- source: https://openapi.tools/ name: Docuo category: Documentation link: https://docuo.spreading.ai/?via=ot repository: https://github.com/spreadingai language: SaaS source_description: >- Docuo is a lightweight platform that transforms your static contents into a modern developer hub, API reference, product guides, and more. v2: false v3: true v3_1: true id: db7874d4a4e4dde0f580b7188c2f60d4 foundInMaster: true - source: https://openapi.tools/ name: API Insights category: - Monitoring - Description Validators link: https://apiinsights.io/ repository: https://github.com/treblle source_description: >- API Insights is a web application and native Mac application that allows you get real time insights into your API using your OpenAPI specification. Think of it like a static analysis tool for your API, using your OpenAPI Specification file. v2: false v3: true id: 7d6c99eb736ef40793d4004ecfbb1661 foundInMaster: true - source: https://openapi.tools/ name: APIMatic Developer Experience Portal category: Documentation link: https://www.apimatic.io/product/publish language: SaaS source_description: >- Customizable developer portals packed with language specific documentation, client libraries, code samples, an API console and much more. v2: true v3: true v3_1: true id: fe6f1a80f0b8b8c43c2c05947b9d6f7a foundInMaster: true - source: https://openapi.tools/ name: APIMatic Transformer category: Converters language: SaaS source_description: >- Transform API Descriptions to and from RAML, API Blueprint, OAI v2/v3, WSDL, etc. link: https://www.apimatic.io/transformer v2: true v3: true v3_1: true id: e455b137b6388767a0bd96244bddcf3b foundInMaster: true - source: https://openapi.tools/ name: Apitally category: Monitoring language: SaaS source_description: >- Simple REST API monitoring tool that helps engineering and product teams understand their APIs. Tracks usage, errors, response times and more using middleware for Python and Node.js frameworks. Also offers uptime monitoring and logging of individual requests. link: https://apitally.io repository: https://github.com/apitally v2: true v3: true v3_1: true id: 93c8a884c50e0448d41e99e5d1de9b1c foundInMaster: true - source: https://openapi.tools/ name: Assertible category: Testing language: SaaS link: https://assertible.com repository: null source_description: >- Import an OpenAPI specification into Assertible to generate tests that validate JSON Schema responses and status codes on every endpoint. v2: true v3: true foundInMaster: true id: da167de4273ba1c7d1a094b9939f3f69 - source: https://openapi.tools/ name: avantation category: - Converters - Testing language: TypeScript link: https://www.avantation.in/ repository: https://github.com/anbuksv/avantation source_description: Generate OpenAPI 3.x specification from HAR. v2: false v3: true repositoryMetadata: base64Readme: >- <h2 align="center">Avantation</h2>
<p align="center">
</p>
<p align="center">
<!--  <a href="https://npmjs.org/package/avantation">
    <img alt="Beta" src="https://img.shields.io/badge/-🚀beta-important.svg?style=flat-square)">
  </a>
-->
  <a href="https://npmjs.org/package/avantation">
    <img alt="Version" target="_blank" src="https://img.shields.io/npm/v/avantation.svg?style=flat-square">
  </a>
  <a href="https://npmjs.org/package/avantation">
    <img alt="downloads" target="_blank" src="https://img.shields.io/npm/dm/avantation.svg?style=flat-square">
  </a>
<!--  <a href="#Code Style">
    <img alt="code style: avantation" src="https://img.shields.io/badge/code_style-avantation-ff69b4.svg?style=flat-square">
  </a>
-->
  <a href="https://npmjs.org/package/avantation">
    <img alt="license" target="_blank" src="https://img.shields.io/npm/l/avantation.svg?style=flat-square">
  </a>
  <a href="https://www.paypal.me/anbuksv">
    <img alt="donation" target="_blank" src="https://img.shields.io/badge/donation-paypal-blueviolet.svg?style=flat-square">
  </a>
</p>

avantation is a tool to generate an OpenAPI 3.0 specification from HTTP Archive format(HAR).

avantation is written and maintained by Anbarasan K (anbuksv@gmail.com).

<ul>
  <li>
   Project home page: <a target="_blank" href="https://www.avantation.in/">https://www.avantation.in/ </a>
  </li>
  <li>
   Code home page: <a target="_blank" href="https://github.com/anbuksv/avantation">https://github.com/anbuksv/avantation</a>
  </li>
  <li>
   Issue tracker: <a target="_blank" href="https://github.com/anbuksv/avantation/issues">https://github.com/anbuksv/avantation/issues</a>
  </li>
</ul>

## Contributing

Bug reports and code and documentation patches are welcome.

## Install

Direct downloads are available through the [release page](https://github.com/anbuksv/avantation/releases/latest).

If you have Node installed on your computer just run npm install

```
npm install -g avantation
```

## Features

-   HTTP sample code generation
-   Dynamic route path conversion
-   Schema generation
-   Supports json/yaml output formats
-   JWT authentication header mapping
-   Static-UI generation

## Usage

```sh-session
Build OpenAPI specification from HAR.

USAGE
  $ avantation HAR

ARGUMENTS
  HAR  http archive(har) path

OPTIONS
  -b, --base-path=base-path                Separate the common path as base path from HTTP requests.
                                           Example:['api/v1']

  -h, --host=host                          Filter the http request from HAR and use it as server url in output.

  -j, --json                               Write output result in JSON format.

  -o, --out=out                            [default: ./openapi.yaml] Write output result at this DEST location.

  -r, --path-param-regex=path-param-regex  [default: ^([0-9]|[-$@!~%^*()_+])+$] Convert Regex matching params into dynamic path

  -s, --security-headers=security-headers  [default: {}] Map matching HTTP headers into security headers on request.

  -t, --template=template                  To override the default template pass the your template file location.

  --disable-tag                            Diable end points grouping based on route path in HAR

  --http-snippet                           Generate HTTP smaple code snippet for request and append it as 'x-code-sample' to OpenAPI path
                                           object.

```

## Demo

### Existing Projects

![Existing Project Demo](demo/Existing_Project_Demo.gif)

**Tools** : [Firefox](https://www.mozilla.org/en-US/firefox/new/) | [Chrome](https://www.google.com/chrome/) | [Safari](https://www.apple.com/safari/) | [Charles Proxy](https://www.charlesproxy.com/)

### New Projects

![New Project Demo](demo/New_Project_Demo.gif)

**Tools** : [Insomnia](https://insomnia.rest/download/)

## User Interface's
<p>
<a target="_blank" href="https://demo.avantation.in/ui/swagger">
  <img alt="Swagger-UI" src="https://img.shields.io/badge/-Swagger-green.svg?style=social&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAACAAAAAgCAYAAABzenr0AAAABHNCSVQICAgIfAhkiAAABiJJREFUWIWtV11sFNcV/s6d2d2Z2cXeXew1Ri547ZQCTtLKYW2soiTEYKA/0FYiP0IhSUv60FblIQ+opIRUSas2UlpVbRVVkSLUJqUPLZHTptSAi9VGmJgQQiKbCGzsxAaXNXi9Xu/O7O7MPX3Y2vJmB+wl+Z5G+s6957v3nnPmHMJiUX+/Fol4thHTQwxaBeJqAYpAMLODOIOuE+FDlvhr/Hr2TYz0WIvZlhYyWNLSvlSH+rwg+Riz0FVNOsvX5hQj6MAflpAOYCYF0pMKrl7wSicrBIgtSHo1bYv9M+eOTtymgJ1KpGX6KSJ+Bkz+lessrGnPoGG9CdXjvszOAZf7NAwcM/DxuxqIpMlMP71m5H6Bnh570QLCrdsqfGT/Q0rx5WjMRNvuFKqirutviviQitOvVmDkjAZmPpW1Ml9LfvBWYkEBoXs2rfCq4qSicnTbjxLU0JIty/EncfEtDcdfDEmZxxiz3HjtzInLNxVQedeGkGYYZ42gU/+N5yepamV5p74ZboyoeP1AmM2EMpZ2cs2psz3XZzkxZ9XU5NUCehcR128/mPjMnAPA0nob2w8mCILrdOHrQlubPsspsx+R+uaniWlX+w+nKBorvnYpGfFBLzJTAv6wvC0R/rBEICxpuE+rDUivkr461D0noKJtS9gLeSQaszwbvp0qWjg5puK170fwXmcANz5S0dRh3tLR4b1V6HmpEh+d1bCiOQufwXNc5I48rl3yYOqqGvPX1b+cvjKcFgCg23gGBO2+7yVLNjz7lwDMKQUN602s3ZxZ8KR3bc1g2eoc/vuhF31/DpTw9z6ZBAn2QirPAYAAdioQvGdth0kV1aXXmxgtvNK9300ueHoAuHNbBg/8YAoAEL/kKeGDyx2s7TCJBB5FU5NX1MSm7geTv7HNvXKaqUKc+vzsyrvBCBUOkjPdC1bjegtg0qqNugcEiHYoHsl1d7sLyGUKAhTP4gXoSxhCYdhZdwF1X7SgeFgSYYcAcGd1Y57cyqudZ5hTCnwBCdW7aP8AgECVg0xCgeOSzaqHUN2YF8RYLUCo94fcU2uot5CutWty5XkHEPl8HiwJF/+tu/JG0AEDtYIlrTRCTonBkf1hdL0Qhh50sOE702BmHN5bhd8/vAyp6wrsPOPQnmq88ngNchbBmiG8vKsGh/ZEICWjbfc09KCD478M4cjT4ZL9/SEHpPDnhBBsC6WEn4NQGIrKcOzCb1eoDEWRYFvAyROAAp+3BKRD0JdIEAG2ReBb1CxSCk9Oy1o3jzS0WSu/un+qyMCaIZx/w4+3/1SBaMzE1w+W/MhuidcPhDF6TkPzt1JofWQGHr04iN/8WRCDp/RLQrIYTU+WXoEWYKx7cAYkGKMf+MpyDgDjA16QYLTuSpU4B4D0pAowxgWYhyaGvFKWhgEUFdAqJGxLILdwDSqCnRXw+hkeF+12njExpDIIFwQDR508idHz7nkmlIL6vClceTfk/1+APD73IBh7X4OTF8QSnWJCTb3B4Pxsyn0SsxXQmlmwfZxDNlOw1SrcBVw+rQHE1oR5pVugt9cE098GjhmcHC+NhcgdhRpw8ndBDHS7i5yP/mM6Tr4UBACEV5RWoeS4gv4unVnij+jvzwkAYAf7HAdO928rSxZseGIaDW0mrl304sIJYxECDAyf1qAHHXxp+0wJ3/2bSrCkHGXxY2BeS1YT2/xzkNjXvjeBps1lRtw8xIdUyDwhsioHIYqfrf+4ju5fh8DAC/G+rn3AvI4obXzhPwFDfnOkz1cdjWXpU3U+VRJExc7jgx78/bkQs6RzcXV6N8bGbGB+Tzh4NJuz5VekQ6nOZ8N8Y0S9LQFumBhW0flsmCFpzJTZLejtnbvioqizxi8nA3WNfbYpdg6cMJRgnU1LXQKpHAye0tB5YCnbJo0yy42T7/zrynzeNbcizVvvhir/SaDaaIuJ1l0pRBrLH0zefm0Jhvv08gaTORGt7TVg5Q8E6gCA+piF1e0ZNLSWN5o5LH4yYWRfLGs0KxLSsmkLQfwKoDUAoPqkXN6UE0bQgV5ZCFQzKZBOKLg68JkOp8V2Nes6mqTARmLcB1CUwFUg1JKA/DTj+f8Ai8OrGYtF9hQAAAAASUVORK5CYII="/>
</a>
<a target="_blank" href="https://demo.avantation.in/ui">
  <img alt="Redoc-UI" src="https://img.shields.io/badge/-Redoc-green.svg?style=social&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAADIAAAAyCAMAAAAp4XiDAAACUlBMVEVMaXEBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtUBQtVluIoZAAAAxXRSTlMABv3H6I4B5CMoDbYX1wWHovsIElb+FvpQD5X5BAceAgMkkO330Ivs5V/49rcU3g7YKYjK9POG0h3gy9RJTAklEPGUzSYY3wra5ktzMaPFMl7CydEL3blG4rR+sRXTjYFjQXI9doOotfJYIREZGyC9O75uwaTvTnSKcT4fwMOvrkSn/DxUV9aP52xRRUPEk5trxoxH6/WYaUKFrOl6L78TN1qwnpygvEAqyFU0SIB9YrocdWCdpu6q1V022zCzuLsuU09NIgXbDUoAAAL/SURBVEjH3Zb1QyJBGIZXEFBPOREBRUUsLOzu7u4+z+7uOluvu7u7u/v9v27Qk/PcGf8An994d5+dnZnvm4Xjdho2Uy5mVCLlJnRDGlwpoZPuURV+Sk5RLI5gG3yF5/z4zmFX6/IyB3JZMaDbIzJyVH2spi4UkBVVOPMlsY3mYJ4nEBgn/v+dledFUcDpRhv6lJqqZDBd3ho7B50NhXs93eESW0KR9ZgXm3hL4J7jzFi7MV94uPDzbAk8Yxk75CRSINmRnxeGYaaW4XTnw7Of41KrRePhu9cI9ibTkKohOM6qhIe+UIs5100bM3KdxOZlOCFnKAEDyNRwz9K9Nioga6jTkJ/EmQrWMDlwsOOkGmPVuazvfUMYLrCU/nY8ocTKOuidGIrGH3dICYgDlJYbKEk9S2tQco2hyD1wkyzzYv6KlREdef4tWFkyFPtotEkNU9rEe3LzbbbiF4GLWo77vZpnbeSDlhPnYYWlJHhBvV7d/yC/4u8i2p6hZPtgkBL3+qNVy1B0aJ8kFbNk+5flH1OGeEmFRobxIgbFck4+A8EG2BdPGmAeno8YypgAlzmua3pBJpOpZAYE70hdzo1gltFluf7INDSMWeyo3c/sX3aEuB6yjM8RFkc30r5iYYLSRuMCDNMHSZtXYDaRF8c/dUfEfarxWa+AMIEX96h94H+Pcr84sjkdEH7fmnfdCFEgamvnO8rTzEddAwXAt9wMiw0CEuWWCYeqhT5ASsMWQ9tcErjfcFhCYJUvNPIgxcPKy52kWaJeXu0Wb3cmQ1Vfyp9Fob4oxk1guBzzZddmykO8VEDScK6Y55iUKgsO9PmSl57USk2MSJ3sUzta3YAob0fqElvYRivwaY53vDbtc0DSIN3hIl+rkPmR/6yXb5H0ilGTFiIZprt5sbO3GySLrIPPGmgR850JH+hZPWkWAckVfpzxBg4drJNvrwpXXclXT7d2hocHGz6Eq8nJfQq0ZTCU2hDWlroVMIepNKVyaUjDUqSdQeY0gpQ77i8S9wdqriCNDOk1SgAAAABJRU5ErkJggg==" />
</a>
</p>

# FAQ

<h3>How to generate Static-UI?</h3>
<p>&nbsp;&nbsp;&nbsp;&nbsp;  avantation-ui bash script will help you to build static user interface for your specs but it's required <a href="https://www.npmjs.com/package/redoc-cli">redoc-cli. </a></p>
 readmeEtag: '"c7318ba2acb80de1609035a12fd652267e9f1c3f"' readmeLastModified: Tue, 22 Jun 2021 06:35:14 GMT repositoryId: 162068352 description: Build OpenAPI3.0 specification from HAR. created: '2018-12-17T03:17:54Z' updated: '2025-10-06T09:26:18Z' language: TypeScript archived: false stars: 91 watchers: 2 forks: 16 owner: anbuksv logo: https://avatars.githubusercontent.com/u/13810481?v=4 license: MIT repoEtag: '"5df568c595c0d065d2f9be00c5b01621aad3f228a9b0c7cb48efd5ba455b09ed"' repoLastModified: Mon, 06 Oct 2025 09:26:18 GMT foundInMaster: true id: 2b13d07e4984e87f5b5e3506ea242963 v3_1_link: https://github.com/anbuksv/avantation/issues/42 - source: https://openapi.tools/ name: Connexion category: - Server - Server Implementations language: Python link: https://connexion.readthedocs.io/en/latest/ repository: https://github.com/spec-first/connexion source_description: >- Connexion is a modern Python web framework that makes spec-first and api-first development easy. No code is generated. With Connexion, you write your server-side API handlers according to a naming convention, and the Connexion engine, given a spec, invokes your code after any input has been validated. v2: true v3: true repositoryMetadata: base64Readme: >-  <a id="top"></a>
 <p align="center">
     <img src="https://raw.githubusercontent.com/spec-first/connexion/main/docs/images/logo_banner.svg" width="100%"/>
 </p>
 <p align="center">
     <a href="https://pypi.org/project/connexion"><img alt="coveralls" src="https://img.shields.io/pypi/status/connexion.svg?style=flat-square&color=brightgreen"></a>
     <a href="https://pypi.org/project/connexion"><img alt="PyPI version" src="https://img.shields.io/pypi/v/connexion?color=brightgreen&style=flat-square"></a>
     <a href="https://pypistats.org/packages/connexion"><img alt="PyPI" src="https://img.shields.io/pypi/dm/connexion?style=flat-square&color=brightgreen"></a>
     <a href="https://github.com/spec-first/connexion/blob/main/LICENSE"><img alt="License" src="https://img.shields.io/pypi/l/connexion?style=flat-square&color=brightgreen"></a>
     <a href="https://github.com/spec-first/connexion/actions/workflows/pipeline.yml"><img alt="GitHub Workflow Status" src="https://img.shields.io/github/actions/workflow/status/spec-first/connexion/pipeline.yml?style=flat-square"></a>
     <a href="https://coveralls.io/github/spec-first/connexion?branch=main"><img alt="Coveralls" src="https://img.shields.io/coverallsCoverage/github/spec-first/connexion?style=flat-square"></a>
     <a href="https://gurubase.io/g/connexion"><img alt="Gurubase" src="https://img.shields.io/badge/Gurubase-Ask%20Connexion%20Guru-brightgreen?style=flat-square"></a>
     <br>
     <br>
     <a href="https://connexion.readthedocs.io/en/stable/"><strong>Explore the docs »</strong></a>
 </p>

---

Connexion is a modern Python web framework that makes spec-first and api-first development easy.
You describe your API in an [OpenAPI][OpenAPI] (or [Swagger][Swagger]) specification with as much 
detail as you want and Connexion will guarantee that it works as you specified.

It works either standalone, or in combination with any ASGI or WSGI-compatible framework!

<p align="center">
   <br>
   <a href="https://connexion.readthedocs.io/en/latest/v3.html"><strong>📢 Connexion 3 was recently released! Read about the changes here »</strong></a>
   <br>
   <br>
</p>

## ✨ Features

Connexion provides the following functionality **based on your specification**:

- 🚏 **Automatic route registration**, no ``@route`` decorators needed
- 🔒 **Authentication**, split from your application logic
- 🔎 **Request and response validation** of headers, parameters, and body
- 📬 **Parameter parsing and injection**, no request object needed
- 📨 **Response serialization**, you can return regular Python objects
- 📺 **A Swagger UI console** with live documentation and ‘try it out’ feature
- 🧩 **Pluggability**, in all dimensions

Connexion also **helps you write your OpenAPI specification** and develop against it by providing a command line interface which lets you test and mock your specification.

```shell
   connexion run openapi.yaml
```

 <p align="right">(<a href="#top">back to top</a>)</p>


## 🫶 Sponsors

<a href="https://www.ml6.eu"><img src="https://raw.githubusercontent.com/spec-first/connexion/main/docs/images/sponsors/ML6.png" title=ML6 height="100"></a>

Sponsors help us dedicate time to maintain Connexion. Want to help?

<a href="https://github.com/sponsors/spec-first"><strong>Explore the options »</strong></a>

<p align="right">(<a href="#top">back to top</a>)</p>

## 🪤 Why Connexion

With Connexion, you write the spec first. Connexion then calls your Python
code, handling the mapping from the specification to the code. This
incentivizes you to write the specification so that all of your
developers can understand what your API does, even before you write a
single line of code.

If multiple teams depend on your APIs, you can use Connexion to easily
send them the documentation of your API. This guarantees that your API will
follow the specification that you wrote. This is a different process from
the one offered by most frameworks, which generate a specification
*after* you've written the code.
Some disadvantages of generating specifications based on code is that
they often end up lacking details or mix your documentation with the implementation
logic of your application.

<p align="right">(<a href="#top">back to top</a>)</p>

## ⚒️ How to Use

### Installation

You can install connexion using pip:

```shell
    $ pip install connexion
```

Connexion provides 'extras' with optional dependencies to unlock additional features:

- `swagger-ui`: Enables a Swagger UI console for your application.
- `uvicorn`: Enables to run the your application using `app.run()` for
  development instead of using an external ASGI server.
- `flask`: Enables the `FlaskApp` to build applications compatible with the Flask
  ecosystem.

You can install them as follows:

```shell
    $ pip install connexion[swagger-ui]
    $ pip install connexion[swagger-ui,uvicorn]
```

<p align="right">(<a href="#top">back to top</a>)</p>

### Creating your application

Connexion can be used either as a standalone application or as a middleware wrapping an existing
ASGI (or WSGI) application written using a different framework. The standalone application can be
built using either the `AsyncApp` or `FlaskApp`.

- The `AsyncApp` is a lightweight application with native asynchronous support. Use it if you
  are starting a new project and have no specific reason to use one of the other options.

  ```Python
      from connexion import AsyncApp

      app = AsyncApp(__name__)
  ```

- The `FlaskApp` leverages the `Flask` framework, which is useful if you're migrating from
  connexion 2.X or you want to leverage the `Flask` ecosystem.

  ```python
      from connexion import FlaskApp

      app = FlaskApp(__name__)
  ```

- The `ConnexionMiddleware` can be wrapped around any existing ASGI or WSGI application.
  Use it if you already have an application written in a different framework and want to add
  functionality provided by connexion

  ```python
      from asgi_framework import App
      from connexion import ConnexionMiddleware

      app = App(__name__)
      app = ConnexionMiddleware(app)
  ```

<p align="right">(<a href="#top">back to top</a>)</p>

### Registering an API

While you can register individual routes on your application, Connexion really shines when you
register an API defined by an OpenAPI (or Swagger) specification.
The operation described in your specification is automatically linked to your Python view function via the ``operationId``

**run.py**

```python
   def post_greeting(name: str, greeting: str):  # Paramaeters are automatically unpacked
       return f"{greeting} {name}", 200          # Responses are automatically serialized

   app.add_api("openapi.yaml")
```

**openapi.yaml**

```yaml
   ...
   paths:
     /greeting/{name}:
       post:
         operationId: run.post_greeting
         responses:
           '200':
             content:
               text/plain:
                 schema:
                   type: string
         parameters:
           - name: name
             in: path
             required: true
             schema:
               type: string
           - name: greeting
             in: query
             required: true
             schema:
               type: string
```

<p align="right">(<a href="#top">back to top</a>)</p>

### Running your application

If you installed connexion using `connexion[uvicorn]`, you can run it using the
`run` method. This is only recommended for development:

```python
    app.run()
```

In production, run your application using an ASGI server such as `uvicorn`. If you defined your
`app` in a python module called `run.py`, you can run it as follows:

```shell
    $ uvicorn run:app
```

Or with gunicorn:

```shell
    $ gunicorn -k uvicorn.workers.UvicornWorker run:app
```

----

Now you're able to run and use Connexion!

See the [examples][examples] folder for more examples.

<p align="right">(<a href="#top">back to top</a>)</p>

## 📜 Changes

A full changelog is maintained on the [GitHub releases page][Releases].

<p align="right">(<a href="#top">back to top</a>)</p>

## 🤲 Contributing

We welcome your ideas, issues, and pull requests. Just follow the
usual/standard GitHub practices.

For easy development, install connexion using poetry with all extras, and
install the pre-commit hooks to automatically run black formatting and static analysis checks.

```shell
    pip install poetry
    poetry install --all-extras
    pre-commit install
```

You can find out more about how Connexion works and where to apply your changes by having a look
at our [architecture][Architecture].

Unless you explicitly state otherwise in advance, any non trivial
contribution intentionally submitted for inclusion in this project by you
to the steward of this repository shall be under the
terms and conditions of Apache License 2.0 written below, without any
additional copyright information, terms or conditions.

<p align="right">(<a href="#top">back to top</a>)</p>

## 🙏 Thanks

We'd like to thank all of Connexion's contributors for working on this
project, Swagger/OpenAPI for their support, and Zalando for originally developing and releasing Connexion.

## 📚 Recommended Resources

About the advantages of working spec-first:

* [Blog Atlassian][Blog Atlassian]
* [API guidelines Zalando][API guidelines Zalando]
* [Blog ML6][Blog ML6]
* [Blog Zalando][Blog Zalando]

Tools to help you work spec-first:

* [Online swagger editor][Online swagger editor]
* [VS Code plugin][VS Code plugin]
* [Pycharm plugin][Pycharm plugin]

[OpenAPI]: https://openapis.org/
[Swagger]: http://swagger.io/open-source-integrations/
[Blog atlassian]: https://www.atlassian.com/blog/technology/spec-first-api-development
[Blog ML6]: https://blog.ml6.eu/why-we-decided-to-help-maintain-connexion-c9f449877083
[Blog Zalando]: https://engineering.zalando.com/posts/2016/12/crafting-effective-microservices-in-python.html
[API guidelines Zalando]: https://opensource.zalando.com/restful-api-guidelines/#api-first
[Online swagger editor]: https://editor.swagger.io/
[VS Code plugin]: https://marketplace.visualstudio.com/items?itemName=42Crunch.vscode-openapi
[Pycharm plugin]: https://plugins.jetbrains.com/plugin/14837-openapi-swagger-editor
[examples]: https://github.com/spec-first/connexion/blob/main/examples
[Releases]: https://github.com/spec-first/connexion/releases
[Architecture]: https://github.com/spec-first/connexion/blob/main/docs/images/architecture.png
 readmeEtag: '"cc069a5d4ed02ea14282a4a02825c583d1c8c21e"' readmeLastModified: Mon, 13 Oct 2025 18:15:57 GMT repositoryId: 35882519 description: >- Connexion is a modern Python web framework that makes spec-first and api-first development easy. created: '2015-05-19T13:05:58Z' updated: '2026-01-30T17:32:33Z' language: Python archived: false stars: 4576 watchers: 75 forks: 781 owner: spec-first logo: https://avatars.githubusercontent.com/u/58754196?v=4 license: Apache-2.0 repoEtag: '"dccaa3e0e2732d9659f9019f019324c5002b602311d68b7bb93db9aaf2032551"' repoLastModified: Fri, 30 Jan 2026 17:32:33 GMT foundInMaster: true homepage: https://github.com/zalando/connexion id: 2d575abdf2671028868058b2442d9131 oldLocations: - https://github.com/zalando/connexion - source: https://openapi.tools/ name: CUE category: DSL language: CUE source_description: >- CUE is an open source language, with a rich set of APIs and tooling, for defining, generating, and validating all kinds of data configuration, APIs, database schemas, code, etc. CUE currently supports generating OpenAPI through its API. link: https://cuelang.org/docs/integrations/openapi/ v2: false v3: true foundInMaster: true id: a3c52256b7de10d04cbf847a9f0f3fb7 - source: https://openapi.tools/ name: JetBrains tools (IntelliJ IDEA, PyCharm etc.) category: GUI Editors language: - Java - Python link: https://plugins.jetbrains.com/plugin/14394-openapi-specifications repository: null source_description: > JetBrains development tools like IntelliJ IDEA, PyCharm and others come with a bundled *OpenAPI Specifications* plugin. The plugin allows you to write the OpenAPI specifications and supports you with validations, formatting, code-completion etc. It supports a *text view* as well as a rendered SwaggerUI-like *graphical interface*. v2: true v3: true id: 7610df08711c820262bf0fa23fe0279f foundInMaster: true - source: https://openapi.tools/ name: Framna Docs category: Documentation link: https://github.com/shapehq/framna-docs/ language: TypeScript source_description: >- Self-hosted web portal that centralizes OpenAPI documentation and facilitates spec-driven development, built with GitHub-based authorization. Integrates with Swagger UI, Stoplight Elements, and Redocly. v2: true v3: true v3_1: true id: d9e93963d6018c3e90a9302baf0a1090 foundInMaster: true - source: https://openapi.tools/ name: Frevo category: - Documentation link: https://frevo.dev?ref=openapitools language: SaaS source_description: >- Keep track of changes in your OpenAPI specifications & generate API references ⛱️ v2: false v3: true v3_1: true id: 4bbbc7dd6703e50c5719dc27fbb4de65 foundInMaster: true - source: https://openapi.tools/ name: Kong Enterprise Edition category: - Documentation - Gateway link: https://konghq.com/kong-enterprise-edition/ language: Lua source_description: >- Highly customizable developer portal with developer onboarding, integrated with the Kong API Gateway v2: true v3: true foundInMaster: true id: 263dd2cb7817fcb9aa55e4fd97b6b852 v3_1_link: https://github.com/Kong/insomnia/issues/4732 v3_1: true - source: - https://openapi.tools/ - openapi3 tags name: Rate My OpenAPI category: - Description Validators - Security - Server Implementations language: Go repository: https://github.com/zuplo/rate-my-openapi link: https://ratemyopenapi.com source_description: Free and open source OpenAPI automated review and validation tool. v3: true v3_1: true id: 54fc718fad968b8a7eecd1e88f3fff99 repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://ratemyopenapi.com/">
    <img src="https://cdn.zuplo.com/static/logos/logo.svg" height="50">
    <h1 align="center">Rate My OpenAPI</h1>
  </a>
</p>

<div align="center">
  <p align="center">
    <a aria-label="Zuplo logo" href="https://zuplo.com">
      <img src="https://img.shields.io/badge/MADE%20BY%20Zuplo-FF00BD.svg?style=for-the-badge&logo=data:image/svg%2bxml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGZpbGw9Im5vbmUiIHZpZXdCb3g9IjAgMCAzNyAzMiIgYXJpYS1oaWRkZW49InRydWUiPgogIDxwYXRoIGZpbGw9IiNGRjAwQkQiIGQ9Ik0yNy4xNDIgMTkuOTc4SDE2LjYyTDI3LjgzIDguNzQ2YS43NTguNzU4IDAgMDAtLjUzNC0xLjI5M0g5LjQ4OFYwaDE5LjUzNGE3LjU3MyA3LjU3MyAwIDAxNC4wNjUgMS4xMjUgNy41OTEgNy41OTEgMCAwMTIuODM2IDMuMTI2IDcuNDAyIDcuNDAyIDAgMDEtMS40NjEgOC4zOThsLTcuMzIgNy4zMjh6Ii8+CiAgPHBhdGggZmlsbD0iI0ZGMDBCRCIgZD0iTTkuNDg5IDExLjA0MmgxMC41MjRsLTExLjE5IDExLjIxYS43NzIuNzcyIDAgMDAuNTQzIDEuMzE2aDE3Ljc1OXY3LjQ1Mkg3LjYxYTcuNTc0IDcuNTc0IDAgMDEtNC4wNjUtMS4xMjVBNy41OTMgNy41OTMgMCAwMS43MSAyNi43NjhhNy40MDMgNy40MDMgMCAwMTEuNDYyLTguMzk3bDcuMzE4LTcuMzI5eiIvPgo8L3N2Zz4K&labelColor=000"></a>
    <a aria-label="NPM version" href="https://www.npmjs.com/package/rmoa"><img alt="NPM Badge" src="https://img.shields.io/npm/v/rmoa.svg?style=for-the-badge&labelColor=000000"></a>
    <a aria-label="License" href="https://github.com/zuplo/rate-my-openapi/blob/main/LICENSE"><img alt="License Badge" src="https://img.shields.io/npm/l/rmoa.svg?style=for-the-badge&labelColor=000000"></a>
    <a aria-label="Join the community on Discord" href="https://discord.com/channels/848913990360629268/1235294876778627246"><img alt="Discrod Badge" src="https://img.shields.io/badge/Chat%20on%20discord-5865F2.svg?style=for-the-badge&logo=discord&labelColor=000000&logoWidth=20"></a>
  </p>
  <a href="https://twitter.com/zuplo">
    <img alt="X (formerly Twitter) Follow" src="https://img.shields.io/twitter/follow/zuplo">
  </a>
  <p align="center">
    <a href="#openapi---introduction"><strong>Introduction</strong></a> · 
    <a href="#website"><strong>Website</strong></a> · 
    <a href="#cli"><strong>CLI</strong></a> · 
    <a href="#github-action"><strong>GitHub Action</strong></a> · 
    <a href="#apis"><strong>APIs</strong></a>
  </p>
</div>

## OpenAPI - Introduction

[OpenAPI](https://www.openapis.org/) is an industry standard to describe HTTP
APIs. When using OpenAPI in your project, you can leverage other tools to help
you generate documentation, code, tests, mock results, or even deploy your API.
It's what's commonly known as the OpenAPI lifecycle, which looks like this:

<div align="center">
<img style="width:50%" src="assets/openapi-lifecycle-light.png#gh-light-mode-only" />
  <img style="width:50%" src="assets/openapi-lifecycle-dark.png#gh-dark-mode-only" />
</div>

## Rate My OpenAPI

At Zuplo we believe that the better the quality of an OpenAPI document, the
better the developer experience will be for the consumers of that API. This
experience is important for the success of an API.

_Rate My OpenAPI_ is a suite of tools designed to help software developers who
are using OpenAPI to design and implement their APIs.

Our tools include a
[website](https://ratemyopenapi.com?utm_source=github&utm_medium=web), a
[CLI](https://www.npmjs.com/package/rmoa), a
[GitHub Action](https://github.com/marketplace/actions/rate-my-openapi-action)
and an [API](https://api.ratemyopenapi.com/docs/routes/introduction), all aimed
at ensuring your APIs meet high standards of quality and usability.

### Categories of Evaluation

Our tools evaluate your OpenAPI definition files and provide a comprehensive
score based on four key categories:

- **Documentation:** Ensure your API is well-documented, making it easy for
  users to understand and use.
- **SDK Generation:** Verify that your API definition supports SDK generation,
  facilitating integration and usage in different programming languages.
- **Security:** Check for best practices and standards to ensure your API is
  secure and protected against common vulnerabilities.
- **Completeness** Ensure your API definition is complete, with all necessary
  endpoints, parameters, and responses accurately defined.

### Website

[https://ratemyopenapi.com](https://ratemyopenapi.com?utm_source=github&utm_medium=web)
offers a user-friendly interface for developers to upload and analyze their
OpenAPI definition files.

Key features include:

- **Linting:** Upload & lint your OpenAPI files to receive detailed feedback.
- **Comprehensive Scoring:** Get a clear, actionable score rating your API's
  documentation, SDK generation, security, and completeness.
- **Detailed Reports:** Access in-depth reports that highlight areas of
  improvement and provide recommendations.
- **Visualization:** Easily visualize the structure and quality of your API with
  in-line feedback.

#### Getting Started

To get started visit
[https://ratemyopenapi.com](https://ratemyopenapi.com?utm_source=github&utm_medium=web),
upload your OpenAPI definition file and review the detailed reports to identify
areas for improvement.

### CLI

The CLI tool is perfect for developers who prefer working from the command line
or need to integrate quality checks into their development workflow.

Key features include:

- <**Automated Checks:** Integrate the CLI into your CI/CD pipeline for
  automated quality checks on every commit.
- **Detailed Output:** Get detailed feedback directly in your terminal, with
  options to further integrate these results into your development flow.

#### Getting Started

To get started install the CLI and start integrating it into your development
workflow.

```bash
npm install rmoa

rmoa lint --filename <openapi-filename> --api-key <API_KEY>
```

CLI source code and documentation can be found at
[packages/cli](/packages/cli/README.md).

To get an `API_KEY` sign up for free at
[https://api.ratemyopenapi.com/docs](https://api.ratemyopenapi.com/docs).

### GitHub Action

Our GitHub Action seamlessly integrates with your repository to ensure your APIs
are consistently of high quality.

Key features include:

- **Automated Linting:** Automatically lint OpenAPI definition files on every
  pull request and push to ensure code quality.
- **Inline Feedback:** Receive feedback directly in your pull requests with
  comments highlighting issues and areas for improvement.
- **Continuous Improvement:** Maintain a high standard of API quality with
  continuous monitoring and feedback.

#### Getting Started

To get started add our GitHub action to your repository and configure it to run
on Pull Requests and Pushes to ensure continuous quality monitoring.

```yaml
steps:
  - uses: actions/checkout@v4
  - uses: zuplo/rmoa-action@v1
    with:
      filepath: "./my-api.json"
      apikey: ${{ secrets.RMOA_API_KEY }}
```

GitHub Action source code and documentation can be found at
[rmoa-action](https://github.com/zuplo/rmoa-action).

#### Getting an API key

To get an `API_KEY` you can sign up for free at
[https://api.ratemyopenapi.com/docs](https://api.ratemyopenapi.com/docs).

### APIs

All our tools make use of our APIs to analyze and provide detailed results.

You can also make direct use of these APIs, which is a great option for those
developers that want to build their own tools or integrate the Rate My OpenAPI
linting capabilities in a way that's not covered the existing tools.

#### Getting Started

To get started, go to
[https://api.ratemyopenapi.com/docs](https://api.ratemyopenapi.com/docs) to get
the detailed documentation on every endpoint available for use.

#### Getting an API key

To get an `API_KEY` you can sign up for free at
[https://api.ratemyopenapi.com/docs](https://api.ratemyopenapi.com/docs).

# License

[MIT License](./LICENSE)
 readmeEtag: '"c0f831b7566b70e4f15b7325e8eff11cccb35602"' readmeLastModified: Thu, 22 Aug 2024 17:48:19 GMT repositoryId: 653466139 description: Find API quality and security issues via your OpenAPI spec created: '2023-06-14T05:39:38Z' updated: '2026-01-18T18:44:23Z' language: TypeScript archived: false stars: 241 watchers: 4 forks: 7 owner: zuplo logo: https://avatars.githubusercontent.com/u/85497839?v=4 license: MIT repoEtag: '"073bb8c12139f8136ca2b2bd0e3401c5817637ebb1074d4fc0fcae79e782f441"' repoLastModified: Sun, 18 Jan 2026 18:44:23 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: Scalar category: Documentation language: Vue.js repository: https://github.com/scalar/scalar link: https://docs.scalar.com/swagger-editor source_description: Beautiful Open-Source API references from Swagger/OAS files ✨ v2: true v3: true v3_1: true id: 9cad3caafc5e01c5c1cd167696f30f10 repositoryMetadata: base64Readme: >- [![CI](https://github.com/scalar/scalar/actions/workflows/ci.yml/badge.svg)](https://github.com/scalar/scalar/actions/workflows/ci.yml)
[![Contributors](https://img.shields.io/github/contributors/scalar/scalar)](https://github.com/scalar/scalar/graphs/contributors)
[![GitHub License](https://img.shields.io/github/license/scalar/scalar)](https://github.com/scalar/scalar/blob/main/LICENSE)
[![Twitter](https://img.shields.io/twitter/follow/scalar)](https://x.com/scalar)
[![Discord](https://img.shields.io/discord/1135330207960678410?style=flat&color=5865F2)](https://discord.gg/scalar)

<h1>
	<p>Scalar</p>
	<p>
		<a href="https://docs.scalar.com/swagger-editor" target="_blank">
			<picture>
				<source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/9b639f2d-0204-4b15-b918-158072a9afa5">
				<source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/be723d07-ec9c-4edb-b657-05a8bc05c7d6">
				<img width="48.494%" height="250" src="https://github.com/user-attachments/assets/9b639f2d-0204-4b15-b918-158072a9afa5#gh-light-mode-only" alt="Scalar API Reference">
			</picture>
	 	</a>
		<a href="https://scalar.com/download" target="_blank">
			<picture>
				<source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/b391b82f-d993-4e97-8a17-1c3c7ccbb19b">
				<source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/579d0853-b2a5-4ea8-b2bd-a98753faeaf3">
				<img width="48.494%" height="250" src="https://github.com/user-attachments/assets/b391b82f-d993-4e97-8a17-1c3c7ccbb19b#gh-light-mode-only" alt="Scalar API client">
			</picture>
		</a>
	</p>
</h1>

<p>
	<img width="830" height="auto" src="https://github.com/user-attachments/assets/3a24fc5c-5d3c-4be4-8b9e-399bd49e4fb7#gh-light-mode-only">
	<img width="830" height="auto" src="https://github.com/user-attachments/assets/76f123ae-f458-409f-84d3-b9ad14fd3fb2#gh-dark-mode-only">
</p>

### Interactive API Reference from OpenAPI/Swagger

└ Renders OpenAPI/Swagger documents<br>
└ Comes with an API testing tool<br>
└ Doesn't look like 2011<br>
└ Generates code examples for many languages & frameworks<br>
└ Integrates with your favorite framework<br>
└ <a href="https://docs.scalar.com/swagger-editor">Try Demo</a>

<!-- minimal line break-->
<p>
	<img width="1200" height="1" src="https://github.com/user-attachments/assets/7d7e7ba7-2b02-49f3-abcd-b24c566a3c16#gh-light-mode-only">
	<img width="1200" height="1" src="https://github.com/user-attachments/assets/341bfb1d-5cb0-4ec6-89eb-4b1dcc07eeb7#gh-dark-mode-only">
</p>
<!-- end minimal line break-->

<p>
	<img width="830" height="auto" src="https://github.com/user-attachments/assets/d9f90c51-82d4-4948-9365-658ccc7d78c4#gh-light-mode-only">
	<img width="830" height="auto" src="https://github.com/user-attachments/assets/e7c2023e-dfbc-428c-bc85-c06d8a6e4b67#gh-dark-mode-only">
</p>

### An offline-first API Client built for OpenAPI

└ Free and open-source<br>
└ First-class OpenAPI/Swagger support<br>
└ Integrates with dozens of API Frameworks (Elysia, Hono, FastAPI, etc)<br>
└ Syncs with your favorite server framework (Watch Mode)<br>
└ Has environment variables, dynamic parameters and everything<br>
└ <a href="https://client.scalar.com/">Try It in the Browser</a>
└ <a href="https://scalar.com/download">Download (Windows, MacOS, Linux)</a><br>

<!-- minimal line break-->
<p>
	<img width="1200" height="1" src="https://github.com/user-attachments/assets/7d7e7ba7-2b02-49f3-abcd-b24c566a3c16#gh-light-mode-only">
	<img width="1200" height="1" src="https://github.com/user-attachments/assets/341bfb1d-5cb0-4ec6-89eb-4b1dcc07eeb7#gh-dark-mode-only">
</p>
<!-- end minimal line break-->

### Trusted by:

<br>

<p>
	<picture>
		<source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/73dcef95-fc4f-4111-bb09-f3ce37453a81#gh-light-mode-only">
		<source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/c792c977-8551-432c-9417-3ea05e408151#gh-dark-mode-only">
		<img width="48.494%" height="330" src="https://github.com/user-attachments/assets/73dcef95-fc4f-4111-bb09-f3ce37453a81#gh-light-mode-only" alt="Scalar API client">
	</picture>
	<picture>
		<source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/1da6b904-34be-4760-b535-0dd2160e6d12#gh-light-mode-only">
		<source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/54a9ea69-c50a-4f49-8c3e-ddd6f8fbe1f3#gh-dark-mode-only">
		<img width="48.494%" height="330" src="https://github.com/user-attachments/assets/1da6b904-34be-4760-b535-0dd2160e6d12#gh-light-mode-only" alt="Scalar API client">
	</picture>
</p>

### Quickstart

All you need is a single HTML file to create an amazing API reference:

```html
<!doctype html>
<html>
  <head>
    <title>Scalar API Reference</title>
    <meta charset="utf-8" />
    <meta
      name="viewport"
      content="width=device-width, initial-scale=1" />
  </head>

  <body>
    <div id="app"></div>

    <!-- Load the Script -->
    <script src="https://cdn.jsdelivr.net/npm/@scalar/api-reference"></script>

    <!-- Initialize the Scalar API Reference -->
    <script>
      Scalar.createApiReference('#app', {
        // The URL of the OpenAPI/Swagger document
        url: 'https://registry.scalar.com/@scalar/apis/galaxy?format=json',
        // Avoid CORS issues
        proxyUrl: 'https://proxy.scalar.com',
      })
    </script>
  </body>
</html>
```

> Need a Custom Header? Check out this example: https://codepen.io/scalarorg/pen/VwOXqam

And there's an ever-growing list of plugins and integrations:

### Integrations

- [HTML/JS API](https://scalar.com/products/api-references/integrations/html-js) (works everywhere)
- [.NET ASP.NET Core](https://scalar.com/products/api-references/integrations/aspnetcore/integration)
- [.NET Aspire](https://scalar.com/products/api-references/integrations/aspire)
- [AdonisJS](https://scalar.com/products/api-references/integrations/adonisjs)
- [Astro](https://scalar.com/products/api-references/integrations/astro)
- [Django Ninja](https://scalar.com/products/api-references/integrations/django-ninja)
- [Django](https://scalar.com/products/api-references/integrations/django)
- [Docker](https://scalar.com/products/api-references/integrations/docker)
- [Docusaurus](https://scalar.com/products/api-references/integrations/docusaurus)
- [Elixir](https://scalar.com/products/api-references/integrations/elixir)
- [Express](https://scalar.com/products/api-references/integrations/express)
- [FastAPI](https://scalar.com/products/api-references/integrations/fastapi)
- [Fastify](https://scalar.com/products/api-references/integrations/fastify)
- [Flask](https://scalar.com/products/api-references/integrations/flask)
- [Go](https://scalar.com/products/api-references/integrations/go)
- [Hapi](https://scalar.com/products/api-references/integrations/hapi)
- [Hono](https://scalar.com/products/api-references/integrations/hono)
- [Laravel Scribe](https://scalar.com/products/api-references/integrations/laravel-scribe)
- [Laravel](https://scalar.com/products/api-references/integrations/laravel)
- [Micronaut](https://scalar.com/products/api-references/integrations/micronaut)
- [NestJS](https://scalar.com/products/api-references/integrations/nestjs)
- [Next.js](https://scalar.com/products/api-references/integrations/nextjs)
- [Nuxt](https://scalar.com/products/api-references/integrations/nuxt)
- [Python](https://scalar.com/products/api-references/integrations/python)
- [React](https://scalar.com/products/api-references/integrations/react)
- [Ruby on Rails](https://scalar.com/products/api-references/integrations/ruby-on-rails)
- [Rust](https://scalar.com/products/api-references/integrations/rust)
- [Spring Boot](https://scalar.com/products/api-references/integrations/spring-boot)
- [SvelteKit](https://scalar.com/products/api-references/integrations/sveltekit)
- [Ts.ED](https://scalar.com/products/api-references/integrations/tsed)
- [Vue.js](https://scalar.com/products/api-references/integrations/vue)

### Built-in Support

The following frameworks have chosen Scalar API Reference as their default OpenAPI documentation UI, recognizing its developer-friendly features and modern design:

- [Effect](https://scalar.com/products/api-references/integrations/effect)
- [ElysiaJS](https://scalar.com/products/api-references/integrations/elysiajs)
- [HappyX](https://github.com/HapticX/happyx)
- [Litestar](https://docs.litestar.dev/latest/usage/openapi/ui_plugins.html)
- [Nitro](https://scalar.com/products/api-references/integrations/nitro)
- [oRPC](https://orpc.unnoq.com/docs/openapi/plugins/openapi-reference)
- [Platformatic](https://scalar.com/products/api-references/integrations/platformatic)

<br>

### Managed Hosting

We're offering a free and paid managed hosting on the edge, too:

- Write your API documentation and publish API references (free)
- `YOUR_NAME_HERE.apidocumentation.com` subdomain and SSL (free)
- [GitHub Sync](https://scalar.com/products/docs/github-sync)
- Write free-form documentation
- Collaborate with your team
- Use any domain

Ready? [Create your Scalar Account](https://scalar.com)

<br>

### Documentation

| Topic                                                                     | Description                          |
| ------------------------------------------------------------------------- | ------------------------------------ |
| [Configuration](https://scalar.com/products/api-references/configuration) | The universal configuration object   |
| [Themes](https://scalar.com/products/api-references/themes)               | Predefined themes, layouts & styling |
| [OpenAPI](https://scalar.com/products/api-references/openapi)             | OpenAPI specification extensions     |
| [Markdown](https://scalar.com/products/api-references/markdown)           | Supported Markdown syntax            |
| [Plugins](https://scalar.com/products/api-references/plugins)             | Extend the functionality             |

<br>

### Projects

| Project                                                                            | Description                      |
| ---------------------------------------------------------------------------------- | -------------------------------- |
| [Scalar API Reference](https://scalar.com/products/api-references/getting-started) | Self-hosted API Reference        |
| [Scalar API Client](https://scalar.com/download)                                   | Postman alternative              |
| [Scalar Registry](https://scalar.com/products/registry/getting-started)            | Manage OpenAPI documents         |
| [Scalar Docs](https://scalar.com/products/docs/getting-started)                    | Hosted documentation             |
| [Scalar Docs Starter](https://github.com/scalar/starter)                           | Starter template for Scalar Docs |
| [Scalar SDKs](https://scalar.com/products/sdks/getting-started)                    | SDK generation                   |
| [Scalar CLI](https://scalar.com/tools/cli/getting-started)                         | Command-line interface           |
| [Scalar Mock Server](https://scalar.com/products/mock-server/getting-started)      | Mock APIs from OpenAPI documents |
| [Scalar Galaxy](packages/galaxy/README.md)                                         | Our OpenAPI Example              |
| [Scalar Editor](https://editor.scalar.com/)                                        | OpenAPI Online Editor            |
| [Scalar OpenAPI Parser](packages/openapi-parser/README.md)                         | Parse OpenAPI documents          |
| [Scalar OpenAPI to Markdown](packages/openapi-to-markdown/README.md)               | OpenAPI > Markdown/HTML          |
| [Scalar OpenAPI Upgrader](packages/openapi-upgrader/README.md)                     | Upgrade OpenAPI documents        |
| [Scalar Void Server](packages/void-server/README.md)                               | HTTP Request Mirror              |

<br>

### Community

We are API nerds. You too? Let's chat on Discord: <https://discord.gg/scalar>

<br>

### Contributions

Contributions are welcome! Read the [`CONTRIBUTING`](CONTRIBUTING.md) guide.

**Top 10 Contributors (April 2025)**

<p>
	<img width="830" height="280" src="https://github.com/user-attachments/assets/8c10d2aa-9eb4-4818-9ca2-625cfed5ca08#gh-light-mode-only">
	<img width="830" height="280" src="https://github.com/user-attachments/assets/50b9b042-107e-4167-9c7f-94497f85d2e0#gh-dark-mode-only">
</p>

<br>
<br>

**Top Contributors (All Time)**

<!-- readme: collaborators,contributors -start -->
<table>
	<tbody>
		<tr>
            <td align="center">
                <a href="https://github.com/hanspagel">
                    <img src="https://avatars.githubusercontent.com/u/1577992?v=4" width="100;" alt="hanspagel"/>
                    <br />
                    <sub><b>hanspagel</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/amritk">
                    <img src="https://avatars.githubusercontent.com/u/2039539?v=4" width="100;" alt="amritk"/>
                    <br />
                    <sub><b>amritk</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/xC0dex">
                    <img src="https://avatars.githubusercontent.com/u/22918366?v=4" width="100;" alt="xC0dex"/>
                    <br />
                    <sub><b>xC0dex</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/antlio">
                    <img src="https://avatars.githubusercontent.com/u/14966155?v=4" width="100;" alt="antlio"/>
                    <br />
                    <sub><b>antlio</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/hwkr">
                    <img src="https://avatars.githubusercontent.com/u/6374090?v=4" width="100;" alt="hwkr"/>
                    <br />
                    <sub><b>hwkr</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/marclave">
                    <img src="https://avatars.githubusercontent.com/u/6176314?v=4" width="100;" alt="marclave"/>
                    <br />
                    <sub><b>marclave</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/cameronrohani">
                    <img src="https://avatars.githubusercontent.com/u/6201407?v=4" width="100;" alt="cameronrohani"/>
                    <br />
                    <sub><b>cameronrohani</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/DemonHa">
                    <img src="https://avatars.githubusercontent.com/u/91160178?v=4" width="100;" alt="DemonHa"/>
                    <br />
                    <sub><b>DemonHa</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/geoffgscott">
                    <img src="https://avatars.githubusercontent.com/u/59206100?v=4" width="100;" alt="geoffgscott"/>
                    <br />
                    <sub><b>geoffgscott</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/marcalexiei">
                    <img src="https://avatars.githubusercontent.com/u/24919330?v=4" width="100;" alt="marcalexiei"/>
                    <br />
                    <sub><b>marcalexiei</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/tmastrom">
                    <img src="https://avatars.githubusercontent.com/u/36525329?v=4" width="100;" alt="tmastrom"/>
                    <br />
                    <sub><b>tmastrom</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/inyourtime">
                    <img src="https://avatars.githubusercontent.com/u/66111030?v=4" width="100;" alt="inyourtime"/>
                    <br />
                    <sub><b>inyourtime</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/bgrcs">
                    <img src="https://avatars.githubusercontent.com/u/18078763?v=4" width="100;" alt="bgrcs"/>
                    <br />
                    <sub><b>bgrcs</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/gevann">
                    <img src="https://avatars.githubusercontent.com/u/6427337?v=4" width="100;" alt="gevann"/>
                    <br />
                    <sub><b>gevann</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/federicobond">
                    <img src="https://avatars.githubusercontent.com/u/138426?v=4" width="100;" alt="federicobond"/>
                    <br />
                    <sub><b>federicobond</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/liamdscalar">
                    <img src="https://avatars.githubusercontent.com/u/225425944?v=4" width="100;" alt="liamdscalar"/>
                    <br />
                    <sub><b>liamdscalar</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ahmedrangel">
                    <img src="https://avatars.githubusercontent.com/u/50090595?v=4" width="100;" alt="ahmedrangel"/>
                    <br />
                    <sub><b>ahmedrangel</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/matthyk">
                    <img src="https://avatars.githubusercontent.com/u/53833818?v=4" width="100;" alt="matthyk"/>
                    <br />
                    <sub><b>matthyk</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/Amorim33">
                    <img src="https://avatars.githubusercontent.com/u/42624869?v=4" width="100;" alt="Amorim33"/>
                    <br />
                    <sub><b>Amorim33</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/baywet">
                    <img src="https://avatars.githubusercontent.com/u/7905502?v=4" width="100;" alt="baywet"/>
                    <br />
                    <sub><b>baywet</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Mason-Little">
                    <img src="https://avatars.githubusercontent.com/u/105008441?v=4" width="100;" alt="Mason-Little"/>
                    <br />
                    <sub><b>Mason-Little</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ivanagas">
                    <img src="https://avatars.githubusercontent.com/u/34755028?v=4" width="100;" alt="ivanagas"/>
                    <br />
                    <sub><b>ivanagas</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/adjsky">
                    <img src="https://avatars.githubusercontent.com/u/49305219?v=4" width="100;" alt="adjsky"/>
                    <br />
                    <sub><b>adjsky</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/RemcoGoy">
                    <img src="https://avatars.githubusercontent.com/u/34539109?v=4" width="100;" alt="RemcoGoy"/>
                    <br />
                    <sub><b>RemcoGoy</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/mpminardi">
                    <img src="https://avatars.githubusercontent.com/u/8587567?v=4" width="100;" alt="mpminardi"/>
                    <br />
                    <sub><b>mpminardi</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/TheVaan">
                    <img src="https://avatars.githubusercontent.com/u/1108485?v=4" width="100;" alt="TheVaan"/>
                    <br />
                    <sub><b>TheVaan</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/mcollina">
                    <img src="https://avatars.githubusercontent.com/u/52195?v=4" width="100;" alt="mcollina"/>
                    <br />
                    <sub><b>mcollina</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/si-net">
                    <img src="https://avatars.githubusercontent.com/u/25386895?v=4" width="100;" alt="si-net"/>
                    <br />
                    <sub><b>si-net</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/nsychev">
                    <img src="https://avatars.githubusercontent.com/u/11004619?v=4" width="100;" alt="nsychev"/>
                    <br />
                    <sub><b>nsychev</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Mohib834">
                    <img src="https://avatars.githubusercontent.com/u/47316464?v=4" width="100;" alt="Mohib834"/>
                    <br />
                    <sub><b>Mohib834</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/lc-soft">
                    <img src="https://avatars.githubusercontent.com/u/1730073?v=4" width="100;" alt="lc-soft"/>
                    <br />
                    <sub><b>lc-soft</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/fuma-nama">
                    <img src="https://avatars.githubusercontent.com/u/76240755?v=4" width="100;" alt="fuma-nama"/>
                    <br />
                    <sub><b>fuma-nama</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Edgaraszs">
                    <img src="https://avatars.githubusercontent.com/u/55696268?v=4" width="100;" alt="Edgaraszs"/>
                    <br />
                    <sub><b>Edgaraszs</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ATREAY">
                    <img src="https://avatars.githubusercontent.com/u/66585295?v=4" width="100;" alt="ATREAY"/>
                    <br />
                    <sub><b>ATREAY</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ledgersteve">
                    <img src="https://avatars.githubusercontent.com/u/90449376?v=4" width="100;" alt="ledgersteve"/>
                    <br />
                    <sub><b>ledgersteve</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/timheuer">
                    <img src="https://avatars.githubusercontent.com/u/4821?v=4" width="100;" alt="timheuer"/>
                    <br />
                    <sub><b>timheuer</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/ykdojo">
                    <img src="https://avatars.githubusercontent.com/u/1811651?v=4" width="100;" alt="ykdojo"/>
                    <br />
                    <sub><b>ykdojo</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/x-delfino">
                    <img src="https://avatars.githubusercontent.com/u/67192579?v=4" width="100;" alt="x-delfino"/>
                    <br />
                    <sub><b>x-delfino</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/horpto">
                    <img src="https://avatars.githubusercontent.com/u/7109350?v=4" width="100;" alt="horpto"/>
                    <br />
                    <sub><b>horpto</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/jpjpjp">
                    <img src="https://avatars.githubusercontent.com/u/4774656?v=4" width="100;" alt="jpjpjp"/>
                    <br />
                    <sub><b>jpjpjp</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/hex0id">
                    <img src="https://avatars.githubusercontent.com/u/5276261?v=4" width="100;" alt="hex0id"/>
                    <br />
                    <sub><b>hex0id</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/yshrsmz">
                    <img src="https://avatars.githubusercontent.com/u/654889?v=4" width="100;" alt="yshrsmz"/>
                    <br />
                    <sub><b>yshrsmz</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/soulchild">
                    <img src="https://avatars.githubusercontent.com/u/59642?v=4" width="100;" alt="soulchild"/>
                    <br />
                    <sub><b>soulchild</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/SebastianBienert">
                    <img src="https://avatars.githubusercontent.com/u/17458785?v=4" width="100;" alt="SebastianBienert"/>
                    <br />
                    <sub><b>SebastianBienert</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/mouhannad-sh">
                    <img src="https://avatars.githubusercontent.com/u/18495740?v=4" width="100;" alt="mouhannad-sh"/>
                    <br />
                    <sub><b>mouhannad-sh</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/mason-at-pieces">
                    <img src="https://avatars.githubusercontent.com/u/123421085?v=4" width="100;" alt="mason-at-pieces"/>
                    <br />
                    <sub><b>mason-at-pieces</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/luc122c">
                    <img src="https://avatars.githubusercontent.com/u/7257092?v=4" width="100;" alt="luc122c"/>
                    <br />
                    <sub><b>luc122c</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/luke-hagar-sp">
                    <img src="https://avatars.githubusercontent.com/u/193985956?v=4" width="100;" alt="luke-hagar-sp"/>
                    <br />
                    <sub><b>luke-hagar-sp</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/handrews">
                    <img src="https://avatars.githubusercontent.com/u/2358015?v=4" width="100;" alt="handrews"/>
                    <br />
                    <sub><b>handrews</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Duncanma">
                    <img src="https://avatars.githubusercontent.com/u/18338424?v=4" width="100;" alt="Duncanma"/>
                    <br />
                    <sub><b>Duncanma</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/WDaan">
                    <img src="https://avatars.githubusercontent.com/u/25707052?v=4" width="100;" alt="WDaan"/>
                    <br />
                    <sub><b>WDaan</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/gsmcdonald">
                    <img src="https://avatars.githubusercontent.com/u/36003378?v=4" width="100;" alt="gsmcdonald"/>
                    <br />
                    <sub><b>gsmcdonald</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/MarioGK">
                    <img src="https://avatars.githubusercontent.com/u/8379079?v=4" width="100;" alt="MarioGK"/>
                    <br />
                    <sub><b>MarioGK</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/hrynevychroman">
                    <img src="https://avatars.githubusercontent.com/u/82209198?v=4" width="100;" alt="hrynevychroman"/>
                    <br />
                    <sub><b>hrynevychroman</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/rpmccarter">
                    <img src="https://avatars.githubusercontent.com/u/63772591?v=4" width="100;" alt="rpmccarter"/>
                    <br />
                    <sub><b>rpmccarter</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/roryschadler">
                    <img src="https://avatars.githubusercontent.com/u/48921090?v=4" width="100;" alt="roryschadler"/>
                    <br />
                    <sub><b>roryschadler</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/rmp135">
                    <img src="https://avatars.githubusercontent.com/u/3882803?v=4" width="100;" alt="rmp135"/>
                    <br />
                    <sub><b>rmp135</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/mirismaili">
                    <img src="https://avatars.githubusercontent.com/u/15800189?v=4" width="100;" alt="mirismaili"/>
                    <br />
                    <sub><b>mirismaili</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/STP5940">
                    <img src="https://avatars.githubusercontent.com/u/21144303?v=4" width="100;" alt="STP5940"/>
                    <br />
                    <sub><b>STP5940</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/captainsafia">
                    <img src="https://avatars.githubusercontent.com/u/1857993?v=4" width="100;" alt="captainsafia"/>
                    <br />
                    <sub><b>captainsafia</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/realsama">
                    <img src="https://avatars.githubusercontent.com/u/46403284?v=4" width="100;" alt="realsama"/>
                    <br />
                    <sub><b>realsama</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ventocis">
                    <img src="https://avatars.githubusercontent.com/u/36198345?v=4" width="100;" alt="ventocis"/>
                    <br />
                    <sub><b>ventocis</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/SamyPesse">
                    <img src="https://avatars.githubusercontent.com/u/845425?v=4" width="100;" alt="SamyPesse"/>
                    <br />
                    <sub><b>SamyPesse</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/dotfortun">
                    <img src="https://avatars.githubusercontent.com/u/11822957?v=4" width="100;" alt="dotfortun"/>
                    <br />
                    <sub><b>dotfortun</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/smoores-dev">
                    <img src="https://avatars.githubusercontent.com/u/5354254?v=4" width="100;" alt="smoores-dev"/>
                    <br />
                    <sub><b>smoores-dev</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/soGit">
                    <img src="https://avatars.githubusercontent.com/u/10682798?v=4" width="100;" alt="soGit"/>
                    <br />
                    <sub><b>soGit</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/Huliiiiii">
                    <img src="https://avatars.githubusercontent.com/u/134658521?v=4" width="100;" alt="Huliiiiii"/>
                    <br />
                    <sub><b>Huliiiiii</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/not-my-profile">
                    <img src="https://avatars.githubusercontent.com/u/73739153?v=4" width="100;" alt="not-my-profile"/>
                    <br />
                    <sub><b>not-my-profile</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/tinchoz49">
                    <img src="https://avatars.githubusercontent.com/u/819446?v=4" width="100;" alt="tinchoz49"/>
                    <br />
                    <sub><b>tinchoz49</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/MatteoH2O1999">
                    <img src="https://avatars.githubusercontent.com/u/82184604?v=4" width="100;" alt="MatteoH2O1999"/>
                    <br />
                    <sub><b>MatteoH2O1999</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/maxehmookau">
                    <img src="https://avatars.githubusercontent.com/u/690977?v=4" width="100;" alt="maxehmookau"/>
                    <br />
                    <sub><b>maxehmookau</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/MaxBreida">
                    <img src="https://avatars.githubusercontent.com/u/18685579?v=4" width="100;" alt="MaxBreida"/>
                    <br />
                    <sub><b>MaxBreida</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/mvt147">
                    <img src="https://avatars.githubusercontent.com/u/9066177?v=4" width="100;" alt="mvt147"/>
                    <br />
                    <sub><b>mvt147</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/malinbranduse">
                    <img src="https://avatars.githubusercontent.com/u/22645119?v=4" width="100;" alt="malinbranduse"/>
                    <br />
                    <sub><b>malinbranduse</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Lazialize">
                    <img src="https://avatars.githubusercontent.com/u/3455147?v=4" width="100;" alt="Lazialize"/>
                    <br />
                    <sub><b>Lazialize</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Olexandr88">
                    <img src="https://avatars.githubusercontent.com/u/93856062?v=4" width="100;" alt="Olexandr88"/>
                    <br />
                    <sub><b>Olexandr88</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/pkucmus">
                    <img src="https://avatars.githubusercontent.com/u/6347418?v=4" width="100;" alt="pkucmus"/>
                    <br />
                    <sub><b>pkucmus</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/petercinibulk">
                    <img src="https://avatars.githubusercontent.com/u/29611892?v=4" width="100;" alt="petercinibulk"/>
                    <br />
                    <sub><b>petercinibulk</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/philsturgeon">
                    <img src="https://avatars.githubusercontent.com/u/67381?v=4" width="100;" alt="philsturgeon"/>
                    <br />
                    <sub><b>philsturgeon</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/rhuanbarreto">
                    <img src="https://avatars.githubusercontent.com/u/283004?v=4" width="100;" alt="rhuanbarreto"/>
                    <br />
                    <sub><b>rhuanbarreto</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/rickihastings">
                    <img src="https://avatars.githubusercontent.com/u/45660?v=4" width="100;" alt="rickihastings"/>
                    <br />
                    <sub><b>rickihastings</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/robert-dean">
                    <img src="https://avatars.githubusercontent.com/u/99672719?v=4" width="100;" alt="robert-dean"/>
                    <br />
                    <sub><b>robert-dean</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/rbange">
                    <img src="https://avatars.githubusercontent.com/u/13252574?v=4" width="100;" alt="rbange"/>
                    <br />
                    <sub><b>rbange</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/unnoq">
                    <img src="https://avatars.githubusercontent.com/u/64189902?v=4" width="100;" alt="unnoq"/>
                    <br />
                    <sub><b>unnoq</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/tsiwek-piwik">
                    <img src="https://avatars.githubusercontent.com/u/117373332?v=4" width="100;" alt="tsiwek-piwik"/>
                    <br />
                    <sub><b>tsiwek-piwik</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/dunklesToast">
                    <img src="https://avatars.githubusercontent.com/u/17279485?v=4" width="100;" alt="dunklesToast"/>
                    <br />
                    <sub><b>dunklesToast</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/stefanprobst">
                    <img src="https://avatars.githubusercontent.com/u/20753323?v=4" width="100;" alt="stefanprobst"/>
                    <br />
                    <sub><b>stefanprobst</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/omerp-explorium">
                    <img src="https://avatars.githubusercontent.com/u/93073822?v=4" width="100;" alt="omerp-explorium"/>
                    <br />
                    <sub><b>omerp-explorium</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/o-az">
                    <img src="https://avatars.githubusercontent.com/u/23618431?v=4" width="100;" alt="o-az"/>
                    <br />
                    <sub><b>o-az</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/milksense">
                    <img src="https://avatars.githubusercontent.com/u/29128703?v=4" width="100;" alt="milksense"/>
                    <br />
                    <sub><b>milksense</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/kwaichanz">
                    <img src="https://avatars.githubusercontent.com/u/36059761?v=4" width="100;" alt="kwaichanz"/>
                    <br />
                    <sub><b>kwaichanz</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/g-airey">
                    <img src="https://avatars.githubusercontent.com/u/35547338?v=4" width="100;" alt="g-airey"/>
                    <br />
                    <sub><b>g-airey</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ga1az">
                    <img src="https://avatars.githubusercontent.com/u/75758899?v=4" width="100;" alt="ga1az"/>
                    <br />
                    <sub><b>ga1az</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/FotieMConstant">
                    <img src="https://avatars.githubusercontent.com/u/42372656?v=4" width="100;" alt="FotieMConstant"/>
                    <br />
                    <sub><b>FotieMConstant</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/danwithabox">
                    <img src="https://avatars.githubusercontent.com/u/144792741?v=4" width="100;" alt="danwithabox"/>
                    <br />
                    <sub><b>danwithabox</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/WilliamBonvini">
                    <img src="https://avatars.githubusercontent.com/u/37834150?v=4" width="100;" alt="WilliamBonvini"/>
                    <br />
                    <sub><b>WilliamBonvini</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/bsdayo">
                    <img src="https://avatars.githubusercontent.com/u/41754841?v=4" width="100;" alt="bsdayo"/>
                    <br />
                    <sub><b>bsdayo</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Chinlinlee">
                    <img src="https://avatars.githubusercontent.com/u/49154622?v=4" width="100;" alt="Chinlinlee"/>
                    <br />
                    <sub><b>Chinlinlee</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/bingbeann">
                    <img src="https://avatars.githubusercontent.com/u/110380613?v=4" width="100;" alt="bingbeann"/>
                    <br />
                    <sub><b>bingbeann</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/zakaria-shahen">
                    <img src="https://avatars.githubusercontent.com/u/35308232?v=4" width="100;" alt="zakaria-shahen"/>
                    <br />
                    <sub><b>zakaria-shahen</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/rolldeep-stepmerrily">
                    <img src="https://avatars.githubusercontent.com/u/98504913?v=4" width="100;" alt="rolldeep-stepmerrily"/>
                    <br />
                    <sub><b>rolldeep-stepmerrily</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Wolf-Syndrome">
                    <img src="https://avatars.githubusercontent.com/u/44598104?v=4" width="100;" alt="Wolf-Syndrome"/>
                    <br />
                    <sub><b>Wolf-Syndrome</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/wilcoschoneveld">
                    <img src="https://avatars.githubusercontent.com/u/3064588?v=4" width="100;" alt="wilcoschoneveld"/>
                    <br />
                    <sub><b>wilcoschoneveld</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/rotciw">
                    <img src="https://avatars.githubusercontent.com/u/21119564?v=4" width="100;" alt="rotciw"/>
                    <br />
                    <sub><b>rotciw</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/WesleySkeen">
                    <img src="https://avatars.githubusercontent.com/u/2134761?v=4" width="100;" alt="WesleySkeen"/>
                    <br />
                    <sub><b>WesleySkeen</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Ma-ve">
                    <img src="https://avatars.githubusercontent.com/u/11227996?v=4" width="100;" alt="Ma-ve"/>
                    <br />
                    <sub><b>Ma-ve</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Mai0313">
                    <img src="https://avatars.githubusercontent.com/u/21039366?v=4" width="100;" alt="Mai0313"/>
                    <br />
                    <sub><b>Mai0313</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Courtcircuits">
                    <img src="https://avatars.githubusercontent.com/u/90451752?v=4" width="100;" alt="Courtcircuits"/>
                    <br />
                    <sub><b>Courtcircuits</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/rektdeckard">
                    <img src="https://avatars.githubusercontent.com/u/26732044?v=4" width="100;" alt="rektdeckard"/>
                    <br />
                    <sub><b>rektdeckard</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/thibaultleouay">
                    <img src="https://avatars.githubusercontent.com/u/13894054?v=4" width="100;" alt="thibaultleouay"/>
                    <br />
                    <sub><b>thibaultleouay</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/yokeTH">
                    <img src="https://avatars.githubusercontent.com/u/66236295?v=4" width="100;" alt="yokeTH"/>
                    <br />
                    <sub><b>yokeTH</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/stratosblue">
                    <img src="https://avatars.githubusercontent.com/u/20887303?v=4" width="100;" alt="stratosblue"/>
                    <br />
                    <sub><b>stratosblue</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/sinh117801">
                    <img src="https://avatars.githubusercontent.com/u/43696715?v=4" width="100;" alt="sinh117801"/>
                    <br />
                    <sub><b>sinh117801</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/DmitriiKhudiakov">
                    <img src="https://avatars.githubusercontent.com/u/148915384?v=4" width="100;" alt="DmitriiKhudiakov"/>
                    <br />
                    <sub><b>DmitriiKhudiakov</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/diegolopes">
                    <img src="https://avatars.githubusercontent.com/u/27782408?v=4" width="100;" alt="diegolopes"/>
                    <br />
                    <sub><b>diegolopes</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/untiny">
                    <img src="https://avatars.githubusercontent.com/u/29054492?v=4" width="100;" alt="untiny"/>
                    <br />
                    <sub><b>untiny</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/lundibundi">
                    <img src="https://avatars.githubusercontent.com/u/9109612?v=4" width="100;" alt="lundibundi"/>
                    <br />
                    <sub><b>lundibundi</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/DavidNiessen">
                    <img src="https://avatars.githubusercontent.com/u/68539499?v=4" width="100;" alt="DavidNiessen"/>
                    <br />
                    <sub><b>DavidNiessen</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/darekaze">
                    <img src="https://avatars.githubusercontent.com/u/32747549?v=4" width="100;" alt="darekaze"/>
                    <br />
                    <sub><b>darekaze</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/danp">
                    <img src="https://avatars.githubusercontent.com/u/2182?v=4" width="100;" alt="danp"/>
                    <br />
                    <sub><b>danp</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/danjohnson95">
                    <img src="https://avatars.githubusercontent.com/u/6871504?v=4" width="100;" alt="danjohnson95"/>
                    <br />
                    <sub><b>danjohnson95</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/damiansan239">
                    <img src="https://avatars.githubusercontent.com/u/42095620?v=4" width="100;" alt="damiansan239"/>
                    <br />
                    <sub><b>damiansan239</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ClanEver">
                    <img src="https://avatars.githubusercontent.com/u/73160783?v=4" width="100;" alt="ClanEver"/>
                    <br />
                    <sub><b>ClanEver</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/chrislearn">
                    <img src="https://avatars.githubusercontent.com/u/5874864?v=4" width="100;" alt="chrislearn"/>
                    <br />
                    <sub><b>chrislearn</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/chadwhitacre">
                    <img src="https://avatars.githubusercontent.com/u/134455?v=4" width="100;" alt="chadwhitacre"/>
                    <br />
                    <sub><b>chadwhitacre</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/carstenlebek">
                    <img src="https://avatars.githubusercontent.com/u/59960385?v=4" width="100;" alt="carstenlebek"/>
                    <br />
                    <sub><b>carstenlebek</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/cactysman">
                    <img src="https://avatars.githubusercontent.com/u/5056880?v=4" width="100;" alt="cactysman"/>
                    <br />
                    <sub><b>cactysman</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Smyrcu">
                    <img src="https://avatars.githubusercontent.com/u/42031920?v=4" width="100;" alt="Smyrcu"/>
                    <br />
                    <sub><b>Smyrcu</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/brunogrcsada">
                    <img src="https://avatars.githubusercontent.com/u/54673205?v=4" width="100;" alt="brunogrcsada"/>
                    <br />
                    <sub><b>brunogrcsada</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/brianheineman">
                    <img src="https://avatars.githubusercontent.com/u/3210678?v=4" width="100;" alt="brianheineman"/>
                    <br />
                    <sub><b>brianheineman</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Borisich">
                    <img src="https://avatars.githubusercontent.com/u/18186889?v=4" width="100;" alt="Borisich"/>
                    <br />
                    <sub><b>Borisich</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/bdpiprava">
                    <img src="https://avatars.githubusercontent.com/u/7871209?v=4" width="100;" alt="bdpiprava"/>
                    <br />
                    <sub><b>bdpiprava</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/BenJeau">
                    <img src="https://avatars.githubusercontent.com/u/22248828?v=4" width="100;" alt="BenJeau"/>
                    <br />
                    <sub><b>BenJeau</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/saithis">
                    <img src="https://avatars.githubusercontent.com/u/1547453?v=4" width="100;" alt="saithis"/>
                    <br />
                    <sub><b>saithis</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/zsilbi">
                    <img src="https://avatars.githubusercontent.com/u/3886658?v=4" width="100;" alt="zsilbi"/>
                    <br />
                    <sub><b>zsilbi</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/arthurfiorette">
                    <img src="https://avatars.githubusercontent.com/u/47537704?v=4" width="100;" alt="arthurfiorette"/>
                    <br />
                    <sub><b>arthurfiorette</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/thisisarko">
                    <img src="https://avatars.githubusercontent.com/u/169167507?v=4" width="100;" alt="thisisarko"/>
                    <br />
                    <sub><b>thisisarko</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/arashsheyda">
                    <img src="https://avatars.githubusercontent.com/u/38922203?v=4" width="100;" alt="arashsheyda"/>
                    <br />
                    <sub><b>arashsheyda</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/sigpwned">
                    <img src="https://avatars.githubusercontent.com/u/1236302?v=4" width="100;" alt="sigpwned"/>
                    <br />
                    <sub><b>sigpwned</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/gizero">
                    <img src="https://avatars.githubusercontent.com/u/1544064?v=4" width="100;" alt="gizero"/>
                    <br />
                    <sub><b>gizero</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/alewin">
                    <img src="https://avatars.githubusercontent.com/u/980844?v=4" width="100;" alt="alewin"/>
                    <br />
                    <sub><b>alewin</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/slash686">
                    <img src="https://avatars.githubusercontent.com/u/4161770?v=4" width="100;" alt="slash686"/>
                    <br />
                    <sub><b>slash686</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ahmadi-akbar">
                    <img src="https://avatars.githubusercontent.com/u/48069695?v=4" width="100;" alt="ahmadi-akbar"/>
                    <br />
                    <sub><b>ahmadi-akbar</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/MathurAditya724">
                    <img src="https://avatars.githubusercontent.com/u/57684218?v=4" width="100;" alt="MathurAditya724"/>
                    <br />
                    <sub><b>MathurAditya724</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/abdulamite">
                    <img src="https://avatars.githubusercontent.com/u/7246246?v=4" width="100;" alt="abdulamite"/>
                    <br />
                    <sub><b>abdulamite</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/ShadiestGoat">
                    <img src="https://avatars.githubusercontent.com/u/48590492?v=4" width="100;" alt="ShadiestGoat"/>
                    <br />
                    <sub><b>ShadiestGoat</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/mrlubos">
                    <img src="https://avatars.githubusercontent.com/u/12529395?v=4" width="100;" alt="mrlubos"/>
                    <br />
                    <sub><b>mrlubos</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/sanscontext">
                    <img src="https://avatars.githubusercontent.com/u/17016388?v=4" width="100;" alt="sanscontext"/>
                    <br />
                    <sub><b>sanscontext</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/kyllian">
                    <img src="https://avatars.githubusercontent.com/u/5831233?v=4" width="100;" alt="kyllian"/>
                    <br />
                    <sub><b>kyllian</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/KevSlashNull">
                    <img src="https://avatars.githubusercontent.com/u/28510368?v=4" width="100;" alt="KevSlashNull"/>
                    <br />
                    <sub><b>KevSlashNull</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/jlnslv">
                    <img src="https://avatars.githubusercontent.com/u/323004?v=4" width="100;" alt="jlnslv"/>
                    <br />
                    <sub><b>jlnslv</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/IHIutch">
                    <img src="https://avatars.githubusercontent.com/u/20825047?v=4" width="100;" alt="IHIutch"/>
                    <br />
                    <sub><b>IHIutch</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/jonataw">
                    <img src="https://avatars.githubusercontent.com/u/29772763?v=4" width="100;" alt="jonataw"/>
                    <br />
                    <sub><b>jonataw</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Jannchie">
                    <img src="https://avatars.githubusercontent.com/u/29743310?v=4" width="100;" alt="Jannchie"/>
                    <br />
                    <sub><b>Jannchie</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/jwangnz">
                    <img src="https://avatars.githubusercontent.com/u/14569?v=4" width="100;" alt="jwangnz"/>
                    <br />
                    <sub><b>jwangnz</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Jefwillems">
                    <img src="https://avatars.githubusercontent.com/u/10141933?v=4" width="100;" alt="Jefwillems"/>
                    <br />
                    <sub><b>Jefwillems</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/Tholdrim">
                    <img src="https://avatars.githubusercontent.com/u/38507238?v=4" width="100;" alt="Tholdrim"/>
                    <br />
                    <sub><b>Tholdrim</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/JacobCoffee">
                    <img src="https://avatars.githubusercontent.com/u/45884264?v=4" width="100;" alt="JacobCoffee"/>
                    <br />
                    <sub><b>JacobCoffee</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/kevinand11">
                    <img src="https://avatars.githubusercontent.com/u/48160414?v=4" width="100;" alt="kevinand11"/>
                    <br />
                    <sub><b>kevinand11</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/tltsutltsu">
                    <img src="https://avatars.githubusercontent.com/u/83283675?v=4" width="100;" alt="tltsutltsu"/>
                    <br />
                    <sub><b>tltsutltsu</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/eltociear">
                    <img src="https://avatars.githubusercontent.com/u/22633385?v=4" width="100;" alt="eltociear"/>
                    <br />
                    <sub><b>eltociear</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/IceyWu">
                    <img src="https://avatars.githubusercontent.com/u/66096254?v=4" width="100;" alt="IceyWu"/>
                    <br />
                    <sub><b>IceyWu</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/iagobalmeida">
                    <img src="https://avatars.githubusercontent.com/u/35234046?v=4" width="100;" alt="iagobalmeida"/>
                    <br />
                    <sub><b>iagobalmeida</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/hpmouton">
                    <img src="https://avatars.githubusercontent.com/u/61012928?v=4" width="100;" alt="hpmouton"/>
                    <br />
                    <sub><b>hpmouton</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/CJHwong">
                    <img src="https://avatars.githubusercontent.com/u/906057?v=4" width="100;" alt="CJHwong"/>
                    <br />
                    <sub><b>CJHwong</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/HelgeSverre">
                    <img src="https://avatars.githubusercontent.com/u/1089652?v=4" width="100;" alt="HelgeSverre"/>
                    <br />
                    <sub><b>HelgeSverre</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Hamidrzash">
                    <img src="https://avatars.githubusercontent.com/u/69453538?v=4" width="100;" alt="Hamidrzash"/>
                    <br />
                    <sub><b>Hamidrzash</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/255kb">
                    <img src="https://avatars.githubusercontent.com/u/7489814?v=4" width="100;" alt="255kb"/>
                    <br />
                    <sub><b>255kb</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/GreyXor">
                    <img src="https://avatars.githubusercontent.com/u/79602273?v=4" width="100;" alt="GreyXor"/>
                    <br />
                    <sub><b>GreyXor</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/galah92">
                    <img src="https://avatars.githubusercontent.com/u/14138433?v=4" width="100;" alt="galah92"/>
                    <br />
                    <sub><b>galah92</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/Fdawgs">
                    <img src="https://avatars.githubusercontent.com/u/43814140?v=4" width="100;" alt="Fdawgs"/>
                    <br />
                    <sub><b>Fdawgs</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/flovouin">
                    <img src="https://avatars.githubusercontent.com/u/5155149?v=4" width="100;" alt="flovouin"/>
                    <br />
                    <sub><b>flovouin</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/filiabel">
                    <img src="https://avatars.githubusercontent.com/u/25987724?v=4" width="100;" alt="filiabel"/>
                    <br />
                    <sub><b>filiabel</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/facus26">
                    <img src="https://avatars.githubusercontent.com/u/18079059?v=4" width="100;" alt="facus26"/>
                    <br />
                    <sub><b>facus26</b></sub>
                </a>
            </td>
		</tr>
		<tr>
            <td align="center">
                <a href="https://github.com/emmanuel-ferdman">
                    <img src="https://avatars.githubusercontent.com/u/35470921?v=4" width="100;" alt="emmanuel-ferdman"/>
                    <br />
                    <sub><b>emmanuel-ferdman</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/elliott-with-the-longest-name-on-github">
                    <img src="https://avatars.githubusercontent.com/u/76245373?v=4" width="100;" alt="elliott-with-the-longest-name-on-github"/>
                    <br />
                    <sub><b>elliott-with-the-longest-name-on-github</b></sub>
                </a>
            </td>
            <td align="center">
                <a href="https://github.com/elliotnash">
                    <img src="https://avatars.githubusercontent.com/u/53949099?v=4" width="100;" alt="elliotnash"/>
                    <br />
                    <sub><b>elliotnash</b></sub>
                </a>
            </td>
		</tr>
	<tbody>
</table>
<!-- readme: collaborators,contributors -end -->
 readmeEtag: '"ec4ca5123ddd0c9f52ff893937e96c65c38a24af"' readmeLastModified: Sat, 31 Jan 2026 01:10:12 GMT repositoryId: 679280065 description: >- Scalar is an open-source API platform:                                       🌐 Modern Rest API Client                                        📖 Beautiful API References                                        ✨ 1st-Class OpenAPI/Swagger Support created: '2023-08-16T13:42:32Z' updated: '2026-02-06T02:22:15Z' language: TypeScript archived: false stars: 13768 watchers: 33 forks: 727 owner: scalar logo: https://avatars.githubusercontent.com/u/301879?v=4 license: MIT repoEtag: '"54e5bfe76d85a3b0d91eb36e2a2baa5eb370a254d4178f7fb4d363ce49bcb3c3"' repoLastModified: Fri, 06 Feb 2026 02:22:15 GMT foundInMaster: true - source: https://openapi.tools/ name: Scramble category: - Auto Generators - Documentation - Server Implementations language: PHP repository: https://github.com/dedoc/scramble link: https://scramble.dedoc.co/ source_description: >- Modern Laravel OpenAPI documentation generator. No PHPDoc annotations required. v2: false v3: false v3_1: true id: 98f20578b2c52e7c4116e9fa61000f60 repositoryMetadata: base64Readme: >- PHA+CiAgPGEgaHJlZj0iaHR0cHM6Ly9zY3JhbWJsZS5kZWRvYy5jbyIgdGFyZ2V0PSJfYmxhbmsiPgogICAgPGltZyBzcmM9Ii4vLmdpdGh1Yi9naC1pbWcucG5nP3Y9MSIgYWx0PSJTY3JhbWJsZSDigJMgTGFyYXZlbCBBUEkgZG9jdW1lbnRhdGlvbiBnZW5lcmF0b3IiLz4KICA8L2E+CjwvcD4KCiMgU2NyYW1ibGUKClNjcmFtYmxlIGdlbmVyYXRlcyBBUEkgZG9jdW1lbnRhdGlvbiBmb3IgTGFyYXZlbCBwcm9qZWN0LiBXaXRob3V0IHJlcXVpcmluZyB5b3UgdG8gbWFudWFsbHkgd3JpdGUgUEhQRG9jIGFubm90YXRpb25zLiBEb2NzIGFyZSBnZW5lcmF0ZWQgaW4gT3BlbkFQSSAzLjEuMCBmb3JtYXQuCgojIyBEb2N1bWVudGF0aW9uCgpZb3UgY2FuIGZpbmQgZnVsbCBkb2N1bWVudGF0aW9uIGF0IFtzY3JhbWJsZS5kZWRvYy5jb10oaHR0cHM6Ly9zY3JhbWJsZS5kZWRvYy5jbykuCgojIyBJbnRyb2R1Y3Rpb24KClRoZSBtYWluIG1vdHRvIG9mIHRoZSBwcm9qZWN0IGlzIGdlbmVyYXRpbmcgeW91ciBBUEkgZG9jdW1lbnRhdGlvbiB3aXRob3V0IHJlcXVpcmluZyB5b3UgdG8gYW5ub3RhdGUgeW91ciBjb2RlLgoKVGhpcyBhbGxvd3MgeW91IHRvIGZvY3VzIG9uIGNvZGUgYW5kIGF2b2lkIGFubm90YXRpbmcgZXZlcnkgcG9zc2libGUgcGFyYW0vZmllbGQgYXMgaXQgbWF5IHJlc3VsdCBpbiBvdXRkYXRlZCBkb2N1bWVudGF0aW9uLiBCeSBnZW5lcmF0aW5nIGRvY3MgYXV0b21hdGljYWxseSBmcm9tIHRoZSBjb2RlIHlvdXIgQVBJIHdpbGwgYWx3YXlzIGhhdmUgdXAtdG8tZGF0ZSBkb2NzIHdoaWNoIHlvdSBjYW4gdHJ1c3QuCgojIyBJbnN0YWxsYXRpb24KWW91IGNhbiBpbnN0YWxsIHRoZSBwYWNrYWdlIHZpYSBjb21wb3NlcjoKYGBgc2hlbGwKY29tcG9zZXIgcmVxdWlyZSBkZWRvYy9zY3JhbWJsZQpgYGAKCiMjIFVzYWdlCkFmdGVyIGluc3RhbGwgeW91IHdpbGwgaGF2ZSAyIHJvdXRlcyBhZGRlZCB0byB5b3VyIGFwcGxpY2F0aW9uOgoKLSBgL2RvY3MvYXBpYCAtIFVJIHZpZXdlciBmb3IgeW91ciBkb2N1bWVudGF0aW9uCi0gYC9kb2NzL2FwaS5qc29uYCAtIE9wZW4gQVBJIGRvY3VtZW50IGluIEpTT04gZm9ybWF0IGRlc2NyaWJpbmcgeW91ciBBUEkuCgpCeSBkZWZhdWx0LCB0aGVzZSByb3V0ZXMgYXJlIGF2YWlsYWJsZSBvbmx5IGluIGBsb2NhbGAgZW52aXJvbm1lbnQuIFlvdSBjYW4gY2hhbmdlIHRoaXMgYmVoYXZpb3IgW2J5IGRlZmluaW5nIGB2aWV3QXBpRG9jc2AgZ2F0ZV0oaHR0cHM6Ly9zY3JhbWJsZS5kZWRvYy5jby91c2FnZS9nZXR0aW5nLXN0YXJ0ZWQjZG9jcy1hdXRob3JpemF0aW9uKS4KCi0tLQoKPHA+CiAgPGEgaHJlZj0iaHR0cHM6Ly9zYXZlbGlmZS5pbi51YS9lbi9kb25hdGUtZW4vIiB0YXJnZXQ9Il9ibGFuayI+CiAgICA8aW1nIHNyYz0iLi8uZ2l0aHViL2doLXByb21vLnN2Zz92PTEiIGFsdD0iRG9uYXRlIi8+CiAgPC9hPgo8L3A+IAo= readmeEtag: '"a7be54662226c74bd5b97fd6b3fe81ece1ea59f0"' readmeLastModified: Fri, 04 Jul 2025 12:37:45 GMT repositoryId: 521720102 description: >- Modern Laravel OpenAPI (Swagger) documentation generator. No PHPDoc annotations required. created: '2022-08-05T17:16:20Z' updated: '2026-02-05T08:32:29Z' language: PHP archived: false stars: 1965 watchers: 12 forks: 186 owner: dedoc logo: https://avatars.githubusercontent.com/u/93313138?v=4 license: MIT repoEtag: '"6416e6b8ce4c0768630ec26b8591e41acb25ab1561a26ae326815c5ed739e404"' repoLastModified: Thu, 05 Feb 2026 08:32:29 GMT foundInMaster: true - source: https://openapi.tools/ name: Scribe category: Documentation link: https://scribe.knuckles.wtf/ repository: https://github.com/knuckleswtf/scribe/ language: PHP source_description: Generate API documentation for humans from your Laravel codebase. v2: false v3: true v3_1: false id: a13f5fae7efc680dc1e5390220f1db3d repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KPGltZyBzcmM9ImxvZ28tc2NyaWJlLnBuZyIgYWx0PSJsb2dvLXNjcmliZSI+PGJyPgoKWyFbTGF0ZXN0IFN0YWJsZSBWZXJzaW9uXShodHRwczovL3Bvc2VyLnB1Z3gub3JnL2tudWNrbGVzd3RmL3NjcmliZS92L3N0YWJsZSldKGh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9rbnVja2xlc3d0Zi9zY3JpYmUpClshW1RvdGFsIERvd25sb2Fkc10oaHR0cHM6Ly9wb3Nlci5wdWd4Lm9yZy9rbnVja2xlc3d0Zi9zY3JpYmUvZG93bmxvYWRzKV0oaHR0cHM6Ly9wYWNrYWdpc3Qub3JnL3BhY2thZ2VzL2tudWNrbGVzd3RmL3NjcmliZSkKWyFbQ29kZSBTdHlsZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlX3N0eWxlLXBpbnQtZjU4ZDMzKV0oaHR0cHM6Ly9naXRodWIuY29tL2xhcmF2ZWwvcGludCkKCjwvZGl2PgoKClNjcmliZSBoZWxwcyB5b3UgZ2VuZXJhdGUgQVBJIGRvY3VtZW50YXRpb24gZm9yIGh1bWFucyBmcm9tIHlvdXIgTGFyYXZlbCBjb2RlYmFzZS4gU2VlIGEgbGl2ZSBleGFtcGxlIGF0IFtkZW1vLnNjcmliZS5rbnVja2xlcy53dGZdKGh0dHBzOi8vZGVtby5zY3JpYmUua251Y2tsZXMud3RmKS4KCiMjIEZlYXR1cmVzCi0gVXNlZnVsIG91dHB1dDoKICAtIFByZXR0eSBzaW5nbGUtcGFnZSBIVE1MIGRvYywgd2l0aCBodW1hbi1mcmllbmRseSB0ZXh0LCBjb2RlIHNhbXBsZXMsIGFuZCBpbi1icm93c2VyIEFQSSB0ZXN0ZXIgKCJUcnkgSXQgT3V0IikKICAtIEdlbmVyYXRlcyBQb3N0bWFuIGNvbGxlY3Rpb24gYW5kIE9wZW5BUEkgc3BlYyAodjMuMC4zIG9yIHYzLjEuMCkKLSBTbWFydHMuIFNjcmliZSBjYW46CiAgLSBleHRyYWN0IHJlcXVlc3QgcGFyYW1ldGVyIGRldGFpbHMgZnJvbSBGb3JtUmVxdWVzdHMgb3IgdmFsaWRhdGlvbiBydWxlcwogIC0gc2FmZWx5IGNhbGwgQVBJIGVuZHBvaW50cyB0byBnZXQgc2FtcGxlIHJlc3BvbnNlcwogIC0gZ2VuZXJhdGUgc2FtcGxlIHJlc3BvbnNlcyBmcm9tIEVsb3F1ZW50IEFQSSBSZXNvdXJjZXMgb3IgVHJhbnNmb3JtZXJzCi0gQ3VzdG9taXphYmxlIHRvIGRpZmZlcmVudCBsZXZlbHM6CiAgLSBDdXN0b21pemUgdGhlIFVJIGJ5IGFkanVzdGluZyB0ZXh0LCBvcmRlcmluZywgZXhhbXBsZXMsIG9yIGNoYW5naW5nIHRoZSBVSSBpdHNlbGYKICAtIEFkZCBjdXN0b20gc3RyYXRlZ2llcyB0byBhZGp1c3QgaG93IGRhdGEgaXMgZXh0cmFjdGVkCiAgLSBTdGF0aWNhbGx5IGRlZmluZSBleHRyYSBlbmRwb2ludHMgb3IgaW5mb3JtYXRpb24gbm90IGluIHlvdXIgY29kZWJhc2UKCj4gWyFUSVBdCj4g8J+RiyBTY3JpYmUgaGVscHMgeW91IGdlbmVyYXRlIGRvY3MgYXV0b21hdGljYWxseSwgYnV0IGlmIHlvdSByZWFsbHkgd2FudCB0byBtYWtlIGZyaWVuZGx5LCBtYWludGFpbmFibGUsIGFuZCB0ZXN0YWJsZSBBUEkgZG9jcywgdGhlcmUgYXJlIHNvbWUgbW9yZSB0aGluZ3MgeW91IG5lZWQgdG8ga25vdy4KPiBTbyBJIG1hZGUgW2EgY291cnNlXShodHRwczovL3NoYWx2YWgudGVhY2hhYmxlLmNvbS9wL2FwaS1kb2N1bWVudGF0aW9uLWZvci1kZXZlbG9wZXJzP3V0bV9zb3VyY2U9c2NyaWJlLWxhcmF2ZWwmdXRtX21lZGl1bT1yZWZlcnJhbCZ1dG1fY2FtcGFpZ249bm9uZSkgZm9yIHlvdS7wn6SXCgojIyBEb2N1bWVudGF0aW9uCkNoZWNrIG91dCB0aGUgZG9jdW1lbnRhdGlvbiBhdCBbc2NyaWJlLmtudWNrbGVzLnd0Zi9sYXJhdmVsXShodHRwOi8vc2NyaWJlLmtudWNrbGVzLnd0Zi9sYXJhdmVsKS4KCiMjIENvbnRyaWJ1dGluZwpDb250cmlidXRpbmcgaXMgZWFzeSEgU2VlIG91ciBbY29udHJpYnV0aW9uIGd1aWRlXShodHRwczovL3NjcmliZS5rbnVja2xlcy53dGYvbGFyYXZlbC9jb250cmlidXRpbmcpLgo= readmeEtag: '"b316cbbeea5323d3874e708d814c13388c02a4ad"' readmeLastModified: Wed, 04 Feb 2026 20:49:04 GMT repositoryId: 260729799 description: Generate API documentation for humans from your Laravel codebase.✍ created: '2020-05-02T16:36:45Z' updated: '2026-02-05T04:55:41Z' language: PHP archived: false stars: 2245 watchers: 15 forks: 357 owner: knuckleswtf logo: https://avatars.githubusercontent.com/u/63367037?v=4 license: MIT repoEtag: '"12c50897b3ffbcdd1e8a5e800f98789080f33fc17713a99d1691be85761a7fed"' repoLastModified: Thu, 05 Feb 2026 04:55:41 GMT foundInMaster: true - source: https://openapi.tools/ name: Supermodel category: DSL language: SaaS source_description: >- Model your data using JSON Schema, refer and remix the models freely, convert to various formats including OAS v2/v3. link: https://supermodel.io v2: true v3: true foundInMaster: true id: b839c288f46fd1ef91135b5544dc4aef - source: https://openapi.tools/ name: ReadMe category: Documentation link: https://readme.com repository: https://github.com/readmeio language: SaaS source_description: Interactive developer hubs that help users succeed with your APIs 🦉 v2: true v3: true v3_1: true id: e2e7cc265f4d5143a413ce6930c59d2d foundInMaster: true - source: https://openapi.tools/ name: ReDoc category: - Documentation - Parsers language: React.js repository: https://github.com/redocly/redoc/ link: https://redocly.github.io/redoc/ source_description: Open source API reference docs from OpenAPI descriptions v2: true v3: true v3_1: true id: fb3dbdf9a2210ae5f30346b6c0094df3 repositoryMetadata: base64Readme: >- <div align="center">
  <img alt="Redoc logo" src="https://raw.githubusercontent.com/Redocly/redoc/main//docs/images/redoc.png" width="400px" />

# Generate beautiful API documentation from OpenAPI

  [![npm](http://img.shields.io/npm/v/redoc.svg)](https://www.npmjs.com/package/redoc) [![License](https://img.shields.io/npm/l/redoc.svg)](https://github.com/Redocly/redoc/blob/main/LICENSE)

  [![bundle size](http://img.badgesize.io/https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js?compression=gzip&max=300000)](https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js) [![npm](https://img.shields.io/npm/dm/redoc.svg)](https://www.npmjs.com/package/redoc) [![jsDelivr status](https://data.jsdelivr.com/v1/package/npm/redoc/badge)](https://www.jsdelivr.com/package/npm/redoc)
</div>


## About Redoc

Redoc is an open source tool for generating documentation from OpenAPI (formerly Swagger) definitions.

By default Redoc offers a three-panel, responsive layout:

- The left panel contains a search bar and navigation menu.
- The central panel contains the documentation.
- The right panel contains request and response examples.

![Redoc demo](https://raw.githubusercontent.com/Redocly/redoc/main/demo/redoc-demo.png)

## Live demo

If you want to see how Redoc renders your OpenAPI definition,
you can try it out online at https://redocly.github.io/redoc/.

A version of the Swagger Petstore API is displayed by default.
To test it with your own OpenAPI definition,
enter the URL for your definition and select **TRY IT**.

## Redoc features

- Responsive three-panel design with menu/scrolling synchronization
- Support for OpenAPI 3.1, OpenAPI 3.0, and Swagger 2.0
- Ability to integrate your API introduction into the side menu
- High-level grouping in side menu with the [`x-tagGroups`](https://redocly.com/docs/api-reference-docs/specification-extensions/x-tag-groups/) specification extension
- [Simple integration with `create-react-app`](https://redocly.com/docs/redoc/quickstart/react/)
- Code samples support (with vendor extension) <br>
  ![code samples in action](docs/images/code-samples-demo.gif)

## Usage

Redoc is provided as a CLI tool (also distributed as a Docker image), HTML tag, and React component.

### Generate documentation from the CLI

If you have Node installed, quickly generate documentation using `npx`:

```bash
npx @redocly/cli build-docs openapi.yaml
```

The tool outputs by default to a file named `redoc-static.html` that you can open in your browser.

> [Redocly CLI](https://github.com/Redocly/redocly-cli/) does more than docs; check it out and add linting, bundling, and more to your API workflow.

### Add an HTML element to the page

Create an HTML page, or edit an existing one, and add the following within the body tags:

```html
    <redoc spec-url="http://petstore.swagger.io/v2/swagger.json"></redoc>
    <script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"> </script>
```

Open the HTML file in your browser, and your API documentation is shown on the page.

Add your own `spec-url` to the `<redoc>` tag; this attribute can also be a local file. The JavaScript library can also be installed locally using `npm` and served from your own server, see the [HTML deployment documentation](https://redocly.com/docs/redoc/deployment/html/) for more details.

### More usage options

Check out the [deployment documentation](./docs/deployment/intro.md) for more options, and detailed documentation for each.

## Redoc vs. Redocly API Reference

Redoc is Redocly's community-edition product. Looking for something more?
We also offer [hosted API reference documentation](https://redocly.com/docs/api-registry/guides/api-registry-quickstart/)
with additional features including:

* Try-it console
* Automated code samples
* Pagination
* Extra theme options

### Documentation and resources

- [Reference docs](https://redocly.com/docs/api-reference-docs/getting-started/) - we take care of the hosting
- [Redoc](https://redocly.com/docs/redoc/) - detailed documentation for this open source project (also in the `docs/` folder)
- [Command-line interface to bundle your docs into a web-ready HTML file](https://redocly.com/docs/cli/commands/build-docs/)
- API linting, bundling, and much more with open source [Redocly CLI](https://redocly.com/docs/cli)

## Showcase

A sample of the organizations using Redocly tools in the wild:

- [Rebilly](https://api-reference.rebilly.com/)
- [Docker Engine](https://docs.docker.com/engine/api/v1.25/)
- [Zuora](https://www.zuora.com/developer/api-reference/)
- [Discourse](http://docs.discourse.org)
- [Commbox](https://www.commbox.io/api/)
- [APIs.guru](https://apis.guru/api-doc/)
- [BoxKnight](https://www.docs.boxknight.com/)
- [Quaderno API](https://developers.quaderno.io/api)

_Pull requests to add your own API page to the list are welcome_

## Configuration

Redoc is highly configurable, see the [configuration documentation](docs/config.md) for details.

### OpenAPI specification extensions
Redoc uses the following [specification extensions](https://redocly.com/docs/api-reference-docs/spec-extensions/):

* [`x-logo`](docs/redoc-vendor-extensions.md#x-logo) - is used to specify API logo
* [`x-traitTag`](docs/redoc-vendor-extensions.md#x-traitTag) - useful for tags that refer to non-navigation properties like Pagination, Rate-Limits, etc
* [`x-codeSamples`](docs/redoc-vendor-extensions.md#x-codeSamples) - specify operation code samples
* [`x-badges`](docs/redoc-vendor-extensions.md#x-badges) - specify operation badges
* [`x-examples`](docs/redoc-vendor-extensions.md#x-examples) - specify JSON example for requests
* [`x-nullable`](docs/redoc-vendor-extensions.md#x-nullable) - mark schema param as a nullable
* [`x-displayName`](docs/redoc-vendor-extensions.md#x-displayname) - specify human-friendly names for the menu categories
* [`x-tagGroups`](docs/redoc-vendor-extensions.md#x-tagGroups) - group tags by categories in the side menu
* [`x-servers`](docs/redoc-vendor-extensions.md#x-servers) - ability to specify different servers for API (backported from OpenAPI 3.0)
* [`x-additionalPropertiesName`](docs/redoc-vendor-extensions.md#x-additionalPropertiesName) - ability to supply a descriptive name for the additional property keys
* [`x-summary`](docs/redoc-vendor-extensions.md#x-summary) - for Response object, use as the response button text, with description rendered under the button
* [`x-explicitMappingOnly`](docs/redoc-vendor-extensions.md#x-explicitMappingOnly) - in Schemas, display a more descriptive property name in objects with additionalProperties when viewing the property list with an object

## Releases

**The README for the `1.x` version is on the [v1.x](https://github.com/Redocly/redoc/tree/v1.x) branch.**

All the 2.x releases are deployed to npm and can be used with Redocly-cdn:
- particular release, for example, `v2.0.0`: https://cdn.redoc.ly/redoc/v2.0.0/bundles/redoc.standalone.js
- `latest` release: https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js

Additionally, all the 1.x releases are hosted on our GitHub Pages-based CDN **(deprecated)**:
- particular release, for example `v1.2.0`: https://rebilly.github.io/ReDoc/releases/v1.2.0/redoc.min.js
- `v1.x.x` release: https://rebilly.github.io/ReDoc/releases/v1.x.x/redoc.min.js
- `latest` release: https://rebilly.github.io/ReDoc/releases/latest/redoc.min.js - points to latest 1.x.x release since 2.x releases are not hosted on this CDN but on unpkg.


## Development
see [CONTRIBUTING.md](.github/CONTRIBUTING.md)
 readmeEtag: '"cb79ec89da4372119bc45d6ca47222a7458168f4"' readmeLastModified: Thu, 30 Jan 2025 12:23:07 GMT repositoryId: 45250726 description: 📘 OpenAPI/Swagger-generated API Reference Documentation created: '2015-10-30T12:53:02Z' updated: '2026-02-06T02:08:40Z' language: TypeScript archived: false stars: 25471 watchers: 295 forks: 2379 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"4138d7edf9238b8142544691e7c435f9ca1ad9293b21d75f82a76a580e7c29fa"' repoLastModified: Fri, 06 Feb 2026 02:08:40 GMT foundInMaster: true - source: https://openapi.tools/ name: RestCase Docs category: Documentation link: https://www.restcase.com/platform/docs language: SaaS source_description: >- An API-first and security-first management platform. Design visually and we will create a beautiful API documentation for your APIs. v2: true v3: true foundInMaster: true id: be441b02201591f69504f5e3c7a1e9ca - source: - https://openapi.tools/ - openapi3 tags name: Nexmo OAS Renderer category: Documentation repository: https://github.com/nexmo/nexmo-oas-renderer language: Ruby source_description: Ruby OpenAPI docs rendering, use standalone or add to your Rails app v3: true repositoryMetadata: base64Readme: >- IyBOZXhtbyBPQVMgUmVuZGVyZXIKCiFbQnVpbGQgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vTmV4bW8vbmV4bW8tb2FzLXJlbmRlcmVyL3dvcmtmbG93cy9DSS9iYWRnZS5zdmcpClshW01JVCBsaWNlbnNlZF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2ZyldKC4vTElDRU5TRS50eHQpCgpTaW5hdHJhIGFwcGxpY2F0aW9uIHRoYXQgcHJvdmlkZXMgYSBwcmV2aWV3IG9mIGhvdyB0aGUgT0FTIGRvY3VtZW50cyB3aWxsIGJlIHJlbmRlcmVkIHdpdGhpbiBbTmV4bW8gRGV2ZWxvcGVyXShodHRwczovL2RldmVsb3Blci5uZXhtby5jb20vKS4KCiogW0RlcGVuZGVuY2llc10oI3JlcXVpcmVtZW50cykKKiBbSW5zdGFsbGF0aW9uIGFuZCBVc2FnZV0oI2luc3RhbGxhdGlvbi1hbmQtdXNhZ2UpCiAgICAqIFtVc2luZyBEb2NrZXJdKCN1c2luZy1kb2NrZXIpCiAgICAqIFtBcyBhIHN0YW5kYWxvbmUgYXBwbGljYXRpb25dKCNhcy1hLXN0YW5kYWxvbmUtYXBwbGljYXRpb24pCiAgICAqIFtNb3VudGVkIGludG8gYSBSYWlscyBhcHBsaWNhdGlvbl0oI21vdW50ZWQtaW50by1hLXJhaWxzLWFwcGxpY2F0aW9uKQogICAgKiBbU3BlY2lmeWluZyB0aGUgcGF0aCB0byB0aGUgZG9jdW1lbnRzXSgjc3BlY2lmeWluZy10aGUtcGF0aC10by10aGUtZG9jdW1lbnRzKQoqIFtDb250cmlidXRpbmddKCNjb250cmlidXRpbmcpCiogW0xpY2Vuc2VdKCNsaWNlbnNlKQoKIyMgRGVwZW5kZW5jaWVzCgoKIyMgSW5zdGFsbGF0aW9uIGFuZCBVc2FnZQoKIyMjIFVzaW5nIERvY2tlcgoKWW91IGNhbiBydW4gdXNpbmcgRG9ja2VyIGFuZCBpdCB3aWxsIHNlcnZlIHRoZSBjdXJyZW50IGRpcmVjdG9yeSAodGhpcyB3aWxsIHVzdWFsbHkgYmUgdGhlIGFwaS1zcGVjaWZpY2F0aW9uIHJlcG8pOgoKIyMjIyBNYWMvTGludXgKCmBgYGJhc2gKZG9ja2VyIHJ1biAtLXJtIC1wIDQ1Njc6NDU2NyAtdiBgcHdkYDovZGVmaW5pdGlvbnMgLWUgJ09BU19QQVRIPS9kZWZpbml0aW9ucycgbmV4bW9kZXYvbmV4bW8tb2FzLXJlbmRlcmVyOmxhdGVzdApgYGAKCkFsdGVybmF0aXZlbHksIGFkZCB0aGUgZm9sbG93aW5nIHRvIHlvdXIgYH4vLmJhc2hyY2AgZmlsZSBhbmQgeW91J2xsIGJlIGFibGUgdG8gcnVuIGBuZXhtby1vYXMtcmVuZGVyZXJgCgpgYGAKZnVuY3Rpb24gbmV4bW8tb2FzLXJlbmRlcmVyKCkgewogIGRvY2tlciBydW4gLS1ybSAtcCA0NTY3OjQ1NjcgLXYgYHB3ZGA6L2RlZmluaXRpb25zIC1lICdPQVNfUEFUSD0vZGVmaW5pdGlvbnMnIG5leG1vZGV2L25leG1vLW9hcy1yZW5kZXJlcjpsYXRlc3QKfQpgYGAKCiMjIyMgV2luZG93cwoKYGBgcHMKZG9ja2VyIHJ1biAtLXJtIC1wIDQ1Njc6NDU2NyAtdiAlQ0QlOi9kZWZpbml0aW9ucyAtZSAnT0FTX1BBVEg9L2RlZmluaXRpb25zJyBuZXhtb2Rldi9uZXhtby1vYXMtcmVuZGVyZXI6bGF0ZXN0CmBgYAoKIyMjIEFzIGEgc3RhbmRhbG9uZSBhcHBsaWNhdGlvbgoKSW5zdGFsbCB0aGUgZ2VtOgoKYGBgIHNoZWxsCiQgZ2VtIGluc3RhbGwgbmV4bW8tb2FzLXJlbmRlcmVyCmBgYAoKQW5kIHNpbXBseSBydW4gdGhlIGV4ZWN1dGFibGUgd2l0aCB0aGUgY29ycmVzcG9uZGluZyBlbnYgdmFyaWFibGVzIHNldCAoc2VlIFtOb3RlXSgjbm90ZSkpOgpgYGAgc2hlbGwKJCBuZXhtby1vYXMtcmVuZGVyZXIKYGBgCgpPciwgaWYgeW91IHdhbnQgdG8gbWFrZSBjb2RlIGNoYW5nZXMsIGp1c3QgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB3aGljaCB3aWxsIHN0YXJ0IHRoZSB3ZWIgc2VydmVyIG9uIGh0dHA6Ly9sb2NhbGhvc3Q6OTM5MzoKYGBgIHNoZWxsCk9BU19QQVRIPS4uLiBidW5kbGUgZXhlYyBzaG90Z3VuIGxpYi9uZXhtby9vYXMvcmVuZGVyZXIvY29uZmlnLnJ1CmBgYAoKIyMjIE1vdW50ZWQgaW50byBhIFJhaWxzIEFwcGxpY2F0aW9uCgpBZGQgdGhpcyB0byB5b3VyIGFwcGxpY2F0aW9uJ3MgYEdlbWZpbGVgOgoKYGBgCmdlbSAnbmV4bW8tb2FzLXJlbmRlcmVyJywgcmVxdWlyZTogZmFsc2UKYGBgCgpBbmQgdGhlbiBydW4gYGJ1bmRsZSBpbnN0YWxsYC4KClJlcXVpcmUgdGhlIGdlbSBpbiBgY29uZmlnL2Vudmlyb25tZW50LnJiYCBhcyBmb2xsb3dzOgoKYGBgIHJ1YnkKIyBMb2FkIHRoZSBSYWlscyBhcHBsaWNhdGlvbi4KcmVxdWlyZV9yZWxhdGl2ZSAnYXBwbGljYXRpb24nCgpyZXF1aXJlICduZXhtby9vYXMvcmVuZGVyZXInCgojIEluaXRpYWxpemUgdGhlIFJhaWxzIGFwcGxpY2F0aW9uLgpSYWlscy5hcHBsaWNhdGlvbi5pbml0aWFsaXplIQpgYGAKCkFuZCBmaW5hbGx5IG1vdW50IHRoZSBTaW5hdHJhIGFwcCBpbnRvIHlvdXIgUmFpbHMgYXBwbGljYXRpb24gYnkgYWRkaW5nIHRoZSBmb2xsb3dpbmcgbGluZSB0byBgY29uZmlnL3JvdXRlcy5yYmA6CgpgYGAgcnVieQptb3VudCBOZXhtbzo6T0FTOjpSZW5kZXJlcjo6QVBJLCBhdDogJy9hcGknCmBgYAoKIyMjIFNwZWNpZnlpbmcgdGhlIHBhdGggdG8gdGhlIGRvY3VtZW50cwoKVGhpcyBnZW0gdXNlcyBbZG90ZW52XShodHRwczovL2dpdGh1Yi5jb20vYmtlZXBlcnMvZG90ZW52KSB0byBoYW5kbGUgZW52aXJvbm1lbnQgdmFyaWFibGVzLCBzbyB5b3Ugc2hvdWxkIGNvcHkgdGhlIGAuZW52LmV4YW1wbGVgIHByb3ZpZGVkIGJ5IHJ1bm5pbmc6CmBgYCBzaGVsbAokIGNwIC5lbnYuZXhhbXBsZSAuZW52CmBgYAphbmQgYXNzaWduIHZhbHVlcyB0byB0aGUgY29ycmVzcG9uZGluZyB2YXJpYWJsZXMuCgojIyMjIE5vdGUKVGhlIGVudiB2YXJpYWJsZSBgT0FTX1BBVEhgIGluZGljYXRlcyB0aGUgcGF0aCB0byB0aGUgZG9jdW1lbnRzIHRoYXQgd2lsbCBiZSByZW5kZXJlZC4KCiMjIENvbnRyaWJ1dGluZwpXZSDinaTvuI8gY29udHJpYnV0aW9ucyBmcm9tIGV2ZXJ5b25lISBbQnVnIHJlcG9ydHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9OZXhtby9uZXhtby1vYXMtcmVuZGVyZXIvaXNzdWVzKSwgW2J1ZyBmaXhlc10oaHR0cHM6Ly9naXRodWIuY29tL05leG1vL25leG1vLW9hcy1yZW5kZXJlci9wdWxscykgYW5kIGZlZWRiYWNrIG9uIHRoZSBsaWJyYXJ5IGlzIGFsd2F5cyBhcHByZWNpYXRlZC4gTG9vayBhdCB0aGUgW0NvbnRyaWJ1dG9yIEd1aWRlbGluZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9OZXhtby9uZXhtby1vYXMtcmVuZGVyZXIvYmxvYi9tYXN0ZXIvQ09OVFJJQlVUSU5HLm1kKSBmb3IgbW9yZSBpbmZvcm1hdGlvbiBhbmQgcGxlYXNlIGZvbGxvdyB0aGUgW0dpdEh1YiBGbG93XShodHRwczovL2d1aWRlcy5naXRodWIuY29tL2ludHJvZHVjdGlvbi9mbG93L2luZGV4Lmh0bWwpLgoKIyMgTGljZW5zZQpUaGlzIHByb2plY3QgaXMgdW5kZXIgdGhlIFtNSVQgTElDRU5TRV0oaHR0cHM6Ly9naXRodWIuY29tL05leG1vL25leG1vLW9hcy1yZW5kZXJlci9ibG9iL21hc3Rlci9MSUNFTlNFKS4K readmeEtag: '"9c04021414112fbdbbd64229ac548ca2d11c9bcb"' readmeLastModified: Wed, 14 Jul 2021 07:45:54 GMT repositoryId: 186776115 description: Render your API references, Nexmo-style! created: '2019-05-15T07:47:32Z' updated: '2025-05-19T18:02:33Z' language: HTML archived: false stars: 45 watchers: 11 forks: 4 owner: Nexmo logo: https://avatars.githubusercontent.com/u/551057?v=4 license: MIT repoEtag: '"1e7a6ec5108b7cca709eeed10fef6210799f01ae75ecf4768581f8291b12530f"' repoLastModified: Mon, 19 May 2025 18:02:33 GMT foundInMaster: true id: 576fdcf39b6a579d65c2e4f63ead9ff4 - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/rookie-luochao/openapi-ui v3: true id: cca834879fa03e42847529ee34ced019 repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgogIE9wZW5BUEkgVUkKPC9oMT4KPHAgYWxpZ249ImNlbnRlciI+Ck9wZW5BUEkvU3dhZ2dlciBVSSBkb2N1bWVudCwgcXVpY2tseSBnZW5lcmF0ZSBtb2NrIHBhcmFtcyBhbmQgY2FsbCBhcGksIGFsc28gc2ltcGxpZmllZCBwb3N0bWFuIHRvb2wKPC9wPgo8cCBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vcm9va2llLWx1b2NoYW8vb3BlbmFwaS11aS9ibG9iL21hc3Rlci9MSUNFTlNFIj4KICAgIDxpbWcgYWx0PSJMaWNlbnNlIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2Uvcm9va2llLWx1b2NoYW8vb3BlbmFwaS11aSI+CiAgPC9hPgogIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9yb29raWUtbHVvY2hhby9vcGVuYXBpLXVpL3JlbGVhc2VzIj4KICAgIDxpbWcgYWx0PSJSZWxlYXNlIChsYXRlc3QgYnkgZGF0ZSkiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvdi9yZWxlYXNlL3Jvb2tpZS1sdW9jaGFvL29wZW5hcGktdWkiPgogIDwvYT4KICA8YSBocmVmPSJodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLXVpLWRpc3QiPgogICAgPGltZyBhbHQ9Im5wbSBvcGVuYXBpLXVpIHBhY2thZ2UiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vcGVuYXBpLXVpLWRpc3Quc3ZnIj4KICA8L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3Jvb2tpZS1sdW9jaGFvL29wZW5hcGktdWkvYWN0aW9ucy93b3JrZmxvd3MvcmVsZWFzZS1jaS55bWwiPgogICAgPGltZyBhbHQ9ImJ1aWxkIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3Jvb2tpZS1sdW9jaGFvL29wZW5hcGktdWkvcmVsZWFzZS1jaS55bWwiPgogIDwvYT4KICA8YSBocmVmPSJodHRwczovL3JlYWN0LmRldiI+CiAgICA8aW1nIGFsdD0iZnJhbWV3b3JrIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZnJhbWV3b3JrLXJlYWN0LWJyaWdodGdyZWVuIj4KICA8L2E+CjwvcD4KPGg0IGFsaWduPSJjZW50ZXIiPgogIDxwPgogICAgPGI+RW5nbGlzaDwvYj4gfAogICAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3Jvb2tpZS1sdW9jaGFvL29wZW5hcGktdWkvYmxvYi9tYXN0ZXIvUkVBRE1FLXpoX0NOLm1kIj7nroDkvZPkuK3mloc8L2E+IAogIDwvcD4KPC9oND4KCiMjIFNjcmVlbiBTaG90CjxkaXYgc3R5bGU9ImRpc3BsYXk6ZmxleCI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3Jvb2tpZS1sdW9jaGFvL29wZW5hcGktdWkvYmxvYi9tYXN0ZXIvc3JjL2Fzc2V0cy9zY3JlZW4tc2hvdC9vcGVuYXBpLXZpZXcucG5nIiBzdHlsZT0id2lkdGg6NTAlIj4KICAgIDxpbWcgYWx0PSJvcGVuYXBpIiBzcmM9Ii4vc3JjL2Fzc2V0cy9zY3JlZW4tc2hvdC9vcGVuYXBpLXZpZXcucG5nIj4KICA8L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3Jvb2tpZS1sdW9jaGFvL29wZW5hcGktdWkvYmxvYi9tYXN0ZXIvc3JjL2Fzc2V0cy9zY3JlZW4tc2hvdC9vcGVuYXBpLXZpZXcyLnBuZyIgc3R5bGU9IndpZHRoOjUwJSI+CiAgICA8aW1nIGFsdD0ib3BlbmFwaSIgc3JjPSIuL3NyYy9hc3NldHMvc2NyZWVuLXNob3Qvb3BlbmFwaS12aWV3Mi5wbmciPgogIDwvYT4KPC9kaXY+CjxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9yb29raWUtbHVvY2hhby9vcGVuYXBpLXVpL2Jsb2IvbWFzdGVyL3NyYy9hc3NldHMvc2NyZWVuLXNob3QvcG9zdG1hbi12aWV3LnBuZyI+CiAgPGltZyBhbHQ9InBvc3RtYW4iIHNyYz0iLi9zcmMvYXNzZXRzL3NjcmVlbi1zaG90L3Bvc3RtYW4tdmlldy5wbmciPgo8L2E+CgojIyBXZWJzaXRlIGRvbWFpbgoqIENOOiBbd3d3Lm9wZW5hcGktdWkuY29tXShodHRwczovL3d3dy5vcGVuYXBpLXVpLmNvbSksIHN1cHBvcnQgaHR0cCxodHRwcwoqIFVTOiBbZG9jLm9wZW5hcGktdWkuY29tXShodHRwczovL2RvYy5vcGVuYXBpLXVpLmNvbSkKKiBVUzI6IFtkb2NzLm9wZW5hcGktdWkuY29tXShodHRwczovL2RvY3Mub3BlbmFwaS11aS5jb20pCgojIyBVc2FnZQojIyMgV2l0aCBDRE4KYHNwZWMtdXJsYCBpcyBgZnVsbCBwYXRoYAoKYGBgdHN4CjwhZG9jdHlwZSBodG1sPgo8aHRtbCBsYW5nPSJlbiI+CiAgPGhlYWQ+CiAgICA8bWV0YSBjaGFyc2V0PSJVVEYtOCIgLz4KICAgIDx0aXRsZT5vcGVuQVBJIFVJPC90aXRsZT4KICA8L2hlYWQ+CiAgPGJvZHk+CiAgICA8ZGl2IGlkPSJvcGVuYXBpLXVpLWNvbnRhaW5lciIgc3BlYy11cmw9Imh0dHBzOi8vcGV0c3RvcmUzLnN3YWdnZXIuaW8vYXBpL3YzL29wZW5hcGkuanNvbiIgdGhlbWU9ImxpZ2h0Ij48L2Rpdj4KICAgIDxzY3JpcHQgc3JjPSJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL29wZW5hcGktdWktZGlzdEBsYXRlc3QvbGliL29wZW5hcGktdWkudW1kLmpzIj48L3NjcmlwdD4KICA8L2JvZHk+CjwvaHRtbD4KYGBgCgpgc3BlYy11cmxgIGlzIGBwYXRoYAoKYGBgdHN4CjwhZG9jdHlwZSBodG1sPgo8aHRtbCBsYW5nPSJlbiI+CiAgPGhlYWQ+CiAgICA8bWV0YSBjaGFyc2V0PSJVVEYtOCIgLz4KICAgIDx0aXRsZT5vcGVuQVBJIFVJPC90aXRsZT4KICA8L2hlYWQ+CiAgPGJvZHk+CiAgICA8ZGl2IGlkPSJvcGVuYXBpLXVpLWNvbnRhaW5lciIgc3BlYy11cmw9Ii9vcGVuYXBpLmpzb24iIHRoZW1lPSJkYXJrIj48L2Rpdj4KICAgIDxzY3JpcHQgc3JjPSJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL29wZW5hcGktdWktZGlzdEBsYXRlc3QvbGliL29wZW5hcGktdWkudW1kLmpzIj48L3NjcmlwdD4KICA8L2JvZHk+CjwvaHRtbD4KYGBgCgojIyMgV2l0aCBSZWFjdChvciBXaXRoIFZ1ZSkKYGBgdHN4CmltcG9ydCB7IHVzZUVmZmVjdCB9IGZyb20gInJlYWN0IjsKCmNvbnN0IFNldFVwT3BlbkFwaVVJID0gKCkgPT4gewogIHVzZUVmZmVjdCgoKSA9PiB7CiAgICBpbXBvcnQoIm9wZW5hcGktdWktZGlzdCIpCiAgfSwgW10pOwoKICByZXR1cm4gKAogICAgPGRpdiBpZD0ib3BlbmFwaS11aS1jb250YWluZXIiIHNwZWMtdXJsPSJodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92My9vcGVuYXBpLmpzb24iIHRoZW1lPSJsaWdodCIgLz4KICApOwp9CgpleHBvcnQgY29uc3Qgb3BlbmFwaVJvdXRlcyA9IHsKICBwYXRoOiAiL29wZW5hcGkiLAogIGlkOiAib3BlbmFwaSIsCiAgZWxlbWVudDogPFNldFVwT3BlbkFwaVVJIC8+LAp9OwpgYGAKIyMjIFdpdGggR28gV2ViIEZyYW1ld29yawotIFtnby1vcGVuYXBpLXVpXShodHRwczovL2dpdGh1Yi5jb20vcm9va2llLWx1b2NoYW8vZ28tb3BlbmFwaS11aSkKCiMjIyBXaXRoIE5vZGVqcyBXZWIgRnJhbWV3b3JrCi0gW25vZGVqcy1vcGVuYXBpLXVpXShodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS11aS9ub2RlanMtb3BlbmFwaS11aSkKCiMjIFF1aWNrIHN0YXJ0CmBgYGJhc2gKIyBub2RlIHZlcnNpb24gPj0gMTgKIyBkb3dubG9hZCBub2RlX21vZHVsZXMKcG5wbSBpbnN0YWxsCiMgb3IgbWFrZSBpbnN0YWxsCgojIHN0YXJ0Cm5wbSBydW4gZGV2CiMgb3IgbWFrZSBkZXYKYGBgCgojIyBTb21lIHNjcmlwdApgYGBiYXNoCiMgYnVpbGQKbnBtIHJ1biBidWlsZAojIG9yIG1ha2UgYnVpbGQKCiMgbWFrZSBkb2NrZXIgaW1hZ2UKbWFrZSBkb2NrZXItYnVpbGQKCiMgcnVuIGRvY2tlciBpbWFnZQptYWtlIGRvY2tlci1ydW4KCiMgbWFrZSBkb2NrZXIgaW1hZ2UgYW5kIHJ1biBkb2NrZXIgaW1hZ2UKbWFrZSBkb2NrZXItYnVpbGQtcnVuCmBgYAoKIyMgU3VwcG9ydCBkYXRhIGZvcm1hdAoqIHN3YWdnZXIyLmpzb24vc3dhZ2dlcjIueW1sCiogb3BlbmFwaTMuanNvbi9vcGVuYXBpMy55bWwKCiMjIEhvdyB0byB1c2UKKiBlbnRlciBzd2FnZ2VyMi9vcGVuYXBpMyBhcGkgZ2F0ZXdheSBVUkwsIHJlZnJlc2ggdGhlIHBhZ2UgdG8gdXBkYXRlIHRoZSBpbnRlcmZhY2UKKiB1cGxvYWQgc3dhZ2dlcjIvb3BlbmFwaTMgZmlsZQoqIGVudGVyIHN3YWdnZXIyL29wZW5hcGkzIHRleHQKCiMjIEdsb2JhbCBjb25maWcKKiBzdXBwb3J0cyBjb25maWd1cmUgcmVxdWVzdCB0aW1lb3V0LCB0aGUgZGVmYXVsdCByZXF1ZXN0IHRpbWVvdXQgaXMgMiBtaW51dGVzCiogc3VwcG9ydHMgY29uZmlndXJlIHJlcXVlc3QgQXV0aG9yaXphdGlvbiwgQXV0aG9yaXphdGlvbiBjYW4gYmUgb3ZlcnJpZGRlbiBpbiB0aGUgY3VycmVudCByZXF1ZXN0CgojIyBTaGFyZSBVUkwKKiB1cmwgY2FuIG9ubHkgYmUgc2hhcmVkIHdoZW4gaW1wb3J0ZWQgdGhyb3VnaCB1cmwgbW9kZQoqIGNvcHkgdGhlIHVybCBhbmQgc2hhcmUgaXQgd2l0aCB0aG9zZSB3aG8gbmVlZCBpdCwgdGhleSBjYW4gZWNobyB0aGUgdXJsIHRvIHRoZSBzcGVjaWZpZWQgaW50ZXJmYWNlCgojIyBNb2NrIHJlcXVlc3QgcGFyYW1zCjEuIGlmIHRoZSBzY2hlbWEgY29udGFpbnMgdGhlIGZvcm1hdCBmaWVsZCwgdGhlbiB1c2UgW29wZW5hcGktc2FtcGxlcl0oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvb3BlbmFwaS1zYW1wbGVyKSB0byBtb2NrIHJlcXVlc3QgcGFyYW1zCjIuIGlmIHRoZSBzY2hlbWEgZG9lcyBub3QgY29udGFpbiB0aGUgZm9ybWF0IGZpZWxkLCB0aGVuIHVzZSBmYWtlciB0byBtb2NrIHJlcXVlc3QgcGFyYW1zCgojIyBSZXF1ZXN0IGVycm9yIG1lc3NhZ2UgZGlzcGxheSBydWxlcwoxLiBpZiB0aGUgcmV0dXJuZWQgc3RydWN0dXJlIGNvbnRhaW5zIGEgbWVzc2FnZSBmaWVsZCwgZGlzcGxheSB0aGUgbWVzc2FnZSBmaWVsZAoyLiBpZiB0aGUgcmV0dXJuZWQgc3RydWN0dXJlIGNvbnRhaW5zIGEgbXNnIGZpZWxkLCBkaXNwbGF5IHRoZSBtc2cgZmllbGQKMy4gaWYgdGhlIHJldHVybmVkIHJlc3VsdCBpcyBhIHN0cmluZywgZGlzcGxheSB0aGUgc3RyaW5nCjQuIGRpc3BsYXkgQXhpb3NSZXNwb25zZS5zdGF0dXNUZXh0IGZpZWxkCjUuIGRpc3BsYXkgQXhpb3NFcnJvci5tZXNzYWdlIGZpZWxkCgojIyBDb25uZWN0IGludHJhbmV0IGFwaQoqIGlmIHVuYWJsZSB0byBjb25uZWN0IGludHJhbmV0IGFwaSwgeW91IGNhbiBydW4gdGhpcyBwcm9qZWN0IGxvY2FsbHkgb3IgdXNlIGRvY2tlciB0byBkZXBsb3kgdGhpcyBwcm9qZWN0IGxvY2FsbHkgb3Igb24gdGhlIHNlcnZlcgoKIyMgU3VwcG9ydCBtdWx0aXBsZSBhcGkgZ2F0ZXdheSBVUkwKKiB0aGUgY2FjaGluZyBzdHJhdGVneSB1c2VkIGlzIHNlc3Npb24gc3RvcmFnZSwgc28geW91IGNhbiBvcGVuIG11bHRpcGxlIHBhZ2VzIGF0IHRoZSBzYW1lIHRpbWUKCiMjIERvY2tlciBkZXBsb3ksIHN1cHBvcnQgZW52IHZhcmlhYmxlIGluamVjdGlvbgpgYGBiYXNoCiMgcHVsbCBEb2NrZXIgaW1hZ2UKZG9ja2VyIHB1bGwgZ2hjci5pby9yb29raWUtbHVvY2hhby9vcGVuYXBpLXVpOmxhdGVzdAoKIyBzdGFydCBjb250YWluZXIsIG5naW54IHJldmVyc2UgcHJveHkgY3VzdG9tIHBvcnQsIGZvciBleGFtcGxlOiBkb2NrZXIgcnVuIC1kIC1wIDgwODE6ODAgZ2hjci5pby9yb29raWUtbHVvY2hhby9vcGVuYXBpLXVpOmxhdGVzdApkb2NrZXIgcnVuIC1kIC1wIDgwOjgwIC1lIEFQUF9DT05GSUc9ZW52PXpoLGFwcE5hbWVaSD3nroDmtIHnvo7op4LnmoTmjqXlj6PmlofmoaMgZ2hjci5pby9yb29raWUtbHVvY2hhby9vcGVuYXBpLXVpOmxhdGVzdApgYGAKCiMjIE5vZGUgdmVyc2lvbgpub2RlID49IDE4 readmeEtag: '"f53d2a377848fff538ed4462a1777f0847b07655"' readmeLastModified: Mon, 12 Aug 2024 15:46:26 GMT repositoryId: 733969338 description: >- OpenAPI/Swagger UI document, quickly generate mock params and call api, also simplified postman tool. 基于 OpenAPI/Swagger 规范的接口文档,快速模拟请求参数并调用接口,也是简化版 postman 工具,欢迎提功能请求、bug created: '2023-12-20T14:51:12Z' updated: '2026-02-04T09:58:19Z' language: TypeScript archived: false stars: 687 watchers: 4 forks: 62 owner: rookie-luochao logo: https://avatars.githubusercontent.com/u/22948077?v=4 license: MIT repoEtag: '"0d26b0f20a8398b2d377cbb4c27d54eb1dd87c8c3e642579afc69c02f35ca687"' repoLastModified: Wed, 04 Feb 2026 09:58:19 GMT category: - Documentation - Server Implementations foundInMaster: true v3_1: true name: OpenAPI-UI source_description: >- Create simple and beautiful OpenAPI/Swagger documentation from OpenAPI files. Generate mock parameters and call APIs. Like Postman, but for OpenAPI specifications. link: https://docs.openapi-ui.com language: TypeScript v2: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/oasdiff/oasdiff v3: true repositoryMetadata: base64Readme: >- ClshW0NJXShodHRwczovL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmL3dvcmtmbG93cy9nby9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmL2FjdGlvbnMpClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9vYXNkaWZmL29hc2RpZmYvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPVk4Qk02WDc3SlkpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvb2FzZGlmZi9vYXNkaWZmKQpbIVtHbyBSZXBvcnQgQ2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL29hc2RpZmYvb2FzZGlmZikKWyFbR29Eb2NdKGh0dHBzOi8vZ29kb2Mub3JnL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmP3N0YXR1cy5zdmcpXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL29hc2RpZmYvb2FzZGlmZikKWyFbRG9ja2VyIEltYWdlIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZG9ja2VyL3YvdHVmaW4vb2FzZGlmZj9zb3J0PXNlbXZlcildKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci90dWZpbi9vYXNkaWZmL3RhZ3MpClshW1NsYWNrXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3NsYWNrLSYjNjQ7b2FzZGlmZi1ncmVlbi5zdmc/bG9nbz1zbGFjayldKGh0dHBzOi8vam9pbi5zbGFjay5jb20vdC9vYXNkaWZmL3NoYXJlZF9pbnZpdGUvenQtMXd2bzd3b2lzLXR0bmNOQm15anlSWHFCenlnflA2b0EpCgohW29hc2RpZmYgYmFubmVyXShodHRwczovL2dpdGh1Yi5jb20veW9uYXRhbm1nci9vYXNkaWZmL2Fzc2V0cy8zMTkxMzQ5NS9hYzliMTU0ZS03MmQxLTQ5NjktYmMzYi1mNTI3YmJlNzc1MWQpCgoKQ29tbWFuZC1saW5lIGFuZCBHbyBwYWNrYWdlIHRvIGNvbXBhcmUgYW5kIGRldGVjdCBicmVha2luZyBjaGFuZ2VzIGluIE9wZW5BUEkgc3BlY3MuCgojIyBJbnN0YWxsYXRpb24KCiMjIyBJbnN0YWxsIHdpdGggR28KYGBgYmFzaApnbyBpbnN0YWxsIGdpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmQGxhdGVzdApgYGAKCiMjIyBJbnN0YWxsIG9uIG1hY09TIHdpdGggQnJldwpgYGBiYXNoCmJyZXcgaW5zdGFsbCBvYXNkaWZmCmBgYAoKIyMjIEluc3RhbGwgb24gbWFjT1MgYW5kIExpbnV4IHVzaW5nIGN1cmwKCmBgYGJhc2gKY3VybCAtZnNTTCBodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vb2FzZGlmZi9vYXNkaWZmL21haW4vaW5zdGFsbC5zaCB8IHNoCmBgYAoKIyMjIEluc3RhbGwgd2l0aCBhc2RmCgpodHRwczovL2dpdGh1Yi5jb20vb2FzZGlmZi9hc2RmLW9hc2RpZmYKCiMjIyBNYW51YWxseSBpbnN0YWxsIG9uIG1hY09TLCBXaW5kb3dzIGFuZCBMaW51eApDb3B5IGJpbmFyaWVzIGZyb20gW2xhdGVzdCByZWxlYXNlXShodHRwczovL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmL3JlbGVhc2VzLykuICAKCiMjIyBVc2UgaW5zdGFsbC5zaApZb3UgY2FuIHVzZSB0aGUgW2luc3RhbGwuc2hdKC4uL2luc3RhbGwuc2gpIHNjcmlwdCB0byBpbnN0YWxsIG9hc2RpZmYuICAKVGhlIHNjcmlwdCB3aWxsIGRvd25sb2FkIHRoZSBsYXRlc3QgdmVyc2lvbiwgb3IgYSBzcGVjaWZpYyB2ZXJzaW9uIG9mIG9hc2RpZmYgYW5kIGluc3RhbGwgaXQgaW4gL3Vzci9sb2NhbC9iaW4uICAKCiMjIFRyeSBpdAoKIyMjIExvY2FsbHkKYGBgCm9hc2RpZmYgY2hhbmdlbG9nIGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9vYXNkaWZmL29hc2RpZmYvbWFpbi9kYXRhL29wZW5hcGktdGVzdDEueWFtbCBodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vb2FzZGlmZi9vYXNkaWZmL21haW4vZGF0YS9vcGVuYXBpLXRlc3Q1LnlhbWwKYGBgCgojIyMgV2l0aCBEb2NrZXIKYGBgCmRvY2tlciBydW4gLS1ybSAtdCB0dWZpbi9vYXNkaWZmIGNoYW5nZWxvZyBodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vb2FzZGlmZi9vYXNkaWZmL21haW4vZGF0YS9vcGVuYXBpLXRlc3QxLnlhbWwgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL29hc2RpZmYvb2FzZGlmZi9tYWluL2RhdGEvb3BlbmFwaS10ZXN0NS55YW1sCmBgYAoKIyMgRmVhdHVyZXMgCi0gRGV0ZWN0IFticmVha2luZyBjaGFuZ2VzXShCUkVBS0lORy1DSEFOR0VTLm1kKQotIERpc3BsYXkgYSB1c2VyLWZyaWVuZGx5IFtjaGFuZ2Vsb2ddKEJSRUFLSU5HLUNIQU5HRVMubWQpIG9mIGFsbCBpbXBvcnRhbnQgQVBJIGNoYW5nZXMKLSBHZW5lcmF0ZSBjb21wcmVoZW5zaXZlIFtkaWZmXShESUZGLm1kKSByZXBvcnRzIGluY2x1ZGluZyBhbGwgYXNwZWN0cyBvZiBbT3BlbkFQSSBTcGVjaWZpY2F0aW9uXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pOiBwYXRocywgb3BlcmF0aW9ucywgcGFyYW1ldGVycywgcmVxdWVzdCBib2RpZXMsIHJlc3BvbnNlcywgc2NoZW1hcywgZW51bXMsIGNhbGxiYWNrcywgc2VjdXJpdHkgZXRjLgotIE91dHB1dCByZXBvcnRzIGluIFlBTUwsIEpTT04sIFRleHQsIE1hcmtkb3duLCBIVE1MLCBKVW5pdCBYTUwgb3IgdGhlIFtnaXRodWIgYWN0aW9ucyBhbm5vdGF0aW9uIGZvcm1hdF0oaHR0cHM6Ly9kb2NzLmdpdGh1Yi5jb20vZW4vYWN0aW9ucy91c2luZy13b3JrZmxvd3Mvd29ya2Zsb3ctY29tbWFuZHMtZm9yLWdpdGh1Yi1hY3Rpb25zI3NldHRpbmctYS13YXJuaW5nLW1lc3NhZ2UpCi0gW0N1c3RvbWl6ZSBIVE1MIGFuZCBNYXJrZG93biBjaGFuZ2Vsb2cgcmVwb3J0c10oVVNBR0VfRVhBTVBMRVMubWQjb3BlbmFwaS1jaGFuZ2Vsb2ctd2l0aC1jdXN0b20tdGVtcGxhdGUpCi0gQ29tcGFyZSBsb2NhbCBzcGVjcyBvciByZW1vdGUgc3BlY3Mgb3ZlciBodHRwL3MKLSBDb21wYXJlIHNwZWNzIGluIFlBTUwgb3IgSlNPTiBmb3JtYXQKLSBbQ29tcGFyZSB0d28gY29sbGVjdGlvbnMgb2Ygc3BlY3NdKENPTVBPU0VELm1kKQotIFtEZXByZWNhdGUgQVBJcyBhbmQgUGFyYW1ldGVyc10oREVQUkVDQVRJT04ubWQpCi0gW0FQSSBzdGFiaWxpdHkgbGV2ZWxzXShTVEFCSUxJVFkubWQpCi0gW011bHRpcGxlIHZlcnNpb25zIG9mIHRoZSBzYW1lIGVuZHBvaW50XShNQVRDSElORy1FTkRQT0lOVFMubWQjZHVwbGljYXRlLWVuZHBvaW50cykKLSBbTWVyZ2UgYWxsT2Ygc2NoZW1hc10oQUxMT0YubWQpCi0gW01lcmdlIGNvbW1vbiAocGF0aC1sZXZlbCkgcGFyYW1ldGVyc10oQ09NTU9OLVBBUkFNUy5tZCkKLSBbQ2FzZS1pbnNlbnNpdGl2ZSBoZWFkZXIgY29tcGFyaXNvbl0oSEVBREVSLURJRkYubWQpCi0gW1BhdGggcHJlZml4IG1vZGlmaWNhdGlvbl0oUEFUSC1QUkVGSVgubWQpCi0gW1BhdGggcGFyYW1ldGVyIHJlbmFtaW5nXShQQVRILVBBUkFNLVJFTkFNRS5tZCkKLSBbRXhjbHVkZSBjZXJ0YWluIGtpbmRzIG9mIGNoYW5nZXNdKERJRkYubWQjZXhjbHVkaW5nLXNwZWNpZmljLWtpbmRzLW9mLWNoYW5nZXMpCi0gW0V4Y2x1ZGUgc3BlY2lmaWMgZXh0ZW5zaW9uIG5hbWVzXShESUZGLm1kI2V4Y2x1ZGluZy1zcGVjaWZpYy1leHRlbnNpb24tbmFtZXMpCi0gW1RyYWNrIGNoYW5nZXMgdG8gT3BlbkFQSSBFeHRlbnNpb25zXShESUZGLm1kI29wZW5hcGktZXh0ZW5zaW9ucykKLSBbRmlsdGVyIGVuZHBvaW50c10oRklMVEVSSU5HLUVORFBPSU5UUy5tZCkKLSBbRXh0ZW5kIGJyZWFraW5nIGNoYW5nZXMgd2l0aCBjdXN0b20gY2hlY2tzXShDVVNUT01JWklORy1DSEVDS1MubWQpCi0gTG9jYWxpemF0aW9uOiB2aWV3IGJyZWFraW5nIGNoYW5nZXMgYW5kIGNoYW5nZWxvZyBtZXNzYWdlcyBpbiBsb2NhbCBsYW5ndWFnZXM6IGVuLCBydSwgcHQtYnIsIGVzCi0gW1J1biB3aXRoIGNvbmZpZ3VyYXRpb24gZmlsZV0oQ09ORklHLUZJTEVTLm1kKQotIFtSdW4gZnJvbSBEb2NrZXJdKERPQ0tFUi5tZCkKLSBbSW50ZWdyYXRlIGluIEdpdEh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL29hc2RpZmYvZ2l0aHViLWRlbW8vdHJlZS9tYWluKQotIFtHaXRIdWIgQWN0aW9uXShodHRwczovL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmLWFjdGlvbikKLSBbUnVuIGFzIGEgU2VydmljZV0oT0FTRElGRi1TRVJWSUNFLm1kKQotIFtPcGVuQVBJIFN5bmM6IEdldCBub3RpZmllZCB3aGVuIGFuIEFQSSBwcm92aWRlciBicmVha3MgdGhlIEFQSV0oaHR0cHM6Ly9naXRodWIuY29tL29hc2RpZmYvc3luYy8pCi0gW0VtYmVkIGluIHlvdXIgZ28gcHJvZ3JhbV0oR08ubWQpCgojIyBEZW1vCjxpbWcgc3JjPSIuL2RlbW8uc3ZnIj4KCiMjIFRoZSBtYWluIGNvbW1hbmRzCi0gW2RpZmZdKERJRkYubWQpOiB0aGUgZGlmZiBiZXR3ZWVuIE9wZW5BUEkgc3BlY3MsIGZ1bGx5IGRldGFpbGVkCi0gW2JyZWFraW5nXShCUkVBS0lORy1DSEFOR0VTLm1kKTogYnJlYWtpbmcgY2hhbmdlcyBiZXR3ZWVuIE9wZW5BUEkgc3BlY3MgIAotIFtjaGFuZ2Vsb2ddKEJSRUFLSU5HLUNIQU5HRVMubWQpOiBpbXBvcnRhbnQgY2hhbmdlcyBiZXR3ZWVuIE9wZW5BUEkgc3BlY3MgaW5jbHVkaW5nIGJyZWFraW5nIGFuZCBub24tYnJlYWtpbmcgY2hhbmdlcwotIFtmbGF0dGVuXShBTExPRi5tZCk6IHJlcGxhY2UgYWxsIGluc3RhbmNlcyBvZiBhbGxPZiBieSBhIG1lcmdlZCBlcXVpdmFsZW50Ci0gY2hlY2tzOiBkaXNwbGF5cyB0aGUgZGlmZmVyZW50IGNoZWNrcyB0aGF0IG9hc2RpZmYgcnVucyB0byBkZXRlY3QgY2hhbmdlcwoKIyMgUm9hZG1hcAotICoqT3BlbkFQSSAzLjEqKiAgCiAgU2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9vYXNkaWZmL29hc2RpZmYvaXNzdWVzLzUyICAKLSAqKklubGluZSB2aXN1YWxpemF0aW9uIG9mIGJyZWFraW5nIGNoYW5nZXMgaW4gdGhlIE9wZW5BUEkgZG9jdW1lbnQqKiAgCiAgSSBhbSB3b3JraW5nIG9uIHRoZSBhYmlsaXR5IHRvIGNvcnJlbGF0ZSBicmVha2luZyBjaGFuZ2VzIGFuZCBjaGFuZ2Vsb2cgbWVzc2FnZXMgd2l0aCB0aGUgdW5kZXJseWluZyBlZGl0cyBpbiB0aGUgb3JpZ2luYWwgWUFNTCBzcGVjaWZpY2F0aW9uLiAgCiAgVGhpcyB3aWxsIGFsbG93IHVzZXJzIHRvIHJldmlldyBjaGFuZ2VzIGluIHRoZWlyIG9yaWdpbmFsIGNvbnRleHQsIGludmVzdGlnYXRlIHRoZWlyIGltcGFjdCwgYW5kIGRlY2lkZSBob3cgdG8gaGFuZGxlIHRoZW0uICAKICBJIGFtIGNvbnNpZGVyaW5nIG9mZmVyaW5nIHRoaXMgY2FwYWJpbGl0eSB1bmRlciBhIHNlcGFyYXRlIGxpY2Vuc2UgYW5kIHdvdWxkIGFwcHJlY2lhdGUgZmVlZGJhY2sgZnJvbSB0aGUgY29tbXVuaXR5LgogIArwn5GJIFtQbGVhc2UgaGVscCBwcmlvcml0aXplIHRoZSByb2FkbWFwXShodHRwczovL2dpdGh1Yi5jb20vb2FzZGlmZi9vYXNkaWZmL2Rpc2N1c3Npb25zLzYzMSkuCgojIyBDcmVkaXRzClRoaXMgcHJvamVjdCByZWxpZXMgb24gdGhlIGV4Y2VsbGVudCBpbXBsZW1lbnRhdGlvbiBvZiBPcGVuQVBJIDMuMCBmb3IgR286IFtraW4tb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL2dldGtpbi9raW4tb3BlbmFwaSkuCgojIyBGZWVkYmFjawpXZSB3ZWxjb21lIHlvdXIgZmVlZGJhY2suICAKSWYgeW91IGhhdmUgaWRlYXMgZm9yIGltcHJvdmVtZW50IG9yIGFkZGl0aW9uYWwgbmVlZHMgYXJvdW5kIEFQSXMsIHBsZWFzZSBbbGV0IHVzIGtub3ddKGh0dHBzOi8vZ2l0aHViLmNvbS9vYXNkaWZmL29hc2RpZmYvZGlzY3Vzc2lvbnMvbmV3P2NhdGVnb3J5PWlkZWFzKS4K readmeEtag: '"57b84c593d2a4748b6989110b18f327c1c0781cc"' readmeLastModified: Thu, 05 Feb 2026 16:25:22 GMT repositoryId: 337689065 description: OpenAPI Diff and Breaking Changes created: '2021-02-10T10:28:36Z' updated: '2026-02-06T04:02:00Z' language: Go archived: false stars: 1068 watchers: 10 forks: 88 owner: oasdiff logo: https://avatars.githubusercontent.com/u/128153573?v=4 license: Apache-2.0 repoEtag: '"c5c8cd21fa8d7efa71cc866b2cf9bef088476c629561196238b6991fe8d814b3"' repoLastModified: Fri, 06 Feb 2026 04:02:00 GMT foundInMaster: true name: oasdiff language: Go source_description: >- Golang module for deep comparison of two OpenAPI specifications. Available also as a command-line. v2: false v3_1: false homepage: https://github.com/tufin/oasdiff id: 9b752ecd94615c375d0324ddf547c930 category: Parsers oldLocations: - https://github.com/tufin/oasdiff - source: - https://openapi.tools/ - openapi3 tags name: - optic diff - optic category: - Learning - Testing language: - Node - cli link: - https://www.useoptic.com/docs/diff-openapi - https://useoptic.com repository: https://github.com/opticdev/optic source_description: - >- Diff the effective API contract between any two versions of your OpenAPI description. Exit 1 on breaking changes. - >- Build your first OpenAPI description from traffic. Use Optic to patch the OpenAPI every time it detects new API behavior. v2: false v3: true repositoryMetadata: base64Readme: >- IAohW0dpdEh1YiBSZXBvIHN0YXJzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9zdGFycy9vcHRpY2Rldi9vcHRpYz9zdHlsZT1zb2NpYWwpICFbR2l0SHViIGNvbnRyaWJ1dG9yc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvY29udHJpYnV0b3JzLWFub24vb3B0aWNkZXYvb3B0aWM/c3R5bGU9c29jaWFsKSAhW25wbV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0vQHVzZW9wdGljL29wZW5hcGktaW8/c3R5bGU9c29jaWFsKSAhW2xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2Uvb3B0aWNkZXYvb3B0aWM/c3R5bGU9c29jaWFsKQoKIyBPcHRpYyAtIE9wZW5BUEkgbWFkZSBlYXN5Cgrwn6egICoqR2VuZXJhdGUgT3BlbkFQSSoqIGZyb20gdGVzdCB0cmFmZmljCgrimqHvuI8gKipLZWVwIE9wZW5BUEkgc3BlYyBhY2N1cmF0ZSoqIHdpdGggYXV0b21hdGljIHNjaGVtYSB0ZXN0aW5nIGFuZCBwYXRjaGVzCgrwn6SpICoqQ2F0Y2ggQnJlYWtpbmcgQ2hhbmdlcyBhbmQgYXBwbHkgTGludCBSdWxlcyoqIHdpdGggdGhlIHNhbWUgdG9vbAoKCj4gT3B0aWMgc3VwcG9ydHMgJHJlZiBhbmQgT3BlbkFQSSBzcGVjcyBzcGxpdCBhY3Jvc3MgbXVsdGlwbGUgZmlsZXMuIEl0IGFsd2F5cyAKCgojIyBbUmVhZCBEb2N1bWVudGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vb3B0aWNkZXYvb3B0aWMvd2lraSkgCgoKIyMjIERlbW9zCgpodHRwczovL2dpdGh1Yi5jb20vdXNlci1hdHRhY2htZW50cy9hc3NldHMvYjM4NmJiNDItMzlhNS00ZGIwLWFlYTUtYzIzOTk1ODhlYmIwCgpodHRwczovL2dpdGh1Yi5jb20vdXNlci1hdHRhY2htZW50cy9hc3NldHMvYmI2YWYwOTgtNmU5MS00NjM2LWI4OGMtMWM5YmM1ODZjY2UwCgpodHRwczovL2dpdGh1Yi5jb20vdXNlci1hdHRhY2htZW50cy9hc3NldHMvMWIwN2JmZDktMzY0Yy00YmJlLWJhZWItNDkzYmVkMzkyY2RmCgojIyMgSW5zdGFsbApgYGBiYXNoCm5wbSBpbnN0YWxsIC1nIEB1c2VvcHRpYy9vcHRpYwpgYGAKCiMjIOKPre+4jyBbR2V0IFN0YXJ0ZWQhXShodHRwczovL2dpdGh1Yi5jb20vb3B0aWNkZXYvb3B0aWMvd2lraSkgCgoKIyMjIExpY2Vuc2UKT3B0aWMgaXMgTUlUIExpY2Vuc2VkIAoKW09wdGljIExhYnMgaXMgbm93IHBhcnQgb2YgQXRsYXNzaWFuXShodHRwczovL3d3dy5hdGxhc3NpYW4uY29tL2Jsb2cvYW5ub3VuY2VtZW50cy9vcHRpYy1hY3F1aXNpdGlvbikK readmeEtag: '"358aa291d9f6d31461560ea8bcd6d83ad1bfd949"' readmeLastModified: Wed, 05 Feb 2025 23:05:43 GMT repositoryId: 123606765 description: >- OpenAPI linting, diffing and testing. Optic helps prevent breaking changes, publish accurate documentation and improve the design of your APIs. created: '2018-03-02T17:02:50Z' updated: '2026-02-04T10:21:13Z' language: TypeScript archived: true stars: 1523 watchers: 9 forks: 90 owner: opticdev logo: https://avatars.githubusercontent.com/u/34556970?v=4 license: MIT repoEtag: '"ebb6a432a89e1eadcaae420b7467945e5a55ff515931ddddb88241159ea7a534"' repoLastModified: Wed, 04 Feb 2026 10:21:13 GMT foundInMaster: true id: 8c3d08406ca659662808d51ddc10b4e4 v3_1: true - source: https://openapi.tools/ name: OpenAPI CRUD Wizard language: Yaml repository: https://github.com/inssch/openapicrudwizard link: https://plugins.jetbrains.com/plugin/19889-openapi-crud-wizard source_description: >- This wizard is an IntelliJ Plugin to create a new OpenAPI document including all CRUD operations based only on a Yaml object. No knowledge about OpenAPI specification needed. v3: true v3_1: true id: 2c2fe4c5fadd80c7fe677b1ad25de5e1 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIENSVUQgV2l6YXJkCgpUaGlzIHJlcG9zaXRvcnkgaXMgYWJvdXQgdGhlIEludGVsbGlKIFBsdWdpbiAiT3BlbkFQSSBDUlVEIFdpemFyZCIKaHR0cHM6Ly9wbHVnaW5zLmpldGJyYWlucy5jb20vcGx1Z2luLzE5ODg5LW9wZW5hcGktY3J1ZC13aXphcmQKCllvdSBjYW4gcmFpc2UgaXNzdWVzIG9yIGZlYXR1cmUgcmVxdWVzdHMgaW4gdGhlICJJc3N1ZXMiIHRhYi4KClRoaXMgc2l0ZSBpcyBpbiBwcm9ncmVzcy4gRnVsbCBkZXNjcmlwdGlvbiBjb21pbmcgc29vbi4KCgo= readmeEtag: '"3d9760d6304818ee7298c9d7a9e4c472c2eb02fa"' readmeLastModified: Tue, 07 Feb 2023 11:38:47 GMT repositoryId: 530957071 description: OpenAPI CRUD Wizard created: '2022-08-31T06:13:23Z' updated: '2024-05-17T16:07:37Z' language: null archived: false stars: 1 watchers: 1 forks: 0 owner: inssch logo: https://avatars.githubusercontent.com/u/109123959?v=4 repoEtag: '"3547aa17af09a237865f8233491388f501661045aed0754a30fa2840866e4dae"' repoLastModified: Fri, 17 May 2024 16:07:37 GMT foundInMaster: true category: Server Implementations - source: https://openapi.tools/ name: OAS RAML Converter category: Converters language: Node.js link: https://mulesoft.github.io/oas-raml-converter/ repository: https://github.com/mulesoft/oas-raml-converter source_description: Converts between OpenAPI and RAML API specifications v2: true v3: true repositoryMetadata: base64Readme: >- IyBPQVMgUkFNTCBDb252ZXJ0ZXIgKERFUFJFQ0FURUQpIFshW25wbSB2ZXJzaW9uXShodHRwczovL2JhZGdlLmZ1cnkuaW8vanMvb2FzLXJhbWwtY29udmVydGVyLnN2ZyldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29hcy1yYW1sLWNvbnZlcnRlcikKCiMgRGVwcmVjYXRpb24gTm90aWNlCioqVGhpcyBwcm9qZWN0IGhhcyBiZWVuIGRlcHJlY2F0ZWQgYW5kIGlzIG5vIGxvbmdlciBtYWludGFpbmVkLiBQbGVhc2UgdXNlIFtXZWJBUEkgUGFyc2VyXShodHRwczovL2dpdGh1Yi5jb20vcmFtbC1vcmcvd2ViYXBpLXBhcnNlcikgY29udmVydGVyIGluc3RlYWQuIGh0dHBzOi8vZ2l0aHViLmNvbS9yYW1sLW9yZy93ZWJhcGktcGFyc2VyKioKCiMjIyBNYXN0ZXIgWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvbXVsZXNvZnQvb2FzLXJhbWwtY29udmVydGVyLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL211bGVzb2Z0L29hcy1yYW1sLWNvbnZlcnRlcikgWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvbXVsZXNvZnQvb2FzLXJhbWwtY29udmVydGVyL2JhZGdlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL211bGVzb2Z0L29hcy1yYW1sLWNvbnZlcnRlcj9icmFuY2g9bWFzdGVyKSAKCiMjIyB2MC4yLnggWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvbXVsZXNvZnQvb2FzLXJhbWwtY29udmVydGVyLnN2Zz9icmFuY2g9djAuMi54KV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL211bGVzb2Z0L29hcy1yYW1sLWNvbnZlcnRlcikgWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvbXVsZXNvZnQvb2FzLXJhbWwtY29udmVydGVyL2JhZGdlLnN2Zz9icmFuY2g9djAuMi54KV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL211bGVzb2Z0L29hcy1yYW1sLWNvbnZlcnRlcj9icmFuY2g9djAuMi54KSAKClRoaXMgcGFja2FnZSBoZWxwcyB0byBjb252ZXJ0IGJldHdlZW4gZGlmZmVyZW50IEFQSSBzcGVjaWZpY2F0aW9ucy4gSXQgd2FzIG9yaWdpbmFsbHkgZm9ya2VkIGZyb20gW3RoZSBzdG9wbGlnaHQuaW8gY29udmVydGVyXShodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vYXBpLXNwZWMtY29udmVydGVyKS4gCgojIyBTdXBwb3J0ZWQgQ29udmVyc2lvbnMgKGJldGEpCgotIE9BUyAoT0FTIDIuMCkgLT4gUkFNTCAxLjA6IFtDb21wbGV0ZSBGdW5jdGlvbmFsIFNwZWNpZmljYXRpb25dKC4vZG9jcy9PQVMyMC10by1SQU1MMTAubWQpCi0gUkFNTCAxLjAgLT4gT0FTIChPQVMgMi4wKTogW0NvbXBsZXRlIEZ1bmN0aW9uYWwgU3BlY2lmaWNhdGlvbl0oLi9kb2NzL1JBTUwxMC10by1PQVMyMC5tZCkKLSBSQU1MIDEuMCAtPiBPQVMgKE9BUyAzLjApOiBbQ29tcGxldGUgRnVuY3Rpb25hbCBTcGVjaWZpY2F0aW9uXSguL2RvY3MvUkFNTDEwLXRvLU9BUzMwLm1kKQotIFJBTUwgMC44IC0+IE9BUyAoT0FTIDIuMCkKLSBSQU1MIDAuOCAtPiBSQU1MIDEuMAoKIyMgVXNpbmcKCiMjIyAxLiBPbmxpbmUgd2ViIHBhZ2UKCkZvciBhbiBvbmxpbmUgY29udmVyc2lvbiwgdXNlOiBbaHR0cHM6Ly9tdWxlc29mdC5naXRodWIuaW8vb2FzLXJhbWwtY29udmVydGVyXShodHRwczovL211bGVzb2Z0LmdpdGh1Yi5pby9vYXMtcmFtbC1jb252ZXJ0ZXIpLgoKIyMjIDIuIENvbW1hbmQgbGluZSB0b29sCgpUaGlzIGNvbW1hbmQgbmVlZHMgdG8gYmUgZXhlY3V0ZSBhZnRlciBjbG9uaW5nIHJlcG9zaXRvcnksIApgYGAKbnBtIHJ1biBidWlsZApgYGAKClRoaXMgdXRpbGl0eSBoZWxwcyB5b3UgY29udmVydGluZyBsb2NhbCBmaWxlcyBmcm9tIHlvdXIgY29tbWFuZCBsaW5lLgoKYGBgCi4vbGliL2Jpbi9jb252ZXJ0ZXIuanMgLS1mcm9tIE9BUzIwIC0tdG8gUkFNTCAuL3BhdGgvdG8vc3dhZ2dlci5qc29uCi4vbGliL2Jpbi9jb252ZXJ0ZXIuanMgLS1mcm9tIE9BUzIwIC0tdG8gUkFNTCAuL3BhdGgvdG8vc3dhZ2dlci5qc29uID4gb3V0cHV0LnJhbWwKCi4vbGliL2Jpbi9jb252ZXJ0ZXIuanMgLS1mcm9tIFJBTUwgLS10byBPQVMyMCAuL3BhdGgvdG8vc291cmNlLnJhbWwgPiBzd2FnZ2VyLmpzb24KYGBgCgpPciBpbnN0YWxsIGdsb2JhbGx5IGFuZCB0aGVuOgoKYGBgCm9hcy1yYW1sLWNvbnZlcnRlciAtLWZyb20gT0FTMjAgLS10byBSQU1MIC4vcGF0aC90by9zd2FnZ2VyLmpzb24Kb2FzLXJhbWwtY29udmVydGVyIC0tZnJvbSBPQVMyMCAtLXRvIFJBTUwgLi9wYXRoL3RvL3N3YWdnZXIuanNvbiA+IG91dHB1dC5yYW1sCgpvYXMtcmFtbC1jb252ZXJ0ZXIgLS1mcm9tIFJBTUwgLS10byBPQVMyMCAuL3BhdGgvdG8vc291cmNlLnJhbWwgPiBzd2FnZ2VyLmpzb24KYGBgCgojIyMgMy4gQXMgYSBzZXJ2aWNlCgpGb3IgYSBSRVNUIEFQSSBvZiB0aGUgY29udmVydGVyLCB5b3UgY2FuIHN0YXJ0IGl0IGluIGFuIGV4cHJlc3Mgc2VydmVyLCBjaGVja291dCB0aGUgW29hcy1yYW1sLWNvbnZlcnRlci1zZXJ2aWNlXShodHRwczovL2dpdGh1Yi5jb20vbXVsZXNvZnQvb2FzLXJhbWwtY29udmVydGVyLXNlcnZpY2UpIHByb2plY3QuCgojIyMgNC4gQXMgYSBkZXBlbmRlbmN5CgojIyMjIEluc3RhbGxhdGlvbiAoTm9kZUpTIG9yIEJyb3dzZXIpCgpgYGBiYXNoCm5wbSBpbnN0YWxsIC0tc2F2ZSBvYXMtcmFtbC1jb252ZXJ0ZXIKYGBgCgojIyMjIEluaXRpYWxpemluZyBhIGNvbnZlcnRlcgoKUmFtbCAxLjAgdG8gT0FTIDIuMDoKYGBganMKdmFyIGNvbnZlcnRlciA9IHJlcXVpcmUoJ29hcy1yYW1sLWNvbnZlcnRlcicpOwp2YXIgcmFtbDEwVG9PYXMyMCA9IG5ldyBjb252ZXJ0ZXIuQ29udmVydGVyKGNvbnZlcnRlci5Gb3JtYXRzLlJBTUwsIGNvbnZlcnRlci5Gb3JtYXRzLk9BUzIwKTsKYGBgCgpPQVMgMi4wIHRvIFJhbWwgMS4wOgpgYGBqcwp2YXIgY29udmVydGVyID0gcmVxdWlyZSgnb2FzLXJhbWwtY29udmVydGVyJyk7CnZhciBvYXMyMFRvUmFtbDEwID0gbmV3IGNvbnZlcnRlci5Db252ZXJ0ZXIoY29udmVydGVyLkZvcm1hdHMuT0FTMjAsIGNvbnZlcnRlci5Gb3JtYXRzLlJBTUwpOwpgYGAKClRoZSBjb252ZXJ0ZXIgZGV0ZWN0cyB0aGUgaW5wdXQgcmFtbCBmb3JtYXQgYXV0b21hdGljYWxseSBieSBwYXNzaW5nIGBSQU1MYCBpbXBvcnQgZm9ybWF0LCBzbzoKUmFtbCAwLjggdG8gT0FTIDIuMDoKYGBganMKdmFyIGNvbnZlcnRlciA9IHJlcXVpcmUoJ29hcy1yYW1sLWNvbnZlcnRlcicpOwp2YXIgcmFtbDA4VG9PYXMyMCA9IG5ldyBjb252ZXJ0ZXIuQ29udmVydGVyKGNvbnZlcnRlci5Gb3JtYXRzLlJBTUwsIGNvbnZlcnRlci5Gb3JtYXRzLk9BUzIwKTsKYGBgClJhbWwgMC44IHRvIFJhbWwgMS4wOgpgYGBqcwp2YXIgY29udmVydGVyID0gcmVxdWlyZSgnb2FzLXJhbWwtY29udmVydGVyJyk7CnZhciByYW1sMDhUb1JhbWwxMCA9IG5ldyBjb252ZXJ0ZXIuQ29udmVydGVyKGNvbnZlcnRlci5Gb3JtYXRzLlJBTUwsIGNvbnZlcnRlci5Gb3JtYXRzLlJBTUwpOwpgYGAKCiMjIyMgQ29udmVydGluZyBmcm9tIGEgZmlsZSBvciB1cmwKCmBgYGpzCm9hczIwVG9SYW1sMTAuY29udmVydEZpbGUoJy9wYXRoL3RvL3N3YWdnZXIuanNvbicpLnRoZW4oZnVuY3Rpb24ocmFtbCkgewogIGNvbnNvbGUubG9nKHJhbWwpOyAvLyByYW1sIGlzIHJhbWwgeWFtbCBzdHJpbmcKfSkKLmNhdGNoKGZ1bmN0aW9uKGVycikgewogIGNvbnNvbGUuZXJyb3IoZXJyKTsKfSk7CmBgYAoKIyMjIyBDb252ZXJ0aW5nIGZyb20gYSBzdHJpbmcgb3IganNvbgoKYGBganMKdmFyIG15T2FzU3RyaW5nID0gJy4uLic7Cm9hczIwVG9SYW1sMTAuY29udmVydERhdGEobXlPYXNTdHJpbmcpLnRoZW4oZnVuY3Rpb24ocmFtbCkgewogIGNvbnNvbGUubG9nKHJhbWwpOyAvLyByYW1sIGlzIHJhbWwgeWFtbCBzdHJpbmcKfSkKLmNhdGNoKGZ1bmN0aW9uKGVycikgewogIGNvbnNvbGUuZXJyb3IoZXJyKTsKfSk7CmBgYAoKIyMjIyBQYXNzaW5nIG9wdGlvbnMKCmBgYGpzCnZhciBvcHRpb25zID0gewogICAgdmFsaWRhdGU6IGZhbHNlLCAvLyBQYXJzZSBib3RoIGlucHV0IGFuZCBvdXRwdXQgdG8gY2hlY2sgdGhhdCBpdHMgYSB2YWxpZCBkb2N1bWVudAogICAgdmFsaWRhdGVJbXBvcnQ6IGZhbHNlLCAvLyBPbmx5IHZhbGlkYXRlIGlucHV0CiAgICB2YWxpZGF0ZUV4cG9ydDogZmFsc2UsIC8vIE9ubHkgdmFsaWRhdGUgb3V0cHV0CiAgICBmb3JtYXQ6ICd5YW1sJywgLy8gT3V0cHV0IGZvcm1hdDoganNvbiAoZGVmYXVsdCBmb3IgT0FTKSBvciB5YW1sIChkZWZhdWx0IGZvciBSQU1MKQogICAgZnM6IHsgLi4uIH0gLy8gVXNlIGEgY3VzdG9tIGZpbGUgc3lzdGVtIHNvbHZlciAobm90IHlldCBhdmFpbGFibGUpCn07CgpvYXMyMFRvUmFtbDEwLmNvbnZlcnRGaWxlKCcvcGF0aC90by9zd2FnZ2VyLmpzb24nLCBvcHRpb25zKS50aGVuKGZ1bmN0aW9uKHJhbWwpIHsKICBjb25zb2xlLmxvZyhyYW1sKTsgLy8gcmFtbCBpcyByYW1sIHlhbWwgc3RyaW5nCn0pCi5jYXRjaChmdW5jdGlvbihlcnIpIHsKICBjb25zb2xlLmVycm9yKGVycik7Cn0pOwpgYGAKCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyBhcmUgd2VsY29tZSEgUGxlYXNlIGNoZWNrIHRoZSBjdXJyZW50IGlzc3VlcyB0byBtYWtlIHN1cmUgd2hhdCB5b3UgYXJlIHRyeWluZyB0byBkbyBoYXMgbm90IGFscmVhZHkgYmVlbiBkaXNjdXNzZWQuCgojIyMgU3RlcHMKCjEuIEZvcmsuCjIuIE1ha2UgY2hhbmdlcy4KMy4gV3JpdGUgdGVzdHMuCjQuIFNlbmQgYSBwdWxsIHJlcXVlc3QuCgojIyMgRGV2ZWxvcAoKSW5zdGFsbCBkZXBlbmRlbmNpZXM6CmBgYGJhc2gKbnBtIGluc3RhbGwKYGBgCgpSdW4gdGVzdHM6CmBgYGJhc2gKbnBtIHRlc3QKYGBgCgpSdW4gZXNsaW50IHRvIGNoZWNrIGxpbnRpbmcgZXJyb3JzOgpgYGBiYXNoCm5wbSBydW4gZXNsaW50CmBgYAo= readmeEtag: '"96668028f8add8a14f4ef8443482f35ccd222055"' readmeLastModified: Thu, 05 Sep 2019 20:39:33 GMT repositoryId: 65008129 description: (DEPRECATED) Converts between OAS and RAML API specifications created: '2016-08-05T09:49:24Z' updated: '2026-01-01T11:59:24Z' language: JavaScript archived: true stars: 73 watchers: 238 forks: 51 owner: mulesoft logo: https://avatars.githubusercontent.com/u/185740?v=4 license: MIT repoEtag: '"0abb667c61ab84138d92c53025a70cea5152bd28a4102e6ff644ad58e7ff9af0"' repoLastModified: Thu, 01 Jan 2026 11:59:24 GMT foundInMaster: true id: 24686ca74b0284507f5f437578166800 - source: https://openapi.tools/ name: OData OpenAPI homepage: https://github.com/oasis-tcs/odata-openapi language: - Node.js - XSLT source_description: OData 4.0, 3.0, and 2.0 to OpenAPI v3.1, v3.0, and v2.0 converter category: Converters repository: https://github.com/oasis-tcs/odata-openapi v2: true v3: true repositoryMetadata: base64Readme: >- WyFbbnBtIERvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZHcvb2RhdGEtb3BlbmFwaSldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29kYXRhLW9wZW5hcGkpCgo8ZGl2Pgo8aDI+UkVBRE1FPC9oMj4KCjxwPk1lbWJlcnMgb2YgdGhlIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL2NvbW1pdHRlZXMvb2RhdGEvIj5PQVNJUyBPcGVuIERhdGEgUHJvdG9jb2wgKE9EYXRhKSBUZWNobmljYWwgQ29tbWl0dGVlPC9hPiBjcmVhdGUgYW5kIG1hbmFnZSB0ZWNobmljYWwgY29udGVudCBpbiB0aGlzIFRDIEdpdEh1YiByZXBvc2l0b3J5ICggPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL29hc2lzLXRjcy9vZGF0YS1vcGVuYXBpIj5odHRwczovL2dpdGh1Yi5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGk8L2E+ICkgYXMgcGFydCBvZiB0aGUgVEMncyBjaGFydGVyZWQgd29yayAoPGk+aS5lLjwvaT4sIHRoZSBwcm9ncmFtIG9mIHdvcmsgYW5kIGRlbGl2ZXJhYmxlcyBkZXNjcmliZWQgaW4gaXRzIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL2NvbW1pdHRlZXMvb2RhdGEvY2hhcnRlci5waHAiPmNoYXJ0ZXI8L2E+KS48L3A+Cgo8cD5PQVNJUyBUQyBHaXRIdWIgcmVwb3NpdG9yaWVzLCBhcyBkZXNjcmliZWQgaW4gPGEgaHJlZj0iaHR0cHM6Ly93d3cub2FzaXMtb3Blbi5vcmcvcmVzb3VyY2VzL3RjYWRtaW4vZ2l0aHViLXJlcG9zaXRvcmllcy1mb3Itb2FzaXMtdGMtbWVtYmVycy1jaGFydGVyZWQtd29yayI+R2l0SHViIFJlcG9zaXRvcmllcyBmb3IgT0FTSVMgVEMgTWVtYmVycycgQ2hhcnRlcmVkIFdvcms8L2E+LCBhcmUgZ292ZXJuZWQgYnkgdGhlIE9BU0lTIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL3BvbGljaWVzLWd1aWRlbGluZXMvdGMtcHJvY2VzcyI+VEMgUHJvY2VzczwvYT4sIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL3BvbGljaWVzLWd1aWRlbGluZXMvaXByIj5JUFIgUG9saWN5PC9hPiwgYW5kIG90aGVyIHBvbGljaWVzLCBzaW1pbGFyIHRvIFRDIFdpa2lzLCBUQyBKSVJBIGlzc3VlcyB0cmFja2luZyBpbnN0YW5jZXMsIFRDIFNWTi9TdWJ2ZXJzaW9uIHJlcG9zaXRvcmllcywgZXRjLiAgV2hpbGUgdGhleSBtYWtlIHVzZSBvZiBwdWJsaWMgR2l0SHViIHJlcG9zaXRvcmllcywgdGhlc2UgVEMgR2l0SHViIHJlcG9zaXRvcmllcyBhcmUgZGlzdGluY3QgZnJvbSA8YSBocmVmPSJodHRwczovL3d3dy5vYXNpcy1vcGVuLm9yZy9yZXNvdXJjZXMvb3Blbi1yZXBvc2l0b3JpZXMiPk9BU0lTIE9wZW4gUmVwb3NpdG9yaWVzPC9hPiwgd2hpY2ggYXJlIHVzZWQgZm9yIGRldmVsb3BtZW50IG9mIG9wZW4gc291cmNlIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL3Jlc291cmNlcy9vcGVuLXJlcG9zaXRvcmllcy9saWNlbnNlcyI+bGljZW5zZWQ8L2E+IGNvbnRlbnQuPC9wPgo8L2Rpdj4KCjxkaXY+CjxoMz5EZXNjcmlwdGlvbjwvaDM+Cgo8cD5UaGUgcHVycG9zZSBvZiB0aGlzIHJlcG9zaXRvcnkgaXMgdG8gc3VwcG9ydCBkZXZlbG9wbWVudCBvZiB0b29scyBmb3IgcHJvZHVjaW5nIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uIj5PcGVuQVBJPC9hPiBkZXNjcmlwdGlvbnMgZm9yIE9EYXRhIHNlcnZpY2VzLjwvcD4KPHA+UGxhbm5lZCB3b3JrIGl0ZW1zIGluY2x1ZGU6Cjx1bD4KPGxpPmNvbnZlcnQgT0RhdGEgQ1NETCBKU09OIG9yIE9EYXRhIENTREwgWE1MIHRvIE9wZW5BUEkgSlNPTjwvbGk+CjxsaT5leGFtcGxlIFhNTCBmaWxlczwvbGk+CjxsaT5leGFtcGxlIG9wZW5hcGkuanNvbiBmaWxlczwvbGk+CjxsaT5leGFtcGxlIGZpbGVzIGZvciB0aGUgbGl2ZSBvZGF0YS5vcmcgc2VydmljZXM8L2xpPgo8L3VsPjwvcD4KCjwvZGl2PgoKPGRpdj4KPGgzPkNvbnRyaWJ1dGlvbnM8L2gzPgo8cD5BcyBzdGF0ZWQgaW4gdGhpcyByZXBvc2l0b3J5J3MgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL29hc2lzLXRjcy9vZGF0YS1vcGVuYXBpL2Jsb2IvbWFpbi9DT05UUklCVVRJTkcubWQiPkNPTlRSSUJVVElORyBmaWxlPC9hPiwgY29udHJpYnV0b3JzIHRvIHRoaXMgcmVwb3NpdG9yeSBhcmUgZXhwZWN0ZWQgdG8gYmUgTWVtYmVycyBvZiB0aGUgT0FTSVMgT0RhdGEgVEMsIGZvciBhbnkgc3Vic3RhbnRpdmUgY2hhbmdlIHJlcXVlc3RzLiAgQW55b25lIHdpc2hpbmcgdG8gY29udHJpYnV0ZSB0byB0aGlzIEdpdEh1YiBwcm9qZWN0IGFuZCA8YSBocmVmPSJodHRwczovL3d3dy5vYXNpcy1vcGVuLm9yZy9qb2luL3BhcnRpY2lwYXRpb24taW5zdHJ1Y3Rpb25zIj5wYXJ0aWNpcGF0ZTwvYT4gaW4gdGhlIFRDJ3MgdGVjaG5pY2FsIGFjdGl2aXR5IGlzIGludml0ZWQgdG8gam9pbiBhcyBhbiBPQVNJUyBUQyBNZW1iZXIuICBQdWJsaWMgZmVlZGJhY2sgaXMgYWxzbyBhY2NlcHRlZCwgc3ViamVjdCB0byB0aGUgdGVybXMgb2YgdGhlIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL3BvbGljaWVzLWd1aWRlbGluZXMvaXByI2FwcGVuZGl4YSI+T0FTSVMgRmVlZGJhY2sgTGljZW5zZTwvYT4uPC9wPgo8L2Rpdj4KCjxkaXY+CjxoMz5MaWNlbnNpbmc8L2gzPgo8cD5QbGVhc2Ugc2VlIHRoZSA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGkvYmxvYi9tYWluL0xJQ0VOU0UubWQiPkxJQ0VOU0U8L2E+IGZpbGUgZm9yIGRlc2NyaXB0aW9uIG9mIHRoZSBsaWNlbnNlIHRlcm1zIGFuZCBPQVNJUyBwb2xpY2llcyBhcHBsaWNhYmxlIHRvIHRoZSBUQydzIHdvcmsgaW4gdGhpcyBHaXRIdWIgcHJvamVjdC4gQ29udGVudCBpbiB0aGlzIHJlcG9zaXRvcnkgaXMgaW50ZW5kZWQgdG8gYmUgcGFydCBvZiB0aGUgT0RhdGEgVEMncyBwZXJtYW5lbnQgcmVjb3JkIG9mIGFjdGl2aXR5LCB2aXNpYmxlIGFuZCBmcmVlbHkgYXZhaWxhYmxlIGZvciBhbGwgdG8gdXNlLCBzdWJqZWN0IHRvIGFwcGxpY2FibGUgT0FTSVMgcG9saWNpZXMsIGFzIHByZXNlbnRlZCBpbiB0aGUgcmVwb3NpdG9yeSA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGkvYmxvYi9tYWluL0xJQ0VOU0UubWQiPkxJQ0VOU0U8L2E+IGZpbGUuPC9wPgo8L2Rpdj4KCjxoMz5GdXJ0aGVyIERlc2NyaXB0aW9uIG9mIHRoaXMgUmVwb3NpdG9yeTwvaDM+CgpUaGUgT0RhdGEgVEMgaGFzIHB1Ymxpc2hlZCB0aGUgW09EYXRhIHRvIE9wZW5BUEkgTWFwcGluZyBWZXJzaW9uIDEuMF0oaHR0cDovL2RvY3Mub2FzaXMtb3Blbi5vcmcvb2RhdGEvb2RhdGEtb3BlbmFwaS92MS4wL29kYXRhLW9wZW5hcGktdjEuMC5odG1sKSwgYSByZWNvbW1lbmRhdGlvbiBvbiBob3cgdG8gY3JlYXRlIE9wZW5BUEkgZGVzY3JpcHRpb25zIGZvciBPRGF0YSBzZXJ2aWNlcy4gVGhpcyBwcm9qZWN0IGNvbnRhaW5zIHR3byBwcm9vZi1vZi1jb25jZXB0IGltcGxlbWVudGF0aW9ucyBvZiB0aGF0IG1hcHBpbmcsIFtvbmUgdXNpbmcgSmF2YVNjcmlwdF0oaHR0cHM6Ly9naXRodWIuY29tL29hc2lzLXRjcy9vZGF0YS1vcGVuYXBpL2Jsb2IvbWFpbi9saWIpLCBhbmQgW29uZSB1c2luZyBYU0xUXShodHRwczovL2dpdGh1Yi5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGkvYmxvYi9tYWluL3Rvb2xzKS4gVGhlIGxhdHRlciBhbHNvIGV2YWx1YXRlcyBjZXJ0YWluICJPRGF0YSBWMiBzdHlsZSIgYW5ub3RhdGlvbnMgdGhhdCBhcmUgZXhwbGFpbmVkIFtoZXJlXShodHRwczovL3NhcC5naXRodWIuaW8vb2RhdGEtdm9jYWJ1bGFyaWVzL2RvY3MvdjItYW5ub3RhdGlvbnMuaHRtbCkuCgpUaGUgW2BleGFtcGxlc2AgZm9sZGVyXShodHRwczovL2dpdGh1Yi5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGkvYmxvYi9tYWluL2V4YW1wbGVzKSBjb250YWlucyBbT3BlbkFQSSAzLjAuMl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24pIGRlc2NyaXB0aW9ucyB0aGF0IGhhdmUgYmVlbiBjcmVhdGVkIGZyb20gdGhlIFhNTCBgJG1ldGFkYXRhYCBkb2N1bWVudHMgb2YgbGl2ZSBhbmQgZXhhbXBsZSBPRGF0YSBzZXJ2aWNlcyB3aXRoIHRoZXNlIHByb29mLW9mLWNvbmNlcHQgaW1wbGVtZW50YXRpb25zLgoKVGhlIGVudGl0eS1yZWxhdGlvbnNoaXAgZGlhZ3JhbXMgdmlzdWFsaXppbmcgdGhlIHJlc291cmNlIG1vZGVscyBvZiBlYWNoIHNlcnZpY2UgYXJlIGdlbmVyYXRlZCBvbi10aGUtZmx5IHdpdGggW3lVTUxdKGh0dHA6Ly95dW1sLm1lLykuCgpPcGVuQVBJIGRlc2NyaXB0aW9ucyBmb3IgbGl2ZSBleGFtcGxlIE9EYXRhIHNlcnZpY2VzIGF0IFt3d3cub2RhdGEub3JnXShodHRwOi8vd3d3Lm9kYXRhLm9yZy8pCgotIFtUcmlwUGluIChyZWFkL3dyaXRlKV0oaHR0cDovL3BldHN0b3JlLnN3YWdnZXIuaW8vP3VybD1odHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGkvbWFpbi9leGFtcGxlcy9UcmlwUGluLm9wZW5hcGkzLmpzb24pCi0gW1NpbXBsZSByZWFkL3dyaXRlIHNlcnZpY2VdKGh0dHA6Ly9wZXRzdG9yZS5zd2FnZ2VyLmlvLz91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL29hc2lzLXRjcy9vZGF0YS1vcGVuYXBpL21haW4vZXhhbXBsZXMvZXhhbXBsZS5vcGVuYXBpMy5qc29uKQotIFtOb3J0aHdpbmQgKHJlYWQpXShodHRwOi8vcGV0c3RvcmUuc3dhZ2dlci5pby8/dXJsPWh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9vYXNpcy10Y3Mvb2RhdGEtb3BlbmFwaS9tYWluL2V4YW1wbGVzL05vcnRod2luZC5vcGVuYXBpMy5qc29uKQoKT3BlbkFQSSBkZXNjcmlwdGlvbnMgZm9yIE9EYXRhIHNlcnZpY2VzIHRoYXQgcmVmZXJlbmNlIGVhY2ggb3RoZXIgKGNyb3NzLXNlcnZpY2UgcmVmZXJlbmNlcykKCi0gW1Blb3BsZV0oaHR0cDovL3BldHN0b3JlLnN3YWdnZXIuaW8vP3VybD1odHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vb2FzaXMtdGNzL29kYXRhLW9wZW5hcGkvbWFpbi9leGFtcGxlcy9QZW9wbGUub3BlbmFwaTMuanNvbikKLSBbUHJvZHVjdHNdKGh0dHA6Ly9wZXRzdG9yZS5zd2FnZ2VyLmlvLz91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL29hc2lzLXRjcy9vZGF0YS1vcGVuYXBpL21haW4vZXhhbXBsZXMvUHJvZHVjdHMub3BlbmFwaTMuanNvbikKCjxkaXY+CjxoMz5Db250YWN0PC9oMz4KPHA+UGxlYXNlIHNlbmQgcXVlc3Rpb25zIG9yIGNvbW1lbnRzIGFib3V0IDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL3Jlc291cmNlcy90Y2FkbWluL2dpdGh1Yi1yZXBvc2l0b3JpZXMtZm9yLW9hc2lzLXRjLW1lbWJlcnMtY2hhcnRlcmVkLXdvcmsiPk9BU0lTIFRDIEdpdEh1YiByZXBvc2l0b3JpZXM8L2E+IHRvIHRoZSBPQVNJUyA8YSBocmVmPSJtYWlsdG86dGMtYWRtaW5Ab2FzaXMtb3Blbi5vcmciPlRDIEFkbWluaXN0cmF0b3I8L2E+LiAgRm9yIHF1ZXN0aW9ucyBhYm91dCBjb250ZW50IGluIHRoaXMgcmVwb3NpdG9yeSwgcGxlYXNlIGNvbnRhY3QgdGhlIFRDIENoYWlyIG9yIENvLUNoYWlycyBhcyBsaXN0ZWQgb24gdGhlIHRoZSBPRGF0YSBUQydzIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm9hc2lzLW9wZW4ub3JnL2NvbW1pdHRlZXMvb2RhdGEvIj5ob21lIHBhZ2U8L2E+LjwvcD4KPC9kaXY+Cg== readmeEtag: '"21870cecf66f16f38866046aabeebf4b9d79eb6f"' readmeLastModified: Mon, 20 Oct 2025 07:05:04 GMT repositoryId: 74067831 description: >- OASIS OData TC: Tools for producing API descriptions for OData services that adhere to the OpenAPI Specification created: '2016-11-17T21:11:22Z' updated: '2026-02-03T08:30:49Z' language: XSLT archived: false stars: 214 watchers: 18 forks: 88 owner: oasis-tcs logo: https://avatars.githubusercontent.com/u/20116735?v=4 license: NOASSERTION repoEtag: '"20c37623b0ca768bc9dee710738ad3047a6756d01986a3846c3176a529397628"' repoLastModified: Tue, 03 Feb 2026 08:30:49 GMT foundInMaster: true v3_1: true id: 9039f84e8e8b63849cbf15578564033e - source: - https://openapi.tools/ - openapi3 tags name: - OpenAPI Filter - openapi-filter category: - Converters - Parsers language: Node.js source_description: - Filter internal components from OpenAPI Descriptions - >- OpenAPI 2.0 and 3.0 filter utility. A CLI/module to filter out internal/private paths, operations, parameters, schemas etc from OpenAPI v1/OpenAPI v2/AsyncAPI definitions. Simply flag any OpenAPI object within the definition with an `x-internal` specification extension or target a OpenAPI property (tags, methods, OperationId), and it will be removed from the output. repository: https://github.com/mermade/openapi-filter v2: true v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWZpbHRlcgoKIVtjaV0oaHR0cHM6Ly9naXRodWIuY29tL01lcm1hZGUvb3BlbmFwaS1maWx0ZXIvd29ya2Zsb3dzL2NpL2JhZGdlLnN2ZykKCkZpbHRlciBpbnRlcm5hbCBwYXRocywgb3BlcmF0aW9ucywgcGFyYW1ldGVycywgc2NoZW1hcyBldGMgZnJvbSBPcGVuQVBJL1N3YWdnZXIvQXN5bmNBUEkgZGVmaW5pdGlvbnMuCgpTaW1wbHkgZmxhZyBhbnkgb2JqZWN0IHdpdGhpbiB0aGUgZGVmaW5pdGlvbiB3aXRoIGFuIGB4LWludGVybmFsYCBzcGVjaWZpY2F0aW9uIGV4dGVuc2lvbiwgYW5kIGl0IHdpbGwgYmUgcmVtb3ZlZCBmcm9tIHRoZSBvdXRwdXQuCgpGb3IgZXhhbXBsZToKCmBgYHlhbWwKb3BlbmFwaTogMy4wLjAKaW5mbzoKICB0aXRsZTogQVBJCiAgdmVyc2lvbjogMS4wLjAKcGF0aHM6CiAgLzoKICAgIGdldDoKICAgICAgeC1pbnRlcm5hbDogdHJ1ZQogICAgICAuLi4KYGBgCgpXb3JrcyB3aXRoIE9wZW5BUEkvU3dhZ2dlciAyLjAgYW5kIDMuMC54IGFuZCBBc3luY0FQSSBkZWZpbml0aW9ucy4KCmBgYApvcGVuYXBpLWZpbHRlci5qcyA8aW5maWxlPiBbb3V0ZmlsZV0KClBvc2l0aW9uYWxzOgogIGluZmlsZSAgIHRoZSBpbnB1dCBmaWxlCiAgb3V0ZmlsZSAgdGhlIG91dHB1dCBmaWxlCgpPcHRpb25zOgoKICAtLWluZm8gICAgICAgICAgIGluY2x1ZGUgY29tcGxldGUgaW5mbyBvYmplY3Qgd2l0aCAtLXZhbGlkICAgICAgICAgICBbYm9vbGVhbl0KICAtLWludmVyc2UsIC1pICAgIG91dHB1dCBmaWx0ZXJlZCBlbGVtZW50cyBvbmx5ICAgICAgICAgICAgICAgICAgICAgICBbYm9vbGVhbl0KICAtLWZsYWdzLCAtZiAgICAgIGZsYWdzIHRvIGZpbHRlciBieSAgICAgICAgICBbYXJyYXldIFtkZWZhdWx0OiBbIngtaW50ZXJuYWwiXV0KICAtLWZsYWdWYWx1ZXMsIC12IGZsYWcgU3RyaW5nIHZhbHVlcyB0byBtYXRjaCAgICAgICAgICAgICBbYXJyYXldIFtkZWZhdWx0OiBbXV0KICAtLWNoZWNrVGFncyAgICAgIGZpbHRlciBpZiBmbGFncyBnaXZlbiBpbiAtLWZsYWdzIGFyZSBpbiB0aGUgdGFncyBhcnJheQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtib29sZWFuXQogIC0tb3ZlcnJpZGVzLCAtbyAgcHJlZml4ZXMgdXNlZCB0byBvdmVycmlkZSBuYW1lZCBwcm9wZXJ0aWVzW2Fycl0gW2RlZmF1bHQ6IFtdXQogIC0tdmFsaWQgICAgICAgICAgdHJ5IHRvIGVuc3VyZSBpbnZlcnNlIG91dHB1dCBpcyB2YWxpZCAgICAgICAgICAgICAgIFtib29sZWFuXQogIC0tc3RyaXAsIC1zICAgICAgc3RyaXAgdGhlIGZsYWdzIGZyb20gdGhlIGZpbmlzaGVkIHByb2R1Y3QgICAgICAgICAgIFtib29sZWFuXQogIC0tc2VydmVycyAgICAgICAgaW5jbHVkZSBjb21wbGV0ZSBzZXJ2ZXJzIG9iamVjdCB3aXRoIC0tdmFsaWQgICAgICAgIFtib29sZWFuXQogIC0tbGluZVdpZHRoLCAtbCAgbWF4IGxpbmUgd2lkdGggb2YgeWFtbCBvdXRwdXQgICAgICAgICAgIFtudW1iZXJdIFtkZWZhdWx0OiAwXQogIC0tbWF4QWxpYXNDb3VudCAgbWF4aW11bSBZQU1MIGFsaWFzZXMgYWxsb3dlZCAgICAgICAgICBbbnVtYmVyXSBbZGVmYXVsdDogMTAwXQogIC0tY29uZmlnRmlsZSAgICAgVGhlIGZpbGUgJiBwYXRoIGZvciB0aGUgZmlsdGVyIG9wdGlvbnMgICAgICAgICAgICAgICAgIFtwYXRoXQogIC0taGVscCAgICAgICAgICAgU2hvdyBoZWxwICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtib29sZWFuXQogIC0tdmVyYm9zZSAgICAgICAgT3V0cHV0IG1vcmUgZGV0YWlscyBvZiB0aGUgZmlsdGVyIHByb2Nlc3MgICAgICAgICAgICAgW2NvdW50XQpgYGAKCnVzZSBgLS1gIHRvIHNlcGFyYXRlIGZsYWdzIG9yIG90aGVyIGFycmF5IG9wdGlvbnMgZnJvbSBmb2xsb3dpbmcgb3B0aW9ucywgaS5lLjoKCmBvcGVuYXBpLWZpbHRlciAtLWZsYWdzIHgtcHJpdmF0ZSB4LWhpZGRlbiAtLSBzb3VyY2UueWFtbCB0YXJnZXQueWFtbGAKCm9yCgpgYGBqYXZhc2NyaXB0CmxldCBvcGVuYXBpRmlsdGVyID0gcmVxdWlyZSgnb3BlbmFwaS1maWx0ZXInKTsKbGV0IG9wdGlvbnMgPSB7fTsgLy8gZGVmYXVsdHMgYXJlIHNob3duCi8vb3B0aW9ucy5pbnZlcnNlID0gZmFsc2U7Ci8vb3B0aW9ucy52YWxpZCA9IGZhbHNlOwovL29wdGlvbnMuZmxhZ3MgPSBbJ3gtaW50ZXJuYWwnXTsKbGV0IHJlcyA9IG9wZW5hcGlGaWx0ZXIuZmlsdGVyKG9iaixvcHRpb25zKTsKYGBgCgpTZWUgdGhlIFt3aWtpXShodHRwczovL2dpdGh1Yi5jb20vTWVybWFkZS9vcGVuYXBpLWZpbHRlci93aWtpKSBmb3IgZnVydGhlciBleGFtcGxlcy4K readmeEtag: '"3b83424f748e8c33bde73e794e3dcbfd7f1ef222"' readmeLastModified: Sun, 15 Oct 2023 13:23:21 GMT repositoryId: 99120783 description: >- Filter internal paths, operations, parameters, schemas etc from OpenAPI/Swagger/AsyncAPI definitions created: '2017-08-02T13:42:57Z' updated: '2025-11-14T11:08:26Z' language: JavaScript archived: false stars: 167 watchers: 3 forks: 34 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"54c1f8c934bb7f5d1c06cc364ded424303caee88148dd5c73291136ea2f79187"' repoLastModified: Fri, 14 Nov 2025 11:08:26 GMT foundInMaster: true id: c00a8a003ace3ce47f5740bb641667fb v3_1: true - source: https://openapi.tools/ name: OData.OpenAPI category: - Converters - Parsers repository: https://github.com/xuzhg/odata.openapi language: .NET source_description: Convert an Edm (Entity Data Model) to OpenAPI 3.0 v3: true repositoryMetadata: base64Readme: >- IyBPRGF0YS5PcGVuQVBJCgpBIHByb2plY3QgdG8gY29udmVydCBhbiBFZG0gKEVudGl0eSBEYXRhIE1vZGVsKSB0byBbT3BlbkFwaSAzLjBdKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykuCgpOb3csIHRoZSBXaG9sZSBwcm9qZWN0IGlzIG1vdmluZyB0bzogaHR0cHM6Ly9naXRodWIuY29tL21pY3Jvc29mdC9vcGVuYXBpLm5ldCBmb3IgYmFzaWMgT3BlbiBBUEkgRE9NIGFuZAoKaHR0cHM6Ly9naXRodWIuY29tL21pY3Jvc29mdC9vcGVuYXBpLm5ldC5vZGF0YSBmb3IgYmFzaWMgQ1NETCB0byBPcGVuIEFQSSBET00uCg== readmeEtag: '"22e3c2a52080e5ae6542c952011e423a4b10ac0d"' readmeLastModified: Mon, 06 Aug 2018 17:03:11 GMT repositoryId: 105206912 description: OData to Open API created: '2017-09-28T22:56:53Z' updated: '2023-05-22T09:10:47Z' language: C# archived: false stars: 7 watchers: 3 forks: 5 owner: xuzhg logo: https://avatars.githubusercontent.com/u/9426627?v=4 repoEtag: '"7420dbc87b4b37201967f18a22cc3341e26b799edad5e4734d9894e60dafbc87"' repoLastModified: Mon, 22 May 2023 09:10:47 GMT foundInMaster: true id: d147d76d31836810da33ac308ffb7615 - source: https://openapi.tools/ name: OpenDocumenter category: - Documentation - Parsers link: https://github.com/ouropencode/OpenDocumenter language: Vue.js source_description: >- OpenDocumenter is a automatic documentation generator for OpenAPI v3 schemas. Simply provide your schema file in JSON or YAML, then sit back and enjoy the documentation. v2: true v3: true foundInMaster: true repository: https://github.com/ouropencode/opendocumenter repositoryMetadata: base64Readme: >- <h1 align="center">OpenDocumenter</h1>

<p align="center">
  <img alt="Version" src="https://img.shields.io/github/package-json/v/ouropencode/OpenDocumenter" />
  <img alt="npm" src="https://img.shields.io/npm/v/opendocumenter" />
  <img alt="License" src="https://img.shields.io/github/license/ouropencode/OpenDocumenter" />
  <img alt="Dependencies" src="https://img.shields.io/librariesio/release/npm/opendocumenter" />
  <img alt="Issues" src="https://img.shields.io/github/issues/ouropencode/OpenDocumenter" />
  <img alt="Node Version" src="https://img.shields.io/node/v/opendocumenter" />
  <img alt="Build Status" src="https://github.com/ouropencode/OpenDocumenter/workflows/Quisque%20API%20Build/badge.svg" />
</p>

<p align="center">
  <a href="https://ouropencode.github.io/OpenDocumenter/" target="_blank">
    <img alt="Node Version" src="https://raw.githubusercontent.com/ouropencode/OpenDocumenter/master/example/example.gif" />
  </a>
</p>

OpenDocumenter is a automatic documentation generator for [OpenAPI v3](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md) schemas. Simply provide your schema file in JSON or YAML, then sit back and enjoy the documentation.

Powered by [nuxt.js](https://nuxtjs.org/https://nuxtjs.org/) and [swagger-parser](https://github.com/swagger-api/swagger-parser).

## Example
For a live preview of documentation generated with OpenDocumenter you can view our demo documentation "**[Quisque API](https://ouropencode.github.io/OpenDocumenter/)**". This example uses lorem-ipsum for all content to provide placeholder text used to demonstrate the form of a document without relying on meaningful content.

## Installation

```bash
> npm install -g opendocumenter
```
or
```bash
> yarn global add opendocumenter
```

## Usage
```
> opendocumenter --help

   _____             ____                            _
  |     |___ ___ ___|    \ ___ ___ _ _ _____ ___ ___| |_ ___ ___
  |  |  | . | -_|   |  |  | . |  _| | |     | -_|   |  _| -_|  _|
  |_____|  _|___|_|_|____/|___|___|___|_|_|_|___|_|_|_| |___|_|
        |_|                                                                      

  OpenDocumenter is a automatic documentation generator for OpenAPI v3 schemas.
  Simply provide your schema file in JSON or YAML, then sit back and enjoy the documentation.

  Powered by nuxt.js and swagger-parser.

  Usage:

    opendocumenter --schema=<file> --output=<dir>

  Arguments:

    --schema=<file>    (required) The OpenAPI 3 format file to generate documentation from.
    --output=<dir>     (required) The output destination directory.
    --config=<file>    A configuration file to load advanced options from.
```

## Configuration
OpenDocumenter can be configured using a `.json` file stored alongside your schema file.

### Merge From Directory
The `mergeFromDirectory` parameter allows you to specify a directory that will be copied over the base template before building. This allows customization of any part of OpenDocumenter to suit your needs.

```json
{
  "mergeFromDirectory": "./overrides"
}
```

Any part of the OpenDocumenter vue source (see: [./src](./src)) can be overridden, just ensure to follow the same directory structure!
```
overrides
 |- assets
 | '- theme.less
 '- components
   |- DocHeader.vue
   '- DocEntry.vue
```

### 'Generated Using' Footer
By default, a small 'Generated Using' message is included on the footer of the generated documentation. Although we'd love you to keep it, you can disable this by setting the `disableGeneratedUsingFooter` parameter to true.

```json
{
  "disableGeneratedUsingFooter": true
}
```

### Aborting on Invalid Schema
OpenDocumenter is capable of generating documentation for OpenAPI schemas that don't match the OpenAPI Specification entirely. When generating we attempt to validate your schema, display any validation warnings, and then continue to generate. If you would like the generation to abort when a schema is invalid you can set the `abortOnInvalidSchema` parameter to true.

```json
{
  "abortOnInvalidSchema": true
}
```

### Shields / Badges
Various shields are included in the generated documentation header, such as the API version. Additional shields can be added using the `shields` parameter. Each shield is an object containing either the `url` key, or a combination of `left`, `right` and `color`. The `translate` parameter can be used (`left`, `right`, `both`) to run the text through the internationalization handler. An optional link can provided with the `href` parameter. All shields are generated using [shields.io](https://shields.io) unless a URL is provided.

```json
{
  "shields": [
    { "url": "https://img.shields.io/badge/test-1.2.3--test-blue" },
    { "left": "test", "right": "1.2.3-test", "color": "blue" },
    {
      "left": "test",
      "right": "1.2.3-test",
      "color": "blue",
      "href": "https://www.example.com"
    }
  ]
}
```

### Build: Modern Mode
By default, the build-chain produces "Modern Mode"  output, shipping native ES2015 code to modern browsers that support it, with auto fallback to a legacy bundle. This can be turned off by setting `vueModernMode` parameter, afterwards the build-chain produces "Legacy" output for older browser compatibility.

```json
{
  "vueModernMode": false
}
```

### Build: Reporting
You can control the reporting output from the build-chain with the `vueReport` parameter. This defaults to `none` and can be set to `json`, `html` or `both`. With this enabled the build-chain will output a report file describing the webpack bundle.

```json
{
  "vueReport": "html"
}
```

### Internationalization
Most of the documentation text is taken directly from the OpenAPI schema file, however, there are various strings throughout the project that cannot be stored within the schema file. All of these strings are customizable by editing the `i18n` parameter.
```json
{
  "i18n": {
    "API_SDK_DOCUMENTATION": "API and SDK Documentation",
    "VERSION": "version",
    "NO_INDEPTH_DOCS_AVAILABLE_ENDPOINT": "No in-depth API documentation is available for this endpoint.",
    "NO_INDEPTH_DOCS_AVAILABLE_TAG": "No in-depth API documentation is available for this section.",
    "CLICK_TO_COPY": "click to copy",
    "COPIED": "copied",
    "REQUEST_BODY": "Request Body",
    "REQUEST_RESPONSES": "Request Responses",
    "DEFINITION": "Definition",
    "DEFINITIONS": "Definitions",
    "SERVER": "Server",
    "LANGUAGE": "Language",
    "GENERATED_USING": "Generated using OpenDocumenter by $ourOpenCode",
    "HAVE_ANY_QUESTIONS_CONTACT": "Have any questions? Please contact",
    "US": "us",
    "OR": "or",
    "VIA_EMAIL": "via email",
    "VIA_OUR_WEBSITE": "via our website",
    "TERMS_OF_SERVICE": "Terms of Service",
    "SEND_US_AN_EMAIL": "Send us an email",
    "VISIT_OUR_WEBSITE": "Visit our website",
    "EXTERNAL_DOCUMENTATION": "External Documentation",
    "DOCUMENTATION": "Documentation"
  }
}
```

> There is a single end-user visible string that isn't configurable in this config. The "Loading Documentation..." text in the page title, briefly shown during page load, is only configurable using the "Merge From Directory" parameter.

## License
Licensed under the MIT license. Please see [LICENSE](LICENSE) for more details.
 readmeEtag: '"d46fd60db4b32123a2d52108804bee01f5ee1640"' readmeLastModified: Mon, 31 Aug 2020 17:28:44 GMT repositoryId: 284009477 description: >- OpenDocumenter is a automatic documentation generator for OpenAPI v3 schemas. Simply provide your schema file in JSON or YAML, then sit back and enjoy the documentation. created: '2020-07-31T10:32:43Z' updated: '2025-11-13T00:09:15Z' language: Vue archived: false stars: 168 watchers: 2 forks: 11 owner: ouropencode logo: https://avatars.githubusercontent.com/u/7923793?v=4 license: MIT repoEtag: '"5c1dc6a27cb01593d43b19cdcfc43ce0f18e80ea995dfdd131e3a2c4282c6131"' repoLastModified: Thu, 13 Nov 2025 00:09:15 GMT id: 11c244f76ef65a503bffd1ef4663f38a v3_1_link: https://github.com/ouropencode/OpenDocumenter/issues/2 - source: https://openapi.tools/ name: oas3-api-snippet-enricher category: - Documentation - Parsers language: JavaScript repository: https://github.com/cdwv/oas3-api-snippet-enricher/ link: https://github.com/cdwv/oas3-api-snippet-enricher/ source_description: Enrich your existing description documents with generated code samples v3: true repositoryMetadata: base64Readme: >- IyBFbnJpY2ggeW91ciBPcGVuQVBJIDMuMCBzY2hlbWEgd2l0aCBleGFtcGxlcwoKVGhhbmtzIHRvIHRoZSB3b25kZXJmdWwgW3N3YWdnZXItc25pcHBldF0oaHR0cHM6Ly9naXRodWIuY29tL0VyaWtXaXR0ZXJuL3N3YWdnZXItc25pcHBldCkgbW9kdWxlIHlvdSBjYW4gbm93IHNpbXBseSBlbnJpY2ggeW91ciBPcGVuQVBJIHNjaGVtYSB3aXRoIGNvZGUgc2FtcGxlcy4gSXQncyBhcyBlYXN5IGFzIDEuMi4zLgoKMS4gYGBucG0gaW5zdGFsbCBzbmlwcGV0LWVucmljaGVyLWNsaWBgCjIuIGBgLi9ub2RlX21vZHVsZXMvLmJpbi9zbmlwcGV0LWVucmljaGVyLWNsaSAtLWlucHV0PXlvdXJfb2FzLmpzb25gYAoKIyMgRXhhbXBsZSBVc2FnZQoKRW5yaWNoIHlvdXIgT0FTIDMuMCBTY2hlbWEKYGBgCi4vbm9kZV9tb2R1bGVzLy5iaW4vc25pcHBldC1lbnJpY2hlci1jbGkgLS1pbnB1dD1vcGVuYXBpLmpzb24gPiBvcGVuYXBpLXdpdGgtZXhhbXBsZXMuanNvbgpgYGAKQWx0ZXJuYXRpdmVseSB5b3UgY2FuIHBvaW50IGl0IHRvIGEgWUFNTC1mb3JtYXR0ZWQgc3BlYzoKYGBgCmN1cmwgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFzdGVyL2V4YW1wbGVzL3YzLjAvcGV0c3RvcmUueWFtbCAtLW91dHB1dCBwZXRzdG9yZS55YW1sCi4vbm9kZV9tb2R1bGVzLy5iaW4vc25pcHBldC1lbnJpY2hlci1jbGkgLS1pbnB1dD1wZXRzdG9yZS55YW1sID4gb3BlbmFwaS13aXRoLWV4YW1wbGVzLmpzb24KYGBgCgpVc2UgdGFyZ2V0cyBvcHRpb25zIHRvIHNwZWNpZmljIGxhbmd1YWdlczoKYGBgCi4vbm9kZV9tb2R1bGVzLy5iaW4vc25pcHBldC1lbnJpY2hlci1jbGkgLS10YXJnZXRzPSJub2RlX3JlcXVlc3Qsc2hlbGxfY3VybCIgLS1pbnB1dD1vcGVuYXBpLmpzb24gPiBvcGVuYXBpLXdpdGgtZXhhbXBsZXMuanNvbgpgYGAKClVzZSBbUmVEb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jLykgdG8gYnVpbGQgYmVhdXRpZnVsIEFQSSBkb2M6CmBgYApyZWRvYy1jbGkgYnVuZGxlIG9wZW5hcGktd2l0aC1leGFtcGxlcy5qc29uCmBgYAoKZW5qb3kuCgohW1JlRG9jIEFQSSBkb2N1bWVudGF0aW9uIHdpdGggY29kZSBzYW1wbGVzXShpbWFnZS5wbmcpCgpDb250cmlidXRpbmcKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCkNvbnRyaWJ1dGlvbnMgYXJlIG1vc3Qgd2VsY29tZSEKCgpMaWNlbnNlCj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CgpNSVQKCk1haW50YWluZXJzCj09PT09PT09PT09CgpbPGltZyB3aWR0aD0iMzAwIiB0aXRsZT0iQ29kZXdhdmUuZXUiIHNyYz0iY2R3di1sb2dvLW5ldy5zdmciPl0oaHR0cDovL2NvZGV3YXZlLmV1KQoKUHJvamVjdCBpcyBjdXJyZW50bHkgbWFpbnRhaW5lZCwgaW4gb3VyIHNwYXJlIHRpbWUsIGJ5IFtjb2Rld2F2ZS5ldV0oaHR0cDovL2NvZGV3YXZlLmV1KSBhbmQgYSBncm93aW5nIG51bWJlciBvZiBDb250cmlidXRvcnMhCg== readmeEtag: '"2e480b8b5b9a3c8516169de7cef40e16dade1c06"' readmeLastModified: Mon, 11 Dec 2023 12:08:59 GMT repositoryId: 220618185 description: Enrich your OpenAPI 3.0 JSON with code samples created: '2019-11-09T09:13:16Z' updated: '2025-12-19T14:23:25Z' language: JavaScript archived: false stars: 97 watchers: 8 forks: 14 owner: cdwv logo: https://avatars.githubusercontent.com/u/6742687?v=4 license: MIT repoEtag: '"07f7ab831d4772605944d76b1543ab16c3be38090dabb5f8cb226bfbdcf2de06"' repoLastModified: Fri, 19 Dec 2025 14:23:25 GMT foundInMaster: true id: 90d603135b448f5444c654b9aba124e2 - source: https://openapi.tools/ name: MrinDoc category: Documentation source_description: OpenAPI description document viewer. link: https://mrin9.github.io/OpenAPI-Viewer/ repository: https://github.com/mrin9/openapi-viewer language: Vue.JS v2: true v3: true repositoryMetadata: base64Readme: >- PGltZyBhbHQ9Ik1yaW5Eb2MgbG9nbyIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vbXJpbjkvT3BlbkFQSS1WaWV3ZXIvYmxvYi9tYXN0ZXIvcHVibGljL2ltYWdlcy9sZWF2ZXNfbG9nbzEuc3ZnIiB3aWR0aD0iNjBweCIgLz4KCiMgTXJpbkRvYwpBbiBPcGVuQVBJIDMuMCBhbmQgMi4wIFNwZWMgdmlld2VyIHdpdGggYnVpbHQgaW4gY29uc29sZSAKCiMjIyBbRGVtb10oaHR0cHM6Ly9tcmluOS5naXRodWIuaW8vT3BlbkFQSS1WaWV3ZXIpCgojIyMgQ2hlY2tvdXQgUmFwaURvYyAtIEFub3RoZXIgcHJvamVjdCBmb3IgVmlld2luZyBPcGVuQVBJIHNwZWMgdXNpbmcgd2ViLUNvbXBvbm5ldAojIyMjIFtDaGVja291dCBSYXBpRG9jXShodHRwczovL21yaW45LmdpdGh1Yi5pby9SYXBpRG9jKQoKIyBGZWF0dXJlcwotIFN1cHBvcnRzIFN3YWdnZXIgVjIuMCBhbmQgT3BlbkFQSSAzLjAKLSBTdXBwb3J0cyBzZWFyY2hpbmcgb2YgZW5kLXBvaW50cyAocGF0aHMpCi0gQnVpbHQgSW4gY29uc29sZSB0byB0cnkgb3V0IEFQSXMKLSBSZXNwb25zaXZlIChzdXBwb3J0IGZvciBkZXNrdG9wIGFuZCB0YWJsZXRzKQotIFN1cHBvcnRzIEF1dGhlbnRpY2F0aW9uCiAgLSBPQXV0aDIKICAtIEFQSSBLZXkoVG9rZW4pCiAgLSBIVFRQIEJlYXJlcgogIC0gSFRUUCBCYXNpYwotIFVYIGZyaWVuZGx5LgogIC0gQWxsIEVuZHBvaW50cyBwYXRocyBhcmUgY2xlYW5seSBwcmVzZW50ZWQKICAtIENsZWFyIHNlcGVyYXRpb24gb2YgUmVxdWVzdCBhbmQgUmVzcG9uc2UgaW5mbyBsYWlkIG91dCBzaWRlIGJ5IHNpZGUgaW4gYSB0d28gY29sdW0gbGF5b3V0CiAgLSBMZXNzZXIgY2xpY2tzIHRvIGRlYWwgd2l0aC4gQWxsIHNjaGVtYSBtb2RlbHMsIHJlc3BvbnNlLWpzb25zIGFuZCBleGFtcGxlcyBhcmUgZXhwYW5kZWQgYnkgZGVmYXVsdHMgd2hpY2ggZWxpbWluYXRlcyB0aGUgbmVlZCBvZiBjbGljayBhbmQgcmV2ZWFsIGVhY2ggbGV2ZWwuCiAgLSBXaGlsZSB1c2luZyBBUEkgY29uc29sZSwgcmVxdWVzdCBkYXRhIGlzIHByZS1maWxsZWQgYmFzZWQgb24gc2NoZW1hCiAgLSBOZWVkcyBqdXN0IG9uZSBjbGljayB0byB0cnkgb3V0IGxpdmUgQVBJcwoKCiMjIFByb2plY3QgQnVpbGQKYGBgCnlhcm4gaW5zdGFsbAp5YXJuIHJ1biBidWlsZCAKYGBgCiMjIyBTY3JlZW5zaG90cwojIyMjIEVuZFBvaW50IExpc3RpbmdzCjxrYmQ+CiAgICA8aW1nIHNyYz0iL3NjcmVlbnNob3RzL2VuZC1wb2ludC1saXN0aW5nLnBuZz9yYXc9dHJ1ZSI+Cjwva2JkPgoKIyMjIyBSZXF1ZXN0L1Jlc3BvbnNlCkxhaWQgb3V0IGluIGEgdHdvIGNvbHVtIGZvciBlYXN5IHZpZXcgJiBjb21wYXJlIAo8a2JkPgogICAgPGltZyBzcmM9Ii9zY3JlZW5zaG90cy9yZXF1ZXN0LXJlc3BvbnNlLnBuZz9yYXc9dHJ1ZSI+Cjwva2JkPgoKIyMjIyBTY2hlbWEgVmlldwpBbGwgUmVxdWVzdC9SZXNwb25zZSBzY2hlbWEgYW5kIFJlc3BvbnNlIEpTT05zIGFyZSBleHBhbmRlZCBieSBkZWZhdWx0LCBzbyB0aGF0IHlvdSBkb250IGhhdmUgdG8gY2xpY2sgYW5kIHJldmVhbCBlYWNoIGxldmVsIGZvciBlYXN5IGJyb3dzaW5nIGFuZCByZWFkCjxrYmQ+CiAgICA8aW1nIHNyYz0iL3NjcmVlbnNob3RzL3Jlc3BvbnNlLXNjaGVtYS5wbmc/cmF3PXRydWUiPgo8L2tiZD4KCiMjIyMgQ29uc29sZQpCdWlsdCBpbiBjb25zb2xlLCBmaWVsZHMgYXJlIHByZS1maWxsZWQgd2l0aCB2YWx1ZXMgaWYgZXhpc3QgaW4gc3BlYywgcmVxdWVzdCBib2R5IGlzIGdlbmVyYXRlZCBhbmQgZmlsbGVkIGJhc2VkIG9uIHNjaGVtYSwgYWxsb3dpbmcgeW91IHRvIHF1aWNrbHkgdHJ5IG91dCB0aGUgZW5kLXBvaW50IHdpdGggYSBzaW5nbGUgY2xpY2sgCjxrYmQ+CiAgICA8aW1nIHNyYz0iL3NjcmVlbnNob3RzL2NvbnNvbGUucG5nP3Jhdz10cnVlIj4KPC9rYmQ+CgojIyMjIEF1dGhlbnRpY2F0aW9uIFN1cHBvcnQKQWxsIHRoZSBhdXRoZW50aWNhdGlvbiBtZXRob2RzIGFyZSBzdXBwb3J0ZWQuIEFsb25nIHdpdGggaW5zdHJ1Y3Rpb25zIG9uIGhvdyB0byB1c2UgaXQKPGtiZD4KICAgIDxpbWcgc3JjPSIvc2NyZWVuc2hvdHMvYXV0aGVudGljYXRpb24xLnBuZz9yYXc9dHJ1ZSI+Cjwva2JkPgo8a2JkPgogICAgPGltZyBzcmM9Ii9zY3JlZW5zaG90cy9hdXRoZW50aWNhdGlvbjIucG5nP3Jhdz10cnVlIj4KPC9rYmQ+Cgo= readmeEtag: '"1c4738347c5e8bb21cf1eeceb9278ddd435882e0"' readmeLastModified: Thu, 28 Feb 2019 23:26:40 GMT repositoryId: 159911001 description: OpenApi viewer Implemented using Vue created: '2018-12-01T05:06:18Z' updated: '2025-11-17T10:45:04Z' language: Vue archived: false stars: 105 watchers: 1 forks: 19 owner: mrin9 logo: https://avatars.githubusercontent.com/u/5056899?v=4 repoEtag: '"2787ba514fb2ae3ae7f0f0d5552d9de4f42b32839fd43997d6779cfb1b457703"' repoLastModified: Mon, 17 Nov 2025 10:45:04 GMT foundInMaster: true id: e8e8917a5716d2c743ba73a1b7ba57c4 - source: - https://openapi.tools/ - openapi3 tags name: RapiPdf category: Documentation source_description: Custom Element to generate PDF from OpenAPI descriptions. link: https://mrin9.github.io/RapiPdf repository: https://github.com/mrin9/rapipdf language: Web Component v2: true v3: true repositoryMetadata: base64Readme: >- PGltZyBhbHQ9Ik1yaW5Eb2MgbG9nbyIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vbXJpbjkvUmFwaVBkZi9ibG9iL21hc3Rlci9sb2dvLnBuZyIgd2lkdGg9IjYwcHgiIC8+CgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSIvPgogICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3NpemUvbXJpbjkvcmFwaXBkZi9kaXN0L3JhcGlwZGYtbWluLmpzLnN2Zz9jb2xvckI9Ymx1ZSZsYWJlbD1taW5pZmllZCZzdHlsZT1mbGF0LXNxdWFyZSI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvc2l6ZS9tcmluOS9yYXBpcGRmL2Rpc3QvcmFwaXBkZi1taW4uanMuZ3ouc3ZnP2NvbG9yQj1ibHVlJmxhYmVsPXppcCZzdHlsZT1mbGF0LXNxdWFyZSI+CjwvcD4KCiMgUmFwaVBERgpDdXN0b20gZWxlbWVudCBmb3IgT3Blbi1BUEkgdG8gUERGIGdlbmVyYXRpb24KCiMjIEZlYXR1cmVzCi0gU3VwcG9ydHMgU3dhZ2dlciAyLjAgYW5kIE9wZW5BUEkgMy4wCi0gR2VuZXJhdGUgUERGIHVzaW5nIFdlYi1Db21wb25lbnQKLSBXb3JrcyB3aXRoIGFueSBmcmFtZXdvcmsgb3Igd2l0aCBubyBmcmFtZXdvcmsKLSBQbGVudHkgb2YgY3VzdG9taXppbmcgb3B0aW9ucywgaW5jbHVkaW5nIHNlbGVjdGlvbiBvZiBicmFuZCBjb2xvcnMKLSBTdXBwb3J0ZWQgb24gQ2hyb21lLCBGaXJlRm94IGFuZCBTYWZhcmkuIChOb3QgeWV0IHRlc3RlZCBvbiBFZGdlKQoKIyMgRG9jdW1lbnRhdGlvbgpbQ2hlY2sgb3V0IHRoZSB1c2FnZSBhbmQgZXhhbXBsZXNdKGh0dHBzOi8vbXJpbjkuZ2l0aHViLmlvL1JhcGlQZGYvKQoKIyMgQnVpbGQgUHJvY2VzcwpXZSByZWNvbW1lbmQgYHlhcm5gIG92ZXIgYG5wbWAgYXMgd2UgdXNlIHlhcm4gW3Jlc29sdXRpb25zXShodHRwczovL3lhcm5wa2cuY29tL2xhbmcvZW4vZG9jcy9zZWxlY3RpdmUtdmVyc2lvbi1yZXNvbHV0aW9ucy8pIHRvIGtlZXAgdGhlIGJ1bmRsZSBzaXplIHNtYWxsZXIuIEFzIG9mIHRoaXMgd3JpdGluZyB0aGlzIGZlYXR1cmUgaXMgbm90IHN1cHBvcnRlZCBpbiBucG0gbmF0aXZlbHkKYGBgYmFzaAojIENsb25lIC8gRG93bmxvYWQgdGhlIHByb2plY3QgdGhlbgp5YXJuIGluc3RhbGwKCiMgYnVpbGQgd2lsbCBnZW5lcmF0ZSByYXBpZG9jLW1pbi5qcywgdGhpcyBpcyB0aGUgb25seSBmaWxlIHlvdSB3aWxsIG5lZWQuCiMgdXNlIGl0IGluIHRoZSBzY3JpcHQgdGFnIG9mIHlvdXIgaHRtbCA8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSJyYXBpZG9jLW1pbi5qcyI+PC9zY3JpcHQ+PC9ib2R5Pgp5YXJuIGJ1aWxkCgojIGZvciBkZXZlbG9wZW1lbnQgdXNlIHlhcm4gc2VydmUgKHRoaXMgd2lsbCBzdGFydCBhbiB3ZWJzZXJ2ZXIgYXQgcG9ydCA4MDgwLCB0aGVuIG5hdmlnYXRlIHRvIGxvY2FsaG9zdDo4MDgwKQp5YXJuIHNlcnZlCgojIGFsdGVybmF0aXZlIHRvIHlhcm4gc2VydmU6ICh0aGlzIHdpbGwgc3RhcnQgYW4gd2Vic2VydmVyIGF0IHBvcnQgODA4MCBsaXN0ZW5pbmcgdG8gYWxsIGFkYXB0ZXJzKQp5YXJuIHNlcnZlLWV2ZXJ5b25lCmBgYA== readmeEtag: '"321c11c632be27a5e95bf6eac4cf7441b3d4f28f"' readmeLastModified: Tue, 28 Jul 2020 07:01:44 GMT repositoryId: 176074336 description: PDF generation from OpenAPI / Swagger Spec created: '2019-03-17T08:21:39Z' updated: '2026-02-04T01:07:07Z' language: JavaScript archived: false stars: 306 watchers: 5 forks: 109 owner: mrin9 logo: https://avatars.githubusercontent.com/u/5056899?v=4 license: MIT repoEtag: '"54c52cb22aefbf0ed744f08f9fbe12d37d8d119cb919c38b164052cfe4d409f0"' repoLastModified: Wed, 04 Feb 2026 01:07:07 GMT foundInMaster: true id: 614c2b5cc8e9e45d47072d21fa7d607f - source: https://openapi.tools/ name: Stoplight Docs category: Documentation link: https://stoplight.io/api-documentation language: SaaS source_description: >- Create beautiful, customizable, interactive API documentation generated from OpenAPI, integrated with Stoplight Studio. v2: true v3: true v3_1: true foundInMaster: true id: a9510ff30812653a4089e453440ac0b7 - source: - https://openapi.tools/ - openapi3 tags name: jekyll-openapi category: - Documentation - Server Implementations link: https://github.com/robertlove/jekyll-openapi language: Jekyll source_description: >- An OpenAPI 3 documentation website generator built with Jekyll for use on GitHub Pages. v3: true repository: https://github.com/robertlove/jekyll-openapi repositoryMetadata: base64Readme: >- IyBKZWt5bGwvT3BlbkFQSQoKIVtPQVNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvT0FTLTMuMC4zLWJyaWdodGdyZWVuLnN2ZykKCioqTm90ZToqKiBUaGlzIHByb2plY3QgaXMgKipub3QgZmluaXNoZWQqKiBhbmQgaXMgdW5kZXIgYWN0aXZlIGRldmVsb3BtZW50LiBDb250cmlidXRpb25zIHdlbGNvbWUhCgpKZWt5bGwvT3BlbkFQSSBpcyBhbiBPcGVuQVBJIDMgZG9jdW1lbnRhdGlvbiB3ZWJzaXRlIGdlbmVyYXRvciBidWlsdCB3aXRoIEpla3lsbCBmb3IgdXNlIG9uIEdpdEh1YiBQYWdlcy4KCltKZWt5bGxdKGh0dHBzOi8vamVreWxscmIuY29tLykgaXMgYSBzdGF0aWMgd2Vic2l0ZSBnZW5lcmF0b3IgdGhhdCB3b3JrcyBzZWFtbGVzc2x5IHdpdGggW0dpdEh1YiBQYWdlc10oaHR0cHM6Ly9wYWdlcy5naXRodWIuY29tLyksIHdoaWxzdCB0aGUgW09wZW5BUEkgU3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24pIChPQVMpIGRlZmluZXMgYSBzdGFuZGFyZCwgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2UtYWdub3N0aWMgaW50ZXJmYWNlIGRlc2NyaXB0aW9uIGZvciBSRVNUIEFQSXMuIFB1dCB0aGlzIGFsbCB0b2dldGhlciBhbmQgeW91IGdldCBhIGZhc3QsIHNlY3VyZSBhbmQgaGlnaGx5IGN1c3RvbWlzYWJsZSBzdGF0aWMgd2Vic2l0ZSB3aXRoIGZyZWUgaG9zdGluZyBmb3IgeW91ciBBUEkgZG9jdW1lbnRhdGlvbi4KCiMjIFRhYmxlIG9mIENvbnRlbnRzCgotIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCi0gW1VzYWdlXSgjdXNhZ2UpCi0gW0NvbnRyaWJ1dGluZ10oI2NvbnRyaWJ1dGluZykKLSBbQ3JlZGl0c10oI2NyZWRpdHMpCi0gW0xpY2Vuc2VdKCNsaWNlbnNlKQoKIyMgSW5zdGFsbGF0aW9uCgpCZWZvcmUgeW91IGJlZ2luLCBlbnN1cmUgeW91J3ZlIGluc3RhbGxlZCBbSmVreWxsXShodHRwczovL2pla3lsbHJiLmNvbS8pLiBUaGVuOgoKMS4gYCQgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9yb2JlcnRsb3ZlL2pla3lsbC1vcGVuYXBpLmdpdGAKMS4gYCQgY2QgamVreWxsLW9wZW5hcGlgCjEuIGAkIGJ1bmRsZSBpbnN0YWxsYAoxLiBgJCBidW5kbGUgZXhlYyBqZWt5bGwgc2VydmVgCgpJZiBhbGwgd2VudCB3ZWxsLCB5b3UnbGwgYmUgYWJsZSB0byB2aWV3IHlvdXIgbmV3IEpla3lsbC9PcGVuQVBJIHdlYnNpdGUgbG9jYWxseSBieSBnb2luZyB0byBodHRwOi8vMTI3LjAuMC4xOjQwMDAvamVreWxsLW9wZW5hcGkvIGluIHlvdXIgYnJvd3Nlci4KCiMjIFVzYWdlCgpVcGRhdGUgYF9kYXRhL29wZW5hcGkuanNvbmAgd2l0aCB5b3VyIG93biBBUEkgZGVzY3JpcHRpb24sIHJlZnJlc2ggeW91ciBicm93c2VyLCBhbmQgdm9pbMOgIQoKKipOb3RlOioqIFRoaXMgcHJvamVjdCBpcyBidWlsdCB3aXRoIFtKZWt5bGwvQm9vdHN0cmFwXShodHRwczovL2dpdGh1Yi5jb20vcm9iZXJ0bG92ZS9qZWt5bGwtYm9vdHN0cmFwKS4gU2VlIFtKZWt5bGwvQm9vdHN0cmFwIFVzYWdlXShodHRwczovL2dpdGh1Yi5jb20vcm9iZXJ0bG92ZS9qZWt5bGwtYm9vdHN0cmFwI3VzYWdlKSBmb3IgbW9yZSB1c2FnZSBpbmZvcm1hdGlvbiAtIGluY2x1ZGluZyBpbnN0cnVjdGlvbnMgb24gaG93IHRvIGNoYW5nZSB0aGUgbG9vayBhbmQgZmVlbCBvZiB5b3VyIEFQSSBkb2N1bWVudGF0aW9uLgoKIyMgQ29udHJpYnV0aW5nCgpTZWUgW0NvbnRyaWJ1dGluZ10oaHR0cHM6Ly9naXRodWIuY29tL3JvYmVydGxvdmUvLmdpdGh1Yi9ibG9iL21hc3Rlci9DT05UUklCVVRJTkcubWQpLgoKIyMgQ3JlZGl0cwoKU2VlIFtDb250cmlidXRvcnNdKGh0dHBzOi8vZ2l0aHViLmNvbS9yb2JlcnRsb3ZlL2pla3lsbC1vcGVuYXBpL2dyYXBocy9jb250cmlidXRvcnMpLgoKIyMgTGljZW5zZQoKU2VlIFtMSUNFTlNFXShMSUNFTlNFKS4K readmeEtag: '"afbe1ae786fa174dab9c5405965eae629c46b695"' readmeLastModified: Fri, 11 Jun 2021 00:22:28 GMT repositoryId: 188228581 description: >- An OpenAPI 3 documentation website generator built with Jekyll for use on GitHub Pages created: '2019-05-23T12:20:07Z' updated: '2026-01-21T21:42:47Z' language: HTML archived: true stars: 42 watchers: 3 forks: 7 owner: robertlove logo: https://avatars.githubusercontent.com/u/1000773?v=4 license: Apache-2.0 repoEtag: '"524e1d38afb58b5118fd3bdbc9c25fec5d081b67d79a375ecab2c7837f6b54b7"' repoLastModified: Wed, 21 Jan 2026 21:42:47 GMT foundInMaster: true id: 5c30d43528b7d785fe7aa3f05bb9f9a3 - source: - https://openapi.tools/ - openapi3 tags name: Spot category: - DSL - Parsers repository: https://github.com/airtasker/spot language: TypeScript source_description: A concise, developer-friendly way to describe your API contract. v2: true v3: true repositoryMetadata: base64Readme: >- # Spot

**Spot** (_"Single Point Of Truth"_) is a concise, developer-friendly way to describe your API contract.

Leveraging the TypeScript syntax, it lets you describe your API and generate other API contract formats you need (OpenAPI, Swagger, JSON Schema).

You don't need to use TypeScript in your codebase to benefit from using Spot.

Example of an API definition file `api.ts` which defines a single `POST` endpoint to create a user:

```typescript
import { api, endpoint, request, response, body } from "@airtasker/spot";

@api({
  name: "My API"
})
class Api {}

@endpoint({
  method: "POST",
  path: "/users"
})
class CreateUser {
  @request
  request(@body body: CreateUserRequest) {}

  @response({ status: 201 })
  response(@body body: CreateUserResponse) {}
}

interface CreateUserRequest {
  firstName: string;
  lastName: string;
}

interface CreateUserResponse {
  firstName: string;
  lastName: string;
  role: string;
}
```

## Getting Started

Get started with writing Spot contracts - [Spot Guide](https://github.com/airtasker/spot/wiki/Spot-Guide)

For all available syntax, see [Spot Syntax](https://github.com/airtasker/spot/wiki/Spot-Syntax)

### Requirements

- Node.js >= 18.12.0
- pnpm (recommended) or npm

### Installation

With [pnpm](https://pnpm.io/) installed and initialized add `@airtasker/spot` to your project:

```sh
pnpm add @airtasker/spot
```

You can pass the definition above to a generator by simply running:

```sh
npx @airtasker/spot generate --contract api.ts
```

# Why we built Spot

At first glance, you may wonder why we bothered building Spot. Why not use OpenAPI (formely known as Swagger) to describe your API?

At the core, we built Spot because we wanted a better developer experience.

## Writing contracts

OpenAPI documents are stored as YAML files, following a very specific schema. You won’t know that you used the wrong field name or forgot to wrap a type definition into a schema object unless you run a good OpenAPI linter. Most developers who aren’t intimately familiar with the OpenAPI specification end up using a visual editor such as Swagger Editor or Stoplight.

Since Spot leverages the TypeScript syntax, all you need is to write valid TypeScript code. Your editor will immediately tell you when your code is invalid. It will tell you what’s missing, and you even get autocomplete for free. We could have picked any other typed language—TypeScript just happened to be one of the most concise and ubiquitous for us.

## Reviewing contracts

We believe that API contracts should be checked into Git, or whichever code versioning system you use. In addition, API contracts should be systematically peer reviewed. It’s far too easy for a backend engineer to incorrectly assume what client engineers expect from an endpoint.

Because of their complex nested structure and the richness of the OpenAPI specification, OpenAPI documents can be difficult to review in a pull request. They’re great for machines, but not always for humans.

Spot aims to be as human-readable as possible. We’ve seen developers become a lot more engaged in discussions on pull requests for Spot contracts, compared to our previous OpenAPI documents.

## Interoperability with various formats

Depending on what you're trying to achieve (testing, documentation, client code generation…), you'll find tools that only work with OpenAPI 2 (Swagger), and newer tools that only support OpenAPI 3. You may also find tools for a different API ecosystem such as JSON Schema or API Blueprint.

We built Spot with this in mind. Instead of having to juggle various API format converters, Spot can generate every major API document format. This is why we called it "Single Point Of Truth".

[![oclif](https://img.shields.io/badge/cli-oclif-brightgreen.svg)](https://oclif.io)
[![Version](https://img.shields.io/npm/v/@airtasker/spot.svg)](https://npmjs.org/package/@airtasker/spot)
[![CircleCI](https://circleci.com/gh/airtasker/spot/tree/master.svg?style=shield)](https://circleci.com/gh/airtasker/spot/tree/master)
[![Downloads/week](https://img.shields.io/npm/dw/@airtasker/spot.svg)](https://npmjs.org/package/@airtasker/spot)
[![License](https://img.shields.io/npm/l/@airtasker/spot.svg)](https://github.com/airtasker/spot/blob/master/package.json)

<!-- toc -->
* [Spot](#spot)
* [Why we built Spot](#why-we-built-spot)
* [Usage](#usage)
* [Commands](#commands)
* [Releases](#releases)
<!-- tocstop -->

# Usage

To get started and set up an API declaration in the current directory, run:

```
npx @airtasker/spot init
```

You can then run a generator with:

```
npx @airtasker/spot generate --contract api.ts
```

## In Memory Usage

```ts
import { Spot } from "@airtasker/spot";

const contract = Spot.parseContract("./api.ts");
const openApi = Spot.OpenApi3.generateOpenAPI3(contract);

console.log(openApi);

/*
{
  openapi: '3.0.2',
  info: { title: 'my-api', description: undefined, version: '0.0.0' },
  paths: { '/users': { post: [Object] } },
  components: {
    schemas: { CreateUserRequest: [Object], CreateUserResponse: [Object] },
    securitySchemes: undefined
  },
  security: undefined
}
*/
```

# Commands

<!-- commands -->
* [`spot checksum SPOT_CONTRACT`](#spot-checksum-spot_contract)
* [`spot docs SPOT_CONTRACT`](#spot-docs-spot_contract)
* [`spot generate`](#spot-generate)
* [`spot help [COMMAND]`](#spot-help-command)
* [`spot init`](#spot-init)
* [`spot lint SPOT_CONTRACT`](#spot-lint-spot_contract)
* [`spot mock SPOT_CONTRACT`](#spot-mock-spot_contract)
* [`spot validate SPOT_CONTRACT`](#spot-validate-spot_contract)
* [`spot validation-server SPOT_CONTRACT`](#spot-validation-server-spot_contract)

## `spot checksum SPOT_CONTRACT`

Generate a checksum for a Spot contract

```
USAGE
  $ spot checksum SPOT_CONTRACT

ARGUMENTS
  SPOT_CONTRACT  path to Spot contract

OPTIONS
  -h, --help  show CLI help

EXAMPLE
  $ spot checksum api.ts
```

_See code: [build/cli/src/commands/checksum.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/checksum.js)_

## `spot docs SPOT_CONTRACT`

Preview Spot contract as OpenAPI3 documentation. The documentation server will start on http://localhost:8080.

```
USAGE
  $ spot docs SPOT_CONTRACT

ARGUMENTS
  SPOT_CONTRACT  path to Spot contract

OPTIONS
  -h, --help       show CLI help
  -p, --port=port  [default: 8080] Documentation server port

EXAMPLE
  $ spot docs api.ts
```

_See code: [build/cli/src/commands/docs.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/docs.js)_

## `spot generate`

Runs a generator on an API. Used to produce client libraries, server boilerplates and well-known API contract formats such as OpenAPI.

```
USAGE
  $ spot generate

OPTIONS
  -c, --contract=contract    (required) Path to a TypeScript Contract definition
  -g, --generator=generator  Generator to run
  -h, --help                 show CLI help
  -l, --language=language    Language to generate
  -o, --out=out              Directory in which to output generated files

EXAMPLE
  $ spot generate --contract api.ts --language yaml --generator openapi3 --out output/
```

_See code: [build/cli/src/commands/generate.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/generate.js)_

## `spot help [COMMAND]`

display help for spot

```
USAGE
  $ spot help [COMMAND]

ARGUMENTS
  COMMAND  command to show help for

OPTIONS
  --all  see all commands in CLI
```

_See code: [@oclif/plugin-help](https://github.com/oclif/plugin-help/blob/v3.3.1/src/commands/help.ts)_

## `spot init`

Generates the boilerplate for an API.

```
USAGE
  $ spot init

OPTIONS
  -h, --help  show CLI help

EXAMPLE
  $ spot init
  Generated the following files:
  - api.ts
  - tsconfig.json
  - package.json
```

_See code: [build/cli/src/commands/init.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/init.js)_

## `spot lint SPOT_CONTRACT`

Lint a Spot contract

```
USAGE
  $ spot lint SPOT_CONTRACT

ARGUMENTS
  SPOT_CONTRACT  path to Spot contract

OPTIONS
  -h, --help                                                     show CLI help
  --has-discriminator=(error|warn|off)                           Setting for has-discriminator
  --has-request-payload=(error|warn|off)                         Setting for has-request-payload
  --has-response=(error|warn|off)                                Setting for has-response
  --has-response-payload=(error|warn|off)                        Setting for has-response-payload
  --no-inline-objects-within-unions=(error|warn|off)             Setting for no-inline-objects-within-unions
  --no-nullable-arrays=(error|warn|off)                          Setting for no-nullable-arrays
  --no-nullable-fields-within-request-bodies=(error|warn|off)    Setting for no-nullable-fields-within-request-bodies
  --no-omittable-fields-within-response-bodies=(error|warn|off)  Setting for no-omittable-fields-within-response-bodies
  --no-trailing-forward-slash=(error|warn|off)                   Setting for no-trailing-forward-slash

EXAMPLES
  $ spot lint api.ts
  $ spot lint --has-descriminator=error
  $ spot lint --no-nullable-arrays=off
```

_See code: [build/cli/src/commands/lint.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/lint.js)_

## `spot mock SPOT_CONTRACT`

Run a mock server based on a Spot contract

```
USAGE
  $ spot mock SPOT_CONTRACT

ARGUMENTS
  SPOT_CONTRACT  path to Spot contract

OPTIONS
  -h, --help                                   show CLI help
  -p, --port=port                              (required) [default: 3010] Port on which to run the mock server
  --pathPrefix=pathPrefix                      Prefix to prepend to each endpoint path

  --proxyBaseUrl=proxyBaseUrl                  If set, the server will act as a proxy and fetch data from the given
                                               remote server instead of mocking it

  --proxyFallbackBaseUrl=proxyFallbackBaseUrl  Like proxyBaseUrl, except used when the requested API does not match
                                               defined SPOT contract. If unset, 404 will always be returned.

  --proxyMockBaseUrl=proxyMockBaseUrl          Like proxyBaseUrl, except used to proxy draft endpoints instead of
                                               returning mocked responses.

EXAMPLE
  $ spot mock api.ts
```

_See code: [build/cli/src/commands/mock.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/mock.js)_

## `spot validate SPOT_CONTRACT`

Validate a Spot contract

```
USAGE
  $ spot validate SPOT_CONTRACT

ARGUMENTS
  SPOT_CONTRACT  path to Spot contract

OPTIONS
  -h, --help  show CLI help

EXAMPLE
  $ spot validate api.ts
```

_See code: [build/cli/src/commands/validate.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/validate.js)_

## `spot validation-server SPOT_CONTRACT`

Start the spot contract validation server

```
USAGE
  $ spot validation-server SPOT_CONTRACT

ARGUMENTS
  SPOT_CONTRACT  path to Spot contract

OPTIONS
  -h, --help       show CLI help
  -p, --port=port  [default: 5907] The port where application will be available

EXAMPLE
  $ spot validation-server api.ts
```

_See code: [build/cli/src/commands/validation-server.js](https://github.com/airtasker/spot/blob/v2.0.2/build/cli/src/commands/validation-server.js)_
<!-- commandsstop -->

# Releases

When we're ready for a new release, following steps in the [Wiki](https://github.com/airtasker/spot/wiki/Releasing-Spot)
 readmeEtag: '"d50e525a2e860cb47c99bf4e6a5444dcb6c3c5f6"' readmeLastModified: Fri, 06 Feb 2026 04:55:33 GMT repositoryId: 152569547 description: Spot is a concise, developer-friendly way to describe your API contract. created: '2018-10-11T09:55:46Z' updated: '2026-02-06T04:55:38Z' language: TypeScript archived: false stars: 568 watchers: 48 forks: 38 owner: airtasker logo: https://avatars.githubusercontent.com/u/1745680?v=4 license: NOASSERTION repoEtag: '"2657daed831d41e81478503fc50ec2b3aaa09376cb2a32580c93d25692761272"' repoLastModified: Fri, 06 Feb 2026 04:55:38 GMT foundInMaster: true id: 7c28c31ae17b20b046f57bb6765c3939 - source: https://openapi.tools/ name: OpenAPI Client Generators category: - Code Generators - SDK link: https://github.com/zijianhuang/openapiclientgen language: C# repository: https://github.com/zijianhuang/openapiclientgen source_description: >- .NET Core command line program to generate strongly typed client API codes in C# on .NET Frameworks and .NET Core, and in TypeScript for Angular 5+, Aurelia, jQuery, AXIOS and Fetch API. v2: true v3: true repositoryMetadata: base64Readme: >- # Strongly Typed OpenAPI Client Generators
Reading an OpenAPI / Swagger YAML/JSON definition file, OpenApiClientGen generates strongly typed client API codes in C# and in TypeScript for Angular, Axios, Fetch API, Aurelia and jQuery, as well as Angular strictly typed reactive forms.

This program is based on Fonlow.TypeScriptCodeDomCore and Fonlow.Poco2TsCore which are core components of [WebApiClientGen](https://github.com/zijianhuang/webapiclientgen), thus the codes generated share similar characteristics.

Generated TypeScript codes conform to the [TypeScript strict mode](https://www.typescriptlang.org/tsconfig#strict), and the generated Angular 2+ codes conform to the [Angular strict mode](https://angular.io/guide/strict-mode).

**Hints**

If you are using ASP.NET or ASP.NET Core to develop Web API and would support C# clients and TypeScript clients, WebApiClientGen may provide an alternative solution that covers what offered by Swashbuckle+NSwag. And in some contexts it is more efficient, effective and comprehensive to use WebApiClientGen in SDLC without involving Open API / Swagger YAML/JSON definitions.

## Examples of Generated Codes

OpenApiClientGen had been tested upon over 2,000 Open API definitions in v3 formats and in v2 formats.

*  [pet.yaml](https://github.com/zijianhuang/openapiclientgen/blob/master/Tests/DemoClientApi/pet.yaml), [C# Client API codes generated](https://github.com/zijianhuang/openapiclientgen/blob/master/Tests/DemoClientApi/PetAuto.cs) of client API lib, and   [Integration Tests](https://github.com/zijianhuang/openapiclientgen/tree/master/Tests/SwagApiTests).
* [Generated TypeScript codes for Angular 5+](https://github.com/zijianhuang/openapiclientgen/blob/master/Tests/SwagTsTests/NG2Results/) and [integration tests for pet.yaml](https://github.com/zijianhuang/openapiclientgen/tree/master/ng2/src)
* [Generated TypeScript codes for Angular Reactive Typed Forms](https://github.com/zijianhuang/openapiclientgen/blob/master/Tests/SwagTsTests/NG2FormGroupResults) 
* [Generated TypeScript codes for Aurelia](https://github.com/zijianhuang/openapiclientgen/tree/master/Tests/SwagTests/AureliaResults) and [integration tests for pet.yaml](https://github.com/zijianhuang/openapiclientgen/tree/master/aurelia/src)
* [Generated TypeScript codes for AXIOS](https://github.com/zijianhuang/openapiclientgen/tree/master/Tests/SwagTests/AxiosResults) and [integration tests for pet.yaml](https://github.com/zijianhuang/openapiclientgen/tree/master/axios/src)
* [Generated TypeScript codes for Fetch API](https://github.com/zijianhuang/openapiclientgen/tree/master/Tests/SwagTests/FetchResults) and [integration tests for pet.yaml](https://github.com/zijianhuang/openapiclientgen/tree/master/fetchapi/src)
* [Generated TypeScript codes for jQuery](https://github.com/zijianhuang/openapiclientgen/tree/master/Tests/SwagTests/JqResults) and [integration tests for pet.yaml](https://github.com/zijianhuang/openapiclientgen/tree/master/jq/src)

## Data Mappings

This section describes only mappings for simple and primitive types in OpenAPI definition, C# and TypeScript.

### To C# Codes

```c#
static readonly Dictionary<string, Type> basicClrTypeDic = new()
{
	{"integer_int32", typeof(int) },
	{"integer_int64", typeof(long) },
	{"integer", typeof(int) },
	{"integer_uint32", typeof(uint) },
	{"number_float", typeof(float) },
	{"number_double", typeof(double) },
	{"number_decimal", typeof(decimal) },
	{"number", typeof(double) }, //C# by default use double for number literal
	{"string", typeof(string) },
	{"boolean", typeof(bool) },
	{"string_date", typeof(DateOnly) },
	{"string_date-time", typeof(DateTimeOffset) },
};
```

### To TypeScript Codes

```c#
static readonly Dictionary<string, string> basicTsTypeDic = new()
{
	{"integer_int32", "number" },
	{"integer_int64", "number" },
	{"integer", "number" },
	{"integer_uint32", "number" },
	{"number_float", "number" },
	{"number_double", "number" },
	{"number_decimal", "number" },
	{"number", "number" },
	{"string", "string" },
	{"boolean", "boolean" },
	{"string_date", "Date" },
	{"string_date-time", "Date" },
};
```

And "integer_int64" is actually mapped to "string" because JS supports integral precision up to 53 bits.

**References:**
* [Dealing with Large Integral Numbers in JavaScript for Integral Types of ASP.NET Core Web API](https://www.codeproject.com/Articles/5377807/Dealing-with-Large-Integral-Numbers-in-JavaScript)

## Validation Mappings

### OpenAPI validation to .NET validation

| Schema | .NET Attribute |
| ------ | ---- |  
| MinLength | MinLength |
| MaxLength | MaxLength |
| MinLength + MaxLength | Length |
| Minimum, Maximum | Range(min, max) or Range(TypeFullName, min, max) |
| MinItems | MinLength |
| MaxItems | MaxLength |
| MinItems + MaxItems | Length |
| Pattern | RegularExpression |

### OpenAPI validation to Angular Reactive Forms Validation

| Schema | Angular Reactive Forms Validation |
| ----- | ---- |
| Required | required |
| MinLength, MaxLength, MinItems, MaxItems | minLength, maxLength |
| Minimum, Maximum | min, max |
| Pattern | pattern |

## Installation
OpenApiClientGen is a .NET Core console app.

**Prerequisites**

* .NET 8 (since v3.2)

**Remarks**

* The generated C# codes could be built with .NET Frameworks in addition to .NET.


### Source Installation
Check out this repository or one of its tags, then do a release build or a Visual Studio's Publish.

**Hints**

The plugin assemblies should be copied accordingly after a release build.

### Binary Download

Download one the [zip files](https://github.com/zijianhuang/openapiclientgen/releases) of the releases and extract to a local folder. The console app includes the following plugins for JavaScript/TypeScript libs or frameworks:

* Fonlow.OpenApiClientGen.Aurelia
* Fonlow.OpenApiClientGen.Axios
* Fonlow.OpenApiClientGen.Fetch
* Fonlow.OpenApiClientGen.JQ
* Fonlow.OpenApiClientGen.NG2
* Fonlow.OpenApiClientGen.NG2FormGroup


#### MacOS
If you are using Mac OS or Linux for development, you may use "dotnet publish", for example:

`dotnet publish -r osx-x64 --output %target% --configuration release --self-contained false`

Under folder "Fonlow.OpenApiClienGen", there are 2 batch files to build a release for MacOS:
* BuildForMac.bat
* BuildForMacSelfContained.bat

On Mac, you may run 

```dotnet Fonlow.OpenApiClientGen.dll my.yaml codegen.json```

or 

```./Fonlow.OpenApiClientGen my.yaml codegen.json```

After running `chmod 755 Fonlow.OpenApiClientGen` once.

## Code Generation

When running the program without valid parameters, you get some simple hints:
```
Parameter 1: Open API YAML/JSON definition file
Parameter 2: Settings file in JSON format.
Example:  
  Fonlow.OpenApiClientGen.exe my.yaml
  Fonlow.OpenApiClientGen.exe my.yaml myproj.json
  Fonlow.OpenApiClientGen.exe my.yaml ..\myproj.json
```

### Settings

A typical CodeGen JSON file is like this ["DemoCodeGen.json"](https://github.com/zijianhuang/openapiclientgen/blob/master/DemoCodeGen.json):
```json
{
	"ClientNamespace": "My.Pet.Client",
	"ClientLibraryProjectFolderName": "./Tests/DemoClientApi",
	"ContainerClassName": "PetClient",
	"ClientLibraryFileName": "PetAuto.cs",
	"ActionNameStrategy": 4,
	"UseEnsureSuccessStatusCodeEx": true,
	"DecorateDataModelWithDataContract": true,
	"DataContractNamespace": "http://pet.domain/2020/03",
	"DataAnnotationsEnabled": true,
	"DataAnnotationsToComments": true,
	"HandleHttpRequestHeaders": true,

	"Plugins": [
		{
			"AssemblyName": "Fonlow.OpenApiClientGen.NG2",
			"TargetDir": "./ng2/src/clientapi",
			"TSFile": "ClientApiAuto.ts",
			"AsModule": true,
			"ContentType": "application/json;charset=UTF-8"
		}
	]

}
```

You may run the code generator through a batch file:
```batch
cd %~dp0
c:\green\OpenApiClientGen\Fonlow.OpenApiClientGen.exe some_service.yaml mycodegen.json
```

The JSON file is mapped to the following [C# codes](https://github.com/zijianhuang/openapiclientgen/blob/master/Fonlow.OpenApiClientGen.ClientTypes/Settings.cs) which describes each setting:
```c#
	public class Settings : ISettings
	{
		/// <summary>
		/// The generated codes should be contained in a namespace. The default is My.Namespace.
		/// </summary>
		public string ClientNamespace { get; set; } = "My.Namespace";

		/// <summary>
		/// To compose client function name through removing path prefix. Typically / or /api. The default is /.
		/// The lenght of the prefix is used to remove path prefix. Applied when ActionNameStrategy is PathMethodQueryParameters.
		/// </summary>
		public string PathPrefixToRemove { get; set; } = "/";

		public ActionNameStrategy ActionNameStrategy { get; set; }

		/// <summary>
		/// The default is \w* for picking up alphanumeric words from some crikey operationIds through matching a list of words 
		/// which will be merged into a function name in Pascal or camel casing. 
		/// Applied when ActionNameStrategy is NorrmalizedOperationId.
		/// </summary>
		public string RegexForNormalizedOperationId { get; set; } = @"\w*";

		public ContainerNameStrategy ContainerNameStrategy { get; set; }

		/// <summary>
		/// Container class name when ContainerNameStrategy is None. The default is Misc.
		/// </summary>
		public string ContainerClassName { get; set; } = "Misc";

		/// <summary>
		/// Suffix of container class name if ContainerNameStrategy is not None. The default is "Client".
		/// </summary>
		public string ContainerNameSuffix { get; set; } = "Client";

		/// <summary>
		/// Assuming the client API project is the sibling of Web API project. Relative path to the WebApi project should be fine. C# only.
		/// </summary>
		public string ClientLibraryProjectFolderName { get; set; }

		/// <summary>
		/// The name of the CS file to be generated under client library project folder. The default is OpenApiClientAuto.cs.
		/// </summary>
		public string ClientLibraryFileName { get; set; } = "OpenApiClientAuto.cs";

		/// <summary>
		/// Generated data types will be decorated with DataContractAttribute and DataMemberAttribute in C#.
		/// </summary>
		public bool DecorateDataModelWithDataContract { get; set; }

		/// <summary>
		/// When DecorateDataModelWithDataContract is true, this is the namespace of DataContractAttribute. For example, "http://mybusiness.com/09/2019
		/// </summary>
		public string DataContractNamespace { get; set; }

		/// <summary>
		/// Serialize enum to string. For C#, effective if DecorateDataModelWithDataContract is true, and the enum type is decorated by
		/// [JsonConverter(typeof(Newtonsoft.Json.Converters.StringEnumConverter))] or [System.Text.Json.Serialization.JsonConverter(typeof(System.Text.Json.Serialization.JsonStringEnumConverter))].
		/// For TypeScript, the output is string enums.
		/// </summary>
		public bool EnumToString { get; set; }

		/// <summary>
		/// Decorate the Data Model with the System.SerializableAttribute attribute
		/// </summary>
		public bool DecorateDataModelWithSerializable { get; set; }

		/// <summary>
		/// For .NET client, generate both async and sync functions for each Web API function. When false, only async.
		/// </summary>
		public bool GenerateBothAsyncAndSync { get; set; }

		/// <summary>
		/// Replace EnsureSuccessStatusCode with EnsureSuccessStatusCodeEx for specific unsuccessful HTTP status handling, which throws YourClientWebApiRequestException.
		/// </summary>
		public bool UseEnsureSuccessStatusCodeEx { get; set; }

		/// <summary>
		/// Default  is true so the code block is included in the generated codes.
		/// Defined if UseEnsureSuccessStatusCodeEx is true. Respective code block will be included the code gen output. However, if you have a few client APIs generated to be used in the same application,
		/// and you may want these client APIs share the same code block, then put the WebApiRequestException code block to an assembly or a standalone CS file.
		/// </summary>
		public bool IncludeEnsureSuccessStatusCodeExBlock { get; set; } = true;

		/// <summary>
		/// System.ComponentModel.DataAnnotations attributes are to be copied over, including Required, Range, MaxLength, MinLength and StringLength. Applied to C# only.
		/// What defined in "default" of YAML will become the default value of respective property.
		/// </summary>
		public bool DataAnnotationsEnabled { get; set; }

		/// <summary>
		/// System.ComponentModel.DataAnnotations attributes are translated into Doc Comments, 
		/// including Required, Range, MaxLength, MinLength, StringLength, DataType and RegularExpression. Applied to C# and TypeScript.
		/// </summary>
		public bool DataAnnotationsToComments { get; set; }

		/// <summary>
		/// Function parameters contain a callback to handle HTTP request headers
		/// </summary>
		public bool HandleHttpRequestHeaders { get; set; }

		/// <summary>
		/// Allow cancellation in Send
		/// </summary>
		public bool CancellationTokenEnabled { get; set; }

		/// <summary>
		/// Use System.Text.Json instead of Newtonsoft.Json
		/// </summary>
		public bool UseSystemTextJson { get; set; }

		/// <summary>
		/// For date type, generate DateOnly property. Default true. If false, generate DateTimeOffset decorated by System.ComponentModel.DataAnnotations.DataTypeAttribute(System.ComponentModel.DataAnnotations.DataType.Date)
		/// </summary>
		public bool DateToDateOnly { get; set; } = true;

		/// <summary>
		/// Generated data types will be decorated with System.Text.Json.Serialization.JsonPropertyNameAttribute or Newtonsoft.Json.JsonPropertyAttribute with the original property name defined in YAML.
		/// </summary>
		public bool DecorateDataModelWithPropertyName { get; set; }

		/// <summary>
		/// OpenApClinetGen declares all value type properties including enum properties as nullable by default in generated C#, and all properties as nullable by default in generated TypeScript codes, unless the property is required. 
		/// This is to prevent serializer from creating payload for properties not assigned.  
		/// There might be situations in which you don't want such default features and want the codegen to respect OpenApi v3 option nullable. Then turn this setting to true, which affects generated C# codes only.
		/// Please note, Some Open API definition files do not define nullable for some premitive types and enum, however, the respective backends do not expect some properties presented in the payload of the request.
		/// therefore you need to build some integration test suites to find out what the backend would like.
		/// If the YAML file defines a reference type property as nullable, the codegen ignores this setting since in C# a nullable reference type property is invalid, unless you set UseNullableReferenceType to true.
		/// </summary>
		public bool DisableSystemNullableByDefault { get; set; }

		/// <summary>
		/// Use T? instead of System.Nullable<T> for value types, while by default System.Nullable<T> is used. C# 2.0 feature
		/// </summary>
		public bool UseNullableQuestionMark { get; set; }

		/// <summary>
		/// Use T? for reference types. C# 8.0 feature: https://docs.microsoft.com/en-us/dotnet/csharp/language-reference/builtin-types/nullable-reference-types
		/// https://docs.microsoft.com/en-us/dotnet/csharp/nullable-migration-strategies: The global nullable context does not apply for generated code files.
		/// Therefore it is up to you application programmers to make the compiler recognize the generated file is not of generated codes.
		/// Check test case: Test_vimeo()
		/// </summary>
		public bool UseNullableReferenceType { get; set; }

		/// <summary>
		/// Create the Model classes only
		/// </summary>
		public bool GenerateModelsOnly { get; set; }

		/// <summary>
		/// Use Guid for format:uuid specifier, defaults to string.
		/// </summary>
		public bool UseGuid { get; set; }

		/// <summary>
		/// By default, array type will be array in generated C#. You may generated IEnumerable and some of its derived types.
		/// </summary>
		public ArrayAsIEnumerableDerived ArrayAs { get; set; }

		/// <summary>
		/// Use pascal case for properties and model class names
		/// </summary>
		public bool UsePascalCase { get; set; }

		/// <summary>
		/// Prefix class names with enclosing Type name. Default True.
		/// </summary>
		public bool PrefixWithTypeName { get; set; } = true;

		/// <summary>
		/// Create destination folder if not exists. Applied to both CS and TS.
		/// </summary>
		public bool CreateFolder { get; set; }

		/// <summary>
		/// Type name with dots will have the dotted segments become namespace. APIs\presalytics.io\ooxml\0.1.0 has this, with namespace and type name collisions.
		/// </summary>
		public bool DotsToNamespaces { get; set; }

		/// <summary>
		/// When generating codes, sort types, member properties and methods by alphabetic order.
		/// </summary>
		public bool SortTypesMembersAndMethods { get; set; }

		/// <summary>
		/// Meta for plugins that generate TypeScript/JavaScript codes.
		/// </summary>
		public JSPlugin[] Plugins { get; set; }
	}

		public enum ActionNameStrategy
	{
		/// <summary>
		/// Either OperationId or MethodQueryParameters
		/// </summary>
		Default = 0,

		OperationId = 1,

		/// <summary>
		/// Compose something like GetSomeWhereById1AndId2. Generally used with ContainerNameStrategy.Path
		/// </summary>
		MethodQueryParameters = 2,

		PathMethodQueryParameters = 3,

		/// <summary>
		/// According to Open API specification, it is RECOMMENDED that the naming of operationId follows common programming naming conventions. 
		/// However, some YAML may name operationId after a valid function name. For example, "list-data-sets", "Search by name" or "SearchByName@WkUld". 
		/// Regular expression (regex) may be needed to pick up alphanumeric words from such operationId and create a valid function name.
		/// The default RegexForNormalizedOperationId is /w*.
		/// </summary>
		NormalizedOperationId = 4,
	}

	public enum ContainerNameStrategy
	{
		/// <summary>
		/// All client functions will be constructed in a god class named after ContainerClassName
		/// </summary>
		None = 0,

		/// <summary>
		/// Use tags
		/// </summary>
		Tags = 1,

		/// <summary>
		/// Use path as resource for grouping, as a container class name.
		/// </summary>
		Path = 2,
	}

	public enum ArrayAsIEnumerableDerived
	{
		Array = 0,

		IEnumerable = 1,
		IList = 2,
		ICollection = 3,
		IReadOnlyList = 4,
		IReadOnlyCollection = 5,

		List = 6,
		Collection = 7,
		ReadOnlyCollection = 8,
	}

	/// <summary>
	/// A DTO class, not part of the CodeGen.json 
	/// </summary>
	public class JSOutput
	{
		public string JSPath { get; set; }

		public bool AsModule { get; set; }

		///// <summary>
		///// HTTP content type used in POST of HTTP of NG2. so text/plain could be used to avoid preflight in CORS.
		///// </summary>
		public string ContentType { get; set; }
	}

	public class JSPlugin
	{
		/// <summary>
		/// Assembly file name without extension dll and dir. The assembly file should be in the same directory of the main program.
		/// </summary>
		public string AssemblyName { get; set; }

		/// <summary>
		/// Related path or absolute path. It is recommended to use slash / to sepparate. If you use backlash \, \\ should be used.
		/// </summary>
		public string TargetDir { get; set; }

		/// <summary>
		/// Generated TS file name. For example, MyNgClientAuto.ts
		/// </summary>
		public string TSFile { get; set; }

		///// <summary>
		///// HTTP content type used in POST of HTTP of NG2. For example, "application/json;charset=UTF-8".
		///// </summary>
		public string ContentType { get; set; }

		/// <summary>
		/// True to have "export namespace"; false to have "namespace". jQuery wants "namespace" while Angular 2+ wants "export namespace".
		/// </summary>
		public bool AsModule { get; set; }
	}

```

For more details, especially the contexts of using these settings, please check [Settings Explained](https://github.com/zijianhuang/openapiclientgen/wiki/Settings-Explained).

## Comparison with NSwag

* [Quick Comparison](https://github.com/zijianhuang/openapiclientgen/wiki/Comparison-with-NSwag)
* [Detailed Comparison](https://github.com/zijianhuang/OpenApiExamples/wiki)

## Articles

In addition to Wiki, please refer to these articles:


* [OpenApiClientGen to Generate Strongly Typed Client API Codes Based on Open API Definitions](https://www.codeproject.com/Articles/5262184/OpenApiClientGen-to-Generate-Strongly-Typed-Client) / [Local Copy](Docs/Articles//OpenApiClientGen%20to%20Generate%20Strongly%20Typed%20Client%20API%20Codes%20Based%20on%20Open%20API%20Definitions%20-%20CodeProject.html)
* [Generate Typed Forms of Angular Reactive Forms from Swagger / OpenAPI definitions](https://www.codeproject.com/Articles/5375552/Generate-Typed-Forms-of-Angular-Reactive-Forms-fro) / [Local Copy](Docs/Articles/Generate%20Typed%20Forms%20of%20Angular%20Reactive%20Forms%20from%20Swagger%20_%20OpenAPI%20definitions%20-%20CodeProject.html) readmeEtag: '"8db8b9b3f8984c83a1cdbbc2806708d0448a5982"' readmeLastModified: Sat, 10 May 2025 05:59:58 GMT repositoryId: 247183556 description: >- Generate strongly typed C# and TypeScript client codes from Open API / Swagger definitions supporting jQuery, Angular, AXIOS, Fetch API, Aurelia and Angular Strictly Typed Forms created: '2020-03-14T00:15:16Z' updated: '2025-12-13T22:21:13Z' language: TypeScript archived: false stars: 76 watchers: 2 forks: 15 owner: zijianhuang logo: https://avatars.githubusercontent.com/u/8484970?v=4 license: MIT repoEtag: '"b4ef186c4c730b01436d54dfbe3c6438d5e4aeab80865d5fface4092bcb140e2"' repoLastModified: Sat, 13 Dec 2025 22:21:13 GMT foundInMaster: true id: 7db93864fa7fbbeccc6ada53c6f9cc62 - source: - https://openapi.tools/ - openapi3 tags name: OpenAPI Generator category: - Code Generators - SDK link: https://openapi-generator.tech language: Java repository: https://github.com/openapitools/openapi-generator source_description: >- A template-driven engine to generate documentation, API clients and server stubs in different languages by parsing your OpenAPI Description (community-driven fork of swagger-codegen) v2: true v3: true repositoryMetadata: base64Readme: >- <h1 align="center">OpenAPI Generator</h1>


<div align="center">

[![Stable releases in Maven Central](https://img.shields.io/maven-metadata/v/https/repo1.maven.org/maven2/org/openapitools/openapi-generator/maven-metadata.xml.svg)](http://search.maven.org/#search%7Cgav%7C1%7Cg%3A%22org.openapitools%22%20AND%20a%3A%22openapi-generator%22)
[![Apache 2.0 License](https://img.shields.io/badge/License-Apache%202.0-orange)](./LICENSE)
[![Open Collective backers](https://img.shields.io/opencollective/backers/openapi_generator?color=orange&label=OpenCollective%20Backers)](https://opencollective.com/openapi_generator)
[![Join the Slack chat room](https://img.shields.io/badge/Slack-Join%20the%20chat%20room-orange)](https://join.slack.com/t/openapi-generator/shared_invite/zt-36ucx4ybl-jYrN6euoYn6zxXNZdldoZA)
[![Follow OpenAPI Generator Twitter account to get the latest update](https://img.shields.io/twitter/follow/oas_generator.svg?style=social&label=Follow)](https://twitter.com/oas_generator)
[![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod)](https://gitpod.io/#https://github.com/OpenAPITools/openapi-generator)
[![Conan Center](https://shields.io/conan/v/openapi-generator)](https://conan.io/center/recipes/openapi-generator)
[![Revved up by Develocity](https://img.shields.io/badge/Revved%20up%20by-Develocity-06A0CE?logo=Gradle&labelColor=02303A)](https://ge.openapi-generator.tech/scans)
</div>

<div align="center">

[Master](https://github.com/OpenAPITools/openapi-generator/tree/master) (`7.20.0`):
[![Integration Test2](https://circleci.com/gh/OpenAPITools/openapi-generator.svg?style=shield)](https://circleci.com/gh/OpenAPITools/openapi-generator)
[![Bitrise](https://img.shields.io/bitrise/4a2b10a819d12b67/master?label=bitrise%3A%20Swift+4,5&token=859FMDR8QHwabCzwvZK6vQ)](https://app.bitrise.io/app/4a2b10a819d12b67)

</div>

<div align="center">

:star::star::star: If you would like to contribute, please refer to [guidelines](CONTRIBUTING.md) and a list of [open tasks](https://github.com/openapitools/openapi-generator/issues?q=is%3Aopen+is%3Aissue+label%3A%22help+wanted%22). :star::star::star:

:bangbang: To migrate from Swagger Codegen to OpenAPI Generator, please refer to the [migration guide](docs/migration-from-swagger-codegen.md) :bangbang:

:notebook_with_decorative_cover: For more information, please refer to the [Wiki page](https://github.com/openapitools/openapi-generator/wiki) and [FAQ](https://github.com/openapitools/openapi-generator/wiki/FAQ) :notebook_with_decorative_cover:

:notebook_with_decorative_cover: The eBook [A Beginner's Guide to Code Generation for REST APIs](https://gum.co/openapi_generator_ebook) is a good starting point for beginners :notebook_with_decorative_cover:

:warning: If the OpenAPI spec, templates or any input (e.g. options, environment variables) is obtained from an untrusted source or environment, please make sure you've reviewed these inputs before using OpenAPI Generator to generate the API client, server stub or documentation to avoid potential security issues (e.g. [code injection](https://en.wikipedia.org/wiki/Code_injection)). For security vulnerabilities, please contact [team@openapitools.org](mailto:team@openapitools.org). :warning:

:bangbang: Both "OpenAPI Tools" (https://OpenAPITools.org - the parent organization of OpenAPI Generator) and "OpenAPI Generator" are not affiliated with OpenAPI Initiative (OAI) :bangbang:

</div>

## Sponsors

If you find OpenAPI Generator useful for work, please consider asking your company to support this Open Source project by [becoming a sponsor](https://opencollective.com/openapi_generator). You can also individually sponsor the project by [becoming a backer](https://opencollective.com/openapi_generator).

#### Thank you to our bronze sponsors!

[![NamSor](https://openapi-generator.tech/img/companies/namsor.png)](https://www.namsor.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[![LightBow](https://openapi-generator.tech/img/companies/lightbow.png)](https://www.lightbow.net/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/docspring.png" width="128" height="128">](https://docspring.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/datadog.png" width="128" height="128">](https://datadoghq.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/thales.jpg" width="128" height="128">](https://cpl.thalesgroup.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/apideck.jpg" width="128" height="128">](https://www.apideck.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/pexa.png" width="128" height="128">](https://www.pexa.com.au/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/numary.png" width="128" height="128">](https://www.numary.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/onesignal.png" width="128" height="128">](https://www.onesignal.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/virtualansoftware.png" width="128" height="128">](https://www.virtualansoftware.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/mergedev.jpeg" width="128" height="128">](https://www.merge.dev/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/burkert.jpg" width="128" height="128">](https://www.burkert.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/finbourne.png" width="128" height="128">](https://www.finbourne.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/bumpsh.png" width="128" height="128">](https://bump.sh/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/bileto.png" width="128" height="128">](https://www.bileto.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/bairesdev.png" width="128" height="128">](https://www.bairesdev.com/sponsoring-open-source-projects/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/dmtech.jpeg" width="128" height="128">](https://www.dmtech.de/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/adyen.png" width="128" height="128">](https://adyen.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/fornex.png" width="128" height="128">](https://fornex.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/alloyautomation.png" width="128" height="128">](https://runalloy.com/signup?utm_source=github&utm_medium=referral&utm_campaign=1524_openapigenerator)
[<img src="https://openapi-generator.tech/img/companies/ssstwitter.png" width="128" height="128">](https://ssstwitter.com/?utm_source=github&utm_medium=referral&utm_campaign=sponsor)
[<img src="https://openapi-generator.tech/img/companies/svix.png" width="128" height="128">](https://www.svix.com/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/litslink.png" width="128" height="128">](https://litslink.com/services/artificial-intelligence?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/designli.jpg" width="128" height="128">](https://designli.co?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/itm.png" width="128" height="128">](https://opensource.muenchen.de?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/kong.png" width="128" height="128">](https://konghq.com/products/kong-konnect?utm_medium=referral&utm_source=github&utm_campaign=platform&utm_content=openapi-generator)
[<img src="https://openapi-generator.tech/img/companies/route4me.png" width="128" height="128">](https://route4me.com/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/dm.png" width="128" height="128">](https://www.dotcom-monitor.com/sponsoring-open-source-projects/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/clickit.jpg" width="128" height="128">](https://www.clickittech.com/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/unified_to.jpg" width="128" height="128">](https://unified.to/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/savetwt.jpg" width="128" height="128">](https://savetwt.com/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)
[<img src="https://openapi-generator.tech/img/companies/serpapi.png" width="128" height="128">](https://serpapi.com/?utm_source=openapi-generator&utm_medium=sponsorship&utm_campaign=oss-sponsorship)

#### Thank you GoDaddy for sponsoring the domain names, Linode for sponsoring the VPS, Checkly for sponsoring the API monitoring and Gradle for sponsoring Develocity

[<img src="https://openapi-generator.tech/img/companies/godaddy.png" width="150">](https://www.godaddy.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[![Linode](https://www.linode.com/media/images/logos/standard/light/linode-logo_standard_light_small.png)](https://www.linode.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://encrypted-tbn0.gstatic.com/images?q=tbn%3AANd9GcRAhEYadUyZYzGUotZiSdXkVMqqLGuohyixLl4eUpUV6pAbUULL" width="150">](https://checklyhq.com/?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)
[<img src="https://upload.wikimedia.org/wikipedia/commons/thumb/c/cb/Gradle_logo.png/320px-Gradle_logo.png" width="150">](https://gradle.org?utm_source=openapi_generator&utm_medium=github_webpage&utm_campaign=sponsor)

## Overview

OpenAPI Generator allows generation of API client libraries (SDK generation), server stubs,  documentation and configuration automatically given an [OpenAPI Spec](https://github.com/OAI/OpenAPI-Specification) (both 2.0 and 3.0 are supported). Currently, the following languages/frameworks are supported:

|                                  | Languages/Frameworks                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                       |
| -------------------------------- |--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| **API clients**                  | **ActionScript**, **Ada**, **Apex**, **Bash**, **C**, **C#** (.net 2.0, 3.5 or later, .NET Standard 1.3 - 2.1, .NET Core 3.1, .NET 5.0. Libraries: RestSharp, GenericHost, HttpClient), **C++** (Arduino, cpp-restsdk, Qt5, Tizen, Unreal Engine 4), **Clojure**, **Crystal**, **Dart**, **Elixir**, **Elm**, **Eiffel**, **Erlang**, **Go**, **Groovy**, **Haskell** (http-client, Servant), **Java** (Apache HttpClient 4.x, Apache HttpClient 5.x, Jersey2.x, OkHttp, Retrofit1.x, Retrofit2.x, Feign, RestTemplate, RESTEasy, Vertx, Google API Client Library for Java, Rest-assured, Spring 5 Web Client, Spring 6 RestClient, MicroProfile Rest Client, Helidon), **Jetbrains HTTP Client**, **Julia**, **k6**, **Kotlin**, **Lua**, **N4JS**, **Nim**, **Node.js/JavaScript** (ES5, ES6, AngularJS with Google Closure Compiler annotations, Flow types, Apollo GraphQL DataStore), **Objective-C**, **OCaml**, **Perl**, **PHP**, **PowerShell**, **Python**, **R**, **Ruby**, **Rust** (hyper, reqwest, rust-server), **Scala** (akka, http4s, scalaz, sttp, swagger-async-httpclient, pekko), **Swift** (2.x, 3.x, 4.x, 5.x, 6.x), **Typescript** (AngularJS, Angular (9.x - 19.x), Aurelia, Axios, Fetch, Inversify, jQuery, Nestjs, Node, redux-query, Rxjs), **XoJo**, **Zapier** |
| **Server stubs**                 | **Ada**, **C#** (ASP.NET Core, Azure Functions), **C++** (Oat++, Pistache, Restbed, Qt5 QHTTPEngine), **Erlang**, **F#** (Giraffe), **Go** (net/http, Gin, Echo), **Haskell** (Servant, Yesod), **Java** (MSF4J, Spring, Undertow, JAX-RS: CDI, CXF, Inflector, Jersey, RestEasy, Play Framework, [PKMST](https://github.com/ProKarma-Inc/pkmst-getting-started-examples), [Vert.x](https://vertx.io/), [Apache Camel](https://camel.apache.org/), [Helidon](https://helidon.io/)), **Julia**, **Kotlin** (Spring Boot, [Ktor](https://github.com/ktorio/ktor), [Vert.x](https://vertx.io/)), **PHP** ([Flight](https://docs.flightphp.com/), Laravel, Lumen, [Mezzio (fka Zend Expressive)](https://github.com/mezzio/mezzio), Slim, Silex, [Symfony](https://symfony.com/)), **Python** (FastAPI, Flask), **NodeJS**, **Ruby** (Sinatra, Rails5), **Rust** ([rust-server](https://openapi-generator.tech/docs/generators/rust-server/)), **Scala** (Akka, [Finch](https://github.com/finagle/finch), [Lagom](https://github.com/lagom/lagom), [Play](https://www.playframework.com/), [Cask](https://github.com/com-lihaoyi/cask), Scalatra)                                                                                                                                                    |
| **API documentation generators** | **HTML**, **Confluence Wiki**, **Asciidoc**, **Markdown**, **PlantUML**                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                    |
| **Configuration files**          | [**Apache2**](https://httpd.apache.org/)                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   |
| **Others**                       | **GraphQL**, **JMeter**, **Ktorm**, **MySQL Schema**, **Postman Collection**, **Protocol Buffer**, **WSDL**                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                |

## Table of contents

- [Sponsors](#sponsors)
    - [Thank you to our bronze sponsors!](#thank-you-to-our-bronze-sponsors)
    - [Thank you GoDaddy for sponsoring the domain names, Linode for sponsoring the VPS, Checkly for sponsoring the API monitoring and Gradle for sponsoring Develocity](#thank-you-godaddy-for-sponsoring-the-domain-names-linode-for-sponsoring-the-vps-checkly-for-sponsoring-the-api-monitoring-and-gradle-for-sponsoring-develocity)
- [Overview](#overview)
- [Table of contents](#table-of-contents)
- [1 - Installation](#1---installation)
  - [1.1 - Compatibility](#11---compatibility)
- [1.2 - Artifacts on Maven Central](#12---artifacts-on-maven-central)
  - [1.3 - Download JAR](#13---download-jar)
  - [Launcher Script](#launcher-script)
  - [1.4 - Build Projects](#14---build-projects)
    - [Nix users](#nix-users)
  - [1.5 - Homebrew](#15---homebrew)
  - [1.6 - Docker](#16---docker)
    - [Public Pre-built Docker images](#public-pre-built-docker-images)
    - [OpenAPI Generator CLI Docker Image](#openapi-generator-cli-docker-image)
    - [OpenAPI Generator Online Docker Image](#openapi-generator-online-docker-image)
    - [Development in docker](#development-in-docker)
      - [Troubleshooting](#troubleshooting)
    - [Run Docker in Vagrant](#run-docker-in-vagrant)
  - [1.7 - NPM](#17---npm)
  - [1.8 - pip](#18---pip)
- [2 - Getting Started](#2---getting-started)
- [3 - Usage](#3---usage)
  - [To generate a sample client library](#to-generate-a-sample-client-library)
  - [3.1 - Customization](#31---customization)
  - [3.2 - Workflow Integration (Maven, Gradle, Github, CI/CD)](#32---workflow-integration-maven-gradle-github-cicd)
  - [3.3 - Online OpenAPI generator](#33---online-openapi-generator)
  - [3.4 - License information on Generated Code](#34---license-information-on-generated-code)
  - [3.5 - IDE Integration](#35---ide-integration)
- [4 - Companies/Projects using OpenAPI Generator](#4---companiesprojects-using-openapi-generator)
- [5 - Presentations/Videos/Tutorials/Books](#5---presentationsvideostutorialsbooks)
- [6 - About Us](#6---about-us)
  - [6.1 - OpenAPI Generator Core Team](#61---openapi-generator-core-team)
    - [Core Team Members](#core-team-members)
    - [Template Creator](#template-creator)
    - [How to join the core team](#how-to-join-the-core-team)
  - [6.2 - OpenAPI Generator Technical Committee](#62---openapi-generator-technical-committee)
    - [Members of Technical Committee](#members-of-technical-committee)
  - [6.3 - History of OpenAPI Generator](#63---history-of-openapi-generator)
    - [Founding Members (alphabetical order):](#founding-members-alphabetical-order)
- [7 - License](#7---license)

## [1 - Installation](#table-of-contents)

### [1.1 - Compatibility](#table-of-contents)

The OpenAPI Specification has undergone 3 revisions since initial creation in 2010.  The openapi-generator project has the following compatibilities with the OpenAPI Specification:

| OpenAPI Generator Version                                                                                                                                 | Release Date | Notes                                             |
| --------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------ | ------------------------------------------------- |
| 7.20.0 (upcoming minor release) [SNAPSHOT](https://github.com/OpenAPITools/openapi-generator/wiki/FAQ#how-to-test-with-the-latest-master-of-openapi-generator) | 20.02.2026   | Minor release with breaking changes (with fallback) |
| [7.19.0](https://github.com/OpenAPITools/openapi-generator/releases/tag/v7.19.0) (latest stable release)                                                    | 20.01.2026   | Minor release with breaking changes (with fallback) |
| [6.6.0](https://github.com/OpenAPITools/openapi-generator/releases/tag/v6.6.0)                                                    | 11.05.2023   | Minor release with breaking changes (with fallback) |
| [5.4.0](https://github.com/OpenAPITools/openapi-generator/releases/tag/v5.4.0)                                                    | 31.01.2022   | Minor release with breaking changes (with fallback) |
| [4.3.1](https://github.com/OpenAPITools/openapi-generator/releases/tag/v4.3.1)                                                    | 06.05.2020   | Patch release (enhancements, bug fixes, etc)                       |

OpenAPI Spec compatibility: 1.0, 1.1, 1.2, 2.0, 3.0, 3.1 (beta support)

(We do not publish daily/nightly build. Please use SNAPSHOT instead)

For old releases, please refer to the [**Release**](https://github.com/OpenAPITools/openapi-generator/releases) page.

For decommissioned generators/libraries/frameworks, please refer to [the "Decommission" label](https://github.com/OpenAPITools/openapi-generator/issues?q=label%3ADecommission+is%3Amerged+) in the pull request page.

## [1.2 - Artifacts on Maven Central](#table-of-contents)

You can find our released artifacts on maven central:

**Core:**
```xml
<dependency>
    <groupId>org.openapitools</groupId>
    <artifactId>openapi-generator</artifactId>
    <version>${openapi-generator-version}</version>
</dependency>
```
See the different versions of the [openapi-generator](https://search.maven.org/artifact/org.openapitools/openapi-generator) artifact available on maven central.

**Cli:**
```xml
<dependency>
    <groupId>org.openapitools</groupId>
    <artifactId>openapi-generator-cli</artifactId>
    <version>${openapi-generator-version}</version>
</dependency>
```
See the different versions of the [openapi-generator-cli](https://search.maven.org/artifact/org.openapitools/openapi-generator-cli) artifact available on maven central.

**Maven plugin:**
```xml
<dependency>
    <groupId>org.openapitools</groupId>
    <artifactId>openapi-generator-maven-plugin</artifactId>
    <version>${openapi-generator-version}</version>
</dependency>
```
* See the different versions of the [openapi-generator-maven-plugin](https://search.maven.org/artifact/org.openapitools/openapi-generator-maven-plugin) artifact available on maven central.
* [Readme](https://github.com/OpenAPITools/openapi-generator/blob/master/modules/openapi-generator-maven-plugin/README.md)

**Gradle plugin:**
```xml
<dependency>
    <groupId>org.openapitools</groupId>
    <artifactId>openapi-generator-gradle-plugin</artifactId>
    <version>${openapi-generator-version}</version>
</dependency>
```
* See the different versions of the [openapi-generator-gradle-plugin](https://search.maven.org/artifact/org.openapitools/openapi-generator-gradle-plugin) artifact available on maven central.
* [Readme](https://github.com/OpenAPITools/openapi-generator/blob/master/modules/openapi-generator-gradle-plugin/README.adoc)

### [1.3 - Download JAR](#table-of-contents)
<!-- RELEASE_VERSION -->
If you're looking for the latest stable version, you can grab it directly from Maven.org (Java 11 runtime at a minimum):

JAR location: `https://repo1.maven.org/maven2/org/openapitools/openapi-generator-cli/7.19.0/openapi-generator-cli-7.19.0.jar`

For **Mac/Linux** users:
```sh
wget https://repo1.maven.org/maven2/org/openapitools/openapi-generator-cli/7.19.0/openapi-generator-cli-7.19.0.jar -O openapi-generator-cli.jar
```

For **Windows** users, you will need to install [wget](http://gnuwin32.sourceforge.net/packages/wget.htm) or you can use Invoke-WebRequest in PowerShell (3.0+), e.g.
```
Invoke-WebRequest -OutFile openapi-generator-cli.jar https://repo1.maven.org/maven2/org/openapitools/openapi-generator-cli/7.19.0/openapi-generator-cli-7.19.0.jar
```

After downloading the JAR, run `java -jar openapi-generator-cli.jar help` to show the usage.

For Mac users, please make sure Java 11 is installed (Tips: run `java -version` to check the version), and export `JAVA_HOME` in order to use the supported Java version:
```sh
export JAVA_HOME=`/usr/libexec/java_home -v 1.11`
export PATH=${JAVA_HOME}/bin:$PATH
```

<!-- /RELEASE_VERSION -->
### Launcher Script

One downside to manual jar downloads is that you don't keep up-to-date with the latest released version. We have a Bash launcher script at [bin/utils/openapi-generator.cli.sh](./bin/utils/openapi-generator-cli.sh) which resolves this issue.

To install the launcher script, copy the contents of the script to a location on your path and make the script executable.

An example of setting this up (NOTE: Always evaluate scripts curled from external systems before executing them).

```
mkdir -p ~/bin/openapitools
curl https://raw.githubusercontent.com/OpenAPITools/openapi-generator/master/bin/utils/openapi-generator-cli.sh > ~/bin/openapitools/openapi-generator-cli
chmod u+x ~/bin/openapitools/openapi-generator-cli
export PATH=$PATH:~/bin/openapitools/
```

Now, `openapi-generator-cli` is "installed". On invocation, it will query the GitHub repository for the most recently released version. If this matches the last downloaded jar,
it will execute as normal. If a newer version is found, the script will download the latest release and execute it.

If you need to invoke an older version of the generator, you can define the variable `OPENAPI_GENERATOR_VERSION` either ad hoc or globally. You can export this variable if you'd like to persist a specific release version.

Examples:

```
# Execute latest released openapi-generator-cli
openapi-generator-cli version

# Execute version 4.1.0 for the current invocation, regardless of the latest released version
OPENAPI_GENERATOR_VERSION=4.1.0 openapi-generator-cli version

# Execute version 4.1.0-SNAPSHOT for the current invocation
OPENAPI_GENERATOR_VERSION=4.1.0-SNAPSHOT openapi-generator-cli version

# Execute version 4.0.2 for every invocation in the current shell session
export OPENAPI_GENERATOR_VERSION=4.0.2
openapi-generator-cli version # is 4.0.2
openapi-generator-cli version # is also 4.0.2

# To "install" a specific version, set the variable in .bashrc/.bash_profile
echo "export OPENAPI_GENERATOR_VERSION=4.0.2" >> ~/.bashrc
source ~/.bashrc
openapi-generator-cli version # is always 4.0.2, unless any of the above overrides are done ad hoc
```

### [1.4 - Build Projects](#table-of-contents)

To build from source, you need the following installed and available in your `$PATH:`

* [Java 11](https://adoptium.net/)

* [Apache Maven 3.8.8 or greater](https://maven.apache.org/) (optional)

After cloning the project, you can build it from source using [maven wrapper](https://maven.apache.org/wrapper/):

- Linux: `./mvnw clean install`
- Windows: `mvnw.cmd clean install`

#### Nix users

If you're a nix user, you can enter OpenAPI Generator shell, by typing:
```sh
nix develop
```
It will enter a shell with Java 11 installed.

Direnv supports automatically loading of the nix developer shell, so if you're using direnv too, type:
```sh
direnv allow
```
and have `java` and `mvn` set up with correct versions each time you enter project directory.

The default build contains minimal static analysis (via CheckStyle). To run your build with PMD and Spotbugs, use the `static-analysis` profile:

- Linux: `./mvnw -Pstatic-analysis clean install`
- Windows: `mvnw.cmd -Pstatic-analysis clean install`

### [1.5 - Homebrew](#table-of-contents)

To install, run `brew install openapi-generator`

Here is an example usage to generate a Ruby client:
```sh
openapi-generator generate -i https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml -g ruby -o /tmp/test/
```

To reinstall with the latest master, run `brew uninstall openapi-generator && brew install --HEAD openapi-generator`

To install OpenJDK (pre-requisites), please run
```sh
brew tap AdoptOpenJDK/openjdk
brew install --cask adoptopenjdk11
export JAVA_HOME=`/usr/libexec/java_home -v 1.11`
```

or download installer via https://adoptium.net/

To install Maven (optional), please run
```sh
brew install maven
```

### [1.6 - Docker](#table-of-contents)

#### Public Pre-built Docker images

 - [https://hub.docker.com/r/openapitools/openapi-generator-cli/](https://hub.docker.com/r/openapitools/openapi-generator-cli/) (official CLI)
 - [https://hub.docker.com/r/openapitools/openapi-generator-online/](https://hub.docker.com/r/openapitools/openapi-generator-online/) (official web service)


#### OpenAPI Generator CLI Docker Image

The OpenAPI Generator image acts as a standalone executable. It can be used as an alternative to installing via homebrew, or for developers who are unable to install Java or upgrade the installed version.

To generate code with this image, you'll need to mount a local location as a volume.

Example:

```sh
docker run --rm -v "${PWD}:/local" openapitools/openapi-generator-cli generate \
    -i https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml \
    -g go \
    -o /local/out/go
```

The generated code will be located under `./out/go` in the current directory.

#### OpenAPI Generator Online Docker Image

The openapi-generator-online image can act as a self-hosted web application and API for generating code. This container can be incorporated into a CI pipeline, and requires at least two HTTP requests and some docker orchestration to access generated code.

Example usage:

```sh
# Start container at port 8888 and save the container id
> CID=$(docker run -d -p 8888:8080 openapitools/openapi-generator-online)

# allow for startup
> sleep 10

# Get the IP of the running container (optional)
GEN_IP=$(docker inspect --format '{{.NetworkSettings.IPAddress}}'  $CID)

# Execute an HTTP request to generate a Ruby client
> curl -X POST --header 'Content-Type: application/json' --header 'Accept: application/json' \
-d '{"openAPIUrl": "https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml"}' \
'http://localhost:8888/api/gen/clients/ruby'

{"code":"c2d483.3.4672-40e9-91df-b9ffd18d22b8","link":"http://localhost:8888/api/gen/download/c2d483.3.4672-40e9-91df-b9ffd18d22b8"}

# Download the generated zip file
> wget http://localhost:8888/api/gen/download/c2d483.3.4672-40e9-91df-b9ffd18d22b8

# Unzip the file
> unzip c2d483.3.4672-40e9-91df-b9ffd18d22b8

# Shutdown the openapi generator image
> docker stop $CID && docker rm $CID
```

#### Development in docker

You can use `run-in-docker.sh` to do all development. This script maps your local repository to `/gen`
in the docker container. It also maps `~/.m2/repository` to the appropriate container location.

To execute `mvn package`:

```sh
git clone https://github.com/openapitools/openapi-generator
cd openapi-generator
./run-in-docker.sh mvn package
```

Build artifacts are now accessible in your working directory.

Once built, `run-in-docker.sh` will act as an executable for openapi-generator-cli. To generate code, you'll need to output to a directory under `/gen` (e.g. `/gen/out`). For example:

```sh
./run-in-docker.sh help # Executes 'help' command for openapi-generator-cli
./run-in-docker.sh list # Executes 'list' command for openapi-generator-cli
./run-in-docker.sh generate -i modules/openapi-generator/src/test/resources/3_0/petstore.yaml \
    -g go -o /gen/out/go-petstore -p packageName=petstore # generates go client, outputs locally to ./out/go-petstore
```

##### Troubleshooting

If an error like this occurs, just execute the **./mvnw clean install -U** command:

> org.apache.maven.lifecycle.LifecycleExecutionException: Failed to execute goal org.apache.maven.plugins:maven-surefire-plugin:2.19.1:test (default-test) on project openapi-generator: A type incompatibility occurred while executing org.apache.maven.plugins:maven-surefire-plugin:2.19.1:test: java.lang.ExceptionInInitializerError cannot be cast to java.io.IOException

```sh
./run-in-docker.sh ./mvnw clean install -U
```

> Failed to execute goal org.fortasoft:gradle-maven-plugin:1.0.8:invoke (default) on project openapi-generator-gradle-plugin-mvn-wrapper: org.gradle.tooling.BuildException: Could not execute build using Gradle distribution 'https://services.gradle.org/distributions/gradle-4.7-bin.zip'

Right now: no solution for this one :|

#### Run Docker in Vagrant
Prerequisite: install [Vagrant](https://www.vagrantup.com/downloads.html) and [VirtualBox](https://www.virtualbox.org/wiki/Downloads).
 ```sh
git clone https://github.com/openapitools/openapi-generator.git
cd openapi-generator
vagrant up
vagrant ssh
cd /vagrant
./run-in-docker.sh ./mvnw package
```

### [1.7 - NPM](#table-of-contents)

There is also an [NPM package wrapper](https://www.npmjs.com/package/@openapitools/openapi-generator-cli) available for different platforms (e.g. Linux, Mac, Windows). (JVM is still required)
Please see the [project's README](https://github.com/openapitools/openapi-generator-cli) there for more information.

Install it globally to get the CLI available on the command line:

```sh
npm install @openapitools/openapi-generator-cli -g
openapi-generator-cli version
```

<!-- RELEASE_VERSION -->
To use a specific version of "openapi-generator-cli"

```sh
openapi-generator-cli version-manager set 7.19.0
```

Or install it as dev-dependency:

```sh
npm install @openapitools/openapi-generator-cli -D
```
<!-- /RELEASE_VERSION -->

You can use [locally built JARs](https://github.com/OpenAPITools/openapi-generator-cli?tab=readme-ov-file#use-locally-built-jar) or [`SNAPSHOT` versions](https://github.com/OpenAPITools/openapi-generator-cli?tab=readme-ov-file#use-nightly-snapshot-build) as well.

### [1.8 - pip](#table-of-contents)


> **Platform(s)**: Linux, macOS, Windows
**Install** via [PyPI](https://pypi.org/) (`java` executable is needed to run):

```
pip install openapi-generator-cli
```

To install a specific version
```
pip install openapi-generator-cli==7.19.0
```

You can also install with [jdk4py](https://github.com/activeviam/jdk4py) instead of java binary. (python>=3.10 is required)

```
pip install openapi-generator-cli[jdk4py]
```

Ref: https://github.com/openAPITools/openapi-generator-pip

## [2 - Getting Started](#table-of-contents)

To generate a PHP client for [petstore.yaml](https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml), please run the following
```sh
git clone https://github.com/openapitools/openapi-generator
cd openapi-generator
./mvnw clean package
java -jar modules/openapi-generator-cli/target/openapi-generator-cli.jar generate \
   -i https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml \
   -g php \
   -o /var/tmp/php_api_client
```
(if you're on Windows, replace the last command with `java -jar modules\openapi-generator-cli\target\openapi-generator-cli.jar generate -i https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml -g php -o c:\temp\php_api_client`)

<!-- RELEASE_VERSION -->
You can also download the JAR (latest release) directly from [maven.org](https://repo1.maven.org/maven2/org/openapitools/openapi-generator-cli/7.19.0/openapi-generator-cli-7.19.0.jar)
<!-- /RELEASE_VERSION -->

To get a list of **general** options available, please run `java -jar modules/openapi-generator-cli/target/openapi-generator-cli.jar help generate`

To get a list of PHP specified options (which can be passed to the generator with a config file via the `-c` option), please run `java -jar modules/openapi-generator-cli/target/openapi-generator-cli.jar config-help -g php`

## [3 - Usage](#table-of-contents)

### To generate a sample client library
You can build a client against the [Petstore API](https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml) as follows:

```sh
./bin/generate-samples.sh ./bin/configs/java-okhttp-gson.yaml
```

(On Windows, please install [GIT Bash for Windows](https://gitforwindows.org/) to run the command above)

This script uses the default library, which is `okhttp-gson`. It will run the generator with this command:

```sh
java -jar modules/openapi-generator-cli/target/openapi-generator-cli.jar generate \
  -i https://raw.githubusercontent.com/openapitools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml \
  -g java \
  -t modules/openapi-generator/src/main/resources/Java \
  --additional-properties artifactId=petstore-okhttp-gson,hideGenerationTimestamp=true \
  -o samples/client/petstore/java/okhttp-gson
```

with a number of options. [The java options are documented here.](docs/generators/java.md)

You can also get the options with the `help generate` command (below only shows partial results):

```
NAME
        openapi-generator-cli generate - Generate code with the specified
        generator.

SYNOPSIS
        openapi-generator-cli generate
                [(-a <authorization> | --auth <authorization>)]
                [--api-name-suffix <api name suffix>] [--api-package <api package>]
                [--artifact-id <artifact id>] [--artifact-version <artifact version>]
                [(-c <configuration file> | --config <configuration file>)] [--dry-run]
                [(-e <templating engine> | --engine <templating engine>)]
                [--enable-post-process-file]
                [(-g <generator name> | --generator-name <generator name>)]
                [--generate-alias-as-model] [--git-host <git host>]
                [--git-repo-id <git repo id>] [--git-user-id <git user id>]
                [--global-property <global properties>...] [--group-id <group id>]
                [--http-user-agent <http user agent>]
                [(-i <spec file> | --input-spec <spec file>)]
                [--ignore-file-override <ignore file override location>]
                [--import-mappings <import mappings>...]
                [--instantiation-types <instantiation types>...]
                [--invoker-package <invoker package>]
                [--language-specific-primitives <language specific primitives>...]
                [--legacy-discriminator-behavior] [--library <library>]
                [--log-to-stderr] [--minimal-update]
                [--model-name-prefix <model name prefix>]
                [--model-name-suffix <model name suffix>]
                [--model-package <model package>]
                [(-o <output directory> | --output <output directory>)] [(-p <additional properties> | --additional-properties <additional properties>)...]
                [--package-name <package name>] [--release-note <release note>]
                [--remove-operation-id-prefix]
                [--reserved-words-mappings <reserved word mappings>...]
                [(-s | --skip-overwrite)] [--server-variables <server variables>...]
                [--skip-validate-spec] [--strict-spec <true/false strict behavior>]
                [(-t <template directory> | --template-dir <template directory>)]
                [--type-mappings <type mappings>...] [(-v | --verbose)]

OPTIONS
        -a <authorization>, --auth <authorization>
            adds authorization headers when fetching the OpenAPI definitions
            remotely. Pass in a URL-encoded string of name:header with a comma
            separating multiple values

...... (results omitted)

        -v, --verbose
            verbose mode

```

You can then compile and run the client, as well as unit tests against it:

```sh
cd samples/client/petstore/java/okhttp-gson
mvn package
```

Other generators have [samples](https://github.com/OpenAPITools/openapi-generator/tree/master/samples) too.

### [3.1 - Customization](#table-of-contents)

Please refer to [customization.md](docs/customization.md) on how to customize the output (e.g. package name, version)

### [3.2 - Workflow Integration (Maven, Gradle, Github, CI/CD)](#table-of-contents)

Please refer to [integration.md](docs/integration.md) on how to integrate OpenAPI generator with Maven, Gradle, sbt, Bazel, Github and CI/CD.

### [3.3 - Online OpenAPI generator](#table-of-contents)

Here are the public online services:

- latest stable version: https://api.openapi-generator.tech
- latest master: https://api-latest-master.openapi-generator.tech (updated with latest master every hour)

The server is sponsored by [Linode](https://www.linode.com/) [![Linode Logo](https://www.linode.com/media/images/logos/standard/light/linode-logo_standard_light_small.png)](https://www.linode.com/)

(These services are beta and do not have any guarantee on service level)

Please refer to [online.md](docs/online.md) on how to run and use the `openapi-generator-online` - a web service for `openapi-generator`.

### [3.4 - License information on Generated Code](#table-of-contents)

The OpenAPI Generator project is intended as a benefit for users of the Open API Specification.  The project itself has the [License](#7---license) as specified. In addition, please understand the following points:

* The templates included with this project are subject to the [License](#7---license).
* Generated code is intentionally _not_ subject to the parent project license

When code is generated from this project, it shall be considered **AS IS** and owned by the user of the software.  There are no warranties--expressed or implied--for generated code.  You can do what you wish with it, and once generated, the code is your responsibility and subject to the licensing terms that you deem appropriate.

### [3.5 - IDE Integration](#table-of-contents)

Here is a list of community-contributed IDE plug-ins that integrate with OpenAPI Generator:

- Eclipse: [Codewind OpenAPI Tools for Eclipse](https://www.eclipse.org/codewind/open-api-tools-for-eclipse.html) by [IBM](https://www.ibm.com)
- IntelliJ IDEA: [OpenAPI Generator](https://plugins.jetbrains.com/plugin/8433-openapi-generator) by [Jim Schubert](https://jimschubert.us/#/)
- IntelliJ IDEA: [Senya Editor](https://plugins.jetbrains.com/plugin/10690-senya-editor) by [senya.io](https://senya.io)
- [RepreZen API Studio](https://www.reprezen.com/)
- Visual Studio: [REST API Client Code Generator](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.ApiClientCodeGenerator) by [Christian Resma Helle](https://christian-helle.blogspot.com/)
- Visual Studio Code: [Codewind OpenAPI Tools](https://marketplace.visualstudio.com/items?itemName=IBM.codewind-openapi-tools) by [IBM](https://marketplace.visualstudio.com/publishers/IBM)


## [4 - Companies/Projects using OpenAPI Generator](#table-of-contents)
Here are some companies/projects (alphabetical order) using OpenAPI Generator in production. To add your company/project to the list, please visit [README.md](README.md) and click on the icon to edit the page.

- [Aalborg University](https://www.aau.dk)
- [act coding](https://github.com/actcoding)
- [Adaptant Solutions AG](https://www.adaptant.io/)
- [adesso SE](https://www.adesso.de/)
- [adorsys GmbH & Co.KG](https://adorsys.com/)
- [Adyen](https://www.adyen.com/)
- [Agoda](https://www.agoda.com/)
- [Airthings](https://www.airthings.com/)
- [Aleri Solutions Gmbh](https://www.aleri.de/)
- [Allianz](https://www.allianz.com)
- [Angular.Schule](https://angular.schule/)
- [Aqovia](https://aqovia.com/)
- [Australia and New Zealand Banking Group (ANZ)](http://www.anz.com/)
- [Arduino](https://www.arduino.cc/)
- [ASKUL](https://www.askul.co.jp)
- [Amazon Web Services (AWS)](https://aws.amazon.com/)
- [b<>com](https://b-com.com/en)
- [百度营销](https://e.baidu.com)
- [Bandwidth](https://dev.bandwidth.com)
- [Banzai Cloud](https://banzaicloud.com)
- [BIMData.io](https://bimdata.io)
- [Bithost GmbH](https://www.bithost.ch)
- [Bosch Connected Industry](https://www.bosch-connected-industry.com)
- [Boxever](https://www.boxever.com/)
- [Brevy](https://www.brevy.com)
- [Bunker Holding Group](https://www.bunker-holding.com/)
- [California State University, Northridge](https://www.csun.edu)
- [CAM](https://www.cam-inc.co.jp/)
- [Camptocamp](https://www.camptocamp.com/en)
- [Carlsberg Group](https://www.carlsberggroup.com/)
- [CERN](https://home.cern/)
- [Christopher Queen Consulting](https://www.christopherqueenconsulting.com/)
- [Cisco](https://www.cisco.com/)
- [codecentric AG](https://www.codecentric.de/)
- [CoinAPI](https://www.coinapi.io/)
- [Commencis](https://www.commencis.com/)
- [ConfigCat](https://configcat.com/)
- [cronn GmbH](https://www.cronn.de/)
- [Crossover Health](https://crossoverhealth.com/)
- [Cupix](https://www.cupix.com/)
- [Datadog](https://www.datadoghq.com)
- [DB Systel](https://www.dbsystel.de)
- [Deeporute.ai](https://www.deeproute.ai/)
- [Devsupply](https://www.devsupply.com/)
- [dmTECH GmbH](https://www.dmTECH.de)
- [DocSpring](https://docspring.com/)
- [dwango](https://dwango.co.jp/)
- [Edge Impulse](https://www.edgeimpulse.com/)
- [Element AI](https://www.elementai.com/)
- [Embotics](https://www.embotics.com/)
- [emineo](https://www.emineo.ch)
- [fastly](https://www.fastly.com/)
- [Fenergo](https://www.fenergo.com/)
- [freee](https://corp.freee.co.jp/en/)
- [FreshCells](https://www.freshcells.de/)
- [Fuse](https://www.fuse.no/)
- [Gantner](https://www.gantner.com)
- [GenFlow](https://github.com/RepreZen/GenFlow)
- [GetYourGuide](https://www.getyourguide.com/)
- [Glovo](https://glovoapp.com/)
- [GMO Pepabo](https://pepabo.com/en/)
- [GoDaddy](https://godaddy.com)
- [Gumtree](https://gumtree.com)
- [Here](https://developer.here.com/)
- [IBM](https://www.ibm.com/)
- [Instana](https://www.instana.com)
- [Interxion](https://www.interxion.com)
- [Inquisico](https://inquisico.com)
- [JustStar](https://www.juststarinfo.com)
- [k6.io](https://k6.io/)
- [Klarna](https://www.klarna.com/)
- [Kronsoft Development](https://www.kronsoft.ro/home/)
- [Kubernetes](https://kubernetes.io)
- [Landeshauptstadt München - it@M](https://muenchen.digital/it-at-m/)
- [Linode](https://www.linode.com/)
- [Logicdrop](https://www.logicdrop.com)
- [Lumeris](https://www.lumeris.com)
- [LVM Versicherungen](https://www.lvm.de)
- [MailSlurp](https://www.mailslurp.com)
- [Manticore Search](https://manticoresearch.com)
- [Mastercard](https://developers.mastercard.com)
- [Médiavision](https://www.mediavision.fr/)
- [Metaswitch](https://www.metaswitch.com/)
- [MoonVision](https://www.moonvision.io/)
- [Myworkout](https://myworkout.com)
- [NamSor](https://www.namsor.com/)
- [Neverfail](https://www.neverfail.com/)
- [NeuerEnergy](https://neuerenergy.com)
- [Nokia](https://www.nokia.com/)
- [OneSignal](https://www.onesignal.com/)
- [Options Clearing Corporation (OCC)](https://www.theocc.com/)
- [Openet](https://www.openet.com/)
- [openVALIDATION](https://openvalidation.io/)
- [Oracle](https://www.oracle.com/)
- [Paxos](https://www.paxos.com)
- [Plaid](https://plaid.com)
- [PLAID, Inc.](https://plaid.co.jp/)
- [Pinterest](https://www.pinterest.com)
- [Ponicode](https://ponicode.dev/)
- [Pricefx](https://www.pricefx.com/)
- [PrintNanny](https://www.print-nanny.com/)
- [Prometheus/Alertmanager](https://github.com/prometheus/alertmanager)
- [Qavar](https://www.qavar.com)
- [QEDIT](https://qed-it.com)
- [Qovery](https://qovery.com)
- [Qulix Systems](https://www.qulix.com)
- [Raksul](https://corp.raksul.com)
- [Raiffeisen Schweiz Genossenschaft](https://www.raiffeisen.ch)
- [RedHat](https://www.redhat.com)
- [RepreZen API Studio](https://www.reprezen.com/swagger-openapi-code-generation-api-first-microservices-enterprise-development)
- [REST United](https://restunited.com)
- [Robocorp](https://www.robocorp.com)
- [Robotinfra](https://www.robotinfra.com)
- [Sarvika Technologies Pvt. Ltd.](https://www.sarvika.com)
- [SearchApi](https://www.searchapi.io/)
- [SmartHR](https://smarthr.co.jp/)
- [Sony Interactive Entertainment](https://www.sie.com/en/index.html)
- [Splitit](https://www.splitit.com/)
- [Stingray](http://www.stingray.com)
- [Suva](https://www.suva.ch/)
- [Svix](https://www.svix.com/)
- [Telstra](https://dev.telstra.com)
- [Tencent](https://www.tencent.com)
- [The University of Aizu](https://www.u-aizu.ac.jp/en/)
- [TINQIN](https://www.tinqin.com/)
- [Translucent ApS](https://www.translucent.dk)
- [TravelTime platform](https://www.traveltimeplatform.com/)
- [TribalScale](https://www.tribalscale.com)
- [Trifork](https://trifork.com)
- [TUI InfoTec GmbH](http://www.tui-infotec.com/)
- [Twilio](https://www.twilio.com/)
- [Twitter](https://twitter.com)
- [unblu inc.](https://www.unblu.com/)
- [Veamly](https://www.veamly.com/)
- [VMWare](https://www.vmware.com/)
- [wbt-solutions](https://www.wbt-solutions.de/)
- [Woleet](https://www.woleet.io/)
- [WSO2](https://wso2.com/)
- [Vouchery.io](https://vouchery.io)
- [Xero](https://www.xero.com/)
- [Yahoo Japan](https://www.yahoo.co.jp/)
- [viadee](https://www.viadee.de/)
- [Vonage](https://vonage.com)
- [YITU Technology](https://www.yitutech.com/)
- [Yelp](https://www.yelp.com/)
- [Zalando](https://www.zalando.com)
- [3DS Outscale](https://www.outscale.com/)

## [5 - Presentations/Videos/Tutorials/Books](#table-of-contents)

- 2018/05/12 - [OpenAPI Generator - community drivenで成長するコードジェネレータ](https://ackintosh.github.io/blog/2018/05/12/openapi-generator/) by [中野暁人](https://github.com/ackintosh)
- 2018/05/15 - [Starting a new open-source project](http://jmini.github.io/blog/2018/2018-05-15_new-open-source-project.html) by [Jeremie Bresson](https://github.com/jmini)
- 2018/05/15 - [REST API仕様からAPIクライアントやスタブサーバを自動生成する「OpenAPI Generator」オープンソースで公開。Swagger Codegenからのフォーク](https://www.publickey1.jp/blog/18/rest_apiapiopenapi_generatorswagger_generator.html) by [Publickey](https://www.publickey1.jp)
- 2018/06/08 - [Swagger Codegen is now OpenAPI Generator](https://angular.schule/blog/2018-06-swagger-codegen-is-now-openapi-generator) by [JohannesHoppe](https://github.com/JohannesHoppe)
- 2018/06/21 - [Connect your JHipster apps to the world of APIs with OpenAPI and gRPC](https://fr.slideshare.net/chbornet/jhipster-conf-2018-connect-your-jhipster-apps-to-the-world-of-apis-with-openapi-and-grpc) by [Christophe Bornet](https://github.com/cbornet) at [JHipster Conf 2018](https://jhipster-conf.github.io/)
- 2018/06/22 - [OpenAPI Generator で Gatling Client を生成してみた](https://rohki.hatenablog.com/entry/2018/06/22/073000) at [ソモサン](https://rohki.hatenablog.com/)
- 2018/06/27 - [Lessons Learned from Leading an Open-Source Project Supporting 30+ Programming Languages](https://speakerdeck.com/wing328/lessons-learned-from-leading-an-open-source-project-supporting-30-plus-programming-languages) - [William Cheng](https://github.com/wing328) at [LinuxCon + ContainerCon + CloudOpen China 2018](http://bit.ly/2waDKKX)
- 2018/07/19 - [OpenAPI Generator Contribution Quickstart - RingCentral Go SDK](https://medium.com/ringcentral-developers/openapi-generator-for-go-contribution-quickstart-8cc72bf37b53) by [John Wang](https://github.com/grokify)
- 2018/08/22 - [OpenAPI Generatorのプロジェクト構成などのメモ](https://yinm.info/20180822/) by [Yusuke Iinuma](https://github.com/yinm)
- 2018/09/12 - [RepreZen and OpenAPI 3.0: Now is the Time](https://www.reprezen.com/blog/reprezen-openapi-3.0-upgrade-now-is-the-time) by [Miles Daffin](https://www.reprezen.com/blog/author/miles-daffin)
- 2018/10/31 - [A node package wrapper for openapi-generator](https://github.com/HarmoWatch/openapi-generator-cli)
- 2018/11/03 - [OpenAPI Generator + golang + Flutter でアプリ開発](http://ryuichi111std.hatenablog.com/entry/2018/11/03/214005) by [Ryuichi Daigo](https://github.com/ryuichi111)
- 2018/11/15 - [基于openapi3.0的yaml文件生成java代码的一次实践](https://blog.csdn.net/yzy199391/article/details/84023982) by [焱魔王](https://me.csdn.net/yzy199391)
- 2018/11/18 - [Generating PHP library code from OpenAPI](https://lornajane.net/posts/2018/generating-php-library-code-from-openapi) by [Lorna Jane](https://lornajane.net/) at [LORNAJANE Blog](https://lornajane.net/blog)
- 2018/11/19 - [OpenAPIs are everywhere](https://youtu.be/-lDot4Yn7Dg) by [Jeremie Bresson (Unblu)](https://github.com/jmini) at [EclipseCon Europe 2018](https://www.eclipsecon.org/europe2018)
- 2018/12/09 - [openapi-generator をカスタマイズする方法](https://qiita.com/watiko/items/0961287c02eac9211572) by [@watiko](https://qiita.com/watiko)
- 2019/01/03 - [Calling a Swagger service from Apex using openapi-generator](https://lekkimworld.com/2019/01/03/calling-a-swagger-service-from-apex-using-openapi-generator/) by [Mikkel Flindt Heisterberg](https://lekkimworld.com)
- 2019/01/13 - [OpenAPI GeneratorでRESTful APIの定義書から色々自動生成する](https://ky-yk-d.hatenablog.com/entry/2019/01/13/234108) by [@ky_yk_d](https://twitter.com/ky_yk_d)
- 2019/01/20 - [Contract-First API Development with OpenAPI Generator and Connexion](https://medium.com/commencis/contract-first-api-development-with-openapi-generator-and-connexion-b21bbf2f9244) by [Anil Can Aydin](https://github.com/anlcnydn)
- 2019/01/30 - [Rapid Application Development With API First Approach Using Open-API Generator](https://dzone.com/articles/rapid-api-development-using-open-api-generator) by [Milan Sonkar](https://dzone.com/users/828329/milan_sonkar.html)
- 2019/02/02 - [平静を保ち、コードを生成せよ 〜 OpenAPI Generator誕生の背景と軌跡 〜](https://speakerdeck.com/akihito_nakano/gunmaweb34) by [中野暁人](https://github.com/ackintosh) at [Gunma.web #34 スキーマ駆動開発](https://gunmaweb.connpass.com/event/113974/)
- 2019/02/20 - [An adventure in OpenAPI V3 code generation](https://mux.com/blog/an-adventure-in-openapi-v3-api-code-generation/) by [Phil Cluff](https://mux.com/blog/author/philc/)
- 2019/02/26 - [Building API Services: A Beginner’s Guide](https://medium.com/google-cloud/building-api-services-a-beginners-guide-7274ae4c547f) by [Ratros Y.](https://medium.com/@ratrosy) in [Google Cloud Platform Blog](https://medium.com/google-cloud)
- 2019/02/26 - [Building APIs with OpenAPI: Continued](https://medium.com/@ratrosy/building-apis-with-openapi-continued-5d0faaed32eb) by [Ratros Y.](https://medium.com/@ratrosy) in [Google Cloud Platform Blog](https://medium.com/google-cloud)
- 2019-03-07 - [OpenAPI Generator で Spring Boot と Angular をタイプセーフに繋ぐ](https://qiita.com/chibato/items/e4a748db12409b40c02f) by [Tomofumi Chiba](https://github.com/chibat)
- 2019-03-16 - [A Quick introduction to manual OpenAPI V3](https://vadosware.io/post/quick-intro-to-manual-openapi-v3/) by [vados](https://github.com/t3hmrman) at [VADOSWARE](https://vadosware.io)
- 2019-03-25 - [Access any REST service with the SAP S/4HANA Cloud SDK](https://blogs.sap.com/2019/03/25/integrate-sap-s4hana-cloud-sdk-with-open-api/) by [Alexander Duemont](https://people.sap.com/alexander.duemont)
- 2019-03-25 - [OpenAPI generatorを試してみる](https://qiita.com/amuyikam/items/e8a45daae59c68be0fc8) by [@amuyikam](https://twitter.com/amuyikam)
- 2019-03-27 - [OpenAPI3を使ってみよう！Go言語でクライアントとスタブの自動生成まで！](https://techblog.zozo.com/entry/openapi3/go) by [@gold_kou](https://twitter.com/gold_kou)
- 2019-04-17 - [OpenAPIによるスキーマファースト開発の実施サンプルとCloud Runについて](https://tech-blog.optim.co.jp/entry/2019/04/17/174000) by [@yukey1031](https://twitter.com/yukey1031)
- 2019-04-18 - [How to use OpenAPI3 for API developer (RubyKaigi 2019)](https://speakerdeck.com/ota42y/how-to-use-openapi3-for-api-developer) by [@ota42y](https://twitter.com/ota42y) at [RubyKaigi 2019](https://rubykaigi.org/2019)
- 2019-04-29 - [A Beginner's Guide to Code Generation for REST APIs (OpenAPI Generator)](https://gum.co/openapi_generator_ebook) by [William Cheng](https://twitter.com/wing328)
- 2019-05-01 - [Design and generate a REST API from Swagger / OpenAPI in Java, Python, C# and more](https://simply-how.com/design-and-generate-api-code-from-openapi) by [Simply How](https://simply-how.com/)
- 2019-05-17 - [Generate Spring Boot REST API using Swagger/OpenAPI](https://www.47northlabs.com/knowledge-base/generate-spring-boot-rest-api-using-swagger-openapi/) by [Antonie Zafirov](https://www.47northlabs.com/author/antonie-zafirov/)
- 2019-05-22 - [REST APIs代码生成指南(OpenAPI Generator)](https://gum.co/openapi_generator_ebook_gb) by [William Cheng](https://twitter.com/wing328), [Xin Meng](https://github.com/xmeng1)
- 2019-05-24 - [REST API 代碼生成指南 (OpenAPI Generator)](https://gum.co/openapi_generator_ebook_big5) by [William Cheng](https://twitter.com/wing328)
- 2019-06-24 - [Kubernetes Clients and OpenAPI Generator](https://speakerdeck.com/wing328/kubernetes-clients-and-openapi-generator) by [William Cheng](https://twitter.com/wing328) at [Kubernetes Contributor Summits Shanghai 2019](https://www.lfasiallc.com/events/contributors-summit-china-2019/)
- 2019-06-28 [Codewind OpenAPI Tools](https://marketplace.eclipse.org/content/codewind-openapi-tools) in [Eclipse Marketplace](https://marketplace.eclipse.org/) by IBM
- 2019-06-29 [Codewind OpenAPI Tools](https://marketplace.visualstudio.com/items?itemName=IBM.codewind-openapi-tools) in [Visual Studio Marketplace](https://marketplace.visualstudio.com/) by IBM
- 2019-07-04 - [REST API のためのコード生成入門 (OpenAPI Generator)](https://gum.co/openapi_generator_ebook_big5) by [William Cheng](https://twitter.com/wing328), [中野暁人](https://github.com/ackintosh), [和田拓朗](https://github.com/taxpon)
- 2019-07-08 - [OpenAPI Generator にコントリビュートしたら社名が載った話。(CAM) - CAM TECH BLOG](https://tech.cam-inc.co.jp/entry/2019/07/08/140000) by [CAM, Inc.](https://www.cam-inc.co.jp/)
- 2019-07-14 - [OpenAPI GeneratorでPythonのクライアントライブラリを作成した](https://qiita.com/yuji38kwmt/items/dfb929316a1335a161c0) by [yuji38kwmt](https://qiita.com/yuji38kwmt)
- 2019-07-19 - [Developer Experience (DX) for Open-Source Projects: How to Engage Developers and Build a Growing Developer Community](https://speakerdeck.com/wing328/developer-experience-dx-for-open-source-projects-english-japanese) by [William Cheng](https://twitter.com/wing328), [中野暁人](https://github.com/ackintosh) at [Open Source Summit Japan 2019](https://events.linuxfoundation.org/events/open-source-summit-japan-2019/)
- 2019-08-14 - [Our OpenAPI journey with Standardizing SDKs](https://bitmovin.com/our-openapi-journey-with-standardizing-sdks/) by [Sebastian Burgstaller](https://bitmovin.com/author/sburgstaller/) at [Bitmovin](https://www.bitmovin.com)
- 2019-08-15 - [APIのコードを自動生成させたいだけならgRPCでなくてもよくない?](https://www.m3tech.blog/entry/2019/08/15/110000) by [M3, Inc.](https://corporate.m3.com/)
- 2019-08-22 - [マイクロサービスにおけるWeb APIスキーマの管理─ GraphQL、gRPC、OpenAPIの特徴と使いどころ](https://employment.en-japan.com/engineerhub/entry/2019/08/22/103000) by [@ota42y](https://twitter.com/ota42y)
- 2019-08-24 - [SwaggerドキュメントからOpenAPI Generatorを使ってモックサーバー作成](https://qiita.com/masayoshi0222/items/4845e4c715d04587c104) by [坂本正義](https://qiita.com/masayoshi0222)
- 2019-08-29 - [OpenAPI初探](https://cloud.tencent.com/developer/article/1495986) by [peakxie](https://cloud.tencent.com/developer/user/1113152) at [腾讯云社区](https://cloud.tencent.com/developer)
- 2019-08-29 - [全面进化：Kubernetes CRD 1.16 GA前瞻](https://www.servicemesher.com/blog/kubernetes-1.16-crd-ga-preview/) by [Min Kim](https://github.com/yue9944882) at [ServiceMesher Blog](https://www.servicemesher.com/blog/)
- 2019-09-01 - [Creating a PHP-Slim server using OpenAPI (Youtube video)](https://www.youtube.com/watch?v=5cJtbIrsYkg) by [Daniel Persson](https://www.youtube.com/channel/UCnG-TN23lswO6QbvWhMtxpA)
- 2019-09-06 - [Vert.x and OpenAPI](https://wissel.net/blog/2019/09/vertx-and-openapi.html) by [Stephan H Wissel](https://twitter.com/notessensei) at [wissel.net blog](https://wissel.net)
- 2019-09-09 - [Cloud-native development - Creating RESTful microservices](https://cloud.ibm.com/docs/cloud-native?topic=cloud-native-rest-api) in [IBM Cloud Docs](https://cloud.ibm.com/docs)
- 2019-09-14 - [Generating and Configuring a Mastercard API Client](https://developer.mastercard.com/platform/documentation/generating-and-configuring-a-mastercard-api-client/) at [Mastercard Developers Platform](https://developer.mastercard.com/platform/documentation/)
- 2019-09-15 - [OpenAPI(Swagger)導入下調べ](https://qiita.com/ShoichiKuraoka/items/f1f7a3c2376f7cd9c56a) by [Shoichi Kuraoka](https://qiita.com/ShoichiKuraoka)
- 2019-09-17 - [Tutorial: Documenting http4k APIs with OpenApi3](https://www.http4k.org/tutorials/documenting_apis_with_openapi/) by [http4k](https://www.http4k.org/)
- 2019-09-22 - [OpenAPI 3を完全に理解できる本](https://booth.pm/ja/items/1571902) by [@ota42y](https://twitter.com/ota42y)
- 2019-09-22 - [RESTful APIs: Tutorial of OpenAPI Specification](https://medium.com/@amirm.lavasani/restful-apis-tutorial-of-openapi-specification-eeada0e3901d) by [Amir Lavasani](https://medium.com/@amirm.lavasani)
- 2019-09-22 - [Redefining SDKs as software diversity kits](https://devrel.net/dev-rel/redefining-sdks-as-software-diversity-kits) by [Sid Maestre (Xero)](https://twitter.com/sidneyallen) at [DevRelCon San Francisco 2019](https://sf2019.devrel.net/)
- 2019-09-23 - [swaggerからOpenApi GeneratorでSpringのコードを自動生成](https://qiita.com/littleFeet/items/492df2ad68a0799a5e5e) by [@littleFeet](https://qiita.com/littleFeet) at [Qiita](https://qiita.com/)
- 2019-09-24 - [Eine Stunde was mit Api First!](https://www.slideshare.net/JanWeinschenker/eine-stunde-was-mit-api-first) by [@janweinschenker](https://twitter.com/janweinschenker) at [Java Forum Nord](https://javaforumnord.de/)
- 2019-10-09 - [openapi-generator で生成した Go クライアントで Bearer 認証をする](https://autopp-tech.hatenablog.com/entry/2019/10/09/222039) by [Akira Tanimura](https://github.com/autopp)
- 2019-10-10 - [Automatic Generation of REST Clients](https://www.meetup.com/fr-FR/Criteo-Labs-Tech-Talks/events/264775768/) by Thomas Peyrard, Senior Software Engineer at Criteo in [Full-Stack Tech Talks (Meetup)](https://www.meetup.com/fr-FR/Criteo-Labs-Tech-Talks/events/264775768/)
- 2019-10-12 - [OpenApi自动生成client](https://blog.csdn.net/wxid2798226/article/details/102527467) by [郑泽洲](https://me.csdn.net/wxid2798226)
- 2019-10-16 - [How to ship APIs faster?](https://medium.com/@accounts_76224/how-to-ship-apis-faster-cabef2f819e4) by [Simon Guilliams @ PoniCode](https://ponicode.dev)
- 2019-10-22 - [OpenAPI + Spring Boot(Kotlin)でファイルダウンロードAPIを作成する](https://qiita.com/boronngo/items/4b78b92526209daeaee9) by [Yuki Furukawa](https://twitter.com/yuki_furukawa5)
- 2019-10-24 - [Microprofile OpenAPI - Code First or Design First?](https://github.com/pe-st/apidocs/blob/master/MicroProfile-OpenAPI-all-slides.pdf) by [Peter [pɛʃə] Steiner](https://twitter.com/pesche) at [eclipsecon Europe 2019](https://www.eclipsecon.org/europe2019/sessions/microprofile-openapi-code-first-or-design-first)
- 2019-11-06 - [Generating API clients based on OpenAPI v3 specifications](https://98elements.com/blog/generating-api-clients-based-on-openapi-v3-specifications) by [Dominik Jastrzębski @ 98elements](https://98elements.com)
- 2019-11-06 - [OpenAPIを利用して自前のAPIサーバー(Sinatra)を移植した時のメモ](https://qiita.com/YasuhiroABE/items/c73920eab2d9d6e97fd9) by [Yasuhiro ABE](https://twitter.com/YasuhiroABE)
- 2019-11-07 - [API First development with OpenAPI - You should you practise it !?](https://www.youtube.com/watch?v=F9iF3a1Z8Y8) by [Nick Van Hoof](https://www.nickvanhoof.com/) at [Devoxx Belgium 2019](https://devoxx.be/)
- 2019-11-08 - [JHipster beyond CRUD - API-First for Enterprises by Enrico Costanzi](https://www.youtube.com/watch?v=m28JFovKQ20) by [Enrico Costanzi](https://twitter.com/enricocostanzi) at [JHipster Conf 2019 in Paris](https://jhipster-conf.github.io/)
- 2019-11-11 - [TypeScript REST APIクライアント](https://qiita.com/unhurried/items/7b74f7d3c43545dadd2b) by [@unhurried](https://qiita.com/unhurried)
- 2019-11-11 - [One Spec to Rule them all - OpenAPI in Action](https://www.youtube.com/watch?v=MMay_nht8ec) by [Andreas Litt](https://github.com/littldr) at [code.talks 2019](https://www.codetalks.com/)
- 2019-11-13 - [OpenAPI 3.0 Editor And Generator With A Spring Boot Example](https://simply-how.com/design-and-generate-api-code-from-openapi) at [Simply How](https://simply-how.com/)
- 2019-11-17 - [OpenAPI Generator YouTube playlist](https://www.youtube.com/playlist?list=PLtJyHVMdzfF6fBkOUV5VDVErP23CGgHIy) at [YouTube](https://www.youtube.com)
- 2019-11-20 - [Introduction to OpenAPI](https://noti.st/lornajane/HvDH7U/introduction-to-openapi) by [Lorna Mitchell](https://twitter.com/lornajane) at [GOTO Copenhagen 2019](https://gotocph.com/2019/)
- 2019-11-20 - [How to Generate Angular code from OpenAPI specifications](https://dotnetthoughts.net/how-to-generate-angular-code-from-openapi-specifications/) by Anuraj
- 2019-11-23 - [Swagger ではない OpenAPI Specification 3.0 による API サーバー開発](https://www.slideshare.net/techblogyahoo/swagger-openapi-specification-30-api) by [Tetsuya Morimoto](https://github.com/t2y) at [JJUG CCC 2019 Fall](https://ccc2019fall.java-users.jp/)
- 2019-11-24 - [Accelerate Flutter development with OpenAPI and Dart code generation](https://medium.com/@irinasouthwell_220/accelerate-flutter-development-with-openapi-and-dart-code-generation-1f16f8329a6a) by [Irina Southwell](https://medium.com/@irinasouthwell_220)
- 2019-11-25 - [openapi-generatorで手軽にスタブサーバとクライアントの生成](https://qiita.com/pochopocho13/items/8db662e1934fb2b408b8) by [@pochopocho13](https://twitter.com/pochopocho13)
- 2019-11-26 - [CordaCon 2019 Highlights: Braid Server and OpenAPI Generator for Corda Client API’s](https://blog.b9lab.com/cordacon-2019-highlights-braid-server-and-openapi-generator-for-corda-flows-api-s-d24179ccb27c) by [Adel Rustum](https://blog.b9lab.com/@adelrestom) at [B9lab](https://blog.b9lab.com/)
- 2019-12-03 - [A Road to Less Coding: Auto-Generate APILibrary](https://www.corda.net/blog/a-road-to-less-coding-auto-generate-apilibrary/) at [Corda Blog](https://www.corda.net/blog/)
- 2019-12-04 - [Angular＋NestJS＋OpenAPI（Swagger）でマイクロサービスを視野に入れた環境を考える](https://qiita.com/teracy55/items/0327c7a170ec772970c6) by [てらしー](https://twitter.com/teracy55)
- 2019-12-05 - [Code generation on the Java VM](https://speakerdeck.com/sullis/code-generation-on-the-java-vm-2019-12-05) by [Sean Sullivan](https://speakerdeck.com/sullis)
- 2019-12-17 - [OpenAPI Generator で OAuth2 アクセストークン発行のコードまで生成してみる](https://www.techscore.com/blog/2019/12/17/openapi-generator-oauth2-accesstoken/) by [TECHSCORE](https://www.techscore.com/blog/)
- 2019-12-23 - [Use Ada for Your Web Development](https://www.electronicdesign.com/technologies/embedded-revolution/article/21119177/use-ada-for-your-web-development) by [Stephane Carrez](https://github.com/stcarrez)
- 2019-12-23 - [OpenAPIのスキーマを分割・構造化していく方法](https://gift-tech.co.jp/articles/structured-openapi-schema) by [小飯塚達也](https://github.com/t2h5) at [GiFT, Inc](https://gift-tech.co.jp/)
- 2020-01-17 - [OpenAPI demo for Pulp 3.0 GA](https://www.youtube.com/watch?v=mFBP-M0ZPfw&t=178s) by [Pulp](https://www.youtube.com/channel/UCI43Ffs4VPDv7awXvvBJfRQ) at [Youtube](https://www.youtube.com/)
- 2020-01-19 - [Why document a REST API as code?](https://dev.to/rolfstreefkerk/why-document-a-rest-api-as-code-5e7p) by [Rolf Streefkerk](https://github.com/rpstreef) at [DEV Community](https://dev.to)
- 2020-01-28 - [Get Your Serverless Swagger Back with OpenAPI](https://dev.to/matttyler/get-your-serverless-swagger-back-with-openapi-48gc) by [Matt Tyler](https://dev.to/matttyler)
- 2020-01-30 - [OpenAPI Generatorへのコントリビュート](https://www.yutaka0m.work/entry/2020/01/30/163905) by [yutaka0m](https://github.com/yutaka0m)
- 2020-02-01 - [Using OpenAPI to Maximise Your Pulp 3 Experience](https://fosdem.org/2020/schedule/event/openapi/) by [Dennis Kliban](https://github.com/dkliban/) at [FOSDEM](https://fosdem.org/)
- 2020-02-07 - [Why you should use OpenAPI for your API design](https://www.youtube.com/watch?v=zhb7vUApLW8&t=927s) by [Nick Van Hoof](https://apiconference.net/speaker/nick-van-hoof/) at [API Conference](https://apiconference.net/)
- 2020-02-17 - [Rubynetes: using OpenAPI to validate Kubernetes configs](https://www.brightbox.com/blog/2020/02/17/using-openapi-to-validate-kubernetes-configs/) by Neil Wilson at [Brightbox](https://www.brightbox.com/)
- 2020-02-20 - [Building SDKs for the future](https://devblog.xero.com/building-sdks-for-the-future-b79ff726dfd6) by [Sid Maestre (Xero)](https://twitter.com/sidneyallen)
- 2020-02-27 - [Nuxt利用プロダクトでIE11と仲良くするためのE2E](https://tech.medpeer.co.jp/entry/e2e-ie11) at [Medpeer.co.jp Tech Blog](https://tech.medpeer.co.jp/)
- 2020-02-29 - [Providing Support to IoT Devices Deployed in Disconnected Rural Environment (Conference paper)](https://link.springer.com/chapter/10.1007/978-3-030-41494-8_14) by Sergio Laso, Daniel Flores-Martín, Juan Luis HerreraCarlos, CanalJuan Manuel, MurilloJavier Berrocal
- 2020-03-02 - [How To Generate Angular & Spring Code From OpenAPI Specification](https://www.mokkapps.de/blog/how-to-generate-angular-and-spring-code-from-open-api-specification/) by [Michael Hoffmann](https://www.mokkapps.de/)
- 2020-03-02 - [OpenAPI Generator + TypeScript で始める自動生成の型に守られた豊かなクライアント生活](https://gift-tech.co.jp/articles/openapi-generator-typescript) by [五百蔵 直樹](https://gift-tech.co.jp/members/naokiioroi) at [GiFT株式会社](https://gift-tech.co.jp/)
- 2020-03-10 - [OpenAPI Generator Meetup #1](https://speakerdeck.com/akihito_nakano/openapi-generator-meetup-number-1) by [中野暁人](https://github.com/ackintosh) at [OpenAPI Generator Meetup #1](https://openapi-generator-meetup.connpass.com/event/168187/)
- 2020-03-15 - [Load Testing Your API with Swagger/OpenAPI and k6](https://k6.io/blog/load-testing-your-api-with-swagger-openapi-and-k6)
- 2020-04-13 - [俺的【OAS】との向き合い方 (爆速でOpenAPIと友達になろう)](https://tech-blog.optim.co.jp/entry/2020/04/13/100000) in [OPTim Blog](https://tech-blog.optim.co.jp/)
- 2020-04-22 - [Introduction to OpenAPI Generator](https://nordicapis.com/introduction-to-openapi-generator/) by [Kristopher Sandoval](https://nordicapis.com/author/sandovaleffect/) in [Nordic APIs](https://nordicapis.com/)
- 2020-04-27 - [How we use Open API v3 specification to auto-generate API documentation, code-snippets and clients](https://medium.com/pdf-generator-api/how-we-use-open-api-v3-specification-to-auto-generate-api-documentation-code-snippets-and-clients-d127a3cea784) by [Tanel Tähepõld](https://medium.com/@tanel.tahepold)
- 2020-05-09 - [OpenAPIでお手軽にモックAPIサーバーを動かす](https://qiita.com/kasa_le/items/97ca6a8dd4605695c25c) by [Sachie Kamba](https://qiita.com/kasa_le)
- 2020-05-18 - [Spring Boot REST with OpenAPI 3](https://dev.to/alfonzjanfrithz/spring-boot-rest-with-openapi-3-59jm) by [Alfonz Jan Frithz](https://dev.to/alfonzjanfrithz)
- 2020-05-19 - [Dead Simple APIs with Open API](https://www.youtube.com/watch?v=sIaXmR6xRAw) by [Chris Tankersley](https://github.com/dragonmantank) at [Nexmo](https://developer.nexmo.com/)
- 2020-05-22 - [TypeScript REST API Client](https://dev.to/unhurried/typescript-rest-api-client-4in3) by ["unhurried"](https://dev.to/unhurried)
- 2020-05-28 - [【使用 lotify + Swagger 建置可共用的 LINE Notify bot】 - #NiJia @ Chatbot Developer Taiwan 第 #19 小聚](https://www.youtube.com/watch?v=agYVz6dzh1I) by [Chatbot Developer Taiwan](https://www.youtube.com/channel/UCxeYUyZNnHmpX23YNF-ewvw)
- 2020-05-28 - [Building APIs with Laravel using OpenAPI](https://www.youtube.com/watch?v=xexLvQqAhiA) by [Chris Tankersley](https://github.com/dragonmantank) at [Laracon EU](https://laracon.eu/)
- 2020-06-12 - [Interoperability by construction: code generation for Arrowhead Clients](https://ieeexplore.ieee.org/document/9274746) by Michele Albano, Brian Nielsen at [2020 IEEE Conference on Industrial Cyberphysical Systems (ICPS)](https://ieeexplore.ieee.org/xpl/conhome/9274544/proceeding)
- 2020-06-23 - [新規サーバーアプリケーションにTypeScriptを採用してみた](https://www.cam-inc.co.jp/news/20200623) at [CAM Tech Blog](https://www.cam-inc.co.jp/news/tech-blog/)
- 2020-06-29 - [Artifact Abstract: Deployment of APIs on Android Mobile Devices and Microcontrollers](https://ieeexplore.ieee.org/document/9127353) by [Sergio Laso ; Marino Linaje ; Jose Garcia-Alonso ; Juan M. Murillo ; Javier Berrocal](https://ieeexplore.ieee.org/document/9127353/authors#authors) at [2020 IEEE International Conference on Pervasive Computing and Communications (PerCom)](https://ieeexplore.ieee.org/xpl/conhome/9125449/proceeding)
- 2020-07-07 - [5 Best API Documentation Tools](https://blog.dreamfactory.com/5-best-api-documentation-tools/) by Susanna Bouse at [DreamFactory Blog](https://blog.dreamfactory.com/)
- 2020-07-12 - [Open API 3.0の定義からgolangのサーバコードのスケルトンを作成する](https://qiita.com/professor/items/4cbd04ec084d13057bc2) by [@professor (Qiita Blog)](https://qiita.com/professor)
- 2020-07-20 - [Datadog API client libraries now available for Java and Go](https://www.datadoghq.com/blog/java-go-libraries/) by Jordan Obey at [Datadog Blog](https://www.datadoghq.com/blog)
- 2020-07-23 - [Generate Client SDK for .NET Core using Open Api](https://dev.to/no0law1/generate-client-sdk-for-net-core-using-open-api-2dgh) by [Nuno Reis](https://dev.to/no0law1)
- 2020-07-26 - [Dartのhttp_interceptorライブラリを使うと配列のクエリパラメータが消えてしまう件の応急処置](https://qiita.com/gyamoto/items/eeeff81b6770487319ed) by [@gyamoto](https://qiita.com/gyamoto)
- 2020-08-01 - [Generate Angular ReactiveForms from Swagger/OpenAPI](https://dev.to/martinmcwhorter/generate-angular-reactiveforms-from-swagger-openapi-35h9) by [Martin McWhorter](https://dev.to/martinmcwhorter)
- 2020-08-03 - [Criando Bibliotecas para APIs RESTful com OpenAPI, Swagger Editor e OpenAPI Generator](https://medium.com/@everisBrasil/criando-bibliotecas-para-apis-restful-com-openapi-swagger-editor-e-openapi-generator-75349a6420fd) by [everis Brasil (an NTT DATA Company)](https://medium.com/@everisBrasil)
- 2020-08-19 - [マイクロサービスを連携してみよう](https://thinkit.co.jp/article/17704) by [岡井 裕矢(おかい ゆうや)](https://thinkit.co.jp/author/17588), [泉 勝(いずみ まさる)](https://thinkit.co.jp/author/17705) at [Think IT（シンクイット）](https://thinkit.co.jp/)
- 2020-08-25 - [OpenAPI Generator と TypeScript で型安全にフロントエンド開発をしている話](https://tech.smarthr.jp/entry/2020/08/25/135631) at [SmartHR Tech Blog](https://tech.smarthr.jp/)
- 2020-09-10 - [Introduction to OpenAPI with Instana](https://www.instana.com/blog/introduction-to-openapi-with-instana/) by [Cedric Ziel](https://www.instana.com/blog/author/cedricziel/) at [Instana Blog](https://www.instana.com/blog/)
- 2020-09-17 - [Generate PowerShellSDK using openapi-generator](https://medium.com/@ghufz.learn/generate-powershellsdk-using-openapi-generator-33b700891e33) by [Ghufran Zahidi](https://medium.com/@ghufz.learn)
- 2020-09-24 - [How to automate API code generation (OpenAPI/Swagger) and boost productivity - Tutorial with React Native featuring TypeScript](https://medium.com/@sceleski/how-to-automate-api-code-generation-openapi-swagger-and-boost-productivity-1176a0056d8a) by [Sanjin Celeski](https://medium.com/@sceleski)
- 2020-09-25 - [Generate OpenAPI Angular Client](https://medium.com/@pguso/generate-openapi-angular-client-8c9288e8bbd4) by [Patric](https://medium.com/@pguso)
- 2020-10-24 - [Working with Microsoft Identity - React Native Client](https://www.josephguadagno.net/2020/10/24/working-with-microsoft-identity-react-native-client) by [Joseph Guadagno](https://www.josephguadagno.net/)
- 2020-10-31 - [[B2] OpenAPI Specification으로 타입-세이프하게 API 개발하기: 희망편 VS 절망편](https://www.youtube.com/watch?v=J4JHLESAiFk) by 최태건 at [FEConf 2020](https://2020.feconf.kr/)
- 2020-11-05 - [Automated REST-Api Code Generation: Wie IT-Systeme miteinander sprechen](https://www.massiveart.com/blog/automated-rest-api-code-generation-wie-it-systeme-miteinander-sprechen) by Stefan Rottensteiner at [MASSIVE ART Blog](https://www.massiveart.com/blog)
- 2020-12-01 - [OpenAPI GeneratorでGoのAPIサーバー/クライアントコードを自動生成する](https://qiita.com/saki-engineering/items/b20d8b6074c4da9664a5) by [@saki-engineering](https://qiita.com/saki-engineering)
- 2020-12-04 - [Scaling the Test Coverage of OpenAPI Generator for 30+ Programming Languages](https://www.youtube.com/watch?v=7Lke9dHRqT0) by [William Cheng](https://github.com/wing328) at [Open Source Summit Japan + Automotive Linux Summit 2020](https://events.linuxfoundation.org/archive/2020/open-source-summit-japan/) ([Slides](https://speakerdeck.com/wing328/scaling-the-test-coverage-of-openapi-generator-for-30-plus-programming-languages))
- 2020-12-09 - [プロジェクトにOpenAPI Generatorで自動生成された型付きAPI Clientを導入した話](https://qiita.com/yoshifujiT/items/905c18700ede23f40840) by [@yoshifujiT](https://github.com/yoshifujiT)
- 2020-12-15 - [Next.js + NestJS + GraphQLで変化に追従するフロントエンドへ 〜 ショッピングクーポンの事例紹介](https://techblog.yahoo.co.jp/entry/2020121530052952/) by [小倉 陸](https://github.com/ogugu9) at [Yahoo! JAPAN Tech Blog](https://techblog.yahoo.co.jp/)
- 2021-01-08 - [Hello, New API – Part 1](https://www.nginx.com/blog/hello-new-api-part-1/) by [Jeremy Schulman](https://www.nginx.com/people/jeremy-schulman/) at [Major League Baseball](https://www.mlb.com)
- 2021-01-18 - [「アプリ開発あるある」を疑うことから始まった、API Clientコードの自動生成【デブスト2020】](https://codezine.jp/article/detail/13406?p=2) by [CodeZine編集部](https://codezine.jp/author/1)
- 2021-02-05 - [REST-API-Roundtrip with SpringDoc and OpenAPI Generator](https://blog.viadee.de/en/rest-api-roundtrip) by [Benjamin Klatt](https://twitter.com/benklatt) at [viadee](https://www.viadee.de/en/)
- 2021-02-17 - [REST-API-Roundtrip with SpringDoc and OpenAPI Generator](https://medium.com/nerd-for-tech/rest-api-roundtrip-with-springdoc-and-openapi-generator-30bd27ccf698) by [cloud @viadee](https://cloud-viadee.medium.com/)
- 2021-03-08 - [OpenAPI Generator 工具的躺坑尝试](https://blog.csdn.net/u013019701/article/details/114531975) by [独家雨天](https://blog.csdn.net/u013019701) at [CSDN官方博客](https://blog.csdn.net/)
- 2021-03-16 - [如何基于 Swagger 使用 OpenAPI Generator 生成 JMeter 脚本？](https://cloud.tencent.com/developer/article/1802704) by [高楼Zee](https://cloud.tencent.com/developer/user/5836255) at [腾讯云专栏](https://cloud.tencent.com/developer/column)
- 2021-03-24 - [openapi-generator-cli による TypeScript 型定義](https://zenn.dev/takepepe/articles/openapi-generator-cli-ts) by [Takefumi Yoshii](https://zenn.dev/takepepe)
- 2021-03-28 - [Trying out NestJS part 4: Generate Typescript clients from OpenAPI documents](https://dev.to/arnaudcortisse/trying-out-nestjs-part-4-generate-typescript-clients-from-openapi-documents-28mk) by [Arnaud Cortisse](https://dev.to/arnaudcortisse)
- 2021-03-31 - [Open API Server Implementation Using OpenAPI Generator](https://www.baeldung.com/java-openapi-generator-server) at [Baeldung](https://www.baeldung.com/)
- 2021-03-31 - [使用OpenAPI Generator實現Open API Server](https://www.1ju.org/article/java-openapi-generator-server) at [億聚網](https://www.1ju.org/)
- 2021-04-19 - [Introducing Twilio’s OpenAPI Specification Beta](https://www.twilio.com/blog/introducing-twilio-open-api-specification-beta) by [GARETH PAUL JONES](https://www.twilio.com/blog/author/gpj) at [Twilio Blog](https://www.twilio.com/blog)
- 2021-04-22 - [Leveraging OpenApi strengths in a Micro-Service environment](https://medium.com/unibuddy-technology-blog/leveraging-openapi-strengths-in-a-micro-service-environment-3d7f9e7c26ff) by Nicolas Jellab at [Unibuddy Technology Blog](https://medium.com/unibuddy-technology-blog)
- 2021-04-27 - [From zero to publishing PowerShell API clients in PowerShell Gallery within minutes](https://speakerdeck.com/wing328/from-zero-to-publishing-powershell-api-clients-in-powershell-gallery-within-minutes) by [William Cheng](https://github.com/wing328) at [PowerShell + DevOps Global Summit 2021](https://events.devopscollective.org/event/powershell-devops-global-summit-2021/)
- 2021-05-31 - [FlutterでOpen Api Generator(Swagger)を使う](https://aakira.app/blog/2021/05/flutter-open-api/) by [AAkira](https://twitter.com/_a_akira)
- 2021-06-22 - [Rest API Documentation and Client Generation With OpenAPI](https://dzone.com/articles/rest-api-documentation-and-client-generation-with) by [Prasanth Gullapalli](https://dzone.com/users/1011797/prasanthnath.g@gmail.com.html)
- 2021-07-16 - [銀行事業のサーバーサイド開発について / LINE 京都開発室 エンジニア採用説明会](https://www.youtube.com/watch?v=YrrKQHxLPpQ) by 野田誠人, Robert Mitchell
- 2021-07-19 - [OpenAPI code generation with kotlin](https://sylhare.github.io/2021/07/19/Openapi-swagger-codegen-with-kotlin.html) by [sylhare](https://github.com/sylhare)
- 2021-07-29 - [How To Rewrite a Huge Codebase](https://dzone.com/articles/how-to-rewrite-a-huge-code-base) by [Curtis Poe](https://dzone.com/users/4565446/publiusovidius.html)
- 2021-08-21 - [Generating Client APIs using Swagger Part 1](https://medium.com/@flowsquad/generating-client-apis-using-swagger-part-1-2d46f13f5e92) by [FlowSquad.io](https://medium.com/@flowsquad)
- 2021-09-11 - [Invoking AWS ParallelCluster API](https://docs.aws.amazon.com/parallelcluster/latest/ug/api-reference-v3.html) at [AWS ParallelCluster API official documentation](https://docs.aws.amazon.com/parallelcluster/latest/ug/api-reference-v3.html)
- 2021-09-20 - [OpenAPI Generator - The Babel Fish of the API World](https://www.youtube.com/watch?v=s2zMtwd5klg) by [Cliffano Subagio (Principal Engineer at Shine Solutions)](https://github.com/cliffano) at [Apidays LIVE Australia 2021](https://www.apidays.global/australia2021/)
- 2021-10-02 - [How to Write Fewer Lines of Code with the OpenAPI Generator](https://hackernoon.com/how-to-write-fewer-lines-of-code-with-the-openapi-generator) by [Mikhail Alfa](https://hackernoon.com/u/alphamikle)
- 2021-10-12 - [OpenAPI Generator : 4000 étoiles sur GitHub et des spaghettis](https://www.youtube.com/watch?v=9hEsNBSqTFk) by [Jérémie Bresson](https://github.com/jmini) at [Devoxx FR 2021](https://cfp.devoxx.fr/2021/speaker/jeremie_bresson)
- 2021-10-17 - [Generate a TypeScript HTTP Client From An OpenAPI Spec In DotNET 5](https://richardwillis.info/blog/generate-a-type-script-http-client-from-an-open-api-spec-in-dot-net-5) by [Richard Willis](https://github.com/badsyntax)
- 2021-11-06 - [スタートアップの開発で意識したこと](https://zenn.dev/woo_noo/articles/5cb09f8e2899ae782ad1) by [woo-noo](https://zenn.dev/woo_noo)
- 2021-11-09 - [Effective Software Development using OpenAPI Generator](https://apexlabs.ai/post/effective-software-development-using-openapi-generator) by Ajil Oomme
- 2021-12-07 - [An Introduction to OpenAPI](https://betterprogramming.pub/4-use-cases-of-openapi-which-are-good-to-know-1a041f4ad71e) by [Na'aman Hirschfeld](https://naamanhirschfeld.medium.com/)
- 2022-01-02 - [Towards a secure API client generator for IoT devices](https://arxiv.org/abs/2201.00270) by Anders Aaen Springborg, Martin Kaldahl Andersen, Kaare Holland Hattel, Michele Albano
- 2022-02-02 - [Use OpenApi generator to share your models between Flutter and your backend](https://www.youtube.com/watch?v=kPW7ccu9Yvk) by [Guillaume Bernos](https://feb2022.fluttervikings.com/speakers/guillaume_bernos) at [Flutter Vikings Conference 2022 (Hybrid)](https://feb2022.fluttervikings.com/)
- 2022-03-15 - [OpenAPI Specでハイフン区切りのEnum値をOpenAPI Generatorで出力すると、ハイフン区切りのまま出力される](https://qiita.com/yuji38kwmt/items/824d74d4889055ab37d8) by [yuji38kwmt](https://qiita.com/yuji38kwmt)
- 2022-04-01 - [OpenAPI Generatorのコード生成とSpring Frameworkのカスタムデータバインディングを共存させる](https://techblog.zozo.com/entry/coexistence-of-openapi-and-spring) in [ZOZO Tech Blog](https://techblog.zozo.com/)
- 2022-04-06 - [Effective Software Development using OpenAPI Generator](https://apexlabs.ai/post/openapi-generator) by Ajil Oommen (Senior Flutter Developer)
- 2022-05-13 - [A Path From an API To Client Libraries](https://www.youtube.com/watch?v=XC8oVn_efTw) by [Filip Srnec](https://www.devoxx.co.uk/talk/?id=11211) at Infobip
- 2022-06-01 - [API First, using OpenAPI and Spring Boot](https://medium.com/xgeeks/api-first-using-openapi-and-spring-boot-2602c04bb0d3) by [Micael Estrázulas Vianna](https://estrazulas.medium.com/)
- 2022-06-10 - [Autogenerating Clients with FastAPI and Github Actions](https://www.propelauth.com/post/autogenerating-clients-with-fastapi-and-github-actions) by [Andrew Israel](https://www.propelauth.com/author/andrew)
- 2022-06-12 - [Mustache templates with OpenAPI specs](https://medium.com/geekculture/mustache-templates-with-openapi-specs-f24711c67dec) by [Beppe Catanese](https://github.com/gcatanese)
- 2022-07-01 - [Generate API contract using OpenAPI Generator Maven plugin](https://huongdanjava.com/generate-api-contract-using-openapi-generator-maven-plugin.html) by [Khanh Nguyen](https://huongdanjava.com/)
- 2022-07-22 - [使用OpenAPI Generator Maven plugin开发api优先的java客户端和服务端代码](https://blog.roccoshi.top/2022/java/openapi-generator%E7%9A%84%E4%BD%BF%E7%94%A8/) by [Lincest](https://github.com/Lincest)
- 2022-08-01 - [Tutorial: Etsy Open API v3 (ruby)](https://blog.tjoyal.dev/etsy-open-api-v3/) by [Thierry Joyal](https://github.com/tjoyal)
- 2022-09-03 - [OpenAPI Generator For Go Web Development](https://blog.kevinhu.me/2022/09/03/03-openapi-generator/) by [Kevin Hu](https://twitter.com/Oldgunix)
- 2022-10-01 - [OpenAPI Generatorをカスタマイズしたコードを生成する（Swagger Codegenとほぼ同じ）](https://nainaistar.hatenablog.com/entry/2022/10/03/120000) by [きり丸](https://twitter.com/nainaistar)
- 2022-10-21 - [Kotlin（Spring Boot）の API を OpenAPI Generator で自動生成](https://zenn.dev/msksgm/articles/20221021-kotlin-spring-openapi-generator) by [msksgm](https://zenn.dev/msksgm)
- 2022-10-26 - [Quarkus Insights #106: Quarkiverse Extension Spotlight: OpenApi Generator](https://www.youtube.com/watch?v=_s_if69t2iQ) by [Quarkusio](https://www.youtube.com/c/Quarkusio)
- 2022-11-28 - [The REST API implementation flow](https://tmsvr.com/openapi-code-generation-for-rest-apis/) by [Imre Tömösvári](https://tmsvr.com/author/imre/)
- 2022-12-13 - [API-First with Spring WebFlux and OpenAPI Generator](https://boottechnologies-ci.medium.com/api-first-with-spring-webflux-and-openapi-generator-38b7804c4ed4) by [Eric Anicet](https://boottechnologies-ci.medium.com/)
- 2023-01-06 - [Major Improvements with Helidon and OpenAPI](https://medium.com/helidon/major-improvements-with-helidon-and-openapi-f76a0951508e) by [Tim Quinn](https://medium.com/@tquinno600)
- 2023-02-02 - [Replacing Postman with the Jetbrains HTTP Client](https://lengrand.fr/replacing-postman-in-seconds-with-the-jetbrains-http-client/) by [julien Lengrand-Lambert](https://github.com/jlengrand)
- 2023-03-15 - [OpenAPI Generatorに適したOpenAPIの書き方](https://techblog.zozo.com/entry/how-to-write-openapi-for-openapi-generator) by [ZOZO Tech Blog](https://techblog.zozo.com/)
- 2023-03-19 - [EXOGEM: Extending OpenAPI Generator for Monitoring of RESTful APIs](https://link.springer.com/chapter/10.1007/978-3-031-26507-5_10) by Daniel Friis Holtebo, Jannik Lucas Sommer, Magnus Mølgaard Lund, Alessandro Tibo, Junior Dongo & Michele Albano at "ICSOC 2022: Service-Oriented Computing – ICSOC 2022 Workshops"
- 2023-03-28 - [API-First Design with OpenAPI Generator](https://www.linkedin.com/pulse/api-first-design-openapi-generator-jonathan-manera/) by [Jonathan Manera](https://www.linkedin.com/in/manerajona/)
- 2023-03-28 - [ハンズオンで学ぶサーバーサイド Kotlin（Spring Boot&Arrow&OpenAPI Generator）v1.0.1](https://zenn.dev/msksgm/books/implementing-server-side-kotlin-development) by [msk](https://zenn.dev/msksgm)
- 2023-04-01 - [OpenAPI Client Code Generation](https://testingboss.com/blog/openapi-client-generation/) by Kwo Ding
- 2023-04-27 - [Create an Angular Client using OpenAPI Specifications](Create an Angular Client using OpenAPI Specifications) by [Patric](https://pguso.medium.com/)
- 2023-05-16 - [Adyen for Java developers](https://www.adyen.com/blog/adyen-java-library) by [Beppe Catanese, Developer Advocate, Adyen](https://github.com/gcatanese)
- 2023-05-18 - [如何基于 Swagger 使用 OpenAPI Generator 生成 JMeter 脚本？](https://blog.51cto.com/u_15181572/6294974) by [高楼（Zee)](https://blog.51cto.com/u_15181572)
- 2023-06-28 - [Generate API contract using OpenAPI Generator Maven plugin](https://huongdanjava.com/generate-api-contract-using-openapi-generator-maven-plugin.html) by [Khanh Nguyen](https://huongdanjava.com/)
- 2023-06-30 - [Generate Client SDKs with OpenApi Generator in Springboot](https://medium.com/@ramavathvinayak/generate-client-sdks-with-openapi-generator-in-springboot-f9f012e73c0b) by [Vinayak Ramavath](https://medium.com/@ramavathvinayak)
- 2023-12-10 - [UnityでOpenAPI Generatorを使う](https://www.youtube.com/watch?v=CbNwKVV5LRM) by [Soup Tori](https://www.youtube.com/@souptori8417)
- 2024-01-24 - [Comment générer des stubs wiremock avec openapi generator](https://www.youtube.com/watch?v=0jhONfBrcKw) by [Alexis Couvreur](https://github.com/acouvreur)
- 2024-03-04 - [Generating TypeScript Types with OpenAPI for REST API Consumption](https://www.pullrequest.com/blog/generating-typescript-types-with-openapi-for-rest-api-consumption/) by [PullRequest](https://www.pullrequest.com/)
- 2024-03-07 - [Fully typed Web Apps with OpenAPI (Part 1)](https://medium.com/@gfox1984/fully-typed-web-apps-with-openapi-part-1-595d55766670) by [Guillaume Renard](https://medium.com/@gfox1984)
- 2024-03-08 - [Laravel OpenAPIによる "辛くない" スキーマ駆動開発](https://fortee.jp/phperkaigi-2024/proposal/9e2e6c38-d078-4efa-99b4-83ebf9033b34) by [KentarouTakeda](https://twitter.com/KentarouTakeda)
- 2024-04-04 - [Working with OpenAPI using Rust](https://www.shuttle.dev/blog/2024/04/04/using-openapi-rust) by [Joshua Mo](https://twitter.com/joshmo_dev)
- 2024-04-08 - [Implement API first strategy with OpenAPI generator plugin](https://medium.com/javarevisited/implement-api-first-strategy-with-openapi-generator-plugin-e4bbe7f0d778) by [Rui Zhou](https://medium.com/@wirelesser)
- 2024-05-06 - [OpenAPI Generator Custom Templates](https://www.javacodegeeks.com/openapi-generator-custom-templates.html) by [Mary Zheng](https://www.javacodegeeks.com/author/mary-zheng)
- 2025-02-09 - [Custom validation with OpenApiGenerator and Spring Boot 3](https://medium.com/@jugurtha.aitoufella/custom-validation-with-openapigenerator-and-spring-boot-3-34a656e815c8) by [Jugurtha Aitoufella](https://medium.com/@jugurtha.aitoufella)
- 2025-02-20 - [Optimizing API Integration in a Large React Application Using OpenAPI Generator](https://www.youtube.com/watch?v=-B33pQnGQUI) by Stefano Marzo


## [6 - About Us](#table-of-contents)

What's the design philosophy or principle behind OpenAPI Generator?

We focus on developer experience. The generators should produce code, config, documentation, and more that are easily understandable and consumable by users. We focused on simple use cases to start with (bottom-up approach). Since then the project and the community have grown a lot: 600k weekly downloads via NPM CLI wrapper, 30M downloads via openapi-generator-cli docker image just to highlight a few. We've gradually supported more features (e.g. oneOf, anyOf introduced in OpenAPI 3.0) in various generators and we will continue this approach to deliver something based on our understanding of user demand and what they want, and continue to add support of new features introduced in OpenAPI specification (such as v3.1 and future versions of the OpenAPI specification).

### [6.1 - OpenAPI Generator Core Team](#table-of-contents)

OpenAPI Generator core team members are contributors who have been making significant contributions (review issues, fix bugs, make enhancements, etc) to the project on a regular basis.

#### Core Team Members
* [@wing328](https://github.com/wing328) (2015/07) [:heart:](https://www.patreon.com/wing328)
* [@jimschubert](https://github.com/jimschubert) (2016/05) [:heart:](https://www.patreon.com/jimschubert)
* [@cbornet](https://github.com/cbornet) (2016/05)
* [@jmini](https://github.com/jmini) (2018/04)  [:heart:](https://www.patreon.com/jmini)
* [@etherealjoy](https://github.com/etherealjoy) (2019/06)

:heart: = Link to support the contributor directly

#### Template Creator

**NOTE**: Embedded templates are only supported in _Mustache_ format. Support for all other formats is experimental and subject to change at any time.

Here is a list of template creators:
 * API Clients:
   * Ada: @stcarrez
   * Apex: @asnelling
   * Bash: @bkryza
   * C: @PowerOfCreation @zhemant [:heart:](https://www.patreon.com/zhemant)
   * C++ Oat++: @Kraust
   * C++ REST: @Danielku15
   * C++ Tiny: @AndersSpringborg @kaareHH @michelealbano @mkakbas
   * C++ UE4: @Kahncode
   * C# (.NET 2.0): @who
   * C# (.NET Standard 1.3 ): @Gronsak
   * C# (.NET 4.5 refactored): @jimschubert [:heart:](https://www.patreon.com/jimschubert)
   * C# (GenericHost): @devhl-labs
   * C# (HttpClient): @Blackclaws
   * Clojure: @xhh
   * Crystal: @wing328
   * Dart: @yissachar
   * Dart (refactor): @joernahrens
   * Dart 2: @swipesight
   * Dart (Jaguar): @jaumard
   * Dart (Dio): @josh-burton
   * Elixir: @niku
   * Elm: @eriktim
   * Eiffel: @jvelilla
   * Erlang: @tsloughter
   * Erlang (PropEr): @jfacorro @robertoaloi
   * Groovy: @victorgit
   * Go: @wing328 [:heart:](https://www.patreon.com/wing328)
   * Go (rewritten in 2.3.0): @antihax
   * Godot (GDScript): @Goutte [:heart:](https://liberapay.com/Goutte)
   * Haskell (http-client): @jonschoning
   * Java (Feign): @davidkiss
   * Java (Retrofit): @0legg
   * Java (Retrofit2): @emilianobonassi
   * Java (Jersey2): @xhh
   * Java (okhttp-gson): @xhh
   * Java (RestTemplate): @nbruno
   * Java (Spring 5 WebClient): @daonomic
   * Java (Spring 6 RestClient): @nicklas2751
   * Java (RESTEasy): @gayathrigs
   * Java (Vertx): @lopesmcc
   * Java (Google APIs Client Library): @charlescapps
   * Java (Rest-assured): @viclovsky
   * Java (Java 11 Native HTTP client): @bbdouglas
   * Java (Apache HttpClient 5.x): @harrywhite4 @andrevegas
   * Java (Helidon): @spericas @tjquinno @tvallin
   * Javascript/NodeJS: @jfiala
   * JavaScript (Apollo DataSource): @erithmetic
   * JavaScript (Closure-annotated Angular) @achew22
   * JavaScript (Flow types) @jaypea
   * Jetbrains HTTP Client : @jlengrand
   * JMeter: @davidkiss
   * Julia: @tanmaykm
   * Kotlin: @jimschubert [:heart:](https://www.patreon.com/jimschubert)
   * Kotlin (MultiPlatform): @andrewemery
   * Kotlin (Volley): @alisters
   * Kotlin (jvm-spring-webclient): @stefankoppier
   * Kotlin (jvm-spring-restclient): @stefankoppier
   * Lua: @daurnimator
   * N4JS: @mmews-n4
   * Nim: @hokamoto
   * OCaml: @cgensoul
   * Perl: @wing328 [:heart:](https://www.patreon.com/wing328)
   * PHP (Guzzle): @baartosz
   * PHP (with Data Transfer): @Articus
   * PowerShell: @beatcracker
   * PowerShell (refactored in 5.0.0): @wing328
   * Python: @spacether [:heart:][spacether sponsorship]
   * Python-Experimental: @spacether [:heart:][spacether sponsorship]
   * Python (refactored in 7.0.0): @wing328
   * R: @ramnov
   * Ruby (Faraday): @meganemura @dkliban
   * Ruby (HTTPX): @honeyryderchuck
   * Rust: @farcaller
   * Rust (rust-server): @metaswitch
   * Scala (scalaz & http4s): @tbrown1979
   * Scala (Akka): @cchafer
   * Scala (sttp): @chameleon82
   * Scala (sttp4): @flsh86
   * Scala (scala-sttp4-jsoniter): @lbialy
   * Scala (Pekko): @mickaelmagniez
   * Scala (http4s): @JennyLeahy
   * Swift: @tkqubo
   * Swift 3: @hexelon
   * Swift 4: @ehyche
   * Swift 5: @4brunu
   * Swift 6: @4brunu
   * Swift Combine: @dydus0x14
   * TypeScript (Angular1): @mhardorf
   * TypeScript (Angular2): @roni-frantchi
   * TypeScript (Angular6): @akehir
   * TypeScript (Angular7): @topce
   * TypeScript (Axios): @nicokoenig
   * TypeScript (Fetch): @leonyu
   * TypeScript (Inversify): @gualtierim
   * TypeScript (jQuery): @bherila
   * TypeScript (Nestjs): @vfrank66
   * TypeScript (Node):  @mhardorf
   * TypeScript (Rxjs): @denyo
   * TypeScript (redux-query): @petejohansonxo
   * Xojo: @Topheee
   * Zapier: @valmoz, @emajo
 * Server Stubs
   * Ada: @stcarrez
   * C# ASP.NET 5: @jimschubert [:heart:](https://www.patreon.com/jimschubert)
   * C# ASP.NET Core 3.0: @A-Joshi
   * C# APS.NET Core 3.1: @phatcher
   * C# Azure functions: @Abrhm7786
   * C# NancyFX: @mstefaniuk
   * C++ (Qt5 QHttpEngine): @etherealjoy
   * C++ Oat++: @Kraust
   * C++ Pistache: @sebymiano
   * C++ Restbed: @stkrwork
   * Erlang Server: @galaxie @nelsonvides
   * F# (Giraffe) Server: @nmfisher
   * Go Server: @guohuang
   * Go Server (refactored in 7.0.0): @lwj5
   * Go (Echo) Server: @ph4r5h4d
   * Go (Gin) Server: @kemokemo
   * GraphQL Express Server: @renepardon
   * Haskell Servant: @algas
   * Haskell Yesod: @yotsuya
   * Java Camel: @carnevalegiacomo
   * Java Dubbo: @redoom
   * Java MSF4J: @sanjeewa-malalgoda
   * Java Spring Boot: @diyfr
   * Java Undertow: @stevehu
   * Java Play Framework: @JFCote
   * Java PKMST: @anshu2185 @sanshuman @rkumar-pk @ninodpillai
   * Java Vert.x: @lwlee2608
   * Java Micronaut: @andriy-dmytruk
   * Java Helidon: @spericas @tjquinno @tvallin
   * Java WireMock: [@acouvreur](https://github.com/acouvreur)
   * JAX-RS RestEasy: @chameleon82
   * JAX-RS CXF: @hiveship
   * JAX-RS CXF (CDI): @nickcmaynard
   * JAX-RS RestEasy (JBoss EAP): @jfiala
   * Julia: @tanmaykm
   * Kotlin: @jimschubert [:heart:](https://www.patreon.com/jimschubert)
   * Kotlin (Spring Boot): @dr4ke616
   * Kotlin (Vertx): @Wooyme
   * Kotlin (JAX-RS): @anttileppa
   * Kotlin Misk: @andrewwilsonnew @guiarn
   * Kotlin WireMock: @stefankoppier
   * NodeJS Express: @YishTish
   * PHP Flight: @daniel-sc
   * PHP Laravel: @renepardon
   * PHP Laravel (refactor in 7.12.0): @gijs-blanken
   * PHP Lumen: @abcsun
   * PHP Mezzio (with Path Handler): @Articus
   * PHP Slim: @jfastnacht
   * PHP Slim4: [@ybelenko](https://github.com/ybelenko)
   * PHP Symfony: @ksm2
   * PHP Symfony6: @BenjaminHae
   * Python FastAPI: @krjakbrjak
   * Python AIOHTTP:
   * Ruby on Rails 5: @zlx
   * Rust (rust-server): @metaswitch
   * Rust (rust-axum): @linxGnu
   * Scala Akka: @Bouillie
   * Scala Cask: @aaronp
   * Scala Finch: @jimschubert [:heart:](https://www.patreon.com/jimschubert)
   * Scala Lagom: @gmkumar2005
   * Scala Play: @adigerber
   * TypeScript NestJS: @aryobenholzner
 * Documentation
   * AsciiDoc: @man-at-home
   * HTML Doc 2: @jhitchcock
   * Confluence Wiki: @jhitchcock
   * PlantUML: @pburls
 * Configuration
   * Apache2: @stkrwork
   * k6: @mostafa
 * Schema
   * Avro: @sgadouar
   * GraphQL: @wing328 [:heart:](https://www.patreon.com/wing328)
   * Ktorm: @Luiz-Monad
   * MySQL: [@ybelenko](https://github.com/ybelenko)
   * PostgreSQL: [@iri](https://github.com/iri)
   * Postman Collection: @gcatanese
   * Protocol Buffer: @wing328
   * WSDL: @adessoDpd

:heart: = Link to support the contributor directly

#### How to join the core team

Here are the requirements to become a core team member:
- rank within top 50 in https://github.com/openapitools/openapi-generator/graphs/contributors
  - to contribute, here are some good [starting points](https://github.com/openapitools/openapi-generator/issues?q=is%3Aopen+is%3Aissue+label%3A%22help+wanted%22)
- regular contributions to the project
  - about 3 hours per week
  - for contribution, it can be addressing issues, reviewing PRs submitted by others, submitting PR to fix bugs or make enhancements, etc
  - must be active in the past 3 months at the time of application

 To join the core team, please reach out to team@openapitools.org for more information.

 To become a Template Creator, simply submit a PR for new API client (e.g. Rust, Elixir) or server stub (e.g. Ruby Grape) generator.

### [6.2 - OpenAPI Generator Technical Committee](#table-of-contents)

Members of the OpenAPI Generator technical committee shoulder the following responsibilities:

- Provides guidance and direction to other users
- Reviews pull requests and issues
- Improves the generator by making enhancements, fixing bugs or updating documentations
- Sets the technical direction of the generator

Who is eligible? Those who want to join must have at least 3 PRs merged into a generator. (Exceptions can be granted to template creators or contributors who have made a lot of code changes with less than 3 merged PRs)

If you want to join the committee, please kindly apply by sending an email to team@openapitools.org with your Github ID.

#### Members of Technical Committee

| Languages/Generators  | Member (join date)                                                                                                                                                                                                                                    |
|:----------------------|:------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| ActionScript          |                                                                                                                                                                                                                                                       |
| Ada                   | @stcarrez (2018/02) @michelealbano (2018/02)                                                                                                                                                                                                          |
| Android               | @jaz-ah (2017/09)                                                                                                                                                                                                                                     |
| Apex                  |                                                                                                                                                                                                                                                       |
| Bash                  | @frol (2017/07) @bkryza (2017/08) @kenjones-cisco (2017/09)                                                                                                                                                                                           |
| C                     | @zhemant (2018/11) @ityuhui (2019/12) @michelealbano (2020/03) @eafer (2024/12)                                                                                                                                                                                        |
| C++                   | @ravinikam (2017/07) @stkrwork (2017/07) @etherealjoy (2018/02) @martindelille (2018/03) @muttleyxd (2019/08) @aminya (2025/05)                                                                                                                                         |
| C#                    | @mandrean (2017/08) @shibayan (2020/02) @Blackclaws (2021/03) @lucamazzanti (2021/05) @iBicha (2023/07)                                                                                                                                          |
| Clojure               |                                                                                                                                                                                                                                                       |
| Crystal               | @cyangle (2021/01)                                                                                                                                                                                                                                    |
| Dart                  | @jaumard (2018/09) @josh-burton (2019/12) @amondnet (2019/12) @sbu-WBT (2020/12) @kuhnroyal (2020/12) @agilob (2020/12) @ahmednfwela (2021/08)                                                                                                        |
| Eiffel                | @jvelilla (2017/09)                                                                                                                                                                                                                                   |
| Elixir                | @mrmstn (2018/12)                                                                                                                                                                                                                                     |
| Elm                   | @eriktim (2018/09)                                                                                                                                                                                                                                    |
| Erlang                | @tsloughter (2017/11) @jfacorro (2018/10) @robertoaloi (2018/10) @nelsonvides (2024/09)                                                                                                                                                               |
| F#                    | @nmfisher (2019/05)                                                                                                                                                                                                                                   |
| Go                    | @antihax (2017/11) @grokify (2018/07) @kemokemo (2018/09) @jirikuncar (2021/01) @ph4r5h4d (2021/04) @lwj5 (2023/04)                                                                                                                                                   |
| GraphQL               | @renepardon (2018/12)                                                                                                                                                                                                                                 |
| Groovy                |                                                                                                                                                                                                                                                       |
| Haskell               |                                                                                                                                                                                                                                                       |
| Java                  | @bbdouglas (2017/07) @sreeshas (2017/08) @jfiala (2017/08) @lukoyanov (2017/09) @cbornet (2017/09) @jeff9finger (2018/01) @karismann (2019/03) @Zomzog (2019/04) @lwlee2608 (2019/10) @martin-mfg (2023/08)                                                                 |
| Java Spring           | @cachescrubber (2022/02) @welshm (2022/02) @MelleD (2022/02) @atextor (2022/02) @manedev79 (2022/02) @javisst (2022/02) @borsch (2022/02) @banlevente (2022/02) @Zomzog (2022/09) @martin-mfg (2023/08)                                                                     |
| JMeter                | @kannkyo (2021/01)                                                                                                                                                                                                                                    |
| Jetbrains HTTP Client | @jlengrand (2023/01)                                                                                                                                                                                                                                  |
| Julia                 | @tanmaykm (2023/01)                                                                                                                                                                                                                                   |
| Kotlin                | @karismann (2019/03) @Zomzog (2019/04) @andrewemery (2019/10) @4brunu (2019/11) @yutaka0m (2020/03) @stefankoppier (2022/06) @e5l (2024/10) @dennisameling (2026/02)                                         |
| Lua                   | @daurnimator (2017/08)                                                                                                                                                                                                                                |
| N4JS                  | @mmews-n4 (2023/03)                                                                                                                                                                                      |
| Nim                   |                                                                                                                                                                                                                                                       |
| NodeJS/Javascript     | @CodeNinjai (2017/07) @frol (2017/07) @cliffano (2017/07)                                                                                                                                                                                             |
| ObjC                  |                                                                                                                                                                                                                                                       |
| OCaml                 | @cgensoul (2019/08), @sir4ur0n (2025/08)                                                                                                                                                                                                              |
| Perl                  | @wing328 (2017/07) [:heart:](https://www.patreon.com/wing328) @yue9944882 (2019/06)                                                                                                                                                                   |
| PHP                   | @jebentier (2017/07), @dkarlovi (2017/07), @mandrean (2017/08), @jfastnacht (2017/09), [@ybelenko](https://github.com/ybelenko) (2018/07), @renepardon (2018/12)                                                                                      |
| PowerShell            | @wing328 (2020/05)                                                                                                                                                                                                                                    |
| Python                | @cbornet (2017/09) @tomplus (2018/10) @krjakbrjak (2023/02) @fa0311 (2023/10) @multani (2023/10) |
| R                     | @Ramanth (2019/07) @saigiridhar21 (2019/07)                                                                                                                                                                                                           |
| Ruby                  | @cliffano (2017/07) @zlx (2017/09) @autopp (2019/02)                                                                                                                                                                                                  |
| Rust                  | @frol (2017/07) @farcaller (2017/08) @richardwhiuk (2019/07) @paladinzh (2020/05) @jacob-pro (2022/10) @dsteeley (2025/07)                                                                                                                                               |
| Scala                 | @clasnake (2017/07), @shijinkui  (2018/01), @ramzimaalej (2018/03), @chameleon82 (2020/03), @Bouillie (2020/04) @fish86 (2023/06)                                                               |
| Swift                 | @jgavris (2017/07) @ehyche (2017/08) @Edubits (2017/09) @jaz-ah (2017/09) @4brunu (2019/11) @dydus0x14 (2023/06)                                                                                                                                                           |
| TypeScript            | @TiFu (2017/07) @taxpon (2017/07) @sebastianhaas (2017/07) @kenisteward (2017/07) @Vrolijkx (2017/09) @macjohnny (2018/01) @topce (2018/10) @akehir (2019/07) @petejohansonxo (2019/11) @amakhrov (2020/02) @davidgamero (2022/03) @mkusaka (2022/04) @joscha (2024/10) @dennisameling (2026/02)   |
| Xojo                  | @Topheee (2023/04)                                                                                                                                                                                                                                    |


Past Members of Technical Committee:
| Languages/Generators         | Member (join date)                                                                                                                                                                                                                |
| :---------------- | :-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Python            | @taxpon (2017/07) @frol (2017/07) @mbohlool (2017/07) @cbornet (2017/09) @kenjones-cisco (2017/11) @tomplus (2018/10) @arun-nalla (2019/11)  |


:heart: = Link to support the contributor directly

### [6.3 - History of OpenAPI Generator](#table-of-contents)

OpenAPI Generator is a fork of [Swagger Codegen](https://github.com/swagger-api/swagger-codegen). In view of the issues with the Swagger Codegen 3.0.0 (beta) release and the disagreement on the project's direction, more than 40 top contributors and template creators of Swagger Codegen decided to fork Swagger Codegen and maintain a community-driven version called "OpenAPI Generator". Please refer to the [Q&A](docs/qna.md) for more information.

#### Founding Members (alphabetical order):

- [Akihito Nakano](https://github.com/ackintosh)
- [Artem Ocheredko](https://github.com/galaxie)
- [Arthur Mogliev](https://github.com/Articus)
- [Bartek Kryza](https://github.com/bkryza)
- [Ben Wells](https://github.com/bvwells)
- [Benjamin Gill](https://github.com/bjgill)
- [Christophe Bornet](https://github.com/cbornet)
- [Cliffano Subagio](https://github.com/cliffano)
- [Daiki Matsudate](https://github.com/d-date)
- [Daniel](https://github.com/Danielku15)
- [Emiliano Bonassi](https://github.com/emilianobonassi)
- [Erik Timmers](https://github.com/eriktim)
- [Esteban Gehring](https://github.com/macjohnny)
- [Gustavo Paz](https://github.com/gustavoapaz)
- [Javier Velilla](https://github.com/jvelilla)
- [Jean-François Côté](https://github.com/JFCote)
- [Jim Schubert](https://github.com/jimschubert)
- [Jon Schoning](https://github.com/jonschoning)
- [Jérémie Bresson](https://github.com/jmini) [:heart:](https://www.patreon.com/jmini)
- [Jörn Ahrens](https://github.com/jayearn)
- [Keni Steward](https://github.com/kenisteward)
- [Marcin Stefaniuk](https://github.com/mstefaniuk)
- [Martin Delille](https://github.com/MartinDelille)
- [Masahiro Yamauchi](https://github.com/algas)
- [Michele Albano](https://github.com/michelealbano)
- [Ramzi Maalej](https://github.com/ramzimaalej)
- [Ravindra Nikam](https://github.com/ravinikam)
- [Ricardo Cardona](https://github.com/ricardona)
- [Sebastian Haas](https://github.com/sebastianhaas)
- [Sebastian Mandrean](https://github.com/mandrean)
- [Sreenidhi Sreesha](https://github.com/sreeshas)
- [Stefan Krismann](https://github.com/stkrwork)
- [Stephane Carrez](https://github.com/stcarrez)
- [Takuro Wada](https://github.com/taxpon)
- [Tomasz Prus](https://github.com/tomplus)
- [Tristan Sloughter](https://github.com/tsloughter)
- [Victor Orlovsky](https://github.com/viclovsky)
- [Victor Trakhtenberg](https://github.com/victorgit)
- [Vlad Frolov](https://github.com/frol)
- [Vladimir Pouzanov](https://github.com/farcaller)
- [William Cheng](https://github.com/wing328)
- [Xin Meng](https://github.com/xmeng1) [:heart:](https://www.patreon.com/user/overview?u=16435385)
- [Xu Hui Hui](https://github.com/xhh)
- [antihax](https://github.com/antihax)
- [beatcracker](https://github.com/beatcracker)
- [daurnimator](https:/github.com/daurnimator)
- [etherealjoy](https://github.com/etherealjoy)
- [jfiala](https://github.com/jfiala)
- [lukoyanov](https://github.com/lukoyanov)

:heart: = Link to support the contributor directly

## [7 - License](#table-of-contents)
-------

Copyright 2018 OpenAPI-Generator Contributors (https://openapi-generator.tech)
Copyright 2018 SmartBear Software

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at [apache.org/licenses/LICENSE-2.0](https://www.apache.org/licenses/LICENSE-2.0)

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

---
 readmeEtag: '"1cf6874abaf9f652f741e5b9b8261417196ff2f6"' readmeLastModified: Wed, 04 Feb 2026 11:44:39 GMT repositoryId: 133134007 description: >- OpenAPI Generator allows generation of API client libraries (SDK generation), server stubs, documentation and configuration automatically given an OpenAPI Spec (v2, v3) created: '2018-05-12T09:57:56Z' updated: '2026-02-06T03:54:45Z' language: Java archived: false stars: 25755 watchers: 223 forks: 7376 owner: OpenAPITools logo: https://avatars.githubusercontent.com/u/37325267?v=4 license: Apache-2.0 repoEtag: '"7971ffac372c800290e8ce0e18b2b37b93e2f205f5479ef8de9b3d178c339644"' repoLastModified: Fri, 06 Feb 2026 03:54:45 GMT foundInMaster: true id: 0418f8afcd0196c648d1a46c0c5c20d1 v3_1_link: https://github.com/OpenAPITools/openapi-generator/issues/9083 - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/openapi-json-schema-tools/openapi-json-schema-generator v3: true id: b03b9d5fc0aef95c46d5d148a05798ce repositoryMetadata: base64Readme: >- # OpenAPI JSON Schema Generator

[![CI Tests](https://dl.circleci.com/status-badge/img/gh/openapi-json-schema-tools/openapi-json-schema-generator/tree/master.svg?style=shield)](https://dl.circleci.com/status-badge/redirect/gh/openapi-json-schema-tools/openapi-json-schema-generator/tree/master)
[![Apache 2.0 License](https://img.shields.io/badge/License-Apache%202.0-orange)](./LICENSE)

Auto generate a client sdk from your openapi 3.0.0-3.1.0 document using Openapi JSON Schema Generator. 
This project is a code generator that focuses on supporting all openapi and json schema features.

## Overview
OpenAPI JSON Schema Generator allows auto-generation of API client libraries (SDK generation) given an
[OpenAPI document](https://github.com/OAI/OpenAPI-Specification) (3.0.0-[3.1.0*](#openapi-v310-support) are supported).
This project focuses on making the output 100% compliant with openapi + JSON schema specs.
The goal is to fully support everything defined in openapi + the included JSON schema specs
so developers can use all of those features.

Currently, the following languages/frameworks are supported:

| Feature                                                                                                              | [Python](docs/generators/python.md) | [Java](docs/generators/java.md) | [Kotlin](docs/generators/kotlin.md) |
|----------------------------------------------------------------------------------------------------------------------|-------------------------------------|---------------------------------|-------------------------------------|
| Generator status                                                                                                     | stable                              | stable                          | stable                                   |
| Openapi v3.0.0-3.1.0 ingestion                                                                                       | X                                   | X                               | X                                   |
| Json Schema 2020-12 Support (components/schemas)                                                                     | X                                   | X                               | X                                   |
| Component schema classes + documentation produced                                                                    | X                                   | X                               | X                                   |
| Documentation produced for other component types:<br>headers, parameters, requestBodies, ressponses, securitySchemes | X                                   | X                               |                                     |
| Methods generated for endpoints <br>that send/receive json + docs generated for them                                 | X                                   | X                               |                                     |

## Reasons To Use the Generators
- Openapi spec support for v3.0.0-3.1.0
  - thorough tests run in CI using json schema test suite, see 3_0_0 and 3_1_0 sample clients
- Static analysis:
  - Python: mypy run in CI against python petstore sample
  - Java: checker framework run w/ NullnessChecker, ensures no null pointer exceptions
- Format support for: int32, int64, float, double, binary, date, datetime, uuid
- Invalid (in language) property names supported like `from`, `1var`, `hi-there` etc in
  - schema property names
  - endpoint parameter names
- Openapi document inline schemas supported at any depth in any location
- Generated Code: Class + method inputs are typed
- Generated Code: Static type checking done in static languages using builder inputs and class property access
- Generated Code: run-time type checking done in all generators (a payload can be validated against n schemas)
- Generated Code re-use built in from the ground up
  - components/schemas/headers etc are generated as separate classes and imported when used via $ref
- Payload values are not coerced when validated, so a date/date-time value can pass other validations that describe the payload only as type string
- String transmission of numbers supported with type: string, format: number, value can be accessed as a Decimal with schemas.as_decimal(inst)
- Multiple content types supported for request and response bodies
- Endpoint response always also includes the raw response
- Interfaces kept consistent across generated languages

## Join Our Community
We use a Discord server as a place to ask questions and help each other. It offers functionality very similar to Slack.
You can join us here: https://discord.gg/mHB8WEQuYQ

### Can I build here?

Yes; contributions are welcome!
Submit a PR if you want to add a new server scaffold, client sdk, or documentation generator in any language.

## Table of contents

  - [OpenAPI JSON Schema Generator](#openapi-json-schema-generator)
  - [Overview](#overview)
  - [Table of Contents](#table-of-contents)
  - [Installation](#installation)
    - [Compatibility](#compatibility)
    - [Build Projects](#build-projects)
    - [Docker](#docker)
  - [Getting Started](#getting-started)
  - [Usage](#usage)
    - [Customization](#customization)
    - [Workflow Integration](#workflow-integration)
    - [License Information](#license-information)
  - [Companies/Projects using OpenAPI JSON Schema Generator](#companiesprojects-using-openapi-json-schema-generator)
  - [About Us](#about-us)
    - [History of OpenAPI JSON Schema Generator](#history-of-openapi-json-schema-generator)
  - [License](#license)

## Installation

### Compatibility

The OpenAPI Specification has undergone 3 revisions since initial creation in 2010.  The openapi-json-schema-generator project has the following compatibilities with the OpenAPI Specification:

| OpenAPI JSON Schema Generator Version | OpenAPI Spec compatibility                   |
|---------------------------------------|----------------------------------------------|
| 3.3.0+                                | 3.0.0 - [3.1.0*](#openapi-v310-support) |
| 3.1.0 - 3.2.1                         | 3.0.0 - [3.1.0](https://github.com/openapi-json-schema-tools/openapi-json-schema-generator/blob/3.1.0/docs/generators/python.md#schema-feature) |
| 1.0.0 - 3.0.0                         | 3.0.0 - 3.0.3                                |

#### OpenAPI v3.1.0 support
OpenAPI v3.1.0 specification support includes these new/updated 2020-12 json schema keywords:
1. const: only string values are working because of bugs in swagger parser
2. contains
3. dependentRequired
4. dependentSchemas
5. else
6. if
7. maxContains
8. minContains
9. patternProperties
10. prefixItems
11. propertyNames
12. then
13. type (array of types supported in addition to one non-array value)
14. unevaluatedItems
15. unevaluatedProperties

Note: these features can also be seen in the generator documentation [schema features](docs/generators/python.md#schema-feature)

### Build Projects

To build from source, you need the following installed and available in your `$PATH:`

* [Java 11](https://www.oracle.com/technetwork/java/index.html)

* [Apache Maven 3.9.3 or greater](https://maven.apache.org/)

After cloning the project, you can build it from source with this command:
```sh
mvn clean install
```

The default build contains minimal static analysis (via CheckStyle). To run your build with PMD and Spotbugs, use the `static-analysis` profile:

```sh
mvn -Pstatic-analysis clean install
```

### Docker

#### Public Pre-built Docker images

 - [https://hub.docker.com/r/openapijsonschematools/openapi-json-schema-generator-cli/](https://hub.docker.com/r/openapijsonschematools/openapi-json-schema-generator-cli/) (official CLI)

#### OpenAPI JSON Schema Generator CLI Docker Image

The docker image acts as a standalone executable. It can be used as an alternative to installing via homebrew, or for developers who are unable to install Java or upgrade the installed version.

To generate code with this image, you'll need to mount a local location as a volume.

Example:

```sh
docker run --rm -v "${PWD}:/local" openapijsonschematools/openapi-json-schema-generator-cli generate \
    -i https://raw.githubusercontent.com/openapi-json-schema-tools/openapi-json-schema-generator/master/src/test/resources/3_0/petstore.yaml \
    -g python \
    -o /local/out/python
```

The generated code will be located under `./out/python` in the current directory.

#### Development in docker

You can use `bin/run-in-docker.sh` to do all development. This script maps your local repository to `/gen`
in the docker container. It also maps `~/.m2/repository` to the appropriate container location.

To execute `mvn package`:

```sh
git clone https://github.com/openapi-json-schema-tools/openapi-json-schema-generator
cd openapi-json-schema-generator
./bin/run-in-docker.sh mvn package
```

Build artifacts are now accessible in your working directory.

Once built, `run-in-docker.sh` will act as an executable for openapi-json-schema-generator-cli. To generate code, you'll need to output to a directory under `/gen` (e.g. `/gen/out`). For example:

```sh
./bin/run-in-docker.sh help # Executes 'help' command for openapi-json-schema-generator-cli
./bin/run-in-docker.sh list # Executes 'list' command for openapi-json-schema-generator-cli
./bin/run-in-docker.sh /gen/bin/python-petstore.sh  # Builds the Python client
./bin/run-in-docker.sh generate -i src/test/resources/3_0/petstore.yaml \
    -g go -o /gen/out/python-petstore -p packageName=petstore_api # generates python client, outputs locally to ./out/python-petstore
```

##### Troubleshooting

If an error like this occurs, just execute the **mvn clean install -U** command:

> org.apache.maven.lifecycle.LifecycleExecutionException: Failed to execute goal org.apache.maven.plugins:maven-surefire-plugin:2.19.1:test (default-test) on project openapi-json-schema-generator: A type incompatibility occurred while executing org.apache.maven.plugins:maven-surefire-plugin:2.19.1:test: java.lang.ExceptionInInitializerError cannot be cast to java.io.IOException

```sh
./run-in-docker.sh mvn clean install -U
```

> Failed to execute goal org.fortasoft:gradle-maven-plugin:1.0.8:invoke (default) on project openapi-json-schema-generator-gradle-plugin-mvn-wrapper: org.gradle.tooling.BuildException: Could not execute build using Gradle distribution 'https://services.gradle.org/distributions/gradle-4.7-bin.zip'

Right now: no solution for this one :|

<!-- /RELEASE_VERSION -->
## Getting Started

To generate a python client for [petstore.yaml](https://raw.githubusercontent.com/openapi-json-schema-tools/openapi-json-schema-generator/master/src/test/resources/3_0/petstore.yaml), please run the following
```sh
git clone https://github.com/openapi-json-schema-tools/openapi-json-schema-generator
cd openapi-json-schema-generator
mvn clean package
java -jar target/openapi-json-schema-generator-cli.jar generate \
   -i https://raw.githubusercontent.com/openapi-json-schema-tools/openapi-json-schema-generator/master/src/test/resources/3_0/petstore.yaml \
   -g python \
   -o /var/tmp/python_api_client
```
(if you're on Windows, replace the last command with `java -jar target\openapi-json-schema-generator-cli.jar generate -i https://raw.githubusercontent.com/openapi-json-schema-tools/openapi-json-schema-generator/master/src/test/resources/3_0/petstore.yaml -g python -o c:\temp\python_api_client`)

To get a list of **general** options available, please run `java -jar target/openapi-json-schema-generator-cli.jar help generate`

To get a list of python specified options (which can be passed to the generator with a config file via the `-c` option), please run `java -jar target/openapi-json-schema-generator-cli.jar config-help -g python`

## Usage

### To generate a sample client library
You can build a client against the [Petstore API](https://raw.githubusercontent.com/openapijsonschematools/openapi-json-schema-generator/master/src/test/resources/3_0/petstore.yaml) as follows:

```sh
./bin/generate-samples.sh ./bin/generate_samples_configs/python.yaml
```

(On Windows, please install [GIT Bash for Windows](https://gitforwindows.org/) to run the command above)

This script will run the generator with this command:

```sh
java -jar target/openapi-json-schema-generator-cli.jar generate \
  -i https://raw.githubusercontent.com/openapijsonschematools/openapi-json-schema-generator/master/src/test/resources/3_0/petstore.yaml \
  -g python \
  -t src/main/resources/python \
  --additional-properties packageName=petstore_api \
  -o samples/client/petstore/python
```

with a number of options. [The python options are documented here.](docs/generators/python.md)

You can also get the options with the `help generate` command (below only shows partial results):

```
NAME
        openapi-json-schema-generator-cli generate - Generate code with the specified
        generator.

SYNOPSIS
        openapi-json-schema-generator-cli generate
                [(-a <authorization> | --auth <authorization>)]
                [--api-name-suffix <api name suffix>] [--api-package <api package>]
                [--artifact-id <artifact id>] [--artifact-version <artifact version>]
                [(-c <configuration file> | --config <configuration file>)] [--dry-run]
                [(-e <templating engine> | --engine <templating engine>)]
                [--enable-post-process-file]
                [(-g <generator name> | --generator-name <generator name>)]
                [--git-host <git host>] [--git-repo-id <git repo id>]
                [--git-user-id <git user id>] [--global-property <global properties>...]
                [--group-id <group id>] [--hide-generation-timestamp]
                [--http-user-agent <http user agent>]
                [(-i <spec file> | --input-spec <spec file>)]
                [--ignore-file-override <ignore file override location>]
                [--ints-allowed-for-float-double-formats]
                [--invoker-package <invoker package>] [--minimal-update]
                [--model-name-prefix <model name prefix>]
                [--model-name-suffix <model name suffix>]
                [(-o <output directory> | --output <output directory>)] [(-p <additional properties> | --additional-properties <additional properties>)...]
                [--package-name <package name>] [--release-note <release note>]
                [--remove-enum-value-prefix] [--remove-operation-id-prefix]
                [(-s | --skip-overwrite)] [--skip-operation-example]
                [--skip-validate-spec] [--strict-spec <true/false strict behavior>]
                [(-t <template directory> | --template-dir <template directory>)]
                [(-v | --verbose)]
```
<details>
<summary>generate OPTIONS</summary>

```text
OPTIONS
        -a <authorization>, --auth <authorization>
            adds authorization headers when fetching the OpenAPI definitions
            remotely. Pass in a URL-encoded string of name:header with a comma
            separating multiple values

        --api-name-suffix <api name suffix>
            Suffix that will be appended to all API names ('tags'). Default:
            Api. e.g. Pet => PetApi. Note: Only ruby, python, jaxrs generators
            support this feature at the moment.

        --api-package <api package>
            package for generated api classes

        --artifact-id <artifact id>
            artifactId in generated pom.xml. This also becomes part of the
            generated library's filename

        --artifact-version <artifact version>
            artifact version in generated pom.xml. This also becomes part of the
            generated library's filename

        -c <configuration file>, --config <configuration file>
            Path to configuration file. It can be JSON or YAML. If file is JSON,
            the content should have the format {"optionKey":"optionValue",
            "optionKey1":"optionValue1"...}. If file is YAML, the content should
            have the format optionKey: optionValue. Supported options can be
            different for each language. Run config-help -g {generator name}
            command for language-specific config options.

        --dry-run
            Try things out and report on potential changes (without actually
            making changes).

        -e <templating engine>, --engine <templating engine>
            templating engine: "handlebars"(default) or "mustache"

        --enable-post-process-file
            Enable post-processing file using environment variables.

        -g <generator name>, --generator-name <generator name>
            generator to use (see list command for list)

        --git-host <git host>
            Git host, e.g. gitlab.com.

        --git-repo-id <git repo id>
            Git repo ID, e.g. openapi-generator.

        --git-user-id <git user id>
            Git user ID, e.g. openapijsonschematools.

        --global-property <global properties>
            sets specified global properties (previously called 'system
            properties') in the format of name=value,name=value (or multiple
            options, each with name=value)

        --group-id <group id>
            groupId in generated pom.xml

        --hide-generation-timestamp
            Hides the generation timestamp when files are generated.

        --http-user-agent <http user agent>
            HTTP user agent, e.g. codegen_csharp_api_client, default to
            'OpenAPI-Generator/{packageVersion}/{language}'

        -i <spec file>, --input-spec <spec file>
            location of the OpenAPI spec, as URL or file (required if not loaded
            via config using -c)

        --ignore-file-override <ignore file override location>
            Specifies an override location for the .openapi-generator-ignore
            file. Most useful on initial generation.

        --ints-allowed-for-float-double-formats
            Integers are allowed in for type: number format:float/double
            payloads

        --invoker-package <invoker package>
            root package for generated code

        --minimal-update
            Only write output files that have changed.

        --model-name-prefix <model name prefix>
            Prefix that will be prepended to all model names.

        --model-name-suffix <model name suffix>
            Suffix that will be appended to all model names.

        -o <output directory>, --output <output directory>
            where to write the generated files (current dir by default)

        -p <additional properties>, --additional-properties <additional
        properties>
            sets additional properties that can be referenced by the mustache
            templates in the format of name=value,name=value. You can also have
            multiple occurrences of this option.

        --package-name <package name>
            package for generated classes (where supported)

        --release-note <release note>
            Release note, default to 'Minor update'.

        --remove-enum-value-prefix
            Remove the common prefix of enum values

        --remove-operation-id-prefix
            Remove prefix of operationId, e.g. config_getId => getId

        -s, --skip-overwrite
            specifies if the existing files should be overwritten during the
            generation.

        --skip-operation-example
            Skip examples defined in operations to avoid out of memory errors.

        --skip-validate-spec
            Skips the default behavior of validating an input specification.

        --strict-spec <true/false strict behavior>
            'MUST' and 'SHALL' wording in OpenAPI spec is strictly adhered to.
            e.g. when false, no fixes will be applied to documents which pass
            validation but don't follow the spec.

        -t <template directory>, --template-dir <template directory>
            folder containing the template files

        -v, --verbose
            verbose mode
```

</details>

You can then use the auto-generated client. The README.md is a good starting point.

Other generators have [samples](samples) too.

### Customization

Please refer to [customization.md](docs/customization.md) on how to customize the output (e.g. package name, version)

### Workflow Integration

Please refer to [integration.md](docs/integration.md) on how to integrate OpenAPI generator with Maven, Gradle,  Github and CI/CD.

### License Information

The OpenAPI JSON Schema Generator project is intended as a benefit for users of the Open API Specification.  The project itself has the [License](#license) as specified. In addition, please understand the following points:

* The templates included with this project are subject to the [License](#license).
* Generated code is intentionally _not_ subject to the parent project license

When code is generated from this project, it shall be considered **AS IS** and owned by the user of the software.  There are no warranties--expressed or implied--for generated code.  You can do what you wish with it, and once generated, the code is your responsibility and subject to the licensing terms that you deem appropriate.

## Companies/Projects using OpenAPI JSON Schema Generator

[Github code search](https://github.com/search?q=DateSchema+DateTimeSchema+language%3APython+path%3A**%2Fschemas.py&type=code)

## About Us

This repo is based on v6.2.0 of OpenAPI Generator. This project focuses on making the output 100% compliant with JSON schema as part of the OpenAPI 3.1 specification with a focus on complex cases (top-down approach). The goal is to fully support everything defined in JSON schema so that developers can leverage JSON schema as well as OpenAPI specification in their API design. Building here allows for more rapid progress supporting new features in OpenAPI 3.X without having to support many older generators which don't use the new features.

### History of OpenAPI JSON Schema Generator

OpenAPI JSON Schema Generator is based on OpenAPI Generator v6.2.0.
The project was created here because the openapi-generator core team required the removal of the python generator 
from their project. The author of the python generator (@spacether) preferred to keep building 
in the openapi-generator repo, but core team refused to consider keeping python in openapi-generator.
Below is a timeline of those events and some of their reasons:

#### Timeline of python generator development
- Jan 4, 2021 - [openapi-generator v5.4.0: python-experimental created](https://github.com/OpenAPITools/openapi-generator/pull/8325). This generator is the beginning of the current python generator in this repo.
- Sept 19, 2022 - meeting to discuss openapi 3.1.0 + python client, removal of python client mentioned as an option, not a requirement
- Sept 22, 2022 - [openapi-generator v6.2.0: new python-experimental switched in as the primary python client](https://github.com/OpenAPITools/openapi-generator/pull/13501)
- Sept 23, 2022 - communication clarified that removal of the python generator is required
- Sept 24, 2022 - [openapi-generator v6.2.0: removal of python generator mentioned in v6.2.0 release](https://github.com/OpenAPITools/openapi-generator/releases/tag/v6.2.0)
- Sept 26, 2022 - [different new repo made for openapi json schema generator in OpenapiTools org](https://github.com/OpenAPITools/openapi-json-schema-generator)
- Oct 2, 2022 - [moved project to this repo](https://github.com/openapi-json-schema-tools/openapi-json-schema-generator) I moved the generator to the new repo because full ownership privledges were not granted on the new repo to me, which had been promised, and because I was not given privledges that allowed docker distribution from the new repo
- May 14, 2023 - [openapi-generator v7.0.0: python generator removed, a diffferent generator becomes the only python client](https://github.com/OpenAPITools/openapi-generator/pull/15486)


#### Removal Reasons
- Core team and @wing328 felt adoption of the python client was reduced from 5.0.0 and onward due to python-prior + python generators
- Some python users in the community did not prefer the new python code
- The fact that other users + companies are using it does not warrant keeping it in the repo
- The fact that it is more fully passing json schema tests (including the feature keywords oneOf/anyOf/allOf/additionalProperties) does not warrant keeping it in the repo
- The openapi-generator core team refused to consider the option of keeping the python generator as another generator option in their repo, and building another python generator that looks more conventional and making that generator primary

## License

-------
Copyright 2023 OpenAPI-Json-Schema-Generator Contributors
Copyright 2018 OpenAPI-Generator Contributors (https://openapi-generator.tech)
Copyright 2018 SmartBear Software

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at [apache.org/licenses/LICENSE-2.0](https://www.apache.org/licenses/LICENSE-2.0)

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.

---
 readmeEtag: '"b0d2141847afd7cc7f88872037794345f545f8da"' readmeLastModified: Sat, 15 Jun 2024 18:18:55 GMT repositoryId: 544314254 description: ' OpenAPI JSON Schema Generator allows auto-generation of API client libraries with a focus on JSON schema given an OpenAPI document' created: '2022-10-02T06:47:59Z' updated: '2025-12-31T04:39:24Z' language: Java archived: true stars: 171 watchers: 4 forks: 14 owner: openapi-json-schema-tools logo: https://avatars.githubusercontent.com/u/114849992?v=4 license: Apache-2.0 repoEtag: '"99bebdb488f45f7f8d567cf876ba6252d192a1dd9e89614ee1c8c270311a43a2"' repoLastModified: Wed, 31 Dec 2025 04:39:24 GMT category: - Code Generators - SDK - Parsers foundInMaster: true name: OpenAPI JSON Schema Generator link: https://github.com/openapi-json-schema-tools/openapi-json-schema-generator language: - Python - Java source_description: >- A template-driven engine to generate API client code + documentation by parsing your OpenAPI Description v2: false v3_1_link: >- https://github.com/openapi-json-schema-tools/openapi-json-schema-generator/issues/131 v3_1: true - source: - https://openapi.tools/ - openapi3 tags name: Kiota Api Client Generator category: - Code Generators - SDK link: https://aka.ms/kiota/docs language: C# repository: https://github.com/microsoft/kiota source_description: >- Kiota is a cross platform API Client code generator that is small, fast, and optimized for API consumers to find APIs and generate client code for just the parts of the API that they need. One tool, for any OpenAPI described API, that delivers a consistent client experience in multiple languages. v2: true v3: true v3_1: false id: c94b9d43825b39f99511edfe70e459ec repositoryMetadata: base64Readme: >- # Project

[![Dotnet](https://github.com/microsoft/kiota/actions/workflows/dotnet.yml/badge.svg)](https://github.com/microsoft/kiota/actions/workflows/dotnet.yml) [![CodeQL](https://github.com/microsoft/kiota/actions/workflows/codeql-analysis.yml/badge.svg)](https://github.com/microsoft/kiota/actions/workflows/codeql-analysis.yml) [![Coverage](https://sonarcloud.io/api/project_badges/measure?project=microsoft_kiota&metric=coverage)](https://sonarcloud.io/dashboard?id=microsoft_kiota) [![Sonarcloud Status](https://sonarcloud.io/api/project_badges/measure?project=microsoft_kiota&metric=alert_status)](https://sonarcloud.io/dashboard?id=microsoft_kiota)

Kiota is a command line tool for generating an API client to call any OpenAPI described API you are interested in. The goal is to eliminate the need to take a dependency on a different API SDK for every API that you need to call. Kiota API clients provide a strongly typed experience with all the features you expect from a high quality API SDK, but without having to learn a new library for every HTTP API.

This library builds on top of the [Microsoft.OpenAPI.NET](https://github.com/microsoft/openapi.net) library to ensure comprehensive support for APIs that use OpenAPI descriptions. One of the goals of the project is to provide the best code generator support possible for OpenAPI and JSON Schema features. The [conceptual documentation](https://learn.microsoft.com/openapi/kiota) describes how kiota works and the high level concepts, this readme documents how to get started with Kiota.

## Getting started

### Generating SDKs

1. Install required tools and dependencies. (refer to the [Supported Languages](#supported-languages) table under the **Required tools & dependencies** column)
1. Get Kiota using one of the [available options](https://learn.microsoft.com/openapi/kiota/install).
1. Generate your API client, checkout the [Parameters reference](https://learn.microsoft.com/openapi/kiota/using) for the different options.
1. Start calling your API using your fluent API Client.

### Supported languages

The following table provides an overview of the languages supported by Kiota and the progress in the implementation of the different components.

| Language | Generation | Abstractions                   | Serialization                                                                                                                                                                                                                                                                                                                                                                                | Authentication | HTTP | Required tools & dependencies |
| -------- | ---------- |--------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| -------------- | ---- | -------------- |
| CSharp | ✔ | [✔](https://github.com/microsoft/kiota-dotnet/tree/main/src/abstractions)     | [FORM](https://github.com/microsoft/kiota-dotnet/tree/main/src/serialization/form), [JSON](https://github.com/microsoft/kiota-dotnet/tree/main/src/serialization/json), [MULTIPART](https://github.com/microsoft/kiota-dotnet/tree/main/src/serialization/multipart), [TEXT](https://github.com/microsoft/kiota-dotnet/tree/main/src/serialization/text)                                     | [Anonymous](https://github.com/microsoft/kiota-dotnet/blob/main/src/abstractions/authentication/AnonymousAuthenticationProvider.cs), [API Key](https://github.com/microsoft/kiota-dotnet/blob/main/src/abstractions/authentication/ApiKeyAuthenticationProvider.cs), [Azure](https://github.com/microsoft/kiota-dotnet/tree/main/src/authentication/azure) | [✔](https://github.com/microsoft/kiota-dotnet/tree/main/src/http/httpClient) | [link](https://learn.microsoft.com/openapi/kiota/quickstarts/dotnet) |
| Dart | [🛠️](https://github.com/microsoft/kiota-dart) | [🛠️](https://github.com/microsoft/kiota-dart/tree/main/packages/microsoft_kiota_abstractions) | [🛠️ FORM](https://github.com/microsoft/kiota-dart/tree/main/packages/microsoft_kiota_serialization_form), [🛠️ JSON](https://github.com/microsoft/kiota-dart/tree/main/packages/microsoft_kiota_serialization_json), [🛠️ MULTIPART](https://github.com/microsoft/kiota-dart/tree/main/packages/microsoft_kiota_serialization_multipart), [🛠️ TEXT](https://github.com/microsoft/kiota-dart/tree/main/packages/microsoft_kiota_serialization_text) | [🛠️ Anonymous](https://github.com/microsoft/kiota-dart/blob/main/packages/microsoft_kiota_abstractions/lib/src/authentication/anonymous_authentication_provider.dart), [🛠️ API Key](https://github.com/microsoft/kiota-dart/blob/main/packages/microsoft_kiota_abstractions/lib/src/authentication/api_key_authentication_provider.dart) | [🛠️](https://github.com/microsoft/kiota-dart/tree/main/packages/microsoft_kiota_http) | [link](https://learn.microsoft.com/en-us/openapi/kiota/quickstarts/dart) |
| Go | ✔ | [✔](https://github.com/microsoft/kiota-abstractions-go)         | [FORM](https://github.com/microsoft/kiota-serialization-form-go), [JSON](https://github.com/microsoft/kiota-serialization-json-go), [MULTIPART](https://github.com/microsoft/kiota-serialization-multipart-go), [TEXT](https://github.com/microsoft/kiota-serialization-text-go)                                                                                                             | [Anonymous](https://github.com/microsoft/kiota-abstractions-go/blob/main/authentication/anonymous_authentication_provider.go), [API Key](https://github.com/microsoft/kiota-abstractions-go/blob/main/authentication/api_key_authentication_provider.go), [Azure](https://github.com/microsoft/kiota-authentication-azure-go/) | [✔](https://github.com/microsoft/kiota-http-go/) | [link](https://learn.microsoft.com/openapi/kiota/quickstarts/go) |
| Java | ✔ | [✔](https://github.com/microsoft/kiota-java/tree/main/components/abstractions)       | [FORM](https://github.com/microsoft/kiota-java/tree/main/components/serialization/form), [JSON](https://github.com/microsoft/kiota-java/tree/main/components/serialization/json), [MULTIPART](https://github.com/microsoft/kiota-java/tree/main/components/serialization/multipart), [TEXT](https://github.com/microsoft/kiota-java/tree/main/components/serialization/text)                 | [Anonymous](https://github.com/microsoft/kiota-java/blob/main/components/abstractions/src/main/java/com/microsoft/kiota/authentication/AnonymousAuthenticationProvider.java), [API Key](https://github.com/microsoft/kiota-java/blob/main/components/abstractions/src/main/java/com/microsoft/kiota/authentication/ApiKeyAuthenticationProvider.java), [Azure](https://github.com/microsoft/kiota-java/tree/main/components/authentication/azure) | [✔](https://github.com/microsoft/kiota-java/tree/main/components/http/okHttp) | [link](https://learn.microsoft.com/openapi/kiota/quickstarts/java) |
| PHP | ✔ | [✔](https://github.com/microsoft/kiota-abstractions-php)          | [JSON](https://github.com/microsoft/kiota-serialization-json-php), [FORM](https://github.com/microsoft/kiota-serialization-form-php), [MULTIPART](https://github.com/microsoft/kiota-serialization-multipart-php), [TEXT](https://github.com/microsoft/kiota-serialization-text-php)                                                                                                         | [Anonymous](https://github.com/microsoft/kiota-abstractions-php/blob/main/src/Authentication/AnonymousAuthenticationProvider.php), [✔️ PHP League](https://github.com/microsoft/kiota-authentication-phpleague-php) | [✔](https://github.com/microsoft/kiota-http-guzzle-php) | [link](https://learn.microsoft.com/openapi/kiota/quickstarts/php) |
| Python | ✔ | [✔](https://github.com/microsoft/kiota-abstractions-python)  | [FORM](https://github.com/microsoft/kiota-serialization-form-python), [JSON](https://github.com/microsoft/kiota-serialization-json-python), [MULTIPART](https://github.com/microsoft/kiota-serialization-multipart-python), [TEXT](https://github.com/microsoft/kiota-serialization-text-python)                                                                                             | [Anonymous](https://github.com/microsoft/kiota-abstractions-python/blob/main/kiota_abstractions/authentication/anonymous_authentication_provider.py), [Azure](https://github.com/microsoft/kiota-authentication-azure-python) | [✔](https://github.com/microsoft/kiota-http-python) | [link](https://learn.microsoft.com/openapi/kiota/quickstarts/python) |
| Ruby | 🛠️ | [🛠️](https://github.com/microsoft/kiota-abstractions-ruby)       | [❌ FORM](https://github.com/microsoft/kiota/issues/2077), [JSON](https://github.com/microsoft/kiota-serialization-json-ruby), [❌ MULTIPART](https://github.com/microsoft/kiota/issues/3032), [❌ TEXT](https://github.com/microsoft/kiota/issues/1049)                                                                                                                                        | [Anonymous](https://github.com/microsoft/kiota-abstractions-ruby/blob/main/lib/microsoft_kiota_abstractions/authentication/anonymous_authentication_provider.rb), [OAuth2](https://github.com/microsoft/kiota-authentication-oauth-ruby) | [🛠️](https://github.com/microsoft/kiota-http-ruby)|  |
| TypeScript/JavaScript | 🛠️ | [🛠️](https://github.com/microsoft/kiota-typescript/tree/main/packages/abstractions) | [FORM](https://github.com/microsoft/kiota-typescript/tree/main/packages/serialization/form), [JSON](https://github.com/microsoft/kiota-typescript/tree/main/packages/serialization/json), [MULTIPART](https://github.com/microsoft/kiota-typescript/tree/main/packages/serialization/multipart), [TEXT](https://github.com/microsoft/kiota-typescript/tree/main/packages/serialization/text) | [Anonymous](https://github.com/microsoft/kiota-typescript/blob/main/packages/abstractions/src/authentication/anonymousAuthenticationProvider.ts), [API Key](https://github.com/microsoft/kiota-typescript/blob/main/packages/abstractions/src/authentication/apiKeyAuthenticationProvider.ts), [Azure](https://github.com/microsoft/kiota-typescript/tree/main/packages/authentication/azure), [SPFx](https://github.com/microsoft/kiota-typescript/tree/main/packages/authentication/spfx) | [🛠️](https://github.com/microsoft/kiota-typescript/tree/main/packages/http/fetch) | [link](https://learn.microsoft.com/openapi/kiota/quickstarts/typescript) |

> Legend: ✔ -> stable, 🛠️ -> in preview, ❌ -> not started, ▶ -> in progress.

### Parameters reference

Parameters are documented [here](https://learn.microsoft.com/openapi/kiota/using).

### Debugging

Make sure you [install the pre-requisites first](CONTRIBUTING.md). If you are using Visual Studio Code as your IDE, the **launch.json** file already contains the configuration to run Kiota. By default this configuration will use the `openApiDocs/v1.0/Mail.yml` under the [PowerShell repository](https://github.com/microsoftgraph/msgraph-sdk-powershell) as the OpenAPI to generate an SDK for. By default this configuration will output the generated files in a graphdotnetv4|graphjavav4|graphtypescriptv4 folder located in the parent folder this repository is cloned in.

Selecting the language you want to generate an API client for in the Visual Studio Debug tab and hitting **F5** will automatically build, start, and attach the debugging process to Kiota.

### Samples

You can find samples of clients generated with Kiota in the [Kiota samples](https://github.com/microsoft/kiota-samples) repository.

An example of an application that is calling multiple API can be found in the [KiotaApp](https://github.com/darrelmiller/KiotaApp) repo

## Contributing

This project welcomes contributions and suggestions.  Most contributions require you to agree to a
Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us
the rights to use your contribution. For details, visit [https://cla.opensource.microsoft.com](https://cla.opensource.microsoft.com).

When you submit a pull request, a CLA bot will automatically determine whether you need to provide
a CLA and decorate the PR appropriately (e.g., status check, comment). Simply follow the instructions
provided by the bot. You will only need to do this once across all repos using our CLA.

This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/).
For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or
contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments.

## Trademarks

This project may contain trademarks or logos for projects, products, or services. Authorized use of Microsoft
trademarks or logos is subject to and must follow
[Microsoft's Trademark & Brand Guidelines](https://www.microsoft.com/legal/intellectualproperty/trademarks/usage/general).
Use of Microsoft trademarks or logos in modified versions of this project must not cause confusion or imply Microsoft sponsorship.
Any use of third-party trademarks or logos are subject to those third-party's policies.
 readmeEtag: '"06d219f12e0e36d56136770cc0c3e2a20333ff21"' readmeLastModified: Wed, 09 Jul 2025 20:42:50 GMT repositoryId: 323665366 description: OpenAPI based HTTP Client code generator created: '2020-12-22T15:35:49Z' updated: '2026-02-05T17:55:22Z' language: C# archived: false stars: 3630 watchers: 40 forks: 299 owner: microsoft logo: https://avatars.githubusercontent.com/u/6154722?v=4 license: MIT repoEtag: '"3bb9f2640c7b3020aaca53c675792de626f562911984e67a0b1e063241d2db59"' repoLastModified: Thu, 05 Feb 2026 17:55:22 GMT foundInMaster: true - source: https://openapi.tools/ name: FabriKt category: - Code Generators - SDK link: https://github.com/cjbooms/fabrikt language: Kotlin repository: https://github.com/fabrikt-io/fabrikt source_description: >- A sophisticated Kotlin code generation library capable of generating Jackson-annotated data classes, Spring Controller interfaces, and fault-tolerant OkHttp clients. Written in Kotlin, this library programatically generates code and is capable of handling advanced OpenApi3 specification features such as polymorphism. v2: false v3: true v3_1: false id: 590c094dc10757222dccc4b807bf562d repositoryMetadata: base64Readme: >- # Fabrikt `/ˈfa-brikt/` - Kotlin code from OpenAPI 3

* [Introduction](#introduction)
* [Features](#features)
* [Examples](#examples)
* [Usage Instructions](#usage-instructions)
  * [Command Line](#command-line)
  * [Gradle w/ custom task](#gradle-w-custom-task)
  * [Gradle w/ plugin](#gradle-w-plugin)
  * [Maven](#maven)
  * [Docker](#docker)
* [Getting the Most from Fabrikt](#getting-the-most-from-fabrikt)
* [Configuration Options](#configuration-options)
* [Original Motivation](#original-motivation)
* [Building Locally](#building-locally)
* [Publishing](#publishing)
* [Specific Features](#specific-features)

## Introduction

This library was built to take advantage of the complex modeling features available in OpenAPI 3. It generates Kotlin data classes with advanced support for features such as: 
 - Null Safety
 - Inlined schema definitions
 - Enumerations 
 - Sealed Classes
 - Polymorphism (`@JsonSubTypes`)
 - Maps of Maps of Maps
 - GraalVM Native Reflection Registration
 - Json Merge Patch (via `JsonNullable`) (add `x-json-merge-patch: true` to schemas)
 - Override Jackson Include NonNull (via `JsonInclude`) (add `x-jackson-include-non-null: true` to schemas)
 
as well as HTTP clients and controllers for a number of popular frameworks (see [Features](#features)).

More than just bootstrapping, this library can be permanently integrated into your build tool and will ensure contract and code always match, even as APIs evolve in complexity.

### Try Fabrikt Online

Try Fabrikt with your own API spec in the [Fabrikt Playground](https://try.fabrikt.io) and see how it can help you generate code for your API clients and servers.

[![Screenshot of Fabrikt Playground](https://github.com/user-attachments/assets/34ac6aff-f27e-4f48-bc53-cbb2a00e3e78)](https://try.fabrikt.io)

### Coordinates

```xml
<dependency>
  <groupId>com.cjbooms</groupId>
  <artifactId>fabrikt</artifactId>
</dependency>
```

## Features

The library currently has support for generating:

* Models
  * **Jackson** annotated **data classes**
  * **Kotlinx.serialization** annotated **data classes**
* Clients
  * **OkHttp Client (w/ Jackson Models)** - with the option for a resilience4j fault-tolerance wrapper
  * **OpenFeign** annotated client interfaces
  * **Ktor Client (w/ Jackson & Kotlin Serialization models)**
  * **Spring HTTP Interface** annotated client interfaces
* Controllers
  * **Spring MVC** annotated controller interfaces
  * **Micronaut** HTTP annotated controller interfaces
  * **Ktor server** routes and controller interfaces

## Examples

Consult test directory for OpenAPI code generation examples. 

It forms a living documentation full of [code examples](src/test/resources/examples) generated from different OpenAPI 3 permutations.

Furthermore, the [end-to-end tests](/end2end-tests) demonstrate how to integrate the library using Gradle.

## Usage Instructions

The library can be used in a variety of ways, including as a command line tool, a Gradle task, or a Maven plugin.

Please refer to [Configuration Options](#configuration-options) section for a list of available parameters.

### Command Line

Fabrikt is packaged as an executable jar, allowing it to be integrated into any build tool. 

The CLI can be invoked as follows:

```
java -jar fabrikt.jar \
    --output-directory '/tmp' \
    --base-package 'com.example' \
    --api-file '/path-to-api/open-api.yaml' \
    --targets 'client' \
    --targets 'http_models' \
    --http-client-opts resilience4j
```

### Gradle w/ custom task

Here is an example of a Gradle task with code generated to the `build/generated` directory, and execution linked to the compile task. 

```kotlin
val fabrikt: Configuration by configurations.creating

val generationDir = "$buildDir/generated"
val apiFile = "$buildDir/path-to-api/open-api.yaml"

sourceSets {
    main { java.srcDirs("$generationDir/src/main/kotlin") }
    test { java.srcDirs("$generationDir/src/test/kotlin") }
    ...
}

tasks {   
    ...
    val generateCode by creating(JavaExec::class) {
        inputs.files(apiFile)
        outputs.dir(generationDir)
        outputs.cacheIf { true }
        classpath(fabrikt)
        mainClass.set("com.cjbooms.fabrikt.cli.CodeGen")
        args = listOf(
            "--output-directory", generationDir,
            "--base-package", "com.example",
            "--api-file", apiFile,
            "--targets", "http_models",
            "--targets", "client",
            "--http-client-opts", "resilience4j"
        )
    }
    withType<org.jetbrains.kotlin.gradle.tasks.KotlinCompile> {
        kotlinOptions.jvmTarget = "17"
        dependsOn(generateCode)
    }
}

dependencies {
     fabrikt("com.cjbooms:fabrikt:+") // This should be pinned  
     ...
}
```

### Gradle w/ plugin

The [Fabrikt Gradle plugin](https://github.com/acanda/fabrikt-gradle-plugin) serves as a convenient wrapper for Fabrikt, 
allowing seamless integration of code generation into a Gradle build.

**Note:** Since the plugin is maintained separately from the Fabrikt library, please refer to the
[Configuration](https://github.com/acanda/fabrikt-gradle-plugin?tab=readme-ov-file#configuration) section of the 
plugin's README for the most up-to-date information on how to use it.

Latest version of the plugin: [![Gradle Plugin Portal Version](https://img.shields.io/gradle-plugin-portal/v/ch.acanda.gradle.fabrikt?style=flat)](https://plugins.gradle.org/plugin/ch.acanda.gradle.fabrikt)

```kotlin
plugins {
    // find latest version: https://github.com/acanda/fabrikt-gradle-plugin/releases
    id("ch.acanda.gradle.fabrikt") version "1.27.3"
}

fabrikt {
    generate("dog") {
        apiFile = file("src/main/openapi/dog.yaml")
        basePackage = "com.example.api"
    }
}
```

### Maven

The [exec-maven-plugin](http://www.mojohaus.org/exec-maven-plugin/examples/example-exec-using-plugin-dependencies.html) is capable of downloading the Fabrikt library from Maven Central and executing its main method with defined arguments.

### Docker

Fabrikt is also available as a Docker image, which can be convenient for CI/CD pipelines or environments where you prefer not to install Java directly.

The Docker image can be invoked as follows:

```bash
docker run --rm -v $(pwd):/workspace ghcr.io/fabrikt-io/fabrikt:latest \
  --output-directory '.' \
  --base-package 'com.example' \
  --api-file 'openapi.yaml' \
  --targets 'http_models'
```

The command mounts your current directory to `/workspace` in the container, where Fabrikt will read the OpenAPI specification and write the generated code.

## Getting the Most from Fabrikt

### 1. Prefer components to inline schemas
While inline schemas are perfectly valid they are not supported by Fabrikt in all circumstances.
This is especially true for request bodies and non-trivial parameters. Instead, define your schemas in the
components section of the OpenAPI spec (`components.parameters` & `components.requestBodies`). [#20](https://github.com/cjbooms/fabrikt/issues/20), [#187](https://github.com/cjbooms/fabrikt/issues/187)

### 2. Use `oneOf` with discriminator for polymorphism
`oneOf` along with the flag `SEALED_INTERFACES_FOR_ONE_OF` will generate polymorphic models with sealed interfaces.
The `discriminator` property is used by Fabrikt to determine the subtypes to be generated.

## Configuration Options

This section documents the available CLI parameters for controlling what gets generated. This documentation is generated using: `./gradlew printCodeGenUsage`

| Parameter                      | Description |
| ------------------------------ | ------------------------------ |
|   `--api-file`                 | This must be a valid Open API v3 spec. All code generation will be based off this input. |
|   `--api-fragment`             | A partial Open API v3 fragment, to be combined with the primary API for code generation purposes. |
| * `--base-package`             | The base package which all code will be generated under. |
|   `--external-ref-resolution`  | Specify to which degree referenced schemas from external files are included in model generation. Default: TARGETED |
|                                | CHOOSE ONE OF: |
|                                |   `TARGETED` - Generate models only for directly referenced schemas in external API files. |
|                                |   `AGGRESSIVE` - Referencing any schema in an external API file triggers generation of every external schema in that file. |
|   `--http-client-opts`         | Select the options for the http client code that you want to be generated. |
|                                | CHOOSE ANY OF: |
|                                |   `RESILIENCE4J` - Generates a fault tolerance service for the client using the following library "io.github.resilience4j:resilience4j-all:+" (only for OkHttp clients) |
|                                |   `SUSPEND_MODIFIER` - This option adds the suspend modifier to the generated client functions (only for OpenFeign clients) |
|                                |   `SPRING_RESPONSE_ENTITY_WRAPPER` - This option adds the Spring-ResponseEntity generic around the response to be able to get response headers and status (only for OpenFeign clients). |
|                                |   `SPRING_CLOUD_OPENFEIGN_STARTER_ANNOTATION` - This option adds the @FeignClient annotation to generated client interface |
|   `--http-client-target`       | Optionally select the target client that you want to be generated. Defaults to OK_HTTP |
|                                | CHOOSE ONE OF: |
|                                |   `OK_HTTP` - Generate OkHttp client. |
|                                |   `OPEN_FEIGN` - Generate OpenFeign client. |
|                                |   `SPRING_HTTP_INTERFACE` - Generate Spring HTTP Interface. |
|                                |   `KTOR` - Generate Ktor client. |
|   `--http-controller-opts`     | Select the options for the controllers that you want to be generated. |
|                                | CHOOSE ANY OF: |
|                                |   `SUSPEND_MODIFIER` - This option adds the suspend modifier to the generated controller functions |
|                                |   `AUTHENTICATION` - This option adds the authentication parameter to the generated controller functions |
|                                |   `COMPLETION_STAGE` - This option makes generated controller functions have Type CompletionStage<T> (works only with Spring Controller generator) |
|   `--http-controller-target`   | Optionally select the target framework for the controllers that you want to be generated. Defaults to Spring Controllers |
|                                | CHOOSE ONE OF: |
|                                |   `SPRING` - Generate for Spring framework. |
|                                |   `MICRONAUT` - Generate for Micronaut framework. |
|                                |   `KTOR` - Generate for Ktor server. |
|   `--http-model-opts`          | Select the options for the http models that you want to be generated. |
|                                | CHOOSE ANY OF: |
|                                |   `X_EXTENSIBLE_ENUMS` - This option treats x-extensible-enums as enums |
|                                |   `JAVA_SERIALIZATION` - This option adds Java Serializable interface to the generated models |
|                                |   `QUARKUS_REFLECTION` - This option adds @RegisterForReflection to the generated models. Requires dependency "'io.quarkus:quarkus-core:+" |
|                                |   `MICRONAUT_INTROSPECTION` - This option adds @Introspected to the generated models. Requires dependency "'io.micronaut:micronaut-core:+" |
|                                |   `MICRONAUT_REFLECTION` - This option adds @ReflectiveAccess to the generated models. Requires dependency "'io.micronaut:micronaut-core:+" |
|                                |   `MICRONAUT_SERDEABLE` - This option adds @Serdeable to the generated models. Requires dependency "'io.micronaut.serde:micronaut-serde-jackson:+" |
|                                |   `INCLUDE_COMPANION_OBJECT` - This option adds a companion object to the generated models. |
|                                |   `SEALED_INTERFACES_FOR_ONE_OF` - This option enables the generation of interfaces for discriminated oneOf types |
|                                |   `NON_NULL_MAP_VALUES` - This option makes map values non-null. The default (since v15) and most spec compliant is make map values nullable |
|   `--http-model-suffix`        | Specify custom suffix for all generated model classes. Defaults to no suffix. |
|   `--instant-library`          | Specify which Instant library to use in generated model classes for kotlinx.serialization. Default: KOTLINX_INSTANT |
|                                | CHOOSE ONE OF: |
|                                |   `KOTLINX_INSTANT` - Use `kotlinx.datetime` Instant in generated classes (default) |
|                                |   `KOTLIN_TIME_INSTANT` - Use `kotlin.time` Instant in generated classes |
|   `--jackson-nullability-mode` | Configure advanced handling when serializing null values with Jackson. Default: NONE |
|                                | CHOOSE ONE OF: |
|                                |   `NONE` - Default Jackson behaviour |
|                                |   `ENFORCE_OPTIONAL_NON_NULL` - Omit null values for optional non-null fields |
|                                |   `ENFORCE_REQUIRED_NULLABLE` - Include null values for required nullable fields |
|                                |   `STRICT` - Combines `ENFORCE_OPTIONAL_NON_NULL` and `ENFORCE_REQUIRED_NULLABLE` for strictest contract enforcement |
|   `--openfeign-client-name`    | Specify openfeign client name for spring-cloud-starter-openfeign. Defaults to 'fabrikt-client'. |
|   `--output-directory`         | Allows the generation dir to be overridden. Defaults to current dir |
|   `--output-opts`              | Select options for the output. |
|                                | CHOOSE ANY OF: |
|                                |   `ADD_FILE_DISCLAIMER` - This option adds a disclaimer to the generated files. |
|   `--resources-path`           | Allows the path for generated resources to be overridden. Defaults to `src/main/resources` |
|   `--serialization-library`    | Specify which serialization library to use for annotations in generated model classes. Default: JACKSON |
|                                | CHOOSE ONE OF: |
|                                |   `JACKSON` - Use Jackson for serialization and deserialization |
|                                |   `KOTLINX_SERIALIZATION` - Use kotlinx.serialization for serialization and deserialization |
|   `--src-path`                 | Allows the path for generated source files to be overridden. Defaults to `src/main/kotlin` |
|   `--targets`                  | Targets are the parts of the application that you want to be generated. |
|                                | CHOOSE ANY OF: |
|                                |   `HTTP_MODELS` - Jackson annotated data classes to represent the schema objects defined in the input. |
|                                |   `CONTROLLERS` - Spring / Micronaut / Ktor HTTP controllers for each of the endpoints defined in the input. |
|                                |   `CLIENT` - Simple http rest client. |
|                                |   `QUARKUS_REFLECTION_CONFIG` - This options generates the reflection-config.json file for quarkus integration projects |
|   `--type-overrides`           | Specify non-default kotlin types for certain OAS types. For example, generate `Instant` instead of `OffsetDateTime` |
|                                | CHOOSE ANY OF: |
|                                |   `DATETIME_AS_INSTANT` - Use `Instant` as the datetime type. Defaults to `OffsetDateTime` |
|                                |   `DATETIME_AS_LOCALDATETIME` - Use `LocalDateTime` as the datetime type. Defaults to `OffsetDateTime` |
|                                |   `BYTE_AS_STRING` - Ignore string format `byte` and use `String` as the type |
|                                |   `BINARY_AS_STRING` - Ignore string format `binary` and use `String` as the type |
|                                |   `URI_AS_STRING` - Ignore string format `uri` and use `String` as the type |
|                                |   `UUID_AS_STRING` - Ignore string format `uuid` and use `String` as the type |
|                                |   `DATE_AS_STRING` - Ignore string format `date` and use `String` as the type |
|                                |   `DATETIME_AS_STRING` - Ignore string format `date-time` and use `String` as the type |
|                                |   `BYTEARRAY_AS_INPUTSTREAM` - Use `InputStream` as ByteArray type. Defaults to `ByteArray` |
|   `--validation-library`       | Specify which validation library to use for annotations in generated model classes. Default: JAVAX_VALIDATION |
|                                | CHOOSE ONE OF: |
|                                |   `JAVAX_VALIDATION` - Use `javax.validation` annotations in generated model classes (default) |
|                                |   `JAKARTA_VALIDATION` - Use `jakarta.validation` annotations in generated model classes |
|                                |   `NO_VALIDATION` - Use no validation annotations in generated model classes |

## Original Motivation

The team that built the first version of this tool initially contributed to the Kotlin code generation ability in
[OpenApiTools](https://github.com/OpenAPITools/openapi-generator), but reached the limits of what could be achieved with
template-based generation. This library leverages the rich OpenAPI 3 model provided by
[KaiZen-OpenApi-Parser](https://github.com/RepreZen/KaiZen-OpenApi-Parser) and uses [Kotlin Poet](https://square.github.io/kotlinpoet/) to
programmatically construct Kotlin classes for maximum flexibility.

This project was started by engineers from [Zalando Tech](https://opensource.zalando.com/) and is battle-tested heavily in production there.

## Specific Features

### Polymorphism via `allOf`

The following example shows how `allOf` can be used to generate polymorphic Kotlin data classes. It does the following:
- A `ChildDefinition` schema defines both the discriminator mapping details and the schema for the discriminator property. In this case the discriminator property is an enumeration
- In each child schema, `allOf` is used to merge the `ChildDefinition` with the child's custom schema. This guarantees that each child schema inherits the correct discriminator property.
- In the `Responses` schema a `oneOf` lists only child schemas. This will be detected by Fabrikt and it will generate the list of parent types: `List<ChildDefinition>`

_NOTE: A new feature has been added that allows Polymorphism to be achieved using only a [discriminated oneOf](src/test/resources/examples/discriminatedOneOf). This feature makes use of Kotlin `sealed interface` and must be explicitly enabled via `--http-model-opts, SEALED_INTERFACES_FOR_ONE_OF`_

```yml
openapi: 3.0.0
components:
  schemas:
    ChildDefinition:
      type: object
      discriminator:
        propertyName: some_enum
        mapping:
          obj_one_only: '#/components/schemas/DiscriminatedChild1'
          obj_two_first: '#/components/schemas/DiscriminatedChild2'
          obj_two_second: '#/components/schemas/DiscriminatedChild2'
          obj_three: '#/components/schemas/discriminated_child_3'
      properties:
        discriminating_property:
          $ref: '#/components/schemas/ChildDiscriminator'

    ChildDiscriminator:
      type: string
      enum:
        - obj_one_only
        - obj_two_first
        - obj_two_second
        - obj_three

    DiscriminatedChild1:
      allOf:
        - $ref: '#/components/schemas/ChildDefinition'
        - type: object
          properties:
            some_prop:
              type: string

    DiscriminatedChild2:
      allOf:
        - $ref: '#/components/schemas/ChildDefinition'
        - type: object
          properties:
            some_prop:
              type: string

    discriminated_child_3:
      allOf:
        - $ref: '#/components/schemas/ChildDefinition'

    Responses:
      type: "object"
      properties:
        entries:
          type: "array"
          items:
            oneOf:
              - $ref: "#/components/schemas/DiscriminatedChild2"
              - $ref: "#/components/schemas/DiscriminatedChild2"
              - $ref: "#/components/schemas/discriminated_child_3"
```
```kotlin
@JsonTypeInfo(
    use = JsonTypeInfo.Id.NAME,
    include = JsonTypeInfo.As.EXISTING_PROPERTY,
    property = "some_enum",
    visible = true
)
@JsonSubTypes(
    JsonSubTypes.Type(
        value = DiscriminatedChild1::class,
        name =
        "obj_one_only"
    ),
    JsonSubTypes.Type(
        value = DiscriminatedChild2::class,
        name =
        "obj_two_first"
    ),
    JsonSubTypes.Type(
        value = DiscriminatedChild2::class,
        name =
        "obj_two_second"
    ),
    JsonSubTypes.Type(value = DiscriminatedChild3::class, name = "obj_three")
)
sealed class ChildDefinition() {
    abstract val someEnum: ChildDiscriminator
}

enum class ChildDiscriminator(
    @JsonValue
    val value: String
) {
    OBJ_ONE_ONLY("obj_one_only"),

    OBJ_TWO_FIRST("obj_two_first"),

    OBJ_TWO_SECOND("obj_two_second"),

    OBJ_THREE("obj_three");

    companion object {
        private val mapping: Map<String, ChildDiscriminator> =
            values().associateBy(ChildDiscriminator::value)

        fun fromValue(value: String): ChildDiscriminator? = mapping[value]
    }
}

data class DiscriminatedChild1(
    @param:JsonProperty("some_prop")
    @get:JsonProperty("some_prop")
    val someProp: String? = null,
    @get:JsonProperty("some_enum")
    @get:NotNull
    override val someEnum: ChildDiscriminator = ChildDiscriminator.OBJ_ONE_ONLY
) : ChildDefinition()

data class DiscriminatedChild2(
    @get:JsonProperty("some_enum")
    @get:NotNull
    override val someEnum: ChildDiscriminator,
    @param:JsonProperty("some_prop")
    @get:JsonProperty("some_prop")
    val someProp: String? = null
) : ChildDefinition()

data class DiscriminatedChild3(
    @get:JsonProperty("some_enum")
    @get:NotNull
    override val someEnum: ChildDiscriminator = ChildDiscriminator.OBJ_THREE
) : ChildDefinition()

data class Responses(
    @param:JsonProperty("entries")
    @get:JsonProperty("entries")
    @get:Valid
    val entries: List<ChildDefinition>? = null
)
```

## Contributing

### Building Locally

Fabrikt is built with Gradle and requires an initialised git repository. The easiest way to build it is to clone the repo locally before executing the build command:
```
git clone git@github.com:cjbooms/fabrikt.git
cd fabrikt/
./gradlew clean build
```

### Adjusting Test Examples

A utility function is available in [GeneratedCodeAsserter.kt](src/test/kotlin/com/cjbooms/fabrikt/util/GeneratedCodeAsserter.kt) to mass change all of the code generation examples in the test resources folder. This is useful when a global change is made to the code generation logic and all of the examples need to be updated.

### Publishing

1. Go to [Release Tab](https://github.com/cjbooms/fabrikt/releases)
2. Select `Draft a new release`.
3. Set tag to a version greater than current using symantic versioning, anticipating whether the changes made could break builds.
4. Click `Generate release notes`. Ensure that the tag and release version match.
5. Click `Publish release` buttom at the bottom.

Github Actions will publish the deployment to [Sonatype Central](https://central.sonatype.com/publishing/deployments). You must then log in to Sonatype and decide to either release or drop that deployment. After 30 minutes or so, Maven Central will have indexed the promoted release.
 readmeEtag: '"f3d553b88c26051e7b395545a29a9094c9ecf463"' readmeLastModified: Wed, 04 Feb 2026 12:37:56 GMT repositoryId: 229844927 description: Generates Kotlin Code from OpenApi3 Specifications created: '2019-12-24T00:56:42Z' updated: '2026-02-04T12:38:00Z' language: Kotlin archived: false stars: 245 watchers: 7 forks: 66 owner: fabrikt-io logo: https://avatars.githubusercontent.com/u/236459801?v=4 license: Apache-2.0 repoEtag: '"be3c784730bd28eabe7f45da0fbd81826d96c980e58861f0f4bde9ccc9715e36"' repoLastModified: Wed, 04 Feb 2026 12:38:00 GMT foundInMaster: true oldLocations: - https://github.com/cjbooms/fabrikt - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/bump-sh/cli v3: true repositoryMetadata: base64Readme: >- # Bump CLI

<p align="center">
  <img width="20%" src="https://bump.sh/icon-default-maskable-large.png" />
</p>

<p align="center">
  <a href="https://docs.bump.sh/help">Help</a> |
  <a href="https://bump.sh/users/sign_up">Sign up</a>
</p>

The Bump.sh CLI is used to interact with API documentation and hubs hosted on Bump.sh from your choice of popular API description formats: OpenAPI, Swagger, or AsyncAPI.

Using [OpenAPI](https://github.com/OAI/OpenAPI-Specification) (v3.x and v2.0) or [AsyncAPI](https://www.asyncapi.com/docs/reference/specification/latest) (2.x), you can do any of the following:

- Validate an API document before publishing to your documentation.
- Publish an API document to your Bump.sh documentation or hubs.
- Compare two API documents to generate a human-readable diff from your API definition.
Under the hood, it uses the API of [developers.bump.sh](https://developers.bump.sh). And is built with the [`oclif`](https://oclif.io) framework in Typescript.

[![Version](https://img.shields.io/npm/v/bump-cli.svg)](https://npmjs.org/package/bump-cli)
[![Tests](https://github.com/bump-sh/cli/actions/workflows/checks.yml/badge.svg)](https://github.com/bump-sh/cli/actions/workflows/checks.yml)
[![License](https://img.shields.io/npm/l/bump-cli.svg)](https://github.com/bump-sh/cli/blob/master/package.json)

## Table of contents

* [Installation](#installation)
* [Usage](#usage)
* [Commands](#commands)
* [Development](#development)
* [Contributing](#contributing)
* [Versioning](#versioning)

## Installation

The Bump.sh CLI is a node package currently distributed via NPM. This means you must have the Node v20+ interpreter installed on your computer or CI servers.

_If you are looking to use Bump.sh in a continuous integration environment you might be interested by [our Github Action](https://github.com/marketplace/actions/bump-sh-api-documentation-changelog)._

> You can download a standalone package directly from the latest
> Github release assets if you don’t use Node.
{: .info}

### Global installation

To install it globally, run the following command with NPM:

```shell
npm install -g bump-cli
```

Or, with Yarn via:

```shell
yarn global add bump-cli
```

### Add Bump.sh to your Node project

As our CLI is a node package, you can easily embed it to your project by adding the package to your `package.json` file, either with NPM:

```shell
npm install --save-dev bump-cli
```

Or with Yarn via:

```shell
yarn add --dev bump-cli
```

You can then use any Bump.sh commands with `npx` (same as `npm exec`):

```shell
npx bump --help
```

### Can I install Bump.sh CLI without using NodeJS?

Unfortunately, at the moment we only support the Node environment. However, you can download a standalone package directly from the [latest Github release](https://github.com/bump-sh/cli/releases) assets which you can run as a standalone binary. Or you can push your documentation using [our API](https://developers.bump.sh/) (advanced usage only).

## Usage

To list all the available commands, just type `bump` in your command line environment.

```shell
$ bump --help
The Bump.sh CLI is used to interact with your API documentation hosted on Bump.sh by using the API of developers.bump.sh

VERSION
  bump-cli/x.y.z linux-x64 node-v20+

USAGE
  $ bump [COMMAND]

COMMANDS
  deploy   Create a new version of your documentation from the given file or URL.
  diff     Get a comparison diff with your documentation from the given file or URL.
  help     Display help for bump.
  overlay  Apply an OpenAPI specified overlay to your API definition.
  preview  Create a documentation preview from the given file or URL.
```

 You can also get some help anytime by adding `--help` to any command. Example: `bump deploy --help`.

### Prepare your Bump.sh account

While some commands don't need any API token (`preview` or `diff`) you will need an access key if you want to interact with your Bump.sh documentation.

Head over to your Documentation settings in the “CI deployment” section or your Account or Organization settings in the “API keys” section to fetch a personal token for later usage.

## Commands

* [`bump deploy [FILE]`](#the-deploy-command)
* [`bump diff [FILE]`](#the-diff-command)
* [`bump preview [FILE]`](#the-preview-command)
* [`bump overlay [DEFINITION_FILE] [OVERLAY_FILE]`](#the-overlay-command)

### The `deploy` command

When an API is updated, the documentation should be updated at the same time. This is what the deploy command is for.

```shell
bump deploy path/to/api-document.yml --doc my-documentation --token $DOC_TOKEN
```

> You can find your own `my-documentation` slug and `$DOC_TOKEN` api key from your [documentation settings](https://bump.sh/docs).
{: .info}

You can also deploy a given API document to a different branch of your documentation with the `--branch <branch-name>` parameter. Please note the branch will be created if it doesn’t exist. More details about the branching feature are available on [this dedicated help page](https://docs.bump.sh/help/branching). E.g. deploy the API document to the `staging` branch of the documentation:

```shell
bump deploy path/to/api-document.yml --doc my-documentation --token $DOC_TOKEN --branch staging
```

#### Deploy a folder all at once

If you already have a hub in your [Bump.sh](https://bump.sh) account, you can automatically create documentation and deploy it into that hub by publishing a whole directory containing multiple API documents in a single command:

```shell
bump deploy dir/path/to/apis/ --auto-create --hub my-hub --token $HUB_TOKEN
```

> You can find your own `my-hub` slug and `$HUB_TOKEN` api key from your [hub settings](https://bump.sh/hubs).
{: .info}

Please note, by default, only files named `{slug}-api.[format]` are published. Where `{slug}` is a name for your API and `[format]` is either `yaml` or `json`. Adjust to your file naming convention using the `--filename-pattern <pattern>` option.

Note that it _can_ include `*` wildcard special character, but **must** include the `{slug}` filter to extract your documentation’s slug from the filename. The pattern can also have any other optional fixed characters.

Here’s a practical example. Let's assume that you have the following files in your `path/to/apis/` directory:

```
path/to/apis
└─ private-api-users-service.json
└─ partner-api-payments-service.yml
└─ public-api-contracts-service.yml
└─ data.json
└─ README.md
```

In order to deploy the 3 services API definition files from this folder (`private-api-users-service.json`, `partner-api-payments-service.yml` and `public-api-contracts-service.yml`), you can execute the following command:

```
bump deploy path/to/apis/ --hub my-hub --filename-pattern '*-api-{slug}-service'
```

#### Validate an API document

Simulate your API document's deployment to ensure it is valid by adding the `--dry-run` flag to the `deploy` command. It is handy in a Continuous Integration environment running a test deployment outside your main branch:

```shell
bump deploy path/to/api-document.yml --dry-run --doc my-documentation --token $DOC_TOKEN
```

Please check `bump deploy --help` for more usage details.

### The `diff` command

Using the `diff` command can help to spot differences between the local API
document and the latest deployed version.

#### Public API diffs

From any two API documents or URLs, you can retrieve a comprehensive changelog
of what has changed between them.

```shell
$ bump diff path/to/your/file.yml path/to/your/second_file.yml
* Comparing the two given definition files... done
Modified: GET /consommations
  Response modified: 200
    [Breaking] Body attribute modified: energie
```

By default the command will always exit with a successful return code. If you
want to use this command in a CI environment and want the command to fail **in
case of a breaking change**, you will need to add the `--fail-on-breaking` flag
to your diff command.

By default if the environment variable `CI=1` is present (in most continuous
integration environment), the flag will be enabled. In that case you can disable
the failures with `--no-fail-on-breaking` flag.

You can also test this feature in our dedicated web application at
<https://api-diff.io/>.

#### GitHub Integration

If you want to receive automatic `bump diff` results on Github Pull Requests you
might be interested by [our Github
Action](https://github.com/marketplace/actions/bump-sh-api-documentation-changelog#deploy-documentation--diff-on-pull-requests)
which has support for the diff command.

#### Authenticated diffs related to your Bump.sh documentation

From an existing Bump.sh documentation, the `diff` command will retrieve a
comparison changelog between your latest published documentation and the given
file or URL:

```shell
bump diff path/to/your/file.yml --doc my-documentation --token $DOC_TOKEN
```

If you want to compare two unpublished versions of your API document, the `diff` command can retrieve a comparison changelog between two given file or URL, “as simple as `git diff`”:

```shell
bump diff path/to/your/file.yml path/to/your/next-file.yml --doc my-documentation --token $DOC_TOKEN
```

Please check `bump diff --help` for full usage details.

#### Diffs with overlayed source files

The `bump diff` command also supports [overlays](#the-overlay-command). This means you can pass the `--overlay my-overlay-file.yml` flag to the command and it will apply the overlay on both files **before** running the diff. E.g.:

```shell
bump diff --overlay my-overlay.yml path/to/your/file.yml path/to/your/next-file.yml
```

### The `preview` command

When writing documentation, you might want to preview how it renders on Bump.sh.
This is precisely the goal of the `preview` command: it will create temporary
documentation with a unique URL, which will be available for a short period (30
minutes).

Usage from a local OpenAPI or AsyncAPI document:

```shell
bump preview path/to/file.json
```

You can also preview a document available via a URL:

```shell
bump preview https://developers.bump.sh/source.yaml
```

#### Live preview

By using the `--live` flag you can stay focused on API design (OpenAPI or AsyncAPI file) while seeing a continuously updated preview each time you save your API document.

- Launch the live preview command in your terminal

```shell
bump preview --live --open api-document.yaml
```

- Edit your `api-document.yaml` file in your favorite text editor.
- Watch the live preview being updated each time you save your file.
- The additional `--open` flag helps to automatically open the preview URL in your browser.

> You can create as many previews as you like without being authenticated. This is a **free and unlimited service**.
{: .info}

Please check `bump preview --help` for more usage details

### The `overlay` command

The [Overlay Specification](https://spec.openapis.org/overlay/v1.0.0.html) from the OpenAPI Initiative makes it possible to modify the content of an API definition by adding a layer on top of it. That layer helps adding, removing or changing some or all of the content of the original definition. 

The `bump overlay` command takes an original API document, applies the changes from the overlay document, and outputs a modified version. No changes are made directly to the original document.

```shell
bump overlay api-document.yaml overlay.yaml
```

To redirect the output of the command to a new file you can run:

```shell
bump overlay api-document.yaml overlay.yaml > modified-api-document.yaml
```

You can also apply the overlay using the [`deploy` command](#the-deploy-command) with the `--overlay` flag:

```shell
bump deploy api-document.yaml --doc my-doc --token my-token --overlay overlay.yaml
```

If there are multiple overlays which need to be applied, the `--overlay` can be passed multiple times.

```shell
bump deploy api-document.yaml \
  --doc my-doc \
  --token my-token \
  --overlay overlay1.yaml \
  --overlay overlay2.yaml
```

## Development

Make sure to have Node.js (At least v20) installed on your machine.

- Install node dependencies with

  ```shell
  npm install
  ```

- Compile the Typescript code


  ```shell
  npm run build
  npm run clean # Remove build artifacts
  ```

- Format the codebase to comply with the linter rules

  ```shell
  npm run fmt
  ```

- Run the test suites

  ```shell
  npm run test
  npm run test-coverage # Run tests with coverage
  ```

### Use package in local environment

You can run the package by executing the file `bin/run.js` locally:

  ```shell
  bin/run.js
  ```

For example to generate a preview:

  ```shell
  ./bin/run.js preview path/to/file.json
  > Your preview is visible at: https://bump.sh/preview/42
  ```

Please note that even if CLI is running locally, by default requests are sent to [Bump.sh API](https://developers.bump.sh/).

If you have a local version of the Bump.sh API, you can run CLI 100% in local environment
by setting the environment variable `BUMP_HOST`:

  ```shell
  BUMP_HOST="http://localhost:3000" ./bin/run.js preview path/to/file.json
  > Your preview is visible at: http://localhost:3000/preview/42
  ```

## License

The Bump CLI project is released under the [MIT License](http://opensource.org/licenses/MIT).

## Contributing

Bug reports and pull requests are welcome on GitHub at <https://github.com/bump-sh/cli>. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.

## Code of Conduct

Everyone interacting in the Bump-CLI project codebases, issue trackers, chat rooms, and mailing lists is expected to follow the [code of conduct](https://github.com/bump-sh/.github/blob/main/CODE_OF_CONDUCT.md).

## Thanks

- [Lorna Mitchel](https://github.com/lornajane/) for [openapi-overlay-js](https://github.com/lornajane/openapi-overlays-js).
- [Rico](https://github.com/rstacruz) for transferring the ownership of the `bump-cli` package name.
 readmeEtag: '"1c739bbc92d4101df4f479422f70a80c197c0690"' readmeLastModified: Thu, 27 Nov 2025 10:50:29 GMT repositoryId: 353317141 description: Bump.sh CLI - Deploy your OpenAPI & AsyncAPI documentations from your CI created: '2021-03-31T10:29:18Z' updated: '2026-01-30T10:25:46Z' language: TypeScript archived: false stars: 64 watchers: 4 forks: 8 owner: bump-sh logo: https://avatars.githubusercontent.com/u/33217836?v=4 license: MIT repoEtag: '"82df7bbd260100ef2913755b51afd7160484b9be46638eebbc90d6a75196cb85"' repoLastModified: Fri, 30 Jan 2026 10:25:46 GMT foundInMaster: true category: - Documentation - Server Implementations id: 591ed1a3d5bb0d3944aa1ab23f82edfa name: Bump.sh homepage: https://github.com/bump-sh/cli language: SaaS source_description: > Bump.sh generates elegant documentation and changelogs from your OpenAPI specifications. Git diff, for your API. Integrates with CI and Slack. link: https://bump.sh v2: true v3_1: true - source: https://openapi.tools/ name: Python OpenAPI Generator category: - Code Generators - Server Implementations link: https://github.com/wael34218/python_openapi_generator language: Python repository: https://github.com/wael34218/python_openapi_generator source_description: This library facilitates creating OpenAPI document for Python projects. v2: false v3: true repositoryMetadata: base64Readme: >- IyBQeXRob24gT3BlbkFQSSAoU3dhZ2dlcikgR2VuZXJhdG9yClRoaXMgbGlicmFyeSBmYWNpbGl0YXRlcyBjcmVhdGluZyBPcGVuQVBJIChTd2FnZ2VyKSBkb2N1bWVudCBmb3IgUHl0aG9uIHByb2plY3RzLgoKSSB3b3JrZWQgZm9yIGEgbG9uZyB0aW1lIGluIGJ1aWxkaW5nIEFQSXMgYW5kIHNlcnZpY2VzLCBhbmQgSSBjYW4gdGVsbCB5b3UgdGhhdCBvbmUgdGhpbmcgSSBnb3Qgc2ljayBvZiBoZWFyaW5nIHdhcyAiV2h5IHlvdXIgQVBJcyBhcmUgbm90IGRvY3VtZW50ZWQ/Ii4KSSBzZWFyY2hlZCBsb25nIGFuZCBoYXJkIGZvciBhIGxpYnJhcnkgdGhhdCBnZW5lcmF0ZXMgZG9jdW1lbnRhdGlvbiB3aXRob3V0IG1lIGdvaW5nIGJhY2sgYW5kIGZvcnRoIGVkaXRpbmcgeWFtbC9qc29uL21hcmtkb3duIGZpbGVzIC4uLiBpdHMgZW5vdWdoIHRoYXQgSSB3b3JyeSBhYm91dCB3cml0dGluZyB1bml0IGFuZCBpbnRlZ3JhdGlvbiB0ZXN0cy4KQWZ0ZXIgbG9uZyBzZWFyY2ggSSBmb3VuZCAwIGxpYnJhcmllcyB0aGF0IGRvZXMgdGhhdC4KSSB3YXMgd29uZGVyaW5nIHdoeSBub2JvZHkgaGFzIHlldCBleHBsb2l0ZWQgdGhpcyBnYXAgaW4gYSB2ZXJ5IGx1Y3JhdGl2ZSBtYXJrZXQsIHdvbmRlciBubyBsb25nZXIuIEhlbHAgaXMgYXQgaGFuZC4KCiMjIEluc3RhbGxhdGlvbgoKYGBgCnBpcCBpbnN0YWxsIG9wZW5hcGktZ2VuZXJhdG9yCmBgYAoKIyMgR2VuZXJhdGUgeW91ciBBUEkgdXNpbmcgc2ltcGxlIHNjcmlwdAoKYGBgCmZyb20gb3BlbmFwaV9nZW5lcmF0b3IgaW1wb3J0IE9wZW5hcGlHZW5lcmF0b3IKaW1wb3J0IHJlcXVlc3RzCgojIFN0ZXAgMTogQ3JlYXRlIGFuIGluc3RhbmNlIG9mIHRoZSBnZW5lcmF0b3IKZ2VuID0gT3BlbmFwaUdlbmVyYXRvcigiVGl0bGUiLCAiVGhpcyBpcyBhIHRlc3RpbmcgZGVzY3JpcHRpb24iLCAidjAuMC4xIiwgc2VydmVyPSJodHRwczovL3N3YXBpLmNvLyIpCgojIFN0ZXAgMjogQWRkIGFsbCByZXNwb25zZXMKcmVzcG9uc2UgPSByZXF1ZXN0cy5nZXQoImh0dHBzOi8vc3dhcGkuY28vYXBpL3BsYW5ldHMvIiwgcGFyYW1zPXsicGFnZSI6IDJ9KQpnZW4uYWRkX3Jlc3BvbnNlKHJlc3BvbnNlLCBkZXNjcmlwdGlvbj0iUmV0cmlldmluZyBwbGFuZXRzIikKCnJlc3BvbnNlID0gcmVxdWVzdHMuZ2V0KCJodHRwczovL3N3YXBpLmNvL2FwaS9wZW9wbGUvIiwgcGFyYW1zPXsicGFnZSI6IDN9LCBkZXNjcmlwdGlvbj0iUmV0cmlldmluZyBwZW9wbGUiKQpnZW4uYWRkX3Jlc3BvbnNlKHJlc3BvbnNlLCBkZXNjcmlwdGlvbj0iUmV0cmlldmluZyBwZW9wbGUiKQoKIyBTdGVwIDM6RXhwb3J0Cmdlbi5leHBvcnQoImV4YW1wbGUueW1sIiwgZXh0ZW5zaW9uPSJ5YW1sIikKIyBBbHNvIGNhbiBzZXQgdGhlIGV4dGVuc2lvbiB0byBganNvbmAKCiMgU3RlcCA0OiBQcm9maXQKYGBgCg== readmeEtag: '"d48e945e4ce335a8b31764e46ccf2b60bd829eb7"' readmeLastModified: Thu, 07 Nov 2019 03:53:13 GMT repositoryId: 173452649 description: >- This library facilitates creating OpenAPI (Swagger) document for Python projects. created: '2019-03-02T13:41:09Z' updated: '2025-10-06T09:27:02Z' language: Python archived: false stars: 12 watchers: 2 forks: 2 owner: wael34218 logo: https://avatars.githubusercontent.com/u/334876?v=4 license: MIT repoEtag: '"fe0641029e16019306a348b1546a71c41fadc1a002acc0dd60ee8d2e103edaa9"' repoLastModified: Mon, 06 Oct 2025 09:27:02 GMT foundInMaster: true id: b7a4b7aa58bd888b38858294394928aa - source: https://openapi.tools/ name: Fix My OpenAPI - A VSCode Extension by APIMatic category: - Text Editors - Description Validators language: Any link: >- https://marketplace.visualstudio.com/items?itemName=apimatic-developers.apimatic-for-vscode source_description: > APIMatic's VSCode extension helps you validate, lint, and auto-fix your OpenAPI files, along with other capabilities such as API transformations, API audit report generation, SDKs generation, and more. v2: true v3: true v3_1: true id: 587a6f1da562ea627d000b675af8c3d1 foundInMaster: true - source: https://openapi.tools/ name: - VSCode OpenAPI Editor - 42Crunch IntelliJ OpenAPI Editor category: - Text Editors - Server Implementations language: Any link: - >- https://marketplace.visualstudio.com/items?itemName=42Crunch.vscode-openapi - https://plugins.jetbrains.com/plugin/14837-openapi-swagger-editor repository: https://github.com/42crunch/vscode-openapi source_description: > OpenAPI extension for Visual Studio Code - new file templates, navigation, intellisense, code snippets. v2: true v3: true repositoryMetadata: base64Readme: >- # OpenAPI Editing with API Quality, Conformance and Security Testing

Manually create, import, navigate and edit OpenAPI contracts, along with using features such as Preview, functional "Try it" testing and IntelliSense. The plugin provides rich support for the OpenAPI Specification (OAS) / Swagger specification. The OpenAPI editor is free and does not require registration or have usage limits.

Additional 42Crunch tools are available in the OpenAPI Editor. These API security testing tools are provided primarily for our enterprise customers. We have also made these tools available on a limited free basis to all our OpenAPI editor users (freemium registration is required). The tools are as follows:

- [API Contract Generator](#api-contract-generator) - Automatically generate OpenAPI contracts from your existing Postman Collections and HAR files
- [API Contract Audit](#activating-api-audit) - Check the quality, conformance and security of your OpenAPI contracts
- [API Scan](#activating-api-scan) - Scan your API for security vulnerabilities

## OpenAPI Editing

The plugin supports code navigation, linting, SwaggerUI or ReDoc preview, IntelliSense, schema enforcement and generation, schema definition links and snippets.
It also supports both OpenAPI v2 and v3.0.x in JSON or YAML format. [QuickStart](#quick-start)

- [Creating OpenAPI files](#creating-openapi-files)
- [Navigating an API definition](#navigating-an-api-definition)
- [Add new elements](#add-new-elements-in-the-openapi-explorer)
- [Use IntelliSense](#use-intellisense)
- [Jump to a reference](#jump-to-a-reference)
- [Sort entries in the navigation pane](#sort-entries-in-the-navigation-pane)
- [Preview OpenAPI documentation](#preview-openapi-documentation)
- [Execute operations with "Try it"](#execute-operations-with-try-it)
- [Generate JSON schemas based on the response content](#generate-json-schemas-based-on-the-response-content)
- [Configure authentication for external references in OpenAPI files](#configure-authentication-for-external-references-in-openapi-files)
- Split your OpenAPI into multiple files linked via $ref
- Use quick fixes to automatically resolve problems in your OpenAPI
- Use Code snippets to add paths, operations, components, security

## API Contract Generator

Designed to reduce time spent by developers on repetitive manual tasks and improve the quality of OpenAPI documentation, this feature enables users to automatically generate OpenAPI contracts from your existing Postman Collections and HAR files (recorded web traffic to the API).

- Allows users to automatically generate API specifications from scratch
- Streamlines the API design process
- Helps maintain consistency across API definitions

To access the API Contract Generator, click on the 42C icon in the 42Crunch OpenAPI (Swagger) Editor in the IDE.

![OpenAPI Explorer](./images/api-contract-generator.png)

## Activating API Audit

A static analysis that lets you check the quality, conformance (to the OpenAPI specification) and security of your API definition. [Video explainer](https://vimeo.com/873235173)

- [Running an audit](#running-an-audit)
- [Navigating the issues in the audit report](#navigating-the-issues-in-the-audit-report)
  - [Priority issues](#priority-issues)
  - [Full issue list](#full-issue-list)
- [Issues details](#issues-details)
- [Fixing issues](#fixing-issues)

## Activating API Scan

A dynamic conformance and security tool that tests the API for conformance to the API definition and security vulnerabilities. The free version of API Scan runs locally in your own environment and does not require your API to be uploaded to the 42Crunch platform.
[Video explainer](https://vimeo.com/884864394/b167153ec4)

- [Conformance to the API Definition](#new-dynamic-api-security-testing)
- [Launching the Scan](#launching-42crunch-scan)

The free version of API Scan runs locally in your own environment and requires no API file to be uploaded.

## Freemium

This service lets users of our OpenAPI editor extension, who are not customers, run API Contract Generator, API Audit and API Scan.

- [Getting a Freemium Token](#getting-a-freemium-token)

## Support and Documentation:

We’ve recently launched our developer community where you’ll be able to help, get tips-n-tricks and keep up to speed with all the latest developments: https://developers.42crunch.com/

## Quick start

After installing the plugin, open any JSON or YAML file that contains an OpenAPI definition. The plugin automatically detects that this is an OpenAPI file, and the **/API** button is shown in the left-hand panel.

![OpenAPI Explorer](./images/OpenAPI%20Explorer.PNG)

We also encourage you to watch [this video](https://42crunch.com/tutorial-openapi-swagger-extension-vs-code/#Introducing-OpenAPI-Editor) that gives you a full tour of the editor and its different features.

## Editor features

This extension makes it easier and faster to navigate your OpenAPI definitions, especially when they get longer.

You can home in on elements in the OpenAPI explorer view, or jump directly to the target of a reference in the API. You can also add new elements to your API directly in the OpenAPI explorer directly where they are needed. Filling in the details is quicker with IntelliSense support for OpenAPI elements.

### Creating OpenAPI files

1. Press **Ctrl+Shift+P** on Windows or Linux, or **Cmd+Shift+P** on a Mac.
2. In the command prompt, start typing `new openapi`, and click the corresponding command to create either an OAS v2 or v3 template file.
3. Use the OpenAPI explorer to populate the template with new paths and other elements as needed.
4. Save the file to your disk to fully enable IntelliSense.

![Create new OpenAPI file from a template](./images/New%20OpenAPI%20file.gif?raw=true)

![](https://img.shields.io/badge/Learning%20More!-red) Watch this [video](https://42crunch.com/tutorial-openapi-swagger-extension-vs-code/#Navigating-OpenAPI-Explorer) on editor basics.

### Navigating an API definition

1. Open an OpenAPI file.
2. Click the OpenAPI button to switch to the OpenAPI explorer view.
3. Expand the sections and elements in the file as needed, and click the ones you want to jump to in the editor.

![Navigation inside the OpenAPI file](./images/Navigation.gif?raw=true)

### Add new elements in the OpenAPI explorer

1. In OpenAPI explorer pane, go to the section where you want to add a new element and right-click on the relevant node to open a context menu.
2. Click the item you want to add from the list.

![Add new API path and verb](./images/Add%20paths%20and%20verbs.gif?raw=true)

### Use IntelliSense

As you start typing OpenAPI elements or their values, the context-sensitive list of available options is displayed in the IntelliSense menu. In JSON OpenAPI files, just type double-quote (`"`) to show the menu, and type further to filter the list. In YAML OpenAPI files, start typing the property name.

You can also use the corresponding VS Code hotkey (**Ctrl+Space** on Windows, **Cmd+Space** on Mac) to open the IntelliSense menu.

![IntelliSense for OpenAPI editing](https://github.com/42Crunch/vscode-openapi/blob/master/images/Intellisense.gif?raw=true)

### Jump to a reference

Use Go to Definition to locate the targets of references easily. To jump to view the definition from a reference in your API, either **Ctrl+click** a reference, or right-click a reference and click **Go to Definition** in the shortcut menu.

![Go to definition](./images/Go%20to%20Definition.gif?raw=true)

### Sort entries in the navigation pane

By default, entries in the OpenAPI Explorer pane are sorted alphabetically. If you want to instead have them sorted in the order they are in the OpenAPI file, change the corresponding setting:

1. On the **File** menu, click **Preferences > Settings**.
2. Expand the **Extensions** section and click **OpenAPI**.
3. Clear the checkbox **Alphabetically sort contents of OpenAPI explorer outlines**.

![](./images/OutlineSettings.png)

### Preview OpenAPI documentation

You can get a documentation-style preview of the API you are editing by clicking the Preview button <img src="./images/Preview_button.png" width=19 /> at the top right:

![OpenAPI Preview Pane](./images/OASPreview.png)

The extension supports two popular OpenAPI documentation generators: [SwaggerUI](https://swagger.io/tools/swagger-ui/) and [ReDoc](https://github.com/Redocly/redoc).

To change the default OpenAPI Preview rendering engine:

1. On the **File** menu, click **Preferences > Settings**.
2. Expand the **Extensions** section and click **OpenAPI**.
3. Pick the option of your choice from the **Default Preview Rendered** dropdown list.

![OpenAPI Preview Pane](./images/Change_OpenAPI_Preview_engine.png)

### Execute operations with "Try it"

With "Try it", you can invoke operations defined in your OpenAPI directly from VS Code:

- Click on "Try it" code lens which is displayed right below the HTTP verb (e.g. "get", "post") of your operation and in the examples section.
- The payload data will be generated based on the request JSON Schema, or the first example available. You can edit this information before sending the request.

![TryIt view](./images/tryit.png)

Try it comes with a number of limitations:

- Sending files is not supported
- It works best with text-based responses, such as JSON
- Binary/image responses will be shown as text

### Generate JSON schemas based on the response content

"Try it" can be used to generate JSON Schema based on the body of the response.

- Select "Tools" tab in the TryIt response
- Click "Generate schema" button.

![TryIt response tools](images/tryit-schema.png)

### Configure authentication for external references in OpenAPI files

If you use references to schemas served by an authenticated HTTP service (such as an Schema Registry service or a repository), you'll need to configure the list of approved hosts in the extension settings. To do this:

1. On the **File** menu, click **Preferences > Settings**.
2. Expand the **Extensions** section and click **OpenAPI**.
3. Locate the **Openapi: Approved Hostnames** setting, click on **Add item**, and write the hostname you need for resolving external references.
   ![Configure approved hosts and authentication](./images/Configure%20approved%20hosts.png)

In case some of the approved hosts requires authentication, you can configure it in the OpenAPI > External References section of the `42Crunch: Open Settings` command view:

![Configure approved hosts and authentication](./images/Configure%20approved%20hosts%20authentication.png)

<br>

> To open this view, go to `View > Command Palette...` in VSCode menu and type in `42Crunch`, you'll see the Open Settings command listed below:
> ![42Crunch: Open Settings](./images/42Crunch%20Open%20Settings.png)
> You can also use keyboard shortcuts for the Command Palette **Ctrl+Shift+P**, or **Cmd+Shift+P** for Mac users.

<br>

After configuring all hosts you need to refer to, all OpenAPI references to any of the approved hosts will be dynamically resolved when linting or previewing your API.

## Static API Security Testing

You can use this OpenAPI extension to check the quality and security of your API definition as you work on it. This feature is powered by 42Crunch [Audit](https://docs.42crunch.com/latest/content/concepts/api_contract_security_audit.htm?utm_campaign=IDE&utm_medium=referral&utm_source=vscode). 42Crunch Audit performs a static analysis of the API definition that includes more than 300 checks on best practices and potential vulnerabilities related to authentication, authorization as well as data constraints.

![](https://img.shields.io/badge/Learning%20More!-green) Watch this [video](https://42crunch.com/free-user-faq/#Free-API-Security-Testing-FAQs-00) to learn more about 42Crunch Audit.

You can run the audit service in freemium or platform mode:

- **Using our freemium centralized service**: this service is a fully featured version of the audit, but with usage limits. In this mode, OAS files are sent to the service, audited and a report is returned. This is a stateless service: **we do not keep the OpenAPI file, nor the report.**
- New! **Using the 42Crunch CLI**: in this mode, audits are performed locally (on the user's machine). This is only available to <u>freemium</u> users for now. Support for local audit/scans using a platform API token will be available soon. In this mode, **OpenAPI files and reports are kept locally**.
- **Using 42Crunch SaaS platform**: this requires an account on a 42Crunch platform, which is available to customers and to prospects evaluating our product. In this case, you need to supply your platform URL and an IDE token (which can be created from the platform home page). You can specify those settings by invoking `42Crunch: Update platform credentials` from the command palette.

![IDE-PlatformIntegration](./images/IDE-PlatformIntegration.png)

### Getting a Freemium Token

To run Security Audit from VS Code, you need a token. The first time you try to audit or scan an API, you are asked to provide your email address or an API token from the platform.

Once you supply the address, the extension requests the token to be sent to your mailbox. Paste the token you received in the prompt in VS Code, and you are all set.

Watch this short [video](https://42crunch.com/free-user-faq/#Free-API-Security-Testing-FAQs-02) which takes you through those steps.

### Running an audit

You can use OpenAPI extension to check the quality of your API as you work on it. You can run the audit directly from VS Code by clicking the Audit button in the toolbar. Alternatively, you can run an audit for an individual endpoint using the code lens.

![](./images/StartAudit.png)

### Navigating the issues in the audit report

After the audit finishes, you get the audit report directly in the VS Code view, side by side with your code. The report viewer provides handy ways to navigate the found issues, even if the report is quite long.

#### Priority issues

Look here for issues that require the most attention.

- **Most Common issues**: this list contains issues that are occuring the most, and how many times they are repeated.
- **Opportunities**: this list contains issues that, if fixed, will most contribute to raise the audit score.

Those two lists will often overlap and in certain cases be identical, but this is totally normal.

![AuditIssuesList](./images/AuditIssuesList.png)

#### Full issue list

The full issue list contains all issues found. The list can be filtered in two ways:

- **Using the security gate (SQG) toggle**: an SQG enforces enterprise compliance and hightlights issues that are offending the requirements established, such as a minimal score, issues severity or specific issues (e.g. an API key is used when only OAuth is allowed across the enterprise). When the toggle is on, only the issues affecting the SQG status are shown.

![](https://img.shields.io/badge/Note%20-blue) SQG results are not visible yet to all Freemium users.

![AuditIssuesList-Full](./images/AuditIssuesList-Full.png)

- **Using the filtering options**: you can also use filtering options to drill-down into the list such as severity, category or even a specific issue type.

#### Issues details

For each issue, you have access to full information about the issue, why it is relevant and recommendations on how to address the issue.

![](https://img.shields.io/badge/Learning%20More!-green) Watch this [video](https://42crunch.com/free-user-faq/#Free-API-Security-Testing-FAQs-2) to learn more about audit and how to navigate issues.

### Fixing issues

Many of the issues reported by 42Crunch Audit have fixes associated with them. These are code snippets that you can insert into the OpenAPI file and then customize with the appropriate value.

1. Cick some of the error lines in your OpenAPI file to bring up the QuickFix blue icon on the left of the line.
2. Click the Quickfix icon and select the fix to apply.
3. Tweak the values in the inserted code snippet as you see fit.

![Quick Fixes in VSCode](./images/Quick_Fixes_in_VSCode.gif)

## (New!) Dynamic API Security testing

42Crunch Audit performs a security analysis that does not require any live API, just the definition itself. 42Crunch Scan leverages the OpenAPI definition to:

- Test the resilience and behavior of APIs by automatically generating security tests from the APIs' OpenAPI definition. Scan reproduces the typical behavior of a hacker by injecting bad payloads, bad tokens, and using invalid HTTP verbs and paths. This helps detect vulnerabilities early in the API life cycle, especially those associated with the [OWASP API Security Top 10](https://apisecurity.io/owasp-api-security-top-10/owasp-api-security-top-10-project/).
- Validate that the implementation of the API conforms to its established contract: Scan checks all responses against the OpenAPI definition and detects unexpected responses and data leaks.

![](https://img.shields.io/badge/Learning%20More!-Blue) Watch this [video](https://42crunch.com/free-user-faq/?utm_campaign=IDE&utm_medium=referral&utm_source=vscode#Free-API-Security-Testing-FAQs-4) to learn more about 42Crunch Scan.

APIs which thoroughly enforce compliance to an established contract are far more resilient to all types of attacks.

![](https://img.shields.io/badge/Important-red) You must only use 42Crunch Scan against APIs that you own, not those of third parties.

### Launching 42Crunch Scan

We recommend you use the 42Crunch API Security Testing Binary to run scans. The alternative is to run a docker image locally. 42Crunch customers can also leverage our [scand manager](https://github.com/42Crunch/scand-manager), by deploying an API-driven scan engine on Kubernetes.

In order to run a scan, you will need :

- **A credential** : most likely, your API is using some form of authentication, like an API Key or token. You need a valid credential to provide to the scan engine.

- **The URL** when the API is deployed.

  ![](https://img.shields.io/badge/Warning-orange) We strongly recommend that you do <u>not</u> target a production system. While the tool does not try to inject malicious payloads, it is possible that the API implementation is not resilient enough to handle the tests and may crash or behave unexpectedly.

When you first launch a scan, you are presented with the scan configuration viewer. The scan configuration is generated automatically from the OpenAPI file you chose to scan.

- For each operation in the OpenAPI file, a request is created. You can test individual requests using the **Try** button top-right.

  ![](./images/ScantryIt.png)

- Requests can be arranged into testing scenarios: a scenario combines one or multiple requests, for example you need to create a resource before you can view it. The editor allows you to extract data as variables from a request execution to inject it into the next step.

  Similarly to requests, you can test an individual scenario to ensure it is built correctly before starting a scan: the scan engine will execute the scenario and if successful, launch automatically dozens of tests using the data provided in the OpenAPI file. The scan will send bad verbs, bad data types, bad data formats as well as authentication tests.

  ![](./images/ScanScenarios.png)

Once the scan has run, you are presented with a results page. The summary shows if the scan got a testing baseline by running the HappyPath test. Additional testing results are visible from the tests list. For each issue, you can easily reproduce the problem using a curl request.

![](./images/ScanReport.gif)

## Miscellaneous commands

- You can load results of Security Audit from a file for a currently open OpenAPI file by running command `42Crunch: Load Security Audit report from file` from the command palette

## Network requirements

To execute the 42Crunch Freemium services, you need access to the following URL: https://stateless.42crunch.com. You may need to ask your administrators to add a firewall rule to allow the connection.

## Known issues

- For new files, IntelliSense does not work until you save the file. File extension must be `.json` or `.yaml`.
- When running 42Crunch Scan on a host with incorrect clock, the scan can fail with the message `cannot send the scan report : rpc error: code = InvalidArgument desc = invalid input`, to resolve the issue make sure your clock is correctly set.

## Feedback

Submit your bug reports at [GitHub project Issues](https://github.com/42Crunch/vscode-openapi/issues).

And, needless to say, your reviews at [VS Code marketplace](https://marketplace.visualstudio.com/items?itemName=42Crunch.vscode-openapi&ssr=false#review-details) mean the world to us!
 readmeEtag: '"68d41a1a15ca8b0744b9e7e36f4ff1cad7d574ee"' readmeLastModified: Thu, 15 Jan 2026 21:56:46 GMT repositoryId: 190430086 description: VisualStudio Code OpenAPI tools created: '2019-06-05T16:27:53Z' updated: '2026-02-04T13:36:04Z' language: TypeScript archived: false stars: 364 watchers: 11 forks: 43 owner: 42Crunch logo: https://avatars.githubusercontent.com/u/25365245?v=4 license: AGPL-3.0 repoEtag: '"110126c6634d2d248342976839ba0bff2dd9844383615367059927416475a848"' repoLastModified: Wed, 04 Feb 2026 13:36:04 GMT foundInMaster: true id: 782ecbb10f1ff8ac9685e0ed24e52e58 v3_1: false - source: - https://openapi.tools/ - openapi3 tags name: KaiZen-OpenAPI-Editor homepage: https://github.com/RepreZen/KaiZen-OpenAPI-Editor language: Java source_description: > Full-featured Eclipse editor for OpenAPI, also available on Eclipse Marketplace. category: - Text Editors - Parsers license: EPL repository: https://github.com/reprezen/kaizen-openapi-editor v2: true v3: true repositoryMetadata: base64Readme: >- <img src="https://cdn2.hubspot.net/hubfs/597611/Assets_Swagger/KZOE.png" alt="KaiZen OpenAPI Editor Logo" height="50%" width="50%"/>

# KaiZen OpenAPI Editor for Eclipse

_KaiZen OpenAPI Editor_ is an Eclipse editor for the [industry standard API description language](http://openapis.org), formerly known as [Swagger](http://swagger.io). It now supports both [Swagger-OpenAPI version 2.0](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md) and [OpenAPI version 3.0](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md).  

KaiZen Editor is a core component of [RepreZen API Studio](http://reprezen.com/swagger-tools), a comprehensive solution for API modeling, documentation, visualization, testing and code generation, built on Eclipse.

We welcome your suggestions and contributions!

## Eclipse Installer

KaiZen OpenAPI Editor is available on [Eclipse Marketplace](https://marketplace.eclipse.org/content/kaizen-openapi-editor). Drag-and-drop this button into Eclipse Oxygen or later to install, or [see below](#installing-kaizen-openapi-editor) for other options:

[![Drag to your running Eclipse workspace.](./etc/img/btn-install.png)](http://marketplace.eclipse.org/marketplace-client-intro?mpc_install=3429028 "Drag to your running Eclipse workspace.")

## NEW! OpenAPI 3.0 Editing

KaiZen OpenAPI Editor now features full support for the [OpenAPI version 3.0](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md) specification. [See here for further details.](https://github.com/RepreZen/KaiZen-OpenAPI-Editor/blob/master/OPEN_API_V3_SUPPORT.md)
 
## Feature Highlights

<img src="/etc/img/ContentAssistQuickOutline.png" alt="Drawing" width="400" />

### Validation
<img src="http://i.imgur.com/GrFw9EM.png" alt="Validation_screenshot" width="400">

### Code Assist
Code templates:  
<img src="http://i.imgur.com/ZtHJX6A.gif" alt="Code_template" width="400">

Keywords and values:  
<img src="http://i.imgur.com/3uZ5bQa.gif" alt="CodeAssist_keys_and_values" width="400">

### Code Assist for References
Code assist for references has several scopes which can be viewed in sequence by pressing `Ctrl`+`Space` repeatedly:

* The first scope shows only elements from the current document.
* The second expands it to elements from the containing project.
* The third shows elements from the entire workspace.

<img src="http://i.imgur.com/P0IWIEt.gif" alt="CodeAssist_for_references" width="400">

Pressing the hotkey a fourth time starts the cycle over again, with document scope.

### Navigation to a Reference
You can navigate to a reference using `Ctrl`+`Click`:  
<img src="http://i.imgur.com/7WpuV3K.gif" alt="Navigation_to_references" width="400">

### Quick Outline
Quick Outline can be invoked with `Ctrl`+`o`. Similar to code assist for references, it has three scopes: model, project, and workspace. It also allows filtering:    
<img src="http://i.imgur.com/jvcoooa.gif" alt="Navigation_to_references" width="400">

### Outline
Outline View shows the contents of the active OpenAPI spec:  
<img src="http://i.imgur.com/iv49CLn.png" alt="Navigation_to_references" width="400">

## Installing KaiZen OpenAPI Editor
KaiZen OpenAPI Editor requires Java 8 (64-bit) and Eclipse Oxygen or later.

### Installing from Eclipse Marketplace
The [Eclipse Marketplace solution](https://marketplace.eclipse.org/content/kaizen-openapi-editor) is the easiest way to install KaiZen Editor into an Eclipse IDE. You can drag-and-drop the Install button from the browser into your Eclipse IDE, or use the built-in Eclipse Marketplace Client.

[![Drag to your running Eclipse workspace.](./etc/img/btn-install.png)](http://marketplace.eclipse.org/marketplace-client-intro?mpc_install=3429028 "Drag to your running Eclipse workspace.")

### Installing from the Update Site 
You can install KaiZen OpenAPI Editor into your Eclipse IDE by clicking `Help > Install New Software... > Add...`
This will show a dialog box where you can select the location of the update site.
Use the update site http://products.reprezen.com/swagedit/latest/ as the URL.

### Installing RepreZen API Studio
KaiZen Editor is included as a core component of RepreZen API Studio, which adds live documentation and diagram views, sandbox testing with the built-in mock service and Swagger-UI, powerful code generation, and other features.

RepreZen API Studio is available through two different installation options:  
* [API Studio Desktop](http://www.reprezen.com/OpenAPI) installs as a standalone desktop application on Windows, MacOS or Linux.
* [API Studio for Eclipse](https://marketplace.eclipse.org/content/reprezen-api-studio) installs into an Eclipse IDE distribution, allowing you to use RepreZen's API design, documentation and development features in your primary Eclipse development environment.<br/>
[![Drag to your running Eclipse workspace.](./etc/img/btn-install.png)](http://marketplace.eclipse.org/marketplace-client-intro?mpc_install=3138718 "Drag to your running Eclipse workspace.")

## Troubleshooting
See the [Troubleshooting Guide](https://github.com/RepreZen/SwagEdit/blob/master/TROUBLESHOOTING.md) for solutions to common problems.

## Contributing to KaiZen OpenAPI Editor
We welcome contributions - documentation, bug reports or bug fixes.
If you are interested in contributing to KaiZen Editor, please see the [Developer's Guide](https://github.com/RepreZen/SwagEdit/blob/master/DEVELOPERS_GUIDE.md). 

We also created a list of [good first bugs](https://github.com/RepreZen/SwagEdit/labels/Good%20First%20Bug)
that are relatively easy to fix.

## License
KaiZen OpenAPI Editor is provided under the [Eclipse Public License v1.0](https://www.eclipse.org/legal/epl-v10.html)

## Video: KaiZen Editor in RepreZen API Studio

[![Editing Swagger-OpenAPI in RepreZen API Studio](http://img.youtube.com/vi/KX_tHp_KQkE/0.jpg)](https://www.youtube.com/watch?v=KX_tHp_KQkE)

_**Note:** KaiZen Editor includes code assist, real-time validation, syntax highlighting, and outline view.<br/>
[Eclipse Color Theme](https://marketplace.eclipse.org/content/eclipse-color-theme) and [EditBox](http://marketplace.eclipse.org/content/nodeclipse-editbox-background-colors-themes-highlight-code-blocks-c-java-javascript-python) are available as separate plugins.<br/>
[RepreZen API Studio](http://reprezen.com/swagger-tools) includes the mock service, live Swagger-UI, advanced code generation, and other features that are not part of KaiZen Editor. See the [feature comparison here](https://support.reprezen.com/support/solutions/articles/24000046272-what-s-the-difference-between-kaizen-openapi-editor-and-reprezen-api-studio-)._
 readmeEtag: '"95dde4753e7016fdba087e97e0e0731224c0a021"' readmeLastModified: Tue, 17 Sep 2019 12:33:47 GMT repositoryId: 43460338 description: Eclipse Editor for the Swagger-OpenAPI Description Language created: '2015-09-30T21:30:07Z' updated: '2026-02-04T21:19:46Z' language: Java archived: false stars: 120 watchers: 15 forks: 13 owner: RepreZen logo: https://avatars.githubusercontent.com/u/15205934?v=4 license: EPL-1.0 repoEtag: '"4595c2049f8c064491637bca3a4e8872f77c8e1b6c4f5d4c0767c8dd353dbf06"' repoLastModified: Wed, 04 Feb 2026 21:19:46 GMT foundInMaster: true id: 3927b9da31ce0c8291351fc07cace0ae - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags name: Swagger Editor homepage: https://github.com/swagger-api/swagger-editor language: Node.js source_description: > Design, describe, and document your API on the first open source editor fully dedicated to OpenAPI-based APIs. category: - Text Editors - Server Implementations repository: https://github.com/swagger-api/swagger-editor v2: true v3: true repositoryMetadata: base64Readme: >- # SwaggerEditor

## Table of Contents

- [Anonymized analytics](#anonymized-analytics)
- [Getting started](#getting-started)
  - [Prerequisites](#prerequisites)
  - [Installation](#installation)
  - [Usage](#usage)
- [Development](#development)
  - [Prerequisites](#prerequisites)
  - [Setting up](#setting-up)
  - [npm scripts](#npm-scripts)
  - [Build artifacts](#build-artifacts)
  - [Package mapping](#package-mapping)
- [Documentation](#documentation)
- [Docker](#docker)
- [License](#license)
- [Software Bill Of Materials (SBOM)](#software-bill-of-materials-sbom)

## Anonymized analytics

Swagger Editor uses [Scarf](https://scarf.sh/) to collect [anonymized installation analytics](https://github.com/scarf-sh/scarf-js?tab=readme-ov-file#as-a-user-of-a-package-using-scarf-js-what-information-does-scarf-js-send-about-me). These analytics help support the maintainers of this library and ONLY run during installation. To [opt out](https://github.com/scarf-sh/scarf-js?tab=readme-ov-file#as-a-user-of-a-package-using-scarf-js-how-can-i-opt-out-of-analytics), you can set the `scarfSettings.enabled` field to `false` in your project's `package.json`:

```
// package.json
{
  // ...
  "scarfSettings": {
    "enabled": false
  }
  // ...
}
```

Alternatively, you can set the environment variable `SCARF_ANALYTICS` to `false` as part of the environment that installs your npm packages, e.g., `SCARF_ANALYTICS=false npm install`.

## Getting started

### Prerequisites

These prerequisites are required both for installing SwaggerEditor as a npm package and local development setup.

- [node-gyp](https://www.npmjs.com/package/node-gyp) with [Python 3.x](https://www.python.org/downloads/)
- [GLIBC](https://www.gnu.org/software/libc/) `>=2.29`
- [emscripten](https://emscripten.org/docs/getting_started/downloads.html) or [docker](https://www.docker.com/) needs to be installed, we recommend going with a docker option


### Installation

Assuming [prerequisites](#prerequisites) are already installed, SwaggerEditor npm package is installable and works with `Node.js >= 12.22.0`.
You can install SwaggerEditor via [npm CLI](https://docs.npmjs.com/cli) by running the following command:

```sh
 $ npm install swagger-editor@alpha
````

> NOTE: when using bundler to build your project which is using swagger-editor@5 npm package,
you might run into following Node.js error: `Reached heap limit Allocation failed - JavaScript heap out of memory`.
It's caused by significant amount of code that needs to be bundled. This error can be resolved
by extending the Node.js max heap limit: `export NODE_OPTIONS="--max_old_space_size=4096"`.

### Usage

Use the package in your application:

**index.js**:

```js
import React from 'react';
import ReactDOM from 'react-dom';
import SwaggerEditor from 'swagger-editor';
import 'swagger-editor/swagger-editor.css';

const url = "https://raw.githubusercontent.com/asyncapi/spec/v2.2.0/examples/streetlights-kafka.yml";

const MyApp = () => (
  <div>
    <h1>SwaggerEditor Integration</h1>
    <SwaggerEditor url={url} />
  </div>
);

self.MonacoEnvironment = {
  /**
   * We're building into the dist/ folder. When application starts on
   * URL=https://example.com then SwaggerEditor will look for
   * `apidom.worker.js` on https://example.com/dist/apidom.worker.js and
   * `editor.worker` on https://example.com/dist/editor.worker.js.
   */
  baseUrl: `${document.baseURI || location.href}dist/`,
}

ReactDOM.render(<MyApp />, document.getElementById('swagger-editor'));
```

**webpack.config.js** (webpack@5)

Install dependencies needed for webpack@5 to properly build SwaggerEditor.

```sh
 $ npm i stream-browserify --save-dev
 $ npm i https-browserify --save-dev
 $ npm i stream-http --save-dev
 $ npm i util --save-dev
 $ npm i buffer --save-dev
 $ npm i process --save-dev
```

```js
const path = require('path');
const webpack = require('webpack');

module.exports = {
  mode: 'production',
  entry: {
    app: './index.js',
    'apidom.worker': 'swagger-editor/apidom.worker',
    'editor.worker': 'swagger-editor/editor.worker',
  },
  output: {
    globalObject: 'self',
    filename: '[name].js',
    path: path.resolve(__dirname, 'dist')
  },
  resolve: {
    fallback: {
      path: false,
      fs: false,
      http: require.resolve('stream-http'), // required for asyncapi parser
      https: require.resolve('https-browserify'), // required for asyncapi parser
      stream: require.resolve('stream-browserify'),
      util: require.resolve('util'),
      url: require.resolve('url'),
      buffer: require.resolve('buffer'),
      zlib: false,
    },
    alias: {
      // This alias make sure we don't pull two different versions of monaco-editor
      'monaco-editor': '/node_modules/monaco-editor',
      // This alias makes sure we're avoiding a runtime error related to this package
      '@stoplight/ordered-object-literal$': '/node_modules/@stoplight/ordered-object-literal/src/index.mjs',
      'react/jsx-runtime.js': 'react/jsx-runtime',
    },
  },
  plugins: [
    new webpack.ProvidePlugin({
      Buffer: ['buffer', 'Buffer'],
      process: ['process'],
    }),
  ],
  module: {
    rules: [
      {
        test: /\.css$/,
        use: ['style-loader', 'css-loader']
      },
      /**
       * The default way in which webpack loads wasm files won’t work in a worker,
       * so we will have to disable webpack’s default handling of wasm files and
       * then fetch the wasm file by using the file path that we get using file-loader.
       *
       * Resource: https://pspdfkit.com/blog/2020/webassembly-in-a-web-worker/
       *
       * Alternatively, WASM file can be bundled directly into JavaScript bundle as data URLs.
       * This configuration reduces the complexity of WASM file loading
       * but increases the overal bundle size:
       *
       * {
       *   test: /\.wasm$/,
       *   type: 'asset/inline',
       * }
       */
      {
        test: /\.wasm$/,
        loader: 'file-loader',
        type: 'javascript/auto', // this disables webpacks default handling of wasm
      },
    ]
  }
};
```

Alternative **webpack.config.js** (webpack@5)

We've already built Web Workers artifacts for you, and they're located inside our npm distribution
package in `dist/umd/` directory. To avoid the complexity of building the Web Worker artifacts, you can
use those artifacts directly. This setup will work both for **production** and **development** (webpack-dev-server)
and will significantly shorten your build process.

Install `copy-webpack-plugin` and other needed dependencies.

```sh
 $ npm i copy-webpack-plugin --save-dev
 $ npm i stream-browserify --save-dev
 $ npm i https-browserify --save-dev
 $ npm i stream-http --save-dev
 $ npm i util --save-dev
 $ npm i buffer --save-dev
 $ npm i process --save-dev
```

```js
const path = require('path');
const webpack = require('webpack');
const CopyWebpackPlugin = require('copy-webpack-plugin');

module.exports = {
  mode: 'production',
  entry: {
    app: './index.js',
  },
  output: {
    globalObject: 'self',
    filename: 'static/js/[name].js',
    path: path.resolve(__dirname, 'dist')
  },
  resolve: {
    fallback: {
      path: false,
      fs: false,
      http: require.resolve('stream-http'), // required for asyncapi parser
      https: require.resolve('https-browserify'), // required for asyncapi parser
      stream: require.resolve('stream-browserify'),
      util: require.resolve('util'),
      url: require.resolve('url'),
      buffer: require.resolve('buffer'),
      zlib: false,
    },
    alias: {
      // This alias make sure we don't pull two different versions of monaco-editor
      'monaco-editor': '/node_modules/monaco-editor',
      // This alias makes sure we're avoiding a runtime error related to this package
      '@stoplight/ordered-object-literal$': '/node_modules/@stoplight/ordered-object-literal/src/index.mjs',
      'react/jsx-runtime.js': 'react/jsx-runtime',
    }
  },
  plugins: [
    new webpack.ProvidePlugin({
      Buffer: ['buffer', 'Buffer'],
      process: ['process'],
    }),
    new CopyWebpackPlugin({
      patterns: [
        {
          from: 'node_modules/swagger-editor/dist/umd/apidom.worker.js',
          to: 'static/js',
        },
        {
          from: 'node_modules/swagger-editor/dist/umd/editor.worker.js',
          to: 'static/js',
        }
      ]
    }),
  ],
  module: {
    rules: [
      {
        test: /\.css$/,
        use: ['style-loader', 'css-loader']
      },
      /**
       * The default way in which webpack loads wasm files won’t work in a worker,
       * so we will have to disable webpack’s default handling of wasm files and
       * then fetch the wasm file by using the file path that we get using file-loader.
       *
       * Resource: https://pspdfkit.com/blog/2020/webassembly-in-a-web-worker/
       *
       * Alternatively, WASM file can be bundled directly into JavaScript bundle as data URLs.
       * This configuration reduces the complexity of WASM file loading
       * but increases the overal bundle size:
       *
       * {
       *   test: /\.wasm$/,
       *   type: 'asset/inline',
       * }
       */
      {
        test: /\.wasm$/,
        loader: 'file-loader',
        type: 'javascript/auto', // this disables webpacks default handling of wasm
      },
    ]
  }
};
```

## Development

### Prerequisites

Assuming [prerequisites](#prerequisites) are already installed, [Node.js](https://nodejs.org/) `>=22.11.0` and `npm >=10.9.0`
are the minimum required versions that this repo runs on, but we recommend using the latest version of Node.js@20.

### Setting up

If you use [nvm](https://github.com/nvm-sh/nvm), running following command inside this repository
will automatically pick the right Node.js version for you:

```sh
 $ nvm use
```

Run the following commands to set up the repository for local development:

```sh
 $ git clone https://github.com/swagger-api/swagger-editor.git
 $ cd swagger-editor
 $ npm i
 $ npm start
```

### npm scripts

**Lint**

```sh
 $ npm run lint
```

**Runs unit and integration tests**

```sh
 $ npm test
```

**Runs E2E Cypress tests**

Usage in **development** environment:

```sh
 $ npm run cy:dev
```

Usage in **Continuous Integration (CI)** environment:

```sh
 $ npm run cy:ci
```

**Build**

```sh
 $ npm run build
````

This script will build all the SwaggerEditor build artifacts - `app`, `esm` and `umd`.

### Build artifacts

After building artifacts, every two new directories will be created: `build/` and `dist/`.

**build/**

```sh
$ npm run build:app
$ npm run build:app:serve
```

Builds and serves standalone SwaggerEditor application and all it's assets on `http://localhost:3050/`.

**dist/esm/**

```sh
$ npm run build:bundle:esm
```

This bundle is suited for consumption by 3rd parties,
which want to use SwaggerEditor as a library in their own applications and have their own build process.

**dist/umd/**

```sh
$ npm run build:bundle:umd
```

SwaggerEditor UMD bundle exports SwaggerEditor symbol on a global object.
It's bundled with React defined as external. This allows the consumer to use his own version of React + ReactDOM and mount SwaggerEditor lazily.

```html
<!DOCTYPE html>
<html lang="en">
<head>
  <meta charset="utf-8" />
  <meta name="viewport" content="width=device-width, initial-scale=1" />
  <meta
    name="description"
    content="SwaggerEditor"
  />
  <title>SwaggerEditor</title>
  <link rel="stylesheet" href="./swagger-editor.css" />
</head>
<body>
  <div id="swagger-editor"></div>
  <script src="https://unpkg.com/react@18/umd/react.production.min.js" crossorigin></script>
  <script src="https://unpkg.com/react-dom@18/umd/react-dom.production.min.js" crossorigin></script>
  <script src="./dist/umd/swagger-editor.js"></script>
  <script>
    const props = {
      url: 'https://raw.githubusercontent.com/asyncapi/spec/v2.2.0/examples/streetlights-kafka.yml',
    };
    const element = React.createElement(SwaggerEditor, props);
    const domContainer = document.querySelector('#swagger-editor');

    ReactDOM.render(element, domContainer);
  </script>
</body>
</html>
```

**npm**

SwaggerEditor is released as `swagger-editor@5` npm package on [npmjs.com](https://npmjs.com).
Package can also be produced manually by running the following commands (assuming you're already followed [setting up](#setting-up) steps):

```sh
 $ npm run build:bundle:esm
 $ npm run build:bundle:umd
 $ npm pack
```

### Package mapping

SwaggerEditor maps its [build artifacts](#build-artifacts) in `package.json` file in the following way:

```json
"unpkg": "./dist/umd/swagger-editor.js",
"module": "./dist/esm/swagger-editor.js",
"browser": "./dist/esm/swagger-editor.js",
"jsnext:main": "./dist/esm/swagger-editor.js",
"exports": {
  "./package.json": "./package.json",
  "./swagger-editor.css": "./dist/swagger-editor.css",
  ".": {
    "browser": "./dist/esm/swagger-editor.js"
  },
  "./plugins/*": {
    "browser": "./dist/esm/plugins/*/index.js",
    "node": "./dist/esm/plugins/*/index.js"
  },
  "./presets/*": {
    "browser": "./dist/esm/presets/*/index.js"
  },
  "./apidom.worker": {
    "browser": "./dist/esm/apidom.worker.js"
  },
  "./editor.worker": {
    "browser": "./dist/esm/editor.worker.js"
  }
}
```

To learn more about these fields please refer to [webpack mainFields documentation](https://webpack.js.org/configuration/resolve/#resolvemainfields)
or to [Node.js Modules: Packages documentation](https://nodejs.org/docs/latest-v16.x/api/packages.html).

## Documentation

### Using older version of React

> [!IMPORTANT]
> By older versions we specifically refer to `React >=17 <18`.

By default [swagger-editor@5](https://www.npmjs.com/package/swagger-editor) npm package comes with latest version of [React@18](https://react.dev/blog/2022/03/29/react-v18).
It's possible to use _swagger-editor@5_ npm package with older version of React.

Let's say my application integrates with _swagger-editor@5_ npm package and uses [React@17.0.2](https://www.npmjs.com/package/react/v/17.0.2).

### npm

In order to inform `swagger-editor@5` npm package that I require it to use my React version, I need to use [npm overrides](https://docs.npmjs.com/cli/v9/configuring-npm/package-json#overrides).

```json
{
  "dependencies": {
    "react": "=17.0.2",
    "react-dom": "=17.0.2"
  },
  "overrides": {
    "swagger-editor": {
      "react": "$react",
      "react": "$react-dom",
      "react-redux": "^8"
    }
  }
}
```

> [!NOTE]
> The React and ReactDOM override are defined as a reference to the dependency. Since _react-redux@9_ only supports `React >= 18`, we need to use _react-redux@8_.

### yarn

In order to inform `swagger-editor@5` npm package that I require it to use my specific React version, I need to use [yarn resolutions](https://yarnpkg.com/cli/set/resolution).

```json
{
  "dependencies": {
    "react": "17.0.2",
    "react-dom": "17.0.2"
  },
  "resolutions": {
    "swagger-editor/react": "17.0.2",
    "swagger-editor/react-dom": "17.0.2",
    "swagger-editor/react-redux": "^8"
  }
}
```

> [!NOTE]
> The React and ReactDOM resolution cannot be defined as a reference to the dependency. Unfortunately *yarn* does not support aliasing like `$react` or `$react-dom` as *npm* does. You'll need to specify the exact versions.

### Customization

- [Plug points](./docs/customization/plug-points/README.md)

#### Syntax Highlighting Modes

SwaggerEditor supports two syntax highlighting modes for the Monaco editor:

1. **Simplified Mode (default)** - Regex-based syntax highlighting using Monaco's Monarch tokenizer
2. **ApiDOM Mode** - Semantic token highlighting provided by ApiDOM Language Service

The simplified mode is enabled by default. If you need more sophisticated semantic highlighting, you can enable ApiDOM mode.

**Using Simplified Mode (default):**

```js
import EditorMonacoLanguageApiDOMPlugin from 'swagger-editor/plugins/editor-monaco-language-apidom';

// Default behavior - uses simplified syntax highlighting
const plugins = [
  EditorMonacoLanguageApiDOMPlugin,
  // ... other plugins
];
```

**Enabling ApiDOM Mode:**

```js
import EditorMonacoLanguageApiDOMPlugin from 'swagger-editor/plugins/editor-monaco-language-apidom';

// Enable ApiDOM semantic token highlighting
const plugins = [
  EditorMonacoLanguageApiDOMPlugin({ useApiDOMSyntaxHighlighting: true }),
  // ... other plugins
];
```

**Visual Differences:**

The two modes produce different syntax highlighting appearances:

- **Simplified mode**:
  - Uses regex-based Monarch tokenizer for syntax coloring
  - Keywords, strings, numbers, and booleans each have distinct colors
  - Does not colorize bracket pairs (brackets are styled as part of the overall token)
  - Color scheme defined by theme token rules: `plain.keyword`, `plain.value.string`, `plain.value.number`, `plain.value.boolean`

- **ApiDOM mode**:
  - Uses semantic token analysis from ApiDOM Language Service
  - Provides context-aware token coloring based on specification structure
  - Enables bracket pair colorization (semantic tokens don't include bracket information, so editor's bracket colorization feature is enabled)
  - Color scheme uses ApiDOM-specific token types with more granular semantic categories

Both modes support:
- OpenAPI 2.0, 3.0, 3.1
- AsyncAPI 2.x, 3.x
- JSON and YAML syntax
- Specification extensions (x- prefixed fields)
- Inline JSON objects and arrays

### Environment Variables

It is possible to use an environment variable to specify a local JSON/YAML file or a remote URL for SwaggerEditor to load on startup.
These environment variables will get baked in during build time into build artifacts.

Environment variables currently available:

| Variable name               |                                                Description                                                 |
|-----------------------------|:----------------------------------------------------------------------------------------------------------:|
| `REACT_APP_DEFINITION_FILE` | Specifies a local file path, and the specified file must also be present in the `/public/static` directory |
| `REACT_APP_DEFINITION_URL`  | Specifies a remote URL. This environment variable currently takes precedence over `REACT_APP_SWAGGER_FILE` |
| `REACT_APP_VERSION`         |              Specifies the version of this app. The version is read from `package.json` file.              |

Sample environment variable values can be found in `.env` file. For more information about using
environment variables, please refer to [adding Custom Environment Variables](https://create-react-app.dev/docs/adding-custom-environment-variables/)
section of Create React App documentation.

### Using preview plugins in SwaggerUI

SwaggerEditor comes with number of `preview` plugins that are responsible for rendering
the definition that's being created in the editor. These plugins include:

- **EditorPreviewAsyncAPIPlugin** - AsyncAPI specification rendering support
- **EditorPreviewAPIDesignSystemsPlugin** - API Design Systems rendering support

With a bit of adapting, we can use these plugins with SwaggerUI to provide an ability
to render AsyncAPI or API Design Systems definitions with SwaggerUI.

```js
import SwaggerUI from 'swagger-ui';
import SwaggerUIStandalonePreset from 'swagger-ui/dist/swagger-ui-standalone-preset';
import 'swagger-editor/swagger-editor.css';
import EditorContentOriginPlugin from 'swagger-editor/plugins/editor-content-origin';
import EditorContentTypePlugin from 'swagger-editor/plugins/editor-content-type';
import EditorPreviewAsyncAPIPlugin from 'swagger-editor/plugins/editor-preview-asyncapi';
import EditorPreviewAPIDesignSystemsPlugin from 'swagger-editor/plugins/editor-preview-api-design-systems';
import SwaggerUIAdapterPlugin from 'swagger-editor/plugins/swagger-ui-adapter';

SwaggerUI({
  url: 'https://petstore.swagger.io/v2/swagger.json',
  dom_id: '#swagger-ui',
  presets: [SwaggerUI.presets.apis, SwaggerUIStandalonePreset],
  plugins: [
    EditorContentOriginPlugin,
    EditorContentTypePlugin,
    EditorPreviewAsyncAPIPlugin,
    EditorPreviewAPIDesignSystemsPlugin,
    SwaggerUIAdapterPlugin,
    SwaggerUI.plugins.DownloadUrl,
  ],
});
```

The key here is `SwaggerUIAdapter` plugin which adapts SwaggerEditor plugins to use
directly with SwaggerUI.

#### Standalone mode

SwaggerUI standalone mode is supported as well. With standalone mode you'll get a `TopBar` with
an input where URL of the definition can be provided and this definition is subsequently loaded
by the SwaggerUI.

```js
import SwaggerUI from 'swagger-ui';
import SwaggerUIStandalonePreset from 'swagger-ui/dist/swagger-ui-standalone-preset';
import 'swagger-ui/dist/swagger-ui.css';
import 'swagger-editor/swagger-editor.css';
import EditorContentOriginPlugin from 'swagger-editor/plugins/editor-content-origin';
import EditorContentTypePlugin from 'swagger-editor/plugins/editor-content-type';
import EditorPreviewAsyncAPIPlugin from 'swagger-editor/plugins/editor-preview-asyncapi';
import EditorPreviewAPIDesignSystemsPlugin from 'swagger-editor/plugins/editor-preview-api-design-systems';
import SwaggerUIAdapterPlugin from 'swagger-editor/plugins/swagger-ui-adapter';

SwaggerUI({
  url: 'https://petstore.swagger.io/v2/swagger.json',
  dom_id: '#swagger-ui',
  presets: [SwaggerUI.presets.apis, SwaggerUIStandalonePreset],
  plugins: [
    EditorContentOriginPlugin,
    EditorContentTypePlugin,
    EditorPreviewAsyncAPIPlugin,
    EditorPreviewAPIDesignSystemsPlugin,
    SwaggerUIAdapterPlugin,
    SwaggerUI.plugins.DownloadUrl,
  ],
  layout: 'StandaloneLayout',
});
```

#### Utilizing preview plugins via [unpkg.com](https://unpkg.com/)

It's possible to utilize preview plugins in a build-free way via [unpkg.com](https://unpkg.com/) to create a standalone
multi-spec supporting version of SwaggerUI.

```html
<!DOCTYPE html>
<html >
  <head>
    <meta charset="utf-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1" />
    <meta name="theme-color" content="#000000" />
    <meta name="description" content="SwaggerUIMultifold" />
    <link rel="stylesheet" href="//unpkg.com/swagger-editor@5.0.0-alpha.102/dist/swagger-editor.css" />
  </head>
  <body style="margin:0; padding:0;">
    <section id="swagger-ui"></section>

    <script src="//unpkg.com/swagger-ui-dist@5.21.0/swagger-ui-bundle.js"></script>
    <script src="//unpkg.com/swagger-ui-dist@5.21.0/swagger-ui-standalone-preset.js"></script>
    <script>
      ui = SwaggerUIBundle({});
      // expose SwaggerUI React globally for SwaggerEditor to use
      window.React = ui.React;
    </script>
    <script src="//unpkg.com/swagger-editor@5.0.0-alpha.102/dist/umd/swagger-editor.js"></script>
    <script>
      SwaggerUIBundle({
        url: 'https://petstore3.swagger.io/api/v3/openapi.json',
        dom_id: '#swagger-ui',
        presets: [
          SwaggerUIBundle.presets.apis,
          SwaggerUIStandalonePreset,
        ],
        plugins: [
          SwaggerEditor.plugins.EditorContentOrigin,
          SwaggerEditor.plugins.EditorContentType,
          SwaggerEditor.plugins.EditorPreviewAsyncAPI,
          SwaggerEditor.plugins.EditorPreviewApiDesignSystems,
          SwaggerEditor.plugins.SwaggerUIAdapter,
          SwaggerUIBundle.plugins.DownloadUrl,
        ],
        layout: 'StandaloneLayout',
      });
    </script>
  </body>
</html>
```

### Composing customized SwaggerEditor version

SwaggerEditor is just a number of SwaggerUI plugins used with [swagger-ui-react](https://www.npmjs.com/package/swagger-ui-react).
Customized SwaggerEditor can be created by composing individual plugins with either [swagger-ui](https://www.npmjs.com/package/swagger-ui) and [swagger-ui-react](https://www.npmjs.com/package/swagger-ui-react).

#### Plugins

List of available plugins:

- dialogs
- dropdown-menu
- dropzone
- editor-content-fixtures
- editor-content-from-file
- editor-content-origin
- editor-content-persistence
- editor-content-read-only
- editor-content-type
- editor-monaco
- editor-monaco-language-apidom
- editor-monaco-yaml-paste
- editor-preview
- editor-preview-api-design-systems
- editor-preview-asyncapi
- editor-preview-swagger-ui
- editor-safe-render
- editor-textarea
- layout
- modals
- props-change-watcher
- splash-screen
- swagger-ui-adapter
- top-bar
- versions

Individual plugins can be imported in the following way:

```js
import EditorContentTypePlugin from 'swagger-editor/plugins/editor-content-type';
import EditorContentReadOnlyPlugin from 'swagger-editor/plugins/editor-content-read-only';
```

#### Presets

Along with plugins, presets are available as well. Preset is a collection of plugins
that are design to work together to provide a compound feature.

List of available presets:

- textarea
- monaco

Individual presets can be imported in the following way:

```js
import TextareaPreset from 'swagger-editor/presets/textarea';
import MonacoPreset from 'swagger-editor/presets/monaco';
```

> NOTE: Please refer to the [Plug points documentation](https://github.com/swagger-api/swagger-ui/blob/master/docs/customization/plug-points.md)
of SwaggerUI to understand how presets are passed to SwaggerUI.

#### Composing with swagger-ui

```js
import SwaggerUI from 'swagger-ui';
import 'swagger-ui/dist/swagger-ui.css';
import UtilPlugin from 'swagger-editor/plugins/util';
import VersionsPlugin from 'swaggereditor/plugins/versions';
import ModalsPlugin from 'swagger-editor/plugins/modals';
import DialogsPlugin from 'swagger-editor/plugins/dialogs';
import DropdownMenuPlugin from 'swagger-editor/plugins/dropdown-menu';
import DropzonePlugin from 'swagger-editor/plugins/dropzone';
import VersionsPlugin from 'swagger-editor/plugins/versions';
import EditorTextareaPlugin from 'swagger-editor/plugins/editor-textarea';
import EditorMonacoPlugin from 'swagger-editor/plugins/editor-monaco';
import EditorMonacoYamlPastePlugin from 'swagger-editor/plugins/editor-monaco-yaml-paste';
import EditorMonacoLanguageApiDOMPlugin from 'swagger-editor/plugins/editor-monaco-language-apidom';
import EditorContentReadOnlyPlugin from 'swagger-editor/plugins/editor-content-read-only';
import EditorContentOriginPlugin from 'swagger-editor/plugins/editor-content-origin';
import EditorContentTypePlugin from 'swagger-editor/plugins/editor-content-type';
import EditorContentPersistencePlugin from 'swagger-editor/plugins/editor-content-persistence';
import EditorContentFixturesPlugin from 'swagger-editor/plugins/editor-content-fixtures';
import EditorContentFromFilePlugin from 'swagger-editor/plugins/editor-content-from-file';
import EditorPreviewPlugin from 'swagger-editor/plugins/editor-preview';
import EditorPreviewSwaggerUIPlugin from 'swagger-editor/plugins/editor-preview-swagger-ui';
import EditorPreviewAsyncAPIPlugin from 'swagger-editor/plugins/editor-preview-asyncapi';
import EditorPreviewApiDesignSystemsPlugin from 'swagger-editor/plugins/editor-preview-api-design-systems';
import TopBarPlugin from 'swagger-editor/plugins/top-bar';
import SplashScreenPlugin from 'swagger-editor/plugins/splash-screen';
import LayoutPlugin from 'swagger-editor/plugins/layout';
import EditorSafeRenderPlugin from 'swagger-editor/plugins/editor-safe-render';

SwaggerUI({
  url: 'https://petstore.swagger.io/v2/swagger.json',
  dom_id: '#swagger-editor',
  plugins: [
    UtilPlugin,
    VersionsPlugin,
    ModalsPlugin,
    DialogsPlugin,
    DropdownMenuPlugin,
    DropzonePlugin,
    VersionsPlugin,
    EditorTextareaPlugin,
    EditorMonacoPlugin,
    EditorMonacoYamlPastePlugin,
    EditorMonacoLanguageApiDOMPlugin,
    EditorContentReadOnlyPlugin,
    EditorContentOriginPlugin,
    EditorContentTypePlugin,
    EditorContentPersistencePlugin,
    EditorContentFixturesPlugin,
    EditorContentFromFilePlugin,
    EditorPreviewPlugin,
    EditorPreviewSwaggerUIPlugin,
    EditorPreviewAsyncAPIPlugin,
    EditorPreviewApiDesignSystemsPlugin,
    TopBarPlugin,
    SplashScreenPlugin,
    LayoutPlugin,
    EditorSafeRenderPlugin,
  ],
  layout: 'StandaloneLayout',
});
```

#### Composing with swagger-ui-react

```js
import React from 'react';
import ReactDOM from 'react-dom';
import SwaggerUI from 'swagger-ui-react';
import 'swagger-ui-react/swagger-ui.css';
import UtilPlugin from 'swagger-editor/plugins/util';
import VersionsPlugin from 'swaggereditor/plugins/versions';
import ModalsPlugin from 'swagger-editor/plugins/modals';
import DialogsPlugin from 'swagger-editor/plugins/dialogs';
import DropdownMenuPlugin from 'swagger-editor/plugins/dropdown-menu';
import DropzonePlugin from 'swagger-editor/plugins/dropzone';
import VersionsPlugin from 'swagger-editor/plugins/versions';
import EditorTextareaPlugin from 'swagger-editor/plugins/editor-textarea';
import EditorMonacoPlugin from 'swagger-editor/plugins/editor-monaco';
import EditorMonacoYamlPastePlugin from 'swagger-editor/plugins/editor-monaco-yaml-paste';
import EditorMonacoLanguageApiDOMPlugin from 'swagger-editor/plugins/editor-monaco-language-apidom';
import EditorContentReadOnlyPlugin from 'swagger-editor/plugins/editor-content-read-only';
import EditorContentOriginPlugin from 'swagger-editor/plugins/editor-content-origin';
import EditorContentTypePlugin from 'swagger-editor/plugins/editor-content-type';
import EditorContentPersistencePlugin from 'swagger-editor/plugins/editor-content-persistence';
import EditorContentFixturesPlugin from 'swagger-editor/plugins/editor-content-fixtures';
import EditorContentFromFilePlugin from 'swagger-editor/plugins/editor-content-from-file';
import EditorPreviewPlugin from 'swagger-editor/plugins/editor-preview';
import EditorPreviewSwaggerUIPlugin from 'swagger-editor/plugins/editor-preview-swagger-ui';
import EditorPreviewAsyncAPIPlugin from 'swagger-editor/plugins/editor-preview-asyncapi';
import EditorPreviewApiDesignSystemsPlugin from 'swagger-editor/plugins/editor-preview-api-design-systems';
import TopBarPlugin from 'swagger-editor/plugins/top-bar';
import SplashScreenPlugin from 'swagger-editor/plugins/splash-screen';
import LayoutPlugin from 'swagger-editor/plugins/layout';
import EditorSafeRenderPlugin from 'swagger-editor/plugins/editor-safe-render';

const SwaggerEditor = () => {
  return (
    <SwaggerUI
      url={url}
      plugins={[
        UtilPlugin,
        VersionsPlugin,
        ModalsPlugin,
        DialogsPlugin,
        DropdownMenuPlugin,
        DropzonePlugin,
        VersionsPlugin,
        EditorTextareaPlugin,
        EditorMonacoPlugin,
        EditorMonacoYamlPastePlugin,
        EditorMonacoLanguageApiDOMPlugin,
        EditorContentReadOnlyPlugin,
        EditorContentOriginPlugin,
        EditorContentTypePlugin,
        EditorContentPersistencePlugin,
        EditorContentFixturesPlugin,
        EditorContentFromFilePlugin,
        EditorPreviewPlugin,
        EditorPreviewSwaggerUIPlugin,
        EditorPreviewAsyncAPIPlugin,
        EditorPreviewApiDesignSystemsPlugin,
        TopBarPlugin,
        SplashScreenPlugin,
        LayoutPlugin,
        EditorSafeRenderPlugin,
      ]}
      layout="StandaloneLayout"
    />
  );
};

ReactDOM.render(<SwaggerEditor />, document.getElementById('swagger-editor'));
```

## Docker

### Pre-built DockerHub image

SwaggerEditor is available as a pre-built docker image hosted on **docker.swagger.io**.

```sh
$ docker pull docker.swagger.io/swaggerapi/swagger-editor:latest
$ docker run -d -p 8080:80 docker.swagger.io/swaggerapi/swagger-editor:latest
```

### Building locally

**Privileged image**:

```sh
 $ npm run build:app
 $ docker build . -t swaggerapi/swagger-editor:latest
 $ docker run -d -p 8080:80 swaggerapi/swagger-editor:latest
```

Now open your browser at `http://localhost:8080/`.

**Unprivileged image**:

```sh
 $ npm run build:app
 $ docker build . -f Dockerfile.unprivileged -t swaggerapi/swagger-editor:latest-unprivileged
 $ docker run -d -p 8080:8080 swaggerapi/swagger-editor:latest-unprivileged
```

Now open your browser at `http://localhost:8080/`.

> **No** custom environment variables are currently supported by SwaggerEditor.

## License

SwaggerEditor is licensed under [Apache 2.0 license](https://github.com/swagger-api/swagger-editor/blob/next/LICENSES/Apache-2.0.txt).
SwaggerEditor comes with an explicit [NOTICE](https://github.com/swagger-api/apidom/blob/next/NOTICE) file
containing additional legal notifications and information.

This project uses [REUSE specification](https://reuse.software/spec/) that defines a standardized method
for declaring copyright and licensing for software projects.

## Software Bill Of Materials (SBOM)

Software Bill Of materials is available in this repository [dependency graph](https://github.com/swagger-api/swagger-editor/network/dependencies).
Click on `Export SBOM` button to download the SBOM in [SPDX format](https://spdx.dev/).
 readmeEtag: '"9c26ed7997e0cd9ee1c251b86c3cd2ba08f3eecf"' readmeLastModified: Mon, 26 Jan 2026 14:49:08 GMT repositoryId: 19390712 description: Swagger Editor created: '2014-05-02T22:47:10Z' updated: '2026-02-06T03:09:04Z' language: JavaScript archived: false stars: 9412 watchers: 249 forks: 2360 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 repoEtag: '"eb5fb52e576677c14f132df7c77cbce025014de43499cb8e7271961339eca714"' repoLastModified: Fri, 06 Feb 2026 03:09:04 GMT foundInMaster: true id: 951cf61d12c812ef9b201ad76b5d278d v3_1: true - source: https://openapi.tools/ name: SwaggerHub category: Text Editors language: - SaaS - On-Premise NodeJS repository: null link: https://swagger.io/tools/swaggerhub/ source_description: > API design and documentation platform to improve collaboration, standardize development workflow and centralize their API discovery and consumption. v2: true v3: true foundInMaster: true id: 4b7adfae5ed14133b59e6f80811da841 - source: https://openapi.tools/ name: Redocly VSCode Extension category: - Text Editors - Parsers language: Node.js link: https://marketplace.visualstudio.com/items?itemName=Redocly.openapi-vs-code repository: https://github.com/redocly/redocly-vs-code source_description: >- Redocly OpenAPI is a Visual Studio Code extension that helps you write, validate, preview, and maintain your OpenAPI documents. v2: true v3: true v3_1: true id: 491454b1f2a266a142515a7eee661b08 repositoryMetadata: base64Readme: >- # Redocly OpenAPI


Redocly OpenAPI is a Visual Studio Code extension that helps you write, validate, and maintain your OpenAPI documents. It warns about errors in OpenAPI definitions and lets you quickly access referenced schemas or open the files that contain them. The extension works with OpenAPI 2.0 and 3.0 definitions, and has basic support for OpenAPI 3.1.

**Feature highlights:**

- Validate your OpenAPI definitions 
- Quickly preview and access referenced schemas
- Work with multi-file definitions
- Preview API documentation side-by-side with your OpenAPI definition
- Access context-aware help about OpenAPI features
- Edit API definitions through interactive forms 


![Peek and go-to definition features](https://redoc.ly/images/vscode/openapi-vscode-peek.gif)

**Found an issue or have an idea for improving the extension? [Let us know!](https://github.com/Redocly/redocly-vs-code/issues/new/choose)**

**The changelog is available [here](https://marketplace.visualstudio.com/items/Redocly.openapi-vs-code/changelog).**

## Before you start

- The extension requires a `redocly.yaml` configuration file in your workspace. Read how to create it [in the Configuration section](#configuration).

- Note that the extension only works with YAML files. Validation for JSON files is supported starting with version `0.2.0` of the extension.


### Limitations

Be aware that the extension may be affected by some limitations of the VS Code editor.

So far, we have identified these limitations:

- Special characters `/`, `$` and `#` do not trigger VS Code suggestions
- VS Code doesn't suggest values while inside a snippet


## Installation

To install the extension in VS Code:

1. Press [Ctrl+Shift+X] or [⇧⌘X] to open the *Extensions* tab.

2. Type `Redocly OpenAPI` to search for the extension in Marketplace.

3. Find the extension in the list and select **Install**.

4. When the installation is complete, select **Enable** to activate the extension in your workspace.


## Configuration

To use the extension, we recommend you create a configuration file called `redocly.yaml` and place it in the root directory of your workspace. If the extension detects that this file doesn't exist, it will prompt you to create it automatically for the current workspace.

Here's an example of what your directory structure could look like:

```
.
├── workspace 
│   ├── openapi-definitions
│   │   ├── production.yaml
│   │   └── test.yaml
│   ├── redocly.yaml
│   ├── some-file.txt

```

The `redocly.yaml` configuration file defines the criteria for validating OpenAPI definitions. 

Add the following example contents to the file and save the changes:

```yaml
apis:
  apiName@version:
    root: ./openapi.yaml

extends:
  - recommended
  
rules:
  assert/operation-description:
    subject: 
      type: Operation
      property: description
    assertions:
      defined: true
      minLength: 30

theme:
  openapi:
    generateCodeSamples:
      languages:
        - lang: curl
        - lang: Node.js
        - lang: Python
```

This will let you start working with the extension. You can modify the file at any point to control the behavior of the extension. When modifying the `redocly.yaml` file, you must save it to disk for your changes to apply.

To learn more about built-in rules you can use, refer to the [Redocly OpenAPI CLI documentation](https://redoc.ly/docs/cli/built-in-rules/). The [linting guide](https://redoc.ly/docs/cli/guides/lint/) provides more information on how to configure the linter.

Note that you can add multiple paths to OpenAPI files under `apiDefinitions`, each in its own line with a custom, unique name as the key. Files listed in this section will be automatically validated by the extension when you open them in VS Code. Changes made to `apiDefinitions` are dynamically reflected in the [preview panel](#live-preview-of-the-api-documentation).


## Usage

The Redocly OpenAPI VS Code extension helps you create and update OpenAPI documents. 

To start using it, either create a new OpenAPI definition or open an existing one in your VS Code.

If this is your first time working with OpenAPI definitions, you can use our default template to get started. Create an empty YAML file and open the *Cursor context* panel. In the panel, select **Add template**. Now you have a fully functional OpenAPI definition you can use with the extension.

![Adding the default OpenAPI template](https://redoc.ly/images/vscode/redocly-vscode-default-template.gif)


### OpenAPI structure autocompletion 

To use the extension for OpenAPI authoring, create a new, empty YAML file. As you start typing `openapi` at the top of the file, the extension will automatically offer suggestions for OpenAPI definition elements [according to the specification](https://github.com/OAI/OpenAPI-Specification). 

![Autocompletion](https://redoc.ly/images/vscode/openapi-vscode-author.gif)

Select a suggestion from the list. The extension will automatically generate the fields supported by the selected OpenAPI section. You can then add your values for each of the fields.

### Value autocompletion 

Starting with version `0.2.0`, the extension supports value autocompletion for references (`$ref` fields). To use this feature, your OpenAPI document must have `components` with a proper type defined. For example, the extension will only suggest values for `$ref` inside `requestBody` if you have components defined in `requestBodies`. Similarly, it will only suggest values from `schemas` for `$ref` fields inside `schema`.

![References completion](https://redoc.ly/images/vscode/openapi-vscode-reference-completion.gif)

Autocompletion for `example` and `default` fields will suggest values you have previously defined in `enum` on the same level.

![Example/Default completion](https://redoc.ly/images/vscode/openapi-vscode-example-default-completion.gif)

The extension will also show suggested values for all other fields that have a predefined set of values according to the OpenAPI specification (like `required`, which has `boolean` type; or `type`, which must be one of `array` | `object` | `number`) and that are supported by [Redocly OpenAPI CLI].

![Value completion](https://redoc.ly/images/vscode/openapi-vscode-enum-boolean-completion.gif)

### Dynamic OpenAPI validation 

The extension continuously validates the OpenAPI definition you're working on. Depending on the rules you've added to your `redocly.yaml` file, the extension will warn you about the following issues:

- indentation
- incorrect type usage
- wrong paths to referenced files
- missing or undefined fields and sections
- and more

Here are some examples illustrating different types of issues and how Redocly OpenAPI points them out.

**Missing fields and sections**

![Redocly OpenAPI warning for the servers section](https://redoc.ly/images/vscode/openapi-vscode-missing-section.png)

![Redocly OpenAPI warning for fields in the operation object](https://redoc.ly/images/vscode/openapi-vscode-required-fields.png)


**Undefined sections**

![Redocly OpenAPI warning for undefined security scheme](https://redoc.ly/images/vscode/openapi-vscode-security-scheme.png)


**Type usage**

![Redocly OpenAPI warning about allowed types](https://redoc.ly/images/vscode/openapi-vscode-type.png)


**Wrong path to referenced file**

![Redocly OpenAPI warning about a referenced file not found](https://redoc.ly/images/vscode/openapi-vscode-incorrect-reference.png)


### Quick navigation

When you [use references](https://redoc.ly/docs/resources/ref-guide/) in your OpenAPI definition, the extension will validate them and warn you if they are incorrect (for example, if the path to a schema in a separate file is incorrect). 

Right-click on any `$ref` value to access additional navigation options: **Go to Definition** and **Peek**.

Selecting **Go to Definition** opens the referenced item in a new VS Code tab (if it's a separate file) or takes you directly to the section where it's defined (if it's in the same file).

Selecting **Peek > Peek definition** opens an inline preview of the referenced item within the current VS Code tab.


### Live preview of the API documentation

The Redocly OpenAPI VS Code extension relies on [Redocly Reference](https://redoc.ly/reference-docs) to generate a preview of the API documentation based on the OpenAPI document you're editing. This lets you see the OpenAPI source and the documentation generated from it side-by-side. When you save a change to your OpenAPI definition, the documentation in the preview panel is automatically updated.

Access the documentation preview in any of the following ways:

- Select the **Open preview** button in the upper right section of your VS Code window.

- Open the *Command Palette*, start typing `redocly`, then select *Redocly OpenAPI: Open preview*.

Both actions open the *Preview* panel.

In the panel, you must provide an API key from Redocly to use the documentation preview feature. To obtain the key, first [create a free Redocly account](https://app.redoc.ly/signup), then [generate a personal API key](https://redoc.ly/docs/settings/personal-api-keys/) for it.

When you have generated your personal API key, paste it into the *API key* in the *Preview* panel and select **Submit**. Documentation previews will be enabled for all your workspaces.

In the tab bar above the panel, select the name of the YAML file for which you want to preview the documentation. If you have defined custom names in the `apiDefinitions` section of the `redocly.yaml` file, those names will be displayed in tabs instead of file names. Selecting a name loads the API documentation preview for the API definition.

![Using the documentation preview feature](https://redoc.ly/images/vscode/openapi-vscode-live-preview.png)

To customize the documentation preview, add [supported Reference configuration options](https://redoc.ly/docs/api-reference-docs/configuration/) to the `referenceDocs` section in your `redocly.yaml` file.

For example, to hide the logo image and change the font size and color, you could add the following:

```yaml
apiDefinitions:
  main: path/to/your-openapi.yaml
  test: path/to/another-openapi.yaml
lint:
  extends:
    - recommended
referenceDocs:
  hideLogo: true
  theme:
    colors:
      text:
        primary: "#FF0000"
    typography:
      fontSize: 16px
```

Remember to save the `redocly.yaml` file for changes to apply.

To exit the documentation preview, close the *Preview* panel. You can open it again at any point.


### Interactive forms and context-aware help for OpenAPI authoring

To help you write valid, specification-compliant API definitions, the Redocly OpenAPI VS Code extension provides dynamic descriptions of OpenAPI features in the cursor context panel. 

Access the cursor context in any of the following ways:

- Select the **Open cursor context** button in the upper right section of your VS Code window.

- Open the *Command Palette*, start typing `redocly`, then select *Redocly OpenAPI: Open cursor context*.

Both actions open the *Cursor context* panel.

As you place your cursor into different sections of your OpenAPI document, the context-aware descriptions in the panel change to match the exact object or property you're editing.

![Using the cursor context panel](https://redoc.ly/images/vscode/openapi-vscode-context-explorer.png)

For supported sections, the *Cursor context* panel can also display a visual editor where you can change the contents of your API definition through interactive forms.

![Using the visual editor](https://redoc.ly/images/vscode/redocly-vscode-visual-editor.gif)

Depending on the currently selected section of the OpenAPI document, the *Cursor context* panel may show triple bar ("hamburger") icons to the left of each input field in the visual editor. Use these icons to change the order of fields in the panel. To reorder a field, select the triple bar icon next to it, then drag-and-drop it up or down in the panel.

To exit, close the *Cursor context* panel. You can open it again at any point.


## Known issues

- Interactive forms are supported only for `info`, `server`, `tags` and `externalDocs` sections in the current version.

- Autocompletion support is limited in the current version.

- The `redocly.yaml` file must be saved to disk (Ctrl+S) for changes to apply.


## Troubleshooting

If you suspect the extension is not working properly, make sure the following conditions are true:

- the extension is enabled in the current VS Code workspace (or globally)
- a non-empty `redocly.yaml` configuration file exists in the root directory of your workspace
- your API definition file is in the YAML format


## License

Installation and usage of the extension are subject to the terms and conditions of the [Redocly Subscription agreement](https://redoc.ly/subscription-agreement/).

 readmeEtag: '"29a324ffdfa0a7a8ee8923371921b405142f8647"' readmeLastModified: Wed, 17 Apr 2024 13:20:02 GMT repositoryId: 424036264 description: Redocly VS Code extension created: '2021-11-03T00:03:16Z' updated: '2026-02-01T15:50:11Z' language: null archived: false stars: 43 watchers: 12 forks: 9 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 repoEtag: '"a1b5bb5adf075b34771c71f53ef702998e0dc43a6e353036c523b7534f7a9a6e"' repoLastModified: Sun, 01 Feb 2026 15:50:11 GMT foundInMaster: true - source: https://openapi.tools/ name: Api-Fiddle category: GUI Editors language: - TypeScript - Saas link: https://www.api-fiddle.com/ source_description: > Opinionated API design platform built for collaboration and exploration. Create API designs for technical documents, specifications, and reviews. v2: false v3: false v3_1: true id: 83324f57ad5dff50e50148a8f4b6eef3 foundInMaster: true - source: https://openapi.tools/ name: Stoplight Studio category: - GUI Editors - Text Editors language: SaaS link: https://stoplight.io/studio source_description: > Stoplight Studio is a powerful GUI/text editor with linting and mocking built right in. The desktop version has been discontinued, but it runs in the browser, powered by your existing GitHub, GitLab, or BitBucket repos. v2: true v3: true v3_1: true foundInMaster: true id: 8348b06da005508392b6fa17c12d3ca4 - source: - https://openapi.tools/ - openapi3 tags name: Hackolade category: - GUI Editors - Parsers language: ReactJS link: https://hackolade.com repository: https://github.com/hackolade/openapi source_description: > A visual editor for OpenAPI v2/v3, from the pioneer in data modeling for NoSQL databases. v2: true v3: true repositoryMetadata: base64Readme: >- # OpenAPI
Plugin to enable OpenAPI 3 as a target in Hackolade data modeling.  Requires prior download of the Hackolade application from our [download page](https://hackolade.com/download.html)

This plugin is for OpenAPI 3.  If instead you need Swagger 2, you need a separate plugin accessible [here](https://github.com/hackolade/Swagger).

Starting with Hackolade Studio v6.10.0, this plugin also supports OAS 3.1.x

<span class="rvts78">OpenAPI Specification</span><span class="rvts77"> or OAS (formerly known as </span><span class="rvts78">Swagger Specification</span><span class="rvts77">) is an open-source format for describing and documenting APIs. It has become a de-facto standard,</span> <span class="rvts6">language-agnostic interface</span> <span class="rvts77">for designing and describing RESTful APIs</span> <span class="rvts6">which allow both humans and computers to discover and understand the capabilities of the service without access to source code, documentation, or through network traffic inspection.  When properly defined via OpenAPI, a consumer can understand and interact with the remote service with a minimal amount of implementation logic.</span><span class="rvts77">The latest version of OpenAPI is </span><span class="rvts78">3.0.2</span><span class="rvts77">. OpenAPI definitions can be written in JSON or YAML.</span>

<span class="rvts6">  
</span>

<span class="rvts6">OpenAPI is a</span> [formal specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md) <span class="rvts6">surrounded by a large ecosystem of tools, which includes everything from front-end user interfaces, low-level code libraries and commercial API management solutions.</span>

<span class="rvts6">  
</span>

<span class="rvts6">To perform model-first design of a REST API using OpenAPI 3 with Hackolade, you must first download the OpenAPI</span> [plugin](DownloadadditionalDBtargetplugin.html)<span class="rvts6">.  This plugin is strictly compliant with version 3.0.2 of the OpenAPI specification.  If you need support version 2 of the specification (a.k.a. Swagger) you need another plugin described</span> [here](SwaggerAPI.html)<span class="rvts6">.</span>

<span class="rvts21">  
</span>

<span class="rvts21">Creating APIs is not easy! And writing OpenAPI documentation in a design-first approach can be tedious at best, generally error-prone and frustrating...  Hackolade takes a visual schema-centric approach so you can focus on the content of requests and responses. The application also assists with all the metadata to produce validated OpenAPI files and test the transactions.  You can also reverse-engineer existing OpenAPI 3 files in JSON or YAML to produce a graphical representation of your APIs.  You may want to consult our </span>[step-by-step tutorial](https://hackolade.com/help/CreateaRESTAPImodel.html)

<span class="rvts6">Hackolade was specially adapted to support the API design of OAS, including all the necessary metadata for the API, the requests and responses.  The application closely follows the terminology of the specification.  The visual tool puts the focus on what really matters in an API: the schema of the information being exchanged between systems.  At the same time, it provides assistance to modelers and does not require perfect mastery of the OpenAPI 3 syntax.  It generates validated files that are syntactically correct and compatible with the specification thereby greatly improving productivity and quality.</span>

<span class="rvts6">  
</span>

<span class="rvts6">The diagram below results from the reverse-engineering of the</span> [Pet Store](https://mermade.org.uk/examples/openapi.json) <span class="rvts6">sample API.</span>

<img src="lib/OpenAPI Workspace.png" width="100%" height="100%">

<span class="rvts78">  
</span>

<span class="rvts77">Note the toolbar button to toogle the level of details displayed in the ER Diagram view.  </span>

<img src="lib/OpenAPI - Toggle field details.png" width="25%" height="25%">

<span class="rvts77">  
</span>

<span class="rvts77">By default Hackolade displays all the various ways an API can be defined in OpenAPI.  Once you're done creating your design, you may wish to hide the extraneous structures and adjust the layout to take advantage of the extra real estate on screen.</span>

<span class="rvts77">  
</span>

## <span class="rvts0"><span class="rvts15">Data Types</span></span>

<span class="rvts6">The OpenAPI specification describes primitives (or scalar) data types which can have an optional property modifier,</span> <span class="rvts69">format</span><span class="rvts6">, plus a file primitive type.  Complex types such as arrays and sub-objects, plus combinations thereof, are also allowed.</span>

<span class="rvts6">  
</span>

<img src="lib/Swagger data types.png" width="25%" height="25%"><img src="lib/Swagger data types - string.png" width="25%" height="25%"><img src="lib/Swagger data types - number.png" width="25%" height="25%"><img src="lib/Swagger data types - integer.png" width="25%" height="25%">


<span class="rvts6">  
</span>

## <span class="rvts0"><span class="rvts15">API metadata</span></span>

<span class="rvts6">The info object, as well as the host, basePath, schemes, consumes, produces, the securityDefinitions object, the security object, the tags object, and externalDocs object are fixed fields treated as metadata and maintained at model-level in Hackolade.</span>

<span class="rvts6">  
</span>

<img src="lib/OpenAPI - Info object 1.png" width="33%" height="33%"><img src="lib/OpenAPI - Info object 2.png" width="33%" height="33%">
<img src="lib/OpenAPI - Info object 3.png" width="33%" height="33%">

<span class="rvts78">  
</span>

## <span class="rvts0"><span class="rvts15">Components</span></span>

[Component objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#components-object) <span class="rvts34">hold a set of reusable objects</span> <span class="rvts80">that can be used across multiple endpoints in the same API</span><span class="rvts34">: schema, parameter, request body, response, example, header, security scheme, link, or callback. All objects defined within the components object will have no effect on the API unless they are explicitly referenced from properties outside the components object by using a $ref reference in any path item.</span>

<span class="rvts34">  
</span>

<span class="rvts79">As you expose more resources and operations against your API, your API may repeat a lot of existing parameters or response descriptions in many different paths and operations.  By creating reusable component objects, you avoid time-consuming rewriting as well as the risk of inconsistencies.</span>

<span class="rvts79">  
</span>

<span class="rvts34">Data types can be objects, but also primitives and arrays. This object is based on the </span>[JSON Schema Specification Wright Draft 00 (a.k.a. Draft-05)](http://json-schema.org/)<span class="rvts34"> and uses a predefined subset of it. On top of this subset, there are extensions provided by this specification to allow for more complete documentation.</span>

<span class="rvts34">  
</span>

<img src="lib/OpenAPI - Components.png" width="66%" height="66%">

<span class="rvts34">  
</span>

<span class="rvts6">Consult</span> [this page](Reusableobjectsdefinitions.html) <span class="rvts6">or more information on how to use definitions.  For OpenAPI, you should limit yourself to Hackolade model definitions.</span>

## <span class="rvts0"><span class="rvts15">Resource</span></span>

<span class="rvts6">The resource path object is a container representing the relative path to an individual endpoint.  The field name must start with a slash ("/").  The path is appended to the basePath in order to construct the full URL.  Path templating (</span><span class="rvts34">usage of curly braces ("{}") to mark a section of a URL path as replaceable using path parameters</span><span class="rvts6">) is allowed.</span>

<span class="rvts6">  
</span>

<span class="rvts6">Each resource contains one or more "</span>[path item objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#pathItemObject)<span class="rvts6">" made of a request and one or more responses:</span>

<img src="lib/OpenAPI - Resource container.png" width="66%" height="66%">

<span class="rvts6">  
</span>

<span class="rvts6">You may create a new resource via right-click anywhere in the ERD view and choosing the contextual menu option:</span>

<img src="lib/Swagger - Add resource contextual menu.png" width="25%" height="25%">

<span class="rvts6">  
</span>

<span class="rvts6">or via the menu:</span>

<img src="lib/Swagger - Add resource action menu.png" width="25%" height="25%">

<span class="rvts6">  
</span>

<span class="rvts6">or the toolbar:</span>

<img src="lib/Swagger - Add resource toolbar button.png" width="25%" height="25%">

<span class="rvts6">  
</span>

## <span class="rvts0"><span class="rvts15">Requests</span></span>

<span class="rvts6">A request is an object with a type, associated data, relationships to other resources, and a set of methods that operate on it.  Only a few standard methods are defined for the resource (corresponding to the standard HTTP GET, POST, PUT and DELETE methods</span><span class="rvts44">.</span><span class="rvts6">)</span>

<span class="rvts6">  
</span>

<span class="rvts6">The</span> [Parameter Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#parameterObject) <span class="rvts6">d</span><span class="rvts74">escribes a single operation parameter defined by a combination of a </span><span class="rvts21">name</span><span class="rvts74"> and </span><span class="rvts21">location</span><span class="rvts74">.  Hackolade provides a handy template of parameter types allowing the description of the payload either by adding adding individual fields or by referencing an existing component:</span>

<img src="lib/OpenAPI - Request Parameter 1.png" width="75%" height="75%">
<img src="lib/OpenAPI - Request Parameter 2.png" width="75%" height="75%">

<span class="rvts78">  
</span>

<span class="rvts6">To create a request within a resource container, you may:</span>

<span class="rvts6">- right-click inside the container area of the ERD pane, and choose the contextual menu option:</span>

<img src="lib/Swagger - Add request contextual menu.png" width="25%" height="25%">

<span class="rvts6">- choose the Action menu:</span>

<img src="lib/Swagger - Add request action menu.png" width="25%" height="25%">

<span class="rvts6">- choose the toolbar button:</span>

<img src="lib/Swagger - Add request toolbar button.png" width="25%" height="25%">

<span class="rvts6">  
</span>

<span class="rvts6">It is easy to maintain the metadata for a request in the properties pane:</span>

<img src="lib/OpenAPI - Request Properties.png" width="25%" height="25%">

<span class="rvts78">  
</span>

## <span class="rvts0"><span class="rvts15">Responses</span></span>

[Response objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#responses-object) <span class="rvts34">describe responses from API operations.  For each request, you may create one or more responses.</span>

<img src="lib/OpenAPI - Responses.png" width="75%" height="75%">

<span class="rvts78">  
</span>

<span class="rvts34">A response may have a schema that is defined as individual fields or references a component:</span>

<img src="lib/OpenAPI - Response schema.png" width="75%" height="75%">

<span class="rvts34">  
</span>

<span class="rvts6">To create a response for a given request, you may:</span>

<span class="rvts6">- right-click on the request in the ERD and choose the contextual menu option:</span>

<img src="lib/Swagger - Add response contextual menu.png" width="10%" height="10%">

<span class="rvts6">- or choose the Action menu:</span>

<img src="lib/Swagger - Add response action menu.png" width="10%" height="10%">

<span class="rvts6">- or choose the toolbar button:</span>

<img src="lib/Swagger - Add response toolbar button.png" width="10%" height="10%">

<span class="rvts6">  
</span>

<span class="rvts6">It is easy to maintain the metadata for a response in the properties pane:</span>

<img src="lib/OpenAPI - Response properties.png" width="15%" height="15%">

<span class="rvts78">  
</span>

## <span class="rvts0"><span class="rvts15">Forward-Engineering</span></span>

<span class="rvts34">The files describing the RESTful API in accordance with the OpenAPI specification OAS are represented as JSON objects and conform to the JSON standards.  Hackolade generates OpenAPI documentation in JSON format or YAML format.  The schema exposes two types of fields. Fixed fields, which have a declared name, and patterned fields, which declare a regex pattern for the field name. Patterned fields can have multiple occurrences as long as each has a unique name.  The OpenAPI representation of the API is made of a single file. However, parts of the definitions can be split into separate files, at the discretion of the user. This is applicable for $ref fields in the specification as follows from the </span>[JSON Schema](http://json-schema.org/) [](http://json-schema.org/) <span class="rvts34">definitions.</span>

<span class="rvts34">  
</span>

<img src="lib/OpenAPI - Forward-Engineering.png" width="100%" height="100%">

<span class="rvts78">  
</span>

<span class="rvts34">An internal OpenAPI syntax validator ensures that the generated file is valid, and the right-hand pane allows interactions with the API and testing.</span>

<span class="rvts6">  
</span>

## <span class="rvts0"><span class="rvts15">Reverse-Engineering</span></span>

<span class="rvts6">This function lets you take a OpenAPI file in JSON or YAML format and generate a Hackolade model.  Then, you may enrich the model with comments, generate standard Hackolade documentation, and make the API evolve to generate a new OpenAPI file through forward-engineering.</span>
 readmeEtag: '"44d2984a839356f1059e17dc156d46a55a0c2aa4"' readmeLastModified: Thu, 25 Jul 2024 12:13:38 GMT repositoryId: 192987405 description: Hackolade(https://hackolade.com) plugin for OpenAPI 3 API documentation created: '2019-06-20T21:05:49Z' updated: '2026-02-04T21:05:24Z' language: JavaScript archived: false stars: 18 watchers: 3 forks: 10 owner: hackolade logo: https://avatars.githubusercontent.com/u/20265734?v=4 license: NOASSERTION repoEtag: '"ea8ee7466edb7759c3cc55546fce22d8ecf2b5f644131f384c3f348262860a9f"' repoLastModified: Wed, 04 Feb 2026 21:05:24 GMT foundInMaster: true id: 0ca14e83c7b680313bbc57c0a56e6c05 v3_1: true - source: https://openapi.tools/ name: Apicurio Studio category: - GUI Editors - Server Implementations language: - Angular 7.0 - Java - Saas link: https://www.apicur.io/ repository: https://github.com/apicurio/apicurio-studio source_description: | Web-Based Open Source API Design via the OpenAPI specification. v2: true v3: true repositoryMetadata: base64Readme: >- IVtWZXJpZnkgQnVpbGQgV29ya2Zsb3ddKGh0dHBzOi8vZ2l0aHViLmNvbS9BcGljdXJpby9hcGljdXJpby1zdHVkaW8vd29ya2Zsb3dzL0J1aWxkJTIwJTI2JTIwVmVyaWZ5L2JhZGdlLnN2ZykKWyFbSm9pbiB0aGUgY2hhdCBhdCBodHRwczovL2FwaWN1cmlvLnp1bGlwY2hhdC5jb20vXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3p1bGlwLWpvaW5fY2hhdC1icmlnaHRncmVlbi5zdmcpXShodHRwczovL2FwaWN1cmlvLnp1bGlwY2hhdC5jb20vKQoKIVtBcGljdXJpbyBTdHVkaW9dKC5hc3NldHMvYXBpY3VyaW9fc3R1ZGlvX2xvZ29fZGVmYXVsdC5zdmcpCgojIFRoZSBhcGljdXJpby1zdHVkaW8gcHJvamVjdCAoT3BlbiBTb3VyY2UgQVBJIERlc2lnbiBTdHVkaW8pCgojIyBTdW1tYXJ5CgpUaGlzIGlzIHRoZSBvZmZpY2lhbCBHaXQgcmVwb3NpdG9yeSBmb3IgdGhlIEFwaWN1cmlvIFN0dWRpbyBwcm9qZWN0OiAgaHR0cDovL3d3dy5hcGljdXIuaW8vc3R1ZGlvCgpUaGUgQXBpY3VyaW8gU3R1ZGlvIHByb2plY3QgaXMgYW4gQVBJIGRlc2lnbiBzdHVkaW8gdGhhdCBjYW4gYmUgdXNlZCB0byBjcmVhdGUKbmV3IG9yIGVkaXQgZXhpc3RpbmcgQVBJIGRlc2lnbnMgKHVzaW5nIHNwZWNpZmljYXRpb25zIGxpa2UgT3BlbkFQSSBvciBBc3luY0FQSSkuCgojIyBHZXQgdGhlIGNvZGUKClRoZSBlYXNpZXN0IHdheSB0byBnZXQgc3RhcnRlZCB3aXRoIHRoZSBjb2RlIGlzIHRvIFtjcmVhdGUgeW91ciBvd24gZm9ya10oaHR0cDovL2hlbHAuZ2l0aHViLmNvbS9mb3JraW5nLykKb2YgdGhpcyByZXBvc2l0b3J5LCBhbmQgdGhlbiBjbG9uZSB5b3VyIGZvcms6CmBgYGJhc2gKICAkIGdpdCBjbG9uZSBnaXRAZ2l0aHViLmNvbTo8eW91Pi9hcGljdXJpby1zdHVkaW8uZ2l0CiAgJCBjZCBhcGljdXJpby1zdHVkaW8KICAkIGdpdCByZW1vdGUgYWRkIHVwc3RyZWFtIGdpdDovL2dpdGh1Yi5jb20vYXBpY3VyaW8vYXBpY3VyaW8tc3R1ZGlvLmdpdApgYGAKQXQgYW55IHRpbWUsIHlvdSBjYW4gcHVsbCBjaGFuZ2VzIGZyb20gdGhlIHVwc3RyZWFtIGFuZCBtZXJnZSB0aGVtIG9udG8geW91ciBtYWluOgpgYGBiYXNoCiAgJCBnaXQgY2hlY2tvdXQgbWFpbiAgICAgICAgICAgICAgICMgc3dpdGNoZXMgdG8gdGhlICdtYWluJyBicmFuY2gKICAkIGdpdCBwdWxsIHVwc3RyZWFtIG1haW4gICAgICAgICAgIyBmZXRjaGVzIGFsbCAndXBzdHJlYW0nIGNoYW5nZXMgYW5kIG1lcmdlcyAndXBzdHJlYW0vbWFpbicgb250byB5b3VyICdtYWluJyBicmFuY2gKICAkIGdpdCBwdXNoIG9yaWdpbiAgICAgICAgICAgICAgICAgICAjIHB1c2hlcyBhbGwgdGhlIHVwZGF0ZXMgdG8geW91ciBmb3JrLCB3aGljaCBzaG91bGQgYmUgaW4tc3luYyB3aXRoICd1cHN0cmVhbScKYGBgClRoZSBnZW5lcmFsIGlkZWEgaXMgdG8ga2VlcCB5b3VyICdtYWluJyBicmFuY2ggaW4tc3luYyB3aXRoIHRoZSAndXBzdHJlYW0vbWFpbicuCgojIyBCdWlsZGluZyBhcGljdXJpby1zdHVkaW8KCiMjIyBSZXF1aXJlbWVudHMKLSBOb2RlLmpzIGFuZCBOUE0KCiMjIyBCdWlsZGluZwpVc2Ugc3RhbmRhcmQgTlBNIGJhc2VkIFVJIHRvb2xpbmcgdG8gYnVpbGQgdGhlIHByb2plY3Q6CgpgYGBiYXNoCmNkIHVpCm5wbSBpbnN0YWxsCm5wbSBydW4gYnVpbGQKbnBtIHJ1biBwYWNrYWdlCmBgYAoKIyMgQ29udHJpYnV0ZSBmaXhlcyBhbmQgZmVhdHVyZXMKCkFwaWN1cmlvIFN0dWRpbyBpcyBvcGVuIHNvdXJjZSwgYW5kIHdlIHdlbGNvbWUgYW55Ym9keSB3aG8gd2FudHMgdG8gcGFydGljaXBhdGUgYW5kIGNvbnRyaWJ1dGUhCgpJZiB5b3Ugd2FudCB0byBmaXggYSBidWcgb3IgbWFrZSBhbnkgY2hhbmdlcywgcGxlYXNlIFtsb2cgYW4gaXNzdWUgaW4gR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vYXBpY3VyaW8vYXBpY3VyaW8tc3R1ZGlvL2lzc3VlcykgZGVzY3JpYmluZyB0aGUgYnVnIG9yIG5ldyBmZWF0dXJlLiBUaGVuIHdlIGhpZ2hseSByZWNvbW1lbmQgbWFraW5nIHRoZSBjaGFuZ2VzIG9uIGEgdG9waWMgYnJhbmNoIG5hbWVkIHdpdGggdGhlIEdpdEh1YiBpc3N1ZSBudW1iZXIuIEZvciBleGFtcGxlLCB0aGlzIGNvbW1hbmQgY3JlYXRlcyBhIGJyYW5jaCBmb3IgYW4gaXNzdWUgd2l0aCBudW1iZXIgMTIzNDoKCmBgYGJhc2gKICAkIGdpdCBjaGVja291dCAtYiBhcGljdXJpby1zdHVkaW8tMTIzNApgYGAKCkFmdGVyIHlvdSdyZSBoYXBweSB3aXRoIHlvdXIgY2hhbmdlcyBhbmQgYSBmdWxsIGJ1aWxkICh3aXRoIHVuaXQgdGVzdHMpIHJ1bnMgc3VjY2Vzc2Z1bGx5LCBjb21taXQgeW91cgpjaGFuZ2VzIG9uIHlvdXIgdG9waWMgYnJhbmNoLiBUaGVuIGl0J3MgdGltZSB0byBjaGVjayBmb3IgYW5kIHB1bGwgYW55IHJlY2VudCBjaGFuZ2VzIHRoYXQgd2VyZSBtYWRlIGluCnRoZSBvZmZpY2lhbCByZXBvc2l0b3J5OgoKYGBgYmFzaAogICQgZ2l0IGNoZWNrb3V0IG1haW4gICAgICAgICAgICAgICAjIHN3aXRjaGVzIHRvIHRoZSAnbWFpbicgYnJhbmNoCiAgJCBnaXQgcHVsbCB1cHN0cmVhbSBtYWluICAgICAgICAgICMgZmV0Y2hlcyBhbGwgJ3Vwc3RyZWFtJyBjaGFuZ2VzIGFuZCBtZXJnZXMgJ3Vwc3RyZWFtL21haW4nIG9udG8geW91ciAnbWFpbicgYnJhbmNoCiAgJCBnaXQgY2hlY2tvdXQgYXBpY3VyaW8tc3R1ZGlvLTEyMzQgICAjIHN3aXRjaGVzIHRvIHlvdXIgdG9waWMgYnJhbmNoCiAgJCBnaXQgcmViYXNlIG1haW4gICAgICAgICAgICAgICAgICMgcmVhcHBsaWVzIHlvdXIgY2hhbmdlcyBvbiB0b3Agb2YgdGhlIGxhdGVzdCBpbiBtYWluCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIyAoaS5lLiwgdGhlIGxhdGVzdCBmcm9tIG1haW4gd2lsbCBiZSB0aGUgbmV3IGJhc2UgZm9yIHlvdXIgY2hhbmdlcykKYGBgCgpJZiB0aGUgcHVsbCBncmFiYmVkIGEgbG90IG9mIGNoYW5nZXMsIHlvdSBzaG91bGQgcmVydW4geW91ciBidWlsZCB0byBtYWtlIHN1cmUgeW91ciBjaGFuZ2VzIGFyZSBzdGlsbCBnb29kLgpZb3UgY2FuIHRoZW4gZWl0aGVyIFtjcmVhdGUgcGF0Y2hlc10oaHR0cDovL3Byb2dpdC5vcmcvYm9vay9jaDUtMi5odG1sKSAob25lIGZpbGUgcGVyIGNvbW1pdCwgc2F2ZWQgaW4gYH4vYXBpY3VyaW8tc3R1ZGlvLTEyMzRgKSB3aXRoCgpgYGBiYXNoCiAgJCBnaXQgZm9ybWF0LXBhdGNoIC1NIC1vIH4vYXBpY3VyaW8tc3R1ZGlvLTEyMzQgb3JnaW4vbWFpbgpgYGAKCmFuZCB1cGxvYWQgdGhlbSB0byB0aGUgSklSQSBpc3N1ZSwgb3IgeW91IGNhbiBwdXNoIHlvdXIgdG9waWMgYnJhbmNoIGFuZCBpdHMgY2hhbmdlcyBpbnRvIHlvdXIgcHVibGljIGZvcmsgcmVwb3NpdG9yeQoKYGBgYmFzaAogICQgZ2l0IHB1c2ggb3JpZ2luIGFwaWN1cmlvLXN0dWRpby0xMjM0ICAgICAgICAgIyBwdXNoZXMgeW91ciB0b3BpYyBicmFuY2ggaW50byB5b3VyIHB1YmxpYyBmb3JrIG9mIGFwaWN1cmlvLXN0dWRpbwpgYGAKCmFuZCBbZ2VuZXJhdGUgYSBwdWxsLXJlcXVlc3RdKGh0dHA6Ly9oZWxwLmdpdGh1Yi5jb20vcHVsbC1yZXF1ZXN0cy8pIGZvciB5b3VyIGNoYW5nZXMuCgpXZSBwcmVmZXIgcHVsbC1yZXF1ZXN0cywgYmVjYXVzZSB3ZSBjYW4gcmV2aWV3IHRoZSBwcm9wb3NlZCBjaGFuZ2VzLCBjb21tZW50IG9uIHRoZW0sCmRpc2N1c3MgdGhlbSB3aXRoIHlvdSwgYW5kIGxpa2VseSBtZXJnZSB0aGUgY2hhbmdlcyByaWdodCBpbnRvIHRoZSBvZmZpY2lhbCByZXBvc2l0b3J5LgoKUGxlYXNlIHRyeSB0byBjcmVhdGUgb25lIGNvbW1pdCBwZXIgZmVhdHVyZSBvciBmaXgsIGdlbmVyYWxseSB0aGUgZWFzaWVzdCB3YXkgdG8gZG8gdGhpcyBpcyB2aWEgW2dpdCBzcXVhc2hdKGh0dHBzOi8vZ2l0LXNjbS5jb20vYm9vay9lbi92Mi9HaXQtVG9vbHMtUmV3cml0aW5nLUhpc3RvcnkjU3F1YXNoaW5nLUNvbW1pdHMpLgpUaGlzIG1ha2VzIHJldmVydGluZyBjaGFuZ2VzIGVhc2llciwgYW5kIGF2b2lkcyBuZWVkbGVzc2x5IHBvbGx1dGluZyB0aGUgcmVwb3NpdG9yeSBoaXN0b3J5IHdpdGggY2hlY2twb2ludCBjb21taXRzLgoKIyMgQ29kZSBGb3JtYXR0aW5nCgpXaGVuIHlvdSBhcmUgaGFja2luZyBvbiBzb21lIGFwaWN1cmlvLXN0dWRpbyBjb2RlLCB3ZSdkIHJlYWxseSBhcHByZWNpYXRlIGl0IGlmIHlvdSBmb2xsb3dlZCB0aGUKYXBpY3VyaW8tc3R1ZGlvIGNvZGluZyBzdGFuZGFyZHMuICBUaGUgcHJvamVjdCB1c2VzIGBlc2xpbnRgIHRvIGVuc3VyZSB0aGVzZSBzdGFuZGFyZHMuICBZb3UgY2FuCmNoZWNrIHlvdXIgY29kZSB1c2luZzoKCmBgYGJhc2gKY2QgdWkKbnBtIHJ1biBsaW50CmBgYA== readmeEtag: '"e46aab3313ead9a7513b2ae860b89873a293352a"' readmeLastModified: Fri, 18 Oct 2024 17:27:00 GMT repositoryId: 69595482 description: Open Source API Design created: '2016-09-29T18:22:33Z' updated: '2026-01-23T03:24:57Z' language: TypeScript archived: true stars: 1061 watchers: 36 forks: 517 owner: Apicurio logo: https://avatars.githubusercontent.com/u/28107283?v=4 license: Apache-2.0 repoEtag: '"c85ce0958fdc5a3e44d2b4b7570c0011a6d2d3fa2d72cb566a6bfcb66e99aafc"' repoLastModified: Fri, 23 Jan 2026 03:24:57 GMT foundInMaster: true homepage: https://github.com/Apicurio/apicurio-studio id: efe9656e80c1bf6304903ca5a60fe5a6 - source: - https://openapi.tools/ - openapi3 tags name: OAIE Sketch category: - GUI Editors - Parsers language: Vue.js link: https://www.github.com/OAIE/oaie-sketch repository: https://github.com/oaie/oaie-sketch source_description: > Browser based OpenApi Integrated Editor with side-by side view of the yaml and an interactive graph. v2: false v3: true repositoryMetadata: base64Readme: >- IyBPQUlFIFNrZXRjaApPQUlFIFNrZXRjaCBpcyBhIGJyb3dzZXItYmFzZWQgZWRpdG9yIGZvciBPQVMzIHNwZWMgZG9jdW1lbnRzLiBJdCBvZmZlcnMgYSBzaWRlLWJ5LXNpZGUgWUFNTCBjb2RlIGVkaXRvciBhbmQgdmlzdWFsIGVkaXRvciB3aXRoIGEgZm9jdXMgb24gdGFjdGlsZSBmZWVkYmFjaywgc2ltcGxpY2l0eSBhbmQgcGVyZm9ybWFuY2UuIEl0cyBwaGlsb3NvcGh5IGlzICJzdGF5IGNsb3NlIHRvIHRoZSBjb2RlIGFuZCB2aXN1YWxpemUiLgoKT25saW5lIERlbW86IGh0dHBzOi8vcmF3LmdpdGhhY2suY29tL09BSUUvb2FpZS1za2V0Y2gvbWFzdGVyL3NrZXRjaC5odG1sCgojIyBJbnN0YWxsYXRpb24KQ3VycmVudGx5IE9BSUUgU2tldGNoIGlzIG9ubHkgYSBmZXcgZmlsZXMgcGx1cyBDRE4gbGlua3MuIFNvIHlvdXIgcHJvY2VzcyB0byBnZXQgaXQgcnVubmluZyBpcyBtYW51YWwuCi0gY2xvbmUgcmVwb3NpdG9yeQotIG9wZW4gc2tldGNoLmh0bWwgaW4gYnJvd3NlcgoKIyMgRmVhdHVyZXMKLSBzaG93cyBlYWNoIG9wZXJhdGlvbiBhcyBncmFwaCBub2RlIChwYXRoLCBvcGVyYXRpb25JZCwgc3VtbWFyeSwgcGFyYW1ldGVycykKLSBzaG93cyBlYWNoIHNjaGVtYSBhcyBncmFwaCBub2RlIChuYW1lLCBkZXNjcmlwdGlvbiwgcHJvcGVydGllcykKLSBzaG93cyBlYWNoIHJlbGF0aW9uIGFzIGdyYXBoIGFycm93IChhcnJheS1yZWxhdGlvbnMgYXMgZG90dGVkKQotIGRyYWdnaW5nIGZvciBncmFwaCBub2RlcwotIG1hc3MtbW92ZSB0byBtYWtlIHNwYWNlIGZvciBuZXcgY29udGVudCAodXAvZG93biBhcnJvd3MgaW4gb3BlcmF0aW9uKQotIHF1aWNrIGFjdGlvbiBidXR0b25zIGZvciBhZGRpbmcgcHJvcGVydGllcyBhbmQgc2NoZW1hcwotIGNsaWNraW5nIHNjaGVtYXMgb3IgcHJvcGVydGllcyB3aWxsIHNjcm9sbCB0aGUgZG9jdW1lbnQKLSBkZXRlY3RzIGNoYW5nZXMgYW5kIGNvbG9ycyBzeW5jIGJ1dHRvbnMKLSBjYW4gYWRkIGEgImJha2VkIiB2aXogaW50byB0aGUgZG9jdW1lbnQgYXMgYSB2aWV3ZXIgc3RvcmFnZQotIHVzZXMgdml6IGFzIG1ldGFkYXRhIChub2RlIHBvc2l0aW9uKSBzdG9yYWdlCi0gc3luYyBncmFwaCBpbnRvIHlhbWwsIHN5bmMgeWFtbCBpbnRvIGdyYXBoCi0gYXV0by1zYXZlIGluIGxvY2Fsc3RvcmFnZSBvZiBicm93c2VyCgohW29haWUtc2tldGNoLTIucG5nXShvYWllLXNrZXRjaC0yLnBuZykKCiMjIE9waW5pb25hdGVkIE9BUzMgWUFNTApSZXF1ZXN0cywgUmVzcG9uc2VzIGFuZCBzdWItb2JqZWN0cyBzaG91bGQgYmUgdGhlaXIgb3duIHNjaGVtYSBkZWZpbml0aW9ucy4gTm90ZSB0aGF0IHRoaXMgaXMgYmVzdCBwcmFjdGljZSBhbnl3YXkgaWYgeW91IGV2ZXIgd2FudCB0byBnZW5lcmF0ZSBjbGFzc2VzIGZyb20geW91ciBzcGVjLgoKICAgIGNvbXBvbmVudHM6CiAgICAgIHNjaGVtYXM6CiAgICAgICAgQ2FwYWJpbGl0aWVzUmVxdWVzdDoKICAgICAgICAgIHJlcXVpcmVkOgogICAgICAgICAgICAtIHNwZWMKICAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgIHNwZWM6CiAgICAgICAgICAgICAgJHJlZjogJyMvY29tcG9uZW50cy9zY2hlbWFzL1NwZWMnCiAgICAgICAgQ2FwYWJpbGl0aWVzUmVzcG9uc2U6CiAgICAgICAgICBwcm9wZXJ0aWVzOgogICAgICAgICAgICBjYXBhYmlsaXRpZXM6CiAgICAgICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgICAgICBpdGVtczoKICAgICAgICAgICAgICAgICRyZWY6ICcjL2NvbXBvbmVudHMvc2NoZW1hcy9DYXBhYmlsaXR5JwogICAgICAgIENhcGFiaWxpdHk6CiAgICAgICAgICBwcm9wZXJ0aWVzOgogICAgICAgICAgICBtZXRob2Q6CiAgICAgICAgICAgICAgdHlwZTogc3RyaW5nCiAgICAgICAgICAgIHNrdVBhdHRlcm46CiAgICAgICAgICAgICAgdHlwZTogc3RyaW5nCiAgICAgICAgU3BlYzoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBPQUlFLm5vSW5BcnJvd3MKICAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgIGlkOgogICAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgICBzY2hlZHVsZXM6CiAgICAgICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgICAgICBpdGVtczoKICAgICAgICAgICAgICAgICRyZWY6ICcjL2NvbXBvbmVudHMvc2NoZW1hcy9TY2hlZHVsZScKICAgICAgICBTY2hlZHVsZToKICAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgIG1ldGhvZDoKICAgICAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgICAgZXhwcmVzc2lvbjoKICAgICAgICAgICAgICB0eXBlOiBzdHJpbmcKCiMjIFZJWgpCZWdpbm5pbmcgZG9jdW1lbnRzIGxpa2UgdGhpcyB3aWxsIGFkZCBhIHZpeiAodmlzdWFsaXNhdGlvbikgaW50byB0aGUgaGVhZGVyICh0aGlzIHdpbGwgc2hvdyB1cCBhcyBhIGdyYXBoaWMgaW4gc3dhZ2dlcik6CgogICAgb3BlbmFwaTogIjMuMC4wIgogICAgaW5mbzoKICAgICAgdmVyc2lvbjogIjAuMC4xIgogICAgICB0aXRsZTogTXkgU2VydmljZQogICAgICBkZXNjcmlwdGlvbjogfAogICAgICAgIE15IGludHJvZHVjdG9yeSBpbmZvcm1hdGlvbgogICAgICAgIDwhLS1PQUlFLnZpei0tPjwhLS0vT0FJRS52aXotLT4KICAgIC4uLgoKU2tldGNoIHBlcnNpc3RzIGdyYXBoIG5vZGUgcG9zaXRpb25zIHRvIHRoZSBicm93c2VyJ3MgbG9jYWxTdG9yYWdlLCBidXQgd2hlbiBhIHZpeiBpcyBwcmVzZW50LCBpdCB3aWxsIHN0b3JlIGFuZCBsb2FkIGdyYXBoIG5vZGUgcG9zaXRpb25zIHRvIGFuZCBmcm9tIHRoaXMgdml6IChpbnNpZGUgdGhlICZsdDshLS1PQUlFLnZpei0tJmd0OyB0YWcpLgoKIyMgQ2F2ZWF0cwotIEN1cnJlbnRseSBvbmx5IHRlc3RlZCBvbiBDaHJvbWUuCi0gT0FJRSBTa2V0Y2ggaGFzIGFuIG9waW5pb24gYWJvdXQgY29ycmVjdCBkb2N1bWVudCBzdHJ1Y3R1cmU6IGVhY2ggc2NoZW1hIHNob3VsZCBiZSBkZWNsYXJlZCBhcyBhIG5hbWVkIHNjaGVtYS4KLSBsb2NhbCBzdG9yYWdlIGlzIHVzZWQgYXMgcGVyc2lzdGVuY2UuIHRoaXMgbWVhbnMgdGhhdCBjdXJyZW50bHkgaXQgaXMgb25seSBmZWFzaWJsZSB0byBlZGl0IG9uZSBzcGVjIGF0IGEgdGltZSAobXVjaCBsaWtlIHRoZSBTd2FnZ2VyIG9ubGluZSBlZGl0b3IpLgotIE9BSUUgU2tldGNoIGlzIGluIGEgdmVyeSBlYXJseSBzdGFnZSwgcGxlYXNlIGJhY2t1cCB5b3VyIGRvY3VtZW50cyBmcmVxdWVudGx5IG91dHNpZGUgb2YgaXQhCi0gU3luYzogd2hlbiBib3RoIGFyZSB5ZWxsb3csIGZpcnN0IGNsaWNrICI8LSB1cGRhdGUgaW5saW5lIHZpeiIsIHRoZW4gInVwZGF0ZSBmcm9tIGVkaXRvciAtPiIKCiMjIFRPRE8KLSBkb2N1bWVudGF0aW9uCi0gZmluZCBjb250cmlidXRvcnMKLSBtYXJrZXRpbmcKLSBzZWUgUHJvamVjdHMgdGFiIGZvciBwbGFubmVkIGZlYXR1cmVzCgojIyBDb250cmlidXRpbmcKLSBXZSB3YW50IHRvIG1haW50YWluIGEgZm9jdXMgb24gdGhlIGNvZGUuCi0gV2Ugd2FudCB0byBmYWNpbGl0YXRlIHdyaXRpbmcgYmV0dGVyIHNwZWNzLgotIFdlIHdhbnQgdG8gbWFrZSB3cml0aW5nIHNwZWNzIGVhc2llciBhbmQgZ2l2ZSBhbiBlYXNpZXIgbGVhcm5pbmcgY3VydmUuCi0gV2Ugd2FudCB0byBiZSBhcyBsaWdodHdlaWdodCBhcyBwb3NzaWJsZSAoY3VycmVudGx5IG5lZWRzIFZ1ZSwgSlMtWUFNTCwgalF1ZXJ5LCBqUXVlcnkgVUkgLSBoZWxwIG9uIHJlbW92aW5nIGpRdWVyeSBhcHByZWNpYXRlZCEpLgotIFdlIHdhbnQgdG8ga2VlcCB0aGUgZGVwZW5kZW5jeSB0byBsaWJyYXJpZXMgYXMgbG93IGFzIHBvc3NpYmxlLgo= readmeEtag: '"5c70fc3576325791048b134c65f5e9b26df07ba9"' readmeLastModified: Wed, 09 Dec 2020 09:27:25 GMT repositoryId: 251551520 description: OpenAPI Visual Editor created: '2020-03-31T09:03:18Z' updated: '2025-09-02T20:09:25Z' language: HTML archived: false stars: 74 watchers: 5 forks: 10 owner: OAIE logo: https://avatars.githubusercontent.com/u/62933192?v=4 license: MIT repoEtag: '"5809b16e76ef933a3044fdb2ba4fdd34ed84fab38b2b44ce8361d1e4dadb7e2a"' repoLastModified: Tue, 02 Sep 2025 20:09:25 GMT foundInMaster: true id: 1559682e19f96b7e2025e28b33915bed - source: https://openapi.tools/ name: OpenAPI Designer category: GUI Editors language: SaaS link: https://openapidesigner.com source_description: >- OpenAPI Designer is an easy-to-use, free, web-based, codeless description document editor with JSON and YAML outputs. v3: true v3_1: true id: 113b3e287451017a1bfd30a88f5aa402 foundInMaster: true - source: https://openapi.tools/ name: ApiBldr category: GUI Editors language: - Angular 9.0 - Saas link: https://www.apibldr.com/ source_description: | Web-Based API Designer for OpenAPI (swagger) and AsyncAPI specifications. v2: true v3: true foundInMaster: true id: 2e1e071de0cefab924eba4860c14f63f - source: https://openapi.tools/ name: RestCase Designer category: GUI Editors language: - Angular 9.0 - Saas link: https://www.restcase.com/platform/design source_description: > A design-first API managment platform with WYSIWYG API Designer for OpenAPI and AsyncAPI specifications. v2: true v3: true foundInMaster: true id: 034f272138f88f0a55801dae0857323a - source: https://openapi.tools/ name: Frogment Editor category: GUI Editors language: - Desktop - SaaS link: https://www.frogment.com source_description: > A free openAPI spec editor and linter that breaks down your spec into fragments to make editing easier and more intuituve. v2: false v3: true v3_1: true id: 372fe64c1e2965a75c3a3e3b1ccc5eaa foundInMaster: true - source: https://openapi.tools/ name: Meeshkan category: - Testing - Mock language: SaaS link: https://meeshkan.com/ source_description: >- Meeshkan is an automated testing and mocking tool. It offers first-class support for GraphQL APIs, but Meeshkan is also built to handle REST APIs and third-party dependencies. v2: false v3: true id: bfa5bf187387df6760e4366cf73a63b4 foundInMaster: true - source: https://openapi.tools/ name: Response2Schema category: - Learning - Parsers language: PHP link: https://github.com/dsuurlant/response2schema source_description: >- Takes any JSON response and generates an OpenAPI definition document with the component schema and a default endpoint. v2: false v3: true foundInMaster: true repository: https://github.com/dsuurlant/response2schema repositoryMetadata: base64Readme: >- IyBSZXNwb25zZTJTY2hlbWEKCkEgcXVpY2sgJ24gZWFzeSB3YXkgdG8gZ2VuZXJhdGUgeW91ciBPcGVuQVBJIHNwZWMgYmFzZWQgb24gYSBKU09OIG9iamVjdC4gVXNlZnVsIGZvciBib290c3RyYXBwaW5nIHlvdXIgY29tcG9uZW50CiBzY2hlbWFzLiBUaGlzIGlzIGludGVuZGVkIHRvIGJlIGEgc3RhcnRpbmcgcG9pbnQgZm9yIHdyaXRpbmcgeW91ciBzcGVjaWZpY2F0aW9uLCBhcyB0aGUgdG9vbCBjYW5ub3QgaW5mZXIgdHlwZXMgc3VjaAogIGFzIGVudW0sIG9uZU9mLCBtYXhpbXVtL21pbmltdW0sIGFuZCBzbyBvbi4gSXQgYWxzbyBkb2VzIG5vdCBzdXBwb3J0IGdlbmVyYXRpbmcgZW5kcG9pbnRzIG9yIGVycm9yIHJlc3BvbnNlcy4gQWxsCiAgIGl0IGRvZXMgaXMgdGFrZSB5b3VyIEpTT04gb2JqZWN0IGFuZCB0dXJucyBpdCBpbnRvIGFuIE9wZW5BUEkgc2NoZW1hIG9iamVjdCwgYWxvbmcgd2l0aCBhIHNpbXBsZSBleGFtcGxlIHNwZWMgdGhhdAogICAgeW91IGNhbiBleHBhbmQgb24uCiAgICAKU3VwcG9ydHMgZ2VuZXJhdGluZyBhbiBPcGVuQVBJIHNwZWMgaW4gYEpTT05gIG9yIGB5YW1sYC4KCiMgSW5zdGFsbGF0aW9uCgpgY29tcG9zZXIgcmVxdWlyZSBkc3V1cmxhbnQvcmVzcG9uc2Uyc2NoZW1hYAoKT3IgZG93bmxvYWQgdGhlIHBoYXIgZnJvbSB0aGUgW1JlbGVhc2VzIHBhZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9kc3V1cmxhbnQvcmVzcG9uc2Uyc2NoZW1hL3JlbGVhc2VzKS4KCiMgVXNhZ2UKCkp1c3QgcG9pbnQgUmVzcG9uc2UyU2NoZW1hIHRvIHlvdXIgaW5wdXQganNvbiBmaWxlLCBhbmQgdGVsbCBpdCB3aGVyZSB0byBwdXQgdGhlIG91dHB1dCBPcGVuQVBJIHNwZWMuCgpJdCB3aWxsIGF1dG9tYXRpY2FsbHkgZm9ybWF0IGl0IHRvIGpzb24gb3IgeWFtbCBiYXNlZCBvbiB0aGUgZXh0ZW5zaW9uIG9mIHRoZSBvdXRwdXQgcGF0aC4KCmAuL3ZlbmRvci9iaW4vcmVzcG9uc2Uyc2NoZW1hIHJlc3BvbnNlLmpzb24gb3BlbmFwaS55YW1sYAoKT3Igd2hlbiB1c2luZyB0aGUgcGhhcjoKCmAuL3Jlc3BvbnNlMnNjaGVtYS5waGFyIHJlc3BvbnNlLmpzb24gb3BlbmFwaS55YW1sYAoKIyBFeGFtcGxlCgpHaXZlbiBhIHZlcnkgc2ltcGxlIHJlc3BvbnNlOgoKYGBganNvbgp7CiAgICAiaWQiOiAxLCAgCiAgICAibmFtZSI6ICJFeGFtcGxlIFJlc3BvbnNlIgp9CmBgYAoKUmVzcG9uc2UyU2NoZW1hIGdlbmVyYXRlcyB0aGUgZm9sbG93aW5nIHNwZWM6CgpgYGB5YW1sCm9wZW5hcGk6IDMuMC4wCmluZm86CiAgdGl0bGU6ICdPcGVuQVBJIHNwZWNpZmljYXRpb24gYXV0b21hdGljYWxseSBnZW5lcmF0ZWQgYnkgUmVzcG9uc2UyU2NoZW1hLicKICBkZXNjcmlwdGlvbjogJ1BsZWFzZSBhZGFwdCB0aGlzIHNwZWNpZmljYXRpb24gdG8geW91ciBvd24gbmVlZHMuJwogIHZlcnNpb246IDEuMC4wCnBhdGhzOgogIC9yZXNvdXJjZToKICAgIGdldDoKICAgICAgZGVzY3JpcHRpb246ICdEZXNjcmlwdGlvbiBvZiB0aGUgZW5kcG9pbnQnCiAgICAgIG9wZXJhdGlvbklkOiBnZXRSZXNvdXJjZQogICAgICByZXNwb25zZXM6CiAgICAgICAgJzIwMCc6CiAgICAgICAgICBkZXNjcmlwdGlvbjogJ0Rlc2NyaXB0aW9uIG9mIHRoaXMgcmVzcG9uc2UuJwogICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICAkcmVmOiAnIy9jb21wb25lbnRzL3NjaGVtYXMvUmVzb3VyY2UnCmNvbXBvbmVudHM6CiAgc2NoZW1hczoKICAgIFJlc291cmNlOgogICAgICB0eXBlOiBvYmplY3QKICAgICAgcHJvcGVydGllczoKICAgICAgICBpZDoKICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgICAgICBuYW1lOgogICAgICAgICAgdHlwZTogc3RyaW5nCmBgYAoKSXQgd2lsbCBhbHdheXMgZ2VuZXJhdGUgdGhlIGV4YW1wbGUgZW5kcG9pbnQgYW5kIGEgc2NoZW1hIG5hbWVkICdSZXNvdXJjZScuCgpUaGUgYmVzdCB3YXkgdG8gdXNlIHRoaXMgdG9vbCBpcyB0byB0YWtlIHRoaXMgYXMgYSBzdGFydGluZyBwb2ludCwgb3IgY29weS1wYXN0ZSB0aGUgc2NoZW1hIGRlZmluaXRpb24gdG8geW91ciBvd24KIE9wZW5BUEkgc3BlYy4KIAojIENyZWRpdHMKCkJ1aWx0IGFuZCBtYWludGFpbmVkIGJ5IFtEYW5pw6tsbGUgU3V1cmxhbnRdKGh0dHBzOi8vZ2l0aHViLmNvbS9kc3V1cmxhbnQpLgoKUmVsaWVzIGhlYXZpbHkgb24gdGhlIGF3ZXNvbWUgUEhQIE9wZW5BUEkgbGlicmFyeSBbY2ViZS9waHAtb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL2NlYmUvcGhwLW9wZW5hcGkpLgoK readmeEtag: '"112e223871be1155102dc38aa7911ef203e0dac3"' readmeLastModified: Mon, 09 Oct 2023 08:32:37 GMT repositoryId: 266575464 description: A quick and easy tool for generating OpenAPI schemas. created: '2020-05-24T16:00:11Z' updated: '2025-10-30T15:48:44Z' language: PHP archived: false stars: 90 watchers: 3 forks: 5 owner: dsuurlant logo: https://avatars.githubusercontent.com/u/4609379?v=4 license: MIT repoEtag: '"2e85d125f46c90a41351d84b1b40f7c49dd81c269da8b4cf605b0e1c185cb5f4"' repoLastModified: Thu, 30 Oct 2025 15:48:44 GMT id: ee1da7e4ea9bb6a42233ac4702ceebf7 - source: https://openapi.tools/ name: InducOapi category: - Learning - Parsers language: Python link: https://pypi.org/project/inducoapi repository: https://github.com/thewall89/inducoapi source_description: >- A simple python module to generate OpenAPI Description Documents by supplying request/response bodies. v2: false v3: true repositoryMetadata: base64Readme: >- # InducOapi

[![ci](https://github.com/TheWall89/inducoapi/actions/workflows/ci.yml/badge.svg?branch=master)](https://github.com/TheWall89/inducoapi/actions/workflows/ci.yml)
[![PyPI version](https://badge.fury.io/py/inducoapi.svg)](https://badge.fury.io/py/inducoapi)

A simple python module to generate OpenAPI Description Documents by supplying request/response bodies.

*Contributions for new features, fixes or improvements are welcome. Feel free to send a pull request.*

## Motivation

Sometimes you have a fully functioning HTTP service without OpenAPI documentation. At some point in time, others may
need to use your service. Writing the documentation by hand is a pain and can feel like an overwhelming job for complex
services.
*inducoapi* helps you generate your OpenAPI Description Documents by taking as input request/response examples plus some
other information.

The generated OpenAPI documentation is validated
with [openapi-spec-validator](https://github.com/p1c2u/openapi-spec-validator).

*Warning*: This program also generates the `example` fields in OpenAPI schemas by default. If you have sensitive data in
your request/response files, disable this feature with `--no-example`.

## Installation

### With `pip`

```shell script
pip install inducoapi
```

### With [poetry](https://python-poetry.org/)

```shell script
git clone git@github.com:TheWall89/inducoapi.git
cd inducoapi
poetry install
```

To run unit-tests:

```shell script
poetry install --with=dev
poetry run pytest
```

## Usage

### From CLI

`inducoapi` provides its own command. You can simply execute it with

```shell script
inducoapi
```

If you get a `command not found` error, try to activate your virtualenv or run `poetry shell` first.

You can also run `inducoapi` in the classic way:

```shell script
python -m inducoapi
```

#### Help

`inducoapi` provides its own help. Check it out with:

```shell script
python -m inducoapi -h
```

#### Examples

Let's consider a simple case: you have an HTTP service managing employees. We want to generate the OpenAPI Description
Document for a GET on all the employees, returning a 200 status code:

```shell script
python -m inducoapi GET /employees 200
```

<details><summary>output</summary>

```yaml
openapi: 3.0.0
info:
  title: Generated by InducOapi
  version: v1
paths:
  /employees:
    get:
      responses:
        200:
          description: ''
```

</details>

Now, a GET request with an empty response is not quite useful. Let's add an argument with a JSON file containing a
response example. Input examples can be found in [examples](examples).

```shell script
python -m inducoapi GET /employees 200 --response examples/employees.json
```

<details><summary>output</summary>

```yaml
openapi: 3.0.0
info:
  title: Generated by InducOapi
  version: v1
paths:
  /employees:
    get:
      responses:
        200:
          description: ''
          content:
            application/json:
              schema:
                type: array
                items:
                  type: object
                  properties:
                    id:
                      type: integer
                      example: 1
                    name:
                      type: string
                      example: Dwight Schrute
                    role:
                      type: string
                      example: salesman
```

</details>

Let's add a parameter to filter the employees by name.

```shell script
python -m inducoapi GET /employees 200 --response examples/employees.json --parameter name,query 
```

<details><summary>output</summary>

```yaml
openapi: 3.0.0
info:
  title: Generated by InducOapi
  version: v1
paths:
  /employees:
    get:
      responses:
        '200':
          description: ''
          content:
            application/json:
              schema:
                type: array
                items:
                  type: object
                  properties:
                    id:
                      type: integer
                      example: 1
                    name:
                      type: string
                      example: Dwight Schrute
                    role:
                      type: string
                      example: salesman
      parameters:
        - name: name
          in: query
          required: false
          description: ''
          schema: { }
```

</details>

Finally, let's try a POST request with both request and response examples.

```shell script
python -m inducoapi POST /employees 201 --request examples/new_employee_req.json --response examples/new_employee_resp.json
```

<details><summary>output</summary>

```yaml
openapi: 3.0.0
info:
  title: Generated by InducOapi
  version: v1
paths:
  /employees:
    post:
      requestBody:
        content:
          application/json:
            schema:
              type: object
              properties:
                name:
                  type: string
                  example: Michael Scott
                role:
                  type: string
                  example: manager
      responses:
        201:
          description: ''
          content:
            application/json:
              schema:
                type: object
                properties:
                  id:
                    type: integer
                    example: 4
                  name:
                    type: string
                    example: Michael Scott
                  role:
                    type: string
                    example: manager
```

</details>

If you want to directly write the generated OpenAPI Description Documents to a YAML file, just
add `--output openapi.yaml`

### From python

[test_inducoapi.py](tests/test_inducoapi.py) provides usage examples of the module from python.

## TODO list

- [x] Add support for request/response files in YAML
- [x] Add support for `application/yaml` content
- [x] Customize title and version in info
- [x] Package module
- [x] Support for `$ref` in response schemas
- [x] Add support for `parameters`
- [ ] ~~Add support for `links`~~ (I don't think it is very useful)
- [ ] ~~Add support for `format`~~ (hard to infer)
 readmeEtag: '"8ed88b3cfc9912b379b3dfab4295625b3934844e"' readmeLastModified: Sun, 06 Jul 2025 17:41:36 GMT repositoryId: 232868853 description: >- A simple python module to generate OpenAPI Description Documents by supplying request/response bodies. created: '2020-01-09T17:43:54Z' updated: '2025-07-30T11:24:37Z' language: Python archived: false stars: 19 watchers: 2 forks: 4 owner: TheWall89 logo: https://avatars.githubusercontent.com/u/2626117?v=4 license: Apache-2.0 repoEtag: '"5d7c4dfaff173d065c568d5ddfd571906a3c9d120349328cdd2d45ead23b403c"' repoLastModified: Wed, 30 Jul 2025 11:24:37 GMT foundInMaster: true id: 2829ee121d0fb8321fc8c0dbe2c74103 v3_1: true - source: - https://openapi.tools/ - openapi3 tags name: Prism category: - Mock - Server Implementations language: cli link: https://stoplight.io/open-source/prism repository: https://github.com/stoplightio/prism source_description: >- Turn any OAI file into an API server with mocking, transformations, validations, and more. v2: true v3: true v3_1: true repositoryMetadata: base64Readme: >- [![Prism - API Mock Servers and Contract Testing](examples/readme-header.svg)][mocking_landing_page]

[![CircleCI][circle_ci_image]][circle_ci]
[![NPM Downloads][npm_image]][npm]
[![Stoplight Forest](https://img.shields.io/ecologi/trees/stoplightinc)][stoplight_forest]

# Prism Overview

Prism is a set of packages for API mocking and contract testing with **OpenAPI v2** (formerly known as Swagger) and **OpenAPI v3.x**.

Prism provides:

- **Mock Servers**: Life-like mock servers from any API specification document.
- **Validation Proxy**: Contract Testing for API consumers and developers.
- **Comprehensive API Specification Support**: OpenAPI v3.1, OpenAPI v3.0, OpenAPI v2.0 (formerly Swagger) and Postman Collections.

## Ways to Use Prism

### Hosted Prism

Stoplight provides hosted mock servers for convenience so that API consumers can experiment with an API without the need for backend code.

Use one of these options for instant, hosted mock servers:

- [Stoplight Platform](https://stoplight.io/?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_prism): Collaborative API Design Platform for designing, developing and documenting APIs with hosted mocking powered by Prism.
- [Stoplight Studio](https://stoplight.io/studio/?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_prism): Free visual OpenAPI designer that comes integrated with mocking powered by Prism.

Learn more in the [hosted Prism documentation](https://docs.stoplight.io/docs/platform/336b74db38c50-work-with-mock-servers).

### Self-hosted Prism

Prism is an open-source HTTP server run from the command-line. It provides mocking, request validation, and content negotiation. Use it standalone tool or in continuous integration.

![Demo of Prism Mock Server being called with curl from the CLI](./examples/prism-cli.svg)

## Installation and Usage

This information refers to Open Source Prism 3.x, which is the current version most likely you will use. If you're looking for the 2.x version, look at the [`2.x` branch][2.x]

### Install Self-hosted Prism

Prism requires

- NodeJS >= 18.20.1
- for NodeJS 18.x, [>= 18.16 is required](https://github.com/stoplightio/prism/issues/2305)

```bash
npm install -g @stoplight/prism-cli

# OR

yarn global add @stoplight/prism-cli
```

For more installation options, see our [installation documentation](./docs/getting-started/01-installation.md).

### Mocking

Prism can help you create a fake "mock" based off an OpenAPI document, which helps people see how your API will work before you even have it built. Run it locally with the `prism mock` command to run your API on a HTTP server you can interact with.

```bash
prism mock https://raw.githack.com/OAI/OpenAPI-Specification/master/examples/v3.0/petstore-expanded.yaml
```

Learn more about [how the mock server works](docs/guides/01-mocking.md).

### Validation Proxy

Prism can help you check for discrepencies between your API implementation and the OpenAPI document that describes, letting you funnel HTTP traffic through it with the `prism proxy` command.

```bash
prism proxy examples/petstore.oas2.yaml https://petstore.swagger.io/v2
```

Learn more about [how the validation proxy works](docs/guides/03-validation-proxy.md).

## 📖 Documentation and Community

- [Documentation](https://docs.stoplight.io/docs/prism/674b27b261c3c-prism-overview)
  - [Getting Started](./docs/getting-started/01-installation.md)
  - [Guides](./docs/guides/01-mocking.md)
- [Community](https://github.com/stoplightio/prism/discussions)

### ❓ FAQs

**Cannot access mock server when using Docker?**

Prism uses localhost by default, which usually means 127.0.0.1. When using docker the mock server will
be unreachable outside of the container unless you run the mock command with `-h 0.0.0.0`.

**Why am I getting 404 errors when I include my basePath?**

OpenAPI v2.0 had a concept called "basePath", which was essentially part of the HTTP path the stuff
after host name and protocol, and before query string. Unlike the paths in your `paths` object, this
basePath was applied to every single URL, so Prism v2.x used to do the same. In OpenAPI v3.0 they
merged the basePath concept in with the server.url, and Prism v3 has done the same.

We treat OAS2 `host + basePath` the same as OAS3 `server.url`, so we do not require them to go in
the URL. If you have a base path of `api/v1` and your path is defined as `hello`, then a request to
`http://localhost:4010/hello` would work, but `http://localhost:4010/api/v1/hello` will fail. This
confuses some, but the other way was confusing to others. Check the default output of Prism CLI to
see what URLs you have available.

### 🚧 Roadmap

- [x] Content Negotiation
- [x] Security Validation
- [x] Validation Proxy
- [ ] [Recording/Learning Mode](https://roadmap.stoplight.io/c/324-learning-recording?utm_source=github&utm_medium=prism&utm_campaign=readme) (create OpenAPI from HTTP traffic)
- [ ] [Data Persistence](https://roadmap.stoplight.io/c/308-persisted-mock-data?utm_source=github&utm_medium=prism&utm_campaign=readme) (allow Prism act like a sandbox)

Submit your ideas for new functionality on the [Stoplight Roadmap](https://roadmap.stoplight.io/?utm_source=github&utm_medium=prism&utm_campaign=readme).

### 🏁 Help Others Utilize Prism

If you're using Prism for an interesting use case, [contact us](mailto:growth@stoplight.io) for a case study. We'll add it to a list here. Spread the goodness 🎉

### 👏 Contributing

If you are interested in contributing to Prism itself, check out our [contributing docs ⇗][contributing] and [code of conduct ⇗][code_of_conduct] to get started.

### 🎉 Thanks

Prism is built on top of lots of excellent packages, and here are a few we'd like to say a special thanks to.

- [ajv](https://www.npmjs.com/package/ajv)
- [faker](https://www.npmjs.com/package/@faker-js/faker)
- [fp-ts](https://www.npmjs.com/package/fp-ts)
- [gavel](https://www.npmjs.com/package/gavel)
- [json-schema-faker](https://www.npmjs.com/package/json-schema-faker)
- [lerna](https://www.npmjs.com/package/lerna)
- [micri](https://www.npmjs.com/package/micri)
- [openapi-sampler](https://www.npmjs.com/package/openapi-sampler)
- [yargs](https://www.npmjs.com/package/yargs)

Check these projects out!

### 🌲 Sponsor Prism by Planting a Tree

If you would like to thank us for creating Prism, we ask that you [**buy the world a tree**](https://ecologi.com/stoplightinc).

[code_of_conduct]: CODE_OF_CONDUCT.md
[contributing]: CONTRIBUTING.md
[download-release]: https://github.com/stoplightio/prism/releases/latest
[core]: https://www.npmjs.com/package/@stoplight/prism-core
[http]: https://www.npmjs.com/package/@stoplight/prism-http
[http-server]: https://www.npmjs.com/package/@stoplight/prism-http-server
[cli]: https://www.npmjs.com/package/@stoplight/prism-cli
[cli-docs]: ./docs/getting-started/03-cli.md
[2.x]: https://github.com/stoplightio/prism/tree/2.x
[http-docs]: packages/http/README.md
[mocking_landing_page]: https://stoplight.io/api-mocking?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_prism
[circle_ci]: https://circleci.com/gh/stoplightio/prism
[circle_ci_image]: https://img.shields.io/circleci/build/github/stoplightio/prism/master
[npm]: https://www.npmjs.com/package/@stoplight/prism-cli
[npm_image]: https://img.shields.io/npm/dw/@stoplight/prism-http?color=blue
[stoplight_forest]: https://ecologi.com/stoplightinc
 readmeEtag: '"6f5fe69fe0b604dfdac6d8014ab3f37ed8a1c19e"' readmeLastModified: Fri, 18 Oct 2024 09:24:35 GMT repositoryId: 54172257 description: >- Turn any OpenAPI2/3 and Postman Collection file into an API server with mocking, transformations and validations. created: '2016-03-18T03:52:09Z' updated: '2026-02-05T20:25:21Z' language: TypeScript archived: false stars: 4833 watchers: 53 forks: 389 owner: stoplightio logo: https://avatars.githubusercontent.com/u/10767217?v=4 license: Apache-2.0 repoEtag: '"e0599d9e2d7c5cb23cd2d10097a713302ba951bd08d31033e57c009fa3207d54"' repoLastModified: Thu, 05 Feb 2026 20:25:21 GMT foundInMaster: true useMasterProperties: true id: 15aad8cd322242e03733b07f2a37890a - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags name: Microcks category: - Mock Testing - SDK language: Java link: https://microcks.io repository: https://github.com/microcks/microcks source_description: >- Open source Kubernetes-native tool for API Mocking and Testing. Turn your OAI contract examples into ready-to-use mocks. Use examples to test and validate implementations according to spec and schema elements. Microcks is a Cloud Native Computing Sandbox project 🚀 v2: true v3: true v3_1: true repositoryMetadata: base64Readme: >- <img src="./microcks-banner.png" width="600"> 

[![GitHub Workflow Status](https://img.shields.io/github/actions/workflow/status/microcks/microcks/build-verify.yml?logo=github&style=for-the-badge)](https://github.com/microcks/microcks/actions)
[![Container](https://img.shields.io/badge/dynamic/json?color=blueviolet&logo=docker&style=for-the-badge&label=Quay.io&query=tags[1].name&url=https://quay.io/api/v1/repository/microcks/microcks/tag/?limit=10&page=1&onlyActiveTags=true)](https://quay.io/repository/microcks/microcks?tab=tags)
[![Version](https://img.shields.io/maven-central/v/io.github.microcks/microcks?color=blue&style=for-the-badge)]((https://search.maven.org/artifact/io.github.microcks/microcks))
[![License](https://img.shields.io/github/license/microcks/microcks?style=for-the-badge&logo=apache)](https://www.apache.org/licenses/LICENSE-2.0)
[![Project Chat](https://img.shields.io/badge/discord-microcks-pink.svg?color=7289da&style=for-the-badge&logo=discord)](https://microcks.io/discord-invite/)
[![Artifact HUB](https://img.shields.io/endpoint?url=https://artifacthub.io/badge/repository/microcks&style=for-the-badge)](https://artifacthub.io/packages/search?repo=microcks)
[![CNCF Landscape](https://img.shields.io/badge/CNCF%20Landscape-5699C6?style=for-the-badge&logo=cncf)](https://landscape.cncf.io/?item=app-definition-and-development--application-definition-image-build--microcks)

# Microcks - Kubernetes native tool for API Mocking & Testing

Microcks is a platform for turning your API and microservices assets - *OpenAPI specs*, *AsyncAPI specs*, *gRPC protobuf*, *GraphQL schema*, *Postman collections*, *SoapUI projects* - into live mocks in seconds.

It also reuses these assets to run contract conformance and non-regression tests against your API implementation. We provide integrations with *Jenkins*, *GitHub Actions*, *Tekton* and many others through a simple CLI.

[![LFX Health Score](https://img.shields.io/static/v1?label=Health%20Score&message=Healthy&color=A7F3D0&logo=linuxfoundation&logoColor=white&style=flat)](https://insights.linuxfoundation.org/project/microcks/repository/microcks-microcks) [![LFX Contributors](https://img.shields.io/static/v1?label=Contributors&message=761&color=0094FF&logo=linuxfoundation&logoColor=white&style=flat)](https://insights.linuxfoundation.org/project/microcks/repository/microcks-microcks) [![LFX Active Contributors](https://img.shields.io/static/v1?label=Active%20contributors%20(1Y)&message=208&color=0094FF&logo=linuxfoundation&logoColor=white&style=flat)](https://insights.linuxfoundation.org/project/microcks/repository/microcks-microcks)

## Getting Started

* [Documentation](https://microcks.io/documentation/tutorials/getting-started/)
* [Microcks Community](https://github.com/microcks/community) and community meeting

To get involved with our community, please familiarize yourself with the project's [Code of Conduct](./CODE_OF_CONDUCT.md).

## Build Status

The current development version is `1.13.3-SNAPSHOT` on branch `1.13.x`. 

[![GitHub Workflow Status](https://img.shields.io/github/actions/workflow/status/microcks/microcks/build-verify.yml?branch=1.11.x&logo=github&style=for-the-badge)](https://github.com/microcks/microcks/actions)

#### Sonarcloud Quality metrics

[![Code Smells](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=code_smells)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)
[![Reliability Rating](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=reliability_rating)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)
[![Bugs](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=bugs)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)
[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=coverage)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)
[![Technical Debt](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=sqale_index)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)
[![Security Rating](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=security_rating)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)
[![Maintainability Rating](https://sonarcloud.io/api/project_badges/measure?project=microcks_microcks&metric=sqale_rating)](https://sonarcloud.io/summary/new_code?id=microcks_microcks)

#### Fossa license and security scans

[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fmicrocks%2Fmicrocks.svg?type=shield&issueType=license)](https://app.fossa.com/projects/git%2Bgithub.com%2Fmicrocks%2Fmicrocks?ref=badge_shield&issueType=license)
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fmicrocks%2Fmicrocks.svg?type=shield&issueType=security)](https://app.fossa.com/projects/git%2Bgithub.com%2Fmicrocks%2Fmicrocks?ref=badge_shield&issueType=security)
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2Fmicrocks%2Fmicrocks.svg?type=small)](https://app.fossa.com/projects/git%2Bgithub.com%2Fmicrocks%2Fmicrocks?ref=badge_small)

#### Signature, Provenance, SBOM

[![Static Badge](https://img.shields.io/badge/supply_chain-documentation-blue?logo=securityscorecard&label=Supply%20Chain&link=https%3A%2F%2Fmicrocks.io%2Fdocumentation%2Freferences%2Fcontainer-images%23software-supply-chain-security)](https://microcks.io/documentation/references/container-images#software-supply-chain-security)

#### OpenSSF best practices

[![CII Best Practices](https://bestpractices.coreinfrastructure.org/projects/7513/badge)](https://bestpractices.coreinfrastructure.org/projects/7513)
[![OpenSSF Scorecard](https://api.securityscorecards.dev/projects/github.com/microcks/microcks/badge)](https://securityscorecards.dev/viewer/?uri=github.com/microcks/microcks)


## Versions

Here are the naming conventions we're using for current releases, ongoing development maintenance activities.

| Status      | Version           | Branch   | Container images tags |
| ----------- |-------------------|----------|-----------------------|
| Stable      | `1.13.2`          | `master` | `1.13.2`, `latest`    |
| Dev         | `1.13.3-SNAPSHOT` | `1.13.x` | `nightly`             |
| Maintenance | `1.12.2-SNAPSHOT` | `1.12.x` | `maintenance`         |

Have a look at our [tested configurations](TESTED_CONFIGURATIONS.md) to know more about the versions of dependencies 
and integrations that we are using for development and testing.

## How to build Microcks

The build instructions are available in the [building guide](BUILDING.md).

## Thanks to the community!

[![Stargazers repo roster for @microcks/microcks](http://reporoster.com/stars/microcks/microcks)](http://github.com/microcks/microcks/stargazers)
[![Forkers repo roster for @microcks/microcks](http://reporoster.com/forks/microcks/microcks)](http://github.com/microcks/microcks/network/members)
 readmeEtag: '"58cdcc7aa20743b1d9324902a023fd73ac38d086"' readmeLastModified: Thu, 08 Jan 2026 08:04:35 GMT repositoryId: 31214867 description: >- The open source, cloud native tool for API Mocking and Testing. Microcks is a Cloud Native Computing Foundation sandbox project 🚀 created: '2015-02-23T15:46:09Z' updated: '2026-02-05T03:48:23Z' language: Java archived: false stars: 1798 watchers: 18 forks: 312 owner: microcks logo: https://avatars.githubusercontent.com/u/11051048?v=4 license: Apache-2.0 repoEtag: '"f7b98eacd56020c6da6da9def3a0bb25cbf1cec5102c897f823a87e5aaf7324d"' repoLastModified: Thu, 05 Feb 2026 03:48:23 GMT foundInMaster: true id: 1c0e918b5a733b0f1726199e3ea101d6 - source: https://openapi.tools/ name: OpenAPI Mocker category: - Mock - Parsers language: nodejs link: https://www.npmjs.com/package/open-api-mocker repository: https://github.com/jormaechea/open-api-mocker source_description: >- Standalone nodejs based OpenAPI 3 mock server, docker-friendly with request validation and autoreload. v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIE1vY2tlcgoKIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9qb3JtYWVjaGVhL29wZW4tYXBpLW1vY2tlci93b3JrZmxvd3MvYnVpbGQvYmFkZ2Uuc3ZnKQpbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL29wZW4tYXBpLW1vY2tlci5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuLWFwaS1tb2NrZXIpClshW01haW50YWluYWJpbGl0eV0oaHR0cHM6Ly9hcGkuY29kZWNsaW1hdGUuY29tL3YxL2JhZGdlcy83OWY2ZWNhN2VhM2Y4ZmU1NTRjMi9tYWludGFpbmFiaWxpdHkpXShodHRwczovL2NvZGVjbGltYXRlLmNvbS9naXRodWIvam9ybWFlY2hlYS9vcGVuLWFwaS1tb2NrZXIvbWFpbnRhaW5hYmlsaXR5KQpbIVtUZXN0IENvdmVyYWdlXShodHRwczovL2FwaS5jb2RlY2xpbWF0ZS5jb20vdjEvYmFkZ2VzLzc5ZjZlY2E3ZWEzZjhmZTU1NGMyL3Rlc3RfY292ZXJhZ2UpXShodHRwczovL2NvZGVjbGltYXRlLmNvbS9naXRodWIvam9ybWFlY2hlYS9vcGVuLWFwaS1tb2NrZXIvdGVzdF9jb3ZlcmFnZSkKWyFbRG9ja2VyIGNvbXBhdGlibGVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZG9ja2VyLWNvbXBhdGlibGUtZ3JlZW4pXShodHRwczovL2h1Yi5kb2NrZXIuY29tL3JlcG9zaXRvcnkvZG9ja2VyL2pvcm1hZWNoZWEvb3Blbi1hcGktbW9ja2VyKQoKQW4gQVBJIG1vY2tlciBiYXNlZCBpbiB0aGUgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbi4KCiMjIEluc3RhbGxhdGlvbiBhbmQgdXNhZ2UKCiMjIyBVc2luZyBucG0KCmBgYApucG0gaSAtZyBvcGVuLWFwaS1tb2NrZXIKCm9wZW4tYXBpLW1vY2tlciAtcyBteS1zY2hlbWEuanNvbiAtdwoKb3Blbi1hcGktbW9ja2VyIC0taGVscCAjIFRvIHByb21wdCBldmVyeSBhdmFpbGFibGUgc2V0dGluZy4KYGBgCgojIyMgVXNpbmcgZG9ja2VyCgpgYGAKZG9ja2VyIHJ1biAtdiAiJFBXRC9teXNjaGVtYS5qc29uOi9hcHAvc2NoZW1hLmpzb24iIC1wICI1MDAwOjUwMDAiIGpvcm1hZWNoZWEvb3Blbi1hcGktbW9ja2VyCmBgYAoKT3IgdG8gcnVuIGFuIHNwZWNpZmljIHZlcnNpb24KCmBgYApkb2NrZXIgcnVuIC12ICIkUFdEL215c2NoZW1hLmpzb246L2FwcC9zY2hlbWEuanNvbiIgLXAgIjUwMDA6NTAwMCIgam9ybWFlY2hlYS9vcGVuLWFwaS1tb2NrZXI6WC5ZLlpgCmBgYAoKWW91IGNhbiBzZXQgYW55IHBhcmFtZXRlciB3aGVuIHJ1bm5pbmcgaW5zaWRlIGEgZG9ja2VyIGNvbnRhaW5lcgoKYGBgCmRvY2tlciBydW4gLXYgIiRQV0QvbXlzY2hlbWEuanNvbjovYXBwL3NjaGVtYS5qc29uIiAtcCAiMzAwMDozMDAwIiBqb3JtYWVjaGVhL29wZW4tYXBpLW1vY2tlcjpYLlkuWiAtcyAvYXBwL3NjaGVtYS5qc29uIC1wIDMwMDBgCmBgYAoKIyMgQ2FwYWJpbGl0aWVzCgotIFt4XSBSZWFkIHlhbWwgYW5kIGpzb24gT3BlbkFQSSB2MyBzY2hlbWFzLgotIFt4XSBQb3J0IGJpbmRpbmcgc2VsZWN0aW9uCi0gW3hdIFJlcXVlc3QgcGFyYW1ldGVycyB2YWxpZGF0aW9uCi0gW3hdIFJlcXVlc3QgYm9keSB2YWxpZGF0aW9uCi0gW3hdIFJlc3BvbnNlIGJvZHkgYW5kIGhlYWRlcnMgZ2VuZXJhdGlvbiBiYXNlZCBvbiBleGFtcGxlcyBvciBzY2hlbWFzCi0gW3hdIFJlc3BvbnNlIHNlbGVjdGlvbiB1c2luZyByZXF1ZXN0IGhlYWRlcjogYFByZWZlcjogc3RhdHVzQ29kZT1YWFhgIG9yIGBQcmVmZXI6IGV4YW1wbGU9bmFtZWAKLSBbeF0gUmVxdWVzdCBhbmQgcmVzcG9uc2UgbG9nZ2luZwotIFt4XSBTZXJ2ZXJzIGJhc2VwYXRoIHN1cHBvcnQKLSBbeF0gU3VwcG9ydCB4LWZha2VyIGFuZCB4LWNvdW50IGV4dGVuc2lvbiBtZXRob2RzIHRvIGN1c3RvbWlzZSBnZW5lcmF0ZWQgcmVzcG9uc2VzCi0gWyBdIEFQSSBBdXRoZW50aWNhdGlvbgoKIyMgQ3VzdG9taXppbmcgR2VuZXJhdGVkIFJlc3BvbnNlcwpUaGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGFsbG93cyBjdXN0b20gcHJvcGVydGllcyB0byBiZSBhZGRlZCB0byBhbiBBUEkgZGVmaW5pdGlvbiBpbiB0aGUgZm9ybSBvZiBfeC0qXy4KT3BlbkFQSSBNb2NrZXIgc3VwcG9ydHMgdGhlIHVzZSBvZiB0d28gY3VzdG9tIGV4dGVuc2lvbnMgdG8gYWxsb3cgZGF0YSB0byBiZSByYW5kb21pc2VkIHdoaWNoIHNob3VsZCBhbGxvdyBmb3IgbW9yZQpyZWFsaXN0aWMgbG9va2luZyBkYXRhIHdoZW4gZGV2ZWxvcGluZyBhIFVJIGFnYWluc3QgYSBtb2NrIEFQSSBmb3IgaW5zdGFuY2UuCgojIyMgeC1mYWtlcgpUaGUgX3gtZmFrZXJfIGV4dGVuc2lvbiBpcyB2YWxpZCBmb3IgdXNlIG9uIHByb3BlcnRpZXMgdGhhdCBoYXZlIGEgcHJpbWl0aXZlIHR5cGUgKGUuZy4gYHN0cmluZ2AvYGludGVnZXJgLCBldGMuKQphbmQgY2FuIGJlIHVzZWQgd2l0aGluIGFuIEFQSSBkZWZpbml0aW9uIHRvIHVzZSBvbmUgb3IgbW9yZSBtZXRob2RzIGZyb20gdGhlIGNvbW11bml0eSBtYW50YWluZWQKW0Zha2VyXShodHRwczovL2Zha2VyanMuZGV2LykgbGlicmFyeSBmb3IgZ2VuZXJhdGluZyByYW5kb20gZGF0YS4KCkdpdmVuIHRoZSBmb2xsb3dpbmcgQVBJIGRlZmluaXRpb246CmBgYHlhbWwKb3BlbmFwaTogJzMuMC4yJwppbmZvOgogIHRpdGxlOiBDYXRzCiAgdmVyc2lvbjogJzEuMCcKc2VydmVyczoKICAtIHVybDogaHR0cHM6Ly9hcGkuY2F0cy50ZXN0L3YxCnBhdGhzOgogIC9jYXQ6CiAgICBnZXQ6CiAgICAgIHJlc3BvbnNlczoKICAgICAgICAnMjAwJzoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBPSwogICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICB0eXBlOiBvYmplY3QKICAgICAgICAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgICAgICAgIGZpcnN0TmFtZToKICAgICAgICAgICAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgICAgICAgICAgICB4LWZha2VyOiBwZXJzb24uZmlyc3ROYW1lCiAgICAgICAgICAgICAgICAgIGxhc3ROYW1lOgogICAgICAgICAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgICAgICAgICAgIHgtZmFrZXI6IHBlcnNvbi5sYXN0TmFtZQogICAgICAgICAgICAgICAgICBmdWxsTmFtZToKICAgICAgICAgICAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgICAgICAgICAgICB4LWZha2VyOiAne3twZXJzb24uZmlyc3ROYW1lfX0ge3twZXJzb24ubGFzdE5hbWV9fScKICAgICAgICAgICAgICAgICAgYWdlOgogICAgICAgICAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgICAgICAgICAgIHgtZmFrZXI6ICdudW1iZXIuaW50KHsgIm1pbiI6IDEsICJtYXgiOiAyMCB9KScKCmBgYAoKQSBKU09OIHJlc3BvbnNlIHNpbWlsYXIgdG8gdGhlIGZvbGxvd2luZyB3b3VsZCBiZSBwcm9kdWNlZDoKYGBgSlNPTgp7CiAgICAiZmlyc3ROYW1lIjogIlRlZCIsCiAgICAibGFzdE5hbWUiOiAiS296ZXkiLAogICAgImZ1bGxOYW1lIjogIkhlcmJlcnQgTG93ZSIsCiAgICAiYWdlIjogMTIKfQpgYGAKClRoZSBfeC1mYWtlcl8gZXh0ZW5zaW9uIGFjY2VwdHMgdmFsdWVzIGluIDMgZm9ybXM6CjEuIF9mYWtlck5hbWVzcGFjZS5tZXRob2RfLiBlLmcuIGBzdHJpbmcudXVpZGAKMi4gX2Zha2VyTmFtZXNwYWNlLm1ldGhvZCh7ICJtZXRob2RBcmdzIjogImluIiwgImpzb24iOiAiZm9ybWF0IiB9KV8uIGUuZy4gYG51bWJlci5pbnQoeyAibWF4IjogMTAwIH0pYAozLiBBIG11c3RhY2hlIHRlbXBsYXRlIHN0cmluZyBtYWtpbmcgdXNlIG9mIHRoZSAyIGZvcm1zIGFib3ZlLiBlLmcuIGBNeSBuYW1lIGlzIHt7cGVyc29uLmZpcnN0TmFtZX19IHt7cGVyc29uLmxhc3ROYW1lfX1gCgoqTk9URSo6IFRvIGF2b2lkIG5ldyBmYWtlIGRhdGEgZnJvbSBiZWluZyBnZW5lcmF0ZWQgb24gZXZlcnkgY2FsbCwgdXAgdG8gMTAgcmVzcG9uc2VzIHBlciBlbmRwb2ludCBhcmUgY2FjaGVkCmJhc2VkIG9uIHRoZSBpbmNvbWluZyBxdWVyeSBzdHJpbmcsIHJlcXVlc3QgYm9keSBhbmQgaGVhZGVycy4KCiMjIyB4LWNvdW50ClRoZSBfeC1jb3VudF8gZXh0ZW5zaW9uIGhhcyBlZmZlY3Qgb25seSB3aGVuIHVzZWQgb24gYW4gYGFycmF5YCB0eXBlIHByb3BlcnR5LgpJZiBlbmNvdW50ZXJlZCwgT3BlbkFQSSBNb2NrZXIgd2lsbCByZXR1cm4gYW4gYXJyYXkgd2l0aCB0aGUgZ2l2ZW4gbnVtYmVyIG9mIGVsZW1lbnRzIGluc3RlYWQgb2YgdGhlIGRlZmF1bHQgb2YgYW4KYXJyYXkgd2l0aCBhIHNpbmdsZSBpdGVtLgoKRm9yIGV4YW1wbGUsIHRoZSBmb2xsb3dpbmcgQVBJIGRlZmluaXRpb246CmBgYHlhbWwKb3BlbmFwaTogJzMuMC4yJwppbmZvOgogIHRpdGxlOiBDYXRzCiAgdmVyc2lvbjogJzEuMCcKc2VydmVyczoKICAtIHVybDogaHR0cHM6Ly9hcGkuY2F0cy50ZXN0L3YxCnBhdGhzOgogIC9jYXQ6CiAgICBnZXQ6CiAgICAgIHJlc3BvbnNlczoKICAgICAgICAnMjAwJzoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBPSwogICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICB0eXBlOiBhcnJheQogICAgICAgICAgICAgICAgeC1jb3VudDogNQogICAgICAgICAgICAgICAgaXRlbXM6CiAgICAgICAgICAgICAgICAgIHR5cGU6IHN0cmluZwpgYGAKCldpbGwgcHJvZHVjZSB0aGUgZm9sbG93aW5nIHJlc3BvbnNlOgpgYGBKU09OClsKICAgICJzdHJpbmciLAogICAgInN0cmluZyIsCiAgICAic3RyaW5nIiwKICAgICJzdHJpbmciLAogICAgInN0cmluZyIKXQpgYGAKCiMjIEFkdmFuY2VkIHVzYWdlCgpTZWUgdGhlIFthZHZhbmNlZCB1c2FnZSBkb2NzXShkb2NzL1JFQURNRS5tZCkgdG8gZXh0ZW5kIG9yIGJ1aWxkIHlvdXIgb3duIGFwcCB1cG9uIE9wZW5BUEkgTW9ja2VyLgoKIyMgVGVzdHMKClNpbXBseSBydW4gYG5wbSB0YAoKIyMgQ29udHJpYnV0aW5nCgpJc3N1ZXMgYW5kIFBScyBhcmUgd2VsY29tZS4K readmeEtag: '"e4336bc8f8f199bdf64e37116044d8f822da68ef"' readmeLastModified: Tue, 30 May 2023 23:48:16 GMT repositoryId: 193262126 description: A mock server based in OpenAPI Specification created: '2019-06-22T17:38:08Z' updated: '2025-11-05T23:21:40Z' language: JavaScript archived: false stars: 126 watchers: 4 forks: 34 owner: jormaechea logo: https://avatars.githubusercontent.com/u/5612500?v=4 license: MIT repoEtag: '"17ff34a9a5e2caa911a5a5d588592041315011cdbd230609d107b4e992535493"' repoLastModified: Wed, 05 Nov 2025 23:21:40 GMT foundInMaster: true id: 025e0dafdc92cfc32c40f410a4c333b7 - source: https://openapi.tools/ name: Wiremock category: - Mock - Testing - Data Validators - Documentation language: SaaS link: https://www.wiremock.io/product repository: https://www.wiremock.io/product source_description: >- WireMock Cloud is a managed, hosted version of WireMock, developed by the same team who wrote the open-source project. It is built on the same technology that powers open source WireMock and is 100% compatible with the WireMock API, with additional features that make it quick and easy to mock any API you depend on. WireMock Cloud also introduces advanced capabilities such as chaos engineering, OpenAPI generation, validation and documentation as well as better collaboration and user management. v2: true v3: true v3_1: true id: f6eb4e8c5fd7bad5d57f440b0ae64ae1 foundInMaster: true - source: https://openapi.tools/ name: Fakeit category: - Mock - Parsers language: - cli - Docker link: https://github.com/justinfeng/fakeit repository: https://github.com/justinfeng/fakeit source_description: >- Create mock server from OpenAPI 3 specification with random response generation and request validation. v3: true repositoryMetadata: base64Readme: >- IyBGYWtlaXQKCltGdWt1b2thIFJ1YnkgQXdhcmQgLSBTcGVjaWFsIEF3YXJkXShodHRwczovL3d3dy5kaWdpdGFsZnVrdW9rYS5qcC9ldmVudF9yZXBvcnRzLzIxP2xvY2FsZT1qYSkKClshW0NpcmNsZUNJXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NpcmNsZWNpL2J1aWxkL2dpdGh1Yi9KdXN0aW5GZW5nL2Zha2VpdC5zdmcpXShodHRwczovL2NpcmNsZWNpLmNvbS9naC9KdXN0aW5GZW5nL2Zha2VpdCkKWyFbQ29kZSBDbGltYXRlIG1haW50YWluYWJpbGl0eV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9jb2RlY2xpbWF0ZS9tYWludGFpbmFiaWxpdHkvSnVzdGluRmVuZy9mYWtlaXQuc3ZnKV0oaHR0cHM6Ly9jb2RlY2xpbWF0ZS5jb20vZ2l0aHViL0p1c3RpbkZlbmcvZmFrZWl0KQpbIVtHZW1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2VtL3YvZmFrZWl0LnN2ZyldKGh0dHBzOi8vcnVieWdlbXMub3JnL2dlbXMvZmFrZWl0KQpbIVtHZW1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2VtL2R0L2Zha2VpdC5zdmcpXShodHRwczovL3J1YnlnZW1zLm9yZy9nZW1zL2Zha2VpdCkKWyFbRG9ja2VyIFB1bGxzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2RvY2tlci9wdWxscy9yZWFsZmVuZ2ppYS9mYWtlaXQuc3ZnKV0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL3JlYWxmZW5namlhL2Zha2VpdCkKWyFbR2l0SHViXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL0p1c3RpbkZlbmcvZmFrZWl0LnN2ZyldKGh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUKQoKQ3JlYXRlIG1vY2sgc2VydmVyIGZyb20gT3BlbmFwaSBzcGVjaWZpY2F0aW9uCgojIyBNb3RpdmF0aW9uCgpPcGVuYXBpIG1vY2sgc2VydmVyIGlzIG9uZSBvZiBjb3JlIGNvbXBvbmVudHMgdG8gc3VwcG9ydCBjb250cmFjdCBiYXNlZCBkZXZlbG9wbWVudCBhbmQgdGVzdGluZy4gQXMgcGFydCBvZiBvdXIgam91cm5hbCwgc2V2ZXJhbCBrZXkgcmVxdWlyZW1lbnRzIGZvciBzdWNoIG1vY2sgc2VydmVyIGhhdmUgYmVlbiBpZGVudGlmaWVkOgoKKiBDb250cm9sIHJlc3BvbnNlIGdlbmVyYXRpb24gaW4gbm9uIGludHJ1c2l2ZSBtYW5uZXIuIGkuZS4gd2l0aG91dCBtb2RpZnlpbmcgZXhhbXBsZSBpbiBjb250cmFjdAoqIFJhbmRvbWx5IGdlbmVyYXRlZCByZXNwb25zZSB0byBzdXBwb3J0IHByb3BlcnR5IGJhc2VkIHRlc3RpbmcKKiBGdWxmaWxsIHByb3BlcnR5IHJlZmVyZW5jZSBpbiByZXNwb25zZSBnZW5lcmF0aW9uIHRvIHN1cHBvcnQgZGV2ZWxvcG1lbnQgYWdhaW5zdCBjb250cmFjdC4gaS5lLiByZWdhcmRpbmcgdGhlIGZvbGxvd2luZyByZXNwb25zZSwgZ3VhcmFudGVlIHRoZSBgc2VsZWN0ZWRJZGAgcHJvcGVydHkgaXMgYWx3YXlzIGEgdmFsaWQgYGlkYCBpbiB0aGUgaXRlbXMKYGBganNvbgp7CiAgInNlbGVjdGVkSWQiOiAxLAogICJpdGVtcyI6IFsKICAgIHsgImlkIjogMSB9LAogICAgeyAiaWQiOiAyIH0KICBdCn0KYGBgCgpBZnRlciB0cmllZCBzZXZlcmFsIGV4aXN0aW5nIG9wdGlvbnMsIHdlIGNhbm5vdCBmaW5kIGEgYmVzdCBzb2x1dGlvbiB0byBtZWV0IGFsbCB0aGUgcmVxdWlyZW1lbnRzLiBTbyB3ZSBlbmQgdXAgd2l0aCBfX0Zha2VpdF9fLgoKIyMgRmVhdHVyZXMKCiogUmFuZG9tbHkgb3Igc3RhdGljYWxseSBnZW5lcmF0ZWQgcmVzcG9uc2UgKGBhcHBsaWNhdGlvbi8uKmpzb25gKQoqIFJlcXVlc3QgdmFsaWRhdGlvbiAoYGFwcGxpY2F0aW9uLy4qanNvbmAsIGBtdWx0aXBhcnQvZm9ybS1kYXRhYCkKKiBMb2FkIHNwZWNpZmljYXRpb24gZnJvbSBsb2NhbCBvciByZW1vdGUKKiBTdXBwb3J0IGhvdCByZWxvYWQgbG9jYWwgc3BlY2lmaWNhdGlvbgoqIFN1cHBvcnQgc3BlY2lmaWNhdGlvbiBpbiB5YW1sIG9yIGpzb24gZm9ybWF0CiogW0V4cGVyaW1lbnRhbF0gU3VwcG9ydCBwbGFpbiB0ZXh0IGFuZCBiaW5hcnkgcmVzcG9uc2UsIGUuZy4gYGFwcGxpY2F0aW9uL3BkZmAKCiMjIEluc3RhbGxhdGlvbgoKSW5zdGFsbCBpdCB3aXRoOgoKICAgICQgZ2VtIGluc3RhbGwgZmFrZWl0CgpPciB1c2UgdGhlIFtkb2NrZXIgaW1hZ2VdKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9yZWFsZmVuZ2ppYS9mYWtlaXQpCgojIyBVc2FnZQoKICAgICQgZmFrZWl0IC0tc3BlYyA8TG9jYWwgZmlsZSBvciByZW1vdGUgdXJsPgoKIyMjIENvbW1hbmQgbGluZSBvcHRpb25zCgogICAgJCBmYWtlaXQgLS1oZWxwCiAgICB1c2FnZToKICAgICAgICAtLXNwZWMgICAgICAgICAgICAgICBzcGVjIGZpbGUgdXJpIChyZXF1aXJlZCkKICAgICAgICAtcCwgLS1wb3J0ICAgICAgICAgICBjdXN0b20gcG9ydAogICAgICAgIC1xLCAtLXF1aWV0ICAgICAgICAgIG11dGUgcmVxdWVzdCBhbmQgcmVzcG9uc2UgbG9nCiAgICAgICAgLWwsIC0tbG9nLWZpbGUgICAgICAgcmVkaXJlY3QgbG9nIHRvIGEgZmlsZQogICAgICAgIC0tcGVybWlzc2l2ZSAgICAgICAgIGxvZyB2YWxpZGF0aW9uIGVycm9yIGFzIHdhcm5pbmcgaW5zdGVhZCBvZiBkZW55aW5nIHJlcXVlc3QKICAgICAgICAtLXVzZS1leGFtcGxlICAgICAgICB1c2UgZXhhbXBsZSBwcm92aWRlZCBpbiBzcGVjIGlmIGV4aXN0cwogICAgICAgIC0tc3RhdGljICAgICAgICAgICAgIGdlbmVyYXRlIHN0YXRpYyByZXNwb25zZQogICAgICAgIC0tc3RhdGljLXR5cGVzICAgICAgIGdlbmVyYXRlIHN0YXRpYyB2YWx1ZSBmb3Igc3BlY2lmaWVkIHR5cGVzLCBlLmcuIC0tc3RhdGljLXR5cGVzIGludGVnZXIsc3RyaW5nCiAgICAgICAgLS1zdGF0aWMtcHJvcGVydGllcyAgZ2VuZXJhdGUgc3RhdGljIHZhbHVlIGZvciBzcGVjaWZpZWQgcHJvcGVydGllcywgZS5nLiAtLXN0YXRpYy1wcm9wZXJ0aWVzIGlkLHV1aWQKICAgICAgICAtLWJhc2UtcGF0aCAgICAgICAgICBtb3VudHMgdGhlIG1vY2sgc2VydmVyIGF0IHRoZSBnaXZlbiBwYXRoLCBlLmcuIC0tYmFzZS1wYXRoIC9hcGkKCiAgICBvdGhlciBvcHRpb25zOgogICAgICAgIC12LCAtLXZlcnNpb24KICAgICAgICAtaCwgLS1oZWxwCgoqKk5vdGVzOioqCiogU2VlIFtoZXJlXShkb2NzL3JhbmRvbS5tZCkgZm9yIE9wZW5hcGkgcHJvcGVydGllcyBzdXBwb3J0ZWQgaW4gcmFuZG9tIHJlc3BvbnNlIGdlbmVyYXRpb24KKiBTZWUgW2hlcmVdKGRvY3Mvc3RhdGljLm1kKSBmb3IgZGVmYXVsdCB2YWx1ZSBpbiBzdGF0aWMgcmVzcG9uc2UgZ2VuZXJhdGlvbgoqIFJlZ2FyZGluZyBgLS11c2UtZXhhbXBsZWAgbW9kZSwgcHJvcGVydHkgd2l0aG91dCBleGFtcGxlIHNwZWNpZmllZCB3aWxsIHN0aWxsIGJlIHJhbmRvbWx5IG9yIHN0YXRpY2FsbHkgZ2VuZXJhdGVkCiogUmFuZG9tIHJlc3BvbnNlIGdlbmVyYXRpb24gY2FuIG5vdCBoYW5kbGUgcmVjdXJzaXZlIHNjaGVtYSByZWZlcmVuY2UuIElmIHlvdSBkbyBuZWVkIGl0IGluIHlvdXIgc3BlYyBmaWxlLCBwbGVhc2UgcHJvdmlkZSBgZXhhbXBsZWAgcHJvcGVydHkgZm9yIHRoZSByZWN1cnNpdmUgcGFydCBvZiBzY2hlbWEgYW5kIHNwZWNpZnkgYC0tdXNlLWV4YW1wbGVgIG9wdGlvbi4KCiMjIyBDb25maWd1cmF0aW9uIGVuZHBvaW50CgpNb2NrIHNlcnZlciBiZWhhdmlvdXIgY2FuIGJlIGNoYW5nZWQgb24gdGhlIGZseQoKUmV0cmlldmUgY3VycmVudCBjb25maWc6CgogICAgR0VUIC9fX2Zha2VpdF9jb25maWdfXwoKVXBkYXRlIGNvbmZpZzoKCiAgICBQVVQgL19fZmFrZWl0X2NvbmZpZ19fCgpSZXF1ZXN0IGFuZCByZXNwb25zZToKYGBganNvbgp7CiAgInBlcm1pc3NpdmUiOiBmYWxzZSwKICAidXNlX2V4YW1wbGUiOiB0cnVlLAogICJzdGF0aWMiOiBmYWxzZSwKICAic3RhdGljX3R5cGVzIjogWwogICAgImludGVnZXIiCiAgXSwKICAic3RhdGljX3Byb3BlcnRpZXMiOiBbCiAgICAiaWQiCiAgXSwKICAiYmFzZV9wYXRoIjogIi9hcGkiCn0KYGBgCgojIyBEZXZlbG9wbWVudAoKQWZ0ZXIgY2hlY2tpbmcgb3V0IHRoZSByZXBvLCBydW4gYGJ1bmRsZSBpbnN0YWxsYCB0byBpbnN0YWxsIGRlcGVuZGVuY2llcy4gVGhlbiwgcnVuIGByYWtlYCB0byBydW4gdGhlIHRlc3RzLgoKVG8gaW5zdGFsbCB0aGlzIGdlbSBvbnRvIHlvdXIgbG9jYWwgbWFjaGluZSwgcnVuIGBidW5kbGUgZXhlYyByYWtlIGluc3RhbGxgLgoKVG8gcmVsZWFzZSBhIG5ldyB2ZXJzaW9uLCB1cGRhdGUgdGhlIHZlcnNpb24gbnVtYmVyIGluIGB2ZXJzaW9uLnJiYCwgYW5kIHRoZW4gcnVuIGBidW5kbGUgZXhlYyByYWtlIHJlbGVhc2VgLCB3aGljaCB3aWxsIGNyZWF0ZSBhIGdpdCB0YWcgZm9yIHRoZSB2ZXJzaW9uLCBwdXNoIGdpdCBjb21taXRzIGFuZCB0YWdzLCBhbmQgcHVzaCB0aGUgYC5nZW1gIGZpbGUgdG8gW3J1YnlnZW1zLm9yZ10oaHR0cHM6Ly9ydWJ5Z2Vtcy5vcmcpLgoKIyMgQ29udHJpYnV0aW5nCgpCdWcgcmVwb3J0cyBhbmQgcHVsbCByZXF1ZXN0cyBhcmUgd2VsY29tZSBvbiBHaXRIdWIgYXQgaHR0cHM6Ly9naXRodWIuY29tL0p1c3RpbkZlbmcvZmFrZWl0LgoKIyMgTGljZW5zZQoKVGhlIGdlbSBpcyBhdmFpbGFibGUgYXMgb3BlbiBzb3VyY2UgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbTUlUIExpY2Vuc2VdKGh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUKS4K readmeEtag: '"527c50eea9720a5f527a8f03d1df652a9162a865"' readmeLastModified: Fri, 02 Aug 2024 01:56:10 GMT repositoryId: 179582526 description: Create mock server from Openapi specification created: '2019-04-04T21:51:59Z' updated: '2025-11-09T08:45:47Z' language: Ruby archived: false stars: 80 watchers: 3 forks: 5 owner: JustinFeng logo: https://avatars.githubusercontent.com/u/1527902?v=4 license: MIT repoEtag: '"ea981a691646f370bf54b830e50aefed08c43ed6bb087b5d24e1b3e59f8e1681"' repoLastModified: Sun, 09 Nov 2025 08:45:47 GMT foundInMaster: true id: 6b67af7dabca973abebd21b956d683e7 - source: https://openapi.tools/ name: Mockintosh category: Mock language: - CLI - Docker link: https://mockintosh.io source_description: > Mocks for CloudNative Environments - Converts OpenAPI files to Mocks and use them to develop in isolated environments and test edge cases, Async call to queues such as Kafka or RabbitMQ or simulate performance & chaos testing v2: true v3: true v3_1: true foundInMaster: true id: c51e06db0cd86207ca6bf9b9a9e781f1 - source: https://openapi.tools/ name: openapi-data-mocker category: - Mock - Parsers language: PHP repository: https://github.com/ybelenko/openapi-data-mocker source_description: > Tiny library to generate basic OpenAPI Data Types. Consider it as extended Faker package. First version able to mock most of the data formats. It doesn't support polymorphism yet, but work in progress. May be useful for writing custom unit tests. v2: false v3: true repositoryMetadata: base64Readme: >- # Openapi Data Mocker

[![Latest Stable Version](https://poser.pugx.org/ybelenko/openapi-data-mocker/v/stable)](https://packagist.org/packages/ybelenko/openapi-data-mocker)
[![Build Status](https://github.com/ybelenko/openapi-data-mocker/actions/workflows/ci.yml/badge.svg)](https://github.com/ybelenko/openapi-data-mocker/actions/workflows/ci.yml)
[![Coverage Status](https://coveralls.io/repos/github/ybelenko/openapi-data-mocker/badge.svg?branch=master)](https://coveralls.io/github/ybelenko/openapi-data-mocker?branch=master)
[![License](https://poser.pugx.org/ybelenko/openapi-data-mocker/license)](https://packagist.org/packages/ybelenko/openapi-data-mocker)

Openapi Data Mocker helps to generate fake data from OpenAPI 3.0 documents. Most of the methods may work with 2.0 version(fka Swagger 2.0), but it's not tested. This package was an enhancement of PHP Slim4 server in [OpenAPI Generator](https://github.com/OpenAPITools/openapi-generator) project, but it easier to maintain it in separated repo.

## Requirements
- PHP ^7.3

__Important notice! While PHP 8.0 declared in composer.json this package hasn't been tested against it.__

## Installation via Composer

Run in terminal:
```console
composer require ybelenko/openapi-data-mocker
```

## Usage example

Imagine we have [OpenAPI Specification 3.0.3 - Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md#schema-object) like this:
```yaml
description: Real world example schema
type: object
properties:
  id:
    type: integer
    format: int32
    minimum: 1
  purchased_items:
    type: array
    items:
      type: object
      properties:
        SKU:
          type: string
          format: uuid
          maxLength: 20
        quantity:
          type: integer
          format: int32
          minimum: 1
          maximum: 5
        price:
          type: object
          properties:
            currency:
              type: string
              minLength: 3
              maxLength: 3
              enum:
              - USD
              - EUR
              - RUB
            value:
              type: number
              format: float
              minimum: 0.01
              maximum: 99.99
        manufacturer:
          type: object
          properties:
            name:
              type: string
              maxLength: 30
            country:
              type: string
              enum:
              - CHN
              - USA
              - RUS
  buyer:
    type: object
    properties:
      first_name:
        type: string
        minLength: 3
        maxLength: 15
      last_name:
        type: string
        minLength: 3
        maxLength: 15
      credit_card:
        type: integer
        minimum: 1000000000000000
        maximum: 10000000000000000
      phone:
        type: integer
        minimum: 10000000000000
        maximum: 99999999999999
      email:
        type: string
        format: email
  status:
    type: string
    enum:
    - registered
    - paid
    - shipped
    - delivered
    default: registered
  created_at:
    type: string
    format: date-time
```
> Notice! While schema object presented in YAML format this library doesn't support YAML or JSON parsing right now. It means that `mockSchemaObject` method expects already decoded JSON value as argument.

When we mock mentioned schema with `mockSchemaObject` method:
```php
require __DIR__ . '/vendor/autoload.php';

use OpenAPIServer\Mock\OpenApiDataMocker as Mocker;
$mocker = new Mocker();
// set model classes namespace for $ref handling
// current example doesn't use $refs in schemas, however
$mocker->setModelsNamespace('JohnDoesPackage\\Model\\');
// class InvoiceTest contains schema mentioned previously
// it returns that schema with getOpenApiSchema() method declared in OpenAPIServer\Mock\BaseModel parent class
$schema = \OpenAPIServer\Mock\Model\InvoiceTest::getOpenApiSchema();
$data = $mocker->mockSchemaObject($schema);
echo json_encode($data, \JSON_PRETTY_PRINT);
```

the output looks like:
```json
{
    "id": 1912777939,
    "purchased_items": [
        {
            "SKU": "5ee78cfde9f05",
            "quantity": 4,
            "price": {
                "currency": "EUR",
                "value": 57.635
            },
            "manufacturer": {
                "name": "Lorem i",
                "country": "USA"
            }
        }
    ],
    "buyer": {
        "first_name": "Lorem ipsum do",
        "last_name": "Lorem ipsum ",
        "credit_card": 2455087473915908,
        "phone": 65526260517693,
        "email": "jfkennedy@example.com"
    },
    "status": "delivered",
    "created_at": "1978-08-08T04:03:09+00:00"
}
```
Of course that output will be slightly different on every call. That's what mocker package has been developed for.

You can check extended example at [examples/extended_example.php](examples/extended_example.php).

## Supported features

All data types supported except specific string formats: `email`, `uuid`, `password` which are poorly implemented.

### Data Types Support

| Data Type | Data Format |      Supported     |
|:---------:|:-----------:|:------------------:|
| `integer` | `int32`     | :white_check_mark: |
| `integer` | `int64`     | :white_check_mark: |
| `number`  | `float`     | :white_check_mark: |
| `number`  | `double`    |                    |
| `string`  | `byte`      | :white_check_mark: |
| `string`  | `binary`    | :white_check_mark: |
| `boolean` |             | :white_check_mark: |
| `string`  | `date`      | :white_check_mark: |
| `string`  | `date-time` | :white_check_mark: |
| `string`  | `password`  | :white_check_mark: |
| `string`  | `email`     | :white_check_mark: |
| `string`  | `uuid`      | :white_check_mark: |

### Data Options Support

| Data Type   |         Option         |      Supported     |
|:-----------:|:----------------------:|:------------------:|
| `string`    | `minLength`            | :white_check_mark: |
| `string`    | `maxLength`            | :white_check_mark: |
| `string`    | `enum`                 | :white_check_mark: |
| `string`    | `pattern`              |                    |
| `integer`   | `minimum`              | :white_check_mark: |
| `integer`   | `maximum`              | :white_check_mark: |
| `integer`   | `exclusiveMinimum`     | :white_check_mark: |
| `integer`   | `exclusiveMaximum`     | :white_check_mark: |
| `number`    | `minimum`              | :white_check_mark: |
| `number`    | `maximum`              | :white_check_mark: |
| `number`    | `exclusiveMinimum`     | :white_check_mark: |
| `number`    | `exclusiveMaximum`     | :white_check_mark: |
| `array`     | `items`                | :white_check_mark: |
| `array`     | `additionalItems`      |                    |
| `array`     | `minItems`             | :white_check_mark: |
| `array`     | `maxItems`             | :white_check_mark: |
| `array`     | `uniqueItems`          |                    |
| `object`    | `properties`           | :white_check_mark: |
| `object`    | `maxProperties`        |                    |
| `object`    | `minProperties`        |                    |
| `object`    | `patternProperties`    |                    |
| `object`    | `additionalProperties` |                    |
| `object`    | `required`             |                    |
| `*`         | `$ref`                 | :white_check_mark: |
| `*`         | `allOf`                |                    |
| `*`         | `anyOf`                |                    |
| `*`         | `oneOf`                |                    |
| `*`         | `not`                  |                    |

## Known Limitations

Avoid circular refs in your schema. Schema below can cause infinite loop and `Out of Memory` PHP error:
```yml
# ModelA has reference to ModelB while ModelB has reference to ModelA.
# Mock server will produce huge nested JSON example and ended with `Out of Memory` error.
definitions:
  ModelA:
    type: object
    properties:
      model_b:
        $ref: '#/definitions/ModelB'
  ModelB:
    type: array
    items:
      $ref: '#/definitions/ModelA'
```

Don't ref scalar types, because generator will not produce models which mock server can find. So schema below will cause error:
```yml
# generated build contains only `OuterComposite` model class which referenced to not existed `OuterNumber`, `OuterString`, `OuterBoolean` classes
# mock server cannot mock `OuterComposite` model and throws exception
definitions:
  OuterComposite:
    type: object
    properties:
      my_number:
        $ref: '#/definitions/OuterNumber'
      my_string:
        $ref: '#/definitions/OuterString'
      my_boolean:
        $ref: '#/definitions/OuterBoolean'
  OuterNumber:
    type: number
  OuterString:
    type: string
  OuterBoolean:
    type: boolean
```

## Links to mentioned technologies

* [OpenAPI Specification 3.0.3](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md)
* [OpenAPI Generator](https://openapi-generator.tech)
* [Composer](https://getcomposer.org/download/)
 readmeEtag: '"3734033c26f4362ef1101866e41d60139b869ca6"' readmeLastModified: Mon, 21 Mar 2022 19:46:27 GMT repositoryId: 253581764 description: Library that generates fake data from OpenAPI 3.0 Spec created: '2020-04-06T18:24:20Z' updated: '2025-11-09T08:45:28Z' language: PHP archived: false stars: 12 watchers: 1 forks: 0 owner: ybelenko logo: https://avatars.githubusercontent.com/u/5541023?v=4 license: MIT repoEtag: '"82729282904f435741d1f673dcb180c202f881666d35303c2ad17e4ce8a0f34f"' repoLastModified: Sun, 09 Nov 2025 08:45:28 GMT foundInMaster: true id: f47895ca9fc1749e90dd75e64d5ab471 - source: https://openapi.tools/ name: tsoa category: - Server - Data Validators - Parsers language: - Node.js - TypeScript link: https://github.com/lukeautry/tsoa repository: https://github.com/lukeautry/tsoa source_description: >- Creates OpenAPI docs and provides free runtime validation for your Koa, Express, Hapi (and more) services v2: true v3: true repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL3Rzb2EtY29tbXVuaXR5LmdpdGh1Yi5pby9kb2NzLyIgdGFyZ2V0PSJibGFuayI+CiAgICA8aDE+dHNvYTwvaDE+CiAgPC9hPgpQcm9ub3VuY2VkIHNvwrd1aAoKT3BlbkFQSS1jb21wbGlhbnQgUkVTVCBBUElzIHVzaW5nIFR5cGVTY3JpcHQgYW5kIE5vZGUKCiFbYnVpbGQgc3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vbHVrZWF1dHJ5L3Rzb2EvYWN0aW9ucy93b3JrZmxvd3MvcnVuVGVzdHNPblB1c2gueW1sL2JhZGdlLnN2ZykKWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3YvdHNvYS9sYXRlc3QpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS90c29hKQoKPC9kaXY+CgojIyBHb2FsCgotIFR5cGVTY3JpcHQgY29udHJvbGxlcnMgYW5kIG1vZGVscyBhcyB0aGUgc2luZ2xlIHNvdXJjZSBvZiB0cnV0aCBmb3IgeW91ciBBUEkKLSBBIHZhbGlkIE9wZW5BUEkgKGZvcm1lcmx5IFN3YWdnZXIpIHNwZWMgKDIuMCBvciAzLjAgaWYgeW91IGNob29zZSDwn5iNKSBpcyBnZW5lcmF0ZWQgZnJvbSB5b3VyIGNvbnRyb2xsZXJzIGFuZCBtb2RlbHMsIGluY2x1ZGluZzoKICAtIFBhdGhzIChlLmcuIEdFVCAvdXNlcnMpCiAgLSBEZWZpbml0aW9ucyBiYXNlZCBvbiBUeXBlU2NyaXB0IGludGVyZmFjZXMgKG1vZGVscykKICAtIFBhcmFtZXRlcnMvbW9kZWwgcHJvcGVydGllcyBtYXJrZWQgYXMgcmVxdWlyZWQgb3Igb3B0aW9uYWwgYmFzZWQgb24gVHlwZVNjcmlwdCAoZS5nLiBteVByb3BlcnR5Pzogc3RyaW5nIGlzIG9wdGlvbmFsIGluIHRoZSBPcGVuQVBJIHNwZWMpCiAgLSBqc0RvYyBzdXBwb3J0ZWQgZm9yIG9iamVjdCBkZXNjcmlwdGlvbnMgKG1vc3Qgb3RoZXIgbWV0YWRhdGEgY2FuIGJlIGluZmVycmVkIGZyb20gVHlwZVNjcmlwdCB0eXBlcykKLSBSb3V0ZXMgYXJlIGdlbmVyYXRlZCBmb3IgbWlkZGxld2FyZSBvZiBjaG9pY2UKICAtIEV4cHJlc3MsIEhhcGksIGFuZCBLb2EgY3VycmVudGx5IHN1cHBvcnRlZCwgb3RoZXIgbWlkZGxld2FyZSBjYW4gYmUgc3VwcG9ydGVkIHVzaW5nIGEgc2ltcGxlIGhhbmRsZWJhcnMgdGVtcGxhdGUKICAtIFZhbGlkYXRlIHJlcXVlc3QgcGF5bG9hZHMKCiMjIFBoaWxvc29waHkKCi0gUmVseSBvbiBUeXBlU2NyaXB0IHR5cGUgYW5ub3RhdGlvbnMgdG8gZ2VuZXJhdGUgQVBJIG1ldGFkYXRhIGlmIHBvc3NpYmxlCi0gSWYgcmVndWxhciB0eXBlIGFubm90YXRpb25zIGFyZW4ndCBhbiBhcHByb3ByaWF0ZSB3YXkgdG8gZXhwcmVzcyBtZXRhZGF0YSwgdXNlIGRlY29yYXRvcnMKLSBVc2UganNkb2MgZm9yIHB1cmUgdGV4dCBtZXRhZGF0YSAoZS5nLiBlbmRwb2ludCBkZXNjcmlwdGlvbnMpCi0gTWluaW1pemUgYm9pbGVycGxhdGUKLSBNb2RlbHMgYXJlIGJlc3QgcmVwcmVzZW50ZWQgYnkgaW50ZXJmYWNlcyAocHVyZSBkYXRhIHN0cnVjdHVyZXMpLCBidXQgY2FuIGFsc28gYmUgcmVwcmVzZW50ZWQgYnkgY2xhc3NlcwotIFJ1bnRpbWUgdmFsaWRhdGlvbiBvZiB0c29hIHNob3VsZCBiZWhhdmUgYXMgY2xvc2VseSBhcyBwb3NzaWJsZSB0byB0aGUgc3BlY2lmaWNhdGlvbnMgdGhhdCB0aGUgZ2VuZXJhdGVkIE9wZW5BUEkgMi8zIHNjaGVtYSBkZXNjcmliZXMuIEFueSBkaWZmZXJlbmNlcyBpbiB2YWxpZGF0aW9uIGxvZ2ljIGFyZSBjbGFyaWZpZWQgYnkgbG9nZ2luZyB3YXJuaW5ncyBkdXJpbmcgdGhlIGdlbmVyYXRpb24gb2YgdGhlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAoT0FTKSBhbmQvb3IgdGhlIHJvdXRlcy4KICAtIFBsZWFzZSBub3RlIHRoYXQgYnkgZW5hYmxpbmcgT3BlbkFQSSAzIHlvdSBtaW5pbWl6ZSB0aGUgY2hhbmNlcyBvZiBkaXZlcmdlbnQgdmFsaWRhdGlvbiBsb2dpYyBzaW5jZSBPcGVuQVBJIDMgaGFzIGEgbW9yZSBleHByZXNzaXZlIHNjaGVtYSBzeW50YXguCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCi0gW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vdHNvYS1jb21tdW5pdHkuZ2l0aHViLmlvL2RvY3MvKQotIFtBUEkgUmVmZXJlbmNlXShodHRwczovL3Rzb2EtY29tbXVuaXR5LmdpdGh1Yi5pby9yZWZlcmVuY2UvKQotIFtHZXR0aW5nIHN0YXJ0ZWQgZ3VpZGVdKGh0dHBzOi8vdHNvYS1jb21tdW5pdHkuZ2l0aHViLmlvL2RvY3MvZ2V0dGluZy1zdGFydGVkKQoKIyMgRXhhbXBsZXMKCkNoZWNrIG91dCB0aGUgW2d1aWRlc10oaHR0cHM6Ly90c29hLWNvbW11bml0eS5naXRodWIuaW8vZG9jcy9nZXR0aW5nLXN0YXJ0ZWQpCgpTZWUgZXhhbXBsZSBjb250cm9sbGVycyBpbiBbdGhlIHRlc3RzXSh0ZXN0cy9maXh0dXJlcy9jb250cm9sbGVycykKClNlZSBleGFtcGxlIG1vZGVscyBpbiBbdGhlIHRlc3RzXSh0ZXN0cy9maXh0dXJlcy90ZXN0TW9kZWwudHMpCgojIyBIZWxwIHdhbnRlZAoKIyMjIENvbnRyaWJ1dGluZyBjb2RlCgpUbyBjb250cmlidXRlICh2aWEgYSBQUiksIHBsZWFzZSBmaXJzdCBzZWUgdGhlIFtDb250cmlidXRpbmcgR3VpZGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9sdWtlYXV0cnkvdHNvYS90cmVlL21hc3Rlci9kb2NzL0NPTlRSSUJVVElORy5tZCkKCiMjIyBCZWNvbWluZyBhIG1haW50YWluZXIKCnRzb2Egd2FudHMgYWRkaXRpb25hbCBtYWludGFpbmVycyEgVGhlIGxpYnJhcnkgaGFzIGluY3JlYXNlZCBpbiBwb3B1bGFyaXR5IGFuZCBoYXMgcXVpdGUgYSBsb3Qgb2YgcHVsbCByZXF1ZXN0cyBhbmQgaXNzdWVzLiBbUGxlYXNlIHBvc3QgaW4gdGhpcyBpc3N1ZV0oaHR0cHM6Ly9naXRodWIuY29tL2x1a2VhdXRyeS90c29hL2lzc3Vlcy8yMzYpIGlmIHlvdSdyZSB3aWxsaW5nIHRvIHRha2Ugb24gdGhlIHJvbGUgb2YgYSBtYWludGFpbmVyLgo= readmeEtag: '"81295b201e9ff99d873dc633f42632deb1156566"' readmeLastModified: Fri, 12 Jul 2024 09:50:15 GMT repositoryId: 61366800 description: Build OpenAPI-compliant REST APIs using TypeScript and Node created: '2016-06-17T10:42:50Z' updated: '2026-02-01T10:44:57Z' language: TypeScript archived: false stars: 3930 watchers: 26 forks: 530 owner: lukeautry logo: https://avatars.githubusercontent.com/u/8770560?v=4 license: MIT repoEtag: '"9f33e1233173d4760923a88eca26a8cbe94dd31a0734474cd985bd1a7f398f51"' repoLastModified: Sun, 01 Feb 2026 10:44:57 GMT foundInMaster: true id: 2529db76d3d9f042b0c624f65b13fc41 - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/hey-api/openapi-ts v3: true id: 2f6d5164db429943c445c5647745a9be repositoryMetadata: base64Readme: >- <div align="center">
  <img alt="Hey API presents openapi-ts" height="214" src="https://heyapi.dev/assets/.gen/openapi-ts-hero-640w.png" width="438">
  <h1><b>OpenAPI TypeScript</b></h1>
  <p><em>“OpenAPI codegen that just works.”</em><br/><sub>— Guillermo Rauch, CEO of Vercel</sub></p>
</div>

<p align="center">
  <a href="https://www.devtrends.dev/trends?c=v1.kZIBIg"><img src="https://api.devtrends.dev/badge/npm/%40hey-api%2Fopenapi-ts?period=month&style=flat&view=value" alt="DevTrends badge for @hey-api/openapi-ts" /></a>
  <a href="https://www.devtrends.dev/trends?c=v1.kZIBIg&v=change"><img src="https://api.devtrends.dev/badge/npm/%40hey-api%2Fopenapi-ts?period=year&style=flat&view=change" alt="DevTrends badge for @hey-api/openapi-ts" /></a>
  <a href="https://github.com/hey-api/openapi-ts/actions?query=branch%3Amain"><img src="https://github.com/hey-api/openapi-ts/actions/workflows/ci.yml/badge.svg?event=push&branch=main" alt="CI status" /></a>
  <a href="https://github.com/hey-api/openapi-ts"><img src="https://img.shields.io/github/stars/hey-api/openapi-ts?style=flat&logo=github&label=GitHub&color=54C82D" alt="GitHub stars" /></a>
  <a href="https://github.com/hey-api/openapi-ts/blob/main/LICENSE.md"><img src="https://img.shields.io/github/license/hey-api/openapi-ts" alt="MIT License"></a>
</p>

<p align="center">
  <a href="https://stackblitz.com/edit/hey-api-example?file=openapi-ts.config.ts,src%2Fclient%2Fschemas.gen.ts,src%2Fclient%2Fsdk.gen.ts,src%2Fclient%2Ftypes.gen.ts">Demo</a>
  <span>&nbsp;•&nbsp;</span>
  <a href="https://heyapi.dev">Manual</a>
  <span>&nbsp;•&nbsp;</span>
  <a href="https://github.com/hey-api/openapi-ts/issues">Issues</a>
  <span>&nbsp;•&nbsp;</span>
  <a href="https://heyapi.dev/openapi-ts/community/contributing">Contribute</a>
</p>

## About

The OpenAPI to TypeScript code generator used by Vercel, OpenCode, and PayPal.

Generate production-ready SDKs, Zod schemas, TanStack Query hooks, or choose from 20+ other plugins.

Part of the Hey API ecosystem.

## Features

- production-ready code that compiles
- runs in any Node.js 20+ environment
- accepts any OpenAPI specification
- core plugins for SDKs, types, and schemas
- HTTP clients for Fetch API, Angular, Axios, Next.js, Nuxt, and more
- 20+ plugins to reduce third-party boilerplate
- highly customizable via plugins
- [sync with Hey API Registry](https://heyapi.dev/openapi-ts/integrations) for spec management

## Contributing

Want to see your code in products used by millions?

Start with our [Contributing](https://heyapi.dev/openapi-ts/community/contributing) guide and release your first feature.

## Sponsors

Hey API is sponsor-funded. If you rely on Hey API in production, consider becoming a [sponsor](https://github.com/sponsors/hey-api) to accelerate the roadmap.

<h3 align="center">Gold</h3>

<table align="center" style="justify-content: center;align-items: center;display: flex;">
  <tbody>
    <tr>
      <td align="center" width="336px">
        <p></p>
        <p>
          <a href="https://kutt.it/pkEZyc" target="_blank">
            <picture height="50px">
              <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/.gen/stainless-logo-wordmark-480w.jpeg">
              <img alt="Stainless logo" height="50px" src="https://heyapi.dev/assets/.gen/stainless-logo-wordmark-480w.jpeg">
            </picture>
          </a>
          <br/>
          Best-in-class developer interfaces for your API.
          <br/>
          <a href="https://kutt.it/pkEZyc" style="text-decoration:none;" target="_blank">
            stainless.com
          </a>
        </p>
        <p></p>
      </td>
      <td align="center" width="336px">
        <p></p>
        <p>
          <a href="https://kutt.it/QM9Q2N" target="_blank">
            <picture height="50px">
              <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/opencode/logo-light.svg">
              <img alt="Opencode logo" height="50px" src="https://heyapi.dev/assets/opencode/logo-dark.svg">
            </picture>
          </a>
          <br/>
          The open source coding agent.
          <br/>
          <a href="https://kutt.it/QM9Q2N" style="text-decoration:none;" target="_blank">
            opencode.ai
          </a>
        </p>
        <p></p>
      </td>
    </tr>
    <tr>
      <td align="center" width="336px">
        <p></p>
        <p>
          <a href="https://kutt.it/6vrYy9" target="_blank">
            <picture height="50px">
              <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/mintlify/logo-light.svg">
              <img alt="Mintlify logo" height="50px" src="https://heyapi.dev/assets/mintlify/logo-dark.svg">
            </picture>
          </a>
          <br/>
          The intelligent knowledge platform.
          <br/>
          <a href="https://kutt.it/6vrYy9" style="text-decoration:none;" target="_blank">
            mintlify.com
          </a>
        </p>
        <p></p>
      </td>
    </tr>
  </tbody>
</table>

<h3 align="center">Silver</h3>

<table align="center" style="justify-content: center;align-items: center;display: flex;">
  <tbody>
    <tr>
      <td align="center" width="172px">
        <a href="https://kutt.it/skQUVd" target="_blank">
          <picture height="40px">
            <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/scalar/logo-light.svg">
            <img alt="Scalar logo" height="40px" src="https://heyapi.dev/assets/scalar/logo-dark.svg">
          </picture>
        </a>
        <br/>
        <a href="https://kutt.it/skQUVd" style="text-decoration:none;" target="_blank">
          scalar.com
        </a>
      </td>
      <td align="center" width="172px">
        <a href="https://kutt.it/Dr9GuW" target="_blank">
          <picture height="40px">
            <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/fastapi/logo-light.svg">
            <img alt="FastAPI logo" height="40px" src="https://heyapi.dev/assets/fastapi/logo-dark.svg">
          </picture>
        </a>
        <br/>
        <a href="https://kutt.it/Dr9GuW" style="text-decoration:none;" target="_blank">
          fastapi.tiangolo.com
        </a>
      </td>
    </tr>
  </tbody>
</table>

<h3 align="center">Bronze</h3>

<table align="center" style="justify-content: center;align-items: center;display: flex;">
  <tbody>
    <tr>
      <td align="center" width="136px">
        <a href="https://kutt.it/YpaKsX" target="_blank">
          <picture height="34px">
            <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/.gen/kinde-logo-wordmark-dark-480w.webp">
            <img alt="Kinde logo" height="34px" src="https://heyapi.dev/assets/.gen/kinde-logo-wordmark-480w.jpeg">
          </picture>
        </a>
      </td>
      <td align="center" width="136px">
        <a href="https://kutt.it/KkqSaw" target="_blank">
          <picture height="34px">
            <source media="(prefers-color-scheme: dark)" srcset="https://heyapi.dev/assets/cella/logo-light.svg">
            <img alt="Cella logo" height="34px" src="https://heyapi.dev/assets/cella/logo-dark.svg">
          </picture>
        </a>
      </td>
    </tr>
  </tbody>
</table>

## Quick Start

The fastest way to use `@hey-api/openapi-ts` is via npx

```sh
npx @hey-api/openapi-ts -i hey-api/backend -o src/client
```

Congratulations on creating your first client! 🎉 You can learn more about the generated files on the [Output](https://heyapi.dev/openapi-ts/output) page.

## Installation

You can download `@hey-api/openapi-ts` from npm using your favorite package manager.

#### npm

```sh
npm install @hey-api/openapi-ts -D -E
```

#### pnpm

```sh
pnpm add @hey-api/openapi-ts -D -E
```

#### yarn

```sh
yarn add @hey-api/openapi-ts -D -E
```

#### bun

```sh
bun add @hey-api/openapi-ts -D -E
```

### Versioning

This package is in [initial development](https://semver.org/#spec-item-4). Please pin an exact version so you can safely upgrade when you're ready.

We publish [migration notes](https://heyapi.dev/openapi-ts/migrating) for every breaking release. You might not be impacted by a breaking change if you don't use the affected features.

## Usage

### CLI

Most people run `@hey-api/openapi-ts` via CLI. To do that, add a script to your `package.json` file which will make `openapi-ts` executable through script.

```json
"scripts": {
  "openapi-ts": "openapi-ts"
}
```

The above script can be executed by running `npm run openapi-ts` or equivalent command in other package managers. Next, we will create a [configuration](https://heyapi.dev/openapi-ts/configuration) file and move our options from Quick Start to it.

### Node.js

You can also generate output programmatically by calling `createClient()` in a JavaScript/TypeScript file.

#### `script.ts`

```ts
import { createClient } from '@hey-api/openapi-ts';

createClient({
  input: 'hey-api/backend', // sign up at app.heyapi.dev
  output: 'src/client',
});
```

## Configuration

`@hey-api/openapi-ts` supports loading configuration from any file inside your project root folder supported by [jiti loader](https://github.com/unjs/c12?tab=readme-ov-file#-features). Below are the most common file formats.

#### `openapi-ts.config.ts`

```js
import { defineConfig } from '@hey-api/openapi-ts';

export default defineConfig({
  input: 'hey-api/backend', // sign up at app.heyapi.dev
  output: 'src/client',
});
```

#### `openapi-ts.config.cjs`

```js
/** @type {import('@hey-api/openapi-ts').UserConfig} */
module.exports = {
  input: 'hey-api/backend', // sign up at app.heyapi.dev
  output: 'src/client',
};
```

#### `openapi-ts.config.mjs`

```js
/** @type {import('@hey-api/openapi-ts').UserConfig} */
export default {
  input: 'hey-api/backend', // sign up at app.heyapi.dev
  output: 'src/client',
};
```

Alternatively, you can use `openapi-ts.config.js` and configure the export statement depending on your project setup.

### Input

You must set the input so we can load your OpenAPI specification. It can be a path or URL, object containing a path or URL, or an object representing an OpenAPI specification. Hey API supports all valid OpenAPI versions and file formats.

> If you use an HTTPS URL with a self-signed certificate in development, you will need to set [`NODE_TLS_REJECT_UNAUTHORIZED=0`](https://github.com/hey-api/openapi-ts/issues/276#issuecomment-2043143501) in your environment.

### Output

You must set the output so we know where to generate your files. It can be a path to the destination folder or an object containing the destination folder path and optional settings.

> You should treat the output folder as a dependency. Do not directly modify its contents as your changes might be erased when you run `@hey-api/openapi-ts` again.

## Parser

We parse your input before making it available to plugins. While configuring the parser is optional, it's the perfect place to modify or validate your input if needed.

## Plugins

Plugins are responsible for generating artifacts from your input. By default, Hey API will generate TypeScript interfaces and SDK from your OpenAPI specification. You can add, remove, or customize any of the plugins. In fact, we highly encourage you to do so!

### Client

Clients are responsible for sending the actual HTTP requests. Using clients is not required, but you must add a client to `plugins` if you're generating SDKs (we default to Fetch).

### Native Clients

- [`@hey-api/client-fetch`](https://heyapi.dev/openapi-ts/clients/fetch)
- [`@hey-api/client-angular`](https://heyapi.dev/openapi-ts/clients/angular)
- [`@hey-api/client-axios`](https://heyapi.dev/openapi-ts/clients/axios)
- [`@hey-api/client-ky`](https://heyapi.dev/openapi-ts/clients/ky)
- [`@hey-api/client-next`](https://heyapi.dev/openapi-ts/clients/next-js)
- [`@hey-api/client-nuxt`](https://heyapi.dev/openapi-ts/clients/nuxt)
- [`@hey-api/client-ofetch`](https://heyapi.dev/openapi-ts/clients/ofetch)

### Planned Clients

The following clients are planned but not in development yet. You can help us prioritize them by voting on [GitHub](https://github.com/hey-api/openapi-ts/labels/RSVP%20%F0%9F%91%8D%F0%9F%91%8E).

- [`@hey-api/client-effect`](https://heyapi.dev/openapi-ts/clients/effect)

Don't see your client? [Build your own](https://heyapi.dev/openapi-ts/clients/custom) or let us know your interest by [opening an issue](https://github.com/hey-api/openapi-ts/issues).

### Native Plugins

These plugins help reduce boilerplate associated with third-party dependencies. Hey API natively supports the most popular packages. Please open an issue on [GitHub](https://github.com/hey-api/openapi-ts/issues) if you'd like us to support your favorite package.

- [`@angular/common`](https://heyapi.dev/openapi-ts/plugins/angular)
- [`@hey-api/schemas`](https://heyapi.dev/openapi-ts/output/json-schema)
- [`@hey-api/sdk`](https://heyapi.dev/openapi-ts/output/sdk)
- [`@hey-api/transformers`](https://heyapi.dev/openapi-ts/transformers)
- [`@hey-api/typescript`](https://heyapi.dev/openapi-ts/output/typescript)
- [`@pinia/colada`](https://heyapi.dev/openapi-ts/plugins/pinia-colada)
- [`@tanstack/angular-query-experimental`](https://heyapi.dev/openapi-ts/plugins/tanstack-query)
- [`@tanstack/react-query`](https://heyapi.dev/openapi-ts/plugins/tanstack-query)
- [`@tanstack/solid-query`](https://heyapi.dev/openapi-ts/plugins/tanstack-query)
- [`@tanstack/svelte-query`](https://heyapi.dev/openapi-ts/plugins/tanstack-query)
- [`@tanstack/vue-query`](https://heyapi.dev/openapi-ts/plugins/tanstack-query)
- [`fastify`](https://heyapi.dev/openapi-ts/plugins/fastify)
- [`valibot`](https://heyapi.dev/openapi-ts/plugins/valibot)
- [`zod`](https://heyapi.dev/openapi-ts/plugins/zod)

### Planned Plugins

The following plugins are planned but not in development yet. You can help us prioritize them by voting on [GitHub](https://github.com/hey-api/openapi-ts/labels/RSVP%20%F0%9F%91%8D%F0%9F%91%8E).

- [Adonis](https://heyapi.dev/openapi-ts/plugins/adonis)
- [Ajv](https://heyapi.dev/openapi-ts/plugins/ajv)
- [Arktype](https://heyapi.dev/openapi-ts/plugins/arktype)
- [Chance](https://heyapi.dev/openapi-ts/plugins/chance)
- [Elysia](https://heyapi.dev/openapi-ts/plugins/elysia)
- [Express](https://heyapi.dev/openapi-ts/plugins/express)
- [Faker](https://heyapi.dev/openapi-ts/plugins/faker)
- [Falso](https://heyapi.dev/openapi-ts/plugins/falso)
- [Hono](https://heyapi.dev/openapi-ts/plugins/hono)
- [Joi](https://heyapi.dev/openapi-ts/plugins/joi)
- [Koa](https://heyapi.dev/openapi-ts/plugins/koa)
- [MSW](https://heyapi.dev/openapi-ts/plugins/msw)
- [Nest](https://heyapi.dev/openapi-ts/plugins/nest)
- [Nock](https://heyapi.dev/openapi-ts/plugins/nock)
- [Superstruct](https://heyapi.dev/openapi-ts/plugins/superstruct)
- [Supertest](https://heyapi.dev/openapi-ts/plugins/supertest)
- [SWR](https://heyapi.dev/openapi-ts/plugins/swr)
- [TypeBox](https://heyapi.dev/openapi-ts/plugins/typebox)
- [Yup](https://heyapi.dev/openapi-ts/plugins/yup)
- [Zustand](https://heyapi.dev/openapi-ts/plugins/zustand)

Don't see your plugin? [Build your own](https://heyapi.dev/openapi-ts/plugins/custom) or let us know your interest by [opening an issue](https://github.com/hey-api/openapi-ts/issues).

## Migrating

You can learn more on the [Migrating](https://heyapi.dev/openapi-ts/migrating) page.

## License

Released under the [MIT License](https://github.com/hey-api/openapi-ts/blob/main/LICENSE.md).
 readmeEtag: '"f308ed5957d2ae6b31435e1335720c59e70ce4ca"' readmeLastModified: Thu, 05 Feb 2026 14:46:51 GMT repositoryId: 741524414 description: >- 🌀 OpenAPI to TypeScript codegen. Production-ready SDKs, Zod schemas, TanStack Query hooks, and 20+ plugins. Used by Vercel, OpenCode, and PayPal. created: '2024-01-10T15:19:25Z' updated: '2026-02-06T03:39:13Z' language: TypeScript archived: false stars: 4033 watchers: 10 forks: 305 owner: hey-api logo: https://avatars.githubusercontent.com/u/164436240?v=4 license: MIT repoEtag: '"028356e8dac5a05dd362013138fe39a5a3bc3741fd65c347cac0b82a070ebb74"' repoLastModified: Fri, 06 Feb 2026 03:39:13 GMT category: - Converters - SDK foundInMaster: true name: '@hey-api/openapi-ts' source_description: Turn your OpenAPI description into a beautiful TypeScript client. language: TypeScript link: https://heyapi.vercel.app/ v2: true v3_1: true - source: https://openapi.tools/ name: Vert.x Web Api Contract category: Server language: - Java - Kotlin - JavaScript - Groovy - Ruby - Ceylon or Scala link: https://vertx.io/docs/#web repository: https://github.com/vert-x3/vertx-web/tree/master/vertx-web-api-contract source_description: >- Create API endpoints with Vert.x 3 and OpenAPI 3 with automatic requests validation v2: false v3: true repositoryMetadata: base64Readme: >- PSBWZXJ0LngtV2ViCgppbWFnZTpodHRwczovL2dpdGh1Yi5jb20vdmVydC14My92ZXJ0eC13ZWIvYWN0aW9ucy93b3JrZmxvd3MvY2ktNS54LnltbC9iYWRnZS5zdmdbIkJ1aWxkIFN0YXR1cyAoNS54KSIsbGluaz0iaHR0cHM6Ly9naXRodWIuY29tL3ZlcnQteDMvdmVydHgtd2ViL2FjdGlvbnMvd29ya2Zsb3dzL2NpLTUueC55bWwiXQppbWFnZTpodHRwczovL2dpdGh1Yi5jb20vdmVydC14My92ZXJ0eC13ZWIvYWN0aW9ucy93b3JrZmxvd3MvY2ktNC54LnltbC9iYWRnZS5zdmdbIkJ1aWxkIFN0YXR1cyAoNC54KSIsbGluaz0iaHR0cHM6Ly9naXRodWIuY29tL3ZlcnQteDMvdmVydHgtd2ViL2FjdGlvbnMvd29ya2Zsb3dzL2NpLTQueC55bWwiXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9pby52ZXJ0eC92ZXJ0eC13ZWIuc3ZnWyJNYXZlbiBDZW50cmFsIl0KaW1hZ2U6aHR0cHM6Ly9iZXN0cHJhY3RpY2VzLmNvcmVpbmZyYXN0cnVjdHVyZS5vcmcvcHJvamVjdHMvNTQwL2JhZGdlWyJDSUkgQmVzdCBQcmFjdGljZXMiLGxpbms9Imh0dHBzOi8vYmVzdHByYWN0aWNlcy5jb3JlaW5mcmFzdHJ1Y3R1cmUub3JnL3Byb2plY3RzLzU0MCJdCgpWZXJ0LngtV2ViIGlzIGEgc2V0IG9mIGJ1aWxkaW5nIGJsb2NrcyBmb3IgYnVpbGRpbmcgd2ViIGFwcGxpY2F0aW9ucyB3aXRoIFZlcnQueC4gVGhpbmsgb2YgaXQgYXMgYSBTd2lzcyBBcm15IEtuaWZlIGZvciBidWlsZGluZwptb2Rlcm4sIHNjYWxhYmxlLCB3ZWIgYXBwcy4KClBsZWFzZSBzZWUgdGhlIG1haW4gZG9jdW1lbnRhdGlvbiBvbiB0aGUgd2ViLXNpdGUgZm9yIGEgZnVsbCBkZXNjcmlwdGlvbjoKCiogaHR0cHM6Ly92ZXJ0eC5pby9kb2NzLyN3ZWJbV2ViLXNpdGUgZG9jdW1lbnRhdGlvbl0KCj09IFRlbXBsYXRlIGVuZ2luZXMKClRlbXBsYXRlIGVuZ2luZSBpbXBsZW1lbnRhdGlvbnMgYXJlIGluIHRoZSB0ZW1wbGF0ZSBlbmdpbmUgc3ViLXByb2plY3QuCg== readmeEtag: '"269b52e98cea5813787f90c67bef329932b70760"' readmeLastModified: Fri, 16 Aug 2024 21:55:56 GMT repositoryId: 26628954 description: HTTP web applications for Vert.x created: '2014-11-14T08:15:20Z' updated: '2026-02-05T20:01:22Z' language: Java archived: false stars: 1144 watchers: 71 forks: 556 owner: vert-x3 logo: https://avatars.githubusercontent.com/u/8124623?v=4 license: Apache-2.0 repoEtag: '"27b312e88d5724b891cdb6068adcfe907cdb8f8c8a312ae3851967e80ed62edc"' repoLastModified: Thu, 05 Feb 2026 20:01:22 GMT foundInMaster: true v3_link: https://github.com/vert-x3/vertx-web/issues/1872 id: 2657bfe401922934e35bc38fbbef6b6d - source: https://openapi.tools/ name: express-openapi category: Server language: - Node.js - Typescript link: https://www.npmjs.com/package/express-openapi repository: >- https://github.com/kogosoftwarellc/open-api/tree/master/packages/express-openapi source_description: >- An unopinionated OpenAPI framework for Express, which supports Promise based middleware, response handlers and Security Filtering. v2: true v3: true v3_1_link: https://github.com/vert-x3/vertx-web/issues/1872 id: d3549059c844ac132a254310d4d7ac01 repositoryMetadata: base64Readme: >- IyBAb3Blbi1hcGkgWyFbQ292ZXJhbGxzIFN0YXR1c11bY292ZXJhbGxzLWltYWdlXV1bY292ZXJhbGxzLXVybF0KPiBBIE1vbm9yZXBvIG9mIHZhcmlvdXMgcGFja2FnZXMgdG8gcG93ZXIgT3BlbkFQSSBpbiBub2RlLgoKIyMgUXVpY2sgU3RhcnQgRXhwcmVzcwoKKiBTZWUgW2V4cHJlc3Mtb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9leHByZXNzLW9wZW5hcGkpJm5ic3A7Jm5ic3A7Jm5ic3A7WyFbZXhwcmVzcy1vcGVuYXBpIERvd25sb2Fkc11bZXhwcmVzcy1vcGVuYXBpLWRvd25sb2Fkcy1pbWFnZV1dW2V4cHJlc3Mtb3BlbmFwaS1ucG0tdXJsXQoKIyMgUXVpY2sgU3RhcnQgS29hCiogU2VlIFtrb2Etb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9rb2Etb3BlbmFwaSkmbmJzcDsmbmJzcDsmbmJzcDtbIVtrb2Etb3BlbmFwaSBEb3dubG9hZHNdW2tvYS1vcGVuYXBpLWRvd25sb2Fkcy1pbWFnZV1dW2tvYS1vcGVuYXBpLW5wbS11cmxdCgojIyBQYWNrYWdlcwoqIFtleHByZXNzLW9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9rb2dvc29mdHdhcmVsbGMvb3Blbi1hcGkvdHJlZS9tYXN0ZXIvcGFja2FnZXMvZXhwcmVzcy1vcGVuYXBpKQoqIFtmZXRjaC1vcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20va29nb3NvZnR3YXJlbGxjL29wZW4tYXBpL3RyZWUvbWFzdGVyL3BhY2thZ2VzL2ZldGNoLW9wZW5hcGkpCiogW2ZzLXJvdXRlc10oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9mcy1yb3V0ZXMpCiogW29wZW5hcGktZGVmYXVsdC1zZXR0ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9rb2dvc29mdHdhcmVsbGMvb3Blbi1hcGkvdHJlZS9tYXN0ZXIvcGFja2FnZXMvb3BlbmFwaS1kZWZhdWx0LXNldHRlcikKKiBbb3BlbmFwaS1mcmFtZXdvcmtdKGh0dHBzOi8vZ2l0aHViLmNvbS9rb2dvc29mdHdhcmVsbGMvb3Blbi1hcGkvdHJlZS9tYXN0ZXIvcGFja2FnZXMvb3BlbmFwaS1mcmFtZXdvcmspCiogW29wZW5hcGktanNvbnNjaGVtYS1wYXJhbWV0ZXJzXShodHRwczovL2dpdGh1Yi5jb20va29nb3NvZnR3YXJlbGxjL29wZW4tYXBpL3RyZWUvbWFzdGVyL3BhY2thZ2VzL29wZW5hcGktanNvbnNjaGVtYS1wYXJhbWV0ZXJzKQoqIFtvcGVuYXBpLXJlcXVlc3QtY29lcmNlcl0oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9vcGVuYXBpLXJlcXVlc3QtY29lcmNlcikKKiBbb3BlbmFwaS1yZXF1ZXN0LXZhbGlkYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9vcGVuYXBpLXJlcXVlc3QtdmFsaWRhdG9yKQoqIFtvcGVuYXBpLXJlc3BvbnNlLXZhbGlkYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9vcGVuYXBpLXJlc3BvbnNlLXZhbGlkYXRvcikKKiBbb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20va29nb3NvZnR3YXJlbGxjL29wZW4tYXBpL3RyZWUvbWFzdGVyL3BhY2thZ2VzL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvcikKKiBbb3BlbmFwaS10eXBlc10oaHR0cHM6Ly9naXRodWIuY29tL2tvZ29zb2Z0d2FyZWxsYy9vcGVuLWFwaS90cmVlL21hc3Rlci9wYWNrYWdlcy9vcGVuYXBpLXR5cGVzKQoKIyMgRGV2ZWxvcG1lbnQKClRoaXMgbW9ub3JlcG8gdXNlcyBsZXJuYSBmb3IgZGV2ZWxvcG1lbnQuICBTZWUgdGhlIHJvb3QgcGFja2FnZS5qc29uIGZvciBoZWxwZnVsIHNjcmlwdHMuCgojIyMgVHlwaWNhbCBXb3JrZmxvdyBmb3IgQ29udHJpYnV0b3JzCgpMZXQncyBzYXkgeW91J3JlIHdvcmtpbmcgb24gYSBwYWNrYWdlIHVuZGVyIFsuL3BhY2thZ2VzXShodHRwczovL2dpdGh1Yi5jb20va29nb3NvZnR3YXJlbGxjL29wZW4tYXBpL3RyZWUvbWFzdGVyL3BhY2thZ2VzKS4gIEhlcmUncyB3aGF0IHlvdSBkbzoKCjEuIGBjZCBvcGVuLWFwaWAKMS4gYG5wbSBydW4gYm9vdHN0cmFwYAoxLiBgbnBtIHRgCjEuIE1ha2UgeW91ciBjaGFuZ2VzLgogIDEuIF9EbyBub3QgYnVtcCB0aGUgdmVyc2lvbiBpbiBwYWNrYWdlLmpzb24uXyAgQSBtYWludGFpbmVyIHdpbGwgaGFuZGxlIHRoYXQgb25jZSB5b3VyIFBSIGlzIG1lcmdlZC4KMS4gT25jZSB5b3UncmUgc2F0aXNmaWVkIHdpdGggeW91ciBjaGFuZ2VzOgogIDEuIENyZWF0ZSBhIG5ldyBicmFuY2ggYGdpdCBjaGVja291dCAtYiBteS1icmFuY2hgIChpbiBjYXNlIHlvdSBoYXZlbid0IGRvbmUgc28gYWxyZWFkeSkuCiAgMS4gYC4vYmluL2NvbW1pdCBwYWNrYWdlcy88cGFja2FnZV95b3UncmVfd29ya2luZ19vbj4gJ2NvbW1pdCBtZXNzYWdlIGRlc2NyaWJpbmcgeW91ciBjaGFuZ2UuICBjYW4gYmUgbXVsdGkgbGluZSBoZXJlLiAganVzdCBjbG9zZSB3aXRoIGEgc2luZ2xlIHF1b3RlIGxpa2Ugc286J2AKICAxLiBQdXNoIHlvdXIgY2hhbmdlIHRvIHlvdXIgZm9yawogIDEuIE9wZW4gYSBQUi4KCiMjIyBiaW4KClNldmVyYWwgc2NyaXB0cyBoYXZlIGJlZW4gY3JlYXRlZCB0byBhaWQgaW4gdGhlIGRldmVsb3BtZW50IG9mIHRoaXMgbW9ub3JlcG8gKHNlZSBbLi9iaW5dKC4vYmluKSkuICBUaGV5IGFzc3VtZSB0aGF0IHlvdXIgYCRQV0RgIGlzIHRoZSByb290IG9mIHRoZSByZXBvc2l0b3J5LiAgSGVyZSBpcyBhIGJyaWVmIHN1bW1hcnkgb2YgY29tbW9uIGFjdGlvbnM6CgoqIENvbW1pdCBjaGFuZ2VzIHRvIGEgcGFja2FnZSAtIGAuL2Jpbi9jb21taXQgcGFja2FnZXMvPHBhY2thZ2VfdG9fY29tbWl0PiAnQ29tbWl0IG1lc3NhZ2UnYCAodGhlIGNvbW1pdCBtZXNzYWdlIHdpbGwgYmUgcHJlcGVuZGVkIHdpdGggdGhlIHBhY2thZ2UgbmFtZSBlLmcuIGA8cGFja2FnZV90b19jb21taXQ+OiBDb21taXQgbWVzc2FnZWAKKiBUaGVzZSByZWR1Y2UgYm9pbGVycGxhdGUgYW5kIGFyZSBjYWxsZWQgZnJvbSBucG0gc2NyaXB0cyBpbiBsZWFmIHJlcG9zLgogICogbnljCiAgKiB0c2MKICAqIG1vY2hhCgojIyBMSUNFTlNFCgpgYGAKVGhlIE1JVCBMaWNlbnNlIChNSVQpCgpDb3B5cmlnaHQgKGMpIDIwMTggS29nbyBTb2Z0d2FyZSBMTEMKClBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhIGNvcHkKb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzICh0aGUgIlNvZnR3YXJlIiksIHRvIGRlYWwKaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0cwp0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsCmNvcGllcyBvZiB0aGUgU29mdHdhcmUsIGFuZCB0byBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBTb2Z0d2FyZSBpcwpmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgoKVGhlIGFib3ZlIGNvcHlyaWdodCBub3RpY2UgYW5kIHRoaXMgcGVybWlzc2lvbiBub3RpY2Ugc2hhbGwgYmUgaW5jbHVkZWQgaW4KYWxsIGNvcGllcyBvciBzdWJzdGFudGlhbCBwb3J0aW9ucyBvZiB0aGUgU29mdHdhcmUuCgpUSEUgU09GVFdBUkUgSVMgUFJPVklERUQgIkFTIElTIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwgRVhQUkVTUyBPUgpJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GIE1FUkNIQU5UQUJJTElUWSwKRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UgQU5EIE5PTklORlJJTkdFTUVOVC4gSU4gTk8gRVZFTlQgU0hBTEwgVEhFCkFVVEhPUlMgT1IgQ09QWVJJR0hUIEhPTERFUlMgQkUgTElBQkxFIEZPUiBBTlkgQ0xBSU0sIERBTUFHRVMgT1IgT1RIRVIKTElBQklMSVRZLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwgVE9SVCBPUiBPVEhFUldJU0UsIEFSSVNJTkcgRlJPTSwKT1VUIE9GIE9SIElOIENPTk5FQ1RJT04gV0lUSCBUSEUgU09GVFdBUkUgT1IgVEhFIFVTRSBPUiBPVEhFUiBERUFMSU5HUyBJTgpUSEUgU09GVFdBUkUuCmBgYAoKW2V4cHJlc3Mtb3BlbmFwaS1kb3dubG9hZHMtaW1hZ2VdOiBodHRwOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2RtL2V4cHJlc3Mtb3BlbmFwaS5zdmcKW2V4cHJlc3Mtb3BlbmFwaS1ucG0tdXJsXTogaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZS9leHByZXNzLW9wZW5hcGkKW2tvYS1vcGVuYXBpLWRvd25sb2Fkcy1pbWFnZV06IGh0dHA6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0va29hLW9wZW5hcGkuc3ZnCltrb2Etb3BlbmFwaS1ucG0tdXJsXTogaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZS9rb2Etb3BlbmFwaQoKW2NvdmVyYWxscy11cmxdOiBodHRwczovL2NvdmVyYWxscy5pby9naXRodWIva29nb3NvZnR3YXJlbGxjL29wZW4tYXBpP2JyYW5jaD1tYWluCltjb3ZlcmFsbHMtaW1hZ2VdOiBodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIva29nb3NvZnR3YXJlbGxjL29wZW4tYXBpL2JhZGdlLnN2Zz9icmFuY2g9bWFpbgo= readmeEtag: '"8f1d406d4ad10b939db247a64e8ebfb658ba4536"' readmeLastModified: Wed, 08 May 2024 21:06:34 GMT repositoryId: 48869446 description: A Monorepo of various packages to power OpenAPI in node created: '2016-01-01T04:28:48Z' updated: '2026-02-04T13:54:17Z' language: JavaScript archived: false stars: 920 watchers: 12 forks: 239 owner: kogosoftwarellc logo: https://avatars.githubusercontent.com/u/15419153?v=4 license: MIT repoEtag: '"d668eb1865c8e9f2dcf8baa1a72c66b992a9e4066dfbaa31151095f4966d36cb"' repoLastModified: Wed, 04 Feb 2026 13:54:17 GMT foundInMaster: true - source: https://openapi.tools/ name: BaucisJS + baucis-openapi3 homepage: https://github.com/metadevpro/baucis-openapi3 language: Node.js source_description: >- Create REST resources with persistence on MongoDB and expose OpenAPI v.3 contracts category: Server link: https://www.npmjs.com/package/baucis-openapi3 repository: https://github.com/metadevpro/baucis-openapi3 v2: false v3: true repositoryMetadata: base64Readme: >- YmF1Y2lzLW9wZW5hcGkzCj09PT09PT09PT09PT09PQoKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL21ldGFkZXZwcm8vYmF1Y2lzLW9wZW5hcGkzLnN2ZyldKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9tZXRhZGV2cHJvL2JhdWNpcy1vcGVuYXBpMykKWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvbWV0YWRldnByby9iYXVjaXMtb3BlbmFwaTMvYmFkZ2Uuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL2NvdmVyYWxscy5pby9naXRodWIvbWV0YWRldnByby9iYXVjaXMtb3BlbmFwaTM/YnJhbmNoPW1hc3RlcikKWyFbRGVwZW5kZW5jeSBTdGF0dXNdKGh0dHBzOi8vZGF2aWQtZG0ub3JnL21ldGFkZXZwcm8vYmF1Y2lzLW9wZW5hcGkzLnN2ZyldKGh0dHBzOi8vZGF2aWQtZG0ub3JnL21ldGFkZXZwcm8vYmF1Y2lzLW9wZW5hcGkzKQpbIVtLbm93biBWdWxuZXJhYmlsaXRpZXNdKGh0dHBzOi8vc255ay5pby90ZXN0L25wbS9iYXVjaXMtb3BlbmFwaTMvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9zbnlrLmlvL3Rlc3QvbnBtL2JhdWNpcy1vcGVuYXBpMykKWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9iYXVjaXMtb3BlbmFwaTMuc3ZnKV0oaHR0cDovL2JhZGdlLmZ1cnkuaW8vanMvYmF1Y2lzLW9wZW5hcGkzKQpbIVtHcmVlbmtlZXBlciBiYWRnZV0oaHR0cHM6Ly9iYWRnZXMuZ3JlZW5rZWVwZXIuaW8vbWV0YWRldnByby9iYXVjaXMtb3BlbmFwaTMuc3ZnKV0oaHR0cHM6Ly9ncmVlbmtlZXBlci5pby8pCgpbIVtOUE1dKGh0dHBzOi8vbm9kZWkuY28vbnBtL2JhdWNpcy1vcGVuYXBpMy5wbmc/ZG93bmxvYWRzPXRydWUmZG93bmxvYWRSYW5rPXRydWUmc3RhcnM9dHJ1ZSldKGh0dHBzOi8vbm9kZWkuY28vbnBtL2JhdWNpcy1vcGVuYXBpMy8pCgoKVGhpcyBtb2R1bGUgZ2VuZXJhdGVzIGN1c3RvbWl6YWJsZSBbT3BlbkFQSSAzLjBdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kKSBkZWZpbml0aW9ucyBmb3IgeW91ciBCYXVjaXMgQVBJLgpVc2UgdGhpcyBtb2R1bGUgaW4gY29uanVuY3Rpb24gd2l0aCBbQmF1Y2lzXShodHRwczovL2dpdGh1Yi5jb20vd3BybC9iYXVjaXMpLgoKClVzYWdlCi0tLS0tCgpJbnN0YWxsIHdpdGg6CgogICAgbnBtIGluc3RhbGwgLS1zYXZlIGJhdWNpcyBiYXVjaXMtb3BlbmFwaTMKCkluY2x1ZGUgdGhlIHBhY2thZ2UgYWZ0ZXIgYmF1Y2lzIGlzIGluY2x1ZGVkLCBhbmQgYmVmb3JlIHlvdXIgQVBJIGlzIGJ1aWx0LgoKYGBgamF2YXNjcmlwdAogICAgdmFyIGV4cHJlc3MgPSByZXF1aXJlKCdleHByZXNzJyk7CiAgICB2YXIgYmF1Y2lzID0gcmVxdWlyZSgnYmF1Y2lzJyk7CiAgICB2YXIgYXBpRG9jID0gcmVxdWlyZSgnYmF1Y2lzLW9wZW5hcGkzJyk7CgogICAgdmFyIGFwcCA9IGV4cHJlc3MoKTsKCiAgICAvLyAuLi4gU2V0IHVwIGEgbW9uZ29vc2Ugc2NoZW1hIC4uLgoKICAgIGJhdWNpcy5yZXN0KCd2ZWdldGFibGUnKTsKICAgIGFwcC51c2UoJy9hcGknLCBiYXVjaXMoKSk7CmBgYAoKVGhlbiwgYWNjZXNzIGUuZy4gYEdFVCBodHRwOi8vbG9jYWxob3N0OjMzMzMvYXBpL29wZW5hcGkuanNvbmAuICBTZWUgdGhlIFtCYXVjaXNdKGh0dHBzOi8vZ2l0aHViLmNvbS93cHJsL2JhdWNpcykgcmVwbyBmb3IgbW9yZSBpbmZvcm1hdGlvbiBhYm91dCBidWlsZGluZyBSRVNUIEFQSXMgd2l0aCBbQmF1Y2lzXShodHRwczovL2dpdGh1Yi5jb20vd3BybC9iYXVjaXMpLgoKVGVzdHMKLS0tLS0KQ2hhbmdlIHRoZSBgdGVzdC9maXh1cmVzL2NvbmZpZy5qc29uYCBpZiBuZWVkZWQsIHRvIHBvaW50IHRvIGEgdmFsaWQgbW9uZ29kYiBkYXRhYmFzZS4KVGhlbiBydW46CgpgYGAKbnBtIHRlc3QKYGBgCgoKRXh0ZW5zaWJpbGl0eQotLS0tLS0tLS0tLS0tCgpJZiB5b3Ugd2FudCB0byBtb2RpZnkgdGhlIE9wZW5BUEkgZGVmaW5pdGlvbiwgZ2VuZXJhdGUgdGhlIGRlZmluaXRpb24gZmlyc3QuICAoVGhpcyB3aWxsIGhhcHBlbiBhdXRvbWF0aWNhbGx5IG90aGVyd2lzZS4pCgpVc2UgdGhlIGBvcGVuQXBpM2AgbWVtYmVyIG9mIHRoZSBjb250cm9sbGVyIHRvIGV4dGVuZCBgcGF0aHNgIGFuZCBgY29tcG9uZW50c2AgcGVyIGNvbnRyb2xsZXIuCgpgYGBqYXZhc2NyaXB0CmNvbnRyb2xsZXIuZ2VuZXJhdGVPcGVuQXBpMygpOwpjb250cm9sbGVyLm9wZW5BcGkzLnBhdGhzLnh5eiA9ICcxMjMnOwpjb250cm9sbGVyLm9wZW5BcGkzLmNvbXBvbmVudHMuc2NoZW1hcy54eXogPSB7fTsKYGBgCgpPciB1c2UgdGhlIGBvcGVuQXBpM0RvY3VtZW50YCBvZiB0aGUgYmF1Y2lzIGluc3RhbmNlIG1vZHVsZSB0byBhY2Nlc3MgYW5kIG1vZGlmeSBkaXJyZWN0eSB0aGUgZnVsbCBkb2N1bWVudCBhZnRlciBjYWxsaW5nIGBnZW5lcmF0ZU9wZW5BcGkzKClgIG9uIHRoZSBBUEkuCgpgYGBqYXZhc2NyaXB0CnZhciBiYXVjaXNJbnN0YW5jZSA9IGJhdWNpcygpOwoKLy9nZW5lcmF0ZSBzdGFuZGFyZCB0ZW1wbGF0ZSBmb3IgT3BlbkFQSTMKYmF1Y2lzSW5zdGFuY2UuZ2VuZXJhdGVPcGVuQXBpMygpOwovL2V4dGVuZCBPcGVuQVBJMyBkZWZpbml0aW9ucwpiYXVjaXNJbnN0YW5jZS5vcGVuQXBpM0RvY3VtZW50LmluZm8udGl0bGUgPSAibXlBcGkiOwoKYXBwLnVzZSgnL2FwaScsIGJhdWNpc0luc3RhbmNlKTsKYGBgCgpCYWNrd2FyZCBjb21wYXRpYmlsaXR5Ci0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCkluIGNhc2UgeW91IHdhbnQgdG8gcHJvdmlkZSBhbiBlYXN5IHRyYW5zaXRpb24gYXMgcG9zc2libGUgZm9yIHlvdXIgY3VycmVudCBBUEkgY2xpZW50cy4gWW91IGNhbiBleHBvc2UgYm90aCBBUEkgZGVzY3JpcHRpb25zIGF0IHRoZSBzYW1lIHRpbWUgaW5jbHVkaW5nIGJvdGggbW9kdWxlczoKCmBgYGphdmFzY3JpcHQKICAgIHZhciBleHByZXNzID0gcmVxdWlyZSgnZXhwcmVzcycpOwogICAgdmFyIGJhdWNpcyA9IHJlcXVpcmUoJ2JhdWNpcycpOwogICAgdmFyIHN3YWdnZXIgPSByZXF1aXJlKCdiYXVjaXMtc3dhZ2dlcicpOwogICAgdmFyIHN3YWdnZXIyID0gcmVxdWlyZSgnYmF1Y2lzLXN3YWdnZXIyJyk7CiAgICB2YXIgb3BlbmFwaTMgPSByZXF1aXJlKCdiYXVjaXMtb3BlbmFwaTMnKTsKCiAgICB2YXIgYXBwID0gZXhwcmVzcygpOwoKICAgIC8vIC4uLiBTZXQgdXAgYSBtb25nb29zZSBzY2hlbWEgLi4uCgogICAgYmF1Y2lzLnJlc3QoJ3ZlZ2V0YWJsZScpOwogICAgYXBwLnVzZSgnL2FwaScsIGJhdWNpcygpKTsKYGBgCgpBZnRlciB0aGF0OgotIFN3YWdnZXIgMS4xIGRvYyB3aWxsIGJlIGV4cG9zZWQgYXQgYC9hcGkvZG9jdW1lbnRhdGlvbmAKLSBTd2FnZ2VyIDIuMCBkb2Mgd2lsbCBiZSBleHBvc2VkIGF0IGAvYXBpL3N3YWdnZXIuanNvbmAKLSBPcGVuQVBJIDMuMCBkb2Mgd2lsbCBiZSBleHBvc2VkIGF0IGAvYXBpL29wZW5hcGkuanNvbmAKCgpDb250YWN0Ci0tLS0tLS0KCiAqIFtAbWV0YWQzdl0oaHR0cHM6Ly90d2l0dGVyLmNvbS9tZXRhZDN2KSBodHRwczovL21ldGFkZXYucHJvCiAqIFtAcG1vbGluYW1dKGh0dHBzOi8vdHdpdHRlci5jb20vcG1vbGluYW0pCgpSZWxlYXNlZCB1bmRlciBNSVQgTGljZW5zZS4gJmNvcHk7IDIwMTctMjAxOCBNZXRhZGV2Cg== readmeEtag: '"911d64ea488af9377794cca8eec31a8a217f35df"' readmeLastModified: Mon, 01 Apr 2019 21:55:30 GMT repositoryId: 83848050 description: BaucisJS pluging for generating OpenAPI 3.0 compliance API contracts. created: '2017-03-03T22:41:11Z' updated: '2025-04-09T12:48:50Z' language: JavaScript archived: false stars: 14 watchers: 2 forks: 2 owner: metadevpro logo: https://avatars.githubusercontent.com/u/24300914?v=4 license: MIT repoEtag: '"2760f8026b3a9114fd5192e806dc46aa306999f9230c8e7ca179fe4a70777f5a"' repoLastModified: Wed, 09 Apr 2025 12:48:50 GMT foundInMaster: true id: e39508ff695c02cf804c421dd228064e - source: - https://openapi.tools/ - openapi3 tags name: '@smartrecruiters/openapi-first' category: - Server - Parsers language: Node.js link: https://www.npmjs.com/package/@smartrecruiters/openapi-first repository: https://github.com/smartrecruiters/openapi-first source_description: > Initializes your API express application with the description in OpenAPI 3.0 format using provided middlewares (parsers, validators, controller, defaults setting) or custom ones v2: false v3: true repositoryMetadata: base64Readme: >- # @smartrecruiters/openapi-first

[![NPM Version][npm-image]][npm-url]
[![NPM Downloads][downloads-image]][downloads-url]
[![Node.js Version][node-version-image]][node-version-url]
[![Licence][license-image]][license-url]
[![Build][travis-image]][travis-url]

Start your node REST app with designing API first!

## Is it for you?

If you:
- use OpenAPI Specification 3.0 to document your REST APIs written in node.js,
- like design first approach regarding REST APIs
- want your specification to be single source of truth of your API,
- want to handle validation and parsing of requests query, path, body, content-type in a unified manner for all API endpoints,

then `@smartrecruiters/openapi-first` is what you are looking for!

This module initializes your API connect-style application with specification in
[OpenAPI Specification 3.0](https://openapis.org/specification) format.

## How to start

Let's say you have specification in OpenAPI Specification 3.0 in `spec.json`:

```javascript
{
    "openapi": "3.0.0",
    "info": {
        "version": "1.0.0",
        "title": "Hello World API"
    },
    "paths": {
        "/hello": {
            "get": {
                "responses": {
                    "200": {
                        "description": "Success",
                        "content": {
                            "application/json": {
                                "schema": {
                                    "$ref": "#/components/schemas/Greeting"
                                }
                            }
                        }
                    }
                },
                "x-swagger-router-controller": "greeting/hello",
                "parameters": [
                    {
                        "in": "query",
                        "name": "name",
                        "schema": {
                            "type": "string",
                            "default": "world"
                        }
                    }
                ]
            }
        }
    },
    "components": {
        "schemas": {
            "Greeting": {
                "type": "object",
                "properties": {
                    "greeting": {
                        "type": "string"
                    }
                }
            }
        }
    }
}
```

Now you can implement connect-style middleware with "business logic" in `greeting/hello.js`:

```javascript
module.exports = function(req, res) {
    res.status(200).json({greeting: `Hello, ${req.query.name}!`})
}
```

Now, let's make an `app.js` file:
```javascript
const openApiFirst = require('@smartrecruiters/openapi-first')
const express = require('express')

// create express app
const app = express()

const spec = require('./spec.json')

// create open api specification initializer
const api = openApiFirst(app, spec)

// to enable setting default values on empty query params
api.use(require('@smartrecruiters/openapi-first/middlewares/query/defaults')())

// to link the specification with code in 'api' directory
api.use(require('@smartrecruiters/openapi-first/middlewares/controllers/by-property')({dir: __dirname}))

app.listen(8080)

```

We can now run the application:
```bash
node app
```
To verify it's working, let's hit the endpoint:
```bash
curl localhost:8080/hello?who=world
```
The response should be 200 with body:
```json
{"greeting":"Hello, world!"}
```

## openapi middlewares

You can use one of the middlewares under `@smartrecruiters/middlewares/*` or create your own. Such middlewares will be
applied to connect-style app for each operation as they are specification and operation aware. For instance,
`@smartrecruiters/middlewares/query/validate` middleware will be applied to any and only operation which has query
parameters defined, passing an Error to `next` callback when `req.query` is invalid.

Currently following middlewares are available:
- request body validation,
- request body parsing (e.g. form string parameters to types specified in API documentation)
- setting default values on request body,
- query parameters validation,
- setting default query parameters values
- removing unspecified query parameters,
- path parameters validation,
- content type validation,
- routing to appropriate controller,
- oauth scopes authorization,
- error handlers (`MissingRequiredScopes`).

### Validation middlewares

Middlewares for request body, query and path validation expects schema validators in order to be created.
The recommended schema validator is [`@smartrecruiters/openapi-schema-validator`](https://www.npmjs.com/package/@smartrecruiters/openapi-schemas-validator)

### Create your own openapi middleware

Adding your own openapi is very simple. Let's say your operation has extension
[OpenAPI Specification 3.0 Specification Extension](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.1.md#specificationExtensions) 'x-only-admin'.
If it is set on, this will mean that only users with admin can use this method.
Assuming some preceding middleware is setting `req.user.role`, you can write a simple openapi middleware
that will gather information from operation object and act accordingly:
```javascript
const onlyAdminMiddleware = operation =>
    (req, res, next) => {
        if(operation['x-only-admin'] && req.user.role !== "admin") {
            res.status(403).json("Access forbidden. For this operation, you need to have admin role")
        }
        return next()
    }
```

## Contributing

Please see our [Code of conduct](docs/CODE_OF_CONDUCT.md) and [Contributing guidelines](docs/CONTRIBUTING.md)

## License

[MIT](LICENSE)

[npm-image]: https://img.shields.io/npm/v/@smartrecruiters/openapi-first.svg
[npm-url]: https://www.npmjs.com/package/@smartrecruiters/openapi-first
[downloads-image]: https://img.shields.io/npm/dm/@smartrecruiters/openapi-first.svg
[downloads-url]: https://www.npmjs.com/package/@smartrecruiters/openapi-first
[node-version-image]: https://img.shields.io/node/v/@smartrecruiters/openapi-first.svg
[node-version-url]: https://nodejs.org/en/download/
[license-url]: https://github.com/smartrecruiters/openapi-first/blob/master/LICENSE
[license-image]: https://img.shields.io/npm/l/@smartrecruiters/openapi-first.svg
[travis-url]: https://travis-ci.com/smartrecruiters/openapi-first
[travis-image]: https://api.travis-ci.com/smartrecruiters/openapi-first.svg?branch=master
 readmeEtag: '"9b52f3708e00f17d78a68292d12aed651637223f"' readmeLastModified: Wed, 07 Aug 2024 14:25:42 GMT repositoryId: 124390318 description: Start your node REST app with designing API first! created: '2018-03-08T12:46:22Z' updated: '2025-11-19T14:40:23Z' language: JavaScript archived: false stars: 40 watchers: 31 forks: 5 owner: smartrecruiters logo: https://avatars.githubusercontent.com/u/1618540?v=4 license: MIT repoEtag: '"0e0fd5cf9526d0221b780afa93b1b5d4e6dc77a3b7ecceee5bb127abcac1ac8a"' repoLastModified: Wed, 19 Nov 2025 14:40:23 GMT foundInMaster: true id: b6882b9c2b3386fd0dac448d07cbcb36 - source: https://openapi.tools/ name: API Platform category: - Server - Server Implementations language: PHP link: https://api-platform.com repository: https://github.com/api-platform/api-platform source_description: REST and GraphQL framework to build modern API-driven projects v2: true v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPjxhIGhyZWY9Imh0dHBzOi8vYXBpLXBsYXRmb3JtLmNvbSI+PGltZyBzcmM9Imh0dHBzOi8vYXBpLXBsYXRmb3JtLmNvbS9pbWFnZXMvbG9nb3MvTG9nb19DaXJjbGUlMjB3ZWJieSUyMHRleHQlMjBibHVlLnBuZyIgYWx0PSJBUEkgUGxhdGZvcm0iIHdpZHRoPSIyNTAiIGhlaWdodD0iMjUwIj48L2E+PC9oMT4KCkFQSSBQbGF0Zm9ybSBpcyBhIG5leHQtZ2VuZXJhdGlvbiB3ZWIgZnJhbWV3b3JrIGRlc2lnbmVkIHRvIGVhc2lseSBjcmVhdGUgQVBJLWZpcnN0IHByb2plY3RzIHdpdGhvdXQgY29tcHJvbWlzaW5nIGV4dGVuc2liaWxpdHkKYW5kIGZsZXhpYmlsaXR5OgoKKiBEZXNpZ24geW91ciBvd24gZGF0YSBtb2RlbCBhcyBwbGFpbiBvbGQgUEhQIGNsYXNzZXMgb3IgWyoqaW1wb3J0IGFuIGV4aXN0aW5nIG9udG9sb2d5KipdKGh0dHBzOi8vYXBpLXBsYXRmb3JtLmNvbS9kb2NzL3NjaGVtYS1nZW5lcmF0b3IpLgoqICoqRXhwb3NlIGluIG1pbnV0ZXMgYSBoeXBlcm1lZGlhIFJFU1Qgb3IgYSBHcmFwaFFMIEFQSSoqIHdpdGggcGFnaW5hdGlvbiwgZGF0YSB2YWxpZGF0aW9uLCBhY2Nlc3MgY29udHJvbCwgcmVsYXRpb24gZW1iZWRkaW5nLAogIGZpbHRlcnMsIGFuZCBlcnJvciBoYW5kbGluZy4uLgoqIEJlbmVmaXQgZnJvbSBDb250ZW50IE5lZ290aWF0aW9uOiBbR3JhcGhRTF0oaHR0cHM6Ly9hcGktcGxhdGZvcm0uY29tL2RvY3MvY29yZS9ncmFwaHFsLyksIFtKU09OLUxEXShodHRwczovL2pzb24tbGQub3JnKSwgW0h5ZHJhXShodHRwczovL2h5ZHJhLWNnLmNvbSksCiAgW0hBTF0oaHR0cHM6Ly9naXRodWIuY29tL21pa2VrZWxseS9oYWxfc3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9oYWxfc3BlY2lmaWNhdGlvbi5tZCksIFtKU09OOkFQSV0oaHR0cHM6Ly9qc29uYXBpLm9yZy8pLCBbWUFNTF0oaHR0cHM6Ly95YW1sLm9yZy8pLCBbSlNPTl0oaHR0cHM6Ly93d3cuanNvbi5vcmcvKSwgW1hNTF0oaHR0cHM6Ly93d3cudzMub3JnL1hNTC8pIGFuZCBbQ1NWXShodHRwczovL3d3dy5pZXRmLm9yZy9yZmMvcmZjNDE4MC50eHQpIGFyZSBzdXBwb3J0ZWQgb3V0IG9mIHRoZSBib3guCiogRW5qb3kgdGhlICoqYmVhdXRpZnVsIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkIEFQSSBkb2N1bWVudGF0aW9uKiogKFtPcGVuQVBJXShodHRwczovL2FwaS1wbGF0Zm9ybS5jb20vZG9jcy9jb3JlL29wZW5hcGkvKSkuCiogQWRkIFsqKmEgY29udmVuaWVudCBNYXRlcmlhbCBEZXNpZ24gYWRtaW5pc3RyYXRpb24gaW50ZXJmYWNlKipdKGh0dHBzOi8vYXBpLXBsYXRmb3JtLmNvbS9kb2NzL2FkbWluKSBidWlsdCB3aXRoIFtSZWFjdF0oaHR0cHM6Ly9yZWFjdGpzLm9yZy8pCiAgd2l0aG91dCB3cml0aW5nIGEgbGluZSBvZiBjb2RlLgoqICoqU2NhZmZvbGQgZnVsbHkgZnVuY3Rpb25hbCBQcm9ncmVzc2l2ZS1XZWItQXBwcyBhbmQgbW9iaWxlIGFwcHMqKiBidWlsdCB3aXRoIFtOZXh0LmpzXShodHRwczovL2FwaS1wbGF0Zm9ybS5jb20vZG9jcy9jbGllbnQtZ2VuZXJhdG9yL25leHRqcy8pIChSZWFjdCksCltOdXh0LmpzXShodHRwczovL2FwaS1wbGF0Zm9ybS5jb20vZG9jcy9jbGllbnQtZ2VuZXJhdG9yL251eHRqcy8pIChWdWUuanMpIG9yIFtSZWFjdCBOYXRpdmVdKGh0dHBzOi8vYXBpLXBsYXRmb3JtLmNvbS9kb2NzL2NsaWVudC1nZW5lcmF0b3IvcmVhY3QtbmF0aXZlLykKdGhhbmtzIHRvIFt0aGUgY2xpZW50IGdlbmVyYXRvcl0oaHR0cHM6Ly9hcGktcGxhdGZvcm0uY29tL2RvY3MvY2xpZW50LWdlbmVyYXRvci8pIChhIFZ1ZS5qcyBnZW5lcmF0b3IgaXMgYWxzbyBhdmFpbGFibGUpLgoqIEluc3RhbGwgYSBkZXZlbG9wbWVudCBlbnZpcm9ubWVudCBhbmQgZGVwbG95IHlvdXIgcHJvamVjdCBpbiBwcm9kdWN0aW9uIHVzaW5nICoqW0RvY2tlcl0oaHR0cHM6Ly9hcGktcGxhdGZvcm0uY29tL2RvY3MvZGlzdHJpYnV0aW9uKSoqCmFuZCBbS3ViZXJuZXRlc10oaHR0cHM6Ly9hcGktcGxhdGZvcm0uY29tL2RvY3MvZGVwbG95bWVudC9rdWJlcm5ldGVzKS4KKiBFYXNpbHkgYWRkICoqW09BdXRoXShodHRwczovL29hdXRoLm5ldC8pIGF1dGhlbnRpY2F0aW9uKiouCiogQ3JlYXRlIHNwZWNzIGFuZCB0ZXN0cyB3aXRoICoqW2EgZGV2ZWxvcGVyIGZyaWVuZGx5IEFQSSB0ZXN0aW5nIHRvb2xdKGh0dHBzOi8vYXBpLXBsYXRmb3JtLmNvbS9kb2NzL2Rpc3RyaWJ1dGlvbi90ZXN0aW5nLykqKi4KClRoZSBvZmZpY2lhbCBwcm9qZWN0IGRvY3VtZW50YXRpb24gaXMgYXZhaWxhYmxlICoqW29uIHRoZSBBUEkgUGxhdGZvcm0gd2Vic2l0ZV0oaHR0cHM6Ly9hcGktcGxhdGZvcm0uY29tKSoqLgoKQVBJIFBsYXRmb3JtIGVtYnJhY2VzIG9wZW4gd2ViIHN0YW5kYXJkcyBhbmQgdGhlCltMaW5rZWQgRGF0YV0oaHR0cHM6Ly93d3cudzMub3JnL3N0YW5kYXJkcy9zZW1hbnRpY3dlYi9kYXRhKSBtb3ZlbWVudC4gWW91ciBBUEkgd2lsbCBhdXRvbWF0aWNhbGx5IGV4cG9zZSBzdHJ1Y3R1cmVkIGRhdGEuCkl0IG1lYW5zIHRoYXQgeW91ciBBUEkgUGxhdGZvcm0gYXBwbGljYXRpb24gaXMgdXNhYmxlICoqb3V0IG9mIHRoZSBib3gqKiB3aXRoIHRlY2hub2xvZ2llcyBvZgp0aGUgc2VtYW50aWMgd2ViLgoKSXQgYWxzbyBtZWFucyB0aGF0ICoqeW91ciBTRU8gd2lsbCBiZSBpbXByb3ZlZCoqIGJlY2F1c2UgKipbR29vZ2xlIGxldmVyYWdlcyB0aGVzZSBmb3JtYXRzXShodHRwczovL2RldmVsb3BlcnMuZ29vZ2xlLmNvbS9zZWFyY2gvZG9jcy9ndWlkZXMvaW50cm8tc3RydWN0dXJlZC1kYXRhKSoqLgoKTGFzdCBidXQgbm90IGxlYXN0LCB0aGUgc2VydmVyIGNvbXBvbmVudCBvZiBBUEkgUGxhdGZvcm0gaXMgYnVpbHQgb24gdG9wIG9mIHRoZSBbU3ltZm9ueV0oaHR0cHM6Ly9zeW1mb255LmNvbSkgZnJhbWV3b3JrLAp3aGlsZSBjbGllbnQgY29tcG9uZW50cyBsZXZlcmFnZSBbUmVhY3RdKGh0dHBzOi8vcmVhY3Rqcy5vcmcvKSAoW1Z1ZS5qc10oaHR0cHM6Ly92dWVqcy5vcmcvKSBmbGF2b3JzIGFyZSBhbHNvIGF2YWlsYWJsZSkuCkl0IG1lYW5zIHRoYXQgeW91IGNhbjoKCiogVXNlICoqdGhvdXNhbmRzIG9mIFN5bWZvbnkgYnVuZGxlcyBhbmQgUmVhY3QgY29tcG9uZW50cyoqIHdpdGggQVBJIFBsYXRmb3JtLgoqIEludGVncmF0ZSBBUEkgUGxhdGZvcm0gaW4gKiphbnkgZXhpc3RpbmcgU3ltZm9ueSwgUmVhY3QsIG9yIFZ1ZSBhcHBsaWNhdGlvbioqLgoqIFJldXNlICoqYWxsIHlvdXIgU3ltZm9ueSBhbmQgSmF2YVNjcmlwdCBza2lsbHMqKiwgYW5kIGJlbmVmaXQgZnJvbSB0aGUgaW5jcmVkaWJsZSBhbW91bnQgb2YgZG9jdW1lbnRhdGlvbiBhdmFpbGFibGUuCiogRW5qb3kgdGhlIHBvcHVsYXIgW0RvY3RyaW5lIE9STV0oaHR0cHM6Ly93d3cuZG9jdHJpbmUtcHJvamVjdC5vcmcvcHJvamVjdHMvb3JtLmh0bWwpICh1c2VkIGJ5IGRlZmF1bHQsIGJ1dCBmdWxseSBvcHRpb25hbDoKICB5b3UgY2FuIHVzZSB0aGUgZGF0YSBwcm92aWRlciB5b3Ugd2FudCwgaW5jbHVkaW5nIGJ1dCBub3QgbGltaXRlZCB0byBNb25nb0RCIGFuZCBFbGFzdGljc2VhcmNoKQoKIyMgSW5zdGFsbAoKW1JlYWQgdGhlIG9mZmljaWFsICJHZXR0aW5nIFN0YXJ0ZWQiIGd1aWRlXShodHRwczovL2FwaS1wbGF0Zm9ybS5jb20vZG9jcy9kaXN0cmlidXRpb24vKS4KCiMjIENyZWRpdHMKCkNyZWF0ZWQgYnkgW0vDqXZpbiBEdW5nbGFzXShodHRwczovL2R1bmdsYXMuZnIpLiBDb21tZXJjaWFsIHN1cHBvcnQgaXMgYXZhaWxhYmxlIGF0IFtMZXMtVGlsbGV1bHMuY29vcF0oaHR0cHM6Ly9sZXMtdGlsbGV1bHMuY29vcCkuCg== readmeEtag: '"76a11567f28b360e0fc0597ffdefb227a6c9cf86"' readmeLastModified: Tue, 06 Aug 2024 14:12:56 GMT repositoryId: 31790438 description: >- 🕸️ Create REST and GraphQL APIs, scaffold Jamstack webapps, stream changes in real-time. created: '2015-03-06T21:46:05Z' updated: '2026-02-04T14:46:42Z' language: TypeScript archived: false stars: 9105 watchers: 210 forks: 979 owner: api-platform logo: https://avatars.githubusercontent.com/u/13420081?v=4 license: MIT repoEtag: '"748cb3ed44d2fc2f18ab17a2423661e1a57c5970dab6ba1584f4bea45d9683a1"' repoLastModified: Wed, 04 Feb 2026 14:46:42 GMT foundInMaster: true id: d14f5904300aabf1ceef4660a6b3d57e - source: https://openapi.tools/ name: Mojolicious::Plugin::OpenApi category: Server language: Perl link: https://metacpan.org/pod/Mojolicious::Plugin::OpenAPI repository: https://github.com/jhthorsen/mojolicious-plugin-openapi source_description: > Mojolicious::Plugin::OpenAPI is a plugin for Mojolicious framework that add routes and input/output validation to your Mojolicious application based on OpenAPI description documents.' v2: true v3: true repositoryMetadata: base64Readme: >- # NAME

Mojolicious::Plugin::OpenAPI - OpenAPI / Swagger plugin for Mojolicious

# SYNOPSIS

    # It is recommended to use Mojolicious::Plugin::OpenAPI with a "full app".
    # See the links after this example for more information.
    use Mojolicious::Lite;

    # Because the route name "echo" matches the "x-mojo-name", this route
    # will be moved under "basePath", resulting in "POST /api/echo"
    post "/echo" => sub {

      # Validate input request or return an error document
      my $c = shift->openapi->valid_input or return;

      # Generate some data
      my $data = {body => $c->req->json};

      # Validate the output response and render it to the user agent
      # using a custom "openapi" handler.
      $c->render(openapi => $data);
    }, "echo";

    # Load specification and start web server
    plugin OpenAPI => {url => "data:///swagger.yaml"};
    app->start;

    __DATA__
    @@ swagger.yaml
    swagger: "2.0"
    info: { version: "0.8", title: "Echo Service" }
    schemes: ["https"]
    basePath: "/api"
    paths:
      /echo:
       post:
         x-mojo-name: "echo"
         parameters:
         - { in: "body", name: "body", schema: { type: "object" } }
         responses:
           200:
             description: "Echo response"
             schema: { type: "object" }

See [Mojolicious::Plugin::OpenAPI::Guides::OpenAPIv2](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3AGuides%3A%3AOpenAPIv2) or
[Mojolicious::Plugin::OpenAPI::Guides::OpenAPIv3](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3AGuides%3A%3AOpenAPIv3) for more in depth
information about how to use [Mojolicious::Plugin::OpenAPI](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI) with a "full app".
Even with a "lite app" it can be very useful to read those guides.

Looking at the documentation for
["x-mojo-to" in Mojolicious::Plugin::OpenAPI::Guides::OpenAPIv2](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3AGuides%3A%3AOpenAPIv2#x-mojo-to) can be especially
useful. (The logic is the same for OpenAPIv2 and OpenAPIv3)

# DESCRIPTION

[Mojolicious::Plugin::OpenAPI](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI) is [Mojolicious::Plugin](https://metacpan.org/pod/Mojolicious%3A%3APlugin) that add routes and
input/output validation to your [Mojolicious](https://metacpan.org/pod/Mojolicious) application based on a OpenAPI
(Swagger) specification. This plugin supports both version [2.0](#schema) and
[3.x](#schema), though 3.x _might_ have some missing features.

Have a look at the ["SEE ALSO"](#see-also) for references to plugins and other useful
documentation.

Please report in [issues](https://github.com/jhthorsen/json-validator/issues)
or open pull requests to enhance the 3.0 support.

# HELPERS

## openapi.spec

    $hash = $c->openapi->spec($json_pointer)
    $hash = $c->openapi->spec("/info/title")
    $hash = $c->openapi->spec;

Returns the OpenAPI specification. A JSON Pointer can be used to extract a
given section of the specification. The default value of `$json_pointer` will
be relative to the current operation. Example:

    {
      "paths": {
        "/pets": {
          "get": {
            // This datastructure is returned by default
          }
        }
      }
    }

## openapi.validate

    @errors = $c->openapi->validate;

Used to validate a request. `@errors` holds a list of
[JSON::Validator::Error](https://metacpan.org/pod/JSON%3A%3AValidator%3A%3AError) objects or empty list on valid input.

Note that this helper is only for customization. You probably want
["openapi.valid\_input"](#openapi-valid_input) in most cases.

## openapi.valid\_input

    $c = $c->openapi->valid_input;

Returns the [Mojolicious::Controller](https://metacpan.org/pod/Mojolicious%3A%3AController) object if the input is valid or
automatically render an error document if not and return false. See
["SYNOPSIS"](#synopsis) for example usage.

# HOOKS

[Mojolicious::Plugin::OpenAPI](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI) will emit the following hooks on the
[application](https://metacpan.org/pod/Mojolicious) object.

## openapi\_routes\_added

Emitted after all routes have been added by this plugin.

    $app->hook(openapi_routes_added => sub {
      my ($openapi, $routes) = @_;

      for my $route (@$routes) {
        ...
      }
    });

This hook is EXPERIMENTAL and subject for change.

# RENDERER

This plugin register a new handler called `openapi`. The special thing about
this handler is that it will validate the data before sending it back to the
user agent. Examples:

    $c->render(json => {foo => 123});    # without validation
    $c->render(openapi => {foo => 123}); # with validation

This handler will also use ["renderer"](#renderer) to format the output data. The code
below shows the default ["renderer"](#renderer) which generates JSON data:

    $app->plugin(
      OpenAPI => {
        renderer => sub {
          my ($c, $data) = @_;
          return Mojo::JSON::encode_json($data);
        }
      }
    );

# ATTRIBUTES

## route

    $route = $openapi->route;

The parent [Mojolicious::Routes::Route](https://metacpan.org/pod/Mojolicious%3A%3ARoutes%3A%3ARoute) object for all the OpenAPI endpoints.

## validator

    $jv = $openapi->validator;

Holds either a [JSON::Validator::Schema::OpenAPIv2](https://metacpan.org/pod/JSON%3A%3AValidator%3A%3ASchema%3A%3AOpenAPIv2) or a
[JSON::Validator::Schema::OpenAPIv3](https://metacpan.org/pod/JSON%3A%3AValidator%3A%3ASchema%3A%3AOpenAPIv3) object.

# METHODS

## register

    $openapi = $openapi->register($app, \%config);
    $openapi = $app->plugin(OpenAPI => \%config);

Loads the OpenAPI specification, validates it and add routes to
[$app](https://metacpan.org/pod/Mojolicious). It will also set up ["HELPERS"](#helpers) and adds a
[before\_render](https://metacpan.org/pod/Mojolicious#before_render) hook for auto-rendering of error
documents. The return value is the object instance, which allow you to access
the ["ATTRIBUTES"](#attributes) after you load the plugin.

`%config` can have:

### coerce

See ["coerce" in JSON::Validator](https://metacpan.org/pod/JSON%3A%3AValidator#coerce) for possible values that `coerce` can take.

Default: booleans,numbers,strings

The default value will include "defaults" in the future, once that is stable enough.

### default\_response

Instructions for
["add\_default\_response\_schema" in JSON::Validator::Schema::OpenAPIv2](https://metacpan.org/pod/JSON%3A%3AValidator%3A%3ASchema%3A%3AOpenAPIv2#add_default_response_schema). (Also used
for OpenAPIv3)

### format

Set this to a default list of file extensions that your API accepts. This value
can be overwritten by
["x-mojo-to" in Mojolicious::Plugin::OpenAPI::Guides::OpenAPIv2](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3AGuides%3A%3AOpenAPIv2#x-mojo-to).

This config parameter is EXPERIMENTAL and subject for change.

### log\_level

`log_level` is used when logging invalid request/response error messages.

Default: "warn".

### op\_spec\_to\_route

`op_spec_to_route` can be provided if you want to add route definitions
without using "x-mojo-to". Example:

    $app->plugin(OpenAPI => {op_spec_to_route => sub {
      my ($plugin, $op_spec, $route) = @_;

      # Here are two ways to customize where to dispatch the request
      $route->to(cb => sub { shift->render(openapi => ...) });
      $route->to(ucfirst "$op_spec->{operationId}#handle_request");
    }});

This feature is EXPERIMENTAL and might be altered and/or removed.

### plugins

A list of OpenAPI classes to extend the functionality. Default is:
[Mojolicious::Plugin::OpenAPI::Cors](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3ACors),
[Mojolicious::Plugin::OpenAPI::SpecRenderer](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3ASpecRenderer) and
[Mojolicious::Plugin::OpenAPI::Security](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3ASecurity).

    $app->plugin(OpenAPI => {plugins => [qw(+Cors +SpecRenderer +Security)]});

You can load your own plugins by doing:

    $app->plugin(OpenAPI => {plugins => [qw(+SpecRenderer My::Cool::OpenAPI::Plugin)]});

### renderer

See ["RENDERER"](#renderer).

### route

`route` can be specified in case you want to have a protected API. Example:

    $app->plugin(OpenAPI => {
      route => $app->routes->under("/api")->to("user#auth"),
      url   => $app->home->rel_file("cool.api"),
    });

### skip\_validating\_specification

Used to prevent calling ["errors" in JSON::Validator::Schema::OpenAPIv2](https://metacpan.org/pod/JSON%3A%3AValidator%3A%3ASchema%3A%3AOpenAPIv2#errors) for the
specification.

### spec\_route\_name

Name of the route that handles the "basePath" part of the specification and
serves the specification. Defaults to "x-mojo-name" in the specification at
the top level.

### spec, url

See ["schema" in JSON::Validator](https://metacpan.org/pod/JSON%3A%3AValidator#schema) for the different `url` formats that is
accepted.

`spec` is an alias for "url", which might make more sense if your
specification is written in perl, instead of JSON or YAML.

Here are some common uses:

    $app->plugin(OpenAPI => {url  => $app->home->rel_file('openapi.yaml'));
    $app->plugin(OpenAPI => {url  => 'https://example.com/swagger.json'});
    $app->plugin(OpenAPI => {spec => JSON::Validator::Schema::OpenAPIv3->new(...)});
    $app->plugin(OpenAPI => {spec => {swagger => "2.0", paths => {...}, ...}});

### version\_from\_class

Can be used to overridden `/info/version` in the API specification, from the
return value from the `VERSION()` method in `version_from_class`.

Defaults to the current `$app`. This can be disabled by setting the
"version\_from\_class" to zero (0).

# AUTHORS

## Project Founder

Jan Henning Thorsen - `jhthorsen@cpan.org`

## Contributors

- Bernhard Graf <augensalat@gmail.com>
- Doug Bell <doug@preaction.me>
- Ed J <mohawk2@users.noreply.github.com>
- Henrik Andersen <hem@fibia.dk>
- Henrik Andersen <hem@hamster.dk>
- Ilya Rassadin <elcamlost@gmail.com>
- Jan Henning Thorsen <jan.henning@thorsen.pm>
- Jan Henning Thorsen <jhthorsen@cpan.org>
- Ji-Hyeon Gim <potatogim@gluesys.com>
- Joel Berger <joel.a.berger@gmail.com>
- Krasimir Berov <k.berov@gmail.com>
- Lars Thegler <lth@fibia.dk>
- Lee Johnson <lee@givengain.ch>
- Linn-Hege Kristensen <linn-hege@stix.no>
- Manuel <manuel@mausz.at>
- Martin Renvoize <martin.renvoize@ptfs-europe.com>
- Mohammad S Anwar <mohammad.anwar@yahoo.com>
- Nick Morrott <knowledgejunkie@gmail.com>
- Renee <reb@perl-services.de>
- Roy Storey <kiwiroy@users.noreply.github.com>
- SebMourlhou <35918953+SebMourlhou@users.noreply.github.com>
- SebMourlhou <sebastien.mourlhou@justice.ge.ch>
- SebMourlhou <sebmourlhou@yahoo.fr>
- Søren Lund <sl@keycore.dk>
- Stephan Hradek <github@hradek.net>
- Stephan Hradek <stephan.hradek@eco.de>

# COPYRIGHT AND LICENSE

Copyright (C) Jan Henning Thorsen

This program is free software, you can redistribute it and/or modify it under
the terms of the Artistic License version 2.0.

# SEE ALSO

- [Mojolicious::Plugin::OpenAPI::Guides::OpenAPIv2](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3AGuides%3A%3AOpenAPIv2)

    Guide for how to use this plugin with OpenAPI version 2.0 spec.

- [Mojolicious::Plugin::OpenAPI::Guides::OpenAPIv3](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3AGuides%3A%3AOpenAPIv3)

    Guide for how to use this plugin with OpenAPI version 3.0 spec.

- [Mojolicious::Plugin::OpenAPI::Cors](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3ACors)

    Plugin to add Cross-Origin Resource Sharing (CORS).

- [Mojolicious::Plugin::OpenAPI::Security](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3ASecurity)

    Plugin for handling security definitions in your schema.

- [Mojolicious::Plugin::OpenAPI::SpecRenderer](https://metacpan.org/pod/Mojolicious%3A%3APlugin%3A%3AOpenAPI%3A%3ASpecRenderer)

    Plugin for exposing your spec in human readable or JSON format.

- [https://www.openapis.org/](https://www.openapis.org/)

    Official OpenAPI website.
 readmeEtag: '"181a2e75e3a4070b490bafd549b9cd816d1466a1"' readmeLastModified: Sun, 26 Feb 2023 01:22:10 GMT repositoryId: 60654453 description: OpenAPI / Swagger plugin for Mojolicious created: '2016-06-07T23:50:24Z' updated: '2025-12-10T16:53:37Z' language: Perl archived: false stars: 56 watchers: 18 forks: 46 owner: jhthorsen logo: https://avatars.githubusercontent.com/u/45729?v=4 repoEtag: '"0bdcc0b8af6e5eac328cdcc9f45855111d62008979897147a01d4f1c22fdfde8"' repoLastModified: Wed, 10 Dec 2025 16:53:37 GMT foundInMaster: true id: edf3e4e2b3dc32976ffd777bb48b410c - source: https://openapi.tools/ name: Fusio homepage: https://github.com/apioo/fusio language: PHP source_description: Open source API management platform category: - Server - Gateway - Server Implementations link: https://www.fusio-project.org/ repository: https://github.com/apioo/fusio v2: false v3: true repositoryMetadata: base64Readme: >- 
<p align="center">
    <a href="https://www.fusio-project.org/" target="_blank"><img src="https://www.fusio-project.org/img/fusio_64px.png"></a>
</p>

# Fusio

Self-Hosted API Management for Builders.

## 🚀 Use Cases

* __Database API Gateway__
  > Unlock legacy databases and expose them via modern REST APIs.

* __Custom Backend Logic for APIs__
  > Build and manage custom business logic tailored to your domain.

* __Gateway for Microservices__
  > Route, orchestrate, and secure traffic between internal services.

* __API Developer Portal__
  > Provide docs, testing, and SDKs for internal or external developers.

* __API Monetization__
  > Enable freemium or tiered access with quotas, limits, and billing hooks.

* __MCP Integration__
  > Leverage the Model Context Protocol to enable AI-driven access and control of API endpoints.

* __API Usage Analytics__
  > Monitor traffic, detect issues early, and understand API consumption.

* __Headless CMS Backend__
  > Manage and expose structured content to any frontend via APIs.

* __SDK Automation__
  > Automatically generate ready-to-use client SDKs (PHP, TypeScript, Python, etc.).

## 📦 Quick Start

### 🛠️ Installation

* __Download artifact__
  > You can either download the official [release](https://github.com/apioo/fusio/releases) or clone the repository.
  ```bash
  git clone https://github.com/apioo/fusio.git
  ```

* __Set up your `.env`__
  > Configure fitting database credentials at the `APP_CONNECTION` variable, all other parameters are optional.
  > * MySQL: `pdo-mysql://root:test1234@localhost/fusio`
  > * PostgreSQL: `pdo-pgsql://postgres:postgres@localhost/fusio`
  > * SQLite: `pdo-sqlite:///fusio.sqlite`

  > It is also recommended to provide the `APP_URL` which contains the domain pointing to the public folder i.e.
    `https://api.my_domain.com` or `https://my_domain.com/fusio`, this is required if you host Fusio inside
    a sub-folder otherwise Fusio tries to detect the domain via the host header.

* __Run migrations__
  ```bash
  php bin/fusio migrate
  ```

* __Create administrator user__
  > After the installation is complete, you have to create a new administrator account. Choose as account type "Administrator".
  ```bash
  php bin/fusio adduser
  ```

* __Install backend app__
  ```bash
  php bin/fusio marketplace:install fusio
  ```

* __Start via PHP built-in server__
  > This should be only used for testing, for production you need a classical Nginx/Apache setup or use Docker, take a look at our [installation documentation](https://docs.fusio-project.org/docs/installation/) for more details.
  ```bash
  php -S 127.0.0.1:8080 -t public
  ```

### 🌐 Web-Installer

Instead of manual installation you can also use the web installer script located at `/install.php`
to complete the installation. After installation, it is recommended to delete this "install" script.

### 🐳 Docker

To run Fusio with Docker you only need the official Fusio [docker image](https://hub.docker.com/r/fusio/fusio)
and a database. The following example shows a minimal `docker-compose.yaml` which you can use to run Fusio.

```yaml
services:
  fusio:
    image: fusio/fusio
    restart: always
    environment:
      FUSIO_PROJECT_KEY: "42eec18ffdbffc9fda6110dcc705d6ce"
      FUSIO_CONNECTION: "pdo-mysql://fusio:61ad6c605975@mysql-fusio/fusio"
      FUSIO_BACKEND_USER: "test"
      FUSIO_BACKEND_EMAIL: "demo@fusio-project.org"
      FUSIO_BACKEND_PW: "test1234"
    links:
      - mysql-fusio
    ports:
      - "8080:80"

  mysql-fusio:
    image: mysql:8.0
    restart: always
    environment:
      MYSQL_RANDOM_ROOT_PASSWORD: "1"
      MYSQL_USER: "fusio"
      MYSQL_PASSWORD: "61ad6c605975"
      MYSQL_DATABASE: "fusio"
    volumes:
      - ./db:/var/lib/mysql
```

## 🧩 Apps

Fusio includes a flexible app system that lets you install various web-based apps to support
different API-related use cases. These apps are typically simple JavaScript frontends that
interact with Fusio's internal API.

You can browse all available apps in the [Fusio Marketplace](https://www.fusio-project.org/marketplace),
and install them using either the CLI:

```bash
php bin/fusio marketplace:install fusio
```

or directly through the backend interface.

### 🖥️ Backend

![Backend](https://www.fusio-project.org/media/backend/dashboard.png)

The backend app is the main app to configure and manage your API located at `/apps/fusio/`.

### 💡 VSCode

Fusio provides a [VSCode extension](https://marketplace.visualstudio.com/items?itemName=Fusio.fusio)
which can be used to simplify action development.

## 🔗 Integration

### 🧰 SDK

To build and integrate applications with Fusio, you can use one of our officially supported SDKs, which simplify interaction with a Fusio instance. Alternatively, you can directly communicate with the REST API for full control and flexibility.

| Language   | GitHub                                                  | Package                                                           | Example                                                      |
|------------|---------------------------------------------------------|-------------------------------------------------------------------|--------------------------------------------------------------|
| C#         | [GitHub](https://github.com/apioo/fusio-sdk-csharp)     | [NuGet](https://www.nuget.org/packages/Fusio.SDK)                 | [Example](https://github.com/apioo/fusio-sample-csharp-cli)  |
| Go         | [GitHub](https://github.com/apioo/fusio-sdk-go)         |                                                                   | [Example](https://github.com/apioo/fusio-sample-go-cli)      |
| Java       | [GitHub](https://github.com/apioo/fusio-sdk-java)       | [Maven](https://mvnrepository.com/artifact/org.fusio-project/sdk) | [Example](https://github.com/apioo/fusio-sample-java-cli)    |
| Javascript | [GitHub](https://github.com/apioo/fusio-sdk-javascript) | [NPM](https://www.npmjs.com/package/fusio-sdk)                    |                                                              |
| PHP        | [GitHub](https://github.com/apioo/fusio-sdk-php)        | [Packagist](https://packagist.org/packages/fusio/sdk)             | [Example](https://github.com/apioo/fusio-sample-php-cli)     |
| Python     | [GitHub](https://github.com/apioo/fusio-sdk-python)     | [PyPI](https://pypi.org/project/fusio-sdk/)                       | [Example](https://github.com/apioo/fusio-sample-python-cli)  |

### 🖥️ Frontend

| Framework | GitHub                                                           | Package                                             | Example |
|-----------|------------------------------------------------------------------|-----------------------------------------------------|---------|
| Angular   | [GitHub](https://github.com/apioo/fusio-sdk-javascript-angular)  | [NPM](https://www.npmjs.com/package/ngx-fusio-sdk)  | [Example](https://github.com/apioo/fusio-sample-javascript-angular)        |

### 📡 REST API

| Domain   | Documentation                                       | Specification                                                                           |
|----------|-----------------------------------------------------|-----------------------------------------------------------------------------------------|
| Backend  | [ReDoc](https://www.fusio-project.org/api/backend)  | [OpenAPI](https://demo.fusio-project.org/system/generator/spec-openapi?filter=backend)  |
| Consumer | [ReDoc](https://www.fusio-project.org/api/consumer) | [OpenAPI](https://demo.fusio-project.org/system/generator/spec-openapi?filter=consumer) |
| System   | [ReDoc](https://www.fusio-project.org/api/system)   | [OpenAPI](https://demo.fusio-project.org/system/generator/spec-openapi?filter=system)   |

## 🌍 Ecosystem

Besides our core product, we offer additional services to augment the functionality of Fusio.

* [Marketplace](https://www.fusio-project.org/marketplace)  
  The Fusio marketplace is the place to share apps and actions with other Fusio users, it helps to quickly build
  your API by using existing code from other users. You can register and configure the credentials at your local
  Fusio installation under System / Config s. `marketplace_client_id` and `marketplace_client_secret` then you can use
  the panel under Development / Marketplace to install apps or actions.
* [SDKgen](https://sdkgen.app/)  
  SDK as a service platform which helps you to generate client SDKs for your API in different languages like `CSharp`,
  `Go`, `Java` and `Python` which helps your customers to interact with your API. Therefor you need to register at the
  SDKgen app and provide the credentials under System / Config s. `sdkgen_client_id` and `sdkgen_client_secret`.
  Then you can generate the SDK directly at the backend under Development / SDK. 
* [TypeHub](https://typehub.cloud/)  
  API and data design platform, basically you can push your API specification to this platform so that users can simply
  discover your API. It tracks all changes of your API so that you have always a clean history how your API evolves.
* [APIgen](https://apigen.app/)  
  Service which generates fully working Fusio APIs based on a data model. It also includes a simple Angular
  frontend app to CRUD your models. It can be seen as low-code generator to quickly generate CRUD APIs but the
  generated code is clean and can be also used as foundation for your next app.
* [APImon](https://apimon.app/)  
  Simple API monitoring service which helps to monitor your Fusio installation. It is optimized for Fusio, but it can be
  also used for different API endpoints. APImon invokes your endpoints in specific intervals and notifies you about
  changes. It also includes an uptime page for your users for example s. https://api.apimon.app/status/fusio_marketplace

## 🏷️ Domains

By default, the entire Fusio project can be hosted on a single domain. In this setup:

* Your API is served from the root path (e.g., https://acme.com/).
* Web apps like the developer portal and admin backend are accessible under the `/apps` directory (e.g., https://acme.com/apps/developer).

This setup is quick to get started with and requires no additional configuration.
For production environments, we recommend a subdomain-based structure:

* __api.acme.com__  
  Hosts only the Fusio API. In this setup, you can safely remove the `apps/` folder from the `public/` directory.
* __developer.acme.com__  
  Hosts the **Developer App**, a portal where third-party developers can register, view documentation, and access their credentials. 
* __fusio.acme.com__ (optional)  
  Hosts the **Backend App**, used to manage your Fusio instance. You can also host this on a separate internal domain.

> Note: This is just a suggested setup. You're free to choose any domain or subdomain structure that best fits your infrastructure.

## 📚 Documentation

Please check out our official documentation website where we bundle all documentation resources:

https://docs.fusio-project.org/

## 🤝 Support

### 💬 Get Help

If you have questions or run into issues while using Fusio:

- Open a [discussion](https://github.com/apioo/fusio/discussions) for general questions, feedback, or feature ideas.
- Report bugs or technical problems via the [issue tracker](https://github.com/apioo/fusio/issues).
- Join our [Discord community](https://discord.gg/eMrMgwsc6e) to chat directly with the developers and other users.

If you're a company or freelancer looking for more tailored help, please check out our **consulting** services below.

---

### 📣 Promotion & Media

Are you a blogger, writer, or run a developer-focused publication? We'd love for you to cover Fusio!

Visit the [Media Page](https://www.fusio-project.org/media) to download official icons for use in your articles or videos.

---

### 🧑‍🏫 Consulting & Workshops

For companies or freelancers who want in-depth guidance on using and integrating Fusio:

- We offer **consulting services** to help you evaluate whether Fusio fits your architecture.
- Our **workshops** walk you through key functionality, answer your specific questions, and help identify the best integration approach.

Feel free to [contact us](https://www.fusio-project.org/contact) for more details.

---

### 💖 Support Fusio

If Fusio helps you build APIs faster or adds value to your projects, please consider supporting our work:

- ⭐ Star the project on GitHub
- ☕ [Sponsor via GitHub](https://github.com/sponsors/chriskapp)
- 💬 Spread the word on social media or write about Fusio

Every bit of support helps us continue improving the platform!

---

### 🤝 Project Partners

We’re grateful to our partners who support the Fusio project and share our vision of advancing open API development.

If your company is interested in becoming a partner and being listed here, consider [becoming a sponsor](https://github.com/sponsors/chriskapp).

<a href="https://jb.gg/OpenSource">
 <picture>
   <source media="(prefers-color-scheme: dark)" srcset="https://www.jetbrains.com/company/brand/img/logo_jb_dos_3.svg">
   <source media="(prefers-color-scheme: light)" srcset="https://resources.jetbrains.com/storage/products/company/brand/logos/jetbrains.svg">
   <img alt="JetBrains logo." src="https://resources.jetbrains.com/storage/products/company/brand/logos/jetbrains.svg">
 </picture>
</a>
 readmeEtag: '"e358e8ac947730bb5bc32678d35cd28f723bdf80"' readmeLastModified: Sat, 15 Nov 2025 20:10:34 GMT repositoryId: 39904377 description: Self-Hosted API Management for Builders created: '2015-07-29T16:34:52Z' updated: '2026-02-01T08:39:47Z' language: PHP archived: false stars: 2072 watchers: 55 forks: 237 owner: apioo logo: https://avatars.githubusercontent.com/u/18172950?v=4 license: Apache-2.0 repoEtag: '"4820cfab2b96c7d7e7259e903d47a46a8111239c145dad828cbf706ea90788bd"' repoLastModified: Sun, 01 Feb 2026 08:39:47 GMT foundInMaster: true id: 5458da680012a142d20e79ac150bc4ab - source: - https://openapi.tools/ - openapi3 tags name: yii2-app-api category: - Server - Mock - Server Implementations language: PHP repository: https://github.com/php-openapi/yii2-app-api source_description: > Generate Server side API code with routing, models, data validation and database schema from an OpenAPI description. Based on Yii Framework. v2: false v3: true repositoryMetadata: base64Readme: >- # yii2-app-api

> OpenAPI Spec to API in 3, 2, 1... done!

Yii Framework Application Template for quickly building API-first applications.

Based on [yii2-openapi](https://github.com/php-openapi/yii2-openapi) (code generator) and [php-openapi](https://github.com/php-openapi/php-openapi) (specification reader and validator).

[![Latest Stable Version](https://poser.pugx.org/php-openapi/yii2-app-api/v/stable)](https://packagist.org/packages/php-openapi/yii2-app-api)
[![Total Downloads](https://poser.pugx.org/php-openapi/yii2-app-api/downloads)](https://packagist.org/packages/php-openapi/yii2-app-api)
[![License](https://poser.pugx.org/php-openapi/yii2-app-api/license)](https://packagist.org/packages/php-openapi/yii2-app-api)


## Demo

[![asciicast](https://asciinema.org/a/GSnZFeL0beAwLvbhIn50oVI9w.svg)](https://asciinema.org/a/GSnZFeL0beAwLvbhIn50oVI9w)


## Overview

This application template does not contain any useful code, it only provides the [directory structure](#structure) for your API project. 
All the code is generated from an [OpenAPI](https://www.openapis.org/about) API description file.

When working with this template you follow the [API Design-First Approach](https://apisyouwonthate.com/blog/api-design-first-vs-code-first).
So to get started building an API with this application template you need an API description first.
If you don't have one yet, you might want to check the following resources first: 

- [OpenAPI 3.0 tutorial](https://idratherbewriting.com/learnapidoc/pubapis_openapi_tutorial_overview)
- [OpenAPI Specification Version 3.0.2](http://spec.openapis.org/oas/v3.0.2) 

If you have an OpenAPI description, you can continue with the next steps:

1. [Setup](#setup)
2. [Generate Code](#generate-code)

More Docs and Guides can be found in [/docs](/docs/README.md).

## Setup <span id="setup"></span>

There are two different ways:

- PHP directly on your machine
- Using Docker

### PHP directly

Having PHP and a database server installed on your machine, you can run the following commands:

    composer create-project php-openapi/yii2-app-api my-api
    cd my-api
    cp env.php.dist env.php
    cp config/components-ENV.local.php config/components-dev.local.php

You need to adjust `config/components-dev.local.php` to configure your database connection.

After that, run `make start` and `make stop` to control the PHP webservers for API and backend.

You may use `XDEBUGW=1` to enable xdebug for the webserver, e.g. `make XDEBUGW=1 start`.

You can now continue with [generating code](#generate-code).

### Using Docker

You need [Docker](https://docs.docker.com/install/) and [Docker Compose](https://docs.docker.com/compose/install/) installed.

Create the repository:

    composer create-project php-openapi/yii2-app-api my-api

For the easiest way you need GNU make, then run:

    make start-docker
    
This uses `docker-compose` to start docker containers for the API and the backend including a database.

You can now continue with [generating code](#generate-code).

> Note: If you don't have GNU make, you need to copy the configuration files as in the PHP directly section, then run:
> 
>     cp docker-compose.override.dist.yml docker-compose.override.yml
>     docker-compose up -d
>     docker-compose exec backend-php sh -c 'cd /app && composer install'


## Generate Code <span id="generate-code"></span>

At this point you can start generating your API code to play with it by following the instructions below.
If you'd like to start a real project instead of just playing around you can find more detailed
documentation in [docs/README.md](docs/README.md) specifically the ["Getting Started"-Section](docs/getting-started.md).

### Console

> Tip: If you use Docker, run `make cli` before running any of these commands.

Run `./yii gii/api` to generate your API code. The `--openApiPath` parameter specifies the path to your OpenAPI
spec file. The following example will generate API code for the [OpenAPI petstore example](https://github.com/OAI/OpenAPI-Specification/blob/3.0.2/examples/v3.0/petstore-expanded.yaml).

    ./yii gii/api --openApiPath=https://raw.githubusercontent.com/OAI/OpenAPI-Specification/3.0.2/examples/v3.0/petstore-expanded.yaml

> Note: If the OpenAPI spec file is present locally on the file system, then `openApiPath` must be specified as an absolute path, relative paths will not work correctly.
> On UNIX based OS it must start with `/`. Example: `/home/user/documents/MyProjectOpenAPISpec.yml`

Run `./yii gii/api --help` for a list of configuration options. You may also adjust the configuration in `config/gii-generators.php`.

Then set up the database:

    ./yii migrate/up
    ./yii faker

### Web

To use the web generator, open <http://localhost:8338/gii> and select the `REST API Generator`.

![Gii - REST API Generator](docs/img/gii-generator.png)

Enter the path or URL to the "OpenAPI 3 Spec file", e.g. `https://raw.githubusercontent.com/OAI/OpenAPI-Specification/3.0.2/examples/v3.0/petstore-expanded.yaml`.

Click "Preview":

![Gii - REST API Generator - Generated files](docs/img/gii-generator-files.png)

Click "Generate" to generate API files.

Then set up the database by running the following commands on the command line:

    ./yii migrate/up
    ./yii faker

## Try it

    cd api
    make start

Your API is now available at `http://localhost:8337/`. Try to access an endpoint of your spec via `curl`:

    $ curl http://localhost:8337/pets
    [
        {
            "name": "Eos rerum modi et quaerat voluptatibus.",
            "tag": "Totam in commodi in est nisi nihil aut et."
        },
        {
            "name": "Voluptas quia eos nisi deleniti itaque aspernatur aspernatur.",
            "tag": "Temporibus id culpa dolorem sequi aut."
        },
        {
            "name": "Facere aut similique laboriosam omnis perferendis et.",
            "tag": "Quo harum quo et ea distinctio non quam."
        },
        ...
    ]


## Linting the OpenAPI specification

You can run a linter to check the validity of your OpenAPI specification file:

    make lint-js  # run spectral lint
    make lint-php # run php-openapi validate
    make lint     # run both of the above commands

You can find more information on spectral at <https://www.npmjs.com/package/@stoplight/spectral-cli#-documentation-and-community>.


## Application structure <span id="structure"></span>

This application template consists of 3 application tiers:

- `api`, contains the Yii application for the REST API.
- `console`, contains the Yii application for console commands, cronjobs or queues (`yii` command).
- `backend`, contains the Yii application for a CRUD backend on the API data.

The following list explains the directory structure in more detail:

- `api/` - API application tier
  - `config/` - configuration for API tier
    - `url-rules.php` - custom URL rules
    - `url-rules.rest.php` - URL rules **generated** from OpenAPI Description
    - `components.php` - application components
    - `app.php` - Yii application config (+ overrides for different environments `app-*.php`)
  - `controllers/` - Controller classes **generated** from OpenAPI Description
  - `web/` - public web directory for API application

- `backend/` - Backend application tier
  - `config/` - configuration for Backend tier
    - `components.php` - application components
    - `app.php` - Yii application config (+ overrides for different environments `app-*.php`)
  - `controllers/` - Controller classes
  - `views/` - View files
  - `web/` - public web directory for Backend application

- `common/` - common code files
  - `models/` - model classes **generated** from OpenAPI Description
  - `migrations/` - database migrations **generated** from OpenAPI Description

- `config/` - Common configuration for all application tiers
  - `components.php` - Yii application components (+ overrides for different environments `components-*.php`)
  - `env.php` - Environment setup (YII_DEBUG, YII_ENV, path aliases, composer autoloader)
  - `events.php` - Class wide event listeners
  - `gii-generators.php` - configuration for the Gii code generator (allows to set default values for the ApiGenerator)
  - `params.php` - Configuration for `Yii::$app->params`

- `console/` - Console application tier
  - `config/` - configuration for Console tier
    - `components.php` - application components
    - `app.php` - Yii application config (+ overrides for different environments `app-*.php`)

- `logs/` - log files
- `runtime/` - temporary runtime files


# Development

Below commands are helpful while developing this project:

```bash
./yii gii/api --openApiPath=/app/openapi/schema.yaml --generateMigrations=0  --generateControllers=0 --generateUrls=0

./yii gii/api --openApiPath=/app/openapi/schema.yaml --generateMigrations=1  --generateControllers=0 --generateUrls=0 --generateModels=0 --generateModelFaker=0
```


# Support

**Need help with your API project?**

Professional support, consulting as well as software development services are available:

https://www.cebe.cc/en/contact

Development of this library is sponsored by [cebe.:cloud: "Your Professional Deployment Platform"](https://cebe.cloud).
 readmeEtag: '"d802cae7aa749c8a93fc3f6e3ff805ea35bcbaea"' readmeLastModified: Thu, 06 Jun 2024 12:21:10 GMT repositoryId: 175403734 description: OpenAPI Spec to API in 3, 2, 1... done! created: '2019-03-13T11:03:55Z' updated: '2025-11-09T08:25:24Z' language: PHP archived: false stars: 108 watchers: 6 forks: 10 owner: php-openapi logo: https://avatars.githubusercontent.com/u/71378648?v=4 license: MIT repoEtag: '"9ca28f901fe8c3cefa7a42d2731062a27523da0c06fd92cd1112a54ad15e953f"' repoLastModified: Sun, 09 Nov 2025 08:25:24 GMT foundInMaster: true id: 97701731f0fe680b9faea9103bf2242e oldLocations: - https://github.com/cebe/yii2-app-api - source: https://openapi.tools/ name: Beeceptor category: Mock language: SaaS link: https://beeceptor.com/ source_description: >- 🐝 An HTTP interceptor and rule-based mocking service for REST APIs. No coding required to create a mock endpoint. No sign-up required. v2: false v3: true id: 3add01284466aef304a66bf0c91b6717 foundInMaster: true - source: https://openapi.tools/ name: '@eropple/nestjs-openapi3' category: Server language: Node.js link: https://github.com/eropple/nestjs-openapi3 repository: https://github.com/eropple/nestjs-openapi3 source_description: > Integrates tightly with a NestJS application to infers complex descriptions and expresses them in its generated OpenAPI document. It then presents that document via ReDoc, and validates inputs for conformance to spec. v2: false v3: true repositoryMetadata: base64Readme: >- # `@eropple/nestjs-openapi3` #
[![npm version](https://badge.fury.io/js/%40eropple%2Fnestjs-openapi3.svg)](https://badge.fury.io/js/%40eropple%2Fnestjs-openapi3)

`@eropple/nestjs-openapi3` is a library for [NestJS]() to generate [OpenAPI 3.x]() documents from your API specification. It attempts to be more integrated with the flow of your application than [@nestjs/swagger]() and to push you towards building clean, well-separated APIs along the way.

## Release History ##
### `0.4.4` ###
- Added `OAS.propPrefabs` that includes some helpful presets for dealing with UUIDs, dates, etc. in proper formatting. Some of these, like dates, are automatically handled _if it's a `Date`_, but if you, say, have `Date | null`, it becomes a pain. Hence things like `OAS.propPrefabs.uuid` and `OAS.propPrefabs.date`.
- Adding the JSON schema text directly to the `window` object for RapiDoc; if you manage to send a bad JSON file for one reason or another, you can at least go `window.specText` and find out what's up.
- Added `example` and `description` to prop arguments.

### `0.4.3` ###
- Added the `validationFailedResponse` option to `OpenapiModule.attach`. This allows you to customize the JSON output from the validation interceptor so you can return predictably-shaped errors. It won't, however, set a `400` default in `defaultResponses`; that's your job. (Also make sure your error filter returns the same shape for a 400!)

### `0.4.2` ###
- Left a logging statement in, 'cause I'm a monster. My apologies to your stdout.

### `0.4.1` ###
- Added `defaultResponses` as an option when creating a document, to allow for standard error types to be returned from your APIs. This does _not_ ensure that these objects are created; you should write an exception filter to do that--but since you can just pass a model type as your response type here, it's pretty easy to just create the object and write it out to your response.
- Retracted ReDoc support; after investigating them more completely I discovered that they intend to keep on-page client functionality as a paid product and I have no interest in encouraging that.
- Added RapiDoc 5.3.0 support. `apiDocs:` now takes `'swagger' | 'rapidoc' | null`. This is self-hosted, with no dependency on a remote resource for its script, so you can use it in environments without Internet access. **`rapidoc` shall become the new default in or before version `1.0.0`.**
- Fixed another bug related to NestJS metadata parsing. (Pulled `0.4.0` after discovery; it was a tough one.)

### <strike>`0.4.0`</strike> ###
- Added ReDoc support. There is a new config option, `apiDocs: 'swagger' | 'redoc' | null`; it defaults to `swagger`. If null is passed, APIs will not be served; this is in addition to `skipApiServing: true`. **This will change before `1.0.0`:** the default will become `redoc` and `skipApiServing: true` will be removed.

### `0.3.0` ###
- Now serving your docs with Swagger UI, located at `/api-docs`. Pass `skipApiServing: true` to `OpenapiModule#attach` to disable it.
- Added `Cookie` parameter decorator. If you already have something like `cookie-parser` set up, this library will use the cookies that it parsed; otherwise, it'll parse and create `req.cookies` with `cookie`.

### `0.2.1` ###
- Fixed some lingering errors in the way endpoint metadata was being set in some cases.
- Fixed a bunch of TSLint errors I'd forgotten to take care of. Next TODO will be setting up Husky.

### `0.2.0` ###
- You can now pass 1-tuple arrays as `content` and other places where schemas are expected; they will be parsed as an array of whatever type or schema is in that array.
- `allOf`, `anyOf`, `oneOf`, and `items` all act as you'd expect--that is, [do-what-I-mean schema conversion](https://github.com/eropple/nestjs-openapi3/#schemalike-do-what-i-mean-schema-objects) now respect functions, arrays, etc. and parse them correctly.

### `0.1.1` ###
- Fixed one of the wildest bugs in recent memory. The long and the short of it is that TypeScript decorators on an unset property render that property as being non-writable. A real solution is probably in the TypeScript folks's court, but in the interim this library now generates a descriptor for a property when applicable to avoid making it unwritable.

### `0.1.0` ###
- Initial release.

## Installation ##
Pop into your NestJS project and `npm install @eropple/nestjs-openapi3` or `yarn add @eropple/nestjs-openapi3`.

You'll need reasonably up-to-date versions of `@nestjs/common`, `@nestjs/core`, `reflect-metadata`, and `rxjs`. **NestJS before version 6.5 is unsupported and shall not be supported.**

To serve your OpenAPI document as an explorable page with "try it out" functionality, you'll also need `swagger-ui-express` (when `apiDocs: 'swagger'`, the default, is set) or `rapidoc` (when `apiDocs: 'rapidoc'` is set). If you don't want to serve this, pass `apiDocs: null`.

## Usage ##
First and foremost, you should be aware that `@eropple/nestjs-openapi3` is a heavily integrated library. This is going to touch a lot of your codebase. If you're new to OpenAPI, you should investigate [the OpenAPI 3.x spec](). If you've used Swagger, whether through `@nestjs/swagger` or another avenue, you should read up on [what's new in OpenAPI 3.x]().

### Attaching `OpenapiModule` ###
You'll need to attach the module to your application. We have to do it this way, rather than as a normal module dependency on `AppModule` or similar, because we need to get access to the completely populated dependency injection container.

```ts
// It's good practice to make this a separate method; you can then
// do handy stuff like generate your OpenAPI document without having
// to start a web server. Take a look at our example app for more
// information.
async function buildOpenapiDocument(app: INestApplication) {
  return OpenapiModule.createDocument(
    {
      app,
      // You can pass `console` here if you're lame, but you
      // should be using Bunyan. Check out `@eropple/nestjs-bunyan`!
      baseLogger: openapiLogger,
    },
    (b) => {
      b.addTitle(`${APPLICATION_NAME} API`);
      b.addVersion(PACKAGE_VERSION);
      b.addServer({
        url: API_URL,
      });
      b.addSecurityScheme('token', {
        name: 'Authorization',
        type: 'apiKey',
        in: 'header',
      });
    },
  );
}

async function configureApp(app: INestApplication) {
  // Wherever you set up CORS, helmet, etc. - just do this

  await OpenapiModule.attach(
    {
      app,
      baseLogger: openapiLogger,
      document: await buildOpenapiDocument(app),
    },
  );
}
```

You can now see your API document at `/openapi.json` and your Swagger UI at `/api-docs`.

### Decorators, Decorators, and More Decorators ###
This package provides a number of decorators and it's probably best to look at [Ed's skeleton project]() to get a works-in-anger example of the full API.

One thing I've found to be very useful is to import decorators under a namespace in order to make it clearer when decorators that _aren't_ from NestJS are being used. My standard approach, then, is to import the entire package:

```ts
import * as OAS from '@eropple/nestjs-openapi3';
```

#### Parent- and Endpoint-Level Decorators ####
Some decorators can be applied to controllers (and all endpoints within that controller) and modules (and all endpoints within all controllers). These include:

- `@OAS.Tags()` - tags operations in the document (often used to more richly define clients)
- `@OAS.Deprecated()` - marks as deprecated
- `@OAS.Ignore()` - skips OpenAPI handling
- `@OAS.Parameter()` - defines a parameter that's used outside of the handler. For example, in [Ed's skeleton project](), it's used to provide `tenantName` in a way that can be passed to `@eropple/nestjs-auth` to determine tenancy, from which a user can be found.
- `@OAS.SecurityScheme()` - adds a security scheme, with scopes for OAuth2 or OIDC schemes

#### Endpoint-Level Decorators ####
In addition to the above, some decorators apply explicitly to the endpoint handler.

- `@OAS.Get()`, `@OAS.Post()`, `@OAS.Put()`, etc. - replaces NestJS's http method decorators (such as `@Get()`) with decorators that accept OpenAPI operator details.
- `@OAS.Operation()` - the "break glass in case of" option. If you want to write a literal operation for inclusion in your OpenAPI document, use this one. You'll need to use a NestJS HTTP method decorator, such as `@Get()`, to make sure that NestJS can route requests.

#### Argument-Level Decorators ####
These decorators only apply to the arguments in an endpoint hander function.

- `@OAS.Body()` - wraps NestJS's `@Body()` and accepts schemas for request bodies.
- `@OAS.Path()` - wraps NestJS's `@Param()`. The name change is _intentional_ to match up with the `in` field of an OpenAPI parameter object.
- `@OAS.Header()` - wraps NestJS's `@Headers()`, only supporting a single selected header.
- `@OAS.Query()` - wraps NestJS's `@Query()`.

### `SchemaLike`: "Do What I Mean" Schema Objects ###
Most of this library doesn't rely directly on the `O3TS.SchemaObject` fetched out of our dependency, `openapi3-ts`. Instead, we rely on the concept of a `SchemaLike`, which is defined as thus:

```ts
export type SchemaLikeSchemaObject =
  & O3TS.SchemaObject
  & {
    allOf: Array<O3TS.SchemaObject | O3TS.ReferenceObject | Ctor | SchemaLikeSchemaObject>;
    anyOf: Array<O3TS.SchemaObject | O3TS.ReferenceObject | Ctor | SchemaLikeSchemaObject>;
    oneOf: Array<O3TS.SchemaObject | O3TS.ReferenceObject | Ctor | SchemaLikeSchemaObject>;

    items: O3TS.SchemaObject | O3TS.ReferenceObject | Ctor | SchemaLikeSchemaObject;
  };

export type SchemaLike =
  SchemaLikeSchemaObject | O3TS.SchemaObject | O3TS.ReferenceObject | Ctor |
  [SchemaLikeSchemaObject | O3TS.SchemaObject | O3TS.ReferenceObject | Ctor];
```

Anywhere you would normally write a [Schema Object](), you can pass any of the above:

- A schema object (`{ type: 'string' }`, etc.) -- but there's a twist: any of `allOf`, `anyOf`, or `oneOf` can be a `SchemaLike`.
- A reference object to a schema declared elsewhere in the API or explicitly against the `OpenApiBuilder` class during the attachment of the module.
- A 1-tuple array with any of the above, which will yield `{ type: 'array', items: YourSubSchemaHere }`.
- A constructor for a class decorated with `@OAS.Model()`, which will yield as a schema entered into `#/components/schemas` and used as a reference (`{ $ref: '#/components/schemas/YourClassName' }`).

### Fetching the OpenAPI Document ###
The OpenAPI specification strongly suggests that the OpenAPI JSON document should be served at `/openapi.json`. We follow that suggestion.

Your API docs are served at `/api-docs`. It's configurable between `rapidoc` and `swagger` (the default, though it will stop being the default in or before version `1.0.0`).

### Validation and Coercion ###
One important difference between `@nestjs/swagger` and `@eropple/nestjs-openapi3` is that this library does type conversion to match your desired schema. By default, in NestJS, the following HTTP handler doesn't do what you'd expect:

```ts
@Controller('foo')
export class FooController {
  @Get()
  doSomething(@Query('v') v: number) {
    console.log(typeof(v)); // This will return `string`, not `number`!
  }
}
```

This stinks, so `@eropple/nestjs-openapi3` tries to do one better. Since OpenAPI schemas are just JSON schemas, we use [ajv]() to validate incoming parameters and request bodies. ajv also supports type coercion, which we've enabled. So instead, you've got something more like this:

```ts
@Controller('foo')
export class FooController {
  @OAS.Get()
  doSomething(@OAS.Query('v') v: number) {
    console.log(typeof(v)); // This will return `number`.
  }
}
```

ajv's type coercion also applies to objects in request bodies.

There are some sharp edges to ajv type coercion. Most notably, if a client passes `null` when a string is expected, it will be coerced to the empty string (`""`). You might want to take a look at the [ajv coercion chart]() to avoid any footguns.

### Upgrading from `@nestjs/swagger` ###
Discussing this in-depth is a big ol' **TODO** item. I've converted projects using Swagger 2.0 to using this library; if you have a basic understanding of a Swagger document and its components, porting is pretty easy. Some notes (please feel free to add more to this list):

- NestJS's built-in routing decorators, like `@Get()` and `@Post()`, can be replaced with `@OAS.Get()` and `@OAS.Post()`. You can specify operation information as an argument to this decorator.
- `@Param()` is replaced with `@OAS.Path()`
- `@Headers()` is still usable to fetch all headers from a request, but `@Headers('specific-header')` is replaced with `@OAS.Header()`.
- `@Query()` is replaced with `@OAS.Query()`.
- `@Body()` is replaced with `@OAS.Body()`.
- Burn all that `@ApiExplicitParam()` stuff with cleansing fire.
- Add `@OAS.Model()` or `@OAS.ModelRaw()` to all your request bodies or responses. (`@OAS.ModelRaw()` is useful for turning off model introspection if you've got an object that hinges on discriminators or other advanced)
- Replace `@ApiModelProperty()` with `@OAS.Prop()` or, if you're writing something that's hard to describe, `@OAS.PropRaw()`.

## What This Library Doesn't (And Probably Won't) Do ##
- [Discriminators]() are not supported and probably never will. I think they're a bad idea and nearly impossible to model in a generic way. If you've got a brilliant PR suggestion, I'd be willing to look at it, but seeing as how it makes a lot of things break (for example, we no longer can use [ajv]() to validate schemas), I'm skeptical.

## Avenues for Contribution ##
Look for `TODO`s in the codebase; they're usually good contribution opportunities!

- Fastify support. A lot of `OpenapiValidationInterceptor` expects to be using Express. Probably not a huge challenge, though.
- Add optional Swagger UI functionality, similar to `@nestjs/swagger`.
- Tests in isolation. I'm not clear on how to adequately unit test against functionality hanging directly off of NestJS; right now this library (along with other highly integrated libraries of mine, like nestjs-auth) relies on integration tests in example projects.
- Only the simplest parameter styles are currently supported: `simple` (the only one for headers, though paths have other options that we don't support) and `form` for query parameters. If somebody out there needs more complex styles, you should
- Cleaner types around some of the API. For example, right now you must use one of `content` or `multiContent` for request bodies, but there's no type checking to assert that one _must_ be used. Similarly, operation responses are a little kludgy. Probably an easy fix for somebody!
- Deeply nested SchemaLikes (see above) may not be being handled correctly in all cases.
- [Example functionality]() is entirely absent. Squaring it with simplified flavors of content objects is hard and I don't use it; pull requests welcome.
- There are a couple of odd behaviors resulting from using [ajv]() to validate client parameters/request bodies against our OpenAPI 3 schemas. In particular, when dealing with a string field, ajv coerces `null` to the empty string. I think this should fail hard instead, but not enough to not use ajv!
- Maybe make `defaultResponses` a little smarter. Say you're using the module's `validationFailedResponse` builder--it doesn't make sense for a `400` to be on all endpoints, even ones where there are no parameters or headers. On the other hand, it _might_ make sense, because it means a user can throw a code 400 `HttpException` even when they're dealing with more intricate details of a bad request. Open to discussion.

[NestJS]: https://nestjs.com/
[Ed's skeleton project]: https://github.com/eropple/nest-and-next-skeleton
[OpenAPI 3.x]: https://swagger.io/docs/specification/about/
[@nestjs/swagger]: https://docs.nestjs.com/recipes/swagger
[Schema Object]: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#schemaObject
[the OpenAPI 3.x spec]: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md
[what's new in OpenAPI 3.x]: https://swagger.io/blog/news/whats-new-in-openapi-3-0/
[Example functionality]: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#exampleObject
[Discriminators]: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#discriminatorObject
[ajv]: https://github.com/epoberezkin/ajv
[ajv coercion chart]: https://ajv.js.org/coercion.html
 readmeEtag: '"b7f5fb24078604d03f75aeed32db07e8b1fb568a"' readmeLastModified: Tue, 19 Nov 2019 20:15:40 GMT repositoryId: 210099348 description: OpenAPI 3.x document generation and serving for NestJS. created: '2019-09-22T06:09:28Z' updated: '2026-01-24T02:29:32Z' language: TypeScript archived: true stars: 25 watchers: 2 forks: 1 owner: eropple logo: https://avatars.githubusercontent.com/u/109262?v=4 repoEtag: '"526a8ca8da7b8bb8e942c0ad72d7e88d950aa4cc70fbb2e468bcf4f98a6f5e1f"' repoLastModified: Sat, 24 Jan 2026 02:29:32 GMT foundInMaster: true id: 45bed8813d1c030edf5f60d19fc7791d - source: https://openapi.tools/ name: '@nestjs/swagger' category: Server language: Node.js link: https://docs.nestjs.com/recipes/swagger repository: https://github.com/nestjs/swagger source_description: > Official OpenAPI (Swagger) module for NestJS. Use decorators to define OpenAPI endpoint documentation, parameters and return types. Integrates tightly with a NestJS application. Ships with Swagger UI and serves OpenAPI v3 spec. v2: false v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+DQogIDxhIGhyZWY9Imh0dHA6Ly9uZXN0anMuY29tLyIgdGFyZ2V0PSJibGFuayI+PGltZyBzcmM9Imh0dHBzOi8vbmVzdGpzLmNvbS9pbWcvbG9nby1zbWFsbC5zdmciIHdpZHRoPSIxMjAiIGFsdD0iTmVzdCBMb2dvIiAvPjwvYT4NCjwvcD4NCg0KPHAgYWxpZ249ImNlbnRlciI+QSBwcm9ncmVzc2l2ZSA8YSBocmVmPSJodHRwOi8vbm9kZWpzLm9yZyIgdGFyZ2V0PSJibGFuayI+Tm9kZS5qczwvYT4gZnJhbWV3b3JrIGZvciBidWlsZGluZyBlZmZpY2llbnQgYW5kIHNjYWxhYmxlIHNlcnZlci1zaWRlIGFwcGxpY2F0aW9ucy48L3A+DQo8cCBhbGlnbj0iY2VudGVyIj4NCiAgPGEgaHJlZj0iaHR0cHM6Ly93d3cubnBtanMuY29tL35uZXN0anNjb3JlIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9AbmVzdGpzL2NvcmUuc3ZnIiBhbHQ9Ik5QTSBWZXJzaW9uIiAvPjwvYT4NCiAgPGEgaHJlZj0iaHR0cHM6Ly93d3cubnBtanMuY29tL35uZXN0anNjb3JlIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vbC9AbmVzdGpzL2NvcmUuc3ZnIiBhbHQ9IlBhY2thZ2UgTGljZW5zZSIgLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9+bmVzdGpzY29yZSI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2RtL0BuZXN0anMvY29yZS5zdmciIGFsdD0iTlBNIERvd25sb2FkcyIgLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vY2lyY2xlY2kuY29tL2doL25lc3Rqcy9uZXN0IiB0YXJnZXQ9Il9ibGFuayI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY2lyY2xlY2kvYnVpbGQvZ2l0aHViL25lc3Rqcy9uZXN0L21hc3RlciIgYWx0PSJDaXJjbGVDSSIgLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vZGlzY29yZC5nZy9HN1Fubmh5IiB0YXJnZXQ9Il9ibGFuayI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZGlzY29yZC1vbmxpbmUtYnJpZ2h0Z3JlZW4uc3ZnIiBhbHQ9IkRpc2NvcmQiLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL25lc3QjYmFja2VyIj48aW1nIHNyYz0iaHR0cHM6Ly9vcGVuY29sbGVjdGl2ZS5jb20vbmVzdC9iYWNrZXJzL2JhZGdlLnN2ZyIgYWx0PSJCYWNrZXJzIG9uIE9wZW4gQ29sbGVjdGl2ZSIgLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL25lc3Qjc3BvbnNvciI+PGltZyBzcmM9Imh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL25lc3Qvc3BvbnNvcnMvYmFkZ2Uuc3ZnIiBhbHQ9IlNwb25zb3JzIG9uIE9wZW4gQ29sbGVjdGl2ZSIgLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vcGF5cGFsLm1lL2thbWlsbXlzbGl3aWVjIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9Eb25hdGUtUGF5UGFsLWRjM2Q1My5zdmciLz48L2E+DQogIDxhIGhyZWY9Imh0dHBzOi8vdHdpdHRlci5jb20vbmVzdGZyYW1ld29yayI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdHdpdHRlci9mb2xsb3cvbmVzdGZyYW1ld29yay5zdmc/c3R5bGU9c29jaWFsJmxhYmVsPUZvbGxvdyI+PC9hPg0KPC9wPg0KICA8IS0tWyFbQmFja2VycyBvbiBPcGVuIENvbGxlY3RpdmVdKGh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL25lc3QvYmFja2Vycy9iYWRnZS5zdmcpXShodHRwczovL29wZW5jb2xsZWN0aXZlLmNvbS9uZXN0I2JhY2tlcikNCiAgWyFbU3BvbnNvcnMgb24gT3BlbiBDb2xsZWN0aXZlXShodHRwczovL29wZW5jb2xsZWN0aXZlLmNvbS9uZXN0L3Nwb25zb3JzL2JhZGdlLnN2ZyldKGh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL25lc3Qjc3BvbnNvciktLT4NCg0KIyMgRGVzY3JpcHRpb24NCg0KW09wZW5BUEkgKFN3YWdnZXIpXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKSBtb2R1bGUgZm9yIFtOZXN0XShodHRwczovL2dpdGh1Yi5jb20vbmVzdGpzL25lc3QpLg0KDQojIyBJbnN0YWxsYXRpb24NCg0KYGBgYmFzaA0KJCBucG0gaSAtLXNhdmUgQG5lc3Rqcy9zd2FnZ2VyDQpgYGANCg0KIyMgUXVpY2sgU3RhcnQNCg0KW092ZXJ2aWV3ICYgVHV0b3JpYWxdKGh0dHBzOi8vZG9jcy5uZXN0anMuY29tL29wZW5hcGkvaW50cm9kdWN0aW9uKQ0KDQojIyBTdXBwb3J0DQoNCk5lc3QgaXMgYW4gTUlULWxpY2Vuc2VkIG9wZW4gc291cmNlIHByb2plY3QuIEl0IGNhbiBncm93IHRoYW5rcyB0byB0aGUgc3BvbnNvcnMgYW5kIHN1cHBvcnQgYnkgdGhlIGFtYXppbmcgYmFja2Vycy4gSWYgeW91J2QgbGlrZSB0byBqb2luIHRoZW0sIHBsZWFzZSBbcmVhZCBtb3JlIGhlcmVdKGh0dHBzOi8vZG9jcy5uZXN0anMuY29tL3N1cHBvcnQpLg0KDQojIyBTdGF5IGluIHRvdWNoDQoNCi0gQXV0aG9yIC0gW0thbWlsIE15xZtsaXdpZWNdKGh0dHBzOi8vdHdpdHRlci5jb20va2FtbXlzbGl3aWVjKQ0KLSBXZWJzaXRlIC0gW2h0dHBzOi8vbmVzdGpzLmNvbV0oaHR0cHM6Ly9uZXN0anMuY29tLykNCi0gVHdpdHRlciAtIFtAbmVzdGZyYW1ld29ya10oaHR0cHM6Ly90d2l0dGVyLmNvbS9uZXN0ZnJhbWV3b3JrKQ0KDQojIyBMaWNlbnNlDQoNCk5lc3QgaXMgW01JVCBsaWNlbnNlZF0oTElDRU5TRSkuDQo= readmeEtag: '"5df0fd0107b57e36136c4d3316d49a93eeeb4b6f"' readmeLastModified: Tue, 04 Mar 2025 03:23:08 GMT repositoryId: 105440153 description: 'OpenAPI (Swagger) module for Nest framework (node.js) :earth_americas:' created: '2017-10-01T12:20:24Z' updated: '2026-02-05T22:15:58Z' language: TypeScript archived: false stars: 1853 watchers: 26 forks: 526 owner: nestjs logo: https://avatars.githubusercontent.com/u/28507035?v=4 license: MIT repoEtag: '"3765f998ee5252438ae27d5d092c4e3f50ad0a39b7d38918c2601e545e2152c5"' repoLastModified: Thu, 05 Feb 2026 22:15:58 GMT foundInMaster: true id: 7071babd9853fc901c5decfa9b465b67 - source: - https://openapi.tools/ - openapi3 tags name: Falcon Heavy category: - Server - Mock - Parsers language: Python link: https://github.com/NotJustAToy/falcon-heavy repository: https://github.com/notjustatoy/falcon-heavy source_description: > The framework for building app backends and microservices via the API design-first workflow. v3: true repositoryMetadata: base64Readme: >- KioqKioqKioqKioqCkZhbGNvbiBIZWF2eQoqKioqKioqKioqKioKCgpBYm91dAojIyMjIwoKVGhlIGZyYW1ld29yayBmb3IgYnVpbGRpbmcgYXBwIGJhY2tlbmRzIGFuZCBtaWNyb3NlcnZpY2VzIGJ5IHNwZWNpZmljYXRpb24tZmlyc3QgQVBJIGRlc2lnbiBhcHByb2FjaCBiYXNlZCBvbiB0aGUgYE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAzIDxodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbj5gX18uCgpGYWxjb24gSGVhdnkgY29udmVydHMgYW5kIHZhbGlkYXRlcyByZXF1ZXN0cyBhbmQgcmVuZGVycyByZXNwb25zZXMgY29ycmVzcG9uZGVkIHNwZWNpZmljYXRpb24uIEl0IGNhbiBiZSB1c2VkIHdpdGggYERqYW5nbyA8aHR0cHM6Ly93d3cuZGphbmdvcHJvamVjdC5jb20vPmBfXywgYEZhbGNvbiA8aHR0cHM6Ly9mYWxjb25mcmFtZXdvcmsub3JnLz5gX18gYW5kIGBGbGFzayA8aHR0cHM6Ly9wYWxsZXRzcHJvamVjdHMuY29tL3AvZmxhc2svPmBfXyB3ZWIgZnJhbWV3b3Jrcy4KCkluc3RhbGxhdGlvbgojIyMjIyMjIyMjIyMKClJlY29tbWVuZGVkIHdheSAodmlhIHBpcCk6CgouLiBjb2RlOjogYmFzaAoKICAgICQgcGlwIGluc3RhbGwgZmFsY29uLWhlYXZ5CgpVc2FnZQojIyMjIwoKMS4gSW1wbGVtZW50IGFsbCBhYnN0cmFjdCBtZXRob2RzIGZyb20gYSBjb3JyZXNwb25kaW5nIGRlY29yYXRvciBjbGFzcy4KMi4gU2V0IHVwIHJvdXRpbmcgYmFzZWQgb24gYSBzcGVjaWZpY2F0aW9uLgozLiBEZWNvcmF0ZSB2aWV3cyB3aXRoIHlvdXIgZGVjb3JhdG9yLgoKTGltaXRhdGlvbnMKIyMjIyMjIyMjIyMKCiogWE1MIGlzIG5vdCBzdXBwb3J0ZWQuCiogQ2FuJ3QgdXNlIHJlc2VydmVkIGNoYXJhY3RlcnMgaW4gcGF0aCBhbmQgcXVlcnkgcGFyYW1ldGVycy4KKiBSZWN1cnNpdmUgZGVwZW5kZW5jaWVzIGRldGVjdGlvbiBub3QgaW1wbGVtZW50ZWQuCgpMaWNlbnNlCiMjIyMjIyMKCkNvcHlyaWdodCAyMDE5LTIwMjAgTm90IEp1c3QgQSBUb3kgQ29ycC4KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCgpVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCmRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCldJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgpTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCmxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgo= readmeEtag: '"d8e2ecbcd8419e851febf59ce35af5d437dd822e"' readmeLastModified: Mon, 02 Mar 2020 08:05:48 GMT repositoryId: 226058346 description: >- The framework for building app backends and microservices by specification-first API design approach based on the OpenAPI Specification 3 created: '2019-12-05T08:57:06Z' updated: '2025-11-09T08:30:24Z' language: Python archived: false stars: 25 watchers: 3 forks: 1 owner: NotJustAToy logo: https://avatars.githubusercontent.com/u/58468738?v=4 license: Apache-2.0 repoEtag: '"844dfdcc7d9395ce0c64484553dd1cb8e66a4a50b089127f2e64bbfed5165fae"' repoLastModified: Sun, 09 Nov 2025 08:30:24 GMT foundInMaster: true id: fec1361e049c40cc2fb370a08b153cb0 - source: - https://openapi.tools/ - openapi3 tags name: Dredd category: Testing link: https://dredd.io/ repository: https://github.com/apiaryio/dredd language: Javascript source_description: >- Language-agnostic command-line tool for validating API description document against backend implementation of the API v2: true v3: true repositoryMetadata: base64Readme: >- IyBEcmVkZCDigJQgSFRUUCBBUEkgVGVzdGluZyBGcmFtZXdvcmsKClshW25wbSB2ZXJzaW9uXShodHRwczovL2JhZGdlLmZ1cnkuaW8vanMvZHJlZGQuc3ZnKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvZHJlZGQpClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvYXBpYXJ5aW8vZHJlZGQvdHJlZS9tYXN0ZXIuc3ZnP3N0eWxlPXN2ZyldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL2FwaWFyeWlvL2RyZWRkL3RyZWUvbWFzdGVyKQpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vY2kuYXBwdmV5b3IuY29tL2FwaS9wcm9qZWN0cy9zdGF0dXMvbjNpeGZ4aDcycXVzaHlyNC9icmFuY2gvbWFzdGVyP3N2Zz10cnVlKV0oaHR0cHM6Ly9jaS5hcHB2ZXlvci5jb20vcHJvamVjdC9BcGlhcnkvZHJlZGQvYnJhbmNoL21hc3RlcikKWyFbRG9jdW1lbnRhdGlvbiBTdGF0dXNdKGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL2RyZWRkL2JhZGdlLz92ZXJzaW9uPWxhdGVzdCldKGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL2RyZWRkL2J1aWxkcy8pClshW0tub3duIFZ1bG5lcmFiaWxpdGllc10oaHR0cHM6Ly9zbnlrLmlvL3Rlc3QvbnBtL2RyZWRkL2JhZGdlLnN2ZyldKGh0dHBzOi8vc255ay5pby90ZXN0L25wbS9kcmVkZCkKCiFbRHJlZGQgLSBIVFRQIEFQSSBUZXN0aW5nIEZyYW1ld29ya10oZG9jcy9fc3RhdGljL2ltYWdlcy9kcmVkZC5wbmc/cmF3PXRydWUpCgo+ICoqRHJlZGQgaXMgYSBsYW5ndWFnZS1hZ25vc3RpYyBjb21tYW5kLWxpbmUgdG9vbCBmb3IgdmFsaWRhdGluZwo+IEFQSSBkZXNjcmlwdGlvbiBkb2N1bWVudCBhZ2FpbnN0IGJhY2tlbmQgaW1wbGVtZW50YXRpb24gb2YgdGhlIEFQSS4qKgoKLSBbRG9jdW1lbnRhdGlvbl1bXQotIFtDaGFuZ2Vsb2ddW10KLSBbQ29udHJpYnV0b3IncyBHdWlkZWxpbmVzXVtdCgpEcmVkZCByZWFkcyB5b3VyIEFQSSBkZXNjcmlwdGlvbiBhbmQgc3RlcCBieSBzdGVwIHZhbGlkYXRlcyB3aGV0aGVyIHlvdXIgQVBJCmltcGxlbWVudGF0aW9uIHJlcGxpZXMgd2l0aCByZXNwb25zZXMgYXMgdGhleSBhcmUgZGVzY3JpYmVkIGluIHRoZQpkb2N1bWVudGF0aW9uLgoKIyMjIFN1cHBvcnRlZCBBUEkgRGVzY3JpcHRpb24gRm9ybWF0cwoKLSBbQVBJIEJsdWVwcmludF1bXQotIFtPcGVuQVBJIDJdW10gKGZvcm1lcmx5IGtub3duIGFzIFN3YWdnZXIpCi0gW09wZW5BUEkgM11bXSAoW2V4cGVyaW1lbnRhbF0oaHR0cHM6Ly9naXRodWIuY29tL2FwaWFyeWlvL2FwaS1lbGVtZW50cy5qcy9ibG9iL21hc3Rlci9wYWNrYWdlcy9vcGVuYXBpMy1wYXJzZXIvU1RBVFVTLm1kKSwgY29udHJpYnV0aW9ucyB3ZWxjb21lISkKCiMjIyBTdXBwb3J0ZWQgSG9va3MgTGFuZ3VhZ2VzCgpEcmVkZCBzdXBwb3J0cyB3cml0aW5nIFtob29rc10oaHR0cHM6Ly9kcmVkZC5vcmcvZW4vbGF0ZXN0L2hvb2tzLykK4oCUIGEgZ2x1ZSBjb2RlIGZvciBlYWNoIHRlc3Qgc2V0dXAgYW5kIHRlYXJkb3duLiBGb2xsb3dpbmcgbGFuZ3VhZ2VzIGFyZSBzdXBwb3J0ZWQ6CgotIFtHb10oaHR0cHM6Ly9kcmVkZC5vcmcvZW4vbGF0ZXN0L2hvb2tzLWdvLykKLSBbTm9kZS5qcyAoSmF2YVNjcmlwdCldKGh0dHBzOi8vZHJlZGQub3JnL2VuL2xhdGVzdC9ob29rcy1ub2RlanMvKQotIFtQZXJsXShodHRwczovL2RyZWRkLm9yZy9lbi9sYXRlc3QvaG9va3MtcGVybC8pCi0gW1BIUF0oaHR0cHM6Ly9kcmVkZC5vcmcvZW4vbGF0ZXN0L2hvb2tzLXBocC8pCi0gW1B5dGhvbl0oaHR0cHM6Ly9kcmVkZC5vcmcvZW4vbGF0ZXN0L2hvb2tzLXB5dGhvbi8pCi0gW1J1YnldKGh0dHBzOi8vZHJlZGQub3JnL2VuL2xhdGVzdC9ob29rcy1ydWJ5LykKLSBbUnVzdF0oaHR0cHM6Ly9kcmVkZC5vcmcvZW4vbGF0ZXN0L2hvb2tzLXJ1c3QvKQotIERpZG4ndCBmaW5kIHlvdXIgZmF2b3JpdGUgbGFuZ3VhZ2U/IF9bQWRkIGEgbmV3IG9uZSFdKGh0dHBzOi8vZHJlZGQub3JnL2VuL2xhdGVzdC9ob29rcy1uZXctbGFuZ3VhZ2UvKV8KCiMjIyBTdXBwb3J0ZWQgU3lzdGVtcwoKLSBMaW51eCwgbWFjT1MsIFdpbmRvd3MsIC4uLgotIFtUcmF2aXMgQ0ldW10sIFtDaXJjbGVDSV1bXSwgW0plbmtpbnNdW10sIFtBcHBWZXlvcl1bXSwgLi4uCgojIyBJbnN0YWxsYXRpb24KCmBgYAokIG5wbSBpbnN0YWxsIC1nIGRyZWRkCmBgYAoKIyMgUXVpY2sgU3RhcnQKCjEuICBDcmVhdGUgYW4gW0FQSSBCbHVlcHJpbnRdW10gZmlsZSBjYWxsZWQgYGFwaS1kZXNjcmlwdGlvbi5hcGliYC4KICAgIEZvbGxvdyBbdHV0b3JpYWwgYXQgQVBJIEJsdWVwcmludCB3ZWJzaXRlXVthcGkgYmx1ZXByaW50IHR1dG9yaWFsXQogICAgb3IganVzdCB0YWtlIG9uZSBvZiB0aGUgW3JlYWR5LW1hZGUgZXhhbXBsZXNdW2FwaSBibHVlcHJpbnQgZXhhbXBsZXNdLgoyLiAgUnVuIGludGVyYWN0aXZlIGNvbmZpZ3VyYXRpb246CgogICAgYGBgc2hlbGwKICAgICQgZHJlZGQgaW5pdAogICAgYGBgCgozLiAgUnVuIERyZWRkOgoKICAgIGBgYHNoZWxsCiAgICAkIGRyZWRkCiAgICBgYGAKCjQuICBUbyBzZWUgaG93IHRvIHVzZSBhbGwgRHJlZGQncyBmZWF0dXJlcywgYnJvd3NlIHRoZQogICAgW2Z1bGwgZG9jdW1lbnRhdGlvbl1bZG9jdW1lbnRhdGlvbl0uCgojIyBIb3d0b3MsIFR1dG9yaWFscywgQmxvZ3Bvc3RzICgzcmQgcGFydHkpCgotIFtNYWludGVuaXIgw6Agam91ciBzYSBkb2N1bWVudGF0aW9uIGQnQVBJIGF2ZWMgRHJlZGQhXShodHRwczovL2Jsb2cuaXRuZXR3b3JrLmZyL2Jsb2ctcG9zdC8yMDE5LzA1LzA2L2RyZWRkLXBhcnRpZS0xLWVjcml0dXJlLWRvY3VtZW50YXRpb24uaHRtbCkgXzA1LzA2LzIwMTlfCi0gW0RyZWRkIC0gTGFuZ3VhZ2UtYWdub3N0aWMgSFRUUCBBUEkgVGVzdGluZyBUb29sIC0gSW50ZXJ2aWV3IHdpdGggSG9uemEgSmF2b3Jla10oaHR0cHM6Ly9zdXJ2aXZlanMuY29tL2Jsb2cvZHJlZGQtaW50ZXJ2aWV3LykgXzAzLzIyLzIwMTlfCi0gW0xhcmF2ZWwgT3BlbkFQSSAzIERvY3VtZW50YXRpb24gVmVyaWZpY2F0aW9uIFVzaW5nIERyZWRkXShodHRwczovL2NvbW1hbmR6LmlvL3NuaXBwZXRzL2xhcmF2ZWwvbGFyYXZlbC1kcmVkZC1vcGVuYXBpLXYzLykgXzAyLzI0LzIwMTlfCi0gW1Rlc3RpbmcgeW91ciBBUEkgd2l0aCBEcmVkZF0oaHR0cHM6Ly9tZWRpdW0uY29tL21vcC1kZXZlbG9wZXJzL3Rlc3RpbmcteW91ci1hcGktd2l0aC1kcmVkZC1jMDJlNmNhMTUxZjIpIF8wOS8yNy8yMDE4XwotIFtXcml0aW5nIFRlc3RhYmxlIEFQSSBEb2N1bWVudGF0aW9uIFVzaW5nIEFQSUIgYW5kIERyZWRkIChSYWlscyldKGh0dHBzOi8vYmxvZy5yZWJhc2VkLnBsLzIwMTgvMDYvMjkvdGVzdGFibGUtYXBpLWRvY3MuaHRtbCkgXzA2LzI5LzIwMThfCi0gW0Rlc2lnbi1maXJzdCBBUEkgU3BlY2lmaWNhdGlvbiBXb3JrZmxvdyBNYXR1cmVzXShodHRwczovL3BoaWxzdHVyZ2Vvbi51ay9hcGkvMjAxOC8wMy8wMS9hcGktc3BlY2lmaWNhdGlvbi13b3JrZmxvdy1tYXR1cmVzLykgXzAzLzAxLzIwMThfCi0gW1dyaXRpbmcgYW5kIHRlc3RpbmcgQVBJIHNwZWNpZmljYXRpb25zIHdpdGggQVBJIEJsdWVwcmludCwgRHJlZGQgYW5kIEFwaWFyeV0oaHR0cHM6Ly9oYWNrZXJub29uLmNvbS93cml0aW5nLWFuZC10ZXN0aW5nLWFwaS1zcGVjaWZpY2F0aW9ucy13aXRoLWFwaS1ibHVlcHJpbnQtZHJlZWQtYW5kLWFwaWFyeS1kZjEzOGFjY2NlNWEpIF8xMi8wNC8yMDE3XwotIFtUZXN0aW5nIGFuIEFQSSBBZ2FpbnN0IGl0cyBEb2N1bWVudGF0aW9uXShodHRwczovL2Rldi50by9hbGJlcnRvZmR6bS90ZXN0aW5nLWFuLWFwaS1hZ2FpbnN0LWRvY3VtZW50YXRpb24tNmNsKSBfMTEvMjMvMjAxN18KLSBbS2VlcGluZyBEb2N1bWVudGF0aW9uIEhvbmVzdF0oaHR0cHM6Ly9ibG9nLmFwaXN5b3V3b250aGF0ZS5jb20va2VlcGluZy1kb2N1bWVudGF0aW9uLWhvbmVzdC1kOWFiNTM1MWRkZDQpIF8xMS8yMS8yMDE3XwotIFtBcGlhcnkgZGVzaWduZWQgQVBJcyB0ZXN0ZWQgdXNpbmcgRHJlZGRdKGh0dHBzOi8vcmVkdGh1bmRlci5ibG9nLzIwMTcvMDkvMjAvYXBpYXJ5LWRlc2lnbmVkLWFwaXMtdGVzdGVkLXVzaW5nLWRyZWRkLykgXzA5LzIwLzIwMTdfCi0gW0RyZWRkICsgU3dhZ2dlciBmb3IgUkVTVCBBUEkgdGVzdGluZ10oaHR0cHM6Ly9jb2RlYnVyc3QuaW8vZHJlZGQtc3dhZ2dlci1mb3ItcmVzdC1hcGktdGVzdGluZy03MTVkMWFmNWU4YzUpIF8wMS8yNC8yMDE3XwotIFtUZXN0aW5nIFlvdXIgQVBJIERvY3VtZW50YXRpb24gV2l0aCBEcmVkZF0oaHR0cHM6Ly9tYXR0aGV3ZGFseS5jby51ay9ibG9nLzIwMTYvMDgvMDgvdGVzdGluZy15b3VyLWFwaS1kb2N1bWVudGF0aW9uLXdpdGgtZHJlZGQvKSBfMDgvMDgvMjAxNl8KLSBbRFJFREQgQVBJIFRlc3RlciB3b3JrcyB3aXRoIEFQSSBCbHVlcHJpbnRzXShodHRwOi8vd3d3LmZpbmtsYWJzLm9yZy9hcnRpY2xlcy9hcGktYmx1ZXByaW50LWRyZWRkLmh0bWwpIF8wNy8wNS8yMDE2XwotIFtEb2N1bWVudGF0aW9uIGRyaXZlbiBBUEkgRGV2ZWxvcG1lbnQgdXNpbmcgTGFyYXZlbCwgRHJlZGQgYW5kIEFwaWFyeV0oaHR0cHM6Ly9tZWRpdW0uY29tL2ZyaWFuYml6L2FwaS1waHAtcGlsb3QlQzMlQTllLXBhci1sYS1kb2MtM2M5ZWI0ZGFhMmFhKSBfMDYvMjEvMjAxNl8KLSBbRHJlZGQgdjEuMS4wOiBBIEJpdCBEaWZmZXJlbnRdKGh0dHBzOi8vcGhpbHN0dXJnZW9uLnVrL2FwaS8yMDE2LzA2LzIwL2RyZWRkLXYxLTEtMC1hLWJpdC1kaWZmZXJlbnQvKSBfMDYvMjAvMjAxNl8KLSBbRHJlZGQ6IERvIFlvdXIgSFRUUCBBUEkgSnVzdGljZV0oaHR0cHM6Ly9waGlsc3R1cmdlb24udWsvYXBpLzIwMTUvMDEvMjgvZHJlZGQtYXBpLXRlc3RpbmctZG9jdW1lbnRhdGlvbi8pIF8wMS8yOC8yMDE1XwoKW2FwaSBibHVlcHJpbnRdOiBodHRwczovL2FwaWJsdWVwcmludC5vcmcvClthcGkgYmx1ZXByaW50IHR1dG9yaWFsXTogaHR0cHM6Ly9hcGlibHVlcHJpbnQub3JnL2RvY3VtZW50YXRpb24vdHV0b3JpYWwuaHRtbApbYXBpIGJsdWVwcmludCBleGFtcGxlc106IGh0dHBzOi8vZ2l0aHViLmNvbS9hcGlhcnlpby9hcGktYmx1ZXByaW50L3RyZWUvbWFzdGVyL2V4YW1wbGVzCltvcGVuYXBpIDJdOiBodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8yLjAubWQKW29wZW5hcGkgM106IGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kCltkb2N1bWVudGF0aW9uXTogaHR0cHM6Ly9kcmVkZC5vcmcvZW4vbGF0ZXN0LwpbY2hhbmdlbG9nXTogaHR0cHM6Ly9naXRodWIuY29tL2FwaWFyeWlvL2RyZWRkL3JlbGVhc2VzCltjb250cmlidXRvcidzIGd1aWRlbGluZXNdOiBodHRwczovL2RyZWRkLm9yZy9lbi9sYXRlc3QvY29udHJpYnV0aW5nLwpbdHJhdmlzIGNpXTogaHR0cHM6Ly90cmF2aXMtY2kub3JnLwpbY2lyY2xlY2ldOiBodHRwczovL2NpcmNsZWNpLmNvbS8KW2plbmtpbnNdOiBodHRwczovL2plbmtpbnMuaW8vClthcHB2ZXlvcl06IGh0dHBzOi8vd3d3LmFwcHZleW9yLmNvbS8K readmeEtag: '"ea4790fbe4413cc1856328d57208245ecfe7cf46"' readmeLastModified: Wed, 19 Jul 2023 20:05:32 GMT repositoryId: 12590052 description: Language-agnostic HTTP API Testing Tool created: '2013-09-04T12:21:35Z' updated: '2026-02-05T19:44:21Z' language: JavaScript archived: true stars: 4227 watchers: 69 forks: 278 owner: apiaryio logo: https://avatars.githubusercontent.com/u/765943?v=4 license: MIT repoEtag: '"19d2347edef41db86d90cb344242c53075bca4852dcd6b2a45168a58556a4161"' repoLastModified: Thu, 05 Feb 2026 19:44:21 GMT foundInMaster: true id: bb2e4b0879fce0d8706dcebfd2c10a4c - source: https://openapi.tools/ name: django-contract-tester category: Data Validators repository: https://github.com/maticardenas/django-contract-tester language: Python source_description: >- Validate Django REST Framework (DRF) and Django Ninja APIs against their OpenAPI specification. v2: true v3: true v3_1: true id: e7a16a54a9f5280da020e35d32fc151c repositoryMetadata: base64Readme: >- [![PyPI](https://img.shields.io/pypi/v/django-contract-tester.svg)](https://pypi.org/project/django-contract-tester/)
[![Coverage](https://codecov.io/gh/maticardenas/django-contract-tester/graph/badge.svg)](https://app.codecov.io/gh/maticardenas/django-contract-tester)
[![Python versions](https://img.shields.io/badge/Python-3.9%2B-blue)](https://pypi.org/project/django-contract-tester/)
[![Django versions](https://img.shields.io/badge/Django-4.2%2B-blue)](https://pypi.org/project/django-contract-tester/)


# Django Contract Tester

This is a test utility to validate `DRF (Django REST Framework)` and `Django Ninja` test requests & responses against `OpenAPI` versions 2.x and 3.x schemas.

It has built-in support for `OpenAPI`version `2.0`, `3.0.x` and `3.1.x`, both `yaml` or `json` schema files.

## Installation

```shell script
pip install django-contract-tester
```

## Usage

Instantiate one or more instances of `SchemaTester`:

```python
from openapi_tester import SchemaTester

schema_tester = SchemaTester()
```

If you are using either [drf-yasg](https://github.com/axnsan12/drf-yasg)
or [drf-spectacular](https://github.com/tfranzel/drf-spectacular) this will be auto-detected, and the schema will be
loaded by the `SchemaTester` automatically.

If you are using schema files, you will need to pass the file path:

```python
from openapi_tester import SchemaTester

# path should be a string
schema_tester = SchemaTester(schema_file_path="./schemas/publishedSpecs.yaml")
```

Once you've instantiated a tester, you can use it to test responses and request bodies:

```python
from openapi_tester.schema_tester import SchemaTester

schema_tester = SchemaTester()


def test_response_documentation(client):
    response = client.get('api/v1/test/1')
    assert response.status_code == 200
    schema_tester.validate_response(response=response)


def test_request_documentation(client):
    response = client.get('api/v1/test/1')
    assert response.status_code == 200
    schema_tester.validate_request(response=response)
```

If you are using the Django testing framework, you can create a base `APITestCase` that incorporates schema validation:

```python
from rest_framework.response import Response
from rest_framework.test import APITestCase

from openapi_tester.schema_tester import SchemaTester

schema_tester = SchemaTester()


class BaseAPITestCase(APITestCase):
    """ Base test class for api views including schema validation """

    @staticmethod
    def assertResponse(response: Response, **kwargs) -> None:
        """ helper to run validate_response and pass kwargs to it """
        schema_tester.validate_response(response=response, **kwargs)
```

Then use it in a test file:

```python
from shared.testing import BaseAPITestCase


class MyAPITests(BaseAPITestCase):
    def test_some_view(self):
        response = self.client.get("...")
        self.assertResponse(response)
```

## Options

You can pass options either globally, when instantiating a `SchemaTester`, or locally, when
invoking `validate_response`:

```python
from openapi_tester import SchemaTester, is_camel_case
from tests.utils import my_uuid_4_validator

schema_test_with_case_validation = SchemaTester(
    case_tester=is_camel_case,
    ignore_case=["IP"],
    validators=[my_uuid_4_validator]
)

```

Or

```python
from openapi_tester import SchemaTester, is_camel_case
from tests.utils import my_uuid_4_validator

schema_tester = SchemaTester()


def my_test(client):
    response = client.get('api/v1/test/1')
    assert response.status_code == 200
    schema_tester.validate_response(
        response=response,
        case_tester=is_camel_case,
        ignore_case=["IP"],
        validators=[my_uuid_4_validator]
    )
```

### case_tester

The case tester argument takes a callable that is used to validate the key case of both schemas and responses. If
nothing is passed, case validation is skipped.

The library currently has 4 built-in case testers:

- `is_pascal_case`
- `is_snake_case`
- `is_camel_case`
- `is_kebab_case`

You can use one of these, or your own.

### ignore_case

List of keys to ignore when testing key case. This setting only applies when case_tester is not `None`.

### validators

List of custom validators. A validator is a function that receives two parameters: schema_section and data, and returns
either an error message or `None`, e.g.:

```python
from typing import Any, Optional
from uuid import UUID


def my_uuid_4_validator(schema_section: dict, data: Any) -> Optional[str]:
    schema_format = schema_section.get("format")
    if schema_format == "uuid4":
        try:
            result = UUID(data, version=4)
            if not str(result) == str(data):
                return f"Expected uuid4, but received {data}"
        except ValueError:
            return f"Expected uuid4, but received {data}"
    return None
```

### field_key_map

You can pass an optional dictionary that maps custom url parameter names into values, for situations where this cannot be
inferred by the DRF `EndpointEnumerator`. A concrete use case for this option is when
the [django i18n locale prefixes](https://docs.djangoproject.com/en/3.1/topics/i18n/translation/#language-prefix-in-url-patterns).

```python
from openapi_tester import SchemaTester

schema_tester = SchemaTester(field_key_map={
  "language": "en",
})
```

## Schema Validation

When the SchemaTester loads a schema, it parses it using an
[OpenAPI spec validator](https://github.com/p1c2u/openapi-spec-validator). This validates the schema.
In case of issues with the schema itself, the validator will raise the appropriate error.

## Django testing client

The library includes an `OpenAPIClient`, which extends Django REST framework's
[`APIClient` class](https://www.django-rest-framework.org/api-guide/testing/#apiclient).
If you wish to validate each request and response against OpenAPI schema when writing
unit tests - `OpenAPIClient` is what you need!

To use `OpenAPIClient` simply pass `SchemaTester` instance that should be used
to validate requests and responses and then use it like regular Django testing client:

```python
schema_tester = SchemaTester()
client = OpenAPIClient(schema_tester=schema_tester)
response = client.get('/api/v1/tests/123/')
```

To force all developers working on the project to use `OpenAPIClient` simply
override the `client` fixture (when using `pytest` with `pytest-django`):

```python
from pytest_django.lazy_django import skip_if_no_django

from openapi_tester.schema_tester import SchemaTester


@pytest.fixture
def schema_tester():
    return SchemaTester()


@pytest.fixture
def client(schema_tester):
    skip_if_no_django()

    from openapi_tester.clients import OpenAPIClient

    return OpenAPIClient(schema_tester=schema_tester)
```

If you are using plain Django test framework, we suggest to create custom
test case implementation and use it instead of original Django one:

```python
import functools

from django.test.testcases import SimpleTestCase
from openapi_tester.clients import OpenAPIClient
from openapi_tester.schema_tester import SchemaTester

schema_tester = SchemaTester()


class MySimpleTestCase(SimpleTestCase):
    client_class = OpenAPIClient
    # or use `functools.partial` when you want to provide custom
    # ``SchemaTester`` instance:
    # client_class = functools.partial(OpenAPIClient, schema_tester=schema_tester)
```

This will ensure you all newly implemented views will be validated against
the OpenAPI schema.

> It is worth noting that for the case of clients the request validation is only performed for **successful** response scenarios. This is to avoid having the package interfering with your functional negative test case suite.
> You can still use `schema_tester.validate_request` method separately for the negative cases in which you would like to validate the request as well.


### Django Ninja Test Client

In case you are using `Django Ninja` and its corresponding [test client](https://github.com/vitalik/django-ninja/blob/master/ninja/testing/client.py#L159), you can use the `OpenAPINinjaClient`, which extends from it, in the same way as the `OpenAPIClient`:

```python
schema_tester = SchemaTester()
client = OpenAPINinjaClient(
        router_or_app=router,
        schema_tester=schema_tester,
    )
response = client.get('/api/v1/tests/123/')
```

Given that the Django Ninja test client works separately from the django url resolver, you can pass the `path_prefix` argument to the `OpenAPINinjaClient` to specify the prefix of the path that should be used to look into the OpenAPI schema.

```python
client = OpenAPINinjaClient(
        router_or_app=router,
        path_prefix='/api/v1',
        schema_tester=schema_tester,
    )
```

## Configuration

This package supports configuration through dedicated configuration files, allowing you to set validation behavior globally for your project. This allows you to disable/enable specific validations, which can be useful in case you have certain designs that are still in progress, or can't be changed for some specific reason

### Methods

You can configure the library using either:

1. **`.django-contract-tester`**
2. **`pyproject.toml`**

If both files exist, `.django-contract-tester` takes precedence over configuration in `pyproject.toml`.

### Structure

#### .django-contract-tester

Create a `.django-contract-tester` file in your project root with the following INI structure:

```ini
[django-contract-tester]
ignore_case = ID, API, URL

[django-contract-tester:validation]
request = true
response = true
types = true
formats = true
query_parameters = true
request_for_non_successful_responses = false
disabled_types = integer, array
disabled_formats = date-time, email
disabled_constraints = enum, pattern, minLength
```

#### pyproject.toml

Add under `[tool.django-contract-tester]` section:

- **`ignore_case`**: List of keys to ignore when testing key case (only applies when `case_tester` is set)

Under `[tool.django-contract-tester.validation]`, you can control various aspects of validation:

**Master Switches** (enable/disable entire validation categories):
- **`request`** (default: `true`): Enable/disable request body validation
- **`request_for_non_successful_responses`** (default: `false`): Enable request validation for non-successful responses (`4xx`, `5xx`)
- **`response`** (default: `true`): Enable/disable response body validation
- **`types`** (default: `true`): Enable/disable all schema type validations (string, integer, array, etc.)
- **`formats`** (default: `true`): Enable/disable all schema format validations (date-time, uuid, email, etc.)
- **`query_parameters`** (default: `true`): Enable/disable query parameter validation

**Granular Controls** (disable specific types/formats/constraints):
- **`disabled_types`**: List of OpenAPI types to skip validating (e.g., `["integer", "array"]`)
- **`disabled_formats`**: List of OpenAPI formats to skip validating (e.g., `["date-time", "email"]`)
- **`disabled_constraints`**: List of OpenAPI constraint keywords to skip validating (e.g., `["enum", "pattern", "minLength"]`)

### Examples

#### .django-contract-tester

```ini
[django-contract-tester]
# Keys to ignore during case validation
ignore_case = ID, API, URL

[django-contract-tester:validation]
# Master switches
request = true
response = true
types = true
formats = true
query_parameters = true

# Disable validation for specific types
disabled_types = integer

# Disable validation for specific formats
disabled_formats = date-time, email

# Disable specific constraint validations
disabled_constraints = enum, pattern, minLength, maxLength, minimum, maximum, multipleOf
```

#### pyproject.toml

```toml
[tool.django-contract-tester]
# Keys to ignore during case validation
ignore_case = ["ID", "API", "URL"]

[tool.django-contract-tester.validation]
# Master switches
request = true
response = true
types = true
formats = true
query_parameters = true

# Disable validation for specific types
disabled_types = ["integer"]

# Disable validation for specific formats
disabled_formats = ["date-time", "email"]

# Disable specific constraint validations
disabled_constraints = [
    "enum",
    "pattern",
    "minLength",
    "maxLength",
    "minimum",
    "maximum",
    "multipleOf"
]
```

#### Disabling Validation for Endpoints or Paths

You can also exclude specific endpoints or entire API paths from validation by adding them to the `excluded_endpoints` option under the `[tool.django-contract-tester.validation]` section of your `pyproject.toml`, or to the corresponding section in a `.django-contract-tester` INI file. For example, if you have endpoints that are being built, changing rapidly, or not yet present in your OpenAPI specs.

**.django-contract-tester**
```
[django-contract-tester:validation]
excluded_endpoints = GET /api/pets/*, POST /api/orders, /api/health
# Exclude all GET requests to subpaths of /api/pets/, e.g. GET /api/pets/{id}
# Exclude the POST /api/orders endpoint
# Exclude all methods for /api/health
```

**pyproject.toml**
```toml
[tool.django-contract-tester.validation]
excluded_endpoints = [
    "GET /api/pets/*",      # Exclude all GET requests to subpaths of /api/pets/, e.g. "GET /api/pets/{id}"
    "POST /api/orders",     # Exclude the POST /api/orders endpoint
    "/api/health",          # Exclude all methods for /api/health
]
```

## Known Issues

* We are using [prance](https://github.com/jfinkhaeuser/prance) as a schema resolver, and it has some issues with the resolution of (very) complex OpenAPI 2.0 schemas.

## Contributing

Contributions are welcome. Please see the [contributing guide](https://github.com/maticardenas/django-contract-tester/blob/master/CONTRIBUTING.md)
 readmeEtag: '"ff81e0ec50c05b76fa84eff7d316fda59ac0a349"' readmeLastModified: Sun, 25 Jan 2026 16:30:05 GMT repositoryId: 561525667 description: >- Test utility for validating Django APIs against their OpenAPI specification created: '2022-11-03T22:07:32Z' updated: '2026-01-25T16:40:07Z' language: Python archived: false stars: 18 watchers: 0 forks: 4 owner: maticardenas logo: https://avatars.githubusercontent.com/u/12075970?v=4 license: BSD-3-Clause repoEtag: '"94067f78c050fcfa49f7045b9d1df6b06116ecfa29960eed199ab500b47f1ba7"' repoLastModified: Sun, 25 Jan 2026 16:40:07 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: express-openapi-validator category: - Description Validators - Data Validators repository: https://github.com/cdimascio/express-openapi-validator language: JavaScript source_description: 🦋 Auto-validate API requests and responses in ExpressJS. v2: false v3: true repositoryMetadata: base64Readme: >- IyDwn6aLIGV4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IKClshW2J1aWxkIHdvcmtmbG93XShodHRwczovL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IvYWN0aW9ucy93b3JrZmxvd3MvZGVmYXVsdC55bWwvYmFkZ2Uuc3ZnKV0oIykgWyFbXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3Iuc3ZnKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvcikgWyFbXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9kbS9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yP2NvbG9yPWJsdWUpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yKSBbIVtBbGwgQ29udHJpYnV0b3JzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9jb250cmlidXRvcnMvY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IKKV0oI2NvbnRyaWJ1dG9ycykgWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IvYmFkZ2Uuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL2NvdmVyYWxscy5pby9naXRodWIvY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3I/YnJhbmNoPW1hc3RlcikgWyFbQ29kYWN5IEJhZGdlXShodHRwczovL2FwaS5jb2RhY3kuY29tL3Byb2plY3QvYmFkZ2UvR3JhZGUvMTU3MGEwNmY2MDkzNDVkZGIyMzcxMTRiYmQ2Y2VlZDcpXShodHRwczovL3d3dy5jb2RhY3kuY29tL21hbnVhbC9jZGltYXNjaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvcj91dG1fc291cmNlPWdpdGh1Yi5jb20mdXRtX21lZGl1bT1yZWZlcnJhbCZ1dG1fY29udGVudD1jZGltYXNjaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvciZ1dG1fY2FtcGFpZ249QmFkZ2VfR3JhZGUpIFshW10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXR0ZXIvcm9vbS9jZGltYXNjaW8tb3NzL2NvbW11bml0eT9jb2xvcj0lMjNlYjIwNWEpXShodHRwczovL2dpdHRlci5pbS9jZGltYXNjaW8tb3NzL2NvbW11bml0eSkgWyFbR2l0cG9kIFJlYWR5LXRvLUNvZGVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvR2l0cG9kLVJlYWR5LS10by0tQ29kZS1ibHVlP2xvZ289Z2l0cG9kKV0oaHR0cHM6Ly9naXRwb2QuaW8vI2h0dHBzOi8vZ2l0aHViLmNvbS9jZGltYXNjaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvcikgIFshW10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9kb2N1bWVudGF0aW9uLXllcy1pbmZvcm1hdGlvbmFsKV0oaHR0cHM6Ly9jZGltYXNjaW8uZ2l0aHViLmlvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3ItZG9jdW1lbnRhdGlvbi8pIFshW10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2ZyldKCNsaWNlbnNlKQoKKipBbiBPcGVuQXBpIHZhbGlkYXRvciBmb3IgRXhwcmVzc0pTKiogdGhhdCBhdXRvbWF0aWNhbGx5IHZhbGlkYXRlcyAqKkFQSSoqIF8qKnJlcXVlc3RzKipfIGFuZCBfKipyZXNwb25zZXMqKl8gdXNpbmcgYW4gKipPcGVuQVBJIDMqKiBzcGVjaWZpY2F0aW9uLgoKPHAgYWxpZ249ImNlbnRlciI+CjxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IvbWFzdGVyL2Fzc2V0cy9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yLWxvZ28tdjIucG5nIiB3aWR0aD0iNjAwIj4KPC9wPgoKW/CfpotleHByZXNzLW9wZW5hcGktdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IpIGlzIGFuIHVub3BpbmlvbmF0ZWQgbGlicmFyeSB0aGF0IGludGVncmF0ZXMgd2l0aCBuZXcgYW5kIGV4aXN0aW5nIEFQSSBhcHBsaWNhdGlvbnMuIGV4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IgbGV0cyB5b3Ugd3JpdGUgY29kZSB0aGUgd2F5IHlvdSB3YW50OyBpdCBkb2VzIG5vdCBpbXBvc2UgYW55IGNvZGluZyBjb252ZW50aW9uIG9yIHByb2plY3QgbGF5b3V0LiBTaW1wbHksIGluc3RhbGwgdGhlIHZhbGlkYXRvciBvbnRvIHlvdXIgZXhwcmVzcyBhcHAsIHBvaW50IGl0IHRvIHlvdXIgKipPcGVuQVBJIDMuMC54Kiogb3IgKiozLjEueCoqIHNwZWNpZmljYXRpb24sIHRoZW4gZGVmaW5lIGFuZCBpbXBsZW1lbnQgcm91dGVzIHRoZSB3YXkgeW91IHByZWZlci4gU2VlIGFuIFtleGFtcGxlXShodHRwczovL2NkaW1hc2Npby5naXRodWIuaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvci1kb2N1bWVudGF0aW9uL2d1aWRlLXN0YW5kYXJkLykuCgoqKkZlYXR1cmVzOioqCgotIOKclO+4jyByZXF1ZXN0IHZhbGlkYXRpb24KLSDinJTvuI8gcmVzcG9uc2UgdmFsaWRhdGlvbiAoanNvbiBvbmx5KQotIPCfka4gc2VjdXJpdHkgdmFsaWRhdGlvbiAvIGN1c3RvbSBzZWN1cml0eSBmdW5jdGlvbnMKLSDwn5G9IDNyZCBwYXJ0eSAvIGN1c3RvbSBmb3JtYXRzIC8gY3VzdG9tIGRhdGEgc2VyaWFsaXphdGlvbi1kZXNlcmlhbGl6YXRpb24KLSDwn6e1IG9wdGlvbmFsbHkgYXV0by1tYXAgT3BlbkFQSSBlbmRwb2ludHMgdG8gRXhwcmVzcyBoYW5kbGVyIGZ1bmN0aW9ucwotIOKcgu+4jyAqKlwkcmVmKiogc3VwcG9ydDsgc3BsaXQgc3BlY3Mgb3ZlciBtdWx0aXBsZSBmaWxlcwotIPCfjoggZmlsZSB1cGxvYWQKLSDinI/vuI8gT3BlbkFQSSAzLjAueCBhbmQgMy4xLnggc3BlYyBzdXBwb3J0Ci0g4pyoIEV4cHJlc3MgNCBhbmQgNSBzdXBwb3J0CgoKCioqRG9jczoqKgotIPCfk5YgW2RvY3VtZW50YXRpb25dKGh0dHBzOi8vY2RpbWFzY2lvLmdpdGh1Yi5pby9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yLWRvY3VtZW50YXRpb24vKQoKWyFbR2l0SHViIHN0YXJzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9zdGFycy9jZGltYXNjaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvci5zdmc/c3R5bGU9c29jaWFsJmxhYmVsPVN0YXImbWF4QWdlPTI1OTIwMDApXShodHRwczovL0dpdEh1Yi5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3Ivc3RhcmdhemVycy8pIFshW1R3aXR0ZXIgVVJMXShodHRwczovL2ltZy5zaGllbGRzLmlvL3R3aXR0ZXIvdXJsL2h0dHBzL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3Iuc3ZnP3N0eWxlPXNvY2lhbCldKGh0dHBzOi8vdHdpdHRlci5jb20vaW50ZW50L3R3ZWV0P3RleHQ9Q2hlY2slMjBvdXQlMjBleHByZXNzLW9wZW5hcGktdmFsaWRhdG9yJTIwYnklMjAlNDBDYXJtaW5lRGlNYXNjaW8lMjBodHRwcyUzQSUyRiUyRmdpdGh1Yi5jb20lMkZjZGltYXNjaW8lMkZleHByZXNzLW9wZW5hcGktdmFsaWRhdG9yJTIwJUYwJTlGJTkxJThEKQoKW0V4cHJlc3MgNV0oaHR0cHM6Ly9leHByZXNzanMuY29tL2VuLzV4L2FwaS5odG1sKSBzdXBwb3J0IGF2YWlsYWJsZSBpbiBgPj12NS41LjBgIQoKW09BUyAzLjFdKGh0dHBzOi8vZ2l0aHViLmNvbS9jZGltYXNjaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvci9wdWxsLzg4Mikgc3VwcG9ydCBhdmFpbGFibGUgaW4gYD49djUuNC4wYCEKCltOZXN0SlNdKGh0dHBzOi8vZ2l0aHViLmNvbS9jZGltYXNjaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvci90cmVlL21hc3Rlci9leGFtcGxlcy85LW5lc3RqcykKW0tvYV0oaHR0cHM6Ly9naXRodWIuY29tL2NkaW1hc2Npby9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yL3RyZWUvbGVybmEtZmFzdGlmeS9wYWNrYWdlcy9rb2Etb3BlbmFwaS12YWxpZGF0b3IpIGFuZCBbRmFzdGlmeV0oaHR0cHM6Ly9naXRodWIuY29tL2NkaW1hc2Npby9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yL3RyZWUvbGVybmEtZmFzdGlmeS9wYWNrYWdlcy9mYXN0aWZ5LW9wZW5hcGktdmFsaWRhdG9yKSBub3cgYXZhaWxhYmxlISDwn5qACgoKIyMgSW5zdGFsbAoKYGBgc2hlbGwKbnBtIGluc3RhbGwgZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvcgpgYGAKCiMjIFVzYWdlCgoxLiBSZXF1aXJlL2ltcG9ydCB0aGUgb3BlbmFwaSB2YWxpZGF0b3IKCmBgYGphdmFzY3JpcHQKY29uc3QgT3BlbkFwaVZhbGlkYXRvciA9IHJlcXVpcmUoJ2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3InKTsKYGBgCgpvcgoKYGBgamF2YXNjcmlwdAppbXBvcnQgKiBhcyBPcGVuQXBpVmFsaWRhdG9yIGZyb20gJ2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3InOwpgYGAKCjIuIEluc3RhbGwgdGhlIG1pZGRsZXdhcmUKCmBgYGphdmFzY3JpcHQKYXBwLnVzZSgKICBPcGVuQXBpVmFsaWRhdG9yLm1pZGRsZXdhcmUoewogICAgYXBpU3BlYzogJy4vb3BlbmFwaS55YW1sJywKICAgIHZhbGlkYXRlUmVxdWVzdHM6IHRydWUsIC8vIChkZWZhdWx0KQogICAgdmFsaWRhdGVSZXNwb25zZXM6IHRydWUsIC8vIGZhbHNlIGJ5IGRlZmF1bHQKICB9KSwKKTsKYGBgCgozLiBSZWdpc3RlciBhbiBlcnJvciBoYW5kbGVyCgpgYGBqYXZhc2NyaXB0CmFwcC51c2UoKGVyciwgcmVxLCByZXMsIG5leHQpID0+IHsKICAvLyBmb3JtYXQgZXJyb3IKICByZXMuc3RhdHVzKGVyci5zdGF0dXMgfHwgNTAwKS5qc29uKHsKICAgIG1lc3NhZ2U6IGVyci5tZXNzYWdlLAogICAgZXJyb3JzOiBlcnIuZXJyb3JzLAogIH0pOwp9KTsKYGBgCgpfKipJbXBvcnRhbnQ6KiogRW5zdXJlIGV4cHJlc3MgaXMgY29uZmlndXJlZCB3aXRoIGFsbCByZWxldmFudCBib2R5IHBhcnNlcnMuIEJvZHkgcGFyc2VyIG1pZGRsZXdhcmUgZnVuY3Rpb25zIG11c3QgYmUgc3BlY2lmaWVkIHByaW9yIHRvIGFueSB2YWxpZGF0ZWQgcm91dGVzLiBTZWUgYW4gW2V4YW1wbGVdKCNleGFtcGxlLWV4cHJlc3MtYXBpLXNlcnZlcilfLgoKIyMgW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vY2RpbWFzY2lvLmdpdGh1Yi5pby9leHByZXNzLW9wZW5hcGktdmFsaWRhdG9yLWRvY3VtZW50YXRpb24vKQoKU2VlIHRoZSBbZG9jXShodHRwczovL2NkaW1hc2Npby5naXRodWIuaW8vZXhwcmVzcy1vcGVuYXBpLXZhbGlkYXRvci1kb2N1bWVudGF0aW9uLykgZm9yIGNvbXBsZXRlIGRvY3VtZW5hdGlvbgoKX2RlcHJlY2F0ZWRfIFtsZWdhY3kgZG9jXShodHRwczovL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3Ivd2lraSkgCgojIyBMaWNlbnNlCgpbTUlUXShMSUNFTlNFKQoKPGEgaHJlZj0iaHR0cHM6Ly93d3cuYnV5bWVhY29mZmVlLmNvbS9tOTd0QTVjIiB0YXJnZXQ9Il9ibGFuayI+PGltZyBzcmM9Imh0dHBzOi8vYm1jLWNkbi5ueWMzLmRpZ2l0YWxvY2VhbnNwYWNlcy5jb20vQk1DLWJ1dHRvbi1pbWFnZXMvY3VzdG9tX2ltYWdlcy9vcmFuZ2VfaW1nLnBuZyIgYWx0PSJCdXkgTWUgQSBDb2ZmZWUiIHN0eWxlPSJoZWlnaHQ6IGF1dG8gIWltcG9ydGFudDt3aWR0aDogYXV0byAhaW1wb3J0YW50OyIgPjwvYT4K readmeEtag: '"d1af81fdbdc0e30e5e68975942c9de87becd6008"' readmeLastModified: Wed, 14 May 2025 14:44:12 GMT repositoryId: 176581314 description: >- 🦋 Auto-validates api requests, responses, and securities using ExpressJS and an OpenAPI 3.1.x or 3.0.x specification created: '2019-03-19T19:08:05Z' updated: '2026-02-06T01:52:53Z' language: TypeScript archived: false stars: 999 watchers: 18 forks: 231 owner: cdimascio logo: https://avatars.githubusercontent.com/u/4706618?v=4 license: MIT repoEtag: '"a5cee7ede0b239511b29ce9f138af8faa024ac5a847f7503ad794965661a42c8"' repoLastModified: Fri, 06 Feb 2026 01:52:53 GMT foundInMaster: true id: 3a27bc1a71642e0c1c020c41dc66e206 - source: https://openapi.tools/ name: openapi-dev-tool category: - Testing - Documentation - Parsers repository: https://github.com/lyra/openapi-dev-tool language: JavaScript source_description: >- OpenAPI Dev Tool proposes to developers a unique tool to address development and industrialization needs! v2: true v3: true repositoryMetadata: base64Readme: >- # OpenAPI Dev Tool

[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)

<!-- AUTO-GENERATED-CONTENT:START (badgePlugin:src=./scripts/badge-template) -->

[![Build Status](https://img.shields.io/travis/lyra/openapi-dev-tool/develop)](https://app.travis-ci.com/github/lyra/openapi-dev-tool)

<!-- AUTO-GENERATED-CONTENT:END -->

[![npm package](https://img.shields.io/npm/v/@lyra-network/openapi-dev-tool/latest.svg?style=flat-square)](https://www.npmjs.com/package/@lyra-network/openapi-dev-tool)
[![npm downloads](https://img.shields.io/npm/dt/@lyra-network/openapi-dev-tool.svg?style=flat-square)](https://www.npmjs.com/package/@lyra-network/openapi-dev-tool)

## Sponsor

`openapi-dev-tool` is a tool developed by [Lyra Network](https://www.lyra.com/)

<img src="doc-assets/logo_lyra_blue.svg" alt="Lyra" width="48"/>

## What is OpenAPI?

[OpenAPI](https://swagger.io/docs/specification/about/) specification is an API description format for REST APIs where you can describe your entire API, including:

- Available endpoints (/users) and operations on each endpoint (GET /users, POST /users)
- Operations input and output parameters
- Authentication methods
- Contact information, license, terms of use and other information

## Aim

Although using OpenAPI comes with a lot of advantages (documentation and code generation, faster API publishing, teams collaboration promotion...), industrializing it remains complicated due to certain shortcomings. Using a single OpenAPI file for several contexts (public and private usage for example) is not supported by the current release of OpenAPI. Using the external references to design your API from several files to facilitate the maintainability is still required. Painlessly publish your specifications into a software repository to be able to reuse it (to generate client or server code). Etc.
Also, the OpenAPI ecosystem is very large (https://openapi.tools/) with a lot of tools with different quality levels. Consequently, to set up a complete development environment, many tools have to be installed and upgraded regularly.

OpenAPI Dev Tool fixes all these issues by offering to developers a unique tool to address development and industrialization needs!

From `openapi-dev-tool` command-line interface, you can:

- design your OpenAPI specifications for several contexts of uses (without any copy-paste) ;
- write your OpenAPI specifications by using several files (to avoid having a unique large and unmaintainable file) ;
- easily publish your OpenAPI files and reuse them to generate your client or your server ;
- view documentation in real-time ;
- validate your OpenAPI specifications and your examples automatically ;
- etc.

![presentation](doc-assets/pres.gif)

## Standard support

OpenAPI Dev Tool supports this following standards version:

- [Swagger 2.0](https://swagger.io/specification/v2/)
- [Openapi 3.0.x](https://swagger.io/specification/)
- [Openapi 3.1.0](https://spec.openapis.org/oas/v3.1.0)

## Installation

1. Install `Node.JS` (https://nodejs.org/)

1. Install `OpenAPI Dev Tool` tool

   - With npm
     > `npm i -g @lyra-network/openapi-dev-tool`
   - With yarn
     > `yarn global add @lyra-network/openapi-dev-tool`

1. Show usage
   > `openapi-dev-tool help`

## Features

### Serving

#### Overview

> Develop your specifications!

- **Serve one or several OpenAPI files** specified in [configuration file](#configuration-file) and written in YAML or JSON
- **Live reloading** to view dynamically your changes in your browser in real-time
- **Bundle OpenAPI files** to merge all the files before serving. By using [OpenAPI remote reference](https://swagger.io/docs/specification/using-ref/), you can work on several files. It improves the maintainability and avoids having a large OpenAPI file!
- **Several documentation viewers embedded** with the possibility to [override the templates](#templates-override) used:
  - Redoc (http://localhost:3000)
  - Swagger UI (http://localhost:3000/swagger-ui)
- **`x-tags` vendor extension** in `Info` object to be able to [tag OpenAPI files](#tags-usage). OpenAPI doesn't support specifications tags and only supports operations tags (for organization purposes). With the `x-tags` vendor extension and for Redoc viewer, it is now possible to categorize and organize specifications.

For example, declare a private and a public specifications:

![presentation](doc-assets/xtag.png)

- **Validate** your specifications and your examples automatically
- **Static files** can be exposed in addition of your openapi documentation
- **Context uses**: it is possible to serve a same specification in [several contexts](#context-usage) (internal, public, etc.)
- **Viewers up-to-date** you can use the latest releases of viewers directly by declaring it from your `package.json` ([swagger-ui-dist](https://www.npmjs.com/package/swagger-ui-dist) & [redoc](https://www.npmjs.com/package/redoc)). `openapi-dev-tool` will automatically use the viewers defined in your app.

#### Usage

Usage can be displayed by typing the command

> `openapi-dev-tool help serve`

```
openapi-dev-tool serve

  Serve one or serveral OpenAPI specification files in YAML or JSON
  format

Command Options

  -b, --skipBundle               Skips bundle openapi files into one before serving or publishing, default is
                                 false
  -p, --port number              Port to use, default is 3000
  -u, --contextPath string       Context used to expose openapi documentations. Has to be start with '/', default is '/'
  -e, --viewsFolder string       Folder that contains views in EJS to override defaults
                                 - "api.ejs": for API List page
                                 - "redoc.ejs": for Redoc page
                                 - "swagger-ui.ejs": for Swagger UI page
  -s, --staticFolders string[]   Exposes a static folder in addition of openapi documentation, for example, if
                                 you want expose general documentation (migration, API overview,...)
                                 Has to be a map like :
                                 /<path1>:<folder1> where path1 and folder1 have to [a-zA-Z0-9_-]+
                                 openapi-dev-tool will create a path "/path1" where static files of folder
                                 "folder1" will be exposed
  -x, --skipValidation           Skips OpenAPI validation process, default is false

Global Options

  -c, --config string                Configuration file in JSON or YAML format where specifications are defined, default
                                     is config.json
  -f, --filter string                Manage some specifications defined by its name (`api.info.title`) and which match
                                     with this regex filter rather use the whole of enabled specifications declared
                                     into configuration file
  -v, --verbose                      Verbose mode, default is false
  -a, --urlDownloadTemplate string   Rather to use specs from local FS, you can specify remote specs (using 'artifact'
                                     config property) which will be downloaded by using this url. From this url template
                                     '[ARTIFACT_ID]', '[GROUP_ID]' and
                                     '[VERSION]' will be replaced.
  --downloadPoolSize number          Pool size used to download artifacts, default is "Infinity".
```

### Publishing

#### Overview

> Publish your specifications!
> Use the version and title of your specifications to publish them directly in your software repository server!

- **Publish one or several OpenAPI files** specified in [configuration file](#configuration-file) and written in YAML or JSON
- **Bundle OpenAPI files** to merge all the files before publishing. By using (OpenAPI remote reference)[https://swagger.io/docs/specification/using-ref/], you can work on several files. It improves the maintainability and avoid having a large OpenAPI file!
- **Context uses**: it is possible to publish a same specification in [several contexts](#context-usage) (internal, public, etc.)

#### Usage

Usage can be displayed by typing the command

> `openapi-dev-tool help publish`

```
openapi-dev-tool publish

  Publish into a software repository server (like Sonatype Nexus) one or serveral OpenAPI specifications
  files. The repo can be a Maven repository or a npm repository.

Command Options

  -b, --skipBundle               Skips bundle openapi files into one before serving or publishing, default is
                                 false
  -t, --repoType string          Repository server type. maven or npm are possible, default is maven
  -g, --groupId string           GroupId used in repo server, default is com.openapi for maven repo and @myCompany for npm repo
  -s, --repoServer string        Repository server url to store OpenAPI specification files
  --repoSnapshotsServer string   Repository server url to store OpenAPI snapshots specification files. If specified,
                                 --repoServer will be used to store OpenAPI releases specification files
  -u, --repoUser string          Repository server username. Authentication by using user/password
  -p, --repoPassword string      Repository server password. Authentication by using user/password
  --repoToken string             Repository server token. Authentication by using token. For npm repo only
  -x, --skipValidation           Skips OpenAPI validation process, default is false

Global Options

  -c, --config string                Configuration file in JSON or YAML format where specifications are defined, default
                                     is config.json
  -f, --filter string                Manage some specifications defined by its name (`api.info.title`) and which match
                                     with this regex filter rather use the whole of enabled specifications declared
                                     into configuration file
  -v, --verbose                      Verbose mode, default is false
  -a, --urlDownloadTemplate string   Rather to use specs from local FS, you can specify remote specs (using 'artifact'
                                     config property) which will be downloaded by using this url. From this url template
                                     '[ARTIFACT_ID]', '[GROUP_ID]' and
                                     '[VERSION]' will be replaced.
  --downloadPoolSize number          Pool size used to download artifacts, default is "Infinity".
```

### Publishing Locally

#### Usage

> **Warning**
>
> Publishing locally is only possible to publish in local Maven repository and it uses the `mvn` command from PATH.

Usage can be displayed by typing the command

> `openapi-dev-tool help publish-local`

```
openapi-dev-tool publish-local

  Publish into the local Maven repository

Command Options

  -b, --skipBundle               Skips bundle openapi files into one before serving or publishing, default is
                                 false
  -g, --groupId string           GroupId used in repo server, default is com.openapi
  -x, --skipValidation           Skips OpenAPI validation process, default is false

Global Options

  -c, --config string                Configuration file in JSON or YAML format where specifications are defined, default
                                     is config.json
  -f, --filter string                Manage some specifications defined by its name (`api.info.title`) and which match
                                     with this regex filter rather use the whole of enabled specifications declared
                                     into configuration file
  -v, --verbose                      Verbose mode, default is false
  -a, --urlDownloadTemplate string   Rather to use specs from local FS, you can specify remote specs (using 'artifact'
                                     config property) which will be downloaded by using this url. From this url template
                                     '[ARTIFACT_ID]', '[GROUP_ID]' and
                                     '[VERSION]' will be replaced.
  --downloadPoolSize number          Pool size used to download artifacts, default is "Infinity".
```

> **Warning**
>
> `openapi-dev-tool` is going to determinate local repository path automatically from `mvn` command. So, for this to work, `mvn` command has to be available in PATH.

### Merging

#### Overview

> Merge your specification!
> To debug, it can be interesting to get the merge result used by `publish`, `publish-local` and `serve` command.

- **Merge split OpenAPI specification files** i.e. components specific files, paths specific files, info, etc. that can be written in YAML or JSON into a single file (per OpenAPI specification)

#### Usage

Usage can be displayed by typing the command

> `openapi-dev-tool help merge`

```
openapi-dev-tool merge

  Merge split OpenAPI specification (components, paths, info, etc.) into a bundled specification file

Command Options

  -o, --output string            Merged file output directory

Global Options

  -c, --config string                Configuration file in JSON or YAML format where specifications are defined,
                                     default is config.json
  -v, --verbose                      Verbose mode, default is false
  -a, --urlDownloadTemplate string   Rather to use specs from local FS, you can specify remote specs (using
                                     'artifact' config property) which will be downloaded by using this url. From
                                     this url template '[ARTIFACT_ID]', '[GROUP_ID]' and '[VERSION]' will be
                                     replaced.
  --downloadPoolSize number          Pool size used to download artifacts, default is "Infinity".
```

## Configuration file

Configuration file is the main item to execute `openapi-dev-tool`. Written in JSON or YAML format, this configuration file contains the specifications list that have to be served, published or merged.
It also describes additional parameters used by specific commands (as described above).

```yaml
{
    "specs": [                                       // Array of specifications (several specifications can be exposed)
        {                                            // First specification file
            "file": "./petstore.yaml",               // Relative path of the specification main file (or from artifact archive). It has to be an OpenAPI file in YAML or JSON.
            "artifact": "groupId:artifactId:version" // Artifact of spec downloaded rather to be referenced locally
            "vFolders": "myFolder",                  // Virtual folders to organize specs documentation into several folder. vFolders can be a string or an array of strings
            "enabled": true,                         // Enable or not specification file to avoid to serve / publish / merge some specification file (default is true) --> can be an environment variable
            "context": {                             // Object used for template generation (see Template usage chapter below)
              ...
            }
        },
        {                                            // Second specification file
            "file": "./petstore2.yaml",
            ...
        }
    ],
    "html-injector": []                              // To change render of pages of serve command you can provide your own template files but you can also use this HTML injector where each lines will be injected into HTML
}
```

## Advanced usage

### Command options from RC File

It is possible to specify the whole of command options of `publish`, `publish-local`, `serve`, `merge` commands by using a rc file (in JSON) rather to specify these from command-line.
For example:

`cat .openapi-dev-toolrc`

```
{
     "repoServer": "https://myServer/repository/openapi",
     "repoUser": "admin",
     "repoPassword": "admin",
     "groupId": "com.myCompany.openapi"
}
```

### Templates override

By default, the serve feature renders HTML content by parsing EJS templates defined in `openapi-dev-tool` module:

- `api.ejs`: for API List page. This page lists all the available APIs before accessing it with Redoc viewer
- `redoc.ejs`: for Redoc page. This page loads Redoc viewer.
- `swagger-ui.ejs`: for Swagger UI page. This page loads Redoc viewer.

From `viewsFolder` command option of `serve` command, you can specify a folder where your own EJS templates are defined the change the HTML generated.

For example, `api.ejs`:

```html
<!DOCTYPE html>
<html>
  <head>
    <title>My Company</title>
  <head>
  <body>
  <% specs.forEach(function(spec){ %>
    <a href="/redoc?specName=<%=spec.name%>"><%=spec.name%></a>
  <% }); %>
  </body>
</html>
```

For each template, the following objects are loaded in EJS context:

- `api.ejs`
  - `specs`: array of OpenAPI specifications where each `spec` item contains the following attributes:
    - `name`: name of specification
    - `version`: version of specification
    - `description`: description converted in HTML (from Markdown)
    - `url`: url where specs can be viewed
    - `file`: file of specification
    - `context`: current context
    - `tags`: tags loaded by using the `x-tags` vendor extension
- `redoc.ejs`
  - `spec`: object with only `url` attribute
- `swagger-ui.ejs`
  - `specs`: array of OpenAPI specifications where each `spec` item contains the following attributes:
    - `name`: name of specification
    - `url`: url where specs can be viewed

### Context usage

It is possible to serve, publish and merge a same API for several contexts.
For example, for some use-cases a same API should be used in several contexts: from customers (public access) and from internal systems (private access). The API is exactly the same, only the authentication layer could be different. To avoid duplicating your OpenAPI specification, you can use the same API executed in two contexts (public and private).

To do that, you can use EJS template directly in your OpenAPI files to specify different behaviors where `context` specified in Configuration file will be used for rendering. For example for public access, we want to add a security layer which doesn't exist for internal access:

`specs.yaml`

```yaml
openapi: 3.0.0
...

components:
<%if (public) { %>
  securitySchemes:
    basicAuth:
      type: http
      scheme: basic
<% } %>

...

<%if (public) { %>
security:
- basicAuth: []
<% } %>
...
```

This specification could be used in several contexts (public or internal):

`config.json`

```json
{
  "specs": [
    {
      "file": "./specs/specs.yaml",
      "context": {
        "public": false
      }
    },
    {
      "file": "./specs/specs.yaml",
      "context": {
        "public": true
      }
    }
  ]
}
```

### Tags usage

In OpenAPI specifications, it is possible to use a `Tag` object to group operations in several categories. It is great to organize our specifications but it is not usable to tag all the OpenAPI files. For example, to indicate that the current specification is public, internal, etc.

To cover this need, we use an OpenAPI vendor extension `x-tags` in `Info` object. This new field is a string array that contains tags of the current specification.

For example:
`specs.yaml`

```yaml
openapi: 3.0.0

info:
  version: 1.0.0
  x-tags:
    - public
```

These tags are displayed in API pages (before Redoc page).

## License

[Apache 2.0 License](LICENSE)
 readmeEtag: '"4443086610d29be5c6f0b6d2a4f3776fc1bb20f0"' readmeLastModified: Thu, 13 Nov 2025 13:47:51 GMT repositoryId: 310225760 description: >- OpenAPI Dev Tool proposes to developers a unique tool to address development and industrialization needs! created: '2020-11-05T07:50:31Z' updated: '2025-11-13T13:47:56Z' language: JavaScript archived: false stars: 48 watchers: 7 forks: 8 owner: lyra logo: https://avatars.githubusercontent.com/u/572508?v=4 license: Apache-2.0 repoEtag: '"dc1552fba804cc29a7c7cbe49bb8bcb18fbc7a07a0639dfd126fa67e8c1a0415"' repoLastModified: Thu, 13 Nov 2025 13:47:56 GMT foundInMaster: true id: 921cd13437e80edc1d65f659e39df975 v3_1: true - source: https://openapi.tools/ name: portman category: - Testing - Converters link: https://getportman.com/ repository: https://github.com/apideck-libraries/portman language: Node.js source_description: >- Port OpenAPI Spec to Postman Collection, with contract & variation tests included! v2: false v3: true repositoryMetadata: base64Readme: >- ![portman-hero](https://user-images.githubusercontent.com/1112129/125833512-c32359d8-af27-495b-8211-744c504146b2.png)

<p align="center">
  <a href="https://www.npmjs.com/package/@apideck/portman"><img src="https://img.shields.io/npm/v/@apideck/portman.svg" alt="Total Downloads"></a>
  <a href="https://www.npmjs.com/package/@apideck/portman"><img src="https://img.shields.io/npm/dw/@apideck/portman.svg" alt="Latest Stable Version"></a>
</p>

# Portman 👨🏽‍🚀

Port OpenAPI Spec to Postman Collection, with contract & variation tests included!

Portman leverages OpenAPI documents, with all its defined API request/response properties, to power your Postman collection.
Let Portman do all the work and inject contract & variation tests with a minimum of configuration.
Customize the Postman requests & variables with a wide range of options to assign & overwrite variables.

> [!IMPORTANT]  
> **Important Change:** If you are using version 1.28.0 with a custom Postman config file specified by the `--postmanConfigFile` flag, please ensure that the `parametersResolution` option is set to either "Example" or "Schema". The options `requestParametersResolution` and `exampleParametersResolution` are deprecated openapi-to-postman options.

## Why use Portman?

Convert your OpenAPI spec to Postman, generate contract & variation tests, upload the Postman collection & run the tests through Newman.
Include the Portman CLI as part of an automated process for injecting the power of Portman directly into your CI/CD pipeline.

[Read the full blog post](https://blog.apideck.com/announcing-portman)

## Features

With Portman, you can:

- [x] Convert an OpenAPI document to a Postman collection
  - [x] Support for OpenAPI 3.0
  - [x] Support for OpenAPI 3.1
- Extend the Postman collection with capabilities
  - [x] Inject Postman Contract Tests - [learn more](./examples/testsuite-contract-tests/readme.md)
  - [x] Assign collection variables - [learn more](./examples/testsuite-assign-variables/readme.md)
  - [x] Inject Postman Variation Tests - [learn more](./examples/testsuite-variation-tests/readme.md)
  - [x] Inject Postman Integration Tests
  - [x] Inject Postman with Pre-request & Tests scripts on a collection or operation level - [learn more](./examples/testsuite-pre-request-scripts/readme.md)
  - [x] Modify Postman requests - learn more [here](./examples/testsuite-overwrites/readme.md) and [here](./examples/testsuite-assign-overwrite/readme.md)
  - [x] Fuzz Postman requests - [learn more](./examples/testsuite-fuzzing-tests/readme.md)
- [x] Upload the Postman collection to your Postman app - [learn more](#configure-automatic-upload-to-postman-app)
- [x] Test the Postman collection with Newman - [learn more](#run-newman-with-newman-options)
- [x] Split the configuration into multiple files using $ref
- [x] Manage everything in config file for easy local or CI/CD usage - [learn more](#pass-all-cli-options-as-jsonyaml-file)

## Getting started

1. [Install Portman](#installation)
2. Initialize Portman CLI configuration by running: `$ portman --init`

OR

1. [Install Portman](#installation)
2. Copy `.env.example` to `.env` and add environment variables you need available to your collection
3. Copy/rename and customize each of the \_\_\_\_.default.json config files in the root directory to suit your needs
4. Start converting your OpenAPI document to Postman

OR

If you have an existing OpenAPI specification, try running Portman without any special setup to see how it can generate a Postman collection with contract tests with it's default configuration.

1. [Install Portman](#installation)
2. Run portman on your OpenAPI spec, ie: 
- `npx portman -l my-openapi-spec.yaml` 
- (if your spec is hosted use the `-u` parameter, ie:
  - `npx portman -u https://petstore3.swagger.io/api/v3/openapi.json`

This will generate a postman collection that contains a request for every method:endpoint combination defined in your spec, and include a set of "Contract Tests" for each one.  You can learn more about contract tests, and how to examine the generated collection [here](./examples/testsuite-contract-tests/readme.md).

(Running portman with no explicit configuration is the same as running it with [this configuration file](./examples/testsuite-contract-tests/default-portman-config.json))

All configuration options to convert from OpenAPI to Postman can be found in the [openapi-to-postman](https://github.com/postmanlabs/openapi-to-postman/blob/develop/OPTIONS.md) package documentation.
All configuration options to filter flags/tags/methods/operations/... from OpenAPI can be found in the [openapi-format](https://github.com/thim81/openapi-format#openapi-filter-options) package documentation or using the online [openapi-format playground](https://openapi-format-playground.vercel.app/).

## Installation

### Local Installation (recommended)

You can add the Portman CLI to the `node_modules` by using:

```shell
$ npm install --save @apideck/portman
```

or using yarn:

```shell
$ yarn add @apideck/portman
```

Note that this will require you to run the Portman CLI with `npx @apideck/portman -l your-openapi-file.yaml` or, if
you are using an older version of npm, `./node_modules/.bin/portman -l your-openapi-file.yaml`.

### Global Installation

```shell
$ npm install -g @apideck/portman
```

### NPX usage

To execute the CLI without installing it via npm, use the npx method.

```shell
$ npx @apideck/portman -l your-openapi-file.yaml
```

## CLI Usage

```
Usage: -u <url> -l <local> -b <baseUrl> -t <includeTests>

Options:
 --help                     Show help                                                                        [boolean]
 --version                  Show version number                                                              [boolean]
 --url,-u                   URL of OAS to port to Postman collection                                         [string]
 --local, -l                Use local OAS to port to Postman collection                                      [string]
 --baseUrl, -b              Override spec baseUrl to use in Postman                                          [string]
 --output, -o               Write the Postman collection to an output file                                   [string]
 --oaOutput                 Write the (filtered) OpenAPI file to an output file                              [string]
 --runNewman, -n            Run Newman on newly created collection                                           [boolean]
 --newmanRunOptions         JSON stringified object to pass options for configuring Newman                   [string]
 --newmanOptionsFile        Path/URL to Newman options file to pass options for configuring Newman           [string]
 --newmanIterationData, -d  Iteration data to run Newman with newly created collection                       [string]
 --localPostman             Use local Postman collection, skips OpenAPI conversion                           [string]
 --syncPostman              Upload generated collection to Postman (default: false)                          [boolean]
 --syncPostmanCollectionIds Synchronises the IDs of newly created postman collections with those already
                            on Postman, useful when you want to use Postman pull request (default: false)    [boolean]
 --postmanFastSync          Postman sync creates new collection (new UID),instead of update (default: false) [boolean]
 --postmanRefreshCache      Postman sync will refresh all local cached Postman API data (default: false)     [boolean]
 --postmanUid, -p           Postman collection UID to upload with the generated Postman collection           [string]
 --postmanWorkspaceName     Postman Workspace name to target the upload of the generated Postman collection  [string]
 --includeTests, -t         Inject Portman test suite (default: true)                                        [boolean]
 --bundleContractTests      Bundle Portman contract tests in a separate folder in Postman (default: false)   [boolean]
 --portmanConfigFile, -c    Path/URL to Portman settings config file (portman-config.json)                   [string]
 --postmanConfigFile,-s     Path to openapi-to-postman config file (postman-config.json)                     [string]
 --filterFile               Path/URL to openapi-format config file (oas-format-filter.json)                  [string]
 --envFile                  Path to the .env file to inject environment variables                            [string]
 --collectionName           Overwrite OpenAPI title to set the Postman collection name                       [string]
 --cliOptionsFile           Path/URL to Portman CLI options file                                             [string]
 --ignoreCircularRefs       Ignore circular references in OpenAPI spec (default: false)                      [boolean]
 --logAssignVariables       Toggle logging of assigned variables (default: true)                             [boolean]
 --warn/--no-warn           Toggle warnings for missing openApiOperationIds (default: true)                  [boolean]
 --init                     Configure Portman CLI options in an interactive manner                           [string]
 --extraUnknownFormats      Add extra unknown formats to json schema tests                                   [array]
```

### Environment variables as Postman variables

Portman uses `dotenv` to not only access variables for functionality, but you can also add environment variables that you'd like declared within your Postman environment.
Simply prefix any variable name with `PORTMAN_`, and it will be available for use in your Postman collection as the camel-cased equivalent. For example:

```
PORTMAN_CONSUMER_ID=test_user_id
```

will be available in your collection or tests by referencing:

```
{{consumerId}}
```

It is possible to set a spec-specific `.env` file, that lives next to your config files. The path can be passed in via `envFile` cli option.
This is useful if you have Portman managing multiple specs that have unique environment requirements.

By default, Portman will leverage any ENVIRONMENT variable that is defined that starts with `PORTMAN_`.

Another option to set variables is by configuring them as `collectionVariables` in the [globals section](#portman---globals-property) of your Portman configuration.

### CLI Options

###### Initialize Portman CLI configuration

```
portman --init
```

The `init` option will help you to configure the cliConfig options and put the default config, env file in place to kick-start the usage of Portman.

###### Pass in the remotely hosted spec

```
portman -u https://specs.apideck.com/crm.yml
```

###### Overwrite the baseUrl in spec and run Newman

```
portman -u https://specs.apideck.com/crm.yml -b http://localhost:3050 -n true
```

###### Path pass to a local data file for Newman to use for iterations

```
portman -u https://specs.apideck.com/crm.yml -b http://localhost:3050 -n true -d ./tmp/newman/data/crm.json
```

###### Pass the path to a local spec (useful when updating your specs) and output Postman collection locally

```
portman -l ./tmp/specs/crm.yml -o ./tmp/specs/crm.postman.json
```

###### Skip tests and just generate collection

```
portman -l ./tmp/specs/crm.yml -t false
```

###### Filter OpenAPI and generate collection

```
portman -u https://specs.apideck.com/crm.yml --filterFile examples/cli-filtering/oas-format-filter.json
```

For more details, review the [cli-filtering example](https://github.com/apideck-libraries/portman/tree/main/examples/cli-filtering).

###### Add extra forms to Json schema validation

```
portman -l ./tmp/specs/crm.yml -o ./tmp/specs/crm.postman.json --extraUnknownFormats ulid one two
```

This makes the schema validation more lenient, and solves problems with unknown formats

###### Upload newly generated collection to Postman, which will upsert the collection, based on the collection name

```
portman -l ./tmp/specs/crm.yml --syncPostman
```

Upload newly generated collection to Postman using the collection UID to overwrite the existing.

```
portman -l ./tmp/specs/crm.yml --syncPostman -p 9601963a-53ff-4aaa-92a0-2e70a8a2a748
```

When a collection gets large, the Postman API will compare all the requests when updating the collection. This can take some time even result in 5xx errors.
To overcome this, you can use the `--postmanFastSync` option. This option will sync your collection to Postman by using "delete" and "create" operations instead of the "update".

REMARK: Using `--postmanFastSync` will result in a new Postman collection and Postman UID for each sync.

```
portman -l ./tmp/specs/crm.yml --syncPostman --postmanFastSync
```

Portman caches a set of Postman API data to facilitate faster lookups and uploads, preventing unnecessary connecting to the Postman API.
In case you need to reset the cache you simply remove the `.portman.cache.json` file or set the `--postmanRefreshCache` option when running the Postman sync.

```
portman -l ./tmp/specs/crm.yml --syncPostman --postmanRefreshCache
```

###### Pass custom paths for config files

All configuration options to convert from OpenAPI to Postman can be on the [openapi-to-postman](https://github.com/postmanlabs/openapi-to-postman/blob/develop/OPTIONS.md) package documentation.
Portman provides a default openapi-to-postman configuration [postman-config.default.json](postman-config.default.json), which will be used if no custom config `--postmanConfigFile` is passed.

Portman configuration file in JSON format:

```
portman -u https://specs.apideck.com/crm.yml -c ./tmp/crm/portman-config.json -s ./common/postman-config.json
```

Portman configuration file in YAML format:

```
portman -u https://specs.apideck.com/crm.yml -c ./tmp/crm/portman-config.yaml -s ./common/postman-config.json
```

###### Pass all CLI options as JSON/YAML file

All the CLI options can be managed in a separate configuration file and passed along to the portman command. This will
make configuration easier, especially in CI/CD implementations.

Portman CLI options settings in JSON format

```
portman --cliOptionsFile ./examples/cli-options/portman-cli-options.json
```

Portman CLI options settings in YAML format

```
portman --cliOptionsFile ./examples/cli-options/portman-cli-options.yaml
```

All the available Portman CLI options can be used in the config file.
By passing the CLI options as parameter, you can overwrite the defined CLI options defined in the file.

For more details, review the [cli-options example](https://github.com/apideck-libraries/portman/tree/main/examples/cli-options).

###### Run Newman with Newman options

All [Newman configuration options](https://learning.postman.com/docs/running-collections/using-newman-cli/command-line-integration-with-newman/#options) to run Newman can be passed along through Portman.

```
portman -u https://specs.apideck.com/crm.yml -c ./tmp/crm/portman-config.json --runNewman --newmanOptionsFile ./tmp/crm/newman-options.json
```

For more details, review the [cli-options example](https://github.com/apideck-libraries/portman/tree/main/examples/cli-options).

NOTE: Newman is set to ignore redirects to allow for testing redirect response codes. If you are running collections within Postman UI, you'll need to ensure Postman is set to the same, or your redirect tests will fail.

> Postman > Preferences > Automatically follow redirects > OFF

### Output

Without specifying the output location, your generated Postman Collection is written to `./tmp/converted/${specName}.json` if you are manually importing to Postman or need to inspect for debugging.

By using `-o` or `--output` parameter, you can define the location where the Postman collection will be written.

```
portman -l ./tmp/specs/crm.yml -o ./tmp/specs/crm.Postman.json
```

## Portman settings

The Portman settings consist out of multiple parts:

- **version** : which refers to the JSON Portman configuration version.
- **tests** : which refers to the definitions for the generated contract & variance tests.
  - **contractTests** : refers to the options to enabled autogenerated contract tests.
  - **contentTests** : refers to the additional Postman tests that check the content.
  - **variationTests** : refers to the options to define variation tests.
  - **integrationTests** : refers to the options to define integration tests.
  - **extendTests** : refers to the custom additions of manually created Postman tests.
- **assignVariables** : which refers to setting Postman collection variables for easier automation.
- **overwrites** : which refers to the custom additions/modifications of the OpenAPI/Postman request data.
- **operationPreRequestScripts** : which refers to injecting Postman Pre-request Scripts for requests.
- **globals** : which refers to the customization that applies for the whole Postman collection.

### Portman targeting

It is possible to inject Postman tests and pre-register scripts, assign variables and overwrite query params, headers, request body data with values.

To be able to do this very specifically, there are options to define the targets:

- **openApiOperationId (String)** : References to the OpenAPI operationId, example: `leadsAll`
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : References to a combination of the OpenAPI method & path, example: `GET::/crm/leads`

- **excludeForOperations (Array)** : References to OpenAPI operations that will be skipped for targeting. It supports both the `openApiOperationId` and `openApiOperation` format, example: `["leadsAdd", "GET::/crm/leads/{id}"]`

An `openApiOperationId` is an optional property. To offer support for OpenAPI documents that don't have operationIds, we
have added the `openApiOperation` definition, which is the unique combination of the OpenAPI method & path, with a `::`
separator symbol. The targeting option `excludeForOperations` is really useful when using wildcards, to allow exclusions from the wildcard.

This will allow targeting for very specific OpenAPI items.

To facilitate managing the filtering, we have included wildcard options for the `openApiOperation` option, supporting
the methods & path definitions.

REMARK: Be sure to put quotes around the target definition.

- **Strict matching** example: `"openApiOperation": "GET::/crm/leads",`
  This will target only the "GET" method and the specific path "/crm/leads"

- **Method wildcard matching** example: `"openApiOperation": "*::/crm/leads",`
  This will target all methods ('get', 'put', 'post', 'delete', 'options', 'head', 'patch', 'trace') and the specific
  path "/crm/leads"

- **Path wildcard matching** example: `"openApiOperation": "GET::/crm/*"`
  This will target only the "GET" method and any path matching any folder behind the "/crm", like "/crm/companies" and
  "/crm/leads".

- **Method & Path wildcard matching** example: `"openApiOperation": "*::/crm/*",`
  A combination of wildcards for the method and path parts is even possible.

### Portman - `tests` properties

The Portman `tests` is where you would define the tests that would be applicable and automatically generated by Portman, based on the OpenAPI document.
The contract tests are grouped in an array of `contractTests`.

#### contractTests options

- **openApiOperationId (String)** : References to the OpenAPI operationId. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : References to a combination of the OpenAPI method & path (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting, example: `["leadsAdd", "GET::/crm/leads/{id}"]`
- **openApiRequest (String | optional)** : References to the OpenAPI request body content-type (supports wildcards like `application/*`) to use for the contract test. If not defined, the default request content-type from OpenAPI will be used, with a preference for `application/json`.
- **openApiResponse (String | optional)** : References to the OpenAPI response object code or `code::content-type` (supports wildcards like `text/*`) for which a contract test will be inserted. Examples: `"404"`, `"200::text/plain"`, `"200::text/*"`. If not defined, the 1st response object from OpenAPI will be taken as expected response.

  Contract tests are always attached to the main request that Portman generates for an operation. When your OpenAPI document lists multiple request or response content types, only one set will be used for that request. You can control which types are selected with the `openApiRequest` and `openApiResponse` options. To create individual requests for every content type, use the variation testing concept described below.

- **statusSuccess (Boolean)** : Adds the test if the response of the Postman request returned a 2xx
- **statusCode (Boolean, HTTP code)** : Adds the test if the response of the Postman request return a specific status code.
- **responseTime (Boolean)** : Adds the test to verify if the response of the Postman request is returned within a number of ms.
  - **maxMs (number)** : Define the expected number of ms for the `responseTime` check.
- **contentType (Boolean)** : Adds the test if the response header is matching the expected content-type defined in the OpenAPI spec.
- **jsonBody (Boolean)** : Adds the test if the response body is matching the expected content-type defined in the OpenAPI spec.
- **schemaValidation (Boolean)** : Adds the test if the response body is matching the JSON schema defined in the OpenAPI spec. The JSON schema is inserted inline in the Postman test.
  - **additionalProperties (Boolean)** : Extend the expected JSON schema used for the `schemaValidation` by setting all the `additionalProperties`.
- **headersPresent (Boolean)** : Adds the test to verify if the Postman response header has the required header names present, like defined in the OpenAPI spec.

For more details, review the [contract-tests example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-contract-tests) and the [testsuite-contract-content-types example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-contract-content-types) for some more details about targeting request body & response content-types.

#### variationTests options

- **openApiOperationId (String)** : References to the OpenAPI operationId for which a variation will be created. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : References to a combination of the OpenAPI method & path for which a variation will be created. (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting, example: `["leadsAdd", "GET::/crm/leads/{id}"]`
- **openApiResponse (String | optional)** : References to the OpenAPI response object code or `code::content-type`. Supports wildcards for both the code (e.g. `4*` or `*`) and the content type (e.g. `text/*` or `*`). Examples: `"404"`, `"200::text/plain"`, `"2*::application/json"`, `"2*::application/*"`, `"*::*"`. If not defined, the 1st response object from OpenAPI will be taken as expected response. If the configured response code is not defined in the OpenAPI document, Portman will not generate a variation for the targeted operations.
- **openApiRequest (String | optional)** : References to the OpenAPI request body content-type (supports wildcards like `application/*`) for which a variation will be created. If not defined, the default request content-type from OpenAPI will be used.

- **overwrites** : which refers to the custom additions/modifications of the OpenAPI/Postman request data, specifically for the variation.
- **fuzzing** : Fuzz testing sets unexpected values for API requests, to cause unexpected behavior and errors in the API response.
- **tests** : which refers to the definitions for the generated contract & variance tests for the variation.
  - **contractTests** : refers to the options to enabled autogenerated contract tests for the variation.
  - **contentTests** : refers to the additional Postman tests that check the content for the variation.
  - **extendTests** : refers to the custom additions of manual created Postman tests to be included in the variation.
- **assignVariables** : This refers to setting Postman collection variables that are assigned based on variation.

For more details, review the [content-variation example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-variation-tests) and the [content-type variation example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-variation-content-types).

#### integrationTests options

- **name (String)** : As Integration tests will normally contain multiple operations, this is the folder name that will be generated in the Integration Tests folder in your Postman collection.
- **operations (Array)** : Array of operations to be performed

### Portman - `contentTests` properties

Content tests will validate if the response property values will match the expected defined values.
While the Portman `tests` verify the "contract" of the API, the `contentTests` will verify the content of the API.

#### contentTests options

- **openApiOperationId (String)** : References to the OpenAPI operationId. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : References to a combination of the OpenAPI method & path (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting, example: `["leadsAdd", "GET::/crm/leads/{id}"]`

- **responseBodyTests (Array)** : Array of key/value pairs of properties & values in the Postman response body.

  - **key (String)** : The key that will be targeted in the response body to check if it exists. 
    - To look up a key within in array of objects, you can use an array index (example `data.websites[0].url`) or a * wildcard (example: `data.websites[*].url`) which uses the `value` to match an object in an array. 
    - To test a response body that is a single value instead of a JSON object, set this to '.'
  - **value (String)** : The value that will be used to check if the value in the response body property matches.
  - **contains (String)** : The value that will be used to check if the value is present in the value of the response body property.
  - **oneOf (String[],Number[],Boolean[])** : The value that will be used to check one of the values is matching the response body property.
  - **length (Number)** : The number that will be used to check if the value of the response body property (string/array) has a length of the defined number.
  - **minLength (Number)** : The number that will be used to check if the value of the response body property (string/array) has a minimum length of the defined number.
  - **maxLength (Number)** : The number that will be used to check if the value of the response body property (string/array) has a maximum length of the defined number.
  - **notExist (Boolean)** : The inverse of the key check that verify if the key does not exist in the response body.
  - **assert (String)** : A custom Postman assertion to check if the value in the response body property matches with the provided assertion (example: `not.to.be.null`).
  
- **responseHeaderTests (Array)** : Array of key/value pairs of properties & values in the Postman response header.
  - **key (String)** : The header name that will be targeted in the response header to check if it exists.
  - **value (String)** : The value that will be used to check if the value in the response header matches.
  - **contains (String)** : The value that will be used to check if the value is present in the value of the response header.
  - **oneOf (String[],Number[],Boolean[])** : The value that will be used to check one of the values is matching the value of the response header.
  - **length (Number)** : The number that will be used to check if the value of the response header has a length of the defined number of characters.
  - **minLength (Number)** : The number that will be used to check if the value of the response header has a minimum length of the defined number of characters.
  - **maxLength (Number)** : The number that will be used to check if the value of the response header has a maximum length of the defined number of characters.
  - **notExist (Boolean)** : The inverse of the key check that verify if the key does not exist in the response header.
  - **assert (String)** : A custom Postman assertion to check if the value in the response header matches with the provided assertion (example: `not.to.be.null`).

For more details, review the [content-tests example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-content-tests).

### Portman - `extendTests` properties

When you need to add additional tests or overwrite the Portman-generated test, you can use the `extendTests` to define the raw Postman tests.
Anything added in the `tests` array will be added to the Postman test scripts.

#### extendTests options

- **openApiOperationId (String)** : References to the OpenAPI operationId. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : References to a combination of the OpenAPI method & path (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting, example: `["leadsAdd", "GET::/crm/leads/{id}"]`

- **tests (Array)** : Array of additional Postman test scripts.  Values can be the script content or path to the script file (with `file:` prefix).
- **overwrite (Boolean true/false | Default: false)** : Resets all generateTests and overwrites them with the defined tests from
  the `tests` array.
- **append (Boolean true/false | Default: true)** : Place the tests after (append) or before (prepend) all generated tests.

<hr>

### Portman - `assignVariables` properties

The "assignVariables" allows you to set Postman collection variables for easier automation.

#### assignVariables options

- **openApiOperationId (String)** : Reference to the OpenAPI operationId for which the Postman pm.collectionVariables will be set. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, for which the Postman pm.collectionVariables will be set. example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : Reference to the combination of the OpenAPI method & path, for which the Postman pm.collectionVariables will be set. (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting, example: `["leadsAdd", "GET::/crm/leads/{id}"]`

- **collectionVariables (Array)** : Array of key/value pairs to set the Postman collection variables.
  - **responseBodyProp (String)** : The property for which the value will be taken from the response body and set the value as the pm.collectionVariables value. To store the root level, use `.` as key.
  - **responseHeaderProp (String)** : The property for which the value will be taken from the response header and set the value as the pm.collectionVariables value.
  - **requestBodyProp (String)** : The property for which the value will be taken from the request body and set the value as the pm.collectionVariables value.
  - **value (String)** : The defined value that will be set as the pm.collectionVariables value. The value can be generated using template expressions.  For the full list of template expressions, check the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions)
  - **name (string OPTIONAL |  Default: <operationId>.<varProp>)** : The desired name that will be used to as the Postman variable name. If the `name` is not provided, Portman will generate a variable name, using the `<operationId>.<varProp>`. You can pass your own template expressions, to dynamically generate variable names. The template can contain the following dynamic expressions: `<operationId>` results in the OpenAPI operation ID (example `leadsAdd`), `<path>` results in the OpenAPI operation ID (example `/crm/leads`), `<pathRef>` results in the Portman operation (example `POST::/crm/leads_POST`), `<method>` results in the OpenAPI method (example `GET`), `<opsRef>` results in the OpenAPI `operationId` with a fallback to the `pathRef` in case the OpenAPI does not contain an operation ID. For the full list of dynamic expressions, check the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).

For more details, review the [Assign variables example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-variables) and [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).

<hr>

### Portman - `overwrites` properties

To facilitate automation, you might want to modify properties with "randomized" or specific values. The overwrites are mapped based on the OpenAPI operationId or OpenAPI Operation reference.

#### overwrites options

- **openApiOperationId (String)** : Reference to the OpenAPI operationId for which the Postman request will be overwritten or extended. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, for which the Postman request will be overwritten or extended (example: `['leadsAll', 'companiesAll', 'contactsAll']`)
- **openApiOperation (String)** : Reference to combination of the OpenAPI method & path, for which the Postman request will be overwritten or extended (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting. (example: `["leadsAdd", "GET::/crm/leads/{id}"]`)

- **overwriteRequestBaseUrl (Object)** :

  Key/value pair to overwrite the Postman Request Base URL.

  - **value (String)** : The value that will be used to overwrite/extend the value in the request base URL. (example: `https://example.com` or `{{baseUrl}}`).
  - **overwrite (Boolean true/false | Default: true)** : Overwrites the request base URL value OR attach the value to the original request base URL value.
  - **remove (Boolean true/false | Default: false)** : Removes the targeted request base URL from Postman.

- **overwriteRequestQueryParams (Array)** :

  Array of key/value pairs to overwrite in the Postman Request Query params.

  - **key (String)** : The key that will be targeted in the request Query Param to overwrite/extend. Supports wildcard * to match any sequence of characters. For example, `filter[*]` matches `filter[0]`, `filter[1]`, etc.
  - **value (String)** : The value that will be used to overwrite/extend the value in the request Query Param OR use the [Postman Dynamic variables](https://learning.Postman.com/docs/writing-scripts/script-references/variables-list/) to use dynamic values like `{{$guid}}` or `{{$randomInt}}`. Supports also templating to generate variable names. The template can contain the following dynamic expressions: `<operationId>` results in the OpenAPI operation ID (example `leadsAdd`), `<path>` results in the OpenAPI operation ID (example `/crm/leads`), `<pathRef>` results in the Portman operation (example `POST::/crm/leads_POST`), `<method>` results in the OpenAPI method (example `GET`), `<opsRef>` results in the OpenAPI `operationId` with a fallback to the `pathRef` in case the OpenAPI does not contain an operation ID. For the full list of dynamic expressions, check the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).
  - **overwrite (Boolean true/false | Default: true)** : Overwrites the request query param value OR attach the value to the original request query param value.
  - **disable (Boolean true/false | Default: false)** : Disables the request query param in Postman.
  - **remove (Boolean true/false | Default: false)** : Removes the targeted request query param from Postman.
  - **insert (Boolean true/false | Default: true)** : Insert additional the request query param in Postman that are not present in OpenAPI.
  - **description (String)** : Overwrites the request query param description in Postman.

- **overwriteRequestPathVariables (Array)** :

  Array of key/value pairs to overwrite in the Postman Request Path Variables.

  - **key (String)** : The key that will be targeted in the request Path variables to overwrite/extend.
  - **value (String)** : The value that will be used to overwrite/extend the value in the request path variable OR use the [Postman Dynamic variables](https://learning.Postman.com/docs/writing-scripts/script-references/variables-list/) to use dynamic values like `{{$guid}}` or `{{$randomInt}}`. Supports also templating to generate variable names. The template can contain the following dynamic expressions: `<operationId>` results in the OpenAPI operation ID (example `leadsAdd`), `<path>` results in the OpenAPI operation ID (example `/crm/leads`), `<pathRef>` results in the Portman operation (example `POST::/crm/leads_POST`), `<method>` results in the OpenAPI method (example `GET`), `<opsRef>` results in the OpenAPI `operationId` with a fallback to the `pathRef` in case the OpenAPI does not contain an operation ID. For the full list of dynamic expressions, check the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).
  - **overwrite (Boolean true/false | Default: true)** : Overwrites the request path variable value OR attaches the value to the original request Path variable value.
  - **remove (Boolean true/false | Default: false)** : Removes the targeted request path variable from Postman.
  - **insert (Boolean true/false | Default: true)** : Insert additional the request path variable in Postman that are not present in OpenAPI.
  - **description (String)** : Optional, Overwrites the request path variable description in Postman.

- **overwriteRequestHeaders (Array)** :

  Array of key/value pairs to overwrite in the Postman Request Headers.

  - **key (String)** : The key that will be targeted in the request Headers to overwrite/extend.
  - **value (String)** : The value that will be used to overwrite/extend the value in the request headers OR use the [Postman Dynamic variables](https://learning.Postman.com/docs/writing-scripts/script-references/variables-list/) to use dynamic values like `{{$guid}}` or `{{$randomInt}}`. Supports also templating to generate variable names. The template can contain the following dynamic expressions: `<operationId>` results in the OpenAPI operation ID (example `leadsAdd`), `<path>` results in the OpenAPI operation ID (example `/crm/leads`), `<pathRef>` results in the Portman operation (example `POST::/crm/leads_POST`), `<method>` results in the OpenAPI method (example `GET`), `<opsRef>` results in the OpenAPI `operationId` with a fallback to the `pathRef` in case the OpenAPI does not contain an operation ID. For the full list of dynamic expressions, check the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).
  - **overwrite (Boolean true/false | Default: true)** : Overwrites the request header value OR attaches the value to the original request header value.
  - **disable (Boolean true/false | Default: false)** : Disables the request header in Postman.
  - **remove (Boolean true/false | Default: false)** : Removes the targeted request header from Postman.
  - **insert (Boolean true/false | Default: true)** : Insert the additional request header in Postman that are not present in OpenAPI.
  - **description (String)** : Overwrites the request header description in Postman.

- **overwriteRequestBody (Array)** :

  Array of key/value pairs to overwrite in the Postman Request Body.

  **Applicable for request body types: JSON/form-data/x-www-form-urlencoded**
  - **key (String)** : The key that will be targeted in the request body to overwrite/extend. Use the `.` notation to target nested properties. To target the root level, use `.` as key.
  - **value (Any)** : The value that will be used to overwrite/extend the key in the request body OR use the [Postman Dynamic variables](https://learning.Postman.com/docs/writing-scripts/script-references/variables-list/) to use dynamic values like `{{$guid}}` or `{{$randomInt}}`. The value can be a text/number/boolean/array/object or Postman variable (to pass the Postman variable as type boolean or number, use `{{{variableName}}}` surrounded by 3x {{{ and 3x }}}). Supports also templating to generate variable names. The template can contain the following dynamic expressions: `<operationId>` results in the OpenAPI operation ID (example `leadsAdd`), `<path>` results in the OpenAPI operation ID (example `/crm/leads`), `<pathRef>` results in the Portman operation (example `POST::/crm/leads_POST`), `<method>` results in the OpenAPI method (example `GET`), `<opsRef>` results in the OpenAPI `operationId` with a fallback to the `pathRef` in case the OpenAPI does not contain an operation ID. For the full list of dynamic expressions, check the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).
  - **overwrite (Boolean true/false | Default: true)** : Overwrites the request body value OR attaches the value to the original request body value.
  - **remove (Boolean true/false | Default: false)** : Removes the request body property, including the value.

  **Applicable for request body types: form-data/x-www-form-urlencoded**
  - **insert (Boolean true/false | Default: true)** : Insert the additional request form key/value in Postman that are not present in OpenAPI.
  - **description (String)** : Overwrites the request form data description in Postman.

- **overwriteRequestSecurity (Object)** :

  A Postman RequestAuthDefinition object that will be applied to the request.The security overwrites provides a number of security types:

  - **remove (Boolean true/false | Default: false)** : Unsets the Authorization type in Postman.

  - **apiKey**: The API key auth will send a key-value pair to the API either in the request headers or query parameters.
    - **value (String)** : The value that will be inserted as the Postman apiKey value. It can be a plain value or a Postman variable.
    - **key (String | optional)** : The "key" value that will be inserted in the Postman apiKey key field. It can be a plain value or a Postman variable.
    - **in (String | optional)** : The "in" value that defines where the Api Key will be added in the Postman request Header or Query params. Postman supports `header` for "Header" or `query` for "Query Params".

  ```json
  "overwriteRequestSecurity": {
        "apiKey": {
          "value": "{{apiKey}}"
        }
      }
  ```

  - **bearer**: The bearer tokens allow requests to authenticate using an access key, such as a JSON Web Token (JWT).
    - **token (String)** : The "token" that will be inserted as the Postman bearer token value. It can be a plain value or a Postman variable.

  ```json
  "overwriteRequestSecurity": {
        "bearer": {
          "token": "{{bearerToken}}"
        }
      }
  ```

  - **basic**: Basic authentication involves sending a verified username and password with your request.
    - **username (String)** : The username that will be inserted as the basic authentication username value
    - **password (String)** : The password that will be inserted as the basic authentication password value
  
  ```json
  "overwriteRequestSecurity": {
        "basic": {
          "username": "{{username}}",
          "password": "{{password}}",
        }
      }
  ```
  
  - **Postman security options**: Overwrite/Insert Postman authorization settings.
    - **Postman Type (Array)** : The Postman authorization option type. Supported types are: `awsv4`, `digest`, `edgegrid`, `ntlm`, `oauth1`, `oauth2`
      - **Attributes** : key/value/type as defined in Postman (the easiest way to define it, is to set it manually in Postman, export the collection and extract the matching values from the JSON file).

For more details, review the [Overwrites example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-overwrites) and [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#template-expressions).

<hr>

### Portman - `fuzzing` properties

> Fuzzing or fuzz testing is an automated software testing technique that involves providing invalid, unexpected, or random data as inputs to a computer program (a REST API in the case of Portman).

Fuzzing changes the requests (body, query params, ... ) to unexpected values in an effort to cause unexpected behavior and errors in the API response.
For Portman, we want to provide a simple form of Fuzzing, with the goal to trigger validation/error responses, which can be contract tested.
The automatic fuzzing is based on the OpenAPI request properties, where for each fuzzing variation a new Postman request will be generated, with optional contract tests.

The Fuzzing options describe the configuration setting for available OpenAPI fuzzing variations.

REMARKS:

- [Postman Dynamic variables](https://learning.Postman.com/docs/writing-scripts/script-references/variables-list/) are rendered before being fuzzed.
- Regular Postman variables are skipped from fuzzing.

#### fuzzing options

- **requestBody (Array)** :

  An array of fuzzing options for the Postman Request Body.

  REMARK: Fuzzing is only applicable for OpenAPI request bodies of media type: "application/json"

  - **requiredFields (Boolean)** : Removes the properties & values from the request body that are marked as "required" in OpenAPI.
  - **minimumNumberFields (Boolean)** : Changes the values of the numeric fields to a lower value than the defined "minimum" property in the OpenAPI document.
  - **maximumNumberFields (Boolean)** : Changes the value of the numeric fields to a higher value than the defined "maximum" property in the OpenAPI document.
  - **minLengthFields (Boolean)** : Changes the length of the value to a lower length than the defined "minLength" property in the OpenAPI document.
  - **maxLengthFields (Boolean)** : Changes the length of the value to a higher length than the defined "maxLength" property in the OpenAPI document.

- **requestQueryParams (Array)** :

  An array of fuzzing options for the Postman Request Query parameters.

  - **requiredFields (Boolean)** : Removes the properties & values from the request query params that are marked as "required" in OpenAPI.
  - **minimumNumberFields (Boolean)** : Changes the values of the numeric fields to a lower value than the defined "minimum" property in the OpenAPI document.
  - **maximumNumberFields (Boolean)** : Changes the value of the numeric fields to a higher value than the defined "maximum" property in the OpenAPI document.
  - **minLengthFields (Boolean)** : Changes the length of the value to a lower length than the defined "minLength" property in the OpenAPI document.
  - **maxLengthFields (Boolean)** : Changes the length of the value to a higher length than the defined "maxLength" property in the OpenAPI document.

- **requestHeaders (Array)** :

  An array of fuzzing options for the Postman Request Headers.

  - **requiredFields (Boolean)** : Removes the properties & values from the request headers that are marked as "required" in OpenAPI.
  - **minimumNumberFields (Boolean)** : Changes the values of the numeric fields to a lower value than the defined "minimum" property in the OpenAPI document.
  - **maximumNumberFields (Boolean)** : Changes the value of the numeric fields to a higher value than the defined "maximum" property in the OpenAPI document.
  - **minLengthFields (Boolean)** : Changes the length of the value to a lower length than the defined "minLength" property in the OpenAPI document.
  - **maxLengthFields (Boolean)** : Changes the length of the value to a higher length than the defined "maxLength" property in the OpenAPI document.

For more details, review the [fuzzing example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-fuzzing-tests).

<hr>

### Portman - `operationPreRequestScripts` properties

The `operationPreRequestScripts` configuration will inject pre-request scripts in the Postman collection, on request level.
Postman executes pre-request scripts before a request runs. If you want to set the Postman Collection pre-request scripts on the collection level, you can use the `globals` > `collectionPreRequestScripts` configuration.
The `operationPreRequestScripts` is inserted on the request level.

#### operationPreRequestScripts options

- **openApiOperationId (String)** : Reference to the OpenAPI operationId on which the "Pre-request Scripts" will be inserted. (example: `leadsAll`)
- **openApiOperationIds (Array)** : References to an array of OpenAPI operationIds, for which the "Pre-request Scripts" will be inserted (example: `['leadsAll', 'companiesAll', 'contactsAll']`
- **openApiOperation (String)** : Reference to combination of the OpenAPI method & path, for which the "Pre-request Scripts" will be inserted (example: `GET::/crm/leads`)
- **excludeForOperations (Array | optional)** : References to OpenAPI operations that will be skipped for targeting. (example: `["leadsAdd", "GET::/crm/leads/{id}"]`)

- **scripts (Array)** : Array of scripts that will be injected as Postman Pre-request Scripts on request level, that will be executed before the targeted requests in this collection. Values can be the script content or path to the script file (with `file:` prefix).

<hr>

### Portman - `globals` property

The configuration defined in the `globals` will be executed on the full Postman collection. This is handy if you need to do mass replacements of variables or specific words/keys/values in the full collection that cannot be overwritten per request.

#### globals options

- **stripResponseExamples (Default: false)** : Strip the response examples from the generated Postman collection.
- **variableCasing** : Change the casing of the auto-generated Postman variables. Supported values are: `camelCase`, `pascalCase`, `kebabCase`, `trainCase`, `snakeCase`, `adaCase`, `constantCase`, `cobolCase`, `dotNotation`. See the [Assign & Overwrite example](https://github.com/apideck-libraries/portman/tree/main/examples/testsuite-assign-overwrite#globals) for the different casing options.
- **separatorSymbol (Default: "::")** : Change the separator symbol for the auto-generated Postman testName description (Example: `[GET]::/crm/leads - Status code is 2xx`). Helpful when using the [postman-to-k6](https://github.com/apideck-libraries/postman-to-k6) converter.
- **collectionPreRequestScripts** : Array of scripts that will be injected as Postman Collection Pre-request Scripts that will be executed by Postman before every request in this collection. Values can be the script content or path to the script file (with `file:` prefix).
- **collectionTestScripts**: Array of scripts that will be injected as Postman Collection Test Scripts will be executed by Postman after every request in this collection. Values can be the script content or path to the script file (with `file:` prefix).
- **collectionVariables**: A map of key value pairs that will inserted as Postman collection variables.
- **keyValueReplacements** : A map of parameter key names that will have their values replaced with the provided Postman variables.
- **valueReplacements** : A map of values that will have their values replaced with the provided values.
- **rawReplacements** : Consider this a "search & replace" utility, that will search a string/object/... and replace it with another string/object/...
  This is very useful to replace data from the OpenAPI specification, before it is used in the Portman test automation generation.
- **portmanReplacements** : The "search & replace" utility right before the final Postman file is written, that will search a string/object/... and replace it with another string/object/...
  This is practical to replace any data from the generated Portman collection, before it is used in Postman / Newman test execution.
- **orderOfOperations** : The `orderOfOperations` is a list of OpenAPI operations, which is used by Portman to sort the Postman requests in the desired order, in their folder. The ordering from `orderOfOperations` is performed per folder. Items that are **not** defined in the `orderOfOperations` list will remain at their current order.
- **orderOfFolders** : The `orderOfFolders` is a list of Postman folder names, which is used by Portman to sort the Postman folders in the desired order. Folders that are **not** defined in the `orderOfFolders` list will remain at their current order, after the re-order folders.
- **securityOverwrites** : Overwrite of the OpenAPI Security Scheme Object (supported types: "apiKey", "http basic auth", "http bearer token") or inject a Postman authorization option (supported types: awsv4, digest, edgegrid, ntlm, oauth1, oauth2) on a collection level. 

  The security overwrites provides a number of security types:

  - **remove (Boolean true/false | Default: false)** : Unsets the Authorization type in Postman.
  
  - **apiKey**: The API key auth will send a key-value pair to the API either in the request headers or query parameters.
    - **value (String)** : The value that will be inserted as the Postman apiKey value. It can be a plain value or a Postman variable.
    - **key (String | optional)** : The "key" value that will be inserted in the Postman apiKey key field. It can be a plain value or a Postman variable.
    - **in (String | optional)** : The "in" value that defines where the Api Key will be added in the Postman request Header or Query params. Postman supports `header` for "Header" or `query` for "Query Params".
  
  ```json
  "securityOverwrites": {
        "apiKey": {
          "value": "{{apiKey}}"
        }
      }
  ```
  
  - **bearer**: The bearer tokens allow requests to authenticate using an access key, such as a JSON Web Token (JWT).
    - **token (String)** : The "token" that will be inserted as the Postman bearer token value. It can be a plain value or a Postman variable.
  
  ```json
  "securityOverwrites": {
        "bearer": {
          "token": "{{bearerToken}}"
        }
      }
  ```
  
  - **basic**: Basic authentication involves sending a verified username and password with your request.
    - **username (String)** : The username that will be inserted as the basic authentication username value
    - **password (String)** : The password that will be inserted as the basic authentication password value
  
  ```json
  "securityOverwrites": {
        "basic": {
          "username": "{{username}}",
          "password": "{{password}}",
        }
      }
  ```
  
  - **Postman security options**: Overwrite/Insert Postman authorization settings.
    - **Postman Type (Array)** : The Postman authorization option type. Supported types are: `awsv4`, `digest`, `edgegrid`, `ntlm`, `oauth1`, `oauth2`
      - **Attributes** : key/value/type as defined in Postman (the easiest way to define it, is to set it manually in Postman, export the collection and extract the matching values from the JSON file). 
  
  ```json
  {
    "globals": {
      "securityOverwrites": {
        "oauth1": [
          {
            "key": "addEmptyParamsToSign",
            "value": true,
            "type": "boolean"
          },
          {
            "key": "timestamp",
            "value": "1461319769",
            "type": "string"
          },
          {
            "key": "nonce",
            "value": "ik3oT5",
            "type": "string"
          },
          {
            "key": "consumerSecret",
            "value": "D+EdQ-gs$-%@2Nu7",
            "type": "string"
          },
          {
            "key": "consumerKey",
            "value": "RKCGzna7bv9YD57c",
            "type": "string"
          },
          {
            "key": "signatureMethod",
            "value": "HMAC-SHA1",
            "type": "string"
          },
          {
            "key": "version",
            "value": "1.0",
            "type": "string"
          },
          {
            "key": "addParamsToHeader",
            "value": false,
            "type": "boolean"
          }
        ]
      }
    }
  }
  ```

For more details on the `globals` configuration options , review the [globals example](https://github.com/apideck-libraries/portman/tree/main/examples/portman-globals) and [ordering example](https://github.com/apideck-libraries/portman/tree/main/examples/postman-ordering)

<hr>

## Configure automatic upload to Postman App

> REMARK: Portman does **not** require you to have a Postman account.

In case you want to sync the generated Postman collection with the Postman app (`portman --syncPostman`), you would need a Postman account since Portman leverages the Postman API to sync the collection.

This can be a "free" Postman account or any of the paid [Postman plans](https://www.postman.com/pricing/).

The generated Postman collection can always be [imported manually](https://learning.postman.com/docs/getting-started/importing-and-exporting-data/#importing-data-into-postman), without a Postman account.

To enable automatic uploads of the generated Postman collection through Portman, follow these steps:

1. Get your Postman API key

![Documentation Pipeline](https://raw.githubusercontent.com/apideck-libraries/portman/main/docs/img/postman-automation-0.png)

![Documentation Pipeline](https://raw.githubusercontent.com/apideck-libraries/portman/main/docs/img/postman-automation-1.png)

![Documentation Pipeline](https://raw.githubusercontent.com/apideck-libraries/portman/main/docs/img/postman-automation-2.png)

2. Goto the root folder of your project

3. Copy [env-postman-app-example](./.env-postman-app.example) as `.env` in the root folder of your project

4. Enter your Postman API key in a local `.env` file, as `POSTMAN_API_KEY=[replace with Postman api key]`

Next to the Postman API key, you can also pass along the Postman Workspace name & the specific Postman Collection UID.

Supported Postman API .ENV variables:

- **POSTMAN_API_KEY** : Postman API key
- **POSTMAN_WORKSPACE_NAME** : Postman Workspace name to target the upload of the generated Postman collection
- **POSTMAN_COLLECTION_UID** : Postman collection UID to upload with the generated Postman collection

The `POSTMAN_WORKSPACE_NAME` & `POSTMAN_COLLECTION_UID` variables can also be set as CLI Options `--postmanWorkspaceName` & `--postmanUid` , which will overrule the variables defined in the .ENV file.

> **RECOMMENDATION**: Do not commit the `.env` file in any versioning system like GIT if it contains confidential credentials.

# Credits

Portman started as a PR on the handy [openapi-to-postman](https://github.com/postmanlabs/openapi-to-postman) package to generate basic Postman tests from the OpenAPI specification.

[Apideck](https://www.apideck.com/) immediately saw the PR's value and collaborated with the original author, [Tim Haselaars](https://github.com/thim81), to adopt the functionality and extend the options & tooling to create "Portman".

The goal of Portman is to drive API automation by 'porting' a static OpenAPI document to a dynamic Postman collection that includes a powerful testing suite with variable requests, bodies and more. All this while being easy to configure & ready to use.

Portman is a valuable tool in any OpenAPI workflow, for local development or as part of a CI/CD automation pipeline.

Credits for this package for the hard work of [Nick Lloyd](https://github.com/nicklloyd) and [Tim Haselaars](https://github.com/thim81).

# Future ideas

- [ ] Make Postman security dynamic

# Resources

A collection of blog posts and resources about Portman

- https://www.andmore.dev/blog/getting-started-portman/ by [andmoredev](https://github.com/andmoredev)
- https://itnext.io/automating-api-testing-with-portman-postman-and-newman-ec1a869cbc99 by Tyler Owen
- https://www.codecentric.de/wissens-hub/blog/charge-your-apis-volume-3-optimizing-api-testing-with-contract-testing by [Daniel Kocot](https://github.com/danielkocot)
- https://blog.apideck.com/portman-api-testing by [Chris Wood](https://www.linkedin.com/in/sensiblewood/)
- https://dev.to/oneadvanced/api-provider-contract-testing-for-all-with-portman-openapi-and-postman-4ll1 by [Alex Savage](https://github.com/savage-alex)
- https://github.com/thim81/spec-driven-openapi-contract-performance-testing by [Tim Haselaars](https://github.com/thim81)
- https://www.codecentric.de/wissens-hub/blog/charge-your-apis-volume-25-contract-testing by Pasquale Brunelli
- https://qase.io/blog/automated-contract-testing/amp/ Contract Testing in Action by Kirill Ivliev
 readmeEtag: '"82cf93ad248ada79c6bcbcf35c8006f230b358ba"' readmeLastModified: Thu, 05 Feb 2026 23:01:08 GMT repositoryId: 364542854 description: >- Port OpenAPI Specs to Postman Collections, inject test suite and run via Newman 👨🏽‍🚀 created: '2021-05-05T10:42:29Z' updated: '2026-02-05T23:01:12Z' language: TypeScript archived: false stars: 677 watchers: 7 forks: 61 owner: apideck-libraries logo: https://avatars.githubusercontent.com/u/73573473?v=4 license: Apache-2.0 repoEtag: '"92584518b57295d412ecb3e9f8a30457c691f0df2d02fe832458e5c252db81fc"' repoLastModified: Thu, 05 Feb 2026 23:01:12 GMT foundInMaster: true id: a03cdb329f630a946a946dfe26802c43 - source: https://openapi.tools/ name: Step CI category: - Testing - Converters - Data Validators language: - CLI - Node.js link: https://stepci.com repository: https://github.com/stepci/stepci source_description: > Open-source framework for API Quality Assurance. Generate multi-step test-scenarios from OpenAPI. Validate responses against description documents and more. v3: true id: 50dd208178d094d51cbcd8e57f209c23 repositoryMetadata: base64Readme: >- IVtTY3JlZW4gUmVjb3JkaW5nIDIwMjMtMTAtMDQgYXQgMTUgNDMgMTddKGh0dHBzOi8vZ2l0aHViLmNvbS9zdGVwY2kvc3RlcGNpL2Fzc2V0cy8xMDQwMDA2NC84ODFlZmQ0OS1mZDkzLTRmZjgtOGU5OS00YjZlMjRmZTEyMjcpCgo+ICoqTm90ZSoqCj4gV2UganVzdCBhbm5vdW5jZWQgW1N1cHBvcnQgUGxhbl0oaHR0cHM6Ly9zdGVwY2kuY29tLyNwcmljaW5nKSBmb3IgU3RlcCBDSQoKPiAqKkltcG9ydGFudCoqCj4gRm9yIHVzZXJzIG1pZ3JhdGluZyBmcm9tIFBvc3RtYW4gYW5kIEluc29tbmlhLCBzZWUgaXNzdWVzIFsjMjldKGh0dHBzOi8vZ2l0aHViLmNvbS9zdGVwY2kvc3RlcGNpL2lzc3Vlcy8yOSkgYW5kIFsjMzBdKGh0dHBzOi8vZ2l0aHViLmNvbS9zdGVwY2kvc3RlcGNpL2lzc3Vlcy8zMCkgcmVzcGVjdGl2ZWx5CgojIFdlbGNvbWUKClN0ZXAgQ0kgaXMgYW4gb3Blbi1zb3VyY2UgQVBJIFF1YWxpdHkgQXNzdXJhbmNlIGZyYW1ld29yawoKLSAqKkxhbmd1YWdlLWFnbm9zdGljKiouIENvbmZpZ3VyZSBlYXNpbHkgdXNpbmcgWUFNTCwgSlNPTiBvciBKYXZhU2NyaXB0Ci0gKipSRVNULCBHcmFwaFFMLCBnUlBDLCB0UlBDLCBTT0FQKiouIFRlc3QgZGlmZmVyZW50IEFQSSB0eXBlcyBpbiBvbmUgd29ya2Zsb3cKLSAqKlNlbGYtaG9zdGVkKiouIFRlc3Qgc2VydmljZXMgb24geW91ciBuZXR3b3JrLCBsb2NhbGx5IGFuZCBDSS9DRAotICoqSW50ZWdyYXRlZCoqLiBQbGF5IG5pY2VseSB3aXRoIG90aGVycwoKW+KGkiAqKlJlYWQgdGhlIERvY3MqKl0oaHR0cHM6Ly9kb2NzLnN0ZXBjaS5jb20pCgpb4oaSICoqVHJ5IHRoZSBPbmxpbmUgUGxheWdyb3VuZCoqXShodHRwczovL3N0ZXBjaS5jb20pCgpb4oaSICoqSm9pbiB1cyBvbiBEaXNjb3JkKipdKGh0dHBzOi8vZGlzY29yZC5nZy9LcUpKekozQlR1KQoKIyMgR2V0IHN0YXJ0ZWQKCjEuIEluc3RhbGwgdGhlIENMSQoKICAgKipVc2luZyBbTm9kZS5qc10oaHR0cHM6Ly9ub2RlanMub3JnL2VuLykqKgoKICAgIGBgYAogICAgbnBtIGluc3RhbGwgLWcgc3RlcGNpCiAgICBgYGAKCiAgICA+ICoqTm90ZSoqOiBNYWtlIHN1cmUgeW91J3JlIHVzaW5nIHRoZSBMVFMgdmVyc2lvbiBvZiBOb2RlLmpzCgogICAgKipVc2luZyBbSG9tZWJyZXddKGh0dHBzOi8vYnJldy5zaC8pKioKCiAgICBgYGAKICAgIGJyZXcgaW5zdGFsbCBzdGVwY2kKICAgIGBgYAoKMi4gQ3JlYXRlIGV4YW1wbGUgd29ya2Zsb3cKCiAgICAqKndvcmtmbG93LnltbCoqCgogICAgYGBgeWFtbAogICAgdmVyc2lvbjogIjEuMSIKICAgIG5hbWU6IFN0YXR1cyBDaGVjawogICAgZW52OgogICAgICBob3N0OiBleGFtcGxlLmNvbQogICAgdGVzdHM6CiAgICAgIGV4YW1wbGU6CiAgICAgICAgc3RlcHM6CiAgICAgICAgICAtIG5hbWU6IEdFVCByZXF1ZXN0CiAgICAgICAgICAgIGh0dHA6CiAgICAgICAgICAgICAgdXJsOiBodHRwczovLyR7e2Vudi5ob3N0fX0KICAgICAgICAgICAgICBtZXRob2Q6IEdFVAogICAgICAgICAgICAgIGNoZWNrOgogICAgICAgICAgICAgICAgc3RhdHVzOiAvXjIwLwogICAgYGBgCgogICAgPiAqKk5vdGUqKjogWW91IGNhbiBhbHNvIGFsc28gdXNlIEpTT04gZm9ybWF0IHRvIGNvbmZpZ3VyZSB5b3VyIHdvcmtmbG93CgozLiBSdW4gdGhlIHdvcmtmbG93CgogICAgYGBgCiAgICBzdGVwY2kgcnVuIHdvcmtmbG93LnltbAogICAgYGBgCgogICAgYGBgCiAgICBQQVNTICBleGFtcGxlCgogICAgVGVzdHM6IDAgZmFpbGVkLCAxIHBhc3NlZCwgMSB0b3RhbAogICAgU3RlcHM6IDAgZmFpbGVkLCAxIHBhc3NlZCwgMSB0b3RhbAogICAgVGltZTogIDAuNTU5cywgZXN0aW1hdGVkIDFzCgogICAgV29ya2Zsb3cgcGFzc2VkIGFmdGVyIDAuNTU5cwogICAgYGBgCgojIyBEb2N1bWVudGF0aW9uCgpEb2N1bWVudGF0aW9uIGlzIGF2YWlsYWJsZSBvbiBbZG9jcy5zdGVwY2kuY29tXShodHRwczovL2RvY3Muc3RlcGNpLmNvbSkKCiMjIEV4YW1wbGVzCgpZb3UgY2FuIGZpbmQgZXhhbXBsZSB3b3JrZmxvd3MgdW5kZXIgW2BleGFtcGxlcy9gXShleGFtcGxlcy8pCgojIyBDb21tdW5pdHkKCkpvaW4gb3VyIGNvbW11bml0eSBvbiBbRGlzY29yZF0oaHR0cHM6Ly9kaXNjb3JkLmdnL0txSkp6SjNCVHUpIGFuZCBbR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vc3RlcGNpL3N0ZXBjaS9kaXNjdXNzaW9ucykKCiMjIENvbnRyaWJ1dGluZwoKQXMgYW4gb3Blbi1zb3VyY2UgcHJvamVjdCwgd2Ugd2VsY29tZSBjb250cmlidXRpb25zIGZyb20gdGhlIGNvbW11bml0eS4gSWYgeW91IGFyZSBleHBlcmllbmNpbmcgYW55IGJ1Z3Mgb3Igd2FudCB0byBhZGQgc29tZSBpbXByb3ZlbWVudHMsIHBsZWFzZSBmZWVsIGZyZWUgdG8gb3BlbiBhbiBpc3N1ZSBvciBwdWxsIHJlcXVlc3QKCiMjIFN1cHBvcnQgUGxhbgoKR2V0IFByby1sZXZlbCBzdXBwb3J0IHdpdGggU0xBLCBvbmJvYXJkaW5nLCBwcmlvcml0aXplZCBmZWF0dXJlLXJlcXVlc3RzIGFuZCBidWdmaXhlcy4KClvihpIgKipMZWFybiBtb3JlKipdKGh0dHBzOi8vc3RlcGNpLmNvbS8jcHJpY2luZykKCjxhIGhyZWY9Imh0dHBzOi8vY2FsLmNvbS91c2hha292L3N0ZXAtY2ktZGVtbyI+PGltZyBhbHQ9IkJvb2sgdXMgd2l0aCBDYWwuY29tIiBzcmM9Imh0dHBzOi8vY2FsLmNvbS9ib29rLXdpdGgtY2FsLWRhcmsuc3ZnIiAvPjwvYT4KCiMjIFByaXZhY3kKCkJ5IGRlZmF1bHQsIHRoZSBDTEkgY29sbGVjdHMgYW5vbnltb3VzIHVzYWdlIGRhdGEsIHdoaWNoIGluY2x1ZGVzOgoKLSBVbmlxdWUgdXNlciBJRAotIE9TIE5hbWUKLSBOb2RlIFZlcnNpb24KLSBDTEkgVmVyc2lvbgotIENvbW1hbmQgKGBzdGVwY2kgaW5pdGAsIGBzdGVwY2kgcnVuYCwgYHN0ZXBjaSBnZW5lcmF0ZWApCi0gRW52aXJvbm1lbnQgKExvY2FsLCBEb2NrZXIsIENJL0NEKQoKPiAqKk5vdGUqKgo+IFRoZSB1c2FnZSBhbmFseXRpY3MgY2FuIGJlIGRpc2FibGVkIGJ5IHNldHRpbmcgYFNURVBDSV9ESVNBQkxFX0FOQUxZVElDU2AgZW52aXJvbm1lbnQgdmFyaWFibGUKCiMjIExpY2Vuc2UKClRoZSBzb3VyY2UgY29kZSBpcyBkaXN0cmlidXRlZCB1bmRlciBNb3ppbGxhIFB1YmxpYyBMaWNlbnNlIHRlcm1zCg== readmeEtag: '"dd8463cf1c9e81e4c7825e4e6f74da1ca906f414"' readmeLastModified: Sat, 03 Aug 2024 09:44:52 GMT repositoryId: 537500924 description: Automated API Testing and Quality Assurance created: '2022-09-16T14:48:21Z' updated: '2026-02-04T07:59:49Z' language: TypeScript archived: false stars: 1835 watchers: 12 forks: 93 owner: stepci logo: https://avatars.githubusercontent.com/u/61350067?v=4 license: MPL-2.0 repoEtag: '"ae37b97369c34f5c926a292fea7f60fe6b4b8f2f86363f29c02bd4417f263e2f"' repoLastModified: Wed, 04 Feb 2026 07:59:49 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: openapi-spring-webflux-validator category: - Description Validators - Data Validators repository: https://github.com/cdimascio/openapi-spring-webflux-validator language: - Java - Kotlin source_description: > A friendly kotlin library to validate API endpoints against an OpenAPI description document. v2: true v3: true repositoryMetadata: base64Readme: >- # openapi-spring-webflux-validator
![](https://travis-ci.org/cdimascio/openapi-spring-webflux-validator.svg?branch=master)[![Maven Central](https://img.shields.io/maven-central/v/io.github.cdimascio/openapi-spring-webflux-validator.svg?label=Maven%20Central)](https://search.maven.org/artifact/io.github.cdimascio/openapi-spring-webflux-validator/4.2.0/jar) [![Codacy Badge](https://api.codacy.com/project/badge/Grade/f78b72ca90104e42b111723a7720adf3)](https://www.codacy.com/app/cdimascio/openapi-spring-webflux-validator?utm_source=github.com&amp;utm_medium=referral&amp;utm_content=cdimascio/openapi-spring-webflux-validator&amp;utm_campaign=Badge_Grade) ![](https://img.shields.io/badge/license-Apache%202.0-blue.svg)<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-6-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END --> 

A friendly kotlin library to validate API endpoints using an _OpenApi 3_ or _Swagger 2_ specification. Great with webflux functional. 
It **works happily with Spring Webflux 6's baseline of Jakarta JVM runtime >=17**. 
<p align="center">
	<img width="300" src="https://raw.githubusercontent.com/cdimascio/openapi-spring-webflux-validator/master/assets/openapi-webflux-validator-logo2.png" width="600"/>
</p>

Supports specifications in _YAML_ and _JSON_

## Prequisites

and Spring Webflux 6 + Java >=17

_For use with Spring Boot 2 and Webflux 5, use `openapi-spring-webflux-validator` version `3.5.0`. Java 8 or greater is required._ 

## Install

### Maven

```xml
<dependency>
    <groupId>io.github.cdimascio</groupId>
    <artifactId>openapi-spring-webflux-validator</artifactId>
    <version>4.2.0</version>
</dependency>
```

### Gradle

```groovy
compile 'io.github.cdimascio:openapi-spring-webflux-validator:4.2.0'
```

For sbt, grape, ivy and more, see [here](https://search.maven.org/#artifactdetails%7Cio.github.cdimascio%7Copenapi-spring-webflux-validator%7C2.0.0%7Cjar)

## Usage (Kotlin)

This section and the next describe usage with Kotlin and Java respectively.

See this [complete Spring Webflux example that uses openapi-spring-webflux-validator](https://github.com/cdimascio/kotlin-swagger-spring-functional-template).

### Configure (Kotlin)

This one-time configuration requires you to provide the _location of the openapi/swagger specification_ and an optional _custom error handler_.

Supports `JSON` and `YAML`

```kotlin
import io.github.cdimascio.openapi.Validate

val validate = Validate.configure("static/api.yaml")
```

with custom error handler

```kotlin
import org.springframework.web.reactive.function.server.ServerRequest

data class MyError(val request: ServerRequest, val code: String, val messages: List<String>)
val validate = Validate.configure("static/api.json") { request, status, messages ->
   MyError(request, status.name, messages)
}
```

with custom ObjectMapper factory:

```kotlin
val validate = Validate.configure(
   openApiSwaggerPath = "api.yaml",
   errorHandler = { request, status, message -> ValidationError(request, status.value(), message[0]) },
   objectMapperFactory = { ObjectMapper()
       .registerKotlinModule()
       .registerModule(JavaTimeModule())
       .configure(SerializationFeature.WRITE_DATES_AS_TIMESTAMPS, false) }
)
```

### Validate a request (Kotlin + Reactor)

You can now validate a request in a coroutine style,
using the `validate` instance created [above](#configure-kotlin):

without a body

```kotlin
validate.request(req) {
    // Do stuff e.g. return a list of names 
    ok().body(Mono.just(listOf("carmine", "alex", "eliana")))
}
```

with body

```kotlin
validate.request(req).withBody(User::class.java) { body ->
    // Note that body is deserialized as User!
    // Now you can do stuff. 
    // For example, lets echo the request as the response 
    ok().body(Mono.just(body))
}
```

with body you want to process as string (e.g. for computing a request signature), or that you want to deserialize somehow specifically

```kotlin
val identity: (String) -> String = { it }
validate.request(req).withBody(String::class.java, readValue = identity) { body ->
    ok().body(Mono.just("content length is ${body.length}"))
}
```

### Validate a request (Kotlin + coroutines)

Or you can validate a request in a coroutine style,
using the `validate` instance created [above](#configure-kotlin):


without a body

```kotlin
validate.requestAndAwait(req) {
    // Do stuff e.g. return a list of names 
    ok().bodyValueAndAwait(listOf("carmine", "alex", "eliana"))
}
```

with body

```kotlin
validate.request(req).awaitBody(User::class.java) { body: User ->
    // Note that body is deserialized as User!
    // Now you can do stuff. 
    // For example, lets echo the request as the response 
    ok().bodyValueAndAwait(body)
}
```

with body you want to process as string (e.g. for computing a request signature), or that you want to deserialize somehow specifically

```kotlin
val identity: (String) -> String = { it }
validate.request(req).awaitBody(String::class.java, identity) { body: String ->
    ok().bodyValueAndAwait("content length is ${body.length}")
}
```

## Usage

### Configure (Java)
This one-time configuration requires you to provide the _location of the openapi/swagger specification_ and an optional _custom error handler_.

```java
import io.github.cdimascio.openapi.Validate;

Validate<ValidationError> validate = Validate.configure("static/api.json")
```

with custom error handler

```java
import org.springframework.web.reactive.function.server.ServerRequest;

class MyError {
    private ServerRequest request;
    private String id;
    private  String messages;
    public MyError(ServerRequest request, String id, List<String> messages) {
        this.request = request;
        this.id = id;
        this.messages = messages;
    }
    public ServerRequest getRequest() {
        return request;
    }
    public String getId() {
        return id;
    }
    public void setId(String id) {
        this.id = id;
    }
    public List<String> getMessages() {
        return messages;
    }
    public void setMessages(List<String> messages) {
        this.messages = messages;
    }     
}
```

```java
Validate<ValidationError> validate = Validate.configure("static/api.json", (request, status, messages) ->
    new MyError(request, status.getName(), messages)
);
```

### Validate a request (Java)

Using the `validate` instance created above, you can now validate a request:

without a body

```java
ArrayList<String> users = new ArrayList<String>() {{
    add("carmine");
    add("alex");
    add("eliana");
}};

validate.request(req, () ->
    // Do stuff e.g. return a list of user names
    ServerResponse.ok().bodyValue(users)
);
```

with body

```java
validate
    .request(req)
    .withBody(User.class, user -> 
        // Note that body is deserialized as User!
        // Now you can do stuff. 
        // For example, lets echo the request as the response
        ServerResponse.ok().bodyValue(user)
    );
```

with body you want to process as string (e.g. for computing a request signature)

```java
validate
    .request(req)
    .withBody(String.class, s -> s, body ->
        ServerResponse.ok().bodyValue("content length is " + body.length())
    );
```

## Example Validation Output

Let's assume a `POST` request to create a user requires the following request body:

```json
{
  "firstname": "carmine",
  "lastname": "dimasico"
}
```

Let's now assume an API user misspells `lastname` as `lastnam`

```shell
curl -X POST http://localhost:8080/api/users -H "Content-Type: application/json" -d'{ 
  "firstname": "c", 
  "lastnam": "d" 
}'
```

`openapi-spring-webflux-validator` automatically validates the request against a Swagger spect and returns:

```json
{
  "code": 400,
  "messages":[
	  "Object instance has properties which are not allowed by the schema: [\"lastnam\"]",
	  "Object has missing required properties ([\"lastname\"])"
  ]
} 
```

**Woah! Cool!!** :-D 

## Example

Let's say you have an endpoint `/users` that supports both `GET` and `POST` operations.

You can create those routes and validate them like so:

**Create the routes in a reactive or coroutine style:**

```kotlin
package myproject.controllers

import org.springframework.core.io.ClassPathResource
import org.springframework.http.MediaType.*
import org.springframework.web.reactive.function.server.ServerResponse.permanentRedirect
import org.springframework.web.reactive.function.server.coRouter
import org.springframework.web.reactive.function.server.plus
import org.springframework.web.reactive.function.server.router
import java.net.URI

class Routes(private val userHandler: UserHandler) {
    fun router() = router {
        "/api".nest {
            accept(APPLICATION_JSON).nest {
                POST("/users", userHandler::create)
            }
            accept(TEXT_EVENT_STREAM).nest {
                GET("/users", userHandler::findAll)
            }
        }
    } + coRouter { 
        "/coApi".nest {
            accept(APPLICATION_JSON).nest {
                POST("/users", userHandler::coCreate)
            }
            accept(TEXT_EVENT_STREAM).nest {
                GET("/users", userHandler::coFindAll)
            }
        }
    }
}
```

```kotlin
package myproject

import io.github.cdimascio.openapi.Validate

val validate = Validate.configure("static/api.yaml")
```

**Validate with openapi-spring-webflux-validator**

```kotlin
package myproject.controllers

import myproject.models.User
import myproject.validate
import org.springframework.web.reactive.function.server.ServerRequest
import org.springframework.web.reactive.function.server.ServerResponse
import org.springframework.web.reactive.function.server.ServerResponse.ok
import org.springframework.web.reactive.function.server.bodyValueAndAwait
import reactor.core.publisher.Flux
import reactor.core.publisher.Mono

class UserHandler {

    fun findAll(req: ServerRequest): Mono<ServerResponse> {
        return validate.request(req) {
            ok().bodyValue(listOf("carmine", "alex", "eliana"))
        }
    }

    fun create(req: ServerRequest): Mono<ServerResponse> {
        return validate.request(req).withBody(User::class.java) {
            // it is the request body deserialized as User
            ok().bodyValue(it)
        }
    }

    suspend fun coFindAll(req: ServerRequest): ServerResponse {
        return validate.requestAndAwait(req) {
            ok().bodyValueAndAwait(listOf("carmine", "alex", "eliana"))
        }
    }

    suspend fun coCreate(req: ServerRequest): ServerResponse {
        return validate.request(req).awaitBody(User::class.java) {
            // it is the request body deserialized as User
            ok().bodyValueAndAwait(it)
        }
    }
}
```

## License

[Apache 2.0](https://www.apache.org/licenses/LICENSE-2.0)

<a href="https://www.buymeacoffee.com/m97tA5c" target="_blank"><img src="https://bmc-cdn.nyc3.digitaloceanspaces.com/BMC-button-images/custom_images/orange_img.png" alt="Buy Me A Coffee" style="height: auto !important;width: auto !important;" ></a>

## Contributors ✨

Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://github.com/cdimascio"><img src="https://avatars1.githubusercontent.com/u/4706618?v=4" width="100px;" alt=""/><br /><sub><b>Carmine DiMascio</b></sub></a><br /><a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=cdimascio" title="Code">💻</a> <a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=cdimascio" title="Tests">⚠️</a> <a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=cdimascio" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/krzykrucz"><img src="https://avatars1.githubusercontent.com/u/18364177?v=4" width="100px;" alt=""/><br /><sub><b>Krzysiek Kruczyński</b></sub></a><br /><a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=krzykrucz" title="Code">💻</a> <a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=krzykrucz" title="Tests">⚠️</a> <a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=krzykrucz" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/chejerlakarthik"><img src="https://avatars0.githubusercontent.com/u/12871079?v=4" width="100px;" alt=""/><br /><sub><b>Chejerla Karthik</b></sub></a><br /><a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=chejerlakarthik" title="Code">💻</a></td>
    <td align="center"><a href="http://www.katielevy.com"><img src="https://avatars0.githubusercontent.com/u/8975181?v=4" width="100px;" alt=""/><br /><sub><b>Katie Levy</b></sub></a><br /><a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=katielevy1" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/reinterpretcat"><img src="https://avatars1.githubusercontent.com/u/1611077?v=4" width="100px;" alt=""/><br /><sub><b>Ilya Builuk</b></sub></a><br /><a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=reinterpretcat" title="Code">💻</a></td>
    <td align="center"><a href="http://simon.zambrovski.org/"><img src="https://avatars0.githubusercontent.com/u/673128?v=4" width="100px;" alt=""/><br /><sub><b>Simon Zambrovski</b></sub></a><br /><a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=zambrovski" title="Code">💻</a> <a href="https://github.com/cdimascio/openapi-spring-webflux-validator/commits?author=zambrovski" title="Tests">⚠️</a></td>
  </tr>
</table>

<!-- markdownlint-enable -->
<!-- prettier-ignore-end -->
<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
 readmeEtag: '"a2e728aee174f34ca6d5605aad0082ccc6702d5c"' readmeLastModified: Wed, 28 Dec 2022 03:11:56 GMT repositoryId: 112494052 description: >- 🌱 A friendly kotlin library to validate API endpoints using an OpenApi 3.0 and Swagger 2.0 specification created: '2017-11-29T15:41:04Z' updated: '2026-02-03T03:42:17Z' language: Kotlin archived: false stars: 99 watchers: 4 forks: 13 owner: cdimascio logo: https://avatars.githubusercontent.com/u/4706618?v=4 license: NOASSERTION repoEtag: '"72b51de16b770b7520b269e91cc8655e75a4c31bbe4c0bf26b20a3a6f6d350a0"' repoLastModified: Tue, 03 Feb 2026 03:42:17 GMT foundInMaster: true id: 73c574b5f31d77e560375bfaef2b65eb - source: - https://openapi.tools/ - openapi3 tags name: Spectral homepage: https://github.com/stoplightio/spectral language: - CLI - TypeScript - JavaScript source_description: > A flexible OpenAPI/AsyncAPI API linter, which can help you build anything from automated API Style Guides for API Governance workflows, or handle any other JSON/YAML object linting. functions. category: Description Validators link: https://stoplight.io/spectral repository: https://github.com/stoplightio/spectral v2: true v3: true v3_1: true repositoryMetadata: base64Readme: >- [![Demo of Spectral linting an OpenAPI document from the CLI](./docs/img/readme-header.svg)](https://stoplight.io/api-governance?utm_source=github&utm_medium=spectral&utm_campaign=readme)
[![CircleCI](https://img.shields.io/circleci/build/github/stoplightio/spectral/develop)](https://circleci.com/gh/stoplightio/spectral) [![npm Downloads](https://img.shields.io/npm/dw/@stoplight/spectral-core?color=blue)](https://www.npmjs.com/package/@stoplight/spectral-core) [![Stoplight Forest](https://img.shields.io/ecologi/trees/stoplightinc)][stoplight_forest]

- **Custom Rulesets**: Create custom rules to lint JSON or YAML objects
- **Ready-to-use Rulesets**: Validate and lint **OpenAPI v2 & v3.x**, **AsyncAPI**, and **Arazzo v1** Documents
- **API Style Guides**: Automated [API Style Guides](https://stoplight.io/api-style-guides-guidelines-and-best-practices?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_spectral) using rulesets improve consistency across all your APIs
- **Ready-to-use Functions**: Built-in set of functions to help [create custom rules](https://meta.stoplight.io/docs/spectral/e5b9616d6d50c-custom-rulesets#adding-rules). Functions include pattern checks, parameter checks, alphabetical ordering, a specified number of characters, provided keys are present in an object, etc.
- **Custom Functions**: Create custom functions for advanced use cases

# Overview

- [🧰 Installation](#-installation)
- [💻 Usage](#-usage)
- [📖 Documentation](#-documentation)
- [ℹ️ Support](#ℹ️-support)
- [🌎 Real-World Rulesets](#-real-world-rulesets)
- [⚙️ Integrations](#️-integrations)
- [👏 Contributing](#-contributing)
- [🌲 Sponsor Spectral by Planting a Tree](#-sponsor-spectral-by-planting-a-tree)

## 🧰 Installation

The easiest way to install spectral is to use either [npm](https://www.npmjs.com/):

```bash
npm install -g @stoplight/spectral-cli
```

Or [yarn](https://yarnpkg.com/):

```
yarn global add @stoplight/spectral-cli
```

There are also [additional installation options](https://meta.stoplight.io/docs/spectral/ZG9jOjYyMDc0Mw-installation).

## 💻 Usage

### 1. Create a local ruleset

Spectral, being a generic YAML/JSON linter, **needs a ruleset** to lint files. A ruleset is a JSON, YAML, or JavaScript/TypeScript file (often the file is called `.spectral.yaml` for a YAML ruleset) that contains a collection of rules, which can be used to lint other JSON or YAML files such as an API description.

To get started, run this command in your terminal to create a `.spectral.yaml` file that uses the Spectral predefined rulesets based on OpenAPI, Arazzo or AsyncAPI:

```bash
echo 'extends: ["spectral:oas", "spectral:asyncapi", "spectral:arazzo"]' > .spectral.yaml
```

If you would like to create your own rules, check out the [Custom Rulesets](https://meta.stoplight.io/docs/spectral/01baf06bdd05a-rulesets) page.

### 2. Lint

Use this command if you have a ruleset file in the same directory as the documents you are linting:

```bash
spectral lint myapifile.yaml
```

Use this command to lint with a custom ruleset, or one that's located in a different directory than the documents being linted:

```bash
spectral lint myapifile.yaml --ruleset myruleset.yaml
```

## 📖 Documentation

- [Documentation](https://meta.stoplight.io/docs/spectral/docs/getting-started/1-concepts.md)
  - [Getting Started](https://meta.stoplight.io/docs/spectral/docs/getting-started/1-concepts.md) - The basics of Spectral.
  - [Rulesets](https://meta.stoplight.io/docs/spectral/01baf06bdd05a-rulesets) - Understand the structure of a ruleset so you can tweak and make your own rules.

Once you've had a look through the getting started material, some of these guides can help you become a power user.

- [Different Workflows](https://meta.stoplight.io/docs/spectral/docs/guides/1-workflows.md) - When and where should you use Spectral? Editors, Git hooks, continuous integration, GitHub Actions, wherever you like!
- [Using the command-line interface](https://meta.stoplight.io/docs/spectral/docs/guides/2-cli.md) - Quickest way to get going with Spectral is in the CLI.
- [Using the JavaScript API](https://meta.stoplight.io/docs/spectral/docs/guides/3-javascript.md) - Access the _raw power_ of Spectral via the JS, or hey, TypeScript if you want.
- [Custom Rulesets](https://meta.stoplight.io/docs/spectral/docs/guides/4-custom-rulesets.md) - Need something more than the core rulesets provide? Fancy building your own API Style Guide? Learn how to create a custom ruleset.
- [Custom Functions](https://meta.stoplight.io/docs/spectral/docs/guides/5-custom-functions.md) - Handle more advanced rules, by writing a little JavaScript/TypeScript and calling it as a function.

## ℹ️ Support

If you need help using Spectral or have any questions, you can use [GitHub Discussions](https://github.com/stoplightio/spectral/discussions), or visit the [Stoplight Community Forum](https://community.stoplight.io/). These communities are a great place to share your rulesets, or show off tools that use Spectral.

If you have a bug or feature request, [create an issue for it](https://github.com/stoplightio/spectral/issues).

## 🌎 Real-World Rulesets

Stoplight has a set of Spectral rulesets that were created to help users get started with Stoplight's Style Guides. You can find them on [API Stylebook](https://apistylebook.stoplight.io/), and you can download the source Spectral file by selecting a style guide on the project sidebar and selecting **Export** -> **Spectral File(s)** on the top-right. A few noteworthy style guides are:

- [OWASP Top 10](https://apistylebook.stoplight.io/docs/owasp-top-10) - Set of rules to enforce [OWASP security guidelines](https://owasp.org/www-project-api-security/).
- [URL Style Guidelines](https://apistylebook.stoplight.io/docs/url-guidelines) - Set of rules to help developers make better and consistent endpoints.
- [Documentation](https://github.com/stoplightio/spectral-documentation) - Scan an OpenAPI description to make sure you're leveraging enough of its features to help documentation tools like Stoplight Elements, ReDoc, and Swagger UI build the best quality API Reference Documentation possible.

There are also rulesets created by many companies to improve their APIs. You can use these as is to lint your OpenAPI descriptions, or use these as a reference to learn more about what rules you would want in your own ruleset:

- [Adidas](https://github.com/adidas/api-guidelines/blob/master/.spectral.yml) - Adidas were one of the first companies to release their API Style Guide in a written guide _and_ a Spectral ruleset. Lots of good rules to try in here.
- [APIs You Won't Hate](https://github.com/apisyouwonthate/style-guide) - An opinionated collection of rules based on advice in the [APIs You Won't Hate](https://apisyouwonthate.com/) community.
- [Azure](https://github.com/Azure/azure-api-style-guide/blob/main/spectral.yaml) - Ruleset and complimentary style guide for creating OpenAPI 2 or 3 definitions of Azure services.
- [Box](https://github.com/box/box-openapi/blob/main/.spectral.yml) - Lots of [Custom Functions](https://meta.stoplight.io/docs/spectral/ZG9jOjI1MTkw-custom-functions) being used to enforce good practices that the Box API governance folks are interested in.
- [DigitalOcean](https://github.com/digitalocean/openapi/blob/main/spectral/ruleset.yml) - Keeping their OpenAPI nice and tidy, enforcing use of `$ref` (probably to minimize conflicts), naming conventions for Operation IDs, and all sorts of other handy OpenAPI tips.
- [Tranascom](https://github.com/transcom/mymove/blob/master/swagger-def/.spectral.yml) - Don't even think about using anything other than `application/json`.
- [Zalando](https://apistylebook.stoplight.io/docs/zalando-restful-api-guidelines) - Based on [Zalando's RESTFUL API Guidelines](https://github.com/zalando/restful-api-guidelines), covers a wide-range of API topics such as versioning standards, property naming standards, the default format for request/response properties, and more.

Check out some additional style guides here:

- [Spectral Rulesets by Stoplight](https://github.com/stoplightio/spectral-rulesets)
- [API Stylebook by Stoplight](https://apistylebook.stoplight.io)

## ⚙️ Integrations

- [GitHub Action](https://github.com/stoplightio/spectral-action) - Lints documents in your repo, built by [Vincenzo Chianese](https://github.com/XVincentX/).
- [JetBrains Plugin](https://plugins.jetbrains.com/plugin/18520-spectral) - Automatic linting of your OpenAPI specifications and highlighting in your editor.
- [Stoplight Studio](https://stoplight.io/studio?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_spectral) - Uses Spectral to validate and lint OpenAPI documents.
- [VS Code Spectral Extension](https://marketplace.visualstudio.com/items?itemName=stoplight.spectral) - All the power of Spectral without leaving VS Code.

## 🏁 Help Others Utilize Spectral

If you're using Spectral for an interesting use case, create an issue with details on how you're using it. We'll add it to a list here. Spread the goodness 🎉

## 👏 Contributing

If you are interested in contributing to Spectral, check out [CONTRIBUTING.md](CONTRIBUTING.md).

## 🎉 Thanks

- [Mike Ralphson](https://github.com/MikeRalphson) for kicking off the Spectral CLI and his work on Speccy
- [Jamund Ferguson](https://github.com/xjamundx) for JUnit formatter
- [Sindre Sorhus](https://github.com/sindresorhus) for Stylish formatter
- [Ava Thorn](https://github.com/amthorn) for the Pretty formatter
- Julian Laval for HTML formatter
- [@nulltoken](https://github.com/nulltoken) for a whole bunch of amazing features

## 📜 License

Spectral is 100% free and open-source, under [Apache License 2.0](LICENSE).

## 🌲 Sponsor Spectral by Planting a Tree

If you would like to thank Stoplight for creating Spectral, [**buy the world a tree**][stoplight_forest].

[stoplight_forest]: https://ecologi.com/stoplightinc
 readmeEtag: '"f20ce650a5581991dfb549a95800dfeea7489b74"' readmeLastModified: Fri, 07 Mar 2025 13:53:46 GMT repositoryId: 144890632 description: >- A flexible JSON/YAML linter for creating automated style guides, with baked in support for OpenAPI (v3.1, v3.0, and v2.0), Arazzo v1.0, as well as AsyncAPI v2.x. created: '2018-08-15T18:43:18Z' updated: '2026-02-05T12:49:17Z' language: TypeScript archived: false stars: 3000 watchers: 32 forks: 273 owner: stoplightio logo: https://avatars.githubusercontent.com/u/10767217?v=4 license: Apache-2.0 repoEtag: '"f371d39e339fb60af9629122a8a8fbfcdde20af53bc4221640b62f199be5d43b"' repoLastModified: Thu, 05 Feb 2026 12:49:17 GMT foundInMaster: true id: fab3cc58898cd8953a30e43cc622e0fe - source: https://openapi.tools/ name: OpenAPI Style Validator category: Description Validators repository: https://github.com/openapitools/openapi-style-validator language: - Java - CLI source_description: > A customizable style validator to make sure your OpenAPI description follows your organization's standards. v2: true v3: true repositoryMetadata: base64Readme: >- ## This project is looking for maintainers. Please refer to the [announcement](https://github.com/OpenAPITools/openapi-style-validator/issues/783) for more information.

# OpenApi Tools: OpenAPI Style Validator

![](website/static/img/logo.png)

A customizable style validator to make sure your OpenApi spec follows your organization's standards.

[![](https://img.shields.io/badge/Buy%20us%20a%20tree-%F0%9F%8C%B3-lightgreen)](https://offset.earth/darkjaff)

![Build Status](https://github.com/OpenAPITools/openapi-style-validator/actions/workflows/build.yml/badge.svg) [![Maven Central](https://img.shields.io/maven-central/v/org.openapitools.openapistylevalidator/openapi-style-validator-cli)](https://mvnrepository.com/artifact/org.openapitools.openapistylevalidator) [![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=OpenAPITools_openapi-style-validator&metric=alert_status)](https://sonarcloud.io/dashboard?id=OpenAPITools_openapi-style-validator) [![Coverage](https://sonarcloud.io/api/project_badges/measure?project=OpenAPITools_openapi-style-validator&metric=coverage)](https://sonarcloud.io/dashboard?id=OpenAPITools_openapi-style-validator)

[![Join the Slack chat room](https://img.shields.io/badge/Slack-Join%20the%20chat%20room-orange)](https://join.slack.com/t/openapi-generator/shared_invite/enQtNzAyNDMyOTU0OTE1LTY5ZDBiNDI5NzI5ZjQ1Y2E5OWVjMjZkYzY1ZGM2MWQ4YWFjMzcyNDY5MGI4NjQxNDBiMTlmZTc5NjY2ZTQ5MGM) [![Website and documentation](https://img.shields.io/badge/Website%20and%20Documentation-Click%20to%20View-green)](https://openapitools.github.io/openapi-style-validator/) [![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod)](https://gitpod.io/#https://github.com/OpenAPITools/openapi-style-validator)

## Summary

In the collaborative world of API development, ensuring consistency and adherence to standards in your OpenAPI (YAML) specification file is a paramount challenge. Even if you've meticulously documented directives and guidelines in a shared PDF or a company wiki, there's no guarantee that every team member will strictly follow them.

One could argue that manual code reviews may catch these deviations, but what if there were a more efficient way to ensure conformity? Enter openapi-style-validator, a tool designed to automate this process. While you may still conduct code reviews to address business logic, requirements, and REST path issues, openapi-style-validator excels at validating the style and adherence to standards, including mandatory examples, naming conventions, descriptions, and more. By integrating this powerful tool into your workflow, you can effortlessly elevate the quality and consistency of your OpenAPI files, leaving you with more time to focus on the critical aspects of your API development.

## :trophy: Sponsors

Thanks to our sponsors for supporting this project!

### :1st_place_medal: Gold

[Select your sponsor tier](https://github.com/sponsors/JFCote) and get your BIG logo here with a link to your website.

### :2nd_place_medal: Silver

[Select your sponsor tier](https://github.com/sponsors/JFCote) and get your logo here with a link to your website.

### :3rd_place_medal: Bronze

[Select your sponsor tier](https://github.com/sponsors/JFCote) and get your company name here with a link to your website.

## What it can validate

### API Info
- API license must be present and non-empty string
- API description must be present and non-empty string
- API contact must be present and non-empty string

### Operations
- OperationId must be present and non-empty string
- Operation description must be present and non-empty string
- Operation must have a tag and non-empty string
- Operation summary must be present and non-empty string

### Models
- All model properties must have examples
- All model properties must have descriptions
- All required properties must be listed among the properties

### Naming convention
- Enforce naming convention for paths, parameters (path, query and cookie), headers and properties
  - underscore_case
  - UNDERSCORE_UPPER_CASE
  - camelCase
  - hyphen-case
  - Hyphen-Upper-Case

## Paid Alternatives

This project was started because when I tried SwaggerHub (which is the paid version of the swagger toolkit), they had
something similar, but it was proprietary. I didn't want to pay for the service so I created this project in my free
times. I did not validate lately if they still have this feature.

## How to use the style validator

For now, the project is a simple command line interface (CLI) and a library. The easiest way to use it right now
is to use the CLI and check the output. It will list all errors found based on the options you provided.

### To build

The project is configured to use gradle. To build the jar, just do:

`gradlew assemble`

or if you want to invoke the jar creation directly

`gradlew shadowJar`

### To change the code

If you want to open the project, I highly suggest that you use IntelliJ IDEA Community (Free) or Ultimate (paid).
This IDE is cross platform so it should work on any OS. This project was created using this tool and you can just open it directly. Maybe you can use other tools to open the project but I will not provide any support.

### To launch

`java -jar openapi-style-validator.jar -s ./path/to/spec.yaml -o ./path/to/options.json`

Example using the default output path for the jar (replace `<version>` with the current version):

`java -jar modules/cli/build/libs/openapi-style-validator-cli-<version>-all.jar -s specs/petstore.yaml -o specs/options.json`

#### Command Line

|Parameter|Required?|Description|
|---|---|---|
|-s, -source|yes|The path to your json/yaml spec file|
|-o, -options|no|The path to your json options file|

#### Options File
The options file is described in json (example in `specs/options.json`), and has the following possible values:

|Option|Type| Possible Values                                                                                                |Default Value|Description|
|---|---|----------------------------------------------------------------------------------------------------------------|---|---|
|validateInfoLicense|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is a license section in the info section|
|validateInfoDescription|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is a description attribute in the info section|
|validateInfoContact|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is a contact section in the info section|
|validateOperationOperationId|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is an operation id for each operation|
|validateOperationDescription|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is a description for each operation|
|validateOperationTag|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is a tag for each operation|
|validateOperationSummary|boolean| `true`, `false`                                                                                                |`true`|Ensures that there is a summary for each operation|
|validateModelPropertiesExample|boolean| `true`, `false`                                                                                                |`true`|Ensures that the properties of the Schemas have an example value defined|
|validateModelPropertiesDescription|boolean| `true`, `false`                                                                                                |`true`|Ensures that the properties of the Schemas have a description value defined|
|validateModelRequiredProperties|boolean| `true`, `false`                                                                                                |`true`|Ensures that all required properties of the Schemas are listed among their properties|
|validateModelNoLocalDef|boolean| `true`, `false`                                                                                                |`true`|Not implemented yet|
|validateNaming|boolean| `true`, `false`                                                                                                |`true`|Ensures the names follow a given naming convention|
|ignoreHeaderXNaming|boolean| `true`, `false`                                                                                                |`true`|Exclude from validation header parameters starting with `x-`|
|pathNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |`HyphenCase`|Naming convention for paths|
|parameterNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |`CamelCase`|Global naming convention for all parameter types (path, query and cookie) [(note)](#parameter-naming-convention-hierarchy)|
|pathParamNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |(same as `parameterNamingConvention`)|Specific naming convention for path parameters [(note)](#parameter-naming-convention-hierarchy)|
|queryParamNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |(same as `parameterNamingConvention`)|Specific naming convention for query parameters [(note)](#parameter-naming-convention-hierarchy)|
|cookieParamNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |(same as `parameterNamingConvention`)|Specific naming convention for cookie parameters [(note)](#parameter-naming-convention-hierarchy)|
|headerNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |`UnderscoreUpperCase`|Naming convention for headers|
|schemaNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |`PascalCase`|Naming convention for schemas|
|propertyNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |`CamelCase`|Naming convention for properties|
|enumNamingConvention|string| `CamelCase`, `PascalCase`, `HyphenUpperCase`, `HyphenCase`, `UnderscoreCase`, `UnderscoreUpperCase`, `AnyCase` |`AnyCase`|Naming convention for enums|
|allowedModelProperties|array| `["_links", "_embedded"]`|`[]`|An array of property names that are authorized even if they do not match to the "propertyNamingConvention"|

#### Parameter Naming Convention hierarchy

The Parameter Naming Convention can be defined by a combination of the following options in the `options.json` file:
- parameterNamingConvention
- pathParamNamingConvention
- queryParamNamingConvention
- cookieParamNamingConvention

The first option (parameterNamingConvention) defines a global convention for all other parameter naming convention options (path, query and cookie), unless they are explicitly defined in the `options.json` file.

In other words, the `pathParamNamingConvention`, `queryParamNamingConvention` and `cookieParamNamingConvention` are specific for the parameter type their names indicate. Thus, they take precedence over `parameterNamingConvention` when they are defined. If not defined, the undefined option is assigned the same value for `parameterNamingConvention` (either explicitly defined or default).

Example 1:
```json
{
  /*
    'parameterNamingConvention' is not defined.
        Then it assumes the 'CamelCase' convention.
   */
  "pathParamNamingConvention": "UnderscoreCase",
  "queryParamNamingConvention": "HyphenCase"
  /*
    'cookieParamNamingConvention' is not defined.
        Then it assumes the same value as 'parameterNamingConvention',
        which is its default value ('CamelCase').
   */
}
```

Example 2:
```json
{
  "parameterNamingConvention": "HyphenCase",
  "pathParamNamingConvention": "UnderscoreCase",
  "queryParamNamingConvention": "HyphenCase"
  /*
    'cookieParamNamingConvention' is not defined.
        Then it assumes the same value as 'parameterNamingConvention',
        which is explicitly defined as 'HyphenCase'.
   */
}
```

## Supported Extensions

We sometimes have some paths in our API that are there for legacy reasons. To support this scenario, there is now the possibility to add an extension property in your spec named `x-style-validator-ignored`. Here is an example:

```yaml
/my-pathWithWeird_NAMING-FoRLeGaCyReAsOnSLOL:
  x-style-validator-ignored: true
  post:
    tags:
      - Whatever
    summary: A summary
    /* missing operating id or other important property */
    responses:
      200:
        description: OK
  get:
    ...
```

The path (GET, POST, etc..) will be completely ignored by the style validator.

## Roadmap

To view the latest version of the roadmap, please see the [official roadmap](https://openapitools.github.io/openapi-style-validator/docs/roadmap/).

## Releases on maven central

The different components of this project are released on Maven Central:

The core library:

```xml
<dependency>
  <groupId>org.openapitools.openapistylevalidator</groupId>
  <artifactId>openapi-style-validator-lib</artifactId>
  <version>${validatorVersion}</version>
</dependency>
```

The command line tool (without dependency):

```xml
<dependency>
  <groupId>org.openapitools.openapistylevalidator</groupId>
  <artifactId>openapi-style-validator-cli</artifactId>
  <version>${validatorVersion}</version>
</dependency>
```

The command line tool is also available as standalone jar (containing its dependencies):

```xml
<dependency>
  <groupId>org.openapitools.openapistylevalidator</groupId>
  <artifactId>openapi-style-validator-cli</artifactId>
  <version>${validatorVersion}</version>
  <classifier>all</classifier>
</dependency>
```

You can download it manually (replace the `<version>` placeholder in following URL):

```
https://repo1.maven.org/maven2/org/openapitools/openapistylevalidator/openapi-style-validator-cli/<version>/openapi-style-validator-cli-<version>-all.jar
```
 readmeEtag: '"5c7194d8ca9fa58fcd321c6e721ef3283aa0252a"' readmeLastModified: Mon, 17 Nov 2025 05:42:16 GMT repositoryId: 101086314 description: >- A customizable style validator to make sure your OpenAPI spec follows your organization's standards. created: '2017-08-22T16:53:13Z' updated: '2026-01-21T01:08:17Z' language: Java archived: false stars: 232 watchers: 7 forks: 47 owner: OpenAPITools logo: https://avatars.githubusercontent.com/u/37325267?v=4 license: Apache-2.0 repoEtag: '"762a1e0221bd5a30044302f9ea5d7a0797735d85972fec09f8d816188b3c9b22"' repoLastModified: Wed, 21 Jan 2026 01:08:17 GMT foundInMaster: true id: 30d322c922366df3f6f8fb9c681b2139 - source: https://openapi.tools/ name: OpenAPI Validator category: Description Validators repository: https://github.com/ibm/openapi-validator language: Node.js source_description: Configurable and extensible validator/linter for OpenAPI documents v2: true v3: true repositoryMetadata: base64Readme: >- [![Build Status](https://github.com/IBM/openapi-validator/actions/workflows/build.yaml/badge.svg)](https://github.com/IBM/openapi-validator/actions/workflows/build.yaml)
[![npm-version](https://img.shields.io/npm/v/ibm-openapi-validator.svg)](https://www.npmjs.com/package/ibm-openapi-validator)
[![semantic-release](https://img.shields.io/badge/%20%20%F0%9F%93%A6%F0%9F%9A%80-semantic--release-e10079.svg)](https://github.com/semantic-release/semantic-release)
[![Gitter](https://badges.gitter.im/openapi-validator/community.svg)](https://gitter.im/openapi-validator/community?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge)
[![Commitizen friendly](https://img.shields.io/badge/commitizen-friendly-brightgreen.svg)](http://commitizen.github.io/cz-cli/)
[![CLA assistant](https://cla-assistant.io/readme/badge/ibm/openapi-validator)](https://cla-assistant.io/ibm/openapi-validator)

# OpenAPI Validator
The IBM OpenAPI Validator lets you validate [OpenAPI 3.0.x](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md)
and [OpenAPI 3.1.x](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.1.0.md) documents for compliance with the OpenAPI specifications, as well as IBM-defined best practices.

#### Prerequisites
- Node 16.0.0+
- NPM 8.3.0+

## Table of contents

<!--
  The TOC below is generated using the `markdown-toc` node package.

      https://github.com/jonschlinkert/markdown-toc

  You should regenerate the TOC after making changes to this file.

      npm run format-readme-toc
  -->

<!-- toc -->

- [Getting Started](#getting-started)
  * [Ruleset](#ruleset)
  * [Customization](#customization)
- [Installation](#installation)
  * [Install with NPM (recommended)](#install-with-npm-recommended)
  * [Download an executable binary](#download-an-executable-binary)
  * [Build from source](#build-from-source)
    + [Build platform-specific binaries](#build-platform-specific-binaries)
  * [Container image](#container-image)
    + [Building your own](#building-your-own)
- [Usage](#usage)
  * [Command Syntax](#command-syntax)
  * [Configuration](#configuration)
    + [Configuration Properties](#configuration-properties)
      - [colorizeOutput](#colorizeoutput)
      - [errorsOnly](#errorsonly)
      - [files](#files)
      - [ignoreFiles](#ignorefiles)
      - [limits](#limits)
      - [logLevels](#loglevels)
      - [outputFormat](#outputformat)
      - [ruleset](#ruleset)
      - [summaryOnly](#summaryonly)
      - [produceImpactScore](#produceimpactscore)
      - [markdownReport](#markdownreport)
  * [Programmatic Usage](#programmatic-usage)
- [Validator Output](#validator-output)
  * [Text](#text)
    + [Additional Customization](#additional-customization)
  * [JSON](#json)
- [Logging](#logging)
- [Contributing](#contributing)
- [License](#license)

<!-- tocstop -->

## Getting Started
The validator analyzes your API definition and reports any problems within. The validator is highly customizable,
and supports OpenAPI 3.0.x and 3.1.x documents.
The tool also supports a number of rules from [Spectral](https://stoplight.io/open-source/spectral/). You can easily extend the tool with custom rules to meet your specific needs and ensure compliance to your standards.

Get started by [installing the tool](#installation), then [run the tool](#usage) on your API definition.

### Ruleset
By default, the validator will use the [IBM Cloud Validation Ruleset](docs/ibm-cloud-rules.md) (npm package `@ibm-cloud/openapi-ruleset`).
However, if the validator detects the presence of any of the standard Spectral ruleset files (`spectral.yaml`, `spectral.yml`, `spectral.json`,
or `spectral.js`) in the current directory (from which the validator is being run) or in any containing directory within the file system,
then that ruleset file will be used instead.
To explicitly specify an alternate ruleset, you can use the `-r`/`--ruleset` option (or the `ruleset` configuration property)
to specify the name of your custom ruleset file.

If one of the standard Spectral ruleset files are present and you'd like to force the use of the IBM Cloud Validation Ruleset instead,
you can use `-r default` or `--ruleset default` (or set the `ruleset` configuration property to the value `'default'`).

Details about these options are provided below in the [Usage](#usage) section.

### Customization

You can modify the behavior of the validator for your project to meet your preferred standards. See the [customization documentation](docs/ibm-cloud-rules.md#customization) for more information.

## Installation
There are three ways to install the validator: using NPM, downloading a platform-specific binary, or building from source.

### Install with NPM (recommended)

`npm install -g ibm-openapi-validator`

The `-g` flag installs the tool globally so that the validator can be run from anywhere in the file system. Alternatively, you can pass no flag or the `--save-dev` flag to add the validator as a dependency to your project and run it from your NPM scripts or JavaScript code.

### Download an executable binary

Platform-specific binary files are packaged with each release for MacOS, Linux, and Windows. See [the releases](https://github.com/IBM/openapi-validator/releases) page to download the executable for your platform. These do not depend on Node.JS being installed.

### Build from source
1. Clone or download this repository
2. Navigate to the root directory of this project.
3. Install the dependencies using `npm install`
4. Build the command line tool by running `npm run link`.

_If you installed the validator using `npm install -g ibm-openapi-validator`, you will need to run `npm uninstall -g ibm-openapi-validator` before running `npm run link`._

#### Build platform-specific binaries
It is also possible to build platform specific binaries from the source code by running `npm run pkg` in the project root directory.  The binaries (lint-openapi-macos, lint-openapi-linux, lint-openapi-windows.exe) will be in the project's `packages/validator/bin` directory.

### Container image

Run the validator with the container image by mounting your API definition.

If it is named `openapi.yaml` in the current directory, then run:
```bash
docker run \
  --rm --tty \
  --volume "$PWD:/data:ro" \
  ibmdevxsdk/openapi-validator:latest \
    openapi.yaml
```

You should **replace `latest` with a specific tagged version** to avoid any surprises when new releases are published.

Flag and argument syntax is the same as described in [Usage](#usage), but file paths are relative to `/data`.

To use a custom ruleset named `ruleset.yaml` in the current directory, run:
```bash
docker run \
  --rm --tty \
  --volume "$PWD:/data:ro" \
  ibmdevxsdk/openapi-validator:latest \
    --ruleset ruleset.yaml \
    openapi.yaml
```

#### Building your own

If the existing image doesn't suit your needs, you could extend it and build your own.

For example, to build a validator image with your own custom ruleset package installed, make a `Dockerfile` like this:

```Dockerfile
FROM ibmdevxsdk/openapi-validator:latest
RUN npm install -g ${your-ruleset-pkg-here}
```

## Usage
### Command Syntax
```bash
Usage: lint-openapi [options] [file...]

Run the validator on one or more OpenAPI 3.x documents

Options:
  -c, --config <file>            use configuration stored in <file> (*.json, *.yaml, *.js)
  -e, --errors-only              include only errors in the output and skip warnings (default is false)
  -i, --ignore <file>            avoid validating <file> (e.g. -i /dir1/ignore-file1.json --ignore /dir2/ignore-file2.yaml ...) (default is []) (default: [])
  -j, --json                     produce JSON output (default is text)
  -l, --log-level <loglevel>     set the log level for one or more loggers (e.g. -l root=info -l ibm-schema-description-exists=debug ...)  (default: [])
  -n, --no-colors                disable colorizing of the output (default is false)
  -r, --ruleset <file>           use Spectral ruleset contained in `<file>` ("default" forces use of default IBM Cloud Validation Ruleset)
  -s, --summary-only             include only the summary information and skip individual errors and warnings (default is false)
  -q, --impact-score             compute scores representing the API impact of rule violations and include with the results (default is false)
  -m, --markdown-report          generate a Markdown file with a report on all validator results (default is false)
  -w, --warnings-limit <number>  set warnings limit to <number> (default is -1)
  --version                      output the version number
  -h, --help                     display help for command
```
where `[file...]` is a space-separated list containing the filenames of one or more OpenAPI 3.x documents to be validated.
The validator supports OpenAPI documents in either JSON or YAML format, using these supported file extensions:
```
.json
.yaml
.yml
```
Assuming your command shell supports the use of wildcards, you can use wildcards when specifying the names of files to be validated.
For example, to run the validator on all `.yaml` files contained in the `/my/apis` directory, you could use
this command:
```bash
lint-openapi /my/apis/*.yaml
```

Note that the `-i`/`--ignore` option can be particularly useful when using wildcards because it allows you to skip the
validation of specific files which might otherwise be included in a validation run.
For example, to validate all `.yaml` files in the `/my/apis` directory, except for `/my/apis/broken-api.yaml` use the command:
```bash
lint-openapi /my/apis/*.yaml -i /my/apis/broken-api.yaml
```

### Configuration
In addition to command-line options, the validator supports the use of a configuration file containing options as well.
A configuration file can be in JSON, YAML or Javascript format, using these supported extensions: `.json`, `.yaml`, `.yml`, and `.js`.
Its structure must comply with [this JSON schema](packages/validator/src/schemas/config-file.yaml).

You can specify the name of your configuration file with the `-c`/`--config` option.  Here's an example:
```bash
lint-openapi -c my-config.yaml my-api.json
```
where `my-config.yaml` might contain the following:
```yaml
errorsOnly: true
limits:
  warnings: 25
outputFormat: 'json'
summaryOnly: true
```
This would be equivalent to this command:
```bash
lint-openapi --errors-only --warnings-limit=25 --json --summary-only my-api.json
```
When using both a configuration file and various command-line options, be aware that the options specified
via the command-line will take precendence and override any corresponding properties specified in the configuration file.

#### Configuration Properties
This section provides information about each of the properties that are supported within a configuration file.

##### colorizeOutput
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>colorizeOutput</code> configuration property corresponds to the <code>-n</code>/<code>--no-colors</code>
command-line option. If set to true, then the validator will colorize its output.</td>
<td><code>true</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
colorizeOutput: false
</pre>
</td>
<td>
<pre>
{
  "colorizeOutput": false
}
</pre>
</td>
<td>
<pre>
module.exports = {
  colorizeOutput: false
};
</pre>
</td>
</tr>
</table>

##### errorsOnly
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>errorsOnly</code> configuration property corresponds to the <code>-e</code>/<code>--errors-only</code> command-line option.
If set to <code>true</code>, the validator will include only errors in its output, while messages of severity warning,
info or hint will be skipped.</td>
<td><code>false</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
errorsOnly: true
</pre>
</td>
<td>
<pre>
{
  "errorsOnly": true
}
</pre>
</td>
<td>
<pre>
module.exports = {
  errorsOnly: true
};
</pre>
</td>
</tr>
</table>

##### files
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>files</code> configuration property corresponds to positional command-line arguments (i.e. <code>[file...]</code>).
You can set this property to the names of the OpenAPI documents to be validated. If any filenames are also entered as positional arguments
on the command-line, they will override any values specified in this configuration property.</td>
<td><code>[]</code>(empty list)</td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
files:
  - file1.json
  - file2.yaml
</pre>
</td>
<td>
<pre>
{
  "files": [
    "file1.json",
    "file2.yaml"
  ]
}
</pre>
</td>
<td>
<pre>
module.exports = {
  files: [
    'file1.json',
    'file2.yaml'
  ]
};
</pre>
</td>
</tr>
</table>

##### ignoreFiles
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>ignoreFiles</code> configuration property corresponds to the <code>-i</code>/<code>--ignore</code> command-line option.
Set this property to the fully-qualified filenames of OpenAPI documents to be excluded from validation.
This property can be particularly useful when using wildcards for specifying the OpenAPI documents to be validated,
because it allows you to skip the validation of specific files which might otherwise be included in a validation run.
For example, to validate all <code>.yaml</code> files in the <code>/my/apis</code> directory, except for
<code>/my/apis/broken-api.yaml</code> use the command:
<pre>
    lint-openapi /my/apis/*.yaml --ignore /my/apis/broken-api.yaml
</pre>
</td>
<td><code>[]</code>(empty list)</td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
ignoreFiles:
  - /my/apis/file1.yml
</pre>
</td>
<td>
<pre>
{
  "ignoreFiles": [
    "/my/apis/file1.yml"
  ]
}
</pre>
</td>
<td>
<pre>
module.exports = {
  ignoreFiles: [
    '/my/apis/file1.yml'
  ]
};
</pre>
</td>
</tr>
</table>

##### limits
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>limits</code> configuration property corresponds to the <code>-w</code>/<code>--warnings-limit</code> command-line option.
Use this property to set the warnings limit.  When validating an OpenAPI document, the validator will compare the number of warnings
it encounters with the warnings limit.  If the number of warnings exceeds the limit, then an error will be logged and the
validator will return exitCode 1, similar to if actual errors were found. If the warnings limit is set to a negative number,
then no warnings limit check will be performed by the validator.</td>
<td><code>{ warnings: -1 }</code>(warnings limit check disabled)</td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
limits:
  warnings: 25
</pre>
</td>
<td>
<pre>
{
  "limits": {
    "warnings": 25
  }
}
</pre>
</td>
<td>
<pre>
module.exports = {
  limits: {
    warnings: 25
  }
};
</pre>
</td>
</tr>
</table>

##### logLevels
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>logLevels</code> property is an object that specifies the logging level
(<code>error</code>, <code>warn</code>, <code>info</code>, or <code>debug</code>)
associated with each logger within the validator.  It corresponds to the <code>-l</code>/<code>--log-level</code> command-line option.</td>
<td><code>{ root: 'info' }</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
logLevels:
  root: error
  ibm-schema-description-exists: debug
</pre>
</td>
<td>
<pre>
{
  "logLevels": {
    "root": "error",
    "ibm-schema-description-exists": "debug"
  }
}
</pre>
</td>
<td>
<pre>
module.exports = {
  logLevels: {
    root: 'error',
    'ibm-schema-description-exists': 'debug'
  }
};
</pre>
</td>
</tr>
</table>

##### outputFormat
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>You can set the <code>outputFormat</code> configuration property to either <code>text</code> or <code>json</code>
to indicate the type of output you want the validator to produce.
This property corresponds to the <code>-j</code>/<code>--json</code> command-line option.</td>
<td><code>text</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
outputFormat: json
</pre>
</td>
<td>
<pre>
{
  "outputFormat": "json"
}
</pre>
</td>
<td>
<pre>
module.exports = {
  outputFormat: 'json'
};
</pre>
</td>
</tr>
</table>

##### ruleset
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>You can use the <code>ruleset</code> configuration property to specify a custom ruleset to be used by the validator.
This corresponds to the <code>-r</code>/<code>--ruleset</code> command-line option.

By default, the validator will look for the standard Spectral ruleset files (<code>.spectral.yaml</code>, <code>.spectral.yml</code>,
<code>.spectral.json</code>, or <code>.spectral.js</code>) in the current working directory and
its parent directories within the filesystem.  If none are found, then the IBM Cloud Validation Ruleset
will be used.

If you want to force the use of the IBM Cloud Validation Ruleset even if one of the standard Spectral ruleset files are present,
you can specify <code>'default'</code> for the <code>ruleset</code> configuration property.
</td>
<td><code>null</code>, which implies that a standard Spectral ruleset file will be used (if present),
otherwise the IBM Cloud Validation Ruleset will be used.</td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
ruleset: my-custom-rules.yaml
</pre>
</td>
<td>
<pre>
{
  "ruleset": "my-custom-rules.yaml"
}
</pre>
</td>
<td>
<pre>
module.exports = {
  ruleset: 'my-custom-rules.yaml'
};
</pre>
</td>
</tr>
</table>

##### summaryOnly
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>summaryOnly</code> configuration property corresponds to the <code>-s</code>/<code>--summary-only</code> command-line option.
If set to true, the validator will include only the summary section in its output.
</td>
<td><code>false</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
summaryOnly: true
</pre>
</td>
<td>
<pre>
{
  "summaryOnly": true
}
</pre>
</td>
<td>
<pre>
module.exports = {
  summaryOnly: true
};
</pre>
</td>
</tr>
</table>

##### produceImpactScore
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>produceImpactScore</code> configuration property corresponds to the
<code>-q</code>/<code>--impact-score</code> command-line option. If set to true,
the validator will, in addition to reporting individual rule violations, use the
rule violation data to produce API impact scores based on the categories of usability,
security, robustness, and cost of evolution. By default, the data demonstrating how
the scores are calculated from each rule is displayed. If this option is combined with
the "summary only" configuration option, only the categorized impact scores are displayed.
These scores are useful for "Automated Quality Screening". See [this documentation](docs/automated-quality-screening.md)
for more information about the purpose of these scores and how they are computed.
</td>
<td><code>false</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
produceImpactScore: true
</pre>
</td>
<td>
<pre>
{
  "produceImpactScore": true
}
</pre>
</td>
<td>
<pre>
module.exports = {
  produceImpactScore: true
};
</pre>
</td>
</tr>
</table>

##### markdownReport
<table border=1>
<tr>
<td><b>Description</b></td>
<td width=25%><b>Default</b></td>
</tr>
<tr>
<td>The <code>markdownReport</code> configuration property corresponds to the
<code>-m</code>/<code>--markdown-report</code> command-line option.
If set to true, the validator will generate a Markdown file containing a report on all of the validator results,
including the individual rule violations, the impact scores, and the data used to compute the impact scores.
It provides a single location to see all of the information produced by the validator.
A default filename is always used and is based on the name of the API definition file provided to the validator.
If a file of the same name already exists, it will be overwritten by default. This is because the primary use-case
of this feature is publishing reports during CI/CD builds, where a single report will need to be updated frequently.
</td>
<td><code>false</code></td>
</tr>
</table>
<b>Examples:</b>
<table border=1>
<tr>
</tr>
<tr>
<td><code>.yaml/.yml</code></td>
<td><code>.json</code></td>
<td><code>.js</code></td>
</tr>
<tr>
<td>
<pre>
markdownReport: true
</pre>
</td>
<td>
<pre>
{
  "markdownReport": true
}
</pre>
</td>
<td>
<pre>
module.exports = {
  markdownReport: true
};
</pre>
</td>
</tr>
</table>

### Programmatic Usage
While the validator does not expose an API for usage within a Node.js program, you can achieve programmatic behavior
consistent with the CLI by using the open-source tool [Spectral's Node API](https://meta.stoplight.io/docs/spectral/eb68e7afd463e-spectral-in-java-script)
and the [IBM OpenAPI Ruleset package](https://www.npmjs.com/package/@ibm-cloud/openapi-ruleset).
Here is a simple example of what that might look like:
```js
const ibmOpenapiRuleset = require('@ibm-cloud/openapi-ruleset');
const { Spectral } = require('@stoplight/spectral-core');

function async runSpectral(openapiDocument) {
  const spectral = new Spectral();
  spectral.setRuleset(ibmOpenapiRuleset);
  results = await spectral.run(openapiDocument);
  console.log(results);
}
```

## Validator Output
The validator can produce output in either text or JSON format.  The default is `text` output, and this can be
controlled with the `-j`/`--json` command-line option or `outputFormat` configuration property.

**NOTE** The text (i.e. "human readable") output is not a part of the tool's contract and may change in a minor
or patch release. If you are building automation or scripts with this tool, use the JSON output, which is stable
and subject to semantic versioning.

### Text
Here is an example of text output:
```
IBM OpenAPI Validator (validator: 0.97.5; ruleset: 0.45.5), @Copyright IBM Corporation 2017, 2023.

Validation Results for /my/directory/my-api.yaml:

Errors:

  Message :   Path contains two or more consecutive path parameter references: /v1/clouds/{cloud_id}/{region_id}
  Rule    :   ibm-no-consecutive-path-parameter-segments
  Path    :   paths./v1/clouds/{cloud_id}/{region_id}
  Line    :   332

Warnings:

  Message :   Operation summaries should not have a trailing period
  Rule    :   ibm-summary-sentence-style
  Path    :   paths./v1/clouds.post.summary
  Line    :   46

  Message :   Operation summaries should not have a trailing period
  Rule    :   ibm-summary-sentence-style
  Path    :   paths./v1/clouds.get.summary
  Line    :   93

Summary:

  Total number of errors   : 1
  Total number of warnings : 2

  Errors:
   1 (100%) : Path contains two or more consecutive path parameter references

  Warnings:
   2 (100%) : Operation summaries should not have a trailing period
```
As you can see, any errors detected by the validator are listed first, then
warnings, and finally a summary section.

#### Additional Customization
- The `-s`/`--summary-only` command-line option or the `summaryOnly` configuration property causes only the summary to be displayed.
- The `-e`/`--errors-only` option or `errorsOnly` configuration property causes only error-level violations to be displayed.
- The `-q`/`--impact-score` option or `produceImpactScore` configuration property causes the validator to show aggregated impact scores. See the example below:

Example of impact score tables that are appended to the standard output:
```
┌────────────────┬───────────┐
│       category │ max score │
├────────────────┼───────────┤
│      usability │   98 /100 │
│       security │  100 /100 │
│     robustness │   97 /100 │
│      evolution │   67 /100 │
│ overall (mean) │   91 /100 │
└────────────────┴───────────┘
┌──────────────────────────────┬───────┬─────────────────┬──────────────────┬─────────────────┬───────────────────┬──────────────────┬────────────┐
│                         rule │ count │            func │ usability impact │ security impact │ robustness impact │ evolution impact │ rule total │
├──────────────────────────────┼───────┼─────────────────┼──────────────────┼─────────────────┼───────────────────┼──────────────────┼────────────┤
│ operation-operationId-unique │     1 │  1×3÷operations │                1 │                 │                 2 │                3 │          6 │
│       ibm-no-array-responses │     2 │ 2×10÷operations │                  │                 │                   │               20 │         20 │
│             no-$ref-siblings │     1 │  1×1÷operations │             0.33 │                 │                   │                  │       0.33 │
└──────────────────────────────┴───────┴─────────────────┴──────────────────┴─────────────────┴───────────────────┴──────────────────┴────────────┘
```

### JSON
When displaying JSON output, the validator will produce a JSON object which complies with
[this JSON schema](packages/validator/src/schemas/results-object.yaml). The JSON data will include information about all rule violations,
as well as all impact score information computed from the rule violations.
Here is an example of JSON output:
```json
{
  "error": {
    "results": [
      {
        "message": "Path contains two or more consecutive path parameter references: /v1/clouds/{cloud_id}/{region_id}",
        "path": [
          "paths",
          "/v1/clouds/{cloud_id}/{region_id}"
        ],
        "rule": "ibm-no-consecutive-path-parameter-segments",
        "line": 332
      }
    ],
    "summary": {
      "total": 1,
      "entries": [
        {
          "generalizedMessage": "Path contains two or more consecutive path parameter references",
          "count": 1,
          "percentage": 100
        }
      ]
    }
  },
  "warning": {
    "results": [
      {
        "message": "Operation summaries should not have a trailing period",
        "path": [
          "paths",
          "/v1/clouds",
          "post",
          "summary"
        ],
        "rule": "ibm-summary-sentence-style",
        "line": 46
      },
      {
        "message": "Operation summaries should not have a trailing period",
        "path": [
          "paths",
          "/v1/clouds",
          "get",
          "summary"
        ],
        "rule": "ibm-summary-sentence-style",
        "line": 93
      }
    ],
    "summary": {
      "total": 2,
      "entries": [
        {
          "generalizedMessage": "Operation summaries should not have a trailing period",
          "count": 2,
          "percentage": 100
        }
      ]
    }
  },
  "info": {
    "results": [],
    "summary": {
      "total": 0,
      "entries": []
    }
  },
  "hint": {
    "results": [],
    "summary": {
      "total": 0,
      "entries": []
    }
  },
  "hasResults": true
  "impactScore": {
    "categorizedSummary": {
      "usability": 94,
      "security": 100,
      "robustness": 100,
      "evolution": 100,
      "overall": 99
    },
    "scoringData": [
      {
        "rule": "ibm-no-consecutive-path-parameter-segments",
        "count": 1,
        "func": "1×10÷operations",
        "demerits": {
          "usability": 3.33,
          "total": 3.33
        }
      },
      {
        "rule": "ibm-summary-sentence-style",
        "count": 2,
        "func": "2×1÷operations",
        "demerits": {
          "usability": 0.67,
          "total": 0.67
        }
      }
    ]
  }
}
```
The JSON output is also affected by the `-s`/`--summary-only` and `-e`/`--errors-only` options as well as the `summaryOnly` and `errorsOnly`
configuration properties. It is _not_ affected by the `-q`/`--impact-score` option or `produceImpactScore` property.

## Logging
The validator uses a *logger* for displaying messages on the console.
The core validator uses a single logger named `root`, while each of the rules contained in the
IBM Cloud Validation Ruleset uses their own unique logger whose name will match the rule's id
(e.g. `ibm-accept-header`, `ibm-schema-description-exists`, etc.).

Each logger has a logging level associated with it: `error`, `warn`, `info`, and `debug`.
Each of these levels implicitly includes the levels that precede it in the list.
For example, if you set the logging level of a logger to `info`, then all messages of type
`info`, `warn`, and `error` will be displayed, but `debug` messages will not.

To set the level of the `root` logger to `info`, you could use this option: `--log-level root=info`.

To set the level of the logger used by the `ibm-accept-header` rule to `debug`,
you could use this option: `-l ibm-accept-header=debug`.

You can also use a glob-like value for a logger name to set the level on multiple loggers.
For example, to set the level for *all* loggers whose name starts with `ibm-property`, try this:
`-l ibm-property*=debug`.

Enabling debug logging for a specific rule might be useful in a situation where the rule is
reporting violations which seem to be inexplicable. In this case, additional debug information
might be helpful in determining why the violations are occurring, and could possibly lead to a solution.
For example, suppose the `ibm-pagination-style` rule is reporting several violations,
but yet at first glance it's not obvious why these violations are occurring.
To enable debug logging for this rule, use a command like this:
```
lint_openapi -l ibm-pagination-style=debug my-new-api.yaml
```

The default log level for the `root` logger is `info`, while the default log level for
rule-specific loggers is `warn`.

## Contributing

See [CONTRIBUTING](CONTRIBUTING.md).

## License

This project is licensed under Apache 2.0.  Full license text is available in [LICENSE](LICENSE).
 readmeEtag: '"b9fd7f5eb697153217217e8c1c99cc446d6370b8"' readmeLastModified: Fri, 10 Jan 2025 22:00:33 GMT repositoryId: 153173690 description: Configurable and extensible validator/linter for OpenAPI documents created: '2018-10-15T19:59:22Z' updated: '2026-02-05T18:07:15Z' language: JavaScript archived: false stars: 610 watchers: 21 forks: 93 owner: IBM logo: https://avatars.githubusercontent.com/u/1459110?v=4 license: Apache-2.0 repoEtag: '"6c3fb24b5b609943caa4f82abb7d07a55050a83e815f9b8b06419ee39072cf26"' repoLastModified: Thu, 05 Feb 2026 18:07:15 GMT foundInMaster: true id: 0bf7dd29a9d07232135bb00764928296 v3_1: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/redocly/redocly-cli v3: true v3_1: true id: dc53a7154c15667d6f54d47cdb300949 repositoryMetadata: base64Readme: >- # Redocly CLI

[@Redocly](https://redocly.com) CLI is your all-in-one API documentation utility.
It builds, manages, improves, and quality-checks your API descriptions, all of which comes in handy for various phases of the API Lifecycle.
Create your own rulesets to make API governance easy, publish beautiful API reference documentation, and more.
Supports OpenAPI 3.2, 3.1, 3.0 and OpenAPI 2.0 (legacy Swagger), AsyncAPI 3.0 and 2.6, Arazzo 1.0.

![build and test](https://github.com/redocly/redocly-cli/actions/workflows/tests.yaml/badge.svg)
![npm (scoped)](https://img.shields.io/npm/v/@redocly/cli)
![NPM](https://img.shields.io/npm/l/@redocly/cli)

![OpenAPI CLI toolset](./media/redocly-cli.gif)

## Migration

Migrating from Redocly CLI v1 to v2?
Here's the [guide](https://redocly.com/docs/cli/guides/migrate-to-v2) to make the transition smoother.

## Usage

### Node

```sh
npx @redocly/cli@latest lint path-to-root-file.yaml
```

Alternatively, install it globally with `npm`:

```sh
npm install @redocly/cli -g
```

Then you can use it as `redocly [command] [options]`, for example:

```sh
redocly lint path-to-root-file.yaml
```

The minimum required versions of Node.js and NPM are 22.12.0 and 10.9.2 respectively.

### Docker

To give the Docker container access to the OpenAPI description files, you need to mount the containing directory as a volume.
Assuming the API description is rooted in the current working directory, you need the following command:

```sh
docker run --rm -v $PWD:/spec redocly/cli lint path-to-root-file.yaml
```

To build and run with a local image, run the following from the project root:

```sh
docker build -t redocly/cli .
docker run --rm -v $PWD:/spec redocly/cli lint path-to-root-file.yaml
```

## Common tasks

### Generate API reference documentation

Redocly CLI is a great way to render API reference documentation.
It uses open source [Redoc](https://github.com/redocly/redoc) to build your documentation.
Use a command like this:

```sh
redocly build-docs openapi.yaml
```

Your API reference docs are in `redoc-static.html` by default.
You can customize this in many ways.
[Read the main docs](https://redocly.com/docs/cli/commands/build-docs) for more information.

> :bulb: Redocly also has [hosted API management solution](https://redocly.com/reunite/), a (commercial) alternative to Redoc.
> Redoc, Revel, Reef, and Realm can be worked on locally using the `preview` command.

### Bundle multiple OpenAPI documents

Having one massive OpenAPI description can be annoying, so most people split them up into multiple documents via `$ref`, only to later find out some tools don't support `$ref` or don't support multiple documents.
Redocly CLI to the rescue! It has a `bundle` command you can use to recombine all of those documents back into one single document.
The bundled output that Redocly CLI provides is clean, tidy, and looks like a human made it.

### Automate API guidelines with Linting

Check that your API matches the expected API guidelines by using the `lint` command.
API guidelines are an important piece of API governance. They help to keep APIs consistent by enforcing the same standards and naming conventions, and they can also guide API teams through potential security hazards and other pitfalls.
Automating API guidelines means you can keep APIs consistent and secure throughout their lifecycle.
Even better, you can shape the design of the API before it even exists by combining API linting with a design-first API workflow.

Our API linter is designed for speed on even large documents, and it's easy to run locally, in CI, or anywhere you need it.
It's also designed for humans, with meaningful error messages to help you get your API right every time.

Try it like this:

```sh
redocly lint openapi.yaml
```

**Configure the rules** as you wish.
Other API Linters use complicated identifiers like JSONPath, but Redocly makes life easy with simple expressions that understand the API specification structure.
You can either use the [built-in rules](https://redocly.com/docs/cli/rules) to mix-and-match your ideal API guidelines, or break out the tools to [build your own](https://redocly.com/docs/cli/rules#rule-ideas).

**Format the output** in whatever way you need.
The `stylish` output is as good as it sounds, but if you need JSON, Markdown, Checkstyle and other outputs to integrate with other tools, the `lint` command can output those too.

**Multiple files supported** so you don't need to bundle your API description to lint it; just point Redocly CLI at the "entry point" (e.g.: `openapi.yaml`) and it handles the rest.

[Learn more about API standards and configuring Redocly rules](https://redocly.com/docs/cli/api-standards).

Looking for more examples? Check out our [Cookbook](https://github.com/Redocly/redocly-cli-cookbook).

### API contract testing with Respect

Ensure your APIs match their OpenAPI descriptions with **Respect**, Redocly's API contract testing system. Respect sends real HTTP requests to your API server and validates that responses match the expectations defined in your OpenAPI description and Arazzo workflows.

- Verify API responses match your OpenAPI schema
- Test complex API sequences using OpenAPI Arazzo format
- Catch API drift early in development and CI/CD
- Ensure external APIs you depend on behave as expected
- Track response times and detect anomalies

#### Quick start

```sh
# Generate test workflows from your OpenAPI spec
redocly generate-arazzo openapi.yaml

# Run contract tests against your API
redocly respect auto-generated.arazzo.yaml --verbose
```

#### Respect vs. Respect Monitoring

Respect is Redocly's community-edition product. Looking for something more? We also offer continuous API monitoring with additional features including:

- Real-time insights
- Automated alerts via email or Slack
- Automated API checks
- Proactive API quality assurance

Learn more about [Respect](https://redocly.com/respect) and [get started with API contract testing](https://redocly.com/docs/respect/get-started).

### Transform an OpenAPI description

If your OpenAPI description isn't everything you hoped it would be, enhance it with the Redocly [decorators](https://redocly.com/docs/cli/decorators) feature.
This allows you to:

- Publish reference docs with a subset of endpoints for public use
- Improve the docs by adding examples and descriptions
- Adapt an existing OpenAPI description, and replace details like URLs for use on staging platforms

## Data collection

This tool [collects data](./docs/usage-data.md) to help Redocly improve our products and services.
You can opt out by setting the `REDOCLY_TELEMETRY` environment variable to `off`.

## Update notifications

Redocly CLI checks for updates on startup.
You can disable this by setting the `REDOCLY_SUPPRESS_UPDATE_NOTICE` environment variable to `true`.

## More resources

[Read the detailed docs](https://redocly.com/docs/cli/).

## Credits

Thanks to [graphql-js](https://github.com/graphql/graphql-js) and [eslint](https://github.com/eslint/eslint) for inspiration of the API description traversal approach and to [Swagger](https://github.com/swagger-api/swagger-editor), [Spectral](https://github.com/stoplightio/spectral), and [OAS-Kit](https://github.com/Mermade/oas-kit) for inspiring the recommended ruleset.

## Development

Contributions are welcome!
All the information you need is in [CONTRIBUTING.md](CONTRIBUTING.md).
 readmeEtag: '"7ffb8715b2fa243155ea86555b3f5f1101452039"' readmeLastModified: Fri, 21 Nov 2025 09:28:04 GMT repositoryId: 207588519 description: >- ⚒️ Redocly CLI makes OpenAPI easy. Lint/validate to any standard, generate beautiful docs, and more. created: '2019-09-10T14:56:25Z' updated: '2026-02-05T16:31:10Z' language: TypeScript archived: false stars: 1357 watchers: 21 forks: 201 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"7c057fdc1397f54938320009028afb65dbddcb572cf7b50c12c210b467c56d82"' repoLastModified: Thu, 05 Feb 2026 16:31:10 GMT category: - Description Validators - Documentation - Server Implementations foundInMaster: true oldLocations: - https://github.com/redocly/openapi-cli name: Redocly CLI link: https://redocly.com/redocly-cli/ language: - CLI - TypeScript source_description: > Bring versatile OpenAPI validation, linting & bundling to your command line with this open-source Swiss-army knife. v2: true - source: https://openapi.tools/ name: - openapi-validator-middleware - fastify-openapi-glue category: - Data Validators - Server link: - https://www.npmjs.com/package/openapi-validator-middleware - https://www.npmjs.com/package/fastify-openapi-glue repository: https://github.com/payu/openapi-validator-middleware language: Node.js source_description: - > Provides data validation within an Express, Koa or Fastify app according to a OpenAPI definition. It uses Ajv under the hood for validation. - >- A plugin for the Fastify webserver to autogenerate a Fastify configuration based on a OpenApi description. v2: true v3: true repositoryMetadata: base64Readme: >- # openapi-validator-middleware

[![NPM Version][npm-image]][npm-url]
![main workflow](https://github.com/PayU/openapi-validator-middleware/actions/workflows/main.yml/badge.svg)
[![NPM Downloads][downloads-image]][downloads-url]
[![Test Coverage][coveralls-image]][coveralls-url]
[![Known Vulnerabilities][snyk-image]][snyk-url]
[![Apache 2.0 License][license-image]][license-url]

This package provides data validation within an Express, Koa or Fastify app according to a [Swagger/OpenAPI definition](https://swagger.io/specification/). It uses [Ajv](https://www.npmjs.com/package/ajv) under the hood for validation.

NOTICE: As this package gone through a long way, as we added support for OpenAPI definitions, while also adding support for more frameworks such as Koa and Fastify, we finally took the step of changing [express-ajv-swagger-validation](https://www.npmjs.com/package/express-ajv-swagger-validation) name to something that describes it better. As of now we'll be using the name [openapi-validator-middleware](https://www.npmjs.com/package/openapi-validator-middleware) instead.  
There are no code changes in `openapi-validator-middleware@2.0.0` compared to `express-ajv-swagger-validation@1.2.0` apart from the name change.

<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
**Table of Contents**  <!-- *generated with [DocToc](https://github.com/thlorenz/doctoc)* -->

- [openapi-validator-middleware](#openapi-validator-middleware)
  - [Installation](#installation)
  - [API](#api)
    - [openapi-validator-middleware.validate(fastifyOptions)](#openapi-validator-middlewarevalidatefastifyoptions)
      - [fastifyOptions](#fastifyoptions)
    - [openapi-validator-middleware.init(pathToSwaggerFile, options)](#openapi-validator-middlewareinitpathtoswaggerfile-options)
    - [openapi-validator-middleware.initAsync(pathToSwaggerFile, options)](#openapi-validator-middlewareinitasyncpathtoswaggerfile-options)
      - [Options](#options)
  - [Usage Example](#usage-example)
    - [Express](#express)
    - [Koa](#koa)
    - [Fastify](#fastify)
    - [Multiple-instances](#multiple-instances)
  - [Important Notes](#important-notes)
    - [Schema Objects](#schema-objects)
    - [Multipart/form-data (files)](#multipartform-data-files)
    - [Fastify support](#fastify-support)
    - [Koa support](#koa-support)
    - [Koa packages](#koa-packages)
  - [Known Issues with OpenAPI 3](#known-issues-with-openapi-3)
  - [Running Tests](#running-tests)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

## Installation

Install using the node package registry:

```bash
npm install --save openapi-validator-middleware
```

Then import the module in your code:

```js
const swaggerValidation = require('openapi-validator-middleware');
```

## API

### openapi-validator-middleware.validate(fastifyOptions)

This middleware function validates the request body, headers, path parameters and query parameters according to the _paths_ definition of the swagger file. Make sure to use this middleware inside a route definition in order to have `req.route.path` assigned to the most accurate express route.

- `fastifyOptions`: Only applicable for `fastify` framework. See below.

#### fastifyOptions

- `skiplist`: Endpoint paths for which validation should not be applied. An array of strings in RegExp format, e. g. `['^/pets$']` 

### openapi-validator-middleware.init(pathToSwaggerFile, options)

Initialize the middleware using a swagger definition.
The function executes synchronously and does not return anything.

- `pathToSwaggerFile`: Path to the swagger definition.
- `options`: Additional options for the middleware ([see below](#options)).

### openapi-validator-middleware.initAsync(pathToSwaggerFile, options)

Initialize the middleware using a swagger definition.
The function executes asynchronously and the resolved promise does not return anything.

This Initilaztion function also supports schema with references to external files. 

- `pathToSwaggerFile`: Path to the swagger definition.
- `options`: Additional options for the middleware ([see below](#options)).

#### Options

Options currently supported:
- `framework` - Defines in which framework the middleware is working ('fastify', 'koa' or 'express'). As default, set to 'express'.
- `formats` - Array of formats that can be added to `ajv` configuration, each element in the array should include `name` and `pattern`.
    ```js
    formats: [
        { name: 'double', pattern: /\d+\.(\d+)+/ },
        { name: 'int64', pattern: /^\d{1,19}$/ },
        { name: 'int32', pattern: /^\d{1,10}$/ }
    ]
    ```

- `keywords` - Array of keywords that can be added to `ajv` configuration, each element in the array can be either an object or a function. 
If the element is an object, it must include `name` and `definition`. If the element is a function, it should accept `ajv` as its first argument and inside the function you need to call `ajv.addKeyword` to add your custom keyword 
- `beautifyErrors`- Boolean that indicates if to beautify the errors, in this case it will create a string from the Ajv error.
    - Examples:
        - `query/limit should be <= 100` - query param
        - `path/petId should NOT be shorter than 3 characters` - path param not in format
        - `body/[0].test.field1 should be string` - Item in an array body
        - `body/test should have required property 'field1'` - nested field
        - `body should have required property 'name'` - Missing field in body

    You can see more examples in the [tests](./test).

- `firstError` - Boolean that indicates if to return only the first error.
- `makeOptionalAttributesNullable` - Boolean that forces preprocessing of Swagger schema to include 'null' as possible type for all non-required properties. Main use-case for this is to ensure correct handling of null values when Ajv type coercion is enabled
- `ajvConfigBody` - Object that will be passed as config to new Ajv instance which will be used for validating request body. Can be useful to e. g. enable type coercion (to automatically convert strings to numbers etc). See Ajv documentation for supported values.
- `ajvConfigParams` - Object that will be passed as config to new Ajv instance which will be used for validating request body. See Ajv documentation for supported values.
- `contentTypeValidation` - Boolean that indicates if to perform content type validation in case `consume` field is specified and the request body is not empty.
- `expectFormFieldsInBody` - Boolean that indicates whether form fields of non-file type that are specified in the schema should be validated against request body (e. g. Multer is copying text form fields to body)
- `errorFormatter` - optional custom function that will be invoked to create a validation error that will be thrown if Ajv validation fails. Function should accept two parameters: `(errors, middlewareOptions)` and return an error that will be thrown.

## Usage Example
### Express
```js
swaggerValidation.init('test/unit-tests/input-validation/pet-store-swagger.yaml');
const app = express();
app.use(bodyParser.json());
app.get('/pets', swaggerValidation.validate, (req, res, next) => {
    return res.json({ result: 'OK' });
});
app.post('/pets', swaggerValidation.validate, (req, res, next) => {
    return res.json({ result: 'OK' });
});
app.get('/pets/:petId', swaggerValidation.validate, (req, res, next) => {
    return res.json({ result: 'OK' });
});

app.use((err, req, res, next) => {
    if (err instanceof swaggerValidation.InputValidationError) {
        return res.status(400).json({ more_info: JSON.stringify(err.errors) });
    }
});

const server = app.listen(serverPort, () => {});
```

### Koa
```js
'use strict';
const Koa = require('koa');
const Router = require('koa-router');
const bodyParser = require('koa-bodyparser');
const inputValidation = require('openapi-validator-middleware');
const app = new Koa();
const router = new Router();
app.use(bodyParser());
app.use(router.routes());
inputValidation.init('test/pet-store-swagger.yaml', { framework: 'koa' });

router.get('/pets', inputValidation.validate, async (ctx, next) => {
    ctx.status = 200;
    ctx.body = { result: 'OK' };
});
router.post('/pets', inputValidation.validate, async (ctx, next) => {
    ctx.status = 200;
    ctx.body = { result: 'OK' };
});
router.get('/pets/:petId', inputValidation.validate, async (ctx, next) => {
    ctx.status = 200;
    ctx.body = { result: 'OK' };
});
router.put('/pets', inputValidation.validate, async (ctx, next) => {
    ctx.status = 200;
    ctx.body = { result: 'OK' };
});

app.use(async (ctx, next) => {
    try {
        return await next();
    } catch (err) {
        if (err instanceof inputValidation.InputValidationError) {
            ctx.status = 400;
            ctx.body = err.errors;
        }
        throw err;
    }
})
app.listen(process.env.PORT || 8888, function () {
    console.log('listening on', this.address());
});
```

### Fastify
```js
'use strict';
const fastify = require('fastify');
const inputValidation = require('openapi-validator-middleware');

async function getApp() {
    inputValidation.init('test/pet-store-swagger.yaml', { 
        framework: 'fastify'
    });
    const app = fastify({ logger: true });

    app.register(inputValidation.validate({
      skiplist: ['^/_metrics$']
    }));
    app.setErrorHandler(async (err, req, reply) => {
        if (err instanceof inputValidation.InputValidationError) {
             return reply.status(400).send({ more_info: JSON.stringify(err.errors) });
        }

        reply.status(500);
        reply.send();
    });

    app.get('/pets', (req, reply) => {
        reply.status(204).send();
    });

    await app.ready();
    return app;
}
```

### multiple-instances
```js
const inputValidation = require('openapi-validator-middleware');
const validatorA = inputValidation.getNewMiddleware('test/pet-store-swaggerA.yaml', {framework: 'express'});
const validatorB = inputValidation.getNewMiddleware('test/pet-store-swaggerB.yaml', {framework: 'express'});

app.get('/pets', validatorA.validate, (req, res, next) => {
    return res.json({ result: 'OK' });
});

app.post('/pets', validatorB.validate, (req, res, next) => {
    return res.json({ result: 'OK' });
});

```
## Important Notes

### Schema Objects

It is important to set the `type` property of any [Schema Objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object) explicitly to `object`. Although it isn't required in the OpenAPI specification, it is necessary in order for [Ajv](https://www.npmjs.com/package/ajv) to work correctly.

### Multipart/form-data (files) 

Multipart/form-data (files) support is based on [`express/multer`](https://github.com/expressjs/multer).

### Fastify support

Fastify support requires `uri-js` dependency to be installed. Make sure to run `npm install uri-js`.
When using this package as a middleware for fastify, the validations errors will be thrown.

### Koa support

When using this package as middleware for koa, the validations errors will be thrown.

### Koa packages

This package supports Koa servers that use [`koa-router`](https://www.npmjs.com/package/koa-router), [`koa-bodyparser`](https://www.npmjs.com/package/koa-bodyparser) and [`koa-multer`](https://www.npmjs.com/package/koa-multer).

## Known Issues with OpenAPI 3 

- Inheritance with a discriminator is supported only if the ancestor object is the discriminator.
- The discriminator support in the inheritance chain stops when getting to a child without a discriminator (a leaf in the inheritance tree), meaning a child without a discriminator cannot point to another child with a discriminator.

## Running Tests

The tests use mocha, istanbul and mochawesome. Run them using the node test script:

```bash
npm test
```

[npm-image]: https://img.shields.io/npm/v/openapi-validator-middleware.svg?style=flat
[npm-url]: https://npmjs.org/package/openapi-validator-middleware
[coveralls-image]: https://coveralls.io/repos/github/PayU/openapi-validator-middleware/badge.svg?branch=master
[coveralls-url]: https://coveralls.io/github/PayU/openapi-validator-middleware?branch=master
[downloads-image]: http://img.shields.io/npm/dm/openapi-validator-middleware.svg?style=flat
[downloads-url]: https://img.shields.io/npm/dm/openapi-validator-middleware.svg
[license-image]: https://img.shields.io/badge/license-Apache_2.0-green.svg?style=flat
[license-url]: LICENSE
[snyk-image]: https://snyk.io/test/github/PayU/openapi-validator-middleware/badge.svg?targetFile=package.json
[snyk-url]: https://snyk.io/test/github/PayU/openapi-validator-middleware?targetFile=package.json
 readmeEtag: '"9c580833b662962db48b0e4df553ad32e2707add"' readmeLastModified: Thu, 24 Mar 2022 08:37:29 GMT repositoryId: 107256133 description: Input validation using Swagger (Open API) and ajv created: '2017-10-17T10:56:52Z' updated: '2025-08-27T13:21:55Z' language: JavaScript archived: false stars: 144 watchers: 30 forks: 50 owner: PayU logo: https://avatars.githubusercontent.com/u/10152107?v=4 license: Apache-2.0 repoEtag: '"ba986b052dbdcc3a8f648065d2bfeb6ae56a9670e0834936d1ebc0a6e057c870"' repoLastModified: Wed, 27 Aug 2025 13:21:55 GMT foundInMaster: true id: a412c258fff439cbb6458a2a341038f9 v3_1: true - source: https://openapi.tools/ name: committee category: Data Validators repository: https://github.com/interagent/committee language: Ruby source_description: >- Validation middleware for Rack server. This gem validates request and response using an OpenAPI Description. And convert parameter string to specific Ruby object (e.g. convert datetime string to DateTime class). v2: true v3: true repositoryMetadata: base64Readme: >- # Committee  [![ci](https://github.com/interagent/committee/actions/workflows/ci.yaml/badge.svg)](https://github.com/interagent/committee/actions/workflows/ci.yaml) [![Gem Version](https://badge.fury.io/rb/committee.svg)](https://badge.fury.io/rb/committee)

A collection of middleware to help build services with JSON Schema, OpenAPI 2, OpenAPI 3.

## Supported Ruby Versions

Committee is tested on the following MRI versions:

- 2.7
- 3.0
- 3.1
- 3.2
- 3.3
- 3.4

## Committee::Middleware::RequestValidation

This feature is supported by all of Hyper-Schema, OpenAPI 2, and OpenAPI 3.

``` ruby
use Committee::Middleware::RequestValidation, schema_path: 'docs/schema.json', coerce_date_times: true
```

This piece of middleware validates the parameters of incoming requests to make sure that they're formatted according to the constraints imposed by a particular schema.

### Configuration Options

| name | Hyper-Schema | OpenAPI 3 | Description |
|-----------:|------------:|------------:| :------------ |
|allow_blank_structures | false | always true | Allow Empty Response Body. supported on Hyper-schema parser but will default to true in next major version. |
|allow_form_params | true | true | Specifies that input can alternatively be specified as `application/x-www-form-urlencoded` parameters when possible. This won't work for more complex schema validations. |
|allow_get_body | true | false | Allow GET request body, which merge to request parameter. See (#211) |
|allow_query_params | true | true | Specifies that query string parameters will be taken into consideration when doing validation. |
|allow_non_get_query_params | false | false | Allow GET and non-GET query string parameters to be taken into consideration when doing validation. |
|check_content_type | true | true | Specifies that `Content-Type` should be verified according to JSON Hyper-schema or OpenAPI 3 definition. |
|check_header | true | true | Check header data using JSON Hyper-schema or OpenAPI 3 definition. |
|coerce_date_times | false | true | Convert the string with `"format": "date-time"` parameter to DateTime object. |
|coerce_form_params| false | true | Tries to convert POST data encoded into an `application/x-www-form-urlencoded` body (where values are all strings) into concrete types required by the schema. This works for `null` (empty value), `integer` (numeric value without decimals), `number` (numeric value) and `boolean` ("true" is converted to `true` and "false" to `false`). If coercion is not possible, the original value is passed unchanged to schema validation. |
|coerce_path_params| false | true | The same as `coerce_form_params`, but tries to coerce parameters encoded in a request's URL path. |
|coerce_query_params| false | true  | The same as `coerce_form_params`, but tries to coerce `GET` parameters encoded in a request's query string. |
|coerce_recursive| false | always true | Coerce data in arrays and other nested objects |
|coerce_response_values| false | false | Enable type coercion for response body validation. When `true`, allows string values like `"726"` to be coerced to numbers. When `false` (default), response bodies must match exact types defined in the schema. |
|optimistic_json| false | false | Will attempt to parse JSON in the request body even without a `Content-Type: application/json` before falling back to other options. |
|raise| false | false | Raise an exception on error instead of responding with a generic error body. |
|strict| false | false | Puts the middleware into strict mode, meaning that paths which are not defined in the schema will be responded to with a 404 instead of being run. |
|strict_query_params| not supported | false | Rejects requests with query parameters not defined in the schema. When enabled, any query parameter not explicitly defined in the OpenAPI specification will result in a 400 error. |
|strict_reference_validation| always false | false | Raises an exception (`OpenAPIParser::MissingReferenceError`) on middleware load if the provided schema file contains unresolvable references (`$ref:"#/something/not/here"`). Not supported on Hyper-schema parser. Defaults to `false` on OpenAPI3 but will default to `true` in next major version. |
|ignore_error| false | false | Validate and ignore result even if validation is error. So always return original data. |

Non-boolean options:

| name | allowed object type | Hyper-Schema | OpenAPI 3 | Description |
|-----------:|------------:|------------:|------------:| :------------ |
|error_class| StandardError | supported | supported | Change validation errors from `Committee::ValidationError`). |
|prefix| String | supported | supported | Mounts the middleware to respond at a configured prefix. (e.g. prefix is '/v1' and request path is '/v1/test' use '/test' definition). |
|schema_path| String | supported | supported | Defines the location of the schema file to use for validation. |
|error_handler| Proc Object | supported | supported | A proc which will be called when error occurs. Take an Error instance as first argument, and request.env as second argument. (e.g. `-> (ex, env) { Raven.capture_exception(ex, extra: { rack_env: env }) }`) |
|accept_request_filter  | Proc Object | supported | supported | A proc that accepts a Request and returns a boolean. It indicates whether to validate the current request, or not. (e.g. `-> (request) { request.path.start_with?('/something') }`) |
|params_key| String | supported | supported | Save checked and merged parameter value to request.env using this key. Default value is `committee.params` |
|headers_key| String | supported | supported | Save checked header value to request.env using this key. Default value is `committee.headers` |
|query_hash_key| String | supported | supported | Save checked query parameter value to request.env using this key. Default value is `rack.request.query_hash` but we will change  `committee.query_hash` in next version |
|path_hash_key| String | supported | supported | Save checked path parameter value to request.env using this key. Default value is `committee.path_hash` |
|request_body_hash_key| String | supported | supported | Save checked request body parameter (json, form) value to request.env using this key. Default value is `committee.request_body_hash` |


Note that Hyper-Schema and OpenAPI 2 get the same defaults for options.

Some examples of use:

``` bash
# missing required parameter
$ curl -X POST http://localhost:9292/account/app-transfers -H "Content-Type: application/json" -d '{"app":"heroku-api"}'
{"id":"bad_request","message":"#/paths/~1account~1app-transfers/post/requestBody/content/application~1json/schema missing required parameters: recipient"}

# missing required parameter (should have &query=...)
$ curl -X GET http://localhost:9292/search?category=all
{"id":"bad_request","message":"#/paths/~1search/get missing required parameters: query"}

# contains an unknown parameter
$ curl -X POST http://localhost:9292/account/app-transfers -H "Content-Type: application/json" -d '{"app":"heroku-api","recipient":"api@heroku.com","sender":"api@heroku.com"}'
{"id":"bad_request","message":"#/paths/~1account~1app-transfers/post/requestBody/content/application~1json/schema does not define properties: sender"}

# invalid type
$ curl -X POST http://localhost:9292/account/app-transfers -H "Content-Type: application/json" -d '{"app":"heroku-api","recipient":7}'
{"id":"bad_request","message":"#/paths/~1account~1app-transfers/post/requestBody/content/application~1json/schema/properties/recipient expected string, but received Integer: 7"}

# invalid format (supports date-time, email, uuid)
$ curl -X POST http://localhost:9292/account/app-transfers -H "Content-Type: application/json" -d '{"app":"heroku-api","recipient":"matz"}'
{"id":"bad_request","message":"#/paths/~1account~1app-transfers/post/requestBody/content/application~1json/schema/properties/recipient email address format does not match value: matz"}

# invalid pattern
$ curl -X POST http://localhost:9292/apps -H "Content-Type: application/json" -d '{"name":"$#%"}'
{"id":"bad_request","message":"#/paths/~1apps/post/requestBody/content/application~1json/schema/properties/name pattern ^[a-z][a-z0-9-]{3,50}$ does not match value: $#%"}
```

## Committee::Middleware::Stub

**Note:** This feature is not yet available for OpenAPI 3.

``` ruby
use Committee::Middleware::Stub, schema_path: 'docs/schema.json'
```

This piece of middleware intercepts any routes that are in the JSON Schema, then builds and returns an appropriate response for them.

``` bash
$ curl -X GET http://localhost:9292/apps
[
  {
    "archived_at":"2012-01-01T12:00:00Z",
    "buildpack_provided_description":"Ruby/Rack",
    "created_at":"2012-01-01T12:00:00Z",
    "git_url":"git@heroku.com/example.git",
    "id":"01234567-89ab-cdef-0123-456789abcdef",
    "maintenance":false,
    "name":"example",
    "owner":[
      {
        "email":"username@example.com",
        "id":"01234567-89ab-cdef-0123-456789abcdef"
      }
    ],
    "region":[
      {
        "id":"01234567-89ab-cdef-0123-456789abcdef",
        "name":"us"
      }
    ],
    "released_at":"2012-01-01T12:00:00Z",
    "repo_size":0,
    "slug_size":0,
    "stack":[
      {
        "id":"01234567-89ab-cdef-0123-456789abcdef",
        "name":"cedar"
      }
    ],
    "updated_at":"2012-01-01T12:00:00Z",
    "web_url":"http://example.herokuapp.com"
  }
]
```

### committee-stub

A bundled executable is also available to easily start up a server that will serve the stub for some particular JSON Schema file:

``` bash
committee-stub -p <port> <path to JSON schema>
```

## Committee::Middleware::ResponseValidation

This feature is supported by all of Hyper-Schema, OpenAPI 2, and OpenAPI 3.

``` ruby
use Committee::Middleware::ResponseValidation, schema_path: 'docs/schema.json'
```

This piece of middleware validates the contents of the response received from up the stack for any route that matches the JSON Schema. A hyper-schema link's `targetSchema` property is used to determine what a valid response looks like.

### Configuration Options

| name | Hyper-Schema | OpenAPI 3 | Description |
|-----------:|------------:|------------:| :------------ |
|raise| false | false | Raise an exception on error instead of responding with a generic error body. |
|validate_success_only| true | false | Also validate non-2xx responses only. |
|ignore_error| false | false | Validate and ignore result even if validation is error. So always return original data. |
|parse_response_by_content_type| true | true | Parse response body to JSON only if Content-Type header is 'application/json'. When false, this always optimistically parses as JSON without checking for Content-Type header. |
|strict| false | false | Puts the middleware into strict mode, meaning that response code and content type does not defined in the schema will be responded to with a 500 instead of application's status code. |
|strict_reference_validation| always false | false | Raises an exception (`OpenAPIParser::MissingReferenceError`) on middleware load if the provided schema file contains unresolvable references (`$ref:"#/something/not/here"`). Not supported on Hyper-schema parser. Defaults to `false` on OpenAPI3 but will default to `true` in next major version. |

No boolean option values:

| name | allowed object type | Hyper-Schema | OpenAPI 3 | Description |
|-----------:|------------:|------------:|------------:| :------------ |
|prefix| String | support | support | Mounts the middleware to respond at a configured prefix. |
|error_class| StandardError | support | support | Specifies the class to use for formatting and outputting validation errors (defaults to `Committee::ValidationError`). |
|error_handler| Proc Object | support | support | A proc which will be called when error occurs. Take an Error instance as first argument, and request.env as second argument. (e.g. `-> (ex, env) { Raven.capture_exception(ex, extra: { rack_env: env }) }`) |
|streaming_content_parsers| Hash[String, Proc] | support | support | A hash mapping content types to custom parser lambdas for streaming responses (e.g. `{ 'text/event-stream' => ->(body) { body } }`). Used for validating streaming response formats like Server-Sent Events. |

Given a simple Sinatra app that responds for an endpoint in an incomplete fashion:

``` ruby
require "committee"
require "sinatra"

use Committee::Middleware::ResponseValidation, schema_path: 'docs/schema.json'

get "/apps" do
  content_type :json
  "[{}]"
end
```

The middleware will raise an error to indicate what the problems are:

``` bash
# missing keys in response
$ curl -X GET http://localhost:9292/apps
{"id":"invalid_response","message":"Missing keys in response: archived_at, buildpack_provided_description, created_at, git_url, id, maintenance, name, owner:email, owner:id, region:id, region:name, released_at, repo_size, slug_size, stack:id, stack:name, updated_at, web_url."}
```

If you want to take log only (for example avoiding false-positive in production), use `ignore_error` and `error_handler` option.

## Validation Errors

Committee will by default respond with a generic error JSON body for validation errors (when the `raise` middleware option is `false`).

Here's an example error to show the default format:

```json
{
  "id":"invalid_response",
  "message":"Missing keys in response: archived_at, buildpack_provided_description, created_at, git_url, id, maintenance, name, owner:email, owner:id, region:id, region:name, released_at, repo_size, slug_size, stack:id, stack:name, updated_at, web_url."
}
```

You can customize this JSON body by setting the `error_class` middleware option. The `error_class` will be instantiated with: `status`, `id`, and `message`.

* `status`: HTTP status code
* `id`: HTTP status name/string
* `message`: error message

Here's an example of a class to format errors according to [JSON API](http://jsonapi.org/format/#errors):

```ruby
module MyAPI
  class ValidationError < Committee::ValidationError
    def error_body
      {
        errors: [
          { status: id, detail: message }
        ]
      }
    end

    def render
      [
        status,
        { "Content-Type" => "application/vnd.api+json" },
        [JSON.generate(error_body)]
      ]
    end
  end
end
```

## Test Assertions

Supported in HyperSchema and OpenAPI 3.

Committee ships with a small set of schema validation test assertions designed to be used along with `rack-test`.

Here's a simple test to demonstrate:

``` ruby
describe Committee::Middleware::Stub do
  include Committee::Test::Methods
  include Rack::Test::Methods

  def app
    Sinatra.new do
      get "/" do
        content_type :json
        JSON.generate({ "foo" => "bar" })
      end
    end
  end

  def committee_options
    @committee_options ||= { schema: Committee::Drivers::load_from_file('docs/schema.json'), prefix: "/v1" }
  end

  def request_object
    last_request
  end

  def response_data
    [last_response.status, last_response.headers, last_response.body]
  end

  describe "GET /" do
    it "conforms to schema with 200 response code" do
      get "/"
      assert_schema_conform(200)
    end

    it "conforms to request schema" do
      get "/"
      assert_request_schema_confirm
    end

    it "conforms to response schema with 200 response code" do
      get "/"
      assert_response_schema_confirm(200)
    end

    it "conforms to response and request schema with 200 response code" do
      @committee_options[:old_assert_behavior] = false
      get "/"
      assert_schema_conform(200)
    end
  end
end
```


## Tips

### Use Ruby on Rails with coerce option
Please set `'action_dispatch.request.request_parameters'` to `params_key` option.

```
use Committee::Middleware::RequestValidation,
      schema_path: 'docs/schema.json',
      coerce_date_times: true,
      params_key: 'action_dispatch.request.request_parameters'
```

Committee has few options which enable convert request data.
By default committee save converted data to `committee.params` and rails does not read it.
So we need to save converted value to `'action_dispatch.request.request_parameters'` because rails create parameter from this value.

### Streaming Response Validation

Committee supports validating streaming responses like Server-Sent Events or custom streaming formats using the `streaming_content_parsers` option:

```ruby
use Committee::Middleware::ResponseValidation,
  schema_path: 'docs/schema.json',
  streaming_content_parsers: {
    'text/event-stream' => ->(body) { body },  # Pass through SSE as string
    'application/x-json-stream' => ->(body) { JSON.parse(body) }  # Parse custom JSON stream
  }
```

This allows you to validate streaming response formats by providing custom parsers for specific content types. The parser lambda receives the complete response body as a string and should return the parsed data for validation.

**Note:** This feature uses `Rack::BodyProxy` internally, which buffers the entire response body in memory before validation. This can consume significant memory for large streaming responses. Consider this limitation when validating large data streams.

Limitation: For standard Content-Types, it is possible to intervene in the actual response when validation fails (e.g., by setting the HTTP status code to 500). However, for Content-Types specified via the `streaming_content_parsers` option, validation is performed only after the entire response body has been read—that is, after it has already been returned to the client—making it impossible to modify the response. In such cases, only `handle_exception` is called, and an error is raised if necessary.

## Using OpenAPI 3

Committee can detect the type of schema (Hyper-Schema, OpenAPI 3, etc.) from the provided file, so there's no need to pass in any additional options:

```ruby
use Committee::Middleware::RequestValidation, schema_path: 'open_api_3/schema.yaml'
```

If you want to select the type manually, pass an `OpenAPI 3` object to the `schema` option manually:

```ruby
open_api = OpenAPIParser.parse(YAML.load_file('open_api_3/schema.yaml'))
schema = Committee::Drivers::OpenAPI3::Driver.new.parse(open_api)
use Committee::Middleware::RequestValidation, schema: schema
```

### Limitations of OpenAPI 3 support

* Stub servers are not yet supported, so neither `Committee::Middleware::Stub` or `Committee::Bin::CommitteeStub` are functional.
* Changing `coerce_recursive` isn't supported. This option is always on.

### Upgrading from Committee 2.* to 3.*

Committee 3.* has many breaking changes so we recommend upgrading to the latest release on 2.* and fixing any deprecation errors you see before upgrading to 3.*. The steps would be roughly as follows:

1. Update to the latest 2.* release (usually by modifying the statement in your `Gemfile` and running `bundle update`).
2. Run your test suite and fix any deprecation warnings that appear.
3. Update to the latest 3.* release.
4. Switch to OpenAPI 3 if you'd like to do so.

Important changes are also described below.

### Upgrading from Committee 4.* to 5.*

Committee 5.* has few breaking changes so we recommend upgrading to the latest release on 4.* and fixing any deprecation errors you see before upgrading.

- set `parse_response_by_content_type=true` by default (old versions set `false`)
- set `parameter_overwrite_by_rails_rule=true` by default (old version set `false`)

#### Future Updates (5.*~)
OpenAPI3 Schema Users: Newly-added `strict_reference_validation` option defaults to `false` if not set.
From a later major version of Committee, we want to change it to `true` (or remove the option entirely and force
`true` to be passed to the OpenAPI3 parser (see: https://github.com/interagent/committee/issues/343).

To remove deprecation warnings, pass `strict_reference_validation: true` as an option when loading the middleware
(because this is about schema parsing/initialization, the setting is valid for both request and response validation).

### Setting schemas in middleware

Committee 2.* supported setting `schema` to a string or a hash like this:

```ruby
# valid
use Committee::Middleware::RequestValidation, schema: JSON.parse(File.read(...))

# valid
use Committee::Middleware::RequestValidation, schema: {json: 'json_data...'}

# valid
use Committee::Middleware::RequestValidation, schema: 'json string'

```

That usage is no longer supported in 3.* Instead, use either `schema_path` or set a parsed schema object to `schema`:

```ruby
# auto-select Hyper-Schema/OpenAPI 2/OpenAPI 3 from file
use Committee::Middleware::RequestValidation, schema_path: 'docs/schema.json' # using file extension

# auto-select Hyper-Schema/OpenAPI 2/OpenAPI 3 from hash
json = JSON.parse(File.read('docs/schema.json'))
use Committee::Middleware::RequestValidation, schema: Committee::Drivers::load_data(json)

# manually select
json = JSON.parse(File.read(...))
schema = Committee::Drivers::HyperSchema::Driver.new.parse(json)
use Committee::Middleware::RequestValidation, schema: schema
```

The auto-select algorithm works roughly like this (so make sure that your file sets one of these attributes correctly):

```ruby
hash = JSON.load(json_path)

# OpenAPI 3 requires the `openapi` key and a version
if hash['openapi']&.start_with?('3.')
  return Committee::Drivers::OpenAPI3::Driver.new.parse(hash)

# OpenAPI 2 requires the `swagger` key
elsif hash['swagger'] == '2.0'
  return Committee::Drivers::OpenAPI2::Driver.new.parse(hash)

else
  return Committee::Drivers::HyperSchema::Driver.new.parse(hash)
end
```

### Test assertions

Committee 3.* drops many of the methods that were
previously available from the `Committee::Test::Methods`
mixin.

Use it by defining a `committee_options` method and having
it return a schema and other options you'd like to use:

```ruby
def committee_options
  @committee_options ||= { schema: Committee::Drivers::load_from_file('docs/schema.json'), prefix: "/v1", validate_success_only: true }
end
```

The default assertion option in 2.* was `validate_success_only=true`, but this becomes `validate_success_only=false` in 3.*. For the smoothest possible upgrade, you should set it to `false` in your test suite before upgrading to 3.*.

### Test schema coverage
You can check how much of your API schema your tests have covered.
NOTICE: Currently committee only supports schema coverage for **openapi** schemas, and only checks coverage on responses, via `assert_response_schema_confirm` or `assert_schema_conform` methods.
Usage:
1. Set schema_coverage option of `committee_options`
2. Use `assert_response_schema_confirm` or `assert_schema_conform`
3. Then use `SchemaCoverage#report` or `SchemaCoverage#report_flatten` to get coverage report

Example:
```ruby
before do
  schema_coverage = Committee::Test::SchemaCoverage.new(openapi_schema)
  @committee_options[:schema_coverage] = schema_coverage
end
it 'covers /some_api' do
  get '/some_api'
  assert_response_schema_confirm # or assert_schema_conform
  coverage_report = schema_coverage.report
  # check coverage expectations of /some_api here
end
it 'covers /other_api schema' do
  get '/other_api'
  assert_response_schema_confirm # or assert_schema_conform
  coverage_report = schema_coverage.report
  # check coverage expectations of /other_api here
end
after do
  coverage_report = schema_coverage.report
  # check coverage expectations of all apis here
end
```

Coverage report structure:
```
/* using #report */
{
  <path> => {
    <method> => {
      'responses' => {
        <status> => <true|false>
      }
    }
  }
}
/* using #report_flatten */
{
  responses: [
    { path: <path>, method: <method>, status: <status>, is_covered: <true|false> },
  ]
}
```

Other helper methods:
* `Committee::Test::SchemaCoverage.merge_report(<Hash>, <Hash>)`: merge 2 coverage reports together
* `Committee::Test::SchemaCoverage.flatten_report(<Hash>)`: flatten a coverage report Hash into flatten structure

### Other changes

* `GET` request bodies are ignored in OpenAPI 3 by default. If you want to use them, set the `allow_get_body` option to `true`.

## Development

Run tests with the following:

```
bundle install
bundle exec rake
```

Run a particular test suite or test:

```
bundle exec ruby -Ilib -Itest test/router_test.rb
bundle exec ruby -Ilib -Itest test/router_test.rb -n /prefix/
```

## Release

1. Update the version in `lib/committee/version.rb` as appropriate for [semantic
   versioning](http://semver.org) and add details to `CHANGELOG.md`.
2. Commit those changes. Use a commit message like `Bump version to 1.2.3`.
3. Run the `release` task:

    ```
    bundle exec rake release
    ```

## Tutorials

- [Validating requests and responses using OpenAPI specification with Committee](https://nicolasiensen.github.io/2022-04-18-validating-requests-and-responses-using-openapi-specification-with-committee/)
 readmeEtag: '"dc647be71a2a55e00a72cc7d7939a9cf2484d0c6"' readmeLastModified: Tue, 06 Jan 2026 03:19:53 GMT repositoryId: 14134534 description: A collection of Rack middleware to support JSON Schema. created: '2013-11-05T07:05:45Z' updated: '2026-02-06T00:31:50Z' language: Ruby archived: false stars: 940 watchers: 14 forks: 140 owner: interagent logo: https://avatars.githubusercontent.com/u/7388387?v=4 license: MIT repoEtag: '"47f3d8d33beb4003d17d98788021b5579e49a0dc9249e876be38f2dd44bad220"' repoLastModified: Fri, 06 Feb 2026 00:31:50 GMT foundInMaster: true id: 64f8f7a816e2d8825abd1a213d9e3fb5 - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/ahx/openapi_first v3: true repositoryMetadata: base64Readme: >- # openapi_first

openapi_first is a Ruby gem for request / response validation and contract-testing against an [OpenAPI](https://www.openapis.org/) 3.0 or 3.1 Openapi API description (OAD). It makes an APIFirst workflow easy and reliable.

## Usage

Configure
```ruby
OpenapiFirst.register('openapi/openapi.yaml')
```

Use an OAD to validate incoming requests:
```ruby
use OpenapiFirst::Middlewares::RequestValidation
```

Turn your request tests into [contract tests](#contract-testing) against an OAD:
```ruby
# spec_helper.rb
require 'openapi_first'
OpenapiFirst::Test.setup

require 'my_app'
RSpec.configure do |config|
  config.include OpenapiFirst::Test::Methods[MyApp], type: :request
end
```

## Contents

<!-- TOC -->

- [Configuration](#configuration)
- [Rack Middlewares](#rack-middlewares)
  - [Request validation](#request-validation)
  - [Response validation](#response-validation)
- [Contract testing](#contract-testing)
  - [Test assertions](#test-assertions)
- [Manual use](#manual-use)
- [Framework integration](#framework-integration)
- [Hooks](#hooks)
- [Alternatives](#alternatives)
- [Frequently Asked Questions](#frequently-asked-questions)
- [Development](#development)
  - [Benchmarks](#benchmarks)
  - [Contributing](#contributing)

<!-- /TOC -->

## Configuration

You should register OADs globally so you don't have to load the file multiple times or to refernce them by Symbol (like :v1 in this example).
```ruby
OpenapiFirst.configure do |config|
  config.register('openapi/openapi.yaml') # :default
  config.register('openapi/v1.openapi.yaml', as: :v1)
end
```

You can configure default options globally:

```ruby
OpenapiFirst.configure do |config|
  # Specify which plugin is used to render error responses returned by the request validation middleware (defaults to :default)
  config.request_validation_error_response = :jsonapi
end
```

or configure per instance:

```ruby
OpenapiFirst.load('openapi.yaml') do |config|
  config.request_validation_error_response = :jsonapi
end
```

## Rack Middlewares

### Request validation

The request validation middleware returns a 4xx if the request is invalid or not defined in the API description. It adds a request object to the current Rack environment at `env[OpenapiFirst::REQUEST]` with the request parameters parsed exactly as described in your API description plus access to meta information from your API description. See _[Manual use](#manual-use)_ for more details about that object.

```ruby
use OpenapiFirst::Middlewares::RequestValidation

# Pass `raise_error: true` to raise an error if request is invalid:
use OpenapiFirst::Middlewares::RequestValidation, raise_error: true
```

#### Error responses

openapi_first produces a useful machine readable error response that can be customized.
The default response looks like this. See also [RFC 9457](https://www.rfc-editor.org/rfc/rfc9457).

```json
http-status: 400
content-type: "application/problem+json"

{
  "title": "Bad Request Body",
  "status": 400,
  "errors": [
    {
      "message": "value at `/data/name` is not a string",
      "pointer": "/data/name",
      "code": "string"
    },
    {
      "message": "number at `/data/numberOfLegs` is less than: 2",
      "pointer": "/data/numberOfLegs",
      "code": "minimum"
    },
    {
      "message": "object at `/data` is missing required properties: mandatory",
      "pointer": "/data",
      "code": "required"
    }
  ]
}
```

openapi_first offers a [JSON:API](https://jsonapi.org/) error response by passing `error_response: :jsonapi`:

```ruby
use OpenapiFirst::Middlewares::RequestValidation, 'openapi.yaml', error_response: :jsonapi
```

<details>
<summary>See details of JSON:API error response</summary>

```json
// http-status: 400
// content-type: "application/vnd.api+json"

{
  "errors": [
    {
      "status": "400",
      "source": {
        "pointer": "/data/name"
      },
      "title": "value at `/data/name` is not a string",
      "code": "string"
    },
    {
      "status": "400",
      "source": {
        "pointer": "/data/numberOfLegs"
      },
      "title": "number at `/data/numberOfLegs` is less than: 2",
      "code": "minimum"
    },
    {
      "status": "400",
      "source": {
        "pointer": "/data"
      },
      "title": "object at `/data` is missing required properties: mandatory",
      "code": "required"
    }
  ]
}
```

</details>

#### Custom error responses

You can build your own custom error response with `error_response: MyCustomClass` that implements `OpenapiFirst::ErrorResponse`.
You can define custom error responses globally by including / implementing `OpenapiFirst::ErrorResponse` and register it via `OpenapiFirst.register_error_response(my_name, MyCustomErrorResponse)` and set `error_response: my_name`.

### Response validation

This middleware raises an error by default if the response is not valid.
This can be useful in a test or staging environment, especially if you are adopting OpenAPI for an existing implementation.

```ruby
use OpenapiFirst::Middlewares::ResponseValidation if ENV['RACK_ENV'] == 'test'

# Pass `raise_error: false` to not raise an error:
use OpenapiFirst::Middlewares::ResponseValidation, raise_error: false
```

If you are adopting OpenAPI you can use these options together with [hooks](#hooks) to get notified about requests/responses that do match your API description.

## Contract Testing

You can see your OpenAPI API description as a contract that your clients can rely on as how your API behaves. There are two aspects of contract testing: Validation and Coverage. By validating requests and responses, you can avoid that your API implementation processes requests or returns responses that don't match your API description. To make sure your _whole_ API description is implemented, openapi_first can check that all of your API description is covered when you test your API with [rack-test](https://github.com/rack/rack-test).

Here is how to set it up:

1. Setup the test mode
    ```ruby
    # spec_helper.rb
    require 'openapi_first'
    OpenapiFirst::Test.setup
    ```
2. Observe your application. You can do this in multiple ways:
    - Add an `app` method to your tests (which is called by rack-test) that wraps your application with silent request / response validation.
      ```ruby
      module RequestSpecHelpers
        def app
          OpenapiFirst::Test.app(MyApp)
        end
      end

      RSpec.configure do |config|
        config.include RequestSpecHelpers, type: :request
      end
      ```

      Or do this by creating a Module and including it to add an "app" method.

      ```ruby
      RSpec.configure do |config|
        config.include OpenapiFirst::Test::Methods[MyApp], type: :request
      end
      ```
3. Run your tests. The Coverage feature will tell you about missing or invalid requests/responses:
      ```
      ✓ GET /stations
        ✓ 200(application/json)
        ❌ 200(application/xml) – No responses tracked!
        ❌ 400(application/problem+json) – No responses tracked!
      ```

      Now add tests for all those "❌" to make them "✓" and you're green!

> [!NOTE]
> Check out [faraday-openapi](https://codeberg.org/ahx/faraday-openapi) to have your API _client_ validate request/responses against an OAD, which is useful to validate HTTP mocks during testing.

### Configure test coverage

You can ignore errors for certain requests/responses. This will stop `OpenapiFirst::Test` from raising an exception if the block returns true. 

```ruby
OpenapiFirst::Test.setup do |test|
  test.ignore_request_error do |validated_request|
    # Ignore unknown requests on certain paths
    validated_request.path.start_with?('/api/v1') && validated_request.unknown?
  end
  
  test.ignore_response_error do |validated_response, rack_request|
    # Ignore invalid response bodies on certain paths
    validated_request.path.start_with?('/api/legacy/stuff') && validated_request.error.type ==  :invalid_body      
  end
end
```


OpenapiFirst::Test raises an error when a response status is not defined except for 404 and 500. You can change this:

```ruby
OpenapiFirst::Test.setup do |test|
  test.ignored_unknown_status << 403
end
```

Or you can ignore all unknown response status:

```ruby
OpenapiFirst::Test.setup do |test|
  test.ignore_unknown_response_status = true
end
```

Exclude certain _responses_ from coverage with `skip_coverage`:

```ruby
OpenapiFirst::Test.setup do |test|
  test.skip_response_coverage do |response_definition|
    response_definition.status == '5XX'
  end
end
```

Skip coverage for a request and all responses alltogether of a route with `skip_coverage`:

```ruby
OpenapiFirst::Test.setup do |test|
  test.skip_coverage do |path, request_method|
    path == '/bookings/{bookingId}' && requests_method == 'DELETE'
  end
end
```

OpenapiFirst::Test raises an error when a request is not defined. You can deactivate this with:

```ruby
OpenapiFirst::Test.setup do |test|
  test.ignore_unknown_requests = true
end
```

### Test assertions

> [!WARNING]
> You probably don't need this. Just setup [Contract testing and API coverage](#contract-testing) and use your normal assertions.

openapi_first ships with a simple but powerful Test method `assert_api_conform` to run request and response validation in your tests without using the middlewares. This is designed to be used with rack-test.

Here is how to use it with RSpec, but MiniTest works just as good:

```ruby
# spec_helper.rb
OpenapiFirst::Test.setup do |test|
  test.register(File.join(__dir__, '../examples/openapi.yaml'), as: :example_app)
end
```


Inside your test :
```ruby
RSpec.describe 'Example App' do
  include Rack::Test::Methods
  include OpenapiFirst::Test::Methods[App]

  it 'is API conform' do
    get '/'
    assert_api_conform(status: 200)
  end
end
```

## Manual use

Load the API description:

```ruby
require 'openapi_first'

definition = OpenapiFirst.load('openapi.yaml')
```

### Validate request

```ruby
validated_request = definition.validate_request(rack_request)

# Inspect the request and access parsed parameters
validated_request.valid?
validated_request.invalid?
validated_request.error # => Failure object or nil
validated_request.parsed_body # => The parsed request body (Hash)
validated_request.parsed_query # A Hash of query parameters that are defined in the API description, parsed exactly as described.
validated_request.parsed_path_parameters
validated_request.parsed_headers
validated_request.parsed_cookies
validated_request.parsed_params # Merged parsed path, query parameters and request body
# Access the Openapi 3 Operation Object Hash
validated_request.operation['x-foo']
validated_request.operation['operationId'] => "getStuff"
# or the whole request definition
validated_request.request_definition.path # => "/pets/{petId}"
validated_request.request_definition.operation_id # => "showPetById"

# Or you can raise an exception if validation fails:
definition.validate_request(rack_request, raise_error: true) # Raises OpenapiFirst::RequestInvalidError or OpenapiFirst::NotFoundError if request is invalid
```

### Validate response

```ruby
validated_response = definition.validate_response(rack_request, rack_response)

# Inspect the response and access parsed parameters and
validated_response.valid?
validated_response.invalid?
validated_response.error # => Failure object or nil
validated_response.status # => 200
validated_response.parsed_body
validated_response.parsed_headers

# Or you can raise an exception if validation fails:
definition.validate_response(rack_request,rack_response, raise_error: true) # Raises OpenapiFirst::ResponseInvalidError or OpenapiFirst::ResponseNotFoundError
```

## Hooks

You can integrate your code at certain points during request/response validation via hooks.

Available hooks:

- `after_request_validation`
- `after_response_validation`
- `after_request_parameter_property_validation`
- `after_request_body_property_validation`
- `after_response_body_property_validation`

Setup per per instance:

```ruby
OpenapiFirst.load('openapi.yaml') do |config|
  config.after_request_validation do |validated_request|
    validated_request.valid? # => true / false
  end
  config.after_response_validation do |validated_response, request|
    if validated_response.invalid?
      warn "#{request.request_method} #{request.path}: #{validated_response.error.message}"
    end
  end
end
```

Setup globally:

```ruby
OpenapiFirst.configure do |config|
  config.after_request_parameter_property_validation do |data, property, property_schema|
    data[property] = Date.iso8601(data[property]) if property_schema['format'] == 'date'
  end
end
```

## Framework integration

Using rack middlewares is supported in probably all Ruby web frameworks.

The contract testing feature is designed to be used via rack-test, which should be compatible all Ruby web frameworks as well.

That aside, closer integration with specific frameworks like Sinatra, Hanami, Roda or others would be great. If you have ideas, pain points or PRs, please don't hesitate to [share](https://github.com/ahx/openapi_first/discussions).

## Alternatives

This gem was inspired by [committee](https://github.com/interagent/committee) (Ruby) and [Connexion](https://github.com/spec-first/connexion) (Python).
Here is a [feature comparison between openapi_first and committee](https://gist.github.com/ahx/1538c31f0652f459861713b5259e366a).

## Frequently Asked Questions

### How can I adapt request paths that don't match my schema?

Let's say you have `openapi.yaml` like this:

```yaml
servers:
  - url: https://yourhost/api
paths:
  # The actual endpoint URL is https://yourhost/api/resource
  /resource:
```

Here your OpenAPI schema defines endpoints starting with `/resource` but your actual application is mounted at `/api/resource`. You can bridge the gap by transforming the path via the `path:` configuration:

```ruby
oad = OpenapiFirst.load('openapi.yaml') do |config|
  config.path = ->(req) { request.path.delete_prefix('/api') }
end
use OpenapiFirst::Middlewares::RequestValidation, oad
```

## Development

Run `git submodule update --init` to initialize the git submodules.

Run `bin/setup` to install dependencies.

See `bundle exec rake` to run the linter and the tests.

Run `bundle exec rspec` to run the tests only.

### Benchmarks

[Results](https://gist.github.com/ahx/e6ffced58bd2e8d5baffb2f4d2c1f823)

Run benchmarks:

```sh
cd benchmarks
bundle
bundle exec ruby benchmarks.rb
```

### Contributing

If you have a question or an idea or found a bug, don't hesitate to create an issue [on Github](https://github.com/ahx/openapi_first) or [Codeberg](https://codeberg.org/ahx/openapi_first) or say hi on [Mastodon (ruby.social)](https://ruby.social/@ahx).

Pull requests are very welcome as well, of course. Feel free to create a "draft" pull request early on, even if your change is still work in progress. 🤗
 readmeEtag: '"bc951a2a1156ca6e4ec76f803900d4031a088d0c"' readmeLastModified: Wed, 07 Jan 2026 12:44:41 GMT repositoryId: 171639126 description: >- openapi_first is a Ruby gem for request / response validation and contract-testing against an OpenAPI API description. It makes APIFirst easy and reliable. created: '2019-02-20T09:11:47Z' updated: '2026-01-27T06:59:30Z' language: Ruby archived: false stars: 208 watchers: 3 forks: 23 owner: ahx logo: https://avatars.githubusercontent.com/u/8669?v=4 license: MIT repoEtag: '"7fa48c8f5a915c257030a3af138fbd91b6430fd43e83db5b233efc7f36f67abc"' repoLastModified: Tue, 27 Jan 2026 06:59:30 GMT foundInMaster: true category: Data Validators id: d8781910209a542c2f846aa8c740baf2 name: openapi_first language: Ruby source_description: >- It validates requests and responses against your API description using rack middlewares or a low-level interface. It gives you access to request parameters that are parsed exactly as described in your API description and produces useful customizable error responses if request validation fails. v3_1: true - source: https://openapi.tools/ name: openVALIDATION category: - Description Validators - Data Validators - Parsers repository: https://github.com/openvalidation/openvalidation-openapi link: https://docs.openvalidation.io/openapi/openapi-specification language: Java source_description: >- Allows complex validation rules to be specified in openAPI spec files using natural language. v3: true repositoryMetadata: base64Readme: >- WyFbQnVpbGQgU3RhdHVzXShodHRwczovL2Rldi5henVyZS5jb20vdmFsaWRhcmlhL29wZW52YWxpZGF0aW9uL19hcGlzL2J1aWxkL3N0YXR1cy9vcGVuVkFMSURBVElPTiUyME9wZW5BUEkvb3BlblZBTElEQVRJT04lMjBPcGVuQVBJJTIwbWFzdGVyP2JyYW5jaE5hbWU9bWFzdGVyKV0oaHR0cHM6Ly9kZXYuYXp1cmUuY29tL3ZhbGlkYXJpYS9vcGVudmFsaWRhdGlvbi9fYnVpbGQvbGF0ZXN0P2RlZmluaXRpb25JZD0zJmJyYW5jaE5hbWU9bWFzdGVyKQohW0F6dXJlIERldk9wcyB0ZXN0cyAoY29tcGFjdCldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYXp1cmUtZGV2b3BzL3Rlc3RzL3ZhbGlkYXJpYS9vcGVudmFsaWRhdGlvbi8zP2NvbXBhY3RfbWVzc2FnZSkKWyFbTWF2ZW4gQ2VudHJhbF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3YvaW8ub3BlbnZhbGlkYXRpb24vb3BlbnZhbGlkYXRpb24tb3BlbmFwaS1nZW5lcmF0b3IpXShodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9Zzppby5vcGVudmFsaWRhdGlvbikKWyFbRm9sbG93IHVzIG9uIFR3aXR0ZXJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdHdpdHRlci9mb2xsb3cvb3BlblZBTElEQVRJT04/c3R5bGU9c29jaWFsKV0oaHR0cHM6Ly90d2l0dGVyLmNvbS9vcGVuVkFMSURBVElPTikKCgojIE9wZW5BUEkgR2VuZXJhdG9yIGZvciBvcGVuVkFMSURBVElPTgoKVGhpcyBwcm9qZWN0IHByb3ZpZGVzIGFuIG9wZW5hcGktZ2VuZXJhdG9yIGludGVncmF0aW9uIHRvIGdlbmVyYXRlIHNlcnZpY2VzIHdpdGggdmFsaWRhdGlvbiBsb2dpYyBwcm92aWRlZCBieSBvcGVuVkFMSURBVElPTi4KCiMjIFdoYXQncyBPcGVuQVBJClRoZSBnb2FsIG9mIE9wZW5BUEkgaXMgdG8gZGVmaW5lIGEgc3RhbmRhcmQsIGxhbmd1YWdlLWFnbm9zdGljIGludGVyZmFjZSB0byBSRVNUIEFQSXMgd2hpY2ggYWxsb3dzIGJvdGggaHVtYW5zIGFuZCBjb21wdXRlcnMgdG8gZGlzY292ZXIgYW5kIHVuZGVyc3RhbmQgdGhlIGNhcGFiaWxpdGllcyBvZiB0aGUgc2VydmljZSB3aXRob3V0IGFjY2VzcyB0byBzb3VyY2UgY29kZSwgZG9jdW1lbnRhdGlvbiwgb3IgdGhyb3VnaCBuZXR3b3JrIHRyYWZmaWMgaW5zcGVjdGlvbi4KV2hlbiBwcm9wZXJseSBkZXNjcmliZWQgd2l0aCBPcGVuQVBJLCBhIGNvbnN1bWVyIGNhbiB1bmRlcnN0YW5kIGFuZCBpbnRlcmFjdCB3aXRoIHRoZSByZW1vdGUgc2VydmljZSB3aXRoIGEgbWluaW1hbCBhbW91bnQgb2YgaW1wbGVtZW50YXRpb24gbG9naWMuClNpbWlsYXIgdG8gd2hhdCBpbnRlcmZhY2VzIGhhdmUgZG9uZSBmb3IgbG93ZXItbGV2ZWwgcHJvZ3JhbW1pbmcsIE9wZW5BUEkgcmVtb3ZlcyB0aGUgZ3Vlc3N3b3JrIGluIGNhbGxpbmcgdGhlIHNlcnZpY2UuCgpDaGVjayBvdXQgW09wZW5BUEktU3BlY10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24pIGZvciBhZGRpdGlvbmFsIGluZm9ybWF0aW9uIGFib3V0IHRoZSBPcGVuQVBJIHByb2plY3QsIGluY2x1ZGluZyBhZGRpdGlvbmFsIGxpYnJhcmllcyB3aXRoIHN1cHBvcnQgZm9yIG90aGVyIGxhbmd1YWdlcyBhbmQgbW9yZS4gCgojIyBXaGF0J3Mgb3BlblZBTElEQVRJT04KCkNoZWNrIG91dCBbb3BlblZBTElEQVRJT05dKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVudmFsaWRhdGlvbi9vcGVudmFsaWRhdGlvbikgdG8gbGVhcm4gYWJvdXQgaHVtYW4gcmVhZGFibGUgdmFsaWRhdGlvbiBydWxlcyBmb3Igc29mdHdhcmUgc29sdXRpb25zLgoKIyMgb3BlblZBTElEQVRJT04gT3BlbkFQSSBDb2RlZ2VuIERvY3VtZW50YXRpb24KCltEb2N1bWVudGF0aW9uXShodHRwczovL2RvY3Mub3BlbnZhbGlkYXRpb24uaW8vb3BlbmFwaS9vcGVuYXBpLXNwZWNpZmljYXRpb24pCgpbVHV0b3JpYWxdKGh0dHBzOi8vZG9jcy5vcGVudmFsaWRhdGlvbi5pby9vcGVuYXBpL29wZW5hcGktdHV0b3JpYWwpCgojIyBIb3cgZG8gSSB1c2UgdGhpcz8KCkRvd25sb2FkIHRoZSBbb3BlblZBTElEQVRJT04gT3BlbkFQSSBnZW5lcmF0b3IgQ0xJXShodHRwczovL2Rvd25sb2FkYXJjaGl2ZS5ibG9iLmNvcmUud2luZG93cy5uZXQvb3BlbnZhbGlkYXRpb24tb3BlbmFwaS1nZW5lcmF0b3Ivb3Ytb3BlbmFwaS1nZW5lcmF0b3ItY2xpLmphcikgKHJlcXVpcmVzIHRoZSBKYXZhIFNFIDggcnVudGltZSBlbnZpcm9ubWVudCkuClVzZSBgb3Ytb3BlbmFwaS1nZW5lcmF0b3ItY2xpLmphcmAgYXMgYSBkcm9wLWluIHJlcGxhY2VtZW50IGZvciB0aGUgYG9wZW5hcGktZ2VuZXJhdG9yLWNsaS5qYXJgCgpOb3cgeW91IGNhbiB1c2Ugb3BlblZBTElEQVRJT04gcnVsZXMgaW4geW91ciBzZXJ2aWNlIGNvbnRyYWN0OgpgYGB5YW1sCnBhdGhzOgogIC86IAogICBwb3N0OgogICAgICByZXF1ZXN0Qm9keToKICAgICAgICBjb250ZW50OgogICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgICRyZWY6ICcjL2NvbXBvbmVudHMvc2NoZW1hcy9hcHBsaWNhbnQnCiAgICAgICAgICAgIHgtb3YtcnVsZXM6CiAgICAgICAgICAgICAgY3VsdHVyZTogZW4KICAgICAgICAgICAgICBydWxlOiB8CiAgICAgICAgICAgICAgICB0aGUgbG9jYXRpb24gb2YgdGhlIGFwcGxpY2FudCBtdXN0IGJlIERvcnRtdW5kCiAgICAgIHJlc3BvbnNlczoKICAgICAgICAnMjAwJzoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBzdWNjZXNzCmNvbXBvbmVudHM6CiAgc2NoZW1hczoKICAgIGFwcGxpY2FudDoKICAgICAgdHlwZTogb2JqZWN0CiAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgbmFtZToKICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgIGFnZToKICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgICAgICBsb2NhdGlvbjoKICAgICAgICAgIHR5cGU6IHN0cmluZwpgYGAKKihGb3IgZnVydGhlciBkZXRhaWxzIGNoZWNrIHRoZSBhYm92ZS1tZW50aW9uZWQgZG9jdW1lbnRhdGlvbiBvciB0aGUgb3BlblZBTElEQVRJT04gcHJvamVjdCBpdHNlbGYuKSoKCkFsdGVybmF0aXZlbHksIHRoZSBgb3BlbnZhbGlkYXRpb24tb3BlbmFwaS1nZW5lcmF0b3JgIGphciBpdHNlbGYgY2FuIGJlIGNhbGxlZCBpbiBjb21iaW5hdGlvbiB3aXRoIHRoZSBPcGVuQVBJIGdlbmVyYXRvci4KCkZvciBtYWMvbGludXg6CmBgYApqYXZhIC1jcCAvcGF0aC90by9vcGVuYXBpLWdlbmVyYXRvci1jbGkuamFyOi9wYXRoL3RvL29wZW52YWxpZGF0aW9uLW9wZW5hcGktZ2VuZXJhdG9yLmphciBvcmcub3BlbmFwaXRvb2xzLmNvZGVnZW4uT3BlbkFQSUdlbmVyYXRvciBnZW5lcmF0ZSAtZyBvdi1qYXZhLXNwcmluZy1zZXJ2ZXIgLWkgL3BhdGgvdG8vb3BlbmFwaS55YW1sIC1vIC4vdGVzdApgYGAKKERvIG5vdCBmb3JnZXQgdG8gcmVwbGFjZSB0aGUgdmFsdWVzIGAvcGF0aC90by9vcGVuYXBpLWdlbmVyYXRvci1jbGkuamFyYCwgYC9wYXRoL3RvL29wZW52YWxpZGF0aW9uLW9wZW5hcGktZ2VuZXJhdG9yLmphcmAgYW5kIGAvcGF0aC90by9vcGVuYXBpLnlhbWxgIGluIHRoZSBwcmV2aW91cyBjb21tYW5kKQoKRm9yIFdpbmRvd3MgdXNlcnMsIHlvdSB3aWxsIG5lZWQgdG8gdXNlIGA7YCBpbnN0ZWFkIG9mIGA6YCBpbiB0aGUgY2xhc3NwYXRoLCBlLmcuCmBgYApqYXZhIC1jcCAvcGF0aC90by9vcGVuYXBpLWdlbmVyYXRvci1jbGkuamFyOy9wYXRoL3RvL29wZW52YWxpZGF0aW9uLW9wZW5hcGktZ2VuZXJhdG9yLmphciBvcmcub3BlbmFwaXRvb2xzLmNvZGVnZW4uT3BlbkFQSUdlbmVyYXRvciBnZW5lcmF0ZSAtZyBvdi1qYXZhLXNwcmluZy1zZXJ2ZXIgLWkgL3BhdGgvdG8vb3BlbmFwaS55YW1sIC1vIC4vdGVzdApgYGAKCk9yIHVzZSBvcGVuVkFMSURBVElPTiBHZW5lcmF0b3IgYXMgTWF2ZW4gUGx1Z2luOgpgYGB4bWwKICAgIDxidWlsZD4KICAgICAgICA8cGx1Z2lucz4KICAgICAgICAgICAgPHBsdWdpbj4KICAgICAgICAgICAgICAgIDxncm91cElkPm9yZy5vcGVuYXBpdG9vbHM8L2dyb3VwSWQ+CiAgICAgICAgICAgICAgICA8YXJ0aWZhY3RJZD5vcGVuYXBpLWdlbmVyYXRvci1tYXZlbi1wbHVnaW48L2FydGlmYWN0SWQ+CiAgICAgICAgICAgICAgICA8dmVyc2lvbj4zLjMuNDwvdmVyc2lvbj4KICAgICAgICAgICAgICAgIDxleGVjdXRpb25zPgogICAgICAgICAgICAgICAgICAgIDxleGVjdXRpb24+CiAgICAgICAgICAgICAgICAgICAgICAgIDxnb2Fscz4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxnb2FsPmdlbmVyYXRlPC9nb2FsPgogICAgICAgICAgICAgICAgICAgICAgICA8L2dvYWxzPgogICAgICAgICAgICAgICAgICAgICAgICA8Y29uZmlndXJhdGlvbj4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxpbnB1dFNwZWM+JHtwcm9qZWN0LmJhc2VkaXJ9L215LnNwZWMueWFtbDwvaW5wdXRTcGVjPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPGdlbmVyYXRvck5hbWU+b3YtamF2YS1ydWxlczwvZ2VuZXJhdG9yTmFtZT4gPCEtIHJ1bGVzIG9ubHkgZ2VuZXJhdGlvbiAtLT4KICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxjb25maWdPcHRpb25zPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIDxpbnZva2VyUGFja2FnZT5teS5jdXN0b20ucGFja2FnZTwvaW52b2tlclBhY2thZ2U+CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgPG1vZGVsUGFja2FnZT5teS5jdXN0b20ucGFja2FnZS5tb2RlbDwvbW9kZWxQYWNrYWdlPgogICAgICAgICAgICAgICAgICAgICAgICAgICAgPC9jb25maWdPcHRpb25zPgogICAgICAgICAgICAgICAgICAgICAgICA8L2NvbmZpZ3VyYXRpb24+CiAgICAgICAgICAgICAgICAgICAgPC9leGVjdXRpb24+CiAgICAgICAgICAgICAgICA8L2V4ZWN1dGlvbnM+CiAgICAgICAgICAgICAgICA8ZGVwZW5kZW5jaWVzPgogICAgICAgICAgICAgICAgICAgIDxkZXBlbmRlbmN5PgogICAgICAgICAgICAgICAgICAgICAgICA8Z3JvdXBJZD5pby5vcGVudmFsaWRhdGlvbjwvZ3JvdXBJZD4KICAgICAgICAgICAgICAgICAgICAgICAgPGFydGlmYWN0SWQ+b3BlbnZhbGlkYXRpb24tb3BlbmFwaS1nZW5lcmF0b3I8L2FydGlmYWN0SWQ+CiAgICAgICAgICAgICAgICAgICAgICAgIDx2ZXJzaW9uPjAuMC4xPC92ZXJzaW9uPgogICAgICAgICAgICAgICAgICAgIDwvZGVwZW5kZW5jeT4KICAgICAgICAgICAgICAgIDwvZGVwZW5kZW5jaWVzPgogICAgICAgICAgICA8L3BsdWdpbj4KICAgICAgICA8L3BsdWdpbnM+CiAgICA8L2J1aWxkPgpgYGAKCgojIyBHZXR0aW5nIGludm9sdmVkCgpQbGVhc2UgcmVmZXIgdG8gb3VyIFtjb250cmlidXRpb24gZ3VpZGVsaW5lc10oQ09OVFJJQlVUSU5HLm1kKS4KCiMjIENvbnRhY3QKCllvdSBjYW4gd3JpdGUgYW4gW0UtTWFpbF0obWFpbHRvOnZhbGlkYXJpYUBvcGVudmFsaWRhdGlvbi5pbykgb3IgbWVudGlvbiBvdXIgdHdpdHRlciBhY2NvdW50IFtAb3BlblZBTElEQVRJT05dKGh0dHBzOi8vdHdpdHRlci5jb20vb3BlbnZhbGlkYXRpb24pLgo= readmeEtag: '"ff79107deef69e96cf5810a882930e2902173b98"' readmeLastModified: Thu, 05 Mar 2020 16:36:31 GMT repositoryId: 204974336 description: >- Generate OpenAPI service stubs with complete validation layer using this openVALIDATION adapter. created: '2019-08-28T16:18:50Z' updated: '2025-04-18T19:35:30Z' language: Java archived: false stars: 5 watchers: 1 forks: 2 owner: openvalidation logo: https://avatars.githubusercontent.com/u/50744669?v=4 license: Apache-2.0 repoEtag: '"7dca9dc961f12ee437efcaf6b1816a4ef03bc188d8a24cce5ad35debaa54c1fd"' repoLastModified: Fri, 18 Apr 2025 19:35:30 GMT foundInMaster: true id: 7a685d5b3cc00e5c26420228fb5f8f00 - source: - https://openapi.tools/ - openapi3 tags name: swagger-parser homepage: https://github.com/swagger-api/swagger-parser language: Java source_description: Swagger Parser reads OpenAPI definitions into current Java POJOs. category: Parsers repository: https://github.com/swagger-api/swagger-parser v2: true v3: true repositoryMetadata: base64Readme: >- # Swagger Parser <img src="https://raw.githubusercontent.com/swagger-api/swagger.io/wordpress/images/assets/SW-logo-clr.png" height="50" align="right">

**NOTE:** If you're looking for `swagger-parser` 1.X and OpenAPI 2.0, please refer to [v1 branch](https://github.com/swagger-api/swagger-parser/tree/v1)

**NOTE:** Since version 2.1.0 Swagger Parser supports OpenAPI 3.1; see [this page](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---OpenAPI-3.1) for details

![Build Master - Java 11, 14 and 17](https://github.com/swagger-api/swagger-parser/workflows/Build%20Test%20Deploy%20master/badge.svg?branch=master)

# Table of contents

  - [Overview](#overview)
  - [Table of Contents](#table-of-contents)
  - [Usage](#usage)
  - [Adding to your project](#adding-to-your-project)
    - [Prerequisites](#prerequisites)
  - [Authentication](#authentication)  
  - [Options](#options)
    - [Resolve](#1-resolve)
    - [ResolveFully](#2-resolvefully)
    - [Flatten](#3-flatten)
    - [ResolveCombinators](#4-resolvecombinators)
  - [Extensions](#extensions)
  - [OpenAPI 3.1 Support](#openapi-31-support)
  - [License](#license)
   
## Overview 

This is the Swagger Parser project, which parses OpenAPI definitions in JSON or YAML format into [swagger-core](https://github.com/swagger-api/swagger-core) representation as [Java POJO](https://github.com/swagger-api/swagger-core/blob/master/modules/swagger-models/src/main/java/io/swagger/v3/oas/models/OpenAPI.java#L36), returning any validation warnings/errors.  

It also provides a simple framework to add additional converters from different formats into the Swagger objects, making the entire toolchain available.


### Usage
Using the Swagger Parser is simple.  Once included in your project, you can read a OpenAPI Specification from any location:

```java
import io.swagger.parser.OpenAPIParser;
import io.swagger.v3.parser.OpenAPIV3Parser;
import io.swagger.v3.parser.core.models.SwaggerParseResult;
import io.swagger.v3.oas.models.OpenAPI;

// ... your code

  // parse a swagger description from the petstore and get the result
  SwaggerParseResult result = new OpenAPIParser().readLocation("https://petstore3.swagger.io/api/v3/openapi.json", null, null);
  
  // or from a file
  //   SwaggerParseResult result = new OpenAPIParser().readLocation("./path/to/openapi.yaml", null, null);
  
  // the parsed POJO
  OpenAPI openAPI = result.getOpenAPI();
  
  if (result.getMessages() != null) result.getMessages().forEach(System.err::println); // validation errors and warnings
  
  if (openAPI != null) {
    ...
  }
  
```

or from a string:

```java
import io.swagger.parser.OpenAPIParser;
import io.swagger.v3.parser.OpenAPIV3Parser;
import io.swagger.v3.parser.core.models.SwaggerParseResult;
import io.swagger.v3.oas.models.OpenAPI;

// ... your code

  // parse a swagger description from the petstore and get the result
  SwaggerParseResult result = new OpenAPIParser().readContents("https://petstore3.swagger.io/api/v3/openapi.json", null, null);
  
  // or from a file
  //   SwaggerParseResult result = new OpenAPIParser().readContents("./path/to/openapi.yaml", null, null);
  
  // the parsed POJO
  OpenAPI openAPI = result.getOpenAPI();
  
  if (result.getMessages() != null) result.getMessages().forEach(System.err::println); // validation errors and warnings
  
  if (openAPI != null) {
    ...
  }
  
```

If you are providing a Swagger/OpenAPI 2.0 document to the parser , e.g.:

```java
SwaggerParseResult result = new OpenAPIParser().readContents("./path/to/swagger.yaml", null, null);
```

the Swagger/OpenAPI 2.0 document will be first converted into a comparable OpenAPI 3.0 one.

You can also directly use `OpenAPIV3Parser` which only handles OpenAPI 3.0 documents, and provides a convenience method to get directly the parsed `OpenAPI object:

```java
import io.swagger.v3.parser.OpenAPIV3Parser;
import io.swagger.v3.oas.models.OpenAPI;

// ... your code

  // read a swagger description from the petstore
    
  OpenAPI openAPI = new OpenAPIV3Parser().read("https://petstore3.swagger.io/api/v3/openapi.json");
  
```

### Adding to your project
You can include this library from Sonatype OSS for SNAPSHOTS, or Maven central for releases.  In your dependencies:

```xml
<dependency>
  <groupId>io.swagger.parser.v3</groupId>
  <artifactId>swagger-parser</artifactId>
  <version>2.1.37</version>
</dependency>
```

#### Prerequisites
You need the following installed and available in your $PATH:

* Java 11
* [Apache maven 3.x](http://maven.apache.org/)

After cloning the project, you can build it from source with this command:

```
mvn package
```

### Authentication

If your OpenAPI definition is protected, you can pass headers in the request:
```java
import io.swagger.v3.parser.core.models.AuthorizationValue;

// ... your code

  // build a authorization value
  AuthorizationValue mySpecialHeader = new AuthorizationValue()
    .keyName("x-special-access")  //  the name of the authorization to pass
    .value("i-am-special")        //  the value of the authorization
    .type("header");              //  the location, as either `header` or `query`

  // or in a single constructor
  AuthorizationValue apiKey = new AuthorizationValue("api_key", "special-key", "header");
  OpenAPI openAPI = new OpenAPIV3Parser().readWithInfo(
    "https://petstore3.swagger.io/api/v3/openapi.json",
    Arrays.asList(mySpecialHeader, apiKey)
  );
```

#### Dealing with self-signed SSL certificates
If you're dealing with self-signed SSL certificates, or those signed by GoDaddy, you'll need to disable SSL Trust 
Manager.  That's done by setting a system environment variable as such:

```
export TRUST_ALL=true
```

And then the Swagger Parser will _ignore_ invalid certificates.  Of course this is generally a bad idea, but if you're 
working inside a firewall or really know what you're doing, well, there's your rope.

#### Dealing with Let's Encrypt
Depending on the version of Java that you use, certificates signed by the [Let's Encrypt](https://letsencrypt.org) certificate authority _may not work_ by default.  If you are using any version of Java prior to 1.8u101, you most likely _must_ install an additional CA in your
JVM.  Also note that 1.8u101 may _not_ be sufficient on it's own.  Some users have reported that certain operating systems are 
not accepting Let's Encrypt signed certificates.

Your options include:

* Accepting all certificates per above
* Installing the certificate manually in your JVM using the keystore using the `keytool` command
* Configuring the JVM on startup to load your certificate

But... this is all standard SSL configuration stuff and is well documented across the web.


### Options
Parser uses options as a way to customize the behavior while parsing:

#### 1. resolve:

```java
ParseOptions parseOptions = new ParseOptions();
parseOptions.setResolve(true); 
final OpenAPI openAPI = new OpenAPIV3Parser().read("a.yaml", null, parseOptions);
```


- When remote or relative references are found in the parsed document, parser will attempt to:

1. resolve the reference in the remote or relative location 
1. parse the resolved reference
1. add the resolved "component" (e.g. parameter, schema, response, etc.) to the resolved `OpenAPI` POJO components section
1. replace the remote/relative reference with a local reference,  e.g. : `#/components/schemas/NameOfRemoteSchema`. 

This applies to schemas, parameters, responses, pretty much everything containing a ref.

#### 2. resolveFully:

```java
ParseOptions parseOptions = new ParseOptions();
parseOptions.setResolve(true); // implicit
parseOptions.setResolveFully(true);
final OpenAPI openAPI = new OpenAPIV3Parser().read("a.yaml", null, parseOptions);
```

- In some scenarios, after references are resolved (with `resolve`, see above), you might need to have all local references removed replacing the reference with the content of the referenced element. This is for example used in [Swagger Inflector](https://github.com/swagger-api/swagger-inflector). Be aware that the result could be more heavy/long due to duplication
    
Original document:

`a.yaml` 
```
openapi: 3.0.1
paths:
  "/newPerson":
    post:
      summary: Create new person
      description: Create new person
      responses:
        '200':
          description: ok
          content:
            "*/*":
              schema:
                "$ref": "./ref-without-component/b.yaml#/components/schemas/CustomerType"
```
`b.yaml`
```
openapi: 3.0.1
components:
  schemas:
    CustomerType:
      type: string
      example: Example value
```

Serialized result after parsing with option `resolveFully(true)`

`a.yaml`
```
openapi: 3.0.1
servers:
- url: /
paths:
  /newPerson:
    post:
      summary: Create new person
      description: Create new person
      responses:
        200:
          description: ok
          content:
            '*/*':
              schema:
                type: string
                example: Example value
components:
  schemas:
    CustomerType:
      type: string
      example: Example value
```

#### 3. flatten: 

```java
ParseOptions parseOptions = new ParseOptions();
parseOptions.setFlatten(true); 
final OpenAPI openAPI = new OpenAPIV3Parser().read("a.yaml", null, parseOptions);
```


This is kind of the opposite of resolveFully, limited to defined schemas.

In some scenarios, you might need to have all schemas defined inline (e.g. a response schema) moved to the `components/schemas` section and replaced with a reference to the newly added schema within `components/schemas`. This is for example used in [Swagger Codegen](https://github.com/swagger-api/swagger-codegen).

Original document:

`flatten.yaml`

```
openapi: 3.0.0
info:
  version: 1.0.0
  title: Swagger Petstore
  license:
    name: MIT
paths:
  /pets:
    get:
      summary: List all pets
      operationId: listPets
      responses:
        '200':
          description: An paged array of pets
          headers:
            x-next:
              description: A link to the next page of responses
              schema:
                type: string
          content:
            application/json:
              schema:
                 type: object
                 properties:
                    id:
                      type: integer
                      format: int64
                    name:
                      type: string
                    tag:
                      type: string
```

Serialized result after parsing with option `flatten(true)`

```
openapi: 3.0.0
info:
  title: Swagger Petstore
  license:
    name: MIT
  version: 1.0.0
servers:
- url: /
paths:
  /pets:
    get:
      tags:
      - pets
      summary: List all pets
      responses:
        200:
          description: An paged array of pets
          headers:
            x-next:
              description: A link to the next page of responses
              style: simple
              explode: false
              schema:
                type: string
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/inline_response_200'
components:
  schemas:
    inline_response_200:
      type: object
      properties:
        id:
          type: integer
          format: int64
        name:
          type: string
        tag:
          type: string
```

#### 4. resolveCombinators: 

```java
ParseOptions parseOptions = new ParseOptions();
parseOptions.setResolve(true); // implicit
parseOptions.setResolveFully(true);
parseOptions.setResolveCombinators(false); // default is true 
final OpenAPI openAPI = new OpenAPIV3Parser().read("a.yaml", null, parseOptions);
```

This option (only available with `resolveFully = true`) allows to customize behaviour related to `allOf/anyOf/oneOf` (composed schemas)  processing. With option set to `true` (default), composed schemas are transformed into "non composed" ones, by having all properties merged into a single resulting schema (see example below).
If option is set to `false`, the resulting schema will instead maintain its "composed" nature, keeping properties within e.g. the `allOf` members.

Please see examples below:

**Unresolved yaml**

```
openapi: 3.0.1
servers:
- url: http://petstore.swagger.io/api

info:
  description: 'This is a sample server Petstore'
  version: 1.0.0
  title: testing source file
  termsOfService: http://swagger.io/terms/

paths:
  "/withInvalidComposedModel":
    post:
      operationId: withInvalidComposedModel
      requestBody:
        content:
          "application/json":
            schema:
              "$ref": "#/components/schemas/ExtendedAddress"
        required: false
      responses:
        '200':
          description: success!
components:
  schemas:
    ExtendedAddress:
      type: object
      allOf:
        - $ref: '#/components/schemas/Address'
        - type: object
          required:
          - gps
          properties:
            gps:
              type: string
    Address:
      required:
      - street
      type: object
      properties:
        street:
          type: string
          example: 12345 El Monte Road
        city:
          type: string
          example: Los Altos Hills
        state:
          type: string
          example: CA
        zip:
          type: string
          example: '94022'
```

**resolvedCombinator = true (default) - Test case**

```
@Test
    public void resolveAllOfWithoutAggregatingParameters(@Injectable final List<AuthorizationValue> auths) {
        ParseOptions options = new ParseOptions();
        options.setResolveFully(true);
        options.setResolveCombinators(true);

        // Testing components/schemas
        OpenAPI openAPI = new OpenAPIV3Parser().readLocation("src/test/resources/composed.yaml",auths,options).getOpenAPI();
        
        ComposedSchema allOf = (ComposedSchema) openAPI.getComponents().getSchemas().get("ExtendedAddress");
        assertEquals(allOf.getAllOf().size(), 2);

        assertTrue(allOf.getAllOf().get(0).get$ref() != null);
        assertTrue(allOf.getAllOf().get(1).getProperties().containsKey("gps"));


        // Testing path item
        ObjectSchema schema = (ObjectSchema) openAPI.getPaths().get("/withInvalidComposedModel").getPost().getRequestBody().getContent().get("application/json").getSchema();

        assertEquals(schema.getProperties().size(), 5);
        assertTrue(schema.getProperties().containsKey("street"));
        assertTrue(schema.getProperties().containsKey("gps"));

    }
```

**resolvedCombinator = true (default) - Resolved Yaml**

```
openapi: 3.0.1
info:
  title: testing source file
  description: This is a sample server Petstore
  termsOfService: http://swagger.io/terms/
  version: 1.0.0
servers:
- url: http://petstore.swagger.io/api
paths:
  /withInvalidComposedModel:
    post:
      operationId: withInvalidComposedModel
      requestBody:
        content:
          application/json:
            schema:
              required:
              - gps
              - street
              type: object
              properties:
                street:
                  type: string
                  example: 12345 El Monte Road
                city:
                  type: string
                  example: Los Altos Hills
                state:
                  type: string
                  example: CA
                zip:
                  type: string
                  example: "94022"
                gps:
                  type: string
        required: false
      responses:
        200:
          description: success!
components:
  schemas:
    ExtendedAddress:
      type: object
      allOf:
      - $ref: '#/components/schemas/Address'
      - required:
        - gps
        type: object
        properties:
          gps:
            type: string
    Address:
      required:
      - street
      type: object
      properties:
        street:
          type: string
          example: 12345 El Monte Road
        city:
          type: string
          example: Los Altos Hills
        state:
          type: string
          example: CA
        zip:
          type: string
          example: "94022"
 ```
 
 **resolvedCombinator = false - Test case**
 
 ```
 @Test
    public void resolveAllOfWithoutAggregatingParameters(@Injectable final List<AuthorizationValue> auths) {
        ParseOptions options = new ParseOptions();
        options.setResolveFully(true);
        options.setResolveCombinators(false);

        // Testing components/schemas
        OpenAPI openAPI = new OpenAPIV3Parser().readLocation("src/test/resources/composed.yaml",auths,options).getOpenAPI();
       
        ComposedSchema allOf = (ComposedSchema) openAPI.getComponents().getSchemas().get("ExtendedAddress");
        assertEquals(allOf.getAllOf().size(), 2);
        assertTrue(allOf.getAllOf().get(0).getProperties().containsKey("street"));
        assertTrue(allOf.getAllOf().get(1).getProperties().containsKey("gps"));

        // Testing path item
        ComposedSchema schema = (ComposedSchema) openAPI.getPaths().get("/withInvalidComposedModel").getPost().getRequestBody().getContent().get("application/json").getSchema();
        // In fact the schema resolved previously is the same of /withInvalidComposedModel
        assertEquals(schema, allOf);
        assertEquals(schema.getAllOf().size(), 2);
        assertTrue(schema.getAllOf().get(0).getProperties().containsKey("street"));
        assertTrue(schema.getAllOf().get(1).getProperties().containsKey("gps"));

    }
  ```
  
  **resolvedCombinator = false - Resolved Yaml**
  
  ```
openapi: 3.0.1
info:
  title: testing source file
  description: This is a sample server Petstore
  termsOfService: http://swagger.io/terms/
  version: 1.0.0
servers:
- url: http://petstore.swagger.io/api
paths:
  /withInvalidComposedModel:
    post:
      operationId: withInvalidComposedModel
      requestBody:
        content:
          application/json:
            schema:
              type: object
              allOf:
              - required:
                - street
                type: object
                properties:
                  street:
                    type: string
                    example: 12345 El Monte Road
                  city:
                    type: string
                    example: Los Altos Hills
                  state:
                    type: string
                    example: CA
                  zip:
                    type: string
                    example: "94022"
              - required:
                - gps
                type: object
                properties:
                  gps:
                    type: string
        required: false
      responses:
        200:
          description: success!
components:
  schemas:
    ExtendedAddress:
      type: object
      allOf:
      - required:
        - street
        type: object
        properties:
          street:
            type: string
            example: 12345 El Monte Road
          city:
            type: string
            example: Los Altos Hills
          state:
            type: string
            example: CA
          zip:
            type: string
            example: "94022"
      - required:
        - gps
        type: object
        properties:
          gps:
            type: string
    Address:
      required:
      - street
      type: object
      properties:
        street:
          type: string
          example: 12345 El Monte Road
        city:
          type: string
          example: Los Altos Hills
        state:
          type: string
          example: CA
        zip:
          type: string
          example: "94022"
```
#### 5. explicitObjectSchema:

```java
ParseOptions parseOptions = new ParseOptions();
parseOptions.setResolve(true); // implicit
parseOptions.setResolveFully(true);
parseOptions.setResolveCombinators(false);
parseOptions.setExplicitObjectSchema(false); // default is true
final OpenAPI openAPI = new OpenAPIV3Parser().read("a.yaml", null, parseOptions);
```

This option allows you to customize the processing of schema properties when the type is not specified. By default it is set to `true`.
- `true` : when the type is not defined for property, ‘object’ is set.
- `false` : the property remains undefined when no type is specified
It's only applied when `resolveFully` is set to `true`.

### Extensions
This project has a core artifact--`swagger-parser`, which uses Java Service Provider Interface (SPI) so additional extensions can be added. 

To build your own extension, you simply need to create a `src/main/resources/META-INF/services/io.swagger.v3.parser.core.extensions.SwaggerParserExtension` file with the full classname of your implementation.  Your class must also implement the `io.swagger.v3.parser.core.extensions.SwaggerParserExtension` interface.  Then, including your library with the `swagger-parser` module will cause it to be triggered automatically.

### OpenAPI 3.1 support

Since version 2.1.0 Swagger Parser supports OpenAPI 3.1; see [this page](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---OpenAPI-3.1) for details

## Security contact

Please disclose any security-related issues or vulnerabilities by emailing [security@swagger.io](mailto:security@swagger.io), instead of using the public issue tracker.
 readmeEtag: '"89cc138da6fc9ccbbf5df1cfa9c2dc963a38dd3d"' readmeLastModified: Thu, 18 Dec 2025 13:31:56 GMT repositoryId: 18997610 description: Swagger Spec to Java POJOs created: '2014-04-21T16:00:22Z' updated: '2026-02-04T09:37:17Z' language: Java archived: false stars: 851 watchers: 43 forks: 539 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"64e7360ad64407cba021be68c30e0045fd0047e36d7b35d9c0c965844023dba7"' repoLastModified: Wed, 04 Feb 2026 09:37:17 GMT foundInMaster: true id: 45f46a8f7ac698cfe3926303bf3dff9d v3_1_link: https://github.com/swagger-api/swagger-parser/pull/1730 v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/apidevtools/swagger-parser v3: true id: 18a1495e6591ba141a4db34b3ae36f32 repositoryMetadata: base64Readme: >- # Swagger 2.0 and OpenAPI 3.0 parser/validator

[![Build Status](https://github.com/APIDevTools/swagger-parser/workflows/CI-CD/badge.svg?branch=master)](https://github.com/APIDevTools/swagger-parser/actions)
[![Coverage Status](https://coveralls.io/repos/github/APIDevTools/swagger-parser/badge.svg?branch=master)](https://coveralls.io/github/APIDevTools/swagger-parser)
[![Tested on APIs.guru](https://api.apis.guru/badges/tested_on.svg)](https://apis.guru/browse-apis/)

[![npm](https://img.shields.io/npm/v/@apidevtools/swagger-parser.svg)](https://www.npmjs.com/package/@apidevtools/swagger-parser)
[![Dependencies](https://david-dm.org/APIDevTools/swagger-parser.svg)](https://david-dm.org/APIDevTools/swagger-parser)
[![License](https://img.shields.io/npm/l/@apidevtools/swagger-parser.svg)](LICENSE)
[![Buy us a tree](https://img.shields.io/badge/Treeware-%F0%9F%8C%B3-lightgreen)](https://shop.protect.earth/)

[![OS and Browser Compatibility](https://apidevtools.com/img/badges/ci-badges-with-ie.svg)](https://github.com/APIDevTools/swagger-parser/actions)

## Features

- Parses Swagger specs in **JSON** or **YAML** format
- Validates against the [Swagger 2.0 schema](https://github.com/OAI/OpenAPI-Specification/blob/master/schemas/v2.0/schema.json) or [OpenAPI 3.0 Schema](https://github.com/OAI/OpenAPI-Specification/blob/master/schemas/v3.0/schema.json)
- [Resolves](https://apidevtools.com/swagger-parser/docs/swagger-parser.html#resolveapi-options-callback) all `$ref` pointers, including external files and URLs
- Can [bundle](https://apidevtools.com/swagger-parser/docs/swagger-parser.html#bundleapi-options-callback) all your Swagger files into a single file that only has _internal_ `$ref` pointers
- Can [dereference](https://apidevtools.com/swagger-parser/docs/swagger-parser.html#dereferenceapi-options-callback) all `$ref` pointers, giving you a normal JavaScript object that's easy to work with
- **[Tested](https://github.com/APIDevTools/swagger-parser/actions)** in Node.js and all modern web browsers on Mac, Windows, and Linux
- Tested on **[over 1,500 real-world APIs](https://apis.guru/browse-apis/)** from Google, Microsoft, Facebook, Spotify, etc.
- Supports [circular references](https://apidevtools.com/swagger-parser/docs/#circular-refs), nested references, back-references, and cross-references
- Maintains object reference equality &mdash; `$ref` pointers to the same value always resolve to the same object instance

## Related Projects

- [Swagger CLI](https://github.com/APIDevTools/swagger-cli)
- [Swagger Express Middleware](https://github.com/APIDevTools/swagger-express-middleware)

## Example

```javascript
SwaggerParser.validate(myAPI, (err, api) => {
  if (err) {
    console.error(err);
  } else {
    console.log("API name: %s, Version: %s", api.info.title, api.info.version);
  }
});
```

Or use `async`/`await` or [Promise](http://javascriptplayground.com/blog/2015/02/promises/) syntax instead. The following example is the same as above:

```javascript
try {
  let api = await SwaggerParser.validate(myAPI);
  console.log("API name: %s, Version: %s", api.info.title, api.info.version);
} catch (err) {
  console.error(err);
}
```

For more detailed examples, please see the [API Documentation](https://apidevtools.com/swagger-parser/docs/)

## Installation

Install using [npm](https://docs.npmjs.com/about-npm/):

```bash
npm install @apidevtools/swagger-parser
```

## Usage

When using Swagger Parser in Node.js apps, you'll probably want to use **CommonJS** syntax:

```javascript
const SwaggerParser = require("@apidevtools/swagger-parser");
```

When using a transpiler such as [Babel](https://babeljs.io/) or [TypeScript](https://www.typescriptlang.org/), or a bundler such as [Webpack](https://webpack.js.org/) or [Rollup](https://rollupjs.org/), you can use **ECMAScript modules** syntax instead:

```javascript
import * as SwaggerParser from "@apidevtools/swagger-parser";
```

## Browser support

Swagger Parser supports recent versions of every major web browser. Older browsers may require [Babel](https://babeljs.io/) and/or [polyfills](https://babeljs.io/docs/en/next/babel-polyfill).

To use Swagger Parser in a browser, you'll need to use a bundling tool such as [Webpack](https://webpack.js.org/), [Rollup](https://rollupjs.org/), [Parcel](https://parceljs.org/), or [Browserify](http://browserify.org/). Some bundlers may require a bit of configuration, such as setting `browser: true` in [rollup-plugin-resolve](https://github.com/rollup/rollup-plugin-node-resolve).

## API Documentation

Full API documentation is available [right here](https://apidevtools.com/swagger-parser/docs/)

## Security

The library, by default, attempts to resolve any files referenced using `$ref`, without considering file extensions or the location of the files. This can result in Local File Inclusion (LFI), thus, potentially sensitive information disclosure. Developers must be cautious when working with documents from untrusted sources. See [here](SECURITY.md) for more details and information on how to mitigate LFI.

## Contributing

I welcome any contributions, enhancements, and bug-fixes. [Open an issue](https://github.com/APIDevTools/swagger-parser/issues) on GitHub and [submit a pull request](https://github.com/APIDevTools/swagger-parser/pulls).

To test the project locally on your computer:

1. **Clone this repo**<br>
   `git clone https://github.com/APIDevTools/swagger-parser.git`

2. **Install dependencies**<br>
   `npm install`

3. **Run the tests**<br>
   `npm test`

4. **Check the code coverage**<br>
   `npm run coverage`

## License

Swagger Parser is 100% free and open-source, under the [MIT license](LICENSE). Use it however you want.

This package is [Treeware](http://treeware.earth). If you use it in production, then we ask that you [**buy the world a tree**](https://shop.protect.earth) to thank us for our work.

## Big Thanks To

Thanks to these awesome companies for their support of Open Source developers ❤

[![GitHub](https://apidevtools.com/img/badges/github.svg)](https://github.com/open-source)
[![NPM](https://apidevtools.com/img/badges/npm.svg)](https://www.npmjs.com/)
[![Coveralls](https://apidevtools.com/img/badges/coveralls.svg)](https://coveralls.io)
 readmeEtag: '"d2e8b27c72ae4bdcdfdc2e480ad8aff868226cb0"' readmeLastModified: Tue, 03 Jun 2025 19:01:44 GMT repositoryId: 25453221 description: Swagger 2.0 and OpenAPI 3.0 parser/validator created: '2014-10-20T06:14:51Z' updated: '2026-02-03T15:44:03Z' language: JavaScript archived: false stars: 1181 watchers: 19 forks: 165 owner: APIDevTools logo: https://avatars.githubusercontent.com/u/43750074?v=4 license: MIT repoEtag: '"5a2662277896a5ff5392ed5525b3531f199df1e6cee0ba75759c65368468dd99"' repoLastModified: Tue, 03 Feb 2026 15:44:03 GMT category: Parsers foundInMaster: true name: APIDevTools/swagger-parser language: Node.js source_description: >- OpenAPI 2.0 and 3.0 parser and validator. Can also bundle multiple documents into one via `$ref`. v2: true oldLocations: - https://github.com/bigstickcarpet/swagger-parser - source: - https://openapi.tools/ - openapi3 tags name: openapi-snippet category: Parsers repository: https://github.com/erikwittern/openapi-snippet language: Node.js source_description: >- Generates code snippets in various languages & tools (cURL, Node, Python, Ruby, Java, Go, C#...), from OpenAPI documents. v2: true v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNuaXBwZXQKKipHZW5lcmF0ZXMgY29kZSBzbmlwcGV0cyBmcm9tIE9wZW4gQVBJIChwcmV2aW91c2x5IFN3YWdnZXIpIGRvY3VtZW50cy4qKgoKVGhpcyBwYWNrYWdlIHRha2VzIGFzIGlucHV0IGFuIE9wZW5BUEkgdjIuMCBvciB2My4wLnggZG9jdW1lbnQuIEl0IHRyYW5zbGF0ZXMgdGhlIGRvY3VtZW50IGludG8gYW4gW0hUVFAgQXJjaGl2ZSAxLjIgcmVxdWVzdCBvYmplY3RdKGh0dHA6Ly93d3cuc29mdHdhcmVpc2hhcmQuY29tL2Jsb2cvaGFyLTEyLXNwZWMvI3JlcXVlc3QpLiBJdCB1c2VzIHRoZSBbSFRUUCBTbmlwcGV0XShodHRwczovL2dpdGh1Yi5jb20vTWFzaGFwZS9odHRwc25pcHBldCkgbGlicmFyeSB0byBnZW5lcmF0ZSBjb2RlIHNuaXBwZXRzIGZvciBldmVyeSBBUEkgZW5kcG9pbnQgKFVSTCBwYXRoICsgSFRUUCBtZXRob2QpIGRlZmluZWQgaW4gdGhlIHNwZWNpZmljYXRpb24gaW4gdmFyaW91cyBsYW5ndWFnZXMgJiB0b29scyAoYGNVUkxgLCBgTm9kZWAsIGBQeXRob25gLCBgUnVieWAsIGBKYXZhYCwgYEdvYCwgYEMjYC4uLiksIG9yIGZvciBzZWxlY3RlZCBlbmRwb2ludHMuCgojIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKbnBtIGkgb3BlbmFwaS1zbmlwcGV0CmBgYAoKIyMgQnVpbGQgT3BlbkFQSSBTbmlwcGV0IChmb3IgdXNlIGluIGJyb3dzZXIpCkNsb25lIHRoaXMgcmVwb3NpdG9yeS4gSW5zdGFsbCByZXF1aXJlZCBkZXBlbmRlbmNpZXM6CgpgYGBiYXNoCm5wbSBpCmBgYAoKQnVpbGQgYSBtaW5pZmllZCB2ZXJzaW9uIG9mIE9wZW5BUEkgU25pcHBldCAoYG9wZW5hcGlzbmlwcGV0Lm1pbi5qc2ApOgoKYGBgYmFzaApucG0gcnVuIGJ1aWxkCmBgYAoKIyMgVXNhZ2UKCiMjIyBBcyBhIG1vZHVsZQoKYGBgamF2YXNjcmlwdApjb25zdCBPcGVuQVBJU25pcHBldCA9IHJlcXVpcmUoJ29wZW5hcGktc25pcHBldCcpCgovLyBkZWZpbmUgaW5wdXQ6CmNvbnN0IG9wZW5BcGkgPSAuLi4gLy8gT3BlbiBBUEkgZG9jdW1lbnQKY29uc3QgdGFyZ2V0cyA9IFsnbm9kZV91bmlyZXN0JywgJ2MnXSAvLyBhcnJheSBvZiB0YXJnZXRzIGZvciBjb2RlIHNuaXBwZXRzLiBTZWUgbGlzdCBiZWxvdy4uLgoKdHJ5IHsKICAvLyBlaXRoZXIsIGdldCBzbmlwcGV0cyBmb3IgQUxMIGVuZHBvaW50czoKICBjb25zdCByZXN1bHRzID0gT3BlbkFQSVNuaXBwZXQuZ2V0U25pcHBldHMob3BlbkFwaSwgdGFyZ2V0cykgLy8gcmVzdWx0cyBpcyBub3cgYXJyYXkgb2Ygc25pcHBldHMsIHNlZSAiT3V0cHV0IiBiZWxvdy4KCiAgLy8gLi4ub3IsIGdldCBzbmlwcGV0cyBmb3IgYSBzaW5nbGUgZW5kcG9pbnQ6CiAgY29uc3QgcmVzdWx0czIgPSBPcGVuQVBJU25pcHBldC5nZXRFbmRwb2ludFNuaXBwZXRzKG9wZW5BcGksICcvdXNlcnMve3VzZXItaWR9L3JlbGF0aW9uc2hpcCcsICdnZXQnLCB0YXJnZXRzKQp9IGNhdGNoIChlcnIpIHsKICAvLyBkbyBzb21ldGhpbmcgd2l0aCBwb3RlbnRpYWwgZXJyb3JzLi4uCn0KYGBgCgojIyMgV2l0aGluIHRoZSBicm93c2VyCgpJbmNsdWRlIHRoZSBgb3BlbmFwaXNuaXBwZXQubWluLmpzYCBmaWxlIGNyZWF0ZWQgYWZ0ZXIgYnVpbGRpbmcgdGhlIHRoZSBsaWJyYXJ5IChzZWUgYWJvdmUpIGluIHlvdXIgSFRNTCBwYWdlOgoKYGBgaHRtbAo8c2NyaXB0IHR5cGU9InRleHQvamF2YXNjcmlwdCIgc3JjPSJwYXRoL3RvL29wZW5hcGlzbmlwcGV0Lm1pbi5qcyI+PC9zY3JpcHQ+CmBgYAoKVXNlIE9wZW5BUEkgU25pcHBldCwgd2hpY2ggbm93IGRlZmluZXMgdGhlIGdsb2JhbCB2YXJpYWJsZSBgT3BlbkFQSVNuaXBwZXRgLgoKCiMjIE91dHB1dApUaGUgb3V0cHV0IGZvciBldmVyeSBlbmRwb2ludCBpcyBhbiBvYmplY3QsIGNvbnRhaW5pbmcgdGhlIGBtZXRob2RgLCBgdXJsYCwgYSBodW1hbi1yZWFkYWJsZSBgZGVzY3JpcHRpb25gLCBhbmQgdGhlIGNvcnJlc3BvbmRpbmcgYHJlc291cmNlYCAtIGFsbCBvZiB0aGVzZSB2YWx1ZXMgc3RlbSBmcm9tIHRoZSBPcGVuQVBJIGRvY3VtZW50LiBJbiBhZGRpdGlvbiwgd2l0aGluIHRoZSBgc25pcHBldHNgIGxpc3QsIGFuIG9iamVjdCBjb250YWluaW5nIGEgY29kZSBzbmlwcGV0IGZvciBldmVyeSBjaG9zZW4gdGFyZ2V0IGlzIHByb3ZpZGVkLiBBcyBvZiB2ZXJzaW9uIGAwLjQuMGAsIHRoZSBzbmlwcGV0cyBpbmNsdWRlIGV4ZW1wbGFyeSBwYXlsb2FkIGRhdGEuCgpJZiBgZ2V0U25pcHBldHNgIGlzIHVzZWQsIGFuIGFycmF5IG9mIHRoZSBhYm92ZSBkZXNjcmliZWQgb2JqZWN0cyBpcyByZXR1cm5lZC4KCkZvciBleGFtcGxlOgoKYGBganMKWwogIC8vIC4uLgogIHsKICAgICJtZXRob2QiOiAiR0VUIiwKICAgICJ1cmwiOiAiaHR0cHM6Ly9hcGkuaW5zdGFncmFtLmNvbS92MS91c2Vycy97dXNlci1pZH0vcmVsYXRpb25zaGlwIiwKICAgICJkZXNjcmlwdGlvbiI6ICJHZXQgaW5mb3JtYXRpb24gYWJvdXQgYSByZWxhdGlvbnNoaXAgdG8gYW5vdGhlciB1c2VyLiIsCiAgICAicmVzb3VyY2UiOiAicmVsYXRpb25zaGlwIiwKICAgICJzbmlwcGV0cyI6IFsKICAgICAgewogICAgICAgICJpZCI6ICJub2RlIiwKICAgICAgICAibWltZVR5cGUiOiAiYXBwbGljYXRpb24vanNvbiIsICAvLyBPbmx5IHNldCBmb3IgbWV0aG9kcyB3aXRoIGEgcmVxdWVzdCBib2R5CiAgICAgICAgInRpdGxlIjogIk5vZGUgKyBOYXRpdmUiLAogICAgICAgICJjb250ZW50IjogInZhciBodHRwID0gcmVxdWlyZShcImh0dHBzXCIpO1xuXG52YXIgb3B0aW9ucyA9IHsuLi4iCiAgICAgIH0KICAgIF0KICB9CiAgLy8gLi4uCl0KYGBgCgojIyBUYXJnZXRzCkN1cnJlbnRseSwgT3BlbkFQSSBTbmlwcGV0IHN1cHBvcnRzIHRoZSBmb2xsb3dpbmcgW3RhcmdldHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9Lb25nL2h0dHBzbmlwcGV0L3RyZWUvbWFzdGVyL3NyYy90YXJnZXRzKSAoZGVwZW5kaW5nIG9uIHRoZSBIVFRQIFNuaXBwZXQgbGlicmFyeSk6CgoqIGBjX2xpYmN1cmxgIChkZWZhdWx0KQoqIGBjc2hhcnBfcmVzdHNoYXJwYCAoZGVmYXVsdCkKKiBgY3NoYXJwX2h0dHBjbGllbnRgCiogYGdvX25hdGl2ZWAgKGRlZmF1bHQpCiogYGphdmFfb2todHRwYAoqIGBqYXZhX3VuaXJlc3RgIChkZWZhdWx0KQoqIGBqYXZhc2NyaXB0X2pxdWVyeWAKKiBgamF2YXNjcmlwdF94aHJgIChkZWZhdWx0KQoqIGBub2RlX25hdGl2ZWAgKGRlZmF1bHQpCiogYG5vZGVfcmVxdWVzdGAKKiBgbm9kZV91bmlyZXN0YAoqIGBvYmpjX25zdXJsc2Vzc2lvbmAgKGRlZmF1bHQpCiogYG9jYW1sX2NvaHR0cGAgKGRlZmF1bHQpCiogYHBocF9jdXJsYCAoZGVmYXVsdCkKKiBgcGhwX2h0dHAxYAoqIGBwaHBfaHR0cDJgCiogYHB5dGhvbl9weXRob24zYCAoZGVmYXVsdCkKKiBgcHl0aG9uX3JlcXVlc3RzYAoqIGBydWJ5X25hdGl2ZWAgKGRlZmF1bHQpCiogYHNoZWxsX2N1cmxgIChkZWZhdWx0KQoqIGBzaGVsbF9odHRwaWVgCiogYHNoZWxsX3dnZXRgCiogYHN3aWZ0X25zdXJsc2Vzc2lvbmAgKGRlZmF1bHQpCgpJZiBvbmx5IHRoZSBsYW5ndWFnZSBpcyBwcm92aWRlZCAoZS5nLiwgYGNgKSwgdGhlIGRlZmF1bHQgbGlicmFyeSB3aWxsIGJlIHNlbGVjdGVkLgoKCkxpY2Vuc2U6IE1JVAo= readmeEtag: '"7e8dcc103099999269c433a3422041583fb19760"' readmeLastModified: Wed, 13 Jul 2022 18:36:58 GMT repositoryId: 66855599 description: Generates code snippets for given Swagger / Open API documents created: '2016-08-29T15:24:00Z' updated: '2026-01-01T17:57:57Z' language: JavaScript archived: false stars: 128 watchers: 4 forks: 69 owner: ErikWittern logo: https://avatars.githubusercontent.com/u/1745998?v=4 license: MIT repoEtag: '"69eec6c1e3ff357d06f373f037d928bd27534d61387fe7a9cc08e0cee46b1e95"' repoLastModified: Thu, 01 Jan 2026 17:57:57 GMT foundInMaster: true id: fbf76c5bdaef3297eb35739ca2ca0954 - source: https://openapi.tools/ name: openapi-snippet-cli category: Parsers repository: https://github.com/richardkabiling/openapi-snippet-cli language: Node.js source_description: >- Adds code snippets in redoc style (x-codeSamples) to OpenAPI documents. This is a CLI wrapper for the "openapi-snippet". v2: true v3: true repositoryMetadata: base64Readme: >- b3BlbmFwaS1zbmlwcGV0LWNsaQo9PT09PT09PT09PT09PT09PT09CgpBZGRzIG9wZW5hcGkgc25pcHBldHMgdXNpbmcgYG9wZW5hcGktc25pcHBldGAgbW9kdWxlIGluIHJlZG9jIHN0eWxlICh4LWNvZGVTYW1wbGVzKS4gVGhpcyBpcyBhIENMSSB3cmFwcGVyIG9uIFtvcGVuYXBpLXNuaXBwZXRdKGh0dHBzOi8vZ2l0aHViLmNvbS9FcmlrV2l0dGVybi9vcGVuYXBpLXNuaXBwZXQpCgpbIVtvY2xpZl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jbGktb2NsaWYtYnJpZ2h0Z3JlZW4uc3ZnKV0oaHR0cHM6Ly9vY2xpZi5pbykKWyFbVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vcGVuYXBpLXNuaXBwZXQtY2xpLnN2ZyldKGh0dHBzOi8vbnBtanMub3JnL3BhY2thZ2Uvb3BlbmFwaS1zbmlwcGV0LWNsaSkKWyFbRG93bmxvYWRzL3dlZWtdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2R3L29wZW5hcGktc25pcHBldC1jbGkuc3ZnKV0oaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZS9vcGVuYXBpLXNuaXBwZXQtY2xpKQpbIVtMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9sL29wZW5hcGktc25pcHBldC1jbGkuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3JpY2hhcmRrYWJpbGluZy9vcGVuYXBpLXNuaXBwZXQtY2xpL2Jsb2IvbWFzdGVyL3BhY2thZ2UuanNvbikKCiogW0dldHRpbmcgU3RhcnRlZF0oI2dldHRpbmctc3RhcnRlZCkKKiBbVXNhZ2VdKCN1c2FnZSkKKiBbQXJndW1lbnRzXSgjYXJndW1lbnRzKQoqIFtPcHRpb25zXSgjb3B0aW9ucykKCiMgR2V0dGluZyBTdGFydGVkClRvIGluc3RhbGw6CgpgYGBzaC1zZXNzaW9uCiQgbnBtIGluc3RhbGwgLWcgb3BlbmFwaS1zbmlwcGV0LWNsaQpgYGAKCiMgVXNhZ2UKIyMgQWRkaW5nIFNuaXBwZXRzIHRvIGEgU2NoZW1hCmBgYHNoLXNlc3Npb24KJCBvcGVuYXBpLXNuaXBwZXQgc2NoZW1hLnlhbWwgLW8gZGlzdC9zY2hlbWEueWFtbApgYGAKClRoZSBleGFtcGxlIGFib3ZlIHNob3VsZCBhZGQgc25pcHBldHMgdG8gYHNjaGVtYS55YW1sYCBhbmQgb3V0cHV0IHRoZSBtb2RpZmllZCBzY2hlbWEgdG8gYSBuZXcgZmlsZSBgZGlzdC9zY2hlbWEueWFtbGAuCgojIyBPdXRwdXR0aW5nIEpTT04KYGBgc2gtc2Vzc2lvbgokIG9wZW5hcGktc25pcHBldCBzY2hlbWEueWFtbCAtZSBqc29uIC1vIGRpc3Qvc2NoZW1hLmpzb24KYGBgCgojIyBDaG9vc2luZyBUYXJnZXRzCmBgYHNoLXNlc3Npb24KJCBvcGVuYXBpLXNuaXBwZXQgc2NoZW1hLnlhbWwgLXQgamF2YSAtdCBjIC1vIGRpc3Qvc2NoZW1hLmpzb24KYGBgCgpUaGUgZXhhbXBsZSBhYm92ZSBzaG91bGQgYWRkIHNuaXBwZXRzIGZvciBgamF2YWAgYW5kIGBjYCB1c2luZyB0aGVpciBkZWZhdWx0IGZyYW1ld29ya3MKCmBgYHNoLXNlc3Npb24KJCBvcGVuYXBpLXNuaXBwZXQgc2NoZW1hLnlhbWwgLXQgamF2YV9va2h0dHAgLW8gZGlzdC9zY2hlbWEuanNvbgpgYGAKClRoaXMgc2hvdWxkIGFkZCBzbmlwcGV0cyBmb3IgJ2phdmFgIHVzaW5nIE9rSHR0cC4KCiMgQXJndW1lbnRzCmBgYApVU0FHRQogICQgb3BlbmFwaS1zbmlwcGV0IFtGSUxFXQoKQVJHVU1FTlRTCiAgRklMRSAgaW5wdXQgb3BlbmFwaSBkb2N1bWVudC4gSXQgd2lsbCBhdHRlbXB0IHRvIHJlc29sdmUgcmVmZXJlbmNlcyAoaW5jbHVkaW5nIGJvdGggaW50ZXJuYWwgYWRuIGV4dGVybmFsIG9uZXMpCmBgYAoKIyBPcHRpb25zCgpgYGAKT1BUSU9OUwogIC1lLCAtLWV4dD15YW1sfGpzb24gICAgW2RlZmF1bHQ6IHlhbWxdIG91dHB1dCBmb3JtYXQKICAtaCwgLS1oZWxwICAgICAgICAgICAgIHNob3cgQ0xJIGhlbHAKICAtbywgLS1vdXRwdXQ9b3V0cHV0ICAgIFtkZWZhdWx0OiBvdXRwdXQueWFtbF0gb3V0cHV0IGZpbGUgbmFtZQoKICAtdCwgLS10YXJnZXRzPXRhcmdldHMgIHRhcmdldCBzbmlwcGV0IGxhbmd1YWdlcyArIGZyYW1ld29ya3MuIENhbiBiZSBwcm92aWRlZCBtdWx0aXBsZSB0aW1lcy4gSWYgaW5wdXR0aW5nIGxhbmd1YWdlIG9ubHksIGRlZmF1bHRzIHRvIG9uZSBvZiB0aGUgZnJhbWV3b3Jrcy4gU3VwcG9ydHMKICAgICAgICAgICAgICAgICAgICAgICAgIGxhbmd1YWdlcyBzdXBwb3J0ZWQgaW4gaHR0cHM6Ly9naXRodWIuY29tL0VyaWtXaXR0ZXJuL29wZW5hcGktc25pcHBldC4gRGVmYXVsdHMgdG8gYWRkaW5nIHNuaXBwZXRzIGZvciBBTEwgc3VwcG9ydGVkIGxhbmd1YWdlcy4KCiAgLXYsIC0tdmVyc2lvbiAgICAgICAgICBzaG93IENMSSB2ZXJzaW9uCmBgYAo= readmeEtag: '"66f502da67823c809b038892e9a42d544b64900a"' readmeLastModified: Fri, 05 Jun 2020 08:06:30 GMT repositoryId: 269371079 description: null created: '2020-06-04T13:45:15Z' updated: '2024-09-17T19:57:16Z' language: TypeScript archived: false stars: 9 watchers: 1 forks: 3 owner: richardkabiling logo: https://avatars.githubusercontent.com/u/2498349?v=4 repoEtag: '"04d1f75d45a9e629c7f1a58d1ab9eccd754d3f60978448e535bf057da749975c"' repoLastModified: Tue, 17 Sep 2024 19:57:16 GMT foundInMaster: true id: e96651ada1af40ecd2c2595e9341e5ea - source: - https://openapi.tools/ - openapi3 tags name: KaiZen OpenAPI Parser repository: https://github.com/reprezen/kaizen-openapi-parser owner: RepreZen category: Parsers language: Java source_description: High-performance Parser, Validator, and Java Object Model for OpenAPI 3.x v3: true repositoryMetadata: base64Readme: >- <img src="https://cdn2.hubspot.net/hubfs/597611/Assets_Swagger/KaiZen%20OpenAPI%20Parser%20Logo%20(medium).png" height="50%" width="50%" alt="KaiZen OpenAPI Parser Logo" />

# KaiZen OpenAPI Parser

## Overview ##

The KaiZen OpenApi Parser from RepreZen is a Java-based validating
parser for OpenAPI 3.0 offering full compliance with the
[OpenAPI 3.0 Specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md),
and a highly uniform read/write programming API.
[OpenAPI](http://openapis.org), formerly known as the Swagger
specification, is the industry-standard format for machine-readable
REST API descriptions.

Feature highlights of KaiZen OpenAPI Parser include:
* **High Performance** - Informal testing shows a 3x-4x performance
  improvement over the current Java Swagger 2.0 parser. This is
  largely attributable to a design based on adapting Jackson
  `JsonNode` objects, rather than deserializing to internal POJOs.
  
* **Read/Write API** - All aspects of a model may be interrogated and
  modified. We also plan to provide fluent builders for all
  model object types. We provide bidirectional navigation throughout the
  model, and every object that is a property value of its containing
  object (whether as a named field or a map entry) knows its own name.
  
* **Tolerant Reader** - The parser yields a fully accessible result
  from any valid JSON or YAML input - whether or not the input is a
  valid OpenAPI specification.
  
* **Separate validation** - All validation beyond basic
  JSON/YAML parsing is performed after the initial parse, and it can
  be disabled for speed. Validation goes beyond checking what can be
  expressed in JSON Schema, to include all requirements described in
  the OpenAPI specification.

* **Serialization** - Serialization to JSON or YAML is supported, and
  by default, round-tripping will not cause any reordering of model
  content.
  
* **Easy Evolution** - A YAML-based DSL is used to capture most of the
  details of the OpenAPI Specification. We use code generation to
  create interfaces and implementation classes. Member-level
  `@Generated` annotations make it possible to augment the generated
  sources with manual code that is preserved during re-generation.
  
* **Flexible Reference Handling** - All references are detected and
  during parsing, including references not technically permitted by
  the OpenAPI specification. References are normally traversed 
  automatically by the API, but full details of references and 
  their resolution status are also available.
  
* **Unpolluted API** - Many features of the parser API are not directly
  accessible from modeled objects, but are accessed via adapter objects.
  This ensures that these features will not collide with generated
  methods of the model API, even as new features are added to the
  OpenAPI specification in the future.
    
## Documentation

The [Getting Started Guide](GettingStarted.md) shows how to build the 
software locally, and contains a simple sample program that shows how 
to use the parser.

The [API Overview](API-Overview.md) describes the APIs presented in
the project, including the parser, the serializer, the read/write
model, and the treatment of references.

## Who's using KaiZen Parser?

Here's a starting list of projects that are currently using KaiZen OpenAPI Parser. If you don't see your project here, please open an issue or pull request to add it:

| Project Link | Description |
| --- | --- | 
| [Eclipse Vert.x](http://vertx.io/) | Eclipse Vert.x is a tool-kit for building reactive applications on the JVM. | 
| [Light-rest-4j](https://github.com/networknt/light-rest-4j) | Light-4j RESTful framework for building fast, lightweight microservices on the JVM. | 
| [RepreZen API Studio](http://reprezen.com/OpenAPI) | RepreZen API Studio is an integrated workbench for API design, documentation and development, built on the Eclipse platform. | 


## Current State

* The parser is currently based on the pre-release [revision 3.0.0-rc0](https://github.com/OAI/OpenAPI-Specification/blob/d232e6d3e1ea4038a533329a82876ae868e9cf13/versions/3.0.md). We are nearly ready with an upgrade to the [3.0.2 draft revision](https://github.com/OAI/OpenAPI-Specification/blob/v3.0.2-dev/versions/3.0.2.md).

* The [JsonOverlay Project](https://github.com/RepreZen/JsonOverlay) is a framework for creating parsers and APIs for YAML/JSON based DSLs. It is the backbone of the KaiZen OpenApi Parser. Features that that it provides include:
 
  * Read-Write API for all model objects, based on a YAML document that describes the OpenAPI model structure
  * Factories for model objects (used internally, but not currently well exposed to user code; that will change shortly)
  * Full handling of all references.
  * Serialization, reference inspection, navigation, and numerous other features via its `Overlay` adapter classes.
  * Position-aware parser providing URL, line and file number for all parsed objects, available through `Overlay` adapters and used by the KaiZen parser in its validation messages.

* Validations are currently all contained within this project, however many routine validations (e.g. validating proper JSON types throughout a model document, checking that required properties are present, etc.) will at some point be moved into the JsonOverlay project.

* Most validations are present, but there are a number that are currently missing, and some existing validations still reflect the OpenAPI specification in its pre-release revision 3.0.0-rc0. Work is underway on Issue #26](https://github.com/RepreZen/KaiZen-OpenApi-Parser/issues/26), which should result in a complete and robust implementation of all model validations, updated to the 3.0.2 revision (currently in draft status).

* Serialization is available via the `Overlay` adapter's `toJson` method. By default references appear in the serialized output, but an option causes references to be followed and inlined in the output. Recursive references cause serialization to blow up if this option is used.
  - A separate component, called "OpenAPI Normalizer," will soon be made available that will provide much greater control over the treatment of references. This is currently a private feature embedded in [RepreZen API Studio](https://www.reprezen.com/). Its primary function is to turn an OpenAPI model spread across multiple files into an equivalent single-file model. Options control which references are inlined, and which are _localized_ as named component objects in the constructed single-file model.
  
* A handful of high-level tests have been implemented:
  - *BigParseTest* parses a large model without validation and checks
    that every value node in the input is accessible in the expected
    fashion in the resulting model object.
  - *ExamplesTest* - Parses and validates all example models currently
    in the the `OAI/OpenAPI-Specification` GitHub repo.

    Many more tests are required!

* Few JavaDocs exist at present, unfortunately. There's an open issue
  to correct that.

### Packages

_Some of these packages are likely to be refactored out into separate
component Maven projects._

All packages are prefixed by `com.reprezen.kaizen`

* `oasparser`: Top-level package, directly includes
  `OpenApiParser` class and some things related to code generation.
  
* `oasparser.model3`: Generated model interfaces (generated by JsonOverlay)

* `oasparser.ovl3`: Generated model implementation classes (generated by JsonOverlay)

* `oasparser.val`: Base classes for validators

* `oasparser.val3`: Validators for all OpenAPI objects.

* `oasparser.test`: The handful of tests that have been
  implemented so far. More needed


## License
KaiZen OpenAPI Parser is provided under the Eclipse Public License (https://www.eclipse.org/legal/epl-v10.html)

## Contributing

We welcome serious contributors. However, we are at an early and
fast-changing phase in this project, so we recommend coordinating
with us before diving into a juicy bit of coding. This might spare 
you the frustration of finding that some sweeping low-level change
has rendered your efforts unusable.

If you would like to work with us, please drop us a line at
API.Community@RepreZen.com, or open a new issue if you have
a suggestion or want to report a bug or omission.

## Resources
* Blog Post: [Introducing KaiZen OpenAPI 3.0 Parser: fast, flexible Java parsing & validation](http://www.reprezen.com/blog/kaizen-openapi-3_0-parser-swagger-java-open-source)
* [Getting Started Guide](GettingStarted.md)
* [API Overview](API-Overview.md)
 readmeEtag: '"65b773015321acdbf3de73891ebfec4619170091"' readmeLastModified: Wed, 17 Jul 2019 13:54:46 GMT repositoryId: 86403618 description: High-performance Parser, Validator, and Java Object Model for OpenAPI 3.x created: '2017-03-28T02:00:01Z' updated: '2025-12-29T02:11:01Z' language: Java archived: false stars: 132 watchers: 8 forks: 31 owner: RepreZen logo: https://avatars.githubusercontent.com/u/15205934?v=4 repoEtag: '"dcd33228a9773935b801365f177c9f5623361cc908aaa7520f1ee73208aa322f"' repoLastModified: Mon, 29 Dec 2025 02:11:01 GMT foundInMaster: true homepage: https://github.com/RepreZen/KaiZen-OpenAPI-Parser id: f542543d6021da9cbce2c66599690853 - source: https://openapi.tools/ name: OpenAPI-TS homepage: https://github.com/metadevpro/openapi3-ts language: TypeScript source_description: TS Model & utils for OpenAPI 3.0.x contracts category: Parsers repository: https://github.com/metadevpro/openapi3-ts v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQXBpMy1UUwoKVHlwZVNjcmlwdCBsaWJyYXJ5IHRvIGhlbHAgYnVpbGRpbmcgT3BlbkFQSSAzLnggY29tcGxpYW50IEFQSSBjb250cmFjdHMuCgpbIVtDb3ZlcmFnZSBTdGF0dXNdKGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9tZXRhZGV2cHJvL29wZW5hcGkzLXRzL2JhZGdlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL21ldGFkZXZwcm8vb3BlbmFwaTMtdHM/YnJhbmNoPW1hc3RlcikKWyFbS25vd24gVnVsbmVyYWJpbGl0aWVzXShodHRwczovL3NueWsuaW8vdGVzdC9naXRodWIvbWV0YWRldnByby9vcGVuYXBpMy10cy9iYWRnZS5zdmc/dGFyZ2V0RmlsZT1wYWNrYWdlLmpzb24pXShodHRwczovL3NueWsuaW8vdGVzdC9naXRodWIvbWV0YWRldnByby9vcGVuYXBpMy10cz90YXJnZXRGaWxlPXBhY2thZ2UuanNvbikKWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9vcGVuYXBpMy10cy5zdmcpXShodHRwOi8vYmFkZ2UuZnVyeS5pby9qcy9vcGVuYXBpMy10cykKClshW05QTV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vb3BlbmFwaTMtdHMucG5nP2Rvd25sb2Fkcz10cnVlJmRvd25sb2FkUmFuaz10cnVlJnN0YXJzPXRydWUpXShodHRwczovL25vZGVpLmNvL25wbS9vcGVuYXBpMy10cy8pCgojIyBWZXJzaW9uIDQKCipCcmVha2luZyBjaGFuZ2Ugbm90aWNlOioKClZlcnNpb24gNC4wIEFkZHMgZXhwbGljaXQgc3VwcG9ydCBmb3IgT0FTIDMuMCBhbmQgT0FTIDMuMSBhcyBzZXBhcmF0ZSBpbXBsZW1lbnRhdGlvbnMuCgojIyMgVG8gdXNlIHZlcnNpb24gMy4xIGltcG9ydAoKYGBganMKaW1wb3J0IHsgb2FzMzEgfSBmcm9tICdvcGVuYXBpMy10cyc7CmBgYAoKT3IgZGlyZWN0bHkgaW1wb3J0IGZyb20gc3VicGF0aDoKCmBgYGpzCmltcG9ydCB7IE9wZW5BUElPYmplY3QsIE9wZW5BcGlCdWlsZGVyIH0gZnJvbSAnb3BlbmFwaTMtdHMvb2FzMzEnOwpgYGAKCiMjIyBUbyB1c2UgdmVyc2lvbiAzLjAgaW1wb3J0CgpgYGBqcwppbXBvcnQgeyBvYXMzMCB9IGZyb20gJ29wZW5hcGkzLXRzJzsKYGBgCgpPciBkaXJlY3RseSBpbXBvcnQgZnJvbSBzdWJwYXRoOgoKYGBganMKaW1wb3J0IHsgT3BlbkFQSU9iamVjdCwgT3BlbkFwaUJ1aWxkZXIgfSBmcm9tICdvcGVuYXBpMy10cy9vYXMzMCc7CmBgYAoKIyMgSW5jbHVkZXMKCiogYC9zcmMvbW9kZWxgIFRTIHR5cGVkIGludGVyZmFjZXMgZm9yIGhlbHBpbmcgYnVpbGRpbmcgYSBjb250cmFjdC4KKiBgL3NyYy9kc2xgIEZsdWVudCBEU0wgZm9yIGJ1aWxkaW5nIGEgY29udHJhY3QuCgojIyBJbnN0YWxsCgpJbnN0YWxsIHBhY2thZ2UgdmlhICoqbnBtKio6CgpgYGBiYXNoCm5wbSBpIC0tc2F2ZSBvcGVuYXBpMy10cwpgYGAKCiMjIERvY3VtZW50YXRpb24sIFZlcnNpb25zLCBhbmQgQ2hhbmdlbG9nCgoqIFtEb2N1bWVudGF0aW9uXShkb2NzL2luZGV4Lm1kKS4KKiBTZWUgW2NoYW5nZWxvZ10oQ2hhbmdlbG9nLm1kKSBmb3IgdmVyc2lvbiBhbmQgY2hhbmdlcy4KCiMjIFJlZmVyZW5jZXMKCiogT3BlbkFQSSBzcGVjIDMuMS4wLiBbaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4xLjAubWRdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMS4wLm1kKQoKIyMgTGljZW5zZQoKTGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLgoKIyMgQ3JlZGl0cwoKKipDb250YWN0OioqIFBlZHJvIEouIE1vbGluYSB8IGdpdGh1YjogW3BqbW9saW5hXShodHRwczovL2dpdGh1Yi5jb20vcGptb2xpbmEpIHwgdHdpdHRlcjogW3Btb2xpbmFtXShodHRwczovL3R3aXR0ZXIuY29tL3Btb2xpbmFtKQoKKGMpIDIwMTctMjAyNC4gW1BlZHJvIEouIE1vbGluYV0oaHR0cDovL3BqbW9saW5hLmNvbSkgYXQgTWV0YWRldiBTLkwuIFtodHRwczovL21ldGFkZXYucHJvXShodHRwczovL21ldGFkZXYucHJvKSAmIGNvbnRyaWJ1dG9ycy4K readmeEtag: '"12f647d08b21592b28224cc4ed886b16d9bc840f"' readmeLastModified: Mon, 10 Jun 2024 15:05:07 GMT repositoryId: 84108450 description: TS Model & utils for creating and exposing OpenAPI 3.x contracts. created: '2017-03-06T18:47:56Z' updated: '2026-02-02T17:17:07Z' language: TypeScript archived: false stars: 540 watchers: 6 forks: 63 owner: metadevpro logo: https://avatars.githubusercontent.com/u/24300914?v=4 license: MIT repoEtag: '"02d25ab98b83f3d4bdd142b6a6434f81fb20907fa8ce4a8f05059e298f10d98f"' repoLastModified: Mon, 02 Feb 2026 17:17:07 GMT foundInMaster: true id: 390339b2a28f2e5660519a2422c29196 - source: - https://openapi.tools/ - openapi3 tags name: kin-openapi homepage: https://github.com/getkin/kin-openapi language: Go source_description: >- OpenAPI 3.0 (and Swagger v2) implementation for Go (parsing, converting, validation, and more) category: - Parsers - Data Validators repository: https://github.com/getkin/kin-openapi v3: true repositoryMetadata: base64Readme: >- [![CI](https://github.com/getkin/kin-openapi/workflows/go/badge.svg)](https://github.com/getkin/kin-openapi/actions)
[![Go Report Card](https://goreportcard.com/badge/github.com/getkin/kin-openapi)](https://goreportcard.com/report/github.com/getkin/kin-openapi)
[![GoDoc](https://godoc.org/github.com/getkin/kin-openapi?status.svg)](https://godoc.org/github.com/getkin/kin-openapi)
[![Join Gitter Chat Channel -](https://badges.gitter.im/getkin/kin.svg)](https://gitter.im/getkin/kin?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)

# Introduction
A [Go](https://golang.org) project for handling [OpenAPI](https://www.openapis.org/) files. We target:
* [OpenAPI `v2.0`](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/2.0.md) (formerly known as Swagger)
* [OpenAPI `v3.0`](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md)
* [OpenAPI `v3.1`](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md) Soon! [Tracking issue here.](https://github.com/getkin/kin-openapi/issues/230)

Licensed under the [MIT License](./LICENSE).

## Contributors, users and sponsors
The project has received pull requests [from many people](https://github.com/getkin/kin-openapi/graphs/contributors). Thanks to everyone!

Please, [give back to this project](https://github.com/sponsors/fenollp) by becoming a sponsor.

Here's some projects that depend on _kin-openapi_:
  * [github.com/go-fuego/fuego](https://github.com/go-fuego/fuego) - "Framework generating OpenAPI 3 spec from source code"
  * [github.com/a-h/rest](https://github.com/a-h/rest) - "Generate OpenAPI 3.0 specifications from Go code without annotations or magic comments"
  * [github.com/Tufin/oasdiff](https://github.com/Tufin/oasdiff) - "A diff tool for OpenAPI Specification 3"
  * [github.com/danielgtaylor/apisprout](https://github.com/danielgtaylor/apisprout) - "Lightweight, blazing fast, cross-platform OpenAPI 3 mock server with validation"
  * [github.com/oapi-codegen/oapi-codegen](https://github.com/oapi-codegen/oapi-codegen) - "Generate Go client and server boilerplate from OpenAPI 3 specifications"
  * [github.com/dunglas/vulcain](https://github.com/dunglas/vulcain) - "Use HTTP/2 Server Push to create fast and idiomatic client-driven REST APIs"
  * [github.com/danielgtaylor/restish](https://github.com/danielgtaylor/restish) - "...a CLI for interacting with REST-ish HTTP APIs with some nice features built-in"
  * [github.com/goadesign/goa](https://github.com/goadesign/goa) - "Design-based APIs and microservices in Go"
  * [github.com/hashicorp/nomad-openapi](https://github.com/hashicorp/nomad-openapi) - "Nomad is an easy-to-use, flexible, and performant workload orchestrator that can deploy a mix of microservice, batch, containerized, and non-containerized applications. Nomad is easy to operate and scale and has native Consul and Vault integrations."
  * [gitlab.com/jamietanna/httptest-openapi](https://gitlab.com/jamietanna/httptest-openapi) ([*blog post*](https://www.jvt.me/posts/2022/05/22/go-openapi-contract-test/)) - "Go OpenAPI Contract Verification for use with `net/http`"
  * [github.com/SIMITGROUP/openapigenerator](https://github.com/SIMITGROUP/openapigenerator) - "Openapi v3 microservices generator"
  * [https://github.com/projectsveltos/addon-controller](https://github.com/projectsveltos/addon-controller) - "Kubernetes add-on controller designed to manage tens of clusters."
  * (Feel free to add your project by [creating an issue](https://github.com/getkin/kin-openapi/issues/new) or a pull request)

## Alternatives
* [libopenapi](https://github.com/pb33f/libopenapi) a fully featured, high performance OpenAPI 3.1, 3.0 and Swagger parser, library, validator and toolkit
* [go-swagger](https://github.com/go-swagger/go-swagger) stated [*OpenAPIv3 won't be supported*](https://github.com/go-swagger/go-swagger/issues/1122#issuecomment-575968499)
* [swaggo](https://github.com/swaggo/swag) has an [open issue on OpenAPIv3](https://github.com/swaggo/swag/issues/386)
* [go-openapi](https://github.com/go-openapi)'s [spec3](https://github.com/go-openapi/spec3)
	* an iteration on [spec](https://github.com/go-openapi/spec) (for OpenAPIv2)
	* see [README](https://github.com/go-openapi/spec3/tree/3fab9faa9094e06ebd19ded7ea96d156c2283dca#oai-object-model---) for the missing parts

Be sure to check [OpenAPI Initiative](https://github.com/OAI)'s [great tooling list](https://github.com/OAI/OpenAPI-Specification/blob/master/IMPLEMENTATIONS.md) as well as [OpenAPI.Tools](https://openapi.tools/).

# Structure
  * _openapi2_ ([godoc](https://godoc.org/github.com/getkin/kin-openapi/openapi2))
    * Support for OpenAPI 2 files, including serialization, deserialization, and validation.
  * _openapi2conv_ ([godoc](https://godoc.org/github.com/getkin/kin-openapi/openapi2conv))
    * Converts OpenAPI 2 files into OpenAPI 3 files.
  * _openapi3_ ([godoc](https://godoc.org/github.com/getkin/kin-openapi/openapi3))
    * Support for OpenAPI 3 files, including serialization, deserialization, and validation.
  * _openapi3filter_ ([godoc](https://godoc.org/github.com/getkin/kin-openapi/openapi3filter))
    * Validates HTTP requests and responses
    * Provides a [gorilla/mux](https://github.com/gorilla/mux) router for OpenAPI operations
  * _openapi3gen_ ([godoc](https://godoc.org/github.com/getkin/kin-openapi/openapi3gen))
    * Generates `*openapi3.Schema` values for Go types.

# Some recipes
## Validating an OpenAPI document
```shell
go run github.com/getkin/kin-openapi/cmd/validate@latest [--circular] [--defaults] [--examples] [--ext] [--patterns] -- <local YAML or JSON file>
```

## Loading OpenAPI document
Use `openapi3.Loader`, which resolves all references:
```go
loader := openapi3.NewLoader()
doc, err := loader.LoadFromFile("my-openapi-spec.json")
```

## Getting OpenAPI operation that matches request
```go
loader := openapi3.NewLoader()
doc, _ := loader.LoadFromData([]byte(`...`))
_ = doc.Validate(loader.Context)
router, _ := gorillamux.NewRouter(doc)
route, pathParams, _ := router.FindRoute(httpRequest)
// Do something with route.Operation
```

## Validating HTTP requests/responses
```go
package main

import (
	"context"
	"fmt"
	"net/http"

	"github.com/getkin/kin-openapi/openapi3"
	"github.com/getkin/kin-openapi/openapi3filter"
	"github.com/getkin/kin-openapi/routers/gorillamux"
)

func main() {
	ctx := context.Background()
	loader := &openapi3.Loader{Context: ctx, IsExternalRefsAllowed: true}
	doc, _ := loader.LoadFromFile(".../My-OpenAPIv3-API.yml")
	// Validate document
	_ = doc.Validate(ctx)
	router, _ := gorillamux.NewRouter(doc)
	httpReq, _ := http.NewRequest(http.MethodGet, "/items", nil)

	// Find route
	route, pathParams, _ := router.FindRoute(httpReq)

	// Validate request
	requestValidationInput := &openapi3filter.RequestValidationInput{
		Request:    httpReq,
		PathParams: pathParams,
		Route:      route,
	}
	_ = openapi3filter.ValidateRequest(ctx, requestValidationInput)

	// Handle that request
	// --> YOUR CODE GOES HERE <--
	responseHeaders := http.Header{"Content-Type": []string{"application/json"}}
	responseCode := 200
	responseBody := []byte(`{}`)

	// Validate response
	responseValidationInput := &openapi3filter.ResponseValidationInput{
		RequestValidationInput: requestValidationInput,
		Status:                 responseCode,
		Header:                 responseHeaders,
	}
	responseValidationInput.SetBodyBytes(responseBody)
	_ = openapi3filter.ValidateResponse(ctx, responseValidationInput)
}
```

## Custom content type for body of HTTP request/response

By default, the library parses a body of the HTTP request and response of [a few content types](https://github.com/getkin/kin-openapi/blob/6da871e0e170b7637eb568c265c08bc2b5d6e7a3/openapi3filter/req_resp_decoder.go#L1264) e.g. `"text/plain"` or `"application/json"`.
To support other content types you must register decoders for them:

```go
func main() {
	// ...

	// Register a body's decoder for content type "application/xml".
	openapi3filter.RegisterBodyDecoder("application/xml", xmlBodyDecoder)

	// Now you can validate HTTP request that contains a body with content type "application/xml".
	requestValidationInput := &openapi3filter.RequestValidationInput{
		Request:    httpReq,
		PathParams: pathParams,
		Route:      route,
	}
	if err := openapi3filter.ValidateRequest(ctx, requestValidationInput); err != nil {
		panic(err)
	}

	// ...

	// And you can validate HTTP response that contains a body with content type "application/xml".
	if err := openapi3filter.ValidateResponse(ctx, responseValidationInput); err != nil {
		panic(err)
	}
}

func xmlBodyDecoder(body io.Reader, h http.Header, schema *openapi3.SchemaRef, encFn openapi3filter.EncodingFn) (decoded any, err error) {
	// Decode body to a primitive, []any, or map[string]any.
}
```

## Custom function to check uniqueness of array items

By default, the library checks unique items using the following predefined function:

```go
func isSliceOfUniqueItems(xs []any) bool {
	s := len(xs)
	m := make(map[string]struct{}, s)
	for _, x := range xs {
		key, _ := json.Marshal(&x)
		m[string(key)] = struct{}{}
	}
	return s == len(m)
}
```

In the predefined function `json.Marshal` is used to generate a string that can
be used as a map key which is to check the uniqueness of an array
when the array items are objects or arrays. You can register
you own function according to your input data to get better performance:

```go
func main() {
	// ...

	// Register a customized function used to check uniqueness of array.
	openapi3.RegisterArrayUniqueItemsChecker(arrayUniqueItemsChecker)

	// ... other validate codes
}

func arrayUniqueItemsChecker(items []any) bool {
	// Check the uniqueness of the input slice
}
```

## Custom function to change schema error messages

By default, the error message returned when validating a value includes the error reason, the schema, and the input value.

For example, given the following schema:

```json
{
  "type": "string",
  "allOf": [
    { "pattern": "[A-Z]" },
    { "pattern": "[a-z]" },
    { "pattern": "[0-9]" },
    { "pattern": "[!@#$%^&*()_+=-?~]" }
  ]
}
```

Passing the input value `"secret"` to this schema will produce the following error message:

```
string doesn't match the regular expression "[A-Z]"
Schema:
  {
    "pattern": "[A-Z]"
  }

Value:
  "secret"
```

Including the original value in the error message can be helpful for debugging, but it may not be appropriate for sensitive information such as secrets.

To disable the extra details in the schema error message, you can set the `openapi3.SchemaErrorDetailsDisabled` option to `true`:

```go
func main() {
	// ...

	// Disable schema error detailed error messages
	openapi3.SchemaErrorDetailsDisabled = true

	// ... other validate codes
}
```

This will shorten the error message to present only the reason:

```
string doesn't match the regular expression "[A-Z]"
```

For more fine-grained control over the error message, you can pass a custom `openapi3filter.Options` object to `openapi3filter.RequestValidationInput` that includes a `openapi3filter.CustomSchemaErrorFunc`.

```go
func validationOptions() *openapi3filter.Options {
	options := &openapi3filter.Options{}
	options.WithCustomSchemaErrorFunc(safeErrorMessage)
	return options
}

func safeErrorMessage(err *openapi3.SchemaError) string {
	return err.Reason
}
```

This will change the schema validation errors to return only the `Reason` field, which is guaranteed to not include the original value.

## Reconciling component $ref types

`ReferencesComponentInRootDocument` is a useful helper function to check if a component reference
coincides with a reference in the root document's component objects fixed fields.

This can be used to determine if two schema definitions are of the same structure, helpful for
code generation tools when generating go type models.

```go
doc, err = loader.LoadFromFile("openapi.yml")

for _, path := range doc.Paths.InMatchingOrder() {
	pathItem := doc.Paths.Find(path)

	if pathItem.Get == nil || pathItem.Get.Responses.Status(200) {
		continue
	}

	for _, s := range pathItem.Get.Responses.Status(200).Value.Content {
		name, match := ReferencesComponentInRootDocument(doc, s.Schema)
		fmt.Println(path, match, name) // /record true #/components/schemas/BookRecord
	}
}
```

## CHANGELOG: Sub-v1 breaking API changes

### v0.131.0
* No longer `openapi3filter.RegisterBodyDecoder` the `openapi3filter.ZipFileBodyDecoder` by default.

### v0.129.0
* `openapi3.Discriminator.Mapping` and `openapi3.OAuthFlow.Scopes` fields went from a `map[string]string` to the new type `StringMap`

### v0.127.0
* Downgraded `github.com/gorilla/mux` dep from `1.8.1` to `1.8.0`.

### v0.126.0
* `openapi3.CircularReferenceError` and `openapi3.CircularReferenceCounter` are removed. `openapi3.Loader` now implements reference backtracking, so any kind of circular references should be properly resolved.
* `InternalizeRefs` now takes a refNameResolver that has access to `openapi3.T` and more properties of the reference needing resolving.
* The `DefaultRefNameResolver` has been updated, choosing names that will be less likely to collide with each other. Because of this internalized specs will likely change slightly.
* `openapi3.Format` and `openapi3.FormatCallback` are removed and the type of `openapi3.SchemaStringFormats` has changed.

### v0.125.0
* The `openapi3filter.ErrFunc` and `openapi3filter.LogFunc` func types now take the validated request's context as first argument.

### v0.124.0
* `openapi3.Schema.Type` & `openapi2.Parameter.Type` fields went from a `string` to the type `*Type` with methods: `Includes`, `Is`, `Permits` & `Slice`.

### v0.122.0
* `Paths` field of `openapi3.T` is now a pointer
* `Responses` field of `openapi3.Operation` is now a pointer
* `openapi3.Paths` went from `map[string]*PathItem` to a struct with an `Extensions` field and methods: `Set`, `Value`, `Len`, `Map`, and `New*`.
* `openapi3.Callback` went from `map[string]*PathItem` to a struct with an `Extensions` field and methods: `Set`, `Value`, `Len`, `Map`, and `New*`.
* `openapi3.Responses` went from `map[string]*ResponseRef` to a struct with an `Extensions` field and methods: `Set`, `Value`, `Len`, `Map`, and `New*`.
* `(openapi3.Responses).Get(int)` renamed to `(*openapi3.Responses).Status(int)`

### v0.121.0
* Introduce `openapi3.RequestBodies` (an alias on `map[string]*openapi3.ResponseRef`) and use it in place of `openapi3.Responses` for field `openapi3.Components.Responses`.

### v0.116.0
* Dropped `openapi3filter.DefaultOptions`. Use `&openapi3filter.Options{}` directly instead.

### v0.113.0
* The string format `email` has been removed by default. To use it please call `openapi3.DefineStringFormat("email", openapi3.FormatOfStringForEmail)`.
* Field `openapi3.T.Components` is now a pointer.
* Fields `openapi3.Schema.AdditionalProperties` and `openapi3.Schema.AdditionalPropertiesAllowed` are replaced by `openapi3.Schema.AdditionalProperties.Schema` and `openapi3.Schema.AdditionalProperties.Has` respectively.
* Type `openapi3.ExtensionProps` is now just `map[string]any` and extensions are accessible through the `Extensions` field.

### v0.112.0
* `(openapi3.ValidationOptions).ExamplesValidationDisabled` has been unexported.
* `(openapi3.ValidationOptions).SchemaFormatValidationEnabled` has been unexported.
* `(openapi3.ValidationOptions).SchemaPatternValidationDisabled` has been unexported.

### v0.111.0
* Changed `func (*_) Validate(ctx context.Context) error` to `func (*_) Validate(ctx context.Context, opts ...ValidationOption) error`.
* `openapi3.WithValidationOptions(ctx context.Context, opts *ValidationOptions) context.Context` prototype changed to `openapi3.WithValidationOptions(ctx context.Context, opts ...ValidationOption) context.Context`.

### v0.101.0
* `openapi3.SchemaFormatValidationDisabled` has been removed in favour of an option `openapi3.EnableSchemaFormatValidation()` passed to `openapi3.T.Validate`. The default behaviour is also now to not validate formats, as the OpenAPI spec mentions the `format` is an open value.

### v0.84.0
* The prototype of `openapi3gen.NewSchemaRefForValue` changed:
	* It no longer returns a map but that is still accessible under the field `(*Generator).SchemaRefs`.
	* It now takes in an additional argument (basically `doc.Components.Schemas`) which gets written to so `$ref` cycles can be properly handled.

### v0.61.0
* Renamed `openapi2.Swagger` to `openapi2.T`.
* Renamed `openapi2conv.FromV3Swagger` to `openapi2conv.FromV3`.
* Renamed `openapi2conv.ToV3Swagger` to `openapi2conv.ToV3`.
* Renamed `openapi3.LoadSwaggerFromData` to `openapi3.LoadFromData`.
* Renamed `openapi3.LoadSwaggerFromDataWithPath` to `openapi3.LoadFromDataWithPath`.
* Renamed `openapi3.LoadSwaggerFromFile` to `openapi3.LoadFromFile`.
* Renamed `openapi3.LoadSwaggerFromURI` to `openapi3.LoadFromURI`.
* Renamed `openapi3.NewSwaggerLoader` to `openapi3.NewLoader`.
* Renamed `openapi3.Swagger` to `openapi3.T`.
* Renamed `openapi3.SwaggerLoader` to `openapi3.Loader`.
* Renamed `openapi3filter.ValidationHandler.SwaggerFile` to `openapi3filter.ValidationHandler.File`.
* Renamed `routers.Route.Swagger` to `routers.Route.Spec`.

### v0.51.0
* Type `openapi3filter.Route` moved to `routers` (and `Route.Handler` was dropped. See https://github.com/getkin/kin-openapi/issues/329)
* Type `openapi3filter.RouteError` moved to `routers` (so did `ErrPathNotFound` and `ErrMethodNotAllowed` which are now `RouteError`s)
* Routers' `FindRoute(...)` method now takes only one argument: `*http.Request`
* `getkin/kin-openapi/openapi3filter.Router` moved to `getkin/kin-openapi/routers/legacy`
* `openapi3filter.NewRouter()` and its related `WithSwaggerFromFile(string)`, `WithSwagger(*openapi3.Swagger)`, `AddSwaggerFromFile(string)` and `AddSwagger(*openapi3.Swagger)` are all replaced with a single `<router package>.NewRouter(*openapi3.Swagger)`
	* NOTE: the `NewRouter(doc)` call now requires that the user ensures `doc` is valid (`doc.Validate() != nil`). This used to be asserted.

### v0.47.0
Field `(*openapi3.SwaggerLoader).LoadSwaggerFromURIFunc` of type `func(*openapi3.SwaggerLoader, *url.URL) (*openapi3.Swagger, error)` was removed after the addition of the field `(*openapi3.SwaggerLoader).ReadFromURIFunc` of type `func(*openapi3.SwaggerLoader, *url.URL) ([]byte, error)`.
 readmeEtag: '"58a612844fadf6ce6b7e728d11bac7728a83b36e"' readmeLastModified: Thu, 02 Oct 2025 09:23:15 GMT repositoryId: 94029207 description: >- OpenAPI 3.0 (and Swagger v2) implementation for Go (parsing, converting, validation, and more) created: '2017-06-11T20:12:06Z' updated: '2026-02-05T19:43:42Z' language: Go archived: false stars: 3153 watchers: 20 forks: 488 owner: getkin logo: https://avatars.githubusercontent.com/u/40365715?v=4 license: MIT repoEtag: '"57777eff14dbf6dc3bc0813ef59dc76c21d4f08a54741639a1e708c7d02868f6"' repoLastModified: Thu, 05 Feb 2026 19:43:42 GMT foundInMaster: true v2: true id: f6838ce08e87fa4c3dc813c97193ece3 oldLocations: - https://github.com/jban332/kin-openapi - source: https://openapi.tools/ name: openapi.tanna.dev/go/validator category: Data Validators repository: https://gitlab.com/jamietanna/httptest-openapi/ language: Go source_description: >- A Go library for making it easier to validate that your OpenAPI contracts match your HTTP request/responses used in your tests. Based on kin-openapi. v3: true id: 839cc8f67fdfb81f410fbc29ab770cc8 foundInMaster: true - source: https://openapi.tools/ name: openapi-psr7-validator category: Data Validators repository: https://github.com/thephpleague/openapi-psr7-validator language: PHP source_description: >- Using a PHP framework that supports PSR-7? Get free validation without writing a bunch of code, by registering this middleware and pointing it at your API description document. v3: true repositoryMetadata: base64Readme: >- [![Latest Stable Version](https://poser.pugx.org/league/openapi-psr7-validator/v/stable)](https://packagist.org/packages/league/openapi-psr7-validator)
[![Build Status](https://travis-ci.org/thephpleague/openapi-psr7-validator.svg?branch=master)](https://travis-ci.org/thephpleague/openapi-psr7-validator)
[![License](https://poser.pugx.org/league/openapi-psr7-validator/license)](https://packagist.org/packages/lezhnev74/openapi-psr7-validator)
![contributions welcome](https://img.shields.io/badge/contributions-welcome-brightgreen.svg)

# OpenAPI PSR-7 Message (HTTP Request/Response) Validator
This package can validate PSR-7 messages against OpenAPI (3.0.x) specifications 
expressed in YAML or JSON. 

![](image.jpg)

## Installation
```
composer require league/openapi-psr7-validator
```

## OpenAPI (OAS) Terms
There are some specific terms that are used in the package. These terms come 
from OpenAPI:
- `specification` - an OpenAPI document describing an API, expressed in JSON or YAML file
- `data` - actual thing that we validate against a specification, including body and metadata
- `schema` - the part of the specification that describes the body of the request / response
- `keyword` - properties that are used to describe the instance are called key
words, or schema keywords
- `path` - a relative path to an individual endpoint
- `operation` - a method that we apply on the path (like `get /password`)
- `response` - described response (includes status code, content types etc)


## How To Validate

### ServerRequest Message
You can validate `\Psr\Http\Message\ServerRequestInterface` instance like this:

```php
$yamlFile = "api.yaml";
$jsonFile = "api.json";

$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getServerRequestValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYaml(file_get_contents($yamlFile))->getServerRequestValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromJson(file_get_contents($jsonFile))->getServerRequestValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromJsonFile($jsonFile)->getServerRequestValidator();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getServerRequestValidator();

$match = $validator->validate($request);
```

As a result you would get and `OperationAddress $match` which has matched the given request. If you already know
the operation which should match your request (i.e you have routing in your project), you can use 
`RouterRequestValidator`

```php
$address = new \League\OpenAPIValidation\PSR7\OperationAddress('/some/operation', 'post');

$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getRoutedRequestValidator();

$validator->validate($address, $request);
```

This would simplify validation a lot and give you more performance.

### Request Message
You can validate `\Psr\Http\Message\RequestInterface` instance like this:

```php
$yamlFile = "api.yaml";
$jsonFile = "api.json";

$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getRequestValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYaml(file_get_contents($yamlFile))->getRequestValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromJson(file_get_contents($jsonFile))->getRequestValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromJsonFile($jsonFile)->getRequestValidator();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getRequestValidator();

$match = $validator->validate($request);
```

### Response Message
Validation of `\Psr\Http\Message\ResponseInterface` is a bit more complicated
. Because you need not only YAML file and Response itself, but also you need 
to know which operation this response belongs to (in terms of OpenAPI).

Example:

```php
$yamlFile = "api.yaml";
$jsonFile = "api.json";

$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getResponseValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYaml(file_get_contents($yamlFile))->getResponseValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromJson(file_get_contents($jsonFile))->getResponseValidator();
#or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromJsonFile($jsonFile)->getResponseValidator();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getResponseValidator();

$operation = new \League\OpenAPIValidation\PSR7\OperationAddress('/password/gen', 'get') ;

$validator->validate($operation, $response);
```

### Reuse Schema After Validation

`\League\OpenAPIValidation\PSR7\ValidatorBuilder` reads and compiles schema in memory as instance of `\cebe\openapi\spec\OpenApi`. Validators use this instance to perform validation logic. You can reuse this instance after the validation like this:

```php
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getServerRequestValidator();
# or
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getResponseValidator();

/** @var \cebe\openapi\spec\OpenApi */
$openApi = $validator->getSchema();
```

### PSR-15 Middleware
PSR-15 middleware can be used like this:

```php
$yamlFile = 'api.yaml';
$jsonFile = 'api.json';

$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYamlFile($yamlFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYaml(file_get_contents($yamlFile))->getValidationMiddleware();
#or
$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJsonFile($jsonFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJson(file_get_contents($jsonFile))->getValidationMiddleware();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \League\OpenAPIValidation\PSR7\ValidationMiddlewareBuilder)->fromSchema($schema)->getValidationMiddleware();
```

### SlimFramework Middleware
Slim framework uses slightly different middleware interface, so here is an 
adapter which you can use like this:

```php
$yamlFile = 'api.yaml';
$jsonFile = 'api.json';

$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYamlFile($yamlFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYaml(file_get_contents($yamlFile))->getValidationMiddleware();
#or
$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJsonFile($jsonFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \League\OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJson(file_get_contents($jsonFile))->getValidationMiddleware();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \League\OpenAPIValidation\PSR7\ValidationMiddlewareBuilder)->fromSchema($schema)->getValidationMiddleware();

$slimMiddleware = new \League\OpenAPIValidation\PSR15\SlimAdapter($psr15Middleware);

/** @var \Slim\App $app */
$app->add($slimMiddleware);
```

### Caching Layer / PSR-6 Support
PSR-7 Validator has a built-in caching layer (based on [PSR-6](https://www.php-fig.org/psr/psr-6/) interfaces) which saves time on parsing OpenAPI specs. It is optional.
You enable caching if you pass a configured Cache Pool Object to the static constructor like this:
```php
// Configure a PSR-6 Cache Pool
$cachePool = new ArrayCachePool();

// Pass it as a 2nd argument
$validator = (new \League\OpenAPIValidation\PSR7\ValidatorBuilder)
    ->fromYamlFile($yamlFile)
    ->setCache($cachePool)
    ->getResponseValidator();
# or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)
    ->fromYamlFile($yamlFile)
    ->setCache($cachePool)
    ->getValidationMiddleware();
```

You can use `->setCache($pool, $ttl)` call for both PSR-7 and PSR-15 builder in order to set 
[proper expiration ttl in seconds (or explicit `null`)](https://www.php-fig.org/psr/psr-6/#definitions)

If you want take control over the cache key for schema item, or your cache does not support cache key generation by itself
you can `->overrideCacheKey('my_custom_key')` to ensure cache uses key you want.

### Standalone OpenAPI Validator
The package contains a standalone validator which can validate any data 
against an OpenAPI schema like this:

```php
$spec = <<<SPEC
schema:
  type: string
  enum:
  - a
  - b
SPEC;
$data = "c";

$spec   = cebe\openapi\Reader::readFromYaml($spec);
# (optional) reference resolving
$spec->resolveReferences(new ReferenceContext($spec, "/"));
$schema = new cebe\openapi\spec\Schema($spec->schema);

try {
    (new \League\OpenAPIValidation\Schema\SchemaValidator())->validate($data, $schema);
} catch(\League\OpenAPIValidation\Schema\Exception\KeywordMismatch $e) {
    // you can evaluate failure details
    // $e->keyword() == "enum"
    // $e->data() == "c"
    // $e->dataBreadCrumb()->buildChain() -- only for nested data
}
```

## Custom Type Formats
As you know, OpenAPI allows you to add formats to types:

```yaml
schema:
  type: string
  format: binary
```

This package contains a bunch of built-in format validators:
- `string` type:
    - `byte`
    - `date`
    - `date-time`
    - `email`
    - `hostname`
    - `ipv4`
    - `ipv6`
    - `uri`
    - `uuid` (uuid4)
- `number` type
    - `float`
    - `double`

You can also add your own formats. Like this:
```php
# A format validator must be a callable
# It must return bool value (true if format matched the data, false otherwise)

# A callable class:
$customFormat = new class()
{
    function __invoke($value): bool
    {
        return $value === "good value";
    }
};

# Or just a closure:
$customFormat = function ($value): bool {
    return $value === "good value";
};

# Register your callable like this before validating your data
\League\OpenAPIValidation\Schema\TypeFormats\FormatsContainer::registerFormat('string', 'custom', $customFormat);
```

## Exceptions
The package throws a list of various exceptions which you can catch and handle. There are some of them:
- Schema related:
    - `\League\OpenAPIValidation\Schema\Exception\KeywordMismatch` - Indicates that data was not matched against a schema's keyword
        - `\League\OpenAPIValidation\Schema\Exception\TypeMismatch` - Validation for `type` keyword failed against a given data. For example `type:string` and value is `12`
        - `\League\OpenAPIValidation\Schema\Exception\FormatMismatch` - data mismatched a given type format. For example `type: string, format: email` won't match `not-email`.
- PSR7 Messages related:
    - `\League\OpenAPIValidation\PSR7\Exception\NoContentType` - HTTP message(request/response) contains no Content-Type header. General HTTP errors.
    - `\League\OpenAPIValidation\PSR7\Exception\NoPath` - path is not found in the spec
    - `\League\OpenAPIValidation\PSR7\Exception\NoOperation` - operation os not found in the path
    - `\League\OpenAPIValidation\PSR7\Exception\NoResponseCode` - response code not found under the operation in the spec
    - Validation exceptions (check parent exception for possible root causes):
        - `\League\OpenAPIValidation\PSR7\Exception\ValidationFailed` - generic exception for failed PSR-7 message
        - `\League\OpenAPIValidation\PSR7\Exception\Validation\InvalidBody` - body does not match schema
        - `\League\OpenAPIValidation\PSR7\Exception\Validation\InvalidCookies` - cookies does not match schema or missing required cookie
        - `\League\OpenAPIValidation\PSR7\Exception\Validation\InvalidHeaders` - header does not match schema or missing required header
        - `\League\OpenAPIValidation\PSR7\Exception\Validation\InvalidPath` - path does not match pattern or pattern values does not match schema
        - `\League\OpenAPIValidation\PSR7\Exception\Validation\InvalidQueryArgs` - query args does not match schema or missing required argument
        - `\League\OpenAPIValidation\PSR7\Exception\Validation\InvalidSecurity` - request does not match security schema or invalid security headers
    - Request related:
        - `\League\OpenAPIValidation\PSR7\Exception\MultipleOperationsMismatchForRequest` - request matched multiple operations in the spec, 
        but validation failed for all of them.

## Testing
You can run the tests with:

```
vendor/bin/phpunit
```

## Contribution Guide
Feel free to open an Issue or add a Pull request. 
There is a certain code style that this package follows: [doctrine/coding-standard](https://www.doctrine-project.org/projects/doctrine-coding-standard/en/latest/reference/index.html#introduction).

To conform to this style please use a git hook, shipped with this package at `.githooks/pre-commit`.

How to use it:
1. Clone the package locally and navigate to the folder
2. Create a symlink to the hook like this: `ln -s -f ../../.githooks/pre-commit .git/hooks/pre-commit`
3. Add execution rights: `chmod +x .git/hooks/pre-commit`
4. Now commit any new changes and the code will be checked and formatted accordingly.
5. If there are any issues with your code, check the log here: `.phpcs-report.txt`

## Credits
People:
- [Dmitry Lezhnev](https://github.com/lezhnev74)
- [Carsten Brandt](https://github.com/cebe)
- [Samuel Nela](https://github.com/samnela)
- [Pavel Batanov](https://github.com/scaytrase)
- [Christopher L Bray](https://github.com/brayniverse)
- [David Pauli](https://github.com/dpauli)
- [Jason Judge](https://github.com/judgej)
- [Yannick Chenot](https://github.com/osteel)
- [TarasBK](https://github.com/TarasBK)
- [Jason B. Standing](https://github.com/jasonbstanding)
- [Dmytro Demchyna](https://github.com/dmytro-demchyna)
- [Will Chambers](https://github.com/willchambers99)
- [Ignacio](https://github.com/imefisto)
- A big thank you to [Henrik Karlström](https://github.com/hkarlstrom) who kind of inspired me to work on this package. 

Resources:
- Icons made by Freepik, licensed by CC 3.0 BY
- [cebe/php-openapi](https://github.com/cebe/php-openapi) package for Reading OpenAPI files
- [slim3-psr15](https://github.com/bnf/slim3-psr15) package for Slim middleware adapter
 
## License
The MIT License (MIT). Please see `License.md` file for more information.

## TODO
- [ ] Support Discriminator Object (note: apparently, this is not so straightforward, as discriminator can point to any external scheme)
 readmeEtag: '"bb4441fcf3a344c1448c23f38f6b5d2b6242e9ce"' readmeLastModified: Wed, 10 Jan 2024 19:11:09 GMT repositoryId: 209473440 description: >- It validates PSR-7 messages (HTTP request/response) against OpenAPI specifications created: '2019-09-19T05:57:05Z' updated: '2026-01-28T14:59:59Z' language: PHP archived: false stars: 553 watchers: 21 forks: 107 owner: thephpleague logo: https://avatars.githubusercontent.com/u/527621?v=4 license: MIT repoEtag: '"54ce419a2bd836e74252536fe0971a769e25c1c8d3c52eca2365c8ee6dddcc61"' repoLastModified: Wed, 28 Jan 2026 14:59:59 GMT foundInMaster: true v3_1_link: https://github.com/thephpleague/openapi-psr7-validator/issues/163 id: 5db76a91d3b4f650045f409b9e0f4e9b v3_1: true - source: - https://openapi.tools/ - openapi3 tags name: php-openapi category: - Parsers - Description Validators repository: https://github.com/cebe/php-openapi language: PHP source_description: A PHP library for manipulating and validating OpenAPI 3.0 Descriptions v3: true repositoryMetadata: base64Readme: >- # php-openapi

Read and write [OpenAPI](https://www.openapis.org/) 3.0.x YAML and JSON files and make the content accessible in PHP objects.

It also provides a CLI tool for validating and converting OpenAPI 3.0.x Description files.

[![Latest Stable Version](https://poser.pugx.org/cebe/php-openapi/v/stable)](https://packagist.org/packages/cebe/php-openapi)
[![Total Downloads](https://poser.pugx.org/cebe/php-openapi/downloads)](https://packagist.org/packages/cebe/php-openapi)
[![Build Status](https://github.com/cebe/php-openapi/workflows/CI/badge.svg)](https://github.com/cebe/php-openapi/actions)


## Install

    composer require cebe/php-openapi

## Requirements

- PHP 7.1 or higher (works fine with PHP 8)

## Used by

This library provides a low level API for reading and writing OpenAPI files. It is used by higher level tools to
do awesome work:

- [cebe/yii2-openapi](https://github.com/cebe/yii2-openapi) Code Generator for REST API from OpenAPI 3 Descriptions, includes fake data generator.
- [cebe/yii2-app-api](https://github.com/cebe/yii2-app-api) Yii framework application template for developing API-first applications.
- [league/openapi-psr7-validator](https://github.com/thephpleague/openapi-psr7-validator) validates PSR-7 messages (HTTP request/response) against OpenAPI descriptions.
- [dsuurlant/response2schema](https://github.com/dsuurlant/response2schema) a quick and easy tool for generating OpenAPI schemas based on example data.
- [hotmeteor/spectator](https://github.com/hotmeteor/spectator) a light-weight OpenAPI testing tool for existing Laravel test suite.
- [googoogajoob/openapi-slim4](https://github.com/googoogajoob/openapi-slim4) Configure the paths of a slim4 application from an openapi definition.
- ... ([add yours](https://github.com/cebe/php-openapi/edit/master/README.md#L24))

## Usage

### CLI Tool

    $ vendor/bin/php-openapi help
    PHP OpenAPI 3 tool
    ------------------
    by Carsten Brandt <mail@cebe.cc>

    Usage:
      php-openapi <command> [<options>] [input.yml|input.json] [output.yml|output.json]

      The following commands are available:

        validate   Validate the API Description in the specified input file against the OpenAPI v3.0 schema.
                   Note: the validation is performed in two steps. The results are composed of
                    (1) structural errors found while reading the API Description file, and
                    (2) violations of the OpenAPI v3.0 schema.

                   If no input file is specified input will be read from STDIN.
                   The tool will try to auto-detect the content type of the input, but may fail
                   to do so. You may specify --read-yaml or --read-json to force the file type.

                   Exits with code 2 on validation errors, 1 on other errors and 0 on success.

        convert    Convert a JSON or YAML input file to JSON or YAML output file.

                   If no input file is specified input will be read from STDIN.
                   If no output file is specified output will be written to STDOUT.
                   The tool will try to auto-detect the content type of the input and output file, but may fail
                   to do so. You may specify --read-yaml or --read-json to force the input file type.
                   and --write-yaml or --write-json to force the output file type.

                   By default all references are resolved (replaced with the object referred to). You can control
                   handling of references with the following arguments:

                   --resolve-none      Do not resolve references.
                   --resolve-external  Only resolve references that point to external files.
                                       This process is often referred to as "inlining".
                   --resolve-all       Resolve all references (default).
                                       Recursive pointers will stay references.

        inline     Convert a JSON or YAML input file to JSON or YAML output file and
                   resolve all external references. The output will be a single API Description file.
                   This is a shortcut for calling convert --resolve-external.

        help       Shows this usage information.

      Options:

        --read-json   force reading input as JSON. Auto-detect if not specified.
        --read-yaml   force reading input as YAML. Auto-detect if not specified.
        --write-json  force writing output as JSON. Auto-detect if not specified.
        --write-yaml  force writing output as YAML. Auto-detect if not specified.
        -s, --silent  silent mode. Will hide all success/information messages and only print errors.


### Reading API Description Files

Read OpenAPI Description from JSON file:

```php
use cebe\openapi\Reader;

// realpath is needed for resolving references with relative Paths or URLs
$openapi = Reader::readFromJsonFile(realpath('openapi.json'));
```

Read OpenAPI Description from YAML:

```php
use cebe\openapi\Reader;

// realpath is needed for resolving references with relative Paths or URLs
$openapi = Reader::readFromYamlFile(realpath('openapi.yaml'));
// you may also specify the URL to your API Description file
$openapi = Reader::readFromYamlFile('https://raw.githubusercontent.com/OAI/OpenAPI-Specification/3.0.2/examples/v3.0/petstore-expanded.yaml');
```

Access API Description data:

```php
echo $openapi->openapi; // openAPI version, e.g. 3.0.0
echo $openapi->info->title; // API title
foreach($openapi->paths as $path => $definition) {
    // iterate path definitions
}
```

Object properties are exactly like in the [OpenAPI Specification](https://github.com/OAI/OpenAPI-Specification/blob/3.0.2/versions/3.0.2.md#openapi-specification).
You may also access additional properties added by [specification extensions](https://github.com/OAI/OpenAPI-Specification/blob/3.0.2/versions/3.0.2.md#specificationExtensions).

Read a component schema and its properties for below spec:

```yaml
openapi: 3.0.0
info:
  title: Test API
  version: 1.0.0
paths:
  /foo:
    put:
      description: create foo
      responses:
        '200':
          description: request succeeded          
components:  
  schemas:
    Foo:
      description: This is an description
      type: object
      properties:
        message:
          type: string
        code:
          type: number
```

```php
# read a component schema
$foo = $openapi->components->schemas['Foo'];

# read a property of schema
$foo->properties['message']
```

### Writing API Description Files

```php
use cebe\openapi\spec\OpenApi;
use cebe\openapi\spec\PathItem;

// create base API Description
$openapi = new OpenApi([
    'openapi' => '3.0.2',
    'info' => [
        'title' => 'Test API',
        'version' => '1.0.0',
    ],
    'paths' => [],
]);
// manipulate description as needed
$openapi->paths['/test'] = new PathItem([
    'description' => 'something'
]);
// ...

$json = \cebe\openapi\Writer::writeToJson($openapi);
```

results in the following JSON data:

```json
{
    "openapi": "3.0.0",
    "info": {
        "title": "Test API",
        "version": "1.0.0"
    },
    "paths": {
        "/test": {
            "description": "something"
        }
    }
}
```

### Writing API Description Files using prepared Objects

Since version 1.2.0, the above example can also be written like this (passing objects instead of arrays):

```php
use cebe\openapi\spec\OpenApi;
use cebe\openapi\spec\PathItem;
use cebe\openapi\spec\Info;


// create base API Description
$openapi = new OpenApi([
    'openapi' => '3.0.2',
    'info' => new Info([
        'title' => 'Test API',
        'version' => '1.0.0',
    ]),
    'paths' => [
        '/test' => new PathItem([
            'description' => 'something'
        ]),
    ],
]);
$json = \cebe\openapi\Writer::writeToJson($openapi);
```

### Reading API Description Files and Resolving References

In the above we have passed the raw JSON or YAML data to the Reader. In order to be able to resolve
references to structures in external files, we must provide the full context.

```php
use cebe\openapi\Reader;
use cebe\openapi\spec\OpenAPI;
use cebe\openapi\ReferenceContext;

// there are two different modes for resolving references:
// ALL: resolve all references, which will result in a large description with a lot of repetition
// but no references (except if there are recursive references, these will stop at some level)
$mode = ReferenceContext::RESOLVE_MODE_ALL;
// INLINE: only references to external files are resolved, references to places in the current file
// are still Reference objects.
$mode = ReferenceContext::RESOLVE_MODE_INLINE;

// an absolute URL or file path is needed to allow resolving external references
$openapi = Reader::readFromJsonFile('https://www.example.com/api/openapi.json', OpenAPI::class, $mode);
$openapi = Reader::readFromYamlFile('https://www.example.com/api/openapi.yaml', OpenAPI::class, $mode);
```

If data has been loaded in a different way you can manually resolve references like this by giving a context:

```php
$openapi->resolveReferences(
    new \cebe\openapi\ReferenceContext($openapi, 'https://www.example.com/api/openapi.yaml')
);
```

### Validation

The library provides simple validation operations, that check basic OpenAPI spec requirements.
This is the same as "structural errors found while reading the API Description file" from the CLI tool.
This validation does not include checking against the OpenAPI v3.0 JSON schema, this is only implemented in the CLI.

```
// return `true` in case no errors have been found, `false` in case of errors.
$specValid = $openapi->validate();
// after validation getErrors() can be used to retrieve the list of errors found.
$errors = $openapi->getErrors();
```

> **Note:** Validation is done on a very basic level and is not complete. So a failing validation will show some errors,
> but the list of errors given may not be complete. Also a passing validation does not necessarily indicate a completely
> valid spec.


## Completeness

This library is currently work in progress, the following list tracks completeness:

- [x] read OpenAPI 3.0 JSON
- [x] read OpenAPI 3.0 YAML
- [ ] OpenAPI 3.0 Schema
  - [x] OpenAPI Object
  - [x] Info Object
  - [x] Contact Object
  - [x] License Object
  - [x] Server Object
  - [x] Server Variable Object
  - [x] Components Object
  - [x] Paths Object
  - [x] Path Item Object
  - [x] Operation Object
  - [x] External Documentation Object
  - [x] Parameter Object
  - [x] Request Body Object
  - [x] Media Type Object
  - [x] Encoding Object
  - [x] Responses Object
  - [x] Response Object
  - [x] Callback Object
  - [x] Example Object
  - [x] Link Object
    - [ ] [Runtime Expressions](https://github.com/OAI/OpenAPI-Specification/blob/3.0.2/versions/3.0.2.md#runtime-expressions)
  - [x] Header Object
  - [x] Tag Object
  - [x] Reference Object
  - [x] Schema Object
    - [x] load/read
    - [ ] validation
  - [x] Discriminator Object
  - [x] XML Object
  - [x] Security Scheme Object
  - [x] OAuth Flows Object
  - [x] OAuth Flow Object
  - [x] Security Requirement Object

# Development

You may use the docker environment for local development:

    docker-compose build
    make IN_DOCKER=1 install
    make IN_DOCKER=1 test
    ...


# Support

**Need help with your API project?**

Professional support, consulting as well as software development services are available:

https://www.cebe.cc/en/contact

Development of this library is sponsored by [cebe.:cloud: "Your Professional Deployment Platform"](https://cebe.cloud).
 readmeEtag: '"d7cd07151e5836fb93b11e7879828520ef036777"' readmeLastModified: Fri, 07 Feb 2025 11:28:04 GMT repositoryId: 154310978 description: >- Read and write OpenAPI yaml/json files and make the content accessible in PHP objects. created: '2018-10-23T10:46:11Z' updated: '2026-01-19T07:05:22Z' language: PHP archived: false stars: 498 watchers: 17 forks: 98 owner: cebe logo: https://avatars.githubusercontent.com/u/189796?v=4 license: MIT repoEtag: '"bc3824ecc2198f528aa28459b8630f661c76b4284027b95346306c2732106176"' repoLastModified: Mon, 19 Jan 2026 07:05:22 GMT foundInMaster: true v3_1_link: https://github.com/cebe/php-openapi/pull/128 id: 3a26d5049f03c42cb9771c179db0f35f - source: https://openapi.tools/ name: Object Oriented OpenAPI Specification homepage: https://github.com/goldspecdigital/oooas language: PHP source_description: >- An object oriented approach to generating OpenAPI Descriptions, implemented in PHP category: Parsers repository: https://github.com/goldspecdigital/oooas v3: true repositoryMetadata: base64Readme: >- <p align="center">
    <a href="https://github.com/goldspecdigital/oooas"><img 
        alt="Object Oriented OpenAPI Specification"
        src="https://svgshare.com/i/D70.svg" width="400px"
    ></a>
</p>

<p align="center">
    <a href="https://github.com/goldspecdigital/oooas"><img 
        alt="GitHub stars" 
        src="https://img.shields.io/github/stars/goldspecdigital/oooas.svg?style=social"
    ></a>
</p>

<p align="center">
    <a href="https://github.com/goldspecdigital/oooas/tags"><img 
        alt="GitHub tag (latest SemVer)" 
        src="https://img.shields.io/github/tag/goldspecdigital/oooas.svg"
    ></a>
    <a href="https://github.com/goldspecdigital/oooas/actions"><img 
        alt="Build status"
        src="https://github.com/goldspecdigital/oooas/workflows/Tests/badge.svg" 
    ></a>
    <a href="https://packagist.org/packages/goldspecdigital/oooas"><img 
        alt="Packagist" 
        src="https://img.shields.io/packagist/dt/goldspecdigital/oooas.svg"
    ></a>
    <img 
        alt="PHP from Packagist" 
        src="https://img.shields.io/packagist/php-v/goldspecdigital/oooas.svg"
    >
    <img 
        alt="Dependency count"
        src="https://img.shields.io/badge/dependencies-0-brightgreen.svg" 
    >
    <img 
        alt="Packagist" 
        src="https://img.shields.io/packagist/l/goldspecdigital/oooas.svg"
    >
</p>

## Introduction

An object oriented approach to generating OpenAPI specs, implemented in PHP. 

You can build up your API spec using immutable PHP classes, and then export the 
spec to JSON (or YAML with the help of another package).

This package is **dependency free** and makes heavy use of **PHP 7 features**, 
mainly being **type hints** and enabling **strict types**. This should make your 
life a lot easier when working with a good IDE that can use this information.

## Installing

You can install the package via composer:
```bash
composer require goldspecdigital/oooas
```

## Example

See the code sample below for the most basic usage:

```php
use GoldSpecDigital\ObjectOrientedOAS\Objects\{
    Info, MediaType, Operation, PathItem, Response, Schema, Tag
};
use GoldSpecDigital\ObjectOrientedOAS\OpenApi;

// Create a tag for all the user endpoints.
$usersTag = Tag::create()
    ->name('Users')
    ->description('All user related endpoints');

// Create the info section.
$info = Info::create()
    ->title('API Specification')
    ->version('v1')
    ->description('For using the Example App API');
    
// Create the user schema.
$userSchema = Schema::object()
    ->properties(
        Schema::string('id')->format(Schema::FORMAT_UUID),
        Schema::string('name'),
        Schema::integer('age')->example(23),
        Schema::string('created_at')->format(Schema::FORMAT_DATE_TIME)
    );
    
// Create the user response.
$userResponse = Response::create()
    ->statusCode(200)
    ->description('OK')
    ->content(
        MediaType::json()->schema($userSchema)
    );
    
// Create the operation for the route (i.e. GET, POST, etc.).
$showUser = Operation::get()
    ->responses($userResponse)
    ->tags($usersTag)
    ->summary('Get an individual user')
    ->operationId('users.show');
    
// Define the /users path along with the supported operations.
$usersPath = PathItem::create()
    ->route('/users')
    ->operations($showUser);
    
// Create the main OpenAPI object composed off everything created above.
$openApi = OpenApi::create()
    ->openapi(OpenApi::OPENAPI_3_0_2)
    ->info($info)
    ->paths($usersPath)
    ->tags($usersTag);
    
header('Content-Type: application/json');
echo $openApi->toJson();
```

### YAML output

Using the same code above will output the following YAML:

> In this example, the YAML may seem simpler to look at, however once the spec
starts to increase in size - the ability to reuse objects and split them into
separate files easily will be a massive help.

```yaml
openapi: 3.0.2
info:
  title: API Specification
  description: For using the Example App API
  version: v1
paths:
  "/users":
    get:
      tags:
      - Users
      summary: Get an individual user
      operationId: users.show
      responses:
        '200':
          description: OK
          content:
            application/json:
              schema:
                type: object
                properties:
                  id:
                    format: uuid
                    type: string
                  name:
                    type: string
                  age:
                    type: integer
                    example: 23
                  created_at:
                    format: date-time
                    type: string
tags:
- name: Users
  description: All user related endpoints
```

### Outputting as JSON or YAML

Built in output to YAML has been omitted on purpose to keep this package
dependency free. However, you can easily convert the array to a YAML string 
using several open source packages. See below for an example of  outputting to 
both JSON and YAML:

```php
use GoldSpecDigital\ObjectOrientedOAS\OpenApi;
use Symfony\Component\Yaml\Yaml;

$openApi = OpenApi::create();

$json = $openApi->toJson();
$array = $openApi->toArray();
$yaml = Yaml::dump($array);
```

## Guidance

If you want to learn more about the OpenAPI schema, then have a look at the 
official [OpenAPI Specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md).

Alternatively, if you would like a quick reference, then check out the 
[OpenAPI Map](https://openapi-map.apihandyman.io/?version=3.0) project created 
by [Arnaud Lauret](http://apihandyman.io/).

You can use this interactive tool to figure out what objects go where and how
they relate to one another.

## Usage

### Setting and unsetting properties

Each object has setter methods for it's supported properties. Most of these 
methods allow `null` values which will need to be explicitly passed (see the 
next example for how to unset using variadic setter methods). This will have the 
effect of unsetting the property:

```php
$info = Info::create()
    ->title('Example API');

$openApi = OpenApi::create()
    ->info($info);
echo $openApi->toJson(); // '{"info": {"title": "Example API"}}'

$openApi = $openApi->info(null);
echo $openApi->toJson(); // '{}'
```

For variadic setter methods, if you call the method and don't supply any
parameters, then this will have the effect of unsetting the property:

```php
$path = PathItem::create()
    ->route('/users');

$openApi = OpenApi::create()
    ->paths($path);
echo $openApi->toJson(); // '{"paths": {"/users": []}}'

$openApi = $openApi->paths();
echo $openApi->toJson(); // '{}'
```

### Retrieving properties

You can easily retrieve a property using a magic getter. These have been
implemented for all properties for every object. DocBlocks have been provided
to give better auto-completion in IDEs:

```php
$info = Info::create()->title('Example API');

echo $info->title; // 'Example API'
```

### Object ID

Every object has an optional `$objectId` property which is a `string` and can 
either be set in the class constructor or the preferred `create()` method. This 
property is used when a parent object needs to use a name for the children.

An example of this in use is when a schema object is composed of other schema
properties:

```php
$schema = Schema::create()
    ->type(Schema::TYPE_OBJECT)
    ->properties(
        Schema::create('username')->type(Schema::TYPE_STRING),
        Schema::create('age')->type(Schema::TYPE_INTEGER)
    );
    
echo $schema->toJson();
/* 
{
  "type": "object",
  "properties": {
    "username": {
      "type": "string"
    },
    "age": {
      "type": "integer"
    }
  }
} 
*/
``` 

If an object contains any helper creation methods, then these methods also allow
you to specify the `$objectId` property as a parameter. The code sample below is
functionally identical to the one above:

```php
$schema = Schema::object()
    ->properties(
        Schema::string('username'),
        Schema::integer('age')
    );
    
echo $schema->toJson();
/* 
{
  "type": "object",
  "properties": {
    "username": {
      "type": "string"
    },
    "age": {
      "type": "integer"
    }
  }
} 
*/
``` 

### $ref

The use of `$ref` has been applied to every single object to use as you wish.
You may substitute any object for a `$ref` by invoking the `ref()` static method 
on the object class: 

```php
$schema = AllOf::create()
    ->schemas(
        Schema::ref('#/components/schemas/ExampleSchema')
    );
    
echo $schema->toJson();
/*
{
  "allOf": [
    ["$ref": "#/components/schemas/ExampleSchema"]
  ]
}
*/
```

### Specification extensions

You can add [specification extensions](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#specificationExtensions)
to all objects:

```php
$schema = Schema::create()
    ->x('foo', 'bar')
    ->x('items', Schema::array()->items(Schema::string()));
    
echo $schema->toJson();
/*
{
  "x-foo": "bar",
  "x-items": {
    "type": "array",
    "items": {
      "type": "string"
    }
  }
}
*/

echo $schema->{'x-foo'}; // 'bar'
```

You can also unset specification extensions by invoking the `x()` method and
only providing the key:

```php
$schema = Schema::create()
    ->x('foo', 'bar')
    ->x('items', Schema::array()->items(Schema::string()));

$schema = $schema->x('foo');
    
echo $schema->toJson();
/*
{
  "x-items": {
    "type": "array",
    "items": {
      "type": "string"
    }
  }
}
*/
```

To retrieve an array of all the specification extensions you can call the `$x`
property:

```php
$schema = Schema::create()
    ->x('foo', 'bar')
    ->x('items', Schema::array()->items(Schema::string()));

echo json_encode($schema->x);
/*
{
  "x-foo": "bar",
  "x-items": {
    "type": "array",
    "items": {
      "type": "string"
    }
  }
}
*/
```

### Validation

In order to perform schema validation you must first install the
`justinrainbow/json-schema` package:

```bash
composer require justinrainbow/json-schema:^5.2
```

Once installed, you may now make use of the `validate()` method on the `OpenApi`
object:

```php
$openApi = OpenApi::create();
$openApi->validate();
```

_If you haven't installed the `justinrainbow/json-schema` package and attempt to 
use the `validate()` method, then a `RuntimeException` will be thrown._

If validation fails for the schema, then a 
`GoldSpecDigital\ObjectOrientedOAS\Exceptions\ValidationException` will be 
thrown. You can use the `getErrors()` method on this exception to retrieve all
of the validation errors.

## Running the tests

To run the test suite you can use the following commands:

```bash
# To run both style and unit tests.
composer test

# To run only style tests.
composer test:style

# To run only unit tests.
composer test:unit
```

If you receive any errors from the style tests, you can automatically fix most,
if not all of the issues with the following command:

```bash
composer fix:style
```

## Contributing

Please read [CONTRIBUTING.md](CONTRIBUTING.md) for details on our code of 
conduct, and the process for submitting pull requests to us.

## Versioning

We use [SemVer](http://semver.org/) for versioning. For the versions available, 
see the [tags on this repository](https://github.com/goldspecdigital/oooas/tags). 

## Authors

* [GoldSpec Digital](https://github.com/goldspecdigital)

See also the list of [contributors](https://github.com/goldspecdigital/oooas/contributors) 
who participated in this project.

## License

This project is licensed under the MIT License - see the [LICENSE.md](LICENSE.md) 
file for details.
 readmeEtag: '"b4d5d56bfb406f0b47465e5ba6a2ab4341aad3e9"' readmeLastModified: Wed, 20 Jul 2022 09:57:04 GMT repositoryId: 150484399 description: >- An object oriented approach to generating OpenAPI specs, implemented in PHP. created: '2018-09-26T20:11:56Z' updated: '2025-11-23T05:30:36Z' language: PHP archived: false stars: 232 watchers: 7 forks: 25 owner: goldspecdigital logo: https://avatars.githubusercontent.com/u/31162012?v=4 license: MIT repoEtag: '"a1568657c4dbaf21a26310f3cf78af18829e271ef155e00dcbc00662aff6c003"' repoLastModified: Sun, 23 Nov 2025 05:30:36 GMT foundInMaster: true id: b572581c8e045165d5121ee9c5bc361e - source: https://openapi.tools/ name: OpenAPI3-Rust category: Parsers repository: https://github.com/adwhit/openapi3-rust language: Rust source_description: Rust serialization library for OpenAPI v3 v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJdjMKClNwZWM6IGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvT3BlbkFQSS5uZXh0L3ZlcnNpb25zLzMuMC5tZAo= readmeEtag: '"ce669f4ef3578a7762c5d175c06a9368bee7d84c"' readmeLastModified: Sun, 09 Dec 2018 15:37:14 GMT repositoryId: 97812110 description: Rust serialization library for OpenAPIv3 created: '2017-07-20T08:43:25Z' updated: '2024-04-22T10:18:12Z' language: Rust archived: true stars: 13 watchers: 1 forks: 4 owner: adwhit logo: https://avatars.githubusercontent.com/u/3189554?v=4 license: MIT repoEtag: '"dfdbb54d520ed1c9508c22b6fb0154f9ba5278ab6165a27890b49205733ca2bd"' repoLastModified: Mon, 22 Apr 2024 10:18:12 GMT foundInMaster: true id: 000be8d50bcd3022727597e652c33e7f - source: https://openapi.tools/ name: psx-api category: Parsers link: https://phpsx.org repository: https://github.com/apioo/psx-api language: PHP source_description: Parse and generate API specification formats v2: true v3: true repositoryMetadata: base64Readme: >- CiMgQVBJCgpUaGlzIGxpYnJhcnkgcHJvdmlkZXMgYW4gYXR0cmlidXRlIHBhcnNlciB0byBkeW5hbWljYWxseSBnZW5lcmF0ZSBhIFtUeXBlQVBJXShodHRwczovL3R5cGVhcGkub3JnLykgc3BlY2lmaWNhdGlvbgpmcm9tIGFueSBjb250cm9sbGVyIChjb2RlLWZpcnN0KS4gQmFzZWQgb24gdGhlIHNwZWNpZmljYXRpb24gaXQgaXMgdGhlbiBwb3NzaWJsZSB0byBnZW5lcmF0ZSBjbGllbnQgU0RLcyBvcgphbiBPcGVuQVBJIHNwZWNpZmljYXRpb24uCgpXZSBwcm92aWRlIGFsc28gYSBob3N0ZWQgdmVyc2lvbiBvZiB0aGlzIFtjb2RlIGdlbmVyYXRvcl0oaHR0cHM6Ly90eXBlYXBpLm9yZy9nZW5lcmF0b3IpLgpGb3IgbW9yZSBpbnRlZ3JhdGlvbiBvcHRpb25zIHlvdSBjYW4gYWxzbyB0YWtlIGEgbG9vayBhdCB0aGUgW1NES2dlbl0oaHR0cHM6Ly9zZGtnZW4uYXBwLykgcHJvamVjdAp3aGljaCBwcm92aWRlcyBhIENMSSBiaW5hcnkgb3IgR2l0SHViIGFjdGlvbiB0byBpbnRlZ3JhdGUgdGhlIGNvZGUgZ2VuZXJhdG9yLgoKIyMgVXNhZ2UKClRoZSByb290IG1vZGVsIG9iamVjdCBpcyBjYWxsZWQgYSBgU3BlY2lmaWNhdGlvbmAgd2hpY2ggY29udGFpbnMgYE9wZXJhdGlvbnNgIGFuZCBgRGVmaW5pdGlvbnNgLiBFYWNoIG9wZXJhdGlvbgptYXBzIHRvIGEgc3BlY2lmaWMgUkVTVCBBUEkgZW5kcG9pbnQgYW5kIHRoZSBkZWZpbml0aW9ucyByZXByZXNlbnQgdGhlIHNjaGVtYXMgdG8gZGVzY3JpYmUgdGhlIEpTT04gcmVxdWVzdCBvciByZXNwb25zZQpwYXlsb2FkLgoKIyMjIEZyYW1ld29yawoKWW91IGNhbiB1c2UgUEhQIGF0dHJpYnV0ZXMgdG8gZGVzY3JpYmUgdGhlIHN0cnVjdHVyZSBvZiB5b3VyIGVuZHBvaW50cy4gWW91IGNhbiB0aGVuIHVzZSB0aGUgYXR0cmlidXRlIHBhcnNlciAoYFBTWFxBcGlcUGFyc2VyXEF0dHJpYnV0ZWApCnRvIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGUgYSBzcGVjaWZpY2F0aW9uIGZvciB5b3VyIGNvbnRyb2xsZXIuIEEgY29udHJvbGxlciBjbGFzcyBjb3VsZCB0aGVuIGxvb2sgbGlrZToKCmBgYHBocAo8P3BocAoKY2xhc3MgTXlDb250cm9sbGVyCnsKICAgICNbR2V0XQogICAgI1tQYXRoKCcvbXkvZW5kcG9pbnQvOmlkJyldCiAgICBwdWJsaWMgZnVuY3Rpb24gZ2V0TW9kZWwoI1tQYXJhbV0gaW50ICRpZCwgI1tRdWVyeV0gaW50ICR5ZWFyKTogXE15XFJlc3BvbnNlXE1vZGVsCiAgICB7CiAgICAgICAgLy8gQFRPRE8gaW1wbGVtZW50CiAgICB9CiAgICAKICAgICNbUG9zdF0KICAgICNbUGF0aCgnL215L2VuZHBvaW50JyldCiAgICBwdWJsaWMgZnVuY3Rpb24gaW5zZXJ0TW9kZWwoI1tCb2R5XSBcTXlcUmVxdWVzdFxNb2RlbCAkbW9kZWwpOiBcTXlcUmVzcG9uc2VcTW9kZWwKICAgIHsKICAgICAgICAvLyBAVE9ETyBpbXBsZW1lbnQKICAgIH0KfQoKYGBgCgpUaGlzIHdvdWxkIGJlIGVub3VnaCBmb3IgdGhlIEFQSSBjb21wb25lbnQgdG8gZ2VuZXJhdGUgZWl0aGVyIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBvciBhIGNsaWVudCBTREsuCk5vdGUgdGhpcyBsaWJyYXJ5IG9ubHkgbmVlZHMgdGhlIG1ldGEgaW5mb3JtYXRpb24sIGlmIHlvdSBjYW4gZ2V0IHRob3NlIG1ldGEgaW5mb3JtYXRpb24gYXQgeW91ciBmcmFtZXdvcmsgaW4gYW5vdGhlcgp3YXkgeW91IGNhbiBhbHNvIGltcGxlbWVudCBhIGN1c3RvbSBgUGFyc2VySW50ZXJmYWNlYC4KCiMjIyBTdGFuZGFsb25lCgpCZXNpZGUgdGhlIGZyYW1ld29yayBpbnRlZ3JhdGlvbiB5b3UgY2FuIHVzZSB0aGlzIGNvbXBvbmVudCBhbHNvIHRvIHNpbXBseSBwYXJzZSBleGlzdGluZyBUeXBlQVBJIHNwZWNpZmljYXRpb24gYW5kCmdlbmVyYXRlIHNwZWNpZmljIG91dHB1dC4gVGhlIGZvbGxvd2luZyBpcyBhIHNpbXBsZSBleGFtcGxlIGhvdyB0byB1c2UgdGhlIFBIUCBBUEkgYW5kIGhvdyB0byBnZW5lcmF0ZSBjb2RlLgoKYGBgcGhwCjw/cGhwCgovLyB1c2UgdGhlIEFQSSBtYW5hZ2VyIHRvIG9idGFpbiBhIHNwZWNpZmljYXRpb24gZnJvbSBkaWZmZXJlbnQgc291cmNlcwokbWFuYWdlciA9IG5ldyBcUFNYXEFwaVxBcGlNYW5hZ2VyKG5ldyBcUFNYXFNjaGVtYVxTY2hlbWFNYW5hZ2VyKCkpOwoKLy8gcmVhZHMgdGhlIFR5cGVBUEkgc3BlY2lmaWNhdGlvbiBhbmQgZ2VuZXJhdGVzIGEgc3BlY2lmaWNhdGlvbgokc3BlY2lmaWNhdGlvbiA9ICRtYW5hZ2VyLT5nZXRBcGkoJy4vdHlwZWFwaS5qc29uJyk7CgovLyBjb250YWlucyBhbGwgc2NoZW1hIHR5cGUgZGVmaW5pdGlvbnMKJGRlZmluaXRpb25zID0gJHNwZWNpZmljYXRpb24tPmdldERlZmluaXRpb25zKCk7CgovLyByZXR1cm5zIHRoZSByZXNvdXJjZSBmb28gZnJvbSB0aGUgc3BlY2lmaWNhdGlvbgokb3BlcmF0aW9uID0gJHNwZWNpZmljYXRpb24tPmdldE9wZXJhdGlvbnMoKS0+Z2V0KCdteS5vcGVyYXRpb24nKTsKCi8vIHJldHVybnMgYWxsIGF2YWlsYWJsZSBhcmd1bWVudHMKJG9wZXJhdGlvbi0+Z2V0QXJndW1lbnRzKCk7CgovLyByZXR1cm5zIHRoZSByZXR1cm4gdHlwZQokb3BlcmF0aW9uLT5nZXRSZXR1cm4oKTsKCi8vIHJldHVybnMgYWxsIGV4Y2VwdGlvbnMgd2hpY2ggYXJlIGRlc2NyaWJlZAokb3BlcmF0aW9uLT5nZXRUaHJvd3MoKTsKCi8vIHJldHVybnMgdGhlIGFzc2lnbmVkIEhUVFAgbWV0aG9kCiRvcGVyYXRpb24tPmdldE1ldGhvZCgpOwoKLy8gcmV0dXJucyB0aGUgYXNzaWduZWQgSFRUUCBwYXRoCiRvcGVyYXRpb24tPmdldFBhdGgoKTsKCi8vIGNyZWF0ZXMgYSBQSFAgY2xpZW50IHdoaWNoIGNvbnN1bWVzIHRoZSBkZWZpbmVkIG9wZXJhdGlvbnMKJHJlZ2lzdHJ5ID0gXFBTWFxBcGlcR2VuZXJhdG9yRmFjdG9yeTo6ZnJvbUxvY2FsKCktPmZhY3RvcnkoKTsKJGdlbmVyYXRvciA9ICRyZWdpc3RyeS0+Z2V0R2VuZXJhdG9yKFxQU1hcQXBpXFJlcG9zaXRvcnlcTG9jYWxSZXBvc2l0b3J5OjpDTElFTlRfUEhQKQoKJHNvdXJjZSA9ICRnZW5lcmF0b3ItPmdlbmVyYXRlKCRyZXNvdXJjZSk7CgpgYGAK readmeEtag: '"772703306d3dceaf7bbc89fff64bcff0f5edad2f"' readmeLastModified: Fri, 11 Oct 2024 18:49:31 GMT repositoryId: 55149439 description: TypeAPI parser and SDK code generator created: '2016-03-31T12:39:05Z' updated: '2025-12-13T17:30:03Z' language: PHP archived: false stars: 36 watchers: 1 forks: 9 owner: apioo logo: https://avatars.githubusercontent.com/u/18172950?v=4 license: Apache-2.0 repoEtag: '"e91c25f35b76b7078696572153f2e8255fcce63b27128bd9035ba0ec997fcde9"' repoLastModified: Sat, 13 Dec 2025 17:30:03 GMT foundInMaster: true id: ee32e280cf10ca5103c9b9e144021b91 - source: https://openapi.tools/ name: Microsoft/OpenAPI.NET category: Parsers repository: https://github.com/microsoft/openapi.net language: .NET source_description: >- C# based parser with OpenAPI Description validation and migration support from V2 v2: true v3: true repositoryMetadata: base64Readme: >- <!-- using the raw image URL so it displays correctly on nuget.org -->
![Category overview screenshot](https://raw.githubusercontent.com/microsoft/OpenAPI.NET/main/docs/images/oainet.png "Microsoft + OpenAPI = Love")

# OpenAPI.NET

|Package|Nuget|
|--|--|
|Models and Writers|[![nuget](https://img.shields.io/nuget/v/Microsoft.OpenApi.svg)](https://www.nuget.org/packages/Microsoft.OpenApi/) |
|YamlReader | [![nuget](https://img.shields.io/nuget/v/Microsoft.OpenApi.YamlReader.svg)](https://www.nuget.org/packages/Microsoft.OpenApi.YamlReader/) |
|Hidi|[![nuget](https://img.shields.io/nuget/v/Microsoft.OpenApi.Hidi.svg)](https://www.nuget.org/packages/Microsoft.OpenApi.Hidi/)

The **OpenAPI.NET** SDK contains a useful object model for OpenAPI documents in .NET along with common serializers to extract raw OpenAPI JSON and YAML documents from the model.

**See more information on the OpenAPI specification and its history here: <a href="https://www.openapis.org">OpenAPI Initiative</a>**

Project Objectives:

- Provide a single shared object model in .NET for OpenAPI descriptions.
- Include the most primitive Reader for ingesting OpenAPI JSON and YAML documents in both V2 and V3 formats.
- Provide OpenAPI description writers for both V2 and V3 specification formats.
- Enable developers to create Readers that translate different data formats into OpenAPI descriptions.

# Installation

- Install core Nuget package [**Microsoft.OpenApi**](https://www.nuget.org/packages/Microsoft.OpenApi)
- Install Yaml Reader Nuget package [**Microsoft.OpenApi.YamlReader**](https://www.nuget.org/packages/Microsoft.OpenApi.YamlReader)

> Note: we just released a new major version of the library, which brings support for OpenAPI 3.2!
> You can read more about the changes of this upcoming version [in the upgrade guide](./docs/upgrade-guide-3.md).

# Processors
The OpenAPI.NET project holds the base object model for representing OpenAPI documents as .NET objects. Some developers have found the need to write processors that convert other data formats into this OpenAPI.NET object model. We'd like to curate that list of processors in this section of the readme. 

The base JSON and YAML processors are built into this project. Below is the list of the other supported processor projects.

- [**C# Comment / Annotation Processor**](https://github.com/Microsoft/OpenAPI.NET.CSharpAnnotations) : Converts standard .NET annotations ( /// comments ) emitted from your build (MSBuild.exe) into OpenAPI.NET document object. 

- [**OData CSDL Processor**](https://github.com/Microsoft/OpenAPI.NET.OData) : Converts the XML representation of the Entity Data Model (EDM) describing an OData Service into OpenAPI.NET document object. 

# Example Usage

Creating an OpenAPI Document

```C#
var document = new OpenApiDocument
{
    Info = new OpenApiInfo
    {
        Version = "1.0.0",
        Title = "Swagger Petstore (Simple)",
    },
    Servers = new List<OpenApiServer>
    {
        new OpenApiServer { Url = "http://petstore.swagger.io/api" }
    },
    Paths = new OpenApiPaths
    {
        ["/pets"] = new OpenApiPathItem
        {
            Operations = new()
            {
                [HttpMethod.Get] = new OpenApiOperation
                {
                    Description = "Returns all pets from the system that the user has access to",
                    Responses = new OpenApiResponses
                    {
                        ["200"] = new OpenApiResponse
                        {
                            Description = "OK"
                        }
                    }
                }
            }
        }
    }
};
```

Reading and writing an OpenAPI description

```C#
var (openApiDocument, _) = await OpenApiDocument.LoadAsync("https://raw.githubusercontent.com/OAI/OpenAPI-Specification/refs/heads/main/_archive_/schemas/v3.0/pass/petstore.yaml");

// Write V2 as JSON
var outputString = await openApiDocument.SerializeAsJsonAsync(OpenApiSpecVersion.OpenApi2_0);
```

# Validating/Testing OpenAPI descriptions
In order to test the validity of an OpenApi document, we avail the following tools:
- [Microsoft.OpenApi.Hidi](https://www.nuget.org/packages/Microsoft.OpenApi.Hidi)

    A commandline tool for validating and transforming OpenAPI descriptions. [Installation guidelines and documentation](https://github.com/microsoft/OpenAPI.NET/blob/main/src/Microsoft.OpenApi.Hidi/readme.md)

- Microsoft.OpenApi.Workbench

    A workbench tool consisting of a GUI where you can test and convert OpenAPI descriptions in both JSON and YAML from v2-->v3 and vice versa.

    #### Installation guidelines:
    1. Clone the repo locally by running this command:
        `git clone https://github.com/microsoft/OpenAPI.NET.git`
    2. Open the solution file `(.sln)` in the root of the project with Visual Studio
    3. Navigate to the `src/Microsoft.OpenApi.Workbench` directory and set it as the startup project
    4. Run the project and you'll see a GUI pop up resembling the one below:
    
    
    ![workbench preview](https://raw.githubusercontent.com/microsoft/OpenAPI.NET/main/docs/images/workbench.png "a screenshot of the workbench application")
    
    5. Copy and paste your OpenAPI descriptions in the **Input Content** window or paste the path to the descriptions file in the **Input File** textbox and click on `Convert` to render the results.

# Contributing

This project welcomes contributions and suggestions.  Most contributions require you to agree to a
Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us
the rights to use your contribution. For details, visit https://cla.microsoft.com.

When you submit a pull request, a CLA-bot will automatically determine whether you need to provide
a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the instructions
provided by the bot. You will only need to do this once across all repos using our CLA.

This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/).
For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or
contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments.

To provide feedback and ask questions you can use Stack Overflow with the [OpenAPI.NET](https://stackoverflow.com/questions/tagged/openapi.net) tag.
 readmeEtag: '"6a7f787f8e6cf6076b83186b54e46c10ddfc4c94"' readmeLastModified: Tue, 11 Nov 2025 21:58:29 GMT repositoryId: 97175798 description: >- The OpenAPI.NET SDK contains a useful object model for OpenAPI documents in .NET along with common serializers to extract raw OpenAPI JSON and YAML documents from the model. created: '2017-07-14T00:24:14Z' updated: '2026-02-05T12:17:03Z' language: C# archived: false stars: 1575 watchers: 62 forks: 282 owner: microsoft logo: https://avatars.githubusercontent.com/u/6154722?v=4 license: MIT repoEtag: '"02e87e29122fea6641bc92d47b69eb0c7a7bdcb9421082dddf2cc5f1bb1d11d0"' repoLastModified: Thu, 05 Feb 2026 12:17:03 GMT foundInMaster: true id: 1066748bb5b1f114a70856a29fee4aa7 - source: - https://openapi.tools/ - openapi3 tags name: oas_parser homepage: https://github.com/Nexmo/oas_parser language: Ruby source_description: A Ruby parser for OpenAPI 3.0+ descriptions. category: Parsers repository: https://github.com/nexmo/oas_parser v2: false v3: true repositoryMetadata: base64Readme: >- IyDimqDvuI8gVGhpcyByZXBvc2l0b3J5IGlzIG5vIGxvbmdlciBtYWludGFpbmVkLgoKLS0tCgojIE9wZW4gQVBJIERlZmluaXRpb24gUGFyc2VyCgo8aW1nIHNyYz0iaHR0cHM6Ly9kZXZlbG9wZXIubmV4bW8uY29tL2Fzc2V0cy9pbWFnZXMvVm9uYWdlX05leG1vLnN2ZyIgaGVpZ2h0PSI0OHB4IiBhbHQ9Ik5leG1vIGlzIG5vdyBrbm93biBhcyBWb25hZ2UiIC8+CgpBIFJ1YnkgcGFyc2VyIGZvciBPcGVuIEFQSSBTcGVjIDMuMCsgZGVmaW5pdGlvbnMuCgojIyMgSW5zdGFsbAoKSW5zdGFsbCB0aGUgZ2VtOgoKYGBgCiQgZ2VtIGluc3RhbGwgb2FzX3BhcnNlcgpgYGAKCk9yIGFkZCBpdCB0byB5b3VyIEdlbWZpbGU6CgpgYGBydWJ5CmdlbSAnb2FzX3BhcnNlcicKYGBgCgojIyMgVXNhZ2UKCkhlcmUgaXMgYSBiYXNpYyBleGFtcGxlIG9mIGhvdyB5b3UgY2FuIHRyYXZlcnNlIHRocm91Z2ggYW4gT3BlbiBBUEkgU3BlYyAzIERlZmluaXRpb246CgpgYGBydWJ5CnJlcXVpcmUgJ29hc19wYXJzZXInCgpkZWZpbml0aW9uID0gT2FzUGFyc2VyOjpEZWZpbml0aW9uLnJlc29sdmUoJ3BldHN0b3JlLnltbCcpCiMgPT4gIzxPYXNQYXJzZXI6OkRlZmluaXRpb24+CgojIEdldCBhIHNwZWNpZmljIHBhdGgKcGF0aCA9IGRlZmluaXRpb24ucGF0aF9ieV9wYXRoKCcvcGV0cycpCiMgPT4gIzxPYXNQYXJzZXI6OlBhdGg+CgojIEdldCBhbGwgcGF0aHMuCmRlZmluaXRpb24ucGF0aHMKIyA9PiBbIzxPYXNQYXJzZXI6OlBhdGg+LCAuLi5dCgojIEdldCBhIHNwZWNpZmljIGVuZHBvaW50IGJ5IG1ldGhvZAplbmRwb2ludCA9IHBhdGguZW5kcG9pbnRfYnlfbWV0aG9kKCdnZXQnKQojID0+ICM8T2FzUGFyc2VyOjpFbmRwb2ludD4KCiMgR2V0IGFsbCBlbmRwb2ludHMKcGF0aC5lbmRwb2ludHMKIyA9PiBbIzxPYXNQYXJzZXI6OkVuZHBvaW50PiwgLi4uXQoKIyBHZXQgZW5kcG9pbnQgZGVzY3JpcHRpb24KZW5kcG9pbnQuZGVzY3JpcHRpb24KIyA9PiAiUmV0dXJucyBhbGwgcGV0cyBmcm9tIHRoZSBzeXN0ZW0gdGhhdCB0aGUgdXNlciBoYXMgYWNjZXNzIHRvIgpgYGAKCkNoZWNrb3V0IHRoZSB0ZXN0cyBhbmQgYGxpYmAgZGlyZWN0b3J5IGZvciBtb3JlIGNsYXNzZXMgYW5kIG1ldGhvZHMuCgojIyMgRGV2ZWxvcG1lbnQKClJ1biB0ZXN0czoKCmBgYAokIHJzcGVjCmBgYAoKIyMjIFB1Ymxpc2hpbmcKCkNsb25lIHRoZSByZXBvIGFuZCBuYXZpZ2F0ZSB0byBpdHMgZGlyZWN0b3J5OgoKYGBgCiQgY2Qgb2FzLXBhcnNlcgpgYGAKCkJ1bXAgdGhlIGxhdGVzdCB2ZXJzaW9uIGluIGBvYXNfcGFyc2VyL2xpYi9vYXNfcGFyc2VyL3ZlcnNpb24ucmJgOgoKYGBgCi8vb2xkCm1vZHVsZSBPYXNQYXJzZXIKICBWRVJTSU9OID0gJzEuMC4wJy5mcmVlemUKZW5kCgovL25ldwptb2R1bGUgT2FzUGFyc2VyCiAgVkVSU0lPTiA9ICcxLjEuMCcuZnJlZXplCmVuZApgYGAKCkJ1aWxkIHRoZSBnZW06CgpgYGAKJCBnZW0gYnVpbGQgb2FzX3BhcnNlci5nZW1zcGVjCmBgYAoKX1RoaXMgd2lsbCBjcmVhdGUgYSBgb2FzX3BhcnNlci0xLjEuMC5nZW1gIGZpbGUuXwoKUHVzaCB0aGUgZ2VtIHRvIHJ1YnlnZW1zLm9yZzoKCmBgYAokIGdlbSBwdXNoIG9hc19wYXJzZXItMS4xLjAuZ2VtCmBgYAoKVmVyaWZ5IHRoZSBjaGFuZ2Ugd2FzIG1hZGUgYnkgY2hlY2tpbmcgZm9yIHRoZSBbbmV3IHZlcnNpb24gb24gcnVieWdlbXMub3JnXShodHRwczovL3J1YnlnZW1zLm9yZy9nZW1zL29hc19wYXJzZXIpCgoKCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyBhcmUgd2VsY29tZSwgcGxlYXNlIGZvbGxvdyBbR2l0SHViIEZsb3ddKGh0dHBzOi8vZ3VpZGVzLmdpdGh1Yi5jb20vaW50cm9kdWN0aW9uL2Zsb3cvaW5kZXguaHRtbCkK readmeEtag: '"c585d4ba158489b3abc75240e2a9651e74086477"' readmeLastModified: Thu, 06 Jul 2023 13:35:37 GMT repositoryId: 114796887 description: An open source Open API Spec 3 Definition Parser created: '2017-12-19T18:12:33Z' updated: '2023-07-06T13:35:44Z' language: Ruby archived: false stars: 51 watchers: 14 forks: 14 owner: Nexmo logo: https://avatars.githubusercontent.com/u/551057?v=4 license: MIT repoEtag: '"0a35a24ca8893655a51b51c53f6728aed92ecabe23fd68115e401f7d963f8aba"' repoLastModified: Thu, 06 Jul 2023 13:35:44 GMT foundInMaster: true id: f2aacc6ec161af737088d825023eb174 - source: https://openapi.tools/ name: openapi3 category: Parsers repository: https://github.com/dorthu/openapi3 language: Python source_description: >- An OpenAPI 3 Specification client, and validator, covering both description validation and limited data validation for Python 3. v2: false v3: true repositoryMetadata: base64Readme: >- b3BlbmFwaTMKPT09PT09PT0KCkEgUHl0aG9uIGBPcGVuQVBJIDMgU3BlY2lmaWNhdGlvbmBfIGNsaWVudCBhbmQgdmFsaWRhdG9yIGZvciBQeXRob24gMy4KCi4uIGltYWdlOjogaHR0cHM6Ly9naXRodWIuY29tL0RvcnRodS9vcGVuYXBpMy9hY3Rpb25zL3dvcmtmbG93cy9tYWluLnltbC9iYWRnZS5zdmcKICAgOmFsdDogR2l0SHViIEFjdGlvbnMgQnVpbGQgQmFkZ2UKCi4uIGltYWdlOjogaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL3B5L29wZW5hcGkzLnN2ZwogICA6dGFyZ2V0OiBodHRwczovL2JhZGdlLmZ1cnkuaW8vcHkvb3BlbmFwaTMKCgpWYWxpZGF0aW9uIE1vZGUKLS0tLS0tLS0tLS0tLS0tCgpUaGlzIG1vZHVsZSBjYW4gYmUgcnVuIGFnYWluc3QgYSBzcGVjIGZpbGUgdG8gdmFsaWRhdGUgaXQgbGlrZSBzbzo6CgogICBweXRob24zIC1tIG9wZW5hcGkzIC9wYXRoL3RvL3NwZWMKClVzYWdlIGFzIGEgQ2xpZW50Ci0tLS0tLS0tLS0tLS0tLS0tCgpUaGlzIGxpYnJhcnkgYWxzbyBmdW5jdGlvbnMgYXMgYW4gaW50ZXJhY3RpdmUgY2xpZW50IGZvciBhcmJpdHJhcnkgT3BlbkFQSSAzCnNwZWNzLiBGb3IgZXhhbXBsZSwgdXNpbmcgYExpbm9kZSdzIE9wZW5BUEkgMyBTcGVjaWZpY2F0aW9uYF8gZm9yIHJlZmVyZW5jZTo6CgogICBmcm9tIG9wZW5hcGkzIGltcG9ydCBPcGVuQVBJCiAgIGltcG9ydCB5YW1sCgogICAjIGxvYWQgdGhlIHNwZWMgZmlsZSBhbmQgcmVhZCB0aGUgeWFtbAogICB3aXRoIG9wZW4oJ29wZW5hcGkueWFtbCcpIGFzIGY6CiAgICAgICBzcGVjID0geWFtbC5zYWZlX2xvYWQoZi5yZWFkKCkpCgogICAjIHBhcnNlIHRoZSBzcGVjIGludG8gcHl0aG9uIC0gdGhpcyB3aWxsIHJhaXNlIGlmIHRoZSBzcGVjIGlzIGludmFsaWQKICAgYXBpID0gT3BlbkFQSShzcGVjKQoKICAgIyBjYWxsIG9wZXJhdGlvbnMgYW5kIHJlY2VpdmUgcmVzdWx0IG1vZGVscwogICByZWdpb25zID0gYXBpLmNhbGxfZ2V0UmVnaW9ucygpCgogICAjIGF1dGhlbnRpY2F0ZSB1c2luZyBhIHNlY3VyaXR5U2NoZW1lIGRlZmluZWQgaW4gdGhlIHNwZWMncyBjb21wb25lbnRzLnNlY3VyaXR5U2NoZW1lcwogICBhcGkuYXV0aGVudGljYXRlKCdwZXJzb25hbEFjY2Vzc1Rva2VuJywgbXlfdG9rZW4pCgogICAjIGNhbGwgYW4gb3BlcmF0aW9uIHRoYXQgcmVxdWlyZXMgYXV0aGVudGljYXRpb24KICAgbGlub2RlcyAgPSBhcGkuY2FsbF9nZXRMaW5vZGVJbnN0YW5jZXMoKQoKICAgIyBjYWxsIGFuIG9wZXJhdGlvbiB3aXRoIHBhcmFtZXRlcnMKICAgbGlub2RlID0gYXBpLmNhbGxfZ2V0TGlub2RlSW5zdGFuY2UocGFyYW1ldGVycz17Imxpbm9kZUlkIjogMTIzfSkKCiAgICMgdGhlIG1vZGVscyByZXR1cm5zIGFyZSBhbGwgb2YgdGhlIHNhbWUgKGdlbmVyYXRlZCkgdHlwZQogICBwcmludCh0eXBlKGxpbm9kZSkpICAgICAgICAgICAgICAgICAgICAgICMgb3BlbmFwaS5zY2hlbWFzLkxpbm9kZQogICB0eXBlKGxpbm9kZSkgPT0gdHlwZShsaW5vZGVzLmRhdGFbMF0pICAgICMgVHJ1ZQoKICAgIyBjYWxsIGFuIG9wZXJhdGlvbiB3aXRoIGEgcmVxdWVzdCBib2R5CiAgIG5ld19saW5vZGUgPSBhcGkuY2FsbF9jcmVhdGVMaW5vZGVJbnN0YW5jZShkYXRhPXsicmVnaW9uIjoidXMtZWFzdCIsInR5cGUiOiJnNi1zdGFuZGFyZC0yIn0pCgogICAjIHRoZSByZXR1cm5lZCBtb2RlbHMgaXMgc3RpbGwgb2YgdGhlIGNvcnJlY3QgdHlwZQogICB0eXBlKG5ld19saW5vZGUpID09IHR5cGUobGlub2RlKSAgICAgIyBUcnVlCgpIVFRQIGJhc2ljIGF1dGhlbnRpY2F0aW9uIGFuZCBIVFRQIGRpZ2VzdCBhdXRoZW50aWNhdGlvbiB3b3JrcyBsaWtlIHRoaXM6OgoKICAgIyBhdXRoZW50aWNhdGUgdXNpbmcgYSBzZWN1cml0eVNjaGVtZSBkZWZpbmVkIGluIHRoZSBzcGVjJ3MgY29tcG9uZW50cy5zZWN1cml0eVNjaGVtZXMKICAgIyBUdXBsZSB3aXRoICh1c2VybmFtZSwgcGFzc3dvcmQpIGFzIHNlY29uZCBhcmd1bWVudAogICBhcGkuYXV0aGVudGljYXRlKCdiYXNpY0F1dGgnLCAoJ3VzZXJuYW1lJywgJ3Bhc3N3b3JkJykpCgpSdW5uaW5nIFRlc3RzCi0tLS0tLS0tLS0tLS0KClRoaXMgcHJvamVjdCBpbmNsdWRlcyBhIHRlc3Qgc3VpdGUsIHJ1biB2aWEgYGBweXRlc3RgYC4gIFRvIHJ1biB0aGUgdGVzdCBzdWl0ZSwKZW5zdXJlIHRoYXQgeW91J3ZlIGluc3RhbGxlZCB0aGUgZGVwZW5kZW5jaWVzIGFuZCB0aGVuIHJ1biBgYHB5dGVzdGBgIGluIHRoZSByb290Cm9mIHRoaXMgcHJvamVjdC4KClJvYWRtYXAKLS0tLS0tLQoKVGhlIGZvbGxvd2luZyBmZWF0dXJlcyBhcmUgcGxhbm5lZCBmb3IgdGhlIGZ1dHVyZToKCiogUmVxdWVzdCBib2R5IG1vZGVscywgY3JlYXRpb24sIGFuZCB2YWxpZGF0aW9uLgoqIFBhcmFtZXRlcnMgaW50ZXJmYWNlIHdpdGggdmFsaWRhdGlvbiBhbmQgZXhwbGljaXQgdHlwaW5nLgoqIFN1cHBvcnQgZm9yIG1vcmUgYXV0aGVudGljYXRpb24gdHlwZXMuCiogU3VwcG9ydCBmb3Igbm9uLWpzb24gcmVxdWVzdC9yZXNwb25zZSBjb250ZW50LgoqIEZ1bGwgc3VwcG9ydCBmb3IgYWxsIG9iamVjdHMgZGVmaW5lZCBpbiB0aGUgc3BlY2lmaWNhdGlvbi4KCi4uIF9PcGVuQVBJIDMgU3BlY2lmaWNhdGlvbjogaHR0cHM6Ly9vcGVuYXBpcy5vcmcKLi4gX0xpbm9kZSdzIE9wZW5BUEkgMyBTcGVjaWZpY2F0aW9uOiBodHRwczovL2RldmVsb3BlcnMubGlub2RlLmNvbS9hcGkvdjQKCg== readmeEtag: '"53e8b32a8eaf9d75e4b358be541b0006e16edabc"' readmeLastModified: Tue, 29 Aug 2023 17:12:35 GMT repositoryId: 132167773 description: A Python3 OpenAPI 3 Spec Parser created: '2018-05-04T17:08:13Z' updated: '2025-11-08T18:47:05Z' language: Python archived: false stars: 121 watchers: 3 forks: 46 owner: Dorthu logo: https://avatars.githubusercontent.com/u/7172953?v=4 license: BSD-3-Clause repoEtag: '"9a83d0d16070a638cc124d3427b374b87d7052603b21952a3ffefce26651661a"' repoLastModified: Sat, 08 Nov 2025 18:47:05 GMT foundInMaster: true id: c4167980072f8c84fc27a9160da0380a - source: https://openapi.tools/ name: openapi3_parser homepage: https://github.com/kevindew/openapi3_parser language: Ruby source_description: >- A Ruby implementation of parser and validator for the OpenAPI 3 Specification. category: Parsers repository: https://github.com/kevindew/openapi3_parser v2: false v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIDMgUGFyc2VyCgohW2NpIHdvcmtmbG93XShodHRwczovL2dpdGh1Yi5jb20va2V2aW5kZXcvb3BlbmFwaTNfcGFyc2VyL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmcpCgpUaGlzIGEgUnVieSBiYXNlZCBwYXJzZXIvdmFsaWRhdG9yIGZvciBbT3BlbkFQSSAzXVtvcGVuYXBpLTNdLiBJdCBpcyB1c2VkIHRvCmNvbnZlcnQgYW4gT3BlbkFQSSBmaWxlIChjYW4gYmUgYSBsb2NhbCBmaWxlLCBhIFVSTCwgYSBzdHJpbmcgb3IgZXZlbiBhIFJ1YnkKaGFzaCkgaW50byBhbiBvYmplY3QgZ3JhcGggd2l0aCBhIHNpbXBsZSBBUEkgdGhhdCBmb2xsb3dzIHRoZSBbT3BlbkFQSQpzcGVjaWZpY2F0aW9uXVtvcGVuYXBpLTMtc3BlY10uCgpCYXNpYyBleGFtcGxlOgoKYGBgcnVieQpyZXF1aXJlICJvcGVuYXBpM19wYXJzZXIiCgpkb2N1bWVudCA9IE9wZW5hcGkzUGFyc2VyLmxvYWRfdXJsKCJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYXN0ZXIvZXhhbXBsZXMvdjMuMC9wZXRzdG9yZS55YW1sIikKCmRvY3VtZW50LnBhdGhzWyIvcGV0cyJdLmdldC5zdW1tYXJ5CiMgPT4gIkxpc3QgYWxsIHBldHMiCmBgYAoKSXQgYWltcyB0byBzdXBwb3J0IDEwMCUgb2YgdGhlIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24sIHdpdGgga2V5IGZlYXR1cmVzCmJlaW5nOgoKLSBTdXBwb3J0cyBsb2FkaW5nIGEgc3BlY2lmaWNhdGlvbiBieSBwYXRoIHRvIGEgZmlsZSwgVVJMLCBSdWJ5IGZpbGUgb2JqZWN0cywKICBhbmQgc3RyaW5ncyBpbiBZQU1MIGFuZCBKU09OIGZvcm1hdHMsIGl0IGV2ZW4gc3VwcG9ydHMgbG9hZGluZyB2aWEgYSBSdWJ5IGhhc2g7Ci0gU3VwcG9ydCBmb3IgbG9hZGluZyByZWZlcmVuY2VzIGZyb20gZXh0ZXJuYWwgZmlsZXMgaW5jbHVkaW5nIFVSTHM7Ci0gSGFuZGxlcyByZWN1cnNpdmUgcmVmZXJlbmNlczsKLSBBbGwgb2YgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIG1hcHBlZCB0byBSdWJ5IG9iamVjdHMsIHByb3ZpZGluZyBhIG5hdHVyYWwKICBSdWJ5IGludGVyZmFjZSB0aGF0IG1hcHMgY2xlYXJseSB0byB0aGUgc3BlY2lmaWNhdGlvbjsKLSBPcGVuQVBJIGZpbGVzIHZhbGlkYXRlZCB3aXRoIGEgc2ltcGxlIEFQSSB0byBxdWlja2x5IGFuZCBzaW1wbHkgc2VlIGFsbAogIHByb2JsZW1zIHdpdGggYSBmaWxlCi0gQnVpbHQtaW4gTWFya2Rvd24gdG8gSFRNTCBjb252ZXJzaW9uOwotIERvY3VtZW50YXRpb24gZm9yIHRoZSBBUEkgdG8gbmF2aWdhdGUgdGhlIE9wZW5BUEkgbm9kZXMgaXMgYXZhaWxhYmxlIG9uCiAgW3J1Ynlkb2MuaW5mb11bZG9jc10uCgpJJ3ZlIHdyb3RlIGEgYmxvZyBwb3N0IHJlZmxlY3Rpbmcgb24gdGhlIGRlY2lzaW9ucyBpbnZvbHZlZCBpbiBidWlsZGluZyB0aGlzCnBhcnNlciBpbiBbSG93IHRvIHdyaXRlIGFuIE9wZW5BUEkgMyBwYXJzZXJdW2Jsb2ddLgoKW29wZW5hcGktM106IGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uCltvcGVuYXBpLTMtc3BlY106IGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4yLm1kI3NwZWNpZmljYXRpb24KW2RvY3NdOiBodHRwOi8vd3d3LnJ1Ynlkb2MuaW5mby9naXRodWIva2V2aW5kZXcvb3BlbmFwaTNfcGFyc2VyL09wZW5hcGkzUGFyc2VyL05vZGUvT3BlbmFwaQpbYmxvZ106IGh0dHBzOi8va2V2aW5kZXcubWUvcG9zdC8xODg2MTE0MjMyMzEvaG93LXRvLXdyaXRlLWFuLW9wZW5hcGktMy1wYXJzZXIKCiMjIFVzYWdlCgojIyMgTG9hZGluZyBhIHNwZWNpZmljYXRpb24KCmBgYHJ1YnkKIyBieSBVUkwKT3BlbmFwaTNQYXJzZXIubG9hZF91cmwoImh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9rZXZpbmRldy9vcGVuYXBpM19wYXJzZXIvbWFpbi9zcGVjL3N1cHBvcnQvZXhhbXBsZXMvcGV0c3RvcmUtZXhwYW5kZWQueWFtbCIpCgojIGJ5IHBhdGggdG8gZmlsZQpPcGVuYXBpM1BhcnNlci5sb2FkX2ZpbGUoInNwZWMvc3VwcG9ydC9leGFtcGxlcy91YmVyLnlhbWwiKQoKIyBieSBGaWxlCk9wZW5hcGkzUGFyc2VyLmxvYWQoRmlsZS5vcGVuKCJzcGVjL3N1cHBvcnQvZXhhbXBsZXMvdWJlci55YW1sIikpCgojIGJ5IFN0cmluZwpPcGVuYXBpM1BhcnNlci5sb2FkKCd7ICJvcGVuYXBpIjogIjMuMC4wIiwgImluZm8iOiB7ICJ0aXRsZSI6ICJBUEkiLCAidmVyc2lvbiI6ICIxLjAuMCIgfSwgInBhdGhzIjoge30gIH0nKQoKIyBieSBIYXNoCk9wZW5hcGkzUGFyc2VyLmxvYWQob3BlbmFwaTogIjMuMC4wIiwgaW5mbzogeyB0aXRsZTogIkFQSSIsIHZlcnNpb246ICIxLjAuMCIgfSwgcGF0aHM6IHt9KQoKYGBgCgojIyMgVmFsaWRhdGluZwoKYGBgcnVieQpkb2N1bWVudCA9IE9wZW5hcGkzUGFyc2VyLmxvYWQob3BlbmFwaTogIjMuMC4wIiwgaW5mbzoge30sIHBhdGhzOiB7fSkKZG9jdW1lbnQudmFsaWQ/CiMgPT4gZmFsc2UKZG9jdW1lbnQuZXJyb3JzCiPCoD0+IE9wZW5hcGkzUGFyc2VyOjpWYWxpZGF0aW9uOjpFcnJvckNvbGxlY3Rpb24oZXJyb3JzOiB7IiMvaW5mbyI9PlsiTWlzc2luZyByZXF1aXJlZCBmaWVsZHM6IHRpdGxlIGFuZCB2ZXJzaW9uIl19KQpgYGAKCiMjIyBUcmF2ZXJzaW5nCgpgYGBydWJ5CmRvY3VtZW50ID0gT3BlbmFwaTNQYXJzZXIubG9hZF91cmwoImh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9rZXZpbmRldy9vcGVuYXBpM19wYXJzZXIvbWFpbi9zcGVjL3N1cHBvcnQvZXhhbXBsZXMvcGV0c3RvcmUtZXhwYW5kZWQueWFtbCIpCgojIGJ5IG9iamVjdHMKCmRvY3VtZW50LmluZm8udGVybXNfb2Zfc2VydmljZQojID0+ICJodHRwOi8vc3dhZ2dlci5pby90ZXJtcy8iCgpkb2N1bWVudC5wYXRocy5rZXlzCiMgPT4gWyIvcGV0cyIsICIvcGV0cy97aWR9Il0KCmRvY3VtZW50LnBhdGhzWyIvcGV0cyJdLmdldC5wYXJhbWV0ZXJzLm1hcCgmOm5hbWUpCiMgPT4gWyJ0YWdzIiwgImxpbWl0Il0KCiMgYnkgaGFzaCBzeW50YXgKCmRvY3VtZW50WyJpbmZvIl1bInRlcm1zT2ZTZXJ2aWNlIl0KPT4gImh0dHA6Ly9zd2FnZ2VyLmlvL3Rlcm1zLyIKCmRvY3VtZW50WyJwYXRocyJdLmtleXMKIyA9PiBbIi9wZXRzIiwgIi9wZXRzL3tpZH0iXQoKZG9jdW1lbnRbInBhdGhzIl1bIi9wZXRzIl1bImdldCJdWyJwYXJhbWV0ZXJzIl0ubWFwKCY6bmFtZSkKIyA9PiBbInRhZ3MiLCAibGltaXQiXQoKIyBieSBhIHBhdGggdG8gYSBub2RlCmRvY3VtZW50Lm5vZGVfYXQoIiMvcGF0aHMvJTJGcGV0cy9nZXQvb3BlcmF0aW9uSWQiKQo9PiAiZmluZFBldHMiCgpkb2N1bWVudC5ub2RlX2F0KCIjL2NvbXBvbmVudHMvc2NoZW1hcy9QZXQvYWxsT2YvMC9yZXF1aXJlZC8wIikKPT4gIm5hbWUiCgojIG9yIGNvbWJpbmluZwoKZG9jdW1lbnQuY29tcG9uZW50cy5zY2hlbWFzWyJQZXQiXS5ub2RlX2F0KCIjLi4vTmV3UGV0IikKPT4gT3BlbmFwaTNQYXJzZXI6Ok5vZGU6OlNjaGVtYSgjL2NvbXBvbmVudHMvc2NoZW1hcy9OZXdQZXQpCmBgYAoKWW91IGNhbiBsZWFybiBtb3JlIGFib3V0IHRoZSBBUEkgb24gW3J1Ynlkb2MuaW5mb11bZG9jc10KCiMjIEluc3RhbGxhdGlvbgoKWW91IGNhbiBpbnN0YWxsIHRoaXMgZ2VtIGludG8geW91ciBidW5kbGVyIGFwcGxpY2F0aW9uIGJ5IGFkZGluZyB0aGlzIGxpbmUgdG8KeW91ciBHZW1maWxlOgoKYGBgCmdlbSAib3BlbmFwaTNfcGFyc2VyIiwgIn4+IDAuMTAuMCIKYGBgCgphbmQgdGhlbiBydW5uaW5nIGAkIGJ1bmRsZSBpbnN0YWxsYAoKT3IgaW5zdGFsbCB0aGUgZ2VtIG9udG8geW91ciBtYWNoaW5lIHZpYSBgJCBnZW0gaW5zdGFsbCBvcGVuYXBpM19wYXJzZXJgCgojIyBTdGF0dXMKClRoaXMgaXMgY3VycmVudGx5IGEgd29yayBpbiBwcm9ncmVzcyBhbmQgd2lsbCByZW1haW4gc28gdW50aWwgaXQgcmVhY2hlcyAxLjAuCgpTZWUgW1RPRE9dKFRPRE8ubWQpIGZvciBkZXRhaWxzIG9mIHRoZSBmZWF0dXJlcyBzdGlsbCB0byBpbXBsZW1lbnQuCgojIyBMaWNlbmNlCgpbTUlUIExpY2Vuc2VdKExJQ0VOQ0UpCg== readmeEtag: '"91b00fb70c82fca634eb45633bb90a4ba8e6349d"' readmeLastModified: Tue, 21 May 2024 08:42:03 GMT repositoryId: 107609542 description: Open API 3 Parser/Validator for Ruby created: '2017-10-19T23:33:52Z' updated: '2025-12-04T15:12:11Z' language: Ruby archived: false stars: 105 watchers: 7 forks: 15 owner: kevindew logo: https://avatars.githubusercontent.com/u/282717?v=4 license: MIT repoEtag: '"4d6aae8751087d6ef04362730b600fd7a1f5d467de0bfa20c74a27e72db2f200"' repoLastModified: Thu, 04 Dec 2025 15:12:11 GMT foundInMaster: true id: 609fe191ac5d7b1b89c772e47fd2deb8 - source: https://openapi.tools/ name: APIMatic Generate category: - Code Generators - SDK language: SaaS link: https://www.apimatic.io/product/generate source_description: >- Bring in your API description (OAI v2/v3, RAML, API Blueprint, WSDL, etc.) to generate fully functional SDKs in over 10 languages. v2: true v3: true v3_1: true id: 7cbb02fd80f5559ac1fd392da1417abe foundInMaster: true - source: https://openapi.tools/ name: docler-labs/api-client-generator category: SDK language: PHP repository: https://github.com/doclerlabs/api-client-generator source_description: >- API client generator is a console application capable of auto-generating a PSR18/PSR7 API client based on OpenAPI specification according to PHP best practices and your code style standards. v3: true id: 0c78f909fbd7707b739560ebfa258b81 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNESyBnZW5lcmF0b3IgLSBBUEkgY2xpZW50IGdlbmVyYXRvcgoKQVBJIGNsaWVudCBnZW5lcmF0b3IgaXMgYSBjb25zb2xlIGFwcGxpY2F0aW9uIGNhcGFibGUgb2YgYXV0by1nZW5lcmF0aW5nIGEgW1BTUjE4XShodHRwczovL3d3dy5waHAtZmlnLm9yZy9wc3IvcHNyLTE4LykvW1BTUjddKGh0dHBzOi8vd3d3LnBocC1maWcub3JnL3Bzci9wc3ItNy8pIGNvbXBsaWFudCBBUEkgY2xpZW50IGJhc2VkIG9uIFtPcGVuQVBJIHYzXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pIHNwZWNpZmljYXRpb24gYWNjb3JkaW5nIHRvIFBIUCBiZXN0IHByYWN0aWNlcyBhbmQgeW91ciBjb2RlIHN0eWxlIHN0YW5kYXJkcy4KClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL0RvY2xlckxhYnMvYXBpLWNsaWVudC1nZW5lcmF0b3Iuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvRG9jbGVyTGFicy9hcGktY2xpZW50LWdlbmVyYXRvcikKWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvRG9jbGVyTGFicy9hcGktY2xpZW50LWdlbmVyYXRvci9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vY292ZXJhbGxzLmlvL2dpdGh1Yi9Eb2NsZXJMYWJzL2FwaS1jbGllbnQtZ2VuZXJhdG9yP2JyYW5jaD1tYXN0ZXIpClshW1BIUFN0YW4gTGV2ZWxdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUEhQU3Rhbi1sZXZlbCUyMDgtYnJpZ2h0Z3JlZW4uc3ZnP3N0eWxlPWZsYXQpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1BIUFN0YW4tbGV2ZWwlMjA4LWJyaWdodGdyZWVuLnN2Zz9zdHlsZT1mbGF0KQoKIyMgUmVxdWlyZW1lbnRzCi0gT3BlbkFQSSA+PSAzLjAKLSBQSFAgPj0gNy4wCgojIyBXaHkgdXNpbmcgaXQ/Ci0gV2l0aCBnZW5lcmF0ZWQgY2xpZW50IHlvdSBhcmUgYWx3YXlzIHN1cmUgdGhhdCB5b3VyIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBpcyB1cC10by1kYXRlLgotIFdvcmsgd2l0aCBvYmplY3RzIGluc3RlYWQgb2YgcmF3IGRhdGEsIGl0IGlzIE9PUCBmcmllbmRseS4KLSBTYXZlcyB5b3VyIHRpbWUuIFlvdSBkb24ndCBuZWVkIHRvIHdyaXRlIGRhdGEgbWFwcGVycyB5b3Vyc2VsZiB0byBwb3B1bGF0ZSB0aG9zZSBvYmplY3RzIHdpdGggdGhlIGRhdGEgZnJvbSB0aGUgcmVzcG9uc2UuCi0gQWxsIHRoZSBiYXNpYyB0eXBlIHZhbGlkYXRpb25zIGluIHRoZSByZXF1ZXN0IGFuZCB0aGUgcmVzcG9uc2UgZG9uZSBhdXRvbWF0aWNhbGx5LgotIERlc3BpdGUgdGhlIGZhY3QgdGhlIGNvZGUgaXMgZ2VuZXJhdGVkIGl0J3MgY2xlYXIgYW5kIHJlYWRhYmxlLCBzaW1wbGUgdG8gZGVidWcgYW5kIHRvIHJlYXNvbiBhYm91dC4KLSBIaWdobHkgY29uZmlndXJhYmxlIGFuZCBleHRlbnNpYmxlLgotIFJlbGlhYmxlIGFuZCB3ZWxsIHRlc3RlZC4KLSBTaW1wbHksICoqaWYgc29tZXRoaW5nIGNhbiBiZSBhdXRvbWF0ZWQgaXQgc2hvdWxkIGJlIGF1dG9tYXRlZC4qKiAgRm9jdXMgb24gaW1wb3J0YW50IHN0dWZmLgoKIyMgRmVhdHVyZXMKLSBTdXBwb3J0cyB5YW1sIG9yIGpzb24gc3BlY2lmaWNhdGlvbiBmaWxlIGZvcm1hdHMuCi0gVmFsaWRhdGVzIHlvdXIgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLgotIFN1cHBvcnRzIG11bHRpcGxlIGNvbnRlbnQgdHlwZXM6CiAgICAqIGFwcGxpY2F0aW9uL2pzb24KICAgICogYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkCiAgICAqIGFwcGxpY2F0aW9uL3htbAotIFN1cHBvcnRzIG5ldyBQSFAgdmVyc2lvbnMgc3ludGF4IGZlYXR1cmVzLgotIEl0IGlzIGJhc2UgY2xpZW50IGluZGVwZW5kZW50LCB5b3UgYXJlIGZyZWUgdG8gY2hvb3NlIGFueSBbZXhpc3RpbmcgUFNSLTE4IGNvbXBsaWFudCBjbGllbnRdKGh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wcm92aWRlcnMvcHNyL2h0dHAtY2xpZW50LWltcGxlbWVudGF0aW9uKS4gSnVzdCBjaG9vc2UgdGhlIG9uZSB3aGljaCB5b3UgYWxyZWFkeSB1c2UsIHNvIGdlbmVyYXRlZCBjbGllbnQgd291bGQgbm90IGNhdXNlIGFueSBjb25mbGljdHMgd2l0aCB5b3VyIGRlcGVuZGVuY2llcy4gQWx0aG91Z2ggbm90IHJlY29tbWVuZGVkLCB5b3UgY2FuIGFsc28gdXNlIG9yIGJ1aWxkIHlvdXIgb3duIFBTUi0xOCBpbXBsZW1lbnRhdGlvbiwgYXMgdGhlIGdlbmVyYXRlZCBjbGllbnQgZGVwZW5kcyBvbiBQU1IgaW50ZXJmYWNlcyBvbmx5LgotIEFwcGxpZXMgY29kZSBzdHlsZSBydWxlcyB0byBnZW5lcmF0ZWQgY29kZSwgeW91IGNhbiBzcGVjaWZ5IHlvdXIgb3duLgotIEdlbmVyYXRlcyBSRUFETUUgYW5kIGNvbXBvc2VyLmpzb24gZmlsZXMgd2l0aCBwb3NzaWJpbGl0eSB0byB1c2UgeW91ciBvd24gdGVtcGxhdGUuCi0gU3VwcG9ydHMgYGFsbE9mYCwgYG9uZU9mYCwgYGFueU9mYCBPcGVuQVBJIHBhcmFtZXRlcnMuCi0gU3VwcG9ydHMgbnVsbGFibGUgb3B0aW9uYWwgc2NoZW1lIHByb3BlcnR5LgoKIyMgRXhhbXBsZQpDaGVjayBvdXQgW2V4YW1wbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9Eb2NsZXJMYWJzL2FwaS1jbGllbnQtZ2VuZXJhdG9yL3RyZWUvbWFzdGVyL2V4YW1wbGUpIGRpcmVjdG9yeSB0byBzZWUgdGhlIGNvZGUgZ2VuZXJhdGVkIGJ5IGFwaS1jbGllbnQtZ2VuZXJhdG9yLgoKVHJ5IGl0IG91dDoKYGBgYmFzaApjZCBleGFtcGxlICYmIFwKY29tcG9zZXIgaW5zdGFsbCAmJiBcCnBocCB0ZXN0LWV4YW1wbGUucGhwCmBgYAoKIyMgVXNhZ2UKIyMjIFdpdGggRG9ja2VyCmBgYGJhc2gKJCBkb2NrZXIgcnVuIC1pdCBcCi12IHtwYXRoLXRvLXNwZWNpZmljYXRpb259L29wZW5hcGkueWFtbDovb3BlbmFwaS55YW1sOnJvIFwKLXYge3BhdGgtdG8tY2xpZW50fS9zb21lLWFwaS1jbGllbnQ6L2NsaWVudCBcCi1lIE5BTUVTUEFDRT1Hcm91cFxcU29tZUFwaUNsaWVudCBcCi1lIE9QRU5BUEk9L29wZW5hcGkueWFtbCBcCi1lIE9VVFBVVF9ESVI9L2NsaWVudCBcCi1lIFBBQ0tBR0U9Z3JvdXAvc29tZS1hcGktY2xpZW50IFwKZGhsYWJzL2FwaS1jbGllbnQtZ2VuZXJhdG9yCmBgYAoKPiBpZiB5b3UncmUgcnVubmluZyB0aGlzIGNvbW1hbmQgb24gV2luZG93cyB5b3UgbWlnaHQgbmVlZCB0byB1c2Ugc2luZ2xlIGJhY2tzbGFzaCBpbnN0ZWFkIGluIC1lIE5BTUVTUEFDRQoKIyMjIFdpdGhvdXQgRG9ja2VyClByZWNvbmRpdGlvbnM6IFBIUCA3LjQKCkNsb25lIHRoZSByZXBvc2l0b3J5IGFuZCBydW46CmBgYGJhc2ggCk9QRU5BUEk9e3BhdGgtdG8tc3BlY2lmaWNhdGlvbn0vb3BlbmFwaS55YW1sIE5BTUVTUEFDRT1Hcm91cFxTb21lQXBpQ2xpZW50IFBBQ0tBR0U9Z3JvdXAvc29tZS1hcGktY2xpZW50IE9VVFBVVF9ESVI9e3BhdGgtdG8tY2xpZW50fS9nZW5lcmF0ZWQgLi9iaW4vYXBpLWNsaWVudC1nZW5lcmF0b3IgZ2VuZXJhdGUKYGBgIAoKIyMgQ29uZmlndXJhdGlvbgpUaGUgZm9sbG93aW5nIGVudmlyb25tZW50IHZhcmlhYmxlcyBhcmUgYXZhaWxhYmxlOgoKfCBWYXJpYWJsZSB8IFJlcXVpcmVkIHwgRGVmYXVsdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCBFbnVtIHwgRXhhbXBsZSAgICAgICAgICAgICAgICAgICAgfCBEZXNjcmlwdGlvbiB8CnwtLS0tLS0tLS0tLS18LS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS18LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CnwgYE5BTUVTUEFDRWAgfCB5ZXMgfCB8IHwgR3JvdXBcXFNvbWVBcGlDbGllbnQgfAp8IGBQQUNLQUdFYCB8IHllcyB8IHwgfCBncm91cC9zb21lLWFwaS1jbGllbnQgfAp8IGBPUEVOQVBJIGAgfCB5ZXMgfCB8IHwgL2FwaS9vcGVuYXBpLnlhbWwgfAp8IGBPVVRQVVRfRElSYCB8IHllcyB8IHwgfCAvY2xpZW50IHwKfCBgQ09ERV9TVFlMRWAgfCBubyB8IHtwYXRoLXRvLXJlcG9zaXRvcnl9Ly5waHAtY3MtZml4ZXIucGhwLmRpc3QgfCB8IC9jbGllbnQvbXlDb2RlU3R5bGUucGhwIHwKfCBgU09VUkNFX0RJUmAgfCBubyB8IHNyYyB8IHwgc3JjIHwKfCBgQ0xJRU5UX1BIUF9WRVJTSU9OYCB8IG5vIHwgNy40IHwgNy4wLCA3LjEsIDcuMiwgNy4zLCA3LjQsIDguMCwgOC4xLCA4LjIsIDguMyB8IDcuNCB8CnwgYENPTVBPU0VSX0pTT05fVEVNUExBVEVfRElSYCB8IG5vIHwge3BhdGgtdG8tcmVwb3NpdG9yeX0vdGVtcGxhdGUvY29tcG9zZXIuanNvbi50d2lnIHwgfCAvcGF0aC9jb21wb3Nlci5qc29uLnR3aWcgfAp8IGBSRUFETUVfTURfVEVNUExBVEVfRElSYCB8IG5vIHwge3BhdGgtdG8tcmVwb3NpdG9yeX0vdGVtcGxhdGUvUkVBRE1FLm1kLnR3aWcgfCB8IC9wYXRoL1JFQURNRS5tZC50d2lnIHwKfCBgSFRUUF9NRVNTQUdFYCB8IG5vIHwgZ3V6emxlIHwgZ3V6emxlLCBueWhvbG0gfCBueWhvbG0gfAp8IGBDT05UQUlORVJgIHwgbm8gfCBwaW1wbGUgfCBwaW1wbGUgfCBwaW1wbGUgfAp8IGBJTkNMVURFX1RBR1NgIHwgbm8gfCB8IHwgdGFnMSx0YWcyLHRhZzMgfCB0YWcgd2hpdGVsaXN0IHRvIHNlbGVjdCBnZW5lcmF0ZWQgb3BlcmF0aW9ucyB8CnwgYEVYQ0xVREVfVEFHU2AgfCBubyB8IHwgfCB0YWcxLHRhZzIsdGFnMyB8IHRhZyBibGFja2xpc3QgdG8gc2VsZWN0IGdlbmVyYXRlZCBvcGVyYXRpb25zIHwKCiMjIFJ1bm5pbmcgdGVzdHMKCmBgYGJhc2gKJCBjb21wb3NlciBpbnN0YWxsCiQgbWFrZSB0ZXN0CmBgYAoKKGNoZWNrIGBtYWtlYCBmb3IgYWxsIGF2YWlsYWJsZSByb3V0aW5lcykuCg== readmeEtag: '"8c1ca279271af3a1c6567ccf856aaafbb88b6cb5"' readmeLastModified: Tue, 14 Jan 2025 11:11:53 GMT repositoryId: 267043092 description: >- API client generator is a console application capable of generating an API client based on OpenAPI(Swagger) specification. created: '2020-05-26T12:57:31Z' updated: '2026-01-29T15:01:51Z' language: PHP archived: false stars: 39 watchers: 13 forks: 22 owner: DoclerLabs logo: https://avatars.githubusercontent.com/u/12913135?v=4 license: MIT repoEtag: '"204baae5d50e4d630edb064482cf9222d95af7f77b4cdd154f280dc88dbec52f"' repoLastModified: Thu, 29 Jan 2026 15:01:51 GMT foundInMaster: true - source: https://openapi.tools/ name: janephp/open-api category: SDK language: PHP repository: https://github.com/janephp/open-api source_description: Generate a PHP Client API (PSR-7 compatible) given a OpenAPI specification. v2: true v3: true repositoryMetadata: base64Readme: >- IyBKYW5lIE9wZW5BUEkKCkdlbmVyYXRlIGEgUEhQIENsaWVudCBBUEkgKFBTUjcgY29tcGF0aWJsZSkgZ2l2ZW4gYSBbT3BlbkFQSSAoU3dhZ2dlcikgc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMi4wLm1kKS4KCiMjIExpY2Vuc2UKClZpZXcgdGhlIFtMSUNFTlNFXShMSUNFTlNFKSBmaWxlIGF0dGFjaCB0byB0aGlzIHByb2plY3QuCgojIyBSZXNvdXJjZXMKCiAqIFtEb2N1bWVudGF0aW9uXShodHRwOi8vamFuZS5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvKQogKiBbQ29udHJpYnV0aW5nXShodHRwczovL2dpdGh1Yi5jb20vamFuZXBocC9qYW5lcGhwL2Jsb2IvbWFzdGVyL0NPTlRSSUJVVElORy5tZCkKICogW1JlcG9ydCBJc3N1ZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9qYW5lcGhwL2phbmVwaHAvaXNzdWVzKSBhbmQgW3NlbmQgUHVsbCBSZXF1ZXN0c10oaHR0cHM6Ly9naXRodWIuY29tL2phbmVwaHAvamFuZXBocC9wdWxscykgCiBpbiB0aGUgW21haW4gSmFuZSBSZXBvc2l0b3J5XShodHRwczovL2dpdGh1Yi5jb20vamFuZXBocC9qYW5lcGhwKQogCiMjIFNwb25zb3IKClshW0pvbGlDb2RlXShodHRwczovL2pvbGljb2RlLmNvbS9pbWFnZXMvbG9nby5zdmcpXShodHRwczovL2pvbGljb2RlLmNvbSkKCk9wZW4gU291cmNlIHRpbWUgc3BvbnNvcmVkIGJ5IEpvbGlDb2RlCgojIyBDcmVkaXRzCgoqIFtBbGwgY29udHJpYnV0b3JzXShodHRwczovL2dpdGh1Yi5jb20vam9saWNvZGUvamFuZS9ncmFwaHMvY29udHJpYnV0b3JzKQo= readmeEtag: '"a1df2b781a7f621d4ed7f6708fa220b5d61ab794"' readmeLastModified: Fri, 20 Mar 2020 07:42:44 GMT repositoryId: 112961308 description: '[READ-ONLY] Subtree split of Jane OpenAPI library' created: '2017-12-03T20:30:50Z' updated: '2025-10-16T12:59:10Z' language: PHP archived: true stars: 45 watchers: 2 forks: 10 owner: janephp logo: https://avatars.githubusercontent.com/u/21155132?v=4 license: MIT repoEtag: '"8640d6f648e0b8f363db6299989faf2085237240df43d1e9e7180395faccad4c"' repoLastModified: Thu, 16 Oct 2025 12:59:10 GMT foundInMaster: true id: 6db70c053b88f67063720245f841e5a6 - source: https://openapi.tools/ name: go-swagger category: - Parsers - SDK - Converters link: https://goswagger.io/ repository: https://github.com/go-swagger/go-swagger language: Go source_description: >- Unmaintained v2.0 only project seeking new maintainer, or probably a fork. Parser, validator, generates descriptions from code, or code from descriptions! v2: true repositoryMetadata: base64Readme: >- # Swagger 2.0

<!-- Badges: status  -->
[![Tests][test-badge]][test-url] [![Coverage][cov-badge]][cov-url] [![CI vuln scan][vuln-scan-badge]][vuln-scan-url] [![CodeQL][codeql-badge]][codeql-url]
<!-- Badges: release & docker images  -->
[![Release][release-badge]][release-url] [![Container Registry on Quay.io][quay-badge]][quay-url] [![Container Registry on Github][ghcr-badge]][ghcr-url]
<!-- Badges: code quality  -->
[![Go Report Card][gocard-badge]][gocard-url] [![CodeFactor Grade][codefactor-badge]][codefactor-url]
<!-- Badges: license & compliance -->
[![License][license-badge]][license-url] [![Open SSF Scorecard][ossf-badge]][ossf-url] [![OpenSSF Best Practices][ossf-cci-badge]][ossf-cci-url] [![OSS licences status][fossa-badge]][fossa-url]
<!-- Badges: documentation & support -->
<!-- Badges: others & stats -->
[![Documentation][doc-badge]][doc-url] [![GoDoc][godoc-badge]][godoc-url] [![Slack Channel][slack-badge]][slack-url] [![go version][goversion-badge]][goversion-url] ![Top language][top-badge] ![Commits since latest release][commits-badge]

---

This project contains a golang implementation of Swagger 2.0 (aka [OpenAPI 2.0](https://github.com/OAI/OpenAPI-Specification/blob/old-v3.2.0-dev/versions/2.0.md)).
It provide tools to work with swagger specifications.

[Swagger](https://swagger.io/) is a simple yet powerful representation of your RESTful API.<br>

## Documentation

<https://goswagger.io>

##  Features

`go-swagger` brings to the go community a complete suite of fully-featured, high-performance, API components to  work with a Swagger API: server, client and data model.

* Generates a server from a swagger specification
* Generates a client from a swagger specification
* Generates a CLI (command line tool) from a swagger specification (alpha stage)
* Supports most features offered by jsonschema and swagger, including polymorphism
* Generates a swagger specification from annotated go code
* Additional tools to work with a swagger spec
* Great customization features, with vendor extensions and customizable templates

Our focus with code generation is to produce idiomatic, fast go code, which plays nice with golint, go vet etc.

[More details](https://goswagger.io/go-swagger/features).

##  Project status

This project supports OpenAPI 2.0. **At this moment it does not support OpenAPI 3.x**.

`go-swagger` is now feature complete and has stabilized its API.

Most features and building blocks are now in a stable state, with a rich set of CI tests.

The go-openapi community actively continues bringing fixes and enhancements to this code base.

There is still much room for improvement: contributors and PR's are welcome. You may also get in touch with maintainers on [our slack channel](https://slackin.goswagger.io).

## Installing

```sh
go install github.com/go-swagger/go-swagger/cmd/swagger@latest
```

`go-swagger` is also available as binary or docker releases as well as from source: [more details](https://goswagger.io/go-swagger/install).

## Try it

Try `go-swagger` in a free online workspace using Gitpod:

[![Open in Gitpod](https://gitpod.io/button/open-in-gitpod.svg)](https://gitpod.io#https://github.com/go-swagger/go-swagger)

## Licensing

The toolkit itself is licensed under an Apache Software License 2.0:
[SPDX-License-Identifier: Apache-2.0](./LICENSE).

Just like swagger, this does not cover code generated by the toolkit. That code is entirely yours to license however you see fit.

### Licence scan on dependencies 

[![FOSSA Status][fossa-badge-large]][fossa-url-large]

<!-- Badges: status  -->
[test-badge]: https://github.com/go-swagger/go-swagger/actions/workflows/test.yaml/badge.svg
[test-url]: https://github.com/go-swagger/go-swagger/actions/workflows/test.yaml
[cov-badge]: https://codecov.io/gh/go-swagger/go-swagger/branch/master/graph/badge.svg
[cov-url]: https://codecov.io/gh/go-swagger/go-swagger
[vuln-scan-badge]: https://github.com/go-swagger/go-swagger/actions/workflows/scanner.yaml/badge.svg
[vuln-scan-url]: https://github.com/go-swagger/go-swagger/actions/workflows/scanner.yaml
[codeql-badge]: https://github.com/go-swagger/go-swagger/actions/workflows/codeql.yaml/badge.svg
[codeql-url]: https://github.com/go-swagger/go-swagger/actions/workflows/codeql.yaml
<!-- Badges: release & docker images  -->
[release-badge]: https://badge.fury.io/gh/go-swagger%2Fgo-swagger.svg
[release-url]: https://badge.fury.io/gh/go-swagger%2Fgo-swagger
[quay-badge]: https://img.shields.io/badge/dynamic/json?url=https%3A%2F%2Fquay.io%2Fapi%2Fv1%2Frepository%2Fgoswagger%2Fswagger%2Ftag%2F%3Flimit%3D1%26onlyActiveTags%3Dtrue%26filter_tag_name%3Dlike%3Av&label=Container%20Registry%20on%20quay.io&query=%24.tags[:1].name&logo=redhatopenshift&logoColor=#EE0000?color=green
[quay-url]: https://quay.io/repository/goswagger/swagger?tab=tags
[ghcr-badge]: https://ghcr-badge-ipv2.onrender.com/go-swagger/go-swagger/latest_tag?ignore=sha-*,edge,master&label=Container%20Registry%20on%20Github
[ghcr-url]: https://github.com/orgs/go-swagger/packages/container/go-swagger/versions?filters[version_type]=tagged
<!-- Badges: code quality  -->
[gocard-badge]: https://goreportcard.com/badge/github.com/go-swagger/go-swagger
[gocard-url]: https://goreportcard.com/report/github.com/go-swagger/go-swagger
[codefactor-badge]: https://img.shields.io/codefactor/grade/github/go-swagger/go-swagger
[codefactor-url]: https://www.codefactor.io/repository/github/go-swagger/go-swagger
<!-- Badges: documentation & support -->
[doc-badge]: https://img.shields.io/badge/doc-site-blue?link=https%3A%2F%2Fgoswagger.io%2Fgo-swagger%2F
[doc-url]: https://goswagger.io/go-swagger
[godoc-badge]: https://godoc.org/github.com/go-swagger/go-swagger?status.svg
[godoc-url]: http://godoc.org/github.com/go-swagger/go-swagger
[slack-badge]: https://slackin.goswagger.io/badge.svg
[slack-url]: https://slackin.goswagger.io
<!-- Badges: license & compliance -->
[license-badge]: http://img.shields.io/badge/license-Apache%20v2-orange.svg
[license-url]: https://github.com/go-swagger/go-swagger/?tab=Apache-2.0-1-ov-file#readme
[ossf-badge]: https://api.securityscorecards.dev/projects/github.com/go-swagger/go-swagger/badge
[ossf-url]: https://securityscorecards.dev/viewer/?uri=github.com/go-swagger/go-swagger
[ossf-cci-badge]: https://www.bestpractices.dev/projects/11359/badge
[ossf-cci-url]: https://www.bestpractices.dev/projects/11359
[fossa-badge]: https://app.fossa.io/api/projects/git%2Bgithub.com%2Fgo-swagger%2Fgo-swagger.svg?type=shield
[fossa-url]: https://app.fossa.io/projects/git%2Bgithub.com%2Fgo-swagger%2Fgo-swagger?ref=badge_shield
[fossa-badge-large]: https://app.fossa.io/api/projects/git%2Bgithub.com%2Fgo-swagger%2Fgo-swagger.svg?type=large
[fossa-url-large]: https://app.fossa.io/projects/git%2Bgithub.com%2Fgo-swagger%2Fgo-swagger?ref=badge_large
[oai-url]: https://raw.githubusercontent.com/swagger-api/swagger-spec/master/LICENSE
<!-- Badges: others & stats -->
[goversion-badge]: https://img.shields.io/github/go-mod/go-version/go-swagger/go-swagger
[goversion-url]: https://github.com/go-swagger/go-swagger/blob/master/go.mod
[top-badge]: https://img.shields.io/github/languages/top/go-swagger/go-swagger
[commits-badge]: https://img.shields.io/github/commits-since/go-swagger/go-swagger/latest
 readmeEtag: '"74f6d1cd7718c8461a573dd85cbdd7876beba7ad"' readmeLastModified: Sun, 09 Nov 2025 20:29:42 GMT repositoryId: 26726495 description: Swagger 2.0 implementation for go created: '2014-11-16T20:13:15Z' updated: '2026-02-05T16:28:58Z' language: Go archived: false stars: 9936 watchers: 112 forks: 1293 owner: go-swagger logo: https://avatars.githubusercontent.com/u/10362650?v=4 license: Apache-2.0 repoEtag: '"2854fd5bea806a164e813794ed0f7d4ef9c71abe1e682b9e641fe12f21b5d7c9"' repoLastModified: Thu, 05 Feb 2026 16:28:58 GMT foundInMaster: true id: ef136df9ffb40bb33ccbd76665d8f5b4 - source: https://openapi.tools/ name: docs category: - Parsers - Converters - SDK link: https://github.com/go-oas/docs repository: https://github.com/go-oas/docs source_description: >- A modern alternative to `go-swagger`. Offers generation and parsing of OpenAPI Specs, depending on the usage. v3: true v3_1: true id: 421481e00ec1f8e0df0d1e7806edbe2d repositoryMetadata: base64Readme: >- IyBkb2NzCgojIyMgQXV0b21hdGljYWxseSBnZW5lcmF0ZSBSRVNUZnVsIEFQSSBkb2N1bWVudGF0aW9uIGZvciBHTyBwcm9qZWN0cyAtIGFsaWduZWQgd2l0aCBbT3BlbiBBUEkgU3BlY2lmaWNhdGlvbiBzdGFuZGFyZF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjMubWQpLgoKPGltZyBhbGlnbj0icmlnaHQiIHdpZHRoPSIxODBweCIgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20va2F5bmV0aWsvZG90ZmlsZXMvbWFzdGVyL3N2Zy1yZXNvdXJjZXMvZ28tZ3JwYy13ZWIuc3ZnIj4KCiFbR29sYW5nQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1vYXMvZG9jcy93b3JrZmxvd3MvZ29sYW5nY2kvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKQohW0J1aWxkXShodHRwczovL2dpdGh1Yi5jb20vZ28tb2FzL2RvY3Mvd29ya2Zsb3dzL0J1aWxkL2JhZGdlLnN2Zz9icmFuY2g9bWFpbikKWyFbVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS92ZXJzaW9uLXYxLjAuNS1zdWNjZXNzLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1vYXMvZG9jcy9yZWxlYXNlcykKWyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL2dvLW9hcy9kb2NzKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL2dvLW9hcy9kb2NzKQpbIVtDb3ZlcmFnZSBTdGF0dXNdKGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9nby1vYXMvZG9jcy9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2NvdmVyYWxscy5pby9naXRodWIvZ28tb2FzL2RvY3M/YnJhbmNoPW1haW4pClshW2NvZGViZWF0IGJhZGdlXShodHRwczovL2NvZGViZWF0LmNvL2JhZGdlcy8zMmI4NjU1Ni04NGUzLTRkYjktOWYxMS05MjNkMTI5OTRmOTApXShodHRwczovL2NvZGViZWF0LmNvL3Byb2plY3RzL2dpdGh1Yi1jb20tZ28tb2FzLWRvY3MtbWFpbikKWyFbR28gUmVmZXJlbmNlXShodHRwczovL3BrZy5nby5kZXYvYmFkZ2UvZ2l0aHViLmNvbS9nby1vYXMvZG9jcy5zdmcpXShodHRwczovL3BrZy5nby5kZXYvZ2l0aHViLmNvbS9nby1vYXMvZG9jcykKWyFbTWVudGlvbmVkIGluIEF3ZXNvbWUgR29dKGh0dHBzOi8vYXdlc29tZS5yZS9tZW50aW9uZWQtYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9hd2Vzb21lLWdvLmNvbSkKCmdvLU9BUyBEb2NzIGNvbnZlcnRzIHN0cnVjdHVyZWQgT0FTMyAoU3dhZ2dlcjMpIG9iamVjdHMgaW50byB0aGUgT3BlbiBBUEkgU3BlY2lmaWNhdGlvbiAmIGF1dG9tYXRpY2FsbHkgc2VydmVzIGl0IG9uCmNob3NlbiByb3V0ZSBhbmQgcG9ydC4gSXQncyBleHRyZW1lbHkgZmxleGlibGUgYW5kIHNpbXBsZSwgc28gYmFzaWNhbGx5IGl0IGNhbiBiZSBpbnRlZ3JhdGVkIGludG8gYW55IGZyYW1ld29yayBvcgpleGlzdGluZyBwcm9qZWN0LgoKV2UgaW52aXRlIGFueW9uZSBpbnRlcmVzdGVkIHRvIGpvaW4gb3VyICoqW0dIIERpc2N1c3Npb25zIGJvYXJkXShodHRwczovL2dpdGh1Yi5jb20vZ28tb2FzL2RvY3MvZGlzY3Vzc2lvbnMpKiouIEhvbmVzdApmZWVkYmFjayB3aWxsIGVuYWJsZSB1cyB0byBidWlsZCBiZXR0ZXIgcHJvZHVjdCBhbmQgYXQgdGhlIHNhbWUgdGltZSBub3Qgd2FzdGUgdmFsdWFibGUgdGltZSBhbmQgZWZmb3J0IG9uIHNvbWV0aGluZwp0aGF0IG1pZ2h0IG5vdCBmaXQgaW50ZW5kZWQgdXNhZ2UuIFNvIGlmIHlvdSBjYW4sIHBsZWFzZSBzcGFyZSBmZXcgbWludXRlcyB0byBnaXZlIHlvdXIgb3BpbmlvbiBvZiB3aGF0IHNob3VsZCBiZSBkb25lCm5leHQsIG9yIHdoYXQgc2hvdWxkIGJlIHRoZSBwcmlvcml0eSBmb3Igb3VyIHJvYWRtYXAuIDptdXNjbGU6IDp0YWRhOgoKLS0tLQoKIyMgVGFibGUgb2YgQ29udGVudHMKCi0gW0dldHRpbmcgU3RhcnRlZF0oI2dldHRpbmctc3RhcnRlZCkKLSBbSG93IHRvIHVzZV0oI2hvdy10by11c2UpCiAgICAqIFtFeGFtcGxlc10oI2V4YW1wbGVzKQotIFtDb250YWN0XSgjY29udGFjdCkKLSBbVGhlIGN1cnJlbnQgcm9hZG1hcCAocGxhbm5lZCBmZWF0dXJlcyldKCNyb2FkbWFwKQoKIyMgR2V0dGluZyBTdGFydGVkCgoxLiBEb3dubG9hZCAqKl9kb2NzXyoqIGJ5IHVzaW5nOgogICBgYGBzaAogICAkIGdvIGdldCAtdSBnaXRodWIuY29tL2dvLW9hcy9kb2NzCiAgIGBgYCAKMi4gQWRkIG9uZSBsaW5lIGFubm90YXRpb24gdG8gdGhlIGhhbmRsZXIgeW91IHdpc2ggdG8gdXNlIGluIHRoZSBmb2xsb3dpbmcKICAgZm9ybWF0OiBgLy8gQE9BUyA8RlVOQ19OQU1FPiA8Uk9VVEU+IDxIVFRQX01FVEhPRD5gCiAgIEV4YW1wbGVzOgogICBgYGAKICAgLy8gQE9BUyBoYW5kbGVDcmVhdGVVc2VyIC91c2VycyBQT1NUCiAgIC8vIEBPQVMgaGFuZGxlR2V0VXNlciAvdXNlcnMgR0VUCiAgIGBgYAozLiBEZWNsYXJlIGFsbCByZXF1aXJlZCBkb2N1bWVudGF0aW9uIGVsZW1lbnRzIHRoYXQgYXJlIHNoYXJlZC4gT3IgcmV1c2Ugb25lcyB0aGF0IGFscmVhZHkgZXhpc3QgaW4gdGhlIGV4YW1wbGVzCiAgIGRpcmVjdG9yeS4KNC4gRGVjbGFyZSBzcGVjaWZpYyBkb2NzIGVsZW1lbnRzIHBlciByb3V0ZS4KCi0tLS0KCiMjIEhvdyB0byB1c2UKCkZvciBtb3JlIGV4cGxpY2l0IGV4YW1wbGUsIHBsZWFzZSByZWZlciB0byBbZG9jcy9leGFtcGxlc10oaHR0cHM6Ly9naXRodWIuY29tL2dvLW9hcy9kb2NzL2V4YW1wbGVzKQoKQWRkIE9BUyBUQUcgdG8geW91ciBleGlzdGluZyBoYW5kbGVyIHRoYXQgaGFuZGxlcyBmZXRjaGluZyBvZiBhIFVzZXI6CgpgYGBnbwpwYWNrYWdlIHVzZXJzCgppbXBvcnQgIm5ldC9odHRwIgoKLy8gQE9BUyBoYW5kbGVHZXRVc2VyIC91c2VycyBHRVQKZnVuYyAocyAqc2VydmljZSkgaGFuZGxlR2V0VXNlcigpIGh0dHAuSGFuZGxlckZ1bmMgewoJcmV0dXJuIGZ1bmModyBodHRwLlJlc3BvbnNlV3JpdGVyLCByICpodHRwLlJlcXVlc3QpIHsKCX0KfQpgYGAKCkNyZWF0ZSBhIHVuaXF1ZSBBUEkgZG9jdW1lbnRhdGlvbiBmdW5jdGlvbiBmb3IgdGhhdCBlbmRwb2ludDoKCmBgYGdvCnBhY2thZ2UgbWFpbgoKaW1wb3J0ICJnaXRodWIuY29tL2dvLW9hcy9kb2NzIgoKZnVuYyBoYW5kbGVHZXRVc2VyUm91dGUob2FzUGF0aEluZGV4IGludCwgb2FzICpkb2NzLk9BUykgewoJcGF0aCA6PSBvYXMuR2V0UGF0aEJ5SW5kZXgob2FzUGF0aEluZGV4KQoKCXBhdGguU3VtbWFyeSA9ICJHZXQgYSBVc2VyIgoJcGF0aC5PcGVyYXRpb25JRCA9ICJnZXRVc2VyIgoJcGF0aC5SZXF1ZXN0Qm9keSA9IGRvY3MuUmVxdWVzdEJvZHl7fQoJcGF0aC5SZXNwb25zZXMgPSBkb2NzLlJlc3BvbnNlc3sKCQlnZXRSZXNwb25zZU9LKCksCgl9CgoJcGF0aC5UYWdzID0gYXBwZW5kKHBhdGguVGFncywgInBldCIpCn0KYGBgCgpCZWFyIGluIG1pbmQgdGhhdCBjcmVhdGluZyBhIHVuaXF1ZSBmdW5jdGlvbiBwZXIgZW5kcG9pbnQgaGFuZGxlciBpcyBub3QgcmVxdWlyZWQsIGJ1dCBzaW1wbHkgcHJvdmlkZXMgZ29vZCB2YWx1ZSBpbgp1c2FiaWxpdHkgb2Ygc2hhcmVkIGRvY3VtZW50YXRpb24gZWxlbWVudHMuCgpPbmNlIHlvdSBjcmVhdGVkIHRoZSBmdW5jdGlvbiwgc2ltcGx5IHJlZ2lzdGVyIGl0IGZvciBwYXJzaW5nIGJ5IHVzaW5nIGBBdHRhY2hSb3V0ZXMoKWAgZGVmaW5lZCB1cG9uIGBPQVNgIHN0cnVjdHVyZS4KRS5nLjoKCmBgYGdvCgpwYWNrYWdlIG1haW4KCmltcG9ydCAoCgkiZ2l0aHViLmNvbS9nby1vYXMvZG9jcyIKKQoKZnVuYyBtYWluKCkgewoJYXBpRG9jIDo9IGRvY3MuTmV3KCkKCWFwaURvYy5BdHRhY2hSb3V0ZXMoW11pbnRlcmZhY2V7fXsKCQloYW5kbGVHZXRVc2VyUm91dGUsCgl9KQpgYGAKCklmIHRoaXMgYXBwcm9hY2ggaXMgdG9vIGZsZXhpYmxlIGZvciB5b3UsIHlvdSBhcmUgYWx3YXlzIGxlZnQgd2l0aCB0aGUgcG9zc2liaWxpdHkgdG8gY3JlYXRlIHlvdXIgb3duIGF0dGFjaGVyIC0gb3IgYW55Cm90aGVyIHBhcnRzIG9mIHRoZSBzeXN0ZW0gZm9yIHRoYXQgbWF0dGVyLgoKIyMjIEV4YW1wbGVzCgpUbyBydW4gZXhhbXBsZXMsIGFuZCBjaGVja291dCBob3N0ZWQgZG9jdW1lbnRhdGlvbiB2aWEgU3dhZ2dlciBVSSwgaXNzdWUgdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgc2gKJCBnbyBydW4gLi9leGFtcGxlcy9maWxlX291dHB1dC8qLmdvCiMgb3IgdW5jb21tZW50IGxpbmUgNDAgYW5kIGNvbW1lbnQgbGluZSAzOCBpbiBpbnRlcm5hbC9kaXN0L2luZGV4Lmh0bWwgYmVmb3JlIHJ1bm5pbmc6CiQgZ28gcnVuIC4vZXhhbXBsZXMvc3RyZWFtX291dHB1dC8qLmdvCmBgYAoKQW5kIG5hdmlnYXRlIHRvIGBodHRwOi8vbG9jYWxob3N0OjMwMDUvZG9jcy9hcGkvYCBpbiBjYXNlIHRoYXQgeW91IGRpZG4ndCBjaGFuZ2UgYW55dGhpbmcgYmVmb3JlIHJ1bm5pbmcgdGhlIGV4YW1wbGUKYWJvdmUuCgotLS0tCgojIyBDb250YWN0CgpDaGVjayBvdXQgdGhlIGN1cnJlbnQgW1Byb2plY3QgYm9hcmRdKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1vYXMvZG9jcy9wcm9qZWN0cy8xKSBvcgpvdXIgKipbR0ggRGlzY3Vzc2lvbnMgYm9hcmRdKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1vYXMvZG9jcy9kaXNjdXNzaW9ucykqKiBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCllvdSBjYW4gam9pbiBvdXIgVGVsZWdyYW0gZ3JvdXAgYXQ6IFtodHRwczovL3QubWUvZ29fb2FzXShodHRwczovL3QubWUvZ29fb2FzKQoKIyMgUm9hZG1hcAoKfCBGZWF0dXJlIChHSCBpc3N1ZXMpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCBEZXNjcmlwdGlvbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgUmVsZWFzZSB8CnwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0gfAp8IFtWYWxpZGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vZ28tb2FzL2RvY3MvaXNzdWVzLzE3KSAgICAgICAgICB8IEFkZCB2YWxpZGF0aW9uIHRvIGFsbCBzdHJ1Y3R1cmVzIGJhc2VkIG9uIE9BUzMuMC4zICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCB2MS4xLjAgIHwKfCBbQ0xJXShodHRwczovL2dpdGh1Yi5jb20vZ28tb2FzL2RvY3MvaXNzdWVzLzE4KSAgICAgICAgICAgICAgICAgfCBBZGQgQ0xJIHN1cHBvcnQgLSBtYWtlIGl0IENMSSBmcmllbmRseSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgdjEuMi4wICB8CnwgW1Bvc3RtYW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1vYXMvZG9jcy9pc3N1ZXMvMTkpICAgICAgICAgICAgIHwgQWRkIHBvc3RtYW4gc3VwcG9ydCB2aWEgUE0gQVBJICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IHYxLjMuMCAgfAp8IFtSZURvY10oaHR0cHM6Ly9naXRodWIuY29tL2dvLW9hcy9kb2NzL2lzc3Vlcy8yMCkgICAgICAgICAgICAgICB8IEFkZCBSZURvYyBzdXBwb3J0IGFzIGFuIGFsdGVybmF0aXZlIHRvIFN3YWdnZXJVSSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCB2MS40LjAgIHwKfCBbRTJFIEF1dG8tZ2VuZXJhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL2dvLW9hcy9kb2NzL2lzc3Vlcy8yMSkgfCBHbyB0ZXN0cyBjb252ZXJzaW9uIHRvIEN5cHJlc3MvS2F0YWxvbiBzdWl0ZXMgKGNvbnZlcnQgbW9ja2VkIHVuaXQgdGVzdHMgaW50byBlMmUgdGVzdHMpIHwgdjEuNS4wICB8Cgo= readmeEtag: '"815b19fa38654d6cd804e799ef507047e040dc05"' readmeLastModified: Wed, 15 Mar 2023 07:41:11 GMT repositoryId: 333859606 description: >- Automatically generate RESTful API documentation for GO projects - aligned with Open API Specification standard created: '2021-01-28T18:51:47Z' updated: '2025-09-27T15:33:34Z' language: Go archived: false stars: 48 watchers: 1 forks: 6 owner: go-oas logo: https://avatars.githubusercontent.com/u/78081803?v=4 license: MIT repoEtag: '"be1e6c936790de5db538901397f449f859142b6359f479eb5c47ba3c931b96b1"' repoLastModified: Sat, 27 Sep 2025 15:33:34 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: restful-react homepage: https://github.com/contiamo/restful-react language: React (Typescript) source_description: >- Generate React hooks with appropriate type-signatures from OpenAPI descriptions category: SDK link: https://github.com/contiamo/restful-react repository: https://github.com/contiamo/restful-react v2: true v3: true repositoryMetadata: base64Readme: >- # `restful-react`

[![npm](https://img.shields.io/npm/v/restful-react.svg)](https://www.npmjs.com/package/restful-react)

⚠️ **restful-react has been deprecated:** We are focusing on a new open api generator that generates `react-query` components: https://github.com/fabien0102/openapi-codegen ⚠️

Building React apps that interact with a RESTful API presents a set of questions, challenges and potential gotchas. This project aims to remove such pitfalls, and provide a pleasant developer experience when crafting such applications.

It can be considered **a thin wrapper around the [fetch API](https://developer.mozilla.org/en/docs/Web/API/Fetch_API) in the form of React components and hooks.**

When used in a setup with [OpenAPI / Swagger](https://en.wikipedia.org/wiki/OpenAPI_Specification) specs and [Typescript](https://www.typescriptlang.org/), **restful-react ensures a reliable and always up to date contract between backend and frontend.** It generates components and types from your specs and can be integrated quite comfortably into your development workflows (featuring for example the import of OpenAPI specs from your github repos).

restful-react is **very well tested, production ready** and powers all of our projects at Contiamo.

<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->

- [Overview](#overview)
- [Getting Started](#getting-started)
- [Features](#features)
  - [Global Configuration](#global-configuration)
    - [`RestfulProvider` API](#restfulprovider-api)
  - [Loading and Error States](#loading-and-error-states)
  - [Lazy Fetching](#lazy-fetching)
  - [Response Resolution](#response-resolution)
  - [Debouncing Requests](#debouncing-requests)
  - [TypeScript Integration](#typescript-integration)
  - [Query Parameters](#query-parameters)
  - [Mutations with `useMutate`](#mutations-with-usemutate)
  - [Mocks](#mocks)
  - [Polling with `Poll`](#polling-with-poll)
    - [Long Polling](#long-polling)
    - [Full `Poll` Component API](#full-poll-component-api)
    - [Polling and Code Generation](#polling-and-code-generation)
  - [Code Generation from OpenAPI / Swagger specs](#code-generation-from-openapi--swagger-specs)
    - [Usage](#usage)
    - [Validation of the OpenAPI specification](#validation-of-the-openapi-specification)
    - [API Versioning](#api-versioning)
    - [Import from URL](#import-from-url)
    - [Import from GitHub](#import-from-github)
    - [Transforming an Original Spec](#transforming-an-original-spec)
    - [Advanced configuration](#advanced-configuration)
      - [Config File Format](#config-file-format)
      - [Config File Example](#config-file-example)
      - [Custom generator](#custom-generator)
      - [Only generating custom code (no react hooks/components)](#only-generating-custom-code-no-react-hookscomponents)
- [Contributing](#contributing)
  - [Code](#code)
  - [How to publish to npm](#how-to-publish-to-npm)
- [`@without-cli` npm package](#without-cli-npm-package)
- [Next Steps](#next-steps)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

## Overview

At its core, `restful-react` exposes a [hook](https://reactjs.org/docs/hooks-intro.html), called `useGet`. This component retrieves data, either on mount or later, and then handles error states, loading states, and other cases for you. As such, you get a component that _gets stuff_ and then does stuff with it. Here's a quick overview what it looks like.

[![Edit restful-react demos](https://codesandbox.io/static/img/play-codesandbox.svg)](https://codesandbox.io/s/restful-react-demos-vjets)

```jsx
import React from "react";
import { useGet } from "restful-react";

const MyComponent = () => {
  const { data: randomDogImage } = useGet({
    path: "https://dog.ceo/api/breeds/image/random",
  });

  return <img alt="Here's a good boye!" src={randomDogImage && randomDogImage.message} />;
};

export default MyComponent;
```

and on React Native,
[Edit restful-react basic demo on Expo](https://snack.expo.io/SJaSAj49r)

```jsx
import { AppRegistry, Image } from "react-native";
import React from "react";

import { useGet } from "restful-react";

const App = () => {
  const { data: randomDogImage } = useGet({
    path: "https://dog.ceo/api/breeds/image/random",
  });
  return (
    <>
      {randomDogImage && (
        <Image
          style={{ width: 250, height: 250 }}
          source={{
            uri: randomDogImage.message,
          }}
        />
      )}
    </>
  );
};

AppRegistry.registerComponent("react-native-app", () => App);
```

## Getting Started

To install and use this library, install it by running `yarn add restful-react`, or `npm i restful-react --save` and you should be good to go. Don't forget to `import { useGet } from "restful-react"` or similar wherever you need it!

## Features

`restful-react` ships with the following features that we think might be useful.

### Global Configuration

REST API endpoints usually sit alongside a base, global URL. As a convenience, the `RestfulProvider` allows top-level configuration of your requests, that are then passed down the React tree to `useGet` hooks.

Consider,

[![Edit restful-react demos](https://codesandbox.io/static/img/play-codesandbox.svg)](https://codesandbox.io/s/restful-react-demos-vjets)

```jsx
// index.js

import React from "react";
import { RestfulProvider } from "restful-react";

import App from "./App.jsx";

const MyRestfulApp = () => (
  <RestfulProvider base="https://dog.ceo/api">
    <App />
  </RestfulProvider>
);

export default MyRestfulApp;
```

Meanwhile, in `./App.jsx`,

```jsx
// App.jsx

import React from "react";
import { useGet } from "restful-react";

const MyComponent = () => {
  const { data: randomDogImage } = useGet({
    // Inferred from RestfulProvider in index.js
    path: "breeds/image/random",
  });

  return <img alt="Here's a good boye!" src={randomDogImage && randomDogImage.message} />;
};

export default MyComponent;
```

Naturally, the request will be sent to the full path `https://dog.ceo/api/breeds/image/random`. The full API of the `RestfulProvider` is outlined below. Each configuration option is composable and _can be_ overridden by `Get` components further down the tree.

#### `RestfulProvider` API

Here's a full overview of the API available through the `RestfulProvider`, along with its defaults.

```tsx
// Interface
export interface RestfulReactProviderProps<T = any> {
  /** The backend URL where the RESTful resources live. */
  base: string;
  /**
   * The path that gets accumulated from each level of nesting
   * taking the absolute and relative nature of each path into consideration
   */
  parentPath?: string;
  /**
   * A function to resolve data return from the backend, most typically
   * used when the backend response needs to be adapted in some way.
   */
  resolve?: ResolveFunction<T>;
  /**
   * Options passed to the fetch request.
   */
  requestOptions?: ((url: string, method: string, requestBody?: string) => Partial<RequestInit>) | Partial<RequestInit>;
  /**
   * Trigger on each error.
   * For `Get` and `Mutation` calls, you can also call `retry` to retry the exact same request.
   * Please note that it's quite hard to retrieve the response data after a retry mutation in this case.
   * Depending of your case, it can be easier to add a `localErrorOnly` on your `Mutate` component
   * to deal with your retry locally instead of in the provider scope.
   */
  onError?: (err: any, retry: () => Promise<T | null>, response?: Response) => void;
  /**
   * Trigger on each request.
   */
  onRequest?: (req: Request) => void;
  /**
   * Trigger on each response.
   */
  onResponse?: (req: Response) => void;
  /**
   * Query parameters passed to each request.
   */
  queryParams?: { [key: string]: any };
  /**
   * Query parameter stringify options applied for each request.
   */
  queryParamStringifyOptions?: IStringifyOptions;
}

// Usage
<RestfulProvider
  base="String!"
  resolve={data => data}
  requestOptions={(url, method, requestBody) => ({ headers: { Authorization: authToken } })}
/>;
```

Here's some docs about the [RequestInit](https://developer.mozilla.org/en-US/docs/Web/API/Request/Request) type of request options.

### Loading and Error States

`useGet` hooks return an object with loading and error states, to allow for state handling. Consider,

[![Edit restful-react demos](https://codesandbox.io/static/img/play-codesandbox.svg)](https://codesandbox.io/s/restful-react-demos-vjets)

```jsx
import React from "react";
import { useGet } from "restful-react";

const MyComponent = () => {
  const { data: randomDogImage, loading } = useGet({
    path: "https://dog.ceo/api/breeds/image/random",
  });

  return loading ? <h1>Loading...</h1> : <img alt="Here's a good boye!" src={randomDogImage.message} />;
};

export default MyComponent;
```

### Lazy Fetching

It is possible to use a `useGet` hook and defer the fetch to a later stage. This is done with the `lazy` boolean property. This is great for displaying UI immediately, and then allowing parts of it to be fetched as a response to an event: like the click of a button, for instance. Consider,

[![Edit restful-react demos](https://codesandbox.io/static/img/play-codesandbox.svg)](https://codesandbox.io/s/restful-react-demos-vjets)

```jsx
import React from "react";
import { useGet } from "restful-react";

const MyComponent = () => {
  const { data: randomDogImage, loading, refetch } = useGet({
    path: "https://dog.ceo/api/breeds/image/random",
    lazy: true,
  });

  return !randomDogImage && loading ? (
    <h1>Loading!</h1>
  ) : (
    <div>
      <div>
        <h1>Welcome to my image getter!</h1>
        <button onClick={() => refetch()}>Get a good boye!</button>
      </div>
      <div>{randomDogImage && <img alt="Here's a good boye!" src={randomDogImage.message} />}</div>
    </div>
  );
};

export default MyComponent;
```

The above example will display your UI, and then load good boyes on demand.

### Response Resolution

Sometimes, your backend responses arrive in a shape that you might want to adapt, validate, or restructure. Other times, maybe your data consistently arrives in a `{ data: {} }` shape, with `data` containing the stuff you want.

At the `RestfulProvider` level, _or_ on the `useGet` level, a `resolve` prop will take the data and _do stuff_ to it, providing the final resolved or unwrapped data to the children. Consider,

[![Edit restful-react demos](https://codesandbox.io/static/img/play-codesandbox.svg)](https://codesandbox.io/s/restful-react-demos-vjets)

```jsx
import React from "react";
import { useGet } from "restful-react";

const MyComponent = () => {
  const { data: imageUrl } = useGet({
    path: "https://dog.ceo/api/breeds/image/random",
    resolve: image => image && image.message,
  });

  return imageUrl && <img alt="Here's a good boye!" src={imageUrl} />;
};

export default MyComponent;
```

### Debouncing Requests

Some requests fire in response to a rapid succession of user events: things like autocomplete or resizing a window. For this reason, users sometimes need to wait until all the keystrokes are typed (until everything's _done_), before sending a request.

`restful-react` exposes a `debounce` prop on `Get` that does exactly this.

Here's an example:

```jsx
const SearchThis = props => {
  const { data } = useGet({
    path: "/hello/world",
    debounce: true,
  });

  return (
    <div>
      <h1>Here's all the things I search</h1>
      <ul>
        {data.map(thing => (
          <li>{thing}</li>
        ))}
      </ul>
    </div>
  );
};
```

Debounce also accepts a number, which tells `useGet` how long to wait until doing the request.

```diff
const SearchThis = props => {
  const { data } = useGet({
    path: "/hello/world",
-    debounce: true,
+    debounce: 200 /* ms */,
  })

  return <div>
        <h1>Here's all the things I search</h1>
        <ul>
          {data.map(thing => (
            <li>{thing}</li>
          ))}
        </ul>
      </div>
}
```

It uses [lodash's debounce](https://lodash.com/docs/4.17.10#debounce) function under the hood, so you get all the benefits of it out of the box like so!

```diff

const SearchThis = props => {
  const { data } = useGet({
    path: "/hello/world",
-    debounce: 200,
+    debounce: { wait: 200, options: { leading: true, maxWait: 300, trailing: false } } /* ms */,
  })

  return <div>
        <h1>Here's all the things I search</h1>
        <ul>
          {data.map(thing => (
            <li>{thing}</li>
          ))}
        </ul>
      </div>
}
```

### TypeScript Integration

One of the most powerful features of `restful-react` is that each component exported is strongly typed, empowering developers through self-documenting APIs.

![Using restful-react in VS Code](assets/labs.gif)

### Query Parameters

All components in this library support query params (`https://my.site/?query=param`) via a `queryParams` prop. Each `useGet`, `useMutate` and `Poll` instance is _generic_, having a type signature of `useGet<TData, TError, TQueryParams>`. If described, the `queryParams` prop is _fully_ type-safe in usage and provides autocomplete functionality.

![Autocompletion on QueryParams](assets/idp.gif)

Please note that the above example was built using our [OpenAPI generator](#code-generation) in order to infer the type of component from the specification and automatically generate the entire type-safe component in a _very_ quick and easy way.

### Mutations with `useMutate`

`restful-react` exposes an additional hook called `useMutate`. These components allow sending requests with other HTTP verbs in order to mutate backend resources.

[![Edit restful-react demos](https://codesandbox.io/static/img/play-codesandbox.svg)](https://codesandbox.io/s/restful-react-demos-vjets)

```jsx
import React from "react";
import { useGet, useMutate } from "restful-react";

const base = "https://jsonplaceholder.typicode.com";

const ListItem = ({ id, children }) => {
  const { mutate: del, loading } = useMutate({
    verb: "DELETE",
    path: `/posts/`,
    base,
  });

  return (
    <li key={id}>
      {loading ? (
        "Deleting..."
      ) : (
        <button onClick={() => del(id).then(() => alert("Deleted successfully. Pretend it got removed from the DOM."))}>
          ❌
        </button>
      )}
      &nbsp;{children}
    </li>
  );
};

const MyHugeList = () => {
  const { data: posts } = useGet({
    path: "/posts",
    base,
  });
  return (
    <div>
      <h1>Posts</h1>
      <ul>
        {posts &&
          posts.map(post => (
            <ListItem key={post.id} id={post.id}>
              {post.title}
            </ListItem>
          ))}
      </ul>
    </div>
  );
};
export default MyHugeList;
```

`useMutate` is strongly typed, and provides intelligent autocompletion out of the box, complete with other available [HTTP verbs](https://developer.mozilla.org/de/docs/Web/HTTP/Methods).

![Mutate](assets/mutate.png)

Each mutation returns a promise that can then be used to update local component state, dispatch an action, or do something else depending on your use case.

### Mocks

No backend support yet for your amazing feature? Need to isolate an edge case? You can easily provide a mock to `useMutate` and `useGet` to bypass the classic flow.

/!\ If `mock` option is provided, no requests will be send to the server. /!\

```jsx
import React from "react";
import { useGet, useMutate } from "restful-react";

const base = "https://jsonplaceholder.typicode.com";

// Mock the `mutate` handler
const { mutate: del, loading } = useMutate({
  verb: "DELETE",
  path: `/posts/`,
  base,
  // This will avoid any server call in favor of mock response
  mock: {
    mutate: id => console.log(`The item ${id} was deleted`),
  },
});

// Mock the `loading`, so it's easy to isolate the loading state
const { data: posts } = useGet({
  path: "/posts",
  base,
  // This will avoid any server call in favor of mock response
  mock: {
    loading: true,
  },
});

// Mock the `error`, so it's easy to isolate the error state
const { data: posts } = useGet({
  path: "/posts",
  base,
  // This will avoid any server call in favor of mock response
  mock: {
    error: "oh no!",
  },
});
```

### Polling with `Poll`

`restful-react` also exports a `Poll` render props component that will poll a backend endpoint over a predetermined interval until a stop condition is met. Consider,

```jsx
import { Poll } from "restful-react"

<Poll path="/deployLogs" resolve={data => data && data.data}>
  {(deployLogs: DeployLog[], { loading }) =>
    loading ? (
      <PageSpinner />
    ) : (
      <DataTable
        columns={["createdAt", "deployId", "status", "sha", "message"]}
        orderBy="createdAt"
        data={deployLogs}
        formatters={{
          createdAt: (d: DeployLog["createdAt"]) => title(formatRelative(d, Date.now())),
          sha: (i: DeployLog["sha"]) => i && i.slice(0, 7),
        }}
      />
    )
  }
</Poll>
```

`Poll` supports:

- an `interval` prop that will poll at a specified interval (defaults to polling 1 second), and
- an `until` prop that accepts a condition expressed as a function that returns a boolean value. When this condition is met, polling will stop.

  The signature of this function is `(data: T, response: ResponseInit) => boolean`. As a developer, you have access to the returned data, along with the response object in case you'd like to stop polling if `response.ok === false`, for example.

Below is a more convoluted example that employs nearly the full power of the `Poll` component.

```jsx
<Poll path="/status" until={(_, response) => response && response.ok} interval={0} lazy>
  {(_, { loading, error, finished, polling }, { start }) => {
    return loading ? (
      <Progress error={error} />
    ) : (
      <Button
        loading={editorLoading || polling}
        condensed
        icon="ExternalLink"
        color="ghost"
        onClick={() => {
          if (finished) {
            return window.open(editor.url);
          }
          requestEditor();
          start();
        }}
      >
        {finished ? "Launch Editor" : "Request Editor"}
      </Button>
    );
  }}
</Poll>
```

Note from the previous example, `Poll` also exposes more states: `finished`, and `polling` that allow better flow control, as well as lazy-start polls that can also be programmatically stopped at a later stage.

#### Long Polling

At Contiamo, we have a [powerful Long Polling specification](docs/contiamo-long-poll.md) in place that allows us to build real-time apps over HTTPS, as opposed to WebSockets. At a glance the specification can be distilled into:

- Web UI sends a request with a `Prefer` header that contains:
  - a time, in seconds, to keep requests open (`60s`), and
  - a **polling index** that is a server-sent hash `ahpiegh`.
  - all together, the client sends a request with a header `Prefer: wait=60s;index=939192`.
- The backend server responds, either with:
  - an empty response with status `304 Not Modified`
  - a successful response with data and a new **polling index**.

The polling index allow the client and the server to stay in sync: the client says "the last stuff I got was at this index". The server says "oh, let me get you up to speed and send you a new index".

Visually, this is represented as below.

![Contiamo Poll](docs/long-poll-flow.png).

To get this functionality in `restful-react`, this means specifying a `wait` prop on your `Poll` component, provided your server implements this specification as well.

#### [Full `Poll` Component API](src/Poll.tsx#L53-L101)

#### Polling and Code Generation

By default we generate a `Poll` component when the `prefer` header is specified in the OpenAPI/Swagger specs (more information about this design decision here -> https://github.com/contiamo/restful-react#long-polling).

We do not generate an equivalent hook version. Polling is quite trivial in a react hook, so we usually just use `useEffect` when we need some polling feature.

Example:

```ts
// Poll data if no completedAt
useEffect(() => {
  if (error) {
    return onError();
  } else if (data && !data.completedAt) {
    const timerId = window.setTimeout(() => refetch(), 1000);
    return () => window.clearTimeout(timerId);
  } else {
    return;
  }
}, [data, refetch, error]);
```

### Code Generation from OpenAPI / Swagger specs

`restful-react` is able to generate React hooks with appropriate type-signatures (TypeScript) from any valid OpenAPI v3 or Swagger v2 specification, either in `yaml` or `json` formats.

#### Usage

Type-safe React data fetchers can be generated from an OpenAPI specification using the following command:

- `restful-react import --file MY_OPENAPI_SPEC.yaml --output my-awesome-generated-types.tsx`

This command can be invoked by _either_:

- Installing `restful-react` globally and running it in the terminal: `npm i -g restful-react`, or
- Adding a `script` to your `package.json` like so:

```diff
      "scripts": {
        "start": "webpack-dev-server",
        "build": "webpack -p",
+       "generate-fetcher": "restful-react import --file MY_SWAGGER_DOCS.json --output FETCHERS.tsx"
      }
```

Your components can then be generated by running `npm run generate-fetcher`. Optionally, we recommend linting/prettifying the output for readability like so:

```diff
      "scripts": {
        "start": "webpack-dev-server",
        "build": "webpack -p",
        "generate-fetcher": "restful-react import --file MY_SWAGGER_DOCS.json --output FETCHERS.tsx",
+       "postgenerate-fetcher": "prettier FETCHERS.d.tsx --write"
      }
```

#### Validation of the OpenAPI specification

To enforce the best quality as possible of specification, we have integrated the amazing [OpenAPI linter from IBM](https://github.com/IBM/openapi-validator). We strongly encourage you to setup your custom rules with a `.validaterc` file, you can find all useful information about this configuration [here](https://github.com/IBM/openapi-validator/#configuration).

To activate this, add a `--validation` flag to your `restful-react` call.

#### API Versioning

The generated file will include an exported constant `SPEC_VERSION` that will contain to the OpenAPI `info.version` property's value.

#### Import from URL

Adding the `--url` flag to `restful-react import` instead of using the `--file` flag will attempt to fetch the spec from that endpoint.

- `restful-react import --url https://api.mine.com/openapi.json --output my-awesome-generated-types.tsx`

#### Import from GitHub

Adding the `--github` flag to `restful-react import` instead of using the `--file` flag allows us to **create React components from an OpenAPI spec _remotely hosted on GitHub._** <sup>_(how is this real life_ 🔥 _)_</sup>

To generate components from remote specifications, you'll need to follow the following steps:

1.  Visit [your GitHub settings](https://github.com/settings/tokens).
1.  Click **Generate New Token** and choose the following:

        Token Description: (enter anything)
        Scopes:
            [X] repo
                [X] repo:status
                [X] repo_deployment
                [X] public_repo
                [X] repo:invite
                [X] security_events

1.  Click **Generate token**.
1.  Copy the generated string.
1.  Open a terminal and run `restful-react import --github username:repo:branch:path/to/openapi.yaml --output MY_FETCHERS.tsx`, substituting things where necessary.
1.  You will be prompted for a token.
1.  Paste your token.
1.  You will be asked if you'd like to save it for later. This is _entirely_ up to you and completely safe: it is saved in your home directory.
1.  You're done! 🎉

Note: For CI environment, you can also provide the github token with the environment variable called `GITHUB_TOKEN`

#### Transforming an Original Spec

In some cases, you might need to augment an existing OpenAPI specification on the fly, for code-generation purposes. Our CLI makes this quite straightforward:

```bash
  restful-react import --file myspec.yaml --output mybettercomponents.tsx --transformer path/to/my-transformer.js
```

The function specified in `--transformer` is pure: it imports your `--file`, transforms it, and passes the augmented OpenAPI specification to `restful-react`'s generator. Here's how it can be used:

```ts
// /path/to/my-transformer.js

/**
 * Transformer function for restful-react.
 *
 * @param {OpenAPIObject} schema
 * @return {OpenAPIObject}
 */
module.exports = inputSchema => ({
  ...inputSchema,
  // Place your augmentations here
  paths: Object.entries(schema.paths).reduce(
    (mem, [path, pathItem]) => ({
      ...mem,
      [path]: Object.entries(pathItem).reduce(
        (pathItemMem, [verb, operation]) => ({
          ...pathItemMem,
          [verb]: {
            ...fixOperationId(path, verb, operation),
          },
        }),
        {},
      ),
    }),
    {},
  ),
});
```

#### Advanced configuration

`restful-react` supports the concept of "schema stitching" in a RESTful ecosystem as well. We are able to tie multiple backends together and generate code using a single configuration file, `restful-react.config.js`

To activate this "advanced mode", replace all flags from your `restful-react` call with the config flag: `--config restful-react.config.js` (or any filename that you want).

⚠️ **Note:** using a config file makes use of all of the options contained therein, and ignores all other CLI flags.

##### Config File Format

```ts
interface RestfulReactConfig {
  [backend: string]: {
    // classic configuration
    output: string;
    file?: string;
    github?: string;
    transformer?: string;
    validation?: boolean;
    skipReact?: boolean;

    // advanced configuration
    customImport?: string;
    customProps?: {
      base?: string;
    };
    pathParametersEncodingMode?: "uriComponent" | "rfc3986";
    customGenerator?: (data: {
      componentName: string;
      verb: string;
      route: string;
      description: string;
      genericsTypes: string;
      operation: OperationObject;
      paramsInPath: string[];
      paramsTypes: string;
    }) => string;
  };
}
```

##### Config File Example

```js
// restful-react.config.js
/**
 * Restful-react configuration.
 *
 * @type {import("restful-react/dist/bin/config").RestfulReactAdvancedConfiguration}
 */
module.exports = {
  myFirstBackend: {
    output: "src/queries/myFirstBackend.tsx",
    file: "specs/my-first-backend.yaml",
    customProps: {
      base: `"http://my-first-backend.com"`,
    },
  },
  configurableBackend: {
    output: "src/queries/configurableBackend.tsx",
    github: "contiamo:restful-react:master:docs/swagger.json",
    customImport: `import { getConfig } from "../components/Config.tsx";`,
    customProps: {
      base: `{getConfig("backendBasePath")}`,
    },
  },
};
```

```json
// package.json
{
  "scripts": {
    "gen": "restful-react import --config restful-react.config.js",
    "gen-first": "restful-react import --config restful-react.config.js myFirstBackend"
  }
}
```

##### Custom generator

To support even more advanced usecases (like a promise base API, mock generator or anything else that can infer from your specs), you can define your own template in `customGenerator`. This function will be call for each route with some useful computed values (see the types above) and the resulted string will be added to the generated file.

You can see a concrete usage inside the `examples` folder and try yourself in this repository with the following command:

- `yarn build`
- `yarn example:advanced petstore-custom-fetch`

You can inspect the result inside `/examples/petstoreFromFileSpecWithCustomFetch.tsx`

##### Only generating custom code (no react hooks/components)

In some cases you might want to use the familiar restful-react to generate code for non-react environments (e.g. promise-based fetchers for nodejs or other frameworks). In this case, you can disable react code generation altogether by passing the `--skipReact` flag or, if you are using a configuration file, setting `skipReact: true`.

When set, only your custom generators will be executed.

## Contributing

All contributions are welcome – especially:

- documentation,
- bug reports and issues,
- code contributions.

### Code

If you'd like to actively develop or help maintain this project then there are existing tests against which you can test the library with. Typically, this looks like

- `git clone git@github.com:contiamo/restful-react.git`
- `cd restful-react`
- `yarn install`
- `yarn test --watch`

From there, you should be able to start developing without problems.

### How to publish to npm

Just update the `version` in `package.json`!

As soon as your branch will be merged to master, a new npm version will be automatically published for you.

## `@without-cli` npm package

If for any reasons you don't want to use our CLI to generate restful-react components, we provide a `without-cli` version of the package.

Just `npm install restful-react@without-cli` to have this light version.

This version will follow `latest` but without the cli part (more details into `publish-without-cli.js`).

## Next Steps

We're actively developing this at Contiamo to meet our use cases as they arise. If you have a use case that you'd like to implement, do it! Open an issue, submit a Pull Request, have fun! We're friendly.
 readmeEtag: '"177e8cd0afe6c2003d24e3c25e1f3b437abba9d3"' readmeLastModified: Thu, 16 Jun 2022 12:35:09 GMT repositoryId: 139149148 description: >- A consistent, declarative way of interacting with RESTful backends, featuring code-generation from Swagger and OpenAPI specs 🔥 created: '2018-06-29T12:52:25Z' updated: '2026-01-26T12:50:12Z' language: TypeScript archived: true stars: 1863 watchers: 16 forks: 108 owner: contiamo logo: https://avatars.githubusercontent.com/u/862286?v=4 license: MIT repoEtag: '"32c54086e52e0578ecf6f40463596fd7a4246a729a47f8962472b52f6695d6d0"' repoLastModified: Mon, 26 Jan 2026 12:50:12 GMT foundInMaster: true id: 7fa6fa97d3fc13594a92cd5594506906 - source: https://openapi.tools/ name: NSwag category: SDK link: https://NSwag.org repository: https://github.com/ricosuter/nswag language: .NET source_description: OpenAPI toolchain for .NET, Web API and TypeScript v2: true v3: true repositoryMetadata: base64Readme: >- ## NSwag: The Swagger/OpenAPI toolchain for .NET, ASP.NET Core and TypeScript

NSwag | [NJsonSchema](http://njsonschema.org) | [Apimundo](https://apimundo.com) | [Namotion.Reflection](https://github.com/RicoSuter/Namotion.Reflection)

[![NuGet Version](https://img.shields.io/nuget/v/NSwag.Core.svg)](https://www.nuget.org/packages?q=NSwag)
[![npm](https://img.shields.io/npm/v/nswag.svg)](https://www.npmjs.com/package/nswag)
[![NuGet Version Preview](https://img.shields.io/nuget/vpre/NSwag.Core.svg)](https://www.nuget.org/packages?q=NSwag)
[![build](https://github.com/RicoSuter/NSwag/actions/workflows/build.yml/badge.svg)](https://github.com/RicoSuter/NSwag/actions/workflows/build.yml)
[![Discord](https://img.shields.io/badge/Discord-join%20chat-1dce73.svg)](https://discord.gg/BxQNy25WF6)
[![StackOverflow](https://img.shields.io/badge/questions-on%20StackOverflow-orange.svg?style=flat)](http://stackoverflow.com/questions/tagged/nswag)
[![Wiki](https://img.shields.io/badge/docs-in%20wiki-orange.svg?style=flat)](https://github.com/RicoSuter/nswag/wiki)
[![Backers on Open Collective](https://opencollective.com/NSwag/backers/badge.svg)](#backers) 
[![Sponsors on Open Collective](https://opencollective.com/NSwag/sponsors/badge.svg)](#sponsors)

:point_right: [**Announcing Apimundo:** An API documentation system based on NSwag and NJsonSchema](https://github.com/RicoSuter/NSwag/issues/3077) :point_left:

NSwag is a Swagger/OpenAPI 2.0 and 3.0 toolchain for .NET, .NET Core, Web API, ASP.NET Core, TypeScript (jQuery, AngularJS, Angular 2+, Aurelia, KnockoutJS and more) and other platforms, written in C#. The [OpenAPI/Swagger specification](https://github.com/OAI/OpenAPI-Specification) uses JSON and JSON Schema to describe a RESTful web API. The NSwag project provides tools to generate OpenAPI specifications from existing ASP.NET Web API controllers and client code from these OpenAPI specifications. 

The project combines the functionality of Swashbuckle (OpenAPI/Swagger generation) and AutoRest (client generation) in one toolchain (these two libs are not needed). This way a lot of incompatibilites can be avoided and features which are not well described by the OpenAPI specification or JSON Schema are better supported (e.g. [inheritance](https://github.com/NJsonSchema/NJsonSchema/wiki/Inheritance), [enum](https://github.com/NJsonSchema/NJsonSchema/wiki/Enums) and reference handling). The NSwag project heavily uses [NJsonSchema for .NET](http://njsonschema.org) for JSON Schema handling and C#/TypeScript class/interface generation. 

![ToolchainDiagram](assets/ToolchainDiagram.png) 

The project is developed and maintained by [Rico Suter](http://rsuter.com) and other contributors.

### Features

- [Generate Swagger 2.0 and OpenAPI 3.0 specifications from C# ASP.NET (Core) controllers](https://github.com/RicoSuter/NSwag/wiki/Middlewares)
- Serve the specs via ASP.NET (Core) middleware, optionally with [Swagger UI](https://github.com/swagger-api/swagger-ui) or [ReDoc](https://github.com/Rebilly/ReDoc)
- Generate C# or TypeScript clients/proxies from these specs
- Everything can be automated via CLI (distributed via NuGet tool or build target; or NPM)
- CLI configured via JSON file or NSwagStudio Windows UI

### Ways to use the toolchain

- Simple to use Windows GUI, [NSwagStudio](https://github.com/RicoSuter/NSwag/wiki/NSwagStudio)
- By using the [OpenAPI or OpenAPI UI OWIN and ASP.NET Core Middlewares](https://github.com/RicoSuter/NSwag/wiki/Middlewares) (also serves the [Swagger UI](https://github.com/swagger-api/swagger-ui)) (recommended)
- Via [command line](https://github.com/RicoSuter/NSwag/wiki/CommandLine) (Windows, Mac and Linux support through [Mono](http://www.mono-project.com/) or .NET Core console binary, also via [NPM package](https://www.npmjs.com/package/nswag))
- In your C# code, via [NuGet](https://www.nuget.org/packages?q=NSwag)
- In your [MSBuild targets](https://github.com/RicoSuter/NSwag/wiki/NSwag.MSBuild)
- With [ServiceProjectReference](https://github.com/RicoSuter/NSwag/wiki/ServiceProjectReference) tags in your .csproj (preview)
- In your [Azure V2 Functions](https://github.com/Jusas/NSwag.AzureFunctionsV2) (external project, might not use latest NSwag version)

### Tutorials

- [Add NSwag to your ASP.NET Core app](https://github.com/RicoSuter/NSwag/wiki/AspNetCore-Middleware)
- [Integrate the NSwag toolchain into your ASP.NET Web API project](https://blog.rsuter.com/nswag-tutorial-integrate-the-nswag-toolchain-into-your-asp-net-web-api-project/)
- [Generate an Angular TypeScript client from an existing ASP.NET Web API web assembly](https://blog.rsuter.com/nswag-tutorial-generate-an-angular-2-typescript-client-from-an-existing-asp-net-web-api-web-assembly/)
- [Video Tutorial: How to integrate NSwag into your ASP.NET Core Web API project (5 mins)](https://www.youtube.com/watch?v=lF9ZZ8p2Ciw)

### OpenAPI/Swagger Generators

- ASP.NET Web API assembly to OpenAPI (supports .NET Core)
    - [AspNetCoreOpenApiDocumentGenerator](https://github.com/RicoSuter/NSwag/wiki/AspNetCoreOpenApiDocumentGenerator)
    - [WebApiOpenApiDocumentGenerator](https://github.com/RicoSuter/NSwag/wiki/WebApiOpenApiDocumentGenerator)
        - Generates an OpenAPI specification for Web API controllers
    - [WebApiToOpenApiCommand](https://github.com/RicoSuter/NSwag/wiki/WebApiToOpenApiCommand)
        - Generates an OpenAPI specification for controllers in an external Web API assembly
        - [Also supports loading of .NET Core assemblies](https://github.com/RicoSuter/NSwag/wiki/Assembly-loading)
    - [TypesToOpenApiCommand](https://github.com/RicoSuter/NSwag/wiki/TypesToOpenApiCommand)
        - Generates an OpenAPI specification containing only types from .NET assemblies

### Code Generators

- **CSharp Client**
	- [CSharpClientGenerator](https://github.com/RicoSuter/NSwag/wiki/CSharpClientGenerator)
		- Generates C# clients from an OpenAPI specification
		- Generates POCOs or classes implementing [INotifyPropertyChanged](https://msdn.microsoft.com/en-us/library/system.componentmodel.inotifypropertychanged(v=vs.110).aspx) supporting DTOs
		- The generated clients can be used with full .NET, .NET Core, Xamarin and .NET Standard 1.4 in general
- **CSharp Controllers** (contract first/schema first development)
	- [CSharpControllerGenerator](https://github.com/RicoSuter/NSwag/wiki/CSharpControllerGenerator)
	    - Generates Web API Controllers based on an OpenAPI specification (ASP.NET Web API and ASP.NET Core)
- **TypeScript Client**
	- [TypeScriptClientGenerator](https://github.com/RicoSuter/NSwag/wiki/TypeScriptClientGenerator)
		- Generates TypeScript clients from an OpenAPI specification
		- Available templates/supported libraries: 
			- JQuery with Callbacks, `JQueryCallbacks`
			- JQuery with promises `JQueryPromises`
			- AngularJS using $http, `AngularJS`
			- Angular (v2+) using the http service, `Angular`
			- window.fetch API and ES6 promises, `Fetch` (use this template in your React/Redux app)
			- Aurelia using the HttpClient from aurelia-fetch-client, `Aurelia` (based on the Fetch template)
			- `Axios` (preview)
	    
### Downloads

- [Download latest **NSwagStudio MSI installer (NSwagStudio.msi)**](https://github.com/RicoSuter/NSwag/releases) (Windows Desktop application)
- [Download latest **NSwag command line tools** and NSwagStudio (NSwag.zip)](https://github.com/RicoSuter/NSwag/releases)

### NPM Packages

- [NSwag](https://www.npmjs.com/package/nswag): Command line tools (.NET and .NET Core) distributed as NPM package

### NuGet Packages

#### Specification

- **[NSwag.Core](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Core/versions/latest)**
    - The OpenAPI/Swagger reader and writer classes, see [OpenApiDocument](https://github.com/RicoSuter/NSwag/wiki/OpenApiDocument) (.NET Standard 1.0 / 2.0 and .NET 4.5)
- **[NSwag.Core.Yaml](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Core.Yaml/versions/latest)** (.NET Standard 1.3 / 2.0 and .NET 4.5)
    - Extensions to read and write YAML OpenAPI specifications
- **[NSwag.Annotations](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Annotations/versions/latest)** (.NET Standard 1.0 / 2.0 and .NET 4.5)
    - Attributes to decorate Web API controllers to control the OpenAPI generation

#### OpenAPI generation

- **[NSwag.Generation](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Generation/versions/latest/)** (.NET Standard 1.0 / 2.0 and .NET 4.5)
    - Classes to generate OpenAPI specifications
- **[NSwag.Generation.WebApi](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Generation.WebApi/versions/latest)** (.NET Standard 1.0 / 2.0 and .NET 4.5)
    - Classes to generate OpenAPI specifications from Web API controllers, see [WebApiOpenApiDocumentGenerator](https://github.com/RicoSuter/NSwag/wiki/WebApiOpenApiDocumentGenerator)
- **[NSwag.Generation.AspNetCore](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Generation.AspNetCore/versions/latest)** (.NET Standard 1.6 / 2.0 and .NET 4.5.1)
    - (Experimental) Classes to generate OpenAPI specifications from ASP.NET Core MVC controllers using the ApiExplorer

#### Code generation

- **[NSwag.CodeGeneration](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.CodeGeneration/versions/latest)** (.NET Standard 1.3 / 2.0 / .NET 4.5.1)
    - Base classes to generate clients from OpenAPI specifications
- **[NSwag.CodeGeneration.CSharp](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.CodeGeneration.CSharp/versions/latest)** (.NET Standard 1.3 and .NET 4.5.1)
    - Classes to generate C# clients from OpenAPI specifications, see [CSharpClientGenerator](https://github.com/RicoSuter/NSwag/wiki/CSharpClientGenerator) and [CSharpControllerGenerator](https://github.com/RicoSuter/NSwag/wiki/CSharpControllerGenerator)
- **[NSwag.CodeGeneration.TypeScript](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.CodeGeneration.TypeScript/versions/latest)** (.NET Standard 1.3 and .NET 4.5.1)
    - Classes to generate TypeScript clients from OpenAPI specifications, see [TypeScriptClientGenerator](https://github.com/RicoSuter/NSwag/wiki/TypeScriptClientGenerator)

#### ASP.NET and ASP.NET Core

- **[NSwag.AspNetCore](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.AspNetCore/versions/latest)** (.NET Standard 1.6 / 2.0 and .NET 4.5.1+)
- **[NSwag.AspNet.Owin](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.AspNet.Owin/versions/latest)** (.NET 4.5+)
    - [ASP.NET Core/OWIN middlewares](https://github.com/RicoSuter/NSwag/wiki/Middlewares) for serving OpenAPI specifications and Swagger UI
- **[NSwag.AspNet.WebApi](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.AspNet.WebApi/versions/latest)** (.NET 4.5+)
    - ASP.NET Web API filter which serializes exceptions ([JsonExceptionFilterAttribute](https://github.com/RicoSuter/NSwag/wiki/JsonExceptionFilterAttribute))

#### Frontends

- **[NSwag.AssemblyLoader](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.AssemblyLoader/versions/latest)** (.NET Standard 1.6 / 2.0 and .NET 4.5.1): 
    - Classes to load assemblies in an isolated AppDomain and generate OpenAPI specs from Web API controllers
- **[NSwag.Commands](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.Commands/versions/latest)** (.NET Standard 1.6 / 2.0 and .NET 4.5.1+): 
    - Commands for the command line tool implementations and UI
- **[NSwag.MSBuild](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.MSBuild/versions/latest)** (MSBuild .targets): 
    - Adds a .targets file to your Visual Studio project, so that you can run the NSwag command line tool in an MSBuild target, see [MSBuild](https://github.com/RicoSuter/NSwag/wiki/MSBuild)
- **[NSwag.ConsoleCore](https://apimundo.com/organizations/nuget-org/nuget-feeds/public/packages/NSwag.ConsoleCore/versions/latest)** (.NET Core 1.0, 1.1, 2.0, 2.1 and 2.2): 
    - Command line tool for .NET Core (`dotnet nswag`)
    - `<DotNetCliToolReference Include="NSwag.ConsoleCore" Version="..." />`
- **[NSwagStudio](https://chocolatey.org/packages/nswagstudio)** (Chocolatey, Windows): 
    - Package to install the NSwagStudio and command line tools via Chocolatey

#### CI NuGet Feed

https://www.myget.org/F/nswag/api/v3/index.json

The NuGet packages may require the **Microsoft.NETCore.Portable.Compatibility** package on .NET Core/UWP targets (if mscorlib is missing). 

![LayerDiagram](assets/LayerDiagram.png)

### Usage in C&#35;

To register the [middlewares](https://github.com/RicoSuter/NSwag/wiki/AspNetCore-Middleware) to generate an OpenAPI spec and render the UI, register NSwag in `Startup.cs`: 

```csharp
public class Startup
{
    ...

    public void ConfigureServices(IServiceCollection services)
    {
        services.AddOpenApiDocument(); // add OpenAPI v3 document
//      services.AddSwaggerDocument(); // add Swagger v2 document
    }

    public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
    {
        ...

        app.UseOpenApi(); // serve OpenAPI/Swagger documents
        app.UseSwaggerUi(); // serve Swagger UI
        app.UseReDoc(); // serve ReDoc UI
    }
}
```

The following code shows how to read an OpenAPI/Swagger specification and generate C# client classes to call the described web services: 
	
```cs
var document = await OpenApiDocument.FromFileAsync("openapi.json");
var clientSettings = new CSharpClientGeneratorSettings 
{
    ClassName = "MyClass",
    CSharpGeneratorSettings = 
    {
        Namespace = "MyNamespace"
    }
};

var clientGenerator = new CSharpClientGenerator(document, clientSettings);
var code = clientGenerator.GenerateFile();
```

Check out the [project Wiki](https://github.com/RicoSuter/NSwag/wiki) for more information.

### NSwagStudio

The generators can be used in a comfortable and simple Windows GUI called [NSwagStudio](https://github.com/RicoSuter/NSwag/wiki/NSwagStudio): 

[![](https://raw.githubusercontent.com/RicoSuter/NSwag/master/assets/screenshots/03_WebAPI_CSharp.png)](https://raw.githubusercontent.com/RicoSuter/NSwag/master/assets/screenshots/03_WebAPI_CSharp.png)

## Sponsors, support and consulting

Companies or individuals which paid a substantial amount for implementing, fixing issues, support or sponsoring are listed below. Thank you for supporting this project! You can also become a financial contributor:

- [Sponsor main contributor Rico Suter via GitHub](https://github.com/sponsors/RicoSuter)
- [Sponsor project via Open Collective for NSwag](https://opencollective.com/nswag)

Please contact [Rico Suter](https://rsuter.com) for paid consulting and support. 

## Contributors

This project exists thanks to all the people who contribute. [[Contribute](CONTRIBUTING.md)].
<a href="https://github.com/RicoSuter/NSwag/graphs/contributors"><img src="https://opencollective.com/NSwag/contributors.svg?width=890&button=false" /></a>

## Sponsors

Support this project by becoming a sponsor. Your logo will show up here with a link to your website. 

**Top sponsors:**

[![](https://images.gotowebinar.com/30dcc42d33945684be9cf66852300d1a)](https://picturepark.com)

**Sponsors:**

<a href="https://opencollective.com/NSwag/sponsor/0/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/0/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/1/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/1/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/2/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/2/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/3/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/3/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/4/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/4/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/5/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/5/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/6/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/6/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/7/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/7/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/8/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/8/avatar.svg"></a>
<a href="https://opencollective.com/NSwag/sponsor/9/website" target="_blank"><img src="https://opencollective.com/NSwag/sponsor/9/avatar.svg"></a>

## Backers

Thank you to all our backers!

<a href="https://opencollective.com/NSwag#backers" target="_blank"><img src="https://opencollective.com/NSwag/backers.svg?width=890"></a>
 
 readmeEtag: '"642164cf56d36dea29ee9425fb719d6e41d0c01c"' readmeLastModified: Wed, 17 Jul 2024 15:18:44 GMT repositoryId: 41822320 description: 'The Swagger/OpenAPI toolchain for .NET, ASP.NET Core and TypeScript. ' created: '2015-09-02T19:11:57Z' updated: '2026-02-05T16:28:23Z' language: C# archived: false stars: 7267 watchers: 159 forks: 1346 owner: RicoSuter logo: https://avatars.githubusercontent.com/u/2603405?v=4 license: MIT repoEtag: '"bc483187fe7dfa43fefae93a58240b0af28a1f3e2d084710c58caf4e05646dce"' repoLastModified: Thu, 05 Feb 2026 16:28:23 GMT foundInMaster: true id: 93636e95435e95198d7d15c9d507f0d3 - source: https://openapi.tools/ name: api-codegen-ts category: - SDK - Data Validators - Parsers link: https://www.npmjs.com/package/@nll/api-codegen-ts repository: https://github.com/nullpub/api-codegen-ts language: TypeScript source_description: >- Generates TypeScript models, response validators, and operation controllers from OpenAPI descriptions v2: true v3: true repositoryMetadata: base64Readme: >- IyBhcGktY29kZWdlbi10cwoKQSBnZW5lcmFsIHR5cGVzY3JpcHQgY29kZSBnZW5lcmF0aW9uIGxpYnJhcnkuIEN1cnJlbnRseSBpdCBjYW4gdGFrZSBzd2FnZ2VyIDIuMCBhbmQgb3BlbmFwaSAzLjAuMCsganNvbiBzcGVjcyBhbmQgdHVybiB0aGVtIGludG8gdHlwZXNjcmlwdCBjb250cm9sbGVycyB0aGF0IHZhbGlkYXRlIGFwaSByZXNwb25zZXMuCgojIyBJbnN0YWxsYXRpb24KCmBucG0gaSAtRCBAbmxsL2FwaS1jb2RlZ2VuLXRzYAoKIyMgQ29uZmlndXJhdGlvbgoKQWRkIGFuIGFwaUNvZGVnZW4gc2VjdGlvbiB0byBgcGFja2FnZS5qc29uYC4KCmBgYGpzb24KewogICJhcGlDb2RlZ2VuIjogewogICAgInNyYyI6ICIuL3BhdGgvdG8veW91ci9zd2FnZ2VyLXNwZWMuanNvbiIsCiAgICAiZHN0IjogIi4vcGF0aC90by95b3VyL2NvZGVnZW4vZGVzdGluYXRpb24iCiAgfQp9CmBgYAoKIyMgVXNhZ2UKCkFmdGVyIGNvbmZpZ3VyaW5nIGFwaS1jb2RlZ2VuLXRzIHlvdSBjYW4gZ2VuZXJhdGUgY29kZSBsaWtlIHNvOgoKYG5weCBAbmxsL2FwaS1jb2RlZ2VuLXRzYAoKQW5kIGl0IHdpbGwgb3V0cHV0IGl0cyBwcm9ncmVzcyBhbmQgb3V0cHV0IGNvZGUuCgojIyBBZHZhbmNlZCB1c2FnZQoKVGhpcyBjb2RlZ2VuIGxpYnJhcnkgYWJzdHJhY3RzIHBhcnNlcnMgYW5kIHByaW50ZXJzLiBBIHBhcnNlciBpcyByZXNwb25zaWJsZSBmb3IgcmVjZWl2aW5nIGFuIGlucHV0IHN0cmluZyAoYSBzcGVjaWZpY2F0aW9uKSBhbmQgZ2VuZXJhdGluZyBhbnkgaW50ZXJtZWRpYXRlIHJlcHJlc2VudGF0aW9uLiBDdXJyZW50bHksIHRoZXJlIGFyZSBwcmltaXRpdmUgc3dhZ2dlciAyLjAgYW5kIG9wZW5hcGkgMy4wLjAgcGFyc2Vycy4gQSBwcmludGVyIHRha2VzIGFuIGludGVybWVkaWF0ZSByZXByZXNlbnRhdGlvbiAoaXQgbXVzdCBtYXRjaCB3aGF0ZXZlciB0aGUgcGFyc2VyIG91dHB1dHMpIGFuZCByZXR1cm5zIHRoZSBmaWxlIHN0cmluZ3MgdG8gd3JpdGUgdG8gZGlzay4gVGhlIGNvcmUgZnVuY3Rpb25hbGl0eSBvZiBhcGktY29kZWdlbi10cyBpcyBtYW5hZ2luZyB0aGlzIHdvcmtmbG93LCBpdCBoYW5kbGVzIHRoZSByZWFkaW5nIG9mIHRoZSBzb3VyY2UgZmlsZXMgYW5kIHRoZSBwcmludGluZyBvZiB0aGUgZ2VuZXJhdGVkIGZpbGVzLgoKIyMgU3VwcG9ydAoKVGhpcyBpcyBteSBmaXJzdCBhdHRlbXB0IGF0IGEgY29kZWdlbiBsaWJyYXJ5LiBJZiB5b3UgZmluZCBidWdzIG9yIHNvbWUgc2V0IG9mIGZlYXR1cmVzIHRoYXQgYXJlbid0IHN1cHBvcnRlZCBwbGVhc2UgY3JlYXRlIGEgZ2l0aHViIGlzc3VlLiBUaGFua3MhCg== readmeEtag: '"406840cedd6d31bf4b9ac20d6743f83bec5c3f59"' readmeLastModified: Wed, 28 Aug 2019 04:48:19 GMT repositoryId: 190281420 description: >- A generalized codegen library by default setup to go from swagger 2.0 to typescript. created: '2019-06-04T21:23:53Z' updated: '2024-11-19T13:06:48Z' language: TypeScript archived: true stars: 6 watchers: 0 forks: 1 owner: nullpub logo: https://avatars.githubusercontent.com/u/22160158?v=4 repoEtag: '"9de8a8fa7ebb7e6b4868e8bef59f6377ab2b5792d49109c60aab7e348a2d0680"' repoLastModified: Tue, 19 Nov 2024 13:06:48 GMT foundInMaster: true id: 753c3c0a8c4ef510d178b3a4a5ff35ca - source: - https://openapi.tools/ - openapi3 tags name: Swagger-Codegen category: Code Generators link: https://swagger.io/swagger-codegen/ repository: https://github.com/swagger-api/swagger-codegen language: Java source_description: >- Swagger Codegen enables generating server stubs and client SDKs for APIs described in OpenAPI v2: true v3: true repositoryMetadata: base64Readme: >- # Swagger Codegen
![Swagger Codegen Logo](./docs/Swagger-Codegen-Logo.png)

This is the Swagger Codegen project, which allows generation of API client libraries (SDK generation), server stubs and documentation automatically given an [OpenAPI Spec](https://github.com/OAI/OpenAPI-Specification).

💚 If you would like to contribute, please refer to [guidelines](https://github.com/swagger-api/swagger-codegen/blob/master/CONTRIBUTING.md) and a list of [open tasks](https://github.com/swagger-api/swagger-codegen/issues?q=is%3Aopen+is%3Aissue+label%3A%22help+wanted%22).💚

📔 For more information, please refer to the [Wiki page](https://github.com/swagger-api/swagger-codegen/wiki) and [FAQ](https://github.com/swagger-api/swagger-codegen/wiki/FAQ) 📔

⚠️ If the OpenAPI Description or Swagger file is obtained from an untrusted source, please make sure you've reviewed the artefact before using Swagger Codegen to generate the API client, server stub or documentation as [code injection](https://en.wikipedia.org/wiki/Code_injection) may occur. ⚠️

## Versioning

> ⚠️ this document refers to version 2.X, check [here](https://github.com/swagger-api/swagger-codegen/tree/3.0.0) for 3.X.

Both **2.X** and **3.X** version lines of Swagger Codegen are available and are independently maintained.

**NOTES:**

- Versions 2.X (`io.swagger`) and 3.X (`io.swagger.codegen.v3`) have **different** group ids.
- OpenAPI 3.0.X is supported from the 3.X version only.

For full versioning information, please refer to the [versioning documentation](./docs/versioning.md).

## Supported Languages and Frameworks

Currently, the following languages/frameworks are supported:

- **API clients**: **ActionScript**, **Ada**, **Apex**, **Bash**, **C#** (.net 2.0, 3.5 or later), **C++** (cpprest, Qt5, Tizen), **Clojure**, **Dart**, **Elixir**, **Elm**, **Eiffel**, **Erlang**, **Go**, **Groovy**, **Haskell** (http-client, Servant), **Java** (Jersey1.x, Jersey2.x, OkHttp, Retrofit1.x, Retrofit2.x, Feign, RestTemplate, RESTEasy, Vertx, Google API Client Library for Java, Rest-assured), **Kotlin**, **Lua**, **Node.js** (ES5, ES6, AngularJS with Google Closure Compiler annotations) **Objective-C**, **Perl**, **PHP**, **PowerShell**, **Python**, **R**, **Ruby**, **Rust** (rust, rust-server), **Scala** (akka, http4s, swagger-async-httpclient), **Swift** (2.x, 3.x, 4.x, 5.x), **Typescript** (Angular1.x, Angular2.x, Fetch, jQuery, Node)
- **Server stubs**: **Ada**, **C#** (ASP.NET Core, NancyFx), **C++** (Pistache, Restbed), **Erlang**, **Go**, **Haskell** (Servant), **Java** (MSF4J, Spring, Undertow, JAX-RS: CDI, CXF, Inflector, RestEasy, Play Framework, [PKMST](https://github.com/ProKarma-Inc/pkmst-getting-started-examples)), **Kotlin**, **PHP** (Lumen, Slim, Silex, [Symfony](https://symfony.com/), [Zend Expressive](https://github.com/zendframework/zend-expressive)), **Python** (Flask), **NodeJS**, **Ruby** (Sinatra, Rails5), **Rust** (rust-server), **Scala** ([Finch](https://github.com/finagle/finch), [Lagom](https://github.com/lagom/lagom), Scalatra)
- **API documentation generators**: **HTML**, **Confluence Wiki**
- **Configuration files**: [**Apache2**](https://httpd.apache.org/)
- **Others**: **JMeter**

Check out [OpenAPI-Spec](https://github.com/OAI/OpenAPI-Specification) for additional information about the OpenAPI project.

## Table of contents

- [Versioning](#versioning)
- [Compatibility](#compatibility)
- [Getting Started](#getting-started)
- [Generators](#generators)
  - [To generate a sample client library](#to-generate-a-sample-client-library)
  - [Generating libraries from your server](#generating-libraries-from-your-server)
  - [Validating your OpenAPI Spec](#validating-your-openapi-spec)
  - [Generating dynamic html api documentation](#generating-dynamic-html-api-documentation)
  - [Generating static html api documentation](#generating-static-html-api-documentation)
- [Workflow Integration](#workflow-integration)
- [Online Generators](#online-generators)
- [Contribution](#contributing)
- [Swagger Codegen Core Team](#swagger-codegen-core-team)


## Compatibility

The OpenAPI Specification has undergone 3 revisions since initial creation in 2010.  The **current stable** versions of Swagger Codegen project have the following compatibilities with the OpenAPI Specification:

Swagger Codegen Version    | Release Date | Swagger / OpenAPI Spec compatibility | Notes
-------------------------- |--------------| -------------------------- | -----
[3.0.71](https://github.com/swagger-api/swagger-codegen/releases/tag/v3.0.71) (**current stable**) | 2025-07-03   | 1.0, 1.1, 1.2, 2.0, 3.0   | [tag v3.0.71](https://github.com/swagger-api/swagger-codegen/tree/v3.0.71)
[2.4.46](https://github.com/swagger-api/swagger-codegen/releases/tag/v2.4.46) (**current stable**) | 2025-06-30   | 1.0, 1.1, 1.2, 2.0   | [tag v2.4.46](https://github.com/swagger-api/swagger-codegen/tree/v2.4.46)

💁 Here's also an overview of what's coming around the corner:

Swagger Codegen Version    | Release Date | Swagger / OpenAPI Spec compatibility | Notes
-------------------------- |--------------| -------------------------- | -----
3.0.72-SNAPSHOT (current 3.0.0, upcoming minor release) [SNAPSHOT](https://central.sonatype.com/service/rest/repository/browse/maven-snapshots/io/swagger/codegen/v3/swagger-codegen-cli/3.0.72-SNAPSHOT/)| TBD          | 1.0, 1.1, 1.2, 2.0, 3.0 | Minor release
2.4.47-SNAPSHOT (current master, upcoming minor release) [SNAPSHOT](https://central.sonatype.com/service/rest/repository/browse/maven-snapshots/io/swagger/swagger-codegen-cli/2.4.47-SNAPSHOT/)| TBD          | 1.0, 1.1, 1.2, 2.0   | Minor release

For detailed breakdown of all versions, please see the [full compatibility listing](./docs/compatibility.md).

### Getting Started

To get up and running with Swagger Codegen, check out the following guides and instructions:

- [Prerequisites](./docs/prerequisites.md)
- [Building](./docs/building.md)
- [Using Docker](./docs/docker.md)

Once you've your environment setup, you're ready to start generating clients and/or servers.

As a quick example, to generate a PHP client for https://petstore.swagger.io/v2/swagger.json, you can run the following:

```sh
git clone https://github.com/swagger-api/swagger-codegen
cd swagger-codegen
mvn clean package
java -jar modules/swagger-codegen-cli/target/swagger-codegen-cli.jar generate \
   -i https://petstore.swagger.io/v2/swagger.json \
   -l php \
   -o /var/tmp/php_api_client
```

**Note:** if you're on Windows, replace the last command with

```sh
java -jar modules\swagger-codegen-cli\target\swagger-codegen-cli.jar generate -i https://petstore.swagger.io/v2/swagger.json -l php -o c:\temp\php_api_client
```

You can also download the JAR (latest release) directly from [maven.org](https://repo1.maven.org/maven2/io/swagger/swagger-codegen-cli/2.4.46/swagger-codegen-cli-2.4.46.jar)

To get a list of **general** options available, you can run the following:

```sh
java -jar modules/swagger-codegen-cli/target/swagger-codegen-cli.jar help generate
```

To get a list of PHP specified options (which can be passed to the generator with a config file via the `-c` option), please run

```sh
java -jar modules/swagger-codegen-cli/target/swagger-codegen-cli.jar config-help -l php
```

## Generators

### To generate a sample client library

You can build a client against the swagger sample [petstore](https://petstore.swagger.io) API as follows:

```sh
./bin/java-petstore.sh
```

(On Windows, run `.\bin\windows\java-petstore.bat` instead)

This will run the generator with this command:

```sh
java -jar modules/swagger-codegen-cli/target/swagger-codegen-cli.jar generate \
  -i https://petstore.swagger.io/v2/swagger.json \
  -l java \
  -o samples/client/petstore/java
```

with a number of options. You can get the options with the `help generate` command (below only shows partial results):

```text
NAME
        swagger-codegen-cli generate - Generate code with chosen lang

SYNOPSIS
        swagger-codegen-cli generate
                [(-a <authorization> | --auth <authorization>)]
                [--additional-properties <additional properties>...]
                [--api-package <api package>] [--artifact-id <artifact id>]
                [--artifact-version <artifact version>]
                [(-c <configuration file> | --config <configuration file>)]
                [-D <system properties>...] [--git-repo-id <git repo id>]
                [--git-user-id <git user id>] [--group-id <group id>]
                [--http-user-agent <http user agent>]
                (-i <spec file> | --input-spec <spec file>)
                [--ignore-file-override <ignore file override location>]
                [--import-mappings <import mappings>...]
                [--instantiation-types <instantiation types>...]
                [--invoker-package <invoker package>]
                (-l <language> | --lang <language>)
                [--language-specific-primitives <language specific primitives>...]
                [--library <library>] [--model-name-prefix <model name prefix>]
                [--model-name-suffix <model name suffix>]
                [--model-package <model package>]
                [(-o <output directory> | --output <output directory>)]
                [--release-note <release note>] [--remove-operation-id-prefix]
                [--reserved-words-mappings <reserved word mappings>...]
                [(-s | --skip-overwrite)]
                [(-t <template directory> | --template-dir <template directory>)]
                [--type-mappings <type mappings>...] [(-v | --verbose)]

OPTIONS
        -a <authorization>, --auth <authorization>
            adds authorization headers when fetching the swagger definitions
            remotely. Pass in a URL-encoded string of name:header with a comma
            separating multiple values

...... (results omitted)

        -v, --verbose
            verbose mode

```

You can then compile and run the client, as well as unit tests against it:

```sh
cd samples/client/petstore/java
mvn package
```

Other languages have petstore samples, too:
```sh
./bin/android-petstore.sh
./bin/java-petstore.sh
./bin/objc-petstore.sh
```

### Generating libraries from your server
It's just as easy--just use the `-i` flag to point to either a server or file.

🔧 Swagger Codegen comes with a tonne of flexibility to support your code generation preferences. Checkout the [generators documentation](./docs/generators.md) which takes you through some of the possibilities as well as showcasing how to generate from local files and ignore file formats.

### Selective generation

You may not want to generate *all* models in your project.  Likewise you may want just one or two apis to be written. If that's the case check the [selective generation](./docs/generation-selective.md) instructions.


### Advanced Generator Customization

There are different aspects of customizing the code generator beyond just creating or modifying templates.  Each language has a supporting configuration file to handle different type mappings, or bring your own models. For more information check out the [advanced configuration docs](./docs/generators-configuration.md).

### Validating your OpenAPI Spec

You have options.  The easiest is to use our [online validator](https://github.com/swagger-api/validator-badge) which not only will let you validate your spec, but with the debug flag, you can see what's wrong with your spec.  For example check out [Swagger Validator](http://online.swagger.io/validator/debug?url=https://petstore.swagger.io/v2/swagger.json).

If you want to have validation directly on your own machine, then [Spectral](https://stoplight.io/open-source/spectral) is an excellent option.

### Generating dynamic html api documentation

To do so, just use the `-l dynamic-html` flag when reading a spec file.  This creates HTML documentation that is available as a single-page application with AJAX.  To view the documentation:

```sh
cd samples/dynamic-html/
npm install
node .
```

Which launches a node.js server so the AJAX calls have a place to go.

### Generating static html api documentation

To do so, just use the `-l html` flag when reading a spec file.  This creates a single, simple HTML file with embedded css so you can ship it as an email attachment, or load it from your filesystem:

```sh
cd samples/html/
open index.html
```

## Workflow Integration

It's possible to leverage Swagger Codegen directly within your preferred CI/CD workflows to streamline your auto-generation requirements. Check out the [workflows integration](./docs/workflow-integration.md) guide to see information on our Maven, Gradle, and GitHub integration options. 🚀

## Online Generators

If you don't want to run and host your own code generation instance, check our our [online generators](./docs/online-generators.md) information.

## Contributing

Please refer to this [page](https://github.com/swagger-api/swagger-codegen/blob/master/CONTRIBUTING.md)

## Swagger Codegen Core Team

Swagger Codegen core team members are contributors who have been making significant contributions (review issues, fix bugs, make enhancements, etc) to the project on a regular basis.

Members of the core team shoulder the following responsibilities:

- Provides guidance and direction to other users
- Reviews pull requests and issues
- Improves the generator by making enhancements, fixing bugs or updating documentations
- Sets the technical direction of the generator

## Security contact

Please disclose any security-related issues or vulnerabilities by emailing [security@swagger.io](mailto:security@swagger.io), instead of using the public issue tracker.

## License information on Generated Code

The Swagger Codegen project is intended as a benefit for users of the Swagger / Open API Specification.  The project itself has the [License](./LICENSE) as specified.  In addition, please understand the following points:

- The templates included with this project are subject to the [License](./LICENSE).
- Generated code is intentionally _not_ subject to the parent project license

When code is generated from this project, it shall be considered **AS IS** and owned by the user of the software.  There are no warranties--expressed or implied--for generated code.  You can do what you wish with it, and once generated, the code is your responsibility and subject to the licensing terms that you deem appropriate.

## Thank You

💚💚💚 We'd like to give a big shout out to all those who've contributed to Swagger Codegen, be that in raising issues, fixing bugs, [authoring templates](./docs/template-creators.md), or crafting useful [content](./docs/public-content.md) for others to benefit from. 💚💚💚
 readmeEtag: '"c5c0eb28c6157dee157f929125a23b9ddfc057e6"' readmeLastModified: Fri, 04 Jul 2025 13:44:09 GMT repositoryId: 2006876 description: >- swagger-codegen contains a template-driven engine to generate documentation, API clients and server stubs in different languages by parsing your OpenAPI / Swagger definition. created: '2011-07-06T14:26:48Z' updated: '2026-02-05T19:25:26Z' language: Mustache archived: false stars: 17694 watchers: 407 forks: 6034 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"66407fef137d53f5d5f72d0ed5d0dc8fb3215ec3f8d4b2aea4efc81d6ca35931"' repoLastModified: Thu, 05 Feb 2026 19:25:26 GMT foundInMaster: true id: eda1e66e1b7805496e42f57851791a32 - source: https://openapi.tools/ name: Azure AutoRest category: SDK repository: https://github.com/azure/autorest/ language: TypeScript source_description: >- Generates client libraries for accessing RESTful web services from an OpenAPI document. Supports C#, PowerShell, Go, Java, Node.js, TypeScript, Python, and Ruby. v2: true v3: true repositoryMetadata: base64Readme: >- # <img align="center" src="./docs/images/logo.png"> AutoRest

The **AutoRest** tool generates client libraries for accessing RESTful web services. Input to _AutoRest_ is a spec that describes the REST API using the [OpenAPI Specification](https://github.com/OAI/OpenAPI-Specification) format.

[Release notes](./docs/releases)

## Packages

| Name                                            | Changelog                       | Latest                                                       | Next                                                              |
| ----------------------------------------------- | ------------------------------- | ------------------------------------------------------------ | ----------------------------------------------------------------- |
| Core functionality                              |
| [autorest][autorest_src]                        | [Changelog][autorest_chg]       | ![](https://img.shields.io/npm/v/autorest)                   | ![](https://img.shields.io/npm/v/autorest/next)                   |
| [@autorest/core][core_src]                      | [Changelog][core_chg]           | ![](https://img.shields.io/npm/v/@autorest/core)             | ![](https://img.shields.io/npm/v/@autorest/core/next)             |
| [@autorest/modelerfour][modelerfour_src]        | [Changelog][modelerfour_chg]    | ![](https://img.shields.io/npm/v/@autorest/modelerfour)      | ![](https://img.shields.io/npm/v/@autorest/modelerfour/next)      |
| Language generators                             |
| [@autorest/csharp][csharp_src]                  | [Changelog][csharp_chg]         | ![](https://img.shields.io/npm/v/@autorest/csharp)           |                                                                   |
| [@autorest/go][go_src]                          | [Changelog][go_chg]             | ![](https://img.shields.io/npm/v/@autorest/go)               |                                                                   |
| [@autorest/java][java_src]                      | [Changelog][java_chg]           | ![](https://img.shields.io/npm/v/@autorest/java)             |                                                                   |
| [@autorest/powershell][powershell_src]          | [Changelog][powershell_chg]     | ![](https://img.shields.io/npm/v/@autorest/powershell)       |                                                                   |
| [@autorest/python][python_src]                  | [Changelog][python_chg]         | ![](https://img.shields.io/npm/v/@autorest/python)           |                                                                   |
| [@autorest/swift][swift_src]                    | [Changelog][swift_chg]          | ![](https://img.shields.io/npm/v/@autorest/swift)            |                                                                   |
| [@autorest/typescript][typescript_src]          | [Changelog][typescript_chg]     | ![](https://img.shields.io/npm/v/@autorest/typescript)       |                                                                   |
| Internal packages                               |
| [@autorest/codemodel][codemodel_src]            | [Changelog][codemodel_chg]      | ![](https://img.shields.io/npm/v/@autorest/codemodel)        | ![](https://img.shields.io/npm/v/@autorest/codemodel/next)        |
| [@autorest/common][common_src]                  | [Changelog][common_chg]         | ![](https://img.shields.io/npm/v/@autorest/common)           | ![](https://img.shields.io/npm/v/@autorest/common/next)           |
| [@autorest/configuration][configuration_src]    | [Changelog][configuration_chg]  | ![](https://img.shields.io/npm/v/@autorest/configuration)    | ![](https://img.shields.io/npm/v/@autorest/configuration/next)    |
| [@autorest/extension-base][extension_base_src]  | [Changelog][extension_base_chg] | ![](https://img.shields.io/npm/v/@autorest/extension-base)   | ![](https://img.shields.io/npm/v/@autorest/extension-base/next)   |
| [@azure-tools/extension][extension_src]         | [Changelog][extension_chg]      | ![](https://img.shields.io/npm/v/@azure-tools/extension)     | ![](https://img.shields.io/npm/v/@azure-tools/extension/next)     |
| [@azure-tools/codegen][codegen_src]             | [Changelog][codemodel_chg]      | ![](https://img.shields.io/npm/v/@azure-tools/codegen)       | ![](https://img.shields.io/npm/v/@azure-tools/codegen/next)       |
| [@azure-tools/openapi][openapi_src]             | [Changelog][openapi_chg]        | ![](https://img.shields.io/npm/v/@azure-tools/openapi)       | ![](https://img.shields.io/npm/v/@azure-tools/openapi/next)       |
| [@azure-tools/deduplication][deduplication_src] | [Changelog][deduplication_chg]  | ![](https://img.shields.io/npm/v/@azure-tools/deduplication) | ![](https://img.shields.io/npm/v/@azure-tools/deduplication/next) |
| [@azure-tools/datastore][datastore_src]         | [Changelog][datastore_chg]      | ![](https://img.shields.io/npm/v/@azure-tools/datastore)     | ![](https://img.shields.io/npm/v/@azure-tools/datastore/next)     |
| [@azure-tools/oai2-to-oai3][oai2-to-oai3_src]   | [Changelog][oai2-to-oai3_chg]   | ![](https://img.shields.io/npm/v/@azure-tools/oai2-to-oai3)  | ![](https://img.shields.io/npm/v/@azure-tools/oai2-to-oai3/next)  |
| [@azure-tools/jsonschema][jsonschema_src]       | [Changelog][jsonschema_chg]     | ![](https://img.shields.io/npm/v/@azure-tools/jsonschema)    | ![](https://img.shields.io/npm/v/@azure-tools/jsonschema/next)    |

[autorest_src]: packages/apps/autorest
[core_src]: packages/extensions/core
[modelerfour_src]: packages/extensions/modelerfour
[csharp_src]: https://github.com/Azure/autorest.csharp
[python_src]: https://github.com/Azure/autorest.python
[go_src]: https://github.com/Azure/autorest.go
[java_src]: https://github.com/Azure/autorest.java
[swift_src]: https://github.com/Azure/autorest.swift
[typescript_src]: https://github.com/Azure/autorest.typescript
[powershell_src]: https://github.com/Azure/autorest.powershell
[codemodel_src]: packages/libs/codemodel
[common_src]: packages/libs/common
[configuration_src]: packages/libs/configuration
[extension_base_src]: packages/libs/extension-base
[oai2-to-oai3_src]: packages/libs/oai2-to-oai3_src
[extension_src]: packages/libs/extension
[codegen_src]: packages/libs/codegen
[openapi_src]: packages/libs/openapi
[deduplication_src]: packages/libs/deduplication
[datastore_src]: packages/libs/datastore
[jsonschema_src]: packages/libs/oai2-to-oai3
[autorest_chg]: packages/apps/autorest/CHANGELOG.md
[core_chg]: packages/extensions/core/CHANGELOG.md
[modelerfour_chg]: packages/extensions/modelerfour/CHANGELOG.md
[csharp_chg]: https://github.com/Azure/autorest.csharp
[python_chg]: https://github.com/Azure/autorest.python/blob/main/packages/autorest.python/ChangeLog.md
[go_chg]: https://github.com/Azure/autorest.go
[java_chg]: https://github.com/Azure/autorest.java/releases
[swift_chg]: https://github.com/Azure/autorest.swift
[typescript_chg]: https://github.com/Azure/autorest.typescript
[powershell_chg]: https://github.com/Azure/autorest.powershell/releases
[codemodel_chg]: packages/libs/codemodel/CHANGELOG.md
[common_chg]: packages/libs/common/CHANGELOG.md
[configuration_chg]: packages/libs/configuration/CHANGELOG.md
[extension_base_chg]: packages/libs/extension-base/CHANGELOG.md
[extension_chg]: packages/libs/extension/CHANGELOG.md
[codegen_chg]: packages/libs/codegen/CHANGELOG.md
[openapi_chg]: packages/libs/openapi/CHANGELOG.md
[deduplication_chg]: packages/libs/deduplication/CHANGELOG.md
[datastore_chg]: packages/libs/datastore/CHANGELOG.md
[oai2-to-oai3_chg]: packages/libs/oai2-to-oai3/CHANGELOG.md
[jsonschema_chg]: packages/libs/jsonschema/CHANGELOG.md

## Support Policy

AutoRest is an open source tool -- if you need assistance, first check the documentation. If you find a bug or need some help, feel free to submit an [issue](https://github.com/Azure/autorest/issues)

## Getting Started using AutoRest ![image](./docs/images/normal.png)

View our [docs readme][docs_readme] as a starting point to find both general information and language-generator specific information

## Contributing

### Contributing guide

Check our [internal developer docs](./docs/developer/readme.md) to learn about our development process, how to propose bugfixes and improvements, and how to build and test your changes to Autorest.

### Code of Conduct

This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/). For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments.

<!--LINKS-->

[docs_readme]: docs/readme.md
 readmeEtag: '"f205b0054148c6362d9b9a5d930c27da17b36eca"' readmeLastModified: Tue, 13 Aug 2024 07:26:42 GMT repositoryId: 31089461 description: >- OpenAPI (f.k.a Swagger) Specification code generator. Supports C#, PowerShell, Go, Java, Node.js, TypeScript, Python created: '2015-02-20T23:02:54Z' updated: '2026-02-05T03:06:15Z' language: TypeSpec archived: false stars: 4801 watchers: 964 forks: 737 owner: Azure logo: https://avatars.githubusercontent.com/u/6844498?v=4 license: MIT repoEtag: '"365392cb2abe4bd2aabbfd42a3a1da9cb8187795b3613fe8162b69a5ea9e2827"' repoLastModified: Thu, 05 Feb 2026 03:06:15 GMT foundInMaster: true id: 648d92bca90bf58cd74407e0bcad2936 - source: https://openapi.tools/ name: spring-openapi homepage: https://github.com/jrcodeza/spring-openapi language: Java source_description: >- OpenAPI v3 generator for Java Spring. Includes also client generation. Supports inheritance with discriminators, Jackson annotations and custom interceptors. category: SDK repository: https://github.com/jrcodeza/spring-openapi v3: true repositoryMetadata: base64Readme: >- # spring-openapi (OpenAPI 3 generator)
![Java CI](https://github.com/jrcodeza/spring-openapi/workflows/Java%20CI/badge.svg)

Spring Boot OpenAPI 3 generator. It scans provided packages 
(model and controller) and generates based on reflection, javax validation
and spring annotations the OpenAPI 3 json. It is able to handle also
inheritance using OpenAPI 3 discriminator possibilities. The inheritance
generation is achieved using jackson annotations.

Generator allows you to also intercept schema, schema field, operation
and operation parameter process, so you can include your own behavior
to enrich already mapped sections (e.g. include own annotation behavior
to descriptions etc.).

So far there is core generator functionality 
**spring-openapi-schema-generator** and maven plugin
**spring-openapi-schema-generator-plugin** available.Soon there will be 
also java client generator finalized, because current tooling is still 
not able to generate client with full OpenAPI 3 discriminator features
(jackson annotations).

There is also short article describing usage of spring-openapi generator here: 
https://medium.com/@remenec.jakub/openapi-3-spec-and-client-generation-on-java-spring-application-38a9ba5a2932.

## Current version
**1.4.6**

Now generator supports also OpenAPI version 2 generation.

Release notes: https://github.com/jrcodeza/spring-openapi/blob/master/CHANGELOG.md 

1.2.1 - to ensure compatiblity with swagger ui for bigger inheritance structures.
This version models inheritance using allOf only if discriminator (JsonSubTypes) is found
in inheritance hierarchy. If not, it takes all attributes from parent classes
and adds them to the current one. See tests for more info.

## Example project
You can check this repository https://github.com/jrcodeza/spring-openapi-example for an example
usage. It shows spec to code and also code to spec generation using maven plugins. It contains
simple REST resources, model (including inheritance), interceptors and examples resolver.

## OpenAPI UI interpreter
If you need to display OpenAPI v3 spec in browser you can have a look also on **oas-ui**
plugin https://github.com/vahanito/oas-ui (which was developed together with spring-openapi) . 
It supports searching of components and resources. Correctly displays inheritance also 
with discriminator info. You can use it in react as plugin or also as standalone bundle js. 

## Annotations
### @OpenApiExample
You can define it on class, method, parameter or field. There are 2 main use cases:

- Providing the example value **in place**
```java
@OpenApiExample(name = "StandardExample", value = "standardExampleValue")
```
- providing **key** to example. You have to manage that key is handled by examples resolver
```java
@OpenApiExample(name = "KeyExample", key = "KEY")
```
Example resolver is a simple interface:
```java
public interface OpenApiExampleResolver {
	
    String resolveExample(String exampleKey);

}
```
The **exampleKey** parameter is a key for which example value should be provided by resolver. It can be 
for example a file name with example request - then goal of the resolver is to read file content and provide
it. 

It's possible to define more examples for one element by wrapping @OpenApiExample annotation to **@OpenApiExamples**
one. Name of the example becomes important in this case, because it will be stored to spec as example name.

### @OpenApiIgnore
Element annotated with this annotation will not be generated to spec.

### @Response
Sometimes you need to specify method (operation) response in your own way. It can 
be necessary when you have several return statuses or response types. Also when 
you want to specify response headers. For this purpose you can use this annotation.

It's also possible to wrap multiple @Response annotations to **@Responses** annotations. E.g.:
```java
@Responses({
	@Response(responseCode = 201, description = "Created", responseBody = ValidationDummy.class,
			headers = @Header(name = "SomeHeader", description = "TestHeader")),
	@Response(responseCode = 200, description = "Replaced", responseBody = ValidationDummy.class)
}) 
```
@Header annotation is also custom spring-openapi annotation.

## Generate spec from code
### Runtime usage
Include dependency

```java
<dependency>
    <groupId>com.github.jrcodeza</groupId>
    <artifactId>spring-openapi-schema-generator</artifactId>
    <version>1.4.6</version>
</dependency>
```

Instantiate **OpenAPIGenerator**

```java
OpenAPIGenerator openAPIGenerator = new OpenAPIGenerator(
                singletonList("org.spring.openapi.schema.generator.plugin.model.*"),
                singletonList("org.spring.openapi.schema.generator.plugin.controller.*"),
                createTestInfo()
        );
```
The first parameter is model package and the second is controller package. It
is possible to define multiple packages for each. The third parameter
is OpenAPI basic info like project title, description, version etc. It is
required in order to create valid OpenAPI 3 output.
Additionally you can define interceptors
* **OperationInterceptor** - executed after controller method is mapped
* **OperationParameterInterceptor** - executed after each method parameter is mapped
* **RequestBodyInterceptor** - executed after RequestBody annotated parameter of a method is mapped
* **SchemaFieldInterceptor** - executed after class field is mapped
* **SchemaInterceptor** - executed after each class is mapped

You can add these interceptors either to constructor after info parameter or
by calling add + interceptorName method on OpenAPIGenerator class. E.g.:

```java
openAPIGenerator.addOperationInterceptor(operationInterceptor);
openAPIGenerator.addOperationParameterInterceptor(operationParameterInterceptor);
openAPIGenerator.addRequestBodyInterceptor(requestBodyInterceptor);
openAPIGenerator.addSchemaFieldInterceptor(schemaFieldInterceptor);
openAPIGenerator.addSchemaInterceptor(schemaInterceptor);
```

You can define your own interceptor as follows:
```java
package com.github.jrcodeza.schema.generator.plugin.interceptors;

import java.lang.reflect.Method;

import com.github.jrcodeza.schema.generator.interceptors.OperationInterceptor;

import io.swagger.v3.oas.models.Operation;

public class TestOperationInterceptor implements OperationInterceptor {

	@Override
	public void intercept(Method method, Operation transformedOperation) {
		transformedOperation.setSummary("Interceptor summary");
	}

}
```

It's possible to define also global headers which are applicable for all resources.
```java
openAPIGenerator.addGlobalHeader("Test-Global-Header", "Some desc", false);
```

OpenAPIGenerator method **generate** can also take OpenApiGeneratorConfig as parameter.
In this config you can define if generation of examples is enabled. You can also define
example resolver, which can be useful if for example you have bigger examples of POST body requests
stored in files. You can define it as follows:

```java
OpenApiGeneratorConfigBuilder.defaultConfig()
                        .withGenerateExamples(true)
                        .withOpenApiExampleResolver(createExampleResolver())
                        .build()
```

Finally when you want to **generate OpenAPI 3 spec** you have to execute
generate method on OpenAPIGenerator instance.
```java
OpenAPI openAPI = openAPIGenerator.generate();
```
and you will receive OpenAPI object (from swagger API) which can be serialized
to string like this
```java
ObjectMapper objectMapper = new ObjectMapper();
objectMapper.setSerializationInclusion(JsonInclude.Include.NON_NULL);
try {
    String generated = objectMapper.writeValueAsString(openAPI);
    JSONAssert.assertEquals(getResourceFileAsString(), generated, false);
} catch (JsonProcessingException | JSONException e) {
    e.printStackTrace();
}
```

### Maven plugin usage
Maven plugin wraps before mentioned functionality into maven plugin.

```xml
<build>
    <plugins>
        <plugin>
            <groupId>com.github.jrcodeza</groupId>
            <artifactId>spring-openapi-schema-generator-plugin</artifactId>
            <configuration>
                <title>Test title</title>
                <description>Test description</description>
                <version>1.0.0-TEST</version>
                <modelPackages>
                    <modelPackage>org.spring.openapi.schema.generator.test.model.*</modelPackage>
                </modelPackages>
                <controllerBasePackages>
                    <controllerBasePackage>org.spring.openapi.schema.generator.test.controller.*</controllerBasePackage>
                </controllerBasePackages>
                <outputDirectory>target/openapi</outputDirectory>
                <schemaFieldInterceptors>
                    <schemaFieldInterceptor>com.github.jrcodeza.schema.generator.plugin.interceptor.TestSchemaFieldInterceptor</schemaFieldInterceptor>
                </schemaFieldInterceptors>
                <operationInterceptors>
                    <operationInterceptor>com.github.jrcodeza.schema.generator.plugin.interceptor.TestOperationInterceptor</operationInterceptor>
                </operationInterceptors>
                <generateExamples>true</generateExamples>
                <openApiExamplesResolver>com.github.jrcodeza.schema.generator.plugin.example.TestExampleResolver</openApiExamplesResolver>
            </configuration>
        </plugin>
    </plugins>
</build>
```

You can see that title, description and version are the OpenAPI info parameters.
Then you have to define also modelPackages and controllerPackages. Additionally
you should define also outputDirectory, in our case target/openapi will be the target folder
where the swagger.json will be saved.

From 1.2.0 version it is also possible to define all interceptors also in plugin. Additionally there
is also option to turn on/off generateExamples and of course specify openApiExamplesResolver.

## Generate client from spec

### Runtime usage
Include dependency

```java
<dependency>
    <groupId>com.github.jrcodeza</groupId>
    <artifactId>spring-openapi-client-generator</artifactId>
    <version>1.4.6</version>
</dependency>
```

Use generator in code

```java
new OpenApiClientGenerator().generateClient(
				"target.package",
				"pathToOpenApi3Spec",
				"targetFolder", // e.g. /target/generatedSources/oas
				true); // should generate interface
```

### Maven plugin usage

1. Include dependency
```java
<dependency>
    <groupId>com.github.jrcodeza</groupId>
    <artifactId>spring-openapi-client-generator-plugin</artifactId>
    <version>1.4.6</version>
</dependency>
```

2. Add plugin to maven plugins section
```xml
<plugin>
    <groupId>com.github.jrcodeza</groupId>
    <artifactId>spring-openapi-client-generator-plugin</artifactId>
    <executions>
        <execution>
            <goals>
                <goal>generateClientFromOpenApi</goal>
            </goals>
            <configuration>
                <outputPackage>test.package</outputPackage>
                <outputPath>target/generated-sources</outputPath>
                <schemaPath>src/main/resources/oas3.json</schemaPath>
                <generateResourceInterface>true</generateResourceInterface>
                <generateDiscriminatorProperty>true</generateDiscriminatorProperty>
            </configuration>
        </execution>
    </executions>
</plugin>
```
**generateResourceInterface** option allows you to turn on/off generation of interface
with methods that represent operations of the API.

**generateDiscriminatorProperty** allows you to turn on/off functionality where
plugin generates the discriminator property explicitly to class (setting it visible=true in jackson).
This is handy when you need to see the discriminator property inside the class
and being able to access it using setter or getter. 

## Contributions
Pull requests are welcome. If you would like to collaborate more feel free to contact
me on remenec.jakub@gmail.com .

## License
License for this tool is GNU GPL V3.
 readmeEtag: '"76b9a39260eb7d089c5fffd6deb2e95d164440cf"' readmeLastModified: Tue, 23 Jun 2020 10:13:33 GMT repositoryId: 199309560 description: Spring Boot OpenAPI generator, container and UI. created: '2019-07-28T16:13:05Z' updated: '2025-12-07T19:24:59Z' language: Java archived: false stars: 58 watchers: 8 forks: 21 owner: jrcodeza logo: https://avatars.githubusercontent.com/u/25500422?v=4 license: MIT repoEtag: '"2117ce7ce165b372ceace4d842357b7087014fa8edf921af82a94b3963792588"' repoLastModified: Sun, 07 Dec 2025 19:24:59 GMT foundInMaster: true id: e66e43593ee243e46c820555721612a7 - source: https://openapi.tools/ name: - oas - oas-normalize language: - JavaScript - TypeScript link: https://openap.is repository: https://github.com/readmeio/oas source_description: - Comprehensive tooling for working with OpenAPI definitions - Comprehensive tooling for working with OpenAPI definitions. v2: true v3: true repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://npm.im/oas">
    <img src="https://raw.githubusercontent.com/readmeio/oas/main/.github/hero.png" alt="oas" />
  </a>
</p>

<p align="center">
  Comprehensive tooling for working with OpenAPI definitions
</p>

<p align="center">
  <a href="https://npm.im/oas"><img src="https://img.shields.io/npm/v/oas?style=for-the-badge" alt="NPM Version" /></a>
  <a href="https://npm.im/oas"><img src="https://img.shields.io/node/v/oas?style=for-the-badge" alt="Node Version" /></a>
  <a href="https://npm.im/oas"><img src="https://img.shields.io/npm/l/oas?style=for-the-badge" alt="MIT License" /></a>
  <a href="https://github.com/readmeio/oas"><img src="https://img.shields.io/github/actions/workflow/status/readmeio/oas/ci.yml?branch=main&style=for-the-badge" alt="Build status" /></a>
</p>

<p align="center">
  <a href="https://readme.com"><img src="https://raw.githubusercontent.com/readmeio/.github/main/oss-badge.svg" /></a>
</p>

`oas` is the library that we've built at [ReadMe](https://readme.com) for powering everything we do related to [OpenAPI](https://www.openapis.org/); from our [Reference Guides](https://readme.com/documentation), to our [Metrics product](https://readme.com/metrics), or other in-house tooling like [code generation](https://npm.im/@readme/oas-to-snippet), [request execution](https://npm.im/@readme/oas-to-har), and [SDK code generation](https://api.readme.dev/).

---

- [Installation](#installation)
- [Usage](#usage)
  - [OpenAPI definitions](#openapi-definitions)
    - [General](#general)
    - [Operations](#operations)
    - [Servers](#servers)
    - [Specification Extensions](#specification-extensions)
    - [User Authentication](#user-authentication)
  - [Operations](#operations-1)
    - [General](#general-1)
    - [Callbacks](#callbacks)
    - [Request Body](#request-body)
    - [Responses](#responses)
    - [Security](#security)
    - [Specification Extensions](#specification-extensions-1)
  - [Callbacks](#callbacks)
    - [General](#general-2)
  - [Webhooks](#webhooks)
- [FAQ](#faq)

## Installation

```
npm install oas
```

## Usage

> [!NOTE]
> If you need to use this library within a browser you'll likely need to use a bundler like [Webpack](https://webpack.js.org/) or [Rollup](https://rollupjs.org/).

`oas` offers a main `Oas` class, which will be your main entrypoint for using the library.

```js
import Oas from 'oas';
import petstoreSpec from '@readme/oas-examples/3.0/json/petstore.json' with { type: 'json' };

const petstore = new Oas(petstoreSpec);
```

Here the `Oas` constructor takes a JSON API definition (`petstoreSpec`). All API definitions you feed it must be JSON and must be an OpenAPI definition. If you have a YAML or Swagger definition you will need to convert it (our [oas-normalize](https://npm.im/oas-normalize) library can do this for you). And if you're in a CJS or non-`import` environment you can pull the library in with `require('oas').default`.

From here, the following APIs are at your disposal.

### OpenAPI definitions

Because this library has full TypeScript types and docblocks this README is not intended to be full documentation so consult the individual method docblocks if you need more information on a specific method.

#### General

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.dereference()` | Dereference the current OpenAPI definition. Note that this will ignore circular references. |
| `.getCircularReferences()` | Retrieve an array of any circular `$ref` pointer that may exist wthin the OpenAPI definition. Note that this requires `.dereference()` to be called first. |
| `.getDefinition()` | Retrieve the OpenAPI definition that was fed into the `Oas` constructor. |
| `.getTags()` | Retrieve an array of all tags that exist within the API definition and are set on operations. |
| `.getPaths()` | Retrieve every operation that exists within the API definition. This returns an array of instances of the `Operation` class. |
| `.getVersion()` | Retrieve the OpenAPI version that this API definition is targeted for. |
| `.getWebhooks()` | Retrieve every webhook operation that exists within the API definition. This returns an array of instances of the `Webhook` class. |
| `#init()` | An alternative for `new Oas()` that you can use if the typing on the `Oas` constructor gives you trouble. Typing OpenAPI definitions is hard! |
<!-- prettier-ignore-end -->

#### Operations

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.findOperation()` | Discover an operation with the current OpenAPI definition that matches a given URL and HTTP method. |
| `.findOperationWithoutMethod()` | Like `.findOperation()` but without supplying an HTTP method. |
| `.getOperation()` | Same as `.findOperation()` but this returns an instance of the `Operation` class. |
| `.getOperationById()` | Retrieve an operation in an OpenAPI definition by an `operationId`. |
| `.operation()` | Retrieve an instance of the `Operation` or `Webhook` classes for a given path and HTTP method. |
<!-- prettier-ignore-end -->

#### Servers

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.defaultVariables()` | Retrieve the default server variables for a specific server URL, while also potentially factoring in user data. You can specify user variable data to the `Oas` constructor. Check out [Using Variables in Documentation](https://docs.readme.com/docs/user-data-options#using-variables-in-documentation) for some background on how we use this. |
| `.replaceUrl()` | Replace a given templated server URL with supplied server variable data. |
| `.splitUrl()` | Chunk out a specific server URL into its individual parts. |
| `.splitVariables` | Chunk out a given URL and if it matches a server URL in the OpenAPI file, extract the matched server variables that are present in the URL. |
| `.url()` | Retrieve a fully composed server URL. You can optionally select which of the defined server URLs to use as well as specify server variable information. |
| `.variables()` | Retrieve all server variables that a specific server URL in the definition has. |
<!-- prettier-ignore-end -->

#### Specification Extensions

> [!NOTE]
> Optionally you can also supply an instance of the `Operation` to both of these methods to see or retrieve a given extension if it exists on that operation but if it exists in both the operation and at the root the operation-level extension will be prioritized.

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getExtension()` | Retrieve a given [specification extension](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#specification-extensions) if it exists at the root of the API definition.  |
| `.hasExtension()` | Determine if a given [specification extension](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#specification-extensions) exists on the root of the API definition. |
| `.validateExtension()` | Determine if a given [ReadMe custom OpenAPI extension](https://docs.readme.com/docs/openapi-extensions) is valid or not. |
| `.validateExtensions()` | Validate all of our [ReadMe custom OpenAPI extension](https://docs.readme.com/docs/openapi-extensions), throwing exceptions when necessary. |
<!-- prettier-ignore-end -->

Information about ReadMe's supported OpenAPI extensions at https://docs.readme.com/docs/openapi-extensions.

#### User Authentication

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getAuth()` | Retrieve the appropriate API keys for the current OpenAPI definition from an object of user data. Check out [Using Variables in Documentation](https://docs.readme.com/docs/user-data-options#using-variables-in-documentation) for some background on how we use this. |
<!-- prettier-ignore-end -->

### Operations

For your convenience, the entrypoint into the `Operation` class should generally always be through `.operation()`. For example:

```js
import Oas from 'oas';
import petstoreSpec from '@readme/oas-examples/3.0/json/petstore.json' with { type: 'json' };

const petstore = new Oas(petstoreSpec);
const operation = petstore.operation('/pet', 'post');
```

#### General

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getContentType()` | Retrieve the primary request body content type. If multiple are present, prefer whichever is JSON-compliant. |
| `.getDescription()` | Retrieve the `description` that's set on this operation. This supports common descriptions that may be set at the [path item level](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#path-item-object). |
| `.getOperationId()` | Retrieve the `operationId` that's present on the operation, and if one is not present one will be created based off the method + path and returned instead. |
| `.hasOperationId()` | Determine if the operation has an `operationId` present. |
| `.isDeprecated()` | Determine if this operation is marked as deprecated. |
| `.isFormUrlEncoded()` | Determine if this operation requires its payload to be delivered as `application/x-www-form-urlencoded`. |
| `.isJson()` | Determine if this operation requires its payload to be delivered as JSON. |
| `.isMultipart()` | Determine if this operation requires its data to be sent as a multipart payload. |
| `.isXml()` | Determine if this operation requires its data to be sent as XML. |
| `.isWebhook()` | Determine if this operation is an instance of the `Webhook` class. |
| `.getExampleGroups()` | Returns an object with groups of all example definitions (body/header/query/path/response/etc.). The examples are grouped by their key when defined via the `examples` map. |
| `.getHeaders()` | Retrieve all headers that can either be sent for or returned from this operation. This includes header-based authentication schemes, common header parameters, and request body and response content types. |
| `.getSummary()` | Retrieve the `summary` that's set on this operation. This supports common summaries that may be set at the [path item level](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#path-item-object). |
| `.getTags()` | Retrieve all tags, and [their metadata](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#tag-object), that exist on this operation. |
<!-- prettier-ignore-end -->

#### Callbacks

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getCallback()` | Retrieve a specific callback on this operation. This will return an instance of the `Callback` class. |
| `.getCallbackExamples()` | Retrieve an array of all calback examples that this operation has defined. |
| `.getCallbacks()` | Retrieve all callbacks that this operation has. Similar to `.getPaths()` returning an array of `Operation` instances this will return an array of `Callback` instances. |
| `.hasCallbacks()` | Determine if this operation has any callbacks defined. |
<!-- prettier-ignore-end -->

#### Parameters

> [!NOTE]
> All parameter accessors here support, and will automatically retrieve and handle, common parameters that may be set at the [path item level](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#path-item-object).

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getParameters()` | Retrieve all parameters that may be used with on this operation. |
| `.getParametersAsJSONSchema()` | Retrieve and convert the operations parameters into an array of JSON Schema schemas for each available type of parameter available on the operation: `path`, `query`, `body`, `cookie`, `formData`, and `header`. |
| `.hasParameters()` | Determine if the operation has any parameters to send. |
| `.hasRequiredParameters()` | Determine if any of the parameters on this operation are required. |
<!-- prettier-ignore-end -->

#### Request Body

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getRequestBody()` | Retrieve the raw request body object for a given content type. If none is specified it will return either the first that's JSON-like, or the first defined. |
| `.getRequestBodyExamples()` | Retrieve an array of all request body examples that this operation has defined. |
| `.getRequestBodyMediaTypes()` | Retrieve a list of all the media/content types that the operation can accept a request body payload for. |
| `.hasRequestBody()` | Determine if this operation has a request body defined. |
| `.hasRequiredRequestBody()` | Determine if this operation has a required request body. |
<!-- prettier-ignore-end -->

#### Responses

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getResponseAsJSONSchema()` | Retrive and convert a response on this operation into JSON Schema. |
| `.getResponseByStatusCode()` | Retrieve the raw response object for a given status code. |
| `.getResponseExamples()` | Retrieve an array of all response examples that this operation has defined. |
| `.getResponseStatusCodes()` | Retrieve all status codes that this operation may respond with. |
<!-- prettier-ignore-end -->

#### Security

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getSecurity()` | Return all security requirements that are applicable for either this operation, or if the operation has none specific to it, then for the entire API. |
| `.getSecurityWithTypes()` | Return a collection of all security schemes applicable to this operation (using `.getSecurity()`), grouped by how the security should be handled (either AND or OR auth requirements). |
| `.prepareSecurity` | Return an object of every security scheme that _can_ be used on this operation, indexed by the type of security scheme it is (eg. `Basic`, `OAuth2`, `APIKey`, etc.). |
<!-- prettier-ignore-end -->

#### Specification Extensions

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.hasExtension()` | Determine if a given [specification extension](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#specification-extensions) exists on this operation. |
<!-- prettier-ignore-end -->

Information about ReadMe's supported OpenAPI extensions at https://docs.readme.com/docs/openapi-extensions.

### Callbacks

The `Callback` class inherits `Operation` so every API available on instances of `Operation` is available here too. Much like `Operation`, we also support common parameters, summaries, and descriptions that may be set at the [path item level](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#path-item-object) within a `callbacks` definition.

#### General

<!-- prettier-ignore-start -->
| Method | Description |
| :--- | :--- |
| `.getIdentifier()` | Retrieve the primary identifier of this callback. |
<!-- prettier-ignore-end -->

### Webhooks

Because our `Webhook` class extends `Operation`, every API that's available on the [Operation](#operation) class is available on webhooks.

### Additional Utilities

Beyond the `Oas`, `Operation`, `Callback` and `Webhook` interfaces, the `oas` library also offers additional tooling for analyzing and troubleshooting OpenAPI definitions.

#### Analyzer

The analyzer, `oas/analyzer`, allows you to run a set of query analyses on your API definition to understand the complexity of your API definition.

```ts
import petstore from '@readme/oas-examples/3.0/json/petstore.json' with { type: 'json' };
import analyzer from 'oas/analyzer';

console.log(await analyzer(petstore));
```

##### General

<!-- prettier-ignore-start -->
| Metric | Description |
| :--- | :--- |
| `dereferencedFileSize` | Size of the definition after resolving all references. |
| `mediaTypes` | What are the different media type shapes that your API operations support? |
| `operationTotal` | The total amount of operations in your definition. |
| `rawFileSize` | Size of the definition in its raw form. |
| `securityTypes` | The different types of security that your API contains. |
<!-- prettier-ignore-end -->

##### OpenAPI Features

<!-- prettier-ignore-start -->
| Metric | Description |
| :--- | :--- |
| `additionalProperties` | Does your API use `additionalProperties`? |
| `callbacks` | Does your API use [callbacks](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#callback-object)? |
| `circularRefs` | Does your API have any circular `$ref` pointers, and if so where are they located? |
| `commonParameters` | Does your API utilize [common parameters](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#path-item-object)? |
| `discriminators` | Does your API use polymorphic [discriminators](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#discriminator-object)? |
| `links` | Does your API use [links](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md#link-object)? |
| `style` | Do any parameters in your API require [style](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#user-content-parameterstyle) serialization?
| `polymorphism` | Does your API use [polymorphism](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#composition-and-inheritance-polymorphism) (`anyOf`, `oneOf`, `allOf`)? |
| `serverVariables` | Does your API use [server variables](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#server-variable-object)? |
| `webhooks` | Does your API use [webhooks](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#oasWebhooks)?
| `xml` | Does any parameter or schema in your API use the [XML object](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#xml-object) for declaring how a schema should be treated in XML? |
<!-- prettier-ignore-end -->

#### Reducer

> [!WARNING]
> This API is still very experimental and should not be used in production environments!

The reducer, `oas/reducer`, can be used to reduce an OpenAPI definition down to only the information necessary for a specific set of tags, paths, or operations. OpenAPI reduction can be helpful to isolate and troubleshoot issues with a very large API definition -- all while still having a fully functional and valid OpenAPI definition.

```ts
import petstore from '@readme/oas-examples/3.0/json/petstore.json' with { type: 'json' };
import reducer from 'oas/reducer';

// This will reduce the `petstore` API definition down to only operations, and
// any referenced schemas, that are a part of the `Store` tag.
console.log(reducer(petstore, { tags: ['Store'] }));

// Reduces the `petstore` down to only the `POST /pet` operation.
console.log(reducer(petstore, { paths: { '/pet': ['post'] } });

// You can also select all of the methods of a given path by using the `*`
// wildcard. The resulting reduced API definition here will contain `POST /pet`
// and `PUT /put`.
console.log(reducer(petstore, { paths: { '/pet': ['*'] } });
```

> [!NOTE]
> Though the reducer does not require you to first dereference your API definition it currently unfortunately cannot, depending on the circumstances, be used to dereference an API operation that has circular `$ref` pointers.

## FAQ

#### Can I create an OpenAPI definition with this?

Though `oas` used to offer functionality related to this, it does no longer. If you need an OpenAPI (or Swagger) definition for your API we recommend checking out the API editing experience within [ReadMe](https://readme.com), manually maintaining JSON/YAML files (it sounds worse than it actually is), or the language-agnostic [swagger-inline](https://npm.im/swagger-inline).
 readmeEtag: '"d62d6dfd8f5bf3fb15ea42bbb03a9a87561232ce"' readmeLastModified: Thu, 31 Jul 2025 02:44:03 GMT repositoryId: 84146520 description: Comprehensive tooling for working with OpenAPI definitions. created: '2017-03-07T02:46:31Z' updated: '2026-02-05T02:21:40Z' language: TypeScript archived: false stars: 373 watchers: 18 forks: 38 owner: readmeio logo: https://avatars.githubusercontent.com/u/6878153?v=4 license: MIT repoEtag: '"20dadbe29e21e1da87b3ce76f1c0592de2393e880dba055a0cbd72f01d54fab1"' repoLastModified: Thu, 05 Feb 2026 02:21:40 GMT foundInMaster: true id: 362a4b56c90877840a8032f920389bf6 v3_1: true category: - Parsers - Description Validators - Converters - source: https://openapi.tools/ name: api category: - Parsers - Description Validators - Converters language: - JavaScript - TypeScript repository: https://github.com/readmeio/api link: https://api.readme.dev source_description: Magical SDK generation from an OpenAPI definition v2: true v3: true v3_1: true id: fe3a10e114b8e1a8d9f03e70c963b89e repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyB3aWR0aD0iNDAwIiBzcmM9Imh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9yZWFkbWVpby9hcGkvbWFpbi9kb2NzL2ltYWdlcy9sb2dvLnN2ZyIgLz4KPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CiAgTWFnaWNhbCBTREsgZ2VuZXJhdGlvbiBmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbiDwn6qECjwvcD4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxhIGhyZWY9Imh0dHBzOi8vbnBtLmltL2FwaSI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3YvYXBpP3N0eWxlPWZvci10aGUtYmFkZ2UiIGFsdD0iTlBNIFZlcnNpb24iPjwvYT4KICA8YSBocmVmPSJodHRwczovL25wbS5pbS9hcGkiPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL25vZGUvdi9hcGk/c3R5bGU9Zm9yLXRoZS1iYWRnZSIgYWx0PSJOb2RlIFZlcnNpb24iPjwvYT4KICA8YSBocmVmPSJodHRwczovL25wbS5pbS9hcGkiPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9sL2FwaT9zdHlsZT1mb3ItdGhlLWJhZGdlIiBhbHQ9Ik1JVCBMaWNlbnNlIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3JlYWRtZWlvL2FwaSI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3JlYWRtZWlvL2FwaS9jaS55bWw/YnJhbmNoPW1haW4mc3R5bGU9Zm9yLXRoZS1iYWRnZSIgYWx0PSJCdWlsZCBzdGF0dXMiPjwvYT4KPC9wPgoKLSBbSW5zdGFsbGF0aW9uXShodHRwczovL2FwaS5yZWFkbWUuZGV2L2RvY3MvaW5zdGFsbGF0aW9uKQotIFtVc2FnZV0oaHR0cHM6Ly9hcGkucmVhZG1lLmRldi9kb2NzL3VzYWdlKQogIC0gW0F1dGhlbnRpY2F0aW9uXShodHRwczovL2FwaS5yZWFkbWUuZGV2L2RvY3MvYXV0aGVudGljYXRpb24pCiAgLSBbUGFyYW1ldGVycyBhbmQgUGF5bG9hZHNdKGh0dHBzOi8vYXBpLnJlYWRtZS5kZXYvZG9jcy9wYXJhbWV0ZXJzLWFuZC1wYXlsb2FkcykKICAtIFtNYWtpbmcgcmVxdWVzdHNdKGh0dHBzOi8vYXBpLnJlYWRtZS5kZXYvZG9jcy9tYWtpbmctcmVxdWVzdHMpCiAgLSBbU2VydmVyIGNvbmZpZ3VyYXRpb25zXShodHRwczovL2FwaS5yZWFkbWUuZGV2L2RvY3Mvc2VydmVyLWNvbmZpZ3VyYXRpb25zKQotIFtIb3cgZG9lcyBpdCB3b3JrP10oaHR0cHM6Ly9hcGkucmVhZG1lLmRldi9kb2NzL2hvdy1pdC13b3JrcykKLSBbRkFRXShodHRwczovL2FwaS5yZWFkbWUuZGV2L2RvY3MvZmFxKQoKYGFwaWAgaXMgYSBsaWJyYXJ5IHRoYXQgZmFjaWxpdGF0ZXMgY3JlYXRpbmcgYW4gU0RLIGZyb20gYW4gT3BlbkFQSSBkZWZpbml0aW9uLiBZb3UgY2FuIHVzZSBpdHMgY29kZWdlbiBvZmZlcmluZyB0byBjcmVhdGUgYW4gb3BpbmlvbmF0ZWQgU0RLIGZvciBUeXBlU2NyaXB0IG9yIEpTICgrIFR5cGVTY3JpcHQgdHlwZXMpLgoKYGBgc2gKJCBucHggYXBpIGluc3RhbGwgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFpbi9leGFtcGxlcy92My4wL3BldHN0b3JlLmpzb24KYGBgCgpgYGBqcwpjb25zdCBwZXRzdG9yZSA9IHJlcXVpcmUoJ0BhcGkvcGV0c3RvcmUnKTsKCnBldHN0b3JlLmxpc3RQZXRzKCkudGhlbigoeyBkYXRhIH0pID0+IHsKICBjb25zb2xlLmxvZyhgTXkgcGV0cyBuYW1lIGlzICR7ZGF0YVswXS5uYW1lfSFgKTsKfSk7CmBgYAoKVGhlIEVTTSBzeW50YXggaXMgc3VwcG9ydGVkIGFzIHdlbGw6CgpgYGBqcwppbXBvcnQgcGV0c3RvcmUgZnJvbSAnQGFwaS9wZXRzdG9yZSc7CgpwZXRzdG9yZS5saXN0UGV0cygpLnRoZW4oKHsgZGF0YSB9KSA9PiB7CiAgY29uc29sZS5sb2coYE15IHBldHMgbmFtZSBpcyAke2RhdGFbMF0ubmFtZX0hYCk7Cn0pOwpgYGAK readmeEtag: '"2fcc971cbd1848188ddab0a4634dc071390696a0"' readmeLastModified: Thu, 01 Aug 2024 17:08:22 GMT repositoryId: 81140437 description: 🚀 Automatic SDK generation from an OpenAPI definition created: '2017-02-06T22:31:07Z' updated: '2026-02-04T01:09:16Z' language: TypeScript archived: false stars: 675 watchers: 18 forks: 28 owner: readmeio logo: https://avatars.githubusercontent.com/u/6878153?v=4 license: MIT repoEtag: '"a08669db019bf17b4b22312788190a35b691bc68392aacc85b2a9c971201dfd2"' repoLastModified: Wed, 04 Feb 2026 01:09:16 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: openapi-cli-tool link: https://pypi.org/project/openapi-cli-tool/ repository: https://github.com/hakopako/openapi-cli-tool language: Python source_description: >- Can list up defined API paths and bundle multi-file into one. Supports multiple file extensions. v2: false v3: true repositoryMetadata: base64Readme: >- IVtvcGVuYXBpLWNsaS10b29sXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vaGFrb3Bha28vb3BlbmFwaS1jbGktdG9vbC9tYXN0ZXIvZG9jL2xvZ28ucG5nKQoKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kuY29tL2hha29wYWtvL29wZW5hcGktY2xpLXRvb2wuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5jb20vaGFrb3Bha28vb3BlbmFwaS1jbGktdG9vbCkKIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3ZlcnNpb24tdjAuMy4wLWdyZWVuLnN2ZyI+CiA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1saWdodGdyYXkuc3ZnIj4gIAo8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9weXRob24tMi43LDMuNDw9LWJsdWUuc3ZnIj4gPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvc3dhZ2dlci0zLngteWVsbG93LnN2ZyI+CgojIG9wZW5hcGktY2xpLXRvb2wKT3BlbkFQSSAoU3dhZ2dlciAzLngpIENMSSBUb29sLiAgCgotIFN1cHBvcnRzIG11bHRpcGxlIGZpbGUgZXh0ZW5zaW9ucyAoanNvbnx5YW1sfHltbCkuCi0gQ2FuIGxpc3QgdXAgZGVmaW5lZCBBUEkgcGF0aHMuCi0gRGlzcGxheSBhbiBBUEkgc3BlY2lmaWNhdGlvbiB3aGljaCBpcyByZXNvbHZlZCAoYCRyZWZgKS4KLSBCdW5kbGUgbXVsdGktZmlsZSBpbnRvIG9uZSAob3V0cHV0IHRvIGpzb258eWFtbHxodG1sKS4KLSBPQVMgaW50ZXJhY3RpdmUgc2NhZmZvbGRpbmcuICAKCiMgUmVxdWlyZW1lbnRzCgpQeXRob24gMi43LCAzLjQgPD0uCgojIEluc3RhbGxhdGlvbgoKV2l0aCBwaXA6CgpgYGBiYXNoCiQgcGlwIGluc3RhbGwgb3BlbmFwaS1jbGktdG9vbApgYGAKTWFudWFsbHk6ICAKQ2xvbmUgdGhlIHJlcG9zaXRvcnkgYW5kIGV4ZWN1dGUgdGhlIFB5dGhvbiBpbnN0YWxsYXRpb24gY29tbWFuZCBvbiB5b3VyIG1hY2hpbmUuICAKCmBgYGJhc2gKJCBwaXAgLXIgcmVxdWlyZW1lbnRzLnR4dCBpbnN0YWxsCiQgcHl0aG9uIHNldHVwLnB5IGluc3RhbGwKYGBgCgpUaGVuIGBvcGVuYXBpLWNsaS10b29sYCBjb21tYW5kIGlzIGluc3RhbGxlZC4KCiMgVXNhZ2UKCmBgYAokIG9wZW5hcGktY2xpLXRvb2wgLS1oZWxwClVzYWdlOiBvcGVuYXBpLWNsaS10b29sIFtPUFRJT05TXSBDT01NQU5EIFtBUkdTXS4uLgoKT3B0aW9uczoKICAtLWhlbHAgIFNob3cgdGhpcyBtZXNzYWdlIGFuZCBleGl0LgoKQ29tbWFuZHM6CiAgYnVuZGxlICAgIEJ1bmRsZSBtdWx0aXBsZSBmaWxlcyBpbnRvIG9uZS4KICBsaXN0ICAgICAgTGlzdCB1cCBBUEkgcGF0aHMgaW4gYSBmaWxlIG9yIGRpcmVjdG9yeS4KICByZXNvbHZlICAgRGlzcGxheSBgJHJlZmAgcmVzb2x2ZWQgQVBJIHNwZWNpZmljYXRpb24uCiAgc2NhZmZvbGQgIEludGVyYWN0aXZlbHkgY3JlYXRlIGEgc2ltcGxlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbi4KYGBgCgojIyBCdW5kbGUKCkJ1bmRsZSBtdWx0aS1maWxlIHNwZWNpZmljYXRpb25zIGludG8gb25lLCByZWdhcmRsZXNzIG9mIGZpbGUgZXh0ZW5zaW9uIChqc29ufHlhbWx8eW1sKS4KCmBgYAokIG9wZW5hcGktY2xpLXRvb2wgYnVuZGxlIC0taGVscApVc2FnZTogb3BlbmFwaS1jbGktdG9vbCBidW5kbGUgW09QVElPTlNdIEZJTEVfUEFUSFMKCiAgQnVuZGxlIG11bHRpcGxlIGZpbGVzIGludG8gb25lLgoKT3B0aW9uczoKICAtZiwgLS1maWxlIFRFWFQgIExvYWQgY29tbW9uIG9iamVjdHMgc3VjaCBhcyBpbmZvIGFuZCBzZXJ2ZXJzIGZyb20gYQogICAgICAgICAgICAgICAgICAgc3BlY2lmaWMgZmlsZS4gRGVmYXVsdCBpcyBhIGZpbGUgd2hpY2ggaXMgdGhlIHRvcCBvZiBsaXN0CiAgICAgICAgICAgICAgICAgICBjb21tYW5kIHJlc3VsdC4KICAtdCwgLS10eXBlIFRFWFQgIEV4cG9ydCBkYXRhIHR5cGUuIHtqc29ufHlhbWx8aHRtbH0gIFtkZWZhdWx0OiBqc29uXQogIC0taGVscCAgICAgICAgICAgU2hvdyB0aGlzIG1lc3NhZ2UgYW5kIGV4aXQuCmBgYAoKZXhhbXBsZToKYGBgCiQgb3BlbmFwaS1jbGktdG9vbCBidW5kbGUgLXQgaHRtbCBmaWxlMS5qc29uIGZpbGUyLnlhbWxgID4gLi9zcGVjaWZpY2F0aW9uLmh0bWwKYGBgCgpJbiB0aGUgaHRtbCBmaWxlLCBhbiB1bnBrZyB2ZXJzaW9uIG9mIFtzd2FnZ2VyLXVpXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkgaXMgZW1iZWRkZWQuIFJlbmRlcmVkIHNjcmVlbnNob3QgYmVsb3c6ICAKCgohW2J1bmRsZS1odG1sLWltZ10oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2hha29wYWtvL29wZW5hcGktY2xpLXRvb2wvbWFzdGVyL2RvYy9idW5kbGUtaHRtbC5wbmcpCgoKIyMgTGlzdAoKTGlzdCB1cCBBUEkgcGF0aHMgZnJvbSBhIGZpbGUvZGlyZWN0b3J5IHJlZ2FyZGxlc3Mgb2YgdGhlIGZpbGUgZXh0ZW5zaW9uIChqc29ufHlhbWx8eW1sKS4KCmBgYGJhc2gKJCBvcGVuYXBpLWNsaS10b29sIGxpc3QgYGZpbmQgLi9zcGVjYAoKTWV0aG9kICAgIFBhdGggICAgICAgRmlsZQotLS0tLS0tLSAgLS0tLS0tLS0tICAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KUFVUICAgICAgIC9hdmF0YXIgICAgLi90ZXN0cy9yZXNvdXJjZXMvc3BlYy9zYW1wbGUueW1sCkdFVCAgICAgICAvZm9sbHdlcnMgIC4vdGVzdHMvcmVzb3VyY2VzL3NwZWMvZm9sZGVyMS9zYW1wbGUyLnlhbWwKUE9TVCAgICAgIC9mb2xsd2VycyAgLi90ZXN0cy9yZXNvdXJjZXMvc3BlYy9mb2xkZXIxL3NhbXBsZTIueWFtbApQVVQgICAgICAgL2ZvbGx3ZXJzICAuL3Rlc3RzL3Jlc291cmNlcy9zcGVjL2ZvbGRlcjEvc2FtcGxlMi55YW1sClBPU1QgICAgICAvcGV0cyAgICAgIC4vdGVzdHMvcmVzb3VyY2VzL3NwZWMvc2FtcGxlLnltbApHRVQgICAgICAgL3Bvc3RzICAgICAuL3Rlc3RzL3Jlc291cmNlcy9zcGVjL2ZvbGRlcjEvc2FtcGxlLmpzb24KUE9TVCAgICAgIC9wb3N0cyAgICAgLi90ZXN0cy9yZXNvdXJjZXMvc3BlYy9mb2xkZXIxL3NhbXBsZS5qc29uCkdFVCAgICAgICAvdXNlcnMgICAgIC4vdGVzdHMvcmVzb3VyY2VzL3NwZWMvZm9sZGVyMS9zYW1wbGUuanNvbgpQT1NUICAgICAgL3VzZXJzICAgICAuL3Rlc3RzL3Jlc291cmNlcy9zcGVjL2ZvbGRlcjEvc2FtcGxlLmpzb24KYGBgCgoKIyMgUmVzb2x2ZQoKRGlzcGxheSBhbiBBUEkgc3BlY2lmaWNhdGlvbiB3aGljaCBpcyByZXNvbHZlZCBmcm9tICBhIG11bHRpLWZpbGUgQVBJIHNwZWNpZmljYXRpb24gdmlhICRyZWYgcG9pbnRlcnMuICAKCmBgYApVc2FnZTogb3BlbmFwaS1jbGktdG9vbCByZXNvbHZlIFtPUFRJT05TXSBNRVRIT0QgUEFUSCBGSUxFX1BBVEhTCgogIERpc3BsYXkgYCRyZWZgIHJlc29sdmVkIEFQSSBzcGVjaWZpY2F0aW9uLgoKT3B0aW9uczoKICAtdCwgLS10eXBlIFRFWFQgIEV4cG9ydCBkYXRhIHR5cGUuIHtqc29ufHlhbWx9ICBbZGVmYXVsdDoganNvbl0KICAtLWhlbHAgICAgICAgICAgIFNob3cgdGhpcyBtZXNzYWdlIGFuZCBleGl0LgpgYGAKCmV4YW1wbGU6CmBgYGJhc2gKJCBvcGVuYXBpLWNsaS10b29sIHJlc29sdmUgcG9zdCAvY2F0cyBgZmluZCAuL3Rlc3RzL3Jlc291cmNlcy9zcGVjYApgYGAKCgojIyBTY2FmZm9sZAoKSW50ZXJhY3RpdmVseSBpbnB1dCBpbmZvcm1hdGlvbiBvZiB5b3VyIEFQSS4gIApBIHNpbXBsZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gaXMgZ2VuZXJhdGVkIGZyb20geW91ciBwcm9tcHQuCgpgYGBiYXNoCiQgb3BlbmFwaS1jbGktdG9vbCBzY2FmZm9sZAoKUGxlYXNlIGVudGVyIHRpdGxlIFsiIl06IHNhbXBsZQpQbGVhc2UgZW50ZXIgdmVyc2lvbiBbdjEuMF06ClBsZWFzZSBlbnRlciBsaWNlbnNlIFtBcGFjaGUgMi4wXToKUGxlYXNlIGVudGVyIHNlcnZlciB1cmwgW2h0dHA6Ly9leGFtcGxlLmNvbV06ClBsZWFzZSBlbnRlciBwYXRoIFsvXTogL2V4YW1wbGUKUGxlYXNlIGVudGVyIG1ldGhvZCBmb3IgL2V4YW1wbGUgW2dldHxwb3N0fHB1dHxkZWxldGV8aGVhZHxvcHRpb258dHJhY2VdOiBnZXQKUGxlYXNlIGVudGVyIGRlc2NyaXB0aW9uIGZvciBnZXQgL2V4YW1wbGUgWyIiXTogc2FtcGxlIGdldCBlbmRwb2ludApQbGVhc2UgZW50ZXIgcmVzcG9uc2UgY29kZSBmb3IgZ2V0IC9leGFtcGxlIFsyMDBdOgpgYGAK readmeEtag: '"4160acff531aee88dd1cb97794807f5b79204ae4"' readmeLastModified: Fri, 13 Dec 2019 15:18:01 GMT repositoryId: 189242303 description: >- OpenAPI (Swagger 3.x) CLI Tool. Supports multiple file extensions. Can list up defined API paths and bundle multi-file into one. created: '2019-05-29T14:31:14Z' updated: '2025-10-06T09:27:54Z' language: Python archived: false stars: 23 watchers: 1 forks: 5 owner: hakopako logo: https://avatars.githubusercontent.com/u/4510192?v=4 license: MIT repoEtag: '"881dfb9e91ef90149249b19fcdf3566470ec23614f0da8ae1d05fe1d6777f605"' repoLastModified: Mon, 06 Oct 2025 09:27:54 GMT foundInMaster: true id: 74cfd1a51e146dbdbfccbf999d194eae category: Parsers - source: https://openapi.tools/ name: laravel-openapi category: - Converters - Testing language: PHP repository: https://github.com/vyuldashev/laravel-openapi source_description: Generate OpenAPI 3 specification for Laravel Applications. v2: false v3: true repositoryMetadata: base64Readme: >- IyBHZW5lcmF0ZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gZm9yIExhcmF2ZWwgQXBwbGljYXRpb25zCgpbIVtMYXRlc3QgVmVyc2lvbiBvbiBQYWNrYWdpc3RdKGh0dHBzOi8vcG9zZXIucHVneC5vcmcvdnl1bGRhc2hldi9sYXJhdmVsLW9wZW5hcGkvdi9zdGFibGU/Zm9ybWF0PWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9wYWNrYWdpc3Qub3JnL3BhY2thZ2VzL3Z5dWxkYXNoZXYvbGFyYXZlbC1vcGVuYXBpKQpbIVtNSVQgTGljZW5zZWRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGljZW5zZS1NSVQtYnJpZ2h0Z3JlZW4uc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKV0oTElDRU5TRS5tZCkKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vdnl1bGRhc2hldi9sYXJhdmVsLW9wZW5hcGkvd29ya2Zsb3dzL1Rlc3RzL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS92eXVsZGFzaGV2L2xhcmF2ZWwtb3BlbmFwaS9hY3Rpb25zKQpbIVtUb3RhbCBEb3dubG9hZHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L2R0L3Z5dWxkYXNoZXYvbGFyYXZlbC1vcGVuYXBpLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy92eXVsZGFzaGV2L2xhcmF2ZWwtb3BlbmFwaSkKCiMjIERvY3VtZW50YXRpb24KCllvdSdsbCBmaW5kIHRoZSBkb2N1bWVudGF0aW9uIG9uIGh0dHBzOi8vdnl1bGRhc2hldi5naXRodWIuaW8vbGFyYXZlbC1vcGVuYXBpLgoKIyMgTGljZW5zZQoKVGhlIE1JVCBMaWNlbnNlIChNSVQpLiBQbGVhc2Ugc2VlIFtMaWNlbnNlIEZpbGVdKExJQ0VOU0UubWQpIGZvciBtb3JlIGluZm9ybWF0aW9uLgo= readmeEtag: '"5f41c782107174831f4d9482f6f3a2dfecec277b"' readmeLastModified: Thu, 04 May 2023 10:01:53 GMT repositoryId: 182520856 description: Generate OpenAPI specification for Laravel Applications created: '2019-04-21T10:45:45Z' updated: '2026-01-12T10:02:16Z' language: PHP archived: false stars: 456 watchers: 12 forks: 111 owner: vyuldashev logo: https://avatars.githubusercontent.com/u/1809081?v=4 license: MIT repoEtag: '"303550c881afeaeaae1f2683ac987fe27d1c6346406538e0b34fd90919c3495a"' repoLastModified: Mon, 12 Jan 2026 10:02:16 GMT foundInMaster: true id: ee05a2f19f3fd6fc886e5e4711ef400f - source: https://openapi.tools/ name: Flotiq - headless CMS with OpenAPI support category: - SDK - GUI Editors link: https://flotiq.com source_description: >- Visually define your Content Types, Flotiq automatically generates your own OpenAPI v3 compatible endpoints, SDKs and Postman collections. v3: true foundInMaster: true id: 478d41ac500dd9ce23e0f0213a656154 - source: https://openapi.tools/ name: Chai OpenAPI Response Validator category: Testing language: Node.js repository: >- https://github.com/openapi-library/openapivalidators/tree/master/packages/chai-openapi-response-validator source_description: >- Simple Chai support for asserting that HTTP responses satisfy an OpenAPI spec. v2: true v3: true repositoryMetadata: base64Readme: >- # OpenAPI Validators

![build status](https://github.com/openapi-library/OpenAPIValidators/actions/workflows/ci.yml/badge.svg)
![style](https://img.shields.io/badge/code%20style-airbnb-ff5a5f.svg)
[![codecov](https://codecov.io/gh/openapi-library/OpenAPIValidators/branch/master/graph/badge.svg)](https://codecov.io/gh/openapi-library/OpenAPIValidators)
[![MIT License](https://img.shields.io/npm/l/openapi-validator.svg?style=flat-square)](https://github.com/openapi-library/OpenAPIValidators/blob/master/LICENSE)
[![contributions welcome](https://img.shields.io/badge/contributions-welcome-brightgreen.svg?style=flat)](https://github.com/openapi-library/OpenAPIValidators/blob/master/CONTRIBUTING.md)

Use Jest or Chai to assert that HTTP responses satisfy an OpenAPI spec.

## Problem 😕

If your server's behaviour doesn't match your API documentation, then you need to correct your server, your documentation, or both. The sooner you know the better.

## Solution 😄

These test plugins let you automatically test whether your server's behaviour and documentation match. They extend Jest and Chai to support the [OpenAPI standard](https://swagger.io/docs/specification/about/) for documenting REST APIs. In your JavaScript tests, you can simply assert `expect(responseObject).toSatisfyApiSpec()`

### [jest-openapi](https://github.com/openapi-library/OpenAPIValidators/tree/master/packages/jest-openapi#readme)

[![downloads](https://img.shields.io/npm/dm/jest-openapi)](https://www.npmjs.com/package/jest-openapi)
[![npm](https://img.shields.io/npm/v/jest-openapi.svg)](https://www.npmjs.com/package/jest-openapi)

### [Chai OpenAPI Response Validator](https://github.com/openapi-library/OpenAPIValidators/tree/master/packages/chai-openapi-response-validator#readme)

[![downloads](https://img.shields.io/npm/dm/chai-openapi-response-validator)](https://www.npmjs.com/package/chai-openapi-response-validator)
[![npm](https://img.shields.io/npm/v/chai-openapi-response-validator.svg)](https://www.npmjs.com/package/chai-openapi-response-validator)

## Contributors ✨

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://github.com/rwalle61"><img src="https://avatars1.githubusercontent.com/u/18170169?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Richard Waller</b></sub></a><br /><a href="#maintenance-rwalle61" title="Maintenance">🚧</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=rwalle61" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=rwalle61" title="Documentation">📖</a> <a href="https://github.com/openapi-library/OpenAPIValidators/pulls?q=is%3Apr+reviewed-by%3Arwalle61" title="Reviewed Pull Requests">👀</a></td>
    <td align="center"><a href="https://github.com/JonnySpruce"><img src="https://avatars3.githubusercontent.com/u/30812276?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Jonny Spruce</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=JonnySpruce" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=JonnySpruce" title="Documentation">📖</a> <a href="https://github.com/openapi-library/OpenAPIValidators/pulls?q=is%3Apr+reviewed-by%3AJonnySpruce" title="Reviewed Pull Requests">👀</a></td>
    <td align="center"><a href="https://github.com/AlexDobeck"><img src="https://avatars2.githubusercontent.com/u/10519388?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Alex Dobeck</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=AlexDobeck" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3AAlexDobeck" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://github.com/BenGu3"><img src="https://avatars2.githubusercontent.com/u/7105857?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ben Guthrie</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=BenGu3" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3ABenGu3" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://martijnvegter.com/"><img src="https://avatars3.githubusercontent.com/u/25134477?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Martijn Vegter</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=mvegter" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/ludeknovy"><img src="https://avatars1.githubusercontent.com/u/13610612?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ludek</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=ludeknovy" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3Aludeknovy" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://github.com/tgiardina"><img src="https://avatars1.githubusercontent.com/u/37459104?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Tommy Giardina</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=tgiardina" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3Atgiardina" title="Bug reports">🐛</a></td>
  </tr>
  <tr>
    <td align="center"><a href="https://xotabu4.github.io/"><img src="https://avatars3.githubusercontent.com/u/3033972?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Oleksandr Khotemskyi</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=Xotabu4" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/amitkeinan9"><img src="https://avatars.githubusercontent.com/u/16577335?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Amit Keinan</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=amitkeinan9" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/DetachHead"><img src="https://avatars.githubusercontent.com/u/57028336?v=4?s=100" width="100px;" alt=""/><br /><sub><b>DetachHead</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3ADetachHead" title="Bug reports">🐛</a></td>
    <td align="center"><a href="http://karlssonkristoffer.com/"><img src="https://avatars.githubusercontent.com/u/20490202?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Kristoffer Karlsson</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=kristofferkarlsson93" title="Documentation">📖</a></td>
  </tr>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->
 readmeEtag: '"4d5a529f7a19b52fab62aaf17e9c5263bd765e2d"' readmeLastModified: Tue, 29 Mar 2022 10:12:21 GMT repositoryId: 199614125 description: Use Jest or Chai to assert that HTTP responses satisfy an OpenAPI spec created: '2019-07-30T08:56:01Z' updated: '2025-12-11T11:06:52Z' language: TypeScript archived: false stars: 194 watchers: 11 forks: 38 owner: openapi-library logo: https://avatars.githubusercontent.com/u/53815441?v=4 license: MIT repoEtag: '"982859aab5b874db7602c4d6d41812452f2b57b69b87a00845a5040995271994"' repoLastModified: Thu, 11 Dec 2025 11:06:52 GMT foundInMaster: true id: b5cd9eb978510085ded374f60b8e1486 - source: https://openapi.tools/ name: jest-openapi category: Testing language: Node.js repository: >- https://github.com/openapi-library/openapivalidators/tree/master/packages/jest-openapi source_description: >- Additional Jest matchers for asserting that HTTP responses satisfy an OpenAPI spec. v2: true v3: true repositoryMetadata: base64Readme: >- # OpenAPI Validators

![build status](https://github.com/openapi-library/OpenAPIValidators/actions/workflows/ci.yml/badge.svg)
![style](https://img.shields.io/badge/code%20style-airbnb-ff5a5f.svg)
[![codecov](https://codecov.io/gh/openapi-library/OpenAPIValidators/branch/master/graph/badge.svg)](https://codecov.io/gh/openapi-library/OpenAPIValidators)
[![MIT License](https://img.shields.io/npm/l/openapi-validator.svg?style=flat-square)](https://github.com/openapi-library/OpenAPIValidators/blob/master/LICENSE)
[![contributions welcome](https://img.shields.io/badge/contributions-welcome-brightgreen.svg?style=flat)](https://github.com/openapi-library/OpenAPIValidators/blob/master/CONTRIBUTING.md)

Use Jest or Chai to assert that HTTP responses satisfy an OpenAPI spec.

## Problem 😕

If your server's behaviour doesn't match your API documentation, then you need to correct your server, your documentation, or both. The sooner you know the better.

## Solution 😄

These test plugins let you automatically test whether your server's behaviour and documentation match. They extend Jest and Chai to support the [OpenAPI standard](https://swagger.io/docs/specification/about/) for documenting REST APIs. In your JavaScript tests, you can simply assert `expect(responseObject).toSatisfyApiSpec()`

### [jest-openapi](https://github.com/openapi-library/OpenAPIValidators/tree/master/packages/jest-openapi#readme)

[![downloads](https://img.shields.io/npm/dm/jest-openapi)](https://www.npmjs.com/package/jest-openapi)
[![npm](https://img.shields.io/npm/v/jest-openapi.svg)](https://www.npmjs.com/package/jest-openapi)

### [Chai OpenAPI Response Validator](https://github.com/openapi-library/OpenAPIValidators/tree/master/packages/chai-openapi-response-validator#readme)

[![downloads](https://img.shields.io/npm/dm/chai-openapi-response-validator)](https://www.npmjs.com/package/chai-openapi-response-validator)
[![npm](https://img.shields.io/npm/v/chai-openapi-response-validator.svg)](https://www.npmjs.com/package/chai-openapi-response-validator)

## Contributors ✨

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://github.com/rwalle61"><img src="https://avatars1.githubusercontent.com/u/18170169?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Richard Waller</b></sub></a><br /><a href="#maintenance-rwalle61" title="Maintenance">🚧</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=rwalle61" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=rwalle61" title="Documentation">📖</a> <a href="https://github.com/openapi-library/OpenAPIValidators/pulls?q=is%3Apr+reviewed-by%3Arwalle61" title="Reviewed Pull Requests">👀</a></td>
    <td align="center"><a href="https://github.com/JonnySpruce"><img src="https://avatars3.githubusercontent.com/u/30812276?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Jonny Spruce</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=JonnySpruce" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=JonnySpruce" title="Documentation">📖</a> <a href="https://github.com/openapi-library/OpenAPIValidators/pulls?q=is%3Apr+reviewed-by%3AJonnySpruce" title="Reviewed Pull Requests">👀</a></td>
    <td align="center"><a href="https://github.com/AlexDobeck"><img src="https://avatars2.githubusercontent.com/u/10519388?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Alex Dobeck</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=AlexDobeck" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3AAlexDobeck" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://github.com/BenGu3"><img src="https://avatars2.githubusercontent.com/u/7105857?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ben Guthrie</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=BenGu3" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3ABenGu3" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://martijnvegter.com/"><img src="https://avatars3.githubusercontent.com/u/25134477?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Martijn Vegter</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=mvegter" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/ludeknovy"><img src="https://avatars1.githubusercontent.com/u/13610612?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ludek</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=ludeknovy" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3Aludeknovy" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://github.com/tgiardina"><img src="https://avatars1.githubusercontent.com/u/37459104?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Tommy Giardina</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=tgiardina" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3Atgiardina" title="Bug reports">🐛</a></td>
  </tr>
  <tr>
    <td align="center"><a href="https://xotabu4.github.io/"><img src="https://avatars3.githubusercontent.com/u/3033972?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Oleksandr Khotemskyi</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=Xotabu4" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/amitkeinan9"><img src="https://avatars.githubusercontent.com/u/16577335?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Amit Keinan</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=amitkeinan9" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/DetachHead"><img src="https://avatars.githubusercontent.com/u/57028336?v=4?s=100" width="100px;" alt=""/><br /><sub><b>DetachHead</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3ADetachHead" title="Bug reports">🐛</a></td>
    <td align="center"><a href="http://karlssonkristoffer.com/"><img src="https://avatars.githubusercontent.com/u/20490202?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Kristoffer Karlsson</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=kristofferkarlsson93" title="Documentation">📖</a></td>
  </tr>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->
 readmeEtag: '"4d5a529f7a19b52fab62aaf17e9c5263bd765e2d"' readmeLastModified: Tue, 29 Mar 2022 10:12:21 GMT repositoryId: 199614125 description: Use Jest or Chai to assert that HTTP responses satisfy an OpenAPI spec created: '2019-07-30T08:56:01Z' updated: '2025-12-11T11:06:52Z' language: TypeScript archived: false stars: 194 watchers: 11 forks: 38 owner: openapi-library logo: https://avatars.githubusercontent.com/u/53815441?v=4 license: MIT repoEtag: '"982859aab5b874db7602c4d6d41812452f2b57b69b87a00845a5040995271994"' repoLastModified: Thu, 11 Dec 2025 11:06:52 GMT foundInMaster: true id: e77dbc0bddfdcddf31c2fb1cff63cb6c - source: https://openapi.tools/ name: hikaku category: Testing language: Kotlin repository: https://github.com/codecentric/hikaku source_description: >- A library that tests if the implementation of a REST-API meets its specification. v2: false v3: true repositoryMetadata: base64Readme: >- IyBoaWtha3UKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2FwaS50cmF2aXMtY2kub3JnL2NvZGVjZW50cmljL2hpa2FrdS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9jb2RlY2VudHJpYy9oaWtha3UpIFshW01hdmVuIENlbnRyYWwgVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3YvZGUuY29kZWNlbnRyaWMuaGlrYWt1L2hpa2FrdS1jb3JlLnN2ZyldKGh0dHBzOi8vc2VhcmNoLm1hdmVuLm9yZy9zZWFyY2g/cT1nOmRlLmNvZGVjZW50cmljLmhpa2FrdSkKCjxwIGFsaWduPSJjZW50ZXIiPgogIDxpbWcgc3JjPSJkb2NzL2ltYWdlcy9oaWtha3UtbG9nby1zbWFsbC5wbmciPgo8L3A+CgpIaWtha3UgKOavlOi8gykgaXMgamFwYW5lc2UgYW5kIG1lYW5zICJjb21wYXJpc29uIi4gVGhpcyBsaWJyYXJ5IHRlc3RzIGlmIGEgUkVTVC1BUEkgaW1wbGVtZW50YXRpb24gbWVldHMgaXRzIHNwZWNpZmljYXRpb24uCgpJZiB5b3UgY3JlYXRlIHlvdXIgUkVTVC1BUEkgY29udHJhY3QtZmlyc3Qgd2l0aG91dCB1c2luZyBhbnkgdHlwZSBvZiBnZW5lcmF0aW9uLCB5b3UgaGF2ZSB0byBtYWtlIHN1cmUgdGhhdCBzcGVjaWZpY2F0aW9uIGFuZCBpbXBsZW1lbnRhdGlvbiBkb24ndCBkaXZlcmdlLgpUaGUgYWltIG9mIHRoaXMgcHJvamVjdCBpcyB0byBtZWV0IHRoaXMgbmVlZCBhbmQgb2ZmZXIgYSBtZWNoYW5pc20gdG8gY2hlY2sgc3BlY2lmaWNhdGlvbiBhbmQgaW1wbGVtZW50YXRpb24gZm9yIGVxdWFsaXR5IHdpdGhvdXQgaGF2aW5nIHRvIGNyZWF0ZSByZXF1ZXN0cyB3aGljaCBhcmUgZmlyZWQgYWdhaW5zdCBhIG1vY2sgc2VydmVyLiBTbyB0aGlzIGxpYnJhcnkgd29uJ3QgY2hlY2sgdGhlIGJlaGF2aW9yIG9mIHRoZSBBUEksIGJ1dCB0aGUgc3RydWN0dXJhbCBjb3JyZWN0bmVzcy4gUGxlYXNlIHNlZSBhbHNvIHRoZSBzZWN0aW9uIFtsaW1pdGF0aW9uc10oI2xpbWl0YXRpb25zKQoKIyMgQ3VycmVudGx5IHN1cHBvcnRlZAoKKyAqKlNwZWNpZmljYXRpb25zKioKICArIFtPcGVuQVBJIDMuMC5YXShvcGVuYXBpL1JFQURNRS5tZCkKICArIFtSQU1MIDEuWF0ocmFtbC9SRUFETUUubWQpCiAgKyBbV0FETF0od2FkbC9SRUFETUUubWQpCisgKipJbXBsZW1lbnRhdGlvbnMqKgogICsgW1NwcmluZyBNVkMgNS4zLlhdKHNwcmluZy9SRUFETUUubWQpCiAgKyBbTWljcm9uYXV0XShtaWNyb25hdXQvUkVBRE1FLm1kKQogICsgW0pBWC1SUyAzLjAuWF0oamF4LXJzL1JFQURNRS5tZCkKICAgICsgW0FwYWNoZSBDWEZdKGh0dHA6Ly9jeGYuYXBhY2hlLm9yZykKICAgICsgW0Ryb3B3aXphcmRdKGh0dHBzOi8vd3d3LmRyb3B3aXphcmQuaW8pCiAgICArIFtKZXJzZXldKGh0dHBzOi8vamVyc2V5LmdpdGh1Yi5pbykKICAgICsgW1Jlc3RlYXN5XShodHRwczovL3Jlc3RlYXN5LmdpdGh1Yi5pbykKICAgICsgW1Jlc3RsZXRdKGh0dHBzOi8vcmVzdGxldC5jb20vb3Blbi1zb3VyY2UvZG9jdW1lbnRhdGlvbi91c2VyLWd1aWRlLzIuMy9leHRlbnNpb25zL2pheHJzKQogICAgKyBbUXVhcmt1c10oaHR0cHM6Ly9xdWFya3VzLmlvKQogIApQbGVhc2UgcmVmZXIgdG8gdGhlIGxpc3Qgb2YgW2FsbCBmZWF0dXJlc10oZG9jcy9mZWF0dXJlcy5tZCkuIFRvIGNoZWNrIHRoZSBmZWF0dXJlIHN1cHBvcnQgZm9yIGVhY2ggY29udmVydGVyLgpJdCBpcyBwb3NzaWJsZSB0aGF0IG5vdCBldmVyeSBjb252ZXJ0ZXIgc3VwcG9ydHMgZXZlcnkgZmVhdHVyZS4gT25seSB0aGUgaW50ZXJzZWN0aW9uIG9mIHRoZSBmZWF0dXJlcyBvZiB0d28gYEVuZHBvaW50Q29udmVydGVyYHMgaXMgdXNlZCBmb3IgdGhlIG1hdGNoaW5nLiBQbGVhc2Uga2VlcCB0aGF0IGluIG1pbmQgcmVnYXJkaW5nIHRoZSBlcXVhbGl0eSBvZiBpbXBsZW1lbnRhdGlvbiBhbmQgc3BlY2lmaWNhdGlvbi4KICAKIyMgVXNhZ2UKClNldHRpbmcgdXAgYSB0ZXN0IHdpdGggaGlrYWt1IGlzIHZlcnkgc2ltcGxlLiBZb3UganVzdCBpbnN0YW50aWF0ZSB0aGUgYEhpa2FrdWAgY2xhc3MgYW5kIHByb3ZpZGUgYW4gYEVuZHBvaW50Q29udmVydGVyYCBmb3IgdGhlIHNwZWNpZmljYXRpb24gYW5kIGFub3RoZXIgb25lIGZvciB0aGUgaW1wbGVtZW50YXRpb24uIE9wdGlvbmFsbHksIHlvdSBjYW4gYWxzbyBwYXNzIGFuIGluc3RhbmNlIG9mIGBIaWtha3VDb25maWdgLiBDaGVjayB0aGUgbGlzdCBvZiBvcHRpb25zIGFuZCBkZWZhdWx0IHZhbHVlcyBvZiB0aGUgW2NvbmZpZ10oZG9jcy9jb25maWcubWQpLiBUaGVuIHlvdSBjYWxsIGBtYXRjaCgpYCBvbiB0aGUgYEhpa2FrdWAgY2xhc3MuClRoZSBtYXRjaCByZXN1bHQgaXMgc2VudCB0byBvbmUgb3IgbXVsdGlwbGUgYFJlcG9ydGVyYC4gSWYgdGhlIHRlc3QgZmFpbHMga290bGluJ3MgYERlZmF1bHRBc3NlcnRlci5mYWlsKClgIG1ldGhvZCBpcyBjYWxsZWQuCgojIyMgRXhhbXBsZQoKVGhlcmUgaXMgYW4gYXJ0aWZhY3QgZm9yIGVhY2ggY29udmVydGVyLiBTbyB3ZSBuZWVkIG9uZSBkZXBlbmRlbmN5IGZvciB0aGUgc3BlY2lmaWNhdGlvbiBhbmQgb25lIGZvciB0aGUgaW1wbGVtZW50YXRpb24uIEluIHRoaXMgZXhhbXBsZSBvdXIgcHJvamVjdCBjb25zaXN0cyBvZiBhbiBPcGVuQVBJIHNwZWNpZmljYXRpb24gYW5kIGEgU3ByaW5nIGltcGxlbWVudGF0aW9uLiBUaGUgc3BlY2lmaWNhdGlvbiBkb2VzIG5vdCBjb250YWluIHRoZSBfL2Vycm9yXyBlbmRwb2ludHMgY3JlYXRlZCBieSBzcHJpbmcsIHNvIHdlIHdhbnQgdG8gb21pdCB0aG9zZS4KRmlyc3QgYWRkIHRoZSBkZXBlbmRlbmNpZXMgZm9yIHRoZSBjb252ZXJ0ZXJzLCB0aGF0IHdlIHdhbnQgdG8gdXNlLiBJbiB0aGlzIGNhc2UgYGhpa2FrdS1vcGVuYXBpYCBhbmQgYGhpa2FrdS1zcHJpbmdgLgoKYGBgZ3JhZGxlCmRlcGVuZGVuY2llcyB7CiAgICB0ZXN0SW1wbGVtZW50YXRpb24gImRlLmNvZGVjZW50cmljLmhpa2FrdTpoaWtha3Utb3BlbmFwaTokaGlrYWt1VmVyc2lvbiIKICAgIHRlc3RJbXBsZW1lbnRhdGlvbiAiZGUuY29kZWNlbnRyaWMuaGlrYWt1Omhpa2FrdS1zcHJpbmc6JGhpa2FrdVZlcnNpb24iCn0KYGBgCgojIyMjIEtvdGxpbgoKQW5kIG5vdyB3ZSBjYW4gY3JlYXRlIHRoZSB0ZXN0IGNhc2U6CgpgYGBrb3RsaW4KQFNwcmluZ0Jvb3RUZXN0CmNsYXNzIFNwZWNpZmljYXRpb25UZXN0IHsKCiAgICBAQXV0b3dpcmVkCiAgICBwcml2YXRlIGxhdGVpbml0IHZhciBzcHJpbmdDb250ZXh0OiBBcHBsaWNhdGlvbkNvbnRleHQKCiAgICBAVGVzdAogICAgZnVuIGBzcGVjaWZpY2F0aW9uIG1hdGNoZXMgaW1wbGVtZW50YXRpb25gKCkgewogICAgICAgIEhpa2FrdSgKICAgICAgICAgICAgICAgIHNwZWNpZmljYXRpb24gPSBPcGVuQXBpQ29udmVydGVyKFBhdGhzLmdldCgib3BlbmFwaS55YW1sIikpLAogICAgICAgICAgICAgICAgaW1wbGVtZW50YXRpb24gPSBTcHJpbmdDb252ZXJ0ZXIoc3ByaW5nQ29udGV4dCksCiAgICAgICAgICAgICAgICBjb25maWcgPSBIaWtha3VDb25maWcoCiAgICAgICAgICAgICAgICAgICAgICAgIGZpbHRlcnMgPSBsaXN0T2YoU3ByaW5nQ29udmVydGVyLklHTk9SRV9FUlJPUl9FTkRQT0lOVCkKICAgICAgICAgICAgICAgICkKICAgICAgICApCiAgICAgICAgLm1hdGNoKCkKICAgIH0KfQpgYGAKCiMjIyMgSmF2YQoKU2FtZSBleGFtcGxlIGluIEphdmE6CgpgYGBqYXZhCkBTcHJpbmdCb290VGVzdApwdWJsaWMgY2xhc3MgU3BlY2lmaWNhdGlvblRlc3QgewoKICBAQXV0b3dpcmVkCiAgcHJpdmF0ZSBBcHBsaWNhdGlvbkNvbnRleHQgc3ByaW5nQ29udGV4dDsKCiAgQFRlc3QKICBwdWJsaWMgdm9pZCBzcGVjaWZpY2F0aW9uX21hdGNoZXNfaW1wbGVtZW50YXRpb24oKSB7CiAgICBMaXN0PEZ1bmN0aW9uMTxFbmRwb2ludCwgQm9vbGVhbj4+IGZpbHRlcnMgPSBuZXcgQXJyYXlMaXN0PD4oKTsKICAgIGZpbHRlcnMuYWRkKFNwcmluZ0NvbnZlcnRlci5JR05PUkVfRVJST1JfRU5EUE9JTlQpOwoKICAgIExpc3Q8UmVwb3J0ZXI+IHJlcG9ydGVycyA9IG5ldyBBcnJheUxpc3Q8PigpOwogICAgcmVwb3J0ZXJzLmFkZChuZXcgQ29tbWFuZExpbmVSZXBvcnRlcigpKTsKCiAgICBuZXcgSGlrYWt1KAogICAgICAgICAgICBuZXcgT3BlbkFwaUNvbnZlcnRlcihQYXRocy5nZXQoIm9wZW5hcGkuanNvbiIpKSwKICAgICAgICAgICAgbmV3IFNwcmluZ0NvbnZlcnRlcihzcHJpbmdDb250ZXh0KSwKICAgICAgICAgICAgbmV3IEhpa2FrdUNvbmZpZygKICAgICAgICAgICAgICAgICAgICByZXBvcnRlcnMsCiAgICAgICAgICAgICAgICAgICAgZmlsdGVycwogICAgICAgICAgICApCiAgICApCiAgICAubWF0Y2goKTsKICB9Cn0KYGBgCgojIyBMaW1pdGF0aW9ucwpIaWtha3UgY2hlY2tzIHRoZSBpbXBsZW1lbnRhdGlvbiB3aXRoIHN0YXRpYyBjb2RlIGFuYWx5c2lzLiBTbyBldmVyeXRoaW5nIHRoYXQgaXMgaGlnaGx5IGR5bmFtaWMgaXMgbm90IGNvdmVyZWQgYnkgaGlrYWt1LiBUaGVyZSBtaWdodCBiZSBvdGhlciBsaWJyYXJpZXMgYW5kIGZyYW1ld29ya3MgdGhhdCBjYW4gY292ZXIgdGhlc2UgYXNwZWN0cyBieSBjaGVja2luZyB0aGUgYmVoYXZpb3IuCgojIyMgaHR0cCBzdGF0dXMgY29kZXMKRm9yIGltcGxlbWVudGF0aW9ucyB0aGUgc3RhdHVzIGNvZGVzIGFyZSB2ZXJ5IGR5bmFtaWMuIFRoZXJlIGFyZSB2YXJpb3VzIHdheXMgdG8gc2V0IGEgaHR0cCBzdGF0dXMuIEZvciBleGFtcGxlIHVzaW5nIGEgYFJlc3BvbnNlRW50aXR5YCBvYmplY3QgaW4gc3ByaW5nIG9yIHVzaW5nIGFkZGl0aW9uYWwgZmlsdGVycyBhbmQgc28gb24uIFRoYXQncyB3aHkgaGlrYWt1IGRvZXMgbm90IHN1cHBvcnQgaHR0cCBzdGF0dXMgY29kZXMuCgojIyMgUmVxdWVzdCBhbmQgcmVzcG9uc2Ugb2JqZWN0CkZvciBpbXBsZW1lbnRhdGlvbnMgYm90aCByZXF1ZXN0IGFuZCByZXNwb25zZSBvYmplY3RzIGFyZSBoaWdobHkgZHluYW1pYy4gRm9yIHJlc3BvbnNlIG9iamVjdHMgdGhlcmUgbWlnaHQgYmUgYSBnZW5lcmljIGBSZXNwb25zZUVudGl0eWAgYXMgd2VsbCBvciBpbnRlcmZhY2VzIHdpdGggZGlmZmVyZW50IGltcGxlbWVudGF0aW9ucyBjYW4gYmUgdXNlZC4gSW4gYm90aCBjYXNlcyAocmVxdWVzdCBhbmQgcmVzcG9uc2UpIHRoZSBvYmplY3RzIGNhbiBiZSBhbHRlcmVkIGJ5IGEgc2VyaWFsaXphdGlvbiBsaWJyYXJ5IGFuZCB0aGVyZSBhIGxvdCBvZiBkaWZmZXJlbnQgbGlicyBvdXQgdGhlcmUuIFRoYXQncyB3aHkgaGlrYWt1IG5laXRoZXIgc3VwcG9ydHMgcmVxdWVzdCBub3IgcmVzcG9uc2Ugb2JqZWN0cy4KCiMjIE1vcmUgSW5mbwoKKiAqKkJsb2cgKGVuZ2xpc2gpOioqIFtTcG90dGluZyBtaXNtYXRjaGVzIGJldHdlZW4geW91ciBzcGVjIGFuZCB5b3VyIFJFU1QtQVBJIHdpdGggaGlrYWt1XShodHRwczovL2Jsb2cuY29kZWNlbnRyaWMuZGUvZW4vMjAxOS8wMy9zcG90LW1pc21hdGNoZXMtYmV0d2Vlbi15b3VyLXNwZWMtYW5kLXlvdXItcmVzdC1hcGkvKQoqICoqQmxvZyAoZ2VybWFuKToqKiBbIEFid2VpY2h1bmdlbiB6d2lzY2hlbiBTcGV6aWZpa2F0aW9uIHVuZCBSRVNULUFQSSBtaXQgaGlrYWt1IGVya2VubmVuXShodHRwczovL2Jsb2cuY29kZWNlbnRyaWMuZGUvMjAxOS8wMy9hYndlaWNodW5nZW4tendpc2NoZW4tcmVzdC1hcGktc3BlemlmaWthdGlvbi1lcmtlbm5lbi8pCiogKipTYW1wbGUgcHJvamVjdCoqIFtBIGNvbXBsZXRlIHNhbXBsZSBwcm9qZWN0XShodHRwczovL2dpdGh1Yi5jb20vY2MtamhyL2hpa2FrdS1zYW1wbGUpCg== readmeEtag: '"9552177ad0f534c779e456e00de9b84f2e23f7b9"' readmeLastModified: Thu, 19 Aug 2021 18:39:18 GMT repositoryId: 167929609 description: >- A library that tests if the implementation of a REST-API meets its specification. created: '2019-01-28T08:54:05Z' updated: '2026-02-02T11:38:29Z' language: Kotlin archived: false stars: 200 watchers: 6 forks: 16 owner: codecentric logo: https://avatars.githubusercontent.com/u/1009716?v=4 license: Apache-2.0 repoEtag: '"58d17b1521736ba35aeab9eccf15c1a744574470963e7e5c7859394624f5540e"' repoLastModified: Mon, 02 Feb 2026 11:38:29 GMT foundInMaster: true id: 0724d08c486c3cc69b4e067f07cdabd8 - source: https://openapi.tools/ name: Tcases for OpenAPI category: Testing language: Java repository: https://github.com/cornutum/tcases/blob/master/tcases-openapi/readme.md source_description: >- Generates test cases directly from an OpenAPI v3 description of your API. Creates tests executable using various test frameworks. Bonus: Semantic linter reports elements that are inconsistent, superfluous, or dubious. v2: false v3: true repositoryMetadata: base64Readme: >- IyBUY2FzZXM6IEEgTW9kZWwtQmFzZWQgVGVzdCBDYXNlIEdlbmVyYXRvciAjCgpbIVtNYXZlbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9tYXZlbi00LjEuMS1ncmVlbi5zdmcpXShodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9dGNhc2VzLXNoZWxsKQpbIVtKYXZhZG9jXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2phdmFkb2MtNC4xLjEtZ3JlZW4uc3ZnKV0oaHR0cHM6Ly9qYXZhZG9jLmlvL2RvYy9vcmcuY29ybnV0dW0udGNhc2VzL3RjYXNlcy1zaGVsbCkKCiMjIFdoYXQncyBOZXc/ICMjCiAgKiBUaGUgbGF0ZXN0IHZlcnNpb24gKFtUY2FzZXMgNC4xLjFdKFJlbGVhc2VOb3Rlcy5tZCM0MTEpKSBpcyBub3cgYXZhaWxhYmxlIGF0IHRoZSBNYXZlbiBDZW50cmFsIFJlcG9zaXRvcnkuCiAgICBTZWUgWypIb3cgVG8gRG93bmxvYWQgVGNhc2VzKl0oSG93VG9Eb3dubG9hZC5tZCkgZm9yIGRvd25sb2FkIGluc3RydWN0aW9ucy4KCiAgKiBIYXZpbmcgdHJvdWJsZSB3aXRoIFRjYXNlcz8gQ2hlY2sgb3V0IFt0aGVzZSB0aXBzXSguL1Ryb3VibGVzaG9vdGluZy1GQVFzLm1kKS4KCiAgKiBHb3QgYSBxdWVzdGlvbj8gTmVlZCBzb21lIGd1aWRhbmNlPyBTdGFydCBhIFtkaXNjdXNzaW9uXShodHRwczovL2dpdGh1Yi5jb20vQ29ybnV0dW0vdGNhc2VzL2Rpc2N1c3Npb25zKS4KCiMjIFdoYXQgRG9lcyBJdCBEbz8gIyMKClRjYXNlcyBpcyBhIHRvb2wgZm9yIGRlc2lnbmluZyB0ZXN0cy4gSXQgZG9lc24ndCBtYXR0ZXIgd2hhdCBraW5kIG9mIHN5c3RlbSB5b3UgYXJlIHRlc3RpbmcgLS0gVUksIGNvbW1hbmQgbGluZSwKW1JFU1QtZnVsIEFQSV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKSwgb3IgYmFja2VuZC4gIE5vciBkb2VzIGl0IG1hdHRlcgp3aGF0IGxldmVsIG9mIHRoZSBzeXN0ZW0geW91IGFyZSB0ZXN0aW5nIC0tIHVuaXQsIHN1YnN5c3RlbSwgb3IgZnVsbCBzeXN0ZW0uIFlvdSBjYW4gdXNlIFRjYXNlcyB0byBkZXNpZ24geW91ciB0ZXN0cyBpbiBhbnkgb2YKdGhlc2Ugc2l0dWF0aW9ucy4gV2l0aCBUY2FzZXMsIHlvdSBkZWZpbmUgdGhlIGlucHV0IHNwYWNlIGZvciB5b3VyIHN5c3RlbS11bmRlci10ZXN0IGFuZCB0aGUgbGV2ZWwgb2YgY292ZXJhZ2UgdGhhdCB5b3UKd2FudC4gVGhlbiBUY2FzZXMgZ2VuZXJhdGVzIGEgbWluaW1hbCBzZXQgb2YgdGVzdCBjYXNlcyB0aGF0IG1lZXRzIHlvdXIgcmVxdWlyZW1lbnRzLgoKVGNhc2VzIGlzIHByaW1hcmlseSBhIHRvb2wgZm9yIGJsYWNrLWJveCB0ZXN0IGRlc2lnbi4gRm9yIHN1Y2ggdGVzdHMsIHRoZSBjb25jZXB0IG9mICJjb3ZlcmFnZSIgaXMgZGlmZmVyZW50IGZyb20gc3RydWN0dXJhbAp0ZXN0aW5nIGNyaXRlcmlhIHN1Y2ggYXMgbGluZSBjb3ZlcmFnZSwgYnJhbmNoIGNvdmVyYWdlLCBldGMuIEluc3RlYWQsIFRjYXNlcyBpcyBndWlkZWQgYnkgY292ZXJhZ2Ugb2YgdGhlIGlucHV0IHNwYWNlIG9mIHlvdXIKc3lzdGVtLgoKVGNhc2VzIGdpdmVzIHlvdSBhIHdheSB0byBkZWZpbmUgdGhlIGlucHV0IHNwYWNlIGZvciB5b3VyIHN5c3RlbSBpbiBhIGZvcm0gdGhhdCBpcyBjb25jaXNlIGJ1dCBjb21wcmVoZW5zaXZlLiBUaGVuIFRjYXNlcyBhbGxvd3MKeW91IHRvIGNvbnRyb2wgdGhlIG51bWJlciBvZiB0ZXN0IGNhc2VzIGluIHlvdXIgc2FtcGxlIHN1YnNldCBieSBzcGVjaWZ5aW5nIHRoZSBsZXZlbCBvZiBjb3ZlcmFnZSB5b3Ugd2FudC4gWW91IGNhbiBzdGFydCB3aXRoIGEKYmFzaWMgbGV2ZWwgb2YgY292ZXJhZ2UsIGFuZCBUY2FzZXMgd2lsbCBnZW5lcmF0ZSBhIHNtYWxsIHNldCBvZiB0ZXN0IGNhc2VzIHRoYXQgdG91Y2hlcyBldmVyeSBzaWduaWZpY2FudCBlbGVtZW50IG9mIHRoZSBpbnB1dApzcGFjZS4gVGhlbiB5b3UgY2FuIGltcHJvdmUgeW91ciB0ZXN0cyBieSBzZWxlY3RpdmVseSBhZGRpbmcgY292ZXJhZ2UgaW4gc3BlY2lmaWMgaGlnaC1yaXNrIGFyZWFzLiBGb3IgZXhhbXBsZSwgeW91IGNhbiBzcGVjaWZ5CnBhaXJ3aXNlIGNvdmVyYWdlIG9yIGhpZ2hlci1vcmRlciBjb21iaW5hdGlvbnMgb2Ygc2VsZWN0ZWQgaW5wdXQgdmFyaWFibGVzLgoKIyMgSG93IERvZXMgSXQgV29yaz8gIyMKCkZpcnN0LCB5b3UgY3JlYXRlIGEgc3lzdGVtIGlucHV0IGRlZmluaXRpb24sIGEgZG9jdW1lbnQgdGhhdCBkZWZpbmVzIHlvdXIgc3lzdGVtIGFzIGEgc2V0IG9mIGZ1bmN0aW9ucy4gRm9yIGVhY2ggc3lzdGVtCmZ1bmN0aW9uLCB0aGUgc3lzdGVtIGlucHV0IGRlZmluaXRpb24gZGVmaW5lcyB0aGUgdmFyaWFibGVzIHRoYXQgY2hhcmFjdGVyaXplIHRoZSBmdW5jdGlvbiBpbnB1dCBzcGFjZS4gSWYgeW91IGFyZSB0ZXN0aW5nIGEgV2ViCnNlcnZpY2UgQVBJLCB5b3UgY2FuIGV2ZW4gW2dlbmVyYXRlIGEgc3lzdGVtIGlucHV0IGRlZmluaXRpb24gYXV0b21hdGljYWxseV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKQpmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbi4KClRoZW4sIHlvdSBjYW4gY3JlYXRlIGEgZ2VuZXJhdG9yIGRlZmluaXRpb24uIFRoYXQncyBhbm90aGVyIGRvY3VtZW50IHRoYXQgZGVmaW5lcyB0aGUgY292ZXJhZ2UgeW91IHdhbnQgZm9yIGVhY2ggc3lzdGVtCmZ1bmN0aW9uLiBUaGUgZ2VuZXJhdG9yIGRlZmluaXRpb24gaXMgb3B0aW9uYWwuIFlvdSBjYW4gc2tpcCB0aGlzIHN0ZXAgYW5kIHN0aWxsIGdldCBhIGJhc2ljIGxldmVsIG9mIGNvdmVyYWdlLgoKRmluYWxseSwgeW91IHJ1biBUY2FzZXMuIFRjYXNlcyBpcyBhIEphdmEgcHJvZ3JhbSB0aGF0IHlvdSBjYW4gcnVuIGZyb20gdGhlIGNvbW1hbmQgbGluZSBvciB1c2luZyB0aGUKW1RjYXNlcyBNYXZlbiBQbHVnaW5dKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL3RjYXNlcy1tYXZlbi1wbHVnaW4vKS4gVGhlIGNvbW1hbmQgbGluZSB2ZXJzaW9uIG9mIFRjYXNlcyBjb21lcyB3aXRoIGJ1aWx0LWluCnN1cHBvcnQgZm9yIHJ1bm5pbmcgdXNpbmcgYSBzaGVsbCBzY3JpcHQgb3IgYW4gYW50IHRhcmdldC4gVXNpbmcgeW91ciBpbnB1dCBkZWZpbml0aW9uIGFuZCB5b3VyIGdlbmVyYXRvciBkZWZpbml0aW9uLCBUY2FzZXMKZ2VuZXJhdGVzIGEgc3lzdGVtIHRlc3QgZGVmaW5pdGlvbi4gVGhlIHN5c3RlbSB0ZXN0IGRlZmluaXRpb24gaXMgYSBkb2N1bWVudCB0aGF0IGxpc3RzLCBmb3IgZWFjaCBzeXN0ZW0gZnVuY3Rpb24sIGEgc2V0IG9mIHRlc3QKY2FzZXMgdGhhdCBwcm92aWRlcyB0aGUgc3BlY2lmaWVkIGxldmVsIG9mIGNvdmVyYWdlLiBFYWNoIHRlc3QgY2FzZSBkZWZpbmVzIGEgc3BlY2lmaWMgdmFsdWUgZm9yIGV2ZXJ5IGZ1bmN0aW9uIGlucHV0CnZhcmlhYmxlLiBUY2FzZXMgZ2VuZXJhdGVzIG5vdCBvbmx5IHZhbGlkIGlucHV0IHZhbHVlcyB0aGF0IGRlZmluZSBzdWNjZXNzZnVsIHRlc3QgY2FzZXMgYnV0IGFsc28gaW52YWxpZCB2YWx1ZXMgZm9yIHRoZSB0ZXN0cwpjYXNlcyB0aGF0IGFyZSBuZWVkZWQgdG8gdmVyaWZ5IGV4cGVjdGVkIGVycm9yIGhhbmRsaW5nLgoKT2YgY291cnNlLCB0aGUgc3lzdGVtIHRlc3QgZGVmaW5pdGlvbiBpcyBub3Qgc29tZXRoaW5nIHlvdSBjYW4gZXhlY3V0ZSBkaXJlY3RseS4gKFVubGVzcyBpdCB3YXMKW2Rlcml2ZWQgYXV0b21hdGljYWxseSBmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbl0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI2hvdy1kby15b3UtcnVuLWdlbmVyYXRlZC1hcGktdGVzdC1jYXNlcykhKQpCdXQgaXQgZm9sbG93cyBhIHdlbGwtZGVmaW5lZCBzY2hlbWEsIHdoaWNoIG1lYW5zIHlvdSBjYW4gdXNlIGEgdmFyaWV0eSBvZiB0cmFuc2Zvcm1hdGlvbiB0b29scyB0byBjb252ZXJ0IGl0IGludG8gYSBmb3JtIHRoYXQKaXMgc3VpdGFibGUgZm9yIHRlc3RpbmcgeW91ciBzeXN0ZW0uIEZvciBleGFtcGxlLCBUY2FzZXMgY29tZXMgd2l0aCBhIGJ1aWx0LWluIHRyYW5zZm9ybWVyIHRoYXQgY29udmVydHMgYSBzeXN0ZW0gdGVzdApkZWZpbml0aW9uIGludG8gYSBKYXZhIHNvdXJjZSBjb2RlIHRlbXBsYXRlIGZvciBhIEpVbml0IG9yIFRlc3RORyB0ZXN0IGNsYXNzLgoKIyMgR2V0IFN0YXJ0ZWQhICMjCgogICogKipUaGUgTG93ZG93bioqCiAgICAqIFtUY2FzZXM6IFRoZSBDb21wbGV0ZSBHdWlkZV0oLi9UY2FzZXMtR3VpZGUubWQjdGNhc2VzLXRoZS1jb21wbGV0ZS1ndWlkZSkKICAgICogW1RjYXNlcyBmb3IgT3BlbkFQSV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKTogVGVzdGluZyBhIFJFU1QtZnVsIEFQST8gR2VuZXJhdGUgdGVzdCBjYXNlcyBkaXJlY3RseSBmcm9tIHlvdXIgT3BlbkFQSSB2MyBkZWZpbml0aW9uLgogICAgKiBbVGhlIFRjYXNlcyBNYXZlbiBQbHVnaW5dKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL3RjYXNlcy1tYXZlbi1wbHVnaW4vKQoKICAqICoqSGVscGZ1bCBHdWlkZXMqKgogICAgKiBbSG93IFRvIERvd25sb2FkIFVzaW5nIE1hdmVuXShIb3dUb0Rvd25sb2FkLm1kKQogICAgKiBbSG93IFRvIFNldHVwIGEgVGNhc2VzIFdlYiBTZXJ2aWNlXSguL1RjYXNlcy1XZWItU2VydmljZS5tZCkKICAgICogW1VzaW5nIFRoZSBUY2FzZXMgQVBJXSguL1VzaW5nLVRjYXNlcy1BUEkubWQpCiAgICAqIFtUcm91Ymxlc2hvb3RpbmcgRkFRXSguL1Ryb3VibGVzaG9vdGluZy1GQVFzLm1kI3Ryb3VibGVzaG9vdGluZy1mYXFzKQogICAgKiBbUmVsZWFzZSBOb3Rlc10oUmVsZWFzZU5vdGVzLm1kKQoKICAqICoqTW9yZSBJbmZvKioKICAgICogW01vZGVsLURyaXZlbiBUZXN0aW5nIFVzaW5nIFRjYXNlc10oTW9kZWxEcml2ZW5UZXN0aW5nRm9yQWdpbGVUZWFtcy5tZCkKICAgICogSmF2YWRvYzogW1RjYXNlcyBBUEldKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL2FwaS9pbmRleC5odG1sKQoKIyMgQ29udHJpYnV0b3JzICMjCgpUaGFua3MgdG8gdGhlIGZvbGxvd2luZyBwZW9wbGUsIHdobyBoYXZlIGNvbnRyaWJ1dGVkIHNpZ25pZmljYW50IGltcHJvdmVtZW50cyB0byBUY2FzZXMuCgogICogW0tlcnJ5IEtpbWJyb3VnaF0oaHR0cHM6Ly9naXRodWIuY29tL2tlcnJ5a2ltYnJvdWdoKSAocHJvamVjdCBmb3VuZGVyKQogICogW0p1Z2xhcl0oaHR0cHM6Ly9naXRodWIuY29tL2p1Z2xhcikKICAqIFtUaGliYXVsdCBLcnVzZV0oaHR0cHM6Ly9naXRodWIuY29tL3RrcnVzZSkK readmeEtag: '"ca9a75dabe602853f00d0e6ee6f9a7326d7f4433"' readmeLastModified: Thu, 27 Nov 2025 22:47:20 GMT repositoryId: 32223373 description: A model-based test case generator created: '2015-03-14T17:22:38Z' updated: '2025-12-18T16:57:03Z' language: Java archived: false stars: 236 watchers: 16 forks: 56 owner: Cornutum logo: https://avatars.githubusercontent.com/u/11477145?v=4 license: MIT repoEtag: '"a1d286eef641716d283e62ff50d46ad41827aaf523f41391a8f106f17a7d330b"' repoLastModified: Thu, 18 Dec 2025 16:57:03 GMT foundInMaster: true id: 64dc602f01798a9a39aba18c09d655ae - source: https://openapi.tools/ name: StackHawk HawkScan category: Security language: SaaS repository: https://github.com/stackhawk source_description: >- StackHawk is an application vulnerability scanner purpose built for developers to use in the DevOps pipeline. It leverages a provided OpenAPI v2 or v3 spec file for route discovery and enhanced scanning. link: https://stackhawk.com/ v2: true v3: true foundInMaster: true id: d27194a62778c331d662fae462dc8045 - source: https://openapi.tools/ name: FireTail category: Security language: SaaS repository: https://github.com/firetail-io source_description: >- FireTail provides discovery, logging, posture management and in-line enforcement of APIs using OpenAPI. API governance is backed by cloud provider integrations and a suite of open-source application libraries. link: https://www.firetail.io/ v2: true v3: true id: cbc88089f856e67eed142049e8a38063 foundInMaster: true - source: https://openapi.tools/ name: 42crunch category: Security language: SaaS source_description: >- A unique set of integrated API security tools that allow discovery, remediation of OpenAPI vulnerabilities and runtime protection against API attacks. link: https://42crunch.com/ v2: true v3: true foundInMaster: true id: 1b7c824e66c86452d1843ccac1465b71 - source: https://openapi.tools/ name: openapi-fuzzer category: - Security - Parsers language: Rust source_description: >- Based on OpenAPI specification, openapi-fuzzer provides random data as inputs to the API endpoints in order to find bugs. link: https://github.com/matusf/openapi-fuzzer v2: false v3: true foundInMaster: true repository: https://github.com/matusf/openapi-fuzzer repositoryMetadata: base64Readme: >- # OpenAPI fuzzer

![ci](https://github.com/matusf/openapi-fuzzer/actions/workflows/ci.yml/badge.svg)

Black-box fuzzer that fuzzes APIs based on [OpenAPI specification](https://github.com/OAI/OpenAPI-Specification/). All you need to do is to supply URL of the API and its specification. Find bugs for free!

![image](https://user-images.githubusercontent.com/18228995/225413315-eab08df2-ed56-4b7a-8c8a-027c18d9a106.png)

## Findings

The fuzzer has been used to find bugs in numerous software. Some of the well-known fuzzed software include[^1]:

<details><summary><b>Kubernetes</b></summary>

- [kubenetes#101350](https://github.com/kubernetes/kubernetes/issues/101350)
- [kubenetes#101348](https://github.com/kubernetes/kubernetes/issues/101348)
- [kubenetes#101355](https://github.com/kubernetes/kubernetes/issues/101355)

</details>

<details><summary><b>Gitea</b></summary>

- [gitea#15357](https://github.com/go-gitea/gitea/issues/15357)
- [gitea#15356](https://github.com/go-gitea/gitea/issues/15356)
- [gitea#15346](https://github.com/go-gitea/gitea/issues/15346)

</details>

<details><summary><b>Vault</b></summary>

- [vault#11310](https://github.com/hashicorp/vault/issues/11310)
- [vault#11311](https://github.com/hashicorp/vault/issues/11311)
- [vault#11313](https://github.com/hashicorp/vault/issues/11313)

</details>

The category of bugs differ, but some of the common are parsing bugs, invalid format bugs and querying non-existent entities. **If you have found bugs with this fuzzer, please reach out to me. I would love to hear from you.** Feel free to submit a PR and add your finding to the list above.

## Building & installing

### From crates.io

To build the fuzzer, you will need to have [rust installed](https://www.rust-lang.org/learn/get-started).

```sh
cargo install openapi-fuzzer
```

### From source

```sh
git clone git@github.com:matusf/openapi-fuzzer.git
cd openapi-fuzzer

# Install to the $PATH
cargo install --path .

# Or build inside the repo
cargo build --release
```

### Using containers

```sh
podman pull ghcr.io/matusf/openapi-fuzzer
```

## Usage

After installation you will have the `openapi-fuzzer` binary available to you, which offers two subcommands - `run` and `resend`.  The `run` subcommand will fuzz the API according to the specification and report any findings. All findings will be stored in a JSON format in a `results` directory (the name of the directory can be specified by `--results-dir` flag).

If the fuzzer finds a bug it will save the seed that leads to the generation of the payload triggering the bug. Those seeds are saved in a regressions file called `openapi-fuzzer.regressions`. The seeds will be used in the next runs of the fuzzer to check if the bug persists. You shall save it alongside your project.

When you are done with fuzzing, you can use `openapi-fuzzer resend` to resend payloads that triggered bugs and examine the cause in depth.

OpenAPI fuzzer supports version 3 of the OpenAPI specification in YAML or JSON format. You can convert older versions at [editor.swagger.io](https://editor.swagger.io/).

### Tips

- When the fuzzer receives an unexpected status code, it will report it as a finding. However, many APIs do not specify client error status codes in the specification. To minimize false positive findings ignore status codes that you are not interested in with `-i` flag. It is advised to fuzz it in two stages. Firstly, run the fuzzer without `-i` flag. Then check the `results` folder for the reported findings. If there are reports from status codes you do not care about, add them via `-i` flag and rerun the fuzzer.
- Most APIs use some base prefix for endpoints like `/v1` or `/api`, however, the specifications are sometimes written without it. Do not forget to **include the path prefix in the url**.
- You may add an extra header with `-H` flag. It may be useful when you would like to increase coverage by providing some sort of authorization. You can use the `-H` flag to add cookies too. e.g. `-H "Cookie: A=1;"`. Use a single `-H` flag when adding multiple cookies as well. e.g. `-H "Cookie: A=1; B=2; C=3;"`.
- Currently, the fuzzer makes 256 requests per endpoint. If all received responses are expected, it declares the endpoint as ok and continues to fuzz the next one. You can adjust this number by setting a `--max-test-case-count` flag.
- To disable the verification of TLS certificates and thus use, for example, self-signed certificates, you can use the `--skip-tls-verify` flag.
- By default, the fuzzer uses rate limiting. If it receives an HTTP status code of 429 or 503, it will wait for a number of seconds specified by the `Retry-After` header. If the header is not present, it will use an exponential backoff algorithm with a starting value of 1 second. After 10 unsuccessful retries, fuzzing of the endpoint is aborted.

```console
$ openapi-fuzzer run --help
Usage: openapi-fuzzer run -s <spec> -u <url> [-i <ignore-status-code>] [-H <header>] [--max-test-case-count <max-test-case-count>] [-o <results-dir>] [--stats-dir <stats-dir>] [--skip-tls-verify] [--no-rate-limiting]

run openapi-fuzzer

Options:
  -s, --spec        path to OpenAPI specification file
  -u, --url         url of api to fuzz
  -i, --ignore-status-code
                    status codes that will not be considered as finding
  -H, --header      additional header to send
  --max-test-case-count
                    maximum number of test cases that will run for each
                    combination of endpoint and method (default: 256)
  -o, --results-dir directory for results with minimal generated payload used
                    for resending requests (default: results).
  --stats-dir       directory for request times statistics. if no value is
                    supplied, statistics will not be saved
  --skip-tls-verify disable verification of TLS certificates
  --no-rate-limiting
                    do not use rate limiting
  --help            display usage information

$ openapi-fuzzer run -s ./spec.yaml -u http://127.0.0.1:8200/v1/ -i 404 -i 400 -H  "Authorization: Basic QWxhZGRpbjpvcGVuIHNlc2FtZQ=="
```

### Replaying findings

When you are done fuzzing you can replay the findings. All findings are stored in the `results` folder. Name of each file consists of concatenated endpoint, HTTP method and received status code. To resend the same payload to API, you need to run `openapi-fuzzer resend` and specify a path to the finding file as an argument and a url of the api. You can overwrite the headers with `-H` flag as well, which is useful, when you need authorization.

```console
$ ls -1 results/
api-v1-componentstatuses-{name}-GET-500.json
api-v1-namespaces-{namespace}-configmaps-GET-500.json
api-v1-namespaces-{namespace}-endpoints-GET-500.json
api-v1-namespaces-{namespace}-events-GET-500.json
api-v1-namespaces-{namespace}-limitranges-GET-500.json
api-v1-namespaces-{namespace}-persistentvolumeclaims-GET-500.json
api-v1-namespaces-{namespace}-pods-GET-500.json
api-v1-namespaces-{namespace}-podtemplates-GET-500.json
api-v1-namespaces-{namespace}-replicationcontrollers-GET-500.json
api-v1-namespaces-{namespace}-resourcequotas-GET-500.json
api-v1-namespaces-{namespace}-secrets-GET-500.json
api-v1-namespaces-{namespace}-serviceaccounts-GET-500.json
api-v1-namespaces-{namespace}-services-GET-500.json
api-v1-watch-namespaces-{name}-GET-500.json
...

$ openapi-fuzzer resend --help
Usage: openapi-fuzzer resend <file> [-H <header...>] -u <url>

resend payload genereted by fuzzer

Positional Arguments:
  file              path to result file generated by fuzzer

Options:
  -H, --header      extra header
  -u, --url         url of api
  --help            display usage information

$ openapi-fuzzer resend --url https://minikubeca:8443 results/api-v1-componentstatuses-\{name\}-GET-500.json -H "Authorization: Bearer $KUBE_TOKEN" | jq
500 (Internal Server Error)
{
  "kind": "Status",
  "apiVersion": "v1",
  "metadata": {},
  "status": "Failure",
  "message": "Component not found: ኊ0",
  "code": 500
}
```

[^1]: not all found bugs are linked
 readmeEtag: '"c9b6f61b75afa2e8df985085f45893790dd4841b"' readmeLastModified: Mon, 16 Oct 2023 16:56:16 GMT repositoryId: 324014659 description: >- Black-box fuzzer that fuzzes APIs based on OpenAPI specification. Find bugs for free! created: '2020-12-23T22:33:38Z' updated: '2026-01-31T14:18:25Z' language: Rust archived: false stars: 574 watchers: 8 forks: 24 owner: matusf logo: https://avatars.githubusercontent.com/u/18228995?v=4 license: AGPL-3.0 repoEtag: '"69ef901b1f62c0be6d950eb8b44891d1e3485c2b732290f9dd1b87f607499eda"' repoLastModified: Sat, 31 Jan 2026 14:18:25 GMT id: 9e0191ba47403298459b30242dbfe9ad - source: https://openapi.tools/ name: cats category: - Security - Server Implementations language: Java source_description: >- CATS is a REST API Fuzzer and negative testing tool for OpenAPI endpoints. CATS automatically generates, runs and reports tests with minimum configuration and no coding effort. Tests are self-healing and do not require maintenance. link: https://github.com/Endava/cats v2: true v3: true v3_1: true repository: https://github.com/endava/cats id: a6ff4bc9abd488fc56bf9b04747b4051 repositoryMetadata: base64Readme: >- <p align="center">
<picture>
  <source media="(prefers-color-scheme: dark)" srcset="https://raw.githubusercontent.com/Endava/cats/master/images/cats_logo_dark.svg" />
  <img alt="CATS logo" src="https://raw.githubusercontent.com/Endava/cats/master/images/cats_logo_light.svg" width="400">
</picture>
</p>
<p align="center">

![CI](https://img.shields.io/github/actions/workflow/status/Endava/cats/main.yml?style=for-the-badge&logo=git&logoColor=white)
[![Commits](https://img.shields.io/github/commit-activity/m/Endava/cats?style=for-the-badge&logo=git&logoColor=white)](https://github.com/Endava/cats/pulse)

[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)
[![Java Version](https://img.shields.io/badge/Java-25+-blue.svg)](https://openjdk.org)
[![GraalVM](https://img.shields.io/badge/GraalVM-Native-orange.svg)](https://www.graalvm.org)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=cats&metric=alert_status)](https://sonarcloud.io/dashboard?id=cats)
[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=cats&metric=coverage)](https://sonarcloud.io/dashboard?id=cats)
[![Bugs](https://sonarcloud.io/api/project_badges/measure?project=cats&metric=bugs)](https://sonarcloud.io/dashboard?id=cats)
[![Code Smells](https://sonarcloud.io/api/project_badges/measure?project=cats&metric=code_smells)](https://sonarcloud.io/dashboard?id=cats)

</p>

> CATS documentation is available at [https://endava.github.io/cats/](https://endava.github.io/cats/)

**REST API fuzzer and negative testing tool. Run thousands of self-healing API tests within minutes with no coding effort!**

- **Comprehensive**: tests are generated automatically based on a large number scenarios and cover **every** field and header
- **Intelligent**: tests are generated based on data types and constraints; each Fuzzer has specific expectations depending on the scenario under test
- **Highly Configurable**: high amount of customization: you can filter specific Fuzzers, HTTP response codes, HTTP methods, request paths, provide business context and a lot more
- **Self-Healing**: as tests are generated, any OpenAPI spec change is picked up automatically
- **Simple to Learn**: flat learning curve, with intuitive configuration and syntax
- **Fast**: automatic process for write, run and report tests which covers thousands of scenarios within minutes

<p align="center"></p>

> Short on time? Check out the [1-minute Quick Start Guide](https://endava.github.io/cats/docs/intro)!

# Overview
By using a simple and minimal syntax, with a flat learning curve, CATS (**C**ontract **A**PI **T**esting and **S**ecurity) enables you to generate thousands of API tests within minutes with **no coding effort**.
All tests are **generated, run and reported automatically** based on a pre-defined set of **100+ Fuzzers**. 
The Fuzzers cover a wide range of boundary testing and negative scenarios from fully random large Unicode values to well crafted, context dependant values based on the request data types and constraints. 
Even more, you can leverage the fact that CATS generates request payloads dynamically and write simple end-to-end functional tests.

## HTML Report

<div align="center">
  <img alt="CATS" width="100%" src="images/tests_result.png"/>
</div>

## Command Line

<div align="center">
  <img alt="CATS" width="100%" src="images/run_result.png"/>
</div>

# Tutorials on how to use CATS

This is a list of articles with step-by-step guides on how to use CATS:
* [Testing the GitHub API with CATS](https://ludovicianul.github.io/posts/github-api-testing/)
* [How to write self-healing functional tests with no coding effort](https://ludovicianul.github.io/posts/self-healing-api-tests/)

# Some bugs found by CATS

- https://github.com/hashicorp/vault/issues/13274 | https://github.com/hashicorp/vault/issues/13273
- https://github.com/hashicorp/vault/issues/13225 | https://github.com/hashicorp/vault/issues/13232
- https://github.com/go-gitea/gitea/issues/19397  | https://github.com/go-gitea/gitea/issues/19398
- https://github.com/go-gitea/gitea/issues/19399

# Installation

## Homebrew

```shell
> brew tap endava/tap
> brew install cats
```

## Manual
CATS is bundled both as an executable JAR or a native binary. The native binaries do not need Java installed. 

After downloading your OS native binary, you can add it to PATH so that you can execute it as any other command line tool:

```shell
sudo cp cats /usr/local/bin/cats
```

You can also get autocomplete by downloading the [cats_autocomplete](cats_autocomplete) script and do:

```shell
source cats_autocomplete
```

To get persistent autocomplete, add the above line in `.zshrc` or `.bashrc`, but make sure you put the fully qualified path for the `cats_autocomplete` script.

You can also check the `cats_autocomplete` source for alternative setup.

There is no native binary for Windows, but you can use the uberjar version. This requires Java 25+ to be installed.

You can run it as `java -jar cats.jar`.

Head to the releases page to download the latest version: [https://github.com/Endava/cats/releases](https://github.com/Endava/cats/releases).

## Build from sources

You can build CATS from sources on you local box. You need [Java 21+](https://sdkman.io/jdks). Maven is already bundled.

> Before running the first build, please make sure you do a `./mvnw clean`. CATS uses a fork of [OKHttp](https://square.github.io/okhttp/) which will install locally
under the `4.11.0-CATS` version, so don't worry about overriding the official versions.


You can use the following Maven command to build the project as an uberjar:

`./mvnw package -Dquarkus.package.type=uber-jar`


You will end up with a `cats-runner.jar` in the `target` folder. You can run it with `java -jar cats-runner.jar ...`.

You can also build native images using a [GraalVM Java version](https://www.graalvm.org/).

`./mvnw package -Pnative`

### Notes on Unit Tests

You may see some `error` log messages while running the Unit Tests. Those are expected behaviour for testing the negative scenarios of the Fuzzers.

## Experimental: Maven dependency for programmatic use
CATS doesn't have explicit support (yet) for programmatic use via JUnit or TestNG. 
You can however experiment with running the `CatsMain` class with the same arguments as you would run in the command line.

You must add these 2 dependencies:

```xml
<dependency>
    <groupId>com.squareup.okhttp3</groupId>
    <artifactId>okhttp</artifactId>
    <version>4.11.0</version>
</dependency>
<dependency>
    <groupId>com.endava</groupId>
    <artifactId>cats</artifactId>
    <version>9.0.3</version>
</dependency>
```

> Please not that you also need to explicitly add the `okhttp` dependency. 
> CATS uses a fork of okhttp that is not published in Maven central.
> When using CATS as a dependency, HTTP header fuzzers that prefix/suffix header values with spaces won't properly work.

# Contributing
Please refer to [CONTRIBUTING.md](CONTRIBUTING.md). 
 readmeEtag: '"5d0117aed79f3ba2392f1c634fd683e55d18d9b0"' readmeLastModified: Tue, 03 Feb 2026 21:15:13 GMT repositoryId: 252459854 description: >- CATS is a REST API Fuzzer and negative testing tool for OpenAPI endpoints. CATS automatically generates, runs and reports tests with minimum configuration and no coding effort. Tests are self-healing and do not require maintenance. created: '2020-04-02T13:14:39Z' updated: '2026-02-05T18:42:13Z' language: Java archived: false stars: 1339 watchers: 10 forks: 86 owner: Endava logo: https://avatars.githubusercontent.com/u/19396140?v=4 license: Apache-2.0 repoEtag: '"b312a91f917e4fe64705828560fad085dbcc717bb452463fc63c1bd49df4c934"' repoLastModified: Thu, 05 Feb 2026 18:42:13 GMT foundInMaster: true - source: https://openapi.tools/ name: API Insights category: Security language: SaaS source_description: >- RestCase executes hundrends of security and quality checks against the API definition, the API insights report provides detailed security scoring for prioritization, and remediation advice to help developers define the best API definition possible. link: https://restcase.com/platform/security v2: true v3: true foundInMaster: true id: aab65db9be3090f79ce65c7fc9808dce - source: https://openapi.tools/ name: OpenAPI Schema to JSON Schema category: - Converters - Parsers language: TypeScript source_description: >- Due to the OpenAPI v3.0 and JSON Schema discrepancy, you can use this JS library to convert OpenAPI Schema objects to proper JSON Schema. link: https://www.npmjs.com/package/@openapi-contrib/openapi-schema-to-json-schema repository: https://github.com/openapi-contrib/openapi-schema-to-json-schema v3: true repositoryMetadata: base64Readme: >- # OpenAPI Schema to JSON Schema

A little NodeJS package to convert OpenAPI Schema Object or Parameter Object to JSON Schema.

Currently converts from [OpenAPI 3.0](https://spec.openapis.org/oas/v3.0.3.html) to [JSON Schema Draft 4](http://json-schema.org/specification-links.html#draft-4).

[![Treeware](https://img.shields.io/badge/dynamic/json?color=brightgreen&label=Treeware&query=%24.total&url=https%3A%2F%2Fpublic.offset.earth%2Fusers%2Ftreeware%2Ftrees)](https://treeware.earth)

## Why?

OpenAPI is a specification for describing RESTful APIs. OpenAPI v3.0 allows us to describe the structures of request and response payloads in a detailed manner. This would, theoretically, mean that we should be able to automatically validate request and response payloads. However, as of writing there aren't many validators around.

The good news is that there are many validators for JSON Schema for different languages. The bad news is that OpenAPI v3.0 is [not entirely compatible with JSON Schema](https://stoplight.io/blog/openapi-json-schema/). The Schema Object of OpenAPI v3.0 is an extended subset of JSON Schema Specification Wright Draft 00 with some differences. This will be resolved in OpenAPI v3.1, but until then... this tool will fill that gap.

There is also a [CLI tool](https://github.com/mikunn/openapi2schema) for creating a JSON of schemas from the whole API specification.

If you need to do the conversion in reverse, checkout [json-schema-to-openapi-schema](https://github.com/openapi-contrib/json-schema-to-openapi-schema).

## Features

- converts OpenAPI v3.0 Schema Object to JSON Schema Draft 4
- converts OpenAPI v3.0 Parameter Object to JSON Schema Draft 4
- deletes `nullable` and adds `"null"` to `type` array if `nullable` is `true`
- supports deep structures with nested `allOf`s etc.
- removes [OpenAPI specific properties](https://spec.openapis.org/oas/v3.0.3.html#fixed-fields-20) such as `discriminator`, `deprecated` etc. unless specified otherwise
- optionally supports `patternProperties` with `x-patternProperties` in the Schema Object

**NOTE**: `$ref`s are not handled in any way, so please use a resolver such as [json-schema-ref-parser](https://github.com/APIDevTools/json-schema-ref-parser) or [swagger-cli bundle](https://www.npmjs.com/package/swagger-cli) prior to using this package.

## Installation

```
npm install --save @openapi-contrib/openapi-schema-to-json-schema
```

### CLI

```bash
npx "@openapi-contrib/openapi-schema-to-json-schema" --input openapi.json --output json-schema.json
```

## Converting OpenAPI schema

Here's a small example to get the idea:

```js
var { openapiSchemaToJsonSchema: toJsonSchema } = require("@openapi-contrib/openapi-schema-to-json-schema");

var schema = {
  type: "string",
  format: "date-time",
  nullable: true,
};

var convertedSchema = toJsonSchema(schema);

console.log(convertedSchema);
```

The example prints out

```js
{
  type: ['string', 'null'],
  format: 'date-time',
  '$schema': 'http://json-schema.org/draft-04/schema#'
}
```

Provide the function the schema object with possible options.

### Options

The function accepts `options` object as the second argument.

#### `cloneSchema` (boolean)

If set to `false`, converts the provided schema in place. If `true`, clones the schema by converting it to JSON and back. The overhead of the cloning is usually negligible. Defaults to `true`.

#### `dateToDateTime` (boolean)

This is `false` by default and leaves `date` format as is. If set to `true`, sets `format: 'date'` to `format: 'date-time'`.

For example

```js
var schema = {
  type: "string",
  format: "date",
};

var convertedSchema = toJsonSchema(schema, { dateToDateTime: true });

console.log(convertedSchema);
```

prints

```js
{
  type: 'string',
  format: 'date-time',
  '$schema': 'http://json-schema.org/draft-04/schema#'
}
```

#### `keepNotSupported` (array)

By default, the following fields are removed from the result schema: `nullable`, `discriminator`, `readOnly`, `writeOnly`, `xml`, `externalDocs`, `example` and `deprecated` as they are not supported by JSON Schema Draft 4. Provide an array of the ones you want to keep (as strings) and they won't be removed.

#### `removeReadOnly` (boolean)

If set to `true`, will remove properties set as `readOnly`. If the property is set as `required`, it will be removed from the `required` array as well. The property will be removed even if `readOnly` is set to be kept with `keepNotSupported`.

#### `removeWriteOnly` (boolean)

Similar to `removeReadOnly`, but for `writeOnly` properties.

#### `supportPatternProperties` (boolean)

If set to `true` and `x-patternProperties` property is present, change `x-patternProperties` to `patternProperties` and call `patternPropertiesHandler`. If `patternPropertiesHandler` is not defined, call the default handler. See `patternPropertiesHandler` for more information.

#### `patternPropertiesHandler` (function)

Provide a function to handle pattern properties and set `supportPatternProperties` to take effect. The function takes the schema where `x-patternProperties` is defined on the root level. At this point `x-patternProperties` is changed to `patternProperties`. It must return the modified schema.

If the handler is not provided, the default handler is used. If `additionalProperties` is set and is an object, the default handler sets it to false if the `additionalProperties` object has deep equality with a pattern object inside `patternProperties`. This is because we might want to define `additionalProperties` in OpenAPI spec file, but want to validate against a pattern. The pattern would turn out to be useless if `additionalProperties` of the same structure were allowed. Create you own handler to override this functionality.

See `test/pattern_properties.test.js` for examples how this works.

#### `definitionKeywords` (array)

By default, definitions are not converted. If your documents follow the convention of having a definitions object at the root of a (sub)schema, you can set definitionKeywords to `['definitions']`.

```js
var schema = {
  definitions: {
    sharedDefinition: {
      type: "object",
      properties: {
        foo: {
          type: "string",
          nullable: true,
        },
      },
    },
  },
};
var convertedSchema = toJsonSchema(schema, {
  definitionKeywords: ["definitions"],
});
console.log(convertedSchema);
```

prints

```js
{
  $schema: 'http://json-schema.org/draft-04/schema#',
  definitions: {
    sharedDefinition: {
      type: 'object',
      properties: {
        foo: {
          type: ['string', 'null']
        }
      }
    }
  }
}
```

## Converting OpenAPI parameters

OpenAPI parameters can be converted:

```js
var { fromParameter } = require("@openapi-contrib/openapi-schema-to-json-schema");

var param = {
  name: "parameter name",
  in: "query",
  schema: {
    type: "string",
    format: "date",
  },
};

var convertedSchema = fromParameter(param);

console.log(convertedSchema);
```

The result is as follows:

```js
{
  type: 'string',
  format: 'date',
  '$schema': 'http://json-schema.org/draft-04/schema#'
}
```

When a parameter has several schemas (one per MIME type) a map is returned instead.

```js
{
  name: 'parameter name',
  in: 'query',
  content: {
    'application/javascript': {
      schema: {
        type: 'string'
      }
    },
    'text/css': {
      schema: {
        type: 'string'
      }
    }
  }
}
```

would be converted to:

```js
{
  'application/javascript': {
    type: 'string',
    '$schema': 'http://json-schema.org/draft-04/schema#'
  },
  'text/css': {
    type: 'string',
    '$schema': 'http://json-schema.org/draft-04/schema#'
  }
}
```

## Treeware

This package is [Treeware](https://treeware.earth). If you use it in production, then we ask that you [**buy the world a tree**](https://plant.treeware.earth/{venfor}/{package}) to thank us for our work. By contributing to the Treeware forest you’ll be creating employment for local families and restoring wildlife habitats.

## Thanks

- [Stoplight][] for donating time and effort to this project, and many more.
- [mikunn][] for originally creating this package.
- [All Contributors][link-contributors]

[stoplight]: https://stoplight.io/
[mikunn]: https://github.com/mikunn
[link-contributors]: https://github.com/openapi-contrib/openapi-schema-to-json-schema/graphs/contributors

## Copyright

Copyright 2023 the [OpenAPI Contrib organization](https://github.com/openapi-contrib). Code released under the [MIT License](https://github.com/openapi-contrib/openapi-schema-to-json-schema/blob/main/LICENSE).
 readmeEtag: '"48037924913e4772a488531513892eaf0a796630"' readmeLastModified: Tue, 11 Jul 2023 22:18:10 GMT repositoryId: 234640931 description: >- Due to the OpenAPI v3.0 and JSON Schema discrepancy, you can use this JS library to convert OpenAPI Schema objects to proper JSON Schema. created: '2020-01-17T21:37:32Z' updated: '2026-01-15T20:55:18Z' language: TypeScript archived: false stars: 268 watchers: 4 forks: 21 owner: openapi-contrib logo: https://avatars.githubusercontent.com/u/49447320?v=4 license: MIT repoEtag: '"a9d1a4d7a0246ba9828f031cb60de411257add82e0dbcf3d5d728b037f7bd43d"' repoLastModified: Thu, 15 Jan 2026 20:55:18 GMT foundInMaster: true id: f6608d418f20d73a1e15b74206edd45a - source: https://openapi.tools/ name: JSON Schema to OpenAPI Schema category: - Converters - Parsers language: TypeScript source_description: >- Due to the OpenAPI v3.0 and JSON Schema discrepancy, you can use this JS library to convert JSON Schema objects to OpenAPI Schema. link: https://www.npmjs.com/package/@openapi-contrib/json-schema-to-openapi-schema repository: https://github.com/openapi-contrib/json-schema-to-openapi-schema v3: true repositoryMetadata: base64Readme: >- # JSON Schema to OpenAPI Schema

A little NodeJS package to convert JSON Schema to a [OpenAPI Schema Object](http://spec.openapis.org/oas/v3.0.3.html#schema-object).

[![Treeware](https://img.shields.io/badge/dynamic/json?color=brightgreen&label=Treeware&query=%24.total&url=https%3A%2F%2Fpublic.offset.earth%2Fusers%2Ftreeware%2Ftrees)](https://treeware.earth)

## Features

- converts JSON Schema Draft 04 to OpenAPI 3.0 Schema Object
- switches `type: ['foo', 'null']` to `type: foo` and `nullable: true`
- supports deep structures with nested `allOf`s etc.
- switches `patternProperties` to `x-patternProperties`
- converts `dependencies` to an allOf + oneOf OpenAPI-valid equivalent

## Installation

```shell
npm install --save @openapi-contrib/json-schema-to-openapi-schema
```

Requires NodeJS v10 or greater.

## Usage

Here's a small example to get the idea:

```ts
import { convert } from '@openapi-contrib/json-schema-to-openapi-schema';

const schema = {
	$schema: 'http://json-schema.org/draft-04/schema#',
	type: ['string', 'null'],
	format: 'date-time',
};

(async () => {
	const convertedSchema = await convert(schema);
	console.log(convertedSchema);
})();
```

The example prints out

```js
{
  type: 'string',
  format: 'date-time',
  nullable: true
}
```

### Options

The function accepts `options` object as the second argument.

#### `cloneSchema` (boolean)

If set to `false`, converts the provided schema in place. If `true`, clones the schema by converting it to JSON and back. The overhead of the cloning is usually negligible. Defaults to `true`.

#### `dereference` (boolean)

If set to `true`, all local and remote references (http/https and file) $refs will be dereferenced. Defaults to `false`.

#### `convertUnreferencedDefinitions` (boolean)

Defaults to true.

If a schema had a definitions property (which is valid in JSONSchema), and only some of those entries are referenced, we'll still try and convert the remaining definitions to OpenAPI. If you do not want this behavior, set this to `false`.

#### `dereferenceOptions` (object = $RefParser.Options)

Options to pass to the dereferencer (@apidevtools/json-schema-ref-parser). To prevent circular references, pass `{ dereference: { circular: 'ignore' } }`.

## Command Line

```sh
Usage:
    json-schema-to-openapi-schema <command> [options] <file>

Commands:
    convert                 Converts JSON Schema Draft 04 to OpenAPI 3.0 Schema Object

Options:
    -h, --help              Show help for any command
    -v, --version           Output the CLI version number
    -d, --dereference       If set all local and remote references (http/https and file) $refs will be dereferenced
```

## Synchronous API

For synchronous conversion without async operations, use `convertSync`:

```ts
import { convertSync } from '@openapi-contrib/json-schema-to-openapi-schema';

const schema = {
	$schema: 'http://json-schema.org/draft-04/schema#',
	type: ['string', 'null'],
	format: 'date-time',
};

const convertedSchema = convertSync(schema);
console.log(convertedSchema);
```

This will output the same result as the async version:

```js
{
  type: 'string',
  format: 'date-time',
  nullable: true
}
```

### Options for convertSync

`convertSync` supports a subset of the options available to `convert`:

#### `cloneSchema` (boolean)

If set to `false`, converts the provided schema in place. If `true`, clones the schema by converting it to JSON and back. The overhead of the cloning is usually negligible. Defaults to `true`.

#### `convertUnreferencedDefinitions` (boolean)

Defaults to `true`.

If a schema has a `definitions` property (valid in JSON Schema), and only some entries are referenced, we'll still try to convert the remaining definitions to OpenAPI. If you do not want this behavior, set this to `false`.

### ⚠️ Limitations

`convertSync` does **not** support dereferencing external references (`$ref` to HTTP/HTTPS URLs or external files). If your schema contains external references, use the async `convert` function with the `dereference: true` option instead.

#### Example:

```ts
// Schema with external reference
const schema = {
	type: 'object',
	properties: {
		user: {
			$ref: 'https://example.com/schemas/user.json',
		},
	},
};

// convertSync leaves the reference unresolved
const result = convertSync(schema);
// Result: { type: 'object', properties: { user: { $ref: 'https://example.com/schemas/user.json' } } }

// Use async convert for proper resolution:
const resolved = await convert(schema, { dereference: true });
```

**Note:** External references will remain as `$ref` pointers in the output schema. They will not be expanded or validated.

## Why?

OpenAPI is often described as an extension of JSON Schema, but both specs have changed over time and grown independently. OpenAPI v2 was based on JSON Schema draft v4 with a long list of deviations, but OpenAPI v3 shrank that list, upping their support to draft v4 and making the list of discrepancies shorter. This has been solved for OpenAPI v3.1, but for those using OpenAPI v3.0, you can use this tool to solve [the divergence](https://apisyouwonthate.com/blog/openapi-and-json-schema-divergence).

![Diagram showing data model (the objects, payload bodies, etc) and service model (endpoints, headers, metadata, etc)](https://cdn-images-1.medium.com/max/1600/0*hijIL-3Xa5EFZ783.png)

This tool sets out to allow folks to convert from JSON Schema (their one source of truth for everything) to OpenAPI (a thing for HTML docs and making SDKs).

## Versions

- **From:** [JSON Schema Draft v5 †](http://json-schema.org/specification-links.html#draft-5)
- **To:** [OpenAPI 3.0](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md)

_† Draft v5 is also known as Draft Wright 00, as the drafts are often named after the author, and this was the first one by A. Wright. Amongst other things, draft v5 aimed to rewrite the meta files, but the experiment failed, meaning we need to continue to use the draft v4 metafiles. Ugh._

## Converting Back

To convert the other way, check out [openapi-schema-to-json-schema], which this package was based on.

## Tests

To run the test-suite:

```shell
npm test
```

## Treeware

This package is [Treeware](https://treeware.earth). If you use it in production, then we ask that you [**buy the world a tree**](https://plant.treeware.earth/{venfor}/{package}) to thank us for our work. By contributing to the Treeware forest you’ll be creating employment for local families and restoring wildlife habitats.

## Thanks

- [Stoplight][] for [donating time and effort](https://stoplight.io/blog/companies-supporting-open-source/) to this project, and many more.
- [mikunn][] for creating [openapi-schema-to-json-schema] which this is based on.
- [Phil Sturgeon][] for flipping that conversion script about face.
- [WeWork][] for giving this a home for a while.
- [All Contributors][link-contributors]

[mikunn]: https://github.com/mikunn
[wework]: https://github.com/wework
[stoplight]: https://stoplight.io/
[phil sturgeon]: https://github.com/philsturgeon
[openapi-schema-to-json-schema]: https://github.com/openapi-contrib/openapi-schema-to-json-schema
[link-contributors]: https://github.com/openapi-contrib/json-schema-to-openapi-schema/graphs/contributors
 readmeEtag: '"a70d7726cfb7f722c93771e5e2e69f5e783d0caa"' readmeLastModified: Mon, 01 Sep 2025 20:03:04 GMT repositoryId: 212850148 description: >- Due to the OpenAPI v3.0 and JSON Schema discrepancy, you can use this JS library to convert JSON Schema objects to OpenAPI Schema. created: '2019-10-04T15:44:03Z' updated: '2026-01-24T04:05:33Z' language: TypeScript archived: false stars: 122 watchers: 4 forks: 18 owner: openapi-contrib logo: https://avatars.githubusercontent.com/u/49447320?v=4 license: MIT repoEtag: '"401cfab09d727365951bf31322f1f4b9b95c9132c0e9fb3540ab04ce2f65c2f3"' repoLastModified: Sat, 24 Jan 2026 04:05:33 GMT foundInMaster: true id: c443b19d564b9a8e21fe8b6192f75f8f - source: - https://openapi.tools/ - openapi3 tags name: Unchase.OpenAPI.Connectedservice category: - SDK - Code Generators link: https://github.com/unchase/Unchase.OpenAPI.Connectedservice repository: https://github.com/unchase/unchase.openapi.connectedservice language: .NET source_description: >- Visual Studio extension to generate C# (TypeScript) HttpClient (or C# Controllers) code for OpenAPI web service with NSwag. v2: true v3: true repositoryMetadata: base64Readme: >- ![Logo](img/Unchase-OpenAPI-Swagger-Connected-Service-Logo.png)

[Unchase OpenAPI (Swagger) Connected Service](https://marketplace.visualstudio.com/items?itemName=Unchase.unchaseOpenAPIConnectedService) is a Visual Studio 2017/2019/2022 extension to generate `C#` (`TypeScript`) `HttpClient` (or `C#` `Controllers`) code for `OpenAPI` (formerly [`Swagger API`](https://swagger.io/docs/specification/about/)) web service with [NSwag](https://github.com/RSuter/NSwag).

> Starting from Visual Studio Community 2019 v16.1.3 extensions based on `Microsoft Connected Services` now work fine.

> The project is developed and maintained by [Nikolay Chebotov (**Unchase**)](https://github.com/unchase).

## Getting Started

#### **[Read How-To on medium.com](https://medium.com/@unchase/how-to-generate-c-or-typescript-client-code-for-openapi-swagger-specification-d882d59e3b77)**

Install from `Tools -> Extensions and Updates` menu inside [Visual Studio](https://visualstudio.microsoft.com/vs/) 2017 (for [VisualStudio](https://visualstudio.microsoft.com/vs/) 2019: `Extensions -> Manage Extensions`) or [download](http://vsixgallery.com/extensions/Unchase.OpenAPI.ConnectedService.63199638-6211-4285-ba8f-75b1f0326c2a/extension.vsix)  as `VSIX` package from VSGallery or [download](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService)  as `VSIX` package from [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=Unchase.unchaseopenapiconnectedservice):

![Adding Unchase OpenAPI (Swagger) Connected Service in Visual Studio](img/Unchase-OpenAPI-Swagger-Connected-Service.gif)

## Builds status

|Status|Value|
|:----|:---:|
|Build|[![Build status](https://ci.appveyor.com/api/projects/status/90oewanfh32fjcr6)](https://ci.appveyor.com/project/unchase/unchase.openapi.connectedservice)
|Buid History|![Build history](https://buildstats.info/appveyor/chart/unchase/unchase-openapi-connectedservice)
|GitHub Release|[![GitHub release](https://img.shields.io/github/release/unchase/Unchase.OpenAPI.Connectedservice.svg)](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/releases/latest)
|GitHub Release Date|[![GitHub Release Date](https://img.shields.io/github/release-date/unchase/Unchase.OpenAPI.Connectedservice.svg)](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/releases/latest)
|GitHub Release Downloads|[![Github Releases](https://img.shields.io/github/downloads/unchase/Unchase.OpenAPI.Connectedservice/total.svg)](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/releases/latest)
|VS Marketplace|[![VS Marketplace](http://vsmarketplacebadge.apphb.com/version-short/unchase.UnchaseOpenAPIConnectedService.svg)](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService)
|VS Marketplace Downloads|[![VS Marketplace Downloads](http://vsmarketplacebadge.apphb.com/downloads-short/unchase.UnchaseOpenAPIConnectedService.svg)](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService)
|VS Marketplace Installs|[![VS Marketplace Installs](http://vsmarketplacebadge.apphb.com/installs-short/unchase.UnchaseOpenAPIConnectedService.svg)](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService)

## Features

- Generate `C#` or `TypeScript` clients/proxies (client code) from Swagger 2.0 and OpenAPI 3.0 specifications
- Generate `C#` ASP.NET Controller from Swagger 2.0 and OpenAPI 3.0 specifications
- **Experimental**: Generate `C#` or `TypeScript` clients/proxies (client code) or ASP.NET Controller from OData specification converted to OpenAPI based on [OpenAPI.NET.OData](https://github.com/microsoft/OpenAPI.NET.OData)
- Generate `.nswag` file for using in [`NSwagStudio`](https://github.com/NSwag/NSwag/wiki/NSwagStudio) (no need to install for generating)
- Add required dependencies for the `C#` client (before generating):
	- Library targeting .NET Standard 1.4+:
		1. Newtonsoft.Json ([NuGet](https://www.nuget.org/packages/Newtonsoft.Json))
		2. System.Net.Http ([NuGet](https://www.nuget.org/packages/System.Net.Http))
		3. System.ComponentModel.Annotations ([NuGet](https://www.nuget.org/packages/System.ComponentModel.Annotations))
	- Library targeting the full .NET:
		1. Newtonsoft.Json ([NuGet](https://www.nuget.org/packages/Newtonsoft.Json))
		2. System.Runtime.Serialization (GAC)
		3. System.ComponentModel.DataAnnotations (GAC)
	- Library targeting PCL 259 (Portable Class Library):
		1. Newtonsoft.Json ([NuGet](https://www.nuget.org/packages/Newtonsoft.Json))
		2. Microsoft.Net.Http ([NuGet](https://www.nuget.org/packages/Microsoft.Net.Http))
		3. Portable.DataAnnotations ([NuGet](https://www.nuget.org/packages/Portable.DataAnnotations))
- Add Required dependences for the `C#` controller (before generating):
	1. Microsoft.AspNetCore.Mvc ([NuGet](https://www.nuget.org/packages/Microsoft.AspNetCore.MVC))
- **Command** to open generated `.nswag` and `.nswag.json` files in [NSWagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio)
- **Command** to compare `.nswag.json` specification file with another `.nswag.json` specification file (or specification given by `endpoint`)
- Storage of the last 10 endpoints (specification path)

## Settings Meaning

Meaning of the Unchase [OpenAPI (Swagger) Connected Service](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService) settings according to [NSwagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio):

![Unchase OpenAPI (Swagger) Connected Service settings meaning](img/Unchase-OpenAPI-Swagger-Connected-Service-Settings-Meaning.png)

## Exclude type names

Since [v1.4.0](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/releases/tag/v1.4.0) you can exclude type names in separate Window for C# client code generation:

![Unchase OpenAPI (Swagger) Connected Service - exclude type names](img/Csharp-Client-Generation-Exclude-Type-Names.png)

## Generate code from OData specification converted to OpenAPI specification

Since [v1.5.0](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/releases/tag/v1.4.0) you can generate code from OData specification converted to OpenAPI specification:

![Unchase OpenAPI (Swagger) Connected Service - generate from OData](img/Csharp-Client-Generation-OData.png)

## Custom Commands

### `Open in NSwagStudio` Command

Since *v1.1.** have been added menu command embedded in Visual Studio Solution Explorer context menu lets you open generated `.nswag` and `.nswag.json` files in [NSwagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio).

This extension is for those times where you generate `.nswag` and `.nswag.json` files and you want to be able to quickly open it in [NSwagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio).

#### Prerequisite

> In order to use this extension, you must have [Visual Studio](https://visualstudio.microsoft.com/vs/) 2017/2019, this connected service as well as [NSwagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio) installed.

#### Solution Explorer

You can open `.nswag` and `.nswag.json` files in [NSWagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio) by simply right-clicking it in Solution Explorer and select **Open in NSwagStudio**:

![Open in NSwagStudio menu Command](img/OpenWithNSwagCommandMenu.png)

#### Path to NSwagStudio.exe

If you installed [NSwagStudio](https://github.com/NSwag/NSwag/wiki/NSwagStudio) at a non-default location, a prompt will ask for the path to `NSwagStudio.exe`.

You can always change the location in *Tools -> Options -> Web -> Unchase OpenAPI (Swagger) Connected Service*:

![Open in NSwagStudio Option](img/UnchaseOpenAPIConnectedServiceCommandsOptions1.png)

### `Compare OpenAPI-specifications...` Command

Since *v1.2.** have been added menu command embedded in Visual Studio Solution Explorer context menu lets you compare generated `.nswag.json` specification-file with another `.nswag.json` specification-file (or with specification given by `endpoint`).

This extension is for those times where you generate `.nswag.json` file and you want to quickly compare it with another specification or specification given by `endpoint`.

#### Prerequisite

> In order to use this extension, you must have [Visual Studio](https://visualstudio.microsoft.com/vs/) 2017/2019 as well as this connected service.

#### Solution Explorer

You can compare `.nswag.json` specification-file with another `.nswag.json` specification-file (or with specification given by `endpoint`) by simply selecting one or two files and right-clicking them in Solution Explorer and select **Compare OpenAPI-specifications...**:

![Compare OpenAPI Specifications Command](img/CompareOpenAPISpecificationsOneFileCommandMenu.png) ![Compare OpenAPI Specifications Command](img/CompareOpenAPISpecificationsTwoFilesCommandMenu.png)

#### Path to the specification `Endpoint`

You can always change the specification Endpoint to compare with in *Tools -> Options -> Web -> Unchase OpenAPI (Swagger) Connected Service*:

![Compare OpenAPI Specifications Option](img/UnchaseOpenAPIConnectedServiceCommandsOptions2.png)

#### Compare View

![Compare OpenAPI Specifications Command result](img/CompareOpenAPISpecificationsCommandResult.png)

## HowTos

- [ ] Add HowTos in a future
- [ ] ... [request for HowTo you need](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/issues/new?title=DOC)

## Troubleshooting

### Can't open .nswag file in NSwagStudio

- You can use **Open in NSwagStudio** menu command
- If generated code corrupted, try to open `.nswag` file in [`NSwagStudio`](https://github.com/RSuter/NSwag/wiki/NSwagStudio) (Windows GUI for editing .*nswag files)
- If it doesn't open, try to create new `.nswag` file in [`NSwagStudio`](https://github.com/RSuter/NSwag/wiki/NSwagStudio) for the same API service link and check the differences

### Installation completes but I can't see the Service in the list of connected services (Visual Studio 2019)

- Relevant [bug report](https://developercommunity.visualstudio.com/content/problem/468751/vs2019-preview-cannot-install-connected-service-ex.html). `Connected Services` restored in the v16.1.3 update to [Visual Studio](https://visualstudio.microsoft.com/vs/) 2019.

## Roadmap

See the [changelog](CHANGELOG.md) for the further development plans and version history.

## Feedback

Please feel free to add your [review](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService&ssr=false#review-details), [request a feature](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/issues/new?title=FEATURE), [ask a question](https://marketplace.visualstudio.com/items?itemName=unchase.unchaseOpenAPIConnectedService&ssr=false#qna) or [report a bug](https://github.com/unchase/Unchase.OpenAPI.Connectedservice/issues/new?title=BUG) including in connected service: 

![Unchase OpenAPI Connected Service Report a Bug](img/Unchase-OpenAPI-Connected-Service-ReportBug.png)

Thank you in advance!

## Thank me!

If you like what I am doing and you would like to thank me, please consider:

[![Buy me a coffe!](img/buymeacoffe.png)](https://www.buymeacoffee.com/nikolaychebotov)

Thank you for your support!

----------

Copyright &copy; 2019 [Nikolay Chebotov (**Unchase**)](https://github.com/unchase) - Provided under the [Apache License 2.0](LICENSE.md).

 readmeEtag: '"de7939daae9afa578f8259c0e47d0ad3c1d2ccf3"' readmeLastModified: Tue, 20 Jun 2023 19:32:43 GMT repositoryId: 178596119 description: >- :scroll: Visual Studio extension to generate OpenAPI (Swagger) web service reference. created: '2019-03-30T18:26:16Z' updated: '2025-12-21T00:20:36Z' language: C# archived: false stars: 109 watchers: 2 forks: 19 owner: unchase logo: https://avatars.githubusercontent.com/u/29679226?v=4 license: Apache-2.0 repoEtag: '"e2d11d3dd70fa8844aee1283ec9f9f13d1ea4946e3c919e752c8bf3f375bff40"' repoLastModified: Sun, 21 Dec 2025 00:20:36 GMT foundInMaster: true id: b552c4cab06ab1125ad610aaea022ee3 - source: https://openapi.tools/ name: VSCode OpenAPI Preview category: - Documentation - Parsers language: TypeScript link: https://marketplace.visualstudio.com/items?itemName=zoellner.openapi-preview repository: https://github.com/zoellner/openapi-preview source_description: >- Preview OpenAPI v3.0 & v3.1 documents in Visual Studio Code using Spotlight Elements. v2: false v3: true v3_1: true id: b17945a4cfa653f43157e65867d551ec repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXByZXZpZXcKCkludGVncmF0ZWQgV2ViIFBhbmVsIHByZXZpZXcgZm9yIGFuIFtPcGVuQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKQpPcGVuQVBJIEZvcm1hdCAzLjEKCiMjIEZlYXR1cmVzCgoqIERpc3BsYXlzIEFQSSBkb2NzIGluIGEgYnVpbGQgaW4gd2ViIHBhbmVsIHVzaW5nIFtTcG90bGlnaHQgRWxlbWVudHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9wbGlnaHRpby9lbGVtZW50cykuCgoKIyMgVXNhZ2UKCiogT3BlbiB0aGUgbWFpbiAqT3BlbkFQSSBZQU1MIGZpbGUqLgoqIFJ1biB0aGUgQ29tbWFuZCBg4oen4oyYUGAgKG9uIE1hYykgb3IgYEN0cmwrU2hpZnQrUGAgKG9uIFdpbmRvd3MpICoqT3BlbkFQSSBQcmV2aWV3KiouCg== readmeEtag: '"7cd2c62d79bed77fc8ed34039610334c16c1f1d3"' readmeLastModified: Thu, 11 Jul 2024 06:28:46 GMT repositoryId: 160854009 description: OpenAPI Preview Extension for VS Code created: '2018-12-07T17:16:12Z' updated: '2025-12-25T15:32:08Z' language: TypeScript archived: false stars: 8 watchers: 2 forks: 2 owner: zoellner logo: https://avatars.githubusercontent.com/u/2665319?v=4 license: MIT repoEtag: '"6620f9f2746bfc58297bcdbde9b66dce7ad632a52c91b2c87f260bb9afb589eb"' repoLastModified: Thu, 25 Dec 2025 15:32:08 GMT foundInMaster: true - source: https://openapi.tools/ name: openapi-processor category: Server language: Java link: https://docs.openapiprocessor.io repository: https://github.com/openapi-processor source_description: > Generates java interfaces & model classes for Spring Boot (annotation based, MVC & WebFlux) or Micronaut (annotation based) from an OpenAPI yaml description. Provides powerful type mapping capabilities to adjust the generated code. Gradle & Maven support. Playground. v2: false v3: true v3_1: true id: 8dde59828e2a316fa7a7f1f7f038cea6 foundInMaster: true - source: https://openapi.tools/ name: openapi-processor/openapi-parser category: - Parsers - Schema Validators language: Java link: https://github.com/openapi-processor/openapi-parser repository: https://github.com/openapi-processor/openapi-parser source_description: > OpenAPI parser with (json schema) validation (currently 3.0 only). Separate apis for OpenAPI 3.0 & 3.1. Easily get resolved $ref objects. Pluggable document reader & json/yaml converter. Minimal dependencies. v2: false v3: true v3_1: true id: 22335853514b99484180f312b1db1fa1 repositoryMetadata: base64Readme: >- Om9wZW5hcGk6IGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8KOnBhcnNlcjogaHR0cHM6Ly9naXRodWIuY29tL29wZW5hcGktcHJvY2Vzc29yL29wZW5hcGktcGFyc2VyL3RyZWUvbWFzdGVyL29wZW5hcGktcGFyc2VyCjpwYXJzZXItYm9tOiBodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1wcm9jZXNzb3Ivb3BlbmFwaS1wYXJzZXIvdHJlZS9tYXN0ZXIvb3BlbmFwaS1wYXJzZXItYm9tCjp2YWxpZGF0b3I6IGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYXBpLXByb2Nlc3Nvci9vcGVuYXBpLXBhcnNlci90cmVlL21hc3Rlci9qc29uLXNjaGVtYS12YWxpZGF0b3IKOnZhbGlkYXRvci1ib206IGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYXBpLXByb2Nlc3Nvci9vcGVuYXBpLXBhcnNlci90cmVlL21hc3Rlci9qc29uLXNjaGVtYS12YWxpZGF0b3ItYm9tCjppby1pbnRlcmZhY2VzOiBodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1wcm9jZXNzb3Ivb3BlbmFwaS1wYXJzZXIvdHJlZS9tYXN0ZXIvaW8taW50ZXJmYWNlcwo6Y29udmVydGVyLWphY2tzb246IGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYXBpLXByb2Nlc3Nvci9vcGVuYXBpLXBhcnNlci90cmVlL21hc3Rlci9pby1qYWNrc29uCjpjb252ZXJ0ZXItc25ha2V5YW1sOiBodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1wcm9jZXNzb3Ivb3BlbmFwaS1wYXJzZXIvdHJlZS9tYXN0ZXIvaW8tc25ha2V5YW1sCjptZW1vcnk6IGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYXBpLXByb2Nlc3Nvci9vcGVuYXBpLXBhcnNlci90cmVlL21hc3Rlci9tZW1vcnktcHJvdG9jb2wKOmphY2tzb246IGh0dHBzOi8vZ2l0aHViLmNvbS9GYXN0ZXJYTUwvamFja3Nvbgo6c25ha2V5YW1sOiBodHRwczovL2JpdGJ1Y2tldC5vcmcvc25ha2V5YW1sL3NuYWtleWFtbC9zcmMvbWFzdGVyLwoKOmFsbC1jaTogaHR0cHM6Ly9naXRodWIuY29tL29wZW5hcGktcHJvY2Vzc29yL29wZW5hcGktcGFyc2VyL2FjdGlvbnM/cXVlcnk9d29ya2Zsb3clM0FidWlsZAo6YWxsLWNpLWJhZGdlOiBodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1wcm9jZXNzb3Ivb3BlbmFwaS1wYXJzZXIvd29ya2Zsb3dzL2J1aWxkL2JhZGdlLnN2ZwoKOmNlbnRyYWwtc2VhcmNoOiBodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9aW8ub3BlbmFwaXByb2Nlc3Nvcgo6cGFyc2VyLWNlbnRyYWwtYmFkZ2U6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2lvLm9wZW5hcGlwcm9jZXNzb3Ivb3BlbmFwaS1wYXJzZXI/bGFiZWw9TWF2ZW4lMjBDZW50cmFsCjpwYXJzZXItYm9tLWNlbnRyYWwtYmFkZ2U6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2lvLm9wZW5hcGlwcm9jZXNzb3Ivb3BlbmFwaS1wYXJzZXItYm9tP2xhYmVsPU1hdmVuJTIwQ2VudHJhbAo6dmFsaWRhdG9yLWNlbnRyYWwtYmFkZ2U6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2lvLm9wZW5hcGlwcm9jZXNzb3IvanNvbi1zY2hlbWEtdmFsaWRhdG9yP2xhYmVsPU1hdmVuJTIwQ2VudHJhbAo6dmFsaWRhdG9yLWJvbS1jZW50cmFsLWJhZGdlOiBodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9pby5vcGVuYXBpcHJvY2Vzc29yL2pzb24tc2NoZW1hLXZhbGlkYXRvci1ib20/bGFiZWw9TWF2ZW4lMjBDZW50cmFsCjppbnRlcmZhY2VzLWNlbnRyYWwtYmFkZ2U6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2lvLm9wZW5hcGlwcm9jZXNzb3IvaW8taW50ZXJmYWNlcz9sYWJlbD1NYXZlbiUyMENlbnRyYWwKOmphY2tzb24tY2VudHJhbC1iYWRnZTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3YvaW8ub3BlbmFwaXByb2Nlc3Nvci9pby1qYWNrc29uP2xhYmVsPU1hdmVuJTIwQ2VudHJhbAo6c25ha2V5YW1sLWNlbnRyYWwtYmFkZ2U6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2lvLm9wZW5hcGlwcm9jZXNzb3IvaW8tc25ha2V5YW1sP2xhYmVsPU1hdmVuJTIwQ2VudHJhbAoKSXQgaXMgdXNhYmxlLCBpdCBpcyB1c2VkIGFzIHRoZSBpbnRlcm5hbCBvcGVuYXBpIHBhcnNlci92YWxpZGF0b3IgYnkgb3BlbmFwaS1wcm9jZXNzb3IuCgo9PSBvcGVuYXBpLXBhcnNlciAmIHZhbGlkYXRvcgoKYSBKYXZhIDExIGJhc2VkIGxpbms6e29wZW5hcGl9W09wZW5BUEldIDMuMiwgMy4xLCAzLjAgcGFyc2VyIHdpdGggdmFsaWRhdGlvbiBhbmQgcGx1Z2dhYmxlIGRvY3VtZW50IHJlYWRlciAmIGpzb24veWFtbCBjb252ZXJ0ZXIuCgoqIHBhcnNlIE9wZW5BUEkgMy4yLCAzLjEgJiAzLjAKKiB2YWxpZGF0ZSBPcGVuQVBJIDMuMiwgMy4xICYgMy4wIChKU09OIHNjaGVtYSB2YWxpZGF0aW9uLCBjYW4gZm9sbG93ICRyZWYncyBpbiB0aGUgT3BlbkFQSSBkb2N1bWVudCkKKiBzZXBhcmF0ZSBhcGlzIGZvciBPcGVuQVBJIDMuMiwgMy4xICYgMy4wCiogZWFzaWx5IGdldCByZXNvbHZlZCAkcmVmIG9iamVjdAoqIGJ1bmRsZSAmIHdyaXRlIChzaW5nbGUgZmlsZSkgT3BlbkFQSSBkb2N1bWVudAoqIGFwcGx5IG92ZXJsYXkgdG8gKGJ1bmRsZWQpIE9wZW5BUEkgZG9jdW1lbnQgKGV4cGVyaW1lbnRhbCkKKiBtaW5pbWFsIGRlcGVuZGVuY2llcwoqIHBsdWdnYWJsZSBkb2N1bWVudCByZWFkZXIKKiBwbHVnZ2FibGUganNvbi95YW1sIGNvbnZlcnRlcgoqIHBsdWdnYWJsZSBgZm9ybWF0YCB2YWxpZGF0b3JzIChwbGFubmVkKQoKPT0gbW9kdWxlcwoKLy8gYmFkZ2VzCmxpbms6e2FsbC1jaX1baW1hZ2U6e2FsbC1jaS1iYWRnZX1bXV0KCgo9PT0gbGluazp7cGFyc2VyfVtvcGVuYXBpLXBhcnNlcl0KCmxpbms6e2NlbnRyYWwtc2VhcmNofVtpbWFnZTp7cGFyc2VyLWNlbnRyYWwtYmFkZ2V9W11dCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvT3BlbkFQSS0zLjIlMkNfMy4xJTJDXzMuMC0lMjMwMGMwMDA/bGFiZWw9T3BlbkFQSVtTdGF0aWMgQmFkZ2VdCgp0aGUgbGluazp7b3BlbmFwaX1bT3BlbkFQSV0gcGFyc2VyLgoKPT09IGxpbms6e3BhcnNlci1ib219W29wZW5hcGktcGFyc2VyLWJvbV0KCmxpbms6e2NlbnRyYWwtc2VhcmNofVtpbWFnZTp7cGFyc2VyLWJvbS1jZW50cmFsLWJhZGdlfVtdXQoKQSAiYmlsbCBvZiBtYXRlcmlhbHMiIFBPTSBmb3Igb3BlbmFwaS1wYXJzZXIgZGVwZW5kZW5jaWVzLgoKPT09IGxpbms6e3ZhbGlkYXRvcn1banNvbi1zY2hlbWEtdmFsaWRhdG9yXQoKbGluazp7Y2VudHJhbC1zZWFyY2h9W2ltYWdlOnt2YWxpZGF0b3ItY2VudHJhbC1iYWRnZX1bXV0KaW1hZ2U6aHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9lbmRwb2ludD91cmw9aHR0cHMlM0ElMkYlMkZib3d0aWUucmVwb3J0JTJGYmFkZ2VzJTJGamF2YS1pby5vcGVuYXBpcHJvY2Vzc29yLmpzb24tc2NoZW1hLXZhbGlkYXRvciUyRnN1cHBvcnRlZF92ZXJzaW9ucy5qc29uW1N1cHBvcnRlZCBEaWFsZWN0c10KCnRoZSBKU09OIFNjaGVtYSB2YWxpZGF0b3IgdXNlZCBieSB0aGUgYG9wZW5hcGktcGFyc2VyYC4KCj09PSBsaW5rOnt2YWxpZGF0b3ItYm9tfVtqc29uLXNjaGVtYS12YWxpZGF0b3ItYm9tXQoKbGluazp7Y2VudHJhbC1zZWFyY2h9W2ltYWdlOnt2YWxpZGF0b3ItYm9tLWNlbnRyYWwtYmFkZ2V9W11dCgpBICJiaWxsIG9mIG1hdGVyaWFscyIgUE9NIGZvciBqc29uLXNjaGVtYS12YWxpZGF0b3IgZGVwZW5kZW5jaWVzLgoKPT09IGxpbms6e2NvbnZlcnRlci1qYWNrc29ufTNbaW8tamFja3NvbjNdCgpsaW5rOntjZW50cmFsLXNlYXJjaH1baW1hZ2U6e2phY2tzb24tY2VudHJhbC1iYWRnZX1bXV0KCmpzb24veWFtbCBgUmVhZGVyYCwgYFdyaXRlcmAgJiBgQ29udmVydGVyYCBiYXNlZCBvbiBsaW5rOntqYWNrc29ufVtKYWNrc29uIDNdLgoKPT09IGxpbms6e2NvbnZlcnRlci1qYWNrc29ufVtpby1qYWNrc29uXQoKbGluazp7Y2VudHJhbC1zZWFyY2h9W2ltYWdlOntqYWNrc29uLWNlbnRyYWwtYmFkZ2V9W11dCgpqc29uL3lhbWwgYFJlYWRlcmAsIGBXcml0ZXJgICYgYENvbnZlcnRlcmAgYmFzZWQgb24gbGluazp7amFja3Nvbn1bSmFja3NvbiAyXS4KCj09PSBsaW5rOntjb252ZXJ0ZXItc25ha2V5YW1sfVtpby1zbmFrZXlhbWxdCgpsaW5rOntjZW50cmFsLXNlYXJjaH1baW1hZ2U6e3NuYWtleWFtbC1jZW50cmFsLWJhZGdlfVtdXQoKeWFtbCBgUmVhZGVyYCwgYFdyaXRlcmAgJiBgQ29udmVydGVyYCBiYXNlZCBvbiBsaW5rOntzbmFrZXlhbWx9W3NuYWtleWFtbF0uCgo9PT0gbGluazp7aW8taW50ZXJmYWNlc31baW8taW50ZXJmYWNlc10KCmxpbms6e2NlbnRyYWwtc2VhcmNofVtpbWFnZTp7aW50ZXJmYWNlcy1jZW50cmFsLWJhZGdlfVtdXQoKZGVmaW5lcyB0aGUgaW50ZXJmYWNlcyBmb3IgdGhlIHBsdWdnYWJsZSBkb2N1bWVudCByZWFkZXIgYW5kIGpzb24veWFtbCBjb252ZXJ0ZXIuIE9ubHkgbmVlZGVkIGZvciBpbXBsZW1lbnRpbmcgYWRkaXRpb25hbCByZWFkZXIgb3IgY29udmVydGVyLgo= readmeEtag: '"56deeaec3c4c2f3a75930d6191403c8d72736339"' readmeLastModified: Sun, 23 Nov 2025 23:07:41 GMT repositoryId: 418185783 description: OpenAPI 3.2, 3.1 & 3.0 Parser & JSON Schema Validator, Java created: '2021-10-17T16:14:37Z' updated: '2026-01-31T15:44:49Z' language: Java archived: false stars: 19 watchers: 2 forks: 2 owner: openapi-processor logo: https://avatars.githubusercontent.com/u/66728774?v=4 license: Apache-2.0 repoEtag: '"d4fc348d13915e674b5e3f1cb37d31e6acea3c8342b230a3946ae9b7e23d8580"' repoLastModified: Sat, 31 Jan 2026 15:44:49 GMT foundInMaster: true - source: https://openapi.tools/ name: mitmproxy2swagger category: - Auto Generators - Server Implementations language: Python link: https://github.com/alufers/mitmproxy2swagger repository: https://github.com/alufers/mitmproxy2swagger source_description: >- A tool for automatically converting mitmproxy captures to OpenAPI 3.0 specifications. Automatically reverse-engineer REST APIs by just running the apps and capturing the traffic. v3: true id: 8799f612e2b1a072f59a35691ca68ff3 repositoryMetadata: base64Readme: >- IyBtaXRtcHJveHkyc3dhZ2dlcgoKWyFbUHlQSSB2ZXJzaW9uXShodHRwczovL2JhZGdlLmZ1cnkuaW8vcHkvbWl0bXByb3h5MnN3YWdnZXIuc3ZnKV0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL3B5L21pdG1wcm94eTJzd2FnZ2VyKQpbIVtBcmNoIExpbnV4IHJlcG9zaXRvcnldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvYXJjaGxpbnV4LW1pdG1wcm94eTJzd2FnZ2VyLWJsdWUpXShodHRwczovL2FyY2hsaW51eC5vcmcvcGFja2FnZXMvZXh0cmEvYW55L21pdG1wcm94eTJzd2FnZ2VyLykKCjxodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS81NDAwOTQwLzE2ODA4NjgxOC1jNDhmNjBhYi0zZjk1LTQyZWItYjQzNS1jOGIxYTYzMjZiODEubXA0PgoKQSB0b29sIGZvciBhdXRvbWF0aWNhbGx5IGNvbnZlcnRpbmcgW21pdG1wcm94eV0oaHR0cHM6Ly9taXRtcHJveHkub3JnLykgY2FwdHVyZXMgdG8gW09wZW5BUEkgMy4wXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pIHNwZWNpZmljYXRpb25zLiBUaGlzIG1lYW5zIHRoYXQgeW91IGNhbiBhdXRvbWF0aWNhbGx5IHJldmVyc2UtZW5naW5lZXIgUkVTVCBBUElzIGJ5IGp1c3QgcnVubmluZyB0aGUgYXBwcyBhbmQgY2FwdHVyaW5nIHRoZSB0cmFmZmljLgoKLS0tCgoqKvCfhpUgTkVXISoqCgpBZGRlZCBzdXBwb3J0IGZvciBwcm9jZXNzaW5nIEhBUiBleHBvcnRlZCBmcm9tIHRoZSBicm93c2VyIERldlRvb2xzLiBTZWUgW1VzYWdlIC0gSEFSXSgjaGFyKSBmb3IgbW9yZSBkZXRhaWxzLgoKLS0tCgojIyBJbnN0YWxsYXRpb24KCkZpcnN0IHlvdSB3aWxsIG5lZWQgcHl0aG9uMyBhbmQgcGlwMy4KCmBgYGJhc2gKJCBwaXAgaW5zdGFsbCBtaXRtcHJveHkyc3dhZ2dlcgojIC4uLiBvciAuLi4KJCBwaXAzIGluc3RhbGwgbWl0bXByb3h5MnN3YWdnZXIKIyAuLi4gb3IgLi4uCiQgZ2l0IGNsb25lIGdpdEBnaXRodWIuY29tOmFsdWZlcnMvbWl0bXByb3h5MnN3YWdnZXIuZ2l0CiQgY2QgbWl0bXByb3h5MnN3YWdnZXIKJCBkb2NrZXIgYnVpbGQgLXQgbWl0bXByb3h5MnN3YWdnZXIgLgpgYGAKClRoZW4gY2xvbmUgdGhlIHJlcG8gYW5kIHJ1biBgbWl0bXByb3h5MnN3YWdnZXJgIGFzIHBlciBleGFtcGxlcyBiZWxvdy4KCiMjIFVzYWdlCgojIyMgTWl0bXByb3h5CgpUbyBjcmVhdGUgYSBzcGVjaWZpY2F0aW9uIGJ5IGluc3BlY3RpbmcgSFRUUCB0cmFmZmljIHlvdSB3aWxsIG5lZWQgdG86CgoxLiBDYXB0dXJlIHRoZSB0cmFmZmljIGJ5IHVzaW5nIHRoZSBtaXRtcHJveHkgdG9vbC4gSSBwZXJzb25hbGx5IHJlY29tbWVuZCB1c2luZyBtaXRtd2ViLCB3aGljaCBpcyBhIHdlYiBpbnRlcmZhY2UgYnVpbHQtaW4gdG8gbWl0bXByb3h5LgoKICAgYGBgYmFzaAogICAkIG1pdG13ZWIKICAgV2ViIHNlcnZlciBsaXN0ZW5pbmcgYXQgaHR0cDovLzEyNy4wLjAuMTo4MDgxLwogICBQcm94eSBzZXJ2ZXIgbGlzdGVuaW5nIGF0IGh0dHA6Ly8qOjk5OTkKICAgLi4uCiAgIGBgYAoKICAgKipJTVBPUlRBTlQqKgoKICAgVG8gY29uZmlndXJlIHlvdXIgY2xpZW50IHRvIHVzZSB0aGUgcHJveHkgZXhwb3NlZCBieSBtaXRtIHByb3h5LCBwbGVhc2UgY29uc3VsdCB0aGUgW21pdG1wcm94eSBkb2N1bWVudGF0aW9uXShodHRwczovL2RvY3MubWl0bXByb3h5Lm9yZy9zdGFibGUvKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCjIuIFNhdmUgdGhlIHRyYWZmaWMgdG8gYSBmbG93IGZpbGUuCgogICBJbiBtaXRtd2ViIHlvdSBjYW4gZG8gdGhpcyBieSB1c2luZyB0aGUgIkZpbGUiIG1lbnUgYW5kIHNlbGVjdGluZyAiU2F2ZSI6CgogICAhW0Egc2NyZWVuc2hvdCBzaG93aW5nIHRoZSBsb2NhdGlvbiBvZiB0aGUgIlNhdmUiIG9wdGlvbiBpbiB0aGUgIkZpbGUiIG1lbnVdKC4vZG9jcy9taXRtd2ViX3NhdmUucG5nKQoKMy4gUnVuIHRoZSBmaXJzdCBwYXNzIG9mIG1pdG1wcm94eTJzd2FnZ2VyOgoKICAgYGBgYmFzaAogICAkIG1pdG1wcm94eTJzd2FnZ2VyIC1pIDxwYXRoX3RvX21pdG1wdG94eV9mbG93PiAtbyA8cGF0aF90b19vdXRwdXRfc2NoZW1hPiAtcCA8YXBpX3ByZWZpeD4KICAgIyAuLi4gb3IgLi4uCiAgICQgZG9ja2VyIHJ1biAtaXQgLXYgJFBXRDovYXBwIG1pdG1wcm94eTJzd2FnZ2VyIG1pdG1wcm94eTJzd2FnZ2VyIC1pIDxwYXRoX3RvX21pdG1wdG94eV9mbG93PiAtbyA8cGF0aF90b19vdXRwdXRfc2NoZW1hPiAtcCA8YXBpX3ByZWZpeD4KICAgYGBgCgogICBQbGVhc2Ugbm90ZSB0aGF0IHlvdSBjYW4gdXNlIGFuIGV4aXN0aW5nIHNjaGVtYSwgaW4gd2hpY2ggY2FzZSB0aGUgZXhpc3Rpbmcgc2NoZW1hIHdpbGwgYmUgZXh0ZW5kZWQgd2l0aCB0aGUgbmV3IGRhdGEuIFlvdSBjYW4gYWxzbyBydW4gaXQgYSBmZXcgdGltZXMgd2l0aCBkaWZmZXJlbnQgZmxvdyBjYXB0dXJlcywgdGhlIGNhcHR1cmVkIGRhdGEgd2lsbCBiZSBzYWZlbHkgbWVyZ2VkLgoKICAgYDxhcGlfcHJlZml4PmAgaXMgdGhlIGJhc2UgdXJsIG9mIHRoZSBBUEkgeW91IHdpc2ggdG8gcmV2ZXJzZS1lbmdpbmVlci4gWW91IHdpbGwgbmVlZCB0byBvYnRhaW4gaXQgYnkgb2JzZXJ2aW5nIHRoZSByZXF1ZXN0cyBiZWluZyBtYWRlIGluIG1pdG1wcm94eS4KCiAgIEZvciBleGFtcGxlIGlmIGFuIGFwcCBoYXMgbWFkZSByZXF1ZXN0cyBsaWtlIHRoZXNlOgoKICAgYGBgaHR0cAogICBodHRwczovL2FwaS5leGFtcGxlLmNvbS92MS9sb2dpbgogICBodHRwczovL2FwaS5leGFtcGxlLmNvbS92MS91c2Vycy8yCiAgIGh0dHBzOi8vYXBpLmV4YW1wbGUuY29tL3YxL3VzZXJzLzIvcHJvZmlsZQogICBgYGAKCiAgIFRoZSBsaWtlbHkgcHJlZml4IGlzIGBodHRwczovL2FwaS5leGFtcGxlLmNvbS92MWAuCgo0LiBSdW5uaW5nIHRoZSBmaXJzdCBwYXNzIHNob3VsZCBoYXZlIGNyZWF0ZWQgYSBzZWN0aW9uIGluIHRoZSBzY2hlbWEgZmlsZSBsaWtlIHRoaXM6CgogICBgYGB5YW1sCiAgIHgtcGF0aC10ZW1wbGF0ZXM6CiAgICAgIyBSZW1vdmUgdGhlIGlnbm9yZTogcHJlZml4IHRvIGdlbmVyYXRlIGFuIGVuZHBvaW50IHdpdGggaXRzIFVSTAogICAgICMgTGluZXMgdGhhdCBhcmUgY2xvc2VyIHRvIHRoZSB0b3AgdGFrZSBwcmVjZWRlbmNlLCB0aGUgbWF0Y2hpbmcgaXMgZ3JlZWR5CiAgICAgLSBpZ25vcmU6L2FkZHJlc3NlcwogICAgIC0gaWdub3JlOi9iYXNrZXQKICAgICAtIGlnbm9yZTovYmFza2V0L2FkZAogICAgIC0gaWdub3JlOi9iYXNrZXQvY2hlY2tvdXRzCiAgICAgLSBpZ25vcmU6L2Jhc2tldC9jb3Vwb25zL2F0dGFjaC97aWR9CiAgICAgLSBpZ25vcmU6L2Jhc2tldC9jb3Vwb25zL2F0dGFjaC8xMDQ3NTQKICAgYGBgCgogICBZb3Ugc2hvdWxkIGVkaXQgdGhlIHNjaGVtYSBmaWxlIHdpdGggYSB0ZXh0IGVkaXRvciBhbmQgcmVtb3ZlIHRoZSBgaWdub3JlOmAgcHJlZml4IGZyb20gdGhlIHBhdGhzIHlvdSB3aXNoIHRvIGJlIGdlbmVyYXRlZC4gWW91IGNhbiBhbHNvIGFkanVzdCB0aGUgcGFyYW1ldGVycyBhcHBlYXJpbmcgaW4gdGhlIHBhdGhzLgoKNS4gUnVuIHRoZSBzZWNvbmQgcGFzcyBvZiBtaXRtcHJveHkyc3dhZ2dlcjoKCiAgIGBgYGJhc2gKICAgJCBtaXRtcHJveHkyc3dhZ2dlciAtaSA8cGF0aF90b19taXRtcHRveHlfZmxvdz4gLW8gPHBhdGhfdG9fb3V0cHV0X3NjaGVtYT4gLXAgPGFwaV9wcmVmaXg+IFstLWV4YW1wbGVzXQogICAjIC4uLiBvciAuLi4KICAgJCBkb2NrZXIgcnVuIC1pdCAtdiAkUFdEOi9hcHAgbWl0bXByb3h5MnN3YWdnZXIgbWl0bXByb3h5MnN3YWdnZXIgLWkgPHBhdGhfdG9fbWl0bXB0b3h5X2Zsb3c+IC1vIDxwYXRoX3RvX291dHB1dF9zY2hlbWE+IC1wIDxhcGlfcHJlZml4PiBbLS1leGFtcGxlc10KICAgYGBgCgogICBSdW4gdGhlIGNvbW1hbmQgYSBzZWNvbmQgdGltZSAod2l0aCB0aGUgc2FtZSBzY2hlbWEgZmlsZSkuIEl0IHdpbGwgcGljayB1cCB0aGUgZWRpdGVkIGxpbmVzIGFuZCBnZW5lcmF0ZSBlbmRwb2ludCBkZXNjcmlwdGlvbnMuCgogICBQbGVhc2Ugbm90ZSB0aGF0IG1pdG1wcm94eTJzd2FnZ2VyIHdpbGwgbm90IG92ZXJ3cml0ZSBleGlzdGluZyBlbmRwb2ludCBkZXNjcmlwdGlvbnMsIGlmIHlvdSB3YW50IHRvIG92ZXJ3cml0ZSB0aGVtLCB5b3UgY2FuIGRlbGV0ZSB0aGVtIGJlZm9yZSBydW5uaW5nIHRoZSBzZWNvbmQgcGFzcy4KCiAgIFBhc3NpbmcgYC0tZXhhbXBsZXNgIHdpbGwgYWRkIGV4YW1wbGUgZGF0YSB0byByZXF1ZXN0cyBhbmQgcmVzcG9uc2VzLiBUYWtlIGNhdXRpb24gd2hlbiB1c2luZyB0aGlzIG9wdGlvbiwgYXMgaXQgbWF5IGFkZCBzZW5zaXRpdmUgZGF0YSAodG9rZW5zLCBwYXNzd29yZHMsIHBlcnNvbmFsIGluZm9ybWF0aW9uIGV0Yy4pIHRvIHRoZSBzY2hlbWEuCiAgIFBhc3NpbmcgYC0taGVhZGVyc2Agd2lsbCBhZGQgaGVhZGVycyBkYXRhIHRvIHJlcXVlc3RzIGFuZCByZXNwb25zZXMuIFRha2UgY2F1dGlvbiB3aGVuIHVzaW5nIHRoaXMgb3B0aW9uLCBhcyBpdCBtYXkgYWRkIHNlbnNpdGl2ZSBkYXRhICh0b2tlbnMsIHBhc3N3b3JkcywgcGVyc29uYWwgaW5mb3JtYXRpb24gZXRjLikgdG8gdGhlIHNjaGVtYS4KCiMjIyBIQVIKCjEuIENhcHR1cmUgYW5kIGV4cG9ydCB0aGUgdHJhZmZpYyBmcm9tIHRoZSBicm93c2VyIERldlRvb2xzLgoKICAgSW4gdGhlIGJyb3dzZXIgRGV2VG9vbHMsIGdvIHRvIHRoZSBOZXR3b3JrIHRhYiBhbmQgY2xpY2sgdGhlICJFeHBvcnQgSEFSIiBidXR0b24uCgogICAhW0Egc2NyZWVuc2hvdCBzaG93aW5nIHdoZXJlIHRoZSBleHBvcnQgaGFyIGJ1dHRvbiBpcyBsb2NhdGVkXSguL2RvY3MvZXhwb3J0X2hhcl9idXR0b24ucG5nKQoKMi4gQ29udGludWUgdGhlIHNhbWUgd2F5IHlvdSB3b3VsZCBkbyB3aXRoIHRoZSBtaXRtcHJveHkgZHVtcC4gYG1pdG1wcm94eTJzd2FnZ2VyYCB3aWxsIGF1dG9tYXRpY2FsbHkgZGV0ZWN0IHRoZSBIQVIgZmlsZSBhbmQgcHJvY2VzcyBpdC4KCiMjIEV4YW1wbGUgb3V0cHV0CgpTZWUgdGhlIFtleGFtcGxlc10oLi9leGFtcGxlX291dHB1dHMvKS4gWW91IHdpbGwgZmluZCBhIGdlbmVyYXRlZCBzY2hlbWEgdGhlcmUgYW5kIGFuIGh0bWwgZmlsZSB3aXRoIHRoZSBnZW5lcmF0ZWQgZG9jdW1lbnRhdGlvbiAodmlhIFtyZWRvYy1jbGldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3JlZG9jLWNsaSkpLgoKU2VlIHRoZSBnZW5lcmF0ZWQgaHRtbCBmaWxlIFtoZXJlXShodHRwczovL3Jhdy5naXRoYWNrLmNvbS9hbHVmZXJzL21pdG1wcm94eTJzd2FnZ2VyL21hc3Rlci9leGFtcGxlX291dHB1dHMvbGlzZWstc3RhdGljLmh0bWwpLgoKIyMgRGV2ZWxvcG1lbnQgYW5kIGNvbnRyaWJ1dGluZwoKVGhpcyBwcm9qZWN0IHVzZXM6CgotIFtwb2V0cnldKGh0dHBzOi8vcHl0aG9uLXBvZXRyeS5vcmcvKSBmb3IgZGVwZW5kZW5jeSBtYW5hZ2VtZW50Ci0gW3ByZS1jb21taXRdKGh0dHBzOi8vcHJlLWNvbW1pdC5jb20vKSBmb3IgY29kZSBmb3JtYXR0aW5nIGFuZCBsaW50aW5nCi0gW3B5dGVzdF0oaHR0cHM6Ly9kb2NzLnB5dGVzdC5vcmcvZW4vc3RhYmxlLykgZm9yIHVuaXQgdGVzdGluZwoKVG8gaW5zdGFsbCB0aGUgZGVwZW5kZW5jaWVzOgoKYGBgYmFzaApwb2V0cnkgaW5zdGFsbApgYGAKClJ1biBsaW50ZXJzOgoKYGBgYmFzaApwcmUtY29tbWl0IHJ1biAtLWFsbC1maWxlcwpgYGAKCkluc3RhbGwgcHJlLWNvbW1pdCBob29rczoKCmBgYGJhc2gKcHJlLWNvbW1pdCBpbnN0YWxsCmBgYAoKUnVuIHRlc3RzOgoKYGBgYmFzaApwb2V0cnkgcnVuIHB5dGVzdApgYGAKClJ1biB0ZXN0cyB3aXRoIGNvdmVyYWdlOgoKYGBgYmFzaApwb2V0cnkgcnVuIHB5dGVzdCAtLWNvdj1taXRtcHJveHkyc3dhZ2dlcgpgYGAKCiMjIExpY2Vuc2UKCk1JVAo= readmeEtag: '"7bc61550d619b1125bbf77e397d6ca00184686ea"' readmeLastModified: Wed, 03 Apr 2024 08:07:41 GMT repositoryId: 491304964 description: Automagically reverse-engineer REST APIs via capturing traffic created: '2022-05-11T23:48:49Z' updated: '2026-02-05T20:53:58Z' language: HTML archived: false stars: 9214 watchers: 27 forks: 344 owner: alufers logo: https://avatars.githubusercontent.com/u/5400940?v=4 repoEtag: '"1c7b9a11bacdf852f9fccc4269d8f350ad79ee66b07f211f81c75cf041eb3e99"' repoLastModified: Thu, 05 Feb 2026 20:53:58 GMT foundInMaster: true - source: https://openapi.tools/ name: har2openapi category: - Auto Generators - Parsers language: TypeScript link: https://github.com/dcarr178/har2openapi repository: https://github.com/dcarr178/har2openapi source_description: >- Automatically generate OpenAPI 3.0 Spec by using network requests captured in one or more HAR files v2: false v3: true repositoryMetadata: base64Readme: >- # har2openapi

This program automatically creates API documentation via a OpenApi Spec (OAS) file by using network requests captured in 
one or more HAR files.

## Step 1 - Create HAR file(s)
There are lots of ways to create HAR files to use as input for this program. One way is to manually click your way
through a web application and capture network requests in Chrome as illustrated in this [handy guide](https://help.o2.verizonmedia.com/hc/en-us/articles/360000074523-How-to-Save-a-HAR-File-Log-in-Google-Chrome).

Another way to generate HAR files is to install [Charles Proxy](https://www.charlesproxy.com/) and then run all your
end-to-end browser tests. Charles Proxy saves its raw captures in a .chls file but you can export that to a har file as Charles Proxy 
outputs har files in a convenient json format. Generating HAR files using your end-to-end tests can save you a lot of
manual clicking!

There are many other ways to generate HAR files so go crazy. Google is your friend.

## Step 2 - Generate request/response examples
Start by copying `config.json.template` to `config.json`. Once you have a valid config.json file, run
``` bash
node index.js generate outputFilename.json inputHarFile1.json inputHarFile2.json inputHarFile3.json...
```

This command consumes one or more HAR files and outputs an OAS file. A few additional files are created for your convenience:
* `output/methodList.txt` - this file contains one line for each method/operation created
* `output/pathList.txt` - this file contains one line for each path created
* `output/outputFilename.json.yaml` - this is a YAML representation of the same OAS object output to outputFilename.json

### Iteratively edit config.json and generate OAS again

You'll probably notice that the OAS file starts out pretty noisy which is why you'll want to edit `config.json` and then rinse 
and repeat generating the OAS file again.

### Replace strings in path
In the config file, you can set your API base path and any number of search/replace commands. For each parameter in
```javascript
...
  "pathReplace": {
    "key": "value",
    "key": "value"
  }
...
```
The program executes as
```
path = path.replace(/key/g, value)
```
Why would we want to do this? I'm glad you asked. There are various answers:

* Remove query string parameters from the path. Query string parameters are detected automatically and moved to path variables in the OAS file.
* Search and replace IDs in the path. A noisy OAS file will contain one endpoint definition for `/account/1` and another endpoint definition
for `/account/2`. By adding replace strings to config.json you can collapse duplicate paths into one endpoint definition
and automatically move the path IDs into path parameters in the OAS file.

### Replace strings anywhere
In addition to path parameters, there are instances when we want to remove certain strings from anywhere in the endpoint defintion
including request examples, response examples, etc. Specifically, you may want to remove secrets like passwords, sessionIds, and other things 
you probably don't want in your OAS file. To do this, add search/replace items here:
```javascript
...
  "replace": {
    "secret": ""
  }
...
```
### Tag endpoints
Each endpoint definition (aka operation) in the OAS file can have a tag which assists renders in grouping related endpoints
together. Config.json allows you to search for text in the path and assign matches to a specific Tag.

```javascript
...
  "tags": [
    ["slides"],
    ["progress", "Task Progress"]
  ]
...
```

Using this example, if `path.includes("slides") then tag = "Slides"`. if `path.includes("progress") then tag = "Task Progress"`  

## Step 3 - Manually edit request and response examples
At this point in the process it's best to manually review the request and response examples in the generated OAS file and shorten them where necessary. For
example, if an API response returns an array of 2,000 elements, perhaps your api documentation could restrict itself to a handful of
elements instead of displaying all 2,000. These descisions need to be made on a case-by-case basis and you can 
edit the generated OAS file directly.

## Step 4 - Generate json schemas and x-code-samples
After you have manually edited your OAS file, you can generate json schemas for all request and response examples and generate 
x-code-samples by running
```bash
node index.js samples output/generated.json output/withSchema.json
```
This command consumes the (presumably edited) generated.json OAS file and outputs a new OAS file containing json schemas and x-code-samples. 
It also creates an additional file for your convenience:
* `output/withSchema.json.yaml` - this is a YAML representation of the same OAS object output to withSchema.json

## Step 5 - Merge generated paths/methods/operations into master OAS file
If this is the first time you are creating your OAS file then you will want to start editing withSchema.json and adding your own
api servers, info object, security, etc. Your edited file should be considered your master OAS file. However, in the future
you may want to re-run har2openapi against new HAR files to identify differences or detect new api endpoints that
weren't picked up previously.

When this happens, you'll want to generate a new OAS file and then carefully merge it into your master OAS file without
damaging the edits you have carefully made. To do this, run

```bash
node index.js merge input/master.json output/withSchema.json output/openapi-spec.json
```
This command consumes the generated withSchema.json OAS file with your master.json file and outputs a new OAS file with them merged. New methods/operations found
in withSchema.json will be added to master.json but nothing in master.json will be replaced. 

This program also creates an additional file for your convenience:
* `output/open-spec.json.yaml` - this is a YAML representation of the same OAS object output to open-spec.json

## History

In April, 2020 I needed to document a python REST api with 150 paths and 250 endpoints (aka methods, operations). My 
preferred approach was to create an [OpenAPI Spec 3.0.3](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md) 
file (OAS) so it could be consumed and rendered using any number of [different tools](https://openapi.tools/).

To begin creating my OAS file, I fired up [Stoplight Studio](https://stoplight.io/studio/) and began cranking
away at building endpoint definitions. After creating 5 or 10 endpoints by hand, I was already burned out. I'm just
not cut out for mind-numbing, repetitive work. Creating every request and response example by hand was massively laborious
not to mention creating json schemas for each one. I'm also very lazy. Creating endpoint documentation by hand could take years!

Besides, I had a bigger problem. The API I was documenting was poorly documented (hence my involvement) and I couldn't really rely on
it to even tell me what all the paths and supported (http) methods were. 

As I laid awake at night wondering how to escape my predicament, I thought why not run all of our [cypress](https://cypress.io) end-to-end tests, capture all the network requests with something like 
[Charles Proxy](https://www.charlesproxy.com/), and figure out a way to use the proxy output to automagically create
my OAS file for me and thus escape months of life-draining monotony?

That is how this project began.
 readmeEtag: '"2197cf34c6e33a3a32f907d556e6cecd3257128f"' readmeLastModified: Tue, 24 Aug 2021 17:42:44 GMT repositoryId: 262170349 description: Generate openapi spec api documentation from captured har files created: '2020-05-07T22:19:19Z' updated: '2026-01-31T12:13:36Z' language: JavaScript archived: false stars: 161 watchers: 6 forks: 25 owner: dcarr178 logo: https://avatars.githubusercontent.com/u/2379611?v=4 repoEtag: '"f0f92d761ab675cafc3dfd456ed9558edd550f08712d82f90228d2036543fe5c"' repoLastModified: Sat, 31 Jan 2026 12:13:36 GMT foundInMaster: true id: 4df801fc431a2975d8981b7c4879ff7a - source: https://openapi.tools/ name: har-to-openapi category: - Auto Generators - Parsers language: TypeScript link: https://github.com/jonluca/har-to-openapi repository: https://github.com/jonluca/har-to-openapi source_description: >- Automatically generate OpenAPI 3.0 Spec by using network requests captured in one or more HAR files v3: true id: 98cb389f9d59b10647b4a8c30ed2c230 repositoryMetadata: base64Readme: >- IyBIQVIgdG8gT3BlbkFQSQoKWyFbbnBtIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3YvaGFyLXRvLW9wZW5hcGkuc3ZnKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvaGFyLXRvLW9wZW5hcGkpIFshW0xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2wvaGFyLXRvLW9wZW5hcGkuc3ZnKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvaGFyLXRvLW9wZW5hcGkpCgpDb252ZXJ0IGEgSEFSIGZpbGUgdG8gYW4gT3BlbkFQSSBzcGVjCgojIEludHJvZHVjdGlvbgoKX1RoaXMgbGlicmFyeSBpcyBsb29zZWx5IGJhc2VkIG9uIFtoYXIyb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL2RjYXJyMTc4L2hhcjJvcGVuYXBpKSwgYnV0IGNsZWFuZWQgdXAgYW5kIGNoYW5nZWQgZm9yIHVzYWdlIGluIGEgbW9yZSBwcm9ncmFtbWF0aWMgZmFzaGlvbl8KCiMgR2V0dGluZyBTdGFydGVkCgpgYGAKeWFybiBhZGQgaGFyLXRvLW9wZW5hcGkKYGBgCgpvcgoKYGBgCm5wbSBpIC0tc2F2ZSBoYXItdG8tb3BlbmFwaQpgYGAKCiMgVXNhZ2UKCmBgYHR5cGVzY3JpcHQKaW1wb3J0IHsgZ2VuZXJhdGVTcGVjIH0gZnJvbSAiaGFyLXRvLW9wZW5hcGkiOwoKLy8gcmVhZCBhIGhhciBmaWxlIGZyb20gd2hlcmV2ZXIgeW91IHdhbnQgLSBpbiB0aGlzIGV4YW1wbGUgaXRzIGp1c3QgYSByb290IGpzb24gb2JqZWN0Ci8vIGNvbnN0IGhhciA9IGF3YWl0IGZzLnJlYWRGaWxlKCJteS5oYXIiKTsKCmNvbnN0IGhhciA9IHsKICBsb2c6IHsKICAgIGVudHJpZXM6IFsKICAgICAgewogICAgICAgIGluZGV4OiAwLAogICAgICAgIHJlcXVlc3Q6IHsKICAgICAgICAgIG1ldGhvZDogIkNVU1RPTSIsCiAgICAgICAgICB1cmw6ICJodHRwOi8vdGVzdC5sb2FkaW1wYWN0LmNvbS9sb2dpbiIsCiAgICAgICAgICBoZWFkZXJzOiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICBuYW1lOiAiQ29udGVudC1UeXBlIiwKICAgICAgICAgICAgICB2YWx1ZTogImFwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZCIsCiAgICAgICAgICAgIH0sCiAgICAgICAgICBdLAogICAgICAgICAgcG9zdERhdGE6IHsKICAgICAgICAgICAgbWltZVR5cGU6ICJhcHBsaWNhdGlvbi94LXd3dy1mb3JtLXVybGVuY29kZWQiLAogICAgICAgICAgICB0ZXh0OiAiZm9vMD1iYXIwJmZvbzE9YmFyMSIsCiAgICAgICAgICAgIHBhcmFtczogWwogICAgICAgICAgICAgIHsKICAgICAgICAgICAgICAgIG5hbWU6ICJmb28wIiwKICAgICAgICAgICAgICAgIHZhbHVlOiAiYmFyMCIsCiAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgXSwKICAgICAgICAgIH0sCiAgICAgICAgfSwKICAgICAgfSwKICAgIF0sCiAgfSwKfTsKCmNvbnN0IG9wZW5hcGkgPSBhd2FpdCBnZW5lcmF0ZVNwZWMoaGFyLCB7IHJlbGF4ZWRNZXRob2RzOiB0cnVlIH0pOwpjb25zdCB7IHNwZWMsIHlhbWxTcGVjIH0gPSBvcGVuYXBpOwovLyBzcGVjID0geyAuLi4gfSBvcGVuYXBpIHNwZWMgc2NoZW1hIGRvY3VtZW50Ci8vIHlhbWxTcGVjID0gc3RyaW5nLCAiaW5mbzogLi4uIgpgYGAKCiMjIE9wdGlvbnMKCmBgYHR5cGVzY3JpcHQKZXhwb3J0IGludGVyZmFjZSBDb25maWcgewogIC8vIGlmIHRydWUsIHdlJ2xsIHRyZWF0IGV2ZXJ5IHVybCBhcyBoYXZpbmcgdGhlIHNhbWUgZG9tYWluLCByZWdhcmRsZXNzIG9mIHdoYXQgaXRzIGFjdHVhbCBkb21haW4gaXMKICAvLyB0aGUgZmlyc3QgZG9tYWluIHdlIHNlZSBpcyB0aGUgZG9tYWluIHdlJ2xsIHVzZQogIGZvcmNlQWxsUmVxdWVzdHNJblNhbWVTcGVjPzogYm9vbGVhbjsKICAvLyBpZiB0cnVlLCBldmVyeSBwYXRoIG9iamVjdCB3aWxsIGhhdmUgaXRzIG93biBzZXJ2ZXJzIGVudHJ5LCBkZWZpbmluZyBpdHMgYmFzZSBwYXRoLiBUaGlzIGlzIHVzZWZ1bCB3aGVuCiAgLy8gZm9yY2VBbGxSZXF1ZXN0c0luU2FtZVNwZWMgaXMgc2V0CiAgYWRkU2VydmVyc1RvUGF0aHM/OiBib29sZWFuOwogIC8vIHRyeSBhbmQgZ3Vlc3MgY29tbW9uIGF1dGggaGVhZGVycwogIGd1ZXNzQXV0aGVudGljYXRpb25IZWFkZXJzPzogYm9vbGVhbjsKICAvLyBpZiB0aGUgcmVzcG9uc2UgaGFzIHRoaXMgc3RhdHVzIGNvZGUsIGlnbm9yZSB0aGUgYm9keQogIGlnbm9yZUJvZGllc0ZvclN0YXR1c0NvZGVzPzogbnVtYmVyW107CiAgLy8gd2hldGhlciBub24gc3RhbmRhcmQgbWV0aG9kcyBzaG91bGQgYmUgYWxsb3dlZCAobGlrZSBIVFRQIE1ZX0NVU1RPTV9NRVRIT0QpCiAgcmVsYXhlZE1ldGhvZHM/OiBib29sZWFuOwogIC8vIHdoZXRoZXIgd2Ugc2hvdWxkIHRyeSBhbmQgcGFyc2Ugbm9uIGFwcGxpY2F0aW9uL2pzb24gcmVzcG9uc2VzIGFzIGpzb24gLSBkZWZhdWx0cyB0byB0cnVlCiAgcmVsYXhlZENvbnRlbnRUeXBlSnNvblBhcnNlPzogYm9vbGVhbjsKICAvLyBhIGxpc3Qgb2YgdGFncyB0aGF0IG1hdGNoIHBhc3NlZCBvbiB0aGUgcGF0aCwgZWl0aGVyIFttYXRjaF9hbmRfdGFnXSBvciBbbWF0Y2gsIHRhZ10KICB0YWdzPzogKFtzdHJpbmddIHwgW3N0cmluZywgc3RyaW5nXSB8IHN0cmluZylbXSB8ICgodXJsOiBzdHJpbmcpID0+IHN0cmluZyB8IHN0cmluZ1tdIHwgdm9pZCk7CiAgLy8gcmVzcG9uc2UgbWltZSB0eXBlcyB0byBmaWx0ZXIgZm9yCiAgbWltZVR5cGVzPzogc3RyaW5nW107CiAgLy8ga25vd24gc2VjdXJpdHkgaGVhZGVycyBmb3IgdGhpcyBoYXIsIHRvIGFkZCB0byBzZWN1cml0eSBmaWVsZCBpbiBvcGVuYXBpIChlLmcuICJYLUF1dGgtVG9rZW4iKQogIHNlY3VyaXR5SGVhZGVycz86IHN0cmluZ1tdOwogIC8vIFdoZXRoZXIgdG8gZmlsdGVyIG91dCBhbGwgc3RhbmRhcmQgaGVhZGVycyBmcm9tIHRoZSBwYXJhbWV0ZXIgbGlzdCBpbiBvcGVuYXBpCiAgZmlsdGVyU3RhbmRhcmRIZWFkZXJzPzogYm9vbGVhbjsKICAvLyBXaGV0aGVyIHRvIGxvZyBlcnJvcnMgdG8gY29uc29sZQogIGxvZ0Vycm9ycz86IGJvb2xlYW47CiAgLy8gYSBzdHJpbmcsIHJlZ2V4LCBvciBjYWxsYmFjayB0byBmaWx0ZXIgdXJscyBmb3IgaW5jbHVzaW9uCiAgdXJsRmlsdGVyPzogc3RyaW5nIHwgUmVnRXhwIHwgKCh1cmw6IHN0cmluZykgPT4gYm9vbGVhbiB8IFByb21pc2U8Ym9vbGVhbj4pOwogIC8vIHdoZW4gd2UgZW5jb3VudGVyIGEgVVJMLCB0cnkgYW5kIHBhcmFtZXRlcml6ZSBpdCwgc3VjaCB0aGF0IHNvbWV0aGluZyBsaWtlCiAgLy8gR0VUIC91dWlkcy8xMjNlNDU2Ny1lODliLTEyZDMtYTQ1Ni00MjY2NTU0NDAwMDAgYmVjb21lcyBHRVQgL3V1aWRzL3t1dWlkfQogIGF0dGVtcHRUb1BhcmFtZXRlcml6ZVVybD86IGJvb2xlYW47CiAgLy8gd2hlbiB3ZSBlbmNvdW50ZXIgYSBwYXRoIHdpdGhvdXQgYSByZXNwb25zZSBvciB3aXRoIGEgcmVzcG9uc2UgdGhhdCBkb2VzIG5vdCBoYXZlIDJ4eCwgZG9udCBpbmNsdWRlIGl0CiAgZHJvcFBhdGhzV2l0aG91dFN1Y2Nlc3NmdWxSZXNwb25zZT86IGJvb2xlYW47Cn0KYGBgCg== readmeEtag: '"b1c6645596b604d10ba4cf9034427636a80c3dbf"' readmeLastModified: Fri, 11 Aug 2023 17:49:57 GMT repositoryId: 519317344 description: HAR to OpenAPI spec generator created: '2022-07-29T18:43:20Z' updated: '2026-01-24T04:24:53Z' language: TypeScript archived: false stars: 124 watchers: 2 forks: 14 owner: jonluca logo: https://avatars.githubusercontent.com/u/13029040?v=4 license: MIT repoEtag: '"837109d53fc1e63e930738eb7b3d56f203c28bdcf6910d80da250dc84b147c26"' repoLastModified: Sat, 24 Jan 2026 04:24:53 GMT foundInMaster: true - source: https://openapi.tools/ name: GranthAi category: - Auto Generators - Documentation language: Javascript link: https://granthai.com repository: https://www.npmjs.com/package/granthai source_description: >- NodeJs OpenAPI 3 based documentation generator which sits between server APIs and anyone calling the APIs v3: true id: 29fe99d026b8f20572226028e338a960 foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/adawg4/openapi-autospec v3: true id: 87df782257819ac984fca13d011b0283 repositoryMetadata: base64Readme: >- <a name="readme-top"></a>

<div align="center">
<img height="120" src="logo.png">

# OpenAPI AutoSpec

Need last-minute docs? Quickly capture API behavior with a server proxy that automatically generates OpenAPI specifications in real-time from any locally running website or service.

<img src="https://img.shields.io/badge/license-MIT-blue" alt="license"/> [![Join Discord](https://img.shields.io/badge/Discord-Join-grey?style=flat&logo=Discord)](https://discord.gg/CRnxg7uduH)
<a target="_blank" href="https://twitter.com/intent/tweet?text=OpenAPI AutoSpec is a proxy server that generates API specs for any app or website on localhost! Try it out: https://github.com/adawg4/openapi-autospec" class="item"><img src="https://img.shields.io/twitter/url?label=Tweet&amp;style=social&amp;url=https://github.com/adawg4/openapi-autospec"></a>
</div>

## About
<p align="center" width="100%">
    <img width="80%" src="autogif.gif">
</p>
OpenAPI AutoSpec is a local server proxy that generates new OpenAPI specifications from network requests. When running, it will connect your local HTTP traffic to the proxy. Once that happens, it will automatically convert network traffic into the specification.

*Features*:
- Generate OpenAPI 3.0 specifications automatically for any local website or application
- Capture and document new requests & responses, including headers, bodies, and query parameters
- Review generated specifications in real-time on your terminal and download them with ease
- Export your OpenAPI file for sharing
- Ignores static file URLs: .js, .css, .svg, .png, .jpeg, .ico.
</br>

## Installation

This is a [Node.js](https://nodejs.org/en/) module available through the
[npm registry](https://www.npmjs.com/).

Before installing, [download and install Node.js](https://nodejs.org/en/download/).
Node.js 0.10 or higher is required.

Installation steps:
[`npm install` command](https://docs.npmjs.com/getting-started/installing-npm-packages-locally):

```console
$ npm install openapi-autospec
```
</br>

## Usage
Start the server, it generates a link for you to use, and then use the new link from the proxy server to catch traffic on a target server. To generate the document you want: visit pages, fill out forms/all the fields or data you wish to track, and perform the actions you want to document from your APIs. For more information on getting documentation from a full-stack server versus both frontend and backend servers - <a href="#readme-misc">use this reference</a>.

To start the server:
```console
$ npx autospec --portTo PORT --portFrom PORT --filePath openapi.json
```

- --portTo choose the port your localhost webserver is running from.
- --portFrom (optional) is for if another port is in use.
- --filePath is where you would like to export the spec.

</br>
Once the server is running, it will automatically begin documenting the API behavior of your local web applications. The documentation process is based on the network requests observed, so you don't need to do any crazy setup within your applications beyond the below.


The server provides real-time printouts of the generated OpenAPI specifications. From here:
- Fill out all fields you wish to be documented for forms
- Export the OpenAPI specification for external use or sharing (exports where you are in the terminal if you don't specify the --filePath flag)
- You should find and replace all instances of ‘localhost’ or ‘127.0.0.1:8000’ in the exported doc with your actual website
- Filter hosts and parameterize paths to fine-tune the documentation properly for server stubs
- Restart the documentation process at any time to refresh the generated specifications
</br>

## Roadmap

Want an easy way to automatically build and manage your SDKs, Zapier/Make integrations, Docs, Webhooks, RPA plugins, and custom plugins to marketplaces like Salesforce, Zoom, GSuite all in one spot while getting your app AI-agent ready? [Contact us](https://calendly.com/aidan_/demo). We can custom tailor this and more with our paid service.

As for the repo:
- [Path parameterization](https://www.abstractapi.com/api-glossary/path-parameters) tools
- HTTPS and OpenAPI 3.1 Specification support
- Run on GCP, AWS, Azure, Docker, and Kubernetes
</br>

## What is OpenAPI?
OpenAPI specifications provide a standardized description of an API's expected requests and responses, making APIs easier to understand and integrate with. Governed by the OpenAPI Initiative and the Linux Foundation, they are the modern standard for documenting RESTful APIs.
</br>
</br>

## Community

[Join our Discord!](https://discord.gg/CRnxg7uduH) We are here to answer questions and help you get the most out of our OpenAPI tool.
</br>
</br>

## Contributing

We welcome community contributions. For guidelines, refer to our [CONTRIBUTING.md](/CONTRIBUTING.md).



Shoutouts to [Awesome API DevTools](https://github.com/yosriady/awesome-api-devtools) and [OpenAPI Devtools](https://chromewebstore.google.com/detail/openapi-devtools/jelghndoknklgabjgaeppjhommkkmdii).
</br>


<a name="readme-misc"></a>
## Running proxies for multiple servers

### Full-Stack Web Frameworks (e.g., Django)

1. Run your website and note down the port and localhost information. Ex. 127.0.0.1:8000
2. Run the script and specify the localhost information where traffic should be routed. Ex. npx autospec --portTo 8000. 
```
$ npx autospec --portTo PORT --portFrom PORT--filePath openapi.json
```
- The --portTo flag should be followed by the localhost information where you want the traffic to be routed. Ex. "127.0.0.1:8000"
- The optional --portFrom flag can be used to specify a different port to run the proxy, in case the default port is already in use. The value following this would be the port you want to use for the proxy. Ex. "3000" 
3. Copy the output URL (Ex. localhost:7928) and use your website or application as described above.
4. Access the newly created file - named with the current timestamp.

### Frontend and Backend Separately (e.g., a Node backend and React frontend)

For applications with a separate frontend and backend, run proxies for both.

1. Run your frontend and note down the host and port information. E.g., 127.0.0.1:3000

2. Operate your backend on a different port and note the host and port information. E.g., 127.0.0.1:5000 (For detailed guidelines on how to change port info, visit <a href="#readme-bottom">here</a>).

3. Run the proxy for the frontend, specifying the frontend's host and port information where traffic should be routed. If necessary, you can specify a different port for the proxy to operate from.

```console
$ npx autospec --portTo FRONTEND_PORT --filePath frontend.json
```

4. Similarly, run the proxy for the backend, specifying the backend's host and port where the frontend normally sends information.

```console
$ npx autospec --portTo NEW_BACKEND_PORT --portFrom NORMAL_BACKEND_PORT --filePath backend.json
```

This will initiate the server that listens to network requests from your locally running websites, automatically documenting their API interactions.

## Changing backend ports

<a name="readme-bottom"></a>
```markdown
### 1. Express.js (Node.js)
To change the port for an Express.js application, you can set the port in your application code before starting the server. For example:

```javascript
const express = require('express');
const app = express();
const port = 3001; // Change to your desired port

app.listen(port, () => {
  console.log(`Example app listening at http://localhost:${port}`);
});
```

### 2. Django (Python)
For Django, you can specify the port when you run the runserver command from the command line:

```shell
python manage.py runserver 8080
```
This command runs the server on port 8080.

### 3. Flask (Python)
In Flask, you can specify the port when calling the app.run() method:

```python
from flask import Flask
app = Flask(__name__)

if __name__ == '__main__':
    app.run(port=5001) // Change to your desired port
```

### 4. Rails (Ruby)
For a Rails application, you can specify the port with the -p option when starting the server:

```shell
rails server -p 4000
```
This command runs the server on port 4000.

### 5. Spring Boot (Java)
In Spring Boot, you can specify the server port in the application.properties or application.yml file located in the src/main/resources directory:

```
server.port=8081
```

### 6. ASP.NET Core (C#)
For an ASP.NET Core application, you can specify the port in the launchSettings.json file found in the Properties folder of your project. Alternatively, you can use the --urls command-line argument:

```shell
dotnet run --urls="http://localhost:5002"
```
This command runs the server on port 5002.

### 7. Laravel (PHP)
In Laravel, you can specify the port with the --port option when serving the application:

```shell
php artisan serve --port=8001
```
This command runs the server on port 8001.

### 8. Vue.js (Node.js)
For a Vue.js project created with vue-cli, you can specify the port in the vue.config.js file:

```javascript
module.exports = {
  devServer: {
    port: 8081
  }
}
```

### 9. React (Node.js)
For a React application created with create-react-app, you can set the port by modifying the start script in your package.json file, or by setting the PORT environment variable before running the npm start command:

```shell
PORT=3001 npm start
```

### 10. Angular (Node.js)
For an Angular application, you can specify the port with the --port option when serving the application:

```shell
ng serve --port 4201
```

These are references - Always refer to the official documentation for the most accurate and up-to-date information.

 readmeEtag: '"c753c9a0e8becf3d65fcff649d634259360122c0"' readmeLastModified: Sat, 04 May 2024 00:18:47 GMT repositoryId: 760891853 description: Proxy server that generates API specs for any app or website on localhost. created: '2024-02-20T21:16:04Z' updated: '2026-02-01T12:21:14Z' language: JavaScript archived: false stars: 663 watchers: 2 forks: 16 owner: Adawg4 logo: https://avatars.githubusercontent.com/u/27599708?v=4 repoEtag: '"d1c2cd26d986480e1aab489fec5863903c41e7965180cdea3c6acb3c8128e525"' repoLastModified: Sun, 01 Feb 2026 12:21:14 GMT category: - Auto Generators - Documentation - Server Implementations foundInMaster: true name: OpenAPI AutoSpec language: JavaScript link: https://www.portway.ai/ source_description: >- Proxy server that generates API descriptions for any app or website on localhost. v2: false v3_1: false - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/andrewwalsh/openapi-devtools v3: true id: 34445bbc3e815731b195a393a2a9d3f4 repositoryMetadata: base64Readme: >- PGEgbmFtZT0icmVhZG1lLXRvcCI+PC9hPgoKWyFbTUlUIExpY2Vuc2VdW2xpY2Vuc2Utc2hpZWxkXV1bbGljZW5zZS11cmxdClshW0Rvd25sb2FkIGluIHRoZSBDaHJvbWUgV2ViIFN0b3JlXVtjaHJvbWUtc2hpZWxkXV1bY2hyb21lLXVybF0KWyFbRG93bmxvYWQgaW4gdGhlIEZpcmVmb3ggQWRkLW9uIFN0b3JlXVtmaXJlZm94LXNoaWVsZF1dW2ZpcmVmb3gtdXJsXQoKCj4gWyFJTVBPUlRBTlRdICAKPiBBIG5ldyBhbmQgZmluYWwgdmVyc2lvbiBvZiB0aGlzIHByb2plY3QgaXMgYXZhaWxhYmxlIGJlbG93LiBJdCBoYXMgYmVlbiByZXdyaXR0ZW4gZnJvbSBzY3JhdGNoIGFuZCBzaWduaWZpY2FudGx5IGV4cGFuZHMgdGhlIGZlYXR1cmUgc2V0IG9mIE9wZW5BUEkgRGV2VG9vbHMuCj4gCj4gTmV3IGZlYXR1cmVzOgo+ICAgLSBJZGVudGlmaWVzIHBhdGggcGFyYW1ldGVycyBhdXRvbWF0aWNhbGx5Cj4gICAtIElzIGF2YWlsYWJsZSBhcyBhIGRlc2t0b3AgYXBwIHRoYXQgdXNlcyBhIHByb3h5IChlLmcuIHNlZSBtaXRtMnN3YWdnZXIpCj4gICAtIEhhcyBhIHdlYnNpdGUgYW5kIGNvbW1hbmQgbGluZSB0b29sIHRoYXQgY2FuIGdlbmVyYXRlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgZnJvbSBIQVIgZmlsZXMgKGUuZy4gaGFyLXRvLW9wZW5hcGkpCj4gICAtIFVzZXMgYSBwdWJsaXNoZWQgbGlicmFyeSB0aGF0IGNhbiBnZW5lcmF0ZSBBUEkgc3BlY2lmaWNhdGlvbnMgZm9yIGFueSBBUEkgaW4gbWlkZGxld2FyZSwgRGV2T3BzIGludGVncmF0aW9ucywgb3IgYW55IGVudmlyb25tZW50Cj4gICAtIENhbiBnZW5lcmF0ZSBjdXJsIGNvbW1hbmRzLCBjbGllbnQgY29kZSwgYW5kIHNlbmQgQVBJIHJlcXVlc3RzIHZpYSBhbiBBUEkgY2xpZW50IHRoYXQgcG9wdWxhdGVzIGF1dG9tYXRpY2FsbHkKPiAgIC0gSXMgYWdub3N0aWMgdG8gQVBJIHNlbWFudGljcyBhbmQgbWF5IHRyYW5zbGF0ZSBpbnRvIGRpZmZlcmVudCB2ZXJzaW9ucyBvZiBPcGVuQVBJLCBvciByZXZlcnNlIGVuZ2luZWVyIG90aGVyIEhUVFAtYmFzZWQgc3RhbmRhcmRzIHN1Y2ggYXMgR3JhcGhRTAo+ICAgLSBJbmNsdWRlcyBhbGwgb3JpZ2luYWwgZmVhdHVyZXMsIGluY2x1ZGluZyB0aGUgYWJpbGl0eSB0byBpbXBvcnQgYW5kIGV4cG9ydAoKLS0tPiBbQ2hlY2sgb3V0IHRoZSBuZXcgdmVyc2lvbiBvZiBPcGVuQVBJIERldlRvb2xzIGhlcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9BbmRyZXdXYWxzaC9kZW15c3RpZnkpCgo8IS0tIFBST0pFQ1QgTE9HTyAtLT4KPGJyIC8+CjxkaXYgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL0FuZHJld1dhbHNoL29wZW5hcGktZGV2dG9vbHMiPgogICAgPGltZyBzcmM9InJlc291cmNlcy9sb2dvLnN2ZyIgYWx0PSJPcGVuIEFQSSBkZXYgdG9vbHMiIHdpZHRoPSIzMDAiIGhlaWdodD0iMjUwIj4KICA8L2E+CgoKICA8cCBhbGlnbj0iY2VudGVyIiBzdHlsZT0ibWF4LXdpZHRoOiA2MDBweDsiPgogICAgRWZmb3J0bGVzc2x5IGRpc2NvdmVyIEFQSSBiZWhhdmlvdXIgd2l0aCBhIGJyb3dzZXIgZXh0ZW5zaW9uIHRoYXQgYXV0b21hdGljYWxseSBnZW5lcmF0ZXMgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBpbiByZWFsIHRpbWUgZm9yIGFueSBhcHAgb3Igd2Vic2l0ZS4KICAgIDxiciAvPgogICAgPGJyIC8+CiAgICA8YSBocmVmPSJodHRwczovL2F3YWxzaC5pby9wb3N0cy9vcGVuYXBpLWRldnRvb2xzLyI+UmVhZCBNb3JlPC9hPgogICAgwrcKICAgIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9BbmRyZXdXYWxzaC9vcGVuYXBpLWRldnRvb2xzL2lzc3VlcyI+UmVwb3J0IEJ1ZzwvYT4KICAgIMK3CiAgICA8YSBocmVmPSJodHRwczovL2Zvcm1zLmdsZS9mQTJFdHZhbVVyc1J4WEtZQSI+R2l2ZSBGZWVkYmFjazwvYT4KICA8L3A+CiAgPC9wPgo8L2Rpdj4KCiMjIEFib3V0IFRoZSBQcm9qZWN0Cgo8cCBhbGlnbj0iY2VudGVyIiB3aWR0aD0iMTAwJSI+CiAgICA8aW1nIHdpZHRoPSI4MCUiIHNyYz0icmVzb3VyY2VzL2RlbW8uZ2lmIj4KPC9wPgoKT3BlbkFQSSBEZXZUb29scyBpcyBhIGJyb3dzZXIgZXh0ZW5zaW9uIHRoYXQgZ2VuZXJhdGVzIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgaW4gcmVhbCB0aW1lIGZyb20gbmV0d29yayByZXF1ZXN0cy4gT25jZSBpbnN0YWxsZWQgaXQgYWRkcyBhIG5ldyB0YWIgdG8gRGV2VG9vbHMgY2FsbGVkIGBPcGVuQVBJYC4gV2hpbGUgdGhlIHRvb2wgaXMgb3BlbiBpdCBhdXRvbWF0aWNhbGx5IGNvbnZlcnRzIG5ldHdvcmsgcmVxdWVzdHMgaW50byBhIHNwZWNpZmljYXRpb24uCgoqRmVhdHVyZXMqOgotIEluc3RhbnRseSBnZW5lcmF0ZSBhbiBPcGVuQVBJIDMuMSBzcGVjaWZpY2F0aW9uIGZvciBhbnkgd2Vic2l0ZSBvciBhcHBsaWNhdGlvbiBqdXN0IGJ5IHVzaW5nIGl0Ci0gQXV0b21hdGljYWxseSBtZXJnZXMgbmV3IHJlcXVlc3QgJiByZXNwb25zZSBoZWFkZXJzLCBib2RpZXMsIGFuZCBxdWVyeSBwYXJhbWV0ZXJzIHBlciBlbmRwb2ludAotIENsaWNrIG9uIGEgW3BhdGggcGFyYW1ldGVyXShodHRwczovL3d3dy5hYnN0cmFjdGFwaS5jb20vYXBpLWdsb3NzYXJ5L3BhdGgtcGFyYW1ldGVycykgYW5kIHRoZSBhcHAgd2lsbCBhdXRvbWF0aWNhbGx5IG1lcmdlIGV4aXN0aW5nIGFuZCBmdXR1cmUgbWF0Y2hpbmcgcmVxdWVzdHMKLSBWaWV3IHRoZSBzcGVjaWZpY2F0aW9uIGluc2lkZSB0aGUgdG9vbCB1c2luZyBbUmVkb2NdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3JlZG9jKSBhbmQgZG93bmxvYWQgd2l0aCBhIGNsaWNrCi0gRXhwb3J0IGFuZCBzYXZlIGEgc2Vzc2lvbiBhdCBhbnkgdGltZSwgb3Igc2hhcmUgaXQgd2l0aCBvdGhlcnMKCjxwIGFsaWduPSJyaWdodCI+KDxhIGhyZWY9IiNyZWFkbWUtdG9wIj5iYWNrIHRvIHRvcDwvYT4pPC9wPgoKIyMgSW5zdGFsbGF0aW9uCgo8cCBhbGlnbj0iY2VudGVyIiB3aWR0aD0iMTAwJSI+CiAgICA8aW1nIHdpZHRoPSI4MCUiIHNyYz0icmVzb3VyY2VzL2RlbW8taW1nLnBuZyI+CjwvcD4KCltEb3dubG9hZCB0aGUgZXh0ZW5zaW9uIGluIHRoZSBDaHJvbWUgV2ViIFN0b3JlXVtjaHJvbWUtdXJsXS4KCltEb3dubG9hZCB0aGUgZXh0ZW5zaW9uIGluIHRoZSBGaXJlZm94IEFkZC1vbiBTdG9yZV1bZmlyZWZveC11cmxdLgoKT3RoZXJ3aXNlLCB0byBpbnN0YWxsIG1hbnVhbGx5OgogIC0gW0Rvd25sb2FkIGFuZCBleHRyYWN0IHRoZSBkaXN0LnppcCBmaWxlIGluIHRoZSBsYXRlc3QgcmVsZWFzZV0oaHR0cHM6Ly9naXRodWIuY29tL0FuZHJld1dhbHNoL29wZW5hcGktZGV2dG9vbHMvcmVsZWFzZXMvbGF0ZXN0L2Rvd25sb2FkL2Rpc3QuemlwKQogIC0gSW4gQ2hyb21lLCBuYXZpZ2F0ZSB0byBgY2hyb21lOi8vZXh0ZW5zaW9uc2AKICAtIEluIHRoZSB0b3AgcmlnaHQgZW5hYmxlIHRoZSBgRGV2ZWxvcGVyIG1vZGVgIHRvZ2dsZQogIC0gSW4gdGhlIHRvcCBsZWZ0IGNsaWNrIGBMb2FkIHVucGFja2VkYCBhbmQgc2VsZWN0IHRoZSBleHRyYWN0ZWQgYGRpc3RgIGRpcmVjdG9yeQogIC0gT3BlbiBhIG5ldyB0YWIgYW5kIHRoZW4gc2VsZWN0IGBPcGVuQVBJYCBpbiB0aGUgZGV2ZWxvcGVyIHRvb2xzIChvcGVuIHdpdGggYGNtZCtpYCBvciBgY3RybCtpYCkKICAtIEZpcmVmb3ggaXMgbW9yZSBjaGFsbGVuZ2luZy4gUGxlYXNlIHVzZSB0aGUgYWRkLW9uIHN0b3JlLgoKPHAgYWxpZ249InJpZ2h0Ij4oPGEgaHJlZj0iI3JlYWRtZS10b3AiPmJhY2sgdG8gdG9wPC9hPik8L3A+CgojIyBVc2FnZQoKVGhlIHNwZWNpZmljYXRpb24gd2lsbCBhdXRvbWF0aWNhbGx5IHBvcHVsYXRlIGJhc2VkIG9uIEpTT04gcmVxdWVzdHMgdGhhdCBmaXJlIGFzIHlvdSBicm93c2UgdGhlIHdlYi4gSW4gdGhlIHNldHRpbmdzIG1lbnUgeW91IGNhbiBmaWx0ZXIgaG9zdHMgYW5kIHBhcmFtZXRlcmlzZSBwYXRocyBpbiBVUkxzLiBPbmNlIHlvdSBkbyBzbyBhbGwgbWF0Y2hpbmcgZXhpc3RpbmcgYW5kIGZ1dHVyZSByZXF1ZXN0cyB0byB0aGF0IGVuZHBvaW50IHdpbGwgYmUgbWVyZ2VkLiBUaGlzIHByb2Nlc3MgaXMgaXJyZXZlcnNpYmxlLCBidXQgeW91IGNhbiBjbGVhciB0aGUgc3BlY2lmaWNhdGlvbiBhbmQgcmVzdGFydCBhdCBhbnkgdGltZS4KCldoZW4gdGhlIHNhbWUgZW5kcG9pbnQgcmVzcG9uZHMgd2l0aCBkaWZmZXJlbnQgZGF0YSwgc3VjaCBhcyBhIHZhbHVlIHRoYXQgaXMgc29tZXRpbWVzIGEgc3RyaW5nIGFuZCBzb21ldGltZXMgbnVsbCwgdGhlIHNwZWNpZmljYXRpb24gZm9yIHRoYXQgdmFsdWUgd2lsbCBiZSAqZWl0aGVyKiBzdHJpbmcgb3IgbnVsbC4gQWxsIGluZm9ybWF0aW9uIGlzIGFjY291bnRlZCBmb3IgaW4gdGhlIGZpbmFsIHNwZWNpZmljYXRpb24uIElmIHlvdSBzZWUgc29tZXRoaW5nIG1pc3NpbmcgZnJvbSBhIHJlcXVlc3QsIHRyaWdnZXIgYSByZXF1ZXN0IHRoYXQgY29udGFpbnMgdGhlIG1pc3NpbmcgaW5mb3JtYXRpb24uCgpUaGUgc2V0dGluZ3MgbWVudSBjb250YWlucyBzZXZlcmFsIG9wdGlvbnMuIEhlcmUgeW91IGNhbiBlbmFibGUgcmVhbCBleGFtcGxlcyBpbiB0aGUgc3BlY2lmaWNhdGlvbi4gWW91IGNhbiBhbHNvIGV4cG9ydCB0aGUgY3VycmVudCBzdGF0ZSBvZiB0aGUgYXBwIGFzIGEgc3RyaW5nLCBzaGFyZSBvciBzdG9yZSBpdCwgYW5kIGltcG9ydCBpdCBsYXRlci4KCjxwIGFsaWduPSJyaWdodCI+KDxhIGhyZWY9IiNyZWFkbWUtdG9wIj5iYWNrIHRvIHRvcDwvYT4pPC9wPgoKIyMgV2hhdCBpcyBPcGVuQVBJPwoKQW4gW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIHNwZWNpZmljYXRpb24gaXMgYSBkZXNjcmlwdGlvbiBvZiB3aGF0IGFuIEFQSSBleHBlY3RzIHRvIHJlY2VpdmUgYW5kIHdoYXQgaXQgd2lsbCByZXNwb25kIHdpdGguIEl0IGlzIGdvdmVybmVkIGJ5IHRoZSBPcGVuQVBJIEluaXRpYXRpdmUgYW5kIHRoZSBMaW51eCBGb3VuZGF0aW9uLiBPcGVuQVBJIHNwZWNpZmljYXRpb25zIGFyZSB0aGUgbW9kZXJuIHN0YW5kYXJkIGZvciBSRVNUZnVsIEFQSXMsIGFuZCBzeXN0ZW1zIHRoYXQgaGF2ZSB0aGVtIGFyZSBmYXIgZWFzaWVyIHRvIHdvcmsgd2l0aC4KCjxwIGFsaWduPSJyaWdodCI+KDxhIGhyZWY9IiNyZWFkbWUtdG9wIj5iYWNrIHRvIHRvcDwvYT4pPC9wPgoKIyMgQ29udHJpYnV0aW5nCgpUbyBkZXZlbG9wIHRoZSBwcm9qZWN0OgotIGBucG0gaW5zdGFsbGAKLSBgbnBtIHJ1biBkZXZgCgo8cCBhbGlnbj0icmlnaHQiPig8YSBocmVmPSIjcmVhZG1lLXRvcCI+YmFjayB0byB0b3A8L2E+KTwvcD4KCjwhLS0gTUFSS0RPV04gTElOS1MgJiBJTUFHRVMgLS0+CjwhLS0gaHR0cHM6Ly93d3cubWFya2Rvd25ndWlkZS5vcmcvYmFzaWMtc3ludGF4LyNyZWZlcmVuY2Utc3R5bGUtbGlua3MgLS0+CltsaWNlbnNlLXVybF06IGh0dHBzOi8vZ2l0aHViLmNvbS9BbmRyZXdXYWxzaC9vcGVuYXBpLWRldnRvb2xzL2Jsb2IvbWFpbi9MSUNFTlNFLnR4dApbbGljZW5zZS1zaGllbGRdOiBodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL0FuZHJld1dhbHNoL29wZW5hcGktZGV2dG9vbHMuc3ZnP3N0eWxlPWZvci10aGUtYmFkZ2UKW2Nocm9tZS11cmxdOiBodHRwczovL2Nocm9tZS5nb29nbGUuY29tL3dlYnN0b3JlL2RldGFpbC9vcGVuYXBpLWRldnRvb2xzL2plbGdobmRva25rbGdhYmpnYWVwcGpob21ta2ttZGlpCltjaHJvbWUtc2hpZWxkXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9Hb29nbGUlMjBDaHJvbWUtNDI4NUY0P3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1Hb29nbGVDaHJvbWUmbG9nb0NvbG9yPXdoaXRlCltmaXJlZm94LXVybF06IGh0dHBzOi8vYWRkb25zLm1vemlsbGEub3JnL2VuLVVTL2ZpcmVmb3gvYWRkb24vb3BlbmFwaS1kZXZ0b29scy8KW2ZpcmVmb3gtc2hpZWxkXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9GaXJlZm94LUZGNzEzOT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289RmlyZWZveC1Ccm93c2VyJmxvZ29Db2xvcj13aGl0ZQ== readmeEtag: '"cb00079efb9342440dc77e645e239cd2706e3deb"' readmeLastModified: Wed, 19 Mar 2025 04:42:20 GMT repositoryId: 709571808 description: Browser extension that generates API specs for any app or website created: '2023-10-25T00:36:06Z' updated: '2026-02-03T07:44:21Z' language: TypeScript archived: false stars: 4276 watchers: 22 forks: 89 owner: AndrewWalsh logo: https://avatars.githubusercontent.com/u/15863952?v=4 license: MIT repoEtag: '"c09b44c9d2577a6d3cb3693a0357b34210fa3b5652fa15bdb230832dcaf407fd"' repoLastModified: Tue, 03 Feb 2026 07:44:21 GMT category: - Auto Generators - Documentation categoryByRequestIndicator: true foundInMaster: true v3_1: true name: OpenAPI DevTools language: TypeScript link: >- https://chrome.google.com/webstore/detail/openapi-devtools/jelghndoknklgabjgaeppjhommkkmdii source_description: Browser extension that generates API specs for any app or website v2: false - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/microsoft/typespec v3: true id: 0a49bbbd9d784c681b8eb737004a2eab repositoryMetadata: base64Readme: >- # TypeSpec

[Official Docs](https://typespec.io/) | [Try TypeSpec Online](https://aka.ms/trytypespec) | [Getting Started](https://typespec.io/docs) | [Language Overview](https://typespec.io/docs/language-basics/overview)

TypeSpec is a language for defining cloud service APIs and shapes. TypeSpec is a highly extensible language with primitives that can describe API shapes common among REST, OpenAPI, gRPC, and other protocols.

TypeSpec is excellent for generating many different API description formats, client and service code, documentation, and other assets while keeping your TypeSpec definition as a single source of truth.

Using TypeSpec, you can create reusable patterns for all aspects of an API and package those reusable patterns into libraries. These patterns establish "guardrails" for API designers and make it easier to follow best practices than to deviate from them. TypeSpec also has a rich linter framework with the ability to flag anti-patterns as well as an emitter framework that lets you control the output to ensure it follows the patterns you want.

TypeSpec is a Microsoft-built, community-supported project. Your ideas, feedbacks, and code make all the difference and we deeply appreciate the support from the community.

## [Installation](https://typespec.io/docs)

```
npm install -g @typespec/compiler
```

#### Tools

The [TypeSpec VS Code extension](https://marketplace.visualstudio.com/items?itemName=typespec.typespec-vscode) can be installed from the VS Code [marketplace](https://marketplace.visualstudio.com/items?itemName=typespec.typespec-vscode) or directly on the command line:

```
tsp code install
```

The [TypeSpec VS Extension](https://marketplace.visualstudio.com/items?itemName=typespec.typespecvs) can be installed from the [VS Marketplace](https://marketplace.visualstudio.com/items?itemName=typespec.typespecvs) or directly on the command line:

```
tsp vs install
```

## [Usage](https://typespec.io/docs#create-first-typespec-project)

### TypeSpec to OpenAPI 3.0 Example

This example uses the `@typespec/http`, `@typespec/rest`, and `@typespec/openapi3` libraries to define a basic REST service and generate an OpenAPI 3.0 document from it.

Run the following command and select "Generic REST API":

```
tsp init
```

Hit enter a few times to confirm the defaults.

Copy the contents below into your **main.tsp**:

```typespec
import "@typespec/http";
import "@typespec/rest";
import "@typespec/openapi3";

using Http;
using Rest;

/** This is a pet store service. */
@service(#{ title: "Pet Store Service" })
@server("https://example.com", "The service endpoint")
namespace PetStore;

@route("/pets")
interface Pets {
  list(): Pet[];
}

model Pet {
  @minLength(100)
  name: string;

  @minValue(0)
  @maxValue(100)
  age: int32;

  kind: "dog" | "cat" | "fish";
}
```

Install the dependencies of main.tsp:

```
tsp install
```

Compile it to OpenAPI 3.0:

```
tsp compile main.tsp --emit @typespec/openapi3
```

You can find the emitted OpenAPI output in `./tsp-output/openapi.json`.

## Advanced Scenarios

### Installing nightly version

On every commit to the main branch, packages with changes are automatically published to npm with the `@next` tag.
The [packages](#packages) section shows which version corresponds to the `next` tag for each package.

To use a `nightly` version of the packages, go over each one of the packages in the `package.json` file and update it to either the latest published `@next` version or `@latest`, whichever is the newest. You can also use the tag `latest` or `next` instead of an explicit version.

After updating the package.json file you can run `npm update --force`. Force is required as there might be some incompatible version requirement.

Example

```json5
// Stable setup
"dependencies": {
  "@typespec/compiler": "~0.30.0",
  "@typespec/http": "~0.14.0",
  "@typespec/rest": "~0.14.0",
  "@typespec/openapi": "~0.9.0",
}

// Consume next version
// In this example: compiler and openapi have changes but rest library has none
"dependencies": {
  "@typespec/compiler": "~0.31.0-dev.5",
  "@typespec/http": "~0.14.0",
  "@typespec/rest": "~0.14.0", // No changes to @typespec/rest library so need to stay the latest.
  "@typespec/openapi": "~0.10.0-dev.2",
}
```

## Packages

| Name                                               | Changelog                        | Latest                                                                                                                                   | Next                                                                      |
| -------------------------------------------------- | -------------------------------- | ---------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------- |
| Core functionality                                 |                                  |                                                                                                                                          |                                                                           |
| [@typespec/compiler][compiler_src]                 | [Changelog][compiler_chg]        | [![](https://img.shields.io/npm/v/@typespec/compiler)](https://www.npmjs.com/package/@typespec/compiler)                                 | ![](https://img.shields.io/npm/v/@typespec/compiler/next)                 |
| TypeSpec Libraries                                 |                                  |                                                                                                                                          |                                                                           |
| [@typespec/http][http_src]                         | [Changelog][http_chg]            | [![](https://img.shields.io/npm/v/@typespec/http)](https://www.npmjs.com/package/@typespec/http)                                         | ![](https://img.shields.io/npm/v/@typespec/http/next)                     |
| [@typespec/rest][rest_src]                         | [Changelog][rest_chg]            | [![](https://img.shields.io/npm/v/@typespec/rest)](https://www.npmjs.com/package/@typespec/rest)                                         | ![](https://img.shields.io/npm/v/@typespec/rest/next)                     |
| [@typespec/openapi][openapi_src]                   | [Changelog][openapi_chg]         | [![](https://img.shields.io/npm/v/@typespec/openapi)](https://www.npmjs.com/package/@typespec/openapi)                                   | ![](https://img.shields.io/npm/v/@typespec/openapi/next)                  |
| [@typespec/openapi3][openapi3_src]                 | [Changelog][openapi3_chg]        | [![](https://img.shields.io/npm/v/@typespec/openapi3)](https://www.npmjs.com/package/@typespec/openapi3)                                 | ![](https://img.shields.io/npm/v/@typespec/openapi3/next)                 |
| [@typespec/versioning][versioning_src]             | [Changelog][versioning_chg]      | [![](https://img.shields.io/npm/v/@typespec/versioning)](https://www.npmjs.com/package/@typespec/versioning)                             | ![](https://img.shields.io/npm/v/@typespec/versioning/next)               |
| TypeSpec Tools                                     |                                  |                                                                                                                                          |                                                                           |
| [@typespec/prettier-plugin-typespec][prettier_src] | [Changelog][prettier_chg]        | [![](https://img.shields.io/npm/v/@typespec/prettier-plugin-typespec)](https://www.npmjs.com/package/@typespec/prettier-plugin-typespec) | ![](https://img.shields.io/npm/v/@typespec/prettier-plugin-typespec/next) |
| [typespec-vs][typespec-vs_src]                     | [Changelog][typespec-vs_chg]     | [![](https://img.shields.io/npm/v/typespec-vs)](https://www.npmjs.com/package/typespec-vs)                                               | ![](https://img.shields.io/npm/v/typespec-vs/next)                        |
| [typespec-vscode][typespec-vscode_src]             | [Changelog][typespec-vscode_chg] | [![](https://img.shields.io/npm/v/typespec-vscode)](https://www.npmjs.com/package/typespec-vscode)                                       | ![](https://img.shields.io/npm/v/typespec-vscode/next)                    |
| [tmlanguage-generator][tmlanguage_src]             | [Changelog][tmlanguage_chg]      | [![](https://img.shields.io/npm/v/tmlanguage-generator)](https://www.npmjs.com/package/tmlanguage-generator)                             | ![](https://img.shields.io/npm/v/tmlanguage-generator/next)               |

[compiler_src]: packages/compiler
[compiler_chg]: packages/compiler/CHANGELOG.md
[http_src]: packages/http
[http_chg]: packages/http/CHANGELOG.md
[rest_src]: packages/rest
[rest_chg]: packages/rest/CHANGELOG.md
[openapi_src]: packages/openapi
[openapi_chg]: packages/openapi/CHANGELOG.md
[openapi3_src]: packages/openapi3
[openapi3_chg]: packages/openapi3/CHANGELOG.md
[versioning_src]: packages/versioning
[versioning_chg]: packages/versioning/CHANGELOG.md
[prettier_src]: packages/prettier-plugin-typespec
[prettier_chg]: packages/prettier-plugin-typespec/CHANGELOG.md
[typespec-vs_src]: packages/typespec-vs
[typespec-vs_chg]: packages/typespec-vs/CHANGELOG.md
[typespec-vscode_src]: packages/typespec-vscode
[typespec-vscode_chg]: packages/typespec-vscode/CHANGELOG.md
[tmlanguage_src]: packages/tmlanguage-generator
[tmlanguage_chg]: packages/tmlanguage-generator/CHANGELOG.md

`@next` version of the package are the latest versions available on the `main` branch.
 readmeEtag: '"23127df411e8f922adee3f7ea9e15a2710aa145b"' readmeLastModified: Thu, 08 May 2025 00:05:14 GMT repositoryId: 381857226 description: null created: '2021-06-30T23:29:49Z' updated: '2026-02-06T00:30:11Z' language: Java archived: false stars: 5571 watchers: 47 forks: 336 owner: microsoft logo: https://avatars.githubusercontent.com/u/6154722?v=4 license: MIT repoEtag: '"374c05ff1296ca0f057f7b6e4c3e265cee343485863d6183160ab684509ccc45"' repoLastModified: Fri, 06 Feb 2026 00:30:11 GMT category: - Auto Generators - Server Implementations foundInMaster: true name: TypeSpec language: TypeScript link: https://typespec.io/ source_description: >- Emit OpenAPI specifications from API descriptions defined in the generic, interoperable, and extensible TypeSpec language. - source: - https://openapi.tools/ - openapi3 tags name: oa-client category: - SDK - Parsers language: TypeScript link: https://github.com/ninofiliu/oa-client repository: https://github.com/ninofiliu/oa-client source_description: >- Flexible client helper for making and validating calls to OpenAPI backends. For Node and the browser. Runtime lib - no need for code generation! v2: false v3: true repositoryMetadata: base64Readme: >- IyBvYS1jbGllbnQKCkhhcm5lc3MgYWxsIHRoZSBwb3dlciBvZiB5b3VyIGJhY2tlbmQncyBPcGVuQVBJIHYzIHNwZWMgZmlsZXMgYnkgZ2VuZXJhdGluZyBhIGNsaWVudCBvYmplY3QgaW4gYSBmZXcgbGluZXMKCiMjIEZlYXR1cmVzCgrwn5qAIENyZWF0ZXMgYXQgcnVudGltZSBhIGNsaWVudCBvYmplY3QgaW4gYSBmZXcgbGluZXMgKHJlYWQgbW9yZSBpbiBbR2V0dGluZyBTdGFydGVkXSgjZ2V0dGluZy1zdGFydGVkKSkKCmBgYGpzCi8vIENyZWF0aW9uCmltcG9ydCB7IGNyZWF0ZUNsaWVudCB9IGZyb20gJ29hLWNsaWVudCc7CmNvbnN0IGNsaWVudCA9IGNyZWF0ZUNsaWVudChzcGVjcywgY2FsbGVycywgewogIG9yaWdpbjogJ2h0dHBzOi8vbXkuYXBpLmNvbScsCiAgdmFsaWRhdGlvbkxldmVsOiAnZXJyb3InLAp9KTsKLy8gVXNhZ2UKY2xpZW50Wy8qIHBhdGggKi9dWy8qIG1ldGhvZCAqL10oLyogb3B0aW9uYWwgcGFyYW1zICovKS50aGVuKGFwaVJlc3BvbnNlID0+IHsgLyogLi4uICovIH0pCmBgYAoK8J+agCBPcHRpb25hbGx5IHRocm93cyBmb3IgaW52YWxpZCBwYXRoLCBxdWVyeSwgb3IgYm9keQoKYGBgeWFtbAojIE9wZW5BUEkgc3BlY3MKcGF0aHM6CiAgL3VzZXJzL3t1c2VySWR9OgogICAgZ2V0OgogICAgICBwYXJhbWV0ZXJzOgogICAgICAgIC0gaW46IHBhdGgKICAgICAgICAgIG5hbWU6IHVzZXJJZAogICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICB0eXBlOiBpbnRlZ2VyCmBgYAoKYGBganMKY2xpZW50WyIvdXNlcnMve3VzZXJJZH0iXS5nZXQoeyBwYXRoUGFyYW1zOiB7IHVzZXJJZDogImpvaG4iIH0gfSk7Ci8vIHRocm93cyBbb2EtY2xpZW50OjEwM10gRGF0YSBkb2VzIG5vdCBwYXNzIHZhbGlkYXRpb246IGRhdGEudXNlcklkIHNob3VsZCBiZSBhbiBpbnRlZ2VyCmBgYAoK8J+agCBDb21waWxlcyB0aGUgcGF0aCBhbmQgcXVlcnkgcGFyYW1zCgpgYGBqcwpjbGllbnRbIi9uZXctdXNlci97am9ifSJdLnBvc3QoewogIHBhdGhQYXJhbXM6IHsgam9iOiAiZGlyZWN0b3IiIH0sCiAgcXVlcnlQYXJhbXM6IHsgbmFtZTogIkdhc3BhciBOb8OpIiB9LAp9KTsKLy8gY2FsbHMgL25ldy11c2VyL2RpcmVjdG9yP25hbWU9R2FzcGFyK05vJUMzJUE5CmBgYAoKIyMgR2V0dGluZyBzdGFydGVkCgojIyMgMS4gSW5zdGFsbCB0aGUgcGFja2FnZQoKYGBgc2gKbnBtIGluc3RhbGwgLS1zYXZlIG9hLWNsaWVudApgYGAKCiMjIyAyLiBJbXBvcnQgdGhlIHBhY2thZ2UKClRoaXMgcGFja2FnZSBpcyBpc29tb3JwaGljOiBpdCBjYW4gYmUgdXNlZCBib3RoIGFzIGFuIEVTTSBvciBhIENvbW1vbkpTCgpgYGBqcwovLyBvawppbXBvcnQgeyBjcmVhdGVDbGllbnQgfSBmcm9tICJvYS1jbGllbnQiOwovLyBhbHNvIG9rCmNvbnN0IHsgY3JlYXRlQ2xpZW50IH0gPSByZXF1aXJlKCJvYS1jbGllbnQiKTsKYGBgCgojIyMgMy4gSGF2ZSBzb21ld2hlcmUgeW91ciBPcGVuQVBJIHNwZWNzIGFzIGEgSlMgb2JqZWN0CgpZb3UgZG9uJ3QgbmVlZCB0byBhZGQgYW55dGhpbmcgY29tcGFyZWQgdG8gbm9ybWFsIHNwZWNzLCBleGNlcHQgZm9yIGFuIG9wdGlvbmFsIGAucGF0aHNbcGF0aF1bbWV0aG9kXVsneC10eXBlJ11gLCB0aGF0IGRlZmluZXMgdGhlIF9jYWxsZXJfLCBtb3JlIG9uIHRoZW0gYmVsb3cuIElmIHRoaXMga2V5IGlzIG9taXR0ZWQsIGl0cyB2YWx1ZSBkZWZhdWx0cyB0byB0aGUgcmVxdWVzdCB0eXBlIChlLmcuIGAiZ2V0ImAgb3IgYCJwb3N0ImApLgoKTm90ZSB0aGF0IGBvYS1jbGllbnRgIGRvZXMgbm90IHJlc29sdmUgc3BlY3MgZm9yIHlvdS4gSWYgeW91IGhhdmUgYCRyZWZzYCwgeW91IHNob3VsZCB1c2UgYSBwYWNrYWdlIGxpa2UgW2pzb24tc2NoZW1hLXJlZi1wYXJzZXJdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL0BhcGlkZXZ0b29scy9qc29uLXNjaGVtYS1yZWYtcGFyc2VyKSB0byByZXNvbHZlIHRoZW0uCgpgYGBqcwpjb25zdCBzcGVjcyA9IHsKICBvcGVuYXBpOiAiMy4wLjAiLAogIGluZm86IHsKICAgIC8qIC4uLiAqLwogIH0sCiAgcGF0aHM6IHsKICAgICIvdXNlcnMve3VzZXJJZH0iOiB7CiAgICAgIGdldDogewogICAgICAgICJ4LXR5cGUiOiAiYXV0aG9yaXplZEdldCIsIC8vIHdpbGwgdXNlIHRoZSAiYXV0aG9yaXplZEdldCIgY2FsbGVyCiAgICAgICAgcGFyYW1ldGVyczogWwogICAgICAgICAgewogICAgICAgICAgICBpbjogInBhdGgiLAogICAgICAgICAgICBuYW1lOiAidXNlcklkIiwKICAgICAgICAgICAgcmVxdWlyZWQ6IHRydWUsCiAgICAgICAgICAgIHNjaGVtYTogewogICAgICAgICAgICAgIHR5cGU6ICJpbnRlZ2VyIiwKICAgICAgICAgICAgfSwKICAgICAgICAgIH0sCiAgICAgICAgXSwKICAgICAgICByZXNwb25zZXM6IHsKICAgICAgICAgIC8qIC4uLiAqLwogICAgICAgIH0sCiAgICAgIH0sCiAgICB9LAogICAgIi9zdGF0dXMiOiB7CiAgICAgIGdldDogewogICAgICAgIC8vIG5vIHgtdHlwZSAtPiB3aWxsIHVzZSB0aGUgImdldCIgY2FsbGVyCiAgICAgICAgcmVzcG9uc2VzOiB7CiAgICAgICAgICAvKiAuLi4gKi8KICAgICAgICB9LAogICAgICB9LAogICAgfSwKICB9LAp9OwpgYGAKCiMjIyA0LiBXcml0ZSB5b3VyIF9jYWxsZXJzXwoKVGhlc2UgYXJlIGdlbmVyaWMgZnVuY3Rpb25zIHRoYXQgaGFuZGxlIHJlcXVlc3RzIGF0IHRoZSBIVFRQIGxldmVsLgoKVGhleSBhcmUgbm90IGhhbmRsZWQgYnkgdGhpcyBwYWNrYWdlLCBiZWNhdXNlIHRoZXkgY2FuIGJlIHZlcnkgZGlmZmVyZW50IGZyb20gb25lIGNvZGViYXNlIHRvIGFub3RoZXI7IGJ1dCB1c3VhbGx5IHlvdSBkb24ndCBoYXZlIHRvIHdyaXRlIGEgbG90IG9mIHRoZW0uCgpgdXJsYCBpcyBhbiBbVVJMXShodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9BUEkvVVJMKSwgYGJvZHlgIGlzIGEgcGxhaW4gSlMgb2JqZWN0LgoKYGBganMKY29uc3QgY2FsbGVycyA9IHsKICBnZXQ6IGFzeW5jICh1cmwpID0+IHsKICAgIGNvbnN0IHJlc3AgPSBhd2FpdCBmZXRjaCh1cmwpOwogICAgY29uc3QganNvbiA9IGF3YWl0IHJlc3AuanNvbigpOwogICAgcmV0dXJuIGpzb247CiAgfSwKICBhdXRob3JpemVkR2V0OiBhc3luYyAodXJsKSA9PiB7CiAgICBjb25zdCBoZWFkZXJzID0gbmV3IEhlYWRlcnMoKTsKICAgIGhlYWRlcnMuYXBwZW5kKCJDb250ZW50LVR5cGUiLCAiYXBwbGljYXRpb24vanNvbiIpOwogICAgaGVhZGVycy5hcHBlbmQoIkF1dGhvcml6YXRpb24iLCBgQmVhcmVyICR7bG9jYWxTdG9yYWdlLnRva2VufWApOwogICAgY29uc3QgcmVzcCA9IGF3YWl0IGZldGNoKHVybCwgeyBoZWFkZXJzIH0pOwogICAgY29uc3QganNvbiA9IGF3YWl0IHJlc3AuanNvbigpOwogICAgcmV0dXJuIGpzb247CiAgfSwKICBhdXRob3JpemVkUG9zdDogYXN5bmMgKHVybCwgYm9keSkgPT4gewogICAgY29uc3QgaGVhZGVycyA9IG5ldyBIZWFkZXJzKCk7CiAgICBoZWFkZXJzLmFwcGVuZCgiQ29udGVudC1UeXBlIiwgImFwcGxpY2F0aW9uL2pzb24iKTsKICAgIGhlYWRlcnMuYXBwZW5kKCJBdXRob3JpemF0aW9uIiwgYEJlYXJlciAke2xvY2FsU3RvcmFnZS50b2tlbn1gKTsKICAgIGNvbnN0IHJlc3AgPSBhd2FpdCBmZXRjaCh1cmwsIHsKICAgICAgbWV0aG9kOiAiUE9TVCIsCiAgICAgIGhlYWRlcnMsCiAgICAgIGJvZHk6IEpTT04uc3RyaW5naWZ5KGJvZHkpLAogICAgfSk7CiAgICBjb25zdCBqc29uID0gYXdhaXQgcmVzcC5qc29uKCk7CiAgICByZXR1cm4ganNvbjsKICB9LAp9OwpgYGAKCiMjIyA1LiBDcmVhdGUgeW91ciBjbGllbnQKCllvdSBkbyB0aGlzIG9uY2UgYW5kIGBjbGllbnRgIGNhbiBiZSB1c2VkIGluIHRoZSByZXN0IG9mIHlvdXIgY29kZSBhZnRlcndhcmQuCgpgb3JpZ2luYCBkZWZhdWx0cyB0byBgc3BlY3Muc2VydmVyc1swXS51cmxgLiBPcHRpb25hbCBpZiBpdCdzIGRlZmluZWQsIGVsc2UgcmVxdWlyZWQuCgpgdmFsaWRhdGlvbkxldmVsYCBpcyBvbmUgb2YgYCdvZmYnYCAoZGVmYXVsdCksIGAnd2FybidgLCBvciBgJ2Vycm9yJ2AuIEl0IGNoZWNrcyB0aGUgcGF0aCBwYXJhbXMsIHRoZSBxdWVyeSBwYXJhbXMsIGFuZCB0aGUgYm9keSBhZ2FpbnN0IHRoZSBzY2hlbWEgcHJlc2VudCBpbiB0aGUgc3BlY3MuCgpgYGBqcwpjb25zdCBjbGllbnQgPSBjcmVhdGVDbGllbnQoc3BlY3MsIGNhbGxlcnMsIHsKICBvcmlnaW46ICJodHRwczovL215LmFwaS5jb20iLAogIHZhbGlkYXRpb25MZXZlbDogImVycm9yIiwKfSk7CmBgYAoKIyMjIDYuIFVzZSB5b3VyIGNsaWVudAoKVGhlcmVhZnRlciwgYG9hLWNsaWVudGAgZG9lcyBhbGwgdGhlIHdvcmsgb2YgYnVpbGRpbmcgdGhlIGZ1bGwgVVJMIGFuZCB2YWxpZGF0aW5nIGlucHV0IGRhdGEgZm9yIHlvdSEKCkluIHRoaXMgZXhhbXBsZSwgdGhpcwoKYGBganMKY29uc3QgZGF0YSA9IGF3YWl0IGNsaWVudFsiL3VzZXJzL3t1c2VySWR9Il0uZ2V0KHsKICBwYXRoUGFyYW1zOiB7IHVzZXJJZDogMTIzIH0sCn0pOwpgYGAKCmlzIGVxdWl2YWxlbnQgdG8KCmBgYGpzCmNvbnN0IHVybCA9IG5ldyBVUkwoImh0dHBzOi8vbXkuYXBpLmNvbS91c2Vycy8xMjMiKTsKY29uc3QgZGF0YSA9IGF3YWl0IGNhbGxlcnMuYXV0aG9yaXplZEdldCh1cmwpOwpgYGAKCiMjIERpZmZlcmVuY2VzIHdpdGggb3BlbmFwaS1jbGllbnQKClRoZSBbb3BlbmFwaS1jbGllbnRdKGh0dHBzOi8vZ2l0aHViLmNvbS9taWtlc3RlYWQvb3BlbmFwaS1jbGllbnQpIHBhY2thZ2UgaXMgc2ltaWxhciBidXQgYWNjb21wbGlzaGVzIHRoaW5ncyBkaWZmZXJlbnRseS4KCmBvcGVuYXBpLWNsaWVudGAgaXMgYSAqKmNvZGUgZ2VuZXJhdGlvbioqIHBhY2thZ2UuIFlvdSB1c2UgaXQgYXMgYSBjb21tYW5kIGxpbmUgc28gdGhhdCBpdCBjb25zdW1lcyBPcGVuQVBJIHNwZWNzIGFuZCBvdXRwdXRzIGNvZGUgdGhhdCB3aWxsIGNhbGwgeW91ciBzZXJ2ZXIuIEl0IGlzIG5vdCBpZGVhbCBiZWNhdXNlIHlvdSBkb24ndCBvd24gYW5kIGNvbnRyb2wgYWxsIG9mIHlvdXIgY29kZSwgYW5kIGl0IGFkZHMgY29tcGxleGl0eS4KCmBvYS1jbGllbnRgIGlzIHNpbXBsZXIgLSBpdCBleHBvc2VzIGBjcmVhdGVDbGllbnRgLCBhICoqZmFjdG9yeSoqIHRoYXQgdGFrZSBzcGVjcyBhcyBpbnB1dCBhbmQgYnVpbGRzIHRoZSBjbGllbnQgYXQgcnVudGltZS4gSWYgeW91ciBBUEkgdXBkYXRlcywgeW91IGRvbid0IGhhdmUgdG8gd3JpdGUgb3IgZ2VuZXJhdGUgYSBzaW5nbGUgbGluZSBvZiBjb2RlLgoKYG9wZW5hcGktY2xpZW50YCBoYW5kbGVzIGFsbCB0aGUgSFRUUCBjYWxscyBhbmQgYXV0aGVudGljYXRpb24gZm9yIHlvdS4gVGhhdCBjYW4gc2VlbSBwb3dlcmZ1bCwgYnV0IGFjdHVhbGx5IHRoZSBzeXN0ZW0gaXMgX3ZlcnlfIHJpZ2lkLCBldmVuIGZvciBzbWFsbCBjdXN0b21pemF0aW9ucywgYW5kIGRvZXNuJ3QgY292ZXIgYWxsIGNhc2VzIHlvdSdsbCBmYWNlIGFsb25nIHRoZSB3YXkuCgpJbiBgb2EtY2xpZW50YCwgeW91IGZ1bGx5IG93biB5b3VyIGdlbmVyaWMgSFRUUCBjYWxsZXJzOiB5b3Ugd3JpdGUgdGhlbSB5b3Vyc2VsZiwgYnV0IHlvdSBwcm9iYWJseSB3b24ndCB3cml0ZSBtb3JlIHRoYW4gZml2ZSBvZiB0aGVtIGR1cmluZyB5b3VyIHdob2xlIHByb2plY3QgbGlmZXRpbWU6IHdobyBuZWVkcyBtb3JlIHRoYW4gZ2V0LCBwb3N0LCBhdXRob3JpemVkIGdldCwgYXV0aG9yaXplZCBwb3N0IGFuZCBmaWxlIHVwbG9hZD8KCmBgYAorLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKfCBXcml0dGVuIHdpdGggPDMgYnkgTmlubyBGaWxpdSB8CnwgIENvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWVkISAgfAorLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSsKICAgICAgICAgXCAgIF5fX14KICAgICAgICAgIFwgIChvbylcX19fX19fXwogICAgICAgICAgICAgKF9fKVwgICAgICAgKVwvXAogICAgICAgICAgICAgICAgIHx8LS0tLXcgfAogICAgICAgICAgICAgICAgIHx8ICAgICB8fApgYGAK readmeEtag: '"6f8355e6d3d524c4d07e934c70a40d077eac6bc1"' readmeLastModified: Fri, 29 Dec 2023 15:35:35 GMT repositoryId: 258327118 description: >- Flexible client helper for making and validating calls to OpenAPI backends. For Node and the browser. Runtime lib - no need for code generation! created: '2020-04-23T20:49:35Z' updated: '2025-08-27T12:23:06Z' language: TypeScript archived: false stars: 50 watchers: 3 forks: 3 owner: ninofiliu logo: https://avatars.githubusercontent.com/u/29477588?v=4 license: MIT repoEtag: '"1fee6f99d107b444c785a8fcb074f7eef90385b937c81d3f411b6c4da339f6b6"' repoLastModified: Wed, 27 Aug 2025 12:23:06 GMT foundInMaster: true id: 90d109eb23c36d0136638c20f8f43cd3 - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags name: Restish category: - Documentation - Testing language: - CLI - Go link: https://rest.sh/ repository: https://github.com/rest-sh/restish source_description: >- A CLI for REST-ish APIs with HTTP/2, built-in auth, content negotiation, caching, and more that understands and can discover OpenAPI descriptions. v3: true repositoryMetadata: base64Readme: >- IVtSZXN0aXNoIExvZ29dKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzEwNjgyNi84MjEwOTkxOC1lYzViMjMwMC05NmVlLTExZWEtOWFmMC04NTE1MzI5ZDU5NjUucG5nKQoKWyFbV29ya3MgV2l0aCBSZXN0aXNoXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1dvcmtzJTIwV2l0aC1SZXN0aXNoLWZmNWY4NyldKGh0dHBzOi8vcmVzdC5zaC8pIFshW1VzZXIgR3VpZGVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvRG9jcy1HdWlkZS01ZmFmZDcpXShodHRwczovL3Jlc3Quc2gvIy9ndWlkZSkgWyFbQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS9yZXN0LXNoL3Jlc3Rpc2gvYWN0aW9ucy93b3JrZmxvd3MvY2kueWFtbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vcmVzdC1zaC9yZXN0aXNoL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnlhbWwpIFshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9yZXN0LXNoL3Jlc3Rpc2gvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL3Jlc3Qtc2gvcmVzdGlzaCkgWyFbRG9jc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9nb2RvYy1yZWZlcmVuY2UtNWZhZmQ3KV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2dpdGh1Yi5jb20vcmVzdC1zaC9yZXN0aXNoP3RhYj1zdWJkaXJlY3RvcmllcykgWyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL3Jlc3Qtc2gvcmVzdGlzaCldKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9yZXN0LXNoL3Jlc3Rpc2gpCgpbUmVzdGlzaF0oaHR0cHM6Ly9yZXN0LnNoLykgaXMgYSBDTEkgZm9yIGludGVyYWN0aW5nIHdpdGggW1JFU1RdKGh0dHBzOi8vYXBpc3lvdXdvbnRoYXRlLmNvbS9ibG9nL3Jlc3QtYW5kLWh5cGVybWVkaWEtaW4tMjAxOSktaXNoIEhUVFAgQVBJcyB3aXRoIHNvbWUgbmljZSBmZWF0dXJlcyBidWlsdC1pbiDigJQgbGlrZSBhbHdheXMgaGF2aW5nIHRoZSBsYXRlc3QgQVBJIHJlc291cmNlcywgZmllbGRzLCBhbmQgb3BlcmF0aW9ucyBhdmFpbGFibGUgd2hlbiB0aGV5IGdvIGxpdmUgb24gdGhlIEFQSSB3aXRob3V0IG5lZWRpbmcgdG8gaW5zdGFsbCBvciB1cGRhdGUgYW55dGhpbmcuCkNoZWNrIG91dCBbaG93IFJlc3Rpc2ggY29tcGFyZXMgdG8gY1VSTCAmIEhUVFBpZV0oaHR0cHM6Ly9yZXN0LnNoLyMvY29tcGFyaXNvbikuCgpTZWUgdGhlIFt1c2VyIGd1aWRlXShodHRwczovL3Jlc3Quc2gvIy9ndWlkZSkgZm9yIGhvdyB0byBpbnN0YWxsIFJlc3Rpc2ggYW5kIGdldCBzdGFydGVkLgoKRmVhdHVyZXMgaW5jbHVkZToKCi0gSFRUUC8yIChbUkZDIDc1NDBdKGh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmM3NTQwKSkgd2l0aCBUTFMgYnkgX2RlZmF1bHRfIHdpdGggZmFsbGJhY2sgdG8gSFRUUC8xLjEKLSBHZW5lcmljIGhlYWQvZ2V0L3Bvc3QvcHV0L3BhdGNoL2RlbGV0ZSB2ZXJicyBsaWtlIGBjdXJsYCBvciBbSFRUUGllXShodHRwczovL2h0dHBpZS5vcmcvKQotIEdlbmVyYXRlZCBjb21tYW5kcyBmb3IgQ0xJIG9wZXJhdGlvbnMsIGUuZy4gYHJlc3Rpc2ggbXktYXBpIGxpc3QtdXNlcnNgCiAgLSBBdXRvbWF0aWNhbGx5IGRpc2NvdmVycyBBUEkgZGVzY3JpcHRpb25zCiAgICAtIFtSRkMgODYzMV0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzg2MzEpIGBzZXJ2aWNlLWRlc2NgIGxpbmsgcmVsYXRpb24KICAgIC0gW1JGQyA1OTg4XShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNTk4OCNzZWN0aW9uLTYuMi4yKSBgZGVzY3JpYmVkYnlgIGxpbmsgcmVsYXRpb24KICAtIFN1cHBvcnRlZCBmb3JtYXRzCiAgICAtIE9wZW5BUEkgWzMuMF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjMubWQpIC8gWzMuMV0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMS4wLmh0bWwpIGFuZCBbSlNPTiBTY2hlbWFdKGh0dHBzOi8vanNvbi1zY2hlbWEub3JnLykKICAtIEF1dG9tYXRpYyBjb25maWd1cmF0aW9uIG9mIEFQSSBhdXRoIGlmIGFkdmVydGlzZWQgYnkgdGhlIEFQSQogIC0gU2hlbGwgY29tbWFuZCBjb21wbGV0aW9uIGZvciBCYXNoLCBGaXNoLCBac2gsIFBvd2Vyc2hlbGwKLSBBdXRvbWF0aWMgcGFnaW5hdGlvbiBvZiByZXNvdXJjZSBjb2xsZWN0aW9ucyB2aWEgW1JGQyA1OTg4XShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNTk4OCkgYHByZXZgIGFuZCBgbmV4dGAgaHlwZXJtZWRpYSBsaW5rcwotIEFQSSBlbmRwb2ludC1iYXNlZCBhdXRoIGJ1aWx0LWluIHdpdGggc3VwcG9ydCBmb3IgcHJvZmlsZXM6CiAgLSBIVFRQIEJhc2ljCiAgLSBBUEkga2V5IHZpYSBoZWFkZXIgb3IgcXVlcnkgcGFyYW0KICAtIE9BdXRoMiBjbGllbnQgY3JlZGVudGlhbHMgZmxvdyAobWFjaGluZS10by1tYWNoaW5lLCBbUkZDIDY3NDldKGh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmM2NzQ5KSkKICAtIE9BdXRoMiBhdXRob3JpemF0aW9uIGNvZGUgKHdpdGggUEtDRSBbUkZDIDc2MzZdKGh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmM3NjM2KSkgZmxvdwogIC0gT24gdGhlIGZseSBhdXRob3JpemF0aW9uIHRocm91Z2ggZXh0ZXJuYWwgdG9vbHMgZm9yIGN1c3RvbSBBUEkgc2lnbmF0dXJlIG1lY2hhbmlzbXMKLSBDb250ZW50IG5lZ290aWF0aW9uLCBkZWNvZGluZyAmIHVubWFyc2hhbGxpbmcgYnVpbHQtaW46CiAgLSBKU09OIChbUkZDIDgyNTldKGh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmM4MjU5KSwgPGh0dHBzOi8vd3d3Lmpzb24ub3JnLz4pCiAgLSBZQU1MICg8aHR0cHM6Ly95YW1sLm9yZy8+KQogIC0gQ0JPUiAoW1JGQyA3MDQ5XShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNzA0OSksIDxodHRwOi8vY2Jvci5pby8+KQogIC0gTWVzc2FnZVBhY2sgKDxodHRwczovL21zZ3BhY2sub3JnLz4pCiAgLSBBbWF6b24gSW9uICg8aHR0cDovL2Ftem4uZ2l0aHViLmlvL2lvbi1kb2NzLz4pCiAgLSBHemlwIChbUkZDIDE5NTJdKGh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmMxOTUyKSksIERlZmxhdGUgKFtSRkMgMTk1MV0oaHR0cHM6Ly9kYXRhdHJhY2tlci5pZXRmLm9yZy9kb2MvaHRtbC9yZmMxOTUxKSksIGFuZCBCcm90bGkgKFtSRkMgNzkzMl0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzc5MzIpKSBjb250ZW50IGVuY29kaW5nCi0gQXV0b21hdGljIHJldHJpZXMgd2l0aCBzdXBwb3J0IGZvciBbYFJldHJ5LUFmdGVyYF0oaHR0cHM6Ly9kZXZlbG9wZXIubW96aWxsYS5vcmcvZW4tVVMvZG9jcy9XZWIvSFRUUC9IZWFkZXJzL1JldHJ5LUFmdGVyKSBhbmQgYFgtUmV0cnktSW5gIGhlYWRlcnMgd2hlbiBBUElzIGFyZSByYXRlLWxpbWl0ZWQuCi0gU3RhbmRhcmRpemVkIFtoeXBlcm1lZGlhXShodHRwczovL3NtYXJ0YmVhci5jb20vbGVhcm4vYXBpLWRlc2lnbi93aGF0LWlzLWh5cGVybWVkaWEvKSBwYXJzaW5nIGludG8gcXVlcnlhYmxlL2ZvbGxvd2FibGUgcmVzcG9uc2UgbGlua3M6CiAgLSBIVFRQIExpbmsgcmVsYXRpb24gaGVhZGVycyAoW1JGQyA1OTg4XShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNTk4OCNzZWN0aW9uLTYuMi4yKSkKICAtIFtIQUxdKGh0dHA6Ly9zdGF0ZWxlc3MuY28vaGFsX3NwZWNpZmljYXRpb24uaHRtbCkKICAtIFtTaXJlbl0oaHR0cHM6Ly9naXRodWIuY29tL2tldmluc3dpYmVyL3NpcmVuKQogIC0gW1RlcnJpZmljYWxseSBTaW1wbGUgSlNPTl0oaHR0cHM6Ly9naXRodWIuY29tL21wbmFsbHkvVGVycmlmaWNhbGx5LVNpbXBsZS1KU09OKQogIC0gW0pTT046QVBJXShodHRwczovL2pzb25hcGkub3JnLykKLSBMb2NhbCBjYWNoaW5nIHRoYXQgcmVzcGVjdHMgW1JGQyA3MjM0XShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNzIzNCkgYENhY2hlLUNvbnRyb2xgIGFuZCBgRXhwaXJlc2AgaGVhZGVycwotIENMSSBbc2hvcnRoYW5kXShodHRwczovL2dpdGh1Yi5jb20vZGFuaWVsZ3RheWxvci9vcGVuYXBpLWNsaS1nZW5lcmF0b3IvdHJlZS9tYXN0ZXIvc2hvcnRoYW5kI2NsaS1zaG9ydGhhbmQtc3ludGF4KSBmb3Igc3RydWN0dXJlZCBkYXRhIGlucHV0IChlLmcuIGZvciBKU09OKQotIFtTaG9ydGhhbmQgcXVlcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9kYW5pZWxndGF5bG9yL3Nob3J0aGFuZCNxdWVyeWluZykgcmVzcG9uc2UgZmlsdGVyaW5nICYgcHJvamVjdGlvbgotIENvbG9yaXplZCBwcmV0dGlmaWVkIHJlYWRhYmxlIG91dHB1dAotIEZhc3QgbmF0aXZlIHplcm8tZGVwZW5kZW5jeSBiaW5hcnkKCkFydGljbGVzOgoKLSBbQSBDTEkgZm9yIFJFU1QgQVBJc10oaHR0cHM6Ly9kZXYudG8vZGFuaWVsZ3RheWxvci9hLWNsaS1mb3ItcmVzdC1hcGlzLXBhcnQtMS0xMDRiKQotIFtNYXBwaW5nIE9wZW5BUEkgdG8gdGhlIENMSV0oaHR0cHM6Ly9kZXYudG8vZGFuaWVsZ3RheWxvci9tYXBwaW5nLW9wZW5hcGktdG8tdGhlLWNsaS0zN3BiKQoKVGhpcyBwcm9qZWN0IHN0YXJ0ZWQgbGlmZSBhcyBhIGZvcmsgb2YgW09wZW5BUEkgQ0xJIEdlbmVyYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL2RhbmllbGd0YXlsb3Ivb3BlbmFwaS1jbGktZ2VuZXJhdG9yKS4K readmeEtag: '"d8995b9baacf83a85ad17f3b566fc1f9a51b34aa"' readmeLastModified: Sun, 04 May 2025 17:37:51 GMT repositoryId: 264117985 description: >- Restish is a CLI for interacting with REST-ish HTTP APIs with some nice features built-in created: '2020-05-15T06:45:59Z' updated: '2026-02-05T19:58:42Z' language: Go archived: false stars: 1203 watchers: 6 forks: 89 owner: rest-sh logo: https://avatars.githubusercontent.com/u/207975437?v=4 license: MIT repoEtag: '"d43f8e270ec990014b5e9bfd6a9158d4196665956c813fa576959c85d77f2e4c"' repoLastModified: Thu, 05 Feb 2026 19:58:42 GMT foundInMaster: true id: 19ff073f2b77ceec34553a5d412e8ba6 v3_1: true oldLocations: - https://github.com/danielgtaylor/restish - source: - https://openapi.tools/ - openapi3 tags name: openapi-examples-validator category: - Description Validators - Data Validators - Parsers repository: https://github.com/codekie/openapi-examples-validator language: JavaScript source_description: Validates embedded JSON-examples in OpenAPI-specs v2: true v3: true repositoryMetadata: base64Readme: >- b3BlbmFwaS1leGFtcGxlcy12YWxpZGF0b3IKPT09PT09PT09PT09PT09PT09PT09PT09PT0KClZhbGlkYXRlcyBlbWJlZGRlZCBKU09OLWV4YW1wbGVzIGluIE9wZW5BUEktc3BlY3MgKHYyIGFuZCB2MyBhcmUgc3VwcG9ydGVkKQoKWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9vcGVuYXBpLWV4YW1wbGVzLXZhbGlkYXRvci5zdmcpXShodHRwczovL2JhZGdlLmZ1cnkuaW8vanMvb3BlbmFwaS1leGFtcGxlcy12YWxpZGF0b3IpClshW1N0YW5kYXJkIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcmVsZWFzZS1zdGFuZGFyZCUyMHZlcnNpb24tYnJpZ2h0Z3JlZW4uc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2NvbnZlbnRpb25hbC1jaGFuZ2Vsb2cvc3RhbmRhcmQtdmVyc2lvbikKWyFbUnVuIHRlc3RzXShodHRwczovL2dpdGh1Yi5jb20vY29kZWtpZS9vcGVuYXBpLWV4YW1wbGVzLXZhbGlkYXRvci9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LWFuZC1kZXBsb3ktZG9ja2VyLnltbC9iYWRnZS5zdmc/cXVlcnk9YnJhbmNoJTNBbWFpbildKGh0dHBzOi8vZ2l0aHViLmNvbS9jb2Rla2llL29wZW5hcGktZXhhbXBsZXMtdmFsaWRhdG9yL2FjdGlvbnMvd29ya2Zsb3dzL3Rlc3QtYW5kLWRlcGxveS1kb2NrZXIueW1sP3F1ZXJ5PWJyYW5jaCUzQW1haW4pClshW0NvdmVyYWdlIFN0YXR1c10oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vcmVwb3MvZ2l0aHViL2NvZGVraWUvb3BlbmFwaS1leGFtcGxlcy12YWxpZGF0b3IvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL2NvZGVraWUvb3BlbmFwaS1leGFtcGxlcy12YWxpZGF0b3I/YnJhbmNoPW1haW4pClshW011dGF0aW9uIHRlc3RpbmcgYmFkZ2VdKGh0dHBzOi8vYmFkZ2Uuc3RyeWtlci1tdXRhdG9yLmlvL2dpdGh1Yi5jb20vY29kZWtpZS9vcGVuYXBpLWV4YW1wbGVzLXZhbGlkYXRvci9tYWluKV0oaHR0cHM6Ly9zdHJ5a2VyLW11dGF0b3IuZ2l0aHViLmlvKQpbIVtNYWludGFpbmFiaWxpdHldKGh0dHBzOi8vYXBpLmNvZGVjbGltYXRlLmNvbS92MS9iYWRnZXMvNTA5NGY2YWM3NzU0ZTVhMThiMWIvbWFpbnRhaW5hYmlsaXR5KV0oaHR0cHM6Ly9jb2RlY2xpbWF0ZS5jb20vZ2l0aHViL2NvZGVraWUvb3BlbmFwaS1leGFtcGxlcy12YWxpZGF0b3IvbWFpbnRhaW5hYmlsaXR5KQpbIVtLbm93biBWdWxuZXJhYmlsaXRpZXNdKGh0dHBzOi8vc255ay5pby90ZXN0L2dpdGh1Yi9jb2Rla2llL29wZW5hcGktZXhhbXBsZXMtdmFsaWRhdG9yL2JhZGdlLnN2ZyldKGh0dHBzOi8vc255ay5pby90ZXN0L2dpdGh1Yi9jb2Rla2llL29wZW5hcGktZXhhbXBsZXMtdmFsaWRhdG9yKQpbIVtEb2NrZXIgSHViXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2RvY2tlci1yZWFkeS1ibHVlLnN2ZyldKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9jb2Rla2llL29wZW5hcGktZXhhbXBsZXMtdmFsaWRhdG9yKQoKUHJlcmVxdWlzaXRlcwotLS0tLS0tLS0tLS0KCi0gW05vZGUuanNdKGh0dHBzOi8vbm9kZWpzLm9yZy8pID49MTguCgpJbnN0YWxsCi0tLS0tLS0KCkluc3RhbGwgdXNpbmcgW25wbV0oaHR0cHM6Ly9kb2NzLm5wbWpzLmNvbS9nZXR0aW5nLXN0YXJ0ZWQvd2hhdC1pcy1ucG0pOgoKICAgIG5wbSBpbnN0YWxsIC1nIG9wZW5hcGktZXhhbXBsZXMtdmFsaWRhdG9yCgpVc2FnZQotLS0tLQoKYGBgCm9wZW5hcGktZXhhbXBsZXMtdmFsaWRhdG9yIFtvcHRpb25zXSA8ZmlsZXBhdGg+CgpWYWxpZGF0ZSBlbWJlZGRlZCBleGFtcGxlcyBpbiBPcGVuQVBJLXNwZWNzIChKU09OIGFuZCBZQU1MIHN1cHBvcnRlZCkuCiAgVG8gdmFsaWRhdGUgZXh0ZXJuYWwgZXhhbXBsZXMsIHVzZSB0aGUgYC1zYCBhbmQgYC1lYCBvcHRpb24uCiAgVG8gcGFzcyBhIG1hcHBpbmctZmlsZSwgdG8gdmFsaWRhdGUgbXVsdGlwbGUgZXh0ZXJuYWwgZXhhbXBsZXMsIHVzZSB0aGUgYC1tYCBvcHRpb24uCgpPcHRpb25zOgogIC1WLCAtLXZlcnNpb24gICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvdXRwdXQgdGhlIHZlcnNpb24gbnVtYmVyCiAgLXMsIC0tc2NoZW1hLWpzb25wYXRoIDxzY2hlbWEtanNvbnBhdGg+ICAgIFBhdGggdG8gT3BlbkFQSS1zY2hlbWEsIHRvIHZhbGlkYXRlIHRoZSBleGFtcGxlIGZpbGUgYWdhaW5zdAogIC1lLCAtLWV4YW1wbGUtZmlsZXBhdGggPGV4YW1wbGUtZmlsZXBhdGg+ICBmaWxlIHBhdGggdG8gZXhhbXBsZSBmaWxlLCB0byBiZSB2YWxpZGF0ZWQKICAtbSwgLS1tYXBwaW5nLWZpbGVwYXRoIDxtYXBwaW5nLWZpbGVwYXRoPiAgZmlsZSBwYXRoIHRvIG1hcCwgY29udGFpbmluZyBzY2hlbWEtcGF0aHMgYXMga2V5IGFuZCB0aGUgZmlsZS1wYXRoKHMpIHRvCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGV4YW1wbGVzIGFzIHZhbHVlLiBJZiB3aWxkY2FyZHMgYXJlIHVzZWQsIHRoZSBwYXJhbWV0ZXIgaGFzIHRvIGJlIHB1dCBpbgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBxdW90ZXMuCiAgLWMsIC0tY3dkLXRvLW1hcHBpbmctZmlsZSAgICAgICAgICAgICAgICAgIGNoYW5nZXMgdG8gdGhlIGRpcmVjdG9yeSBvZiB0aGUgbWFwcGluZy1maWxlLCBiZWZvcmUgcmVzb2x2aW5nIHRoZQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBleGFtcGxlJ3MgcGF0aHMuIFVzZSB0aGlzIG9wdGlvbiwgaWYgeW91ciBtYXBwaW5nLWZpbGVzIHVzZSByZWxhdGl2ZSBwYXRocwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBmb3IgdGhlIGV4YW1wbGVzCiAgLW4sIC0tbm8tYWRkaXRpb25hbC1wcm9wZXJ0aWVzICAgICAgICAgICAgIGRvbid0IGFsbG93IHByb3BlcnRpZXMgdGhhdCBhcmUgbm90IGRlc2NyaWJlZCBpbiB0aGUgc2NoZW1hCiAgLXIsIC0tYWxsLXByb3BlcnRpZXMtcmVxdWlyZWQgICAgICAgICAgICAgIG1ha2UgYWxsIHRoZSBwcm9wZXJ0aWVzIGluIHRoZSBzY2hlbWEgcmVxdWlyZWQKICAtbywgLS1pZ25vcmUtZm9ybWF0cyA8aWdub3JlZC1mb3JtYXRzLi4uPiAgRGF0YXR5cGUgZm9ybWF0cyB0byBpZ25vcmUgKHRvIHByZXZlbnQgInVua25vd24gZm9ybWF0IiBlcnJvcnMuKQogIC1oLCAtLWhlbHAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvdXRwdXQgdXNhZ2UgaW5mb3JtYXRpb24KYGBgYAoKVGhlIHZhbGlkYXRvciB3aWxsIHNlYXJjaCB0aGUgT3BlbkFQSS1zcGVjIGZvciByZXNwb25zZS1leGFtcGxlcyBhbmQgdmFsaWRhdGUgdGhlbSBhZ2FpbnN0IGl0cyBzY2hlbWEuCgpJZiBhbiBleHRlcm5hbCBleGFtcGxlIGhhcyB0byBiZSB2ZXJpZmllZCwgdGhlIGAtc2AgYW5kIGAtZWAgb3B0aW9uIGhhcyB0byBiZSB1c2VkLgoKRm9yIGV4YW1wbGU6CgpgYGAKJCBvcGVuYXBpLWV4YW1wbGVzLXZhbGlkYXRvciAtcyAkLnBhdGhzLi8uZ2V0LnJlc3BvbnNlcy4yMDAuc2NoZW1hIC1lIGV4YW1wbGUuanNvbiBvcGVuYXBpLXNwZWMuanNvbgpgYGAKClRvIHZhbGlkYXRlIG11bHRpcGxlIGV4dGVybmFsIGV4YW1wbGVzLCBwYXNzIGEgbWFwcGluZyBmaWxlIHdpdGggYSBzaW1pbGFyIHN0cnVjdHVyZSBhbG9uZyB3aXRoIHRoZSBgLW1gIG9wdGlvbjoKCmBgYGpzb24KewogICIkLnBhdGhzLi8uZ2V0LnJlc3BvbnNlcy4yMDAuc2NoZW1hIjogWwogICAgInRlc3QvZGF0YS9leHRlcm5hbC1leGFtcGxlcy12YWxpZC1leGFtcGxlMS5qc29uIiwKICAgICJ0ZXN0L2RhdGEvZXh0ZXJuYWwtZXhhbXBsZXMtdmFsaWQtZXhhbXBsZTIuanNvbiIsCiAgICAidGVzdC9kYXRhL2V4dGVybmFsLWV4YW1wbGVzLWludmFsaWQtdHlwZS5qc29uIgogIF0sCiAgIiQucGF0aHMuLy5nZXQucmVzcG9uc2VzLjMwMC5zY2hlbWEiOiAidGVzdC9kYXRhL2V4dGVybmFsLWV4YW1wbGVzLWludmFsaWQtbWlzc2luZy1saW5rLmpzb24iLAogICIkLnBhdGhzLi8ucG9zdC5wYXJhbWV0ZXJzWz8oQC5pbj09PSJib2R5IildLnNjaGVtYSI6ICJ0ZXN0L2RhdGEvdjIvcG9zdC1yZXF1ZXN0LyouanNvbiIKfQpgYGAKSXQgaXMgcG9zc2libGUgdG8gdXNlIHdpbGRjYXJkcyBpbiB0aGUgY29tbWFuZCBsaW5lIHRvIHNlbGVjdCBtdWx0aXBsZSBtYXBwaW5nIGZpbGVzLCBhbmQgaXQgaXMgcG9zc2libGUgdG8gdXNlIHdpbGRjYXJkcyBpbnNpZGUgdGhlIG1hcHBpbmcgZmlsZSB0byBzZWxlY3QgbXVsdGlwbGUgZXhhbXBsZXMuCgpFcnJvcnMgd2lsbCBiZSB3cml0dGVuIHRvIGBzdGRlcnJgLgoKU2FtcGxlIG91dHB1dCBvZiB2YWxpZGF0aW9uIGVycm9yczoKCmBgYGpzb24KWwogICAgewogICAgICAgICJrZXl3b3JkIjogInR5cGUiLAogICAgICAgICJpbnN0YW5jZVBhdGgiOiAiL3ZlcnNpb25zLzAvaWQiLAogICAgICAgICJzY2hlbWFQYXRoIjogIiMvcHJvcGVydGllcy92ZXJzaW9ucy9pdGVtcy9wcm9wZXJ0aWVzL2lkL3R5cGUiLAogICAgICAgICJwYXJhbXMiOiB7CiAgICAgICAgICAgICJ0eXBlIjogInN0cmluZyIKICAgICAgICB9LAogICAgICAgICJtZXNzYWdlIjogIm11c3QgYmUgc3RyaW5nIiwKICAgICAgICAiZXhhbXBsZVBhdGgiOiAiL34xL2dldC9yZXNwb25zZXMvMjAwL2V4YW1wbGVzL2FwcGxpY2F0aW9ufjFqc29uIgogICAgfQpdCmBgYAoKRG9ja2VyCi0tLS0tLQoKRXhhbXBsZSB1c2FnZToKCmBgYHNoZWxsCiQgZG9ja2VyIHJ1biAtLXJtIC1pIFwKICAgIC0tdXNlcj0kKGlkIC11KSBcCiAgICAtdiAke1BXRH06L2RhdGEgXAogICAgY29kZWtpZS9vcGVuYXBpLWV4YW1wbGVzLXZhbGlkYXRvcjpsYXRlc3QgXAogICAgL2RhdGEvdGVzdC9kYXRhL3YzL3NpbXBsZS1hcGktd2l0aC1leGFtcGxlcy13aXRoLXJlZnMtaW52YWxpZC55bWwKYGBgCgpDYXZlYXQKLS0tLS0tCgotIFRoZSBmb3JtYXRzIGBpbnQzMmAsIGBmbG9hdGAgYW5kIGBkb3VibGVgIGFyZSBzdXBwb3J0ZWQgZm9yIHRoZSB0eXBlIGBudW1iZXJgLiBUaGUgZm9ybWF0IGBpbnQ2NGAgaXMgb25seSBhdmFpbGFibGUKICBmb3IgdGhlIHR5cGUgYHN0cmluZ2AsIHRob3VnaCAoZHVlIHRvIHRoZSBwcmVjaXNpb24tbGltaXRhdGlvbnMgb2YgSmF2YXNjcmlwdCkuCi0gVGhlIG9wdGlvbnMgYC0tbm8tYWRkaXRpb25hbC1wcm9wZXJ0aWVzYCBhbmQgYC0tYWxsLXByb3BlcnRpZXMtcmVxdWlyZWRgIGFyZSBub3QgY29tcGF0aWJsZSB3aXRoIFtzdWItc2NoZW1hcyBjb21iaW5lciBrZXl3b3JkXShodHRwczovL2pzb24tc2NoZW1hLm9yZy91bmRlcnN0YW5kaW5nLWpzb24tc2NoZW1hL3JlZmVyZW5jZS9jb21iaW5pbmcuaHRtbCkuIEEgd2FybmluZyB3aWxsIGJlIGxvZ2dlZCBpZiBvbmUgbW9kZWwgaXMgc2tpcHBlZCBiZWNhdXNlIGl0IGNvbnRhaW5zIGEgY29tYmluZXIga2V5d29yZC4KClRlc3QKLS0tLQoKVG8gcnVuIHRoZSB0ZXN0cywgZXhlY3V0ZQoKICAgIG5wbSB0ZXN0CgpvciB0byBjaGVjayB0aGUgY292ZXJhZ2UKCiAgICBucG0gcnVuIGNvdmVyYWdlCg== readmeEtag: '"38cd5254578e2dff6458749ef1bbd301ec3c85c2"' readmeLastModified: Wed, 20 Nov 2024 12:08:41 GMT repositoryId: 78840041 description: Validates embedded examples in OpenAPI-files created: '2017-01-13T10:34:47Z' updated: '2025-12-25T09:37:02Z' language: JavaScript archived: false stars: 61 watchers: 2 forks: 14 owner: codekie logo: https://avatars.githubusercontent.com/u/1512840?v=4 license: MIT repoEtag: '"955c7bf3c1b975a7328d5c9b39257ab7a46d352c84c0208f17738408b26eebf5"' repoLastModified: Thu, 25 Dec 2025 09:37:02 GMT foundInMaster: true id: 301e3f380f248daf345a88c8971d2a71 - source: - https://openapi.tools/ - openapi3 tags name: openapi-to-postman category: - Converters - Parsers repository: https://github.com/postmanlabs/openapi-to-postman language: JavaScript source_description: Convert OpenAPI and Swagger specs to the Postman Collection (v2) format v2: true v3: true repositoryMetadata: base64Readme: >- 
<img src="https://voyager.postman.com/logo/postman-logo-orange.svg" width="320" alt="The Postman Logo">

*Supercharge your API workflow.*
*Modern software is built on APIs. Postman helps you develop APIs faster.*

# OpenAPI 3.0, 3.1 and Swagger 2.0 to Postman Collection

![Build Status](https://github.com/postmanlabs/openapi-to-postman/actions/workflows/integration.yml/badge.svg)

<a href="https://www.npmjs.com/package/openapi-to-postmanv2" alt="Latest Stable Version">![npm](https://img.shields.io/npm/v/openapi-to-postmanv2.svg)</a>
<a href="https://www.npmjs.com/package/openapi-to-postmanv2" alt="Total Downloads">![npm](https://img.shields.io/npm/dw/openapi-to-postmanv2.svg)</a>

#### Contents

1. [Getting Started](#getting-started)
2. [Command Line Interface](#command-line-interface)
    1. [Options](#options)
    2. [Usage](#usage)
    3. [Syncing Collections](#syncing-collections)
3. [Using the converter as a NodeJS module](#using-the-converter-as-a-nodejs-module)
    1. [Convert Function](#convert)
    2. [Sync Collection Function](#sync-collection)
    3. [Options](#options)
    4. [ConversionResult](#conversionresult)
    5. [Sample usage](#sample-usage)
    6. [Validate function](#validate-function)
4. [Conversion Schema](#conversion-schema)

---

---

### 🚀 We now also support OpenAPI 3.1 and Swagger 2.0 along with OpenAPI 3.0.
---
---

<h2 id="getting-started">💭 Getting Started</h2>

To use the converter as a Node module, you need to have a copy of the NodeJS runtime. The easiest way to do this is through npm. If you have NodeJS installed you have npm installed as well.

```terminal
$ npm install openapi-to-postmanv2
```

If you want to use the converter in the CLI, install it globally with NPM:

```terminal
$ npm i -g openapi-to-postmanv2
```


<h2 id="command-line-interface">📖 Command Line Interface</h2>

The converter can be used as a CLI tool as well. The following [command line options](#options) are available.

`openapi2postmanv2 [options]`

### Options

- `-s <source>`, `--spec <source>`
  Used to specify the OpenAPI specification (file path) which is to be converted

- `-o <destination>`, `--output <destination>`
  Used to specify the destination file in which the collection is to be written

- `-p`, `--pretty`
  Used to pretty print the collection object while writing to a file

- `-i`, `--interface-version`
  Specifies the interface version of the converter to be used. Value can be 'v2' or 'v1'. Default is 'v2'.

- `-O`, `--options`
  Used to supply options to the converter, for complete options details see [here](/OPTIONS.md)

- `-c`, `--options-config`
  Used to supply options to the converter through config file, for complete options details see [here](/OPTIONS.md)

- `-t`, `--test`
  Used to test the collection with an in-built sample specification

- `-v`, `--version`
  Specifies the version of the converter

- `-h`, `--help`
  Specifies all the options along with a few usage examples on the terminal

- `--sync <collectionFile>`
  Sync spec changes with an existing Postman Collection file

- `--sync-options <syncOptions>`
  Comma-separated list of sync options (e.g., `syncExamples=true`)

- `--sync-options-config <syncOptionsConfig>`
  JSON file containing sync options


###  Usage

- Takes a specification (spec.yaml) as an input and writes to a file (collection.json) with pretty printing and using provided options
```terminal
$ openapi2postmanv2 -s spec.yaml -o collection.json -p -O folderStrategy=Tags,includeAuthInfoInExample=false
```

- Takes a specification (spec.yaml) as an input and writes to a file (collection.json) with pretty printing and using provided options via config file
```terminal
$ openapi2postmanv2 -s spec.yaml -o collection.json -p  -c ./examples/cli-options-config.json
```

- Takes a specification (spec.yaml) as an input and writes to a file (collection.json) with pretty printing and using provided options (Also avoids any `"<Error: Too many levels of nesting to fake this schema>"` kind of errors present in converted collection)
```terminal
$ openapi2postmanv2 -s spec.yaml -o collection.json -p -O folderStrategy=Tags,requestParametersResolution=Example,optimizeConversion=false,stackLimit=50
```

- Testing the converter
```terminal
$ openapi2postmanv2 --test
```

### Syncing Collections

The sync functionality allows you to update an existing Postman collection with changes from your OpenAPI specification.

#### Basic Sync

```terminal
$ openapi2postmanv2 -s spec.yaml --sync collection.json -o synced-collection.json
```

#### Sync with Options

```terminal
$ openapi2postmanv2 -s spec.yaml --sync collection.json --sync-options syncExamples=true -o synced-collection.json
```

#### Sync with Config File

```terminal
$ openapi2postmanv2 -s spec.yaml --sync collection.json --sync-options-config sync-options.json -o synced-collection.json
```

**sync-options.json:**
```json
{
  "syncExamples": true
}
```

For a complete list of sync options and their usage, see [SYNC_OPTIONS.md](/SYNC_OPTIONS.md)


<h2 id="using-the-converter-as-a-nodejs-module">🛠 Using the converter as a NodeJS module</h2>

In order to use the convert in your node application, you need to import the package using `require`.

```javascript
var Converter = require('openapi-to-postmanv2')
```

The converter provides the following functions:

### Convert

The convert function takes in your OpenAPI 3.0, 3.1 and Swagger 2.0 specification (YAML / JSON) and converts it to a Postman collection.

Signature: `convert (data, options, callback);`

**data:**

```javascript
{ type: 'file', data: 'filepath' }
OR
{ type: 'string', data: '<entire OpenAPI string - JSON or YAML>' }
OR
{ type: 'json', data: OpenAPI-JS-object }
```

**options:**
```javascript
{
  schemaFaker: true,
  requestNameSource: 'fallback',
  indentCharacter: ' '
}
/*
All three properties are optional. Check the options section below for possible values for each option.
*/
```
Note: All possible values of options and their usage can be found over here: [OPTIONS.md](/OPTIONS.md)

**callback:**
```javascript
function (err, result) {
  /*
  result = {
    result: true,
    output: [
      {
        type: 'collection',
        data: {..collection object..}
      }
    ]
  }
  */
}
```

### Sync Collection

The syncCollection function syncs changes from an OpenAPI specification with an existing Postman collection.

Signature: `syncCollection (data, options, collection, syncOptions, callback);`

**data:**

```javascript
{ type: 'file', data: 'filepath' }
OR
{ type: 'string', data: '<entire OpenAPI string - JSON or YAML>' }
OR
{ type: 'json', data: OpenAPI-JS-object }
```

**options:**
```javascript
{
  parametersResolution: 'Example',
  folderStrategy: 'Tags'
}
/*
Regular conversion options. Check the options section below for possible values.
*/
```

**collection:**
```javascript
// Existing Postman collection object (JSON)
const collection = {
  info: { name: 'My API', schema: '...' },
  item: [...]
}
```

**syncOptions:**
```javascript
{
  syncExamples: true
}
/*
Sync-specific options that control the merge behavior.
All sync options are optional. Check SYNC_OPTIONS.md for details.
*/
```
Note: All possible values of sync options can be found here: [SYNC_OPTIONS.md](/SYNC_OPTIONS.md)

**callback:**
```javascript
function (err, result) {
  /*
  result = {
    result: true,
    output: [
      {
        type: 'collection',
        data: {..synced collection object..}
      }
    ]
  }
  */
}
```

### Options

Check out complete list of options and their usage at [OPTIONS.md](/OPTIONS.md)

### ConversionResult

- `result` - Flag responsible for providing a status whether the conversion was successful or not.

- `reason` - Provides the reason for an unsuccessful conversion, defined only if result if `false`.

- `output` - Contains an array of Postman objects, each one with a `type` and `data`. The only type currently supported is `collection`.



### Sample Usage

#### Converting a specification

```javascript
const fs = require('fs'),
  Converter = require('openapi-to-postmanv2'),
  openapiData = fs.readFileSync('sample-spec.yaml', {encoding: 'UTF8'});

Converter.convert({ type: 'string', data: openapiData },
  {}, (err, conversionResult) => {
    if (!conversionResult.result) {
      console.log('Could not convert', conversionResult.reason);
    }
    else {
      console.log('The collection object is: ', conversionResult.output[0].data);
    }
  }
);
```

#### Syncing a specification with an existing collection

```javascript
const fs = require('fs'),
  Converter = require('openapi-to-postmanv2'),
  openapiData = fs.readFileSync('sample-spec.yaml', {encoding: 'UTF8'}),
  existingCollection = JSON.parse(fs.readFileSync('collection.json', {encoding: 'UTF8'}));

const syncOptions = {
  syncExamples: true
};

Converter.syncCollection(
  { type: 'string', data: openapiData },
  {}, // conversion options
  existingCollection,
  syncOptions,
  (err, syncResult) => {
    if (!syncResult.result) {
      console.log('Could not sync', syncResult.reason);
    }
    else {
      console.log('The collection object is: ', syncResult.output[0].data);
    }
  }
);
```

### Validate Function

The validate function is meant to ensure that the data that is being passed to the [convert function](#convert-function) is a valid JSON object or a valid (YAML/JSON) string.

The validate function is synchronous and returns a status object which conforms to the following schema

#### Validation object schema

```javascript
{
  type: 'object',
  properties: {
    result: { type: 'boolean'},
    reason: { type: 'string' }
  },
  required: ['result']
}
```

##### Validation object explanation
- `result` - true if the data looks like OpenAPI and can be passed to the convert function

- `reason` - Provides a reason for an unsuccessful validation of the specification

<h2 id="conversion-schema">🧭 Conversion Schema</h2>

| *postman* | *openapi* | *related options* |
| --- | --- | :---: |
| collectionName | info.title | - |
| description | info.description + info.contact | - |
| collectionVariables| server.variables + pathVariables | - |
| folderName | paths.path / tags.name | folderStrategy |
| requestName | operationItem(method).summary / operationItem(method).operationId / url | requestNameSource |
| request.method | path.method | - |
| request.headers | parameter (`in = header`) | - | [link](#Header/Path-param-conversion-example) |
| request.body | operationItem(method).requestBody | requestParametersResolution, exampleParametersResolution |
| request.url.raw | server.url (path level server >> openapi server) + path | - |
| request.url.variables | parameter (`in = path`) | - |
| request.url.params | parameter (`in = query`) | - |
| api_key in (query or header) | components.securitySchemes.api_key | includeAuthInfoInExample |
 readmeEtag: '"0e43225da907420b41c2babd5dd293548fdde191"' readmeLastModified: Wed, 28 Jan 2026 13:09:59 GMT repositoryId: 161469924 description: >- Plugin for converting OpenAPI 3.0 specs to the Postman Collection (v2) format created: '2018-12-12T10:13:46Z' updated: '2026-02-02T04:38:45Z' language: JavaScript archived: false stars: 1042 watchers: 20 forks: 233 owner: postmanlabs logo: https://avatars.githubusercontent.com/u/10251060?v=4 license: Apache-2.0 repoEtag: '"355905b4366d13f5a453820b5d4126c9b314f831f305d2b0e0b874a3e88656eb"' repoLastModified: Mon, 02 Feb 2026 04:38:45 GMT foundInMaster: true id: 31b717a2084e3d92f6cda2c33484c789 v3_link: true v3_1: true - source: https://openapi.tools/ name: openapi-format category: - Parsers - Converters repository: https://github.com/thim81/openapi-format link: https://www.npmjs.com/package/openapi-format language: Node.js source_description: > A CLI to format an OpenAPI document by ordering fields in a hierarchical order, with the option to filter out flags, tags, methods, operationIDs; including the option to convert an OpenAPI 3.0 document to an OpenAPI version 3.1. v2: false v3: true repositoryMetadata: base64Readme: >- ![openapi-format icon](./assets/openapi-format-logo.svg)

<a href="https://www.npmjs.com/package/openapi-format" alt="Latest Stable Version">![npm](https://img.shields.io/npm/v/openapi-format.svg)</a>
<a href="https://www.npmjs.com/package/openapi-format" alt="Total Downloads">![npm](https://img.shields.io/npm/dw/openapi-format.svg)</a>

# openapi-format

Format an OpenAPI document by ordering, formatting, filtering, and applying overlays to OpenAPI documents.

The openapi-format CLI can sort the OpenAPI fields by ordering them in a hierarchical order, format the casing of the fields and
output cleanly indented JSON or YAML.

Additional features include powerful filtering options based on flags, tags, methods, operationIDs, and even unused components.
To quickly standardize OpenAPI documents there is support for generating the operationIds and apply casing rules for consistency.

The CLI can split large OpenAPI documents into modular, multi-file structures for easier management. 
For upgrades, the openapi-format CLI offers the option to convert an OpenAPI 3.0 or 3.1 document to OpenAPI 3.1 or 3.2.

With the newly added OpenAPI Overlay support, users can overlay changes onto existing OpenAPI documents, to extend and customize the OpenAPI document.

## Table of content
* [Use-cases](#use-cases)
* [Features](#features)
* [Online playground](#online-playground)
* [Installation](#installation)
    + [Local Installation (recommended)](#local-installation-recommended)
    + [Global Installation](#global-installation)
    + [NPX usage](#npx-usage)
* [Command Line Interface](#command-line-interface)
* [OpenAPI format CLI options](#openapi-format-cli-options)
* [OpenAPI sort configuration options](#openapi-sort-configuration-options)
* [OpenAPI formatting configuration options](#openapi-formatting-configuration-options)
* [OpenAPI filter options](#openapi-filter-options)
* [OpenAPI generate elements](#openapi-generate-options)
* [CLI sort usage](#cli-sort-usage)
* [CLI filter usage](#cli-filter-usage)
* [CLI OpenAPI overlay usage](#cli-openapi-overlay-usage)
* [CLI generate usage](#cli-generate-usage)
* [CLI casing usage](#cli-casing-usage)
* [CLI split & bundle usage](#cli-bundle--split-usage)
* [CLI rename usage](#cli-rename-usage)
* [CLI convertTo usage](#cli-convertto-usage)
* [CLI configuration usage](#cli-configuration-usage)
* [Programmatic usage](#programmatic-usage)
* [Credits](#credits)

## Use-cases

**Public documentation:**
An OpenAPI document is a specification that evolves and changes. To facilitate working with the specification and publishing the
document as public documentation, you want to deliver a clean and structured specification. OpenAPI-format helps you to
organize the fields by sorting, formatting and filtering specific elements from the OpenAPI like internal endpoints, beta tags, ...
and even unused schemas, examples, responses, ... with a clean and optimized OpenAPI document as a result.

**Maintenance:**
When working on large OpenAPI documents or with multiple team members, the file can be become messy and difficult to
compare changes. By sorting & formatting from time to time, the fields are all ordered in a structured manner & properly cased, which will help you
to maintain the file with greater ease.

**CI/CD pipelines:**
OpenAPI-format can be useful in CI/CD pipelines, where the OpenAPI is used as the source for other documents like Web documentation,
Postman collections, test suites, ...

## Features

- [x] Order OpenAPI fields in a default order
- [x] Order OpenAPI fields in a custom order
- [x] Order Components elements by alphabet
- [x] Format the casing (camelCase,PascalCase, ...) of component elements
- [x] Filter OpenAPI files based on methods
- [x] Filter OpenAPI files based on flags
- [x] Filter OpenAPI files based on flags values
- [x] Filter OpenAPI files based on tags
- [x] Filter OpenAPI files based on operationID's
- [x] Filter OpenAPI files based on operations definition
- [x] Filter OpenAPI files based on response content-types
- [x] Apply OpenAPI overlay actions
- [x] Strip flags from OpenAPI files
- [x] Strip unused components from OpenAPI files
- [x] Generate OpenAPI elements for consistency
- [x] Bundle local and remote references in the OpenAPI document
- [x] Split the OpenAPI document into a multi-file structure
- [x] Convert OpenAPI 3.0 documents to OpenAPI 3.1 or 3.2
- [x] Convert OpenAPI 3.1 documents to OpenAPI 3.2
- [x] Rename the OpenAPI title
- [x] Support OpenAPI documents in JSON format
- [x] Support OpenAPI documents in YAML format
- [x] Format via CLI
- [x] Format via local or remote config files
- [x] Use as a Module
- [x] Aligned YAML parsing style with Stoplight Studio style
- [x] Support for OpenAPI 3.0
- [x] Support for OpenAPI 3.1
- [x] Support for OpenAPI 3.2
- [x] Online playground (https://openapi-format-playground.vercel.app/)

## Online playground

The [OpenAPI-Format Playground](https://openapi-format-playground.vercel.app/) is a web-based tool for formatting and sorting OpenAPI documents, powered by the openapi-format CLI.

<a href="https://openapi-format-playground.vercel.app/" target="_blank" title="OpenAPI-format Playground" rel="nofollow">
<img src="https://raw.githubusercontent.com/thim81/openapi-format/main/assets/openapi-format-playground.png" alt="OpenAPI-format Playground" width="50%"><br>
<img src="https://raw.githubusercontent.com/thim81/openapi-format/main/assets/openapi-format-playground-diff.png" alt="OpenAPI-format Playground Diff viewer" width="25%">
<img src="https://raw.githubusercontent.com/thim81/openapi-format/main/assets/openapi-format-playground-filter.png" alt="OpenAPI-format Playground Filter UI" width="25%">
</a>

More info about the features and usage, can be found in the [readme](https://github.com/thim81/openapi-format-playground?tab=readme-ov-file#features).

## Installation

### Local Installation (recommended)

While possible to install globally, we recommend that you add the openapi-format CLI to the `node_modules` by using:

```shell
$ npm install --save openapi-format
```

or using yarn...

```shell
$ yarn add openapi-format
```

Note that this will require you to run the openapi-format CLI with `npx openapi-format your-openapi-file.yaml` or, if
you are using an older versions of npm, `./node_modules/.bin/openapi-format your-openapi-file.yaml`.

### Global Installation

```shell
$ npm install -g openapi-format
```

### NPX usage

To execute the CLI without installing it via npm, use the npx method

```shell
$ npx openapi-format your-openapi-file.yaml
```

## Command Line Interface

```
openapi-format.js <input-file> -o [output-file] [options]

Arguments:
  input-file   the OpenAPI document can be a local or remote file in JSON or YAML format
  output-file  the output file is optional and can be either a .json or .yaml file. 

Options:

  --output, -o          Save the formatted OpenAPI file as JSON/YAML            [path]

  --sortFile            The file to specify custom OpenAPI fields ordering      [path]
  --casingFile          The file to specify casing rules                        [path]
  --generateFile        The file to specify generate rules                      [path]  
  --filterFile          The file to specify filter rules                        [path]
  --overlayFile         The file to specify OpenAPI overlay actions             [path]

  --no-sort             Don't sort the OpenAPI file                          [boolean]
  --keepComments        Don't remove the comments from the OpenAPI YAML file [boolean]
  --sortComponentsFile  The file with components to sort alphabetically         [path]
  --sortComponentsProps Sort properties within schema components alphabetically [boolean]

  --no-bundle           Don't bundle the local and remote $ref               [boolean]
  --split               Split OpenAPI document into a multi-file structure   [boolean]

  --rename              Rename the OpenAPI title                              [string]

  --convertTo           convert the OpenAPI document to OpenAPI version 3.1 or 3.2   [string]

  --configFile          The file with the OpenAPI-format CLI options            [path]

  --lineWidth           Max line width of YAML output                         [number]

  --json                Prints the file to stdout as JSON                    [boolean]
  --yaml                Prints the file to stdout as YAML                    [boolean]

  --playground, -p      Open config in online playground
  --help                Show help                                            [boolean]
  --version             Output the version number                            
  --verbose             Output more details of the filter process              [count]
```

## OpenAPI format CLI options

| Parameter             | Alias         | Description                                                                 | Input type   | Default                    | Info     |
|-----------------------|---------------|-----------------------------------------------------------------------------|--------------|----------------------------|----------|
| file                  |               | the OpenAPI document can be a local or remote file in JSON or YAML format   | path to file |                            | required |
| --output              | -o            | save the formatted OpenAPI file as JSON/YAML                                | path to file |                            | optional |
| --sortFile            | -s            | the file to specify custom OpenAPI fields ordering                          | path to file | defaultSort.json           | optional |
| --filterFile          | -f            | the file to specify filter setting                                          | path to file | defaultFilter.json         | optional |
| --casingFile          | -k            | the file to specify casing setting                                          | path to file |                            | optional |
| --generateFile        | -g            | the file to specify generate rules                                          | path to file |                            | optional |
| --overlayFile         | -l            | the file to specify OpenAPI overlay actions                                 | path to file |                            | optional |
| --no-sort             |               | don't sort the OpenAPI file                                                 | boolean      | FALSE                      | optional |
| --keepComments        |               | don't remove the comments from the OpenAPI YAML file                        | boolean      | FALSE                      | optional |
| --sortComponentsFile  |               | sort the items of the components (schemas, parameters, ...) by alphabet     | path to file | defaultSortComponents.json | optional |
| --sortComponentsProps |               | sort properties within schema components alphabetically                     | boolean      | FALSE                      | optional |
| --no-bundle           |               | don't bundle the local and remote $ref in the OpenAPI document              | boolean      | FALSE                      | optional |
| --split               |               | split the OpenAPI document into a multi-file structure                      | boolean      | FALSE                      | optional |
| --rename              |               | rename the OpenAPI title                                                    | string       |                            | optional |
| --convertTo           |               | convert the OpenAPI document to OpenAPI version 3.1 or 3.2                  | string       |                            | optional |
| --configFile          | -c            | the file with all the format config options                                 | path to file |                            | optional |
| --lineWidth           |               | max line width of YAML output                                               | number       | -1 (Infinity)              | optional |
| --json                |               | prints the file to stdout as JSON                                           |              | FALSE                      | optional |
| --yaml                |               | prints the file to stdout as YAML                                           |              | FALSE                      | optional |
| --playground          | -p            | open config in online playground                                            |              |                            | optional |
| --version             |               | output the version number                                                   |              |                            | optional |
| --verbose             | -v, -vv, -vvv | verbosity that can be increased, which will show more output of the process |              |                            | optional |
| --help                | h             | display help for command                                                    |              |                            | optional |

## OpenAPI sort configuration options

The CLI will sort the OpenAPI document in the defined order liked defined per OpenAPI key/field/property/element. The fields that are
not specified will keep their order like it is in the original OpenAPI document, so only defined fields will be
re-ordered.

The default sorting of the different fields based on the defined order (listed in the table below), which is stored in
the [defaultSort.json](https://github.com/thim81/openapi-format/blob/main/defaultSort.json) file.

### OpenAPI sort fields

You can easily modify this by specifying your own ordering per key, which can be passed on to the CLI (see below for an
example on how to do this).

| Key         | Ordered by                                                                                                      | OpenAPI reference                                                         |
| ----------- | ----------------------------------------------------------------------------------------------------------------| ------------------------------------------------------------------------- |
| root        | - openapi<br>\- info<br>\- servers<br>\- paths<br>\- components<br>\- tags<br>\- x-tagGroups<br>\- externalDocs | [openapi-object](https://spec.openapis.org/oas/v3.0.3.html#openapi-object)                  |
| get         | - operationId<br>\- summary<br>\- description<br>\- parameters<br>\- requestBody<br>\- responses                | [operationObject](https://spec.openapis.org/oas/v3.0.3.html#operationObject)                |
| post        | - operationId<br>\- summary<br>\- description<br>\- parameters<br>\- requestBody<br>\- responses                | [operationObject](https://spec.openapis.org/oas/v3.0.3.html#operationObject)                 |
| put         | - operationId<br>\- summary<br>\- description<br>\- parameters<br>\- requestBody<br>\- responses                | [operationObject](https://spec.openapis.org/oas/v3.0.3.html#operationObject)                 |
| patch       | - operationId<br>\- summary<br>\- description<br>\- parameters<br>\- requestBody<br>\- responses                | [operationObject](https://spec.openapis.org/oas/v3.0.3.html#operationObject)                 |
| delete      | - operationId<br>\- summary<br>\- description<br>\- parameters<br>\- requestBody<br>\- responses                | [operationObject](https://spec.openapis.org/oas/v3.0.3.html#operationObject)                 |
| parameters  | - name<br>\- in<br>\- description<br>\- required<br>\- schema                                                   | [parameterObject](https://spec.openapis.org/oas/v3.0.3.html#parameterObject)                 |
| requestBody | - description<br>\- headers<br>\- content<br>\- links                                                           | [request-body-object](https://spec.openapis.org/oas/v3.0.3.html#request-body-object)             |
| responses   | - description<br>\- headers<br>\- content<br>\- links                                                           | [responses-object](https://spec.openapis.org/oas/v3.0.3.html#responses-object)                |
| content     | (By alphabet)                                                                                                   | [responses-object](https://spec.openapis.org/oas/v3.0.3.html#responses-object)                |
| components  | - parameters<br>\- schemas                                                                                      | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object)               |
| schema      | - description<br>\- type<br>\- items<br>\- properties<br>\- format<br>\- example<br>\- default                  | [schemaObject](https://spec.openapis.org/oas/v3.0.3.html#schemaObject)                    |
| schemas     | - description<br>\- type<br>\- items<br>\- properties<br>\- format<br>\- example<br>\- default                  |                                                                           |
| properties  | - description<br>\- type<br>\- items<br>\- format<br>\- example<br>\- default<br>\- enum                        |                                                                           |

Have a look at the folder [yaml-default](test/yaml-default) and compare the "output.yaml" (sorted document) with the "input.yaml" (original document), to see how openapi-format have sorted the OpenAPI document.

### OpenAPI sort Paths

You can change the order of the paths defined in the OpenAPI specification and sort them alphabetically (`path`) or by the first tag of the first method (`tags`).

Options to sort by:

- `original` (default): keep the order as defined in the OpenAPI specification
- `path`: order the paths alphabetically by the path parts
- `tags`: order by the first tag of the first method

| Key         | Options                      | OpenAPI reference                                                      |
|-------------|------------------------------|------------------------------------------------------------------------|
| sortPathsBy | 'original' / 'path' / 'tags' | [paths-object](https://spec.openapis.org/oas/v3.0.3.html#paths-object) |



## OpenAPI filter options

By specifying the desired filter values for the available filter types, the openapi-format CLI will strip out any
matching item from the OpenAPI document. You can combine multiple types to filter out a range of OpenAPI items.

| Type                   | Description                                 | Type    | Examples                                  |
|------------------------|---------------------------------------------|---------|-------------------------------------------|
| methods                | OpenAPI methods.                            | array   | ['get','post','put']                      |
| inverseMethods         | OpenAPI methods that will be kept           | array   | ['get','post','put']                      |
| tags                   | OpenAPI tags                                | array   | ['pet','user']                            |
| inverseTags            | OpenAPI tags that will be kept              | array   | ['pet','user']                            |
| operationIds           | OpenAPI operation ID's                      | array   | ['findPetsByStatus','updatePet']          |
| inverseOperationIds    | OpenAPI operation ID's that will be kept    | array   | ['findPetsByStatus','updatePet']          |
| operations             | OpenAPI operations                          | array   | ['GET::/pets','PUT::/pets']               |
| flags                  | Custom flags                                | array   | ['x-exclude','x-internal']                |
| inverseFlags           | Custom flags that will kept                 | array   | ['x-exclude','x-internal']                |
| flagValues             | Custom flags with a specific value          | array   | ['x-version: 1.0','x-version: 3.0']       |
| inverseFlagValues      | Custom flags with a value that will be kept | array   | ['x-version: 1.0','x-version: 3.0']       |
| responseContent        | Response Content types                      | array   | ['application/json','application/html']   |
| inverseResponseContent | Response Content types that will kept       | array   | ['application/ld+json']                   |
| requestContent         | Request Body Content types                  | array   | ['application/json','application/html']   |
| inverseRequestContent  | Request Body Content types that will kept   | array   | ['application/ld+json']                   |
| unusedComponents       | Unused components                           | array   | ['examples','schemas']                    |
| stripFlags             | Custom flags that will be stripped          | array   | ['x-exclude','x-internal']                |
| preserveEmptyObjects   | Preserve empty object                       | boolean | true OR ['schema']                        |
| textReplace            | Search & replace values to replace          | array   | [{'searchFor':'Pet','replaceWith':'Dog'}] |

Some more details on the available filter types:

### Filter - methods/inverseMethods

=> **methods**: Refers to the [Path Item Object](http://spec.openapis.org/oas/v3.0.3.html#operationObject)

This will remove all fields and attached fields that match the verbs. In the example below, this would mean that
all `get`, `put`, `post` items would be removed from the OpenAPI document.

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
            summary: Finds Pets by status
        put:
            summary: Update an existing pet
```

=> **inverseMethods**: This option does the inverse filtering, by keeping only the verbs defined and remove all other verbs.

### Filter - tags

=> **tags**: Refers to the "tags" field from the "Operation
  Object" https://spec.openapis.org/oas/v3.0.3.html#operationObject

This will remove all fields and attached fields that match the tags. In the example below, this would mean that all
items with the tags `pet` or `user` would be removed from the OpenAPI document.

For example:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        put:
            tags:
                - pet
            summary: Update an existing pet
```

=> **inverseTags**: This option does the inverse filtering, by keeping only the tags defined and remove all other tags, including the operations without a tags.

### Filter - operationIds

=> **operationIds**: Refers to the "operationId" field from the [Operation Object](https://spec.openapis.org/oas/v3.0.3.html#operationObject)

This will remove specific fields and attached fields that match the operation ID's. In the example below, this would
mean that the item with operationID `findPetsByStatus` would be removed from the OpenAPI document.

For example:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
            operationId: findPetsByStatus
```

=> **inverseTags**: This option does the inverse filtering, by keeping only the operationIds defined and remove all other operationIds, including the operations without an operationId.

### Filter - operations

=> **operations**: Refers to a combination of a OpenAPI method & path from the [Path Object](https://spec.openapis.org/oas/v3.0.3.html#paths-object)
& [Path item](https://spec.openapis.org/oas/v3.0.3.html#path-item-object)

This will remove specific path items that match the operation definition `PUT::/pets`. In the example below, this would
mean that the item with the path '/pets' and method 'PUT' would be removed from the OpenAPI document.

For example:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
            summary: Finds Pets by status
        put:
            summary: Update an existing pet
```

An `operationId` is an optional property. To offer support for OpenAPI documents that don't have operationIds, we have
added the `operation` definition which is the unique combination of the OpenAPI method & path, with a `::` separator
symbol.

This will allow filtering for very specific OpenAPI items, without the need of adding operationIds to the OpenAPI
document.

To facilitate managing the filtering, we have included wildcard options for the `operations` option, supporting the
methods & path definitions.

REMARK: Be sure to put quotes around the target definition.

Strict matching example: `"GET::/pets"`
This will target only the "GET" method and the specific path "/pets"

Method wildcard matching example: `"*::/pets"`
This will target all methods ('get', 'query', 'put', 'post', 'delete', 'options', 'head', 'patch', 'trace') and the specific
path "/pets"

Path wildcard matching example: `"GET::/pets/*"`
This will target only the "GET" method and any path matching any folder behind the "/pets", like "/pets/123" and
"/pets/123/buy".

Method & Path wildcard matching example: `"*::/pets/*"`
A combination of wildcards for the method and path parts is even possible.

### Filter - flags/inverseFlags

=> **flags**: Refers to a custom property that can be set on any field in the OpenAPI document.

This will remove all fields and attached fields that match the flags. In the example below, this would mean that all
items with the flag `x-exclude` would be removed from the OpenAPI document.

For example:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
            x-exclude: true
```

=> **inverseFlags**: This option does the inverse filtering, by keeping only the path operations, tags, x-tagGroups that match the flag(s). This is a very aggressive option to keep only the items that are needed but it will keep the components. The components can best be filtered out, by using the "[unusedComponents](https://github.com/thim81/openapi-format#filter---unusedcomponents)" filter option.

### Filter - flagValues/inverseFlagValues

=> **flagValues**: Refers to a flag, custom property which can be set on any field in the OpenAPI document, and the combination with the value for that flag.

This will remove all fields and attached fields that match the flag with the specific value.

A `flagValues` example:

```yaml
flagValues:
    - x-version: 1.0
    - x-version: 3.0
```
In the example below, this would mean that all items with the flag `x-version` that matches `x-version: 1.0` OR `x-version: 3.0` would be removed from the OpenAPI document.

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
            x-version: 1.0
```

The filter option `flagValues` also will remove flags that contain an array of values in the OpenAPI document.

A `flagValues` example:

```yaml
flagValues:
    - x-versions: 1.0
    - x-versions: 2.0
```

In the example below, this would mean that all items with the flag `x-versions`, which is an array, that match `x-version: 1.0` OR `x-version: 3.0` would be removed from the OpenAPI document.

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
            x-versions:
                - 1.0
                - 3.0
                - 5.0
```

Have a look at [flagValues](test/yaml-filter-custom-flagsvalue-value) and [flagValues for array values](test/yaml-filter-custom-flagsvalue-array) for a practical example.

=> **inverseFlagValues**: This option does the inverse filtering, by keeping only the path operations, tags, x-tagGroups that match the flag with the specific value. This is a very aggressive option to keep only the items that are needed but it will keep the components. The components can best be filtered out, by using the "[unusedComponents](https://github.com/thim81/openapi-format#filter---unusedcomponents)" filter option.

### Filter - responseContent/inverseResponseContent

=> **ResponseContent**: Refers to the [Response Object's content](https://spec.openapis.org/oas/v3.0.3.html#response-object)

A responses `content` filter example:

```yaml
responseContent:
  - application/json
```

This will remove all the content that match the media types from the responses. In the example below, this would mean that all `application/json`
content items would be removed from the OpenAPI document

Example before:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
  /pet:
    post:
      tags:
        - pet
      summary: Add a new pet to the store
      description: Add a new pet to the store
      operationId: addPet
      x-visibility: true
      responses:
        '200':
          description: Successful operation
          content:
            application/xml:
              schema:
                $ref: '#/components/schemas/Pet'
            application/json:
              schema:
                $ref: '#/components/schemas/Pet'
        '405':
          description: Invalid input
```

Example after:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
  /pet:
    post:
      tags:
        - pet
      summary: Add a new pet to the store
      description: Add a new pet to the store
      operationId: addPet
      x-visibility: true
      responses:
        '200':
          description: Successful operation
          content:
            application/xml:
              schema:
                $ref: '#/components/schemas/Pet'
        '405':
          description: Invalid input
```

=> **inverseResponseContent**: This option does the inverse filtering, by keeping only the content with media types defined and remove all other content.

### Filter - requestContent/inverseRequestContent

=> **requestContent**: Refers to the [Request Body Object's content](https://spec.openapis.org/oas/v3.0.3.html#request-body-object)

A  request body `content` filter example:

```yaml
requestContent:
  - application/json
```

This will remove all the content that match the media types from the request body. In the example below, this would mean that all `application/json`
content items would be removed from the OpenAPI document.

Example before:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
  /pet:
    post:
      tags:
        - pet
      summary: Add a new pet to the store
      description: Add a new pet to the store
      operationId: addPet
      x-visibility: true
      requestBody: 
          content:
            application/xml:
              schema:
                $ref: '#/components/schemas/Pet'
            application/json:
              schema:
                $ref: '#/components/schemas/Pet'

```

Example after:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
  /pet:
    post:
      tags:
        - pet
      summary: Add a new pet to the store
      description: Add a new pet to the store
      operationId: addPet
      x-visibility: true
      requestBody:
          content:
            application/xml:
              schema:
                $ref: '#/components/schemas/Pet'
```

=> **inverseRequestContent**: This option does the inverse filtering, by keeping only the content with media types defined and remove all other content types from the request body.

### Filter - unusedComponents

=> **unusedComponents**: Refers to a list of [reusable component types]( https://spec.openapis.org/oas/v3.0.3.html#components-object), from which unused items will be removed.

This option allows you to strip the OpenAPI document from any unused items of the targeted `components` types.
Any item in the list of OpenAPI `components` that is not referenced as `$ref`, will get marked and removed from the OpenAPI document.

REMARK: We will recursively strip all unused components, with a maximum depth of 10 times. This means that "nested" components, that become unused, will also get removed

Supported component types that can be marked as "unused":
- schemas
- parameters
- examples
- headers
- requestBodies
- responses
- mediaTypes

### Filter - textReplace

=> **textReplace**: "search & replace" option to replace text in the OpenAPI specification

The `textReplace` provides a "search & replace" method, that will search for a text/word/characters in the OpenAPI description, summary, URL fields and replace it with another text/word/characters.
This is very useful to replace data in the OpenAPI specification.

A `textReplace` example:

```yaml
textReplace:
    - searchFor: 'Pets'
      replaceWith: 'Dogs'
    - searchFor: 'swagger.io'
      replaceWith: 'openapis.org'
```

This will replace all "Pets" with "Dogs" & "swagger.io" with "openapi.org" in the OpenAPI document.

### Filter - stripFlags

=> **stripFlags**: Refers to a list of custom properties that can be set on any field in the OpenAPI document.

The `stripFlags` will remove only the flags, the linked parent and properties will remain. In the example below, this would mean that all
flags `x-exclude` itself would be stripped from the OpenAPI document.

Example before:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
          x-exclude: true
          summary: Finds Pets by status
```

Example after:

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        get:
          summary: Finds Pets by status
```

### Filter - preserveEmptyObjects

=> **preserveEmptyObjects**: Refers to any properties of your OpenAPI document, from which empty object values would be kept.

The `preserveEmptyObjects` will preserve all empty objects if set to `true`.

You can also pass a list of keys from which preserve empty objects. For instance a `['schema']` value would only prevent removal of empty objects having for key `schema`.

REMARK: Please note that openapi-format default behavior is to remove all empty objects from your document, except for items of examples, security, schemas, default, oneOf, allOf.

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        post:
          requestBody:
            description: Create a new pet in the store
            required: true
            content:
              application/json:
                schema: {}
```

Example after (with `preserveEmptyObjects: false`):

```yaml
openapi: 3.0.0
info:
    title: API
    version: 1.0.0
paths:
    /pets:
        post:
          requestBody:
            description: Create a new pet in the store
            required: true
            content:
              application/json: {}
```

## OpenAPI formatting configuration options

Tools like [spectral](https://github.com/stoplightio/spectral) or [speccy](https://speccy.io/), or any of the [linting tools](https://nordicapis.com/8-openapi-linters/), provide a manner to validate & lint OpenAPI specifications to be uniform. The linting tool informs about the incorrect usage of OpenAPI properties & inconsistent field names.
This is very useful and helps to guard the quality of the OpenAPI specification. They inform which fields to correct so that the specification will comply with all the defined linting rules.

The openapi-format CLI formatting option can assist with keeping the field names consistent by automatically changing the casing of the properties/keys/names for the different elements in the OpenAPI document.
The desired casing can be defined per OpenAPI key/element (see list below).
The keys that are not specified will keep their casing like it is in the original OpenAPI document, so only for defined fields, the casing will be changed.

| Key                        | Description                                                                                          | OpenAPI reference                                                         |
|----------------------------|------------------------------------------------------------------------------------------------------| ------------------------------------------------------------------------- |
| operationId                | Changes operation ID's that are part of the Operations Object                                        | [operation-object](https://spec.openapis.org/oas/v3.0.3.html#operation-object)|
| properties                 | Changes property keys of the schemas of the inline response/requestBody & components                 | [schemaObject](https://spec.openapis.org/oas/v3.0.3.html#schemaObject) |
| parametersCookie           | Changes the cookie name of the parameters inline & models in the components                          | [parameter-object](https://spec.openapis.org/oas/v3.0.3.html#parameter-object) |
| parametersPath             | Changes the path name of the parameters inline & models in the components                            | [parameter-object](https://spec.openapis.org/oas/v3.0.3.html#parameter-object) |
| parametersHeader           | Changes the header name of the parameters inline & models in the components                          | [parameter-object](https://spec.openapis.org/oas/v3.0.3.html#parameter-object) |
| parametersQuery            | Changes the query name of the parameters inline & models in the components                           | [parameter-object](https://spec.openapis.org/oas/v3.0.3.html#parameter-object) |
| componentsParametersCookie | Changes the key of the cookie models in the components parameters sections & "$ref" links            | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsParametersPath   | Changes the key of the path models in the components parameters sections & "$ref" links              | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsParametersQuery  | Changes the key of the query models in the components parameters sections & "$ref" links             | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsParametersHeader | Changes the key of the header models in the components parameters sections & "$ref" links            | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsSchemas          | Changes the key of the schema models in the components sections & "$ref" links                       | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsExamples         | Changes the key of the example models in the components sections & "$ref" links                      | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsHeaders          | Changes the key of the header models in the components sections & "$ref" links                       | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsResponses        | Changes the key of the response models in the components sections & "$ref" links                     | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsRequestBodies    | Changes the key of the request body models in the components sections & "$ref" links                 | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |
| componentsSecuritySchemes  | Changes the key of the security schemes in the components sections & "$ref" links                    | [components-object](https://spec.openapis.org/oas/v3.0.3.html#components-object) |

### Casing options

| Casing type      | Casing alias | Description                                       | Example          |
| -----------------| ------------ | ------------------------------------------------- | --------------- |
| 🐪 camelCase     | camelCase    | converts a strings to `camelCase`                 | `openapiFormat`  |
| 👨‍🏫 PascalCase    | PascalCase   | converts a strings to `PascalCase`                | `OpenapiFormat`  |
| 🥙 kebab-case    | kebabCase    | converts a strings to `kebab-case`                | `openapi-format` |
| 🚂 Train-Case    | TrainCase    | converts a strings to `Train-Case`                | `Openapi-Format` |
| 🐍 snake_case    | snakeCase    | converts a strings to `snake_case`                | `openapi_format` |
| 🕊 Ada_Case      | AdaCase      | converts a strings to `Ada_Case`                  | `Openapi_Format` |
| 📣 CONSTANT_CASE | constantCase | converts a strings to `CONSTANT_CASE`             | `OPENAPI_FORMAT` |
| 👔 COBOL-CASE    | cobolCase    | converts a strings to `COBOL-CASE`                | `OPENAPI-FORMAT` |
| 📍 Dot.notation  | dotNotation  | converts a strings to `Dot.notation`              | `openapi.format` |
| 🛰 Space case    | spaceCase    | converts a strings to `Space case` (with spaces)  | `openapi format` |
| 🏛 Capital Case  | capitalCase  | converts a strings to `Capital Case` (with spaces)| `Openapi Format` |
| 🔡 lower case    | lowerCase    | converts a strings to `lower case` (with spaces)  | `openapi format` |
| 🔠 UPPER CASE    | upperCase    | converts a strings to `UPPER CASE` (with spaces)  | `OPENAPI FORMAT` |

> REMARK: All special characters are stripped during conversion, except for the `@` and `$`, since they can be part of the query strings.

The casing options are provided by the nano NPM [case-anything](https://github.com/mesqueeb/case-anything) package.

### Format casing - operationId

=> **operationId**: Refers to the `operationId` properties in the OpenAPI document.

Formatting casing example:

```yaml
operationId: kebab-case
```

Example before:

```yaml
openapi: 3.0.3
paths:
    /pets:
        get:
          operationId: getPets
```

openapi-format will format the "getPets" from the original camelcase to kebab-case.

Example after:

```yaml
openapi: 3.0.3
paths:
    /pets:
        get:
          operationId: get-pets
```

### Format casing - model & schema properties

=> **properties**: Refers to all the schema properties, that are defined inline in the paths request bodies & responses and the models in the components section of the OpenAPI document.

Formatting casing example:

```yaml
properties: snake_case
```

Example before:

```yaml
openapi: 3.0.3
components:
    schemas:
        UserModel:
            type: object
            properties:
                id:
                    type: integer
                    example: 10
                emailAddress:
                    type: string
                    example: john@doe.com
                firstName:
                    type: string
                    example: John
```

The CLI will format all the properties like: "id", "username", "firstName" from the original camelcase to snake_case.

Example after:

```yaml
openapi: 3.0.3
components:
    schemas:
        UserModel:
            type: object
            properties:
                id:
                    type: integer
                    example: 10
                email_address:
                    type: string
                    example: john@doe.com
                first_name:
                    type: string
                    example: John
```

### Format casing - component keys

=> **componentsSchemas / componentsExamples / componentsParametersCookie / componentsParametersHeader / componentsParametersQuery / componentsParametersQuery / componentsParametersPath / componentsHeaders / componentsResponses / componentsRequestBodies / componentsSecuritySchemes**: Refers to all the model objects that are defined in the components section of the OpenAPI document.

Formatting casing example:

```yaml
componentsSchemas: PascalCase
```

Example before:

```yaml
openapi: 3.0.3
paths:
    /orders:
        get:
            responses:
                content:
                    application/json:
                        schema:
                            $ref: '#/components/schemas/order-model'
components:
    schemas:
        userModel:
            type: object
        order-model:
            type: object
        pet_model:
            type: object
```

openapi-format will format all the component keys like: "userModel", "order-model", "pet_model" to PascalCase, including formatting all the "$ref" used in the OpenAPI document.

Example after:

```yaml
openapi: 3.0.3
paths:
    /orders:
        get:
            responses:
                content:
                    application/json:
                        schema:
                            $ref: '#/components/schemas/OrderModel'
components:
    schemas:
        UserModel:
            type: object
        OrderModel:
            type: object
        PetModel:
            type: object
```

### Format casing - parameter names

=> **componentsParametersCookie / componentsParametersPath / componentsParametersQuery / componentsParametersHeader**: Refers to "name" in the Parameters types: Path, Query or Header, which can be defined inline in the Path or as a reference in the components of the OpenAPI document.

Formatting casing example:

```yaml
componentsParametersPath: kebab-case
```

Example before:

```yaml
openapi: 3.0.3
paths:
    '/pet/{petId}':
        get:
            parameters:
                - name: petId
                  in: path
                  description: ID of pet to return
                - $ref: '#/components/parameters/LimitParam'
components:
    parameters:
        LimitParam:
            name: limitParam
            in: query
            description: max records to return
```

The CLI will format the "name" of the parameters: Path, Query or Header like: "petId", "limitParam" to kebab-case in the OpenAPI document.

Example after:

```yaml
openapi: 3.0.3
paths:
    '/pet/{petId}':
        get:
            parameters:
                - name: pet-id
                  in: path
                  description: ID of pet to return
               - $ref: '#/components/parameters/LimitParam'
components:
    parameters:
        LimitParam:
            name: limit-param
            in: query
            description: max records to return
```

### OpenAPI Generate Options

The OpenAPI formatting tool allows you to generate various elements such as `operationId`, and more using customizable templates. These templates enable dynamic generation of missing or incomplete values in your OpenAPI specification, ensuring consistency and adherence to your conventions.

Options for generating elements:

- `operationIdTemplate`: Generate the `operationId` using placeholders like `<method>`, `<pathPart2>`, etc.
- `overwriteExisting`: Set to `true` or `false` to control whether existing values should be overwritten (default: `false`).

| Key                 | Options                        | OpenAPI Reference                                                              |
|---------------------|--------------------------------|--------------------------------------------------------------------------------|
| operationIdTemplate | Customizable with placeholders | [operation-object](https://spec.openapis.org/oas/v3.0.3.html#operation-object) |
| overwriteExisting   | `true` / `false`               | N/A                                                                            |

See [CLI generate usage](#cli-generate-usage) for an example and the available template options.

## CLI sort usage

- Format a spec with the default sorting and saves it as a new JSON file

```shell
$ openapi-format openapi.json -o openapi-formatted.json
```

- Format a remote spec with the default sorting and saves it as a new JSON file

```shell
$ openapi-format https://raw.githubusercontent.com/OAI/OpenAPI-Specification/main/examples/v3.0/api-with-examples.json -o openapi-formatted.json
```

- Format an OpenAPI JSON document with the default sorting and saves it as a new YAML file

```shell
$ openapi-format openapi.json -o openapi.yaml
```

- Format an OpenAPI document using a configuration file containing all the options that would otherwise be passed via the CLI. 

```shell
$ openapi-format openapi.yaml --configFile openapi-format-options.json
```

The formatting will happen based on all the options set in the `openapi-format-options.json` file. All the
available [OpenAPI format options](https://github.com/thim81/openapi-format#openapi-format-options) can be used in the config file.

- Format an OpenAPI document with the default sorting and output it as JSON to STDOUT

```shell
$ openapi-format openapi.json --json
```

- Format an OpenAPI document with the default sorting and output it as YAML to STDOUT

```shell
$ openapi-format openapi.json --yaml
```

- Format an OpenAPI JSON document with the default sorting and save it as YAML

```shell
$ openapi-format openapi.json -o openapi.yaml
```

- Format an OpenAPI document but skip the sorting and save it as a new JSON file

```shell
$ openapi-format openapi.json -o openapi-formatted.json --no-sort
```

This should keep the OpenAPI fields in the same order. This can be needed, when you only want to do a filtering or
rename action.

- Convert the OpenAPI 3.0 document to OpenAPI 3.1 but skip the sorting and save it as a new YAML file

```shell
$ openapi-format openapi.yaml -o openapi-3.1.yaml --no-sort --convertTo "3.1"
```

This will convert the OpenAPI 3.0 document into version 3.1 of OpenAPI, without any ordering or filtering.
During the conversion, openapi-format will transform all OpenAPI 3.0 properties into the OpenAPI 3.1 properties, as described in the [migration guide from
Phil Sturgeon](https://www.openapis.org/blog/2021/02/16/migrating-from-openapi-3-0-to-3-1-0).

- Format an OpenAPI document, including sorting all elements in the components section

```shell
$ openapi-format openapi.json -o openapi-formatted.json --sortComponentsFile ./test/json-sort-components/customSortComponents.json
```

This will sort all elements in the components (components/schemas, components/parameters, components/headers,
components/requestBodies, components/responses, ...) section alphabetically. The file specified by `--sortComponentsFile` should contain an array of component types to sort.

Example of a customSortComponents.json file:
```json
["schemas", "parameters", "headers", "requestBodies", "responses", "securitySchemes"]
```

**before**
```yaml
components:
  schemas:
    Order:
      type: object
      # properties...
    Customer:
      type: object
      # properties...
    Address:
      type: object
      # properties...
  securitySchemes:
    petstore_auth:
      type: oauth2
      # configuration...
    api_key:
      type: apiKey
      # configuration...
```

**after**
```yaml
components:
  schemas:
    Address:
      type: object
      # properties...
    Customer:
      type: object
      # properties...
    Order:
      type: object
      # properties...
  securitySchemes:
    api_key:
      type: apiKey
      # configuration...
    petstore_auth:
      type: oauth2
      # configuration...
```

- Format an OpenAPI document, including sorting properties within schema components alphabetically

```shell
$ openapi-format openapi.json -o openapi-formatted.json --sortComponentsProps
```

This will sort all properties within schema components alphabetically. For example:

**before**
```yaml
components:
  schemas:
    UserDto:
      type: object
      properties:
        lastName:
          type: string
        firstName:
          type: string
```

**after**
```yaml
components:
  schemas:
    UserDto:
      type: object
      properties:
        firstName:
          type: string
        lastName:
          type: string
```

## CLI filter usage

- Format an OpenAPI document by filtering fields, default sorting and saves it as a new file

When you want to strip certain methods ,tags, operationIds, operations, flags you can pass a `filterFile` which contains the
specific values for the methods ,tags, operationIds, operations, flags.

This can be useful to combine with the sorting, to end up with an order and filtered OpenAPI document.

example:

```shell
$ openapi-format openapi.json -o openapi-formatted.json --filterFile customFilter.yaml
```

where the `customFilter.yaml` would contain a combination of all the elements you want to filter out.

```yaml
flags:
    - x-visibility
flagValues: [ ]
tags: [ ]
operationIds:
    - addPet
    - findPetsByStatus
```

## CLI OpenAPI Overlay Usage

The OpenAPI Overlay functionality allows users to apply actions such as updates and removals to an OpenAPI Specification (OAS). This feature is useful for dynamically modifying OAS documents during development, testing, or publishing workflows.

### What is an OpenAPI Overlay?
An [OpenAPI Overlay](https://spec.openapis.org/overlay/v1.0.0.html) is a specification that defines a structured set of actions to be applied to an existing OpenAPI document. It enables:

- Updating existing fields, such as descriptions, parameters, or endpoints.
- Adding new fields or objects to the OpenAPI document.
- Removing fields or objects that are no longer relevant.

An overlay document follows the structure below:

```
overlay: 1.0.0
info:
  title: Example Overlay
  version: 1.0.0
actions:
  - target: "$"   // JSONPath definition of the targetted element of the document
    update: // The action to be applied: update or remove
      info:
        description: "Updated description for the OpenAPI specification."
  - target: "$.paths['/example']"
    update:
      get:
        description: "Updated GET description for /example endpoint."
  - target: "$.paths['/example'].get.parameters"
    remove: true   # Example of removing an element
```

Fore more information about the OpenAPI Overlay options, see [OpenAPI Overlay Specification 1.0.0](https://www.openapis.org/blog/2024/10/22/announcing-overlay-specification) 

Use the `--overlayFile` option to specify the overlay file and apply it to your OpenAPI document.

example:
```shell
$ openapi-format openapi.yaml --overlayFile overlay.yaml -o openapi-updated.yaml
```

You can also let the overlay declare the base OpenAPI document using the top-level `extends` property. When `extends` is present, the input file argument becomes optional:

```yaml
overlay: 1.0.0
info:
  title: Overlay for Tic Tac Toe
  version: 1.0.0
extends: 'https://raw.githubusercontent.com/OAI/learn.openapis.org/refs/heads/main/examples/v3.1/tictactoe.yaml'
# actions: [...]  # optional
```

CLI usage with `extends` (no input file argument):

```shell
$ openapi-format --overlayFile overlay.yaml -o openapi-updated.yaml
```

Notes:
- `extends` supports both local paths and remote `http(s)` URLs.
- Local relative paths are resolved relative to the overlay file’s location.

## CLI generate usage

- Generate OpenAPI elements and saves it as a new file

The OpenAPI formatting tool allows you to generate OpenAPI elements, such as `operationId`, `summary`, and more, based on configurable templates. The generated elements will be saved to the output OpenAPI file.

You can also combine generation with filtering and sorting to customize the output.

example:

```shell
$ openapi-format openapi.json -o openapi-formatted.json --generateFile customGenerate.yaml
```

where the `customGenerate.yaml` would contain a combination of all the elements you to generate.

```yaml
operationIdTemplate: "<method>_<pathPart2>"
overwriteExisting: false
```

**Template Options:**
In the customGenerate.yaml, you can define templates for various OpenAPI properties using dynamic placeholders. These placeholders will be replaced by actual values from the OpenAPI operations. Below is a list of available placeholders and what they represent:

- `<operationId>` : The operationId of the OpenAPI operation. Example: leadsAll
- `<method>` : The HTTP method of the OpenAPI operation. Example: GET
- `<path>` : The path of the OpenAPI operation. Example: /crm/leads
- `<pathRef>` : The path reference of the OpenAPI operation. Example: GET::/crm/leads
- `<tag>` : The first tag name of the OpenAPI operation. Example: Leads
- `<tag1>` : The first tag name of the OpenAPI operation. Example: Leads
- `<tag2>` : The second tag name of the OpenAPI operation, if more than one tag is defined. Example: CRM
- `<tagn>` : The nth tag name of the OpenAPI operation if more than one tag is defined.
- `<pathPart1>` : The first part of the path of the OpenAPI operation. Example: crm
- `<pathPart2>` : The second part of the path of the OpenAPI operation. Example: leads
- `<pathPartn>` : The nth part of the path of the OpenAPI operation.

You can also include static text in the templates, which will be merged with the dynamic placeholders. For example:

```yaml
operationIdTemplate: "Prefix_<method>_<pathPart2>_Handler"
```

**Configuration Options:**

- `operationIdTemplate`: Template for generating `operationId`. Use dynamic placeholders like <method> and <pathPart2>.
- `overwriteExisting`: Set to `true` or `false` to control whether existing values should be overwritten. Default: `false`.

## CLI casing usage

- Generate OpenAPI elements and saves it as a new file

The OpenAPI formatting tool allows you to enforce consistent casing styles across various OpenAPI elements, such as `operationId`, `summary`, `parameters`, and more. The specified casing preferences will be applied to the relevant elements and saved to the output OpenAPI file.

example:

```shell
$ openapi-format openapi.json -o openapi-formatted.json --casingFile customCasing.yaml
```

where the `customCasing.yaml` would contain a casing preferences for the specified of the elements.

In this example, the customCasing.yaml file would contain the desired casing preferences for specific OpenAPI elements.

```yaml
operationId: snake_case
properties: camelCase

parametersQuery: kebab-case
parametersHeader: kebab-case
parametersPath: snake_case

componentsExamples: PascalCase
componentsSchemas: camelCase
componentsHeaders: kebab-case
componentsResponses: snake_case
componentsRequestBodies: snake_case
componentsSecuritySchemes: PascalCase

componentsParametersQuery: snake_case
componentsParametersHeader: kebab-case
componentsParametersPath: camelCase
```

**Casing Options:**
In the customCasing.yaml, you can define the casing style for various OpenAPI properties, allowing you to customize the appearance of your document consistently.

- `operationId`: Defines the casing for operation IDs. Example: snake_case, PascalCase, or camelCase.
- `properties`: Sets the casing for properties within components. Example: camelCase.
- `parametersQuery`, `parametersHeader`, `parametersPath`: Define different casing styles for parameters based on their location (query, header, path). Example: snake_case, kebab-case.
- and many more

See [OpenAPI formatting configuration options](#openapi-formatting-configuration-options) for the full list of casing options

## CLI Bundle & Split usage

- **Bundling**: Create a self-contained OpenAPI file that can be used for documentation generation or API validation tools that don't support external references.

- **Splitting**: Generate a modular OpenAPI structure during development or testing, making it easier to manage changes to individual paths or components without altering the entire document.

### Splitting the OpenAPI Document

The `--split` option splits the OpenAPI document into a modular multi-file structure. This structure makes it easier to manage larger specifications by separating paths, components (schemas, paramaters, ...) into individual files.

Example: Splitting the Document

```shell
$ openapi-format openapi.json -o ./openapi-split/openapi.yaml --split
```

This command will take the openapi.json and split it into multiple files, stored under the ./openapi-split/ directory. 

The resulting structure might look like this:

```bash
./openapi-split/
├── openapi.yaml
├── paths/
│   ├── /pets.yaml
│   └── /pets/{petId}.yaml
├── components/
├── schemas/
│   ├── Pet.yaml
│   └── Error.yaml
├── parameters/
│   └── petId.yaml
```

The main openapi.yaml file will contain references to these newly created files using $ref, making the structure modular and easier to navigate.

###  Bundling the OpenAPI Document

The `--no-bundle` option allows you to control whether local and remote $ref references are bundled into the final document. 

By default, all $ref references are dereferenced, resulting in a single, self-contained OpenAPI file. However, in some cases, you might prefer to keep the $ref references intact, especially if you're working with external references or want to maintain a modular structure.

Example: Default Bundling
```shell
$ openapi-format input.json -o bundled-openapi.json
```

This example produces a fully dereferenced dereferenced-openapi.json, where all local and remote $ref references are resolved into the file.
This is the default behaviour.

Example: No Bundling

```shell
$ openapi-format openapi.json -o openapi.json --no-bundle
```

In this case, the resulting bundled-openapi.json will preserve all $ref references as they are in the original document.

## CLI rename usage

- Format an OpenAPI document by changing the title and saves it as a new JSON file

During CI/CD pipelines, you might want to create different results of the OpenAPI document. Having the option to rename
them might make it easier to work with the results, so that is why we provide this command option.

```shell
$ openapi-format openapi.json -o openapi.json --rename "OpenAPI Petstore - OpenAPI 3.0"
```

which results in

**before**

```json
{
    "openapi": "3.0.2",
    "info": {
        "title": "Petstore - OpenAPI 3.0",
```

**after**

```json
{
    "openapi": "3.0.2",
    "info": {
        "title": "OpenAPI Petstore - OpenAPI 3.0",
```

## CLI convertTo usage

- Format & convert the OpenAPI document to OpenAPI version 3.1 or 3.2

openapi-format can help you to upgrade your current OpenAPI 3.0.x document to OpenAPI 3.1 or 3.2.

```shell
$ openapi-format openapi.json -o openapi-3.1.json --convertTo "3.1"
$ openapi-format openapi.json -o openapi-3.2.json --convertTo "3.2"
```

Using `3.1` results in all the changes described in the [migration guide from Phil Sturgeon](https://www.openapis.org/blog/2021/02/16/migrating-from-openapi-3-0-to-3-1-0), while the `3.2` target aligns with the new capabilities highlighted in [OpenAPI 3.2 is here](https://quobix.com/articles/openapi-3.2/).

**before**

```json
{
    "openapi": "3.0.2",
    "info": {
        "title": "Petstore - OpenAPI",
```

**after**

```json
{
    "openapi": "3.1.0",
    "info": {
        "title": "OpenAPI Petstore - OpenAPI",
```

When converting to 3.2 the `openapi` field will be set to `3.2.0`, preparing the document for features like hierarchical tags, the `QUERY` HTTP method, and reusable media types introduced in the 3.2 release.

## CLI configuration usage

The openapi-format CLI supports bundling all options in a single configuration file. This can simplify management, especially for CI/CD pipelines where configurations are stored in version control systems.

### Using the --configFile option

You can pass a configuration file containing all the options that would otherwise be passed via the CLI. This helps in centralizing the configuration for your OpenAPI formatting operations.

Example:

```shell
$ openapi-format openapi.json --configFile openapi-format-options.json
```
The formatting will happen based on all the options set in the `openapi-format-options.json` file. All the
available [OpenAPI format options](https://github.com/thim81/openapi-format#openapi-format-options) can be used in the config file.

The openapi-format-options.json file might look like this:

```json
{
  "sort": true,
  "casingSet": {
    "operationId": "camelCase",
    "properties": "snake_case"
  },
  "filterSet": {
    "tags": ["internal", "beta"]
  },
  "generateSet": {
    "operationIdTemplate": "<method>_<pathPart2>_Handler"
  }
}
```

Alternatively, you can reference external files for each setting using the corresponding File properties.

```json
{
    "sortFile": "customSort.json",
    "casingFile": "casing-rules.json",
    "filterFile": "filter-rules.json",
    "generateFile": "generate-rules.json"
}
```

In this case, the settings will be loaded from the external files, and they override any inline configurations.

**Define sort, filter, casing, generate options**

You can either pass the settings inline or reference an external file using the appropriate File property:

- **sortSet** / **sortFile**: Sort the fields in the OpenAPI document based on the order defined in the sort settings.

  - Inline: Pass the sort order directly using sortSet in the config file.
  - File: Use sortFile to specify the path to a local or remote JSON/YAML file containing custom sorting rules.

- **casingSet** / **casingFile**: Define the casing convention for operationId, parameters, properties, etc.

  - Inline:
    ```json
    "casingSet": {
      "operationId": "camelCase",
      "properties": "PascalCase"
    }
    ```

  - File: Use casingFile to specify the path to a local or remote JSON/YAML file containing casing rules.

- **filterSet** / **filterFile**: Filter out specific tags, paths, or components from the OpenAPI document.

  - Inline:
    ```json
    "filterSet": {
      "tags": ["internal", "beta"]
    }
    ```

  - File: Use filterFile to specify the path to a local or remote JSON/YAML file containing filter rules.

- **generateSet** / **generateFile**: Automatically generate operationId, summary, and other elements based on predefined templates.

  - Inline:
    ```json
    "generateSet": {
      "operationIdTemplate": "<method>_<pathPart2>_Handler"
    }
    ```

  - File: Use generateFile to specify the path to a local or remote JSON/YAML file containing generate rules.


### Using .openapiformatrc

In addition to specifying a configuration file using `--configFile`, openapi-format also supports automatically loading a configuration file named `.openapiformatrc` from the current directory. If this file is present, it will be used as the configuration source, and individual options passed via the CLI will override the settings from this file.

Example of a .openapiformatrc file:

```json
{
  "output": "openapi-final.yaml",  
  "sort": true,
  "filterSet": {
      "tags": ["internal", "beta"]
  }
}
```

## Programmatic usage

The CLI is the primary interface, but all formatting utilities are exposed as module functions as well:

```js
const {
  openapiSort,
  getDefaultSortSet,
  getDefaultSortComponentsSet
} = require('openapi-format');

const document = /* your OpenAPI document */;
const sorted = await openapiSort(document, {
  sortSet: await getDefaultSortSet(),
  sortComponentsSet: await getDefaultSortComponentsSet()
});
```

Both `sortSet` and `sortComponentsSet` are optional when you call `openapiSort`. When omitted, openapi-format automatically applies the built-in defaults (the same ones the CLI uses). The helper functions `getDefaultSortSet` and `getDefaultSortComponentsSet` are provided in case you want to start from the defaults and tweak them before invoking `openapiSort` in your own scripts.

### Sorting with minimal setup

```js
const {
  parseFile,
  stringify,
  writeFile,
  openapiSort
} = require('openapi-format');

const input = await parseFile('spec.json'); // local path or remote URL
const {data} = await openapiSort(input, {sort: true});

const output = await stringify(data, {format: 'json'});
await writeFile('spec.sorted.json', output, {format: 'json'});
```

### Sorting with custom tweaks

```js
const {
  parseFile,
  stringify,
  writeFile,
  openapiSort,
  getDefaultSortSet
} = require('openapi-format');

const document = await parseFile('spec.json');

const sortSet = await getDefaultSortSet();
sortSet.get = ['summary', 'description', 'responses']; // override GET priority

const {data} = await openapiSort(document, {
  sort: true,
  sortSet,
  sortComponentsSet: ['schemas', 'responses']
});

const output = await stringify(data, {format: 'json'});
await writeFile('spec.sorted.json', output, {format: 'json'});
```

### Mixing in other utilities

```js
const {openapiFilter, openapiGenerate} = require('openapi-format');

let draft = await parseFile('spec.json');
draft = (await openapiFilter(draft, {
  filterSet: {tags: ['public']}
})).data;

draft = (await openapiGenerate(draft, {
  generateSet: {
    operationIdTemplate: '<method>_<pathPart1>_<pathPart2>'
  }
})).data;
```

### Using the file helpers for YAML/JSON OpenAPI documents

The module also exports the same helpers the CLI relies on for smart parsing/writing (large-number handling, YAML comments, remote loading, etc.) of OpenAPI documents:

```js
const {
  parseFile,
  stringify,
  writeFile,
  openapiSort
} = require('openapi-format');

const input = await parseFile('openapi.yaml'); // local path or remote URL
const {data} = await openapiSort(input, {sort: true});

const output = await stringify(data, {format: 'yaml', lineWidth: -1});
await writeFile('openapi.sorted.yaml', output, {format: 'yaml'});
```

## AsyncAPI documents

For handling AsyncAPI documents, we have created a separate
package [asyncapi-format](https://github.com/thim81/asyncapi-format) to allow customisation specific for AsyncAPI
use-cases.

## Stoplight Studio

We have adopted the YAML parsing style from [Stoplight Studio](https://stoplight.io/studio/), by leveraging
the [@stoplight/yaml](https://www.npmjs.com/package/@stoplight/yaml) package for handling the parsing of OpenAPI YAML
files.

By using the Stoplight YAML parsing, the results will be slightly different from when using a normal YAML parsing
library, like [js-to-yaml](https://www.npmjs.com/package/js-yaml). We appreciate the Stoplight Studio tool, since it is
an excellent GUI for working with OpenAPI documents for non-OpenAPI experts who will be contributing changes. By
adopting the Stoplight Studio YAML parsing, the potential risk of merge conflicts will be lowered, which is the main
reason why we opted for using the @stoplight/yaml package.

## Credits

This package is inspired by
the [@microsoft.azure/format-spec](https://www.npmjs.com/package/@microsoft.azure/format-spec) from [@fearthecowboy](https://github.com/fearthecowboy). The
original code was not available on GitHub, with the last update being 3 years ago, so to improve support and extend it we
tried to reproduce the original functionality.

The filter capabilities from `openapi-format` are inspired by the work from [@MikeRalphson](https://github.com/mikeralphson) on
the [openapi-filter](https://github.com/Mermade/openapi-filter) package.

The casing options available in `openapi-format` are powered by the excellent [case-anything](https://github.com/mesqueeb/case-anything) nano package from Luca Ban ([@mesqueeb](https://github.com/mesqueeb)).

<a href="https://www.jetbrains.com/" target="_blank">
<img src="https://resources.jetbrains.com/storage/products/company/brand/logos/jetbrains.png" alt="JetBrains logo." width="200px">
</a>

Special thanks to [JetBrains](https://www.jetbrains.com/) for their continuous sponsorship of this project over the last 3 years, and for their support to open-source software (OSS) initiatives.
 readmeEtag: '"e0db0a2877f9376b042fb85613d0f3f56fce33fb"' readmeLastModified: Mon, 22 Dec 2025 11:27:55 GMT repositoryId: 345077537 description: Format an OpenAPI document by ordering, formatting and filtering fields. created: '2021-03-06T11:31:52Z' updated: '2026-01-22T10:41:37Z' language: JavaScript archived: false stars: 152 watchers: 3 forks: 22 owner: thim81 logo: https://avatars.githubusercontent.com/u/952446?v=4 license: MIT repoEtag: '"52003f8b5e97f2f09a3c5c868bc991e0bc06b4d0996b5bdc1c9bcd8c3c6a6ef9"' repoLastModified: Thu, 22 Jan 2026 10:41:37 GMT foundInMaster: true id: 788dd625de099d0d3403fe30067f3743 v3_1: true - source: https://openapi.tools/ name: super-linter category: - Description Validators - Server Implementations repository: https://github.com/github/super-linter language: - CLI - Docker source_description: >- GitHub Action to lint repositories as part of CI/CD. Implements the latest version of Spectral. v2: true v3: true repositoryMetadata: base64Readme: >- ClRoaXMgcmVwb3NpdG9yeSBpcyBhIGZvcmsgb2YgdGhlIHVwc3RyZWFtIGh0dHBzOi8vZ2l0aHViLmNvbS9zdXBlci1saW50ZXIvc3VwZXItbGludGVyLiBQbGVhc2UgZGlyZWN0IGFsbCBwdWxsIHJlcXVlc3RzIGFuZCBpc3N1ZXMgdGhlcmUgYXMgdGhpcyByZXBvc2l0b3J5IHdpbGwgbm90IGFsd2F5cyBiZSB1cCB0byBkYXRlLgo= readmeEtag: '"7939d5a68eedbc0f5647e25027156d13fe577f47"' readmeLastModified: Thu, 29 Aug 2024 18:27:11 GMT repositoryId: 633129333 description: Combination of multiple linters to install as a GitHub Action created: '2023-04-26T21:02:26Z' updated: '2026-01-05T07:14:56Z' language: Shell archived: false stars: 142 watchers: 4 forks: 26 owner: github logo: https://avatars.githubusercontent.com/u/9919?v=4 license: MIT repoEtag: '"8a2efc17f55a7923c163fcbe9e8ee3ed78c55692c36ac4c08d6159d687a7c182"' repoLastModified: Mon, 05 Jan 2026 07:14:56 GMT foundInMaster: true id: ec6a7a1e4a69a9beeb95bdebb88cb36d - source: https://openapi.tools/ name: SpringFox category: Server language: - Java - Kotlin - Groovy - or Ruby link: https://springfox.io repository: https://github.com/springfox/springfox source_description: Automated JSON API documentation for APIs built with Spring and SpringBoot v1: true v2: true v3: true repositoryMetadata: base64Readme: >- # Springfox

[![Join the chat at https://gitter.im/springfox/springfox](https://badges.gitter.im/Join%20Chat.svg)](https://gitter.im/springfox/springfox?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
[![FOSSA Status](https://app.fossa.io/api/projects/git%2Bgithub.com%2Fspringfox%2Fspringfox.svg?type=shield)](https://app.fossa.io/projects/git%2Bgithub.com%2Fspringfox%2Fspringfox?ref=badge_shield)
[ ![Download](https://api.bintray.com/packages/springfox/maven-repo/springfox/images/download.svg) ](https://bintray.com/springfox/maven-repo/springfox/_latestVersion) 
[![Project Stats](https://www.openhub.net/p/springfox/widgets/project_thin_badge.gif)](https://www.openhub.net/p/springfox)

| Build Status  | Coverage   | Code Analysis |
|---|---|---|
|[![Circle CI](https://circleci.com/gh/springfox/springfox/tree/master.svg?style=svg)](https://circleci.com/gh/springfox/springfox/tree/master)|[![codecov](https://codecov.io/gh/springfox/springfox/branch/master/graph/badge.svg)](https://codecov.io/gh/springfox/springfox) |[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=alert_status)](https://sonarcloud.io/dashboard?id=springfox_springfox)|

| Sonar Cloud |
|------------ |
|[![Security Rating](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=security_rating)](https://sonarcloud.io/dashboard?id=springfox_springfox)|
|[![Vulnerabilities](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=vulnerabilities)](https://sonarcloud.io/dashboard?id=springfox_springfox)|
|[![Maintainability Rating](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=sqale_rating)](https://sonarcloud.io/dashboard?id=springfox_springfox)|
|[![Reliability Rating](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=reliability_rating)](https://sonarcloud.io/dashboard?id=springfox_springfox)|
|[![Technical Debt](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=sqale_index)](https://sonarcloud.io/dashboard?id=springfox_springfox)|
|[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=springfox_springfox&metric=coverage)](https://sonarcloud.io/dashboard?id=springfox_springfox)|

### About
For more information on this project visit the [Springfox Website](http://springfox.io) or
 [http://springfox.github.io/springfox/](http://springfox.github.io/springfox/)

### Useful links
- [Reference Documentation](http://springfox.io)
- [Examples repository](https://github.com/springfox/springfox-demos)
- [Contribution Guidelines](https://github.com/springfox/springfox/wiki/Contribution-guidelines)
- [Core contributors](http://springfox.github.io/springfox/contributors.html)
- [Development and contribution guidelines](https://github.com/martypitt/swagger-springmvc/wiki/Development)
- [Change log](docs/release-notes.md)
- [Development Environment](http://springfox.github.io/springfox/docs/current/#development-environment)
- [Release Instructions](http://springfox.github.io/springfox/docs/current/#releasing)

### Getting Started

#### For new projects
For Maven
```xml 
<dependency>
    <groupId>io.springfox</groupId>
    <artifactId>springfox-boot-starter</artifactId>
    <version>3.0.0</version>
</dependency>

```

For Gradle
```gradle 
  implementation "io.springfox:springfox-boot-starter:<version>"
```
#### Migrating from earlier snapshot 
#### Spring Boot Applications
NOTE: Would love feedback to make this better
1. Remove explicit dependencies on `springfox-swagger2`
2. Remove any `@EnableSwagger2...` annotations
3. Add the `springfox-boot-starter` dependency
4. Springfox 3.x removes dependencies on guava and other 3rd party libraries (not zero dep yet! depends on spring plugin
and open api libraries for annotations and models) so if you used guava predicates/functions those will need to 
transition to java 8 function interfaces.

#### Migrating from existing 2.x version
#### Spring Boot Applications
NOTE: Would love feedback to make this better
1. Remove explicit dependencies on `springfox-swagger2`
2. Remove the `@EnableSwagger2` annotations
3. Add the `springfox-boot-starter` dependency
4. Springfox 3.x removes dependencies on guava and other 3rd party libraries (not zero dep yet! depends on spring plugin
and open api libraries for annotations and models) so if you used guava predicates/functions those will need to 
transition to java 8 function interfaces 
5. If you are using WebMvc but you don't use the [`@EnableWebMvc`](https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/servlet/config/annotation/EnableWebMvc.html) annotation yet, add this annotation.

#### Regular spring mvc  
NOTE: Would love feedback to make this experience better
1. Remove explicit dependencies on `springfox-swagger2`
2. Add `@EnableOpenApi` for open API (and `@EnableSwagger2WebMvc` or `@EnableSwagger2WebFlux` for older versions)  
3. Added the `springfox-oas` library 
4. Springfox 3.x removes dependencies on guava and other 3rd party libraries (not zero dep yet! depends on spring plugin
and open api libraries for annotations and models) so if you used guava predicates/functions those will need to 
transition to java 8 function interfaces 


License
-------

Copyright 2015 Marty Pitt - [@martypitt](https://github.com/martypitt), Dilip Krishnan - [@dilipkrish](https://github.com/dilipkrish),
Adrian Kelly -  [@adrianbk](https://github.com/adrianbk),

Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at [apache.org/licenses/LICENSE-2.0](http://www.apache.org/licenses/LICENSE-2.0)

Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.


## License
[![FOSSA Status](https://app.fossa.io/api/projects/git%2Bgithub.com%2Fspringfox%2Fspringfox.svg?type=large)](https://app.fossa.io/projects/git%2Bgithub.com%2Fspringfox%2Fspringfox?ref=badge_large)
 readmeEtag: '"fd6ee0fdca5a5307c2655e0354220fb22686b380"' readmeLastModified: Wed, 14 Oct 2020 01:49:23 GMT repositoryId: 4238977 description: Automated JSON API documentation for API's built with Spring created: '2012-05-06T06:38:57Z' updated: '2026-02-04T01:24:12Z' language: Java archived: false stars: 5942 watchers: 258 forks: 1518 owner: springfox logo: https://avatars.githubusercontent.com/u/11529357?v=4 license: Apache-2.0 repoEtag: '"b7b624202da9e5570ef99dfc3f9cc547a193672857d3913b5244d926cb540795"' repoLastModified: Wed, 04 Feb 2026 01:24:12 GMT foundInMaster: true id: 2addde4c497bf582f20025ccac2cda86 - source: - https://openapi.tools/ - openapi3 tags name: php-openapi-faker language: PHP repository: https://github.com/canvural/php-openapi-faker source_description: >- Library to generate fake data for OpenAPI 3.x requests, responses and schemas. v2: false v3: true repositoryMetadata: base64Readme: >- IyBwaHAtb3BlbmFwaS1mYWtlcgoKIVtUZXN0c10oaHR0cHM6Ly9naXRodWIuY29tL2NhbnZ1cmFsL3BocC1vcGVuYXBpLWZha2VyL3dvcmtmbG93cy9UZXN0cy9iYWRnZS5zdmcpClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9jYW52dXJhbC9waHAtb3BlbmFwaS1mYWtlci9icmFuY2gvbWFzdGVyL2dyYXBoL2JhZGdlLnN2ZyldKGh0dHBzOi8vY29kZWNvdi5pby9naC9jYW52dXJhbC9waHAtb3BlbmFwaS1mYWtlcikKWyFbSW5mZWN0aW9uIE1TSV0oaHR0cHM6Ly9iYWRnZS5zdHJ5a2VyLW11dGF0b3IuaW8vZ2l0aHViLmNvbS9jYW52dXJhbC9waHAtb3BlbmFwaS1mYWtlci9tYXN0ZXIpXShodHRwczovL2luZmVjdGlvbi5naXRodWIuaW8pClshW1BIUFN0YW5dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUEhQU3Rhbi1MZXZlbCUyME1heC1icmlnaHRncmVlbi5zdmc/c3R5bGU9ZmxhdCZsb2dvPXBocCldKGh0dHBzOi8vcGhwc3Rhbi5vcmcpCgpMaWJyYXJ5IHRvIGdlbmVyYXRlIGZha2UgZGF0YSBmb3IgeW91ciBPcGVuQVBJIHJlcXVlc3RzLCByZXNwb25zZXMgYW5kIHNjaGVtYXMuCgpgYGBwaHAKJGZha2VyID0gXFZ1cmFsXE9wZW5BUElGYWtlclxPcGVuQVBJRmFrZXI6OmNyZWF0ZUZyb21Kc29uKCR5b3VyU2NoZW1hQXNKc29uKTsKJGZha2VEYXRhID0gJGZha2VyLT5tb2NrUmVzcG9uc2UoJy90b2RvcycsJ0dFVCcpOwpgYGAKCiMjIEluc3RhbGxhdGlvbgoKWW91IGNhbiBpbnN0YWxsIHRoZSBwYWNrYWdlIHZpYSBjb21wb3NlcjoKCmBgYGJhc2gKY29tcG9zZXIgcmVxdWlyZSAtLWRldiBjYW52dXJhbC9waHAtb3BlbmFwaS1mYWtlcgpgYGAKCiMjIFVzYWdlCgpGaXJzdCB5b3UgbmVlZCB0byBjcmVhdGUgYW4gaW5zdGFuY2Ugb2YgYE9wZW5BUElGYWtlcmAgd2l0aCB5b3VyIHNjaGVtYSB0aGF0IHlvdSB3YW50IHRvIGZha2UgZGF0YSBmcm9tLiBZb3UgY2FuIHVzZSBgY3JlYXRlRnJvbUpzb25gLCBgY3JlYXRlRnJvbVlhbWxgIG9yIGBjcmVhdGVGcm9tU2NoZW1hYCB0byBjcmVhdGUgYW4gaW5zdGFuY2Ugb2YgYE9wZW5BUElGYWtlcmAuCmBgYHBocAokZmFrZXIgPSBcVnVyYWxcT3BlbkFQSUZha2VyXE9wZW5BUElGYWtlcjo6Y3JlYXRlRnJvbUpzb24oJHlvdXJTY2hlbWFBc0pzb24pOwpgYGAKClRoZW4geW91IGNhbiB1c2UgYG1vY2tSZXNwb25zZWAsIGBtb2NrUmVzcG9uc2VGb3JFeGFtcGxlYCwgYG1vY2tSZXF1ZXN0YCwgYG1vY2tSZXF1ZXN0Rm9yRXhhbXBsZWAsIGBtb2NrQ29tcG9uZW50U2NoZW1hYCBhbmQgYG1vY2tDb21wb25lbnRTY2hlbWFGb3JFeGFtcGxlYCBtZXRob2RzIG9uIGl0IHRvIGdlbmVyYXRlIGZha2UgZGF0YSBmb3IgeW91ciByZXF1ZXN0cywgcmVzcG9uc2VzIGFuZCBzY2hlbWFzLiBMaWtlIHNvOgoKYGBgcGhwCiRmYWtlRGF0YSA9ICRmYWtlci0+bW9ja1Jlc3BvbnNlKCcvdG9kb3MnLCdHRVQnKTsKYGBgCgojIyMgT3B0aW9ucwoKVGhlcmUgYXJlIHNvbWUgb3B0aW9ucyB5b3UgY2FuIHVzZSB0byBtb2RpZnkgc29tZSBiZWhhdmlvdXIuIFlvdSBjYW4gcGFzcyBvcHRpb25zIGFzIGFuIGFzc29jaWF0aXZlIGFycmF5IHRvIGBzZXRPcHRpb25zYCBtZXRob2QgaW4gYE9wZW5BUElGYWtlcmAuIEZvciBleGFtcGxlOgoKYGBgcGhwCiRmYWtlciA9IFxWdXJhbFxPcGVuQVBJRmFrZXJcT3BlbkFQSUZha2VyOjpjcmVhdGVGcm9tSnNvbigkeW91clNjaGVtYUFzSnNvbikKICAgIC0+c2V0T3B0aW9ucyhbJ21pbkl0ZW1zJyA9PiA1XSk7CmBgYAoKQmVsb3cgeW91IGNhbiBmaW5kIGV4cGxhbmF0aW9uIGZvciBlYWNoIG9wdGlvbi4KCiMjIyMgYG1pbkl0ZW1zYApPdmVycmlkZXMgYG1pbkl0ZW1zYCBwcm9wZXJ0eSBpZiBpdCdzIGxlc3MgdGhhbiB0aGlzIHZhbHVlLgoKIyMjIyBgbWF4SXRlbXNgCk92ZXJyaWRlIGBtYXhJdGVtc2AgaWYgaXQncyBncmVhdGVyIHRoYW4gdGhpcyB2YWx1ZS4KCiMjIyBgYWx3YXlzRmFrZU9wdGlvbmFsc2AKCklmIGVuYWJsZWQsIGV2ZXJ5IHByb3BlcnR5IG9yIGl0ZW0gd2lsbCBiZSBnZW5lcmF0ZWQgcmVnYXJkbGVzcyBpZiB0aGV5IGFyZSByZXF1aXJlZCBvciBub3QuICoqRGVmYXVsdCoqOiBgZmFsc2VgCgojIyMgYHN0cmF0ZWd5YApCeSBkZWZhdWx0LCBgT3BlbkFQSUZha2VyYCB1c2VzIGEgZHluYW1pYyBnZW5lcmF0aW9uIHN0cmF0ZWd5LiBZb3UgY2FuIGVuYWJsZSB0aGUgc3RhdGljIGV4YW1wbGVzIGdlbmVyYXRpb24gYnkgdXNpbmcgdGhlIGBzdGF0aWNgIHN0cmF0ZWd5LgoqKkRlZmF1bHQqKjogYGR5bmFtaWNgCgojIyBDaGFuZ2Vsb2cKClBsZWFzZSBzZWUgW0NIQU5HRUxPR10oQ0hBTkdFTE9HLm1kKSBmb3IgbW9yZSBpbmZvcm1hdGlvbiBvbiB3aGF0IGhhcyBjaGFuZ2VkIHJlY2VudGx5LgoKIyMgQ29udHJpYnV0aW5nCgpQbGVhc2Ugc2VlIFtDT05UUklCVVRJTkddKENPTlRSSUJVVElORy5tZCkgZm9yIGRldGFpbHMuCgojIyBDcmVkaXRzCgpQZW9wbGU6Ci0gW0NhbiBWdXJhbF0oaHR0cHM6Ly9naXRodWIuY29tL2NhbnZ1cmFsKQotIFtBbGwgQ29udHJpYnV0b3JzXSguLi8uLi9jb250cmlidXRvcnMpCgpSZXNvdXJjZXM6Ci0gW2NlYmUvcGhwLW9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9jZWJlL3BocC1vcGVuYXBpKQotIFtsZWFndWUvb3BlbmFwaS1wc3I3LXZhbGlkYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL3RoZXBocGxlYWd1ZS9vcGVuYXBpLXBzcjctdmFsaWRhdG9yKQoKIyMgTGljZW5zZQoKVGhlIE1JVCBMaWNlbnNlIChNSVQpLiBQbGVhc2Ugc2VlIFtMaWNlbnNlIEZpbGVdKExJQ0VOU0UubWQpIGZvciBtb3JlIGluZm9ybWF0aW9uLgo= readmeEtag: '"f6d07df0eec133990026e724552ba3c346d964ad"' readmeLastModified: Sun, 07 May 2023 08:42:23 GMT repositoryId: 271818243 description: Library to generate fake data for OpenAPI request/response/schemas created: '2020-06-12T14:31:49Z' updated: '2025-06-26T19:24:11Z' language: PHP archived: false stars: 93 watchers: 3 forks: 2 owner: canvural logo: https://avatars.githubusercontent.com/u/1574232?v=4 license: MIT repoEtag: '"d7e1ebff87f3287fca3412efa8d25191d0c33ffcd48301c6bdf596ba5ceab63d"' repoLastModified: Thu, 26 Jun 2025 19:24:11 GMT foundInMaster: true id: 4fdfc8e7383c27745a0bff820fab179e category: Parsers - source: https://openapi.tools/ name: OWASP ZAP category: - Security - SDK language: Java repository: https://github.com/zaproxy/zaproxy source_description: >- OWASP ZAP is a free and open source web security tool that can be used manually or completely automated. It supports importing OpenAPI v2 and v3 definitions to allow an API to be thoroughly security tested. link: https://www.zaproxy.org/ v2: true v3: true repositoryMetadata: base64Readme: >- IyBbIVtdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS93aWtpL3phcHJveHkvemFwcm94eS9pbWFnZXMvemFwLWJ5LWNoZWNrbWFyeC5wbmcpXShodHRwczovL3d3dy56YXByb3h5Lm9yZykKWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLUFwYWNoZSUyMDItNEVCMUJBLnN2ZyldKGh0dHBzOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAuaHRtbCkKWyFbR2l0SHViIHJlbGVhc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3JlbGVhc2UvemFwcm94eS96YXByb3h5LnN2ZyldKGh0dHBzOi8vd3d3LnphcHJveHkub3JnL2Rvd25sb2FkLykKWyFbSmF2YSBDSV0oaHR0cHM6Ly9naXRodWIuY29tL3phcHJveHkvemFwcm94eS9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3phcHJveHkvemFwcm94eS9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWwpClshW0NJSSBCZXN0IFByYWN0aWNlc10oaHR0cHM6Ly9iZXN0cHJhY3RpY2VzLmNvcmVpbmZyYXN0cnVjdHVyZS5vcmcvcHJvamVjdHMvMjQvYmFkZ2UpXShodHRwczovL2Jlc3RwcmFjdGljZXMuY29yZWluZnJhc3RydWN0dXJlLm9yZy9wcm9qZWN0cy8yNCkKWyFbR2l0aHViIFJlbGVhc2VzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9kb3dubG9hZHMvemFwcm94eS96YXByb3h5L2xhdGVzdC90b3RhbC5zdmc/bWF4QWdlPTI1OTIwMDApXShodHRwczovL3phcGJvdC5naXRodWIuaW8vemFwLW1nbXQtc2NyaXB0cy9kb3dubG9hZHMuaHRtbCkKWyFbamF2YWRvY10oaHR0cHM6Ly9qYXZhZG9jLmlvL2JhZGdlMi9vcmcuemFwcm94eS96YXAvamF2YWRvYy5zdmcpXShodHRwczovL2phdmFkb2MuaW8vZG9jL29yZy56YXByb3h5L3phcCkKWyFbQ29kZVFMXShodHRwczovL2dpdGh1Yi5jb20vemFwcm94eS96YXByb3h5L2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVxbC55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3phcHJveHkvemFwcm94eS9hY3Rpb25zL3dvcmtmbG93cy9jb2RlcWwueW1sKQpbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD16YXByb3h5X3phcHJveHkmbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9emFwcm94eV96YXByb3h5KQpbIVtPcGVuIFNvdXJjZSBIZWxwZXJzXShodHRwczovL3d3dy5jb2RldHJpYWdlLmNvbS96YXByb3h5L3phcHJveHkvYmFkZ2VzL3VzZXJzLnN2ZyldKGh0dHBzOi8vd3d3LmNvZGV0cmlhZ2UuY29tL3phcHJveHkvemFwcm94eSkKWyFbVHdpdHRlciBGb2xsb3ddKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdHdpdHRlci9mb2xsb3cvemFwcm94eS5zdmc/c3R5bGU9c29jaWFsJmxhYmVsPUZvbGxvdyZtYXhBZ2U9MjU5MjAwMCldKGh0dHBzOi8vdHdpdHRlci5jb20vemFwcm94eSkKCiFbSW50ZWdyYXRpb24gVGVzdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS96YXByb3h5L3phcHJveHkvYWN0aW9ucy93b3JrZmxvd3MvcnVuLWludGVncmF0aW9uLXRlc3RzLnltbC9iYWRnZS5zdmcpCiFbRG9ja2VyIExpdmUgUmVsZWFzZV0oaHR0cHM6Ly9naXRodWIuY29tL3phcHJveHkvemFwcm94eS9hY3Rpb25zL3dvcmtmbG93cy9yZWxlYXNlLWxpdmUtZG9ja2VyLnltbC9iYWRnZS5zdmcpCgpUaGUgWmVkIEF0dGFjayBQcm94eSAoWkFQKSBieSBDaGVja21hcnggaXMgdGhlIHdvcmxk4oCZcyBtb3N0IHdpZGVseSB1c2VkIHdlYiBhcHAgc2Nhbm5lci4gCkZyZWUgYW5kIG9wZW4gc291cmNlLiBBIGNvbW11bml0eSBiYXNlZCBHaXRIdWIgVG9wIDEwMDAgcHJvamVjdCB0aGF0IGFueW9uZSBjYW4gY29udHJpYnV0ZSB0by4KCkl0IGNhbiBoZWxwIHlvdSBhdXRvbWF0aWNhbGx5IGZpbmQgc2VjdXJpdHkgdnVsbmVyYWJpbGl0aWVzIGluIHlvdXIgd2ViIGFwcGxpY2F0aW9ucyB3aGlsZSB5b3UgYXJlIGRldmVsb3BpbmcgYW5kIHRlc3RpbmcgeW91ciBhcHBsaWNhdGlvbnMuIApJdCdzIGFsc28gYSBncmVhdCB0b29sIGZvciBleHBlcmllbmNlZCBwZW50ZXN0ZXJzIHRvIHVzZSBmb3IgbWFudWFsIHNlY3VyaXR5IHRlc3RpbmcuCgpbIVtdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS93aWtpL3phcHJveHkvemFwcm94eS9pbWFnZXMvWkFQLURvd25sb2FkLnBuZyldKGh0dHBzOi8vd3d3LnphcHJveHkub3JnL2Rvd25sb2FkLykKCkZvciBtb3JlIGRldGFpbHMgYWJvdXQgWkFQIHNlZSB0aGUgd2Vic2l0ZTogW3phcHJveHkub3JnXShodHRwczovL3d3dy56YXByb3h5Lm9yZy8pCgpbIVtdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS93aWtpL3phcHJveHkvemFwcm94eS9pbWFnZXMvemFwLXdlYnNpdGUucG5nKV0oaHR0cHM6Ly93d3cuemFwcm94eS5vcmcvKQo= readmeEtag: '"e5efcf16bf926466a0dbb8e29f146884df227f05"' readmeLastModified: Tue, 24 Sep 2024 12:03:48 GMT repositoryId: 36817565 description: The ZAP by Checkmarx Core project created: '2015-06-03T16:55:01Z' updated: '2026-02-06T04:24:36Z' language: Java archived: false stars: 14711 watchers: 398 forks: 2502 owner: zaproxy logo: https://avatars.githubusercontent.com/u/6716868?v=4 license: Apache-2.0 repoEtag: '"81e1b53233cc43e7f2cb32f437bde8e1e9d720e5c174149ab15f5d11c6abb46f"' repoLastModified: Fri, 06 Feb 2026 04:24:36 GMT foundInMaster: true id: 9eedc1be44837a65f912bcf562844f39 - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/oxlip-lang/oal v3: true id: fd805efb152540f08abf74a1601a5da0 repositoryMetadata: base64Readme: >- IVtCdWlsZF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvYWN0aW9ucy93b3JrZmxvdy9zdGF0dXMvb3hsaXAtbGFuZy9vYWwvY2kueW1sP2JyYW5jaD1tYXN0ZXIpClshW0xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGljZW5zZS1BcGFjaGVfMi4wLWJsdWUuc3ZnKV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9BcGFjaGUtMi4wKQoKIyBUaGUgT3hsaXAgQVBJIExhbmd1YWdlCk94bGlwIGlzIGEgaGlnaC1sZXZlbCBmdW5jdGlvbmFsIHByb2dyYW1taW5nIGxhbmd1YWdlIGZvciBkZXNpZ25pbmcgW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy93aGF0LWlzLW9wZW5hcGkpIGRlZmluaXRpb25zLgpBcyBhbiBbSW50ZXJmYWNlIERlc2NyaXB0aW9uIExhbmd1YWdlXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9JbnRlcmZhY2VfZGVzY3JpcHRpb25fbGFuZ3VhZ2UpLCBpdCBpcyBub3QgZ2VuZXJhbCBwdXJwb3NlLgpUaGUgbW90aXZhdGlvbiBpcyB0byBhbGxldmlhdGUgdGhlIHBhaW4gb2YgbWFuYWdpbmcgT3BlbkFQSSBpbiBKU09OIG9yIFlBTUwgYnkgaGFuZCBhbmQgYXQgc2NhbGUuCk94bGlwIGRlZmluZXMgYWxnZWJyYWljIGFic3RyYWN0aW9ucyBvdmVyIFtSRVNUXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9SZXByZXNlbnRhdGlvbmFsX3N0YXRlX3RyYW5zZmVyKSBjb25jZXB0cywgbm90IHRvbyBkaXNzaW1pbGFyIHRvIFtTYXNzL1NDU1Mgb3ZlciBDU1NdKGh0dHBzOi8vc2Fzcy1sYW5nLmNvbS8pLgoKVGhlcmUgYXJlIHByb3MgYW5kIGNvbnMgdG8gYm90aCBfQVBJLWRlc2lnbi1maXJzdF8gYW5kIE9wZW5BUEkgZ2VuZXJhdGVkIGZyb20gaW1wbGVtZW50YXRpb24uCkFzIE9wZW5BUEkgaXMgYmV0dGVyIHByb2R1Y2VkIG9yIGNvbnN1bWVkIGJ5IG1hY2hpbmVzIHJhdGhlciB0aGFuIGh1bWFucywgT3hsaXAgdHJpZXMgdG8gaGVscCBfQVBJLWRlc2lnbi1maXJzdF8gdGVhbXMgd2l0aCBiZXR0ZXIgdG9vbGluZy4KCiMjIFtEb2N1bWVudGF0aW9uXShodHRwczovL3d3dy5veGxpcC1sYW5nLm9yZy8pCgojIyBbUGxheWdyb3VuZF0oaHR0cHM6Ly9veGxpcC1sYW5nLmdpdGh1Yi5pby9veGxpcC1wbGF5Z3JvdW5kKQoKIyMgSW5zdGFsbGF0aW9uClRoaXMgc3RlcCByZXF1aXJlcyBhIFtsb2NhbCBSdXN0IGFuZCBDYXJnbyBpbnN0YWxsYXRpb25dKGh0dHBzOi8vZG9jLnJ1c3QtbGFuZy5vcmcvY2FyZ28vZ2V0dGluZy1zdGFydGVkL2luc3RhbGxhdGlvbi5odG1sKS4KCmBgYAptYWtlIGluc3RhbGwKYGBgCk9wdGlvbmFsOiBhIFtWU0NvZGUgbGFuZ3VhZ2UgZXh0ZW5zaW9uXShodHRwczovL2dpdGh1Yi5jb20vb3hsaXAtbGFuZy9vYWwtdnNjb2RlKSBpcyBhdmFpbGFibGUgZm9yIHN5bnRheCBoaWdobGlnaHRpbmcgYW5kIElERSBjYXBhYmlsaXRpZXMuCgojIyBVc2FnZQpgYGAKICAgIG9hbC1jbGkgW09QVElPTlNdCgpPUFRJT05TOgogICAgLWIsIC0tYmFzZSA8QkFTRT4gICAgICAgIFRoZSByZWxhdGl2ZSBVUkwgdG8gYSBiYXNlIE9wZW5BUEkgZGVzY3JpcHRpb24KICAgIC1jLCAtLWNvbmYgPENPTkZJRz4gICAgICBUaGUgcGF0aCB0byB0aGUgY29uZmlndXJhdGlvbiBmaWxlCiAgICAtaCwgLS1oZWxwICAgICAgICAgICAgICAgUHJpbnQgaGVscCBpbmZvcm1hdGlvbgogICAgLW0sIC0tbWFpbiA8TUFJTj4gICAgICAgIFRoZSByZWxhdGl2ZSBVUkwgdG8gdGhlIG1haW4gcHJvZ3JhbQogICAgLXQsIC0tdGFyZ2V0IDxUQVJHRVQ+ICAgIFRoZSByZWxhdGl2ZSBVUkwgdG8gdGhlIHRhcmdldCBPcGVuQVBJIGRlc2NyaXB0aW9uCmBgYAoKIyMjIENvbXBpbGluZyB0aGUgZXhhbXBsZSBwcm9ncmFtCmBgYApvYWwtY2xpIC0tY29uZiBleGFtcGxlcy9vYWwudG9tbApgYGAKCiMjIEV4cGVyaW1lbnRhbDogV2ViQXNzZW1ibHkgc3VwcG9ydApSZWxlYXNlIHRvIFdlYkFzc2VtYmx5IHJlcXVpcmVzIHRoZSBpbnN0YWxsYXRpb24gb2YgW2B3YXNtLXBhY2tgXShodHRwczovL3J1c3R3YXNtLmdpdGh1Yi5pby93YXNtLXBhY2svaW5zdGFsbGVyLykuCgpgYGAKbWFrZSB3YXNtCmBgYA== readmeEtag: '"e4083bdd30e501dd47c14c5a41b6cc7bea2b8257"' readmeLastModified: Sun, 28 Jan 2024 10:24:11 GMT repositoryId: 425256008 description: A high-level language for OpenAPI created: '2021-11-06T13:50:41Z' updated: '2025-10-03T13:24:58Z' language: Rust archived: false stars: 37 watchers: 1 forks: 3 owner: oxlip-lang logo: https://avatars.githubusercontent.com/u/143259218?v=4 license: Apache-2.0 repoEtag: '"bf0822f680d2c83c3c5fc90f30d520ced7d892a44bd13ffa5248d2989fba0767"' repoLastModified: Fri, 03 Oct 2025 13:24:58 GMT category: - DSL - Parsers foundInMaster: true name: Oxlip API Language link: https://www.oxlip-lang.org/ language: OAL source_description: > Oxlip is a high-level functional programming language for designing OpenAPI definitions. It defines algebraic abstractions over REST concepts to alleviate the pain of managing OpenAPI in JSON or YAML by hand and at scale. v2: false v3_1: false - source: https://openapi.tools/ name: OpenAPI3 Fuzzer category: - Security - Parsers link: https://pypi.org/project/openapi3-fuzzer/ repository: https://github.com/vwt-digital/openapi3-fuzzer/tree/master language: Python source_description: >- Simple fuzzer for OpenAPI 3 specification based APIs. Verifies responses and sends various attack patterns. v3: true repositoryMetadata: base64Readme: >- IyBTaW1wbGUgZnV6emVyIGZvciBPcGVuQVBJIDMgc3BlY2lmaWNhdGlvbiBiYXNlZCBBUElzCgojIyBXaGF0IGRvZXMgdGhpcyBmdXp6ZXIgZG8/CgoxLiBTZW5kcyB2YXJpb3VzIGF0dGFjayBwYXR0ZXJucyB0byBhbGwgdGhlIHBhdGhzIGRlZmluZWQgaW4gYW4gT3BlbkFQSSAzIGRlZmluaXRpb24gZmlsZSwgdXNpbmcgdGhlIE9BUzMgZGVmaW5pdGlvbiB0byBjcmVhdGUgcG9wdWxhdGUgcmVxdWVzdHMuCjIuIFZlcmlmaWVzIGlmIHRoZSByZXNwb25zZXMgbWF0Y2hlcyB0aG9zZSBkZWZpbmVkIGluIHRoZSBPQVMzIGRlZmluaXRpb24gZmlsZSwgY29tcGxhaW5zIGFuZCBleGl0KDIpIGlmIGl0IGRvZXNuJ3QuCjMuIENvbXBsYWlucyBsb3VkbHkgYW5kIGV4aXQoMSkgaWYgYSBwYXRoIHJldHVybnMgYW4gaW50ZXJuYWwgc2VydmVyIGVycm9yIChzdGF0dXMgY29kZSA1MDAgYW5kIGhpZ2hlcikKCiMjIFdoeSBkb2VzIHRoaXMgT3BlbkFQSSBmdXp6ZXIgZXhpc3Q/CgpUbyBtYWtlIGl0IGVhc3kgdG8gaW50ZWdyYXRlIGFuIE9wZW5BUEkgMyBmdXp6ZXIgaW4gYW4gZXhpc3RpbmcgQVBJLgoKIyMgSG93IGRvIEkgdXNlIHRoaXM/CgoxLiBJbnN0YWxsIHRoZSBmdXp6ZXIgdXNpbmcgaXRzIFtwaXAgcGFja2FnZV0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L29wZW5hcGkzLWZ1enplci8pCjIuIEFkZCBhdCBsZWFzdCB0aGUgZm9sbG93aW5nIHBhY2thZ2VzIHRvIHJlcXVpcmVtZW50cy10ZXN0LnR4dDoKYGBgYHB5dGhvbgpjb3ZlcmFnZT09NS4wLjMKb3BlbmFwaTMtZnV6emVyCmFkYWw9PTEuMi4yCkZsYXNrLVRlc3Rpbmc9PTAuNy4xCmBgYGAKMy4gR2VuZXJhdGUgT3BlbkFQSSAoaHR0cHM6Ly9naXRodWIuY29tL09wZW5BUElUb29scy9vcGVuYXBpLWdlbmVyYXRvcikKNC4gQ3JlYXRlIGEgdGVzdF9mdXp6aW5nIGZpbGUgaW4gdGhlIHRlc3QgbG9jYXRpb24gdXNpbmcgdGhlIHRlbXBsYXRlIGJlbG93OgpgYGBgcHl0aG9uCmltcG9ydCBhZGFsCgppbXBvcnQgY29uZmlnCmZyb20gb3BlbmFwaTNfZnV6emVyIGltcG9ydCBGdXp6SXQKZnJvbSBvcGVuYXBpX3NlcnZlci50ZXN0IGltcG9ydCBCYXNlVGVzdENhc2UKCgpkZWYgZ2V0X3Rva2VuKCk6CiAgICAiIiIKICAgIENyZWF0ZSBhIHRva2VuIGZvciB0ZXN0aW5nCiAgICA6cmV0dXJuOgogICAgIiIiCiAgICBvYXV0aF9leHBlY3RlZF9hdXRoZW50aWNhdG9yID0gYXV0aGVudGljYXRvcnVyaQogICAgY2xpZW50X2lkID0gYXBwaWQKICAgIGNsaWVudF9zZWNyZXQgPSBzZWNyZXQKICAgIHJlc291cmNlID0gcmVzb3VyY2UvYXVkaWVuY2UKCiAgICAjIGdldCBhbiBBenVyZSBhY2Nlc3MgdG9rZW4gdXNpbmcgdGhlIGFkYWwgbGlicmFyeQogICAgY29udGV4dCA9IGFkYWwuQXV0aGVudGljYXRpb25Db250ZXh0KG9hdXRoX2V4cGVjdGVkX2F1dGhlbnRpY2F0b3IpCiAgICB0b2tlbl9yZXNwb25zZSA9IGNvbnRleHQuYWNxdWlyZV90b2tlbl93aXRoX2NsaWVudF9jcmVkZW50aWFscygKICAgICAgICByZXNvdXJjZSwgY2xpZW50X2lkLCBjbGllbnRfc2VjcmV0KQoKICAgIGFjY2Vzc190b2tlbiA9IHRva2VuX3Jlc3BvbnNlLmdldCgnYWNjZXNzVG9rZW4nKQogICAgcmV0dXJuIGFjY2Vzc190b2tlbgoKCmNsYXNzIFRlc3R2QVBJKEJhc2VUZXN0Q2FzZSk6CgogICAgZGVmIHRlc3RfZnV6emluZyhzZWxmKToKICAgICAgICBGdXp6SXQoIm9wZW5hcGkueWFtbCIsIGdldF90b2tlbigpLCBzZWxmKQoKYGBgYAo1LiBSdW4gdXNpbmcgb3VyIFt1bml0dGVzdCBjb250YWluZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS92d3QtZGlnaXRhbC9jbG91ZGJ1aWxkZXItdW5pdHRlc3QpIG9yIHZpYSB0aGUgW1B5dGhvbiBVbml0dGVzdCBGcmFtZXdvcmtdKGh0dHBzOi8vZG9jcy5weXRob24ub3JnLzMvbGlicmFyeS91bml0dGVzdC5odG1sKQoKIyMgV2hhdCBPQVMzIGl0ZW1zIGFyZSBzdXBwb3J0ZWQ/CgpCYXNlZCBvbiBbT3BlbkFQSSBzcGVjaWZpY2F0aW9uIDMuMC4yXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMi5tZCk6CgpPcGVyYXRpb24gfCBTdXBwb3J0ZWQKLS0tLS0tLS0tLXwtLS0tLS0tLS0tCkdFVCAgICAgICB8IFllcwpQT1NUICAgICAgfCBZZXMKUFVUICAgICAgIHwgWWVzCkRFTEVURSAgICB8IFllcwpIRUFEICAgICAgfCBZZXMKT1BUSU9OUyAgIHwgbm8KUEFUQ0ggICAgIHwgbm8KVFJBQ0UgICAgIHwgbm8KClBhcmFtZXRlciBpbiB8IFN1cHBvcnRlZAotLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0KcGF0aCAgICAgICAgIHwgWWVzCnF1ZXJ5ICAgICAgICB8IG5vCmhlYWRlciAgICAgICB8IG5vCmNvb2tpZSAgICAgICB8IG5vCgpQcm9wZXJ0eSB0eXBlcyB8IFN1cHBvcnRlZAotLS0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLQpzdHJpbmcgICAgICAgICB8IFllcwppbnRlZ2VyICAgICAgICB8IFllcwpudW1iZXIgICAgICAgICB8IFllcwphcnJheSAgICAgICAgICB8IFllcwpub25lICAgICAgICAgICB8IFllcwpib29sZWFuICAgICAgICB8IG5vCgojIyBFeGFtcGxlIG91dHB1dAoKSW50ZXJuYWwgc2VydmVyIGVycm9yOgoKYGBgYApHRVQgZnV6emluZyAvbWFuYWdlcnMvZXhwZW5zZXMve2V4cGVuc2VzX2lkfS9hdHRhY2htZW50cwoKKiBJTlRFUk5BTCBTRVJWRVIgRVJST1IKICBFbmRwb2ludCByZXR1cm5lZCA1MDAgYnV0IGV4cGVjdGVkIG9uZSBvZiBbMjAwXQogIEdFVCBodHRwczovL2Rldi5teWFwaS5leGFtcGxlL21hbmFnZXJzL2V4cGVuc2VzLzk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5OTk5L2F0dGFjaG1lbnRzCmBgYGAKClJlc3BvbnNlIGRvZXNuJ3QgY29uZm9ybSB0byB0aGUgT0FTMyBzcGVjOgoKYGBgYAotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpHRVQgZnV6emluZyAvZW1wbG95ZWVzL2V4cGVuc2VzL3tleHBlbnNlc19pZH0KCi0gVW5leHBlY3RlZCBzdGF0dXMgY29kZQogIEVuZHBvaW50IHJldHVybmVkIDQwNCBidXQgZXhwZWN0ZWQgb25lIG9mIFsyMDAsICdkZWZhdWx0J10KICBHRVQgaHR0cHM6Ly9kZXYubXlhcGkuZXhhbXBsZS9lbXBsb3llZXMvZXhwZW5zZXMvKSQjKioqXgpgYGBgCgpgYGBgClBPU1QgZnV6emluZyAvZW1wbG95ZWVzL2V4cGVuc2VzL3tleHBlbnNlc19pZH0KCi0gVW5leHBlY3RlZCBzdGF0dXMgY29kZQogIEVuZHBvaW50IHJldHVybmVkIDQwMCBidXQgZXhwZWN0ZWQgb25lIG9mIFsyMDEsICdkZWZhdWx0J10KICBQT1NUIGh0dHBzOi8vZGV2Lm15YXBpLmV4YW1wbGUvZW1wbG95ZWVzL2V4cGVuc2VzCnsKICAgICJhbW91bnQiOiAiMTIzIiwKICAgICJjb3N0X3R5cGUiOiAiMTIzIiwKICAgICJub3RlIjogIjtzbGVlcCAxMCIsCiAgICAidHJhbnNhY3Rpb25fZGF0ZSI6ICIxMjMiCn0KYGBgYAojIyBDb250cmlidXRvcnMKQSBzcGVjaWFsIHRoYW5rcyB0byB0aGUgY29udHJpYnV0b3JzIG91dHNpZGUgb2YgVldUIERpZ2l0YWwuCnxOYW1lfENvbnRyaWJ1dGlvbnwKfC0tLXwtLS18CnxbSm9ycml0IEZvbG1lcl0oaHR0cHM6Ly9naXRodWIuY29tL2pvcnJpdGZvbG1lcil8U3RhcnRlZCB0aGUgcHJvamVjdCBhbmQgY3JlYXRlZCBhIGJhc2UgZm9yIHRoZSBmdXp6ZXIufAojIyBMSUNFTlNFCgpHUEwzCg== readmeEtag: '"ab9290c2f3d5444830a630b4149fbd23f70a443e"' readmeLastModified: Thu, 22 Jul 2021 15:59:51 GMT repositoryId: 244578763 description: Simple fuzzer for OpenAPI 3 specification based APIs created: '2020-03-03T08:15:47Z' updated: '2025-07-25T20:48:07Z' language: Python archived: false stars: 22 watchers: 3 forks: 4 owner: vwt-digital logo: https://avatars.githubusercontent.com/u/51355845?v=4 license: GPL-3.0 repoEtag: '"4eb527c13d22c90e1472028fa8cd3147d28555965fd52be63884e6fd7cce3474"' repoLastModified: Fri, 25 Jul 2025 20:48:07 GMT foundInMaster: true id: f357b9cac35c5789bafaa767f4fc3b50 - source: https://openapi.tools/ name: vREST NG category: Testing language: JavaScript link: https://ng.vrest.io repository: null source_description: >- vREST NG is a simple and powerful application for API Automation. It Allows to use OpenAPI specification into vREST NG to drive your API testing that validates the API responses against JSON Schema and also provides powerful response validation capabilities. v2: true v3: true foundInMaster: true id: ea2266de94cddafe3121aac607d00ec7 - source: https://openapi.tools/ name: openapi-validator-bundle category: - Data Validators - Parsers repository: https://github.com/cydrickn/openapi-validator-bundle language: PHP source_description: Validates Request and Response using Symfony Framework v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQXBpIFZhbGlkYXRvciBCdW5kbGUKClN5bWZvbnkgQnVuZGxlIGZvciB2YWxpZGF0aW5nIFJlcXVlc3QgYW5kIFJlc3BvbnNlIGJhc2VkIG9uIFtvcGVuIGFwaSBzcGVjaWZpY2F0aW9uXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pIDMuCgojIyMgUmVxdWlyZW1lbnRzCgotIFN5bWZvbnkgPj0gNQotIFBIUCA+PSA3LjQKLSBQSFAgRXh0ZW5zaW9uIEpTT04KCiMjIyMgT3B0aW9uYWwKCi0gbmVsbWlvL2FwaS1kb2MtYnVuZGxlID49IDQuMCAoQ3VycmVudGx5IGluIEJldGEpCiAgICAtIFdoeSA0LjAsIHZlcnNpb24gMyBhbmQgYmVsb3cgb25seSBzdXBwb3J0cyB2ZXJzaW9uIDIgb2YgT3BlbkFQSSBTcGVjaWZpY2F0aW9uLgoKCiMjIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKY29tcG9zZXIgcmVxdWlyZSBjeWRyaWNrbi9vcGVuYXBpLXZhbGlkYXRvci1idW5kbGUKYGBgCgojIyMjIFNldHRpbmcgVXAKCkFkZCB0aGUgYnVuZGxlIGluIHlvdXIgYGNvbmZpZy9idW5kbGVzLnBocGAKCmBgYHBocAo8P3BocAoKcmV0dXJuIFsKICAgIC8vIC4uLgogICAgQ3lkcmlja25cT3BlbkFwaVZhbGlkYXRvckJ1bmRsZVxDeWRyaWNrbk9wZW5BcGlWYWxpZGF0b3JCdW5kbGU6OmNsYXNzID0+IFsnYWxsJyA9PiB0cnVlXSwKXTsKYGBgCgpBZGQgY29uZmlndXJhdGlvbiBgY29uZmlnL3BhY2thZ2VzL2N5ZHJpY2tuX29wZW5hcGlfdmFsaWRhdG9yLnltbGAKYGBgeWFtbApjeWRyaWNrbl9vcGVuX2FwaV92YWxpZGF0b3I6CiAgICB2YWxpZGF0ZV9yZXF1ZXN0OiB0cnVlCiAgICB2YWxpZGF0ZV9yZXNwb25zZTogdHJ1ZQogICAgc2NoZW1hOgogICAgICBmYWN0b3J5OiB5YW1sLWZpbGUKICAgICAgZmlsZTogJWtlcm5lbC5wcm9qZWN0X2RpciUvY29uZmlnL29wZW5hcGkvc3BlYy55YW1sCmBgYAoKIyMjIyBDb25maWd1cmF0aW9ucwoKfENvbmZpZyAgICAgICAgICAgfFR5cGUgICB8UmVxdWlyZWR8QWNjZXB0ZWQgVmFsdWUgICAgICAgICAgICAgICAgfERlZmF1bHQgIHxEZXNjcmlwdGlvbnwKfC0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS18LS0tLS0tLS18LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLXwtLS0tLS0tLS0tLXwKfHZhbGlkYXRlX3JlcXVlc3QgfEJvb2xlYW58WWVzICAgICB8dHJ1ZSBvciBmYWxzZSAgICAgICAgICAgICAgICAgfHRydWUgICAgIHxFbmFibGUgdmFsaWRhdGluZyBvZiByZXF1ZXN0fAp8dmFsaWRhdGVfcmVzcG9uc2V8Qm9vbGVhbnxZZXMgICAgIHx0cnVlIG9yIGZhbHNlICAgICAgICAgICAgICAgICB8dHJ1ZSAgICAgfEVuYWJsZSB2YWxpZGF0aW5nIG9mIHJlc3BvbnNlfAp8c2NoZW1hLmZhY3RvcnkgICB8U3RyaW5nIHxZZXMgICAgIHx5YW1sLWZpbGUsIGpzb24tZmlsZSBvciBuZWxtaW98eWFtbC1maWxlfEZhY3RvcnkgdG8gdXNlIHRvIGdlbmVyYXRlIHRoZSBzY2hlbWEgZm9yIHZhbGlkYXRpb258CnxzY2hlbWEuZmlsZSAgICAgIHxTdHJpbmcgfFJlcXVpcmVkIG9ubHkgZm9yIHlhbWwtZmlsZSBhbmQganNvbi1maWxlfHx8RmlsZSBwYXRoIG9mIHRoZSBzcGVjaWZpY2F0aW9ufAoKIyMjIFRPRE8KCi0gW3hdIEFkZCBOZWxtaW8gQXBpIFNjaGVtYSBGYWN0b3J5Ci0gW3hdIEFkZCBEeW5hbWljIENvbmZpZ3VyYXRpb24KLSBbeF0gQWRkIGRvY3VtZW50Ci0gWyBdIEFkZCBSb3V0ZSBTY2hlbWEgRmFjdG9yeQotIFsgXSBBZGQgUEhQIGZpbGUgU2NoZW1hIEZhY3RvcnkKLSBbIF0gQ29kZSBjb3ZlcmFnZSBvZiAxMDAlCi0gWyBdIEFkZCBDSQotIFsgXSBTdXBwb3J0IExvd2VyIFBIUCBWZXJzaW9uID49IDcuMQotIFsgXSBTdXBwb3J0IExvd2VyIFN5bWZvbnkgVmVyc2lvbiA+PSAzCg== readmeEtag: '"e050dcb00429047d02dfb59bab01327330eccf73"' readmeLastModified: Thu, 20 Jan 2022 14:37:40 GMT repositoryId: 284669411 description: >- Symfony Bundle for validating Request and Response base on open api specification. created: '2020-08-03T10:21:26Z' updated: '2026-01-16T08:23:57Z' language: PHP archived: false stars: 12 watchers: 1 forks: 6 owner: cydrickn logo: https://avatars.githubusercontent.com/u/6417652?v=4 repoEtag: '"225c696d7c75d0dee742b8b16509bd69cd84be9a67dca6394d924e91d2dbb126"' repoLastModified: Fri, 16 Jan 2026 08:23:57 GMT foundInMaster: true id: 70d091df07cce505a35dbd6fa1789796 - source: https://openapi.tools/ name: Goa category: DSL language: Go source_description: >- Goa provides a holistic approach for developing remote APIs and microservices in Go. implementers don't have to worry about the documentation getting out of sync as Goa takes care of generating OpenAPI specifications for HTTP based services and gRPC protocol buffer files for gRPC based services link: https://goa.design v2: true v3: true foundInMaster: true id: bda876c6645d589b7595411a464db1c3 - source: - https://openapi.tools/ - openapi3 tags name: OpenAPI HttpFoundation Testing category: Data Validators language: PHP source_description: >- Validate your HttpFoundation requests and responses (Symfony, Laravel, Drupal...) against OpenAPI definitions repository: https://github.com/osteel/openapi-httpfoundation-testing v2: false v3: true repositoryMetadata: base64Readme: >- # OpenAPI HttpFoundation Testing

[![Build Status](https://github.com/osteel/openapi-httpfoundation-testing/workflows/CI/badge.svg)](https://github.com/osteel/openapi-httpfoundation-testing/actions)
[![Latest Stable Version](https://img.shields.io/packagist/v/osteel/openapi-httpfoundation-testing)](https://packagist.org/packages/osteel/openapi-httpfoundation-testing)
[![License](https://img.shields.io/packagist/l/osteel/openapi-httpfoundation-testing)](https://packagist.org/packages/osteel/openapi-httpfoundation-testing)
[![Downloads](https://img.shields.io/packagist/dt/osteel/openapi-httpfoundation-testing)](https://packagist.org/packages/osteel/openapi-httpfoundation-testing)


Validate HttpFoundation requests and responses against OpenAPI (3+) definitions.

See [this post](https://tech.osteel.me/posts/openapi-backed-api-testing-in-php-projects-a-laravel-example "OpenAPI-backed API testing in PHP projects – a Laravel example") for more details and [this repository](https://github.com/osteel/openapi-httpfoundation-testing-laravel-example) for an example use in a Laravel project.

> [!IMPORTANT]
> While you can safely use this package for your projects, as long as version `1.0` has not been released "minor" version patches can contain breaking changes. Make sure to check the [release section](../../releases) before you upgrade.

## Why?

[OpenAPI](https://swagger.io/specification/) is a specification intended to describe RESTful APIs in a way that can be understood by both humans and machines.

By validating an API's requests and responses against the OpenAPI definition that describes it, we guarantee that the API is used correctly and behaves in accordance with the documentation we provide, thus making the OpenAPI definition the single source of truth.

The [HttpFoundation component](https://symfony.com/doc/current/components/http_foundation.html) is developed and maintained as part of the [Symfony framework](https://symfony.com/). It is used to handle HTTP requests and responses in projects such as Symfony, Laravel, Drupal, and [many others](https://symfony.com/components/HttpFoundation).

## How does it work?

This package is built on top of [OpenAPI PSR-7 Message Validator](https://github.com/thephpleague/openapi-psr7-validator), which validates [PSR-7 messages](https://www.php-fig.org/psr/psr-7/) against OpenAPI definitions.

It converts HttpFoundation request and response objects to PSR-7 messages using Symfony's [PSR-7 Bridge](https://symfony.com/doc/current/components/psr7.html) and [Tobias Nyholm](https://github.com/Nyholm)'s [PSR-7 implementation](https://github.com/Nyholm/psr7), before passing them on to OpenAPI PSR-7 Message Validator.

## Installation

> [!NOTE]
> This package supports PHP 8.0 and above.

Via Composer:

```bash
composer require --dev osteel/openapi-httpfoundation-testing
```

## Usage

> [!NOTE]
> This package is mostly intended to be used as part of an API test suite.

Import the builder class:

```php
use Osteel\OpenApi\Testing\ValidatorBuilder;
```

Use the builder to create a [`\Osteel\OpenApi\Testing\Validator`](/src/Validator.php) object, using one of the available factory methods for YAML or JSON:

```php
// From a local file:

$validator = ValidatorBuilder::fromYamlFile($yamlFile)->getValidator();
$validator = ValidatorBuilder::fromJsonFile($jsonFile)->getValidator();

// From a string:

$validator = ValidatorBuilder::fromYamlString($yamlString)->getValidator();
$validator = ValidatorBuilder::fromJsonString($jsonString)->getValidator();

// From a URL:

$validator = ValidatorBuilder::fromYamlUrl($yamlUrl)->getValidator();
$validator = ValidatorBuilder::fromJsonUrl($jsonUrl)->getValidator();

// Automatic detection (slower):

$validator = ValidatorBuilder::fromYaml($yamlFileOrStringOrUrl)->getValidator();
$validator = ValidatorBuilder::fromJson($jsonFileOrStringOrUrl)->getValidator();
```

> [!TIP]
> You can also use a dependency injection container to bind the `ValidatorBuilder` class to the [`ValidatorBuilderInterface`](/src/ValidatorBuilderInterface.php) interface it implements and inject the interface instead, which would also be useful for testing and mocking.

You can now validate `\Symfony\Component\HttpFoundation\Request` and `\Symfony\Component\HttpFoundation\Response` objects for a given [path](https://swagger.io/specification/#paths-object) and method:

```php
$validator->validate($response, '/users', 'post');
```

> [!TIP]
> For convenience, objects implementing `\Psr\Http\Message\ServerRequestInterface` or `\Psr\Http\Message\ResponseInterface` are also accepted.

In the example above, we check that the response matches the OpenAPI definition for a `POST` request on the `/users` path.

Each of OpenAPI's [supported HTTP methods](https://swagger.io/docs/specification/paths-and-operations/ "Paths and Operations") (`DELETE`, `GET`, `HEAD`, `OPTIONS`, `PATCH`, `POST`, `PUT` and `TRACE`) also has a shortcut method that calls `validate` under the hood, meaning the line above could also be written this way:

```php
$validator->post($response, '/users');
```

Validating a request object works exactly the same way:

```php
$validator->post($request, '/users');
```

In the example above, we check that the request matches the OpenAPI definition for a `POST` request on the `/users` path.

The `validate` method returns `true` in case of success, and throw a [`\Osteel\OpenApi\Testing\Exceptions\ValidationException`](/src/Exceptions/ValidationException.php) exception in case of error.

## Caching

This package supports caching to speed up the parsing of OpenAPI definitions. Simply pass your [PSR-6](https://www.php-fig.org/psr/psr-6/) or [PSR-16](https://www.php-fig.org/psr/psr-16/) cache object to the `setCache` method of the [`ValidatorBuilder`](/src/ValidatorBuilder.php) class.

Here is an example using Symfony's [Array Cache Adapter](https://symfony.com/doc/current/components/cache/adapters/array_cache_adapter.html "Array Cache Adapter"):

```php
use Osteel\OpenApi\Testing\ValidatorBuilder;
use Symfony\Component\Cache\Adapter\ArrayAdapter;

$cache = new ArrayAdapter();
$validator = ValidatorBuilder::fromYamlFile($yamlFile)->setCache($cache)->getValidator();
```

## Extending the package

There are two main extension points – message adapters and cache adapters.

### Message adapters

The [`ValidatorBuilder`](/src/ValidatorBuilder.php) class uses the [`HttpFoundationAdapter`](/src/Adapters/HttpFoundationAdapter.php) class as its default HTTP message adapter. This class converts HttpFoundation request and response objects to their PSR-7 counterparts.

If you need to change the adapter's logic, or if you need a new adapter altogether, create a class implementing the [`MessageAdapterInterface`](/src/Adapters/MessageAdapterInterface.php) interface and pass it to the `setMessageAdapter` method of the [`ValidatorBuilder`](/src/ValidatorBuilder.php) class:

```php
$validator = ValidatorBuilder::fromYamlFile($yamlFile)
    ->setMessageAdapter($yourAdapter)
    ->getValidator();
```

### Cache adapters

The [`ValidatorBuilder`](/src/ValidatorBuilder.php) class uses the [`Psr16Adapter`](/src/Cache/Psr16Adapter.php) class as its default cache adapter. This class converts PSR-16 cache objects to their PSR-6 counterparts.

If you need to change the adapter's logic, or if you need a new adapter altogether, create a class implementing the [`CacheAdapterInterface`](/src/Cache/CacheAdapterInterface.php) interface and pass it to the `setCacheAdapter` method of the [`ValidatorBuilder`](/src/ValidatorBuilder.php) class:

```php
$validator = ValidatorBuilder::fromYamlFile($yamlFile)
    ->setCacheAdapter($yourAdapter)
    ->getValidator();
```

### Other interfaces

The [`ValidatorBuilder`](/src/ValidatorBuilder.php) and [`Validator`](/src/Validator.php) classes are `final` but they implement the [`ValidatorBuilderInterface`](/src/ValidatorBuilderInterface.php) and [`ValidatorInterface`](/src/ValidatorInterface.php) interfaces respectively for which you can provide your own implementations if you need to.

## Change log

Please see the [Releases section](../../releases) for details.

## Contributing

Please see [CONTRIBUTING](/.github/CONTRIBUTING.md) for details.

## Credits

**People**

- [Yannick Chenot](https://github.com/osteel)
- [Patrick Rodacker](https://github.com/lordrhodos)
- [Johnathan Michael Dell](https://github.com/johnathanmdell)
- [Paul Mitchum](https://github.com/paul-m)
- [All Contributors](../../contributors)

Special thanks to [Pavel Batanov](https://github.com/scaytrase) for his advice on structuring the package.

**Packages**

- [OpenAPI PSR-7 Message Validator](https://github.com/thephpleague/openapi-psr7-validator)
- [The PSR-7 Bridge](https://symfony.com/doc/current/components/psr7.html)
- [PSR-7 implementation](https://github.com/Nyholm/psr7)

## License

The MIT License (MIT). Please see [License File](LICENSE.md) for more information.
 readmeEtag: '"af0be1a339c00740e84aeec6d6eb2579bf7c638f"' readmeLastModified: Thu, 04 Dec 2025 11:14:59 GMT repositoryId: 306618893 description: >- Validate your HttpFoundation requests and responses against OpenAPI (3+) definitions created: '2020-10-23T11:42:16Z' updated: '2026-01-26T08:55:27Z' language: PHP archived: false stars: 119 watchers: 4 forks: 15 owner: osteel logo: https://avatars.githubusercontent.com/u/436467?v=4 license: MIT repoEtag: '"6039ad8c2d60c64ab412a26e786986ab961cda6d0c474c8b19d470df9ffdb2ce"' repoLastModified: Mon, 26 Jan 2026 08:55:27 GMT foundInMaster: true id: 940ce85dbc99fab7a0ea193ac4ade24c v3_1: true - source: - https://openapi.tools/ - openapi3 tags name: VSCode OpenAPI Snippets category: - Text Editors - Parsers language: Any source_description: >- OpenAPI Snippets for Visual Studio Code editor, includes split file validation v3: true link: https://marketplace.visualstudio.com/items?itemName=proohit.openapi-snippets repository: https://github.com/proohit/openapi-snippets repositoryMetadata: base64Readme: >- IyBPcGVuQXBpIFNuaXBwZXRzCgojIyBVc2FnZQoKIVtzbmlwcGV0LXByZXZpZXddKC4vb3BlbmFwaS1zbmlwcGV0cy1wcmV2aWV3LmdpZikKCiMjIyBzbmlwcGV0cwoKY3VycmVudGx5IHN1cHBvcnRlZCBzbmlwcGV0czoKCnwgICAgICAgICAgICBQcmVmaXggfCBDb250ZW50ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0tLS06IHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwKfCAgICAgICAgIGBwYXRoZ2V0YCB8IEdFVCBwYXRoIG9iamVjdCAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgICAgYHBhdGhnZXRwYXJhbWAgfCBHRVQgcGF0aCBvYmplY3Qgd2l0aCByZXF1ZXN0IHBhcmFtZXRlcnMgICAgfAp8ICAgICAgICBgcGF0aHBvc3RgIHwgUE9TVCBwYXRoIG9iamVjdCAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCAgICAgICAgIGBwYXRocHV0YCB8IFBVVCBwYXRoIG9iamVjdCAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgICAgYHBhdGhwdXRwYXJhbWAgfCBQVVQgcGF0aCBvYmplY3Qgd2l0aCByZXF1ZXN0IHBhcmFtZXRlcnMgICAgfAp8ICAgICAgYHBhdGhkZWxldGVgIHwgREVMRVRFIHBhdGggb2JqZWN0ICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgcGF0aGRlbGV0ZXBhcmFtYCB8IERFTEVURSBwYXRoIG9iamVjdCB3aXRoIHJlcXVlc3QgcGFyYW1ldGVycyB8CnwgICAgYHNjaGVtYU9iamVjdGAgfCBTY2hlbWEgb2JqZWN0ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8ICBgcHJvcGVydHlPYmplY3RgIHwgUHJvcGVydHkgb2JqZWN0IGZvciBwcm9wZXJ0aWVzIG9mIGEgc2NoZW1hIHwKfCAgICAgYGNvbnRlbnRUeXBlYCB8IFNldmVyYWwgY29udGVudCB0eXBlcyAgICAgICAgICAgICAgICAgICAgICB8CgojIyMgVmFsaWRhdGlvbgoKSlNPTiBhbmQgWUFNTC9ZTUwgZmlsZXMgYXJlIGJlaW5nIHZhbGlkYXRlZCBhY2NvcmRpbmcgdG8gdGhlaXIgcmVzcGVjdGl2ZSB1c2FnZSAoZS5nLiBTY2hlbWEgT2JqZWN0cykuIEZvciBzY2hlbWEgc3BlY2lmaWNhdGlvbnMsIHNlZSBbT3BlbkFwaSBTcGVjaWZpY2F0aW9uIHYzXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYXN0ZXIvc2NoZW1hcy92My4wL3NjaGVtYS5qc29uKQoKKipOb3RlKio6IEZvciBJbnRlbGxpU2Vuc2UgdG8gdmFsaWRhdGUgdGhlIGZpbGVzLCBmaWxlbmFtZXMgaGF2ZSB0byBtYXRjaCB0aGVpciByZXNwZWN0aXZlIHNjaGVtYToKCnwgICAgICAgRmlsZSBuYW1lIHBhdHRlcm4gfCBFeGFtcGxlICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tLS06IHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwKfCBgW2NvbXBvbmVudHMvc2NoZW1hcy9cKiB8IFwqLXNjaGVtYV0uKGpzb24seWFtbCx5bWwpYCB8IGNvbXBvbmVudHMvc2NoZW1hcy91c2Vycy5qc29uIHwKfCAgICAgICAgICAgICAgYFtwYXRocy9cKiB8IFwqLXBhdGhdLihqc29uLHlhbWwseW1sKWAgICB8IHVzZXJzLXBhdGguanNvbiB8CgpZb3UgY2FuIG5hdmlnYXRlIHRocm91Z2ggcHJlZGVmaW5lZCBhbmNob3IgcG9pbnRzIHdpdGggVEFCIGtleS4gVGhpcyBvbmx5IHdvcmtzIHJpZ2h0IGFmdGVyIGluc2VydGluZyBhIHNuaXBwZXQuCgoqKk5vdGUqKjogRm9yIFlBTUwvWU1MIGZpbGVzIHRvIGJlIHN1cHBvcnRlZCwgdGhlIFtZQU1MIEV4dGVuc2lvbl0oaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPXJlZGhhdC52c2NvZGUteWFtbCkgaGFzIHRvIGJlIGluc3RhbGxlZCEKCiMjIyBLZXliaW5kaW5ncwoKWW91IGNhbiBjb25maWd1cmUgY3VzdG9tIGtleWJpbmRpbmdzIGZvciB0aGUgc25pcHBldHMuIEZvciB0aGlzLCBvcGVuIHRoZSBrZXliaW5kaW5ncy5qc29uIGZpbGUgb2YgVlMgQ29kZSAoU2hvcnRjdXQgYEN0cmwrSyBDdHJsK1NgLCB0aGVuIGNsaWNrIG9uIHRoZSBgT3BlbiBLZXlib2FyZCBTaG9ydGN1dHMgKEpTT04pYCBidXR0b24gaW4gdGhlIHVwcGVyIHJpZ2h0IGNvcm5lcikuCkZvciBhbnkga2V5YmluZGluZywgdXNlIHRoaXMgdGVtcGxhdGUgYW5kIGVkaXQgdGhlIGZpZWxkcyB3aGljaCBhcmUgbWFya2VkIGJ5IGA8PD4+YDoKCmZvciBqc29uIGZpbGVzOgoKYGBganNvbgp7CiAgImtleSI6ICI8PFlvdXIgZGVzaXJlZCBzaG9ydGN1dD4+IiwKICAiY29tbWFuZCI6ICJlZGl0b3IuYWN0aW9uLmluc2VydFNuaXBwZXQiLAogICJ3aGVuIjogImVkaXRvclRleHRGb2N1cyIsCiAgImFyZ3MiOiB7CiAgICAibGFuZ0lkIjogImpzb24iLAogICAgIm5hbWUiOiAiPDxTbmlwcGV0IHByZWZpeD4+IgogIH0KfQpgYGAKCmZvciB5YW1sIGZpbGVzOgoKYGBganNvbgp7CiAgImtleSI6ICI8PFlvdXIgZGVzaXJlZCBzaG9ydGN1dD4+IiwKICAiY29tbWFuZCI6ICJlZGl0b3IuYWN0aW9uLmluc2VydFNuaXBwZXQiLAogICJ3aGVuIjogImVkaXRvclRleHRGb2N1cyIsCiAgImFyZ3MiOiB7CiAgICAibGFuZ0lkIjogInlhbWwiLAogICAgIm5hbWUiOiAiPDxTbmlwcGV0IHByZWZpeD4+IgogIH0KfQpgYGAKCkZvciBgbmFtZWAgeW91IGNhbiB1c2UgYW55IFtzbmlwcGV0IHByZWZpeF0oI3NuaXBwZXRzKS4KCiMjIEluc3RhbGxhdGlvbgoKSnVzdCBkb3dubG9hZCBhbmQgaW5zdGFsbCBpdCBmcm9tIFtWUyBNYXJrZXRwbGFjZV0oaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPXByb29oaXQub3BlbmFwaS1zbmlwcGV0cykuIFRoZXJlIGFyZSBubyBtb3JlIHNldHRpbmdzLCBzbyB0aGUgZXh0ZW5zaW9uIGlzIHJlYWR5IHRvIHVzZSBhZnRlciBpbnN0YWxsYXRpb24uIEl0IGZ1bGx5IHdvcmtzIHdpdGggYW5kIGV4dGVuZHMgW09wZW5BUEkgKFN3YWdnZXIpIEVkaXRvcl0oaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPTQyQ3J1bmNoLnZzY29kZS1vcGVuYXBpKQo= readmeEtag: '"97afdc1cec810f47e55845ecd4340a725323cdb2"' readmeLastModified: Fri, 30 Oct 2020 20:24:53 GMT repositoryId: 256450021 description: OpenAPI snippets and external file validation extension for VSCode created: '2020-04-17T08:46:16Z' updated: '2025-10-06T09:31:09Z' language: null archived: false stars: 9 watchers: 2 forks: 2 owner: proohit logo: https://avatars.githubusercontent.com/u/46965017?v=4 repoEtag: '"db3979142ecacd71ed470732599d89a475839b9fe19ba44a08e49c3ccbcb0649"' repoLastModified: Mon, 06 Oct 2025 09:31:09 GMT foundInMaster: true id: 641631c1edd1eaf61a89b81a4ae7c8eb - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/arbs-io/vscode-openapi-viewer v3: true id: e10b173576edb293957b5e57936fce83 repositoryMetadata: base64Readme: >- IyB2c2NvZGUtb3BlbmFwaS12aWV3ZXIKCiFbVmlzdWFsIFN0dWRpbyBNYXJrZXRwbGFjZSBWZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL3Zpc3VhbC1zdHVkaW8tbWFya2V0cGxhY2Uvdi9BbmRyZXdCdXRzb24udnNjb2RlLW9wZW5hcGktdmlld2VyKQohWy5naXRodWIvd29ya2Zsb3dzL2NvZGVxbC1hbmFseXNpc10oaHR0cHM6Ly9naXRodWIuY29tL2FyYnMtaW8vdnNjb2RlLW9wZW5hcGktdmlld2VyL2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVxbC1hbmFseXNpcy55bWwvYmFkZ2Uuc3ZnKQohW1Zpc3VhbCBTdHVkaW8gTWFya2V0cGxhY2UgSW5zdGFsbHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdmlzdWFsLXN0dWRpby1tYXJrZXRwbGFjZS9pL0FuZHJld0J1dHNvbi52c2NvZGUtb3BlbmFwaS12aWV3ZXIpClshW0dpdEh1YiBpc3N1ZXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy9hcmJzLWlvL3ZzY29kZS1vcGVuYXBpLXZpZXdlci5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vYXJicy1pby92c2NvZGUtb3BlbmFwaS12aWV3ZXIvaXNzdWVzKQohW1Zpc3VhbCBTdHVkaW8gTWFya2V0cGxhY2UgUmF0aW5nXShodHRwczovL2ltZy5zaGllbGRzLmlvL3Zpc3VhbC1zdHVkaW8tbWFya2V0cGxhY2Uvci9BbmRyZXdCdXRzb24udnNjb2RlLW9wZW5hcGktdmlld2VyKQoKWyFbTWFpbnRhaW5hYmlsaXR5IFJhdGluZ10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9YXJicy1pb192c2NvZGUtb3BlbmFwaS12aWV3ZXImbWV0cmljPXNxYWxlX3JhdGluZyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPWFyYnMtaW9fdnNjb2RlLW9wZW5hcGktdmlld2VyKQpbIVtTZWN1cml0eSBSYXRpbmddKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PWFyYnMtaW9fdnNjb2RlLW9wZW5hcGktdmlld2VyJm1ldHJpYz1zZWN1cml0eV9yYXRpbmcpXShodHRwczovL3NvbmFyY2xvdWQuaW8vc3VtbWFyeS9uZXdfY29kZT9pZD1hcmJzLWlvX3ZzY29kZS1vcGVuYXBpLXZpZXdlcikKWyFbUmVsaWFiaWxpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1hcmJzLWlvX3ZzY29kZS1vcGVuYXBpLXZpZXdlciZtZXRyaWM9cmVsaWFiaWxpdHlfcmF0aW5nKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9YXJicy1pb192c2NvZGUtb3BlbmFwaS12aWV3ZXIpClshW0J1Z3NdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PWFyYnMtaW9fdnNjb2RlLW9wZW5hcGktdmlld2VyJm1ldHJpYz1idWdzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9YXJicy1pb192c2NvZGUtb3BlbmFwaS12aWV3ZXIpClshW1Z1bG5lcmFiaWxpdGllc10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9YXJicy1pb192c2NvZGUtb3BlbmFwaS12aWV3ZXImbWV0cmljPXZ1bG5lcmFiaWxpdGllcyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPWFyYnMtaW9fdnNjb2RlLW9wZW5hcGktdmlld2VyKQoKIyMgV2hhdCBpcyBWaXN1YWwgU3R1ZGlvIENvZGUgT3BlbkFQSSB2aWV3ZXI/CgpUaGlzIGV4dGVuc2lvbiBhbGxvd3MgdXNlcnMgdG8gdmlldyBPcGVuQVBJIHNwZWNpZmljYXRpb25zLiBUaGUgZXh0ZW5zaW9uIHN1cHBvcnQgYm90aCBganNvbmAgYW5kIGB5YW1sYCBmb3JtYXQuIEl0IHByb3ZpZGVzIGEgcHJldmlldyBvZiB0aGUgc3BlY2lmaWNhdGlvbiBpbiB0aGUgZm9ybSBvZiBhbiBpbnRlcmFjdGl2ZSBkb2N1bWVudCwgd2hpY2ggY2FuIGJlIHVzZWQgdG8gdGVzdCBhbmQgZGVidWcgQVBJcy4gVGhlIHByZXZpZXcgY2FuIGJlIGFjY2Vzc2VkIGZyb20gd2l0aGluIHRoZSBWaXN1YWwgU3R1ZGlvIGVkaXRvciwgbWFraW5nIGl0IGNvbnZlbmllbnQgZm9yIGRldmVsb3BlcnMgdG8gdmlldyBhbmQgZWRpdCB0aGUgc3BlY2lmaWNhdGlvbiBhcyB0aGV5IHdvcmsgb24gdGhlaXIgY29kZS4KClRoZSBleHRlbnNpb24gaXMgaGFuZHkgZm9yIGRldmVsb3BlcnMgYW5kIGFyY2hpdGVjdHMgd29ya2luZyBvbiBBUEktYmFzZWQgcHJvamVjdHMsIGFsbG93aW5nIHRoZW0gdG8gc2VlIGhvdyB0aGVpciBBUElzIHdpbGwgYmVoYXZlIGFuZCBpbnRlcmFjdCB3aXRoIG90aGVyIGNvbXBvbmVudHMuIEl0IGNhbiBhbHNvIGJlIHVzZWQgdG8gZ2VuZXJhdGUgZG9jdW1lbnRhdGlvbiBmb3IgQVBJcywgd2hpY2ggY2FuIGJlIGhlbHBmdWwgZm9yIGRldmVsb3BlcnMgd2hvIGFyZSB3b3JraW5nIG9uIHByb2plY3RzIHdpdGggbXVsdGlwbGUgdGVhbXMgb3Igc2hhcmluZyBBUElzIHdpdGggZXh0ZXJuYWwgY2xpZW50cy4KClRoZSB2c2NvZGUtb3BlbmFwaS12aWV3ZXIgZXh0ZW5zaW9uIGlzIGEgdmFsdWFibGUgdG9vbCBmb3IgZGV2ZWxvcGVycyBhbmQgYXJjaGl0ZWN0cyB3b3JraW5nIHdpdGggT3BlbkFQSSBzcGVjaWZpY2F0aW9ucy4gSXQgcHJvdmlkZXMgYSBjb252ZW5pZW50IGFuZCBpbnRlcmFjdGl2ZSB3YXkgdG8gcHJldmlldywgdGVzdCwgYW5kIGRlYnVnIEFQSXMgd2l0aGluIFZpc3VhbCBTdHVkaW8uCgojIyBHZXR0aW5nIHN0YXJ0ZWQKCjEuIEZpcnN0LCBvcGVuIFZpc3VhbCBTdHVkaW8gYW5kIGdvIHRvIHRoZSAiRXh0ZW5zaW9ucyIgdGFiIGluIHRoZSB0b29sYmFyLgoyLiBJbiB0aGUgc2VhcmNoIGJhciwgdHlwZSAidnNjb2RlLW9wZW5hcGktdmlld2VyIiBhbmQgcHJlc3MgZW50ZXIuCjMuIENsaWNrIG9uIHRoZSAidnNjb2RlLW9wZW5hcGktdmlld2VyIiBleHRlbnNpb24gaW4gdGhlIHNlYXJjaCByZXN1bHRzIGFuZCBjbGljayAiRG93bmxvYWQiLgo0LiBPbmNlIHRoZSBleHRlbnNpb24gaGFzIGJlZW4gZG93bmxvYWRlZCwgY2xpY2sgIkluc3RhbGwiIHRvIGluc3RhbGwgaXQuCjUuIEFmdGVyIHRoZSBpbnN0YWxsYXRpb24sIHlvdSB3aWxsIG5lZWQgdG8gcmVzdGFydCBWaXN1YWwgU3R1ZGlvIGZvciB0aGUgY2hhbmdlcyB0byB0YWtlIGVmZmVjdC4KNi4gT25jZSBWaXN1YWwgU3R1ZGlvIGhhcyBiZWVuIHJlc3RhcnRlZCwgeW91IGNhbiBvcGVuIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBmaWxlIGJ5IGdvaW5nIHRvICJGaWxlID4gT3BlbiA+IEZpbGUiIGFuZCBzZWxlY3RpbmcgdGhlIG9wZW5hcGkgc3BlY2lmaWNhdGlvbiBmaWxlIGZyb20geW91ciBwcm9qZWN0Lgo3LiBUaGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIHdpbGwgYmUgZGlzcGxheWVkIGluIHRoZSBWaXN1YWwgU3R1ZGlvIGVkaXRvci4gSWYgdGhlIHNwZWNpZmljYXRpb24gaXMgdmFsaWQsIGEgcHJldmlldyBidXR0b24gd2lsbCBhcHBlYXIuIFlvdSBjYW4gdXNlIHRoaXMgZG9jdW1lbnQgdG8gdGVzdCBhbmQgZGVidWcgeW91ciBBUElzIGFuZCBnZW5lcmF0ZSBkb2N1bWVudGF0aW9uIGZvciB0aGVtLgoKVGhhdCdzIGl0ISBZb3Ugc2hvdWxkIG5vdyBiZSBhYmxlIHRvIHVzZSB0aGUgdnNjb2RlLW9wZW5hcGktdmlld2VyIGV4dGVuc2lvbiBpbiBWaXN1YWwgU3R1ZGlvIENvZGUgdG8gdmlldyBhbmQgZWRpdCB5b3VyIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgdmlzdWFsbHksIGFuZCBpbnR1aXRpdmVseS4KCiFbdnNjb2RlLW9wZW5hcGktdmlld2VyLmdpZl0oaW1hZ2VzL3ZzY29kZS1vcGVuYXBpLXZpZXdlci5naWYpCgojIyAqKkhvdyBjYW4gSSBoZWxwPyoqCgpJZiB5b3UgZW5qb3kgdXNpbmcgdGhlIGV4dGVuc2lvbiwgcGxlYXNlIGdpdmUgaXQgYSByYXRpbmcgb24gdGhlIFtWaXN1YWwgU3R1ZGlvIE1hcmtldHBsYWNlXShodHRwczovL21hcmtldHBsYWNlLnZpc3VhbHN0dWRpby5jb20vaXRlbXM/aXRlbU5hbWU9QW5kcmV3QnV0c29uLnZzY29kZS1vcGVuYXBpLXZpZXdlcikuCgpTaG91bGQgeW91IGVuY291bnRlciBidWdzIG9yIGlmIHlvdSBoYXZlIGZlYXR1cmUgcmVxdWVzdHMsIGhlYWQgb24gb3ZlciB0byB0aGUgW0dpdEh1YiByZXBvXShodHRwczovL2dpdGh1Yi5jb20vYXJicy1pby92c2NvZGUtb3BlbmFwaS12aWV3ZXIpIHRvIG9wZW4gYW4gaXNzdWUgaWYgb25lIGRvZXNuJ3QgYWxyZWFkeSBleGlzdC4KUHVsbCByZXF1ZXN0cyBhcmUgYWxzbyB2ZXJ5IHdlbGNvbWUgc2luY2UgSSBjYW4ndCBhbHdheXMgZ2V0IGFyb3VuZCB0byBmaXhpbmcgYWxsIGJ1Z3MgbXlzZWxmLgoKVGhpcyBpcyBhIHBlcnNvbmFsIHBhc3Npb24gcHJvamVjdCwgc28gbXkgdGltZSBpcyBsaW1pdGVkLgoKQW5vdGhlciB3YXkgdG8gaGVscCBvdXQgaXMgdG8gW3Nwb25zb3IgbWUgb24gR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vc3BvbnNvcnMvYXJicy1pbykuCg== readmeEtag: '"b1a9c2222218ea9bd23dc3e0bc9a00fecc8872b8"' readmeLastModified: Mon, 05 Aug 2024 03:59:13 GMT repositoryId: 584545758 description: >- The vscode-openapi-viewer extension is a valuable tool for developers and architects working with OpenAPI specifications, as it provides a convenient and interactive way to preview, test, and debug APIs within Visual Studio Code. created: '2023-01-02T21:56:59Z' updated: '2025-12-17T19:30:01Z' language: TypeScript archived: false stars: 14 watchers: 1 forks: 5 owner: arbs-io logo: https://avatars.githubusercontent.com/u/71968891?v=4 license: MIT repoEtag: '"51baddc412a84a515b774c8f43cadfadbd5e77677cbc8c145c8269577ee9455d"' repoLastModified: Wed, 17 Dec 2025 19:30:01 GMT category: - Documentation - Text Editors - SDK foundInMaster: true name: VSCode OpenAPI Viewer language: TypeScript link: >- https://marketplace.visualstudio.com/items?itemName=AndrewButson.vscode-openapi-viewer source_description: >- Preview and edit JSON or YAML OpenAPI specifications in Visual Studio Code using RapiDoc. v2: true v3_1: true - source: https://openapi.tools/ name: Spectator category: Testing language: PHP source_description: >- Spectator provides light-weight OpenAPI testing tools you can use within your existing Laravel test suite. v2: false v3: true link: https://github.com/hotmeteor/spectator repository: https://github.com/hotmeteor/spectator repositoryMetadata: base64Readme: >- <p align="center"><img src="https://spectator.s3.us-east-2.amazonaws.com/spectator-logo.png" width="300"></p>

# Spectator

Spectator provides light-weight OpenAPI testing tools you can use within your existing Laravel test suite.

Write tests that verify your API spec doesn't drift from your implementation.

![Tests](https://github.com/hotmeteor/spectator/workflows/Tests/badge.svg)
[![Latest Version on Packagist](https://img.shields.io/packagist/vpre/hotmeteor/spectator.svg?style=flat-square)](https://packagist.org/packages/hotmeteor/spectator)
![PHP from Packagist](https://img.shields.io/packagist/php-v/hotmeteor/spectator)

## Requirements

- PHP 8.1+
- Laravel 10+

## Installation

You can install the package through Composer.

```bash
composer require hotmeteor/spectator --dev
```

Then, publish the config file of this package with this command:

```bash
php artisan vendor:publish --provider="Spectator\SpectatorServiceProvider"
```

The config file will be published in `config/spectator.php`.

### Upgrading from v1 to v2

**Important:** Spectator v2 requires PHP 8.1 and Laravel 10. If you are using an older version of PHP or Laravel, you should not upgrade to v2.

While this should typically be a straightforward upgrade, you should be aware of some of the changes that have been made.

Please read the [UPGRADE.md](UPGRADE.md) file for more information.

## Configuration

### Sources

**Sources** are references to where your API spec lives. Depending on the way you or your team works, or where your spec lives, you may want to configure different sources for different environments.

As you can see from the config, there's three source types available: `local`, `remote`, and `github`. Each source requires the folder where your spec lives to be defined, not the spec file itself. This provides flexibility when working with multiple APIs in one project, or an API fragmented across multiple spec files.

---

#### Local

```env
## Spectator config

SPEC_SOURCE=local
SPEC_PATH=/spec/reference
```

---

#### Remote

_This is using the raw access link from Github, but any remote source can be specified. The SPEC_URL_PARAMS can be used to append any additional parameters required for the remote url._

```env
## Spectator config

SPEC_PATH="https://raw.githubusercontent.com/path/to/repo"
SPEC_URL_PARAMS="?token=ABEDC3E5AQ3HMUBPPCDTTMDAFPMSM"
```

---

#### Github

_This uses the Github Personal Access Token which allows you access to a remote repo containing your contract._

You can view instructions on how to obtain your Personal Access Token from Github at [this link](https://docs.github.com/en/github/authenticating-to-github/creating-a-personal-access-token) .

**Important to note than the SPEC_GITHUB_PATH must included the branch (ex: main) and then the path to the directory containing your contract.**

```env
## Spectator config

SPEC_GITHUB_PATH='main/contracts'
SPEC_GITHUB_REPO='orgOruser/repo'
SPEC_GITHUB_TOKEN='your personal access token'
```

---

### Specifying the Target Spec File

In your tests you will declare the spec file you want to test against:

```php
public function testBasicExample()
{
    Spectator::using('Api.v1.json');

    // ...
```

## Testing

### Paradigm Shift

**Now, on to the good stuff.**

At first, spec testing, or contract testing, may seem counter-intuitive, especially when compared with "feature" or "functional" testing as supported by Laravel's [HTTP Tests](https://laravel.com/docs/8.x/http-tests). 

While _functional_ tests are ensuring that your request validation, controller behavior, events, responses, etc. all behave the way you expect when people interact with your API, _contract_ tests are ensuring that **requests and responses are spec-compliant** - _and that's it_. The data itself could be wrong, but that's outside the scope of a contract test.

### Writing Tests

Spectator introduces a few simple tools to compliment the existing Laravel testing toolbox.

Here's an example of a typical JSON API test:

```php
<?php

class ExampleTest extends TestCase
{
    /**
     * A basic functional test example.
     *
     * @return void
     */
    public function testBasicExample()
    {
        $response = $this->postJson('/user', ['name' => 'Sally']);

        $response
            ->assertStatus(201)
            ->assertJson([
                'created' => true,
            ]);
    }
}
```

And here's an example of a contract test:

```php
<?php

use Spectator\Spectator;

class ExampleTest extends TestCase
{
    /**
     * A basic functional test example.
     *
     * @return void
     */
    public function testBasicExample()
    {
        Spectator::using('Api.v1.json');

        $response = $this->postJson('/user', ['name' => 'Sally']);

        $response
            ->assertValidRequest()
            ->assertValidResponse(201);
    }
}
```

The test is verifying that both the request and the response are valid according to the spec, in this case located in `Api.v1.json`. This type of testing promotes TDD: you can write endpoint contract tests against your endpoints _first_, and then ensure your spec and implementation are aligned.

Within your spec, each possible response should be documented. For example, a single `POST` endpoint may result in a `2xx`, `4xx`, or even `5xx` code response. Additionally, your endpoints will likely have particular parameter validation that needs to be adhered to. 

This is what makes contract testing different from functional testing:

- in **functional testing**, successful and failed responses are tested for outcomes
- in **contract testing**, requests and responses are tested for conformity and outcomes don't matter.

### Debugging

For certain validation errors, a special exception message is thrown which shows error message(s) displayed alongside the expected schema. For example:

```
  ---

The properties must match schema: data
All array items must match schema
The required properties (name) are missing

object++ <== The properties must match schema: data
    status*: string
    data*: array <== All array items must match schema
        object <== The required properties (name) are missing
            id*: string
            name*: string
            slug: string?

  ---
```

A few custom symbols are used:

- "++": Object supports `additionalProperties`
- "\*": Item is `required`
- "?": Item can be `nullable`

## Usage

### Providing a Spec

Define the spec file to test against. This can be defined in your `setUp()` method or in a specific test method.

```php
<?php

use Spectator\Spectator;

class ExampleTest extends TestCase
{
    public function setUp(): void
    {
        parent::setUp();

        Spectator::using('Api.v1.json');
    }

    public function testApiEndpoint()
    {
        // Test request and response...
    }

    public function testDifferentApiEndpoint()
    {
        Spectator::using('Other.v1.json');

        // Test request and response...
    }
}
```

### Testing Requests

When testing endpoints, there are a few new methods:

```php
$this->assertValidRequest();
$this->assertValidResponse($status = null);
$this->assertValidationMessage('Expected validation message');
$this->assertErrorsContain('Check for single error');
$this->assertErrorsContain(['Check for', 'Multiple Errors']);
```

Of course, you can continue to use all existing HTTP test methods:

```php
$this
    ->actingAs($user)
    ->postJson('/comments', [
        'message' => 'Just over here spectating',
    ])
    ->assertCreated()
    ->assertValidRequest()
    ->assertValidResponse();
```

That said, mixing functional and contract testing may become more difficult to manage and read later. It's strongly advised to keep the two types of tests separate.

### Testing Responses

Instead of using the built-in `->assertStatus($status)` method, you may also verify the response that is valid is actually the response you want to check. For example, you may receive a `200` **or** a `202` from a single endpoint, and you want to ensure you're validating the correct response.

```php
$this
    ->actingAs($user)
    ->postJson('/comments', [
        'message' => 'Just over here spectating',
    ])
    ->assertValidRequest()
    ->assertValidResponse(201);
```

When exceptions are thrown that are not specific to this package's purpose, e.g. typos or missing imports, the output will be formatted by default with a rather short message and no stack trace.
This can be changed by disabling Laravel's built-in validation handler which allows for easier debugging when running tests.

This can be done in a few different ways:

```php
class ExampleTestCase
{
    public function setUp(): void
    {
        parent::setUp();

        Spectator::using('Api.v1.json');

        // Disable exception handling for all tests in this file
        $this->withoutExceptionHandling();
    }

    // ...
}
```

```php
class ExampleTestCase
{
    public function test_some_contract_test_example(): void
    {
        // Only disable exception handling for this test
        $this->withouthExceptionHandling();

        // Test request and response ...

    }
}
```

### Deactivating Spectator

If you want to deactivate Spectator for a specific test, you can use the `Spectator::reset` method:

```php
<?php

use Spectator\Spectator;

class ExampleTest extends TestCase
{
    public function setUp(): void
    {
        parent::setUp();

        Spectator::using('Api.v1.json');
    }

    public function testWithoutSpectator()
    {
        Spectator::reset();
        
        // Run your test without Spectator
    }
}
```

## Core Concepts

### Approach

Spectator works by registering a custom middleware that performs request and response validation against a spec.

### Dependencies

For those interested in contributing to Spectator, it is worth familiarizing yourself with the core dependencies used for spec testing:

- `cebe/php-openapi`: Used to parse specs into usable arrays
- `opis/json-schema`: Used to perform validation of an object/array against a spec

## Sponsors

A huge thanks to all our sponsors who help push Spectator development forward!

If you'd like to become a sponsor, please [see here for more information](https://github.com/sponsors/hotmeteor). 💪

## Credits

- Created by [Adam Campbell](https://github.com/hotmeteor)
- Maintained by [Bastien Philippe](https://github.com/bastien-phi), [Jarrod Parkes](https://github.com/jarrodparkes), and [Adam Campbell](https://github.com/hotmeteor)
- Inspired by [Laravel OpenAPI](https://github.com/mdwheele/laravel-openapi) package by [Dustin Wheeler](https://github.com/mdwheele)
- [All Contributors](../../contributors)

<a href = "https://github.com/hotmeteor/spectator/graphs/contributors">
  <img src = "https://contrib.rocks/image?repo=hotmeteor/spectator"/>
</a>

Made with [contributors-img](https://contrib.rocks).

## License

The MIT License (MIT). Please see [License File](LICENSE.md) for more information.
 readmeEtag: '"37d8d44117659bfd1933755f42a09d42e6fc3c5f"' readmeLastModified: Tue, 19 Mar 2024 11:57:51 GMT repositoryId: 267042343 description: OpenAPI testing for PHP created: '2020-05-26T12:54:10Z' updated: '2026-01-21T10:45:24Z' language: PHP archived: false stars: 299 watchers: 6 forks: 56 owner: hotmeteor logo: https://avatars.githubusercontent.com/u/378585?v=4 license: MIT repoEtag: '"268246589da944b059adda25e57ec70e2c673bffc295e3362cd7abc7b15a7b88"' repoLastModified: Wed, 21 Jan 2026 10:45:24 GMT foundInMaster: true v3_1_link: https://github.com/hotmeteor/spectator/issues/100 id: 21ebf4f7492590e7ac32e77be29af173 - source: https://openapi.tools/ name: APIFuzzer category: Data Validators language: Python source_description: >- Fuzz test your application using your OpenAPI definition without coding. Integrate into CI/CD, get Junit XML test result and JSON report of failures v2: true v3: true link: https://github.com/KissPeter/APIFuzzer repository: https://github.com/kisspeter/apifuzzer repositoryMetadata: base64Readme: >- [![Join the chat at https://gitter.im/API-Fuzzer/Lobby](https://badges.gitter.im/API-Fuzzer/Lobby.svg)](https://gitter.im/API-Fuzzer/Lobby?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
[![Test Coverage](https://api.codeclimate.com/v1/badges/bfc9bda00deb5002b665/test_coverage)](https://codeclimate.com/github/KissPeter/APIFuzzer/test_coverage)
[![Codacy Badge](https://api.codacy.com/project/badge/Grade/eab6434d9bd742e3880d8f589a9cc0a6)](https://www.codacy.com/app/KissPeter/APIFuzzer?utm_source=github.com&utm_medium=referral&utm_content=KissPeter/APIFuzzer&utm_campaign=badger)
[![Maintainability](https://api.codeclimate.com/v1/badges/bfc9bda00deb5002b665/maintainability)](https://codeclimate.com/github/KissPeter/APIFuzzer/maintainability)
[![Pypi downloads](https://img.shields.io/pypi/dw/APIFuzzer)](https://pypistats.org/packages/apifuzzer)
[![CI](https://github.com/KissPeter/APIFuzzer/actions/workflows/python-app.yml/badge.svg)](https://github.com/KissPeter/APIFuzzer/actions)

# APIFuzzer — HTTP API Testing Framework

APIFuzzer reads your API description and step by step fuzzes the fields to validate 
if you application can cope with the fuzzed parameters. Does not require coding.

## APIFuzzer main features

* Parse API definition from local file or remote URL
* JSON and YAML file format support
* All HTTP methods are supported
* Fuzzing of request body, query string, path parameter and request header are supported
* Relies on random mutations
* Support CI integration 
    * Generate JUnit XML test report format
    * Send request to alternative URL
    * Support HTTP basic auth from configuration
    * Save report of failed test in JSON format into the pre-configured folder
    * Log to stdout instead of syslog
* Configurable log level

### Supported API definition formats
- [Swagger][]
- [OpenAPI][]

### Planned
- [GraphQL][]
- [API Blueprint][]

## Installation

### With PIP

#### Pre-requirements
1. Python3
2. sudo apt install libcurl4-openssl-dev libssl-dev libcurl4-nss-dev (on Ubuntu 18.04, required by pycurl)
3. sudo apt install gcc libcurl4-nss-dev (on Ubuntu 20.04, required by pycurl)

Latest version:

```shell
pip3 install APIFuzzer
```
Development version: 
Fetch the most recent code from GitHub
```shell
$ git clone https://github.com/KissPeter/APIFuzzer.git
```
Install requirements. If you don't have pip installed, then sudo apt-get install python3-pip -y 
```shell
$ pip3 install -r APIFuzzer/requirements.txt
```

### Using Docker

```shell
$ docker pull kisspeter/apifuzzer:latest
```

## Quick Start

Check the help (some of them are not implemented yet):
```shell

$$ usage: APIFuzzer [-h] [-s SRC_FILE] [--src_url SRC_URL] [-r REPORT_DIR] [--level LEVEL] [-u ALTERNATE_URL] [-t TEST_RESULT_DST]
                 [--log {critical,fatal,error,warn,warning,info,debug,notset}] [--basic_output BASIC_OUTPUT] [--headers HEADERS] [-v ,--version]

APIFuzzer configuration

optional arguments:
  -h, --help            show this help message and exit
  -s SRC_FILE, --src_file SRC_FILE
                        API definition file path. JSON and YAML format is supported
  --src_url SRC_URL     API definition url. JSON and YAML format is supported
  -r REPORT_DIR, --report_dir REPORT_DIR
                        Directory where error reports will be saved. Default is temporally generated directory
  --level LEVEL         Test deepness: [1,2], the higher is the deeper (In progress)
  -u ALTERNATE_URL, --url ALTERNATE_URL
                        Use CLI defined url instead compile the url from the API definition. Useful for testing
  -t TEST_RESULT_DST, --test_report TEST_RESULT_DST
                        JUnit test result xml save path
  --log {critical,fatal,error,warn,warning,info,debug,notset}
                        Use different log level than the default WARNING
  --basic_output BASIC_OUTPUT
                        Use basic output for logging (useful if running in jenkins). Example --basic_output=True
  --headers HEADERS     Http request headers added to all request. Example: '[{"Authorization": "SuperSecret"}, {"Auth2": "asd"}]'

```

## Usage examples:

### Installed package

Start the sample application (install the necessary packages listed in test/requirements_for_test.txt):

```shell
$ python3 test/test_application.py
```
Start the fuzzer:

```shell
$ APIFuzzer -s test/test_api/openapi_v2.json -u http://127.0.0.1:5000/ -r /tmp/reports/ --log debug 
```
Check the reports:

```shell
$ ls -1 /tmp/reports/
```
Report example:

```shell
$ json_pp < /tmp/reports/79_1573993485.5391517.json
{
   "response" : "Test application exception: invalid literal for int() with base 10: '0\\x00\\x10'",
   "sub_reports" : [],
   "parsed_status_code" : 500,
   "state" : "COMPLETED",
   "test_number" : 79,
   "request_body" : null,
   "reason" : "failed",
   "name" : "target",
   "request_url" : "http://127.0.0.1:5000/exception/0\u0000\u0010",
   "request_method" : "GET",
   "status" : "failed",
   "request_headers" : "{\"User-Agent\": \"APIFuzzer\", \"Accept-Encoding\": \"gzip, deflate\", \"Accept\": \"*/*\", \"Connection\": \"keep-alive\"}"
}
```

### Docker

#### Tested service runs on docker host

Notes 
> * Use  http://host.docker.internal instead of http://127.0.0.1 or http://localhost in the references. Read [Docker cocumentation](https://docs.docker.com/desktop/networking/#i-want-to-connect-from-a-container-to-a-service-on-the-host) for further explanation
> * You need to attach a volume like in this example to share files and folders with the container:

```shell
docker run --volume results:/results/ kisspeter/apifuzzer --src_url http://host.docker.internal:8000/openapi.json --url http://host.docker.internal:8000 --test_report /results/junit.xml --report /results/report/ ```
```
#### Tested service runs in other docker container
Notes 
> * Define `--net` at startup to attach this docker to an existing network. Read [Docker cocumentation](https://docs.docker.com/network/network-tutorial-standalone/#use-user-defined-bridge-networks) for further explanation
> * Use  http://CONTAINERNAME instead of http://127.0.0.1 or http://localhost in the references. 
> * You need to attach a volume like in this example to share files and folders with the container:

```shell
docker run --volume results:/results/ kisspeter/apifuzzer --net fastapi-performance-optimization_default kisspeter/apifuzzer --src_url http://fastapi-performance-optimization:8000/openapi.json -u http://fastapi-performance-optimization:8000 --test_report /results/junit.xml --report /results/report/```
```

[API Blueprint]: https://apiblueprint.org/
[Swagger]: http://swagger.io/
[OpenAPI]: https://swagger.io/docs/specification/about/
[GraphQL]: https://graphql.org/
 readmeEtag: '"3763bbba6ba5ba662de109a5882d73f11fdd92b9"' readmeLastModified: Tue, 16 Jul 2024 11:41:35 GMT repositoryId: 116009897 description: >- Fuzz test your application using your OpenAPI or Swagger API definition without coding created: '2018-01-02T11:49:13Z' updated: '2026-01-20T02:36:16Z' language: Python archived: false stars: 465 watchers: 7 forks: 69 owner: KissPeter logo: https://avatars.githubusercontent.com/u/19633417?v=4 license: GPL-3.0 repoEtag: '"ac39c7da14869cf027cf3672a477580c5d01553266491c699fc42bd7d3a0c72a"' repoLastModified: Tue, 20 Jan 2026 02:36:16 GMT foundInMaster: true id: bd38b4a53d7dcb21f05b494be14573e6 - source: https://openapi.tools/ name: Django REST Framework language: Python link: https://www.django-rest-framework.org/api-guide/schemas/ repository: https://github.com/encode/django-rest-framework source_description: > Automates generation of OpenAPI 3 description documents either as a static file (via CLI command) or a dynamic view within the Django REST Framework (DRF) application. v2: false v3: true repositoryMetadata: base64Readme: >- # [Django REST framework][docs]

[![build-status-image]][build-status]
[![coverage-status-image]][codecov]
[![pypi-version]][pypi]

**Awesome web-browsable Web APIs.**

Full documentation for the project is available at [https://www.django-rest-framework.org/][docs].

---

# Overview

Django REST framework is a powerful and flexible toolkit for building Web APIs.

Some reasons you might want to use REST framework:

* The Web browsable API is a huge usability win for your developers.
* [Authentication policies][authentication] including optional packages for [OAuth1a][oauth1-section] and [OAuth2][oauth2-section].
* [Serialization][serializers] that supports both [ORM][modelserializer-section] and [non-ORM][serializer-section] data sources.
* Customizable all the way down - just use [regular function-based views][functionview-section] if you don't need the [more][generic-views] [powerful][viewsets] [features][routers].
* [Extensive documentation][docs], and [great community support][group].

**Below**: *Screenshot from the browsable API*

![Screenshot][image]

----

# Requirements

* Python 3.10+
* Django 4.2, 5.0, 5.1, 5.2, 6.0

We **highly recommend** and only officially support the latest patch release of
each Python and Django series.

# Installation

Install using `pip`...

    pip install djangorestframework

Add `'rest_framework'` to your `INSTALLED_APPS` setting.

```python
INSTALLED_APPS = [
    # ...
    "rest_framework",
]
```

# Example

Let's take a look at a quick example of using REST framework to build a simple model-backed API for accessing users and groups.

Startup up a new project like so...

    pip install django
    pip install djangorestframework
    django-admin startproject example .
    ./manage.py migrate
    ./manage.py createsuperuser


Now edit the `example/urls.py` module in your project:

```python
from django.contrib.auth.models import User
from django.urls import include, path
from rest_framework import routers, serializers, viewsets


# Serializers define the API representation.
class UserSerializer(serializers.HyperlinkedModelSerializer):
    class Meta:
        model = User
        fields = ["url", "username", "email", "is_staff"]


# ViewSets define the view behavior.
class UserViewSet(viewsets.ModelViewSet):
    queryset = User.objects.all()
    serializer_class = UserSerializer


# Routers provide a way of automatically determining the URL conf.
router = routers.DefaultRouter()
router.register(r"users", UserViewSet)

# Wire up our API using automatic URL routing.
# Additionally, we include login URLs for the browsable API.
urlpatterns = [
    path("", include(router.urls)),
    path("api-auth/", include("rest_framework.urls", namespace="rest_framework")),
]
```

We'd also like to configure a couple of settings for our API.

Add the following to your `settings.py` module:

```python
INSTALLED_APPS = [
    # ... make sure to include the default installed apps here.
    "rest_framework",
]

REST_FRAMEWORK = {
    # Use Django's standard `django.contrib.auth` permissions,
    # or allow read-only access for unauthenticated users.
    "DEFAULT_PERMISSION_CLASSES": [
        "rest_framework.permissions.DjangoModelPermissionsOrAnonReadOnly",
    ]
}
```

That's it, we're done!

    ./manage.py runserver

You can now open the API in your browser at `http://127.0.0.1:8000/`, and view your new 'users' API. If you use the `Login` control in the top right corner you'll also be able to add, create and delete users from the system.

You can also interact with the API using command line tools such as [`curl`](https://curl.haxx.se/). For example, to list the users endpoint:

    $ curl -H 'Accept: application/json; indent=4' -u admin:password http://127.0.0.1:8000/users/
    [
        {
            "url": "http://127.0.0.1:8000/users/1/",
            "username": "admin",
            "email": "admin@example.com",
            "is_staff": true,
        }
    ]

Or to create a new user:

    $ curl -X POST -d username=new -d email=new@example.com -d is_staff=false -H 'Accept: application/json; indent=4' -u admin:password http://127.0.0.1:8000/users/
    {
        "url": "http://127.0.0.1:8000/users/2/",
        "username": "new",
        "email": "new@example.com",
        "is_staff": false,
    }

# Documentation & Support

Full documentation for the project is available at [https://www.django-rest-framework.org/][docs].

For questions and support, use the [REST framework discussion group][group], or `#restframework` on libera.chat IRC.

# Security

Please see the [security policy][security-policy].

[build-status-image]: https://github.com/encode/django-rest-framework/actions/workflows/main.yml/badge.svg
[build-status]: https://github.com/encode/django-rest-framework/actions/workflows/main.yml
[coverage-status-image]: https://img.shields.io/codecov/c/github/encode/django-rest-framework/main.svg
[codecov]: https://codecov.io/github/encode/django-rest-framework?branch=main
[pypi-version]: https://img.shields.io/pypi/v/djangorestframework.svg
[pypi]: https://pypi.org/project/djangorestframework/
[group]: https://groups.google.com/forum/?fromgroups#!forum/django-rest-framework

[funding]: https://fund.django-rest-framework.org/topics/funding/
[sponsors]: https://fund.django-rest-framework.org/topics/funding/#our-sponsors

[oauth1-section]: https://www.django-rest-framework.org/api-guide/authentication/#django-rest-framework-oauth
[oauth2-section]: https://www.django-rest-framework.org/api-guide/authentication/#django-oauth-toolkit
[serializer-section]: https://www.django-rest-framework.org/api-guide/serializers/#serializers
[modelserializer-section]: https://www.django-rest-framework.org/api-guide/serializers/#modelserializer
[functionview-section]: https://www.django-rest-framework.org/api-guide/views/#function-based-views
[generic-views]: https://www.django-rest-framework.org/api-guide/generic-views/
[viewsets]: https://www.django-rest-framework.org/api-guide/viewsets/
[routers]: https://www.django-rest-framework.org/api-guide/routers/
[serializers]: https://www.django-rest-framework.org/api-guide/serializers/
[authentication]: https://www.django-rest-framework.org/api-guide/authentication/
[image]: https://www.django-rest-framework.org/img/quickstart.png

[docs]: https://www.django-rest-framework.org/
[security-policy]: https://github.com/encode/django-rest-framework/security/policy
 readmeEtag: '"f875f65dd65e244aff200dfcc120d70c3fcb87b7"' readmeLastModified: Fri, 05 Dec 2025 12:01:25 GMT repositoryId: 1431547 description: Web APIs for Django. 🎸 created: '2011-03-02T17:13:56Z' updated: '2026-02-06T04:32:07Z' language: Python archived: false stars: 29857 watchers: 607 forks: 7063 owner: encode logo: https://avatars.githubusercontent.com/u/19159390?v=4 license: NOASSERTION repoEtag: '"28b7808305d74fab8ed0e721bb335c043395feade6734ac216abae4463282ae5"' repoLastModified: Fri, 06 Feb 2026 04:32:07 GMT foundInMaster: true id: fb8994c078a380cc3510b024f6ed1202 category: Server Implementations - source: https://openapi.tools/ name: RESTest category: Testing language: Java source_description: > RESTest is a framework for automated black-box testing of RESTful web APIs. It follows a model-based approach, where test cases are automatically derived from the OpenAPI description document (OAS) of the API under test. v2: true v3: true v3_1: false link: https://github.com/isa-group/RESTest repository: https://github.com/isa-group/restest repositoryMetadata: base64Readme: >- <img src="https://github.com/isa-group/RESTest/blob/master/RESTLogo_Black.png?raw=true" alt="RESTest Logo" width="200px"/>

[![Maven Central](https://maven-badges.herokuapp.com/maven-central/es.us.isa/restest/badge.svg)](https://maven-badges.herokuapp.com/maven-central/es.us.isa/restest)
[![javadoc](https://javadoc.io/badge2/es.us.isa/restest/javadoc.svg)](https://javadoc.io/doc/es.us.isa/restest)
<!--
[![Build Status](https://circleci.com/gh/isa-group/RESTest.svg?style=svg)](https://circleci.com/gh/isa-group/RESTest)
[![Test coverage](https://sonarcloud.io/api/project_badges/measure?project=isa-group_RESTest&metric=coverage)](https://sonarcloud.io/component_measures?id=isa-group_RESTest&metric=Coverage)
[![Maintainability](https://sonarcloud.io/api/project_badges/measure?project=isa-group_RESTest&metric=sqale_rating)](https://sonarcloud.io/component_measures?id=isa-group_RESTest&metric=Maintainability)
[![Reliability](https://sonarcloud.io/api/project_badges/measure?project=isa-group_RESTest&metric=reliability_rating)](https://sonarcloud.io/component_measures?id=isa-group_RESTest&metric=Reliability)
[![Security](https://sonarcloud.io/api/project_badges/measure?project=isa-group_RESTest&metric=security_rating)](https://sonarcloud.io/component_measures?id=isa-group_RESTest&metric=Security)
-->

RESTest is a framework for automated black-box testing of RESTful web APIs. It follows a model-based approach, where test cases are automatically derived from the [OpenAPI Specification (OAS)](https://www.openapis.org/) of the API under test. No access to the source code is required, which makes it possible to test APIs written in any programming language, running in local or remote servers.

## Index
1. [RESTest Wiki](https://github.com/isa-group/RESTest#restest-wiki)

2. [Quick examples](https://github.com/isa-group/RESTest#quick-examples)
   1. [Using the system programmatically](https://github.com/isa-group/RESTest#using-the-system-programmatically)
      1. [Generating test cases](https://github.com/isa-group/RESTest#generating-test-cases)
      2. [Generating and running test cases](https://github.com/isa-group/RESTest#generating-and-running-test-cases)
      3. [Generating and running test cases](https://github.com/isa-group/RESTest#running-test-cases)
   2. [Using the system via the command-line interface](https://github.com/isa-group/RESTest#using-the-system-via-the-command-line-interface)
      1. [Generate, Execute, and Generate Reports based on OpenAPI specification](https://github.com/isa-group/RESTest#generate,-execute,-and-generate-reports-based-on-openapi-specification)
      2. [Create a Test Configuration File](https://github.com/isa-group/RESTest#create-a-test-configuration-file)
      3. [Generate Test Cases](https://github.com/isa-group/RESTest#generate-test-cases)
      4. [Execute Test Cases](https://github.com/isa-group/RESTest#execute-test-cases)
      5. [Generate and Execute Test Cases](https://github.com/isa-group/RESTest#generate-and-execute-test-cases)
3. [How does it work?](https://github.com/isa-group/RESTest#how-does-it-work)
4. [What can I do with RESTest?](https://github.com/isa-group/RESTest#what-can-i-do-with-restest)
5. [Running RESTest as a JAR](https://github.com/isa-group/RESTest#running-restest-as-a-jar)
   1. [Option 1: Build RESTest from source](https://github.com/isa-group/RESTest#option-1-build-restest-from-source)
   2. [Option 2: Download the latest release](https://github.com/isa-group/RESTest#option-2-download-the-latest-release)
6. [Citing RESTest](https://github.com/isa-group/RESTest#citing-restest)
7. [License](https://github.com/isa-group/RESTest#license)
   1. [Icon credits](https://github.com/isa-group/RESTest#icon-credits)

## RESTest Wiki
In this page you can find a brief description of how RESTest works and an illustrating example. If you want to read the full documentation, please visit the [Wiki](https://github.com/isa-group/RESTest/wiki).




## Quick examples:

### Using the system programmatically

#### Generating test cases 

In this example, we illustrate how to generate a set of test cases using the Constraint-Based test case generator and write them to a file using the RESTAssured writer. Test cases are generated by randomly selecting values for each input parameter, ensuring that dependencies between parameters are met (or intentionally violated). This is made possible through the use of a constraint solver (Choco) and IDL, a domain-specific language for specifying dependencies between parameters as part of the OAS specification.

The complete code of the example is provided above.

```java 
public class Ex3_CBTGeneration {

    public static String propertyFilePath="src/main/resources/Examples/Ex3_CBTGeneration/user_config.properties"; 		// Path to user properties file with configuration options

    public static void main(String[] args) throws RESTestException {
        // Load properties
        RESTestLoader loader = new RESTestLoader(propertyFilePath);

        // Create test case generator
        ConstraintBasedTestCaseGenerator generator = (ConstraintBasedTestCaseGenerator) loader.createGenerator();
        Collection<TestCase> testCases = generator.generate();

        // Create target directory for test cases if it does not exist
        createDir(loader.getTargetDirJava());

        // Write (RestAssured) test cases
        RESTAssuredWriter writer = (RESTAssuredWriter) loader.createWriter();
        writer.write(testCases);

        System.out.println(testCases.size() + " test cases generated and written to " + loader.getTargetDirJava());
    }
}

```

This code demonstrates how to load the properties, create a constraint-based test case generator, generate test cases, and write them using the RESTAssured writer. Now, let's examine one of the test cases generated using RESTAssured.
```java

import io.restassured.RestAssured;
import io.restassured.response.Response;
import org.junit.Test;
import static org.junit.Assert.fail;

public class Ex3_CBTGeneration {

   // ... Previous code ...

   @Test
   public void test_qec4n4cuyvcj_getMultiHotelOffers() {
      String testResultId = "test_qec4n4cuyvcj_getMultiHotelOffers";

      // Filter settings for specific test case
      nominalOrFaultyTestCaseFilter.updateFaultyData(true, true, "individual_parameter_constraint:Violated 'min' constraint of integer parameter radius");
      statusCode5XXFilter.updateFaultyData(true, true, "individual_parameter_constraint:Violated 'min' constraint of integer parameter radius");
      csvFilter.setTestResultId(testResultId);
      statusCode5XXFilter.setTestResultId(testResultId);
      nominalOrFaultyTestCaseFilter.setTestResultId(testResultId);
      validationFilter.setTestResultId(testResultId);

      try {
         // Build and send request using RestAssured
         Response response = RestAssured
                 .given()
                 .queryParam("page[limit]", "25")
                 .queryParam("amenities", "BABY-SITTING,AIR_CONDITIONING")
                 .queryParam("hotelIds", "HSFELAAF")
                 .queryParam("roomQuantity", "5")
                 .queryParam("chains", "WV,EC,WW")
                 .queryParam("includeClosed", "false")
                 .queryParam("radius", "-3")
                 .queryParam("radiusUnit", "MILE")
                 .queryParam("checkInDate", "2023-11-14")
                 .queryParam("rateCodes", "WKD,STP,TUR")
                 .queryParam("paymentPolicy", "DEPOSIT")
                 .filter(allureFilter)
                 .filter(statusCode5XXFilter)
                 .filter(nominalOrFaultyTestCaseFilter)
                 .filter(validationFilter)
                 .filter(csvFilter)
                 .when()
                 .get("/shopping/hotel-offers");

         // Verify the response
         response.then();
         System.out.println("El caso de prueba pasó exitosamente.");
      } catch (RuntimeException ex) {
         System.err.println(ex.getMessage());
         fail(ex.getMessage());
      }
   }
}
```

#### Running test cases 
This example will illustrate how to execute a set of tests that have been previously generated using ResTestExecutor. Below is an example of how this process would be carried out.
```
public class Ex8_TestExecution {


    public static final String PROPERTY_FILE_PATH = "src/main/resources/Examples/Ex8_TestExecution/user_config.properties"; 		// Path to user properties file with configuration options

    public static void main(String[] args) throws RESTestException {

        // Execute tests

        RESTestExecutor executor = new RESTestExecutor(PROPERTY_FILE_PATH);
        executor.execute();


    }
}
```


#### Generating and running test cases
In this example, we will explore the generation and execution of test cases using RESTestRunner, a tool that enables the creation of an automated workflow to manage the entire testing process. Below is an example illustrating how this workflow would be executed.

```java

public class Ex9_Generation_Execution {

   public static final String PROPERTY_FILE_PATH = "src/main/resources/Examples/Ex9_Generation_Execution/user_config.properties"; 		// Path to user properties file with configuration options

   public static final Logger logger = Logger.getLogger(Ex9_Generation_Execution.class.getName());

   public static void main(String[] args) throws RESTestException {
      // Load properties
      RESTestRunner runner = new RESTestRunner(PROPERTY_FILE_PATH);

      // Run workflow
      runner.run();

      if (logger.isLoggable(java.util.logging.Level.INFO)) {
         String message1 = String.format("%d test cases generated and written to %s", runner.getNumberOfTestCases(), runner.getTargetDirJava());
         String message2 = String.format("Allure report available at %s", runner.getAllureReportsPath());
         String message3 = String.format("CSV stats available at %s/%s", PropertyManager.readProperty("data.tests.dir"), runner.getExperimentName());
         String message4 = String.format("Coverage report available at %s/%s", PropertyManager.readProperty("data.coverage.dir"), runner.getExperimentName());
         logger.info(message1);
         logger.info(message2);
         logger.info(message3);
         logger.info(message4);
      }

   }
}


```

Upon completion, the program will display detailed information in the console. This will include the number of generated test cases, the location of Allure reports, CSV statistics, and the coverage report, providing a comprehensive overview of the results obtained during the execution.

Finally, test failures are collected and they can be easily spotted and analyzed in a user-friendly GUI, built with [Allure](http://allure.qatools.ru/). To do so, open the file `src/main/resources/Examples/Ex8_Generation_Execution/allure_report/anApiOfIceAndFire/index.html` in your browser:

![Allure](docs/img.png)

### Using the system via the command-line interface

The RESTest CLI supports various options for interacting with the system. Here are the available options: 

* **-h, --help:** Display help information about available options and usage. 
* **-o, --openapi [path]:** Generates and executes a series of tests based on an OpenAPI file. 
* **-c, --create-conf [path]:** Generate a test configuration file from the provided OpenAPI file. 
* **-g, --generate [path]:** Generate test cases based on the provided configuration file. 
* **-e, --execute [path]:** Execute the test cases defined in the provided configuration file. 

Here are some quick examples of using the RESTest CLI: 

#### Generate, Execute, and Generate Reports based on OpenAPI specification: 

```
java -jar .\target\restest-cli.jar -o [path_to_oas_file] 
```
  

#### Create a Test Configuration File: 

```
java -jar .\target\restest-cli.jar -c [path_to_oas_file] 
```
  

#### Generate Test Cases: 

```
java -jar .\target\restest-cli.jar -g [path_to_properties_file] 
```
  

#### Execute Test Cases: 

```
java -jar .\target\restest-cli.jar -e [path_to_properties_file] 
```
  

#### Generate and Execute Test Cases: 

```
java -jar .\target\restest-cli.jar -g -e [path_to_properties_file]
```


## Detailed example

In this section, we will delve into a specific example to illustrate in detail how RESTEST operates. Throughout the following steps, we will break down a practical scenario, providing step-by-step explanations of how the system functions.

In this example, we will conduct tests on the Ice and Fire API using RESTest. To perform these tests, we will rely on the OpenAPI specification of the API, which is available at the following [link](src/main/resources/Examples/Ex10_Iterative_Generation_Execution/spec_iceandfire.yaml).

In the next step, we will proceed to obtain the configuration file that will encompass all the essential settings for our tests. This file plays a fundamental role as it sets key parameters, such as the types of generators to use, specific configurations for the Ice and Fire API, and any other relevant information for the successful execution of tests with RESTest.
This is the configuration file for this example:

```yaml
---
auth:
  required: true
  queryParams: {}
  headerParams: {}
  apiKeysPath: null
  headersPath: null
testConfiguration:
  operations:
  - testPath: /api
    operationId: getRoot
    method: get
    testParameters: null
    expectedResponse: 200
  - testPath: "/api/books/{book-id}"
    operationId: findBookById
    method: get
    testParameters:
    - name: book-id
      in: path
      weight: null
      generators:
      - type: RandomNumber
        genParameters:
        - name: type
          values:
          - integer
          objectValues: null
        - name: min
          values:
          - 1
          objectValues: null
        - name: max
          values:
          - 100
          objectValues: null
        valid: true
    expectedResponse: 200
  - testPath: "/api/characters/{character-id}"
    operationId: findCharacterById
    method: get
    testParameters:
    - name: character-id
      in: path
      weight: null
      generators:
      - type: RandomNumber
        genParameters:
        - name: type
          values:
          - integer
          objectValues: null
        - name: min
          values:
          - 1
          objectValues: null
        - name: max
          values:
          - 100
          objectValues: null
        valid: true
    expectedResponse: 200
  - testPath: "/api/houses/{house-id}"
    operationId: findHouseById
    method: get
    testParameters:
    - name: house-id
      in: path
      weight: null
      generators:
      - type: RandomNumber
        genParameters:
        - name: type
          values:
          - integer
          objectValues: null
        - name: min
          values:
          - 1
          objectValues: null
        - name: max
          values:
          - 100
          objectValues: null
        valid: true
    expectedResponse: 200

```


The following [link](src/main/java/es/us/isa/restest/examples/Ex2_CreateTestConf.java) provides an example of how we can obtain this configuration file.

Once we have both files, the OAS specification, and the configuration file, we will create a new .properties file in which we will specify the path where these files are located, along with other configurations for the generation and execution of tests. This file looks like the following:
````java

# OAS specification
oas.path=src/main/resources/Examples/Ex9_Iterative_Generation_Execution/spec_iceandfire.yaml

# Test configuration file
conf.path=src/main/resources/Examples/Ex9_Iterative_Generation_Execution/test_conf.yaml

# Directory where the test cases will be generated
test.target.dir=src/generation/java/anApiOfIceAndFire

# Package name
test.target.package=anApiOfIceAndFire

# Experiment name (for naming related folders and files)
experiment.name=anApiOfIceAndFire

# Set true for running the generated test cases
experiment.execute=true

# Name of the test class to be generated
testclass.name=AnApiOfIceAndFireTest

# Ratio of faulty test cases to be generated doe to broken individual dependencies (ex. missing required parameter). Helpful for negative testing
faulty.ratio=0.05

# Number of test cases to be generated per operation
testsperoperation=5

# Maximum number of test cases to be generated (approximately)
numtotaltestcases=100

# Delay between requests in seconds (-1 for no delay)
delay=30

# Set to true for validating the test cases with OASValidator before executing them
testcases.check=true


# =================================
# CONSTRAINT-BASED TESTING (CBT)
# =================================

# Test case generator
generator=CBT

# Ratio of faulty test cases to generate due to the violation of inter-parameter dependencies
faulty.dependency.ratio=0.4

# Number of requests using the same randomly generated input data
reloadinputdataevery=10

# Number of values used for each parameter when reloading input data
inputdatamaxvalues=100

# =================
# ALLURE REPORTS
# =================

# Set to true for generating Allure reports
allure.report=true

# Path to the directory where Allure will save the report
allure.report.dir=src/main/resources/Examples/Ex9_Iterative_Generation_Execution/allure_report

# ==================
# CSV STATS REPORTS
# ==================

# Set to true for generating CSV stats
stats.csv=true

# ===================
# COVERAGE REPORTS
# ===================

# Set to true for computing input coverage
coverage.input=true

# Set to true for computing output coverage
coverage.output=true

# Path to the directory where coverage results will be saved
data.coverage.dir=target/coverage-data


````

Once we have this file, we will be ready to carry out the generation and execution of our tests. To achieve this, we will use the following example as a guide:

```java

public class Ex9_Generation_Execution {

   public static final String PROPERTY_FILE_PATH = "src/main/resources/Examples/Ex9_Generation_Execution/user_config.properties"; 		// Path to user properties file with configuration options

   public static final Logger logger = Logger.getLogger(Ex9_Generation_Execution.class.getName());

   public static void main(String[] args) throws RESTestException {
      // Load properties
      RESTestRunner runner = new RESTestRunner(PROPERTY_FILE_PATH);

      // Run workflow
      runner.run();

      if (logger.isLoggable(java.util.logging.Level.INFO)) {
         String message1 = String.format("%d test cases generated and written to %s", runner.getNumberOfTestCases(), runner.getTargetDirJava());
         String message2 = String.format("Allure report available at %s", runner.getAllureReportsPath());
         String message3 = String.format("CSV stats available at %s/%s", PropertyManager.readProperty("data.tests.dir"), runner.getExperimentName());
         String message4 = String.format("Coverage report available at %s/%s", PropertyManager.readProperty("data.coverage.dir"), runner.getExperimentName());
         logger.info(message1);
         logger.info(message2);
         logger.info(message3);
         logger.info(message4);
      }

   }
}

```

As we can see, by providing the path where the .properties file is located, the system will create a workflow that will perform the generation of test cases, their execution, and the generation of reports and summaries about the tests. Additionally, after executing the workflow, the following information will be printed on the screen:

* Number of test cases generated and written to a specific location.
* Location of the Allure report.
* CSV stats available at a specific path.
* Coverage report available at another specific path.

These details provide a quick and useful insight into the success and results of the conducted tests, facilitating the tracking and evaluation of the performance of the generated test cases.

## How does it work?
The figure below shows how RESTest works:

1. **Test model generation**: RESTest takes as input the OAS specification of the API under test, considered the *system model*. A [*test model*](https://github.com/isa-group/RESTest/wiki/Test-configuration-files) is automatically generated from the system model including test-specific configuration data. The default test model can be manually enriched with fine-grained configuration details such as test data generation settings.

2. **Abstract test case generation**: The system and the test models drive the generation of abstract test cases following user-defined test case generation strategies such as random testing. If the API under test contains [inter-parameter dependencies](https://github.com/isa-group/RESTest/wiki/Inter-parameter-dependencies), then constraint-based testing can be applied, specifying the dependencies in the OAS specification using the IDL4OAS extension (see examples [here](https://github.com/isa-group/IDLReasoner/blob/master/src/test/resources/OAS_example.yaml#L45) and [here](https://github.com/isa-group/IDLReasoner/tree/master/src/test/resources)). Requests satisfying all inter-parameter dependencies are automatically generated thanks to [IDLReasoner](https://github.com/isa-group/IDLReasoner).

3. **Test case generation**: The abstract test cases are instantiated into a specific programming language or testing framework using a [test writer](https://github.com/isa-group/RESTest/wiki/Test-writers). RESTest currently supports the generation of [REST Assured](http://rest-assured.io/) and [Postman](https://www.postman.com/) test cases.

4. **Test case execution**: The test cases are executed and a set of reports and stats are generated. Stats are machine-readable, and the test reports can be graphically visualized thanks to [Allure](http://allure.qatools.ru/).

5. **Feedback collection**: [Test case generators](https://github.com/isa-group/RESTest/wiki/Test-case-generators) and other components can react to the test outputs (i.e., the stats generated in the previous step) to create more sophisticated test cases. Examples of this are the stateful data generators (e.g., the [BodyGenerator](https://github.com/isa-group/RESTest/blob/master/src/main/java/es/us/isa/restest/inputs/stateful/BodyGenerator.java)) and the [StatsReportManager](https://github.com/isa-group/RESTest/blob/master/src/main/java/es/us/isa/restest/reporting/StatsReportManager.java), both of which generate new test data based on previous API responses.

![RESTest](docs/RESTest_v3.png)

## What can I do with RESTest?
Check out the following demo video, where we discuss some of the things that you can do with RESTest, both from the user and the developer point of view. The showcase shown in the video is available at http://betty.us.es/restest-showcase-demo/.

<a href="https://youtu.be/TnGkwMDBDt4" target="_blank"><img src="docs/play_video.png" alt="RESTest demo video" width="300" /></a>

## Running RESTest as a JAR
Instead of from an IDE like IntelliJ IDEA, you can also run RESTest as a fat JAR. You have two options:

### Option 1: Build RESTest from source

To package RESTest as a fat JAR file, run the following command in the root directory:

```
mvn clean install -DskipTests
```

Then, run the JAR file passing as argument the path to the properties file, for example:

```
java -jar target/restest.jar src/test/resources/Restcountries/restcountries_demo.properties
```

### Option 2: Download the latest release

Go to the [releases page](https://github.com/isa-group/RESTest/releases) and download the latest one. RESTest releases consist of ZIP files which, once uncompressed, provide the directory structure and the necessary resources to run RESTest as a JAR. You can test the same example shown in the quickstart guide by running the following command:

```
java -jar restest.jar src/test/resources/Folder/api.properties
```

## Citing RESTest

If you want to cite RESTest in your research, please use the BibTeX entry below. [Here's a link to a preprint of the paper](https://www.researchgate.net/publication/352835570_RESTest_Automated_Black-Box_Testing_of_RESTful_Web_APIs).


```bibtex
@inproceedings{MartinLopez2021Restest,
	title= {{RESTest: Automated Black-Box Testing of RESTful Web APIs}},
	author= {Alberto Martin-Lopez and Sergio Segura and Antonio Ruiz-Cort\'{e}s},
	booktitle= {Proceedings of the 30th ACM SIGSOFT International Symposium on Software Testing and Analysis},
	series= {ISSTA '21},
	publisher= {Association for Computing Machinery},
	year= {2021}
}
```

## License
RESTest is distributed under the [GNU Lesser General Public License v3.0](LICENSE).

RESTest includes Allure Framework &copy; 2019 Qameta Software OÜ. It is used under the terms of the Apache 2.0 License, which can be obtained from http://www.apache.org/licenses/LICENSE-2.0.

### Icon credits
This README and some pages of the Wiki use icons provided by [Freepik](https://www.flaticon.com/authors/freepik), available at [Flaticon](https://www.flaticon.com/).
 readmeEtag: '"b09720e784cd44db9643aeb2c84cdff066cd0ce3"' readmeLastModified: Fri, 21 Nov 2025 15:02:25 GMT repositoryId: 120752811 description: 'RESTest: Automated Black-Box Testing of RESTful Web APIs' created: '2018-02-08T11:27:45Z' updated: '2026-01-27T21:33:06Z' language: Java archived: false stars: 228 watchers: 15 forks: 37 owner: isa-group logo: https://avatars.githubusercontent.com/u/2708867?v=4 license: LGPL-3.0 repoEtag: '"aa92ff2525218ec50daa6291be9576ea788432b9a087e9669042bbc35a34fcb1"' repoLastModified: Tue, 27 Jan 2026 21:33:06 GMT foundInMaster: true id: 9115f26fa9e7b17b844a0f10b9814db7 - source: https://openapi.tools/ name: ReadyAPI category: Testing language: Java source_description: > an end to end API functional, security, performance and virtualization tool where OAS description documents can be utilized to automate the creation and validation of end to end tests, running them manually or at any point in your CI/CD pipeline. pipelines. v2: true v3: true v3_1: false link: https://smartbear.com/product/ready-api/overview/ foundInMaster: true id: 2e4959c1f56a5c235f1e82b2d4cbc505 - source: https://openapi.tools/ name: DeveloperHub category: Documentation link: https://developerhub.io language: SaaS source_description: Collaboration platform for product and API documentation v2: true v3: true v3_1: true foundInMaster: true id: 9c4871865101e2bb2ee5a4e48084b75e - source: https://openapi.tools/ name: OAuth2 as OpenAPI Spec 3.0 components language: Any repository: https://github.com/ybelenko/oauth2_as_oas3_components source_description: >- OAuth2 token endpoint described with OAS3 schema. All grants documented. Can be installed as NPM or Composer package. v2: false v3: true repositoryMetadata: base64Readme: >- IyBPQXV0aDIgYXMgT3BlbkFQSSBTcGVjIDMuMCBjb21wb25lbnRzCgpUaGUgZXhhbXBsZSBmaWxlIHdoaWNoIGRlc2NyaWJlcyBPQXV0aDIgdG9rZW4gZW5kcG9pbnRzIGxvY2F0ZWQgW2Rpc3Qvb2F1dGgyX2VuZHBvaW50cy55bWxdKGRpc3Qvb2F1dGgyX2VuZHBvaW50cy55bWwpLgoKIyMgV2h5IHRoaXMgcGFja2FnZSBleGlzdHMKU2luY2UgW1JGQyA2NzQ5IE9BdXRoMl0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzY3NDkpIHNlcnZlciBpbXBsZW1lbnRhdGlvbiBtYXkgYmUgdmVyeSBkaWZmZXJlbnQob3B0aW9uYWwvcmVjb21tZW5kZWQgcmVzcG9uc2UgZmllbGRzLCBleHRlbmRlZCBncmFudCkgaXQgbWlnaHQgbWUgdXNlZnVsIHRvIGRlc2NyaWJlIHlvdXIgdW5pcXVlIGltcGxlbWVudGF0aW9uIHdpdGhpbiB5b3VyIE9BUzMgZmlsZS4gSXQncyBhbHNvIHZlcnkgaGFuZHkgdG8gc2VlIGV4YW1wbGVzIG9mIHlvdXIgdG9rZW4gYW5kIGVycm9yIHJlc3BvbnNlLCBiZWNhdXNlIFJGQzY3NDkgaXMgYSB0ZXh0IGRvY3VtZW50IHdpdGhvdXQgYW55IHBpY3R1cmVzIG9yIGdyYXBocy4gVGhlIGV4YW1wbGUgZmlsZSBtZW50aW9uZWQgYmVmb3JlIGNvbnRhaW5zIGRlc2NyaXB0aW9uIG9mIHRva2VuIGVuZHBvaW50cyBmb3IgZWFjaCBhdXRob3JpemF0aW9uIGdyYW50LCBjb25zaWRlciBpdCBhcyBzdGFydGluZyBwb2ludC4KCkV4YW1wbGUgZmlsZSBvbWl0cyBbYXV0aG9yaXphdGlvbiBlbmRwb2ludF0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzY3NDkjc2VjdGlvbi0zLjEpIGVuZHBvaW50IG9uIHB1cnBvc2UuIEkgZG9uJ3Qga25vdyBob3cgdG8gZGVzY3JpYmUgaXQgd2l0aCBPQVMzIHNpbmNlIGVuZHBvaW50IHJlc3BvbnNlIGlzbid0IEpTT04oaHRtbCBwYWdlKS4gSWYgeW91IGhhdmUgYW55IHN1Z2dlc3Rpb24gcGxlYXNlIHN1Ym1pdCBhbiBpc3N1ZSB0byB0aGlzIHJlcG8uCgpTaW5jZSBbUkZDIDY3NDkgLSBUaGUgT0F1dGgyLjAgQXV0aG9yaXphdGlvbiBGcmFtZXdvcmsgLSAyLjMuMS4gQ2xpZW50IFBhc3N3b3JkXShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNjc0OSNzZWN0aW9uLTIuMy4xKSBkb2Vzbid0IHJlY29tbWVuZCB0byBzZW5kIGNsaWVudCBwYXNzd29yZCBpbiByZXF1ZXN0IGJvZHkgdGhlbiBvdXIgZXhhbXBsZSBleHBlY3RzIGJhc2ljIGF1dGhvcml6YXRpb24gaW4gYWxsIGVuZHBvaW50cy4KCiMjIEluc3RhbGxhdGlvbgoKIyMjIENvcHkgUGFzdGUKU2luY2UgaXQncyBub3QgYWN0dWFsbHkgYSBjb2RlLCBidXQgbWFya3VwIHlvdSBjYW4ganVzdCBjb3B5IGFueXRoaW5nIHlvdSB3YW50IGZyb20gW2Rpc3Qvb2F1dGgyX2VuZHBvaW50cy55bWxdKGRpc3Qvb2F1dGgyX2VuZHBvaW50cy55bWwpLgoKVGhlcmUgaXMgYWxzbyBlbmhhbmNlZCBleGFtcGxlIHdpdGggcG9seW1vcnBoaXNtIGF0IFtkaXN0L29hdXRoMl9lbmRwb2ludHNfcG9seW1vcnBoaXNtLnltbF0oZGlzdC9vYXV0aDJfZW5kcG9pbnRzX3BvbHltb3JwaGlzbS55bWwpLiAqKlBsZWFzZSwgbWFrZSBzdXJlIHRoYXQgeW91ciB0b29scyBzdXBwb3J0cyBwb2x5bW9ycGhpc20uIFRoaXMgZXhhbXBsZSBjb250YWlucyBuZXcgZmVhdHVyZXMgbm90IGhpZ2hseSBhZG9wdGVkIHlldC4gQ2hlY2sgYXQgbGVhc3QgYG9uZU9mYCBwcm9wZXJ0eSBzdXBwb3J0LiBJIHdvdWxkbid0IHVzZSBpdCBteXNlbGYgYmVjYXVzZSBtb3N0IG9mIHRoZSBPcGVuQVBJIHRvb2xzIGhhdmUgaXNzdWVzIHdpdGggcG9seW1vcnBoaXNtIG5vdywgYnV0IHRoaXMgZXhhbXBsZSBtYXkgYmUgdXNlZnVsIGluIGZvcnNlZWFibGUgZnV0dXJlLioqCgojIyMgQ29tcG9zZXIKSW5zdGFsbCBbQ29tcG9zZXIgLSBEZXBlbmRlbmN5IE1hbmFnZXIgZm9yIFBIUF0oaHR0cHM6Ly9nZXRjb21wb3Nlci5vcmcvZG93bmxvYWQvKQoKVGhlbiBydW4gaW4gdGVybWluYWw6CmBgYGNvbnNvbGUKY29tcG9zZXIgcmVxdWlyZSB5YmVsZW5rby9vYXV0aDJfYXNfb2FzM19jb21wb25lbnRzCmBgYAoKVXNlIHByb3ZpZGVkIGNvbXBvbmVudHMgdmlhIGAkcmVmYCBhdHRyaWJ1dGUgbGlrZToKCmBgYHlhbWwKcGF0aHM6CiAgL3Rva2VuOgogICAgcG9zdDoKICAgICAgc3VtbWFyeTogT2J0YWluIGFjY2VzcyB0b2tlbiB3aXRoICJhdXRob3JpemF0aW9uX2NvZGUiIGdyYW50LgogICAgICByZXF1ZXN0Qm9keToKICAgICAgICAkcmVmOiAnLi92ZW5kb3IveWJlbGVua28vb2F1dGgyX2FzX29hczNfY29tcG9uZW50cy9kaXN0L2NvbXBvbmVudHMvcmVxdWVzdEJvZGllcy9Ub2tlblJlcXVlc3RDb2RlR3JhbnQueW1sJwogICAgICByZXNwb25zZXM6CiAgICAgICAgJzIwMCc6CiAgICAgICAgICAkcmVmOiAnLi92ZW5kb3IveWJlbGVua28vb2F1dGgyX2FzX29hczNfY29tcG9uZW50cy9kaXN0L2NvbXBvbmVudHMvcmVzcG9uc2VzL09BdXRoMlRva2VuU3VjY2Vzc1Jlc3BvbnNlLnltbCcKICAgICAgICAnNFhYJzoKICAgICAgICAgICRyZWY6ICcuL3ZlbmRvci95YmVsZW5rby9vYXV0aDJfYXNfb2FzM19jb21wb25lbnRzL2Rpc3QvY29tcG9uZW50cy9yZXNwb25zZXMvT0F1dGgyVG9rZW5FcnJvclJlc3BvbnNlLnltbCcKYGBgCgpFeHRlbmRlZCBleGFtcGxlIHdpdGggcmVmcyBbZGlzdC9vYXV0aDJfZW5kcG9pbnRzX3dpdGhfcmVmcy55bWxdKGRpc3Qvb2F1dGgyX2VuZHBvaW50c193aXRoX3JlZnMueW1sKQoKIyMjIE5QTQpbSW5zdGFsbCBOUE0gYW5kIE5vZGUuanNdKGh0dHBzOi8vZG9jcy5ucG1qcy5jb20vZG93bmxvYWRpbmctYW5kLWluc3RhbGxpbmctbm9kZS1qcy1hbmQtbnBtKQoKVGhlbiBydW4gaW4gdGVybWluYWw6CmBgYGNvbnNvbGUKbnBtIGkgLS1zYXZlIG9hdXRoMl9hc19vYXMzX2NvbXBvbmVudHMKYGBgCgpVc2UgcHJvdmlkZWQgY29tcG9uZW50cyB2aWEgYCRyZWZgIGF0dHJpYnV0ZSBsaWtlOgoKYGBgeWFtbApwYXRoczoKICAvdG9rZW46CiAgICBwb3N0OgogICAgICBzdW1tYXJ5OiBPYnRhaW4gYWNjZXNzIHRva2VuIHdpdGggImF1dGhvcml6YXRpb25fY29kZSIgZ3JhbnQuCiAgICAgIHJlcXVlc3RCb2R5OgogICAgICAgICRyZWY6ICcuL25vZGVfbW9kdWxlcy9vYXV0aDJfYXNfb2FzM19jb21wb25lbnRzL2Rpc3QvY29tcG9uZW50cy9yZXF1ZXN0Qm9kaWVzL1Rva2VuUmVxdWVzdENvZGVHcmFudC55bWwnCiAgICAgIHJlc3BvbnNlczoKICAgICAgICAnMjAwJzoKICAgICAgICAgICRyZWY6ICcuL25vZGVfbW9kdWxlcy9vYXV0aDJfYXNfb2FzM19jb21wb25lbnRzL2Rpc3QvY29tcG9uZW50cy9yZXNwb25zZXMvT0F1dGgyVG9rZW5TdWNjZXNzUmVzcG9uc2UueW1sJwogICAgICAgICc0WFgnOgogICAgICAgICAgJHJlZjogJy4vbm9kZV9tb2R1bGVzL29hdXRoMl9hc19vYXMzX2NvbXBvbmVudHMvZGlzdC9jb21wb25lbnRzL3Jlc3BvbnNlcy9PQXV0aDJUb2tlbkVycm9yUmVzcG9uc2UueW1sJwpgYGAKCkV4dGVuZGVkIGV4YW1wbGUgd2l0aCByZWZzIFtkaXN0L29hdXRoMl9lbmRwb2ludHNfd2l0aF9yZWZzLnltbF0oZGlzdC9vYXV0aDJfZW5kcG9pbnRzX3dpdGhfcmVmcy55bWwpCgojIyBDb250cmlidXRpbmcKCklmIHlvdSBoYXZlIGFueSBzdWdnZXN0aW9ucyBwbGVhc2Ugc3VibWl0IGFuIGlzc3VlLgoKIyMgTGljZW5zZQpbTUlUIExpY2Vuc2VdKExJQ0VOU0UpCg== readmeEtag: '"6c8026d25e06120124e8248435017d4bb49efc43"' readmeLastModified: Tue, 20 Apr 2021 20:45:12 GMT repositoryId: 358547893 description: OAuth2 definitions as OpenAPI Spec 3.0 components created: '2021-04-16T09:35:37Z' updated: '2025-09-09T01:57:05Z' language: null archived: false stars: 9 watchers: 2 forks: 1 owner: ybelenko logo: https://avatars.githubusercontent.com/u/5541023?v=4 license: MIT repoEtag: '"aefdc7f7ef586b8ac321161e8314413e23d909d5eef5df2aeb5ed2916c4de4e6"' repoLastModified: Tue, 09 Sep 2025 01:57:05 GMT foundInMaster: true id: 943b84998524f65e45768ca73ecb3841 category: Parsers - source: https://openapi.tools/ name: Mayhem for API category: - Testing - Data Validators - Security - Server Implementations link: https://forallsecure.com/mayhem-for-api repository: https://github.com/forallsecure/mapi-action language: Any source_description: >- Probe your REST API with an infinite stream of test cases generated automatically from your OpenAPI specification. v2: true v3: true id: ce2ef79e2c9f1a119ca823d0afd8258b repositoryMetadata: base64Readme: >- # Mayhem for API GitHub Action

[![Mayhem for API](./imgs/mapi-logo-full-color.svg)](https://www.mayhem.security/api-security)

A GitHub Action for using Mayhem for API to check for reliability,
performance and security issues in your APIs.

## About Mayhem for API

🧪 Modern App Testing: Mayhem for API is a dynamic testing tool that
catches reliability, performance and security bugs before they hit
production.

🧑‍💻 For Developers, by developers: The engineers building
software are the best equipped to fix bugs, including security bugs. As
engineers ourselves, we're building tools that we wish existed to make
our job easier!

🤖 Simple to Automate in CI: Tests belong in CI, running on every commit
and PRs. We make it easy, and provide results right in your PRs where
you want them. Adding Mayhem for API to a DevOps pipeline is easy.

Want to try it? [Sign up for free](https://www.mayhem.security/api-security) today!

## Usage

1. Get a Mayhem for API token

    a. [Sign up for Mayhem for free and install `mapi`](https://app.mayhem.security)

    b. Create an [API token](https://app.mayhem.security/docs/api-testing/getting-set-up/generating-api-tokens/)

    c. Add your API token with name `MAYHEM_TOKEN` on your repository's GitHub page at
       `Settings` → `Secrets` → `New repository secret`

2. Create a file in _your_ GitHub repository at:
```
    .github/workflows/ForAllSecure-Mayhem-for-API.yml
```

3. Add the following text to the file and tweak it for your codebase.

    **Note**: To auto-detect diffs, Mayhem needs a deeper repository clone than the default of actions/checkout@v3.  Set `fetch-depth` to 0 for a full clone, or deeper clones to fetch enough commit history to compute a merge base for the branch.

```yaml
name: Mayhem for API
on:
    push:
    pull_request:
jobs:
  security:
    runs-on: ubuntu-latest
    steps:
    - uses: actions/checkout@v3
      with:
        # fetch entire history to compute diffs between jobs
        fetch-depth: 0

    - name: Start your API
      run: ./run_your_api.sh & # <----------------------------------- UPDATE THIS

    - name: Run Mayhem for API to check for vulnerabilities
      uses: ForAllSecure/mapi-action@v2
      with:
        mayhem-token: ${{ secrets.MAYHEM_TOKEN }}
        api-url: http://localhost:8000 # <--------------------------- UPDATE THIS
        api-spec: your-openapi-spec-or-postman-collection.json # <--- UPDATE THIS
```

4. Commit the new file and push it up to GitHub

5. Your new Mayhem for API action will be visible at

    `https://github.com/<USERNAME>/<REPO_NAME>/actions`

---

This repo contains a [full example](workflow.yml) for reference.

The action accepts the follow inputs:

| Required | Input Name | Type | Description | Default
| --- | --- | --- | --- | ---
| ✔️ | `mayhem-token` | string | Mayhem API token |
| ✔️ | `api-url` | string | URL to your running API. *Example:* http://localhost:8000/api/v1 |
| ✔️ | `api-spec` | string | Path or URL to your Swagger spec, OpenAPI spec, or Postman collection file, or Postman [collection id](https://support.postman.com/hc/en-us/articles/5063785095319-How-to-find-the-ID-of-an-element-in-Postman).|
|   | `target` | string | The organization-scoped name of your target, such as `forallsecure/mapi-action-example` | auto-generated from your GitHub Repository name
|   | `postman-api-key` | string | Postman [API key](https://learning.postman.com/docs/developer/intro-api/) for api specs that are private postman collection ids.
|   | `postman-environment` | string | Path or id of a Postman [Environment](https://learning.postman.com/docs/sending-requests/managing-environments/).
|   | `zap-api-scan` | boolean | Include results from [ZAP - API Scan](https://www.zaproxy.org/docs/docker/api-scan/) | false |
|   | `duration` | number/string | Duration of scan. 'auto' for automatic duration. Otherwise time (ie: '30sec', '5min', '1h', '1h30m') | auto
|   | `html-report` | string | Path to the generated SARIF report |
|   | `sarif-report` | string | Path to the generated HTML report |
|   | `run-args` | string | Additional arguments to provide to the `mapi run` command.  Argument values should be separated on new lines. <br><br>e.g.<br> <pre>  run-args: \|<br>    # Basic Auth<br>    --basic-auth<br>    login:password</pre><br>⚠️ Avoid wrapping values in quotes, as these will be escaped and included in the value passed to `mapi`.<br><br>⛔️ `"login:password"` <br>✅ `login:password` |
|   | `mayhem-url` | string | Mayhem API override | https://app.mayhem.security


### Continuing on error

The above examples will fail the workflow when issues are found. If you
want to ensure the Action continues, even if Mayhem for API found
issues, then continue-on-error can be used.

```yaml
name: Mayhem for API
on:
    push:
    pull_request:
jobs:
  security:
    runs-on: ubuntu-latest
    steps:
    - uses: actions/checkout@v3
      with:
        # fetch a greater number of commits for computing diffs between jobs
        fetch-depth: 50

    - name: Start your API
      run: ./run_your_api.sh &

    - name: Run Mayhem for API to check for vulnerabilities
      uses: ForAllSecure/mapi-action@v1
      continue-on-error: true # <-----------------------------------------------
      with:
        mayhem-token: ${{ secrets.MAYHEM_TOKEN }}
        api-url: http://localhost:8000 # <- update this
        api-spec: your-openapi-spec-or-postman-collection.json
        # Additional 'mapi run' arguments
        run-args: |
          # Basic Auth
          --basic-auth
          login:password
          # Do not fuzz the '/logout' endpoint
          --ignore-endpoint
          /logout
          # Treat all warnings as errors
          --warnaserror

```

# Reports

Mayhem for API generate reports when you pass `sarif-report` or
`html-report` to the input. Make sure to pass `continue-on-error` to the
Mayhem for API step if you want to process the reports in follow-up
steps.

## Artifact HTML Report

![HTML Report](./imgs/sample-report.png)

To create an artifact of the report in your build, add this step to your pipeline:

```yaml
- name: Run Mayhem for API to check for vulnerabilities
  uses: ForAllSecure/mapi-action@v1
  continue-on-error: true
  with:
    mayhem-token: ${{ secrets.MAYHEM_TOKEN }}
    api-url: http://localhost:8000 # <- update this
    api-spec: your-openapi-spec-or-postman-collection.json # <- update this
    html-report: mapi.html

# Archive HTML report
- name: Archive Mayhem for API report
  uses: actions/upload-artifact@v4
  with:
    name: mapi-report
    path: mapi.html
```

## GitHub Code Scanning support

![Mayhem for API issue in your
PR](./imgs/sarif-github.png)

Uploading SARIF reports to GitHub allows you to see any issue found by
Mayhem for API right on your PR, as well as in the "Security" tab of
your repository. This currently requires you to have a GitHub Enterprise
Plan or have a public repository. To upload the SARIF report, add this
step to your pipeline:

```yaml
- name: Run Mayhem for API to check for vulnerabilities
  uses: ForAllSecure/mapi-action@v1
  continue-on-error: true
  with:
    mayhem-token: ${{ secrets.MAYHEM_TOKEN }}
    api-url: http://localhost:8000 # <- update this
    api-spec: your-openapi-spec-or-postman-collection.json # <- update this
    sarif-report: mapi.sarif

# Upload SARIF file (only available on public repos or github enterprise)
- name: Upload SARIF file
  uses: github/codeql-action/upload-sarif@v1
  with:
    sarif_file: mapi.sarif
```

If your API server sends back stacktraces in the 500 Internal Server
Error (only do this in a test environment -- never in production!),
Mayhem for API will try to map issues it finds to the exact line of code
that triggered the issue.
 readmeEtag: '"a93412e0937a08badd3215d4241c37032454cc01"' readmeLastModified: Mon, 25 Nov 2024 10:11:51 GMT repositoryId: 355229366 description: ' 🤖 Run a Mayhem for API scan in GitHub Actions' created: '2021-04-06T14:58:24Z' updated: '2025-05-31T15:36:41Z' language: TypeScript archived: false stars: 24 watchers: 8 forks: 4 owner: ForAllSecure logo: https://avatars.githubusercontent.com/u/12204221?v=4 license: MIT repoEtag: '"311115291cf4e9b073bfc09632ca25884080d7d777cc88a63c559bfb3f9f4efe"' repoLastModified: Sat, 31 May 2025 15:36:41 GMT foundInMaster: true v3_1: true - source: https://openapi.tools/ name: JSONSchema::Validator category: Data Validators language: Perl link: https://metacpan.org/pod/JSONSchema::Validator repository: https://github.com/skbkontur/perl-jsonschema-validator source_description: >- A Perl library which validates request/response according to an OpenAPI specification v3: true repositoryMetadata: base64Readme: >- IyBOQU1FCgpKU09OU2NoZW1hOjpWYWxpZGF0b3IgLSBWYWxpZGF0b3IgZm9yIEpTT04gU2NoZW1hIERyYWZ0NC9EcmFmdDYvRHJhZnQ3IGFuZCBPcGVuQVBJIFNwZWNpZmljYXRpb24gMy4wCgojIFZFUlNJT04KCnZlcnNpb24gMC4wMTEKCiMgU1lOT1BTSVMKCiAgICAjIHRvIGdldCBPcGVuQVBJIHZhbGlkYXRvciBpbiBZQU1MIGZvcm1hdAogICAgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHJlc291cmNlID0+ICdmaWxlOi8vL3NvbWUvcGF0aC90by9vYXMzMC55bWwnKTsKICAgIG15ICgkcmVzdWx0LCAkZXJyb3JzLCAkd2FybmluZ3MpID0gJHZhbGlkYXRvci0+dmFsaWRhdGVfcmVxdWVzdCgKICAgICAgICBtZXRob2QgPT4gJ0dFVCcsCiAgICAgICAgb3BlbmFwaV9wYXRoID0+ICcvdXNlci97aWR9L3Byb2ZpbGUnLAogICAgICAgIHBhcmFtZXRlcnMgPT4gewogICAgICAgICAgICBwYXRoID0+IHsKICAgICAgICAgICAgICAgIGlkID0+IDEyMzQKICAgICAgICAgICAgfSwKICAgICAgICAgICAgcXVlcnkgPT4gewogICAgICAgICAgICAgICAgZGV0YWlscyA9PiAnc2hvcnQnCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIGhlYWRlciA9PiB7CiAgICAgICAgICAgICAgICBoZWFkZXIgPT4gJ2hlYWRlciB2YWx1ZScKICAgICAgICAgICAgfSwKICAgICAgICAgICAgY29va2llID0+IHsKICAgICAgICAgICAgICAgIG5hbWUgPT4gJ3ZhbHVlJwogICAgICAgICAgICB9LAogICAgICAgICAgICBib2R5ID0+IFskaXNfZXhpc3RzLCAkY29udGVudF90eXBlLCAkZGF0YV0KICAgICAgICB9CiAgICApOwogICAgbXkgKCRyZXN1bHQsICRlcnJvcnMsICR3YXJuaW5ncykgPSAkdmFsaWRhdG9yLT52YWxpZGF0ZV9yZXNwb25zZSgKICAgICAgICBtZXRob2QgPT4gJ0dFVCcsCiAgICAgICAgb3BlbmFwaV9wYXRoID0+ICcvdXNlci97aWR9L3Byb2ZpbGUnLAogICAgICAgIHN0YXR1cyA9PiAnMjAwJywKICAgICAgICBwYXJhbWV0ZXJzID0+IHsKICAgICAgICAgICAgaGVhZGVyID0+IHsKICAgICAgICAgICAgICAgIGhlYWRlciA9PiAnaGVhZGVyIHZhbHVlJwogICAgICAgICAgICB9LAogICAgICAgICAgICBib2R5ID0+IFskaXNfZXhpc3RzLCAkY29udGVudF90eXBlLCAkZGF0YV0KICAgICAgICB9CiAgICApCgogICAgIyB0byBnZXQgSlNPTiBTY2hlbWEgRHJhZnQ0L0RyYWZ0Ni9EcmFmdDcgdmFsaWRhdG9yIGluIEpTT04gZm9ybWF0CiAgICAkdmFsaWRhdG9yID0gSlNPTlNjaGVtYTo6VmFsaWRhdG9yLT5uZXcocmVzb3VyY2UgPT4gJ2h0dHA6Ly9leGFtcGxlLmNvbS9kcmFmdDQvc2NoZW1hLmpzb24nKQogICAgbXkgKCRyZXN1bHQsICRlcnJvcnMpID0gJHZhbGlkYXRvci0+dmFsaWRhdGVfc2NoZW1hKCRvYmplY3RfdG9fdmFsaWRhdGUpCgojIERFU0NSSVBUSU9OCgpPcGVuQVBJIHNwZWNpZmljYXRpb24gYW5kIEpTT04gU2NoZW1hIERyYWZ0NC9EcmFmdDYvRHJhZnQ3IHZhbGlkYXRvcnMgd2l0aCBtaW5pbXVtIGRlcGVuZGVuY2llcy4KCiMgTUVUSE9EUwoKIyMgbmV3CgpDcmVhdGVzIG9uZSBvZiB0aGUgZm9sbG93aW5nIHZhbGlkYXRvcnM6IEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6RHJhZnQ0LCBKU09OU2NoZW1hOjpWYWxpZGF0b3I6OkRyYWZ0NiwgSlNPTlNjaGVtYTo6VmFsaWRhdG9yOjpEcmFmdDcsIEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6T0FTMzAuCgogICAgbXkgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHJlc291cmNlID0+ICdmaWxlOi8vL3NvbWUvcGF0aC90by9vYXMzMC55bWwnKTsKICAgIG15ICR2YWxpZGF0b3IgPSBKU09OU2NoZW1hOjpWYWxpZGF0b3ItPm5ldyhyZXNvdXJjZSA9PiAnaHR0cDovL2V4YW1wbGUuY29tL2RyYWZ0NC9zY2hlbWEuanNvbicpOwogICAgbXkgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHNjaGVtYSA9PiB7JyRzY2hlbWEnID0+ICdwYXRoL3RvL3NjaGVtYScsIC4uLn0pOwogICAgbXkgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHNjaGVtYSA9PiB7Li4ufSwgc3BlY2lmaWNhdGlvbiA9PiAnRHJhZnQ0Jyk7CgppZiBwYXJhbWV0ZXIgYHNwZWNpZmljYXRpb25gIGlzIG5vdCBzcGVjaWZpZWQgdGhlbiB0eXBlIG9mIHZhbGlkYXRvciB3aWxsIGJlIGRldGVybWluZWQgYnkgYCRzY2hlbWFgIGtleQpmb3IgSlNPTiBTY2hlbWEgRHJhZnQ0L0RyYWZ0Ni9EcmFmdDcgYW5kIGJ5IGBvcGVuYXBpYCBrZXkgZm9yIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLjAgaW4gYHNjaGVtYWAgcGFyYW1ldGVyLgoKUGFyYW1ldGVyczoKCi0gcmVzb3VyY2VzCgogICAgVG8gZ2V0IHNjaGVtYSBieSB1cmkKCi0gc2NoZW1hCgogICAgVG8gZ2V0IGV4cGxpY2l0bHkgc3BlY2lmaWVkIHNjaGVtYQoKLSBzcGVjaWZpY2F0aW9uCgogICAgVG8gc3BlY2lmeSBzcGVjaWZpY2F0aW9uIG9mIHNjaGVtYQoKLSB2YWxpZGF0ZVxfc2NoZW1hCgogICAgRG8gbm90IHZhbGlkYXRlIHNwZWNpZmllZCBzY2hlbWEKCi0gYmFzZVxfdXJpCgogICAgVG8gc3BlY2lmeSBiYXNlIHVyaSBvZiBzY2hlbWEuCiAgICBUaGlzIHBhcmFtZXRlciB1c2VkIHRvIGJ1aWxkIGFic29sdXRlIHBhdGggYnkgcmVsYXRpdmUgcmVmZXJlbmNlIGluIHNjaGVtYS4KICAgIEJ5IGRlZmF1bHQgYGJhc2VfdXJpYCBpcyBlcXVhbCB0byB0aGUgcmVzb3VyY2UgcGF0aCBpZiB0aGUgcmVzb3VyY2UgcGFyYW1ldGVyIGlzIHNwZWNpZmllZCBvdGhlcndpc2UgdGhlIGAkaWRgIGtleSBpbiB0aGUgc2NoZW1hLgoKQWRkaXRpb25hbCBwYXJhbWV0ZXJzIG5lZWQgdG8gYmUgbG9va2VkIGF0IGluIGEgc3BlY2lmaWMgdmFsaWRhdG9yIGNsYXNzLgpDdXJyZW50bHkgdGhlcmUgYXJlIHZhbGlkYXRvcnM6IEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6RHJhZnQ0LCBKU09OU2NoZW1hOjpWYWxpZGF0b3I6OkRyYWZ0NiwgSlNPTlNjaGVtYTo6VmFsaWRhdG9yOjpEcmFmdDcsIEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6T0FTMzAuCgojIyB2YWxpZGF0ZVxfcGF0aHMKClZhbGlkYXRlcyBhbGwgZmlsZXMgc3BlY2lmaWVkIGJ5IHBhdGggZ2xvYnMuCgogICAgbXkgJHJlc3VsdCA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+dmFsaWRhdGVfcGF0aHMoWycvc29tZS9wYXRoL3RvL29wZW5hcGkuKi55YW1sJywgJy9zb21lL3BhdGgvdG8vanNvbnNjaGVtYS4qLmpzb24nXSk7CiAgICBmb3IgbXkgJGZpbGUgKGtleXMgJSRyZXN1bHQpIHsKICAgICAgICBteSAoJHJlcywgJGVycm9ycykgPSBAeyRyZXN1bHQtPnskZmlsZX19OwogICAgfQoKIyMgdmFsaWRhdGVcX3Jlc291cmNlCgojIyB2YWxpZGF0ZVxfcmVzb3VyY2VcX3NjaGVtYQoKIyBDQVZFQVRTCgojIyBZQU1MICYgYm9vbGVhbnMKCldoZW4gcmVhZGluZyBzY2hlbWEgZGVmaW5pdGlvbnMgZnJvbSBZQU1MLCBwbGVhc2Ugbm90ZSB0aGF0IHRoZSBzdGFuZGFyZApiZWhhdmlvdXIgb2YgW1lBTUw6OlBQXShodHRwczovL21ldGFjcGFuLm9yZy9wb2QvWUFNTCUzQSUzQVBQKSBhbmQgW1lBTUw6OlhTXShodHRwczovL21ldGFjcGFuLm9yZy9wb2QvWUFNTCUzQSUzQVhTKSBpcyB0byByZWFkIHZhbHVlcyB3aGljaCBldmFsdWF0ZQp0byBgdHJ1ZWAgb3IgYGZhbHNlYCBpbiBhIHBlcmwgY29udGV4dC4gVGhlc2UgdmFsdWVzIGhhdmUgbm8gcmVjb2duaXphYmxlCidib29sZWFuIHR5cGUnLiBUaGlzIGlzIGluc3VmZmljaWVudCBmb3IgSlNPTiBzY2hlbWEgdmFsaWRhdGlvbi4KClRvIG1ha2UgdGhlIFlBTUwgcmVhZGVycyBhbmQgYm9vbGVhbnMgd29yayB3aXRoIGBKU09OU2NoZW1hOjpWYWxpZGF0b3JgLAp5b3UgbmVlZCB0byB1c2UgdGhlIGBKU09OOjpQUGAgKGluY2x1ZGVkIGluIFBlcmwncyBzdGFuZGFyZCBsaWJyYXJ5KSBtb2R1bGUKYXMgZm9sbG93czoKCiAgICAjIGZvciBZQU1MOjpQUAogICAgdXNlIFlBTUw6OlBQOwoKICAgIG15ICRyZWFkZXIgPSBZQU1MOjpQUC0+bmV3KCBib29sZWFuID0+ICdKU09OOjpQUCcgKTsKICAgICMgZnJvbSBoZXJlLCB5b3UgY2FuIGZyZWVseSB1c2UgdGhlIHJlYWRlciB0bwogICAgIyByZWFkICYgd3JpdGUgYm9vbGVhbnMgYXMgJ3RydWUnIGFuZCAnZmFsc2UnCgoKICAgICMgZm9yIFlBTUw6OlhTCiAgICB1c2UgWUFNTDo6WFM7CgogICAgbXkgJHJlYWRlciA9IFlBTUw6OlhTLT5uZXc7CgogICAgIyBhbmQgd2hlbmV2ZXIgeW91IHJlYWQgWUFNTCB3aXRoIHRoaXMgcmVhZGVyLCBkbzoKICAgIG15ICR5YW1sID0gZG8gewogICAgICBsb2NhbCAkWUFNTDo6WFM6OkJvb2xlYW4gPSAnSlNPTjo6UFAnOwogICAgICAkcmVhZGVyLT5Mb2FkKCRzdHJpbmcpOyAjIG9yICRyZWFkZXItPkxvYWRGaWxlKCdmaWxlbmFtZScpOwogICAgfTsKClRoaXMgaXNuJ3QgYSBwcm9ibGVtIHdoZW4geW91IHVzZSB0aGUgYHJlc291cmNlYCBhcmd1bWVudCB0byB0aGUKYEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6bmV3YCBjb25zdHJ1Y3RvciwgYnV0IGlmIHlvdSByZWFkIHlvdXIgb3duCnNjaGVtYSBhbmQgdXNlIHRoZSBgc2NoZW1hYCBhcmd1bWVudCwgdGhpcyBpcyBzb21ldGhpbmcgdG8gYmUgYXdhcmUgb2YuCgojIyBhbGxvd1xfYmlnbnVtID0+IDEKClRoZSBgYWxsb3dfYmlnbnVtID1gIDE+IHNldHRpbmcgKGF2YWlsYWJsZSBvbiBbSlNPTjo6WFNdKGh0dHBzOi8vbWV0YWNwYW4ub3JnL3BvZC9KU09OJTNBJTNBWFMpIGFuZApbQ3BhbmVsOjpKU09OOjpYU10oaHR0cHM6Ly9tZXRhY3Bhbi5vcmcvcG9kL0NwYW5lbCUzQSUzQUpTT04lM0ElM0FYUykpIG9uIGRlc2VyaWFsaXplcnMgaXMgbm90IHN1cHBvcnRlZC4KCldoZW4gZGVzZXJpYWxpemluZyBhIHJlcXVlc3QgYm9keSB3aXRoIGEgSlNPTiBwYXJzZXIgY29uZmlndXJlZCB3aXRoCmBhbGxvd19iaWdudW0gPWAgMT4sIGZsb2F0cyAtIGV2ZW4gb25lcyB3aGljaCBmaXQgaW50byB0aGUgcmVndWxhcgpmbG9hdCByYW5nZXMgLSB3aWxsIGJlIGRlc2VyaWFsaXplZCBhcyBgTWF0aDo6QmlnRmxvYXRgLiBTaW1pbGFybHksCmludGVnZXJzIG91dHNpZGUgb2YgdGhlIGludGVybmFsIGludGVnZXIgcmFuZ2UgYXJlIGRlc2VyaWFsaXplZCBhcwpgTWF0aDo6QmlnSW50YC4gTnVtYmVycyByZXByZXNlbnRlZCBhcyBgTWF0aDo6QmlnKmAgb2JqZWN0cyBhcmUgbm90CnJlY29nbml6ZWQgYXMgYWN0dWFsIG51bWJlcnMgYW5kIHdpbGwgZmFpbCB2YWxpZGF0aW9uLgoKIyBBVVRIT1JTCgotIEFsZXhleSBTdGF2cm92IDxsb2dpb25pekB5YS5ydT4KLSBJdmFuIFB1dGludHNldiA8dWlkQHJ5ZGxhYi5ydT4KLSBBbnRvbiBGZWRvdG92IDx0b3NoYS5mZWRvdG92LjIwMDBAZ21haWwuY29tPgotIERlbmlzIEliYWV2IDxkaW9ueXNAZ21haWwuY29tPgotIEFuZHJleSBLaG96b3YgPGFuZHJleUByeWRsYWIucnU+CgojIENPTlRSSUJVVE9SUwoKLSBFcmlrIEh1ZWxzbWFubiA8ZWh1ZWxzQGdtYWlsLmNvbT4KLSBKYW1lcyBXYXRlcnMgPGphbWVzQGpjd2F0ZXJzLmNvLnVrPgoKIyBDT1BZUklHSFQgQU5EIExJQ0VOU0UKClRoaXMgc29mdHdhcmUgaXMgQ29weXJpZ2h0IChjKSAyMDIxIGJ5IEFsZXhleSBTdGF2cm92LgoKVGhpcyBpcyBmcmVlIHNvZnR3YXJlLCBsaWNlbnNlZCB1bmRlcjoKCiAgICBUaGUgTUlUIChYMTEpIExpY2Vuc2UK readmeEtag: '"32131c216eadce854b63f2683e969d148b8f6a5b"' readmeLastModified: Fri, 09 Sep 2022 20:43:45 GMT repositoryId: 382338785 description: JSON Schema and OpenAPI data validator for Perl created: '2021-07-02T12:25:14Z' updated: '2025-08-22T15:27:27Z' language: Perl archived: false stars: 7 watchers: 7 forks: 4 owner: skbkontur logo: https://avatars.githubusercontent.com/u/5087073?v=4 license: NOASSERTION repoEtag: '"6bff7480de4e1f12236d8fbd8da528b861b6c3fdf205d487bac073d7c142f5fe"' repoLastModified: Fri, 22 Aug 2025 15:27:27 GMT foundInMaster: true id: 66dd4d5692dee0c9db5adb6f7788057a - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags name: openapi-python-client category: - Converters - Parsers link: https://github.com/openapi-generators/openapi-python-client repository: https://github.com/openapi-generators/openapi-python-client language: Python source_description: Generate modern Python clients from OpenAPI 3.0 documents. v2: false v3: true repositoryMetadata: base64Readme: >- ![Run Checks](https://github.com/openapi-generators/openapi-python-client/workflows/Run%20Checks/badge.svg)
[![codecov](https://codecov.io/gh/openapi-generators/openapi-python-client/branch/main/graph/badge.svg)](https://codecov.io/gh/triaxtec/openapi-python-client)
[![MIT license](https://img.shields.io/badge/License-MIT-blue.svg)](https://lbesson.mit-license.org/)
[![Generic badge](https://img.shields.io/badge/type_checked-mypy-informational.svg)](https://mypy.readthedocs.io/en/stable/introduction.html)
[![PyPI version shields.io](https://img.shields.io/pypi/v/openapi-python-client.svg)](https://pypi.python.org/pypi/openapi-python-client/)
[![Downloads](https://static.pepy.tech/personalized-badge/openapi-python-client?period=total&units=international_system&left_color=blue&right_color=green&left_text=Downloads)](https://pepy.tech/project/openapi-python-client)

# openapi-python-client

Generate modern Python clients from OpenAPI 3.0 and 3.1 documents.

_This generator does not support OpenAPI 2.x FKA Swagger. If you need to use an older document, try upgrading it to
version 3 first with one of many available converters._

**This project is still in development and does not support all OpenAPI features**

## Why This?

This tool focuses on creating the best developer experience for Python developers by:

1. Using all the latest and greatest Python features like type annotations and dataclasses.
2. Having documentation and usage instructions specific to this one generator.
3. Being written in Python with Jinja2 templates, making it easier to improve and extend for Python developers. It's also much easier to install and use if you already have Python.

## Installation

I recommend you install with [pipx](https://pipxproject.github.io/pipx/) so you don't conflict with any other packages you might have: `pipx install openapi-python-client --include-deps`.

> Note the `--include-deps` option makes `ruff` available in your path so that `openapi-python-client` can use it to clean up the generated code.

**If you use `pipx run` then the post-generation hooks will not be available unless you install them manually.**

You can also install with normal pip: `pip install openapi-python-client`

Then, if you want tab completion: `openapi-python-client --install-completion`

## Usage

### Create a new client

`openapi-python-client generate --url https://my.api.com/openapi.json`

This will generate a new client library named based on the title in your OpenAPI spec. For example, if the title
of your API is "My API", the expected output will be "my-api-client". You can change that directory name with the config file (documented below) or with `--output-path`.

If the directory to generate already exists, you'll get an error unless you use `--overwrite`.

You can use an OpenAPI file instead of a URL like `openapi-python-client generate --path location/on/disk/openapi.json`.

### Using custom templates

This feature leverages Jinja2's [ChoiceLoader](https://jinja.palletsprojects.com/en/2.11.x/api/#jinja2.ChoiceLoader) and [FileSystemLoader](https://jinja.palletsprojects.com/en/2.11.x/api/#jinja2.FileSystemLoader). This means you do _not_ need to customize every template. Simply copy the template(s) you want to customize from [the default template directory](openapi_python_client/templates) to your own custom template directory (file names _must_ match exactly) and pass the template directory through the `custom-template-path` flag to the `generate` command:

```
openapi-python-client generate \
  --url https://my.api.com/openapi.json \
  --custom-template-path=relative/path/to/mytemplates
```

_Be forewarned, this is a beta-level feature in the sense that the API exposed in the templates is undocumented and unstable._

## What You Get

1. A `pyproject.toml` file, optionally with [Poetry] metadata (default), [PDM] (with `--meta=pdm`), or only [Ruff] config.
2. A `README.md` you'll most definitely need to update with your project's details
3. A Python module named just like the auto-generated project name (e.g. "my_api_client") which contains:
   1. A `client` module which will have both a `Client` class and an `AuthenticatedClient` class. You'll need these
      for calling the functions in the `api` module.
   2. An `api` module which will contain one module for each tag in your OpenAPI spec, as well as a `default` module
      for endpoints without a tag. Each of these modules in turn contains one function for calling each endpoint.
   3. A `models` module which has all the classes defined by the various schemas in your OpenAPI spec
4. A `setup.py` file _if_ you use `--meta=setup` (default is `--meta=poetry`)

For a full example you can look at the `end_to_end_tests` directory which has `baseline_openapi_3.0.json` and `baseline_openapi_3.1.yaml` files.
The "golden-record" in that same directory is the generated client from either of those OpenAPI documents.

## Configuration

You can pass a YAML (or JSON) file to openapi-python-client with the `--config` option in order to change some behavior.
The following parameters are supported:

### class_overrides

Used to change the name of generated model classes. This param should be a mapping of existing class name
(usually a key in the "schemas" section of your OpenAPI document) to class_name and module_name. As an example, if the
name of a model in OpenAPI (and therefore the generated class name) was something like "_PrivateInternalLongName"
and you want the generated client's model to be called "ShortName" in a module called "short_name" you could do this:

Example:

```yaml
class_overrides:
  _PrivateInternalLongName:
    class_name: ShortName
    module_name: short_name
```

The easiest way to find what needs to be overridden is probably to generate your client and go look at everything in the `models` folder.

### docstrings_on_attributes

By default, when `openapi-python-client` generates a model class, it includes a list of attributes and their
descriptions in the docstring for the class. If you set this option to `true`, then the attribute descriptions
will be put in docstrings for the attributes themselves, and will not be in the class docstring.

```yaml
docstrings_on_attributes: true
```

### literal_enums

By default, `openapi-python-client` generates classes inheriting for `Enum` for enums. It can instead use `Literal` 
values for enums by setting this to `true`:

```yaml
literal_enums: true
```

This is especially useful if enum values, when transformed to their Python names, end up conflicting due to case sensitivity or special symbols.

### generate_all_tags

`openapi-python-client` generates module names within the `api` module based on the OpenAPI `tags` of each endpoint. 
By default, only the _first_ tag is generated. If you want to generate **duplicate** endpoint functions using _every_ tag 
listed, you can enable this option:

```yaml
generate_all_tags: true
```

### project_name_override and package_name_override

Used to change the name of generated client library project/package. If the project name is changed but an override for the package name
isn't provided, the package name will be converted from the project name using the standard convention (replacing `-`'s with `_`'s).

Example:

```yaml
project_name_override: my-special-project-name
package_name_override: my_extra_special_package_name
```

### field_prefix

When generating properties, the `name` attribute of the OpenAPI schema will be used. When the `name` is not a valid Python identifier (e.g. begins with a number) this string will be prepended. Defaults to "field\_". It will also be used to prefix fields in schema starting with "_" in order to avoid ambiguous semantics.

Example:

```yaml
field_prefix: attr_
```

### package_version_override

Specify the package version of the generated client. If unset, the client will use the version of the OpenAPI spec.

Example:

```yaml
package_version_override: 1.2.3
```

### post_hooks

In the config file, there's an easy way to tell `openapi-python-client` to run additional commands after generation. Here's an example showing the default commands (using [Ruff]) that will run if you don't override them in config:

```yaml
post_hooks:
   - "ruff check . --fix-only"
   - "ruff format ."
```

### use_path_prefixes_for_title_model_names

By default, `openapi-python-client` generates class names which include the full path to the schema, including any parent-types. This can result in very long class names like `MyRouteSomeClassAnotherClassResponse`—which is very unique and unlikely to cause conflicts with future API additions, but also super verbose.

If you are carefully curating your `title` properties already to ensure no duplicate class names, you can turn off this prefixing feature by setting `use_path_prefixes_for_title_model_names` to `false` in your config file. This will use the `title` property of any object that has it set _without_ prefixing.

If this option results in conflicts, you will need to manually override class names instead via the `class_overrides` option.

### http_timeout

By default, the timeout for retrieving the schema file via HTTP is 5 seconds. In case there is an error when retrieving the schema, you might try and increase this setting to a higher value.

### content_type_overrides

Normally, `openapi-python-client` will skip any bodies or responses that it doesn't recognize the content type for.
This config tells the generator to treat a given content type like another.

```yaml
content_type_overrides:
  application/zip: application/octet-stream
```

## Supported Extensions

### x-enum-varnames

This extension has been adopted by similar projects such as [OpenAPI Tools](https://github.com/OpenAPITools/openapi-generator/pull/917).
It is intended to provide user-friendly names for integer Enum members that get generated.
It is critical that the length of the array matches that of the enum values.

```
"Colors": {
   "type": "integer",
   "format": "int32",
   "enum": [
       0,
       1,
       2
   ], 
  "x-enum-varnames": [
      "Red",
      "Green",
      "Blue"
   ]
}
```

Results in:
```
class Color(IntEnum):
    RED = 0
    GREEN = 1
    BLUE = 2
```

[changelog.md]: CHANGELOG.md
[poetry]: https://python-poetry.org/
[PDM]: https://pdm-project.org/latest/
[Ruff]: https://docs.astral.sh/ruff/
 readmeEtag: '"4a886d2992f16a1e5b0ca8c63868476618a80829"' readmeLastModified: Tue, 04 Nov 2025 01:05:20 GMT repositoryId: 240776275 description: Generate modern Python clients from OpenAPI created: '2020-02-15T19:33:46Z' updated: '2026-02-05T23:16:31Z' language: Python archived: false stars: 1901 watchers: 15 forks: 264 owner: openapi-generators logo: https://avatars.githubusercontent.com/u/84925606?v=4 license: MIT repoEtag: '"99c7fe952f1471f600eb315d73c3507b13bbc181550da324693e3b77b4cd9f71"' repoLastModified: Thu, 05 Feb 2026 23:16:31 GMT foundInMaster: true id: 784ec4adac4e47e4211ea74ef36dd5a5 v3_1: true - source: - https://openapi.tools/ - openapi3 tags name: Elements category: Documentation link: https://stoplight.io/open-source/elements repository: https://github.com/stoplightio/elements language: - Javascript - Custom Element source_description: >- Build beautiful, interactive API Docs with embeddable React or Web Components, powered by OpenAPI and Markdown v2: true v3: true v3_1: true repositoryMetadata: base64Readme: >- [![Elements - OpenAPI Powered API Documentation](docs/images/readme-header.svg)][elements_landing_page]

[![Storybook](https://cdn.jsdelivr.net/gh/storybookjs/brand@master/badge/badge-storybook.svg)](https://stoplight-elements.netlify.app)
[![CircleCI][circle_ci_image]][circle_ci]
[![NPM Downloads][circle_ci_image]][npm]
[![Stoplight Forest](https://img.shields.io/ecologi/trees/stoplightinc)][stoplight_forest]

Beautiful API documentation powered by OpenAPI and Markdown. Use these UI components to create API reference documentation, or more complete documentation with Markdown articles covering tutorials, how-to guides, etc.

Available as React Components, or Web Components, you can use Elements all together to build beautiful three-column "Stripe-esque" documentation, or stacked documentation thats easier for integrating into existing Content Management Systems with their own navigation.

# Overview

- [Overview](#overview)
  - [📖 Community](#-community)
  - [👁️🗨 ️️Examples](#️-️️examples)
  - [🏁 Usage](#-usage)
    - [React Component](#react-component)
    - [Web Component](#web-component)
  - [🚧 Roadmap](#-roadmap)
  - [⚙️ Integrations](#️-integrations)
  - [🏁 Help Others Utilize Elements](#-help-others-utilize-elements)
  - [👏 Contributing](#-contributing)
  - [🎉 Thanks](#-thanks)
  - [🌲 Sponsor Elements by Planting a Tree](#-sponsor-elements-by-planting-a-tree)

## 📖 Community

Let's chat about features, ideas, what you're doing with Elements, on [GitHub Discussions](https://github.com/stoplightio/elements/discussions).

## 👁️🗨 ️️Examples

Stoplight Elements comes with a few example integration projects, showing you how to utilize Elements with different frameworks.
- **[react-cra](./examples/react-cra)** - An example app built using Create React App utilizing Stoplight Elements.
- **[angular](./examples/angular)** - An angular app utilizing the Web Components distribution of Elements.
- **[bootstrap](./examples/bootstrap)** - A single HTML page utilizing the Web Components distribution via a global script tag.

To run these examples yourself:
1. Clone this repo.
2. Go to `examples` folder and open an example, e.g.: `examples/angular`.
3. Run `yarn` to install all dependencies.
4. Run `yarn start` to run the example.

> **Note:** for **bootstrap** example just go straight to its directory and open the HTML file.

## 🏁 Usage

The examples will hopefully help show Elements working in close to real world situations, but the most bare bones examples of Elements can be found below.

### React Component

```bash
$ npm install @stoplight/elements
```

```js
import { API } from "@stoplight/elements";

<API
  apiDescriptionUrl="https://api.apis.guru/v2/specs/github.com/1.1.4/openapi.yaml"
  router="history"
/>
```

For more information on using Elements as a React component, head over to our [React documentation](docs/getting-started/elements/react.md).
### Web Component

```html
<!doctype html>
<html lang="en">
  <head>
    <meta charset="utf-8">
    <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
    <title>Elements in HTML</title>
    <!-- Embed elements Elements via Web Component -->
    <script src="https://unpkg.com/@stoplight/elements/web-components.min.js"></script>
    <link rel="stylesheet" href="https://unpkg.com/@stoplight/elements/styles.min.css">
  </head>
  <body>

    <elements-api
      apiDescriptionUrl="https://api.apis.guru/v2/specs/github.com/1.1.4/openapi.yaml"
      router="hash"
      layout="sidebar"
    />

  </body>
</html>
```

Load this page up in your browser and you should see the [GitHub REST API](https://docs.github.com/en/rest) documented in Stoplight Elements.

For more information on using Elements as a Web Component, head over to our [Web Component documentation](docs/getting-started/elements/html.md).

## 🚧 Roadmap

- [x] API Console (a.k.a "Try it!")
- [x] Automatic Code Samples
- [x] Automatic Examples
- [x] React & Web Component Support
- [x] OpenAPI Support
  - [x] OpenAPI v3.1
  - [x] OpenAPI v3.0
  - [x] OpenAPI v2.0
- [x] Callbacks
- [x] Webhooks
- [x] Multiple APIs (a.k.a "Dev Portal")

Submit your ideas for new functionality on the [Stoplight Roadmap](https://roadmap.stoplight.io/?utm_source=github&utm_medium=elements&utm_campaign=readme).

## ⚙️ Integrations

- [Stoplight Studio](https://stoplight.io/studio/?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_elements) - Free visual OpenAPI designer that uses Elements to preview your API descriptions on the fly.
- [Stoplight Platform](https://stoplight.io/?utm_source=github.com&utm_medium=referral&utm_campaign=github_repo_elements) - Collaborative API Design Platform for designing, developing and documenting APIs with hosted documentation powered by Elements.
- [LaravelPHP Elements](https://packagist.org/packages/juststeveking/laravel-stoplight-elements) - A simple API documentation package for Laravel using OpenAPI and Stoplight Elements.

## 🏁 Help Others Utilize Elements

If you're using Elements for an interesting use case, [contact us](mailto:growth@stoplight.io) for a case study. We'll add it to a list here. Spread the goodness 🎉

## 👏 Contributing

If you are interested in contributing to Elements itself, check out our [contributing docs ⇗][contributing] and [code of conduct ⇗][code_of_conduct] to get started.

## 🎉 Thanks

Elements is built on top of lots of excellent packages, and here are a few we'd like to say a special thanks to.

- [httpsnippet](https://www.npmjs.com/package/httpsnippet) by [Kong](https://github.com/Kong).
- [openapi-sampler](https://www.npmjs.com/package/openapi-sampler) by [ReDocly](https://redoc.ly/).

Check these projects out!

## 🌲 Sponsor Elements by Planting a Tree

If you would like to thank us for creating Elements, we ask that you [**buy the world a tree**][stoplight_forest].

[stoplight_forest]: https://ecologi.com/stoplightinc

[code_of_conduct]: CODE_OF_CONDUCT.md
[contributing]: CONTRIBUTING.md
[download-release]: https://github.com/stoplightio/elements/releases/latest
[elements_landing_page]: https://elements-demo.stoplight.io?utm_source=github&utm_medium=elements&utm_campaign=readme
[circle_ci]: https://circleci.com/gh/stoplightio/elements
[circle_ci_image]: https://img.shields.io/circleci/build/github/stoplightio/elements/main
[npm]: https://www.npmjs.com/package/@stoplight/elements
[npm_image]: https://img.shields.io/npm/dw/@stoplight/elements?color=blue
[stoplight_forest]: https://ecologi.com/stoplightinc
 readmeEtag: '"3efe9d7f56d3a16ec93c29aea6e06b56789631a6"' readmeLastModified: Fri, 16 Aug 2024 12:21:04 GMT repositoryId: 184680614 description: >- Build beautiful, interactive API Docs with embeddable React or Web Components, powered by OpenAPI and Markdown. created: '2019-05-03T01:05:01Z' updated: '2026-02-05T11:18:58Z' language: TypeScript archived: false stars: 2349 watchers: 17 forks: 253 owner: stoplightio logo: https://avatars.githubusercontent.com/u/10767217?v=4 license: Apache-2.0 repoEtag: '"4c1900b2a60cc9a614f73da86711109405382cd001ab450feebc70f62f0b6263"' repoLastModified: Thu, 05 Feb 2026 11:18:58 GMT foundInMaster: true id: 2f94cc1fc980953b17a9979d4ff2e7ec - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/nfroidure/whook v3: true repositoryMetadata: base64Readme: >- [//]: # ( )
[//]: # (This file is automatically generated by a `metapak`)
[//]: # (module. Do not change it  except between the)
[//]: # (`content:start/end` flags, your changes would)
[//]: # (be overridden.)
[//]: # ( )
# whook
> Build strong and efficient REST web services.

[![GitHub license](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/nfroidure/whook/blob/main/LICENSE)


[//]: # (::contents:start)

Why write code when you have an OpenAPI 3.1 definition?

![The Whook's logo](./whook.svg)

## Summary

Whook eats your documentation and provides you with a performant router that
takes care of running the right code for the right operation.

By using the [OpenAPI](https://www.openapis.org/) standard and the dependency
injection pattern, Whook provides a convenient, highly modular and easily
testable back end framework.

## Quickstart

To start a new Whook project:

```sh
# Initialize the project
npm init @whook;
cd my_project_name;

# Check install with a dry run of the server
npm run dev -- __inject httpServer,process,dryRun

# Run tests
npm t

# Start developing
npm run watch

# Build the project
npm run build

# Start the compiled project
npm start

# Start the built project
node builds/local/server/start.js

# Create a new handler/service/provider
npm run whook -- create
```

## Why use Whook?

- robust: types, functional programming
- highly modular, extendable and reusable
- fully integrated and production ready
- can be deployed anywhere (serverless, docker, microservices): enter the
  anylith era
- easy to test: TDD, E2E tests made easy
- feature complete for most API use cases
- ease your work but embrace projects complexity

## Usage

A tutorial is still to be written, expect it to come very soon. The above
[quickstart command](#quickstart) is a good starting point.

That said you can check the following "How to" PRs:

- [add GraphQL](https://github.com/nfroidure/whook/pull/62)
- [deploy with Docker](https://github.com/nfroidure/whook/pull/164)
- [deploy on AWS Lambda](https://github.com/nfroidure/whook/pull/54)
- [deploy with GCP Cloud Functions](https://github.com/nfroidure/whook/pull/66)

Also, the [`packages/` folder](./packages) contains a lot of easy to setup
modules with well detailed readmes and setup instructions.

Finally, search for Whook's package easily with the
[NPM's Whook tag](https://www.npmjs.com/search?q=keywords:whook). Also the
[Knifecycle tag](https://www.npmjs.com/search?q=keywords:knifecycle) can be
useful to find projects using the same dependency injection library.

If you have any question or issue using Whook, post your help request to stack
overflow with the
[Whook tag](https://stackoverflow.com/questions/ask?tags=whook). Questions with
this tag will be regularly checked by Whook's authors.

Finally, if you encounter any bug (unexpected error, feature requests, OpenAPI
specification violation), please fill an issue!

## Principles

[Check this deck](https://slides.com/nfroidure/introducing-whook) for a complete
introduction to Whook's principles!

### Global overview

This projects aims to make creating well documented and highly customizable REST
APIs a breeze. It is based on years of experience
[building REST APIs with NodeJS](https://insertafter.com/en/blog/http_rest_apis_with_nodejs.html).

By relying on the [OpenAPI schemas](https://www.openapis.org/) to declare a new
endpoint, this project forces documentation before code. It also is highly
customizable since based on the dependency injection with inversion of control
pattern allowing you to override or wrap its main constituents.

![Architecture Overview](./overview.svg)

The Whook route handling flow is very simple.

First, we have a HTTPServer that handles requests and serves responses (the
`httpServer` service).

Then, the `httpTransaction` transform the NodeJS requests into raw serializable
ones (raw objects with no methods nor internal states, useful for testing).

Then the router (`httpRouter`) deal with that request to test which handler
needs to be run by comparing the method/path couple with the OpenAPI operations
declarations.

Once found, it simply runs the right route handler with the OpenAPI parameters
value filled from the serializable request. The handler simply have to return a
serializable response object in turn.

If any error occurs within this process, than the `errorHandler` is responsible
for providing the now lacking response object based on the error it catches.

And that's it, you have your REST API. We have
[no middleware](http://insertafter.com/en/blog/no_more_middlewares.html) concept
here. Instead, every handler is a simple function taking parameters and
returning a response. It makes those functions very easily composable (in a
functional programming sense).

You may add global wrappers to change every routes input/output on the fly or
add a local wrapper specifically to one or a few routes.

### Core concepts

Whook work by adding ingredients to you API:

- **configuration**: Whook look ups for `config/{NODE_ENV}/config.js` files. It
  creates constants you can inject in your routes and services.
- **API**: It defines the various endpoint of your API and how to map these to
  routes thanks to the well known OpenAPI format (formerly Swagger),
- **routes**: to define an implement API endpoints,
- **crons**: to define and implement periodic tasks,
- **consumers**: to implement queues/streams consumers,
- **services**: various services that deal with side effects,
- **wrappers**: higher order functions you can apply to routes, crons or
  consumers (CORS, authentication...).

You can see a lot of those concepts implemented in the
[Whook example](./packages/whook-example) folder.

Whook's DI system relies on the
[Knifecyle](https://github.com/nfroidure/knifecycle) module. It is great for
adding or easily override/wrap a lot of its core component and brings
instrumentation and testability to your code bases.

## Contributing

Contributors are very welcome to help pushing Whook forward!

Clone this project's repository and run:

```sh
npm i
npm run build
npm run metapak
npm t
```

The repository is based on LernaJS that allows to host several NPM packages in a
single repository. That said, to keep it simple it only proxies the packages
commands.

Install those
[VSCode extensions](https://insertafter.com/en/blog/my_vscode_configuration.html)
to get a smooth developer experience.

For committing run:

```sh
npm run cz
```

## Publishing

```sh
NODE_ENV=cli npm run lerna  -- publish
```

[//]: # (::contents:end)

# Authors
- [Nicolas Froidure](http://insertafter.com/en/index.html)
- [Vincent Da Silva](https://dasilvavincent.github.io/PortFolio/)
- [Ayoub HAD-DAD](https://github.com/AubHaddad)

# License
[MIT](https://github.com/nfroidure/whook/blob/main/LICENSE)
 readmeEtag: '"79803fe7c304e5b2f2c416e68718b90702127c1f"' readmeLastModified: Mon, 17 Mar 2025 15:50:52 GMT repositoryId: 33482446 description: Build strong and efficient REST web services. created: '2015-04-06T13:09:13Z' updated: '2025-11-25T13:20:56Z' language: TypeScript archived: false stars: 33 watchers: 4 forks: 6 owner: nfroidure logo: https://avatars.githubusercontent.com/u/229633?v=4 license: MIT repoEtag: '"558794ecbffee93df824153cce5fccda5fb1635b8f496d2ab2aa27d5c55b8ccc"' repoLastModified: Tue, 25 Nov 2025 13:20:56 GMT foundInMaster: true name: Whook homepage: https://github.com/nfroidure/whook language: - Javascript - TypeScript source_description: OpenAPI 3 based NodeJS server. category: - Server - Server Implementations id: 939a255e49c264ce8bc1417d3621a5e5 link: https://github.com/nfroidure/whook v2: false v3_1: true - source: https://openapi.tools/ name: openapi-ts-sdk-builder homepage: https://github.com/nfroidure/openapi-ts-sdk-builder language: - Javascript - TypeScript source_description: Generate a TypeScript SDK from OpenAPI 3 definitions. category: SDK foundInMaster: true repository: https://github.com/nfroidure/openapi-ts-sdk-builder repositoryMetadata: base64Readme: >- [//]: # ( )
[//]: # (This file is automatically generated by a `metapak`)
[//]: # (module. Do not change it  except between the)
[//]: # (`content:start/end` flags, your changes would)
[//]: # (be overridden.)
[//]: # ( )
# openapi-ts-sdk-builder
> Create a TypeScript SDK from an OpenAPI 3 definition

[![GitHub license](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/nfroidure/openapi-ts-sdk-builder/blob/main/LICENSE)


[//]: # (::contents:start)

A TypeScript rewrite of
[openapi-js-sdk-builder](https://github.com/sencrop/openapi-js-sdk-builder).

It basically brings a minimal TypeScript SDK from an OpenAPI3 file with no OOP
inside. It works with any HTTP client.

# Usage

With a raw Node script:

```js
import { generateSDKFromOpenAPI } from 'openapi-ts-sdk-builder';
import { readFileSync, writeFileSync } from 'fs';

const openAPIContents = readFileSync('openapi.json', 'utf-8');
const sdkContents = generateSDKFromOpenAPI(
  openAPIContents,
  {
    sdkVersion: 'v1.1.1',
    ignoredParametersNames: ['cookie', 'X-API-Version', 'X-SDK-Version'],
    undocumentedParametersNames: ['X-Application-Version'],
  },
  {
    generateUnusedSchemas: true,
    brandedTypes: [
      'SensorUUID',
      'UUID',
      'Locale',
      'TimeZone',
      'ValueName',
      'SensorVariable',
    ],
    generateRealEnums: true,
    exportNamespaces: true,
  },
);

writeFileSync('src/sdk.ts', sdkContents, 'utf-8');
```

Sample usage with `axios`:

```ts
import BaseAPI, { APIStatuses } from './sdk.ts';
import axios from 'axios';
import querystring from 'querystring';
import type { RequestExecutor } from './sdk.ts';
import type { AxiosRequestConfig } from 'axios';

const executeRequest: RequestExecutor<AxiosRequestConfig> = async (
  httpRequest,
  operationId,
  options,
) => {
  const callOptions = {
    ...options,
    baseURL: 'http://localhost:3000',
    url: httpRequest.path,
    method: httpRequest.method,
    headers: {
      ...(options.headers || {}),
      ...(httpRequest.headers || {}),
    },
    params: httpRequest.params,
    data: httpRequest.body,
    paramsSerializer: querystring.stringify.bind(querystring),
    validateStatus: (status: number) =>
      APIStatuses[operationId].includes(status),
  };
  const response = await axios(callOptions);

  return {
    status: response.status,
    headers: response.headers,
    body: response.data,
  };
};

// Use the API
await BaseAPI.getPing(executeRequest);
await BaseAPI.getUser(executeRequest, { userId: '123' });

// Generate URIs only use the API then
await APIURIBuilders.buildGetPingURI({
  /*...*/
});

// To know which method is used by an endpoint
APIMethods.getPing; // => get

// To know which status codes can be returned by an endpoint
APIStatuses.getPing; // => ["default", 200]

// Generate a complete endpoint input
// (may be useful when you want to pass
// HTTP requests to another process )
APIInputBuilders.buildGetPingInput({
  /*...*/
});
```

You can also safely operate on the API by doing so:

```ts
import BaseAPI, { APIStatuses } from './sdk.ts';
import config from './config';
import YError from 'yerror';
import type { RequestExecutor, Components } from './sdk.ts';
import type { AxiosRequestConfig } from 'axios';

export { Enums };
export type { Components };

type AuthTokenInput = { token?: string };

const API = Object.keys(BaseAPI).reduce((FinalAPI, operationId) => {
  FinalAPI[operationId] = async (
    { token, ...input }: unknown & AuthTokenInput,
    options: AxiosRequestConfig = {},
  ) => {
    try {
      const response =  await BaseAPI[operationId](
        executeRequest,
        {
          ...input,
          xApplicationVersion: config.applicationVersion,
        },
        {
          ...options,
          baseURL: config.apiURL,
          headers: {
            ...options.headers,
            ...(token
              ? {
                  authorization: `Bearer ${token}`,
                }
              : {}),
          },
        },
      );
      return response;
    } catch (err) {
      console.error('Got an API error:', err.stack);
      throw new YError(
        err.response?.data?.error ? 'E_API_ERROR' : 'E_UNEXPECTED_ERROR',
        err.response?.data,
      );
    }
  };
  return FinalAPI;
}, {}) as {
  [P in keyof typeof BaseAPI]: (
    input: Parameters<typeof BaseAPI[P]>[1] & AuthTokenInput,
    config?: AxiosRequestConfig,
  ) => Promise<ReturnType<typeof BaseAPI[P]>>;
};

export default API;
```

Finally, you may appreciate using it with the
[`useSSR` React hook](https://github.com/vercel/swr) to benefit from your SDK
types:

```ts
import useSWR from 'swr';
import API from './api';

type Handler<I, O> = (input: I) => Promise<O>;
type HandlerInput<T> = T extends Handler<infer I, unknown> ? I : never;
type HandlerOutput<T> = T extends Handler<unknown, infer I> ? I : never;

const API_KEYS: Record<any, string> = Object.keys(API).reduce((hash, key) => {
  hash[API[key]] = key;
  return hash;
}, {});

export default function useAPISWR<T extends Handler<any, any>>(
  swrCouple: [T, HandlerInput<T>],
  options?: Parameters<typeof useSWR>[2],
) {
  const uniqueKey = swrCouple
    ? Object.keys(swrCouple[1]).reduce(
        (finalKey, key) => finalKey + key + JSON.stringify(swrCouple[1][key]),
        // Sadly, here, we cannot rely on `swrCouple[0].name` to
        // build the unicity key since the build destroys it
        API_KEYS[swrCouple[0]] + '-',
      )
    : null;

  return useSWR<
    Awaited<HandlerOutput<T>> extends { body: infer D } ? D : never
  >(
    uniqueKey,
    async () => (await swrCouple[0](swrCouple[1])).body,
    options as any,
  );
}
```

[//]: # (::contents:end)

# API
<a name="module_openapi-ts-sdk-builder"></a>

## openapi-ts-sdk-builder
<a name="module_openapi-ts-sdk-builder..generateSDKFromOpenAPI"></a>

### openapi-ts-sdk-builder~generateSDKFromOpenAPI(openAPIContent, options, [typeOptions]) ⇒ <code>Promise.&lt;string&gt;</code>
Build a JS SDK from an OpenAPI file

**Kind**: inner method of [<code>openapi-ts-sdk-builder</code>](#module_openapi-ts-sdk-builder)  
**Returns**: <code>Promise.&lt;string&gt;</code> - The SDK JS code  

| Param | Type | Description |
| --- | --- | --- |
| openAPIContent | <code>string</code> |  |
| options | <code>Object</code> |  |
| options.sdkVersion | <code>string</code> | The SDK version |
| [options.sdkName] | <code>string</code> | The SDK name (default to API) |
| [options.ignoredParametersNames] | <code>Array.&lt;string&gt;</code> | Provide a list of parameters to ignore |
| [options.undocumentedParametersNames] | <code>Array.&lt;string&gt;</code> | Provide a list of parameters to keep undocumented |
| [typeOptions] | <code>Object</code> | Options to be passed to the type generator |


# Authors
- [Nicolas Froidure](https://insertafter.com/en/index.html)

# License
[MIT](https://github.com/nfroidure/openapi-ts-sdk-builder/blob/main/LICENSE)
 readmeEtag: '"bed392d8c61f5f5b070e1f5c5a763a419fec90a4"' readmeLastModified: Thu, 18 Jul 2024 12:08:27 GMT repositoryId: 292779894 description: Create a TypeScript SDK from an OpenAPI 3 definition created: '2020-09-04T07:26:46Z' updated: '2025-11-06T15:55:32Z' language: TypeScript archived: false stars: 16 watchers: 1 forks: 2 owner: nfroidure logo: https://avatars.githubusercontent.com/u/229633?v=4 license: MIT repoEtag: '"fc2733554b45b28722a8dd2769019e28371806b32f67dd1ef4c7ef8037f8ab69"' repoLastModified: Thu, 06 Nov 2025 15:55:32 GMT id: 6d4b61a2ffce070a86ca3005371b8050 link: https://github.com/nfroidure/openapi-ts-sdk-builder v2: false v3: true v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/nfroidure/schema2dts v3: true repositoryMetadata: base64Readme: >- [//]: # ( )
[//]: # (This file is automatically generated by a `metapak`)
[//]: # (module. Do not change it  except between the)
[//]: # (`content:start/end` flags, your changes would)
[//]: # (be overridden.)
[//]: # ( )
# schema2dts
> A very simple JSONSchema7/OpenAPI3 to TypeScript types definitions generator



[//]: # (::contents:start)

It is intended to support JSONSchema7/OpenAPI3 but may work for some if not most
JSONSchema versions.

This module assumes your JSONSchema / OpenAPI3 documents are valid. It also
doesn't support external references at the moment (and probably for ever) and
expect a single object whose definitions are all relative to the root object.

It is also meant to be a building block for higher level generators.

## Usage

```ts
import { readFile, writeFile } from 'node:fs/promises';
import {
  generateJSONSchemaTypes,
  generateOpenAPITypes,
  toSource,
} from 'schema2dts';

// Open API
const openAPISchema = JSON.parse(readFileSync('openapi.json').toString());

await writeFile(
  'API.d.ts',
  toSource(await generateOpenAPITypes(openAPISchema)),
);

// JSON Schema
const jsonSchema = JSON.parse(readFileSync('schema.json').toString());

await writeFile(
  'API.d.ts',
  toSource(await generateJSONSchemaTypes(jsonSchema)),
);
```

If you find some cases with unexpected results, please add the fixtures to this
repository in a pull request and describe the problem you encounter.

## Options

You can change the API main namespace in order to be able to use several
generated types in the same repository. Just provide its namespace a the second
argument to `generateOpenAPITypes`.

The third argument is for options:

- you can generate the unused schemas (especially useful when in development
  mode) to be able to use them in your code despite the fact they ain't used in
  you API at that moment. Just set `generateUnusedSchemas` to `true`.
- you can also filter the statuses you wish to generate by setting
  `filterStatuses` to `[200, 201, 202, 300]` for example so that the 500 errors
  responses ain't taken in count.

## Known issues

There is some differences between the JSONSchema `anyOf`, `allOf` and `oneOf`
keywords (learn
[more here on combining schemas](https://json-schema.org/understanding-json-schema/reference/combining.html)).

The current way to handle this in this library is to:

- convert `oneOf` to an
  [union type](https://www.typescriptlang.org/docs/handbook/unions-and-intersections.html#union-types)
  which is valid
- convert `anyOf` to an union type too which is not really what it means in JSON
  Schema
- convert `allOf` to an
  [intersection type](https://www.typescriptlang.org/docs/handbook/unions-and-intersections.html#intersection-types)
  which is completly wrong and will work only with JSON Schemas meant to be used
  that way. By example, combining an existing object schema with another object
  to make some properties required will only work if you set its type to
  `object` in both schemas explicitly:

```json
{
  "allOf": [
    {
      "$ref": "#/definitions/User"
    },
    { "type": "object", "required": ["id"] }
  ]
}
```

Currently, the `if`/`then` keywords of JSONSchema do not work. You should be
able to replace most of its use per a `oneOf` form.

[//]: # (::contents:end)

# API
## Functions

<dl>
<dt><a href="#generateOpenAPITypes">generateOpenAPITypes(schema, options)</a> ⇒ <code>TypeScript.NodeArray</code></dt>
<dd><p>Create the TypeScript types declarations from an Open API document</p>
</dd>
<dt><a href="#generateJSONSchemaTypes">generateJSONSchemaTypes(schema, options)</a> ⇒ <code>TypeScript.NodeArray</code></dt>
<dd><p>Create the TypeScript types declarations from a JSONSchema document</p>
</dd>
<dt><a href="#toSource">toSource(nodes)</a> ⇒</dt>
<dd><p>Returns source from a list of TypeScript statements</p>
</dd>
</dl>

<a name="generateOpenAPITypes"></a>

## generateOpenAPITypes(schema, options) ⇒ <code>TypeScript.NodeArray</code>
Create the TypeScript types declarations from an Open API document

**Kind**: global function  

| Param | Type | Description |
| --- | --- | --- |
| schema | <code>JSONSchema.Document</code> |  |
| options | <code>Object</code> |  |
| options.baseName | <code>string</code> |  |
| options.filterStatuses | <code>Array.&lt;number&gt;</code> |  |
| options.generateUnusedSchemas | <code>boolean</code> |  |
| options.camelizeInputs | <code>boolean</code> |  |
| options.brandedTypes | <code>Array.&lt;string&gt;</code> | Brand types by names |
| options.brandedFormats | <code>Array.&lt;string&gt;</code> | Brand formats by names |
| options.typedFormats | <code>Object</code> | Substitute string format by a type |
| options.generateRealEnums | <code>boolean</code> |  |
| options.tuplesFromFixedArraysLengthLimit | <code>number</code> |  |
| options.exportNamespaces | <code>boolean</code> |  |
| options.requireCleanAPI | <code>boolean</code> |  |

<a name="generateJSONSchemaTypes"></a>

## generateJSONSchemaTypes(schema, options) ⇒ <code>TypeScript.NodeArray</code>
Create the TypeScript types declarations from a JSONSchema document

**Kind**: global function  

| Param | Type | Description |
| --- | --- | --- |
| schema | <code>JSONSchema.Document</code> |  |
| options | <code>Object</code> |  |
| options.baseName | <code>string</code> |  |
| options.brandedTypes | <code>Array.&lt;string&gt;</code> | Brand types by names |
| options.brandedFormats | <code>Array.&lt;string&gt;</code> | Brand formats by names |
| options.typedFormats | <code>Object</code> | Substitute string format by a type |
| options.generateRealEnums | <code>boolean</code> |  |
| options.tuplesFromFixedArraysLengthLimit | <code>number</code> |  |
| options.exportNamespaces | <code>boolean</code> |  |

<a name="toSource"></a>

## toSource(nodes) ⇒
Returns source from a list of TypeScript statements

**Kind**: global function  
**Returns**: string  

| Param | Type |
| --- | --- |
| nodes | <code>TypedPropertyDescriptor.NodeArray</code> | 


# Authors
- [Nicolas Froidure](https://insertafter.com/en/index.html)

# License
[ISC](https://github.com/nfroidure/schema2dts/blob/main/LICENSE)
 readmeEtag: '"2e904c7ebb830489a6a7901975088936aa231174"' readmeLastModified: Tue, 06 May 2025 11:42:18 GMT repositoryId: 291741148 description: A very simple JSONSchema to TypeScript types generator created: '2020-08-31T14:41:10Z' updated: '2025-11-06T15:40:42Z' language: TypeScript archived: false stars: 7 watchers: 2 forks: 2 owner: nfroidure logo: https://avatars.githubusercontent.com/u/229633?v=4 license: MIT repoEtag: '"cb3c72b5d4f636f3bbafab9c453a85cc5e50484acc0108bf43ddbe6fd2271e1a"' repoLastModified: Thu, 06 Nov 2025 15:40:42 GMT foundInMaster: true name: schema2dts homepage: https://github.com/nfroidure/schema2dts language: - Javascript - TypeScript source_description: Create types definitions from an OpenAPI schema. id: 8dbfb11a57e76d28203908d316294149 link: https://github.com/nfroidure/schema2dts v2: false category: Parsers - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/blst-security/cherrybomb v3: true repositoryMetadata: base64Readme: >- CjxkaXYgIGFsaWduPSJjZW50ZXIiPgoKIVtjaGVycnlfYm9tYl92MS4wXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYmxzdC1zZWN1cml0eS9jaGVycnlib21iL21haW4vaW1hZ2VzL2NoZXJyeWJvbWJfZ2l0aHViX2FydF92Mi0xJTIwKDEpLnBuZykKCiAgCgo8aDE+U3RvcCBoYWxmLWRvbmUgQVBJIHNwZWNpZmljYXRpb25zPC9oMT4KClshW01haW50YWluZWQgYnkgYmxzdCBzZWN1cml0eV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9tYWludGFpbmVkJTIwYnktYmxzdCUyMHNlY3VyaXR5LTRGNDZFNSldKCkKClshW2RvY3NdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZG9jcy1wYXNzaW5nLWJyaWdodGdyZWVuKV0oKQoKWyFbRGlzY29yZCBTaGllbGRdKGh0dHBzOi8vZGlzY29yZGFwcC5jb20vYXBpL2d1aWxkcy85MTQ4NDY5MzczMjc0OTczMDcvd2lkZ2V0LnBuZz9zdHlsZT1zaGllbGQpXShodHRwczovL2Rpc2NvcmQuZ2cvV2RIaHY0RHF3VSkKCjwvZGl2PgoKICAKCiMg8J+SoyBXaGF0IGlzIENoZXJyeWJvbWI/CgpDaGVycnlib21iIGlzIGFuIENMSSB0b29sIHdyaXR0ZW4gIGluIFJ1c3QgdGhhdCBoZWxwcyBwcmV2ZW50IGluY29ycmVjdCBjb2RlIGltcGxlbWVudGF0aW9uIGVhcmx5IGluIGRldmVsb3BtZW50LiBJdCB3b3JrcyBieSB2YWxpZGF0aW5nIGFuZCB0ZXN0aW5nIHlvdXIgQVBJIHVzaW5nIGFuIE9wZW5BUEkgZmlsZS4gSXRzIG1haW4gZ29hbCBpcyB0byByZWR1Y2Ugc2VjdXJpdHkgZXJyb3JzIGFuZCBlbnN1cmUgeW91ciBBUEkgZnVuY3Rpb25zIGFzIGludGVuZGVkLgoKICAKICAKCiMg8J+UqCBIb3cgZG9lcyBpdCB3b3JrPwoKCkNoZXJyeWJvbWIgbWFrZXMgc3VyZSB5b3VyIEFQSSBpcyB3b3JraW5nIGNvcnJlY3RseS4gSXQgY2hlY2tzIHlvdXIgQVBJJ3Mgc3BlYyBmaWxlIChPcGVuQVBJIFNwZWNpZmljYXRpb24pIGZvciBnb29kIHByYWN0aWNlcyBhbmQgbWFrZXMgc3VyZSBpdCBmb2xsb3dzIHRoZSBPQVMgcnVsZXMuIFRoZW4sIGl0IHRlc3RzIHlvdXIgQVBJIGZvciBjb21tb24gaXNzdWVzIGFuZCB2dWxuZXJhYmlsaXRpZXMuIElmIGFueSBwcm9ibGVtcyBhcmUgZm91bmQsIENoZXJyeWJvbWIgZ2l2ZXMgeW91IGEgZGV0YWlsZWQgcmVwb3J0IHdpdGggdGhlIGV4YWN0IGxvY2F0aW9uIG9mIHRoZSBwcm9ibGVtIHNvIHlvdSBjYW4gZml4IGl0IGVhc2lseS4KCiAgCgojIPCfkL4gR2V0IFN0YXJ0ZWQKCiMjIEluc3RhbGxhdGlvbgoKCgojIyMjIyBMaW51eC9NYWNPUzoKCmBgYAoKREVQUkVDQVRFRCBGT1IgTk9XCmBgYAoKVGhlIHNjcmlwdCByZXF1aXJlcyBzdWRvIHBlcm1pc3Npb25zIHRvIG1vdmUgdGhlIGNoZXJyeWJvbWIgYmluIGludG8gPGI+L3Vzci9sb2NhbC9iaW4vPC9iPi48L2JyPgoKKElmIHlvdSB3YW50IHRvIHZpZXcgdGhlIHNoZWxsIHNjcmlwdChvciBldmVuIGhlbHAgdG8gaW1wcm92aW5nIGl0IC0gWy9zY3JpcHRzL2luc3RhbGwuc2hdKC9zY3JpcHRzL2luc3RhbGwuc2gpKQoKICMjIyMjIENvbnRhaW5lcml6ZWQgdmVyc2lvbgogWW91IGNhbiBnZXQgQ2hlcnJ5Ym9tYiB0aHJvdWdoIGl0cyBjb250YWluZXJpemVkIHZlcnNpb24gd2hpY2ggaXMgaG9zdGVkIG9uIEFXUyBFQ1IsIGFuZCByZXF1aXJlcyBhbiBBUEkga2V5IHRoYXQgeW91IGNhbiBnZXQgb24gdGhhdCBhZGRyZXNzKHRoZSBsb2FkaW5nIGlzIGEgYml0IHNsb3cpIC0gREVQUkVDQVRFRCBGT1IgTk9XCgpgYGAKZG9ja2VyIHJ1biAtLW1vdW50IHR5cGU9YmluZCxzb3VyY2U9W1BBVEggVE8gT0FTXSxkZXN0aW5hdGlvbj0vaG9tZSBwdWJsaWMuZWNyLmF3cy9ibHN0LXNlY3VyaXR5L2NoZXJyeWJvbWI6bGF0ZXN0IGNoZXJyeWJvbWIgLWYgL2hvbWUvW09BUyBOQU1FXSAtLWFwaS1rZXk9W0FQSS1LRVldCmBgYAoKIyMjIyBHZXQgaXQgZnJvbSBjcmF0ZXMuaW8KCmBgYGJhc2gKCmNhcmdvIGluc3RhbGwgY2hlcnJ5Ym9tYgoKYGBgCgpJZiB5b3UgZG9uJ3QgaGF2ZSBjYXJnbyBpbnN0YWxsZWQsIHlvdSBjYW4gaW5zdGFsbCBpdCBmcm9tIFtoZXJlXShodHRwczovL2RvYy5ydXN0LWxhbmcub3JnL2NhcmdvL2dldHRpbmctc3RhcnRlZC9pbnN0YWxsYXRpb24uaHRtbCkKCgoKIyMjIyBCdWlsZGluZyBmcm9tIFNvdXJjZXMKCllvdSBjYW4gYWxzbyBidWlsZCBDaGVycnlib21iIGZyb20gc291cmNlcyBieSBjbG9uaW5nIHRoaXMgcmVwbywgYW5kIGJ1aWxkaW5nIGl0IHVzaW5nIGNhcmdvLgoKYGBgCgpnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2Jsc3Qtc2VjdXJpdHkvY2hlcnJ5Ym9tYiAmJiBjZCBjaGVycnlib21iCgpgYGAKVGhlIG1haW4gYnJhbmNoJ3MgQ2FyZ28udG9tbCBmaWxlIHVzZXMgYGNoZXJyeWJvbWItZW5naW5lYCBhbmQgYGNoZXJyeWJvbWItb2FzYCBmcm9tIGNyYXRlcy5pby4gCgppZiB5b3Ugd2FudCBidWlsZCB0aG9zZSBmcm9tIHNvdXJjZSB0b28sIHlvdSBjYW4gY2hhbmdlIHRoZSBmb2xsb3dpbmcgZmlsZXM6CgoocmVtb3ZlIHRoZSB2ZXJzaW9uIG51bWJlciBhbmQgcmVwbGFjZSB3aXRoIHRoZSBwYXRoIHRvIHRoZSBsb2NhbCByZXBvKQoKCgpgYGAKY2hlcnJ5Ym9tYi9DYXJnby50b21sOgpjaGVycnlib21iLWVuZ2luZSA9IHZlcnNpb24gPT4geyBwYXRoID0gImNoZXJyeWJvbWItZW5naW5lIiB9CmBgYAogCmBgYApjaGVycnlib21iL2NoZXJyeWJvbWItZW5naW5lL0NhcmdvLnRvbWw6CmNoZXJyeWJvbWItb2FzID0gdmVyc2lvbiA9PiB7IHBhdGggPSAiLi4vY2hlcnJ5Ym9tYi1vYXMiIH0KYGBgCgpgYGAKY2FyZ28gYnVpbGQgLS1yZWxlYXNlCnN1ZG8gbXYgLi90YXJnZXQvcmVsZWFzZS9jaGVycnlib21iIC91c3IvbG9jYWwvYmluICMgb3IgYW55IG90aGVyIGRpcmVjdG9yeSBpbiB5b3VyIFBBVEgKYGBgCiAgCgogIAojIyMgUHJvZmlsZSAKIApQcm9maWxlcyBhbGxvdyB5b3UgdG8gY2hvb3NlIHRoZSB0eXBlIG9mIGNoZWNrIHlvdSB3YW50IHRvIHVzZS4KYGBgCi0gaW5mbzogb25seSBnZW5lcmF0ZXMgcGFyYW0gYW5kIGVuZHBvaW50IHRhYmxlcwotIG5vcm1hbDogIGJvdGggYWN0aXZlIGFuZCBwYXNzaXZlCi0gaW50cnVzaXZlOiBhY3RpdmUgYW5kIGludHJ1c2l2ZSBbaW4gZGV2ZWxvcG1lbnRdCi0gcGFzc2l2ZTogb25seSBwYXNzaXZlIHRlc3RzCi0gZnVsbDogYWxsIHRoZSBvcHRpb25zCmBgYAoKIyMjIENvbmZpZyAKCgoKV2l0aCBhIGNvbmZpZ3VyYXRpb24gZmlsZSwgeW91IGNhbiBlYXNpbHkgZWRpdCwgdmlldywgQ2hlcnJ5Ym9tYidzIG9wdGlvbnMuClRoZSBjb25maWcgZmlsZSBhbGxvd3MgeW91IHRvIHNldCB0aGUgcnVubmluZyBwcm9maWxlLCBsb2NhdGlvbiBvZiB0aGUgb2FzIGZpbGUsIHRoZSB2ZXJib3NpdHkgYW5kIGlnbm9yZSB0aGUgVExTIGVycm9yLgoKQ29uZmlnIGFsc28gYWxsb3dzIHlvdSB0byBvdmVycmlkZSB0aGUgc2VydmVyJ3MgVVJMIHdpdGggYW4gYXJyYXkgb2Ygc2VydmVycywgYW5kIGFkZCBzZWN1cml0eSB0byB0aGUgcmVxdWVzdCBbaW4gZGV2ZWxvcG1lbnRdLiAKCk5vdGljZSB0aGF0IENMSSBhcmd1bWVudHMgcGFyYW1ldGVyIHdpbGwgb3ZlcnJpZGUgY29uZmlnIG9wdGlvbnMgaWYgYm90aCBhcmUgc2V0LgoKWW91IGNhbiBhbHNvIGFkZCBvciByZW1vdmUgY2hlY2tzIGZyb20gYSBwcm9maWxlIHVzaW5nIGBwYXNzaXZlL2FjdGl2ZS1pbmNsdWRlL2V4Y2x1ZGVgLiBbaW4gZGV2ZWxvcG1lbnRdCgpgYGAKY2hlcnJ5Ym9tYiAtLWNvbmZpZyAgPENPTkZJR19GSUxFPgpgYGAKCgpTdHJ1Y3R1cmUgb2YgY29uZmlnIGZpbGU6CmBgYAp7CiJmaWxlIiA6ICJvcGVuLWFwaS5qc29uIiwKInZlcmJvc2l0eSIgOiAibm9ybWFsLCAKInByb2ZpbGUiIDogIiAiTm9ybWFsIiwKInBhc3NpdmVfaW5jbHVkZSIgOiBbImNoZWNrMSwgY2hlY2tzMiJdLAoiYWN0aXZlX2luY2x1ZGUiOiBbImNoZWNrMywgY2hlY2s0Il0sCiJzZXJ2ZXJzX292ZXJyaWRlIiAsIFsiaHR0cDovL3NlcnZlci8iXSwKInNlY3VyaXR5IjogIFt7CiAgICAiYXV0aF90eXBlIjogIkJhc2ljIiwKICAgICJhdXRoX3ZhbHVlIiA6IHRva2VuX3ZhbHVlLAogICAgImF1dGhfc2NvcGUiIDogc2NvcGVfbmFtZQogICAgfV0sCiJpZ25vcmVfdGxzX2Vycm9ycyIgOiB0cnVlLCAKIm5vX2NvbG9yIiA6IGZhbHNlLAp9CmBgYAoKCgojIFVzYWdlCgpBZnRlciBpbnN0YWxsaW5nLCB2ZXJpZnkgaXQncyB3b3JraW5nIGJ5IHJ1bm5pbmcKCmBgYApjaGVycnlib21iIC0tdmVyc2lvbgoKYGBgCgojIyMgT3BlbkFQSSBzcGVjaWZpY2F0aW9uCgoKYGBgIGNoZXJyeWJvbWIgLS1maWxlIDxQQVRIPiAtLXByb2ZpbGUgcGFzc2l2ZSBgYGAKClBhc3NpdmUgT3V0cHV0IGV4YW1wbGU6CgohW3Bhc3NpdmVfb3V0cHV0XShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYmxzdC1zZWN1cml0eS9jaGVycnlib21iL21haW4vaW1hZ2VzL3Bhc3NpdmUxXzAucG5nKQoKCiMjIyBHZW5lcmF0ZSBJbmZvIFRhYmxlCgoKYGBgCmNoZXJyeWJvbWIgLS1maWxlIDxQQVRIPiAtLXByb2ZpbGUgaW5mbwoKYGBgClBhcmFtZXRlciB0YWJsZSBvdXRwdXQ6CgogICFbcGFyYW1ldGVyX291dHB1dF0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2Jsc3Qtc2VjdXJpdHkvY2hlcnJ5Ym9tYi9tYWluL2ltYWdlcy9wYXJhbV92MS5wbmcpCgpFbmRwb2ludCB0YWJsZSBvdXRwdXQ6CgogICFbZW5kcG9pbnRfb3V0cHV0XShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYmxzdC1zZWN1cml0eS9jaGVycnlib21iL21haW4vaW1hZ2VzL2VuZHBvaW50X3YxLnBuZykKCgoKCiMg8J+NuyBJbnRlZ3JhdGlvbgoKREVQUkVDQVRFRCBGT1IgTk9XIC0gV0lMTCBCRSBSRVBMQUNFRCBTT09OCgpZb3UgY2FuIGVtYmVkIGl0IGludG8geW91ciBDSSBwaXBlbGluZSwgYW5kIElmIHlvdSBwbGFuIG9uIGRvaW5nIHRoYXQgSSB3b3VsZCByZWNvbW1lbmQgdGhhdCB5b3UgZ28gdG8gb3VyIFt3ZWJzaXRlXSgpLCBzaWduIHVwLCBnbyB0aHJvdWdoIHRoZSBbQ0kgcGlwZWxpbmUgaW50ZWdyYXRpb24gd2l6YXJkXSgpLCBhbmQgY29weSB0aGUgZ3Jvb3Z5L0dpdEh1YiBhY3Rpb25zIHNuaXBwZXQgYnVpbHQgZm9yIHlvdS4KCjwvYnI+RXhhbXBsZToKCiFbQ0kgcGlwZWxpbmUgYnVpbGRlciBvdXRwdXRdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9ibHN0LXNlY3VyaXR5L2NoZXJyeWJvbWIvbWFpbi9pbWFnZXMvY2lfb3V0cHV0LnBuZykKCiMg8J+SqiBTdXBwb3J0CgogIAoKIyMjIEdldCBoZWxwCgpJZiB5b3UgaGF2ZSBhbnkgcXVlc3Rpb25zIHlvdSBjYW4gYXNrIHVzIG9uIG91ciBbZGlzY29yZCBzZXJ2ZXJdKGh0dHBzOi8vZGlzY29yZC5nZy9XZEhodjREcXdVKS4KCgpZb3UgYXJlIGFsc28gd2VsY29tZSB0byBvcGVuIGFuIElzc3VlIGhlcmUgb24gR2l0SHViLgo= readmeEtag: '"1d2f3098a3e03c3c375f1ee43b453d76587fa2ae"' readmeLastModified: Fri, 25 Oct 2024 10:03:59 GMT repositoryId: 428941557 description: >- Stop half-done APIs! Cherrybomb is a CLI tool that helps you avoid undefined user behaviour by auditing your API specifications, validating them and running API security tests. created: '2021-11-17T07:02:22Z' updated: '2026-02-03T20:06:51Z' language: Rust archived: false stars: 1231 watchers: 12 forks: 82 owner: blst-security logo: https://avatars.githubusercontent.com/u/54233087?v=4 license: Apache-2.0 repoEtag: '"674d6fcd3b7aeb8fbbb80d8f12fe2f2f62184c225cfe9d3611eff833fe4da3ff"' repoLastModified: Tue, 03 Feb 2026 20:06:51 GMT foundInMaster: true name: Cherrybomb category: - Description Validators - Server Implementations language: Rust link: https://www.blstsecurity.com/cherrybomb source_description: >- A CLI tool that helps avoid undefined user behaviour by validating your API descriptions, to make sure key parts are not missing or vague. v2: false v3_1: true id: ddbbcb536a870b27a61df0b41529385d - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/ignatandrei/blocklyautomation v3: true repositoryMetadata: base64Readme: >- # VisualAPI - Add ( to your |  inside your ) application LowCode Macros for YOUR API / HTTP Calls

<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-2-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END -->

Blockly Automation :  is a tool for LowCode / automation of sites (Swagger / OpenAPI / RPA ) and PC made of Blockly. 

# How to use if you are

## a backend developer with an REST / OpenAPI / Swagger enabled site and I want to show the use of my site


Use the appropiate package: 

### [NetCore](https://github.com/ignatandrei/netcoreblockly/)  , made by Andrei Ignat

.NET Core: [![Nuget](https://img.shields.io/nuget/dt/NetCore2Blockly)](https://www.nuget.org/packages/NetCore2Blockly/)

More details at https://github.com/ignatandrei/netcoreblockly/


### [Java](https://github.com/eciuca/blockly-automation-starter-webmvc-ui) , Full example by Emanuel Ciuca 

Java Spring : [![Maven Central](https://img.shields.io/maven-central/v/io.github.eciuca.blockly/blockly-automation-starter-webmvc-ui)](https://central.sonatype.com/artifact/io.github.eciuca.blockly/blockly-automation-starter-webmvc-ui/)

The Maven is https://central.sonatype.com/artifact/io.github.eciuca.blockly/blockly-automation-starter-webmvc-ui/

The repo with example is https://github.com/eciuca/blockly-automation-starter-webmvc-ui 


### [Node](https://github.com/ignatandrei/BlocklyAutomation/wiki/node) , made by Andrei Ignat

Node: [![npm](https://img.shields.io/npm/v/node2-blockly)](https://www.npmjs.com/package/node-blockly?)

More details at https://github.com/ignatandrei/BlocklyAutomation/wiki/node

### [PHP](https://github.com/Tynael/laravel-blockly-automation) , made by Carol Pelu



Packagist : [![php](https://img.shields.io/packagist/v/carolpelu/blockly-automation)](https://packagist.org/packages/carolpelu/blockly-automation)

More details at https://github.com/Tynael/laravel-blockly-automation




## Docker Extension

https://open.docker.com/extensions/marketplace?extensionId=ignatandrei/blockly-automation


### Manual

Download the HTML release [BlocklyAutomation](https://github.com/ignatandrei/BlocklyAutomation/releases/latest/download/releaseBlocklyAutomation.zip/) .

Follow the instructions in the wiki file.  

#### [Manual Java](https://github.com/cosminpopescu14/math-operations-swagger) , Manual example by Cosmin Popescu 

Example by Cosmin Popescu at https://github.com/cosminpopescu14/math-operations-swagger



## a frontend  developer that finds a bug in a REST / OpenAPI / Swagger enabled site

You want to show to the backend developer how to reproduce the problem .

If you control the site , then you can use the [BlocklyAutomation](https://github.com/ignatandrei/BlocklyAutomation/releases/latest/download/releaseBlocklyAutomation.zip/) to reproduce the problem.

Follow the instructions in the wiki file.  

If you do not control the site, then install the application from http://ba.serviciipeweb.ro/ and then use it to reproduce the problem. ( wiki site coming with details) 

## a web site application tester
If you control the site , then can use the https://github.com/ignatandrei/BlocklyAutomation/releases/latest/download/releaseBlocklyAutomation.zip/  and then use it to reproduce the problem. ( wiki site coming with details) .

If you does not control the site, then install the application from http://ba.serviciipeweb.ro/ and then use it to make test cases. ( wiki site coming with details) 


## [Docker Extension](https://github.com/ignatandrei/BlocklyAutomation/wiki/DockerExtension)

Please see [Docker Extension](https://github.com/ignatandrei/BlocklyAutomation/wiki/DockerExtension)
See 
## not a programmer.  I want to automate/gather data from several sites ( public or private )

You want to obtain some data from the web. For example, extract the exchange between EUR / RON .

Install the application from http://ba.serviciipeweb.ro/ .

Follow the instructions in the wiki file.


## not a programmer andI want to automate things on my PC. 

You want to obtain some data from your PC. For example, extract and export to CSV the Chrome Bookmarks.

Install the application from http://ba.serviciipeweb.ro/ .

Follow the instructions in the wiki file.



## Suggest a public API for BlocklyAutomation

If you have a public API/site that you want to automate, please send file an issue at https://github.com/ignatandrei/BlocklyAutomation/issues 


## How to see a preview


To see the whole potential , please go to http://ba.serviciipeweb.ro/ and click Launch .

If you want to see some web Demo , please go to https://ignatandrei.github.io/BlocklyAutomation/ and test various HTTP request, Swaggers and more

## How to install on your PC

See [releases tab](https://github.com/ignatandrei/BlocklyAutomation/releases)

You can find the 

1. Angular site , ready to be deployed to any server
2. IIS site   , ready to be deployed on IIS
3. A .NET Core site, ready to be deployed on Linux or Windows, standalone ( as a service )

# You have a site with OpenAPI / Swagger. What should I do  ?

Download from [releases tab](https://github.com/ignatandrei/BlocklyAutomation/releases) the Angular site. Put index.html and all other files into a BLocklyAutomation folder inside your wwwroot  folder ( or in your project root ) and you are good to go.

1. It could be an idea to map everything that start with /BlocklyAutomation to /BlocklyAutomation/index.html - see src\Loaders\SimpleSite to have an C# example 
   
2. Modify assets/settings.json to change the name and the starting blocks
   
3. Modify assets/loadAtStartup/swaggers.json to add your swaggers

4. Modify assets/showUsage/demoBlocks/all.txt to add your demo for the blocks . 
   ( You can construct and then download and save as files)

5. Send me an email to help you  if something  is not working.
## How to customize

To customize the title , introduction and start blocks , see assets/settings.json

To load swaggers at startup, see  assets/loadAtStartup/swaggers.json

To customize demo blocks,download your blocks, put the txt file in  assets/showUsage/demoBlocks/ and modify assets/showUsage/demoBlocks/all.txt

##  How to contribute

If you are a beginner to blockly, see 

https://github.com/ignatandrei/BlocklyAutomation/issues?q=is%3Aissue+is%3Aopen+label%3A%22good+first+issue%22

![GitHub issues by-label](https://img.shields.io/github/issues/ignatandrei/BlocklyAutomation/good%20first%20issue)

You will be mentioned below ;-)


## Contributors ✨

Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://aenyx-designs.com/"><img src="https://avatars.githubusercontent.com/u/33196341?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Bogdan Bobe</b></sub></a><br /><a href="#design-arealshadow" title="Design">🎨</a> <a href="https://github.com/ignatandrei/BlocklyAutomation/commits?author=arealshadow" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/adrian-badulescu"><img src="https://avatars.githubusercontent.com/u/49490946?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Adrian Badulescu</b></sub></a><br /><a href="#example-adrian-badulescu" title="Examples">💡</a></td>
  </tr>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome! readmeEtag: '"0b8a3b0a75a8247f4625843bda3826f655a7eb35"' readmeLastModified: Tue, 23 Jul 2024 17:48:16 GMT repositoryId: 427082891 description: >- VisualAPI - LowCode Macros for YOUR API / HTTP Calls : https://visualapi.azurewebsites.net/ created: '2021-11-11T17:14:47Z' updated: '2025-03-23T17:11:34Z' language: JavaScript archived: false stars: 40 watchers: 3 forks: 10 owner: ignatandrei logo: https://avatars.githubusercontent.com/u/153982?v=4 license: MIT repoEtag: '"b66fb339c972682afb29c2a3b793dd7f021da2db621a9b899a2fe2e1025a692a"' repoLastModified: Sun, 23 Mar 2025 17:11:34 GMT foundInMaster: true category: - Code Generators - SDK - Documentation - Testing id: 44b9d7b1e68471c05b67c3ff6c02172a name: BlocklyAutomation link: https://ignatandrei.github.io/BlocklyAutomation/ language: - Javascript - .NET source_description: >- Input any OpenAPI document to have generated Blocks in Blockly form to test and generate documentation. v3_1: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/embraser01/typoas v3: true repositoryMetadata: base64Readme: >- # Typoas

![npm](https://img.shields.io/npm/v/@typoas/cli?label=cli%40npm)  
![npm](https://img.shields.io/npm/v/@typoas/generator?label=generator%40npm)  
![npm](https://img.shields.io/npm/v/@typoas/runtime?label=runtime%40npm)  
![GitHub](https://img.shields.io/github/license/embraser01/typoas)  
![npm bundle size](https://img.shields.io/bundlephobia/minzip/@typoas/runtime)

Typoas is an OpenAPI 3.X generator for Typescript. It's inspired by [openapi-generator](https://openapi-generator.tech/)  
but is written in Typescript for Typescript. The generator uses the Typescript AST to generate code instead on relaying  
on templates which allows better schemas definitions and other cool stuff.

Main features are:

- Fully typed and fully customizable
- References `$ref` handling (including cyclic refs)
- **Tree Shaking** out of the box
- **React Query** integration
- Support for `allOf`, `oneOf` and `anyOf` schemas.
- Automatically convert `format: 'date-time'` to JS `Date`
- Handle **API Key**, **HTTP Config**, **OAuth2**<sup>1</sup> and **OIDC**<sup>1</sup> auth security schemes
- JSDoc for schemas and operations
- Override system for generated types
- Uses `fetch` api (can be customized)
- Non JSON content type support
- Small bundle size
- And more...

> <sup>1</sup>: OAuth2 and OpenIDConnect scheme do not handle flows to retrieve an `accessToken`.  
> You need to provide your own `accessToken` through the `provider.getConfig()` function.

The project is split into 4 packages:

- [`@typoas/generator`](./packages/typoas-generator) is used to generate the API specific code.
- [`@typoas/cli`](./packages/typoas-cli) is a CLI entry point built on top of `@typoas/generator`.
- [`@typoas/runtime`](./packages/typoas-runtime) is the package that will be used by the generated code.
- [`@typoas/react-query`](./packages/typoas-react-query) integrates _Typoas_ with [React Query](https://tanstack.com/query/latest).

---

- [Installation](#installation)
- [Usage](#usage)
  - [Use the generator from the CLI](#use-the-generator-from-the-cli)
  - [Use the generator from the API](#use-the-generator-from-the-api)
  - [Use the generated code](#use-the-generated-code)
    - [Customize server configuration](#customize-server-configuration)
    - [Customize fetch implementation](#customize-fetch-implementation)
    - [Handling authentification](#handling-authentification)
    - [Customizing the serialization](#customizing-the-serialization)
  - [Using with React Query](#using-with-react-query)
- [Examples](#examples)
- [Notes](#notes)
  - [Overrides](#overrides)
  - [External references](#external-references)
  - [Parameters serialization](#parameters-serialization)
  - [Migrating from v1 to v2](#migrating-from-v1-to-v2)
- [Contributing](#contributing)
- [License](#license)

## Installation

It will generate a single TS file containing all the code specific to the underlying API.  
This file only has a single dependency on `@typoas/runtime`.  
**You need to manually** add `@typoas/runtime` to your `dependencies`.  
It handles common things like serialization/authentification

## Usage

### Use the generator from the CLI

You can generate the TS client from the spec from the command line:

```bash
yarn dlx @typoas/cli generate -i my-spec.json -o src/client.ts
npx @typoas/cli generate -i my-spec.json -o src/client.ts
```

Here is a short list of supported command line options:

```
    -i, --input [path/url]         Path or URL to the OpenAPI JSON specification (yaml/json format)
    -o, --output [path]            Path where to write the generated TS file
    -e,--generate-enums            Generate enums instead of literal string types where possible
    -p,--prettier                  If set, the generated file will be formatted with Prettier
    -r,--full-response-mode        Enabled by default, generate functions that only throws on network errors, can be disabled using --no-full-response-mode
    --js-doc, --no-js-doc          Whether to add JS Doc to the generated code (default: true)
    --wrap-lines-at                Define a maximum width for JS Doc comments, 0 to disable (default: 120)
    --only-types                   Use it to only generate types in #components/schemas/
    --no-fetcher-options           Use it to disable the additional param added to every operations
    --override                     You can define a list of types that will be imported from a custom file instead of being generated. Must be used with --override-import
    --override-import              Path to the file that will be imported to resolve overrides
    --version                      Output the version number
    -h, --help                     Display help for command
```

or you can use it in code:

```ts
import cli from '@typoas/cli';

await cli.run(
  [
    'generate',
    '-i',
    'https://raw.githubusercontent.com/github/rest-api-description/main/descriptions/api.github.com/api.github.com.yaml',
    '-o',
    `./src/github.ts`,
  ],
  {
    stdin: process.stdin,
    stdout: process.stdout,
    stderr: process.stderr,
  },
);
```

### Use the generator from the API

> The API is still at an **alpha** stage, so it may break between minors.

It uses `typescript` API to generate usable code:

```typescript
import { readFileSync, writeFileSync } from 'fs';
import { createPrinter, NewLineKind, SourceFile } from 'typescript';
import { generateClient, getStringFromSourceFile } from '@typoas/generator';

const specs = JSON.parse(readFileSync('path/to/github-openapi.json', 'utf8'));
const src = generateClient(specs, {
  /* options */
});
const data = getStringFromSourceFile(src);

writeFileSync('./src/client.ts', data, 'utf8');
```

### Use the generated code

Once the file is generated you'll be able to use it like this:

```typescript
import { ServerConfiguration, ok } from '@typoas/runtime';
// Client generated from:
// yarn dlx @typoas/cli generate -i https://raw.githubusercontent.com/github/rest-api-description/main/descriptions/api.github.com/api.github.com.yaml -o client.ts
import { createContext, pullsList } from './client';

const ctx = createContext();

// By default, returns a StatusResponse that contains statusCode, data and headers.
const res = await pullsList(ctx, { repo: 'typoas', owner: 'embraser01' });
if (res.ok) {
  res.data.forEach((pr) => console.log(pr.title));
}

// An helper `ok` function can help throw on non 2xx status code and return the data
// The second argument is an optional expected status code (useful when multiple success status code are possible)
const prs = await ok(
  pullsList(ctx, { repo: 'typoas', owner: 'embraser01' }),
  200,
);
```

> You can directly generate functions that throw on non 2xx status code by using the `--no-full-response-mode` option.
> This feature may be removed in the future and isn't the recommended way.

You can customize multiple things in the `createContext` function.

#### Customize server configuration

By default, the `createContext` function will use the first `servers` entry in the spec.
You can override the server configuration by passing a `ServerConfiguration` object:

```typescript
import { ServerConfiguration } from '@typoas/runtime';
import { createContext } from './client';

const ctx = createContext({
  serverConfiguration: new ServerConfiguration(
    'https://{env}.api.example.com',
    {
      env: 'staging',
    },
  ),
});
```

> The way it's done currently is not the best. It will be improved in the future.

#### Customize fetch implementation

The default fetcher implementation [uses](./packages/typoas-runtime/src/fetcher/index.ts) the `fetch` API.
You can override the fetcher implementation if you want to customize things (like adding a retry policy, headers, etc.).

For example, you can use `axios` instead of the `fetch` API:

```typescript
import axios, { AxiosError, AxiosResponse, Method } from 'axios';
import {
  Fetcher,
  RequestContext,
  ResponseContext,
  BaseFetcherData,
} from '@typoas/runtime';
import { createContext } from './client';

class AxiosHttpLibrary implements Fetcher<BaseFetcherData> {
  async send(request: RequestContext): Promise<ResponseContext> {
    let resp: AxiosResponse;
    try {
      resp = await axios.request({
        url: request.getUrl(),
        method: request.getHttpMethod() as Method,
        headers: request.getHeaders(),
        data: request.getBody(),
      });
    } catch (e) {
      const err = e as AxiosError;
      if (!err.response) {
        throw e;
      }
      // Typoas handles errors itself
      resp = err.response;
    }

    return new ResponseContext(resp.status, resp.headers, {
      text: async () => resp.data as string,
      binary: async () => resp.data as Blob,
      json: async () => resp.data,
    });
  }
}

const ctx = createContext({ fetcher: new AxiosHttpLibrary() });
```

> Node.js >= 18.0.0 includes the `fetch` implementation, so Typoas should work out of the box.

#### Handling authentification

The `createContext` function can take an `authProviders` object to handle authentification.

For example, for the petstore sample:

```typescript
import { createContext } from './petstore';

const ctx = createContext({
  authProviders: {
    api_key: {
      async getConfig() {
        return 'MyAPIKey';
      },
    },
    petstore_auth: {
      getConfig() {
        const accessToken = globalStore.getAccessToken();

        if (accessToken) {
          return null;
        }
        return { accessToken };
      },
    },
  },
});
```

It supports 5 types of security schemes:

- `apiKey` mode
- `http` bearer and basic mode
- `oauth2` mode
- `openIdConnect` mode

The `getConfig` function should return the configuration needed to authenticate the request. Returning `null` will skip the authentification.

#### Customizing the serialization

You can switch some serialization options by passing a `serializerOptions` object to the `createContext` function.

> Not every serialization option is supported. See [#11](https://github.com/Embraser01/typoas/issues/11) for more information.

### Using with React Query

Documentation is available in the [`@typoas/react-query`](./packages/typoas-react-query) package.

## Examples

You can find examples in the [`examples`](./examples) folder.

## Notes

Here is some notes on some known issues.

### Overrides

Sometimes you may want to override some of the generated types.
For example, Typoas will not generate Template Literal Types or Generics as it's hard to be defined in an OpenAPI spec.

This works by listing the types you want to override and giving the import path you want to use.

For example, if you want to override the PetStore `Category` type of your spec:

```typescript
// my-overrides.ts
export type Category = {
  id?: number;
  name?: string;
  description?: string;
};
```

When you generate the client, you can pass the `--overrides` option:

```bash
yarn dlx @typoas/cli generate -i my-spec.json -o src/client.ts --override-import my-overrides.ts --override Category
```

This will not generate the `Category` type but import it from the `my-overrides.ts` file. This allows easy customization of generated types.

There are a couple of things to note:

- The override name is the **sanitized** name of the type. You can first generate the client and see the generated types to know the name.
- _Typoas_ will not check the override file. If the file isn't there or the type isn't exported, it will throw a TypeScript error.
- You will have to export the type yourself, _Typoas_ will not do it for you (contrary to the generated types).

### External references

External references are not supported. Every `$ref` must be in the spec.  
An issue is open [here](https://github.com/Embraser01/typoas/issues/37).

### Parameters serialization

_Typoas_ has partial support for serialization specified here: https://swagger.io/docs/specification/serialization/

- It **does** support array serialization for **query**.
- It **does NOT** support serialization for **path** parameters with `style` `label` or `matrix`.
- It **does NOT** support serialization for **query** parameters with nested objects. It will be JSON.stringify
- It **does NOT** support serialization for **headers** or **cookie** parameters.

On query serialization, there can only be one style for a full operation. The first query param will set the style for  
the whole operation.

### Migrating from v1 to v2

In v1, the whole API was generated in a single class. In V2 this was replaced by individual function  
which allow [Tree Shaking](https://webpack.js.org/guides/tree-shaking/). To get a similar result,  
you can use `wrapApi` helper:

```typescript
import { ServerConfiguration, wrapApi } from '@typoas/runtime';
import { createContext, pullsList, issuesList } from './client';

const ctx = createContext();
const ghClient = wrapApi(ctx, {
  pullsList,
  issuesList,
  // ...
});

ghClient
  .pullsList(ctx, {
    repo: 'typoas',
    owner: 'embraser01',
  })
  .then((list) => console.log('List of PRs', list))
  .catch((err) => console.error('Error while getting PRs', err));
```

## Contributing

Pull requests are welcome. For major changes, please open an issue first to discuss what you would like to change.

## License

[MIT](https://choosealicense.com/licenses/mit/)
 readmeEtag: '"3730b83a6017f2e5a05be738dfc73fa5ee07a56a"' readmeLastModified: Mon, 10 Feb 2025 11:03:02 GMT repositoryId: 350501832 description: Open API (OAS 3.0) Typescript generator created: '2021-03-22T22:01:11Z' updated: '2025-08-11T20:51:42Z' language: TypeScript archived: false stars: 43 watchers: 2 forks: 6 owner: Embraser01 logo: https://avatars.githubusercontent.com/u/8802277?v=4 license: MIT repoEtag: '"f81f97d79406aca7cc292a310d6311b9e84fed87a27b9d508a6105775e9495d5"' repoLastModified: Mon, 11 Aug 2025 20:51:42 GMT foundInMaster: true category: - SDK - Parsers id: daab13b9706c2ce0fb810cc426f403b1 name: Typoas link: https://github.com/Embraser01/typoas language: Typescript source_description: Fully typed OpenAPI Typescript generator v2: false v3_1: true - source: https://openapi.tools/ name: Zuplo (OpenAPI-based gateway and documentation) category: - GUI Editors - Gateway - Documentation language: - Web - SaaS link: https://www.zuplo.com source_description: >- Zuplo is an API gateway designed for developers. Natively powered by OpenAPI (3.1 or 3.0), zuplo offers an OpenAPI design surface, API documentation and a serverless, programmable edge gateway that includes request validation, auth, rate-limiting and more. v2: false v3: true v3_1: true id: 94f410082dd5e91b5d75df09846e8687 foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: kotlin-openapi3-dsl category: - DSL - Parsers language: Kotlin source_description: >- kotlin-openapi3-dsl is a DSL written in Kotlin to write OpenAPI descriptions in plain Kotlin. link: https://github.com/derveloper/kotlin-openapi3-dsl v2: false v3: true repository: https://github.com/derveloper/kotlin-openapi3-dsl id: f4ab0e00ab487873f90c7f20db81bf8b repositoryMetadata: base64Readme: >- IyBrb3RsaW4tb3BlbmFwaTMtZHNsCgohW0phdmEgQ0kgd2l0aCBHcmFkbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9kZXJ2ZWxvcGVyL2tvdGxpbi1vcGVuYXBpMy1kc2wvd29ya2Zsb3dzL0phdmElMjBDSSUyMHdpdGglMjBHcmFkbGUvYmFkZ2Uuc3ZnKQpbIVtNYXZlbiBDZW50cmFsXShodHRwczovL21hdmVuLWJhZGdlcy5oZXJva3VhcHAuY29tL21hdmVuLWNlbnRyYWwvY2MudmlsZWRhL2tvdGxpbi1vcGVuYXBpMy1kc2wvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9tYXZlbi1iYWRnZXMuaGVyb2t1YXBwLmNvbS9tYXZlbi1jZW50cmFsL2NjLnZpbGVkYS9rb3RsaW4tb3BlbmFwaTMtZHNsKQoKQnVpbGQgeW91ciBPcGVuQXBpMyBzcGVjIGluIGtvdGxpbiEKCiMjIGltcG9ydAoKa290bGluLW9wZW5hcGkzLWRzbCBpcyBhdmFpbGFibGUgb24gbWF2ZW4gY2VudHJhbAoKIyMjIGdyYWRsZQoKYGBgZ3Jvb3Z5CmNvbXBpbGUgImNjLnZpbGVkYTprb3RsaW4tb3BlbmFwaTMtZHNsOjEuNS4wIgpgYGAKCiMjIyBtYXZlbgpgYGB4bWwKPGRlcGVuZGVuY3k+CiAgICA8Z3JvdXBJZD5jYy52aWxlZGE8L2dyb3VwSWQ+CiAgICA8YXJ0aWZhY3RJZD5rb3RsaW4tb3BlbmFwaTMtZHNsPC9hcnRpZmFjdElkPgogICAgPHZlcnNpb24+MS41LjA8L3ZlcnNpb24+CjwvZGVwZW5kZW5jeT4KYGBgCgojIyBleGFtcGxlCgpmb3IgYSBjb21wbGV0ZSBleGFtcGxlIFtsb29rIGF0IHRoZSB0ZXN0XShzcmMvdGVzdC9rb3RsaW4vY2MvdmlsZWRhL29wZW5hcGkvZHNsL09wZW5BcGlEc2xUZXN0Lmt0KQoKIyMgbGljZW5zZQpgYGAKQ29weXJpZ2h0IDIwMTcgVHJpc3RhbiBMZW8KCkxpY2Vuc2VkIHVuZGVyIHRoZSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAgKHRoZSAiTGljZW5zZSIpOwp5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuCllvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdAoKICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCgpVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCmRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCldJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgpTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCmxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgpgYGAK readmeEtag: '"65dd1080cc18c6cc7b4e4129e2bccf8d1d3456ad"' readmeLastModified: Wed, 14 Aug 2024 18:07:02 GMT repositoryId: 108694147 description: Build your OpenApi3 spec in kotlin! created: '2017-10-29T00:33:35Z' updated: '2026-02-05T13:40:14Z' language: Kotlin archived: false stars: 68 watchers: 4 forks: 6 owner: derveloper logo: https://avatars.githubusercontent.com/u/18556?v=4 license: Apache-2.0 repoEtag: '"eb078af0dc6fb3ed1aa7df04352e445e05254f8b28d1ae8483a662b0767491dd"' repoLastModified: Thu, 05 Feb 2026 13:40:14 GMT foundInMaster: true - source: https://openapi.tools/ name: Mintlify category: - Documentation - Server Implementations language: - Web - SaaS link: https://mintlify.com repository: https://github.com/mintlify/starter source_description: >- Modern standard for API documentation. Beautiful and easy-to-maintain UI components and interactive playground. v2: false v3: true v3_1: true id: eadde2613f7d452d5370b68ef15882bc repositoryMetadata: base64Readme: >- IyBNaW50bGlmeSBTdGFydGVyIEtpdAoKVXNlIHRoZSBzdGFydGVyIGtpdCB0byBnZXQgeW91ciBkb2NzIGRlcGxveWVkIGFuZCByZWFkeSB0byBjdXN0b21pemUuCgpDbGljayB0aGUgZ3JlZW4gKipVc2UgdGhpcyB0ZW1wbGF0ZSoqIGJ1dHRvbiBhdCB0aGUgdG9wIG9mIHRoaXMgcmVwbyB0byBjb3B5IHRoZSBNaW50bGlmeSBzdGFydGVyIGtpdC4gVGhlIHN0YXJ0ZXIga2l0IGNvbnRhaW5zIGV4YW1wbGVzIHdpdGgKCi0gR3VpZGUgcGFnZXMKLSBOYXZpZ2F0aW9uCi0gQ3VzdG9taXphdGlvbnMKLSBBUEkgcmVmZXJlbmNlIHBhZ2VzCi0gVXNlIG9mIHBvcHVsYXIgY29tcG9uZW50cwoKKipbRm9sbG93IHRoZSBmdWxsIHF1aWNrc3RhcnQgZ3VpZGVdKGh0dHBzOi8vc3RhcnRlci5taW50bGlmeS5jb20vcXVpY2tzdGFydCkqKgoKIyMgRGV2ZWxvcG1lbnQKCkluc3RhbGwgdGhlIFtNaW50bGlmeSBDTEldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL21pbnQpIHRvIHByZXZpZXcgeW91ciBkb2N1bWVudGF0aW9uIGNoYW5nZXMgbG9jYWxseS4gVG8gaW5zdGFsbCwgdXNlIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYApucG0gaSAtZyBtaW50CmBgYAoKUnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCBhdCB0aGUgcm9vdCBvZiB5b3VyIGRvY3VtZW50YXRpb24sIHdoZXJlIHlvdXIgYGRvY3MuanNvbmAgaXMgbG9jYXRlZDoKCmBgYAptaW50IGRldgpgYGAKClZpZXcgeW91ciBsb2NhbCBwcmV2aWV3IGF0IGBodHRwOi8vbG9jYWxob3N0OjMwMDBgLgoKIyMgUHVibGlzaGluZyBjaGFuZ2VzCgpJbnN0YWxsIG91ciBHaXRIdWIgYXBwIGZyb20geW91ciBbZGFzaGJvYXJkXShodHRwczovL2Rhc2hib2FyZC5taW50bGlmeS5jb20vc2V0dGluZ3Mvb3JnYW5pemF0aW9uL2dpdGh1Yi1hcHApIHRvIHByb3BhZ2F0ZSBjaGFuZ2VzIGZyb20geW91ciByZXBvIHRvIHlvdXIgZGVwbG95bWVudC4gQ2hhbmdlcyBhcmUgZGVwbG95ZWQgdG8gcHJvZHVjdGlvbiBhdXRvbWF0aWNhbGx5IGFmdGVyIHB1c2hpbmcgdG8gdGhlIGRlZmF1bHQgYnJhbmNoLgoKIyMgTmVlZCBoZWxwPwoKIyMjIFRyb3VibGVzaG9vdGluZwoKLSBJZiB5b3VyIGRldiBlbnZpcm9ubWVudCBpc24ndCBydW5uaW5nOiBSdW4gYG1pbnQgdXBkYXRlYCB0byBlbnN1cmUgeW91IGhhdmUgdGhlIG1vc3QgcmVjZW50IHZlcnNpb24gb2YgdGhlIENMSS4KLSBJZiBhIHBhZ2UgbG9hZHMgYXMgYSA0MDQ6IE1ha2Ugc3VyZSB5b3UgYXJlIHJ1bm5pbmcgaW4gYSBmb2xkZXIgd2l0aCBhIHZhbGlkIGBkb2NzLmpzb25gLgoKIyMjIFJlc291cmNlcwotIFtNaW50bGlmeSBkb2N1bWVudGF0aW9uXShodHRwczovL21pbnRsaWZ5LmNvbS9kb2NzKQo= readmeEtag: '"055c983adbe012e7990378a0d2f784a6379aa0d6"' readmeLastModified: Wed, 17 Sep 2025 18:19:57 GMT repositoryId: 581018208 description: 📖 The starter kit for your Mintlify docs created: '2022-12-22T03:50:30Z' updated: '2026-02-05T04:44:47Z' language: MDX archived: false stars: 1617 watchers: 10 forks: 509 owner: mintlify logo: https://avatars.githubusercontent.com/u/93011474?v=4 license: MIT repoEtag: '"f896e004e3d91a3d307ec38927213db0f05894258f4b05830e51d7283c04b8d4"' repoLastModified: Thu, 05 Feb 2026 04:44:47 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/blueswen/mkdocs-redoc-tag v3: true id: 04369ce76dd1df0b16316691678f55bd repositoryMetadata: base64Readme: >- IyBNa0RvY3MgUmVkb2MgVGFnIAoKPGEgdGFyZ2V0PSJfYmxhbmsiIGhyZWY9Imh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9ta2RvY3MtcmVkb2MtdGFnIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3YvbWtkb2NzLXJlZG9jLXRhZy5zdmciIGFsdD0iUHlQSSB2ZXJzaW9uIi8+PC9hPgo8YSB0YXJnZXQ9Il9ibGFuayIgaHJlZj0iaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L21rZG9jcy1yZWRvYy10YWciPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvZG0vbWtkb2NzLXJlZG9jLXRhZy5zdmciIGFsdD0iUHlQSSBkb3dubG9hZHMiLz48L2E+CjxhIHRhcmdldD0iX2JsYW5rIiBocmVmPSJodHRwczovL2NvZGVjb3YuaW8vZ2gvYmx1ZXN3ZW4vbWtkb2NzLXJlZG9jLXRhZyI+PGltZyBzcmM9Imh0dHBzOi8vY29kZWNvdi5pby9naC9ibHVlc3dlbi9ta2RvY3MtcmVkb2MtdGFnL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2ZyIgYWx0PSJDb2RlY292Ii8+PC9hPgoKQSBNa0RvY3MgcGx1Z2luIHN1cHBvcnRzIGFkZGluZyBbUmVkb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKSB0byB0aGUgcGFnZS4KCiMjIEZlYXR1cmVzCgoxLiBPcGVuQVBJIFNwZWNpZmljYXRpb24gZmlsZSBmcm9tIG9ubGluZSBvdmVyIFVSTCBvciBzdGF0aWMgZmlsZSBpbiBkb2NzCjIuIEFsbCBkZXBlbmRlbmNpZXMgYXJlIHVzaW5nIHN0YXRpYyBmaWxlcyBoYW5kbGVkIGJ5IHRoZSBwbHVnaW4gbm90IGZyb20gQ0ROLCBlc3BlY2lhbGx5IHN1aXRhYmxlIGZvciB0aG9zZSBkb2N1bWVudHMgYmVlbiBkZXBsb3llZCBpbiB0aGUgaW50cmFuZXQKMy4gU3luY2hyb25pemVkIGRhcmsgbW9kZSB3aXRoIFtNYXRlcmlhbCBmb3IgTWtEb2NzXShodHRwczovL3NxdWlkZnVuay5naXRodWIuaW8vbWtkb2NzLW1hdGVyaWFsLykKCiMjIERlcGVuZGVuY3kKCjEuIFB5dGhvbiBQYWNrYWdlCiAgICAxLiBiZWF1dGlmdWxzb3VwND49NC4xMS4xCjIuIFJlZG9jIHN0YW5kYWxvbmUgamF2YXNjcmlwdCBmcm9tIFtvZmZpY2lhbCBDRE5dKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jP3RhYj1yZWFkbWUtb3YtZmlsZSNyZWxlYXNlcykKICAgIDEuIHJlZG9jPT0yLjQuMAoKIyMgVXNhZ2UKCjEuIEluc3RhbGwgdGhlIHBsdWdpbiBmcm9tIFB5cGkKCiAgICBgYGBiYXNoCiAgICBwaXAgaW5zdGFsbCBta2RvY3MtcmVkb2MtdGFnCiAgICBgYGAKCjIuIEFkZCBgYGByZWRvYy10YWdgYGAgcGx1Z2luIGludG8geW91ciBta2RvY3MueW1sIHBsdWdpbnMgc2VjdGlvbnM6CgogICAgYGBgeWFtbAogICAgcGx1Z2luczoKICAgICAgIC0gcmVkb2MtdGFnCiAgICBgYGAKMy4gQWRkIGBgYHJlZG9jYGBgIHRhZyBpbiBtYXJrZG93biB0byBpbmNsdWRlIFJlZG9jOgoKICAgIGBgYGh0bWwKICAgIDxyZWRvYyBzcmM9Imh0dHBzOi8vcGV0c3RvcmUuc3dhZ2dlci5pby92Mi9zd2FnZ2VyLmpzb24iLz4KICAgIGBgYAoKICAgICFbUmVkb2MgU2FtcGxlIEltYWdlXShodHRwczovL2JsdWVzd2VuLmdpdGh1Yi5pby9ta2RvY3MtcmVkb2MtdGFnL3NhbXBsZS5wbmcpCgo0LiBZb3UgbWF5IGN1c3RvbWl6ZSB0aGUgcGx1Z2luIGJ5IHBhc3Npbmcgb3B0aW9ucyBpbiBta2RvY3MueW1sOgoKICAgIGBgYHlhbWwKICAgIHBsdWdpbnM6CiAgICAgICAtIHJlZG9jLXRhZzoKICAgICAgICAgICAgYmFja2dyb3VuZDogV2hpdGUKICAgIGBgYAoKICAgIHwgT3B0aW9ucyB8IFR5cGUgfCBEZXNjcmlwdGlvbiB8CiAgICB8LS0tfC0tLXwtLS18CiAgICB8IGJhY2tncm91bmQgfCBTdHJpbmcgfCBEZWZhdWx0OiAiIi4gUmVkb2MgaWZyYW1lIGJvZHkgYmFja2dyb3VuZCBhdHRyaWJ1dGUgdmFsdWUuIFlvdSBjYW4gdXNlIGFueSBjc3MgdmFsdWUgZm9yIGJhY2tncm91bmQgZm9yIGV4YW1wbGUgIiM3NGI5ZmYiIG9yICJHYWluc2Jvcm8iIG9yICIiIGZvciBub3RoaW5nLiB8CiAgICB8IGhlaWdodCB8IFN0cmluZyB8IERlZmF1bHQ6ICI4MHZoIi4gSGVpZ2h0IG9mIFJlZG9jIGlmcmFtZS4gfAoKIyMgSG93IGl0IHdvcmtzCgoxLiBDb3B5IFJlZG9jIHNjcmlwdCBmaWxlIGludG8gYHNpdGUvYXNzZXRzL2phdmFzY3JpcHRzL2AgZGlyZWN0b3J5CjIuIFNlYXJjaCBhbGwgcmVkb2MgdGFncywgdGhlbiBjb252ZXJ0IHRoZW0gdG8gYW4gaWZyYW1lIHRhZyBhbmQgZ2VuZXJhdGUgdGhlIGlmcmFtZSB0YXJnZXQgaHRtbCB3aXRoIHRoZSBnaXZlbiBPcGVuQVBJIFNwZWNpZmljYXRpb24gc3JjIHBhdGgKCiMjIExpY2Vuc2UKClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UgLSBzZWUgdGhlIFtMSUNFTlNFLm1kXShodHRwczovL2dpdGh1Yi5jb20vQmx1ZXN3ZW4vbWtkb2NzLXJlZG9jLXRhZy9ibG9iL21haW4vTElDRU5TRSkgZmlsZSBmb3IgZGV0YWlscy4KCiMjIFJlZmVyZW5jZQoKMS4gW3JlZGFya10oaHR0cHM6Ly9naXRodWIuY29tL2RpbGFueC9yZWRhcmspOiBzb3VyY2Ugb2YgZGFyayBtb2RlIGphdmFzY3JpcHQgYW5kIGNzcwo= readmeEtag: '"ee6a42e7fafc56ed9967f8e240890449107a7ddc"' readmeLastModified: Sat, 05 Apr 2025 14:56:11 GMT repositoryId: 727734575 description: A MkDocs plugin supports adding Redoc to the page. created: '2023-12-05T13:20:17Z' updated: '2025-05-31T04:21:30Z' language: Python archived: false stars: 3 watchers: 1 forks: 2 owner: blueswen logo: https://avatars.githubusercontent.com/u/1564148?v=4 license: MIT repoEtag: '"1479cf8df39f4e8e8d2180c26726e228f79bbe5ba8cc4b73b63e88f45c37cd3c"' repoLastModified: Sat, 31 May 2025 04:21:30 GMT category: Documentation foundInMaster: true name: MkDocs Redoc Tag language: Python link: https://blueswen.github.io/mkdocs-redoc-tag/ source_description: A MkDocs plugin supports adding Redoc to the page. v2: true v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/blueswen/mkdocs-swagger-ui-tag v3: true id: 210c459f395c1a11f8160fc69bd12fee repositoryMetadata: base64Readme: >- # MkDocs Swagger UI Tag

<p align="center">
<a target="_blank" href="https://pypi.org/project/mkdocs-swagger-ui-tag"><img src="https://img.shields.io/pypi/v/mkdocs-swagger-ui-tag.svg" alt="PyPI version"/></a>
<a target="_blank" href="https://pypi.org/project/mkdocs-swagger-ui-tag"><img src="https://img.shields.io/pypi/dm/mkdocs-swagger-ui-tag.svg" alt="PyPI downloads"/></a>
<a target="_blank" href="https://codecov.io/gh/blueswen/mkdocs-swagger-ui-tag"><img src="https://codecov.io/gh/blueswen/mkdocs-swagger-ui-tag/branch/main/graph/badge.svg?token=1D1B0GAQN1" alt="Codecov"/></a>
</p>

A MkDocs plugin supports adding [Swagger UI](https://github.com/swagger-api/swagger-ui) to the page.

[Live demo](https://blueswen.github.io/mkdocs-swagger-ui-tag/) with [Material for MkDocs](https://squidfunk.github.io/mkdocs-material/).

## Features

1. OpenAPI Specification file from online over URL or static file in docs
2. All dependencies are using static files handled by the plugin not from CDN, especially suitable for those documents been deployed in the intranet
3. Multiple Swagger UI on the same page
4. Synchronized dark mode with [Material for MkDocs](https://squidfunk.github.io/mkdocs-material/)
5. Configure [Swagger UI configuration](https://swagger.io/docs/open-source-tools/swagger-ui/usage/configuration/) through plugin options and tag attributes
6. Support multiple OAS in a single Swagger UI with a top bar selector
7. Support Swagger UI [initOAuth](https://swagger.io/docs/open-source-tools/swagger-ui/usage/oauth2/) method

## Dependency

1. Python Package
    1. beautifulsoup4>=4.13.3
2. [Swagger UI dist](https://www.npmjs.com/package/swagger-ui-dist) javascript file and CSS file
    1. swagger-ui-dist==5.27.1

## Usage

1. Install the plugin from PyPI

    ```bash
    pip install mkdocs-swagger-ui-tag
    ```

2. Add ```swagger-ui-tag``` plugin to your mkdocs.yml plugins sections:

    ```yaml
    plugins:
       - swagger-ui-tag
    ```

3. Add ```swagger-ui``` tag in markdown to include Swagger UI

    ```markdown
    <swagger-ui src="https://petstore.swagger.io/v2/swagger.json"/>
    ```

    ![Swagger UI Sample Image](https://blueswen.github.io/mkdocs-swagger-ui-tag/sample.png)

4. You may customize the plugin by passing options in mkdocs.yml, check more details on [options](https://blueswen.github.io/mkdocs-swagger-ui-tag/options/):

    ```yaml
    plugins:
       - swagger-ui-tag:
            background: White
            docExpansion: none
            filter: ""
            syntaxHighlightTheme: monokai
            tryItOutEnabled: ['get', 'post']
    ```

    | Options | Type | Description |
    |---|---|---|
    | background | String | Default: "". Swagger UI iframe body background attribute value. You can use any css value for background for example "#74b9ff" or "Gainsboro" or "" for nothing. |
    | docExpansion | String | Default: "list". Controls the default expansion setting for the operations and tags. It can be "list" (expands only the tags), "full" (expands the tags and operations) or "none" (expands nothing). |
    | filter | String or Boolean | Default: False. If set, enables filtering. The top bar will show an edit box that you can use to filter the tagged operations that are shown. Can be Boolean to enable or disable, or a string, in which case filtering will be enabled using that string as the filter expression. Filtering is case sensitive matching the filter expression anywhere inside the tag. |
    | syntaxHighlightTheme | String | Default: "agate". [Highlight.js](https://highlightjs.org/static/demo/) syntax coloring theme to use. It can be "agate", "arta", "monokai", "nord", "obsidian" or "tomorrow-night" |
    | tryItOutEnabled | Boolean | Default: False. This setting determines the default editability of the "Try it out" section, including parameters or body. |
    | oauth2RedirectUrl | String | Default: Absolute URL of "/assets/swagger-ui/oauth2-redirect.html" relative with site_url in mkdocs.yml or document root path on site without site_url, e.g. "[https://blueswen.github.io/mkdocs-swagger-ui-tag/assets/swagger-ui/oauth2-redirect.html](https://blueswen.github.io/mkdocs-swagger-ui-tag/assets/swagger-ui/oauth2-redirect.html)". OAuth redirect URL. |
    | supportedSubmitMethods | Array | Default: All Http Methods. Array=["get", "put", "post", "delete", "options", "head", "patch", "trace"]. List of HTTP methods that have the "Try it out" feature enabled. An empty array disables "Try it out" for all operations. This does not filter the operations from the display. |
    | validatorUrl | String | Default: "https://validator.swagger.io/validator". By default, Swagger UI attempts to validate specs against swagger.io's online validator in multiple OAS Swagger UI. You can use this parameter to set a different validator URL, for example for locally deployed validators ([Validator Badge](https://github.com/swagger-api/validator-badge)). Setting it "none" to disable validation. |
    | extra_css | Array | Default: []. List of additional CSS files to include in Swagger UI iframes. |
    | dark_scheme_name | String | Default: "slate". The color scheme name used for dark mode detection with Material for MkDocs theme. |
    | filter_files | Array | Default: []. List of file paths to filter processing. If specified, only files in this list will be processed for swagger-ui tags. |

## How it works

1. Copies the Swagger UI script file into `site/assets/javascripts/` directory, the CSS file into `site/assets/stylesheets/` directory, and the [default Oauth2 redirect html](https://github.com/blueswen/mkdocs-swagger-ui-tag/blob/main/mkdocs_swagger_ui_tag/swagger-ui/oauth2-redirect.html) into `site/assets/swagger-ui/` directory
2. Search all swagger-ui tags, then convert them to an iframe tag and generate the iframe target HTML with the given OpenAPI Specification src path and options

## Development

### Upgrading the Swagger-UI version

To upgrade the version of `swagger-ui-dist` update the version in the [package.json](./package.json) and install with `npm i`. 

After this, run the provided [update swagger ui script](./update-swagger-ui.sh) to move the files to the appropriate folders and commit the changes.

## License

This project is licensed under the MIT License - see the [LICENSE.md](https://github.com/Blueswen/mkdocs-swagger-ui-tag/blob/main/LICENSE) file for details.

## Reference

1. [Amoenus Swagger Dark Theme](https://github.com/Amoenus/SwaggerDark/): source of dark mode css
 readmeEtag: '"d23373e86ef64551840d459e9bd2be30e46f89e2"' readmeLastModified: Sat, 23 Aug 2025 16:23:37 GMT repositoryId: 502576077 description: A MkDocs plugin supports adding Swagger UI to the page. created: '2022-06-12T09:49:37Z' updated: '2026-02-03T15:44:15Z' language: CSS archived: false stars: 109 watchers: 1 forks: 13 owner: blueswen logo: https://avatars.githubusercontent.com/u/1564148?v=4 license: MIT repoEtag: '"20aeb8caae282cc246e87adaeb66c51e8cff2aaf1dddbb5bc68f8fc8e01690d7"' repoLastModified: Tue, 03 Feb 2026 15:44:15 GMT category: - Documentation - Parsers foundInMaster: true name: MkDocs Swagger UI Tag language: Python link: https://blueswen.github.io/mkdocs-swagger-ui-tag/ source_description: A MkDocs plugin supports for add Swagger UI in page. v2: true v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/bcoughlan/openapi-commander v3: true id: 182561588480789f47720cb799634f03 repositoryMetadata: base64Readme: >- # OpenAPI Commander

Generate a Node.js command line tool from an OpenAPI definition using the
[commander](https://www.npmjs.com/package/commander) library.

```
$ node petstore.js

Usage: petstore [options] [command]

Options:
  -p, --print <mode>     Print the HTTP request instead of sending it. (choices: "curl", "plain")
  -s, --server <server>  Server to use
  -a, --auth <auth>      Authorization header to send
  -h, --help             display help for command

Commands:
  pet                    Everything about your Pets
  user                   Operations about user
  help [command]         display help for command

$ node petstore.js pet getPetById 1
Status: 200
{
  "id": 1,
  "category": {
    "id": 1,
    "name": "Hola"
  },
}
```

It can be used as a starter for writing command line tools that talk to APIs,
or as an alternative to using curl to work with APIs.

<!-- generated with "markdown-toc -i README.md" -->
<!-- toc -->

- [Features](#features)
- [Setup](#setup)
- [Examples](#examples)
  * [Commands grouped by tag](#commands-grouped-by-tag)
  * [Basic GET example](#basic-get-example)
  * [Global options](#global-options)
    + [Custom server](#custom-server)
    + [Set Authorization header](#set-authorization-header)
    + [Print request without sending](#print-request-without-sending)
    + [Print a curl command of the request](#print-a-curl-command-of-the-request)
  * [Print example request bodies](#print-example-request-bodies)
- [Build standalone binaries](#build-standalone-binaries)
- [A very incomplete TODO list...](#a-very-incomplete-todo-list)
- [Contributing](#contributing)

<!-- tocstop -->


# Features

- Generate clean code from your spec.
- Supports Swagger 2, OpenAPI 3.0 and 3.1.
- Show examples of request bodies, using examples from the spec or generated with [openapi-sampler](https://github.com/Redocly/openapi-sampler).
- Verbose mode to see response headers.
- Print a curl command of the request.

# Setup

1. Create an npm project with `npm init` or use an existing one. Your Node.js version must be 16+

2. Install dependencies

** Note: If your project still uses CommonJS, install `openapi-commander@3` and use `.js` for the output extension.**

```
npm install commander
npm install --save-dev openapi-commander
```

3. Generate the CLI code

```
npx openapi-commander generate <path or URL to spec> <output file>
```

e.g.

```
npx openapi-commander generate https://raw.githubusercontent.com/OAI/OpenAPI-Specification/old-v3.0.4-dev/examples/v3.0/petstore.yaml petstore.mjs
```

4. Run

```
node petstore.js ...
```

# Examples

## Subcommands

The commands are grouped by their first tag in the OpenAPI spec (the same logic
that Swagger UI uses).

```
~/petstore$ node petstore.js pet
Usage: petstore pet [options] [command]

Everything about your Pets

Options:
  -h, --help                           display help for command

Commands:
  updatePet [options] <body>           Update an existing pet
  addPet [options] <body>              Add a new pet to the store
  findPetsByStatus [options]           Finds Pets by status
  findPetsByTags [options]             *DEPRECATED* Finds Pets by tags
  getPetById <petId>                   Find pet by ID
  updatePetWithForm [options] <petId>  Updates a pet in the store with form data
  deletePet [options] <petId>          Deletes a pet
  uploadFile [options] <petId>         uploads an image
  help [command]                       display help for command
```


## Basic GET example

```
~/petstore$ node petstore.js pet getPetById 1
Status: 200
{
  "id": 1,
  "category": {
    "id": 1,
    "name": "Hola"
  },
  "name": "Perrito",
  "photoUrls": [
    "/tmp/inflector995596007222725944.tmp"
  ],
  "tags": [
    {
      "id": 1,
      "name": "Bizcocho"
    }
  ],
  "status": "available"
}
```

You can also add the `-v` flag to show response headers.

## Global options

### Custom server

```
~/petstore$ node petstore.js -s https://mypetstore.example pet getPetById 1
```

Alternatively you can override it by setting the environment variable `{COMMANDNAME}_SERVER`, e.g. `PETSTORE_SERVER`.

### Set Authorization header

```
~/petstore$ node petstore.js -a 'Bearer mytoken' getPetById 1
```

Alternatively you can override it by setting the environment variable `{COMMANDNAME}_AUTH`, e.g. `PETSTORE_AUTH`.

### Print request without sending

```
~/petstore$ node petstore.js pet -a 'Bearer mytoken' -s https://mypetstore.example getPetById 1 --print plain
GET https://mypetstore.example/pet/1
Authorization: Bearer mytoken
Accept: application/json
```

### Print a curl command of the request

```
~/petstore$ node petstore.js pet -a 'Bearer mytoken' -s https://mypetstore.example getPetById 1 --print curl
curl -X GET -H 'Authorization: Bearer mytoken' -H 'Accept: application/json' 'https://mypetstore.example/pet/1'
```

## Print example request bodies

```
~/petstore$ node petstore.js pet updatePet examples
Example for application/json:
{
  "id": 10,
  "name": "doggie",
  "category": {
    "id": 1,
    "name": "Dogs"
  },
  "photoUrls": [
    "string"
  ],
  "tags": [
    {
      "id": 0,
      "name": "string"
    }
  ],
  "status": "available"
}
```

# Build standalone binaries

To build multi-platform standalone binaries platform I recommend [vercel/pkg](https://github.com/vercel/pkg).

# A very incomplete TODO list...

The top priority is to beef up the test suite.

- Syntax highlighting of examples
- Implement different security/auth types.
- Auto-detect request body type from file.
- Show JSON & YAML examples with comments for field descriptions.
  -> Strip comments from JSON when passing in
- Strip html/markdown from description
- More serialization options https://swagger.io/docs/specification/serialization/
- application/x-www-form-urlencoded content types.
- Server templating.
- Autocomplete.

# Contributing

This is a hobby side project so I have limited time to work on issues but I will 
do my best to discuss issues, review PRs and keep the project maintained. Please
open an issue for discussion before opening any significant PRs to avoid any
disappointment about project scope.
 readmeEtag: '"2377cc967c9ee21b8d079ca47e1e0cef0afa81e4"' readmeLastModified: Mon, 18 Aug 2025 15:15:15 GMT repositoryId: 497158492 description: Generate a Node.js command line tool from an OpenAPI definition created: '2022-05-27T22:55:07Z' updated: '2025-11-04T09:25:13Z' language: TypeScript archived: false stars: 29 watchers: 1 forks: 4 owner: bcoughlan logo: https://avatars.githubusercontent.com/u/421683?v=4 license: MIT repoEtag: '"9ec184fae1141dbab3241601d3a9b48ccb7a047f695206e8926f0c690a685730"' repoLastModified: Tue, 04 Nov 2025 09:25:13 GMT category: - Code Generators - SDK - Parsers foundInMaster: true name: OpenAPI Commander language: - Node.js - CLI link: https://www.npmjs.com/package/openapi-commander source_description: Generate a Node.js command line tool from an OpenAPI definition. v2: false v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/pmcelhaney/counterfact v3: true id: 2c27ee61398c89f17aa5be7ffa08c72a repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIiBtYXJrZG93bj0iMSI+Cgo8aW1nIHNyYz0iLi9jb3VudGVyZmFjdC5zdmciIGFsdD0iQ291bnRlcmZhY3QiIGJvcmRlcj0wPgo8YnI+PGJyPjxicj4KPC9kaXY+CgoqKlNwaW4gdXAgYSBtb2NrIHNlcnZlciBpbnN0YW50bHkuIE5vIGNvbmZpZywgbm8gZnVzcy4gVHJ5IGl0IG5vdzoqKgoKYGBgc2ggY29weQpucHggY291bnRlcmZhY3RAbGF0ZXN0IGh0dHBzOi8vcGV0c3RvcmUzLnN3YWdnZXIuaW8vYXBpL3YzL29wZW5hcGkuanNvbiBtb2NrLWFwaQpgYGAKClRoaXMgY29tbWFuZCByZWFkcyBhbiBPcGVuQVBJIHNwZWMgKHRoZSBbU3dhZ2dlciBQZXRzdG9yZV0oaHR0cHM6Ly9wZXRzdG9yZS5zd2FnZ2VyLmlvKSksIGdlbmVyYXRlcyBUeXBlU2NyaXB0IGNvZGUgZm9yIGEgbW9jayBzZXJ2ZXIgaW4gYG1vY2stYXBpYCwgYW5kIHN0YXJ0cyB0aGUgc2VydmVyLgoKPGRldGFpbHM+CjxzdW1tYXJ5PkV4YW1wbGUgb2YgZ2VuZXJhdGVkIGNvZGU8L3N1bW1hcnk+CgpgYGB0cwovLyAuL21vY2stYXBpL3JvdXRlcy9zdG9yZS9vcmRlci97b3JkZXJJRH0udHMKaW1wb3J0IHR5cGUgeyBIVFRQX0dFVCB9IGZyb20gIi4uLy4uLy4uL3R5cGVzL3BhdGhzL3N0b3JlL29yZGVyL3tvcmRlcklkfS50eXBlcy5qcyI7CmltcG9ydCB0eXBlIHsgSFRUUF9ERUxFVEUgfSBmcm9tICIuLi8uLi8uLi90eXBlcy9wYXRocy9zdG9yZS9vcmRlci97b3JkZXJJZH0udHlwZXMuanMiOwoKZXhwb3J0IGNvbnN0IEdFVDogSFRUUF9HRVQgPSAoJCkgPT4gewogIHJldHVybiAkLnJlc3BvbnNlWzIwMF0ucmFuZG9tKCk7Cn07CgpleHBvcnQgY29uc3QgREVMRVRFOiBIVFRQX0RFTEVURSA9ICgkKSA9PiB7CiAgcmV0dXJuICQucmVzcG9uc2VbMjAwXTsKfTsKYGBgCgo8L2RldGFpbHM+CgpXYW50IGNvbnRyb2w/IEVkaXQgdGhlIGdlbmVyYXRlZCByb3V0ZSBmaWxlcyAoZS5nLiBgLi9tb2NrLWFwaS9yb3V0ZXMvc3RvcmUvb3JkZXIve29yZGVySUR9LnRzYCkgYW5kIGRlZmluZSByZXNwb25zZXMgZGlyZWN0bHkuIEEgdHlwZS1zYWZlIEFQSSBmcm9tIHlvdXIgc3BlYyBzcGVlZHMgdXAgcHJvdG90eXBpbmcuCgo8ZGV0YWlscz4KPHN1bW1hcnk+RWRpdCB0aGUgY29kZSB0byBkZWZpbmUgY3VzdG9tIGJlaGF2aW9yIGFuZCByZXNwb25zZXM8L3N1bW1hcnk+CgpgYGB0cwovLyAuL21vY2stYXBpL3JvdXRlcy9zdG9yZS9vcmRlci97b3JkZXJJRH0udHMKaW1wb3J0IHsgT3JkZXIgfSBmcm9tICIuLi8uLi8uLi90eXBlcy9jb21wb25lbnRzL3NjaGVtYXMvT3JkZXIuanMiOwppbXBvcnQgdHlwZSB7IEhUVFBfR0VUIH0gZnJvbSAiLi4vLi4vLi4vdHlwZXMvcGF0aHMvc3RvcmUvb3JkZXIve29yZGVySWR9LnR5cGVzLmpzIjsKaW1wb3J0IHR5cGUgeyBIVFRQX0RFTEVURSB9IGZyb20gIi4uLy4uLy4uL3R5cGVzL3BhdGhzL3N0b3JlL29yZGVyL3tvcmRlcklkfS50eXBlcy5qcyI7CgpleHBvcnQgY29uc3QgR0VUOiBIVFRQX0dFVCA9ICgkKSA9PiB7CiAgY29uc3Qgb3JkZXJzOiBSZWNvcmQ8bnVtYmVyLCBPcmRlcj4gPSB7CiAgICAxOiB7CiAgICAgIHBldElkOiAxMDAsCiAgICAgIHN0YXR1czogInBsYWNlZCIsCiAgICB9LAogICAgMjogewogICAgICBwZXRJZDogOTk5LAogICAgICBzdGF0dXM6ICJhcHByb3ZlZCIsCiAgICB9LAogICAgMzogewogICAgICBwZXRJZDogMTIzNCwKICAgICAgc3RhdHVzOiAiZGVsaXZlcmVkIiwKICAgIH0sCiAgfTsKCiAgY29uc3Qgb3JkZXIgPSBvcmRlcnNbJC5yZXF1ZXN0Lm9yZGVySURdOwoKICBpZiAob3JkZXIgPT09IHVuZGVmaW5lZCkgewogICAgcmV0dXJuICQucmVzcG9uc2VbNDA0XTsKICB9CgogIHJldHVybiAkLnJlc3BvbnNlWzIwMF0uanNvbihvcmRlcik7Cn07CgpleHBvcnQgY29uc3QgREVMRVRFOiBIVFRQX0RFTEVURSA9ICgkKSA9PiB7CiAgcmV0dXJuICQucmVzcG9uc2VbMjAwXTsKfTsKYGBgCgo8L2RldGFpbHM+CgpZb3UgY2FuIGFsc28gKipwcm94eSBzb21lIHBhdGhzIHRvIHRoZSByZWFsIEFQSSoqIHdoaWxlIG1vY2tpbmcgb3RoZXJzIOKAlCBwZXJmZWN0IHdoZW4gcGFydCBvZiB0aGUgYmFja2VuZCBpc27igJl0IGZpbmlzaGVkIG9yIHlvdSBuZWVkIHRvIHNpbXVsYXRlIHRyaWNreSBzY2VuYXJpb3MuIFNlZSBbUHJveHlpbmddKC4vZG9jcy91c2FnZS5tZCNwcm94eWluZy10by1hLXJlYWwtYXBpKSBmb3IgZGV0YWlscy4KClVzZSBgbnB4IGNvdW50ZXJmYWN0IC0taGVscGAgdG8gdmlldyBDTEkgZmxhZ3MgZm9yIGN1c3RvbWl6aW5nIGJlaGF2aW9yIChlLmcuIGAtLXBvcnRgLCBgLS1wcm94eWAsIGAtLXdhdGNoYCkuCgpHbyBmdXJ0aGVyIHdpdGggc3RhdGVmdWwgbW9ja3M6IFBPU1QgZGF0YSwgdGhlbiBHRVQgaXQgYmFjay4gVHdlYWsgYmFja2VuZCBkYXRhIGxpdmUgd2l0aCBhIFJFUEwuIFVwZGF0ZSBjb2RlIHdpdGhvdXQgcmVzdGFydGluZyB0aGUgc2VydmVyIG9yIGxvc2luZyBzdGF0ZS4KCldlIGJlbGlldmUgc3Ryb25nbHkgaW4gQVBJLWZpcnN0IGRldmVsb3BtZW50IGFuZCB0aGUgRGVwZW5kZW5jeSBJbnZlcnNpb24gUHJpbmNpcGxlLiBXaGVuIGZyb250ZW5kIGFuZCBiYWNrZW5kIGltcGxlbWVudCB0aGUgc2FtZSBBUEkgc3BlYywgdGhleSBjYW4gYmUgYnVpbHQgYW5kIHRlc3RlZCBpbmRlcGVuZGVudGx5LiBNb3N0IG9mIHRoZSB0aW1lLCBpdOKAmXMgYmVzdCB0byB0ZXN0IHRoZSBmcm9udCBlbmQgYWdhaW5zdCB0aGUgcmVhbCBBUEkgYW5kIGZ1bGwgc3RhY2suIEJ1dCB0aGVyZSdzIGFsd2F5cyBzb21lIHBhcnQgb2YgdGhlIGJhY2tlbmQgdGhhdCBpc24ndCBmaW5pc2hlZCB5ZXQsIG9yIGEgc2NlbmFyaW8gdGhhdCdzIGN1bWJlcnNvbWUgdG8gcmVwcm9kdWNlLiBGb3IgdGhpcyByZWFzb24sIENvdW50ZXJmYWN0IHN1cHBvcnRzIHByb3h5aW5nIHRvIHRoZSByZWFsIEFQSSBmb3Igc29tZSBwYXRocyBhbmQgdXNpbmcgbW9ja3MgZm9yIG90aGVycy4KCkNvdW50ZXJmYWN0IHdhcyBieSBidWlsdCBhbiBlbmdpbmVlciB3aG8gc3BlbnQgZGVjYWRlcyB3cml0aW5nIGZyb250ZW5kIGNvZGUgYW5kIGdvdCB0aXJlZCBvZiBiZWluZyBzbG93ZWQgZG93biBieSB1bmZpbmlzaGVkIG9yIHVud2llbGR5IGJhY2tlbmQgQVBJcy4gVGhpcyBpcyB0aGUgZmFzdGVzdCB3YXkgdG8gZ2V0IHVuYmxvY2tlZCwgcHJvdG90eXBlIGlkZWFzLCBhbmQgcmVtZW1iZXIgd2hhdCBpdCBmZWVscyBsaWtlIHRvIGVuam95IGNyZWF0aW5nIHN0dWZmLgoKPiBSZXF1aXJlcyBOb2RlIOKJpSAxNy4wLjAKCjxkaXYgYWxpZ249ImNlbnRlciIgbWFya2Rvd249IjEiPgoKW0RvY3VtZW50YXRpb25dKC4vZG9jcy91c2FnZS5tZCkgfCBbQ2hhbmdlbG9nXSguL0NIQU5HRUxPRy5tZCkgfCBbQ29udHJpYnV0aW5nXSguL0NPTlRSSUJVVElORy5tZCkKCjwvZGl2PgoKPGJyPgo8ZGl2IGFsaWduPSJjZW50ZXIiICBtYXJrZG93bj0iMSI+CgohW01JVCBMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2xpY2Vuc2UtTUlULWJsdWUpIFshW1R5cGVTY3JpcHRdKC4vdHlwZXNjcmlwdC1iYWRnZS5wbmcpXShodHRwczovL2dpdGh1Yi5jb20vZWxsZXJicm9jay90eXBlc2NyaXB0LWJhZGdlcy8pIFshW0NvdmVyYWdlIFN0YXR1c10oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vcmVwb3MvZ2l0aHViL3BtY2VsaGFuZXkvY291bnRlcmZhY3QvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL3BtY2VsaGFuZXkvY291bnRlcmZhY3QpCgo8L2Rpdj4K readmeEtag: '"d9d1d66eb27316e77fead24c60c5e5e753aec596"' readmeLastModified: Sat, 11 Oct 2025 20:06:16 GMT repositoryId: 479515560 description: OpenAPI / Swagger to TypeScript generator and mock server created: '2022-04-08T19:32:15Z' updated: '2026-02-06T02:25:18Z' language: TypeScript archived: false stars: 135 watchers: 4 forks: 17 owner: pmcelhaney logo: https://avatars.githubusercontent.com/u/51504?v=4 license: MIT repoEtag: '"ee3b42e13abd23b75be1b3e6e7339fec212ec58894df3f1e36362148becb45cc"' repoLastModified: Fri, 06 Feb 2026 02:25:18 GMT category: - Mock - SDK - Server Implementations foundInMaster: true name: Counterfact link: https://counterfact.dev language: - TypeScript - Node source_description: >- Counterfact is a revolutionary free and open source mock server. If you have Node installed, all you have to do is run `npx counterfact@latest path-or-url-to-your-openapi-document`. Out of the box, you'll get a mock server that returns random data. With a TypeScript API that's optimized for mocking, you can add more realistic behavior. It supports mix and matching mocks with calls to the real API (sans CORS headaches). With hot reloading and a REPL, you can change things up on the fly. v3_1: true v2: true - source: https://openapi.tools/ name: Atlassian OpenAPI Request Validators category: Testing language: Java link: https://bitbucket.org/atlassian/swagger-request-validator/src/master/ repository: https://bitbucket.org/atlassian/swagger-request-validator/src/master/ v2: true v3: true source_description: >- A set of Java libraries which allow you to integrate OpenAPI Description Document validation into your testing or clients with tools like WireMock/RestAssured/MockMVC/etc... id: 1c4a1d5d85bbfdfa72511b0d14337f06 foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/daveshanley/vacuum v3: true repositoryMetadata: base64Readme: >- ![logo](logo.png)

# vacuum - The world's fastest OpenAPI & Swagger linter.

![build](https://github.com/daveshanley/vacuum/workflows/Build/badge.svg)
[![Go Report Card](https://goreportcard.com/badge/github.com/daveshanley/vacuum)](https://goreportcard.com/report/github.com/daveshanley/vacuum)
[![discord](https://img.shields.io/discord/923258363540815912)](https://discord.gg/UAcUF78MQN)
[![Docs](https://img.shields.io/badge/godoc-reference-5fafd7)](https:/-/pkg.go.dev/github.com/daveshanley/vacuum)
[![npm](https://img.shields.io/npm/dm/@quobix/vacuum?style=flat-square&label=npm%20downloads)](https://www.npmjs.com/package/@quobix/vacuum)
[![Docker Pulls](https://img.shields.io/docker/pulls/dshanley/vacuum?style=flat-square)](https://hub.docker.com/r/dshanley/vacuum)
[![Mentioned in Awesome Go](https://awesome.re/mentioned-badge-flat.svg)](https://github.com/avelino/awesome-go)

An **ultra-super-fast**, lightweight OpenAPI linter and quality checking tool, written in golang and inspired by [Spectral](https://github.com/stoplightio/spectral).

It's **fully compatible** with existing [Spectral](https://github.com/stoplightio/spectral) rulesets.

## Install using [homebrew](https://brew.sh) tap

```
brew install daveshanley/vacuum/vacuum
```

## Install using [npm](https://npmjs.com)

```
npm i -g @quobix/vacuum
```

## Install using [yarn](https://yarnpkg.com/)

```
yarn global add @quobix/vacuum
```

## Install using curl

```bash
curl -fsSL https://quobix.com/scripts/install_vacuum.sh | sh
```

### For CI/CD environments 

To avoid GitHub API rate limiting in automated environments, set a GitHub token:

```bash
# Using repository token (GitHub Actions)
GITHUB_TOKEN=${{ secrets.GITHUB_TOKEN }} curl -fsSL https://quobix.com/scripts/install_vacuum.sh | sh

# Using personal access token
GITHUB_TOKEN=your_github_token curl -fsSL https://quobix.com/scripts/install_vacuum.sh | sh
```

#### GitHub Actions example

```yaml
- name: Install vacuum
  env:
    GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}  # Increases rate limit from 60 to 5000 requests/hour
  run: |
    curl -fsSL https://quobix.com/scripts/install_vacuum.sh | sh
```

> **Note**: The GitHub token prevents intermittent installation failures in CI/CD environments caused by API rate limiting. 
> No additional permissions are required, the token only accesses public repository information.

## Install using [Docker](https://hub.docker.com/r/dshanley/vacuum)

The image is available at: https://hub.docker.com/r/dshanley/vacuum

```
docker pull dshanley/vacuum
```

> **Multi-platform support**: Docker images are available for both `linux/amd64` and `linux/arm64` architectures, including native ARM64 support for Apple Silicon Macs.

To run, mount the current working dir to the container and use a relative path to your spec, like so

```
docker run --rm -v $PWD:/work:ro dshanley/vacuum lint <your-openapi-spec.yaml>
```
Alternatively, you can pull it from
[Github packages](https://github.com/daveshanley/vacuum/pkgs/container/vacuum).
To do that, replace `dshanley/vacuum` with `ghcr.io/daveshanley/vacuum` in the above commands.

## Run with Go

If you have go >= 1.16 installed, you can use `go run` to build and run it:

```
go run github.com/daveshanley/vacuum@latest lint <your-openapi-spec.yaml>
```

---

## Sponsors
If your company is using `vacuum`, please considering [supporting this project](https://github.com/sponsors/daveshanley),
like our _very kind_ sponsors, past and present:


<a href="https://speakeasyapi.dev/?utm_source=vacuum+repo&utm_medium=github+sponsorship">
<picture>
  <source media="(prefers-color-scheme: dark)" srcset=".github/sponsors/speakeasy-github-sponsor-dark.svg">
  <img alt="speakeasy'" src=".github/sponsors/speakeasy-github-sponsor-light.svg">
</picture>
</a>

[Speakeasy](https://speakeasyapi.dev/?utm_source=vacuum+repo&utm_medium=github+sponsorship)

<a href="https://scalar.com">
<picture>
  <source media="(prefers-color-scheme: dark)" srcset=".github/sponsors/scalar-dark.png">
  <img alt="scalar" src=".github/sponsors/scalar-light.png">
</picture>
</a>

[scalar](https://scalar.com)

<a href="https://apideck.com">
<picture>
  <source media="(prefers-color-scheme: dark)" srcset=".github/sponsors/apideck-dark.png">
  <img alt="apideck'" src=".github/sponsors/apideck-light.png">
</picture>
</a>

[apideck](https://apideck.com)

---

## Come chat with us

Need help? Have a question? Want to share your work? [Join our discord](https://discord.gg/UAcUF78MQN) and
come say hi!

## Documentation

🔥 **New in** `v0.23` 🔥: **OpenAPI Overlay Support**

Ever needed to tweak an OpenAPI spec for different environments without maintaining multiple copies? 

Maybe swap out server URLs between dev, staging, and production? Or perhaps strip out internal endpoints before publishing the API docs?

OpenAPI Overlays are the answer. They let us make non-destructive modifications to specs using JSONPath expressions to 
target exactly what we want to change. vaccum now supports a new `apply-overlay` command.

- [Learn more about the apply-overlay command](https://quobix.com/vacuum/commands/apply-overlay)

---


`v0.22`: **Async Functions / Promises, Fetch & Batch mode in Custom JS Functions**

Do you want to call remote APIs in your vacuum javascript functions? What about async processing or the ability to use Promises?

vacuum now has its own event loop and will happily support `async` and `await`. Combined with a full implementation of [Fetch](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API).

Also added **Batch Mode**. This allows custom functions to receive the entire list of nodes, instead of firing the function 
for each result, so you can send all your data off to an API or an LLM, and have the ability parse and process everything at once
vs individually. 

- [Read all about async JS functions in vacuum](https://quobix.com/vacuum/api/custom-javascript-functions/#async-functions--promises)
- [Learn about using fetch in vacuum](https://quobix.com/vacuum/api/custom-javascript-functions/#fetch-api)
- [find out how batch mode works](https://quobix.com/vacuum/api/custom-javascript-functions/#batch-mode-processing)

---

`v0.21`: **Change detection filtering**

Want to see linting results on **just** the changes you have made to an OpenAPI document? Or want to see just the results on the differences between two documents? 
comes with a **what changed** mode. Using the new `--original` and `--changes-summary` global flags, you can filter out all the noise. 

[documentation for change detection](https://quobix.com/vacuum/commands/change-detection/) is available to learn more.

vacuum now supports [JSON Path Plus](https://github.com/JSONPath-Plus/JSONPath) annotations. This means that vacuum is 
compliant with **All Spectral Paths**. One of the last remaining gaps between vacuum and spectral has been closed. 

[See all the newly supported annotations](https://quobix.com/vacuum/rulesets/custom-rulesets/#anatomy-of-a-rule)

---

`v0.20`: **Support for auto fixing custom rules**

Got some rules that don't really need a human to look at?

Well now you can define an `AutoFixFunction` for your rules, and when you run with the `--fix` flag, the fixes will be applied to the file, or use `--fix-file` to write them to a different file.

See [Auto-Fixing Rule Violations](#auto-fixing-rule-violations) for more specifics.

---

`v0.19`: **Ignore rules with `x-lint-ignore`**

Got an error in your spec you know about but can't get round to fixing yet?
Migrating from zally and wanting to keep your existing `x-zally-ignore` issues silenced?

Now you can! Just add `x-lint-ignore: rule-id` to the yaml node reporting the failure (or `x-lint-ignore: [rule-one, rule-two]` if there are multiple issues to ignore).

---

`v0.18`: **New dashboard, new lint command, new rules!**.

Upgrades all around. There is a completely new `dashboard` command with a completely new dashboard terminal UI. It's 
completely interactive and allows you to explore, and filter violations, view full docs and see code. The `dashboard` command
also adds a new `-w` / `--watch` flag that will watch your OpenAPI file for changes and re-lint and re-render results automatically.

A re-written `lint` command that has a whole new rendering engine and output. Everything is much more readable, 
easier to see on a screen, matches the new `dashboard` style. It's 100% backwards compatible with previous versions, all flags as they were. 

New rules:

 - [no-request-body](https://quobix.com/vacuum/rules/operations/no-request-body/) - Ensures `GET` and `DELETE` operations do not have request bodies.
 - [duplicate-paths](https://quobix.com/vacuum/rules/operations/duplicate-paths/) - Ensures there are no duplicate paths exist
 - [no-unnecessary-combinator](https://quobix.com/vacuum/rules/schemas/no-unnecessary-combinator/) - Ensures no `allOf`, `oneOf` or `anyOf` combinators exist with a single schema inside them.
 - [camel-case-properties](https://quobix.com/vacuum/rules/schemas/camel-case-properties/) - Ensures all schema properties are `camelCase`.

---

`v0.17`: **Github Action**.

vacuum now has an official Github Action. [Read the docs](https://quobix.com/vacuum/github-action/), or check it out
in the [GitHub Marketplace](https://github.com/marketplace/actions/vacuum-openapi-linter-and-quality-analysis-tool).

---


`v0.16.11`: **Composed bundling mode**.

A different way to bundle exploded OpenAPI specifications into a single file. [Read the docs](https://quobix.com/vacuum/commands/bundle/).

---

`v0.16+` : **RFC 9535 Compliant**.

`v0.21+` : **JSON Path Plus Support**.

vacuum now supports both [RFC 9535](https://www.rfc-editor.org/rfc/rfc9535) JSONPath and **JSON Path Plus** extensions - full Spectral compatibility!

---

`v0.15+`: **Fixes, New Rules, Functions and Command**.

There is a new command `generate-ignorefile` that will generate an ignore file from a linting report.

New rule `no-request-body` checks for incorrect request bodies in operations, and `path-item-refs` checks for
$refs being used in path items.

---

### [Quick Start Guide 🚀](https://quobix.com/vacuum/start)

See all the documentation at https://quobix.com/vacuum

- [Installing vacuum](https://quobix.com/vacuum/installing/)
- [About vacuum](https://quobix.com/vacuum/about/)
- [Why should you care?](https://quobix.com/vacuum/why/)
- [Concepts](https://quobix.com/vacuum/concepts/)
- [FAQ](https://quobix.com/vacuum/faq/)
- [CLI Commands](https://quobix.com/vacuum/commands/)
  - [lint](https://quobix.com/vacuum/commands/lint/)
  - [vacuum report](https://quobix.com/vacuum/commands/report/)
  - [dashboard](https://quobix.com/vacuum/commands/dashboard/)
  - [html-report](https://quobix.com/vacuum/commands/html-report/)
  - [bundle](https://quobix.com/vacuum/commands/bundle/)
  - [spectral-report](https://quobix.com/vacuum/commands/spectral-report/)
  - [language-server](https://quobix.com/vacuum/commands/language-server/)
  - [apply-overlay](https://quobix.com/vacuum/commands/apply-overlay/)
  - [Change Detection](https://quobix.com/vacuum/commands/change-detection/)
- [Developer API](https://quobix.com/vacuum/api/getting-started/)
  - [Using The Index](https://quobix.com/vacuum/api/spec-index/)
  - [RuleResultSet](https://quobix.com/vacuum/api/rule-resultset/)
  - [Loading a RuleSet](https://quobix.com/vacuum/api/loading-ruleset/)
  - [Linting Non-OpenAPI Files](https://quobix.com/vacuum/api/non-openapi/)
  - [Custom Golang Functions](https://quobix.com/vacuum/api/custom-functions/)
  - [Custom JavaScript Functions](https://quobix.com/vacuum/api/custom-javascript-functions/)
- [Rules](https://quobix.com/vacuum/rules/)
  - [Examples](https://quobix.com/vacuum/rules/examples/)
  - [Tags](https://quobix.com/vacuum/rules/tags/)
  - [Descriptions](https://quobix.com/vacuum/rules/descriptions/)
  - [Schemas](https://quobix.com/vacuum/rules/schemas/)
  - [Spec Information](https://quobix.com/vacuum/rules/information/)
  - [Operations & Paths](https://quobix.com/vacuum/rules/operations/)
  - [Validation](https://quobix.com/vacuum/rules/validation/)
  - [Security](https://quobix.com/vacuum/rules/security/)
  - [OWASP](https://quobix.com/vacuum/rules/owasp/)
- [Functions](https://quobix.com/vacuum/functions/)
  - [Core Functions](https://quobix.com/vacuum/functions/core/) 
  - [OpenAPI Functions](https://quobix.com/vacuum/functions/openapi/)
  - [OWASP Functions](https://quobix.com/vacuum/functions/owasp/)
- [Understanding RuleSets](https://quobix.com/vacuum/rulesets/understanding/)
  - [Sharing RuleSets](https://quobix.com/vacuum/rulesets/sharing/)
  - [All Rules](https://quobix.com/vacuum/rulesets/all/)
  - [No Rules](https://quobix.com/vacuum/rulesets/no-rules/)
  - [Recommended Rules](https://quobix.com/vacuum/rulesets/recommended/)
  - [Custom Rules](https://quobix.com/vacuum/rulesets/custom-rulesets/)

---

> **vacuum can suck all the lint of a 5mb OpenAPI spec in milliseconds.**

Designed to reliably lint OpenAPI specifications, **very, very quickly**. Including _very large_ ones. Spectral can be quite slow
when used as an API and does not scale for enterprise applications.

vacuum will tell you what is wrong with your spec, why, where, and how to fix it. 

vacuum will work at scale and is designed as a CLI (with a web or console UI) and a library to be consumed in other applications.

### Dashboard

vacuum comes with an interactive dashboard (`vacuum dashboard <your-openapi-spec.yaml>`) allowing you to explore
rules and violations in a console, without having to scroll through thousands of results.

<a href="https://quobix.com/vacuum/commands/dashboard/">
<picture>
  <source media="(prefers-color-scheme: dark)" srcset=".github/assets/dashboard.gif">
  <img alt="speakeasy'" src=".github/sponsors/speakeasy-github-sponsor-light.svg">
</picture>
</a>

To read about the dashboard, see the [dashboard command docs](https://quobix.com/vacuum/commands/dashboard/).

### HTML Report

vacuum can generate an easy to navigate and understand HTML report. Like the dashboard
you can explore broken rules and violations, but in your browser.

No external dependencies, the HTML report will run completely offline.

![vacuum html-report](html-report-screenshot.png)

---

> **_Supports OpenAPI Version 2 (Swagger) and Version 3+_**

You can use either **YAML** or **JSON**, vacuum supports both formats.

## Using vacuum with pre-commit

Vacuum can be used with [pre-commit](https://pre-commit.com).

To do that, add to your `.pre-commit-config.yaml`:

```yaml
repos:
  - repo: https://github.com/daveshanley/vacuum
    rev: # a tag or a commit hash from this repo, see https://github.com/daveshanley/vacuum/releases
    hooks:
      - id: vacuum
```

See the [hook definition](./.pre-commit-hooks.yaml) here for details on what options the hook uses and what files it checks by default.

If no filenames or more than one filename in your repository matches the default `files` pattern in the hook definition,
the pattern needs to be overridden in your config so that it matches exactly one filename to lint at a time.
To lint multiple files, specify the hook multiple times with the appropriate overrides.

## Build an interactive HTML report 

```
./vacuum html-report <your-openapi-spec.yaml | vacuum-report.json.gz> <report-name.html>
```

You can replace `report-name.html` with your own choice of filename. Open the report
in your favorite browser and explore the results. 


## See full linting report 

```
./vacuum lint -d <your-openapi-spec.yaml>
```


## Lint multiple files at once

```
./vacuum lint -d <spec1.yaml> <spec2.yaml> <spec3.yaml>
```

## Lint multiple files using a glob pattern

```
./vacuum lint -d some/path/**/*.yaml
```

## See full linting report with inline code snippets

```
./vacuum lint -d -s <your-openapi-spec.yaml>
```

## See just the linting errors

```
./vacuum lint -d -e <your-openapi-spec.yaml>
```

## See just a specific category of report


```
./vacuum lint -d -c schemas <your-openapi-spec.yaml>
```

The options here are:

- `examples`
- `operations`
- `information`
- `descriptions`
- `schemas`
- `security`
- `tags`
- `validation`
- `owasp`

## Generate a Spectral compatible report

If you're already using Spectral JSON reports, and you want to use vacuum instead, use the `spectral-report` command

```
./vacuum spectral-report <your-openapi-spec.yaml> <report-output-name.json>
```

The report file name is _optional_. The default report output name is `vacuum-spectral-report.json`


## Generate a `vacuum report`

Vacuum reports are complete snapshots in time of a linting report for a specification. These reports can be 'replayed' 
back through vacuum. Use the `dashboard` or the `html-report` commands to 'replay' the report and explore the results
as they were when the report was generated.

```
./vacuum report -c <your-openapi-spec.yaml> <report-prefix>
```

The default name of the report will be `vacuum-report-MM-DD-YY-HH_MM_SS.json`. You can change the prefix by supplying
it as the second argument to the `report` command. 

Ideally, **you should compress the report using `-c`**. This shrinks down the size significantly. vacuum automatically
recognizes a compressed report file and will deal with it automatically when reading.

> When using compression, the file name will be `vacuum-report-MM-DD-YY-HH_MM_SS.json.gz`. vacuum uses gzip internally.

## Ignoring specific linting errors

You can ignore specific linting errors by providing an `--ignore-file` argument to the `lint` and `report` commands.

```
./vacuum lint --ignore-file <path-to-ignore-file.yaml> -d <your-openapi-spec.yaml>
```

```
./vacuum report --ignore-file <path-to-ignore-file.yaml> -c <your-openapi-spec.yaml> <report-prefix>
```

The ignore-file should point to a .yaml file that contains a list of errors to be ignored by vacuum. The structure of the
yaml file is as follows:

```
<rule-id-1>:
  - <json_path_to_error_or_warning_1>
  - <json_path_to_error_or_warning_2>
<rule-id-2>:
  - <json_path_to_error_or_warning_1>
  - <json_path_to_error_or_warning_2>
  ...
```

Ignoring errors is useful for when you want to implement new rules to existing production APIs. In some cases, 
correcting the lint errors would result in a breaking change. Having a way to ignore these errors allows you to implement
the new rules for new APIs while maintaining backwards compatibility for existing ones.

---

## Try out the dashboard

This is an early, but working console UI for vacuum. The code isn't great, it needs a lot of clean up, but
if you're interested in seeing how things are progressing, it's available.

```
./vacuum dashboard <your-openapi-spec.yaml | vacuum-report.json.gz>
```

---
## Supply your own Spectral compatible ruleset

If you're already using Spectral and you have your own [custom ruleset](https://meta.stoplight.io/docs/spectral/e5b9616d6d50c-custom-rulesets#custom-rulesets),
then you can use it with vacuum! 

The `lint`, `dashboard` and `spectral-report` commands all accept a `-r` or `--ruleset` flag, defining the path to your ruleset file.

### Here are some examples you can try

**_All rules turned off_**
```
./vacuum lint -r rulesets/examples/norules-ruleset.yaml <your-openapi-spec.yaml>
```

**_Only recommended rules_**
```
./vacuum lint -r rulesets/examples/recommended-ruleset.yaml <your-openapi-spec.yaml>
```

**_Enable specific rules only_**
```
./vacuum lint -r rulesets/examples/specific-ruleset.yaml <your-openapi-spec.yaml>
```

**_Custom rules_**
```
./vacuum lint -r rulesets/examples/custom-ruleset.yaml <your-openapi-spec.yaml>
```

**_All rules, all of them!**
```
./vacuum lint -r rulesets/examples/all-ruleset.yaml <your-openapi-spec.yaml>
```

---

## Configuration

### File
You can configure vacuum using a configuration file named `vacuum.conf.yaml`

By default, vacuum searches for this file in the following directories
1. Working directory
2. `$XDG_CONFIG_HOME`
3. `${HOME}/.config`

You can also specify a path to a file using the `--config` flag

Global flags are configured as top level nodes
```yaml
time: true
base: 'http://example.com'
...
```
Command specific flags are configured under a node with the commands name
```yaml
...
lint:
  silent: true
  ...
```

### Environmental variables

You can configure global vacuum flags using environmental variables in the form of: `VACUUM_<flag>`

If a flag, has a `-` in it, replace with `_`


## Auto-fixing rule violations

If you have a rule that doesn't need a human to look at it, and the change can be reliably automated you can configure an `AutoFixFunction` on the rule. When you then run the `lint` command you can pass the `--fix` flag and the violation will be automatically fixed.

### Set up

1. Define a rule that has an `autoFixFunction`, e.g.:
```yaml
rules:
  use-compatible-extensions:
    autoFixFunction: useExtensibleEnum
    description: Prefer compatible extensions
    id: use-compatible-extensions
    given: "$.components.schemas[?@.enum]"
    severity: warn
    message: Use x-extensible-enum instead of enum for better compatibility
    then:
      field: enum
      function: falsy
```

This rule flags any usage of `enum` and recommends they are updated to `x-extensible-enum`.
A simple change which can be easily auto fixed!

2. Create a function which performs the auto-fix.
```go
func useExtensibleEnum(
	node *yaml.Node,
	document *yaml.Node,
	context *model.RuleFunctionContext,
) (*yaml.Node, error) {
	if node.Kind != yaml.MappingNode {
		return node, nil
	}

	for i := 0; i < len(node.Content); i += 2 {
		if i+1 >= len(node.Content) {
			break
		}

		keyNode := node.Content[i]

		if keyNode.Value == "enum" {
			keyNode.Value = "x-extensible-enum"

			return node, nil
		}
	}

	return node, nil
}
```

> [!NOTE]
> The auto fix function must satisfy the `AutoFixFunction` type.
> It should take in the `*yaml.Node` of the violation, the root `*yaml.Node` of the document and the `RuleFunctionContext`.
> It should return the fixed `*yaml.Node` and an error.

3. Configure your `RuleSetExecution` to use the auto fix function.
```go
func Lint(rulesFile string, specFile string) error {
	rules, err := rulesets.LoadLocalRuleSet(ctx, rulesFile)
	if err != nil {
		return fmt.Errorf("error loading ruleset: %w", err)
	}

	rs := rulesets.BuildDefaultRuleSetsWithLogger(slog.Logger).
		GenerateRuleSetFromSuppliedRuleSet(rules)

	// NOTE: only showing the fields on the RuleSetExecution relevant to auto-fixing.
	results := motor.ApplyRulesToRuleSet(&motor.RuleSetExecution{
		AutoFixFunctions: map[string]model.AutoFixFunction{
			"useExtensibleEnum": useExtensibleEnum,
		},
		ApplyAutoFixes:         true,
		RuleSet:                rs,
	})

	// Write back to file if fixes were applied
	if len(lintResults.FixedResults) > 0 && autoFix {
		fileInfo, _ := os.Stat(specFile)

		err = os.WriteFile(specFile, result.ModifiedSpec, fileInfo.Mode())
		if err != nil {
			return fmt.Errorf("failed to write file %s: %w", c.file, err)
		}
	}

	return nil
}
```

When the auto fix function runs, if it returns an error the fix will not be applied, the error will be logged, and the violation will be reported in the standard results.

If the auto fix function succeeds the yaml node flagged by the violation will be replaced with the transformed version returned by the auto fix function.

> [!TIP]
> When using `vacuum` as a library You can access the fixed yaml content in the `RuleSetExecutionResult.ModifiedSpec`, and choose where to write the file.
> 
> When using `vacuum` as a cli, the `--fix` flag will overwrite the spec file in place, and `--fix-file` flag lets you specify an alternative file to write the content to, if you want to compare the outputs.

### Usage


> Logo gopher is modified, originally from [egonelbre](https://github.com/egonelbre/gophers)
 readmeEtag: '"3e7ab3c5c4e90dbd0d92a5f63ef807f6c33078a5"' readmeLastModified: Wed, 31 Dec 2025 01:30:28 GMT repositoryId: 415667153 description: >- vacuum is the worlds fastest OpenAPI 3, OpenAPI 2 / Swagger linter and quality analysis tool. Built in go, it tears through API specs faster than you can think. vacuum is compatible with Spectral rulesets and generates compatible reports. created: '2021-10-10T18:24:19Z' updated: '2026-02-05T21:44:57Z' language: Go archived: false stars: 999 watchers: 7 forks: 77 owner: daveshanley logo: https://avatars.githubusercontent.com/u/187345?v=4 license: MIT repoEtag: '"7a02b6ea4a797ba2fc1f554a4f9f80693158407a45ec8503061978bfe5538872"' repoLastModified: Thu, 05 Feb 2026 21:44:57 GMT foundInMaster: true category: - Description Validators - Data Validators id: 45eaedfa8d8b1fcb106649241c2929d6 name: vacuum link: https://quobix.com/vacuum language: go source_description: >- A blazing fast OpenAPI linter and validator that is compatible with Spectral rulesets, and designed for enterprise-grade speed and scale. v2: true v3_1: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/vladkens/apigen-ts v3: true v3_1: true id: fe23c57d8d96bc45f44ec987d4758ed9 repositoryMetadata: base64Readme: >- # apigen-ts

<div align="center">

[<img src="https://badges.ws/npm/v/apigen-ts" alt="version" />](https://npmjs.org/package/apigen-ts)
[<img src="https://badges.ws/packagephobia/publish/apigen-ts" alt="size" />](https://packagephobia.now.sh/result?p=apigen-ts)
[<img src="https://badges.ws/npm/dm/apigen-ts" alt="downloads" />](https://npmjs.org/package/apigen-ts)
[<img src="https://badges.ws/github/license/vladkens/apigen-ts" alt="license" />](https://github.com/vladkens/apigen-ts/blob/main/LICENSE)
[<img src="https://badges.ws/badge/-/buy%20me%20a%20coffee/ff813f?icon=buymeacoffee&label" alt="donate" />](https://buymeacoffee.com/vladkens)

</div>

<div align="center">
  <img src="./logo.svg" alt="apigen-ts logo" height="80" />
</div>

## Features

- Generates ready to use `ApiClient` with types (using `fetch`)
- Single output file, minimal third-party code
- Loads schemas from JSON / YAML, locally and remote
- Ability to customize `fetch` with your custom function
- Automatic formating with Prettier
- Can parse dates from date-time format (`--parse-dates` flag)
- Support OpenAPI v2, v3, v3.1
- Can be used with npx as well

## Install

```sh
npm install apigen-ts --save-dev
```

```sh
yarn add -D apigen-ts
```

## Usage

### 1. Generate

```sh
# From file
npx apigen-ts ./openapi.json ./api-client.ts

# From url
npx apigen-ts https://petstore3.swagger.io/api/v3/openapi.json ./api-client.ts

# From protected url
npx apigen-ts https://secret-api.example.com ./api-client.ts -H "x-api-key: secret-key"
```

Run `npx apigen-ts --help` for more options. Examples of generated clients [here](./examples/).

### 2. Import

```ts
import { ApiClient } from "./api-client"

const api = new ApiClient({
  baseUrl: "https://example.com/api",
  headers: { Authorization: "secret-token" },
})
```

### 3. Use

```ts
// GET /pet/{petId}
await api.pet.getPetById(1) // -> Pet

// GET /pet/findByStatus?status=sold
await api.pet.findPetsByStatus({ status: "sold" }) // -> Pets[]

// PUT /user/{username}; second arg body with type User
await api.user.updateUser("username", { firstName: "John" })
```

## Advanced

### Login flow

```ts
const { token } = await api.auth.login({ usename, password })
api.Config.headers = { Authorization: token }

await api.protectedRoute.get() // here authenticated
```

### Automatic date parsing

```sh
npx apigen-ts ./openapi.json ./api-client.ts --parse-dates
```

```ts
const pet = await api.pet.getPetById(1)
const createdAt: Date = pet.createdAt // date parsed from string with format=date-time
```

### String union as enums

You can generate string literal union instead of native enums in case you want to run in Node.js environment with [type-stripping](https://nodejs.org/api/typescript.html#type-stripping). To achive this pass `--inline-enums` command line argument or use `inlineEnums: true` in Node.js API.

```sh
npx apigen-ts ./openapi.json ./api-client.ts --inline-enums
```

This will generate:

```ts
type MyEnum = "OptionA" | "OptionB"

// instead of
enum MyEnum = {
  OptionA = "OptionA",
  OptionB = "OptionB"
}
```

### Errors handling

An exception will be thrown for all unsuccessful return codes.

```ts
try {
  const pet = await api.pet.getPetById(404)
} catch (e) {
  console.log(e) // parse error depend of your domain model, e is awaited response.json()
}
```

Also you can define custom function to parse error:

```ts
class MyClient extends ApiClient {
  async ParseError(rep: Response) {
    // do what you want
    return { code: "API_ERROR" }
  }
}

try {
  const api = new MyClient()
  const pet = await api.pet.getPetById(404)
} catch (e) {
  console.log(e) // e is { code: "API_ERROR" }
}
```

### Base url resolving

You can modify how the endpoint url is created. By default [URL constructor](https://developer.mozilla.org/en-US/docs/Web/API/URL/URL) used to resolve endpoint url like: `new URL(path, baseUrl)` which has specific resolving [rules](https://developer.mozilla.org/en-US/docs/Web/API/URL_API/Resolving_relative_references). E.g.:

- `new URL("/v2/cats", "https://example.com/v1/") // -> https://example.com/v2/cats`
- `new URL("v2/cats", "https://example.com/v1/") // -> https://example.com/v1/v2/cats`

If you want to have custom endpoint url resolving rules, you can override `PrepareFetchUrl` method. For more details see [issue](https://github.com/vladkens/apigen-ts/issues/2).

```ts
class MyClient extends ApiClient {
  PrepareFetchUrl(path: string) {
    return new URL(`${this.Config.baseUrl}/${path}`.replace(/\/{2,}/g, "/"))
  }
}

const api = new MyClient({ baseUrl: "https://example.com/v1" })
// will call: https://example.com/v1/pet/ instead of https://example.com/pet/
const pet = await api.pet.getPetById(404)
```

### Node.js API

Create file like `apigen.mjs` with content:

```js
import { apigen } from "apigen-ts"

await apigen({
  source: "https://petstore3.swagger.io/api/v3/openapi.json",
  output: "./api-client.ts",
  // everything below is optional
  name: "MyApiClient", // default "ApiClient"
  parseDates: true, // default false
  inlineEnums: false, // default false, use string literal union instead of enum
  headers: { "x-api-key": "secret-key" }, // Custom HTTP headers to use when fetching schema
  resolveName(ctx, op, proposal) {
    // proposal is [string, string] which represents module.funcName
    if (proposal[0] === "users") return // will use default proposal

    const [a, b] = op.name.split("/").slice(3, 5) // eg. /api/v1/store/items/search
    return [a, `${op.method}_${b}`] // [store, 'get_items'] -> apiClient.store.get_items()
  },
})
```

Then run with: `node apigen.mjs`

## Usage with different backend frameworks

### FastAPI

By default `apigen-ts` generates noisy method names when used with FastAPI. This can be fixed by [custom resolving](https://fastapi.tiangolo.com/advanced/path-operation-advanced-configuration/#using-the-path-operation-function-name-as-the-operationid) for operations ids.

```py
from fastapi import FastAPI
from fastapi.routing import APIRoute

app = FastAPI()

# add your routes here

def update_operation_ids(app: FastAPI) -> None:
    for route in app.routes:
        if isinstance(route, APIRoute):
            ns = route.tags[0] if route.tags else "general"
            route.operation_id = f"{ns}_{route.name}".lower()


# this function should be after all routes added
update_operation_ids(app)
```

_Note: If you want FastAPI to be added as preset, open PR please._

## Compare

- [openapi-typescript-codegen](https://github.com/ferdikoomen/openapi-typescript-codegen) ([npm](https://www.npmjs.com/package/openapi-typescript-codegen)): no single file mode [#1263](https://github.com/ferdikoomen/openapi-typescript-codegen/issues/1263#issuecomment-1502890838)
- [openapi-typescript](https://github.com/drwpow/openapi-typescript) ([npm](https://www.npmjs.com/package/openapi-typescript)): low level api; no named types export to use in client code
- [openapi-generator-cli](https://github.com/OpenAPITools/openapi-generator-cli) ([npm](https://www.npmjs.com/package/@openapitools/openapi-generator-cli)): wrapper around java lib
- [swagger-typescript-api](https://github.com/acacode/swagger-typescript-api) ([npm](https://www.npmjs.com/package/swagger-typescript-api)): complicated configuration; user-api breaking changes between versions

## Development

- https://ts-ast-viewer.com
- https://jsonschemalint.com
- https://redocly.github.io/redoc/
- https://swagger.io/docs/specification/basic-structure/
 readmeEtag: '"023a9604086f0838eb5f7dacaa9674e90c3c74c7"' readmeLastModified: Thu, 15 May 2025 01:33:23 GMT repositoryId: 725796572 description: >- Typed, single-file API client generator for OpenAPI schemas with minimal dependencies and maximum type safety. created: '2023-11-30T22:25:17Z' updated: '2026-01-05T22:51:13Z' language: TypeScript archived: false stars: 53 watchers: 3 forks: 10 owner: vladkens logo: https://avatars.githubusercontent.com/u/825754?v=4 license: MIT repoEtag: '"b88a00dbcfa131dda0f3d67813274fbe08090f85eebc981aef4550459c5d732b"' repoLastModified: Mon, 05 Jan 2026 22:51:13 GMT category: SDK foundInMaster: true name: apigen-ts language: TypeScript link: https://www.npmjs.com/package/apigen-ts source_description: Typed HTTP client generator as single file without extra dependencies v2: true - source: https://openapi.tools/ name: APIGit category: - Documentation - Mock - GUI Editors - Testing language: SaaS repository: https://github.com/apigitlabs link: https://apigit.com source_description: >- the native Git based collaboration platform for API design, document, mock, testing and share. v2: true v3: true v3_1: true id: eff815e9a2d41cfd4464f36c69bc11ce foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/pb33f/libopenapi v3: true id: 78ab0e104df30dd8b36325f7708a55ae repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+Cgk8aW1nIHNyYz0ibGlib3BlbmFwaS1sb2dvLnBuZyIgYWx0PSJsaWJvcGVuYXBpIiBoZWlnaHQ9IjMwMHB4IiB3aWR0aD0iNDUwcHgiLz4KPC9wPgoKIyBsaWJvcGVuYXBpIC0gZW50ZXJwcmlzZSBncmFkZSBPcGVuQVBJIHRvb2xzIGZvciBnb2xhbmcuCgoKIVtQaXBlbGluZV0oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL2xpYm9wZW5hcGkvd29ya2Zsb3dzL0J1aWxkL2JhZGdlLnN2ZykKWyFbR29SZXBvcnRDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS9wYjMzZi9saWJvcGVuYXBpKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL3BiMzNmL2xpYm9wZW5hcGkpClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9wYjMzZi9saWJvcGVuYXBpL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2Zz8pXShodHRwczovL2NvZGVjb3YuaW8vZ2gvcGIzM2YvbGlib3BlbmFwaSkKWyFbZGlzY29yZF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kaXNjb3JkLzkyMzI1ODM2MzU0MDgxNTkxMildKGh0dHBzOi8vZGlzY29yZC5nZy94N1ZBQ1Z1RUdQKQpbIVtEb2NzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2dvZG9jLXJlZmVyZW5jZS01ZmFmZDcpXShodHRwczovL3BrZy5nby5kZXYvZ2l0aHViLmNvbS9wYjMzZi9saWJvcGVuYXBpKQoKbGlib3BlbmFwaSBoYXMgZnVsbCBzdXBwb3J0IGZvciBPcGVuQVBJIDMsIDMuMSBhbmQgMy4yLiBJdCBjYW4gaGFuZGxlIHRoZSBsYXJnZXN0IGFuZCBtb3N0CmNvbXBsZXggc3BlY2lmaWNhdGlvbnMgeW91IGNhbiB0aGluayBvZi4KCi0tLQoKIyMgU3BvbnNvcnMgJiB1c2VycwpJZiB5b3VyIGNvbXBhbnkgaXMgdXNpbmcgYGxpYm9wZW5hcGlgLCBwbGVhc2UgY29uc2lkZXJpbmcgW3N1cHBvcnRpbmcgdGhpcyBwcm9qZWN0XShodHRwczovL2dpdGh1Yi5jb20vc3BvbnNvcnMvZGF2ZXNoYW5sZXkpLCAKbGlrZSBvdXIgX3Zlcnkga2luZF8gc3BvbnNvcnM6Cgo8YSBocmVmPSJodHRwczovL3NwZWFrZWFzeS5jb20vZWRpdG9yP3V0bV9zb3VyY2U9bGlib3BlbmFwaStyZXBvJnV0bV9tZWRpdW09Z2l0aHViK3Nwb25zb3JzaGlwIj4KPHBpY3R1cmU+CiAgPHNvdXJjZSBtZWRpYT0iKHByZWZlcnMtY29sb3Itc2NoZW1lOiBkYXJrKSIgc3Jjc2V0PSIuZ2l0aHViL3Nwb25zb3JzL3NwZWFrZWFzeS1naXRodWItc3BvbnNvci1kYXJrLnN2ZyI+CiAgPGltZyBhbHQ9InNwZWFrZWFzeSciIHNyYz0iLmdpdGh1Yi9zcG9uc29ycy9zcGVha2Vhc3ktZ2l0aHViLXNwb25zb3ItbGlnaHQuc3ZnIj4KPC9waWN0dXJlPgo8L2E+CgpbU3BlYWtlYXN5XShodHRwczovL3NwZWFrZWFzeS5jb20vZWRpdG9yP3V0bV9zb3VyY2U9bGlib3BlbmFwaStyZXBvJnV0bV9tZWRpdW09Z2l0aHViK3Nwb25zb3JzaGlwKQoKPGEgaHJlZj0iaHR0cHM6Ly9zY2FsYXIuY29tIj4KPHBpY3R1cmU+CiAgPHNvdXJjZSBtZWRpYT0iKHByZWZlcnMtY29sb3Itc2NoZW1lOiBkYXJrKSIgc3Jjc2V0PSIuZ2l0aHViL3Nwb25zb3JzL3NjYWxhci1kYXJrLnBuZyI+CiAgPGltZyBhbHQ9InNjYWxhciciIHNyYz0iLmdpdGh1Yi9zcG9uc29ycy9zY2FsYXItbGlnaHQucG5nIj4KPC9waWN0dXJlPgo8L2E+Cgpbc2NhbGFyXShodHRwczovL3NjYWxhci5jb20pCgo8YSBocmVmPSJodHRwczovL2FwaWRlY2suY29tIj4KPHBpY3R1cmU+CiAgPHNvdXJjZSBtZWRpYT0iKHByZWZlcnMtY29sb3Itc2NoZW1lOiBkYXJrKSIgc3Jjc2V0PSIuZ2l0aHViL3Nwb25zb3JzL2FwaWRlY2stZGFyay5wbmciPgogIDxpbWcgYWx0PSJhcGlkZWNrJyIgc3JjPSIuZ2l0aHViL3Nwb25zb3JzL2FwaWRlY2stbGlnaHQucG5nIj4KPC9waWN0dXJlPgo8L2E+CgpbYXBpZGVja10oaHR0cHM6Ly9hcGlkZWNrLmNvbSkKCi0tLQoKIyMgQ29tZSBjaGF0IHdpdGggdXMKCk5lZWQgaGVscD8gSGF2ZSBhIHF1ZXN0aW9uPyBXYW50IHRvIHNoYXJlIHlvdXIgd29yaz8gW0pvaW4gb3VyIGRpc2NvcmRdKGh0dHBzOi8vZGlzY29yZC5nZy94N1ZBQ1Z1RUdQKSBhbmQKY29tZSBzYXkgaGkhCgojIyBDaGVjayBvdXQgdGhlIGBsaWJvcGVuYXBpLXZhbGlkYXRvcmAKCk5lZWQgdG8gdmFsaWRhdGUgcmVxdWVzdHMsIHJlc3BvbnNlcywgcGFyYW1ldGVycyBvciBzY2hlbWFzPyBVc2UgdGhlIG5ldyAKW2xpYm9wZW5hcGktdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20vcGIzM2YvbGlib3BlbmFwaS12YWxpZGF0b3IpIG1vZHVsZS4KCiMjIERvY3VtZW50YXRpb24KClNlZSBhbGwgdGhlIGRvY3VtZW50YXRpb24gYXQgaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpLwoKLSBbSW5zdGFsbGluZyBsaWJvcGVuYXBpXShodHRwczovL3BiMzNmLmlvL2xpYm9wZW5hcGkvaW5zdGFsbGluZy8pCi0gW1VzaW5nIE9wZW5BUEldKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9vcGVuYXBpLykKLSBbVXNpbmcgU3dhZ2dlcl0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3N3YWdnZXIvKQotIFtUaGUgRGF0YSBNb2RlbF0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL21vZGVsLykKLSBbVmFsaWRhdGlvbl0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3ZhbGlkYXRpb24vKQotIFtNb2RpZnlpbmcgLyBNdXRhdGluZyB0aGUgT3BlbkFQSSBNb2RlbF0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL21vZGlmeWluZy8pCi0gW01vY2tpbmcgLyBDcmVhdGluZyBFeGFtcGxlc10oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL21vY2tzLykKLSBbVXNpbmcgVmVuZG9yIEV4dGVuc2lvbnNdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9leHRlbnNpb25zLykKLSBbVGhlIEluZGV4XShodHRwczovL3BiMzNmLmlvL2xpYm9wZW5hcGkvaW5kZXgvKQotIFtUaGUgUmVzb2x2ZXJdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9yZXNvbHZlci8pCi0gW1RoZSBSb2xvZGV4XShodHRwczovL3BiMzNmLmlvL2xpYm9wZW5hcGkvcm9sb2RleC8pCi0gW0NpcmN1bGFyIFJlZmVyZW5jZXNdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9jaXJjdWxhci1yZWZlcmVuY2VzLykKLSBbQnVuZGxpbmcgU3BlY3NdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9idW5kbGluZy8pCi0gW1doYXQgQ2hhbmdlZCAvIERpZmYgRW5naW5lXShodHRwczovL3BiMzNmLmlvL2xpYm9wZW5hcGkvd2hhdC1jaGFuZ2VkLykKLSBbT3ZlcmxheXNdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9vdmVybGF5cy8pCi0gW0ZBUV0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL2ZhcS8pCi0gW0Fib3V0IGxpYm9wZW5hcGldKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS9hYm91dC8pCi0tLQoKIyMjIFF1aWNrLXN0YXJ0IHR1dG9yaWFsCgrwn5GAICoqR2V0IHJvbGxpbmcgZmFzdCB1c2luZyBgbGlib3BlbmFwaWAgd2l0aCB0aGUgCltQYXJzaW5nIE9wZW5BUEkgZmlsZXMgdXNpbmcgZ29dKGh0dHBzOi8vcXVvYml4LmNvbS9hcnRpY2xlcy9wYXJzaW5nLW9wZW5hcGktdXNpbmctZ28vKSoqIGd1aWRlIPCfkYAKCk9yLCBmb2xsb3cgdGhlc2Ugc3RlcHMgYW5kIHNlZSBzb21ldGhpbmcgaW4gYSBmZXcgc2Vjb25kcy4KCiMjIyMgU3RlcCAxOiBHcmFiIHRoZSBwZXRzdG9yZQoKYGBgYmFzaApjdXJsIGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL21haW4vX2FyY2hpdmVfL3NjaGVtYXMvdjMuMC9wYXNzL3BldHN0b3JlLnlhbWwgPiBwZXRzdG9yZXYzLmpzb24KYGBgCgojIyMjIFN0ZXAgMjogR3JhYiBsaWJvcGVuYXBpCgpgYGBiYXNoCmdvIGdldCBnaXRodWIuY29tL3BiMzNmL2xpYm9wZW5hcGkKYGBgCgojIyMjIFN0ZXAgMzogUGFyc2UgdGhlIHBldHN0b3JlIHVzaW5nIGxpYm9wZW5hcGkKCkNvcHkgYW5kIHBhc3RlIHRoaXMgY29kZSBpbnRvIGEgYG1haW4uZ29gIGZpbGUuCgpgYGBnbwpwYWNrYWdlIG1haW4KCmltcG9ydCAoCgkiZm10IgoJIm9zIgoJImdpdGh1Yi5jb20vcGIzM2YvbGlib3BlbmFwaSIKKQoKZnVuYyBtYWluKCkgewoJcGV0c3RvcmUsIF8gOj0gb3MuUmVhZEZpbGUoInBldHN0b3JldjMuanNvbiIpCglkb2N1bWVudCwgZXJyIDo9IGxpYm9wZW5hcGkuTmV3RG9jdW1lbnQocGV0c3RvcmUpCglpZiBlcnIgIT0gbmlsIHsKCQlwYW5pYyhmbXQuU3ByaW50ZigiY2Fubm90IGNyZWF0ZSBuZXcgZG9jdW1lbnQ6ICVlIiwgZXJyKSkKCX0KCWRvY01vZGVsLCBlcnIgOj0gZG9jdW1lbnQuQnVpbGRWM01vZGVsKCkKCWlmIGVyciAhPSBuaWwgewoJCXBhbmljKGZtdC5TcHJpbnRmKCJjYW5ub3QgY3JlYXRlIHYzIG1vZGVsIGZyb20gZG9jdW1lbnQ6ICVlIiwgZXJyKSkKCX0KCgkvLyBUaGUgZm9sbG93aW5nIGZhaWxzIGFmdGVyIHRoZSBmaXJzdCBpdGVyYXRpb24KCWZvciBzY2hlbWFOYW1lLCBzY2hlbWEgOj0gcmFuZ2UgZG9jTW9kZWwuTW9kZWwuQ29tcG9uZW50cy5TY2hlbWFzLkZyb21PbGRlc3QoKSB7CgkJaWYgc2NoZW1hLlNjaGVtYSgpLlByb3BlcnRpZXMgIT0gbmlsIHsKCQkJZm10LlByaW50ZigiU2NoZW1hICclcycgaGFzICVkIHByb3BlcnRpZXNcbiIsIHNjaGVtYU5hbWUsIHNjaGVtYS5TY2hlbWEoKS5Qcm9wZXJ0aWVzLkxlbigpKQoJCX0KCX0KfQpgYGAKClJ1biBpdCwgd2hpY2ggc2hvdWxkIHByaW50IG91dDoKCmBgYGJhc2gKU2NoZW1hICdQZXQnIGhhcyAzIHByb3BlcnRpZXMKU2NoZW1hICdFcnJvcicgaGFzIDIgcHJvcGVydGllcwpgYGAKCgo+IFJlYWQgdGhlIGZ1bGwgZG9jcyBhdCBbaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL10oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpLykKCi0tLQoKTG9nbyBnb3BoZXIgaXMgbW9kaWZpZWQsIG9yaWdpbmFsbHkgZnJvbSBbZWdvbmVsYnJlXShodHRwczovL2dpdGh1Yi5jb20vZWdvbmVsYnJlL2dvcGhlcnMpCg== readmeEtag: '"ec9efa6a77bec2aee6aaad4273b44a8e96b01760"' readmeLastModified: Tue, 23 Dec 2025 16:03:54 GMT repositoryId: 514939258 description: >- libopenapi is a fully featured, high performance OpenAPI 3.2, 3.1, 3.0 and Swagger parser, library, validator and toolkit for golang applications. created: '2022-07-17T19:41:10Z' updated: '2026-02-05T15:16:47Z' language: Go archived: false stars: 780 watchers: 7 forks: 98 owner: pb33f logo: https://avatars.githubusercontent.com/u/104016643?v=4 license: MIT repoEtag: '"6f231ea9757392b8a3fabe5c51e95dd8d73c62f1c5b49dc55f3b96d76f0b038f"' repoLastModified: Thu, 05 Feb 2026 15:16:47 GMT category: Parsers foundInMaster: true name: libopenapi link: https://pb33f.io/libopenapi language: go source_description: >- Enterprise grade, fully featured OpenAPI 3.1, 3.0 and Swagger parser library for go. A complete toolset for reading and parsing OpenAPI and Swagger specifications. Comes complete with high and low-level APIs, diff engine, index and resolver. v2: true v3_1: true - source: https://openapi.tools/ name: go-openapi-spec-code-diffs category: - Validator - Parsers link: https://github.com/RHEcosystemAppEng/go-openapi-spec-code-diffs language: go repository: https://github.com/rhecosystemappeng/go-openapi-spec-code-diffs source_description: >- A golang validation tool that compares given OpenAPI specs (e.g. openapi.yaml) vis-a-vis routes (e.g. /api/v1/customer/:id) defined in golang source code and reports differences. This is useful in scenarios where you want to keep the OpenAPI specs and Code in synch. v3: true v3_1: true id: e7024c0e9e4e358fa3a62d42b0f1b052 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNzIFZhbGlkYXRvcgoKIyMgU3VtbWFyeQpBIHZhbGlkYXRpb24gdG9vbCB0aGF0IGNvbXBhcmVzIE9wZW5BUEkgc3BlY3MgdmlzLWEtdmlzIHJvdXRlcyAoZS5nLiAvYXBpL3YxL2N1c3RvbWVyLzppZCkgZGVmaW5lZCBpbiBnb2xhbmcgc291cmNlIGNvZGUuIFVzZWZ1bCBpbiBzY2VuYXJpb3Mgd2hlcmUgeW91IHdhbnQgdG8gZW5zdXJlIHRoZSBPcGVuQVBJIHNwZWNzIGFuZCBDb2RlIGFyZSBpbiBzeW5jaC4KCiMjIFdoYXQKVGhpcyB2YWxpZGF0b3IgdmFsaWRhdGVzIEFQSSBFbmRwb2ludHMgZGVmaW5lZCBpbiBnb2xhbmcgc291cmNlIGNvZGUgd2l0aCBPcGVuQVBJIHNwZWNpZmljYXRpb25zIGFuZCByZXBvcnRzIGZvbGxvd2luZyBkaWZmZXJlbmNlcy4KKiBPcGVuQVBJIHNwZWNzIGZvdW5kIGluIHNwZWNpZmljYXRpb25zIGJ1dCBub3QgZm91bmQgaW4gZ29sYW5nIHNvdXJjZSBjb2RlCiogUm91dGVzIGZvdW5kIGluIGdvbGFuZyBzb3VyY2UgY29kZSBidXQgbm90IGZvdW5kIGluIE9wZW5BUEkgc3BlY3MuCgojIyBXaHkKKiBIZWxwcyB0byBrZWVwIE9wZW5BUEkgc3BlY3MgYW5kIGdvbGFuZyBzb3VyY2UgY29kZSBpbiBzeW5jaC4KKiBEbyBub3QgZm9yZ2V0IHRvIGltcGxlbWVudCBwYXRocyBkZWZpbmVkIGluIE9wZW5BUEkgc3BlY3MgaW4gZ29sYW5nIHNvdXJjZSBjb2RlLgoqIERvIG5vdCBmb3JnZXQgdG8gaW5jbHVkZSBBUElzIGltcGxlbWVudGVkIGluIGdvbGFuZyBzb3VyY2UgY29kZSBpbiBPcGVuQVBJIHNwZWNzIGluLgoKIyMgSG93CiogVGhpcyB0b29sIHRha2VzIGdvbGFuZyBzb3VyY2Ugcm9vdCBkaXJlY3Rvcnkgd2hpY2ggaW1wbGVtZW50cyB5b3VyIEFQSSBhbmQgT3BlbkFQSSBzcGVjcyBmaWxlIGFzIGlucHV0cy4KKiBUaGUgdG9vbCBlc3NlbnRpYWxseSBidWlsZHMgdHdvIGxpc3RzOiAKICAxLiBMaXN0IG9mIHJvdXRlcy9wYXRocyBmb3VuZCBpbiBnb2xhbmcgc291cmNlIGNvZGUuCiAgMi4gTGlzdCBvZiByb3V0ZXMvcGF0aHMgZGVmaW5lZCBpbiB5b3VyIE9wZW5BUEkgc3BlY3MgZmlsZS4KICAqIEJvdGggdGhlIGxpc3RzIHNob3VsZCBtYXRjaCBpZiBub3QgdGhlIHRvb2wgd2lsbCByZXBvcnQgZGlmZmVyZW5jZXMuCiogVGhlIHRvb2wgdXNlcyByZWd1bGFyIGV4cHJlc3Npb25zIHRvIGZpbmQgcGF0aHMgZGVmaW5lZCBpbiBnb2xhbmcgc291cmNlIGNvZGUgZS5nLiAiL2hlYWx0aC9yZWFkeSIsICIvdXNlcnMiIGV0Yy4KICAqIE9uY2Ugc3VjaCBhIHBhdGggaXMgZm91bmQsIHRoZSBjb3JyZXNwb25kaW5nIGxpbmUgaXMgc2Nhbm5lZCB0byBsb29rIGZvciBhIGh0dHBtZXRob2Qgc3VjaCBhcyBHRVQsIFBVVCwgUE9TVCwgREVMRVRFLCBIRUFELCBPUFRJT05TIGFuZCBQQVRDSC4KICAqIElmIHRoZSBodHRwbWV0aG9kIGlzIGZvdW5kIG9uIHRoZSBsaW5lIHRoZW4gdG9vbCBjb25zaWRlcnMgdGhlIHBhdGggZm91bmQgaW4gdGhlIGxpbmUgYXMgYSB2YWxpZCByb3V0ZS9wYXRoIGRlZmluaXRpb24uCiAgKiBUaHVzIGZvbGxvd2luZyBjb2RlIGlzIGNvbnNpZGVyZWQgYSB2YWxpZCByb3V0ZS9wYXRoIGRlZmluaXRpb24gYnkgdGhlIHRvb2wuCmBgYGdvCnJvdXRlIDo9IHJvdXRlUmVnaXN0cmF0aW9ueyJQVVQiLCAiL3VzZXIvOmlkL2FkbWluLzppc0FkbWluIiwgaGFuZGxlcnMuU2V0QWRtaW5TdGF0dXN9CmBgYAogICogSG93ZXZlciB0aGUgZm9sbG93aW5nIGxpbmUgaXMgbm90IGNvbnNpZGVyZWQgYSB2YWxpZCByb3V0ZS9wYXRoIGRlZmluaXRpb24gYXMgdGhpcyBpcyB0aGUgbGluZSBoYXZpbmcgYSBnb2xhbmcga2V5d29yZCAnaWYnLgpgYGBnbwppZiAiL3RoaXMvaXMvbm90L2NvbnNpZGVyZWQvYS9wYXRoL2RlZmluaXRpb24iID09ICJERUxFVEUiIHsKYGBgCiAgKiBZb3UgY2FuIG1ha2UgdXNlIG9mIHRoZSBmb2xsb3dpbmcgaWdub3JlIGVsZW1lbnRzIHdoaWxlIHNjYW5uaW5nIGdvbGFuZyBzb3VyY2UgY29kZSwgdG8gZnVydGhlciBmaW5lIHR1bmUgdGhlIHJ1biBvZiB0aGUgdG9vbC4gCgp8IEVsZW1lbnQgICAgICAgfCBEZXNjcmlwdGlvbiB8CnwtLS0tLS0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLS0tLXwKfCBpZ25vcmVkRGlyc0ZpbGUgIHwgRGlyZWN0b3JpZXMgdG8gYmUgaWdub3JlZCB3aGVuIHNjYW5uaW5nIGdvbGFuZyBzb3VyY2UgY29kZSB8CnwgaWdub3JlZEZpbGVzRmlsZSB8IEZpbGVzIHRvIGJlIGlnbm9yZWQgfAp8IGlnbm9yZWRMaW5lc0ZpbGUgfCBMaW5lcyB0byBiZSBpZ25vcmVkIHwKfCBpZ25vcmVkUGF0aHNGaWxlIHwgQVBJIFBhdGhzIHRvIGJlIGlnbm9yZWQgZnJvbSBjb21wYXJpc29uIHN1Y2ggYXMgL2hlYWx0aC9yZWFkeSBvciAvaGVhbHRoL2xpdmUgfAoKIyMgV2h5IHRoZSBpZ25vcmVkIGVsZW1lbnRzIGFib3ZlIGFyZSBuZWVkZWQ/CiogVGhlIHRvb2wgcmVjdXJzaXZlbHkgd29ya3Mgb24gLmdvIGZpbGVzIGZvdW5kIGluIGdvbGFuZyBzb3VyY2UgY29kZSBkaXJlY3RvcnkuCiogWW91IGNhbiBzbGlnaHRseSBpbXByb3ZlIHRoZSBwZXJmb3JtYW5jZSBieSBzcGVjaWZ5aW5nIGRpcmVjdG9yaWVzIHdoaWNoIHR5cGljYWxseSBkbyBub3QgY29udGFpbiAuZ28gZmlsZXMgc3VjaCBhcyBiaW4sIG91dCBhbmQgLmdpdCBlLmcuCiogWW91IGNhbiB1c2UgaWdub3JlZEZpbGVzRmlsZSB0byBmdXJ0aGVyIGV4Y2x1ZGUgc3BlY2lmaWMgZmlsZXMgZnJvbSBzY2FubmluZy4KKiBJZiB0aGUgaWdub3JlZCBkaXJlY3RvcmllcyBhbmQgZmlsZXMgYXJlIG5vdCBzdWZmaWNpZW50LCBpZiB0b29sIGNvbmZ1c2VzIGNlcnRhaW4gY29kZSBsaW5lcyB0byBjb250YWluIHZhbGlkIHBhdGhzL3JvdXRlcyBidXQgaWYgdGhleSBhcmUgbm90IHZhbGlkIHBhdGhzL3JvdXRlcywgc2ltcGx5IGNvcHkgdGhvc2UgbGluZSBpbiBpZ25vcmVkTGluZXNGaWxlIGFuZCB0aG9zZSBsaW5lcyB3aWxsIGJlIGlnbm9yZWQgZnJvbSBzY2FuLgoKIyMgVXNhZ2UKIyMjIEFzIHNoZWxsIGNvbW1hbmQKKiBUaGUgY29tbWFuZCBsaW5lIHZlcnNpb24gbWFrZXMgdXNlIG9mIG5hbWVkIGFyZ3VtZW50cy4gCiogQXQgYW55IHBvaW50IHRvIHNlZSB0aGUgaGVscCBmb3IgdGhlIGNvbW1hbmQgdXNlIHRoZSBmb2xsb3dpbmcKYGBgc2hlbGwKb3BlbmFwaV9zcGVjX2NvZGVfZGlmZnMgLS1oZWxwCmBgYAoqIEZvcm1hdApgYGBzaGVsbApvcGVuYXBpX3NwZWNfY29kZV9kaWZmcyAvCiAgICAtb3BlbkFQSVNwZWNzRmlsZSAncGF0aC90by9vcGVuYXBpL3NwZWNzL2ZpbGVuYW1lJyAvCiAgICAtZ29Tb3VyY2VEaXIgJ3BhdGgvdG8vZ29sYW5nL3NvdXJjZS9kaXInIC8KICAgIC1pZ25vcmVkRGlyc0ZpbGUgJ3BhdGgvdG8vaWdub3JlZC9kaXJlY3Rvcmllcy9maWxlbmFtZScgLyAKICAgIC1pZ25vcmVkRmlsZXNGaWxlICdwYXRoL3RvL2lnbm9yZWQvZmlsZXMvZmlsZW5hbWUnIC8KICAgIC1pZ25vcmVkTGluZXNGaWxlICdwYXRoL3RvL2lnbm9yZWQvbGluZXMvZmlsZW5hbWUnIC8gCiAgICAtaWdub3JlZFBhdGhzRmlsZSAncGF0aC90by9pZ25vcmVkL3BhdGhzL2ZpbGVuYW1lJyAvIAogICAgLWxvZ0xldmVsICdsb2ctbGV2ZWwgY2FuIGJlIG9uZSBvZjogZGlzYWJsZWQsIGluZm8sIGRlYnVnLCBlcnJvcicKYGBgCgojIyMgQXMgYSBwYWNrYWdlL2xpYnJhcnkgaW4gZ29sYW5nIHNvdXJjZSBjb2RlCiogSW1wb3J0IHRoZSBwYWNrYWdlL2xpYnJhcnkKYGBgZ28KaW1wb3J0ICJnaXRodWIuY29tL1JIRWNvc3lzdGVtQXBwRW5nL29wZW5hcGlfc3BlY19jb2RlX2RpZmZzL3ZhbGlkYXRvciIKYGBgCgoqIFVzZSB0aGUgdmFsaWRhdG9yIGUuZy4gaW4gdGVzdCBjb2RlIGFzIGZvbGxvd3MuIFBhdGhzIGJlbG93IGFyZSByZWxhdGl2ZSB0byB0aGUgcGF0aCBvZiB0aGUgdGVzdCBmaWxlIGZyb20gd2hpY2ggdGVzdHMgYXJlIGdvaW5nIHRvIGJlIGV4ZWN1dGVkIGZyb20uCmBgYGdvCmZ1bmMgdmFsaWRhdGVPcGVuQVBJU3BlY3ModCAqdGVzdGluZy5UKSB7CglvYXNTdGF0aWNWYWxpZGF0b3IgOj0gdmFsaWRhdG9yLk5ld09wZW5BUElTcGVjQ29kZURpZmZzVmFsaWRhdG9yKCIuL29hc1N0YXRpY1ZhbGlkYXRvci8uZGlyaWdub3JlIiwgIi4vb2FzU3RhdGljVmFsaWRhdG9yLy5zcGVjaWdub3JlIiwgIi4uLy4uLyIsICIuLi8uLi9vcGVuYXBpLnlhbWwiKQoJZXJyLCByZXN1bHQgOj0gb2FzU3RhdGljVmFsaWRhdG9yLlZhbGlkYXRlKCkKCglhc3NlcnQuTmlsKHQsIGVyciwgIk5vIGVycm9ycyByZXR1cm5lZCBmcm9tIG9wZW5hcGkgdmFsaWRhdGlvbiIpCglhc3NlcnQuRXF1YWwodCwgMCwgbGVuKHJlc3VsdC5TcGVjRGVmc05vdEluQ29kZSksICJGb3VuZCBzcGVjIGRlZnMgbm90IGltcGxlbWVudGVkIGluIGNvZGUiLCBsZW4ocmVzdWx0LlNwZWNEZWZzTm90SW5Db2RlKSkKCWFzc2VydC5FcXVhbCh0LCAwLCBsZW4ocmVzdWx0LkNvZGVEZWZzTm90U3BlYyksICJGb3VuZCBjb2RlIGRlZnMgbm90IHJlZmxlY3RlZCBpbiBzcGVjcyIsIGxlbihyZXN1bHQuQ29kZURlZnNOb3RTcGVjKSkKfQpgYGAKCiMjIFJ1bm5pbmcgdGVzdHMKKiBVc2UgdGhlIHN0YW5kYXJkIGdvIGNvbW1hbmQgdG8gcnVuIHRlc3RzIGFzIGZvbGxvd3MsIHdoaWNoIGFyZSBkaXZpZGVkIGFzIHBvc2l0aXZlIGFuZCBuZWdhdGl2ZSB0ZXN0czoKYGBgc2hlbGwKIGdvIHRlc3QgLXYgLi90ZXN0cy9wb3NpdGl2ZSAuL3Rlc3RzL25lZ2F0aXZlCmBgYA== readmeEtag: '"bab8c8dd6a236ec5043056e8f9d10c2f4320c0b1"' readmeLastModified: Mon, 24 Jul 2023 20:40:53 GMT repositoryId: 666011427 description: null created: '2023-07-13T13:55:40Z' updated: '2025-09-18T23:39:10Z' language: Go archived: false stars: 3 watchers: 2 forks: 1 owner: RHEcosystemAppEng logo: https://avatars.githubusercontent.com/u/82061998?v=4 license: Apache-2.0 repoEtag: '"d26e9f4c10363c1e23401780a2c3bc12ef66d0dc143b0f03c2305a20259fe59c"' repoLastModified: Thu, 18 Sep 2025 23:39:10 GMT foundInMaster: true - source: https://openapi.tools/ name: draig category: SDK link: https://gitlab.com/robarago/draig language: Java repository: https://gitlab.com/robarago/draig source_description: >- DRAIG is an OpenAPI3 based design-first custom generator that creates fully functional microservices from an OAS API description document. You can use DRAIG with openapi-generator as a backend generator or use DRAIG CLI or REPL to generate your microservices. With simple user-customizable templates, you can extended and improve implementation code so that generated code can be compiled and deployed immediatelly. v2: false v3: true v3_1: false id: 3a9dfe0c9cec65c2ce3c97d0b8e8186f foundInMaster: true - source: https://openapi.tools/ name: draig-car category: - Server - Testing - Text Editors link: https://gitlab.com/robarago/draig-car language: TypeScript repository: https://gitlab.com/robarago/draig-car source_description: >- This is a REPL OpenAPI3 description document text editor. It provides both a CLI and and a REPL you can use to test your APIs, but you can also test database queries, start and stop database (containers), generate fake data automatically, start your test microservice and much more. You don't have to exit from your REPL! Or you can use the CLI to do the same if you want, giving you the option to automatize your contract-first based dev cycles. v2: false v3: true v3_1: false id: ce0dc163dc2b30744344ee24800647a0 foundInMaster: true - source: https://openapi.tools/ name: ABAP OpenAPI Client & Server Generator category: - SDK - Server - Parsers language: ABAP repository: https://github.com/abap-openapi/abap-openapi link: https://abap-openapi.github.io/web-openapi-client/ source_description: >- ABAP OpenAPI Client & Server Generator written in ABAP. Code generation runs also in the Browser and on command line using NodeJS. v3: true id: 9f36fdafa836d6b231bd4fe0cd2b34b7 repositoryMetadata: base64Readme: >- IyBBQkFQIE9wZW5BUEkgQ2xpZW50ICYgU2VydmVyIEdlbmVyYXRvcgoKQUJBUCBPcGVuQVBJIGlzIGFuIFtPcGVuQVBJXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcpIGdlbmVyYXRvciB0b29sIGRlc2lnbmVkIHRvIGNyZWF0ZSBBUEkgY2xpZW50cywgSUNGIHNlcnZlciBoYW5kbGVycywgYW5kIElDRiBzZXJ2ZXIgaW1wbGVtZW50YXRpb24gc3R1YnMgZnJvbSBPcGVuQVBJIGRvY3VtZW50cy4KCiMjIEdlbmVyYXRlCgpJdCBpcyBwb3NzaWJsZSB0byBnZW5lcmF0ZSB0aGUgb2JqZWN0cyB1c2luZyBkaWZmZXJlbnQgcHJvY2Vzc2VzOgotIEdlbmVyYXRlIHZpYSBvdXIgW3dlYiBjbGllbnRdKGh0dHBzOi8vYWJhcC1vcGVuYXBpLmdpdGh1Yi5pby93ZWItb3BlbmFwaS1jbGllbnQvKQotIEdlbmVyYXRlIHZpYSBjb21tYW5kIGxpbmUgKE5vZGVKUykKLSBHZW5lcmF0ZSB2aWEgQUJBUAoKIyMgRmVhdHVyZXMKCnwgRmVhdHVyZSB8ICB8CnwgLS0tIHwgLS0tIHwKfCBPcGVuQVBJIEZpbGUgVHlwZXMgfCBKU09OIHwKfCBPcGVuQVBJIFZlcnNpb25zIHwgdjJcKiwgdjMgfAp8IEFCQVAgVmVyc2lvbnMgfCB2NzAyIGFuZCB1cCAgfAp8IE9iamVjdCBDcmVhdGlvbiB8IG9uZSBzZWxmLWNvbnRhaW5lZCBnbG9iYWwgY2xhc3MgJiBpbnRlcmZhY2UgcGVyIE9wZW5BUEkgZGVmaW5pdGlvbiB8CgpcKiBPcGVuQVBJIHYyIGlzIGN1cnJlbnRseSBvbmx5IGNhcGFibGUgYnkgY29udmVydGluZyB0aGUgdjIgZmlsZSB0byBhIHYzIGZpbGUuIFRoaXMgY2FuIGJlIGRvbmUgbWFudWFsbHkgdXNpbmcgdGhlIFtTd2FnZ2VyIEVkaXRvcl0oaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby8pLCBvciBwcm9ncmFtbWF0aWNhbGx5IHVzaW5nIFtTd2FnZ2VyIENvbnZlcnRlcl0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29udmVydGVyKQoKTk9URTogZ2VuZXJhdGVkIGNvZGUgY3VycmVudGx5IHVzZXMgWkNMX09BUElfSlNPTiwgc3VnZ2VzdCBjb3B5aW5nIHRoZSBpbXBsZW1lbnRhdGlvbiB0byBhIGxvY2FsIGNsYXNzIGluIHRoZSBnZW5lcmF0ZWQgZ2xvYmFsIGNsYXNzCgojIyBVc2UgQ2FzZXMKIyMjIEFQSSBDcmVhdG9yCi0gV3JpdGUgdGhlIE9wZW5BUEkgZG9jdW1lbnQgZm9yIHlvdXIgQVBJIHNvIHRoYXQgeW91IGNhbiBnZW5lcmF0ZSB0aGUgSUNGIGhhbmRsZXIgYW5kIFNlcnZlciBJbXBsZW1lbnRhdGlvbiBib2lsZXJwbGF0ZQotIFVzZSB0aGUgT3BlbkFQSSBkb2N1bWVudCB0byBjcmVhdGUgYXV0b21hdGljIGRvY3VtZW50YXRpb24gZm9yIHlvdXIgQVBJCgojIyMgQVBJIFVzZXIKLSBVc2UgYW55IE9wZW5BUEkgZG9jdW1lbnQgdG8gY3JlYXRlIHRoZSBjbGllbnQgY2xhc3MgdG8gY29uc3VtZSBleHRlcm5hbCBBUElzCgojIyBCdWlsZGluZy9EZXZlbG9waW5nCiMjIyBQcmVyZXF1aXNpdGVzCltOb2RlSlNdKGh0dHBzOi8vbm9kZWpzLm9yZykgMTYrCgojIyMgU2V0dXAgCi0gY2xvbmUgdGhpcyByZXBvc2l0b3J5Ci0gcnVuIGBucG0gaW5zdGFsbGAKCiMjIyBUZXN0aW5nClVuaXQgVGVzdHM6IGBucG0gdGVzdGAKSW50ZXJncmF0aW9uIFRlc3RzOiBgbnBtIHJ1biBpbnRlZ3JhdGlvbl90ZXN0YAoKWW91IGNhbiB0cnkgb3V0IHRoZSBnZW5lcmF0aW9uIHVzaW5nIFN3YWdnZXIncyBQZXRzdG9yZSBFeGFtcGxlOgotIEp1c3QgcnVuIGBucG0gcnVuIHBldHN0b3JlYAotIFRoZSBvdXRwdXQgZmlsZXMgd2lsbCBiZSBnZW5lcmF0ZWQgaW4gYC4vdGVzdF92MS9nZW5lcmF0ZWQvYAoKIyMgRXhhbXBsZQoKYGBgc2gKIyEvdXNyL2Jpbi9lbnYgYmFzaAoKcm0gLXJmIGFiYXAtb3BlbmFwaQpnaXQgY2xvbmUgLS1kZXB0aD0xIGh0dHBzOi8vZ2l0aHViLmNvbS9hYmFwLW9wZW5hcGkvYWJhcC1vcGVuYXBpCnJtIC1yZiBhYmFwLW9wZW5hcGkvLmdpdApjZCBhYmFwLW9wZW5hcGkKbnBtIGNpCm5wbSBydW4gdHJhbnNwaWxlCnJtIC4uL3NyYy9hcGkvKi5hYmFwCm5vZGUgdGVzdF92Mi9pbmRleC5tanMgPGZpbGVuYW1lPiAuLi9zcmMvYXBpIDxuYW1lPgpgYGAK readmeEtag: '"62f0f75d0377e6de12a6bf5a97fd9b67cb12f236"' readmeLastModified: Thu, 12 Sep 2024 13:07:42 GMT repositoryId: 331034649 description: ABAP OpenAPI Client and Server generator in ABAP created: '2021-01-19T16:03:52Z' updated: '2026-01-22T02:59:45Z' language: ABAP archived: false stars: 94 watchers: 11 forks: 17 owner: abap-openapi logo: https://avatars.githubusercontent.com/u/77685491?v=4 license: MIT repoEtag: '"4642426452f19e7e9341a60793b110bb5ea810a8e6a579ad4bf9ce3ed0f7062c"' repoLastModified: Thu, 22 Jan 2026 02:59:45 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/kubeshop/kusk-gateway v3: true repositoryMetadata: base64Readme: >- PiBEZWFyIHVzZXJzIGFuZCBjb250cmlidXRvcnMgb2YgS3VzayBhbmQgS3VzayBHYXRld2F5IQo+Cj4gV2UgY29udGludWUgdG8gYmVsaWV2ZSBpbiB0aGUgaWRlYSBvZiBPcGVuQVBJIGRyaXZlbiBHYXRld2F5LiBIb3dldmVyLCB3ZSBoYXZlIGRlY2lkZWQgdG8gcGF1c2Ugb25nb2luZyBhY3RpdmUgZGV2ZWxvcG1lbnQgYW5kIHN1cHBvcnQgZm9yIHRoZSB0aW1lIGJlaW5nLgo+IAo+IFdlIHdhbnQgdG8gZXhwcmVzcyBvdXIgc2luY2VyZSBncmF0aXR1ZGUgdG8gZXZlcnlvbmUgd2hvIGhhcyBjb250cmlidXRlZCB0byBLdXNrLCB3aGV0aGVyIGJ5IHN1Ym1pdHRpbmcgY29kZSwgcmVwb3J0aW5nIGlzc3Vlcywgb3Igc2ltcGx5IHVzaW5nIGl0LiBXZSBhcmUgcHJvdWQgb2YgdGhlIHdvcmsgd2UgYWNjb21wbGlzaGVkIHRvZ2V0aGVyLCBhbmQgaG9wZSB0aGF0IG91ciBlZmZvcnRzIGhhdmUgaGFkIGEgcG9zaXRpdmUgaW1wYWN0IG9uIHlvdXIgd29yayBhbmQgeW91ciBjb21tdW5pdHkuCj4gCj4gSWYgeW91J3JlIHdpbGxpbmcgdG8gaW1wcm92ZSB0aGUgcHJvamVjdCwgcGxlYXNlIGZlZWwgZnJlZSB0byBjb250cmlidXRlLiBXZSB3aWxsIGRvIG91ciBiZXN0IHRvIHJldmlldyBhbmQgYXBwcm92ZSBpbmNvbWluZyBQUnMgYXMgb3VyIHRpbWUgYW5kIHJlc291cmNlcyBhbGxvdy4KPiAKPiBUaGFuayB5b3UgZm9yIHlvdXIgdW5kZXJzdGFuZGluZyBhbmQgc3VwcG9ydCEKPiAKPiBUaGUgS3VzayBUZWFtIGF0IEt1YmVzaG9wCgo8cCBhbGlnbj0iY2VudGVyIj4KICA8aW1nIHN0eWxlPSJ3aWR0aDo2NiUiIHNyYz0iYXNzZXRzL2t1c2stY29sb3ItbGcucG5nI2doLWxpZ2h0LW1vZGUtb25seSIgYWx0PSJLdXNrIExvZ28gTGlnaHQiLz4KICA8aW1nIHN0eWxlPSJ3aWR0aDo2NiUiIHNyYz0iYXNzZXRzL2t1c2std2hpdGUtbGcucG5nI2doLWRhcmstbW9kZS1vbmx5IiBhbHQ9Ikt1c2sgTG9nbyBEYXJrIiAvPgo8L3A+Cgo8cCBhbGlnbj0iY2VudGVyIj4KICBXZWxjb21lIHRvIEt1c2sgR2F0ZXdheSAtIGEgc2VsZi1zZXJ2aWNlIEFQSSBnYXRld2F5IHBvd2VyZWQgYnkgPGEgaHJlZj0iaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLyI+T3BlbkFQSTwvYT4gYW5kIDxhIGhyZWY9Imh0dHBzOi8vd3d3LmVudm95cHJveHkuaW8iPkVudm95PC9hPgo8L3A+Cgo8cCBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL2t1c2suaW8iPldlYnNpdGU8L2E+Jm5ic3A7fCZuYnNwOwogIDxhIGhyZWY9Imh0dHBzOi8va3ViZXNob3AuZ2l0aHViLmlvL2t1c2stZ2F0ZXdheSI+RG9jdW1lbnRhdGlvbjwvYT4mbmJzcDt8Jm5ic3A7CiAgPGEgaHJlZj0iaHR0cHM6Ly90d2l0dGVyLmNvbS9LdXNrX2lvIj5Ud2l0dGVyPC9hPiZuYnNwO3wmbmJzcDsKICA8YSBocmVmPSJodHRwczovL2Rpc2NvcmQuZ2cvaGZxNDR3dFI2USI+RGlzY29yZDwvYT4mbmJzcDt8Jm5ic3A7CiAgPGEgaHJlZj0iaHR0cHM6Ly9rdWJlc2hvcC5pby9jYXRlZ29yeS9rdXNrIj5CbG9nPC9hPgo8L3A+Cgo8ZGl2IGFsaWduPSJjZW50ZXIiPgogIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9rdWJlc2hvcC9rdXNrLWdhdGV3YXkvcmVsZWFzZXMiPjxpbWcgdGl0bGU9IlJlbGVhc2UiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvdi9yZWxlYXNlL2t1YmVzaG9wL2t1c2stZ2F0ZXdheSIvPjwvYT4KICA8YSBocmVmPSIiPjxpbWcgdGl0bGU9IkdvIHZlcnNpb24iIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvZ28tbW9kL2dvLXZlcnNpb24va3ViZXNob3Ava3Vzay1nYXRld2F5Ii8+PC9hPgogIDxhIGhyZWY9IiI+PGltZyB0aXRsZT0iQ29kZSBidWlsZHMiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvd29ya2Zsb3cvc3RhdHVzL2t1YmVzaG9wL2t1c2stZ2F0ZXdheS9idWlsZCUyMGFuZCUyMHRlc3QiLz48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2t1YmVzaG9wL2t1c2stZ2F0ZXdheS9yZWxlYXNlcyI+PGltZyB0aXRsZT0iUmVsZWFzZSBkYXRlIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3JlbGVhc2UtZGF0ZS9rdWJlc2hvcC9rdXNrLWdhdGV3YXkiLz48L2E+CiAgPGEgaHJlZj0iIj48aW1nIHRpdGxlPSJTdGFycyIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9zdGFycy9rdWJlc2hvcC9rdXNrLWdhdGV3YXk/Y29sb3I9Ymx1ZXZpb2xldCZsYWJlbD1TdGFyZ2F6ZXJzIi8+PC9hPgoKICBbIVtTbGFja10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9zbGFjay1qb2luJTIwY2hhdC1lMDE1NjMuc3ZnP2xvZ289c2xhY2spXShodHRwczovL2t1YmVzaG9wLW9zcy5zbGFjay5jb20vYXJjaGl2ZXMvQzAzUjRKQlRNVVopCiAgWyFbRGlzY29yZF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kaXNjb3JkLzg4NDQ2NDU0OTM0NzA3NDA0OT9sYWJlbD1kaXNjb3JkJmxvZ289ZGlzY29yZCldKGh0dHBzOi8vZGlzY29yZC5jb20vY2hhbm5lbHMvODg0NDY0NTQ5MzQ3MDc0MDQ5KQo8L2Rpdj4KCiMgV2VsY29tZSB0byBLdXNrIEdhdGV3YXkKCkt1c2sgR2F0ZXdheSBlbmFibGVzIHlvdSB0byBkZXNpZ24gYW5kIGNvbmZpZ3VyZSB5b3VyIEFQSXMgZnJvbSBzaW5nbGUgT3BlbkFQSSBtYW5pZmVzdCBhbmQgeW91IGNhbjoKCi0g4pqhICoqUmFwaWRseSBwcm90b3R5cGUgeW91ciBBUElzKiogYnkgbW9ja2luZyB5b3VyIEFQSSByZXNwb25zZXMsIGFsbG93aW5nIHlvdXIgdGVhbXMgdG8gaW5zdGFudGx5IHN0YXJ0IGJ1aWxkaW5nIG9uIHRvcCBvZiB5b3VyIEFQSXMgd2l0aG91dCB5b3VyIHNlcnZpY2VzIGJlaW5nIGltcGxlbWVudGVkCi0g8J+UkiBQcm90ZWN0IHlvdXIgZW5kcG9pbnRzIHdpdGggKiphdXRvbWF0aWMgcmVxdWVzdCBhbmQgcmVzcG9uc2UgdmFsaWRhdGlvbnMqKgotIOKaoO+4jyBDb25maWd1cmUgY3JpdGljYWwgcG9saWNpZXMgbGlrZSByZXF1ZXN0IHRpbWVvdXRzIGFuZCBDT1JTIHdpdGggKipubyBjb2RpbmcgcmVxdWlyZWQqKgotIPCfp5AgQ2VudHJhbGx5IGNvbnRyb2wgeW91ciBBUElzIGZyb20gYW4gT3BlbiBTb3VyY2UgZGFzaGJvYXJkCgojIyBHZXQgU3RhcnRlZAoKQ2hlY2sgb3V0IG91ciBbR2V0dGluZyBTdGFydGVkIHNlY3Rpb25dKGh0dHBzOi8va3ViZXNob3AuZ2l0aHViLmlvL2t1c2stZ2F0ZXdheS9nZXR0aW5nLXN0YXJ0ZWQpIHRvIGluc3RhbGwgS3VzayBHYXRld2F5LCBkZXBsb3kgYW4gZXhhbXBsZSBBUEkgYW5kIGNvbm5lY3QgYSBzZXJ2aWNlIHRvIHRoZSBBUEkuCgojIyBDb250cmlidXRpbmcKCkdvIHRvIGNvbnRyaWJ1dGlvbiBkb2N1bWVudCB0byByZWFkIG1vcmUgaG93IGNhbiB5b3UgaGVscCB1cyDwn5SlCgojIyBGZWVkYmFjawoKV2hldGhlciBpdCBoZWxwcyB5b3Ugb3Igbm90IC0gd2UnZCBMT1ZFIHRvIGhlYXIgZnJvbSB5b3UuIFBsZWFzZSBsZXQgdXMga25vdyB3aGF0IHlvdSB0aGluayBhbmQgb2YgY291cnNlLCBob3cgd2UgY2FuIG1ha2UgaXQgYmV0dGVyLiBQbGVhc2Ugam9pbiBvdXIgZ3Jvd2luZyBjb21tdW5pdHkgb24gW0Rpc2NvcmRdKGh0dHBzOi8vZGlzY29yZC5jb20vaW52aXRlLzZ6dXBDWkZRYmUpLgoKIyMgRGV2ZWxvcG1lbnQKClNlZSBbYFNLQUZGT0xELm1kYF0oLi9TS0FGRk9MRC5tZCkuCg== readmeEtag: '"eed208d1215ce66a8cf64443577e0139c5ff56aa"' readmeLastModified: Tue, 21 Feb 2023 10:07:52 GMT repositoryId: 408727871 description: Kusk-gateway is an OpenAPI-driven API Gateway for Kubernetes created: '2021-09-21T07:33:29Z' updated: '2025-12-16T18:21:00Z' language: Go archived: false stars: 281 watchers: 15 forks: 24 owner: kubeshop logo: https://avatars.githubusercontent.com/u/82541796?v=4 license: MIT repoEtag: '"3ca431cd28a98dc4733017d0d8fcb5c242f1faf0b0cd020193699d349cb6f1d2"' repoLastModified: Tue, 16 Dec 2025 18:21:00 GMT foundInMaster: true category: - Documentation - Mock - Gateway - Server Implementations id: b0e18adcd3ab375ea18858db2f04ec17 name: Kusk Gateway link: https://docs.kusk.io/ language: Kubernetes source_description: >- Kusk-Gateway is an OpenAPI-driven API Gateway for Kubernetes. It empowers you to develop, validate, mock and deploy your APIs in a matter of minutes using both manual and automated GitOps/APIOps workflows. v2: true v3_1: false - source: https://openapi.tools/ name: openapi-comparator language: C# repository: https://github.com/criteo/openapi-comparator link: https://github.com/criteo/openapi-comparator source_description: C# library for comparing two OpenAPI specifications. v3: true id: 01c7663c51ca9c29913eec0e1c6ccd8c repositoryMetadata: base64Readme: >- IyBPcGVuIEFQSSBDb21wYXJhdG9yCgpBbiBPcGVuQVBJIHRvb2wgdG8gY29tcGFyZSBPcGVuQVBJIFNwZWNpZmljYXRpb25zLgoKIyMgQyMgTGlicmFyeQoKVGhlIHRvb2wgaXMgYXZhaWxhYmxlIGFzIGEgW251Z2V0IHBhY2thZ2VdKGh0dHBzOi8vd3d3Lm51Z2V0Lm9yZy9wYWNrYWdlcy9Dcml0ZW8uT3BlbkFwaS5Db21wYXJhdG9yKSwgZGlyZWN0bHkgdXNhYmxlIGludG8geW91ciBDIyBhcHBsaWNhdGlvbi4KClRvIGluc3RhbGwgaXQgcnVuIHRoZSBjb21tYW5kOgpgYGBiYXNoCmRvdG5ldCBhZGQgcGFja2FnZSBDcml0ZW8uT3BlbkFwaS5Db21wYXJhdG9yCmBgYAoKSGVyZSBpcyBhbiBleGFtcGxlIG9mIGhvdyB0byB1c2UgdGhlIENvbXBhcmF0b3I6CmBgYEMjCnZhciBkaWZmZXJlbmNlcyA9IE9wZW5BcGlDb21wYXJhdG9yLkNvbXBhcmUoCiAgICBvbGRPcGVuQXBpU3BlYywKICAgIG5ld09wZW5BcGlTcGVjCik7CmBgYAoKIyMgQ29tbWFuZCBsaW5lIHRvb2wKClRoZSBjb21wYXJhdG9yIGlzIGFsc28gYXZhaWxhYmxlIGFzIGEgW2NvbW1hbmQgbGluZSB0b29sXShodHRwczovL3d3dy5udWdldC5vcmcvcGFja2FnZXMvQ3JpdGVvLk9wZW5BcGkuQ29tcGFyYXRvci5DbGkvMC4xLjApLgoKVG8gaW5zdGFsbCBpdCwgcnVuIHRoZSBjb21tYW5kOgpgYGBiYXNoCmRvdG5ldCB0b29sIGluc3RhbGwgLWcgQ3JpdGVvLk9wZW5BcGkuQ29tcGFyYXRvci5DbGkKYGBgCgpZb3UgY2FuIHRoZW4gdXNlIHRoZSB0b29sIHRocm91Z2ggdGhlIGBvcGVuYXBpLWNvbXBhcmVgIGNvbW1hbmQ6CmBgYGJhc2gKb3BlbmFwaS1jb21wYXJlIC1vIG5ld19vYXMuanNvbiAtbiBvbGRfb2FzLmpzb24gLWYgSnNvbgpgYGAKCkF2YWlsYWJsZSBvcHRpb25zOgp8IE9wdGlvbiAgICAgICAgICAgfCBTbWFsbCAgIHwgUmVxdWlyZWQgfCBEZXNjcmlwdGlvbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfC0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS18LS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CnwgYC0tb2xkYCAgICAgICAgICB8IGAtb2AgICAgfCBgdHJ1ZWAgICB8IFBhdGggb3IgVVJMIHRvIG9sZCBPcGVuQVBJIFNwZWNpZmljYXRpb24uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IGAtLW5ld2AgICAgICAgICAgfCBgLW5gICAgIHwgYHRydWVgICAgfCBQYXRoIG9yIFVSTCB0byBuZXcgT3BlbkFQSSBTcGVjaWZpY2F0aW9uLiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgLS1vdXRwdXRGb3JtYXRgIHwgYC1mYCAgICB8IGBmYWxzZWAgIHwgKERlZmF1bHQ6IGBKc29uYCkgU3BlY2lmaWVzIGluIHdoaWNoIGZvcm1hdCB0aGUgZGlmZmVyZW5jZXMgc2hvdWxkIGJlIGRpc3BsYXllZC4gUG9zc2libGUgdmFsdWVzOiBgSnNvbmAgXHwgYFRleHRgLiB8CnwgYC0tc3RyaWN0YCAgICAgICB8IGAtc2AgICAgfCBgZmFsc2VgICB8IChEZWZhdWx0OiBgZmFsc2VgKSBFbmFibGUgc3RyaWN0IG1vZGU6IGJyZWFraW5nIGNoYW5nZXMgYXJlIGVycm9ycyBpbnN0ZWFkIG9mIHdhcm5pbmdzLiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IGAtLWhlbHBgICAgICAgICAgfCBgLWhgICAgIHwgYGZhbHNlYCAgfCBMb2cgYXZhaWxhYmxlIG9wdGlvbnMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKCiMjIENvbXBhcmlzb24gcnVsZXMKCkVhY2ggY29tcGFyaXNvbiBydWxlIGlzIGRvY3VtZW50ZWQgaW4gdGhlIFtkb2N1bWVudGF0aW9uIHNlY3Rpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9jcml0ZW8vb3BlbmFwaS1jb21wYXJhdG9yL3RyZWUvbWFpbi9kb2N1bWVudGF0aW9uKS4KCiMjIE9wZW5BUEkgdmVyc2lvbiBzdXBwb3J0CgpJbnRlcm5hbGx5LCB0aGUgY29tcGFyYXRvciB1c2VzIFttaWNyb3NvZnQvT3BlbkFQSS5ORVRdKGh0dHBzOi8vZ2l0aHViLmNvbS9taWNyb3NvZnQvT3BlbkFQSS5ORVQvKSB3aGljaCBjdXJyZW50bHkgc3VwcG9ydHMgT3BlbkFQSSAyLjAgdG8gMy4wLjAuCgojIyBDb250cmlidXRpbmcKCkFueSBjb250cmlidXRpb24gaXMgbW9yZSB0aGFuIHdlbGNvbWVkLiBGb3Igbm93LCBubyBzcGVjaWZpYyBydWxlIG11c3QgYmUgYXBwbGllZCB0byBjb250cmlidXRlLCBqdXN0IGNyZWF0ZSBhbiBJc3N1ZSBvciBhIFB1bGwgUmVxdWVzdCBhbmQgd2UnbGwgdHJ5IHRvIGhhbmRsZSBpdCBBU0FQLgoKIyMgTGljZW5zZQoKT3BlbkFwaSBDb21wYXJhdG9yIGlzIGFuIE9wZW4gU291cmNlIHNvZnR3YXJlIHJlbGVhc2VkIHVuZGVyIHRoZSBbQXBhY2hlIDIuMCBsaWNlbnNlXShodHRwczovL2dpdGh1Yi5jb20vY3JpdGVvL29wZW5hcGktY29tcGFyYXRvci9ibG9iL21haW4vTElDRU5DRSkuCgojIyBEZXZlbG9wZXIgZ3VpZGUKClNpbXBseSB1c2UgdGhlIGRvdG5ldCBjbGkuIEZvciBleGFtcGxlLCB0byBydW4gdGhlIHRlc3RzOgpgYGBiYXNoCmRvdG5ldCB0ZXN0CmBgYAo= readmeEtag: '"84b46d434f5d40f40d7d37729fcb5d742da5e9ec"' readmeLastModified: Thu, 20 Feb 2025 16:43:16 GMT repositoryId: 506186592 description: null created: '2022-06-22T09:46:19Z' updated: '2026-02-06T02:49:51Z' language: C# archived: false stars: 35 watchers: 8 forks: 12 owner: criteo logo: https://avatars.githubusercontent.com/u/1713646?v=4 license: Apache-2.0 repoEtag: '"e9d161ecd235a74308cc87418df9a2d11079e15086a7e5dfa7e520ccf5bf8b4a"' repoLastModified: Fri, 06 Feb 2026 02:49:51 GMT foundInMaster: true category: Parsers - source: https://openapi.tools/ name: openapi-validator-junit category: - Description Validators - Parsers link: https://github.com/stefankoppier/openapi-validator repository: https://github.com/stefankoppier/openapi-validator language: Kotlin source_description: >- A JUnit extension for validating a wide range of properties of a specification. v2: true v3: true v3_1: true id: 2d3fe34fda3d9de22ff8948b50e4617d repositoryMetadata: base64Readme: >- WyFbR3JhZGxlIFBsdWdpbiBQb3J0YWxdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ3JhZGxlLXBsdWdpbi1wb3J0YWwvdi9pby5naXRodWIuc3RlZmFua29wcGllci5vcGVuYXBpLnZhbGlkYXRvcildKGh0dHBzOi8vcGx1Z2lucy5ncmFkbGUub3JnL3BsdWdpbi9pby5naXRodWIuc3RlZmFua29wcGllci5vcGVuYXBpLnZhbGlkYXRvcikKWyFbTWF2ZW4gQ2VudHJhbF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3YvaW8uZ2l0aHViLnN0ZWZhbmtvcHBpZXIub3BlbmFwaS52YWxpZGF0b3IvY29yZSldKGh0dHBzOi8vbXZucmVwb3NpdG9yeS5jb20vYXJ0aWZhY3QvaW8uZ2l0aHViLnN0ZWZhbmtvcHBpZXIub3BlbmFwaS52YWxpZGF0b3IpClshW1F1YWxpdHkgR2F0ZSBTdGF0dXNdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PXN0ZWZhbmtvcHBpZXJfb3BlbmFwaS12YWxpZGF0b3ImbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPXN0ZWZhbmtvcHBpZXJfb3BlbmFwaS12YWxpZGF0b3IpClshW0NvdmVyYWdlXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1zdGVmYW5rb3BwaWVyX29wZW5hcGktdmFsaWRhdG9yJm1ldHJpYz1jb3ZlcmFnZSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPXN0ZWZhbmtvcHBpZXJfb3BlbmFwaS12YWxpZGF0b3IpCgojIE9wZW5BUEkgVmFsaWRhdG9yCgpUaGUgT3BlbkFQSSBWYWxpZGF0b3IgYWxsb3dzIHlvdSB0byB2YWxpZGF0ZSB5b3VyIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBieSB3cml0aW5nIGEgZGVzY3JpcHRpdmUgcnVsZXNldCBpbiBLb3RsaW4hClRoZXJlIGlzIGFuIGV4dGVuc2l2ZSBzZXQgb2YgYnVpbHQtaW4gcnVsZXMgYW5kIGlzIGV4dGVuc2libGUsIGFzIGl0IGFsbG93cyB5b3UgdG8gYWRkIGN1c3RvbSBydWxlcyBlYXNpbHkuIAoKVmlzaXQgW3RoZSBwcm9qZWN0IGRvY3VtZW50YXRpb25dKGh0dHBzOi8vc3RlZmFua29wcGllci5naXRodWIuaW8vb3BlbmFwaS12YWxpZGF0b3IvKSBmb3IgYSBjb21wbGV0ZSBvdmVydmlldyBvZiB0aGUgCmZ1bmN0aW9uYWxpdHkuCgojIyBGZWF0dXJlcwoKIyMjIFJ1bGVzCkFsbCBwcm9wZXJ0aWVzIG9mIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gaGF2ZSBhIGNvcnJlc3BvbmRpbmcgcnVsZS4gRm9yIGV4YW1wbGUsIGlmIHlvdSB3YW50IHRvIHZhbGlkYXRlIHRoYXQgdGhlCnBhdGggYC9wZXQvZmluZEJ5U3RhdHVzYCBoYXMgYSBnZXQgb3BlcmF0aW9uIHdpdGggdGhlIHJlcXVpcmVkIHF1ZXJ5IHBhcmFtZXRlciBgc3RhdHVzYCB5b3UgY2FuIHdyaXRlCmBgYGtvdGxpbgpvcGVuQVBJIHsKICAgIHBhdGhzIHsKICAgICAgICByZXF1aXJlZCgpCiAgICAgICAgcGF0aChuYW1lZCA9ICIvcGV0L2ZpbmRCeVN0YXR1cyIpIHsKICAgICAgICAgICAgZ2V0IHsgCiAgICAgICAgICAgICAgICBwYXJhbWV0ZXJzIHsKICAgICAgICAgICAgICAgICAgICBwYXJhbWV0ZXIobmFtZWQgPSAic3RhdHVzIikgewogICAgICAgICAgICAgICAgICAgICAgcmVxdWlyZWQoKQogICAgICAgICAgICAgICAgICAgICAgYGluYCB7IGV4YWN0bHkoInF1ZXJ5IikgfQogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KfQpgYGAKCiMjIyBRdWFudGlmaWVyIFJ1bGVzClRoZXJlIGlzIHN1cHBvcnQgZm9yIHF1bnRpZmljYXRpb24gb24gdGhlIHJlbGV2YW50IHR5cGVzLiBGb3IgZXhhbXBsZQpgYGBrb3RsaW4Kb3BlbkFQSSB7CiAgICBwYXRocyB7CiAgICAgICAgYWxsIHsKICAgICAgICAgICAgc3RhcnRzV2l0aCgicHJlZml4IikKICAgICAgICB9CiAgICB9Cn0KYGBgCnN0YXRlcyB0aGF0IGFsbCBwYXRocyBtdXN0IHN0YXJ0IHdpdGggdGhlIHN0cmluZyBgcHJlZml4YC4KCiMjIyBQcmVjb25kaXRpb25zClRoZXJlIGlzIHN1cHBvcnQgZm9yIHByZWNvbmRpdGlvbnMuIFRoZSBnZW5lcmFsIHByZWNvbmRpdGlvbiBgZ2l2ZW5gIHZhbGlkYXRlcyBhIGdpdmVuIHJ1bGUgb25seSBpZgp0aGUgcHJlZGljYXRlIGlzIHRydWUuIEZvciBleGFtcGxlCmBgYGtvdGxpbgpvcGVuQVBJIHsKICAgIGluZm8gewogICAgICAgIGdpdmVuKHsgaXQgIT0gbnVsbCAmJiBpdC5sZW5ndGggPiAyMCB9KSB7CiAgICAgICAgICAgIGRlc2NyaXB0aW9uIHsgbG93ZXJjYXNlKCkgfQogICAgICAgIH0KICAgIH0KfQpgYGAKc3RhdGVzIHRoYXQgdGhlIGZpZWxkIGBkZXNjcmlwdGlvbmAgb2YgdGhlIGBpbmZvYCBmaWVsZCBtdXN0IGxvd2VyY2FzZSB3aGVuIGl0IGlzIG5vdCBudWxsLCBhbmQgaXQncyBsZW5ndGggaXMKZ3JlYXRlciB0aGFuIDIwLgoKVGhlcmUgaXMgYWxzbyB0aGUgYHNpbmNlYCBwcmVjb25kaXRpb24sIHdoaWNoIHN0YXRlcyB0aGF0IHNvbWV0aGluZyBzaG91bGQgaG9sZCBvbmx5IGlmIHRoZSBnaXZlbiBkYXRlIGlzIGluCnRoZSBwYXN0LiBGb3IgZXhhbXBsZQpgYGBrb3RsaW4Kb3BlbkFQSSB7CiAgICBpbmZvIHsKICAgICAgICBzaW5jZShMb2NhbERhdGUub2YoMjAyNCwgMSwgMSkpIHsKICAgICAgICAgICAgdGl0bGUgeyBsb3dlcmNhc2UoKSB9CiAgICAgICAgfQogICAgfQp9CmBgYApzdGF0ZXMgdGhhdCB0aGUgZmllbGQgYHRpdGxlYCBvZiB0aGUgYGluZm9gIGZpZWxkIG11c3QgYmUgaW4gbG93ZXJjYXNlIGFmdGVyIEphbnVhcnkgMcui4bWXIG9mIDIwMjQuICAKCiMjIFVzYWdlCgpUaGUgT3BlbkFQSSBWYWxpZGF0b3IgYWxsb3dzIGZvciB0d28gd2F5cyB0byBleGVjdXRlIHRoZSB2YWxpZGF0aW9uOiB2aWEgR3JhZGxlIG9yIHZpYSBKVW5pdC4KCiMjIyBHcmFkbGUgSW50ZWdyYXRpb24KCkFwcGx5IHRoZSBwbHVnaW4KYGBga290bGluCnBsdWdpbnMgewogICAgaWQoImlvLmdpdGh1Yi5zdGVmYW5rb3BwaWVyLm9wZW5hcGkudmFsaWRhdG9yIikgdmVyc2lvbiAieC55LnoiCn0KYGBgCmFuZCB0aGVuIGNvbmZpZ3VyZSB0aGUgZG9jdW1lbnQgdG8gYmUgdmVyaWZpZWQgdXNpbmcgdGhlIGdpdmVuIHJ1bGVzIHVzaW5nCmBgYGtvdGxpbgpvcGVuQVBJVmFsaWRhdGUgewogICAgZG9jdW1lbnQuc2V0KHVyaSgicGV0c3RvcmUueWFtbCIpKQogICAgcnVsZXMuc2V0KAogICAgICAgIG9wZW5BUEkgewogICAgICAgICAgICBpbmZvIHsKICAgICAgICAgICAgICAgIHRpdGxlIHsgZXhhY3RseSgiT3BlbkFQSSBQZXRzdG9yZSIpIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgICkKfQpgYGAKCiMjIyBKVW5pdCBJbnRlZ3JhdGlvbgoKQWRkIHRoZSBwYWNrYWdlIHRvIHlvdXIgZGVwZW5kZW5jaWVzLgoKVGhlbiBleHRlbmQgeW91ciB0ZXN0IGNsYXNzIHdpdGggYEBFeHRlbmRXaXRoKE9wZW5BUElWYWxpZGF0aW9uRXh0ZW5zaW9uOjpjbGFzcylgIHRvIGFkZCBsb2FkaW5nIG9mIGEgc3BlY2lmaWNhdGlvbi4gClRoZSBzcGVjaWZpY2F0aW9uIGNhbiBiZSBsb2FkZWQgdXNpbmcgdGhlIGFubm90YXRpb24gYEBPcGVuQVBJVGVzdChyZWxhdGl2ZVVybCA9ICJzcmMvdGVzdC9yZXNvdXJjZXMvcGV0c3RvcmUueWFtbCIpYCB0byAKc3BlY2lmeSB0aGUgbG9jYXRpb24gb2YgdGhlIHNwZWNpZmljYXRpb24uIEZvciBleGFtcGxlCmBgYGtvdGxpbgpARXh0ZW5kV2l0aChPcGVuQVBJVmFsaWRhdGlvbkV4dGVuc2lvbjo6Y2xhc3MpCkBPcGVuQVBJVGVzdChyZWxhdGl2ZVVybCA9ICJzcmMvdGVzdC9yZXNvdXJjZXMvcGV0c3RvcmUueWFtbCIpCmNsYXNzIE9wZW5BUElWYWxpZGF0aW9uRXh0ZW5zaW9uVGVzdCB7CgogICAgQFRlc3QKICAgIGZ1biBgbXkgcGV0c3RvcmUgdGVzdGAoKSB7CiAgICAgICAgYXNzZXJ0RG9jdW1lbnRJc1ZhbGlkRm9yIHsKICAgICAgICAgICAgb3BlbkFQSSgiTXkgc3BlY2lmaWNhdGlvbiIpIHsKICAgICAgICAgICAgICAgIGluZm8gewogICAgICAgICAgICAgICAgICAgIHRpdGxlIHsgZXhhY3RseSgiT3BlbkFQSSBQZXRzdG9yZSIpIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KfQpgYGAK readmeEtag: '"418b10cd8b745efa68624d45799c241e01f3a0f0"' readmeLastModified: Wed, 01 Nov 2023 06:30:02 GMT repositoryId: 646046176 description: Library for validating OpenAPI specifications created: '2023-05-27T05:41:25Z' updated: '2024-08-01T11:03:01Z' language: Kotlin archived: false stars: 2 watchers: 1 forks: 0 owner: stefankoppier logo: https://avatars.githubusercontent.com/u/5744697?v=4 license: Apache-2.0 repoEtag: '"6e46560abb476ec775203698e454e4ae07dca69d3e33330e045b30e4d1126db6"' repoLastModified: Thu, 01 Aug 2024 11:03:01 GMT foundInMaster: true - source: https://openapi.tools/ name: Hot Potato category: - Data Validators - Description Validators - Server - Testing language: C# link: https://github.com/davidmbillie/Hot-Potato repository: https://github.com/davidmbillie/hot-potato source_description: >- A reverse proxy that validates responses against an OpenAPI description. It can be used as either a standalone .NET tool, or installed as NuGet packages for an ASP.NET Core integration testing project. id: bdfcf4d5e833ed87ca9e7a80a42cd36d repositoryMetadata: base64Readme: >- # Hot Potato Proxy

![Build Status](https://github.com/HylandSoftware/Hot-Potato/workflows/hot-potato-ci/badge.svg)

The Hot Potato is an ASP.NET Core reverse proxy that will validate an API's conformance to an OpenAPI spec.

## Setup

To use the complete tool you will need to download the `HotPotato.AspNetCore.Host` NuGet package from https://www.nuget.org/packages/HotPotato.AspNetCore.Host/. Since Hot Potato is a dotnet global tool you can easily download it from Powershell or Command Prompt. The most common way of utilizing this tool can be found in the [Postman](#postman) section below.

If you have an automated API testing project with a mock server you would like to conformance test, you also have the option of installing the `HotPotato.AspNetCore.Middleware` NuGet package from https://www.nuget.org/packages/HotPotato.AspNetCore.Middleware/. More information about writing conformance tests with a mock server can be found below in the [Middleware](#middleware) section.

### Install .NET Tool

To install Hot Potato use the following command:
```sh
dotnet tool install -g hotpotato.aspnetcore.host
```
There are other options that can be utilized when downloading a dotnet tool. A complete list of options can be found here: https://docs.microsoft.com/en-us/dotnet/core/tools/dotnet-tool-install

If the install is successful you will see a message like this:  
```sh
You can invoke the tool using the following command: HotPotato
Tool 'hotpotato.aspnetcore.host' (version '2.0.0') was successfully installed.
```
### Start Hot Potato

You can now start the tool by using the command `HotPotato`. Add the arguments for your testing situation and you can utilize `HotPotato` from the command line.
```sh
HotPotato --RemoteEndpoint http://example.com/my/endpoint --SpecLocation http://example.com/my/specification.yaml
```

### SSL Validation Issues

We have also also provided an environment variable named `HttpClientSettings__IgnoreClientHttpsCertificateValidationErrors` that can be set to `true` in the case of of persistent SSL certificate validation issues. 

### Specs with Token Validation

In the case of something like accessing a raw file in a private repo on Github, a token is needed to access a specification. For cases like this, we have included a `SpecToken` environment variable that can be used as a secret. If you're running Hot Potato locally via Visual Studio, an easy way of setting the secret can be found here: [Manage User Secrets](https://docs.microsoft.com/en-us/aspnet/core/security/app-secrets?view=aspnetcore-5.0&tabs=windows#json-structure-flattening-in-visual-studio).

<a name="results"></a>
## Results

In order to retrieve results from the proxy, we have exposed a `/results` endpoint. This endpoint will return a JSON-formatted object that shows all of the requests that have come through and whether or not they're conformant. The response will also include a `X-Status` header which will be either `Pass`, `Fail`, or `Inconclusive`.

__Pass Result__
```json
[
    {
        "Path":"/endpoint",
        "Method":"GET",
        "StatusCode":200,
        "State":"Pass"
    }
]
```

__Fail Result__
```json
[
    {        
        "Path":"/endpoint",
        "Method":"GET",
        "StatusCode":404,
        "State":"Fail",
        "Reasons":["InvalidBody"],
        "ValidationErrors":
        [
            {
                "Message":"Error",
                "Kind":"Unknown",
                "Property":"Property",
                "LineNumber":5,
                "LinePosition":10
            }
        ]
    }
]
```

### Custom Result Headers

Hot Potato also allows users to add custom headers to results objects. To do so, users can can add the prefix "X-HP-" to a header key in a request, and it will appear at the top of the result in a "custom" array. The custom array only appears if custom headers are provided.

```json
[
	{
        "custom": {
            "X-HP-Name": [
                "LandingPage"
            ]
        },
        "state": "Pass",
        "path": "/",
        "method": "get",
        "statusCode": 200
    }
]
```

## Structure

The proxy is broken down into a number of components to allow flexibility for developers.

### HotPotato.AspNetCore.Host

This is an ASP.NET Core host configured to use the Hot Potato Middleware. It is stood up as a separate server that listens by default on port `3232`. There is an `appsettings.json` to allow the developer to set the remote endpoint to forward requests to and the location of the OpenAPI specification to validate conformance. These values can also be passed into the command line via the following command:

`hotpotato --RemoteEndpoint http://example.com/my/endpoint --SpecLocation http://example.com/my/specification.yaml`

### HotPotato.Core

`HotPotato.Core` contains all of the models, HTTP logic, the proxy forwarder, and the interface for the proxy processor. This library is a dependency of `HotPotato.AspNetCore.Middleware` but will likely never need to be directly consumed.

### HotPotato.OpenApi

`HotPotato.OpenApi` contains all of the OpenAPI functionality including the processor implementation that validates an HTTP request/response pair against a spec, a result collection, logic to find paths, and so on. Currently, we are using the packages `NSwag`, `NJsonSchema`, and `NJsonSchema.Yaml` to consume a spec and to validate the contained schemas.

<a name="middleware"></a>
### HotPotato.AspNetCore.Middleware

This is an ASP.NET Core middleware that can be used in situations where test suites are directly starting up the server startup or using [`TestServer`](https://docs.microsoft.com/en-us/dotnet/api/microsoft.aspnetcore.testhost.testserver).

The main factor in setting up a test suite to use the isolated middleware is that because Hot Potato's default behavior is to create its own client through DI setup, an instance of the API Under Test's client must be created first to be able to be injected into DI.

#### Using the Middleware with TestServer

In our example [test fixture](https://github.com/HylandSoftware/Hot-Potato/blob/master/test/HotPotato.TestServer.Test/TestFixture.cs), we use two instances of TestServer: one to create the client representing the API Under Test, and one that consumes this client to create a new client housing the Hot Potato proxy.

First, we define a TestFixture class to use a generic Startup reference type:
 ```csharp
public class TestFixture<TStartup> : IDisposable where TStartup : class
 ```

Next, TestServer doesn't actually listen on an address, but it needs a placeholder BaseAddress to be used by the HttpRequest constructors. In our example, we use localhost:5000 for the TestServer housing the API and localhost:3232 for the TestServer housing the Hot Potato proxy to be consistent with the rest of our testing. These can be set to anything as long as they are valid URIs.

```
private const string ApiServerAddress = "http://localhost:5000";
private const string HotPotatoAddress = "http://localhost:3232";
```

Then we create a TestServer instance using the Startup type to create a client:
```csharp
var apiBuilder = new WebHostBuilder()
    .UseStartup<TStartup>();

apiServer = new TestServer(apiBuilder);
apiServer.BaseAddress = new Uri(TestServerAddress);

HotPotatoClient apiClient = new HotPotatoClient(apiServer.CreateClient());
```

Now that we have a client created for the API Under Test, we can build our web host and inject it into the Test Server.
This is done in our fixture setup, but can also be placed in a custom Startup class:

```csharp
var hotPotatoBuilder = new WebHostBuilder()
    //Setting this here instead of in appsettings.json so it always matches the BaseAddress on TestServer
    .UseSetting("RemoteEndpoint", TestServerAddress)
    .ConfigureAppConfiguration((hostingContext, config) =>
    {
        config.SetBasePath(hostingContext.HostingEnvironment.ContentRootPath)
            .AddJsonFile("appsettings.json", optional: true);
    })
    .ConfigureServices(services =>
    {
        services.ConfigureMiddlewareServices(apiClient);
    })
    .Configure(builder =>
    {
        builder.UseMiddleware<HotPotatoMiddleware>();
    });

    hotPotatoServer = new TestServer(hotPotatoBuilder);
    hotPotatoServer.BaseAddress = new Uri(HotPotatoAddress);
```

An important part of this builder is the line:
```csharp
services.ConfigureMiddlewareServices(apiClient);
```
`ConfigureMiddlewareServices` is an extension method found in the AspNetCore.Middleware project that adds the client to DI as well as all the services necessary to use the Middleware.

These services are `IProxy`, `ISpecificationProivder`, `IResultCollector`, and `IProcessor`.

We set the fixture's public members of Results to the List<Result> member of the IResultCollector and Client to the client created with the Hot Potato TestServer:
```csharp
Results = hotPotatoServer.Host.Services.GetService<IResultCollector>().Results;
Client = new HotPotatoClient(hotPotatoServer.CreateClient());
```

Then we use them like so in a test to send requests and verify the validation results:
```csharp
await client.SendAsync(req);

Result result = results.ElementAt(0);

Assert.Equal(State.Pass, result.State);
```

Make sure to call `results.Clear()` in a `Dispose()` method in XUnit or a `[Teardown]` method in NUnit. Another option is to call `results.Clear` in the `finally` block of a try-finally statement containing the test fixture. 

The full example test can be found at [RawPotatoTest.cs](https://github.com/HylandSoftware/Hot-Potato/blob/master/test/HotPotato.TestServer.Test/RawPotatoTest.cs).

#### Using Middleware/TestServer with Startups in separate test projects

As mentioned above, the fixture setup can be done in a Startup class, and to avoid re-writing the same code in multiple test projects, the Startup class from one test project can be referenced by another. When choosing this route, the test Startup class will need to be registered as a MVC Application Part with the line `services.AddMvc().AddApplicationPart(typeof(Startup).Assembly);`.

```csharp
public override void ConfigureServices(IServiceCollection services)
{
    startup.ConfigureServices(services);

    services.AddMvc().AddApplicationPart(typeof(Startup).Assembly);

    //custom code here
}
```

<a name="postman"></a>
## Testing with Postman

End-to-End tests using Hot Potato can be run with Postman both locally and through a pipeline also using Newman.

To use Postman locally, you must have instances of both the Hot Potato server and your API server running.

For our test project, we provided our own sample Hot Potato API, which can be found [here](https://github.com/HylandSoftware/Hot-Potato/tree/master/test/HotPotato.Api).

Once your System Under Test is ready, you may start writing Postman requests with the base address of localhost:3232.
To check the results of these requests, you can query the results endpoint as shown in the [Results](#results) section above.

You may also create these requests as part of a collection, which will allow for the creation of test sets, and the ability for them to be exported and run by a pipeline.

If you are not familiar with creating collections and writing tests in Postman, more information can be found in the links below:

[Collections](https://learning.getpostman.com/docs/postman/collections/creating-collections/)

[Tests](https://learning.getpostman.com/docs/postman/scripts/test_scripts/)

Tests will usually check for critical information such as if the correct status code and body are being returned correctly in the response.

Examples can be found in our HappyPath collection [here](https://github.com/HylandSoftware/Hot-Potato/blob/master/test/HappyPathTests.postman_collection.json).

**Check that the response contains the correct status code and expected body**
```javascript
pm.test(\"LandingPage returns 200 OK\", function () {
	pm.response.to.have.status(200)
})

pm.test(\"LandingPage returns expected body \", function () {
	pm.response.to.have.body(\"https://github.com/HylandSoftware/Hot-Potato\")
})
```

At the end of each collection, a GET request can be sent to the /results endpoint so that the list of results can be tested.
We split our test suite into three different collections of HappyPath, Non-Conformant, and NotInSpec tests, so that we could easily check if each collection only contained either all 'Pass' or all 'Fail' results.

**Check that the HappyPath results do not contain fail results**
```javascript
pm.test(\"Results should not contain Fail\", function () {
	pm.expect(pm.response.text()).to.not.include(\"Fail\")
})
```

Make sure to send a DELETE request at the end of your collection so that the results from collection do not carry over to the next.
 readmeEtag: '"b6e9c3c9fc8432ca35d8f53c2db9cfcf8ff39fa2"' readmeLastModified: Fri, 15 Sep 2023 01:15:45 GMT repositoryId: 623048998 description: >- ASP.NET Core proxy that will validate an API's conformance to an OpenAPI spec. created: '2023-04-03T15:34:48Z' updated: '2025-11-09T08:05:40Z' language: C# archived: false stars: 6 watchers: 1 forks: 1 owner: davidmbillie logo: https://avatars.githubusercontent.com/u/13511813?v=4 license: MIT-0 repoEtag: '"e026b9988d49f45bb8bef94e106dadb18c5f34565920c269323e20b60144b40a"' repoLastModified: Sun, 09 Nov 2025 08:05:40 GMT foundInMaster: true - source: https://openapi.tools/ name: Specmatic category: - Mock - Testing language: - Any - CLI repository: https://github.com/specmatic/specmatic link: https://specmatic.in/ source_description: >- Specmatic enables teams to practice Contract-Driven Development by leveraging OpenAPI specifications as Executable Contracts to reduce time-to-market by allowing teams to build, test and deploy their MicroServices and MicroFrontends independently and Shift-Left the identification of compatibility issues between Components / Services in complex interdependent systems such as those built with MicroServices architecture. In the process it reduces / removes the need for Integration Testing. v3: true id: 8b74578a6a8f45ec1dd9606eac23cc1a repositoryMetadata: base64Readme: >- U3BlY21hdGljCj09PT09PT09PQpbIVtNYXZlbiBDZW50cmFsXShodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9pby5zcGVjbWF0aWMvc3BlY21hdGljLWNvcmUuc3ZnKV0oaHR0cHM6Ly9tdm5yZXBvc2l0b3J5LmNvbS9hcnRpZmFjdC9pby5zcGVjbWF0aWMvc3BlY21hdGljLWNvcmUpIFshW0dpdEh1YiByZWxlYXNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3JlbGVhc2Uvc3BlY21hdGljL3NwZWNtYXRpYy5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vc3BlY21hdGljL3NwZWNtYXRpYy9yZWxlYXNlcykgIVtDSSBCdWlsZF0oaHR0cHM6Ly9naXRodWIuY29tL3NwZWNtYXRpYy9zcGVjbWF0aWMvd29ya2Zsb3dzL0NJJTIwQnVpbGQvYmFkZ2Uuc3ZnKSBbIVtUd2l0dGVyIEZvbGxvd10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90d2l0dGVyL2ZvbGxvdy9zcGVjbWF0aWMuc3ZnP3N0eWxlPXNvY2lhbCZsYWJlbD1Gb2xsb3cpXShodHRwczovL3R3aXR0ZXIuY29tL3NwZWNtYXRpYykgWyFbRG9ja2VyIFB1bGxzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2RvY2tlci9wdWxscy9zcGVjbWF0aWMvc3BlY21hdGljLnN2ZyldKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9zcGVjbWF0aWMvc3BlY21hdGljKQoKIyMjIyMgU2hpcCBBSS1SZWFkeSBBUElzIDEweCBGYXN0ZXIgd2l0aCBaZXJvIEludGVncmF0aW9uIEhlYWRhY2hlcwpFbGltaW5hdGUgQVBJIGludGVncmF0aW9uIGhlYWRhY2hlcyB3aXRoIFNwZWNtYXRpYydzIG5vLWNvZGUgQUktcG93ZXJlZCBBUEkgZGV2ZWxvcG1lbnQgc3VpdGUuIFRlYW1zIHNoaXAgQVBJcyAxMHggZmFzdGVyIGJ5IHRyYW5zZm9ybWluZyBzcGVjaWZpY2F0aW9ucyBpbnRvIGV4ZWN1dGFibGUgY29udHJhY3RzIGluc3RhbnRseeKAlG5vIGNvZGluZyByZXF1aXJlZCwgbm8gaW50ZWdyYXRpb24gc3VycHJpc2VzLgoKIyMjIENvbnRleHQKCkluIGEgY29tcGxleCwgaW50ZXJkZXBlbmRlbnQgZWNvc3lzdGVtLCB3aGVyZSBlYWNoIHNlcnZpY2UgaXMgZXZvbHZpbmcgcmFwaWRseSwgd2Ugd2FudCB0byBtYWtlIHRoZSBkZXBlbmRlbmNpZXMgYmV0d2VlbiB0aGVtIGV4cGxpY2l0IGluIHRoZSBmb3JtIG9mIGV4ZWN1dGFibGUgY29udHJhY3RzLiBbQ29udHJhY3QgRHJpdmVuIERldmVsb3BtZW50XShodHRwczovL3NwZWNtYXRpYy5pby9jb250cmFjdF9kcml2ZW5fZGV2ZWxvcG1lbnQuaHRtbCkgbGV2ZXJhZ2VzIEFQSSBzcGVjaWZpY2F0aW9ucyBsaWtlIFtPcGVuQVBJXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnLyNvcGVuYXBpLXNwZWNpZmljYXRpb24pLCBbQXN5bmNBUEldKGh0dHBzOi8vd3d3LmFzeW5jYXBpLmNvbS8pLCBbR3JhcGhRTF0oaHR0cHM6Ly9ncmFwaHFsLm9yZy8pIFNETCBmaWxlcywgW2dSUENdKGh0dHBzOi8vZ3JwYy5pby8pIFByb3RvIGZpbGVzLCBldGMuIGFzIGV4ZWN1dGFibGUgY29udHJhY3RzIGFsbG93aW5nIHRlYW1zIHRvIGdldCBpbnN0YW50YW5lb3VzIGZlZWRiYWNrIHdoaWxlIG1ha2luZyBjaGFuZ2VzIHRvIGF2b2lkIGFjY2lkZW50YWwgYnJlYWthZ2UuCgpXaXRoIHRoaXMgYWJpbGl0eSwgd2UgY2FuIG5vdyBpbmRlcGVuZGVudGx5IGRlcGxveSwgYXQgd2lsbCwgYW55IHNlcnZpY2UgYXQgYW55IHRpbWUgd2l0aG91dCBoYXZpbmcgdG8gZGVwZW5kIG9uIGV4cGVuc2l2ZSBhbmQgZnJhZ2lsZSBpbnRlZ3JhdGlvbiB0ZXN0cy4KCkxlYXJuIG1vcmUgYXQgW3NwZWNtYXRpYy5pb10oaHR0cHM6Ly9zcGVjbWF0aWMuaW8vI2ZlYXR1cmVzKSDwn4yQCgpbR2V0IHN0YXJ0ZWQgbm93XShodHRwczovL3NwZWNtYXRpYy5pby9nZXR0aW5nX3N0YXJ0ZWQuaHRtbCkg8J+agAoKWyFbU3BlY21hdGljIC0gQ29udHJhY3QgRHJpdmVuIERldmVsb3BtZW50IC0gWW91VHViZSBwbGF5bGlzdF0oaHR0cHM6Ly9pbWcueW91dHViZS5jb20vdmkvSzVCWXhvT05nWG8vMC5qcGcpXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PUs1Qll4b09OZ1hvJmxpc3Q9UEw5Wi1KZ2lUc09ZUkVSY3N5OW8zeTZuc2k1eUszSUJfdykKCltZb3VUdWJlIHBsYXlsaXN0XShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PUs1Qll4b09OZ1hvJmxpc3Q9UEw5Wi1KZ2lUc09ZUkVSY3N5OW8zeTZuc2k1eUszSUJfdykg8J+Tugo= readmeEtag: '"e77d844b6a4cbce952f6e840284e016189dbf7b6"' readmeLastModified: Wed, 03 Sep 2025 11:34:09 GMT repositoryId: 247710440 description: >- Eliminate API integration headaches with Specmatic's no-code AI-powered API development suite. Teams ship APIs 10x faster by transforming specifications into executable contracts instantly—no coding required, no integration surprises. created: '2020-03-16T13:33:07Z' updated: '2026-02-05T05:38:36Z' language: Kotlin archived: false stars: 354 watchers: 10 forks: 60 owner: specmatic logo: https://avatars.githubusercontent.com/u/80689899?v=4 license: MIT repoEtag: '"1f72b1d2b4f721ad303d9d1c55792c320125a13e94ac40b2797e113ed61d9236"' repoLastModified: Thu, 05 Feb 2026 05:38:36 GMT foundInMaster: true oldLocations: - https://github.com/znsio/specmatic - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/pb33f/openapi-changes v3: true id: fc8c62141b92d4a5935d734bb4b51767 repositoryMetadata: base64Readme: >- CiFbbG9nb10ob3BlbmFwaS1jaGFuZ2VzLWxvZ28ud2VicCkKClshW2Rpc2NvcmRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZGlzY29yZC85MjMyNTgzNjM1NDA4MTU5MTIpXShodHRwczovL2Rpc2NvcmQuZ2cveDdWQUNWdUVHUCkKWyFbR2l0SHViIGRvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvZG93bmxvYWRzL3BiMzNmL29wZW5hcGktY2hhbmdlcy90b3RhbD9sYWJlbD1naXRodWIlMjBkb3dubG9hZHMmc3R5bGU9ZmxhdC1zcXVhcmUpXShodHRwczovL2dpdGh1Yi5jb20vcGIzM2Yvd2lyZXRhcC9yZWxlYXNlcykKWyFbbnBtXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9kbS9AcGIzM2Yvb3BlbmFwaS1jaGFuZ2VzP3N0eWxlPWZsYXQtc3F1YXJlJmxhYmVsPW5wbSUyMGRvd25sb2FkcyldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL0BwYjMzZi9vcGVuYXBpLWNoYW5nZXMpClshW0RvY2tlciBQdWxsc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kb2NrZXIvcHVsbHMvcGIzM2Yvb3BlbmFwaS1jaGFuZ2VzP3N0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL3BiMzNmL29wZW5hcGktY2hhbmdlcykKCiMgT3BlbkFQSSBDaGFuZ2VzCgojIyBUaGUgd29ybGQncyAqKl9zZXhpZXN0XyoqIE9wZW5BUEkgZGlmZiB0b29sLgoKV2Ugd2lsbCBnZXQgdG8gdGhlIHNleHkgcGFydCBpbiBhIG1vbWVudCwgYnV0IGluIGEgbnV0c2hlbGwgYG9wZW5hcGktY2hhbmdlc2AgYWxsb3dzIHlvdSAKdG8gc2VlIGFuZCBleHBsb3JlIHdoYXQgaGFzIGNoYW5nZWQgd2l0aCB5b3VyIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiwgYmV0d2VlbiBhIHNpbmdsZSBjaGFuZ2UsIApvciBmb3IgYWxsIHRpbWUhIAoKRXhwbG9yZSBPcGVuQVBJIGNoYW5nZSBoaXN0b3J5IHVzaW5nIGEgdGVybWluYWwgb3IgYSBicm93c2VyLCB5b3UgZGVjaWRlIHdoaWNoIGV4cGVyaWVuY2Ugc3VpdHMgeW91LiBQZXJmZWN0IGZvcgppbmRpdmlkdWFsIHVzZSwgb3IgaW4gYSBDSS9DRCBwaXBlbGluZSBmb3IgYXV0b21hdGlvbi4KCj4gVGhpcyBpcyBhbiBlYXJseSB0b29sIGFuZCBpcyBhY3RpdmUsIGRhaWx5IGRldmVsb3BtZW50CgojIyBIb3cgaXMgaXQgdGhlICdzZXhpZXN0Jz8KCkhhdmUgeW91IGV2ZXIgc2VlbiBhbiBPcGVuQVBJIGRpZmYgcmVwb3J0IGxvb2sgbGlrZSB0aGlzPwoKIVtdKGh0dHBzOi8vZ2l0aHViLmNvbS9wYjMzZi9vcGVuYXBpLWNoYW5nZXMvYmxvYi9tYWluLy5naXRodWIvYXNzZXRzL3VpLmdpZikKCldoYXQgYWJvdXQgYSB0ZXJtaW5hbCBVSSB0aGF0IGRvZXMgdGhlIHNhbWU/CgohW10oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL29wZW5hcGktY2hhbmdlcy9ibG9iL21haW4vLmdpdGh1Yi9hc3NldHMvY29uc29sZS5naWYpCgotLS0KCgojIyBEb2N1bWVudGF0aW9uCgojIyMg8J+RiSBb4pqhIFRyeSB0aGUgb25saW5lIERlbW8g4pqhXShodHRwczovL3BiMzNmLmlvL29wZW5hcGktY2hhbmdlcy9kZW1vLykg8J+RiAoKIyMjIFtRdWljayBTdGFydCBHdWlkZSDwn5qAXShodHRwczovL3BiMzNmLmlvL29wZW5hcGktY2hhbmdlcy9xdWlja3N0YXJ0LykKClNlZSBhbGwgdGhlIGRvY3VtZW50YXRpb24gYXQgaHR0cHM6Ly9wYjMzZi5pby9vcGVuYXBpLWNoYW5nZXMvCgotIFtJbnN0YWxsaW5nIG9wZW5hcGktY2hhbmdlc10oaHR0cHM6Ly9wYjMzZi5pby9vcGVuYXBpLWNoYW5nZXMvaW5zdGFsbGluZy8pCi0gW0NvbmZpZ3VyaW5nIGJyZWFraW5nIGNoYW5nZXNdKGh0dHBzOi8vcGIzM2YuaW8vb3BlbmFwaS1jaGFuZ2VzL2NvbmZpZ3VyaW5nLykKLSBbQ29tbWFuZCBhcmd1bWVudHNdKGh0dHBzOi8vcGIzM2YuaW8vb3BlbmFwaS1jaGFuZ2VzL2NvbW1hbmQtYXJndW1lbnRzLykKLSBDTEkgQ29tbWFuZHMKICAgIC0gW2Bjb25zb2xlYCBjb21tYW5kXShodHRwczovL3BiMzNmLmlvL29wZW5hcGktY2hhbmdlcy9jb25zb2xlLykKICAgIC0gW2BodG1sLXJlcG9ydGAgY29tbWFuZF0oaHR0cHM6Ly9wYjMzZi5pby9vcGVuYXBpLWNoYW5nZXMvaHRtbC1yZXBvcnQvKQogICAgLSBbYHJlcG9ydGAgY29tbWFuZF0oaHR0cHM6Ly9wYjMzZi5pby9vcGVuYXBpLWNoYW5nZXMvcmVwb3J0LykKICAgIC0gW2BzdW1tYXJ5YCBjb21tYW5kXShodHRwczovL3BiMzNmLmlvL29wZW5hcGktY2hhbmdlcy9zdW1tYXJ5LykKLSBbQWJvdXQgb3BlbmFwaS1jaGFuZ2VzXShodHRwczovL3BiMzNmLmlvL29wZW5hcGktY2hhbmdlcy9hYm91dC8pCgotLS0KIApBcmUgeW91IHJlYWR5IHRvIHRyeSBpdCBvdXQ/CgojIyBJbnN0YWxsIHVzaW5nIGhvbWVicmV3IHRhcAoKYGBgYmFzaApicmV3IGluc3RhbGwgcGIzM2YvdGFwcy9vcGVuYXBpLWNoYW5nZXMKYGBgCgotLS0KCiMjIEluc3RhbGwgdXNpbmcgbnBtIG9yIHlhcm4KCmBgYGJhc2gKbnBtIGkgLWcgQHBiMzNmL29wZW5hcGktY2hhbmdlcwpgYGAKCklmIHlvdSBwcmVmZXIgeWFybiBfKHJlY29tbWVuZGVkKV8KCmBgYGJhc2gKeWFybiBnbG9iYWwgYWRkIEBwYjMzZi9vcGVuYXBpLWNoYW5nZXMKYGBgCgotLS0KCiMjIEluc3RhbGwgdXNpbmcgY1VSTAoKYGBgYmFzaApjdXJsIC1mc1NMIGh0dHBzOi8vcGIzM2YuaW8vb3BlbmFwaS1jaGFuZ2VzL2luc3RhbGwuc2ggfCBzaCAKYGBgCi0tLQoKIyMgSW5zdGFsbC9ydW4gdXNpbmcgRG9ja2VyCgpgYGBiYXNoCmRvY2tlciBwdWxsIHBiMzNmL29wZW5hcGktY2hhbmdlcwpgYGAKCkRvY2tlciBpbWFnZXMgYXJlIGF2YWlsYWJsZSBmb3IgYm90aCBgbGludXgvYW1kNjRgIGFuZCBgbGludXgvYXJtNjRgIGFyY2hpdGVjdHVyZXMuCgpUbyBydW4sIG1vdW50IHRoZSBjdXJyZW50IHdvcmtpbmcgZGlyIHRvIHRoZSBjb250YWluZXIgbGlrZSBzbzoKCmBgYApkb2NrZXIgcnVuIC0tcm0gLXYgJFBXRDovd29yazpydyBwYjMzZi9vcGVuYXBpLWNoYW5nZXMgc3VtbWFyeSAuIHNhbXBsZS1zcGVjcy9wZXRzdG9yZXYzLmpzb24KYGBgCj4gVGhlIGBjb25zb2xlYCBjYW5ub3QgcnVuIHZpYSBkb2NrZXIuCgotLS0KCiMjIEN1c3RvbSBCcmVha2luZyBSdWxlcyBDb25maWd1cmF0aW9uCgo+IFN1cHBvcnRlZCBpbiBgdjAuOTErYAoKb3BlbmFwaS1jaGFuZ2VzIHVzZXMgW2xpYm9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9wYjMzZi9saWJvcGVuYXBpKSdzIGNvbmZpZ3VyYWJsZSBicmVha2luZyBjaGFuZ2UKZGV0ZWN0aW9uIHN5c3RlbS4gWW91IGNhbiBjdXN0b21pemUgd2hpY2ggY2hhbmdlcyBhcmUgY29uc2lkZXJlZCAiYnJlYWtpbmciIGJ5IHByb3ZpZGluZyBhIGNvbmZpZ3VyYXRpb24gZmlsZS4KCiMjIyBVc2luZyBhIENvbmZpZyBGaWxlCgpgYGBiYXNoCiMgVXNlIGV4cGxpY2l0IGNvbmZpZyBmaWxlCm9wZW5hcGktY2hhbmdlcyBzdW1tYXJ5IC1jIG15LXJ1bGVzLnlhbWwgb2xkLnlhbWwgbmV3LnlhbWwKCiMgT3IgcGxhY2UgY2hhbmdlcy1ydWxlcy55YW1sIGluIGN1cnJlbnQgZGlyZWN0b3J5IChhdXRvLWRldGVjdGVkKQpvcGVuYXBpLWNoYW5nZXMgc3VtbWFyeSBvbGQueWFtbCBuZXcueWFtbApgYGAKCiMjIyBEZWZhdWx0IENvbmZpZyBMb2NhdGlvbnMKCm9wZW5hcGktY2hhbmdlcyBzZWFyY2hlcyBmb3IgYGNoYW5nZXMtcnVsZXMueWFtbGAgaW46CjEuIEN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkgKGAuL2NoYW5nZXMtcnVsZXMueWFtbGApCjIuIFVzZXIgY29uZmlnIGRpcmVjdG9yeSAoYH4vLmNvbmZpZy9jaGFuZ2VzLXJ1bGVzLnlhbWxgKQoKIyMjIEV4YW1wbGUgQ29uZmlndXJhdGlvbgoKQ3JlYXRlIGEgYGNoYW5nZXMtcnVsZXMueWFtbGAgZmlsZToKCmBgYHlhbWwKIyBDdXN0b20gYnJlYWtpbmcgcnVsZXMgY29uZmlndXJhdGlvbgojIE9ubHkgc3BlY2lmeSBvdmVycmlkZXMgLSB1bnNwZWNpZmllZCBydWxlcyB1c2UgZGVmYXVsdHMKCiMgTWFrZSBvcGVyYXRpb24gcmVtb3ZhbCBub24tYnJlYWtpbmcgKGZvciBkZXByZWNhdGlvbiB3b3JrZmxvd3MpCnBhdGhJdGVtOgogIGdldDoKICAgIHJlbW92ZWQ6IGZhbHNlCiAgcG9zdDoKICAgIHJlbW92ZWQ6IGZhbHNlCiAgcHV0OgogICAgcmVtb3ZlZDogZmFsc2UKICBkZWxldGU6CiAgICByZW1vdmVkOiBmYWxzZQoKIyBNYWtlIGVudW0gdmFsdWUgcmVtb3ZhbCBub24tYnJlYWtpbmcKc2NoZW1hOgogIGVudW06CiAgICByZW1vdmVkOiBmYWxzZQoKIyBNYWtlIHBhcmFtZXRlciBjaGFuZ2VzIG5vbi1icmVha2luZwpwYXJhbWV0ZXI6CiAgcmVxdWlyZWQ6CiAgICBtb2RpZmllZDogZmFsc2UKYGBgCgojIyMgQ29uZmlndXJhdGlvbiBTdHJ1Y3R1cmUKCkVhY2ggcnVsZSBoYXMgdGhyZWUgb3B0aW9uczoKLSBgYWRkZWRgOiBJcyBhZGRpbmcgdGhpcyBwcm9wZXJ0eSBhIGJyZWFraW5nIGNoYW5nZT8gKHRydWUvZmFsc2UpCi0gYG1vZGlmaWVkYDogSXMgbW9kaWZ5aW5nIHRoaXMgcHJvcGVydHkgYSBicmVha2luZyBjaGFuZ2U/ICh0cnVlL2ZhbHNlKQotIGByZW1vdmVkYDogSXMgcmVtb3ZpbmcgdGhpcyBwcm9wZXJ0eSBhIGJyZWFraW5nIGNoYW5nZT8gKHRydWUvZmFsc2UpCgojIyMgQXZhaWxhYmxlIENvbXBvbmVudHMKCllvdSBjYW4gY29uZmlndXJlIHJ1bGVzIGZvciB0aGVzZSBPcGVuQVBJIGNvbXBvbmVudHM6Cgp8IENvbXBvbmVudCAgICAgICAgICAgICB8IERlc2NyaXB0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CnwgYHBhdGhzYCAgICAgICAgICAgICAgIHwgUGF0aCBkZWZpbml0aW9ucyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IGBwYXRoSXRlbWAgICAgICAgICAgICB8IE9wZXJhdGlvbnMgKGdldCwgcG9zdCwgcHV0LCBkZWxldGUsIGV0Yy4pICAgICAgICAgIHwKfCBgb3BlcmF0aW9uYCAgICAgICAgICAgfCBPcGVyYXRpb24gZGV0YWlscyAob3BlcmF0aW9uSWQsIHJlcXVlc3RCb2R5LCBldGMuKSB8CnwgYHBhcmFtZXRlcmAgICAgICAgICAgIHwgUGFyYW1ldGVyIHByb3BlcnRpZXMgKG5hbWUsIHJlcXVpcmVkLCBzY2hlbWEpICAgICAgfAp8IGBzY2hlbWFgICAgICAgICAgICAgICB8IFNjaGVtYSBwcm9wZXJ0aWVzICh0eXBlLCBmb3JtYXQsIGVudW0sIHByb3BlcnRpZXMpIHwKfCBgcmVzcG9uc2VgICAgICAgICAgICAgfCBSZXNwb25zZSBkZWZpbml0aW9ucyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgYHNlY3VyaXR5U2NoZW1lYCAgICAgIHwgU2VjdXJpdHkgc2NoZW1lIHByb3BlcnRpZXMgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IGBzZWN1cml0eVJlcXVpcmVtZW50YCB8IFNlY3VyaXR5IHJlcXVpcmVtZW50cyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKCkZvciB0aGUgY29tcGxldGUgbGlzdCBvZiBjb25maWd1cmFibGUgcHJvcGVydGllcyBhbmQgbW9yZSBleGFtcGxlcywgc2VlIHRoZQpbZnVsbCBjb25maWd1cmF0aW9uIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vcGIzM2YuaW8vb3BlbmFwaS1jaGFuZ2VzL2NvbmZpZ3VyaW5nLykuCgotLS0KCkNoZWNrIG91dCBhbGwgdGhlIGRvY3MgYXQgaHR0cHM6Ly9wYjMzZi5pby9vcGVuYXBpLWNoYW5nZXMvCg== readmeEtag: '"b36e97124926c37ce46b0755e31b2562975be300"' readmeLastModified: Wed, 17 Dec 2025 19:05:44 GMT repositoryId: 571681922 description: >- The world's sexiest OpenAPI breaking changes detector. Discover what changed between two OpenAPI specs, or a single spec over time. Supports OpenAPI 3.1, 3.0 and Swagger created: '2022-11-28T16:53:14Z' updated: '2026-02-01T08:45:55Z' language: Go archived: false stars: 323 watchers: 2 forks: 29 owner: pb33f logo: https://avatars.githubusercontent.com/u/104016643?v=4 license: NOASSERTION repoEtag: '"e57aaac704d91090e81b912238fac7272ab84f76de0ae61dcaddfb48fa1e029e"' repoLastModified: Sun, 01 Feb 2026 08:45:55 GMT category: - Data Validators - Testing - Documentation foundInMaster: true name: openapi-changes link: https://pb33f.io/openapi-changes language: go source_description: >- The world's fastest, most beautiful, powerful and detailed OpenAPI change detection tool on earth. Explore every change in An OpenAPI spec. See changes over time, go back into history and see what changed with each commit. Identifies all breaking changes, Powered by libopenapi. v2: true v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/tuwilof/fitting v3: true id: f0e56305d3796a01c1c2b5a5166bc11b repositoryMetadata: base64Readme: >- # Fitting

<img align="right" width="192" height="192"
alt="Fitting avatar: Documents with hangers"
src="./images/logo.png">



Library add improve test log, validate its according to your API documentation, show the documentation coverage with log.

Test log setting supports RSpec test and WebMock stubbing for Ruby On Rails application, API documentation supports API Blueprint and OpenAPI.

This reduces the costs of support, testers and analysts.

[Telegram community for any Fitting related questions](https://t.me/+NFojMTCFJ-1iNDYy)

Log
```text
FITTING incoming request {"method":"POST","path":"/public/api/v1/inboxes/tEX5JiZyceiwuKMi1oN9Sf8S/contacts","body":{},"response":{"status":200,"content_type":"application/json","body":{"source_id":"00dbf18d-879e-47cb-ac45-e9aece266eb1","pubsub_token":"ktn6YwPus57JDf4e59eFPom5","id":3291,"name":"shy-surf-401","email":null,"phone_number":null}},"title":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb:9","group":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb","host":"www.example.com"}
FITTING outgoing request {"method":"POST","path":"/v1/organizations/org_id/meeting","body":{},"response":{"status":200,"content_type":"application/json","body":{"success":true,"data":{"meeting":{"id":"meeting_id","roomName":"room_name"}}}},"title":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb:50","group":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb","host":"api.cluster.dyte.in"}
```

validation
```console
FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.FFF..FFFFFFFFFF....F.......F...FF.....F...F....F..............................FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF.F..FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF........FFF...FFFF......FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF........FFFFFFFFFFF..FFFFFF..FFFFFFFFFFFFFFFFF.......FFFFFF.............FFFFFFFFFFFF....F........FFF.F...FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF............FF........FFF......FFFFFFFFFFFFFFFFFFFFFF....FFFFFF......F............FFFF........FFFFFFFFFFFFFF.....FFFFFFFFFFFFFFFFFFFFFFF..FF.....FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.....FF..........FFFFFFFFFFFFFFFFFF...FFFF...............F.F....FF..FFFFFFFF

  1) Fitting::Doc::NotFound log error:

host: www.example.com
method: POST
path: /public/api/v1/inboxes/{inbox_identifier}/contacts
code: 200

content-type: application/json

json-schema: {
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "properties": {
    "id": {
      "type": "integer",
      "description": "Id of the contact"
    },
    "source_id": {
      "type": "string",
      "description": "The session identifier of the contact"
    },
    "name": {
      "type": "string",
      "description": "Name of the contact"
    },
    "email": {
      "type": "string",
      "description": "Email of the contact"
    },
    "pubsub_token": {
      "type": "string",
      "description": "The token to be used to connect to chatwoot websocket"
    }
  }
}

body: {
  "source_id": "c9e8c31f-06df-49b4-8fb9-4466457ae65b",
  "pubsub_token": "Zgc7DEvaj5TkgZ1a4C7AvJXo",
  "id": 3293,
  "name": "restless-snowflake-670",
  "email": null,
  "phone_number": null
}

error [
  "The property '#/email' of type null did not match the following type: string in schema e56b7e65-d70c-5f7a-a96c-982df5f8f2f7"
]

...

804 examples, 565 failure, 0 pending

Coverage: 65.51%
```

and cover
![exmaple](images/b1.png)

![exmaple](images/b2.png)

![exmaple](images/w1.png)

![exmaple](images/w2.png)

## Installation
Add this line to your application's Gemfile:
```ruby
gem 'fitting'
```

After that execute:
```bash
$ bundle
```

Or install the gem by yourself:
```bash
$ gem install fitting
```

## Usage
### Log
Firstly, improve `test.log`.

To your `spec_helper.rb`:

```ruby
require 'fitting'

Fitting.logger
```

Delete all files `log/*.log` and run rspec

You get more information about incoming and outgoing request in `log/fitting*.log`.

```text
FITTING incoming request {"method":"POST","path":"/public/api/v1/inboxes/tEX5JiZyceiwuKMi1oN9Sf8S/contacts","body":{},"response":{"status":200,"content_type":"application/json","body":{"source_id":"00dbf18d-879e-47cb-ac45-e9aece266eb1","pubsub_token":"ktn6YwPus57JDf4e59eFPom5","id":3291,"name":"shy-surf-401","email":null,"phone_number":null}},"title":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb:9","group":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb","host":"www.example.com"}
FITTING outgoing request {"method":"POST","path":"/v1/organizations/org_id/meeting","body":{},"response":{"status":200,"content_type":"application/json","body":{"success":true,"data":{"meeting":{"id":"meeting_id","roomName":"room_name"}}}},"title":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb:50","group":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb","host":"api.cluster.dyte.in"}
```

### Validation
Secondly, validate the log to the documentation.

Add this to your `.fitting.yml`:

```yaml
APIs:
  - host: www.example.com
    type: openapi2
    path: swagger/swagger.json
```

Run 
```bash
bundle e rake fitting:validate
```

Console output

```console
FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.FFF..FFFFFFFFFF....F.......F...FF.....F...F....F..............................FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF.F..FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF........FFF...FFFF......FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF........FFFFFFFFFFF..FFFFFF..FFFFFFFFFFFFFFFFF.......FFFFFF.............FFFFFFFFFFFF....F........FFF.F...FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF............FF........FFF......FFFFFFFFFFFFFFFFFFFFFF....FFFFFF......F............FFFF........FFFFFFFFFFFFFF.....FFFFFFFFFFFFFFFFFFFFFFF..FF.....FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.....FF..........FFFFFFFFFFFFFFFFFF...FFFF...............F.F....FF..FFFFFFFF

  1) Fitting::Doc::NotFound log error:

host: www.example.com
method: POST
path: /public/api/v1/inboxes/{inbox_identifier}/contacts
code: 200

content-type: application/json

json-schema: {
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "properties": {
    "id": {
      "type": "integer",
      "description": "Id of the contact"
    },
    "source_id": {
      "type": "string",
      "description": "The session identifier of the contact"
    },
    "name": {
      "type": "string",
      "description": "Name of the contact"
    },
    "email": {
      "type": "string",
      "description": "Email of the contact"
    },
    "pubsub_token": {
      "type": "string",
      "description": "The token to be used to connect to chatwoot websocket"
    }
  }
}

body: {
  "source_id": "c9e8c31f-06df-49b4-8fb9-4466457ae65b",
  "pubsub_token": "Zgc7DEvaj5TkgZ1a4C7AvJXo",
  "id": 3293,
  "name": "restless-snowflake-670",
  "email": null,
  "phone_number": null
}

error [
  "The property '#/email' of type null did not match the following type: string in schema e56b7e65-d70c-5f7a-a96c-982df5f8f2f7"
]

...

804 examples, 565 failure, 0 pending

Coverage: 65.51%
```

### Coverage
And task will create HTML (`coverage/fitting.html`) reports.

![exmaple](images/b1.png)

![exmaple](images/b2.png)

More information on action coverage

![exmaple2](images/w1.png)

![exmaple2](images/w2.png)

## Settings

### APIs

#### type

##### OpenAPI 2.0
Swagger

```yaml
APIs:
  - host: www.example.com
    type: openapi2
    path: doc/api.json
```

##### OpenAPI 3.0
Also OpenAPI

```yaml
APIs:
  - host: www.example.com
    type: openapi3
    path: doc/api.json
```

##### API Blueprint
First you need to install [drafter](https://github.com/apiaryio/drafter) or [crafter](https://github.com/funbox/crafter).
Works after conversion from API Blueprint to API Elements (in YAML file) with Drafter or Crafter.

That is, I mean that you first need to do this

```bash
drafter doc.apib -o doc.yaml
```

or

```bash
node_modules/.bin/crafter doc.apib > doc.yaml
```

and then

```yaml
APIs:
  - host: www.example.com
    type: drafter
    path: doc/api.yaml
```

or

```yaml
APIs:
  - host: www.example.com
    type: crafter
    path: doc/api.yaml
```

##### Tomograph

To use additional features of the pre-converted [tomograph](https://github.com/tuwilof/tomograph)

example

```bash
bundle exec tomograph -d crafter --exclude-description doc/api.yml doc/api.json
```

and then

```yaml
APIs:
  - host: www.example.com
    type: tomogram
    path: doc/api.json
```

#### prefix

Setting the prefix name is optional. For example, you can do this:

```yaml
APIs:
  - host: www.example.com
    prefix: /api/v3
    type: openapi2
    path: swagger/swagger.json
```

### SkipValidation

#### host

It is not necessary to immediately describe each host in detail, you can only specify its name and skip it until you are ready to documented it

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
```

#### prefix

If you want to skip a specific prefix in the host

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
    prefix: /admin/api
```

#### method and path

If you want to skip a specific request in the host

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
    method: GET
    path: /api/v1/cars
```

### NoCov

It is not necessary to immediately test each doc in detail, you can only specify its name and skip it until you are ready to test it

#### host
```yaml
NoCov:
  - host: sso.test
```

#### method
```yaml
NoCov:
  - host: sso.test
    method: GET
```

#### path
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
```

#### code
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
```

#### content-type
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
```

#### combination
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
    combination: oneOf.0
```

#### combination_next
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
    combination: oneOf.0
    combination_next: oneOf.0.required.users
```

### Debug

If you find bug, you can debug it or create task in this github project  with new file `coverage/fitting.debug.yml`

```yaml
Debug:
  - host: www.example.com
    method: GET
    path: /api/v3/users
    code: 200
    content-type: application/json
```

## Community

Join us on [Telegram](https://t.me/+NFojMTCFJ-1iNDYy).

## Contributing

Bug reports and pull requests are welcome on GitHub at [github.com/tuwilof/fitting](https://github.com/tuwilof/fitting).
This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.

## License

The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).

[Sponsored by FunBox](https://funbox.ru)
 readmeEtag: '"39b0d151f49626ae338174765093384ce3b6f35c"' readmeLastModified: Tue, 09 Jul 2024 20:36:02 GMT repositoryId: 81070703 description: >- Library add improve test log for RSpec and WebMock, validate its according to API Blueprint and Open API, show the documentation coverage with log. created: '2017-02-06T09:32:39Z' updated: '2025-12-02T08:48:15Z' language: Ruby archived: false stars: 64 watchers: 6 forks: 9 owner: tuwilof logo: https://avatars.githubusercontent.com/u/5956924?v=4 license: MIT repoEtag: '"1d22576b181eb532a0b7893ea9d842877f176a8ebedeba05c7efce774adbbe21"' repoLastModified: Tue, 02 Dec 2025 08:48:15 GMT category: - Testing - Data Validators - Learning foundInMaster: true name: Fitting language: Ruby link: https://github.com/tuwilof/fitting source_description: >- Library add improve test log for RSpec and WebMock, validate its according to API Blueprint and Open API, show the documentation coverage with log. v2: true v3_1: true - source: https://openapi.tools/ name: KrakenD API Gateway category: - Documentation - Gateway link: https://www.krakend.io/enterprise/ language: Golang source_description: >- KrakenD is a high-performance, stateless API Gateway, seamlessly integrated with OpenAPI. It facilitates automatic API documentation and provides the capability to import existing specifications for generating configuration and mock servers. v2: true v3: true id: 1c349424400e28e0b3ed6ddc2a675c45 foundInMaster: true - source: https://openapi.tools/ name: Serverless API Gateway category: - Gateway - Server Implementations link: https://serverlessapigateway.com repository: https://github.com/irensaltali/serverlessapigateway language: TypeScript source_description: >- Serverless API Gateway is a serverless, open-source, and free API Gateway that is designed to be used with any HTTP source. v2: true v3: true v3_1: true id: 2cad52c6f066edce2de23d8b28481995 repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KCTxpbWcgIHNyYz0iZG9jcy9pbWcvaGVyby5qcGciPgogICAgPGgxPiAKICAgICAgICA8c3Ryb25nPlNlcnZlcmxlc3MgQVBJIEdhdGV3YXk8L3N0cm9uZz4KICAgIDwvaDE+CjwvZGl2PgoKWzxpbWcgc3JjPSJodHRwczovL2FwaS5naXRzcG9uc29ycy5jb20vYXBpL2JhZGdlL2ltZz9pZD03Mzc1MjYyNzQiIGhlaWdodD0iMjAiPl0oaHR0cHM6Ly9hcGkuZ2l0c3BvbnNvcnMuY29tL2FwaS9iYWRnZS9saW5rP3A9ekdGMG12QjRFVnV1TGNHMjBhSmNHY2JZWU10UjIwL1JHOC9uOFVxNkFxM2NQZ1VjRTVNK0JEU2Y5RzhseS9EREJQVmk4ZWNKM05UK0dJdWoyK2g4Ky90YTJOdGg0OVNKQW5FOTZzVFlzazcwQmhhZU1NTXBvSE51OFI5eWM4aG9kR3BFNW1sU0luUEMvdUVBSklkd0VRPT0pCgpbIVtKV1QgQ09NUEFUSUJMRV0oaHR0cHM6Ly9qd3QuaW8vaW1nL2JhZGdlLWNvbXBhdGlibGUuc3ZnKV0oKQoKV2VsY29tZSB0byB0aGUgU2VydmVybGVzcyBBUEkgR2F0ZXdheSwgYW4gaW5ub3ZhdGl2ZSB0b29sIGRlc2lnbmVkIHRvIHN0cmVhbWxpbmUgeW91ciBBUEkgbWFuYWdlbWVudCB0YXNrcyB1c2luZyB0aGUgcG93ZXJmdWwgY2FwYWJpbGl0aWVzIG9mIENsb3VkZmxhcmUgV29ya2Vycy4KCiMjIEZlYXR1cmVzCgotICoqSlMgV29ya2VycyoqOiBXcml0ZSBzZXJ2ZXJsZXNzIEphdmFTY3JpcHQgd29ya2VycyB0aGF0IGludGVyY2VwdCBhbmQgbW9kaWZ5IHlvdXIgQVBJIHJlcXVlc3RzIGFuZCByZXNwb25zZXMgb24gdGhlIGZseS4KLSAqKlJvdXRpbmcgKFBhdGggYW5kIE1ldGhvZCkqKjogU2ltcGxpZnkgeW91ciBBUEkgYXJjaGl0ZWN0dXJlIHdpdGggZmxleGlibGUgcGF0aCBhbmQgbWV0aG9kLWJhc2VkIHJvdXRpbmcgZm9yIGRpcmVjdGluZyB0cmFmZmljIHRvIHRoZSBhcHByb3ByaWF0ZSBlbmRwb2ludHMuCi0gKipDT1JTIChCYXNpYykqKjogTWFuYWdlIGNyb3NzLW9yaWdpbiByZXNvdXJjZSBzaGFyaW5nIHNldHRpbmdzIHdpdGggZWFzZSwgZW5zdXJpbmcgeW91ciBBUElzIGNhbiBzZWN1cmVseSBoYW5kbGUgcmVxdWVzdHMgZnJvbSBkaWZmZXJlbnQgb3JpZ2lucy4KLSAqKkF1dGggKEpXVCkqKjogU2VjdXJlIHlvdXIgQVBJcyBieSBpbXBsZW1lbnRpbmcgSlNPTiBXZWIgVG9rZW4gKEpXVCkgYmFzZWQgYXV0aGVudGljYXRpb24gdG8gdmFsaWRhdGUgYW5kIG1hbmFnZSB1c2VyIGFjY2VzcyBlZmZpY2llbnRseS4KLSAqKlNlcnZpY2UgQmluZGluZyoqOiBCaW5kIHlvdXIgQVBJIHRvIGEgc2VydmljZSBhcyBXb3JrZXJzLCBzbyB5b3UgY2FuIHVzZSB0aGUgV29ya2VycyBjYXBhYmlsaXRpZXMgd2l0aGluIHlvdXIgQVBJLgotICoqVmFsdWUgTWFwcGluZyoqOiBNYXAgdmFsdWVzIGZyb20gc291cmNlcyB0byBkZXN0aW5hdGlvbnMsIGFsbG93aW5nIHlvdSB0byBlYXNpbHkgdHJhbnNmb3JtIHlvdXIgZGF0YS4KCiMjIE1vdGl2YXRpb24KCkFQSXMgYXJlIHBpdm90YWwgaW4gdGhlIGxhbmRzY2FwZSBvZiBtb2Rlcm4gYXBwbGljYXRpb25zLCBidXQgdGhleSBicmluZyBmb3J0aCBhIHVuaXF1ZSBzZXQgb2YgY2hhbGxlbmdlcyByZWdhcmRpbmcgc2VjdXJpdHksIHJvdXRpbmcsIGFuZCBvdmVyYWxsIG1hbmFnZW1lbnQuIFRoZSBTZXJ2ZXJsZXNzIEFQSSBHYXRld2F5IGVtZXJnZWQgZnJvbSB0aGUgbmVlZCB0byBhZGRyZXNzIHRoZXNlIGlzc3VlcyBpbiBhIHJlbGlhYmxlLCBtYW5hZ2VhYmxlLCBhbmQgY29zdC1lZmZlY3RpdmUgd2F5LiBCdWlsdCB1cG9uIENsb3VkZmxhcmUncyBzZXJ2ZXJsZXNzIGluZnJhc3RydWN0dXJlLCB0aGlzIHByb2plY3QgcHJvdmlkZXMgZGV2ZWxvcGVycyB3aXRoIGEgbGlnaHR3ZWlnaHQgeWV0IHJvYnVzdCB0b29sa2l0IHRoYXQgYWRhcHRzIHRvIHRoZSB1bnByZWRpY3RhYmlsaXR5IG9mIGludGVybmV0IHNjYWxlIGFuZCB0cmFmZmljLiBPdXIgbWlzc2lvbiBpcyB0byBlbXBvd2VyIGRldmVsb3BlcnMgdG8gc2VjdXJlbHkgYW5kIGVmZmljaWVudGx5IG1hbmFnZSB0aGVpciBBUElzIHdpdGhvdXQgdGhlIG92ZXJoZWFkIG9mIG1hbmFnaW5nIGluZnJhc3RydWN0dXJlLgoKCkZvciBkZXRhaWxlZCBzZXR1cCBhbmQgdXNhZ2UgaW5zdHJ1Y3Rpb25zLCBwbGVhc2Ugc2VlIHRoZSBbU2VydmVybGVzcyBBUEkgR2F0ZXdheSBkb2N1bWVudGF0aW9uXShodHRwczovL2RvY3Muc2VydmVybGVzc2FwaWdhdGV3YXkuY29tKS4KCiMjIENvbnRyaWJ1dGluZwoKSWYgeW91IHdhbnQgdG8gY29udHJpYnV0ZSB0byB0aGUgcHJvamVjdCwgeW91IGNhbiBzdGFydCBieSBjaGVja2luZyB0aGUgW2NvbnRyaWJ1dGluZyBndWlkZWxpbmVzXShDT05UUklCVVRJTkcubWQpLgoKCiMjIEFja25vd2xlZGdtZW50cwoKQSBzaG91dG91dCB0byB0aGUgY29udHJpYnV0b3JzLCBjb21tdW5pdHkgbWVtYmVycywgYW5kIHRoZSBtYWludGFpbmVycyBvZiBDbG91ZGZsYXJlIFdvcmtlcnMgZm9yIHRoZWlyIHN1cHBvcnQgYW5kIGluc3BpcmF0aW9uIGluIG1ha2luZyB0aGlzIHByb2plY3QgYSByZWFsaXR5LgoKVGhlIFNlcnZlcmxlc3MgQVBJIEdhdGV3YXkgaXMgbm90IGp1c3QgYW5vdGhlciBBUEkgdG9vbDsgaXQncyBjcmVhdGVkIGJ5IGRldmVsb3BlcnMsIGZvciBkZXZlbG9wZXJzLCB3aXRoIHRoZSB2aXNpb24gb2YgbWFraW5nIEFQSSBtYW5hZ2VtZW50IGEgYnJlZXplLiBMZXQncyBidWlsZCB0b2dldGhlci4KCgojIyBTdXBwb3J0CgpJJ20gYWx3YXlzIGhhcHB5IHRvIGhlbHAgd2l0aCBhbnkgcXVlc3Rpb25zIG9yIGNvbmNlcm5zIHlvdSBtYXkgaGF2ZS4gRmVlbCBmcmVlIHRvIHJlYWNoIG91dCB0byBtZSBmcm9tIG9uIFtUd2l0dGVyXShodHRwczovL3R3aXR0ZXIuY29tL2lyZW5zYWx0YWxpKSBvciBbTGlua2VkSW5dKGh0dHBzOi8vd3d3LmxpbmtlZGluLmNvbS9pbi9pcmVuc2FsdGFsaS8pLgoKSWYgeW91IG5lZWQgYSBtb3JlIGV4dGVuc2l2ZSBzdXBwb3J0IHlvdSBjYW4gYWx3YXlzIGJvb2sgb24gW1N1cGVycGVlcl0oaHR0cHM6Ly9zdXBlcnBlZXIuY29tL2lyZW5zYWx0YWxpLy0vc2VydmVybGVzcy1hcGktZ2F0ZXdheSkKCiMgQ29tcGFuaWVzIHRoYXQgdXNlIFNlcnZlcmxlc3MgQVBJIEdhdGV3YXkKCjxkaXYgYWxpZ249ImNlbnRlciI+Cgk8YSBocmVmPSJodHRwczovL3dvcGUuY29tIj4gPGltZyB3aWR0aD0iMjAwIiBzcmM9ImRvY3MvaW1nL3dvcGUucG5nIj4gPC9hPgo8L2Rpdj4KCkxldCB1cyBrbm93IGlmIHlvdSBhcmUgdXNpbmcgU2VydmVybGVzcyBBUEkgR2F0ZXdheSBhbmQgd2UgY2FuIGFkZCB5b3VyIGNvbXBhbnkgaGVyZS4KCiMgQ29udHJpYnV0b3JzCgo8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vaXJlbnNhbHRhbGkvc2VydmVybGVzc2FwaWdhdGV3YXkvZ3JhcGhzL2NvbnRyaWJ1dG9ycyI+CiAgPGltZyBzcmM9Imh0dHBzOi8vY29udHJpYi5yb2Nrcy9pbWFnZT9yZXBvPWlyZW5zYWx0YWxpL3NlcnZlcmxlc3NhcGlnYXRld2F5IiAvPgo8L2E+CgoKIyBGZWViYWNrIFN1cnZleQoKUGxlYXNlIHRha2UgYSBmZXcgbWludXRlcyB0byBmaWxsIG91dCB0aGUgW2ZlZWRiYWNrIHN1cnZleV0oaHR0cHM6Ly9yMzlyYTU1YjBzbC50eXBlZm9ybS5jb20vdG8vZXg4SE15VEgpIHRvIGhlbHAgdXMgaW1wcm92ZSB0aGUgU2VydmVybGVzcyBBUEkgR2F0ZXdheS4K readmeEtag: '"02b85dddb7a806618da9c9c0adb921ec81bf65a7"' readmeLastModified: Sat, 18 Jan 2025 15:10:12 GMT repositoryId: 737526274 description: Serverless API Gateway created: '2023-12-31T11:37:10Z' updated: '2026-02-02T12:25:15Z' language: JavaScript archived: false stars: 70 watchers: 2 forks: 11 owner: irensaltali logo: https://avatars.githubusercontent.com/u/6968747?v=4 license: NOASSERTION repoEtag: '"35cc1066280a2a5e0e6ded7b5e9435d23ba525ddf90e4d2403ebb5b8823fc41f"' repoLastModified: Mon, 02 Feb 2026 12:25:15 GMT foundInMaster: true - source: https://openapi.tools/ name: Treblle category: - Documentation - Security - Monitoring link: https://www.treblle.com/ source_description: >- Treblle is a lightweight SDK that helps Engineering and Product teams build, ship, and maintain REST based APIs faster. v3: true v3_1: true id: cee984cf9c829d96ef1ec921ff5b8db2 foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags name: openapi-request-response-validation category: - Data Validators - Parsers repository: https://github.com/gcatanese/openapi-request-response-validation language: Java source_description: >- Runtime validation of requests and responses of your API according to the OpenAPI specs, returning (if any) the list of errors found. It is integrated via a simple REST API therefore usable by Java and non-Java applications and CI/CD. v2: true v3: true v3_1: true id: 9c4692fcf610859bb87b0e09cf848e83 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHZhbGlkYXRvcgoKVGhpcyB0b29sIGFsbG93cyB0aGUgdmFsaWRhdGlvbiBhdCBydW50aW1lIG9mIHRoZSBBUEkgcmVxdWVzdHMgcmVzcG9uc2VzIGFjY29yZGluZyB0byB0aGUgT3BlbkFQSSBzcGVjcy4gIAoKVGhlcmUgYXJlIHNldmVyYWwgdG9vbHMgdGhhdCBjYW4gdmFsaWRhdGUgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uLCBidXQgdGhlcmUgYXJlIG5vIG1hbnkgb3B0aW9ucyB0byBlbnN1cmUgdGhhdCB0aGUgQVBJIGNvbnRyYWN0cyBhcmUgaG9ub3VyZWQgYnkgdGhlIEFQSSB3ZSBhcmUgZGV2ZWxvcGluZy4KClRoaXMgdG9vbCBtYWtlIHN1cmUgdGhhdCB0aGUgQVBJIHJlcXVlc3RzIGFuZCByZXNwb25zZXMgYXJlIHZhbGlkIGFjY29yZGluZyB0byB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIG9mIHRoZSBBUEkuCgpNb3JlIG9uIFtWYWxpZGF0aW5nIEFQSSByZXF1ZXN0cyBhbmQgcmVzcG9uc2VzXShodHRwczovL2JlcHBlY2F0YW5lc2UuaGFzaG5vZGUuZGV2L3ZhbGlkYXRpbmctYXBpLXJlcXVlc3RzLWFuZC1yZXNwb25zZXMpCgpUaGUgYG9wZW5hcGktcmVxdWVzdC1yZXNwb25zZS12YWxpZGF0b3JgIGlzIGEgU3ByaW5nQm9vdCAoSmF2YSkgYXBwbGljYXRpb24gaW1wbGVtZW50aW5nIGEgUkVTVCBjb250cm9sbGVyIHRvIGFsbG93IFBvc3RtYW4gc2NyaXB0cyAob3Igb3RoZXIgY2xpZW50cykgdG8gc2VuZCB0aGUgcGF5bG9hZCB0byBiZSB2YWxpZGF0ZWQuIFRoZSBPcGVuQVBJIGZpbGUgY2FuIGJlIHN1cHBsaWVkIGF0IHN0YXJ0dXAuCgojIyBIb3cgZG9lcyBpdCB3b3JrPwoKWW91IHdvcmsgd2l0aCBQb3N0bWFuIHRvIHRlc3QgdGhlIEFQSSBlbmRwb2ludHMsIHNlbmRpbmcgYSByZXF1ZXN0IGFuZCB2ZXJpZnkgdGhlIHJlc3BvbnNlLiBUaGFua3MgdG8gUG9zdG1hbiBbVGVzdCBTY3JpcHRzXShodHRwczovL2xlYXJuaW5nLnBvc3RtYW4uY29tL2RvY3Mvd3JpdGluZy1zY3JpcHRzL3Rlc3Qtc2NyaXB0cy8pIGl0IGlzIHBvc3NpYmxlIHRvIGFkZCBjdXN0b20gc2NyaXB0cyB0byBhY2Nlc3MgdGhlICBgcmVxdWVzdGAsIGByZXNwb25zZWAgYW5kIGBoZWFkZXJzYCBwcm9ncmFtbWF0aWNhbGx5IGFuZCBzZW5kIHRoZW0gdG8gdGhlIE9wZW5BUEkgUmVxdWVzdC1SZXNwb25zZSBWYWxpZGF0b3IuClBvc3RtYW4gdGVzdHMgKHdpdGggYXNzZXJ0aW9ucykgY2FuIGJlIGRlZmluZWQgdG8gY29uZmlybSB0aGUgSlNPTiBwYXlsb2FkcyBhcmUgdmFsaWQgYWNjb3JkaW5nIHRvIHRoZSBBUEkgc3BlY2lmaWNhdGlvbi4KCiFbT3BlbkFQSSBWYWxpZGF0b3JdKGRvYy9vcGVuYXBpLXZhbGlkYXRvci5wbmcpCgpUaGUgb3V0Y29tZSBvZiB0aGUgdmFsaWRhdGlvbiAodG9nZXRoZXIgd2l0aCB0aGUgbGlzdCBvZiBlcnJvcnMg4oCUIGlmIGFueSkgaXMgcmV0dXJuZWQgdG8gUG9zdG1hbiAoZGlzcGxheWVkIGluIHRoZSBQb3N0bWFuIGNvbnNvbGUpIGFuZCBsb2dnZWQgYnkgdGhlIGFwcGxpY2F0aW9uLgoKCiMjIEhvdyB0byBydW4KClN0ZXBzOgoqIGFkZCB0aGUgc25pcHBldCBiZWxvdyBpbiB0aGUgQ29sbGVjdGlvbiBUZXN0cwoqIHByb3ZpZGUgdGhlIE9wZW5BUEkgZmlsZQoqIGxhdW5jaCB0aGUgYG9wZW5hcGktcmVxdWVzdC1yZXNwb25zZS12YWxpZGF0aW9uYCB0b29sIChbSmF2YSBhcHBdKCNzdGFydC10aGUtdG9vbC1qYXZhKSBvciB1c2luZyBbRG9ja2VyXSgjc3RhcnQtdGhlLXRvb2wtZG9ja2VyKSkgCiogcnVuIHRoZSBQb3N0bWFuIHJlcXVlc3RzIGFnYWluc3QgeW91ciBzZXJ2aWNlIG9yIGFwcGxpY2F0aW9uIAoKIyMjIENvbGxlY3Rpb24gVGVzdCBzbmlwcGV0CgpJbiB0aGUgKipDb2xsZWN0aW9uIFRlc3RzKiogYWRkIHRoZSBzbmlwcGV0IGJlbG93LiBJdCB3aWxsIHJ1biBhZnRlciBldmVyeSByZXF1ZXN0IGluIHRoZSBjb2xsZWN0aW9uLiAgCgpXaGF0IGRvZXMgaXQgZG8/IEFmdGVyIGV4ZWN1dGluZyB0aGUgcmVxdWVzdCB0aGUgVGVzdCBTY3JpcHQgd2lsbCBzZW5kIGByZXF1ZXN0YCwgYHJlc3BvbnNlYCBhbmQgYGhlYWRlcnNgIHRvIHRoZSB2YWxpZGF0b3IuCgpgYGAKb3BlbmFwaVJlcXVlc3RSZXNwb25zZVZhbGlkYXRpb24gPSB7CiAgICB2YWxpZGF0ZTogZnVuY3Rpb24ocG0pIHsKICAgIAogICAgICAgIC8vIGJ1aWxkIHBhdGggd2l0aG91dCBiYXNlVXJsCiAgICAgICAgdmFyIGJhc2VVcmwgPSBwbS5jb2xsZWN0aW9uVmFyaWFibGVzLmdldCgiYmFzZVVybCIpOwogICAgICAgIGJhc2VVcmwgPSBiYXNlVXJsLnJlcGxhY2UoJ2h0dHBzOi8vJywnJyk7CiAgICAgICAgYmFzZVVybCA9IGJhc2VVcmwucmVwbGFjZShwbS5yZXF1ZXN0LnVybC5nZXRIb3N0KCksJycpOwoKICAgICAgICB2YXIgcGF0aCA9IHBtLnJlcXVlc3QudXJsLmdldFBhdGgoKS5yZXBsYWNlKGJhc2VVcmwsJycpOwoKICAgICAgICBjb25zb2xlLmxvZygnVmFsaWRhdGlvbiBmb3IgJyArIHBhdGgpOwoKICAgICAgICBjb25zdCBwb3N0UmVxdWVzdCA9IHsKICAgICAgICAgICAgdXJsOiAnaHR0cDovL2xvY2FsaG9zdDo4MDgwL3ZhbGlkYXRlJywKICAgICAgICAgICAgbWV0aG9kOiAnUE9TVCcsCiAgICAgICAgICAgIGhlYWRlcjogeydDb250ZW50LVR5cGUnOiAnYXBwbGljYXRpb24vanNvbid9LAogICAgICAgICAgICBib2R5OiB7CiAgICAgICAgICAgIG1vZGU6ICdyYXcnLAogICAgICAgICAgICByYXc6IEpTT04uc3RyaW5naWZ5KHsgCiAgICAgICAgICAgICAgICBtZXRob2Q6IHBtLnJlcXVlc3QubWV0aG9kLCAKICAgICAgICAgICAgICAgIHBhdGg6IHBhdGgsCiAgICAgICAgICAgICAgICBoZWFkZXJzOiBwbS5yZXF1ZXN0LmhlYWRlcnMsCiAgICAgICAgICAgICAgICByZXF1ZXN0QXNKc29uOiAocG0ucmVxdWVzdC5ib2R5ICE9ICIiKSA/IHBtLnJlcXVlc3QuYm9keS5yYXcgOiBudWxsLAogICAgICAgICAgICAgICAgcmVzcG9uc2VBc0pzb246IHBtLnJlc3BvbnNlLnRleHQoKSwKICAgICAgICAgICAgICAgIHN0YXR1c0NvZGU6IHBtLnJlc3BvbnNlLmNvZGUKICAgICAgICAgICAgICAgIH0pCiAgICAgICAgICAgIH0KICAgICAgICB9OwoKICAgICAgICBwbS5zZW5kUmVxdWVzdChwb3N0UmVxdWVzdCwgKGVycm9yLCByZXNwb25zZSkgPT4gewogICAgICAgICAgICBpZihlcnJvciAhPSB1bmRlZmluZWQpIHsKICAgICAgICAgICAgICAgIHBtLmV4cGVjdC5mYWlsKCdVbmV4cGVjdGVkIGVycm9yICcgKyBlcnJvcik7CiAgICAgICAgICAgIH0gZWxzZSB7CiAgICAgICAgICAgICAgICB2YXIgZGF0YSA9IHJlc3BvbnNlLmpzb24oKTsKCiAgICAgICAgICAgICAgICBpZihkYXRhLnZhbGlkID09IGZhbHNlKSB7CiAgICAgICAgICAgICAgICAgICAgY29uc29sZS5sb2coZGF0YS5lcnJvcnMpOwogICAgICAgICAgICAgICAgfQoKICAgICAgICAgICAgICAgIHBtLnRlc3QoIk9wZW5BUEkgdmFsaWRhdGlvbiIsICgpID0+IHsKICAgICAgICAgICAgICAgICAgICBwbS5leHBlY3QoZGF0YS52YWxpZCwgIkludmFsaWQgcmVxdWVzdC9yZXNwb25zZSAoY2hlY2sgQ29uc29sZSkiKS50by5lcXVhbCh0cnVlKTsKICAgICAgICAgICAgICAgIH0pOwoKICAgICAgICAgICAgfQogICAgICAgIH0pOyAgCiAgICB9Cgp9OwoKb3BlbmFwaVJlcXVlc3RSZXNwb25zZVZhbGlkYXRpb24udmFsaWRhdGUocG0pOwpgYGAKCiMjIyBQcm92aWRlIHRoZSBPcGVuQVBJIHNwZWMgZmlsZQoKQ29weS9yZW5hbWUgeW91ciBPcGVuQVBJIHNwZWNzIHRvIGBvcGVuYXBpL29wZW5hcGkueWFtbGAgb3IgYG9wZW5hcGkvb3BlbmFwaS5qc29uYAoKIyMjIFN0YXJ0IHRoZSB0b29sIChKYXZhKQoKUnVuIHRoZSBKYXZhIGFwcGxpY2F0aW9uIApgYGBzaGVsbApqYXZhIC1qYXIgdGFyZ2V0L29wZW5hcGktcmVxdWVzdC1yZXNwb25zZS12YWxpZGF0b3IuamFyCmBgYAoKUnVuIHRoZSBKYXZhIGFwcGxpY2F0aW9uIHdpdGggY3VzdG9tIHBvcnQgYW5kIHNwZWMgZmlsZQpgYGBzaGVsbApqYXZhIC1qYXIgdGFyZ2V0L29wZW5hcGktcmVxdWVzdC1yZXNwb25zZS12YWxpZGF0b3IuamFyIC0tc2VydmVyLnBvcnQ9ODg4OCAtLUlOUFVUX1NQRUNTPS9wYXRoL3RvL215b3BlbmFwaS55YW1sCmBgYAoKIyMjIFN0YXJ0IHRoZSB0b29sIChEb2NrZXIpCgpZb3UgY2FuIHJ1biB0aGUgdG9vbCBvbiBEb2NrZXIKCmBgYAojIHJ1biB1c2luZyBkZWZhdWx0IG9wZW5hcGkvb3BlbmFwaS55YW1sIG9yIG9wZW5hcGkvb3BlbmFwaS5qc29uCmRvY2tlciBydW4gLXYgJChwd2QpOi9vcGVuYXBpIC1pdCAtLXJtIC0tbmFtZSBvcGVuYXBpLXJlcXVlc3QtcmVzcG9uc2UtdmFsaWRhdGlvbiBcCiBnY2F0YW5lc2Uvb3BlbmFwaS1yZXF1ZXN0LXJlc3BvbnNlLXZhbGlkYXRpb24KCiMgcnVuIHVzaW5nIGN1c3RvbSBsb2NhdGlvbiBvZiB0aGUgT3BlbkFQSSBmaWxlCmRvY2tlciBydW4gLXYgJChwd2QpOi9vcGVuYXBpIC1lIElOUFVUX1NQRUNTPS90bXAvb3BlbmFwaS55YW1sIFwKICAtaXQgLS1ybSAtLW5hbWUgb3BlbmFwaS1yZXF1ZXN0LXJlc3BvbnNlLXZhbGlkYXRpb24gXAogICAgZ2NhdGFuZXNlL29wZW5hcGktcmVxdWVzdC1yZXNwb25zZS12YWxpZGF0aW9uCmBgYAoKIyMjIFJ1biBQb3N0bWFuIHJlcXVlc3RzCgpSdW4gdGhlIFBvc3RtYW4gcmVxdWVzdHMgYW5kIGNoZWNrIHRoZSBUZXN0IHRhYgoKIVtQb3N0bWFuIFRlc3QgUmVzdWx0c10oZG9jL3Bvc3RtYW4tdGVzdC1yZXN1bHRzLnBuZykKCgoKLS0tClVzaW5nIFtBdGxhc3NpYW4gU3dhZ2dlciBWYWxpZGF0b3JdKGh0dHBzOi8vYml0YnVja2V0Lm9yZy9hdGxhc3NpYW4vc3dhZ2dlci1yZXF1ZXN0LXZhbGlkYXRvci8pLCBbUG9zdG1hbl0oaHR0cHM6Ly9wb3N0bWFuLmNvbSkgCmFuZCBbRG9ja2VyXShodHRwczovL2RvY2tlci5jb20pCgo= readmeEtag: '"4282cd24432d1a248aeb1c3855592bf8ff3360cb"' readmeLastModified: Wed, 04 Dec 2024 19:53:44 GMT repositoryId: 584163624 description: Validation of request/response according to OpenAPI specs created: '2023-01-01T16:44:30Z' updated: '2025-08-29T12:16:18Z' language: Java archived: false stars: 15 watchers: 1 forks: 2 owner: gcatanese logo: https://avatars.githubusercontent.com/u/1771700?v=4 license: Apache-2.0 repoEtag: '"4e5316c503bc6030926c2aefd366cd6418b41aec026fb969146b561d963286a3"' repoLastModified: Fri, 29 Aug 2025 12:16:18 GMT foundInMaster: true - source: https://openapi.tools/ name: api-smart-diff repository: https://github.com/udamir/api-smart-diff language: TypeScript source_description: >- Compare two Json based API documents (OpenAPI, AsyncAPI, JsonSchema, GraphAPI) v2: true v3: true v3_1: true id: d4a9b510fa698579d4641e0190f532af repositoryMetadata: base64Readme: >- # api-smart-diff
<img alt="npm" src="https://img.shields.io/npm/v/api-smart-diff"> <img alt="npm" src="https://img.shields.io/npm/dm/api-smart-diff?label=npm"> <img alt="npm type definitions" src="https://img.shields.io/npm/types/api-smart-diff"> <img alt="GitHub" src="https://img.shields.io/github/license/udamir/api-smart-diff">

This package provides utils to compute the diff between two Json based API documents - [online demo](https://udamir.github.io/api-smart-diff/)

## Purpose
- Generate API changelog
- Identify breaking changes
- Ensure API versioning consistency

## Supported API specifications

- [OpenApi 3.0](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md)
- [AsyncApi 2.x](https://v2.asyncapi.com/docs/reference)
- [JsonSchema](https://json-schema.org/draft/2020-12/json-schema-core.html)
- GraphQL via [GraphApi](https://github.com/udamir/graphapi)
- ~~[Swagger 2.0](https://github.com/swagger-api/swagger-spec/blob/master/versions/2.0.md)~~
- ~~[AsyncApi 3.x](https://www.asyncapi.com/docs/specifications/)~~ (Roadmap)
- ~~gRPC~~ (Roadmap)

## Features
- Generate diff for supported specifications
- Generate merged document with changes in metadata 
- Classify all changes as breaking, non-breaking, deprecated and annotation
- Human-readable change description
- Supports custom classification rules
- Supports custom comparison or match rules
- Supports custom transformations
- Supports custom human-readable changes annotation
- Resolves all $ref pointers, including circular
- Typescript syntax support out of the box
- Can be used in nodejs or browser

## External $ref
If schema contains an external $ref, you should bundle it via [api-ref-bundler](https://github.com/udamir/api-ref-bundler) first.

## Installation
```SH
npm install api-smart-diff --save
```
or
```SH
yarn add api-smart-diff
```

## Usage

### Nodejs
```ts
import { apiCompare } from 'api-smart-diff'

const { diffs, merged } = apiCompare(before, after)
// diff: 
// {
//   action: "add" | "remove" | "replace" | "rename",
//   after: 'value in after',
//   before: 'value in before',
//   description: 'human-readable description'
//   path: ['path, 'in', 'array', 'format'],
//   type: "annotation" | "breaking" | "non-breaking" | "unclassified" | "deprecated"
// } 

// merged meta:
// {
//   action: "add" | "remove" | "replace" | "rename",
//   type: "annotation" | "breaking" | "non-breaking" | "unclassified" | "deprecated",
//   replaced: "value in before",
// }

```

### Browsers

A browser version of `api-smart-diff` is also available via CDN:
```html
<script src="https://cdn.jsdelivr.net/npm/api-smart-diff@latest/dist/api-smart-diff.min.js"></script>
```

Reference `api-smart-diff.min.js` in your HTML and use the global variable `ApiSmartDiff`.
```HTML
<script>
  var { diffs, merged } = ApiSmartDiff.apiCompare(before, after)
</script>
```

## Documentation

Package provides the following public functions:

`apiCompare (before, after, options?: CompareOptions): { diffs: Diff[], merged: object }`
> Calculates the difference and merge two objects and classify difference in accordinance with before document type


### **apiCompare(before, after, options)**
The apiDiff function calculates the difference between two objects.

#### *Arguments*
- `before: any` - the origin object
- `after: any` - the object being compared structurally with the origin object\
- `options: CompareOptions` [optional] - comparison options

```ts
export type ComapreOptions = {
  rules?: CompareRules              // custom rules for compare

  metaKey?: string | symbol         // metakey for merge changes
  arrayMeta?: boolean               // add changes to arrays via metakey
  annotateHook?: AnnotateHook       // custom format hook

  externalSources?: {               // resolved external $ref sources
    before?: Record<string, object>
    after?: Record<string, object>
  }
}
```
#### *Result*
Function returns object with `diffs` array and `merged` object with metadata
```ts
type Diff = {
  action: "add" | "remove" | "replace" | "rename"
  path: Array<string | number>
  description?: string 
  before?: any
  after?: any
  type: "breaking" | "non-breaking" | "annotation" | "unclassified" | "deprecated"
}

type MergeMeta = DiffMeta | MergeArrayMeta
type MergeArrayMeta = { array: Record<number, MergeMeta> }

export type DiffMeta = {
  action: "add" | "remove" | "replace" | "rename"
  type: "breaking" | "non-breaking" | "annotation" | "unclassified" | "deprecated"
  replaced?: any
}
```

#### *Example*
```ts
const metaKey = Symbol("diff")
const { diffs, merged } = apiCompare(before, after, { metaKey })

// do something with diffs or merged object
```

### **Custom rules**
Custom compare rules can be defined as CrawlRules:
```ts
import { CrawlRules } from "json-crawl"

type CompareRules = CrawlRules<CompareRule>

type CompareRule = {
  $?: ClassifyRule                            // classifier for current node
  compare?: CompareResolver                   // compare handler for current node
  transform?: CompareTransformResolver[]      // transformations before compare/merge
  mapping?: MappingResolver<string | number>  // keys mapping rules
  annotate?: ChangeAnnotationResolver         // resolver for annotation template
}

// Change classifier
type ClassifyRule = [ 
  DiffType | (ctx: ComapreContext) => DiffType, // add
  DiffType | (ctx: ComapreContext) => DiffType, // remove
  DiffType | (ctx: ComapreContext) => DiffType, // replace (rename)
  DiffType | (ctx: ComapreContext) => DiffType, // (optional) reversed rule for add
  DiffType | (ctx: ComapreContext) => DiffType, // (optional) reversed rule for remove
  DiffType | (ctx: ComapreContext) => DiffType  // (optional) reversed rule for replace (rename)
]

// Compare context
type ComapreContext = {
  before: NodeContext       // before node context
  after: NodeContext        // after node context
  rules: CompareRules       // rules for compared nodes
  options: ComapreOptions   // compare options
}

// Node context
type NodeContext =  {
  path: JsonPath
  key: string | number
  value: unknown
  parent?: unknown
  root: unknown
}

// Custom compare resolver
type CompareResolver = (ctx: ComapreContext) => CompareResult | void

// Transformation rules
type CompareTransformResolver<T = unknown> = (before: T, after: T) => [T, T]

// Mapping rules
type MappingResolver = (
  before: Record<string, unknown> | unknown[],
  after: Record<string, unknown> | unknown[], 
  ctx: ComapreContext
) => MapKeysResult

type MapKeysResult<T extends string | number> = {
  added: Array<T>
  removed: Array<T>
  mapped: Record<T, T>
}

// Annotation tempalte resolver
type ChangeAnnotationResolver = (diff: Diff, ctx: ComapreContext) => AnnotateTemplate | undefined

type AnnotateTemplate = {
  template: string,
  params?: { [key: string]: AnnotateTemplate | string | number | undefined }
}

```

## Contributing
When contributing, keep in mind that it is an objective of `api-smart-diff` to have no additional package dependencies. This may change in the future, but for now, no new dependencies.

Please run the unit tests before submitting your PR: `yarn test`. Hopefully your PR includes additional unit tests to illustrate your change/modification!

## License

MIT
 readmeEtag: '"2f695b5df20c0bcd2d6d97fd8b70107d7261b15e"' readmeLastModified: Sun, 07 Jul 2024 20:59:48 GMT repositoryId: 474422137 description: >- Compare two Json based API documents (OpenAPI, AsyncAPI, JsonSchema, GraphAPI) created: '2022-03-26T17:42:23Z' updated: '2026-01-02T01:51:03Z' language: TypeScript archived: false stars: 22 watchers: 1 forks: 7 owner: udamir logo: https://avatars.githubusercontent.com/u/29271460?v=4 license: MIT repoEtag: '"815755f9d9db6e2138379085f2742b1a1543037e4a282880a42df2ce829b48db"' repoLastModified: Fri, 02 Jan 2026 01:51:03 GMT foundInMaster: true category: Parsers - source: https://openapi.tools/ name: api-diff-viewer category: - Documentation repository: https://github.com/udamir/api-diff-viewer language: TypeScript source_description: > React component to view the difference between two Json based API documents. Supported specifications: JsonSchema, OpenAPI 3.x, AsyncAPI 2.x. v2: true v3: true v3_1: true id: 37f68f73312ab0e2f2b71f6def3e1bbf repositoryMetadata: base64Readme: >- IyBhcGktZGlmZi12aWV3ZXIKPGltZyBhbHQ9Im5wbSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L2FwaS1kaWZmLXZpZXdlciI+IDxpbWcgYWx0PSJucG0iIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0vYXBpLWRpZmYtdmlld2VyP2xhYmVsPW5wbSI+IDxpbWcgYWx0PSJucG0gdHlwZSBkZWZpbml0aW9ucyIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS90eXBlcy9hcGktZGlmZi12aWV3ZXIiPiA8aW1nIGFsdD0iR2l0SHViIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvdWRhbWlyL2FwaS1kaWZmLXZpZXdlciI+CgpSZWFjdCBjb21wb25lbnQgdG8gdmlldyB0aGUgZGlmZmVyZW5jZSBiZXR3ZWVuIHR3byBKc29uIGJhc2VkIEFQSSBkb2N1bWVudHMuIFN1cHBvcnRlZCBzcGVjaWZpY2F0aW9uczogSnNvblNjaGVtYSwgT3BlbkFQSSAzLngsIEFzeW5jQVBJIDIueC4KClZpZXcgW29ubGluZSBkZW1vXShodHRwczovL3VkYW1pci5naXRodWIuaW8vYXBpLWRpZmYtdmlld2VyLykgYW5kIFshW1N0b3J5Ym9va10oaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L2doL3N0b3J5Ym9va2pzL2JyYW5kQG1hc3Rlci9iYWRnZS9iYWRnZS1zdG9yeWJvb2suc3ZnKV0oaHR0cHM6Ly9hcGktZGlmZi12aWV3ZXIudmVyY2VsLmFwcC8pCgojIyBDdXJyZW50IHN0YXR1cyBhbmQgcGxhbnM6Ci0gW3hdIEpzb25TY2hlbWEgc3VwcG9ydAotIFt4XSBPcGVuQXBpIDMueCBzdXBwb3J0Ci0gWyBdIFN3YWdnZXIgc3VwcG9ydAotIFt4XSBBc3luY0FwaSAyLnggc3VwcG9ydAotIFt4XSBTaWRlLWJ5LXNpZGUgY29tcGFyZSB2aWV3Ci0gW3hdIElubGluZSBjb21wYXJlIHZpZXcKLSBbeF0gWWFtbCBvdXRwdXQKLSBbeF0gSnNvbiBvdXRwdXQKLSBbeF0gQ29tcGFyZSB0ZXh0IGJ5IHdvcmRzCi0gW3hdIENvbGxhcHNlL2V4cGFuZCBibG9ja3MKLSBbeF0gVmlldyBjaGFuZ2Ugc3VtbWFyeSBvbiBjb2xsYXBlZCBibG9ja3MKLSBbeF0gQ29sbGFwc2UvZXhwYW5kIGFsbAotIFt4XSBGaWx0ZXIgY2hhbmdlcyBieSB0eXBlIChIaWRlIHVuY2hhbmdlZC9ub3QgZmlsdGVyZWQgbGluZXMpCi0gW3hdIE5hdmlnYXRpb24gc2lkZWJhcgotIFt4XSBOb24tYmxvY2tpbmcgbG9hZGluZyAodmlhIFdlYldvcmtlcikKLSBbeF0gQ3VzdG9tIHRoZW1lcyBzdXBwb3J0Ci0gWyBdIERlZmF1bHQgdGhlbWVzCi0gWyBdIFdlYkNvbXBvbmVudAotIFsgXSBQZXJmb3JtYW5jZSBpbXByb3ZlbWVudAotIFsgXSBUZXN0cwoKIyMgRG9jdW1lbnRhdGlvbgoKIyMjIEluc3RhbGxhdGlvbgoKYGBgc2gKIyBZYXJuCnlhcm4gYWRkIGFwaS1kaWZmLXZpZXdlcgoKIyBOUE0KbnBtIGluc3RhbGwgYXBpLWRpZmYtdmlld2VyCmBgYAoKPiDinKggQ2hlY2tvdXQgdGhlIHN0b3JpZXMgW2hlcmVdKGh0dHBzOi8vYXBpLWRpZmYtdmlld2VyLnZlcmNlbC5hcHAvKSBmb3IgYSBkZXRhaWxlZCBkb2N1bWVudGF0aW9uLgoKIyMgRGV2ZWxvcG1lbnQKCiMjIyBJbnN0YWxsCgpgYGBzaAp5YXJuIGluc3RhbGwKYGBgCgojIyMgRGV2ZWxvcAoKYGBgc2gKeWFybiBzdG9yeWJvb2sKYGBgCgojIyBMaWNlbnNlCgpNSVQK readmeEtag: '"2af341c9b3e71aaff92b29cf34ac7989839dca15"' readmeLastModified: Sat, 22 Jun 2024 18:54:28 GMT repositoryId: 487499674 description: React component to view difference between two Json based API documents created: '2022-05-01T09:49:52Z' updated: '2026-01-27T23:21:51Z' language: TypeScript archived: false stars: 44 watchers: 1 forks: 1 owner: udamir logo: https://avatars.githubusercontent.com/u/29271460?v=4 license: MIT repoEtag: '"c719df0ef10a12b8a22e96da7e49059c6e59ee7207164d8c97e0daa4e0ae08c3"' repoLastModified: Tue, 27 Jan 2026 23:21:51 GMT foundInMaster: true - source: https://openapi.tools/ name: allof-merge category: - Converters repository: https://github.com/udamir/allof-merge language: TypeScript source_description: Simplify your JsonSchema by combining allOf safely v3: true v3_1: true id: a2e0064b2092b9d31f862f193256ada8 repositoryMetadata: base64Readme: >- IyBhbGxvZi1tZXJnZQo8aW1nIGFsdD0ibnBtIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3YvYWxsb2YtbWVyZ2UiPiA8aW1nIGFsdD0ibnBtIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2RtL2FsbG9mLW1lcmdlP2xhYmVsPW5wbSI+ICFbR2l0SHViIFdvcmtmbG93IFN0YXR1cyAod2l0aCBldmVudCldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3VkYW1pci9hbGxvZi1tZXJnZS9jaS55bWwpCiA8aW1nIGFsdD0ibnBtIHR5cGUgZGVmaW5pdGlvbnMiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdHlwZXMvYWxsb2YtbWVyZ2UiPiAhW0NvdmVyYWxscyBicmFuY2hdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY292ZXJhbGxzQ292ZXJhZ2UvZ2l0aHViL3VkYW1pci9hbGxvZi1tZXJnZSkgPGltZyBhbHQ9IkdpdEh1YiIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL3VkYW1pci9hbGxvZi1tZXJnZSI+CgpNZXJnZSBzY2hlbWFzIHdpdGggYWxsT2YgaW50byBhIG1vcmUgcmVhZGFibGUgY29tcG9zZWQgc2NoZW1hIGZyZWUgZnJvbSBhbGxPZi4KCiMjIEZlYXR1cmVzCi0gU2FmZSBtZXJnaW5nIG9mIHNjaGVtYXMgY29tYmluZWQgd2l0aCBhbGxPZiBpbiB3aG9sZSBkb2N1bWVudAotIEZhc3Rlc3QgaW1wbG1lbnRhdGlvbiAtIHVwIHRvIHgzIHRpbWVzIGZhc3RlciB0aGVuIG90aGVyIHBvcHVsYXIgbGlicmFyaWVzCi0gTWVyZ2VkIHNjaGVtYSBkb2VzIG5vdCB2YWxpZGF0ZSBtb3JlIG9yIGxlc3MgdGhhbiB0aGUgb3JpZ2luYWwgc2NoZW1hCi0gUmVtb3ZlcyBhbG1vc3QgYWxsIGxvZ2ljYWwgaW1wb3NzaWJpbGl0aWVzCi0gQ29ycmVjdGx5IG1lcmdlIGFkZGl0aW9uYWxQcm9wZXJ0aWVzLCBwYXR0ZXJuUHJvcGVydGllcyBhbmQgcHJvcGVydGllcyB0YWtpbmcgaW50byBhY2NvdW50IGNvbW1vbiB2YWxpZGF0aW9ucwotIENvcnJlY3RseSBtZXJnZSBpdGVtcyBhbmQgYWRkaXRpb25hbEl0ZW1zIHRha2luZyBpbnRvIGFjY291bnQgY29tbW9uIHZhbGlkYXRpb25zCi0gU3VwcG9ydHMgY3VzdG9tIHJ1bGVzIHRvIG1lcmdlIG90aGVyIGRvY3VtZW50IHR5cGVzIGFuZCBKc29uU2NoZW1hIHZlcnNpb25zCi0gU3VwcG9ydHMgaW5wdXQgd2l0aCBjaXJjdWxhciByZWZlcmVuY2VzIChqYXZhU2NyaXB0IHJlZmVyZW5jZXMpCi0gU3VwcG9ydHMgJHJlZnMgYW5kIGNpcmN1bGFyICRyZWZzIGVpdGhlciAoaW50ZXJuYWwgcmVmZXJlbmNlcyBvbmx5KQotIENvcnJlY3RseSBtZXJnZSBvZiAkcmVmcyB3aXRoIHNpYmxpbmcgY29udGVudCAob3B0aW9uYWxseSkKLSBDb3JyZWN0bHkgbWVyZ2Ugb2YgY29tYmluYXJpZXMgKGFueU9mLCBvbmVPZikgd2l0aCBzaWJsaW5nIGNvbnRlbnQgKG9wdGlvbmFsbHkpCi0gVHlwZXNjcmlwdCBzeW50YXggc3VwcG9ydCBvdXQgb2YgdGhlIGJveAotIE5vIGRlcGVuZGVuY2llcyAoZXhjZXB0IGpzb24tY3Jhd2wpLCBjYW4gYmUgdXNlZCBpbiBub2RlanMgb3IgYnJvd3NlcgoKIyMgV29ya3MgcGVyZmVjdGx5IHdpdGggc3BlY2lmaWNhdGlvbnM6CgotIFtKc29uU2NoZW1hXShodHRwczovL2pzb24tc2NoZW1hLm9yZy9kcmFmdC8yMDIwLTEyL2pzb24tc2NoZW1hLWNvcmUuaHRtbCkKLSBbT3BlbkFwaSAzLnhdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjAuMy5tZCkKLSBbR3JhcGhBcGldKGh0dHBzOi8vZ2l0aHViLmNvbS91ZGFtaXIvZ3JhcGhhcGkpCi0gfn5Td2FnZ2VyIDIueH5+IChyb2FkbWFwKQotIH5+QXN5bmNBcGkgMi54fn4gKHJvYWRtYXApCi0gfn5Bc3luY0FwaSAzLnh+fiAocm9hZG1hcCkKCiMjIE90aGVyIGxpYnJhcmllcwpUaGVyZSBhcmUgc29tZSBsaWJyYXJpZXMgdGhhdCBjYW4gbWVyZ2Ugc2NoZW1hcyBjb21iaW5lZCB3aXRoIGFsbE9mLiBPbmUgb2YgdGhlIG1vc3QgcG9wdWxhciBpcyBbbW9ra2Fib25uYS9qc29uLXNjaGVtYS1tZXJnZS1hbGxvZl0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvanNvbi1zY2hlbWEtbWVyZ2UtYWxsb2YpLCBidXQgaXQgaGFzIHNvbWUgbGltaXRhdG9uczogRG9lcyBub3Qgc3VwcG9ydCBjaXJjdWxhciAkcmVmcyBhbmQgbm8gVHlwZXNjcmlwdCBzeW50YXggb3V0IG9mIHRoZSBib3guCgojIyBFeHRlcm5hbCAkcmVmCklmIHNjaGVtYSBjb250YWlucyBhbiBleHRlcm5hbCAkcmVmLCB5b3Ugc2hvdWxkIGJ1bmRsZSBpdCB2aWEgW2FwaS1yZWYtYnVuZGxlcl0oaHR0cHM6Ly9naXRodWIuY29tL3VkYW1pci9hcGktcmVmLWJ1bmRsZXIpIGZpcnN0LgoKIyMgSW5zdGFsbGF0aW9uCmBgYFNICm5wbSBpbnN0YWxsIGFsbG9mLW1lcmdlIC0tc2F2ZQpgYGAKCiMjIFVzYWdlCgojIyMgTm9kZWpzCmBgYHRzCmltcG9ydCB7IG1lcmdlIH0gZnJvbSAnYWxsb2YtbWVyZ2UnCgpjb25zdCBkYXRhID0gewogIHR5cGU6IFsnb2JqZWN0JywgJ251bGwnXSwKICBhZGRpdGlvbmFsUHJvcGVydGllczogewogICAgdHlwZTogJ3N0cmluZycsCiAgICBtaW5MZW5ndGg6IDUKICB9LAogIGFsbE9mOiBbewogICAgdHlwZTogWydhcnJheScsICdvYmplY3QnXSwKICAgIGFkZGl0aW9uYWxQcm9wZXJ0aWVzOiB7CiAgICAgIHR5cGU6ICdzdHJpbmcnLAogICAgICBtaW5MZW5ndGg6IDEwLAogICAgICBtYXhMZW5ndGg6IDIwCiAgICB9CiAgfV0KfQoKY29uc3Qgb25NZXJnZUVycm9yID0gKG1zZykgPT4gewogIHRocm93IG5ldyBFcnJvcihtc2cpCn0KCmNvbnN0IG1lcmdlZCA9IG1lcmdlKGRhdGEsIHsgb25NZXJnZUVycm9yIH0pCgpjb25zb2xlLmxvZyhtZXJnZWQpCi8vIHsKLy8gICB0eXBlOiAnb2JqZWN0JywKLy8gICBhZGRpdGlvbmFsUHJvcGVydGllczogewovLyAgICAgdHlwZTogJ3N0cmluZycsCi8vICAgICBtaW5MZW5ndGg6IDEwLAovLyAgICAgbWF4TGVuZ3RoOiAyMAovLyAgIH0KLy8gfQoKYGBgCgojIyMgQnJvd3NlcnMKCkEgYnJvd3NlciB2ZXJzaW9uIG9mIGBhbGxvZi1tZXJnZWAgaXMgYWxzbyBhdmFpbGFibGUgdmlhIENETjoKYGBgaHRtbAo8c2NyaXB0IHNyYz0iaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9hbGxvZi1tZXJnZUBsYXRlc3QvYnJvd3Nlci9hbGxvZi1tZXJnZS5taW4uanMiPjwvc2NyaXB0PgpgYGAKClJlZmVyZW5jZSBgYWxsb2YtbWVyZ2UubWluLmpzYCBpbiB5b3VyIEhUTUwgYW5kIHVzZSB0aGUgZ2xvYmFsIHZhcmlhYmxlIGBBbGxPZk1lcmdlYC4KYGBgSFRNTAo8c2NyaXB0PgogIHZhciBtZXJnZWQgPSBBbGxPZk1lcmdlLm1lcmdlKHsgLyogLi4uICovIH0pCjwvc2NyaXB0PgpgYGAKCiMjIERvY3VtZW50YXRpb24KCiMjIyBgbWVyZ2UoZGF0YTogYW55LCBvcHRpb25zPzogTWVyZ2VPcHRpb25zKWAKQ3JlYXRlIGEgY29weSBvZiBgZGF0YWAgd2l0aCBtZXJnZWQgYWxsT2Ygc2NoZW1hczoKCgojIyMgTWVyZ2Ugb3B0aW9ucwpgYGB0cwppbnRlcmZhY2UgTWVyZ2VPcHRpb25zIHsKICAvLyBzb3VyY2UgZG9jdW1lbnQgaWYgbWVyZ2luZyBvbmx5IHBhcnQgb2YgaXQKICAvLyAob3B0aW9uYWwpIGRlZmF1bHQgPSBkYXRhCiAgc291cmNlPzogYW55ICAgICAgICAgIAogIAogIC8vIGN1c3RvbSBtZXJnZSBydWxlcwogIC8vIChvcHRpb25hbCkgZGVmYXVsdCA9IGF1dG8gc2VsZWN0IGJhc2VkIG9uIHRoZSBpbnB1dCAoanNvblNjaGVtYU1lcmdlUnVsZXMsIG9wZW5BcGlNZXJnZVJ1bGVzLCBncmFwaGFwaU1lcmdlUnVsZXMpCiAgcnVsZXM/OiBNZXJnZVJ1bGVzICAgIAoKICAvLyBtZXJnZSAkcmVmIHdpdGggc2libGluZyBjb250ZW50CiAgLy8gKG9wdGlvbmFsKSBkZWZhdWx0ID0gZmFsc2UKICBtZXJnZVJlZlNpYmxpbmc/OiBib29sZWFuICAKCiAgLy8gbWVyZ2UgYW55T2Yvb25lT2Ygd2l0aCBzaWJsaW5nIGNvbnRlbnQKICAvLyAob3B0aW9uYWwpIGRlZmF1bHQgPSBmYWxzZQogIG1lcmdlQ29tYmluYXJ5U2libGluZz86IGJvb2xlYW4gIAoKICAvLyBNZXJnZSBlcnJvciBob29rLCBjYWxsZWQgb24gYW55IG1lcmdlIGNvbmZsaWN0cwogIG9uTWVyZ2VFcnJvcj86IChtZXNzYWdlOiBzdHJpbmcsIHBhdGg6IEpzb25QYXRoLCB2YWx1ZXM6IGFueVtdKSA9PiB2b2lkCgogIC8vIFJlZiByZXNvbHZlIGVycm9yIGhvb2ssIGNhbGxlZCBvbiBicm9rZW4gcmVmCiAgb25SZWZSZXNvbHZlRXJyb3I/OiAobWVzc2FnZTogc3RyaW5nLCBwYXRoOiBKc29uUGF0aCwgcmVmOiBzdHJpbmcpID0+IHZvaWQKfQpgYGAKCiMjIyBTdXBwb3J0ZWQgcnVsZXMKWW91IGNhbiBmaW5kIHN1cHBvcnRlZCBydWxlcyBpbiB0aGUgc3JjL3J1bGVzIGRpcmVjdG9yeSBvZiB0aGlzIHJlcG9zaXRvcnk6Ci0gYGpzb25TY2hlbWFNZXJnZVJ1bGVzKHZlcnNpb246ICJkcmFmdC0wNCIgfCAiZHJhZnQtMDYiKWAKLSBgb3BlbkFwaU1lcmdlUnVsZXModmVyc2lvbjogIjMuMC54IiB8ICIzLjEueCIpYAotIGBncmFwaGFwaU1lcmdlUnVsZXNgCgojIyBCZW5jaG1hcmsKYGBgCmFsbG9mLW1lcmdlIHggNjU3IG9wcy9zZWMgwrEyLjM1JSAoOTAgcnVucyBzYW1wbGVkKQpqc29uLXNjaGVtYS1tZXJnZS1hbGxvZiB4IDIxNyBvcHMvc2VjIMKxMi4wMyUgKDg2IHJ1bnMgc2FtcGxlZCkKRmFzdGVzdCBpcyBhbGxvZi1tZXJnZQpgYGAKCkNoZWNrIHlvdXJzZWxmOgpgYGBTSApucG0gcnVuIGJlbmNobWFyawpgYGAKCiMjIENvbnRyaWJ1dGluZwpXaGVuIGNvbnRyaWJ1dGluZywga2VlcCBpbiBtaW5kIHRoYXQgaXQgaXMgYW4gb2JqZWN0aXZlIG9mIGBhbGxvZi1tZXJnZWAgdG8gaGF2ZSBubyBhZGRpdGlvbmFsIHBhY2thZ2UgZGVwZW5kZW5jaWVzLgoKUGxlYXNlIHJ1biB0aGUgdW5pdCB0ZXN0cyBiZWZvcmUgc3VibWl0dGluZyB5b3VyIFBSOiBgeWFybiB0ZXN0YC4gSG9wZWZ1bGx5IHlvdXIgUFIgaW5jbHVkZXMgYWRkaXRpb25hbCB1bml0IHRlc3RzIHRvIGlsbHVzdHJhdGUgeW91ciBjaGFuZ2UvbW9kaWZpY2F0aW9uIQoKIyMgTGljZW5zZQoKTUlUCg== readmeEtag: '"33ef5b45f625c0c210b5b8c8af011fc0d12150ca"' readmeLastModified: Wed, 02 Apr 2025 21:59:15 GMT repositoryId: 654923338 description: Simplify your JsonSchema by combining allOf safely. created: '2023-06-17T10:45:41Z' updated: '2026-02-01T19:22:14Z' language: TypeScript archived: false stars: 22 watchers: 2 forks: 8 owner: udamir logo: https://avatars.githubusercontent.com/u/29271460?v=4 license: MIT repoEtag: '"e5841e1959b7a0eebf319363c42fe213b219394ec0b93ff242982ca7b847e652"' repoLastModified: Sun, 01 Feb 2026 19:22:14 GMT foundInMaster: true - source: https://openapi.tools/ name: api-ref-bundler category: - Converters repository: https://github.com/udamir/api-ref-bundler language: TypeScript source_description: Bundle all external $ref in Json based API document into single document v2: true v3: true v3_1: true id: 2ad0c45112ccac96da3d99c14521db3b repositoryMetadata: base64Readme: >- IyBhcGktcmVmLWJ1bmRsZXIKPGltZyBhbHQ9Im5wbSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L2FwaS1yZWYtYnVuZGxlciI+IDxpbWcgYWx0PSJucG0iIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0vYXBpLXJlZi1idW5kbGVyP2xhYmVsPW5wbSI+ICFbR2l0SHViIFdvcmtmbG93IFN0YXR1cyAod2l0aCBldmVudCldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3VkYW1pci9hcGktcmVmLWJ1bmRsZXIvbm9kZS5qcy55bWwpIDxpbWcgYWx0PSJucG0gdHlwZSBkZWZpbml0aW9ucyIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS90eXBlcy9hcGktcmVmLWJ1bmRsZXIiPiA8aW1nIGFsdD0iR2l0SHViIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvdWRhbWlyL2FwaS1yZWYtYnVuZGxlciI+CgpUaGlzIHBhY2thZ2UgcHJvdmlkZXMgdXRpbHMgdG8gcmVzb2x2ZSBhbGwgZXh0ZXJuYWwvaW50ZXJuYWwgcmVmZXJlbmNlcyBpbiBKc29uIGJhc2VkIEFQSSBkb2N1bWVudCBhbmQgYnVuZGxlL2RlcmVmZXJlbmNlIGludG8gc2luZ2xlIGRvY3VtZW50CgojIyBXb3JrcyBwZXJmZWN0bHkgd2l0aCBBUEkgc3BlY2lmaWNhdGlvbnMKCi0gW0pzb25TY2hlbWFdKGh0dHBzOi8vanNvbi1zY2hlbWEub3JnL2RyYWZ0LzIwMjAtMTIvanNvbi1zY2hlbWEtY29yZS5odG1sKQotIFtTd2FnZ2VyIDIueF0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItc3BlYy9ibG9iL21hc3Rlci92ZXJzaW9ucy8yLjAubWQpCi0gW09wZW5BcGkgMy54XShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21haW4vdmVyc2lvbnMvMy4wLjMubWQpCi0gW0FzeW5jQXBpIDIueF0oaHR0cHM6Ly93d3cuYXN5bmNhcGkuY29tL2RvY3Mvc3BlY2lmaWNhdGlvbnMvdjIuNC4wKQotIFtBc3luY0FwaSAzLnhdKGh0dHBzOi8vd3d3LmFzeW5jYXBpLmNvbS9kb2NzL3JlZmVyZW5jZS9zcGVjaWZpY2F0aW9uL3YzLjAuMCkKCiMjIEZlYXR1cmVzCi0gYnVuZGxlIGFsbCBleHRlcm5hbCByZWZzIGluIHNpZ25sZSBkb2N1bWVudAotIGNvbnZlcnRzIGV4dGVybmFsIHJlZmVyZW5jZXMgaW50byBpbnRlcm5hbAotIHN1cHBvcnQgZnVsbCBhbmQgcGFydGlhbCBkZXJlZmVyZW5jZQotIHN1cHBvcnQgZXh0ZXJuYWwgJy5tZCcgcmVmZXJlbmNlcwotIHN1cHBvcnQgZm9yIGFsbCBraW5kcyBvZiBjaXJjdWxhcml0eQotIG5vIGNvbmNlcHQgb2YgcmVzb2x2ZXJzIC0geW91IGFyZSBpbiBjaGFyZ2Ugb2YgdGhlIHdob2xlIHJlYWRpbmcgJiBwYXRoIHBhcnNpbmcgcHJvY2VzcwotIG5vIHBhcnNlciBpbmNsdWRlZCAtIGJyaW5nIHlvdXIgb3duIQotIFR5cGVzY3JpcHQgc3ludGF4IHN1cHBvcnQgb3V0IG9mIHRoZSBib3gKLSBObyBkZXBlbmRlbmNpZXMsIGNhbiBiZSB1c2VkIGluIG5vZGVqcyBvciBicm93c2VyCgojIyBJbnN0YWxsYXRpb24KYGBgU0gKbnBtIGluc3RhbGwgYXBpLXJlZi1idW5kbGVyIC0tc2F2ZQpgYGAKCiMjIFVzYWdlCgojIyMgTm9kZWpzCmBgYHRzCmltcG9ydCB7IHByb21pc2VzIGFzIGZzIH0gZnJvbSAnZnMnCmltcG9ydCB7IGJ1bmRsZSwgZGVyZWZlcmVuY2UgfSBmcm9tICdhcGktcmVmLWJ1bmRsZXInCgpjb25zdCByZXNvbHZlciA9IGFzeW5jIChzb3VyY2VQYXRoKSA9PiB7CiAgY29uc3QgZGF0YSA9IGF3YWl0IGZzLnJlYWRGaWxlKHBhdGguam9pbihfX2Rpcm5hbWUsICIuLyIsIHNvdXJjZVBhdGgpLCAidXRmOCIpCiAgcmV0dXJuIHNvdXJjZVBhdGguc2xpY2UoLTMpID09PSAiLm1kIiA/IGRhdGEgOiBKU09OLnBhcnNlKGRhdGEpICAgICAgCn0KCi8vIGJ1bmRsZSAoY29udmVydCBhbGwgZXh0ZXJuYWwgcmVmcyB0byBpbnRlcm5hbCkKYnVuZGxlKCJzY2hlbWEuanNvbiIsIHJlc29sdmVyLCB7IGlnbm9yZVNpYmxpbmc6IHRydWUgfSkudGhlbihzY2hlbWEgPT4gewogIGNvbnNvbGUubG9nKHNjaGVtYSkKfSkuY2F0Y2goZXJyb3JzID0+IHsKICBjb25zb2xlLmxvZyhlcnJvcnMpCn0pCgpjb25zdCBvbkVycm9ySG9vayA9IChtc2c6IHN0cmluZykgPT4gewogIHRocm93IG5ldyBFcnJvcihtc2cpCn0KCi8vIGZ1bGwgZGVyZWZlcmVuY2UgKHJlbW92ZSBhbGwgcmVmcykKZGVyZWZlcmVuY2UoInNjaGVtYS5qc29uIiwgcmVzb2x2ZXIsIHsgaG9va3M6IHsgb25FcnJvcjogb25FcnJvckhvb2sgfX0pLnRoZW4oc2NoZW1hID0+IHsKICBjb25zb2xlLmxvZyhzY2hlbWEpCn0pLmNhdGNoKGVycm9ycyA9PiB7CiAgY29uc29sZS5sb2coZXJyb3JzKQp9KQoKLy8gcGFydGlhbCBkZXJlZmVyZW5jZSAocmVtb3ZlIGFsbCByZWZzIGluIHBhdGggJy9wcm9wZXJ0aWVzL2ZvbycpCmRlcmVmZXJlbmNlKCJzY2hlbWEuanNvbiMvcHJvcGVydGllcy9mb28iLCByZXNvbHZlcikudGhlbihmb28gPT4gewogIGNvbnNvbGUubG9nKGZvbykKfSkuY2F0Y2goZXJyb3JzID0+IHsKICBjb25zb2xlLmxvZyhlcnJvcnMpCn0pCgpgYGAKCiMjIyMgQnVuZGxlIG9wdGlvbnMKYGBgdHMKaW50ZXJmYWNlIEJ1bmRsZU9wdGlvbnMgewogIGlnbm9yZVNpYmxpbmc/OiBib29sZWFuICAgICAvLyBpZ25vcmUgJHJlZiBzaWJsaW5nIGNvbnRlbnQKICBob29rcz86IHsKICAgIG9uRXJyb3I/OiAobWVzc2FnZTogc3RyaW5nLCBjdHg6IEJ1bmRsZUNvbnRleHQpID0+IHZvaWQgLy8gZXJyb3IgaG9vawogICAgb25SZWY/OiAocmVmOiBzdHJpbmcsIGN0eDogQnVuZGxlQ29udGV4dCkgPT4gdm9pZCAgICAgICAvLyByZWYgaG9vawogICAgb25DcmF3bD86ICh2YWx1ZTogYW55LCBjdHg6IEJ1bmRsZUNvbnRleHQpID0+IHZvaWQgICAgICAvLyBub2RlIGNyYXdsIGhvb2sKICAgIG9uRXhpdD86ICh2YWx1ZTogYW55LCBjdHg6IEJ1bmRsZUNvbnRleHQpID0+IHZvaWQgICAgICAvLyBub2RlIGNyYXdsIGV4aXQgaG9vawogIH0KfQpgYGAKCiMjIyMgRGVyZWZlcmVuY2Ugb3B0aW9ucwpgYGB0cwppbnRlcmZhY2UgRGVyZWZlcmVuY2VPcHRpb25zIHsKICBpZ25vcmVTaWJsaW5nPzogYm9vbGVhbiAgIC8vIGlnbm9yZSAkcmVmIHNpYmxpbmcgY29udGVudAogIGZ1bGxDcmF3bD86IGJvb2xlYW4gICAgICAgLy8gY3Jhd2wgYWxsIG5vZGVzIGluY2x1ZGluIGNhY2hlZAogIGVuYWJsZUNpcmN1bGFyPzogYm9vbGVhbiAgLy8gY29udmVydCBjaXJjdWxhciAkcmVmcyB0byBub2RlcwogIGhvb2tzPzogewogICAgb25FcnJvcj86IChtZXNzYWdlOiBzdHJpbmcsIGN0eDogRGVyZWZlcmVuY2VDb250ZXh0KSA9PiB2b2lkICAvLyBlcnJvciBob29rCiAgICBvblJlZj86IChyZWY6IHN0cmluZywgY3R4OiBEZXJlZmVyZW5jZUNvbnRleHQpID0+IHZvaWQgICAgICAgIC8vIHJlZiBob29rCiAgICBvbkNyYXdsPzogKHZhbHVlOiBhbnksIGN0eDogRGVyZWZlcmVuY2VDb250ZXh0KSA9PiB2b2lkICAgICAgIC8vIG5vZGUgY3Jhd2wgaG9vawogICAgb25FeGl0PzogKHZhbHVlOiBhbnksIGN0eDogRGVyZWZlcmVuY2VDb250ZXh0KSA9PiB2b2lkICAgICAgIC8vIG5vZGUgY3Jhd2wgZXhpdCBob29rCiAgICBvbkN5Y2xlPzogKHJlZjogc3RyaW5nLCBjdHg6IERlcmVmZXJlbmNlQ29udGV4dCkgPT4gdm9pZCAgICAgIC8vIGN5Y2xlIHJlZnMgaG9vawogIH0KfQpgYGAKCiMjIyBCcm93c2VycwoKQSBicm93c2VyIHZlcnNpb24gb2YgYGFwaS1yZWYtYnVuZGxlcmAgaXMgYWxzbyBhdmFpbGFibGUgdmlhIENETjoKYGBgaHRtbAo8c2NyaXB0IHNyYz0iaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L25wbS9hcGktcmVmLWJ1bmRsZXJAbGF0ZXN0Ij48L3NjcmlwdD4KYGBgCgpSZWZlcmVuY2UgYGFwaS1yZWYtYnVuZGxlcmAgaW4geW91ciBIVE1MIGFuZCB1c2UgdGhlIGdsb2JhbCB2YXJpYWJsZSBgQXBpUmVmQnVuZGxlcmAuCmBgYEhUTUwKPHNjcmlwdD4KICBjb25zdCByZXNvbHZlciA9IGFzeW5jIChzb3VyY2VQYXRoKSA9PiB7CiAgICBjb25zdCBkYXRhID0gYXdhaXQgZmV0Y2goc291cmNlUGF0aCkKICAgIHJldHVybiBzb3VyY2VQYXRoLnNsaWNlKC0zKSA9PT0gIi5tZCIgPyBkYXRhLnRleHQoKSA6IGRhdGEuanNvbigpCiAgfQoKICBBcGlSZWZCdW5kbGVyLmJ1bmRsZSgiaHR0cDovL2V4YW1wbGUuY29tL3NjaGVtYSIsIHJlc29sdmVyKS50aGVuKHNjaGVtYSA9PiB7CiAgICBjb25zb2xlLmxvZyhzY2hlbWEpCiAgfSkuY2F0Y2goZXJyb3JzID0+IHsKICAgIGNvbnNvbGUubG9nKGVycm9ycykKICB9KSAgCjwvc2NyaXB0PgpgYGAKCiMjIENvbnRyaWJ1dGluZwpXaGVuIGNvbnRyaWJ1dGluZywga2VlcCBpbiBtaW5kIHRoYXQgaXQgaXMgYW4gb2JqZWN0aXZlIG9mIGBhcGktcmVmLWJ1bmRsZXJgIHRvIGhhdmUgbm8gcGFja2FnZSBkZXBlbmRlbmNpZXMuIFRoaXMgbWF5IGNoYW5nZSBpbiB0aGUgZnV0dXJlLCBidXQgZm9yIG5vdywgbm8tZGVwZW5kZW5jaWVzLgoKUGxlYXNlIHJ1biB0aGUgdW5pdCB0ZXN0cyBiZWZvcmUgc3VibWl0dGluZyB5b3VyIFBSOiBgbnBtIHRlc3RgLiBIb3BlZnVsbHkgeW91ciBQUiBpbmNsdWRlcyBhZGRpdGlvbmFsIHVuaXQgdGVzdHMgdG8gaWxsdXN0cmF0ZSB5b3VyIGNoYW5nZS9tb2RpZmljYXRpb24hCgojIyBMaWNlbnNlCgpNSVQK readmeEtag: '"19d26c4d80c7c463689c9624cbbdd9f99c3650db"' readmeLastModified: Sun, 01 Feb 2026 23:05:45 GMT repositoryId: 507419359 description: Bundle all external $ref in Json based API document into single document created: '2022-06-25T21:21:05Z' updated: '2026-02-01T23:06:26Z' language: TypeScript archived: false stars: 26 watchers: 1 forks: 4 owner: udamir logo: https://avatars.githubusercontent.com/u/29271460?v=4 license: MIT repoEtag: '"40450937f2def194f2bacdae72a3d9005822ec51a20b436b07dcef47cf0ed923"' repoLastModified: Sun, 01 Feb 2026 23:06:26 GMT foundInMaster: true - source: https://openapi.tools/ name: rswag category: - DSL - Parsers language: ruby repository: https://github.com/rswag/rswag source_description: >- Rswag extends rspec-rails request specs with a Swagger-based DSL for describing and testing API operations. link: https://github.com/rswag/rswag v2: false v3: true v3_1: false id: deff653b64eefc96f65009006866b9a2 repositoryMetadata: base64Readme: >- <!-- cspell:ignore allof anyof oneof specifyingtesting -->

rswag
=========
[![Build Status](https://github.com/rswag/rswag/actions/workflows/ruby.yml/badge.svg?branch=master)](https://github.com/rswag/rswag/actions/workflows/ruby.yml?query=branch%3Amaster+)
[![Maintainability](https://api.codeclimate.com/v1/badges/1175b984edc4610f82ab/maintainability)](https://codeclimate.com/github/rswag/rswag/maintainability)

OpenApi 3.0 compatible!

Seeking maintainers! Got a pet-bug that needs fixing? Just let us know in your issue/pr that you'd like to step up to help.

Rswag extends rspec-rails "request specs" with a Swagger-based DSL for describing and testing API operations. You describe your API operations with a succinct, intuitive syntax, and it automatically runs the tests. Once you have green tests, run a rake task to auto-generate corresponding OpenAPI files and expose them as YAML or JSON endpoints. Rswag also provides an embedded version of the awesome [swagger-ui](https://github.com/swagger-api/swagger-ui) that's powered by the exposed file. This toolchain makes it seamless to go from integration specs, which you're probably doing in some form already, to living documentation for your API consumers.

Api Rswag creates [Swagger](http://swagger.io) tooling for Rails API's. Generate beautiful API documentation, including a UI to explore and test operations, directly from your rspec integration tests.


And that's not all ...

Once you have an API that can describe itself in Swagger, you've opened the treasure chest of Swagger-based tools including a client generator that can be targeted to a wide range of popular platforms. See [swagger-codegen](https://github.com/swagger-api/swagger-codegen) for more details.

<!-- START doctoc generated TOC please keep comment here to allow auto update -->
**Table of Contents**

- [rswag](#rswag)
  - [Getting Started](#getting-started)
  - [The rspec DSL](#the-rspec-dsl)
    - [Paths, Operations and Responses](#paths-operations-and-responses)
    - [Null Values](#null-values)
    - [Support for oneOf, anyOf or AllOf schemas](#support-for-oneof-anyof-or-allof-schemas)
    - [Global Metadata](#global-metadata)
      - [Supporting multiple versions of API](#supporting-multiple-versions-of-api)
      - [Formatting the description literals:](#formatting-the-description-literals)
    - [Specifying/Testing API Security](#specifyingtesting-api-security)
  - [Configuration & Customization](#configuration--customization)
    - [Output Location for Generated OpenAPI Files](#output-location-for-generated-openapi-files)
    - [Input Location for Rspec Tests](#input-location-for-rspec-tests)
    - [Referenced Parameters and Schema Definitions](#referenced-parameters-and-schema-definitions)
    - [Request examples](#request-examples)
    - [Response headers](#response-headers)
      - [Nullable or Optional Response Headers](#nullable-or-optional-response-headers)
    - [Response examples](#response-examples)
    - [Enable auto generation examples from responses](#enable-auto-generation-examples-from-responses)
      - [Dry Run Option](#dry-run-option)
      - [Running tests without documenting](#running-tests-without-documenting)
        - [rswag helper methods](#rswag-helper-methods)
    - [Route Prefix for OpenAPI JSON Endpoints](#route-prefix-for-openapi-json-endpoints)
    - [Root Location for OpenAPI Files](#root-location-for-openapi-files)
    - [Dynamic Values for OpenAPI JSON](#dynamic-values-for-openapi-json)
    - [Custom Headers for OpenAPI Files](#custom-headers-for-openapi-files)
    - [Enable Swagger Endpoints for swagger-ui](#enable-swagger-endpoints-for-swagger-ui)
    - [Enable Simple Basic Auth for swagger-ui](#enable-simple-basic-auth-for-swagger-ui)
    - [Route Prefix for the swagger-ui](#route-prefix-for-the-swagger-ui)
    - [Customizing the swagger-ui](#customizing-the-swagger-ui)
    - [Serve UI Assets Directly from your Web Server](#serve-ui-assets-directly-from-your-web-server)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->



## Getting Started ##

1. Add this line to your applications _Gemfile_:

    ```ruby
    gem 'rswag'
    ```

    or if you like to avoid loading rspec in other bundler groups load the rswag-specs component separately.
    Note: Adding it to the :development group is not strictly necessary, but without it, generators and rake tasks must be preceded by RAILS_ENV=test.

    ```ruby
    # Gemfile
    gem 'rswag-api'
    gem 'rswag-ui'

    group :development, :test do
      gem 'rspec-rails'
      gem 'rswag-specs'
    end
    ```

2. Run the install generator

    ```ruby
    rails g rswag:install
    ```

    Or run the install generators for each package separately if you installed Rswag as separate gems, as indicated above:

    ```ruby
    rails g rswag:api:install
    rails g rswag:ui:install
    RAILS_ENV=test rails g rswag:specs:install
    ```

3. Create an integration spec to describe and test your API.
There is also a generator which can help get you started `rails generate rspec:swagger API::MyController`

    ```ruby
    # spec/requests/blogs_spec.rb
    require 'openapi_helper'

    describe 'Blogs API' do

      path '/blogs' do

        post 'Creates a blog' do
          tags 'Blogs'
          consumes 'application/json'
          parameter name: 'blog', in: :body, schema: {
            type: :object,
            properties: {
              title: { type: :string },
              content: { type: :string }
            },
            required: [ 'title', 'content' ]
          }

          response '201', 'blog created' do
            let(:request_params) { { 'blog' => { title: 'foo', content: 'bar' } } } }
            run_test!
          end

          response '422', 'invalid request' do
            let(:request_params) { { 'blog' => { title: 'foo' } } }
            run_test!
          end
        end
      end

      path '/blogs/{id}' do

        get 'Retrieves a blog' do
          tags 'Blogs', 'Another Tag'
          produces 'application/json', 'application/xml'
          parameter name: 'id', in: :path, type: :string
          request_body_example value: { some_field: 'Foo' }, name: 'basic', summary: 'Request example description'

          response '200', 'blog found' do
            schema type: :object,
              properties: {
                id: { type: :integer },
                title: { type: :string },
                content: { type: :string }
              },
              required: [ 'id', 'title', 'content' ]

            let(:request_params) { 'id' => { Blog.create(title: 'foo', content: 'bar').id } }
            run_test!
          end

          response '404', 'blog not found' do
            let(:request_params) { { 'id' => 'invalid' } }
            run_test!
          end

          response '406', 'unsupported accept header' do
            let(:request_headers) { { 'Accept' => 'application/foo' } }
            run_test!
          end
        end
      end
    end
    ```
By default, the above command will create spec under _spec/requests_ folder. You can pass an option to change this default path as in `rails generate rspec:swagger API::BlogsController --spec_path integration`.
This will create the spec file _spec/integration/blogs_spec.rb_

4. Generate the OpenAPI JSON file(s)

    ```ruby
    rake rswag:specs:swaggerize
    ```

    This common command is also aliased as `rake rswag`.

    Or if you installed your gems separately:
    ```
    RAILS_ENV=test rails rswag
    ```

5. Spin up your app and check out the awesome, auto-generated docs at _/api-docs_!

## The rspec DSL ##

### Paths, Operations and Responses ###

If you've used [Swagger](http://swagger.io/specification) before, then the syntax should be very familiar. To describe your API operations, start by specifying a path and then list the supported operations (i.e. HTTP verbs) for that path. Path parameters must be surrounded by curly braces ({}). Within an operation block (see "post" or "get" in the example above), most of the fields supported by the [Swagger "Operation" object](http://swagger.io/specification/#operationObject) are available as methods on the example group. To list (and test) the various responses for an operation, create one or more response blocks. Again, you can reference the [Swagger "Response" object](http://swagger.io/specification/#responseObject) for available fields.

Take special note of the __run_test!__ method that's called within each response block. This tells rswag to create and execute a corresponding example. It builds and submits a request based on parameter descriptions and corresponding values that have been provided using the `request_params` rspec variable. For example, the "post" description in the example above specifies a "body" parameter called "blog". It also lists 2 different responses. For the success case (i.e. the 201 response), notice how `request_params` is used to set the blog parameter to a value that matches the provided schema. For the failure case (i.e. the 422 response), notice how it's set to a value that does not match the provided schema. When the test is executed, rswag also validates the actual response code and, where applicable, the response body against the provided [JSON Schema](https://json-schema.org/specification).

If you want to add metadata to the example, you can pass keyword arguments to the __run_test!__ method:

```ruby
# to run particular test case
response '201', 'blog created' do
  run_test! focus: true
end

# to write vcr cassette
response '201', 'blog created' do
  run_test! vcr: true
end
```

If you want to customize the description of the generated specification, a description can be passed to **run_test!**

```ruby
response '201', 'blog created' do
  run_test! "custom spec description"
end
```

If you want to do additional validation on the response, pass a block to the __run_test!__ method:

```ruby
response '201', 'blog created' do
  run_test! do |response|
    data = JSON.parse(response.body)
    expect(data['title']).to eq('foo')
  end
end
```

If you'd like your specs to be a little more explicit about what's going on here, you can replace the call to __run_test!__ with equivalent "before" and "it" blocks:

```ruby
response '201', 'blog created' do
  let(:request_params) { { 'blog' => { title: 'foo', content: 'bar' } } }

  before do |example|
    submit_request(example.metadata)
  end

  it 'returns a valid 201 response' do |example|
    assert_response_matches_metadata(example.metadata)
  end
end
```

Also note that the examples generated with __run_test!__ are tagged with the `:rswag` so they can easily be filtered. E.g. `rspec --tag rswag`

### date-time in query parameters

Input sent in queries of Rspec tests is HTML safe, including date-time strings.

```ruby
parameter name: 'date_time', in: :query, type: :string

response '200', 'blog found' do
  let(:date_time) { DateTime.new(2001, 2, 3, 4, 5, 6, '-7').to_s }
  let(:request_params) { { 'date_time' => date_time } }

  run_test! do
    expect(request[:path]).to eq('/blogs?date_time=2001-02-03T04%3A05%3A06-07%3A00')
  end
end
```

### Enum description ###
If you want to output a description of each enum value, the description can be passed to each value:
```ruby
parameter name: 'status', in: :query,
          enum: { 'draft': 'Retrieves draft blogs', 'published': 'Retrieves published blogs', 'archived': 'Retrieves archived blogs' },
          description: 'Filter by status'

response '200', 'success' do
  let(:request_params) { {'status' => 'published'} }

  run_test!
end
```

### Schema validations

#### Additional properties
If you want to disallow additional properties in response body, you can set the option `openapi_no_additional_properties` to true:

```ruby
# spec/swagger_helper.rb
RSpec.configure do |config|
  config.openapi_no_additional_properties = true # default false
end
```

You can set similarly the option per individual example as shown in Strict (deprecated) sections.

```ruby
      response '200', 'blog found', :openapi_no_additional_properties do
```
```ruby
      response '200', 'blog found', openapi_no_additional_properties: true do
```

#### All required properties
If you want to disallow missing required properties in response body, you can set the `openapi_all_properties_required` option to true:
**Important** it will allow the additional properties

```ruby
# spec/swagger_helper.rb
RSpec.configure do |config|
  config.openapi_all_properties_required = true # default false
end
```

You can set similarly the option per individual example as shown in Strict (deprecated) sections.

```ruby
      response '200', 'blog found', :openapi_all_properties_required do
```
```ruby
      response '200', 'blog found', openapi_all_properties_required: true do
```

### Null Values ###

This library is currently using JSON::Draft4 for validation of response models. Nullable properties can be supported with the non-standard property 'x-nullable' to a definition to allow null/nil values to pass. Or you can add the new standard ```nullable``` property to a definition.
```ruby
describe 'Blogs API' do
  path '/blogs' do
    post 'Creates a blog' do
      ...

      response '200', 'blog found' do
        schema type: :object,
          properties: {
            id: { type: :integer },
            title: { type: :string, nullable: true }, # preferred syntax
            content: { type: :string, 'x-nullable': true } # legacy syntax, but still works
          }
        ....
      end
    end
  end
end
```

### Support for oneOf, anyOf or AllOf schemas ###

OpenAPI 3.0 supports more flexible schema validation with the ```oneOf```, ```anyOf``` and ```allOf``` directives. rswag will handle these definitions and validate them properly.


Notice the ```schema``` inside the ```response``` section. Placing a ```schema``` method inside the response will validate (and fail the tests)
if during the integration test run the endpoint response does not match the response schema. This test validation can handle anyOf and allOf as well. See below:

```ruby

  path '/blogs/flexible' do
    post 'Creates a blog flexible body' do
      tags 'Blogs'
      description 'Creates a flexible blog from provided data'
      operationId 'createFlexibleBlog'
      consumes 'application/json'
      produces 'application/json'

      parameter name: 'blog', in: :body, schema: {
          oneOf: [
            { '$ref' => '#/components/schemas/blog' },
            { '$ref' => '#/components/schemas/flexible_blog' }
          ]
        }

      response '201', 'flexible blog created' do
        schema oneOf: [{ '$ref' => '#/components/schemas/blog' }, { '$ref' => '#/components/schemas/flexible_blog' }]
        run_test!
      end
    end
  end

```
This automatic schema validation is a powerful feature of rswag.

### Global Metadata ###

In addition to paths, operations and responses, OpenAPI also supports global API metadata. When you install rswag, a file called _openapi_helper.rb_ is added to your spec folder. This is where you define one or more OpenAPI documents and provide global metadata. Again, the format is based on OpenAPI so most of the global fields supported by the top level ["Swagger" object](http://swagger.io/specification/#swaggerObject) can be provided with each document definition. As an example, you could define an OpenAPI document for each version of your API and in each case specify a title, version string.

```ruby
# spec/openapi_helper.rb
RSpec.configure do |config|
  config.openapi_root = Rails.root.to_s + '/openapi'

  config.openapi_specs = {
    'v1/openapi.json' => {
      openapi: '3.0.1',
      info: {
        title: 'API V1',
        version: 'v1',
        description: 'This is the first version of my API'
      },
      servers: [
        {
          url: 'https://{defaultHost}',
          variables: {
            defaultHost: {
                default: 'www.example.com'
            }
          }
        }
      ]
    },

    'v2/openapi.yaml' => {
      openapi: '3.0.1',
      info: {
        title: 'API V2',
        version: 'v2',
        description: 'This is the second version of my API'
      },
      servers: [
        {
          url: '{protocol}://{defaultHost}',
          variables: {
            protocol: {
              default: :https
            },
            defaultHost: {
                default: 'www.example.com'
            }
          }
        }
      ]
    }
  }
end
```

#### Supporting multiple versions of API ####
By default, the paths, operations and responses defined in your spec files will be associated with the first OpenAPI document in _openapi_helper.rb_. If your API has multiple versions, you should be using separate documents to describe each of them. In order to assign a file with a given version of API, you'll need to add the ```openapi_spec``` tag to each spec specifying its target document name:

```ruby
# spec/requests/v2/blogs_spec.rb
describe 'Blogs API', openapi_spec: 'v2/openapi.yaml' do

  path '/blogs' do
  ...

  path '/blogs/{id}' do
  ...
end
```

#### Supporting YAML format ####

By default, the OpenAPI specs are generated in JSON format. If you want to generate them in YAML format, you can specify the OpenAPI format in the `openapi_helper.rb` file:

```ruby
# spec/openapi_helper.rb
RSpec.configure do |config|
  config.openapi_root = Rails.root.to_s + '/openapi'

  # Use if you want to see which test is running
  # config.formatter = :documentation

  # Generate OpenAPI docs in YAML format
  config.openapi_format = :yaml

  config.openapi_specs = {
    'v1/openapi.yaml' => {
      openapi: '3.0.1',
      info: {
        title: 'API V1',
        version: 'v1',
        description: 'This is the first version of my API'
      },
      servers: [
        {
          url: 'https://{defaultHost}',
          variables: {
            defaultHost: {
                default: 'www.example.com'
            }
          }
        }
      ]
    },
  }
end
```

#### Formatting the description literals: ####
OpenAPI supports the Markdown syntax to format strings. This can be especially handy if you were to provide a long description of a given API version or endpoint. Use [this guide](https://github.com/adam-p/markdown-here/wiki/Markdown-Cheatsheet) for reference.

__NOTE:__ There is one difference between the official Markdown syntax and OpenAPI interpretation, namely tables. To create a table like this:

| Column1 | Column2 |
| ------- | ------- |
| cell1   | cell2   |

you should use the following syntax, making sure there is no whitespace at the start of any of the lines:

```
&#13;
| Column1 | Column2 | &#13; |
| ------- | ------- |&#13;
| cell1   | cell2    |&#13;
&#13;
```

### Specifying/Testing API Security ###

OpenAPI allows for the specification of different security schemes and their applicability to operations in an API.
To leverage this in rswag, you define the schemes globally in _openapi_helper.rb_ and then use the "security" attribute at the operation level to specify which schemes, if any, are applicable to that operation.
OpenAPI supports :basic, :bearer, :apiKey and :oauth2 and :openIdConnect scheme types. See [the spec](https://swagger.io/docs/specification/authentication/) for more info.

```ruby
# spec/openapi_helper.rb
RSpec.configure do |config|
  config.openapi_root = Rails.root.to_s + '/openapi'

  config.openapi_specs = {
    'v1/openapi.json' => {
      ...  # note the new Open API 3.0 compliant security structure here, under "components"
      components: {
        securitySchemes: {
          basic_auth: {
            type: :http,
            scheme: :basic
          },
          api_key: {
            type: :apiKey,
            name: 'api_key',
            in: :header
          }
        }
      }
    }
  }
end

# spec/requests/blogs_spec.rb
describe 'Blogs API' do

  path '/blogs' do

    post 'Creates a blog' do
      tags 'Blogs'
      security [ basic_auth: [] ]
      ...

      response '201', 'blog created' do
        let(:request_headers) { { 'Authorization' => "Basic #{::Base64.strict_encode64('jsmith:jspass')}" } }
        run_test!
      end

      response '401', 'authentication failed' do
        let(:request_headers) { { 'Authorization' => "Basic #{::Base64.strict_encode64('bogus:bogus')}" } }
        run_test!
      end
    end
  end
end

# example of documenting an endpoint that handles basic auth and api key based security
describe 'Auth examples API' do
  path '/auth-tests/basic-and-api-key' do
    post 'Authenticates with basic auth and api key' do
      tags 'Auth Tests'
      operationId 'testBasicAndApiKey'
      security [{ basic_auth: [], api_key: [] }]

      response '204', 'Valid credentials' do
        let(:request_headers) {
          {
            'Authorization' => "Basic #{::Base64.strict_encode64('jsmith:jspass')}"
            'api_key' => 'foobar',
          }
        }
        run_test!
      end

      response '401', 'Invalid credentials' do
        let(:request_headers) {
          {
            'Authorization' => "Basic #{::Base64.strict_encode64('jsmith:jspass')}"
            'api_key' => 'bar-foo',
          }
        }
        run_test!
      end
    end
  end
end


```

__NOTE:__ Depending on the scheme types, you'll be required to assign a corresponding parameter value with each example.
For example, :basic auth is required above and so the :Authorization (header) parameter must be set accordingly

## Configuration & Customization ##

The steps described above will get you up and running with minimal setup. However, rswag offers a lot of flexibility to customize as you see fit. Before exploring the various options, you'll need to be aware of its different components. The following table lists each of them and the files that get added/updated as part of a standard install.

| Gem             | Description                                                                                                                  | Added/Updated                                        |
| --------------- | ---------------------------------------------------------------------------------------------------------------------------- | ---------------------------------------------------- |
| __rswag-specs__ | Swagger-based DSL for rspec & accompanying rake task for generating OpenAPI files                                            | _spec/openapi_helper.rb_                             |
| __rswag-api__   | Rails Engine that exposes your OpenAPI files as JSON endpoints                                                               | _config/initializers/rswag_api.rb, config/routes.rb_ |
| __rswag-ui__    | Rails Engine that includes [swagger-ui](https://github.com/swagger-api/swagger-ui) and powers it from your Swagger endpoints | _config/initializers/rswag_ui.rb, config/routes.rb_  |

### Output Location for Generated OpenAPI Files ###

You can adjust this in the _openapi_helper.rb_ that's installed with __rswag-specs__:

```ruby
# spec/openapi_helper.rb
RSpec.configure do |config|
  config.openapi_root = Rails.root.to_s + '/your-custom-folder-name'
  ...
end
```

__NOTE__: If you do change this, you'll also need to update the rswag_api.rb initializer (assuming you're using rswag-api). More on this later.

### Input Location for Rspec Tests ###

By default, rswag will search for integration tests in _spec/requests_, _spec/api_ and _spec/integration_. If you want to use tests from other locations, provide the PATTERN argument to rake:

```ruby
# search for tests in spec/openapi
rake rswag:specs:swaggerize PATTERN="spec/openapi/**/*_spec.rb"
```

### Additional rspec options

You can add additional rspec parameters using the ADDITIONAL_RSPEC_OPTS env variable:

```ruby
# Only include tests tagged "rswag"
rake rswag:specs:swaggerize ADDITIONAL_RSPEC_OPTS="--tag rswag"
```

### Referenced Parameters and Schema Definitions ###

OpenAPI allows you to describe JSON structures inline with your operation descriptions OR as referenced globals.
For example, you might have a standard response structure for all failed operations.
Rather than repeating the schema in every operation spec, you can define it globally and provide a reference to it in each spec:

```ruby
# spec/openapi_helper.rb
config.openapi_specs = {
  'v1/openapi.json' => {
    openapi: '3.0.0',
    info: {
      title: 'API V1'
    },
    components: {
      schemas: {
        errors_object: {
          type: 'object',
          properties: {
            errors: { '$ref' => '#/components/schemas/errors_map' }
          }
        },
        errors_map: {
          type: 'object',
          additionalProperties: {
            type: 'array',
            items: { type: 'string' }
          }
        },
        blog: {
          type: 'object',
          properties: {
            id: { type: 'integer' },
            title: { type: 'string' },
            content: { type: 'string', nullable: true },
            thumbnail: { type: 'string', nullable: true }
          },
          required: %w[id title]
        },
        new_blog: {
          type: 'object',
          properties: {
            title: { type: 'string' },
            content: { type: 'string', nullable: true },
            thumbnail: { type: 'string', format: 'binary', nullable: true }
          },
          required: %w[title]
        }
      }
    }
  }
}

# spec/requests/blogs_spec.rb
describe 'Blogs API' do

  path '/blogs' do

    post 'Creates a blog' do

      parameter name: 'new_blog', in: :body, schema: { '$ref' => '#/components/schemas/new_blog' }

      response 422, 'invalid request' do
        schema '$ref' => '#/components/schemas/errors_object'
  ...
end

# spec/requests/comments_spec.rb
describe 'Blogs API' do

  path '/blogs/{blog_id}/comments' do

    post 'Creates a comment' do

      response 422, 'invalid request' do
        schema '$ref' => '#/components/schemas/errors_object'
  ...
end
```

### Request examples ###

```ruby
# spec/integration/blogs_spec.rb
describe 'Blogs API' do

  path '/blogs/{blog_id}' do

    get 'Retrieves a blog' do

      request_body_example value: { some_field: 'Foo' }, name: 'request_example_1', summary: 'A request example'

      response 200, 'blog found' do
        ...
```

to use the actual request from the spec as the example:

```ruby
config.after(:each, operation: true, use_as_request_example: true) do |spec|
  spec.metadata[:operation][:request_examples] ||= []

  example = {
    value: JSON.parse(request.body.string, symbolize_names: true),
    name: 'request_example_1',
    summary: 'A request example'
  }

  spec.metadata[:operation][:request_examples] << example
end
```

### Response headers ###

In Rswag, you could use `header` method inside the response block to specify header objects for this response.
Rswag will validate your response headers with those header objects and inject them into the generated OpenAPI spec:

```ruby
# spec/requests/comments_spec.rb
describe 'Blogs API' do

  path '/blogs/{blog_id}/comments' do

    post 'Creates a comment' do

      response 422, 'invalid request' do
        header 'X-Rate-Limit-Limit', schema: { type: :integer }, description: 'The number of allowed requests in the current period'
        header 'X-Rate-Limit-Remaining', schema: { type: :integer }, description: 'The number of remaining requests in the current period'
  ...
end
```

#### Nullable or Optional Response Headers ####

You can include `nullable` or `required` to specify whether a response header must be present or may be null. When `nullable` is not included, the headers validation validates that the header response is non-null. When `required` is not included, the headers validation validates the the header response is passed.

```ruby
# spec/integration/comments_spec.rb
describe 'Blogs API' do

  path '/blogs/{blog_id}/comments' do

    get 'Gets a list of comments' do

      response 200, 'blog found' do
        header 'X-Cursor', schema: { type: :string, nullable: true }, description: 'The cursor to get the next page of comments.'
        header 'X-Per-Page', schema: { type: :integer }, required: false, description: 'The number of comments per page.'
  ...
end
```

### Response examples ###

You can provide custom response examples to the generated OpenAPI spec by calling the method `examples` inside the response block:
However, auto generated example responses are now enabled by default in rswag. See below.

```ruby
# spec/requests/blogs_spec.rb
describe 'Blogs API' do

  path '/blogs/{blog_id}' do

    get 'Retrieves a blog' do

      response 200, 'blog found' do
        example 'application/json', :example_key, {
            id: 1,
            title: 'Hello world!',
            content: '...'
          }
        example 'application/json', :example_key_2, {
            id: 1,
            title: 'Hello world!',
            content: '...'
          }, "Summary of the example", "Longer description of the example"
  ...
end
```


### Enable auto generation examples from responses ###


To enable examples generation from responses add callback above run_test! like:

```ruby
after do |example|
  content = example.metadata[:response][:content] || {}
  example_spec = {
    "application/json"=>{
      examples: {
        test_example: {
          value: JSON.parse(response.body, symbolize_names: true)
        }
      }
    }
  }
  example.metadata[:response][:content] = content.deep_merge(example_spec)
end
```

Note you will need to disable the `--dry-run` option for Rspec 3.0.0 or higher to
run the after callback. See below for more information.

#### Dry Run Option ####

The `--dry-run` option is enabled by default for Rspec 3, but if you need to
disable it you can use the environment variable `RSWAG_DRY_RUN=0` during the
generation command or add the following to your `config/environments/test.rb`:

```ruby
RSpec.configure do |config|
  config.rswag_dry_run = false
end
```

#### Running tests without documenting ####

If you want to use Rswag for testing without adding it to you OpenAPI specs, you can provide the document tag:
```ruby
describe 'Blogs API' do
  path '/blogs/{blog_id}' do
    get 'Retrieves a blog' do
      # documentation is now disabled for this response only
      response 200, 'blog found', document: false do
        ...
```

You can also reenable documentation for specific responses only:
```ruby
# documentation is now disabled
describe 'Blogs API', document: false do
  path '/blogs/{blog_id}' do
    get 'Retrieves a blog' do
      # documentation is reenabled for this response only
      response 200, 'blog found', document: true do
        ...
      end

      response 401, 'special case' do
        ...
      end
```

##### rswag helper methods #####
<!--
There are some helper methods to help with documenting request bodies.
```ruby
describe 'Blogs API', type: :request, openapi_spec: 'v1/openapi.json' do
  let(:request_headers) { { 'api_key' => 'fake_key' } }

  path '/blogs' do
    post 'Creates a blog' do
      tags 'Blogs'
      description 'Creates a new blog from provided data'
      operationId 'createBlog'
      consumes 'application/json'
      produces 'application/json'

      request_body_json schema: { '$ref' => '#/components/schemas/blog' },
                        examples: :blog

      request_body_text_plain
      request_body_xml schema: { '$ref' => '#/components/schemas/blog' }

      let(:blog) { { blog: { title: 'foo', content: 'bar' } } }

      response '201', 'blog created' do
        schema '$ref' => '#/components/schemas/blog'
        run_test!
      end

      response '422', 'invalid request' do
        schema '$ref' => '#/components/schemas/errors_object'
        let(:blog) { { blog: { title: 'foo' } } }

        run_test! do |response|
          expect(response.body).to include("can't be blank")
        end
      end
    end
  end
end
```

In the above example, we see methods ```request_body_json``` ```request_body_plain``` ```request_body_xml```.
These methods can be used to describe json, plain text and xml body. They are just wrapper methods to setup posting JSON, plain text or xml into your endpoint.
The simplest most common usage is for json formatted body to use the schema: to specify the location of the schema for the request body
and the examples: :blog which will create a named example "blog" under the "requestBody / content / application/json / examples" section.
Again, documenting request response examples changed in OpenAPI 3.0. The example above would generate a openapi.json snippet that looks like this:

```json
        ...
        {"requestBody": {
          "required": true,
          "content": {
            "application/json": {
              "examples": {
                "blog": {  // takes the name from  examples: :blog above
                  "value": {  // this is OpenAPI 3.0 structure -> https://swagger.io/docs/specification/adding-examples/
                    "blog": { // here is the actual JSON payload that is submitted to the service, and shows up in swagger UI as an example
                      "title": "foo",
                      "content": "bar"
                    }
                  }
                }
              },
              "schema": {
                "$ref": "#/components/schemas/blog"
              }
            },
            "test/plain": {
              "schema": {
                "type": "string"
              }
            },
            "application/xml": {
              "schema": {
                "$ref": "#/components/schemas/blog"
              }
            }
          }
        },
        }
```

*NOTE:* for this example request body to work in the tests properly, you need to ``let`` a variable named *blog*.
The variable with the matching name (blog in this case) is eval-ed and captured to be placed in the examples section.
This ```let``` value is used in the integration test to run the test AND captured and injected into the requestBody section.

##### rswag response examples #####

In the same way that requestBody examples can be captured and injected into the output, response examples can also be captured.
Using the above example, when the integration test is run - the OpenAPI spec would include the following snippet providing more useful real world examples
capturing the response from the execution of the integration test.

```json
       ...  "responses": {
          "201": {
            "description": "blog created",
            "content": {
              "application/json": {
                "example": {
                  "id": 1,
                  "title": "foo",
                  "content": "bar",
                  "thumbnail": null
                },
                "schema": {
                  "$ref": "#/components/schemas/blog"
                }
              }
            }
          },
          "422": {
            "description": "invalid request",
            "content": {
              "application/json": {
                "example": {
                  "errors": {
                    "content": [
                      "can't be blank"
                    ]
                  }
                },
                "schema": {
                  "$ref": "#/components/schemas/errors_object"
                }
              }
            }
          }
        }
```
 -->
### Route Prefix for OpenAPI JSON Endpoints ###

The functionality to expose OpenAPI files, such as those generated by rswag-specs, as JSON endpoints is implemented as a Rails Engine. As with any Engine, you can change its mount prefix in _routes.rb_:

```ruby
TestApp::Application.routes.draw do
  ...

  mount Rswag::Api::Engine => 'your-custom-prefix'
end
```

Assuming a OpenAPI file exists at &lt;openapi_root&gt;/v1/openapi.json, this configuration would expose the file as the following JSON endpoint:

```
GET http://<hostname>/your-custom-prefix/v1/openapi.json
```

### Root Location for OpenAPI Files ###

You can adjust this in the _rswag_api.rb_ initializer that's installed with __rspec-api__:

```ruby
Rswag::Api.configure do |c|
  c.openapi_root = Rails.root.to_s + '/your-custom-folder-name'
  ...
end
```

__NOTE__: If you're using rswag-specs to generate OpenAPI files, you'll want to ensure they both use the same &lt;openapi_root&gt;. The reason for separate settings is to maintain independence between the two gems. For example, you could install rswag-api independently and create your OpenAPI files manually.

### Dynamic Values for OpenAPI JSON ##

There may be cases where you need to add dynamic values to the OpenAPI JSON that's returned by rswag-api. For example, you may want to provide an explicit host name. Rather than hardcoding it, you can configure a filter that's executed prior to serializing every OpenAPI document:

```ruby
Rswag::Api.configure do |c|
  ...

  c.openapi_filter = lambda { |openapi, env| openapi['host'] = env['HTTP_HOST'] }
end
```

Note how the filter is passed the rack env for the current request. This provides a lot of flexibility. For example, you can assign the "host" property (as shown) or you could inspect session information or an Authorization header and remove operations based on user permissions.

### Custom Headers for OpenAPI Files ###

You can specify custom headers for serving your generated OpenAPI JSON. For example you may want to force a specific charset for the 'Content-Type' header. You can configure a hash of headers to be sent with the request:

```ruby
Rswag::Api.configure do |c|
  ...

  c.openapi_headers = { 'Content-Type' => 'application/json; charset=UTF-8' }
end
```

Take care when overriding Content-Type if you serve both YAML and JSON files as it will no longer switch the Content-Type header correctly.


### Enable Swagger Endpoints for swagger-ui ###

You can update the _rswag_ui.rb_ initializer, installed with rswag-ui, to specify which Swagger endpoints should be available to power the documentation UI. If you're using rswag-api, these should correspond to the Swagger endpoints it exposes. When the UI is rendered, you'll see these listed in a drop-down to the top right of the page:

```ruby
Rswag::Ui.configure do |c|
  c.openapi_endpoint '/api-docs/v1/openapi.json', 'API V1 Docs'
  c.openapi_endpoint '/api-docs/v2/openapi.json', 'API V2 Docs'
end
```

### Enable Simple Basic Auth for swagger-ui

You can also update the _rswag_ui.rb_ initializer, installed with rswag-ui to specify a username and password should you want to keep your documentation private.

```ruby
Rswag::Ui.configure do |c|
  c.basic_auth_enabled = true
  c.basic_auth_credentials 'username', 'password'
end
```

### Route Prefix for the swagger-ui ###

Similar to rswag-api, you can customize the swagger-ui path by changing its mount prefix in _routes.rb_:

```ruby
TestApp::Application.routes.draw do
  ...

  mount Rswag::Api::Engine => 'api-docs'
  mount Rswag::Ui::Engine => 'your-custom-prefix'
end
```

### Customizing the swagger-ui ###

The swagger-ui provides several options for customizing its behavior, all of which are documented here https://github.com/swagger-api/swagger-ui/tree/2.x#swaggerui. If you need to tweak these or customize the overall look and feel of your swagger-ui, then you'll need to provide your own version of index.html. You can do this with the following generator.

```ruby
rails g rswag:ui:custom

```

This will add a local version that you can modify at _app/views/rswag/ui/home/index.html.erb_. For example, it will let you to add your own `<title>` and favicon.

To replace the *"Swagger sponsored by"* brand image, you can add the following script to the generated file:

```html
<script>
  (function () {
  window.addEventListener("load", function () {
      setTimeout(function () {

          var logo = document.getElementsByClassName('link');

          logo[0].children[0].alt = "My API";
          logo[0].children[0].src = "/favicon.png";
      });
  }); })();
</script>
```

The above script would expect to find an image named `favicon.png` in the public folder.

You can also customize the swagger ui with _configuration objects._ Rswag's `config_object` can be used with swagger's configuration parameters found here: https://swagger.io/docs/open-source-tools/swagger-ui/usage/configuration/
The ui can be configured in the _config/initializers/rswag-ui.rb_ file like so:

```ruby
Rswag::Ui.configure do |c|
  c.config_object['defaultModelsExpandDepth'] = 1
end

```

### Serve UI Assets Directly from your Web Server

Rswag ships with an embedded version of the [swagger-ui](https://github.com/swagger-api/swagger-ui), which is a static collection of JavaScript and CSS files. These assets are served by the rswag-ui middleware. However, for optimal performance you may want to serve them directly from your web server (e.g. Apache or NGINX). To do this, you'll need to copy them to the web server root. This is the "public" folder in a typical Rails application.

```
bundle exec rake rswag:ui:copy_assets[public/api-docs]
```

__NOTE:__: The provided subfolder MUST correspond to the UI mount prefix - "api-docs" by default.


Notes to test OpenAPI output locally with Swagger editor
```
docker pull swaggerapi/swagger-editor
```
```
docker run -d -p 80:8080 swaggerapi/swagger-editor
```
This will run the swagger editor in the docker daemon and can be accessed
at ```http://localhost```. From here, you can use the UI to load the generated openapi.json to validate the output.

### Linting with RuboCop RSpec

When you lint your RSpec spec files with `rubocop-rspec`, it will fail to detect RSpec aliases that Rswag defines.
Make sure to use `rubocop-rspec` 2.0 or newer and add the following to your `.rubocop.yml`:

```yaml
inherit_gem:
  rswag-specs: .rubocop_rspec_alias_config.yml
```
 readmeEtag: '"11d1ce2d520cb69736023d1b6a072ddb2fb5ca9a"' readmeLastModified: Wed, 09 Apr 2025 20:54:02 GMT repositoryId: 29151578 description: Seamlessly adds a Swagger to Rails-based API's created: '2015-01-12T19:08:10Z' updated: '2026-02-03T00:50:07Z' language: Ruby archived: false stars: 2181 watchers: 28 forks: 458 owner: rswag logo: https://avatars.githubusercontent.com/u/55741034?v=4 license: MIT repoEtag: '"035e8dbe3f87f592411bb584f749b6317d5ab04e03bf9f162dbcf3769673f4e5"' repoLastModified: Tue, 03 Feb 2026 00:50:07 GMT foundInMaster: true v3_1_link: https://github.com/rswag/rswag/pull/714 - source: https://openapi.tools/ name: 🌿 Fern category: - Code Generators - SDK - Documentation - Server link: https://buildwithfern.com language: - TypeScript - Java - Python - Go - Ruby - C# - PHP - Swift - and Rust - Express.js - FastAPI - Spring Boot source_description: - >- Generate SDKs in popular languages and publish them to package managers (like npm). - null - >- Generate server-side boilerplate code from OpenAPI and then implement your business logic. v2: true v3: true repository: https://github.com/fern-api/fern id: 9a06b3cc23e65a92ab1f012b26743de1 repositoryMetadata: base64Readme: >- <br/>
<div align="center">
  <a href="https://www.buildwithfern.com/?utm_source=github&utm_medium=readme&utm_campaign=fern&utm_content=logo">
    <picture>
      <source media="(prefers-color-scheme: dark)" srcset="/fern/images/logo-white.svg">
      <source media="(prefers-color-scheme: light)" srcset="/fern/images/logo-primary.svg">
      <img alt="logo" src="/fern/images/logo-primary.svg" height="80" align="center">
    </picture>
  </a>
<br/>

<br/>

[![2023 Y Combinator Startup](https://img.shields.io/badge/Y%20Combinator-2023-orange)](https://www.ycombinator.com/companies/fern)
![License](https://img.shields.io/badge/license-Apache%202.0-blue)

[![Slack](https://img.shields.io/badge/slack-pink.svg)](https://join.slack.com/t/fern-community/shared_invite/zt-2dpftfmif-MuAegl8AfP_PK8s2tx350Q)
[![Documentation](https://img.shields.io/badge/Read%20our%20Documentation-black?logo=book)](https://buildwithfern.com/learn/home?utm_source=fern-api/fern/readme-read-our-documentation)

</div>

# 🌿 What is Fern?

Fern is a platform that transforms your API definitions into production-ready SDKs and beautiful documentation in minutes. 

With Fern, you can offer your users:

- 🧩 **Type-safe SDKs** in multiple languages, including TypeScript, Python, Java, Go, Ruby, PHP, and C#
- 📘 **Developer documentation** featuring an interactive UI and auto-generated API + SDK references
- ✨ **AI Search** powered by an assistant trained on your docs, APIs, and SDKs that can instantly answer a developer's questions

Fern supports leading API specifications including OpenAPI (REST, Webhooks), AsyncAPI (WebSockets), Protobuf (gRPC), and OpenRPC.

<div align="center">
    <a href="/fern/images/overview-dark.png" target="_blank">
        <picture>
            <source srcset="/fern/images/overview-dark.png" media="(prefers-color-scheme: dark)">
            <source srcset="/fern/images/overview-light.png" media="(prefers-color-scheme: light)">
            <img src="/fern/images/overview-light.png" width="700" alt="Overview Diagram">
        </picture>
    </a>
</div>

## 🌿 SDKs

The Fern platform is available via a command line interface (CLI) and requires Node 18+. To install it, run:

```bash
npm install -g fern-api
```

Initialize Fern with your OpenAPI spec:

```bash
fern init --openapi ./path/to/openapi.yml
# or
fern init --openapi https://link.buildwithfern.com/plantstore-openapi
```

Your directory should look like the following:

```yaml
fern/
├─ fern.config.json
├─ generators.yml # generators you're using
└─ openapi/
  └─ openapi.json # your openapi document
```

Finally, to invoke the generator, run:

```bash
fern generate
```

🎉 Once the command completes, you'll see your SDK in `/generated/sdks/typescript`.

## 🌿 API Documentation

Fern can also build and host a documentation website with an auto-generated API reference. Write additional pages in markdown and have them versioned with git. 
Search, SEO, dark mode, and popular components are provided out-of-the-box. Plus, you can customize the colors, font, logo, and domain name.

Check out docs built with Fern:

- [elevenlabs.io/docs](https://elevenlabs.io/docs)
- [launchdarkly.com/docs](https://launchdarkly.com/docs/home)
- [docs.hume.ai](https://docs.hume.ai/)

Get started [here](https://github.com/fern-api/docs-starter).

## 🌿 Generators

Generators are process that take your API Definition as input and output artifacts (SDKs,
Postman Collections, Server boilerplate, etc.). To add a generator, run `fern add <generator id>`.

### SDK Generators

| Generator ID                       | Latest Version                                                                                    | Entrypoint                                                                    |
| ---------------------------------- | ------------------------------------------------------------------------------------------------- | ----------------------------------------------------------------------------- |
| `fernapi/fern-typescript-sdk`      | ![Typescript Generator Version](https://img.shields.io/docker/v/fernapi/fern-typescript-sdk)      | [cli.ts](./generators/typescript/sdk/cli/src/cli.ts)                      |
| `fernapi/fern-python-sdk`          | ![Python Generator Version](https://img.shields.io/docker/v/fernapi/fern-python-sdk)              | [cli.py](./generators/python/src/fern_python/generators/sdk/cli.py)           |
| `fernapi/fern-java-sdk`            | ![Java Generator Version](https://img.shields.io/docker/v/fernapi/fern-java-sdk)                  | [Cli.java](./generators/java/sdk/src/main/java/com/fern/java/client/Cli.java) |
| `fernapi/fern-ruby-sdk`            | ![Ruby Generator Version](https://img.shields.io/docker/v/fernapi/fern-ruby-sdk)                  | [cli.ts](./generators/ruby/sdk/src/cli.ts)                                    |
| `fernapi/fern-go-sdk`              | ![Go Generator Version](https://img.shields.io/docker/v/fernapi/fern-go-sdk)                      | [main.go](./generators/go/cmd/fern-go-sdk/main.go)                            |
| `fernapi/fern-csharp-sdk`          | ![C# Generator Version](https://img.shields.io/docker/v/fernapi/fern-csharp-sdk)                  | [cli.ts](./generators/csharp/sdk/src/cli.ts)                                  |
| `fernapi/fern-php-sdk`             | ![PHP Generator Version](https://img.shields.io/docker/v/fernapi/fern-php-sdk)                    | [cli.ts](./generators/php/sdk/src/cli.ts)                                     |
| `fernapi/fern-swift-sdk`           | ![Swift Generator Version](https://img.shields.io/docker/v/fernapi/fern-swift-sdk)                | [cli.ts](./generators/swift/sdk/src/cli.ts)                                   |
| `fernapi/fern-rust-sdk`            | ![Rust Generator Version](https://img.shields.io/docker/v/fernapi/fern-rust-sdk)                  | [cli.ts](./generators/rust/sdk/src/cli.ts)                                  |

### Server-side Generators

Fern's server-side generators output boilerplate application code (models and networking logic). This is intended for spec-first or API-first developers, who write their API definition (as an OpenAPI spec or Fern definition) and want to generate backend code.

| Generator ID                      | Latest Version                                                                                                  | Entrypoint                                                                       |
| --------------------------------- | --------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------------------- |
| `fernapi/fern-typescript-express` | ![Typescript Express Server Generator Version](https://img.shields.io/docker/v/fernapi/fern-typescript-express) | [cli.ts](./generators/typescript/express/cli/src/cli.ts)                         |
| `fernapi/fern-fastapi-server`     | ![Python FastAPI Server Generator Version](https://img.shields.io/docker/v/fernapi/fern-fastapi-server)         | [cli.py](./generators/python/src/fern_python/generators/sdk/cli.py)              |
| `fernapi/fern-java-spring`        | ![Java Spring Server Generator Version](https://img.shields.io/docker/v/fernapi/fern-java-spring)               | [Cli.java](./generators/java/spring/src/main/java/com/fern/java/spring/Cli.java) |

### Model Generators

Fern's model generators will output schemas or types defined in your OpenAPI spec or Fern Definition.

| Generator ID                  | Latest Version                                                                                   | Entrypoint                                                                    |
| ----------------------------- | ------------------------------------------------------------------------------------------------ | ----------------------------------------------------------------------------- |
| `fernapi/fern-pydantic-model` | ![Pydantic Model Generator Version](https://img.shields.io/docker/v/fernapi/fern-pydantic-model) | [cli.py](./generators/python/src/fern_python/generators/sdk/cli.py)           |
| `fernapi/fern-java-model`     | ![Java Model Generator Version](https://img.shields.io/docker/v/fernapi/fern-java-model)         | [Cli.java](./generators/java/sdk/src/main/java/com/fern/java/client/Cli.java) |
| `fernapi/fern-ruby-model`     | ![Ruby Model Generator Version](https://img.shields.io/docker/v/fernapi/fern-ruby-model)         | [cli.ts](./generators/ruby/model/src/cli.ts)                                  |
| `fernapi/fern-go-model`       | ![Go Model Generator Version](https://img.shields.io/docker/v/fernapi/fern-go-model)             | [main.go](./generators/go/cmd/fern-go-model/main.go)                          |

### Spec Generators

Fern's spec generators can output an OpenAPI spec or a Postman collection.

> **Note**: The OpenAPI spec generator is primarily intended for Fern Definition users. This prevents lock-in so that one can always export to OpenAPI.

| Generator ID           | Latest Version                                                                     | Entrypoint                                |
| ---------------------- | ---------------------------------------------------------------------------------- | ----------------------------------------- |
| `fernapi/fern-openapi` | ![OpenAPI Generator Version](https://img.shields.io/docker/v/fernapi/fern-openapi) | [cli.ts](./generators/openapi/src/cli.ts) |
| `fernapi/fern-postman` | ![Postman Generator Version](https://img.shields.io/docker/v/fernapi/fern-postman) | [cli.ts](./generators/postman/src/cli.ts) |

## 🌿 CLI Commands

Here's a quick look at the most popular CLI commands. View the documentation for [all CLI commands](https://buildwithfern.com/learn/cli-api/cli-reference/commands).

`fern init`: adds a new starter API to your repository.

`fern check`: validate your API definition and Fern configuration.

`fern generate`: run the generators specified in `generators.yml` in the cloud.

`fern generate --local`: run the generators specified in `generators.yml` in docker locally.

`fern add <generator>`: include a new generator in your `generators.yml`. For example, `fern add fern-python-sdk`.

## Advanced

### API First

Fern supports developers and teams that want to be API-first or Spec-first.

Define your API, and use Fern to generate models, networking code and boilerplate application code. The generated code adds
type safety to your API implementation - if your backend doesn't implement the API correctly, it won't compile.

Frameworks currently supported:

- [Express](./generators/typescript)
- [Spring Boot](./generators/java)
- [FastAPI](./generators/python)

### Fern Definition

While we are big fans of OpenAPI, we know it isn't the _easiest_ format to read and write. If you're looking for an alternative,
give the Fern Definition a try.

Install the Fern CLI and initialize a Fern Project.

```bash
npm install -g fern-api
fern init
```

This will create the following folder structure in your project:

```yaml
fern/
├─ fern.config.json # root-level configuration
├─ generators.yml # generators you're using
└─ definition/
  ├─ api.yml  # API-level configuration
  └─ imdb.yml # endpoints, types, and errors
```

Here's what the `imdb.yml` starter file looks like:

```yaml
types:
  MovieId: string

  Movie:
    properties:
      id: MovieId
      title: string
      rating:
        type: double
        docs: The rating scale is one to five stars

  CreateMovieRequest:
    properties:
      title: string
      rating: double

service:
  auth: false
  base-path: /movies
  endpoints:
    createMovie:
      docs: Add a movie to the database
      method: POST
      path: /create-movie
      request: CreateMovieRequest
      response: MovieId

    getMovie:
      method: GET
      path: /{movieId}
      path-parameters:
        movieId: MovieId
      response: Movie
      errors:
        - MovieDoesNotExistError

errors:
  MovieDoesNotExistError:
    status-code: 404
    type: MovieId
```

Checkout open source projects that are using Fern Definitions:

- [Metriport](https://github.com/metriport/metriport/tree/develop/fern/definition)
- [Rivet](https://github.com/rivet-gg/rivet/tree/main/sdks/api/fern/definition)

## Inspiration

Fern is inspired by internal tooling built to enhance the developer experience. We stand on the shoulders of giants. While teams were responsible for building the following tools, we want to give a shout out to Mark Elliot (creator of Conjure at Palantir), Michael Dowling (creator of Smithy at AWS), and Ian McCrystal (creator of Stripe Docs).

## Community

[Join our Slack!](https://join.slack.com/t/fern-community/shared_invite/zt-2q7ev4mki-mhO5anKslwRowp4oExWf4A) We are here to answer questions and help you get the most out of Fern.

## Contributing

We welcome community contributions. For guidelines, refer to our [CONTRIBUTING.md](/CONTRIBUTING.md). To contribute to our documentation, refer to our [docs](https://github.com/fern-api/docs) repo.

### Development Environment

This repository uses [DevBox](https://www.jetify.com/devbox) for reproducible development environments. DevBox provides cross-platform support (Mac, Linux, Windows via WSL) with exact version pinning based on Nix.

To get started:

```bash
# Install DevBox (https://www.jetify.com/devbox/docs/installing_devbox/)
curl -fsSL https://get.jetify.com/devbox | bash

# Enter the development environment
devbox shell
```

DevBox automatically installs all required dependencies including Node.js, pnpm, Go, Python, Poetry, JDK, and buf with pinned versions matching CI.

![Fern Contributors](https://contrib.rocks/image?repo=fern-api/fern)
 readmeEtag: '"0e9e04b35a2450d798deb7ab092ee26686bbf82f"' readmeLastModified: Fri, 16 Jan 2026 03:24:36 GMT repositoryId: 476820102 description: Input OpenAPI. Output SDKs and Docs. created: '2022-04-01T17:56:54Z' updated: '2026-02-06T03:21:46Z' language: TypeScript archived: false stars: 3495 watchers: 17 forks: 277 owner: fern-api logo: https://avatars.githubusercontent.com/u/102944815?v=4 license: Apache-2.0 repoEtag: '"84735dee9f9411a3e7edaad9a2b057d20e0af5e6d5d4ccc148899c8e7ffca108"' repoLastModified: Fri, 06 Feb 2026 03:21:46 GMT foundInMaster: true sponsored: true testimonial: > I'm a huge fan of Fern. It's a force multiplier for API teams - Mike Bifulco, APIs You Won't Hate cofounder v3_1: true sponsoredDate: - 2023-11-01T06:00:00.000Z - 2023-11-01T06:00:00.000Z - 2023-11-01T06:00:00.000Z - source: https://openapi.tools/ name: ✨ Stainless sponsored: true sponsoredDate: 2024-01-01T06:00:00.000Z category: - SDK - Parsers repository: https://github.com/stainless-api/stl-api link: https://stainlessapi.com language: - TypeScript - Python - Go - Java - and Kotlin source_description: >- Generate SDKs in popular languages and publish them to package managers (like npm). v2: false v3: true v3_1: true id: 585753eb5cd634d1a8d2d02cff7e371e repositoryMetadata: base64Readme: >- # Stainless: a framework for robust & polished REST APIs

Stainless helps you ship quality, typesafe REST APIs from any TypeScript backend.

You declare the shape and behavior of your API in one place,
and get an OpenAPI spec, docs, and typed frontend client without a build step.

You can use it as a pluggable, batteries-included web framework for APIs (managing auth, pagination, observability, etc) or sprinkle it on top of your existing API in any framework for better OpenAPI support and/or full-stack typesafety.

You can also opt into Stainless's Stripe-inspired [pristine API design conventions](#pristine) and get rich pagination, consistent errors, field inclusion & selection, and (WIP) normalized caching on the frontend for free.

Stainless draws inspiration with gratitude from tRPC, FastAPI, GraphQL/Relay, and (heavily) from the internal API Framework we worked on at Stripe.

> **Note**
> This project is currently in its **alpha stage of development**.
> Features may be changed or removed at any time without warning, and production use is not recommended.
> Hobbyists welcome!

For example:

<!-- TODO: this is too long / too much code… -->

```ts
// server.ts
import { Stl } from "stainless";
import { makePrismaPlugin } from "@stl-api/prisma";
import { makeNextPlugin } from "@stl-api/next";
import { makeNextAuthPlugin } from "@stl-api/next-auth";
import { authOptions } from "~/pages/api/auth/[...nextauth]";
import prisma from "~/libs/prisma";
import { z } from "stainless";

const plugins = {
  next: makeNextPlugin(),
  nextAuth: makeNextAuthPlugin({ authOptions }),
  prisma: makePrismaPlugin(),
};

export const stl = new Stl({
  plugins,
});

const User = z.object({
  id: z.string(),
  name: z.string(),
});

const update = stl.endpoint({
  endpoint: "POST /users/:id",
  description: "Update a user. Currently only updating the name is supported.",
  authenticated: true,
  response: User,

  path: z.object({
    id: z.string(),
  }),
  body: z.object({
    name: z.string(),
  }),

  async handler({ id, name }) {
    return await prisma.users.updateOne(id, { name });
  },
});

export const api = stl.api({
  openapi: {
    endpoint: "GET /api/openapi",
  },
  resources: {
    users: stl.resource({
      actions: {
        update,
      },
      models: {
        User,
      },
    }),
  },
});

// client.ts
import { createClient } from "stainless";
import type { api } from "./server";

const client = createClient<typeof api>("http://localhost:3000/api");

// params are fully typed:
const user = await client.users.update("id", { name: "my new name" });
// user object is fully typed.

// A full OpenAPI spec is available by default at "get /openapi":
const openapi = await client.getOpenapi();
console.log(openapi.paths["/users/:id"].post);
console.log(openapi.components.schemas.User);
```

# Getting started

See [`stainless` package docs](/packages/stainless/README.md#getting-started) to get started!

# Packages

- [stainless](/packages/stainless)
- [@stl-api/express](/packages/express)
- [@stl-api/next](/packages/next)
- [@stl-api/next-auth](/packages/next-auth)
- [@stl-api/prisma](/packages/prisma)

# Pristine

Pristine is an API Standard by Stainless, providing opinions on API design so teams don't have to bikeshed, and so tools can expect consistent API shapes.

Following the Pristine Standard helps your API offer an interface like Stripe's,
with best-practices baked in. Like the Relay standard for GQL, Pristine can also help tooling like frontend clients cache data, paginate requests, handle errors, and so on.

<!-- You can opt your API into Pristine like so:

```ts
const stl = new Stainless({ pristine: true });
```

This will enforce the Pristine conventions, and provide easier access to tooling which depends on it.

If you're starting with an existing API and don't want to go straight to a v2,
you can gradually adopt Pristine standards and tooling. In the future, we plan to offer lint rules for your OpenAPI spec and an overall "Lighthouse score" indicating the degree of compliance your API offers.
 -->

## Pristine Conventions

Here is a list of Pristine API design conventions:

- [Pagination](/packages/stainless/docs/pagination.md)
- [Inclusion](/packages/stainless/docs/inclusion.md)
- [Selection](/packages/stainless/docs/selection.md)

# Using Stainless in an existing codebase

If you'd like a maintainable way of declaring your OpenAPI spec
in TypeScript, right alongside your application code, and getting
great docs, end-to-end typesafety, and backend API client libraries (SDKs),
you can adopt the `stainless` library gradually in minimally-invasive ways.

For example, in an Express app, you can add annotations near a handler to get an OpenAPI spec and client:

```ts
// app/routes/users.ts

const User = z.object({
  id: z.string(),
  name: z.string(),
});

const create = stl.endpoint({
  endpoint: "POST /users",
  response: User,
  body: z.object({ name: z.string() }),
});

app.post("/users", async (req, rsp) => {
  const user = await db.users.create({ name });
  rsp.status(200).json(user);
});
```

<summary>You'll also need a small amount of server code</summary>

```ts
// app/api.ts
const users = stl.resource({
  models: { User },
  actions: {
    create,
  },
});

const api = stl.api({
  resources: {
    users,
  },
});

// and voila, you get an OpenAPI spec!
app.get("/openapi", (req, rsp) => {
  rsp.json(api.openapi.spec);
});
```

<summary>
For typesafety and validation of parameters, you can also sprinkle in
param parsing, response generation, and more:
</summary>

```ts
app.post("/users", async (req, rsp) => {
  const { name } = create.validateParams(req);

  const user = await db.users.create({ name });

  rsp.send(create.makeResponse(user));
});
```

Doing this helps TypeScript ensure that your OpenAPI spec is an accurate reflection of your runtime behavior. It can also help return consistent response shapes and validation error messages to the user.

Note that `validateParams` raises `BadRequestError` if params don't match.

To handle errors like this, add middleware:

```ts
app.use((err, req, rsp, next) => {
  if (err instanceof stl.Error) {
    rsp.status(err.statusCode).send(stl.makeError(err));
  }
  // …
});
```

`stl.makeError` is will return a JSON object with `type`, `message`, and other information. (TODO add/encourage things like request ID's…)
 readmeEtag: '"a89884d04ce73d65e3e381b34c408a5d4e33f17e"' readmeLastModified: Mon, 29 Apr 2024 17:35:13 GMT repositoryId: 644096733 description: Stainless full-stack API Framework created: '2023-05-22T20:07:58Z' updated: '2026-02-05T23:09:49Z' language: TypeScript archived: false stars: 151 watchers: 7 forks: 7 owner: stainless-api logo: https://avatars.githubusercontent.com/u/88061651?v=4 repoEtag: '"57ed128205ae0625347bbd59ed0a17386444a5c8be0ea41ad75d0a275635d460"' repoLastModified: Thu, 05 Feb 2026 23:09:49 GMT foundInMaster: true - source: https://openapi.tools/ name: // Speakeasy sponsored: true sponsoredDate: 2024-02-09T06:00:00.000Z category: - SDK - Documentation repository: https://github.com/speakeasy-api link: https://speakeasyapi.dev/ language: - TypeScript - Python - Go - Java - Terraform - C# - PHP - Ruby - Swift - Unity source_description: >- Generate & publish SDKs in 10+ languages, Terraform Providers, and docs from your OpenAPI v3: true v3_1: true id: e9c581a8a3595b87ec6837e848852b65 foundInMaster: true - source: https://openapi.tools/ name: Swift OpenAPI Generator category: - SDK - Server language: Swift repository: https://github.com/apple/swift-openapi-generator link: https://github.com/apple/swift-openapi-generator source_description: >- Generate Swift client and server code from an OpenAPI document. Includes support for type-safe JSON event streaming, multipart, Swift concurrency, customizable middlewares, and pluggable HTTP libraries. v2: false v3: true v3_1: true id: 8c9fc3b3b20c0629891b754a2e3c79e8 repositoryMetadata: base64Readme: >- # Swift OpenAPI Generator

[![](https://img.shields.io/badge/sswg-incubating-yellow.svg)](https://www.swift.org/sswg/)
[![](https://img.shields.io/badge/docc-read_documentation-blue)](https://swiftpackageindex.com/apple/swift-openapi-generator/documentation)
[![](https://img.shields.io/github/v/release/apple/swift-openapi-generator)](https://github.com/apple/swift-openapi-generator/releases)
[![](https://img.shields.io/endpoint?url=https%3A%2F%2Fswiftpackageindex.com%2Fapi%2Fpackages%2Fapple%2Fswift-openapi-generator%2Fbadge%3Ftype%3Dswift-versions)](https://swiftpackageindex.com/apple/swift-openapi-generator)

Generate Swift client and server code from an OpenAPI document.

## Overview

[OpenAPI][openapi] is a specification for documenting HTTP services. An OpenAPI document is written in either YAML or JSON, and can be read by tools to help automate workflows, such as generating the necessary code to send and receive HTTP requests.

Swift OpenAPI Generator is a Swift package plugin that can generate the ceremony code required to make API calls, or implement API servers.

The code is generated at build-time, so it's always in sync with the OpenAPI document and doesn't need to be committed to your source repository.

## Features

- Works with OpenAPI Specification versions 3.0 and 3.1 and has preliminary support for version 3.2.
- Streaming request and response bodies enabling use cases such as JSON event streams, and large payloads without buffering.
- Support for JSON, multipart, URL-encoded form, base64, plain text, and raw bytes, represented as value types with type-safe properties.
- Client, server, and middleware abstractions, decoupling the generated code from the HTTP client library and web framework.

To see these features in action, check out the list of [example projects][examples-generator].

## Usage

Swift OpenAPI Generator can be used to generate API clients and server stubs.

Below you can see some example code, or you can follow one of the [step-by-step tutorials][tutorials-generator].

### Using a generated API client

The generated `Client` type provides a method for each HTTP operation defined in the OpenAPI document[^example-openapi-yaml] and can be used with any HTTP library that provides an implementation of `ClientTransport`.

```swift
import OpenAPIURLSession
import Foundation

let client = Client(
    serverURL: URL(string: "http://localhost:8080/api")!,
    transport: URLSessionTransport()
)
let response = try await client.getGreeting()
print(try response.ok.body.json.message)
```

### Using generated API server stubs

To implement a server, define a type that conforms to the generated `APIProtocol`, providing a method for each HTTP operation defined in the OpenAPI document[^example-openapi-yaml].

The server can be used with any web framework that provides an implementation of `ServerTransport`, which allows you to register your API handlers with the HTTP server.

```swift
import OpenAPIRuntime
import OpenAPIVapor
import Vapor

struct Handler: APIProtocol {
    func getGreeting(_ input: Operations.GetGreeting.Input) async throws -> Operations.GetGreeting.Output {
        let name = input.query.name ?? "Stranger"
        return .ok(.init(body: .json(.init(message: "Hello, \(name)!"))))
    }
}

@main struct HelloWorldVaporServer {
    static func main() async throws {
        let app = try await Application.make()
        let transport = VaporTransport(routesBuilder: app)
        let handler = Handler()
        try handler.registerHandlers(on: transport, serverURL: URL(string: "/api")!)
        try await app.execute()
    }
}
```

## Package ecosystem

The Swift OpenAPI Generator project is split across multiple repositories to enable extensibility and minimize dependencies in your project.

| Repository                                                 | Description                                        |
| ----------                                                 | -----------                                        |
| [apple/swift-openapi-generator][repo-generator]            | Swift package plugin and CLI                       |
| [apple/swift-openapi-runtime][repo-runtime]                | Runtime library used by the generated code         |
| [apple/swift-openapi-urlsession][repo-urlsession]          | `ClientTransport` using [URLSession][urlsession]   |
| [swift-server/swift-openapi-async-http-client][repo-ahc]   | `ClientTransport` using [AsyncHTTPClient][ahc]     |
| [vapor/swift-openapi-vapor][repo-vapor]                    | `ServerTransport` using [Vapor][vapor]             |
| [hummingbird-project/swift-openapi-hummingbird][repo-hb]   | `ServerTransport` using [Hummingbird][hb]          |
| [awslabs/swift-openapi-lambda][repo-lambda]                | `ServerTransport` using [AWS Lambda][lambda]       |

## Requirements and supported features

| Generator versions | Supported OpenAPI versions  |
| ------------------ | --------------------------- |
| `1.0.0` ... `main` | 3.0, 3.1, 3.2 (preliminary) |

See also [Supported OpenAPI features][supported-openapi-features].

### Supported platforms and minimum versions

The generator is used during development and is supported on macOS, Linux, and Windows.

The generated code, runtime library, and transports are supported on more
platforms, listed below.

| Component                           | macOS     | Linux, Windows | iOS    | tvOS   | watchOS | visionOS | Android |
| ----------------------------------: | :---      | :------------  | :-     | :--    | :-----  | :------  | :------ |
| Generator plugin and CLI            | ✅ 10.15+ | ✅              | ✖️     | ✖️     | ✖️      | ✖️        | ✖️      |
| Generated code and runtime library  | ✅ 10.15+ | ✅              | ✅ 13+ | ✅ 13+ | ✅ 6+   | ✅ 1+     | ✅      |

## Documentation and example projects

To get started, check out the [documentation][docs-generator], which contains
[step-by-step tutorials][tutorials-generator].

You can also experiment with [example projects][examples-generator] that use
Swift OpenAPI Generator and integrate with other packages in the ecosystem.

Or if you prefer to watch a video, check out [Meet Swift OpenAPI
Generator](https://developer.apple.com/wwdc23/10171) from WWDC23.

[openapi]: https://openapis.org
[repo-generator]: https://github.com/apple/swift-openapi-generator
[docs-generator]: https://swiftpackageindex.com/apple/swift-openapi-generator/documentation
[tutorials-generator]: https://swiftpackageindex.com/apple/swift-openapi-generator/tutorials/swift-openapi-generator
[supported-openapi-features]: https://swiftpackageindex.com/apple/swift-openapi-generator/documentation/swift-openapi-generator/supported-openapi-features
[examples-generator]: https://github.com/apple/swift-openapi-generator/blob/main/Examples/README.md
[repo-runtime]: https://github.com/apple/swift-openapi-runtime
[repo-urlsession]: https://github.com/apple/swift-openapi-urlsession
[urlsession]: https://developer.apple.com/documentation/foundation/urlsession
[repo-ahc]: https://github.com/swift-server/swift-openapi-async-http-client
[ahc]: https://github.com/swift-server/async-http-client
[repo-vapor]: https://github.com/vapor/swift-openapi-vapor
[vapor]: https://github.com/vapor/vapor
[repo-hb]: https://github.com/hummingbird-project/swift-openapi-hummingbird
[hb]: https://github.com/hummingbird-project/hummingbird
[repo-lambda]: https://github.com/awslabs/swift-openapi-lambda
[lambda]: https://docs.aws.amazon.com/lambda/latest/dg/welcome.html
[^example-openapi-yaml]: <details><summary>Example OpenAPI document (click to expand)</summary>

    ```yaml
    openapi: '3.1.0'
    info:
      title: GreetingService
      version: 1.0.0
    servers:
      - url: https://example.com/api
        description: Example service deployment.
    paths:
      /greet:
        get:
          operationId: getGreeting
          parameters:
            - name: name
              required: false
              in: query
              description: The name used in the returned greeting.
              schema:
                type: string
          responses:
            '200':
              description: A success response with a greeting.
              content:
                application/json:
                  schema:
                    $ref: '#/components/schemas/Greeting'
    components:
      schemas:
        Greeting:
          type: object
          description: A value with the greeting contents.
          properties:
            message:
              type: string
              description: The string representation of the greeting.
          required:
            - message
    ```
    </details>
 readmeEtag: '"c6f2863a66c1f46cb9332a1b677a4c0e02389e6b"' readmeLastModified: Thu, 11 Dec 2025 01:51:52 GMT repositoryId: 630566794 description: Generate Swift client and server code from an OpenAPI document. created: '2023-04-20T16:50:53Z' updated: '2026-01-30T00:10:47Z' language: Swift archived: false stars: 1834 watchers: 114 forks: 153 owner: apple logo: https://avatars.githubusercontent.com/u/10639145?v=4 license: Apache-2.0 repoEtag: '"fe4e8a0ac4a0f12b4035a87291a37c4a2727e17876227007555a1c923d1c951f"' repoLastModified: Fri, 30 Jan 2026 00:10:47 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/pb33f/wiretap v3: true id: 2dbdc7a84f951a79e04f72967a3342be repositoryMetadata: base64Readme: >- IyB3aXJldGFwCgohW2xvZ29dKC5naXRodWIvd2lyZXRhcC1oZXJvLnBuZykKClshW2Rpc2NvcmRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZGlzY29yZC85MjMyNTgzNjM1NDA4MTU5MTIpXShodHRwczovL2Rpc2NvcmQuZ2cveDdWQUNWdUVHUCkKWyFbR2l0SHViIGRvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvZG93bmxvYWRzL3BiMzNmL3dpcmV0YXAvdG90YWw/bGFiZWw9Z2l0aHViJTIwZG93bmxvYWRzJnN0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL3dpcmV0YXAvcmVsZWFzZXMpClshW25wbV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0vQHBiMzNmL3dpcmV0YXA/c3R5bGU9ZmxhdC1zcXVhcmUmbGFiZWw9bnBtJTIwZG93bmxvYWRzKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQHBiMzNmL3dpcmV0YXApClshW0RvY2tlciBQdWxsc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kb2NrZXIvcHVsbHMvcGIzM2Yvd2lyZXRhcD9zdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9wYjMzZi93aXJldGFwKQoKQSBsb2NhbCBhbmQgcGlwZWxpbmUgYmFzZWQgdG9vbCB0byBzbmlmZiBBUEkgcmVxdWVzdCBhbmQgcmVzcG9uc2VzIGZyb20gY2xpZW50cyBhbmQgc2VydmVycwp0byBkZXRlY3QgT3BlbkFQSSBjb250cmFjdCB2aW9sYXRpb25zIGFuZCBjb21wbGlhbmNlLgoKQSBzaGlmdCBsZWZ0IHRvb2wsIGZvciB0aG9zZSB3aG8gd2FudCB0byBrbm93IGlmIHRoZWlyIGFwcGxpY2F0aW9ucwphcmUgYWN0dWFsbHkgY29tcGxpYW50IHdpdGggYW4gQVBJLgoKPiBUaGlzIGlzIGFuIGVhcmx5IHRvb2wgYW5kIGluIGFjdGl2ZSBkZXZlbG9wbWVudCwgV2h5IG5vdCB0cnkgaXQgb3V0IGFuZCBnaXZlIHVzIHNvbWUgZmVlZGJhY2s/CgohW10oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL3dpcmV0YXAvYmxvYi9tYWluLy5naXRodWIvYXNzZXRzL3dpcmV0YXAtcHJldmlldy5naWYpCgotLS0KIyBSZWFkIHRoZSBxdWlja3N0YXJ0IGd1aWRlCgpb8J+agCBRdWljayBTdGFydCBHdWlkZSDwn5qAXShodHRwczovL3BiMzNmLmlvL3dpcmV0YXAvcXVpY2tzdGFydC8pCgotLS0KIyBJbnN0YWxsIHdpcmV0YXAgZm9yIHlvdXIgcGxhdGZvcm0KCiMjIEluc3RhbGxpbmcgdXNpbmcgaG9tZWJyZXcKClRoZSBlYXNpZXN0IHdheSB0byBpbnN0YWxsIGB3aXJldGFwYCBpcyB0byB1c2UgKipbaG9tZWJyZXddKGh0dHBzOi8vYnJldy5zaCkqKiBpZiB5b3UncmUgb24gT1NYIG9yIExpbnV4LgoKV2UgaGF2ZSBvdXIgb3duIHRhcCBhdmFpbGFibGUgdGhhdCBnaXZlcyB0aGUgbGF0ZXN0IGFuZCBncmVhdGVzdCB2ZXJzaW9uLgoKYGBgc2hlbGwKYnJldyBpbnN0YWxsIHBiMzNmL3RhcHMvd2lyZXRhcApgYGAKCi0tLQoKIyMgSW5zdGFsbGluZyB1c2luZyBucG0gb3IgeWFybgoKQnVpbGRpbmcgYSBKYXZhU2NyaXB0IC8gVHlwZVNjcmlwdCBhcHBsaWNhdGlvbj8gTm8gcHJvYmxlbSwgZ3JhYiB5b3VyIGNvcHkgb2YgYHdpcmV0YXBgIHVzaW5nIHlvdXIgcHJlZmVyZW5jZQpvZiAqKlt5YXJuXShodHRwczovL3lhcm5wa2cuY29tLykqKiBvciAqKltucG1dKGh0dHBzOi8vbnBtanMuY29tKSoqLgoKYGBgc2hlbGwKeWFybiBhZGQgZ2xvYmFsIEBwYjMzZi93aXJldGFwCmBgYAoKb3IuLi4KCmBgYHNoZWxsCm5wbSAtaSAtZyBAcGIzM2Yvd2lyZXRhcApgYGAKCi0tLQoKIyMgSW5zdGFsbGluZyB1c2luZyBjVVJMCgpEbyB5b3Ugd2FudCB0byB1c2UgYHdpcmV0YXBgIGluIGEgbGludXggb25seSBvciBDSS9DRCBwaXBlbGluZSBvciB3b3JrZmxvdz8gT3IgeW91IGRvbid0IHdhbnQgdG8vY2FuJ3QgdXNlCmEgcGFja2FnZSBtYW5hZ2VyIGxpa2UgYnJldz8KCk5vIHByb2JsZW0uIFVzZSAqKmNVUkwqKiB0byBkb3dubG9hZCBhbmQgcnVuIG91ciBpbnN0YWxsZXIgc2NyaXB0LgoKYGBgc2hlbGwKY3VybCAtZnNTTCBodHRwczovL3BiMzNmLmlvL3dpcmV0YXAvaW5zdGFsbC5zaCB8IHNoCmBgYAoKLS0tCgojIyBJbnN0YWxsaW5nL3J1bm5pbmcgdXNpbmcgRG9ja2VyCgpMb3ZlIGNvbnRhaW5lcnM/IERvbid0IHdhbnQgdG8gaW5zdGFsbCBhbnl0aGluZz8gTm8gcHJvYmxlbSwgdXNlIG91ciBEb2NrZXIgaW1hZ2UuCgpgYGBzaGVsbApkb2NrZXIgcHVsbCBwYjMzZi93aXJldGFwCmBgYAoKYGBgCmRvY2tlciBydW4gLXAgOTA5MDo5MDkwIC1wIDkwOTE6OTA5MSAtcCA5MDkyOjkwOTIgLS1ybSAtdiAgXAogICAgJFBXRDovd29yazpydyBwYjMzZi93aXJldGFwIC11IGh0dHBzOi8vc29tZWhvc3RvdXR0aGVyZS5jb20KYGBgCgpXZSBlbmFibGUgdGhlIGZvbGxvd2luZyBkZWZhdWx0IHBvcnRzIGA5MDkwYCwgYDkwOTFgLCBhbmQgYDkwOTJgIGZvciB0aGUgZGFlbW9uLCBtb25pdG9yLCBhbmQgd2Vic29ja2V0cyB1c2VkCmJ5IFtyYW5jaF0oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL3JhbmNoKSByZXNwZWN0aXZlbHkuCgotLS0KCiMjIEluc3RhbGxpbmcgb24gV2luZG93cwoKVG8gZ3JhYiB5b3VyIGNvcHkgb2YgYHdpcmV0YXBgIGZvciBXaW5kb3dzLCB5b3UgY2FuIHB1bGwgaXQgZnJvbSB0aGUKKipbbGF0ZXN0IHJlbGVhc2VzIG9uIGdpdGh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL3dpcmV0YXAvcmVsZWFzZXMpKioKYW5kIGRvd25sb2FkIHRoZSBXaW5kb3dzIHZlcnNpb24gZm9yIHlvdXIgQ1BVIHR5cGUuCgotLS0KCiMgUnVubmluZyB3aXJldGFwCgpUbyBnZXQgdXAgYW5kIHJ1bm5pbmcgd2l0aCB0aGUgYWJzb2x1dGUgZGVmYXVsdHMgKHdoaWNoIGlzIHRvIHNuaWZmIGFsbCB0cmFmZmljIG9uIHBvcnQgOTA5MCkKYW5kIHByb3h5IHRvIGBodHRwczovL2FwaS5wYjMzZi5jb21gIHlvdSBjYW4gcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZC4KCmBgYHNoZWxsCndpcmV0YXAgLXUgaHR0cHM6Ly9hcGkucGIzM2YuY29tCmBgYAoKIyMgQWRkaW5nIGFuIE9wZW5BUEkgY29udHJhY3QKCmBgYHNoZWxsCndpcmV0YXAgLXUgaHR0cHM6Ly9hcGkucGIzM2YuY29tIC1zIG15LW9wZW5hcGktc3BlYy55YW1sCmBgYAoKIyBEb2N1bWVudGF0aW9uCgotIPCfmoAgW1F1aWNrIFN0YXJ0XShodHRwczovL3BiMzNmLmlvL3dpcmV0YXAvcXVpY2tzdGFydC8pIPCfmoAKLSBbSW5zdGFsbGluZ10oaHR0cHM6Ly9wYjMzZi5pby93aXJldGFwL3F1aWNrc3RhcnQvKQotIFtDb25maWd1cmluZ10oaHR0cHM6Ly9wYjMzZi5pby93aXJldGFwL2NvbmZpZ3VyaW5nLykKLSBbTW9uaXRvciBVSV0oaHR0cHM6Ly9wYjMzZi5pby93aXJldGFwL21vbml0b3IvKQotIFtTZXJ2aW5nIHN0YXRpYyBjb250ZW50XShodHRwczovL3BiMzNmLmlvL3dpcmV0YXAvc3RhdGljLWNvbnRlbnQvKQotIFtHaWZ0U2hvcCBleGFtcGxlIEFQSV0oaHR0cHM6Ly9wYjMzZi5pby93aXJldGFwL2dpZnRzaG9wLWFwaS8pCi0gW0NvbnRyaWJ1dGluZ10oaHR0cHM6Ly9wYjMzZi5pby93aXJldGFwL2NvbnRyaWJ1dGluZy8pCgo= readmeEtag: '"696e61108aedfbfbb4a75332f5ed6eac2e5e3baa"' readmeLastModified: Thu, 08 Aug 2024 13:12:05 GMT repositoryId: 634974410 description: >- The world's coolest API Validation and compliance tool. Validate APIs against OpenAPI specifications and much more created: '2023-05-01T17:19:15Z' updated: '2026-01-12T10:12:50Z' language: Go archived: false stars: 205 watchers: 2 forks: 26 owner: pb33f logo: https://avatars.githubusercontent.com/u/104016643?v=4 license: NOASSERTION repoEtag: '"1dd4b7398834c7813447fab1a5877db102efc37187bdda6d5775fb192b735b1c"' repoLastModified: Mon, 12 Jan 2026 10:12:50 GMT category: - Testing Tools - Testing - Data Validators - Gateway foundInMaster: true name: wiretap link: https://pb33f.io/wiretap/ language: go source_description: >- Wiretap is a feature packed OpenAPI compliance testing tool. Validate that a client and server are compliant with an OpenAPI spec. Rewrite paths, inject headers, and even host a UI locally to test an OpenAPI implementation. v2: true v3_1: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/seriousme/openapi-schema-validator v3: true repositoryMetadata: base64Readme: >- # OpenAPI schema validator

[![CI status](https://github.com/seriousme/openapi-schema-validator/workflows/Node.js%20CI/badge.svg)](https://github.com/seriousme/openapi-schema-validator/actions?query=workflow%3A%22Node.js+CI%22)
[![Coverage Status](https://codecov.io/github/seriousme/openapi-schema-validator/graph/badge.svg?token=2LJKNGU0N6)](https://codecov.io/github/seriousme/openapi-schema-validator)
[![NPM version](https://img.shields.io/npm/v/@seriousme/openapi-schema-validator.svg)](https://www.npmjs.com/package/@seriousme/openapi-schema-validator)
![npm](https://img.shields.io/npm/dm/@seriousme/openapi-schema-validator)
[![Tested on APIs.guru](https://api.apis.guru/badges/tested_on.svg)](https://apis.guru/browse-apis/)

A JSON schema validator for [OpenAPI](https://www.openapis.org/) specifications,
it currently supports:

- [2.0](https://spec.openapis.org/oas/v2.0)
- [3.0.x](https://spec.openapis.org/oas/v3.0.3)
- [3.1.x](https://spec.openapis.org/oas/v3.1.0)
- [3.2.x](https://spec.openapis.org/oas/v3.2.0)

Tested on [over 2,000 real-world APIs](https://apis.guru) from AWS,
Microsoft, Google etc.

<a name="install"></a>

## Install

```bash
npm install @seriousme/openapi-schema-validator
```

<a name="Usage"></a>

## Usage

This module is ESM only, if you need to use commonJS please see below.

```javascript
// ESM
import { Validator } from "@seriousme/openapi-schema-validator";

console.log(Validator.supportedVersions.has("3.2"));
// prints true

const validator = new Validator();
const res = await validator.validate("./petstore.json");
const specification = validator.specification;
// specification now contains a Javascript object containing the specification
if (res.valid) {
  console.log("Specification matches schema for version", validator.version);
  const schema = validator.resolveRefs();
  // schema now contains a Javascript object containing the dereferenced schema
} else {
  console.log("Specification does not match Schema");
  console.log(res.errors);
}
```

This module can be used in CommonJS code via:

```javascript
// CommonJS
const { Validator } = await (import("@seriousme/openapi-schema-validator"));
```

See also the [upgrading guide](UPGRADING.md) if you come from a previous major version.

<a name="CLI"></a>

### CLI for API validation

Run with global install:

```bash
npm install @seriousme/openapi-schema-validator -g
validate-api <filename>
```

Run without install:

```bash
npx -p @seriousme/openapi-schema-validator validate-api <filename>
```

Where `<filename>` refers to a YAML or JSON file containing the specification.

### CLI for API bundling

To make it easier to combine multiple YAML or JSON files into a single specification file there is the `bundle-api` command.
(see the [validateBundle](#validateBundle) section below)

```bash
npm install @seriousme/openapi-schema-validator -g
bundle-api <specFiles> 
```

Run without install:

```bash
npx -p @seriousme/openapi-schema-validator bundle-api <spec files> 
```

The output will be a validated JSON specification.
Options:
  -o  --output <filename>   the filename to save the output to, default is STDOUT.
  -t  --type  [JSON|YAML]   the output format, default is JSON.

<a name="API"></a>

## API

- [`new Validator(ajvOptions)`](#newValidator)
- [`<instance>.validate(specification)`](#validate)
- [`<instance>.specification`](#specification)
- [`<instance>.version`](#version)
- [`<instance>.resolveRefs(options)`](#resolveRefs)
- [`<instance>.addSpecRef(subSpecification, uri)`](#addSpecRef)
- [`<instance>.validateBundle([specification,subspecification, ...])`](#validateBundle)
- [`Validator.supportedVersions`](#supportedVersions)

<a name="newValidator"></a>

### `new Validator(ajvOptions)`

The constructor returns an instance of `Validator`. By passing an ajv options
object it is possible to influence the behavior of the
[AJV schema validator](https://ajv.js.org/). AJV fails to process the openApi
schemas if you set `strict:true` therefore this set to `false` if present. This
is not a bug but a result of the complexity of the openApi JSON schemas.

<a name="validate"></a>

### `<instance>.validate(specification)`

This function tries to validate a specification against the OpenApi schemas.
`specification` can be one of:

- a JSON object
- a JSON object encoded as string
- a YAML string
- a filename

External references are _not_ automatically resolved so you need to inline them
yourself if required e.g by using `<instance>.addSpecRef()` The result is an
object:

```
{
  valid: <boolean>,
  errors: <any>  // only present if valid is false
}
```

<a name="specification"></a>

### `<instance>.specification`

If the validator managed to extract data from the specification parameter then
the extracted specification is available in this property as javascript object.
E.g. if you supplied a filename of a YAML file and the file was successfully read
and its YAML decoded then the contents is here. Even if validation failed.

<a name="version"></a>

### `<instance>.version`

If validation is successful this will return the openApi version found e.g.
("2.0","3.0","3.1","3.2"). The openApi specification only specifies major/minor
versions as separate schemas. So "3.0.3" results in "3.0".

<a name="resolveRefs"></a>

### `<instance>.resolveRefs(options)`

This function tries to resolve all internal references. External references are
_not_ automatically resolved so you need to inline them yourself if required e.g
by using `<instance>.addSpecRef()`. By default it will use the last
specification passed to `<instance>.validate()` but you can explicitly pass a
specification by passing `{specification:<object>}` as options. The result is an
`object` where all references have been resolved. Resolution of references is
`shallow` This should normally not be a problem for this use case.

<a name="addSpecRef"></a>

### `<instance>.addSpecRef(subSpecification, uri)`

`subSpecification` can be one of:

- a JSON object
- a JSON object encoded as string
- a YAML string
- a filename

`uri` must be a string (e.g. `http://www.example.com/subspec`), but
  is not required if the subSpecification holds a `$id` attribute at top level.
  If you specify a value for `uri` it will overwrite the definition in the `$id`
  attribute at top level.

Sometimes a specification is composed of multiple files that each contain parts
of the specification. The specification refers to these sub specifications using
`external references`. Since references are based on URI's (so `Identifier` and not
`Location` as in URL's!) there needs to be a way to tell the validator how to
resolve those references. This is where this function comes in:

E.g.: we have a main specification in `main-spec.yaml` containing:

```yaml
...
paths:
  /pet:
    post:
      tags:
        - pet
      summary: Add a new pet to the store
      description: ''
      operationId: addPet
      responses:
        '405':
          description: Invalid input
      requestBody:
        $ref: 'http://www.example.com/subspec#/components/requestBodies/Pet'
```

And the reference is in `sub-spec.yaml`, containing:

```yaml
components:
  requestBodies:
    Pet:
      content:
        application/json:
          schema:
            $ref: '#/components/schemas/Pet'
        application/xml:
          schema:
            $ref: '#/components/schemas/Pet'
      description: Pet object that needs to be added to the store
      required: true
  ...
```

Then the validation can be performed as follows:

```javascript
import { Validator } from "@seriousme/openapi-schema-validator";
const validator = new Validator();
await validator.addSpecRef("./sub-spec.yaml", "http://www.example.com/subspec");
const res = await validator.validate("./main-spec.yaml");
// res now contains the results of the validation across main-spec and sub-spec
const specification = validator.specification;
// specification now contains a Javascript object containing the specification
// with the subspec inlined
```

<a name="validateBundle"></a>

### `<instance>.validateBundle([specification,subspecification, ...])`

This offers an alternative to the combination of `addSpecRef` and `validate`.
You can pass an array of (sub)specifications where each can be one of:

- a JSON object
- a JSON object encoded as string
- a YAML string
- a filename

There can only be one main specification present (starting with swagger/openapi) but it does not have to be the first one. If you provide filenames and the files do not have `$id` attributes, then the `$id` attribute will be generated from the filename.

If we take the YAML specifications from the previous example then validation can be performed as follows:

```javascript
import { Validator } from "@seriousme/openapi-schema-validator";
const validator = new Validator();
const res = await validator.validateBundle([ "./sub-spec.yaml", "./main-spec.yaml"]);
// res now contains the results of the validation across main-spec and sub-spec
const specification = validator.specification;
// specification now contains a Javascript object containing the specification
// with the subspec inlined
```

<a name="supportedVersions"></a>

### `Validator.supportedVersions`

This static property returns the OpenApi versions supported by this package as a
`Set`. If present, the result of `<instance>.version` is a member of this `Set`.

<a name="notes"></a>

# Notes

The [JSONschemas](/schemas/) are copied from the [OpenAPI specification JSONschemas](https://github.com/OAI/OpenAPI-Specification/tree/gh-pages) which might differ from the [OpenAPI specification HTML pages](https://spec.openapis.org/oas/latest.html)! 
If you find a bug in a schema (e.g. because it behaves differently than specified in the HTML) then 
please raise an issue at https://github.com/OAI/OpenAPI-Specification.
Shortly after the specification writers update their schema the automation will pick it up 
and include the updated version in this module.

<a name="license"></a>

# License

Licensed under the [MIT license](LICENSE.txt)
 readmeEtag: '"b4d7c96284cd0679313186812440fdaa248e90e1"' readmeLastModified: Fri, 26 Sep 2025 13:19:54 GMT repositoryId: 365588968 description: OpenApi schema validation for OpenApi versions v2, v3.0.x and v3.1.x created: '2021-05-08T18:50:37Z' updated: '2026-02-05T16:56:06Z' language: JavaScript archived: false stars: 63 watchers: 2 forks: 10 owner: seriousme logo: https://avatars.githubusercontent.com/u/3322396?v=4 license: MIT repoEtag: '"166b01cc0ce444882e86c9707bd7dba2896b10e856360b3d14d43ab849f87a09"' repoLastModified: Thu, 05 Feb 2026 16:56:06 GMT foundInMaster: true id: 983210df2c4289b2210365de0362ab66 name: openapi-schema-validator category: Description Validators link: https://www.npmjs.com/package/@seriousme/openapi-schema-validator language: Javascript source_description: OpenApi description validation v2: true v3_1: true - source: https://openapi.tools/ name: typeconv category: - Converters - Parsers link: https://github.com/grantila/typeconv/ language: typescript repository: https://github.com/grantila/typeconv/ source_description: >- typeconv is an extremely fast silver bullet type conversion utility. It converts between any of its supported types, bidirectionally. typeconv lets you convert between type systems which have core-types converters, such as JSON Schema, TypeScript, GraphQL, Open API and SureType. This package can be used as an API programatically or as an application (installed in node_modules/.bin or by using e.g. npx). v3: true id: 5a1209f6c9dd70a9f9d904b0db5924f4 repositoryMetadata: base64Readme: >- [![npm version][npm-image]][npm-url]
[![downloads][downloads-image]][npm-url]
[![build status][build-image]][build-url]
[![coverage status][coverage-image]][coverage-url]
[![Node.JS version][node-version]][node-url]


<img src="https://raw.githubusercontent.com/grantila/typeconv/master/assets/logo.svg" width="100%" />

**typeconv** is an extremely fast *silver bullet* type conversion utility.

It converts between any of its supported types, bidirectionally.

typeconv lets you convert between type systems which have [`core-types`][core-types-github-url] converters, such as JSON Schema, TypeScript, GraphQL, Open API and [SureType][suretype-github-url]. This package can be used as an API programatically or as an application (installed in `node_modules/.bin` or by using e.g. [`npx`](https://www.npmjs.com/package/npx)).

By taking advantage of the [`core-types`][core-types-github-url] ([npm][core-types-npm-url]) toolbox for generic type handling, typeconv can convert and maintain source code location information, comments, descriptions etc. when converting between the different type systems. It is using the following converter packages:
 * [`core-types-json-schema`][core-types-json-schema-github-url] ([npm][core-types-json-schema-npm-url])
 * [`core-types-ts`][core-types-ts-github-url] ([npm][core-types-ts-npm-url])
 * [`core-types-graphql`][core-types-graphql-github-url] ([npm][core-types-graphql-npm-url])
 * [`core-types-suretype`][core-types-suretype-github-url] ([npm][core-types-suretype-npm-url])

These type systems don't share the same set of types and constraints. For example, JSON Schema has *value* constraints (like *"a string must be longer than 5 characters*") and GraphQL doesn't have `null` or key-value objects as a first-class type. Convertions will therefore produce the smallest common denominator of type information, but still be very useful. See [`core-types`][core-types-github-url] for more information on its supported types, and why not implement a new conversion package yourself!


# TL;DR CLI

Convert files from TypeScript (*"ts"*) to GraphQL (*"gql"*), put the generated files in the `gql-schemas` directory in the same directory structure as the source files:

```zsh
$ typeconv -f ts -t gql -o gql-schemas 'types/**/*.ts'
```

This generates `gql-schemas/*.graphql` for each `.ts` file in `types/` (and sub-directories).

*Note that when using glob patterns, put them in quotes to not have the shell try to expand the pattern - typeconv will do it a lot better!*


## SureType

When converting *from* SureType, typeconv will extract all *exported* validators.


## Versions

Since v2:
 * The package is a pure ESM module, no more CommonJS support. This will likely not affect CLI usages.
 * Node 12 support is dropped.


# Contents

 * [Conversion example](#conversion-example)
 * [Usage](#usage)
   * [Command line](#command-line)
   * [As API](#as-api)
     * [JSON Schema](#json-schema)
     * [Open API](#open-api)
     * [TypeScript](#typescript)
     * [GraphQL](#graphql)
     * [SureType](#suretype)


# Conversion example

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>Converting the following JSON Schema:</summary>
<p>

```json
{
	"definitions": {
		"User": {
			"type": "object",
			"title": "User type",
			"description": "This type holds the user information, such as name",
			"properties": { "name": { "type": "string", "title": "The real name" } },
			"required": [ "name" ]
		},
		"ChatLine": {
			"type": "object",
			"title": "A chat line",
			"properties": {
				"user": { "$ref": "#/definitions/User" },
				"line": { "type": "string" }
			},
			"required": [ "user", "line" ]
		}
	}
}
```

</p>
</details>

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>... to TypeScript will produce:</summary>
<p>

```ts
/*User type

This type holds the user information, such as name*/
export interface User {
	/*The real name*/
	name: string;
}

/*A chat line*/
export interface ChatLine {
	user: User;
	line: string;
}
```

or if converted into TypeScript _declarations_ (for `.d.ts` files), `export interface` will be `export declare interface`.

</p>
</details>

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>... and to GraphQL will produce:</summary>
<p>

```graphql
"""
# User type

This type holds the user information, such as name
"""
type User {
	"The real name"
	name: String!
}

"A chat line"
type ChatLine {
	user: User!
	line: String!
}
```

</p>
</details>

Conversions are bi-directional, so any of these type systems can convert to any other.


# Usage


## Command line

You can depend on `typeconv`, and if so, you'll have `node_modules/.bin/typeconv` installed. Or you run it with on-the-fly installation using `npx`, as `npx typeconv args...`.

Use `-f` (or `--from-type`) to specify *from* which type system to convert, and `-t` (or `--to-type`) to specify which type system to convert *to*. Other options can be used to configure each configuration (both the *from* and the *to*) and these options are usually only available for a specific type system.

The types supported are `gql` (GraphQL), `ts` (TypeScript), `jsc` (JSON Schema), `oapi` (Open API) and `st` (SureType).

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary><code>$ typeconv --help</code></summary>
<p>

```
Usage: typeconv [options] file ...

   Options:

   -h, --help                       Print (this) help screen
   --version                        Print the program version
   -v, --verbose                    Verbose informational output (default: false)
   --dry-run                        Prepare and perform conversion, but write no output (default: false)
   --(no-)hidden                    Include hidden files, i.e. files in .gitignore,
                                    files beginning with '.' and the '.git' directory
                                     (default: true)
   -f, --from-type <type>           Type system to convert from

                                       Values:
                                          ts    TypeScript
                                          jsc   JSON Schema
                                          gql   GraphQL
                                          oapi  Open API
                                          st    SureType
                                          ct    core-types

   -t, --to-type <type>             Type system to convert to

                                       Values:
                                          ts    TypeScript
                                          jsc   JSON Schema
                                          gql   GraphQL
                                          oapi  Open API
                                          st    SureType
                                          ct    core-types

   --(no-)shortcut                  Shortcut conversion if possible (bypassing core-types).
                                    This is possible between SureType, JSON Schema and Open API
                                    to preserve all features which would otherwise be erased.
                                     (default: true)
   -o, --output-directory <dir>     Output directory. Defaults to the same as the input files.
   -O, --output-extension <ext>     Output filename extension to use.
                                    Defaults to 'ts'/'d.ts', 'json', 'yaml' and 'graphql'.
                                    Use '-' to not save a file, but output to stdout instead.
   --(no-)strip-annotations         Removes all annotations (descriptions, comments, ...) (default: false)
   TypeScript
     --(no-)ts-declaration          Output TypeScript declarations (default: false)
     --(no-)ts-disable-lint-header  Output comments for disabling linting (default: true)
     --(no-)ts-descriptive-header   Output the header comment (default: true)
     --(no-)ts-use-unknown          Use 'unknown' type instead of 'any' (default: true)
     --ts-non-exported <method>     Strategy for non-exported types (default: include-if-referenced)

                                       Values:
                                          fail                   Fail conversion
                                          ignore                 Don't include non-exported types,
                                                                 even if referenced
                                          include                Include non-exported types
                                          inline                 Don't include non-exported types,
                                                                 inline them if necessary.
                                                                 Will fail on cyclic types
                                          include-if-referenced  Include non-exported types only if they
                                                                 are referenced from exported types

     --ts-namespaces <method>       Namespace strategy. (default: ignore)

                                       Values:
                                          ignore           Ignore namespaces entirely (default).
                                                           - When converting from TypeScript, types in namespaces
                                                           aren't exported.
                                                           - When converting to TypeScript, no attempt to
                                                           reconstruct namespaces is performed.
                                          hoist            When converting from TypeScript, hoist types inside
                                                           namespaces to top-level, so that the types are
                                                           included, but without their namespace.
                                                           This can cause conflicts, in which case deeper
                                                           declarations will be dropped in favor of more top-
                                                           level declarations.
                                                           In case of same-level (namespace depth) declarations
                                                           with the same name, only one will be exported in a
                                                           non-deterministic manner.
                                          dot              When converting from TypeScript, join the namespaces
                                                           and the exported type with a dot (.).
                                                           When converting to TypeScript, try to reconstruct
                                                           namespaces by splitting the name on dot (.).
                                          underscore       When converting from TypeScript, join the namespaces
                                                           and the exported type with an underscore (_).
                                                           When converting to TypeScript, try to reconstruct
                                                           namespaces by splitting the name on underscore (_).
                                          reconstruct-all  When converting to TypeScript, try to reconstruct
                                                           namespaces by splitting the name on both dot and
                                                           underscore.

   GraphQL
     --gql-unsupported <method>     Method to use for unsupported types

                                       Values:
                                          ignore  Ignore (skip) type
                                          warn    Ignore type, but warn
                                          error   Throw an error

     --gql-null-typename <name>     Custom type name to use for null
   Open API
     --oapi-format <fmt>            Output format for Open API (default: yaml)

                                       Values:
                                          json  JSON
                                          yaml  YAML ('yml' is also allowed)

     --oapi-title <title>           Open API title to use in output document.
                                    Defaults to the input filename.
     --oapi-version <version>       Open API document version to use in output document. (default: 1)
   SureType
     --st-ref-method <method>       SureType reference export method (default: provided)

                                       Values:
                                          no-refs   Don't ref anything, inline all types
                                          provided  Reference types that are explicitly exported
                                          ref-all   Ref all provided types and those with names

     --st-missing-ref <method>      What to do when detecting an unresolvable reference (default: warn)

                                       Values:
                                          ignore  Ignore; skip type or cast to any
                                          warn    Same as 'ignore', but warn
                                          error   Fail conversion

     --(no-)st-inline-types         Inline pretty typescript types aside validator code (default: true)
     --(no-)st-export-type          Export the deduced types (or the pretty types,
                                    depending on --st-inline-types)
                                     (default: true)
     --(no-)st-export-schema        Export validator schemas (default: false)
     --(no-)st-export-validator     Export regular validators (default: true)
     --(no-)st-export-ensurer       Export 'ensurer' validators (default: true)
     --(no-)st-export-type-guard    Export type guards (is* validators) (default: true)
     --(no-)st-use-unknown          Use 'unknown' type instead of 'any' (default: true)
     --(no-)st-forward-schema       Forward the JSON Schema, and create an untyped validator schema
                                    with the raw JSON Schema under the hood
                                     (default: false)
```

</p>
</details>


## As API

To convert from one type system to another, you create a *reader* for the type system to convert **from** and a *writer* for the type system to convert **to**. The readers and writers for the different type systems have their own set of options. Some have no options (like the JSON Schema reader), some require options (like the Open API writer).


### makeConverter

Making a converter is done using `makeConverter(reader, writer, options?)`, which takes a reader and a writer, and optionally options:

#### **cwd** (string)

The current working directory, only useful when working with files.

#### **simplify** (boolean) (default true)

When simplify is true, the converter will let core-types
[*compress*](https://github.com/grantila/core-types#simplify) the
types after having converted from {reader} format to core-types.
This is usually recommended, but may cause some annotations (comments)
to be dropped.

#### **map** (ConvertMapFunction)

Custom map function for transforming each type after it has been
converted *from* the source type (and after it has been simplified),
but before it's written to the target type system.

Type: `(node: NamedType, index: number, array: ReadonlyArray<NamedType>) => NamedType`
([NamedType ref](https://github.com/grantila/core-types#specification))

If `filter` is used as well, this runs before `filter`.

If `transform` is used as well, this runs before `transform`.

#### **filter** (ConvertFilterFunction)

Custom filter function for filtering types after they have been
converted *from* the source type.

Type: `(node: NamedType, index: number, array: ReadonlyArray<NamedType>) => boolean`
([NamedType ref](https://github.com/grantila/core-types#specification))

If `map` is used as well, this runs after `map`.

If `transform` is used as well, this runs before `transform`.

#### **transform** (ConvertTransformFunction)

Custom filter function for filtering types after they have been
converted *from* the source type.

Type: `( doc: NodeDocument ) => NodeDocument`
([NodeDocument ref](https://github.com/grantila/core-types#specification))

If `map` is used as well, this runs after `map`.

If `filter` is used as well, this runs after `filter`.

#### **shortcut** (boolean) (default true)

Shortcut reader and writer if possible (bypassing core-types).



### Basic example conversion

```ts
import {
  getTypeScriptReader,
  getOpenApiWriter,
  makeConverter,
} from 'typeconv'

const reader = getTypeScriptReader( );
const writer = getOpenApiWriter( { format: 'yaml', title: 'My API', version: 'v1' } );
const { convert } = makeConverter( reader, writer );
const { data } = await convert( { data: "export type Foo = string | number;" } );
data; // This is the Open API yaml as a string
```


### JSON Schema

There are two exported functions for JSON Schema:

```ts
import { getJsonSchemaReader, getJsonSchemaWriter } from 'typeconv'

const reader = getJsonSchemaReader( );
const writer = getJsonSchemaWriter( );
```

They don't have any options.

typeconv expects the JSON Schema to contain **definitions**, i.e. to be in the form:

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>JSON Schema</summary>
<p>

```json
{
	"definitions": {
		"User": {
			"type": "object",
			"properties": { "name": { "type": "string" } },
			"required": [ "name" ]
		},
		"ChatLine": {
			"type": "object",
			"properties": {
				"user": { "$ref": "#/definitions/User" },
				"line": { "type": "string" }
			},
			"required": [ "user", "line" ]
		}
	}
}
```

</p>
</details>

typeconv doesn't support external references (to other files). If you have that, you need to use a reference parser and merge it into one inline-referenced file before using typeconv.


### Open API

Converting to or from Open API can be done with both JSON and YAML. The default is JSON.

When reading, if the filename ends with `.yml` or `.yaml`, typeconv will interpret the input as YAML.

Writing however, is decided in the writer factory and provided to `getOpenApiWriter`.

```ts
import { getOpenApiReader, getOpenApiWriter } from 'typeconv'

const reader = getOpenApiReader( );
const writer = getOpenApiWriter( {
  format: 'yaml',
  title: 'My API',
  version: 'v1',
  schemaVersion: '3.0.0',
} );
```

The options to `getOpenApiWriter` is:

```ts
interface {
  format?: string;
  title: string;
  version: string;
  schemaVersion?: string;
}
```


### TypeScript

TypeScript conversion is done using:

```ts
import { getTypeScriptReader, getTypeScriptWriter } from 'typeconv'

const reader = getTypeScriptReader( );
const writer = getTypeScriptWriter( );
```

Both these take an optional argument.

The `getTypeScriptReader` takes an optional
[`FromTsOptions`](https://github.com/grantila/core-types-ts#typescript-to-core-types)
object from [`core-types-ts`][core-types-ts-github-url], although `warn` isn't necessary since it's set by typeconv internally.

The `getTypeScriptWriter` takes an optional
[`ToTsOptions`](https://github.com/grantila/core-types-ts#core-types-to-typescript)
object from [`core-types-ts`][core-types-ts-github-url], although `warn`, `filename`, `sourceFilename`, `userPackage` and `userPackageUrl` aren't necessary since they're set by typeconv internally.


### GraphQL

GraphQL conversion is done using;

```ts
import { getGraphQLReader, getGraphQLWriter } from 'typeconv'

const reader = getGraphQLReader( );
const writer = getGraphQLWriter( );
```

Both these take an optional argument.

The `getGraphQLReader` takes an optional
[`GraphqlToCoreTypesOptions`](https://github.com/grantila/core-types-graphql#graphql-to-core-types)
object from [`core-types-graphql`][core-types-graphql-github-url], although `warn` isn't necessary since it's set by typeconv internally.

The `getGraphQLWriter` takes an optional
[`CoreTypesToGraphqlOptions`](https://github.com/grantila/core-types-graphql#core-types-to-graphql)
object from [`core-types-graphql`][core-types-graphql-github-url], although `warn`, `filename`, `sourceFilename`, `userPackage` and `userPackageUrl` aren't necessary since they're set by typeconv internally.


### SureType

SureType conversion is done using;

```ts
import { getSureTypeReader, getSureTypeWriter } from 'typeconv'

const reader = getSureTypeReader( );
const writer = getSureTypeWriter( );
```

Both these take an optional argument from the [`core-types-suretype`][core-types-suretype-github-url] package.

The `getSureTypeReader` takes an optional
[`SuretypeToJsonSchemaOptions`](https://github.com/grantila/core-types-suretype#suretype-to-core-types).

The `getSureTypeWriter` takes an optional
[`JsonSchemaToSuretypeOptions`](https://github.com/grantila/core-types-suretype#core-types-to-suretype).


[npm-image]: https://img.shields.io/npm/v/typeconv.svg
[npm-url]: https://npmjs.org/package/typeconv
[downloads-image]: https://img.shields.io/npm/dm/typeconv.svg
[build-image]: https://img.shields.io/github/actions/workflow/status/grantila/typeconv/master.yml?branch=master
[build-url]: https://github.com/grantila/typeconv/actions?query=workflow%3AMaster
[coverage-image]: https://coveralls.io/repos/github/grantila/typeconv/badge.svg?branch=master
[coverage-url]: https://coveralls.io/github/grantila/typeconv?branch=master
[node-version]: https://img.shields.io/node/v/typeconv
[node-url]: https://nodejs.org/en/

[core-types-npm-url]: https://npmjs.org/package/core-types
[core-types-github-url]: https://github.com/grantila/core-types
[core-types-json-schema-npm-url]: https://npmjs.org/package/core-types-json-schema
[core-types-json-schema-github-url]: https://github.com/grantila/core-types-json-schema
[core-types-ts-npm-url]: https://npmjs.org/package/core-types-ts
[core-types-ts-github-url]: https://github.com/grantila/core-types-ts
[core-types-graphql-npm-url]: https://npmjs.org/package/core-types-graphql
[core-types-graphql-github-url]: https://github.com/grantila/core-types-graphql
[core-types-suretype-npm-url]: https://npmjs.org/package/core-types-suretype
[core-types-suretype-github-url]: https://github.com/grantila/core-types-suretype
[suretype-github-url]: https://github.com/grantila/suretype
 readmeEtag: '"b7043a416d92ebe262c0d06cc82bc8aae498de44"' readmeLastModified: Sun, 04 Jun 2023 08:11:37 GMT repositoryId: 293245471 description: Convert between JSON Schema, TypeScript, GraphQL, Open API and SureType created: '2020-09-06T09:40:04Z' updated: '2026-01-11T08:51:56Z' language: TypeScript archived: false stars: 447 watchers: 2 forks: 10 owner: grantila logo: https://avatars.githubusercontent.com/u/5362579?v=4 license: MIT repoEtag: '"a2c72fd4df09b1e3588fded6157437d93dc8af2aec098fd5b275c952c0451cac"' repoLastModified: Sun, 11 Jan 2026 08:51:56 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/traefik/traefik-hub-helm-chart v3: true id: a37e8ac9d7a778549392a1f379fc7529 repositoryMetadata: base64Readme: >- PGEgbmFtZT0icmVhZG1lLXRvcCI+PC9hPgoKPiBTaW5jZSB0aGUgZW5kIG9mIE1heSAyMDI0LCBUcmFlZmlrIEh1YiBhbmQgVHJhZWZpayBQcm94eSBoYXZlIGJlZW4gdXNpbmcgdGhlIHNhbWUgW1RyYWVmaWsgSGVsbSBDaGFydF0oaHR0cHM6Ly9naXRodWIuY29tL3RyYWVmaWsvdHJhZWZpay1oZWxtLWNoYXJ0KSBmb3IgaW5zdGFsbGF0aW9uLgoKPGJyLz4KCjxkaXYgYWxpZ249ImNlbnRlciIgc3R5bGU9Im1hcmdpbjogMzBweDsiPgo8YSBocmVmPSJodHRwczovL2h1Yi50cmFlZmlrLmlvLyI+CiAgPGltZyBzcmM9Imh0dHBzOi8vZG9jLnRyYWVmaWsuaW8vdHJhZWZpay1odWIvaW1nL2h1Yi1sb2dvLWxpZ2h0LnN2ZyIgc3R5bGU9IndpZHRoOjI1MHB4OyIgYWxpZ249ImNlbnRlciIgLz4KPC9hPgo8YnIgLz4KPGJyIC8+Cgo8ZGl2IGFsaWduPSJjZW50ZXIiPgogICAgPGEgaHJlZj0iaHR0cHM6Ly90cmFlZmlrLmlvL3RyYWVmaWstaHViLyI+IFdlYnNpdGU8L2E+IHwKICAgIDxhIGhyZWY9Imh0dHBzOi8vaHViLnRyYWVmaWsuaW8iPkxvZyBJbjwvYT4gfAogICAgPGEgaHJlZj0iaHR0cHM6Ly9kb2MudHJhZWZpay5pby90cmFlZmlrLWh1Yi8iPkRvY3VtZW50YXRpb248L2E+CjwvZGl2Pgo8L2Rpdj4KCjxiciAvPgoKPGRpdiBhbGlnbj0iY2VudGVyIj48c3Ryb25nPlRyYWVmaWsgSHViIEhlbG0gQ2hhcnQ8L3N0cm9uZz4KCjxiciAvPgo8YnIgLz4KPC9kaXY+CgojIyBJbnRyb2R1Y3Rpb24KClRoaXMgY2hhcnQgaW5zdGFsbHMgdGhlIFRyYWVmaWsgSHViIGFnZW50IGluICpJbmdyZXNzIENvbnRyb2xsZXIqIG1vZGUgb24gYSBLdWJlcm5ldGVzIGNsdXN0ZXIuCgojIyBJbnN0YWxsYXRpb24KCiMjIyBQcmVyZXF1aXNpdGVzCgoxLiBbeF0gSGVsbSAqKnYzKiogW2luc3RhbGxlZF0oaHR0cHM6Ly9oZWxtLnNoL2RvY3MvdXNpbmdfaGVsbS8jaW5zdGFsbGluZy1oZWxtKTogYGhlbG0gdmVyc2lvbmAKMi4gW3hdIFRyYWVmaWsgTGFicyBjaGFydCByZXBvc2l0b3J5OiBgaGVsbSByZXBvIGFkZCB0cmFlZmlrIGh0dHBzOi8vdHJhZWZpay5naXRodWIuaW8vY2hhcnRzYAoKIyMjIERlcGxveWluZyBIdWIgSW5ncmVzIENvbnRyb2xsZXIgZm9yIEt1YmVybmV0ZXMKCmBgYHNoZWxsCmt1YmVjdGwgY3JlYXRlIHNlY3JldCBnZW5lcmljIGh1Yi1hZ2VudC10b2tlbiAtLWZyb20tbGl0ZXJhbD10b2tlbj1YWFhYCgpoZWxtIHVwZ3JhZGUgLS1pbnN0YWxsIHRyYWVmaWstaHViIHRyYWVmaWsvdHJhZWZpay1odWIKYGBgCgpZb3UgY2FuIGN1c3RvbWl6ZSB0aGUgaW5zdGFsbGF0aW9uIHdpdGggYSBgdmFsdWVzYCBmaWxlLgpGaW5kIHRoZSBjb21wbGV0ZSBkb2N1bWVudGF0aW9uIG9uIGFsbCBwYXJhbWV0ZXJzIGluIHRoZSBbZGVmYXVsdCB2YWx1ZSBmaWxlXSguL3RyYWVmaWstaHViL3ZhbHVlcy55YW1sKS4KCjxwIGFsaWduPSJyaWdodCI+KDxhIGhyZWY9IiNyZWFkbWUtdG9wIj5iYWNrIHRvIHRvcDwvYT4pPC9wPgoKIyMgVXBncmFkaW5nIEh1YiBBZ2VudCBmb3IgS3ViZXJuZXRlcwoKT25lIGNhbiBjaGVjayB3aGF0IGhhcyBjaGFuZ2VkIGluIHRoZSBbcmVsZWFzZV0oaHR0cHM6Ly9naXRodWIuY29tL3RyYWVmaWsvdHJhZWZpay1odWItaGVsbS1jaGFydC9yZWxlYXNlcykgb3ZlcnZpZXcgcGFnZS4KClVwZGF0ZSB0aGUgcmVwb3NpdG9yeToKCmBgYHNoZWxsCmhlbG0gcmVwbyB1cGRhdGUKYGBgCgpDaGVjayB0aGUgY3VycmVudCBDaGFydCAmIFRyYWVmaWsgSHViIHZlcnNpb246CgpgYGBzaGVsbApoZWxtIHNlYXJjaCByZXBvIHRyYWVmaWsvdHJhZWZpay1odWIKYGBgCgpVcGdyYWRlIFRyYWVmaWsgSHViOgoKYGBgc2hlbGwKaGVsbSB1cGdyYWRlIHRyYWVmaWstaHViIHRyYWVmaWsvdHJhZWZpay1odWIKYGBgCgo8cCBhbGlnbj0icmlnaHQiPig8YSBocmVmPSIjcmVhZG1lLXRvcCI+YmFjayB0byB0b3A8L2E+KTwvcD4KCiMjIFVwZ3JhZGluZyBDUkRzCgpXaXRoIEhlbG0gdjMsIENSRHMgY3JlYXRlZCBieSB0aGlzIGNoYXJ0IGNhbiBub3QgYmUgdXBkYXRlZCwgY29uc3VsdCB0aGUgW0hlbG0gRG9jdW1lbnRhdGlvbiBvbiBDUkRzXShodHRwczovL2hlbG0uc2gvZG9jcy9jaGFydF9iZXN0X3ByYWN0aWNlcy9jdXN0b21fcmVzb3VyY2VfZGVmaW5pdGlvbnMpIGZvciBtb3JlIGluZm9ybWF0aW9uLgoKPiBQbGVhc2UgcmVhZCBjYXJlZnVsbHkgdGhlIFtyZWxlYXNlIG5vdGVzXShodHRwczovL2dpdGh1Yi5jb20vdHJhZWZpay90cmFlZmlrLWh1Yi1oZWxtLWNoYXJ0L3JlbGVhc2VzKSBvZiB0aGlzIGNoYXJ0IGJlZm9yZSB1cGdyYWRpbmcgQ1JEcyEKCmBgYHNoZWxsCmt1YmVjdGwgYXBwbHkgLS1zZXJ2ZXItc2lkZSAtLWZvcmNlLWNvbmZsaWN0cyAtayBodHRwczovL2dpdGh1Yi5jb20vdHJhZWZpay90cmFlZmlrLWh1Yi1oZWxtLWNoYXJ0L3RyYWVmaWstaHViL2NyZHMvCmBgYAoKPHAgYWxpZ249InJpZ2h0Ij4oPGEgaHJlZj0iI3JlYWRtZS10b3AiPmJhY2sgdG8gdG9wPC9hPik8L3A+CgojIyBVcGdyYWRpbmcgZnJvbSB2MS54LngKClRvIHVwZ3JhZGUgZnJvbSB2MS54LngsIGEgc2VjcmV0IG5lZWRzIHRvIGJlIG1hbnVhbGx5IHJlbW92ZWQgZmlyc3Q6CgpgYGBzaGVsbAprdWJlY3RsIGRlbGV0ZSBzZWNyZXQgaHViLWFnZW50LWNlcnQgLW4gdHJhZWZpay1odWIKYGBgCgo8cCBhbGlnbj0icmlnaHQiPig8YSBocmVmPSIjcmVhZG1lLXRvcCI+YmFjayB0byB0b3A8L2E+KTwvcD4KCiMjIFVuaW5zdGFsbAoKYGBgc2hlbGwKaGVsbSB1bmluc3RhbGwgdHJhZWZpay1odWIKYGBgCgpJZiBUcmFlZmlrIEh1YiB3YXMgaW5zdGFsbGVkIGluIGEgc3BlY2lmaWMgbmFtZXNwYWNlOgoKYGBgc2hlbGwKaGVsbSB1bmluc3RhbGwgdHJhZWZpay1odWIgLS1uYW1lc3BhY2UgbXktbmFtZXNwYWNlCmBgYAoKPHAgYWxpZ249InJpZ2h0Ij4oPGEgaHJlZj0iI3JlYWRtZS10b3AiPmJhY2sgdG8gdG9wPC9hPik8L3A+CgojIyBDb250cmlidXRpbmcKCiMjIyBWZXJzaW9uaW5nCgpXZSB1c2UgW1NlbWFudGljIFZlcnNpb25pbmddKGh0dHBzOi8vc2VtdmVyLm9yZy8pLgoKUHVsbCByZXF1ZXN0cyBtdXN0IGJ1bXAgdGhlIGB2ZXJzaW9uYCBvZiB0aGUgY2hhcnQgc3BlY2lmaWVkIGluIHRoZSBbQ2hhcnQueWFtbF0oLi90cmFlZmlrLWh1Yi9DaGFydC55YW1sKSBmaWxlOgoKLSBUaGUgbmV3IHZlcnNpb24gbXVzdCBiZSBhbiBhbHBoYSBwcmUtcmVsZWFzZSAoZS5nLiAxLjYuMC1hbHBoYS4xKQotIFRoZSBuZXcgdmVyc2lvbiBtdXN0IHJlZmxlY3QgdGhlIG5hdHVyZSBvZiB0aGUgY2hhbmdlLCBhY2NvcmRpbmcgdG8gdGhlIFNlbVZlciBzcGVjaWZpY2F0aW9uCgpBIGNoYXJ0IGNhbiBiZSBtYWRlIGF2YWlsYWJsZSBwdWJsaWNseSBieSByZW1vdmluZyB0aGUgcHJlLXJlbGVhc2Ugc3VmZml4LCB0aGlzIG11c3QgYmUgZG9uZSBvbiBhIHNlcGFyYXRlIFBSIGJ5IGEgbWFpbnRhaW5lci4KCkV2ZXJ5IHZlcnNpb24gYnVtcCBpcyBwdWJsaXNoZWQgb24gdGhlIEhlbG0gQ2hhcnQgUmVnaXN0cnkuCgpUaGUgbGF0ZXN0IHByZS1yZWxlYXNlIHZlcnNpb24gb2YgdGhlIENoYXJ0IGNhbiBiZSB1c2VkIGJ5IHNwZWNpZnlpbmcgYC0tZGV2ZWxgIG9uIHRoZSBgaW5zdGFsbGAgYW5kIGB1cGdyYWRlYCBjb21tYW5kcy4KCiMjIyBMYXVuY2ggdW5pdCB0ZXN0cwoKYGBgc2hlbGwKbWFrZSB0ZXN0CmBgYAoKPHAgYWxpZ249InJpZ2h0Ij4oPGEgaHJlZj0iI3JlYWRtZS10b3AiPmJhY2sgdG8gdG9wPC9hPik8L3A+CgojIyBMaWNlbnNlCgpEaXN0cmlidXRlZCB1bmRlciB0aGUgQXBhY2hlIHYyIExpY2Vuc2UuClNlZSBbTElDRU5TRV0oLi9MSUNFTlNFKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCjxwIGFsaWduPSJyaWdodCI+KDxhIGhyZWY9IiNyZWFkbWUtdG9wIj5iYWNrIHRvIHRvcDwvYT4pPC9wPgo= readmeEtag: '"d1162fe3e633bee28072ea733ca6e3a8da73eb55"' readmeLastModified: Tue, 28 May 2024 13:30:11 GMT repositoryId: 662208402 description: >- Traefik Hub is a Kubernetes-native API Management solution for publishing, securing, and managing APIs. Configuration is driven by Kubernetes CRDs, labels, and selectors for effective GitOps. created: '2023-07-04T15:34:00Z' updated: '2025-08-17T20:16:49Z' language: Makefile archived: true stars: 2 watchers: 8 forks: 9 owner: traefik logo: https://avatars.githubusercontent.com/u/14280338?v=4 license: Apache-2.0 repoEtag: '"811466cdb1bec02914af4a77eb5886a7a49cf9663a02e3beb49063687f5ef16b"' repoLastModified: Sun, 17 Aug 2025 20:16:49 GMT category: - Documentation - Gateway - Server Implementations foundInMaster: true name: Traefik Hub link: https://doc.traefik.io/traefik-hub language: Go source_description: >- Traefik Hub is a Kubernetes-native API Management solution for publishing, securing, and managing APIs. Configuration is driven by Kubernetes CRDs, labels, and selectors for effective GitOps. v2: true - source: https://openapi.tools/ name: HopFront category: - Documentation link: https://github.com/hopfront/hopfront repository: https://github.com/hopfront/hopfront language: TypeScript source_description: >- HopFront automatically builds an user friendly UI from a collection of OpenAPI specifications. You can setup custom dashboard tailored to the way your interact with your APIs. v3: true id: 3b8ac40bdadbc38647a04c95f45fb3bf repositoryMetadata: base64Readme: >- PGJyLz4gIAo8cGljdHVyZT4KICA8c291cmNlIG1lZGlhPSIocHJlZmVycy1jb2xvci1zY2hlbWU6IGRhcmspIiBzcmNzZXQ9Imh0dHBzOi8vZ2l0aHViLmNvbS9ob3Bmcm9udC9ob3Bmcm9udC9hc3NldHMvMjc0MzU0MS9mNmY0MjU3My1kNDZjLTRkY2MtYjIzZS0zOTMzYTIwN2JmNzIiPgogIDxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vaG9wZnJvbnQvaG9wZnJvbnQvYXNzZXRzLzI3NDM1NDEvODFiNzljZTctNzE0OC00N2JjLWFiZDctZDRjYTMwZWYyMGQyIiBoZWlnaHQ9IjYwIj4KPC9waWN0dXJlPgoKLS0tCgpbSG9wRnJvbnRdKGh0dHBzOi8vaG9wZnJvbnQuY29tLykgaXMgdGhlIGJlc3Qgd2F5IHRvIGludGVyYWN0IHdpdGggQVBJcy4KCiFbSG9wRnJvbnQgU2NyZWVuc2hvdF0oaHR0cHM6Ly9hc3NldHMtZ2xvYmFsLndlYnNpdGUtZmlsZXMuY29tLzY1MDljYTExZTNiNmRhNTNiZjRmYzVmMy82NTU3NTliY2E4NzBhNTg4MjM3ZGQyNDhfR3JvdXAlMjAzMC1wLTEwODAucG5nKQoKIyMgRmVhdHVyZXMKLSBTZXQgdXAgaW4gX2xpdGVyYWxseV8gMiBtaW51dGVzLgogIC0geW91IGp1c3QgbmVlZCBhbiBbT3BlbkFQSSBzcGVjXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy9sYXRlc3QuaHRtbCkgKHdoaWNoIGNhbiBiZSBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlZCBpZiB5b3UgZG9uJ3QgYWxyZWFkeSBoYXZlIG9uZSkuCi0gQnJvd3NlICYgcXVlcnkgeW91ciBBUEkgaW4gYSBIdW1hbi1mcmllbmRseSB3YXkuCiAgLSBIb3BGcm9udCB1bmRlcnN0YW5kcyByZWxhdGlvbnNoaXBzIGJldHdlZW4geW91ciBlbmRwb2ludHMsIG5vIG1vcmUgY29weS9wYXN0aW5nIHlvdXIgaWRzIGV2ZXJ5d2hlcmUhCi0gQnVpbGQgY3VzdG9tIGFuZCBkeW5hbWljIGRhc2hib2FyZHMuCiAgLSBVc2VmdWwgZm9yIHJhcGlkbHkgYnVpbGRpbmcgaW50ZXJuYWwgdG9vbHMgb3IgcGVyc29uYWxpemVkIFVJcyBiYXNlZCBvbiB5b3VyIGRhaWx5IEFQSSBpbnRlcmFjdGlvbnMuCgojIyBJbnN0YWxsYXRpb24KVEw7RFI6IGBkb2NrZXIgcnVuIC1wIDMwMDA6MzAwMCAtdiBob3Bmcm9udDovZXRjL2hvcGZyb250IGhvcGZyb250L2hvcGZyb250OmxhdGVzdGAKCkhvcEZyb250IGlzIGN1cnJlbnRseSBhdmFpbGFibGUgYXMgYSBEb2NrZXIgaW1hZ2UsIGZvbGxvdyB0aGUgaW5zdGFsbGF0aW9uIGd1aWRlIFtoZXJlXShodHRwczovL2h1Yi5kb2NrZXIuY29tL3IvaG9wZnJvbnQvaG9wZnJvbnQpLgoK readmeEtag: '"888912d465217f6cbd07184684969595dc8f39f9"' readmeLastModified: Thu, 14 Mar 2024 08:26:16 GMT repositoryId: 703097382 description: Turn any API into a workable app in 2 minutes created: '2023-10-10T15:33:01Z' updated: '2026-02-03T22:08:17Z' language: TypeScript archived: false stars: 79 watchers: 2 forks: 6 owner: hopfront logo: https://avatars.githubusercontent.com/u/147492760?v=4 license: NOASSERTION repoEtag: '"14f691b5665e3f7148a67ae18b90d45b524b716702b6b03ce7d7d3be8a3ec9ae"' repoLastModified: Tue, 03 Feb 2026 22:08:17 GMT foundInMaster: true - source: https://openapi.tools/ name: Xapi Platform category: - GUI Editors - Text Editors - Testing language: Saas link: https://xapihub.io/ source_description: >- Xapi is an integrated platform that enables you to implement a streamlined API-first design strategy with standardized API artifacts and collaborative teams. v2: false v3: false v3_1: true id: a899f827ee5c657acd105a5110757e0a foundInMaster: true - source: https://openapi.tools/ name: Fuego category: - Auto Generators - Data Validators - Server language: Go link: https://go-fuego.github.io/fuego/ repository: https://github.com/go-fuego/fuego source_description: >- Golang Fuego - web & API framework generating OpenAPI 3 spec from source code v3: true id: 4be418d9c134b194919087fc1e16a8cf repositoryMetadata: base64Readme: >- <!-- markdownlint-disable MD041 -->
<p align="center">
  <img src="./static/fuego.svg" height="200" alt="Fuego Logo" />
</p>

# Fuego 🔥

[![Go Reference](https://pkg.go.dev/badge/github.com/go-fuego/fuego.svg)](https://pkg.go.dev/github.com/go-fuego/fuego)
[![Go Report Card](https://goreportcard.com/badge/github.com/go-fuego/fuego)](https://goreportcard.com/report/github.com/go-fuego/fuego)
[![Coverage Status](https://coveralls.io/repos/github/go-fuego/fuego/badge.svg?branch=main)](https://coveralls.io/github/go-fuego/fuego?branch=main)
[![CodSpeed Badge](https://img.shields.io/endpoint?url=https://codspeed.io/badge.json)](https://codspeed.io/go-fuego/fuego?utm_source=badge)
[![Discord Gophers](https://img.shields.io/badge/Discord%20Gophers-%23fuego-%237289da?)](https://discord.gg/9csuB6WNsq)

> The framework for busy Go developers

🚀 **Explore and contribute to our [2025 Roadmap](https://github.com/go-fuego/fuego/discussions/263)!** 🚀

Production-ready Go API framework generating OpenAPI documentation from code.
Inspired by Nest, built for Go developers.

Also empowers templating with `html/template`, `a-h/templ` and `maragudk/gomponents`:
see [the example](./examples/full-app-gourmet) [running live](https://gourmet.quimerch.com).

## Why Fuego?

Chi, Gin, Fiber and Echo are great frameworks.
But since they were designed a long time ago,
[their current API does not allow them][gin-gonic-issue] to deduce
OpenAPI types from signatures, things that are now possible with generics.
Fuego offers a lot of "modern Go based" features that make it easy
to develop APIs and web applications.

## Features

- **OpenAPI**: Fuego automatically generates OpenAPI documentation
  from **code** - _not from comments nor YAML files!_
- **100% `net/http` compatible** (no lock-in): Fuego is built on top of `net/http`,
  so you can use any `http.Handler` middleware or handler! Fuego also supports
  `log/slog`, `context` and `html/template`.
- **Routing**: Fuego router is based on Go 1.22 `net/http`, with grouping and
  middleware support
- **Serialization/Deserialization**: Fuego automatically serializes and
  deserializes JSON, XML and HTML Forms based on user-provided structs
  (or not, if you want to do it yourself)
- **Validation**: Fuego provides a simple and fast validator based on `go-playground/validator`
- **Transformation**: easily transform your data by implementing the
  `fuego.InTransform` and `fuego.OutTransform` interfaces - also useful for
  custom **validation**
- **Middlewares**: easily add a custom `net/http` middleware
  or use the provided middlewares.
- **Error handling**: Fuego provides centralized error handling with
  the standard [RFC 9457](https://www.rfc-editor.org/rfc/rfc9457).
- **Rendering**: Fuego provides a simple and fast rendering system based on
  `html/template` - you can still also use your own template system like
  `templ` or `gomponents`
- **Adaptors**: Fuego can be plugged to an existing Gin or Echo server to
  generate OpenAPI documentation

## Examples

### Hello World

```go
package main

import "github.com/go-fuego/fuego"

func main() {
	s := fuego.NewServer()

	fuego.Get(s, "/", func(c fuego.ContextNoBody) (string, error) {
		return "Hello, World!", nil
	})

	s.Run()
}
```

### Simple POST

```go
package main

import "github.com/go-fuego/fuego"

type MyInput struct {
	Name string `json:"name" validate:"required"`
}

type MyOutput struct {
	Message string `json:"message"`
}

func main() {
	s := fuego.NewServer()

	// Automatically generates OpenAPI documentation for this route
	fuego.Post(s, "/user/{user}", myController)

	s.Run()
}

func myController(c fuego.ContextWithBody[MyInput]) (*MyOutput, error) {
	body, err := c.Body()
	if err != nil {
		return nil, err
	}

	return &MyOutput{Message: "Hello, " + body.Name}, nil
}
```

### With transformation & custom validation

```go
type MyInput struct {
	Name string `json:"name" validate:"required"`
}

// Will be called just before returning c.Body()
func (r *MyInput) InTransform(context.Context) error {
	r.Name = strings.ToLower(r.Name)

	if r.Name == "fuego" {
		return errors.New("fuego is not a valid name for this input")
	}

	return nil
}
```

### More OpenAPI documentation

```go
package main

import (
	"github.com/go-fuego/fuego"
	"github.com/go-fuego/fuego/option"
	"github.com/go-fuego/fuego/param"
)

func main() {
	s := fuego.NewServer()

	// Custom OpenAPI options
	fuego.Post(s, "/", myController,
		option.Description("This route does something..."),
		option.Summary("This is my summary"),
		option.Tags("MyTag"), // A tag is set by default according to the return type (can be deactivated)
		option.Deprecated(),  // Marks the route as deprecated in the OpenAPI spec

		option.Query("name", "Declares a query parameter with default value", param.Default("Carmack")),
		option.Header("Authorization", "Bearer token", param.Required()),
		optionPagination,
		optionCustomBehavior,
	)

	s.Run()
}

var optionPagination = option.Group(
	option.QueryInt("page", "Page number", param.Default(1), param.Example("1st page", 1), param.Example("42nd page", 42)),
	option.QueryInt("perPage", "Number of items per page"),
)

var optionCustomBehavior = func(r *fuego.BaseRoute) {
	r.XXX = "YYY"
}
```

### Std lib compatibility

```go
package main

import (
	"net/http"

	"github.com/go-fuego/fuego"
)

func main() {
	s := fuego.NewServer()

	// Standard net/http middleware
	fuego.Use(s, func(next http.Handler) http.Handler {
		return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
			w.Header().Set("X-Hello", "World")
			next.ServeHTTP(w, r)
		})
	})

	// Standard net/http handler with automatic OpenAPI route declaration
	fuego.GetStd(s, "/std", func(w http.ResponseWriter, r *http.Request) {
		w.Write([]byte("Hello, World!"))
	})

	s.Run()
}
```

### Real-world examples

Please see the [`/examples` folder](./examples/) for more examples.

- [Simple CRUD with OpenAPI](./examples/petstore)
- [Full app with HTML rendering](./examples/full-app-gourmet)

<details>
<summary>All features</summary>

```go
package main

import (
	"context"
	"errors"
	"net/http"
	"strings"

	chiMiddleware "github.com/go-chi/chi/v5/middleware"
	"github.com/go-fuego/fuego"
	"github.com/rs/cors"
)

type Received struct {
	Name string `json:"name" validate:"required"`
}

type MyResponse struct {
	Message       string `json:"message"`
	BestFramework string `json:"best"`
}

func main() {
	s := fuego.NewServer(
		fuego.WithAddr("localhost:8088"),
	)

	fuego.Use(s, cors.Default().Handler)
	fuego.Use(s, chiMiddleware.Compress(5, "text/html", "text/css"))

	// Fuego 🔥 handler with automatic OpenAPI generation, validation, (de)serialization and error handling
	fuego.Post(s, "/", func(c fuego.ContextWithBody[Received]) (MyResponse, error) {
		data, err := c.Body()
		if err != nil {
			return MyResponse{}, err
		}

		c.Response().Header().Set("X-Hello", "World")

		return MyResponse{
			Message:       "Hello, " + data.Name,
			BestFramework: "Fuego!",
		}, nil
	})

	// Standard net/http handler with automatic OpenAPI route declaration
	fuego.GetStd(s, "/std", func(w http.ResponseWriter, r *http.Request) {
		w.Write([]byte("Hello, World!"))
	})

	s.Run()
}

// InTransform will be called when using c.Body().
// It can be used to transform the entity and raise custom errors
func (r *Received) InTransform(context.Context) error {
	r.Name = strings.ToLower(r.Name)
	if r.Name == "fuego" {
		return errors.New("fuego is not a name")
	}
	return nil
}

// OutTransform will be called before sending data
func (r *MyResponse) OutTransform(context.Context) error {
	r.Message = strings.ToUpper(r.Message)
	return nil
}
```

```bash
curl http://localhost:8088/std
# Hello, World!
curl http://localhost:8088 -X POST -d '{"name": "Your Name"}' -H 'Content-Type: application/json'
# {"message":"HELLO, YOUR NAME","best":"Fuego!"}
curl http://localhost:8088 -X POST -d '{"name": "Fuego"}' -H 'Content-Type: application/json'
# {"error":"cannot transform request body: cannot transform request body: fuego is not a name"}
```

</details>

## From net/http to Fuego in 10s

<https://github.com/go-fuego/fuego/assets/46993939/7438a71c-75a4-4e88-a584-71da6362c575>

<details>
<summary>Views</summary>

### Before

<img width="946" alt="image" src="https://github.com/go-fuego/fuego/assets/46993939/394fed17-a1e2-4b67-89b2-8e6c9eeb771b">

#### After

<img width="1010" alt="image" src="https://github.com/go-fuego/fuego/assets/46993939/321088d7-bec4-46cc-a7ee-9a0fa45d7711">

#### Diff

<img width="1413" alt="image" src="https://github.com/go-fuego/fuego/assets/46993939/18796a59-b2e4-4e01-81d1-88c581de3466">

#### Benefits of using Fuego views (controllers returning HTML)

- Never forget to return after an error
- OpenAPI schema generated, listing all the routes
- Deserialization and validation are easier
- Transition to Fuego is easy and fast

</details>

## Contributing

See the [contributing guide](CONTRIBUTING.md).
Thanks to [everyone who has contributed][contributors-url] to this project! ❤️

<a href="https://github.com/go-fuego/fuego/graphs/contributors">
  <img src="https://contrib.rocks/image?repo=go-fuego/fuego" alt="Graph of contributors" />
</a>

<small>Made with [contrib.rocks](https://contrib.rocks)</small>

## Roadmap

See the [board](https://github.com/orgs/go-fuego/projects/1).

## Disclaimer for experienced gophers

I know you might prefer to use `net/http` directly,
but if having a frame can convince my company to use
Go instead of Node, I'm happy to use it.

## License

[MIT](./LICENSE.txt)

[gin-gonic-issue]: https://github.com/gin-gonic/gin/issues/155
[contributors-url]: https://github.com/go-fuego/fuego/graphs/contributors
 readmeEtag: '"86f196c41e91d5188f299ad5aba305b84ec0d81e"' readmeLastModified: Wed, 03 Dec 2025 18:40:14 GMT repositoryId: 706256859 description: >- Golang Fuego - Web framework generating OpenAPI 3 spec from source code - Pluggable to existing Gin & Echo APIs created: '2023-10-17T15:36:43Z' updated: '2026-02-05T23:37:43Z' language: Go archived: false stars: 1657 watchers: 12 forks: 122 owner: go-fuego logo: https://avatars.githubusercontent.com/u/148255240?v=4 license: MIT repoEtag: '"f19fab8b8d85ce36aaf95bc4397b730d3c911eb85d37a9916800f1953b704dd8"' repoLastModified: Thu, 05 Feb 2026 23:37:43 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags - openapi31 tags repository: https://github.com/danielgtaylor/huma v3: true repositoryMetadata: base64Readme: >- <a href="#">
	<picture>
		<source media="(prefers-color-scheme: dark)" srcset="https://huma.rocks/huma-dark.png" />
		<source media="(prefers-color-scheme: light)" srcset="https://huma.rocks/huma.png" />
		<img alt="Huma Logo" src="https://huma.rocks/huma.png" />
	</picture>
</a>

[![HUMA Powered](https://img.shields.io/badge/Powered%20By-HUMA-f40273)](https://huma.rocks/) [![CI](https://github.com/danielgtaylor/huma/workflows/CI/badge.svg?branch=main)](https://github.com/danielgtaylor/huma/actions?query=workflow%3ACI+branch%3Amain++) [![codecov](https://codecov.io/gh/danielgtaylor/huma/branch/main/graph/badge.svg)](https://codecov.io/gh/danielgtaylor/huma) [![Docs](https://godoc.org/github.com/danielgtaylor/huma/v2?status.svg)](https://pkg.go.dev/github.com/danielgtaylor/huma/v2?tab=doc) [![Go Report Card](https://goreportcard.com/badge/github.com/danielgtaylor/huma/v2)](https://goreportcard.com/report/github.com/danielgtaylor/huma/v2)

[**🌎中文文档**](./README_CN.md)
[**🇯🇵日本語ドキュメント**](./README_JA.md)

- [What is huma?](#intro)
- [Install](#install)
- [Example](#example)
- [Documentation](#documentation)

<a name="intro"></a>
A modern, simple, fast & flexible micro framework for building HTTP REST/RPC APIs in Go backed by OpenAPI 3 and JSON Schema. Pronounced IPA: [/'hjuːmɑ/](https://en.wiktionary.org/wiki/Wiktionary:International_Phonetic_Alphabet). The goals of this project are to provide:

- Incremental adoption for teams with existing services
  - Bring your own router (including Go 1.22+), middleware, and logging/metrics
  - Extensible OpenAPI & JSON Schema layer to document existing routes
- A modern REST or HTTP RPC API backend framework for Go developers
  - Described by [OpenAPI 3.1](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.1.0.md) & [JSON Schema](https://json-schema.org/)
- Guard rails to prevent common mistakes
- Documentation that can't get out of date
- High-quality generated developer tooling

Features include:

- Declarative interface on top of your router of choice:
  - Operation & model documentation
  - Request params (path, query, header, or cookie)
  - Request body
  - Responses (including errors)
  - Response headers
- JSON Errors using [RFC9457](https://datatracker.ietf.org/doc/html/rfc9457) and `application/problem+json` by default (but can be changed)
- Per-operation request size limits with sane defaults
- [Content negotiation](https://developer.mozilla.org/en-US/docs/Web/HTTP/Content_negotiation) between server and client
  - Support for JSON ([RFC 8259](https://tools.ietf.org/html/rfc8259)) and optionally CBOR ([RFC 7049](https://tools.ietf.org/html/rfc7049)) content types via the `Accept` header with the default config.
- Conditional requests support, e.g. `If-Match` or `If-Unmodified-Since` header utilities.
- Optional automatic generation of `PATCH` operations that support:
  - [RFC 7386](https://www.rfc-editor.org/rfc/rfc7386) JSON Merge Patch
  - [RFC 6902](https://www.rfc-editor.org/rfc/rfc6902) JSON Patch
  - [Shorthand](https://github.com/danielgtaylor/shorthand) patches
- Annotated Go types for input and output models
  - Generates JSON Schema from Go types
  - Static typing for path/query/header params, bodies, response headers, etc.
  - Automatic input model validation & error handling
- Documentation generation using [Stoplight Elements](https://stoplight.io/open-source/elements)
- Optional CLI built-in, configured via arguments or environment variables
  - Set via e.g. `-p 8000`, `--port=8000`, or `SERVICE_PORT=8000`
  - Startup actions & graceful shutdown built-in
- Generates OpenAPI for access to a rich ecosystem of tools
  - Mocks with [API Sprout](https://github.com/danielgtaylor/apisprout) or [Prism](https://stoplight.io/open-source/prism)
  - SDKs with [OpenAPI Generator](https://github.com/OpenAPITools/openapi-generator) or [oapi-codegen](https://github.com/deepmap/oapi-codegen)
  - CLI with [Restish](https://rest.sh/)
  - And [plenty](https://openapi.tools/) [more](https://apis.guru/awesome-openapi3/category.html)
- Generates JSON Schema for each resource using optional `describedby` link relation headers as well as optional `$schema` properties in returned objects that integrate into editors for validation & completion.

This project was inspired by [FastAPI](https://fastapi.tiangolo.com/). Logo & branding designed by Kari Taylor.

## Sponsors

A big thank you to our current & former sponsors!

<div>
					<img width="1000" height="0" />
					<a href="https://zuplo.link/huma-gh ">
					<picture>
							<!-- <source media="(prefers-color-scheme: dark)" srcset="docs/zuplo-dark.png"> -->
							<img src="https://github.com/user-attachments/assets/aace5aa7-32bd-45cf-a8f8-2e352feaf017" alt="Zuplo" width="260" align="right">
					</picture>
					</a>
					<h3>Zuplo: Scale, Protect, and Productize your Huma API</h3>
					<p>
							Our API Gateway allows you to secure your API, scale it globally, generate documentation from your OpenAPI, and monetize your users.
					</p>
					<a href="https://zuplo.link/huma-gh ">Start for Free</a>
</div>
<hr/>

- [@bclements](https://github.com/bclements)
- [@bekabaz](https://github.com/bekabaz)
- [@victoraugustolls](https://github.com/victoraugustolls)
- [@phoenixtechnologies-io](https://github.com/phoenixtechnologies-io)
- [@chenjr0719](https://github.com/chenjr0719)
- [@vinogradovkonst](https://github.com/vinogradovkonst)
- [@miyamo2](https://github.com/miyamo2)
- [@nielskrijger](https://github.com/nielskrijger)

## Testimonials

> This is by far my favorite web framework for Go. It is inspired by FastAPI, which is also amazing, and conforms to many RFCs for common web things ... I really like the feature set, the fact that it [can use] Chi, and the fact that it is still somehow relatively simple to use. I've tried other frameworks and they do not spark joy for me. - [Jeb_Jenky](https://www.reddit.com/r/golang/comments/zhitcg/comment/izmg6vk/?utm_source=reddit&utm_medium=web2x&context=3)

> After working with #Golang for over a year, I stumbled upon Huma, the #FastAPI-inspired web framework. It’s the Christmas miracle I’ve been hoping for! This framework has everything! - [Hana Mohan](https://twitter.com/unamashana/status/1733088066053583197)

> I love Huma. Thank you, sincerely, for this awesome package. I’ve been using it for some time now and it’s been great! - [plscott](https://www.reddit.com/r/golang/comments/1aoshey/comment/kq6hcpd/?utm_source=reddit&utm_medium=web2x&context=3)

> Thank you Daniel for Huma. Superbly useful project and saves us a lot of time and hassle thanks to the OpenAPI gen — similar to FastAPI in Python. - [WolvesOfAllStreets](https://www.reddit.com/r/golang/comments/1aqj99d/comment/kqfqcml/?utm_source=reddit&utm_medium=web2x&context=3)

> Huma is wonderful, I've started working with it recently, and it's a pleasure, so thank you very much for your efforts 🙏 - [callmemicah](https://www.reddit.com/r/golang/comments/1b32ts4/comment/ksvr9h7/?utm_source=reddit&utm_medium=web2x&context=3)

> It took us 3 months to build our platform in Python with FastAPI, SQL Alchemy and only 3 weeks to rewrite it in Go with Huma and SQL C. Things just work and I seldomly have to debug where in Python I spent a majority of my time debugging. - [Bitclick\_](https://www.reddit.com/r/golang/comments/1cj2znb/comment/l2e4u6y/)

> Look at Huma, it's great. A nice slim layer on top of stdlib mux/chi and automatic body and parameter serialization, kinda feels like doing dotnet web APIs, but forces you to actually design request and response structs, which is great imo. - [Kirides](https://www.reddit.com/r/golang/comments/1fnn5c2/comment/lokuvpo/)

# Install

Install via `go get`. Note that Go 1.23 or newer is required.

```sh
# After: go mod init ...
go get -u github.com/danielgtaylor/huma/v2
```

# Example

Here is a complete basic hello world example in Huma, that shows how to initialize a Huma app complete with CLI, declare a resource operation, and define its handler function.

```go
package main

import (
	"context"
	"fmt"
	"net/http"

	"github.com/danielgtaylor/huma/v2"
	"github.com/danielgtaylor/huma/v2/adapters/humachi"
	"github.com/danielgtaylor/huma/v2/humacli"
	"github.com/go-chi/chi/v5"

	_ "github.com/danielgtaylor/huma/v2/formats/cbor"
)

// Options for the CLI. Pass `--port` or set the `SERVICE_PORT` env var.
type Options struct {
	Port int `help:"Port to listen on" short:"p" default:"8888"`
}

// GreetingOutput represents the greeting operation response.
type GreetingOutput struct {
	Body struct {
		Message string `json:"message" example:"Hello, world!" doc:"Greeting message"`
	}
}

func main() {
	// Create a CLI app which takes a port option.
	cli := humacli.New(func(hooks humacli.Hooks, options *Options) {
		// Create a new router & API
		router := chi.NewMux()
		api := humachi.New(router, huma.DefaultConfig("My API", "1.0.0"))

		// Add the operation handler to the API.
		huma.Get(api, "/greeting/{name}", func(ctx context.Context, input *struct{
			Name string `path:"name" maxLength:"30" example:"world" doc:"Name to greet"`
		}) (*GreetingOutput, error) {
			resp := &GreetingOutput{}
			resp.Body.Message = fmt.Sprintf("Hello, %s!", input.Name)
			return resp, nil
		})

		// Tell the CLI how to start your router.
		hooks.OnStart(func() {
			http.ListenAndServe(fmt.Sprintf(":%d", options.Port), router)
		})
	})

	// Run the CLI. When passed no commands, it starts the server.
	cli.Run()
}
```

> [!TIP]
> Replace `chi.NewMux()` → `http.NewServeMux()` and `humachi.New` → `humago.New` to use the standard library router from Go 1.22+. Just make sure your `go.mod` has `go 1.22` or newer listed in it. Everything else stays the same! Switch whenever you are ready.

You can test it with `go run greet.go` (optionally pass `--port` to change the default) and make a sample request using [Restish](https://rest.sh/) (or `curl`):

```sh
# Get the message from the server
$ restish :8888/greeting/world
HTTP/1.1 200 OK
...
{
	$schema: "http://localhost:8888/schemas/GreetingOutputBody.json",
	message: "Hello, world!"
}
```

Even though the example is tiny you can also see some generated documentation at http://localhost:8888/docs. The generated OpenAPI is available at http://localhost:8888/openapi.json or http://localhost:8888/openapi.yaml.

Check out the [Huma tutorial](https://huma.rocks/tutorial/installation/) for a step-by-step guide to get started.

# Documentation

See the [https://huma.rocks/](https://huma.rocks/) website for full documentation in a presentation that's easier to navigate and search then this README. You can find the source for the site in the `docs` directory of this repo.

Official Go package documentation can always be found at https://pkg.go.dev/github.com/danielgtaylor/huma/v2.

# Articles & Mentions

- [APIs in Go with Huma 2.0](https://dgt.hashnode.dev/apis-in-go-with-huma-20)
- [Reducing Go Dependencies: A case study of dependency reduction in Huma](https://dgt.hashnode.dev/reducing-go-dependencies)
- [Golang News & Libs & Jobs shared on Twitter/X](https://twitter.com/golangch/status/1752175499701264532)
- Featured in Go Weekly [#495](https://golangweekly.com/issues/495) & [#498](https://golangweekly.com/issues/498)
- [Bump.sh Deploying Docs from Huma](https://docs.bump.sh/guides/bump-sh-tutorials/huma/)
- Mentioned in [Composable HTTP Handlers Using Generics](https://www.willem.dev/articles/generic-http-handlers/)

Be sure to star the project if you find it useful!

<a href="https://star-history.com/#danielgtaylor/huma&Date">
	<picture>
		<source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/svg?repos=danielgtaylor/huma&type=Date&theme=dark" />
		<source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/svg?repos=danielgtaylor/huma&type=Date" />
		<img alt="Star History Chart" src="https://api.star-history.com/svg?repos=danielgtaylor/huma&type=Date" />
	</picture>
</a>
 readmeEtag: '"448a830b5adc390545662eeea6de85aee9b681dc"' readmeLastModified: Fri, 12 Sep 2025 04:54:47 GMT repositoryId: 245763170 description: Huma REST/HTTP API Framework for Golang with OpenAPI 3.1 created: '2020-03-08T06:19:51Z' updated: '2026-02-05T23:40:20Z' language: Go archived: false stars: 3795 watchers: 27 forks: 242 owner: danielgtaylor logo: https://avatars.githubusercontent.com/u/106826?v=4 license: MIT repoEtag: '"8b235155914ed233e5b619eacdd2fb7b971173885b1942302bc602caebf38499"' repoLastModified: Thu, 05 Feb 2026 23:40:20 GMT foundInMaster: true category: - Auto Generators - Data Validators - Server id: 96ae035d7b486f67e7bae7f6208179f9 v3_1: true name: Huma language: Go link: https://huma.rocks/ source_description: >- A modern, simple, fast & flexible micro framework for building HTTP REST/RPC APIs in Go backed by OpenAPI 3 and JSON Schema. - source: https://openapi.tools/ name: openapi-merge category: - Converters - Server Implementations language: TypeScript repository: https://github.com/robertmassaioli/openapi-merge link: https://github.com/robertmassaioli/openapi-merge source_description: >- A simple tool to merge multiple independent OpenAPI description files into one file. Useful when you have a number of microservices and you decide to expose them through a gateway. v2: false v3: true v3_1: false id: d02b4c51ae616ca5f33e3d4d36dab531 repositoryMetadata: base64Readme: >- IyMgVGhlIG9wZW5hcGktbWVyZ2UgcmVwb3NpdG9yeQoKV2VsY29tZSB0byB0aGUgb3BlbmFwaS1tZXJnZSByZXBvc2l0b3J5LiBUaGlzIGxpYnJhcnkgaXMgaW50ZW5kZWQgdG8gYmUgdXNlZCBmb3IgbWVyZ2luZyBtdWx0aXBsZSBPcGVuQVBJIDMuMCBmaWxlcyB0b2dldGhlci4gVGhlIG1vc3QgY29tbW9uIHJlYXNvbiB0aGF0IGRldmVsb3BlcnMgd2FudCB0byBkbyB0aGlzIGlzIGJlY2F1c2UgdGhleSBoYXZlIG11bHRpcGxlIHNlcnZpY2VzIHRoYXQgdGhleSB3aXNoIHRvIGV4cG9zZSB1bmRlcm5lYXRoIGEgc2luZ2xlIEFQSSBHYXRld2F5LiBUaGVyZWZvcmUsIGV2ZW4gdGhvdWdoIHRoaXMgbWVyZ2luZyBsb2dpYyBpcyBzdWZmaWNpZW50bHkgZ2VuZXJpYyB0byBiZSB1c2VkIGZvciBtb3N0IHVzZSBjYXNlcywgc29tZSBvZiB0aGUgZmVhdHVyZSBkZWNpc2lvbnMgYXJlIHRhaWxvcmVkIGZvciB0aGF0IHNwZWNpZmljIHVzZSBjYXNlLgoKIyMjIFNjcmVlbnNob3RzCgohW0ltZ3VyXShodHRwczovL2kuaW1ndXIuY29tL0dqblNYQ1MucG5nKQooQW4gZXhhbXBsZSBvZiBjcmVhdGluZyBhbiBvcGVuYXBpLW1lcmdlLmpzb24gY29uZmlndXJhdGlvbiBmaWxlIGZvciB0aGUgQ0xJIHRvb2wpCgojIyMgQWJvdXQgdGhpcyByZXBvc2l0b3J5CgpUaGlzIGlzIGEgbXVsdGktcGFja2FnZSByZXBvc2l0b3J5IHRoYXQgY29udGFpbnM6CgoqIFRoZSBvcGVuYXBpLW1lcmdlIGxpYnJhcnk6IFshW25wbV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vcGVuYXBpLW1lcmdlP2xhYmVsPW9wZW5hcGktbWVyZ2UmbG9nbz1ucG0pXShodHRwczovL2JpdC5seS8yV25JeXRGKQoqIFRoZSBvcGVuYXBpLW1lcmdlIENMSSB0b29sOiBbIVtucG1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3Yvb3BlbmFwaS1tZXJnZS1jbGk/bGFiZWw9b3BlbmFwaS1tZXJnZS1jbGkmbG9nbz1ucG0pXShodHRwczovL2JpdC5seS8zYkVWcTNmKQoKRGVwZW5kaW5nIG9uIHlvdXIgdXNlLWNhc2UsIHlvdSBtYXkgd2lzaCB0byB1c2UgdGhlIENMSSB0b29sIG9yIHRoZSBsaWJyYXJ5IGluIHlvdXIgcHJvamVjdC4gUGxlYXNlIHNlZSB0aGUgcmVhZG1lIGZpbGUgb2YgdGhlIHNwZWNpZmljIHBhY2thZ2UgZm9yIG1vcmUgZGV0YWlscy4KCiMjIyBEZXZlbG9waW5nIG9uIG9wZW5hcGktbWVyZ2UKClRoaXMgcHJvamVjdCBpcyBhIG11bHRpLXBhY2thZ2UgcmVwb3NpdG9yeSBhbmQgdXNlcyB0aGUgW2JvbHRdWzFdIHRvb2wgdG8gbWFuYWdlIHRoZXNlIHBhY2thZ2VzIGluIG9uZSBkZXZlbG9wbWVudCBleHBlcmllbmNlLgoKQWZ0ZXIgY2hlY2tpbmcgb3V0IHRoaXMgcmVwb3NpdG9yeSwgeW91IGNhbiBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kIHRvIGluc3RhbGwgdGhlIHJlcXVpcmVkIGRlcGVuZGVuY2llczoKCmBgYCBzaGVsbApib2x0IGluc3RhbGwKYGBgCgpZb3UgY2FuIHRoZW4gdGVzdCBydW5uaW5nIHRoZSBDTEkgdG9vbCBieSBydW5uaW5nOgoKYGBgIHNoZWxsCnlhcm4gY2xpCmBgYAoKSWYgeW91IHdpc2ggdG8gZW5zdXJlIHRoYXQgeW91IGNhbiBkZXZlbG9wIG9uIHRoZSBgb3BlbmFwaS1tZXJnZWAgbGlicmFyeSBpbiBwYXJhbGxlbCB0byB0aGUgYG9wZW5hcGktbWVyZ2UtY2xpYCB0b29sCnRoZW4geW91IG11c3QgcnVuIHRoZSBUeXBlc2NyaXB0IGJ1aWxkIGZvciBgb3BlbmFwaS1tZXJnZWAgaW4gd2F0Y2ggbW9kZS4gWW91IGNhbiBkbyB0aGlzIGJ5OgoKYGBgIHNoZWxsCmJvbHQgdyBvcGVuYXBpLW1lcmdlIGJ1aWxkIC13CmBgYAoKVGhpcyB3aWxsIGVuc3VyZSB0aGF0IHRoZSBUeXBlc2NyaXB0IGlzIGNvbXBpbGVkIGludG8gSmF2YVNjcmlwdCBzbyB0aGF0IGl0IGNhbiBiZSB1c2VkIGJ5IHRoZSBgb3BlbmFwaS1tZXJnZS1jbGlgIHRvb2wuCgpGb3IgdGhlIG90aGVyIG9wZXJhdGlvbnMgdGhhdCB5b3Ugd2lzaCB0byBwZXJmb3JtLCBwbGVhc2Ugc2VlIHRoZSBwYWNrYWdlLmpzb24gb2YgdGhlIG90aGVyIHBhY2thZ2VzIGluIHRoaXMgcmVwb3NpdG9yeS4KCiBbMV06IGh0dHBzOi8vZ2l0aHViLmNvbS9ib2x0cGtnL2JvbHQ= readmeEtag: '"97186676a02261a5d5d40f2f90071c23a7a68751"' readmeLastModified: Thu, 06 Jun 2024 04:47:59 GMT repositoryId: 369944083 description: >- Merge multiple OpenAPI 3.0 (Swagger) files together via a library or CLI tool. created: '2021-05-23T02:23:17Z' updated: '2026-01-26T13:59:55Z' language: TypeScript archived: false stars: 171 watchers: 3 forks: 31 owner: robertmassaioli logo: https://avatars.githubusercontent.com/u/149178?v=4 license: MIT repoEtag: '"15898bc072f9db7569c6d0f83ee6d643a8e3de71b04cb23ba2040ff3788fa36d"' repoLastModified: Mon, 26 Jan 2026 13:59:55 GMT foundInMaster: true - source: https://openapi.tools/ name: OpenAPI::Modern category: - Description Validators - Data Validators repository: https://github.com/karenetheridge/openapi-modern link: https://metacpan.org/dist/OpenAPI-Modern language: Perl source_description: >- OpenAPI v3.1 validator and JSON Schema validator (draft7, draft2019-09, draft2020-12) v3_1: true id: ca155f2693c6aa619365c0854c0125f5 repositoryMetadata: base64Readme: >- =pod

=encoding UTF-8

=head1 NAME

OpenAPI::Modern - Validate HTTP requests and responses against an OpenAPI v3.0, v3.1 or v3.2 document

=head1 VERSION

version 0.126

=head1 SYNOPSIS

  my $openapi = OpenAPI::Modern->new(
    openapi_uri => 'https://prod.example.com',  # adjust for each deployment environment
    openapi_schema => YAML::PP->new(boolean => 'JSON::PP')->load_string(<<'YAML'));
  $self: /api         # canonical_uri will become https://prod.example.com/api
  openapi: 3.2.0
  info:
    title: Test API
    version: 1.2.3
  paths:
    /foo/{foo_id}:
      servers:
        - url: https://{host}.example.com
          variables:
            host:
              default: prod
              enum: [dev, stg, prod]
      parameters:
      - name: foo_id
        in: path
        required: true
        schema:
          pattern: ^[a-z]+$
      - name: bar_id
        in: path
        required: true
        schema:
          pattern: ^\d+$
      post:
        operationId: my_foo_request
        parameters:
        - name: My-Request-Header
          in: header
          required: true
          schema:
            pattern: ^[0-9]+$
        requestBody:
          required: true
          content:
            application/json:
              schema:
                type: object
                properties:
                  hello:
                    type: string
                    pattern: ^[0-9]+$
        responses:
          200:
            description: success
            headers:
              My-Response-Header:
                required: true
                schema:
                  pattern: ^[0-9]+$
            content:
              application/json:
                schema:
                  type: object
                  required: [ status ]
                  properties:
                    status:
                      const: ok
  YAML

  say 'request:';
  my $request = Mojo::Message::Request->new;
  $request->url->parse("https://dev.example.com/foo/bar");
  $request->headers('My-Request-Header' => '123');
  $request->headers('Content-Type' => 'application/json');
  $request->body('{"hello": 123}');
  my $results = $openapi->validate_request($request);
  say $results;
  say ''; # newline
  say JSON::MaybeXS->new(convert_blessed => 1, canonical => 1, pretty => 1, indent_length => 2)->encode($results);

  say 'response:';
  my $response = Mojo::Message::Response->new(code => 200, message => 'OK');
  $response->headers->content_type('application/json');
  $response->headers->header('My-Response-Header', '123');
  $response->body('{"status": "ok"}');
  $results = $openapi->validate_response($response, { request => $request });
  say $results;
  say ''; # newline
  say JSON::MaybeXS->new(convert_blessed => 1, canonical => 1, pretty => 1, indent_length => 2)->encode($results);

prints:

  request:
  '/request/body/hello': got integer, not string
  '/request/body': not all properties are valid

  {
    "errors" : [
      {
        "absoluteKeywordLocation" : "https://production.example.com/api#/paths/~1foo~1%7Bfoo_id%7D~1bar~1%7Bbar_id%7D/post/requestBody/content/application~1json/schema/properties/hello/type",
        "error" : "got integer, not string",
        "instanceLocation" : "/request/body/hello",
        "keywordLocation" : "/paths/~1foo~1{foo_id}~1bar~1%7Bbar_id%7D/post/requestBody/content/application~1json/schema/properties/hello/type"
      },
      {
        "absoluteKeywordLocation" : "https://production.example.com/api#/paths/~1foo~1%7Bfoo_id%7D~1bar~1%7Bbar_id%7D/post/requestBody/content/application~1json/schema/properties",
        "error" : "not all properties are valid",
        "instanceLocation" : "/request/body",
        "keywordLocation" : "/paths/~1foo~1{foo_id}~1bar~1%7Bbar_id%7D/post/requestBody/content/application~1json/schema/properties"
      }
    ],
    "valid" : false
  }

  response:
  valid

  {
    "valid" : true
  }

=head1 DESCRIPTION

This module provides various tools for working with an
L<OpenAPI Specification v3.1 or v3.2 document|https://spec.openapis.org/oas/latest#openapi-document> within
your application. The JSON Schema evaluator is fully specification-compliant; the OpenAPI evaluator
aims to be but some features are not yet available.

=for Pod::Coverage BUILDARGS FREEZE THAW

=for stopwords schema jsonSchemaDialect metaschema subschema perlish operationId openapi Mojolicious

=head1 CONSTRUCTOR ARGUMENTS

If construction of the object is not successful, for example the document has a syntax error, the
call to C<new()> will throw an exception, which will likely be a L<JSON::Schema::Modern::Result>
object containing details.

Unless otherwise noted, these are also available as read-only accessors.

=head2 openapi_uri

Optional.

The URI that identifies the OpenAPI document; an alias to C<< ->openapi_document->canonical_uri >>.
See L<JSON::Schema::Modern::Document::OpenAPI/canonical_uri>.
Ignored if L</openapi_document> is provided.

This URI will be used at runtime to resolve relative URIs used in
the OpenAPI document, such as for C<jsonSchemaDialect> or C<servers url> values and C<$ref>
locations, as well as used for locations in L<JSON::Schema::Modern::Result> objects (see below).

The value of C<$self> in the document (if present) is resolved against this value.
It is strongly recommended that this resulting URI is absolute.

=head2 openapi_schema

The data structure describing the OpenAPI document (as specified at
L<https://spec.openapis.org/oas/latest>; an alias to C<< ->openapi_document->schema >>.
See L<JSON::Schema::Modern::Document::OpenAPI/schema>.
Ignored if L</openapi_document> is provided, otherwise required.

=head2 openapi_document

The L<JSON::Schema::Modern::Document::OpenAPI> document that holds the OpenAPI information to be
used for validation. If it is not provided to the constructor, then
L</openapi_schema> B<MUST> be provided, and L</evaluator> will also be used if provided.

=head2 evaluator

The L<JSON::Schema::Modern> object to use for all URI resolution and JSON Schema evaluation.
Optional (a default is constructed when omitted).

This must be prepared in advance if custom metaschemas are to be used, as the evaluator is what
holds the information about all available schemas (which are used by keywords such as
C<jsonSchemaDialect> and C<$ref>.

=head2 debug

Boolean; defaults to false (can also be set via C<$DEBUG>).

When set, useful diagnostic information is stored in the C<$options> hash used in public methods
described below.

=head1 METHODS

=head2 document_get

  my $parameter_data = $openapi->document_get('/paths/~1foo~1{foo_id}~1bar~1%7Bbar_id%7D/post/parameters/0');

Fetches the portion of the main OpenAPI document located at the provided JSON pointer.
Proxies to L<JSON::Schema::Modern::Document::OpenAPI/get>.
This is not recursive (does not follow C<$ref> chains) -- for that, use
C<< $openapi->recursive_get(Mojo::URL->new->fragment($json_pointer)) >>, see
L</recursive_get>.

=head2 validate_request

  $result = $openapi->validate_request(
    $request,
    # optional second argument can contain any combination of these keys+values:
    my $options = {
      path_template => '/foo/{foo_id}/bar/{bar_id}',
      operation_id => 'my_foo_request',
      path_captures => { foo_id => 'abc', bar_id => 2 },
      method => 'POST',
    },
  );

Validates an L<HTTP::Request>, L<Plack::Request>, L<Catalyst::Request>,
L<Dancer2::Core::Request> or L<Mojo::Message::Request>
object against the corresponding OpenAPI document, returning a
L<JSON::Schema::Modern::Result> object.

Absolute URIs in the result object are constructed by resolving the openapi document path against
the L</openapi_uri> (which is derived from the document's C<$self> keyword as well as the URI
provided to the document constructor).

The second argument is an optional hashref that contains extra information about the request,
corresponding to the values expected by L</find_path_item> below. The method populates the hashref
with some information about the request:
save it and pass it to a later L</validate_response> call (that corresponds to a response for this request)
to improve performance.

=head2 validate_response

  $result = $openapi->validate_response(
    $response,
    {
      request => $request,
    },
  );

Validates an L<HTTP::Response>, L<Plack::Response>, L<Catalyst::Response>,
L<Dancer2::Core::Response> or L<Mojo::Message::Response>
object against the corresponding OpenAPI document, returning a
L<JSON::Schema::Modern::Result> object.

Absolute URIs in the result object are constructed by resolving the openapi document path against
the L</openapi_uri> (which is derived from the document's C<$self> keyword as well as the URI
provided to the document constructor).

The second argument is an optional hashref that contains extra information about the request
corresponding to the response, as in L</validate_request> and L</find_path_item>.

C<request> in the hashref represents the original request object that
corresponds to this response, which can be used to find the appropriate section of the document if
other values (such as C<operationId>) are not known.

In addition, this values are populated into the hashref:

=for :items * C<response>: the L<Mojo::Message::Response> object that was provided or converted from the
  provided response

=head2 find_path_item

=for Pod::Coverage find_path

  $result = $self->find_path_item($options);

Finds the appropriate L<path-item|https://spec.openapis.org/oas/latest#path-item-object> entry
in the OpenAPI document corresponding to a request. Called
internally by both L</validate_request> and L</validate_response>.

The single argument is a hashref that contains information about the request. Various combinations
of values can be provided; possible values are:

=over 4

=item *

C<request>: the object representing the HTTP request. Supported types are: L<HTTP::Request>, L<Plack::Request>, L<Catalyst::Request>, L<Dancer2::Core::Request>, L<Mojo::Message::Request>. Converted to a L<Mojo::Message::Request> (the value in this hashref will be overwritten).

=item *

C<uri>: the URI of the HTTP request. Converted from any string-compatible type to a L<Mojo::URL>.

=item *

C<method>: the HTTP method used by the request (case-sensitive)

=item *

C<path_template>: a string representing the (possibly partial) path portion of the request URI, with placeholders in braces (e.g. C</pets/{petId}>); see L<https://spec.openapis.org/oas/latest#paths-object>.

=item *

C<operation_id>: a string corresponding to the L<operationId|https://learn.openapis.org/specification/paths.html#the-endpoints-list> at a particular path-template and HTTP location under C</paths>. In the case of ambiguous matches (such as the possibility of more than one C<path_template> matching the request URI), providing this value will serve to unambiguously state which path-item and operation are intended, and is also more efficient as not all path-item entries need to be searched to find a match.

=item *

C<path_captures>: a hashref mapping placeholders in the path template to their literal values in the request URI (not percent-decoded; not recommended to be provided for use in path matching due to inconsistencies with url-escaping, and this capability may be removed in the future)

=back

All values are optional, and will be derived from each other as needed (albeit less
efficiently than if they were provided). All passed-in values MUST be consistent with each other and
the request or the return value from this method is false and appropriate errors will be included
in the C<$options> hash.

When successful, the options hash will be populated (or updated) with all the above keys,
and the return value is true.
When not successful, the options hash will be populated with key C<errors>, an arrayref containing
a L<JSON::Schema::Modern::Error> object, and the return value is false.

In addition, these values are also populated in the options hash (when available):

=for stopwords punycode

=over 4

=item *

C<uri_captures>: a hashref mapping placeholders in the entire uri template (server url plus path template) to their actual values in the request URI; for values coming from the host, these will be puny-decoded; for values coming from the path via a server url these will be url-unescaped, but path values coming from the path template are B<NOT> url-unescaped.

=item *

C<operation_uri>: a URI indicating the document location of the operation object for the request, after following any references (usually something under C</paths/>, but may be in another document). Use C<< $openapi->evaluator->get($uri) >> to fetch this content (see L<JSON::Schema::Modern/get>). Note that this is the same as:

    $openapi->document_get(Mojo::URL->new($openapi->openapi_uri)->fragment($path_to_operation);

(See the L<documentation for an operation|https://learn.openapis.org/specification/paths.html#the-endpoints-list>
or in
L<§4.10 of the specification|https://spec.openapis.org/oas/latest#operation-object>.)

=item *

C<debug>: when C<$OpenAPI::Modern::DEBUG> or L</debug> is set on the OpenAPI::Modern object, additional diagnostic information is stored here in separate keys:

=over 4

=item *

C<uri_patterns>: an arrayref of patterns that were attempted to be matched against the URI, in match order

=back

=back

You can find the associated operation object in the OpenAPI document by using either C<operation_uri>,
or by calling C<< $openapi->openapi_document->operationId_path($operation_id) >>
(see L<JSON::Schema::Modern::Document::OpenAPI/operationId_path>) (note that the latter will
be changed in a subsequent release, in order to support operations existing in other documents).

=head2 recursive_get

Given a uri or uri-reference (resolved against the main OpenAPI document's C<canonical_uri>),
get the definition at that location, following any C<$ref>s along the
way. Include the expected definition type
(one of C<schema>, C<response>, C<parameter>, C<example>, C<request-body>, C<header>,
C<security-scheme>, C<link>, C<callbacks>, or C<path-item>)
for validation of the entire reference chain.

Returns the data in scalar context, or a tuple of the data and the canonical URI of the
referenced location in list context.

If the provided location is relative, the main OpenAPI document is used for the base URI.
If you have a local json pointer you want to resolve, you can turn it into a uri-reference by
prepending C<#> and url-encoding it, e.g. C<< Mojo::URL->new->fragment($jsonp) >>.

  my $param = $openapi->recursive_get('#/components/parameters/Content-Encoding', 'parameter');
  my $operation_schema = $openapi->recursive_get(Mojo::URL->new('https://example.com/api.json')
    ->fragment(jsonp(qw(/paths /foo/{foo_id} get requestBody content application/json schema)));

=head2 canonical_uri

An accessor that delegates to L<JSON::Schema::Modern::Document/canonical_uri>.

=head2 schema

An accessor that delegates to L<JSON::Schema::Modern::Document/schema>.

=head2 get_media_type

An accessor that delegates to L<JSON::Schema::Modern/get_media_type>.

=head2 add_media_type

A setter that delegates to L<JSON::Schema::Modern/add_media_type>.

=head1 CACHING

=for stopwords preforking

Very large OpenAPI documents may take a noticeable time to be
loaded and parsed. You can reduce the impact to your preforking application by loading all necessary
documents at startup, and impact can be further reduced by saving objects to cache and then
reloading them (perhaps by using a timestamp or checksum to determine if a fresh reload is needed).

  sub get_openapi (...) {
    my $serialized_file = Mojo::File->new($serialized_filename);
    my $openapi_file = Mojo::File->new($openapi_filename);
    my $openapi;
    if ($serialized_file->stat->mtime < $openapi_file->stat->mtime)) {
      $openapi = OpenAPI::Modern->new(
        openapi_uri => '/api',
        openapi_schema => decode_json($openapi_file->slurp_raw), # your openapi document
      );
      $openapi->evaluator->add_schema(decode_json(...));  # any other needed schemas
      my $frozen = Sereal::Encoder->new({ freeze_callbacks => 1 })->encode($openapi);
      $serialized_file->spew_raw($frozen);
    }
    else {
      my $frozen = $serialized_file->slurp_raw;
      $openapi = Sereal::Decoder->new->decode($frozen);
    }

    # add custom format validations, media types and encodings here
    $openapi->evaluator->add_media_type(...);

    return $openapi;
  }

See also L<JSON::Schema::Modern/CACHING>.

=head1 BUNDLED SCHEMA DOCUMENTS

This distribution comes bundled with all the metaschema documents you need to build your
application, or build custom schemas on top of. These can be referenced via constants in
L<OpenAPI::Modern::Utilities> if you are building complex cross-referencing documents.
We always use the latest versions of the documents that are published for each minor specification
version; if you need earlier versions (but you should ask yourself why), you can find them at
L<https://spec.openapis.org/#openapi-specification-schemas>.

The default metaschema used by this tool does not permit the use of C<$schema> keywords
in subschemas (unless the value is equal to the default OAS dialect), but a more permissive
dialect is also available (or you can define your own), which you declare by providing the
L<C<jsonSchemaDialect>|https://spec.openapis.org/oas/latest#openapi-object> property in your OpenAPI
Document.

The schemas are also available under the URIs C<< s/<date>/latest/ >> so you don't have to change your
code or configurations to keep pace with internal changes.

An even stricter schema and dialect are available, via the metaschema_uri
C<https://raw.githubusercontent.com/karenetheridge/OpenAPI-Modern/master/share/3.2/strict-schema.json>,
which treats the `format` keyword as an assertion, and also prevents any unknown keywords from being
used in JSON Schemas. This is useful to avoid
spelling mistakes from going unnoticed and resulting in false positive results.

The OpenAPI v3.0 schema document is also available, for use in validating v3.0.x OpenAPI
descriptions, but parsing of the description (e.g. for use in request and response validation) is
not supported. It can be used directly as a schema (e.g. by using its URI in a C<$schema> or C<$ref>
keyword) after calling L<JSON::Schema::Modern::Utilities/load_cached_document>.

=head1 ON THE USE OF JSON SCHEMAS

Embedded JSON Schemas, through the use of the C<schema> keyword, are fully draft2020-12-compliant,
as per the spec, and implemented with L<JSON::Schema::Modern>. Unless overridden with the use of the
L<jsonSchemaDialect|https://spec.openapis.org/oas/latest#specifying-schema-dialects> keyword, their
metaschema is the "dialect" schema listed at L<https://spec.openapis.org/oas/#schema-iterations>, which allows for use of the
OpenAPI-specific keywords (C<discriminator>, C<xml>, C<externalDocs>, and C<example>), as defined in
L<the specification|https://spec.openapis.org/oas/latest#schema-object>. Format validation is turned
B<on>, and the use of content* keywords is off (see
L<JSON::Schema::Modern/validate_content_schemas>).

As references (with the C<$ref> keyword) may reference any position within the entire OpenAPI
document, json pointers are relative to the B<root> of the document, not the root of the subschema
itself. References to other documents are also permitted, provided those documents have been loaded
into the evaluator in advance (see L<JSON::Schema::Modern/add_schema>).

When deserializing parameter values (whose decoding is not strictly defined with a media-type),
values are treated as strings by default. However, if the schema contains a C<type>, C<const> or
C<enum> keyword, the value will
(attempted to be) coerced into that type before being passed to the JSON Schema evaluator. C<allOf>,
C<anyOf>, C<oneOf>, C<not>, C<$ref> and C<$dynamicRef> keywords are also followed in an attempt to
infer the correct desired type, and C<$id> and C<$schema> are respected.
This process includes inspection of subschemas for object values or array items, for parameter
values that are deserialized into those types.
When no type constraint is present, the value will remain as a string; otherwise when multiple types
are permitted, deserialization is attempted in this order: C<object>, C<array>, C<null>, C<boolean>,
C<number>, C<string>.

=head1 LIMITATIONS

All message validation is done using L<Mojolicious> objects (L<Mojo::Message::Request> and
L<Mojo::Message::Response>). If messages of other types are passed, conversion is done on a
best-effort basis, but since different implementations have different levels of adherence to the RFC
specs, some validation errors may occur e.g. if a certain required header is missing on the
original. For best results in validating real messages from the network, parse them directly into
Mojolicious messages (see L<Mojo::Message/parse>).

Only certain permutations of OpenAPI documents are supported at this time:

=over 4

=item *

for query parameters, only C<style: form> and C<explode: true> is supported, only the first value of each parameter name is considered, and C<allowEmptyValue> and C<allowReserved> are not checked

=item *

cookie parameters are not checked at all yet

=item *

C<multipart/*> messages are not yet supported

=item *

OpenAPI descriptions must be contained in a single document; while C<$ref>erences to other documents (such as within a C</components> structure) are supported, C</paths> entries in other documents are not considered at this time.

=item *

The use of C<$ref> within a path-item object is only allowed when not adjacent to any other path-item properties (C<parameters>, C<servers>, request methods)

=item *

Security schemes in the OpenAPI description, and the use of any C<Authorization> headers in requests, are not currently supported.

=back

=head1 SEE ALSO

=over 4

=item *

L<Mojolicious::Plugin::OpenAPI::Modern>

=item *

L<Test::Mojo::Role::OpenAPI::Modern>

=item *

L<JSON::Schema::Modern::Document::OpenAPI>

=item *

L<JSON::Schema::Modern>

=item *

L<https://json-schema.org>

=item *

L<https://www.openapis.org>

=item *

L<https://learn.openapis.org>

=item *

L<https://spec.openapis.org/oas/v3.0>

=item *

L<https://spec.openapis.org/oas/v3.1>

=item *

L<https://spec.openapis.org/oas/v3.2>

=item *

L<https://spec.openapis.org/oas>

=back

=head1 GIVING THANKS

=for stopwords MetaCPAN GitHub

If you found this module to be useful, please show your appreciation by
adding a +1 in L<MetaCPAN|https://metacpan.org/dist/OpenAPI-Modern>
and a star in L<GitHub|https://github.com/karenetheridge/OpenAPI-Modern>.

=head1 SUPPORT

Bugs may be submitted through L<https://github.com/karenetheridge/OpenAPI-Modern/issues>.

I am also usually active on irc, as 'ether' at C<irc.perl.org> and C<irc.libera.chat>.

=for stopwords OpenAPI

You can also find me on the L<JSON Schema Slack server|https://json-schema.slack.com> and L<OpenAPI
Slack server|https://open-api.slack.com>, which are also great resources for finding help.

=head1 AUTHOR

Karen Etheridge <ether@cpan.org>

=head1 COPYRIGHT AND LICENCE

This software is copyright (c) 2021 by Karen Etheridge.

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.

Some schema files have their own licence, in share/oas/LICENSE.

=cut
 readmeEtag: '"4848954d45b1d4b91bca391d293ff4c7f6a1d109"' readmeLastModified: Sat, 31 Jan 2026 01:05:37 GMT repositoryId: 422955010 description: >- Validate HTTP requests and responses against an OpenAPI v3.0, v3.1 or v3.2 document created: '2021-10-30T18:01:46Z' updated: '2026-01-31T01:05:43Z' language: Perl archived: false stars: 8 watchers: 3 forks: 4 owner: karenetheridge logo: https://avatars.githubusercontent.com/u/303051?v=4 license: NOASSERTION repoEtag: '"fd2ffa92a16b5dc0d2f97c0e55a7395f4fd13dce33fdb908cf93f2474df3b1e6"' repoLastModified: Sat, 31 Jan 2026 01:05:43 GMT foundInMaster: true - source: https://openapi.tools/ name: OpenAPI Qraft category: - Code Generators - SDK repository: https://github.com/openapi-qraft/openapi-qraft link: https://openapi-qraft.github.io/openapi-qraft/ language: TypeScript source_description: >- Generate a type-safe TanStack Query client for React from an OpenAPI Document. v2: false v3: true v3_1: true id: c65f9ae24d4cb8721415a8b3b513cce4 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFFyYWZ0CgpUaGlzIHJlcG9zaXRvcnkgaG91c2VzIHRoZSBPcGVuQVBJIFFyYWZ0IHRvb2xzLCBkZXNpZ25lZCB0byBlbmhhbmNlIHRoZSBkZXZlbG9wbWVudCBleHBlcmllbmNlIGJ5IHByb3ZpZGluZwpnZW5lcmF0ZWQgdHlwZXNhZmUgVGFuU3RhY2sgUXVlcnkgd3JhcHBlci4KCiMjIPCfk6YgUGFja2FnZXMKCiMjIyBbYEBvcGVuYXBpLXFyYWZ0L2NsaWAg8J+Ul10oLi9wYWNrYWdlcy9jbGkvUkVBRE1FLm1kKQoKQSBwb3dlcmZ1bCBjb21tYW5kLWxpbmUgdXRpbGl0eSBmb3IgZ2VuZXJhdGluZyBzZXJ2aWNlIGRlY2xhcmF0aW9ucyBhbmQgdHlwZWQgKipUYW5TdGFjayBRdWVyeSoqIGludGVyZmFjZXMgZGlyZWN0bHkgZnJvbSBhbgoqKk9wZW5BUEkgU2NoZW1hKiouCgpbUmVhZCBNb3JlXSguL3BhY2thZ2VzL2NsaS9SRUFETUUubWQpCgojIyMgW2BAb3BlbmFwaS1xcmFmdC9yZWFjdGAg8J+Ul10oLi9wYWNrYWdlcy9yZWFjdC1jbGllbnQvUkVBRE1FLm1kKQoKQSBtb2R1bGFyICoqVHlwZVNjcmlwdCoqIGNsaWVudCB0aGF0IGZhY2lsaXRhdGVzIHR5cGUtc2FmZSBBUEkgcmVxdWVzdHMgaW4gUmVhY3QgYXBwbGljYXRpb25zLCBsZXZlcmFnaW5nICoqVGFuU3RhY2sgUXVlcnkqKi4gSXQKdXNlcyBhIFByb3h5LWJhc2VkIGFyY2hpdGVjdHVyZSB0byBkeW5hbWljYWxseSBjYWxsICoqUmVhY3QgSG9va3Mgd2l0aCB0eXBlZCBwYXJhbWV0ZXJzKiouCgoqKk1haW4gRmVhdHVyZXM6KioKCi0gVHlwZS1zYWZlIEFQSSBSZXF1ZXN0cwotIER5bmFtaWMgUHJveHktQmFzZWQgSG9va3MgZm9yIFRhblN0YWNrIFJlYWN0IFF1ZXJ5Ci0gU2FmZSBhbmQgY29tcGF0aWJsZSBpbnRlZ3JhdGlvbiB3aXRoIFRhblN0YWNrIFF1ZXJ5CgpbUmVhZCBNb3JlXSguL3BhY2thZ2VzL3JlYWN0LWNsaWVudC9SRUFETUUubWQpCgojIyDwn6SdIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyB0byB0aGUgT3BlbkFQSSBRcmFmdCBhcmUgaGlnaGx5IGFwcHJlY2lhdGVkLiBXaGV0aGVyIGl0J3MgaW1wcm92aW5nIGRvY3VtZW50YXRpb24sIGFkZGluZyBuZXcKZmVhdHVyZXMsIG9yIHJlcG9ydGluZyBpc3N1ZXMsIHlvdXIgaGVscCBtYWtlcyBhIGJpZyBkaWZmZXJlbmNlIGluIG1ha2luZyB0aGVzZSB0b29scyBiZXR0ZXIgZm9yIGV2ZXJ5b25lLgoKIyMg8J+ThCBMaWNlbnNlCgpBbGwgcGFja2FnZXMgd2l0aGluIHRoaXMgcmVwb3NpdG9yeSBhcmUgbGljZW5zZWQgdW5kZXIgdGhlIFtNSVQgTGljZW5zZV0oLi9MSUNFTlNFLnR4dCkuCg== readmeEtag: '"9861733ebd5f8a8429226a02bda48cfb18d725b9"' readmeLastModified: Wed, 16 Oct 2024 02:01:43 GMT repositoryId: 755634410 description: ⚡ Full TanStack Query power in type-safe OpenAPI hooks for React created: '2024-02-10T19:15:33Z' updated: '2026-02-03T09:38:50Z' language: TypeScript archived: false stars: 94 watchers: 1 forks: 5 owner: OpenAPI-Qraft logo: https://avatars.githubusercontent.com/u/159651230?v=4 license: MIT repoEtag: '"dc4fef8ef98b84282d659d18f15a8a71106aa3b0ad4881c95a5a0bde955496c0"' repoLastModified: Tue, 03 Feb 2026 09:38:50 GMT foundInMaster: true - source: https://openapi.tools/ name: Mojolicious::Plugin::OpenAPI::Modern category: Server repository: https://github.com/karenetheridge/mojolicious-plugin-openapi-modern link: https://metacpan.org/dist/Mojolicious-Plugin-OpenAPI-Modern language: Perl source_description: Mojolicious plugin for OpenAPI::Modern v3_1: true id: 77699d6a49674ab7e178a8ff8bafb2b2 repositoryMetadata: base64Readme: >- =pod

=encoding UTF-8

=head1 NAME

Mojolicious::Plugin::OpenAPI::Modern - Mojolicious plugin providing access to an OpenAPI document and parser

=head1 VERSION

version 0.019

=head1 SYNOPSIS

  $app->config({
    openapi => {
      document_filename => 'data/openapi.yaml',
      after_response => sub ($c) { ... },
    },
    ...
  });

  # or:

  $app->config({
    openapi => {
      document_uri => 'https://example.com/api/main.json',
      schema => {
        openapi => '3.2.0',
        info => {
          title => 'Test API with raw schema',
          version => '1.2.3',
        },
        ...
      },
    },
    ...
  });

  $app->plugin('OpenAPI::Modern', $app->config->{openapi});

  # in a controller...
  my $result = $c->openapi->validate_request($c->req);

=head1 DESCRIPTION

This L<Mojolicious> plugin makes an L<OpenAPI::Modern> object available to the application.

There are many features to come.

=for stopwords openapi operationId subref

=head1 CONFIGURATION OPTIONS

=head2 schema

The literal, unblessed Perl data structure containing the OpenAPI document. See
L<OpenAPI::Modern/openapi_schema>; passed to the L<OpenAPI::Modern> constructor.

Only used if L</openapi_obj> is not provided.

=head2 document_uri

A uri (string or Mojo::URL object) which identifies the OpenAPI document, and is used for resolving
any relative URIs within the document (as well as locations in any error objects from validation).
Note: as of OpenAPI 3.2, you can also set this value within the document itself, using the C<$self>
keyword. This is passed as C<openapi_uri> to the L<OpenAPI::Modern> constructor.

Note that in tests, you should use a relative URI, otherwise request URIs will not match. This is
because L<Test::Mojo> uses a randomly-generated port for test requests, which cannot be predicted in
advance to be included in C<openapi_uri>.

Only used if L</openapi_obj> is not provided.

=head2 document_filename

A filename indicating from where to load the OpenAPI document. Supports YAML and json file formats.
Only used if L</schema> is not provided; this value will also be used as the L</document_uri> if one
was not explicitly provided.

Only used if neither L</schema> nor L</openapi_obj> provided.

=head2 openapi_obj

An L<OpenAPI::Modern> object to use. If provided, all other options above are ignored.

=head2 after_response

A subref which runs after the response has been finalized, to allow you to perform validation on it.
You B<must not> mutate the response here, nor swap it out for a different response, so use this only
for telemetry and logging.

  my $after_response = sub ($c) {
    my $result = $c->validate_response;
    if ($result->valid) {
      $c->log->debug('response is valid');
    }
    else {
      # see JSON::Schema::Modern::Result for different output formats
      $c->log->error("response is invalid:\n", $result);
    }
  };

=head1 METHODS

=head2 register

Instantiates an L<OpenAPI::Modern> object and provides an accessor to it.

=head1 HELPERS

These methods are made available on the C<$c> object (the invocant of all controller methods,
and therefore other helpers).

=head2 openapi

The L<OpenAPI::Modern> object; it holds your OpenAPI specification and is reused between requests.

=head2 validate_request

  my $result = $c->openapi->validate_request;

Passes C<< $c->req >> to L<OpenAPI::Modern/validate_request> and returns a
L<JSON::Schema::Modern::Result> object.

Note that the matching L<Mojo::Routes::Route> object for this request is I<not> used to find the
OpenAPI path-item that corresponds to this request: only information in the request URI itself is
used (although some information in the route may be used in future features).

You might want to define an C<under> route action that calls C<validate_request> and short-circuits
with an HTTP 400 response on validation failure.

=head2 validate_response

  my $result = $c->openapi->validate_response;

Passes C<< $c->res >> and C<< $c->req >> to L<OpenAPI::Modern/validate_response> and returns a
L<JSON::Schema::Modern::Result> object.

As this can only be called in the parts of the dispatch flow where the response has already been
rendered and finalized, a hook has been set up for you; you can access it by providing a subref to the
L</after_response> configuration value:

  $app->config->{openapi}{after_response} //= sub ($c) {
    my $result = $c->validate_response;
    # ... do something with the validation result
  };

Note that the matching L<Mojo::Routes::Route> object for this request is I<not> used to find the
OpenAPI path-item that corresponds to this request and response: only information in the request URI
itself is used (although some information in the route may be used in future features).

=head1 STASH VALUES

This plugin stores all its data under the C<openapi> hashref, e.g.:

  my $operation_id = $c->stash->{openapi}{operation_id};

Keys starting with underscore are for I<internal use only> and should not be relied upon to behave
consistently across release versions. Values that may be used by controllers and templates are:

=over 4

=item *

C<path_template>: Set by the first call to L</validate_request> or L</validate_response>. A string representing the request URI, with placeholders in braces (e.g. C</pets/{petId}>); see L<https://spec.openapis.org/oas/latest#paths-object>.

=item *

C<path_captures>: Set by the first call to L</validate_request> or L</validate_response>. A hashref mapping placeholders in the path to their actual values in the request URI.

=item *

C<operation_id>: Set by the first call to L</validate_request> or L</validate_response>. Contains the corresponding L<operationId|https://swagger.io/docs/specification/paths-and-operations/#operationid> of the current endpoint.

=item *

C<method>: Set by the first call to L</validate_request> or L</validate_response>. The HTTP method used by the request, lower-cased.

=back

=head1 SEE ALSO

=over 4

=item *

L<OpenAPI::Modern>

=item *

L<Test::Mojo::Role::OpenAPI::Modern>

=item *

L<JSON::Schema::Modern::Document::OpenAPI>

=item *

L<JSON::Schema::Modern>

=item *

L<https://json-schema.org>

=item *

L<https://www.openapis.org/>

=item *

L<https://learn.openapis.org/>

=item *

L<https://spec.openapis.org/oas/latest.html>

=back

=head1 GIVING THANKS

=for stopwords MetaCPAN GitHub

If you found this module to be useful, please show your appreciation by
adding a +1 in L<MetaCPAN|https://metacpan.org/dist/Mojolicious-Plugin-OpenAPI-Modern>
and a star in L<GitHub|https://github.com/karenetheridge/Mojolicious-Plugin-OpenAPI-Modern>.

=head1 SUPPORT

Bugs may be submitted through L<https://github.com/karenetheridge/Mojolicious-Plugin-OpenAPI-Modern/issues>.

I am also usually active on irc, as 'ether' at C<irc.perl.org> and C<irc.libera.chat>.

You can also find me on the L<JSON Schema Slack server|https://json-schema.slack.com> and
L<OpenAPI Slack server|https://open-api.slack.com>, which are also great resources for finding help.

=head1 AUTHOR

Karen Etheridge <ether@cpan.org>

=head1 COPYRIGHT AND LICENCE

This software is copyright (c) 2021 by Karen Etheridge.

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.

=cut
 readmeEtag: '"73b37b25194b5cc022f3871656619f847f027860"' readmeLastModified: Thu, 16 Oct 2025 18:06:18 GMT repositoryId: 436109317 description: Mojolicious plugin providing access to an OpenAPI document and parser created: '2021-12-08T03:55:07Z' updated: '2025-10-16T18:06:26Z' language: Perl archived: false stars: 3 watchers: 2 forks: 1 owner: karenetheridge logo: https://avatars.githubusercontent.com/u/303051?v=4 license: NOASSERTION repoEtag: '"6a0077e9413aa94536a19b7231c927be74d5417707e0ea51d256e723486eab4c"' repoLastModified: Thu, 16 Oct 2025 18:06:26 GMT foundInMaster: true - source: - https://openapi.tools/ - openapi3 tags name: widdershins category: - Documentation - Parsers link: https://mermade.github.io/shins repository: https://github.com/mermade/widdershins language: Node.js source_description: Generate Slate/Shins markdown from OpenAPI 2.0/3.0.x v2: true v3: true repositoryMetadata: base64Readme: >- # widdershins
OpenAPI / Swagger / AsyncAPI / Semoasa definition to [Slate](https://github.com/slatedocs/slate) /
[ReSlate](https://github.com/mermade/reslate) compatible markdown

![Build](https://img.shields.io/travis/Mermade/widdershins/master.svg) [![Tested on APIs.guru](https://api.apis.guru/badges/tested_on.svg)](https://APIs.guru) [![Tested on Mermade OpenAPIs](https://img.shields.io/badge/Additional%20Specs-419-brightgreen.svg)](https://github.com/mermade/OpenAPI_specifications)
[![Known Vulnerabilities](https://snyk.io/test/npm/widdershins/badge.svg)](https://snyk.io/test/npm/widdershins)

<img src="http://mermade.github.io/widdershins/logo.png" width="247px" height="250px" />

### Widdershins *adverb*:
* In a direction contrary to the sun's course;
* anticlockwise;
* helping you produce static documentation from your OpenAPI 3.0 / Swagger 2.0 / AsyncAPI 1.x / Semoasa 0.1.0 definition

![Widdershins screenshot](https://mermade.github.io/widdershins/screenshot.png)

### News

* Version 4.0 changes:
  * Now uses Promises not callbacks
  * Option to output html directly, and to ReSpec format
  * Unified JavaScript and Node.js code-samples, PHP added
  * `restrictions` column (`readOnly`/`writeOnly`) added to schema templates
  * Numerous bug fixes
* As of v3.0.0 Widdershins no longer expands the definition of OpenAPI body parameters / requestBodies by default, unless they have an inline schema. You can restore the old behaviour by using the `--expandBody` option.
* You may limit the depth of schema examples using the `--maxDepth` option. The default is 10.
* To omit schemas entirely, please copy and customise the `main.dot` template.
* As of v3.1.0 Widdershins includes a generated `Authorization` header in OpenAPI code samples. If you wish to omit this, see [here](/templates/openapi3/README.md).

### To install

* Clone the git repository, and `npm i` to install dependencies, or
* `npm install -g widdershins` to install globally

### Getting started

Widdershins is generally used as a stage in an API documentation pipeline. The pipeline begins with an API definition in OpenAPI 3.x, OpenAPI 2.0 (fka Swagger), API Blueprint, AsyncAPI or Semoasa format. Widdershins converts this description into markdown suitable for use by a **renderer**, such as [Slate](https://github.com/slatedocs/slate), [ReSlate](https://github.com/mermade/reslate), [Shins](https://github.com/mermade/shins)  (*deprecated*) or html suitable for use with [ReSpec](https://github.com/w3c/respec).

If you need to create your input API definition, [this list of available editors](https://apis.guru/awesome-openapi3/category.html#editors) may be useful.

More in-depth documentation is [available here](https://mermade.github.io/widdershins).

### Examples

```
node widdershins --search false --language_tabs 'ruby:Ruby' 'python:Python' --summary defs/petstore3.json -o petstore3.md
```

### Options

| CLI parameter name | JavaScript parameter name | Type | Default value | Description |
| --- | --- | --- | --- | --- |
| --clipboard | options.clipboard | `boolean` | `true` | Sets the value of the `code_clipboard` property in the heading, so that markdown processors can include clipboard support. |
| --customApiKeyValue | options.customApiKeyValue | `string` | `ApiKey` | Set a custom API key value to use as the API key in generated code examples. |
| --expandBody | options.expandBody | `boolean` | `false` | If a method's requestBody parameter refers to a schema by reference (not with a inline schema), by default, Widdershins shows only a reference to this parameter. Set this option to true to expand the schema and show all properties in the request body. |
| --headings | options.headings | `integer` | 2 | Set the value of the `headingLevel` parameter in the header so markdown processors know how many heading levels to show in the table of contents. Currently supported only by Shins, not by Slate, which lacks this feature. |
| --omitBody | options.omitBody | `boolean` | `false` | By default, Widdershins includes the body parameter as a row in the parameters table before the rows that represent the fields in the body. Set this parameter to omit that body parameter row. |
| --omitHeader | options.omitHeader | `boolean` | `false` | Omit the header / YAML front-matter in the generated Markdown file. |
| --resolve | options.resolve | `boolean` | `false` | Resolve external $refs, using the `source` parameter or the input file as the base location. |
| --shallowSchemas | options.shallowSchemas | `boolean` | `false` | When referring to a schema with a $ref, don't show the full contents of the schema. |
| N/A | options.source | `string` | None | The absolute location or URL of the source file to use as the base to resolve relative references ($refs) from; required if options.resolve is set to true. For CLI commands, Widdershins uses the input file as the base for the $refs. |
| --summary | options.tocSummary | `boolean` | `false` | Use the operation summary as the TOC entry instead of the ID. |
| --useBodyName | options.useBodyName | `boolean` | Use original param name for OpenAPI 2.0 body parameter. |
| -v, --verbose | options.verbose | `boolean` | `false` | Increase verbosity. |
| -h, --help | options.help | `boolean` | `false` | Show help. |
| --version | options.version | `boolean` | `false` | Show version number. |
| -c, --code | options.codeSamples | `boolean` | `false` | Omit generated code samples. |
| --httpsnippet | options.httpsnippet | `boolean` | `false` | Use httpsnippet to generate code samples. |
| -d, --discovery | options.discovery | `boolean` | `false` | Include schema.org WebAPI discovery data. |
| -e, --environment | N/A | `string` | None | File to load config options from. |
| -i, --includes | options.includes | `string` | None | List of files to put in the `include` header of the output Markdown. Processors such as Shins can then import the contents of these files. |
| -l, --lang | options.lang | `boolean` | `false` | Generate the list of languages for code samples based on the languages used in the source file's `x-code-samples` examples. |
| --language_tabs | options.language_tabs | `string` | (Differs for each input type) | List of language tabs for code samples using language[:label[:client]] format, such as `javascript:JavaScript:request`. |
| -m, --maxDepth | options.maxDepth | `integer` | 10 | Maximum depth to show for schema examples. |
| -o, --outfile | N/A | `string` | None | File to write the output markdown to. If left blank, Widdershins sends the output to stdout. |
| -r, --raw | **inverse** of options.sample | `boolean` | `false` | Output raw schemas instead of example values. |
| -s, --search | options.search | `boolean` | `true` | Set the value of the `search` parameter in the header so Markdown processors like Slate include search or not in their output. |
| -t, --theme | options.theme | `string` | darkula | Syntax-highlighter theme to use. |
| -u, --user_templates | options.user_templates | `string` | None | Directory to load override templates from. |
| -x, --experimental | options.experimental | `boolean` |  | Use httpSnippet for multipart mediatypes. |
| -y, --yaml | options.yaml | `boolean` | `false` | Display JSON schemas in YAML format. |
|  | options.templateCallback | `function` | None | A `function` that is called before and after each template (JavaScript code only). |
|  | options.toc_footers | `object` | A map of `url`s and `description`s to be added to the ToC footers array (JavaScript code only). |

In Node.JS code, create an options object and pass it to the Widdershins `convert` function, as in this example:

```javascript
const converter = require('widdershins');
let options = {}; // defaults shown
options.codeSamples = true;
options.httpsnippet = false;
//options.language_tabs = [];
//options.language_clients = [];
//options.loadedFrom = sourceUrl; // only needed if input document is relative
//options.user_templates = './user_templates';
options.templateCallback = function(templateName,stage,data) { return data };
options.theme = 'darkula';
options.search = true;
options.sample = true; // set false by --raw
options.discovery = false;
options.includes = [];
options.shallowSchemas = false;
options.tocSummary = false;
options.headings = 2;
options.yaml = false;
//options.resolve = false;
//options.source = sourceUrl; // if resolve is true, must be set to full path or URL of the input document
converter.convert(apiObj,options)
.then(str => {
  // str contains the converted markdown
})
.catch(err => {
  console.error(err);
});
```

To only include a subset of the pre-defined language-tabs, or to rename their display-names, you can override the `options.language_tabs`:

```javascript
options.language_tabs = [{ 'go': 'Go' }, { 'http': 'HTTP' }, { 'javascript': 'JavaScript' }, { 'javascript--node': 'Node.JS' }, { 'python': 'Python' }, { 'ruby': 'Ruby' }];
```

The `--environment` option specifies a JSON or YAML-formatted `options` object, for example:

```json
{
  "language_tabs": [{ "go": "Go" }, { "http": "HTTP" }, { "javascript": "JavaScript" }, { "javascript--node": "Node.JS" }, { "python": "Python" }, { "ruby": "Ruby" }],
  "verbose": true,
  "tagGroups": [
    {
      "title": "Companies",
      "tags": ["companies"]
    },
    {
      "title": "Billing",
      "tags": ["invoice-create", "invoice-close", "invoice-delete"]
    }
  ]
}
```

You can also use the environment file to group OAS/Swagger tagged paths together to create a more elegant table of contents, and overall page structure.

If you need to support a version of Slate \<v1.5.0 (or a renderer which also doesn't support display-names for language-tabs, such as `node-slate`, `slate-node` or `whiteboard`), you can use the `--environment` option with the included `whiteboard_env.json` file to simply achieve this.

If you are using the `httpsnippet` option to generate code samples, you can specify the client library used to perform the requests for each language by overriding the `options.language_clients`:

```javascript
options.language_clients = [{ 'shell': 'curl' }, { 'node': 'request' }, { 'java': 'unirest' }];
```

If the language name differs between the markdown name required to syntax highlight and the httpsnippet required target, both can be specified in the form `markdown--target`.

To see the list of languages and clients supported by httpsnippet, [click here](https://github.com/Kong/httpsnippet/tree/master/src/targets).

The `loadedFrom` option is only needed where the OpenAPI / Swagger definition does not specify a host, and (as per the OpenAPI [specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#fixed-fields)) the API endpoint is deemed to be based on the source URL
the definition was loaded from.

To see the list of highlight-js syntax highlighting themes, [click here](https://highlightjs.org/static/demo/).

Schema.org WebAPI discovery data is included if the `discovery` option above is set `true`. See the W3C [WebAPI Discovery Community Group](https://www.w3.org/community/web-api-discovery/) for more information.

## Language tabs

Widdershins supports the `x-code-samples` [vendor-extension](https://github.com/Rebilly/ReDoc/blob/master/docs/redoc-vendor-extensions.md#operation-object-vendor-extensions) to completely customise your documentation. Alternatively, you can edit the default code-samples in the `templates` sub-directory, or override them using the `user_templates` option to specify a directory containing your templates.

Widdershins supports the use of multiple language tabs with the same language (i.e. plain Javascript and Node.Js). To use this support you must be using Slate (or one of its ports compatible with) version 1.5.0 or higher.

## Templates

By default, Widdershins uses the templates in its `templates/` folder to generate the Markdown output. To customize the templates, copy some or all of them to a folder and pass their location to the `user_templates` parameter.

The templates include `.dot` templates and `.def` partials. To override a `.dot` template, you must copy it and the child `.def` partials that the template references. Similarly, to override a `.def` partial, you must also copy the parent `.dot` template. For OpenAPI 3, the primary template is `main.dot` and its main child partials are `parameters.def`, `responses.def`, and `callbacks.def`.

This means that it is usually easiest to copy all `.dot` and `.def` files to your user templates directory so you don't skip a template or partial. To bring in changes from Widdershins updates, you can use a visual `diff` tool which can run across two directories, such as [Meld](http://meldmerge.org/) or [WinMerge](http://winmerge.org).

### Template syntax

Templates are compiled with [doT.js](https://github.com/olado/doT#readme).

Templates have access to a `data` object with a range of properties based on the document context. For information about the parameters, see the README file for the appropriate templates:

* [Swagger 2.0 / OpenAPI 3.0.x template parameters](/templates/openapi3/README.md)
* [AsyncAPI 1.x template parameters](/templates/asyncapi1/README.md)
* [Semoasa 0.1.0 template parameters](/templates/semoasa/README.md)

To print the value of a parameter or variable in a template, use the code `{{=parameterName}}`. For example, to print the title of an OpenAPI 3 spec (from its `info.title` field), use the code `{{=data.api.info.title}}`.

To loop through values in an array, use the code `{{~ arrayName :tempVariable}}` to start the loop and the code `{{~}}` to close the loop. For example, the OpenAPI 3 partial `parameters.def` uses this code to create a table of the parameters in an operation:
```
|Name|In|Type|Required|Description|
|---|---|---|---|---|
{{~ data.parameters :p}}|{{=p.name}}|{{=p.in}}|{{=p.safeType}}|{{=p.required}}|{{=p.shortDesc || 'none'}}|
{{~}}
```

For if/then logic, use the code `{{? booleanExpression}}` to start the code block and the code `{{?}}` to close the block. For example, the OpenAPI 3 `main.dot` template calls the `security.def` partial to show information about the security schemes if the OpenAPI spec includes a `securitySchemes` section:
```
{{? data.api.components && data.api.components.securitySchemes }}
{{#def.security}}
{{?}}
```

You can run arbitrary JavaScript within a template by inserting a code block within curly braces. For example, this code creates a variable and references it with normal doT.js syntax later in the template:
```
{{ {
let message = "Hello!";
} }}

{{=message}}
```

### Template callbacks

The `templateCallback` parameter points to a function that Widdershins calls before and after each template runs. The callback function receives a `data` object that contains the spec that Widdershins is processing; the function must return this object. You can use callback functions only if you are calling Widdershins from JavaScript code, not from the command line.

Widdershins passes these variables to the callback function:
- `templateName`: The name of the template, such as `main`.
- `stage`: Whether Widdershins is calling the callback function before (`pre`) or after (`post`) the template.
- `data`: An object that contains the data that Widdershins is processing. You can mutate the `data` object in any way you see fit, but the function must return it whether it changes it or not. Content that you put in the `data.append` property is appended to the current output stream.

For example, this JavaScript code prints the name of the template and the processing stage in the output Markdown:
```javascript
'use strict';

const converter = require('widdershins');
const fs = require('fs');

let options = {};
options.templateCallback = myCallBackFunction;

function myCallBackFunction(templateName, stage, data) {
  let statusString = "Template name: " + templateName + "\n";
  statusString += "Stage: " + stage + "\n";
  data.append = statusString;
  return data;
}

const apiObj = JSON.parse(fs.readFileSync('defs/petstore3.json'));

converter.convert(apiObj, options)
.then(str => {
  fs.writeFileSync('petstore3Output.md', str, 'utf8');
});
```

## Tests

To run a test-suite:

```
node testRunner {path-to-APIs}
```

The test harness currently expects `.yaml` or `.json` files and has been tested against

* [APIs.guru](https://github.com/APIs-guru/OpenAPI-directory)
* [Mermade OpenAPI definitions collection](https://github.com/mermade/OpenAPI-definitions)

### Comparison between this and other OpenAPI / Swagger to Slate tools

[Blog posting](https://dev.to/mikeralphson/comparison-of-various-openapiswagger-to-slate-conversion-tools) by the author of Widdershins.

### Acknowledgements

* [@latgeek](https://github.com/LatGeek) for the logo.
* [@vfernandestoptal](https://github.com/vfernandestoptal) for the httpsnippet support.

### Widdershins in the wild

Please feel free to add a link to your API documentation here.

* [GOV.UK Content API v1.0.0](https://content-api.publishing.service.gov.uk/reference.html)
* [GOV UK Digital Marketplace API v1.0.0](https://alphagov.github.io/digitalmarketplace-api-docs/#digital-marketplace-api-v1-0-0)
* [Capital One API](https://www.capitalone.co.uk/developer/api/)
* [Cognite Data API](http://doc.cognitedata.com/)
* [SpeckleWorks API](https://speckleworks.github.io/SpeckleSpecs)
* [Bank by API](https://tbicr.github.io/bank-api/bank-api.html)
* [Open EO API](https://open-eo.github.io/openeo-api-poc/apireference/index.html)
* [Split Payments API](http://docs.split.cash/)
* [LeApp daemon API](https://leapp-to.github.io/shins/index.html)
* [Shutterstock API](https://api-reference.shutterstock.com/)
* [Shotstack Video Editing API](https://shotstack.io/docs/api/index.html)
* [Admetricks API](http://dev.admetricks.com/)
* [Eqivo API](https://eqivo.org/)

### Widdershins and ReSlate

* `Widdershins` works well with Slate, but for a solely Node.js-based experience, why not try the [ReSlate](https://github.com/mermade/reslate) port?
 readmeEtag: '"ffd46c7c9450860c650ad1506e1cf0f2f9953b28"' readmeLastModified: Fri, 26 Aug 2022 14:47:38 GMT repositoryId: 70106620 description: >- OpenAPI / Swagger, AsyncAPI & Semoasa definitions to (re)Slate compatible markdown created: '2016-10-05T23:09:13Z' updated: '2026-02-02T04:02:59Z' language: JavaScript archived: false stars: 1564 watchers: 15 forks: 321 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: MIT repoEtag: '"87b5c93178e3efdbbc97404d804aab3d5e657af6d3559c2981138606bf0a7c46"' repoLastModified: Mon, 02 Feb 2026 04:02:59 GMT foundInMaster: true homepage: https://github.com/Mermade/widdershins id: ab4fb414065011325683c7b582a9e582 - source: https://openapi.tools/ name: Zudoku category: Documentation link: https://zudoku.dev repository: https://github.com/zuplo/zudoku language: React source_description: >- OpenAPI powered, highly customizable, API reference and documentation framework with CDN packaged and self-hosted options. v2: false v3: true v3_1: true id: c804492e3311ef6b7453d6b28971b1eb repositoryMetadata: base64Readme: >- <div align=center>

<a href="https://zudoku.dev" alt="Zudoku">
  <img src="./assets/github-hero.png" width=630 alt="Zudoku Docs & Developer Portal">
</a>

[![MIT License](https://img.shields.io/badge/license-mit-green?style=flat&labelColor=000000)](https://github.com/zuplo/zudoku/license.md)
[![Zudoku Version](https://img.shields.io/npm/v/zudoku.svg?style=flat&labelColor=000000)](https://www.npmjs.com/package/zudoku)
[![Create Zudoku App Version](https://img.shields.io/npm/v/create-zudoku?label=cli&style=flat&labelColor=000000)](https://www.npmjs.com/package/create-zudoku)
[![Made with love by Zuplo](https://img.shields.io/badge/Made%20by%20Zuplo-FF00BD.svg?style=flat&logo=data:image/svg%2bxml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIGZpbGw9Im5vbmUiIHZpZXdCb3g9IjAgMCAzNyAzMiIgYXJpYS1oaWRkZW49InRydWUiPgogIDxwYXRoIGZpbGw9IiNGRjAwQkQiIGQ9Ik0yNy4xNDIgMTkuOTc4SDE2LjYyTDI3LjgzIDguNzQ2YS43NTguNzU4IDAgMDAtLjUzNC0xLjI5M0g5LjQ4OFYwaDE5LjUzNGE3LjU3MyA3LjU3MyAwIDAxNC4wNjUgMS4xMjUgNy41OTEgNy41OTEgMCAwMTIuODM2IDMuMTI2IDcuNDAyIDcuNDAyIDAgMDEtMS40NjEgOC4zOThsLTcuMzIgNy4zMjh6Ii8+CiAgPHBhdGggZmlsbD0iI0ZGMDBCRCIgZD0iTTkuNDg5IDExLjA0MmgxMC41MjRsLTExLjE5IDExLjIxYS43NzIuNzcyIDAgMDAuNTQzIDEuMzE2aDE3Ljc1OXY3LjQ1Mkg3LjYxYTcuNTc0IDcuNTc0IDAgMDEtNC4wNjUtMS4xMjVBNy41OTMgNy41OTMgMCAwMS43MSAyNi43NjhhNy40MDMgNy40MDMgMCAwMTEuNDYyLTguMzk3bDcuMzE4LTcuMzI5eiIvPgo8L3N2Zz4K&labelColor=000)](https://zuplo.com)

</div>

# Zudoku

API documentation should always be free.

<a href="#-installation"><strong>Installation</strong></a> ·
<a href="https://zudoku.dev/docs/quickstart"><strong>Docs</strong></a> ·
<a href="#-examples"><strong>Examples</strong></a> ·
<a href="#-contributing--community"><strong>Contributing</strong></a> ·
<a href="#-motivation"><strong>Motivation</strong></a>

## Introduction

**Zudoku** (pronounced "zoo-doh-koo") is an open-source, highly customizable API documentation
framework for building quality developer experiences around OpenAPI and, soon, GraphQL documents.

Because great API documentation frameworks should be:

🌍 Free & Open Source<br /> ✅ OpenAPI powered<br /> 🔩 Extensible with Plugins<br /> ⚡ Easy to
setup & blazing fast to work with<br /> 🔧 Easy to maintain

## 🤩 Try it, right now!

You can test Zudoku with your own OpenAPI document at
[zudoku.dev](https://zudoku.dev?utm_source=github&utm_medium=web&utm_content=link&ref=github) and
see how good your documentation can look!

## ✨ Features

- 🚀 Generate documentation from a single or multiple [OpenAPI](https://swagger.io/specification/)
  schema
- 📄 Create [MDX pages](https://mdxjs.com/) for anything you want to document
- 🔐 Integrate your users with authentication via OpenID or OAuth2
- 🧪 Let users test their API calls using the Integrated Playground (includes authentication!)
- 🌑 Dark mode (of course!)

Zudoku is quick to implement, easy to configure and is highly composable with sensible defaults.

## ⚙️ Installation

You can use the CLI to generate a new project or use it standalone via CDN as a React component.

### ⚡️ Quick start

Fire up your new API docs using the command line generator:

```
npm create zudoku@latest
```

### 📦 Standalone via CDN

Add the package and styles to your `<head>` and pass the URL for your API schema to the
`data-api-url` property, as shown here:

```html
<!doctype html>
<html>
  <head>
    <title>Zudoku Demo</title>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />

    <link rel="icon" type="image/svg+xml" href="https://cdn.zudoku.dev/logos/favicon.svg" />
    <script type="module" src="https://cdn.zudoku.dev/latest/main.js" crossorigin></script>
    <link rel="stylesheet" href="https://cdn.zudoku.dev/latest/style.css" crossorigin />
  </head>
  <body>
    <div data-api-url="https://api.example.com/openapi.json"></div>
  </body>
</html>
```

### 🧱 Getting started templates

To get started with some of the features Zudoku offers you can use one of these
[example templates](https://github.com/zuplo/zudoku/tree/main/examples):

| Template                                                                                | What it does                                          |
| --------------------------------------------------------------------------------------- | ----------------------------------------------------- |
| [many-apis](https://github.com/zuplo/zudoku/tree/main/examples/many-apis)               | Using more than one OpenAPI document with Zudoku      |
| [with-auth0](https://github.com/zuplo/zudoku/tree/main/examples/with-auth0)             | Authenticate users in docs with the Auth0 plugin      |
| [with-config](https://github.com/zuplo/zudoku/tree/main/examples/with-config)           | Barebones config, ready for you to setup how you like |
| [with-vite-config](https://github.com/zuplo/zudoku/tree/main/examples/with-vite-config) | Use Zudoku with your Vite config (Advanced)           |

## 🎓 Examples

- [Rick & Morty API](https://zudoku.dev/demo?api-url=https://rickandmorty.zuplo.io/openapi.json)
- [Pet Store API](https://zudoku.dev/demo?api-url=https://zudoku.dev/petstore.oas.json)
- [Zuplo API Documentation](https://zuplo.com/docs)

### Zudoku use cases

Zudoku is a flexible and highly customizable framework that can be used to create many things,
including:

- Standalone documentation websites
- OpenAPI powered API references
- A developer portal with documentation, fully functional API reference for testing and
  authentication support for your user accounts.
- Internal documentation

## 🔧 Contributing & Community

For details on how to contribute to Zudoku, see the [contributing guide](CONTRIBUTING.md).

## Changelog

Details of the latest updates to Zudoku can be found in the [changelog](CHANGELOG.md).

## 🎯 Motivation

At Zuplo, we couldn’t find an open-source solution that met our high standards for both
trustworthiness and programmability, so we decided to create our own. And since no one chooses Zuplo
solely because of our documentation, we felt great about open-sourcing this tool and making it easy
for anyone to self-host.

We hope that if you use it, you’ll think fondly of Zuplo, and one day, when you’re looking for a
gateway or API management product, you’ll consider us as a vendor to evaluate.

Zudoku will always be open-source. It will always be free.

## License

Zudoku is licensed under MIT. See the full [license](LICENSE.md).

<a href="https://twitter.com/zuplo">
  <img alt="X (formerly Twitter) Follow" src="https://img.shields.io/twitter/follow/zuplo">
</a>
 readmeEtag: '"8fc8f14245a1fe95d910a6d892cbace438e41357"' readmeLastModified: Mon, 06 Oct 2025 14:04:39 GMT repositoryId: 843437350 description: Framework for building high quality, interactive API documentation. created: '2024-08-16T14:09:50Z' updated: '2026-02-05T18:35:48Z' language: TypeScript archived: false stars: 445 watchers: 6 forks: 64 owner: zuplo logo: https://avatars.githubusercontent.com/u/85497839?v=4 license: MIT repoEtag: '"d1042fa14291abe0bb816d64e572c46bce32bf5f79faeb56704c09c11b97538f"' repoLastModified: Thu, 05 Feb 2026 18:35:48 GMT foundInMaster: true - source: openapi3 tags name: FastAPI homepage: https://github.com/tiangolo/fastapi language: Python source_description: >- OpenAPI 3 based, high performance, Python 3.6+ API framework with automatic data validation, serialization and great editor support. category: Server Implementations repository: https://github.com/fastapi/fastapi v3: true repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://fastapi.tiangolo.com"><img src="https://fastapi.tiangolo.com/img/logo-margin/logo-teal.png" alt="FastAPI"></a>
</p>
<p align="center">
    <em>FastAPI framework, high performance, easy to learn, fast to code, ready for production</em>
</p>
<p align="center">
<a href="https://github.com/fastapi/fastapi/actions?query=workflow%3ATest+event%3Apush+branch%3Amaster" target="_blank">
    <img src="https://github.com/fastapi/fastapi/actions/workflows/test.yml/badge.svg?event=push&branch=master" alt="Test">
</a>
<a href="https://coverage-badge.samuelcolvin.workers.dev/redirect/fastapi/fastapi" target="_blank">
    <img src="https://coverage-badge.samuelcolvin.workers.dev/fastapi/fastapi.svg" alt="Coverage">
</a>
<a href="https://pypi.org/project/fastapi" target="_blank">
    <img src="https://img.shields.io/pypi/v/fastapi?color=%2334D058&label=pypi%20package" alt="Package version">
</a>
<a href="https://pypi.org/project/fastapi" target="_blank">
    <img src="https://img.shields.io/pypi/pyversions/fastapi.svg?color=%2334D058" alt="Supported Python versions">
</a>
</p>

---

**Documentation**: <a href="https://fastapi.tiangolo.com" target="_blank">https://fastapi.tiangolo.com</a>

**Source Code**: <a href="https://github.com/fastapi/fastapi" target="_blank">https://github.com/fastapi/fastapi</a>

---

FastAPI is a modern, fast (high-performance), web framework for building APIs with Python based on standard Python type hints.

The key features are:

* **Fast**: Very high performance, on par with **NodeJS** and **Go** (thanks to Starlette and Pydantic). [One of the fastest Python frameworks available](#performance).
* **Fast to code**: Increase the speed to develop features by about 200% to 300%. *
* **Fewer bugs**: Reduce about 40% of human (developer) induced errors. *
* **Intuitive**: Great editor support. <abbr title="also known as auto-complete, autocompletion, IntelliSense">Completion</abbr> everywhere. Less time debugging.
* **Easy**: Designed to be easy to use and learn. Less time reading docs.
* **Short**: Minimize code duplication. Multiple features from each parameter declaration. Fewer bugs.
* **Robust**: Get production-ready code. With automatic interactive documentation.
* **Standards-based**: Based on (and fully compatible with) the open standards for APIs: <a href="https://github.com/OAI/OpenAPI-Specification" class="external-link" target="_blank">OpenAPI</a> (previously known as Swagger) and <a href="https://json-schema.org/" class="external-link" target="_blank">JSON Schema</a>.

<small>* estimation based on tests conducted by an internal development team, building production applications.</small>

## Sponsors

<!-- sponsors -->
### Keystone Sponsor

<a href="https://fastapicloud.com" target="_blank" title="FastAPI Cloud. By the same team behind FastAPI. You code. We Cloud."><img src="https://fastapi.tiangolo.com/img/sponsors/fastapicloud.png"></a>

### Gold and Silver Sponsors

<a href="https://blockbee.io?ref=fastapi" target="_blank" title="BlockBee Cryptocurrency Payment Gateway"><img src="https://fastapi.tiangolo.com/img/sponsors/blockbee.png"></a>
<a href="https://github.com/scalar/scalar/?utm_source=fastapi&utm_medium=website&utm_campaign=main-badge" target="_blank" title="Scalar: Beautiful Open-Source API References from Swagger/OpenAPI files"><img src="https://fastapi.tiangolo.com/img/sponsors/scalar.svg"></a>
<a href="https://www.propelauth.com/?utm_source=fastapi&utm_campaign=1223&utm_medium=mainbadge" target="_blank" title="Auth, user management and more for your B2B product"><img src="https://fastapi.tiangolo.com/img/sponsors/propelauth.png"></a>
<a href="https://zuplo.link/fastapi-gh" target="_blank" title="Zuplo: Deploy, Secure, Document, and Monetize your FastAPI"><img src="https://fastapi.tiangolo.com/img/sponsors/zuplo.png"></a>
<a href="https://liblab.com?utm_source=fastapi" target="_blank" title="liblab - Generate SDKs from FastAPI"><img src="https://fastapi.tiangolo.com/img/sponsors/liblab.png"></a>
<a href="https://docs.render.com/deploy-fastapi?utm_source=deploydoc&utm_medium=referral&utm_campaign=fastapi" target="_blank" title="Deploy & scale any full-stack web app on Render. Focus on building apps, not infra."><img src="https://fastapi.tiangolo.com/img/sponsors/render.svg"></a>
<a href="https://www.coderabbit.ai/?utm_source=fastapi&utm_medium=badge&utm_campaign=fastapi" target="_blank" title="Cut Code Review Time & Bugs in Half with CodeRabbit"><img src="https://fastapi.tiangolo.com/img/sponsors/coderabbit.png"></a>
<a href="https://subtotal.com/?utm_source=fastapi&utm_medium=sponsorship&utm_campaign=open-source" target="_blank" title="The Gold Standard in Retail Account Linking"><img src="https://fastapi.tiangolo.com/img/sponsors/subtotal.svg"></a>
<a href="https://docs.railway.com/guides/fastapi?utm_medium=integration&utm_source=docs&utm_campaign=fastapi" target="_blank" title="Deploy enterprise applications at startup speed"><img src="https://fastapi.tiangolo.com/img/sponsors/railway.png"></a>
<a href="https://serpapi.com/?utm_source=fastapi_website" target="_blank" title="SerpApi: Web Search API"><img src="https://fastapi.tiangolo.com/img/sponsors/serpapi.png"></a>
<a href="https://www.greptile.com/?utm_source=fastapi&utm_medium=sponsorship&utm_campaign=fastapi_sponsor_page" target="_blank" title="Greptile: The AI Code Reviewer"><img src="https://fastapi.tiangolo.com/img/sponsors/greptile.png"></a>
<a href="https://databento.com/?utm_source=fastapi&utm_medium=sponsor&utm_content=display" target="_blank" title="Pay as you go for market data"><img src="https://fastapi.tiangolo.com/img/sponsors/databento.svg"></a>
<a href="https://speakeasy.com/editor?utm_source=fastapi+repo&utm_medium=github+sponsorship" target="_blank" title="SDKs for your API | Speakeasy"><img src="https://fastapi.tiangolo.com/img/sponsors/speakeasy.png"></a>
<a href="https://www.svix.com/" target="_blank" title="Svix - Webhooks as a service"><img src="https://fastapi.tiangolo.com/img/sponsors/svix.svg"></a>
<a href="https://www.stainlessapi.com/?utm_source=fastapi&utm_medium=referral" target="_blank" title="Stainless | Generate best-in-class SDKs"><img src="https://fastapi.tiangolo.com/img/sponsors/stainless.png"></a>
<a href="https://www.permit.io/blog/implement-authorization-in-fastapi?utm_source=github&utm_medium=referral&utm_campaign=fastapi" target="_blank" title="Fine-Grained Authorization for FastAPI"><img src="https://fastapi.tiangolo.com/img/sponsors/permit.png"></a>
<a href="https://www.interviewpal.com/?utm_source=fastapi&utm_medium=open-source&utm_campaign=dev-hiring" target="_blank" title="InterviewPal - AI Interview Coach for Engineers and Devs"><img src="https://fastapi.tiangolo.com/img/sponsors/interviewpal.png"></a>
<a href="https://dribia.com/en/" target="_blank" title="Dribia - Data Science within your reach"><img src="https://fastapi.tiangolo.com/img/sponsors/dribia.png"></a>

<!-- /sponsors -->

<a href="https://fastapi.tiangolo.com/fastapi-people/#sponsors" class="external-link" target="_blank">Other sponsors</a>

## Opinions

"_[...] I'm using **FastAPI** a ton these days. [...] I'm actually planning to use it for all of my team's **ML services at Microsoft**. Some of them are getting integrated into the core **Windows** product and some **Office** products._"

<div style="text-align: right; margin-right: 10%;">Kabir Khan - <strong>Microsoft</strong> <a href="https://github.com/fastapi/fastapi/pull/26" target="_blank"><small>(ref)</small></a></div>

---

"_We adopted the **FastAPI** library to spawn a **REST** server that can be queried to obtain **predictions**. [for Ludwig]_"

<div style="text-align: right; margin-right: 10%;">Piero Molino, Yaroslav Dudin, and Sai Sumanth Miryala - <strong>Uber</strong> <a href="https://eng.uber.com/ludwig-v0-2/" target="_blank"><small>(ref)</small></a></div>

---

"_**Netflix** is pleased to announce the open-source release of our **crisis management** orchestration framework: **Dispatch**! [built with **FastAPI**]_"

<div style="text-align: right; margin-right: 10%;">Kevin Glisson, Marc Vilanova, Forest Monsen - <strong>Netflix</strong> <a href="https://netflixtechblog.com/introducing-dispatch-da4b8a2a8072" target="_blank"><small>(ref)</small></a></div>

---

"_I’m over the moon excited about **FastAPI**. It’s so fun!_"

<div style="text-align: right; margin-right: 10%;">Brian Okken - <strong><a href="https://pythonbytes.fm/episodes/show/123/time-to-right-the-py-wrongs?time_in_sec=855" target="_blank">Python Bytes</a> podcast host</strong> <a href="https://x.com/brianokken/status/1112220079972728832" target="_blank"><small>(ref)</small></a></div>

---

"_Honestly, what you've built looks super solid and polished. In many ways, it's what I wanted **Hug** to be - it's really inspiring to see someone build that._"

<div style="text-align: right; margin-right: 10%;">Timothy Crosley - <strong><a href="https://github.com/hugapi/hug" target="_blank">Hug</a> creator</strong> <a href="https://news.ycombinator.com/item?id=19455465" target="_blank"><small>(ref)</small></a></div>

---

"_If you're looking to learn one **modern framework** for building REST APIs, check out **FastAPI** [...] It's fast, easy to use and easy to learn [...]_"

"_We've switched over to **FastAPI** for our **APIs** [...] I think you'll like it [...]_"

<div style="text-align: right; margin-right: 10%;">Ines Montani - Matthew Honnibal - <strong><a href="https://explosion.ai" target="_blank">Explosion AI</a> founders - <a href="https://spacy.io" target="_blank">spaCy</a> creators</strong> <a href="https://x.com/_inesmontani/status/1144173225322143744" target="_blank"><small>(ref)</small></a> - <a href="https://x.com/honnibal/status/1144031421859655680" target="_blank"><small>(ref)</small></a></div>

---

"_If anyone is looking to build a production Python API, I would highly recommend **FastAPI**. It is **beautifully designed**, **simple to use** and **highly scalable**, it has become a **key component** in our API first development strategy and is driving many automations and services such as our Virtual TAC Engineer._"

<div style="text-align: right; margin-right: 10%;">Deon Pillsbury - <strong>Cisco</strong> <a href="https://www.linkedin.com/posts/deonpillsbury_cisco-cx-python-activity-6963242628536487936-trAp/" target="_blank"><small>(ref)</small></a></div>

---

## FastAPI mini documentary

There's a <a href="https://www.youtube.com/watch?v=mpR8ngthqiE" class="external-link" target="_blank">FastAPI mini documentary</a> released at the end of 2025, you can watch it online:

<a href="https://www.youtube.com/watch?v=mpR8ngthqiE" target="_blank"><img src="https://fastapi.tiangolo.com/img/fastapi-documentary.jpg" alt="FastAPI Mini Documentary"></a>

## **Typer**, the FastAPI of CLIs

<a href="https://typer.tiangolo.com" target="_blank"><img src="https://typer.tiangolo.com/img/logo-margin/logo-margin-vector.svg" style="width: 20%;"></a>

If you are building a <abbr title="Command Line Interface">CLI</abbr> app to be used in the terminal instead of a web API, check out <a href="https://typer.tiangolo.com/" class="external-link" target="_blank">**Typer**</a>.

**Typer** is FastAPI's little sibling. And it's intended to be the **FastAPI of CLIs**. ⌨️ 🚀

## Requirements

FastAPI stands on the shoulders of giants:

* <a href="https://www.starlette.dev/" class="external-link" target="_blank">Starlette</a> for the web parts.
* <a href="https://docs.pydantic.dev/" class="external-link" target="_blank">Pydantic</a> for the data parts.

## Installation

Create and activate a <a href="https://fastapi.tiangolo.com/virtual-environments/" class="external-link" target="_blank">virtual environment</a> and then install FastAPI:

<div class="termy">

```console
$ pip install "fastapi[standard]"

---> 100%
```

</div>

**Note**: Make sure you put `"fastapi[standard]"` in quotes to ensure it works in all terminals.

## Example

### Create it

Create a file `main.py` with:

```Python
from fastapi import FastAPI

app = FastAPI()


@app.get("/")
def read_root():
    return {"Hello": "World"}


@app.get("/items/{item_id}")
def read_item(item_id: int, q: str | None = None):
    return {"item_id": item_id, "q": q}
```

<details markdown="1">
<summary>Or use <code>async def</code>...</summary>

If your code uses `async` / `await`, use `async def`:

```Python hl_lines="7  12"
from fastapi import FastAPI

app = FastAPI()


@app.get("/")
async def read_root():
    return {"Hello": "World"}


@app.get("/items/{item_id}")
async def read_item(item_id: int, q: str | None = None):
    return {"item_id": item_id, "q": q}
```

**Note**:

If you don't know, check the _"In a hurry?"_ section about <a href="https://fastapi.tiangolo.com/async/#in-a-hurry" target="_blank">`async` and `await` in the docs</a>.

</details>

### Run it

Run the server with:

<div class="termy">

```console
$ fastapi dev main.py

 ╭────────── FastAPI CLI - Development mode ───────────╮
 │                                                     │
 │  Serving at: http://127.0.0.1:8000                  │
 │                                                     │
 │  API docs: http://127.0.0.1:8000/docs               │
 │                                                     │
 │  Running in development mode, for production use:   │
 │                                                     │
 │  fastapi run                                        │
 │                                                     │
 ╰─────────────────────────────────────────────────────╯

INFO:     Will watch for changes in these directories: ['/home/user/code/awesomeapp']
INFO:     Uvicorn running on http://127.0.0.1:8000 (Press CTRL+C to quit)
INFO:     Started reloader process [2248755] using WatchFiles
INFO:     Started server process [2248757]
INFO:     Waiting for application startup.
INFO:     Application startup complete.
```

</div>

<details markdown="1">
<summary>About the command <code>fastapi dev main.py</code>...</summary>

The command `fastapi dev` reads your `main.py` file, detects the **FastAPI** app in it, and starts a server using <a href="https://www.uvicorn.dev" class="external-link" target="_blank">Uvicorn</a>.

By default, `fastapi dev` will start with auto-reload enabled for local development.

You can read more about it in the <a href="https://fastapi.tiangolo.com/fastapi-cli/" target="_blank">FastAPI CLI docs</a>.

</details>

### Check it

Open your browser at <a href="http://127.0.0.1:8000/items/5?q=somequery" class="external-link" target="_blank">http://127.0.0.1:8000/items/5?q=somequery</a>.

You will see the JSON response as:

```JSON
{"item_id": 5, "q": "somequery"}
```

You already created an API that:

* Receives HTTP requests in the _paths_ `/` and `/items/{item_id}`.
* Both _paths_ take `GET` <em>operations</em> (also known as HTTP _methods_).
* The _path_ `/items/{item_id}` has a _path parameter_ `item_id` that should be an `int`.
* The _path_ `/items/{item_id}` has an optional `str` _query parameter_ `q`.

### Interactive API docs

Now go to <a href="http://127.0.0.1:8000/docs" class="external-link" target="_blank">http://127.0.0.1:8000/docs</a>.

You will see the automatic interactive API documentation (provided by <a href="https://github.com/swagger-api/swagger-ui" class="external-link" target="_blank">Swagger UI</a>):

![Swagger UI](https://fastapi.tiangolo.com/img/index/index-01-swagger-ui-simple.png)

### Alternative API docs

And now, go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>.

You will see the alternative automatic documentation (provided by <a href="https://github.com/Rebilly/ReDoc" class="external-link" target="_blank">ReDoc</a>):

![ReDoc](https://fastapi.tiangolo.com/img/index/index-02-redoc-simple.png)

## Example upgrade

Now modify the file `main.py` to receive a body from a `PUT` request.

Declare the body using standard Python types, thanks to Pydantic.

```Python hl_lines="2  7-10 23-25"
from fastapi import FastAPI
from pydantic import BaseModel

app = FastAPI()


class Item(BaseModel):
    name: str
    price: float
    is_offer: bool | None = None


@app.get("/")
def read_root():
    return {"Hello": "World"}


@app.get("/items/{item_id}")
def read_item(item_id: int, q: str | None = None):
    return {"item_id": item_id, "q": q}


@app.put("/items/{item_id}")
def update_item(item_id: int, item: Item):
    return {"item_name": item.name, "item_id": item_id}
```

The `fastapi dev` server should reload automatically.

### Interactive API docs upgrade

Now go to <a href="http://127.0.0.1:8000/docs" class="external-link" target="_blank">http://127.0.0.1:8000/docs</a>.

* The interactive API documentation will be automatically updated, including the new body:

![Swagger UI](https://fastapi.tiangolo.com/img/index/index-03-swagger-02.png)

* Click on the button "Try it out", it allows you to fill the parameters and directly interact with the API:

![Swagger UI interaction](https://fastapi.tiangolo.com/img/index/index-04-swagger-03.png)

* Then click on the "Execute" button, the user interface will communicate with your API, send the parameters, get the results and show them on the screen:

![Swagger UI interaction](https://fastapi.tiangolo.com/img/index/index-05-swagger-04.png)

### Alternative API docs upgrade

And now, go to <a href="http://127.0.0.1:8000/redoc" class="external-link" target="_blank">http://127.0.0.1:8000/redoc</a>.

* The alternative documentation will also reflect the new query parameter and body:

![ReDoc](https://fastapi.tiangolo.com/img/index/index-06-redoc-02.png)

### Recap

In summary, you declare **once** the types of parameters, body, etc. as function parameters.

You do that with standard modern Python types.

You don't have to learn a new syntax, the methods or classes of a specific library, etc.

Just standard **Python**.

For example, for an `int`:

```Python
item_id: int
```

or for a more complex `Item` model:

```Python
item: Item
```

...and with that single declaration you get:

* Editor support, including:
    * Completion.
    * Type checks.
* Validation of data:
    * Automatic and clear errors when the data is invalid.
    * Validation even for deeply nested JSON objects.
* <abbr title="also known as: serialization, parsing, marshalling">Conversion</abbr> of input data: coming from the network to Python data and types. Reading from:
    * JSON.
    * Path parameters.
    * Query parameters.
    * Cookies.
    * Headers.
    * Forms.
    * Files.
* <abbr title="also known as: serialization, parsing, marshalling">Conversion</abbr> of output data: converting from Python data and types to network data (as JSON):
    * Convert Python types (`str`, `int`, `float`, `bool`, `list`, etc).
    * `datetime` objects.
    * `UUID` objects.
    * Database models.
    * ...and many more.
* Automatic interactive API documentation, including 2 alternative user interfaces:
    * Swagger UI.
    * ReDoc.

---

Coming back to the previous code example, **FastAPI** will:

* Validate that there is an `item_id` in the path for `GET` and `PUT` requests.
* Validate that the `item_id` is of type `int` for `GET` and `PUT` requests.
    * If it is not, the client will see a useful, clear error.
* Check if there is an optional query parameter named `q` (as in `http://127.0.0.1:8000/items/foo?q=somequery`) for `GET` requests.
    * As the `q` parameter is declared with `= None`, it is optional.
    * Without the `None` it would be required (as is the body in the case with `PUT`).
* For `PUT` requests to `/items/{item_id}`, read the body as JSON:
    * Check that it has a required attribute `name` that should be a `str`.
    * Check that it has a required attribute `price` that has to be a `float`.
    * Check that it has an optional attribute `is_offer`, that should be a `bool`, if present.
    * All this would also work for deeply nested JSON objects.
* Convert from and to JSON automatically.
* Document everything with OpenAPI, that can be used by:
    * Interactive documentation systems.
    * Automatic client code generation systems, for many languages.
* Provide 2 interactive documentation web interfaces directly.

---

We just scratched the surface, but you already get the idea of how it all works.

Try changing the line with:

```Python
    return {"item_name": item.name, "item_id": item_id}
```

...from:

```Python
        ... "item_name": item.name ...
```

...to:

```Python
        ... "item_price": item.price ...
```

...and see how your editor will auto-complete the attributes and know their types:

![editor support](https://fastapi.tiangolo.com/img/vscode-completion.png)

For a more complete example including more features, see the <a href="https://fastapi.tiangolo.com/tutorial/">Tutorial - User Guide</a>.

**Spoiler alert**: the tutorial - user guide includes:

* Declaration of **parameters** from other different places as: **headers**, **cookies**, **form fields** and **files**.
* How to set **validation constraints** as `maximum_length` or `regex`.
* A very powerful and easy to use **<abbr title="also known as components, resources, providers, services, injectables">Dependency Injection</abbr>** system.
* Security and authentication, including support for **OAuth2** with **JWT tokens** and **HTTP Basic** auth.
* More advanced (but equally easy) techniques for declaring **deeply nested JSON models** (thanks to Pydantic).
* **GraphQL** integration with <a href="https://strawberry.rocks" class="external-link" target="_blank">Strawberry</a> and other libraries.
* Many extra features (thanks to Starlette) as:
    * **WebSockets**
    * extremely easy tests based on HTTPX and `pytest`
    * **CORS**
    * **Cookie Sessions**
    * ...and more.

### Deploy your app (optional)

You can optionally deploy your FastAPI app to <a href="https://fastapicloud.com" class="external-link" target="_blank">FastAPI Cloud</a>, go and join the waiting list if you haven't. 🚀

If you already have a **FastAPI Cloud** account (we invited you from the waiting list 😉), you can deploy your application with one command.

Before deploying, make sure you are logged in:

<div class="termy">

```console
$ fastapi login

You are logged in to FastAPI Cloud 🚀
```

</div>

Then deploy your app:

<div class="termy">

```console
$ fastapi deploy

Deploying to FastAPI Cloud...

✅ Deployment successful!

🐔 Ready the chicken! Your app is ready at https://myapp.fastapicloud.dev
```

</div>

That's it! Now you can access your app at that URL. ✨

#### About FastAPI Cloud

**<a href="https://fastapicloud.com" class="external-link" target="_blank">FastAPI Cloud</a>** is built by the same author and team behind **FastAPI**.

It streamlines the process of **building**, **deploying**, and **accessing** an API with minimal effort.

It brings the same **developer experience** of building apps with FastAPI to **deploying** them to the cloud. 🎉

FastAPI Cloud is the primary sponsor and funding provider for the *FastAPI and friends* open source projects. ✨

#### Deploy to other cloud providers

FastAPI is open source and based on standards. You can deploy FastAPI apps to any cloud provider you choose.

Follow your cloud provider's guides to deploy FastAPI apps with them. 🤓

## Performance

Independent TechEmpower benchmarks show **FastAPI** applications running under Uvicorn as <a href="https://www.techempower.com/benchmarks/#section=test&runid=7464e520-0dc2-473d-bd34-dbdfd7e85911&hw=ph&test=query&l=zijzen-7" class="external-link" target="_blank">one of the fastest Python frameworks available</a>, only below Starlette and Uvicorn themselves (used internally by FastAPI). (*)

To understand more about it, see the section <a href="https://fastapi.tiangolo.com/benchmarks/" class="internal-link" target="_blank">Benchmarks</a>.

## Dependencies

FastAPI depends on Pydantic and Starlette.

### `standard` Dependencies

When you install FastAPI with `pip install "fastapi[standard]"` it comes with the `standard` group of optional dependencies:

Used by Pydantic:

* <a href="https://github.com/JoshData/python-email-validator" target="_blank"><code>email-validator</code></a> - for email validation.

Used by Starlette:

* <a href="https://www.python-httpx.org" target="_blank"><code>httpx</code></a> - Required if you want to use the `TestClient`.
* <a href="https://jinja.palletsprojects.com" target="_blank"><code>jinja2</code></a> - Required if you want to use the default template configuration.
* <a href="https://github.com/Kludex/python-multipart" target="_blank"><code>python-multipart</code></a> - Required if you want to support form <abbr title="converting the string that comes from an HTTP request into Python data">"parsing"</abbr>, with `request.form()`.

Used by FastAPI:

* <a href="https://www.uvicorn.dev" target="_blank"><code>uvicorn</code></a> - for the server that loads and serves your application. This includes `uvicorn[standard]`, which includes some dependencies (e.g. `uvloop`) needed for high performance serving.
* `fastapi-cli[standard]` - to provide the `fastapi` command.
    * This includes `fastapi-cloud-cli`, which allows you to deploy your FastAPI application to <a href="https://fastapicloud.com" class="external-link" target="_blank">FastAPI Cloud</a>.

### Without `standard` Dependencies

If you don't want to include the `standard` optional dependencies, you can install with `pip install fastapi` instead of `pip install "fastapi[standard]"`.

### Without `fastapi-cloud-cli`

If you want to install FastAPI with the standard dependencies but without the `fastapi-cloud-cli`, you can install with `pip install "fastapi[standard-no-fastapi-cloud-cli]"`.

### Additional Optional Dependencies

There are some additional dependencies you might want to install.

Additional optional Pydantic dependencies:

* <a href="https://docs.pydantic.dev/latest/usage/pydantic_settings/" target="_blank"><code>pydantic-settings</code></a> - for settings management.
* <a href="https://docs.pydantic.dev/latest/usage/types/extra_types/extra_types/" target="_blank"><code>pydantic-extra-types</code></a> - for extra types to be used with Pydantic.

Additional optional FastAPI dependencies:

* <a href="https://github.com/ijl/orjson" target="_blank"><code>orjson</code></a> - Required if you want to use `ORJSONResponse`.
* <a href="https://github.com/esnme/ultrajson" target="_blank"><code>ujson</code></a> - Required if you want to use `UJSONResponse`.

## License

This project is licensed under the terms of the MIT license.
 readmeEtag: '"963de51edf98fcdb4aef937697fcce0d623daed6"' readmeLastModified: Wed, 04 Feb 2026 17:36:12 GMT repositoryId: 160919119 description: >- FastAPI framework, high performance, easy to learn, fast to code, ready for production created: '2018-12-08T08:21:47Z' updated: '2026-02-06T04:44:53Z' language: Python archived: false stars: 94838 watchers: 728 forks: 8632 owner: fastapi logo: https://avatars.githubusercontent.com/u/156354296?v=4 license: MIT repoEtag: '"1732ec16ea0224f93388a338b48c407cb8473d68fcfd8612df0fa5db289b3e3a"' repoLastModified: Fri, 06 Feb 2026 04:44:53 GMT foundInMaster: true id: 01f8d446752c39de698f67e8d2352229 oldLocations: - https://github.com/tiangolo/fastapi - source: - openapi3 tags - openapi31 tags repository: https://github.com/swagger-api/swagger-ui v3: true repositoryMetadata: base64Readme: >- # <img src="https://raw.githubusercontent.com/swagger-api/swagger.io/wordpress/images/assets/SWU-logo-clr.png" width="300">

[![NPM version](https://badge.fury.io/js/swagger-ui.svg)](http://badge.fury.io/js/swagger-ui)
[![Build Status](https://jenkins.swagger.io/view/OSS%20-%20JavaScript/job/oss-swagger-ui-master/badge/icon?subject=jenkins%20build)](https://jenkins.swagger.io/view/OSS%20-%20JavaScript/job/oss-swagger-ui-master/)
[![npm audit](https://jenkins.swagger.io/buildStatus/icon?job=oss-swagger-ui-security-audit&subject=npm%20audit)](https://jenkins.swagger.io/job/oss-swagger-ui-security-audit/lastBuild/console)
[![total GitHub contributors](https://img.shields.io/github/contributors-anon/swagger-api/swagger-ui.svg)](https://github.com/swagger-api/swagger-ui/graphs/contributors)

[![monthly npm installs](https://img.shields.io/npm/dm/swagger-ui.svg?label=npm%20downloads)](https://www.npmjs.com/package/swagger-ui)
![docker registry](https://img.shields.io/badge/docker-docker.swagger.io%2Fswaggerapi%2Fswagger--ui-blue)
![monthly packagist installs](https://img.shields.io/packagist/dm/swagger-api/swagger-ui.svg?label=packagist%20installs)
[![gzip size](https://img.shields.io/bundlephobia/minzip/swagger-ui.svg?label=gzip%20size)](https://bundlephobia.com/package/swagger-ui)

## Introduction
[Swagger UI](https://swagger.io/tools/swagger-ui/) allows anyone — be it your development team or your end consumers — to visualize and interact with the API’s resources without having any of the implementation logic in place. It’s automatically generated from your OpenAPI (formerly known as Swagger) Specification, with the visual documentation making it easy for back end implementation and client side consumption.

## General
**👉🏼 Want to score an easy open-source contribution?** Check out our [Good first issue](https://github.com/swagger-api/swagger-ui/issues?q=is%3Aissue+is%3Aopen+label%3A%22Good+first+issue%22) label.

**🕰️ Looking for the older version of Swagger UI?** Refer to the [*2.x* branch](https://github.com/swagger-api/swagger-ui/tree/2.x).


This repository publishes three different NPM modules:

* [swagger-ui](https://www.npmjs.com/package/swagger-ui) is a traditional npm module intended for use in single-page applications that are capable of resolving dependencies (via Webpack, Browserify, etc.).
* [swagger-ui-dist](https://www.npmjs.com/package/swagger-ui-dist) is a dependency-free module that includes everything you need to serve Swagger UI in a server-side project, or a single-page application that can't resolve npm module dependencies.
* [swagger-ui-react](https://www.npmjs.com/package/swagger-ui-react) is Swagger UI packaged as a React component for use in React applications.

We strongly suggest that you use `swagger-ui` instead of `swagger-ui-dist` if you're building a single-page application, since `swagger-ui-dist` is significantly larger.

If you are looking for plain ol' HTML/JS/CSS, [download the latest release](https://github.com/swagger-api/swagger-ui/releases/latest) and copy the contents of the `/dist` folder to your server.


## Compatibility
The OpenAPI Specification has undergone 5 revisions since initial creation in 2010.  Compatibility between Swagger UI and the OpenAPI Specification is as follows:

| Swagger UI Version | Release Date | OpenAPI Spec compatibility                           | Notes                                                                 |
|--------------------|--------------|------------------------------------------------------|-----------------------------------------------------------------------|
| 5.19.0             | 2025-02-17   | 2.0, 3.0.0, 3.0.1, 3.0.2, 3.0.3, 3.0.4, 3.1.0, 3.1.1 | [tag v5.19.0](https://github.com/swagger-api/swagger-ui/tree/v5.19.0) |
| 5.0.0              | 2023-06-12   | 2.0, 3.0.0, 3.0.1, 3.0.2, 3.0.3, 3.1.0               | [tag v5.0.0](https://github.com/swagger-api/swagger-ui/tree/v5.0.0)   |
| 4.0.0              | 2021-11-03   | 2.0, 3.0.0, 3.0.1, 3.0.2, 3.0.3                      | [tag v4.0.0](https://github.com/swagger-api/swagger-ui/tree/v4.0.0)   |
| 3.18.3             | 2018-08-03   | 2.0, 3.0.0, 3.0.1, 3.0.2, 3.0.3                      | [tag v3.18.3](https://github.com/swagger-api/swagger-ui/tree/v3.18.3) |
| 3.0.21             | 2017-07-26   | 2.0                                                  | [tag v3.0.21](https://github.com/swagger-api/swagger-ui/tree/v3.0.21) |
| 2.2.10             | 2017-01-04   | 1.1, 1.2, 2.0                                        | [tag v2.2.10](https://github.com/swagger-api/swagger-ui/tree/v2.2.10) |
| 2.1.5              | 2016-07-20   | 1.1, 1.2, 2.0                                        | [tag v2.1.5](https://github.com/swagger-api/swagger-ui/tree/v2.1.5)   |
| 2.0.24             | 2014-09-12   | 1.1, 1.2                                             | [tag v2.0.24](https://github.com/swagger-api/swagger-ui/tree/v2.0.24) |
| 1.0.13             | 2013-03-08   | 1.1, 1.2                                             | [tag v1.0.13](https://github.com/swagger-api/swagger-ui/tree/v1.0.13) |
| 1.0.1              | 2011-10-11   | 1.0, 1.1                                             | [tag v1.0.1](https://github.com/swagger-api/swagger-ui/tree/v1.0.1)   |

## Anonymized analytics

SwaggerUI uses [Scarf](https://scarf.sh/) to collect [anonymized installation analytics](https://github.com/scarf-sh/scarf-js?tab=readme-ov-file#as-a-user-of-a-package-using-scarf-js-what-information-does-scarf-js-send-about-me). These analytics help support the maintainers of this library and ONLY run during installation. To [opt out](https://github.com/scarf-sh/scarf-js?tab=readme-ov-file#as-a-user-of-a-package-using-scarf-js-how-can-i-opt-out-of-analytics), you can set the `scarfSettings.enabled` field to `false` in your project's `package.json`:

```
// package.json
{
  // ...
  "scarfSettings": {
    "enabled": false
  }
  // ...
}
```

Alternatively, you can set the environment variable `SCARF_ANALYTICS` to `false` as part of the environment that installs your npm packages, e.g., `SCARF_ANALYTICS=false npm install`.

## Documentation

#### Usage
- [Installation](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/installation.md)
- [Configuration](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/configuration.md)
- [CORS](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/cors.md)
- [OAuth2](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/oauth2.md)
- [Deep Linking](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/deep-linking.md)
- [Limitations](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/limitations.md)
- [Version detection](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/usage/version-detection.md)

#### Customization
- [Overview](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/customization/overview.md)
- [Plugin API](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/customization/plugin-api.md)
- [Custom layout](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/customization/custom-layout.md)

#### Development
- [Setting up](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/development/setting-up.md)
- [Scripts](https://github.com/swagger-api/swagger-ui/blob/HEAD/docs/development/scripts.md)

#### Contributing
- [Contributing](https://github.com/swagger-api/.github/blob/HEAD/CONTRIBUTING.md)

##### Integration Tests

You will need JDK of version 7 or higher as instructed here
https://nightwatchjs.org/guide/getting-started/installation.html#install-selenium-server

Integration tests can be run locally with `npm run e2e` - be sure you aren't running a dev server when testing!

### Browser support
Swagger UI works in the latest versions of Chrome, Safari, Firefox, and Edge.

### Known Issues

To help with the migration, here are the currently known issues with 3.X. This list will update regularly, and will not include features that were not implemented in previous versions.

- Only part of the parameters previously supported are available.
- The JSON Form Editor is not implemented.
- Support for `collectionFormat` is partial.
- l10n (translations) is not implemented.
- Relative path support for external files is not implemented.

## Security contact

Please disclose any security-related issues or vulnerabilities by emailing [security@swagger.io](mailto:security@swagger.io), instead of using the public issue tracker.

## License

SwaggerUI is licensed under [Apache 2.0 license](https://github.com/swagger-api/swagger-ui/blob/master/LICENSE).
SwaggerUI comes with an explicit [NOTICE](https://github.com/swagger-api/swagger-ui/blob/master/NOTICE) file
containing additional legal notices and information.
 readmeEtag: '"80bca1690eeec3be5fe4c5ca7f0bd4ce57a79b76"' readmeLastModified: Fri, 13 Jun 2025 07:18:33 GMT repositoryId: 2055965 description: >- Swagger UI is a collection of HTML, JavaScript, and CSS assets that dynamically generate beautiful documentation from a Swagger-compliant API. created: '2011-07-15T22:56:39Z' updated: '2026-02-05T06:37:24Z' language: JavaScript archived: false stars: 28607 watchers: 626 forks: 9256 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"d30d31bdf1bdf50bb031210c7904e3e3ff74f8a56282441ade7cd5185ef73b23"' repoLastModified: Thu, 05 Feb 2026 06:37:24 GMT foundInMaster: true name: swagger-ui homepage: https://github.com/swagger-api/swagger-UI language: JavaScript source_description: Web-Based interface for visualizing and testing OpenAPI\Swagger definitions category: - User Interfaces - Code Generators id: ae625180381304ae800725a36dde6320 v3_1: true - source: - openapi3 tags - openapi31 tags name: Redoc homepage: https://github.com/Redocly/redoc language: JavaScript source_description: >- A React-based renderer with deep support for OAS v2 and v3 and zero dev-dependency category: - User Interfaces - Parsers repository: https://github.com/redocly/redoc v3: true repositoryMetadata: base64Readme: >- <div align="center">
  <img alt="Redoc logo" src="https://raw.githubusercontent.com/Redocly/redoc/main//docs/images/redoc.png" width="400px" />

# Generate beautiful API documentation from OpenAPI

  [![npm](http://img.shields.io/npm/v/redoc.svg)](https://www.npmjs.com/package/redoc) [![License](https://img.shields.io/npm/l/redoc.svg)](https://github.com/Redocly/redoc/blob/main/LICENSE)

  [![bundle size](http://img.badgesize.io/https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js?compression=gzip&max=300000)](https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js) [![npm](https://img.shields.io/npm/dm/redoc.svg)](https://www.npmjs.com/package/redoc) [![jsDelivr status](https://data.jsdelivr.com/v1/package/npm/redoc/badge)](https://www.jsdelivr.com/package/npm/redoc)
</div>


## About Redoc

Redoc is an open source tool for generating documentation from OpenAPI (formerly Swagger) definitions.

By default Redoc offers a three-panel, responsive layout:

- The left panel contains a search bar and navigation menu.
- The central panel contains the documentation.
- The right panel contains request and response examples.

![Redoc demo](https://raw.githubusercontent.com/Redocly/redoc/main/demo/redoc-demo.png)

## Live demo

If you want to see how Redoc renders your OpenAPI definition,
you can try it out online at https://redocly.github.io/redoc/.

A version of the Swagger Petstore API is displayed by default.
To test it with your own OpenAPI definition,
enter the URL for your definition and select **TRY IT**.

## Redoc features

- Responsive three-panel design with menu/scrolling synchronization
- Support for OpenAPI 3.1, OpenAPI 3.0, and Swagger 2.0
- Ability to integrate your API introduction into the side menu
- High-level grouping in side menu with the [`x-tagGroups`](https://redocly.com/docs/api-reference-docs/specification-extensions/x-tag-groups/) specification extension
- [Simple integration with `create-react-app`](https://redocly.com/docs/redoc/quickstart/react/)
- Code samples support (with vendor extension) <br>
  ![code samples in action](docs/images/code-samples-demo.gif)

## Usage

Redoc is provided as a CLI tool (also distributed as a Docker image), HTML tag, and React component.

### Generate documentation from the CLI

If you have Node installed, quickly generate documentation using `npx`:

```bash
npx @redocly/cli build-docs openapi.yaml
```

The tool outputs by default to a file named `redoc-static.html` that you can open in your browser.

> [Redocly CLI](https://github.com/Redocly/redocly-cli/) does more than docs; check it out and add linting, bundling, and more to your API workflow.

### Add an HTML element to the page

Create an HTML page, or edit an existing one, and add the following within the body tags:

```html
    <redoc spec-url="http://petstore.swagger.io/v2/swagger.json"></redoc>
    <script src="https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js"> </script>
```

Open the HTML file in your browser, and your API documentation is shown on the page.

Add your own `spec-url` to the `<redoc>` tag; this attribute can also be a local file. The JavaScript library can also be installed locally using `npm` and served from your own server, see the [HTML deployment documentation](https://redocly.com/docs/redoc/deployment/html/) for more details.

### More usage options

Check out the [deployment documentation](./docs/deployment/intro.md) for more options, and detailed documentation for each.

## Redoc vs. Redocly API Reference

Redoc is Redocly's community-edition product. Looking for something more?
We also offer [hosted API reference documentation](https://redocly.com/docs/api-registry/guides/api-registry-quickstart/)
with additional features including:

* Try-it console
* Automated code samples
* Pagination
* Extra theme options

### Documentation and resources

- [Reference docs](https://redocly.com/docs/api-reference-docs/getting-started/) - we take care of the hosting
- [Redoc](https://redocly.com/docs/redoc/) - detailed documentation for this open source project (also in the `docs/` folder)
- [Command-line interface to bundle your docs into a web-ready HTML file](https://redocly.com/docs/cli/commands/build-docs/)
- API linting, bundling, and much more with open source [Redocly CLI](https://redocly.com/docs/cli)

## Showcase

A sample of the organizations using Redocly tools in the wild:

- [Rebilly](https://api-reference.rebilly.com/)
- [Docker Engine](https://docs.docker.com/engine/api/v1.25/)
- [Zuora](https://www.zuora.com/developer/api-reference/)
- [Discourse](http://docs.discourse.org)
- [Commbox](https://www.commbox.io/api/)
- [APIs.guru](https://apis.guru/api-doc/)
- [BoxKnight](https://www.docs.boxknight.com/)
- [Quaderno API](https://developers.quaderno.io/api)

_Pull requests to add your own API page to the list are welcome_

## Configuration

Redoc is highly configurable, see the [configuration documentation](docs/config.md) for details.

### OpenAPI specification extensions
Redoc uses the following [specification extensions](https://redocly.com/docs/api-reference-docs/spec-extensions/):

* [`x-logo`](docs/redoc-vendor-extensions.md#x-logo) - is used to specify API logo
* [`x-traitTag`](docs/redoc-vendor-extensions.md#x-traitTag) - useful for tags that refer to non-navigation properties like Pagination, Rate-Limits, etc
* [`x-codeSamples`](docs/redoc-vendor-extensions.md#x-codeSamples) - specify operation code samples
* [`x-badges`](docs/redoc-vendor-extensions.md#x-badges) - specify operation badges
* [`x-examples`](docs/redoc-vendor-extensions.md#x-examples) - specify JSON example for requests
* [`x-nullable`](docs/redoc-vendor-extensions.md#x-nullable) - mark schema param as a nullable
* [`x-displayName`](docs/redoc-vendor-extensions.md#x-displayname) - specify human-friendly names for the menu categories
* [`x-tagGroups`](docs/redoc-vendor-extensions.md#x-tagGroups) - group tags by categories in the side menu
* [`x-servers`](docs/redoc-vendor-extensions.md#x-servers) - ability to specify different servers for API (backported from OpenAPI 3.0)
* [`x-additionalPropertiesName`](docs/redoc-vendor-extensions.md#x-additionalPropertiesName) - ability to supply a descriptive name for the additional property keys
* [`x-summary`](docs/redoc-vendor-extensions.md#x-summary) - for Response object, use as the response button text, with description rendered under the button
* [`x-explicitMappingOnly`](docs/redoc-vendor-extensions.md#x-explicitMappingOnly) - in Schemas, display a more descriptive property name in objects with additionalProperties when viewing the property list with an object

## Releases

**The README for the `1.x` version is on the [v1.x](https://github.com/Redocly/redoc/tree/v1.x) branch.**

All the 2.x releases are deployed to npm and can be used with Redocly-cdn:
- particular release, for example, `v2.0.0`: https://cdn.redoc.ly/redoc/v2.0.0/bundles/redoc.standalone.js
- `latest` release: https://cdn.redoc.ly/redoc/latest/bundles/redoc.standalone.js

Additionally, all the 1.x releases are hosted on our GitHub Pages-based CDN **(deprecated)**:
- particular release, for example `v1.2.0`: https://rebilly.github.io/ReDoc/releases/v1.2.0/redoc.min.js
- `v1.x.x` release: https://rebilly.github.io/ReDoc/releases/v1.x.x/redoc.min.js
- `latest` release: https://rebilly.github.io/ReDoc/releases/latest/redoc.min.js - points to latest 1.x.x release since 2.x releases are not hosted on this CDN but on unpkg.


## Development
see [CONTRIBUTING.md](.github/CONTRIBUTING.md)
 readmeEtag: '"cb79ec89da4372119bc45d6ca47222a7458168f4"' readmeLastModified: Thu, 30 Jan 2025 12:23:07 GMT repositoryId: 45250726 description: 📘 OpenAPI/Swagger-generated API Reference Documentation created: '2015-10-30T12:53:02Z' updated: '2026-02-06T02:08:40Z' language: TypeScript archived: false stars: 25471 watchers: 295 forks: 2379 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"4138d7edf9238b8142544691e7c435f9ca1ad9293b21d75f82a76a580e7c29fa"' repoLastModified: Fri, 06 Feb 2026 02:08:40 GMT foundInMaster: true v3_1: true id: 2725d6db6977c1dd12ceaf9701787e07 oldLocations: - https://github.com/rebilly/redoc/ - source: openapi3 tags repository: https://github.com/swagger-api/swagger-core v3: true repositoryMetadata: base64Readme: >- # Swagger Core <img src="https://raw.githubusercontent.com/swagger-api/swagger.io/wordpress/images/assets/SW-logo-clr.png" height="50" align="right">

**NOTE:** If you're looking for Swagger Core 1.5.X and OpenAPI 2.0, please refer to [1.5 branch](https://github.com/swagger-api/swagger-core/tree/1.5).

**NOTE:** Since version 2.1.7, Swagger Core also supports the Jakarta namespace. There are a parallel set of artifacts with the `-jakarta` suffix, providing the same functionality as the unsuffixed (i.e.: `javax`) artifacts.
Please see the [Wiki](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---Getting-started) for more details.

**NOTE:** Since version 2.2.0 Swagger Core supports OpenAPI 3.1; see [this page](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---OpenAPI-3.1) for details

![Build Test Deploy](https://github.com/swagger-api/swagger-core/workflows/Build%20Test%20Deploy%20master/badge.svg?branch=master)
[![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.swagger.core.v3/swagger-project/badge.svg?style=plastic)](https://maven-badges.herokuapp.com/maven-central/io.swagger.core.v3/swagger-project)

Swagger Core is a Java implementation of the OpenAPI Specification. Current version supports *JAX-RS2* (`javax` and `jakarta` namespaces).

## Get started with Swagger Core!
See the guide on [getting started with Swagger Core](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---Getting-started) to get started with adding Swagger to your API.

## See the Wiki!
The [github wiki](https://github.com/swagger-api/swagger-core/wiki) contains documentation, samples, contributions, etc. Start there.

## Compatibility
The OpenAPI Specification has undergone several revisions since initial creation in 2010.  The Swagger Core project has the following compatibilities with the OpenAPI Specification:

Swagger core Version      | Release Date | OpenAPI Spec compatibility | Notes | Status
------------------------- | ------------ | -------------------------- | ----- | ----
2.2.42 (**current stable**)| 2026-01-19   | 3.x           | [tag v2.2.42](https://github.com/swagger-api/swagger-core/tree/v2.2.42) | Supported
2.2.41                     | 2025-11-24   | 3.x           | [tag v2.2.41](https://github.com/swagger-api/swagger-core/tree/v2.2.41) | Supported
2.2.40                     | 2025-10-28   | 3.x           | [tag v2.2.40](https://github.com/swagger-api/swagger-core/tree/v2.2.40) | Supported
2.2.39                     | 2025-10-13   | 3.x           | [tag v2.2.39](https://github.com/swagger-api/swagger-core/tree/v2.2.39) | Supported
2.2.38                     | 2025-09-29   | 3.x           | [tag v2.2.38](https://github.com/swagger-api/swagger-core/tree/v2.2.38) | Supported
2.2.37                     | 2025-09-16   | 3.x           | [tag v2.2.37](https://github.com/swagger-api/swagger-core/tree/v2.2.37) | Supported
2.2.36                     | 2025-08-18   | 3.x           | [tag v2.2.36](https://github.com/swagger-api/swagger-core/tree/v2.2.36) | Supported
2.2.35                     | 2025-07-31   | 3.x           | [tag v2.2.35](https://github.com/swagger-api/swagger-core/tree/v2.2.35) | Supported
2.2.34                     | 2025-06-20   | 3.x           | [tag v2.2.34](https://github.com/swagger-api/swagger-core/tree/v2.2.34) | Supported
2.2.33                     | 2025-06-12   | 3.x           | [tag v2.2.33](https://github.com/swagger-api/swagger-core/tree/v2.2.33) | Supported
2.2.32                     | 2025-05-14   | 3.x           | [tag v2.2.32](https://github.com/swagger-api/swagger-core/tree/v2.2.32) | Supported
2.2.31                     | 2025-05-13   | 3.x           | [tag v2.2.31](https://github.com/swagger-api/swagger-core/tree/v2.2.31) | Supported
2.2.30                     | 2025-04-07   | 3.x           | [tag v2.2.30](https://github.com/swagger-api/swagger-core/tree/v2.2.30) | Supported
2.2.29                     | 2025-03-10   | 3.x           | [tag v2.2.29](https://github.com/swagger-api/swagger-core/tree/v2.2.29) | Supported
2.2.28                     | 2025-01-16   | 3.x           | [tag v2.2.28](https://github.com/swagger-api/swagger-core/tree/v2.2.28) | Supported
2.2.27                     | 2024-12-11   | 3.x           | [tag v2.2.27](https://github.com/swagger-api/swagger-core/tree/v2.2.27) | Supported
2.2.26                     | 2024-11-18   | 3.x           | [tag v2.2.26](https://github.com/swagger-api/swagger-core/tree/v2.2.26) | Supported
2.2.25                     | 2024-10-02   | 3.x           | [tag v2.2.25](https://github.com/swagger-api/swagger-core/tree/v2.2.25) | Supported
2.2.24                     | 2024-09-23   | 3.x           | [tag v2.2.24](https://github.com/swagger-api/swagger-core/tree/v2.2.24) | Supported
2.2.23                     | 2024-08-28   | 3.x           | [tag v2.2.23](https://github.com/swagger-api/swagger-core/tree/v2.2.23) | Supported
2.2.22                     | 2024-05-15   | 3.x           | [tag v2.2.22](https://github.com/swagger-api/swagger-core/tree/v2.2.22) | Supported
2.2.21                     | 2024-03-20   | 3.x           | [tag v2.2.21](https://github.com/swagger-api/swagger-core/tree/v2.2.21) | Supported
2.2.20                     | 2023-12-19   | 3.x           | [tag v2.2.20](https://github.com/swagger-api/swagger-core/tree/v2.2.20) | Supported
2.2.19                     | 2023-11-10   | 3.x           | [tag v2.2.19](https://github.com/swagger-api/swagger-core/tree/v2.2.19) | Supported
2.2.18                     | 2023-10-25   | 3.x           | [tag v2.2.18](https://github.com/swagger-api/swagger-core/tree/v2.2.18) | Supported
2.2.17                     | 2023-10-12   | 3.x           | [tag v2.2.17](https://github.com/swagger-api/swagger-core/tree/v2.2.17) | Supported
2.2.16                     | 2023-09-18   | 3.x           | [tag v2.2.16](https://github.com/swagger-api/swagger-core/tree/v2.2.16) | Supported
2.2.15                     | 2023-07-08   | 3.x           | [tag v2.2.15](https://github.com/swagger-api/swagger-core/tree/v2.2.15) | Supported
2.2.14                     | 2023-06-26   | 3.x           | [tag v2.2.14](https://github.com/swagger-api/swagger-core/tree/v2.2.14) | Supported
2.2.13                     | 2023-06-24   | 3.x           | [tag v2.2.13](https://github.com/swagger-api/swagger-core/tree/v2.2.13) | Supported
2.2.12                     | 2023-06-13   | 3.x           | [tag v2.2.12](https://github.com/swagger-api/swagger-core/tree/v2.2.12) | Supported
2.2.11                     | 2023-06-01   | 3.x           | [tag v2.2.11](https://github.com/swagger-api/swagger-core/tree/v2.2.11) | Supported
2.2.10                     | 2023-05-15   | 3.x           | [tag v2.2.10](https://github.com/swagger-api/swagger-core/tree/v2.2.10) | Supported
2.2.9                     | 2023-03-20  | 3.x           | [tag v2.2.9](https://github.com/swagger-api/swagger-core/tree/v2.2.9)                                             | Supported
2.2.8                     | 2023-01-06  | 3.x           | [tag v2.2.8](https://github.com/swagger-api/swagger-core/tree/v2.2.8)                                             | Supported
2.2.7                     | 2022-11-15  | 3.0           | [tag v2.2.7](https://github.com/swagger-api/swagger-core/tree/v2.2.7)                                             | Supported
2.2.6                     | 2022-11-02  | 3.0           | [tag v2.2.6](https://github.com/swagger-api/swagger-core/tree/v2.2.6)                                             | Supported
2.2.5                     | 2022-11-02  | 3.0           | [tag v2.2.5](https://github.com/swagger-api/swagger-core/tree/v2.2.5)                                             | Supported
2.2.4                     | 2022-10-16  | 3.0           | [tag v2.2.4](https://github.com/swagger-api/swagger-core/tree/v2.2.4)                                             | Supported
2.2.3                     | 2022-09-27  | 3.0           | [tag v2.2.3](https://github.com/swagger-api/swagger-core/tree/v2.2.3)                                             | Supported
2.2.2                     | 2022-07-20  | 3.0           | [tag v2.2.2](https://github.com/swagger-api/swagger-core/tree/v2.2.2)                                             | Supported
2.2.1                     | 2022-06-15  | 3.0           | [tag v2.2.1](https://github.com/swagger-api/swagger-core/tree/v2.2.1)                                             | Supported
2.2.0                     | 2022-04-04  | 3.0           | [tag v2.2.0](https://github.com/swagger-api/swagger-core/tree/v2.2.0)                                             | Supported
2.1.13                     | 2022-02-07  | 3.0           | [tag v2.1.13](https://github.com/swagger-api/swagger-core/tree/v2.1.13)                                           | Supported
2.1.12                     | 2021-12-23  | 3.0           | [tag v2.1.12](https://github.com/swagger-api/swagger-core/tree/v2.1.12)                                           | Supported
2.1.11                     | 2021-09-29  | 3.0           | [tag v2.1.11](https://github.com/swagger-api/swagger-core/tree/v2.1.11)                                           | Supported
2.1.10                     | 2021-06-28  | 3.0           | [tag v2.1.10](https://github.com/swagger-api/swagger-core/tree/v2.1.10)                                           | Supported
2.1.9                     | 2021-04-20  | 3.0           | [tag v2.1.9](https://github.com/swagger-api/swagger-core/tree/v2.1.9)                                             | Supported
2.1.8                     | 2021-04-18  | 3.0           | [tag v2.1.8](https://github.com/swagger-api/swagger-core/tree/v2.1.8)                                             | Supported
2.1.7                     | 2021-02-18  | 3.0           | [tag v2.1.7](https://github.com/swagger-api/swagger-core/tree/v2.1.7)                                             | Supported
2.1.6                     | 2020-12-04  | 3.0           | [tag v2.1.6](https://github.com/swagger-api/swagger-core/tree/v2.1.6)                                             | Supported
2.1.5                     | 2020-10-01  | 3.0           | [tag v2.1.5](https://github.com/swagger-api/swagger-core/tree/v2.1.5)                                             | Supported
2.1.4                     | 2020-07-24  | 3.0           | [tag v2.1.4](https://github.com/swagger-api/swagger-core/tree/v2.1.4)                                             | Supported
2.1.3                     | 2020-06-27  | 3.0           | [tag v2.1.3](https://github.com/swagger-api/swagger-core/tree/v2.1.3)                                             | Supported
2.1.2                     | 2020-04-01  | 3.0           | [tag v2.1.2](https://github.com/swagger-api/swagger-core/tree/v2.1.2)                                             | Supported
2.1.1                     | 2020-01-02  | 3.0           | [tag v2.1.1](https://github.com/swagger-api/swagger-core/tree/v2.1.1)                                             | Supported
2.1.0                     | 2019-11-16  | 3.0           | [tag v2.1.0](https://github.com/swagger-api/swagger-core/tree/v2.1.0)                                             | Supported
2.0.10                    | 2019-10-11  | 3.0           | [tag v2.0.10](https://github.com/swagger-api/swagger-core/tree/v2.0.10)                                           | Supported
2.0.9                     | 2019-08-22  | 3.0           | [tag v2.0.9](https://github.com/swagger-api/swagger-core/tree/v2.0.9)                                             | Supported
2.0.8                     | 2019-04-24  | 3.0           | [tag v2.0.8](https://github.com/swagger-api/swagger-core/tree/v2.0.8)                                             | Supported
2.0.7                     | 2019-02-18  | 3.0           | [tag v2.0.7](https://github.com/swagger-api/swagger-core/tree/v2.0.7)                                             | Supported
2.0.6                     | 2018-11-27  | 3.0           | [tag v2.0.6](https://github.com/swagger-api/swagger-core/tree/v2.0.6)                                             | Supported
2.0.5                     | 2018-09-19  | 3.0           | [tag v2.0.5](https://github.com/swagger-api/swagger-core/tree/v2.0.5)                                             | Supported
2.0.4                     | 2018-09-05  | 3.0           | [tag v2.0.4](https://github.com/swagger-api/swagger-core/tree/v2.0.4)                                             | Supported
2.0.3                     | 2018-08-09  | 3.0           | [tag v2.0.3](https://github.com/swagger-api/swagger-core/tree/v2.0.3)                                             | Supported
1.6.14 (**current stable**)| 2024-03-19   | 2.0           | [tag v1.6.14](https://github.com/swagger-api/swagger-core/tree/v1.6.14)                                           | Supported
1.6.13                    | 2024-01-26   | 2.0           | [tag v1.6.13](https://github.com/swagger-api/swagger-core/tree/v1.6.13)                                           | Supported
1.6.12                    | 2023-10-14   | 2.0           | [tag v1.6.12](https://github.com/swagger-api/swagger-core/tree/v1.6.12)                                           | Supported
1.6.11                    | 2023-05-15  | 2.0           | [tag v1.6.11](https://github.com/swagger-api/swagger-core/tree/v1.6.11)                                           | Supported
1.6.10                    | 2023-03-21  | 2.0           | [tag v1.6.10](https://github.com/swagger-api/swagger-core/tree/v1.6.10)                                           | Supported
1.6.9                     | 2022-11-15  | 2.0           | [tag v1.6.9](https://github.com/swagger-api/swagger-core/tree/v1.6.9)                                             | Supported
1.6.8                     | 2022-10-16  | 2.0           | [tag v1.6.8](https://github.com/swagger-api/swagger-core/tree/v1.6.8)                                             | Supported
1.6.7                     | 2022-09-27  | 2.0           | [tag v1.6.7](https://github.com/swagger-api/swagger-core/tree/v1.6.7)                                             | Supported
1.6.6                     | 2022-04-04  | 2.0           | [tag v1.6.6](https://github.com/swagger-api/swagger-core/tree/v1.6.6)                                             | Supported
1.6.5                     | 2022-02-07  | 2.0           | [tag v1.6.5](https://github.com/swagger-api/swagger-core/tree/v1.6.5)                                             | Supported
1.6.4                     | 2021-12-23  | 2.0           | [tag v1.6.4](https://github.com/swagger-api/swagger-core/tree/v1.6.4)                                             | Supported
1.6.3                     | 2021-09-29  | 2.0           | [tag v1.6.3](https://github.com/swagger-api/swagger-core/tree/v1.6.3)                                             | Supported
1.6.2                     | 2020-07-01  | 2.0           | [tag v1.6.2](https://github.com/swagger-api/swagger-core/tree/v1.6.2)                                             | Supported
1.6.1                     | 2020-04-01  | 2.0           | [tag v1.6.1](https://github.com/swagger-api/swagger-core/tree/v1.6.1)                                             | Supported
1.6.0                     | 2019-11-16  | 2.0           | [tag v1.6.0](https://github.com/swagger-api/swagger-core/tree/v1.6.0)                                             | Supported
1.5.24                    | 2019-10-11  | 2.0           | [tag v1.5.24](https://github.com/swagger-api/swagger-core/tree/v1.5.24)                                           | Supported
1.5.23                    | 2019-08-22  | 2.0           | [tag v1.5.23](https://github.com/swagger-api/swagger-core/tree/v1.5.23)                                           | Supported
1.5.22                    | 2019-02-18  | 2.0           | [tag v1.5.22](https://github.com/swagger-api/swagger-core/tree/v1.5.22)                                           | Supported
1.5.21                    | 2018-08-09  | 2.0           | [tag v1.5.21](https://github.com/swagger-api/swagger-core/tree/v1.5.21)                                           | Supported
1.5.20                    | 2018-05-23  | 2.0           | [tag v1.5.20](https://github.com/swagger-api/swagger-core/tree/v1.5.20)                                           | Supported
2.0.2                     | 2018-05-23  | 3.0           | [tag v2.0.2](https://github.com/swagger-api/swagger-core/tree/v2.0.2)                                             | Supported
2.0.1                     | 2018-04-16  | 3.0           | [tag v2.0.1](https://github.com/swagger-api/swagger-core/tree/v2.0.1)                                             | Supported
1.5.19                    | 2018-04-16  | 2.0           | [tag v1.5.19](https://github.com/swagger-api/swagger-core/tree/v1.5.19)                                           | Supported
2.0.0                     | 2018-03-20  | 3.0           | [tag v2.0.0](https://github.com/swagger-api/swagger-core/tree/v2.0.0)                                             | Supported
2.0.0-rc4                 | 2018-01-22  | 3.0           | [tag v2.0.0-rc4](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc4)                                     | Supported
2.0.0-rc3                 | 2017-11-21  | 3.0           | [tag v2.0.0-rc3](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc3)                                     | Supported
2.0.0-rc2                 | 2017-09-29  | 3.0           | [tag v2.0.0-rc2](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc2)                                     | Supported
2.0.0-rc1                 | 2017-08-17  | 3.0           | [tag v2.0.0-rc1](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc1)                                     | Supported
1.5.18                    | 2018-01-22  | 2.0           | [tag v1.5.18](https://github.com/swagger-api/swagger-core/tree/v1.5.18)                                           | Supported
1.5.17                    | 2017-11-21  | 2.0           | [tag v1.5.17](https://github.com/swagger-api/swagger-core/tree/v1.5.17)                                           | Supported
1.5.16                    | 2017-07-15  | 2.0           | [tag v1.5.16](https://github.com/swagger-api/swagger-core/tree/v1.5.16)                                           | Supported
1.3.12                    | 2014-12-23  | 1.2           | [tag v1.3.12](https://github.com/swagger-api/swagger-core/tree/v1.3.12)                                           | Supported
1.2.4                     | 2013-06-19  | 1.1           | [tag swagger-project_2.10.0-1.2.4](https://github.com/swagger-api/swagger-core/tree/swagger-project_2.10.0-1.2.4) | Deprecated
1.0.0                     | 2011-10-16  | 1.0           | [tag v1.0](https://github.com/swagger-api/swagger-core/tree/v1.0)                                                 | Deprecated


### Change History
If you're interested in the change history of swagger and the Swagger Core framework, see [here](https://github.com/swagger-api/swagger-core/releases).

### Prerequisites
You need the following installed and available in your $PATH:

* Java 11
* Apache maven 3.0.4 or greater
* Jackson 2.4.5 or greater


### To build from source (currently 2.2.43-SNAPSHOT)
```
# first time building locally
mvn -N
```

Subsequent builds:
```
mvn install
```

This will build the modules.

Of course if you don't want to build locally you can grab artifacts from maven central:

`https://repo1.maven.org/maven2/io/swagger/core/`

## Sample Apps
The samples have moved to [a new repository](https://github.com/swagger-api/swagger-samples/tree/2.0) and contain various integrations and configurations.

## Security contact

Please disclose any security-related issues or vulnerabilities by emailing [security@swagger.io](mailto:security@swagger.io), instead of using the public issue tracker.
 readmeEtag: '"27cbc7c379b4a3e20ecccaaa8cddff40247bffe5"' readmeLastModified: Mon, 19 Jan 2026 14:58:21 GMT repositoryId: 2003641 description: >- Examples and server integrations for generating the Swagger API Specification, which enables easy access to your REST API created: '2011-07-05T23:44:11Z' updated: '2026-02-06T03:05:35Z' language: Java archived: false stars: 7516 watchers: 307 forks: 2249 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"2d6e76914a17c8cb9b177d2f09ecbdf7fcccb32aefc4c38e0d55c3fef7ddb502"' repoLastModified: Fri, 06 Feb 2026 03:05:35 GMT foundInMaster: true category: Code Generators id: 2093f5c5f56617123b73ae823791176d - source: openapi3 tags repository: https://github.com/mockoon/mockoon v3: true repositoryMetadata: base64Readme: >- <div align="center">
  <a href="https://mockoon.com" alt="mockoon logo">
    <img width="200" height="200" src="https://mockoon.com/images/logo-square-app.png">
  </a>
  <br>
  <a href="https://mockoon.com/"><img src="https://img.shields.io/badge/Website-Go-green.svg?style=flat-square&colorB=1997c6"/></a>
  <a href="https://mockoon.com/newsletter/"><img src="https://img.shields.io/badge/Newsletter-Subscribe-green.svg?style=flat-square"/></a>
  <br>
  <br>
  <h1>Mockoon: awesome API mocking</h1>
</div>

Mockoon is the easiest and quickest way to design and run mock APIs. No remote deployment, no account required, free and open-source.

It combines a [desktop application](https://mockoon.com/download/) to design and run mock servers locally, and a [CLI](https://mockoon.com/cli/) to self-host your fake APIs. A [cloud](#subscribe-to-mockoon-cloud) is also available to collaborate with your team, keep your data in sync, and deploy your mock APIs.

API mocking helps you speed up development and third-party API integration by reducing dependency on external services and their limitations: rate limits, costs, availability, etc.
It also allows you to test your applications in a controlled environment with predictable responses, status codes, and latencies, and easily simulate edge cases and error scenarios.
Finally, you can onboard new team members faster by providing them with a consistent and reliable environment to test and develop their applications.

➡️ [Download](https://mockoon.com/download/)

<div align="center">
  <img width="50%" src="https://mockoon.com/images/hero-repo.png">
</div>

## Features

Mockoon offers many features:

- Unlimited number of mock local servers and routes
- CLI to run your mock in headless environments, CI, etc.
- Complete control on routes definition: HTTP methods and statuses, regex paths, file serving, custom headers, etc.
- OpenAPI compatibility
- Record/logs of all entering and forwarded requests
- JSON templating
- Proxy forwarding mode
- HTTPS support

You can check the [complete list](https://mockoon.com/features/) on the website.

## Download the desktop application

You can get Mockoon desktop's [latest release](https://github.com/mockoon/mockoon/releases/latest) directly from this repository or on the official [website](https://mockoon.com/download/). Mockoon desktop is also available through:

MacOS:

- [_Homebrew_](https://formulae.brew.sh/cask/mockoon): `brew install --cask mockoon`.

Windows:

- _winget_: `winget install mockoon`.
- [_Chocolatey_](https://community.chocolatey.org/packages/mockoon): `choco install mockoon`.
- [_Windows Store_](https://www.microsoft.com/en-us/p/mockoon/9pk8dmsn00jj)

Linux:

- [_Snap store_](https://snapcraft.io/mockoon): `snap install mockoon`.
- [_AUR_](https://aur.archlinux.org/packages/mockoon-bin): `yay -S mockoon-bin`.

## Install the CLI

Mockoon CLI is available as an [NPM package](https://www.npmjs.com/package/@mockoon/cli). Please check our [dedicated documentation](https://github.com/mockoon/mockoon/blob/main/packages/cli/README.md) to learn how to install and use it.

## Use in cloud functions and serverless environments

Mockoon's Serverless [NPM package](https://www.npmjs.com/package/@mockoon/serverless) provides an easy way to run Mockoon's mock APIs in cloud functions and serverless environments: AWS Lambda, GCP Functions, Firebase Functions, etc.

Please check our [dedicated documentation](https://github.com/mockoon/mockoon/blob/main/packages/serverless/README.md) to learn how to use it.

## Support us!

Mockoon is proudly **independent** and **open-source**, maintained without external funding. We rely on both **sponsorships** and **Mockoon Cloud subscriptions** to keep improving the project and building new features. A **big thank you** to the companies below for supporting our work and helping us grow (and all the [sponsors](https://github.com/mockoon/mockoon/blob/main/backers.md) who helped this project over time!):

### Platinum

<div align="center" style="margin-top:20px;margin-bottom:20px;">
  <a href="https://github.blog/2023-04-12-github-accelerator-our-first-cohort-and-whats-next/">
      <picture>
      <source media="(prefers-color-scheme: dark)" srcset="https://mockoon.com/images/sponsors/light/github.png">
      <source media="(prefers-color-scheme: light)" srcset="https://mockoon.com/images/sponsors/github.png">
      <img src="https://mockoon.com/images/sponsors/light/github.png" alt="GitHub logo" />
      </picture>
  </a>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
  <a href="https://localazy.com/register?ref=a9CiDC61gOac-azO">
      <picture>
      <source media="(prefers-color-scheme: dark)" srcset="https://mockoon.com/images/sponsors/light/localazy.png">
      <source media="(prefers-color-scheme: light)" srcset="https://mockoon.com/images/sponsors/localazy.png">
      <img src="https://mockoon.com/images/sponsors/light/localazy.png" alt="Localazy logo" />
      </picture>
  </a>
</div>

### Gold

<div align="center" style="margin-top:20px;margin-bottom:20px;">
 
  <a href="https://www.testmu.ai/">
      <picture>
      <source media="(prefers-color-scheme: dark)" srcset="https://mockoon.com/images/sponsors/light/testmuai.png">
      <source media="(prefers-color-scheme: light)" srcset="https://mockoon.com/images/sponsors/testmuai.png">
      <img src="https://mockoon.com/images/sponsors/light/testmuai.png" alt="TestMu AI logo" />
      </picture>
  </a>
</div>

### Silver

<div align="center" style="margin-top:20px;margin-bottom:20px;">  
  <a href="https://www.emqx.io/">  
      <img src="https://mockoon.com/images/sponsors/emqx.png" alt="emqx logo" />
  </a>
</div>

If you'd like to **support Mockoon** as well, you can **become a sponsor** or **subscribe to Mockoon Cloud**, every contribution helps keep the project alive and evolving. Thank you!

<div align="center" style="margin-top:20px;margin-bottom:20px;">
<a href="https://github.com/sponsors/mockoon"><img src="https://mockoon.com/images/sponsor-btn.png?" width="250" alt="sponsor button" /></a>
</div>

## Subscribe to Mockoon Cloud

With advanced features for solo developers and teams, Mockoon Cloud supercharges your API development:

- ☁️ [cloud deployments](https://mockoon.com/cloud/docs/api-mock-cloud-deployments/)
- 🔄️ [data synchronization and real-time collaboration](https://mockoon.com/cloud/docs/data-synchronization-team-collaboration/)
- 🤖 [AI powered API mocking](https://mockoon.com/ai-powered-api-mocking/)
- 📃 Access to dozens of [ready-to-use JSON templates](https://mockoon.com/templates/).
- 💬 Priority support and training.

Upgrade today and take your API development to the next level.

<div align="center" style="margin-top:20px;margin-bottom:20px;">
<a href="https://mockoon.com/cloud/"><img src="https://mockoon.com/images/cloud-btn.png?" width="250" alt="cloud button" /></a>
</div>

## Mockoon's documentation

You will find Mockoon's [documentation](https://mockoon.com/docs/latest/about/) on the official website. It covers Mockoon's most complex features. Feel free to contribute or ask for new topics to be covered.

## Changelogs

You will find Mockoon applications [changelogs](https://mockoon.com/releases/) on the official website.

## Support/feedback

You can discuss all things related to Mockoon and ask for help on the [official community](https://github.com/mockoon/mockoon/discussions). It's also a good place to discuss bugs and feature requests before opening an issue on this repository.

## Contributing

If you are interested in contributing to Mockoon, please take a look at the [contributing guidelines](https://github.com/mockoon/mockoon/blob/main/CONTRIBUTING.md).

Please also take a look at our [Code of Conduct](https://github.com/mockoon/.github/blob/main/CODE_OF_CONDUCT.md).

## Roadmap

If you want to know what will be coming in the next release you can check the global [Roadmap](https://mockoon.com/public-roadmap/) or [subscribe to our newsletter](https://mockoon.com/newsletter/).
 readmeEtag: '"6e53bba6ac8e5249923f8e7af0942e4f77720b7f"' readmeLastModified: Fri, 23 Jan 2026 12:04:11 GMT repositoryId: 97478419 description: >- Mockoon is the easiest and quickest way to run mock APIs locally. No remote deployment, no account required, open source. created: '2017-07-17T13:19:16Z' updated: '2026-02-05T15:31:49Z' language: TypeScript archived: false stars: 8054 watchers: 62 forks: 451 owner: mockoon logo: https://avatars.githubusercontent.com/u/49429147?v=4 license: MIT repoEtag: '"34bb1554a3191296bd2a9e68dcbd9f73b20cd4d2db6cb770f6e13971253448fb"' repoLastModified: Thu, 05 Feb 2026 15:31:49 GMT foundInMaster: true category: Testing id: 8728dfca2c1908dbecdcddd31597fad4 - source: openapi3 tags name: OpenAPI Server Code Generator (oapi-codegen) category: - Code Generators link: https://github.com/deepmap/oapi-codegen repository: https://github.com/oapi-codegen/oapi-codegen language: Go source_description: >- Generate a client, server, and HTTP types for various Go HTTP servers, from an OpenAPI v3 specification v3: true repositoryMetadata: base64Readme: >- # `oapi-codegen`

[![OpenSSF Best Practices](https://www.bestpractices.dev/projects/9450/badge)](https://www.bestpractices.dev/projects/9450)

`oapi-codegen` is a command-line tool and library to convert OpenAPI specifications to Go code, be it [server-side implementations](#generating-server-side-boilerplate), [API clients](#generating-api-clients), or simply [HTTP models](#generating-api-models).

Using `oapi-codegen` allows you to reduce the boilerplate required to create or integrate with services based on [OpenAPI 3.0](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.0.md), and instead focus on writing your business logic, and working on the real value-add for your organisation.

With `oapi-codegen`, there are a few [Key Design Decisions](#key-design-decisions) we've made, including:

- idiomatic Go, where possible
- fairly simple generated code, erring on the side of duplicate code over nicely refactored code
- supporting as much of OpenAPI 3.x as is possible, alongside Go's type system

`oapi-codegen` is one part of a wider ecosystem, which can be found described in further detail in the [oapi-codegen organisation on GitHub](https://github.com/oapi-codegen).

⚠️ This README may be for the latest development version, which may contain unreleased changes. Please ensure you're looking at the README for the latest release version.

## Action Required: The repository for this project has changed

As announced in [May 2024](https://github.com/oapi-codegen/oapi-codegen/discussions/1605),
we have moved the project from the deepmap organization to our own organization, and you will need to update your
import paths to pull updates past this point. You need to do a recursive search/replace from
`github.com/deepmap/oapi-codegen/v2` to `github.com/oapi-codegen/oapi-codegen/v2`.

> [!IMPORTANT]
> `oapi-codegen` moved to its new home with the version tag `v2.3.0`.

If you are using `v2.2.0` or below, please install like so:

```sh
# for the binary install
go install github.com/deepmap/oapi-codegen/v2/cmd/oapi-codegen@v2.2.0
```

If you are using `v2.3.0` or above, please install like so, using the new module import path:

```sh
# for the binary install
go install github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen@latest
```

## Install

### For Go 1.24+

It is recommended to follow [the `go tool` support available from Go 1.24+](https://www.jvt.me/posts/2025/01/27/go-tools-124/) for managing the dependency of `oapi-codegen` alongside your core application.

To do this, you run `go get -tool`:

```sh
$ go get -tool github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen@latest
# this will then modify your `go.mod`
```

From there, each invocation of `oapi-codegen` would be used like so:

```go
//go:generate go tool oapi-codegen -config cfg.yaml ../../api.yaml
```

### Prior to Go 1.24

It is recommended to follow [the `tools.go` pattern](https://www.jvt.me/posts/2022/06/15/go-tools-dependency-management/) for managing the dependency of `oapi-codegen` alongside your core application.

This would give you a `tools/tools.go`:

```go
//go:build tools
// +build tools

package main

import (
	_ "github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen"
)
```

Then, each invocation of `oapi-codegen` would be used like so:

```go
//go:generate go run github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen --config=config.yaml ../../api.yaml
```

Alternatively, you can install it as a binary with:

```sh
$ go install github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen@latest
$ oapi-codegen -version
```

Which then means you can invoke it like so:

```go
//go:generate oapi-codegen --config=config.yaml ../../api.yaml
```

Note that you can also [move your `tools.go` into its own sub-module](https://www.jvt.me/posts/2024/09/30/go-tools-module/) to reduce the impact on your top-level `go.mod`.

### Pinning to commits

While the project does not ([yet](https://github.com/oapi-codegen/oapi-codegen/issues/1519)) have a defined release cadence, there may be cases where you want to pull in yet-unreleased changes to your codebase.

Therefore, you may want to pin your dependency on `oapi-codegen` to a given commit hash, rather than a tag.

This is **officially recommended** for consumers of `oapi-codegen`, who want features/bug fixes that haven't yet been released.

We aim to keep the default branch ready-to-release so you should be able to safely pin.

To do so, you can run:

```sh
# pin to the latest version on the default branch
$ go get github.com/oapi-codegen/oapi-codegen/v2@main
# alternatively, to a commit hash i.e. https://github.com/oapi-codegen/oapi-codegen/commit/71e916c59688a6379b5774dfe5904ec222b9a537
$ go get github.com/oapi-codegen/oapi-codegen/v2@71e916c59688a6379b5774dfe5904ec222b9a537
```

This will then make a change such as:

```diff
diff --git go.mod go.mod
index 44f29a4..436a780 100644
--- go.mod
+++ go.mod
@@ -2,21 +2,20 @@
-require github.com/oapi-codegen/oapi-codegen/v2 v2.1.0
+require github.com/oapi-codegen/oapi-codegen/v2 v2.1.1-0.20240331212514-80f0b978ef16
```

## Usage

`oapi-codegen` is largely configured using a YAML configuration file, to simplify the number of flags that users need to remember, and to make reading the `go:generate` command less daunting.

For full details of what is supported, it's worth checking out [the GoDoc for `codegen.Configuration`](https://pkg.go.dev/github.com/oapi-codegen/oapi-codegen/v2/pkg/codegen#Configuration).

We also have [a JSON Schema](configuration-schema.json) that can be used by IDEs/editors with the Language Server Protocol (LSP) to perform intelligent suggestions, i.e.:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: api
# ...
```

### Backwards compatibility

Although we strive to retain backwards compatibility - as a project that's using a stable API per SemVer - there are sometimes opportunities we must take to fix a bug that could cause a breaking change for [people relying upon the behaviour](https://xkcd.com/1172/).

In this case, we will expose a [compatibility option](https://pkg.go.dev/github.com/oapi-codegen/oapi-codegen/v2/pkg/codegen#CompatibilityOptions) to restore old behaviour.

## Features

At a high level, `oapi-codegen` supports:

- Generating server-side boilerplate for [a number of servers](#supported-servers) ([docs](#generating-server-side-boilerplate))
- Generating client API boilerplate ([docs](#generating-api-clients))
- Generating the types ([docs](#generating-api-models))
- Splitting large OpenAPI specs across multiple packages([docs](#import-mapping))
  - This is also known as "Import Mapping" or "external references" across our documentation / discussion in GitHub issues

## What does it look like?

Below we can see a trimmed down example taken from the OpenAPI Petstore [example](examples/petstore-expanded/stdhttp/api/petstore.gen.go):

```go
// generated code

type ServerInterface interface {
	// ...
	// Returns all pets
	// (GET /pets)
	FindPets(w http.ResponseWriter, r *http.Request, params FindPetsParams)
	// ...
}

// FindPets operation middleware
func (siw *ServerInterfaceWrapper) FindPets(w http.ResponseWriter, r *http.Request) {

	var err error

	// Parameter object where we will unmarshal all parameters from the context
	var params FindPetsParams

	// ------------- Optional query parameter "tags" -------------

	err = runtime.BindQueryParameter("form", true, false, "tags", r.URL.Query(), &params.Tags)
	if err != nil {
		siw.ErrorHandlerFunc(w, r, &InvalidParamFormatError{ParamName: "tags", Err: err})
		return
	}

	// ------------- Optional query parameter "limit" -------------

	err = runtime.BindQueryParameter("form", true, false, "limit", r.URL.Query(), &params.Limit)
	if err != nil {
		siw.ErrorHandlerFunc(w, r, &InvalidParamFormatError{ParamName: "limit", Err: err})
		return
	}

	handler := http.Handler(http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
		siw.Handler.FindPets(w, r, params)
	}))

	for _, middleware := range siw.HandlerMiddlewares {
		handler = middleware(handler)
	}

	handler.ServeHTTP(w, r)
}

// HandlerWithOptions creates http.Handler with additional options
func HandlerWithOptions(si ServerInterface, options StdHTTPServerOptions) http.Handler {
	m := options.BaseRouter

	if m == nil {
		m = http.NewServeMux()
	}
	if options.ErrorHandlerFunc == nil {
		options.ErrorHandlerFunc = func(w http.ResponseWriter, r *http.Request, err error) {
			http.Error(w, err.Error(), http.StatusBadRequest)
		}
	}

	wrapper := ServerInterfaceWrapper{
		Handler:            si,
		HandlerMiddlewares: options.Middlewares,
		ErrorHandlerFunc:   options.ErrorHandlerFunc,
	}

	m.HandleFunc("GET "+options.BaseURL+"/pets", wrapper.FindPets)

	return m
}
```

Then, in your own code, you implement the underlying logic for the `FindPets` implementation:

```go
type PetStore struct {
	Pets   map[int64]Pet
	NextId int64
	Lock   sync.Mutex
}

// Make sure we conform to ServerInterface

var _ ServerInterface = (*PetStore)(nil)

func NewPetStore() *PetStore {
	return &PetStore{
		Pets:   make(map[int64]Pet),
		NextId: 1000,
	}
}

// FindPets implements all the handlers in the ServerInterface
func (p *PetStore) FindPets(w http.ResponseWriter, r *http.Request, params FindPetsParams) {
	p.Lock.Lock()
	defer p.Lock.Unlock()

	var result []Pet

	for _, pet := range p.Pets {
		if params.Tags != nil {
			// If we have tags,  filter pets by tag
			for _, t := range *params.Tags {
				if pet.Tag != nil && (*pet.Tag == t) {
					result = append(result, pet)
				}
			}
		} else {
			// Add all pets if we're not filtering
			result = append(result, pet)
		}

		if params.Limit != nil {
			l := int(*params.Limit)
			if len(result) >= l {
				// We're at the limit
				break
			}
		}
	}

	w.WriteHeader(http.StatusOK)
	_ = json.NewEncoder(w).Encode(result)
}
```

As we can see, `oapi-codegen` simplifies some of the boilerplate by taking parameters out of the request and instead allows us to focus on the implementation.

You'll note that there's still a bit more marshaling of request/response data, which is further reduced by using the [Strict server](#strict-server) functionality.

<a name="what-does-it-look-like-strict"></a>
When using the strict server, you'll have the following generated code:

```go
// StrictServerInterface represents all server handlers.
type StrictServerInterface interface {
	// ...
	// Returns all pets
	// (GET /pets)
	FindPets(ctx context.Context, request FindPetsRequestObject) (FindPetsResponseObject, error)
	// ...
}

func NewStrictHandlerWithOptions(ssi StrictServerInterface, middlewares []StrictMiddlewareFunc, options StrictHTTPServerOptions) ServerInterface {
	return &strictHandler{ssi: ssi, middlewares: middlewares, options: options}
}

// FindPets operation middleware
func (sh *strictHandler) FindPets(w http.ResponseWriter, r *http.Request, params FindPetsParams) {
	var request FindPetsRequestObject

	request.Params = params

	handler := func(ctx context.Context, w http.ResponseWriter, r *http.Request, request interface{}) (interface{}, error) {
		return sh.ssi.FindPets(ctx, request.(FindPetsRequestObject))
	}
	for _, middleware := range sh.middlewares {
		handler = middleware(handler, "FindPets")
	}

	response, err := handler(r.Context(), w, r, request)

	if err != nil {
		sh.options.ResponseErrorHandlerFunc(w, r, err)
	} else if validResponse, ok := response.(FindPetsResponseObject); ok {
		if err := validResponse.VisitFindPetsResponse(w); err != nil {
			sh.options.ResponseErrorHandlerFunc(w, r, err)
		}
	} else if response != nil {
		sh.options.ResponseErrorHandlerFunc(w, r, fmt.Errorf("unexpected response type: %T", response))
	}
}
```

Then, in your own code, you implement the underlying logic for the `FindPets` implementation:

```go
// Make sure we conform to StrictServerInterface

var _ StrictServerInterface = (*PetStore)(nil)

func NewPetStore() *PetStore {
	return &PetStore{
		Pets:   make(map[int64]Pet),
		NextId: 1000,
	}
}

// FindPets implements all the handlers in the ServerInterface
func (p *PetStore) FindPets(ctx context.Context, request FindPetsRequestObject) (FindPetsResponseObject, error) {
	p.Lock.Lock()
	defer p.Lock.Unlock()

	var result []Pet

	for _, pet := range p.Pets {
		if request.Params.Tags != nil {
			// If we have tags,  filter pets by tag
			for _, t := range *request.Params.Tags {
				if pet.Tag != nil && (*pet.Tag == t) {
					result = append(result, pet)
				}
			}
		} else {
			// Add all pets if we're not filtering
			result = append(result, pet)
		}

		if request.Params.Limit != nil {
			l := int(*request.Params.Limit)
			if len(result) >= l {
				// We're at the limit
				break
			}
		}
	}

	return FindPets200JSONResponse(result), nil
}
```

We can see that this provides the best means to focus on the implementation of the business logic within the endpoint, rather than (un)marshalling types to and from JSON, or wrangling cookies or headers.

## Key design decisions

- Produce an interface that can be satisfied by your implementation, with reduced boilerplate
- Bulk processing and parsing of OpenAPI document in Go
- Resulting output is using Go's `text/template`s, which are user-overridable
- Attempts to produce Idiomatic Go
- Single-file output
- Support multiple OpenAPI files by having a package-per-OpenAPI file
- Support of OpenAPI 3.0
  - OpenAPI 3.1 support is [awaiting upstream support](https://github.com/oapi-codegen/oapi-codegen/issues/373)
  - Note that this does not include OpenAPI 2.0 (aka Swagger)
- Extract parameters from requests, to reduce work required by your implementation
- Implicit `additionalProperties` are ignored by default ([more details](#additional-properties-additionalproperties))
- Prune unused types by default

## Generating server-side boilerplate

`oapi-codegen` shines by making it fairly straightforward (note that this is a purposeful choice of wording here - we want to avoid words like "easy") to generate the server-side boilerplate for a backend API.

Below you can find the supported servers, and more information about how to implement a server using them.

To provide you a fully Test Driven Development style test harness to confirm you are following the specification, you could use a tool such as [openapi.tanna.dev/go/validator](https://openapi.tanna.dev/go/validator/), or craft your own.

### Supported Servers

Right now, we support the following servers, and are supportive of adding new servers, too!

<table>

<tr>
<th>
Server
</th>
<th>
<code>generate</code> flag to enable code generation
</th>
<th>
Example usage
</th>
</tr>

<tr>
<td>

[Chi](https://github.com/go-chi/chi)

</td>
<td>
<code>chi-server</code>
</td>
<td>


For a Chi server, you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  chi-server: true
  models: true
output: gen.go
```

To implement this, check out [the Chi docs](#impl-chi).

</td>
</tr>

<tr>
<td>

[Echo](https://github.com/labstack/echo)

</td>
<td>
<code>echo-server</code>
</td>
<td>

For an Echo server, you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  echo-server: true
  models: true
output: gen.go
```

To implement this, check out [the Echo docs](#impl-echo).

</td>
</tr>

<tr>
<td>

[Fiber](https://github.com/gofiber/fiber)

</td>
<td>
<code>fiber-server</code>
</td>

<td>

For a Fiber server, you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  fiber-server: true
  models: true
output: gen.go
```

To implement this, check out [the Fiber docs](#impl-fiber).

</td>
</tr>


<tr>
<td>

[Gin](https://github.com/gin-gonic/gin)

</td>
<td>
<code>gin-server</code>
</td>
<td>

For a Gin server, you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  gin-server: true
  models: true
output: gen.go
```

To implement this, check out [the Gin docs](#impl-gin).

</td>

</tr>

<tr>
<td>

[gorilla/mux](https://github.com/gorilla/mux)

</td>
<td>
<code>gorilla-server</code>
</td>

<td>

For a gorilla/mux server, you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  gorilla-server: true
  models: true
output: gen.go
```

To implement this, check out [the gorilla/mux docs](#impl-gorillamux).

</td>
</tr>

<tr>
<td>

[Iris](https://github.com/kataras/iris)

</td>
<td>
<code>iris-server</code>
</td>

<td>

For a Iris server, you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  iris-server: true
  models: true
output: gen.go
```

To implement this, check out [the Iris docs](#impl-iris).

</td>
</tr>

<tr>
<td>

[1.22+ `net/http`](https://pkg.go.dev/net/http)

</td>
<td>
<code>std-http-server</code>
</td>

<td>

To use purely `net/http` (for Go 1.22+), you will want a configuration file such as:

```yaml
# yaml-language-server: ...
package: api
generate:
  std-http-server: true
  models: true
output: gen.go
```

To implement this, check out [the Go 1.22+ `net/http` docs](#impl-stdhttp).

</td>
</tr>

</table>

### Go 1.22+ `net/http`
<a name="impl-stdhttp"></a>

As of Go 1.22, enhancements have been made to the routing of the `net/http` package in the standard library, which makes it a great starting point for implementing a server with, before needing to reach for another router or a full framework.

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(w http.ResponseWriter, r *http.Request)
}

func HandlerFromMux(si ServerInterface, m ServeMux) http.Handler {
	return HandlerWithOptions(si, StdHTTPServerOptions{
		BaseRouter: m,
	})
}

// HandlerWithOptions creates http.Handler with additional options
func HandlerWithOptions(si ServerInterface, options StdHTTPServerOptions) http.Handler {
	m := options.BaseRouter

	// ... omitted for brevity

	m.HandleFunc("GET "+options.BaseURL+"/ping", wrapper.GetPing)

	return m
}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/stdhttp/api/impl.go):

```go
import (
	"encoding/json"
	"net/http"
)

// optional code omitted

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(w http.ResponseWriter, r *http.Request) {
	resp := Pong{
		Ping: "pong",
	}

	w.WriteHeader(http.StatusOK)
	_ = json.NewEncoder(w).Encode(resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"
	"net/http"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/stdhttp/api"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	r := http.NewServeMux()

	// get an `http.Handler` that we can use
	h := api.HandlerFromMux(server, r)

	s := &http.Server{
		Handler: h,
		Addr:    "0.0.0.0:8080",
	}

	// And we serve HTTP until the world ends.
	log.Fatal(s.ListenAndServe())
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

> [!NOTE]
> If you feel like you've done everything right, but are still receiving `404 page not found` errors, make sure that you've got the `go` directive in your `go.mod` updated to:

```go.mod
go 1.22
```

</details>

### Chi
<a name="impl-chi"></a>

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(w http.ResponseWriter, r *http.Request)
}

// HandlerFromMux creates http.Handler with routing matching OpenAPI spec based on the provided mux.
func HandlerFromMux(si ServerInterface, r *mux.Router) http.Handler {
	return HandlerWithOptions(si, ChiServerOptions{
		BaseRouter: r,
	})
}

// HandlerWithOptions creates http.Handler with additional options
func HandlerWithOptions(si ServerInterface, options ChiServerOptions) http.Handler {
	r := options.BaseRouter

	// ...

	r.Group(func(r chi.Router) {
		r.Get(options.BaseURL+"/ping", wrapper.GetPing)
	})

	return r
}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/chi/api/impl.go):

```go
import (
	"encoding/json"
	"net/http"
)

// optional code omitted

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(w http.ResponseWriter, r *http.Request) {
	resp := Pong{
		Ping: "pong",
	}

	w.WriteHeader(http.StatusOK)
	_ = json.NewEncoder(w).Encode(resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"
	"net/http"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/chi/api"
	"github.com/go-chi/chi/v5"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	r := chi.NewMux()

	// get an `http.Handler` that we can use
	h := api.HandlerFromMux(server, r)

	s := &http.Server{
		Handler: h,
		Addr:    "0.0.0.0:8080",
	}

	// And we serve HTTP until the world ends.
	log.Fatal(s.ListenAndServe())
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

### gorilla/mux
<a name="impl-gorillamux"></a>

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(w http.ResponseWriter, r *http.Request)
}

// HandlerFromMux creates http.Handler with routing matching OpenAPI spec based on the provided mux.
func HandlerFromMux(si ServerInterface, r *mux.Router) http.Handler {
	return HandlerWithOptions(si, GorillaServerOptions{
		BaseRouter: r,
	})
}

// HandlerWithOptions creates http.Handler with additional options
func HandlerWithOptions(si ServerInterface, options GorillaServerOptions) http.Handler {
	r := options.BaseRouter

	// ...

	r.HandleFunc(options.BaseURL+"/ping", wrapper.GetPing).Methods("GET")

	return r
}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/gorillamux/api/impl.go):

```go
import (
	"encoding/json"
	"net/http"
)

// optional code omitted

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(w http.ResponseWriter, r *http.Request) {
	resp := Pong{
		Ping: "pong",
	}

	w.WriteHeader(http.StatusOK)
	_ = json.NewEncoder(w).Encode(resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"
	"net/http"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/gorillamux/api"
	"github.com/gorilla/mux"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	r := mux.NewRouter()

	// get an `http.Handler` that we can use
	h := api.HandlerFromMux(server, r)

	s := &http.Server{
		Handler: h,
		Addr:    "0.0.0.0:8080",
	}

	// And we serve HTTP until the world ends.
	log.Fatal(s.ListenAndServe())
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

### Echo server
<a name="impl-echo"></a>

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(ctx echo.Context) error
}

// This is a simple interface which specifies echo.Route addition functions which
// are present on both echo.Echo and echo.Group, since we want to allow using
// either of them for path registration
type EchoRouter interface {
	// ...
	GET(path string, h echo.HandlerFunc, m ...echo.MiddlewareFunc) *echo.Route
	// ...
}

// RegisterHandlers adds each server route to the EchoRouter.
func RegisterHandlers(router EchoRouter, si ServerInterface) {
	RegisterHandlersWithBaseURL(router, si, "")
}

// Registers handlers, and prepends BaseURL to the paths, so that the paths
// can be served under a prefix.
func RegisterHandlersWithBaseURL(router EchoRouter, si ServerInterface, baseURL string) {
	// ...

	router.GET(baseURL+"/ping", wrapper.GetPing)

}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/echo/api/impl.go):

```go
import (
	"net/http"

	"github.com/labstack/echo/v4"
)

// optional code omitted

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(ctx echo.Context) error {
	resp := Pong{
		Ping: "pong",
	}

	return ctx.JSON(http.StatusOK, resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/echo/api"
	"github.com/labstack/echo/v4"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	e := echo.New()

	api.RegisterHandlers(e, server)

	// And we serve HTTP until the world ends.
	log.Fatal(e.Start("0.0.0.0:8080"))
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

### Fiber server
<a name="impl-fiber"></a>

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(c *fiber.Ctx) error
}

// RegisterHandlers creates http.Handler with routing matching OpenAPI spec.
func RegisterHandlers(router fiber.Router, si ServerInterface) {
	RegisterHandlersWithOptions(router, si, FiberServerOptions{})
}

// RegisterHandlersWithOptions creates http.Handler with additional options
func RegisterHandlersWithOptions(router fiber.Router, si ServerInterface, options FiberServerOptions) {
	// ...

	router.Get(options.BaseURL+"/ping", wrapper.GetPing)
}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/fiber/api/impl.go):

```go
import (
	"net/http"

	"github.com/gofiber/fiber/v2"
)

// ensure that we've conformed to the `ServerInterface` with a compile-time check
var _ ServerInterface = (*Server)(nil)

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(ctx *fiber.Ctx) error {
	resp := Pong{
		Ping: "pong",
	}

	return ctx.
		Status(http.StatusOK).
		JSON(resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/fiber/api"
	"github.com/gofiber/fiber/v2"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	app := fiber.New()

	api.RegisterHandlers(app, server)

	// And we serve HTTP until the world ends.
	log.Fatal(app.Listen("0.0.0.0:8080"))
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

### Gin server
<a name="impl-gin"></a>

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(c *gin.Context)
}

// RegisterHandlers creates http.Handler with routing matching OpenAPI spec.
func RegisterHandlers(router gin.IRouter, si ServerInterface) {
	RegisterHandlersWithOptions(router, si, GinServerOptions{})
}

// RegisterHandlersWithOptions creates http.Handler with additional options
func RegisterHandlersWithOptions(router gin.IRouter, si ServerInterface, options GinServerOptions) {
	// ...

	router.GET(options.BaseURL+"/ping", wrapper.GetPing)
}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/gorillamux/api/impl.go):

```go
import (
	"net/http"

	"github.com/gin-gonic/gin"
)

// optional code omitted

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(ctx *gin.Context) {
	resp := Pong{
		Ping: "pong",
	}

	ctx.JSON(http.StatusOK, resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"
	"net/http"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/gin/api"
	"github.com/gin-gonic/gin"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	r := gin.Default()

	api.RegisterHandlers(r, server)

	// And we serve HTTP until the world ends.

	s := &http.Server{
		Handler: r,
		Addr:    "0.0.0.0:8080",
	}

	// And we serve HTTP until the world ends.
	log.Fatal(s.ListenAndServe())
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

### Iris server
<a name="impl-iris"></a>

For instance, let's take this straightforward specification:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Minimal ping API server
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
components:
  schemas:
    # base types
    Pong:
      type: object
      required:
        - ping
      properties:
        ping:
          type: string
          example: pong
```

This then generates code such as:

```go
// Pong defines model for Pong.
type Pong struct {
	Ping string `json:"ping"`
}

// ServerInterface represents all server handlers.
type ServerInterface interface {

	// (GET /ping)
	GetPing(ctx iris.Context)
}

// RegisterHandlers creates http.Handler with routing matching OpenAPI spec.
func RegisterHandlers(router *iris.Application, si ServerInterface) {
	RegisterHandlersWithOptions(router, si, IrisServerOptions{})
}

// RegisterHandlersWithOptions creates http.Handler with additional options
func RegisterHandlersWithOptions(router *iris.Application, si ServerInterface, options IrisServerOptions) {
	// ...

	router.Get(options.BaseURL+"/ping", wrapper.GetPing)

	router.Build()
}
```

To implement this HTTP server, we need to write the following code in our [`api/impl.go`](examples/minimal-server/gorillamux/api/impl.go):

```go
import (
	"net/http"

	"github.com/kataras/iris/v12"
)

// optional code omitted

type Server struct{}

func NewServer() Server {
	return Server{}
}

// (GET /ping)
func (Server) GetPing(ctx iris.Context) {
	resp := Pong{
		Ping: "pong",
	}

	ctx.StatusCode(http.StatusOK)
	_ = ctx.JSON(resp)
}
```

Now we've got our implementation, we can then write the following code to wire it up and get a running server:

```go
import (
	"log"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/minimal-server/iris/api"
	"github.com/kataras/iris/v12"
)

func main() {
	// create a type that satisfies the `api.ServerInterface`, which contains an implementation of every operation from the generated code
	server := api.NewServer()

	i := iris.Default()

	api.RegisterHandlers(i, server)

	// And we serve HTTP until the world ends.
	log.Fatal(i.Listen("0.0.0.0:8080"))
}
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

### Strict server

`oapi-codegen` also supports generating a server that is much more strict with the contract that the implementer requires, and takes inspiration from server-side code generation for RPC servers.

This takes the boilerplate reduction from the non-strict servers and adds additional boilerplate reduction, allowing you to make the following changes to your function signatures:

```diff
-FindPets(w http.ResponseWriter, r *http.Request, params FindPetsParams)
+FindPets(ctx context.Context, request FindPetsRequestObject) (FindPetsResponseObject, error)
```

This is the highest level of strictness that `oapi-codegen` supports right now, and it's a good idea to start with this if you want the most guardrails to simplify developing your APIs.

The strict server has support for:

- multiple request/response media types and status codes on a given operation
- first-class support for `multipart/form-data` and `application/x-www-form-urlencoded` requests
- returning an [HTTP 500 Internal Server Error](https://http.cat/500), when an `error` is returned from a function
- automagic (un)marshalling of request/responses, and setting `content-type` and HTTP status codes on responses
- binding request values to a struct, a `multipart.Reader` or providing a `io.Reader`

You can see a little more detail of the generated code in the ["What does it look like"](#what-does-it-look-like-strict) section.

> [!NOTE]
> To configure the strict server generation, you must specify another server to be generated. For instance:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: api
generate:
  # NOTE another server must be added!
  chi-server: true
  strict-server: true
output: server.gen.go
```

> [!NOTE]
> This doesn't include [validation of incoming requests](#requestresponse-validation-middleware).

## Generating API clients

As well as generating the server-side boilerplate, `oapi-codegen` can also generate API clients.

This aims to be an API client that can be used to interact with the methods of the API, and is perfectly suited for production usage.

However, if you were looking for a slightly more SDK-style approach, or a mix of generated tests and/or documentation, this API client may not be for you, and you may want to look at alternate tooling.

For instance, given an `api.yaml`:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Generate models
paths:
  /client:
    get:
      operationId: getClient
      responses:
        200:
          content:
            application/json:
              schema:
                $ref: "#/components/schemas/ClientType"
    put:
      operationId: updateClient
      responses:
        400:
          content:
            application/json:
              schema:
                type: object
                properties:
                  code:
                    type: string
                required:
                - code
components:
  schemas:
    ClientType:
      type: object
      required:
        - name
      properties:
        name:
          type: string
    # NOTE that this is not generated by default because it's not referenced. If you want it, you need to use the following YAML configuration:
    #
    # output-options:
    #   skip-prune: true
    Unreferenced:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
```

And a `cfg.yaml`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: client
output: client.gen.go
generate:
  models: true
  client: true
```

And a `generate.go`:

```go
package client

//go:generate go run github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen -config cfg.yaml api.yaml
```

This would then generate:

```go
package client

// ...

// ClientType defines model for ClientType.
type ClientType struct {
	Name string `json:"name"`
}

// ...

// Client which conforms to the OpenAPI3 specification for this service.
type Client struct {
	// The endpoint of the server conforming to this interface, with scheme,
	// https://api.deepmap.com for example. This can contain a path relative
	// to the server, such as https://api.deepmap.com/dev-test, and all the
	// paths in the swagger spec will be appended to the server.
	Server string

	// Doer for performing requests, typically a *http.Client with any
	// customized settings, such as certificate chains.
	Client HttpRequestDoer

	// A list of callbacks for modifying requests which are generated before sending over
	// the network.
	RequestEditors []RequestEditorFn
}

// ...

// The interface specification for the client above.
type ClientInterface interface {
	// GetClient request
	GetClient(ctx context.Context, reqEditors ...RequestEditorFn) (*http.Response, error)

	// UpdateClient request
	UpdateClient(ctx context.Context, reqEditors ...RequestEditorFn) (*http.Response, error)
}

// ...

// ClientWithResponsesInterface is the interface specification for the client with responses above.
type ClientWithResponsesInterface interface {
	// GetClientWithResponse request
	GetClientWithResponse(ctx context.Context, reqEditors ...RequestEditorFn) (*GetClientResponse, error)

	// UpdateClientWithResponse request
	UpdateClientWithResponse(ctx context.Context, reqEditors ...RequestEditorFn) (*UpdateClientResponse, error)
}

type GetClientResponse struct {
	Body         []byte
	HTTPResponse *http.Response
	JSON200      *ClientType
}

// ...
```

With this generated client, it is then possible to construct and utilise the client, for instance:

```go
package client_test

import (
	"context"
	"fmt"
	"log"
	"net/http"

	"github.com/oapi-codegen/oapi-codegen/v2/examples/client"
)

func TestClient_canCall() {
	// custom HTTP client
	hc := http.Client{}

	// with a raw http.Response
	{
		c, err := client.NewClient("http://localhost:1234", client.WithHTTPClient(&hc))
		if err != nil {
			log.Fatal(err)
		}

		resp, err := c.GetClient(context.TODO())
		if err != nil {
			log.Fatal(err)
		}
		if resp.StatusCode != http.StatusOK {
			log.Fatalf("Expected HTTP 200 but received %d", resp.StatusCode)
		}
	}

	// or to get a struct with the parsed response body
	{
		c, err := client.NewClientWithResponses("http://localhost:1234", client.WithHTTPClient(&hc))
		if err != nil {
			log.Fatal(err)
		}

		resp, err := c.GetClientWithResponse(context.TODO())
		if err != nil {
			log.Fatal(err)
		}
		if resp.StatusCode() != http.StatusOK {
			log.Fatalf("Expected HTTP 200 but received %d", resp.StatusCode())
		}

		fmt.Printf("resp.JSON200: %v\n", resp.JSON200)
	}

}
```

### With Server URLs

An OpenAPI specification makes it possible to denote Servers that a client can interact with, such as:

```yaml
servers:
- url: https://development.gigantic-server.com/v1
  description: Development server
- url: https://{username}.gigantic-server.com:{port}/{basePath}
  description: The production API server
  variables:
    username:
      # note! no enum here means it is an open value
      default: demo
      description: this value is assigned by the service provider, in this example `gigantic-server.com`
    port:
      enum:
        - '8443'
        - '443'
      default: '8443'
    basePath:
      # open meaning there is the opportunity to use special base paths as assigned by the provider, default is `v2`
      default: v2
```

It is possible to opt-in to the generation of these Server URLs with the following configuration:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: serverurls
output: gen.go
generate:
  # NOTE that this uses default settings - if you want to use initialisms to generate i.e. `ServerURLDevelopmentServer`, you should look up the `output-options.name-normalizer` configuration
  server-urls: true
```

This will then generate the following boilerplate:

```go
// (the below does not include comments that are generated)

const ServerUrlDevelopmentServer = "https://development.gigantic-server.com/v1"

type ServerUrlTheProductionAPIServerBasePathVariable string
const ServerUrlTheProductionAPIServerBasePathVariableDefault = "v2"

type ServerUrlTheProductionAPIServerPortVariable string
const ServerUrlTheProductionAPIServerPortVariable8443 ServerUrlTheProductionAPIServerPortVariable = "8443"
const ServerUrlTheProductionAPIServerPortVariable443 ServerUrlTheProductionAPIServerPortVariable = "443"
const ServerUrlTheProductionAPIServerPortVariableDefault ServerUrlTheProductionAPIServerPortVariable = ServerUrlTheProductionAPIServerPortVariable8443

type ServerUrlTheProductionAPIServerUsernameVariable string
const ServerUrlTheProductionAPIServerUsernameVariableDefault = "demo"

func ServerUrlTheProductionAPIServer(basePath ServerUrlTheProductionAPIServerBasePathVariable, port ServerUrlTheProductionAPIServerPortVariable, username ServerUrlTheProductionAPIServerUsernameVariable) (string, error) {
    // ...
}
```

Notice that for URLs that are not templated, a simple `const` definition is created.

However, for more complex URLs that defined `variables` in them, we generate the types (and any `enum` values or `default` values), and instead use a function to create the URL.

For a complete example see [`examples/generate/serverurls`](examples/generate/serverurls).

### Duplicate types generated for clients's response object types

When generating the types for interacting with the generated client, `oapi-codegen` will use the `operationId` and add on a `Request` or `Response` suffix.

However, this can clash if you have named your component schemas in a similar way.

For instance:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: "Show that generated client boilerplate can clash if schemas are well named i.e. `*Request` and `*Response`"
paths:
  /client:
    put:
      operationId: updateClient
      requestBodies:
        application/json:
          schema:
            $ref: '#/components/schemas/UpdateClientRequest'
      responses:
        400:
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/UpdateClientResponse'
components:
  schemas:
    UpdateClientRequest:
      type: object
      properties:
        code:
          type: string
      required:
      - code
    UpdateClientResponse:
      type: object
      required:
        - name
      properties:
        name:
          type: string
```

If you were to generate with this configuration:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: client
output: client.gen.go
generate:
  models: true
  client: true
```

This would then result in `go build` failures:

```
# github.com/oapi-codegen/oapi-codegen/v2/examples/clienttypenameclash
./client.gen.go:184:6: UpdateClientResponse redeclared in this block
        ./client.gen.go:17:6: other declaration of UpdateClientResponse
./client.gen.go:192:7: r.HTTPResponse undefined (type UpdateClientResponse has no field or method HTTPResponse)
./client.gen.go:193:12: r.HTTPResponse undefined (type UpdateClientResponse has no field or method HTTPResponse)
./client.gen.go:200:7: r.HTTPResponse undefined (type UpdateClientResponse has no field or method HTTPResponse)
./client.gen.go:201:12: r.HTTPResponse undefined (type UpdateClientResponse has no field or method HTTPResponse)
./client.gen.go:224:3: unknown field Body in struct literal of type UpdateClientResponse
./client.gen.go:225:3: unknown field HTTPResponse in struct literal of type UpdateClientResponse
./client.gen.go:234:12: response.JSON400 undefined (type *UpdateClientResponse has no field or method JSON400)
```

To fix this, use the `response-type-suffix` Output Option:

```diff
 # yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
 package: client
 output: client.gen.go
 generate:
   models: true
   client: true
+output-options:
+  response-type-suffix: Resp
```

This will then rename the generated types from the default to use the new suffix:

```diff
-type UpdateClientResponse struct {
+type UpdateClientResp struct {
        Body         []byte
        HTTPResponse *http.Response
        JSON400      *UpdateClientResponse
 }
```

There is no currently planned work to change this behaviour.

## Generating API models

If you're looking to only generate the models for interacting with a remote service, for instance if you need to hand-roll the API client for whatever reason, you can do this as-is.

> [!TIP]
> Try to define as much as possible within the `#/components/schemas` object, as `oapi-codegen` will generate all the types here.
>
> Although we can generate some types based on inline definitions in i.e. a path's response type, it isn't always possible to do this, or if it is generated, can be a little awkward to work with as it may be defined as an anonymous struct.

For instance, given an `api.yaml`:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Generate models
paths:
  /client:
    get:
      operationId: getClient
      responses:
        200:
          content:
            application/json:
              schema:
                # NOTE that Client is generated here, because it's within #/components/schemas
                $ref: "#/components/schemas/Client"
    put:
      operationId: updateClient
      responses:
        400:
          content:
            application/json:
              # NOTE that this anonymous object is /not/ generated because it's an anonymous, but would be generated if using `generate: client`
              # See https://github.com/oapi-codegen/oapi-codegen/issues/1512
              schema:
                type: object
                properties:
                  code:
                    type: string
                required:
                - code
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
    # NOTE that this is not generated by default because it's not referenced. If you want it, you need to use the following YAML configuration:
    #
    # output-options:
    #   skip-prune: true
    Unreferenced:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
```

And a `cfg.yaml`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: onlymodels
output: only-models.gen.go
generate:
  models: true
```

And a `generate.go`:

```go
package onlymodels

//go:generate go run github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen -config cfg.yaml api.yaml
```

This would then generate:

```go
package onlymodels

// Client defines model for Client.
type Client struct {
	Name string `json:"name"`
}
```

If you wish to also generate the `Unreferenced` type, you would need the following `cfg.yaml`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: onlymodels
output: only-models.gen.go
generate:
  models: true
output-options:
  # NOTE that this is only required for the `Unreferenced` type
  skip-prune: true
```

For a complete example see [`examples/only-models`](examples/only-models).

## Splitting large OpenAPI specs across multiple packages (aka "Import Mapping" or "external references")
<a name=import-mapping></a>

When you've got a large OpenAPI specification, you may find it useful to split the contents of the spec across multiple files, using external references, such as:

```yaml
      responses:
        200:
          description: Success
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/User'
```

This is supported by `oapi-codegen`, through the ability to perform "Import Mapping".

For instance, let's say that we have a large API, which has a user-facing API and an admin API, both of which use a common set of API models.

In this case, we may have an Admin API that looks like:

```yaml
# admin/api.yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Admin API
  description: The admin-only portion of the API, which has its own separate OpenAPI spec
tags:
  - name: admin
    description: Admin API endpoints
  - name: user
    description: API endpoint that pertains to user data
paths:
  /admin/user/{id}:
    get:
      tags:
        - admin
        - user
      summary: Get a user's details
      operationId: getUserById
      parameters:
        - name: id
          in: path
          required: true
          schema:
            type: string
            format: uuid
      responses:
        200:
          description: Success
          content:
            application/json:
              schema:
                $ref: '../common/api.yaml#/components/schemas/User'
```

This references the common spec:

```yaml
# common/api.yaml
components:
  schemas:
    User:
      type: object
      additionalProperties: false
      properties:
        name:
          type: string
      required:
        - name
```

So how do we get `oapi-codegen` to generate our code?

### Using a single package with multiple OpenAPI specs

<a name=import-mapping-self></a>

> [!TIP]
> Since `oapi-codegen` v2.4.0, it is now possible to split large OpenAPI specifications into the same Go package, using the "self" mapping (denoted by a `-`) when using Import Mapping.
>
> This is an improvement on the previous model, which would require splitting files across multiple packages.

> [!NOTE]
> You still need to have multiple `go generate`s, and any other configuration files.

To get `oapi-codegen`'s single-package support working, we need multiple calls to `oapi-codegen`, one call per OpenAPI spec file:

```sh
$ go run github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen -config cfg-api.yaml ../admin/api.yaml
$ go run github.com/oapi-codegen/oapi-codegen/v2/cmd/oapi-codegen -config cfg-user.yaml ../common/api.yaml
```

This therefore means that we need multiple configuration files, such as `cfg-api.yaml`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: samepackage
output: server.gen.go
generate:
  models: true
  chi-server: true
  strict-server: true
output-options:
  # to make sure that all types are generated
  skip-prune: true
import-mapping:
  user.yaml: "-"
```

And then our `cfg-user.yaml`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: samepackage
output: user.gen.go
generate:
  models: true
output-options:
  # to make sure that all types are generated
  skip-prune: true
```

From here, `oapi-codegen` will generate multiple Go files, all within the same package, which can be used to break down your large OpenAPI specifications, and generate only the subsets of code needed for each part of the spec.

Check out [the import-mapping/samepackage example](examples/import-mapping/samepackage) for the full code.

### Using multiple packages, with one OpenAPI spec per package

To get `oapi-codegen`'s multi-package support working, we need to set up our directory structure:

```
├── admin
│   ├── cfg.yaml
│   └── generate.go
└── common
    ├── cfg.yaml
    └── generate.go
```

We could start with our configuration file for our admin API spec:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
# admin/cfg.yaml
package: admin
output: server.gen.go
generate:
  models: true
  chi-server: true
output-options:
  # to make sure that all types are generated
  skip-prune: true
# NOTE that this won't work, as it's missing `import-mapping`
```

If we were to run `oapi-codegen`, this will fail with the following error

```
error generating code: error creating operation definitions: error generating response definitions: error generating request body definition: error turning reference (../common/api.yaml#/components/schemas/User) into a Go type: unrecognized external reference '../common/api.yaml'; please provide the known import for this reference using option --import-mapping
```

This is because `oapi-codegen` requires the `import-mapping`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: admin
output: server.gen.go
generate:
  models: true
  chi-server: true
output-options:
  # to make sure that all types are generated
  skip-prune: true
import-mapping:
  # for a given file/URL that is $ref'd, point `oapi-codegen` to the Go package that this spec is generated into, to perform Go package imports
  ../common/api.yaml: github.com/oapi-codegen/oapi-codegen/v2/examples/import-mapping/common
```

This will then generate the following code:

```go
package admin

import (
	// ...
	externalRef0 "github.com/oapi-codegen/oapi-codegen/v2/examples/import-mapping/common"
)

// User defines model for User.
type User = externalRef0.User
```

If you don't want to do this, an alternate option is to [use a single package, with multiple OpenAPI spec files for that given package](#import-mapping-self) or to [bundle your multiple OpenAPI files](https://www.jvt.me/posts/2022/02/10/bundle-openapi/) into a single spec.

Check out [the import-mapping/multiplepackages example](examples/import-mapping/multiplepackages/) for the full code.

## Modifying the input OpenAPI Specification (with OpenAPI Overlay)

Prior to `oapi-codegen` v2.4.0, users wishing to override specific configuration, for instance taking advantage of extensions such as `x-go-type`  would need to modify the OpenAPI specification they are using.

In a lot of cases, this OpenAPI specification would be produced by a different team to the consumers (or even a different company) and so asking them to make changes like this were unreasonable.

This would lead to the API consumers needing to vendor the specification from the producer (which is [our recommendation anyway](#https-paths)) and then make any number of local changes to the specification to make it generate code that looks reasonable.

However, in the case that a consumer would update their specification, they would likely end up with a number of merge conflicts.

Now, as of `oapi-codegen` v2.4.0, it is now possible to make changes to the input OpenAPI specification _without needing to modify it directly_.

This takes advantage of the [OpenAPI Overlay specification](https://github.com/OAI/Overlay-Specification), which is a stable specification.

> [!CAUTION]
> Beware! Here (may) be dragons.
>
> The Overlay specification requires the use of JSON Path, which some users may find difficult to write and/or maintain.
>
> We still heavily recommend using Overlay functionality, but would like users to be aware of this.
>
> There is a [proposed modification to the specification](https://github.com/OAI/Overlay-Specification/pull/32) which would relax the need for JSON Path as the targeting mechanism.

For instance, let's say that we have the following OpenAPI specification, which provides insight into an internal endpoint that we should not be generating any code for (denoted by `x-internal`):

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: "Example to indicate how to use the OpenAPI Overlay specification (https://github.com/OAI/Overlay-Specification)"
paths:
  /ping:
    get:
      responses:
        '200':
          description: pet response
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pong'
    delete:
      x-internal: true
      responses:
        '202':
          content: {}
```

If we were to run `oapi-codegen` with out-of-the-box functionality, this would then lead to the DELETE endpoint being generated, which we don't want.

Instead, we can define the following `overlay.yaml`:


```yaml
overlay: 1.0.0
info:
  title: Overlay
  version: 0.0.0
actions:
- target: "$"
  description: Perform a structural overlay, which can be more readable, as it's clear what the shape of the document is
  update:
    info:
      x-overlay-applied: structured-overlay
    paths:
      /ping:
        get:
          responses:
            '200':
              description: Perform a ping request
- target: $.paths.*[?(@.x-internal)]
  description: Remove internal endpoints (noted by x-internal)
  remove: true
- target: $.paths.*.*[?(@.x-internal)]
  description: Remove internal endpoints (noted by x-internal)
  remove: true
```

And our configuration file for `oapi-codegen`:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: api
output: ping.gen.go
generate:
  models: true
  gorilla-server: true
  embedded-spec: true
output-options:
  overlay:
    path: overlay.yaml
```

This then completely removes the DELETE endpoint _before_ we even start to parse the specification in `oapi-codegen`, so it's as if your specification was provided without that endpoint.

Additionally, we can override other pieces of metadata, such as the description for operations.

Check out [the overlay example](examples/overlay/) for the full code, and some more complex examples.

## Generating Nullable types

It's possible that you want to be able to determine whether a field isn't sent, is sent as `null` or has a value.

For instance, if you had the following OpenAPI property:

```yaml
S:
  type: object
  properties:
    Field:
      type: string
      nullable: true
    required: []
```

The default behaviour in `oapi-codegen` is to generate:

```go
type S struct {
	Field *string `json:"field,omitempty"`
}
```

However, you lose the ability to understand the three cases, as there's no way to distinguish two of the types from each other:

- is this field not sent? (Can be checked with `S.Field == nil`)
- is this field `null`? (Can be checked with `S.Field == nil`)
- does this field have a value? (`S.Field != nil && *S.Field == "123"`)

As of `oapi-codegen` [v2.1.0](https://github.com/oapi-codegen/oapi-codegen/releases/tag/v2.1.0) it is now possible to represent this with the `nullable.Nullable` type from [our new library, oapi-codegen/nullable](https://github.com/oapi-codegen/nullable).

If you configure your generator's Output Options to opt-in to this behaviour, as so:

```yaml
output-options:
  nullable-type: true
```

You will now receive the following output:

```go
type S struct {
    // note that there's no pointer here, just `omitempty`
    Field nullable.Nullable[string] `json:"field,omitempty"`
}
```

## OpenAPI extensions

As well as the core OpenAPI support, we also support the following OpenAPI extensions, as denoted by the [OpenAPI Specification Extensions](https://spec.openapis.org/oas/v3.0.3#specification-extensions).

The following extensions are supported:

<table>

<tr>
<th>
Extension
</th>
<th>
Description
</th>
</tr>

<tr>
<td>

`x-go-type` <br>
`x-go-type-import`

</td>
<td>
Override the generated type definition (and optionally, add an import from another package)
</td>
</tr>

<tr>
<td>

`x-go-type-skip-optional-pointer`

</td>
<td>
Do not add a pointer type for optional fields in structs
</td>
</tr>

<tr>
<td>

`x-go-name`

</td>
<td>
Override the generated name of a field or a type
</td>
</tr>

<tr>
<td>

`x-go-type-name`

</td>
<td>
Override the generated name of a type
</td>
</tr>

<tr>
<td>

`x-omitempty`

</td>
<td>
Force the presence of the JSON tag `omitempty` on a field
</td>
</tr>

<tr>
<td>

`x-omitzero`

</td>
<td>
Force the presence of the JSON tag `omitzero` on a field
</td>
</tr>

<tr>
<td>

`x-go-json-ignore`

</td>
<td>
When (un)marshaling JSON, ignore field(s)
</td>
</tr>

<tr>
<td>

`x-oapi-codegen-extra-tags`

</td>
<td>
Generate arbitrary struct tags to fields
</td>
</tr>

<tr>
<td>

`x-enum-varnames` / `x-enumNames`

</td>
<td>
Override generated variable names for enum constants
</td>
</tr>

<tr>
<td>

`x-deprecated-reason`

</td>
<td>
Add a GoDoc deprecation warning to a type
</td>
</tr>

<tr>
<td>

`x-order`

</td>
<td>
Explicitly order struct fields
</td>
</tr>

<tr>
<td>

`x-oapi-codegen-only-honour-go-name`

</td>
<td>
Only honour the `x-go-name` when generating field names
</td>
</tr>

</table>


### `x-go-type` / `x-go-type-import` - override the generated type definition (and optionally, add an import from another package)

Using the `x-go-type` (and optionally, `x-go-type-import` when you need to import another package) allows overriding the type that `oapi-codegen` determined the generated type should be.

We can see this at play with the following schemas:

```yaml
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          type: string
          # this is a bit of a contrived example, as you could instead use
          # `format: uuid` but it explains how you'd do this when there may be
          # a clash, for instance if you already had a `uuid` package that was
          # being imported, or ...
          x-go-type: googleuuid.UUID
          x-go-type-import:
            path: github.com/google/uuid
            name: googleuuid
        id:
          type: number
          # ... this is also a bit of a contrived example, as you could use
          # `type: integer` but in the case that you know better than what
          # oapi-codegen is generating, like so:
          x-go-type: int64
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id   *int64          `json:"id,omitempty"`
	Name googleuuid.UUID `json:"name"`
}
```

You can see this in more detail in [the example code](examples/extensions/xgotype/).

### `x-go-type-skip-optional-pointer` - do not add a pointer type for optional fields in structs

<a name="ext-x-go-type-skip-optional-pointer"></a>

> [!TIP]
> If you prefer this behaviour, and prefer to not have to annotate your whole OpenAPI spec for this behaviour, you can use `output-options.prefer-skip-optional-pointer=true` to default this behaviour for all fields.
>
> It is then possible to override this on a per-type/per-field basis where necessary.

By default, `oapi-codegen` will generate a pointer for optional fields.

Using the `x-go-type-skip-optional-pointer` extension allows omitting that pointer.

We can see this at play with the following schemas:

```yaml
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
          x-go-type-skip-optional-pointer: true
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id   float32 `json:"id,omitempty"`
	Name string  `json:"name"`
}
```

You can see this in more detail in [the example code](examples/extensions/xgotypeskipoptionalpointer/).

### `x-go-name` - override the generated name of a field or a type

By default, `oapi-codegen` will attempt to generate the name of fields and types in as best a way it can.

However, sometimes, the name doesn't quite fit what your codebase standards are, or the intent of the field, so you can override it with `x-go-name`.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-go-name
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      # can be used on a type
      x-go-name: ClientRenamedByExtension
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
          # or on a field
          x-go-name: AccountIdentifier
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientRenamedByExtension defines model for ClientWithExtension.
type ClientRenamedByExtension struct {
	AccountIdentifier *float32 `json:"id,omitempty"`
	Name              string   `json:"name"`
}
```

You can see this in more detail in [the example code](examples/extensions/xgoname/).

### `x-go-type-name` - Override the generated name of a type

> [!NOTE]
> Notice that this is subtly different to the `x-go-name`, which also applies to _fields_ within `struct`s.

By default, `oapi-codegen` will attempt to generate the name of types in as best a way it can.

However, sometimes, the name doesn't quite fit what your codebase standards are, or the intent of the field, so you can override it with `x-go-name`.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-go-type-name
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      x-go-type-name: ClientRenamedByExtension
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
          # NOTE attempting a `x-go-type-name` here is a no-op, as we're not producing a _type_ only a _field_
          x-go-type-name: ThisWillNotBeUsed
```

From here, we now get two different models and a type alias:

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension = ClientRenamedByExtension

// ClientRenamedByExtension defines model for .
type ClientRenamedByExtension struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}
```

You can see this in more detail in [the example code](examples/extensions/xgotypename/).

### `x-omitempty` - force the presence of the JSON tag `omitempty` on a field

In a case that you may want to add the JSON struct tag `omitempty` to types that don't have one generated by default - for instance a required field - you can use the `x-omitempty` extension.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-omitempty
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          type: string
          # for some reason, you may want this behaviour, even though it's a required field
          x-omitempty: true
        id:
          type: number
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name,omitempty"`
}
```

You can see this in more detail in [the example code](examples/extensions/xomitempty/).

### `x-omitzero` - force the presence of the JSON tag `omitzero` on a field

> [!NOTE]
> `omitzero` was added in Go 1.24. If you're not using Go 1.24 in your project, this won't work.

In a case that you may want to add the JSON struct tag `omitzero` to types, you can use the `x-omitempty` extension.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-omitempty
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
          x-omitzero: true
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id   *float32 `json:"id,omitempty,omitzero"`
	Name string   `json:"name"`
}
```

You can see this in more detail in [the example code](examples/extensions/xomitzero/).

### `x-go-json-ignore` - when (un)marshaling JSON, ignore field(s)

By default, `oapi-codegen` will generate `json:"..."` struct tags for all fields in a struct, so JSON (un)marshaling works.

However, sometimes, you want to omit fields, which can be done with the `x-go-json-ignore` extension.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-go-json-ignore
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        complexField:
          type: object
          properties:
            name:
              type: string
            accountName:
              type: string
          # ...
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        complexField:
          type: object
          properties:
            name:
              type: string
            accountName:
              type: string
          # ...
          x-go-json-ignore: true
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	ComplexField *struct {
		AccountName *string `json:"accountName,omitempty"`
		Name        *string `json:"name,omitempty"`
	} `json:"complexField,omitempty"`
	Name string `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	ComplexField *struct {
		AccountName *string `json:"accountName,omitempty"`
		Name        *string `json:"name,omitempty"`
	} `json:"-"`
	Name string `json:"name"`
}
```

Notice that the `ComplexField` is still generated in full, but the type will then be ignored with JSON marshalling.

You can see this in more detail in [the example code](examples/extensions/xgojsonignore/).

### `x-oapi-codegen-extra-tags` - generate arbitrary struct tags to fields

If you're making use of a field's struct tags to i.e. apply validation, decide whether something should be logged, etc, you can use `x-oapi-codegen-extra-tags` to set additional tags for your generated types.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-oapi-codegen-extra-tags
components:
  schemas:
    Client:
      type: object
      required:
        - name
        - id
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
        - id
      properties:
        name:
          type: string
        id:
          type: number
          x-oapi-codegen-extra-tags:
            validate: "required,min=1,max=256"
            safe-to-log: "true"
            gorm: primarykey
```

From here, we now get two different models:

```go
// Client defines model for Client.
type Client struct {
	Id   float32 `json:"id"`
	Name string  `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id   float32 `gorm:"primarykey" json:"id" safe-to-log:"true" validate:"required,min=1,max=256"`
	Name string  `json:"name"`
}
```

You can see this in more detail in [the example code](examples/extensions/xoapicodegenextratags/).

### `x-enum-varnames` / `x-enumNames` - override generated variable names for enum constants

When consuming an enum value from an external system, the name may not produce a nice variable name. Using the `x-enum-varnames` extension allows overriding the name of the generated variable names.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-enumNames and x-enum-varnames
components:
  schemas:
    ClientType:
      type: string
      enum:
        - ACT
        - EXP
    ClientTypeWithNamesExtension:
      type: string
      enum:
        - ACT
        - EXP
      x-enumNames:
        - Active
        - Expired
    ClientTypeWithVarNamesExtension:
      type: string
      enum:
        - ACT
        - EXP
      x-enum-varnames:
        - Active
        - Expired
```

From here, we now get two different forms of the same enum definition.

```go
// Defines values for ClientType.
const (
	ACT ClientType = "ACT"
	EXP ClientType = "EXP"
)

// Defines values for ClientTypeWithNamesExtension.
const (
	ClientTypeWithNamesExtensionActive  ClientTypeWithNamesExtension = "ACT"
	ClientTypeWithNamesExtensionExpired ClientTypeWithNamesExtension = "EXP"
)

// Defines values for ClientTypeWithVarNamesExtension.
const (
	ClientTypeWithVarNamesExtensionActive  ClientTypeWithVarNamesExtension = "ACT"
	ClientTypeWithVarNamesExtensionExpired ClientTypeWithVarNamesExtension = "EXP"
)

// ClientType defines model for ClientType.
type ClientType string

// ClientTypeWithNamesExtension defines model for ClientTypeWithNamesExtension.
type ClientTypeWithNamesExtension string

// ClientTypeWithVarNamesExtension defines model for ClientTypeWithVarNamesExtension.
type ClientTypeWithVarNamesExtension string
```

You can see this in more detail in [the example code](examples/extensions/xenumnames/).

### `x-deprecated-reason` - add a GoDoc deprecation warning to a type

When an OpenAPI type is deprecated, a deprecation warning can be added in the GoDoc using `x-deprecated-reason`.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-deprecated-reason
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          type: string
          deprecated: true
          x-deprecated-reason: Don't use because reasons
        id:
          type: number
          # NOTE that this doesn't generate, as no `deprecated: true` is set
          x-deprecated-reason: NOTE you shouldn't see this, as you've not deprecated this field
```

From here, we now get two different forms of the same enum definition.

```go
// Client defines model for Client.
type Client struct {
	Id   *float32 `json:"id,omitempty"`
	Name string   `json:"name"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id *float32 `json:"id,omitempty"`
	// Deprecated: Don't use because reasons
	Name string `json:"name"`
}
```

Notice that because we've not set `deprecated: true` to the `name` field, it doesn't generate a deprecation warning.

You can see this in more detail in [the example code](examples/extensions/xdeprecatedreason/).

### `x-order` - explicitly order struct fields

Whether you like certain fields being ordered before others, or you want to perform more efficient packing of your structs, the `x-order` extension is here for you.

Note that `x-order` is 1-indexed - `x-order: 0` is not a valid value.

We can see this at play with the following schemas:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-order
components:
  schemas:
    Client:
      type: object
      required:
        - name
      properties:
        a_name:
          type: string
        id:
          type: number
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        a_name:
          type: string
          x-order: 2
        id:
          type: number
          x-order: 1
```

From here, we now get two different forms of the same type definition.

```go
// Client defines model for Client.
type Client struct {
	AName *string  `json:"a_name,omitempty"`
	Id    *float32 `json:"id,omitempty"`
}

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	Id    *float32 `json:"id,omitempty"`
	AName *string  `json:"a_name,omitempty"`
}
```

You can see this in more detail in [the example code](examples/extensions/xorder/).

### `x-oapi-codegen-only-honour-go-name` - only honour the `x-go-name` when generating field names


> [!WARNING]
> Using this option may lead to cases where `oapi-codegen`'s rewriting of field names to prevent clashes with other types, or to prevent including characters that may not be valid Go field names.

In some cases, you may not want use the inbuilt options for converting an OpenAPI field name to a Go field name, such as the `name-normalizer: "ToCamelCaseWithInitialisms"`, and instead trust the name that you've defined for the type better.

In this case, you can use `x-oapi-codegen-only-honour-go-name` to enforce this, alongside specifying the `allow-unexported-struct-field-names` compatibility option.

This allows you to take a spec such as:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: x-oapi-codegen-only-honour-go-name
components:
  schemas:
    TypeWithUnexportedField:
      description: A struct will be output where one of the fields is not exported
      properties:
        name:
          type: string
        id:
          type: string
          # NOTE that there is an explicit usage of a lowercase character
          x-go-name: accountIdentifier
          x-oapi-codegen-extra-tags:
            json: "-"
          x-oapi-codegen-only-honour-go-name: true
```

And we'll generate:

```go
// TypeWithUnexportedField A struct will be output where one of the fields is not exported
type TypeWithUnexportedField struct {
	accountIdentifier *string `json:"-"`
	Name              *string `json:"name,omitempty"`
}
```

You can see this in more detail in [the example code](examples/extensions/xoapicodegenonlyhonourgoname).

## Request/response validation middleware

The generated code that `oapi-codegen` produces has some validation for some incoming data, such as checking for required headers, and when using the [strict server](#strict-server) you get some more validation around the correct usage of the response types.

However, this leaves a lot of validation that needs to be done, which can be tedious to hand-write this logic, especially for large or complex OpenAPI specifications.

To simplify this, we use a middleware, which provides the request validation. The middleware you want to use depends on the server you're using:

<table>

<tr>
<th>
Server
</th>
<th>
Middleware library
</th>
</tr>

<tr>
<td>

[Chi](https://github.com/go-chi/chi)

</td>
<td>

[nethttp-middleware](https://github.com/oapi-codegen/nethttp-middleware)

</td>
</tr>

<tr>
<td>

[Echo](https://github.com/labstack/echo)

</td>
<td>

[echo-middleware](https://github.com/oapi-codegen/echo-middleware)

</td>
</tr>

<tr>
<td>

[Fiber](https://github.com/gofiber/fiber)

</td>
<td>

[fiber-middleware](https://github.com/oapi-codegen/fiber-middleware)

</td>

</tr>


<tr>
<td>

[Gin](https://github.com/gin-gonic/gin)

</td>
<td>

[gin-middleware](https://github.com/oapi-codegen/gin-middleware)

</td>
</tr>

<tr>
<td>

[gorilla/mux](https://github.com/gorilla/mux)

</td>
<td>

[nethttp-middleware](https://github.com/oapi-codegen/nethttp-middleware)

</td>

</tr>

<tr>
<td>

[Iris](https://github.com/kataras/iris)

</td>
<td>

[iris-middleware](https://github.com/oapi-codegen/iris-middleware)

</td>

</tr>

<tr>
<td>

[1.22+ `net/http`](https://pkg.go.dev/net/http)

</td>
<td>

[nethttp-middleware](https://github.com/oapi-codegen/nethttp-middleware)

</td>

</tr>

<tr>
<td>

Any other server (which conforms to `net/http`)

</td>
<td>

[nethttp-middleware](https://github.com/oapi-codegen/nethttp-middleware)

</td>

</tr>

</table>

> [!NOTE]
> It is [not currently possible](https://github.com/oapi-codegen/oapi-codegen/issues/1038) to validate the HTTP response with a middleware.

> [!NOTE]
> We're also [exploring](https://github.com/oapi-codegen/exp/issues/1) the use of [libopenapi-validator](https://github.com/pb33f/libopenapi-validator/) for request/response validation middleware

## Implementing security

If you're using a specification with [Security Schemes](https://spec.openapis.org/oas/v3.0.3#security-scheme-object) and [Security Requirements](https://spec.openapis.org/oas/v3.0.3#security-requirement-object), you'll want to authenticate and authorize requests.

### On the server

> [!NOTE]
> Out-of-the-box, the server-side code generated by `oapi-codegen` does not provide security validation.
>
> To perform authentication, you will need to use the [validation middleware](#requestresponse-validation-middleware).
>
> In the future, we plan to [implement server-side validation in the generated code](https://github.com/oapi-codegen/oapi-codegen/issues/1524)

To see how this can work, check out the [authenticated API example](examples/authenticated-api/echo).

### On the client

With a generated client, you'll want to use the client's generated `WithRequestEditorFn` function to pass in a given request editor `RequestEditorFn`.

For instance:

```go
import (
	"context"
	"fmt"
	"log"

	"github.com/oapi-codegen/oapi-codegen/v2/pkg/securityprovider"
)

func main() {
	basicAuth, err := securityprovider.NewSecurityProviderBasicAuth("my_user", "my_pass")
	if err != nil {
		log.Fatal(err)
	}

	client, err := NewClient("https://....", WithRequestEditorFn(basicAuth.Intercept))
	if err != nil {
		log.Fatal(err)
	}

	resp, err := client.GetClient(context.TODO())
	if err != nil {
		log.Fatal(err)
	}
	fmt.Printf("resp.StatusCode: %v\n", resp.StatusCode)
}
```

Notice that we're using a pre-built provider from the [`pkg/securityprovider` package](https://pkg.go.dev/github.com/oapi-codegen/oapi-codegen/v2/pkg/securityprovider), which has some inbuilt support for other types of authentication, too.

## Custom code generation

It is possible to extend the inbuilt code generation from `oapi-codegen` using Go's `text/template`s.

You can specify, through your configuration file, the `output-options.user-templates` setting to override the inbuilt templates and use a user-defined template.

> [!NOTE]
> Filenames given to the `user-templates` configuration must **exactly** match the filename that `oapi-codegen` is looking for

### Local paths

Within your configuration file, you can specify relative or absolute paths to a file to reference for the template, such as:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
# ...
output-options:
  user-templates:
    client-with-responses.tmpl: ./custom-template.tmpl
    additional-properties.tmpl: /tmp/foo.bar
    typedef.tmpl: no-prefix.tmpl
```

> [!WARNING]
> We do not interpolate `~` or `$HOME` (or other environment variables) in paths given

### HTTPS paths

It is also possible to use HTTPS URLs.

> [!WARNING]
> Although possible, this does lead to `oapi-codegen` executions not necessarily being reproducible. It's recommended to vendor (copy) the OpenAPI spec into your codebase and reference it locally
>
> See [this blog post](https://www.jvt.me/posts/2024/04/27/github-actions-update-file/) for an example of how to use GitHub Actions to manage the updates of files across repos
>
> This will be disabled by default (but possible to turn back on via configuration) [in the future](https://github.com/oapi-codegen/oapi-codegen/issues/1564)

To use it, you can use the following configuration:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
# ...
output-options:
  user-templates:
    # The following are referencing a version of the default client-with-responses.tmpl file, but loaded in through GitHub's raw.githubusercontent.com. The general form to use raw.githubusercontent.com is as follows https://raw.githubusercontent.com/<username>/<project>/<commitish>/path/to/template/template.tmpl

    # Alternatively using raw.githubusercontent.com with a hash
    client-with-responses.tmpl: https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/ad5eada4f3ccc28a88477cef62ea21c17fc8aa01/pkg/codegen/templates/client-with-responses.tmpl
    # Alternatively using raw.githubusercontent.com with a tag
    client-with-responses.tmpl: https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/v2.1.0/pkg/codegen/templates/client-with-responses.tmpl
    # Alternatively using raw.githubusercontent.com with a branch
    client-with-responses.tmpl: https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/master/pkg/codegen/templates/client-with-responses.tmpl
```

> [!WARNING]
> If using URLs that pull locations from a Git repo, such as `raw.githubusercontent.com`, it is strongly encouraged to use a tag or a raw commit hash instead of a branch like `main`. Tracking a branch can lead to unexpected API drift, and loss of the ability to reproduce a build.

### Inline template

It's also possible to set the templates inline in the configuration file:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
# ...
output-options:
  user-templates:
    # NOTE the use of the `|` (pipe symbol) here to denote that this is a
    # multi-line statement that should preserve newlines. More reading:
    # https://stackoverflow.com/a/18708156/2257038 and
    # https://stackoverflow.com/a/15365296/2257038
    client-with-responses.tmpl: |
        // ClientWithResponses builds on ClientInterface to offer response payloads
        type ClientWithResponses struct {
            ClientInterface
        }
        ...
```

### Using the Go package

Alternatively, you are able to use the underlying code generation as a package, which [will be documented in the future](https://github.com/oapi-codegen/oapi-codegen/issues/1487).

## Additional Properties (`additionalProperties`)

[OpenAPI Schemas](https://spec.openapis.org/oas/v3.0.3.html#schema-object) implicitly accept `additionalProperties`, meaning that any fields provided, but not explicitly defined via properties on the schema are accepted as input, and propagated. When unspecified, OpenAPI defines that the `additionalProperties` field is assumed to be `true`.

For simplicity, and to remove a fair bit of duplication and boilerplate, `oapi-codegen` decides to ignore the implicit `additionalProperties: true`, and instead requires you to specify the `additionalProperties` key to generate the boilerplate.

> [!NOTE]
> In the future [this will be possible](https://github.com/oapi-codegen/oapi-codegen/issues/1514) to disable this functionality, and honour the implicit `additionalProperties: true`

Below you can see some examples of how `additionalProperties` affects the generated code.

### Implicit `additionalProperties: true` / no `additionalProperties` set

```yaml
components:
  schemas:
    Thing:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
      # implicit additionalProperties: true
```

Will generate:

```go
// Thing defines model for Thing.
type Thing struct {
	Id int `json:"id"`
}

// with no generated boilerplate nor the `AdditionalProperties` field
```

### Explicit `additionalProperties: true`

```yaml
components:
  schemas:
    Thing:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
      # explicit true
      additionalProperties: true
```

Will generate:

```go
// Thing defines model for Thing.
type Thing struct {
	Id                   int                    `json:"id"`
	AdditionalProperties map[string]interface{} `json:"-"`
}

// with generated boilerplate below
```

<details>

<summary>Boilerplate</summary>

```go

// Getter for additional properties for Thing. Returns the specified
// element and whether it was found
func (a Thing) Get(fieldName string) (value interface{}, found bool) {
	if a.AdditionalProperties != nil {
		value, found = a.AdditionalProperties[fieldName]
	}
	return
}

// Setter for additional properties for Thing
func (a *Thing) Set(fieldName string, value interface{}) {
	if a.AdditionalProperties == nil {
		a.AdditionalProperties = make(map[string]interface{})
	}
	a.AdditionalProperties[fieldName] = value
}

// Override default JSON handling for Thing to handle AdditionalProperties
func (a *Thing) UnmarshalJSON(b []byte) error {
	object := make(map[string]json.RawMessage)
	err := json.Unmarshal(b, &object)
	if err != nil {
		return err
	}

	if raw, found := object["id"]; found {
		err = json.Unmarshal(raw, &a.Id)
		if err != nil {
			return fmt.Errorf("error reading 'id': %w", err)
		}
		delete(object, "id")
	}

	if len(object) != 0 {
		a.AdditionalProperties = make(map[string]interface{})
		for fieldName, fieldBuf := range object {
			var fieldVal interface{}
			err := json.Unmarshal(fieldBuf, &fieldVal)
			if err != nil {
				return fmt.Errorf("error unmarshaling field %s: %w", fieldName, err)
			}
			a.AdditionalProperties[fieldName] = fieldVal
		}
	}
	return nil
}

// Override default JSON handling for Thing to handle AdditionalProperties
func (a Thing) MarshalJSON() ([]byte, error) {
	var err error
	object := make(map[string]json.RawMessage)

	object["id"], err = json.Marshal(a.Id)
	if err != nil {
		return nil, fmt.Errorf("error marshaling 'id': %w", err)
	}

	for fieldName, field := range a.AdditionalProperties {
		object[fieldName], err = json.Marshal(field)
		if err != nil {
			return nil, fmt.Errorf("error marshaling '%s': %w", fieldName, err)
		}
	}
	return json.Marshal(object)
}
```

</details>


### `additionalProperties` as `integer`s

```yaml
components:
  schemas:
    Thing:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
      # simple type
      additionalProperties:
        type: integer
```

Will generate:

```go
// Thing defines model for Thing.
type Thing struct {
	Id                   int            `json:"id"`
	AdditionalProperties map[string]int `json:"-"`
}

// with generated boilerplate below
```

<details>

<summary>Boilerplate</summary>

```go
// Getter for additional properties for Thing. Returns the specified
// element and whether it was found
func (a Thing) Get(fieldName string) (value int, found bool) {
	if a.AdditionalProperties != nil {
		value, found = a.AdditionalProperties[fieldName]
	}
	return
}

// Setter for additional properties for Thing
func (a *Thing) Set(fieldName string, value int) {
	if a.AdditionalProperties == nil {
		a.AdditionalProperties = make(map[string]int)
	}
	a.AdditionalProperties[fieldName] = value
}

// Override default JSON handling for Thing to handle AdditionalProperties
func (a *Thing) UnmarshalJSON(b []byte) error {
	object := make(map[string]json.RawMessage)
	err := json.Unmarshal(b, &object)
	if err != nil {
		return err
	}

	if raw, found := object["id"]; found {
		err = json.Unmarshal(raw, &a.Id)
		if err != nil {
			return fmt.Errorf("error reading 'id': %w", err)
		}
		delete(object, "id")
	}

	if len(object) != 0 {
		a.AdditionalProperties = make(map[string]int)
		for fieldName, fieldBuf := range object {
			var fieldVal int
			err := json.Unmarshal(fieldBuf, &fieldVal)
			if err != nil {
				return fmt.Errorf("error unmarshaling field %s: %w", fieldName, err)
			}
			a.AdditionalProperties[fieldName] = fieldVal
		}
	}
	return nil
}

// Override default JSON handling for Thing to handle AdditionalProperties
func (a Thing) MarshalJSON() ([]byte, error) {
	var err error
	object := make(map[string]json.RawMessage)

	object["id"], err = json.Marshal(a.Id)
	if err != nil {
		return nil, fmt.Errorf("error marshaling 'id': %w", err)
	}

	for fieldName, field := range a.AdditionalProperties {
		object[fieldName], err = json.Marshal(field)
		if err != nil {
			return nil, fmt.Errorf("error marshaling '%s': %w", fieldName, err)
		}
	}
	return json.Marshal(object)
}
```

</details>

### `additionalProperties` with an object

```yaml
components:
  schemas:
    Thing:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
      # object
      additionalProperties:
        type: object
        properties:
          foo:
            type: string
```

Will generate:

```go
// Thing defines model for Thing.
type Thing struct {
	Id                   int `json:"id"`
	AdditionalProperties map[string]struct {
		Foo *string `json:"foo,omitempty"`
	} `json:"-"`
}

// with generated boilerplate below
```

<details>

<summary>Boilerplate</summary>

```go
// Getter for additional properties for Thing. Returns the specified
// element and whether it was found
func (a Thing) Get(fieldName string) (value struct {
	Foo *string `json:"foo,omitempty"`
}, found bool) {
	if a.AdditionalProperties != nil {
		value, found = a.AdditionalProperties[fieldName]
	}
	return
}

// Setter for additional properties for Thing
func (a *Thing) Set(fieldName string, value struct {
	Foo *string `json:"foo,omitempty"`
}) {
	if a.AdditionalProperties == nil {
		a.AdditionalProperties = make(map[string]struct {
			Foo *string `json:"foo,omitempty"`
		})
	}
	a.AdditionalProperties[fieldName] = value
}

// Override default JSON handling for Thing to handle AdditionalProperties
func (a *Thing) UnmarshalJSON(b []byte) error {
	object := make(map[string]json.RawMessage)
	err := json.Unmarshal(b, &object)
	if err != nil {
		return err
	}

	if raw, found := object["id"]; found {
		err = json.Unmarshal(raw, &a.Id)
		if err != nil {
			return fmt.Errorf("error reading 'id': %w", err)
		}
		delete(object, "id")
	}

	if len(object) != 0 {
		a.AdditionalProperties = make(map[string]struct {
			Foo *string `json:"foo,omitempty"`
		})
		for fieldName, fieldBuf := range object {
			var fieldVal struct {
				Foo *string `json:"foo,omitempty"`
			}
			err := json.Unmarshal(fieldBuf, &fieldVal)
			if err != nil {
				return fmt.Errorf("error unmarshaling field %s: %w", fieldName, err)
			}
			a.AdditionalProperties[fieldName] = fieldVal
		}
	}
	return nil
}

// Override default JSON handling for Thing to handle AdditionalProperties
func (a Thing) MarshalJSON() ([]byte, error) {
	var err error
	object := make(map[string]json.RawMessage)

	object["id"], err = json.Marshal(a.Id)
	if err != nil {
		return nil, fmt.Errorf("error marshaling 'id': %w", err)
	}

	for fieldName, field := range a.AdditionalProperties {
		object[fieldName], err = json.Marshal(field)
		if err != nil {
			return nil, fmt.Errorf("error marshaling '%s': %w", fieldName, err)
		}
	}
	return json.Marshal(object)
}
```

</details>

## Globally skipping the "optional pointer"

One of the key things `oapi-codegen` does is to use an "optional pointer", following idiomatic Go practices, to indicate that a field/type is optional.

This can be tuned on a per-field basis, using the [`x-go-type-skip-optional-pointer` extension](#ext-x-go-type-skip-optional-pointer), but it can be a bit repetitive, or can be more complex when using an OpenAPI Overlay.

As of `oapi-codegen` v2.5.0, this can be tuned in two specific ways, via the following `output-options:`:

- `prefer-skip-optional-pointer`: a global default that you do _not_ want the "optional pointer" generated. Optional fields will not have an "optional pointer", and will have an `omitempty` JSON tag
- `prefer-skip-optional-pointer-with-omitzero`: when used in conjunction with `prefer-skip-optional-pointer`, any optional fields are generated with an `omitzero` JSON tag. **Requires Go 1.24+**

In both cases, there is control on a per-field level to set `x-go-type-skip-optional-pointer: false` or `x-omitzero: false` to undo these to field(s).

For example, when combining both options:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
package: preferskipoptionalpointerwithomitzero
output: gen.go
generate:
  # ...
output-options:
  # ...
  prefer-skip-optional-pointer: true
  prefer-skip-optional-pointer-with-omitzero: true
```

When we have the following spec:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: prefer-skip-optional-pointer-with-omitzero
components:
  schemas:
    ClientWithExtension:
      type: object
      required:
        - name
      properties:
        name:
          description: This field is required, so will never have an optional pointer, nor `omitzero`.
          type: string
        id:
          description: This field is optional, but the `prefer-skip-optional-pointer` Output Option ensures that this should not have an optional pointer. However, it will receive `omitzero`.
          type: number
        pointer_id:
          type: number
          description: This field should have an optional pointer, as the field-level definition of `x-go-type-skip-optional-pointer` overrides the `prefer-skip-optional-pointer` Output Option. This will also not receive an `omitzero`.
          # NOTE that this overrides the global preference
          x-go-type-skip-optional-pointer: false
        no_omit:
          type: number
          description: This field is optional, but the `prefer-skip-optional-pointer` Output Option ensures that this should not have an optional pointer. This will not receive `omitzero`, as the field-level definition of `x-omitzero` overrides the `prefer-skip-optional-pointer-with-omitzero` Output Option.
          # NOTE that this overrides the global preference
          x-omitzero: false
```

We then generate the following Go code:

```go
// ...

// ClientWithExtension defines model for ClientWithExtension.
type ClientWithExtension struct {
	// Id This field is optional, but the `prefer-skip-optional-pointer` Output Option ensures that this should not have an optional pointer. However, it will receive `omitzero`.
	Id float32 `json:"id,omitempty,omitzero"`

	// Name This field is required, so will never have an optional pointer, nor `omitzero`.
	Name string `json:"name"`

	// NoOmit This field is optional, but the `prefer-skip-optional-pointer` Output Option ensures that this should not have an optional pointer. This will not receive `omitzero`, as the field-level definition of `x-omitzero` overrides the `prefer-skip-optional-pointer-with-omitzero` Output Option.
	NoOmit float32 `json:"no_omit,omitempty"`

	// PointerId This field should have an optional pointer, as the field-level definition of `x-go-type-skip-optional-pointer` overrides the `prefer-skip-optional-pointer` Output Option. This will also not receive an `omitzero`.
	PointerId *float32 `json:"pointer_id,omitempty"`
}
```

You can see this in more detail in [the example code for `prefer-skip-optional-pointer`](examples/output-options/preferskipoptionalpointer/) and [example code for `prefer-skip-optional-pointer-with-omitzero`](examples/output-options/preferskipoptionalpointerwithomitzero/)

## Changing the names of generated types

As of `oapi-codegen` v2.2.0, it is now possible to use the `output-options` configuration's `name-normalizer` to define the logic for how to convert an OpenAPI name (i.e. an Operation ID or a Schema name) and construct a Go type name.

<details>

<summary>Example, using default configuration</summary>

By default, `oapi-codegen` will perform camel-case conversion, so for a spec such as:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Example code for the `name-normalizer` output option
paths:
  /api/pets/{petId}:
    get:
      summary: Get pet given identifier.
      operationId: getHttpPet
      parameters:
      - name: petId
        in: path
        required: true
        schema:
          type: string
      responses:
        '200':
          description: valid pet
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pet'
components:
  schemas:
    Pet:
      type: object
      required:
        - uuid
        - name
      properties:
        uuid:
          type: string
          description: The pet uuid.
        name:
          type: string
          description: The name of the pet.
    Error:
      required:
        - code
        - message
      properties:
        code:
          type: integer
          format: int32
          description: Error code
        message:
          type: string
          description: Error message
    OneOf2things:
      description: "Notice that the `things` is not capitalised"
      oneOf:
        - type: object
          required:
            - id
          properties:
            id:
              type: integer
        - type: object
          required:
            - id
          properties:
            id:
              type: string
              format: uuid
```

This will produce:

```go
// OneOf2things Notice that the `things` is not capitalised
type OneOf2things struct {
	union json.RawMessage
}

// Pet defines model for Pet.
type Pet struct {
	// Name The name of the pet.
	Name string `json:"name"`

	// Uuid The pet uuid.
	Uuid string `json:"uuid"`
}

// The interface specification for the client above.
type ClientInterface interface {
	// GetHttpPet request
	GetHttpPet(ctx context.Context, petId string, reqEditors ...RequestEditorFn) (*http.Response, error)
}
```

</details>

<details>

<summary>Example, using <code>ToCamelCaseWithInitialisms</code></summary>

By default, `oapi-codegen` will perform camel-case conversion, so for a spec such as:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Example code for the `name-normalizer` output option
paths:
  /api/pets/{petId}:
    get:
      summary: Get pet given identifier.
      operationId: getHttpPet
      parameters:
      - name: petId
        in: path
        required: true
        schema:
          type: string
      responses:
        '200':
          description: valid pet
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Pet'
components:
  schemas:
    Pet:
      type: object
      required:
        - uuid
        - name
      properties:
        uuid:
          type: string
          description: The pet uuid.
        name:
          type: string
          description: The name of the pet.
    Error:
      required:
        - code
        - message
      properties:
        code:
          type: integer
          format: int32
          description: Error code
        message:
          type: string
          description: Error message
    OneOf2things:
      description: "Notice that the `things` is not capitalised"
      oneOf:
        - type: object
          required:
            - id
          properties:
            id:
              type: integer
        - type: object
          required:
            - id
          properties:
            id:
              type: string
              format: uuid
```

This will produce:

```go
// OneOf2things Notice that the `things` is not capitalised
type OneOf2things struct {
	union json.RawMessage
}

// Pet defines model for Pet.
type Pet struct {
	// Name The name of the pet.
	Name string `json:"name"`

	// UUID The pet uuid.
	UUID string `json:"uuid"`
}

// The interface specification for the client above.
type ClientInterface interface {
	// GetHTTPPet request
	GetHTTPPet(ctx context.Context, petID string, reqEditors ...RequestEditorFn) (*http.Response, error)
}
```

</details>


For more details of what the resulting code looks like, check out [the test cases](internal/test/outputoptions/name-normalizer/).

## Examples

The [examples directory](examples) contains some additional cases which are useful examples for how to use `oapi-codegen`, including how you'd take the Petstore API and implement it with `oapi-codegen`.

You could also find some cases of how the project can be used by checking out our [internal test cases](internal/test) which are real-world usages that make up our regression tests.

### Blog posts

We love reading posts by the community about how to use the project.

Here are a few we've found around the Web:

- [Building a Go RESTful API with design-first OpenAPI contracts](https://www.jvt.me/posts/2022/07/12/go-openapi-server/)
- [A Practical Guide to Using oapi-codegen in Golang API Development with the Fiber Framework](https://medium.com/@fikihalan/a-practical-guide-to-using-oapi-codegen-in-golang-api-development-with-the-fiber-framework-bce2a59380ae)
- [Generating Go server code from OpenAPI 3 definitions](https://ldej.nl/post/generating-go-from-openapi-3/)
- [Go Client Code Generation from Swagger and OpenAPI](https://medium.com/@kyodo-tech/go-client-code-generation-from-swagger-and-openapi-a0576831836c)
- [Go oapi-codegen + request validation](https://blog.commitsmart.com/go-oapi-codegen-request-validation-285398b37dc8)
- [Streamlining Go + Chi Development: Generating Code from an OpenAPI Spec](https://i4o.dev/blog/oapi-codegen-with-chi-router)

Got one to add? Please raise a PR!

## Frequently Asked Questions (FAQs)

### Does `oapi-codegen` support OpenAPI 3.1?

No, we don't currently.

OpenAPI 3.1 support is [awaiting upstream support](https://github.com/oapi-codegen/oapi-codegen/issues/373).

In the meantime, you could follow [steps from this blog post](https://www.jvt.me/posts/2025/05/04/oapi-codegen-trick-openapi-3-1/) to [use OpenAPI Overlay](#modifying-the-input-openapi-specification-with-openapi-overlay) to "downgrade" the OpenAPI 3.1 spec to OpenAPI 3.0.

### How does `oapi-codegen` handle `anyOf`, `allOf` and `oneOf`?

`oapi-codegen` supports `anyOf`, `allOf` and `oneOf` for generated code.

For instance, through the following OpenAPI spec:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Using complex schemas
  description: An example of `anyOf`, `allOf` and `oneOf`
components:
  schemas:
    # base types
    Client:
      type: object
      required:
        - name
      properties:
        name:
          type: string
    Identity:
      type: object
      required:
        - issuer
      properties:
        issuer:
          type: string

    # allOf performs a union of all types defined
    ClientWithId:
      allOf:
        - $ref: '#/components/schemas/Client'
        - properties:
            id:
              type: integer
          required:
            - id

    # allOf performs a union of all types defined, but if there's a duplicate field defined, it'll be overwritten by the last schema
    # https://github.com/oapi-codegen/oapi-codegen/issues/1569
    IdentityWithDuplicateField:
      allOf:
        # `issuer` will be ignored
        - $ref: '#/components/schemas/Identity'
        # `issuer` will be ignored
        - properties:
            issuer:
              type: integer
        # `issuer` will take precedence
        - properties:
            issuer:
              type: object
              properties:
                name:
                  type: string
              required:
                - name

    # anyOf results in a type that has an `AsClient`/`MergeClient`/`FromClient` and an `AsIdentity`/`MergeIdentity`/`FromIdentity` method so you can choose which of them you want to retrieve
    ClientAndMaybeIdentity:
      anyOf:
        - $ref: '#/components/schemas/Client'
        - $ref: '#/components/schemas/Identity'

    # oneOf results in a type that has an `AsClient`/`MergeClient`/`FromClient` and an `AsIdentity`/`MergeIdentity`/`FromIdentity` method so you can choose which of them you want to retrieve
    ClientOrIdentity:
      oneOf:
        - $ref: '#/components/schemas/Client'
        - $ref: '#/components/schemas/Identity'
```

This results in the following types:

<details>

<summary>Base types</summary>

```go
// Client defines model for Client.
type Client struct {
	Name string `json:"name"`
}

// Identity defines model for Identity.
type Identity struct {
	Issuer string `json:"issuer"`
}
```

</details>

<details>

<summary><code>allOf</code></summary>

```go
// ClientWithId defines model for ClientWithId.
type ClientWithId struct {
	Id   int    `json:"id"`
	Name string `json:"name"`
}

// IdentityWithDuplicateField defines model for IdentityWithDuplicateField.
type IdentityWithDuplicateField struct {
	Issuer struct {
		Name string `json:"name"`
	} `json:"issuer"`
}
```

</details>

<details>

<summary><code>anyOf</code></summary>

```go
import (
	"encoding/json"

	"github.com/oapi-codegen/runtime"
)

// ClientAndMaybeIdentity defines model for ClientAndMaybeIdentity.
type ClientAndMaybeIdentity struct {
	union json.RawMessage
}

// AsClient returns the union data inside the ClientAndMaybeIdentity as a Client
func (t ClientAndMaybeIdentity) AsClient() (Client, error) {
	var body Client
	err := json.Unmarshal(t.union, &body)
	return body, err
}

// FromClient overwrites any union data inside the ClientAndMaybeIdentity as the provided Client
func (t *ClientAndMaybeIdentity) FromClient(v Client) error {
	b, err := json.Marshal(v)
	t.union = b
	return err
}

// MergeClient performs a merge with any union data inside the ClientAndMaybeIdentity, using the provided Client
func (t *ClientAndMaybeIdentity) MergeClient(v Client) error {
	b, err := json.Marshal(v)
	if err != nil {
		return err
	}

	merged, err := runtime.JSONMerge(t.union, b)
	t.union = merged
	return err
}

// AsIdentity returns the union data inside the ClientAndMaybeIdentity as a Identity
func (t ClientAndMaybeIdentity) AsIdentity() (Identity, error) {
	var body Identity
	err := json.Unmarshal(t.union, &body)
	return body, err
}

// FromIdentity overwrites any union data inside the ClientAndMaybeIdentity as the provided Identity
func (t *ClientAndMaybeIdentity) FromIdentity(v Identity) error {
	b, err := json.Marshal(v)
	t.union = b
	return err
}

// MergeIdentity performs a merge with any union data inside the ClientAndMaybeIdentity, using the provided Identity
func (t *ClientAndMaybeIdentity) MergeIdentity(v Identity) error {
	b, err := json.Marshal(v)
	if err != nil {
		return err
	}

	merged, err := runtime.JSONMerge(t.union, b)
	t.union = merged
	return err
}

func (t ClientAndMaybeIdentity) MarshalJSON() ([]byte, error) {
	b, err := t.union.MarshalJSON()
	return b, err
}

func (t *ClientAndMaybeIdentity) UnmarshalJSON(b []byte) error {
	err := t.union.UnmarshalJSON(b)
	return err
}


```

</details>

<details>

<summary><code>oneOf</code></summary>

```go
// AsClient returns the union data inside the ClientOrIdentity as a Client
func (t ClientOrIdentity) AsClient() (Client, error) {
	var body Client
	err := json.Unmarshal(t.union, &body)
	return body, err
}

// FromClient overwrites any union data inside the ClientOrIdentity as the provided Client
func (t *ClientOrIdentity) FromClient(v Client) error {
	b, err := json.Marshal(v)
	t.union = b
	return err
}

// MergeClient performs a merge with any union data inside the ClientOrIdentity, using the provided Client
func (t *ClientOrIdentity) MergeClient(v Client) error {
	b, err := json.Marshal(v)
	if err != nil {
		return err
	}

	merged, err := runtime.JSONMerge(t.union, b)
	t.union = merged
	return err
}

// AsIdentity returns the union data inside the ClientOrIdentity as a Identity
func (t ClientOrIdentity) AsIdentity() (Identity, error) {
	var body Identity
	err := json.Unmarshal(t.union, &body)
	return body, err
}

// FromIdentity overwrites any union data inside the ClientOrIdentity as the provided Identity
func (t *ClientOrIdentity) FromIdentity(v Identity) error {
	b, err := json.Marshal(v)
	t.union = b
	return err
}

// MergeIdentity performs a merge with any union data inside the ClientOrIdentity, using the provided Identity
func (t *ClientOrIdentity) MergeIdentity(v Identity) error {
	b, err := json.Marshal(v)
	if err != nil {
		return err
	}

	merged, err := runtime.JSONMerge(t.union, b)
	t.union = merged
	return err
}

func (t ClientOrIdentity) MarshalJSON() ([]byte, error) {
	b, err := t.union.MarshalJSON()
	return b, err
}

func (t *ClientOrIdentity) UnmarshalJSON(b []byte) error {
	err := t.union.UnmarshalJSON(b)
	return err
}
```

</details>

For more info, check out [the example code](examples/anyof-allof-oneof/).

### How can I ignore parts of the spec I don't care about?

By default, `oapi-codegen` will generate everything from the specification.

If you'd like to reduce what's generated, you can use one of a few options in [the configuration file](#usage) to tune the generation of the resulting output:

```yaml
# yaml-language-server: $schema=https://raw.githubusercontent.com/oapi-codegen/oapi-codegen/HEAD/configuration-schema.json
output-options:
  include-tags: []
  exclude-tags: []
  include-operation-ids: []
  exclude-operation-ids: []
  exclude-schemas: []
```

Check [the docs](https://pkg.go.dev/github.com/oapi-codegen/oapi-codegen/v2/pkg/codegen#OutputOptions) for more details of usage.

### Should I commit the generated code?

We recommend doing so, yes, for the following reasons:

- It means it's easier to view the impact of a change - be it due to an upgrade of `oapi-codegen`, or a change to your spec - and has helped catch (possibly) breaking changes in the past more easily
- It then allows your codebase to be consumed as a library, as all the files are committed

This means you'll need to have your CI/CD pipeline validate that generated files are all up-to-date, but that's a fairly straightforward piece of work.

### Should I lint the generated code?

We really ask that you don't. Although it intends to be idiomatic Go code, it's not expected to pass all the various linting rules that your project may apply.

> [!NOTE]
> We will, on occasion, improve the generated code to fix some linting warnings, such as those from `go vet`, but this should not be an expected change.

### I've just updated my version of `kin-openapi`, and now I can't build my code 😠

The [kin-openapi](https://github.com/getkin/kin-openapi) project - which we 💜 for providing a great library and set of tooling for interacting with OpenAPI - is a pre-v1 release, which means that they're within their rights to push breaking changes.

This may lead to breakage in your consuming code, and if so, sorry that's happened!

We'll be aware of the issue, and will work to update both the core `oapi-codegen` and the middlewares accordingly.

## Contributors

We're very appreciative of [the many contributors over the years](https://github.com/oapi-codegen/oapi-codegen/graphs/contributors) and the ongoing use of the project 💜

<a href="https://github.com/oapi-codegen/oapi-codegen/graphs/contributors">
  <img src="https://contrib.rocks/image?repo=oapi-codegen/oapi-codegen" />
</a>

## Sponsors

For the most part, `oapi-codegen` is maintained in two busy peoples' free time. As noted in [Creating a more sustainable model for `oapi-codegen` in the future](https://github.com/oapi-codegen/oapi-codegen/discussions/1606), we're looking to make this a more sustainable project in the future.

Please consider sponsoring us through GitHub Sponsors either [on the organisation](https://github.com/sponsors/oapi-codegen/) or [directly for Jamie](https://github.com/sponsors/jamietanna/), which helps work towards us being able to maintain the project long term.

See [this blog post from Tidelift](https://blog.tidelift.com/paying-maintainers-the-howto) for more details on how to talk to your company about sponsoring maintainers of (Open Source) projects you depend on.

In addition, we are also generously sponsored by the following folks, each of whom provide sponsorship for 1 hour of work a month:

<p align="center">
	<a href="https://www.devzero.io/lp/dev-environment?utm_campaign=github&utm_source=oapi-codegen%20repo&utm_medium=github%20sponsorship">
		<picture>
		  <source media="(prefers-color-scheme: light)" srcset=".github/sponsors/devzero-light.svg">
		  <source media="(prefers-color-scheme: dark)" srcset=".github/sponsors/devzero-dark.svg">
		  <img alt="DevZero logo" src=".github/sponsors/devzero-dark.svg" height="100px">
		</picture>
	</a>
</p>

<p align="center">
	<a href="https://sandbox.speakeasy.com/?s=iQ5hEdrjLCii&utm_source=oapi-codegen+repo&utm_medium=github+sponsorship">
		<picture>
		  <source media="(prefers-color-scheme: light)" srcset=".github/sponsors/speakeasy-light.svg">
		  <source media="(prefers-color-scheme: dark)" srcset=".github/sponsors/speakeasy-dark.svg">
		  <img alt="Speakeasy logo" src=".github/sponsors/speakeasy-dark.svg" height="60px">
		</picture>
	</a>
</p>

<p align="center">
	<a href="https://cybozu.co.jp/?utm_source=oapi-codegen+repo&utm_medium=github+sponsorship">
		<img alt="Cybozu logo" src=".github/sponsors/cybozu.svg" height="100px">
	</a>
</p>

<p align="center">
	<a href="https://livepeer.org/?utm_source=oapi-codegen+repo&utm_medium=github+sponsorship">
		<picture>
		  <source media="(prefers-color-scheme: light)" srcset=".github/sponsors/livepeer-light.svg">
		  <source media="(prefers-color-scheme: dark)" srcset=".github/sponsors/livepeer-dark.svg">
		  <img alt="Livepeer logo" src=".github/sponsors/livepeer-dark.svg" height="50px">
		</picture>
	</a>
</p>

(Note that the order of appearance the order in which sponsorship was received)
 readmeEtag: '"9cb70c7c3dac8f923818a19ba6c0fff42b9b17ce"' readmeLastModified: Mon, 03 Nov 2025 13:32:47 GMT repositoryId: 173009358 description: Generate Go client and server boilerplate from OpenAPI 3 specifications created: '2019-02-27T23:59:59Z' updated: '2026-02-06T04:24:27Z' language: Go archived: false stars: 8035 watchers: 41 forks: 1011 owner: oapi-codegen logo: https://avatars.githubusercontent.com/u/142752710?v=4 license: Apache-2.0 repoEtag: '"dfdc98028c12e7f949e740d4c128e253559de39140ac121a03f2df03da2b3706"' repoLastModified: Fri, 06 Feb 2026 04:24:27 GMT foundInMaster: true id: 770a221110cc4ef0e399089f5caf8694 oldLocations: - https://github.com/deepmap/oapi-codegen - source: openapi3 tags name: OpenAPI TypeScript category: - Converters - SDK language: TypeScript source_description: >- Convert static OpenAPI schemas to TypeScript types quickly using pure Node.js. Fast, lightweight, (almost) dependency-free, and no Java/node-gyp/running OpenAPI servers necessary. link: https://github.com/drwpow/openapi-typescript v2: true v3: true v3_1: true repository: https://github.com/openapi-ts/openapi-typescript id: 9880f7b9f35e02f8dfae31a519803f18 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXR5cGVzY3JpcHQKClRvb2xzIGZvciBjb25zdW1pbmcgT3BlbkFQSSBzY2hlbWFzIGluIFR5cGVTY3JpcHQuCgojIyDwn5OmIFBhY2thZ2VzCgo8YSBocmVmPSIuL3BhY2thZ2VzL29wZW5hcGktdHlwZXNjcmlwdCI+PGltZyBzcmM9Ii4vZG9jcy9wdWJsaWMvYXNzZXRzL29wZW5hcGktdHMuc3ZnIiBhbHQ9Im9wZW5hcGktdHlwZXNjcmlwdCIgd2lkdGg9IjIwMCIgaGVpZ2h0PSI0MCIgLz48YnIgLz4KR2VuZXJhdGUgVHlwZVNjcmlwdCB0eXBlcyBmcm9tIHN0YXRpYyBPcGVuQVBJIHNjaGVtYXMKPC9hPgoKPGEgaHJlZj0iLi9wYWNrYWdlcy9vcGVuYXBpLWZldGNoIj48aW1nIHNyYz0iLi9kb2NzL3B1YmxpYy9hc3NldHMvb3BlbmFwaS1mZXRjaC5zdmciIGFsdD0ib3BlbmFwaS1mZXRjaCIgd2lkdGg9IjIxNiIgaGVpZ2h0PSI0MCIgLz48YnIgLz4KVWx0cmEtZmFzdCBmZXRjaGluZyBmb3IgVHlwZVNjcmlwdCBnZW5lcmF0ZWQgYXV0b21hdGljYWxseSBmcm9tIHlvdXIgT3BlbkFQSSBzY2hlbWEuCjwvYT4KCiMjIOKtkCBTcG9uc29ycwoKIyMjIPCfpYcgR29sZCBTcG9uc29ycwoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly96dXBsby5saW5rL29wZW5hcGktdHMtZ2giPjxpbWcgd2lkdGg9IjI4OCIgaGVpZ2h0PSIxMjAiIGFsdD0iWnVwbG8iIHNyYz0iLi9kb2NzL3B1YmxpYy9hc3NldHMvenVwbG8uc3ZnIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly93d3cuc3BlYWtlYXN5LmNvbS9lZGl0b3I/dXRtX3NvdXJjZT1wb3dfb3BlbmFwaV90cyIgdGFyZ2V0PSJfYmxhbmsiPjxpbWcgd2lkdGg9IjI4OCIgaGVpZ2h0PSIxMjAiIHNyYz0iLi9kb2NzL3B1YmxpYy9hc3NldHMvc3BlYWtlYXN5LnBuZyIgLz48L2E+CjwvcD4KCiMjIyDwn6WIIFNpbHZlciBTcG9uc29ycwoKPHAgYWxpZ249ImNlbnRlciI+PGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL25hbmFiaXQtaW5jIj48aW1nIHdpZHRoPSIxMjgiIGhlaWdodD0iMTI4IiBhbHQ9Im5hbmFiaXQiIHNyYz0iaHR0cHM6Ly9hdmF0YXJzLmdpdGh1YnVzZXJjb250ZW50LmNvbS91LzE1NDEyNjk3Nj9zPTIwMCZ2PTQiPjwvYT48L3A+CgojIyMgQmFja2VycwoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2N1c3RvbWVyaW8iPjxpbWcgd2lkdGg9IjY0IiBoZWlnaHQ9IjY0IiBhbHQ9IkBjdXN0b21lcmlvIG9uIEdpdEh1YiIgc3JjPSJodHRwczovL2F2YXRhcnMuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3UvMTE1MjA3OT9zPTIwMCZ2PTQiPjwvYT4KICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vc2hhdW5wZXJzYWQiPjxpbWcgd2lkdGg9IjY0IiBoZWlnaHQ9IjY0IiBhbHQ9IkBzaGF1bnBlcnNhZCBvbiBHaXRIdWIiIHNyYz0iaHR0cHM6Ly9hdmF0YXJzLmdpdGh1YnVzZXJjb250ZW50LmNvbS91LzE3MDI5NzY/cz01MiZ2PTQiPjwvYT4KICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20veW9zaGkybm8iPjxpbWcgd2lkdGg9IjY0IiBoZWlnaHQ9IjY0IiBhbHQ9IkB5b3NoaTJubyBvbiBHaXRIdWIiIHNyYz0iaHR0cHM6Ly9hdmF0YXJzLmdpdGh1YnVzZXJjb250ZW50LmNvbS91LzU3MDU5NzA1P3Y9NCI+PC9hPgogIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9zaWtldHlhbiI+PGltZyB3aWR0aD0iNjQiIGhlaWdodD0iNjQiIGFsdD0iQHNpa2V0eWFuIG9uIEdpdEh1YiIgc3JjPSJodHRwczovL2F2YXRhcnMuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3UvMTI3NzIxMTg/dj00Ij48L2E+CjwvcD4KCkJlY29tZSBhIHNwb25zb3IgYnkgc3VwcG9ydGluZyB0aGlzIHByb2plY3Qgb24gW09wZW5Db2xsZWN0aXZlXShodHRwczovL29wZW5jb2xsZWN0aXZlLmNvbS9vcGVuYXBpLXRzKSEKCjxhIGhyZWY9Imh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL29wZW5hcGktdHMvZG9uYXRlIiB0YXJnZXQ9Il9ibGFuayI+CiAgPGltZyBzcmM9Imh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL29wZW5hcGktdHMvZG9uYXRlL2J1dHRvbkAyeC5wbmc/Y29sb3I9Ymx1ZSIgd2lkdGg9IjMwMCIgLz4KPC9hPgoKIyMg8J+knSBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIGFwcHJlY2lhdGVkIGFuZCB3ZWxjb21lISBTZWUgdGhlIGFwcHJvcHJpYXRlIGd1aWRlIGZvciBlYWNoIHBhY2thZ2U6CgotIFtDb250cmlidXRpbmcgdG8gb3BlbmFwaS10eXBlc2NyaXB0XSguL3BhY2thZ2VzL29wZW5hcGktdHlwZXNjcmlwdC9DT05UUklCVVRJTkcubWQpCi0gW0NvbnRyaWJ1dGluZyB0byBvcGVuYXBpLWZldGNoXSguL3BhY2thZ2VzL29wZW5hcGktZmV0Y2gvQ09OVFJJQlVUSU5HLm1kKQotIFtDb250cmlidXRpbmcgdG8gZG9jc10oLi9kb2NzL0NPTlRSSUJVVElORy5tZCkKCiMjIOKZpe+4jyBUaGFua3MKCi0gVGhhbmtzIHRvIFt0aGUgUHJvamVjdCBTcG9uc29yc10oIy1zcG9uc29ycykgZm9yIGtlZXBpbmcgdGhpcyBwcm9qZWN0IGdvaW5nIQotIFRoYW5rcyB0byBbZG96ZW5zIG9mIGxvdmVseSwgc21hcnQgY29udHJpYnV0b3JzXShodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS10cy9vcGVuYXBpLXR5cGVzY3JpcHQvZ3JhcGhzL2NvbnRyaWJ1dG9ycykgdGhhdCBtYWRlIHRoaXMgbGlicmFyeSBwb3NzaWJsZQotIFRoYW5rcyB0byBbVml0ZXByZXNzXShodHRwczovL3ZpdGVwcmVzcy5kZXYvKSBmb3IgdGhlIGRvY3Mgc2l0ZQotIFRoYW5rcyB0byBbQ2xvdWRmbGFyZSBQYWdlc10oaHR0cHM6Ly9wYWdlcy5jbG91ZGZsYXJlLmNvbS8pIGZvciBkb2NzIHNpdGUgaG9zdGluZwotIFRoYW5rcyB0byBbQWxnb2xpYV0oaHR0cHM6Ly93d3cuYWxnb2xpYS5jb20vKSBmb3IgdGhlIGRvY3Mgc2l0ZSBzZWFyY2gK readmeEtag: '"dbbc6be6fa734d153ec40acf7b8fa76a85a6fa9a"' readmeLastModified: Mon, 11 Aug 2025 04:11:36 GMT repositoryId: 161691324 description: Generate TypeScript types from OpenAPI 3 specs created: '2018-12-13T20:28:01Z' updated: '2026-02-06T04:03:41Z' language: TypeScript archived: false stars: 7882 watchers: 23 forks: 615 owner: openapi-ts logo: https://avatars.githubusercontent.com/u/172838200?v=4 license: MIT repoEtag: '"f77973cfc28b3dc82d018fd9e97f4c8825fa089806d4807379c5b838cfe4fccd"' repoLastModified: Fri, 06 Feb 2026 04:03:41 GMT foundInMaster: true oldLocations: - https://github.com/drwpow/openapi-typescript - source: openapi3 tags repository: https://github.com/xiaoymin/knife4j v3: true id: e1d4be4e3f97f5154f27fa1601e56818 repositoryMetadata: base64Readme: >- a25pZmU0auaYr+S4ukphdmEgTVZD5qGG5p626ZuG5oiQU3dhZ2dlcueUn+aIkEFwaeaWh+aho+eahOWinuW8uuino+WGs+aWueahiCzliY3ouqvmmK9zd2FnZ2VyLWJvb3RzdHJhcC11aSzlj5blkI1rbmlmZTRq5piv5biM5pyb5aW56IO95YOP5LiA5oqK5YyV6aaW5LiA5qC35bCP5benLOi9u+mHjyzlubbkuJTlip/og73lvLrmgo0hCgprbmlmZTRq55qE5YmN6Lqr5pivYHN3YWdnZXItYm9vdHN0cmFwLXVpYO+8jOS4uuS6huWlkeWQiOW+ruacjeWKoeeahOaetuaehOWPkeWxlSznlLHkuo7ljp/mnaVgc3dhZ2dlci1ib290c3RyYXAtdWlg6YeH55So55qE5piv5ZCO56uvSmF2YeS7o+eggSvliY3nq69Vaea3t+WQiOaJk+WMheeahOaWueW8jyzlnKjlvq7mnI3liqHmnrbmnoTkuIvmmL7nmoTlvojoh4Pogr8s5Zug5q2k6aG555uu5q2j5byP5pu05ZCN5Li6YGtuaWZlNGpgCgrmm7TlkI3lkI7kuLvopoHkuJPms6jnmoTmlrnpnaIKCi0g5YmN5ZCO56uvSmF2YeS7o+eggeS7peWPiuWJjeerr1Vp5qih5Z2X6L+b6KGM5YiG56a7LOWcqOW+ruacjeWKoeaetuaehOS4i+S9v+eUqOabtOWKoOeBtea0uwotIOaPkOS+m+S4k+azqOS6jlN3YWdnZXLnmoTlop7lvLrop6PlhrPmlrnmoYgs5LiN5ZCM5LqO5Y+q5piv5pS55ZaE5aKe5by65YmN56uvVWnpg6jliIYKLSDmj5Dkvpvmm7TlpJrngbXmtLvnmoTkuK3pl7Tku7bmlrnmoYhc5bel5YW3CgojIyDpobnnm67mqKHlnZcKCuebruWJjeS7k+W6k+S4u+imgeeahOaooeWdl+ivtOaYju+8mgoKfCDmqKHlnZflkI3np7AgICAgICAgICAgICAgfCDor7TmmI4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgLS0tLS0tLS0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAp8IGtuaWZlNGogICAgICAgICAgICAgIHwg5Li6SmF2YSBNVkPmoYbmnrbpm4bmiJBTd2FnZ2Vy55qE5aKe5by66Kej5Yaz5pa55qGI77yMSmF2YeeUn+aAgeS4i+eahOS4remXtOS7tuWwgeijhSAgICAgICAgICAgICAgICAgICAgICB8Cnwga25pZmU0ai1pbnNpZ2h0IHzlvIDnrrHljbPnlKjnmoTni6znq4vop6PlhrPmlrnmoYgs5o+Q5L6b5a6Y5pa5W0RvY2tlcumVnOWDj10oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yZXBvc2l0b3J5L2RvY2tlci94aWFveW1pbi9rbmlmZTRqL2dlbmVyYWwpLOWfuuS6jlNwcmluZyBCb290IDMuMOe8luWGme+8jOafpeeci1vkvb/nlKjmlofmoaNdKGh0dHBzOi8vZG9jLnhpYW9taW5mby5jb20vZG9jcy9taWRkbGV3YXJlLXNvdXJjZXMvZGVza3RvcC1pbnRyb2R1Y3Rpb24pfAp8IGtuaWZlNGotZG9jICAgIHwga25pZmU0auWumOaWueaWh+aho++8jOWfuuS6jkRvY3VzYXVydXPnvJblhpnvvIzlj4LkuI7otKHnjK7or7db5Y+C6ICD5paH5qGjXShodHRwczovL2RvYy54aWFvbWluZm8uY29tL2RvY3MvY29tbXVuaXR5L2pvaW51cykgICAgICB8Cnwga25pZmU0ai12dWUgICAgICB85b2T5YmNS25pZmU0aueahOWJjeerr+a6kOegge+8jOWfuuS6jlZ1ZTIuMOe8luWGmSAgICAgICAgICAgICAgICAgICAgICAgICB8Cnwga25pZmU0ai12dWUzICAgICAgfOW9k+WJjUtuaWZlNGrnmoTliY3nq6/mupDnoIHvvIzln7rkuo5WdWUzLjDnvJblhpnvvIzor6Xku6PnoIHlupPmnaXoh6rotKHnjK7ogIXvvIznm67liY3lsJrmnKrmipXlhaXkvb/nlKggICAgICAgICAgICAgICAgICAgICAgICB8Cnwga25pZmU0ai1mcm9udCAgICAgICAgfCBrbmlmZTRq55qE5YmN56uv5p625p6E5Luj56CBLOebruWJjeaYr+inhOWIkumYtuaute+8jOivpeaooeWdl+Wwmuacque8lueggSzmnInmg7Pms5XnmoTlj6/ku6XpgJrov4fkuqTmtYHnvqTkuI7kvZzogIXmsp/pgJogICAgfAoKCgoKIyMg6I635Y+W5biu5YqpCgoKKirlrpjnvZHmlofmoaPvvJoqKiBbaHR0cHM6Ly9kb2MueGlhb21pbmZvLmNvbS9dKGh0dHBzOi8vZG9jLnhpYW9taW5mby5jb20vKQoKKirpooTop4jlnLDlnYA6KiogW2h0dHBzOi8vZG9jLnhpYW9taW5mby5jb20vZGVtby9kb2MuaHRtbF0oaHR0cHM6Ly9kb2MueGlhb21pbmZvLmNvbS9kZW1vL2RvYy5odG1sKQoKKipEZW1v56S65L6LOioqIFtodHRwczovL2dpdGVlLmNvbS94aWFveW0vc3dhZ2dlci1ib290c3RyYXAtdWktZGVtb10oaHR0cHM6Ly9naXRlZS5jb20veGlhb3ltL3N3YWdnZXItYm9vdHN0cmFwLXVpLWRlbW8pCgoqKkRlbW/or7TmmI7vvJoqKiBbaHR0cHM6Ly9kb2MueGlhb21pbmZvLmNvbS9kb2NzL2FjdGlvbi9hY3Rpb24tc2ltcGxlXShodHRwczovL2RvYy54aWFvbWluZm8uY29tL2RvY3MvYWN0aW9uL2FjdGlvbi1zaW1wbGUpCgoqKuS9nOiAheS6pOa1ge+8mioqIOWFs+azqOWFrOS8l+WPtyJLbmlmZTRqIu+8jOeCueWHu+iPnOWNleKAnOS6pOa1gee+pOKAneiOt+WPluWKoOe+pOS6jOe7tOeggQoKCiFb6L6T5YWl5Zu+54mH6K+05piOXShodHRwczovL2ZvcnVkYS5naXRlZS5jb20vaW1hZ2VzLzE2NjEwNTM4Njc1Njk0ODAzMTAv5omr56CBX+aQnOe0ouiBlOWQiOS8oOaSreagt+W8jy3moIflh4boibLniYgucG5nICLmiavnoIFf5pCc57Si6IGU5ZCI5Lyg5pKt5qC35byPLeagh+WHhuiJsueJiC5wbmciKQoKCgoKIyMg54m55Yir5aOw5piOCgrkuI3nrqHmmK9rbmlmZTRq6L+Y5pivc3dhZ2dlci1ib290c3RyYXAtdWkKCuWvueWkluaPkOS+m+eahOWcsOWdgOS+neeEtuaYr2RvYy5odG1sCgrorr/pl67vvJpodHRwOi8vaXA6cG9ydC9kb2MuaHRtbAoK5Y2z5Y+v5p+l55yL5paH5qGjCgoqKui/meaYr+awuOi/nOS4jeS8muaUueWPmOeahCoqCgoKCiMjIOeVjOmdouaViOaenAoKIVvmjqXlj6Por7TmmI5dKHN0YXRpYy8xLnBuZykKCiFb5o6l5Y+j6LCD6K+VXShzdGF0aWMvOC5wbmcpCgogIAojIyDwn6SdIOeJueWIq+m4o+iwoiAKCi0g5oSf6LCiIFtKZXRCcmFpbnNdKGh0dHBzOi8vamIuZ2cvT3BlblNvdXJjZVN1cHBvcnQpIOaPkOS+m+eahOWFjei0ueW8gOa6kCBMaWNlbnNl77yaCgo8aW1nIHNyYz0iaHR0cHM6Ly9yZXNvdXJjZXMuamV0YnJhaW5zLmNvbS9zdG9yYWdlL3Byb2R1Y3RzL2NvbXBhbnkvYnJhbmQvbG9nb3MvamJfYmVhbS5wbmciICB3aWR0aD0iMTAwIiBoZWlnaHQ9IjEwMCIgLz4KCgo8aDIgYWxpZ249ImNlbnRlciI+CvCfkqogQ29udHJpYnV0b3JzIPCfkqoKPC9oMj4KCjxwIGFsaWduPSJjZW50ZXIiPgpPdXIgY29udHJpYnV0b3JzIGhhdmUgbWFkZSB0aGlzIHByb2plY3QgcG9zc2libGUuIFRoYW5rIHlvdSEg8J+Zjwo8L3A+CgogCjxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS94aWFveW1pbi9rbmlmZTRqL2dyYXBocy9jb250cmlidXRvcnMiPgogIDxpbWcgc3JjPSJodHRwczovL2NvbnRyaWIucm9ja3MvaW1hZ2U/cmVwbz14aWFveW1pbi9rbmlmZTRqIiB3aWR0aD0iMTAwJSIvPgo8L2E+CjxkaXYgYWxpZ249ImNlbnRlciI+CjxzdWI+TWFkZSB3aXRoIDxhIGhyZWY9Imh0dHBzOi8vY29udHJpYi5yb2NrcyI+Y29udHJpYi5yb2NrczwvYT4uPC9zdWI+CjwvZGl2PgogCgo8cCBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL3N0YXItaGlzdG9yeS5jb20vI3hpYW95bWluL2tuaWZlNGomRGF0ZSI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9hcGkuc3Rhci1oaXN0b3J5LmNvbS9zdmc/cmVwb3M9eGlhb3ltaW4va25pZmU0aiZ0eXBlPURhdGUiIGFsdD0iU3RhciBIaXN0b3J5IENoYXJ0Ij4KICA8L2E+CjwvcD4K readmeEtag: '"71cfc0b45339fa88e06fa99776d7f8b68a1b5903"' readmeLastModified: Tue, 13 Aug 2024 02:24:00 GMT repositoryId: 88701696 description: Knife4j is a set of Swagger2 and OpenAPI3 All-in-one enhancement solution created: '2017-04-19T04:44:28Z' updated: '2026-02-06T01:52:22Z' language: HTML archived: false stars: 4482 watchers: 75 forks: 660 owner: xiaoymin logo: https://avatars.githubusercontent.com/u/7894406?v=4 license: Apache-2.0 repoEtag: '"f598d7140885df17ce18ce7560466bf8ea288adf2f383d7ff2aaf540050c39ad"' repoLastModified: Fri, 06 Feb 2026 01:52:22 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/apis-guru/openapi-directory v3: true repositoryMetadata: base64Readme: >- ![banner]

[![APIs in collection][numApis-image]][apisDir-link]
[![OpenAPI definitions][numSpecs-image]][apisDir-link]
[![Endpoints][endpoints-image]][apisDir-link]

[![Share on Twitter][twitter-image]][twitter-link]
[![Follow on Twitter][twitterFollow-image]][twitterFollow-link]
<a href="#backers" alt="sponsors on Open Collective"><img src="https://opencollective.com/openapi-directory/backers/badge.svg" /></a> <a href="#sponsors" alt="Sponsors on Open Collective"><img src="https://opencollective.com/openapi-directory/sponsors/badge.svg" /></a> <a href="#partners" alt="Partners on Open Collective"><img src="https://opencollective.com/openapi-directory/partner/badge.svg" /></a>

Directory of API definitions in [OpenAPI(fka Swagger)](https://openapis.org) [2.0](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/2.0.md) and [3.x](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md) formats.<BR>
API access to collection: [Go!][apiDoc-link] - We also have RSS feeds for [added](https://api.apis.guru/v2/added.rss) and [updated](http://api.apis.guru/v2/list.rss) APIs.

[![Add API][addAPI-image]][addAPI-link]

Our goal is to create the most comprehensive, standards-compliant and up-to-date directory of machine-readable API definitions, with the following principles:
- Open source, community driven project.
- Only publicly available APIs (free or paid).
- Anyone can add or change an API, not only API owners.
- All data can be accessed through a [REST API][apiDoc-link].

If you reference this project and you need an icon or even a banner, check our [branding guide](https://github.com/APIs-guru/branding).

APIs.guru provide services and consultancy around the GraphQL and OpenAPI specifications and APIs in general.
You can contact us at founders@apis.guru

[![Become a partner](https://opencollective.com/openapi-directory/tiers/partner.svg?avatarHeight=36&width=600)](https://opencollective.com/openapi-directory)

What does APIs.guru do?
--------------------------
* Filter out private and non-reliable APIs
* Convert non-OpenAPI formats into OpenAPI 3.x
* Fix mistakes, ~80% of definitions have some
* Add additional data, like: logo, categories, …
* Update definitions on at least a weekly basis

API definition acceptance criteria
----------------------------------
* Public - anyone can access it as long as they follow some clearly defined steps (subscribe, pay fees, etc.).
* Persistent - API is made with long-lived goal, and not for a particular event (conference, hackathon, etc.).
* Useful - API should provide useful functionality not only for its owner.

Update procedure
--------------------------
All definitions are automatically updated from their original source.
You can see this in the `x-origin` property within each [openapi.yaml](https://github.com/APIs-guru/openapi-directory/search?utf8=%E2%9C%93&q=x-origin+filename%3Aopenapi.yaml) or [swagger.yaml](https://github.com/APIs-guru/openapi-directory/search?utf8=%E2%9C%93&q=x-origin+filename%3Aswagger.yaml) file.
We run our update script at least weekly and automatically revalidate before committing.
If you see some APIs are not updated for more than 2 weeks please open [an issue](https://github.com/APIs-guru/openapi-directory/issues/new).

[@seriousme](https://github.com/seriousme) kindly runs a backup validation process based on the official OpenAPI-Specification [JSON Schemas](https://github.com/OAI/OpenAPI-Specification/tree/main/schemas) [here](https://github.com/seriousme/openapi-schema-validator/blob/master/test/realworld/failed.md#results-of-real-world-testing).

Existing integrations
--------------------------

[![Become a backer](https://opencollective.com/openapi-directory/tiers/backer.svg?avatarHeight=36&width=600)](https://opencollective.com/openapi-directory)

 - [HTTP Toolkit](https://httptoolkit.tech/) **Beautiful & open-source tools to debug, test and develop with HTTP(S)**
 - [OpenAPI-Directory-JS](https://github.com/httptoolkit/openapi-directory-js) - The OpenAPI directory precompiled & indexed for JS usage & searching
 - [Apideck](https://apideck.com) - Faster data integration through Unified APIs
 - [Microsoft Kiota](https://microsoft.github.io/kiota/) - Generate an API client to call any OpenAPI described API
 - [API Tracker](https://apitracker.io/) - Aggregates 14,000+ APIs, SDKs, API specifications, integrations and DX profiles. It aims to help developers access the information they need to integrate APIs faster
 - [API Watch](https://www.apiwatch.io) - Keep track of the APIs you use
 - https://pipedream.com/ - The integration platform built for developers
 - https://any-api.com - Documentation and Test Consoles for Public APIs
 - https://datafire.io - Allows you to process and transfer data between APIs, databases, and more
 - [https://sdks.io](https://sdks.io/Search/FindSDKs?Bridge=APIs.guru) - Explore Automatically Generated SDKs
 - [https://cenit.io](https://cenit.io/directory?spec=swagger) - Data integration platform
 - [commandcar](https://github.com/tikalk/commandcar#installing-from-api-models) - Curl on steroids
 - [Material Swagger UI Hub](https://darosh.github.io/angular-swagger-ui-material/hub/) - testing and exploring hub for Material Swagger UI demo
 - [Paw](https://paw.cloud/) - HTTP client for Mac - You can import definitions and directly play with the APIs, [see how](https://paw.cloud/docs/examples/search-apis)
 - [Bitscoop](https://bitscoop.com/) - A better way to create and maintain integrations
 - [WinSwag](https://github.com/SvenEV/WinSwag) - A UWP app for loading Swagger definitions and exploring REST APIs
 - [ReadMe.io](https://preview.readme.io) - Beautiful, personalized, interactive developer hubs
 - [Velosimo](https://connect.velosimo.io/open_api_directory) - Integration platform for eGovernment
 - [Mockcoon](https://mockoon.com/mock-samples/category/all/) - API mock samples for your project

Also used as test suite in the following projects:
 - [Speakeasy](https://www.speakeasyapi.dev/?utm_source=apigurus+repo&utm_medium=github+sponsorship) - **Generate SDKs Now. Gives your users the DevEx that makes API integrations easy**
 - [ReDoc](https://github.com/Redocly/redoc) - OpenAPI-generated API Reference Documentation
 - [swagger-parser](https://github.com/BigstickCarpet/swagger-parser) - OpenAPI parser and validator for Node and browsers
 - [OpenAPI-schema-validator](https://github.com/seriousme/openapi-schema-validator) - OpenApi schema validation for OpenApi versions v2, v3.0.x and v3.1.x
 - [SwaggerProvider](https://github.com/sergey-tihon/SwaggerProvider) - F# Type Provider for Swagger
 - [ardoq-swagger-addon](https://github.com/ardoq/ardoq-swagger-addon) - Ardoq OpenAPI addon
 - [swagvali](https://github.com/subeeshcbabu/swagvali/) - Module to build validators for OpenAPI Request parameters and Response objects
 - [swagger-search](https://github.com/IG-Group/swagger-search) - An application that collects and indexes swagger docs from your microservices architecture


Integration with 3rd-party services
--------------------------
We discourage you from using Github RAW links or Git directly, the repository structure may change in the future.
Instead, we strongly recommend you to use our [REST API][apiDoc-link].

Licenses
--------------------------
All API definitions contributed to project by authors are covered by the [CC01.0](https://creativecommons.org/publicdomain/zero/1.0/) license.<br>
All API definitions acquired from public sources under the [Fair use](http://en.wikipedia.org/wiki/Fair_use) principle.

Definition sources
--------------------------
Some definitions are taken from Open Source projects:
 - [darklynx/swagger-api-collection](https://github.com/darklynx/swagger-api-collection) - OpenAPI description for Instagram API
 - [Mermade/bbcapis](https://github.com/Mermade/bbcapis) - OpenAPI definitions for the BBC Nitro and iBL APIs
 - [amardeshbd/medium-api-specification](https://github.com/amardeshbd/medium-api-specification) - OpenAPI 2.0 description for Medium API
 - [faragorn/open-api-specs](https://github.com/faragorn/open-api-specs) - OpenAPI definition for the Giphy API

[banner]: https://apis.guru/branding/banner.svg "APIs.guru"
[twitter-image]: https://img.shields.io/twitter/url/http/APIs.guru.svg?style=social
[twitter-link]: https://twitter.com/intent/tweet?text=http%3A%2F%2FAPIs.guru%20-%20Wikipedia%20for%20%23Web%20%23APIs%20by%20@APIs_guru%20pic.twitter.com/UhlhbMw1NP
[twitterFollow-image]: https://img.shields.io/twitter/follow/APIs_guru.svg?style=social
[twitterFollow-link]: https://twitter.com/intent/follow?screen_name=APIs_guru
[numApis-image]: https://api.apis.guru/badges/apis_in_collection.svg
[numSpecs-image]: https://api.apis.guru/badges/openapi_specs.svg
[endpoints-image]: https://api.apis.guru/badges/endpoints.svg
[apisDir-link]: ./APIs
[addAPI-image]: https://cloud.githubusercontent.com/assets/8336157/15861614/7e31511a-2cd5-11e6-8b79-38ad0f61e598.png
[addAPI-link]: https://apis.guru/add-api/
[apiDoc-link]: https://apis.guru/api-doc/

## Contributors

This project exists thanks to all the people who [contribute](CONTRIBUTING.md).
<a href="https://github.com/APIs-guru/openapi-directory/graphs/contributors"><img src="https://opencollective.com/openapi-directory/contributors.svg?width=890&button=false" /></a>

## Asynchronous APIs

See also [AsyncAPI-Directory](https://apis.guru/asyncapi-directory/APIs/)

## Backers

Thank you to all our backers! 🙏 [Become a backer](https://opencollective.com/openapi-directory#backer).

<a href="https://opencollective.com/openapi-directory#backers" target="_blank"><img src="https://opencollective.com/openapi-directory/backers.svg?width=890"></a>


## Sponsors

Support this project by becoming a sponsor. Your logo will show up here with a link to your website. [Become a sponsor](https://opencollective.com/openapi-directory#sponsor).

<a href="https://opencollective.com/openapi-directory/sponsor/0/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/0/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/1/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/1/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/2/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/2/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/3/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/3/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/4/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/4/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/5/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/5/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/6/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/6/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/7/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/7/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/8/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/8/avatar.svg"></a>
<a href="https://opencollective.com/openapi-directory/sponsor/9/website" target="_blank"><img src="https://opencollective.com/openapi-directory/sponsor/9/avatar.svg"></a>

## Partners

Support this project by becoming a partner. Your logo will be displayed prominently in this project and we will work with you to publicise and showcase your projects. [Become a partner](https://opencollective.com/openapi-directory#partner).
 readmeEtag: '"11ced47fb43727ccddd2633ba19da4308ec90b84"' readmeLastModified: Sat, 23 Aug 2025 21:25:15 GMT repositoryId: 31177593 description: >- 🌐 Wikipedia for Web APIs. Directory of REST API definitions in OpenAPI 2.0/3.x format created: '2015-02-22T19:58:12Z' updated: '2026-02-03T15:34:20Z' language: null archived: false stars: 4355 watchers: 111 forks: 656 owner: APIs-guru logo: https://avatars.githubusercontent.com/u/10975548?v=4 license: CC0-1.0 repoEtag: '"96d50b522873474eb3f33a9cd8e4fda4d281c5a61b6a688dee4923ec478a6fbe"' repoLastModified: Tue, 03 Feb 2026 15:34:20 GMT foundInMaster: true category: SDK id: 667f2aef78d9d19418fbd6c0a7835abb - source: openapi3 tags name: springdoc-openapi homepage: https://github.com/springdoc/springdoc-openapi language: Java source_description: >- Library that produces OpenAPI 3.x specification documentation for spring-boot applications. category: - Low-level Tooling - Server Implementations repository: https://github.com/springdoc/springdoc-openapi v3: true repositoryMetadata: base64Readme: >- ![Octocat](https://springdoc.org/img/banner-logo.svg)
[![Build Status](https://ci-cd.springdoc.org:8443/buildStatus/icon?job=springdoc-openapi-starter-IC)](https://ci-cd.springdoc.org:8443/view/springdoc-openapi/job/springdoc-openapi-starter-IC/)
[![Quality Gate](https://sonarcloud.io/api/project_badges/measure?project=springdoc_springdoc-openapi&metric=alert_status)](https://sonarcloud.io/dashboard?id=springdoc_springdoc-openapi)
[![Known Vulnerabilities](https://snyk.io/test/github/springdoc/springdoc-openapi.git/badge.svg)](https://snyk.io/test/github/springdoc/springdoc-openapi.git)
[![Stack Exchange questions](https://img.shields.io/stackexchange/stackoverflow/t/springdoc)](https://stackoverflow.com/questions/tagged/springdoc?tab=Votes)

IMPORTANT: ``springdoc-openapi v1.8.0`` is the latest Open Source release supporting
Spring Boot 2.x and 1.x.

An extended support for [*springdoc-openapi v1*](https://springdoc.org/v1)
project is now available for organizations that need support beyond 2023.

For more details, feel free to reach
out: [sales@springdoc.org](mailto:sales@springdoc.org)

``springdoc-openapi`` is on [Open Collective](https://opencollective.com/springdoc). If
you ❤️ this project consider becoming a [sponsor](https://github.com/sponsors/springdoc).

This project is sponsored by

<p align="center">
<a href="https://opensource.mercedes-benz.com/" target="_blank">
    <img src="https://springdoc.org/img/mercedes-benz.png" height="10%" width="10%" />
</a>
&nbsp;&nbsp;

<a href="https://www.dm-jobs.com/dmTECH/?locale=de_DE&wt_mc=.display.github.sponsoring.logo" target="_blank">
     <img src="https://springdoc.org/img/dmTECH_Logo.jpg" height="10%" width="10%" />
</a>

<a href="https://www.contrastsecurity.com/" target="_blank">
   <img src="https://springdoc.org/img/contrastsecurity.svg" height="10%" width="30%" />
</a>

 <a href="https://www.lvm.de/" target="_blank">
   <img src="https://springdoc.org/img/LVM_Versicherung_2010_logo.svg.png" height="10%" width="25%" />
  </a>
 <a href="https://gdnext.com/" target="_blank">
   <img src="https://springdoc.org/img/gdnext.png" height="10%" width="10%" />
  </a>
</p>

# Table of Contents

- [Full documentation](#full-documentation)
- [**Introduction**](#introduction)
- [**Getting Started**](#getting-started)
    - [Library for springdoc-openapi integration with spring-boot and swagger-ui](#library-for-springdoc-openapi-integration-with-spring-boot-and-swagger-ui)
    - [Spring-boot with OpenAPI Demo applications.](#spring-boot-with-openapi-demo-applications)
        - [Source Code for Demo Applications.](#source-code-for-demo-applications)
        - [Demo Spring Boot 2 Web MVC with OpenAPI 3.](#demo-spring-boot-2-web-mvc-with-openapi-3)
        - [Demo Spring Boot 2 WebFlux with OpenAPI 3.](#demo-spring-boot-2-webflux-with-openapi-3)
        - [Demo Spring Boot 2 WebFlux with Functional endpoints OpenAPI 3.](#demo-spring-boot-2-webflux-with-functional-endpoints-openapi-3)
        - [Demo Spring Boot 2 and Spring Hateoas with OpenAPI 3.](#demo-spring-boot-2-and-spring-hateoas-with-openapi-3)
    - [Integration of the library in a Spring Boot 3.x project without the swagger-ui:](#integration-of-the-library-in-a-spring-boot-3x-project-without-the-swagger-ui)
    - [Error Handling for REST using @ControllerAdvice](#error-handling-for-rest-using-controlleradvice)
    - [Adding API Information and Security documentation](#adding-api-information-and-security-documentation)
    - [spring-webflux support with Annotated Controllers](#spring-webflux-support-with-annotated-controllers)
    - [Using a separate management port (Spring Boot 3)](#using-a-separate-management-port-spring-boot-3)
    - [When Spring Security is enabled](#when-spring-security-is-enabled)
- [Acknowledgements](#acknowledgements)
    - [Contributors](#contributors)
    - [Additional Support](#additional-support)

# [Full documentation](https://springdoc.org/)

# **Introduction**

The springdoc-openapi Java library helps automating the generation of API documentation
using Spring Boot projects.
springdoc-openapi works by examining an application at runtime to infer API semantics
based on Spring configurations, class structure and various annotations.

The library automatically generates documentation in JSON/YAML and HTML formatted pages.
The generated documentation can be complemented using `swagger-api` annotations.

This library supports:

* OpenAPI 3
* Spring-boot v3 (Java 17 & Jakarta EE 9)
* JSR-303, specifically for @NotNull, @Min, @Max, and @Size.
* Swagger-ui
* OAuth 2
* GraalVM native images

The following video introduces the Library:

* [https://youtu.be/utRxyPfFlDw](https://youtu.be/utRxyPfFlDw)

For *spring-boot v3* support, make sure you
use [springdoc-openapi v2](https://springdoc.org/)

This is a community-based project, not maintained by the Spring Framework Contributors (
Pivotal)

# **Getting Started**

## Library for springdoc-openapi integration with spring-boot and swagger-ui

* Automatically deploys swagger-ui to a Spring Boot 3.x application
* Documentation will be available in HTML format, using the
  official [swagger-ui jars](https://github.com/swagger-api/swagger-ui.git).
* The Swagger UI page should then be available at http://server:
  port/context-path/swagger-ui.html and the OpenAPI description will be available at the
  following url for json format: http://server:port/context-path/v3/api-docs
    * `server`: The server name or IP
    * `port`: The server port
    * `context-path`: The context path of the application
* Documentation can be available in yaml format as well, on the following path:
  `/v3/api-docs.yaml`
* Add the `springdoc-openapi-ui` library to the list of your project dependencies (No
  additional configuration is needed):

Maven

```xml
   <dependency>
      <groupId>org.springdoc</groupId>
      <artifactId>springdoc-openapi-starter-webmvc-ui</artifactId>
      <version>last-release-version</version>
   </dependency>
```

Gradle

```groovy
implementation 'org.springdoc:springdoc-openapi-starter-webmvc-ui:latest'
```

* This step is optional: For custom path of the swagger documentation in HTML format, add
  a custom springdoc property, in your spring-boot configuration file:

```properties
# swagger-ui custom path
springdoc.swagger-ui.path=/swagger-ui.html
```

## Spring-boot with OpenAPI Demo applications.

### [Source Code for Demo Applications](https://github.com/springdoc/springdoc-openapi-demos/tree/master).

## [Demo Spring Boot 3 Web MVC with OpenAPI 3](https://demos.springdoc.org/demo-spring-boot-3-webmvc).

## [Demo Spring Boot 3 WebFlux with OpenAPI 3](https://demos.springdoc.org/demo-spring-boot-3-webflux/swagger-ui.html).

## [Demo Spring Boot 3 WebFlux with Functional endpoints OpenAPI 3](https://demos.springdoc.org/demo-spring-boot-3-webflux-functional/swagger-ui.html).

## [Demo Spring Boot 3 and Spring Cloud Function Web MVC](https://demos.springdoc.org/spring-cloud-function-webmvc).

## [Demo Spring Boot 3 and Spring Cloud Function WebFlux](http://158.101.191.70:8085/swagger-ui.html).

## [Demo Spring Boot 3 and Spring Cloud Gateway](https://demos.springdoc.org/demo-microservices/swagger-ui.html).

![Branching](https://springdoc.org/img/pets.png)

## Integration of the library in a Spring Boot 3.x project without the swagger-ui:

* Documentation will be available at the following url for json format: http://server:
  port/context-path/v3/api-docs
    * `server`: The server name or IP
    * `port`: The server port
    * `context-path`: The context path of the application
* Documentation will be available in yaml format as well, on the following
  path : `/v3/api-docs.yaml`
* Add the library to the list of your project dependencies. (No additional configuration
  is needed)

Maven

```xml
   <dependency>
      <groupId>org.springdoc</groupId>
      <artifactId>springdoc-openapi-starter-webmvc-api</artifactId>
      <version>last-release-version</version>
   </dependency>
```

Gradle

```groovy
implementation 'org.springdoc:springdoc-openapi-starter-webmvc-ui:latest'
```

* This step is optional: For custom path of the OpenAPI documentation in Json format, add
  a custom springdoc property, in your spring-boot configuration file:

```properties
# /api-docs endpoint custom path
springdoc.api-docs.path=/api-docs
```

* This step is optional: If you want to disable `springdoc-openapi` endpoints, add a
  custom springdoc property, in your `spring-boot` configuration file:

```properties
# disable api-docs
springdoc.api-docs.enabled=false
```

## Error Handling for REST using @ControllerAdvice

To generate documentation automatically, make sure all the methods declare the HTTP Code
responses using the annotation: @ResponseStatus.

## Adding API Information and Security documentation

The library uses spring-boot application auto-configured packages to scan for the
following annotations in spring beans: OpenAPIDefinition and Info.
These annotations declare, API Information: Title, version, licence, security, servers,
tags, security and externalDocs.
For better performance of documentation generation, declare `@OpenAPIDefinition`
and `@SecurityScheme` annotations within a Spring managed bean.

## spring-webflux support with Annotated Controllers

* Documentation can be available in yaml format as well, on the following path :
  /v3/api-docs.yaml
* Add the library to the list of your project dependencies (No additional configuration
  is needed)

Maven

```xml
   <dependency>
      <groupId>org.springdoc</groupId>
      <artifactId>springdoc-openapi-starter-webflux-ui</artifactId>
      <version>last-release-version</version>
   </dependency>
```

Gradle

```groovy
implementation 'org.springdoc:springdoc-openapi-starter-webmvc-ui:latest'
```

* This step is optional: For custom path of the swagger documentation in HTML format, add
  a custom springdoc property, in your spring-boot configuration file:

```properties
# swagger-ui custom path
springdoc.swagger-ui.path=/swagger-ui.html
```

The `springdoc-openapi` libraries are hosted on maven central repository.
The artifacts can be viewed accessed at the following locations:

Releases:

* [https://central.sonatype.com/search?q=g:org.springdoc)](https://central.sonatype.com/search?q=g:org.springdoc)
  .

Snapshots:

* [https://central.sonatype.com/service/rest/repository/browse/maven-snapshots/org/springdoc/](https://central.sonatype.com/service/rest/repository/browse/maven-snapshots/org/springdoc/)
  .

## Using a separate management port (Spring Boot 3)

Some Spring Boot apps run **Actuator** on a separate management port. In that case:

- **Application port** (e.g., `8080`) serves your app and springdoc endpoints:
  - `http://localhost:8080/v3/api-docs`
  - `http://localhost:8080/swagger-ui/index.html`

- **Management port** (e.g., `9090`) serves Actuator:
  - `http://localhost:9090/actuator`
  - `http://localhost:9090/actuator/health`

Minimal `application.yml`:

```yaml
server:
  port: 8080

management:
  server:
    port: 9090
  endpoints:
    web:
      exposure:
        include: health,info

# springdoc is enabled by default with the starter;
# endpoints remain on the application port.
# (OpenAPI JSON = /v3/api-docs, Swagger UI = /swagger-ui/index.html)
```

### When Spring Security is enabled

With Spring Boot 3, `/v3/api-docs` and Swagger UI are served on the **application port**, while Actuator runs on the **management port**.  
If Spring Security is enabled, explicitly permit the docs paths on the **application port**:

```java
@Bean
SecurityFilterChain api(HttpSecurity http) throws Exception {
  http
    .authorizeHttpRequests(auth -> auth
      .requestMatchers(
        "/v3/api-docs/**",   
        "/v3/api-docs.yaml", 
        "/swagger-ui/**",
        "/swagger-ui.html"
      ).permitAll()
      .anyRequest().authenticated()
    );
  return http.build();
}
```

# Acknowledgements

## Contributors

springdoc-openapi is relevant and updated regularly due to the valuable contributions from
its [contributors](https://github.com/springdoc/springdoc-openapi/graphs/contributors).

<a href="https://github.com/springdoc/springdoc-openapi/graphs/contributors">
  <img src="https://contrib.rocks/image?repo=springdoc/springdoc-openapi" width="50%"/>
</a>

Thanks you all for your support!

## Additional Support

* [Spring Team](https://spring.io/team) - Thanks for their support by sharing all relevant
  resources around Spring projects.
* [JetBrains](https://www.jetbrains.com/?from=springdoc-openapi) - Thanks a lot for
  supporting springdoc-openapi project.

![JenBrains logo](https://resources.jetbrains.com/storage/products/company/brand/logos/jetbrains.svg)
 readmeEtag: '"e3409ef99f573a7fcdf6616eb93cc75ffe26f53b"' readmeLastModified: Sat, 06 Dec 2025 18:27:57 GMT repositoryId: 196475719 description: Library for OpenAPI 3 with spring-boot created: '2019-07-11T23:08:20Z' updated: '2026-02-05T13:04:57Z' language: Java archived: false stars: 3663 watchers: 43 forks: 580 owner: springdoc logo: https://avatars.githubusercontent.com/u/52804136?v=4 license: Apache-2.0 repoEtag: '"faa831b2d868f47e34ffdb6f07516a7f16224f3d7161defad9ec979351fa684e"' repoLastModified: Thu, 05 Feb 2026 13:04:57 GMT foundInMaster: true id: 901410cec129651fcf7374c79f824f27 - source: openapi3 tags repository: https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker v3: true repositoryMetadata: base64Readme: >- ## DEPRECATED 🚨

This Docker image is now deprecated. There's no need to use it, you can just use Uvicorn with `--workers`. ✨

Read more about it below.

---

[![Test](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/actions/workflows/test.yml/badge.svg)](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/actions/workflows/test.yml) [![Deploy](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/workflows/Deploy/badge.svg)](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/actions?query=workflow%3ADeploy)

## Supported tags and respective `Dockerfile` links

* [`python3.11`, `latest` _(Dockerfile)_](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/blob/master/docker-images/python3.11.dockerfile)
* [`python3.10`, _(Dockerfile)_](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/blob/master/docker-images/python3.10.dockerfile)
* [`python3.11-slim` _(Dockerfile)_](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/blob/master/docker-images/python3.11-slim.dockerfile)
* [`python3.10-slim` _(Dockerfile)_](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/blob/master/docker-images/python3.10-slim.dockerfile)

## Deprecated tags

🚨 These tags are no longer supported or maintained, they are removed from the GitHub repository, but the last versions pushed might still be available in Docker Hub if anyone has been pulling them:

* `python3.9`
* `python3.9-slim`
* `python3.8`
* `python3.8-slim`
* `python3.7`
* `python3.9-alpine3.14`
* `python3.8-alpine3.10`
* `python3.7-alpine3.8`
* `python3.6`
* `python3.6-alpine3.8`

The last date tags for these versions are:

* `python3.9-2025-11-09`
* `python3.9-slim-2025-11-09`
* `python3.8-2024-11-02`
* `python3.8-slim-2024-11-02`
* `python3.7-2024-11-02`
* `python3.9-alpine3.14-2024-03-11`
* `python3.8-alpine3.10-2024-01-29`
* `python3.7-alpine3.8-2024-03-11`
* `python3.6-2022-11-25`
* `python3.6-alpine3.8-2022-11-25`

---

**Note**: There are [tags for each build date](https://hub.docker.com/r/tiangolo/uvicorn-gunicorn-fastapi/tags). If you need to "pin" the Docker image version you use, you can select one of those tags. E.g. `tiangolo/uvicorn-gunicorn-fastapi:python3.11-2024-11-02`.

# uvicorn-gunicorn-fastapi

[**Docker**](https://www.docker.com/) image with [**Uvicorn**](https://www.uvicorn.org/) managed by [**Gunicorn**](https://gunicorn.org/) for high-performance [**FastAPI**](https://fastapi.tiangolo.com/) web applications in **[Python](https://www.python.org/)** with performance auto-tuning.

**GitHub repo**: [https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker)

**Docker Hub image**: [https://hub.docker.com/r/tiangolo/uvicorn-gunicorn-fastapi/](https://hub.docker.com/r/tiangolo/uvicorn-gunicorn-fastapi/)

## Description

**FastAPI** has shown to be a Python web framework with [one of the best performances, as measured by third-party benchmarks](https://www.techempower.com/benchmarks/#section=test&runid=a979de55-980d-4721-a46f-77298b3f3923&hw=ph&test=fortune&l=zijzen-7), thanks to being based on and powered by [**Starlette**](https://www.starlette.io/).

The achievable performance is on par with (and in many cases superior to) **Go** and **Node.js** frameworks.

This image has an **auto-tuning** mechanism included to start a number of worker processes based on the available CPU cores. That way you can just add your code and get **high performance** automatically, which is useful in **simple deployments**.

## 🚨 WARNING: You Probably Don't Need this Docker Image

You are probably using **Kubernetes** or similar tools. In that case, you probably **don't need this image** (or any other **similar base image**). You are probably better off **building a Docker image from scratch** as explained in the docs for [FastAPI in Containers - Docker: Build a Docker Image for FastAPI](https://fastapi.tiangolo.com/deployment/docker/#replication-number-of-processes).

### Cluster Replication

If you have a cluster of machines with **Kubernetes**, Docker Swarm Mode, Nomad, or other similar complex system to manage distributed containers on multiple machines, then you will probably want to **handle replication** at the **cluster level** instead of using a **process manager** (like Gunicorn with Uvicorn workers) in each container, which is what this Docker image does.

In those cases (e.g. using Kubernetes) you would probably want to build a **Docker image from scratch**, installing your dependencies, and running **a single Uvicorn process** instead of this image.

For example, your `Dockerfile` could look like:

```Dockerfile
FROM python:3.11

WORKDIR /code

COPY ./requirements.txt /code/requirements.txt

RUN pip install --no-cache-dir --upgrade -r /code/requirements.txt

COPY ./app /code/app

CMD ["uvicorn", "app.main:app", "--host", "0.0.0.0", "--port", "80"]
```

You can read more about this in the [FastAPI documentation about: FastAPI in Containers - Docker](https://fastapi.tiangolo.com/deployment/docker/#replication-number-of-processes).

### Multiple Workers

If you definitely want to have multiple workers on a single container, Uvicorn now supports handling subprocesses, including restarting dead ones. So there's no need for Gunicorn to manage multiple workers in a single container.

You could modify the example `Dockerfile` from above, adding the `--workers` option to Uvicorn, like:

```Dockerfile
FROM python:3.11

WORKDIR /code

COPY ./requirements.txt /code/requirements.txt

RUN pip install --no-cache-dir --upgrade -r /code/requirements.txt

COPY ./app /code/app

CMD ["uvicorn", "app.main:app", "--host", "0.0.0.0", "--port", "80", "--workers", "4"]
```

That's all you need. You don't need this Docker image at all. 😅

You can read more about it in the [FastAPI Docs about Deployment with Docker](https://fastapi.tiangolo.com/deployment/docker/).

## Technical Details

Uvicorn didn't have support for managing worker processing including restarting dead workers. But now it does.

Before that, Gunicorn could be used as a process manager, running Uvicorn workers. This added complexity that is no longer necessary.

## Legacy Docs

The rest of this document is kept for historical reasons, but you probably don't need it. 😅

### `tiangolo/uvicorn-gunicorn-fastapi`

This image will set a sensible configuration based on the server it is running on (the amount of CPU cores available) without making sacrifices.

It has sensible defaults, but you can configure it with environment variables or override the configuration files.

There are also slim versions. If you want one of those, use one of the tags from above.

### `tiangolo/uvicorn-gunicorn`

This image (`tiangolo/uvicorn-gunicorn-fastapi`) is based on [**tiangolo/uvicorn-gunicorn**](https://github.com/tiangolo/uvicorn-gunicorn-docker).

That image is what actually does all the work.

This image just installs FastAPI and has the documentation specifically targeted at FastAPI.

If you feel confident about your knowledge of Uvicorn, Gunicorn and ASGI, you can use that image directly.

### `tiangolo/uvicorn-gunicorn-starlette`

There is a sibling Docker image: [**tiangolo/uvicorn-gunicorn-starlette**](https://github.com/tiangolo/uvicorn-gunicorn-starlette-docker)

If you are creating a new [**Starlette**](https://www.starlette.io/) web application and you want to discard all the additional features from FastAPI you should use [**tiangolo/uvicorn-gunicorn-starlette**](https://github.com/tiangolo/uvicorn-gunicorn-starlette-docker) instead.

**Note**: FastAPI is based on Starlette and adds several features on top of it. Useful for APIs and other cases: data validation, data conversion, documentation with OpenAPI, dependency injection, security/authentication and others.

## How to use

You don't need to clone the GitHub repo.

You can use this image as a base image for other images.

Assuming you have a file `requirements.txt`, you could have a `Dockerfile` like this:

```Dockerfile
FROM tiangolo/uvicorn-gunicorn-fastapi:python3.11

COPY ./requirements.txt /app/requirements.txt

RUN pip install --no-cache-dir --upgrade -r /app/requirements.txt

COPY ./app /app
```

It will expect a file at `/app/app/main.py`.

Or otherwise a file at `/app/main.py`.

And will expect it to contain a variable `app` with your FastAPI application.

Then you can build your image from the directory that has your `Dockerfile`, e.g:

```bash
docker build -t myimage ./
```

## Quick Start

### Build your Image

* Go to your project directory.
* Create a `Dockerfile` with:

```Dockerfile
FROM tiangolo/uvicorn-gunicorn-fastapi:python3.11

COPY ./requirements.txt /app/requirements.txt

RUN pip install --no-cache-dir --upgrade -r /app/requirements.txt

COPY ./app /app
```

* Create an `app` directory and enter in it.
* Create a `main.py` file with:

```Python
from fastapi import FastAPI

app = FastAPI()


@app.get("/")
def read_root():
    return {"Hello": "World"}


@app.get("/items/{item_id}")
def read_item(item_id: int, q: str = None):
    return {"item_id": item_id, "q": q}
```

* You should now have a directory structure like:

```
.
├── app
│   └── main.py
└── Dockerfile
```

* Go to the project directory (in where your `Dockerfile` is, containing your `app` directory).
* Build your FastAPI image:

```bash
docker build -t myimage .
```

* Run a container based on your image:

```bash
docker run -d --name mycontainer -p 80:80 myimage
```

Now you have an optimized FastAPI server in a Docker container. Auto-tuned for your current server (and number of CPU cores).

### Check it

You should be able to check it in your Docker container's URL, for example: <a href="http://192.168.99.100/items/5?q=somequery" target="_blank">http://192.168.99.100/items/5?q=somequery</a> or <a href="http://127.0.0.1/items/5?q=somequery" target="_blank">http://127.0.0.1/items/5?q=somequery</a> (or equivalent, using your Docker host).

You will see something like:

```JSON
{"item_id": 5, "q": "somequery"}
```

### Interactive API docs

Now you can go to <a href="http://192.168.99.100/docs" target="_blank">http://192.168.99.100/docs</a> or <a href="http://127.0.0.1/docs" target="_blank">http://127.0.0.1/docs</a> (or equivalent, using your Docker host).

You will see the automatic interactive API documentation (provided by <a href="https://github.com/swagger-api/swagger-ui" target="_blank">Swagger UI</a>):

![Swagger UI](https://fastapi.tiangolo.com/img/index/index-01-swagger-ui-simple.png)

### Alternative API docs

And you can also go to <a href="http://192.168.99.100/redoc" target="_blank">http://192.168.99.100/redoc</a> or <a href="http://127.0.0.1/redoc" target="_blank">http://127.0.0.1/redoc</a>(or equivalent, using your Docker host).

You will see the alternative automatic documentation (provided by <a href="https://github.com/Rebilly/ReDoc" target="_blank">ReDoc</a>):

![ReDoc](https://fastapi.tiangolo.com/img/index/index-02-redoc-simple.png)

## Dependencies and packages

You will probably also want to add any dependencies for your app and pin them to a specific version, probably including Uvicorn, Gunicorn, and FastAPI.

This way you can make sure your app always works as expected.

You could install packages with `pip` commands in your `Dockerfile`, using a `requirements.txt`, or even using [Poetry](https://python-poetry.org/).

And then you can upgrade those dependencies in a controlled way, running your tests, making sure that everything works, but without breaking your production application if some new version is not compatible.

### Using Poetry

Here's a small example of one of the ways you could install your dependencies making sure you have a pinned version for each package.

Let's say you have a project managed with [Poetry](https://python-poetry.org/), so, you have your package dependencies in a file `pyproject.toml`. And possibly a file `poetry.lock`.

Then you could have a `Dockerfile` using Docker multi-stage building with:

```Dockerfile
FROM python:3.11 as requirements-stage

WORKDIR /tmp

RUN pip install poetry

COPY ./pyproject.toml ./poetry.lock* /tmp/

RUN poetry export -f requirements.txt --output requirements.txt --without-hashes

FROM tiangolo/uvicorn-gunicorn-fastapi:python3.11

COPY --from=requirements-stage /tmp/requirements.txt /app/requirements.txt

RUN pip install --no-cache-dir --upgrade -r /app/requirements.txt

COPY ./app /app
```

That will:

* Install poetry and configure it for running inside of the Docker container.
* Copy your application requirements.
    * Because it uses `./poetry.lock*` (ending with a `*`), it won't crash if that file is not available yet.
* Install the dependencies.
* Then copy your app code.

It's important to copy the app code *after* installing the dependencies, that way you can take advantage of Docker's cache. That way it won't have to install everything from scratch every time you update your application files, only when you add new dependencies.

This also applies for any other way you use to install your dependencies. If you use a `requirements.txt`, copy it alone and install all the dependencies on the top of the `Dockerfile`, and add your app code after it.

## Advanced usage

### Environment variables

These are the environment variables that you can set in the container to configure it and their default values:

#### `MODULE_NAME`

The Python "module" (file) to be imported by Gunicorn, this module would contain the actual application in a variable.

By default:

* `app.main` if there's a file `/app/app/main.py` or
* `main` if there's a file `/app/main.py`

For example, if your main file was at `/app/custom_app/custom_main.py`, you could set it like:

```bash
docker run -d -p 80:80 -e MODULE_NAME="custom_app.custom_main" myimage
```

#### `VARIABLE_NAME`

The variable inside of the Python module that contains the FastAPI application.

By default:

* `app`

For example, if your main Python file has something like:

```Python
from fastapi import FastAPI

api = FastAPI()


@api.get("/")
def read_root():
    return {"Hello": "World"}
```

In this case `api` would be the variable with the FastAPI application. You could set it like:

```bash
docker run -d -p 80:80 -e VARIABLE_NAME="api" myimage
```

#### `APP_MODULE`

The string with the Python module and the variable name passed to Gunicorn.

By default, set based on the variables `MODULE_NAME` and `VARIABLE_NAME`:

* `app.main:app` or
* `main:app`

You can set it like:

```bash
docker run -d -p 80:80 -e APP_MODULE="custom_app.custom_main:api" myimage
```

#### `GUNICORN_CONF`

The path to a Gunicorn Python configuration file.

By default:

* `/app/gunicorn_conf.py` if it exists
* `/app/app/gunicorn_conf.py` if it exists
* `/gunicorn_conf.py` (the included default)

You can set it like:

```bash
docker run -d -p 80:80 -e GUNICORN_CONF="/app/custom_gunicorn_conf.py" myimage
```

You can use the [config file from the base image](https://github.com/tiangolo/uvicorn-gunicorn-docker/blob/master/docker-images/gunicorn_conf.py) as a starting point for yours.

#### `WORKERS_PER_CORE`

This image will check how many CPU cores are available in the current server running your container.

It will set the number of workers to the number of CPU cores multiplied by this value.

By default:

* `1`

You can set it like:

```bash
docker run -d -p 80:80 -e WORKERS_PER_CORE="3" myimage
```

If you used the value `3` in a server with 2 CPU cores, it would run 6 worker processes.

You can use floating point values too.

So, for example, if you have a big server (let's say, with 8 CPU cores) running several applications, and you have a FastAPI application that you know won't need high performance. And you don't want to waste server resources. You could make it use `0.5` workers per CPU core. For example:

```bash
docker run -d -p 80:80 -e WORKERS_PER_CORE="0.5" myimage
```

In a server with 8 CPU cores, this would make it start only 4 worker processes.

**Note**: By default, if `WORKERS_PER_CORE` is `1` and the server has only 1 CPU core, instead of starting 1 single worker, it will start 2. This is to avoid bad performance and blocking applications (server application) on small machines (server machine/cloud/etc). This can be overridden using `WEB_CONCURRENCY`.

#### `MAX_WORKERS`

Set the maximum number of workers to use.

You can use it to let the image compute the number of workers automatically but making sure it's limited to a maximum.

This can be useful, for example, if each worker uses a database connection and your database has a maximum limit of open connections.

By default it's not set, meaning that it's unlimited.

You can set it like:

```bash
docker run -d -p 80:80 -e MAX_WORKERS="24" myimage
```

This would make the image start at most 24 workers, independent of how many CPU cores are available in the server.

#### `WEB_CONCURRENCY`

Override the automatic definition of number of workers.

By default:

* Set to the number of CPU cores in the current server multiplied by the environment variable `WORKERS_PER_CORE`. So, in a server with 2 cores, by default it will be set to `2`.

You can set it like:

```bash
docker run -d -p 80:80 -e WEB_CONCURRENCY="2" myimage
```

This would make the image start 2 worker processes, independent of how many CPU cores are available in the server.

#### `HOST`

The "host" used by Gunicorn, the IP where Gunicorn will listen for requests.

It is the host inside of the container.

So, for example, if you set this variable to `127.0.0.1`, it will only be available inside the container, not in the host running it.

It's is provided for completeness, but you probably shouldn't change it.

By default:

* `0.0.0.0`

#### `PORT`

The port the container should listen on.

If you are running your container in a restrictive environment that forces you to use some specific port (like `8080`) you can set it with this variable.

By default:

* `80`

You can set it like:

```bash
docker run -d -p 80:8080 -e PORT="8080" myimage
```

#### `BIND`

The actual host and port passed to Gunicorn.

By default, set based on the variables `HOST` and `PORT`.

So, if you didn't change anything, it will be set by default to:

* `0.0.0.0:80`

You can set it like:

```bash
docker run -d -p 80:8080 -e BIND="0.0.0.0:8080" myimage
```

#### `LOG_LEVEL`

The log level for Gunicorn.

One of:

* `debug`
* `info`
* `warning`
* `error`
* `critical`

By default, set to `info`.

If you need to squeeze more performance sacrificing logging, set it to `warning`, for example:

You can set it like:

```bash
docker run -d -p 80:8080 -e LOG_LEVEL="warning" myimage
```

#### `WORKER_CLASS`

The class to be used by Gunicorn for the workers.

By default, set to `uvicorn.workers.UvicornWorker`.

The fact that it uses Uvicorn is what allows using ASGI frameworks like FastAPI, and that is also what provides the maximum performance.

You probably shouldn't change it.

But if for some reason you need to use the alternative Uvicorn worker: `uvicorn.workers.UvicornH11Worker` you can set it with this environment variable.

You can set it like:

```bash
docker run -d -p 80:8080 -e WORKER_CLASS="uvicorn.workers.UvicornH11Worker" myimage
```

#### `TIMEOUT`

Workers silent for more than this many seconds are killed and restarted.

Read more about it in the [Gunicorn docs: timeout](https://docs.gunicorn.org/en/stable/settings.html#timeout).

By default, set to `120`.

Notice that Uvicorn and ASGI frameworks like FastAPI are async, not sync. So it's probably safe to have higher timeouts than for sync workers.

You can set it like:

```bash
docker run -d -p 80:8080 -e TIMEOUT="20" myimage
```

#### `KEEP_ALIVE`

The number of seconds to wait for requests on a Keep-Alive connection.

Read more about it in the [Gunicorn docs: keepalive](https://docs.gunicorn.org/en/stable/settings.html#keepalive).

By default, set to `2`.

You can set it like:

```bash
docker run -d -p 80:8080 -e KEEP_ALIVE="20" myimage
```

#### `GRACEFUL_TIMEOUT`

Timeout for graceful workers restart.

Read more about it in the [Gunicorn docs: graceful-timeout](https://docs.gunicorn.org/en/stable/settings.html#graceful-timeout).

By default, set to `120`.

You can set it like:

```bash
docker run -d -p 80:8080 -e GRACEFUL_TIMEOUT="20" myimage
```

#### `ACCESS_LOG`

The access log file to write to.

By default `"-"`, which means stdout (print in the Docker logs).

If you want to disable `ACCESS_LOG`, set it to an empty value.

For example, you could disable it with:

```bash
docker run -d -p 80:8080 -e ACCESS_LOG= myimage
```

#### `ERROR_LOG`

The error log file to write to.

By default `"-"`, which means stderr (print in the Docker logs).

If you want to disable `ERROR_LOG`, set it to an empty value.

For example, you could disable it with:

```bash
docker run -d -p 80:8080 -e ERROR_LOG= myimage
```

#### `GUNICORN_CMD_ARGS`

Any additional command line settings for Gunicorn can be passed in the `GUNICORN_CMD_ARGS` environment variable.

Read more about it in the [Gunicorn docs: Settings](https://docs.gunicorn.org/en/stable/settings.html#settings).

These settings will have precedence over the other environment variables and any Gunicorn config file.

For example, if you have a custom TLS/SSL certificate that you want to use, you could copy them to the Docker image or mount them in the container, and set [`--keyfile` and `--certfile`](http://docs.gunicorn.org/en/latest/settings.html#ssl) to the location of the files, for example:

```bash
docker run -d -p 80:8080 -e GUNICORN_CMD_ARGS="--keyfile=/secrets/key.pem --certfile=/secrets/cert.pem" -e PORT=443 myimage
```

**Note**: instead of handling TLS/SSL yourself and configuring it in the container, it's recommended to use a "TLS Termination Proxy" like [Traefik](https://docs.traefik.io/). You can read more about it in the [FastAPI documentation about HTTPS](https://fastapi.tiangolo.com/deployment/#https).

#### `PRE_START_PATH`

The path where to find the pre-start script.

By default, set to `/app/prestart.sh`.

You can set it like:

```bash
docker run -d -p 80:8080 -e PRE_START_PATH="/custom/script.sh" myimage
```

### Custom Gunicorn configuration file

The image includes a default Gunicorn Python config file at `/gunicorn_conf.py`.

It uses the environment variables declared above to set all the configurations.

You can override it by including a file in:

* `/app/gunicorn_conf.py`
* `/app/app/gunicorn_conf.py`
* `/gunicorn_conf.py`

### Custom `/app/prestart.sh`

If you need to run anything before starting the app, you can add a file `prestart.sh` to the directory `/app`. The image will automatically detect and run it before starting everything.

For example, if you want to add Alembic SQL migrations (with SQLALchemy), you could create a `./app/prestart.sh` file in your code directory (that will be copied by your `Dockerfile`) with:

```bash
#! /usr/bin/env bash

# Let the DB start
sleep 10;
# Run migrations
alembic upgrade head
```

and it would wait 10 seconds to give the database some time to start and then run that `alembic` command.

If you need to run a Python script before starting the app, you could make the `/app/prestart.sh` file run your Python script, with something like:

```bash
#! /usr/bin/env bash

# Run custom Python script before starting
python /app/my_custom_prestart_script.py
```

You can customize the location of the prestart script with the environment variable `PRE_START_PATH` described above.

### Development live reload

The default program that is run is at `/start.sh`. It does everything described above.

There's also a version for development with live auto-reload at:

```bash
/start-reload.sh
```

#### Details

For development, it's useful to be able to mount the contents of the application code inside of the container as a Docker "host volume", to be able to change the code and test it live, without having to build the image every time.

In that case, it's also useful to run the server with live auto-reload, so that it re-starts automatically at every code change.

The additional script `/start-reload.sh` runs Uvicorn alone (without Gunicorn) and in a single process.

It is ideal for development.

#### Usage

For example, instead of running:

```bash
docker run -d -p 80:80 myimage
```

You could run:

```bash
docker run -d -p 80:80 -v $(pwd):/app myimage /start-reload.sh
```

* `-v $(pwd):/app`: means that the directory `$(pwd)` should be mounted as a volume inside of the container at `/app`.
    * `$(pwd)`: runs `pwd` ("print working directory") and puts it as part of the string.
* `/start-reload.sh`: adding something (like `/start-reload.sh`) at the end of the command, replaces the default "command" with this one. In this case, it replaces the default (`/start.sh`) with the development alternative `/start-reload.sh`.

#### Development live reload - Technical Details

As `/start-reload.sh` doesn't run with Gunicorn, any of the configurations you put in a `gunicorn_conf.py` file won't apply.

But these environment variables will work the same as described above:

* `MODULE_NAME`
* `VARIABLE_NAME`
* `APP_MODULE`
* `HOST`
* `PORT`
* `LOG_LEVEL`

## 🚨 Alpine Python Warning

In short: You probably shouldn't use Alpine for Python projects, instead use the `slim` Docker image versions.

---

Do you want more details? Continue reading 👇

Alpine is more useful for other languages where you build a static binary in one Docker image stage (using multi-stage Docker building) and then copy it to a simple Alpine image, and then just execute that binary. For example, using Go.

But for Python, as Alpine doesn't use the standard tooling used for building Python extensions, when installing packages, in many cases Python (`pip`) won't find a precompiled installable package (a "wheel") for Alpine. And after debugging lots of strange errors you will realize that you have to install a lot of extra tooling and build a lot of dependencies just to use some of these common Python packages. 😩

This means that, although the original Alpine image might have been small, you end up with a an image with a size comparable to the size you would have gotten if you had just used a standard Python image (based on Debian), or in some cases even larger. 🤯

And in all those cases, it will take much longer to build, consuming much more resources, building dependencies for longer, and also increasing its carbon footprint, as you are using more CPU time and energy for each build. 🌳

If you want slim Python images, you should instead try and use the `slim` versions that are still based on Debian, but are smaller. 🤓

## Tests

All the image tags, configurations, environment variables and application options are tested.

## Release Notes

### Latest Changes

#### Upgrades

* ⬆ Bump uvicorn[standard] from 0.38.0 to 0.40.0. PR [#415](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/415) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump fastapi[all] from 0.121.0 to 0.128.0. PR [#417](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/417) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump uvicorn[standard] from 0.35.0 to 0.38.0. PR [#390](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/390) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump fastapi[all] from 0.116.0 to 0.121.0. PR [#397](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/397) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump fastapi[all] from 0.115.12 to 0.116.0. PR [#376](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/376) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump uvicorn[standard] from 0.34.2 to 0.35.0. PR [#375](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/375) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump fastapi[all] from 0.115.9 to 0.115.12. PR [#368](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/368) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump uvicorn[standard] from 0.34.0 to 0.34.2. PR [#370](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/370) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump fastapi[all] from 0.115.4 to 0.115.9. PR [#366](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/366) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump uvicorn[standard] from 0.32.0 to 0.34.0. PR [#361](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/361) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump fastapi[all] from 0.88.0 to 0.115.4. PR [#354](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/354) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump uvicorn[standard] from 0.20.0 to 0.32.0. PR [#352](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/352) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 🔥 Drop support for Python 3.7 and 3.8. PR [#355](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/355) by [@tiangolo](https://github.com/tiangolo).
* ⬆ Bump gunicorn from 22.0.0 to 23.0.0. PR [#300](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/300) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump gunicorn from 21.2.0 to 22.0.0. PR [#287](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/287) by [@dependabot[bot]](https://github.com/apps/dependabot).

#### Docs

* 📝 Deprecate this Docker image, use Uvicorn with `--workers` ✨. PR [#303](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/303) by [@tiangolo](https://github.com/tiangolo).
* 📝 Add security policy. PR [#283](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/283) by [@tiangolo](https://github.com/tiangolo).

#### Internal

* 👷 Upgrade actions/checkout from v5 to v6. PR [#406](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/406) by [@tiangolo](https://github.com/tiangolo).
* 👷 Upgrade `latest-changes` GitHub Action and pin `actions/checkout@v5`. PR [#404](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/404) by [@tiangolo](https://github.com/tiangolo).
* 🔥 Drop support for Python 3.9. PR [#398](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/398) by [@tiangolo](https://github.com/tiangolo).
* ⬆ Bump tiangolo/issue-manager from 0.5.1 to 0.6.0. PR [#387](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/387) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump actions/setup-python from 5 to 6. PR [#380](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/380) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump actions/labeler from 5 to 6. PR [#381](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/381) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump actions/checkout from 4 to 5. PR [#379](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/379) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump tiangolo/latest-changes from 0.3.2 to 0.4.0. PR [#378](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/378) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump tiangolo/latest-changes from 0.3.1 to 0.3.2. PR [#357](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/357) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 👷 Update labeler config. PR [#365](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/365) by [@tiangolo](https://github.com/tiangolo).
* 👷 Add CI Labeler. PR [#364](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/364) by [@tiangolo](https://github.com/tiangolo).
* 🔥 Remove old unused files. PR [#356](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/356) by [@tiangolo](https://github.com/tiangolo).
* ⬆ Bump tiangolo/issue-manager from 0.5.0 to 0.5.1. PR [#344](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/344) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 👷 Update `issue-manager.yml`. PR [#343](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/343) by [@tiangolo](https://github.com/tiangolo).
* 👷 Update `latest-changes` GitHub Action. PR [#340](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/340) by [@tiangolo](https://github.com/tiangolo).
* ⬆ Bump docker/build-push-action from 5 to 6. PR [#293](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/293) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 📌 Unpin testing dependencies. PR [#304](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/304) by [@tiangolo](https://github.com/tiangolo).
* ⬆ Bump docker/login-action from 1 to 3. PR [#280](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/280) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump docker/setup-buildx-action from 1 to 3. PR [#279](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/279) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump docker/build-push-action from 2 to 5. PR [#278](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/278) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump actions/setup-python from 4 to 5. PR [#277](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/277) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Update black requirement from ^22.10 to ^23.3. PR [#268](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/268) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 🔧 Add GitHub templates for discussions and templates. PR [#281](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/281) by [@tiangolo](https://github.com/tiangolo).
* 🔧 Update `latest-changes.yml`. PR [#276](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/276) by [@alejsdev](https://github.com/alejsdev).

### 0.8.0

#### Features

* ✨ Add support for multi-arch builds, including support for arm64 (e.g. Mac M1). PR [#273](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/273) by [@tiangolo](https://github.com/tiangolo).

#### Docs

* 📝 Update test badge in `README.md`. PR [#275](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/275) by [@alejsdev](https://github.com/alejsdev).
* 📝 Update test badge in `README.md`. PR [#274](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/274) by [@alejsdev](https://github.com/alejsdev).

#### Upgrades

* ⬆ Bump gunicorn from 20.1.0 to 21.2.0. PR [#270](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/270) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Bump fastapi[all] from 0.87.0 to 0.88.0. PR [#222](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/222) by [@dependabot[bot]](https://github.com/apps/dependabot).

#### Internal

* ⬆ Update mypy requirement from ^0.991 to ^1.4. PR [#269](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/269) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump actions/checkout from 3 to 4. PR [#266](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/266) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump peter-evans/dockerhub-description from 3 to 4. PR [#267](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/267) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump actions/setup-python from 4.3.0 to 5.0.0. PR [#265](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/265) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆ Bump tiangolo/issue-manager from 0.4.0 to 0.5.0. PR [#264](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/264) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 👷 Update dependabot. PR [#253](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/253) by [@tiangolo](https://github.com/tiangolo).
* 👷 Update token for latest changes. PR [#247](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/247) by [@tiangolo](https://github.com/tiangolo).
* 👷 Add GitHub Action for Docker Hub description. PR [#221](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/221) by [@tiangolo](https://github.com/tiangolo).
* ⬆️ Update mypy requirement from ^0.971 to ^0.991. PR [#214](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/214) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update autoflake requirement from ^1.3.1 to ^2.0.0. PR [#215](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/215) by [@dependabot[bot]](https://github.com/apps/dependabot).

### 0.7.0

Highlights of this release:

* Support for Python 3.10 and 3.11.
* Deprecation of Python 3.6.
    * The last Python 3.6 image tag was pushed and is available in Docker Hub, but it won't be updated or maintained anymore.
    * The last image with a date tag is `python3.6-2022-11-25`.
* Upgraded versions of all the dependencies.

#### Features

* ✨ Add support for Python 3.10 and 3.11. PR [#220](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/220) by [@tiangolo](https://github.com/tiangolo).
* ✨ Add Python 3.9 and Python 3.9 Alpine. PR [#67](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/67) by [@graue70](https://github.com/graue70).

#### Breaking Changes

* 🔥 Deprecate and remove Python 3.6. PR [#211](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/211) by [@tiangolo](https://github.com/tiangolo).

#### Upgrades

* ⬆️ Upgrade FastAPI and Uvicorn versions. PR [#212](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/212) by [@tiangolo](https://github.com/tiangolo).
* ⬆️ Upgrade packages to the last version that supports Python 3.6. PR [#207](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/207) by [@tiangolo](https://github.com/tiangolo).

#### Docs

* 📝 Add note to discourage Alpine with Python. PR [#122](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/122) by [@tiangolo](https://github.com/tiangolo).
* 📝 Add warning for Kubernetes, when to use this image. PR [#121](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/121) by [@tiangolo](https://github.com/tiangolo).
* ✏ Fix typo, repeated word on README. PR [#96](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/96) by [@shelbylsmith](https://github.com/shelbylsmith).

#### Internal

* ⬆️ Update black requirement from ^20.8b1 to ^22.10. PR [#216](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/216) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update docker requirement from ^5.0.3 to ^6.0.1. PR [#217](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/217) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 🔥 Remove old Travis file. PR [#219](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/219) by [@tiangolo](https://github.com/tiangolo).
* ⬆️ Upgrade CI OS. PR [#218](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/218) by [@tiangolo](https://github.com/tiangolo).
* 🔧 Update Dependabot config. PR [#213](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/213) by [@tiangolo](https://github.com/tiangolo).
* 👷 Add scheduled CI. PR [#210](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/210) by [@tiangolo](https://github.com/tiangolo).
* 👷 Add alls-green GitHub Action. PR [#209](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/209) by [@tiangolo](https://github.com/tiangolo).
* 👷 Do not run double CI, run on push only on master. PR [#208](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/208) by [@tiangolo](https://github.com/tiangolo).
* ⬆️ Bump actions/setup-python from 4.1.0 to 4.3.0. PR [#201](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/201) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update black requirement from ^19.10b0 to ^20.8b1. PR [#113](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/113) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update docker requirement from ^4.2.0 to ^5.0.3. PR [#125](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/125) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Bump actions/checkout from 2 to 3.1.0. PR [#194](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/194) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update mypy requirement from ^0.770 to ^0.971. PR [#184](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/184) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update isort requirement from ^4.3.21 to ^5.8.0. PR [#116](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/116) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Bump tiangolo/issue-manager from 0.2.0 to 0.4.0. PR [#110](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/110) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Bump actions/setup-python from 1 to 4.1.0. PR [#182](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/182) by [@dependabot[bot]](https://github.com/apps/dependabot).
* ⬆️ Update pytest requirement from ^5.4.1 to ^7.0.1. PR [#153](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/153) by [@dependabot[bot]](https://github.com/apps/dependabot).
* 📌 Add external dependencies and Dependabot to get automatic upgrade PRs. PR [#109](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/109) by [@tiangolo](https://github.com/tiangolo).
* 👷 Update Latest Changes. PR [#108](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/108) by [@tiangolo](https://github.com/tiangolo).
* 👷 Allow GitHub workflow dispatch to trigger test and deploy. PR [#93](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/93) by [@tiangolo](https://github.com/tiangolo).
* 👷 Add latest-changes GitHub action, update issue-manager, add funding. PR [#70](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/70) by [@tiangolo](https://github.com/tiangolo).

### 0.6.0

* Add docs about installing and pinning dependencies. PR [#41](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/41).
* Add `slim` version. PR [#40](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/40).
* Update and refactor bringing all the new features from the base image. Includes:
    * Centralize, simplify, and deduplicate code and setup
    * Move CI to GitHub actions
    * Add Python 3.8 (and Alpine)
    * Add new configs and docs:
        * `WORKER_CLASS`
        * `TIMEOUT`
        * `KEEP_ALIVE`
        * `GRACEFUL_TIMEOUT`
        * `ACCESS_LOG`
        * `ERROR_LOG`
        * `GUNICORN_CMD_ARGS`
        * `MAX_WORKERS`
    * PR [#39](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/39).
* Disable pip cache during installation. PR [#38](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/38).
* Migrate local development from Pipenv to Poetry. PR [#34](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/34).
* Add docs for custom `PRE_START_PATH` env var. PR [#33](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/33).

### 0.5.0

* Refactor tests to use env vars and add image tags for each build date, like `tiangolo/uvicorn-gunicorn-fastapi:python3.7-2019-10-15`. PR [#17](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/17).
* Upgrade Travis. PR [#9](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/9).

### 0.4.0

* Add support for live auto-reload with an additional custom script `/start-reload.sh`, check the [updated documentation](https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker#development-live-reload). PR <a href="https://github.com/tiangolo/uvicorn-gunicorn-docker/pull/6" target="_blank">#6</a> in parent image.

### 0.3.0

* Set `WORKERS_PER_CORE` by default to `1`, as it shows to have the best performance on benchmarks.
* Make the default web concurrency, when `WEB_CONCURRENCY` is not set, to a minimum of 2 workers. This is to avoid bad performance and blocking applications (server application) on small machines (server machine/cloud/etc). This can be overridden using `WEB_CONCURRENCY`. This applies for example in the case where `WORKERS_PER_CORE` is set to `1` (the default) and the server has only 1 CPU core. PR <a href="https://github.com/tiangolo/uvicorn-gunicorn-fastapi-docker/pull/6" target="_blank">#6</a> and PR <a href="https://github.com/tiangolo/uvicorn-gunicorn-docker/pull/5" target="_blank">#5</a> in parent image.

### 0.2.0

* Make `/start.sh` run independently, reading and generating used default environment variables. And remove `/entrypoint.sh` as it doesn't modify anything in the system, only reads environment variables. PR <a href="https://github.com/tiangolo/uvicorn-gunicorn-docker/pull/4" target="_blank">#4</a> in parent image.

### 0.1.0

* Add support for `/app/prestart.sh`.

## License

This project is licensed under the terms of the MIT license.
 readmeEtag: '"d46e9484e3d36e8b5cb751c74c0b79b59f90e092"' readmeLastModified: Wed, 31 Dec 2025 09:04:42 GMT repositoryId: 164136678 description: >- Docker image with Uvicorn managed by Gunicorn for high-performance FastAPI web applications in Python with performance auto-tuning. created: '2019-01-04T17:33:29Z' updated: '2026-02-04T15:12:54Z' language: Python archived: false stars: 2909 watchers: 21 forks: 342 owner: tiangolo logo: https://avatars.githubusercontent.com/u/1326112?v=4 license: MIT repoEtag: '"6310c527e6a40f33d4e95c7109cd3a80ba51d61491316bef499b1c624ed0992d"' repoLastModified: Wed, 04 Feb 2026 15:12:54 GMT foundInMaster: true category: Server Implementations id: d0f3fc0bd35fcadacc8ed2c531780739 - source: - openapi3 tags - openapi31 tags repository: https://github.com/swagger-api/swagger-js v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIENsaWVudCA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3N3YWdnZXItYXBpL3N3YWdnZXIuaW8vd29yZHByZXNzL2ltYWdlcy9hc3NldHMvU1ctbG9nby1jbHIucG5nIiBoZWlnaHQ9IjUwIiBhbGlnbj0icmlnaHQiPgoKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1qcy9hY3Rpb25zL3dvcmtmbG93cy9ub2RlanMueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWpzL2FjdGlvbnMpCgoqKlN3YWdnZXIgQ2xpZW50KiogaXMgYSBKYXZhU2NyaXB0IG1vZHVsZSB0aGF0IGFsbG93cyB5b3UgdG8gZmV0Y2gsIHJlc29sdmUsIGFuZCBpbnRlcmFjdCB3aXRoIFN3YWdnZXIvT3BlbkFQSSBkb2N1bWVudHMuCgojIyBOZXchCgoqKlRoaXMgaXMgdGhlIG5ldyB2ZXJzaW9uIG9mIHN3YWdnZXItanMsIDMueC4qKiBUaGUgbmV3IHZlcnNpb24gc3VwcG9ydHMgU3dhZ2dlciAyLjAgYXMgd2VsbCBhcyBPcGVuQVBJIDMuCgpXYW50IHRvIGxlYXJuIG1vcmU/IENoZWNrIG91dCBvdXIgW0ZBUV0oZG9jcy9taWdyYXRpb24vbWlncmF0aW9uLTIteC10by0zLXgubWQpLgoKRm9yIGZlYXR1cmVzIGtub3duIHRvIGJlIG1pc3NpbmcgZnJvbSAzLnggcGxlYXNlIHNlZSB0aGUgW0dyYXZleWFyZF0oZG9jcy9taWdyYXRpb24vZ3JhdmV5YXJkLTMteC5tZCkuCgoKRm9yIHRoZSBvbGRlciB2ZXJzaW9uIG9mIHN3YWdnZXItanMsIHJlZmVyIHRvIHRoZSBbKjIueCBicmFuY2gqXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1qcy90cmVlLzIueCkuCgo+ICpUaGUgbnBtIHBhY2thZ2UgaXMgY2FsbGVkIGBzd2FnZ2VyLWNsaWVudGAgYW5kIHRoZSBHaXRIdWIgcmVwb3NpdG9yeSBpcyBgc3dhZ2dlci1qc2AuCldlJ2xsIGJlIGNvbnNvbGlkYXRpbmcgdGhhdCBzb29uLiBKdXN0IGdpdmluZyB5b3UgdGhlIGhlYWRzLXVwLiBZb3UgbWF5IHNlZSByZWZlcmVuY2VzIHRvIGJvdGggbmFtZXMuKgoKIyMgQ29tcGF0aWJpbGl0eQpUaGUgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIGhhcyB1bmRlcmdvbmUgbXVsdGlwbGUgcmV2aXNpb25zIHNpbmNlIGluaXRpYWwgY3JlYXRpb24gaW4gMjAxMC4gCkNvbXBhdGliaWxpdHkgYmV0d2VlbiBTd2FnZ2VyIENsaWVudCBhbmQgdGhlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBpcyBhcyBmb2xsb3dzOgoKU3dhZ2dlciBDbGllbnQgVmVyc2lvbiB8IFJlbGVhc2UgRGF0ZSB8IE9wZW5BUEkgU3BlYyBjb21wYXRpYmlsaXR5ICAgICAgICAgICAgICAgICAgICB8IE5vdGVzCi0tLS0tLS0tLS0tLS0tLS0tLSB8LS0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18IC0tLS0tCjMuMzMueCB8IDIwMjQtMTItMzAgICB8IDIuMCwgMy4wLjAsIDMuMC4xLCAzLjAuMiwgMy4wLjMsIDMuMC40LCAzLjEuMCB8IFt0YWcgdjMuMzMuMF0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItanMvcmVsZWFzZXMvdGFnL3YzLjMzLjApCjMuMTkueCB8IDIwMjMtMDEtMjMgICB8IDIuMCwgMy4wLjAsIDMuMC4xLCAzLjAuMiwgMy4wLjMsIDMuMS4wICAgICAgICB8IFt0YWcgdjMuMTkuMC1hbHBoYS4zXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1qcy9yZWxlYXNlcy90YWcvdjMuMTkuMC1hbHBoYS4zKQozLjEwLnggfCAyMDIwLTAxLTE3ICAgfCAyLjAsIDMuMC4wLCAzLjAuMSwgMy4wLjIsIDMuMC4zICAgICAgICAgICAgICAgfCBbdGFnIHYzLjEwLjBdKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWpzL3RyZWUvdjMuMTAuMCkKMi4xLjMyIHwgMjAxNy0wMS0xMiAgIHwgMS4wLCAxLjEsIDEuMiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgW3RhZyB2Mi4xLjMyXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1qcy90cmVlL3YyLjEuMzIpLiBUaGlzIFtyZWxlYXNlXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1qcy9yZWxlYXNlcy90YWcvdjIuMS4zMikgaXMgb25seSBhdmFpbGFibGUgb24gR2l0SHViLgoKIyMgQW5vbnltaXplZCBhbmFseXRpY3MKClN3YWdnZXIgQ2xpZW50IHVzZXMgW1NjYXJmXShodHRwczovL3NjYXJmLnNoLykgdG8gY29sbGVjdCBbYW5vbnltaXplZCBpbnN0YWxsYXRpb24gYW5hbHl0aWNzXShodHRwczovL2dpdGh1Yi5jb20vc2NhcmYtc2gvc2NhcmYtanM/dGFiPXJlYWRtZS1vdi1maWxlI2FzLWEtdXNlci1vZi1hLXBhY2thZ2UtdXNpbmctc2NhcmYtanMtd2hhdC1pbmZvcm1hdGlvbi1kb2VzLXNjYXJmLWpzLXNlbmQtYWJvdXQtbWUpLiBUaGVzZSBhbmFseXRpY3MgaGVscCBzdXBwb3J0IHRoZSBtYWludGFpbmVycyBvZiB0aGlzIGxpYnJhcnkgYW5kIE9OTFkgcnVuIGR1cmluZyBpbnN0YWxsYXRpb24uIFRvIFtvcHQgb3V0XShodHRwczovL2dpdGh1Yi5jb20vc2NhcmYtc2gvc2NhcmYtanM/dGFiPXJlYWRtZS1vdi1maWxlI2FzLWEtdXNlci1vZi1hLXBhY2thZ2UtdXNpbmctc2NhcmYtanMtaG93LWNhbi1pLW9wdC1vdXQtb2YtYW5hbHl0aWNzKSwgeW91IGNhbiBzZXQgdGhlIGBzY2FyZlNldHRpbmdzLmVuYWJsZWRgIGZpZWxkIHRvIGBmYWxzZWAgaW4geW91ciBwcm9qZWN0J3MgYHBhY2thZ2UuanNvbmA6CgpgYGAKLy8gcGFja2FnZS5qc29uCnsKICAvLyAuLi4KICAic2NhcmZTZXR0aW5ncyI6IHsKICAgICJlbmFibGVkIjogZmFsc2UKICB9CiAgLy8gLi4uCn0KYGBgCgpBbHRlcm5hdGl2ZWx5LCB5b3UgY2FuIHNldCB0aGUgZW52aXJvbm1lbnQgdmFyaWFibGUgYFNDQVJGX0FOQUxZVElDU2AgdG8gYGZhbHNlYCBhcyBwYXJ0IG9mIHRoZSBlbnZpcm9ubWVudCB0aGF0IGluc3RhbGxzIHlvdXIgbnBtIHBhY2thZ2VzLCBlLmcuLCBgU0NBUkZfQU5BTFlUSUNTPWZhbHNlIG5wbSBpbnN0YWxsYC4KCiMjIERvY3VtZW50YXRpb24KCiMjIyMgVXNhZ2UKCi0gW0luc3RhbGxhdGlvbl0oZG9jcy91c2FnZS9pbnN0YWxsYXRpb24ubWQpCi0gW1RhZ3MgSW50ZXJmYWNlXShkb2NzL3VzYWdlL3RhZ3MtaW50ZXJmYWNlLm1kKQotIFtIVFRQIGNsaWVudCBmb3IgT0FTIG9wZXJhdGlvbnNdKGRvY3MvdXNhZ2UvaHR0cC1jbGllbnQtZm9yLW9hcy1vcGVyYXRpb25zLm1kKQotIFtPcGVuQVBJIERlZmluaXRpb24gUmVzb2x2ZXJdKGRvY3MvdXNhZ2Uvb3BlbmFwaS1kZWZpbml0aW9uLXJlc29sdmVyLm1kKQotIFtIVFRQIENsaWVudF0oZG9jcy91c2FnZS9odHRwLWNsaWVudC5tZCkKLSBbU3dhZ2dlciBDbGllbnQgQVBJXShkb2NzL3VzYWdlL2FwaS5tZCkKCiMjIyMgRGV2ZWxvcG1lbnQKCi0gW0NvbnRyaWJ1dGluZ10oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpLy5naXRodWIvYmxvYi9tYXN0ZXIvQ09OVFJJQlVUSU5HLm1kKQotIFtTZXR0aW5nIHVwXShkb2NzL2RldmVsb3BtZW50L3NldHRpbmctdXAubWQpCi0gW1NjcmlwdHNdKGRvY3MvZGV2ZWxvcG1lbnQvc2NyaXB0cy5tZCkKCiMjIyMgTWlncmF0aW9ucyAKCi0gW01pZ3JhdGlvbiBndWlkZV0oZG9jcy9taWdyYXRpb24vbWlncmF0aW9uLTIteC10by0zLXgubWQpCi0gW0dyYXZleWFyZF0oZG9jcy9taWdyYXRpb24vZ3JhdmV5YXJkLTMteC5tZCkKCiMjIyBSdW50aW1lIAoKIyMjIE5vZGUuanMKCmBzd2FnZ2VyLWNsaWVudGAgcmVxdWlyZXMgTm9kZS5qcyBgPj0xMi4yMC4wYCBhbmQgdXNlcyBkaWZmZXJlbnQgYGZldGNoYCBpbXBsZW1lbnRhdGlvbiBkZXBlbmRpbmcKb24gTm9kZS5qcyB2ZXJzaW9uLgoKLSBgPj0xMi4yMC4wIDwxOGAgLSBbbm9kZS1mZXRjaEAzXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9ub2RlLWZldGNoKSAKLSBgPj0xOGAgLSBbbmF0aXZlIE5vZGUuanMgZmV0Y2hdKGh0dHBzOi8vbm9kZWpzLm9yZy9kaXN0L2xhdGVzdC12MTgueC9kb2NzL2FwaS9nbG9iYWxzLmh0bWwjZmV0Y2gpCgo+IE5PVEU6IHN3YWdnZXItY2xpZW50IG1pbmltdW0gTm9kZS5qcyBydW50aW1lIHZlcnNpb24gYWxpZ25zIHdpdGggW05vZGUuanMgUmVsZWFzZXNdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi9hYm91dC9yZWxlYXNlcy8pCj4gd2hpY2ggbWVhbnMgdGhhdCB3ZSBjYW4gZHJvcCBzdXBwb3J0IGZvciAqKkVPTCoqIChFbmQgT2YgTGlmZSkgTm9kZS5qcyB2ZXJzaW9ucyB3aXRob3V0IGRvaW5nIG1ham9yIHZlcnNpb24gYnVtcC4KCiMjIyBCcm93c2VycwoKYHN3YWdnZXItY2xpZW50YCB3b3JrcyBpbiB0aGUgbGF0ZXN0IHZlcnNpb25zIG9mIENocm9tZSwgU2FmYXJpLCBGaXJlZm94LCBhbmQgRWRnZQphbmQgdXNlcyBbbmF0aXZlIGZldGNoXShodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9BUEkvRmV0Y2hfQVBJKSBpbXBsZW1lbnRhdGlvbgpwcm92aWRlZCBieSBlYWNoIHN1cHBvcnRlZCBicm93c2VyLgoKIyMgU2VjdXJpdHkgY29udGFjdAoKUGxlYXNlIGRpc2Nsb3NlIGFueSBzZWN1cml0eS1yZWxhdGVkIGlzc3VlcyBvciB2dWxuZXJhYmlsaXRpZXMgYnkgZW1haWxpbmcgW3NlY3VyaXR5QHN3YWdnZXIuaW9dKG1haWx0bzpzZWN1cml0eUBzd2FnZ2VyLmlvKSwgaW5zdGVhZCBvZiB1c2luZyB0aGUgcHVibGljIGlzc3VlIHRyYWNrZXIuCg== readmeEtag: '"4a7bff9bc920afcf75a7511133b16c659e30feb3"' readmeLastModified: Tue, 31 Dec 2024 00:00:27 GMT repositoryId: 3455070 description: >- Javascript library to connect to swagger-enabled APIs via browser or nodejs created: '2012-02-16T00:08:43Z' updated: '2026-01-29T14:41:53Z' language: JavaScript archived: false stars: 2692 watchers: 84 forks: 760 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"c272d6077a6aa769ae9532e2603037933c92feb7982f38a92b7a7cdf20f45419"' repoLastModified: Thu, 29 Jan 2026 14:41:53 GMT foundInMaster: true category: Code Generators id: 5f98b79464c0723b688cff46ee3500c6 v3_1: true - source: openapi3 tags repository: https://github.com/jcrist/msgspec v3: true id: a6f4189b9c0c83612075c0752859ff29 repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9qY3Jpc3RoYXJpZi5jb20vbXNnc3BlYy8iPgogICAgPGltZyBzcmM9Imh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9qY3Jpc3QvbXNnc3BlYy9tYWluL2RvY3MvX3N0YXRpYy9tc2dzcGVjLWxvZ28tbGlnaHQuc3ZnIiB3aWR0aD0iMzUlIiBhbHQ9Im1zZ3NwZWMiPgogIDwvYT4KPC9wPgoKPGRpdiBhbGlnbj0iY2VudGVyIj4KClshW0NJXShodHRwczovL2dpdGh1Yi5jb20vamNyaXN0L21zZ3NwZWMvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9qY3Jpc3QvbXNnc3BlYy9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWwpClshW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZG9jcy1sYXRlc3QtYmx1ZS5zdmcpXShodHRwczovL2pjcmlzdGhhcmlmLmNvbS9tc2dzcGVjLykKWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9qY3Jpc3QvbXNnc3BlYy5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vamNyaXN0L21zZ3NwZWMvYmxvYi9tYWluL0xJQ0VOU0UpClshW1B5UEkgVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3YvbXNnc3BlYy5zdmcpXShodHRwczovL3B5cGkub3JnL3Byb2plY3QvbXNnc3BlYy8pClshW0NvbmRhIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY29uZGEvdm4vY29uZGEtZm9yZ2UvbXNnc3BlYy5zdmcpXShodHRwczovL2FuYWNvbmRhLm9yZy9jb25kYS1mb3JnZS9tc2dzcGVjKQpbIVtDb2RlIENvdmVyYWdlXShodHRwczovL2NvZGVjb3YuaW8vZ2gvamNyaXN0L21zZ3NwZWMvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9hcHAuY29kZWNvdi5pby9naC9qY3Jpc3QvbXNnc3BlYykKCjwvZGl2PgoKYG1zZ3NwZWNgIGlzIGEgKmZhc3QqIHNlcmlhbGl6YXRpb24gYW5kIHZhbGlkYXRpb24gbGlicmFyeSwgd2l0aCBidWlsdGluCnN1cHBvcnQgZm9yIFtKU09OXShodHRwczovL2pzb24ub3JnKSwgW01lc3NhZ2VQYWNrXShodHRwczovL21zZ3BhY2sub3JnKSwKW1lBTUxdKGh0dHBzOi8veWFtbC5vcmcpLCBhbmQgW1RPTUxdKGh0dHBzOi8vdG9tbC5pby9lbi8pLiBJdCBmZWF0dXJlczoKCi0g8J+agCAqKkhpZ2ggcGVyZm9ybWFuY2UgZW5jb2RlcnMvZGVjb2RlcnMqKiBmb3IgY29tbW9uIHByb3RvY29scy4gVGhlIEpTT04gYW5kCiAgTWVzc2FnZVBhY2sgaW1wbGVtZW50YXRpb25zIHJlZ3VsYXJseQogIFtiZW5jaG1hcmtdKGh0dHBzOi8vamNyaXN0aGFyaWYuY29tL21zZ3NwZWMvYmVuY2htYXJrcy5odG1sKSBhcyB0aGUgZmFzdGVzdAogIG9wdGlvbnMgZm9yIFB5dGhvbi4KCi0g8J+OiSAqKlN1cHBvcnQgZm9yIGEgd2lkZSB2YXJpZXR5IG9mIFB5dGhvbiB0eXBlcyoqLiBBZGRpdGlvbmFsIHR5cGVzIG1heSBiZQogIHN1cHBvcnRlZCB0aHJvdWdoCiAgW2V4dGVuc2lvbnNdKGh0dHBzOi8vamNyaXN0aGFyaWYuY29tL21zZ3NwZWMvZXh0ZW5kaW5nLmh0bWwpLgoKLSDwn5SNICoqWmVyby1jb3N0IHNjaGVtYSB2YWxpZGF0aW9uKiogdXNpbmcgZmFtaWxpYXIgUHl0aG9uIHR5cGUgYW5ub3RhdGlvbnMuIEluCiAgW2JlbmNobWFya3NdKGh0dHBzOi8vamNyaXN0aGFyaWYuY29tL21zZ3NwZWMvYmVuY2htYXJrcy5odG1sKSBgbXNnc3BlY2AKICBkZWNvZGVzICphbmQqIHZhbGlkYXRlcyBKU09OIGZhc3RlciB0aGFuCiAgW29yanNvbl0oaHR0cHM6Ly9naXRodWIuY29tL2lqbC9vcmpzb24pIGNhbiBkZWNvZGUgaXQgYWxvbmUuCgotIOKcqCAqKkEgc3BlZWR5IFN0cnVjdCB0eXBlKiogZm9yIHJlcHJlc2VudGluZyBzdHJ1Y3R1cmVkIGRhdGEuIElmIHlvdSBhbHJlYWR5CiAgdXNlIFtkYXRhY2xhc3Nlc10oaHR0cHM6Ly9kb2NzLnB5dGhvbi5vcmcvMy9saWJyYXJ5L2RhdGFjbGFzc2VzLmh0bWwpIG9yCiAgW2F0dHJzXShodHRwczovL3d3dy5hdHRycy5vcmcvZW4vc3RhYmxlLyksCiAgW3N0cnVjdHNdKGh0dHBzOi8vamNyaXN0aGFyaWYuY29tL21zZ3NwZWMvc3RydWN0cy5odG1sKSBzaG91bGQgZmVlbCBmYW1pbGlhci4KICBIb3dldmVyLCB0aGV5J3JlCiAgWzUtNjB4IGZhc3Rlcl0oaHR0cHM6Ly9qY3Jpc3RoYXJpZi5jb20vbXNnc3BlYy9iZW5jaG1hcmtzLmh0bWwjc3RydWN0cykKICBmb3IgY29tbW9uIG9wZXJhdGlvbnMuCgpBbGwgb2YgdGhpcyBpcyBpbmNsdWRlZCBpbiBhCltsaWdodHdlaWdodCBsaWJyYXJ5XShodHRwczovL2pjcmlzdGhhcmlmLmNvbS9tc2dzcGVjL2JlbmNobWFya3MuaHRtbCNsaWJyYXJ5LXNpemUpCndpdGggbm8gcmVxdWlyZWQgZGVwZW5kZW5jaWVzLgoKLS0tCgpgbXNnc3BlY2AgbWF5IGJlIHVzZWQgZm9yIHNlcmlhbGl6YXRpb24gYWxvbmUsIGFzIGEgZmFzdGVyIEpTT04gb3IKTWVzc2FnZVBhY2sgbGlicmFyeS4gRm9yIHRoZSBncmVhdGVzdCBiZW5lZml0IHRob3VnaCwgd2UgcmVjb21tZW5kIHVzaW5nCmBtc2dzcGVjYCB0byBoYW5kbGUgdGhlIGZ1bGwgc2VyaWFsaXphdGlvbiAmIHZhbGlkYXRpb24gd29ya2Zsb3c6CgoqKkRlZmluZSoqIHlvdXIgbWVzc2FnZSBzY2hlbWFzIHVzaW5nIHN0YW5kYXJkIFB5dGhvbiB0eXBlIGFubm90YXRpb25zLgoKYGBgcHl0aG9uCj4+PiBpbXBvcnQgbXNnc3BlYwoKPj4+IGNsYXNzIFVzZXIobXNnc3BlYy5TdHJ1Y3QpOgouLi4gICAgICIiIkEgbmV3IHR5cGUgZGVzY3JpYmluZyBhIFVzZXIiIiIKLi4uICAgICBuYW1lOiBzdHIKLi4uICAgICBncm91cHM6IHNldFtzdHJdID0gc2V0KCkKLi4uICAgICBlbWFpbDogc3RyIHwgTm9uZSA9IE5vbmUKYGBgCgoqKkVuY29kZSoqIG1lc3NhZ2VzIGFzIEpTT04sIG9yIG9uZSBvZiB0aGUgbWFueSBvdGhlciBzdXBwb3J0ZWQgcHJvdG9jb2xzLgoKYGBgcHl0aG9uCj4+PiBhbGljZSA9IFVzZXIoImFsaWNlIiwgZ3JvdXBzPXsiYWRtaW4iLCAiZW5naW5lZXJpbmcifSkKCj4+PiBhbGljZQpVc2VyKG5hbWU9J2FsaWNlJywgZ3JvdXBzPXsiYWRtaW4iLCAiZW5naW5lZXJpbmcifSwgZW1haWw9Tm9uZSkKCj4+PiBtc2cgPSBtc2dzcGVjLmpzb24uZW5jb2RlKGFsaWNlKQoKPj4+IG1zZwpiJ3sibmFtZSI6ImFsaWNlIiwiZ3JvdXBzIjpbImFkbWluIiwiZW5naW5lZXJpbmciXSwiZW1haWwiOm51bGx9JwpgYGAKCioqRGVjb2RlKiogbWVzc2FnZXMgYmFjayBpbnRvIFB5dGhvbiBvYmplY3RzLCB3aXRoIG9wdGlvbmFsIHNjaGVtYSB2YWxpZGF0aW9uLgoKYGBgcHl0aG9uCj4+PiBtc2dzcGVjLmpzb24uZGVjb2RlKG1zZywgdHlwZT1Vc2VyKQpVc2VyKG5hbWU9J2FsaWNlJywgZ3JvdXBzPXsiYWRtaW4iLCAiZW5naW5lZXJpbmcifSwgZW1haWw9Tm9uZSkKCj4+PiBtc2dzcGVjLmpzb24uZGVjb2RlKGIneyJuYW1lIjoiYm9iIiwiZ3JvdXBzIjpbMTIzXX0nLCB0eXBlPVVzZXIpClRyYWNlYmFjayAobW9zdCByZWNlbnQgY2FsbCBsYXN0KToKICBGaWxlICI8c3RkaW4+IiwgbGluZSAxLCBpbiA8bW9kdWxlPgptc2dzcGVjLlZhbGlkYXRpb25FcnJvcjogRXhwZWN0ZWQgYHN0cmAsIGdvdCBgaW50YCAtIGF0IGAkLmdyb3Vwc1swXWAKYGBgCgpgbXNnc3BlY2AgaXMgZGVzaWduZWQgdG8gYmUgYXMgcGVyZm9ybWFudCBhcyBwb3NzaWJsZSwgd2hpbGUgcmV0YWluaW5nIHNvbWUgb2YKdGhlIG5pY2l0aWVzIG9mIHZhbGlkYXRpb24gbGlicmFyaWVzIGxpa2UKW3B5ZGFudGljXShodHRwczovL2RvY3MucHlkYW50aWMuZGV2L2xhdGVzdC8pLiBGb3Igc3VwcG9ydGVkIHR5cGVzLAplbmNvZGluZy9kZWNvZGluZyBhIG1lc3NhZ2Ugd2l0aCBgbXNnc3BlY2AgY2FuIGJlClt+MTAtODB4IGZhc3RlciB0aGFuIGFsdGVybmF0aXZlIGxpYnJhcmllc10oaHR0cHM6Ly9qY3Jpc3RoYXJpZi5jb20vbXNnc3BlYy9iZW5jaG1hcmtzLmh0bWwpLgoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9qY3Jpc3RoYXJpZi5jb20vbXNnc3BlYy9iZW5jaG1hcmtzLmh0bWwiPgogICAgPGltZyBzcmM9Imh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9qY3Jpc3QvbXNnc3BlYy9tYWluL2RvY3MvX3N0YXRpYy9iZW5jaC12YWxpZGF0aW9uLnN2ZyI+CiAgPC9hPgo8L3A+CgpTZWUgW3RoZSBkb2N1bWVudGF0aW9uXShodHRwczovL2pjcmlzdGhhcmlmLmNvbS9tc2dzcGVjLykgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgoKIyMgTElDRU5TRQoKTmV3IEJTRC4gU2VlIHRoZQpbTGljZW5zZSBGaWxlXShodHRwczovL2dpdGh1Yi5jb20vamNyaXN0L21zZ3NwZWMvYmxvYi9tYWluL0xJQ0VOU0UpLgo= readmeEtag: '"60c39bfdd18456e986858137e4a7e00b1ec72775"' readmeLastModified: Sun, 16 Nov 2025 22:02:09 GMT repositoryId: 332952543 description: >- A fast serialization and validation library, with builtin support for JSON, MessagePack, YAML, and TOML created: '2021-01-26T02:53:57Z' updated: '2026-02-06T03:29:32Z' language: Python archived: false stars: 3569 watchers: 26 forks: 132 owner: jcrist logo: https://avatars.githubusercontent.com/u/2783717?v=4 license: BSD-3-Clause repoEtag: '"014a75bcbb684dd41c486d661a041c930149ddb5f959db483ea901de8c705d91"' repoLastModified: Fri, 06 Feb 2026 03:29:32 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/tfranzel/drf-spectacular v3: true repositoryMetadata: base64Readme: >- ===============
drf-spectacular
===============

|build-status| |codecov| |docs| |pypi-version| |pypi-dl|

Sane and flexible `OpenAPI`_ (`3.0.3`_ & `3.1`_) schema generation for `Django REST framework`_.

This project has 3 goals:
    1. Extract as much schema information from DRF as possible.
    2. Provide flexibility to make the schema usable in the real world (not only toy examples).
    3. Generate a schema that works well with the most popular client generators.

The code is a heavily modified fork of the
`DRF OpenAPI generator <https://github.com/encode/django-rest-framework/blob/master/rest_framework/schemas/openapi.py/>`_,
which is/was lacking all of the below listed features.

Features
    - Serializers modelled as components. (arbitrary nesting and recursion supported)
    - `@extend_schema <https://drf-spectacular.readthedocs.io/en/latest/drf_spectacular.html#drf_spectacular.utils.extend_schema>`_ decorator for customization of APIView, Viewsets, function-based views, and ``@action``
        - additional parameters
        - request/response serializer override (with status codes)
        - polymorphic responses either manually with ``PolymorphicProxySerializer`` helper or via ``rest_polymorphic``'s PolymorphicSerializer)
        - ... and more customization options
    - Authentication support (DRF natives included, easily extendable)
    - Custom serializer class support (easily extendable)
    - `SerializerMethodField() <https://drf-spectacular.readthedocs.io/en/latest/customization.html#step-3-extend-schema-field-and-type-hints>`_ type via type hinting or ``@extend_schema_field``
    - i18n support
    - Tags extraction
    - Request/response/parameter examples
    - Description extraction from ``docstrings``
    - Vendor specification extensions (``x-*``) in info, operations, parameters, components, and security schemes
    - Sane fallbacks
    - Sane ``operation_id`` naming (based on path)
    - Schema serving with ``SpectacularAPIView`` (Redoc and Swagger-UI views are also available)
    - Optional input/output serializer component split
    - Callback operations
    - OpenAPI 3.1 support (via setting ``OAS_VERSION``)
    - Included support for:
        - `django-polymorphic <https://github.com/django-polymorphic/django-polymorphic>`_ / `django-rest-polymorphic <https://github.com/apirobot/django-rest-polymorphic>`_
        - `SimpleJWT <https://github.com/jazzband/djangorestframework-simplejwt>`_
        - `DjangoOAuthToolkit <https://github.com/jazzband/django-oauth-toolkit>`_
        - `djangorestframework-jwt <https://github.com/jpadilla/django-rest-framework-jwt>`_ (tested fork `drf-jwt <https://github.com/Styria-Digital/django-rest-framework-jwt>`_)
        - `dj-rest-auth <https://github.com/iMerica/dj-rest-auth>`_ (maintained fork of `django-rest-auth <https://github.com/Tivix/django-rest-auth>`_)
        - `djangorestframework-camel-case <https://github.com/vbabiy/djangorestframework-camel-case>`_ (via postprocessing hook ``camelize_serializer_fields``)
        - `django-filter <https://github.com/carltongibson/django-filter>`_
        - `drf-nested-routers <https://github.com/alanjds/drf-nested-routers>`_
        - `djangorestframework-recursive <https://github.com/heywbj/django-rest-framework-recursive>`_
        - `djangorestframework-dataclasses <https://github.com/oxan/djangorestframework-dataclasses>`_
        - `django-rest-framework-gis <https://github.com/openwisp/django-rest-framework-gis>`_
        - `Pydantic (>=2.0) <https://github.com/pydantic/pydantic>`_


For more information visit the `documentation <https://drf-spectacular.readthedocs.io/>`_.

License
-------

Provided by `T. Franzel <https://github.com/tfranzel>`_. `Licensed under 3-Clause BSD <https://github.com/tfranzel/drf-spectacular/blob/master/LICENSE>`_.

Requirements
------------

-  Python >= 3.7
-  Django (2.2, 3.2, 4.0, 4.1, 4.2, 5.0, 5.1, 5.2)
-  Django REST Framework (3.10.3, 3.11, 3.12, 3.13, 3.14, 3.15, 3.16)

Installation
------------

Install using ``pip``\ ...

.. code:: bash

    $ pip install drf-spectacular

then add drf-spectacular to installed apps in ``settings.py``

.. code:: python

    INSTALLED_APPS = [
        # ALL YOUR APPS
        'drf_spectacular',
    ]


and finally register our spectacular AutoSchema with DRF.

.. code:: python

    REST_FRAMEWORK = {
        # YOUR SETTINGS
        'DEFAULT_SCHEMA_CLASS': 'drf_spectacular.openapi.AutoSchema',
    }

drf-spectacular ships with sane `default settings <https://drf-spectacular.readthedocs.io/en/latest/settings.html>`_
that should work reasonably well out of the box. It is not necessary to
specify any settings, but we recommend to specify at least some metadata.

.. code:: python

    SPECTACULAR_SETTINGS = {
        'TITLE': 'Your Project API',
        'DESCRIPTION': 'Your project description',
        'VERSION': '1.0.0',
        'SERVE_INCLUDE_SCHEMA': False,
        # OTHER SETTINGS
    }

.. _self-contained-ui-installation:

Self-contained UI installation
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

Certain environments have no direct access to the internet and as such are unable
to retrieve Swagger UI or Redoc from CDNs. `drf-spectacular-sidecar`_ provides
these static files as a separate optional package. Usage is as follows:

.. code:: bash

    $ pip install drf-spectacular[sidecar]

.. code:: python

    INSTALLED_APPS = [
        # ALL YOUR APPS
        'drf_spectacular',
        'drf_spectacular_sidecar',  # required for Django collectstatic discovery
    ]
    SPECTACULAR_SETTINGS = {
        'SWAGGER_UI_DIST': 'SIDECAR',  # shorthand to use the sidecar instead
        'SWAGGER_UI_FAVICON_HREF': 'SIDECAR',
        'REDOC_DIST': 'SIDECAR',
        # OTHER SETTINGS
    }


Release management
^^^^^^^^^^^^^^^^^^

*drf-spectacular* deliberately stays below version *1.x.x* to signal that every
new version may potentially break you. For production we strongly recommend pinning the
version and inspecting a schema diff on update.

With that said, we aim to be extremely defensive w.r.t. breaking API changes. However,
we also acknowledge the fact that even slight schema changes may break your toolchain,
as any existing bug may somehow also be used as a feature.

We define version increments with the following semantics. *y-stream* increments may contain
potentially breaking changes to both API and schema. *z-stream* increments will never break the
API and may only contain schema changes that should have a low chance of breaking you.


Take it for a spin
------------------

Generate your schema with the CLI:

.. code:: bash

    $ ./manage.py spectacular --color --file schema.yml
    $ docker run -p 8080:8080 -e SWAGGER_JSON=/schema.yml -v ${PWD}/schema.yml:/schema.yml swaggerapi/swagger-ui

If you also want to validate your schema add the ``--validate`` flag. Or serve your schema directly
from your API. We also provide convenience wrappers for ``swagger-ui`` or ``redoc``.

.. code:: python

    from drf_spectacular.views import SpectacularAPIView, SpectacularRedocView, SpectacularSwaggerView
    urlpatterns = [
        # YOUR PATTERNS
        path('api/schema/', SpectacularAPIView.as_view(), name='schema'),
        # Optional UI:
        path('api/schema/swagger-ui/', SpectacularSwaggerView.as_view(url_name='schema'), name='swagger-ui'),
        path('api/schema/redoc/', SpectacularRedocView.as_view(url_name='schema'), name='redoc'),
    ]

Usage
-----

*drf-spectacular* works pretty well out of the box. You might also want to set some metadata for your API.
Just create a ``SPECTACULAR_SETTINGS`` dictionary in your ``settings.py`` and override the defaults.
Have a look at the `available settings <https://drf-spectacular.readthedocs.io/en/latest/settings.html>`_.

The toy examples do not cover your cases? No problem, you can heavily customize how your schema will be rendered.

Customization by using ``@extend_schema``
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

Most customization cases should be covered by the ``extend_schema`` decorator. We usually get
pretty far with specifying ``OpenApiParameter`` and splitting request/response serializers, but
the sky is the limit.

.. code:: python

    from drf_spectacular.utils import extend_schema, OpenApiParameter, OpenApiExample
    from drf_spectacular.types import OpenApiTypes

    class AlbumViewset(viewset.ModelViewset):
        serializer_class = AlbumSerializer

        @extend_schema(
            request=AlbumCreationSerializer,
            responses={201: AlbumSerializer},
        )
        def create(self, request):
            # your non-standard behaviour
            return super().create(request)

        @extend_schema(
            # extra parameters added to the schema
            parameters=[
                OpenApiParameter(name='artist', description='Filter by artist', required=False, type=str),
                OpenApiParameter(
                    name='release',
                    type=OpenApiTypes.DATE,
                    location=OpenApiParameter.QUERY,
                    description='Filter by release date',
                    examples=[
                        OpenApiExample(
                            'Example 1',
                            summary='short optional summary',
                            description='longer description',
                            value='1993-08-23'
                        ),
                        ...
                    ],
                ),
            ],
            # override default docstring extraction
            description='More descriptive text',
            # provide Authentication class that deviates from the views default
            auth=None,
            # change the auto-generated operation name
            operation_id=None,
            # or even completely override what AutoSchema would generate. Provide raw Open API spec as Dict.
            operation=None,
            # attach request/response examples to the operation.
            examples=[
                OpenApiExample(
                    'Example 1',
                    description='longer description',
                    value=...
                ),
                ...
            ],
        )
        def list(self, request):
            # your non-standard behaviour
            return super().list(request)

        @extend_schema(
            request=AlbumLikeSerializer,
            responses={204: None},
            methods=["POST"]
        )
        @extend_schema(description='Override a specific method', methods=["GET"])
        @action(detail=True, methods=['post', 'get'])
        def set_password(self, request, pk=None):
            # your action behaviour
            ...

More customization
^^^^^^^^^^^^^^^^^^

Still not satisfied? You want more! We still got you covered.
Visit `customization <https://drf-spectacular.readthedocs.io/en/latest/customization.html>`_ for more information.


Testing
-------

Install testing requirements.

.. code:: bash

    $ pip install -r requirements.txt

Run with runtests.

.. code:: bash

    $ ./runtests.py

You can also use the excellent `tox`_ testing tool to run the tests
against all supported versions of Python and Django. Install tox
globally, and then simply run:

.. code:: bash

    $ tox

.. _Django REST framework: https://www.django-rest-framework.org/
.. _OpenAPI: https://swagger.io/
.. _3.0.3: https://spec.openapis.org/oas/v3.0.3
.. _3.1: https://spec.openapis.org/oas/v3.1.0
.. _tox: https://tox.wiki/
.. _drf-spectacular-sidecar: https://github.com/tfranzel/drf-spectacular-sidecar

.. |build-status| image:: https://github.com/tfranzel/drf-spectacular/actions/workflows/ci.yml/badge.svg
   :target: https://github.com/tfranzel/drf-spectacular/actions/workflows/ci.yml
.. |pypi-version| image:: https://img.shields.io/pypi/v/drf-spectacular.svg
   :target: https://pypi.org/project/drf-spectacular/
.. |codecov| image:: https://codecov.io/gh/tfranzel/drf-spectacular/branch/master/graph/badge.svg
   :target: https://codecov.io/gh/tfranzel/drf-spectacular
.. |docs| image:: https://readthedocs.org/projects/drf-spectacular/badge/
   :target: https://drf-spectacular.readthedocs.io/
.. |pypi-dl| image:: https://img.shields.io/pypi/dm/drf-spectacular
   :target: https://pypi.org/project/drf-spectacular/
 readmeEtag: '"2feea8b7159b116605948dfc3081fec9061ea298"' readmeLastModified: Sun, 21 Sep 2025 20:46:56 GMT repositoryId: 244234763 description: Sane and flexible OpenAPI 3 schema generation for Django REST framework. created: '2020-03-01T22:36:54Z' updated: '2026-02-05T21:33:48Z' language: Python archived: false stars: 2789 watchers: 9 forks: 307 owner: tfranzel logo: https://avatars.githubusercontent.com/u/13507857?v=4 license: BSD-3-Clause repoEtag: '"c9f86ed76a85d77f72a75ab3940f97f023bde34854687e528176f42fbc8e0f05"' repoLastModified: Thu, 05 Feb 2026 21:33:48 GMT foundInMaster: true category: Server Implementations id: 520f5c66c3e42a7b3fbface4f156c1dc - source: openapi3 tags name: Schemathesis homepage: https://github.com/schemathesis/schemathesis language: Python source_description: >- A modern API testing tool for web applications built with OpenAPI and GraphQL specifications category: - Testing Tools - Server Implementations repository: https://github.com/schemathesis/schemathesis v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vc2NoZW1hdGhlc2lzL3NjaGVtYXRoZXNpcy9hY3Rpb25zIiB0YXJnZXQ9Il9ibGFuayI+CiAgICAgICAgPGltZyBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9zY2hlbWF0aGVzaXMvc2NoZW1hdGhlc2lzL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnltbC9iYWRnZS5zdmciIGFsdD0iQnVpbGQiPgogICAgPC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9jb2RlY292LmlvL2doL3NjaGVtYXRoZXNpcy9zY2hlbWF0aGVzaXMvYnJhbmNoL21hc3RlciIgdGFyZ2V0PSJfYmxhbmsiPgogICAgICAgIDxpbWcgc3JjPSJodHRwczovL2NvZGVjb3YuaW8vZ2gvc2NoZW1hdGhlc2lzL3NjaGVtYXRoZXNpcy9icmFuY2gvbWFzdGVyL2dyYXBoL2JhZGdlLnN2ZyIgYWx0PSJDb3ZlcmFnZSI+CiAgICA8L2E+CiAgICA8YSBocmVmPSJodHRwczovL3B5cGkub3JnL3Byb2plY3Qvc2NoZW1hdGhlc2lzLyIgdGFyZ2V0PSJfYmxhbmsiPgogICAgICAgIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvdi9zY2hlbWF0aGVzaXMuc3ZnIiBhbHQ9IlZlcnNpb24iPgogICAgPC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L3NjaGVtYXRoZXNpcy8iIHRhcmdldD0iX2JsYW5rIj4KICAgICAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3B5dmVyc2lvbnMvc2NoZW1hdGhlc2lzLnN2ZyIgYWx0PSJQeXRob24gdmVyc2lvbnMiPgogICAgPC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9kaXNjb3JkLmdnL1I5QVNSQW1IbkEiIHRhcmdldD0iX2JsYW5rIj4KICAgICAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kaXNjb3JkLzkzODEzOTc0MDkxMjM2OTc1NSIgYWx0PSJEaXNjb3JkIj4KICAgIDwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUIiB0YXJnZXQ9Il9ibGFuayI+CiAgICAgICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9sL3NjaGVtYXRoZXNpcy5zdmciIGFsdD0iTGljZW5zZSI+CiAgICA8L2E+CjwvcD4KCiMjIFNjaGVtYXRoZXNpcwoKPiAqKkNhdGNoIEFQSSBidWdzIGJlZm9yZSB5b3VyIHVzZXJzIGRvLioqIAoKU2NoZW1hdGhlc2lzIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVzIHRob3VzYW5kcyBvZiB0ZXN0IGNhc2VzIGZyb20geW91ciBPcGVuQVBJIG9yIEdyYXBoUUwgc2NoZW1hIGFuZCBmaW5kcyBlZGdlIGNhc2VzIHRoYXQgYnJlYWsgeW91ciBBUEkuCgo8cCBhbGlnbj0iY2VudGVyIj4KICA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NjaGVtYXRoZXNpcy9zY2hlbWF0aGVzaXMvbWFzdGVyL2ltZy9kZW1vLmdpZiIgYWx0PSJTY2hlbWF0aGVzaXMgYXV0b21hdGljYWxseSBmaW5kaW5nIGEgc2VydmVyIGVycm9yIi8+CiAgPGJyPgogIDxpPkZpbmRpbmcgYnVncyB0aGF0IG1hbnVhbCB0ZXN0aW5nIG1pc3NlZDwvaT4KPC9wPgoKIyMgVHJ5IGl0IG5vdwoKYGBgY29uc29sZQojIFRlc3QgYSBkZW1vIEFQSSAtIGZpbmRzIHJlYWwgYnVncyBpbiAzMCBzZWNvbmRzCnV2eCBzY2hlbWF0aGVzaXMgcnVuIGh0dHBzOi8vZXhhbXBsZS5zY2hlbWF0aGVzaXMuaW8vb3BlbmFwaS5qc29uCgojIFRlc3QgeW91ciBvd24gQVBJCnV2eCBzY2hlbWF0aGVzaXMgcnVuIGh0dHBzOi8veW91ci1hcGkuY29tL29wZW5hcGkuanNvbgpgYGAKCgojIyBXaGF0IHByb2JsZW1zIGRvZXMgaXQgc29sdmU/CgotIPCfkqUgKio1MDAgZXJyb3JzKiogdGhhdCBjcmFzaCB5b3VyIEFQSSBvbiBlZGdlIGNhc2UgaW5wdXRzCi0g8J+TiyAqKlNjaGVtYSB2aW9sYXRpb25zKiogd2hlcmUgeW91ciBBUEkgcmV0dXJucyBkaWZmZXJlbnQgZGF0YSB0aGFuIGRvY3VtZW50ZWQKLSDwn5qqICoqVmFsaWRhdGlvbiBieXBhc3NlcyoqIHdoZXJlIGludmFsaWQgZGF0YSBnZXRzIGFjY2VwdGVkCi0g8J+UlyAqKkludGVncmF0aW9uIGZhaWx1cmVzKiogd2hlbiByZXNwb25zZXMgZG9uJ3QgbWF0Y2ggY2xpZW50IGV4cGVjdGF0aW9ucwotIPCflIQgKipTdGF0ZWZ1bCBidWdzKiogd2hlcmUgb3BlcmF0aW9ucyB3b3JrIGluZGl2aWR1YWxseSBidXQgZmFpbCBpbiByZWFsaXN0aWMgd29ya2Zsb3dzCgoKPiDimqDvuI8gKipVcGdyYWRpbmcgZnJvbSBvbGRlciB2ZXJzaW9ucz8qKiBDaGVjayBvdXIgW01pZ3JhdGlvbiBHdWlkZV0oaHR0cHM6Ly9naXRodWIuY29tL3NjaGVtYXRoZXNpcy9zY2hlbWF0aGVzaXMvYmxvYi9tYXN0ZXIvTUlHUkFUSU9OLm1kKSBmb3Iga2V5IGNoYW5nZXMuCgojIEluc3RhbGxhdGlvbiAmIFVzYWdlCgoqKkNvbW1hbmQgTGluZToqKgpgYGBjb25zb2xlCnV2IHBpcCBpbnN0YWxsIHNjaGVtYXRoZXNpcwpzY2hlbWF0aGVzaXMgcnVuIGh0dHBzOi8veW91ci1hcGkuY29tL29wZW5hcGkuanNvbgpgYGAKCioqUHl0aG9uIFRlc3RzOioqCmBgYHB5dGhvbgppbXBvcnQgc2NoZW1hdGhlc2lzCgpzY2hlbWEgPSBzY2hlbWF0aGVzaXMub3BlbmFwaS5mcm9tX3VybCgiaHR0cHM6Ly95b3VyLWFwaS5jb20vb3BlbmFwaS5qc29uIikKCkBzY2hlbWEucGFyYW1ldHJpemUoKQpkZWYgdGVzdF9hcGkoY2FzZSk6CiAgICAjIFRlc3RzIHdpdGggcmFuZG9tIGRhdGEsIGVkZ2UgY2FzZXMsIGFuZCBpbnZhbGlkIGlucHV0cwogICAgY2FzZS5jYWxsX2FuZF92YWxpZGF0ZSgpCgojIFN0YXRlZnVsIHRlc3Rpbmc6IFRlc3RzIHdvcmtmbG93cyBsaWtlOiBjcmVhdGUgdXNlciAtPiBnZXQgdXNlciAtPiBkZWxldGUgdXNlcgpBUElXb3JrZmxvdyA9IHNjaGVtYS5hc19zdGF0ZV9tYWNoaW5lKCkKIyBDcmVhdGVzIGEgdGVzdCBjbGFzcyBmb3IgcHl0ZXN0L3VuaXR0ZXN0ClRlc3RBUEkgPSBBUElXb3JrZmxvdy5UZXN0Q2FzZQpgYGAKCioqQ0kvQ0Q6KioKYGBgeWFtbAotIHVzZXM6IHNjaGVtYXRoZXNpcy9hY3Rpb25AdjIKICB3aXRoOgogICAgc2NoZW1hOiAiaHR0cHM6Ly95b3VyLWFwaS5jb20vb3BlbmFwaS5qc29uIgpgYGAKCiMjIFdobyB1c2VzIGl0CgpVc2VkIGJ5IHRlYW1zIGF0ICoqW1Nwb3RpZnldKGh0dHBzOi8vZ2l0aHViLmNvbS9iYWNrc3RhZ2UvYmFja3N0YWdlKSoqLCAqKltXb3JkUHJlc3NdKGh0dHBzOi8vZ2l0aHViLmNvbS9Xb3JkUHJlc3Mvb3BlbnZlcnNlKSoqLCAqKkpldEJyYWlucyoqLCAqKlJlZCBIYXQqKiwgYW5kIGRvemVucyBvZiBvdGhlciBjb21wYW5pZXMuCgoKPiAiX1NjaGVtYXRoZXNpcyBpcyB0aGUgYmVzdCB0b29sIGZvciBmdXp6IHRlc3Rpbmcgb2YgUkVTVCBBUElzIG9uIHRoZSBtYXJrZXQuIFdlIGF0IFJlZCBIYXQgdXNlIGl0IGZvciBleGFtaW5pbmcgb3VyIGFwcGxpY2F0aW9ucyBpbiBmdW5jdGlvbmFsIGFuZCBpbnRlZ3JhdGlvbiB0ZXN0aW5nIGxldmVscy5fIiAtIERtaXRyeSBNaXNoYXJvdiwgUmVkSGF0CgojIyBTZWUgaXQgaW4gYWN0aW9uCgrwn5SsICoqW0xpdmUgQmVuY2htYXJrc10oaHR0cHM6Ly93b3JrYmVuY2guc2NoZW1hdGhlc2lzLmlvKSoqIHNob3dpbmcgY29udGludW91cyB0ZXN0aW5nIHJlc3VsdHMgZnJvbSByZWFsLXdvcmxkIEFQSXM6CgotIENvZGUgJiBBUEkgc2NoZW1hIGNvdmVyYWdlIGFjaGlldmVkCi0gSXNzdWVzIGZvdW5kIHdpdGggZGV0YWlsZWQgY2F0ZWdvcml6YXRpb24KLSBQZXJmb3JtYW5jZSBhY3Jvc3MgZGlmZmVyZW50IGZ1enppbmcgc3RyYXRlZ2llcwoKIyMgRG9jdW1lbnRhdGlvbgoK8J+TmiAqKltEb2N1bWVudGF0aW9uXShodHRwczovL3NjaGVtYXRoZXNpcy5yZWFkdGhlZG9jcy5pby9lbi9zdGFibGUvKSoqIHdpdGggZ3VpZGVzLCBleGFtcGxlcywgYW5kIEFQSSByZWZlcmVuY2UuCgojIyBHZXQgSGVscAoKLSDwn5KsIFtEaXNjb3JkIGNvbW11bml0eV0oaHR0cHM6Ly9kaXNjb3JkLmdnL1I5QVNSQW1IbkEpCi0g8J+QmyBbR2l0SHViIGlzc3Vlc10oaHR0cHM6Ly9naXRodWIuY29tL3NjaGVtYXRoZXNpcy9zY2hlbWF0aGVzaXMvaXNzdWVzKQoKIyMgQ29udHJpYnV0aW5nCgpXZSB3ZWxjb21lIGNvbnRyaWJ1dGlvbnMhIFNlZSBvdXIgW2NvbnRyaWJ1dGluZyBndWlkZWxpbmVzXShDT05UUklCVVRJTkcubWQpIGFuZCBqb2luIGRpc2N1c3Npb25zIGluIFtpc3N1ZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9zY2hlbWF0aGVzaXMvc2NoZW1hdGhlc2lzL2lzc3Vlcykgb3IgW0Rpc2NvcmRdKGh0dHBzOi8vZGlzY29yZC5nZy9SOUFTUkFtSG5BKS4KCiMjIEFja25vd2xlZGdlbWVudHMKClNjaGVtYXRoZXNpcyBpcyBidWlsdCBvbiB0b3Agb2YgPGEgaHJlZj0iaHR0cHM6Ly9oeXBvdGhlc2lzLndvcmtzLyIgdGFyZ2V0PSJfYmxhbmsiPkh5cG90aGVzaXM8L2E+LCBhIHBvd2VyZnVsIHByb3BlcnR5LWJhc2VkIHRlc3RpbmcgbGlicmFyeSBmb3IgUHl0aG9uLgoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgW01JVCBsaWNlbnNlXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkuCg== readmeEtag: '"4433d725270a09d142c5b505077029a2802586ab"' readmeLastModified: Sun, 23 Nov 2025 18:09:11 GMT repositoryId: 203157084 description: Catch API bugs before your users do created: '2019-08-19T11:13:01Z' updated: '2026-02-06T02:40:21Z' language: Python archived: false stars: 3019 watchers: 18 forks: 197 owner: schemathesis logo: https://avatars.githubusercontent.com/u/70383271?v=4 license: MIT repoEtag: '"8ea36cb90ea3765aaf29696d331d7c7aec4526116addf82a2c4f93a2285cbcd2"' repoLastModified: Fri, 06 Feb 2026 02:40:21 GMT foundInMaster: true id: f4032fb371a703acffa691c0f1239d45 oldLocations: - https://github.com/kiwicom/schemathesis - source: openapi3 tags repository: https://github.com/google/gnostic v3: true repositoryMetadata: base64Readme: >- [![Go Actions Status](https://github.com/google/gnostic/workflows/Go/badge.svg)](https://github.com/google/gnostic/actions)

# ⨁ gnostic

This repository contains a Go command line tool which converts JSON and YAML
[OpenAPI](https://github.com/OAI/OpenAPI-Specification) descriptions to and
from equivalent Protocol Buffer representations.

[Protocol Buffers](https://developers.google.com/protocol-buffers/) provide a
language-neutral, platform-neutral, extensible mechanism for serializing
structured data. **gnostic**'s Protocol Buffer models for the OpenAPI
Specification can be used to generate code that includes data structures with
explicit fields for the elements of an OpenAPI description. This makes it
possible for developers to work with OpenAPI descriptions in type-safe ways,
which is particularly useful in strongly-typed languages like Go and
[Dart](https://dart.dev/).

**gnostic** reads OpenAPI descriptions into these generated data structures,
reports errors, resolves internal dependencies, and writes the results in a
binary form that can be used in any language that is supported by the Protocol
Buffer tools. A plugin interface simplifies integration with API tools written
in a variety of different languages, and when necessary, Protocol Buffer
OpenAPI descriptions can be reexported as JSON or YAML.

**gnostic** compilation code and OpenAPI Protocol Buffer models are
automatically generated from an
[OpenAPI JSON Schema](https://github.com/OAI/OpenAPI-Specification/blob/master/schemas/v2.0/schema.json).
Source code for the generator is in the [generate-gnostic](generate-gnostic)
directory.

## Related Repositories

[google/gnostic-models](https://github.com/google/gnostic-models) contains a
lightweight distribution of the protobuf models generated by this project.
Where a low-dependency integration of just these models is needed, Go projects
can import packages from `gnostic-models` instead of `gnostic`.

[google/gnostic-grpc](https://github.com/google/gnostic-grpc) contains a
gnostic plugin that can generate an annotated Protocol Buffer description of an
API that, when transcoded, produces an API that conforms to a specified OpenAPI
document. To go from protobuf to OpenAPI, see the
[protoc-gen-openapi](cmd/protoc-gen-openapi) tool in this project.

[google/gnostic-go-generator](https://github.com/google/gnostic-go-generator)
contains an experimental gnostic plugin that generates a Go client for an API
described by a specified OpenAPI document.

## Disclaimer

Feedback and contributions are welcome! Until there is a 1.0 release, please
consider this prerelease software and work in progress. To ensure stable
builds, we request that dependent projects always refer to tagged releases of
**gnostic**.

## Requirements

**gnostic** can be run in any environment that supports [Go](http://golang.org)
and the
[Protocol Buffer Compiler](https://github.com/protocolbuffers/protobuf).

## Installation and Getting Started

The following instructions are for installing **gnostic** using
[Go modules](https://blog.golang.org/using-go-modules), supported by Go 1.11
and later.

1.  Get this package by downloading it with `git clone`.

        git clone https://github.com/google/gnostic
        cd gnostic

2.  Verify that you have a local installation of `protoc`. You can get `protoc`
    [here](https://github.com/protocolbuffers/protobuf).

3.  Build **gnostic** with `make`. This uses
    [go generate](https://blog.golang.org/generate) to build support code
    including code generated by `protoc` and the Go `protoc` plugin, which is
    automatically downloaded from
    [github.com/golang/protobuf](https://github.com/golang/protobuf) by the
    [COMPILE-PROTOS.sh](COMPILE-PROTOS.sh) script. This also builds all plugins
    and associated tools in this repo.

4.  Verify **gnostic** with `make test`. These tests are run by **gnostic**'s
    continuous integration, so you should expect them to pass for all release
    versions.

5.  Run **gnostic**. This sample invocation creates a file in the current
    directory named `petstore.pb` that contains a binary Protocol Buffer
    description of a sample API.

            gnostic --pb-out=. examples/v2.0/json/petstore.json

6.  You can also compile files that you specify with a URL. Here's another way
    to compile the previous example. This time we're creating `petstore.text`,
    which contains a textual representation of the Protocol Buffer description.
    This is mainly for use in testing and debugging.

            gnostic --text-out=petstore.text https://raw.githubusercontent.com/google/gnostic/master/examples/v2.0/json/petstore.json

7.  For a sample application, see apps/report. This reads a binary Protocol
    Buffer encoding created by **gnostic**.

        go install ./apps/report ## automatically installed by the top-level Makefile
        report petstore.pb

8.  **gnostic** also supports plugins. **gnostic**'s plugin interface is
    modeled on `protoc`'s
    [plugin.proto](https://github.com/protocolbuffers/protobuf/blob/master/src/google/protobuf/compiler/plugin.proto)
    and is described in [plugins/plugin.proto](plugins/plugin.proto). Several
    plugins are implemented in the `plugins` directory. Others, like
    [gnostic-grpc](https://github.com/google/gnostic-grpc) and
    [gnostic-go-generator](https://github.com/google/gnostic-go-generator),
    are published in their own repositories. One such plugin is
    [gnostic-vocabulary](plugins/gnostic-vocabulary), which produces a summary
    of the word usage in an APIs interfaces. You can run `gnostic-vocabulary`
    with the following:

            gnostic examples/v2.0/json/petstore.json --vocabulary_out=.

    This will produce files named `vocabulary.pb` and `vocabulary.json` in
    `examples/v2.0/json`. For the format of `vocabulary.pb`, see
    [metrics/vocabulary.proto](metrics/vocabulary.proto).

9.  [Optional] A large part of **gnostic** is automatically-generated by the
    [generate-gnostic](generate-gnostic) tool. This uses JSON schemas to
    generate Protocol Buffer language files that describe supported API
    specification formats and Go-language files of code that will read JSON or
    YAML API descriptions into the generated protocol buffer models.
    Pre-generated versions of these files are checked into the
    [openapiv2](openapiv2), [openapiv3](openapiv3), and [discovery](discovery)
    directories. You can regenerate this code with the following:

        go install ./generate-gnostic
        generate-gnostic --v2
        generate-gnostic --v3
        generate-gnostic --discovery

## Copyright

Copyright 2017-2020, Google LLC.

## License

Released under the Apache 2.0 license.
 readmeEtag: '"2ea85c46ff9c52bc42fa33d54ac43fb1353eb957"' readmeLastModified: Thu, 29 Feb 2024 22:55:22 GMT repositoryId: 73847381 description: >- A compiler for APIs described by the OpenAPI Specification with plugins for code generation and other API support tasks. created: '2016-11-15T19:21:53Z' updated: '2026-02-04T12:04:16Z' language: Go archived: false stars: 2265 watchers: 35 forks: 272 owner: google logo: https://avatars.githubusercontent.com/u/1342004?v=4 license: Apache-2.0 repoEtag: '"ce03df1fa6f0b322335dde288841be59a62dd4ca60d8afaeb392a2b02b33bb99"' repoLastModified: Wed, 04 Feb 2026 12:04:16 GMT foundInMaster: true category: Converters id: 46e070512676eb0fd196ee199e6a0b34 oldLocations: - https://github.com/googleapis/gnostic - source: openapi3 tags repository: https://github.com/rapi-doc/rapidoc v3: true repositoryMetadata: base64Readme: >- PGltZyBhbHQ9Ik1yaW5Eb2MgbG9nbyIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vcmFwaS1kb2MvUmFwaURvYy9ibG9iL21hc3Rlci9sb2dvLnBuZyIgd2lkdGg9IjYwcHgiIC8+CgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSIvPgogICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3NpemUvcmFwaS1kb2MvcmFwaWRvYy9kaXN0L3JhcGlkb2MtbWluLmpzLnN2Zz9jb2xvckI9Ymx1ZSZsYWJlbD1taW5pZmllZCZzdHlsZT1mbGF0LXNxdWFyZSI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvc2l6ZS9yYXBpLWRvYy9yYXBpZG9jL2Rpc3QvcmFwaWRvYy1taW4uanMuZ3ouc3ZnP2NvbG9yQj1ibHVlJmxhYmVsPXppcCZzdHlsZT1mbGF0LXNxdWFyZSI+CiAgICA8YSBocmVmPSJodHRwczovL3d3dy53ZWJjb21wb25lbnRzLm9yZy9lbGVtZW50L3JhcGlkb2MiIGFsdD0icHVibGlzaGVkIG9uIHdlYmNvbXBvbmVudHMub3JnIj4KICAgICAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS93ZWJjb21wb25lbnRzLm9yZy1yYXBpZG9jLWJsdWUuc3ZnP3N0eWxlPXNvY2lhbCIvPgogICAgPC9hPgo8L3A+ICAgICAgICAKCiMgUmFwaURvYwpDdXN0b20gRWxlbWVudCBmb3IgT3Blbi1BUEkgc3BlYyB2aWV3aW5nCgo+IFNwb25zb3JlZCBieSBbWnVwbG9dKGh0dHBzOi8venVwbG8ubGluay9yYXBpZG9jLTIpIC0gR2V0IGEgU3RyaXBlLWxpa2UgQVBJIGV4cGVyaWVuY2UgZm9yIHlvdXIgY3VzdG9tZXJzIGluIG1pbnV0ZXMgLSBkb2N1bWVudGF0aW9uLCByYXRlLWxpbWl0aW5nIGFuZCBBUEkta2V5IGF1dGggaW4gbWludXRlcy4gCgojIyBGZWF0dXJlcwotIFN1cHBvcnRzIFN3YWdnZXIgMi4wLCBPcGVuQVBJIDMueC54IAotIFdvcmtzIHdpdGggYW55IGZyYW1ld29yayBvciB3aXRoIG5vIGZyYW1ld29yawotIEFsbG93cyBtYWtpbmcgQVBJIGNhbGxzCi0gQmV0dGVyIFVzYWJpbGl0eSwgCiAgLSBhbGwgTW9kZWxzIGFuZCBFeGFtcGxlcyBhcmUgZXhwYW5kZWQgYnkgZGVmYXVsdCwgZWxpbWluYXRlcyB0aGUgbmVlZCB0byBjbGljayBhbmQgcmV2ZWFsLgogIC0gUmVxdWVzdCBmaWVsZHMgYXJlIHByZS1wb3B1bGF0ZWQgd2l0aCBzYW1wbGUgZGF0YQogIC0gVGFrZXMgb25seSBvbmUgY2xpY2sgdG8gbWFrZSBhbiBBUEkgY2FsbAogIC0gUmVxdWVzdCBhbmQgcmVzcG9uc2UgY2FuIGJlIHBsYWNlZCBzaWRlLWJ5LXNpZGUgZm9yIGVhc3kgY29tcGFyaXNvbgotIEJyYW5kaW5nIGFuZCBQZXJzb25hbGl6YXRpb24gZmVhdHVyZXMgbWFrZXMgaXQgZWFzeSB0byBmb2xsb3cgYW55IHN0eWxlIGd1aWRlCiAgLSBDb21lcyB3aXRoIDIgVGhlbWVzIChEYXJrIGFuZCBMaWdodCkKICAtIFJlcGxhY2UgZGVmYXVsdCBsb2dvIHdpdGggeW91cnMKICAtIFR5cG9ncmFwaHksIGFsbG93cyBjaGFuZ2luZyBmb250cwogIC0gQWxsb3dzIGNoYW5naW5nIHRleHQtY29sb3IsIGJ1dHRvbi1jb2xvciwgaGVhZGVyLWNvbG9yIGFuZCBjb2xvciBvZiBvdGhlciBVSSBlbGVtZW50cwotIFBsZW50eSBvZiBjdXN0b21pemF0aW9uIG9wdGlvbnMgCiAgLSBBZGQgZXh0ZXJuYWwgY29udGVudHMgYXQgdGhlIHRvcCBhbmQgYm90dG9tIG9mIHRoZSBkb2N1bWVudCwgIHlvdSBtYXkgYWRkIGltYWdlcywgbGluaywgdGV4dCwgZm9ybXMgZXRjCiAgLSBBbGxvd3MgZGlzYWJsaW5nIEFQSSBjYWxsaW5nIGZlYXR1cmUKICAtIEhpZGUgdGhlIGhlYWRlciwgc28gdGhlIHVzZXIgY2FuJ3QgbG9hZCBhbnkgb3RoZXIgT3BlbkFQSSBTcGVjCiAgLSBIaWRlIEF1dGhlbnRpY2F0aW9uIGFuZCBwcm92aWRlIHlvdXIgb3duIHByZS1nZW5lcmF0ZWQgYXBpLWtleSAKICAtIEVtYmVkIGl0IGluc2lkZSBhbm90aGVyIEhUTUwgZG9jdW1lbnQKICAtIFVzZSBpdCBpbnNpZGUgYW5vdGhlciBmcmFtZXdvcmsgKHJlYWN0LCB2dWUsIGFuZ3VsYXIsIGxpdC1lbGVtZW50KQogIC0gVXNlIEphdmFTY3JpcHQgdG8gY2hhbmdlIGl0cyBhdHRyaWJ1dGVzLCBhbmQgaXQgd2lsbCByZWFjdCB0byB0aG9zZSBjaGFuZ2VzCiAgLSBTdHlsZSB0aGUgZWxlbWVudCB3aXRoIHN0YW5kYXJkIGNzcyAoY2hhbmdlIHBhZGRpbmcsIHBvc2l0aW9uLCBib3JkZXIsIG1hcmdpbiApCi0gTGlnaHR3ZWlnaHQgYW5kIGZhc3QKLSBMb2FkIGxvY2FsIGpzb24gc3BlYyBmcm9tIHRoZSBkaXNrCi0gU3VwcG9ydGVkIG9uIENocm9tZSwgRmlyZUZveCBhbmQgU2FmYXJpLiAoTm90IHlldCB0ZXN0ZWQgb24gRWRnZSkKCgojIyBEb2N1bWVudGF0aW9uCltDaGVjayBvdXQgdGhlIHVzYWdlIGFuZCBkZW1vc10oaHR0cHM6Ly9yYXBpLWRvYy5naXRodWIuaW8vUmFwaURvYy8pCgojIyBFeGFtcGxlcwpbRXhhbXBsZXMgYW5kIFRlc3QgY2FzZXNdKGh0dHBzOi8vcmFwaS1kb2MuZ2l0aHViLmlvL1JhcGlEb2MvbGlzdC5odG1sKQoKCiMjIEJ1aWxkIFByb2Nlc3MKYGBgYmFzaAojIENsb25lIC8gRG93bmxvYWQgdGhlIHByb2plY3QgdGhlbgpucG0gaW5zdGFsbAoKIyBidWlsZCB3aWxsIGdlbmVyYXRlIHJhcGlkb2MtbWluLmpzLCB0aGlzIGlzIHRoZSBvbmx5IGZpbGUgeW91IHdpbGwgbmVlZC4KIyB1c2UgaXQgaW4gdGhlIHNjcmlwdCB0YWcgb2YgeW91ciBodG1sIDxzY3JpcHQgdHlwZT0idGV4dC9qYXZhc2NyaXB0IiBzcmM9InJhcGlkb2MtbWluLmpzIj48L3NjcmlwdD48L2JvZHk+Cm5wbSBydW4gYnVpbGQgCgojIGZvciBkZXZlbG9wbWVudCB1c2UgeWFybiBzZXJ2ZSAodGhpcyB3aWxsIHN0YXJ0IGFuIHdlYnNlcnZlciBhdCBwb3J0IDgwODAsIHRoZW4gbmF2aWdhdGUgdG8gbG9jYWxob3N0OjgwODApIApucG0gcnVuIHNlcnZlCmBgYAoKIyMgUm9hZG1hcAotIOKchSBVcGdyYWRlIHRvIHVzZSBgbGl0IDJgCi0gUHJlIFJlbmRlcmluZyAvIFNlcnZlciBTaWRlIFJlbmRlcmluZyAoU1NSKSBvZiBSYXBpRG9jCi0gSW1wcm92ZSBQREYgZ2VuZXJhdGlvbiBvZiBPcGVuQVBJIHNwZWMgVXNpbmcgUmFwaVBERgotIENyZWF0ZSBhIENvbW1hbmQgTGluZSBpbnRlcmZhY2UgKENMSSkgdG8gbWFrZSBpdCBhdXRvbWF0aW9uIGZyaWVuZGx5Ci0gQWRkIGxvY2FsaXphdGlvbiBzdXBwb3J0Ci0gRm9sbG93IFdlYiBDb250ZW50IEFjY2Vzc2liaWxpdHkgR3VpZGVsaW5lcyAoV0NBRyAyKQotIEltcHJvdmUgSG9tZXBhZ2UsIERvY3VtZW50YXRpb24gYW5kIEV4YW1wbGVzCi0gQ3JlYXRlIGdyb3VuZHdvcmsgZm9yIGF1dG9tYXRlZCB0ZXN0aW5nCi0gQ3JlYXRlIGEgbGl2ZSBwbGF5Z3JvdW5kIChTb21ldGhpbmcgbGlrZSBzd2FnZ2VyIGVkaXRvcikKLSBDcmVhdGUgYSBzaG9ydCBhbmltYXRpb24vdmlkZW8gdG8gc2hvdyBob3cgUmFwaURvYy9SYXBpUERGIHdvcmtzCgo= readmeEtag: '"adae48ecaae89a839f0b2a2fe36d62ee9eb12e72"' readmeLastModified: Thu, 04 Jul 2024 15:51:19 GMT repositoryId: 165951750 description: RapiDoc -WebComponent for OpenAPI Spec created: '2019-01-16T01:29:58Z' updated: '2026-02-03T07:42:34Z' language: JavaScript archived: false stars: 1879 watchers: 16 forks: 316 owner: rapi-doc logo: https://avatars.githubusercontent.com/u/104180085?v=4 license: MIT repoEtag: '"cdcb2a896f066b96127f772c5129fd3f19f5de9d607a6530753537dd481c4120"' repoLastModified: Tue, 03 Feb 2026 07:42:34 GMT foundInMaster: true category: - Documentation - Server Implementations id: b903cad8b99b490dbecf6c6951d5b500 oldLocations: - https://github.com/mrin9/rapidoc - source: openapi3 tags repository: https://github.com/ogen-go/ogen v3: true id: 3d3e1f300ede27ed165969a8737ba867 repositoryMetadata: base64Readme: >- <p align="center">
<img width="256" height="256" src="_logo/logo.svg" alt="ogen svg logo">
</p>

# ogen [![Go Reference](https://img.shields.io/badge/go-pkg-00ADD8)](https://pkg.go.dev/github.com/ogen-go/ogen#section-documentation) [![codecov](https://img.shields.io/codecov/c/github/ogen-go/ogen?label=cover)](https://codecov.io/gh/ogen-go/ogen) [![stable](https://img.shields.io/badge/-stable-brightgreen)](https://go-faster.org/docs/projects/status#stable)

OpenAPI v3 Code Generator for Go.

- [Getting started](https://ogen.dev/docs/intro)
- [Sample project](https://github.com/ogen-go/example)
- [Security policy](https://github.com/ogen-go/ogen/blob/-/SECURITY.md)
- [Telegram group `@ogen_dev`](https://t.me/ogen_dev)

# Install

```console
go install -v github.com/ogen-go/ogen/cmd/ogen@latest
```

# Usage

```go
//go:generate go run github.com/ogen-go/ogen/cmd/ogen --target target/dir -package api --clean schema.json
```

or using container:
```shell
docker run --rm \
  --volume ".:/workspace" \
  ghcr.io/ogen-go/ogen:latest --target workspace/petstore --clean workspace/petstore.yml
```

# Features

- No reflection or `interface{}`
  - The json encoding is code-generated, optimized and uses [go-faster/jx](https://github.com/go-faster/jx) for speed and overcoming `encoding/json` limitations
  - Validation is code-generated according to spec
- Code-generated static radix router
- No more boilerplate
  - Structures are generated from OpenAPI v3 specification
  - Arguments, headers, url queries are parsed according to specification into structures
  - String formats like `uuid`, `date`, `date-time`, `uri` are represented by go types directly
- Statically typed client and server
- Convenient support for optional, nullable and optional nullable fields
  - No more pointers
  - Generated Optional[T], Nullable[T] or OptionalNullable[T] wrappers with helpers
  - Special case for array handling with `nil` semantics relevant to specification
    - When array is optional, `nil` denotes absence of value
    - When nullable, `nil` denotes that value is `nil`
    - When required, `nil` currently the same as `[]`, but is actually invalid
    - If both nullable and required, wrapper will be generated (TODO)
- Support for untyped parameters (any)
  - Parameters with no `type` specified in schema are represented as Go `any`
  - Decoded as strings from URI (path, query, header, cookie)
  - Client encoding uses `fmt.Sprint` for flexible value conversion
  - Useful for legacy APIs or dynamic parameter types
- Generated sum types for oneOf
  - Primitive types (`string`, `number`) are detected by type
  - Discriminator field is used if defined in schema
  - Type is inferred by unique fields if possible
    - Field name discrimination: variants with different field names
    - Field type discrimination: variants with same field names but different types (e.g., `{id: string}` vs `{id: integer}`)
    - Field value discrimination: variants with same field names and types but different enum values
- Extra Go struct field tags in the generated types
- OpenTelemetry tracing and metrics

Example generated structure from schema:

```go
// Pet describes #/components/schemas/Pet.
type Pet struct {
	Birthday     time.Time     `json:"birthday"`
	Friends      []Pet         `json:"friends"`
	ID           int64         `json:"id"`
	IP           net.IP        `json:"ip"`
	IPV4         net.IP        `json:"ip_v4"`
	IPV6         net.IP        `json:"ip_v6"`
	Kind         PetKind       `json:"kind"`
	Name         string        `json:"name"`
	Next         OptData       `json:"next"`
	Nickname     NilString     `json:"nickname"`
	NullStr      OptNilString  `json:"nullStr"`
	Rate         time.Duration `json:"rate"`
	Tag          OptUUID       `json:"tag"`
	TestArray1   [][]string    `json:"testArray1"`
	TestDate     OptTime       `json:"testDate"`
	TestDateTime OptTime       `json:"testDateTime"`
	TestDuration OptDuration   `json:"testDuration"`
	TestFloat1   OptFloat64    `json:"testFloat1"`
	TestInteger1 OptInt        `json:"testInteger1"`
	TestTime     OptTime       `json:"testTime"`
	Type         OptPetType    `json:"type"`
	URI          url.URL       `json:"uri"`
	UniqueID     uuid.UUID     `json:"unique_id"`
}
```

Example generated server interface:

```go
// Server handles operations described by OpenAPI v3 specification.
type Server interface {
	PetGetByName(ctx context.Context, params PetGetByNameParams) (Pet, error)
	// ...
}
```

Example generated client method signature:

```go
type PetGetByNameParams struct {
    Name string
}

// GET /pet/{name}
func (c *Client) PetGetByName(ctx context.Context, params PetGetByNameParams) (res Pet, err error)
```

## Generics

Instead of using pointers, `ogen` generates generic wrappers.

For example, `OptNilString` is `string` that is optional (no value) and can be `null`.

```go
// OptNilString is optional nullable string.
type OptNilString struct {
	Value string
	Set   bool
	Null  bool
}
```

Multiple convenience helper methods and functions are generated, some of them:

```go
func (OptNilString) Get() (v string, ok bool)
func (OptNilString) IsNull() bool
func (OptNilString) IsSet() bool

func NewOptNilString(v string) OptNilString
```

## Recursive types

If `ogen` encounters recursive types that can't be expressed in go, pointers are used as fallback.

## Sum types

For `oneOf` sum-types are generated. `ID` that is one of `[string, integer]` will be represented like that:

```go
type ID struct {
	Type   IDType
	String string
	Int    int
}

// Also, some helpers:
func NewStringID(v string) ID
func NewIntID(v int) ID
```

### Discriminator Inference

ogen automatically infers how to discriminate between oneOf variants using several strategies:

**1. Type-based discrimination** (for primitive types)

Variants with different JSON types are discriminated by checking the JSON type at runtime:

```json
{
  "oneOf": [
    {"type": "string"},
    {"type": "integer"}
  ]
}
```

**2. Explicit discriminator** (when discriminator field is specified)

When a discriminator field is defined in the schema, ogen uses it directly:

```json
{
  "oneOf": [...],
  "discriminator": {
    "propertyName": "type",
    "mapping": {"user": "#/components/schemas/User", ...}
  }
}
```

**3. Field-based discrimination** (automatic inference from unique fields)

ogen analyzes the fields in each variant to find discriminating characteristics:

- **Field name discrimination**: Variants have different field names

```json
{
  "oneOf": [
    {"type": "object", "required": ["userId"], "properties": {"userId": {"type": "string"}}},
    {"type": "object", "required": ["orderId"], "properties": {"orderId": {"type": "string"}}}
  ]
}
```

- **Field type discrimination**: Variants have fields with the same name but different types

```json
{
  "oneOf": [
    {
      "type": "object",
      "required": ["id", "value"],
      "properties": {
        "id": {"type": "string"},
        "value": {"type": "string"}
      }
    },
    {
      "type": "object",
      "required": ["id", "value"],
      "properties": {
        "id": {"type": "integer"},
        "value": {"type": "number"}
      }
    }
  ]
}
```

In this case, ogen checks the JSON type of the `id` field at runtime to determine which variant to decode.

- **Field value discrimination**: Variants have fields with the same name and type but different enum values

```json
{
  "oneOf": [
    {
      "type": "object",
      "required": ["status"],
      "properties": {
        "status": {"type": "string", "enum": ["active", "pending"]}
      }
    },
    {
      "type": "object",
      "required": ["status"],
      "properties": {
        "status": {"type": "string", "enum": ["inactive", "deleted"]}
      }
    }
  ]
}
```

In this case, ogen checks the actual string value of the `status` field at runtime and matches it against each variant's enum values. The enum values must be disjoint (non-overlapping) for this to work. If enum values overlap, ogen will report an error and suggest using an explicit discriminator.

## Const values

ogen supports the JSON Schema [`const`](https://json-schema.org/understanding-json-schema/reference/generic#constant-values) keyword, which specifies that a field must have a fixed value (introduced in [JSON Schema draft 6](https://json-schema.org/draft-06/json-schema-release-notes.html) and supported in OpenAPI 3.0+). When a field has a `const` value, it is encoded directly in the generated JSON encoder without requiring the struct field to be set.

### Example schema with const values

```yaml
components:
  schemas:
    ErrorResponse:
      type: object
      properties:
        code:
          type: integer
          const: 400
        status:
          type: string
          const: "error"
        message:
          type: string
```

### Generated code

The generated struct includes the field, but the encoder hardcodes the const value:

```go
type ErrorResponse struct {
    Code    int64  `json:"code"`    // const: 400
    Status  string `json:"status"`  // const: "error"
    Message string `json:"message"`
}

func (s *ErrorResponse) encodeFields(e *jx.Encoder) {
    {
        e.FieldStart("code")
        e.Int64(400)  // Const value encoded directly
    }
    {
        e.FieldStart("status")
        e.Str("error")  // Const value encoded directly
    }
    {
        e.FieldStart("message")
        e.Str(s.Message)  // Regular field
    }
}
```

### Benefits

- **Simplified initialization**: You don't need to set const fields when creating struct instances
- **Type safety**: Const values are validated at code generation time
- **Performance**: Const values are encoded directly without runtime lookups
- **Works with allOf**: Const values are preserved when merging schemas with `allOf`

### Supported const value types

- Primitives: `integer`, `number`, `string`, `boolean`
- Special values: `null`, empty strings (`""`), zero values (`0`, `false`)
- Complex types: `object`, `array` (when specified in schema)

## Extension properties

OpenAPI enables [Specification Extensions](https://spec.openapis.org/oas/v3.1.0#specification-extensions),
which are implemented as patterned fields that are always prefixed by `x-`.

### Server name

Optionally, server name can be specified by `x-ogen-server-name`, for example:

```json
{
  "openapi": "3.0.3",
  "servers": [
    {
      "x-ogen-server-name": "production",
      "url": "https://{region}.example.com/{val}/v1",
    },
    {
      "x-ogen-server-name": "prefix",
      "url": "/{val}/v1",
    },
    {
      "x-ogen-server-name": "const",
      "url": "https://cdn.example.com/v1"
    }
  ],
(...)
```

### Custom type name

Optionally, type name can be specified by `x-ogen-name`, for example:

```json
{
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "x-ogen-name": "Name",
  "properties": {
    "foobar": {
      "$ref": "#/$defs/FooBar"
    }
  },
  "$defs": {
    "FooBar": {
      "x-ogen-name": "FooBar",
      "type": "object",
      "properties": {
        "foo": {
          "type": "string"
        }
      }
    }
  }
}
```

### Custom field name

Optionally, type name can be specified by `x-ogen-properties`, for example:

```yaml
components:
  schemas:
    Node:
      type: object
      properties:
        parent:
          $ref: "#/components/schemas/Node"
        child:
          $ref: "#/components/schemas/Node"
      x-ogen-properties:
        parent:
          name: "Prev"
        child:
          name: "Next"
```

The generated source code looks like:

```go
// Ref: #/components/schemas/Node
type Node struct {
    Prev *Node `json:"parent"`
    Next *Node `json:"child"`
}
```

### Extra struct field tags

Optionally, additional Go struct field tags can be specified by `x-oapi-codegen-extra-tags`, for example:

```yaml
components:
  schemas:
    Pet:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
          format: int64
          x-oapi-codegen-extra-tags:
            gorm: primaryKey
            valid: customIdValidator
```

The generated source code looks like:

```go
// Ref: #/components/schemas/Pet
type Pet struct {
    ID   int64     `gorm:"primaryKey" valid:"customNameValidator" json:"id"`
}
```

### Streaming JSON encoding

By default, ogen loads the entire JSON body into memory before decoding it.
Optionally, streaming JSON encoding can be enabled by `x-ogen-json-streaming`, for example:

```yaml
requestBody:
  required: true
  content:
    application/json:
      x-ogen-json-streaming: true
      schema:
        type: array
        items:
          type: number
```

### Custom validation

Optionally, custom validation can be specified by `x-ogen-validate`, for example:

```yaml
components:
  schemas:
    Product:
      type: object
      properties:
        name:
          type: string
          x-ogen-validate:
            minWords: 2
        tags:
          type: array
          items:
            type: string
          x-ogen-validate:
            uniqueItems: true
        metadata:
          type: object
          additionalProperties: true
          x-ogen-validate:
            fieldCount:
              min: 1
              max: 10
```

Custom validators must be registered before validation is performed:

```go
import "github.com/ogen-go/ogen/validate"

// Register validators
validate.RegisterValidator("minWords", func(value any, params any) error {
    // ... validate minimum word count
})

validate.RegisterValidator("uniqueItems", func(value any, params any) error {
    // ... validate array has no duplicate items
})

validate.RegisterValidator("fieldCount", func(value any, params any) error {
    // ... validate object field count within min/max range
})
```

### Operation groups

Optionally, operations can be grouped so a handler interface will be generated for each group of operations.
This is useful for organizing operations for large APIs.

The group for operations on a path or individual operations can be specified by `x-ogen-operation-group`, for example:

```yaml
paths:
  /images:
    x-ogen-operation-group: Images
    get:
      operationId: listImages
      ...
  /images/{imageID}:
    x-ogen-operation-group: Images
    get:
      operationId: getImageByID
      ...
  /users:
    x-ogen-operation-group: Users
    get:
      operationId: listUsers
      ...
```

The generated handler interfaces look like this:

```go
// x-ogen-operation-group: Images
type ImagesHandler interface {
    ListImages(ctx context.Context, req *ListImagesRequest) (*ListImagesResponse, error)
    GetImageByID(ctx context.Context, req *GetImagesByIDRequest) (*GetImagesByIDResponse, error)
}

// x-ogen-operation-group: Users
type UsersHandler interface {
    ListUsers(ctx context.Context, req *ListUsersRequest) (*ListUsersResponse, error)
}

type Handler interface {
    ImagesHandler
    UsersHandler
    // All un-grouped operations will be on this interface
}
```

## JSON

Code generation provides very efficient and flexible encoding and decoding of json:

```go
// Decode decodes Error from json.
func (s *Error) Decode(d *jx.Decoder) error {
	if s == nil {
		return errors.New("invalid: unable to decode Error to nil")
	}
	return d.ObjBytes(func(d *jx.Decoder, k []byte) error {
		switch string(k) {
		case "code":
			if err := func() error {
				v, err := d.Int64()
				s.Code = int64(v)
				if err != nil {
					return err
				}
				return nil
			}(); err != nil {
				return errors.Wrap(err, "decode field \"code\"")
			}
		case "message":
			if err := func() error {
				v, err := d.Str()
				s.Message = string(v)
				if err != nil {
					return err
				}
				return nil
			}(); err != nil {
				return errors.Wrap(err, "decode field \"message\"")
			}
		default:
			return d.Skip()
		}
		return nil
	})
}
```

# Links

- [Getting started](https://ogen.dev/docs/intro)
- [Sample project](https://github.com/ogen-go/example)
- [Security policy](https://github.com/ogen-go/ogen/blob/-/SECURITY.md)
- [Telegram chat `@ogen_dev`](https://t.me/ogen_dev)
 readmeEtag: '"d2cfcfa29e520817c8a0198953e84c50008ca2e7"' readmeLastModified: Tue, 03 Feb 2026 13:42:13 GMT repositoryId: 372032505 description: OpenAPI v3 code generator for go created: '2021-05-29T17:25:48Z' updated: '2026-02-05T18:13:38Z' language: Go archived: false stars: 1993 watchers: 16 forks: 155 owner: ogen-go logo: https://avatars.githubusercontent.com/u/85122432?v=4 license: Apache-2.0 repoEtag: '"b4b48d4f1ef6a7f4987ddc63235fffb5be544d7a5e538f77b51e1db6f1955914"' repoLastModified: Thu, 05 Feb 2026 18:13:38 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/openapitools/openapi-generator-cli v3: true repositoryMetadata: base64Readme: >- # @openapitools/openapi-generator-cli

[![Join the Slack chat room](https://img.shields.io/badge/Slack-Join%20the%20chat%20room-orange)](https://join.slack.com/t/openapi-generator/shared_invite/zt-12jxxd7p2-XUeQM~4pzsU9x~eGLQqX2g)

![Build](https://github.com/OpenAPITools/openapi-generator-cli/workflows/Build/badge.svg)
[![Renovate enabled](https://img.shields.io/badge/renovate-enabled-brightgreen.svg)](https://renovateapp.com/)
[![semantic-release](https://img.shields.io/badge/%20%20%F0%9F%93%A6%F0%9F%9A%80-semantic--release-e10079.svg)](https://github.com/semantic-release/semantic-release)

OpenAPI Generator allows generation of API client libraries (SDK generation), server stubs, documentation and 
configuration automatically given an OpenAPI Spec (both 2.0 and 3.0 are supported). Please see
[OpenAPITools/openapi-generator](https://github.com/OpenAPITools/openapi-generator).

The OpenAPI Generator is a Java project. `openapi-generator-cli` will download the appropriate JAR file and invoke the `java` executable to run the OpenAPI Generator. You must have the `java` binary executable available on your `PATH` for this to work. (JDK 11 is the minimal version supported. To install OpenJDK, please visit https://adoptium.net/)

If you find this tool useful, please consider sponsoring this project financially via https://opencollective.com/openapi_generator or directly to [Kay Schecker](https://github.com/sponsors/kay-schecker) (the author of this tool) :pray:

---

## Version 2.x.x

#### [update] The command has been renamed

You need to execute `openapi-generator-cli` instead of `openapi-generator` from now on.

#### [added] [semver](https://semver.org/) support! 🎉

<p align="center"><img src="https://github.com/OpenAPITools/openapi-generator-cli/blob/master/img/vm.gif?raw=true"/></p>

To make that happen, a version management was added to the package.
The first time you run the command `openapi-generator-cli` the last stable version 
of [OpenAPITools/openapi-generator](https://github.com/OpenAPITools/openapi-generator) is downloaded by default. 

That version is saved in the file *openapitools.json*. Therefore, you should include this file in your version control, 
to ensure that the correct version is being used next time you call the command.

If you would like to use a different version of the [OpenAPITools/openapi-generator](https://github.com/OpenAPITools/openapi-generator), 
you could change it by using one of the following commands:
 
- `openapi-generator-cli version-manager list` 
- `openapi-generator-cli version-manager set  <versionTags...>`

#### [added] generator config

You will now be able to configure the code generation in *openapitools.json*. 
This makes it more convenient to generate code for every file that matches the given glob expression.
For more information, [please check out the configuration documentation below](#configuration).

## Installation

### Locally (recommended)

```sh
npm install @openapitools/openapi-generator-cli
```

or using yarn

```sh
yarn add @openapitools/openapi-generator-cli
```

After the installation has finished you can run `npx openapi-generator-cli` or add a script like this:

```json
{
  "name": "my-cool-package",
  "version": "0.0.0",
  "scripts": {
    "my-awesome-script-name": "openapi-generator-cli generate -i docs/openapi.yaml -g typescript-angular -o generated-sources/openapi --additional-properties=ngVersion=6.1.7,npmName=restClient,supportsES6=true,npmVersion=6.9.0,withInterfaces=true"
  }
}
```

Note the whitespace sensitivity when using multiple additional-properties:

```text
--additional-properties=ngVersion=6.1.7,npmName=restClient,supportsES6=true,npmVersion=6.9.0,withInterfaces=true
```

### Globally

```sh
npm install -g @openapitools/openapi-generator-cli
```

or using yarn

```sh
yarn global add @openapitools/openapi-generator-cli
```

After the installation has finished you can run `openapi-generator-cli`

## Usage

Mac/Linux:
```
openapi-generator-cli generate -g ruby -i https://raw.githubusercontent.com/OpenAPITools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml -o /var/tmp/ruby-client
```

Windows:
```
openapi-generator-cli generate -g ruby -i https://raw.githubusercontent.com/OpenAPITools/openapi-generator/master/modules/openapi-generator/src/test/resources/3_0/petstore.yaml -o C:\temp\ruby-client
```

## Configuration

If you have installed the package locally and executed the command `openapi-generator-cli` at least once, 
you will find a new file called *openapitools.json* along with the *package.json*. **Please add this file to your VCS.** 

Initially the file has the following content:

```json5
{
  "$schema": "node_modules/@openapitools/openapi-generator-cli/config.schema.json",
  "spaces": 2,
  "generator-cli": {
    "version": "7.8.0" // or the current latest version ;)
  }
}
```

This configuration indicates the following:

- the json file shall be formatted using **2 spaces**
- the jar files shall be downloaded to *./my/custom/storage/dir*
- the generator-cli version 7.8.0 is used

Further it is also possible to configure generators, for example:

```json5
{
  "$schema": "node_modules/@openapitools/openapi-generator-cli/config.schema.json",
  "spaces": 2,
  "generator-cli": {
    "version": "7.8.0",
    "storageDir": "~/my/custom/storage/dir", // optional
    "generators": { // optional
      "v2.0": { // any name you like (just printed to the console log or reference it using --generator-key) 
        "generatorName": "typescript-angular",
        "output": "#{cwd}/output/v2.0/#{ext}/#{name}",
        "glob": "examples/v2.0/{json,yaml}/*.{json,yaml}",
        "additionalProperties": {
          "ngVersion": "6.1.7",
          "npmName": "restClient",
          "supportsES6": "true",
          "npmVersion": "6.9.0",
          "withInterfaces": true
        }
      },
      "v3.0": { // any name you like (just printed to the console log or reference it using --generator-key) 
        "generatorName": "typescript-fetch",
        "output": "#{cwd}/output/v3.0/#{ext}/#{name}",
        "glob": "examples/v3.0/petstore.{json,yaml}"
      }
    }
  }
}
```

If `openapi-generator-cli generate` is called without further arguments, then the configuration 
is automatically used to generate your code. 🎉


##### Available placeholders
     
| placeholder | description                                                   | example                                               |
|-------------|---------------------------------------------------------------|-------------------------------------------------------|
| name        | just file name                                                | auth                                                  |
| Name        | just file name, but starting with a capital letter            | Auth                                                  |
| cwd         | the current cwd                                               | /Users/some-user/projects/some-project                |
| base        | file name and extension                                       | auth.yaml                                             |
| path        | full path and filename                                        | /Users/some-user/projects/some-project/docs/auth.yaml |
| dir         | path without the filename                                     | /Users/some-user/projects/some-project/docs           |
| relDir      | directory name of file relative to the glob provided          | docs                                                  |
| relPath     | file name and extension of file relative to the glob provided | docs/auth.yaml                                        |
| ext         | just file extension                                           | yaml                                                  |
| env.<name>  | environment variable (use ${env.name} syntax)                 |                                                       |

### Using custom / private maven registry 

If you're using a private maven registry you can configure the `downloadUrl` and `queryUrl` like this:

```json
{
  "$schema": "node_modules/@openapitools/openapi-generator-cli/config.schema.json",
  "spaces": 2,
  "generator-cli": {
    "version": "7.8.0",
    "repository": {
      "queryUrl": "https://private.maven.intern/solrsearch/select?q=g:${group.id}+AND+a:${artifact.id}&core=gav&start=0&rows=200",
      "downloadUrl": "https://private.maven.intern/maven2/${groupId}/${artifactId}/${versionName}/${artifactId}-${versionName}.jar"
    }
  }
}
```

If the `version` property param is set it is not necessary to configure the `queryUrl`.

`queryUrl` and `downloadUrl` can use the following placeholders:

| placeholder | description                                        |
|-------------|----------------------------------------------------|
| groupId     | maven groupId where '.' has been replaced with /   |
| artifactId  | maven artifactId where '.' has been replace with / |
| versionName | maven version (only for downloadUrl)               |
| group.id    | maven groupId                                      |
| artifact.id | maven artifactId                                   |
| env.<name>  | environment variable name                          |

### Use locally built JAR
In order to use a locally built jar of the generator CLI, you can copy the jar from your local build (i.e. if you were to `build` the [OpenAPITools/openapi-generator](https://github.com/OpenAPITools/openapi-generator) repository it would be in `~/openapi-generator/modules/openapi-generator-cli/target/openapi-generator-cli.jar`) into `./node_modules/@openapitools/openapi-generator-cli/versions/` and change the `version` in the `openapitools.json` file to the base name of the jar file.
E.g.:
```sh
cd openapi-generator
./mvnw clean package
cp ./modules/openapi-generator-cli/target/openapi-generator-cli.jar /your/project/node_modules/@openapitools/openapi-generator-cli/versions/my-local-snapshot.jar
```
and then:
```json
{
  "$schema": "./node_modules/@openapitools/openapi-generator-cli/config.schema.json",
  "spaces": 2,
  "generator-cli": {
    "version": "my-local-snapshot"
  }
}
```

### Use nightly `SNAPSHOT` build
Change your `openapitools.json` to:

```json
{
  "$schema": "node_modules/@openapitools/openapi-generator-cli/config.schema.json",
  "spaces": 2,
  "generator-cli": {
    "version": "7.17.0-20251003.020930-8",
    "repository": {
      "downloadUrl": "https://central.sonatype.com/repository/maven-snapshots/org/openapitools/openapi-generator-cli/7.17.0-SNAPSHOT/openapi-generator-cli-${version}.jar"
    }
  }
}
```

Example is with a snapshot of `7.17.0`, please change the `version` and `downloadUrl` accordingly.
You can find the published snapshots in the build log of the [Publish to Maven Central GitHub workflow](https://github.com/OpenAPITools/openapi-generator/actions/workflows/maven-release.yml) in OpenAPI Generator repo, e.g.

```
[INFO] Uploading to central: https://central.sonatype.com/repository/maven-snapshots/org/openapitools/openapi-generator-cli/7.17.0-SNAPSHOT/openapi-generator-cli-7.17.0-20251003.020930-8.jar
```

## Run specific generators

| cmd                                                      | v3.0 runs | v2.0 runs |
|----------------------------------------------------------|-----------|-----------|
| openapi-generator-cli generate --generator-key v3.0      | yes       | no        |
| openapi-generator-cli generate --generator-key v3.0 v2.0 | yes       | yes       |
| openapi-generator-cli generate --generator-key foo       | no        | no        |

## Use Docker instead of running java locally

```json
{
  "$schema": "node_modules/@openapitools/openapi-generator-cli/config.schema.json",
  "spaces": 2,
  "generator-cli": {
    "useDocker": true
  }
}
```
If `useDocker` option is used, the spec file gets mounted to path `/local/<your-spec-file-location>` within container. So, if you would configure spec file as 
`-i openapi/my-api.yml` if running locally, with useDocker option set, you would have to configure it like this: `-i /local/openapi/my-api.yml`.

## Custom Generators

Custom generators can be used by passing the `--custom-generator=/my/custom-generator.jar` argument.

## Bypassing maven repo search

To bypass maven repo search for all stable versions (as central.sonatype.com or search.maven.org could be down due to various reasons), one can set the environment variable `OPENAPI_GENERATOR_CLI_SEARCH_URL` to `DEFAULT`:

Linux: `export OPENAPI_GENERATOR_CLI_SEARCH_URL=DEFAULT`

DOS: `set OPENAPI_GENERATOR_CLI_SEARCH_URL=DEFAULT`

## Further Documentation

Please refer to the [official openapi-generator docs](https://github.com/OpenAPITools/openapi-generator#3---usage) for
more information about the possible arguments and a detailed usage manual of the command line interface.

## Install previous version

```sh
npm install @openapitools/openapi-generator-cli@previous
npm i @openapitools/openapi-generator-cli@1.0.18-4.3.1
```

or using yarn

```sh
yarn add @openapitools/openapi-generator-cli@previous
yarn add @openapitools/openapi-generator-cli@1.0.18-4.3.1
```

## You like the package?

Please leave a star.




 readmeEtag: '"14acae4cdb17f6bf8b12341944f43cb9aadc4015"' readmeLastModified: Tue, 30 Dec 2025 16:15:11 GMT repositoryId: 154817001 description: >- A node package wrapper for https://github.com/OpenAPITools/openapi-generator created: '2018-10-26T10:21:23Z' updated: '2026-02-06T04:59:35Z' language: TypeScript archived: false stars: 1882 watchers: 17 forks: 192 owner: OpenAPITools logo: https://avatars.githubusercontent.com/u/37325267?v=4 license: Apache-2.0 repoEtag: '"4f9d39455d12505ba9881c17362ea07d5f72dc58bbc5e5491abaa36de471c8a9"' repoLastModified: Fri, 06 Feb 2026 04:59:35 GMT foundInMaster: true category: - Code Generators - Parsers id: 8f568f0b10837fef38571b68999adc0f - source: openapi3 tags name: OpenAPI-gui homepage: https://github.com/Mermade/openapi-gui language: Node.js source_description: GUI / visual editor for creating and editing OpenAPI definitions category: - Editors - Server Implementations repository: https://github.com/mermade/openapi-gui v3: true repositoryMetadata: base64Readme: >- T3BlbkFQSS1HVUkKPT09PT09PT09PT0KCk9wZW5BUEktR1VJIGlzIGEgR1VJIGZvciBjcmVhdGluZyBhbmQgZWRpdGluZyBPcGVuQVBJIHZlcnNpb24gMy4wLnggSlNPTi9ZQU1MIGRlZmluaXRpb25zLiBJbiBpdHMgY3VycmVudCBmb3JtIGl0IGlzIG1vc3QgdXNlZnVsIGFzIGEgdG9vbCBmb3Igc3RhcnRpbmcgb2ZmIGFuZCBlZGl0aW5nIHNpbXBsZSBPcGVuQVBJIGRlZmluaXRpb25zLiBJbXBvcnRlZCBPcGVuQVBJIDIuMCBkZWZpbml0aW9ucyBhcmUgYXV0b21hdGljYWxseSBjb252ZXJ0ZWQgdG8gdjMuMC4KCiFbU2NyZWVuc2hvdF0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL01lcm1hZGUvb3BlbmFwaS1ndWkvbWFpbi9pbWcvc2NyZWVuc2hvdC5wbmcpCgpGb3IgdGhlIHByZXZpb3VzIFN3YWdnZXIgLyBPcGVuQVBJIDIuMC1vbmx5IHZlcnNpb24gc2VlIFtoZXJlXShodHRwczovL21pa2VyYWxwaHNvbi5naXRodWIuaW8vb3BlbmFwaS1ndWkpLiBUaGF0IHZlcnNpb24gaXMgY3VycmVudGx5ICoqdW5tYWludGFpbmVkKiogYXBhcnQgZnJvbSBzZWN1cml0eSBmaXhlcy4KClRoaXMgcHJvamVjdCB3YXMgaW5pdGlhbGx5IGEgZm9yayBvZiBbRGFyeWwgS3VobidzIElPRG9jdG9yXShodHRwczovL2dpdGh1Yi5jb20vZGFycnlsa3Vobi9pb2RvY3Rvci90cmVlL2FuZ3VsYXItcG9ydCksIHdoaWNoIGluIHR1cm4gd2FzIGluc3BpcmVkIGJ5IFtJT0RvY3RvciBieSBCcmFuZG9uIFdlc3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9icmFuZG9ubXdlc3QvaW9kb2N0b3IpIHdoaWNoIHdhcyB3cml0dGVuIGluIFJ1YnkuIFRoZSBjb21wbGV0ZSBoaXN0b3J5IG9mIHRoZSBwcm9qZWN0IGlzIG1haW50YWluZWQgb24gR2l0SHViLgoKRGVzY3JpcHRpb24KLS0tLS0tLS0tLS0KIyMjIEhvdyBJdCBXb3JrcwoKU2VsZWN0IGFuIGV4aXN0aW5nIE9wZW5BUEkgMi4wIG9yIDMuMC54IGRlZmluaXRpb24gdG8gdXBsb2FkLCBvciBjcmVhdGUgYSBuZXcgZGVmaW5pdGlvbiBhbmQgc3RhcnQgYWRkaW5nIFBhdGhzLCBPcGVyYXRpb25zLCBhbmQgUGFyYW1ldGVycy4gV2hlbiBhbiBleGlzdGluZyBkZWZpbml0aW9uIGlzIHVzZWQsIGl0IGlzIHBhcnNlZCBhbmQgZm9ybXMgZm9yIGVkaXRpbmcgZWFjaCBQYXRoLCBPcGVyYXRpb24gYW5kIFBhcmFtZXRlciB3aWxsIGJlIGNyZWF0ZWQuCgpZb3UgY2FuIGxvYWQgYW4gZXhpc3RpbmcgZGVmaW5pdGlvbiBieSBhcHBlbmRpbmcgYSBgP3VybD1gIHF1ZXJ5IHBhcmFtZXRlciB0byB0aGUgaW5pdGlhbCBzdGFydCBwYWdlIHdpdGggdGhlIHZhbHVlIGJlaW5nIHRoZSBVUkwgdG8gdGhlIGRlZmluaXRpb24geW91IHdpc2ggdG8gbG9hZC4gV2hlbiB1c2luZyBEb2NrZXIsIHlvdSBjYW4gc2V0IHRoZSB2YWx1ZSBvZiB0aGUgYHVybGAgcXVlcnkgcGFyYW1ldGVyIHRvIGAlMmZzZXJ2ZWAgdG8gcGljayB1cCB0aGUgZGVmaW5pdGlvbiB5b3UgcGFzc2VkIGluIG9uIHRoZSBEb2NrZXIgY29tbWFuZGxpbmUuCgpDbGljayBhbiBpdGVtIGZyb20gdGhlIG1lbnUgb24gdGhlIGxlZnQgdG8gYmVnaW4gZWRpdGluZy4gVmlldyB0aGUgSlNPTi9ZQU1MIG91dHB1dCBhdCBhbnkgdGltZSBieSBzZWxlY3Rpbmcgb25lIG9mIHRoZSAiRXhwb3J0IiB0YWJzLiBXaGVuIGZpbmlzaGVkLCBkb3dubG9hZCB0aGUgb3V0cHV0IHRvIHNhdmUgaXQgbG9jYWxseSBvciBjb3B5IGl0IHlvdXIgY2xpcGJvYXJkLiBPcGVuQVBJLUdVSSBvbmx5IHN0b3JlcyBvbmUgZGVmaW5pdGlvbiBhdCBhIHRpbWUsIGFuZCB0aGlzIGlzIGluIHlvdXIgYnJvd3NlcidzIGxvY2FsLXN0b3JhZ2UuIE1ha2Ugc3VyZSB5b3Ugc2F2ZSB5b3VyIEpTT04vWUFNTCBvdXRwdXQgbG9jYWxseS4KCkJlZm9yZSBwZXJmb3JtaW5nIGEgZGVzdHJ1Y3RpdmUgYWN0aW9uLCBPcGVuQVBJLUdVSSBzYXZlcyB0aGUgY3VycmVudCBzdGF0ZSBvZiB0aGUgZGVmaW5pdGlvbi4gQXQgYWxsIG90aGVyIHRpbWVzIHlvdSBtdXN0IHJlbWVtYmVyIHRvIHNlbGVjdCBTYXZlIG1hbnVhbGx5LgoKIyMjIFRlY2hub2xvZ3kKCk9wZW5BUEktR1VJIHJ1bnMgZW50aXJlbHkgY2xpZW50LXNpZGUgdXNpbmcgYSBudW1iZXIgb2YgSmF2YXNjcmlwdCBmcmFtZXdvcmtzIGluY2x1ZGluZyBbVnVlLkpTXShodHRwczovL3Z1ZWpzLm9yZy8pLCBbalF1ZXJ5XShodHRwczovL2pxdWVyeS5jb20vKSBhbmQgW0J1bG1hXShodHRwOi8vYnVsbWEuaW8vKSBmb3IgQ1NTLgoKVG8gZ2V0IHRoZSBhcHAgdXAgYW5kIHJ1bm5pbmcganVzdCBicm93c2UgdG8gW3RoZSBsaXZlIHZlcnNpb24gb24gR2l0SHViIHBhZ2VzXShodHRwczovL21lcm1hZGUuZ2l0aHViLmlvL29wZW5hcGktZ3VpKSwgZGVwbG95IGEgY2xvbmUgdG8gR2l0SHViIHBhZ2VzLCBkZXBsb3kgdG8gSGVyb2t1IHVzaW5nIHRoZSBidXR0b24gYmVsb3csIG9yIGNsb25lIHRoZSByZXBvIGFuZCBwb2ludCBhIGJyb3dzZXIgYXQgYGluZGV4Lmh0bWxgIG9yIGhvc3QgaXQgeW91cnNlbGYgLSBjb3VsZG4ndCBiZSBzaW1wbGVyLiBNb3JlIFt0ZWNobmljYWwgaW5mb3JtYXRpb24gaGVyZV0oZG9jcy90ZWNobmljYWwubWQpLgoKWW91IG9ubHkgbmVlZCB0byBgbnBtIGluc3RhbGxgIHRoZSBOb2RlLmpzIG1vZHVsZXMgaWYgeW91IHdpc2ggdG8gdXNlIHRoZSBgb3BlbmFwaS1ndWlgIGVtYmVkZGVkIHdlYiBzZXJ2ZXIgKGkuZS4gbm90IGlmIHlvdSBhcmUgcnVubmluZyB5b3VyIG93biB3ZWItc2VydmVyKSwgb3RoZXJ3aXNlIHRoZXkgYXJlIG9ubHkgdGhlcmUgZm9yIFBhYVMgZGVwbG95bWVudHMuCgpbIVtEZXBsb3ldKGh0dHBzOi8vd3d3Lmhlcm9rdWNkbi5jb20vZGVwbG95L2J1dHRvbi5zdmcpXShodHRwczovL2hlcm9rdS5jb20vZGVwbG95KQoKIyMjIyBDTEkgb3B0aW9ucwoKYGBgCi1kLCAtLWRlZmluaXRpb24gICAgc2VydmUgdGhlIGdpdmVuIE9BUyBkZWZpbml0aW9uCi1sLCAtLWxhdW5jaCAgICAgICAgc3RhcnQgYSB3ZWItYnJvd3NlciBwb2ludGluZyB0byB0aGUgR1VJCi1wLCAtLXBvcnQgICAgICAgICAgc3BlY2lmeSB0aGUgcG9ydCB0byBydW4gb24sIGRlZmF1bHRzIHRvICRQT1JUIG9yIDMwMDAKLXcsIC0td3JpdGUgICAgICAgICBlbmFibGUgd3JpdGluZyBiYWNrIHRvIHRoZSBzb3VyY2UgZGVmaW5pdGlvbgpgYGAKCiMjIyMgUnVubmluZyB3aXRoIERvY2tlcgoKSWYgeW91IGRvbid0IGhhdmUgYSBsb2NhbCBOb2RlIGRldmVsb3BtZW50IGVudmlyb25tZW50LCBvciBpZiB5b3Ugd291bGQgcHJlZmVyIHRvIHJ1biBPcGVuQVBJLUdVSSBpbiBhIERvY2tlciBjb250YWluZXIsIHlvdSBjYW4gZG8gaW4gd2l0aCBhIGZldyBzaW1wbGUgc3RlcHM6CgoxLiBDbG9uZSB0aGUgcmVwb3NpdG9yeS4KMi4gUnVuIGBkb2NrZXIgYnVpbGQgLXQgbWVybWFkZS9vcGVuYXBpLWd1aSAuYCB0byBidWlsZCB0aGUgRG9ja2VyIGltYWdlIChgbWVybWFkZS9vcGVuYXBpLWd1aWApCjMuIFJ1biBgZG9ja2VyIHJ1biAtLW5hbWUgb3BlbmFwaS1ndWkgLXAgODA4MDozMDAwIC1kIG1lcm1hZGUvb3BlbmFwaS1ndWlgIHRvIHJ1biB0aGUgc2VydmVyIG9uIHBvcnQgODA4MC4KNC4gTmF2aWdhdGUgdG8gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MGAgaW4geW91ciBmYXZvcml0ZSBicm93c2VyLgo1LiBXaGVuIHlvdSdyZSBkb25lLCBzaHV0IGRvd24gdGhlIHNlcnZlciBieSBydW5uaW5nIGBkb2NrZXIgc3RvcCBvcGVuYXBpLWd1aSAmJiBkb2NrZXIgcm0gb3BlbmFwaS1ndWlgCgpPciB5b3UgY2FuIHB1bGwgdGhlIHByZS1idWlsdCBEb2NrZXIgaW1hZ2U6CgoqIGBkb2NrZXIgcHVsbCBtZXJtYWRlL29wZW5hcGktZ3VpYAoKIyMjIExpbWl0YXRpb25zCgoqIE9wZW5BUEktR1VJIHdpbGwgZGUtcmVmZXJlbmNlIHNoYXJlZCBwYXJhbWV0ZXJzLgoqIFRoZSBkZWZpbml0aW9uIG11c3QgYmUgc2VsZi1jb250YWluZWQgd2l0aCBubyBleHRlcm5hbCBgJHJlZmBzLiAqKlRoaXMgaXMgbGlrZWx5IHRvIGJlIHJlc29sdmVkIChoby1obykgc29vbioqLgoqIEVkaXRpbmcgYSByZXNwb25zZSAvIGV4YW1wbGUgLyBib2R5IHNjaGVtYSB3aWxsIGRlcmVmZXJlbmNlIGl0LgoqIE9wZW5BUEktR1VJIHdpbGwgbm90IGFsd2F5cyBwcmVzZXJ2ZSB2ZW5kb3ItZXh0ZW5zaW9ucywgZS5nLiBpZiBhIHBhcmFtZXRlciBpcyBkZWxldGVkIGFuZCByZWNyZWF0ZWQuCiogT3BlbkFQSS1HVUkgd2lsbCBub3QgcHJlc2VydmUgY29tbWVudHMgZnJvbSBkZWZpbml0aW9ucyBpbXBvcnRlZCBpbiBZQU1MIGZvcm1hdC4KClRPRE8KLS0tLQoKKiBTZWUgdGhlIFtUT0RPIGxpc3RdKC9kb2NzL1RPRE8ubWQpIGFuZCB0aGUgY3VycmVudCBzdGF0dXMgb2YgW09wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gc3VwcG9ydF0oZG9jcy9vcGVuYXBpMy1zdXBwb3J0Lm1kKS4KCg== readmeEtag: '"fe9390b576bcf3a1e239f9399ce0c894964d859d"' readmeLastModified: Sun, 15 Oct 2023 11:54:20 GMT repositoryId: 71986592 description: GUI / visual editor for creating and editing OpenAPI / Swagger definitions created: '2016-10-26T09:15:55Z' updated: '2026-01-31T22:28:37Z' language: JavaScript archived: false stars: 1479 watchers: 40 forks: 172 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: MIT repoEtag: '"c954fbaf5b94260fa4b954f8387cf7961a720a9ae839eaad9a9c112d034f7016"' repoLastModified: Sat, 31 Jan 2026 22:28:37 GMT foundInMaster: true id: 71f1a528dc0a6366fb580d1b8c703ea8 - source: openapi3 tags repository: https://github.com/marshmallow-code/apispec v3: true repositoryMetadata: base64Readme: >- *******
apispec
*******

|pypi| |build-status| |docs| |marshmallow-support| |openapi|

.. |pypi| image:: https://badgen.net/pypi/v/apispec
    :target: https://pypi.org/project/apispec/
    :alt: PyPI package

.. |build-status| image:: https://github.com/marshmallow-code/apispec/actions/workflows/build-release.yml/badge.svg
    :target: https://github.com/marshmallow-code/webargs/actions/workflows/build-release.yml
    :alt: Build status

.. |docs| image:: https://readthedocs.org/projects/apispec/badge/
   :target: https://apispec.readthedocs.io/
   :alt: Documentation

.. |marshmallow-support| image:: https://badgen.net/badge/marshmallow/3,4?list=1
    :target: https://marshmallow.readthedocs.io/en/latest/upgrading.html
    :alt: marshmallow 3|4 compatible

.. |openapi| image:: https://badgen.net/badge/OAS/2,3?list=1&color=cyan
    :target: https://github.com/OAI/OpenAPI-Specification
    :alt: OpenAPI Specification 2/3 compatible


A pluggable API specification generator. Currently supports the `OpenAPI Specification <https://github.com/OAI/OpenAPI-Specification>`_ (f.k.a. the Swagger specification).

Features
========

- Supports the OpenAPI Specification (versions 2 and 3)
- Framework-agnostic
- Built-in support for `marshmallow <https://marshmallow.readthedocs.io/>`_
- Utilities for parsing docstrings

Installation
============

::

    $ pip install -U apispec

When using the marshmallow plugin, ensure a compatible marshmallow version is used: ::

    $ pip install -U apispec[marshmallow]

Example Application
===================

.. code-block:: python

    from apispec import APISpec
    from apispec.ext.marshmallow import MarshmallowPlugin
    from apispec_webframeworks.flask import FlaskPlugin
    from flask import Flask
    from marshmallow import Schema, fields


    # Create an APISpec
    spec = APISpec(
        title="Swagger Petstore",
        version="1.0.0",
        openapi_version="3.0.2",
        plugins=[FlaskPlugin(), MarshmallowPlugin()],
    )


    # Optional marshmallow support
    class CategorySchema(Schema):
        id = fields.Int()
        name = fields.Str(required=True)


    class PetSchema(Schema):
        category = fields.List(fields.Nested(CategorySchema))
        name = fields.Str()


    # Optional security scheme support
    api_key_scheme = {"type": "apiKey", "in": "header", "name": "X-API-Key"}
    spec.components.security_scheme("ApiKeyAuth", api_key_scheme)


    # Optional Flask support
    app = Flask(__name__)


    @app.route("/random")
    def random_pet():
        """A cute furry animal endpoint.
        ---
        get:
          description: Get a random pet
          security:
            - ApiKeyAuth: []
          responses:
            200:
              content:
                application/json:
                  schema: PetSchema
        """
        pet = get_random_pet()
        return PetSchema().dump(pet)


    # Register the path and the entities within it
    with app.test_request_context():
        spec.path(view=random_pet)


Generated OpenAPI Spec
----------------------

.. code-block:: python

    import json

    print(json.dumps(spec.to_dict(), indent=2))
    # {
    #   "paths": {
    #     "/random": {
    #       "get": {
    #         "description": "Get a random pet",
    #         "security": [
    #           {
    #             "ApiKeyAuth": []
    #           }
    #         ],
    #         "responses": {
    #           "200": {
    #             "content": {
    #               "application/json": {
    #                 "schema": {
    #                   "$ref": "#/components/schemas/Pet"
    #                 }
    #               }
    #             }
    #           }
    #         }
    #       }
    #     }
    #   },
    #   "tags": [],
    #   "info": {
    #     "title": "Swagger Petstore",
    #     "version": "1.0.0"
    #   },
    #   "openapi": "3.0.2",
    #   "components": {
    #     "parameters": {},
    #     "responses": {},
    #     "schemas": {
    #       "Category": {
    #         "type": "object",
    #         "properties": {
    #           "name": {
    #             "type": "string"
    #           },
    #           "id": {
    #             "type": "integer",
    #             "format": "int32"
    #           }
    #         },
    #         "required": [
    #           "name"
    #         ]
    #       },
    #       "Pet": {
    #         "type": "object",
    #         "properties": {
    #           "name": {
    #             "type": "string"
    #           },
    #           "category": {
    #             "type": "array",
    #             "items": {
    #               "$ref": "#/components/schemas/Category"
    #             }
    #           }
    #         }
    #       }
    #       "securitySchemes": {
    #          "ApiKeyAuth": {
    #            "type": "apiKey",
    #            "in": "header",
    #            "name": "X-API-Key"
    #         }
    #       }
    #     }
    #   }
    # }

    print(spec.to_yaml())
    # components:
    #   parameters: {}
    #   responses: {}
    #   schemas:
    #     Category:
    #       properties:
    #         id: {format: int32, type: integer}
    #         name: {type: string}
    #       required: [name]
    #       type: object
    #     Pet:
    #       properties:
    #         category:
    #           items: {$ref: '#/components/schemas/Category'}
    #           type: array
    #         name: {type: string}
    #       type: object
    #   securitySchemes:
    #     ApiKeyAuth:
    #       in: header
    #       name: X-API-KEY
    #       type: apiKey
    # info: {title: Swagger Petstore, version: 1.0.0}
    # openapi: 3.0.2
    # paths:
    #   /random:
    #     get:
    #       description: Get a random pet
    #       responses:
    #         200:
    #           content:
    #             application/json:
    #               schema: {$ref: '#/components/schemas/Pet'}
    #       security:
    #       - ApiKeyAuth: []
    # tags: []


Documentation
=============

Documentation is available at https://apispec.readthedocs.io/ .

Ecosystem
=========

A list of apispec-related libraries can be found at the GitHub wiki here:

https://github.com/marshmallow-code/apispec/wiki/Ecosystem

Support apispec
===============

apispec is maintained by a group of
`volunteers <https://apispec.readthedocs.io/en/latest/authors.html>`_.
If you'd like to support the future of the project, please consider
contributing to our Open Collective:

.. image:: https://opencollective.com/marshmallow/donate/button.png
    :target: https://opencollective.com/marshmallow
    :width: 200
    :alt: Donate to our collective

Professional Support
====================

Professionally-supported apispec is available through the
`Tidelift Subscription <https://tidelift.com/subscription/pkg/pypi-apispec?utm_source=pypi-apispec&utm_medium=referral&utm_campaign=readme>`_.

Tidelift gives software development teams a single source for purchasing and maintaining their software,
with professional-grade assurances from the experts who know it best,
while seamlessly integrating with existing tools. [`Get professional support`_]

.. _`Get professional support`: https://tidelift.com/subscription/pkg/pypi-apispec?utm_source=pypi-apispec&utm_medium=referral&utm_campaign=readme

.. image:: https://user-images.githubusercontent.com/2379650/45126032-50b69880-b13f-11e8-9c2c-abd16c433495.png
    :target: https://tidelift.com/subscription/pkg/pypi-apispec?utm_source=pypi-apispec&utm_medium=referral&utm_campaign=readme
    :alt: Get supported apispec with Tidelift

Security Contact Information
============================

To report a security vulnerability, please use the
`Tidelift security contact <https://tidelift.com/security>`_.
Tidelift will coordinate the fix and disclosure.

Project Links
=============

- Docs: https://apispec.readthedocs.io/
- Changelog: https://apispec.readthedocs.io/en/latest/changelog.html
- Contributing Guidelines: https://apispec.readthedocs.io/en/latest/contributing.html
- PyPI: https://pypi.python.org/pypi/apispec
- Issues: https://github.com/marshmallow-code/apispec/issues


License
=======

MIT licensed. See the bundled `LICENSE <https://github.com/marshmallow-code/apispec/blob/dev/LICENSE>`_ file for more details.
 readmeEtag: '"d472589f1773e1559028a9171e8e5d3442bc2395"' readmeLastModified: Tue, 07 Jan 2025 05:10:58 GMT repositoryId: 25413181 description: >- A pluggable API specification generator. Currently supports the OpenAPI Specification (f.k.a. the Swagger specification).. created: '2014-10-18T23:48:49Z' updated: '2026-02-03T17:39:01Z' language: Python archived: false stars: 1215 watchers: 21 forks: 185 owner: marshmallow-code logo: https://avatars.githubusercontent.com/u/10334301?v=4 license: MIT repoEtag: '"d4a983372c78f54baeaf27543d5ac80c7a8e94ee30b97ad37742dd75f8196f4a"' repoLastModified: Tue, 03 Feb 2026 17:39:01 GMT foundInMaster: true category: Parsers id: c1528764f05dd585ca47886dd6dabb86 - source: openapi3 tags repository: https://github.com/asteasolutions/zod-to-openapi v3: true id: 360d2acbe3dfa2f35dbc74551fda7829 repositoryMetadata: base64Readme: >- # Zod to OpenAPI

[![npm version](https://img.shields.io/npm/v/@asteasolutions/zod-to-openapi)](https://www.npmjs.com/package/@asteasolutions/zod-to-openapi)
[![npm downloads](https://img.shields.io/npm/dm/@asteasolutions/zod-to-openapi)](https://www.npmjs.com/package/@asteasolutions/zod-to-openapi)

> [!IMPORTANT]
> **For Zod v3 support, please use the v7.3.4 version. However keep in mind that we do not intend to actively support that version going forward** Install with: `npm install @asteasolutions/zod-to-openapi@7.3.4`

A library that uses [zod schemas](https://github.com/colinhacks/zod) to generate an Open API Swagger documentation.

1. [Purpose and quick example](#purpose-and-quick-example)
2. [Usage](#usage)
   1. [Installation](#installation)
   2. [The `openapi` method](#the-openapi-method)
   3. [The Registry](#the-registry)
   4. [The Generator](#the-generator)
   5. [Defining schemas](#defining-schemas)
   6. [Defining routes & webhooks](#defining-routes--webhooks)
   7. [Defining custom components](#defining-custom-components)
   8. [A full example](#a-full-example)
   9. [Adding it as part of your build](#adding-it-as-part-of-your-build)
   10. [Using schemas vs a registry](#using-schemas-vs-a-registry)
   11. [Generation options](#generation-options)
3. [Zod schema types](#zod-schema-types)
   1. [Supported types](#supported-types)
   2. [Unsupported types](#unsupported-types)
4. [Technologies](#technologies)

We keep a changelog as part of the [GitHub releases](https://github.com/asteasolutions/zod-to-openapi/releases).

## Purpose and quick example

We at [Astea Solutions](https://asteasolutions.com/) made this library because we use [zod](https://github.com/colinhacks/zod) for validation in our APIs and are tired of the duplication to also support a separate OpenAPI definition that must be kept in sync. Using `zod-to-openapi`, we generate OpenAPI definitions directly from our zod schemas, thus having a single source of truth.

Simply put, it turns this:

```ts
const UserSchema = z
  .object({
    id: z.string().openapi({ example: '1212121' }),
    name: z.string().openapi({ example: 'John Doe' }),
    age: z.number().openapi({ example: 42 }),
  })
  .openapi('User');

registry.registerPath({
  method: 'get',
  path: '/users/{id}',
  summary: 'Get a single user',
  request: {
    params: z.object({ id: z.string() }),
  },

  responses: {
    200: {
      description: 'Object with user data.',
      content: {
        'application/json': {
          schema: UserSchema,
        },
      },
    },
  },
});
```

into this:

```yaml
components:
  schemas:
    User:
      type: object
      properties:
        id:
          type: string
          example: '1212121'
        name:
          type: string
          example: John Doe
        age:
          type: number
          example: 42
      required:
        - id
        - name
        - age

/users/{id}:
  get:
    summary: Get a single user
    parameters:
      - in: path
        name: id
        schema:
          type: string
        required: true
    responses:
      '200':
        description: Object with user data
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/User'
```

and you can still use `UserSchema` and the `request.params` object to validate the input of your API.

## Usage

### Installation

```shell
npm install @asteasolutions/zod-to-openapi
# or
yarn add @asteasolutions/zod-to-openapi
```

### The `openapi` method

To keep openapi definitions natural, we add an `openapi` method to all Zod objects. Its idea is to provide a convenient way to provide OpenApi specific data.
It has three overloads:

1. `.openapi({ [key]: value })` - this way we can specify any OpenApi fields. For example `z.number().openapi({ example: 3 })` would add `example: 3` to the generated schema.
2. `.openapi("<schema-name>")` - this way we specify that the underlying zod schema should be "registered" i.e added into `components/schemas` with the provided `<schema-name>`
3. `.openapi("<schema-name>", { [key]: value })` - this unites the two use cases above so that we can specify both a registration `<schema-name>` and additional metadata

For this to work, you need to call `extendZodWithOpenApi` once in your project.

This should be done only once in a common-entrypoint file of your project (for example an `index.ts`/`app.ts`). If you're using tree-shaking with Webpack, mark that file as having side-effects.

It can be bit tricky to achieve this in your codebase, because *require* is synchronous and *import* is a async.

#### Using zod's .meta
Starting from v8 (and zod v4) you can also use zod's .meta to provide metadata and we will read it accordingly.

With zod's new option for generating JSON schemas and maintaining registries we've added a pretty much seamless support for all metadata information coming from `.meta` calls as if that was metadata passed into `.openapi`.

So the following 2 schemas produce exactly the same results:
```ts
const schema = z
  .string()
  .openapi('Schema', { description: 'Name of the user', example: 'Test' });

const schema2 = z
  .string()
  .meta({ id: 'Schema2', description: 'Name of the user', example: 'Test' });
```

> Note: This also means that you unless you are using some of our more complicated scenarios you could even generate a schema without using `extendZodWithOpenApi` in your codebase and only rely on `.meta` to provide additional metadata information and schema names (using the `id` property).

#### Scenarios that require using `extendZodWithOpenApi` and `.openapi`
1. When extending registered schemas that are both registered and want the extended one to use `anyOf` i.e:

```ts
const schema = z.object({ name: z.string() }).openapi('Schema');

const schema2 = schema.extend({ age: z.number() }).openapi('Schema2'); // this one would have anyOf and a reference to the first one
```
2. Defining parameter metadata. So for example when doing:
```ts
registry.registerPath({
  // ...
  request: {
    query: z.object({
      name: z.string().openapi({
        description: 'Schema level description',
        param: { description: 'Param level description' },
      }),
    }),
  },
});
```

the result would be:
```ts
  "parameters": [
      {
        "schema": {
          "type": "string",
          "description": "Schema level description" // comes directly from description
        },
        "required": true,
        "description": "Param level description", // comes from param.description
        "name": "name",
        "in": "query"
      }
  ],
```


### The basic idea

```ts
import { extendZodWithOpenApi } from '@asteasolutions/zod-to-openapi';
import { z } from 'zod';

extendZodWithOpenApi(z);

// We can now use `.openapi()` to specify OpenAPI metadata
z.string().openapi({ description: 'Some string' });
```

### Example 1: Calling the openapi-extension using tsx

```
//zod-extend.ts

import { extendZodWithOpenApi } from '@asteasolutions/zod-to-openapi';
import { z } from 'zod';

extendZodWithOpenApi(z);

// package.json

  "scripts": {
    "start": "tsx --import ./zod-extend.ts ./index.ts",
```

### Example 2 - require-syntax

```
import { extendZodWithOpenApi } from '@asteasolutions/zod-to-openapi';
import { z } from 'zod';

extendZodWithOpenApi(z);

const { startServer } = require('./server/start');
startServer();
```


### The Registry

The `OpenAPIRegistry` is a utility that can be used to collect definitions which would later be passed to a `OpenApiGeneratorV3` or `OpenApiGeneratorV31` instance.

```ts
import {
  OpenAPIRegistry,
  OpenApiGeneratorV3,
} from '@asteasolutions/zod-to-openapi';

const registry = new OpenAPIRegistry();

// Register definitions here

const generator = new OpenApiGeneratorV3(registry.definitions);

return generator.generateComponents();
```

### The Generator

There are two generators that can be used - `OpenApiGeneratorV3` and `OpenApiGeneratorV31`. They share the same interface but internally generate schemas that correctly follow the data format for the specific Open API version - `3.0.x` or `3.1.x`. The Open API version affects how some components are generated.

For example: changing the generator from `OpenApiGeneratorV3` to `OpenApiGeneratorV31` would result in following differences:

```ts
z.string().nullable().openapi({refId: 'name'});
```

```yml
# 3.1.0
# nullable is invalid in 3.1.0 but type arrays are invalid in previous versions
name:
  type:
    - 'string'
    - 'null'

# 3.0.0
name:
  type: 'string'
  nullable: true
```

Both generators take a single argument in their constructors - an array of definitions - i.e results from the registry or regular zod schemas.

The public methods of both generators are as follows:

`generateComponents` will generate only the `/components` section of an OpenAPI document (e.g. only `schemas` and `parameters`), not generating actual routes.

`generateDocument` will generate the whole OpenAPI document.

### Defining schemas

An OpenApi schema should be registered by using the `.openapi` method and providing a name:

```ts
const UserSchema = z
  .object({
    id: z.string().openapi({ example: '1212121' }),
    name: z.string().openapi({ example: 'John Doe' }),
    age: z.number().openapi({ example: 42 }),
  })
  .openapi('User');

const generator = new OpenApiGeneratorV3([UserSchema]);
```

The same can be achieved by using the `register` method of an `OpenAPIRegistry` instance. For more check the ["Using schemas vs a registry"](#using-schemas-vs-a-registry) section

```ts
const UserSchema = registry.register(
  'User',
  z.object({
    id: z.string().openapi({ example: '1212121' }),
    name: z.string().openapi({ example: 'John Doe' }),
    age: z.number().openapi({ example: 42 }),
  })
);

const generator = new OpenApiGeneratorV3(registry.definitions);
```

If run now, `generator.generateComponents()` will generate the following structure:

```yaml
components:
  schemas:
    User:
      type: object
      properties:
        id:
          type: string
          example: '1212121'
        name:
          type: string
          example: John Doe
        age:
          type: number
          example: 42
      required:
        - id
        - name
        - age
```

The key for the schema in the output is the first argument passed to `.openapi` method (or the `.register`) - in this case: `User`.

Note that `generateComponents` does not return YAML but a JS object - you can then serialize that object into YAML or JSON depending on your use-case.

The resulting schema can then be referenced by using `$ref: #/components/schemas/User` in an existing OpenAPI JSON. This will be done automatically for Routes defined through the registry.

Note by default a Zod object will result in `"additionalProperties": true` as per the Open API spec unless using `strict` or `catchall`, this is in contrast to normal Zod object usage where `zod.parse` is used.

### Defining routes & webhooks

#### Registering a path or webhook

An OpenAPI path is registered using the `registerPath` method of an `OpenAPIRegistry` instance. An OpenAPI webhook is registered using the `registerWebhook` method and takes the same parameters as `registerPath`.

```ts
registry.registerPath({
  method: 'get',
  path: '/users/{id}',
  description: 'Get user data by its id',
  summary: 'Get a single user',
  request: {
    params: z.object({
      id: z.string().openapi({ example: '1212121' }),
    }),
  },
  responses: {
    200: {
      description: 'Object with user data.',
      content: {
        'application/json': {
          schema: UserSchema,
        },
      },
    },
    204: {
      description: 'No content - successful operation',
    },
  },
});
```

The YAML equivalent of the schema above would be:

```yaml
'/users/{id}':
  get:
    description: Get user data by its id
    summary: Get a single user
    parameters:
      - in: path
        name: id
        schema:
          type: string
          example: '1212121'
        required: true
    responses:
      '200':
        description: Object with user data.
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/User'
      '204':
        description: No content - successful operation
```

The library specific properties for `registerPath` are `method`, `path`, `request` and `responses`. Everything else gets directly appended to the path definition.

- `method` - One of `get`, `post`, `put`, `delete` and `patch`;
- `path` - a string - being the path of the endpoint;
- `request` - an optional object with optional `body`, `params`, `query` and `headers` keys,
  - `query`, `params` - being instances of `ZodObject`
  - `body` - an object with a `description` and a `content` record where:
    - the key is a `mediaType` string like `application/json`
    - and the value is an object with a `schema` of any `zod` type
  - `headers` - instances of `ZodObject` or an array of any `zod` instances
- `responses` - an object where the key is the status code or `default` and the value is an object with a `description` and a `content` record where:
  - the key is a `mediaType` string like `application/json`
  - and the value is an object with a `schema` of any `zod` type

#### Defining route parameters

If you don't want to inline all parameter definitions, you can define them separately with `registerParameter` and then reference them:

```ts
const UserIdParam = registry.registerParameter(
  'UserId',
  z.string().openapi({
    param: {
      name: 'id',
      in: 'path',
    },
    example: '1212121',
  })
);

registry.registerPath({
  ...
  request: {
    params: z.object({
      id: UserIdParam
    }),
  },
  responses: ...
});
```

The YAML equivalent would be:

```yaml
components:
  parameters:
    UserId:
      in: path
      name: id
      schema:
        type: string
        example: '1212121'
      required: true

'/users/{id}':
  get:
    ...
    parameters:
      - $ref: '#/components/parameters/UserId'
    responses: ...
```

Note: In order to define properties that apply to the parameter itself, use the `param` property of `.openapi`. Any properties provided outside of `param` would be applied to the schema for this parameter.

#### Generating the full document

A full OpenAPI document can be generated using the `generateDocument` method of an `OpenApiGeneratorV3` or `OpenApiGeneratorV31` instance. It takes one argument - the document config. It may look something like this:

```ts
return generator.generateDocument({
  openapi: '3.0.0',
  info: {
    version: '1.0.0',
    title: 'My API',
    description: 'This is the API',
  },
  servers: [{ url: 'v1' }],
});
```

### Defining custom components

You can define components that are not OpenAPI schemas, including security schemes, response headers and others. See [this test file](spec/custom-components.spec.ts) for examples.

### A full example

A full example code can be found [here](./example/index.ts). And the YAML representation of its result - [here](./example/openapi-docs.yml)

### Using schemas vs a registry

Schemas are automatically being registered when referenced. That means that if you have a schema like:

```ts
const schema = z.object({ key: z.string().openapi('Test') }).openapi('Object');
```

you'd have the following resulting structure:

```yaml
components:
  schemas:
    Test:
      type: 'string',
    Object:
      type: 'object',
      properties:
        key:
          $ref: '#/components/schemas/Test'
      required: ['key']
```

This does not require any usages of an `OpenAPIRegistry` instance.

However the same output can be achieved with the following code:

```ts
const registry = new OpenAPIRegistry();
const schema = registry.register(
  'Object',
  z.object({ key: z.string().openapi('Test') })
);
```

The main benefit of the `.registry` method is that you can use the registry as a "collection" where you would put all such schemas.

With `.openapi`:

```ts
// file1.ts
export const Schema1 = ...

// file2.ts
export const Schema2 = ...

new OpenApiGeneratorV3([Schema1, Schema2])
```

Adding a `NewSchema` into `file3.ts` would require you to pass that schema manually into the array of the generator constructor.
Note: If a `NewSchema` is referenced by any other schemas or a route/webhook definition it would still appear in the resulting document.

With `registry.register`:

```ts
// registry.ts
export const registry = new OpenAPIRegistry()

// file1.ts
export const Schema1 = registry.register(...)

// file2.ts
export const Schema2 = registry.register(...)

new OpenApiGeneratorV3(registry.definitions)
```

Adding a `NewSchema` into `file3.ts` and using `registry.register` would NOT require you to do any changes to the generator constructor.

#### Conclusion

Using an `OpenAPIRegistry` instance is mostly useful if you would want your resulting document to contain unreferenced schemas.
That can sometimes be useful - for example when you are slowly integrating an already existing documentation with `@asteasolutions/zod-to-openapi` and you are migrating small pieces at a time. Those pieces can then be referenced directly from an existing documentation.

#### Adding it as part of your build

In a file inside your project you can have a file like so:

```ts
export const registry = new OpenAPIRegistry();

export function generateOpenAPI() {
  const config = {...}; // your config comes here

  return new OpenApiGeneratorV3(registry.definitions).generateDocument(config);
}
```

You then use the exported `registry` object to register all schemas, parameters and routes where appropriate.

Then you can create a script that executes the exported `generateOpenAPI` function. This script can be executed as a part of your build step so that it can write the result to some file like `openapi-docs.json`.

### Generation options

Schema generation can be altered in certain scenarios. This can be done by either:

#### Passing a global configuration as second argument for the generator:

```ts
const generator = new OpenApiGeneratorV3(registry.definitions, options);
```


There list of currently supported global options is:
```ts
const options = {
  unionPreferredType: 'oneOf' | 'anyOf' // configures whether oneOf or anyOf is used when generating a schema for a zod union
  sortComponents?: 'alphabetically'; // if sortComponents is passed with the value 'alphabetically' it would sort all schemas and parameters.
                                     // If not - they would appear in the order they were defined
}
```


#### Passing options for a one-off usage for a single schema:

```ts
// Note it is valid for metadata to be undefined in both of the below cases:

schema.openapi('Schema', metadata, options); // when registering a schema or
schema.openapi(metadata, options) // when simply adding some metadata to it
```

There list of currently supported one-off options is:
```ts
const options = {
  unionPreferredType: 'oneOf' | 'anyOf' // configures whether oneOf or anyOf is used when generating a schema for a zod union
}
```

## Zod schema types

### Supported types

The list of all supported types as of now is:

- `ZodAny`
- `ZodArray`
- `ZodBigInt`
- `ZodBoolean`
- `ZodDate`
- `ZodDefault`
- `ZodPrefault`
- `ZodDiscriminatedUnion`
  - including `discriminator` mapping when all Zod objects in the union are registered with `.register()` or contain a `refId`.
- `ZodEffects`
- `ZodEnum`
- `ZodIntersection`
- `ZodLiteral`
- `ZodNativeEnum`
- `ZodNullable`
- `ZodNumber`
  - including `z.number().int()` being inferred as `type: 'integer'`
- `ZodObject`
  - including `.catchall` resulting in the respective `additionalProperties` schema
  - also including `strict` resulting in the respective `additionalProperties` schema
- `ZodOptional`
- `ZodPipeline`
- `ZodReadonly`
- `ZodRecord`
- `ZodString`
  - adding `format` for:
    - `.emoji()`
    - `.cuid()`
    - `.cuid2()`
    - `.ulid()`
    - `.ip()`
    - `.date()`
    - `.datetime()`
    - `.uuid()`
    - `.email()`
    - `.url()`
  - adding `pattern` for `.regex()` is also supported

- `ZodTuple`
- `ZodUnion`
- `ZodUnknown`

Extending an instance of `ZodObject` is also supported and results in an OpenApi definition with `allOf`

### Unsupported types

In case you try to create an OpenAPI schema from a zod schema that is not one of the aforementioned types then you'd receive an `UnknownZodTypeError`.

You can still register such schemas on your own by providing a `type` via the `.openapi` method. In case you think that the desired behavior can be achieved automatically do not hesitate to reach out to us by describing your case via Github Issues.

#### Known issues

1. `z.nullable(schema)` [does not generate a $ref for underlying registered schemas](https://github.com/asteasolutions/zod-to-openapi/issues/141).
  - This is an implementation limitation.
  - However you can simply use `schema.nullable()` which has the exact same effect `zod` wise but it is also fully supported on our end.

## Technologies

- [Typescript](https://www.typescriptlang.org/)
- [Zod 4.x](https://github.com/colinhacks/zod)
- [OpenAPI 3.x TS](https://github.com/metadevpro/openapi3-ts)
 readmeEtag: '"2d79a562f62910c46c20782a747c233ca2d9dbdd"' readmeLastModified: Sun, 04 Jan 2026 11:33:09 GMT repositoryId: 481579846 description: A library that generates OpenAPI (Swagger) docs from Zod schemas created: '2022-04-14T11:32:57Z' updated: '2026-02-05T05:56:51Z' language: TypeScript archived: false stars: 1464 watchers: 12 forks: 97 owner: asteasolutions logo: https://avatars.githubusercontent.com/u/5244095?v=4 license: MIT repoEtag: '"e23715aeab7390fadd5642611c810284a5656769366686424747adceff588019"' repoLastModified: Thu, 05 Feb 2026 05:56:51 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/apiflask/apiflask v3: true id: 2c668cdfc766c1f598c7c7d845b9120c repositoryMetadata: base64Readme: >- 
![](https://apiflask.com/_assets/apiflask-logo.png)

# APIFlask

[![Build status](https://github.com/apiflask/apiflask/actions/workflows/tests.yml/badge.svg)](https://github.com/apiflask/apiflask/actions) [![codecov](https://codecov.io/gh/apiflask/apiflask/branch/main/graph/badge.svg?token=2CFPCZ1DMY)](https://codecov.io/gh/apiflask/apiflask)

APIFlask is a lightweight Python web API framework based on [Flask](https://github.com/pallets/flask). It's easy to use, highly customizable, ORM/ODM-agnostic, and 100% compatible with the Flask ecosystem.

APIFlask supports both [marshmallow](https://github.com/marshmallow-code/marshmallow) schemas and [Pydantic](https://docs.pydantic.dev/) models through a pluggable schema adapter system, giving you the flexibility to choose the validation approach that best fits your project.

With APIFlask, you will have:

- More sugars for view function (`@app.input()`, `@app.output()`, `@app.get()`, `@app.post()` and more)
- Automatic request validation and deserialization
- Automatic response formatting and serialization
- Automatic [OpenAPI Specification](https://github.com/OAI/OpenAPI-Specification) (OAS, formerly Swagger Specification) document generation
- Automatic interactive API documentation
- API authentication support (with [Flask-HTTPAuth](https://github.com/miguelgrinberg/flask-httpauth))
- Automatic JSON response for HTTP errors


## Requirements

- Python 3.9+
- Flask 2.1+


## Installation

For Linux and macOS:

```bash
$ pip3 install apiflask
```

For Windows:

```bash
> pip install apiflask
```


## Links

- Website: <https://apiflask.com>
- Documentation: <https://apiflask.com/docs>
- 中文文档: <https://zh.apiflask.com/docs>
- PyPI Releases: <https://pypi.python.org/pypi/APIFlask>
- Change Log: <https://apiflask.com/changelog>
- Source Code: <https://github.com/apiflask/apiflask>
- Issue Tracker: <https://github.com/apiflask/apiflask/issues>
- Discussion: <https://github.com/apiflask/apiflask/discussions>
- 中文论坛: <https://codekitchen.community>
- Twitter: <https://twitter.com/apiflask>
- Open Collective: <https://opencollective.com/apiflask>


## Donate

If you find APIFlask useful, please consider [donating](https://opencollective.com/apiflask) today. Your donation keeps APIFlask maintained and evolving.

Thank you to all our backers and sponsors!

### Backers

[![](https://opencollective.com/apiflask/backers.svg?width=890)](https://opencollective.com/apiflask)

### Sponsors

[![](https://opencollective.com/apiflask/sponsors.svg?width=890)](https://opencollective.com/apiflask)

## Example

```python
from apiflask import APIFlask, Schema, abort
from apiflask.fields import Integer, String
from apiflask.validators import Length, OneOf

app = APIFlask(__name__)

pets = [
    {'id': 0, 'name': 'Kitty', 'category': 'cat'},
    {'id': 1, 'name': 'Coco', 'category': 'dog'}
]


class PetIn(Schema):
    name = String(required=True, validate=Length(0, 10))
    category = String(required=True, validate=OneOf(['dog', 'cat']))


class PetOut(Schema):
    id = Integer()
    name = String()
    category = String()


@app.get('/')
def say_hello():
    # returning a dict or list equals to use jsonify()
    return {'message': 'Hello!'}


@app.get('/pets/<int:pet_id>')
@app.output(PetOut)
def get_pet(pet_id):
    if pet_id > len(pets) - 1:
        abort(404)
    # you can also return an ORM/ODM model class instance directly
    # APIFlask will serialize the object into JSON format
    return pets[pet_id]


@app.patch('/pets/<int:pet_id>')
@app.input(PetIn(partial=True))  # -> json_data
@app.output(PetOut)
def update_pet(pet_id, json_data):
    # the validated and parsed input data will
    # be injected into the view function as a dict
    if pet_id > len(pets) - 1:
        abort(404)
    for attr, value in json_data.items():
        pets[pet_id][attr] = value
    return pets[pet_id]
```

<details>
<summary>You can use Pydantic models for type-hint based validation</summary>

```python
from enum import Enum

from apiflask import APIFlask, abort
from pydantic import BaseModel, Field

app = APIFlask(__name__)


class PetCategory(str, Enum):
    DOG = 'dog'
    CAT = 'cat'


class PetOut(BaseModel):
    id: int
    name: str
    category: PetCategory


class PetIn(BaseModel):
    name: str = Field(min_length=1, max_length=50)
    category: PetCategory


pets = [
    {'id': 0, 'name': 'Kitty', 'category': 'cat'},
    {'id': 1, 'name': 'Coco', 'category': 'dog'}
]


@app.get('/')
def say_hello():
    return {'message': 'Hello, Pydantic!'}


@app.get('/pets')
@app.output(list[PetOut])
def get_pets():
    return pets


@app.get('/pets/<int:pet_id>')
@app.output(PetOut)
def get_pet(pet_id: int):
    if pet_id > len(pets) or pet_id < 1:
        abort(404)
    return pets[pet_id - 1]


@app.post('/pets')
@app.input(PetIn, location='json')
@app.output(PetOut, status_code=201)
def create_pet(json_data: PetIn):
    # the validated and parsed input data will
    # be injected into the view function as a Pydantic model instance
    new_id = len(pets) + 1
    new_pet = PetOut(id=new_id, name=json_data.name, category=json_data.category)
    pets.append(new_pet)
    return new_pet
```

</details>

<details>
<summary>Or use <code>async def</code></summary>

```bash
$ pip install -U "apiflask[async]"
```

```python
import asyncio

from apiflask import APIFlask

app = APIFlask(__name__)


@app.get('/')
async def say_hello():
    await asyncio.sleep(1)
    return {'message': 'Hello!'}
```

See <em><a href="https://flask.palletsprojects.com/async-await">Using async and await</a></em> for the details of the async support in Flask 2.0.

</details>

Save this as `app.py`, then run it with:

```bash
$ flask run --debug
```

Now visit the interactive API documentation (Swagger UI) at <http://localhost:5000/docs>:

![](https://apiflask.com/_assets/swagger-ui.png)

Or you can change the API documentation UI when creating the APIFlask instance with the `docs_ui` parameter:

```py
app = APIFlask(__name__, docs_ui='redoc')
```

Now <http://localhost:5000/docs> will render the API documentation with Redoc.

Supported `docs_ui` values (UI libraries) include:

- `swagger-ui` (default value): [Swagger UI](https://github.com/swagger-api/swagger-ui)
- `redoc`: [Redoc](https://github.com/Redocly/redoc)
- `elements`: [Elements](https://github.com/stoplightio/elements)
- `rapidoc`: [RapiDoc](https://github.com/rapi-doc/RapiDoc)
- `rapipdf`: [RapiPDF](https://github.com/mrin9/RapiPdf)

The auto-generated OpenAPI spec file is available at <http://localhost:5000/openapi.json>. You can also get the spec with [the `flask spec` command](https://apiflask.com/openapi/#the-flask-spec-command):

```bash
$ flask spec
```

For some complete examples, see [/examples](https://github.com/apiflask/apiflask/tree/main/examples).


## Relationship with Flask

APIFlask is a thin wrapper on top of Flask. You only need to remember the following differences (see *[Migrating from Flask](https://apiflask.com/migrations/flask/)* for more details):

- When creating an application instance, use `APIFlask` instead of `Flask`.
- When creating a blueprint instance, use `APIBlueprint` instead of `Blueprint`.
- The `abort()` function from APIFlask (`apiflask.abort`) returns JSON error response.

For a minimal Flask application:

```python
from flask import Flask, request
from markupsafe import escape

app = Flask(__name__)

@app.route('/')
def hello():
    name = request.args.get('name', 'Human')
    return f'Hello, {escape(name)}'
```

Now change to APIFlask:

```python
from apiflask import APIFlask  # step one
from flask import request
from markupsafe import escape

app = APIFlask(__name__)  # step two

@app.route('/')
def hello():
    name = request.args.get('name', 'Human')
    return f'Hello, {escape(name)}'
```

In a word, to make Web API development in Flask more easily, APIFlask provides `APIFlask` and `APIBlueprint` to extend Flask's `Flask` and `Blueprint` objects and it also ships with some helpful utilities. Other than that, you are actually using Flask.


## Credits

APIFlask starts as a fork of [APIFairy](https://github.com/miguelgrinberg/APIFairy) and is inspired by [flask-smorest](https://github.com/marshmallow-code/flask-smorest) and [FastAPI](https://github.com/tiangolo/fastapi) (see *[Comparison and Motivations](https://apiflask.com/comparison)* for the comparison between these projects).
 readmeEtag: '"02b0bb3919b7318a1c0f666420d0e77673b87ba7"' readmeLastModified: Sun, 16 Nov 2025 09:21:13 GMT repositoryId: 329190167 description: A lightweight Python web API framework. created: '2021-01-13T04:08:48Z' updated: '2026-02-06T03:00:42Z' language: Python archived: false stars: 1119 watchers: 18 forks: 143 owner: apiflask logo: https://avatars.githubusercontent.com/u/78075725?v=4 license: MIT repoEtag: '"b295bd423ebd140a7ed7bb00586d0904321b4563d136f673f9707d4ba554e56e"' repoLastModified: Fri, 06 Feb 2026 03:00:42 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/greyli/apiflask - source: openapi3 tags repository: https://github.com/networknt/json-schema-validator v3: true repositoryMetadata: base64Readme: >- [Stack Overflow](https://stackoverflow.com/questions/tagged/light-4j) |
[Google Group](https://groups.google.com/forum/#!forum/light-4j) |
[Gitter Chat](https://gitter.im/networknt/json-schema-validator) |
[Subreddit](https://www.reddit.com/r/lightapi/) |
[Youtube](https://www.youtube.com/channel/UCHCRMWJVXw8iB7zKxF55Byw) |
[Documentation](https://doc.networknt.com/library/json-schema-validator/) |
[Contribution Guide](https://doc.networknt.com/contribute/) |

[![CI](https://github.com/networknt/json-schema-validator/actions/workflows/ci.yml/badge.svg)](https://github.com/networknt/json-schema-validator/actions/workflows/ci.yml)
[![Maven Central](https://img.shields.io/maven-central/v/com.networknt/json-schema-validator.svg)](http://search.maven.org/#search%7Cga%7C1%7Cg%3Acom.networknt%20a%3Ajson-schema-validator)
[![codecov.io](https://codecov.io/github/networknt/json-schema-validator/coverage.svg?branch=master)](https://codecov.io/github/networknt/json-schema-validator?branch=master)
[![Javadocs](http://www.javadoc.io/badge/com.networknt/json-schema-validator.svg)](https://www.javadoc.io/doc/com.networknt/json-schema-validator)

This is a Java implementation of the [JSON Schema Core Draft v4, v6, v7, v2019-09 and v2020-12](https://json-schema.org/specification) specification for JSON schema validation. This implementation supports [Customizing Dialects, Vocabularies, Keywords and Formats](doc/custom-dialect.md).

The JSON parser used is the [Jackson](https://github.com/FasterXML/jackson) parser.

[OpenAPI](doc/openapi.md) 3 request/response validation is supported with the use of the appropriate dialect.

As it is a key component in our [light-4j](https://github.com/networknt/light-4j) microservices framework to validate request/response against OpenAPI specification for [light-rest-4j](http://www.networknt.com/style/light-rest-4j/) and RPC schema for [light-hybrid-4j](http://www.networknt.com/style/light-hybrid-4j/) at runtime, performance is the most important aspect in the design.

## JSON Schema Specification compatibility

[![Supported Dialects](https://img.shields.io/endpoint?url=https%3A%2F%2Fbowtie.report%2Fbadges%2Fjava-com.networknt-json-schema-validator%2Fsupported_versions.json)](https://bowtie.report/#/implementations/java-networknt-json-schema-validator)
[![Draft 2020-12](https://img.shields.io/endpoint?url=https%3A%2F%2Fbowtie.report%2Fbadges%2Fjava-com.networknt-json-schema-validator%2Fcompliance%2Fdraft2020-12.json)](https://bowtie.report/#/dialects/draft2020-12)
[![Draft 2019-09](https://img.shields.io/endpoint?url=https%3A%2F%2Fbowtie.report%2Fbadges%2Fjava-com.networknt-json-schema-validator%2Fcompliance%2Fdraft2019-09.json)](https://bowtie.report/#/dialects/draft2019-09)
[![Draft 7](https://img.shields.io/endpoint?url=https%3A%2F%2Fbowtie.report%2Fbadges%2Fjava-com.networknt-json-schema-validator%2Fcompliance%2Fdraft7.json)](https://bowtie.report/#/dialects/draft7)
[![Draft 6](https://img.shields.io/endpoint?url=https%3A%2F%2Fbowtie.report%2Fbadges%2Fjava-com.networknt-json-schema-validator%2Fcompliance%2Fdraft6.json)](https://bowtie.report/#/dialects/draft6)
[![Draft 4](https://img.shields.io/endpoint?url=https%3A%2F%2Fbowtie.report%2Fbadges%2Fjava-com.networknt-json-schema-validator%2Fcompliance%2Fdraft4.json)](https://bowtie.report/#/dialects/draft4)

Information on the compatibility support for each version, including known issues, can be found in the [Compatibility with JSON Schema versions](doc/compatibility.md) document.

Since [Draft 2019-09](https://json-schema.org/draft/2019-09/json-schema-validation#rfc.section.7) the `format` keyword only generates annotations by default and does not generate assertions.

This behavior can be overridden to generate assertions by setting the `formatAssertionsEnabled` to `true` in `SchemaRegistryConfig` or `ExecutionConfig`.

## Upgrading to new versions

This library can contain breaking changes in `minor` version releases that may require code changes.

Information on notable or breaking changes when upgrading the library can be found in the [Upgrading to new versions](doc/upgrading.md) document.

The [Releases](https://github.com/networknt/json-schema-validator/releases) page will contain information on the latest versions.

## Comparing against other implementations

The [JSON Schema Validation Comparison](https://github.com/creek-service/json-schema-validation-comparison) project from Creek has an informative [Comparison of JVM based Schema Validation Implementations](https://www.creekservice.org/json-schema-validation-comparison/) which compares both the functional and performance characteristics of a number of different Java implementations.

- [Functional comparison](https://www.creekservice.org/json-schema-validation-comparison/functional#summary-results-table)
- [Performance comparison](https://www.creekservice.org/json-schema-validation-comparison/performance#json-schema-test-suite-benchmark)

The [Bowtie](https://github.com/bowtie-json-schema/bowtie) project has a [report](https://bowtie.report/) that compares functional characteristics of different implementations, including non-Java implementations, but does not do any performance benchmarking.

## Why this library

#### Performance

The following is the benchmark results from the [JSON Schema Validator Perftest](https://github.com/networknt/json-schema-validator-perftest) project that uses the [Java Microbenchmark Harness](https://github.com/openjdk/jmh).

Note that the benchmark results are highly dependent on the input data workloads and schemas used for the validation.

In this case this workload is using the Draft 4 specification and largely tests the performance of the evaluating the `properties` keyword. You may refer to [Results of performance comparison of JVM based JSON Schema Validation Implementations](https://www.creekservice.org/json-schema-validation-comparison/performance#json-schema-test-suite-benchmark) for benchmark results that use the [JSON Schema Test Suite](https://github.com/json-schema-org/JSON-Schema-Test-Suite).

If performance is an important consideration, the specific sample workloads should be benchmarked, as there are different performance characteristics when certain keywords are used. For instance the use of the `unevaluatedProperties` or `unevaluatedItems` keyword will trigger annotation collection in the related validators, such as the `properties` or `items` validators, and annotation collection will adversely affect performance.

Special attention should also be made for inefficient schemas using deeply nested `oneOf` or `anyOf` that do not have a condition to short-circuit the evaluation using `if` and `then`. The validator has no choice but to perform all the evaluations, and the error messages would be typically very confusing as it will return all the messages from the children.

##### NetworkNT 2.0.0

```
Benchmark                                     Mode  Cnt       Score     Error   Units
NetworkntBenchmark.basic                     thrpt   10    5297.105 ± 290.078   ops/s
NetworkntBenchmark.basic:gc.alloc.rate       thrpt   10    1618.328 ±  88.626  MB/sec
NetworkntBenchmark.basic:gc.alloc.rate.norm  thrpt   10  320360.020 ±   0.002    B/op
NetworkntBenchmark.basic:gc.count            thrpt   10     365.000            counts
NetworkntBenchmark.basic:gc.time             thrpt   10     130.000                ms
```

###### Everit 1.14.6

```
Benchmark                                     Mode  Cnt       Score     Error   Units
EveritBenchmark.basic                        thrpt   10    4615.637 ± 151.195   ops/s
EveritBenchmark.basic:gc.alloc.rate          thrpt   10    2097.810 ±  68.708  MB/sec
EveritBenchmark.basic:gc.alloc.rate.norm     thrpt   10  476592.023 ±   0.001    B/op
EveritBenchmark.basic:gc.count               thrpt   10     521.000            counts
EveritBenchmark.basic:gc.time                thrpt   10     170.000                ms
```

#### Functionality

This implementation is tested against the [JSON Schema Test Suite](https://github.com/json-schema-org/JSON-Schema-Test-Suite). As tests are continually added to the suite, these test results may not be current.

| Implementations | Overall                                                              | DRAFT_03 | DRAFT_04                                                           | DRAFT_06                                                           | DRAFT_07                                                           | DRAFT_2019_09                                                       | DRAFT_2020_12                                                       |
| --------------- | -------------------------------------------------------------------- | -------- | ------------------------------------------------------------------ | ------------------------------------------------------------------ | ------------------------------------------------------------------ | ------------------------------------------------------------------- | ------------------------------------------------------------------- |
| NetworkNt       | pass: r:4840 (100.0%) o:2421 (100.0%)<br>fail: r:0 (0.0%) o:0 (0.0%) |          | pass: r:610 (100.0%) o:255 (100.0%)<br>fail: r:0 (0.0%) o:0 (0.0%) | pass: r:829 (100.0%) o:322 (100.0%)<br>fail: r:0 (0.0%) o:0 (0.0%) | pass: r:913 (100.0%) o:554 (100.0%)<br>fail: r:0 (0.0%) o:0 (0.0%) | pass: r:1227 (100.0%) o:639 (100.0%)<br>fail: r:0 (0.0%) o:0 (0.0%) | pass: r:1261 (100.0%) o:651 (100.0%)<br>fail: r:0 (0.0%) o:0 (0.0%) |

- Note that this uses the `JoniRegularExpressionFactory` for the `pattern` and `format` `regex` tests.

#### Jackson Parser

This library uses [Jackson](https://github.com/FasterXML/jackson) which is a Java JSON parser that is widely used in other projects. If you are already using the Jackson parser in your project, it is natural to choose this library over others for schema validation.

#### YAML Support

The library works with JSON and YAML on both schema definitions and input data.

#### OpenAPI Support

The OpenAPI 3.0 specification is using JSON schema to validate the request/response. The library has support for the OpenAPI 3.0 and OpenAPI 3.1 dialects.

#### Minimal Dependencies

Following the design principle of the Light Platform, this library has minimal dependencies to ensure there are no dependency conflicts when using it.

##### Required Dependencies

The following are the dependencies that will automatically be included when this library is included.

```xml
<dependency>
    <!-- Used for logging -->
    <groupId>org.slf4j</groupId>
    <artifactId>slf4j-api</artifactId>
    <version>${version.slf4j}</version>
</dependency>

<dependency>
    <!-- Used to process JSON -->
    <groupId>com.fasterxml.jackson.core</groupId>
    <artifactId>jackson-databind</artifactId>
    <version>${version.jackson}</version>
</dependency>

<dependency>
    <!-- Used to process YAML -->
    <groupId>com.fasterxml.jackson.dataformat</groupId>
    <artifactId>jackson-dataformat-yaml</artifactId>
    <version>${version.jackson}</version>
</dependency>

<dependency>
    <!-- Used to validate RFC 3339 date and date-time -->
    <groupId>com.ethlo.time</groupId>
    <artifactId>itu</artifactId>
    <version>${version.itu}</version>
</dependency>
```

##### Optional Dependencies

The following are the optional dependencies that may be required for certain options.

These are not automatically included and setting the relevant option without adding the library will result in a `ClassNotFoundException`.

```xml
<dependency>
    <!-- Used to validate ECMA 262 regular expressions -->
    <!-- Approximately 50 MB in dependencies -->
    <!-- GraalJSRegularExpressionFactory -->
    <groupId>org.graalvm.js</groupId>
    <artifactId>js</artifactId>
    <version>${version.graaljs}</version>
</dependency>

<dependency>
    <!-- Used to validate ECMA 262 regular expressions -->
    <!-- Approximately 2 MB in dependencies -->
    <!-- JoniRegularExpressionFactory -->
    <groupId>org.jruby.joni</groupId>
    <artifactId>joni</artifactId>
    <version>${version.joni}</version>
</dependency>
```

##### Excludable Dependencies

The following are required dependencies that are automatically included, but can be explicitly excluded if they are not required.

The YAML dependency can be excluded if this is not required. Attempting to process schemas or input that are YAML will result in a `ClassNotFoundException`.

```xml
<dependency>
    <groupId>com.networknt</groupId>
    <artifactId>json-schema-validator</artifactId>
    <exclusions>
        <exclusion>
            <groupId>com.fasterxml.jackson.dataformat</groupId>
            <artifactId>jackson-dataformat-yaml</artifactId>
        </exclusion>
    </exclusions>
</dependency>
```

The Ethlo Time dependency can be excluded if accurate validation of the `date-time` format is not required. The `date-time` format will then use `java.time.OffsetDateTime` to determine if the `date-time` is valid .

```xml
<dependency>
    <groupId>com.networknt</groupId>
    <artifactId>json-schema-validator</artifactId>
    <exclusions>
        <exclusion>
            <groupId>com.ethlo.time</groupId>
            <artifactId>itu</artifactId>
        </exclusion>
    </exclusions>
</dependency>
```

#### Community

This library is very active with a lot of contributors. New features and bug fixes are handled quickly by the team members. Because it is an essential dependency of the [light-4j](https://github.com/networknt/light-4j) framework in the same GitHub organization, it will be evolved and maintained along with the framework.

## Prerequisite

The library supports Java 8 and up. If you want to build from the source code, you need to install JDK 8 locally. To support multiple version of JDK, you can use [SDKMAN](https://www.networknt.com/tool/sdk/)

## Usage

### Adding the dependency

The `json-schema-validator` package is published to Maven Central and is available in two major release lines, depending on your Java and Jackson versions:

- **2.x.x** — Compatible with **Java 8+** and **Jackson 2.x**
- **3.x.x** — Compatible with **Java 17+** and **Jackson 3.x**

#### Maven:

**Jackson 2 (Java 8+)**

```xml
<dependency>
    <groupId>com.networknt</groupId>
    <artifactId>json-schema-validator</artifactId>
    <version>2.0.1</version>
</dependency>
```


**Jackson 3 (Java 17+)**

```xml
<dependency>
    <groupId>com.networknt</groupId>
    <artifactId>json-schema-validator</artifactId>
    <version>3.0.0</version>
</dependency>
```

#### Gradle:


**Jackson 2 (Java 8+)**


```java
dependencies {
    implementation(group: 'com.networknt', name: 'json-schema-validator', version: '2.0.1');
}
```


**Jackson 3 (Java 17+)**


```java
dependencies {
    implementation(group: 'com.networknt', name: 'json-schema-validator', version: '3.0.0');
}
```


### Validating inputs against a schema

The following example demonstrates how inputs are validated against a schema. It comprises the following steps.

- Creating a configuration for controlling validator behavior.
- Creating a schema registry with the default schema dialect and how the schemas can be retrieved.
  - Configuring mapping the `$id` to a retrieval IRI using `schemaIdResolvers`.
  - Configuring how the schemas are loaded using the retrieval IRI.
    For instance a `Map<String, String> schemas` containing a mapping of retrieval URI to schema data as a `String` can by configured using `builder.schemas(schemas)`. This also accepts a `Function<String, String> schemaRetrievalFunction`.
- Loading a schema from a schema location.
- Using the schema to validate the data along with setting any execution specific configuration like for instance the locale or whether format assertions are enabled.

```java
/*
 * The SchemaRegistryConfig can be optionally used to configure certain aspects
 * of how the validation is performed.
 *
 * By default the JDK regular expression implementation which is not ECMA 262
 * compliant is used. The GraalJSRegularExpressionFactory.getInstance() offers
 * the best compliance followed by JoniRegularExpressionFactory.getInstance()
 * but both require additional optional dependencies.
 */
SchemaRegistryConfig schemaRegistryConfig = SchemaRegistryConfig.builder()
        .regularExpressionFactory(JoniRegularExpressionFactory.getInstance()).build();

/*
 * This creates a schema registry that supports all the standard dialects for
 * cross-dialect validation and will use Draft 2020-12 as the default if $schema
 * is not specified in the schema data. If $schema is specified in the schema
 * data then that schema dialect will be used instead and this version is
 * ignored.
 */
SchemaRegistry schemaRegistry = SchemaRegistry.withDefaultDialect(SpecificationVersion.DRAFT_2020_12,
        builder -> builder.schemaRegistryConfig(schemaRegistryConfig)
                /*
                 * This creates a mapping from $id which starts with
                 * https://www.example.org/schema to the retrieval IRI classpath:schema.
                 */
                .schemaIdResolvers(schemaIdResolvers -> schemaIdResolvers
                        .mapPrefix("https://www.example.com/schema", "classpath:schema")));

/*
 * Due to the mapping the schema will be retrieved from the classpath at
 * classpath:schema/example-main.json. If the schema data does not specify an
 * $id the absolute IRI of the schema location will be used as the $id. If the
 * schema data does not specify a dialect using $schema the default dialect
 * specified when creating the schema registry.
 */
Schema schema = schemaRegistry.getSchema(SchemaLocation.of("https://www.example.com/schema/example-main.json"));
String input = "{\r\n"
 + "  \"main\": {\r\n"
 + "    \"common\": {\r\n"
 + "      \"field\": \"invalidfield\"\r\n"
 + "    }\r\n"
 + "  }\r\n"
 + "}";

List<Error> errors = schema.validate(input, InputFormat.JSON, executionContext -> {
    /*
     * By default since Draft 2019-09 the format keyword only generates annotations
     * and not assertions.
     */
    executionContext.executionConfig(executionConfig -> executionConfig.formatAssertionsEnabled(true));
});
```

### Validating a schema against a meta-schema

The following example demonstrates how a schema is validated against a meta-schema of a dialect.

This is actually the same as validating inputs against a schema except in this case the input is the schema and the schema used is the meta-schema.

Note that the meta-schemas for Draft 4, Draft 6, Draft 7, Draft 201
9-09 and Draft 2020-12 are bundled with the library and these classpath resources will be used by default.

```java
SchemaRegistry schemaRegistry = SchemaRegistry.withDialect(Dialects.getDraft202012());
/*
 * Due to the mapping the meta-schema for the dialect will be retrieved from the
 * classpath at classpath:draft/2020-12/schema.
 */
Schema schema = schemaRegistry.getSchema(SchemaLocation.of(Dialects.getDraft202012().getId()));
String input = "{\r\n"
    + "  \"type\": \"object\",\r\n"
    + "  \"properties\": {\r\n"
    + "    \"key\": {\r\n"
    + "      \"title\" : \"My key\",\r\n"
    + "      \"type\": \"invalidtype\"\r\n"
    + "    }\r\n"
    + "  }\r\n"
    + "}";
List<Error> errors = schema.validate(input, InputFormat.JSON, executionContext -> {
    /*
     * By default since Draft 2019-09 the format keyword only generates annotations
     * and not assertions.
     */
    executionContext.executionConfig(executionConfig -> executionConfig.formatAssertionsEnabled(true));
});
```

### Require schema dialect to be specified

The specification allows for the `$schema` keyword not to be specified, in which case the schema will default to the default dialect specified.

The following example creates a `SchemaRegistry` that does not specify a default dialect and will throw a `MissingSchemaKeywordException` if the schema does not specify a dialect using the `$schema` keyword.

```java
SchemaRegistry registry = SchemaRegistry.withDefaultDialectId(null);
```

### Results and output formats

#### Results

The following types of results are generated by the library.

| Type        | Description                                                                                                                                                                                                                                                                                                                                                                                                                  |
| ----------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Assertions  | Validation errors generated by an assertion keyword on a particular input data instance. This is generally described in an `Error` or in a `OutputUnit`. Note that since Draft 2019-09 the `format` keyword no longer generates assertions by default and instead generates only annotations unless configured otherwise using a configuration option or by using a dialect that uses the appropriate vocabulary.            |
| Annotations | Additional information generated by an annotation keyword for a particular input data instance. This is generally described in a `OutputUnit`. Annotation collection and reporting is turned off by default. Annotations required by keywords such as `unevaluatedProperties` or `unevaluatedItems` are always collected for evaluation purposes and cannot be disabled but will not be reported unless configured to do so. |

The following information is used to describe both types of results.

| Type              | Description                                                                                                                                                                                                   |
| ----------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Evaluation Path   | This is the set of keys from the root through which evaluation passes to reach the schema for evaluating the instance. This includes `$ref` and `$dynamicRef`. eg. `/properties/bar/$ref/properties/bar-prop` |
| Schema Location   | This is the canonical IRI of the schema plus the JSON pointer fragment to the schema that was used for evaluating the instance. eg. `https://json-schema.org/schemas/example#/$defs/bar/properties/bar-prop`  |
| Instance Location | This is the JSON pointer fragment to the instance data that was being evaluated. eg. `/bar/bar-prop`                                                                                                          |

Assertions contains the following additional information

| Type          | Description                                                                                                                                                                                                                                                                                                                   |
| ------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Message       | The validation error message.                                                                                                                                                                                                                                                                                                 |
| Message Key   | The message key used for generating the message for localization.                                                                                                                                                                                                                                                             |
| Arguments     | The arguments used for generating the message.                                                                                                                                                                                                                                                                                |
| Keyword       | The keyword that generated the message.                                                                                                                                                                                                                                                                                       |
| Schema Node   | The `JsonNode` pointed to by the Schema Location. This is the schema data that caused the input data to fail. It is possible to get the location information by configuring the `SchemaRegistry` with a `NodeReader` that uses the `LocationJsonNodeFactoryFactory` and using `JsonNodes.tokenStreamLocationOf(schemaNode)`. |
| Instance Node | The `JsonNode` pointed to by the Instance Location. This is the input data that failed validation. It is possible to get the location information by configuring the `SchemaRegistry` with a `NodeReader` that uses the `LocationJsonNodeFactoryFactory` and using `JsonNodes.tokenStreamLocationOf(instanceNode)`.          |
| Error         | The error.                                                                                                                                                                                                                                                                                                                    |
| Details       | Additional details that can be set by custom keyword validator implementations. The library will set the `property` and `index` details for certain errors. For instane the `required` keyword will set the `property`. Note that this is not part of the instance location as that points to the instance node.              |

Annotations contains the following additional information

| Type  | Description                    |
| ----- | ------------------------------ |
| Value | The annotation value generated |

##### Line and Column Information

The library can be configured to store line and column information in the `JsonNode` instances for the instance and schema nodes. This will adversely affect performance and is not configured by default.

This is done by configuring a `NodeReader` that uses the `LocationJsonNodeFactoryFactory` on the `SchemaRegistry`. The `JsonLocation` information can then be retrieved using `JsonNodes.tokenStreamLocationOf(jsonNode)`.

```java
String schemaData = "{\r\n"
                + "  \"$id\": \"https://schema/myschema\",\r\n"
                + "  \"properties\": {\r\n"
                + "    \"startDate\": {\r\n"
                + "      \"format\": \"date\",\r\n"
                + "      \"minLength\": 6\r\n"
                + "    }\r\n"
                + "  }\r\n"
                + "}";
String inputData = "{\r\n"
                + "  \"startDate\": \"1\"\r\n"
                + "}";
SchemaRegistry schemaRegistry = SchemaRegistry.withDialect(Dialects.getDraft202012(),
        builder -> builder.nodeReader(nodeReader -> nodeReader.locationAware()));

Schema schema = schemaRegistry.getSchema(schemaData, InputFormat.JSON);
List<Error> errors = schema.validate(inputData, InputFormat.JSON, executionContext -> {
    executionContext.executionConfig(executionConfig -> executionConfig.formatAssertionsEnabled(true));
});
Error format = errors.get(0);
JsonLocation formatInstanceNodeTokenLocation = JsonNodes.tokenStreamLocationOf(format.getInstanceNode());
JsonLocation formatSchemaNodeTokenLocation = JsonNodes.tokenStreamLocationOf(format.getSchemaNode());
Error minLength = errors.get(1);
JsonLocation minLengthInstanceNodeTokenLocation = JsonNodes.tokenStreamLocationOf(minLength.getInstanceNode());
JsonLocation minLengthSchemaNodeTokenLocation = JsonNodes.tokenStreamLocationOf(minLength.getSchemaNode());

assertEquals("format", format.getKeyword());
assertEquals("date", format.getSchemaNode().asString());
assertEquals(5, formatSchemaNodeTokenLocation.getLineNr());
assertEquals(17, formatSchemaNodeTokenLocation.getColumnNr());
assertEquals("1", format.getInstanceNode().asString());
assertEquals(2, formatInstanceNodeTokenLocation.getLineNr());
assertEquals(16, formatInstanceNodeTokenLocation.getColumnNr());
assertEquals("minLength", minLength.getKeyword());
assertEquals("6", minLength.getSchemaNode().asString());
assertEquals(6, minLengthSchemaNodeTokenLocation.getLineNr());
assertEquals(20, minLengthSchemaNodeTokenLocation.getColumnNr());
assertEquals("1", minLength.getInstanceNode().asString());
assertEquals(2, minLengthInstanceNodeTokenLocation.getLineNr());
assertEquals(16, minLengthInstanceNodeTokenLocation.getColumnNr());
assertEquals(16, minLengthInstanceNodeTokenLocation.getColumnNr());
```

#### Output formats

This library implements the Flag, List and Hierarchical output formats defined in the [Specification for Machine-Readable Output for JSON Schema Validation and Annotation](https://github.com/json-schema-org/json-schema-spec/blob/8270653a9f59fadd2df0d789f22d486254505bbe/jsonschema-validation-output-machines.md).

The List and Hierarchical output formats are particularly helpful for understanding how the system arrived at a particular result.

| Output Format | Description                                                                                                                                                                                                                                     |
| ------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| Default       | Generates the list of assertions.                                                                                                                                                                                                               |
| Boolean       | Returns `true` if the validation is successful. Note that the fail fast option is turned on by default for this output format.                                                                                                                  |
| Flag          | Returns an `OutputFlag` object with `valid` having `true` if the validation is successful. Note that the fail fast option is turned on by default for this output format.                                                                       |
| List          | Returns an `OutputUnit` object with `details` with a list of `OutputUnit` objects with the assertions and annotations. Note that annotations are not collected by default and it has to be enabled as it will impact performance.               |
| Hierarchical  | Returns an `OutputUnit` object with a hierarchy of `OutputUnit` objects for the evaluation path with the assertions and annotations. Note that annotations are not collected by default and it has to be enabled as it will impact performance. |

The following example shows how to generate the hierarchical output format with annotation collection and reporting turned on and format assertions turned on.

```java
SchemaRegistry schemaRegistry = SchemaRegistry.withDialect(Dialects.getDraft202012());
Schema schema = schemaRegistry.getSchema(SchemaLocation.of("https://json-schema.org/schemas/example"));

OutputUnit outputUnit = schema.validate(inputData, InputFormat.JSON, OutputFormat.HIERARCHICAL, executionContext -> {
    executionContext.executionConfig(executionConfig -> executionConfig
            .annotationCollectionEnabled(true)
            .annotationCollectionFilter(keyword -> true)
            .formatAssertionsEnabled(true));
});
```

The following is sample output from the Hierarchical format.

```json
{
  "valid": false,
  "evaluationPath": "",
  "schemaLocation": "https://json-schema.org/schemas/example#",
  "instanceLocation": "",
  "droppedAnnotations": {
    "properties": ["foo", "bar"],
    "title": "root"
  },
  "details": [
    {
      "valid": false,
      "evaluationPath": "/properties/foo/allOf/0",
      "schemaLocation": "https://json-schema.org/schemas/example#/properties/foo/allOf/0",
      "instanceLocation": "/foo",
      "errors": {
        "required": "required property 'unspecified-prop' not found"
      }
    },
    {
      "valid": false,
      "evaluationPath": "/properties/foo/allOf/1",
      "schemaLocation": "https://json-schema.org/schemas/example#/properties/foo/allOf/1",
      "instanceLocation": "/foo",
      "droppedAnnotations": {
        "properties": ["foo-prop"],
        "title": "foo-title",
        "additionalProperties": ["foo-prop", "other-prop"]
      },
      "details": [
        {
          "valid": false,
          "evaluationPath": "/properties/foo/allOf/1/properties/foo-prop",
          "schemaLocation": "https://json-schema.org/schemas/example#/properties/foo/allOf/1/properties/foo-prop",
          "instanceLocation": "/foo/foo-prop",
          "errors": {
            "const": "must be a constant value 1"
          },
          "droppedAnnotations": {
            "title": "foo-prop-title"
          }
        }
      ]
    },
    {
      "valid": false,
      "evaluationPath": "/properties/bar/$ref",
      "schemaLocation": "https://json-schema.org/schemas/example#/$defs/bar",
      "instanceLocation": "/bar",
      "droppedAnnotations": {
        "properties": ["bar-prop"],
        "title": "bar-title"
      },
      "details": [
        {
          "valid": false,
          "evaluationPath": "/properties/bar/$ref/properties/bar-prop",
          "schemaLocation": "https://json-schema.org/schemas/example#/$defs/bar/properties/bar-prop",
          "instanceLocation": "/bar/bar-prop",
          "errors": {
            "minimum": "must have a minimum value of 10"
          },
          "droppedAnnotations": {
            "title": "bar-prop-title"
          }
        }
      ]
    }
  ]
}
```

## Configuration

### Execution Configuration

| Name                          | Description                                                                                                                                                                                                                     | Default Value         |
| ----------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | --------------------- |
| `annotationCollectionEnabled` | Controls whether annotations are collected during processing. Note that collecting annotations will adversely affect performance.                                                                                               | `false`               |
| `annotationCollectionFilter`  | The predicate used to control which keyword to collect and report annotations for. This requires `annotationCollectionEnabled` to be `true`.                                                                                    | `keyword -> false`    |
| `locale`                      | The locale to use for generating messages in `Error`. Note that this value is copied from `SchemaRegistryConfig` for each execution.                                                                                            | `Locale.getDefault()` |
| `failFast`                    | Whether to return failure immediately when an assertion is generated. Note that this value is copied from `SchemaRegistryConfig` for each execution but is automatically set to `true` for the Boolean and Flag output formats. | `false`               |
| `formatAssertionsEnabled`     | The default is to generate format assertions from Draft 4 to Draft 7 and to only generate annotations from Draft 2019-09. Setting to `true` or `false` will override the default behavior.                                      | `null`                |
| `readOnly`                    | Used to indicate that the property should not be sent as part of the request payload, but only in the response payload. This affects the `readOnly` keyword used for the OpenAPI dialect.                                       | `null`                |
| `writeOnly`                   | Used to indicate that the property should not be sent as part of the response payload, but only in the request payload. This affects the `writeOnly` keyword used for the OpenAPI dialect.                                      | `null`                |

### Schema Registry Configuration

| Name                         | Description                                                                                                                                                                                                                                       | Default Value                               |
| ---------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------- |
| `cacheRefs`                  | Whether the schemas loaded from refs will be cached and reused for subsequent runs. Setting this to `false` will affect performance.                                                                                                              | `true`                                      |
| `errorMessageKeyword`        | The keyword to use for custom error messages in the schema. If not set this features is disabled. This is typically set to `errorMessage` or `message`.                                                                                           | `null`                                      |
| `executionContextCustomizer` | This can be used to customize the `ExecutionContext` generated by the `Schema` for each validation run.                                                                                                                                           | `null`                                      |
| `failFast`                   | Whether to return failure immediately when an assertion is generated.                                                                                                                                                                             | `false`                                     |
| `formatAssertionsEnabled`    | The default is to generate format assertions from Draft 4 to Draft 7 and to only generate annotations from Draft 2019-09. Setting to `true` or `false` will override the default behavior.                                                        | `null`                                      |
| `locale`                     | The locale to use for generating messages in `Error`.                                                                                                                                                                                             | `Locale.getDefault()`                       |
| `losslessNarrowing`          | Whether lossless narrowing is used for the `type` keyword. Since Draft 6 a value of `1.0` is interpreted as an integer whether or not this is enabled.                                                                                            | `false`                                     |
| `messageSource`              | This is used to retrieve the locale specific messages.                                                                                                                                                                                            | `DefaultMessageSource.getInstance()`        |
| `pathType`                   | The path type to use for reporting the instance location and evaluation path. Set to `PathType.JSON_PATH` to use JSON Path.                                                                                                                       | `PathType.JSON_POINTER`                     |
| `preloadSchema`              | Whether the schema will be preloaded before processing any input. This will use memory but the execution of the validation will be faster.                                                                                                        | `true`                                      |
| `regularExpressionFactory`   | The factory to use to create regular expressions for instance `JoniRegularExpressionFactory` or `GraalJSRegularExpressionFactory`. This requires the dependency to be manually added to the project or a `ClassNotFoundException` will be thrown. | `JDKRegularExpressionFactory.getInstance()` |
| `schemaIdValidator`          | This is used to customize how the `$id` values are validated. Note that the default implementation allows non-empty fragments where no base IRI is specified and also allows non-absolute IRI `$id` values in the root schema.                    | `JsonSchemaIdValidator.DEFAULT`             |
| `strict`                     | This is set whether keywords are strict in their validation. What this does depends on the individual validators.                                                                                                                                 |                                             |
| `typeLoose`                  | Whether types are interpreted in a loose manner. If set to true, a single value can be interpreted as a size 1 array. Strings may also be interpreted as number, integer or boolean.                                                              | `false`                                     |

### Walk Configuration

| Name                         | Description                                                                                 | Default Value                                         |
| ---------------------------- | ------------------------------------------------------------------------------------------- | ----------------------------------------------------- |
| `applyDefaultsStrategy`      | The strategy for applying defaults when walking when missing or null nodes are encountered. | `ApplyDefaultsStrategy.EMPTY_APPLY_DEFAULTS_STRATEGY` |
| `keywordWalkHandler`         | The `WalkHandler` triggered for keywords.                                                   | `NoOpWalkHandler.getInstance()`                       |
| `propertyWalkHandler`        | The `WalkHandler` triggered for properties.                                                 | `NoOpWalkHandler.getInstance()`                       |
| `itemWalkHandler`            | The `WalkHandler` triggered for items.                                                      | `NoOpWalkHandler.getInstance()`                       |

## Performance Considerations

Special attention should be made for inefficient schemas using deeply nested `oneOf` or `anyOf` that do not have a condition to short-circuit the evaluation using `if` and `then`. The validator has no choice but to perform all the evaluations, and the error messages would be typically very confusing as it will return all the messages from the children.

Instances for `SchemaRegistry` and the `Schema` created from it are designed to be thread-safe provided its configuration is not modified and should be cached and reused. Not reusing the `Schema` means that the schema data needs to be repeated parsed with validator instances created and references resolved. When references are resolved, the validators created will be cached.

Collecting annotations will adversely affect validation performance.

The earlier draft specifications contain less keywords that can potentially impact performance. For instance the use of the `unevaluatedProperties` or `unevaluatedItems` keyword will trigger annotation collection in the related validators, such as the `properties` or `items` validators.

This does not mean that using a schema with a later draft specification will automatically cause a performance impact. For instance, the `properties` validator will perform checks to determine if annotations need to be collected, and checks if the dialect contains the `unevaluatedProperties` keyword and whether the `unevaluatedProperties` keyword exists adjacent the evaluation path.

## Security Considerations

The library assumes that the schemas being loaded are trusted. This security model assumes the use case where the schemas are bundled with the application on the classpath.

| Issue               | Description                                                                                                                                                                                                | Mitigation                                                                                                                                                                     |
| ------------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ |
| Schema Loading      | The library by default will load schemas from the classpath and over the internet if needed.                                                                                                               | The `SchemaLoader` can be configured to block or allow certain IRIs for schema retrieval.                                                                                      |
| Schema Caching      | The library by default preloads and caches references when loading schemas.                                                                                                                                | Set `cacheRefs` option in `SchemaRegistryConfig` to false.                                                                                                                     |
| Regular Expressions | The library does not validate if a given regular expression is susceptable to denial of service ([ReDoS](https://owasp.org/www-community/attacks/Regular_expression_Denial_of_Service_-_ReDoS)).           | An `AllowRegularExpressionFactory` can be configured to perform validation on the regular expressions that are allowed.                                                        |
| Validation Errors   | The library by default attempts to return all validation errors. The use of applicators such as `allOf` with a large number of schemas may result in a large number of validation errors taking up memory. | Set `failFast` option in `SchemaRegistryConfig` to immediately return when the first error is encountered. The `OutputFormat.BOOLEAN` or `OutputFormat.FLAG` also can be used. |

## [Quick Start](doc/quickstart.md)

## [Customizing Schema Retrieval](doc/schema-retrieval.md)

## [Customizing Dialects, Vocabularies, Keywords and Formats](doc/custom-dialect.md)

## [OpenAPI Specification](doc/openapi.md)

## [Schema Walkers](doc/walkers.md)

## [Regular Expressions](doc/ecma-262.md)

## [Custom Error Messages](doc/error-message.md)

## [Multiple Language](doc/multiple-language.md)

## [MetaSchema Validation](doc/metaschema-validation.md)

## Projects

The [light-rest-4j](https://github.com/networknt/light-rest-4j), [light-graphql-4j](https://github.com/networknt/light-graphql-4j) and [light-hybrid-4j](https://github.com/networknt/light-hybrid-4j) use this library to validate the request and response based on the specifications. If you are using other frameworks like Spring Boot, you can use the [OpenApiValidator](https://github.com/mservicetech/openapi-schema-validation), a generic OpenAPI 3.0 validator based on the OpenAPI 3.0 specification.

If you have a project using this library, please submit a PR to add your project below.

- [mpenet/legba](https://github.com/mpenet/legba/) - OpenAPI service library for clojure, adhering to the [RING spec](https://github.com/ring-clojure/ring)

## Contributors

Thanks to the following people who have contributed to this project. If you are using this library, please consider to be a sponsor for one of the contributors.

[@stevehu](https://github.com/sponsors/stevehu)

[@justin-tay](https://github.com/justin-tay)

[@fdutton](https://github.com/fdutton)

[@prashanth-chaitanya](https://github.com/prashanth-chaitanya)

[@BalloonWen](https://github.com/BalloonWen)

[@jiachen1120](https://github.com/jiachen1120)

[@valfirst](https://github.com/valfirst)

[@ddobrin](https://github.com/ddobrin)

[@eskabetxe](https://github.com/eskabetxe)

[@ehrmann](https://github.com/ehrmann)

[@prashanthjos](https://github.com/prashanthjos)

[@Subhajitdas298](https://github.com/Subhajitdas298)

[@FWiesner](https://github.com/FWiesner)

[@jawaff](https://github.com/jawaff)

[@aznan2](https://github.com/aznan2)

[@rhwood](https://github.com/rhwood)

For all contributors, please visit https://github.com/networknt/json-schema-validator/graphs/contributors

If you are a contributor, please join the [GitHub Sponsors](https://github.com/sponsors) and switch the link to your sponsors dashboard via a PR.

## Sponsors

### Individual Sponsors

### Corporation Sponsors
 readmeEtag: '"d7378e4bd4e007f4c748c3bfc12c2679dfc826d1"' readmeLastModified: Mon, 05 Jan 2026 03:02:31 GMT repositoryId: 68291529 description: >- A fast Java JSON schema validator that supports draft V4, V6, V7, V2019-09 and V2020-12 created: '2016-09-15T12:24:16Z' updated: '2026-02-03T15:53:12Z' language: Java archived: false stars: 1026 watchers: 38 forks: 335 owner: networknt logo: https://avatars.githubusercontent.com/u/8740739?v=4 license: Apache-2.0 repoEtag: '"03e28aed8dbe5d7877077d798d734bdb1b0a07863b350d205c5c1718329a5d26"' repoLastModified: Tue, 03 Feb 2026 15:53:12 GMT foundInMaster: true category: Parsers id: 69c6f349899d198b4045edb7ffa3fb74 - source: openapi3 tags repository: https://github.com/openapitools/openapi-diff v3: true repositoryMetadata: base64Readme: >- # OpenAPI-diff 

Compare two OpenAPI specifications (3.x) and render the difference to HTML plain text, Markdown files, or JSON files.

[![Build](https://github.com/OpenAPITools/openapi-diff/workflows/Main%20Build/badge.svg)](https://github.com/OpenAPITools/openapi-diff/actions?query=branch%3Amaster+workflow%3A"Main+Build")
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=OpenAPITools_openapi-diff&metric=alert_status)](https://sonarcloud.io/dashboard?id=OpenAPITools_openapi-diff)

[![Maven Central](https://img.shields.io/maven-central/v/org.openapitools.openapidiff/openapi-diff-core)](https://search.maven.org/artifact/org.openapitools.openapidiff/openapi-diff-core)

[![Contribute with Gitpod](https://img.shields.io/badge/Contribute%20with-Gitpod-908a85?logo=gitpod)](https://gitpod.io/#https://github.com/OpenAPITools/openapi-diff)
[![Join the Slack chat room](https://img.shields.io/badge/Slack-Join%20the%20chat%20room-orange)](https://join.slack.com/t/openapi-generator/shared_invite/zt-12jxxd7p2-XUeQM~4pzsU9x~eGLQqX2g)

[![Docker Automated build](https://img.shields.io/docker/automated/openapitools/openapi-diff)](https://hub.docker.com/r/openapitools/openapi-diff)
[![Docker Image Version](https://img.shields.io/docker/v/openapitools/openapi-diff?sort=semver)](https://hub.docker.com/r/openapitools/openapi-diff/tags)

# Requirements

* Java 8

# Feature

* Supports OpenAPI spec v3.0.
* In-depth comparison of parameters, responses, endpoint, http method (GET,POST,PUT,DELETE...)
* Supports swagger api Authorization
* Render difference of property with Expression Language
* HTML, Markdown, Asciidoc & JSON render

# Maven

Available on [Maven Central](https://search.maven.org/artifact/org.openapitools.openapidiff/openapi-diff-core)

```xml
<dependency>
  <groupId>org.openapitools.openapidiff</groupId>
  <artifactId>openapi-diff-core</artifactId>
  <version>${openapi-diff-version}</version>
</dependency>
```

# Homebrew
Available for Mac users on [brew](https://formulae.brew.sh/formula/openapi-diff)

```bash
brew install openapi-diff
```
Usage instructions in [Usage -> Command line](#command-line)

# Docker

Available on [Docker Hub](https://hub.docker.com/r/openapitools/openapi-diff/) as `openapitools/openapi-diff`.

```bash
# docker run openapitools/openapi-diff:latest
usage: openapi-diff <old> <new>
    --asciidoc <file>           export diff as asciidoc in given file
    --debug                     Print debugging information
    --error                     Print error information
    --fail-on-changed           Fail if API changed but is backward
                                compatible
    --fail-on-incompatible      Fail only if API changes broke backward
                                compatibility
    --config-file               Config file to override default behavior. Supported file formats: .yaml
    --config-prop               Config property to override default behavior with key:value format (e.g. my.prop:true)
 -h,--help                      print this message
    --header <property=value>   use given header for authorisation
    --html <file>               export diff as html in given file
    --info                      Print additional information
    --json <file>               export diff as json in given file
 -l,--log <level>               use given level for log (TRACE, DEBUG,
                                INFO, WARN, ERROR, OFF). Default: ERROR
    --markdown <file>           export diff as markdown in given file
    --off                       No information printed
    --query <property=value>    use query param for authorisation
    --state                     Only output diff state: no_changes,
                                incompatible, compatible
    --text <file>               export diff as text in given file
    --trace                     be extra verbose
    --version                   print the version information and exit
    --warn                      Print warning information
```

## Build the image

This is only required if you want to try new changes in the Dockerfile of this project.

```bash
docker build -t local-openapi-diff .
```

You can replace the local image name `local-openapi-diff` by any name of your choice.

## Run an instance

In this example the `$(pwd)/core/src/test/resources` directory is mounted in the `/specs` directory of the container
in readonly mode (`ro`).

```bash
docker run --rm -t \
  -v $(pwd)/core/src/test/resources:/specs:ro \
  openapitools/openapi-diff:latest /specs/path_1.yaml /specs/path_2.yaml
```

The remote name `openapitools/openapi-diff` can be replaced with `local-openapi-diff` or the name you gave to your local image.

# Usage

openapi-diff can read OpenAPI specs from JSON files or HTTP URLs.

## Command Line

```bash
$ openapi-diff --help
usage: openapi-diff <old> <new>
    --asciidoc <file>           export diff as asciidoc in given file
    --debug                     Print debugging information
    --error                     Print error information
 -h,--help                      print this message
    --header <property=value>   use given header for authorisation
    --html <file>               export diff as html in given file
    --info                      Print additional information
    --json <file>               export diff as json in given file
 -l,--log <level>               use given level for log (TRACE, DEBUG,
                                INFO, WARN, ERROR, OFF). Default: ERROR
    --markdown <file>           export diff as markdown in given file
    --off                       No information printed
    --query <property=value>    use query param for authorisation
    --state                     Only output diff state: no_changes,
                                incompatible, compatible
    --fail-on-incompatible      Fail only if API changes broke backward compatibility
    --fail-on-changed           Fail if API changed but is backward compatible
    --config-file               Config file to override default behavior. Supported file formats: .yaml
    --config-prop               Config property to override default behavior with key:value format (e.g. my.prop:true)
    --trace                     be extra verbose
    --version                   print the version information and exit
    --warn                      Print warning information
```

## Maven Plugin

Add openapi-diff to your POM to show diffs when you test your Maven project. You may opt to throw an error if you have broken backwards compatibility or if your API has changed.  

```xml
<plugin>
  <groupId>org.openapitools.openapidiff</groupId>
  <artifactId>openapi-diff-maven</artifactId>
  <version>${openapi-diff-version}</version>
  <executions>
    <execution>
      <goals>
        <goal>diff</goal>
      </goals>
      <configuration>
        <!-- Reference specification (perhaps your prod schema) -->
        <oldSpec>https://petstore3.swagger.io/api/v3/openapi.json</oldSpec>
        <!-- Specification generated by your project in the compile phase -->
        <newSpec>${project.basedir}/target/openapi.yaml</newSpec>
        <!-- Fail only if API changes broke backward compatibility (default: false) -->
        <failOnIncompatible>true</failOnIncompatible>
        <!-- Fail if API changed (default: false) -->
        <failOnChanged>true</failOnChanged>
        <!-- Supply file path for console output to file if desired. -->
        <consoleOutputFileName>${project.basedir}/../maven/target/diff.txt</consoleOutputFileName>
        <!-- Supply file path for json output to file if desired. -->
        <jsonOutputFileName>${project.basedir}/../maven/target/diff.json</jsonOutputFileName>
        <!-- Supply file path for markdown output to file if desired. -->
        <markdownOutputFileName>${project.basedir}/../maven/target/diff.md</markdownOutputFileName>
        <!-- Supply config file(s), e.g. to disable incompatibility checks. Later files override earlier files -->
        <configFiles>
          <configFile>my/config-file.yaml</configFile>
        </configFiles>
        <!-- Supply config properties, e.g. to disable incompatibility checks. Overrides configFiles. -->
        <configProps>
          <incompatible.response.enum.increased>false</incompatible.response.enum.increased>
        </configProps>
      </configuration>
    </execution>
  </executions>
</plugin>
```

## Direct Invocation

```java
public class Main {
    public static final String OPENAPI_DOC1 = "petstore_v3_1.json";
    public static final String OPENAPI_DOC2 = "petstore_v2_2.yaml";
        
    public static void main(String[] args) {
        ChangedOpenApi diff = OpenApiCompare.fromLocations(OPENAPI_DOC1, OPENAPI_DOC2);

        //...
    }
}
```

### Path Matching while comparing two OpenAPI paths

Path matching controls how paths from the old and new specs are paired during comparison (PathsDiff.java). The default matcher (DefaultPathMatcher) obfuscates path parameter names, meaning `/users/{id}` matches `/users/{userId}`. The default matcher fails on ambiguous signatures if the spec contains a few paths semantically identical. In case this behaviour is not fitting your use case, you can implement your own matching strategy.

You can plug in a custom matcher via `OpenApiDiffOptions` implementing the `PathMatcher` interface:

```java
OpenApiDiffOptions options = OpenApiDiffOptions
    .builder()
    .pathMatcher(new MyCustomPathMatcher())
    .build();

ChangedOpenApi diff = OpenApiCompare.fromLocations(oldSpec, newSpec, null, options);
```

### Render difference

---
#### HTML

```java
HtmlRender htmlRender = new HtmlRender("Changelog", "http://deepoove.com/swagger-diff/stylesheets/demo.css");
FileOutputStream outputStream = new FileOutputStream("testDiff.html");
OutputStreamWriter outputStreamWriter = new OutputStreamWriter(outputStream);
htmlRender.render(diff, outputStreamWriter);
```

#### Markdown

```java
MarkdownRender markdownRender = new MarkdownRender();
FileOutputStream outputStream = new FileOutputStream("testDiff.md");
OutputStreamWriter outputStreamWriter = new OutputStreamWriter(outputStream);
markdownRender.render(diff, outputStreamWriter);
```

#### Asciidoc

```java
AsciidocRender asciidocRender = new AsciidocRender();
FileOutputStream outputStream = new FileOutputStream("testDiff.adoc");
OutputStreamWriter outputStreamWriter = new OutputStreamWriter(outputStream);
asciidocRender.render(diff, outputStreamWriter);
```

#### JSON

```java
JsonRender jsonRender = new JsonRender();
FileOutputStream outputStream = new FileOutputStream("testDiff.json");
OutputStreamWriter outputStreamWriter = new OutputStreamWriter(outputStream);
jsonRender.render(diff, outputStreamWriter);
```

### Extensions

This project uses Java Service Provider Interface (SPI) so additional extensions can be added. 

To build your own extension, you simply need to create a `src/main/resources/META-INF/services/org.openapitools.openapidiff.core.compare.ExtensionDiff` file with the full classname of your implementation.
Your class must also implement the `org.openapitools.openapidiff.core.compare.ExtensionDiff` interface.
Then, including your library with the `openapi-diff` module will cause it to be triggered automatically.

# Examples

### CLI Output

```text
==========================================================================
==                            API CHANGE LOG                            ==
==========================================================================
                             Swagger Petstore                             
--------------------------------------------------------------------------
--                              What's New                              --
--------------------------------------------------------------------------
- GET    /pet/{petId}

--------------------------------------------------------------------------
--                            What's Deleted                            --
--------------------------------------------------------------------------
- POST   /pet/{petId}

--------------------------------------------------------------------------
--                          What's Deprecated                           --
--------------------------------------------------------------------------
- GET    /user/logout

--------------------------------------------------------------------------
--                            What's Changed                            --
--------------------------------------------------------------------------
- PUT    /pet
  Request:
        - Deleted application/xml
        - Changed application/json
          Schema: Backward compatible
- POST   /pet
  Parameter:
    - Add tags in query
  Request:
        - Changed application/xml
          Schema: Backward compatible
        - Changed application/json
          Schema: Backward compatible
- GET    /pet/findByStatus
  Parameter:
    - Deprecated status in query
  Return Type:
    - Changed 200 OK
      Media types:
        - Changed application/xml
          Schema: Broken compatibility
        - Changed application/json
          Schema: Broken compatibility
- GET    /pet/findByTags
  Return Type:
    - Changed 200 OK
      Media types:
        - Changed application/xml
          Schema: Broken compatibility
        - Changed application/json
          Schema: Broken compatibility
- DELETE /pet/{petId}
  Parameter:
    - Add newHeaderParam in header
- POST   /pet/{petId}/uploadImage
  Parameter:
    - Changed petId in path
- POST   /user
  Request:
        - Changed application/json
          Schema: Backward compatible
- POST   /user/createWithArray
  Request:
        - Changed application/json
          Schema: Backward compatible
- POST   /user/createWithList
  Request:
        - Changed application/json
          Schema: Backward compatible
- GET    /user/login
  Parameter:
    - Delete password in query
- GET    /user/logout
- GET    /user/{username}
  Return Type:
    - Changed 200 OK
      Media types:
        - Changed application/xml
          Schema: Broken compatibility
        - Changed application/json
          Schema: Broken compatibility
- PUT    /user/{username}
  Request:
        - Changed application/json
          Schema: Backward compatible
--------------------------------------------------------------------------
--                                Result                                --
--------------------------------------------------------------------------
                 API changes broke backward compatibility                 
--------------------------------------------------------------------------
```

### Markdown

```markdown
### What's New
---
* `GET` /pet/{petId} Find pet by ID

### What's Deleted
---
* `POST` /pet/{petId} Updates a pet in the store with form data

### What's Deprecated
---
* `GET` /user/logout Logs out current logged in user session

### What's Changed
---
* `PUT` /pet Update an existing pet  
    Request

        Deleted request body : [application/xml]
        Changed response : [application/json]
* `POST` /pet Add a new pet to the store  
    Parameter

        Add tags //add new query param demo
    Request

        Changed response : [application/xml]
        Changed response : [application/json]
* `GET` /pet/findByStatus Finds Pets by status  
    Parameter

    Return Type

        Changed response : [200] //successful operation
* `GET` /pet/findByTags Finds Pets by tags  
    Return Type

        Changed response : [200] //successful operation
* `DELETE` /pet/{petId} Deletes a pet  
    Parameter

        Add newHeaderParam
* `POST` /pet/{petId}/uploadImage uploads an image for pet  
    Parameter

        petId Notes ID of pet to update change into ID of pet to update, default false
* `POST` /user Create user  
    Request

        Changed response : [application/json]
* `POST` /user/createWithArray Creates list of users with given input array  
    Request

        Changed response : [application/json]
* `POST` /user/createWithList Creates list of users with given input array  
    Request

        Changed response : [application/json]
* `GET` /user/login Logs user into the system  
    Parameter

        Delete password //The password for login in clear text
* `GET` /user/logout Logs out current logged in user session  
* `PUT` /user/{username} Updated user  
    Request

        Changed response : [application/json]
* `GET` /user/{username} Get user by user name  
    Return Type

        Changed response : [200] //successful operation
```

### JSON

```json
{
    "changedElements": [...],
    "changedExtensions": null,
    "changedOperations": [...],
    "compatible": false,
    "deprecatedEndpoints": [...],
    "different": true,
    "incompatible": true,
    "missingEndpoints": [...],
    "newEndpoints": [
        {
            "method": "GET",
            "operation": {
                "callbacks": null,
                "deprecated": null,
                "description": "Returns a single pet",
                "extensions": null,
                "externalDocs": null,
                "operationId": "getPetById",
                "parameters": [
                    {
                        "$ref": null,
                        "allowEmptyValue": null,
                        "allowReserved": null,
                        "content": null,
                        "deprecated": null,
                        "description": "ID of pet to return",
                        "example": null,
                        "examples": null,
                        "explode": false,
                        "extensions": null,
                        "in": "path",
                        "name": "petId",
                        "required": true,
                        "schema": {...},
                        "style": "SIMPLE"
                    }
                ],
                "requestBody": null,
                "responses": {...},
                "security": [
                    {
                        "api_key": []
                    }
                ],
                "servers": null,
                "summary": "Find pet by ID",
                "tags": [
                    "pet"
                ]
            },
            "path": null,
            "pathUrl": "/pet/{petId}",
            "summary": "Find pet by ID"
        }
    ],
    "newSpecOpenApi": {...},
    "oldSpecOpenApi": {...},
    "unchanged": false
}
```

# License

openapi-diff is released under the Apache License 2.0.

# Thanks

* Adarsh Sharma / [adarshsharma](https://github.com/adarshsharma)
* Quentin Desramé / [quen2404](https://github.com/quen2404)
* [Sayi](https://github.com/Sayi) for his project [swagger-diff](https://github.com/Sayi/swagger-diff) 
  which was a source of inspiration for this tool
 readmeEtag: '"8da667ec1f567f7eee7f7df8ba4e60ef7937f1eb"' readmeLastModified: Mon, 13 Oct 2025 11:21:49 GMT repositoryId: 113924383 description: Utility for comparing two OpenAPI specifications. created: '2017-12-12T00:39:30Z' updated: '2026-01-30T20:51:50Z' language: Java archived: false stars: 1044 watchers: 20 forks: 181 owner: OpenAPITools logo: https://avatars.githubusercontent.com/u/37325267?v=4 license: Apache-2.0 repoEtag: '"24f0fa5f61128f1c505f9e12bf4622ec3d308f9823901911713ed3d99c09a215"' repoLastModified: Fri, 30 Jan 2026 20:51:50 GMT foundInMaster: true category: Parsers id: c128907e9d949a67c1bda0e0ceda755a oldLocations: - https://github.com/quen2404/openapi-diff - source: openapi3 tags repository: https://github.com/stoplightio/studio v3: true repositoryMetadata: base64Readme: >- IyBTdG9wbGlnaHQgU3R1ZGlvIAoKU3R1ZGlvIGlzIFN0b3BsaWdodCdzIG5leHQgZ2VuZXJhdGlvbiBhcHBsaWNhdGlvbiBmb3IgQVBJIGRlc2lnbiwgbW9kZWxpbmcsIGFuZCB0ZWNobmljYWwgd3JpdGluZy4gQSBwcmltYXJ5IGdvYWwgb2YgU3R1ZGlvIGlzIHRvIGVucmljaCwgbm90IHJlcGxhY2UsIHlvdXIgZXhpc3Rpbmcgd29ya2Zsb3dzLiBXaGVuIHJ1bm5pbmcgbG9jYWxseSBpdCB3b3JrcyBmdWxseSBvZmZsaW5lLCB3aXRoIGZvbGRlcnMgYW5kIGZpbGVzIG9uIHlvdXIgY29tcHV0ZXIganVzdCBsaWtlIHlvdXIgZmF2b3JpdGUgSURFLiBXaGVuIHJ1bm5pbmcgaW4gdGhlIGJyb3dzZXIsIHRoZSB3ZWItbmF0aXZlIEdpdCBzdXBwb3J0IGFsbG93cyB5b3UgdG8gZWZmb3J0bGVzc2x5IHdvcmsgd2l0aCB5b3VyIGV4aXN0aW5nIHJlcG9zaXRvcmllcyBzYWZlbHkgYW5kIGVmZmljaWVudGx5LgoKIyMgRG9jdW1lbnRhdGlvbgoKVG8gbGVhcm4gbW9yZSBhYm91dCBTdHVkaW8gYW5kIHRoZSBTdG9wbGlnaHQgcGxhdGZvcm0sIHNlZSBvdXIgW1BsYXRmb3JtIERvY3VtZW50YXRpb25dKGh0dHBzOi8vbWV0YS5zdG9wbGlnaHQuaW8vKS4KCiMjIEZlYXR1cmVzCgojIyMgRnVsbCBTdXBwb3J0IGZvciBPcGVuQVBJIHYyIGFuZCB2MwoKU3R1ZGlvIGNvbWVzIHdpdGggZnVsbCBzdXBwb3J0IGZvciB0aGUgT3BlbkFQSSB2ZXJzaW9ucyAyIGFuZCAzIHNwZWNpZmljYXRpb24gZm9ybWF0cyBmb3IgYWxsIGZ1bmN0aW9uYWxpdHkuIFRoYXQgbWVhbnMgZnVsbCB2YWxpZGF0aW9uLCBtb2NraW5nLCBhbmQgbW9kZWxpbmcgc3VwcG9ydCBmb3IgYm90aCB2ZXJzaW9ucyBvZiB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLgoKIVtTdHVkaW8gbG92ZXMgU3dhZ2dlciArIE9wZW5BUEldKGFzc2V0cy9pbWFnZXMvb3BlbmFwaV9zd2FnZ2VyX2VxdWFsX2hlYXJ0LnBuZykKCiMjIyBHcmFwaGljYWwgQVBJIERlc2lnbgoKRm9ybS1iYXNlZCBkZXNpZ25pbmcgbWVhbnMgeW91IGRvbid0IG5lZWQgdG8gYmUgYW4gT3BlbkFQSSBleHBlcnQgdG8gZ2V0IHN0YXJ0ZWQuIFN0dWRpbyBoYXMgYSAid3JpdGUiIChjb2RlKSBtb2RlIHdpdGggZnVsbCBPcGVuQVBJIGF1dG9jb21wbGV0ZSwgYW5kIGEgInJlYWQiIG1vZGUgZm9yIHZpc3VhbGl6aW5nIEhUVFAgb3BlcmF0aW9ucyBhbmQgbW9kZWxzLgoKIVtHcmFwaGljYWwgRGVzaWduIGZvciBPcGVuQVBJXShhc3NldHMvaW1hZ2VzL2Zvcm0tZWRpdG9yLnBuZykKClRvIGZpbmQgb3V0IG1vcmUgYWJvdXQgaG93IHlvdSBjYW4gcXVpY2tseSBkZXNpZ24gYW5kIHByb3RvdHlwZSBBUElzIHdpdGhvdXQgd3JpdGluZyBhIHNpbmdsZSBsaW5lIG9mIEpTT04gb3IgWUFNTCwgc2VlIFtoZXJlXShodHRwczovL21ldGEuc3RvcGxpZ2h0LmlvL2RvY3MvcGxhdGZvcm0vWkc5ak9qTTJPVE0zTWpnNS1vdmVydmlldykuCgojIyMgSlNPTiBTY2hlbWEgTW9kZWxpbmcKClN0dWRpbyBpcyBub3QganVzdCBmb3IgQVBJcywgeW91IGNhbiBhbHNvIGNyZWF0ZSBhbmQgbW9kaWZ5IHN0YW5kYWxvbmUgSlNPTiBTY2hlbWEgZmlsZXMgZm9yIHN0b3JpbmcgZGF0YSBtb2RlbHMuIERpZCB3ZSBtZW50aW9uIHRoYXQgdGhpcyBpcyBhbHNvIHBvd2VyZWQgYnkgdGhlIGdyYXBoaWNhbCBpbnRlcmZhY2U/IE5vIG1vcmUgc2VhcmNoaW5nIGZvciB0aGUgbWF0Y2hpbmcgYnJhY2tldCBvciB0aGUgbWlzc2luZyBzcGFjZSwgYW5kIG5vIG5lZWQgdG8gYmUgZmFtaWxpYXIgd2l0aCB0aGUgaW4ncyBhbmQgb3V0J3Mgb2YgSlNPTiBTY2hlbWEuCgohW10oYXNzZXRzL2ltYWdlcy9qc2Utc2FtcGxlLnBuZykKCiMjIyBFbmNvdXJhZ2UgUmV1c2UsIEF2b2lkIER1cGxpY2F0aW9uCgpXaGVuIGl0IGNvbWVzIHRvIEFQSSBtb2RlbGluZywgYXZvaWRpbmcgZHVwbGljYXRpb24gb2YgZWZmb3J0IGNhbiBiZSB0aGUgZGlmZmVyZW5jZSBiZXR3ZWVuIHN1Y2Nlc3MgYW5kIGZhaWx1cmUuIEhvdyBjYW4geW91IGVuZm9yY2UgY29uc2lzdGVuY3kgaWYgYWxsIG9mIHlvdXIgZW5kcG9pbnRzIHJlLWNyZWF0ZSB0aGUgc2FtZSBtb2RlbCBpbiBzbGlnaHRseSBkaWZmZXJlbnQgd2F5cz8gKGhpbnQsIHlvdSBjYW4ndCkKCiFbU3R1ZGlvJ3MgZ3JhcGhpY2FsIEpTT04gc2NoZW1hIGVkaXRvciBhbGxvd3MgeW91IHRvIHF1aWNrbHkgZmluZCBtb2RlbHMgdG8gcmVmZXJlbmNlXShhc3NldHMvaW1hZ2VzL2pzZS1zYW1wbGUyLnBuZykKClN0dWRpbyBhbGxvd3MgeW91IHRvIHF1aWNrbHkgYW5kIGVhc2lseSBmaW5kIGFuZCByZXVzZSB0aGUgb2JqZWN0cyB5b3UgbmVlZCwgYXMgeW91IG5lZWQgdGhlbS4gTm8gbW9yZSByZWNyZWF0aW5nIG1vZGVscyBmb3IgZGlmZmVyZW50IGVuZHBvaW50cywgbm8gbW9yZSBoYXZpbmcgdG8gdXBkYXRlIGRvemVucyBvZiBkaWZmZXJlbnQgZW5kcG9pbnRzIGJlY2F1c2UgYSBuZXcgZmllbGQgd2FzIGFkZGVkLgoKVG8gZmluZCBvdXQgbW9yZSBhYm91dCBob3cgeW91IGNhbiBsZXZlcmFnZSByZWZlcmVuY2VzIHRvIHNjYWxlIHlvdXIgQVBJIGNvbnNpc3RlbmN5LCBzZWUgW2hlcmVdKGh0dHBzOi8vbWV0YS5zdG9wbGlnaHQuaW8vZG9jcy9wbGF0Zm9ybS9aRzlqT2pJeU5UWTVNekl3LXNoYXJlZC1jb21wb25lbnRzKS4KCiMjIyBUZWNobmljYWwgRG9jdW1lbnRhdGlvbgoKTWl4IEFQSSBSZWZlcmVuY2UgRG9jdW1lbnRhdGlvbiBhbmQgTWFya2Rvd24tYmFzZWQgZ3VpZGVzLCBob3ctdG9zLCBnZXR0aW5nIHN0YXJ0ZWQgaW5mb3JtYXRpb24sIGV0Yy4gQWxsIG9mIHlvdXIgZG9jdW1lbnRhdGlvbiBjYW4gbGl2ZSB0b2dldGhlciBpbiB0aGUgc2FtZSBwcm9qZWN0LiBTdHVkaW8gaW5jbHVkZXMgYSBidWlsdC1pbiBNYXJrZG93biBlZGl0b3IsIGltYWdlIG1hbmFnZXIsIGFuZCB0aGUgYWJpbGl0eSB0byBwdWJsaXNoIGRvY3VtZW50YXRpb24gdG8gU3RvcGxpZ2h0J3MgbmV3IGRvY3VtZW50YXRpb24gcGxhdGZvcm0uCgohW0NyZWF0ZSBiZWF1dGlmdWwgYW5kIGVhc3ktdG8tdXNlIEFQSSByZWZlcmVuY2UgZG9jdW1lbnRhdGlvbl0oYXNzZXRzL2ltYWdlcy90ZWNobmljYWwtZG9jdW1lbnRhdGlvbi5wbmcpCgpZb3UgY2FuIGV2ZW4gaG9zdCB0aGUgZmlsZXMgaW4geW91ciBvd24gR2l0IHJlcG9zaXRvcnksIGFuZCB0aGVuIHB1Ymxpc2ggd2hlbiB5b3UncmUgcmVhZHkgdG8gc2hvdyBvZmYgeW91ciBsYXRlc3QgYW5kIGdyZWF0ZXN0LgoKVG8gZmluZCBvdXQgbW9yZSBhYm91dCB3cml0aW5nIHRlY2huaWNhbCBkb2N1bWVudGF0aW9uIGluIFN0dWRpbywgc2VlIG91ciBnZXR0aW5nIHN0YXJ0ZWQgZ3VpZGUgW2hlcmVdKGh0dHBzOi8vbWV0YS5zdG9wbGlnaHQuaW8vZG9jcy9wbGF0Zm9ybS9aRzlqT2pRMk9URTROamszLWRvY3VtZW50YXRpb24tcXVpY2stc3RhcnQpLgoKIyMjIFN0eWxlIEd1aWRlcyBhbmQgVmFsaWRhdGlvbgoKRW5mb3JjZSBjb3JyZWN0bmVzcyBhbmQgYmVzdCBwcmFjdGljZXMgd2l0aCBbU3R5bGUgR3VpZGVzXShodHRwczovL21ldGEuc3RvcGxpZ2h0LmlvL2RvY3MvcGxhdGZvcm0vWkc5ak9qTXdNREkzTWpBMi1zdHlsZS1ndWlkZS1vdmVydmlldyksIHBvd2VyZWQgYnkgdGhlIG5hdGl2ZSBbU3BlY3RyYWxdKGh0dHBzOi8vc3RvcGxpZ2h0LmlvL3NwZWN0cmFsLz91dG1fc291cmNlPWdpdGh1Yi5jb20mdXRtX21lZGl1bT1yZWZlcnJhbCZ1dG1fY2FtcGFpZ249Z2l0aHViX3JlcG9fc3R1ZGlvKSBpbnRlZ3JhdGlvbiB0aGF0IGFsZXJ0cyB5b3UgdG8gZXJyb3JzIHRoZSBtb21lbnQgdGhleSBhcmUgY3JlYXRlZC4KCiFbU3BlY3RyYWwgdmFsaWRhdGVzIGFuZCBsaW50cyB5b3VyIEFQSXMgdG8gZW5zdXJlIHRoZXkgYXJlIGNvcnJlY3QgYW5kIGZ1bmN0aW9uYWxdKGFzc2V0cy9pbWFnZXMvc3BlY3RyYWwxLnBuZykKCkNsaWNraW5nIG9uIGVycm9ycyBvciB3YXJuaW5ncyBhbHNvIGJyaW5ncyB5b3UgdG8gZXhhY3RseSB3aGVyZSB0aGV5IGFyZSBsb2NhdGVkIGluIHRoZSBkb2N1bWVudCwgbWFraW5nIGl0IGVhc3kgdG8gZml4IGVycm9ycyBhdCB0aGUgc291cmNlLgoKIyMjIEJ1aWx0LWluIE1vY2tpbmcKCldoZW4gcnVubmluZyBsb2NhbGx5LCBTdHVkaW8gd2lsbCBhdXRvbWF0aWNhbGx5IHN0YXJ0IGEgbG9jYWwgW1ByaXNtIG1vY2sgc2VydmVyXShodHRwczovL3N0b3BsaWdodC5pby9wcmlzbS8/dXRtX3NvdXJjZT1naXRodWIuY29tJnV0bV9tZWRpdW09cmVmZXJyYWwmdXRtX2NhbXBhaWduPWdpdGh1Yl9yZXBvX3N0dWRpbykgZm9yIGV2ZXJ5IEFQSSBkZWZpbmVkIGluIHlvdXIgcHJvamVjdCwgYW5kIGtlZXAgdGhhdCBtb2NrIHNlcnZlciB1cCB0byBkYXRlIGFzIHlvdSBjaGFuZ2UgeW91ciBkZXNpZ25zLgoKIVtNb2NraW5nIGFsbG93cyB5b3UgdG8gcXVpY2tseSB0ZXN0IHRoZSBsb29rIGFuZCBmZWVsIG9mIHlvdXIgQVBJIGJlZm9yZSBqdW1waW5nIGludG8gdGhlIGNvZGVdKGFzc2V0cy9pbWFnZXMvc3R1ZGlvLW1vY2tpbmcucG5nKQoKVG8gZmluZCBvdXQgbW9yZSBhYm91dCBQcmlzbSBhbmQgaG93IG1vY2tpbmcgY2FuIGJlIHVzZWQgdG8gc3RyZWFtbGluZSB5b3VyIEFQSSBkZXZlbG9wbWVudCBwcm9jZXNzLCBzZWUgW2hlcmVdKGh0dHBzOi8vbWV0YS5zdG9wbGlnaHQuaW8vZG9jcy9wbGF0Zm9ybS9aRzlqT2pNMk9UTTNNamc0LXdvcmstd2l0aC1tb2NrLXNlcnZlcnMpLgoKIyMjIEJyaW5nIFlvdXIgT3duIFJlcG9zaXRvcnkKClNpbmNlIFN0dWRpbyB3b3JrcyB3aXRoIHlvdXIgbG9jYWwgZmlsZSBzeXN0ZW0sIHlvdSBjYW4gb3BlbiB1cCB5b3VyIEFQSSBwcm9qZWN0cyBhbmQgc3RhcnQgYWRkaW5nIGRvY3MgYW5kIGRlc2lnbnMgYWxvbmdzaWRlIHRoZSBhY3R1YWwgaW1wbGVtZW50YXRpb24gdGhleSBhcmUgbWVhbnQgdG8gZGVzY3JpYmUuIE9uY2UgeW91J3JlIGRvbmUsIGNoZWNrIGl0IGFsbCBpbnRvIEdpdCB3aXRoIHlvdXIgZmF2b3JpdGUgR2l0IGNsaWVudCEKCiMjIExpY2Vuc2UKCjxhIHJlbD0ibGljZW5zZSIgaHJlZj0iaHR0cHM6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL2xpY2Vuc2VzL2J5LW5kLzQuMC8iPjxpbWcgYWx0PSJDcmVhdGl2ZSBDb21tb25zIExpY2Vuc2UiIHN0eWxlPSJib3JkZXItd2lkdGg6MCIgc3JjPSJodHRwczovL2kuY3JlYXRpdmVjb21tb25zLm9yZy9sL2J5LW5kLzQuMC84OHgzMS5wbmciIC8+PC9hPjxiciAvPjxzcGFuIHhtbG5zOmRjdD0iaHR0cDovL3B1cmwub3JnL2RjL3Rlcm1zLyIgcHJvcGVydHk9ImRjdDp0aXRsZSI+U3RvcGxpZ2h0IFN0dWRpbzwvc3Bhbj4gYnkgPGEgeG1sbnM6Y2M9Imh0dHBzOi8vY3JlYXRpdmVjb21tb25zLm9yZy9ucyMiIGhyZWY9Imh0dHBzOi8vc3RvcGxpZ2h0LmlvL3N0dWRpbz91dG1fc291cmNlPWdpdGh1Yi5jb20mdXRtX21lZGl1bT1yZWZlcnJhbCZ1dG1fY2FtcGFpZ249Z2l0aHViX3JlcG9fc3R1ZGlvIiBwcm9wZXJ0eT0iY2M6YXR0cmlidXRpb25OYW1lIiByZWw9ImNjOmF0dHJpYnV0aW9uVVJMIj5TdG9wbGlnaHQuaW88L2E+IGlzIGxpY2Vuc2VkIHVuZGVyIGEgPGEgcmVsPSJsaWNlbnNlIiBocmVmPSJodHRwczovL2NyZWF0aXZlY29tbW9ucy5vcmcvbGljZW5zZXMvYnktbmQvNC4wLyI+Q3JlYXRpdmUgQ29tbW9ucyBBdHRyaWJ1dGlvbi1Ob0Rlcml2YXRpdmVzIDQuMCBJbnRlcm5hdGlvbmFsIExpY2Vuc2U8L2E+Lgo= readmeEtag: '"70966732d8cb9dcc0d216fc328ccf33bb2ce4282"' readmeLastModified: Tue, 20 Sep 2022 18:48:05 GMT repositoryId: 192797744 description: The modern editor for API Design and Technical Writing. created: '2019-06-19T20:09:49Z' updated: '2026-02-03T02:38:41Z' language: null archived: false stars: 864 watchers: 27 forks: 55 owner: stoplightio logo: https://avatars.githubusercontent.com/u/10767217?v=4 license: NOASSERTION repoEtag: '"a8919da275a2fa1d422dbc8e6c9fe2cb987ef6056bd37b9e0c82fec5a915e988"' repoLastModified: Tue, 03 Feb 2026 02:38:41 GMT foundInMaster: true category: - Description Validators - Server Implementations id: 0e510c4e495bc626b13dba4aa09d83dc - source: - openapi3 tags - openapi31 tags repository: https://github.com/zaghaghi/openapi-tui v3: true v3_1: true id: ef37267c73c0fdc4c3902843cddc05c3 repositoryMetadata: base64Readme: >- IVtsb2dvXShzdGF0aWMvbG9nby5wbmcpCiMgb3BlbmFwaS10dWkKClshW0NJXShodHRwczovL2dpdGh1Yi5jb20vemFnaGFnaGkvb3BlbmFwaS10dWkvd29ya2Zsb3dzL0NJL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS96YWdoYWdoaS9vcGVuYXBpLXR1aS9hY3Rpb25zKQoKVGVybWluYWwgVUkgdG8gbGlzdCwgYnJvd3NlIGFuZCBydW4gQVBJcyBkZWZpbmVkIHdpdGggT3BlbkFQSSB2My4wIGFuZCB2My4xIHNwZWMuCgoKIyBVc2FnZQpgYGBiYXNoCuKdryBvcGVuYXBpLXR1aSAtLWhlbHAKVGhpcyBUVUkgYWxsb3dzIHlvdSB0byBsaXN0IGFuZCBicm93c2UgQVBJcyBkZXNjcmliZWQgYnkgdGhlIG9wZW5hcGkgc3BlY2lmaWNhdGlvbi4KClVzYWdlOiBvcGVuYXBpLXR1aSAtLWlucHV0IDxQQVRIPgoKT3B0aW9uczoKICAtaSwgLS1pbnB1dCA8UEFUSD4gIElucHV0IGZpbGUgb3IgdXJsLCBpbiBqc29uIG9yIHlhbWwgZm9ybWF0IHdpdGggb3BlbmFwaSBzcGVjaWZpY2F0aW9uCiAgLWgsIC0taGVscCAgICAgICAgICBQcmludCBoZWxwCiAgLVYsIC0tdmVyc2lvbiAgICAgICBQcmludCB2ZXJzaW9uCmBgYAoKIyMgRXhhbXBsZXMKYGBgYmFzaAojIG9wZW4gbG9jYWwgeWFtbCBmaWxlCuKdryBvcGVuYXBpLXR1aSAtaSBleGFtcGxlcy9zdHJpcGUvc3BlYy55bWwKCiMgb3BlbiBsb2NhbCBqc29uIGZpbGUK4p2vIG9wZW5hcGktdHVpIC1pIGV4YW1wbGVzL3BldHN0b3JlLmpzb24KCiMgb3BlbiByZW1vdGUgZmlsZQrina8gb3BlbmFwaS10dWkgLWkgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2dpdGh1Yi9yZXN0LWFwaS1kZXNjcmlwdGlvbi9tYWluL2Rlc2NyaXB0aW9ucy1uZXh0L2FwaS5naXRodWIuY29tL2FwaS5naXRodWIuY29tLnlhbWwKYGBgCgoKIyBEZW1vCiFbZGVtb10oc3RhdGljL2RlbW8uZ2lmKQoKIyBPdGhlciBGZWF0dXJlIEFuaW1hdGlvbnMKPGRldGFpbHM+CiAgPHN1bW1hcnk+U2hvdyBtb3JlPC9zdW1tYXJ5PgoKIyMgTmVzdGVkIENvbXBvbmVudHMKIVtuZXN0ZWQtcmVmcmVuY2VzXShzdGF0aWMvbmVzdGVkLXJlZnMuZ2lmKQoKIyMgRnVsbHNjcmVlbgohW2Z1bGxzY3JlZW5dKHN0YXRpYy9mdWxsc2NyZWVuLmdpZikKCiMjIFdlYmhvb2tzCiFbd2ViaG9va3NdKHN0YXRpYy93ZWJob29rcy5naWYpCgojIyBGaWx0ZXIKIVtmaWx0ZXJdKHN0YXRpYy9maWx0ZXIuZ2lmKQoKIyMgQ2FsbCBFbmRwb2ludHMKIVtjYWxsXShzdGF0aWMvY2FsbC5naWYpCgojIyBNdWx0aXBsZSBTZXJ2ZXIgU3VwcG9ydAohW2NhbGxdKHN0YXRpYy9zd2l0Y2gtc2VydmVyLmdpZikKCjwvZGV0YWlscz4KCjxiciAvPgoKCiMgSW5zdGFsbGF0aW9uCkluc3RhbGwgZnJvbSBzb3VyY2U6CmBgYGJhc2gK4p2vIGNhcmdvIGluc3RhbGwgb3BlbmFwaS10dWkKYGBgCk9yIGRvd25sb2FkIHByZS1idWlsdCBhcnRpZmFjdCBmcm9tIHJlbGVhc2UgcGFnZS4KCiMjIERvY2tlcgpKdXN0IHJ1biB0aGUgYXBwbGljYXRpb24gd2l0aCBkb2NrZXIuCgpgYGBiYXNoCiMgb3BlbiBsb2NhbCBmaWxlCuKdryBkb2NrZXIgcnVuIC0tcm0gLXRpIC12JChwd2QpL2V4YW1wbGVzOi9vcHQgemFnaGFnaGkvb3BlbmFwaS10dWkgLWkgL29wdC9wZXRzdG9yZS5qc29uCgojIG9wZW4gcmVtb3RlIGZpbGUK4p2vIGRvY2tlciBydW4gLS1ybSAtaXQgemFnaGFnaGkvb3BlbmFwaS10dWkgLWkgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2dpdGh1Yi9yZXN0LWFwaS1kZXNjcmlwdGlvbi9tYWluL2Rlc2NyaXB0aW9ucy1uZXh0L2FwaS5naXRodWIuY29tL2FwaS5naXRodWIuY29tLnlhbWwKYGBgCiMjIERpc3RybyBQYWNrYWdlcwoKPGRldGFpbHM+CiAgPHN1bW1hcnk+UGFja2FnaW5nIHN0YXR1czwvc3VtbWFyeT4KClshW1BhY2thZ2luZyBzdGF0dXNdKGh0dHBzOi8vcmVwb2xvZ3kub3JnL2JhZGdlL3ZlcnRpY2FsLWFsbHJlcG9zL29wZW5hcGktdHVpLnN2ZyldKGh0dHBzOi8vcmVwb2xvZ3kub3JnL3Byb2plY3Qvb3BlbmFwaS10dWkvdmVyc2lvbnMpCgo8L2RldGFpbHM+CgojIyMgQXJjaCBMaW51eAoKWW91IGNhbiBpbnN0YWxsIHVzaW5nIGBwYWNtYW5gIGFzIGZvbGxvd3M6CgpgYGBiYXNoCuKdryBwYWNtYW4gLVMgb3BlbmFwaS10dWkKYGBgCgojIyMgTml4T1MKCllvdSBjYW4gaW5zdGFsbCB0aGUgYG9wZW5hcGktdHVpYCBwYWNrYWdlIGRpcmVjdGx5IHdpdGggdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgYmFzaApuaXggcHJvZmlsZSBpbnN0YWxsIGdpdGh1Yjp6YWdoYWdoaS9vcGVuYXBpLXR1aQpgYGAKCllvdSBjYW4gYWxzbyBpbnN0YWxsIGBvcGVuYXBpLXR1aWAgYnkgYWRkaW5nIGl0IHRvIHlvdXIgYGNvbmZpZ3VyYXRpb24ubml4YCBmaWxlLgoKYGBgbml4CiMgZmxha2Uubml4Cgp7CiAgaW5wdXRzLm9wZW5hcGktdHVpLnVybCA9ICJnaXRodWI6emFnaGFnaGkvb3BlbmFwaS10dWkiOwogICMgLi4uCgogIG91dHB1dHMgPSB7bml4cGtncywgLi4ufSBAIGlucHV0czogewogICAgbml4b3NDb25maWd1cmF0aW9ucy48eW91ci1ob3N0bmFtZT4gPSBuaXhwa2dzLmxpYi5uaXhvc1N5c3RlbSB7CiAgICAgIHNwZWNpYWxBcmdzID0geyBpbmhlcml0IGlucHV0czsgfTsgIyB0aGlzIGlzIHRoZSBpbXBvcnRhbnQgcGFydAogICAgICBtb2R1bGVzID0gWwogICAgICAgIC4vY29uZmlndXJhdGlvbi5uaXgKICAgICAgXTsKICAgIH07CiAgfQp9CmBgYAoKVGhlbiwgYWRkIGBvcGVuYXBpLXR1aWAgdG8geW91ciBgY29uZmlndXJhdGlvbi5uaXhgCgpgYGBuaXgKIyBjb25maWd1cmF0aW9uLm5peAoKe2lucHV0cywgcGtncywgLi4ufTogewogIGVudmlyb25tZW50LnN5c3RlbVBhY2thZ2VzID0gd2l0aCBwa2dzOyBbCiAgICBpbnB1dHMub3BlbmFwaS10dWkucGFja2FnZXMuJHtwa2dzLnN5c3RlbX0ub3BlbmFwaS10dWkKICBdOwp9CmBgYAoKCiMgS2V5YmluZGluZ3MKCnwgS2V5IHwgQWN0aW9ufAp8Oi0tLS18Oi0tLS0tfAp8IGDihpJgLCBgbGB8IE1vdmUgdG8gbmV4dCBwYW5lIHwKfCBg4oaQYCwgYGhgIHwgTW92ZSB0byBwcmV2aW91cyBwYW5lIHwKfCBg4oaTYCwgYGpgIHwgTW92ZSBkb3duIGluIGxpc3RzIHwKfCBg4oaRYCwgYGtgIHwgTW92ZSB1cCBpbiBsaXN0cyB8CnwgYDEuLi45YCB8IE1vdmUgYmV0d2VlbiB0YWJzIHwKfCBgXWAgfCBNb3ZlIHRvIG5leHQgdGFiIHwKfCBgW2AgfCBNb3ZlIHRvIHByZXZpb3VzIHRhYiB8CnwgYGZgIHwgVG9nZ2xlIGZ1bGxzY3JlZW4gcGFuZXwKfCBgZ2AgfCBHbyBpbiBuZXN0ZWQgaXRlbXMgaW4gbGlzdHN8CnwgYHFgIHwgUXVpdHwKfCBgL2AgfCBGaWx0ZXIgYXBpc3wKfCBgOmAgfCBSdW4gY29tbWFuZHN8CnwgYEJhY2tzcGFjZWAsIGBiYCB8IEdldCBvdXQgb2YgbmVzdGVkIGl0ZW1zIGluIGxpc3RzfAoKIyBDb21tYW5kcyBNYWluIFBhZ2UKfCBDb21tYW5kIHwgRGVzY3JpcHRpb24gfAp8Oi0tLS0tLS0tfDotLS0tLS0tLS0tLS18CnwgYHFgIHwgUXVpdCB8CnwgYHJlcXVlc3RgLCBgcmAgfCBHbyB0byByZXF1ZXN0IHBhZ2V8CnwgYGhpc3RvcnlgIHwgUmVxdWVzdCBoaXN0b3J5fAoKIyBDb21tYW5kcyBSZXF1ZXN0IFBhZ2UKfCBDb21tYW5kIHwgRGVzY3JpcHRpb24gfAp8Oi0tLS0tLS0tfDotLS0tLS0tLS0tLS18CnwgYHFgIHwgUXVpdCB8CnwgYHNlbmRgLCBgc2AgfCBTZW5kIHJlcXVlc3QgfAp8IGBxdWVyeWAsIGBxYCB8IEFkZCBvciByZW1vdmUgcXVlcnkgc3RyaW5ncy4gc3ViLWNvbW1hbmRzIGFyZSBgYWRkYCBvciBgcm1gLiBlLmcuIGBxdWVyeSBhZGQgcGFnZWAgfAp8IGBoZWFkZXJgLCBgaGAgfCBBZGQgb3IgcmVtb3ZlIGhlYWRlcnMuIHN1Yi1jb21tYW5kcyBhcmUgYGFkZGAgb3IgYHJtYC4gZS5nLiBgaGVhZGVyIGFkZCB4LWFwaS1rZXlgIHwKfCBgcmVxdWVzdGAsIGByYCB8IExvYWQgcmVxdWVzdCBwYXlsb2FkLiBlLmcuIGByZXF1ZXN0IG9wZW4gL2hvbWUvaGFtZWQvcGF5bG9hZC5qc29uYCB8CnwgYHJlc3BvbnNlYCwgYHNgIHwgU2F2ZSByZXNwb25zZSBwYXlsb2FkIGUuZy8gYHJlc3BvbnNlIHNhdmUgL2hvbWUvaGFtZWQvcmVzdWx0Lmpzb25gIHwKCiMgRW52aXJvbm1lbnQgVmFyaWFibGVzCnwgVmFyaWFibGUgfCBEZXNjcmlwdGlvbiB8Cnw6LS0tLS0tLS0tfDotLS0tLS0tLS0tLS18CnwgYE9QRU5BUElfVFVJX0RFRkFVTFRfU0VSVkVSYCB8IEFkZCBhIGN1c3RvbSBzZXJ2ZXIgdXJsIHRvIHRoZSBsaXN0IG9mIHNlcnZlcnN8CgoKIyBJbXBsZW1lbnRlZCBGZWF0dXJlcwotIFtYXSBWaWV3ZXIKLSBbWF0gT3BlbkFQSSB2My4xCi0gW1hdIERpc3BsYXkgV2ViaG9va3MKLSBbWF0gRGlzcGxheSBJbmZvIGFuZCBWZXJzaW9uCi0gW1hdIEZpbHRlciBBUElzCi0gW1hdIFJlbW90ZSBBUEkgc3BlY2lmaWNhdGlvbgotIFtYXSBNZXJnZSBQYXJhbWV0ZXJzIEJhc2VkIG9uIGBpbmAKLSBbWF0gUGFuZSBGdWxsc2NyZWVuIE1vZGUKLSBbWF0gTmVzdGVkIENvbXBvbmVudHMKLSBbWF0gU3RhdHVzIExpbmUKLSBbWF0gUGhvbmUgUGFnZQotIFtYXSBDYWxsIEhpc3RvcnkKLSBbWF0gUmVxdWVzdCBQbGFpbiBFZGl0b3IKLSBbWF0gSGVhZGVyIElucHV0IChObyBWYWxpZGF0aW9uKQotIFtYXSBQYXRoIElucHV0IChObyBWYWxpZGF0aW9uKQotIFtYXSBDYWxsaW5nCi0gW1hdIFBsYWluIFJlc3BvbnNlIFZpZXdlciAoU3RhdHVzICsgSGVhZGVycyArIEJvZHkpCi0gW1hdIEhpc3Rvcnkgdmlld2VyCi0gW1hdIFJlZmFjdG9yIGZvb3RlciwgYWRkIGZsYXNoIGZvb3RlciBtZXNzYWdlcwotIFtYXSBJbXBvcnQgcmVxdWVzdCBib2R5IGZpbGUKLSBbWF0gU2F2ZSByZXNwb25zZSBib2R5IGFuZCBoZWFkZXIKLSBbWF0gQ29tbWFuZCBoaXN0b3J5IHdpdGgg4oaRL+KGkwotIFtYXSBTdXBwb3J0IGFycmF5IHF1ZXJ5IHN0cmluZ3MKLSBbWF0gU3VwcGVydCBleHRyYSBoZWFkZXJzCi0gW1hdIFN1cHBvcnQgbXVsdGlwbGUgc2VydmVycwoKIyBCYWNrbG9nCi0gWyBdIFNjaGVtYSBUeXBlcyAob3BlbmFwaS0zMSkKLSBbIF0gRGlzcGxheSBLZXkgTWFwcGluZ3MgaW4gUG9wdXAKLSBbIF0gUmVhZCBTcGVjIGZyb20gU1RESU4gCi0gWyBdIFJlcXVlc3QgcHJvZ3Jlc3MgYmFy readmeEtag: '"ea08df49cf9af797f61cc0faa08ce9b98da107ac"' readmeLastModified: Fri, 04 Oct 2024 15:27:53 GMT repositoryId: 766082951 description: Terminal UI to list, browse and run APIs defined with openapi spec. created: '2024-03-02T09:42:10Z' updated: '2026-02-05T20:19:38Z' language: Rust archived: false stars: 1053 watchers: 6 forks: 24 owner: zaghaghi logo: https://avatars.githubusercontent.com/u/111259?v=4 license: MIT repoEtag: '"67d690e2465f773793aa790cd07097a27b41d8ad3582bfa3a95209793b89936c"' repoLastModified: Thu, 05 Feb 2026 20:19:38 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/stackql/stackql v3: true id: 9cde100cf19bd12ff9483aa23bed6a2d repositoryMetadata: base64Readme: >- <!-- web assets -->
[logo]: https://stackql.io/img/stackql-logo-bold.png "stackql logo"
[homepage]: https://stackql.io/
[docs]: https://stackql.io/docs
[blog]: https://stackql.io/blog
[registry]: https://github.com/stackql/stackql-provider-registry
[variables]: https://stackql.io/docs/getting-started/variables
[macpkg]: https://storage.googleapis.com/stackql-public-releases/latest/stackql_darwin_multiarch.pkg
[winmsi]: https://releases.stackql.io/stackql/latest/stackql_windows_amd64.msi
[winzip]: https://releases.stackql.io/stackql/latest/stackql_windows_amd64.zip
[tuxzip]: https://releases.stackql.io/stackql/latest/stackql_linux_amd64.zip
<!-- docker links -->
[dockerhub]: https://hub.docker.com/u/stackql
[dockerstackql]: https://hub.docker.com/r/stackql/stackql
[dockerjupyter]: https://hub.docker.com/r/stackql/stackql-jupyter-demo
<!-- github actions links -->
[setupaction]: https://github.com/marketplace/actions/stackql-studios-setup-stackql
[execaction]: https://github.com/marketplace/actions/stackql-studios-stackql-exec
<!-- badges -->
[badge1]: https://img.shields.io/badge/platform-windows%20macos%20linux-brightgreen "Platforms"
[badge2]: https://github.com/stackql/stackql/workflows/Go/badge.svg "Go"
[badge3]: https://img.shields.io/github/license/stackql/stackql "License"
[badge4]: https://img.shields.io/tokei/lines/github/stackql/stackql "Lines"    
<!-- github links -->
[issues]: https://github.com/stackql/stackql/issues/new?assignees=&labels=bug&template=bug_report.md&title=%5BBUG%5D
[features]: https://github.com/stackql/stackql/issues/new?assignees=&labels=enhancement&template=feature_request.md&title=%5BFEATURE%5D
[developers]: /docs/developer_guide.md
[registrycont]: /docs/registry_contribution.md
[designdocs]: /docs/high-level-design.md
[contributing]: /CONTRIBUTING.md
[discussions]: https://github.com/orgs/stackql/discussions
<!-- repo assets -->
[darkmodeterm]: /docs/images/stackql-light-term.gif#gh-dark-mode-only
[lightmodeterm]: /docs/images/stackql-dark-term.gif#gh-light-mode-only
<!-- misc links -->
[twitter]: https://twitter.com/stackql

<!-- language: lang-none -->
<div align="center">

[![logo]][homepage]  
![badge1]
![badge2]
![badge3]
![badge4]

</div>

<div align="center">

![homebrew downloads](https://img.shields.io/homebrew/installs/dy/stackql?label=homebrew%20downloads)
![homebrew version](https://img.shields.io/homebrew/v/stackql?label=homebrew%20version)
![GitHub Downloads (all assets, all releases)](https://img.shields.io/github/downloads/stackql/stackql/total?label=github%20release%20downloads)
![GitHub Release](https://img.shields.io/github/v/release/stackql/stackql?label=github%20release)
![Docker Pulls](https://img.shields.io/docker/pulls/stackql/stackql)
![Docker Image Version](https://img.shields.io/docker/v/stackql/stackql?label=docker%20version)
![Chocolatey Downloads](https://img.shields.io/chocolatey/dt/stackql?label=chocolatey%20downloads)
![Chocolatey Version](https://img.shields.io/chocolatey/v/stackql)
![PyPI - Downloads](https://img.shields.io/pypi/dm/stackql-deploy?label=pypi%20downloads)

</div>
<div align="center">

### Deploy, manage and query cloud resources and interact with APIs using SQL
<!-- <h3 align="center">SQL based XOps, observability and middleware framework</h3> -->

<p align="center">

[__Read the docs »__][docs]  
[Raise an Issue][issues] · 
[Request a Feature][features] · 
[Developer Guide][developers] · 
[BYO Providers][registrycont]

</p>
</div>

<details open="open">
<summary>Contents</summary>
<ol>
<li><a href="#about-the-project">About The Project</a></li>
<li><a href="#installation">Installation</a></li>
<li><a href="#usage">Usage</a></li>
<!-- <li><a href="#roadmap">Roadmap</a></li> -->
<li><a href="#contributing">Contributing</a></li>
<li><a href="#license">License</a></li>
<li><a href="#contact">Contact</a></li>
<li><a href="#acknowledgements">Acknowledgements</a></li>
</ol>
</details>

## About The Project

[__StackQL__][homepage] is an open-source project built with Golang that allows you to create, modify and query the state of services and resources across different local and remote interfaces, using SQL semantics.  Such interfaces canonically include, but are not limited to, cloud and SaaS providers (Google, AWS, Azure, Okta, GitHub, etc.).
<br />
<br />

![stackql-shell][darkmodeterm]
![stackql-shell][lightmodeterm]

### How it works

StackQL is a standalone application that can be used in client mode (via __`exec`__ or __`shell`__) or accessed via a Postgres wire protocol client (`psycopg2`, etc.) using server mode (__`srv`__).  

StackQL parses SQL statements and transpiles them into API requests to the (cloud) resource provider.  The API calls are then executed and the results are returned to the user.  

StackQL provider interfaces are canonically defined in OpenAPI extensions to the providers' specification.  These definitions are then used to generate the SQL schema and the API client.  The source for the provider definitions are stored in the [__StackQL Registry__][registry].  The semantics of provider interactions are defined in [our `any-sdk` library](https://github.com/stackql/any-sdk).  For more detail on nuts and bolts, please see [the local `AGENTS.md` file](/AGENTS.md) and [that of `any-sdk`](https://github.com/stackql/any-sdk/blob/main/AGENTS.md).

<details>
<summary><b>StackQL Context Diagram</b></summary>
<br />
The following context diagram describes the StackQL architecture at a high level:  

<!-- ![StackQL Context Diagram](http://www.plantuml.com/plantuml/proxy?cache=no&src=https://raw.githubusercontent.com/stackql/test-readme/main/puml/stackql-c4-context.iuml) -->

```mermaid
flowchart LR
  subgraph StackQL
    direction BT
    subgraph ProviderDefs
        Registry[Provider Registry Docs]    
    end
    subgraph App
        Proc[$ stackql exec\n$ stackql shell\n$ stackql srv]
        style Proc fill:#000,stroke:#000,color:#fff,text-align:left;

        %% ,font-family:'Courier New', Courier, monospace
    end
  end
  User((User)) <--> StackQL <--> Provider[Cloud Provider API]
  ProviderDefs --> App
```

More detailed design documentation can be found in the [here][designdocs].

</details>

## Installation

StackQL is available for Windows, MacOS, Linux, Docker, GitHub Actions and more.  See the installation instructions below for your platform.  

<details>
<summary><b>Installing on MacOS</b></summary>

- Homebrew (`amd64` and `arm64`)
  - `brew install stackql` *or* `brew tap stackql/tap && brew install stackql/tap/stackql`
- MacOS PKG Installer (`amd64` and `arm64`)
  - download the latest [MacOS PKG installer for StackQL][macpkg]
  - run the installer and follow the prompts

</details>

<details>
<summary><b>Installing on Windows</b></summary>

- MSI Installer
  - download the latest [MSI installer for StackQL][winmsi]
  - run the installer and follow the prompts
- Chocolatey
  - install [Chocolatey](https://chocolatey.org/install)
  - run `choco install stackql`
- ZIP Archive
  - download the latest [Windows ZIP archive for StackQL][winzip]
  - extract the archive (code signed `stackql.exe` file) to a directory of your choice
  - add the directory to your `PATH` environment variable (optional)

</details>

<details>
<summary><b>Installing on Linux</b></summary>

- ZIP Archive
  - download the latest [Linux ZIP archive for StackQL][tuxzip]
    - or via `curl -L https://bit.ly/stackql-zip -O && unzip stackql-zip`
  - extract the archive (`stackql` file) to a directory of your choice
  - add the directory to your `PATH` environment variable (optional)

</details>

<details>
<summary><b>Getting StackQL from DockerHub</b></summary>

> View all available StackQL images on [__DockerHub__][dockerhub].  Images available include [__`stackql`__][dockerstackql], [__`stackql-jupyter-demo`__][dockerjupyter] and more.  Pull the latest StackQL base image using:  

```bash
docker pull stackql/stackql
```

</details>

<details>
<summary><b>Using StackQL with GitHub Actions</b></summary>

> Use StackQL in your GitHub Actions workflows to automate cloud infrastructure provisioning, IaC assurance, or compliance/security.  Available GitHub Actions include: [`setup-stackql`][setupaction], [`stackql-exec`][execaction] and more

</details>

## Usage

StackQL can be used via the interactive REPL shell, or via the `exec` command or ran as a server using the [Postgres wire protocol](https://www.postgresql.org/docs/current/protocol.html).  

> ℹ️ StackQL does not require or install a database.

* Interactive Shell
  ```sh
  # run interactive stackql queries
  stackql shell --auth="${AUTH}"
  ```
* Execute a statement or file
  ```sh
  stackql exec --auth="${AUTH}" -i myscript.iql --iqldata vars.jsonnet --output json
  
  # or
  
  stackql exec --auth="${AUTH}" "SELECT id, status FROM aws.ec2.instances WHERE region = 'us-east-1'"
  ```

  > ℹ️ output options of `json`, `csv`, `table` and `text` are available for the `exec` command using the `--output` flag

  > ℹ️ StackQL supports passing parameters using `jsonnet` or `json`, see [__Using Variables__][variables]
* Server
  ```sh
  # serve client requests over the Postgres wire protocol (psycopg2, etc.) 
  stackql srv --auth="${AUTH}"
  ```

_For more examples, please check our [Blog][blog]_

<!-- ## Roadmap

See our [__roadmap__](https://github.com/othneildrew/Best-README-Template/issues) to see where we are going with the project. -->

## Contributing

Contributions are welcome and encouraged.  For more information on how to contribute, please see our [__contributing guide__][contributing].

## License

Distributed under the MIT License. See [`LICENSE`](https://github.com/stackql/stackql/blob/main/LICENSE) for more information.  Licenses for third party software we are using are included in the [/docs/licenses](/docs/licenses) directory.

## Contact

Get in touch with us via Twitter at [__@stackql__][twitter], email us at [__info@stackql.io__](info@stackql.io) or start a conversation using [__discussions__][discussions].

## Acknowledgements
Forks of the following support our work:

* [vitess](https://vitess.io/)
* [kin-openapi](https://github.com/getkin/kin-openapi)
* [gorilla/mux](https://github.com/gorilla/mux)
* [readline](https://github.com/chzyer/readline)
* [psql-wire](https://github.com/jeroenrinzema/psql-wire)
* [mcp-postgres](https://github.com/gldc/mcp-postgres)
* [the `golang` MCP SDK](https://github.com/modelcontextprotocol/go-sdk)
* ...and more.  Please excuse us for any omissions.

We gratefully acknowledge these pieces of work.

 readmeEtag: '"88200627d768f11d359577c4d94e896567de5897"' readmeLastModified: Sun, 12 Oct 2025 22:48:14 GMT repositoryId: 443987542 description: >- Query, provision and operate Cloud and SaaS resources and APIs using an extensible SQL based framework created: '2022-01-03T08:20:06Z' updated: '2026-02-02T05:22:22Z' language: Go archived: false stars: 835 watchers: 11 forks: 78 owner: stackql logo: https://avatars.githubusercontent.com/u/95105302?v=4 license: MIT repoEtag: '"70122ac450f39825e5c69369db387ec87cdb4376aab94efcd758a9fd2dc0470f"' repoLastModified: Mon, 02 Feb 2026 05:22:22 GMT category: Server Implementations foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/apisyouwonthate/openapi.tools v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJLlRvb2xzCgpbIVtOZXRsaWZ5IFN0YXR1c10oaHR0cHM6Ly9hcGkubmV0bGlmeS5jb20vYXBpL3YxL2JhZGdlcy8xZWNhNjRiNC1mMjhjLTQyNTAtYjg4NC1jMzA2MjdhYjY1ZDgvZGVwbG95LXN0YXR1cyldKGh0dHBzOi8vYXBwLm5ldGxpZnkuY29tL3NpdGVzL29wZW5hcGktdG9vbHMvZGVwbG95cykKWyFbQnV5IHVzIGEgdHJlZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9CdXklMjB1cyUyMGElMjB0cmVlLSVGMCU5RiU4QyVCMy1saWdodGdyZWVuKV0oaHR0cHM6Ly9zaG9wLnByb3RlY3QuZWFydGgvcHJvZHVjdHMvdHJlZSkKCkEgY29tcHJlaGVuc2l2ZSBkaXJlY3Rvcnkgb2YgKiozMDArIHRvb2xzKiogZm9yIHdvcmtpbmcgd2l0aCBPcGVuQVBJIGRlc2NyaXB0aW9ucyDigJQgZWRpdG9ycywgbGludGVycywgcGFyc2VycywgY29kZSBnZW5lcmF0b3JzLCBkb2N1bWVudGF0aW9uLCB0ZXN0aW5nLCBtb2NraW5nLCBhbmQgbW9yZS4KCioqW1Zpc2l0IG9wZW5hcGkudG9vbHMg4oaSXShodHRwczovL29wZW5hcGkudG9vbHMpKioKCiMjIEZlYXR1cmVzCgotICoqQ29tcHJlaGVuc2l2ZSBsaXN0aW5ncyoqIOKAlCBUb29scyBvcmdhbml6ZWQgYnkgY2F0ZWdvcnkgd2l0aCBmaWx0ZXJpbmcgYnkgbGFuZ3VhZ2UsIE9wZW5BUEkgdmVyc2lvbiBzdXBwb3J0LCBhbmQgbW9yZQotICoqQ3VyYXRlZCBjb2xsZWN0aW9ucyoqIOKAlCBCcm93c2Ugb3BlbiBzb3VyY2UgdG9vbHMsIFNhYVMgcHJvZHVjdHMsIG9yIGxlZ2FjeSAocHJlLTMuMSkgdG9vbHMKLSAqKlNwb25zb3IgaGlnaGxpZ2h0cyoqIOKAlCBTdXBwb3J0aW5nIGNvbXBhbmllcyBnZXQgcHJpb3JpdHkgcGxhY2VtZW50IGFuZCBiYWRnZXMKLSAqKkFsd2F5cyB1cCB0byBkYXRlKiog4oCUIENvbW11bml0eS1tYWludGFpbmVkIHdpdGggcmVndWxhciBjb250cmlidXRpb25zCgojIyBDb250cmlidXRpbmcKCldhbnQgdG8gYWRkIGEgdG9vbCBvciBpbXByb3ZlIHRoZSBzaXRlPyBQbGVhc2UgcmVhZCBbQ09OVFJJQlVUSU5HLm1kXSguL0NPTlRSSUJVVElORy5tZCkuCgojIyBTcG9uc29yc2hpcAoKT3BlbkFQSS50b29scyBpcyBtYWludGFpbmVkIGJ5IFtBUElzIFlvdSBXb24ndCBIYXRlXShodHRwczovL2FwaXN5b3V3b250aGF0ZS5jb20pLiBTcG9uc29yc2hpcCBoZWxwcyBjb3ZlciBjb3N0cyBhbmQgZW5hYmxlIGNvbnRpbnVlZCBkZXZlbG9wbWVudC4KCioqQmVuZWZpdHMgZm9yIHNwb25zb3JzOioqCgotIFByaW9yaXR5IHBsYWNlbWVudCBpbiBjYXRlZ29yeSBsaXN0aW5ncwotICJTcG9uc29yZWQiIGJhZGdlIG9uIHlvdXIgdG9vbAotIEVtYmVkZGFibGUgYmFkZ2VzIGZvciB5b3VyIHdlYnNpdGUgKFtzZWUgb3B0aW9uc10oaHR0cHM6Ly9vcGVuYXBpLnRvb2xzL2JhZGdlcykpCgpbQmVjb21lIGEgc3BvbnNvciDihpJdKGh0dHBzOi8vb3BlbmFwaS50b29scy9zcG9uc29yKQoKIyMgVGVjaCBTdGFjawoKLSBbQXN0cm9dKGh0dHBzOi8vYXN0cm8uYnVpbGQpIOKAlCBTdGF0aWMgc2l0ZSBnZW5lcmF0b3IKLSBbUmVhY3RdKGh0dHBzOi8vcmVhY3QuZGV2KSDigJQgSW50ZXJhY3RpdmUgY29tcG9uZW50cwotIFtUYWlsd2luZCBDU1NdKGh0dHBzOi8vdGFpbHdpbmRjc3MuY29tKSDigJQgU3R5bGluZwotIFtOZXRsaWZ5XShodHRwczovL25ldGxpZnkuY29tKSDigJQgSG9zdGluZwoKIyMgTGljZW5zZQoKTUlUCg== readmeEtag: '"82d57db53266f3274deb72d2e40484ee91d2419f"' readmeLastModified: Thu, 08 Jan 2026 23:21:51 GMT repositoryId: 114417462 description: >- A collection of Editors, Linters, Parsers, Code Generators, Documentation, Testing created: '2017-12-15T22:51:00Z' updated: '2026-02-05T19:46:56Z' language: TypeScript archived: false stars: 807 watchers: 30 forks: 373 owner: apisyouwonthate logo: https://avatars.githubusercontent.com/u/35378327?v=4 license: MIT repoEtag: '"0645988c353fb1720e674be3d445339f0492dfdf0c4ecb08a58a205ce693304b"' repoLastModified: Thu, 05 Feb 2026 19:46:56 GMT foundInMaster: true category: Description Validators id: bf8b6a3a6298d701eabc857d42d5c371 v3_1: true - source: openapi3 tags repository: https://github.com/mermade/oas-kit v3: true repositoryMetadata: base64Readme: >- IyBPQVMtS2l0CgpUaGlzIGlzIHRoZSBtb25vLXJlcG8gZm9yIHRoZSBmb2xsb3dpbmcgcmVsYXRlZCBwcm9qZWN0cwoKKiBbc3dhZ2dlcjJvcGVuYXBpXShwYWNrYWdlcy9zd2FnZ2VyMm9wZW5hcGkvUkVBRE1FLm1kKQoqIFtvYXMtdmFsaWRhdG9yXShwYWNrYWdlcy9vYXMtdmFsaWRhdG9yL1JFQURNRS5tZCkKKiBbb2FzLWxpbnRlcl0ocGFja2FnZXMvb2FzLWxpbnRlci9SRUFETUUubWQpCiogW29hcy1yZXNvbHZlcl0ocGFja2FnZXMvb2FzLXJlc29sdmVyL1JFQURNRS5tZCkKKiBbb2FzLXNjaGVtYS13YWxrZXJdKHBhY2thZ2VzL29hcy1zY2hlbWEtd2Fsa2VyL1JFQURNRS5tZCkKKiBbb2FzLWtpdC1jb21tb25dKHBhY2thZ2VzL29hcy1raXQtY29tbW9uL1JFQURNRS5tZCkKKiBbcmVmdG9vbHNdKHBhY2thZ2VzL3JlZnRvb2xzL1JFQURNRS5tZCkKCiMjIERvY3VtZW50YXRpb24KCiogW01haW4gc2l0ZV0oaHR0cHM6Ly9tZXJtYWRlLmdpdGh1Yi5pby9vYXMta2l0KQoqIFtDSEFOR0VMT0ddKGh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL29hcy1raXQvYmxvYi9tYXN0ZXIvQ0hBTkdFTE9HLm1kI2NoYW5nZS1sb2cpCgojIyBPbmxpbmUgY29udmVydGVyL3ZhbGlkYXRvcgoKKiBbT3BlbkFQSS13ZWJjb252ZXJ0ZXJdKGh0dHBzOi8vbWVybWFkZS5vcmcudWsvb3BlbmFwaS1jb252ZXJ0ZXIpCgojIyBTdXBwb3J0ZWQgTm9kZS5qcyB2ZXJzaW9ucwoKQW55IExUUyB2ZXJzaW9uLiBJdCBpcyAqKm5vdCoqIHJlY29tbWVuZGVkIHRvIHVzZSBOb2RlLmpzIDEyLjE3LngsMTIuMTgueCBvciAxMi4xOS54IGR1ZSB0byBhbiBbaHR0cDIgYnVnXShodHRwczovL2dpdGh1Yi5jb20vbm9kZWpzL25vZGUvaXNzdWVzLzI4MDAxKS4KCiMjIERldmVsb3BtZW50CgoqIGNsb25lIHRoZSByZXBvc2l0b3J5CiogYG5wbSBpYCBpbiB0aGUgdG9wIGxldmVsIGRpcmVjdG9yeQoqIGBucHggbGVybmEgYm9vdHN0cmFwYAoKUGxlYXNlIHRyeSBhbmQga2VlcCBjb21taXRzIHJlbGF0ZWQgdG8gYSBzaW5nbGUgcGFja2FnZSBvciBwaWVjZSBvZiBmdW5jdGlvbmFsaXR5LiBQbGVhc2UgcmV2aWV3IHRoZQpbQ09OVFJJQlVUSU5HLm1kXShDT05UUklCVVRJTkcubWQpIGZvciBhZGRpdGlvbmFsIGRldGFpbHMuCgojIyBTdXBwb3J0aW5nIGRldmVsb3BtZW50CgoqIFtBUElzLmd1cnUgb3Blbi1jb2xsZWN0aXZlXShodHRwczovL29wZW5jb2xsZWN0aXZlLmNvbS9vcGVuYXBpLWRpcmVjdG9yeSkKKiBbTGlub2RlIFZQUyByZWZlcnJhbCBsaW5rXShodHRwczovL3d3dy5saW5vZGUuY29tLz9yPTU3MzRiZTQ2N2NjNTAxYjIzMjY3Y2Y2NmQ0NTFiYzMzOTA0MmRkZmEpCg== readmeEtag: '"cd704faeca79a10585a963bfa21eb1994069dc88"' readmeLastModified: Wed, 07 Jul 2021 12:13:38 GMT repositoryId: 71464935 description: Convert Swagger 2.0 definitions to OpenAPI 3.0 and resolve/validate/lint created: '2016-10-20T13:17:42Z' updated: '2026-01-21T09:06:09Z' language: JavaScript archived: false stars: 739 watchers: 13 forks: 119 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"e322fec36c266205419b5f5fec838b52b48ebce78f71b7e04f8c1b02862f0234"' repoLastModified: Wed, 21 Jan 2026 09:06:09 GMT foundInMaster: true category: - Converters - Parsers id: 6f45c86bb645fe0967b0b35672cb2a57 oldLocations: - https://github.com/mermade/swagger2openapi - source: openapi3 tags name: API Sprout homepage: https://github.com/danielgtaylor/apisprout language: - cli - Docker source_description: >- Lightweight, blazing fast, cross-platform OpenAPI 3 mock server with validation category: - Mock - Server Implementations link: https://github.com/danielgtaylor/apisprout repository: https://github.com/danielgtaylor/apisprout v3: true repositoryMetadata: base64Readme: >- PGltZyBzcmM9Imh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzEwNjgyNi80MzExOTQ5NC03OGJlOTIyNC04ZWNiLTExZTgtOWQxYS05ZmM2ZjMwMTRiOTEucG5nIiB3aWR0aD0iMzAwIiBhbHQ9IkFQSSBTcHJvdXQiLz4KClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS9kYW5pZWxndGF5bG9yL2FwaXNwcm91dCldKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9kYW5pZWxndGF5bG9yL2FwaXNwcm91dCkgWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQpIFshW0dpdEh1YiB0YWcgKGxhdGVzdCBTZW1WZXIpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi90YWcvZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2RhbmllbGd0YXlsb3IvYXBpc3Byb3V0L3JlbGVhc2VzKSBbIVtEb2NrZXIgUHVsbHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZG9ja2VyL3B1bGxzL2RhbmllbGd0YXlsb3IvYXBpc3Byb3V0LnN2ZyldKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9kYW5pZWxndGF5bG9yL2FwaXNwcm91dC8pCgpBIHNpbXBsZSwgcXVpY2ssIGNyb3NzLXBsYXRmb3JtIEFQSSBtb2NrIHNlcnZlciB0aGF0IHJldHVybnMgZXhhbXBsZXMgc3BlY2lmaWVkIGluIGFuIEFQSSBkZXNjcmlwdGlvbiBkb2N1bWVudC4gRmVhdHVyZXMgaW5jbHVkZToKCi0gT3BlbkFQSSAzLnggc3VwcG9ydAogIC0gVXNlcyBvcGVyYXRpb24gYGV4YW1wbGVzYCBvciBnZW5lcmF0ZXMgZXhhbXBsZXMgZnJvbSBgc2NoZW1hYAotIExvYWQgZnJvbSBhIFVSTCBvciBsb2NhbCBmaWxlIChhdXRvIHJlbG9hZCB3aXRoIGAtLXdhdGNoYCkKLSBDT1JTIGhlYWRlcnMgZW5hYmxlZCBieSBkZWZhdWx0Ci0gQWNjZXB0IGhlYWRlciBjb250ZW50IG5lZ290aWF0aW9uCiAgLSBFeGFtcGxlOiBgQWNjZXB0OiBhcHBsaWNhdGlvbi8qYAotIFByZWZlciBoZWFkZXIgdG8gc2VsZWN0IHJlc3BvbnNlIHRvIHRlc3Qgc3BlY2lmaWMgY2FzZXMKICAtIEV4YW1wbGU6IGBQcmVmZXI6IHN0YXR1cz00MDlgCi0gU2VydmVyIHZhbGlkYXRpb24gKGVuYWJsZWQgd2l0aCBgLS12YWxpZGF0ZS1zZXJ2ZXJgKQogIC0gVmFsaWRhdGVzIHNjaGVtZSwgaG9zdG5hbWUvcG9ydCwgYW5kIGJhc2UgcGF0aAogIC0gU3VwcG9ydHMgYGxvY2FsaG9zdGAgb3V0IG9mIHRoZSBib3gKICAtIFVzZSB0aGUgYC0tYWRkLXNlcnZlcmAgZmxhZywgaW4gY29uanVuY3Rpb24gd2l0aCBgLS12YWxpZGF0ZS1zZXJ2ZXJgLCB0byBkeW5hbWljYWxseSBpbmNsdWRlIG1vcmUgc2VydmVycyBpbiB0aGUgdmFsaWRhdGlvbiBsb2dpYwotIFJlcXVlc3QgcGFyYW1ldGVyICYgYm9keSB2YWxpZGF0aW9uIChlbmFibGVkIHdpdGggYC0tdmFsaWRhdGUtcmVxdWVzdGApCi0gQ29uZmlndXJhdGlvbiB2aWE6CiAgLSBGaWxlcyAoYC9ldGMvYXBpc3Byb3V0L2NvbmZpZy5qc29ufHlhbWxgKQogIC0gRW52aXJvbm1lbnQgKHByZWZpeGVkIHdpdGggYFNQUk9VVF9gLCBlLmcuIGBTUFJPVVRfVkFMSURBVEVfU0VSVkVSYCkKICAtIENvbW1hbmRsaW5lIGZsYWdzCgpVc2FnZSBpcyBzaW1wbGU6CgpgYGBzaAojIExvYWQgZnJvbSBhIGxvY2FsIGZpbGUKYXBpc3Byb3V0IG15LWFwaS55YW1sCgojIFZhbGlkYXRlIHNlcnZlciBuYW1lIGFuZCB1c2UgYmFzZSBwYXRoCmFwaXNwcm91dCAtLXZhbGlkYXRlLXNlcnZlciBteS1hcGkueWFtbAoKIyBEeW5hbWljYWxseSBJbmNsdWRlIGEgbmV3IHNlcnZlciAvIHBhdGggaW4gdGhlIHZhbGlkYXRpb24KYXBpc3Byb3V0IC0tYWRkLXNlcnZlciBodHRwOi8vbG9jYWxob3N0OjgwODAvbW9jayAtLXZhbGlkYXRlLXNlcnZlciBteS1hcGkueWFtbAoKIyBMb2FkIGZyb20gYSBVUkwKYXBpc3Byb3V0IGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL21hc3Rlci9leGFtcGxlcy92My4wL2FwaS13aXRoLWV4YW1wbGVzLnlhbWwKYGBgCgojIyBEb2NrZXIgSW1hZ2UKCkEgaG9zdGVkIFtBUEkgU3Byb3V0IERvY2tlciBpbWFnZV0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL2RhbmllbGd0YXlsb3IvYXBpc3Byb3V0LykgaXMgcHJvdmlkZWQgdGhhdCBtYWtlcyBpdCBlYXN5IHRvIGRlcGxveSBtb2NrIHNlcnZlcnMgb3IgcnVuIGxvY2FsbHkuIEZvciBleGFtcGxlOgoKYGBgc2gKZG9ja2VyIHB1bGwgZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQKZG9ja2VyIHJ1biAtcCA4MDAwOjgwMDAgZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQgaHR0cDovL2V4YW1wbGUuY29tL215LWFwaS55YW1sCmBgYAoKQ29uZmlndXJhdGlvbiBjYW4gYmUgcGFzc2VkIHZpYSBlbnZpcm9ubWVudCB2YXJpYWJsZXMsIGUuZy4gc2V0dGluZyBgU1BST1VUX1ZBTElEQVRFX1JFUVVFU1Q9MWAsIG9yIGJ5IHBhc3NpbmcgY29tbWFuZGxpbmUgZmxhZ3MuIEl0IGlzIGFsc28gcG9zc2libGUgdG8gdXNlIGEgbG9jYWwgQVBJIGRlc2NyaXB0aW9uIGZpbGUgdmlhIFtEb2NrZXIgVm9sdW1lc10oaHR0cHM6Ly9kb2NzLmRvY2tlci5jb20vc3RvcmFnZS92b2x1bWVzLyk6CgpgYGAKIyBSZW1lbWJlciB0byBwdXQgdGhlIGZ1bGwgcGF0aCB0byBsb2NhbCBhcmNoaXZlIFlBTUwgaW4gLXYKZG9ja2VyIHJ1biAtcCA4MDAwOjgwMDAgLXYgJEZVTExQQVRIL2xvY2FsZmlsZS55YW1sOi9hcGkueWFtbCBkYW5pZWxndGF5bG9yL2FwaXNwcm91dCAvYXBpLnlhbWwKYGBgCgojIyBJbnN0YWxsYXRpb24KCkRvd25sb2FkIHRoZSBhcHByb3ByaWF0ZSBiaW5hcnkgZnJvbSB0aGUgW3JlbGVhc2VzXShodHRwczovL2dpdGh1Yi5jb20vZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQvcmVsZWFzZXMpIHBhZ2UuCgpBbHRlcm5hdGl2ZWx5LCB5b3UgY2FuIHVzZSBgZ28gZ2V0YDoKCmBgYHNoCmdvIGdldCBnaXRodWIuY29tL2RhbmllbGd0YXlsb3IvYXBpc3Byb3V0CmBgYAoKIyMgRXh0cmEgRmVhdHVyZXMKCiMjIyBSZW1vdGUgUmVsb2FkCgpJZiB5b3VyIEFQSSBzcGVjIGlzIGxvYWRlZCBmcm9tIGEgcmVtb3RlIFVSTCwgeW91IGNhbiBsaXZlLXJlbG9hZCBpdCBieSBoaXR0aW5nIHRoZSBgL19fcmVsb2FkYCBlbmRwb2ludC4KCiMjIyBIZWFsdGggQ2hlY2sKCkEgc2ltcGxlIGVuZHBvaW50IHdoaWNoIHJldHVybnMgc3RhdHVzIGNvZGUgYDIwMGAgaXMgYXZhaWxhYmxlIGF0IGAvX19oZWFsdGhgLiBUaGlzIGVuZHBvaW50IHN1Y2Nlc3NmdWxseSByZXR1cm5zIGAyMDBgIGV2ZW4gaWYgYC0tdmFsaWRhdGUtc2VydmVyYCBpcyB0dXJuZWQgb24sIGFuZCB0aGUgZW5kcG9pbnQgaXMgYmVpbmcgYWNjZXNzZWQgZnJvbSBhIG5vbi12YWxpZGF0ZWQgaG9zdC4KCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyBhcmUgdmVyeSB3ZWxjb21lLiBQbGVhc2Ugb3BlbiBhIHRyYWNraW5nIGlzc3VlIG9yIHB1bGwgcmVxdWVzdCBhbmQgd2UgY2FuIHdvcmsgdG8gZ2V0IHRoaW5ncyBtZXJnZWQgaW4uCgojIyBSZWxlYXNlIFByb2Nlc3MKClRoZSBmb2xsb3dpbmcgZGVzY3JpYmVzIHRoZSBzdGVwcyB0byBtYWtlIGEgbmV3IHJlbGVhc2Ugb2YgQVBJIFNwcm91dC4KCjEuIE1lcmdlIG9wZW4gUFJzIHlvdSB3YW50IHRvIHJlbGVhc2UuCjEuIFNlbGVjdCBhIG5ldyBzZW12ZXIgdmVyc2lvbiBudW1iZXIgKG1ham9yL21pbm9yL3BhdGNoIGRlcGVuZGluZyBvbiBjaGFuZ2VzKS4KMS4gVXBkYXRlIGBDSEFOR0VMT0cubWRgIHRvIGRlc2NyaWJlIGNoYW5nZXMuCjEuIENyZWF0ZSBhIGNvbW1pdCBmb3IgdGhlIHJlbGVhc2UuCjEuIFRhZyB0aGUgY29tbWl0IHdpdGggYGdpdCB0YWcgLWEgLW0gJ1RhZ2dpbmcgeC55LnogcmVsZWFzZScgdngueS56YC4KMS4gQnVpbGQgcmVsZWFzZSBiaW5hcmllcyB3aXRoIGAuL3JlbGVhc2Uuc2hgLgoxLiBQdXNoIHRoZSBjb21taXQgYW5kIHRhZ3MuCjEuIFVwbG9hZCB0aGUgcmVsZWFzZSBiaW5hcmllcy4KCiMjIExpY2Vuc2UKCkNvcHlyaWdodCAmY29weTsgMjAxOC0yMDE5IERhbmllbCBHLiBUYXlsb3IKCmh0dHA6Ly9kZ3QubWl0LWxpY2Vuc2Uub3JnLwo= readmeEtag: '"66a669d19eba31ae592125a14d05a92dc68fbd57"' readmeLastModified: Thu, 31 Oct 2019 06:11:48 GMT repositoryId: 142108855 description: >- Lightweight, blazing fast, cross-platform OpenAPI 3 mock server with validation created: '2018-07-24T05:29:01Z' updated: '2025-12-30T09:04:54Z' language: Go archived: false stars: 715 watchers: 11 forks: 73 owner: danielgtaylor logo: https://avatars.githubusercontent.com/u/106826?v=4 license: MIT repoEtag: '"fccefa3c5280336ea4de3cc0be9527660439e5edc01105dc28d5d8f1c857eb9e"' repoLastModified: Tue, 30 Dec 2025 09:04:54 GMT foundInMaster: true id: 1ceeb35cbfe549bf7ed38d6e576301db - source: - openapi3 tags - openapi31 tags repository: https://github.com/redocly/create-openapi-repo v3: true repositoryMetadata: base64Readme: >- IyBjcmVhdGUtb3BlbmFwaS1yZXBvCgpBIHRvb2wgZm9yIGdlbmVyYXRpbmcgbXVsdGktZmlsZSBPcGVuQVBJIGRlZmluaXRpb25zLgoKWyFbTlBNIHZlcnNpb25dW25wbS1pbWFnZV1dW25wbS11cmxdIFshW0RlcGVuZGVuY3kgU3RhdHVzXVtkYXZpZGRtLWltYWdlXV1bZGF2aWRkbS11cmxdCgo8cCBhbGlnbj0iY2VudGVyIj4KPGltZyBzcmM9Ii4vbG9nby5wbmciIHdpZHRoPSI1MDBweCIvPgo8L3A+CgpOZWVkIHRvIHdyaXRlIG9yIGNvbnRyaWJ1dGUgdG8gYW4gT3BlbkFQSSBkZWZpbml0aW9uPyBgY3JlYXRlLW9wZW5hcGktcmVwb2AgY2FuIGhlbHAgd2l0aCB0aGF0IQoKIyMgRmVhdHVyZXMKCiAtIE9wZW5BUEkgMy4wIHN1cHBvcnQKIC0gU3BsaXQgYW4gZXhpc3RpbmcgT3BlbkFQSSBkZWZpbml0aW9uIGludG8gbXVsdGlwbGUgZmlsZXMKIC0gQnVuZGxlIGEgbXVsdGktZmlsZSBkZWZpbml0aW9uIGludG8gYSBzaW5nbGUgZmlsZQogLSBWYWxpZGF0ZSB5b3VyIE9wZW5BUEkgZGVmaW5pdGlvbiB1c2luZyBvdXIgZnJlZSBbb3BlbmFwaS1jbGkgdG9vbF0oaHR0cHM6Ly9naXRodWIuY29tL3JlZG9jbHkvb3BlbmFwaS1jbGkpCiAtIEF1dG9tYXRlIGRlcGxveW1lbnQgb2YgeW91ciBBUEkgcmVmZXJlbmNlIGRvY3MgdXNpbmcgQ0kvQ0Qgd29ya2Zsb3dzCiAtIE1haW50YWluIGNvZGUgc2FtcGxlcyBhcyBzZXBhcmF0ZSBmaWxlcwogLSBMaXZlIGVkaXRpbmcgaW4geW91ciBlZGl0b3Igb2YgY2hvaWNlIDpoZWFydF9leWVzOgoKIyMgRXhhbXBsZXMKLSBbUmViaWxseV0oaHR0cHM6Ly9naXRodWIuY29tL1JlYmlsbHkvUmViaWxseUFQSSkKLSBbVGhpbmdmdWxdKGh0dHBzOi8vZ2l0aHViLmNvbS90aGluZ2Z1bC9vcGVuYXBpLXNwZWMpCgojIyBQcmVyZXF1aXNpdGVzCgpCZWZvcmUgeW91IGJlZ2luLCBtYWtlIHN1cmUgeW91IGhhdmUgdGhlIGZvbGxvd2luZyBwcmVyZXF1aXNpdGVzOgoKIC0gW05vZGUuanNdKGh0dHBzOi8vbm9kZWpzLm9yZy8pCiAtIFtHaXRodWIgcmVwb3NpdG9yeV0oaHR0cHM6Ly9oZWxwLmdpdGh1Yi5jb20vYXJ0aWNsZXMvY3JlYXRlLWEtcmVwby8jY3JlYXRlLWEtbmV3LXJlcG9zaXRvcnktb24tZ2l0aHViKSAobmV3IE9wZW5BUEkgZGVmaW5pdGlvbnMgb25seSkKCiMjIEluc3RhbGxhdGlvbgoKMS4gTmF2aWdhdGUgdG8gdGhlIGxvY2F0aW9uIHdoZXJlIHlvdSB3YW50IHRvIGNyZWF0ZSB0aGUgcmVwb3NpdG9yeS4KMi4gUnVuIG9uZSBvZiB0aGUgZm9sbG93aW5nIGNvbW1hbmRzOgogICAtIEluc3RhbGwgYGNyZWF0ZS1vcGVuYXBpLXJlcG9gIGdsb2JhbGx5OgoKICAgICAgICBgYGBiYXNoCiAgICAgICAgbnBtIGluc3RhbGwgLWcgY3JlYXRlLW9wZW5hcGktcmVwbwogICAgICAgIGBgYAoKICAgLSBJbnN0YWxsIGBjcmVhdGUtb3BlbmFwaS1yZXBvYCB1c2luZyBbYG5weGBdKGh0dHBzOi8vbWVkaXVtLmNvbS9AbWF5YmVrYXR6L2ludHJvZHVjaW5nLW5weC1hbi1ucG0tcGFja2FnZS1ydW5uZXItNTVmN2Q0YmQyODJiKToKCiAgICAgICAgYGBgYmFzaAogICAgICAgIG5weCBjcmVhdGUtb3BlbmFwaS1yZXBvCiAgICAgICAgYGBgCgozLiBGb2xsb3cgdGhlIFtpbnRlcmFjdGl2ZSBwcm9tcHRzXShodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9jcmVhdGUtb3BlbmFwaS1yZXBvI3VzYWdlKSB0byBjb21wbGV0ZSB0aGUgaW5zdGFsbGF0aW9uLgoKIyMjIFVwZ3JhZGluZyBmcm9tIGEgcHJpb3IgdmVyc2lvbgoKVG8gdXBncmFkZSBmcm9tIGEgcHJpb3IgdmVyc2lvbiBvZiBgY3JlYXRlLW9wZW5hcGktcmVwb2AsIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQgaW4gdGhlIHJvb3QgZm9sZGVyIG9mIHlvdXIgcmVwb3NpdG9yeToKCmBgYGJhc2gKbnB4IGNyZWF0ZS1vcGVuYXBpLXJlcG8gLS1taWdyYXRlLTItMwpgYGAKCioqTm90ZToqKiBQbHVnaW5zIGFyZW4ndCBpbmNsdWRlZCBpbiB0aGUgbWlncmF0aW9uLiBZb3UnbGwgbmVlZCB0byBtYW51YWxseSBhZGQgdGhlbSB0byB0aGUgYHRyYW5zZm9ybWVyc2AgZm9sZGVyLgoKIyMgVXNhZ2UKCmBjcmVhdGUtb3BlbmFwaS1yZXBvYCBwcm92aWRlcyBpbnRlcmFjdGl2ZSBwcm9tcHRzIHRvIGhlbHAgZ3VpZGUgeW91IHRocm91Z2ggdGhlIGluc3RhbGxhdGlvbiBwcm9jZXNzLiBUd28gYmFzaWMgd29ya2Zsb3dzIGFyZSBzdXBwb3J0ZWQ6CgojIyMgU3BsaXQgYW4gZXhpc3RpbmcgT3BlbkFQSSBkZWZpbml0aW9uCgpUaGUgaW50ZXJhY3RpdmUgcHJvbXB0cyBhbGxvdyB5b3UgdG8gc3BlY2lmeSB0aGUgcGF0aCB0byB0aGUgZmlsZSBvbiB5b3VyIGxvY2FsIG1hY2hpbmUsIGFzIHdlbGwgYXMgcmVuYW1lIHRoZSBBUEkgKG9wdGlvbmFsKS4gQWZ0ZXIgeW91IGNob29zZSB0byBwcm9jZWVkLCBgY3JlYXRlLW9wZW5hcGktcmVwb2AgaW5pdGlhbGl6ZXMgdGhlIHJlcG9zaXRvcnkgYW5kIHNwbGl0cyB5b3VyIE9wZW5BUEkgZGVmaW5pdGlvbiBpbnRvIG11bHRpcGxlIGZpbGVzLgoKIyMjIENyZWF0ZSBhIG5ldyBPcGVuQVBJIGRlZmluaXRpb24KClRoZSBpbnRlcmFjdGl2ZSBwcm9tcHRzIGFsbG93IHlvdSB0byBzcGVjaWZ5IGEgbmFtZSBmb3IgdGhlIEFQSSBhbmQgY2hvb3NlIHdoZXRoZXIgdG8gY3JlYXRlIGEgc3ViLWZvbGRlciBmb3IgY29kZSBzYW1wbGUgZmlsZXMuIEFmdGVyIHlvdSBjaG9vc2UgdG8gcHJvY2VlZCwgYGNyZWF0ZS1vcGVuYXBpLXJlcG9gIGluaXRpYWxpemVzIHRoZSByZXBvc2l0b3J5IGFuZCBwb3B1bGF0ZXMgaXQgd2l0aCBwbGFjZWhvbGRlciBmaWxlcyBhbmQgZm9sZGVycy4gCgojIyBEaXJlY3Rvcnkgc3RydWN0dXJlCgpUaGUgZGlyZWN0b3J5IHN0cnVjdHVyZSB3aWxsIGxvb2sgc2ltaWxhciB0byB0aGlzOgoKKipOb3RlOioqIFlvdSBjYW4gbW9kaWZ5IHRoZSBkaXJlY3Rvcnkgc3RydWN0dXJlIHRvIG1lZXQgeW91ciBzcGVjaWZpYyByZXF1aXJlbWVudHMuCgpgYGAKICAgIOKUnOKUgOKUgCAucmVkb2NseS55YW1sCiAgICDilJzilIDilIAgTElDRU5TRQogICAg4pSc4pSA4pSAIFJFQURNRS5tZAogICAg4pSc4pSA4pSAIGRvY3MKICAgIOKUgiAgIOKUnOKUgOKUgCBmYXZpY29uLnBuZwogICAg4pSCICAg4pSU4pSA4pSAIGluZGV4Lmh0bWwKICAgIOKUnOKUgOKUgCBvcGVuYXBpCiAgICDilIIgICDilJzilIDilIAgUkVBRE1FLm1kCiAgICDilIIgICDilJzilIDilIAgY29kZV9zYW1wbGVzCiAgICDilIIgICDilIIgICDilJzilIDilIAgQyMKICAgIOKUgiAgIOKUgiAgIOKUgiAgIOKUlOKUgOKUgCBlY2hvCiAgICDilIIgICDilIIgICDilIIgICAgICAg4pSU4pSA4pSAIHBvc3QuY3MKICAgIOKUgiAgIOKUgiAgIOKUnOKUgOKUgCBQSFAKICAgIOKUgiAgIOKUgiAgIOKUgiAgIOKUlOKUgOKUgCBlY2hvCiAgICDilIIgICDilIIgICDilIIgICAgICAg4pSU4pSA4pSAIHBvc3QucGhwCiAgICDilIIgICDilIIgICDilJTilIDilIAgUkVBRE1FLm1kCiAgICDilIIgICDilJzilIDilIAgY29tcG9uZW50cwogICAg4pSCICAg4pSCICAg4pSU4pSA4pSAIFJFQURNRS5tZAogICAg4pSCICAg4pSU4pSA4pSAIHBhdGhzCiAgICDilIIgICAgICAg4pSU4pSA4pSAIFJFQURNRS5tZAogICAg4pSU4pSA4pSAIHBhY2thZ2UuanNvbgpgYGAKCiAtIGAucmVkb2NseS55YW1sYDogQ29uZmlndXJhdGlvbiBmaWxlIGZvciBkZWZpbmluZyBzZXR0aW5ncyBmb3IgdmFyaW91cyBSZWRvY2x5IHRvb2xzLCBpbmNsdWRpbmcgdGhlIGxpbnQgdG9vbCBhbmQgcmVmZXJlbmNlIGRvY3MgZW5naW5lLgogLSBgb3BlbmFwaWA6IFRvcC1sZXZlbCBmb2xkZXIgdGhhdCBjb250YWlucyB5b3VyIE9wZW5BUEkgZGVmaW5pdGlvbiwgYG9wZW5hcGkueWFtbGAgZW50cnlwb2ludCBmaWxlLCBhbmQgc3ViLWZvbGRlcnMgZm9yIGBwYXRoc2AsIGBjb21wb25lbnRzYCwgYW5kIGBjb2RlX3NhbXBsZXNgLgogLSBgY29kZV9zYW1wbGVzYDogRm9sZGVyIGZvciBvcmdhbml6aW5nIGNvZGUgc2FtcGxlcyBpbnRvIHN1Yi1mb2xkZXJzLCBzdWNoIGFzIEMjIGFuZCBQSFAuCiAtIGBjb21wb25lbnRzYDogRm9sZGVyIGZvciBvcmdhbml6aW5nIHJldXNhYmxlIGNvbXBvbmVudHMgaW50byBzdWItZm9sZGVycywgc3VjaCBhcyBgc2NoZW1hYCBhbmQgYHJlc3BvbnNlYCBvYmplY3RzLgogLSBgcGF0aHNgOiBGb2xkZXIgZm9yIG9yZ2FuaXppbmcgcGF0aCBkZWZpbml0aW9ucy4gRWFjaCBwYXRoIHNob3VsZCBiZSByZWZlcmVuY2VkIGZyb20gdGhlIGBvcGVuYXBpLnlhbWxgIGVudHJ5cG9pbnQgZmlsZS4KCiMjIENvbW1hbmRzCgpUaGUgZ2VuZXJhdGVkIHJlcG9zaXRvcnkgaW5zdGFsbHMgYSBkZXBlbmRlbmN5IGZvciBvdXIgYHJlZG9jbHktY2xpYCB0b29sIHdoaWNoIHN1cHBvcnRzIHRoZSBmb2xsb3dpbmcgY29tbWFuZHM6CgogLSBgbnBtIHN0YXJ0YDogU3RhcnRzIHRoZSBwcmV2aWV3IHNlcnZlcgogLSBgbnBtIHJ1biBidWlsZGA6IEJ1bmRsZXMgYSBtdWx0aS1maWxlIE9wZW5BUEkgZGVmaW5pdGlvbiBpbnRvIGEgc2luZ2xlIGZpbGUKIC0gYG5wbSB0ZXN0YDogVmFsaWRhdGVzIHRoZSBPcGVuQVBJIGRlZmluaXRpb24KCioqTm90ZToqKiBBZGRpdGlvbmFsIHNjcmlwdGVkIHNob3J0Y3V0cyBhcmUgZGVmaW5lZCBpbiB0aGUgcmVwb3NpdG9yeSdzIGBwYWNrYWdlLmpzb25gIGZpbGUuCgojIyBDb250cmlidXRlCgpJbnRlcmVzdGVkIGluIGNvbnRyaWJ1dGluZyB0byB0aGlzIHByb2plY3Q/IEhlcmUgYXJlIHNvbWUgd2F5cyB5b3UgY2FuIHN1cHBvcnQgdXM6CgogLSBTdWJtaXQgYSBwdWxsIHJlcXVlc3QuCiAtIFN0YXIgdXMgb24gR2l0aHViLgogLSBUZWxsIGEgZnJpZW5kIG9yIGNvbGxlYWd1ZSBhYm91dCB1cyAob3IgVHdlZXQgYWJvdXQgdXMpLgogLSBXcml0ZSBhbiBhcnRpY2xlIG9yIGJsb2cgcG9zdC4gTGV0IHVzIGtub3cgYnkgb3BlbmluZyBhbiBpc3N1ZSB3aXRoIGEgbGluayB0byB0aGUgYXJ0aWNsZS4KIC0gTG9va2luZyB0byBidWlsZCBhIG1vZGVybiBkb2N1bWVudGF0aW9uIHdvcmtmbG93PyBPdXIgW2NvbW1lcmNpYWwgcHJvZHVjdHNdKGh0dHBzOi8vcmVkb2MubHkpIGNhbiBoZWxwIHlvdSBtYWludGFpbiBhbmQgZGVwbG95IEFQSSByZWZlcmVuY2UgZG9jcyBhbmQgZGV2ZWxvcGVyIHBvcnRhbHMuCgpbbnBtLWltYWdlXTogaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL2dlbmVyYXRvci1vcGVuYXBpLXJlcG8uc3ZnCltucG0tdXJsXTogaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZS9nZW5lcmF0b3Itb3BlbmFwaS1yZXBvCltkYXZpZGRtLWltYWdlXTogaHR0cHM6Ly9kYXZpZC1kbS5vcmcvUmViaWxseS9nZW5lcmF0b3Itb3BlbmFwaS1yZXBvLnN2Zz90aGVtZT1zaGllbGRzLmlvCltkYXZpZGRtLXVybF06IGh0dHBzOi8vZGF2aWQtZG0ub3JnL1JlYmlsbHkvZ2VuZXJhdG9yLW9wZW5hcGktcmVwbwo= readmeEtag: '"fb93a46a6a07aa48f1014b29af30d0c056858f5e"' readmeLastModified: Tue, 09 Jul 2024 12:48:45 GMT repositoryId: 58973101 description: >- 🤖 Generator for GH repo to help you manage the OpenAPI definition lifecycle created: '2016-05-16T22:45:23Z' updated: '2026-02-04T05:17:36Z' language: JavaScript archived: false stars: 691 watchers: 21 forks: 111 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"312fd1e5d2ac176d9d8df5cd78923046f8a6899496f5a78956ce0068e751d55e"' repoLastModified: Wed, 04 Feb 2026 05:17:36 GMT foundInMaster: true v3_1: true category: - Code Generators - Parsers id: 63b494773063a90f98a1731dc9061bec oldLocations: - https://github.com/rebilly/generator-openapi-repo - source: openapi3 tags repository: https://github.com/apis-guru/awesome-openapi3 v3: true repositoryMetadata: base64Readme: >- IyBhd2Vzb21lLW9wZW5hcGkzIFshW0F3ZXNvbWVdKGh0dHBzOi8vY2RuLnJhd2dpdC5jb20vc2luZHJlc29yaHVzL2F3ZXNvbWUvZDczMDVmMzhkMjlmZWQ3OGZhODU2NTJlM2E2M2UxNTRkZDhlODgyOS9tZWRpYS9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vc2luZHJlc29yaHVzL2F3ZXNvbWUpCkEgbGlzdCBvZiBhd2Vzb21lIHByb2plY3RzIHJlbGF0ZWQgdG8gT3BlbkFQSSAzLjAueC4KCjxkaXYgYWxpZ249ImNlbnRlciI+CiAgPGltZyBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL2F3ZXNvbWUtb3BlbmFwaTMvYmxvYi9tYXN0ZXIvb3BlbmFwaV9hd2Vzb21lMS5wbmc/cmF3PXRydWUiLz4KICA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL0FQSXMtZ3VydS9vcGVuYXBpLWRpcmVjdG9yeS9tYXN0ZXIvYnJhbmRpbmcvaWNvbi05Nng5Ni5wbmciLz4KPC9kaXY+CgpXaHkgbm90IG1ha2UgeW91ciBwcm9qZWN0IGRpc2NvdmVyYWJsZSBieSB1c2luZyB0aGUgdG9waWMgW29wZW5hcGkzXShodHRwczovL2dpdGh1Yi5jb20vc2VhcmNoP3V0Zjg9JUUyJTlDJTkzJnE9dG9waWMlM0FvcGVuYXBpMyZ0eXBlPVJlcG9zaXRvcmllcyZyZWY9YWR2c2VhcmNoJmw9Jmw9KSBvbiBHaXRIdWIgYW5kIHVzaW5nIHRoZSBoYXNodGFncyAqKiNvcGVuYXBpMyoqIGFuZCAqKiNPQVN2MyoqIG9uIHNvY2lhbCBtZWRpYT8KCiMjIFRvb2xzCgoqIFBsZWFzZSBzZWUgW0FQSXMuZ3VydSBBd2Vzb21lLU9wZW5BUEkzXShodHRwczovL2FwaXMuZ3VydS9hd2Vzb21lLW9wZW5hcGkzLykKCiMjIENvbnRyaWJ1dGluZwoKVGhlIGJlc3Qgd2F5IHRvIGdldCB5b3VyIHByb2plY3QgYWRkZWQgdG8gdGhlIGxpc3QgaXMgdG8gdGFnIGl0IHdpdGggdGhlIGdpdGh1YiB0b3BpYyBgb3BlbmFwaTNgLiAKClB1bGwgcmVxdWVzdHMgc2hvdWxkIG9ubHkgYmUgZm9yIHZpc3VhbCAvIGZ1bmN0aW9uYWwgY2hhbmdlcywgb3IgcHJvamVjdHMvcHJvZHVjdHMgbm90IGhvc3RlZCBvbiBHaXRIdWIuCgojIyBBUEkgQWNjZXNzCgoqIFtjYXRlZ29yaWVzLmpzb25dKGh0dHBzOi8vYXBpcy5ndXJ1L2F3ZXNvbWUtb3BlbmFwaTMvYXBpL2NhdGVnb3JpZXMuanNvbikKKiBbdG9vbHMuanNvbl0oaHR0cHM6Ly9hcGlzLmd1cnUvYXdlc29tZS1vcGVuYXBpMy9hcGkvdG9vbHMuanNvbikKClRoZSByYXcgZGF0YSBjb250YWlucyBPcGVuQVBJIDIuMCBhbmQgU3dhZ2dlciAxLngtcmVsYXRlZCBwcm9qZWN0cy4KCiMjIFJTUyBGZWVkCgoqIFtmZWVkLnhtbF0oaHR0cHM6Ly9hcGlzLmd1cnUvYXdlc29tZS1vcGVuYXBpMy9yc3MvZmVlZC54bWwpCgo= readmeEtag: '"65628e18be602c1fee2850522453b55ca1181446"' readmeLastModified: Sun, 16 Apr 2023 14:15:56 GMT repositoryId: 83716646 description: >- 😎 A list of awesome projects related to OpenAPI 3.0.x, curated by the community created: '2017-03-02T19:31:41Z' updated: '2026-02-06T00:03:58Z' language: JavaScript archived: false stars: 697 watchers: 32 forks: 60 owner: APIs-guru logo: https://avatars.githubusercontent.com/u/10975548?v=4 license: BSD-3-Clause repoEtag: '"82791c657f55a8e2a6801184f95f22f7d3993748cbb80dd9857cd2d47b4ccff7"' repoLastModified: Fri, 06 Feb 2026 00:03:58 GMT foundInMaster: true category: Parsers id: 64d4029fd6b82879b746263cac7ce274 - source: openapi3 tags repository: https://github.com/cdimascio/generator-express-no-stress v3: true repositoryMetadata: base64Readme: >- # 🚂 generator-express-no-stress

![](https://img.shields.io/badge/status-stable-green.svg) ![](https://img.shields.io/npm/v/generator-express-no-stress.svg) [![Codacy Badge](https://api.codacy.com/project/badge/Grade/56c006ccc44c47f49d12b6b35fcf35da)](https://www.codacy.com/app/cdimascio/generator-express-no-stress?utm_source=github.com&utm_medium=referral&utm_content=cdimascio/generator-express-no-stress&utm_campaign=Badge_Grade) [![](https://img.shields.io/gitter/room/cdimascio-oss/community?color=%23eb205a)](https://gitter.im/cdimascio-oss/community) [![All Contributors](https://img.shields.io/badge/all_contributors-4-orange.svg?style=flat-square)](#contributors-) ![](https://img.shields.io/badge/license-MIT-blue.svg)

Create awesome [Express.js](http://www.expressjs.com) applications with best of breed tech including ES.next via [Babel.js](https://babeljs.io/), structured logging with [Pino](https://github.com/pinojs/pino), API validation and interactive documentation using an [OpenAPI 3.x](https://swagger.io/specification/), environment based config with [dotenv](https://github.com/motdotla/dotenv), and linting with [ESLint](http://eslint.org/).

[![GitHub stars](https://img.shields.io/github/stars/cdimascio/generator-express-no-stress.svg?style=social&label=Star&maxAge=2592000)](https://GitHub.com/cdimascio/generator-express-no-stress/stargazers/) [![Twitter URL](https://img.shields.io/twitter/url/https/github.com/cdimascio/generator-express-no-stress.svg?style=social)](https://twitter.com/intent/tweet?text=Check%20out%20express-no-stress%20by%20%40CarmineDiMascio%20https%3A%2F%2Fgithub.com%2Fcdimascio%2Fgenerator-express-no-stress%20%F0%9F%91%8D)

<p align="center">
<img src="https://raw.githubusercontent.com/cdimascio/generator-express-no-stress/master/assets/express-no-stress-logo-v.png">
</p>

generator-express-no-stress gets you up and running in seconds. It's ridiculously easy to configure. Heck, just take the defaults. Start it. Write code.

This generator scaffolds a fully functioning REST API server, complete with interactive documentation, API request and response validation, structured logging, environment driven config, and more. Simply run the generator and smile :-D

[Here's what you get!](#what-you-get)

## Install

_Requires Node 8 or greater_

```shell
npm install -g yo@4.3.0 generator-express-no-stress
```

- See [here](#usage-cli) for use with Yarn and/or Docker
- See [here](#faqs) for Node 6 support

## Scaffold

```shell
yo express-no-stress myapp
cd myapp
```

## Run

Run in _development mode_:

```shell
npm run dev
```

Package and run in _production mode_

```shell
npm start
```

## Test

```shell
npm test
```

## Debug

Run one of the following, then attach your favorite inspector e.g. [VSCode](#debug-in-vscode):

```shell
# debug the server
npm run dev:debug

# debug the tests
npm run test:debug
```

## Try it!

- Interactive API doc at [http://localhost:3000/api-explorer](http://localhost:3000/api-explorer)
- Landing page at [http://localhost:3000](http://localhost:3000)

---

## Usage: CLI

```shell
yo express-no-stress [appname] [--yarn] [--docker]
```

| Option     | default | Description                                                                |
| ---------- | ------- | -------------------------------------------------------------------------- |
| `appname`  | myapp   | The application folder                                                     |
| `--yarn`   | -       | Use the [`yarn`](https://yarnpkg.com) package manager, instead of `npm`    |
| `--docker` |         | Install [Docker](https://www.docker.com/) artifacts including a Dockerfile |

## Usage: Project

The sections below describe all usage options available once the project is generated/scaffolded.

### npm targets

| Target               | Description                                                              |
| -------------------- | ------------------------------------------------------------------------ |
| `npm run dev`        | Run in _development_ mode                                                |
| `npm run dev:debug`  | Debug in _development_ mode                                              |
| `npm run test`       | Run tests                                                                |
| `npm run test:debug` | Debug tests                                                              |
| `npm run lint`       | View a listing of all errors discovered by the linter                    |
| `npm run lint:fix`   | Fix all errors discovered by the linter                                  |
| `npm start`          | Run the in _production_ mode. |

### Debug in VSCode

Add these [contents](https://github.com/cdimascio/generator-express-no-stress/blob/next/assets/.vscode/launch.json) to your `.vscode/launch.json` file

### Debug in WebStorm

1. Start debug in _development_ mode via `npm run dev:debug`
2. From the "Run" menu, select "Debug"
3. Select "Edit Configurations..."
4. From the list of Templates on the left side of the dialog, select "Attach to Node.js/Chrome"
5. Press the "Debug" button to attach the WebStorm debugger


### Deploy to the Cloud

e.g. CloudFoundry

```
cf push myapp
```

### Use Yarn

```
# scaffold
yo express-no-stress myapp --yarn
cd myapp

# run in development mode
yarn run dev

# run in production mode
yarn start

# test
yarn test
```

---

## What you get!

- [Express.js](http://www.expressjs.com) - Fast, unopinionated
  , minimalist web framework for Node.js
- [Pino](https://github.com/pinojs/pino) - Extremely fast node.js logger, inspired by Bunyan. It also includes a shell utility to pretty-print its log files
- [dotenv](https://github.com/motdotla/dotenv) - Loads environment variables from .env for nodejs projects
- [ESLint](http://eslint.org/) - a pluggable linting utility for JavaScript and JSX

  Choose from the following ESLint lint rules:

     - [Airbnb](https://github.com/airbnb/javascript) - A mostly reasonable approach to JavaScript
     - [Prettier](https://github.com/prettier/prettier) - Prettier is an opinionated code formatter

- [Swagger](http://swagger.io/) - is a simple yet powerful representation of your RESTful API
- [SwaggerUI](http://swagger.io/) - dynamically generate beautiful documentation and sandbox from a Swagger-compliant API

### API Validation

Simply describe your APIs with Swagger and automagically get for free:

- Interactive documentation
- API request validation (OpenAPI 3.x)
- API response validation
  - To enable set `OPENAPI_ENABLE_RESPONSE_VALIDATION=true` in `.env`
  

#### Interactive API Doc

![](https://raw.githubusercontent.com/cdimascio/generator-express-no-stress/master/assets/interactive-doc1.png)

#### API Validation!

Oops! I the API caller forgot to pass a `name` field, no stress, we've got this!

![](https://raw.githubusercontent.com/cdimascio/generator-express-no-stress/master/assets/api-validation.png)

### Structured Logging

Structured logging out of the box!

#### raw

![](https://raw.githubusercontent.com/cdimascio/generator-express-no-stress/master/assets/logging-raw.png)

#### pretty

Structured logging pretty printed by default - great for dev!

![](https://raw.githubusercontent.com/cdimascio/generator-express-no-stress/master/assets/logging-pretty.png)

### API Validation Example

Simply describe your APIs with Swagger and automatically get:

- API request validation
- Interactive documentation

### example

#### Swagger API spec

```yaml
swagger: '2.0'
info:
  version: 1.0.0
  title: myapp
  description: My cool app
basePath: /api/v1
tags:
  - name: Examples
    description: Simple example endpoints
  - name: Specification
    description: The swagger API specification

consumes:
  - application/json
produces:
  - application/json

definitions:
  ExampleBody:
    type: object
    title: example
    required:
      - name
    properties:
      name:
        type: string
        example: no_stress

paths:
  /examples:
    get:
      tags:
        - Examples
      description: Fetch all examples
      responses:
        200:
          description: Returns all examples
    post:
      tags:
        - Examples
      description: Create a new example
      parameters:
        - name: example
          in: body
          description: an example
          required: true
          schema:
            $ref: '#/definitions/ExampleBody'
      responses:
        200:
          description: Returns all examples

  /examples/{id}:
    get:
      tags:
        - Examples
      parameters:
        - name: id
          in: path
          required: true
          description: The id of the example to retrieve
          type: integer
      responses:
        200:
          description: Return the example with the specified id
        404:
          description: Example not found

  /spec:
    get:
      tags:
        - Specification
      responses:
        200:
          description: Return the API specification
```

#### Invoke a POST request via the Interactive doc

![](https://raw.githubusercontent.com/cdimascio/generator-express-no-stress/master/assets/interactive-doc.png)

### Linting

express-no-stress uses [ESLint](http://eslint.org/) and provides two choices, Airbnb or Prettier.

To add your own ESLint customizations, edit`eslint.conf.mjs`.

Note that the Airbnb variant provides a slightly modified Airbnb [base](https://github.com/airbnb/javascript/tree/master/packages/eslint-config-airbnb-base) configuration.

## FAQ
**Q**: How do I modify the example API and make it my own?

**A**: There are two key files that enable you to customize and describe your API:
1. `server/routes.js` - This references the implementation of all of your routes. Add as many routes as you like and point each route your express handler functions.
2. `server/common/api.yaml` - This file contains your [OpenAPI spec](https://swagger.io/specification/). Describe your API here. It's recommended that you to declare any and all validation logic in this YAML. `express-no-stress-typescript`  uses [express-openapi-validator](https://github.com/cdimascio/express-openapi-validator) to automatically handle all API validation based on what you've defined in the spec.

**Q**: I previously generated an app, but I want to change the API root. How do I do this?

**A**: You need to make to small changes
  1. Modify `server/routes.js`
  ```javascript
     // Change your original path e.g. /api/v1/examples, to:
     app.use('/api/v2/examples', examplesRouter);
   ```

  2. Modify `server/common/api.yaml` and update the api root:
  ```yaml
    # Change e.g. /api/v1 to /api/v2
    servers:
    - url: /api/v2   
  ```

## License

[MIT](LICENSE)


<a href="https://www.buymeacoffee.com/m97tA5c" target="_blank"><img src="https://bmc-cdn.nyc3.digitaloceanspaces.com/BMC-button-images/custom_images/yellow_img.png" alt="Buy Me A Coffee" style="height: auto !important;width: auto !important;" ></a>

## Contributors ✨

Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://github.com/leosuncin"><img src="https://avatars1.githubusercontent.com/u/4307697?v=4" width="100px;" alt=""/><br /><sub><b>Jaime Leonardo Suncin Cruz</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=leosuncin" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/amygdaloideum"><img src="https://avatars2.githubusercontent.com/u/18416252?v=4" width="100px;" alt=""/><br /><sub><b>Daniel Bornstrand</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=amygdaloideum" title="Code">💻</a></td>
    <td align="center"><a href="http://www.about.me/jasoncorns"><img src="https://avatars2.githubusercontent.com/u/3839416?v=4" width="100px;" alt=""/><br /><sub><b>Jason Corns</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=JasonAllenCorns" title="Documentation">📖</a></td>
    <td align="center"><a href="http://frankcalise.com"><img src="https://avatars0.githubusercontent.com/u/374022?v=4" width="100px;" alt=""/><br /><sub><b>Frank Calise</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=frankcalise" title="Documentation">📖</a></td>
    <td align="center"><a href="https://blog.daskepon.com/"><img src="https://avatars0.githubusercontent.com/u/6374952?v=4" width="100px;" alt=""/><br /><sub><b>Daisuke Tsuji</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=daskepon" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/uronly14me"><img src="https://avatars2.githubusercontent.com/u/5186814?v=4" width="100px;" alt=""/><br /><sub><b>Sangbeom Han</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=uronly14me" title="Documentation">📖</a></td>
    <td align="center"><a href="http://mkly.io"><img src="https://avatars1.githubusercontent.com/u/965353?v=4" width="100px;" alt=""/><br /><sub><b>Mike Lay</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=mkly" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/jodejar214"><img src="https://avatars2.githubusercontent.com/u/9385902?v=4" width="100px;" alt=""/><br /><sub><b>jodejar214</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress/commits?author=jodejar214" title="Code">💻</a></td>
  </tr>
</table>

<!-- markdownlint-enable -->
<!-- prettier-ignore-end -->
<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
 readmeEtag: '"f100c4a0fb6a317a89d1d1a431a9baac0cba9fe9"' readmeLastModified: Sun, 09 Mar 2025 03:13:18 GMT repositoryId: 80230875 description: 🚂 A Yeoman generator for Express.js based 12-factor apps and apis created: '2017-01-27T17:45:14Z' updated: '2026-01-21T09:44:37Z' language: JavaScript archived: false stars: 601 watchers: 21 forks: 81 owner: cdimascio logo: https://avatars.githubusercontent.com/u/4706618?v=4 license: MIT repoEtag: '"45660b1c07789b5ac5d09dfc63561343260118db9033d86da379d28f60f53f9e"' repoLastModified: Wed, 21 Jan 2026 09:44:37 GMT foundInMaster: true category: Description Validators id: a4618ff5d41070e7337f3e3d8c2805d7 - source: openapi3 tags repository: https://github.com/mixmoe/hibiapi v3: true repositoryMetadata: base64Readme: >- <!-- spell-checker: disable -->
<!-- markdownlint-disable MD033 MD041 -->

<img src=".github/logo.svg" align="right">

<div align="left">

# HibiAPI

**_一个实现了多种常用站点的易用化 API 的程序._**

**_A program that implements easy-to-use APIs for a variety of commonly used sites._**

[![Demo Version](https://img.shields.io/badge/dynamic/json?label=demo%20status&query=%24.info.version&url=https%3A%2F%2Fapi.obfs.dev%2Fopenapi.json&style=for-the-badge&color=lightblue)](https://api.obfs.dev)

![Lint](https://github.com/mixmoe/HibiAPI/workflows/Lint/badge.svg)
![Test](https://github.com/mixmoe/HibiAPI/workflows/Test/badge.svg)
[![Coverage](https://codecov.io/gh/mixmoe/HibiAPI/branch/main/graph/badge.svg)](https://codecov.io/gh/mixmoe/HibiAPI)

[![PyPI](https://img.shields.io/pypi/v/hibiapi)](https://pypi.org/project/hibiapi/)
![PyPI - Downloads](https://img.shields.io/pypi/dm/hibiapi)
![PyPI - Python Version](https://img.shields.io/pypi/pyversions/hibiapi)
![PyPI - License](https://img.shields.io/pypi/l/hibiapi)

![GitHub last commit](https://img.shields.io/github/last-commit/mixmoe/HibiAPI)
![GitHub commit activity](https://img.shields.io/github/commit-activity/m/mixmoe/hibiapi)
![Lines of code](https://img.shields.io/tokei/lines/github/mixmoe/hibiapi)
[![GitHub stars](https://img.shields.io/github/stars/mixmoe/HibiAPI)](https://github.com/mixmoe/HibiAPI/stargazers)
[![GitHub forks](https://img.shields.io/github/forks/mixmoe/HibiAPI)](https://github.com/mixmoe/HibiAPI/network)
[![GitHub issues](https://img.shields.io/github/issues/mixmoe/HibiAPI)](https://github.com/mixmoe/HibiAPI/issues)

</div>

---

## 前言

- `HibiAPI`提供多种网站公开内容的 API 集合, 它们包括:

  - Pixiv 的图片和小说相关信息获取和搜索
  - Bilibili 的视频/番剧等信息获取和搜索
  - 网易云音乐的音乐/MV 等信息获取和搜索
  - 百度贴吧的帖子内容的获取
  - [爱壁纸](https://adesk.com/)的横版和竖版壁纸获取
  - 哔咔漫画的漫画信息获取和搜索
  - …

- 该项目的前身是 Imjad API[^1]
  - 由于它的使用人数过多, 致使调用超出限制, 所以本人希望提供一个开源替代来供社区进行自由地部署和使用, 从而减轻一部分该 API 的使用压力

[^1]: [什么是 Imjad API](https://github.com/mixmoe/HibiAPI/wiki/FAQ#%E4%BB%80%E4%B9%88%E6%98%AFimjad-api)

## 优势

### 开源

- 本项目以[Apache-2.0](./LICENSE)许可开源, 请看[开源许可](#开源许可)一节

### 高效

- 使用 Python 的[异步机制](https://docs.python.org/zh-cn/3/library/asyncio.html), 由[FastAPI](https://fastapi.tiangolo.com/)驱动, 带来高效的使用体验 ~~虽然性能瓶颈压根不在这~~

### 稳定

- 在代码中广泛使用了 Python 的[类型提示支持](https://docs.python.org/zh-cn/3/library/typing.html), 使代码可读性更高且更加易于维护和调试

- 在开发初期起就一直使用多种现代 Python 开发工具辅助开发, 包括:

  - 使用 [PyLance](https://marketplace.visualstudio.com/items?itemName=ms-python.vscode-pylance) 进行静态类型推断
  - 使用 [Flake8](https://flake8.pycqa.org/en/latest/) 对代码格式进行检查
  - 使用 [Black](https://black.readthedocs.io/en/stable/) 格式化代码以提升代码可读性

- 不直接使用第三方开发的 API 调用库, 而是全部用更加适合 Web 应用的逻辑重写第三方 API 请求, 更加可控 ~~疯狂造轮子~~

## 已实现 API[^2]

[^2]: 请查看 [#1](https://github.com/mixmoe/HibiAPI/issues/1)

- [x] Pixiv
- [x] 网易云音乐
- [ ] ~~一言~~ (其代替方案<https://hitokoto.cn>提供的方案已足够好, 暂不考虑支持)
- [x] Bilibili
- [x] 二维码
- [ ] ~~企鹅 FM~~ (似乎用的人不是很多)
- [x] 百度贴吧
- [x] 爱壁纸
- [x] 哔咔漫画

## 部署指南

- 手动部署指南: **[点击此处查看](https://github.com/mixmoe/HibiAPI/wiki/Deployment)**

## 应用实例

**我有更多的应用实例?** [立即 PR!](https://github.com/mixmoe/HibiAPI/pulls)

- [`journey-ad/pixiv-viewer`](https://github.com/journey-ad/pixiv-viewer)

  - **又一个 Pixiv 阅览工具**

- 公开搭建实例
  | **站点名称** | **网址** | **状态** |
  | :--------------------------: | :-----------------------------: | :---------------------: |
  | **官方 Demo[^3]** | <https://api.obfs.dev> | ![official][official] |
  | [MyCard](https://mycard.moe) | <https://hibi.moecube.com> | ![mycard][mycard] |

[^3]: 为了减轻服务器负担, Demo 服务器已开启了 Cloudflare 全站缓存, 如果有实时获取更新的需求, 请自行搭建或使用其他部署实例

[official]: https://img.shields.io/website?url=https%3A%2F%2Fapi.obfs.dev%2Fopenapi.json
[mycard]: https://img.shields.io/website?url=https%3A%2F%2Fhibi.moecube.com%2Fopenapi.json

## 特别鸣谢

[**@journey-ad**](https://github.com/journey-ad) 大佬的 **Imjad API**, 它是本项目的起源

### 参考项目

> **正是因为有了你们, 这个项目才得以存在**

- Pixiv: [`Mikubill/pixivpy-async`](https://github.com/Mikubill/pixivpy-async) [`upbit/pixivpy`](https://github.com/upbit/pixivpy)

- Bilibili: [`SocialSisterYi/bilibili-API-collect`](https://github.com/SocialSisterYi/bilibili-API-collect) [`soimort/you-get`](https://github.com/soimort/you-get)

- 网易云音乐: [`metowolf/NeteaseCloudMusicApi`](https://github.com/metowolf/NeteaseCloudMusicApi) [`greats3an/pyncm`](https://github.com/greats3an/pyncm) [`Binaryify/NeteaseCloudMusicApi`](https://github.com/Binaryify/NeteaseCloudMusicApi)

- 百度贴吧: [`libsgh/tieba-api`](https://github.com/libsgh/tieba-api)

- 哔咔漫画：[`niuhuan/pica-rust`](https://github.com/niuhuan/pica-rust) [`abbeyokgo/PicaComic-Api`](https://github.com/abbeyokgo/PicaComic-Api)

### 贡献者们

感谢这些为这个项目作出贡献的各位大佬:

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="http://kyomotoi.moe"><img src="https://avatars.githubusercontent.com/u/37587870?v=4?s=100" width="100px;" alt="Kyomotoi"/><br /><sub><b>Kyomotoi</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=Kyomotoi" title="Documentation">📖</a> <a href="https://github.com/mixmoe/HibiAPI/commits?author=Kyomotoi" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://thdog.moe"><img src="https://avatars.githubusercontent.com/u/46120251?v=4?s=100" width="100px;" alt="城倉奏"/><br /><sub><b>城倉奏</b></sub></a><br /><a href="#example-shirokurakana" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://skipm4.com"><img src="https://avatars.githubusercontent.com/u/40311581?v=4?s=100" width="100px;" alt="SkipM4"/><br /><sub><b>SkipM4</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=SkipM4" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/leaf7th"><img src="https://avatars.githubusercontent.com/u/38352552?v=4?s=100" width="100px;" alt="Nook"/><br /><sub><b>Nook</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=leaf7th" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jiangzhuochi"><img src="https://avatars.githubusercontent.com/u/50538375?v=4?s=100" width="100px;" alt="Jocky Chiang"/><br /><sub><b>Jocky Chiang</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=jiangzhuochi" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/cleoold"><img src="https://avatars.githubusercontent.com/u/13920903?v=4?s=100" width="100px;" alt="midori"/><br /><sub><b>midori</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=cleoold" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://www.2yo.cc"><img src="https://avatars.githubusercontent.com/u/41198038?v=4?s=100" width="100px;" alt="Pretty9"/><br /><sub><b>Pretty9</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=Pretty9" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://nocilol.me/"><img src="https://avatars.githubusercontent.com/u/16256221?v=4?s=100" width="100px;" alt="Jad"/><br /><sub><b>Jad</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/issues?q=author%3Ajourney-ad" title="Bug reports">🐛</a> <a href="#ideas-journey-ad" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://nanoka.top"><img src="https://avatars.githubusercontent.com/u/31837214?v=4?s=100" width="100px;" alt="Yumine Sakura"/><br /><sub><b>Yumine Sakura</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=asadahimeka" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/yeyang52"><img src="https://avatars.githubusercontent.com/u/107110851?v=4?s=100" width="100px;" alt="yeyang"/><br /><sub><b>yeyang</b></sub></a><br /><a href="https://github.com/mixmoe/HibiAPI/commits?author=yeyang52" title="Code">💻</a></td>
    </tr>
  </tbody>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

_本段符合 [all-contributors](https://github.com/all-contributors/all-contributors) 规范_

## 开源许可

    Copyright 2020-2021 Mix Technology

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

        http://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
 readmeEtag: '"89299b320ffcb534742e534a402ce9ace61aa4a7"' readmeLastModified: Tue, 23 Apr 2024 15:15:36 GMT repositoryId: 322624820 description: >- 一个实现了多种常用站点的易用化API的程序 / A program that implements easy-to-use APIs for a variety of commonly used sites. created: '2020-12-18T14:56:31Z' updated: '2026-02-02T03:51:47Z' language: Python archived: true stars: 575 watchers: 7 forks: 84 owner: mixmoe logo: https://avatars.githubusercontent.com/u/68723928?v=4 license: Apache-2.0 repoEtag: '"2825226cef9f5dfca4c1b5b27b6424d93794b7fdc1fe36ab45e16beb2e0ff405"' repoLastModified: Mon, 02 Feb 2026 03:51:47 GMT foundInMaster: true category: Testing id: a5ce8925c7b03da0bff459d65f63458a - source: openapi3 tags repository: https://github.com/fabien0102/openapi-codegen v3: true id: 949c5e572da015484f84ceadd84e19ef repositoryMetadata: base64Readme: >- <br>
<br>
<br>
<div align="center" style="margin-bottom: 16px">
  <img src="openapi-codegen-logo.svg" width="400px" />
</div>
<br>
<br>

  <div align="center">
    <a href="https://www.npmjs.com/package/@openapi-codegen/cli">
      <img alt="npm" src="https://img.shields.io/npm/v/@openapi-codegen/cli.svg?style=for-the-badge">
    </a>
    <a href="https://github.com/fabien0102/openapi-codegen/blob/main/LICENCE.md">
      <img alt="Read the documentation" src="https://img.shields.io/npm/l/@openapi-codegen/cli.svg?style=for-the-badge">
    </a>
<br>
<br>
    Tooling to give you full type-safety around OpenAPI specs.
  
  </div>

<br>

### You can generate

- TypeScript types
- Type-safe Generic Fetchers
- Type-safe React Query hooks (https://github.com/tanstack/query)

## Getting started

1. **Initialize the generator**

   ```bash
   npx @openapi-codegen/cli init
   ```

   <img style="max-width: 400px" src="https://user-images.githubusercontent.com/271912/194000679-5a4501b8-5fc0-430c-9217-028bf91a5dcd.gif">

   If you wish to change any of the selections made, you can do so in the generated `openapi-codegen.config.ts` file later..

2. **Start Generation**

   ```bash
   npx openapi-codegen gen {namespace}
   ```

   After the code generation is done, you will notice the following files:

   - `{namespace}Fetcher.ts` - defines a function that will make requests to your API.
   - `{namespace}Context.tsx` - the context that provides `{namespace}Fetcher` to other components.
   - `{namespace}Components.tsx` - generated React Query components (if you selected React Query as part of initialization).
   - `{namespace}Schemas.ts` - the generated Typescript types from the provided Open API schemas.

   &nbsp;

   > **Warning**
   >
   > If `{namespace}Fetcher.ts` or `{namespace}Context.tsx` already exist in the output folder, they will not be replaced. However, `{namespace}Components.tsx` and `{namespace}Schemas.ts` will be re-generated each time based on the Open API spec file provided.

3. **Configure the Fetcher** (optional)

   After the first step you should see a file called `{namespace}Fetcher.ts` in your ouput directory.

   By default it uses the built-in Fetch API, you are free to change this to your fetching library of choice (Axios, Got etc.)

   If your Open API spec contains a configured server, then the base URL for all requests will default to that server's URL. If no such configuration exists, you'll need to specify the base URL value.

4. **Install and Configure React Query** (optional)

   If during generator setup you picked `> React Query components`, then you will need to install and configure React Query in order for the generated React hooks to work properly:

   - [Install the React Query library](https://tanstack.com/query/latest/docs/framework/react/installation)
   - [Wire up the QueryClientProvider component](https://tanstack.com/query/latest/docs/framework/react/reference/QueryClientProvider) to connect and provide a QueryClient to your application

## Usage

### React Query components

Using [giphy specs](https://api.apis.guru/v2/specs/giphy.com/1.0/openapi.yaml) as example

This will generate lot of ready-to-use hooks like:

- `useRandomGif` -> Wrapper around `useQuery` with injected types
- `useSuspenseRandomGif` -> Same but with `useSuspense`

And you will have some `useMutation` if the api expose some (not the case with this example)

Here an example of usage of this generated api:

```tsx
import { useRandomGif } from "./giphy";

export function GifExplorer() {
  const [tag, setTag] = useState("");
  const { data, error, isError, isPending } = useRandomGif({
    queryParams: {
      tag,
    },
  });

  return (
    <div>
      <input value={tag} onChange={(e) => setTag(e.currentTarget.value)} />
      {isPending ? (
        <div>Loading…</div>
      ) : isError ? (
        <div>
          <pre>{error.payload ?? "Unknown error"}</pre>
        </div>
      ) : (
        // This is typed!
        <img src={data.data?.url} />
      )}
    </div>
  );
}
```

This also support `reactQuery.skipToken` to stay type-safe when you are waiting for params:

```diff
+ import { skipToken } from "@tanstack/react-query";

- const { data, error, isError, isPending } = useRandomGif({
-     queryParams: {
-       tag,
-     },
-   });
+ const { data, error, isError, isPending } = useRandomGif(
+     tag
+       ? skipToken
+       : {
+           queryParams: {
+             tag,
+           },
+         }
+   );
```

You can also use directly the queryFn for more advanced use cases:

```tsx
import { randomGifQuery } from "./giphy/giphyComponents";

const queryClient = new QueryClient();

queryClient.fetchQuery(randomGifQuery({}));
```

## Configuration

The only thing you need to manage is the configuration.
Everything is typed and self-documented, but just in case, you can find here example configuration below:

### Example Config

```ts
// openapi-codegen.config.ts
import { defineConfig } from "@openapi-codegen/cli";
import {
  generateSchemaTypes,
  generateReactQueryComponents,
} from "@openapi-codegen/typescript";

export default defineConfig({
  example: {
    // can be overridden from cli
    from: {
      source: "github",
      owner: "fabien0102",
      repository: "openapi-codegen",
      ref: "main",
      specPath: "examples/spec.yaml",
    },

    // can be overridden from cli
    outputDir: "src/queries",

    to: async (context) => {
      // You can transform the `context.openAPIDocument` here, can be useful to remove internal routes or fixing some known issues in the specs ;)

      // Generate all the schemas types (components & responses)
      const { schemasFiles } = await generateSchemaTypes(context, {
        /* config */
      });

      // Generate all react-query components
      await generateReactQueryComponents(context, {
        /* config*/
        schemasFiles,
      });
    },
  },
});
```

### Plugins

the `@openapi-codegen/cli` supports these generator plugins:

#### **generateSchemaTypes** (frontend/backend)

generate all schema types for your specification:

```ts
const { schemasFiles } = await generateSchemaTypes(context, {
  /* config */
});
```

output: `{namespace}Schemas.ts`

#### **generateFetchers** (frontend)

generate all fetchers with types for your specification

```ts
await generateFetchers(context, {
  /* config */
  schemasFiles,
});
```

output: `{namespace}Fetchers.ts`

#### **generateReactQueryComponents** (frontend)

generate all React Query Components for useQuery() and useMutation()

```ts
await generateReactQueryComponents(context, {
  /* config*/
  schemasFiles,
});
```

output: `{namespace}Components.ts`

This is also generate a query function that can be used in most of React Query functions.

Example with `queryClient.fetchQuery` (data loader case):

```ts
await queryClient.fetchQuery(getYourQueryNameQuery({}));
```

You can import any generator into the `to` section, those can be the ones provided by this project or your own custom ones. You have full control of what you are generating!

Have fun!

## Contributors ✨

Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://fabien0102.com/"><img src="https://avatars.githubusercontent.com/u/1761469?v=4?s=100" width="100px;" alt="Fabien BERNARD"/><br /><sub><b>Fabien BERNARD</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=fabien0102" title="Code">💻</a> <a href="#design-fabien0102" title="Design">🎨</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=fabien0102" title="Documentation">📖</a> <a href="#ideas-fabien0102" title="Ideas, Planning, & Feedback">🤔</a> <a href="#projectManagement-fabien0102" title="Project Management">📆</a> <a href="https://github.com/fabien0102/openapi-codegen/pulls?q=is%3Apr+reviewed-by%3Afabien0102" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/mpotomin"><img src="https://avatars.githubusercontent.com/u/639406?v=4?s=100" width="100px;" alt="mpotomin"/><br /><sub><b>mpotomin</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=mpotomin" title="Code">💻</a> <a href="#ideas-mpotomin" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/fabien0102/openapi-codegen/pulls?q=is%3Apr+reviewed-by%3Ampotomin" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/micha-f"><img src="https://avatars.githubusercontent.com/u/200647?v=4?s=100" width="100px;" alt="Michael Franzkowiak"/><br /><sub><b>Michael Franzkowiak</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=micha-f" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/SferaDev"><img src="https://avatars.githubusercontent.com/u/2181866?v=4?s=100" width="100px;" alt="Alexis Rico"/><br /><sub><b>Alexis Rico</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=SferaDev" title="Code">💻</a> <a href="#ideas-SferaDev" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://ned.im/"><img src="https://avatars.githubusercontent.com/u/271912?v=4?s=100" width="100px;" alt="Nedim Arabacı"/><br /><sub><b>Nedim Arabacı</b></sub></a><br /><a href="#question-needim" title="Answering Questions">💬</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=needim" title="Code">💻</a> <a href="#ideas-needim" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/antoniel"><img src="https://avatars.githubusercontent.com/u/17225358?v=4?s=100" width="100px;" alt="Antoniel Magalhães"/><br /><sub><b>Antoniel Magalhães</b></sub></a><br /><a href="#example-antoniel" title="Examples">💡</a> <a href="https://github.com/fabien0102/openapi-codegen/issues?q=author%3Aantoniel" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/DreierF"><img src="https://avatars.githubusercontent.com/u/5631865?v=4?s=100" width="100px;" alt="Florian Dreier"/><br /><sub><b>Florian Dreier</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=DreierF" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="http://fabianalthaus.de"><img src="https://avatars.githubusercontent.com/u/2795534?v=4?s=100" width="100px;" alt="Fabian Althaus"/><br /><sub><b>Fabian Althaus</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=el-j" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ci-vamp"><img src="https://avatars.githubusercontent.com/u/116516277?v=4?s=100" width="100px;" alt="ci-vamp"/><br /><sub><b>ci-vamp</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/issues?q=author%3Aci-vamp" title="Bug reports">🐛</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=ci-vamp" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://twitter.com/oalanoliv"><img src="https://avatars.githubusercontent.com/u/4368481?v=4?s=100" width="100px;" alt="Alan Oliveira"/><br /><sub><b>Alan Oliveira</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=alan-oliv" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Cellule"><img src="https://avatars.githubusercontent.com/u/4157103?v=4?s=100" width="100px;" alt="Michael &quot;Mike&quot; Ferris"/><br /><sub><b>Michael &quot;Mike&quot; Ferris</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=Cellule" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/rajzik"><img src="https://avatars.githubusercontent.com/u/10364836?v=4?s=100" width="100px;" alt="Jan Šilhan"/><br /><sub><b>Jan Šilhan</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/issues?q=author%3Arajzik" title="Bug reports">🐛</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=rajzik" title="Code">💻</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=rajzik" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/dan-cooke"><img src="https://avatars.githubusercontent.com/u/22816887?v=4?s=100" width="100px;" alt="Daniel Cooke"/><br /><sub><b>Daniel Cooke</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/issues?q=author%3Adan-cooke" title="Bug reports">🐛</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=dan-cooke" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Kusmeroglu"><img src="https://avatars.githubusercontent.com/u/1638544?v=4?s=100" width="100px;" alt="Linden Wright"/><br /><sub><b>Linden Wright</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=Kusmeroglu" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://tverdohleb.com/"><img src="https://avatars.githubusercontent.com/u/172711?v=4?s=100" width="100px;" alt="Valeriy"/><br /><sub><b>Valeriy</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/issues?q=author%3Atverdohleb" title="Bug reports">🐛</a> <a href="https://github.com/fabien0102/openapi-codegen/commits?author=tverdohleb" title="Code">💻</a> <a href="#ideas-tverdohleb" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://zenn.dev/watahaya"><img src="https://avatars.githubusercontent.com/u/15213369?v=4?s=100" width="100px;" alt="Isco"/><br /><sub><b>Isco</b></sub></a><br /><a href="https://github.com/fabien0102/openapi-codegen/commits?author=hayawata3626" title="Code">💻</a></td>
    </tr>
  </tbody>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
 readmeEtag: '"f72351ba76be36251b11a7144ede997c49b28a06"' readmeLastModified: Wed, 01 Oct 2025 08:48:09 GMT repositoryId: 417437020 description: A tool for generating code base on an OpenAPI schema. created: '2021-10-15T09:08:51Z' updated: '2026-02-05T17:21:17Z' language: TypeScript archived: false stars: 625 watchers: 4 forks: 73 owner: fabien0102 logo: https://avatars.githubusercontent.com/u/1761469?v=4 repoEtag: '"cbb7cfec48a84ec592e204c3800cdfb3d3a2df4d3f50f38496ab28e2320a638a"' repoLastModified: Thu, 05 Feb 2026 17:21:17 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/cloud-annotations/docusaurus-openapi v3: true repositoryMetadata: base64Readme: >- <h1 align="center">Docusaurus OpenAPI</h1>

<div align="center">

OpenAPI plugin for generating API reference docs in Docusaurus.

[![license](https://img.shields.io/badge/license-MIT-blue.svg)](https://github.com/cloud-annotations/docusaurus-plugin-openapi/blob/HEAD/LICENSE)
[![npm latest package](https://img.shields.io/npm/v/docusaurus-preset-openapi/latest.svg)](https://www.npmjs.com/package/docusaurus-preset-openapi)
[![npm downloads](https://img.shields.io/npm/dm/docusaurus-plugin-openapi.svg)](https://www.npmjs.com/package/docusaurus-preset-openapi)
[![build](https://github.com/cloud-annotations/docusaurus-plugin-openapi/actions/workflows/validate.yaml/badge.svg)](https://github.com/cloud-annotations/docusaurus-plugin-openapi/actions/workflows/validate.yaml)
<br/>
[![prettier](https://img.shields.io/badge/code_style-prettier-ff69b4.svg)](https://github.com/prettier/prettier)
[![Cypress.io](https://img.shields.io/badge/tested%20with-Cypress-04C38E.svg)](https://www.cypress.io/)
[![jest](https://jestjs.io/img/jest-badge.svg)](https://github.com/facebook/jest)
[![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg)](https://github.com/cloud-annotations/docusaurus-plugin-openapi/blob/HEAD/CONTRIBUTING.md#pull-requests)

</div>

<p align="center">

![](https://user-images.githubusercontent.com/4212769/85324376-b9e3d900-b497-11ea-9765-c42a8ad1ff61.png)

</p>

## Quick Overview

```sh
npx create-docusaurus-openapi my-website
cd my-website
npm start
```

> Coming from `v0.1.0`? See the [migration guide](https://github.com/cloud-annotations/docusaurus-plugin-openapi/releases/tag/v0.2.0).

_([npx](https://medium.com/@maybekatz/introducing-npx-an-npm-package-runner-55f7d4bd282b) comes with npm 5.2+ and higher)_

Then open [http://localhost:3000/](http://localhost:3000/) to see your site.<br>
When you’re ready to deploy to production, create a minified bundle with `npm run build`.

## Creating a new Site

**You’ll need to have Node 14.0.0 or later version on your local development machine** (but it’s not required on the server). We recommend using the latest LTS version. You can use [nvm](https://github.com/creationix/nvm#installation) (macOS/Linux) or [nvm-windows](https://github.com/coreybutler/nvm-windows#node-version-manager-nvm-for-windows) to switch Node versions between different projects.

To create a new site, you may choose one of the following methods:

- ### npx

  ```sh
  npx create-docusaurus-openapi my-website
  ```

  _([npx](https://medium.com/@maybekatz/introducing-npx-an-npm-package-runner-55f7d4bd282b) is a package runner tool that comes with npm 5.2+ and higher)_

- ### npm

  ```sh
  npm init docusaurus-openapi my-website
  ```

  _`npm init <initializer>` is available in npm 6+_

- ### yarn

  ```sh
  yarn create docusaurus-openapi my-website
  ```

  _[`yarn create <starter-kit-package>`](https://yarnpkg.com/lang/en/docs/cli/create/) is available in Yarn 0.25+_

It will create a directory called `my-website` inside the current folder.<br>
Inside that directory, it will generate the initial project structure and install the transitive dependencies:

```
my-website
├── blog
│   ├── 2019-05-28-hola.md
│   ├── 2019-05-29-hello-world.md
│   └── 2020-05-30-welcome.md
├── docs
│   ├── doc1.md
│   ├── doc2.md
│   ├── doc3.md
│   └── mdx.md
├── src
│   ├── css
│   │   └── custom.css
│   └── pages
│       ├── styles.module.css
│       └── index.js
├── static
│   └── img
├── .gitignore
├── openapi.json
├── docusaurus.config.js
├── babel.config.js
├── package.json
├── sidebars.js
└── README.md
```

- `/docusaurus.config.js` - A config file containing the site configuration. This can be used to configure the OpenAPI preset
- `/openapi.json` - The default OpenAPI definition that will be served _(path can be configured in `docusaurus.config.js`)_.

For more info see [project structure rundown](https://docusaurus.io/docs/installation#project-structure-rundown).

Once the installation is done, you can open your project folder:

```sh
cd my-website
```

Inside the newly created project, you can run some built-in commands:

### `npm start` or `yarn start`

Runs the site in development mode.<br>
Open [http://localhost:3000](http://localhost:3000) to view it in the browser.

The page will automatically reload if you make changes to the code.

### `npm run build` or `yarn build`

Builds the site for production to the `build` folder.

Docusaurus is a modern static website generator that will build the website into a directory of static contents, which can be copied to any static file hosting service like [GitHub pages](https://pages.github.com/), [Vercel](https://vercel.com/) or [Netlify](https://www.netlify.com/).

## Add to an existing Site

1. Install the dependency

   ```sh
   npm install docusaurus-preset-openapi
   ```

2. Edit your `docusaurus.config.js` file to use this preset
   ```js
   presets: [
     [
       "docusaurus-preset-openapi",
       /** @type {import('docusaurus-preset-openapi').Options} */
       {
         api: {
           path: "<PATH_TO_YOUR_OPENAPI_DOCUMENT>",
           routeBasePath: "/api",
         },
         docs: {
           sidebarPath: require.resolve("./sidebars.js"),
           routeBasePath: "/",
         },
         theme: {
           customCss: require.resolve("./src/css/custom.css"),
         },
       },
     ],
   ];
   ```

## Popular Alternatives

Docusaurus OpenAPI is great for:

- **Static generation** All OpenAPI operations will be rendered as static pages at build time, giving many performance benefits.
- **Demo panel** Allow users to try out your API with an interactive demo panel.
- **Native Docusaurus feel** Built from scratch to work with Docusaurus.

Here are a few common cases where you might want to try something else:

- If you need better support for more advanced OpenAPI features, check out [Redocusaurus](https://github.com/rohit-gohri/redocusaurus/). Redocusaurus embeds the much more mature OpenAPI documentation generator, [Redoc](https://github.com/Redocly/redoc), into Docusaurus as a React component.

## Contributing

We encourage you to contribute to Docusaurus OpenAPI! Please check out the
[Contributing to Docusaurus OpenAPI guide](https://github.com/cloud-annotations/docusaurus-plugin-openapi/blob/main/CONTRIBUTING.md) for guidelines about how to proceed.

## License

Docusaurus OpenAPI is released under the [MIT License](https://opensource.org/licenses/MIT).
 readmeEtag: '"fb57583ef6d0b6fe8c3ace74ef386b379a77d012"' readmeLastModified: Thu, 02 Jan 2025 22:02:55 GMT repositoryId: 273958287 description: 🦕 OpenAPI plugin for generating API reference docs in Docusaurus. created: '2020-06-21T18:06:52Z' updated: '2026-01-17T20:34:22Z' language: TypeScript archived: false stars: 537 watchers: 8 forks: 84 owner: cloud-annotations logo: https://avatars.githubusercontent.com/u/47490031?v=4 license: MIT repoEtag: '"bce561231a8197e53a758a6da7204e7dc719cfe8179c6a5860abd62c5c66e02a"' repoLastModified: Sat, 17 Jan 2026 20:34:22 GMT foundInMaster: true category: SDK id: 5acf9bbe5338a800925f2b0e23eca0a1 - source: openapi3 tags repository: https://github.com/oazapfts/oazapfts v3: true id: b866db159ac2a8a130a7c78d142b32bc repositoryMetadata: base64Readme: >- # 🍻 oazapfts!

[![CI](https://github.com/oazapfts/oazapfts/actions/workflows/ci.yml/badge.svg)](https://github.com/oazapfts/oazapfts/actions/workflows/ci.yml)
[![semantic-release: angular](https://img.shields.io/badge/semantic--release-angular-e10079?logo=semantic-release)](https://github.com/semantic-release/semantic-release)

Generate TypeScript clients to tap into OpenAPI servers.

![](https://avatars.githubusercontent.com/u/119607625?s=200&v=4)

## Features

- **AST-based**:
  Unlike other code generators `oazapfts` does not use templates to generate code but uses TypeScript's built-in API to generate and pretty-print an abstract syntax tree.
- **Fast**: The CLI does not use any of the common Java-based tooling, so the code generation is super fast.
- **Single file**: All functions and types are co-located in one single self-contained file.
- **Tree-shakeable**: Individually exported functions allow you to bundle only the ones you actually use.
- **Human friendly signatures**: The generated API methods don't leak any HTTP-specific implementation details. For example, all optional parameters are grouped together in one object, no matter whether they end up in the headers, path or query-string.

## Installation

```
npm install oazapfts
```

> **Note**
> With version 3.0.0 oazapfts has become a runtime dependency and the generated code does no longer include all the fetch logic.  
> As of 6.0.0 the runtime has been moved to a separate package, `@oazapfts/runtime`.

## Usage

```
oazapfts <spec> [filename]

Options:
--exclude, -e tag to exclude
--include, -i tag to include
--optimistic
--useEnumType
--useUnknown
--mergeReadWriteOnly
--argumentStyle=<positional | object> (default: positional)
--allSchemas
--futureStripLegacyMethods
```

Where `<spec>` is the URL or local path of an OpenAPI or Swagger spec (in either json or yml) and `<filename>` is the location of the `.ts` file to be generated. If the filename is omitted, the code is written to stdout.

### Options

- `--optimistic` generate a client in [optimistic mode](#optimistic-mode)

- `--useEnumType` generate enums instead of union types

- `--useUnknown` generate `unknown` instead of `any` for empty schemas

- `--mergeReadWriteOnly` by default oazapfs will generate separate types for read-only and write-only properties. This option will merge them into one type.

- `--argumentStyle` if "object" generated functions take single object style argument for parameters and requestBody, by default it's "positional" and parameters are separate as positional arguments

- `--allSchemas` generate types for all schemas included in the spec

- `--futureStripLegacyMethods` skip generating deprecated legacy method aliases.  
  By default, when an `operationId` contains special characters (like dots or colons, e.g. `scope1.userAccount.get`), oazapfts generates both a normalized method name (`scope1UserAccountGet`) and a deprecated fallback using the HTTP verb and path (`getUsersById`) for backward compatibility. Use this flag to only generate the normalized operationId-based names.
  The next major version will default to this behavior.

## Consuming the generated API

For each operation defined in the spec the generated API will export a function with a name matching the `operationId`. If no ID is specified, a reasonable name is generated from the HTTP verb and the path.

```ts
import * as api from "./my-generated-api.ts";
const res = api.getPetById(1);
```

> **Note**
> If your API is large, and you want to take advantage of tree-shaking to exclude unused code, use individual named imports instead:

```ts
import { getPetById } from "./my-generated-api.ts";
```

## Fetch options

The **last argument** of each function is an optional [`RequestOpts`](https://github.com/oazapfts/oazapfts/blob/27b296c6fc28fec4869f1b7e1a4a5585ebbd5ee9/src/runtime/index.ts#L5) object that can be used to pass options to the `fetch` call, for example to pass additional `headers` or an `AbortSignal` to cancel the request later on.

```ts
const res = getPetById(1, {
  credentials: "include",
  headers: {
    Authorization: `Bearer ${token}`,
  },
});
```

You can also use this to override the default `baseUrl` or to provide a custom `fetch` implementation.

> **Note**
> Instead of passing custom options to each function call, consider [overwriting the global defaults](#overriding-the-defaults).

## Optimistic vs. explicit responses

Oazapfts supports two different modes to handle results,
an [explicit](#explicit-mode) mode (the default) and an [optimistic](#optimistic-mode) mode, that makes the response handling less verbose.

## Explicit mode

By default, each function returns an `ApiResponse` object that exposes the `status` code, response `headers` and the `data`.

> **Note**
> This mode is best suited for APIs that return different types for different response codes or APIs where you need to access not only the response body, but also the response headers. If your API is simple, and you don't need this flexibility, consider using the [optimistic mode](#optimistic-mode) instead.

In explicit mode, each function returns a Promise for an `ApiResponse` which is an object with a `status` and a `data` property, holding the HTTP status code and the properly typed data from the response body.

Since an operation can return different types depending on the status code, the actual return type is a _union_ of all possible responses, discriminated by their status.

Consider the following code generated from the `petstore.json` example:

```ts
/**
 * Find pet by ID
 */
export function getPetById(petId: number, opts?: Oazapfts.RequestOpts) {
  return oazapfts.fetchJson<
    | {
        status: 200;
        data: Pet;
      }
    | {
        status: 400;
        data: string;
      }
    | {
        status: 404;
      }
  >(`/pet/${encodeURIComponent(petId)}`, {
    ...opts,
  });
}
```

In this case, the `data` property is typed as `Pet|string`. We can use a type guard to narrow down the type to `Pet`:

```ts
const res = await api.getPetById(1);
if (res.status === 200) {
  const pet = res.data;
  // pet is properly typed as Pet
}
if (res.status === 404) {
  const message = res.data;
  // message is a string
} else {
  // handle the error
}
```

The above code can be simplified by using the `handle` helper:

```ts
import { handle } from "@oazapfts/runtime";

await handle(api.getPetById(1), {
  200(pet) {
    // pet is properly typed as Pet
  },
  404(message) {
    // message is as string
  },
});
```

The helper will throw an `HttpError` error for any unhandled status code, unless you add a `default` handler:

```ts
await handle(api.getPetById(1), {
  200(pet) {
    // ...
  },
  default(status, data) {
    // handle error
  },
});
```

## Optimistic mode

You can opt into the _optimistic mode_ by using the `--optimistic` command line argument.

In this mode, each function will return a Promise for the happy path, i.e. the type specified for the first `2xx` response.

Looking back at our Pet Store example from above, consuming the response is now much easier and less verbose:

```ts
const pet = await api.getPetById(1);
// pet is now typed as Pet!
```

In case of a response other than `200` the promise will be rejected with a `HttpError`.

## Mixing both modes

Sometimes you might want to use the optimistic mode for some of your API calls, but need the full `ApiResponse` for others.

In that case, you can use the `ok`-helper function to selectively apply optimistic response handling:

```ts
import { ok } from "@oazapfts/runtime";

const pet = await ok(api.getPetById(1));
```

## Overriding the defaults

The generated file exports a `defaults` constant that can be used to override the `basePath`, provide a custom `fetch` implementation or to send additional `headers` with each request. Basically, you can set a default for any [fetch option](https://developer.mozilla.org/en-US/docs/Web/API/fetch#options) you want.

```ts
import * as api from "./api.ts";
import nodeFetch from "node-fetch";

// Override the spec's basePath
api.defaults.basePath = "https://example.com/api";

// Send this header with each request
api.defaults.headers = {
  access_token: "secret",
};

// Include credentials in CORS requests, too
api.defaults.credentials = "include";

// Use this instead of the global fetch
api.defaults.fetch = nodeFetch;
```

## Alternatives and integrations

If this library doesn't fit your needs, take a look at [openapi-typescript-codegen](https://github.com/ferdikoomen/openapi-typescript-codegen) which follows a similar philosophy but creates many individual files instead of one single self-contained file.

If your frontend uses React, take a look at [react-api-query](https://www.npmjs.com/package/react-api-query) which makes it easy to use an oazapfts client with React hooks in a convenient and type-safe way.

## About the name

The name comes from a combination of syllables **oa** (OpenAPI) and **ts** (TypeScript) and is [pronounced 🗣](https://www.youtube.com/watch?v=chvb-K95rBE) like the Bavarian _O'zapt'is!_ (it's tapped), the famous words that mark the beginning of the Oktoberfest.

# License

MIT
 readmeEtag: '"beb2a90720901a9e20a1056dbf30050bdfdeb8c6"' readmeLastModified: Mon, 26 Jan 2026 15:37:30 GMT repositoryId: 182983428 description: Generate TypeScript clients to tap into OpenAPI servers created: '2019-04-23T09:40:41Z' updated: '2026-02-06T01:43:39Z' language: TypeScript archived: false stars: 566 watchers: 5 forks: 90 owner: oazapfts logo: https://avatars.githubusercontent.com/u/119607625?v=4 repoEtag: '"a29643f350e8647d284b1eaea67f8d25d3273e8c2b713aa394b46c0ffd270991"' repoLastModified: Fri, 06 Feb 2026 01:43:39 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/cellular/oazapfts - source: openapi3 tags repository: https://github.com/muonsoft/openapi-mock v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIE1vY2sgU2VydmVyCgohW0NJXShodHRwczovL2dpdGh1Yi5jb20vbXVvbnNvZnQvb3BlbmFwaS1tb2NrL3dvcmtmbG93cy9DSS9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcikKIVtHaXRIdWIgcmVsZWFzZSAobGF0ZXN0IGJ5IGRhdGUpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3JlbGVhc2UvbXVvbnNvZnQvb3BlbmFwaS1tb2NrKQpbIVtTY3J1dGluaXplciBDb2RlIFF1YWxpdHldKGh0dHBzOi8vc2NydXRpbml6ZXItY2kuY29tL2cvbXVvbnNvZnQvb3BlbmFwaS1tb2NrL2JhZGdlcy9xdWFsaXR5LXNjb3JlLnBuZz9iPW1hc3RlcildKGh0dHBzOi8vc2NydXRpbml6ZXItY2kuY29tL2cvbXVvbnNvZnQvb3BlbmFwaS1tb2NrLz9icmFuY2g9bWFzdGVyKQpbIVtNYWludGFpbmFiaWxpdHldKGh0dHBzOi8vYXBpLmNvZGVjbGltYXRlLmNvbS92MS9iYWRnZXMvMTU4ZGViMzQzNGE4NDkyNGRhZGUvbWFpbnRhaW5hYmlsaXR5KV0oaHR0cHM6Ly9jb2RlY2xpbWF0ZS5jb20vZ2l0aHViL211b25zb2Z0L29wZW5hcGktbW9jay9tYWludGFpbmFiaWxpdHkpClshW1Rlc3QgQ292ZXJhZ2VdKGh0dHBzOi8vYXBpLmNvZGVjbGltYXRlLmNvbS92MS9iYWRnZXMvMTU4ZGViMzQzNGE4NDkyNGRhZGUvdGVzdF9jb3ZlcmFnZSldKGh0dHBzOi8vY29kZWNsaW1hdGUuY29tL2dpdGh1Yi9tdW9uc29mdC9vcGVuYXBpLW1vY2svdGVzdF9jb3ZlcmFnZSkKCk9wZW5BUEkgQVBJIG1vY2sgc2VydmVyIHdpdGggcmFuZG9tIGRhdGEgZ2VuZXJhdGlvbiBieSBzcGVjaWZpZWQgc2NoZW1hcy4KCiogT3BlbkFQSSAzLnggc3VwcG9ydC4KKiBMb2FkIHNwZWNpZmljYXRpb24gZnJvbSBhIGxvY2FsIGZpbGUgb3IgVVJMLgoqIEpTT04gYW5kIFlBTUwgZm9ybWF0IHN1cHBvcnRlZC4KKiBHZW5lcmF0ZXMgZmFrZSByZXNwb25zZSBkYXRhIGJ5IHByb3ZpZGVkIHNjaGVtYXMgb3IgYnkgZXhhbXBsZXMuCiogQ29udGVudCBuZWdvdGlhdGlvbiBieSBBY2NlcHQgaGVhZGVyLgoqIENhbiBiZSB1c2VkIGFzIHN0YW5kYWxvbmUgYXBwbGljYXRpb24gKExpbnV4IGFuZCBXaW5kb3dzKSBvciBjYW4gYmUgcnVuIHZpYSBEb2NrZXIgY29udGFpbmVyLgoKIyMgU3VwcG9ydGVkIGZlYXR1cmVzCgp8IEZlYXR1cmUgfCBTdXBwb3J0IHN0YXR1cyB8CnwgLS0tIHwgLS0tIHwKfCBnZW5lcmF0aW5nIHhtbCByZXNwb25zZSB8IGJhc2ljIHN1cHBvcnQgKFt3aXRob3V0IHhtbCB0YWdzXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2RhdGEtbW9kZWxzL3JlcHJlc2VudGluZy14bWwvKSkgfAp8IGdlbmVyYXRpbmcganNvbiByZXNwb25zZSB8IHN1cHBvcnRlZCB8CnwgZ2VuZXJhdGlvbiBvZiBbYmFzaWMgdHlwZXNdKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vZGF0YS1tb2RlbHMvZGF0YS10eXBlcy8pIHwgc3VwcG9ydGVkIHwKfCBnZW5lcmF0aW9uIG9mIFtlbnVtc10oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9kYXRhLW1vZGVscy9lbnVtcy8pIHwgc3VwcG9ydGVkIHwKfCBnZW5lcmF0aW9uIG9mIFthc3NvY2lhdGl2ZSBhcnJheXNdKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vZGF0YS1tb2RlbHMvZGljdGlvbmFyaWVzLykgfCBzdXBwb3J0ZWQgfAp8IGdlbmVyYXRpb24gb2YgW2NvbWJpbmVkIHR5cGVzXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2RhdGEtbW9kZWxzL29uZW9mLWFueW9mLWFsbG9mLW5vdC8pIHwgc3VwcG9ydGVkICh3aXRob3V0IHRhZyBgbm90YCBhbmQgZGlzY3JpbWluYXRvcikgfAp8IGxvY2FsIHJlZmVyZW5jZSByZXNvbHZpbmcgfCBzdXBwb3J0ZWQgfAp8IHJlbW90ZSByZWZlcmVuY2UgcmVzb2x2aW5nIHwgbm90IHN1cHBvcnRlZCB8CnwgVVJMIHJlZmVyZW5jZSByZXNvbHZpbmcgfCBub3Qgc3VwcG9ydGVkIHwKfCB2YWxpZGF0aW5nIHJlcXVlc3QgZGF0YSB8IG5vdCBzdXBwb3J0ZWQgfAp8IGZvcmNlIHVzaW5nIGN1c3RvbSByZXNwb25zZSBzY2hlbWEgfCBub3Qgc3VwcG9ydGVkIChzY2hlbWEgZGV0ZWN0ZWQgYXV0b21hdGljYWxseSkgfAoKIyMgUXVpY2sgc3RhcnQKCkRvd25sb2FkIGxhdGVzdCBiaW5hcnkgYW5kIHJ1biBhIHNlcnZlci4KCmBgYGJhc2gKIyBydW5zIGEgbG9jYWwgc2VydmVyIG9uIHBvcnQgODA4MAouL29wZW5hcGktbW9jayBzZXJ2ZSAtLXNwZWNpZmljYXRpb24tdXJsIGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL21hc3Rlci9leGFtcGxlcy92My4wL3BldHN0b3JlLnlhbWwKCiMgdG8gdGVzdCB0aGF0IHRoZSBzZXJ2ZXIgc3VjY2Vzc2Z1bGx5IHJhbgpjdXJsICdodHRwOi8vbG9jYWxob3N0OjgwODAvdjEvcGV0cycKYGBgCgpBbHRlcm5hdGl2ZWx5LCB5b3UgY2FuIHVzZSBbRG9ja2VyXShodHRwczovL3d3dy5kb2NrZXIuY29tLykgaW1hZ2UuCgpgYGBiYXNoCiMgZG93bmxvYWRzIGFuIGltYWdlCmRvY2tlciBwdWxsIG11b25zb2Z0L29wZW5hcGktbW9jawoKIyBydW5zIGEgZG9ja2VyIGNvbnRhaW5lciB3aXRoIGV4cG9ydGVkIHBvcnQgODA4MApkb2NrZXIgcnVuIC1wIDgwODA6ODA4MCAtZSAiT1BFTkFQSV9NT0NLX1NQRUNJRklDQVRJT05fVVJMPWh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL21hc3Rlci9leGFtcGxlcy92My4wL3BldHN0b3JlLnlhbWwiIC0tcm0gbXVvbnNvZnQvb3BlbmFwaS1tb2NrCgojIHRvIHRlc3QgdGhhdCB0aGUgc2VydmVyIHN1Y2Nlc3NmdWxseSByYW4KY3VybCAnaHR0cDovL2xvY2FsaG9zdDo4MDgwL3YxL3BldHMnCmBgYAoKQWxzbywgeW91IGNhbiB1c2UgW0RvY2tlciBDb21wb3NlXShodHRwczovL2RvY3MuZG9ja2VyLmNvbS9jb21wb3NlLykuIEV4YW1wbGUgb2YgYGRvY2tlci1jb21wb3NlLnltbGAKCmBgYHlhbWwKdmVyc2lvbjogJzMuMCcKCnNlcnZpY2VzOgogIG9wZW5hcGlfbW9jazoKICAgIGNvbnRhaW5lcl9uYW1lOiBvcGVuYXBpX21vY2sKICAgIGltYWdlOiBtdW9uc29mdC9vcGVuYXBpLW1vY2sKICAgIGVudmlyb25tZW50OgogICAgICBPUEVOQVBJX01PQ0tfU1BFQ0lGSUNBVElPTl9VUkw6ICdodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYXN0ZXIvZXhhbXBsZXMvdjMuMC9wZXRzdG9yZS55YW1sJwogICAgcG9ydHM6CiAgICAgIC0gIjgwODA6ODA4MCIKYGBgCgpUbyBzdGFydCB1cCBhIGNvbnRhaW5lciBydW4gY29tbWFuZC4KCmBgYGJhc2gKZG9ja2VyLWNvbXBvc2UgdXAgLWQKYGBgCgpJZiB5b3Ugd2FudCB0byByZWZlcmVuY2UgYSBsb2NhbCBmaWxlIGluIGRvY2tlciBjb21wb3NlOgoKKiB5b3UgbXVzdCBmaXJzdCBtb3VudCB0aGUgaG9zdCBkaXIgaW50byBjb250YWluZXIgLSBgLi9vcGVuYXBpOi9ldGMvb3BlbmFwaWAKKiBvbmx5IHRoZW4gY2FuIHlvdSByZWZlcmVuY2UgaXQKCmBgYHlhbWwKdmVyc2lvbjogJzMuMCcKCnNlcnZpY2VzOgogIG9wZW5hcGlfbW9jazoKICAgIGNvbnRhaW5lcl9uYW1lOiBvcGVuYXBpX21vY2sKICAgIGltYWdlOiBtdW9uc29mdC9vcGVuYXBpLW1vY2sKICAgIHZvbHVtZXM6CiAgICAtIC4vb3BlbmFwaTovZXRjL29wZW5hcGkKICAgIGVudmlyb25tZW50OgogICAgICBPUEVOQVBJX01PQ0tfU1BFQ0lGSUNBVElPTl9VUkw6ICcvZXRjL29wZW5hcGkvcGV0c3RvcmUueWFtbCcKICAgIHBvcnRzOgogICAgICAtICI4MDgwOjgwODAiCmBgYAoKIyMgVXNhZ2UgZ3VpZGUKCiogW0NvbnNvbGUgY29tbWFuZHNdKGRvY3MvdXNhZ2VfZ3VpZGUubWQjY29uc29sZS1jb21tYW5kcykKKiBbU2V0dGluZyB1cCBhIGNvbmZpZ3VyYXRpb25dKGRvY3MvdXNhZ2VfZ3VpZGUubWQjc2V0dGluZy11cC1hLWNvbmZpZ3VyYXRpb24pCiogW0NvbmZpZ3VyYXRpb24gZmlsZSBleGFtcGxlXShkb2NzL3VzYWdlX2d1aWRlLm1kI2NvbmZpZ3VyYXRpb24tZmlsZS1leGFtcGxlKQoqIFtDb25maWd1cmF0aW9uIG9wdGlvbnNdKGRvY3MvdXNhZ2VfZ3VpZGUubWQjY29uZmlndXJhdGlvbi1vcHRpb25zKQoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZSAtIHNlZSB0aGUgTElDRU5TRSBmaWxlIGZvciBkZXRhaWxzLgo= readmeEtag: '"324cd90728c76f314276a102e1cf83b89da5e3db"' readmeLastModified: Sat, 11 Mar 2023 09:19:53 GMT repositoryId: 145602302 description: OpenAPI mock server with random data generation created: '2018-08-21T18:16:27Z' updated: '2026-01-27T07:58:01Z' language: Go archived: false stars: 524 watchers: 8 forks: 59 owner: muonsoft logo: https://avatars.githubusercontent.com/u/32521718?v=4 license: MIT repoEtag: '"0009be0d6e8bd7c8cf34c6a44d1ad7900359411376551c1b2587a93233223444"' repoLastModified: Tue, 27 Jan 2026 07:58:01 GMT foundInMaster: true category: - Description Validators - Parsers id: ca63af26f6da094f0ce57d398e968526 - source: openapi3 tags repository: https://github.com/swagger-autogen/swagger-autogen v3: true id: 2fbe0ef8e6fd0bf2f43ab4e7c7d80970 repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyLWF1dG9nZW4KCiMjIE92ZXJ2aWV3CgpUaGlzIG1vZHVsZSBwZXJmb3JtcyBhdXRvbWF0aWMgY29uc3RydWN0aW9uIG9mIFN3YWdnZXIgZG9jdW1lbnRhdGlvbi4gSXQgY2FuIGlkZW50aWZ5IHRoZSBlbmRwb2ludHMgYW5kIGF1dG9tYXRpY2FsbHkgY2FwdHVyZSBtZXRob2RzIHN1Y2ggYXMgZ2V0LCBwb3N0LCBwdXQsIGFuZCBzbyBvbi4gSXQgYWxzbyBpZGVudGlmaWVzIHBhdGhzLCByb3V0ZXMsIG1pZGRsZXdhcmVzLCByZXNwb25zZSBzdGF0dXMgY29kZXMsIHBhcmFtZXRlcnMgaW4gdGhlIHBhdGgsIGhlYWRlciwgcXVlcnkgYW5kIGJvZHkuIEl0IGlzIHBvc3NpYmxlIHRvIGFkZCBpbmZvcm1hdGlvbiBzdWNoIGFzIGVuZHBvaW50IGRlc2NyaXB0aW9uLCBwYXJhbWV0ZXIgZGVzY3JpcHRpb24sIHNjaGVtYXMsIHNlY3VyaXR5LCBhbW9uZyBvdGhlcnMgdXNpbmcgY29tbWVudHMgaW4gY29kZS4gQXQgdGhlIGVuZCwgaXQgZ2VuZXJhdGVzIHRoZSAqLmpzb24qIGZpbGUgY29udGFpbmluZyB0aGUgU3dhZ2dlciBmb3JtYXQgc3BlY2lmaWNhdGlvbi4KClshW05QTSBWZXJzaW9uXShodHRwOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3Yvc3dhZ2dlci1hdXRvZ2VuLnN2Zz9zdHlsZT1mbGF0KV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvc3dhZ2dlci1hdXRvZ2VuKQpbIVtOUE0gRG93bmxvYWRzXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9kbS9zd2FnZ2VyLWF1dG9nZW4uc3ZnP3N0eWxlPWZsYXQpXShodHRwczovL25wbWNoYXJ0cy5jb20vY29tcGFyZS9zd2FnZ2VyLWF1dG9nZW4/bWluaW1hbD10cnVlKQoKIyMgQ29udGVudHMKCi0gW0luc3RhbGxhdGlvbl0oI2luc3RhbGxhdGlvbikKLSBbRG9jdW1lbnRhdGlvbl0oI2RvY3VtZW50YXRpb24pCi0gW0NoYW5nZWxvZ10oI2NoYW5nZWxvZykKLSBbTGljZW5zZV0oI2xpY2Vuc2UpCgojIyBJbnN0YWxsYXRpb24KClRoaXMgaXMgYSBbTm9kZS5qc10oaHR0cHM6Ly9ub2RlanMub3JnL2VuLykgbW9kdWxlIGF2YWlsYWJsZSB0aHJvdWdoIHRoZSBbbnBtXShodHRwczovL3d3dy5ucG1qcy5jb20vKS4KCmBgYGJhc2gKJCBucG0gaW5zdGFsbCAtLXNhdmUtZGV2IHN3YWdnZXItYXV0b2dlbgpgYGAKCklmIHlvdSdyZSB1c2luZyBDb21tb25KUzoKCmBgYGpzCmNvbnN0IHN3YWdnZXJBdXRvZ2VuID0gcmVxdWlyZSgnc3dhZ2dlci1hdXRvZ2VuJykoKTsKYGBgCgpPciBpZiB5b3UncmUgdXNpbmcgRVMgbW9kdWxlczoKCmBgYGpzCmltcG9ydCBzd2FnZ2VyQXV0b2dlbiBmcm9tICdzd2FnZ2VyLWF1dG9nZW4nOwoKYGBgCgojIyBEb2N1bWVudGF0aW9uCgpQbGVhc2UgcmVmZXIgdG8gdGhlIGRvY3VtZW50YXRpb24gd2Vic2l0ZSBvbiBodHRwczovL3N3YWdnZXItYXV0b2dlbi5naXRodWIuaW8uCgojIyBDaGFuZ2Vsb2cKCkNoZWNrIHRoZSBbR2l0SHViIFJlbGVhc2VzIHBhZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWF1dG9nZW4vc3dhZ2dlci1hdXRvZ2VuL3JlbGVhc2VzKS4KCiMjIElzc3VlcwoKRHVlIHRvIHRoZSBudW1iZXIgb2Ygb3BlbiBpc3N1ZXMgYW5kIHRoZSBsYWNrIG9mIHRpbWUgdG8gcmVzcG9uZCB3aXRoIHF1YWxpdHksIEkndmUgZGlzYWJsZWQgdGhlIG9wZW5pbmcgb2YgaXNzdWVzIGZvciBub3cuIEknbSBzdGlsbCBrZWVwaW5nIG15IGV5ZXMgb24gdGhlIHByb2plY3QgYW5kIEkgaW50ZW5kIHRvIHByaW9yaXRpemUgdGhlIGlzc3VlcyBhbHJlYWR5IG9wZW4uIE9uY2UgSSBoYXZlIGNsb3NlZCBhIHNhdGlzZmFjdG9yeSBudW1iZXIgb2YgaXNzdWVzLCBJIHdpbGwgb3BlbiBpdCBhZ2Fpbi4gVGhhbmsgeW91IGZvciB1bmRlcnN0YW5kaW5nLgoKIyMgTGljZW5zZQoKW01JVF0oTElDRU5TRSkgTGljZW5zZQo= readmeEtag: '"2a11c5b385e9cefa3244c662352b041359767f04"' readmeLastModified: Thu, 28 Dec 2023 02:56:36 GMT repositoryId: 255205097 description: >- This module performs automatic construction of Swagger documentation. It can identify the endpoints and automatically capture methods such as get, post, put, and so on. It also identifies paths, routes, middlewares, response status codes and parameters. At the end, it generates the .json file containing the Swagger format specification. created: '2020-04-13T01:37:33Z' updated: '2026-01-29T00:31:35Z' language: JavaScript archived: false stars: 506 watchers: 4 forks: 70 owner: swagger-autogen logo: https://avatars.githubusercontent.com/u/137681762?v=4 license: MIT repoEtag: '"5e136fd706036c3fcad8b714b511460cb5894862f826f4c288b9a64b54979793"' repoLastModified: Thu, 29 Jan 2026 00:31:35 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/davibaltar/swagger-autogen - source: openapi3 tags repository: https://github.com/rithvikvibhu/ghlocalapi v3: true repositoryMetadata: base64Readme: >- IyBbR29vZ2xlIEhvbWUgTG9jYWwgQVBJIERvY3VtZW50YXRpb25dKGh0dHBzOi8vcml0aHZpa3ZpYmh1LmdpdGh1Yi5pby9HSExvY2FsQXBpLykKCioqVXBkYXRlZCBvbjogMjAyMC0wNi0yOCoqCgpBdmFpbGFibGUgaGVyZTogaHR0cHM6Ly9yaXRodmlrdmliaHUuZ2l0aHViLmlvL0dITG9jYWxBcGkvCgpbIVtHaXRIdWIgc3RhcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3N0YXJzL3JpdGh2aWt2aWJodS9HSExvY2FsQXBpKV0oaHR0cHM6Ly9naXRodWIuY29tL3JpdGh2aWt2aWJodS9HSExvY2FsQXBpL3N0YXJnYXplcnMpIFshW0dpdEh1YiBsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL3JpdGh2aWt2aWJodS9HSExvY2FsQXBpKV0oaHR0cHM6Ly9naXRodWIuY29tL3JpdGh2aWt2aWJodS9HSExvY2FsQXBpL2Jsb2IvbWFzdGVyL0xJQ0VOU0UubWQpIFshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL3JpdGh2aWt2aWJodS9HSExvY2FsQXBpLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL3JpdGh2aWt2aWJodS9HSExvY2FsQXBpKQoKRG9jdW1lbnRhdGlvbiBmb3IgR29vZ2xlIEhvbWUncyBsb2NhbCBBUEksIHNwZWNzIGluIFtPcGVuQVBJIDMuMF0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKS4gUmVuZGVyZWQgd2l0aCBbUmVkb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKS4KClRyYXZpcyBDSSBnZW5lcmF0ZXMgdGhlIHN0YXRpYyB3ZWJzaXRlIHdpdGggUmVkb2MuCgotIGBucG0gcnVuIGJ1aWxkYCBnZW5lcmF0ZXMgdGhlIHN0YXRpYyBzaXRlIGluIHRoZSAnZGlzdCcgZm9sZGVyIChNYWtlIHN1cmUgdGhlIGZvbGRlciBleGlzdHMpLgotIGBucG0gcnVuIHNlcnZlYCBzZXJ2ZXMgdGhlIHBhZ2Ugd2l0aCBhIHdlYiBzZXJ2ZXIuCgojIyBDb250cmlidXRpbmcKClRoZSBPcGVuIEFQSSBzcGVjIGlzIGV4cG9ydGVkIGZyb20gQXBpbWF0aWMuIEluc3RlYWQgb2Ygc3VibWl0dGluZyBwdWxsIHJlcXVlc3RzIHdpdGggY2hhbmdlcyB0byB0aGlzIGZpbGUsIHBsZWFzZSBjcmVhdGUgYW4gaXNzdWUgd2l0aCB0aGUgY2hhbmdlcyBpbnN0ZWFkLgoKIyMgQ3JlZGl0cwoKSHVnZSB0aGFua3MgdG8KCi0gW0BMb2hvdXNlXShodHRwczovL2dpdGh1Yi5jb20vTG9ob3VzZSkgZm9yIGZpbmRpbmcgYW5kIHBvc3RpbmcgYW4gaW5pdGlhbCBsaXN0IG9mIEFQSSBlbmRwb2ludHMgW29uIFJlZGRpdCByL2dvb2dsZWhvbWVdKGh0dHBzOi8vd3d3LnJlZGRpdC5jb20vci9nb29nbGVob21lL2NvbW1lbnRzLzdxZzZlZi91c2VfdGhlX2FwaV9vbl95b3VyX2dvb2dsZV9ob21lX3RvX3ZpZXdfdGhpbmdzLykKLSBbQExlYXRoZXJmYWNlNzVdKGh0dHBzOi8vZ2l0aHViLmNvbS9MZWF0aGVyZmFjZTc1KSwgW0BEcmFrdWxpeF0oaHR0cHM6Ly9naXRodWIuY29tL0RyYWt1bGl4KSwgW0BtYWdpY3NlXShodHRwczovL2dpdGh1Yi5jb20vbWFnaWNzZSksIFtAWG9NRVhdKGh0dHBzOi8vZ2l0aHViLmNvbS9Yb01FWCksIFtAQW5nZWxvRDIwMjJdKGh0dHBzOi8vZ2l0aHViLmNvbS9BbmdlbG9EMjAyMikgYW5kIGV2ZXJ5b25lIG9uIFt0aGlzIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vcml0aHZpa3ZpYmh1L0dITG9jYWxBcGkvaXNzdWVzLzM5KSBmb3IgaGVscGluZyBSRSB0aGUgYXV0aGVudGljYXRpb24gc3lzdGVtLgotIFtAc2ltb24td2ViZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9zaW1vbi13ZWJlcikgZm9yIHRoZSBbYGdwc29hdXRoYF0oaHR0cHM6Ly9naXRodWIuY29tL3NpbW9uLXdlYmVyL2dwc29hdXRoKSBsaWJyYXJ5Ci0gQWxsIGNvbnRyaWJ1dG9ycyB3aG8ndmUgcmVwb3J0ZWQgaW5jb25zaXN0ZW5jaWVzIGFuZCBQUidkIGZpeGVzCg== readmeEtag: '"c38b9fa3c56fd64f2ca96bcecc24613a9ee015bb"' readmeLastModified: Sun, 14 Aug 2022 19:42:42 GMT repositoryId: 117561740 description: (Unofficial) Google Home local API documentation. created: '2018-01-15T15:22:58Z' updated: '2026-01-29T13:52:00Z' language: Handlebars archived: false stars: 464 watchers: 30 forks: 38 owner: rithvikvibhu logo: https://avatars.githubusercontent.com/u/5113343?v=4 license: MIT repoEtag: '"c9da77be577ec27ea464ebceb943293977c1bd24dbc4b0b09688236afd74bfa7"' repoLastModified: Thu, 29 Jan 2026 13:52:00 GMT foundInMaster: true category: SDK id: 6f3b994d76d4f6770123b20ae9bd03a2 - source: openapi3 tags repository: https://github.com/tiangolo/full-stack-fastapi-couchbase v3: true repositoryMetadata: base64Readme: >- ## 🚨 DEPRECATION WARNING 🚨

I'm currently not actively using Couchbase nor this generator for any project.

You might still find some internal pieces of code useful for your own use cases, but I won't be able to fix bugs and add features.

If you are starting a new project from scratch, check the alternatives at the [FastAPI docs: Project Generation](https://fastapi.tiangolo.com/project-generation/).

You are still free to use this project if you want to, you might still find some internal pieces of code useful for your own use case. And if you already have a project generated with it that's fine as well (and you probably already updated it to suit your needs).

# Full Stack FastAPI Couchbase - project generator

[![Build Status](https://travis-ci.org/tiangolo/full-stack-fastapi-couchbase.svg?branch=master)](https://travis-ci.org/tiangolo/full-stack-fastapi-couchbase)

### To have in mind

Here are some extra thoughts you might want to consider if you decide to go with Couchbase and/or this project generator.

#### Pros

Couchbase has a great set of features that is not easily or commonly found in alternatives.

It's a distributed database, so, if you have a cluster of several nodes with the data replicated, you don't have a single point of failure.

It provides very high performance.

It has built-in full-text search integrated (using Bleve). It's possible to make all the records be automatically indexed in the search engine without additional code or extra components.

It can be connected/integrated with the Couchbase Sync Gateway, that can be used to synchronize a subset of documents (records) with mobile devices. And with some effort, it can be connected with PouchBD for frontend synchronization (although not querying and/or searching from the frontend, as would be possible with CouchDB).

#### Cons

The documentation is very scarce. This is sadly a big problem. Many features are not well documented, or not documented at all.

Many configurations are not documented and have to be inferred from comments in forums, reading source code, or checking the configurations for the C client and trying different query parameters in URLs, that becomes quite error prone.

Couchbase didn't seem to be designed to be integrated into CI systems as in this project (or alternatives), at least until the last time I used it. It expects all the set up to be done once via the web UI, not automated. The official Docker image can't be configured. So, this project does all the configuration by sending the HTTP requests from the backend code to the container replicating the HTTP requests done in the web UI. But those steps are not documented, there's no "official" way to configure and start it without using the web UI, so the integration with CI could be error prone.

The Couchbase Sync Gateway official Docker image is not designed to be configured much either, so, the `Dockerfile` included in this project adds a good amount of custom logic to support that, but that's not really official.

As it uses N1QL, a flavor of SQL, and there's no easy integration with Python, you have to do all the operations in N1QL strings and integrate them with your own code. Without editor support, completion, nor syntax checks for N1QL.

---

Generate a backend and frontend stack using Python, including interactive API documentation.

[![Screenshot](https://fastapi.tiangolo.com/img/index/index-03-swagger-02.png)](https://github.com/tiangolo/full-stack-fastapi-couchbase)

## Features

* Full **Docker** integration (Docker based).
* Docker Swarm Mode deployment.
* **Docker Compose** integration and optimization for local development.
* **Production ready** Python web server using Uvicorn and Gunicorn.
* Python **[FastAPI](https://github.com/tiangolo/fastapi)** backend:
    * **Fast**: Very high performance, on par with **NodeJS** and **Go** (thanks to Starlette and Pydantic).
    * **Intuitive**: Great editor support. <abbr title="also known as auto-complete, autocompletion, IntelliSense">Completion</abbr> everywhere. Less time debugging.
    * **Easy**: Designed to be easy to use and learn. Less time reading docs.
    * **Short**: Minimize code duplication. Multiple features from each parameter declaration.
    * **Robust**: Get production-ready code. With automatic interactive documentation.
    * **Standards-based**: Based on (and fully compatible with) the open standards for APIs: <a href="https://github.com/OAI/OpenAPI-Specification" target="_blank">OpenAPI</a> and <a href="http://json-schema.org/" target="_blank">JSON Schema</a>.
    * [**Many other features**](https://github.com/tiangolo/fastapi) including automatic validation, serialization, interactive documentation, authentication with OAuth2 JWT tokens, etc.
* **Secure password** hashing by default.
* **JWT token** authentication.
* **CORS** (Cross Origin Resource Sharing).
* **Celery** worker that can import and use code from the rest of the backend selectively (you don't have to install the complete app in each worker).
* **NoSQL Couchbase** database that supports direct synchronization via Couchbase Sync Gateway for offline-first applications.
* **Full Text Search** integrated, using Couchbase.
* REST backend tests based on Pytest, integrated with Docker, so you can test the full API interaction, independent on the database. As it runs in Docker, it can build a new data store from scratch each time (so you can use ElasticSearch, MongoDB, or whatever you want, and just test that the API works).
* Easy Python integration with **Jupyter** Kernels for remote or in-Docker development with extensions like Atom Hydrogen or Visual Studio Code Jupyter.
* **Email notifications** for account creation and password recovery, compatible with:
    * Mailgun
    * SparkPost
    * SendGrid
    * ...any other provider that can generate standard SMTP credentials.
* **Vue** frontend:
    * Generated with Vue CLI.
    * **JWT Authentication** handling.
    * Login view.
    * After login, main dashboard view.
    * Main dashboard with user creation and edition.
    * Self user edition.
    * **Vuex**.
    * **Vue-router**.
    * **Vuetify** for beautiful material design components.
    * **TypeScript**.
    * Docker server based on **Nginx** (configured to play nicely with Vue-router).
    * Docker multi-stage building, so you don't need to save or commit compiled code.
    * Frontend tests ran at build time (can be disabled too).
    * Made as modular as possible, so it works out of the box, but you can re-generate with Vue CLI or create it as you need, and re-use what you want.
* **Flower** for Celery jobs monitoring.
* Load balancing between frontend and backend with **Traefik**, so you can have both under the same domain, separated by path, but served by different containers.
* Traefik integration, including Let's Encrypt **HTTPS** certificates automatic generation.
* GitLab **CI** (continuous integration), including frontend and backend testing.

## How to use it

Go to the directory where you want to create your project and run:

```bash
pip install cookiecutter
cookiecutter https://github.com/tiangolo/full-stack-fastapi-couchbase
```

### Generate passwords

You will be asked to provide passwords and secret keys for several components. Open another terminal and run:

```bash
openssl rand -hex 32
# Outputs something like: 99d3b1f01aa639e4a76f4fc281fc834747a543720ba4c8a8648ba755aef9be7f
```

Copy the contents and use that as password / secret key. And run that again to generate another secure key.


### Input variables

The generator (cookiecutter) will ask you for some data, you might want to have at hand before generating the project.

The input variables, with their default values (some auto generated) are:

* `project_name`: The name of the project
* `project_slug`: The development friendly name of the project. By default, based on the project name
* `domain_main`: The domain in where to deploy the project for production (from the branch `production`), used by the load balancer, backend, etc. By default, based on the project slug.
* `domain_staging`: The domain in where to deploy while staging (before production) (from the branch `master`). By default, based on the main domain.

* `docker_swarm_stack_name_main`: The name of the stack while deploying to Docker in Swarm mode for production. By default, based on the domain.
* `docker_swarm_stack_name_staging`: The name of the stack while deploying to Docker in Swarm mode for staging. By default, based on the domain.

* `secret_key`: Backend server secret key. Use the method above to generate it.
* `first_superuser`: The first superuser generated, with it you will be able to create more users, etc. By default, based on the domain.
* `first_superuser_password`: First superuser password. Use the method above to generate it.
* `backend_cors_origins`: Origins (domains, more or less) that are enabled for CORS (Cross Origin Resource Sharing). This allows a frontend in one domain (e.g. `https://dashboard.example.com`) to communicate with this backend, that could be living in another domain (e.g. `https://api.example.com`). It can also be used to allow your local frontend (with a custom `hosts` domain mapping, as described in the project's `README.md`) that could be living in `http://dev.example.com:8080` to communicate with the backend at `https://stag.example.com`. Notice the `http` vs `https` and the `dev.` prefix for local development vs the "staging" `stag.` prefix. By default, it includes origins for production, staging and development, with ports commonly used during local development by several popular frontend frameworks (Vue with `:8080`, React, Angular).
* `smtp_port`: Port to use to send emails via SMTP. By default `587`.
* `smtp_host`: Host to use to send emails, it would be given by your email provider, like Mailgun, Sparkpost, etc.
* `smtp_user`: The user to use in the SMTP connection. The value will be given by your email provider.
* `smtp_password`: The password to be used in the SMTP connection. The value will be given by the email provider.
* `smtp_emails_from_email`: The email account to use as the sender in the notification emails, it would be something like `info@your-custom-domain.com`.
 
* `couchbase_user`: Couchbase main user to be used by the application (code). By default `admin`.
* `couchbase_password`: Password of the main user, for the backend code. Generate it with the method above.
* `couchbase_sync_gateway_cors`: List of CORS origins that the Sync Gateway should allow to talk to directly. Similar to `backend_cors_origins`.
* `couchbase_sync_gateway_user`: User to be created for the Couchbase Sync Gateway. This is what allows synchronization using the CouchDB protocol, with Couchbase Lite in mobile apps and PouchDB in the web and hybrid mobile apps.
* `couchbase_sync_gateway_password`: Couchbase Sync Gateway password. Generate it with the method above.
 
* `traefik_constraint_tag`: The tag to be used by the internal Traefik load balancer (for example, to divide requests between backend and frontend) for production. Used to separate this stack from any other stack you might have. This should identify each stack in each environment (production, staging, etc).
* `traefik_constraint_tag_staging`: The Traefik tag to be used while on staging. 

* `traefik_public_network`: This assumes you have another separate publicly facing Traefik at the server / cluster level. This is the network that main Traefik lives in.
* `traefik_public_constraint_tag`: The tag that should be used by stack services that should communicate with the public.

* `flower_auth`: Basic HTTP authentication for flower, in the form`user:password`. By default: "`root:changethis`".

* `sentry_dsn`: Key URL (DSN) of Sentry, for live error reporting. If you are not using it yet, you should, is open source. E.g.: `https://1234abcd:5678ef@sentry.example.com/30`.

* `docker_image_prefix`: Prefix to use for Docker image names. If you are using GitLab Docker registry it would be based on your code repository. E.g.: `git.example.com/development-team/my-awesome-project/`.
* `docker_image_backend`: Docker image name for the backend. By default, it will be based on your Docker image prefix, e.g.: `git.example.com/development-team/my-awesome-project/backend`. And depending on your environment, a different tag will be appended ( `prod`, `stag`, `branch` ). So, the final image names used will be like: `git.example.com/development-team/my-awesome-project/backend:prod`.
* `docker_image_celeryworker`: Docker image for the celery worker. By default, based on your Docker image prefix.
* `docker_image_frontend`: Docker image for the frontend. By default, based on your Docker image prefix.
* `docker_image_sync_gateway`: Docker image for the Sync Gateway. By default, based on your Docker image prefix.

## How to deploy

This stack can be adjusted and used with several deployment options that are compatible with Docker Compose, but it is designed to be used in a cluster controlled with pure Docker in Swarm Mode with a Traefik main load balancer proxy handling automatic HTTPS certificates, using the ideas from <a href="https://dockerswarm.rocks" target="_blank">DockerSwarm.rocks</a>.

Please refer to <a href="https://dockerswarm.rocks" target="_blank">DockerSwarm.rocks</a> to see how to deploy such a cluster in 20 minutes.

## More details

After using this generator, your new project (the directory created) will contain an extensive `README.md` with instructions for development, deployment, etc. You can pre-read [the project `README.md` template here too](./{{cookiecutter.project_slug}}/README.md).

## Sibling project generators

* Based on PostgreSQL: [https://github.com/tiangolo/full-stack-fastapi-postgresql](https://github.com/tiangolo/full-stack-fastapi-postgresql).

## Release Notes

### Latest Changes

* Fix Windows line endings for shell scripts after project generation with Cookiecutter hooks. PR [#28](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/28).
* Remove `package-lock.json` to let everyone lock their own versions (depending on OS, etc).
* Simplify Traefik labels for services. PR [#27](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/27).
* Fix Flower Docker configuration. PR [#24](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/24).
* Update testing scripts and typo.
* Add normal user Pytest fixture. PR [#23](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/23).
* Update Dockerfiles to use Couchbase from Debian image. PR [#20](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/20) by [@Gjacquenot](https://github.com/Gjacquenot).
* Use new Pydantic types. PR [#21](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/21) by [@tehtbl](https://github.com/tehtbl).

### 0.4.0

* Fix security on resetting a password. Receive `password` and `token` as body, not query. PR [#16](https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/16).

* Forward arguments from script to `pytest` inside container.

* Update Jupyter Lab installation and util script/environment variable for local development.

### 0.3.0

* PR <a href="https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/15" target="_blank">#15</a>:
    * Update CRUD utils to use types better.
    * Simplify Pydantic model names, from `UserInCreate` to `UserCreate`, etc.
    * Upgrade packages.
    * Add new generic "Items" models, crud utils, endpoints, and tests. To facilitate re-using them to create new functionality. As they are simple and generic (not like Users), it's easier to copy-paste and adapt them to each use case.
    * Update endpoints/*path operations* to simplify code and use new utilities, prefix and tags in `include_router`.
    * Update testing utils.
    * Update linting rules, relax vulture to reduce false positives.
    * Add full text search for items.
    * Update project README.md with tips about how to start with backend.

### 0.2.1

* Fix frontend hijacking /docs in development. Using latest https://github.com/tiangolo/node-frontend with custom Nginx configs in frontend. <a href="https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/14" target="_blank">PR #14</a>.

* Update generated README. Minor typos.

* Update Couchbase installation, to include Couchbase command line tools.

* Set `/start-reload.sh` as a command override for development by default.

### 0.2.0

**<a href="https://github.com/tiangolo/full-stack-fastapi-couchbase/pull/7" target="_blank">PR #7</a>**:

* Simplify and update backend `Dockerfile`s.
* Refactor and simplify backend code, improve naming, imports, modules and "namespaces".
* Improve and simplify Vuex integration with TypeScript accessors.
* Standardize frontend components layout, buttons order, etc.
* Add local development scripts (to develop this project generator itself).
* Add logs to startup modules to detect errors early.
* Improve FastAPI dependency utilities, to simplify and reduce code (to require a superuser).
* Fix/update logic to update users.

## License

This project is licensed under the terms of the MIT license.
 readmeEtag: '"abba64326b23cc45521d6a9eddbe4100cd04b917"' readmeLastModified: Mon, 14 Nov 2022 15:41:30 GMT repositoryId: 161297482 description: >- Full stack, modern web application generator. Using FastAPI, Couchbase as database, Docker, automatic HTTPS and more. created: '2018-12-11T07:53:25Z' updated: '2026-02-02T17:22:16Z' language: Python archived: true stars: 455 watchers: 6 forks: 83 owner: tiangolo logo: https://avatars.githubusercontent.com/u/1326112?v=4 license: MIT repoEtag: '"a874edf4eca2f677fe359fe15e768a2a3ae7ab542cd074f50f95a88144855ff5"' repoLastModified: Mon, 02 Feb 2026 17:22:16 GMT foundInMaster: true category: Server Implementations id: 5b8908148637a85dc44c083694fd548a - source: openapi3 tags repository: https://github.com/leosussan/fastapi-gino-arq-uvicorn v3: true repositoryMetadata: base64Readme: >- IyBmYXN0YXBpLWdpbm8tYXJxLXV2aWNvcm4KSGlnaC1wZXJmb3JtYW5jZSBBc3luYyBSRVNUIEFQSSwgaW4gUHl0aG9uLiBGYXN0QVBJICsgR0lOTyArIEFycSArIFV2aWNvcm4gKHBvd2VyZWQgYnkgUmVkaXMgJiBQb3N0Z3JlU1FMKS4KCiMjIENvbnRlbnRzCi0gW2Zhc3RhcGktZ2luby1hcnEtdXZpY29ybl0oI2Zhc3RhcGktZ2luby1hcnEtdXZpY29ybikKICAtIFtDb250ZW50c10oI2NvbnRlbnRzKQogIC0gW0dldCBTdGFydGVkXSgjZ2V0LXN0YXJ0ZWQpCiAgICAtIFtTZXR1cF0oI3NldHVwKQogICAgLSBbUnVuXSgjcnVuKQogICAgICAtIFtSdW4gTG9jYWxseV0oI3J1bi1sb2NhbGx5KQogICAgICAtIFtSdW4gTG9jYWxseSB3aXRoIERvY2tlci1Db21wb3NlLl0oI3J1bi1sb2NhbGx5LXdpdGgtZG9ja2VyLWNvbXBvc2UpCiAgICAtIFtCdWlsZCBZb3VyIEFwcGxpY2F0aW9uXSgjYnVpbGQteW91ci1hcHBsaWNhdGlvbikKICAtIFtGZWF0dXJlc10oI2ZlYXR1cmVzKQogICAgLSBbQ29yZSBEZXBlbmRlbmNpZXNdKCNjb3JlLWRlcGVuZGVuY2llcykKICAgICAgLSBbQWRkaXRpb25hbCBEZXBlbmRlbmNpZXNdKCNhZGRpdGlvbmFsLWRlcGVuZGVuY2llcykKCiMjIEdldCBTdGFydGVkCiMjIyBTZXR1cAoxLiBDbG9uZSB0aGlzIFJlcG9zaXRvcnkuIGBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2xlb3N1c3Nhbi9mYXN0YXBpLWdpbm8tYXJxLXV2aWNvcm4uZ2l0YAoyLiBJbnN0YWxsIGBQeXRob24gMy44YCBhbmQgYHBvZXRyeWAuCiAgICAqIFJlY29tbWVuZGVkIE1ldGhvZDogYGFzZGZgIC0gYSB1bml2ZXJzYWwgdmVyc2lvbiBtYW5hZ2VyICh0aGluayBgbnZtYCBvciBgcHllbnZgKQogICAgICAgICogRm9sbG93IFt0aGVzZSBpbnN0cnVjdGlvbnNdKGh0dHBzOi8vYXNkZi12bS5jb20vIy9jb3JlLW1hbmFnZS1hc2RmLXZtP2lkPWluc3RhbGwtYXNkZi12bSkgdG8gaW5zdGFsbCBgYXNkZmAuCiAgICAgICAgKiBSdW4gdGhlIGZvbGxvd2luZyBjb21tYW5kcyBmcm9tIHRoZSBwcm9qZWN0IHJvb3Q6CiAgICAgICAgICAgICogYGFzZGYgcGx1Z2luIGFkZCBweXRob25gCiAgICAgICAgICAgICogYGFzZGYgcGx1Z2luIGFkZCBwb2V0cnlgCiAgICAgICAgICAgICogYGFzZGYgaW5zdGFsbGAgLS0gd2lsbCBkb3dubG9hZCAmIGNvbmZpZ3VyZSB0aGlzIHByb2plY3QncyBgUHl0aG9uYCArIGBwb2V0cnlgIHNldHVwCiAgICAgICAgKiBffk5PVEVfOiB5b3VyIG1hY2hpbmUgbXVzdCBoYXZlIGEgc3lzdGVtIHZlcnNpb24gb2YgUHl0aG9uIGluc3RhbGxlZC4gSWYgeW91IGRvbid0LCBydW4gdGhlIGZvbGxvd2luZzogYGFzZGYgaW5zdGFsbCBweXRob24gMy44LjIgJiYgYXNkZiBnbG9iYWwgcHl0aG9uIDMuOC4yYAogICAgKiBJZiB5b3UgaGF2ZSBgUHl0aG9uIDMuOGAgYW5kIGBwb2V0cnlgIGluc3RhbGxlZCBhbHJlYWR5LCBwbGVhc2UgZmVlbCBmcmVlIHRvIHNraXAuCjMuIEluc3RhbGwgZGVwZW5kZW5jaWVzIChgcG9ldHJ5IGluc3RhbGxgKS4KNC4gQWN0aXZhdGUgcHJlLWNvbW1pdCBob29rcyAoaW4gYHBvZXRyeSBzaGVsbGAsIHJ1biBgcHJlLWNvbW1pdCBpbnN0YWxsYCkuCjUuIE1ha2UgYSBjb3B5IG9mIGAuZGlzdC5lbnZgLCByZW5hbWUgdG8gYC5lbnZgLiBGaWxsIGluIFBvc3RncmVTUUwsIFJlZGlzLCBTZW50cnkgKG9wdGlvbmFsKSB2YXJpYWJsZXMuCjYuIEdlbmVyYXRlIERCIE1pZ3JhdGlvbnM6IGluIGBwb2V0cnkgc2hlbGxgLCBydW4gYGFsZW1iaWMgcmV2aXNpb24gLS1hdXRvZ2VuZXJhdGVgLiAKICAgICogQXBwbHkgbWlncmF0aW9ucyBtYW51YWxseSB3aXRoIGBhbGVtYmljIHVwZ3JhZGUgaGVhZGAuCiAgICAqIElmIHVzaW5nIHRoZSBEb2NrZXJmaWxlLCBtaWdyYXRpb25zIGFyZSBhcHBsaWVkIGF0IHN0YXJ0dXAuCgojIyMgUnVuCgojIyMjIFJ1biBMb2NhbGx5Cl9OT1RFOiBZb3UgbXVzdCBoYXZlIFBvc3RncmVTUUwgJiBSZWRpcyBydW5uaW5nIGxvY2FsbHkuXwoKMS4gTWFrZSBzdXJlIFBvc3RncmVTUUwgJiBSZWRpcyBhcmUgcnVubmluZyBsb2NhbGx5LgoyLiBSdW46CiAgICAtIEZhc3RBUEkgQXBwbGljYXRpb246CiAgICAgICAgKiBfRm9yIEFjdGl2ZSBEZXZlbG9wbWVudCAody8gYXV0by1yZWxvYWQpOl8gUnVuIGxvY2FsbHkgd2l0aCBgcG9ldHJ5IHJ1biB0YXNrIGFwcGAKICAgICAgICAqIF9Gb3IgRGVidWdnaW5nIChjb21wYXRpYmxlIHcvIGRlYnVnZ2Vycywgbm8gYXV0by1yZWxvYWQpOl8gQ29uZmlndXJlIGRlYnVnZ2VyIHRvIHJ1biBgcHl0aG9uIGFwcC9tYWluLnB5YC4KICAgIC0gQmFja2dyb3VuZCBUYXNrIFdvcmtlcjoKICAgICAgICAqIF9Gb3IgQWN0aXZlIERldmVsb3BtZW50Ol8gUnVuICBgcG9ldHJ5IHJ1biB0YXNrIHdvcmtlcmAKCiMjIyMgUnVuIExvY2FsbHkgd2l0aCBEb2NrZXItQ29tcG9zZS4KMS4gTWFrZSBzdXJlIGBEb2NrZXJgIGlzIHJ1bm5pbmcgbG9jYWxseS4KMi4gUnVuIGBwb2V0cnkgcnVuIHRhc2sgY29tcG9zZS11cGAqLgogICAtIFJ1biBgcG9ldHJ5IHJ1biB0YXNrIGNvbXBvc2UtZG93bmAgdG8gc3BpbiBkb3duLCBjbGVhbiB1cC4KCipgYXBwL3NldHRpbmdzL3ByZXN0YXJ0LnNoYCB3aWxsIHJ1biBtaWdyYXRpb25zIGZvciB5b3UgYmVmb3JlIHRoZSBhcHAgc3RhcnRzLgoKIyMjIEJ1aWxkIFlvdXIgQXBwbGljYXRpb24KKiBDcmVhdGUgcm91dGVzIGluIGAvYXBwL3JvdXRlc2AsIGltcG9ydCAmIGFkZCB0aGVtIHRvIHRoZSBgUk9VVEVSU2AgY29uc3RhbnQgaW4gIGAvYXBwL21haW4ucHlgCiogQ3JlYXRlIGRhdGFiYXNlIG1vZGVscyB0byBgL2FwcC9tb2RlbHMvb3JtYCwgYWRkIHRoZW0gdG8gYC9hcHAvbW9kZWxzL29ybS9taWdyYXRpb25zL2Vudi5weWAgZm9yIG1pZ3JhdGlvbnMKKiBDcmVhdGUgcHlkYW50aWMgbW9kZWxzIGluIGAvYXBwL21vZGVscy9weWRhbnRpY2AKKiBTdG9yZSBjb21wbGV4IGRiIHF1ZXJpZXMgaW4gYC9hcHAvbW9kZWxzL29ybS9xdWVyaWVzYAoqIFN0b3JlIGNvbXBsZXggdGFza3MgaW4gYGFwcC90YXNrc2AuCiogQWRkIC8gZWRpdCBnbG9iYWxzIHRvIGAvLmVudmAsIGV4cG9zZSAmIGltcG9ydCB0aGVtIGZyb20gYC9hcHAvc2V0dGluZ3MvZ2xvYmFscy5weWAKICAgICogVXNlIGFueSBjb3JvdXRpbmUgYXMgYSBiYWNrZ3JvdW5kIGZ1bmN0aW9uOiBzdG9yZSBhIHJlZmVyZW5jZSBpbiB0aGUgYEFSUV9CQUNLR1JPVU5EX0ZVTkNUSU9OU2AgZW52LgogICAgKiBTZXQgYFNFTlRSWV9EU05gIGluIHlvdXIgZW52aXJvbm1lbnQgdG8gZW5hYmxlIFNlbnRyeS4KKiBEZWZpbmUgY29kZSB0byBydW4gYmVmb3JlIGxhdW5jaCAobWlncmF0aW9ucywgc2V0dXAsIGV0YykgaW4gYC9hcHAvc2V0dGluZ3MvcHJlc3RhcnQuc2hgCgojIyBGZWF0dXJlcwojIyMgQ29yZSBEZXBlbmRlbmNpZXMKKiAqKkZhc3RBUEk6KiogdG91dHMgcGVyZm9ybWFuY2Ugb24tcGFyIHdpdGggTm9kZUpTICYgR28gKyBhdXRvbWF0aWMgU3dhZ2dlciArIFJlRG9jIGdlbmVyYXRpb24uIAoqICoqR0lOTzoqKiBidWlsdCBvbiBTUUxBbGNoZW15IGNvcmUuIExpZ2h0d2VpZ2h0LCBzaW1wbGUsIGFzeW5jaHJvbm91cyBPUk0gZm9yIFBvc3RncmVTUUwuCiogKipBcnE6KiogQXN5bmNpbyArIFJlZGlzID0gZmFzdCwgcmVzb3VyY2UtbGlnaHQgam9iIHF1ZXVpbmcgJiBSUEMuCiogKipVdmljb3JuOioqIExpZ2h0bmluZy1mYXN0LCBhc3luY2hyb25vdXMgQVNHSSBzZXJ2ZXIuCiogKipPcHRpbWl6ZWQgRG9ja2VyZmlsZToqKiBPcHRpbWl6ZWQgRG9ja2VyZmlsZSBmb3IgQVNHSSBhcHBsaWNhdGlvbnMsIGZyb20gaHR0cHM6Ly9naXRodWIuY29tL3RpYW5nb2xvL3V2aWNvcm4tZ3VuaWNvcm4tZG9ja2VyLgoKIyMjIyBBZGRpdGlvbmFsIERlcGVuZGVuY2llcwoqICoqUG9zdGdyZVNRTDoqKiBSb2J1c3QsIGZ1bGx5LWZlYXR1cmVkLCBzY2FsYWJsZSwgb3Blbi1zb3VyY2UuCiogKipSZWRpczoqKiBGYXN0LCBzaW1wbGUsIGJyb2tlciBmb3IgdGhlIEFycSB0YXNrIHF1ZXVlLgoqICoqUHlkYW50aWM6KiogQ29yZSB0byBGYXN0QVBJLiBEZWZpbmUgaG93IGRhdGEgc2hvdWxkIGJlIGluIHB1cmUsIGNhbm9uaWNhbCBweXRob247IHZhbGlkYXRlIGl0IHdpdGggcHlkYW50aWMuIAoqICoqQWxlbWJpYzoqKiBIYW5kbGVzIGRhdGFiYXNlIG1pZ3JhdGlvbnMuIENvbXBhdGlibGUgd2l0aCBHSU5PLgoqICoqU1FMQWxjaGVteV9VdGlsczoqKiBQcm92aWRlcyBlc3NlbnRpYWwgaGFuZGxlcyAmIGRhdGF0eXBlcy4gQ29tcGF0aWJsZSB3aXRoIEdJTk8uCiogKipTZW50cnk6KiogT3Blbi1zb3VyY2UsIGNsb3VkLWhvc3RlZCBlcnJvciArIGV2ZW50IG1vbml0b3JpbmcuCiogKipQcmUtQ29tbWl0OioqIGF1dG9tYXRpYyBmb3JtYXR0aW5nIChgYmxhY2tgICsgYGlzb3J0YCkgYW5kIGxpbnRpbmcgKGBmbGFrZThgKS4KKiAqKlRhc2tpcHk6KiogU21hbGwsIGZsZXhpYmxlIHRhc2sgcnVubmVyIGZvciBQb2V0cnkuCg== readmeEtag: '"3cfcd59db1b45ce6f02867b4282c5e9cc15a329d"' readmeLastModified: Fri, 19 Mar 2021 05:12:56 GMT repositoryId: 193383620 description: >- High-performance Async REST API, in Python. FastAPI + GINO + Arq + Uvicorn (w/ Redis and PostgreSQL). created: '2019-06-23T18:44:16Z' updated: '2026-02-03T15:53:42Z' language: Python archived: false stars: 559 watchers: 12 forks: 46 owner: leosussan logo: https://avatars.githubusercontent.com/u/7993496?v=4 license: MIT repoEtag: '"6e6e94735765bd0ac67e5f9e9def4bec5ca071f54d0f1a498f40d97664ba1940"' repoLastModified: Tue, 03 Feb 2026 15:53:42 GMT foundInMaster: true category: Server Implementations id: ca4835106497a7abc28ad7ff8d789dfd - source: openapi3 tags repository: https://github.com/grantila/typeconv v3: true repositoryMetadata: base64Readme: >- [![npm version][npm-image]][npm-url]
[![downloads][downloads-image]][npm-url]
[![build status][build-image]][build-url]
[![coverage status][coverage-image]][coverage-url]
[![Node.JS version][node-version]][node-url]


<img src="https://raw.githubusercontent.com/grantila/typeconv/master/assets/logo.svg" width="100%" />

**typeconv** is an extremely fast *silver bullet* type conversion utility.

It converts between any of its supported types, bidirectionally.

typeconv lets you convert between type systems which have [`core-types`][core-types-github-url] converters, such as JSON Schema, TypeScript, GraphQL, Open API and [SureType][suretype-github-url]. This package can be used as an API programatically or as an application (installed in `node_modules/.bin` or by using e.g. [`npx`](https://www.npmjs.com/package/npx)).

By taking advantage of the [`core-types`][core-types-github-url] ([npm][core-types-npm-url]) toolbox for generic type handling, typeconv can convert and maintain source code location information, comments, descriptions etc. when converting between the different type systems. It is using the following converter packages:
 * [`core-types-json-schema`][core-types-json-schema-github-url] ([npm][core-types-json-schema-npm-url])
 * [`core-types-ts`][core-types-ts-github-url] ([npm][core-types-ts-npm-url])
 * [`core-types-graphql`][core-types-graphql-github-url] ([npm][core-types-graphql-npm-url])
 * [`core-types-suretype`][core-types-suretype-github-url] ([npm][core-types-suretype-npm-url])

These type systems don't share the same set of types and constraints. For example, JSON Schema has *value* constraints (like *"a string must be longer than 5 characters*") and GraphQL doesn't have `null` or key-value objects as a first-class type. Convertions will therefore produce the smallest common denominator of type information, but still be very useful. See [`core-types`][core-types-github-url] for more information on its supported types, and why not implement a new conversion package yourself!


# TL;DR CLI

Convert files from TypeScript (*"ts"*) to GraphQL (*"gql"*), put the generated files in the `gql-schemas` directory in the same directory structure as the source files:

```zsh
$ typeconv -f ts -t gql -o gql-schemas 'types/**/*.ts'
```

This generates `gql-schemas/*.graphql` for each `.ts` file in `types/` (and sub-directories).

*Note that when using glob patterns, put them in quotes to not have the shell try to expand the pattern - typeconv will do it a lot better!*


## SureType

When converting *from* SureType, typeconv will extract all *exported* validators.


## Versions

Since v2:
 * The package is a pure ESM module, no more CommonJS support. This will likely not affect CLI usages.
 * Node 12 support is dropped.


# Contents

 * [Conversion example](#conversion-example)
 * [Usage](#usage)
   * [Command line](#command-line)
   * [As API](#as-api)
     * [JSON Schema](#json-schema)
     * [Open API](#open-api)
     * [TypeScript](#typescript)
     * [GraphQL](#graphql)
     * [SureType](#suretype)


# Conversion example

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>Converting the following JSON Schema:</summary>
<p>

```json
{
	"definitions": {
		"User": {
			"type": "object",
			"title": "User type",
			"description": "This type holds the user information, such as name",
			"properties": { "name": { "type": "string", "title": "The real name" } },
			"required": [ "name" ]
		},
		"ChatLine": {
			"type": "object",
			"title": "A chat line",
			"properties": {
				"user": { "$ref": "#/definitions/User" },
				"line": { "type": "string" }
			},
			"required": [ "user", "line" ]
		}
	}
}
```

</p>
</details>

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>... to TypeScript will produce:</summary>
<p>

```ts
/*User type

This type holds the user information, such as name*/
export interface User {
	/*The real name*/
	name: string;
}

/*A chat line*/
export interface ChatLine {
	user: User;
	line: string;
}
```

or if converted into TypeScript _declarations_ (for `.d.ts` files), `export interface` will be `export declare interface`.

</p>
</details>

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>... and to GraphQL will produce:</summary>
<p>

```graphql
"""
# User type

This type holds the user information, such as name
"""
type User {
	"The real name"
	name: String!
}

"A chat line"
type ChatLine {
	user: User!
	line: String!
}
```

</p>
</details>

Conversions are bi-directional, so any of these type systems can convert to any other.


# Usage


## Command line

You can depend on `typeconv`, and if so, you'll have `node_modules/.bin/typeconv` installed. Or you run it with on-the-fly installation using `npx`, as `npx typeconv args...`.

Use `-f` (or `--from-type`) to specify *from* which type system to convert, and `-t` (or `--to-type`) to specify which type system to convert *to*. Other options can be used to configure each configuration (both the *from* and the *to*) and these options are usually only available for a specific type system.

The types supported are `gql` (GraphQL), `ts` (TypeScript), `jsc` (JSON Schema), `oapi` (Open API) and `st` (SureType).

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary><code>$ typeconv --help</code></summary>
<p>

```
Usage: typeconv [options] file ...

   Options:

   -h, --help                       Print (this) help screen
   --version                        Print the program version
   -v, --verbose                    Verbose informational output (default: false)
   --dry-run                        Prepare and perform conversion, but write no output (default: false)
   --(no-)hidden                    Include hidden files, i.e. files in .gitignore,
                                    files beginning with '.' and the '.git' directory
                                     (default: true)
   -f, --from-type <type>           Type system to convert from

                                       Values:
                                          ts    TypeScript
                                          jsc   JSON Schema
                                          gql   GraphQL
                                          oapi  Open API
                                          st    SureType
                                          ct    core-types

   -t, --to-type <type>             Type system to convert to

                                       Values:
                                          ts    TypeScript
                                          jsc   JSON Schema
                                          gql   GraphQL
                                          oapi  Open API
                                          st    SureType
                                          ct    core-types

   --(no-)shortcut                  Shortcut conversion if possible (bypassing core-types).
                                    This is possible between SureType, JSON Schema and Open API
                                    to preserve all features which would otherwise be erased.
                                     (default: true)
   -o, --output-directory <dir>     Output directory. Defaults to the same as the input files.
   -O, --output-extension <ext>     Output filename extension to use.
                                    Defaults to 'ts'/'d.ts', 'json', 'yaml' and 'graphql'.
                                    Use '-' to not save a file, but output to stdout instead.
   --(no-)strip-annotations         Removes all annotations (descriptions, comments, ...) (default: false)
   TypeScript
     --(no-)ts-declaration          Output TypeScript declarations (default: false)
     --(no-)ts-disable-lint-header  Output comments for disabling linting (default: true)
     --(no-)ts-descriptive-header   Output the header comment (default: true)
     --(no-)ts-use-unknown          Use 'unknown' type instead of 'any' (default: true)
     --ts-non-exported <method>     Strategy for non-exported types (default: include-if-referenced)

                                       Values:
                                          fail                   Fail conversion
                                          ignore                 Don't include non-exported types,
                                                                 even if referenced
                                          include                Include non-exported types
                                          inline                 Don't include non-exported types,
                                                                 inline them if necessary.
                                                                 Will fail on cyclic types
                                          include-if-referenced  Include non-exported types only if they
                                                                 are referenced from exported types

     --ts-namespaces <method>       Namespace strategy. (default: ignore)

                                       Values:
                                          ignore           Ignore namespaces entirely (default).
                                                           - When converting from TypeScript, types in namespaces
                                                           aren't exported.
                                                           - When converting to TypeScript, no attempt to
                                                           reconstruct namespaces is performed.
                                          hoist            When converting from TypeScript, hoist types inside
                                                           namespaces to top-level, so that the types are
                                                           included, but without their namespace.
                                                           This can cause conflicts, in which case deeper
                                                           declarations will be dropped in favor of more top-
                                                           level declarations.
                                                           In case of same-level (namespace depth) declarations
                                                           with the same name, only one will be exported in a
                                                           non-deterministic manner.
                                          dot              When converting from TypeScript, join the namespaces
                                                           and the exported type with a dot (.).
                                                           When converting to TypeScript, try to reconstruct
                                                           namespaces by splitting the name on dot (.).
                                          underscore       When converting from TypeScript, join the namespaces
                                                           and the exported type with an underscore (_).
                                                           When converting to TypeScript, try to reconstruct
                                                           namespaces by splitting the name on underscore (_).
                                          reconstruct-all  When converting to TypeScript, try to reconstruct
                                                           namespaces by splitting the name on both dot and
                                                           underscore.

   GraphQL
     --gql-unsupported <method>     Method to use for unsupported types

                                       Values:
                                          ignore  Ignore (skip) type
                                          warn    Ignore type, but warn
                                          error   Throw an error

     --gql-null-typename <name>     Custom type name to use for null
   Open API
     --oapi-format <fmt>            Output format for Open API (default: yaml)

                                       Values:
                                          json  JSON
                                          yaml  YAML ('yml' is also allowed)

     --oapi-title <title>           Open API title to use in output document.
                                    Defaults to the input filename.
     --oapi-version <version>       Open API document version to use in output document. (default: 1)
   SureType
     --st-ref-method <method>       SureType reference export method (default: provided)

                                       Values:
                                          no-refs   Don't ref anything, inline all types
                                          provided  Reference types that are explicitly exported
                                          ref-all   Ref all provided types and those with names

     --st-missing-ref <method>      What to do when detecting an unresolvable reference (default: warn)

                                       Values:
                                          ignore  Ignore; skip type or cast to any
                                          warn    Same as 'ignore', but warn
                                          error   Fail conversion

     --(no-)st-inline-types         Inline pretty typescript types aside validator code (default: true)
     --(no-)st-export-type          Export the deduced types (or the pretty types,
                                    depending on --st-inline-types)
                                     (default: true)
     --(no-)st-export-schema        Export validator schemas (default: false)
     --(no-)st-export-validator     Export regular validators (default: true)
     --(no-)st-export-ensurer       Export 'ensurer' validators (default: true)
     --(no-)st-export-type-guard    Export type guards (is* validators) (default: true)
     --(no-)st-use-unknown          Use 'unknown' type instead of 'any' (default: true)
     --(no-)st-forward-schema       Forward the JSON Schema, and create an untyped validator schema
                                    with the raw JSON Schema under the hood
                                     (default: false)
```

</p>
</details>


## As API

To convert from one type system to another, you create a *reader* for the type system to convert **from** and a *writer* for the type system to convert **to**. The readers and writers for the different type systems have their own set of options. Some have no options (like the JSON Schema reader), some require options (like the Open API writer).


### makeConverter

Making a converter is done using `makeConverter(reader, writer, options?)`, which takes a reader and a writer, and optionally options:

#### **cwd** (string)

The current working directory, only useful when working with files.

#### **simplify** (boolean) (default true)

When simplify is true, the converter will let core-types
[*compress*](https://github.com/grantila/core-types#simplify) the
types after having converted from {reader} format to core-types.
This is usually recommended, but may cause some annotations (comments)
to be dropped.

#### **map** (ConvertMapFunction)

Custom map function for transforming each type after it has been
converted *from* the source type (and after it has been simplified),
but before it's written to the target type system.

Type: `(node: NamedType, index: number, array: ReadonlyArray<NamedType>) => NamedType`
([NamedType ref](https://github.com/grantila/core-types#specification))

If `filter` is used as well, this runs before `filter`.

If `transform` is used as well, this runs before `transform`.

#### **filter** (ConvertFilterFunction)

Custom filter function for filtering types after they have been
converted *from* the source type.

Type: `(node: NamedType, index: number, array: ReadonlyArray<NamedType>) => boolean`
([NamedType ref](https://github.com/grantila/core-types#specification))

If `map` is used as well, this runs after `map`.

If `transform` is used as well, this runs before `transform`.

#### **transform** (ConvertTransformFunction)

Custom filter function for filtering types after they have been
converted *from* the source type.

Type: `( doc: NodeDocument ) => NodeDocument`
([NodeDocument ref](https://github.com/grantila/core-types#specification))

If `map` is used as well, this runs after `map`.

If `filter` is used as well, this runs after `filter`.

#### **shortcut** (boolean) (default true)

Shortcut reader and writer if possible (bypassing core-types).



### Basic example conversion

```ts
import {
  getTypeScriptReader,
  getOpenApiWriter,
  makeConverter,
} from 'typeconv'

const reader = getTypeScriptReader( );
const writer = getOpenApiWriter( { format: 'yaml', title: 'My API', version: 'v1' } );
const { convert } = makeConverter( reader, writer );
const { data } = await convert( { data: "export type Foo = string | number;" } );
data; // This is the Open API yaml as a string
```


### JSON Schema

There are two exported functions for JSON Schema:

```ts
import { getJsonSchemaReader, getJsonSchemaWriter } from 'typeconv'

const reader = getJsonSchemaReader( );
const writer = getJsonSchemaWriter( );
```

They don't have any options.

typeconv expects the JSON Schema to contain **definitions**, i.e. to be in the form:

<details style="padding-left: 32px;border-left: 4px solid gray;">
<summary>JSON Schema</summary>
<p>

```json
{
	"definitions": {
		"User": {
			"type": "object",
			"properties": { "name": { "type": "string" } },
			"required": [ "name" ]
		},
		"ChatLine": {
			"type": "object",
			"properties": {
				"user": { "$ref": "#/definitions/User" },
				"line": { "type": "string" }
			},
			"required": [ "user", "line" ]
		}
	}
}
```

</p>
</details>

typeconv doesn't support external references (to other files). If you have that, you need to use a reference parser and merge it into one inline-referenced file before using typeconv.


### Open API

Converting to or from Open API can be done with both JSON and YAML. The default is JSON.

When reading, if the filename ends with `.yml` or `.yaml`, typeconv will interpret the input as YAML.

Writing however, is decided in the writer factory and provided to `getOpenApiWriter`.

```ts
import { getOpenApiReader, getOpenApiWriter } from 'typeconv'

const reader = getOpenApiReader( );
const writer = getOpenApiWriter( {
  format: 'yaml',
  title: 'My API',
  version: 'v1',
  schemaVersion: '3.0.0',
} );
```

The options to `getOpenApiWriter` is:

```ts
interface {
  format?: string;
  title: string;
  version: string;
  schemaVersion?: string;
}
```


### TypeScript

TypeScript conversion is done using:

```ts
import { getTypeScriptReader, getTypeScriptWriter } from 'typeconv'

const reader = getTypeScriptReader( );
const writer = getTypeScriptWriter( );
```

Both these take an optional argument.

The `getTypeScriptReader` takes an optional
[`FromTsOptions`](https://github.com/grantila/core-types-ts#typescript-to-core-types)
object from [`core-types-ts`][core-types-ts-github-url], although `warn` isn't necessary since it's set by typeconv internally.

The `getTypeScriptWriter` takes an optional
[`ToTsOptions`](https://github.com/grantila/core-types-ts#core-types-to-typescript)
object from [`core-types-ts`][core-types-ts-github-url], although `warn`, `filename`, `sourceFilename`, `userPackage` and `userPackageUrl` aren't necessary since they're set by typeconv internally.


### GraphQL

GraphQL conversion is done using;

```ts
import { getGraphQLReader, getGraphQLWriter } from 'typeconv'

const reader = getGraphQLReader( );
const writer = getGraphQLWriter( );
```

Both these take an optional argument.

The `getGraphQLReader` takes an optional
[`GraphqlToCoreTypesOptions`](https://github.com/grantila/core-types-graphql#graphql-to-core-types)
object from [`core-types-graphql`][core-types-graphql-github-url], although `warn` isn't necessary since it's set by typeconv internally.

The `getGraphQLWriter` takes an optional
[`CoreTypesToGraphqlOptions`](https://github.com/grantila/core-types-graphql#core-types-to-graphql)
object from [`core-types-graphql`][core-types-graphql-github-url], although `warn`, `filename`, `sourceFilename`, `userPackage` and `userPackageUrl` aren't necessary since they're set by typeconv internally.


### SureType

SureType conversion is done using;

```ts
import { getSureTypeReader, getSureTypeWriter } from 'typeconv'

const reader = getSureTypeReader( );
const writer = getSureTypeWriter( );
```

Both these take an optional argument from the [`core-types-suretype`][core-types-suretype-github-url] package.

The `getSureTypeReader` takes an optional
[`SuretypeToJsonSchemaOptions`](https://github.com/grantila/core-types-suretype#suretype-to-core-types).

The `getSureTypeWriter` takes an optional
[`JsonSchemaToSuretypeOptions`](https://github.com/grantila/core-types-suretype#core-types-to-suretype).


[npm-image]: https://img.shields.io/npm/v/typeconv.svg
[npm-url]: https://npmjs.org/package/typeconv
[downloads-image]: https://img.shields.io/npm/dm/typeconv.svg
[build-image]: https://img.shields.io/github/actions/workflow/status/grantila/typeconv/master.yml?branch=master
[build-url]: https://github.com/grantila/typeconv/actions?query=workflow%3AMaster
[coverage-image]: https://coveralls.io/repos/github/grantila/typeconv/badge.svg?branch=master
[coverage-url]: https://coveralls.io/github/grantila/typeconv?branch=master
[node-version]: https://img.shields.io/node/v/typeconv
[node-url]: https://nodejs.org/en/

[core-types-npm-url]: https://npmjs.org/package/core-types
[core-types-github-url]: https://github.com/grantila/core-types
[core-types-json-schema-npm-url]: https://npmjs.org/package/core-types-json-schema
[core-types-json-schema-github-url]: https://github.com/grantila/core-types-json-schema
[core-types-ts-npm-url]: https://npmjs.org/package/core-types-ts
[core-types-ts-github-url]: https://github.com/grantila/core-types-ts
[core-types-graphql-npm-url]: https://npmjs.org/package/core-types-graphql
[core-types-graphql-github-url]: https://github.com/grantila/core-types-graphql
[core-types-suretype-npm-url]: https://npmjs.org/package/core-types-suretype
[core-types-suretype-github-url]: https://github.com/grantila/core-types-suretype
[suretype-github-url]: https://github.com/grantila/suretype
 readmeEtag: '"b7043a416d92ebe262c0d06cc82bc8aae498de44"' readmeLastModified: Sun, 04 Jun 2023 08:11:37 GMT repositoryId: 293245471 description: Convert between JSON Schema, TypeScript, GraphQL, Open API and SureType created: '2020-09-06T09:40:04Z' updated: '2026-01-11T08:51:56Z' language: TypeScript archived: false stars: 447 watchers: 2 forks: 10 owner: grantila logo: https://avatars.githubusercontent.com/u/5362579?v=4 license: MIT repoEtag: '"a2c72fd4df09b1e3588fded6157437d93dc8af2aec098fd5b275c952c0451cac"' repoLastModified: Sun, 11 Jan 2026 08:51:56 GMT foundInMaster: true category: Parsers id: 31c759927a7f06228ec927500d9722c5 - source: openapi3 tags repository: https://github.com/0xtheprodev/fastapi-clean-example v3: true id: db8a3c5869dbcd0c85b08af9e691e2ad repositoryMetadata: base64Readme: >- IyBmYXN0YXBpLWNsZWFuLWV4YW1wbGUKClshW1B5dGhvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9weXRob24tMzY3MEEwP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1weXRob24mbG9nb0NvbG9yPWZmZGQ1NCldKGh0dHBzOi8vZG9jcy5weXRob24ub3JnLzMvKQpbIVtGYXN0QVBJXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0Zhc3RBUEktMDA1NTcxP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1mYXN0YXBpKV0oaHR0cHM6Ly9mYXN0YXBpLnRpYW5nb2xvLmNvbS8pClshW09wZW5BUEldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvb3BlbmFwaS02QkE1Mzk/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPW9wZW5hcGktaW5pdGlhdGl2ZSZsb2dvQ29sb3I9ZmZmKV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykKWyFbU3dhZ2dlcl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS8tU3dhZ2dlci0lMjNDbG9qdXJlP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1zd2FnZ2VyJmxvZ29Db2xvcj13aGl0ZSldKGh0dHBzOi8vc3dhZ2dlci5pby8pClshW0dyYXBoUUxdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvLUdyYXBoUUwtRTEwMDk4P3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1ncmFwaHFsJmxvZ29Db2xvcj13aGl0ZSldKGh0dHBzOi8vZ3JhcGhxbC5vcmcvKQpbIVtDb2RlIHN0eWxlOiBibGFja10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlJTIwc3R5bGUtYmxhY2stMDAwMDAwLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9ibGFjay5yZWFkdGhlZG9jcy5pby9lbi9zdGFibGUvKQpbIVtUeXBlZCB3aXRoOiBweWRhbnRpY10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS90eXBlZCUyMHdpdGgtcHlkYW50aWMtQkE2MDBGLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9kb2NzLnB5ZGFudGljLmRldi8pClshW09wZW4gSXNzdWVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtcmF3LzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZS9pc3N1ZXMpClshW0Nsb3NlZCBJc3N1ZXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy1jbG9zZWQtcmF3LzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZS9pc3N1ZXM/cT1pcyUzQWlzc3VlK2lzJTNBY2xvc2VkKQpbIVtPcGVuIFB1bGxzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtcHItcmF3LzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZS9wdWxscykKWyFbQ2xvc2VkIFB1bGxzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtcHItY2xvc2VkLXJhdy8weFRoZVByb0Rldi9mYXN0YXBpLWNsZWFuLWV4YW1wbGU/c3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9mYXN0YXBpLWNsZWFuLWV4YW1wbGUvcHVsbHM/cT1pcyUzQXByK2lzJTNBY2xvc2VkKQpbIVtDb250cmlidXRvcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2NvbnRyaWJ1dG9ycy8weFRoZVByb0Rldi9mYXN0YXBpLWNsZWFuLWV4YW1wbGU/c3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9mYXN0YXBpLWNsZWFuLWV4YW1wbGUvZ3JhcGhzL2NvbnRyaWJ1dG9ycykKWyFbQWN0aXZpdHldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhc3QtY29tbWl0LzB4VGhlUHJvRGV2L2Zhc3RhcGktY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxhYmVsPW1vc3QlMjByZWNlbnQlMjBhY3Rpdml0eSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9mYXN0YXBpLWNsZWFuLWV4YW1wbGUvcHVsc2UpCgojIyBEZXNjcmlwdGlvbgoKX0V4YW1wbGUgQXBwbGljYXRpb24gSW50ZXJmYWNlIHVzaW5nIEZhc3RBUEkgZnJhbWV3b3JrIGluIFB5dGhvbiAzXwoKVGhpcyBleGFtcGxlIHNob3djYXNlcyBSZXBvc2l0b3J5IFBhdHRlcm4gaW4gSGV4YWdvbmFsIEFyY2hpdGVjdHVyZSBfKGFsc28ga25vd24gYXMgQ2xlYW4gQXJjaGl0ZWN0dXJlKV8uIEhlcmUgd2UgaGF2ZSB0d28gRW50aXRpZXMgLSBCb29rcyBhbmQgQXV0aG9ycywgd2hvc2UgcmVsYXRpb25zaGlwcyBoYXZlIGJlZW4gZXhwbG9pdGVkIHRvIGNyZWF0ZSBDUlVEIGVuZHBvaW50IGluIFJFU1QgdW5kZXIgT3BlbkFQSSBzdGFuZGFyZC4KCiMjIEluc3RhbGxhdGlvbgoKLSBJbnN0YWxsIGFsbCB0aGUgcHJvamVjdCBkZXBlbmRlbmN5IHVzaW5nIFtQaXBlbnZdKGh0dHBzOi8vcGlwZW52LnB5cGEuaW8pOgoKICBgYGBzaAogICQgcGlwZW52IGluc3RhbGwgLS1kZXYKICBgYGAKCi0gUnVuIHRoZSBhcHBsaWNhdGlvbiBmcm9tIGNvbW1hbmQgcHJvbXB0OgoKICBgYGBzaAogICQgcGlwZW52IHJ1biB1dmljb3JuIG1haW46YXBwIC0tcmVsb2FkCiAgYGBgCgotIFlvdSBjYW4gYWxzbyBvcGVuIGEgc2hlbGwgaW5zaWRlIHZpcnR1YWwgZW52aXJvbm1lbnQ6CgogIGBgYHNoCiAgJCBwaXBlbnYgc2hlbGwKICBgYGAKCi0gT3BlbiBgbG9jYWxob3N0OjgwMDAvZG9jc2AgZm9yIEFQSSBEb2N1bWVudGF0aW9uCgotIE9wZW4gYGxvY2FsaG9zdDo4MDAwL2dyYXBocWxgIGZvciBHcmFwaFFMIERvY3VtZW50YXRpb24KCl8qTm90ZToqIEluIGNhc2UgeW91IGFyZSBub3QgYWJsZSB0byBhY2Nlc3MgYHBpcGVudmAgZnJvbSB5b3UgYFBBVEhgIGxvY2F0aW9ucywgcmVwbGFjZSBhbGwgaW5zdGFuY2VzIG9mIGBwaXBlbnZgIHdpdGggYHB5dGhvbjMgLW0gcGlwZW52YC5fCgojIyBUZXN0aW5nCgpGb3IgVGVzdGluZywgYHVuaXR0ZXN0YCBtb2R1bGUgaXMgdXNlZCBmb3IgVGVzdCBTdWl0ZSBhbmQgQXNzZXJ0aW9uLCB3aGVyZWFzIGBweXRlc3RgIGlzIGJlaW5nIHVzZWQgZm9yIFRlc3QgUnVubmVyIGFuZCBDb3ZlcmFnZSBSZXBvcnRlci4KCi0gUnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB0byBpbml0aWF0ZSB0ZXN0OgogIGBgYHNoCiAgJCBwaXBlbnYgcnVuIHB5dGVzdAogIGBgYAotIFRvIGluY2x1ZGUgQ292ZXJhZ2UgUmVwb3J0aW5nIGFzIHdlbGw6CiAgYGBgc2gKICAkIHBpcGVudiBydW4gcHl0ZXN0IC0tY292LXJlcG9ydCB4bWwgLS1jb3YgLgogIGBgYAoKIyMgTGljZW5zZQoKJmNvcHk7IE1JVCBMaWNlbnNlCg== readmeEtag: '"bef2ee4db7b02a92c0311125aa4cb2829b3cb22c"' readmeLastModified: Fri, 12 May 2023 09:03:57 GMT repositoryId: 479982630 description: Clean Architecture Example using FastAPI framework created: '2022-04-10T10:23:51Z' updated: '2026-02-03T10:16:58Z' language: Python archived: false stars: 498 watchers: 5 forks: 66 owner: 0xTheProDev logo: https://avatars.githubusercontent.com/u/14367736?v=4 license: MIT repoEtag: '"f1dd655f986ad38f12138d5d098f5dd1fe1756cd9cf32e3cedc489aac20c1292"' repoLastModified: Tue, 03 Feb 2026 10:16:58 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/progyan1997/fastapi-clean-example - https://github.com/progyan1997/fastapi-example - source: openapi3 tags repository: https://github.com/madkote/fastapi-plugins v3: true repositoryMetadata: base64Readme: >- <p align="center">
    <em>Plugins for FastAPI framework, high performance, easy to learn, fast to code, ready for production</em>
</p>
<p align="center">
<a href="https://travis-ci.org/madkote/fastapi-plugins" target="_blank">
    <img src="https://travis-ci.org/madkote/fastapi_plugins.svg?branch=master" alt="Build Status">
</a>
<a href="https://codecov.io/gh/madkote/fastapi-plugins" target="_blank">
    <img src="https://codecov.io/gh/madkote/fastapi_plugins/branch/master/graph/badge.svg" alt="Coverage">
</a>
<a href="https://pypi.org/project/fastapi-plugins" target="_blank">
    <img src="https://img.shields.io/pypi/v/fastapi_plugins.svg" alt="Package version">
</a>
<a href="https://gitter.im/tiangolo/fastapi?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge" target="_blank">
    <img src="https://badges.gitter.im/tiangolo/fastapi.svg" alt="Join the chat at https://gitter.im/tiangolo/fastapi">
</a>
</p>

# fastapi-plugins
FastAPI framework plugins - simple way to share `fastapi` code and utilities across applications.

The concept is `plugin` - plug a functional utility into your application without or with minimal effort.

* [Cache](./docs/cache.md)
  * [Memcached](./docs/cache.md#memcached)
  * [Redis](./docs/cache.md#redis)
* [Scheduler](./docs/scheduler.md)
* [Control](./docs/control.md)
  * [Version](./docs/control.md#version)
  * [Environment](./docs/control.md#environment)
  * [Health](./docs/control.md#health)
  * [Heartbeat](./docs/control.md#heartbeat)
* [Application settings/configuration](./docs/settings.md)
* [Logging](./docs/logger.md)
* Celery
* MQ
* and much more is already in progress...

## Changes
See [release notes](CHANGES.md)

## Installation
* by default contains
  * [Redis](./docs/cache.md#redis)
  * [Scheduler](./docs/scheduler.md)
  * [Control](./docs/control.md)
  * [Logging](./docs/logger.md)
* `memcached` adds [Memcached](#memcached)
* `all` add everything above

```sh
pip install fastapi-plugins
pip install fastapi-plugins[memcached]
pip install fastapi-plugins[all]
```

## Quick start
### Plugin
Add information about plugin system.
### Application settings
Add information about settings.
### Application configuration
Add information about configuration of an application
### Complete example
```python
import fastapi
import fastapi_plugins

from fastapi_plugins.memcached import MemcachedSettings
from fastapi_plugins.memcached import memcached_plugin, TMemcachedPlugin

import asyncio
import aiojobs
import aioredis
import contextlib
import logging

@fastapi_plugins.registered_configuration
class AppSettings(
        fastapi_plugins.ControlSettings,
        fastapi_plugins.RedisSettings,
        fastapi_plugins.SchedulerSettings,
        fastapi_plugins.LoggingSettings,
        MemcachedSettings,
):
    api_name: str = str(__name__)
    logging_level: int = logging.DEBUG
    logging_style: fastapi_plugins.LoggingStyle = fastapi_plugins.LoggingStyle.logjson


@fastapi_plugins.registered_configuration(name='sentinel')
class AppSettingsSentinel(AppSettings):
    redis_type = fastapi_plugins.RedisType.sentinel
    redis_sentinels = 'localhost:26379'


@contextlib.asynccontextmanager
async def lifespan(app: fastapi.FastAPI):
    config = fastapi_plugins.get_config()
    await fastapi_plugins.config_plugin.init_app(app, config)
    await fastapi_plugins.config_plugin.init()
    await fastapi_plugins.log_plugin.init_app(app, config, name=__name__)
    await fastapi_plugins.log_plugin.init()
    await memcached_plugin.init_app(app, config)
    await memcached_plugin.init()
    await fastapi_plugins.redis_plugin.init_app(app, config=config)
    await fastapi_plugins.redis_plugin.init()
    await fastapi_plugins.scheduler_plugin.init_app(app=app, config=config)
    await fastapi_plugins.scheduler_plugin.init()
    await fastapi_plugins.control_plugin.init_app(
        app,
        config=config,
        version=__version__,
        environ=config.model_dump()
    )
    await fastapi_plugins.control_plugin.init()
    yield
    await fastapi_plugins.control_plugin.terminate()
    await fastapi_plugins.scheduler_plugin.terminate()
    await fastapi_plugins.redis_plugin.terminate()
    await memcached_plugin.terminate()
    await fastapi_plugins.log_plugin.terminate()
    await fastapi_plugins.config_plugin.terminate()


app = fastapi_plugins.register_middleware(fastapi.FastAPI(lifespan=lifespan))


@app.get("/")
async def root_get(
        cache: fastapi_plugins.TRedisPlugin,
        conf: fastapi_plugins.TConfigPlugin,
        logger: fastapi_plugins.TLoggerPlugin
) -> typing.Dict:
    ping = await cache.ping()
    logger.debug('root_get', extra=dict(ping=ping, api_name=conf.api_name))
    return dict(ping=ping, api_name=conf.api_name)


@app.post("/jobs/schedule/<timeout>")
async def job_post(
    timeout: int=fastapi.Query(..., title='the job sleep time'),
    cache: fastapi_plugins.TRedisPlugin,
    scheduler: fastapi_plugins.TSchedulerPlugin,
    logger: fastapi_plugins.TLoggerPlugin
) -> str:
    async def coro(job_id, timeout, cache):
        await cache.set(job_id, 'processing')
        try:
            await asyncio.sleep(timeout)
            if timeout == 8:
                logger.critical('Ugly erred job %s' % job_id)
                raise Exception('ugly error')
        except asyncio.CancelledError:
            await cache.set(job_id, 'canceled')
            logger.warning('Cancel job %s' % job_id)
        except Exception:
            await cache.set(job_id, 'erred')
            logger.error('Erred job %s' % job_id)
        else:
            await cache.set(job_id, 'success')
            logger.info('Done job %s' % job_id)

    job_id = str(uuid.uuid4()).replace('-', '')
    logger = await fastapi_plugins.log_adapter(logger, extra=dict(job_id=job_id, timeout=timeout))    # noqa E501
    logger.info('New job %s' % job_id)
    await cache.set(job_id, 'pending')
    logger.debug('Pending job %s' % job_id)
    await scheduler.spawn(coro(job_id, timeout, cache))
    return job_id


@app.get("/jobs/status/<job_id>")
async def job_get(
    job_id: str=fastapi.Query(..., title='the job id'),
    cache: fastapi_plugins.TRedisPlugin,
) -> typing.Dict:
    status = await cache.get(job_id)
    if status is None:
        raise fastapi.HTTPException(
            status_code=starlette.status.HTTP_404_NOT_FOUND,
            detail='Job %s not found' % job_id
        )
    return dict(job_id=job_id, status=status)


@app.post("/memcached/demo/<key>")
async def memcached_demo_post(
    key: str=fastapi.Query(..., title='the job id'),
    cache: fastapi_plugins.TMemcachedPlugin,
) -> typing.Dict:
    await cache.set(key.encode(), str(key + '_value').encode())
    value = await cache.get(key.encode())
    return dict(ping=(await cache.ping()).decode(), key=key, value=value)
```

# Development
Issues and suggestions are welcome through [issues](https://github.com/madkote/fastapi-plugins/issues)

# License
This project is licensed under the terms of the MIT license.
 readmeEtag: '"0a423b3a922daaf582fc80759adb5b12ba91c44f"' readmeLastModified: Mon, 03 Jun 2024 15:39:27 GMT repositoryId: 222924508 description: FastAPI framework plugins created: '2019-11-20T11:41:50Z' updated: '2026-02-04T08:12:56Z' language: Python archived: false stars: 609 watchers: 2 forks: 25 owner: madkote logo: https://avatars.githubusercontent.com/u/4921881?v=4 license: MIT repoEtag: '"9571b32e213b6bae5ea7e785f7f4b1a0f5fbf8ab2376440d3f0d78ca81b5be2d"' repoLastModified: Wed, 04 Feb 2026 08:12:56 GMT foundInMaster: true category: Server Implementations id: 5b8a96994e594bd26bca9505bbc19018 - source: openapi3 tags name: rest homepage: https://github.com/swaggest/rest language: Go source_description: >- API server with automatic request/response mapping/validation and OpenAPI schema category: Server Implementations repository: https://github.com/swaggest/rest v3: true repositoryMetadata: base64Readme: >- # REST with Clean Architecture for Go

[![Build Status](https://github.com/swaggest/rest/workflows/test-unit/badge.svg)](https://github.com/swaggest/rest/actions?query=branch%3Amaster+workflow%3Atest-unit)
[![Coverage Status](https://codecov.io/gh/swaggest/rest/branch/master/graph/badge.svg)](https://codecov.io/gh/swaggest/rest)
[![GoDevDoc](https://img.shields.io/badge/dev-doc-00ADD8?logo=go)](https://pkg.go.dev/github.com/swaggest/rest)
[![Ask DeepWiki](https://deepwiki.com/badge.svg)](https://deepwiki.com/swaggest/rest)
[![Time Tracker](https://wakatime.com/badge/github/swaggest/rest.svg)](https://wakatime.com/badge/github/swaggest/rest)

This module implements HTTP transport level for [`github.com/swaggest/usecase`](https://github.com/swaggest/usecase) 
to build REST services.

## Goals

* Maintain single source of truth for documentation, validation and input/output of HTTP API.
* Avoid dependency on compile time code generation.
* Improve productivity and reliability by abstracting HTTP details with simple API for common case.
* Allow low-level customizations for advanced cases.
* Maintain reasonable performance with low GC impact.

## Non-Goals

* Support for legacy documentation schemas like Swagger 2.0 or RAML.
* Zero allocations.
* Explicit support for XML in request or response bodies.

## Features

* Compatible with `net/http`.
* Built with [`github.com/go-chi/chi`](https://github.com/go-chi/chi) router.
* Modular flexible structure.
* HTTP [request mapping](#request-decoder) into Go value based on field tags.
* Decoupled business logic with Clean Architecture use cases.
* Automatic type-safe OpenAPI 3.0/3.1 documentation with [`github.com/swaggest/openapi-go`](https://github.com/swaggest/openapi-go).
* Single source of truth for the documentation and endpoint interface.
* Automatic request/response JSON schema validation with [`github.com/santhosh-tekuri/jsonschema`](https://github.com/santhosh-tekuri/jsonschema).
* Dynamic gzip compression and fast pass through mode.
* Optimized performance.
* Embedded [Swagger UI](https://swagger.io/tools/swagger-ui/).
* Generic interface for [use case interactors](https://pkg.go.dev/github.com/swaggest/usecase#NewInteractor). 

## Usage

Please check this [tutorial](https://dev.to/vearutop/tutorial-developing-a-restful-api-with-go-json-schema-validation-and-openapi-docs-2490) for end-to-end usage example.

### Request Decoder

Go struct with field tags defines input port. 
Request decoder populates field values from `http.Request` data before use case interactor is invoked. 

```go
// Declare input port type.
type helloInput struct {
    Locale string `query:"locale" default:"en-US" pattern:"^[a-z]{2}-[A-Z]{2}$" enum:"ru-RU,en-US"`
    Name   string `path:"name" minLength:"3"` // Field tags define parameter location and JSON schema constraints.

    // Field tags of unnamed fields are applied to parent schema, 
	// they are optional and can be used to disallow unknown parameters.
    // For non-body params, name tag must be provided explicitly.
    // E.g. here no unknown `query` and `cookie` parameters allowed,
    // unknown `header` params are ok.
    _ struct{} `query:"_" cookie:"_" additionalProperties:"false"`
}
```

Input data can be located in:
* `path` parameter in request URI, e.g. `/users/{name}`,
* `query` parameter in request URI, e.g. `/users?locale=en-US`,
* `formData` parameter in request body with `application/x-www-form-urlencoded` or `multipart/form-data` content,
* `form` parameter acts as `formData` or `query`,
* `json` parameter in request body with `application/json` content,
* `cookie` parameter in request cookie,
* `header` parameter in request header,
* `contentType` of matching raw request body.

For more explicit separation of concerns between use case and transport it is possible to provide request mapping 
separately when initializing handler (please note, such mapping is [not applied](https://github.com/swaggest/rest/issues/61#issuecomment-1059851553) to `json` body).

```go
// Declare input port type.
type helloInput struct {
    Locale string `default:"en-US" pattern:"^[a-z]{2}-[A-Z]{2}$"`
    Name   string `minLength:"3"` // Field tags define parameter location and JSON schema constraints.
}
```

```go
// Add use case handler with custom input mapping to router.
r.Method(http.MethodGet, "/hello/{name}", nethttp.NewHandler(u,
    nethttp.RequestMapping(new(struct {
       Locale string `query:"locale"`
       Name   string `path:"name"` // Field tags define parameter location and JSON schema constraints.
    })),
))
```

Additional field tags describe JSON schema constraints, please check 
[documentation](https://pkg.go.dev/github.com/swaggest/jsonschema-go#Reflector.Reflect).

More schema customizations are possible with [`github.com/swaggest/jsonschema-go interfaces`](https://github.com/swaggest/jsonschema-go#implementing-interfaces-on-a-type).

By default `default` tags are only contributing to documentation, 
if [`request.DecoderFactory.ApplyDefaults`](https://pkg.go.dev/github.com/swaggest/rest/request#DecoderFactory) is 
set to `true`, fields of request structure that don't have explicit value but have `default` will be populated with 
default value.

If input structure implements [`request.Loader`](https://pkg.go.dev/github.com/swaggest/rest/request#Loader),  
then `LoadFromHTTPRequest(r *http.Request) error` method will be invoked to populate input structure instead 
of automatic decoding. This allows low level control for cases that need it.

<details>
<summary>Request decoder can be used standalone, in already existing `ServeHTTP`.</summary>

```go
type MyRequest struct {
    Foo int    `header:"X-Foo"`
    Bar string `formData:"bar"`
    Baz bool   `query:"baz"`
}

// A decoder for particular structure, can be reused for multiple HTTP requests.
myDecoder := request.NewDecoderFactory().MakeDecoder(http.MethodPost, new(MyRequest), nil)

// Request and response writer from ServeHTTP.
var (
    rw  http.ResponseWriter
    req *http.Request
)

// This code would presumably live in ServeHTTP.
var myReq MyRequest

if err := myDecoder.Decode(req, &myReq, nil); err != nil {
    http.Error(rw, err.Error(), http.StatusBadRequest)
}
```
</details>

### Response Encoder

Go struct with field tags defines output port.
Response encoder writes data from output to `http.ResponseWriter` after use case interactor invocation finishes.

```go
// Declare output port type.
type helloOutput struct {
    Now     time.Time `header:"X-Now" json:"-"`
    Message string    `json:"message"`
    Sess    string    `cookie:"sess,httponly,secure,max-age=86400,samesite=lax"`
}
```

Output data can be located in:
* `json` for response body with `application/json` content,
* `header` for values in response header,
* `cookie` for cookie values, cookie fields can have configuration in field tag (same as in actual cookie, but with comma separation),
* `contentType` for a non-empty string value that should be used as response body with given content type.

For more explicit separation of concerns between use case and transport it is possible to provide response header mapping 
separately when initializing handler.

```go
// Declare output port type.
type helloOutput struct {
    Now     time.Time `json:"-"`
    Message string    `json:"message"`
}
```

```go
// Add use case handler with custom output headers mapping to router.
r.Method(http.MethodGet, "/hello/{name}", nethttp.NewHandler(u,
    nethttp.ResponseHeaderMapping(new(struct {
        Now     time.Time `header:"X-Now"`
    })),
))
```

Additional field tags describe JSON schema constraints, please check 
[documentation](https://pkg.go.dev/github.com/swaggest/jsonschema-go#Reflector.Reflect).

### Creating Use Case Interactor

HTTP transport is decoupled from business logic by adapting
[use case interactors](https://pkg.go.dev/github.com/swaggest/usecase#Interactor).

Use case interactor can define input and output ports that are used to map data between Go values and transport.
It can provide information about itself that will be exposed in generated documentation.

```go
// Create use case interactor with references to input/output types and interaction function.
u := usecase.NewInteractor(func(ctx context.Context, input helloInput, output *helloOutput) error {
    msg, available := messages[input.Locale]
    if !available {
        return status.Wrap(errors.New("unknown locale"), status.InvalidArgument)
    }

    output.Message = fmt.Sprintf(msg, input.Name)
    output.Now = time.Now()

    return nil
})
```

### Initializing Web Service

[Web Service](https://pkg.go.dev/github.com/swaggest/rest/web#DefaultService) is an instrumented facade in front of 
router, it simplifies configuration and provides more compact API to add use cases.

```go
// Service initializes router with required middlewares.
service := web.NewService(openapi31.NewReflector())

// It allows OpenAPI configuration.
service.OpenAPISchema().SetTitle("Albums API")
service.OpenAPISchema().SetDescription("This service provides API to manage albums.")
service.OpenAPISchema().SetVersion("v1.0.0")

// Additional middlewares can be added.
service.Use(
    middleware.StripSlashes,

    // cors.AllowAll().Handler, // "github.com/rs/cors", 3rd-party CORS middleware can also be configured here.
)

// Use cases can be mounted using short syntax .<Method>(...).
service.Post("/albums", postAlbums(), nethttp.SuccessStatus(http.StatusCreated))

log.Println("Starting service at http://localhost:8080")

if err := http.ListenAndServe("localhost:8080", service); err != nil {
    log.Fatal(err)
}

```

Usually, `web.Service` API is sufficient, but if it is not, router can be configured manually, please check 
the documentation below.


## Security Setup

Example with HTTP Basic Auth.

```go
// Prepare middleware with suitable security schema.
// It will perform actual security check for every relevant request.
adminAuth := middleware.BasicAuth("Admin Access", map[string]string{"admin": "admin"})

// Prepare API schema updater middleware.
// It will annotate handler documentation with security schema.
adminSecuritySchema := nethttp.HTTPBasicSecurityMiddleware(apiSchema, "Admin", "Admin access")

// Endpoints with admin access.
r.Route("/admin", func(r chi.Router) {
    r.Group(func(r chi.Router) {
        r.Use(adminAuth, adminSecuritySchema) // Add both middlewares to routing group to enforce and document security.
        r.Method(http.MethodPut, "/hello/{name}", nethttp.NewHandler(u))
    })
})
```

Example with cookie.

```go
// Security middlewares.
//  - sessMW is the actual request-level processor,
//  - sessDoc is a handler-level wrapper to expose docs.
sessMW := func(handler http.Handler) http.Handler {
    return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
        if c, err := r.Cookie("sessid"); err == nil {
            r = r.WithContext(context.WithValue(r.Context(), "sessionID", c.Value))
        }

        handler.ServeHTTP(w, r)
    })
}

sessDoc := nethttp.APIKeySecurityMiddleware(s.OpenAPICollector, "User",
    "sessid", oapi.InCookie, "Session cookie.")

// Security schema is configured for a single top-level route.
s.With(sessMW, sessDoc).Method(http.MethodGet, "/root-with-session", nethttp.NewHandler(dummy()))

// Security schema is configured on a sub-router.
s.Route("/deeper-with-session", func(r chi.Router) {
    r.Group(func(r chi.Router) {
        r.Use(sessMW, sessDoc)

        r.Method(http.MethodGet, "/one", nethttp.NewHandler(dummy()))
        r.Method(http.MethodGet, "/two", nethttp.NewHandler(dummy()))
    })
})

```

See [example](./_examples/task-api/internal/infra/nethttp/router.go).

## Handler Setup

Handler is a generalized adapter for use case interactor, so usually setup is trivial.

```go
// Add use case handler to router.
r.Method(http.MethodGet, "/hello/{name}", nethttp.NewHandler(u))
```

## Example

For non-generic use case, see another [example](./_examples/basic/main.go).

```go
package main

import (
	"context"
	"errors"
	"fmt"
	"log"
	"net/http"
	"time"

	"github.com/swaggest/openapi-go/openapi31"
	"github.com/swaggest/rest/response/gzip"
	"github.com/swaggest/rest/web"
	swgui "github.com/swaggest/swgui/v5emb"
	"github.com/swaggest/usecase"
	"github.com/swaggest/usecase/status"
)

func main() {
	s := web.NewService(openapi31.NewReflector())

	// Init API documentation schema.
	s.OpenAPISchema().SetTitle("Basic Example")
	s.OpenAPISchema().SetDescription("This app showcases a trivial REST API.")
	s.OpenAPISchema().SetVersion("v1.2.3")

	// Setup middlewares.
	s.Wrap(
		gzip.Middleware, // Response compression with support for direct gzip pass through.
	)

	// Declare input port type.
	type helloInput struct {
		Locale string `query:"locale" default:"en-US" pattern:"^[a-z]{2}-[A-Z]{2}$" enum:"ru-RU,en-US"`
		Name   string `path:"name" minLength:"3"` // Field tags define parameter location and JSON schema constraints.

		// Field tags of unnamed fields are applied to parent schema.
		// they are optional and can be used to disallow unknown parameters.
		// For non-body params, name tag must be provided explicitly.
		// E.g. here no unknown `query` and `cookie` parameters allowed,
		// unknown `header` params are ok.
		_ struct{} `query:"_" cookie:"_" additionalProperties:"false"`
	}

	// Declare output port type.
	type helloOutput struct {
		Now     time.Time `header:"X-Now" json:"-"`
		Message string    `json:"message"`
	}

	messages := map[string]string{
		"en-US": "Hello, %s!",
		"ru-RU": "Привет, %s!",
	}

	// Create use case interactor with references to input/output types and interaction function.
	u := usecase.NewInteractor(func(ctx context.Context, input helloInput, output *helloOutput) error {
		msg, available := messages[input.Locale]
		if !available {
			return status.Wrap(errors.New("unknown locale"), status.InvalidArgument)
		}

		output.Message = fmt.Sprintf(msg, input.Name)
		output.Now = time.Now()

		return nil
	})

	// Describe use case interactor.
	u.SetTitle("Greeter")
	u.SetDescription("Greeter greets you.")

	u.SetExpectedErrors(status.InvalidArgument)

	// Add use case handler to router.
	s.Get("/hello/{name}", u)

	// Swagger UI endpoint at /docs.
	s.Docs("/docs", swgui.New)

	// Start server.
	log.Println("http://localhost:8011/docs")
	if err := http.ListenAndServe("localhost:8011", s); err != nil {
		log.Fatal(err)
	}
}

```

![Documentation Page](./_examples/basic/screen.png)

## Additional Integrations

* [`github.com/gorilla/mux`](https://github.com/gorilla/mux), see [example](./gorillamux/example_openapi_collector_test.go).

## Performance Optimization

If top performance is critical for the service or particular endpoints, you can trade 
simplicity for performance by implementing manual request loader on input type.

```go
func (i *myInput) LoadFromHTTPRequest(r *http.Request) (err error) {
	i.Header = r.Header.Get("X-Header")

	return nil
}
```

If `request.Loader` is implemented, it will be called instead of both automatic decoding and validation.

Check advanced [example](https://github.com/swaggest/rest/blob/v0.2.29/_examples/advanced-generic/json_body_manual.go#L58).

To further improve performance you may try to use `fasthttp` instead of `net/http` with 
[`rest-fasthttp`](https://github.com/swaggest/rest-fasthttp) fork.

## Versioning

This project adheres to [Semantic Versioning](https://semver.org/#semantic-versioning-200).

Before version `1.0.0`, breaking changes are tagged with `MINOR` bump, features and fixes are tagged with `PATCH` bump.
After version `1.0.0`, breaking changes are tagged with `MAJOR` bump.

Breaking changes are described in [UPGRADE.md](./UPGRADE.md).

## Advanced Usage

[Advanced Usage](./ADVANCED.md)
 readmeEtag: '"321f338627aeeb2432c1e438292b05b7b43a3b39"' readmeLastModified: Sat, 24 May 2025 08:40:37 GMT repositoryId: 306110879 description: Web services with OpenAPI and JSON Schema done quick in Go created: '2020-10-21T18:16:16Z' updated: '2026-01-21T12:58:40Z' language: Go archived: false stars: 476 watchers: 5 forks: 27 owner: swaggest logo: https://avatars.githubusercontent.com/u/19609628?v=4 license: MIT repoEtag: '"4ef87cd1db3c48dccd9b2d49e06cfd9bd296cd208c9f164907b46507f1693f67"' repoLastModified: Wed, 21 Jan 2026 12:58:40 GMT foundInMaster: true id: ecd2fef7bb4079bda209503f9fb9820b - source: openapi3 tags repository: https://github.com/endpoints4s/endpoints4s v3: true id: 8eb71a5192e00663da80338fa967ab89 repositoryMetadata: base64Readme: >- IyBlbmRwb2ludHM0cyAKClshW0dpdHRlcl0oaHR0cHM6Ly9iYWRnZXMuZ2l0dGVyLmltLy5zdmcpXShodHRwczovL2dpdHRlci5pbS9lbmRwb2ludHM0cy9lbmRwb2ludHM0cykKWyFbY29kZWNvdi5pb10oaHR0cDovL2NvZGVjb3YuaW8vZ2l0aHViL2VuZHBvaW50czRzL2VuZHBvaW50czRzL2NvdmVyYWdlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cDovL2NvZGVjb3YuaW8vZ2l0aHViL2VuZHBvaW50czRzL2VuZHBvaW50czRzP2JyYW5jaD1tYXN0ZXIpClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2VuZHBvaW50czRzL2VuZHBvaW50czRzLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2VuZHBvaW50czRzL2VuZHBvaW50czRzKQoKCmVuZHBvaW50czRzIGlzIGEgU2NhbGEgbGlicmFyeSBmb3IgZGVmaW5pbmcgY29tbXVuaWNhdGlvbiBwcm90b2NvbHMgb3ZlciBIVFRQIGJldHdlZW4KYXBwbGljYXRpb25zLgoKU2VlIHRoZSBbZG9jdW1lbnRhdGlvbl0oaHR0cDovL2VuZHBvaW50czRzLmdpdGh1Yi5pbykgdG8gbGVhcm4gbW9yZS4KCiMjIFJ1bm5pbmcgdGhlIEV4YW1wbGVzCgp+fn4KJCBzYnQKPiArKzIuMTMuMgo+IDxleGFtcGxlPi9yZVN0YXJ0Cn5+fgoKV2hlcmUgYDxleGFtcGxlPmAgY2FuIGJlIGVpdGhlcgpbYGV4YW1wbGUtY3Fyc2BdKGRvY3VtZW50YXRpb24vZXhhbXBsZXMvY3FycyksCltgZXhhbXBsZS1kb2N1bWVudGVkYF0oZG9jdW1lbnRhdGlvbi9leGFtcGxlcy9kb2N1bWVudGVkKQpvciBbYGV4YW1wbGUtcXVpY2tzdGFydC1zZXJ2ZXJgXShkb2N1bWVudGF0aW9uL2V4YW1wbGVzL3F1aWNrc3RhcnQpLgoKQW5kIHRoZW4gYnJvd3NlIGh0dHA6Ly9sb2NhbGhvc3Q6OTAwMC4KCiMjIENvbnRyaWJ1dGluZwoKU2VlIHRoZSBbb3BlbiBpc3N1ZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9lbmRwb2ludHM0cy9lbmRwb2ludHM0cy9pc3N1ZXMpLgoKIyMgU3BvbnNvcnMKCklmIHlvdSBmaW5kIGVuZHBvaW50czRzIHVzZWZ1bCwgcGxlYXNlIGNvbnNpZGVyIGFza2luZyB5b3VyIGNvbXBhbnkgdG8gc3VwcG9ydCB0aGlzIHByb2plY3QgYnkgW2JlY29taW5nIGEgc3BvbnNvcl0oaHR0cHM6Ly9vcGVuY29sbGVjdGl2ZS5jb20vZW5kcG9pbnRzNHMpLiBZb3VyIGRvbmF0aW9ucyB3aWxsIGhlbHAgdGhlIHByb2plY3Qgc3RheSB1cCB0byBkYXRlIGFuZCBmdW5kIHdvcmsgb24gaW1wb3J0YW50IFtpc3N1ZXMgb3IgZmVhdHVyZSByZXF1ZXN0c10oaHR0cHM6Ly9naXRodWIuY29tL2VuZHBvaW50czRzL2VuZHBvaW50czRzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBYm91bnR5KS4KCiMjIFN1cHBvcnQKCltKZXRCcmFpbnNdKGh0dHBzOi8vd3d3LmpldGJyYWlucy5jb20vP2Zyb209c2NhbGEtZW5kcG9pbnRzKSBvZmZlcnMgb25lIE9wZW4gU291cmNlIGxpY2Vuc2UuCgojIyBMaWNlbnNlCgpUaGlzIGNvbnRlbnQgaXMgcmVsZWFzZWQgdW5kZXIgdGhlIFtNSVQgTGljZW5zZV0oaHR0cDovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL21pdC1saWNlbnNlLnBocCkuCg== readmeEtag: '"c10e79373f051a7ce3741a57ec29dd09a4acf1d8"' readmeLastModified: Tue, 21 May 2024 17:47:22 GMT repositoryId: 57041055 description: >- Describe HTTP endpoints in Scala and derive clients, servers, and documentation created: '2016-04-25T12:48:00Z' updated: '2026-01-15T22:38:22Z' language: Scala archived: false stars: 416 watchers: 10 forks: 95 owner: endpoints4s logo: https://avatars.githubusercontent.com/u/67427463?v=4 license: MIT repoEtag: '"7d2f062d1196286e096bdcddca48a4ecb424ddc5b567d9f8d904a91511b69e98"' repoLastModified: Thu, 15 Jan 2026 22:38:22 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/thomaxxl/safrs v3: true repositoryMetadata: base64Readme: >- [![Latest Version](https://img.shields.io/pypi/v/safrs.svg)](https://pypi.python.org/pypi/safrs/)
[![Supported Python versions](https://img.shields.io/pypi/pyversions/safrs.svg)](https://pypi.python.org/pypi/safrs/)
[![License: GPL v3](https://img.shields.io/badge/License-GPLv3-blue.svg)](https://www.gnu.org/licenses/gpl-3.0)
![Python application](https://github.com/thomaxxl/safrs/workflows/Python%20application/badge.svg)
[![Codacy Badge](https://app.codacy.com/project/badge/Grade/de12c50717e8487db5dcc31907e627f7)](https://www.codacy.com/gh/thomaxxl/safrs/dashboard?utm_source=github.com&amp;utm_medium=referral&amp;utm_content=thomaxxl/safrs)
[![Downloads](https://pepy.tech/badge/safrs)](https://pepy.tech/project/safrs)

# SAFRS: Python OpenAPI & JSON:API Framework

![demo](docs/images/safrs.gif)

<a class="mk-toclify" id="table-of-contents"></a>

- [Introduction](#overview)
- [Installation](#installation)
- [JSON:API Interface](#http-methods)
- [Resource Objects](#resource-objects)
- [Relationships](#relationships)
- [Methods](#methods)
    - [Custom Methods](#custom-methods)
    - [Class Methods](#class-methods)
- [Initialization](#initialization)
- [Endpoint Naming](#endpoint-naming)
- [Configuration](#configuration)
- [Exposing Existing Databases](#expose-existing)
- [More Examples and Use Cases](#more-examples-and-use-cases)
- [Advanced Functionality](#advanced-usage)
    - [Filtering](#filtering)
- [Customization](#customization)
    - [Custom Serialization](#custom-serialization)
    - [Excluding Attributes and Relationships](#excluding-attrs-rels)
    - [HTTP Decorators](#http-decorators)
    - [API Methods](#api-methods)
    - [Custom Swagger](#custom-swagger)
    - [Classes Without SQLAlchemy Models](#Classes-Without-Models)

<a class="mk-toclify" id="overview"></a>
## Introduction

SAFRS exposes SQLAlchemy database models as a [JSON:API](https://jsonapi.org) webservice and generates the corresponding [swagger/OpenAPI](https://swagger.io/).

Documentation can be found in the [wiki](https://github.com/thomaxxl/safrs/wiki).

__We created a service to generate APIs using GenAI__, check it out [HERE](https://apifabric.ai)!

__A [LIVE DEMO](https://thomaxxl.pythonanywhere.com) is available__, where much of the basic functionality is implemented using a simple [example](examples/demo_pythonanywhere_com.py).


<a class="mk-toclify" id="installation"></a>
## Installation

SAFRS can be installed as a [pip package](https://pypi.python.org/pypi/safrs/) or by downloading the latest version from github, for example:

```bash
git clone https://github.com/thomaxxl/safrs
cd safrs
pip install .
```

Once the dependencies are installed, the [examples](examples) can be started, for example
```
python examples/demo_relationship.py "your-interface-ip"
```

<a class="mk-toclify" id="http-methods"></a>
## JSON:API Interface

Exposed resource objects can be queried using the [JSON:API format](http://jsonapi.org/format/). The API supports following HTTP operations:

- GET : Retrieve an object or a list of objects
- PATCH : Update an object.
- DELETE: Delete an object.
- POST : Create an object. 

Please check the [JSON:API spec](http://jsonapi.org/format/) for more implementation details. 
You can also try out the interface in the [live demo](http://thomaxxl.pythonanywhere.com/api).

<a class="mk-toclify" id="resource-objects"></a>
## Resource Objects

Database objects are implemented as subclasses of the SAFRSBase and SQLAlchemy model classes. The SQLAlchemy columns are serialized to JSON when the corresponding REST API is invoked. 

Following example [app](examples/demo_relationship.py) illustrates how the API is built and documented:

```python
class User(SAFRSBase, db.Model):
    """
        description: User description
    """

    __tablename__ = "Users"
    id = db.Column(db.Integer, primary_key=True)
    name = db.Column(db.String)
    email = db.Column(db.String)

```

The User class is implemented as a subclass of 
- db.Model: SQLAlchemy base
- SAFRSBase: Implements JSON serialization for the object and generates (swagger) API documentation

This User object is then exposed through the web interface using the Api object

```python 
api.expose_object(User)
``` 

The User object REST methods are available on /User, the swagger schema is available on /api/swagger.json and the UI is available on /api/:
![User Swagger](docs/images/USER_swagger.png)


<a class="mk-toclify" id="relationships"></a>
## Relationships

Database object such as the User class from the demo.py example can be extended to include relationships with other objects. The demo_relationship.py contains following extension of the User class where a relationship with the Book class is implemented:

```python
class User(SAFRSBase, db.Model):
    '''
        description: User description
    '''
    __tablename__ = 'Users'
    id = db.Column(db.String, primary_key=True)
    name = db.Column(db.String, default='')
    email = db.Column(db.String, default='')
    books = db.relationship('Book', back_populates="user")
...
``` 

A many-to-one database association is declared by the back_populates relationship argument.
The Book class is simply another subclass of SAFRSBase and db.Model, similar to the previous User class:

```python
class Book(SAFRSBase, db.Model):
    '''
        description: Book description
    '''
    __tablename__ = 'Books'
    id = db.Column(db.String, primary_key=True)
    name = db.Column(db.String, default='')
    user_id = db.Column(db.String, db.ForeignKey('Users.id'))
    user = db.relationship('User', back_populates='books')
```

The User.book relationship can be queried in the API through the following endpoints:
![Relations Swagger](docs/images/Relations_swagger.png)

- POST adds an item to the relationship
- DELETE removes an item from the relationship
- GET retrieves a list of item ids

The relationship API endpoints work similarly for one-to-many relationships.

Relationship members can also be included in the response when querying an instance, by specifying the relationship names as a comma separated list in the `include` query argument.

![relationship include swagger](docs/images/rel_include.PNG)

For example, to retrieve all items in the `books_read` relationship from the People endpoint, you may add the `include=books_read` url parameter


http://thomaxxl.pythonanywhere.com/api/People/?include=books_read

To retrieve nested relationship items, you can specify the nested relationship name after the '.', to retrieve the authors of the books_read instances for instance, you can use

http://thomaxxl.pythonanywhere.com/api/People/?include=books_read.author

<a class="mk-toclify" id="methods"></a>
## Methods

<a class="mk-toclify" id="custom-methods"></a>
### Custom Methods

Safrs allows the user to implement custom methods on the exposed objects. This methods can be invoked through the json API by sending an HTTP POST request to the method endpoint
The following example implements a "send_mail" method fro example:

```python
class User(SAFRSBase, db.Model):
    '''
        description: User description
    '''
    __tablename__ = 'Users'
    id = Column(String, primary_key=True)
    name = Column(String, default='')
    email = Column(String, default='')

    # Following method is exposed through the REST API 
    # This means it can be invoked with a HTTP POST
    @jsonapi_rpc(http_methods=['POST','GET'])
    def send_mail(self, email):
        '''
            description : Send an email
            args:
                email:
                    type : string 
                    example : test email
        '''
        content = 'Mail to {} : {}\n'.format(self.name, email)
        return { 'result' : 'sent {}'.format(content)}

```

This method shows up in the swagger interface:

![Method Swagger](docs/images/method_swagger.PNG)

The ```send_mail``` method is documented with the ```jsonapi_rpc``` decorator. 
This decorator generates a schema based on the function documentation. This documentation contains yaml specification of the API which is used by the swagger UI. 

[api_methods.py](safrs/api_methods.py) contains a couple of methods that can be used in your models.

The yaml specification has to be in the first part of the function and class comments. These parts are delimited by four dashes ("----") . The rest of the comment may contain additional documentation.

<a class="mk-toclify" id="class-methods"></a>
### Class Methods

Two class-level methods have been defined to facilitate object retrieval:

* **lookup** : retrieve a list of objects that match the argument list. For example, following HTTP POST request to a container will retrieve a list of itemswhere the name is "thomas"
```json
{
  "method": "lookup",
  "args": {
    "name": "thomas"
  }
}
```

* **get_list** : retrieve a list of the items with the specified ID's

<a class="mk-toclify" id="initialization"></a>
## Application Initialization
The API can be initialized like this:
```python
api = SafrsApi(app, host=HOST, port=PORT, prefix=API_PREFIX)
```
Then you can expose objects with `expose_object`
```python
api.expose_object(User)    
```

An example that uses the flask app factory pattern is implement in [examples/mini_app.py](examples/mini_app.py)

<a class="mk-toclify" id="endpoint-naming"></a>
## Endpoint Naming
As can be seen in the swagger UI:
- the endpoint collection path names are the SQLAlchemy \_\_tablename\_\_ properties (e.g. /Users )
- the parameter names are derived from the SAFRSBase class names (e.g. {UserId} )
- the the relationship names are the SAFRSBase class relationship names (e.g /books )
The URL path format is [configurable](#configuration)

<a class="mk-toclify" id="configuration"></a>
## Configuration

Some configuration parameters can be set in [config.py](safrs/config.py):
- USE_API_METHODS: set this to false in case you want to disable the `jsonapi_rpc` functionality
- INSTANCE_URL_FMT: This parameter declares the instance url path format
- RELATIONSHIP_URL_FMT: This parameter declares the relationship endpoint path format

<a class="mk-toclify" id="expose-existing"></a>
## Exposing Existing Databases
Safrs allows you to Expose existing databases as jsona:api services with the [expose_existing.py](expose_existing/expose_existing.py) script, for example:

```bash
python3 expose_existing.py mysql+pymysql://root:password@localhost/sakila --host localhost
```

This script uses sqlacodegen to generate a source file containing the SQLAlchemy and `SAFRSBase` database models and starts the API webservice.

More details [here](docs/ExposeDB.md). This approach is used by the [ApiLogicServer](https://github.com/valhuber/ApiLogicServer) project.

<a class="mk-toclify" id="more-examples-and-use-cases"></a>
## More Examples and Use Cases
The [examples](examples) folder contains more example scripts:
- Using a sha hash as primary key (id)
- CORS usage
- Flask-Admin integration example, eg.:
![demo](docs/images/flask-admin.png)

A docker image can be found here:
[https://github.com/thomaxxl/safrs-example](https://github.com/thomaxxl/safrs-example)

<a class="mk-toclify" id="advanced-usage"></a>
## Advanced Functionality

<a class="mk-toclify" id="filtering"></a>
### Filtering
The swagger shows the jsonapi filters that can be used in the url query arguments. Items with an exact match of the specified attribute value can be fetched by specifying the corresponding key-value query parameter. For example, suppose the `User` class, exposed at `/Users` has a `name` attribute, to retrieve all instances with the name "John", you can use a `GET` request to 
`/Users?filter[name]=John`.

It is also possible to use more generic filters by specifiying a JSON string, for example `filter=[{"name":"timestamp","op":"gt","val":"2020-08-01"},{"name":"timestamp","op":"lt","val":"2020-08-02"}]`.

More info can be found in the [wiki](https://github.com/thomaxxl/safrs/wiki/API-Functionality#filtering).

<a class="mk-toclify" id="custom-serialization"></a>
### Custom Serialization
Serialization and deserialization are implemented by the SAFRSBase `to_dict` and `__init__` : you can extend these methods as usual.
For example, if you would like to add some attributes to the json payload of the User object, you can override the to_dict method:

```python
class User(SAFRSBase, db.Model):
    '''
        description: User description
    '''
    __tablename__ = 'Users'
    id = db.Column(db.String, primary_key=True)
    name = db.Column(db.String, default='')
    email = db.Column(db.String, default='')
    books = db.relationship('Book', back_populates="user")

    def to_dict(self):
        result = SAFRSBase.to_dict(self)
        result['custom_field'] = 'custom'
        return result
``` 

This will add the `custom_field` attribute to the result attributes:
```json
"attributes": {
    "custom_field": "custom",
    "email": "reader_email0",
    "name": "Reader 0"
}
```

<a class="mk-toclify" id="customization"></a>
## Customization

<a class="mk-toclify" id="excluding-attrs-rels"></a>
### Excluding Attributes and Relationships
It is possible to specify attributes and relationships that should not be serialized by specifying the respective `exclude_attrs` and èxclude_rels` class attributes in your SAFRSBase instances.
Examples can be found [here](examples/demo_pythonanywhere_com.py#L81) and [here](examples/demo_http_get.py#L21)

<a class="mk-toclify" id="limit-http-verbs"></a>
### Limiting HTTP Methods
It is possible to limit the HTTP methods that are allowed by overriding the `http_methods` class attribute. An example can be found [here](examples/demo_http_get.py#L20)

<a class="mk-toclify" id="HTTP-decorators"></a>
### HTTP Decorators
The `decorators` class attribute list can be used to add custom decorators to the HTTP endpoints. An example of this functionality is implemented
in the [authentication examples](examples/authentication).


<a class="mk-toclify" id="api-methods"></a>
### API Methods
Some additional API RPC methods are implemented in [api_methods.py](safrs/api_methods.py), e.g. mysql regex search. 

<a class="mk-toclify" id="custom-swagger"></a>
### Custom swagger
The swagger schema can be merged with a modified schema dictionary by supplying the to-be-merged dictionary as the `custom_swagger` argument to `SafrsApi`, e.g.
```python
custom_swagger = {"info": {"title" : "New Title" }} # Customized swagger title will be merged
api = SafrsApi(app, host=swagger_host, port=PORT, prefix=OAS_PREFIX, api_spec_url=OAS_PREFIX+'/swagger',
               custom_swagger=custom_swagger, schemes=['http', 'https'], description=description)
```

<a class="mk-toclify" id="Classes-Without-Models"></a>
### Classes Without SQLAlchemy Models
You can implement a serializable class without a model but this requires some extra work because safrs needs to know which attributes and relationships to serialize. An example is implemented [here](examples/demo_stateless.py)

### More Customization
The documentation is being moved to the [wiki](https://github.com/thomaxxl/safrs/wiki)

<details>
<summary>About</summary>
SAFRS is an acronym for **S**ql**A**lchemy **F**lask-**R**estful **S**wagger. The purpose of this framework is to help python developers create a self-documenting JSON API for sqlalchemy database objects and relationships. These objects can be serialized to JSON and can be created, retrieved, updated and deleted through the JSON API. 
Optionally, custom resource object methods can be exposed and invoked using JSON.
Class and method descriptions and examples can be provided in yaml syntax in the code comments. The description is parsed and shown in the swagger web interface. 

The result is an easy-to-use [swagger/OpenAPI](https://swagger.io/) and [JSON:API](https://jsonapi.org) compliant API implementation.
</details>

<details>
<summary>limitations & Todos</summary>
This code was developed for a specific use-case and may not be flexible enough for everyone's needs. A lot of the functionality is available but not documented for the sake of brevity.
Performance is reasonable for regular databases, but once you start exposing really big tables you may run into problems, for example: the `count()` for mysql innodb is slow on large(1M rows) tables, a workaround can be implemented by querying the `sys` tables or using werkzeug caching. 
Feel free to open an issue or drop [me](mailto:thomas.pollet+no+spam+@gmail.com) an email if you run into problems or something isn't clear!
</details>

<details>
<summary>References</summary>

- [JSON:API specification](http://jsonapi.org/format/)
- [OpenApi (Swagger)](https://www.openapis.org/)
- [Flask](http://flask.pocoo.org/)
- [SQLAlchemy](https://www.sqlalchemy.org/)
</details>

<details>
<summary>Thanks</summary>
I developed this code when I worked at [Excellium Services](https://www.excellium-services.com/). They allowed me to open source it when I stopped working there.
</details>
 readmeEtag: '"b5c8f46ac4d3c27b2f0ba4ba92bebc5d8b11b6d0"' readmeLastModified: Fri, 11 Oct 2024 06:41:22 GMT repositoryId: 104184487 description: SqlAlchemy Flask-Restful Swagger Json:API OpenAPI created: '2017-09-20T07:56:55Z' updated: '2025-10-27T07:48:10Z' language: Python archived: false stars: 413 watchers: 12 forks: 73 owner: thomaxxl logo: https://avatars.githubusercontent.com/u/4190750?v=4 license: GPL-3.0 repoEtag: '"5f40e895e301cf7f91679f78292b6ff5bbb122e2ac57cafcc61d2d93bec5640d"' repoLastModified: Mon, 27 Oct 2025 07:48:10 GMT foundInMaster: true category: Server Implementations id: 1ccc4d969c768f04576d518679fb6489 - source: openapi3 tags repository: https://github.com/cloudflare/chanfana v3: true id: b38a05ac4acb961910fe6e4ad445ba6a repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL2NoYW5mYW5hLnBhZ2VzLmRldi8iPgogICAgPGltZyBzcmM9Imh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9jbG91ZGZsYXJlL2NoYW5mYW5hL3JlZnMvaGVhZHMvbWFpbi9kb2NzL3B1YmxpYy9hc3NldHMvbG9nby5wbmciIHdpZHRoPSI1MDAiIGhlaWdodD0iYXV0byIgYWx0PSJjaGFuZmFuYSIvPgogIDwvYT4KPC9kaXY+CgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8ZW0+T3BlbkFQSSAzIGFuZCAzLjEgc2NoZW1hIGdlbmVyYXRvciBhbmQgdmFsaWRhdG9yIGZvciA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vaG9ub2pzL2hvbm8iIHRhcmdldD0iX2JsYW5rIj5Ib25vPC9hPiwgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2t3aGl0bGV5L2l0dHktcm91dGVyIiB0YXJnZXQ9Il9ibGFuayI+aXR0eS1yb3V0ZXI8L2E+IGFuZCBtb3JlITwvZW0+CjwvcD4KCjxociAvPgoKKipEb2N1bWVudGF0aW9uKio6IDxhIGhyZWY9Imh0dHBzOi8vY2hhbmZhbmEucGFnZXMuZGV2LyI+Y2hhbmZhbmEucGFnZXMuZGV2PC9hPgoKKipTb3VyY2UgQ29kZSoqOiA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vY2xvdWRmbGFyZS9jaGFuZmFuYS8iPmdpdGh1Yi5jb20vY2xvdWRmbGFyZS9jaGFuZmFuYTwvYT4KCjxociAvPgoKW2NoYW5mYW5hXShodHRwczovL2dpdGh1Yi5jb20vY2xvdWRmbGFyZS9jaGFuZmFuYSkgKioocHJldmlvdXNseSBrbm93biBhcyBpdHR5LXJvdXRlci1vcGVuYXBpKSoqIGlzIGEgbGlicmFyeSB0aGF0IGFkZHMKT3BlbkFQSSBzY2hlbWEgZ2VuZXJhdGlvbiBhbmQgdmFsaWRhdGlvbiB0byBhbnkgcm91dGVyICg8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vaG9ub2pzL2hvbm8iIHRhcmdldD0iX2JsYW5rIj4KSG9ubzwvYT4sIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9rd2hpdGxleS9pdHR5LXJvdXRlciIgdGFyZ2V0PSJfYmxhbmsiPml0dHktcm91dGVyPC9hPiwgZXRjKSwgbWVhbnQgdG8gYmUgYQpwb3dlcmZ1bCBhbmQgbGlnaHR3ZWlnaHQKbGlicmFyeSBmb3IgQ2xvdWRmbGFyZSBXb3JrZXJzIGJ1dCBydW5zIG9uIGFueSBydW50aW1lIHN1cHBvcnRlZCBieSB0aGUgYmFzZSByb3V0ZXIuCgpUaGUga2V5IGZlYXR1cmVzIGFyZToKCi0gT3BlbkFQSSAzIGFuZCAzLjEgc2NoZW1hIGdlbmVyYXRvciBhbmQgdmFsaWRhdG9yCi0gRnVsbHkgd3JpdHRlbiBpbiB0eXBlc2NyaXB0Ci0gW0NsYXNzLWJhc2VkIGVuZHBvaW50c10oaHR0cHM6Ly9jaGFuZmFuYS5wYWdlcy5kZXYvZW5kcG9pbnRzL2RlZmluaW5nLWVuZHBvaW50cykKLSBRdWVyeSwgUGF0aCwgSGVhZGVycywgYW5kIEJvZHkgdHlwZXNjcmlwdCBpbmZlcmVuY2UsIHNlZSBbUGFyYW1ldGVyc10oaHR0cHM6Ly9jaGFuZmFuYS5wYWdlcy5kZXYvZW5kcG9pbnRzL3BhcmFtZXRlcnMpIGFuZCBbUmVxdWVzdCBWYWxpZGF0aW9uXShodHRwczovL2NoYW5mYW5hLnBhZ2VzLmRldi9lbmRwb2ludHMvcmVxdWVzdC12YWxpZGF0aW9uKQotIEV4dGVuZCBleGlzdGluZyBbSG9ub10oaHR0cHM6Ly9jaGFuZmFuYS5wYWdlcy5kZXYvcm91dGVyLWFkYXB0ZXJzKSwgW2l0dHktcm91dGVyXShodHRwczovL2NoYW5mYW5hLnBhZ2VzLmRldi9yb3V0ZXItYWRhcHRlcnMpLCBldGMgYXBwbGljYXRpb24sIHdpdGhvdXQgdG91Y2hpbmcgb2xkIHJvdXRlcwotIENMSSB0byBleHRyYWN0IHRoZSBzY2hlbWEuanNvbiBmcm9tIHlvdXIgY29kZSAobnB4IGNoYW5mYW5hKSAtIHNlZSBbQ0xJIERvY3VtZW50YXRpb25dKGh0dHBzOi8vY2hhbmZhbmEucGFnZXMuZGV2L2NsaSkKCiMjIEdldHRpbmcgc3RhcnRlZAoKR2V0IHN0YXJ0ZWQgd2l0aCBhIHRlbXBsYXRlIHdpdGggdGhpcyBjb21tYW5kOgoKYGBgYmFzaApucG0gY3JlYXRlIGNsb3VkZmxhcmVAbGF0ZXN0IC0tIC0tdHlwZSBvcGVuYXBpCmBgYAoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCm5wbSBpIGNoYW5mYW5hIC0tc2F2ZQpgYGAKCiMjIENvbW1hbmQtTGluZSBJbnRlcmZhY2UgKENMSSkKCkNoYW5mYW5hIGluY2x1ZGVzIGEgQ0xJIHRvb2wgdG8gaGVscCB5b3UgZXh0cmFjdCB0aGUgT3BlbkFQSSBzY2hlbWEgZnJvbSB5b3VyIENsb3VkZmxhcmUgV29ya2VyIHByb2plY3QuIFRoaXMgaXMgdXNlZnVsIGZvciBnZW5lcmF0aW5nIGEgc3RhdGljIHNjaGVtYSBmaWxlLCBmb3IgZXhhbXBsZSwgaW4gQ0kvQ0QgcGlwZWxpbmVzLgoKVGhlIGJhc2ljIGNvbW1hbmQgaXM6CmBgYGJhc2gKbnB4IGNoYW5mYW5hCmBgYApUaGlzIHdpbGwgc3RhcnQgeW91ciB3b3JrZXIgdXNpbmcgYHdyYW5nbGVyIGRldmAsIGZldGNoIHRoZSBzY2hlbWEgZnJvbSBgL29wZW5hcGkuanNvbmAsIGFuZCBzYXZlIGl0IHRvIGBzY2hlbWEuanNvbmAuCgpGb3IgbW9yZSBkZXRhaWxzIG9uIG9wdGlvbnMgYW5kIHVzYWdlLCBwbGVhc2UgcmVmZXIgdG8gdGhlIFsqKkNMSSBEb2N1bWVudGF0aW9uKipdKGh0dHBzOi8vY2hhbmZhbmEucGFnZXMuZGV2L2NsaSkuCgojIyBNaW5pbWFsIEhvbm8gRXhhbXBsZQoKYGBgdHMKaW1wb3J0IHsgZnJvbUhvbm8sIE9wZW5BUElSb3V0ZSB9IGZyb20gJ2NoYW5mYW5hJwppbXBvcnQgeyBIb25vIH0gZnJvbSAnaG9ubycKaW1wb3J0IHsgeiB9IGZyb20gJ3pvZCcKCmV4cG9ydCB0eXBlIEVudiA9IHsKICAgIC8vIEV4YW1wbGUgYmluZGluZ3MKICAgIERCOiBEMURhdGFiYXNlCiAgICBCVUNLRVQ6IFIyQnVja2V0Cn0KZXhwb3J0IHR5cGUgQXBwQ29udGV4dCA9IENvbnRleHQ8eyBCaW5kaW5nczogRW52IH0+CgpleHBvcnQgY2xhc3MgR2V0UGFnZU51bWJlciBleHRlbmRzIE9wZW5BUElSb3V0ZSB7CiAgc2NoZW1hID0gewogICAgcmVxdWVzdDogewogICAgICBwYXJhbXM6IHoub2JqZWN0KHsKICAgICAgICBpZDogei5zdHJpbmcoKS5taW4oMikubWF4KDEwKSwKICAgICAgfSksCiAgICAgIHF1ZXJ5OiB6Lm9iamVjdCh7CiAgICAgICAgcGFnZTogei5udW1iZXIoKS5pbnQoKS5taW4oMCkubWF4KDIwKSwKICAgICAgfSksCiAgICB9LAogIH0KCiAgYXN5bmMgaGFuZGxlKGM6IEFwcENvbnRleHQpIHsKICAgIGNvbnN0IGRhdGEgPSBhd2FpdCB0aGlzLmdldFZhbGlkYXRlZERhdGE8dHlwZW9mIHRoaXMuc2NoZW1hPigpCgogICAgcmV0dXJuIGMuanNvbih7CiAgICAgIGlkOiBkYXRhLnBhcmFtcy5pZCwKICAgICAgcGFnZTogZGF0YS5xdWVyeS5wYWdlLAogICAgfSkKICB9Cn0KCi8vIFN0YXJ0IGEgSG9ubyBhcHAKY29uc3QgYXBwID0gbmV3IEhvbm88eyBCaW5kaW5nczogRW52IH0+KCkKCi8vIFNldHVwIE9wZW5BUEkgcmVnaXN0cnkKY29uc3Qgb3BlbmFwaSA9IGZyb21Ib25vKGFwcCkKCi8vIFJlZ2lzdGVyIE9wZW5BUEkgZW5kcG9pbnRzICh0aGlzIHdpbGwgYWxzbyByZWdpc3RlciB0aGUgcm91dGVzIGluIEhvbm8pCm9wZW5hcGkuZ2V0KCcvZW50cnkvOmlkJywgR2V0UGFnZU51bWJlcikKCi8vIEV4cG9ydCB0aGUgSG9ubyBhcHAKZXhwb3J0IGRlZmF1bHQgYXBwCmBgYAoKIyMgRmVlZGJhY2sgYW5kIGNvbnRyaWJ1dGlvbnMKCltjaGFuZmFuYV0oaHR0cHM6Ly9naXRodWIuY29tL2Nsb3VkZmxhcmUvY2hhbmZhbmEpIGFpbXMgdG8gYmUgYXQgdGhlIGNvcmUgb2YgbmV3IEFQSXMgYnVpbHQgdXNpbmcKV29ya2VycyBhbmQgZGVmaW5lIGEgcGF0dGVybiB0byBhbGxvdyBldmVyeW9uZSB0bwpoYXZlIGFuIE9wZW5BUEktY29tcGxpYW50IHNjaGVtYSB3aXRob3V0IHdvcnJ5aW5nIGFib3V0IGltcGxlbWVudGF0aW9uIGRldGFpbHMgb3IgcmVpbnZlbnRpbmcgdGhlIHdoZWVsLgoKY2hhbmZhbmEgaXMgY29uc2lkZXJlZCBzdGFibGUgYW5kIHByb2R1Y3Rpb24gcmVhZHkgYW5kIGlzIGJlaW5nIHVzZWQgd2l0aAp0aGUgW1JhZGFyIDIuMCBwdWJsaWMgQVBJXShodHRwczovL2RldmVsb3BlcnMuY2xvdWRmbGFyZS5jb20vcmFkYXIvKSBhbmQgbWFueSBvdGhlciBDbG91ZGZsYXJlIHByb2R1Y3RzLgoKWW91IGNhbiBhbHNvIHRhbGsgdG8gdXMgaW4gdGhlIFtDbG91ZGZsYXJlIENvbW11bml0eV0oaHR0cHM6Ly9jb21tdW5pdHkuY2xvdWRmbGFyZS5jb20vKSBvcgp0aGUgW1JhZGFyIERpc2NvcmQgQ2hhbm5lbF0oaHR0cHM6Ly9kaXNjb3JkLmNvbS9jaGFubmVscy81OTUzMTc5OTAxOTEzOTg5MzMvMTAzNTU1MzcwNzExNjQ3ODQ5NSkK readmeEtag: '"c7b6103098b7ea1122aac907f29b3cb6b4227b50"' readmeLastModified: Wed, 18 Jun 2025 18:26:17 GMT repositoryId: 563328514 description: >- OpenAPI 3 and 3.1 schema generator and validator for Hono, itty-router and more! created: '2022-11-08T11:43:41Z' updated: '2026-02-06T03:08:18Z' language: TypeScript archived: false stars: 688 watchers: 9 forks: 66 owner: cloudflare logo: https://avatars.githubusercontent.com/u/314135?v=4 license: MIT repoEtag: '"657131d1418990021d3d868cc6b3edadf872df44ea125e2a6c11713d7cb24918"' repoLastModified: Fri, 06 Feb 2026 03:08:18 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/cloudflare/itty-router-openapi - source: openapi3 tags repository: https://github.com/dgarcia360/openapi-boilerplate v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgogIDxkaXYgc3R5bGU9ImRpc3BsYXk6aW5saW5lLWJsb2NrO3ZlcnRpY2FsLWFsaWduOiBtaWRkbGU7Ij4KICAgICAgPGltZyBzcmM9ImRvY3MvaGVhZGVyLnBuZyIgd2lkdGg9IjU1MCIvPgogIDwvZGl2Pgo8L2gxPgoKWyFbT3BlbkFQSSBsaW50ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9kZ2FyY2lhMzYwL29wZW5hcGktYm9pbGVycGxhdGUvYWN0aW9ucy93b3JrZmxvd3MvbGludGVyLnlhbWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2RnYXJjaWEzNjAvb3BlbmFwaS1ib2lsZXJwbGF0ZS9hY3Rpb25zL3dvcmtmbG93cy9saW50ZXIueWFtbCkKWyFbTGljZW5zZTogTUlUXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtTUlULXllbGxvdy5zdmcpXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkKCiMgT3BlbkFQSSBib2lsZXJwbGF0ZQoKQSBzdGFydGVyIHRlbXBsYXRlIGZvciAqKk9wZW5BUEkgU3BlY2lmaWNhdGlvbioqIChPQVMpIHByb2plY3RzLgoKVGhpcyBwcm9qZWN0IHNwbGl0cyB0aGUgW1N3YWdnZXIgUGV0c3RvcmVdKGh0dHBzOi8vcGV0c3RvcmUuc3dhZ2dlci5pby8pIGV4YW1wbGUgZnJvbSB0aGUgb2ZmaWNpYWwgZG9jdW1lbnRhdGlvbiBpbnRvIHNtYWxsZXIgZmlsZXMuIEl0IGFsc28gYWRkcyBoYW5keSBjb21tYW5kcyB0byBidWlsZCwgbGludCwgYW5kIHByZXZpZXcgdGhlIE9wZW5BUEkgZG9jdW1lbnQgZnJvbSB0aGUgdGVybWluYWwuCgpZb3UgY2FuIHVzZSB0aGlzIHRlbXBsYXRlIHRvIGd1aWRlIHRoZSBvcmdhbml6YXRpb24gb2YgeW91ciBwcm9qZWN0LiBFaXRoZXIgaWYgeW91IHdhbnQgdG8gY3JlYXRlIGEgbmV3IE9wZW5BUEkgZG9jdW1lbnQgZnJvbSBzY3JhdGNoIG9yIGlmIHlvdSBhbHJlYWR5IGhhdmUgaXQgZGVmaW5lZC4gCgojIyBGZWF0dXJlcwoKKiDwn5OdIFdyaXRlIE9wZW5BUEkgZGVmaW5pdGlvbnMgaW4gZGlmZmVyZW50IGZpbGVzLgoqIPCflIAgQ29tYmluZSBhbGwgZmlsZXMgd2l0aCBbcmVkb2NseS1jbGldKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jbHktY2xpKS4KKiDinIUgVmFsaWRhdGUgYW5kIGxpbnQgdGhlIE9wZW5BUEkgZG9jdW1lbnQgd2l0aCBbc3RvcGxpZ2h0aW8vc3BlY3RyYWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9wbGlnaHRpby9zcGVjdHJhbCkuCiog4pyoIFB1Ymxpc2ggcmVmZXJlbmNlIGRvY3Mgd2l0aCBbUmVkb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKSAmIEdpdEh1YiBQYWdlcy4KCiMjIFdoeT8KCldoZW4gSSB1c2VkIHRvIGRvY3VtZW50IEFQSXMgZm9sbG93aW5nIHRoZSBPcGVuQVBJIHNwZWMsIEkgYWx3YXlzIGVuZGVkIHVwIHdpdGggZG9jdW1lbnRzIG9mIHRob3VzYW5kcyBvZiBsaW5lcywgd2hpY2ggd2VyZSBhIG5pZ2h0bWFyZSB0byBtYWludGFpbi4KCkZvciB0aGlzIHJlYXNvbiwgSSBleHBsb3JlZCBob3cgdG8gc3BsaXQgT3BlbkFQSSBkb2N1bWVudHMuIEp1bXAgb3ZlciB0byBteSBbYmxvZ10oaHR0cHM6Ly9kYXZpZGdhcmNpYS5kZXYvcG9zdHMvaG93LXRvLXNwbGl0LW9wZW4tYXBpLXNwZWMtaW50by1tdWx0aXBsZS1maWxlcy8pIHRvIGxlYXJuIG1vcmUgYWJvdXQgdGhlIHByb2Nlc3MuIEJhc2VkIG9uIG15IHJlc2VhcmNoLCBJIGNyZWF0ZWQgdGhpcyBvcGluaW9uYXRlZCB0ZW1wbGF0ZSB0byBkZWZpbmUsIHRlc3QsIGFuZCBwdWJsaXNoIG1vZHVsYXIgT3BlbkFQSSBwcm9qZWN0cy4KCk9yZ2FuaXppbmcgeW91ciBwcm9qZWN0IGVmZmVjdGl2ZWx5IG9mZmVycyBzaWduaWZpY2FudCBiZW5lZml0cy4gU3BsaXR0aW5nIGEgbGFyZ2UgT3BlbkFQSSBzcGVjIGludG8gbXVsdGlwbGUgZmlsZXMgbWFrZXMgaXQgZWFzaWVyIHRvIG1hbmFnZSBhbmQgbWFpbnRhaW4uIEl0IGFsc28gbWFrZXMgdGhlIGRvY3VtZW50YXRpb24gcHJvY2VzcyBtb3JlIGVmZmljaWVudCBhbmQgbGVzcyBvdmVyd2hlbG1pbmcuIEkndmUgZm91bmQgdGhhdCB3aXRoIHRoaXMgc3RydWN0dXJlLCBvdGhlciBkZXZzIGFyZSBtb3JlIGluY2xpbmVkIHRvIGNvbnRyaWJ1dGUgYW5kIHN1Z2dlc3QgY2hhbmdlcyB0byB0aGUgZG9jdW1lbnRhdGlvbiwgYXMgaXQgZmVlbHMgbW9yZSBhcHByb2FjaGFibGUgYW5kIGxlc3MgaW50aW1pZGF0aW5nLgoKIyMgR2V0dGluZyBzdGFydGVkCgojIyMgUmVxdWlyZW1lbnRzCgoqIE5vZGUuanMgMTYgKGN1cnJlbnQpCgojIyMgSW5zdGFsbGF0aW9uCgoxLiBDbG9uZSB0aGUgcmVwb3NpdG9yeS4KCiAgICBgYGAKICAgIGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vZGdhcmNpYTM2MC9vcGVuYXBpLWJvaWxlcnBsYXRlLmdpdAogICAgYGBgCgoyLiBJbnN0YWxsIHRoZSBwcm9qZWN0IGRlcGVuZGVuY2llcy4KCiAgICBgYGAKICAgIG5wbSBpbnN0YWxsCiAgICBgYGAKCjMuIEVkaXQgYG9wZW5hcGkueWFtbGAgdG8gZml0IHlvdXIgQVBJIGRlZmluaXRpb24uIElmIHlvdeKAmXJlIG5vdCBmYW1pbGlhciB3aXRoIHRoZSBPcGVuQVBJIFNwZWNpZmljYXRpb24sIHJlYWQgW0dldHRpbmcgc3RhcnRlZCB3aXRoIE9BU10oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NvbHV0aW9ucy9nZXR0aW5nLXN0YXJ0ZWQtd2l0aC1vYXMvKSBmaXJzdC4KCiMjIFVzZWZ1bCBjb21tYW5kcwoKVGhlIHByb2plY3Qgd2lsbCBidWlsZCwgbGludCwgYW5kIHByZXZpZXcgdGhlIE9wZW5BUEkgZG9jdW1lbnQgZnJvbSB0aGUgdGVybWluYWwsIHdpdGggdGhlIGZvbGxvd2luZyBjb21tYW5kczoKCiMjIyBCdWlsZAoKVGhlIGNvbW1hbmQgYnVuZGxlcyB0aGUgc3BlYyBhcyBvbmUgYC55YW1sYCBmaWxlLgoKYGBgCm5wbSBydW4gYnVpbGQKYGBgCgpUaGUgbWluaWZpZWQgZG9jdW1lbnQgaXMgc3RvcmVkIGluIGBfYnVpbGQvb3BlbmFwaS55YW1sYC4KCiMjIyBUZXN0CgpUaGUgY29tbWFuZCBjaGVja3MgaWYgdGhlIGRvY3VtZW50IGZvbGxvd3MgdGhlIE9wZW5BUEkgMy4wIFNwZWNpZmljYXRpb24uCgpgYGAKbnBtIHJ1biB0ZXN0CmBgYAoKIyMjIFByZXZpZXcKClRoZSBjb21tYW5kIGJ1aWxkcyBhIGRvY3Mgc2l0ZSBzbyB0aGF0IHlvdSBjYW4gdmlldyB0aGUgcmVuZGVyaW5nIG9uIHlvdXIgbG9jYWwgYnJvd3Nlci4KCmBgYApucG0gcnVuIHByZXZpZXcKYGBgCgpUaGUgc2VydmVyIHN0YXJ0cyBvbiBodHRwOi8vMTI3LjAuMC4xOjgwODAuCgpUaGUgc2l0ZSBpcyBnZW5lcmF0ZWQgd2l0aCBbUmVEb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKS4KSGVyZSdzIGEgcHJldmlldyBvZiBhIHNpdGUgZ2VuZXJhdGVkIHdpdGggdGhpcyBjb21tYW5kOiBbU3dhZ2dlciBQZXRzdG9yZSBSZWZlcmVuY2UgRG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9kZ2FyY2lhMzYwLmdpdGh1Yi5pby9vcGVuYXBpLWJvaWxlcnBsYXRlLykuCgojIyBSZWFkeS10by11c2Ugd29ya2Zsb3dzCgpUaGUgcHJvamVjdCB1c2VzIFtHaXRIdWIgQWN0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL2ZlYXR1cmVzL2FjdGlvbnMpIGZvciBDb250aW51b3VzIEludGVncmF0aW9uIChDSSkuCgpPbiBldmVyeSBuZXcgcHVsbCByZXF1ZXN0LCB0aGUgT3BlbkFQSSBkb2N1bWVudCBpcyBsaW50ZWQgd2l0aCBbc3BlY3RyYWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9wbGlnaHRpby9zcGVjdHJhbCkuIElmIHRoZXJlIGFyZSBjaGFuZ2VzIHRoYXQgaW50cm9kdWNlIGVycm9ycywgdGhlIGJvdCB3aWxsIGhpZ2hsaWdodCB0aGVtIHJlcGx5aW5nIHRvIHRoZSBwdWxsIHJlcXVlc3QuCgpXaGVuIHRoZSBkZWZhdWx0IGJyYW5jaCAoZS5nLiBgbWFzdGVyYCkgcmVjZWl2ZXMgYW4gdXBkYXRlLCBhIHdvcmtmbG93IGF1dG9tYXRpY2FsbHkgcHVibGlzaGVzIHRoZSBBUEkgcmVmZXJlbmNlIGRvY3VtZW50YXRpb24gc2l0ZSB0byBHaXRIdWIgUGFnZXMuCgpTZWUgYC5naXRodWIvd29ya2Zsb3dzYCB0byBjdXN0b21pemUgdGhlIGF2YWlsYWJsZSB3b3JrZmxvd3MuIElmIHlvdSBkb24ndCBwbGFuIHRvIHVzZSBHaXRIdWIgdG8gaG9zdCB5b3VyIHNwZWMgb3IgcHJlZmVyIHRvIGtlZXAgZG9jcyBwcml2YXRlLCBkZWxldGUgdGhlIGAuZ2l0aHViYCBmb2xkZXIuCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWUgYW5kIGFwcHJlY2lhdGVkISAKSWYgeW91IHdhbnQgdG8gZW5oYW5jZSB0aGUgYm9pbGVycGxhdGUsIHBsZWFzZSByZWFkIFtDT05UUklCVVRJTkcubWRdKENPTlRSSUJVVElORy5tZCkgZmlsZSBmaXJzdC4KCiMjIExpY2Vuc2UKCkNvcHlyaWdodCAoYykgMjAxOS1wcmVzZW50IERhdmlkIEdhcmNpYSAoW0BkZ2FyY2lhMzYwXShodHRwczovL2RhdmlkZ2FyY2lhLmRldikpLiBMaWNlbnNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXShMSUNFTlNFLm1kKS4KClRoZSBQZXRTdG9yZSBleGFtcGxlIHVzZWQgaXMgZGVyaXZlZCBmcm9tIFtPQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9leGFtcGxlcy92My4wL3BldHN0b3JlLnlhbWwpLCBDb3B5cmlnaHQgVGhlIExpbnV4IEZvdW5kYXRpb24sIExpY2Vuc2VkIHVuZGVyIHRoZSBbQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9MSUNFTlNFKS4K readmeEtag: '"5e63e66202cb0bcfbea0f96bf605f8a2dbe56387"' readmeLastModified: Sat, 18 Jan 2025 01:18:13 GMT repositoryId: 239108172 description: 📘 Multi-file boilerplate for Open API Specification created: '2020-02-08T10:23:15Z' updated: '2025-12-05T17:14:20Z' language: Shell archived: false stars: 399 watchers: 6 forks: 88 owner: dgarcia360 logo: https://avatars.githubusercontent.com/u/9107969?v=4 license: NOASSERTION repoEtag: '"54668576429663b1b945c73c1fe63b68025b555016cf159784817c3c98e5b223"' repoLastModified: Fri, 05 Dec 2025 17:14:20 GMT foundInMaster: true category: - Description Validators - Parsers id: a62d924d146072268eebd4abdf8d51fc - source: openapi3 tags repository: https://github.com/cdimascio/generator-express-no-stress-typescript v3: true repositoryMetadata: base64Readme: >- # generator-express-no-stress-typescript

![](https://img.shields.io/badge/status-stable-green.svg) ![](https://img.shields.io/npm/v/generator-express-no-stress-typescript.svg) [![Codacy Badge](https://api.codacy.com/project/badge/Grade/21e362094c5a4c0da4841fc172cee95d)](https://www.codacy.com/app/cdimascio/generator-express-no-stress-typescript?utm_source=github.com&utm_medium=referral&utm_content=cdimascio/generator-express-no-stress-typescript&utm_campaign=Badge_Grade) [![](https://img.shields.io/gitter/room/cdimascio-oss/community?color=%23eb205a)](https://gitter.im/cdimascio-oss/community) <!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-10-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END --> ![](https://img.shields.io/badge/license-MIT-blue.svg)
Create awesome [Express.js](http://www.expressjs.com) applications with best of breed tech including [Typescript](https://www.typescriptlang.org/), structured logging with [Pino](https://github.com/pinojs/pino), API validation and interactive documentation via an [OpenAPI 3](https://swagger.io/specification/) or [Swagger 2](https://swagger.io/specification/v2/) spec, environment based config with [dotenv](https://github.com/motdotla/dotenv).

<p align="center">
<img src="https://raw.githubusercontent.com/cdimascio/generator-express-no-stress-typescript/master/assets/express-no-stress-logo-ts.jpg"
</p>

generator-express-no-stress-typescript gets you up and running in seconds. It's ridiculously easy to configure. Heck, just take the defaults. Start it. Write code.

This generator scaffolds a fully functioning REST API server complete with interactive documentation, API validation, structured logging, environment driven config, and more. Simply run the generator and smile :-D

This project is a Typescript variant of [generator-express-no-stress](https://github.com/cdimascio/generator-express-no-stress)

[Here's what you get!](#what-you-get)

## Install

_Requires Node 6 or greater_

```shell
npm install -g yo generator-express-no-stress-typescript
```

- See [here](#usage-cli) for use with Yarn and/or Docker

## Scaffold

```shell
yo express-no-stress-typescript myapp
cd myapp
```

## Run

Run in _development mode_:

```shell
npm run dev
```

Package and run in _production mode_

```shell
npm run compile
npm start
```

## Test

```shell
npm test
```

## Debug

Run one of the following, then attach your favorite inspector.

```shell
# debug the server
npm run dev:debug

# debug the tests
npm run test:debug
```

## Try it!

- Interactive API doc at [http://localhost:3000/api-explorer](http://localhost:3000/api-explorer)
- Landing page at [http://localhost:3000](http://localhost:3000)

---

## Usage: CLI

```shell
yo express-no-stress-typescript [appname] [--yarn] [--docker]
```

| Option     | default | Description                                                                |
| ---------- | ------- | -------------------------------------------------------------------------- |
| `appname`  | myapp   | The application folder                                                     |
| `--yarn`   | -       | Use the [`yarn`](https://yarnpkg.com) package manager, instead of `npm`    |
| `--docker` |         | Install [Docker](https://www.docker.com/) artifacts including a Dockerfile |

## Usage: Project

The sections below describe all usage options available once the project is generated/scaffolded.

### npm targets

| Target               | Description                                                              |
| -------------------- | ------------------------------------------------------------------------ |
| `npm run dev`        | Run in _development_ mode                                                |
| `npm run dev:debug`  | Debug in _development_ mode                                              |
| `npm run test`       | Run tests                                                                |
| `npm run test:debug` | Debug tests                                                              |
| `npm run compile`    | Transpile source code for production use                                 |
| `npm start`          | Run the in _production_ mode. \*Requires running `npm run compile` first |

### Deploy to the Cloud

e.g. CloudFoundry

```
cf push myapp
```

### Use Yarn

```
# scaffold
yo express-no-stress-typescript myapp --yarn

# start
cd myapp
npm start
```

---

## What you get!

- [Typescript](https://www.typescriptlang.org/) - Typescript is a typed superset of JavaScript that compiles to plain JavaScript
- [Express.js](https://www.expressjs.com) - Fast, unopinionated
  , minimalist web framework for Node.js
- [Pino](https://github.com/pinojs/pino) - Extremely fast node.js logger, inspired by Bunyan. It also includes a shell utility to pretty-print its log files
- [dotenv](https://github.com/motdotla/dotenv) - Loads environment variables from .env for nodejs projects
- [Swagger](http://swagger.io/) - is a simple yet powerful representation of your RESTful API.
- [SwaggerUI](http://swagger.io/) - dynamically generate beautiful documentation and sandbox from a Swagger-compliant API

### API Validation

Simply describe your APIs with Swagger and automagically get for free:

- Interactive documentation
- API validation

#### Interactive API Doc

![](https://github.com/cdimascio/generator-express-no-stress-typescript/raw/master/assets/interactive-doc1.png)

#### API Validation!

Oops! I the API caller forgot to pass a `name` field, no stress, we've got this!

![](https://github.com/cdimascio/generator-express-no-stress-typescript/raw/master/assets/api-validation.png)

### Structured Logging

Structured logging out of the box!

#### raw

![](https://github.com/cdimascio/generator-express-no-stress-typescript/raw/master/assets/logging-raw.png)

#### pretty

Structured logging pretty printed by default - great for dev!

![](https://github.com/cdimascio/generator-express-no-stress-typescript/raw/master/assets/logging-pretty.png)

### API Validation Example

Simply describe your APIs with Swagger and automatically get:

- API request validation
- Interactive documentation

### example

#### Swagger API spec

```yaml
swagger: '2.0'
info:
  version: 1.0.0
  title: myapp
  description: My cool app
basePath: /api/v1
tags:
  - name: Examples
    description: Simple example endpoints
  - name: Specification
    description: The swagger API specification

consumes:
  - application/json
produces:
  - application/json

definitions:
  ExampleBody:
    type: object
    title: example
    required:
      - name
    properties:
      name:
        type: string
        example: no_stress

paths:
  /examples:
    get:
      tags:
        - Examples
      description: Fetch all examples
      responses:
        200:
          description: Returns all examples
    post:
      tags:
        - Examples
      description: Create a new example
      parameters:
        - name: example
          in: body
          description: an example
          required: true
          schema:
            $ref: '#/definitions/ExampleBody'
      responses:
        200:
          description: Returns all examples

  /examples/{id}:
    get:
      tags:
        - Examples
      parameters:
        - name: id
          in: path
          required: true
          description: The id of the example to retrieve
          type: integer
      responses:
        200:
          description: Return the example with the specified id
        404:
          description: Example not found

  /spec:
    get:
      tags:
        - Specification
      responses:
        200:
          description: Return the API specification
```

#### Invoke a POST request via the Interactive doc

![](https://github.com/cdimascio/generator-express-no-stress-typescript/raw/master/assets/interactive-doc.png)

## FAQ
**Q**: How do I modify the example API and make it my own?

**A**: There are two key files that enable you to customize and describe your API:
1. `server/routes.ts` - This references the implementation of all of your routes. Add as many routes as you like and point each route your express handler functions.
2. `server/common/api.yaml` - This file contains your [OpenAPI spec](https://swagger.io/specification/). Describe your API here. It's recommended that you to declare any and all validation logic in this YAML. `express-no-stress-typescript`  uses [express-openapi-validator](https://github.com/cdimascio/express-openapi-validator) to automatically handle all API validation based on what you've defined in the spec.

**Q**: I previously generated an app, but I want to change the API root. How do I do this?

**A**: You need to make to small changes
  1. Modify `server/routes.ts`
  ```javascript
     // Change your original path e.g. /api/v1/examples, to:
     app.use('/api/v2/examples', examplesRouter);
   ```

  2. Modify `server/common/api.yaml` and update the api root:
  ```yaml
    # Change e.g. /api/v1 to /api/v2
    servers:
    - url: /api/v2   
  ```

## License

[MIT](LICENSE)

<a href="https://www.buymeacoffee.com/m97tA5c" target="_blank"><img src="https://bmc-cdn.nyc3.digitaloceanspaces.com/BMC-button-images/custom_images/orange_img.png" alt="Buy Me A Coffee" style="height: auto !important;width: auto !important;" ></a>

## Contributors ✨

Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://blog.brakmic.com"><img src="https://avatars1.githubusercontent.com/u/56779?v=4" width="100px;" alt=""/><br /><sub><b>Harris Brakmić</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=brakmic" title="Code">💻</a></td>
    <td align="center"><a href="https://twitter.com/pubkeypubkey"><img src="https://avatars3.githubusercontent.com/u/8926560?v=4" width="100px;" alt=""/><br /><sub><b>Daniel Meyer</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=pubkey" title="Documentation">📖</a></td>
    <td align="center"><a href="https://otaku.codes"><img src="https://avatars0.githubusercontent.com/u/13603045?v=4" width="100px;" alt=""/><br /><sub><b>Viraj Trivedi</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=inf3cti0n95" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/uronly14me"><img src="https://avatars2.githubusercontent.com/u/5186814?v=4" width="100px;" alt=""/><br /><sub><b>Sangbeom Han</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=uronly14me" title="Documentation">📖</a></td>
    <td align="center"><a href="http://www.stefdegoey.nl"><img src="https://avatars2.githubusercontent.com/u/3907488?v=4" width="100px;" alt=""/><br /><sub><b>Stef de Goey</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=Steffion" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/Rogermax"><img src="https://avatars1.githubusercontent.com/u/2633254?v=4" width="100px;" alt=""/><br /><sub><b>Roger</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=Rogermax" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/bagofjuice"><img src="https://avatars3.githubusercontent.com/u/1395172?v=4" width="100px;" alt=""/><br /><sub><b>Vikash Chauhan</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=bagofjuice" title="Code">💻</a> <a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=bagofjuice" title="Documentation">📖</a></td>
  </tr>
  <tr>
    <td align="center"><a href="https://github.com/tsekityam"><img src="https://avatars3.githubusercontent.com/u/3814422?v=4" width="100px;" alt=""/><br /><sub><b>Tse Kit Yam</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=tsekityam" title="Documentation">📖</a> <a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=tsekityam" title="Code">💻</a></td>
    <td align="center"><a href="https://medium.com/@rlkamradt"><img src="https://avatars3.githubusercontent.com/u/6883379?v=4" width="100px;" alt=""/><br /><sub><b>Randy Kamradt</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=rkamradt" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/2u6z3r0"><img src="https://avatars0.githubusercontent.com/u/10853329?v=4" width="100px;" alt=""/><br /><sub><b>Vitthal Patil</b></sub></a><br /><a href="https://github.com/cdimascio/generator-express-no-stress-typescript/commits?author=2u6z3r0" title="Code">💻</a></td>
  </tr>
</table>

<!-- markdownlint-enable -->
<!-- prettier-ignore-end -->
<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
 readmeEtag: '"b63ae9822a116621de960c355e030d3d66bc82f2"' readmeLastModified: Mon, 30 May 2022 18:08:34 GMT repositoryId: 92780080 description: >- 🚄 A Yeoman generator for Express.js based 12-factor apps and apis using Typescript created: '2017-05-29T22:33:47Z' updated: '2025-10-25T21:54:41Z' language: TypeScript archived: false stars: 363 watchers: 7 forks: 40 owner: cdimascio logo: https://avatars.githubusercontent.com/u/4706618?v=4 license: MIT repoEtag: '"7422c5769ab75b668faafae8726f997ddd2afb9ea3f4dc87569328af9390a5fe"' repoLastModified: Sat, 25 Oct 2025 21:54:41 GMT foundInMaster: true category: Description Validators id: ec24bd83d219c38af4319203892ff3cb - source: openapi3 tags repository: https://github.com/postmanlabs/observability-cli v3: true repositoryMetadata: base64Readme: >- IyBXZWxjb21lISDwn5GLCgpUaGUgZmFzdGVzdCBhbmQgZWFzaWVzdCB3YXkgdG8gdW5kZXJzdGFuZCB5b3VyIEFQSXMuCgpCdWlsdCBmb3IgYnVzeSBkZXZlbG9wZXIgdGVhbXMgd2hvIGRvbid0IGhhdmUgdGltZSB0byBiZWNvbWUgZXhwZXJ0cyBpbiBtb25pdG9yaW5nIGFuZCBvYnNlcnZhYmlsaXR5LCBBa2l0YSBtYWtlcyBpdCBwb3NzaWJsZSB0byBxdWlja2x5IGRpc2NvdmVyIGFsbCB5b3VyIEFQSSBlbmRwb2ludHMsIHNlZSB3aGljaCBhcmUgc2xvd2VzdCwgYW5kIGxlYXJuIHdoaWNoIGhhdmUgZXJyb3JzLiBObyBTREtzIG9yIGNvZGUgY2hhbmdlcyBuZWNlc3NhcnkuCgogICogKipHZXQgcGx1Zy1hbmQtcGxheSBBUEkgbW9uaXRvcmluZy4qKiBFeHBsb3JlIGFuZCBzaGFyZSBwZXItZW5kcG9pbnQgdm9sdW1lLCBsYXRlbmN5LCBhbmQgZXJyb3JzLiBTZXQgcGVyLWVuZHBvaW50IGFsZXJ0cy4KICAqICoqU2VlIEFQSSBlbmRwb2ludHMuKiogQXV0b21hdGljYWxseSBnZXQgYSBzZWFyY2hhYmxlIG1hcCBvZiB5b3VyIEFQSSBlbmRwb2ludHMgaW4gdXNlLiBFeHBsb3JlIGJ5IGxhdGVuY3ksIGVycm9ycywgYW5kIHVzYWdlLiBFeHBvcnQgYXMgT3BlbkFQSSBzcGVjcy4KCkRyb3AgQWtpdGEgaW50byB5b3VyIHN5c3RlbSB0byB1bmRlcnN0YW5kIHlvdXIgYXBwbGljYXRpb27igJlzIGJlaGF2aW9yLCB3aXRob3V0IGhhdmluZyB0byBpbnN0cnVtZW50IGNvZGUgb3IgYnVpbGQgeW91ciBvd24gZGFzaGJvYXJkcy4KCldlJ3JlIGluIG9wZW4gYmV0YSBhbmQgd291bGQgbG92ZSB0byBoYXZlIHlvdSB0cnkgdXMgb3V0ISBbQ3JlYXRlIGFuIGFjY291bnQgaW4gdGhlIEFraXRhIEFwcF0oaHR0cHM6Ly9hcHAuYWtpdGEuc29mdHdhcmUvbG9naW4/c2lnbl91cCkgdG8gZ2V0IHN0YXJ0ZWQuCgogIFtBYm91dCB0aGlzIHJlcG9dKCNhYm91dC10aGlzLXJlcG8pCnwgW1J1bm5pbmcgdGhpcyByZXBvXSgjcnVubmluZy10aGlzLXJlcG8pCnwgW0dldHRpbmcgaW52b2x2ZWRdKCNnZXR0aW5nLWludm9sdmVkKQp8IFtSZWxhdGVkIGxpbmtzXSgjcmVsYXRlZC1saW5rcykKCiMjIEFib3V0IHRoaXMgcmVwbwpUaGlzIGlzIHRoZSBvcGVuLXNvdXJjZSByZXBvc2l0b3J5IGZvciB0aGUgY29tbXVuaXR5IHZlcnNpb24gb2Ygb3VyIENMSSwgYW5kIGlzCmludGVuZGVkIGZvciB1c2Ugd2l0aCB0aGUgQWtpdGEgY29uc29sZS4gVGhpcyBjb21tdW5pdHkgdmVyc2lvbiBvZiB0aGUgQ0xJIGRvZXMKbm90IGluY2x1ZGUgZnVuY3Rpb25hbGl0eSBmb3IgaW5mZXJyaW5nIHR5cGVzIGFuZCBkYXRhIGZvcm1hdHMuIFRoaXMKZnVuY3Rpb25hbGl0eSBpcyBhdmFpbGFibGUgb25seSBpbiB0aGUgYGFraXRhYCBiaW5hcnkgdGhhdCB3ZSBkaXN0cmlidXRlLgoKIyMgUnVubmluZyB0aGlzIHJlcG8KCiMjIyBIb3cgdG8gYnVpbGQKUnVubmluZyB0aGUgZm9sbG93aW5nIGNvbW1hbmRzIHdpbGwgZ2VuZXJhdGUgdGhlIGBha2l0YS1jbGlgIGJpbmFyeToKMS4gSW5zdGFsbCBbR28gMS4xOCBvciBhYm92ZV0oaHR0cHM6Ly9nb2xhbmcub3JnL2RvYy9pbnN0YWxsKS4gCjIuIEluc3RhbGwgYGxpYnBjYXBgCiAgICAtIEZvciBIb21lYnJldyBvbiBtYWM6IGBicmV3IGluc3RhbGwgbGlicGNhcGAKICAgIC0gRm9yIFVidW50dS9EZWJpYW46IGBhcHQtZ2V0IGluc3RhbGwgbGlicGNhcC1kZXZgCjMuIGBtYWtlYAoKCiMjIyBIb3cgdG8gdGVzdAoKMS4gSW5zdGFsbCBbZ29tb2NrXShodHRwczovL2dpdGh1Yi5jb20vZ29sYW5nL21vY2spOiBgZ28gZ2V0IGdpdGh1Yi5jb20vZ29sYW5nL21vY2svbW9ja2dlbmAKMi4gYG1ha2UgdGVzdGAKCiMjIyBIb3cgdG8gdXNlCgpTZWUgb3VyIGRvY3M6IFtTaW5nbGUgSG9zdC9WTV0oaHR0cHM6Ly9kb2NzLmFraXRhLnNvZnR3YXJlL2RvY3MvcnVuLWxvY2FsbHkpLgoKTm90ZTogaWYgeW91J3JlIHBsYW5uaW5nIHRvIHVzZSB0aGUgQWtpdGEgQ0xJIHdpdGggdGhlIEFraXRhIENvbnNvbGUsIHdlIHJlY29tbWVuZCB1c2luZyBvdXIgW3N0YXRpY2FsbHkgbGlua2VkIGJpbmFyaWVzXShodHRwczovL2dpdGh1Yi5jb20vYWtpdGFzb2Z0d2FyZS9ha2l0YS1jbGkvcmVsZWFzZXMpIGlmIHBvc3NpYmxlLgoKIyMgR2V0dGluZyBpbnZvbHZlZAoqIFBsZWFzZSBmaWxlIGJ1Z3MgYXMgaXNzdWVzIHRvIHRoaXMgcmVwb3NpdG9yeS4KKiBXZSB3ZWxjb21lIGNvbnRyaWJ1dGlvbnMhIElmIHlvdSB3YW50IHRvIG1ha2UgY2hhbmdlcyBvciBidWlsZCB5b3VyIG93bgogIGV4dGVuc2lvbnMgdG8gdGhlIENMSSBvbiB0b3Agb2YgdGhlCiAgW0FraXRhIElSXShodHRwczovL2dpdGh1Yi5jb20vYWtpdGFzb2Z0d2FyZS9ha2l0YS1pciksIHBsZWFzZSBzZWUgb3VyCiAgW0NPTlRSSUJVVElOR10oQ09OVFJJQlVUSU5HLm1kKSBkb2MuCiogV2UncmUgYWx3YXlzIGhhcHB5IHRvIGFuc3dlciBhbnkgcXVlc3Rpb25zIGFib3V0IHRoZSBDTEksIG9yIGFib3V0IGhvdyB5b3UKICBjYW4gY29udHJpYnV0ZS4gRW1haWwgdXMgYXQgYG9wZW5zb3VyY2UgW2F0XSBha2l0YXNvZnR3YXJlIFtkb3RdIGNvbWAgb3IKICBbcmVxdWVzdCB0byBqb2luIG91ciBTbGFja10oaHR0cHM6Ly9kb2NzLmdvb2dsZS5jb20vZm9ybXMvZC9lLzFGQUlwUUxTZkYtTWY0TGlfRHF5c0NIeTA0MklCZnZ0cFVESEdZclY2RE9IWmxKY1FWOE9JbEFBL3ZpZXdmb3JtP3VzcD1zZl9saW5rKSEKCiMjIFJlbGF0ZWQgbGlua3MKKiBbQWtpdGEgYmxvZ10oaHR0cHM6Ly93d3cuYWtpdGFzb2Z0d2FyZS5jb20vYmxvZykKKiBbQWtpdGEgZG9jc10oaHR0cHM6Ly9kb2NzLmFraXRhLnNvZnR3YXJlLykKKiBbSm9pbiBvcGVuIGJldGFdKGh0dHBzOi8vYXBwLmFraXRhLnNvZnR3YXJlL2xvZ2luP3NpZ25fdXApCg== readmeEtag: '"7d8eae25730c84ff929c923fd1c8096380108121"' readmeLastModified: Tue, 21 May 2024 10:07:09 GMT repositoryId: 340217357 description: >- The Akita CLI helps you make sense of API traffic. Passively watch API traffic with apidump. Model API behavior with apispec. Compare API behavior with apidiff. created: '2021-02-19T00:43:53Z' updated: '2026-02-04T10:25:09Z' language: Go archived: true stars: 365 watchers: 6 forks: 38 owner: postmanlabs logo: https://avatars.githubusercontent.com/u/10251060?v=4 license: Apache-2.0 repoEtag: '"671dd9f643e4d438d55bb98e467b77defe7d9523491060bc06ed22c87bfff6c0"' repoLastModified: Wed, 04 Feb 2026 10:25:09 GMT foundInMaster: true category: - SDK - Server Implementations id: d68f71ef1b50d8a54f410d403edf4ec3 oldLocations: - https://github.com/akitasoftware/akita-cli - source: openapi3 tags repository: https://github.com/ethereum/beacon-apis v3: true repositoryMetadata: base64Readme: >- IyBFdGhlcmV1bSBCZWFjb24gQVBJcwoKWyFbQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS9ldGhlcmV1bS9iZWFjb24tQVBJcy93b3JrZmxvd3MvQ0kvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2V0aGVyZXVtL2JlYWNvbi1BUElzL2FjdGlvbnMvd29ya2Zsb3dzL21haW4ueW1sP3F1ZXJ5PWJyYW5jaCUzQW1hc3RlcikKCkNvbGxlY3Rpb24gb2YgUkVTVGZ1bCBBUElzIHByb3ZpZGVkIGJ5IEV0aGVyZXVtIEJlYWNvbiBub2RlcwoKQVBJIGJyb3dzZXI6IFtodHRwczovL2V0aGVyZXVtLmdpdGh1Yi5pby9iZWFjb24tQVBJcy9dKGh0dHBzOi8vZXRoZXJldW0uZ2l0aHViLmlvL2JlYWNvbi1BUElzLykKCiMjIE91dGxpbmUKClRoaXMgZG9jdW1lbnQgb3V0bGluZXMgYW4gYXBwbGljYXRpb24gcHJvZ3JhbW1pbmcgaW50ZXJmYWNlIChBUEkpIHdoaWNoIGlzIGV4cG9zZWQgYnkgYSBiZWFjb24gbm9kZSBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgRXRoZXJldW0gW2NvbnNlbnN1cyBsYXllciBzcGVjaWZpY2F0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL2V0aGVyZXVtL2NvbnNlbnN1cy1zcGVjcykuCgpUaGUgQVBJIGlzIGEgUkVTVCBpbnRlcmZhY2UsIGFjY2Vzc2VkIHZpYSBIVFRQLiBUaGUgQVBJIHNob3VsZCBub3QsIHVubGVzcyBwcm90ZWN0ZWQgYnkgYWRkaXRpb25hbCBzZWN1cml0eSBsYXllcnMsIGJlIGV4cG9zZWQgdG8gdGhlIHB1YmxpYyBJbnRlcm5ldCBhcyB0aGUgQVBJIGluY2x1ZGVzIG11bHRpcGxlIGVuZHBvaW50cyB3aGljaCBjb3VsZCBvcGVuIHlvdXIgbm9kZSB0byBkZW5pYWwtb2Ytc2VydmljZSAoRG9TKSBhdHRhY2tzIHRocm91Z2ggZW5kcG9pbnRzIHRyaWdnZXJpbmcgaGVhdnkgcHJvY2Vzc2luZy4KIEN1cnJlbnRseSwgdGhlIG9ubHkgc3VwcG9ydGVkIHJldHVybiBkYXRhIHR5cGUgaXMgSlNPTi4KClRoZSBiZWFjb24gbm9kZSAoQk4pIG1haW50YWlucyB0aGUgc3RhdGUgb2YgdGhlIGJlYWNvbiBjaGFpbiBieSBjb21tdW5pY2F0aW5nIHdpdGggb3RoZXIgYmVhY29uIG5vZGVzIGluIHRoZSBFdGhlcmV1bSBuZXR3b3JrLgpDb25jZXB0dWFsbHksIGl0IGRvZXMgbm90IG1haW50YWluIGtleXBhaXJzIHRoYXQgcGFydGljaXBhdGUgd2l0aCB0aGUgYmVhY29uIGNoYWluLgoKVGhlIHZhbGlkYXRvciBjbGllbnQgKFZDKSBpcyBhIGNvbmNlcHR1YWxseSBzZXBhcmF0ZSBlbnRpdHkgd2hpY2ggdXRpbGl6ZXMgcHJpdmF0ZSBrZXlzCnRvIHBlcmZvcm0gdmFsaWRhdG9yIHJlbGF0ZWQgdGFza3MsIGNhbGxlZCAiZHV0aWVzIiwgb24gdGhlIGJlYWNvbiBjaGFpbi4KIFRoZXNlIGR1dGllcyBpbmNsdWRlIHRoZSBwcm9kdWN0aW9uIG9mIGJlYWNvbiBibG9ja3MgYW5kIHNpZ25pbmcgb2YgYXR0ZXN0YXRpb25zLgoKVGhlIGdvYWwgb2YgdGhpcyBzcGVjaWZpY2F0aW9uIGlzIHRvIHByb21vdGUgaW50ZXJvcGVyYWJpbGl0eSBiZXR3ZWVuIHZhcmlvdXMgYmVhY29uIG5vZGUgaW1wbGVtZW50YXRpb25zLgoKIyMgUmVuZGVyClRvIHJlbmRlciBzcGVjIGluIGJyb3dzZXIgeW91IHdpbGwgbmVlZCBhbnkgaHR0cCBzZXJ2ZXIgdG8gbG9hZCBgaW5kZXguaHRtbGAgZmlsZQppbiByb290IG9mIHRoZSByZXBvLgoKIyMjIyMgUHl0aG9uCmBgYApweXRob24gLW0gaHR0cC5zZXJ2ZXIgODA4MApgYGAKQW5kIGFwaSBzcGVjIHdpbGwgcmVuZGVyIG9uIFtodHRwOi8vbG9jYWxob3N0OjgwODBdKGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MCkuCgojIyMjIyBOb2RlSnMKYGBgCm5wbSBpbnN0YWxsIHNpbXBsZWh0dHBzZXJ2ZXIgLWcKCiMgT1IKCnlhcm4gZ2xvYmFsIGFkZCBzaW1wbGVodHRwc2VydmVyCgpzaW1wbGVodHRwc2VydmVyCmBgYApBbmQgYXBpIHNwZWMgd2lsbCByZW5kZXIgb24gW2h0dHA6Ly9sb2NhbGhvc3Q6ODAwMF0oaHR0cDovL2xvY2FsaG9zdDo4MDAwKS4KCiMjIyBVc2FnZQoKTG9jYWwgY2hhbmdlcyB3aWxsIGJlIG9ic2VydmFibGUgaWYgImRldiIgaXMgc2VsZWN0ZWQgaW4gdGhlICJTZWxlY3QgYSBkZWZpbml0aW9uIiBkcm9wLWRvd24gaW4gdGhlIHdlYiBVSS4KClVzZXJzIG1heSBuZWVkIHRvIHRpY2sgdGhlICJEaXNhYmxlIENhY2hlIiBib3ggaW4gdGhlaXIgYnJvd3NlcidzIGRldmVsb3BlciB0b29scyB0byBzZWUgY2hhbmdlcyBhZnRlciBtb2RpZnlpbmcgdGhlIHNvdXJjZS4KCiMjIENvbnRyaWJ1dGluZwpBcGkgc3BlYyBpcyBjaGVja2VkIGZvciBsaW50IGVycm9ycyBiZWZvcmUgbWVyZ2UuCgpUbyBydW4gbGludCBsb2NhbGx5LCBpbnN0YWxsIGxpbnRlciB3aXRoCmBgYApucG0gaW5zdGFsbCAtZyBAcmVkb2NseS9jbGkKCiMgT1IKCnlhcm4gZ2xvYmFsIGFkZCBAcmVkb2NseS9jbGkKYGBgCmFuZCBydW4gbGludCB3aXRoCmBgYApyZWRvY2x5IGxpbnQgYmVhY29uLW5vZGUtb2FwaS55YW1sCmBgYAoKIyMgUmVsZWFzaW5nCgpUaGlzIHJlcG9zaXRvcnkgc3VwcG9ydHMgYm90aCBzdGFibGUgYW5kIHByZS1yZWxlYXNlcy4gVGhlIHZlcnNpb24gb2YgdGhlIG5leHQgcmVsZWFzZSBoYXMgdG8gYmUKZGV0ZXJtaW5lZCBiYXNlZCBvbiB0aGUgY2hhbmdlcyBpbiBgbWFzdGVyYCBicmFuY2ggc2luY2UgdGhlIGxhc3Qgc3RhYmxlIHJlbGVhc2UuIEl0IGlzIHJlY29tbWVuZGVkCnRvIGNyZWF0ZSBhIHByZS1yZWxlYXNlIGJlZm9yZSByZWxlYXNpbmcgYSBuZXcgc3RhYmxlIHZlcnNpb24uCgojIyMgU3RhYmxlIHJlbGVhc2VzCgpTdGVwcyB0byBjcmVhdGUgYSBuZXcgc3RhYmxlIHJlbGVhc2U6CgotIENyZWF0ZSBhbmQgcHVzaCBhIHRhZyB3aXRoIHRoZSB2ZXJzaW9uIG9mIHRoZSByZWxlYXNlLCBlLmcuIGB2My4wLjBgCi0gQ0Qgd2lsbCBjcmVhdGUgdGhlIGdpdGh1YiByZWxlYXNlLCB1cGxvYWQgYnVuZGxlZCBzcGVjIGZpbGVzLCBhbmQgb3BlbiBhIHJlbGVhc2UgUFIKLSBSZXZpZXcsIGFwcHJvdmUgYW5kIG1lcmdlIHRoZSByZWxlYXNlIFBSIHRvIGBtYXN0ZXJgIGJyYW5jaAoKIyMjIFByZS1yZWxlYXNlcwoKU3RlcHMgdG8gY3JlYXRlIGEgbmV3IHByZS1yZWxlYXNlOgoKLSBDcmVhdGUgYW5kIHB1c2ggYSB0YWcgd2l0aCB0aGUgdmVyc2lvbiBvZiB0aGUgcmVsZWFzZSwgZS5nLiBgdjMuMC4wLWFscGhhLjBgCi0gQ0Qgd2lsbCBjcmVhdGUgdGhlIGdpdGh1YiByZWxlYXNlIGFuZCB1cGxvYWQgYnVuZGxlZCBzcGVjIGZpbGVzCgpQcmUtcmVsZWFzZXMgd2lsbCBub3QgYmUgbGlzdGVkIGluIGBpbmRleC5odG1sYCBhbmQgYXJlIGludGVuZGVkIHRvIGFsbG93IGVhcmx5IHRlc3RpbmcgYWdhaW5zdCB0aGUgc3BlYy4K readmeEtag: '"7c32a91af8c8125dba03b2a013db6a4d45755c75"' readmeLastModified: Wed, 20 Aug 2025 08:52:45 GMT repositoryId: 202613633 description: Collection of RESTful APIs provided by Ethereum Beacon nodes created: '2019-08-15T21:27:12Z' updated: '2026-02-05T21:31:21Z' language: HTML archived: false stars: 376 watchers: 35 forks: 207 owner: ethereum logo: https://avatars.githubusercontent.com/u/6250754?v=4 license: CC0-1.0 repoEtag: '"7d1028c87b149c2ab7dbf331ae0032f6d647c9376c29c4cc04e1e02b8e1c6d69"' repoLastModified: Thu, 05 Feb 2026 21:31:21 GMT foundInMaster: true category: - Documentation - Server Implementations id: e04baea2e4cda6dd0521496ba97d0ab5 - source: openapi3 tags repository: https://github.com/samchungy/zod-openapi v3: true id: c86edd8c7126bca7973ca68a5b5bd9e1 repositoryMetadata: base64Readme: >- <p align="center">
  <img src="zod-openapi.png" width="200px" align="center" alt="zod-openapi logo" />
  <h1 align="center">zod-openapi</h1>
</p>
<p align="center">
A TypeScript library which uses <a href="https://github.com/colinhacks/zod">Zod</a> schemas to generate OpenAPI v3.x documentation.
</p>
<div align="center">
<a href="https://www.npmjs.com/package/zod-openapi"><img src="https://img.shields.io/npm/v/zod-openapi"/></a>
<a href="https://www.npmjs.com/package/zod-openapi"><img src="https://img.shields.io/npm/dm/zod-openapi"/></a>
<a href="https://nodejs.org/en/"><img src="https://img.shields.io/node/v/zod-openapi"/></a>
<a href="https://github.com/samchungy/zod-openapi/actions/workflows/test.yml"><img src="https://github.com/samchungy/zod-openapi/actions/workflows/test.yml/badge.svg"/></a>
<a href="https://github.com/samchungy/zod-openapi/actions/workflows/release.yml"><img src="https://github.com/samchungy/zod-openapi/actions/workflows/release.yml/badge.svg"/></a>
<a href="https://github.com/seek-oss/skuba"><img src="https://img.shields.io/badge/🤿%20skuba-powered-009DC4"/></a>
</div>
<br>

## Installation

Install via `npm`, `yarn`, or `pnpm`:

```bash
npm install zod zod-openapi
# or
yarn add zod zod-openapi
# or
pnpm install zod zod-openapi
```

## Usage

### `.meta()`

Use Zod's native `.meta()` method to add OpenAPI metadata to your schemas. This library leverages Zod's built-in metadata functionality - no monkey patching or additional setup is required. Simply call `.meta()` on any Zod schema and it accepts an object with the following options:

> **Note:** For full TypeScript support of OpenAPI properties in `.meta()`, simply import `zod-openapi` in your project or add `/// <reference types="zod-openapi" />` anywhere in your project. This provides compile-time type definitions for properties like `id`, `param`, `header`, etc.

| Option     | Description                                                                                                      |
| ---------- | ---------------------------------------------------------------------------------------------------------------- |
| `id`       | Registers a schema as a reusable OpenAPI component.                                                              |
| `header`   | Adds metadata for [response headers](#response-headers).                                                         |
| `param`    | Adds metadata for [request parameters](#parameters).                                                             |
| `override` | Allows you to override the rendered OpenAPI schema. This takes either an object or a function.                   |
| `outputId` | Allows you to set a different ID for the output schema. This is useful when the input and output schemas differ. |
| `unusedIO` | Allows you to set the `io` for an unused schema added to the components section. Defaults to `output`.           |

You can also set standard OpenAPI properties directly in Zod's native `.meta()` method, such as:

```typescript
z.string().meta({
  description: 'A text field',
  example: 'Example value',
});
```

You can set additional metadata to the rendered schema in a few different ways. Zod's native `.meta()` method allows you to directly set OpenAPI properties to the schema. However, if you wanted to override any field that Zod generates, this may not work as expected. In this case, you can use the `override` option to customize the rendered OpenAPI schema.

eg.

```typescript
z.string().datetime().meta({
  description: 'A date field',
  format: 'MY-FORMAT',
  'x-custom-field': 'custom value',
});
```

Would render the following OpenAPI schema. Note that format is not overridden with our custom format value.

```json
{
  "type": "string",
  "format": "date-time",
  "description": "A date field",
  "x-custom-field": "custom value"
}
```

In order to override fields which are generated by Zod or this library, you can use the `override` option in Zod's `.meta()` method. This allows you to customize the rendered OpenAPI schema after we have both processed it.

##### override object

This does a naive merge of the override object with the rendered OpenAPI schema. This is useful for simple overrides where you want to add or modify properties without complex logic.

```typescript
z.string.datetime().meta({
  description: 'A date field',
  override: {
    format: 'MY-FORMAT',
  },
});
```

##### override function

The `override` function allows you to deeply customize the rendered OpenAPI schema. Eg. if you wanted to render a Zod union as a `oneOf` instead of `anyOf` for a single schema, you can do so by using the `override` function. View the documentation for `override` in the [CreateDocumentOptions](#createdocumentoptions) section for more information.

```typescript
z.union([z.string(), z.number()]).meta({
  description: 'A union of string and number',
  override: ({ jsonSchema }) => {
    jsonSchema.anyOf = jsonSchema.oneOf;
    delete jsonSchema.oneOf;
  },
});
```

### `createDocument`

Generates an OpenAPI documentation object.

```typescript
import * as z from 'zod/v4';
// or import * as z from 'zod'; if using Zod 4.0.0+

import { createDocument } from 'zod-openapi';

const jobId = z.string().meta({
  description: 'A unique identifier for a job',
  example: '12345',
  id: 'jobId',
});

const title = z.string().meta({
  description: 'Job title',
  example: 'My job',
});

const document = createDocument({
  openapi: '3.1.0',
  info: {
    title: 'My API',
    version: '1.0.0',
  },
  paths: {
    '/jobs/{jobId}': {
      put: {
        requestParams: { path: z.object({ jobId }) },
        requestBody: {
          content: {
            'application/json': { schema: z.object({ title }) },
          },
        },
        responses: {
          '200': {
            description: '200 OK',
            content: {
              'application/json': { schema: z.object({ jobId, title }) },
            },
          },
        },
      },
    },
  },
});
```

<details>
  <summary>Creates the following object:</summary>
  
  ```json
  {
    "openapi": "3.1.0",
    "info": {
      "title": "My API",
      "version": "1.0.0"
    },
    "paths": {
      "/jobs/{jobId}": {
        "put": {
          "parameters": [
            {
              "in": "path",
              "name": "jobId",
              "description": "A unique identifier for a job",
              "schema": {
                "$ref": "#/components/schemas/jobId"
              }
            }
          ],
          "requestBody": {
            "content": {
              "application/json": {
                "schema": {
                  "type": "object",
                  "properties": {
                    "title": {
                      "type": "string",
                      "description": "Job title",
                      "example": "My job"
                    }
                  },
                  "required": ["title"]
                }
              }
            }
          },
          "responses": {
            "200": {
              "description": "200 OK",
              "content": {
                "application/json": {
                  "schema": {
                    "type": "object",
                    "properties": {
                      "jobId": {
                        "$ref": "#/components/schemas/jobId"
                      },
                      "title": {
                        "type": "string",
                        "description": "Job title",
                        "example": "My job"
                      }
                    },
                    "required": ["jobId", "title"]
                  }
                }
              }
            }
          }
        }
      }
    },
    "components": {
      "schemas": {
        "jobId": {
          "type": "string",
          "description": "A unique identifier for a job",
          "example": "12345"
        }
      }
    }
  }
  ```
</details>

`createDocument` takes an optional options argument which can be used to modify how the document is created

```typescript
createDocument(doc, {
  override: ({ jsonSchema, zodSchema, io }) => {
    const def = zodSchema._zod.def;
    if (def.type === 'date' && io === 'output') {
      jsonSchema.type = 'string';
      jsonSchema.format = 'date-time';
    }
  },
  allowEmptySchema: {
    custom: true,
  },
});
```

#### CreateDocumentOptions

| Option             | Type                | Description                                                                                                                               |
| ------------------ | ------------------- | ----------------------------------------------------------------------------------------------------------------------------------------- |
| `override`         | `Function`          | Override rendered schema with a function``                                                                                                |
| `outputIdSuffix`   | `string`            | Suffix for output schema IDs when the schema is used in both a request and response. Defaults to ``'Output'`                              |
| `allowEmptySchema` | `Object`            | Control whether empty schemas are allowed for specific Zod schema types                                                                   |
| `cycles`           | `'ref' \| 'throw'`  | How to handle cycles in schemas.<br>- `'ref'` — Break cycles using $defs<br>- `'throw'` — Error on cycles. Defaults to `'ref'`            |
| `reused`           | `'ref' \| 'inline'` | How to handle reused schemas.<br>- `'ref'` — Reused schemas as references<br>- `'inline'` — Inline reused schemas. Defaults to `'inline'` |
|                    |

##### `override`

The `override` function allows you to customize the rendered OpenAPI schema. It receives an object with the following properties:

| Property     | Type                  | Description                                                                                                       |
| ------------ | --------------------- | ----------------------------------------------------------------------------------------------------------------- |
| `jsonSchema` | `Object`              | The OpenAPI schema object being generated. You can modify this object to change the rendered schema.              |
| `zodSchema`  | `ZodType`             | The original Zod schema being converted to OpenAPI. You can use this to access Zod-specific properties.           |
| `io`         | `'input' \| 'output'` | The context in which the schema is being rendered. `'input'` for request bodies/params, `'output'` for responses. |

This allows you to transform the schema based on your own rules or requirements. For example, if you

1. Wanted to change how dates are rendered in the output context because a typical JSON serializer will transform them to a string
2. Change all union outputs to be oneOf instead of anyOf

```typescript
createDocument(doc, {
  override: ({ jsonSchema, zodSchema, io }) => {
    const def = zodSchema._zod.def;
    if (def.type === 'date' && io === 'output') {
      jsonSchema.type = 'string';
      jsonSchema.format = 'date-time';
      return;
    }
    if (def.type === 'union') {
      jsonSchema.oneOf = jsonSchema.anyOf;
      delete jsonSchema.anyOf;
      return;
    }
  },
});
```

##### `outputIdSuffix`

The `outputIdSuffix` option allows you to set a suffix for output schema IDs when the schema is used in both a request and response context. This is useful for distinguishing between input and output schemas.

##### `allowEmptySchema`

The `allowEmptySchema` option allows you to control whether or not this library will throw when it encounters an empty schema. By default this library will not throw if it encounters an `z.any()` or `z.unknown()` schema.

This allows you to customize how this library handles [unrepresentable](https://zod.dev/json-schema?id=unrepresentable) Zod schemas.

eg. If you wanted to allow `z.custom()` to be rendered as an empty schema, you can set the `allowEmptySchema` option as follows:

```typescript
createDocument(doc, {
  allowEmptySchema: {
    custom: true, // Allow empty schemas for `z.custom()` in all contexts
    set: { output: true }, // Allow empty schemas for `z.set()` only in an output context
  },
});
```

##### `cycles` and `reused`

These options are exposed directly from the Zod. Read the [documentation](https://zod.dev/json-schema?id=unrepresentable#cycles) for more information on how to use these options.

#####

### `createSchema`

Creates an OpenAPI Schema Object along with any registered components. OpenAPI 3.1.0 Schema Objects are fully compatible with JSON Schema.

```typescript
import * as z from 'zod/v4';
import { createSchema } from 'zod-openapi';

const jobId = z.string().meta({
  description: 'A unique identifier for a job',
  example: '12345',
  id: 'jobId',
});

const title = z.string().meta({
  description: 'Job title',
  example: 'My job',
});

const job = z.object({
  jobId,
  title,
});

const { schema, components } = createSchema(job);
```

<details>
  <summary>Creates the following object:</summary>
  
  ```json
  {
    "schema": {
      "type": "object",
      "properties": {
        "jobId": {
          "$ref": "#/components/schemas/jobId"
        },
        "title": {
          "type": "string",
          "description": "Job title",
          "example": "My job"
        }
      },
      "required": ["jobId", "title"]
    },
    "components": {
      "jobId": {
        "type": "string",
        "description": "A unique identifier for a job",
        "example": "12345"
      }
    }
  }
  ```
</details>

#### CreateSchemaOptions

`createSchema` takes an optional `CreateSchemaOptions` parameter which includes all options from [CreateDocumentOptions](#createdocumentoptions) plus the following:

```typescript
const { schema, components } = createSchema(job, {
  // Input/Output context - controls how schemas are generated
  io: 'input', // 'input' for request bodies/params, 'output' for responses
  // Component handling
  schemaComponents: { jobId: z.string() }, // Pre-defined components to use
  schemaComponentRefPath: '#/definitions/', // Custom path prefix for component references
  opts: {}, // Create Document Options,
});
```

### Request Parameters

Query, Path, Header & Cookie parameters can be created using the `requestParams` key under the `method` key as follows:

```typescript
createDocument({
  paths: {
    '/jobs/{a}': {
      put: {
        requestParams: {
          path: z.object({ a: z.string() }),
          query: z.object({ b: z.string() }),
          cookie: z.object({ cookie: z.string() }),
          header: z.object({ 'custom-header': z.string() }),
        },
      },
    },
  },
});
```

If you would like to declare parameters in a more traditional way you may also declare them using the [parameters](https://swagger.io/docs/specification/describing-parameters/) key. The definitions will then all be combined.

```ts
createDocument({
  paths: {
    '/jobs/{a}': {
      put: {
        parameters: [
          z.string().meta({
            param: {
              name: 'job-header',
              in: 'header',
            },
          }),
        ],
      },
    },
  },
});
```

### Request Body

Where you would normally declare the [media type](https://swagger.io/docs/specification/media-types/), set the `schema` as your Zod Schema as follows.

```typescript
createDocument({
  paths: {
    '/jobs': {
      get: {
        requestBody: {
          content: {
            'application/json': { schema: z.object({ a: z.string() }) },
          },
        },
      },
    },
  },
});
```

If you wish to use OpenAPI syntax for your schemas, simply add an OpenAPI schema to the `schema` field instead.

### Responses

Similarly to the [Request Body](#request-body), simply set the `schema` as your Zod Schema as follows. You can set the response headers using the `headers` key.

```typescript
createDocument({
  paths: {
    '/jobs': {
      get: {
        responses: {
          200: {
            description: '200 OK',
            content: {
              'application/json': { schema: z.object({ a: z.string() }) },
            },
            headers: z.object({
              'header-key': z.string(),
            }),
          },
        },
      },
    },
  },
});
```

### Callbacks

```typescript
createDocument({
  paths: {
    '/jobs': {
      get: {
        callbacks: {
          onData: {
            '{$request.query.callbackUrl}/data': {
              post: {
                requestBody: {
                  content: {
                    'application/json': { schema: z.object({ a: z.string() }) },
                  },
                },
                responses: {
                  200: {
                    description: '200 OK',
                    content: {
                      'application/json': {
                        schema: z.object({ a: z.string() }),
                      },
                    },
                  },
                },
              },
            },
          },
        },
      },
    },
  },
});
```

### Creating Components

OpenAPI allows you to define reusable [components](https://swagger.io/docs/specification/components/) and this library allows you to replicate that in two separate ways.

1. Auto registering schema
2. Manually registering schema

#### Schema

If we take the example in `createDocument` and instead create `title` as follows

##### Auto Registering Schema

```typescript
const title = z.string().meta({
  description: 'Job title',
  example: 'My job',
  id: 'jobTitle', // <- new field
});
```

Wherever `title` is used in schemas across the document, it will instead be created as a reference.

```json
{ "$ref": "#/components/schemas/jobTitle" }
```

`title` will then be outputted as a schema within the components section of the documentation.

```json
{
  "components": {
    "schemas": {
      "jobTitle": {
        "type": "string",
        "description": "Job title",
        "example": "My job"
      }
    }
  }
}
```

This is a great way to create less repetitive Open API documentation. There are some Open API features like [discriminator mapping](https://swagger.io/docs/specification/data-models/inheritance-and-polymorphism/) which require all schemas in the union to contain a ref.

##### Manually Registering Schema

Another way to register schema instead of adding a `ref` is to add it to the components directly. This will still work in the same way as `ref`. So whenever we run into that Zod type we will replace it with a reference.

eg.

```typescript
const title = z.string().meta({
  description: 'Job title',
  example: 'My job',
});
createDocument({
  components: {
    schemas: {
      jobTitle: title, // this will register this Zod Schema as jobTitle unless `id` in `.meta()` is specified on the type
    },
  },
});
```

#### Parameters

Query, Path, Header & Cookie parameters can be similarly registered:

```typescript
// Easy auto registration
const jobId = z.string().meta({
  description: 'Job ID',
  example: '1234',
  param: { id: 'jobRef' },
});

createDocument({
  paths: {
    '/jobs/{jobId}': {
      put: {
        requestParams: {
          header: z.object({
            jobId,
          }),
        },
      },
    },
  },
});

// or more verbose auto registration
const jobId = z.string().meta({
  description: 'Job ID',
  example: '1234',
  param: { in: 'header', name: 'jobId', id: 'jobRef' },
});

createDocument({
  paths: {
    '/jobs/{jobId}': {
      put: {
        parameters: [jobId],
      },
    },
  },
});

// or manual registration
const otherJobId = z.string().meta({
  description: 'Job ID',
  example: '1234',
  param: { in: 'header', name: 'jobId' },
});

createDocument({
  components: {
    parameters: {
      jobRef: jobId,
    },
  },
});
```

#### Response Headers

Response headers can be similarly registered:

```typescript
const header = z.string().meta({
  description: 'Job ID',
  example: '1234',
  header: { id: 'some-header' },
});

// or

const jobIdHeader = z.string().meta({
  description: 'Job ID',
  example: '1234',
});

createDocument({
  components: {
    headers: {
      someHeaderRef: jobIdHeader,
    },
  },
});
```

#### Responses

Entire Responses can also be registered

```typescript
const response: ZodOpenApiResponseObject = {
  description: '200 OK',
  content: {
    'application/json': {
      schema: z.object({ a: z.string() }),
    },
  },
  id: 'some-response',
};

//or

const response: ZodOpenApiResponseObject = {
  description: '200 OK',
  content: {
    'application/json': {
      schema: z.object({ a: z.string() }),
    },
  },
};

createDocument({
  components: {
    responses: {
      'some-response': response,
    },
  },
});
```

#### Callbacks

Callbacks can also be registered

```typescript
const callback: ZodOpenApiCallbackObject = {
  id: 'some-callback'
  post: {
    responses: {
      200: {
        description: '200 OK',
        content: {
          'application/json': {
            schema: z.object({ a: z.string() }),
          },
        },
      },
    },
  },
};

//or

const callback: ZodOpenApiCallbackObject = {
  post: {
    responses: {
      200: {
        description: '200 OK',
        content: {
          'application/json': {
            schema: z.object({ a: z.string() }),
          },
        },
      },
    },
  },
};

createDocument({
  components: {
    callbacks: {
      'some-callback': callback,
    },
  },
});
```

#### Path Items

Path Items can also be registered

```typescript
const pathItem: ZodOpenApiPathItemObject = {
  id: 'some-path-item',
  get: {
    responses: {
      200: {
        description: '200 OK',
        content: {
          'application/json': {
            schema: z.object({ a: z.string() }),
          },
        },
      },
    },
  },
};

// or

createDocument({
  components: {
    pathItems: {
      'some-path-item': pathItem,
    },
  },
});
```

#### Security Schemes

Security Schemes can be registered for authentication methods:

```typescript
createDocument({
  components: {
    securitySchemes: {
      bearerAuth: {
        type: 'http',
        scheme: 'bearer',
        bearerFormat: 'JWT',
        description: 'JWT Authentication',
      },
    },
  },
});
```

#### Links

Links can be registered to describe relationships between operations:

```typescript
const link: ZodOpenApiLinkObject = {
  id: 'getUserById',
  operationId: 'getUser',
  parameters: {
    userId: '$request.path.id',
  },
  description: 'Link to get user by id',
};

// or

createDocument({
  components: {
    links: {
      getUserById: {
        operationId: 'getUser',
        parameters: {
          userId: '$request.path.id',
        },
        description: 'Link to get user by id',
      },
    },
  },
});
```

#### Examples

Examples can be registered to provide sample values for schemas:

```typescript
const example: ZodOpenApiExampleObject = {
  id: 'userExample',
  summary: 'A sample user',
  value: {
    id: '123',
    name: 'Jane Doe',
    email: 'jane@example.com',
  },
};

// or

createDocument({
  components: {
    examples: {
      userExample: {
        summary: 'A sample user',
        value: {
          id: '123',
          name: 'Jane Doe',
          email: 'jane@example.com',
        },
      },
    },
  },
});
```

### Zod Types

Zod types are composed of two different parts: the input and the output. This library decides which type to create based on if it is used in a request or response context.

Input:

- Request Parameters (query, path, header, cookie)
- Request Body

Output:

- Response Body
- Response Headers

In general, you want to avoid using a registered input schema in an output context and vice versa. This is because the rendered input and output schemas of a simple Zod schema will differ, even with a simple Zod schema like `z.object()`.

```typescript
const schema = z.object({
  name: z.string(),
});
```

Input schemas (request bodies, parameters):

```json
{
  "type": "object",
  "properties": {
    "name": {
      "type": "string"
    }
  },
  "required": ["name"]
}
```

Output schemas (responses):

```json
{
  "type": "object",
  "properties": {
    "name": {
      "type": "string"
    }
  },
  "required": ["name"],
  "additionalProperties": false
}
```

When the same schema is referenced in both input and output contexts, the library generates two separate component schemas. This happens automatically when a schema with an ID is used in both contexts.

You can customize the output schema name by providing an `outputId`:

```ts
const schema = z
  .object({
    name: z.string(),
  })
  .meta({
    id: 'MyObject',
    outputId: 'MyObjectResponse', // Customize the output schema name
  });
```

You can also set a global suffix for output schemas or use `z.looseObject()` and `z.strictObject()` to have explicit control over the schema behavior.

> **⚠️ Note:** If your registered schema contains dynamically created lazy components, they won't be reused between input and output schemas.

## Supported OpenAPI Versions

Currently the following versions of OpenAPI are supported

- `3.1.0` (minimum version)
- `3.1.1`

Setting the `openapi` field will change how the some of the components are rendered.

```ts
createDocument({
  openapi: '3.1.0',
});
```

As an example `z.string().nullable()` will be rendered differently

`3.0.0`

```json
{
  "type": "string",
  "nullable": true
}
```

`3.1.0`

```json
{
  "type": ["string", "null"]
}
```

## Examples

See the library in use in the [examples](./examples/) folder.

- Simple - [setup](./examples/simple/createSchema.ts) | [openapi.yml](./examples/simple/openapi.yml) | [redoc documentation](https://samchungy.github.io/zod-openapi/examples/simple/redoc-static.html)

## Ecosystem

- [fastify-zod-openapi](https://github.com/samchungy/fastify-zod-openapi) - Fastify plugin for zod-openapi. This includes type provider, Zod schema validation, Zod schema serialization and Swagger UI support.

- [eslint-plugin-zod-openapi](https://github.com/samchungy/eslint-plugin-zod-openapi) - Eslint rules for zod-openapi. This includes features which can autogenerate Typescript comments for your Zod types based on your `description`, `example` and `deprecated` fields.

## Version Information

For information about changes and migration from v4 to v5, see the [v5 migration guide](./docs/v5.md).

## Development

### Prerequisites

- Node.js LTS
- pnpm

```shell
pnpm
pnpm build
```

### Test

```shell
pnpm test
```

### Lint

```shell
# Fix issues
pnpm format

# Check for issues
pnpm lint
```
 readmeEtag: '"c4cae5be2ba57f3a5d6315ba4461d8c3eade30b6"' readmeLastModified: Mon, 12 Jan 2026 06:34:50 GMT repositoryId: 628179878 description: Use Zod Schemas to create OpenAPI v3.x documentation created: '2023-04-15T06:28:51Z' updated: '2026-01-29T11:51:26Z' language: TypeScript archived: false stars: 599 watchers: 4 forks: 29 owner: samchungy logo: https://avatars.githubusercontent.com/u/18017094?v=4 license: MIT repoEtag: '"57118213d75de0fdf68f92cf08e6fc00e2b94eab2fb436d17a547498582c6997"' repoLastModified: Thu, 29 Jan 2026 11:51:26 GMT category: Parsers foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/python-openapi/openapi-spec-validator v3: true id: 55e26cf16a1bedc681a4a5932afc9d43 repositoryMetadata: base64Readme: >- KioqKioqKioqKioqKioqKioqKioqKgpPcGVuQVBJIFNwZWMgdmFsaWRhdG9yCioqKioqKioqKioqKioqKioqKioqKioKCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kb2NrZXIvdi9weXRob25vcGVuYXBpL29wZW5hcGktc3BlYy12YWxpZGF0b3Iuc3ZnP2NvbG9yPSUyMzA4NkRENyZsYWJlbD1kb2NrZXIlMjBodWImc29ydD1zZW12ZXIKICAgICA6dGFyZ2V0OiBodHRwczovL2h1Yi5kb2NrZXIuY29tL3IvcHl0aG9ub3BlbmFwaS9vcGVuYXBpLXNwZWMtdmFsaWRhdG9yCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3Yvb3BlbmFwaS1zcGVjLXZhbGlkYXRvci5zdmcKICAgICA6dGFyZ2V0OiBodHRwczovL3B5cGkucHl0aG9uLm9yZy9weXBpL29wZW5hcGktc3BlYy12YWxpZGF0b3IKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1zcGVjLXZhbGlkYXRvci9tYXN0ZXIuc3ZnP3N0eWxlPWZsYXQKICAgICA6dGFyZ2V0OiBodHRwczovL2NvZGVjb3YuaW8vZ2l0aHViL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc3BlYy12YWxpZGF0b3I/YnJhbmNoPW1hc3RlcgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9weXZlcnNpb25zL29wZW5hcGktc3BlYy12YWxpZGF0b3Iuc3ZnCiAgICAgOnRhcmdldDogaHR0cHM6Ly9weXBpLnB5dGhvbi5vcmcvcHlwaS9vcGVuYXBpLXNwZWMtdmFsaWRhdG9yCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL2Zvcm1hdC9vcGVuYXBpLXNwZWMtdmFsaWRhdG9yLnN2ZwogICAgIDp0YXJnZXQ6IGh0dHBzOi8vcHlwaS5weXRob24ub3JnL3B5cGkvb3BlbmFwaS1zcGVjLXZhbGlkYXRvcgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9zdGF0dXMvb3BlbmFwaS1zcGVjLXZhbGlkYXRvci5zdmcKICAgICA6dGFyZ2V0OiBodHRwczovL3B5cGkucHl0aG9uLm9yZy9weXBpL29wZW5hcGktc3BlYy12YWxpZGF0b3IKCkFib3V0CiMjIyMjCgpPcGVuQVBJIFNwZWMgVmFsaWRhdG9yIGlzIGEgQ0xJLCBwcmUtY29tbWl0IGhvb2sgYW5kIHB5dGhvbiBwYWNrYWdlIHRoYXQgdmFsaWRhdGVzIE9wZW5BUEkgU3BlY3MKYWdhaW5zdCB0aGUgYE9wZW5BUEkgMi4wIChha2EgU3dhZ2dlcikKPGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzIuMC5tZD5gX18sCmBPcGVuQVBJIDMuMCA8aHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjMubWQ+YF9fCmFuZCBgT3BlbkFQSSAzLjEgPGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjEuMC5tZD5gX18Kc3BlY2lmaWNhdGlvbi4gVGhlIHZhbGlkYXRvciBhaW1zIHRvIGNoZWNrIGZvciBmdWxsIGNvbXBsaWFuY2Ugd2l0aCB0aGUgU3BlY2lmaWNhdGlvbi4KCgpEb2N1bWVudGF0aW9uCiMjIyMjIyMjIyMjIyMKCkNoZWNrIGRvY3VtZW50YXRpb24gdG8gc2VlIG1vcmUgZGV0YWlscyBhYm91dCB0aGUgZmVhdHVyZXMuIEFsbCBkb2N1bWVudGF0aW9uIGlzIGluIHRoZSAiZG9jcyIgZGlyZWN0b3J5IGFuZCBvbmxpbmUgYXQgYG9wZW5hcGktc3BlYy12YWxpZGF0b3IucmVhZHRoZWRvY3MuaW8gPGh0dHBzOi8vb3BlbmFwaS1zcGVjLXZhbGlkYXRvci5yZWFkdGhlZG9jcy5pbz5gX18KCgpJbnN0YWxsYXRpb24KIyMjIyMjIyMjIyMjCgouLiBjb2RlLWJsb2NrOjogY29uc29sZQoKICAgIHBpcCBpbnN0YWxsIG9wZW5hcGktc3BlYy12YWxpZGF0b3IKCkFsdGVybmF0aXZlbHkgeW91IGNhbiBkb3dubG9hZCB0aGUgY29kZSBhbmQgaW5zdGFsbCBmcm9tIHRoZSByZXBvc2l0b3J5OgoKLi4gY29kZS1ibG9jazo6IGJhc2gKCiAgIHBpcCBpbnN0YWxsIC1lIGdpdCtodHRwczovL2dpdGh1Yi5jb20vcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1zcGVjLXZhbGlkYXRvci5naXQjZWdnPW9wZW5hcGlfc3BlY192YWxpZGF0b3IKCgpVc2FnZQojIyMjIwoKQ0xJIChDb21tYW5kIExpbmUgSW50ZXJmYWNlKQoqKioqKioqKioqKioqKioqKioqKioqKioqKioqCgpTdHJhaWdodCBmb3J3YXJkIHdheToKCi4uIGNvZGUtYmxvY2s6OiBiYXNoCgogICAgb3BlbmFwaS1zcGVjLXZhbGlkYXRvciBvcGVuYXBpLnlhbWwKCnBpcGVzIHdheToKCi4uIGNvZGUtYmxvY2s6OiBiYXNoCgogICAgY2F0IG9wZW5hcGkueWFtbCB8IG9wZW5hcGktc3BlYy12YWxpZGF0b3IgLQoKZG9ja2VyIHdheToKCi4uIGNvZGUtYmxvY2s6OiBiYXNoCgogICAgZG9ja2VyIHJ1biAtdiBwYXRoL3RvL29wZW5hcGkueWFtbDovb3BlbmFwaS55YW1sIC0tcm0gcHl0aG9ub3BlbmFwaS9vcGVuYXBpLXNwZWMtdmFsaWRhdG9yIC9vcGVuYXBpLnlhbWwKCm9yIG1vcmUgcHl0aG9uaWMgd2F5OgoKLi4gY29kZS1ibG9jazo6IGJhc2gKCiAgICBweXRob24gLW0gb3BlbmFwaV9zcGVjX3ZhbGlkYXRvciBvcGVuYXBpLnlhbWwKCkZvciBtb3JlIGRldGFpbHMsIHJlYWQgYWJvdXQgYENMSSAoQ29tbWFuZCBMaW5lIEludGVyZmFjZSkgPGh0dHBzOi8vb3BlbmFwaS1zcGVjLXZhbGlkYXRvci5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvY2xpLmh0bWw+YF9fLgoKcHJlLWNvbW1pdCBob29rCioqKioqKioqKioqKioqKgoKLi4gY29kZS1ibG9jazo6IHlhbWwKCiAgIHJlcG9zOgogICAtICAgcmVwbzogaHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc3BlYy12YWxpZGF0b3IKICAgICAgIHJldjogMC44LjBiMSAjIFRoZSB2ZXJzaW9uIHRvIHVzZSBvciAnbWFzdGVyJyBmb3IgbGF0ZXN0CiAgICAgICBob29rczoKICAgICAgIC0gICBpZDogb3BlbmFwaS1zcGVjLXZhbGlkYXRvcgoKRm9yIG1vcmUgZGV0YWlscywgcmVhZCBhYm91dCBgcHJlLWNvbW1pdCBob29rIDxodHRwczovL29wZW5hcGktc3BlYy12YWxpZGF0b3IucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0L2hvb2suaHRtbD5gX18uCgpQeXRob24gcGFja2FnZQoqKioqKioqKioqKioqKgoKLi4gY29kZTo6IHB5dGhvbgoKICAgIGZyb20gb3BlbmFwaV9zcGVjX3ZhbGlkYXRvciBpbXBvcnQgdmFsaWRhdGUKICAgIGZyb20gb3BlbmFwaV9zcGVjX3ZhbGlkYXRvci5yZWFkZXJzIGltcG9ydCByZWFkX2Zyb21fZmlsZW5hbWUKCiAgICBzcGVjX2RpY3QsIGJhc2VfdXJpID0gcmVhZF9mcm9tX2ZpbGVuYW1lKCdvcGVuYXBpLnlhbWwnKQoKICAgICMgSWYgbm8gZXhjZXB0aW9uIGlzIHJhaXNlZCBieSB2YWxpZGF0ZSgpLCB0aGUgc3BlYyBpcyB2YWxpZC4KICAgIHZhbGlkYXRlKHNwZWNfZGljdCkKCiAgICB2YWxpZGF0ZSh7J29wZW5hcGknOiAnMy4xLjAnfSkKCiAgICBUcmFjZWJhY2sgKG1vc3QgcmVjZW50IGNhbGwgbGFzdCk6CiAgICAgICAgLi4uCiAgICBPcGVuQVBJVmFsaWRhdGlvbkVycm9yOiAnaW5mbycgaXMgYSByZXF1aXJlZCBwcm9wZXJ0eQoKRm9yIG1vcmUgZGV0YWlscywgcmVhZCBhYm91dCBgUHl0aG9uIHBhY2thZ2UgPGh0dHBzOi8vb3BlbmFwaS1zcGVjLXZhbGlkYXRvci5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvcHl0aG9uLmh0bWw+YF9fLgoKUmVsYXRlZCBwcm9qZWN0cwojIyMjIyMjIyMjIyMjIyMjCgoqIGBvcGVuYXBpLWNvcmUgPGh0dHBzOi8vZ2l0aHViLmNvbS9weXRob24tb3BlbmFwaS9vcGVuYXBpLWNvcmU+YF9fCiAgIFB5dGhvbiBsaWJyYXJ5IHRoYXQgYWRkcyBjbGllbnQtc2lkZSBhbmQgc2VydmVyLXNpZGUgc3VwcG9ydCBmb3IgdGhlIE9wZW5BUEkgdjMuMCBhbmQgT3BlbkFQSSB2My4xIHNwZWNpZmljYXRpb24uCiogYG9wZW5hcGktc2NoZW1hLXZhbGlkYXRvciA8aHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvcj5gX18KICAgUHl0aG9uIGxpYnJhcnkgdGhhdCB2YWxpZGF0ZXMgc2NoZW1hIGFnYWluc3QgdGhlIE9wZW5BUEkgU2NoZW1hIFNwZWNpZmljYXRpb24gdjMuMCBhbmQgT3BlbkFQSSBTY2hlbWEgU3BlY2lmaWNhdGlvbiB2My4xLgoKTGljZW5zZQojIyMjIyMjCgpDb3B5cmlnaHQgKGMpIDIwMTctMjAyMywgQXJ0dXIgTWFjaWFnLCBBbGwgcmlnaHRzIHJlc2VydmVkLiBBcGFjaGUgdjIK readmeEtag: '"31126c0cadb25585ec33cacfdd5e4c60bfc2b5d1"' readmeLastModified: Tue, 06 Jan 2026 16:23:52 GMT repositoryId: 102581313 description: >- OpenAPI Spec Validator is a CLI, pre-commit hook and python package that validates OpenAPI Specs against the OpenAPI 2.0 (aka Swagger), OpenAPI 3.0 and OpenAPI 3.1 specification. created: '2017-09-06T08:08:25Z' updated: '2026-02-01T14:49:20Z' language: Python archived: false stars: 385 watchers: 8 forks: 71 owner: python-openapi logo: https://avatars.githubusercontent.com/u/126442889?v=4 license: Apache-2.0 repoEtag: '"38cea59137f18bf5bdc3454caa47b21e6f8ccd32b029e9918cf34670747cc70e"' repoLastModified: Sun, 01 Feb 2026 14:49:20 GMT category: Parsers oldLocations: - https://github.com/p1c2u/openapi-spec-validator foundInMaster: true v3_1: true - source: openapi3 tags repository: https://github.com/asyncapi/modelina v3: true id: 8525db62087986733db0fc4221c4b719 repositoryMetadata: base64Readme: >- [![AsyncAPI Modelina](./docs/img/readme-banner.png)](https://www.modelina.org)
[![Coverage Status](https://coveralls.io/repos/github/asyncapi/modelina/badge.svg?branch=master)](https://coveralls.io/github/asyncapi/modelina?branch=master)
[![Project Status: Active – The project has reached a stable, usable state and is being actively developed.](https://www.repostatus.org/badges/latest/active.svg)](https://www.repostatus.org/#active)
![Maintenance score](https://img.shields.io/npms-io/maintenance-score/@asyncapi/modelina)
[![Npm latest version](https://img.shields.io/npm/v/@asyncapi/modelina)](https://www.npmjs.com/package/@asyncapi/modelina)
[![License](https://img.shields.io/github/license/asyncapi/modelina)](https://github.com/asyncapi/modelina/blob/master/LICENSE)
[![last commit](https://img.shields.io/github/last-commit/asyncapi/modelina)](https://github.com/asyncapi/modelina/commits/master)
[![Discussions](https://img.shields.io/github/discussions/asyncapi/modelina)](https://github.com/asyncapi/modelina/discussions)
[![Website](https://img.shields.io/website?label=website&url=https%3A%2F%2Fwww.modelina.org)](https://www.modelina.org)
[![Playground](https://img.shields.io/website?label=playground&url=https%3A%2F%2Fwww.modelina.org%2Fplayground)](https://www.modelina.org/playground) <!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-113-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END -->


Your one-stop tool for generating accurate and well-tested models for representing the message payloads. Use it as a tool in your development workflow, or a library in a larger integrations, entirely in your control.

---

<!-- toc is generated with GitHub Actions do not remove toc markers -->

<!-- toc -->

- [Installing Modelina](#installing-modelina)
- [Features](#features)
- [Requirements](#requirements)
- [Documentation](#documentation)
- [Examples](#examples)
- [Versioning and maintenance](#versioning-and-maintenance)
- [Development](#development)
- [Contributing](#contributing)
- [Contributors](#contributors)

<!-- tocstop -->

## Installing Modelina

Install Modelina directly as a dependency to your project:

```bash
npm install @asyncapi/modelina
```

Or if you want to [run Modelina, use the CLI](./modelina-cli/README.md).

```bash
modelina generate <language> ./asyncapi.json
```

<h2 align="center">What Does Modelina Do?</h2>

<p align="center">Modelina put YOU in control of your data models, here is how...</p>

<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td><b>Modelina lets you generate data models from many types of <a href="#inputs">inputs</a></b></td>
<td>

```typescript
const asyncapi = ...
const jsonschema = ...
const openapi = ... 
const metamodel = ... 
...
const models = await generator.generate(
  asyncapi | jsonschema | openapi | metamodel
);
```
</td>
  </tr>
    <tr>
    <td><b>Use the same inputs across a range of different <a href="#outputs">generators</a></b></td>
<td>

```typescript
const generator = new TypeScriptGenerator();
const generator = new CsharpGenerator();
const generator = new JavaGenerator();
const generator = new RustGenerator();
...
const models = await generator.generate(input);
```
</td>
  </tr>
    <tr>
    <td><b>Easily let you interact with the generated models.</b> 

- Want to show the generated models on a website? Sure! 
- Want to generate the models into files? Sure! 
- Want to combine all the models into one single file? Sure! 

Whatever interaction you need, you can create.</td>
<td>

```typescript
const models = await generator.generate(input);
for (const model in models) { 
  const generatedCode = model.result;
  const dependencies = model.dependencies;
  const modeltype = model.type;
  const modelName = model.modelName;
  ...
}
```
</td>
  </tr>
  <tr>
    <td><b>Easily modify how models are <a href="./docs/constraints/README.md">constrained</a> into the output</b></td>

<td>

```typescript
const generator = new TypeScriptGenerator({
  constraints: {
    modelName: ({modelName}) => {
      // Implement your own constraining logic
      return modelName;
    }
  }
});
```
</td>
  </tr>
  <tr>
    <td><b>Seamlessly layer additional or replacement code <a href="./docs/presets.md">on top of each other to customize the models</a> to your use-case</b></td>

<td>

```typescript
const generator = new TypeScriptGenerator({
  presets: [
    {
      class: {
        additionalContent({ content }) {
          return `${content}
public myCustomFunction(): string {
  return 'A custom function for each class';
}`;
        },
      }
    }
  ]
});
const models = await generator.generate(input);
```
</td>
  </tr>
  <tr>
    <td><b>Seamlessly lets you <a href="./docs/presets.md">combine multiple layers of additional or replacement code</a></b></td>

<td>

```typescript
const myCustomFunction1 = {
  class: {
    additionalContent({ content }) {
      return `${content}
public myCustomFunction(): string {
return 'A custom function for each class';
}`;
    },
  }
};
const myCustomFunction2 = {...};
const generator = new TypeScriptGenerator({
  presets: [
    myCustomFunction1,
    myCustomFunction2
  ]
});
const models = await generator.generate(input);
```
</td>
  </tr>
</table>

## Features

The following table provides a short summary of available features for supported output languages. To see the complete feature list for each language, please click the individual links for each language.

<a id="inputs"></a>

<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <th>Supported inputs</th>
    <th></th>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-models-from-asyncapi-documents">AsyncAPI</a></td>
    <td>We support the following AsyncAPI versions: <em>2.0.0 -> 3.0.0</em>, which generates models for all the defined message payloads. It supports the following schemaFormats AsyncAPI Schema object, JSON Schema draft 7, <a href="./examples/asyncapi-avro-schema">AVRO 1.9</a>, <a href="./examples/asyncapi-raml-schema">RAML 1.0 data type</a>, and <a href="./examples/asyncapi-openapi-schema">OpenAPI 3.0 Schema</a>.</td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-models-from-avro-schema-documents">Avro Schema<a></td>
    <td>We support the following Avro versions: <a href="./examples/avro-schema-from-object">v1.x</a></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-models-from-json-schema-documents">JSON Schema</a></td>
    <td>We support the following JSON Schema versions: <em>Draft-4, Draft-6 and Draft-7</em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-models-from-openapi-documents">OpenAPI</a></td>
    <td>We support the following OpenAPI versions: <em><a href="./docs/usage.md#generate-models-from-swagger-20-documents">Swagger 2.0</a>, <a href="./docs/usage.md#generate-models-from-openapi-documents">OpenAPI 3.0 and 3.1</a></em>, which generates models for all the defined request and response payloads.</td>
  </tr>
  <tr>
    <td><a href="./examples/xsd-to-typescript">XSD (XML Schema Definition)</a></td>
    <td>We support XML Schema Definition (XSD) with common elements including simple types, complex types, sequences, choices, attributes, and enumerations.</td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-model-from-typescript-type-files">TypeScript</a></td>
    <td>We currently support TypeScript types as file input for model generation</td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-models-from-meta-models">Meta model</a></td>
    <td>This is the internal representation of a model for Modelina, it is what inputs gets converted to, and what generators are provided to generate code. Instead of relying on an input processor, you can create your own models from scratch and still take advantage on the generators and the features.</td>
  </tr>
</table>

<a id="outputs"></a>
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <th>Supported outputs</th>
    <th></th>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-java-models">Java</a></td>
    <td>Class and enum generation: <em>generation of equals, hashCode, toString, Jackson annotation, custom indentation type and size, etc</em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-typescript-models">TypeScript</a></td>
    <td>Class, interface and enum generation: <em>generation of example code, un/marshal functions, custom indentation type and size, etc</em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-c#-models">C#</a></td>
    <td>Class and enum generation: <em>generation of example code, serializer and deserializer functions, custom indentation type and size, etc</em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-go-models">Go</a></td>
    <td>Struct and enum generation: <em>custom indentation type and size, etc </em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-javascript-models">JavaScript</a></td>
    <td>Class generation: <em>custom indentation type and size, etc</em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-dart-models">Dart</a></td>
    <td>Class and enum generation: json_annotation</td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-rust-models">Rust</a></td>
    <td>Struct/tuple and enum generation: <em>generation of `implement Default`, generate serde macros, custom indentation type and size, etc</em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-python-models">Python</a></td>
    <td>Class and enum generation: <em>custom indentation type and size, etc </em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-kotlin-models">Kotlin</a></td>
    <td>Class and enum generation: <em>use of data classes where appropriate, custom indentation type and size, etc </em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-cplusplus-models">C++</a></td>
    <td>Class and enum generation: <em>custom indentation type and size, etc </em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-php-models">PHP</a></td>
    <td>Class and enum generation: <em>custom indentation type and size, descriptions, etc </em></td>
  </tr>
  <tr>
    <td><a href="./docs/usage.md#generate-scala-models">Scala</a></td>
    <td>Class and enum generation: <em>custom indentation type and size, descriptions, etc </em></td>
  </tr>
</table>

## Requirements
The following are a requirement in order to use Modelina.

- [NodeJS](https://nodejs.org/en/) >= 18

## Documentation
A feature in Modelina cannot exists without an example and documentation for it. You can find all the [documentation here](./docs/README.md).

## Examples
Do you need to know how to use the library in certain scenarios? 

We have gathered all the examples in a separate folder and they can be found under the [examples folder](./examples). 

## Versioning and maintenance
As of version 1, Modelina has a very strict set of changes we are allowed to do before it requires a major version change. In short, any changes that change the generated outcome are not allowed as it's a breaking change for the consumer of the generated models. 

Here is a list of changes we are allowed to do that would not require a breaking change:
- Adding new features (that do not change existing output), such as generators, presets, input processors, etc.
- Change existing features, by providing options that default to current behavior. This could be a preset that adapts the output based on options, as long as the API of Modelina and the API of the generated models does not have any breaking changes.
- Bug fixes where the generated code is otherwise unusable (syntax errors, etc).

Breaking changes are allowed and expected at a frequent rate, of course where it makes sense we will try to bundle multiple changes together.

We of course will do our best to uphold this, but mistakes can happen, and if you notice any breaking changes please let us know!

Because of the number of the limited number of champions, only the most recent major version will be maintained.

Major versions are currently happening at a 3-month cadence (in a similar fashion as the AsyncAPI specification), this will happen in January, April, June, and September. 

## Development
We try to make it as easy for you as possible to set up your development environment to contribute to Modelina. You can find the development documentation [here](./docs/development.md).

## Contributing
Without contributions, Modelina would not exist, it's a community project we build together to create the best possible building blocks, and we do this through [champions](./docs/champions.md).

We have made quite a [comprehensive contribution guide](./docs/contributing.md) to give you a lending hand in how different features and changes are introduced.

If no documentation helps you, here is how you can reach out to get help:
- On the [official AsycnAPI slack](https://asyncapi.com/slack-invite) under the `#04_tooling` channel
- Tag a specific [CODEOWNER](./CODEOWNERS) in your PR
- Generally, it's always a good idea to do everything in public, but in some cases, it might not be possible. In those circumstances you can contact the following: 
  - [jonaslagoni](https://github.com/jonaslagoni) (on [AsyncAPI Slack](https://asyncapi.com/slack-invite), [Twitter](https://twitter.com/jonaslagoni), [Email](mailto:jonas-lt@live.dk), [LinkedIn](https://www.linkedin.com/in/jonaslagoni/))

## Contributors 

Thanks go out to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/magicmatatjahu"><img src="https://avatars.githubusercontent.com/u/20404945?v=4?s=100" width="100px;" alt="Maciej Urbańczyk"/><br /><sub><b>Maciej Urbańczyk</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Amagicmatatjahu" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=magicmatatjahu" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=magicmatatjahu" title="Documentation">📖</a> <a href="#ideas-magicmatatjahu" title="Ideas, Planning, & Feedback">🤔</a> <a href="#maintenance-magicmatatjahu" title="Maintenance">🚧</a> <a href="#question-magicmatatjahu" title="Answering Questions">💬</a> <a href="https://github.com/asyncapi/modelina/commits?author=magicmatatjahu" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Amagicmatatjahu" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/czlowiek488"><img src="https://avatars.githubusercontent.com/u/34620109?v=4?s=100" width="100px;" alt="czlowiek488"/><br /><sub><b>czlowiek488</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Aczlowiek488" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Aczlowiek488" title="Reviewed Pull Requests">👀</a> <a href="#ideas-czlowiek488" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/smoya"><img src="https://avatars.githubusercontent.com/u/1083296?v=4?s=100" width="100px;" alt="Sergio Moya"/><br /><sub><b>Sergio Moya</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Asmoya" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=smoya" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=smoya" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=smoya" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Asmoya" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jonaslagoni"><img src="https://avatars.githubusercontent.com/u/13396189?v=4?s=100" width="100px;" alt="Jonas Lagoni"/><br /><sub><b>Jonas Lagoni</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Ajonaslagoni" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=jonaslagoni" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=jonaslagoni" title="Documentation">📖</a> <a href="#ideas-jonaslagoni" title="Ideas, Planning, & Feedback">🤔</a> <a href="#maintenance-jonaslagoni" title="Maintenance">🚧</a> <a href="#question-jonaslagoni" title="Answering Questions">💬</a> <a href="https://github.com/asyncapi/modelina/commits?author=jonaslagoni" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Ajonaslagoni" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://dev.to/derberg"><img src="https://avatars.githubusercontent.com/u/6995927?v=4?s=100" width="100px;" alt="Lukasz Gornicki"/><br /><sub><b>Lukasz Gornicki</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Aderberg" title="Reviewed Pull Requests">👀</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Aderberg" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=derberg" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/arjungarg07"><img src="https://avatars.githubusercontent.com/u/53009722?v=4?s=100" width="100px;" alt="Arjun Garg"/><br /><sub><b>Arjun Garg</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=arjungarg07" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://www.fmvilas.com"><img src="https://avatars.githubusercontent.com/u/242119?v=4?s=100" width="100px;" alt="Fran Méndez"/><br /><sub><b>Fran Méndez</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Afmvilas" title="Reviewed Pull Requests">👀</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Blakelist7"><img src="https://avatars.githubusercontent.com/u/54525960?v=4?s=100" width="100px;" alt="Kanwal Singh"/><br /><sub><b>Kanwal Singh</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Blakelist7" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://www.alejandraquetzalli.com"><img src="https://avatars.githubusercontent.com/u/19964402?v=4?s=100" width="100px;" alt="Alejandra Quetzalli "/><br /><sub><b>Alejandra Quetzalli </b></sub></a><br /><a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Aalequetzalli" title="Reviewed Pull Requests">👀</a> <a href="https://github.com/asyncapi/modelina/commits?author=alequetzalli" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/md-saif-husain"><img src="https://avatars.githubusercontent.com/u/70682968?v=4?s=100" width="100px;" alt="MD SAIF  HUSAIN"/><br /><sub><b>MD SAIF  HUSAIN</b></sub></a><br /><a href="#example-md-saif-husain" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=md-saif-husain" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=md-saif-husain" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=md-saif-husain" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://sudipto.ghosh.pro"><img src="https://avatars.githubusercontent.com/u/11232940?v=4?s=100" width="100px;" alt="Sudipto Ghosh"/><br /><sub><b>Sudipto Ghosh</b></sub></a><br /><a href="#example-sudiptog81" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=sudiptog81" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=sudiptog81" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/panwauu"><img src="https://avatars.githubusercontent.com/u/62597223?v=4?s=100" width="100px;" alt="panwauu"/><br /><sub><b>panwauu</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=panwauu" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=panwauu" title="Tests">⚠️</a> <a href="#example-panwauu" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=panwauu" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Apanwauu" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://www.linkedin.com/in/stefanemayer/"><img src="https://avatars.githubusercontent.com/u/15031950?v=4?s=100" width="100px;" alt="Stefan E. Mayer"/><br /><sub><b>Stefan E. Mayer</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Astefanerwinmayer" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/HashTalmiz"><img src="https://avatars.githubusercontent.com/u/55018280?v=4?s=100" width="100px;" alt="Talmiz Ahmed"/><br /><sub><b>Talmiz Ahmed</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=HashTalmiz" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=HashTalmiz" title="Tests">⚠️</a> <a href="#example-HashTalmiz" title="Examples">💡</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ThanksForAllTheFish"><img src="https://avatars.githubusercontent.com/u/2169655?v=4?s=100" width="100px;" alt="Marco"/><br /><sub><b>Marco</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3AThanksForAllTheFish" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/quadrrem"><img src="https://avatars.githubusercontent.com/u/8450873?v=4?s=100" width="100px;" alt="quadrrem"/><br /><sub><b>quadrrem</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=quadrrem" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=quadrrem" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://kamko.dev"><img src="https://avatars.githubusercontent.com/u/17074375?v=4?s=100" width="100px;" alt="Kamil Janeček"/><br /><sub><b>Kamil Janeček</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=kamko" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Akamko" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=kamko" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/mahakporwal02"><img src="https://avatars.githubusercontent.com/u/56486682?v=4?s=100" width="100px;" alt="mahakporwal02"/><br /><sub><b>mahakporwal02</b></sub></a><br /><a href="#example-mahakporwal02" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=mahakporwal02" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=mahakporwal02" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=mahakporwal02" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ron-debajyoti"><img src="https://avatars.githubusercontent.com/u/22571664?v=4?s=100" width="100px;" alt="Debajyoti Halder"/><br /><sub><b>Debajyoti Halder</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ron-debajyoti" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=ron-debajyoti" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=ron-debajyoti" title="Documentation">📖</a> <a href="#example-ron-debajyoti" title="Examples">💡</a> <a href="#maintenance-ron-debajyoti" title="Maintenance">🚧</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://ritik307.github.io/portfolio/"><img src="https://avatars.githubusercontent.com/u/22374829?v=4?s=100" width="100px;" alt="Ritik Rawal"/><br /><sub><b>Ritik Rawal</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ritik307" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=ritik307" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=ritik307" title="Tests">⚠️</a> <a href="#example-ritik307" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Ishan-Saini"><img src="https://avatars.githubusercontent.com/u/54525602?v=4?s=100" width="100px;" alt="Ishan"/><br /><sub><b>Ishan</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Ishan-Saini" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Ishan-Saini" title="Tests">⚠️</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://samridhi-98.github.io/Portfolio"><img src="https://avatars.githubusercontent.com/u/54466041?v=4?s=100" width="100px;" alt="Samriddhi"/><br /><sub><b>Samriddhi</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Samridhi-98" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Samridhi-98" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=Samridhi-98" title="Documentation">📖</a> <a href="#example-Samridhi-98" title="Examples">💡</a> <a href="#maintenance-Samridhi-98" title="Maintenance">🚧</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/gabormagyar"><img src="https://avatars.githubusercontent.com/u/63397303?v=4?s=100" width="100px;" alt="Gábor Magyar"/><br /><sub><b>Gábor Magyar</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=gabormagyar" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=gabormagyar" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=gabormagyar" title="Documentation">📖</a> <a href="#example-gabormagyar" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ibernabeudev"><img src="https://avatars.githubusercontent.com/u/74215074?v=4?s=100" width="100px;" alt="ibernabeudev"/><br /><sub><b>ibernabeudev</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ibernabeudev" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=ibernabeudev" title="Tests">⚠️</a> <a href="#example-ibernabeudev" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=ibernabeudev" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/slowikowskiarkadiusz"><img src="https://avatars.githubusercontent.com/u/97508930?v=4?s=100" width="100px;" alt="Arkadiusz Słowikowski"/><br /><sub><b>Arkadiusz Słowikowski</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=slowikowskiarkadiusz" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=slowikowskiarkadiusz" title="Tests">⚠️</a> <a href="#example-slowikowskiarkadiusz" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=slowikowskiarkadiusz" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Polygens"><img src="https://avatars.githubusercontent.com/u/3582318?v=4?s=100" width="100px;" alt="Willem Gillis"/><br /><sub><b>Willem Gillis</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Polygens" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Polygens" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3APolygens" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/rmasarovic"><img src="https://avatars.githubusercontent.com/u/1522834?v=4?s=100" width="100px;" alt="rmasarovic"/><br /><sub><b>rmasarovic</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=rmasarovic" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=rmasarovic" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=rmasarovic" title="Tests">⚠️</a> <a href="#example-rmasarovic" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/owais34"><img src="https://avatars.githubusercontent.com/u/37238759?v=4?s=100" width="100px;" alt="Owais Hasnath Ahmed"/><br /><sub><b>Owais Hasnath Ahmed</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=owais34" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=owais34" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=owais34" title="Documentation">📖</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/PanMan"><img src="https://avatars.githubusercontent.com/u/978501?v=4?s=100" width="100px;" alt="PanMan"/><br /><sub><b>PanMan</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=PanMan" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/artur-ciocanu"><img src="https://avatars.githubusercontent.com/u/743192?v=4?s=100" width="100px;" alt="artur-ciocanu"/><br /><sub><b>artur-ciocanu</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=artur-ciocanu" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Aartur-ciocanu" title="Bug reports">🐛</a> <a href="#example-artur-ciocanu" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=artur-ciocanu" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=artur-ciocanu" title="Tests">⚠️</a> <a href="#maintenance-artur-ciocanu" title="Maintenance">🚧</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://micro-jumbo.eu/"><img src="https://avatars.githubusercontent.com/u/11511697?v=4?s=100" width="100px;" alt="Cyprian Gracz"/><br /><sub><b>Cyprian Gracz</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=micro-jumbo" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=micro-jumbo" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Amicro-jumbo" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://www.printnanny.ai"><img src="https://avatars.githubusercontent.com/u/2601819?v=4?s=100" width="100px;" alt="Leigh Johnson"/><br /><sub><b>Leigh Johnson</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=leigh-johnson" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=leigh-johnson" title="Tests">⚠️</a> <a href="#example-leigh-johnson" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=leigh-johnson" title="Documentation">📖</a> <a href="#maintenance-leigh-johnson" title="Maintenance">🚧</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Aleigh-johnson" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/nitintejuja"><img src="https://avatars.githubusercontent.com/u/95347924?v=4?s=100" width="100px;" alt="Nitin Tejuja"/><br /><sub><b>Nitin Tejuja</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=nitintejuja" title="Tests">⚠️</a> <a href="#example-nitintejuja" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kennethaasan"><img src="https://avatars.githubusercontent.com/u/1437394?v=4?s=100" width="100px;" alt="Kenneth Aasan"/><br /><sub><b>Kenneth Aasan</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=kennethaasan" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=kennethaasan" title="Tests">⚠️</a> <a href="#maintenance-kennethaasan" title="Maintenance">🚧</a> <a href="https://github.com/asyncapi/modelina/commits?author=kennethaasan" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/amit-ksh"><img src="https://avatars.githubusercontent.com/u/91947037?v=4?s=100" width="100px;" alt="Amit Kumar Sharma"/><br /><sub><b>Amit Kumar Sharma</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=amit-ksh" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=amit-ksh" title="Documentation">📖</a> <a href="#example-amit-ksh" title="Examples">💡</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/zaytsevand"><img src="https://avatars.githubusercontent.com/u/5207748?v=4?s=100" width="100px;" alt="Andrey Zaytsev"/><br /><sub><b>Andrey Zaytsev</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=zaytsevand" title="Code">💻</a> <a href="#example-zaytsevand" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=zaytsevand" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=zaytsevand" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/codingtenshi"><img src="https://avatars.githubusercontent.com/u/116377630?v=4?s=100" width="100px;" alt="Tenshi Codes"/><br /><sub><b>Tenshi Codes</b></sub></a><br /><a href="#infra-codingtenshi" title="Infrastructure (Hosting, Build-Tools, etc)">🚇</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://yushiomote.org/"><img src="https://avatars.githubusercontent.com/u/3733915?v=4?s=100" width="100px;" alt="Yushi OMOTE"/><br /><sub><b>Yushi OMOTE</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3AYushiOMOTE" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=YushiOMOTE" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://malcherczyk.pl"><img src="https://avatars.githubusercontent.com/u/17534504?v=4?s=100" width="100px;" alt="Zbigniew Malcherczyk"/><br /><sub><b>Zbigniew Malcherczyk</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3AFerror" title="Bug reports">🐛</a> <a href="#infra-Ferror" title="Infrastructure (Hosting, Build-Tools, etc)">🚇</a> <a href="https://github.com/asyncapi/modelina/commits?author=Ferror" title="Code">💻</a> <a href="#example-Ferror" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=Ferror" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3AFerror" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/200Puls"><img src="https://avatars.githubusercontent.com/u/6918360?v=4?s=100" width="100px;" alt="200Puls"/><br /><sub><b>200Puls</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=200Puls" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=200Puls" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://linktr.ee/anaysarkar7"><img src="https://avatars.githubusercontent.com/u/53341181?v=4?s=100" width="100px;" alt="Anay Sarkar"/><br /><sub><b>Anay Sarkar</b></sub></a><br /><a href="#example-anaysarkar7" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=anaysarkar7" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=anaysarkar7" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/LouisXhaferi"><img src="https://avatars.githubusercontent.com/u/52397677?v=4?s=100" width="100px;" alt="Louis Xhaferi"/><br /><sub><b>Louis Xhaferi</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=LouisXhaferi" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/sambhavgupta0705"><img src="https://avatars.githubusercontent.com/u/81870866?v=4?s=100" width="100px;" alt="Sambhav Gupta"/><br /><sub><b>Sambhav Gupta</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=sambhavgupta0705" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Asambhavgupta0705" title="Reviewed Pull Requests">👀</a> <a href="#design-sambhavgupta0705" title="Design">🎨</a> <a href="https://github.com/asyncapi/modelina/commits?author=sambhavgupta0705" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Asambhavgupta0705" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/prayutsu"><img src="https://avatars.githubusercontent.com/u/54636525?v=4?s=100" width="100px;" alt="Abhay Garg"/><br /><sub><b>Abhay Garg</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=prayutsu" title="Code">💻</a> <a href="#example-prayutsu" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=prayutsu" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=prayutsu" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/henrikjon"><img src="https://avatars.githubusercontent.com/u/27212232?v=4?s=100" width="100px;" alt="henrikjon"/><br /><sub><b>henrikjon</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=henrikjon" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=henrikjon" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=henrikjon" title="Documentation">📖</a> <a href="#example-henrikjon" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://yasirdeveloper.netlify.app/"><img src="https://avatars.githubusercontent.com/u/74600745?v=4?s=100" width="100px;" alt="Mohammad Yasir"/><br /><sub><b>Mohammad Yasir</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Yasir761" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Savio629"><img src="https://avatars.githubusercontent.com/u/91362589?v=4?s=100" width="100px;" alt="Savio Dias"/><br /><sub><b>Savio Dias</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Savio629" title="Code">💻</a> <a href="#infra-Savio629" title="Infrastructure (Hosting, Build-Tools, etc)">🚇</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3ASavio629" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ishaan812"><img src="https://avatars.githubusercontent.com/u/70190533?v=4?s=100" width="100px;" alt="Ishaan Shah"/><br /><sub><b>Ishaan Shah</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ishaan812" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Shreyas0410"><img src="https://avatars.githubusercontent.com/u/70795867?v=4?s=100" width="100px;" alt="Shreyas0410"/><br /><sub><b>Shreyas0410</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Shreyas0410" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Shreyas0410" title="Documentation">📖</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/beku-epitome"><img src="https://avatars.githubusercontent.com/u/115151513?v=4?s=100" width="100px;" alt="beku-epitome"/><br /><sub><b>beku-epitome</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=beku-epitome" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Abeku-epitome" title="Reviewed Pull Requests">👀</a> <a href="https://github.com/asyncapi/modelina/commits?author=beku-epitome" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/swordmaster2k"><img src="https://avatars.githubusercontent.com/u/3354016?v=4?s=100" width="100px;" alt="Joshua Michael Daly"/><br /><sub><b>Joshua Michael Daly</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Aswordmaster2k" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/dlkj"><img src="https://avatars.githubusercontent.com/u/243059?v=4?s=100" width="100px;" alt="Daniel KJ"/><br /><sub><b>Daniel KJ</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=dlkj" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=dlkj" title="Tests">⚠️</a> <a href="#example-dlkj" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Adlkj" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://codeyt.com/"><img src="https://avatars.githubusercontent.com/u/73033511?v=4?s=100" width="100px;" alt="Bhavik Agarwal"/><br /><sub><b>Bhavik Agarwal</b></sub></a><br /><a href="#design-Bhavik-ag" title="Design">🎨</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kaushik-rishi"><img src="https://avatars.githubusercontent.com/u/52498617?v=4?s=100" width="100px;" alt="Rishi"/><br /><sub><b>Rishi</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=kaushik-rishi" title="Code">💻</a> <a href="#design-kaushik-rishi" title="Design">🎨</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://rohithboppey.netlify.app"><img src="https://avatars.githubusercontent.com/u/73538974?v=4?s=100" width="100px;" alt="Rohith Boppey"/><br /><sub><b>Rohith Boppey</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=RohithBoppey" title="Code">💻</a> <a href="#design-RohithBoppey" title="Design">🎨</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://ashishpadhy.live"><img src="https://avatars.githubusercontent.com/u/100484401?v=4?s=100" width="100px;" alt="Ashish Padhy"/><br /><sub><b>Ashish Padhy</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Shurtu-gal" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Shurtu-gal" title="Tests">⚠️</a> <a href="#infra-Shurtu-gal" title="Infrastructure (Hosting, Build-Tools, etc)">🚇</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="http://jfcote.github.io"><img src="https://avatars.githubusercontent.com/u/14336900?v=4?s=100" width="100px;" alt="Jean-François Côté"/><br /><sub><b>Jean-François Côté</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=JFCote" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=JFCote" title="Tests">⚠️</a> <a href="#example-JFCote" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=JFCote" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/SumantxD"><img src="https://avatars.githubusercontent.com/u/65810424?v=4?s=100" width="100px;" alt="Sumant.xD"/><br /><sub><b>Sumant.xD</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=SumantxD" title="Tests">⚠️</a> <a href="#infra-SumantxD" title="Infrastructure (Hosting, Build-Tools, etc)">🚇</a> <a href="https://github.com/asyncapi/modelina/commits?author=SumantxD" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/aryanas159"><img src="https://avatars.githubusercontent.com/u/114330931?v=4?s=100" width="100px;" alt="Aryan Singh"/><br /><sub><b>Aryan Singh</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=aryanas159" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://markus.poerschke.nrw"><img src="https://avatars.githubusercontent.com/u/1222377?v=4?s=100" width="100px;" alt="Markus Poerschke"/><br /><sub><b>Markus Poerschke</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=markuspoerschke" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=markuspoerschke" title="Tests">⚠️</a> <a href="#example-markuspoerschke" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=markuspoerschke" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jamesmoey"><img src="https://avatars.githubusercontent.com/u/457472?v=4?s=100" width="100px;" alt="James Moey"/><br /><sub><b>James Moey</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=jamesmoey" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=jamesmoey" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/tomwolanski"><img src="https://avatars.githubusercontent.com/u/68085653?v=4?s=100" width="100px;" alt="tomwolanski"/><br /><sub><b>tomwolanski</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Atomwolanski" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Ksisa"><img src="https://avatars.githubusercontent.com/u/53404771?v=4?s=100" width="100px;" alt="Kristupas"/><br /><sub><b>Kristupas</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Ksisa" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Devansh-Bhatt"><img src="https://avatars.githubusercontent.com/u/94732079?v=4?s=100" width="100px;" alt="Devansh-Bhatt"/><br /><sub><b>Devansh-Bhatt</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Devansh-Bhatt" title="Tests">⚠️</a> <a href="#infra-Devansh-Bhatt" title="Infrastructure (Hosting, Build-Tools, etc)">🚇</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/pipliya"><img src="https://avatars.githubusercontent.com/u/56186142?v=4?s=100" width="100px;" alt="Ansh Pancholi"/><br /><sub><b>Ansh Pancholi</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=pipliya" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Qypol342"><img src="https://avatars.githubusercontent.com/u/37497007?v=4?s=100" width="100px;" alt="Maeght Loan"/><br /><sub><b>Maeght Loan</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Qypol342" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Qypol342" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://princerajpoot.com"><img src="https://avatars.githubusercontent.com/u/44585452?v=4?s=100" width="100px;" alt="Prince Rajpoot"/><br /><sub><b>Prince Rajpoot</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=princerajpoot20" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/harshit-senpai"><img src="https://avatars.githubusercontent.com/u/93075068?v=4?s=100" width="100px;" alt="harshit mishra "/><br /><sub><b>harshit mishra </b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=harshit-senpai" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/PeteAudinate"><img src="https://avatars.githubusercontent.com/u/99274874?v=4?s=100" width="100px;" alt="PeteAudinate"/><br /><sub><b>PeteAudinate</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=PeteAudinate" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jano-petras"><img src="https://avatars.githubusercontent.com/u/1147848?v=4?s=100" width="100px;" alt="jano-petras"/><br /><sub><b>jano-petras</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=jano-petras" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=jano-petras" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=jano-petras" title="Tests">⚠️</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/nilkanth987"><img src="https://avatars.githubusercontent.com/u/12413581?v=4?s=100" width="100px;" alt="Nilkanth Parmar"/><br /><sub><b>Nilkanth Parmar</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=nilkanth987" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=nilkanth987" title="Tests">⚠️</a> <a href="#example-nilkanth987" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=nilkanth987" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://jaisarita.vercel.app/"><img src="https://avatars.githubusercontent.com/u/43639341?v=4?s=100" width="100px;" alt="Ashmit JaiSarita Gupta"/><br /><sub><b>Ashmit JaiSarita Gupta</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=devilkiller-ag" title="Code">💻</a> <a href="#design-devilkiller-ag" title="Design">🎨</a> <a href="https://github.com/asyncapi/modelina/commits?author=devilkiller-ag" title="Documentation">📖</a> <a href="#maintenance-devilkiller-ag" title="Maintenance">🚧</a> <a href="https://github.com/asyncapi/modelina/commits?author=devilkiller-ag" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Adevilkiller-ag" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://harshil.prose.sh"><img src="https://avatars.githubusercontent.com/u/79367883?v=4?s=100" width="100px;" alt="Harshil Jani"/><br /><sub><b>Harshil Jani</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Harshil-Jani" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Harshil-Jani" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://mintu-portfolio.netlify.app/"><img src="https://avatars.githubusercontent.com/u/127925465?v=4?s=100" width="100px;" alt="Mintu Gogoi"/><br /><sub><b>Mintu Gogoi</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Min2who" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/devansh-m12"><img src="https://avatars.githubusercontent.com/u/86195162?v=4?s=100" width="100px;" alt="Devansh Mahant"/><br /><sub><b>Devansh Mahant</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=devansh-m12" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/officialasishkumar"><img src="https://avatars.githubusercontent.com/u/87874775?v=4?s=100" width="100px;" alt="Asish Kumar"/><br /><sub><b>Asish Kumar</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=officialasishkumar" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ankur0904"><img src="https://avatars.githubusercontent.com/u/98346896?v=4?s=100" width="100px;" alt="Ankur Singh"/><br /><sub><b>Ankur Singh</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ankur0904" title="Documentation">📖</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/RowlandBanks"><img src="https://avatars.githubusercontent.com/u/9962551?v=4?s=100" width="100px;" alt="RowlandBanks"/><br /><sub><b>RowlandBanks</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=RowlandBanks" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=RowlandBanks" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3ARowlandBanks" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://moritzkalwa.dev/"><img src="https://avatars.githubusercontent.com/u/62842654?v=4?s=100" width="100px;" alt="Moritz Kalwa"/><br /><sub><b>Moritz Kalwa</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=moritzkalwa" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=moritzkalwa" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Amoritzkalwa" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=moritzkalwa" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/akkshitgupta"><img src="https://avatars.githubusercontent.com/u/96991785?v=4?s=100" width="100px;" alt="Akshit Gupta"/><br /><sub><b>Akshit Gupta</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=akkshitgupta" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Aakkshitgupta" title="Bug reports">🐛</a> <a href="#example-akkshitgupta" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=akkshitgupta" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=akkshitgupta" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Athul0491"><img src="https://avatars.githubusercontent.com/u/65664185?v=4?s=100" width="100px;" alt="Athul Tulasidasan"/><br /><sub><b>Athul Tulasidasan</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Athul0491" title="Code">💻</a> <a href="#example-Athul0491" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=Athul0491" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=Athul0491" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Souvikns"><img src="https://avatars.githubusercontent.com/u/41781438?v=4?s=100" width="100px;" alt="souvik"/><br /><sub><b>souvik</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Souvikns" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Souvikns" title="Tests">⚠️</a> <a href="#example-Souvikns" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=Souvikns" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Louis-PhilippeGentile"><img src="https://avatars.githubusercontent.com/u/43426946?v=4?s=100" width="100px;" alt="Louis-PhilippeGentile"/><br /><sub><b>Louis-PhilippeGentile</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Louis-PhilippeGentile" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Louis-PhilippeGentile" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=Louis-PhilippeGentile" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3ALouis-PhilippeGentile" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/AkhilJ321"><img src="https://avatars.githubusercontent.com/u/98508374?v=4?s=100" width="100px;" alt="Akhil Jamwal"/><br /><sub><b>Akhil Jamwal</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=AkhilJ321" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=AkhilJ321" title="Tests">⚠️</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ahem"><img src="https://avatars.githubusercontent.com/u/163283?v=4?s=100" width="100px;" alt="Anders Hellerup Madsen"/><br /><sub><b>Anders Hellerup Madsen</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ahem" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=ahem" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Aahem" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/rquinio1A"><img src="https://avatars.githubusercontent.com/u/58322910?v=4?s=100" width="100px;" alt="rquinio1A"/><br /><sub><b>rquinio1A</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=rquinio1A" title="Code">💻</a> <a href="#example-rquinio1A" title="Examples">💡</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/gitankit7"><img src="https://avatars.githubusercontent.com/u/34441936?v=4?s=100" width="100px;" alt="gitankit7"/><br /><sub><b>gitankit7</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=gitankit7" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/janne-slamcore"><img src="https://avatars.githubusercontent.com/u/112867847?v=4?s=100" width="100px;" alt="Janne Husberg"/><br /><sub><b>Janne Husberg</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=janne-slamcore" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=janne-slamcore" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Ajanne-slamcore" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/borrull"><img src="https://avatars.githubusercontent.com/u/5275400?v=4?s=100" width="100px;" alt="borrull"/><br /><sub><b>borrull</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=borrull" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=borrull" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://www.yokeplatform.com"><img src="https://avatars.githubusercontent.com/u/2106178?v=4?s=100" width="100px;" alt="Kamil Zuzda"/><br /><sub><b>Kamil Zuzda</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=kamaz" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jespitae"><img src="https://avatars.githubusercontent.com/u/23041146?v=4?s=100" width="100px;" alt="jespitae"/><br /><sub><b>jespitae</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=jespitae" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=jespitae" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Ajespitae" title="Bug reports">🐛</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://pike.github.io/"><img src="https://avatars.githubusercontent.com/u/43494?v=4?s=100" width="100px;" alt="Axel Hecht"/><br /><sub><b>Axel Hecht</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Pike" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Pike" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/tomdevroomen"><img src="https://avatars.githubusercontent.com/u/4637986?v=4?s=100" width="100px;" alt="Tom de Vroomen"/><br /><sub><b>Tom de Vroomen</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=tomdevroomen" title="Code">💻</a> <a href="#example-tomdevroomen" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=tomdevroomen" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=tomdevroomen" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/memdal"><img src="https://avatars.githubusercontent.com/u/77838566?v=4?s=100" width="100px;" alt="memdal"/><br /><sub><b>memdal</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=memdal" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=memdal" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/maxplatov"><img src="https://avatars.githubusercontent.com/u/14889696?v=4?s=100" width="100px;" alt="maxplatov"/><br /><sub><b>maxplatov</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/pulls?q=is%3Apr+reviewed-by%3Amaxplatov" title="Reviewed Pull Requests">👀</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/emmanuel-ferdman"><img src="https://avatars.githubusercontent.com/u/35470921?v=4?s=100" width="100px;" alt="Emmanuel Ferdman"/><br /><sub><b>Emmanuel Ferdman</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=emmanuel-ferdman" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://jonjomckay.com"><img src="https://avatars.githubusercontent.com/u/456645?v=4?s=100" width="100px;" alt="Jonjo McKay"/><br /><sub><b>Jonjo McKay</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=jonjomckay" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Ajonjomckay" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=jonjomckay" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/TertiumOrganum1"><img src="https://avatars.githubusercontent.com/u/51286827?v=4?s=100" width="100px;" alt="TertiumOrganum1"/><br /><sub><b>TertiumOrganum1</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=TertiumOrganum1" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=TertiumOrganum1" title="Documentation">📖</a> <a href="https://github.com/asyncapi/modelina/commits?author=TertiumOrganum1" title="Tests">⚠️</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Shriya-Chauhan"><img src="https://avatars.githubusercontent.com/u/78415084?v=4?s=100" width="100px;" alt="Shriya Chauhan"/><br /><sub><b>Shriya Chauhan</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Shriya-Chauhan" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/fr-th"><img src="https://avatars.githubusercontent.com/u/61010557?v=4?s=100" width="100px;" alt="fr-th"/><br /><sub><b>fr-th</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=fr-th" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=fr-th" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/commits?author=fr-th" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/istvan-nagy-epam"><img src="https://avatars.githubusercontent.com/u/15124019?v=4?s=100" width="100px;" alt="Istvan Zoltan Nagy"/><br /><sub><b>Istvan Zoltan Nagy</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=istvan-nagy-epam" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=istvan-nagy-epam" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Manancode"><img src="https://avatars.githubusercontent.com/u/144525586?v=4?s=100" width="100px;" alt="Manan Arora"/><br /><sub><b>Manan Arora</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Manancode" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Manancode" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/timeder-xitaso"><img src="https://avatars.githubusercontent.com/u/149670052?v=4?s=100" width="100px;" alt="timeder-xitaso"/><br /><sub><b>timeder-xitaso</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=timeder-xitaso" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=timeder-xitaso" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Atimeder-xitaso" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Sm0keySa1m0n"><img src="https://avatars.githubusercontent.com/u/26169906?v=4?s=100" width="100px;" alt="Joseph Tarbit"/><br /><sub><b>Joseph Tarbit</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Sm0keySa1m0n" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Sm0keySa1m0n" title="Tests">⚠️</a> <a href="#example-Sm0keySa1m0n" title="Examples">💡</a> <a href="https://github.com/asyncapi/modelina/commits?author=Sm0keySa1m0n" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/rmaced0"><img src="https://avatars.githubusercontent.com/u/33237379?v=4?s=100" width="100px;" alt="Rigo"/><br /><sub><b>Rigo</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=rmaced0" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=rmaced0" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Armaced0" title="Bug reports">🐛</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ThomasVerhoeven1998"><img src="https://avatars.githubusercontent.com/u/35959140?v=4?s=100" width="100px;" alt="Thomas Verhoeven"/><br /><sub><b>Thomas Verhoeven</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=ThomasVerhoeven1998" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=ThomasVerhoeven1998" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3AThomasVerhoeven1998" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=ThomasVerhoeven1998" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/xeho91"><img src="https://avatars.githubusercontent.com/u/18627568?v=4?s=100" width="100px;" alt="Mateusz Kadlubowski"/><br /><sub><b>Mateusz Kadlubowski</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/issues?q=author%3Axeho91" title="Bug reports">🐛</a> <a href="https://github.com/asyncapi/modelina/commits?author=xeho91" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/reinaut-krekels-acerta"><img src="https://avatars.githubusercontent.com/u/93973899?v=4?s=100" width="100px;" alt="Reinaut Krekels"/><br /><sub><b>Reinaut Krekels</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=reinaut-krekels-acerta" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=reinaut-krekels-acerta" title="Tests">⚠️</a> <a href="https://github.com/asyncapi/modelina/issues?q=author%3Areinaut-krekels-acerta" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/412b"><img src="https://avatars.githubusercontent.com/u/12436202?v=4?s=100" width="100px;" alt="Kirill Plyashkevich"/><br /><sub><b>Kirill Plyashkevich</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=412b" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=412b" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Exar04"><img src="https://avatars.githubusercontent.com/u/109629956?v=4?s=100" width="100px;" alt="Dhadve Yash"/><br /><sub><b>Dhadve Yash</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Exar04" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=Exar04" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jadinm"><img src="https://avatars.githubusercontent.com/u/4190063?v=4?s=100" width="100px;" alt="Mathieu Jadin"/><br /><sub><b>Mathieu Jadin</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=jadinm" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=jadinm" title="Tests">⚠️</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/AarishMansur"><img src="https://avatars.githubusercontent.com/u/170957949?v=4?s=100" width="100px;" alt="Aarish mansur"/><br /><sub><b>Aarish mansur</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=AarishMansur" title="Code">💻</a> <a href="https://github.com/asyncapi/modelina/commits?author=AarishMansur" title="Tests">⚠️</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Varadraj75"><img src="https://avatars.githubusercontent.com/u/67114314?v=4?s=100" width="100px;" alt="Varad Raj Agrawal"/><br /><sub><b>Varad Raj Agrawal</b></sub></a><br /><a href="https://github.com/asyncapi/modelina/commits?author=Varadraj75" title="Code">💻</a></td>
    </tr>
  </tbody>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind are welcome!
 readmeEtag: '"b621835e293e958927dd73780dced76feb3c68bb"' readmeLastModified: Sun, 01 Feb 2026 14:26:33 GMT repositoryId: 331620831 description: >- A library for generating typed models based on inputs such as AsyncAPI, OpenAPI, and JSON Schema documents with high customization created: '2021-01-21T12:27:51Z' updated: '2026-02-05T00:48:26Z' language: TypeScript archived: false stars: 417 watchers: 6 forks: 228 owner: asyncapi logo: https://avatars.githubusercontent.com/u/16401334?v=4 license: Apache-2.0 repoEtag: '"c8c4689479f9d2ad64342aac67bf238a96cb387d7ff48451e2ebb1dfdd4a6036"' repoLastModified: Thu, 05 Feb 2026 00:48:26 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/dymmond/ravyn v3: true id: e99674d9cb609f3f13c0249ae3da99b4 repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://ravyn.dev"><img src="https://res.cloudinary.com/dymmond/image/upload/v1759490296/ravyn/img/logo_pb3fis.png" alt='Ravyn'></a>
</p>

<p align="center">
    <em>A next-generation async Python framework for building high-performance APIs, microservices, and web applications with type safety and elegance. 🚀</em>
</p>

<p align="center">
<a href="https://github.com/dymmond/ravyn/actions/workflows/test-suite.yml/badge.svg?event=push&branch=main" target="_blank">
    <img src="https://github.com/dymmond/ravyn/actions/workflows/test-suite.yml/badge.svg?event=push&branch=main" alt="Test Suite">
</a>

<a href="https://pypi.org/project/ravyn" target="_blank">
    <img src="https://img.shields.io/pypi/v/ravyn?color=%2334D058&label=pypi%20package" alt="Package version">
</a>

<a href="https://pypi.org/project/ravyn" target="_blank">
    <img src="https://img.shields.io/pypi/pyversions/ravyn.svg?color=%2334D058" alt="Supported Python versions">
</a>
</p>

---

**Documentation**: [https://ravyn.dev](https://www.ravyn.dev) 📚

**Source Code**: [https://github.com/dymmond/ravyn](https://github.com/dymmond/ravyn)

**The official supported version is always the latest released**.

!!! Info "Coming from Esmerald?"
    If you came looking for Esmerald, you are in the right place. Esmerald was rebranded to Ravyn. All features remain and continue to grow.

---

## Quick Start

Get your first Ravyn API running in minutes.

### Installation

```shell
pip install ravyn[standard]
```

This installs Ravyn with recommended extras. You'll also need an ASGI server:

```shell
pip install uvicorn
```

### Your First API

Create a file called `app.py`:

```python
from ravyn import Ravyn, get, JSONResponse

app = Ravyn()

@app.get("/")
def welcome() -> JSONResponse:
    return JSONResponse({"message": "Welcome to Ravyn!"})

@app.get("/hello/{name}")
def greet(name: str) -> JSONResponse:
    return JSONResponse({"message": f"Hello, {name}!"})
```

### Run It

```shell
uvicorn app:app --reload
```

Visit [http://127.0.0.1:8000/hello/World](http://127.0.0.1:8000/hello/World) and you'll see:

```json
{"message": "Hello, World!"}
```

### Explore the Docs

Ravyn automatically generates interactive API documentation:

- **Swagger UI**: [http://127.0.0.1:8000/docs/swagger](http://127.0.0.1:8000/docs/swagger)
- **ReDoc**: [http://127.0.0.1:8000/docs/redoc](http://127.0.0.1:8000/docs/redoc)
- **Stoplight Elements**: [http://127.0.0.1:8000/docs/elements](http://127.0.0.1:8000/docs/elements)

**Congratulations!** 🎉 You've built your first Ravyn API.

---

## Why Ravyn?

Ravyn combines the best ideas from FastAPI, Django, Flask, and NestJS into a framework designed for real-world applications. from prototypes to enterprise systems.

### Key Features

- **⚡ Fast**: Built on [Lilya](https://lilya.dev/) and [Pydantic](https://pydantic-docs.helpmanual.io/), with async-first design
- **🎯 Type-Safe**: Full Python 3.10+ type hints for better IDE support and fewer bugs
- **🧩 Flexible**: Choose OOP (controllers) or functional style. or mix both
- **🔋 Batteries Included**: Dependency injection, middleware, permissions, schedulers, and more
- **Database Ready**: Native support for [Edgy ORM][edgy_orm] and [Mongoz ODM][mongoz_odm]
- **🧪 Testable**: Built-in test client for easy testing
- **📖 Auto-Documented**: OpenAPI/Swagger docs generated automatically

---

## Core Concepts

### Routes and Handlers

Ravyn uses **decorators** or **Gateway objects** to define routes.

!!! warning "Critical Requirements"
    1. **At least one route is required**: An empty `Ravyn()` application does nothing. You must define routes to handle requests.
    2. **Return types are important**: Always specify return type hints (e.g., `-> dict`, `-> JSONResponse`). Ravyn uses these to:
        - Serialize your data correctly
        - Generate accurate API documentation
        - Validate responses

#### Decorator Style (Recommended for Simple APIs)

```python
from ravyn import Ravyn, get, post

app = Ravyn()

@app.get("/users")
def list_users() -> dict:
    return {"users": ["Alice", "Bob"]}

@app.post("/users")
def create_user(name: str) -> dict:
    return {"created": name}
```

#### Gateway Style (Recommended for Larger Apps)

```python
from ravyn import Ravyn, Gateway, get

@get()
def list_users() -> dict:
    return {"users": ["Alice", "Bob"]}

app = Ravyn(
    routes=[
        Gateway("/users", handler=list_users)
    ]
)
```

!!! tip
    Use decorators for quick prototypes. Use Gateway + Include for scalable, organized applications.

### Dependency Injection

Inject dependencies at any level. from application-wide to individual routes.

```python
from ravyn import Ravyn, Gateway, Inject, Injects, get

def get_database():
    return {"db": "connected"}

@get()
def users(db: dict = Injects()) -> dict:
    return {"users": [], "db_status": db}

app = Ravyn(
    routes=[Gateway("/users", handler=users)],
    dependencies={"db": Inject(get_database)}
)
```

Learn more in the [Dependencies](./dependencies.md) guide.

### Settings Management

Ravyn uses environment-based settings inspired by Django.

```python
from ravyn import RavynSettings
from ravyn.conf.enums import EnvironmentType

class DevelopmentSettings(RavynSettings):
    app_name: str = "My App (Dev)"
    environment: str = EnvironmentType.DEVELOPMENT
    debug: bool = True
```

Load your settings via environment variable:

```shell
# MacOS/Linux
RAVYN_SETTINGS_MODULE='myapp.settings.DevelopmentSettings' uvicorn app:app --reload

# Windows
$env:RAVYN_SETTINGS_MODULE="myapp.settings.DevelopmentSettings"; uvicorn app:app --reload
```

If no `RAVYN_SETTINGS_MODULE` is set, Ravyn uses sensible defaults.

Learn more in [Application Settings](./application/settings.md).

---

## Organizing Larger Applications

As your app grows, use **Include** to organize routes into modules.

### Project Structure

```
myapp/
├── app.py
├── urls.py
└── accounts/
    ├── controllers.py
    └── urls.py
```

### accounts/controllers.py

```python
from ravyn import get, post

@get()
def list_accounts() -> dict:
    return {"accounts": []}

@post()
def create_account(name: str) -> dict:
    return {"created": name}
```

### accounts/urls.py

```python
from ravyn import Gateway
from .controllers import list_accounts, create_account

route_patterns = [
    Gateway("/", handler=list_accounts),
    Gateway("/create", handler=create_account),
]
```

### urls.py

```python
from ravyn import Include

route_patterns = [
    Include("/accounts", namespace="myapp.accounts.urls"),
]
```

### app.py

```python
from ravyn import Ravyn

app = Ravyn(routes="myapp.urls")
```

Now your routes are organized:

- `GET /accounts/` → list_accounts
- `POST /accounts/create` → create_account

Learn more in [Routing](./routing/routes.md).

---

## Additional Installation Options

### Testing Support

```shell
pip install ravyn[test]
```

Includes the `RavynTestClient` for testing your application.

### JWT Support

```shell
pip install ravyn[jwt]
```

For JWT-based authentication.

### Scheduler Support

```shell
pip install ravyn[schedulers]
```

For background task scheduling.

### Interactive Shell

```shell
pip install ravyn[ipython]  # IPython shell
pip install ravyn[ptpython]  # ptpython shell
```

Learn more about the [shell](./directives/shell.md).

---

## Start a Project with Scaffolding

!!! warning
    This is for users comfortable with Python project structures. If you're new to Ravyn, continue learning the basics first.

Generate a **simple** project scaffold:

```shell
ravyn createproject myproject --simple
```

Or generate a **complete** scaffold (recommended for enterprise apps):

```shell
ravyn createproject myproject
```

This creates a ready-to-go structure with:

- Pre-configured application
- Sample routes
- Test setup

Learn more in [Directives](./directives/directives.md).

---

## Next Steps

Now that you have Ravyn running, explore these topics:

### Essential Concepts
- [Dependencies](./dependencies.md) - Master dependency injection
- [Routing](./routing/routes.md) - Advanced routing patterns
- [Responses](./responses.md) - Different response types
- [Testing](./testclient.md) - Test your application

### Building Features
- [Middleware](./middleware/index.md) - Add request/response processing
- [Permissions](./permissions/index.md) - Secure your endpoints
- [Database Integration](./databases/edgy/motivation.md) - Connect to databases
- [Background Tasks](./background-tasks.md) - Run async tasks

### Going to Production
- [Settings](./application/settings.md) - Environment configuration
- [Deployment](./deployment/intro.md) - Deploy your application
- [OpenAPI Configuration](./configurations/openapi/config.md) - Customize API docs

---

## Requirements

- **Python 3.10+**

Ravyn is built on:

- <a href="https://lilya.dev/" class="external-link" target="_blank">Lilya</a> - High-performance ASGI framework
- <a href="https://pydantic-docs.helpmanual.io/" class="external-link" target="_blank">Pydantic</a> - Data validation

---

## About Ravyn

### History

Ravyn is the evolution of Esmerald, rebranded to align with a growing ecosystem of tools. **Esmerald continues to exist** for its specific use cases, while Ravyn represents the next generation with improved consistency and future-focused design.

### Motivation

While frameworks like FastAPI, Flask, and Django solve 99% of common problems, they sometimes leave gaps in structure and business logic organization. Ravyn was built to fill those gaps while keeping the best features from:

- **FastAPI** - API design and automatic documentation
- **Django** - Permissions and settings management
- **Flask** - Simplicity and flexibility
- **NestJS** - Controllers and dependency injection
- **Starlite** - Transformers and signature models

Learn more in [About Ravyn](./about.md).

---

## Join the Community

Ravyn is an open source project and we love your contribution!

<p align="center">
    <a href="https://github.com/dymmond/ravyn" target="_blank">
        <img src="https://img.shields.io/github/stars/dymmond/ravyn?style=social" alt="GitHub stars">
    </a>
    <a href="https://discord.gg/eMrM9sWWvu" target="_blank">
        <img src="https://img.shields.io/discord/1018998928332570634?logo=discord&logoColor=ffffff&color=7389D8&labelColor=6A7EC2" alt="Discord">
    </a>
    <a href="https://twitter.com/ravyn_framework" target="_blank">
        <img src="https://img.shields.io/twitter/follow/ravyn_framework?style=social" alt="Twitter">
    </a>
</p>

- **Star us on GitHub** to show your support! ⭐️
- **Join our Discord** to ask questions and share your projects.
- **Follow us on X (Twitter)** for the latest updates.

## Sponsors

Currently there are no sponsors of Ravyn but you can financially help and support the author though
[GitHub sponsors](https://github.com/sponsors/tarsil) and become a **Special one** or a **Legend**.

### Powered by

Worth mentioning who is helping us.

**JetBrains**

[![JetBrains logo.](https://resources.jetbrains.com/storage/products/company/brand/logos/jetbrains.svg)](https://jb.gg/OpenSourceSupport)

[edgy_orm]: https://ravyn.dev/databases/edgy/motivation
[mongoz_odm]: https://ravyn.dev/databases/mongoz/motivation

[edgy_orm]: ./databases/edgy/motivation.md
[mongoz_odm]: ./databases/mongoz/motivation.md
 readmeEtag: '"21b2980145db1c55fd93cadc9f3973cd6ef7fa99"' readmeLastModified: Thu, 08 Jan 2026 11:36:06 GMT repositoryId: 553553797 description: >- Ravyn combines performance, type safety, and elegance. A next-generation async Python framework for APIs, microservices, and web applications. created: '2022-10-18T11:50:26Z' updated: '2026-02-02T08:25:01Z' language: Python archived: false stars: 384 watchers: 9 forks: 22 owner: dymmond logo: https://avatars.githubusercontent.com/u/65982469?v=4 license: BSD-3-Clause repoEtag: '"f816aeff241f79a00be9f3a7990842d94136dd47b112c1d4d3fa35efa37627b7"' repoLastModified: Mon, 02 Feb 2026 08:25:01 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/dymmond/esmerald - source: openapi3 tags repository: https://github.com/7nohe/openapi-react-query-codegen v3: true id: cb6960da16fa36f3be9fdcb19adfb75f repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFJlYWN0IFF1ZXJ5IENvZGVnZW4KCj4gQ29kZSBnZW5lcmF0b3IgZm9yIGNyZWF0aW5nIFtSZWFjdCBRdWVyeSAoYWxzbyBrbm93biBhcyBUYW5TdGFjayBRdWVyeSldKGh0dHBzOi8vdGFuc3RhY2suY29tL3F1ZXJ5KSBob29rcyBiYXNlZCBvbiB5b3VyIE9wZW5BUEkgc2NoZW1hLgoKWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy8lNDA3bm9oZSUyRm9wZW5hcGktcmVhY3QtcXVlcnktY29kZWdlbi5zdmcpXShodHRwczovL2JhZGdlLmZ1cnkuaW8vanMvJTQwN25vaGUlMkZvcGVuYXBpLXJlYWN0LXF1ZXJ5LWNvZGVnZW4pCgojIyBGZWF0dXJlcwoKLSBHZW5lcmF0ZXMgY3VzdG9tIHJlYWN0IGhvb2tzIHRoYXQgdXNlIFJlYWN0IFF1ZXJ5J3MgYHVzZVF1ZXJ5YCwgYHVzZVN1c3BlbnNlUXVlcnlgLCBgdXNlTXV0YXRpb25gIGFuZCBgdXNlSW5maW5pdGVRdWVyeWAgaG9va3MKLSBHZW5lcmF0ZXMgY3VzdG9tIGZ1bmN0aW9ucyB0aGF0IHVzZSBSZWFjdCBRdWVyeSdzIGBlbnN1cmVRdWVyeURhdGFgIGFuZCBgcHJlZmV0Y2hRdWVyeWAgZnVuY3Rpb25zCi0gR2VuZXJhdGVzIHF1ZXJ5IGtleXMgYW5kIGZ1bmN0aW9ucyBmb3IgcXVlcnkgY2FjaGluZwotIEdlbmVyYXRlcyBwdXJlIFR5cGVTY3JpcHQgY2xpZW50cyBnZW5lcmF0ZWQgYnkgW0BoZXktYXBpL29wZW5hcGktdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9oZXktYXBpL29wZW5hcGktdHMpCg== readmeEtag: '"652c5ea38037709c8ad59e8e4272caaa9637c426"' readmeLastModified: Mon, 14 Oct 2024 14:16:49 GMT repositoryId: 516560962 description: >- OpenAPI React Query Codegen is a code generator for creating React Query (also known as TanStack Query) hooks based on your OpenAPI schema. created: '2022-07-22T00:34:17Z' updated: '2026-02-02T12:28:25Z' language: TypeScript archived: false stars: 423 watchers: 3 forks: 40 owner: 7nohe logo: https://avatars.githubusercontent.com/u/9207663?v=4 license: MIT repoEtag: '"0a019db910ad5392d094683fe70ecb92cae8b48199002385ec0335c000de500c"' repoLastModified: Mon, 02 Feb 2026 12:28:25 GMT category: SDK foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/python-openapi/openapi-core v3: true id: d81cd6bc100ca208f3155bff6bccc824 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWNvcmUKCjxhIGhyZWY9Imh0dHBzOi8vcHlwaS5weXRob24ub3JnL3B5cGkvb3BlbmFwaS1jb3JlIiB0YXJnZXQ9Il9ibGFuayI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3Yvb3BlbmFwaS1jb3JlLnN2ZyIgYWx0PSJQYWNrYWdlIHZlcnNpb24iPgo8L2E+CjxhIGhyZWY9Imh0dHBzOi8vdHJhdmlzLWNpLm9yZy9weXRob24tb3BlbmFwaS9vcGVuYXBpLWNvcmUiIHRhcmdldD0iX2JsYW5rIj4KICAgIDxpbWcgc3JjPSJodHRwczovL3RyYXZpcy1jaS5vcmcvcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1jb3JlLnN2Zz9icmFuY2g9bWFzdGVyIiBhbHQ9IkNvbnRpbnVvdXMgSW50ZWdyYXRpb24iPgo8L2E+CjxhIGhyZWY9Imh0dHBzOi8vY29kZWNvdi5pby9naXRodWIvcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1jb3JlP2JyYW5jaD1tYXN0ZXIiIHRhcmdldD0iX2JsYW5rIj4KICAgIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1jb3JlL21hc3Rlci5zdmc/c3R5bGU9ZmxhdCIgYWx0PSJUZXN0cyBjb3ZlcmFnZSI+CjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly9weXBpLnB5dGhvbi5vcmcvcHlwaS9vcGVuYXBpLWNvcmUiIHRhcmdldD0iX2JsYW5rIj4KICAgIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvcHl2ZXJzaW9ucy9vcGVuYXBpLWNvcmUuc3ZnIiBhbHQ9IlB5dGhvbiB2ZXJzaW9ucyI+CjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly9weXBpLnB5dGhvbi5vcmcvcHlwaS9vcGVuYXBpLWNvcmUiIHRhcmdldD0iX2JsYW5rIj4KICAgIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvZm9ybWF0L29wZW5hcGktY29yZS5zdmciIGFsdD0iUGFja2FnZSBmb3JtYXQiPgo8L2E+CjxhIGhyZWY9Imh0dHBzOi8vcHlwaS5weXRob24ub3JnL3B5cGkvb3BlbmFwaS1jb3JlIiB0YXJnZXQ9Il9ibGFuayI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3N0YXR1cy9vcGVuYXBpLWNvcmUuc3ZnIiBhbHQ9IkRldmVsb3BtZW50IHN0YXR1cyI+CjwvYT4KCiMjIEFib3V0CgpPcGVuYXBpLWNvcmUgaXMgYSBQeXRob24gbGlicmFyeSB0aGF0IHByb3ZpZGVzIGNsaWVudC1zaWRlIGFuZCBzZXJ2ZXItc2lkZSBzdXBwb3J0CmZvciB0aGUgW09wZW5BUEkgdjMuMF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjMubWQpCmFuZCBbT3BlbkFQSSB2My4xXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21haW4vdmVyc2lvbnMvMy4xLjAubWQpIHNwZWNpZmljYXRpb25zLgoKCiMjIEtleSBmZWF0dXJlcwoKLSAqKlZhbGlkYXRpb24qKiBhbmQgKip1bm1hcnNoYWxsaW5nKiogb2YgcmVxdWVzdCBhbmQgcmVzcG9uc2UgZGF0YSAoaW5jbHVkaW5nIHdlYmhvb2tzKQotICoqSW50ZWdyYXRpb24qKiB3aXRoIHBvcHVsYXIgbGlicmFyaWVzIChSZXF1ZXN0cywgV2Vya3pldWcpIGFuZCBmcmFtZXdvcmtzIChEamFuZ28sIEZhbGNvbiwgRmxhc2ssIFN0YXJsZXR0ZSkKLSBDdXN0b21pemF0aW9uIHdpdGggbWVkaWEgdHlwZSAqKmRlc2VyaWFsaXplcnMqKiBhbmQgZm9ybWF0ICoqdW5tYXJzaGFsbGVycyoqCi0gKipTZWN1cml0eSoqIGRhdGEgcHJvdmlkZXJzIChBUEkga2V5cywgQ29va2llLCBCYXNpYywgYW5kIEJlYXJlciBIVFRQIGF1dGhlbnRpY2F0aW9ucykKCgojIyBEb2N1bWVudGF0aW9uCgpDaGVjayBkb2N1bWVudGF0aW9uIHRvIHNlZSBtb3JlIGRldGFpbHMgYWJvdXQgdGhlIGZlYXR1cmVzLiBBbGwgZG9jdW1lbnRhdGlvbiBpcyBpbiB0aGUgImRvY3MiIGRpcmVjdG9yeSBhbmQgb25saW5lIGF0IFtvcGVuYXBpLWNvcmUucmVhZHRoZWRvY3MuaW9dKGh0dHBzOi8vb3BlbmFwaS1jb3JlLnJlYWR0aGVkb2NzLmlvKQoKCiMjIEluc3RhbGxhdGlvbgoKUmVjb21tZW5kZWQgd2F5ICh2aWEgcGlwKToKCmBgYCBjb25zb2xlCnBpcCBpbnN0YWxsIG9wZW5hcGktY29yZQpgYGAKCkFsdGVybmF0aXZlbHkgeW91IGNhbiBkb3dubG9hZCB0aGUgY29kZSBhbmQgaW5zdGFsbCBmcm9tIHRoZSByZXBvc2l0b3J5OgoKYGBgIGNvbnNvbGUKcGlwIGluc3RhbGwgLWUgZ2l0K2h0dHBzOi8vZ2l0aHViLmNvbS9weXRob24tb3BlbmFwaS9vcGVuYXBpLWNvcmUuZ2l0I2VnZz1vcGVuYXBpX2NvcmUKYGBgCgoKIyMgRmlyc3Qgc3RlcHMKCkZpcnN0LCBjcmVhdGUgeW91ciBPcGVuQVBJIG9iamVjdC4KCmBgYCBweXRob24KZnJvbSBvcGVuYXBpX2NvcmUgaW1wb3J0IE9wZW5BUEkKCm9wZW5hcGkgPSBPcGVuQVBJLmZyb21fZmlsZV9wYXRoKCdvcGVuYXBpLmpzb24nKQpgYGAKCk5vdyB5b3UgY2FuIHVzZSBpdCB0byB2YWxpZGF0ZSBhbmQgdW5tYXJzaGFsIGFnYWluc3QgcmVxdWVzdHMgYW5kL29yIHJlc3BvbnNlcy4gCgpgYGAgcHl0aG9uCiMgcmFpc2VzIGFuIGVycm9yIGlmIHRoZSByZXF1ZXN0IGlzIGludmFsaWQKcmVzdWx0ID0gb3BlbmFwaS51bm1hcnNoYWxfcmVxdWVzdChyZXF1ZXN0KQpgYGAKClJldHJpZXZlIHZhbGlkYXRlZCBhbmQgdW5tYXJzaGFsbGVkIHJlcXVlc3QgZGF0YS4KCmBgYCBweXRob24KIyBnZXQgcGFyYW1ldGVycwpwYXRoX3BhcmFtcyA9IHJlc3VsdC5wYXJhbWV0ZXJzLnBhdGgKcXVlcnlfcGFyYW1zID0gcmVzdWx0LnBhcmFtZXRlcnMucXVlcnkKY29va2llc19wYXJhbXMgPSByZXN1bHQucGFyYW1ldGVycy5jb29raWVzCmhlYWRlcnNfcGFyYW1zID0gcmVzdWx0LnBhcmFtZXRlcnMuaGVhZGVycwojIGdldCBib2R5CmJvZHkgPSByZXN1bHQuYm9keQojIGdldCBzZWN1cml0eSBkYXRhCnNlY3VyaXR5ID0gcmVzdWx0LnNlY3VyaXR5CmBgYAoKVGhlIHJlcXVlc3Qgb2JqZWN0IHNob3VsZCBpbXBsZW1lbnQgdGhlIE9wZW5BUEkgUmVxdWVzdCBwcm90b2NvbC4gQ2hlY2sgW0ludGVncmF0aW9uc10oaHR0cHM6Ly9vcGVuYXBpLWNvcmUucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0L2ludGVncmF0aW9ucy5odG1sKSB0byBmaW5kIG9mZmljaWFsbHkgc3VwcG9ydGVkIGltcGxlbWVudGF0aW9ucy4KCkZvciBtb3JlIGRldGFpbHMgcmVhZCBhYm91dCB0aGUgW1VubWFyc2hhbGxpbmddKGh0dHBzOi8vb3BlbmFwaS1jb3JlLnJlYWR0aGVkb2NzLmlvL2VuL2xhdGVzdC91bm1hcnNoYWxsaW5nLmh0bWwpIHByb2Nlc3MuCgpJZiB5b3UganVzdCB3YW50IHRvIHZhbGlkYXRlIHlvdXIgcmVxdWVzdC9yZXNwb25zZSBkYXRhIHdpdGhvdXQgdW5tYXJzaGFsbGluZywgcmVhZCBhYm91dCBbVmFsaWRhdGlvbl0oaHR0cHM6Ly9vcGVuYXBpLWNvcmUucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0L3ZhbGlkYXRpb24uaHRtbCkgaW5zdGVhZC4KCgojIyBSZWxhdGVkIHByb2plY3RzCgotIFtvcGVuYXBpLXNwZWMtdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20vcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1zcGVjLXZhbGlkYXRvcikKICA6IEEgUHl0aG9uIGxpYnJhcnkgdGhhdCB2YWxpZGF0ZXMgT3BlbkFQSSBTcGVjcyBhZ2FpbnN0IHRoZSBPcGVuQVBJIDIuMCAoYWthIFN3YWdnZXIpLCBPcGVuQVBJIDMuMCwgYW5kIE9wZW5BUEkgMy4xIHNwZWNpZmljYXRpb24uIFRoZSB2YWxpZGF0b3IgYWltcyB0byBjaGVjayBmb3IgZnVsbCBjb21wbGlhbmNlIHdpdGggdGhlIFNwZWNpZmljYXRpb24uCi0gW29wZW5hcGktc2NoZW1hLXZhbGlkYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvcikKICA6IEEgUHl0aG9uIGxpYnJhcnkgdGhhdCB2YWxpZGF0ZXMgc2NoZW1hIGFnYWluc3QgdGhlIE9wZW5BUEkgU2NoZW1hIFNwZWNpZmljYXRpb24gdjMuMCBhbmQgT3BlbkFQSSBTY2hlbWEgU3BlY2lmaWNhdGlvbiB2My4xLgotIFtib3R0bGUtb3BlbmFwaS0zXShodHRwczovL2dpdGh1Yi5jb20vY29wZS1zeXN0ZW1zL2JvdHRsZS1vcGVuYXBpLTMpCiAgOiBPcGVuQVBJIDMuMCBTdXBwb3J0IGZvciB0aGUgQm90dGxlIFdlYiBGcmFtZXdvcmsKLSBbcHlyYW1pZF9vcGVuYXBpM10oaHR0cHM6Ly9naXRodWIuY29tL25pdGVvd2ViL3B5cmFtaWRfb3BlbmFwaTMpCiAgOiBQeXJhbWlkIGFkZG9uIGZvciBPcGVuQVBJMyB2YWxpZGF0aW9uIG9mIHJlcXVlc3RzIGFuZCByZXNwb25zZXMuCi0gW3Rvcm5hZG8tb3BlbmFwaTNdKGh0dHBzOi8vZ2l0aHViLmNvbS9jb3JyZWwvdG9ybmFkby1vcGVuYXBpMykKICA6IFRvcm5hZG8gT3BlbkFQSSAzIHJlcXVlc3QgYW5kIHJlc3BvbnNlIHZhbGlkYXRpb24gbGlicmFyeS4KCiMjIExpY2Vuc2UKClRoZSBwcm9qZWN0IGlzIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgQlNEIDMtQ2xhdXNlIExpY2Vuc2UuCg== readmeEtag: '"4021788dc08b815edec8f27145970bdd1be1a4cd"' readmeLastModified: Wed, 12 Mar 2025 00:29:49 GMT repositoryId: 104200746 description: >- Openapi-core is a Python library that adds client-side and server-side support for the OpenAPI v3.0 and OpenAPI v3.1 specification. created: '2017-09-20T10:25:35Z' updated: '2026-02-04T13:27:19Z' language: Python archived: false stars: 357 watchers: 6 forks: 136 owner: python-openapi logo: https://avatars.githubusercontent.com/u/126442889?v=4 license: BSD-3-Clause repoEtag: '"a3f33c7414acccbab5d40bfac487787aef7ce35a2a30d0cf45b5db26d090119d"' repoLastModified: Wed, 04 Feb 2026 13:27:19 GMT category: - Data Validators - Parsers oldLocations: - https://github.com/p1c2u/openapi-core foundInMaster: true v3_1: true - source: openapi3 tags repository: https://github.com/epiphone/routing-controllers-openapi v3: true repositoryMetadata: base64Readme: >- # routing-controllers-openapi

[![codecov](https://codecov.io/gh/epiphone/routing-controllers-openapi/branch/master/graph/badge.svg)](https://codecov.io/gh/epiphone/routing-controllers-openapi) [![npm version](https://badge.fury.io/js/routing-controllers-openapi.svg)](https://badge.fury.io/js/routing-controllers-openapi)

Runtime OpenAPI v3 schema generation for [routing-controllers](https://github.com/typestack/routing-controllers).

## Installation

`npm install --save routing-controllers-openapi`

## Usage

```typescript
import { getMetadataArgsStorage } from 'routing-controllers'
import { routingControllersToSpec } from 'routing-controllers-openapi'

// Define your controllers as usual:

@JsonController('/users')
class UsersController {
  @Get('/:userId')
  getUser(@Param('userId') userId: string) {
    // ...
  }

  @HttpCode(201)
  @Post('/')
  createUser(@Body() body: CreateUserBody) {
    // ...
  }
}

// Generate a schema:

const storage = getMetadataArgsStorage()
const spec = routingControllersToSpec(storage)
console.log(spec)
```

prints out the following specification:

```json
{
  "components": {
    "schemas": {}
  },
  "info": {
    "title": "",
    "version": "1.0.0"
  },
  "openapi": "3.0.0",
  "paths": {
    "/users/{userId}": {
      "get": {
        "operationId": "UsersController.getUser",
        "parameters": [
          {
            "in": "path",
            "name": "userId",
            "required": true,
            "schema": {
              "type": "string"
            }
          }
        ],
        "responses": {
          "200": {
            "content": {
              "application/json": {}
            },
            "description": "Successful response"
          }
        },
        "summary": "List users",
        "tags": ["Users"]
      }
    },
    "/users/": {
      "post": {
        "operationId": "UsersController.createUser",
        "requestBody": {
          "content": {
            "application/json": {
              "schema": {
                "$ref": "#/components/schemas/CreateUserBody"
              }
            }
          },
          "description": "CreateUserBody",
          "required": false
        },
        "responses": {
          "201": {
            "content": {
              "application/json": {}
            },
            "description": "Successful response"
          }
        },
        "summary": "Create user",
        "tags": ["Users"]
      }
    }
  }
}
```

Check [`/sample`](/sample) for a complete sample application.

### Configuration

`routingControllersToSpec` has the following type signature:

```typescript
export function routingControllersToSpec(
  storage: MetadataArgsStorage,
  routingControllerOptions: RoutingControllersOptions = {},
  additionalProperties: Partial<OpenAPIObject> = {}
): OpenAPIObject
```

`routingControllerOptions` refers to the options object used to configure routing-controllers. Pass in the same options here to have your [`routePrefix`](https://github.com/typestack/routing-controllers/#prefix-all-controllers-routes) and [`defaults`](https://github.com/typestack/routing-controllers/#default-settings) options reflected in the resulting OpenAPI spec.

`additionalProperties` is a partial [OpenAPI object](https://swagger.io/specification/#openapi-object-17) that gets merged into the result spec. You can for example set your own [`info`](https://swagger.io/specification/#openapi-object-19) or [`components`](https://swagger.io/specification/#components-object-33) keywords here.

### Validation classes

Use [class-validator-jsonschema](https://github.com/epiphone/class-validator-jsonschema) to convert your validation classes into OpenAPI-compatible schemas:

```typescript
import { validationMetadatasToSchemas } from 'class-validator-jsonschema'

// ...

const schemas = validationMetadatasToSchemas({
  refPointerPrefix: '#/components/schemas/',
})

const spec = routingControllersToSpec(storage, routingControllerOptions, {
  components: { schemas },
  info: { title: 'My app', version: '1.2.0' },
})
```

### Decorating with additional keywords

Use the `@OpenAPI` decorator to supply your actions with additional keywords:

```typescript
import { OpenAPI } from 'routing-controllers-openapi'

@JsonController('/users')
export class UsersController {
  @Get('/')
  @OpenAPI({
    description: 'List all available users',
    responses: {
      '400': {
        description: 'Bad request',
      },
    },
  })
  listUsers() {
    // ...
  }
}
```

The parameter object consists of any number of properties from the [Operation object](https://swagger.io/specification/#operationObject). These properties are then merged into the spec, overwriting any existing values.

Alternatively you can call `@OpenAPI` with a function of type `(source: OperationObject, route: IRoute) => OperationObject`, i.e. a function receiving the existing spec as well as the target route, spitting out an updated spec. This function parameter can be used to implement for example your own merging logic or custom decorators.

#### Multiple `@OpenAPI` decorators

A single handler can be decorated with multiple `@OpenAPI`s. Note though that since decorators are applied top-down, any possible duplicate keys are overwritten by subsequent decorators:

```typescript
  @OpenAPI({
    summary: 'This value will be overwritten!',
    description: 'This value will remain'
  })
  @OpenAPI({
    summary: 'This value will remain'
  })
  listUsers() {
    // ...
  }
```

Multiple `@OpenAPI`s are merged together with [`lodash/merge`](https://lodash.com/docs/4.17.11#merge) which has [a few interesting properties](https://github.com/lodash/lodash/issues/1313) to keep in mind when it comes to arrays. Use the function parameter described above when strict control over merging logic is required.

#### Class `@OpenAPI` decorator

Using `@OpenAPI` on the controller class effectively applies given spec to each class method. Method-level `@OpenAPI`s are merged into class specs, with the former having precedence:

```typescript
@OpenAPI({
  security: [{ basicAuth: [] }], // Applied to each method
})
@JsonController('/users')
export class UsersController {
  // ...
}
```

### Annotating response schemas

Extracting response types automatically in runtime isn't currently allowed by Typescript's reflection system. Specifically the problem is that `routing-controllers-openapi` can't unwrap generic types like Promise<MyModel> or Array<MyModel>: see e.g. [here](https://github.com/Microsoft/TypeScript/issues/10576) for discussion. As a workaround you can use the `@ResponseSchema` decorator to supply the response body schema:

```typescript
import { ResponseSchema } from 'routing-controllers-openapi'

@JsonController('/users')
export class UsersController {
  @Get('/:id')
  @ResponseSchema(User)
  getUser() {
    // ...
  }
}
```

`@ResponseSchema` takes as an argument either a class-validator class or a plain string schema name. You can also supply an optional secondary `options` argument:

```typescript
  @Post('/')
  @ResponseSchema(User, {
    contentType: 'text/csv',
    description: 'A list of created user objects',
    isArray: true
    statusCode: '201'})
  createUsers() {
    // ...
  }
```

`contentType` and `statusCode` default to routing-controller's `@ContentType` and `@HttpCode` values. To specify a response schema of an array, set `options.isArray` as `true`. You can also annotate a single handler with multiple `ResponseSchema`s to specify responses with different status codes.

Note that when using `@ResponseSchema` together with `@JSONSchema`, the outer decorator will overwrite keys of inner decorators. So in the following example, information from `@ResponseSchema` would be overwritten by `@JSONSchema`:

```typescript
@JSONSchema({responses: {
  '200': {
    'content': {
      'application/json': {
        schema: {
          '$ref': '#/components/schemas/Pet'
        }
      }
    }
  }
}})
@ResponseSchema(SomeResponseObject)
handler() { ... }
```

#### Multiple ResponseSchemas

Multiple ResponseSchemas with different status codes are supported as follows.

```typescript
@ResponseSchema(Response1)
@ResponseSchema(Response2, {statusCode: '400'})
```

In case of multiple ResponseSchemas being registered with the same status code, we resolve them
using the [oneOf](https://swagger.io/docs/specification/data-models/oneof-anyof-allof-not/#oneof) operator.

```typescript
@ResponseSchema(Response1)
@ResponseSchema(Response2)
```

will generate

```json
"200": {
  "content": {
    "application/json":{
      "schema": {
        "oneOf": [
          {$ref: "#/components/schemas/Response1"},
          {$ref: "#/components/schemas/Response2"}
        ]
      }
    }
  }
}
```

## Supported features

- `@Controller`/`@JsonController` base route and default content-type
- `options.routePrefix`
- `@Get`, `@Post` and other action decorators
- Parse path parameters straight from path strings and optionally supplement with `@Param` decorator
  - Regex and optional path parameters (e.g. `/users/:id(\d+)/:type?`) are also supported
- `@QueryParam` and `@QueryParams`
- `@HeaderParam` and `@HeaderParams`
- `@Body` and `@BodyParam`
- Parse response keywords from `@HttpCode` and `@ContentType` values
- Global `options.defaults.paramOptions.required` option and local override with `{required: true}` in decorator params
- Parse `summary`, `operationId` and `tags` keywords from controller/method names

## Future work

- Support for routing-controller's [authorization features](https://github.com/typestack/routing-controllers#using-authorization-features)

Feel free to submit a PR!

## Related projects

- Inspired by [tsoa](https://github.com/lukeautry/tsoa) and [trafficlight](https://github.com/swimlane/trafficlight)
- Include your Mongoose models in the spec with [mongoose-schema-jsonschema](https://github.com/DScheglov/mongoose-schema-jsonschema)
- Generate JSON schema from your Typescript sources with [typescript-json-schema](https://github.com/YousefED/typescript-json-schema)
- [openapi3-ts](https://github.com/metadevpro/openapi3-ts/) provides handy OpenAPI utilities for Typescript
- Convert OpenAPI 3 spec to **Swagger 2** with [api-spec-converter](https://github.com/LucyBot-Inc/api-spec-converter)
- Generate Typescript interface definitions from SQL database schema with [schemats](https://github.com/SweetIQ/schemats)
 readmeEtag: '"4147853ee4d7a12f6b36c044d8039663e340e82c"' readmeLastModified: Sat, 31 Dec 2022 02:24:52 GMT repositoryId: 111706277 description: Runtime OpenAPI v3 schema generation for routing-controllers. created: '2017-11-22T16:13:37Z' updated: '2026-01-19T00:53:56Z' language: TypeScript archived: false stars: 316 watchers: 5 forks: 64 owner: epiphone logo: https://avatars.githubusercontent.com/u/1923531?v=4 license: MIT repoEtag: '"7c976a880dffc147b9e3b6fa2627a2f31fcbf3205e2c3240c8f8749144a187ce"' repoLastModified: Mon, 19 Jan 2026 00:53:56 GMT foundInMaster: true category: Parsers id: 12451687e58e310275a58099d6b7a849 - source: openapi3 tags repository: https://github.com/mermade/openapi-codegen v3: true repositoryMetadata: base64Readme: >- # OpenAPI-CodeGen

[![Build status](https://travis-ci.org/Mermade/openapi-codegen.svg?branch=master)](https://travis-ci.org/Mermade/openapi-codegen)

Node.js-based codegen for OpenAPI documents. This project was initially a 24-hour hackathon. The local model adaptor code is entirely original and has been reverse-engineered from the existing documentation and template usage.

**Work in progress**

Supports OpenAPI 3.0.x natively, and Swagger/OpenAPI 1.2 and 2.0 by internal conversion. Node.js LTS versions are supported.

## Usage

### Installing

`npm i -g openapi-codegen`

or

* clone the repository, and
* `npm i`

or

`npx -p openapi-codegen cg ...`

### CLI

```
cg [options] {[path]configName} {openapi-definition}

Options:
  --help           Show help                                           [boolean]
  --version        Show version number                                 [boolean]
  --filter         Filter term to use with --list                       [string]
  --list           List available templates for provider (og or sc)     [string]
  -d, --debug      Turn on debugging information in the model          [boolean]
  -f, --flat       Do not include config-name in output directory structure
                                                                       [boolean]
  -l, --lint       Lint input definition                               [boolean]
  -o, --output     Specify output directory         [string] [default: "./out/"]
  -s, --stools     Use swagger-tools to validate OpenAPI 2.0 definitions
                                                                       [boolean]
  -t, --templates  Specify templates directory                          [string]
  -v, --verbose    Increase verbosity                                  [boolean]
  -z, --zip        Create a .zip file instead of individual files      [boolean]
```

e.g.

```
node cg --verbose nodejs defs/generator.yaml
```

In this case, the generated code will be written to the `.out/nodejs` directory.

You can also load the OpenAPI definition from a URL.

### API

```javascript
const renderer = require('openapi-codegen');
// load a config and a definition
renderer.main(definition,config,configName);
```

## Templates

The local templates were taken directly from `swagger-codegen`. This project is also licensed under [Apache-2.0](LICENSE) for this reason. Generated code is explicitly covered by the [Unlicense](templates/_common/UNLICENSE). Code to downconvert OpenAPI 3.0 definitions is taken from [Angular-Swagger-UI](https://github.com/Orange-OpenSource/angular-swagger-ui) and is MIT licensed.

You can also use the latest online templates from two providers: `og` ([openapi-generator](https://github.com/OpenAPITools/openapi-generator)) and `sc` ([swagger-codegen](https://github.com/swagger-api/swagger-codegen)). The `--list` and `--filter` options allow you to see which templates are available. Note that using the online templates involves sending your API definition to a remote server.

### Contributors

See [here](https://github.com/swagger-api/swagger-codegen#template-creator) for a partial list of template contributors.

### Status of the template configurations

The local templates with a status have a working (if not necessarily tested) configuration in the **configs** directory. Contributions are welcomed from the community of new and updated configurations and template updates.

<details>
<summary>Click here to expand...</summary>

|Template|Type|Status|README|Authors (TODO)|Config Maintainer|
|---|---|---|---|---|---|
|**\_common**|meta| *contains Apache-2.0 and Unlicense licenses*||
|**Ada**|client|**Untested**
|akka-scala||
|android||
|**apache2**|configuration|**needs work**||
|apex||
|aspnetcore||
|**bash**|client|**Syntax ok, needs testing**||@bkryza|@MikeRalphson
|**clojure**|client|**Untested**|
|**codegen**|meta|**Demo only**|||@MikeRalphson
|**confluenceWikiDocs**|documentation|**Tested** with Docker [server](https://hub.docker.com/r/atlassian/confluence-server/)||
|cpprest||
|csharp||
|**csharp-dotnet2**|client|**Untested**||
|dart||
|**debug**|meta|*used for dumping the model state*||@Mermade|@MikeRalphson
|Eiffel||
|elixir||
|**erlang-client**|client|**Untested**||
|erlang-server|server|
|finch||
|flash||
|**flaskConnexion**|server|**Needs testing**||
|**go**|client|**Builds, needs testing**||
|**go-server**|server|**Builds and runs**||
|**Groovy**|?|**untested**||
|haskell-http-client|client||||
|**haskell-servant**|server|**Untested**||
|**htmlDocs**|documentation|*Appears to work*||
|**htmlDocs2**|documentation|*Appears to work, no console errors logged*||
|Java||
|JavaInflector||
|JavaJaxRS||
|JavaPlayFramework||
|**Javascript**|client|**Untested**||
|**Javascript-Closure-Angular**|client|**Untested**
|JavaSpring||
|JavaVertXServer||
|**JMeter**|meta|**Untested**||
|kotlin-client||
|**lua**|client|**Compiles OK**|
|lumen||
|MSF4J||
|nancyfx||
|**nodejs**|server|**tested** :white_check_mark:||@jfiala|@MikeRalphson|
|objc||
|**openapi**|meta|*outputs the input definition (in OpenAPI 3.0.x form)* :white_check_mark:||@Mermade|@MikeRalphson
|perl||
|php||
|**php-silex**|?|**untested**||
|php-symfony||
|pistache-server||
|powershell||
|**python**|client|**needs testing**|||@mpnordland
|qt5cpp||
|r||
|rails5||
|**restbed**|server|**Untested**||
|ruby||
|rust||
|rust-server||
|scala||
|scalatra||
|scalaz|client|**Untested**||
|**sinatra**|server|**Syntax checks OK**||
|**slim**|server|**Untested**||
|**swagger**|meta|*outputs the input definition (in original form if OpenAPI 2.0)* :white_check_mark:||
|**swagger-static**|documentation|**tested** *template modified to include partials*||
|swift||
|swift3||
|swift4||
|tizen||
|typescript-angular||
|typescript-angularjs||
|**typescript-axios**|client|**tested**||jaredpalmer|
|typescript-aurelia||
|**typescript-fetch**|client|**compiles with tsc ok**||
|typescript-jquery||
|**typescript-node**|client|**compiles with tsc ok**||
|undertow||
|**validator**|meta|*uses swagger2openapi's OpenAPI 3.0 validator internally* :white_check_mark:||
|ze-ph|

### New Templates

These templates are examples of how features of OpenAPI Codegen may be used, and best-practices in naming model properties.

|Template|Type|Status|README|Authors|Config Maintainer|
|---|---|---|---|---|---|
|testing.dredd|testing|**In progress**|[README](templates/testing.dredd/README.md.mustache)|@Mermade|@MikeRalphson|
</details>

## Documentation

* [See here](docs/README.md) - contributions welcome

 readmeEtag: '"ac4bc8812e541e929b3c48cb72ba2966a552431e"' readmeLastModified: Tue, 24 Aug 2021 07:54:24 GMT repositoryId: 110697603 description: OpenAPI 3.0 CodeGen plus Node.js minus the Java and emojis created: '2017-11-14T14:02:19Z' updated: '2025-12-13T13:01:53Z' language: JavaScript archived: false stars: 295 watchers: 10 forks: 46 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: Apache-2.0 repoEtag: '"0a5b0fbcf5db2d45a13996da67b307fa193316437a640c8463a833979755d740"' repoLastModified: Sat, 13 Dec 2025 13:01:53 GMT foundInMaster: true category: Converters id: a5af66e79d95d6e1b9baa57668630864 - source: openapi3 tags repository: https://github.com/abersheeran/kui v3: true id: 74491e84e078ac54aa398a58a3cd5462 repositoryMetadata: base64Readme: >- IyBLdcOtCgpbIVtQeVBJIC0gUHl0aG9uIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9weXZlcnNpb25zL2t1aT9sYWJlbD1TdXBwb3J0JTIwUHl0aG9uJTIwVmVyc2lvbiZzdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9rdWkvKQoKQW4gZWFzeS10by11c2Ugd2ViIGZyYW1ld29yay4gQmFzZWQgb24gW2JhaXplXShodHRwczovL2JhaXplLmFiZXIuc2gpIGFuZCBbcHlkYW50aWNdKGh0dHBzOi8vZG9jcy5weWRhbnRpYy5kZXYvKS4KCiMjIEluc3RhbGwKCmBgYApwaXAgaW5zdGFsbCBrdWkKYGBgCg== readmeEtag: '"599cea9f9be5c132bbf0375e94034edd8053d815"' readmeLastModified: Wed, 31 Jul 2024 06:07:28 GMT repositoryId: 191898408 description: >- An easy-to-use web framework. Supports both WSGI and ASGI modes. Gevent or asyncio, this is the question. created: '2019-06-14T07:39:36Z' updated: '2025-12-31T19:53:57Z' language: Python archived: false stars: 299 watchers: 6 forks: 24 owner: abersheeran logo: https://avatars.githubusercontent.com/u/25364828?v=4 license: Apache-2.0 repoEtag: '"d916ad4a5a8b8f12872e8d58c5941f02bb506fde031540616d83e46db84e7316"' repoLastModified: Wed, 31 Dec 2025 19:53:57 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/index-py/index.py - source: openapi3 tags name: openapi-go homepage: https://github.com/swaggest/openapi-go language: Go source_description: Type-safe OpenAPI 3.x bindings and generator from code category: - Low-level Tooling - Parsers repository: https://github.com/swaggest/openapi-go v3: true repositoryMetadata: base64Readme: >- # OpenAPI structures for Go

<img align="right" width="250px" src="/resources/logo.png">

This library provides Go structures to marshal/unmarshal and reflect [OpenAPI Schema](https://swagger.io/resources/open-api/) documents.

For automated HTTP REST service framework built with this library please check [`github.com/swaggest/rest`](https://github.com/swaggest/rest).

[![Build Status](https://github.com/swaggest/openapi-go/workflows/test/badge.svg)](https://github.com/swaggest/openapi-go/actions?query=branch%3Amaster+workflow%3Atest)
[![Coverage Status](https://codecov.io/gh/swaggest/openapi-go/branch/master/graph/badge.svg)](https://codecov.io/gh/swaggest/openapi-go)
[![GoDevDoc](https://img.shields.io/badge/dev-doc-00ADD8?logo=go)](https://pkg.go.dev/github.com/swaggest/openapi-go)
[![time tracker](https://wakatime.com/badge/github/swaggest/openapi-go.svg)](https://wakatime.com/badge/github/swaggest/openapi-go)
![Code lines](https://sloc.xyz/github/swaggest/openapi-go/?category=code)
![Comments](https://sloc.xyz/github/swaggest/openapi-go/?category=comments)

## Features

* Type safe mapping of OpenAPI 3 documents with Go structures generated from schema.
* Type-based reflection of Go structures to OpenAPI 3.0 or 3.1 schema.
* Schema control with field tags
    * `json` for request bodies and responses in JSON
    * `query`, `path` for parameters in URL
    * `header`, `cookie`, `formData`, `file` for other parameters
    * `form` acts as `query` and `formData`
    * `contentType` indicates body content type
    * [field tags](https://github.com/swaggest/jsonschema-go#field-tags) named after JSON Schema/OpenAPI 3 Schema constraints
    * `collectionFormat` to unpack slices from string
        * `csv` comma-separated values,
        * `ssv` space-separated values,
        * `pipes` pipe-separated values (`|`),
        * `multi` ampersand-separated values (`&`),
        * `json` additionally to slices unpacks maps and structs,
* Flexible schema control with [`jsonschema-go`](https://github.com/swaggest/jsonschema-go#implementing-interfaces-on-a-type)

## Example

[Other examples](https://pkg.go.dev/github.com/swaggest/openapi-go/openapi3#pkg-examples).

```go
reflector := openapi3.Reflector{}
reflector.Spec = &openapi3.Spec{Openapi: "3.0.3"}
reflector.Spec.Info.
    WithTitle("Things API").
    WithVersion("1.2.3").
    WithDescription("Put something here")

type req struct {
    ID     string `path:"id" example:"XXX-XXXXX"`
    Locale string `query:"locale" pattern:"^[a-z]{2}-[A-Z]{2}$"`
    Title  string `json:"string"`
    Amount uint   `json:"amount"`
    Items  []struct {
        Count uint   `json:"count"`
        Name  string `json:"name"`
    } `json:"items"`
}

type resp struct {
    ID     string `json:"id" example:"XXX-XXXXX"`
    Amount uint   `json:"amount"`
    Items  []struct {
        Count uint   `json:"count"`
        Name  string `json:"name"`
    } `json:"items"`
    UpdatedAt time.Time `json:"updated_at"`
}

putOp, err := reflector.NewOperationContext(http.MethodPut, "/things/{id}")
handleError(err)

putOp.AddReqStructure(new(req))
putOp.AddRespStructure(new(resp), func(cu *openapi.ContentUnit) { cu.HTTPStatus = http.StatusOK })
putOp.AddRespStructure(new([]resp), func(cu *openapi.ContentUnit) { cu.HTTPStatus = http.StatusConflict })

reflector.AddOperation(putOp)

getOp, err := reflector.NewOperationContext(http.MethodGet, "/things/{id}")
handleError(err)

getOp.AddReqStructure(new(req))
getOp.AddRespStructure(new(resp), func(cu *openapi.ContentUnit) { cu.HTTPStatus = http.StatusOK })

reflector.AddOperation(getOp)

schema, err := reflector.Spec.MarshalYAML()
if err != nil {
    log.Fatal(err)
}

fmt.Println(string(schema))
```

Output:

```yaml
openapi: 3.0.3
info:
  description: Put something here
  title: Things API
  version: 1.2.3
paths:
  /things/{id}:
    get:
      parameters:
      - in: query
        name: locale
        schema:
          pattern: ^[a-z]{2}-[A-Z]{2}$
          type: string
      - in: path
        name: id
        required: true
        schema:
          example: XXX-XXXXX
          type: string
      responses:
        "200":
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Resp'
          description: OK
    put:
      parameters:
      - in: query
        name: locale
        schema:
          pattern: ^[a-z]{2}-[A-Z]{2}$
          type: string
      - in: path
        name: id
        required: true
        schema:
          example: XXX-XXXXX
          type: string
      requestBody:
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/Req'
      responses:
        "200":
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Resp'
          description: OK
        "409":
          content:
            application/json:
              schema:
                items:
                  $ref: '#/components/schemas/Resp'
                type: array
          description: Conflict
components:
  schemas:
    Req:
      properties:
        amount:
          minimum: 0
          type: integer
        items:
          items:
            properties:
              count:
                minimum: 0
                type: integer
              name:
                type: string
            type: object
          nullable: true
          type: array
        string:
          type: string
      type: object
    Resp:
      properties:
        amount:
          minimum: 0
          type: integer
        id:
          example: XXX-XXXXX
          type: string
        items:
          items:
            properties:
              count:
                minimum: 0
                type: integer
              name:
                type: string
            type: object
          nullable: true
          type: array
        updated_at:
          format: date-time
          type: string
      type: object
```
 readmeEtag: '"23d334cceadd247d93908fbe4cce6a854899fe5e"' readmeLastModified: Fri, 14 Feb 2025 23:19:57 GMT repositoryId: 232207964 description: OpenAPI structures for Go created: '2020-01-07T00:07:41Z' updated: '2026-01-14T19:55:18Z' language: Go archived: false stars: 340 watchers: 4 forks: 31 owner: swaggest logo: https://avatars.githubusercontent.com/u/19609628?v=4 license: MIT repoEtag: '"82a2ffe76bac57a6aad1ee3574699da381d9d032542bff5976c6da96c331d168"' repoLastModified: Wed, 14 Jan 2026 19:55:18 GMT foundInMaster: true id: 042e78fafa5f6cb0dbca01c64778ab3f - source: openapi3 tags repository: https://github.com/swagger-api/swagger-petstore v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIFBldHN0b3JlIFNhbXBsZQoKIyMgT3ZlcnZpZXcKVGhpcyBpcyB0aGUgcGV0IHN0b3JlIHNhbXBsZSBob3N0ZWQgYXQgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby4gRm9yIG90aGVyIHZlcnNpb25zLCBjaGVjayB0aGUgYnJhbmNoZXMuCldlIHdlbGNvbWUgc3VnZ2VzdGlvbiBib3RoIHRoZSBjb2RlIGFuZCB0aGUgQVBJIGRlc2lnbi4KVG8gbWFrZSBjaGFuZ2VzIHRvIHRoZSBkZXNpZ24gaXRzZWxmLCB0YWtlIGEgbG9vayBhdCBodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1wZXRzdG9yZS9ibG9iL21hc3Rlci9zcmMvbWFpbi9yZXNvdXJjZXMvb3BlbmFwaS55YW1sLgoKVGhpcyBpcyBhIGphdmEgcHJvamVjdCB0byBidWlsZCBhIHN0YW5kLWFsb25lIHNlcnZlciB3aGljaCBpbXBsZW1lbnRzIHRoZSBPcGVuQVBJIDMgU3BlYy4gIFlvdSBjYW4gZmluZCBvdXQKbW9yZSBhYm91dCBib3RoIHRoZSBzcGVjIGFuZCB0aGUgZnJhbWV3b3JrIGF0IGh0dHA6Ly9zd2FnZ2VyLmlvLgoKVGhpcyBzYW1wbGUgaXMgYmFzZWQgb24gW3N3YWdnZXItaW5mbGVjdG9yXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1pbmZsZWN0b3IpLCBhbmQgcHJvdmlkZXMgYW4gZXhhbXBsZSBvZiBzd2FnZ2VyIC8gT3BlbkFQSSAzIHBldHN0b3JlLgoKIyMjIFRvIHJ1biAod2l0aCBNYXZlbikKVG8gcnVuIHRoZSBzZXJ2ZXIsIHJ1biB0aGlzIHRhc2s6CgpgYGAKbXZuIHBhY2thZ2UgamV0dHk6cnVuCmBgYAoKVGhpcyB3aWxsIHN0YXJ0IEpldHR5IGVtYmVkZGVkIG9uIHBvcnQgODA4MC4KCiMjIyBUbyBydW4gKHZpYSBEb2NrZXIpCgpFeHBvc2UgcG9ydCA4MDgwIGZyb20gdGhlIGltYWdlIGFuZCBhY2Nlc3MgcGV0c3RvcmUgdmlhIHRoZSBleHBvc2VkIHBvcnQuIFlvdSBjYW4gdGhlbiBhZGQgYW5kIGRlbGV0ZSBwZXRzIGFzIHlvdSBzZWUgZml0LgoKCipFeGFtcGxlKjoKCmBgYApkb2NrZXIgYnVpbGQgLXQgc3dhZ2dlcmFwaS9wZXRzdG9yZTM6dW5zdGFibGUgLgpgYGAKCmBgYApkb2NrZXIgcHVsbCBzd2FnZ2VyYXBpL3BldHN0b3JlMzp1bnN0YWJsZQpkb2NrZXIgcnVuICAtLW5hbWUgc3dhZ2dlcmFwaS1wZXRzdG9yZTMgLWQgLXAgODA4MDo4MDgwIHN3YWdnZXJhcGkvcGV0c3RvcmUzOnVuc3RhYmxlCmBgYAoKCiMjIyBUZXN0aW5nIHRoZSBzZXJ2ZXIKT25jZSBzdGFydGVkLCB5b3UgY2FuIG5hdmlnYXRlIHRvIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGkvdjMvb3BlbmFwaS5qc29uIHRvIHZpZXcgdGhlIFN3YWdnZXIgUmVzb3VyY2UgTGlzdGluZy4KVGhpcyB0ZWxscyB5b3UgdGhhdCB0aGUgc2VydmVyIGlzIHVwIGFuZCByZWFkeSB0byBkZW1vbnN0cmF0ZSBTd2FnZ2VyLgoKIyMjIFVzaW5nIHRoZSBVSQpUaGVyZSBpcyBhbiBIVE1MNS1iYXNlZCBBUEkgdG9vbCBidW5kbGVkIGluIHRoaXMgc2FtcGxlLS15b3UgY2FuIHZpZXcgaXQgaXQgYXQgW2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MF0oaHR0cDovL2xvY2FsaG9zdDo4MDgwKS4gVGhpcyBsZXRzIHlvdSBpbnNwZWN0IHRoZSBBUEkgdXNpbmcgYW4gaW50ZXJhY3RpdmUgVUkuICBZb3UgY2FuIGFjY2VzcyB0aGUgc291cmNlIG9mIHRoaXMgY29kZSBmcm9tIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkK readmeEtag: '"e5785f36018cb57032061dd4aa947a0f4a9cefea"' readmeLastModified: Tue, 19 Mar 2024 10:08:17 GMT repositoryId: 206982975 description: null created: '2019-09-07T14:54:31Z' updated: '2026-01-23T22:29:42Z' language: Java archived: false stars: 315 watchers: 11 forks: 420 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"e3bca214de93bd8f30227b115a2e20f6d702cffcd3026b03dc8f89d780bb4a9b"' repoLastModified: Fri, 23 Jan 2026 22:29:42 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 627140381c10e9bd5f0466dbf6f3b01d - source: openapi3 tags repository: https://github.com/fsprojects/swaggerprovider v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyUHJvdmlkZXIKClshW051R2V0IFZlcnNpb25dKGh0dHBzOi8vYmFkZ2VuLm5ldC9udWdldC92L1N3YWdnZXJQcm92aWRlcildKGh0dHBzOi8vd3d3Lm51Z2V0Lm9yZy9wYWNrYWdlcy9Td2FnZ2VyUHJvdmlkZXIpIFshW051R2V0IERvd25sb2Fkc10oaHR0cHM6Ly9iYWRnZW4ubmV0L251Z2V0L2R0L1N3YWdnZXJQcm92aWRlcildKGh0dHBzOi8vd3d3Lm51Z2V0Lm9yZy9wYWNrYWdlcy9Td2FnZ2VyUHJvdmlkZXIpCgpUaGlzIFN3YWdnZXJQcm92aWRlciBjYW4gYmUgdXNlZCB0byBhY2Nlc3MgUkVTVGZ1bCBBUEkgZ2VuZXJhdGVkIHVzaW5nIFtTd2FnZ2VyLmlvXShodHRwOi8vc3dhZ2dlci5pbykKCkRvY3VtZW50YXRpb246IDxodHRwOi8vZnNwcm9qZWN0cy5naXRodWIuaW8vU3dhZ2dlclByb3ZpZGVyLz4KCioqU2VjdXJpdHk6KiogU1NSRiBwcm90ZWN0aW9uIGlzIGVuYWJsZWQgYnkgZGVmYXVsdC4gRm9yIGxvY2FsIGRldmVsb3BtZW50LCB1c2Ugc3RhdGljIHBhcmFtZXRlciBgU3NyZlByb3RlY3Rpb249ZmFsc2VgLgoKIyMgU3dhZ2dlciBSRVNUZnVsIEFQSSBEb2N1bWVudGF0aW9uIFNwZWNpZmljYXRpb24KClN3YWdnZXIgaXMgYXZhaWxhYmxlIGZvciBBU1AuTkVUIFdlYkFQSSBBUElzIHdpdGggW1N3YXNoYnVja2xlXShodHRwczovL2dpdGh1Yi5jb20vZG9tYWluZHJpdmVuZGV2L1N3YXNoYnVja2xlKS4KQWxzbyBwb3NzaWJsZSB0byBbQ3JlYXRlIGFuIEFTUC5ORVQgQVBJIGFwcCBpbiBBenVyZSBBcHAgU2VydmljZV0oaHR0cHM6Ly9henVyZS5taWNyb3NvZnQuY29tL2VuLXVzL2RvY3VtZW50YXRpb24vYXJ0aWNsZXMvYXBwLXNlcnZpY2UtZG90bmV0LWNyZWF0ZS1hcGktYXBwLykKCiMjIE1haW50YWluZXIocykKCi0gW0BzZXJnZXktdGlob25dKGh0dHBzOi8vZ2l0aHViLmNvbS9zZXJnZXktdGlob24pCgpUaGUgZGVmYXVsdCBtYWludGFpbmVyIGFjY291bnQgZm9yIHByb2plY3RzIHVuZGVyICJmc3Byb2plY3RzIiBpcyBAZnNwcm9qZWN0c2dpdCAtIEYjIENvbW11bml0eSBQcm9qZWN0IEluY3ViYXRpb24gU3BhY2UgKHJlcG8gbWFuYWdlbWVudCkK readmeEtag: '"fc242745c9a4651d8002dabaa1a3509c567387b6"' readmeLastModified: Wed, 24 Dec 2025 19:07:36 GMT repositoryId: 32864387 description: F# generative Type Provider for Swagger created: '2015-03-25T12:54:10Z' updated: '2026-02-01T13:59:58Z' language: F# archived: false stars: 271 watchers: 10 forks: 60 owner: fsprojects logo: https://avatars.githubusercontent.com/u/6001315?v=4 license: Unlicense repoEtag: '"6bc07ffc7146b99a42b8f33a20901a3c4db8acbb19235629624e8bd0afbf69e8"' repoLastModified: Sun, 01 Feb 2026 13:59:58 GMT foundInMaster: true category: SDK id: f1cc73d13ad58d7db2b0a9fb9428dbc3 - source: openapi3 tags repository: https://github.com/testingisdocumenting/znai v3: true repositoryMetadata: base64Readme: >- IyBEaXNjb3JkIFNlcnZlcgoKSm9pbiBgVGVzdGluZyBJcyBEb2N1bWVudGluZ2AgW0Rpc2NvcmQgU2VydmVyXShodHRwczovL2Rpc2NvcmQuZ2cvYUVIYnpYVFg2TikgdG8gYXNrIHF1ZXN0aW9ucyBhbmQgZGlzY3VzcyBmZWF0dXJlcy9idWdzCgojIFpuYWkKCkJ1aWxkIGZ1bmN0aW9uYWwsIG1haW50YWluYWJsZSwgYmVhdXRpZnVsIFVzZXIgR3VpZGVzIHdpdGggYFpuYWlgLgoKKiBNYXJrZG93biB3aXRoIGN1c3RvbSBleHRlbnNpb25zIGFuZCBkb3plbnMgb2YgcGx1Z2luczogCiAgKiBDb250ZW50IGZyb20gZXh0ZXJuYWwgZmlsZXMgd2l0aCBtYXJrZXJzIGFuZCBmaWx0ZXJzIHN1cHBvcnQKICAqIFNpbXBsaWZpZWQgZXh0cmFjdGlvbiBvZiBhIGZ1bmN0aW9uIGJvZHkgY29udGVudCAod29ya2luZyB3aXRoIGV4YW1wbGVzKSAKICAqIEVtYmVkZGluZyBvZiBKYXZhRG9jL1B5RG9jIGRvY3VtZW50YXRpb24gdGV4dCwgcHJlc2VydmluZyBzdHlsZXMKICAqIEJlYXV0aWZ1bCBBUEkgZG9jdW1lbnRhdGlvbiBjYXBhYmlsaXRpZXMgIAogICogVHdvIFNpZGVzIFBhZ2UgTGF5b3V0IE9wdGlvbiB3aXRoIGNvbnZlbmllbnQgZXhhbXBsZSBsYW5ndWFnZXMgc3dpdGNoIAogICogUmljaCB2aXN1YWxzIGxpa2UgZmxvdyBkaWFncmFtcywgY2hhcnRzIGFuZCBjYXJkcwogICogZXRjCiogSW5zdGFudCBwYWdlcyBuYXZpZ2F0aW9uIAoqIExvY2FsIHNlYXJjaAoqIE11bHRpcGxlIGludGVncmF0aW9ucyB0byB3b3JrIHdpdGggUHl0aG9uLCBKYXZhLCBPcGVuQVBJLCBldGMKKiBQcmVzZW50YXRpb24gTW9kZSB0byBhdXRvbWF0aWNhbGx5IHR1cm4geW91ciBkb2N1bWVudGF0aW9uIGludG8gc2xpZGVzLCB1c2luZyB0aGUgc2FtZSBjb250ZW50CiogRGFyay9MaWdodCBydW50aW1lIHRoZW1lcwoqIExldmVyYWdlIGF1dG9tYXRlZCB0ZXN0IHJlc3VsdHMgdG8gZW5yaWNoIHlvdXIgZG9jdW1lbnRhdGlvbiB3aXRoIGFubm90YXRlZCBpbWFnZXMsIFJFU1QgcmVzcG9uc2VzLCBDTEkgb3V0cHV0cywgZXRjCiogR2l0SHViIHBhZ2VzIGZyaWVuZGx5CgohW2dlbmVyYXRlZCBzaXRlIGV4YW1wbGVdKHpuYWktZG9jcy9yZWFkbWUvem5haS1vdmVydmlldy5wbmcpCgpFbWJlZGRpbmcgb2YgW0V4dGVybmFsIFJlc291cmNlc10oaHR0cHM6Ly90ZXN0aW5naXNkb2N1bWVudGluZy5vcmcvem5haS9zbmlwcGV0cy9leHRlcm5hbC1jb2RlLXNuaXBwZXRzKSBhdCBidWlsZCB0aW1lCgohW2dlbmVyYXRlZCBzaXRlIGV4YW1wbGUgb2YgZXh0ZXJuYWwgY29kZV0oem5haS1kb2NzL3JlYWRtZS96bmFpLWV4dGVybmFsLWNvZGUucG5nKQoKUmljaCBWaXN1YWxzIHN1Y2ggYXMgW0NoYXJ0c10oaHR0cHM6Ly90ZXN0aW5naXNkb2N1bWVudGluZy5vcmcvem5haS92aXN1YWxzL2NoYXJ0cyksCltGbG93IERpYWdyYW1zXShodHRwczovL3Rlc3Rpbmdpc2RvY3VtZW50aW5nLm9yZy96bmFpL3Zpc3VhbHMvZmxvdy1kaWFncmFtcykgYW5kCltDYXJkc10oaHR0cHM6Ly90ZXN0aW5naXNkb2N1bWVudGluZy5vcmcvem5haS92aXN1YWxzL2NhcmRzKQoKIVtnZW5lcmF0ZWQgc2l0ZSBleGFtcGxlIG9mIGNoYXJ0c10oem5haS1kb2NzL3JlYWRtZS96bmFpLWNoYXJ0cy5wbmcpCgpEYXJrL0xpZ2h0IG1vZGUgc3dpdGNoaW5nIAoKIVtnZW5lcmF0ZWQgc2l0ZSBleGFtcGxlIG9mIGZsb3cgZGlhZ3JhbV0oem5haS1kb2NzL3JlYWRtZS96bmFpLWZsb3ctZGlhZ3JhbS5wbmcpCgpDYXJkcyBzdXBwb3J0CgohW2dlbmVyYXRlZCBzaXRlIGV4YW1wbGUgb2YgY2FyZHNdKHpuYWktZG9jcy9yZWFkbWUvem5haS1jYXJkcy5wbmcpCgpCZWF1dGlmdWwgd2F5IHRvIGRvY3VtZW50IHlvdXIgQVBJIAoKIVtnZW5lcmF0ZWQgc2l0ZSBleGFtcGxlIG9mIGFwaSBwYXJhbXNdKHpuYWktZG9jcy9yZWFkbWUvem5haS1hcGktcGFyYW1ldGVycy5wbmcpCgpVc2UgW3R3byBzaWRlc10oaHR0cHM6Ly90ZXN0aW5naXNkb2N1bWVudGluZy5vcmcvem5haS9sYXlvdXQvdHdvLXNpZGVzLXRhYnMpIGxheW91dCBvcHRpb24gdG8gcmVuZGVyIGV4YW1wbGVzIGFuZCBzdXBwb3J0aW5nIGluZm9ybWF0aW9uIHNpZGUgYnkgc2lkZQp3aXRoIGNvbnZlbmllbnQgZXhhbXBsZXMgbGFuZ3VhZ2Ugc3dpdGNoCgohW2dlbmVyYXRlZCBzaXRlIGV4YW1wbGUgb2YgdHdvIHNpZGVzIGxheW91dF0oem5haS1kb2NzL3JlYWRtZS96bmFpLXR3by1zaWRlcy10YWJzLnBuZykKCkJsYXppbmcgZmFzdCBsb2NhbCBzZWFyY2ggd2l0aCByZXN1bHRzIHByZXZpZXcKCiFbZ2VuZXJhdGVkIHNpdGUgZXhhbXBsZSBvZiBzZWFyY2hdKHpuYWktZG9jcy9yZWFkbWUvem5haS1zZWFyY2gucG5nKQoKQXV0b21hdGljIFByZXNlbnRhdGlvbiBmcm9tICoqdGhlIHNhbWUgY29udGVudCoqIHdpdGggemVybyBjb25maWcKCiFbZ2VuZXJhdGVkIHNpdGUgZXhhbXBsZSBvZiBwcmVzZW50YXRpb24gbW9kZV0oem5haS1kb2NzL3JlYWRtZS96bmFpLXByZXNlbnRhdGlvbi5wbmcpCgojIFByZXZpZXcKCmBabmFpYCBoYXMgYSBwcmV2aWV3IG1vZGUgdG8gYXV0b21hdGljYWxseSBsaXN0ZW4gdG8gbG9jYWwgZmlsZXMgY2hhbmdlcy4gSXQgYXV0b21hdGljYWxseSBuYXZpZ2F0ZXMgdG8gYSBwYWdlIHRoYXQgd2FzIApjaGFuZ2VkIChkaXJlY3RseSBvciBpbmRpcmVjdGx5KSBhbmQgaGlnaGxpZ2h0cyB0aGUgY2hhbmdlcy4KCiMgRG9jcyBIdWIKCmBabmFpYCBoYXMgYSBmcmVlLCBvcGVuLXNvdXJjZSBlbnRlcnByaXNlIHN1cHBvcnQuIFlvdSBjYW4gaG9zdCBtdWx0aXBsZSBkb2N1bWVudGF0aW9ucyB1c2luZyBzaW5nbGUgc2l0ZS4gSXQgcHJvdmlkZXMKbGFuZGluZyBwYWdlIHRvIGxpc3QgYWxsIHRoZSBjb21wYW55IGlubmVyIGd1aWRlcy4gCgpJdCBhbHNvIGF1dG9tYXRpY2FsbHkgZGV0ZWN0cyBkb2N1bWVudGF0aW9ucyBpbiB0aGUgYnVpbGRzIG91dHB1dHMgYW5kIHVwZGF0ZXMgaG9zdGVkIGRvY3VtZW50YXRpb25zLgoKRXhhbXBsZXMgKGluY2x1ZGluZyB1c2luZyBHb29nbGUgQ2xvdWQpIGFuZCBzZXR1cCBkb2N1bWVudGF0aW9uIGFyZSBjb21pbmcuIFBsZWFzZSBjcmVhdGUgYSBHaXRIdWIgaXNzdWUgaWYgeW91IHdhbnQKdG8gbGVhcm4gbW9yZSBhYm91dCBFbnRlcnByaXNlIHNldHVwLiBJdCB3aWxsIGhlbHAgbWUgcHJpb3JpdGl6ZS4gIAoKIyBSZWFkIE1vcmUKCltabmFpIGRvY3VtZW50YXRpb24gd2Vic2l0ZSBidWlsdCB3aXRoIFpuYWldKGh0dHBzOi8vdGVzdGluZ2lzZG9jdW1lbnRpbmcub3JnL3puYWkvKQoKRm9yIGN1cmlvcywgem5haSB3b3JkIG1lYW5pbmc6IGB6bmFpYCBpcyBhIGB0byBrbm93YCBpbiBVa3JhbmlhbiBhbmQgaXMgYSBwcmVmaXggZm9yIHdvcmRzIGxpa2UgYGtub3dlbGRnZWAgYXJlIGJ1aWx0IG9uLgoKIyBEZXZlbG9wZXJzIFNldHVwCgpbTG9jYWwgQnVpbGRdKGh0dHBzOi8vdGVzdGluZ2lzZG9jdW1lbnRpbmcub3JnL3puYWkvem5haS1kZXZlbG9wbWVudC9sb2NhbC1idWlsZCkK readmeEtag: '"c3f0053680453328de1144f3e127dc430a6acfee"' readmeLastModified: Wed, 10 Jul 2024 02:13:15 GMT repositoryId: 76669415 description: >- Build functional, maintainable, beautiful User Guides with markdown and Znai plugins. Instant pages navigation. Local search. Multiple integrations to work with Python, Java, C++, OpenAPI, etc. Transform "getting started" sections into slideshow for your workshops. Manage multiple documentations with self-deployed znai hub. created: '2016-12-16T16:42:12Z' updated: '2026-02-03T15:56:36Z' language: Java archived: false stars: 275 watchers: 9 forks: 18 owner: testingisdocumenting logo: https://avatars.githubusercontent.com/u/54154955?v=4 license: Apache-2.0 repoEtag: '"d72bb6dc6cd65f4f526458e58b2ad10b1ad94396c640a168979b89a6244e0a6e"' repoLastModified: Tue, 03 Feb 2026 15:56:36 GMT foundInMaster: true category: Documentation id: e96c15d0478cd8c861932cf44e659757 - source: openapi3 tags repository: https://github.com/apibrew/apibrew v3: true id: 7ed00c333f0718c955cbaca68ff2b139 repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CjxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYXBpYnJldy9hcGlicmV3L21hc3Rlci9zdGF0aWMvQXBpQnJldyUyMExvZ28uc3ZnI2doLWxpZ2h0LW1vZGUtb25seSI+CjxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYXBpYnJldy9hcGlicmV3L21hc3Rlci9zdGF0aWMvQXBpQnJldyUyMExvZ28tZGFyay5zdmcjZ2gtZGFyay1tb2RlLW9ubHkiPgo8L3A+CgojIEFwaSBCcmV3ICAgIC0gICAgaHR0cHM6Ly9hcGlicmV3LmlvClshW2J1aWxkXShodHRwczovL2dpdGh1Yi5jb20vYXBpYnJldy9hcGlicmV3L2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnltbC9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vZ2l0aHViLmNvbS9hcGlicmV3L2FwaWJyZXcvYWN0aW9ucy93b3JrZmxvd3MvYnVpbGQueW1sKQpbIVtHbyBSZXBvcnQgQ2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vYXBpYnJldy9hcGlicmV3KV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL2FwaWJyZXcvYXBpYnJldykKWyFbR28gUmVmZXJlbmNlXShodHRwczovL3BrZy5nby5kZXYvYmFkZ2UvZ2l0aHViLmNvbS9hcGlicmV3L2FwaWJyZXcuc3ZnKV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2dpdGh1Yi5jb20vYXBpYnJldy9hcGlicmV3KQpbIVtEb2NrZXIgUHVsbHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZG9ja2VyL3B1bGxzL3Rpc2xpYi9hcGlicmV3KV0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL3Rpc2xpYi9hcGlicmV3KQpbIVtEb2NrZXIgSW1hZ2UgU2l6ZSAobGF0ZXN0IGJ5IGRhdGUpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2RvY2tlci9pbWFnZS1zaXplL3Rpc2xpYi9hcGlicmV3KV0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL3Rpc2xpYi9hcGlicmV3KQohW0dpdEh1YiByZWxlYXNlIChsYXRlc3QgYnkgZGF0ZSldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3YvcmVsZWFzZS9hcGlicmV3L2FwaWJyZXcpCiFbR2l0SHViXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL2FwaWJyZXcvYXBpYnJldykKCkRvY3M6IGh0dHBzOi8vYXBpYnJldy5pby9kb2NzL2dldHRpbmctc3RhcnRlZAoKTmFubyBDb2RlIFJlcG9zaXRvcnk6IGh0dHBzOi8vZ2l0aHViLmNvbS9hcGlicmV3L2FwaWJyZXcvbW9kdWxlcy9uYW5vCgpHZXR0aW5nIFN0YXJ0ZWQgVmlkZW86IGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9NmlKNXFDVWJkZ3MKCgojIyBJbnRyb2R1Y3Rpb24KCioqRXZlcnl0aGluZyBpcyBhICpSZXNvdXJjZSoqKi4gQW5kICoqRXZlcnl0aGluZyBoYXMgYSAqQ1JVRCBBUEkqKioKCldpdGggQVBJIEJyZXcsIHlvdSBjYW4gY3JlYXRlIENSVUQgQVBJcyBmb3IgeW91ciBkYXRhIGluIGEgZmV3IG1pbnV0ZXMKCiMjIyBSdW4geW91ciBpbnN0YW5jZQoKYGBgc2hlbGwKZG9ja2VyIHJ1biAtdiAkKHB3ZCkvZGF0YTovdmFyL2xpYi9wb3N0Z3Jlc3FsL2RhdGEgLXAgOTAwOTo5MDA5IHRpc2xpYi9hcGlicmV3OmZ1bGwtbGF0ZXN0CmBgYAoKWW91ciBpbnN0YW5jZSBpcyByZWFkeTogU3dhZ2dlciBkb2NzOiBodHRwOi8vbG9jYWxob3N0OjkwMDkvZG9jcy9zd2FnZ2VyCgojIyMgQ3JlYXRlIGZpcnN0IHJlc291cmNlCgpDb25maWd1cmUgQXBiciBjbGk6IGh0dHBzOi8vYXBpYnJldy5pby9kb2NzL2NsaQoKY291bnRyeS55bWwKYGBgeWFtbAp0eXBlOiByZXNvdXJjZQpuYW1lOiBCb29rCnByb3BlcnRpZXM6CiAgdGl0bGU6CiAgICB0eXBlOiBTVFJJTkcKICAgIHVuaXF1ZTogdHJ1ZQogICAgcmVxdWlyZWQ6IHRydWUKICBkZXNjcmlwdGlvbjoKICAgIHR5cGU6IFNUUklORwpgYGAKCmBgYGJhc2gKYXBiciBhcHBseSAtZiBjb3VudHJ5LnltbApgYGAKU28geW91IGFyZSByZWFkeSwgeW91IGhhdmUgZnVsbHkgZXN0YWJsaXNoZWQgUmVzdCBBUEkgZm9yIGJvb2sgcmVzb3VyY2UKCjxpbWcgc3JjPSJodHRwczovL2FwaWJyZXcuaW8vZmlsZXMvYm9vay1zd2FnZ2VyLnBuZyIgd2lkdGg9IjMwMCIvPgoKWW91IGNhbiBidWlsZCBlbnRpcmUgYXBwbGljYXRpb24gd2l0aCByZXNvdXJjZXMgYW5kIHJlZmVyZW5jZXMgYmV0d2VlbiB0aGVtIChsaWtlIHJlbGF0aW9ucyBpbiByZWxhdGlvbmFsIGRhdGFiYXNlcykKCiMjIyBDaGFuZ2UgaXRzIGJlaGF2aW91ciB3aXRoIHBvd2VyIG9mIG5hbm8gY29kZQoKRXZlcnl0aGluZyBjYW4gYmUgd3JpdHRlbiBieSByZXNvdXJjZXMsIG5vdD8KKipMZXQncyBleHRlbmQgb3VyIEJvb2sgcmVzb3VyY2Ugd2l0aCBoZWxwIG9mIG5hbm8gY29kZSoqCgpCb29rTG9naWMuanMKYGBgamF2YXNjcmlwdApjb25zdCBib29rID0gcmVzb3VyY2UoJ0Jvb2snKQoKYm9vay5iZWZvcmVDcmVhdGUoKGJvb2spID0+IHsKICBpZiAoIWJvb2suZGVzY3JpcHRpb24pIHsKICAgIGJvb2suZGVzY3JpcHRpb24gPSAnTm8gZGVzY3JpcHRpb24nCiAgfQp9KTsKYGBgCmBgYGJhc2gKYXBiciBkZXBsb3kgLWYgQm9va0xvZ2ljLmpzIC0tb3ZlcnJpZGUKYGBgCgpTZWUgdGhlIGRvY3MgZm9yIG5hbm86IGh0dHBzOi8vYXBpYnJldy5pby9kb2NzL25hbm8KClNvIHdlIGhhdmUgZXh0ZW5kZWQgb3VyIGJvb2sgcmVzb3VyY2Ugd2l0aCBoZWxwIG9mIG5hbm8gY29kZQoKU28sIHdpdGggKipBUEkgQnJldyoqLCB5b3UgY2FuIGNyZWF0ZSB5b3VyIGFwcGxpY2F0aW9uIHdpdGggcmVzb3VyY2VzIGFuZCB5b3UgY2FuIGN1c3RvbWl6ZSBiZWhhdmlvciBvZiB5b3VyIHJlc291cmNlcyB3aXRoIGV4dGVuc2lvbnMKCiMjIEFib3V0CkFQSSBCcmV3IGlzIGEgKipMb3cgQ29kZSBzb2Z0d2FyZSoqIHRoYXQgYWxsb3dzIHRvIGNyZWF0ZSB2YXJpb3VzIEdycGMgYW5kIFJlc3QgQVBJcyBmcm9tIHZhcmlvdXMgZGF0YWJhc2UgcGxhdGZvcm1zCgojIyBGZWF0dXJlcwoKKiAqKipEZWNsYXJhdGl2ZSoqKiAtICpBUEkgQnJldyogaXMgZGVjbGFyYXRpdmUuIFlvdSBjYW4gZGVmaW5lIHlvdXIgc2NoZW1hIGluIGEgZGVjbGFyYXRpdmUgd2F5LCBpdCB3aWxsIGNyZWF0ZSB5b3VyIEFQSXMKKiAqKipMb3cgQ29kZSoqKiAtIFdpdGggQVBJIEJyZXcsIHlvdSBjYW4gY3JlYXRlIEFQSXMgZm9yIHlvdXIgZGF0YSB3aXRob3V0IGNvZGluZy4gQnV0IHlvdSBjYW4gYWxzbyBleHRlbmQgeW91ciBBUElzIHdpdGgKICBleHRlbnNpb25zLCBzbyB5b3UgY2FuIGN1c3RvbWl6ZSBiZWhhdmlvciBvZiB5b3VyIFJlc291cmNlcy9BcGlzCiogKioqUmVzdCBBUEkqKiogLSBBcyB5b3UgY3JlYXRlIHJlc291cmNlcywgUmVzdCBBcGlzIGZvciB0aGVtIGlzIG1hZGUgYXV0b21hdGljYWxseQoqICoqKkdycGMqKiogLSBBcyB5b3UgY3JlYXRlIHJlc291cmNlcywgR3JwYyBBcGlzIGZvciB0aGVtIGlzIG1hZGUgYXV0b21hdGljYWxseQoqICoqKkRhdGFiYXNlIGFnbm9zdGljKioqIC0gQVBJIEJyZXcgaXMgdXNpbmcgUG9zdGdyZXNxbCBkYXRhYmFzZSBieSBkZWZhdWx0LCBidXQgaXQgYWxzbyBzdXBwb3J0cyB2YXJpb3VzIGRhdGFiYXNlcy4gSW5jbHVkaW5nIE1vbmdvLCBNeXNxbCwgUmVkaXMsIGV0Yy4KKiAqKipDUlVEKioqIC0gQ3J1ZCBpcyBvbiB0aGUgaGVhcnQgb2YgQVBJIEJyZXcuCiogKioqU3dhZ2dlcioqKiAtIFN3YWdnZXIgZG9jcyBhcmUgZ2VuZXJhdGVkIGF1dG9tYXRpY2FsbHkKKiAqKipBdXRoZW50aWNhdGlvbioqKiAtIEFQSSBCcmV3IHN1cHBvcnRzIHZhcmlvdXMgYXV0aGVudGljYXRpb24gbWV0aG9kcy4gSW5jbHVkaW5nIEpXVCBhdXRoZW50aWNhdGlvbiBldGMuCiogKioqQXV0aG9yaXphdGlvbioqKiAtIEFQSSBCcmV3IHN1cHBvcnRzIGF1dGhvcml6YXRpb24uIFlvdSBjYW4gZGVmaW5lIHBlcm1pc3Npb25zIGZvciB5b3VyIHJlc291cmNlcwoqICoqKk11bHRpIERhdGFiYXNlKioqIC0gWW91IGNhbiBkZWZpbmUgbXVsdGlwbGUgZGF0YWJhc2VzIGFuZCBkbyBvcGVyYXRpb25zIG9uIHRvcCBvZiB0aGVtCiogKioqU2NhbGFibGUqKiogLSBBUEkgQnJldyBpcyBzY2FsYWJsZS4gWW91IGNhbiBydW4gaXQgb24gbXVsdGlwbGUgaW5zdGFuY2VzLCBhbmQgaXQgd2lsbCB3b3JrIGFzIGV4cGVjdGVkLCBhcyBBUEkgQnJldyBkb2VzIG5vdCBoYXZlIGFueSBkYXRhIGludGVybmFsbHksIHlvdSBjYW4gc2NhbGUgaXQuCiogKioqRXh0ZW5zaWJsZSoqKiAtIEFQSSBCcmV3IGlzIGV4dGVuc2libGUuIFlvdSBjYW4gZXh0ZW5kIHlvdXIgcmVzb3VyY2VzIHdpdGggZXh0ZW5zaW9ucy4gWW91IGNhbiBhbHNvIGV4dGVuZCB5b3VyIEFQSXMgd2l0aCBleHRlbnNpb25zCiogKioqQ0xJIHN1cHBvcnQqKiogLSBBUEkgQnJldyBoYXMgYSBjbGkgdG9vbCB0byBtYW5hZ2UgeW91ciByZXNvdXJjZXMsIGRhdGFTb3VyY2VzLCBldGMuIEl0IGlzIGNhbGxlZCBgYXBicmAKKiAqKipEb2NrZXIqKiogLSBBUEkgQnJldyBpcyBkb2NrZXJpemVkLiBZb3UgY2FuIHJ1biBpdCBvbiBkb2NrZXIKKiAqKipEb2NrZXIgQ29tcG9zZSoqKiAtIEFQSSBCcmV3IGlzIGRvY2tlci1jb21wb3NlIHJlYWR5LiBZb3UgY2FuIHJ1biBpdCBvbiBkb2NrZXItY29tcG9zZQoqICoqKkt1YmVybmV0ZXMqKiogLSBBUEkgQnJldyBpcyBrdWJlcm5ldGVzIHJlYWR5LiBZb3UgY2FuIHJ1biBpdCBvbiBrdWJlcm5ldGVzCgojIyBVc2UgQ2FzZXMKCiogQ3JlYXRpbmcgYmFja2VuZCBmb3IgeW91ciBtb2JpbGUgYXBwbGljYXRpb24gb3Igd2Vic2l0ZQoqIENyZWF0aW5nIGJhY2tlbmQgZm9yIHlvdXIgZXhpc3RpbmcgZGF0YWJhc2UKKiBNYW5hZ2luZyB5b3VyIGRhdGEgaW4gYSBDUlVEIGZhc2hpb24KKiBDcmVhdGluZyBTdGFuZGFyZGl6ZWQsIHdlbGwgZG9jdW1lbnRlZCBBUElzIGZvciB5b3VyIGRhdGEK readmeEtag: '"84c352cdc7641ef48c579a8a49fd1a3f8d14cfc1"' readmeLastModified: Wed, 07 May 2025 20:49:03 GMT repositoryId: 591665560 description: APIBrew is Low code software to automate building CRUDs from yaml files created: '2023-01-21T13:22:13Z' updated: '2025-12-23T01:05:25Z' language: Go archived: false stars: 252 watchers: 3 forks: 8 owner: apibrew logo: https://avatars.githubusercontent.com/u/132176743?v=4 license: MIT repoEtag: '"2096005d5b58c1b0f6f7611fe4ae3f53caedd39582cb20d9c7ed5d012bad1bc4"' repoLastModified: Tue, 23 Dec 2025 01:05:25 GMT category: Server Implementations oldLocations: - https://github.com/tislib/apibrew foundInMaster: true - source: openapi3 tags repository: https://github.com/speakeasy-api/speakeasy v3: true id: b04ab57cb5fd068ea45591850bba85fc repositoryMetadata: base64Readme: >- <div align="center">
 <a href="https://www.speakeasy.com/" target="_blank">
  <img width="1500" height="500" alt="Speakeasy" src="https://github.com/user-attachments/assets/0e56055b-02a3-4476-9130-4be299e5a39c" />
 </a>
 <br />
 <br />
  <div>
   <a href="https://speakeasy.com/docs/create-client-sdks/" target="_blank"><b>Docs Quickstart</b></a>&nbsp;&nbsp;//&nbsp;&nbsp;<a href="https://go.speakeasy.com/slack" target="_blank"><b>Join us on Slack</b></a>
  </div>
 <br />

 <br />

[![LW24 participant](https://img.shields.io/badge/featured-LW24-8957E5.svg?style=flat-square&labelColor=0D1117&logo=data:image/svg%2bxml;base64,PHN2ZyB3aWR0aD0iMzYwIiBoZWlnaHQ9IjM2MCIgdmlld0JveD0iMCAwIDM2MCAzNjAiIGZpbGw9Im5vbmUiIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyI+IDxyZWN0IHdpZHRoPSI2MCIgaGVpZ2h0PSIzMDAiIGZpbGw9IndoaXRlIi8+IDxyZWN0IHg9IjYwIiB5PSIzMDAiIHdpZHRoPSIxMjAiIGhlaWdodD0iNjAiIGZpbGw9IndoaXRlIi8+IDxyZWN0IHg9IjI0MCIgeT0iMzAwIiB3aWR0aD0iNjAiIGhlaWdodD0iNjAiIGZpbGw9IndoaXRlIi8+IDxyZWN0IHg9IjMwMCIgd2lkdGg9IjYwIiBoZWlnaHQ9IjMwMCIgZmlsbD0id2hpdGUiLz4gPHJlY3QgeD0iMTgwIiB3aWR0aD0iNjAiIGhlaWdodD0iMzAwIiBmaWxsPSJ3aGl0ZSIvPiA8L3N2Zz4=)](https://launchweek.dev/lw/2024/mega#participants)
  
</div>

<hr />
<br />
<div align="left">
  <h4><b>Trusted By:</b></h4>
  <h4>
   <p>
    <a href="https://github.com/mistralai/client-python" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/bca19603-0a18-4425-8d0e-aabc81849e74">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/a5cde465-8be4-4865-b1d3-03222cdea0a1">
      <img height="44px" src="https://github.com/user-attachments/assets/bca19603-0a18-4425-8d0e-aabc81849e74#gh-light-mode-only" alt="Mistral">
     </picture>
    </a>
    &nbsp; &nbsp; &nbsp; &nbsp;
    <a href="https://github.com/vercel/sdk" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/9f499468-d679-43d6-9a4c-bb6d8391ebeb">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/7d1b97ed-22d8-4dff-bac5-66fe5dead2fe">
      <img height="44px" src="https://github.com/user-attachments/assets/9f499468-d679-43d6-9a4c-bb6d8391ebeb#gh-light-mode-only" alt="Vercel">
     </picture>
     </a>
    &nbsp; &nbsp; &nbsp; &nbsp;
    <a href="https://github.com/gleanwork/api-client-python" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/79db6f55-7061-41c5-9ba0-dd66eed86743">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/a0f0852f-a491-450b-9183-94ad6f77da3c">
      <img height="44px" src="https://github.com/user-attachments/assets/79db6f55-7061-41c5-9ba0-dd66eed86743#gh-light-mode-only" alt="Glean">
     </picture>
    </a>
    &nbsp; &nbsp; &nbsp; &nbsp;
    <a href="https://github.com/clerk/clerk-sdk-python" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/cc42df48-e54a-4c2b-bb54-83033a806e38">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/dc4ee90f-7a48-4d7c-abe0-fd347addf785">
      <img height="44px" src="https://github.com/user-attachments/assets/cc42df48-e54a-4c2b-bb54-83033a806e38#gh-light-mode-only" alt="Clerk">
     </picture>
    </a>
    &nbsp; &nbsp; &nbsp; &nbsp;
    <a href="https://github.com/OpenRouterTeam/typescript-sdk" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/d7db99df-7882-481f-93e2-50e7acf51b92">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/1b63bec3-bd07-41d2-97c7-70e9f3bb1fb5">
      <img height="44px" src="https://github.com/user-attachments/assets/d7db99df-7882-481f-93e2-50e7acf51b92#gh-light-mode-only" alt="OpenRouter">
     </picture>
    </a>

      
   </p>
    <a href="https://speakeasy.com/customers"><p>
      Full showcase →
    </p></a>
  </h4>
</div>
<br />

## A Modern OpenAPI Native Toolchain

<p>Polished and type-safe SDKs, Terraform providers and Contract Tests for your API. 10 Languages and counting.</p>
<a href="https://app.speakeasy.com/"><img src="https://custom-icon-badges.demolab.com/badge/-Start%20Generating%20-000?style=for-the-badge&logoColor=fffff&logo=speakeasy-api&labelColor=545454" alt="Start Generating"/></a>
<a href="https://youtu.be/-cSZGUvT5-8?si=VwJBPcOGq0g2R4cI"><img src="https://img.shields.io/static/v1?label=Docs&message=Watch%20Demo&color=000&style=for-the-badge" alt="Watch Demo" /></a>

<div align="left">
  <h1>
   <p>
    <a href="https://www.speakeasy.com/docs/sdk-design/typescript/methodology-ts" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/72392c17-5a6d-4aad-a0e2-b411fc5c733e">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/92990464-eea5-446f-9084-39fc5172e9c2">
      <img width="70px" src="https://github.com/user-attachments/assets/72392c17-5a6d-4aad-a0e2-b411fc5c733e#gh-light-mode-only" alt="TypeScript">
     </picture>
     </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/sdk-design/python/methodology-python" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/51b6860a-ebba-4fa2-8445-95bde33dea7e">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/f9c02d4c-874d-4d4f-b9e7-b8add4447e14">
      <img width="70px" src="https://github.com/user-attachments/assets/51b6860a-ebba-4fa2-8445-95bde33dea7e#gh-light-mode-only" alt="Python">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/sdk-design/golang/methodology-go" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/551a1f03-2ffa-4aa4-b738-966a66967a85">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/fea1f0e8-8876-4246-98f0-ea07aef26f7a">
      <img width="70px" src="https://github.com/user-attachments/assets/551a1f03-2ffa-4aa4-b738-966a66967a85#gh-light-mode-only" alt="Golang">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/create-terraform" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/21f87545-2b49-491b-8239-4513d27f254a">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/e63b652b-ce02-47f6-aa9d-b8fb2ff52f9c">
      <img width="70px" src="https://github.com/user-attachments/assets/21f87545-2b49-491b-8239-4513d27f254a#gh-light-mode-only" alt="Terraform">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/sdk-design/java/methodology-java" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/f14e0e61-df8b-45fb-8634-c7f7a327d24b">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/663376a1-530e-4466-93c0-d6d81106a650">
      <img width="70px" src="https://github.com/user-attachments/assets/f14e0e61-df8b-45fb-8634-c7f7a327d24b#gh-light-mode-only" alt="Java">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/sdk-design/csharp/methodology-csharp" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/0d9aec55-502b-491f-b6ca-5432917e26e0">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/915d6c9c-8f26-492c-92ef-a485064254d9">
      <img width="70px" src="https://github.com/user-attachments/assets/0d9aec55-502b-491f-b6ca-5432917e26e0#gh-light-mode-only" alt="Csharp">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/sdk-design/php/methodology-php" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/b0e1b486-de15-488c-b367-ce1c8c4bdf36">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/dfe55b2c-aaa4-42b0-9b13-592c5964fd5f">
      <img width="70px" src="https://github.com/user-attachments/assets/b0e1b486-de15-488c-b367-ce1c8c4bdf36#gh-light-mode-only" alt="PHP">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/sdk-design/ruby/methodology-ruby" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/87f12e15-e61d-40c4-8690-ccd2dd12268f">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/2206de37-79c7-4f70-947d-7c5c41337edb">
      <img width="70px" src="https://github.com/user-attachments/assets/87f12e15-e61d-40c4-8690-ccd2dd12268f#gh-light-mode-only" alt="Ruby">
     </picture>
    </a>
      &nbsp;
    <a href="https://www.speakeasy.com/docs/languages/unity/methodology-unity" target="_blank">
     <picture>
      <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/1966f876-1ca3-44d7-92c7-c2830e8a32e8">
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/445b5109-1d32-4253-a9f4-fa720da5a490">
      <img width="70px" src="https://github.com/user-attachments/assets/1966f876-1ca3-44d7-92c7-c2830e8a32e8#gh-light-mode-only" alt="Unity">
     </picture>
    </a>
   </p>
  </h1>
</div>

<br/>
<h2>
 <p>How it works</p>
 
   <picture>
    <source media="(prefers-color-scheme: light)" srcset="https://github.com/user-attachments/assets/41f1f1f2-2bf8-4505-83f2-aa4746a0bf95">
    <source media="(prefers-color-scheme: dark)" srcset="https://github.com/user-attachments/assets/18978133-c8b5-46c3-95e4-b68c321f8ae4">
    <img src="https://github.com/user-attachments/assets/41f1f1f2-2bf8-4505-83f2-aa4746a0bf95#gh-light-mode-only" alt="How it works diagram">
   </picture>
  </a>
</h2>
<br />

### Features

- SDK code that looks like you wrote it. Optimised for performance, debuggability and modern idiomatics.
- Complete Terraform Providers built on a Type-safe Go SDK.
- Contract Test generation with a pre built mock-server ([Powered by Arazzo](https://www.speakeasy.com/openapi/arazzo))
- Generate clean code-samples for syncing with API docs.
- Make `npm install your-api`. Manage versioning and publishing to package managers
- Modern OpenAPI 3.X toolchain for linting, cleaning, diff-ing and editing specs. ([Powered by Overlays](https://www.speakeasy.com/openapi/overlays))

Check out the [roadmap](https://www.speakeasyapi.dev/roadmap) for whats coming up soon!
<br />
<br />

## CLI  

### Installation

Install Speakeasy CLI via:

- Homebrew
- Winget
- Chocolatey
- Shell Script / GitHub Actions
- Docker

Refer to the [Speakeasy CLI installation documentation](https://www.speakeasy.com/docs/create-client-sdks#install-the-speakeasy-cli) for more information. CLI releases are also directly available in the [repository releases](https://github.com/speakeasy-api/speakeasy/releases).

### Usage

Refer to the [Speakeasy CLI Reference](https://www.speakeasy.com/docs/speakeasy-reference/cli) for usage documentation. Additionally, every CLI command and subcommand supports a `--help` flag for usage information.

## Agent Skills

Use Speakeasy with AI coding agents via our [Agent Skills](https://github.com/speakeasy-api/agent-skills) package.

### Installation

```bash
npx skills add speakeasy-api/agent-skills
```

### Supported Platforms

- [Claude Code](https://claude.ai/code)
- [Cursor](https://cursor.sh/)
- [GitHub Copilot](https://github.com/features/copilot)
- [Gemini CLI](https://github.com/google-gemini/gemini-cli)
- And [15+ other platforms](https://agentskills.io/)

### Usage

Once installed, skills are available with the `speakeasy:` namespace:

```bash
/speakeasy:start-new-sdk-project     # Initialize a new SDK project
/speakeasy:validate-openapi-spec     # Validate an OpenAPI spec
/speakeasy:regenerate-sdk            # Re-run SDK generation
```

See the [agent-skills repository](https://github.com/speakeasy-api/agent-skills) for the full list of available skills.
 readmeEtag: '"153b67ac4c22e67a33b654e37f268fef37628182"' readmeLastModified: Sat, 24 Jan 2026 00:26:53 GMT repositoryId: 543052826 description: >- Build APIs your users love ❤️ with Speakeasy. ✨ Polished and type-safe SDKs. 🌐 Terraform providers and Contract Tests for your API. OpenAPI native. created: '2022-09-29T10:13:15Z' updated: '2026-02-05T21:45:19Z' language: JavaScript archived: false stars: 381 watchers: 14 forks: 30 owner: speakeasy-api logo: https://avatars.githubusercontent.com/u/91446104?v=4 license: NOASSERTION repoEtag: '"90575a0d5fea755e5878b7317b33563b0f8ec1a5b11e653dfeed3a9e928df505"' repoLastModified: Thu, 05 Feb 2026 21:45:19 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/wrannaman/generators v3: true repositoryMetadata: base64Readme: >- IyBSZXN0L0dyYXBoUUwgQVBJcyArIFJlYWN0IENvbXBvbmVudHMgR2VuZXJhdG9yIPCflKUKClshW0J1eSBMaWdpdCBMaWNlbnNlXShodHRwczovL3MzLnVzLXdlc3QtMS53YXNhYmlzeXMuY29tL3B1YmxpYy5zdWdhcmt1YmVzL2xpZ2l0X2VtYmVkLnN2ZyldKGh0dHBzOi8vbGlnaXQuZGV2KQpbIVtGb2xsb3cgb24gVHdpdHRlcl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90d2l0dGVyL2ZvbGxvdy9hbmRyZXdwaWVybm8uc3ZnP2xhYmVsPWZvbGxvdyldKGh0dHBzOi8vdHdpdHRlci5jb20vYW5kcmV3cGllcm5vKQpbIVtBdmVyYWdlIHRpbWUgdG8gcmVzb2x2ZSBhbiBpc3N1ZV0oaHR0cDovL2lzaXRtYWludGFpbmVkLmNvbS9iYWRnZS9yZXNvbHV0aW9uL3N1Z2Fya3ViZXMvZ2VuZXJhdG9ycy5zdmcpXShodHRwOi8vaXNpdG1haW50YWluZWQuY29tL3Byb2plY3Qvc3VnYXJrdWJlcy9nZW5lcmF0b3JzICJBdmVyYWdlIHRpbWUgdG8gcmVzb2x2ZSBhbiBpc3N1ZSIpClshW1BlcmNlbnRhZ2Ugb2YgaXNzdWVzIHN0aWxsIG9wZW5dKGh0dHA6Ly9pc2l0bWFpbnRhaW5lZC5jb20vYmFkZ2Uvb3Blbi9zdWdhcmt1YmVzL2dlbmVyYXRvcnMuc3ZnKV0oaHR0cDovL2lzaXRtYWludGFpbmVkLmNvbS9wcm9qZWN0L3N1Z2Fya3ViZXMvZ2VuZXJhdG9ycyAiUGVyY2VudGFnZSBvZiBpc3N1ZXMgc3RpbGwgb3BlbiIpClshW25wbSBwYWNrYWdlXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L3N1Z2FyLWdlbmVyYXRlL2xhdGVzdC5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9zdWdhci1nZW5lcmF0ZSkKWyFbTlBNIERvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZHQvc3VnYXItZ2VuZXJhdGUuc3ZnP3N0eWxlPWZsYXQpXShodHRwczovL25wbWNoYXJ0cy5jb20vY29tcGFyZS9zdWdhci1nZW5lcmF0ZT9taW5pbWFsPXRydWUpCgoKIVtTdWdhciBHZW5lcmF0b3IgLSBBUEkgRWRpdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL3N1Z2Fya3ViZXMvZ2VuZXJhdG9ycy9ibG9iL21hc3Rlci9sb2dvLnBuZz9yYXc9dHJ1ZSkKClshW0FwcCBEZW1vXShodHRwczovL2ltZy55b3V0dWJlLmNvbS92aS9FN19BQks3blpUOC8wLmpwZyldKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9RTdfQUJLN25aVDgpCgoKVGhpcyBwcm9qZWN0IGlzIHNwb25zb3JlZCBieQoKWyFbU3BvbnNvcmVkQnkuZGV2XShodHRwczovL2FwaS5zcG9uc29yZWRieS5kZXYvaW1nL2Q4YzQzMDdiLWJhNDItNDAwZS1iNDc1LTk0ODdjNzZkMTVjOC5wbmcpXShodHRwczovL2FwaS5zcG9uc29yZWRieS5kZXYvbGluay9kOGM0MzA3Yi1iYTQyLTQwMGUtYjQ3NS05NDg3Yzc2ZDE1YzgpCgojIFF1aWNrIFN0YXJ0CgoKMS4gSW5zdGFsbCB0aGUgbnBtIG1vZHVsZQoKYGBgc2gKIyBpbnN0YWxsCm5wbSBpIC1nIHN1Z2FyLWdlbmVyYXRlCmBgYAoKMi4gQ3JlYXRlIGEganNvbiBzY2hlbWEgc2F2ZSwgdGhpcyB0byAqKm1vbmtleS5qc29uKioKCgpgYGBqc29uCnsKICAibmFtZSI6ICJtb25rZXkiLAogICJzY2hlbWEiOiB7CiAgICAibmFtZSI6IHsKICAgICAgInR5cGUiOiAiU3RyaW5nIiwKICAgICAgImRlZmF1bHQiOiAiIgogICAgfSwKICAgICJhbGl2ZSI6IHsKICAgICAgInR5cGUiOiAiQm9vbGVhbiIsCiAgICAgICJkZWZhdWx0IjogZmFsc2UKICAgIH0sCiAgICAiYWdlIjogewogICAgICAidHlwZSI6ICJOdW1iZXIiLAogICAgICAiZGVmYXVsdCI6IGZhbHNlCiAgICB9CiAgfQp9CmBgYAoKMy4gR2VuZXJhdGUgeW91ciBhcGkgYW5kIGFwcAoKYGBgc2gKc3VnYXItZ2VuZXJhdGUgXAotLXNjaGVtYSBtb25rZXkuanNvbiBcCi0tZGVzdGluYXRpb24gLi9teS1tb25rZXlzCmBgYAoKQm9vbSwgeW91IG5vdyBoYXZlOgoKQVBJOgotIEdyYXBoUUwgQVBJCi0gUkVTVCBBUEkKLSBXb3JraW5nIFRlc3RzCgpBUFA6Ci0gUmVhY3QgY3JlYXRlIGl0ZW0gZm9ybQotIFJlYWN0IHRhYmxlIHRoYXQgc3VwcG9ydHMKICAtIHNlYXJjaAogIC0gc29ydAogIC0gZmlsdGVyCiAgLSBwYWdpbmF0aW9uCiAgLSBlZGl0IGl0ZW0KICAtIGNyZWF0ZSBpdGVtCgoKIyMjIFN0YXJ0IHRoZSBBUEkKCmBgYHNoCmNkIC4vbXktbW9ua2V5cy9hcGkKbnBtIGkKbnBtIHJ1biBzdGFydAoKIyBodHRwOi8vbG9jYWxob3N0Ojc3NzcKYGBgCgojIyMgU3RhcnQgdGhlIEFQUAoKYGBgc2gKY2QgLi9teS1tb25rZXlzL2FwcApucG0gaQpucG0gcnVuIGRldgoKIyBodHRwOi8vbG9jYWxob3N0OjMwMDAKYGBgCgojIyMgQmVob2xkIE1hZ2ljCgpBIGZ1bGx5IGZ1bmN0aW9uaW5nIHJlYWN0IHRhYmxlIGFuZCBmb3JtIHdpdGggc2VhcmNoaW5nIHNvcnRpbmcgZmlsdGVyaW5nLCBlZGl0aW5nLCBhZGRpbmcsIGdsb2JhbCBzZWFyY2gsIGRvd25sb2FkLCBhbmQgcmVmcmVzaC4KCiFbU3VnYXJLdWJlcyBHZW5lcmF0ZWQgQXBwXShodHRwczovL2dpdGh1Yi5jb20vc3VnYXJrdWJlcy9nZW5lcmF0b3JzL2Jsb2IvbWFzdGVyL3RhYmxlLWNvbXBvbmVudC5wbmc/cmF3PXRydWUpCgojIExpbmtzCgpbR3JhcGhRTCBpcyBvbiBsb2NhbGhvc3Q6Nzc3Ny9ncmFwaHFsXShodHRwOi8vbG9jYWxob3N0Ojc3NzcvZ3JhcGhxbCkKCltTd2FnZ2VyIGlzIG9uIGxvY2FsaG9zdDo3Nzc3XShodHRwOi8vbG9jYWxob3N0Ojc3NzcpCgpbQVBQIGlzIG9uIGxvY2FsaG9zdDozMDAwXShodHRwOi8vbG9jYWxob3N0OjMwMDApCgpbQVBJIGlzIG9uIGxvY2FsaG9zdDo3Nzc3XShodHRwOi8vbG9jYWxob3N0OjMwMDApCgojIERvY3VtZW50YXRpb24KCltBUEkgRG9jdW1lbnRhdGlvbiAoZ2VuZXJhdGVkIGJhY2sgZW5kKV0oaHR0cHM6Ly9naXRodWIuY29tL3N1Z2Fya3ViZXMvZ2VuZXJhdG9ycy93aWtpL0FQSSkKCgpbQXBwIERvY3VtZW50YXRpb24gKGdlbmVyYXRlZCBmcm9udCBlbmQpXShodHRwczovL2dpdGh1Yi5jb20vc3VnYXJrdWJlcy9nZW5lcmF0b3JzL3dpa2kvQVBQKQoKIyMgRXhwZXJpbWVudGFsIEVtYmVkZGFibGUgQ29tcG9uZW50cwoKKipXaGF0IGlmIHlvdSBjb3VsZCByZW1vdGVseSB1cGRhdGUgeW91ciBjb21wb25lbnRzIHdpdGhvdXQgaGF2aW5nIHRvIHB1c2ggbmV3IGNvZGU/KioKClRoYXRzIG9uZSBxdWVzdGlvbiB3ZSdyZSBleHBsb3Jpbmcgd2l0aCB0aGUgZXhwZXJpbWVudGFsIGVtYmVkZGFibGUgcmVhY3QgY29tcG9uZW50cy4gVGhlcmUgYXJlIG9mIGNvdXJzZSBjb29sIHdheXMgdG8gc2VydmUgc2luZ2xlIHBhZ2VzIGFzIHNlcnZlcmxlc3MgZnVuY3Rpb25zIGJ1dCB3aGF0J3MgY29vbGVyIHdvdWxkIGJlIGEgd2F5IGZvciBldmVuIG5vbi10ZWNobmljYWwgcGVvcGxlIHRvIHVwZGF0ZSBhIGRhdGFiYXNlIHNjaGVtYSBhbmQgYSBmb3JtIG9yIHRhYmxlIGluIHJlYWwgdGltZSB3aXRob3V0IHdyaXRpbmcgYW55IGNvZGUuCgoKLSBFYWNoIGNvbXBvbmVudCBjb21lcyBvdXQgaW4gYW4gZW1iZWRkYWJsZSBmb3JtYXQgdW5kZXIgKi9lbWJlZCoKLSBUQkQgLSBIb3cgdG8gZWFzaWx5IGRlcGxveSB0aGVzZSBhbmQgdXNlIHRoZW0uCi0KCgojIyBVcGRhdGVzCgotIDgvNS8xOSBzdXBwb3J0IGZvciBtb25nbyBhcnJheXMgaW4gZG9jdW1lbnRzCg== readmeEtag: '"e3945d8d8d83bdcfd3d4d4008752f4b9a5f11fa8"' readmeLastModified: Tue, 03 Mar 2020 22:51:34 GMT repositoryId: 189799018 description: >- API Generator - instantly generate REST and GraphQL APIs (openapi (OAS) 3.0.0) created: '2019-06-02T03:05:54Z' updated: '2026-01-24T02:25:22Z' language: JavaScript archived: false stars: 234 watchers: 1 forks: 29 owner: wrannaman logo: https://avatars.githubusercontent.com/u/5325165?v=4 license: MIT repoEtag: '"e6281517772ed08cdc667f8350a64d34739dfc8aa42da1eca8d1e596d50d0673"' repoLastModified: Sat, 24 Jan 2026 02:25:22 GMT foundInMaster: true category: SDK id: 824dcdea013394109fc958b1d93b91e6 - source: openapi3 tags repository: https://github.com/christianhelle/refitter v3: true id: 17db61563d8b069fd3d4a63d36459a68 repositoryMetadata: base64Readme: >- [![Build](https://github.com/christianhelle/refitter/actions/workflows/build.yml/badge.svg)](https://github.com/christianhelle/refitter/actions/workflows/build.yml)
[![Smoke Tests](https://github.com/christianhelle/refitter/actions/workflows/smoke-tests.yml/badge.svg)](https://github.com/christianhelle/refitter/actions/workflows/smoke-tests.yml)
[![NuGet](https://img.shields.io/nuget/v/refitter?color=blue)](https://www.nuget.org/packages/refitter)
[![Docker Image Version](https://img.shields.io/docker/v/christianhelle/refitter?label=docker)](https://hub.docker.com/r/christianhelle/refitter)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_refitter&metric=alert_status)](https://sonarcloud.io/summary/new_code?id=christianhelle_refitter)
[![codecov](https://codecov.io/gh/christianhelle/refitter/graph/badge.svg?token=242YT1N6T2)](https://codecov.io/gh/christianhelle/refitter)

<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-88-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END -->

# Refitter

Refitter is a tool for generating a C# REST API Client using the [Refit](https://github.com/reactiveui/refit) library. Refitter can generate the Refit interface and contracts from OpenAPI specifications. Refitter could format the generated Refit interface to be managed by [Apizr](https://www.apizr.net) (v6+) and generate some registration helpers too.

Refitter comes in 3 forms:

- A [.NET CLI Tool](#cli-tool) distributed via [nuget.org](http://www.nuget.org/packages/refitter) that outputs a single C# file on disk
- An [MSBuild Task](#msbuild) via the [Refitter.MSBuild](http://www.nuget.org/packages/refitter.msbuild) package that integrates seamlessly into your build pipeline and automatically generates code at build time based on [.refitter](#.refitter-file-format) files within the project directory
- A [C# Source Generator](#source-generator) via the [Refitter.SourceGenerator](http://www.nuget.org/packages/refitter.sourcegenerator) package that generates code on compile time based on a [.refitter](#.refitter-file-format) within the project directory

## CLI Tool

### Installation

The tool is packaged as a .NET Tool and is published to nuget.org. You can install the latest version of this tool like this:

```shell
dotnet tool install --global Refitter
```

Alternatively, you can use the Docker image:

```shell
docker pull christianhelle/refitter
```

Or run it directly:

```shell
docker run --rm -v $(pwd):/src christianhelle/refitter ./openapi.json --output ./GeneratedCode.cs
```

### Usage

```shell
refitter --help
```

```text
USAGE:
    refitter [URL or input file] [OPTIONS]

EXAMPLES:
    refitter ./openapi.json
    refitter https://petstore3.swagger.io/api/v3/openapi.yaml
    refitter ./openapi.json --settings-file ./openapi.refitter --output ./GeneratedCode.cs
    refitter ./openapi.json --namespace "Your.Namespace.Of.Choice.GeneratedCode" --output ./GeneratedCode.cs
    refitter ./openapi.json --namespace "Your.Namespace.Of.Choice.GeneratedCode" --internal
    refitter ./openapi.json --output ./IGeneratedCode.cs --interface-only
    refitter ./openapi.json --output ./GeneratedContracts.cs --contract-only
    refitter ./openapi.json --use-api-response
    refitter ./openapi.json --cancellation-tokens
    refitter ./openapi.json --no-operation-headers
    refitter ./openapi.json --ignored-operation-headers "header-one" --ignored-operation-headers "Header-Two"
    refitter ./openapi.json --no-accept-headers
    refitter ./openapi.json --use-iso-date-format
    refitter ./openapi.json --additional-namespace "Your.Additional.Namespace" --additional-namespace "Your.Other.Additional.Namespace"
    refitter ./openapi.json --multiple-interfaces ByEndpoint
    refitter ./openapi.json --tag Pet --tag Store --tag User
    refitter ./openapi.json --match-path '^/pet/.*'
    refitter ./openapi.json --trim-unused-schema
    refitter ./openapi.json --trim-unused-schema  --keep-schema '^Model$' --keep-schema '^Person.+'
    refitter ./openapi.json --no-deprecated-operations
    refitter ./openapi.json --operation-name-template '{operationName}Async'
    refitter ./openapi.json --optional-nullable-parameters
    refitter ./openapi.json --use-polymorphic-serialization
    refitter ./openapi.json --collection-format Csv
    refitter ./openapi.json --simple-output
    refitter ./openapi.json --no-inline-json-converters

ARGUMENTS:
    [URL or input file]    URL or file path to OpenAPI Specification file

OPTIONS:
                                                DEFAULT
    -h, --help                                                   Prints help information
    -v, --version                                                Prints version information
    -s, --settings-file                                          Path to .refitter settings file. Specifying this will ignore all other settings (except for --output)
    -n, --namespace                             GeneratedCode    Default namespace to use for generated types
        --contracts-namespace                                    Default namespace to use for generated contracts
    -o, --output                                Output.cs        Path to Output file or folder (if multiple files are generated)
        --contracts-output                                       Output path for generated contracts. Enabling this automatically enables generating multiple files
        --no-auto-generated-header                               Don't add <auto-generated> header to output file
        --no-accept-headers                                      Don't add <Accept> header to output file
        --interface-only                                         Don't generate contract types
        --contract-only                                          Don't generate clients
        --use-api-response                                       Return Task<IApiResponse<T>> instead of Task<T>
        --use-observable-response                                Return IObservable instead of Task
        --internal                                               Set the accessibility of the generated types to 'internal'
        --cancellation-tokens                                    Use cancellation tokens
        --no-operation-headers                                   Don't generate operation headers
        --no-logging                                             Don't log errors or collect telemetry
        --additional-namespace                                   Add additional namespace to generated types
        --exclude-namespace                                      Exclude namespace on generated types
        --use-iso-date-format                                    Explicitly format date query string parameters in ISO 8601 standard date format using delimiters (2023-06-15)
        --multiple-interfaces                                    Generate a Refit interface for each endpoint. May be one of ByEndpoint, ByTag
        --multiple-files                                         Generate multiple files instead of a single large file.
                                                                 The output files can be the following:
                                                                 - RefitInterfaces.cs
                                                                 - DependencyInjection.cs
                                                                 - Contracts.cs
        --match-path                                             Only include Paths that match the provided regular expression. May be set multiple times
        --tag                                                    Only include Endpoints that contain this tag. May be set multiple times and result in OR'ed evaluation
        --skip-validation                                        Skip validation of the OpenAPI specification
        --no-deprecated-operations                               Don't generate deprecated operations
        --operation-name-template                                Generate operation names using pattern. When using --multiple-interfaces ByEndpoint, this is name of the Execute() method in the interface where all instances of the string '{operationName}' is replaced with 'Execute'
        --optional-nullable-parameters                           Generate nullable parameters as optional parameters
        --trim-unused-schema                                     Removes unreferenced components schema to keep the generated output to a minimum
        --keep-schema                                            Force to keep matching schema, uses regular expressions. Use together with "--trim-unused-schema". Can be set multiple times
        --include-inheritance-hierarchy                          Keep all possible inherited types/union types even if they are not directly used
        --no-banner                                              Don't show donation banner
        --simple-output                                          Generate simple, plain-text console output without ASCII art, tables, emojis, or color formatting (suitable for IDE output windows)
        --skip-default-additional-properties                     Set to true to skip default additional properties
        --collection-format                      Multi           Determines the format of collection parameters. May be one of Multi, Csv, Ssv, Tsv, Pipes
        --operation-name-generator              Default          The NSwag IOperationNameGenerator implementation to use.
                                                                 May be one of:
                                                                 - Default
                                                                 - MultipleClientsFromOperationId
                                                                 - MultipleClientsFromPathSegments
                                                                 - MultipleClientsFromFirstTagAndOperationId
                                                                 - MultipleClientsFromFirstTagAndOperationName
                                                                 - MultipleClientsFromFirstTagAndPathSegments
                                                                 - SingleClientFromOperationId
                                                                 - SingleClientFromPathSegments
                                                                 See https://refitter.github.io/api/Refitter.Core.OperationNameGeneratorTypes.html for more information
        --immutable-records                                      Generate contracts as immutable records instead of classes
        --use-apizr                                              Use Apizr by:
                                                                 - Adding a final IApizrRequestOptions options parameter to all generated methods
                                                                 - Providing cancellation tokens by Apizr request options instead of a dedicated parameter
                                                                 - Using method overloads instead of optional parameters
                                                                 See https://refitter.github.io for more information and https://www.apizr.net to get started with Apizr
        --use-dynamic-querystring-parameters                     Enable wrapping multiple query parameters into a single complex one. Default is no wrapping.
                                                                 See https://github.com/reactiveui/refit?tab=readme-ov-file#dynamic-querystring-parameters for more information
        --use-polymorphic-serialization                          Use System.Text.Json polymorphic serialization.
                                                                 Replaces NSwag JsonInheritanceConverter attributes with System.Text.Json JsonPolymorphicAttributes.
                                                                 To have the native support of inheritance (de)serialization and fallback to base types when
                                                                 payloads with (yet) unknown types are offered by newer versions of an API
                                                                 See https://learn.microsoft.com/en-us/dotnet/standard/serialization/system-text-json/polymorphism for more information
        --disposable                                             Generate refit clients that implement IDisposable
        --no-inline-json-converters                              Don't inline JsonConverter attributes for enum properties. When disabled, enum properties will not have [JsonConverter(typeof(JsonStringEnumConverter))] attributes
        --integer-type                           int             The .NET type to use for OpenAPI integer types without a format specifier. Common values: 'int' (default), 'long'
        --custom-template-directory                              Custom directory with NSwag fluid templates for code generation. Default is null which uses the default NSwag templates. See <https://github.com/RicoSuter/NSwag/wiki/Templates>
```

To generate code from an OpenAPI specifications file, run the following:

```shell
refitter [path to OpenAPI spec file] --namespace "[Your.Namespace.Of.Choice.GeneratedCode]"
```

This will generate a file called `Output.cs` which contains the Refit interface and contract classes generated using [NSwag](https://github.com/RicoSuter/NSwag)

### CLI Tool Output Example

Here's what the console output looks like when running the Refitter CLI tool:

![Console Output](images/console-output.png)

### Simple Output Mode

For integration with IDEs and build tools that don't support rich console formatting, Refitter provides a `--simple-output` option:

```shell
refitter ./openapi.json --simple-output
```

This mode generates plain text output without:

- ASCII art and banners
- Colored text and rich formatting
- Unicode characters and emojis
- Tables and panels

This is particularly useful when:

- Running Refitter from Visual Studio extensions
- Integrating with build systems that capture console output
- Using tools that don't support ANSI escape sequences
- Running in environments with limited console capabilities


## .Refitter File format

The following is an example `.refitter` file

```js
{
  "openApiPath": "/path/to/your/openAPI", // Required
  "namespace": "Org.System.Service.Api.GeneratedCode", // Optional. Default=GeneratedCode
  "contractsNamespace": "Org.System.Service.Api.GeneratedCode.Contracts", // Optional. Default=GeneratedCode
  "naming": {
    "useOpenApiTitle": false, // Optional. Default=true
    "interfaceName": "MyApiClient" // Optional. Default=ApiClient
  },
  "generateContracts": true, // Optional. Default=true
  "generateClients": true, // Optional. Default=true
  "generateXmlDocCodeComments": true, // Optional. Default=true
  "generateStatusCodeComments": true, // Optional. Default=true
  "addAutoGeneratedHeader": true, // Optional. Default=true
  "addAcceptHeaders": true, // Optional. Default=true
  "addContentTypeHeaders": true, // Optional. Default=true
  "returnIApiResponse": false, // Optional. Default=false
  "responseTypeOverride": { // Optional. Default={}
    "File_Upload": "IApiResponse",
    "File_Download": "System.Net.Http.HttpContent"
  },
  "generateOperationHeaders": true, // Optional. Default=true
  "ignoredOperationHeaders": ["apiKey"], // Optional. Default=[]
  "typeAccessibility": "Public", // Optional. Values=Public|Internal. Default=Public
  "useCancellationTokens": false, // Optional. Default=false
  "useIsoDateFormat": false, // Optional. Default=false
  "multipleInterfaces": "ByEndpoint", // Optional. May be one of "ByEndpoint" or "ByTag"
  "generateDeprecatedOperations": false, // Optional. Default=true
  "operationNameTemplate": "{operationName}Async", // Optional. Must contain {operationName}. When multipleInterfaces == "ByEndpoint", this is name of the Execute() method in the interface where all instances of the string '{operationName}' is replaced with 'Execute'
  "optionalParameters": false, // Optional. Default=false
  "outputFolder": "../CustomOutput" // Optional. Default=./Generated
  "outputFilename": "RefitInterface.cs", // Optional. Default=Output.cs for CLI tool
  "contractsOutputFolder": "../Contracts", // Optional. Default=NULL
  "generateMultipleFiles": false, // Optional. Default=false
  "additionalNamespaces": [ // Optional
    "Namespace1",
    "Namespace2"
  ],
  "includeTags": [ // Optional. OpenAPI Tag to include when generating code
    "Pet",
    "Store",
    "User"
  ],
  "includePathMatches": [ // Optional. Only include Paths that match the provided regular expression
    "^/pet/.*",
    "^/store/.*"
  ],
  "trimUnusedSchema": false, // Optional. Default=false
  "keepSchemaPatterns": [ // Optional. Force to keep matching schema, uses regular expressions. Use together with trimUnusedSchema=true
    "^Model$",
    "^Person.+"
  ],
  "includeInheritanceHierarchy": false, // Optional. Default=false. Set to true to keep all possible type-instances of inheritance/union types. This works in conjunction with trimUnusedSchema.
  "generateDefaultAdditionalProperties": true, // Optional. default=true
  "operationNameGenerator": "Default", // Optional. May be one of Default, MultipleClientsFromOperationId, MultipleClientsFromPathSegments, MultipleClientsFromFirstTagAndOperationId, MultipleClientsFromFirstTagAndOperationName, MultipleClientsFromFirstTagAndPathSegments, SingleClientFromOperationId, SingleClientFromPathSegments
  "immutableRecords": false,
  "useDynamicQuerystringParameters": true, // Optional. Default=false
  "usePolymorphicSerialization": true, // Optional. Default=false
  "generateDisposableClients": true, // Optional. Default=false
  "dependencyInjectionSettings": { // Optional
    "baseUrl": "https://petstore3.swagger.io/api/v3", // Optional. Leave this blank to set the base address manually
    "httpMessageHandlers": [ // Optional
        "AuthorizationMessageHandler",
        "TelemetryMessageHandler"
    ],
    "usePolly": true, // DEPRECATED - Use "transientErrorHandler": "None|Polly|HttpResilience" instead
    "transientErrorHandler": "HttpResilience", // Optional. Set this to configure transient error handling with a retry policy that uses a jittered backoff. May be one of None, Polly, HttpResilience
    "maxRetryCount": 3, // Optional. Default=6
    "firstBackoffRetryInSeconds": 0.5 // Optional. Default=1.0
  },
  "apizrSettings": { // Optional
    "withRequestOptions": true, // Optional. Default=true
    "withRegistrationHelper": true, // Optional. Default=false
    "withCacheProvider": "InMemory", // Optional. Values=None|Akavache|MonkeyCache|InMemory|DistributedAsString|DistributedAsByteArray. Default=None
    "withPriority": true, // Optional. Default=false
    "withMediation": true, // Optional. Default=false
    "withOptionalMediation": true, // Optional. Default=false
    "withMappingProvider": "AutoMapper", // Optional. Values=None|AutoMapper|Mapster. Default=None
    "withFileTransfer": true // Optional. Default=false
  },
  "codeGeneratorSettings": { // Optional. Default settings are the values set in this example
    "requiredPropertiesMustBeDefined": true,
    "generateDataAnnotations": true,
    "anyType": "object",
    "dateType": "System.DateTimeOffset",
    "dateTimeType": "System.DateTimeOffset",
    "timeType": "System.TimeSpan",
    "timeSpanType": "System.TimeSpan",
    "arrayType": "System.Collections.Generic.ICollection",
    "dictionaryType": "System.Collections.Generic.IDictionary",
    "arrayInstanceType": "System.Collections.ObjectModel.Collection",
    "dictionaryInstanceType": "System.Collections.Generic.Dictionary",
    "arrayBaseType": "System.Collections.ObjectModel.Collection",
    "dictionaryBaseType": "System.Collections.Generic.Dictionary",
    "integerType": "Int32", // Optional. Default="Int32". The .NET type for OpenAPI integers without a format. Possible values: "Int32", "Int64"
    "propertySetterAccessModifier": "",
    "generateImmutableArrayProperties": false,
    "generateImmutableDictionaryProperties": false,
    "handleReferences": false,
    "jsonSerializerSettingsTransformationMethod": null,
    "generateJsonMethods": false,
    "enforceFlagEnums": false,
    "inlineNamedDictionaries": false,
    "inlineNamedTuples": true,
    "inlineNamedArrays": false,
    "generateOptionalPropertiesAsNullable": false,
    "generateNullableReferenceTypes": false,
    "generateNativeRecords": false,
    "generateDefaultValues": true,
    "inlineNamedAny": false,
    "inlineJsonConverters": true, // Optional. Default=true. Set to false to not generate JsonConverter attributes for enum properties
    "dateFormat": "yyyy-MM-dd",
    "dateTimeFormat": "yyyy-MM-dd",
    "excludedTypeNames": [
      "ExcludedTypeFoo",
      "ExcludedTypeBar"
    ],
    "customTemplateDirectory": "./path/to/directory/" // Optional. See <https://github.com/RicoSuter/NSwag/wiki/Templates>
  }
}
```

- `openApiPath` - points to the OpenAPI Specifications file. This can be the path to a file stored on disk, relative to the `.refitter` file. This can also be a URL to a remote file that will be downloaded over HTTP/HTTPS
- `namespace` - the namespace used in the generated code. If not specified, this defaults to `GeneratedCode`
- `naming.useOpenApiTitle` - a boolean indicating whether the OpenApi title should be used. Default is `true`
- `naming.interfaceName` - the name of the generated interface. The generated code will automatically prefix this with `I` so if this set to `MyApiClient` then the generated interface is called `IMyApiClient`. Default is `ApiClient`
- `generateContracts` - a boolean indicating whether contracts should be generated. A use case for this is several API clients use the same contracts. Default is `true`
- `generateClients`:  - a boolean indicating whether clients should be generated. A use case for this is to seperate clients and contracts in two generation
- `generateXmlDocCodeComments` - a boolean indicating whether XML doc comments should be generated. Default is `true`
- `generateStatusCodeComments` - a boolean indicating whether the XML docs for `ApiException` and `IApiResponse` contain detailed descriptions for every documented status code. Default is `true`
- `addAutoGeneratedHeader` - a boolean indicating whether XML doc comments should be generated. Default is `true`
- `addAcceptHeaders` -  a boolean indicating whether to add accept headers [Headers("Accept: application/json")]. Default is `true`
- `returnIApiResponse` - a boolean indicating whether to return `IApiResponse<T>` objects. Default is `false`
- `responseTypeOverride` - a dictionary with operation ids (as specified in the OpenAPI document) and a particular return type to use. The types are wrapped in a task, but otherwise unmodified (so make sure to specify or import their namespaces). Default is `{}`
- `generateOperationHeaders` - a boolean indicating whether to use operation headers in the generated methods. Default is `true`
- `ignoredOperationHeaders` - A collection of headers to omit from operation signatures. Default is `[]`
- `typeAccessibility` - the generated type accessibility. Possible values are `Public` and `Internal`. Default is `Public`
- `useCancellationTokens` - Use cancellation tokens in the generated methods. Default is `false`
- `useIsoDateFormat` - Set to `true` to explicitly format date query string parameters in ISO 8601 standard date format using delimiters (for example: 2023-06-15). Default is `false`
- `multipleInterfaces` - Set to `ByEndpoint` to generate an interface for each endpoint, or `ByTag` to group Endpoints by their Tag (like SwaggerUI groups them).
- `outputFolder` - a string describing a relative path to a desired output folder. Default is `./Generated`
- `outputFilename` - Output filename. Default is `Output.cs` when used from the CLI tool, otherwise its the .refitter filename. So `Petstore.refitter` becomes `Petstore.cs`.
- `additionalNamespaces` - A collection of additional namespaces to include in the generated file. A use case for this is when you want to reuse contracts from a different namespace than the generated code. Default is empty
- `includeTags` - A collection of tags to use a filter for including endpoints that contain this tag.
- `includePathMatches` - A collection of regular expressions used to filter paths.
- `generateDeprecatedOperations` - a boolean indicating whether deprecated operations should be generated or skipped. Default is `true`
- `operationNameTemplate` - Generate operation names using pattern. This must contain the string {operationName}. An example usage of this could be `{operationName}Async` to suffix all method names with Async. When using multiple interfaces with `ByEndpoint`, this is name of the Execute() method in the interface where all instances of the string '{operationName}' is replaced with 'Execute'
- `optionalParameters` - Generate non-required parameters as nullable optional parameters
- `trimUnusedSchema` - Removes unreferenced components schema to keep the generated output to a minimum
- `keepSchemaPatterns`: A collection of regular expressions to force to keep matching schema. This is used together with `trimUnusedSchema`
- `includeInheritanceHierarchy`: Set to true to keep all possible type-instances of inheritance/union types. If this is false only directly referenced types will be kept. This works in conjunction with `trimUnusedSchema`
- `generateDefaultAdditionalProperties`: Set to `false` to skip default additional properties. Default is `true`
- `operationNameGenerator`: The NSwag `IOperationNameGenerator` implementation to use. See <https://refitter.github.io/api/Refitter.Core.OperationNameGeneratorTypes.html>
- `immutableRecords`: Set to `true` to generate contracts as immutable records instead of classes. Default is `false`
- `useDynamicQuerystringParameters`: Set to `true` to wrap multiple query parameters into a single complex one. Default is `false` (no wrapping). See <https://github.com/reactiveui/refit?tab=readme-ov-file#dynamic-querystring-parameters> for more information.
- `dependencyInjectionSettings` - Setting this will generated extension methods to `IServiceCollection` for configuring Refit clients
  - `baseUrl` - Used as the HttpClient base address. Leave this blank to manually set the base URL
  - `httpMessageHandlers` - A collection of `HttpMessageHandler` that is added to the HttpClient pipeline
  - `usePolly` - Set this to `true` to configure the HttpClient to use Polly using a retry policy with a jittered backoff.  This is **DEPRECATED**, use `transientErrorHandler` instead
  - `transientErrorHandler`: Set this to configure transient error handling with a retry policy that uses a jittered backoff. See <https://refitter.github.io/api/Refitter.Core.TransientErrorHandler.html>
  - `firstBackoffRetryInSeconds` - This is the duration of the initial retry backoff. Default is 1 second
- `apizrSettings` - Setting this will format Refit interface to be managed by Apizr. See <https://www.apizr.net> for more information
  - `withRequestOptions` - Tells if the Refit interface methods should have a final IApizrRequestOptions options parameter
  - `withRegistrationHelper` - Tells if Refitter should generate Apizr registration helpers (extended with dependencyInjectionSettings set, otherwise static)
  - `withCacheProvider` - Set the cache provider to be used
  - `withPriority` - Tells if Apizr should handle request priority
  - `withMediation` - Tells if Apizr should handle request mediation (extended only)
  - `withOptionalMediation` - Tells if Apizr should handle optional request mediation (extended only)
  - `withMappingProvider` - Set the mapping provider to be used
  - `withFileTransfer` - Tells if Apizr should handle file transfer
- `codeGeneratorSettings` - Setting this allows customization of the NSwag generated types and contracts
  - `requiredPropertiesMustBeDefined` - Default is true,
  - `generateDataAnnotations` - Default is true,
  - `anyType` - Default is `object`,
  - `dateType` - Default is `System.DateTimeOffset`,
  - `dateTimeType` - Default is `System.DateTimeOffset`,
  - `timeType` - Default is `System.TimeSpan`,
  - `timeSpanType` - Default is `System.TimeSpan`,
  - `arrayType` - Default is `System.Collections.Generic.ICollection`,
  - `dictionaryType` - Default is `System.Collections.Generic.IDictionary`,
  - `arrayInstanceType` - Default is `System.Collections.ObjectModel.Collection`,
  - `dictionaryInstanceType` - Default is `System.Collections.Generic.Dictionary`,
  - `arrayBaseType` - Default is `System.Collections.ObjectModel.Collection`,
  - `dictionaryBaseType` - Default is `System.Collections.Generic.Dictionary`,
  - `integerType` - Default is `Int32`. The .NET type to use for OpenAPI integer types without a format specifier. Possible values: `Int32`, `Int64`
  - `propertySetterAccessModifier` - Default is ``,
  - `generateImmutableArrayProperties` - Default is false,
  - `generateImmutableDictionaryProperties` - Default is false,
  - `handleReferences` - Default is false,
  - `jsonSerializerSettingsTransformationMethod` - Default is null,
  - `generateJsonMethods` - Default is false,
  - `enforceFlagEnums` - Default is false,
  - `inlineNamedDictionaries` - Default is false,
  - `inlineNamedTuples` - Default is true,
  - `inlineNamedArrays` - Default is false,
  - `generateOptionalPropertiesAsNullable` - Default is false,
  - `generateNullableReferenceTypes` - Default is false,
  - `generateNativeRecords` - Default is false
  - `generateDefaultValues` - Default is true
  - `inlineNamedAny` - Default is false
  - `inlineJsonConverters` - Default is true. When set to false, enum properties will not have `[JsonConverter(typeof(JsonStringEnumConverter))]` attributes
  - `dateFormat` - Default is null
  - `dateTimeFormat` - Default is null
  - `excludedTypeNames` - Default is empty

## MSBuild

This is the recommended approach for generating code from OpenAPI specifications at build time. The MSBuild approach integrates seamlessly into your build pipeline, automatically generates code at build time

### Why Choose MSBuild over Source Generator?

The MSBuild task offers several advantages over the Source Generator:

- **Build-time Generation**: Code is generated during the pre-build process, which ensures that the Refit source generator will work without needing to rebuild a second time.
- **No Manual Commits**: Generated files are created automatically during build, eliminating the need to commit generated code to source control
- **No Additional Tools Required**: Works out of the box without requiring separate CLI tool installation

### Installation

The MSBuild task is distributed as a NuGet package:

```shell
dotnet add package Refitter.MSBuild
```

### Usage

Once installed, the MSBuild task will automatically scan your project for `.refitter` files and generate code during build. Simply create a `.refitter` file in your project directory with your OpenAPI specification settings.

The MSBuild package includes a custom `.target` file which executes the `RefitterGenerateTask` custom task:

```xml
<UsingTask TaskName="RefitterGenerateTask"
           AssemblyFile="$(MSBuildThisFileDirectory)Refitter.MSBuild.dll"
           Condition="Exists('$(MSBuildThisFileDirectory)Refitter.MSBuild.dll')" />
<Target Name="RefitterGenerate" BeforeTargets="BeforeCompile">
    <RefitterGenerateTask ProjectFileDirectory="$(MSBuildProjectDirectory)"
                          DisableLogging="$(RefitterNoLogging)"
                          SkipValidation="$(RefitterSkipValidation)">
        <Output TaskParameter="GeneratedFiles" ItemName="RefitterGeneratedFiles" />
    </RefitterGenerateTask>
    <ItemGroup>
        <Compile Include="@(RefitterGeneratedFiles)" />
    </ItemGroup>
</Target>
```

The `RefitterGenerateTask` task will scan the project folder for `.refitter` files and execute them all.

### Configuration

By default, telemetry collection is enabled. To opt-out, add the following to your `.csproj` file:

```xml
<PropertyGroup>
  <RefitterNoLogging>true</RefitterNoLogging>
</PropertyGroup>
```

You can also skip OpenAPI validation by setting:

```xml
<PropertyGroup>
  <RefitterSkipValidation>true</RefitterSkipValidation>
</PropertyGroup>
```

### Example

Create a `.refitter` file in your project:

```json
{
  "openApiPath": "https://petstore3.swagger.io/api/v3/openapi.json",
  "namespace": "Petstore.Api",
  "outputFolder": "./Generated"
}
```

Now, every time you build your project, Refitter will automatically generate the API client code based on your OpenAPI specification.

### Alternative: Using CLI with MSBuild Exec Task

If you prefer more control or need to use the CLI tool directly, you can invoke the Refitter CLI from MSBuild pre-build events:

```xml
<Target Name="Refitter" AfterTargets="PreBuildEvent">
    <Exec WorkingDirectory="$(ProjectDir)" Command="dotnet tool restore" />
    <Exec WorkingDirectory="$(ProjectDir)" Command="refitter --settings-file .refitter --skip-validation" />
</Target>
```

This approach requires that Refitter is installed as a local tool using a manifest file, as described in [this tutorial](https://learn.microsoft.com/en-us/dotnet/core/tools/local-tools-how-to-use?WT.mc_id=DT-MVP-5004822).

## Source Generator

> **Note:** We recommend using the [MSBuild](#msbuild) approach instead of the Source Generator because the code is generated at pre-compile, ensuring that the Refit source generator will generated code from the Refit interfaces added to the compilation

Refitter is available as a C# Source Generator that uses the [Refitter.Core](https://github.com/christianhelle/refitter/tree/main/src/Refitter.Core) library for generating a REST API Client using the [Refit](https://github.com/reactiveui/refit) library. Refitter can generate the Refit interface from OpenAPI specifications. Refitter could format the generated Refit interface to be managed by [Apizr](https://www.apizr.net) and generate some registration helpers too.

The Refitter source generator is a bit untraditional in a sense that it creates a folder called `Generated` in the same location as the `.refitter` file and generates files to disk under the `Generated` folder (can be changed with `--outputFolder`). The source generator output should be included in the project and committed to source control. This is done because there is no other way to trigger the Refit source generator to pickup the Refitter generated code

***(Translation: I couldn't for the life of me figure how to get that to work, sorry)***

### Installation

The source generator is distributed as a NuGet package and should be installed to the project that will contain the generated code

```shell
dotnet add package Refitter.SourceGenerator
```

### Usage

This source generator generates code based on any `.refitter` file included to the project as `AdditionalFiles`.

The generator can automatically detect all `.refitter` files inside the project that referenced the `Refitter.SourceGenerator` package and there is no need to include them manually as `AdditionalFiles`

# Using the generated code

Here's an example generated output from the [Swagger Petstore example](https://petstore3.swagger.io) using the default settings

**CLI Tool**

```bash
refitter ./openapi.json --namespace "Your.Namespace.Of.Choice.GeneratedCode"
```

**Source Generator ***.refitter*** file**

```json
{
  "openApiPath": "./openapi.json",
  "namespace": "Your.Namespace.Of.Choice.GeneratedCode"
}
```

**Output (Snippet)**

Full output is available [here](docs/DefaultOutput.cs)

```cs
using Refit;
using System.Collections.Generic;
using System.Text.Json.Serialization;
using System.Threading.Tasks;

namespace Your.Namespace.Of.Choice.GeneratedCode
{
    [System.CodeDom.Compiler.GeneratedCode("Refitter", "1.0.0.0")]
    public partial interface ISwaggerPetstore
    {
        /// <summary>Update an existing pet</summary>
        /// <remarks>Update an existing pet by Id</remarks>
        /// <param name="body">Update an existent pet in the store</param>
        /// <returns>Successful operation</returns>
        /// <exception cref="ApiException">
        /// Thrown when the request returns a non-success status code:
        /// <list type="table">
        /// <listheader>
        /// <term>Status</term>
        /// <description>Description</description>
        /// </listheader>
        /// <item>
        /// <term>400</term>
        /// <description>Invalid ID supplied</description>
        /// </item>
        /// <item>
        /// <term>404</term>
        /// <description>Pet not found</description>
        /// </item>
        /// <item>
        /// <term>405</term>
        /// <description>Validation exception</description>
        /// </item>
        /// </list>
        /// </exception>
        [Headers("Accept: application/xml, application/json")]
        [Put("/pet")]
        Task<Pet> UpdatePet([Body] Pet body);

        ...
    }
}
```

Here's an example generated output from the [Swagger Petstore example](https://petstore3.swagger.io) configured to wrap the return type in `IApiResponse<T>`

**CLI Tool**

```bash
refitter ./openapi.json --namespace "Your.Namespace.Of.Choice.GeneratedCode" --use-api-response
```

**Source Generator ***.refitter*** file**

```json
{
  "openApiPath": "./openapi.json",
  "namespace": "Your.Namespace.Of.Choice.GeneratedCode",
  "returnIApiResponse": true
}
```

**Output (Snippet)**

Full output is available [here](docs/IApiResponseOutput.cs)

```cs
using Refit;
using System.Collections.Generic;
using System.Text.Json.Serialization;
using System.Threading.Tasks;

namespace Your.Namespace.Of.Choice.GeneratedCode
{
    [System.CodeDom.Compiler.GeneratedCode("Refitter", "1.0.0.0")]
    public partial interface ISwaggerPetstore
    {
        /// <summary>Update an existing pet</summary>
        /// <remarks>Update an existing pet by Id</remarks>
        /// <param name="body">Update an existent pet in the store</param>
        /// <returns>
        /// A <see cref="Task"/> representing the <see cref="IApiResponse"/> instance containing the result:
        /// <list type="table">
        /// <listheader>
        /// <term>Status</term>
        /// <description>Description</description>
        /// </listheader>
        /// <item>
        /// <term>200</term>
        /// <description>Successful operation</description>
        /// </item>
        /// <item>
        /// <term>400</term>
        /// <description>Invalid ID supplied</description>
        /// </item>
        /// <item>
        /// <term>404</term>
        /// <description>Pet not found</description>
        /// </item>
        /// <item>
        /// <term>405</term>
        /// <description>Validation exception</description>
        /// </item>
        /// </list>
        /// </returns>
        [Headers("Accept: application/xml, application/json")]
        [Put("/pet")]
        Task<IApiResponse<Pet>> UpdatePet([Body] Pet body);

        ...
    }
}
```

Here's an example generated output from the [Swagger Petstore example](https://petstore3.swagger.io) configured to generate an interface for each endpoint

**CLI Tool**

```bash
refitter ./openapi.json --namespace "Your.Namespace.Of.Choice.GeneratedCode" --multiple-interfaces ByEndpoint
```

**Source Generator ***.refitter*** file**

```json
{
  "openApiPath": "./openapi.json",
  "namespace": "Your.Namespace.Of.Choice.GeneratedCode",
  "multipleInterfaces": "ByEndpoint"
}
```

**Output (Snippet)**

Full output is available [here](docs/IApiResponseOutput.cs)

```cs
/// <summary>Update an existing pet</summary>
[System.CodeDom.Compiler.GeneratedCode("Refitter", "1.0.0.0")]
public partial interface IUpdatePetEndpoint
{
    /// <summary>Update an existing pet</summary>
    /// <remarks>Update an existing pet by Id</remarks>
    /// <param name="body">Update an existent pet in the store</param>
    /// <returns>Successful operation</returns>
    /// <exception cref="ApiException">
    /// Thrown when the request returns a non-success status code:
    /// <list type="table">
    /// <listheader>
    /// <term>Status</term>
    /// <description>Description</description>
    /// </listheader>
    /// <item>
    /// <term>400</term>
    /// <description>Invalid ID supplied</description>
    /// </item>
    /// <item>
    /// <term>404</term>
    /// <description>Pet not found</description>
    /// </item>
    /// <item>
    /// <term>405</term>
    /// <description>Validation exception</description>
    /// </item>
    /// </list>
    /// </exception>
    [Headers("Accept: application/xml, application/json")]
    [Put("/pet")]
    Task<Pet> Execute([Body] Pet body);
}
```

Here's an example generated output from the [Swagger Petstore example](https://petstore3.swagger.io) configured to generate an interface with dynamic querystring parameters

**CLI Tool**

```bash
refitter ./openapi.json --namespace "Your.Namespace.Of.Choice.GeneratedCode" --use-dynamic-querystring-parameters
```

**Output (Snippet)**

Full output is available [here](docs/DynamicQueryStringParameters.cs)

```cs
[System.CodeDom.Compiler.GeneratedCode("Refitter", "1.0.0.0")]
public partial interface ISwaggerPetstoreOpenAPI30
{
    /// <summary>Updates a pet in the store with form data</summary>
    /// <param name="petId">ID of pet that needs to be updated</param>
    /// <param name="queryParams">The dynamic querystring parameter wrapping all others.</param>
    /// <returns>A <see cref="Task"/> that completes when the request is finished.</returns>
    /// <exception cref="ApiException">
    /// Thrown when the request returns a non-success status code:
    /// <list type="table">
    /// <listheader>
    /// <term>Status</term>
    /// <description>Description</description>
    /// </listheader>
    /// <item>
    /// <term>405</term>
    /// <description>Invalid input</description>
    /// </item>
    /// </list>
    /// </exception>
    [Post("/pet/{petId}")]
    Task UpdatePetWithForm(long petId, [Query] UpdatePetWithFormQueryParams queryParams);
}

public class UpdatePetWithFormQueryParams
{
    /// <summary>
    /// Name of pet that needs to be updated
    /// </summary>
    [Query]
    public string Name { get; set; }

    /// <summary>
    /// Status of pet that needs to be updated
    /// </summary>
    [Query]
    public string Status { get; set; }
}
```

## RestService

Here's an example usage of the generated code above

```cs
using Refit;
using System;
using System.Threading.Tasks;

namespace Your.Namespace.Of.Choice.GeneratedCode;

internal class Program
{
    private static async Task Main(string[] args)
    {
        var client = RestService.For<ISwaggerPetstore>("https://petstore3.swagger.io/api/v3");
        var pet = await client.GetPetById(1);

        Console.WriteLine("## Using Task<T> as return type ##");
        Console.WriteLine($"Name: {pet.Name}");
        Console.WriteLine($"Category: {pet.Category.Name}");
        Console.WriteLine($"Status: {pet.Status}");
        Console.WriteLine();

        var client2 = RestService.For<WithApiResponse.ISwaggerPetstore>("https://petstore3.swagger.io/api/v3");
        var response = await client2.GetPetById(2);

        Console.WriteLine("## Using Task<IApiResponse<T>> as return type ##");
        Console.WriteLine($"HTTP Status Code: {response.StatusCode}");
        Console.WriteLine($"Name: {response.Content.Name}");
        Console.WriteLine($"Category: {response.Content.Category.Name}");
        Console.WriteLine($"Status: {response.Content.Status}");
    }
}
```

The `RestService` class generates an implementation of `ISwaggerPetstore` that uses `HttpClient` to make its calls.

The code above when run will output something like this:

```
## Using Task<T> as return type ##
Name: Gatitotototo
Category: Chaucito
Status: Sold

## Using Task<IApiResponse<T>> as return type ##
HTTP Status Code: OK
Name: Gatitotototo
Category: Chaucito
Status: Sold
```

## ASP.NET Core and HttpClientFactory

Here's an example Minimal API with the [`Refit.HttpClientFactory`](https://www.nuget.org/packages/Refit.HttpClientFactory) library:

```cs
using Refit;
using Your.Namespace.Of.Choice.GeneratedCode;

var builder = WebApplication.CreateBuilder(args);
builder.Services.AddEndpointsApiExplorer();
builder.Services.AddSwaggerGen();
builder.Services
    .AddRefitClient<ISwaggerPetstore>()
    .ConfigureHttpClient(c => c.BaseAddress = new Uri("https://petstore3.swagger.io/api/v3"));

var app = builder.Build();
app.MapGet(
        "/pet/{id:long}",
        async (ISwaggerPetstore petstore, long id) =>
        {
            try
            {
                return Results.Ok(await petstore.GetPetById(id));
            }
            catch (Refit.ApiException e)
            {
                return Results.StatusCode((int)e.StatusCode);
            }
        })
    .WithName("GetPetById")
    .WithOpenApi();

app.UseHttpsRedirection();
app.UseSwaggerUI();
app.UseSwagger();
app.Run();
```

.NET Core supports registering the generated `ISwaggerPetstore` interface via `HttpClientFactory`

The following request to the API above

```shell
curl -X 'GET' 'https://localhost:5001/pet/1' -H 'accept: application/json'
```

Returns a response that looks something like this:

```json
{
  "id": 1,
  "name": "Special_char_owner_!@#$^&()`.testing",
  "photoUrls": [
    "https://petstore3.swagger.io/resources/photos/623389095.jpg"
  ],
  "tags": [],
  "status": "Sold"
}
```

## Dependency Injection

Refitter supports generating bootstrapping code that allows the user to conveniently configure all generated Refit interfaces by calling a single extension method to `IServiceCollection`.

This is enabled through the `.refitter` settings file like this:

```json
{
  "openApiPath": "../OpenAPI/v3.0/petstore.json",
  "namespace": "Petstore",
  "dependencyInjectionSettings": {
    "baseUrl": "https://petstore3.swagger.io/api/v3",
    "httpMessageHandlers": [ "TelemetryDelegatingHandler" ],
    "transientErrorHandler": "Polly",
    "maxRetryCount": 3,
    "firstBackoffRetryInSeconds": 0.5
  }
}
```

which will generate an extension method to `IServiceCollection` called `ConfigureRefitClients()`. The generated extension method depends on [`Refit.HttpClientFactory`](https://www.nuget.org/packages/Refit.HttpClientFactory) library and looks like this:

```cs
public static IServiceCollection ConfigureRefitClients(
    this IServiceCollection services,
    Action<IHttpClientBuilder>? builder = default,
    RefitSettings? settings = default)
{
    var clientBuilderISwaggerPetstore = services
        .AddRefitClient<ISwaggerPetstore>(settings)
        .ConfigureHttpClient(c => c.BaseAddress = new Uri("https://petstore3.swagger.io/api/v3"))
        .AddHttpMessageHandler<TelemetryDelegatingHandler>();

    clientBuilderISwaggerPetstore
        .AddPolicyHandler(
            HttpPolicyExtensions
                .HandleTransientHttpError()
                .WaitAndRetryAsync(
                    Backoff.DecorrelatedJitterBackoffV2(
                        TimeSpan.FromSeconds(0.5),
                        3)));

    builder?.Invoke(clientBuilderISwaggerPetstore);

    return services;
}
```

This comes in handy especially when generating multiple interfaces, by tag or endpoint. For example, the following `.refitter` settings file

```json
{
  "openApiPath": "../OpenAPI/v3.0/petstore.json",
  "namespace": "Petstore",
  "multipleInterfaces": "ByTag",
  "dependencyInjectionSettings": {
    "baseUrl": "https://petstore3.swagger.io/api/v3",
    "httpMessageHandlers": [ "TelemetryDelegatingHandler" ],
    "transientErrorHandler": "Polly",
    "maxRetryCount": 3,
    "firstBackoffRetryInSeconds": 0.5
  }
}
```

Will generate a single `ConfigureRefitClients()` extension methods that may contain dependency injection configuration code for multiple interfaces like this

```csharp
public static IServiceCollection ConfigureRefitClients(
    this IServiceCollection services,
    Action<IHttpClientBuilder>? builder = default,
    RefitSettings? settings = default)
{
    var clientBuilderIPetApi = services
        .AddRefitClient<IPetApi>(settings)
        .ConfigureHttpClient(c => c.BaseAddress = new Uri("https://petstore3.swagger.io/api/v3"))
        .AddHttpMessageHandler<TelemetryDelegatingHandler>();

    clientBuilderIPetApi
        .AddPolicyHandler(
            HttpPolicyExtensions
                .HandleTransientHttpError()
                .WaitAndRetryAsync(
                    Backoff.DecorrelatedJitterBackoffV2(
                        TimeSpan.FromSeconds(0.5),
                        3)));

    builder?.Invoke(clientBuilderIPetApi);

    var clientBuilderIStoreApi = services
        .AddRefitClient<IStoreApi>(settings)
        .ConfigureHttpClient(c => c.BaseAddress = new Uri("https://petstore3.swagger.io/api/v3"))
        .AddHttpMessageHandler<TelemetryDelegatingHandler>();

    clientBuilderIStoreApi
        .AddPolicyHandler(
            HttpPolicyExtensions
                .HandleTransientHttpError()
                .WaitAndRetryAsync(
                    Backoff.DecorrelatedJitterBackoffV2(
                        TimeSpan.FromSeconds(0.5),
                        3)));

    builder?.Invoke(clientBuilderIStoreApi);

    var clientBuilderIUserApi = services
        .AddRefitClient<IUserApi>(settings)
        .ConfigureHttpClient(c => c.BaseAddress = new Uri("https://petstore3.swagger.io/api/v3"))
        .AddHttpMessageHandler<TelemetryDelegatingHandler>();

    clientBuilderIUserApi
        .AddPolicyHandler(
            HttpPolicyExtensions
                .HandleTransientHttpError()
                .WaitAndRetryAsync(
                    Backoff.DecorrelatedJitterBackoffV2(
                        TimeSpan.FromSeconds(0.5),
                        3)));

    builder?.Invoke(clientBuilderIUserApi);

    return services;
}
```

Personally, they I use Refitter is to generate an interface per endpoint, so when generating code for a large and complex API, I might have several interfaces.

## Apizr

[Apizr](https://www.apizr.net) is a Refit client manager that provides a set of features to enhance requesting experience with resilience, caching, priority, mediation, mapping, logging, authentication, file transfer capabilities and many more...

### Generating the interfaces

Refitter supports generating Apizr formatted Refit interfaces that can be managed then by Apizr (v6+).

You can enable Apizr formatted Refit interface generation either:

- With the `--use-apizr` command line argument
- By setting the `apizrSettings` section in the `.refitter` settings file

Note that `--use-apizr` uses default Apizr settings with `withRequestOptions` set to `true` as recommended, while the `.refitter` settings file allows you to configure it deeper.

In both cases, it will format the generated Refit interfaces to be Apizr ready by:

- Adding a final IApizrRequestOptions options parameter to all generated methods (if `withRequestOptions` is set to `true`)
- Providing cancellation tokens by Apizr request options instead of a dedicated parameter (if `withRequestOptions` is set to `true`)
- Using method overloads instead of optional parameters (note that setting `useDynamicQuerystringParameters` to `true` improve overloading experience)

From here, you're definitely free to use the formatted interface with Apizr by registering, configuring and using it following the [Apizr documentation](https://www.apizr.net). But Refitter can go further by generating some helpers to make the configuration easier.

### Generating the helpers

Refitter supports generating Apizr (v6+) bootstrapping code that allows the user to conveniently configure all generated Apizr formatted Refit interfaces by calling a single method.
It could be either an extension method to `IServiceCollection` if DependencyInjectionSettings are set, or a static builder method if not.

### [Extended](#tab/tabid-extended)

To enable Apizr registration code generation for `IServiceCollection`, you need at least to set the `withRegistrationHelper` property to `true` and configure the `DependencyInjectionSettings` section in the `.refitter` settings file.
This is what the `.refitter` settings file may look like, depending on you configuration:

```json
{
  "openApiPath": "../OpenAPI/v3.0/petstore.json",
  "namespace": "Petstore",
  "useDynamicQuerystringParameters": true,
  "dependencyInjectionSettings": {
    "baseUrl": "https://petstore3.swagger.io/api/v3",
    "httpMessageHandlers": [ "MyDelegatingHandler" ],
    "transientErrorHandler": "HttpResilience",
    "maxRetryCount": 3,
    "firstBackoffRetryInSeconds": 0.5
  },
  "apizrSettings": {
    "withRequestOptions": true, // Recommended to include an Apizr request options parameter to Refit interface methods
    "withRegistrationHelper": true, // Mandatory to actually generate the Apizr registration extended method
    "withCacheProvider": "InMemory", // Optional, default is None
    "withPriority": true, // Optional, default is false
    "withMediation": true, // Optional, default is false
    "withOptionalMediation": true, // Optional, default is false
    "withMappingProvider": "AutoMapper", // Optional, default is None
    "withFileTransfer": true // Optional, default is false
  }
}
```

which will generate an extension method to `IServiceCollection` called `ConfigurePetstoreApiApizrManager()`. The generated extension method depends on [`Apizr.Extensions.Microsoft.DependencyInjection`](https://www.nuget.org/packages/Apizr.Extensions.Microsoft.DependencyInjection) library and looks like this:

```cs
public static IServiceCollection ConfigurePetstoreApiApizrManager(
    this IServiceCollection services,
    Action<IApizrExtendedManagerOptionsBuilder>? optionsBuilder = null)
{
    optionsBuilder ??= _ => { }; // Default empty options if null
    optionsBuilder += options => options
        .WithBaseAddress("https://petstore3.swagger.io/api/v3", ApizrDuplicateStrategy.Ignore)
        .WithDelegatingHandler<MyDelegatingHandler>()
        .ConfigureHttpClientBuilder(builder => builder
            .AddStandardResilienceHandler(config =>
            {
                config.Retry = new HttpRetryStrategyOptions
                {
                    UseJitter = true,
                    MaxRetryAttempts = 3,
                    Delay = TimeSpan.FromSeconds(0.5)
                };
            }))
        .WithInMemoryCacheHandler()
        .WithAutoMapperMappingHandler()
        .WithPriority()
        .WithOptionalMediation()
        .WithFileTransferOptionalMediation();

    return services.AddApizrManagerFor<IPetstoreApi>(optionsBuilder);
}
```

This comes in handy especially when generating multiple interfaces, by tag or endpoint. For example, the following `.refitter` settings file

```json
{
  "openApiPath": "../OpenAPI/v3.0/petstore.json",
  "namespace": "Petstore",
  "useDynamicQuerystringParameters": true,
  "multipleInterfaces": "ByTag",
  "naming": {
    "useOpenApiTitle": false,
    "interfaceName": "Petstore"
  },
  "dependencyInjectionSettings": {
    "baseUrl": "https://petstore3.swagger.io/api/v3",
    "httpMessageHandlers": [ "MyDelegatingHandler" ],
    "transientErrorHandler": "HttpResilience",
    "maxRetryCount": 3,
    "firstBackoffRetryInSeconds": 0.5
  },
  "apizrSettings": {
    "withRequestOptions": true, // Recommended to include an Apizr request options parameter to Refit interface methods
    "withRegistrationHelper": true, // Mandatory to actually generate the Apizr registration extended method
    "withCacheProvider": "InMemory", // Optional, default is None
    "withPriority": true, // Optional, default is false
    "withMediation": true, // Optional, default is false
    "withOptionalMediation": true, // Optional, default is false
    "withMappingProvider": "AutoMapper", // Optional, default is None
    "withFileTransfer": true // Optional, default is false
  }
}
```

Will generate a single `ConfigurePetstoreApizrManagers()` extension method that may contain dependency injection configuration code for multiple interfaces like this

```csharp
public static IServiceCollection ConfigurePetstoreApizrManagers(
    this IServiceCollection services,
    Action<IApizrExtendedCommonOptionsBuilder>? optionsBuilder = null)
{
    optionsBuilder ??= _ => { }; // Default empty options if null
    optionsBuilder += options => options
        .WithBaseAddress("https://petstore3.swagger.io/api/v3", ApizrDuplicateStrategy.Ignore)
        .WithDelegatingHandler<MyDelegatingHandler>()
        .ConfigureHttpClientBuilder(builder => builder
            .AddStandardResilienceHandler(config =>
            {
                config.Retry = new HttpRetryStrategyOptions
                {
                    UseJitter = true,
                    MaxRetryAttempts = 3,
                    Delay = TimeSpan.FromSeconds(0.5)
                };
            }))
        .WithInMemoryCacheHandler()
        .WithAutoMapperMappingHandler()
        .WithPriority()
        .WithOptionalMediation()
        .WithFileTransferOptionalMediation();

    return services.AddApizr(
        registry => registry
            .AddManagerFor<IPetApi>()
            .AddManagerFor<IStoreApi>()
            .AddManagerFor<IUserApi>(),
        optionsBuilder);

}
```

Here, `IPetApi`, `IStoreApi` and `IUserApi` are the generated interfaces which share the same common configuration defined from the `.refitter` file.

### [Static](#tab/tabid-static)

To enable Apizr static builder code generation, you need at least to set the `withRegistrationHelper` property to `true` and leave the `DependencyInjectionSettings` section to null in the `.refitter` settings file.
This is what the `.refitter` settings file may look like, depending on you configuration:

```json
{
  "openApiPath": "../OpenAPI/v3.0/petstore.json",
  "namespace": "Petstore",
  "useDynamicQuerystringParameters": true,
  "apizrSettings": {
    "withRequestOptions": true, // Recommended to include an Apizr request options parameter to Refit interface methods
    "withRegistrationHelper": true, // Mandatory to actually generate the Apizr registration extended method
    "withCacheProvider": "Akavache", // Optional, default is None
    "withPriority": true, // Optional, default is false
    "withMappingProvider": "AutoMapper", // Optional, default is None
    "withFileTransfer": true // Optional, default is false
  }
}
```

which will generate a static builder method called `BuildPetstore30ApizrManager()`. The generated builder method depends on [`Apizr`](https://www.nuget.org/packages/Apizr) library and looks like this:

```cs
public static IApizrManager<ISwaggerPetstoreOpenAPI30> BuildPetstore30ApizrManager(Action<IApizrManagerOptionsBuilder> optionsBuilder)
{
    optionsBuilder ??= _ => { }; // Default empty options if null
    optionsBuilder += options => options
        .WithAkavacheCacheHandler()
        .WithAutoMapperMappingHandler(new MapperConfiguration(config => { /* YOUR_MAPPINGS_HERE */ }))
        .WithPriority();

    return ApizrBuilder.Current.CreateManagerFor<ISwaggerPetstoreOpenAPI30>(optionsBuilder);
}
```

This comes in handy especially when generating multiple interfaces, by tag or endpoint. For example, the following `.refitter` settings file

```json
{
  "openApiPath": "../OpenAPI/v3.0/petstore.json",
  "namespace": "Petstore",
  "useDynamicQuerystringParameters": true,
  "multipleInterfaces": "ByTag",
  "naming": {
    "useOpenApiTitle": false,
    "interfaceName": "Petstore"
  },
  "dependencyInjectionSettings": {
    "baseUrl": "https://petstore3.swagger.io/api/v3",
    "httpMessageHandlers": [ "MyDelegatingHandler" ],
    "transientErrorHandler": "HttpResilience",
    "maxRetryCount": 3,
    "firstBackoffRetryInSeconds": 0.5
  },
  "apizrSettings": {
    "withRequestOptions": true, // Recommended to include an Apizr request options parameter to Refit interface methods
    "withRegistrationHelper": true, // Mandatory to actually generate the Apizr registration extended method
    "withCacheProvider": "InMemory", // Optional, default is None
    "withPriority": true, // Optional, default is false
    "withMediation": true, // Optional, default is false
    "withOptionalMediation": true, // Optional, default is false
    "withMappingProvider": "AutoMapper", // Optional, default is None
    "withFileTransfer": true // Optional, default is false
  }
}
```

Will generate a single `BuildPetstoreApizrManagers()` builder method that may contain configuration code for multiple interfaces like this

```csharp
public static IApizrRegistry BuildPetstoreApizrManagers(Action<IApizrCommonOptionsBuilder> optionsBuilder)
{
    optionsBuilder ??= _ => { }; // Default empty options if null
    optionsBuilder += options => options
        .WithAkavacheCacheHandler()
        .WithAutoMapperMappingHandler(new MapperConfiguration(config => { /* YOUR_MAPPINGS_HERE */ }))
        .WithPriority();

    return ApizrBuilder.Current.CreateRegistry(
        registry => registry
            .AddManagerFor<IPetApi>()
            .AddManagerFor<IStoreApi>()
            .AddManagerFor<IUserApi>(),
        optionsBuilder);
}
```

Here, `IPetApi`, `IStoreApi` and `IUserApi` are the generated interfaces which share the same common configuration defined from the `.refitter` file.

***

### Customizing the configuration

You may want to adjust apis configuration, for example, to add a custom header to requests. This can be done using the `Action<TApizrOptionsBuilder>` parameter while calling the generated method.
To know how to make Apizr fit your needs, please refer to the [Apizr documentation](https://www.apizr.net).

### Using the managers

Once you called the generated method, you will get an `IApizrManager<T>` instance that you can use to make requests to the API. Here's an example of how to use it:

```csharp
var result = await petstoreManager.ExecuteAsync((api, opt) => api.GetPetById(1, opt),
    options => options // Whatever final request options you want to apply
        .WithPriority(Priority.Background)
        .WithHeaders(["HeaderKey1: HeaderValue1"])
        .WithRequestTimeout("00:00:10")
        .WithCancellation(cts.Token));
```

Please head to the [Apizr documentation](https://www.apizr.net) to get more.

## System requirements

.NET 8.0 or .NET 9.0

## Testing

Refitter uses [TUnit](https://github.com/thomhurst/TUnit) as its testing framework instead of xUnit. TUnit was chosen for its superior performance, providing 3x faster test execution compared to xUnit. This significantly improves the developer experience when running the test suite locally and in CI/CD pipelines.

To run the tests:

```bash
dotnet test --solution src/Refitter.slnx -c Release
```

## Contributing

Please read our [contribution guidelines](CONTRIBUTING.md) if you'd like to contribute to the project.

## Contributors

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/neoGeneva"><img src="https://avatars.githubusercontent.com/u/804724?v=4?s=100" width="100px;" alt="Philip Cox"/><br /><sub><b>Philip Cox</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=neoGeneva" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://cam.macfar.land/"><img src="https://avatars.githubusercontent.com/u/1298847?v=4?s=100" width="100px;" alt="Cameron MacFarland"/><br /><sub><b>Cameron MacFarland</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=distantcam" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://kgame.tw"><img src="https://avatars.githubusercontent.com/u/3646532?v=4?s=100" width="100px;" alt="kgame"/><br /><sub><b>kgame</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=kgamecarter" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://yrki.no"><img src="https://avatars.githubusercontent.com/u/11573601?v=4?s=100" width="100px;" alt="Thomas Pettersen / Yrki"/><br /><sub><b>Thomas Pettersen / Yrki</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/1kvin"><img src="https://avatars.githubusercontent.com/u/46425980?v=4?s=100" width="100px;" alt="Artem"/><br /><sub><b>Artem</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3A1kvin" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/m7clarke"><img src="https://avatars.githubusercontent.com/u/47439144?v=4?s=100" width="100px;" alt="m7clarke"/><br /><sub><b>m7clarke</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Am7clarke" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kirides"><img src="https://avatars.githubusercontent.com/u/13602143?v=4?s=100" width="100px;" alt="kirides"/><br /><sub><b>kirides</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Akirides" title="Bug reports">🐛</a> <a href="https://github.com/christianhelle/refitter/commits?author=kirides" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/guillaumeserale"><img src="https://avatars.githubusercontent.com/u/6672406?v=4?s=100" width="100px;" alt="guillaumeserale"/><br /><sub><b>guillaumeserale</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=guillaumeserale" title="Code">💻</a> <a href="https://github.com/christianhelle/refitter/issues?q=author%3Aguillaumeserale" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Roflincopter"><img src="https://avatars.githubusercontent.com/u/1690243?v=4?s=100" width="100px;" alt="Dennis Brentjes"/><br /><sub><b>Dennis Brentjes</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=Roflincopter" title="Code">💻</a> <a href="#ideas-Roflincopter" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://www.linkedin.com/in/hickeydamian/"><img src="https://avatars.githubusercontent.com/u/57436?v=4?s=100" width="100px;" alt="Damian Hickey"/><br /><sub><b>Damian Hickey</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Adamianh" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/richardhu-lmg"><img src="https://avatars.githubusercontent.com/u/126430787?v=4?s=100" width="100px;" alt="richardhu-lmg"/><br /><sub><b>richardhu-lmg</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Arichardhu-lmg" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/brease-colin"><img src="https://avatars.githubusercontent.com/u/47358935?v=4?s=100" width="100px;" alt="brease-colin"/><br /><sub><b>brease-colin</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Abrease-colin" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/angelofb"><img src="https://avatars.githubusercontent.com/u/2032257?v=4?s=100" width="100px;" alt="angelofb"/><br /><sub><b>angelofb</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=angelofb" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/NoGRo"><img src="https://avatars.githubusercontent.com/u/5665273?v=4?s=100" width="100px;" alt="Dim Nogro"/><br /><sub><b>Dim Nogro</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=NoGRo" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/yadanilov19"><img src="https://avatars.githubusercontent.com/u/17161065?v=4?s=100" width="100px;" alt="yadanilov19"/><br /><sub><b>yadanilov19</b></sub></a><br /><a href="#ideas-yadanilov19" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=yadanilov19" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/danpowell88"><img src="https://avatars.githubusercontent.com/u/1100397?v=4?s=100" width="100px;" alt="Daniel Powell"/><br /><sub><b>Daniel Powell</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Adanpowell88" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Ekkeir"><img src="https://avatars.githubusercontent.com/u/36194685?v=4?s=100" width="100px;" alt="Ekkeir"/><br /><sub><b>Ekkeir</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=Ekkeir" title="Documentation">📖</a> <a href="https://github.com/christianhelle/refitter/issues?q=author%3AEkkeir" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/waylonmtz"><img src="https://avatars.githubusercontent.com/u/4604579?v=4?s=100" width="100px;" alt="Waylon Martinez"/><br /><sub><b>Waylon Martinez</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Awaylonmtz" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/vinaymadupathi"><img src="https://avatars.githubusercontent.com/u/38102900?v=4?s=100" width="100px;" alt="vkmadupa"/><br /><sub><b>vkmadupa</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Avinaymadupathi" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Noblix"><img src="https://avatars.githubusercontent.com/u/22076883?v=4?s=100" width="100px;" alt="Noblix"/><br /><sub><b>Noblix</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=Noblix" title="Code">💻</a> <a href="#ideas-Noblix" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://www.veezla.io"><img src="https://avatars.githubusercontent.com/u/230432?v=4?s=100" width="100px;" alt="Attila Hajdrik"/><br /><sub><b>Attila Hajdrik</b></sub></a><br /><a href="#ideas-attilah" title="Ideas, Planning, & Feedback">🤔</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/bielik01"><img src="https://avatars.githubusercontent.com/u/920950?v=4?s=100" width="100px;" alt="bielik01"/><br /><sub><b>bielik01</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Abielik01" title="Bug reports">🐛</a> <a href="#ideas-bielik01" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/naaeef"><img src="https://avatars.githubusercontent.com/u/43339071?v=4?s=100" width="100px;" alt="naaeef"/><br /><sub><b>naaeef</b></sub></a><br /><a href="#ideas-naaeef" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/alrz"><img src="https://avatars.githubusercontent.com/u/3105979?v=4?s=100" width="100px;" alt="Alireza Habibi"/><br /><sub><b>Alireza Habibi</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Aalrz" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/EEParker"><img src="https://avatars.githubusercontent.com/u/15874076?v=4?s=100" width="100px;" alt="Jeff Parker, PE"/><br /><sub><b>Jeff Parker, PE</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3AEEParker" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jods4"><img src="https://avatars.githubusercontent.com/u/3832820?v=4?s=100" width="100px;" alt="jods"/><br /><sub><b>jods</b></sub></a><br /><a href="#ideas-jods4" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/issues?q=author%3Ajods4" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/edimarquez"><img src="https://avatars.githubusercontent.com/u/41791719?v=4?s=100" width="100px;" alt="Edimarquez Medeiros"/><br /><sub><b>Edimarquez Medeiros</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=edimarquez" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/safakkesikci"><img src="https://avatars.githubusercontent.com/u/7067252?v=4?s=100" width="100px;" alt="safakkesikci"/><br /><sub><b>safakkesikci</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Asafakkesikci" title="Bug reports">🐛</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/folbrecht"><img src="https://avatars.githubusercontent.com/u/145537846?v=4?s=100" width="100px;" alt="folbrecht"/><br /><sub><b>folbrecht</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Afolbrecht" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/mortenlaursen"><img src="https://avatars.githubusercontent.com/u/28759737?v=4?s=100" width="100px;" alt="mortenlaursen"/><br /><sub><b>mortenlaursen</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=mortenlaursen" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/manuel-fernandez-rodriguez"><img src="https://avatars.githubusercontent.com/u/48520205?v=4?s=100" width="100px;" alt="manuel-fernandez-rodriguez"/><br /><sub><b>manuel-fernandez-rodriguez</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Amanuel-fernandez-rodriguez" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/eliyammine"><img src="https://avatars.githubusercontent.com/u/6644807?v=4?s=100" width="100px;" alt="Eli Yammine"/><br /><sub><b>Eli Yammine</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Aeliyammine" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kami-poi"><img src="https://avatars.githubusercontent.com/u/47322691?v=4?s=100" width="100px;" alt="kami-poi"/><br /><sub><b>kami-poi</b></sub></a><br /><a href="#ideas-kami-poi" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Xeevis"><img src="https://avatars.githubusercontent.com/u/5835044?v=4?s=100" width="100px;" alt="Xeevis"/><br /><sub><b>Xeevis</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3AXeevis" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/DJ4ddi"><img src="https://avatars.githubusercontent.com/u/1696102?v=4?s=100" width="100px;" alt="DJ4ddi"/><br /><sub><b>DJ4ddi</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=DJ4ddi" title="Code">💻</a> <a href="#ideas-DJ4ddi" title="Ideas, Planning, & Feedback">🤔</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/direncancatalkaya"><img src="https://avatars.githubusercontent.com/u/57223732?v=4?s=100" width="100px;" alt="direncancatalkaya"/><br /><sub><b>direncancatalkaya</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=direncancatalkaya" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/robpalm"><img src="https://avatars.githubusercontent.com/u/14939530?v=4?s=100" width="100px;" alt="Robert Palmqvist"/><br /><sub><b>Robert Palmqvist</b></sub></a><br /><a href="#ideas-robpalm" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=robpalm" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/TimothyMakkison"><img src="https://avatars.githubusercontent.com/u/49349513?v=4?s=100" width="100px;" alt="Tim M"/><br /><sub><b>Tim M</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=TimothyMakkison" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/janfolbrecht"><img src="https://avatars.githubusercontent.com/u/42186604?v=4?s=100" width="100px;" alt="janfolbrecht"/><br /><sub><b>janfolbrecht</b></sub></a><br /><a href="#ideas-janfolbrecht" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=janfolbrecht" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/osc-nseguin"><img src="https://avatars.githubusercontent.com/u/133910309?v=4?s=100" width="100px;" alt="Nick Seguin"/><br /><sub><b>Nick Seguin</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=osc-nseguin" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/david-brink-talogy"><img src="https://avatars.githubusercontent.com/u/43828739?v=4?s=100" width="100px;" alt="David Brink"/><br /><sub><b>David Brink</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Adavid-brink-talogy" title="Bug reports">🐛</a> <a href="https://github.com/christianhelle/refitter/commits?author=david-brink-talogy" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/dammitjanet"><img src="https://avatars.githubusercontent.com/u/952723?v=4?s=100" width="100px;" alt="Stu Wilson"/><br /><sub><b>Stu Wilson</b></sub></a><br /><a href="#ideas-dammitjanet" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=dammitjanet" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/sharpzilla"><img src="https://avatars.githubusercontent.com/u/16763634?v=4?s=100" width="100px;" alt="sharpzilla"/><br /><sub><b>sharpzilla</b></sub></a><br /><a href="#ideas-sharpzilla" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Jappinen"><img src="https://avatars.githubusercontent.com/u/44408497?v=4?s=100" width="100px;" alt="Tatu"/><br /><sub><b>Tatu</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3AJappinen" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://www.respawnsive.com"><img src="https://avatars.githubusercontent.com/u/4272307?v=4?s=100" width="100px;" alt="Jérémy BRUN-PICARD"/><br /><sub><b>Jérémy BRUN-PICARD</b></sub></a><br /><a href="#ideas-JeremyBP" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=JeremyBP" title="Code">💻</a> <a href="https://github.com/christianhelle/refitter/commits?author=JeremyBP" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ebarnard"><img src="https://avatars.githubusercontent.com/u/1059683?v=4?s=100" width="100px;" alt="Ed Barnard"/><br /><sub><b>Ed Barnard</b></sub></a><br /><a href="#ideas-ebarnard" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/bastiennoel93"><img src="https://avatars.githubusercontent.com/u/18250350?v=4?s=100" width="100px;" alt="bastien.noel"/><br /><sub><b>bastien.noel</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Abastiennoel93" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/MeikelLP"><img src="https://avatars.githubusercontent.com/u/11669846?v=4?s=100" width="100px;" alt="Meikel Philipp"/><br /><sub><b>Meikel Philipp</b></sub></a><br /><a href="#ideas-MeikelLP" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://berkselvi.dev"><img src="https://avatars.githubusercontent.com/u/54676516?v=4?s=100" width="100px;" alt="Berk Selvi"/><br /><sub><b>Berk Selvi</b></sub></a><br /><a href="#ideas-berkslv" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=berkslv" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://www.thebuildsheet.com/"><img src="https://avatars.githubusercontent.com/u/9909564?v=4?s=100" width="100px;" alt="Joshua Ozeri"/><br /><sub><b>Joshua Ozeri</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3AXeClutch" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://kancane.nl/"><img src="https://avatars.githubusercontent.com/u/701534?v=4?s=100" width="100px;" alt="Ryan Heath"/><br /><sub><b>Ryan Heath</b></sub></a><br /><a href="#ideas-ryanheath" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=ryanheath" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Briaoeuidhtns"><img src="https://avatars.githubusercontent.com/u/7530156?v=4?s=100" width="100px;" alt="Brian Brunner"/><br /><sub><b>Brian Brunner</b></sub></a><br /><a href="#ideas-Briaoeuidhtns" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/fsamiec"><img src="https://avatars.githubusercontent.com/u/13166020?v=4?s=100" width="100px;" alt="Frank Samiec"/><br /><sub><b>Frank Samiec</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=fsamiec" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/fabioloreggian"><img src="https://avatars.githubusercontent.com/u/31480768?v=4?s=100" width="100px;" alt="Fabio Loreggian"/><br /><sub><b>Fabio Loreggian</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Afabioloreggian" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/geometrikal"><img src="https://avatars.githubusercontent.com/u/1187528?v=4?s=100" width="100px;" alt="geometrikal"/><br /><sub><b>geometrikal</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Ageometrikal" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/shubinp"><img src="https://avatars.githubusercontent.com/u/25763776?v=4?s=100" width="100px;" alt="Shubin Pavel"/><br /><sub><b>Shubin Pavel</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=shubinp" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://www.netindustry.nl"><img src="https://avatars.githubusercontent.com/u/303034?v=4?s=100" width="100px;" alt="Wiebe Tijsma"/><br /><sub><b>Wiebe Tijsma</b></sub></a><br /><a href="#ideas-zidad" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/hgdemers"><img src="https://avatars.githubusercontent.com/u/169190876?v=4?s=100" width="100px;" alt="Henri Demers"/><br /><sub><b>Henri Demers</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Ahgdemers" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Fargekritt"><img src="https://avatars.githubusercontent.com/u/11448941?v=4?s=100" width="100px;" alt="Amund"/><br /><sub><b>Amund</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=Fargekritt" title="Code">💻</a> <a href="https://github.com/christianhelle/refitter/issues?q=author%3AFargekritt" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/velvolue"><img src="https://avatars.githubusercontent.com/u/155445968?v=4?s=100" width="100px;" alt="Vegard Løkken"/><br /><sub><b>Vegard Løkken</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=velvolue" title="Code">💻</a> <a href="https://github.com/christianhelle/refitter/issues?q=author%3Avelvolue" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/brad-technologik"><img src="https://avatars.githubusercontent.com/u/159975063?v=4?s=100" width="100px;" alt="brad-technologik"/><br /><sub><b>brad-technologik</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Abrad-technologik" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/jaroslaw-dutka"><img src="https://avatars.githubusercontent.com/u/40070136?v=4?s=100" width="100px;" alt="Jarosław Dutka"/><br /><sub><b>Jarosław Dutka</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=jaroslaw-dutka" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/maksionkin"><img src="https://avatars.githubusercontent.com/u/11482043?v=4?s=100" width="100px;" alt="maksionkin"/><br /><sub><b>maksionkin</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Amaksionkin" title="Bug reports">🐛</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/wocasella"><img src="https://avatars.githubusercontent.com/u/37601504?v=4?s=100" width="100px;" alt="wocasella"/><br /><sub><b>wocasella</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Awocasella" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Metziell"><img src="https://avatars.githubusercontent.com/u/33788844?v=4?s=100" width="100px;" alt="Petr Zika"/><br /><sub><b>Petr Zika</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3AMetziell" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/sebastian-wachsmuth"><img src="https://avatars.githubusercontent.com/u/87482207?v=4?s=100" width="100px;" alt="Sebastian Wachsmuth"/><br /><sub><b>Sebastian Wachsmuth</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=sebastian-wachsmuth" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/qrzychu"><img src="https://avatars.githubusercontent.com/u/4633810?v=4?s=100" width="100px;" alt="qrzychu"/><br /><sub><b>qrzychu</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Aqrzychu" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://stbl.com"><img src="https://avatars.githubusercontent.com/u/43214081?v=4?s=100" width="100px;" alt="Александр"/><br /><sub><b>Александр</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Alowern1ght" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/AragornHL"><img src="https://avatars.githubusercontent.com/u/18478623?v=4?s=100" width="100px;" alt="AragornHL"/><br /><sub><b>AragornHL</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=AragornHL" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kmfd3s"><img src="https://avatars.githubusercontent.com/u/5180898?v=4?s=100" width="100px;" alt="kmfd3s"/><br /><sub><b>kmfd3s</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=kmfd3s" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/pfeigl"><img src="https://avatars.githubusercontent.com/u/237891?v=4?s=100" width="100px;" alt="Philipp Feigl"/><br /><sub><b>Philipp Feigl</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Apfeigl" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://tommihaapa.fi/"><img src="https://avatars.githubusercontent.com/u/47896768?v=4?s=100" width="100px;" alt="Tommi Haapa"/><br /><sub><b>Tommi Haapa</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Atommieemeli" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/MrScottyTay"><img src="https://avatars.githubusercontent.com/u/24463321?v=4?s=100" width="100px;" alt="Scott Taylor"/><br /><sub><b>Scott Taylor</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3AMrScottyTay" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/sb-chericks"><img src="https://avatars.githubusercontent.com/u/160796564?v=4?s=100" width="100px;" alt="sb-chericks"/><br /><sub><b>sb-chericks</b></sub></a><br /><a href="#ideas-sb-chericks" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/issues?q=author%3Asb-chericks" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/SWarnberg"><img src="https://avatars.githubusercontent.com/u/26246346?v=4?s=100" width="100px;" alt="Staffan Wärnberg"/><br /><sub><b>Staffan Wärnberg</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3ASWarnberg" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/lilinus"><img src="https://avatars.githubusercontent.com/u/78953007?v=4?s=100" width="100px;" alt="Linus Hamlin"/><br /><sub><b>Linus Hamlin</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Alilinus" title="Bug reports">🐛</a> <a href="https://github.com/christianhelle/refitter/commits?author=lilinus" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://marcohern.com/en/#about"><img src="https://avatars.githubusercontent.com/u/5376517?v=4?s=100" width="100px;" alt="Marco Hernandez"/><br /><sub><b>Marco Hernandez</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Amarcohern" title="Bug reports">🐛</a> <a href="#ideas-marcohern" title="Ideas, Planning, & Feedback">🤔</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/david-pw"><img src="https://avatars.githubusercontent.com/u/178433875?v=4?s=100" width="100px;" alt="david-pw"/><br /><sub><b>david-pw</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Adavid-pw" title="Bug reports">🐛</a> <a href="https://github.com/christianhelle/refitter/commits?author=david-pw" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/eoma-knowit"><img src="https://avatars.githubusercontent.com/u/150660175?v=4?s=100" width="100px;" alt="eoma-knowit"/><br /><sub><b>eoma-knowit</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=eoma-knowit" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/0xced"><img src="https://avatars.githubusercontent.com/u/51363?v=4?s=100" width="100px;" alt="Cédric Luthi"/><br /><sub><b>Cédric Luthi</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=0xced" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://christophdebaene.be"><img src="https://avatars.githubusercontent.com/u/642207?v=4?s=100" width="100px;" alt="Christoph De Baene"/><br /><sub><b>Christoph De Baene</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Achristophdebaene" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/7amou3"><img src="https://avatars.githubusercontent.com/u/993610?v=4?s=100" width="100px;" alt="7amou3"/><br /><sub><b>7amou3</b></sub></a><br /><a href="#ideas-7amou3" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://haldencollier.com"><img src="https://avatars.githubusercontent.com/u/25365699?v=4?s=100" width="100px;" alt="Halden"/><br /><sub><b>Halden</b></sub></a><br /><a href="#ideas-HGCollier" title="Ideas, Planning, & Feedback">🤔</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/mhartmair-cubido"><img src="https://avatars.githubusercontent.com/u/164479174?v=4?s=100" width="100px;" alt="mhartmair-cubido"/><br /><sub><b>mhartmair-cubido</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Amhartmair-cubido" title="Bug reports">🐛</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/karoberts"><img src="https://avatars.githubusercontent.com/u/7482126?v=4?s=100" width="100px;" alt="Keith Roberts"/><br /><sub><b>Keith Roberts</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3Akaroberts" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kmc059000"><img src="https://avatars.githubusercontent.com/u/660670?v=4?s=100" width="100px;" alt="Kenneth Crawford"/><br /><sub><b>Kenneth Crawford</b></sub></a><br /><a href="#ideas-kmc059000" title="Ideas, Planning, & Feedback">🤔</a> <a href="https://github.com/christianhelle/refitter/commits?author=kmc059000" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/0x2badc0de"><img src="https://avatars.githubusercontent.com/u/173240018?v=4?s=100" width="100px;" alt="0x2badc0de"/><br /><sub><b>0x2badc0de</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/issues?q=author%3A0x2badc0de" title="Bug reports">🐛</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://meteion.ca"><img src="https://avatars.githubusercontent.com/u/6116333?v=4?s=100" width="100px;" alt="Tylor Pater"/><br /><sub><b>Tylor Pater</b></sub></a><br /><a href="https://github.com/christianhelle/refitter/commits?author=frogcrush" title="Code">💻</a></td>
    </tr>
  </tbody>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

#

For tips and tricks on software development, check out [my blog](https://christianhelle.com)

If you find this useful and feel a bit generous then feel free to [buy me a coffee ☕](https://www.buymeacoffee.com/christianhelle)
 readmeEtag: '"3ff1d9f32093a89b55599493f184a57b2647bba9"' readmeLastModified: Thu, 22 Jan 2026 21:08:52 GMT repositoryId: 598846021 description: >- A tool for generating Refit interfaces and contracts from OpenAPI (Swagger) specifications created: '2023-02-07T23:18:12Z' updated: '2026-02-05T22:44:54Z' language: C# archived: false stars: 370 watchers: 2 forks: 63 owner: christianhelle logo: https://avatars.githubusercontent.com/u/710400?v=4 license: MIT repoEtag: '"b72cc13613ed3893344c9e5bc0208892b574fe4f4562508cfdd5fd9491a1094b"' repoLastModified: Thu, 05 Feb 2026 22:44:54 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/paypal/paypal-rest-api-specifications v3: true id: fe6bc87a1d5d10f013a6fcfe40ea9186 repositoryMetadata: base64Readme: >- IyBQYXlQYWwgUkVTVCBBUEkgU3BlY2lmaWNhdGlvbnMKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyB0aGUgc3BlY2lmaWNhdGlvbiBmaWxlcyBmb3IgW1BheVBhbCBSRVNUIEFQSXNdKGh0dHBzOi8vZGV2ZWxvcGVyLnBheXBhbC5jb20vYXBpL3Jlc3QvKS4KCllvdSBjYW4gdHJ5IG91ciBSRVNUIEFQSXMgaW4gUG9zdG1hbiB3aXRob3V0IGEgUGF5UGFsIERldmVsb3BlciBhY2NvdW50LiBMZWFybiBtb3JlIGluIG91ciBbUG9zdG1hbiBndWlkZV0oaHR0cHM6Ly9kZXZlbG9wZXIucGF5cGFsLmNvbS9hcGkvcmVzdC9wb3N0bWFuLykuCgpbIVtSdW4gaW4gUG9zdG1hbl0oaHR0cHM6Ly9ydW4ucHN0bW4uaW8vYnV0dG9uLnN2ZyldKGh0dHBzOi8vZ29kLmd3LnBvc3RtYW4uY29tL3J1bi1jb2xsZWN0aW9uLzE5MDI0MTIyLTkyYTg1ZDBlLTUxZTctNDdkYS05ZjgzLWM0NWRjYjFjZGYyND9hY3Rpb249Y29sbGVjdGlvbiUyRmZvcmsmY29sbGVjdGlvbi11cmw9ZW50aXR5SWQlM0QxOTAyNDEyMi05MmE4NWQwZS01MWU3LTQ3ZGEtOWY4My1jNDVkY2IxY2RmMjQlMjZlbnRpdHlUeXBlJTNEY29sbGVjdGlvbiUyNndvcmtzcGFjZUlkJTNEMzQ1MzAwZTYtMzQ2ZS00MmUwLWFlZDEtNTM3MTc5MTlhZWYwKQoKIyMgU3RlcHMgdG8gZ2VuZXJhdGUgY29kZSBmcm9tIHRoZSBzcGVjaWZpY2F0aW9uCgo+IF9Ob3RlOiBTcGVjIGZpbGUgdXNlZDogb3BlbmFwaS9jaGVja291dF9vcmRlcnNfdjIuanNvbl8KCiMjIyBKYXZhCiAxLiBgbnBtIHJ1biBjb2RlZ2VuLWphdmEgLS0gb3BlbmFwaS9jaGVja291dF9vcmRlcnNfdjIuanNvbiAtLWFydGlmYWN0LWlkIG9yZGVyc2AKIDIuIGBjZCBnZW5lcmF0ZWQvamF2YWAKIDMuIGBtdm4gY2xlYW4gaW5zdGFsbGAKClRoZSBhYm92ZSB3aWxsIGluc3RhbGwgdGhlIGBvcmRlcnNgIGFydGlmYWN0IHRvIHRoZSBsb2NhbCBtYXZlbiByZXBvc2l0b3J5CgojIyBBdmFpbGFibGUgc2NyaXB0cwoKYGBgCiBucG0gcnVuClNjcmlwdHMgYXZhaWxhYmxlIGluIEBwYXlwYWwvcGF5cGFsLXJlc3QtYXBpLXNwZWNpZmljYXRpb25zQDEuMC4wIHZpYSBgbnBtIHJ1bi1zY3JpcHRgOgogIHByZXZpZXcKICAgIHJlZG9jbHkgcHJldmlldy1kb2NzCiAgYnVuZGxlCiAgICByZWRvY2x5IGJ1bmRsZQogIGJ1bmRsZS1kZXJlZgogICAgcmVkb2NseSBidW5kbGUgLWQgdHJ1ZQogIGxpbnQtcmVkb2NseQogICAgcmVkb2NseSAtLWZvcm1hdCBzdW1tYXJ5IGxpbnQKICBsaW50LXNwZWN0cmFsCiAgICBzcGVjdHJhbCAtLWZvcm1hdCBwcmV0dHkgbGludAogIGxpbnQtb3BlbmFwaQogICAgb3BlbmFwaS1nZW5lcmF0b3ItY2xpIHZhbGlkYXRlIC1pCiAgY29kZWdlbi1qYXZhCiAgICBvcGVuYXBpLWdlbmVyYXRvci1jbGkgZ2VuZXJhdGUgLWcgamF2YSAtbyBnZW5lcmF0ZWQvamF2YSAtaQogIGNvZGVnZW4tdHlwZXNjcmlwdC1ub2RlCiAgICBvcGVuYXBpLWdlbmVyYXRvci1jbGkgZ2VuZXJhdGUgLWcgdHlwZXNjcmlwdC1ub2RlIC1vIGdlbmVyYXRlZC90eXBlc2NyaXB0LW5vZGUgLWkKYGBgCgojIyBBdmFpbGFibGUgc3BlY2lmaWNhdGlvbnMKCiMjIyBPcGVuQVBJIDMuMC4zCgpUaGUgW09wZW5BUEldKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uL3YzLykgc3BlY2lmaWNhdGlvbiBpcyBhIHByb2dyYW1taW5nIGxhbmd1YWdlIGFnbm9zdGljIHN0YW5kYXJkIHRoYXQgbGV0cyBwcm92aWRlcnMgZGVzY3JpYmUgdGhlIGludGVyZmFjZSBvZiB0aGVpciBIVFRQIEFQSXMuIFRoaXMgYWxsb3dzIGJvdGggaHVtYW5zIGFuZCBtYWNoaW5lcyB0byBkaXNjb3ZlciB0aGUgY2FwYWJpbGl0aWVzIG9mIGFuIEFQSSB3aXRob3V0IG5lZWRpbmcgdG8gZmlyc3QgcmVhZCBkb2N1bWVudGF0aW9uIG9yIHVuZGVyc3RhbmQgdGhlIGltcGxlbWVudGF0aW9uLgoKfCBBUEkgTmFtZSAgICAgICAgICAgICAgfCBWZXJzaW9uIHwgTGlua3MgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwKfCBBZGQgVHJhY2tpbmcgICAgICAgICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9zaGlwcGluZ19zaGlwbWVudF90cmFja2luZ192MS5qc29uKSwgW0FQSSBSZWZlcmVuY2VdKGh0dHBzOi8vZGV2ZWxvcGVyLnBheXBhbC5jb20vZG9jcy9hcGkvdHJhY2tpbmcvdjEvKSAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBDYXRhbG9nIFByb2R1Y3RzICAgICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9jYXRhbG9nc19wcm9kdWN0c192MS5qc29uKSwgW0FQSSBSZWZlcmVuY2VdKGh0dHBzOi8vZGV2ZWxvcGVyLnBheXBhbC5jb20vZG9jcy9hcGkvY2F0YWxvZy1wcm9kdWN0cy92MS8pICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBEaXNwdXRlcyAgICAgICAgICAgICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9jdXN0b21lcl9kaXNwdXRlc192MS5qc29uKSwgW0FQSSBSZWZlcmVuY2VdKGh0dHBzOi8vZGV2ZWxvcGVyLnBheXBhbC5jb20vZG9jcy9hcGkvY3VzdG9tZXItZGlzcHV0ZXMvdjEvKSAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBJbnZvaWNlcyAgICAgICAgICAgICAgfCB2MiAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9pbnZvaWNpbmdfdjIuanNvbiksIFtBUEkgUmVmZXJlbmNlXShodHRwczovL2RldmVsb3Blci5wYXlwYWwuY29tL2RvY3MvYXBpL2ludm9pY2luZy92Mi8pICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBPcmRlcnMgICAgICAgICAgICAgICAgfCB2MiAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9jaGVja291dF9vcmRlcnNfdjIuanNvbiksIFtBUEkgUmVmZXJlbmNlXShodHRwczovL2RldmVsb3Blci5wYXlwYWwuY29tL2RvY3MvYXBpL29yZGVycy92Mi8pICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBQYXJ0bmVyIFJlZmVycmFscyAgICAgfCB2MiAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9jdXN0b21lcl9wYXJ0bmVyX3JlZmVycmFsc192Mi5qc29uKSwgW0FQSSBSZWZlcmVuY2VdKGh0dHBzOi8vZGV2ZWxvcGVyLnBheXBhbC5jb20vZG9jcy9hcGkvcGFydG5lci1yZWZlcnJhbHMvdjIvKSAgICAgICAgICAgICAgICAgIHwKfCBQYXltZW50IEV4cGVyaWVuY2UgICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9wYXltZW50LWV4cGVyaWVuY2Vfd2ViX2V4cGVyaWVuY2VfcHJvZmlsZXNfdjEuanNvbiksIFtBUEkgUmVmZXJlbmNlXShodHRwczovL2RldmVsb3Blci5wYXlwYWwuY29tL2RvY3MvYXBpL3BheW1lbnQtZXhwZXJpZW5jZS92MS8pIHwKfCBQYXltZW50IE1ldGhvZCBUb2tlbnMgfCB2MyAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS92YXVsdF9wYXltZW50X3Rva2Vuc192My5qc29uKSwgW0FQSSBSZWZlcmVuY2VdKGh0dHBzOi8vZGV2ZWxvcGVyLnBheXBhbC5jb20vZG9jcy9hcGkvcGF5bWVudC10b2tlbnMvdjMvKSAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBQYXltZW50cyAgICAgICAgICAgICAgfCB2MiAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9wYXltZW50c19wYXltZW50X3YyLmpzb24pLCBbQVBJIFJlZmVyZW5jZV0oaHR0cHM6Ly9kZXZlbG9wZXIucGF5cGFsLmNvbS9kb2NzL2FwaS9wYXltZW50cy92Mi8pICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBQYXlvdXRzICAgICAgICAgICAgICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9wYXltZW50c19wYXlvdXRzX2JhdGNoX3YxLmpzb24pLCBbQVBJIFJlZmVyZW5jZV0oaHR0cHM6Ly9kZXZlbG9wZXIucGF5cGFsLmNvbS9kb2NzL2FwaS9wYXltZW50cy5wYXlvdXRzLWJhdGNoL3YxLykgICAgICAgICAgICAgICAgIHwKfCBTdWJzY3JpcHRpb25zICAgICAgICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9iaWxsaW5nX3N1YnNjcmlwdGlvbnNfdjEuanNvbiksIFtBUEkgUmVmZXJlbmNlXShodHRwczovL2RldmVsb3Blci5wYXlwYWwuY29tL2RvY3MvYXBpL3N1YnNjcmlwdGlvbnMvdjEvKSAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBUcmFuc2FjdGlvbiBTZWFyY2ggICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9yZXBvcnRpbmdfdHJhbnNhY3Rpb25zX3YxLmpzb24pLCBbQVBJIFJlZmVyZW5jZV0oaHR0cHM6Ly9kZXZlbG9wZXIucGF5cGFsLmNvbS9kb2NzL2FwaS90cmFuc2FjdGlvbi1zZWFyY2gvdjEvKSAgICAgICAgICAgICAgICAgICAgIHwKfCBXZWJob29rcyBNYW5hZ2VtZW50ICAgfCB2MSAgICAgIHwgW09wZW5BUEldKC4vb3BlbmFwaS9ub3RpZmljYXRpb25zX3dlYmhvb2tzX3YxLmpzb24pLCBbQVBJIFJlZmVyZW5jZV0oaHR0cHM6Ly9kZXZlbG9wZXIucGF5cGFsLmNvbS9kb2NzL2FwaS93ZWJob29rcy92MS8pICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKCgojIyMgU3VwcG9ydAoKW1BheVBhbCdzIERldmVsb3BlciBzdXBwb3J0XShodHRwczovL2RldmVsb3Blci5wYXlwYWwuY29tL2RvY3Mvc3VwcG9ydC8pCgoKIAo= readmeEtag: '"61e2bf90daf75f6fd2c3b8e5c4c9d9d6e780f987"' readmeLastModified: Fri, 25 Jul 2025 15:47:17 GMT repositoryId: 673944158 description: This repository contains the specification files for PayPal REST APIs. created: '2023-08-02T19:26:26Z' updated: '2026-02-05T19:33:42Z' language: null archived: false stars: 359 watchers: 31 forks: 274 owner: paypal logo: https://avatars.githubusercontent.com/u/476675?v=4 license: Apache-2.0 repoEtag: '"e94552745bc62ebbd0a8b1922a871ba020034fa19c5a76044886f1c712824e79"' repoLastModified: Thu, 05 Feb 2026 19:33:42 GMT category: Code Generators foundInMaster: true - source: openapi3 tags repository: https://github.com/grafana/thema v3: true repositoryMetadata: base64Readme: >- IyBUaGVtYQoKPiBbIVdBUk5JTkddCj4gVGhpcyByZXBvc2l0b3J5IGlzIG5vIGxvbmdlciBhY3RpdmVseSBtYWludGFpbmVkIG9yIHN1cHBvcnRlZC4KPiBJZiB5b3UgYXJlIGludGVyZXN0ZWQgaW4gY29udGludWluZyB0aGUgZGV2ZWxvcG1lbnQgb2YgdGhpcyBwcm9qZWN0LCBmZWVsIGZyZWUgdG8gZm9yayB0aGUgcmVwb3NpdG9yeSBhbmQgY3VzdG9taXplIGl0IHRvIHlvdXIgbmVlZHMuCgpUaGVtYSBpcyBhIHN5c3RlbSBmb3Igd3JpdGluZyBzY2hlbWFzLiBNdWNoIGxpa2UgSlNPTiBTY2hlbWEgb3IgT3BlbkFQSSwgaXQgaXMgZ2VuZXJhbC1wdXJwb3NlIGFuZCBpdHMgbW9zdCBvYnZpb3VzIGFwcGxpY2F0aW9uIGlzIGFzIGFuIFtJRExdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0ludGVyZmFjZV9kZXNjcmlwdGlvbl9sYW5ndWFnZSkuIEhvd2V2ZXIsIHRob3NlIHN5c3RlbXMgdHJlYXQgX2NoYW5naW5nXyBzY2hlbWFzIGFzIG91dCBvZiBzY29wZTogYSBzaW5nbGUgdmVyc2lvbiBvZiBhIHNjaGVtYSBmb3Igc29tZSBvYmplY3QgaXMgdGhlIGF0b21pYyB1bml0LCBhbmQgdmVyc2lvbmluZyBpcyBsZWZ0IHRvIG9wYXF1ZSBzdHJpbmdzIGluIGV4dGVybmFsIHN5c3RlbXMgbGlrZSBnaXQgb3IgSFRUUC4gVGhlbWEsIGJ5IGNvbnRyYXN0LCBtYWtlcyBzY2hlbWEgY2hhbmdlIGEgZmlyc3QtY2xhc3Mgc3lzdGVtIHByb3BlcnR5OiB0aGUgYXRvbWljIHVuaXQgaXMgdGhlIF9zZXRfIG9mIHNjaGVtYSBmb3Igc29tZSBvYmplY3QsIGl0ZXJhdGl2ZWx5IGFwcGVuZGVkIHRvIG92ZXIgdGltZSBhcyByZXF1aXJlbWVudHMgZXZvbHZlLgoKVGhlbWEncyBhcHByb2FjaCBpcyBub3ZlbCwgc28gYW4gYW5hbG9neSB0byB0aGUgZmFtaWxpYXIgbWF5IGhlbHAuIFsiQnJhbmNoaW5nIGJ5IGFic3RyYWN0aW9uIl0oaHR0cHM6Ly9tYXJ0aW5mb3dsZXIuY29tL2JsaWtpL0JyYW5jaEJ5QWJzdHJhY3Rpb24uaHRtbCkgc3VnZ2VzdHMgdGhhdCB5b3UgcmVmYWN0b3IgbGFyZ2UgYXBwbGljYXRpb25zIG5vdCB3aXRoIGxvbmctcnVubmluZyBWQ1MgYnJhbmNoZXMgYW5kIGJpZy1iYW5nIG1lcmdlcywgYnV0IGJ5IGxldHRpbmcgb2xkIGFuZCBuZXcgY29kZSBsaXZlIHNpZGUtYnktc2lkZSBvbiBgbWFpbmAsIGFuZCBjaG9vc2luZyBiZXR3ZWVuIHRoZW0gd2l0aCBsb2dpY2FsIGdhdGVzLCBsaWtlIFtmZWF0dXJlIGZsYWdzXShodHRwczovL2ZlYXR1cmVmbGFncy5pby9mZWF0dXJlLWZsYWdzLykuIFRoZW1hIGlzICJzY2hlbWEgdmVyc2lvbmluZyBieSBhYnN0cmFjdGlvbiI6IGFsbCB2ZXJzaW9ucyBvZiBhIHNjaGVtYSBsaXZlIHNpZGUtYnktc2lkZSBvbiBgbWFpbmAsIHdpdGhpbiBsb2dpY2FsIHN0cnVjdHVyZXMgVGhlbWEgZGVmaW5lcy4KClRoaXMgaG9saXN0aWMgdmlldyBhbGxvd3MgVGhlbWEgdG8gYWN0IGxpa2UgYSB0eXBlY2hlY2tlciwgYnV0IGZvciBjaGFuZ2Utc2FmZXR5IF9iZXR3ZWVuXyBzY2hlbWEgdmVyc2lvbnM6IGVpdGhlciBzY2hlbWEgdmVyc2lvbnMgbXVzdCBiZSBiYWNrd2FyZHMgY29tcGF0aWJsZSwgb3IgdGhlcmUgbXVzdCBleGlzdCBsb2dpYyB0byB0cmFuc2xhdGUgYSB2YWxpZCBpbnN0YW5jZSBvZiBzY2hlbWEgZnJvbSBvbmUgc2NoZW1hIHZlcnNpb24gdG8gdGhlIG5leHQuIFtDVUVdKGh0dHBzOi8vY3VlbGFuZy5vcmcpLCB0aGUgbGFuZ3VhZ2UgaW4gd2hpY2ggVGhlbWEgc2NoZW1hcyBhcmUgd3JpdHRlbiwgYWxsb3dzIFRoZW1hIHRvIFttZWNoYW5pY2FsbHkgdmVyaWZ5IHRoZXNlIHByb3BlcnRpZXNdKCNNYXR1cml0eSkuCgpUaGVzZSBjYXBhYmlsaXRpZXMgbWFrZSBUaGVtYSBhIGdlbmVyYWwgZnJhbWV3b3JrIGZvciBkZWNvdXBsaW5nIHRoZSBldm9sdXRpb24gb2YgY29tbXVuaWNhdGluZyBzeXN0ZW1zLiBUaGlzIGNhbiBiZSBvdXR3YXJkLWZhY2luZzogVGhlbWEncyBndWFyZHJhaWxzIGFsbG93IGFueW9uZSB0byBjcmVhdGUgQVBJcyB3aXRoIFN0cmlwZSdzIHJlbm93bmVkIFtiYWNrd2FyZHMgY29tcGF0aWJpbGl0eV0oaHR0cHM6Ly9zdHJpcGUuY29tL2RvY3MvdXBncmFkZXMpIGd1YXJhbnRlZXMuIE9yIGl0IGNhbiBiZSBpbndhcmQtZmFjaW5nOiBvciB0byBjaGFuZ2UgdGhlIG1lc3NhZ2VzIHBhc3NlZCBpbiBhIG1lc2ggb2YgbWljcm9zZXJ2aWNlcyB3aXRob3V0IGludHJpY2F0ZWx5IG9yY2hlc3RyYXRpbmcgZGVwbG95bWVudC4KCkxlYXJuIG1vcmUgaW4gb3VyIFtkb2NzXShodHRwczovL2dpdGh1Yi5jb20vZ3JhZmFuYS90aGVtYS90cmVlL21haW4vZG9jcyksIG9yIGluIHRoaXMgW292ZXJ2aWV3IHZpZGVvXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PVBwb1NfVGhudEVNKSEgKFNvbWUgdGhpbmdzIGhhdmUgYmVlbiByZW5hbWVkIHNpbmNlIHRoYXQgdmlkZW8sIGJ1dCB0aGUgbG9naWMgaXMgdW5jaGFuZ2VkLikKCiMjIFVzYWdlCgpUaGVtYSBkZWZpbmVzIHRoZSB3YXkgc2NoZW1hcyBhcmUgd3JpdHRlbiwgb3JnYW5pemluZyBlYWNoIG9iamVjdCdzIGhpc3RvcnkgaW50byBhICJsaW5lYWdlLiIgT25jZSBhdXRob3JlZCwgVGhlbWEgYWxzbyBwcm92aWRlcyB0b29scyBmb3Igd29ya2luZyB3aXRoIGxpbmVhZ2VzIHZpYSBhIGZldyBbYmFzaWMgb3BlcmF0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL2dyYWZhbmEvdGhlbWEvYmxvYi9tYWluL2RvY3Mvb3ZlcnZpZXcubWQjYWJvdXQtdGhlbWEtb3BlcmF0aW9ucykuIFRoZXJlIGFyZSBhIGZldyBkaWZmZXJlbnQgdXNhZ2UgcGF0dGVybnMsIGFsbCBsYXJnZWx5IGVxdWl2YWxlbnQgaW4gY2FwYWJpbGl0eToKCiogKipDTEk6KiogYSBDTEkgY29tbWFuZCB0aGF0IHByb3ZpZGVzIGFjY2VzcyB0byBUaGVtYSdzIGJhc2ljIG9wZXJhdGlvbnMsIG9uZSBsaW5lYWdlIHBlciBpbnZvY2F0aW9uLiBVc2UgaXQgZm9yIGZhc3QgZXhwbG9yYXRpb24gYW5kIHRlc3Rpbmcgb2Ygc2NoZW1hcywgb3IgYXMgYSB0b29sIGluIENJLgoqICoqU2VydmVyOioqIEFuIEhUVFAgc2VydmVyIHRoYXQgcHJvdmlkZXMgYWNjZXNzIHRvIFRoZW1hJ3MgYmFzaWMgb3BlcmF0aW9ucyBmb3IgYSBjb25maWd1cmFibGUgc2V0IG9mIGxpbmVhZ2VzLiBSdW4gaXQgYXMgYSBzdGF0ZWxlc3Mgc2lkZWNhciBpbiB5b3VyIGluZnJhc3RydWN0dXJlIG9yIG1pY3Jvc2VydmljZSBtZXNoLgoqICoqTGlicmFyeToqKiBhIGxpYnJhcnksIGltcG9ydGFibGUgaW4geW91ciBhcHBsaWNhdGlvbiBjb2RlLCB0aGF0IHByb3ZpZGVzIGEgY29udmVuaWVudCBpbnRlcmZhY2UgdG8gVGhlbWEncyBiYXNpYyBvcGVyYXRpb25zLCBhcyB3ZWxsIGFzIGhlbHBlcnMgZm9yIGNvbW1vbiB1c2FnZSBwYXR0ZXJucy4gTmF0dXJhbGx5IHRoZSBtb3N0IGZsZXhpYmxlLCBhbmQgdGhlIHJlY29tbWVuZGVkIGFwcHJvYWNoIGZvciBjcmVhdGluZyBuZXcgaGVscGVycywgc3VjaCBhcyBjb2RlIGdlbmVyYXRvcnMsIEFQSSBnZW5lcmF0b3JzLCBvciBhIHdob2xlIEt1YmVybmV0ZXMgb3BlcmF0b3IgZnJhbWV3b3JrLiAoQ3VycmVudGx5IG9ubHkgZm9yIEdvW15ldmFsdWF0b3JdKQoKVGhlIENMSSBhbmQgc2VydmVyIG1vZGVzIGFyZSBidW5kbGVkIHRvZ2V0aGVyIGluIHRoZSBgdGhlbWFgIGNvbW1hbmQuIFRvIGluc3RhbGw6CgpgYGBiYXNoCmdvIGluc3RhbGwgZ2l0aHViLmNvbS9ncmFmYW5hL3RoZW1hL2NtZC90aGVtYUBsYXRlc3QKYGBgCgojIyBNYXR1cml0eQoKVGhlbWEgaXMgYSB5b3VuZyBwcm9qZWN0LiBUaGUgZ29hbHMgYXJlIGxhcmdlLCBidXQgYm91bmRlZDogd2Ugd2lsbCBrbm93IHdoZW4gdGhlIGNvcmUgc3lzdGVtIGlzIGNvbXBsZXRlLiBBbmQgaXQgbW9zdGx5IGlzLCBub3cgLSB0aG91Z2ggc29tZSBicmVha2luZyBjaGFuZ2VzIHRvIGhvdyBzY2hlbWFzIGFyZSB3cml0dGVuIGFyZSBwbGFubmVkIGJlZm9yZSByZWFjaGluZyBzdGFiaWxpdHkuCgpJdCBpcyBub3QgeWV0IHJlY29tbWVuZGVkIHRvIHJlcGxhY2UgZXN0YWJsaXNoZWQsIHN0YWJsZSBzeXN0ZW1zIHdpdGggVGhlbWEsIGJ1dCBleHBlcmltZW50aW5nIHdpdGggZG9pbmcgc28gaXMgcmVhc29uYWJsZSAoYW5kIGFwcHJlY2lhdGVkISkuIEZvciBuZXdlciBwcm9qZWN0cywgVGhlbWEgbWF5IGJlIGEgZ29vZCBjaG9pY2UgdG9kYXk7IHRoZSBkZWNpc2lvbiBpcyBsaWtlbHkgdG8gY29tZSBkb3duIHRvIHdoZXRoZXIgdGhlIGxvbmctdGVybSBiZW5lZml0IG9mIGEgc2ltcGxlciBhcmNoaXRlY3R1cmUgZm9yIGF1dGhvcmluZywgY29tcG9zaW5nIGFuZCBldm9sdmluZyBzY2hlbWEgd2lsbCBvZmZzZXQgdGhlIHNob3J0LXRlcm0gY29zdCBvZiBzb21lIGluY29tcGxldGUgZnVuY3Rpb25hbGl0eSBhbmQgYnJlYWtpbmcgY2hhbmdlcy4KCiMjIFByaW9yL1JlbGF0ZWQgQXJ0CgpBIG51bWJlciBvZiBzeXN0ZW1zIHBhcnRpYWxseSBvdmVybGFwIHdpdGggVGhlbWEgLSBmb3Igc29tZSBkYXRhLCByb2xsaW5nIHRvZ2V0aGVyIGEgc2V0IG9mIHNjaGVtYSB3aXRoIHRoZSByZWxhdGlvbnMgYmV0d2VlbiB0aG9zZSBzY2hlbWEuCgoqIFtQcm9qZWN0IENhbWJyaWFdKGh0dHBzOi8vd3d3Lmlua2FuZHN3aXRjaC5jb20vY2FtYnJpYS8pIC0gVGhlbWEncyBjbG9zZXN0IGFuYWxvZ3VlLiBMaW1pdGVkIGluIHZlcmlmaWFiaWxpdHkgYnkgKGludGVudGlvbmFsbHkpIGJlaW5nIHdpdGhvdXQgYSBub3Rpb24gb2YgbGluZWFyIHNjaGVtYSBvcmRlcmluZyBhbmQgdmVyc2lvbmluZywgYW5kIGJlY2F1c2Ugc2NoZW1hIGFuZCB0cmFuc2xhdGlvbnMgYXJlIHdyaXR0ZW4gaW4gYSBUdXJpbmcgY29tcGxldGUgbGFuZ3VhZ2UgKFR5cGVzY3JpcHQpLgoqIFtLdWJlcm5ldGVzIHJlc291cmNlcyBhbmQgd2ViaG9vayBjb252ZXJzaW9uc10oaHR0cHM6Ly9rdWJlcm5ldGVzLmlvL2RvY3MvdGFza3MvZXh0ZW5kLWt1YmVybmV0ZXMvY3VzdG9tLXJlc291cmNlcy9jdXN0b20tcmVzb3VyY2UtZGVmaW5pdGlvbi12ZXJzaW9uaW5nLyNzcGVjaWZ5LW11bHRpcGxlLXZlcnNpb25zKSAtIFNpbWlsYXIgZ29hbHM6IG11bHRpcGxlIHZlcnNpb25zIG9mIHJlc291cmNlcyAoc2NoZW1hKSBhbmQgY29udmVydGliaWxpdHkgYmV0d2VlbiB0aGVtLiBMaW1pdGVkIGluIHZlcmlmaWFiaWxpdHkgYnkgcmVseWluZyBvbiBjb252ZW50aW9uIGZvciBncm91cGluZyBzY2hlbWFzLCBhbmQgYnkgZXhwcmVzc2luZyB0cmFuc2xhdGlvbiBpbiBhIFR1cmluZyBjb21wbGV0ZSBsYW5ndWFnZSAoR28pLgoqIFtTdHJpcGUncyBIVFRQIEFQSV0oaHR0cHM6Ly9zdHJpcGUuY29tL2RvY3MvdXBncmFkZXMpIC0gZXhoaWJpdHMgdGhlIGJhY2t3YXJkcyBjb21wYXRpYmlsaXR5IHByb3BlcnRpZXMgYW4gQVBJIGNhbiBoYXZlIHRoYXQgYXJpc2UgZnJvbSBhIHNjaGVtYSBzeXN0ZW0gd2l0aCB0cmFuc2xhdGFiaWxpdHkuCgoKW15ldmFsdWF0b3JdOgogICAgVXNpbmcgVGhlbWEgYXMgYSBsaWJyYXJ5IGluIGEgbGFuZ3VhZ2UgZGVwZW5kcyBvbiBhIENVRSBldmFsdWF0b3IgZm9yIHRoYXQgbGFuZ3VhZ2UuIEN1cnJlbnRseSwgdGhlIG9ubHkgQ1VFIGV2YWx1YXRvciBpcyB3cml0dGVuIGluIEdvLgo= readmeEtag: '"3c9e095f399071232622e0315a96a31db765d154"' readmeLastModified: Wed, 05 Jun 2024 11:00:52 GMT repositoryId: 419709887 description: A CUE-based framework for portable, evolvable schema created: '2021-10-21T12:16:22Z' updated: '2025-10-26T18:01:18Z' language: Go archived: false stars: 232 watchers: 142 forks: 11 owner: grafana logo: https://avatars.githubusercontent.com/u/7195757?v=4 license: Apache-2.0 repoEtag: '"e9b668b046340b6c251cf0b73074a176473b15c77fa7474ed0551f78529b1133"' repoLastModified: Sun, 26 Oct 2025 18:01:18 GMT foundInMaster: true category: - Server - Server Implementations id: db932e8e382d5215f30e4dca6ab5a378 - source: openapi3 tags repository: https://github.com/ronnypfannschmidt/prance v3: true repositoryMetadata: base64Readme: >- |License| |PyPI| |Python Versions| |Package Format| |Package Status|

|Logo|

Prance provides parsers for `Swagger/OpenAPI
2.0 and 3.0 <http://swagger.io/specification/>`__ API specifications in Python.
It uses `openapi\_spec\_validator <https://github.com/p1c2u/openapi-spec-validator>`__,
`swagger\_spec\_validator <https://github.com/Yelp/swagger_spec_validator>`__ or
`flex <https://github.com/pipermerriam/flex>`__
to validate specifications, but additionally resolves `JSON
references <https://tools.ietf.org/html/draft-pbryan-zyp-json-ref-03>`__
in accordance with the OpenAPI spec.

Mostly the latter involves handling non-URI references; OpenAPI is fine
with providing relative file paths, whereas JSON references require URIs
at this point in time.

Usage
=====

Installation
------------

Prance is available from PyPI, and can be installed via pip:

.. code:: bash

    $ pip install prance

Note that this will install the code, but additional subpackages must be specified
to unlock various pieces of functionality. At minimum, a parsing backend must be
installed. For the CLI functionality, you need further dependencies.

The recommended installation installs the CLI, uses ICU and installs one validation
backend:

.. code:: bash

    $ pip install prance[osv,icu,cli]

Make sure you have `ICU Unicode Library <http://site.icu-project.org/home>`__ installed,
as well as Python dev library before running the commands above. If not, use the
following commands:

.. code:: bash

    $ sudo apt-get install libicu-dev python3-dev # Ubuntu/Debian
    $ sudo dnf install libicu-devel python3-devel # Fedora


Command Line Interface
----------------------

After installing prance, a CLI is available for validating (and resolving
external references in) specs:

.. code:: bash

    # Validates with resolving
    $ prance validate path/to/swagger.yml

    # Validates without resolving
    $ prance validate --no-resolve path/to/swagger.yml

    # Fetch URL, validate and resolve.
    $ prance validate http://petstore.swagger.io/v2/swagger.json
    Processing "http://petstore.swagger.io/v2/swagger.json"...
     -> Resolving external references.
    Validates OK as Swagger/OpenAPI 2.0!

Validation is not the only feature of prance. One of the side effects of
resolving is that from a spec with references, one can create a fully resolved
output spec. In the past, this was done via options to the ``validate`` command,
but now there's a specific command just for this purpose:

.. code:: bash

    # Compile spec
    $ prance compile path/to/input.yml path/to/output.yml


Lastly, with the arrival of OpenAPI 3.0.0, it becomes useful for tooling to
convert older specs to the new standard. Instead of re-inventing the wheel,
prance just provides a CLI command for passing specs to the web API of
`swagger2openapi <https://github.com/Mermade/swagger2openapi>`__ - a working
internet connection is therefore required for this command:

.. code:: bash

    # Convert spec
    $ prance convert path/to/swagger.yml path/to/openapi.yml


Code
----

Most likely you have spec file and want to parse it:

.. code:: python

    from prance import ResolvingParser
    parser = ResolvingParser('path/to/my/swagger.yaml')
    parser.specification  # contains fully resolved specs as a dict

Prance also includes a non-resolving parser that does not follow JSON
references, in case you prefer that.

.. code:: python

    from prance import BaseParser
    parser = BaseParser('path/to/my/swagger.yaml')
    parser.specification  # contains specs as a dict still containing JSON references

On Windows, the code reacts correctly if you pass posix-like paths
(``/c:/swagger``) or if the path is relative.  If you pass absolute
windows path (like ``c:\swagger.yaml``), you can use
``prance.util.fs.abspath`` to convert them.

URLs can also be parsed:

.. code:: python

    parser = ResolvingParser('http://petstore.swagger.io/v2/swagger.json')

Largely, that's it. There is a whole slew of utility code that you may
or may not find useful, too. Look at the `full documentation
<https://prance.readthedocs.io/en/latest/#api-modules>`__ for details.


Compatibility
-------------

*Python Versions*

Version 0.16.2 is the last version supporting Python 2. It was released on
Nov 12th, 2019. Python 2 reaches end of life at the end of 2019. If you wish
for updates to the Python 2 supported packages, please contact the maintainer
directly.

Until fairly recently, we also tested with `PyPy <https://www.pypy.org/>`__.
Unfortunately, Travis isn't very good at supporting this. So in the absence
of spare time, they're disabled. `Issue 50 <https://github.com/jfinkhaeuser/prance/issues/50>`__
tracks progress on that.

Similarly, but less critically, Python 3.4 is no longer receiving a lot of
love from CI vendors, so automated builds on that version are no longer
supported.

*Backends*

Different validation backends support different features.

+------------------------+----------------+-----------------+-------------+-------------------------------------------------------+----------------+-----------------------------------------------------------------------------------+
| Backend                | Python Version | OpenAPI Version | Strict Mode | Notes                                                 | Available From | Link                                                                              |
+========================+================+=================+=============+=======================================================+================+===================================================================================+
| swagger-spec-validator | 2 and 3        | 2.0 only        | yes         | Slow; does not accept integer keys (see strict mode). | prance 0.1     | `swagger\_spec\_validator <https://github.com/Yelp/swagger_spec_validator>`__     |
+------------------------+----------------+-----------------+-------------+-------------------------------------------------------+----------------+-----------------------------------------------------------------------------------+
| flex                   | 2 and 3        | 2.0 only        | n/a         | Fastest; unfortunately deprecated.                    | prance 0.8     | `flex <https://github.com/pipermerriam/flex>`__                                   |
+------------------------+----------------+-----------------+-------------+-------------------------------------------------------+----------------+-----------------------------------------------------------------------------------+
| openapi-spec-validator | 2 and 3        | 2.0 and 3.0     | yes         | Slow; does not accept integer keys (see strict mode). | prance 0.11    | `openapi\_spec\_validator <https://github.com/p1c2u/openapi-spec-validator>`__    |
+------------------------+----------------+-----------------+-------------+-------------------------------------------------------+----------------+-----------------------------------------------------------------------------------+

You can select the backend in the constructor of the parser(s):

.. code:: python

    parser = ResolvingParser('http://petstore.swagger.io/v2/swagger.json', backend = 'openapi-spec-validator')


No backend is included in the dependencies; they are detected at run-time. If you install them,
they can be used:

.. code:: bash

    $ pip install openapi-spec-validator
    $ pip install prance
    $ prance validate --backend=openapi-spec-validator path/to/spec.yml

*A note on flex usage:* While flex is the fastest validation backend, unfortunately it is no longer
maintained and there are issues with its dependencies. For one thing, it depends on a version of `PyYAML`
that contains security flaws. For another, it depends explicitly on older versions of `click`.

If you use the flex subpackage, therefore, you do so at your own risk.

*Compatibility*

See `COMPATIBILITY.rst <https://github.com/jfinkhaeuser/prance/blob/master/COMPATIBILITY.rst>`__
for a list of known issues.


Partial Reference Resolution
----------------------------

It's possible to instruct the parser to only resolve some kinds of references.
This allows e.g. resolving references from external URLs, whilst keeping local
references (i.e. to local files, or file internal) intact.

.. code:: python

    from prance import ResolvingParser
    from prance.util.resolver import RESOLVE_HTTP

    parser = ResolvingParser('/path/to/spec', resolve_types = RESOLVE_HTTP)


Multiple types can be specified by OR-ing constants together:

.. code:: python

    from prance import ResolvingParser
    from prance.util.resolver import RESOLVE_HTTP, RESOLVE_FILES

    parser = ResolvingParser('/path/to/spec', resolve_types = RESOLVE_HTTP | RESOLVE_FILES)


Extensions
----------

Prance includes the ability to reference outside swagger definitions
in outside Python packages. Such a package must already be importable
(i.e. installed), and be accessible via the
`ResourceManager API <https://setuptools.readthedocs.io/en/latest/pkg_resources.html#resourcemanager-api>`__
(some more info `here <https://setuptools.readthedocs.io/en/latest/setuptools.html#including-data-files>`__).

For example, you might create a package ``common_swag`` with the file
``base.yaml`` containing the definition

.. code:: yaml

    definitions:
      Severity:
        type: string
        enum:
        - INFO
        - WARN
        - ERROR
        - FATAL

In the ``setup.py`` for ``common_swag`` you would add lines such as

.. code:: python

    packages=find_packages('src'),
    package_dir={'': 'src'},
    package_data={
        '': '*.yaml'
    }

Then, having installed ``common_swag`` into some application, you could
now write

.. code:: yaml

    definitions:
      Message:
        type: object
        properties:
          severity:
            $ref: 'python://common_swag/base.yaml#/definitions/Severity'
          code:
            type: string
          summary:
            type: string
          description:
            type: string
        required:
        - severity
        - summary

Contributing
============

See `CONTRIBUTING.md <https://github.com/jfinkhaeuser/prance/blob/master/CONTRIBUTING.md>`__ for details.

Professional support is available through `finkhaeuser consulting <https://finkhaeuser.de>`__.

License
=======

Licensed under MIT. See the `LICENSE.txt <https://github.com/RonnyPfannschmidt/prance/blob/master/LICENSE.txt>`__ file for details.

"Prancing unicorn" logo image Copyright (c) Jens Finkhaeuser.
Made by `Moreven B <http://morevenb.com/>`__. Use of the logo is permitted under
the `Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International license <https://creativecommons.org/licenses/by-nc-sa/4.0/>`__.


.. |License| image:: https://img.shields.io/pypi/l/prance.svg
   :target: https://pypi.python.org/pypi/prance/
.. |PyPI| image:: https://img.shields.io/pypi/v/prance.svg
   :target: https://pypi.python.org/pypi/prance/
.. |Package Format| image:: https://img.shields.io/pypi/format/prance.svg
   :target: https://pypi.python.org/pypi/prance/
.. |Python Versions| image:: https://img.shields.io/pypi/pyversions/prance.svg
   :target: https://pypi.python.org/pypi/prance/
.. |Package Status| image:: https://img.shields.io/pypi/status/prance.svg
   :target: https://pypi.python.org/pypi/prance/
.. |Logo| image:: https://raw.githubusercontent.com/RonnyPfannschmidt/prance/master/docs/images/prance_logo_256.png
 readmeEtag: '"3a47ba4b9a16416998163676ba68d5e302e7dd0f"' readmeLastModified: Wed, 21 Jun 2023 20:02:29 GMT repositoryId: 67692955 description: Resolving Swagger/OpenAPI 2.0 and 3.0 Parser created: '2016-09-08T10:23:39Z' updated: '2026-01-27T17:46:10Z' language: Python archived: false stars: 260 watchers: 7 forks: 46 owner: RonnyPfannschmidt logo: https://avatars.githubusercontent.com/u/156838?v=4 license: NOASSERTION repoEtag: '"28947c59867ef85d1626f352aa772668972d6b35fe061493b52c5cfd50c95824"' repoLastModified: Tue, 27 Jan 2026 17:46:10 GMT foundInMaster: true id: 06b2469a97224b07993ee5f7d921ea36 - source: openapi3 tags repository: https://github.com/cornutum/tcases v3: true repositoryMetadata: base64Readme: >- IyBUY2FzZXM6IEEgTW9kZWwtQmFzZWQgVGVzdCBDYXNlIEdlbmVyYXRvciAjCgpbIVtNYXZlbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9tYXZlbi00LjEuMS1ncmVlbi5zdmcpXShodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9dGNhc2VzLXNoZWxsKQpbIVtKYXZhZG9jXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2phdmFkb2MtNC4xLjEtZ3JlZW4uc3ZnKV0oaHR0cHM6Ly9qYXZhZG9jLmlvL2RvYy9vcmcuY29ybnV0dW0udGNhc2VzL3RjYXNlcy1zaGVsbCkKCiMjIFdoYXQncyBOZXc/ICMjCiAgKiBUaGUgbGF0ZXN0IHZlcnNpb24gKFtUY2FzZXMgNC4xLjFdKFJlbGVhc2VOb3Rlcy5tZCM0MTEpKSBpcyBub3cgYXZhaWxhYmxlIGF0IHRoZSBNYXZlbiBDZW50cmFsIFJlcG9zaXRvcnkuCiAgICBTZWUgWypIb3cgVG8gRG93bmxvYWQgVGNhc2VzKl0oSG93VG9Eb3dubG9hZC5tZCkgZm9yIGRvd25sb2FkIGluc3RydWN0aW9ucy4KCiAgKiBIYXZpbmcgdHJvdWJsZSB3aXRoIFRjYXNlcz8gQ2hlY2sgb3V0IFt0aGVzZSB0aXBzXSguL1Ryb3VibGVzaG9vdGluZy1GQVFzLm1kKS4KCiAgKiBHb3QgYSBxdWVzdGlvbj8gTmVlZCBzb21lIGd1aWRhbmNlPyBTdGFydCBhIFtkaXNjdXNzaW9uXShodHRwczovL2dpdGh1Yi5jb20vQ29ybnV0dW0vdGNhc2VzL2Rpc2N1c3Npb25zKS4KCiMjIFdoYXQgRG9lcyBJdCBEbz8gIyMKClRjYXNlcyBpcyBhIHRvb2wgZm9yIGRlc2lnbmluZyB0ZXN0cy4gSXQgZG9lc24ndCBtYXR0ZXIgd2hhdCBraW5kIG9mIHN5c3RlbSB5b3UgYXJlIHRlc3RpbmcgLS0gVUksIGNvbW1hbmQgbGluZSwKW1JFU1QtZnVsIEFQSV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKSwgb3IgYmFja2VuZC4gIE5vciBkb2VzIGl0IG1hdHRlcgp3aGF0IGxldmVsIG9mIHRoZSBzeXN0ZW0geW91IGFyZSB0ZXN0aW5nIC0tIHVuaXQsIHN1YnN5c3RlbSwgb3IgZnVsbCBzeXN0ZW0uIFlvdSBjYW4gdXNlIFRjYXNlcyB0byBkZXNpZ24geW91ciB0ZXN0cyBpbiBhbnkgb2YKdGhlc2Ugc2l0dWF0aW9ucy4gV2l0aCBUY2FzZXMsIHlvdSBkZWZpbmUgdGhlIGlucHV0IHNwYWNlIGZvciB5b3VyIHN5c3RlbS11bmRlci10ZXN0IGFuZCB0aGUgbGV2ZWwgb2YgY292ZXJhZ2UgdGhhdCB5b3UKd2FudC4gVGhlbiBUY2FzZXMgZ2VuZXJhdGVzIGEgbWluaW1hbCBzZXQgb2YgdGVzdCBjYXNlcyB0aGF0IG1lZXRzIHlvdXIgcmVxdWlyZW1lbnRzLgoKVGNhc2VzIGlzIHByaW1hcmlseSBhIHRvb2wgZm9yIGJsYWNrLWJveCB0ZXN0IGRlc2lnbi4gRm9yIHN1Y2ggdGVzdHMsIHRoZSBjb25jZXB0IG9mICJjb3ZlcmFnZSIgaXMgZGlmZmVyZW50IGZyb20gc3RydWN0dXJhbAp0ZXN0aW5nIGNyaXRlcmlhIHN1Y2ggYXMgbGluZSBjb3ZlcmFnZSwgYnJhbmNoIGNvdmVyYWdlLCBldGMuIEluc3RlYWQsIFRjYXNlcyBpcyBndWlkZWQgYnkgY292ZXJhZ2Ugb2YgdGhlIGlucHV0IHNwYWNlIG9mIHlvdXIKc3lzdGVtLgoKVGNhc2VzIGdpdmVzIHlvdSBhIHdheSB0byBkZWZpbmUgdGhlIGlucHV0IHNwYWNlIGZvciB5b3VyIHN5c3RlbSBpbiBhIGZvcm0gdGhhdCBpcyBjb25jaXNlIGJ1dCBjb21wcmVoZW5zaXZlLiBUaGVuIFRjYXNlcyBhbGxvd3MKeW91IHRvIGNvbnRyb2wgdGhlIG51bWJlciBvZiB0ZXN0IGNhc2VzIGluIHlvdXIgc2FtcGxlIHN1YnNldCBieSBzcGVjaWZ5aW5nIHRoZSBsZXZlbCBvZiBjb3ZlcmFnZSB5b3Ugd2FudC4gWW91IGNhbiBzdGFydCB3aXRoIGEKYmFzaWMgbGV2ZWwgb2YgY292ZXJhZ2UsIGFuZCBUY2FzZXMgd2lsbCBnZW5lcmF0ZSBhIHNtYWxsIHNldCBvZiB0ZXN0IGNhc2VzIHRoYXQgdG91Y2hlcyBldmVyeSBzaWduaWZpY2FudCBlbGVtZW50IG9mIHRoZSBpbnB1dApzcGFjZS4gVGhlbiB5b3UgY2FuIGltcHJvdmUgeW91ciB0ZXN0cyBieSBzZWxlY3RpdmVseSBhZGRpbmcgY292ZXJhZ2UgaW4gc3BlY2lmaWMgaGlnaC1yaXNrIGFyZWFzLiBGb3IgZXhhbXBsZSwgeW91IGNhbiBzcGVjaWZ5CnBhaXJ3aXNlIGNvdmVyYWdlIG9yIGhpZ2hlci1vcmRlciBjb21iaW5hdGlvbnMgb2Ygc2VsZWN0ZWQgaW5wdXQgdmFyaWFibGVzLgoKIyMgSG93IERvZXMgSXQgV29yaz8gIyMKCkZpcnN0LCB5b3UgY3JlYXRlIGEgc3lzdGVtIGlucHV0IGRlZmluaXRpb24sIGEgZG9jdW1lbnQgdGhhdCBkZWZpbmVzIHlvdXIgc3lzdGVtIGFzIGEgc2V0IG9mIGZ1bmN0aW9ucy4gRm9yIGVhY2ggc3lzdGVtCmZ1bmN0aW9uLCB0aGUgc3lzdGVtIGlucHV0IGRlZmluaXRpb24gZGVmaW5lcyB0aGUgdmFyaWFibGVzIHRoYXQgY2hhcmFjdGVyaXplIHRoZSBmdW5jdGlvbiBpbnB1dCBzcGFjZS4gSWYgeW91IGFyZSB0ZXN0aW5nIGEgV2ViCnNlcnZpY2UgQVBJLCB5b3UgY2FuIGV2ZW4gW2dlbmVyYXRlIGEgc3lzdGVtIGlucHV0IGRlZmluaXRpb24gYXV0b21hdGljYWxseV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKQpmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbi4KClRoZW4sIHlvdSBjYW4gY3JlYXRlIGEgZ2VuZXJhdG9yIGRlZmluaXRpb24uIFRoYXQncyBhbm90aGVyIGRvY3VtZW50IHRoYXQgZGVmaW5lcyB0aGUgY292ZXJhZ2UgeW91IHdhbnQgZm9yIGVhY2ggc3lzdGVtCmZ1bmN0aW9uLiBUaGUgZ2VuZXJhdG9yIGRlZmluaXRpb24gaXMgb3B0aW9uYWwuIFlvdSBjYW4gc2tpcCB0aGlzIHN0ZXAgYW5kIHN0aWxsIGdldCBhIGJhc2ljIGxldmVsIG9mIGNvdmVyYWdlLgoKRmluYWxseSwgeW91IHJ1biBUY2FzZXMuIFRjYXNlcyBpcyBhIEphdmEgcHJvZ3JhbSB0aGF0IHlvdSBjYW4gcnVuIGZyb20gdGhlIGNvbW1hbmQgbGluZSBvciB1c2luZyB0aGUKW1RjYXNlcyBNYXZlbiBQbHVnaW5dKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL3RjYXNlcy1tYXZlbi1wbHVnaW4vKS4gVGhlIGNvbW1hbmQgbGluZSB2ZXJzaW9uIG9mIFRjYXNlcyBjb21lcyB3aXRoIGJ1aWx0LWluCnN1cHBvcnQgZm9yIHJ1bm5pbmcgdXNpbmcgYSBzaGVsbCBzY3JpcHQgb3IgYW4gYW50IHRhcmdldC4gVXNpbmcgeW91ciBpbnB1dCBkZWZpbml0aW9uIGFuZCB5b3VyIGdlbmVyYXRvciBkZWZpbml0aW9uLCBUY2FzZXMKZ2VuZXJhdGVzIGEgc3lzdGVtIHRlc3QgZGVmaW5pdGlvbi4gVGhlIHN5c3RlbSB0ZXN0IGRlZmluaXRpb24gaXMgYSBkb2N1bWVudCB0aGF0IGxpc3RzLCBmb3IgZWFjaCBzeXN0ZW0gZnVuY3Rpb24sIGEgc2V0IG9mIHRlc3QKY2FzZXMgdGhhdCBwcm92aWRlcyB0aGUgc3BlY2lmaWVkIGxldmVsIG9mIGNvdmVyYWdlLiBFYWNoIHRlc3QgY2FzZSBkZWZpbmVzIGEgc3BlY2lmaWMgdmFsdWUgZm9yIGV2ZXJ5IGZ1bmN0aW9uIGlucHV0CnZhcmlhYmxlLiBUY2FzZXMgZ2VuZXJhdGVzIG5vdCBvbmx5IHZhbGlkIGlucHV0IHZhbHVlcyB0aGF0IGRlZmluZSBzdWNjZXNzZnVsIHRlc3QgY2FzZXMgYnV0IGFsc28gaW52YWxpZCB2YWx1ZXMgZm9yIHRoZSB0ZXN0cwpjYXNlcyB0aGF0IGFyZSBuZWVkZWQgdG8gdmVyaWZ5IGV4cGVjdGVkIGVycm9yIGhhbmRsaW5nLgoKT2YgY291cnNlLCB0aGUgc3lzdGVtIHRlc3QgZGVmaW5pdGlvbiBpcyBub3Qgc29tZXRoaW5nIHlvdSBjYW4gZXhlY3V0ZSBkaXJlY3RseS4gKFVubGVzcyBpdCB3YXMKW2Rlcml2ZWQgYXV0b21hdGljYWxseSBmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbl0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI2hvdy1kby15b3UtcnVuLWdlbmVyYXRlZC1hcGktdGVzdC1jYXNlcykhKQpCdXQgaXQgZm9sbG93cyBhIHdlbGwtZGVmaW5lZCBzY2hlbWEsIHdoaWNoIG1lYW5zIHlvdSBjYW4gdXNlIGEgdmFyaWV0eSBvZiB0cmFuc2Zvcm1hdGlvbiB0b29scyB0byBjb252ZXJ0IGl0IGludG8gYSBmb3JtIHRoYXQKaXMgc3VpdGFibGUgZm9yIHRlc3RpbmcgeW91ciBzeXN0ZW0uIEZvciBleGFtcGxlLCBUY2FzZXMgY29tZXMgd2l0aCBhIGJ1aWx0LWluIHRyYW5zZm9ybWVyIHRoYXQgY29udmVydHMgYSBzeXN0ZW0gdGVzdApkZWZpbml0aW9uIGludG8gYSBKYXZhIHNvdXJjZSBjb2RlIHRlbXBsYXRlIGZvciBhIEpVbml0IG9yIFRlc3RORyB0ZXN0IGNsYXNzLgoKIyMgR2V0IFN0YXJ0ZWQhICMjCgogICogKipUaGUgTG93ZG93bioqCiAgICAqIFtUY2FzZXM6IFRoZSBDb21wbGV0ZSBHdWlkZV0oLi9UY2FzZXMtR3VpZGUubWQjdGNhc2VzLXRoZS1jb21wbGV0ZS1ndWlkZSkKICAgICogW1RjYXNlcyBmb3IgT3BlbkFQSV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKTogVGVzdGluZyBhIFJFU1QtZnVsIEFQST8gR2VuZXJhdGUgdGVzdCBjYXNlcyBkaXJlY3RseSBmcm9tIHlvdXIgT3BlbkFQSSB2MyBkZWZpbml0aW9uLgogICAgKiBbVGhlIFRjYXNlcyBNYXZlbiBQbHVnaW5dKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL3RjYXNlcy1tYXZlbi1wbHVnaW4vKQoKICAqICoqSGVscGZ1bCBHdWlkZXMqKgogICAgKiBbSG93IFRvIERvd25sb2FkIFVzaW5nIE1hdmVuXShIb3dUb0Rvd25sb2FkLm1kKQogICAgKiBbSG93IFRvIFNldHVwIGEgVGNhc2VzIFdlYiBTZXJ2aWNlXSguL1RjYXNlcy1XZWItU2VydmljZS5tZCkKICAgICogW1VzaW5nIFRoZSBUY2FzZXMgQVBJXSguL1VzaW5nLVRjYXNlcy1BUEkubWQpCiAgICAqIFtUcm91Ymxlc2hvb3RpbmcgRkFRXSguL1Ryb3VibGVzaG9vdGluZy1GQVFzLm1kI3Ryb3VibGVzaG9vdGluZy1mYXFzKQogICAgKiBbUmVsZWFzZSBOb3Rlc10oUmVsZWFzZU5vdGVzLm1kKQoKICAqICoqTW9yZSBJbmZvKioKICAgICogW01vZGVsLURyaXZlbiBUZXN0aW5nIFVzaW5nIFRjYXNlc10oTW9kZWxEcml2ZW5UZXN0aW5nRm9yQWdpbGVUZWFtcy5tZCkKICAgICogSmF2YWRvYzogW1RjYXNlcyBBUEldKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL2FwaS9pbmRleC5odG1sKQoKIyMgQ29udHJpYnV0b3JzICMjCgpUaGFua3MgdG8gdGhlIGZvbGxvd2luZyBwZW9wbGUsIHdobyBoYXZlIGNvbnRyaWJ1dGVkIHNpZ25pZmljYW50IGltcHJvdmVtZW50cyB0byBUY2FzZXMuCgogICogW0tlcnJ5IEtpbWJyb3VnaF0oaHR0cHM6Ly9naXRodWIuY29tL2tlcnJ5a2ltYnJvdWdoKSAocHJvamVjdCBmb3VuZGVyKQogICogW0p1Z2xhcl0oaHR0cHM6Ly9naXRodWIuY29tL2p1Z2xhcikKICAqIFtUaGliYXVsdCBLcnVzZV0oaHR0cHM6Ly9naXRodWIuY29tL3RrcnVzZSkK readmeEtag: '"ca9a75dabe602853f00d0e6ee6f9a7326d7f4433"' readmeLastModified: Thu, 27 Nov 2025 22:47:20 GMT repositoryId: 32223373 description: A model-based test case generator created: '2015-03-14T17:22:38Z' updated: '2025-12-18T16:57:03Z' language: Java archived: false stars: 236 watchers: 16 forks: 56 owner: Cornutum logo: https://avatars.githubusercontent.com/u/11477145?v=4 license: MIT repoEtag: '"a1d286eef641716d283e62ff50d46ad41827aaf523f41391a8f106f17a7d330b"' repoLastModified: Thu, 18 Dec 2025 16:57:03 GMT foundInMaster: true category: Testing id: 0843a8b0533f5128a9b0e3cf353e82cd - source: openapi3 tags repository: https://github.com/epiphone/class-validator-jsonschema v3: true repositoryMetadata: base64Readme: >- # class-validator-jsonschema

[![npm version](https://badge.fury.io/js/class-validator-jsonschema.svg)](https://badge.fury.io/js/class-validator-jsonschema)

Convert [class-validator](https://github.com/typestack/class-validator)-decorated classes into OpenAPI-compatible JSON Schema. The aim is to provide a best-effort conversion: since some of the `class-validator` decorators lack a direct JSON Schema counterpart, the conversion is bound to be somewhat opinionated. To account for this multiple extension points are available.

## Installation

`npm install class-validator-jsonschema`

**Note the peer dependency versions** in [package.json](./package.json). Try installing a previous major version of `class-validator-jsonschema` in case you're stuck with older peer dependencies.

## Usage

```typescript
import { IsOptional, IsString, MaxLength } from 'class-validator'
import { validationMetadatasToSchemas } from 'class-validator-jsonschema'

class BlogPost {
  @IsString() id: string

  @IsOptional()
  @MaxLength(20, { each: true })
  tags: string[]
}

const schemas = validationMetadatasToSchemas()
console.log(schemas)
```

which prints out:

```json
{
  "BlogPost": {
    "properties": {
      "id": {
        "type": "string"
      },
      "tags": {
        "items": {
          "maxLength": 20,
          "type": "string"
        },
        "type": "array"
      }
    },
    "required": ["id"],
    "type": "object"
  }
}
```

`validationMetadatasToSchemas` takes an `options` object as an optional parameter. Check available configuration objects and defaults at [`options.ts`](src/options.ts).

### Adding and overriding default converters

With `options.additionalConverters` you can add new validation metadata converters or override [the existing ones](src/defaultConverters.ts). Let's say we want to, for example, add a handy `description` field to each `@IsString()`-decorated property:

```typescript
import { ValidationTypes } from 'class-validator'

// ...

const schemas = validationMetadatasToSchemas({
  additionalConverters: {
    [ValidationTypes.IS_STRING]: {
      description: 'A string value',
      type: 'string',
    },
  },
})
```

which now outputs:

```json
{
  "BlogPost": {
    "properties": {
      "id": {
        "description": "A string value",
        "type": "string"
      }
      // ...
    }
  }
}
```

An additional converter can also be supplied in form of a function that receives the validation metadata item and global options, outputting a JSON Schema property object (see below for usage):

```typescript
type SchemaConverter = (
  meta: ValidationMetadata,
  options: IOptions
) => SchemaObject | void
```

### Custom validation classes

`class-validator` allows you to define [custom validation classes](https://github.com/typestack/class-validator#custom-validation-classes). You might for example validate that a string's length is between given two values:

```typescript
import {
  Validate,
  ValidationArguments,
  ValidatorConstraint,
  ValidatorConstraintInterface,
} from 'class-validator'

// Implementing the validator:

@ValidatorConstraint()
export class CustomTextLength implements ValidatorConstraintInterface {
  validate(text: string, validationArguments: ValidationArguments) {
    const [min, max] = validationArguments.constraints
    return text.length >= min && text.length <= max
  }
}

// ...and putting it to use:

class Post {
  @Validate(CustomTextLength, [0, 11])
  title: string
}
```

Now to handle your custom validator's JSON Schema conversion include a `CustomTextLength` converter in `options.additionalConverters`:

```typescript
const schemas = validationMetadatasToSchemas({
  additionalConverters: {
    CustomTextLength: (meta) => ({
      maxLength: meta.constraints[1],
      minLength: meta.constraints[0],
      type: 'string',
    }),
  },
})
```

### Decorating with additional properties

Validation classes can also be supplemented with the `JSONSchema` decorator. `JSONSchema` can be applied both to classes and individual properties; any given keywords are then [merged](https://lodash.com/docs/4.17.4#merge) into the JSON Schema derived from class-validator decorators:

```typescript
import { JSONSchema } from 'class-validator-jsonschema'

@JSONSchema({
  description: 'A User object',
  example: { id: '123' },
})
class BlogPost {
  @IsString()
  @JSONSchema({
    description: 'User primary key',
    format: 'custom-id',
  })
  id: string
}
```

Results in the following schema:

```json
{
  "BlogPost": {
    "description": "A User object",
    "example": { "id": "123" },
    "properties": {
      "id": {
        "description": "User primary key",
        "format": "custom-id",
        "type": "string"
      }
    },
    "required": ["id"],
    "type": "object"
  }
}
```

`JSONSchema` decorators also flow down from parent classes into [inherited validation decorators](https://github.com/typestack/class-validator#inheriting-validation-decorators). Note though that if the inherited class uses `JSONSchema` to redecorate a property from the parent class, the parent class `JSONSchema` gets overwritten - i.e. there's no merging logic.

#### Custom handlers

Alternatively `JSONSchema` can take a function of type `(existingSchema: SchemaObject, options: IOptions) => SchemaObject`. The return value of this function is then **not** automatically merged into existing schema (i.e. the one derived from `class-validator` decorators). Instead you can handle merging yourself in whichever way is preferred, the idea being that removal of existing keywords and other more complex overwrite scenarios can be implemented here.

### @ValidateNested and arrays

`class-validator` supports validating nested objects via the [`@ValidateNested` decorator](https://github.com/typestack/class-validator#validating-nested-objects). Likewise JSON Schema generation is supported out-of-the-box for nested properties such as

```typescript
@ValidateNested()
user: UserClass
```

However, due to [limitations in Typescript's reflection system](https://github.com/Microsoft/TypeScript/issues/10576) we cannot infer the inner type of a generic class. In effect this means that properties like

```typescript
@ValidateNested({ each: true })
users: UserClass[]

@ValidateNested()
user: Promise<UserClass>
```

would resolve to classes `Array` and `Promise` in JSON Schema. To work around this limitation we can use `@Type` from `class-transformer` to explicitly define the nested property's inner type:

```typescript
import { Type } from 'class-transformer'
import { validationMetadatasToSchemas } from 'class-validator-jsonschema'
const { defaultMetadataStorage } = require('class-transformer/cjs/storage') // See https://github.com/typestack/class-transformer/issues/563 for alternatives

class User {
  @ValidateNested({ each: true })
  @Type(() => BlogPost) // 1) Explicitly define the nested property type
  blogPosts: BlogPost[]
}

const schemas = validationMetadatasToSchemas({
  classTransformerMetadataStorage: defaultMetadataStorage, // 2) Define class-transformer metadata in options
})
```

Note also how the `classTransformerMetadataStorage` option has to be defined for `@Type` decorator to take effect.

### Using a custom validation metadataStorage

Under the hood we grab validation metadata from the default storage returned by `class-validator`'s `getMetadataStorage()`. In case of a version clash or something you might want to manually pass in the storage:

```typescript
const schemas = validationMetadatasToSchemas({
  classValidatorMetadataStorage: myCustomMetadataStorage,
})
```

## Limitations

There's no handling for `class-validator`s **validation groups** or **conditional decorator** (`@ValidateIf`) out-of-the-box. The above-mentioned extension methods can be used to fill the gaps if necessary.

The OpenAPI spec doesn't currently support the new JSON Schema **draft-06 keywords** `const` and `contains`. This means that constant value decorators such as `@IsEqual()` and `@ArrayContains()` translate to quite [complicated schemas](https://github.com/sahava/gtm-datalayer-test/issues/4). Hopefully [in a not too distant future](https://github.com/OAI/OpenAPI-Specification/issues/1313#issuecomment-335893062) these keywords are adopted into the spec and we'll be able to provide neater conversion.

Handling **null values** is also tricky since OpenAPI doesn't support JSON Schema's `type: null`, providing its own `nullable` keyword instead. The default `@IsEmpty()` converter for example opts for `nullable` but you can use `type: null` instead via `options.additionalConverters`:

```typescript
// ...
additionalConverters: {
  [ValidationTypes.IS_EMPTY]: {
    anyOf: [
      {type: 'string', enum: ['']},
      {type: 'null'}
    ]
  }
}
```

## TODO

- [x] handle `skipMissingProperties` and `@isDefined()`
- [x] decorators for overwriting prop schemas
- [ ] optional property descriptions (e.g. `A Base64-encoded string`)
- [ ] optional draft-06 keywords
 readmeEtag: '"d4ad0ae40424022585dbaf50f0f98c496b35f964"' readmeLastModified: Tue, 04 Mar 2025 01:17:50 GMT repositoryId: 112473994 description: Convert class-validator-decorated classes into JSON schema created: '2017-11-29T12:41:25Z' updated: '2026-01-27T23:20:16Z' language: TypeScript archived: false stars: 240 watchers: 5 forks: 37 owner: epiphone logo: https://avatars.githubusercontent.com/u/1923531?v=4 license: MIT repoEtag: '"9781ae75147ea0ea360dc007b8e60e79628661b9bf0ee595a699b30d74e0825d"' repoLastModified: Tue, 27 Jan 2026 23:20:16 GMT foundInMaster: true category: - Data Validators - Parsers id: fc507a0f888124f714fc5c42ee5979b0 - source: openapi3 tags repository: https://github.com/brikev/express-jsdoc-swagger v3: true repositoryMetadata: base64Readme: >- ![npm](https://img.shields.io/npm/v/express-jsdoc-swagger)
![Node.js Package](https://github.com/BRIKEV/express-jsdoc-swagger/workflows/Build/badge.svg)
[![Known Vulnerabilities](https://snyk.io/test/github/BRIKEV/express-jsdoc-swagger/badge.svg)](https://snyk.io/test/github/BRIKEV/express-jsdoc-swagger)
[![Maintainability](https://api.codeclimate.com/v1/badges/6d5565df0c9c10e75b59/maintainability)](https://codeclimate.com/github/BRIKEV/express-jsdoc-swagger/maintainability)
[![Test Coverage](https://api.codeclimate.com/v1/badges/6d5565df0c9c10e75b59/test_coverage)](https://codeclimate.com/github/BRIKEV/express-jsdoc-swagger/test_coverage)
![License: MIT](https://img.shields.io/badge/License-MIT-green.svg)
![npm](https://img.shields.io/npm/dm/express-jsdoc-swagger)

# express-jsdoc-swagger

With this library, you can document your express endpoints using swagger [OpenAPI 3 Specification](https://swagger.io/specification/) without writing YAML or JSON. You can write comments similar to `jsdoc` on each endpoint, and the dependecy is going to create the swagger UI.

## Table of Contents

1. [Prerequisites](#Prerequisites)
2. [Installation](#Installation)
3. [Basic Usage](#Basic-Usage)
4. [Basic Examples](#Basic-Examples)
	- [Advanced examples](https://github.com/BRIKEV/express-jsdoc-swagger/tree/master/examples)
	- [Official docs](https://brikev.github.io/express-jsdoc-swagger-docs/#/)
5. [Validator](#Validator)
6. [VSCode extension](https://marketplace.visualstudio.com/items?itemName=brikev.express-jsdoc-swagger-snippets)

## Prerequisites

This library assumes you are using:

1. [NodeJS](https://nodejs.org)
2. [Express.js](http://www.expressjs.com)

## Installation

```
npm i express-jsdoc-swagger
```

## Basic Usage

```javascript
// index.js file
const express = require('express');
const expressJSDocSwagger = require('express-jsdoc-swagger');

const options = {
  info: {
    version: '1.0.0',
    title: 'Albums store',
    license: {
      name: 'MIT',
    },
  },
  security: {
    BasicAuth: {
      type: 'http',
      scheme: 'basic',
    },
  },
  // Base directory which we use to locate your JSDOC files
  baseDir: __dirname,
  // Glob pattern to find your jsdoc files (multiple patterns can be added in an array)
  filesPattern: './**/*.js',
  // URL where SwaggerUI will be rendered
  swaggerUIPath: '/api-docs',
  // Expose OpenAPI UI
  exposeSwaggerUI: true,
  // Expose Open API JSON Docs documentation in `apiDocsPath` path.
  exposeApiDocs: false,
  // Open API JSON Docs endpoint.
  apiDocsPath: '/v3/api-docs',
  // Set non-required fields as nullable by default
  notRequiredAsNullable: false,
  // You can customize your UI options.
  // you can extend swagger-ui-express config. You can checkout an example of this
  // in the `example/configuration/swaggerOptions.js`
  swaggerUiOptions: {},
  // multiple option in case you want more that one instance
  multiple: true,
};

const app = express();
const PORT = 3000;

expressJSDocSwagger(app)(options);

/**
 * GET /api/v1
 * @summary This is the summary of the endpoint
 * @return {object} 200 - success response
 */
app.get('/api/v1', (req, res) => res.json({
  success: true,
}));

app.listen(PORT, () => console.log(`Example app listening at http://localhost:${PORT}`));
```

## Basic Examples

1. Basic configuration options.

```javascript
const options = {
  info: {
    version: '1.0.0',
    title: 'Albums store',
    license: {
      name: 'MIT',
    },
  },
  security: {
    BasicAuth: {
      type: 'http',
      scheme: 'basic',
    },
  },
  baseDir: __dirname,
  // Glob pattern to find your jsdoc files (multiple patterns can be added in an array)
  filesPattern: './**/*.js',
};
```

2. Components definition

```javascript
/**
 * A song type
 * @typedef {object} Song
 * @property {string} title.required - The title
 * @property {string} artist - The artist
 * @property {number} year - The year - double
 */
```

3. Endpoint which returns a `Songs` model array in the response.

```javascript
/**
 * GET /api/v1/albums
 * @summary This is the summary of the endpoint
 * @tags album
 * @return {array<Song>} 200 - success response - application/json
 */
app.get('/api/v1/albums', (req, res) => (
  res.json([{
    title: 'abum 1',
  }])
));
```

3. Endpoint PUT with body and path params which returns a `Songs` model array in the response.

```javascript
/**
 * PUT /api/v1/albums/{id}
 * @summary Update album
 * @tags album
 * @param {string} name.path - name param description
 * @param {Song} request.body.required - songs info
 * @return {array<Song>} 200 - success response - application/json
 */
app.put('/api/v1/albums/:id', (req, res) => (
  res.json([{
    title: 'abum 1',
  }])
));
```

4. Basic endpoint definition with tags, params and basic authentication

```javascript
/**
 * GET /api/v1/album
 * @summary This is the summary of the endpoint
 * @security BasicAuth
 * @tags album
 * @param {string} name.query.required - name param description
 * @return {object} 200 - success response - application/json
 * @return {object} 400 - Bad request response
 */
app.get('/api/v1/album', (req, res) => (
  res.json({
    title: 'abum 1',
  })
));
```

5. Basic endpoint definition with code example for response body

```javascript
/**
 * GET /api/v1/albums
 * @summary This is the summary of the endpoint
 * @tags album
 * @return {array<Song>} 200 - success response - application/json
 * @example response - 200 - success response example
 * [
 *   {
 *     "title": "Bury the light",
 *     "artist": "Casey Edwards ft. Victor Borba",
 *     "year": 2020
 *   }
 * ]
 */
app.get('/api/v1/albums', (req, res) => (
  res.json([{
    title: 'track 1',
  }])
));
```

You can find more examples [here](https://github.com/BRIKEV/express-jsdoc-swagger/tree/master/examples), or visit our [documentation](https://brikev.github.io/express-jsdoc-swagger-docs/#/).


## Validator

We developed a new package works as a validator of your API endpoints and the documentation you create with this package. This package is [express-oas-validator](https://github.com/BRIKEV/express-oas-validator).

**Example**

Install using the node package registry:

```
npm install --save express-oas-validator
```

We have to wait until we have the full swagger schema to initiate the validator.

```js
// validator.js
const { init } = require('express-oas-validator');

const validators = instance => new Promise((resolve, reject) => {
  instance.on('finish', (swaggerDef) => {
    const { validateRequest, validateResponse } = init(swaggerDef);
    resolve({ validateRequest, validateResponse });
  });

  instance.on('error', (error) => {
    reject(error);
  });
});

module.exports = validators;

```

You can check out this also in our [example folder](https://github.com/BRIKEV/express-jsdoc-swagger/tree/master/examples/validator).

```js
// index.js
const express = require('express');
const expressJSDocSwagger = require('express-jsdoc-swagger');
const validator = require('./validator');

const options = {
  info: {
    version: '1.0.0',
    title: 'Albums store',
    license: {
      name: 'MIT',
    },
  },
  filesPattern: './**.js',
  baseDir: __dirname,
};

const app = express();
const instance = expressJSDocSwagger(app)(options);

const serverApp = async () => {
  const { validateRequest, validateResponse } = await validator(instance);
  app.use(express.urlencoded({ extended: true }));
  app.use(express.json());
  /**
   * A song
   * @typedef {object} Song
   * @property {string} title.required - The title
   * @property {string} artist - The artist
   * @property {integer} year - The year
   */

  /**
   * POST /api/v1/songs
   * @param {Song} request.body.required - song info
   * @return {object} 200 - song response
   */
  app.post('/api/v1/songs', validateRequest(), (req, res) => res.send('You save a song!'));

  /**
   * POST /api/v1/name
   * @param {string} request.body.required - name body description
   * @return {object} 200 - song response
   */
  app.post('/api/v1/name', (req, res, next) => {
    try {
      // Validate response
      validateResponse('Error string', req);
      return res.send('Hello World!');
    } catch (error) {
      return next(error);
    }
  });

  /**
   * GET /api/v1/authors
   * @summary This is the summary or description of the endpoint
   * @param {string} name.query.required - name param description - enum:type1,type2
   * @param {array<string>} license.query - name param description
   * @return {object} 200 - success response - application/json
   */
  app.get('/api/v1/authors', validateRequest({ headers: false }), (req, res) => (
    res.json([{
      title: 'album 1',
    }])
  ));

  // eslint-disable-next-line no-unused-vars
  app.use((err, req, res, next) => {
    res.status(err.status).json(err);
  });

  return app;
};

const PORT = process.env.PORT || 4000;

serverApp()
  .then(app => 
    app.listen(PORT, () =>
      console.log(`Listening PORT: ${PORT}`)
    ))
  .catch((err) => {
    console.error(err);
    process.exit(1);
  });
```

You can visit our [documentation](https://brikev.github.io/express-jsdoc-swagger-docs/#/validator).

## Contributors ✨

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/bri06"><img src="https://avatars0.githubusercontent.com/u/24435223?v=4?s=100" width="100px;" alt="Briam Martinez Escobar"/><br /><sub><b>Briam Martinez Escobar</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=bri06" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://twitter.com/kjmesc"><img src="https://avatars2.githubusercontent.com/u/12685053?v=4?s=100" width="100px;" alt="Kevin Julián Martínez Escobar"/><br /><sub><b>Kevin Julián Martínez Escobar</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=kevinccbsg" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/hoonga"><img src="https://avatars3.githubusercontent.com/u/10708927?v=4?s=100" width="100px;" alt="Heung-yeon Oh"/><br /><sub><b>Heung-yeon Oh</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=hoonga" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/LonelyPrincess"><img src="https://avatars1.githubusercontent.com/u/17673317?v=4?s=100" width="100px;" alt="Sara Hernández"/><br /><sub><b>Sara Hernández</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=LonelyPrincess" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://servatj.me"><img src="https://avatars0.githubusercontent.com/u/3521485?v=4?s=100" width="100px;" alt="Josep Servat"/><br /><sub><b>Josep Servat</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=servatj" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/thuydx55"><img src="https://avatars2.githubusercontent.com/u/1469984?v=4?s=100" width="100px;" alt="Nick Dong"/><br /><sub><b>Nick Dong</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=thuydx55" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Stosiu"><img src="https://avatars1.githubusercontent.com/u/10252063?v=4?s=100" width="100px;" alt="Aleksander Stós"/><br /><sub><b>Aleksander Stós</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=Stosiu" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kdankert"><img src="https://avatars0.githubusercontent.com/u/46489624?v=4?s=100" width="100px;" alt="Kjell Dankert"/><br /><sub><b>Kjell Dankert</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=kdankert" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/juliendu11"><img src="https://avatars0.githubusercontent.com/u/18739442?v=4?s=100" width="100px;" alt="juliendu11"/><br /><sub><b>juliendu11</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=juliendu11" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://me.io"><img src="https://avatars.githubusercontent.com/u/45731?v=4?s=100" width="100px;" alt="Mohamed Meabed"/><br /><sub><b>Mohamed Meabed</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=meabed" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ofarukaydin"><img src="https://avatars.githubusercontent.com/u/32788963?v=4?s=100" width="100px;" alt="Faruk Aydın"/><br /><sub><b>Faruk Aydın</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=ofarukaydin" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/dahlmo"><img src="https://avatars.githubusercontent.com/u/23076026?v=4?s=100" width="100px;" alt="Dahlmo"/><br /><sub><b>Dahlmo</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=dahlmo" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/gandazgul"><img src="https://avatars.githubusercontent.com/u/108850?v=4?s=100" width="100px;" alt="Carlos Ravelo"/><br /><sub><b>Carlos Ravelo</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=gandazgul" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/paulish"><img src="https://avatars.githubusercontent.com/u/1762032?v=4?s=100" width="100px;" alt="Paul Ishenin"/><br /><sub><b>Paul Ishenin</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=paulish" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/sbingner"><img src="https://avatars.githubusercontent.com/u/354533?v=4?s=100" width="100px;" alt="Sam Bingner"/><br /><sub><b>Sam Bingner</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=sbingner" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://stackoverflow.com/users/5059657/alexander-staroselsky"><img src="https://avatars.githubusercontent.com/u/34102969?v=4?s=100" width="100px;" alt="Alexander Staroselsky"/><br /><sub><b>Alexander Staroselsky</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=alexstaroselsky" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://joelabrahamsson.com"><img src="https://avatars.githubusercontent.com/u/218986?v=4?s=100" width="100px;" alt="Joel Abrahamsson"/><br /><sub><b>Joel Abrahamsson</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=joelabrahamsson" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/MakakWasTaken"><img src="https://avatars.githubusercontent.com/u/11789635?v=4?s=100" width="100px;" alt="Markus Moltke"/><br /><sub><b>Markus Moltke</b></sub></a><br /><a href="https://github.com/BRIKEV/express-jsdoc-swagger/commits?author=MakakWasTaken" title="Code">💻</a></td>
    </tr>
  </tbody>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome!
 readmeEtag: '"9434a33d59abea965d65ea63bdf053144581e04c"' readmeLastModified: Mon, 23 Oct 2023 11:36:03 GMT repositoryId: 266211454 description: Swagger OpenAPI 3.x generator created: '2020-05-22T21:31:03Z' updated: '2026-01-22T20:03:25Z' language: JavaScript archived: false stars: 224 watchers: 3 forks: 31 owner: BRIKEV logo: https://avatars.githubusercontent.com/u/59850028?v=4 license: MIT repoEtag: '"8bc9e64b58796bc18a26c3ed23762b29549e12af145f7f60efa759f53333fe46"' repoLastModified: Thu, 22 Jan 2026 20:03:25 GMT foundInMaster: true category: Testing id: af37c15464a7e49379863279ecd6694b - source: openapi3 tags repository: https://github.com/wi2l/fizz v3: true id: 04bf0d13931d22601cceca157cd8cccb repositoryMetadata: base64Readme: >- 
<h1 align="center">Fizz</h1>
<p align="center"><img src="images/lemon.png" height="200px" width="auto" alt="Gin Fizz"></p><p align="center">Fizz is a wrapper for <strong>Gin</strong> based on <strong><a href="https://github.com/loopfz/gadgeto/tree/master/tonic">gadgeto/tonic</a></strong>.</p>
<p align="center">It generates wrapping gin-compatible handlers that do all the repetitive work and wrap the call to your handlers. It can also generates an <i>almost</i> complete <strong>OpenAPI 3</strong> specification of your API.</p>
<p align="center">
   <a href="https://pkg.go.dev/github.com/wI2L/fizz?tab=doc"><img src="https://img.shields.io/static/v1?label=godev&message=reference&color=00add8&logo=go"></a>
   <a href="https://goreportcard.com/report/wI2L/fizz"><img src="https://goreportcard.com/badge/github.com/wI2L/fizz"></a>
   <a href="https://github.com/wI2L/fizz/actions"><img src="https://github.com/wI2L/fizz/workflows/CI/badge.svg"></a>
   <a href="https://codecov.io/gh/wI2L/fizz"><img src="https://codecov.io/gh/wI2L/fizz/branch/master/graph/badge.svg"/></a>
   <a href="https://github.com/wI2L/fizz/releases"><img src="https://img.shields.io/github/v/tag/wI2L/fizz?color=blueviolet&label=version&sort=semver"></a>
   <a href="LICENSE"><img src="https://img.shields.io/badge/license-MIT-blue.svg"></a>
   <a href="https://github.com/avelino/awesome-go"><img src="https://awesome.re/mentioned-badge.svg"></a>
<br>
</p>

---

### Getting started

To create a Fizz instance, you can pass an existing *Gin* engine to `fizz.NewFromEngine`, or use `fizz.New` that will use a new default *Gin* engine.

```go
engine := gin.Default()
engine.Use(...) // register global middlewares

f := fizz.NewFromEngine(engine)
```

A Fizz instance implements the `http.HandlerFunc` interface, which means it can be used as the base handler of your HTTP server.
```go
srv := &http.Server{
   Addr:    ":4242",
   Handler: f,
}
srv.ListenAndServe()
```

### Handlers

Fizz abstracts the `GET`, `POST`, `PUT`, `PATCH`, `DELETE`, `OPTIONS`, `HEAD` and `TRACE` methods of a *Gin* engine. These functions accept a variadic list of handlers as the last parameter, but since Fizz relies on *tonic* to retrieve the informations required to generate the *OpenAPI* specification of the operation, **only one of the handlers registered MUST be wrapped with Tonic**.

In the following example, the `BarHandler` is a simple middleware that will be executed before the `FooHandler`, but the generator will use the input/output type of the `FooHandler` to generate the specification of the operation.

```go
func BarHandler(c *gin.Context) { ... }
func FooHandler(*gin.Context, *Foo) (*Bar, error) { ... }

fizz := fizz.New()
fizz.GET("/foo/bar", nil, BarHandler, tonic.Handler(FooHandler, 200))
```

However, registering only standard handlers that follow the `gin.HandlerFunc` signature is accepted, but the *OpenAPI* generator will ignore the operation and it won't appear in the specification.

### Operation informations

To enrich an operation, you can pass a list of optional `OperationOption` functions as the second parameters of the `GET`, `POST`, `PUT`, `PATCH`, `DELETE`, `OPTIONS` and `HEAD` methods.

```go
// Set the default response description.
// A default status text will be created from the code if it is omitted.
fizz.StatusDescription(desc string)

// Set the summary of the operation.
fizz.Summary(summary string)
fizz.Summaryf(format string, a ...interface{})

// Set the description of the operation.
fizz.Description(desc string)
fizz.Descriptionf(format string, a ...interface{})

// Override the ID of the operation.
// Must be a unique string used to identify the operation among
// all operations described in the API.
fizz.ID(id string)

// Mark the operation as deprecated.
fizz.Deprecated(deprecated bool)

// Add an additional response to the operation.
// The example argument will populate a single example in the response schema.
// For populating multiple examples, use fizz.ResponseWithExamples.
// Notice that example and examples fields are mutually exclusive.
// model, header, and example may be `nil`.
fizz.Response(statusCode, desc string, model interface{}, headers []*ResponseHeader, example interface{})

// ResponseWithExamples is a variant of Response that supports providing multiple examples.
// Examples argument will populate multiple examples in the response schema.
// For populating a single example, use fizz.Response.
// Notice that example and examples fields are mutually exclusive.
// model, header, and examples may be `nil`.
fizz.ResponseWithExamples(statusCode, desc string, model interface{}, headers []*ResponseHeader, examples map[string]interface{})

// Add an additional header to the default response.
// Model can be of any type, and may also be `nil`,
// in which case the string type will be used as default.
fizz.Header(name, desc string, model interface{})

// Override the binding model of the operation.
fizz.InputModel(model interface{})

// Overrides the top-level security requirement of an operation.
// Note that this function can be used more than once to add several requirements.
fizz.Security(security *openapi.SecurityRequirement)

// Add an empty security requirement to this operation to make other security requirements optional.
fizz.WithOptionalSecurity()

// Remove any top-level security requirements for this operation.
fizz.WithoutSecurity()

// Add a Code Sample to the operation.
fizz.XCodeSample(codeSample *XCodeSample)

// Mark the operation as internal. The x-internal flag is interpreted by third-party tools and it only impacts the visual documentation rendering.
fizz.XInternal()
```

**NOTES:**
* `fizz.InputModel` allows to override the operation input regardless of how the handler implementation really binds the request parameters. It is the developer responsibility to ensure that the binding matches the OpenAPI specification.
* The first argument of the `fizz.Reponse` method which represents an HTTP status code is of type *string* because the spec accept the value `default`. See the [Responses Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#responsesObject) documentation for more informations.

To help you declare additional headers, predefined variables for Go primitives types that you can use as the third argument of the `fizz.Header` method are available:
```go
var (
   Integer  int32
   Long     int64
   Float    float32
   Double   float64
   String   string
   Byte     []byte
   Binary   []byte
   Boolean  bool
   DateTime time.Time
)
```

### Groups

Exactly like you would do with *Gin*, you can create a group of routes using the method `Group`. Unlike *Gin* own method, Fizz's one takes two other optional arguments, `name` and `description`. These parameters will be used to create a tag in the **OpenAPI** specification that will be applied to all the routes added to the group.

```go
grp := f.Group("/subpath", "MyGroup", "Group description", middlewares...)
```
If the `name` parameter is empty, the tag won't be created and it won't be used.

Subgroups of subgroups can be created to an infinite depth, according yo your needs.

```go
foo := f.Group("/foo", "Foo", "Foo group")

// all routes registered on group bar will have
// a relative path starting with /foo/bar
bar := f.Group("/bar", "Bar", "Bar group")

// /foo/bar/{barID}
bar.GET("/:barID", nil, tonic.Handler(MyBarHandler, 200))
```

The `Use` method can be used with groups to register middlewares after their creation.
```go
grp.Use(middleware1, middleware2, ...)
```

## Tonic

The subpackage *tonic* handles path/query/header/body parameters binding in a single consolidated input object which allows you to remove all the boilerplate code that retrieves and tests the presence of various parameters. The *OpenAPI* generator make use of the input/output types informations of a tonic-wrapped handler reported by *tonic* to document the operation in the specification.

The handlers wrapped with *tonic* must follow the following signature.
```go
func(*gin.Context, [input object ptr]) ([output object], error)
```
Input and output objects are both optional, as such, the minimal accepted signature is:
```go
func(*gin.Context) error
```

To wrap a handler with *tonic*, use the `tonic.Handler` method. It takes a function that follow the above signature and a default status code and return a `gin.HandlerFunc` function that can be used when you register a route with Fizz of *Gin*.

Output objects can be of any type, and will be marshalled to the desired media type.
Note that the input object **MUST always be a pointer to a struct**, or the tonic wrapping will panic at runtime.

If you use closures as handlers, please note that they will all have the same name, and the generator will return an error. To overcome this problem, you have to explicitely set the ID of an operation when you register the handler.

```go
func MyHandler() gin.HandlerFunc {
   return tonic.Handler(func(c *gin.Context) error {}, 200)
}

fizz.GET("/foo", []fizz.OperationOption{
   fizz.ID("MyOperationID")
}, MyHandler())
```

### Location tags

*tonic* uses three struct tags to recognize the parameters it should bind to the input object of your tonic-wrapped handlers:
- `path`: bind from the request path
- `query`: bind from the request query string
- `header`: bind from the request headers

The fields that doesn't use one of these tags will be considered as part of the request body.

The value of each struct tag represents the name of the field in each location, with options.
```go
type MyHandlerParams struct {
   ID  int64     `path:"id"`
   Foo string    `query:"foo"`
   Bar time.Time `header:"x-foo-bar"`
}
```

*tonic* will automatically convert the value extracted from the location described by the tag to the appropriate type before binding.

**NOTE**: A path parameter is always required and will appear required in the spec regardless of the `validate` tag content.

### Additional tags

You can use additional tags. Some will be interpreted by *tonic*, others will be exclusively used to enrich the *OpenAPI* specification.

| name          | description                                                                                                                                                                                                                                                                           |
| ------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| `default`     | *tonic* will bind this value if none was passed with the request. This should not be used if a field is also required. Read the [documentation](https://swagger.io/docs/specification/describing-parameters/) (section _Common Mistakes_) for more informations about this behaviour. |
| `description` | Add a description of the field in the spec.                                                                                                                                                                                                                                           |
| `deprecated`  | Indicates if the field is deprecated. Accepted values are `1`, `t`, `T`, `TRUE`, `true`, `True`, `0`, `f`, `F`, `FALSE`. Invalid value are considered to be false.                                                                                                                    |
| `enum`        | A coma separated list of acceptable values for the parameter.                                                                                                                                                                                                                         |
| `example`     | An example value to be used in OpenAPI specification. See [section below](#Providing-Examples-for-Custom-Types) for the demonstration on how to provide example for custom types.                                                                                                     |
| `format`      | Override the format of the field in the specification. Read the [documentation](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#dataTypeFormat) for more informations.                                                                                     |
| `validate`    | Field validation rules. Read the [documentation](https://godoc.org/gopkg.in/go-playground/validator.v8) for more informations.                                                                                                                                                        |
| `explode`     | Specifies whether arrays should generate separate parameters for each array item or object property (limited to query parameters with *form* style). Accepted values are `1`, `t`, `T`, `TRUE`, `true`, `True`, `0`, `f`, `F`, `FALSE`. Invalid value are considered to be false.     |

### JSON/XML

The JSON/XML encoders usually omit a field that has the tag `"-"`. This behaviour is reproduced by the *OpenAPI* generator ; a field with this tag won't appear in the properties of the schema.

In the following example, the field `Input` is used only for binding request body parameters and won't appear in the output encoding while `Output` will be marshaled but will not be used for parameters binding.
```go
type Model struct {
	Input  string `json:"-"`
	Output string `json:"output" binding:"-"`
}
```

### Request body

If you want to make a request body field mandatory, you can use the tag `validate:"required"`. The validator used by *tonic* will ensure that the field is present.
To be able to make a difference between a missing value and the zero value of a type, use a pointer.

To explicitly ignore a parameter from the request body, use the tag `binding:"-"`.

Note that the *OpenAPI* generator will ignore request body parameters for the routes with a method that is one of `GET`, `DELETE` or `HEAD`.
   > GET, DELETE and HEAD are no longer allowed to have request body because it does not have defined semantics as per [RFC 7231](https://tools.ietf.org/html/rfc7231#section-4.3).
	[*source*](https://swagger.io/docs/specification/describing-request-body/)

### Schema validation

The *OpenAPI* generator recognize some tags of the [go-playground/validator.v8](https://gopkg.in/go-playground/validator.v8) package and translate those to the [properties of the schema](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.1.md#properties) that are taken from the [JSON Schema definition](http://json-schema.org/latest/json-schema-validation.html#rfc.section.6).

The supported tags are: [len](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Length), [max](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Maximum), [min](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Mininum), [eq](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Equals), [gt](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Greater_Than), [gte](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Greater_Than_or_Equal), [lt](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Less_Than), [lte](https://godoc.org/gopkg.in/go-playground/validator.v8#hdr-Less_Than_or_Equal).

Based on the type of the field that carry the tag, the fields `maximum`, `minimum`, `minLength`, `maxLength`, `minItems`, `maxItems`, `minProperties` and `maxProperties` of its **JSON Schema** will be filled accordingly.

## OpenAPI specification

To serve the generated OpenAPI specification in either `JSON` or `YAML` format, use the handler returned by the `fizz.OpenAPI` method.

To enrich the specification, you can provide additional informations. Head to the [OpenAPI 3 spec](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#infoObject) for more informations about the API informations that you can specify, or take a look at the type `openapi.Info` in the file [`openapi/spec.go`](openapi/spec.go#L25).

```go
f := fizz.New()
infos := &openapi.Info{
   Title:       "Fruits Market",
   Description: `This is a sample Fruits market server.`,
   Version:     "1.0.0",
}
f.GET("/openapi.json", nil, f.OpenAPI(infos, "json"))
```
**NOTE**: The generator will never panic. However, it is strongly recommended to call `fizz.Errors` to retrieve and handle the errors that may have occured during the generation of the specification before starting your API.

#### Servers information

If the OpenAPI specification refers to an API that is not hosted on the same domain, or using a path prefix not included in the spec, you will have to declare server information. This can be achieved using the `f.Generator().SetServers` method.

```go
f := fizz.New()
f.Generator().SetServers([]*openapi.Server{
   {
      Description: "Fruits Market - production",
      URL:         "https://example.org/api/1.0",
   },
})
```

#### Security schemes

If your API requires authentication, you have to declare the security schemes that can be used by the operations. This can be achieved using the `f.Generator().SetSecuritySchemes` method.

```go
f := fizz.New()
f.Generator().SetSecuritySchemes(map[string]*openapi.SecuritySchemeOrRef{
   "apiToken": {
      SecurityScheme: &openapi.SecurityScheme{
         Type: "apiKey",
         In:   "header",
         Name: "x-api-token",
      },
   },
})
```

Once defined, the security schemes will be available for all operations. You can override them on an per-operation basis using the `fizz.Security()` function.

```go
fizz.Security(&openapi.SecurityRequirement{
   "apiToken": []string{},
})
```

#### Components

The output types of your handlers are registered as components within the generated specification. By default, the name used for each component is composed of the package and type name concatenated using _CamelCase_ style, and does not contain the full import path. As such, please ensure that you don't use the same type name in two eponym package in your application.

The names of the components can be customized in two different ways.

##### Global override

Override the name of a type globally before registering your handlers. This has the highest precedence.
```go
f := fizz.New()
f.Generator().OverrideTypeName(reflect.TypeOf(T{}), "OverridedName")
```

##### Interface

Implements the `openapi.Typer` interface on your types.
```go
func (*T) TypeName() string { return "OverridedName" }
```
**WARNING:** You **MUST** not rely on the method receiver to return the name, because the method will be called on a new instance created by the generator with the `reflect` package.

#### Custom schemas

The spec generator creates OpenAPI schemas for your types based on their [reflection kind](https://golang.org/pkg/reflect/#Kind).
If you want to control the output schema of a type manually, you can implement the `DataType` interface for this type.

For example, given a UUID version 4 type, declared as a struct, that should appear as a string with a custom format.
```go
type UUIDv4 struct { ... }

func (*UUIDv4) Format() string { return "uuid" }
func (*UUIDv4) Type() string { return "string" }
```

The schema of the type will look like the following instead of describing all the fields of the struct.
```json
{
   "type": "string",
   "format": "uuid"
}
```

If you want to override the `nullable` property of a type, you can implement the `Nullable` interface for this type.

For example, if [`sql.NullString`](https://pkg.go.dev/database/sql#NullString) is not referenced by a pointer in your model but you still want it to be "nullable":

```go
type NullString sql.NullString

func (NullString) Nullable() bool { return true }
```

**WARNING:** You **MUST** not rely on the method receivers to return the type and format, because these methods will be called on a new instance created by the generator with the `reflect` package.

You can also override manually the type and format using `OverrideDataType()`. This has the highest precedence.
```go
fizz.Generator().OverrideDataType(reflect.TypeOf(&UUIDv4{}), "string", "uuid")
```

##### Native and imported types support

Fizz supports some native and imported types. A schema with a proper type and format will be generated automatically, removing the need for creating your own custom schema.

* [`time.Time`](https://golang.org/pkg/time/#Time)
* [`time.Duration`](https://golang.org/pkg/time/#Duration)
* [`net.URL`](https://golang.org/pkg/net/url/#URL)
* [`net.IP`](https://golang.org/pkg/net/#IP)

Note that, according to the doc, the inherent version of the address is a semantic property, and thus cannot be determined by Fizz. Therefore, the format returned is simply `ip`. If you want to specify the version, you can use the tags `format:"ipv4"` or `format:"ipv6"`.
* [`uuid.UUID`](https://godoc.org/github.com/gofrs/uuid#UUID)

#### Markdown

> Throughout the specification description fields are noted as supporting CommonMark markdown formatting. Where OpenAPI tooling renders rich text it MUST support, at a minimum, markdown syntax as described by CommonMark 0.27. Tooling MAY choose to ignore some CommonMark features to address security concerns.
[*source*](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.1.md#rich-text-formatting)

To help you write markdown descriptions in Go, a simple builder is available in the sub-package `markdown`. This is quite handy to avoid conflicts with backticks that are both used in Go for litteral multi-lines strings and code blocks in markdown.

#### Providing Examples for Custom Types
To be able to provide examples for custom types, they must implement the `json.Marshaler` and/or `yaml.Marshaler` and the following interface:
```go
type Exampler interface {
       ParseExample(v string) (interface{}, error)
}
```

If the custom type implements the interface, Fizz will pass the value from the `example` tag to the `ParseExample` method and use the return value as the example in the OpenAPI specification.

## Known limitations

- Since *OpenAPI* is based on the *JSON Schema* specification itself, objects (Go maps) with keys that are not of type `string` are not supported and will be ignored during the generation of the specification.
- Recursive embedding of the same type is not supported, at any level of recursion. The generator will warn and skip the offending fields.
   ```go
   type A struct {
      Foo int
      *A   // ko, embedded and same type as parent
      A *A // ok, not embedded
      *B   // ok, different type
   }

   type B struct {
      Bar string
      *A // ko, type B is embedded in type A
      *C // ok, type C does not contains an embedded field of type A
   }

   type C struct {
      Baz bool
   }
   ```

## Examples

A simple runnable API is available in `examples/market`.
```shell
go build
./market
# Retrieve the specification marshaled in JSON.
curl -i http://localhost:4242/openapi.json
```

## Credits

Fizz is based on [gin-gonic/gin](https://github.com/gin-gonic/gin) and use [gadgeto/tonic](https://github.com/loopfz/gadgeto/tree/master/tonic). :heart:

<p align="right"><img src="https://forthebadge.com/images/badges/built-with-swag.svg"></p>
 readmeEtag: '"be995e1a7e2f133043ba3ac7be220bf9243c64a7"' readmeLastModified: Wed, 12 Oct 2022 15:12:36 GMT repositoryId: 124098943 description: ':lemon: Gin wrapper with OpenAPI 3 spec generation' created: '2018-03-06T15:32:24Z' updated: '2026-01-28T15:11:44Z' language: Go archived: false stars: 232 watchers: 8 forks: 58 owner: wI2L logo: https://avatars.githubusercontent.com/u/6519569?v=4 license: MIT repoEtag: '"0c567b0394656e697d807a54cc06dd3529059bc7888696aa5ea46d978059e7fe"' repoLastModified: Wed, 28 Jan 2026 15:11:44 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/swagger-api/validator-badge v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIFZhbGlkYXRvciBCYWRnZSA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3N3YWdnZXItYXBpL3N3YWdnZXIuaW8vd29yZHByZXNzL2ltYWdlcy9hc3NldHMvU1ctbG9nby1jbHIucG5nIiBoZWlnaHQ9IjUwIiBhbGlnbj0icmlnaHQiPgoKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2plbmtpbnMvYnVpbGQuc3ZnP2pvYlVybD1odHRwczovL2plbmtpbnMuc3dhZ2dlci5pby9qb2Ivb3NzLXN3YWdnZXItdmFsaWRhdG9yLWJhZGdlLW1hc3RlcildKGh0dHBzOi8vamVua2lucy5zd2FnZ2VyLmlvL3ZpZXcvT1NTJTIwLSUyMEphdmEvam9iL29zcy1zd2FnZ2VyLXZhbGlkYXRvci1iYWRnZS1tYXN0ZXIpCgpUaGlzIHByb2plY3Qgc2hvd3MgYSAidmFsaWQgc3dhZ2dlciIgYmFkZ2Ugb24geW91ciBzaXRlLCBzdXBwb3J0aW5nIFN3YWdnZXIvT3BlbkFQSSAyLjAgYW5kIE9wZW5BUEkgMy54IHNwZWNpZmljYXRpb25zLiAgCgpUaGVyZSBpcyBhbiBvbmxpbmUgdmVyc2lvbiBob3N0ZWQgb24gaHR0cDovL3ZhbGlkYXRvci5zd2FnZ2VyLmlvLiAgCgojIyMgVXNpbmcgRG9ja2VyCgpZb3UgY2FuIGFsc28gcHVsbCBhIGRvY2tlciBpbWFnZSBvZiB0aGUgdmFsaWRhdG9yIGRpcmVjdGx5IGZyb20gW0RvY2tlckh1Yl0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL3N3YWdnZXJhcGkvc3dhZ2dlci12YWxpZGF0b3ItdjIvKSwgZS5nLjoKCmBgYApkb2NrZXIgcHVsbCBzd2FnZ2VyYXBpL3N3YWdnZXItdmFsaWRhdG9yLXYyOnYyLjEuNwpkb2NrZXIgcnVuIC1pdCAtcCA4MDgwOjgwODAgLS1uYW1lIHN3YWdnZXItdmFsaWRhdG9yLXYyIHN3YWdnZXJhcGkvc3dhZ2dlci12YWxpZGF0b3ItdjI6djIuMS43CmBgYAoKU2luY2UgdmVyc2lvbiBgMi4wLjJgIGxvY2FsIGFuZCBub24gaHR0cC9odHRwcyB1cmxzIGFyZSByZWplY3RlZCBieSBkZWZhdWx0LCBhbG9uZyB3aXRoIHJlZGlyZWN0czsgdGhpcyBpcyBjb250cm9sbGFibGUgd2l0aCBkb2NrZXIgZW52IHZhcmlhYmxlcyAvIGphdmEgc3lzdGVtIHByb3BlcnRpZXM6CgpgYGAKZG9ja2VyIHJ1biAtaXQgLXAgODA4MDo4MDgwIC1lICJSRUpFQ1RfTE9DQUw9ZmFsc2UiIC1lICJSRUpFQ1RfUkVESVJFQ1Q9ZmFsc2UiIC0tbmFtZSBzd2FnZ2VyLXZhbGlkYXRvci12MiBzd2FnZ2VyYXBpL3N3YWdnZXItdmFsaWRhdG9yLXYyOnYyLjEuNwpgYGAKCkluIG5vbiBkb2NrZXIgZW52aXJvbm1lbnRzLCBzeXN0ZW0gcHJvcGVydGllcyBgcmVqZWN0TG9jYWxgIGFuZCBgcmVqZWN0UmVkaXJlY3RgIGNhbiBiZSB1c2VkLgoKCgpXZWIgVUkgaXMgcmVhY2hhYmxlIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9pbmRleC5odG1sIGFuZCBPcGVuQVBJIHNwZWMgYXQgaHR0cDovL2xvY2FsaG9zdDo4MDgwL3ZhbGlkYXRvci9vcGVuYXBpLmpzb24KCgoKWW91IGNhbiB2YWxpZGF0ZSBPcGVuQVBJIHNwZWNpZmljYXRpb25zIHZlcnNpb24gMi4wIChTd2FnZ2VyKSwgMy4wIGFuZCAzLjEuIFtTd2FnZ2VyIFBhcnNlcl0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItcGFyc2VyL2Jsb2IvbWFzdGVyL1JFQURNRS5tZCkgaXMgdXNlZCBmb3Igc2VtYW50aWMgdmFsaWRhdGlvbi4KRGVwZW5kaW5nIG9uIGBqc29uU2NoZW1hVmFsaWRhdGlvbmAgcXVlcnkgcGFyYW1ldGVyIHZhbHVlIGFsc28gSlNPTiBTY2hlbWEgdmFsaWRhdGlvbiBjYW4gYmUgZXhlY3V0ZWQgKGRlZmF1bHQgdG8gYHRydWVgKSAKCkFkZGl0aW9uYWwgcGFyYW1ldGVycyBhbGxvdyB0byBjdXN0b21pemUgcGFyc2luZyBhbmQgdmFsaWRhdGlvbiBtb2RlLgoKYGBgCjxpbWcgc3JjPSJodHRwczovL3ZhbGlkYXRvci5zd2FnZ2VyLmlvL3ZhbGlkYXRvcj91cmw9e1lPVVJfVVJMfSI+CmBgYAoKT2YgY291cnNlIHRoZSBgWU9VUl9VUkxgIG5lZWRzIHRvIGJlIGFkZHJlc3NhYmxlIGJ5IHRoZSB2YWxpZGF0b3IgKGkuZS4gd29uJ3QgZmluZCBhbnl0aGluZyBvbiBsb2NhbGhvc3QpLiAgSWYgaXQgdmFsaWRhdGVzLCB5b3UnbGwgZ2V0IGEgbmljZSBncmVlbiBWQUxJRCBsb2dvLiAgRmFpbHVyZXMgd2lsbCBnaXZlIGFuIElOVkFMSUQgbG9nbywgYW5kIGlmIHRoZXJlIGFyZSBlcnJvcnMgcGFyc2luZyB0aGUgc3BlY2lmaWNhdGlvbiBvciByZWFjaGluZyBpdCwgYW4gdWdseSByZWQgRVJST1IgbG9nby4KCkZvciBleGFtcGxlLCB1c2luZyBbaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFzdGVyL2V4YW1wbGVzL3YyLjAvanNvbi9wZXRzdG9yZS1leHBhbmRlZC5qc29uXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYXN0ZXIvZXhhbXBsZXMvdjIuMC9qc29uL3BldHN0b3JlLWV4cGFuZGVkLmpzb24pIGFzIGEgc291cmNlLCB3ZSBnZXQgLi4uCgohW10oaHR0cHM6Ly92YWxpZGF0b3Iuc3dhZ2dlci5pby92YWxpZGF0b3I/dXJsPWh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL21hc3Rlci9leGFtcGxlcy92Mi4wL2pzb24vcGV0c3RvcmUtZXhwYW5kZWQuanNvbikKCklmIHlvdXIgc3BlY2lmaWNhdGlvbiBmYWlscyB0byB2YWxpZGF0ZSBmb3Igc29tZSByZWFzb24sIG9yIGlmIHRoZXJlIGlzIGFuIGVycm9yLCB5b3UgY2FuIGdldCBtb3JlIGluZm9ybWF0aW9uIG9uIHdoeSBieSB2aXNpdGluZyBgYGBodHRwczovL3ZhbGlkYXRvci5zd2FnZ2VyLmlvL3ZhbGlkYXRvci9kZWJ1Zz91cmw9e1lPVVJfVVJMfWBgYC4KClNpbmNlIHRoZSB2YWxpZGF0b3IgdXNlcyBhIGJyb3dzZXJsZXNzIGJhY2stZW5kIHRvIGZldGNoIHRoZSBjb250ZW50cyBhbmQgc2NoZW1hLCBpdCdzIG5vdCBzdWJqZWN0IHRvIHRoZSB0ZXJyaWJsZSB3b3JsZCBvZiBDT1JTLgoKIyMjIFVzaW5nIGNVUkwKCllvdSBjYW4gYWxzbyBwb3N0IGEgc3BlYyB1cCB0byB0aGUgc2VydmljZSB3aXRoIGNVUkw6CgpgYGBiYXNoCmN1cmwgLVggUE9TVCAtZCBAc3dhZ2dlci5qc29uIC1IICdDb250ZW50LVR5cGU6YXBwbGljYXRpb24vanNvbicgaHR0cHM6Ly92YWxpZGF0b3Iuc3dhZ2dlci5pby92YWxpZGF0b3IvZGVidWcKYGBgCgpJbiB0aGlzIGV4YW1wbGUsIGBzd2FnZ2VyLmpzb25gIGlzIHRoZSBzd2FnZ2VyIGRlZmluaXRpb24gaW4gSlNPTiBmb3JtYXQsIGluIHRoZSBDV0QuCgpJZiB5b3VyIHN3YWdnZXIgZGVmaW5pdGlvbiBmaWxlIGlzIGluIFlBTUwgZm9ybWF0LCB0aGUgY29tbWFuZCBuZWVkcyB0byBiZSBhZGFwdGVkIGxpa2Ugc286CgpgYGBiYXNoCmN1cmwgLS1kYXRhLWJpbmFyeSBAc3dhZ2dlci55YW1sIC1IICdDb250ZW50LVR5cGU6YXBwbGljYXRpb24veWFtbCcgaHR0cHM6Ly92YWxpZGF0b3Iuc3dhZ2dlci5pby92YWxpZGF0b3IvZGVidWcKYGBgCgpOb3RlIHRoZSB1c2Ugb2YgYC0tZGF0YS1iaW5hcnlgIHRvIGF2b2lkIHN0cmlwcGluZyBuZXdsaW5lcywgYWxvbmcgd2l0aCBhIGRpZmZlcmVudCBgQ29udGVudC1UeXBlYCBoZWFkZXIuCgojIyMgTm90ZQoKQWxsIG9mIHRoZSBhYm92ZSBpcyBhbHNvIGFwcGxpY2FibGUgdG8gT3BlbkFQSSAzLnggc3BlY2lmaWNhdGlvbnM7IGZvciBleGFtcGxlLCB1c2luZyBbaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uXShodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92My9vcGVuYXBpLmpzb24pIGFzIGEgc291cmNlLCB3ZSBnZXQgLi4uCgohW10oaHR0cHM6Ly92YWxpZGF0b3Iuc3dhZ2dlci5pby92YWxpZGF0b3I/dXJsPWh0dHBzOi8vcGV0c3RvcmUzLnN3YWdnZXIuaW8vYXBpL3YzL29wZW5hcGkuanNvbikKClNpbmNlIHZlcnNpb24gMi4xLjAgYSBgL3BhcnNlQnlVcmxgIGFuZCBgL3BhcnNlQnlDb250ZW50YCBhcmUgYXZhaWxhYmxlLCByZXR1cm5pbmcgYSBzZXJpYWxpemVkIHBhcnNlZCBzcGVjaWZpY2F0aW9uLCB3aXRoIHBhcnNpbmcgYW5kIHJlc3VsdCBjb25maWd1cmFibGUgYnkKcGFyYW1ldGVycywgZS5nLiBwYXNzaW5nIGByZXNvbHZlYCwgZXRjLiBTZWUgW1N3YWdnZXIgUGFyc2VyXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1wYXJzZXIvYmxvYi9tYXN0ZXIvUkVBRE1FLm1kI29wdGlvbnMpLgoKIyMjIFJ1bm5pbmcgbG9jYWxseQoKWW91IGNhbiBidWlsZCBhbmQgcnVuIHRoZSB2YWxpZGF0b3IgbG9jYWxseToKCmBgYGJhc2gKbXZuIHBhY2thZ2UgamV0dHk6cnVuCmBgYAoKQW5kIGFjY2VzcyB0aGUgdmFsaWRhdG9yIGxpa2Ugc3VjaDoKCmBgYApodHRwOi8vbG9jYWxob3N0OjgwODAvdmFsaWRhdG9yP3VybD17VVJMfQpgYGAKCm9yCgpgYGAKaHR0cDovL2xvY2FsaG9zdDo4MDgwL3ZhbGlkYXRvcj91cmw9aHR0cDovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjIvc3dhZ2dlci5qc29uCmBgYApgYGAKaHR0cDovL2xvY2FsaG9zdDo4MDgwL3ZhbGlkYXRvcj91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFzdGVyL2V4YW1wbGVzL3YzLjAvcGV0c3RvcmUueWFtbApgYGAKCiMjIFNlY3VyaXR5IGNvbnRhY3QKClBsZWFzZSBkaXNjbG9zZSBhbnkgc2VjdXJpdHktcmVsYXRlZCBpc3N1ZXMgb3IgdnVsbmVyYWJpbGl0aWVzIGJ5IGVtYWlsaW5nIFtzZWN1cml0eUBzd2FnZ2VyLmlvXShtYWlsdG86c2VjdXJpdHlAc3dhZ2dlci5pbyksIGluc3RlYWQgb2YgdXNpbmcgdGhlIHB1YmxpYyBpc3N1ZSB0cmFja2VyLgo= readmeEtag: '"130564c9011e16f3af40a9478c12cbeee097bbad"' readmeLastModified: Mon, 19 May 2025 07:23:19 GMT repositoryId: 23099185 description: Validate your Swagger JSON/YAML today! created: '2014-08-19T06:18:38Z' updated: '2026-01-29T13:11:20Z' language: Java archived: false stars: 220 watchers: 13 forks: 82 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"0fe8740bfa41fe1aa7cf7d7f5ee0ae52f30b80177d54f5debbeb93455615d663"' repoLastModified: Thu, 29 Jan 2026 13:11:20 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: f4f082f1673ead88d9098ccd59574327 - source: openapi3 tags repository: https://github.com/seriousme/fastify-openapi-glue v3: true repositoryMetadata: base64Readme: >- # Fastify OpenApi Glue
[![CI status](https://github.com/seriousme/fastify-openapi-glue/workflows/Node.js%20CI/badge.svg)](https://github.com/seriousme/fastify-openapi-glue/actions?query=workflow%3A%22Node.js+CI%22)
[![Coverage Status](https://codecov.io/gh/seriousme/fastify-openapi-glue/graph/badge.svg?token=ACSanhkzo5)](https://codecov.io/gh/seriousme/fastify-openapi-glue)
[![NPM version](https://img.shields.io/npm/v/fastify-openapi-glue.svg)](https://www.npmjs.com/package/fastify-openapi-glue)
![npm](https://img.shields.io/npm/dm/fastify-openapi-glue)



A plugin for [fastify](https://fastify.dev) to auto-generate a configuration based on a [OpenApi](https://www.openapis.org/)(v2/v3) specification.

It aims at facilitating ["design first" API development](https://swagger.io/blog/api-design/design-first-or-code-first-api-development/) i.e. you write or obtain an API specification and use that to generate code. Given an OpenApi specification Fastify-openapi-glue handles the fastify configuration of routes and validation schemas etc. You can also [generate](#generator) your own project from a OpenApi specification.

<a name="upgrading"></a>
## Upgrading

If you are upgrading from a previous major version of `fastify-openapi-glue` then please check out [UPGRADING.md](UPGRADING.md).

<a name="install"></a>
## Install 
```
npm i fastify-openapi-glue --save
```
<a name="plugin"></a>
## Plugin
<a name="pluginUsage"></a>
### Usage

Add the plugin to your project with `register` and pass it some basic options and you are done !
```javascript
import openapiGlue from "fastify-openapi-glue";
import { Service } from "./service.js";
import { Security } from "./security.js";

const options = {
  specification: `${currentDir}/petstore-openapi.v3.json`,
  serviceHandlers: new Service(),
  securityHandlers: new Security(),
  prefix: "v1",
};


fastify.register(openapiGlue, options);
```

All schema and routes will be taken from the OpenApi specification listed in the options. No need to specify them in your code. 
<a name="pluginOptions"></a>
### Options
  - `specification`: this can be a JSON object, or the name of a JSON or YAML file containing a valid OpenApi(v2/v3) file 
  - `serviceHandlers`: this can be a javascript object or class instance. See the [serviceHandlers documentation](docs/serviceHandlers.md) for more details.
  - `securityHandlers`: this can be a javascript object or class instance. See the [securityHandlers documentation](docs/securityHandlers.md) for more details.
  - `prefix`: this is a string that can be used to prefix the routes, it is passed verbatim to fastify. E.g. if the path to your operation is specified as "/operation" then a prefix of "v1" will make it available at "/v1/operation". This setting overrules any "basePath" setting in a v2 specification. See the [servers documentation](docs/servers.md) for more details on using prefix with a v3 specification.
 - `operationResolver`: a custom operation resolver function, `(operationId, method, openapiPath) => handler | routeOptions` where method is the uppercase HTTP method (e.g. "GET") and openapiPath is the path taken from the specification without prefix (e.g. "/operation"). Mutually exclusive with `serviceHandlers`. See the [operationResolver documentation](docs/operationResolver.md) for more details.
 - `addEmptySchema`: a boolean that allows empty bodies schemas to be passed through. This might be useful for status codes like 204 or 304. Default is `false`.

`specification` and either `serviceHandlers` or `operationResolver` are mandatory, `securityHandlers` and `prefix` are optional.
See the [examples](#examples) section for a demo.

Please be aware that `this` will refer to your serviceHandlers object or your securityHandler object and not to Fastify as explained in the [bindings documentation](docs/bindings.md)

<a name="pluginApiExtensions"></a>
### OpenAPI extensions
The OpenAPI specification supports [extending an API spec](https://spec.openapis.org/oas/latest.html#specification-extensions) to describe extra functionality that isn't covered by the official specification. Extensions start with `x-` (e.g., `x-myapp-logo`) and can contain a primitive, an array, an object, or `null`.

The following extensions are provided by the plugin:
- `x-fastify-config` (object): any properties will be added to the `routeOptions.config` property of the Fastify route.

  For example, if you wanted to use the fastify-raw-body plugin to compute a checksum of the request body, you could add the following extension to your OpenAPI spec to signal the plugin to specially handle this route:

  ```yaml
  paths:
    /webhooks:
      post:
        operationId: processWebhook
        x-fastify-config:
          rawBody: true
        responses:
          204:
            description: Webhook processed successfully
  ```

- `x-no-fastify-config` (true): this will ignore this specific route as if it was not present in your OpenAPI specification:

```yaml
  paths:
    /webhooks:
      post:
        operationId: processWebhook
        x-no-fastify-config: true
        responses:
          204:
            description: Webhook processed successfully
  ```

You can also set custom OpenAPI extensions (e.g., `x-myapp-foo`) for use within your app's implementation. These properties are passed through unmodified to the Fastify route on `{req,reply}.routeOptions.config`. Extensions specified on a schema are also accessible (e.g., `routeOptions.schema.body` or `routeOptions.schema.responses[<statusCode>]`).

<a name="generator"></a>
## Generator

To make life even more easy there is the `openapi-glue` cli. The `openapi-glue` cli takes a valid OpenApi (v2/v3) file (JSON or YAML) and generates a project including a fastify plugin that you can use on any fastify server, a stub of the serviceHandlers class and a skeleton of a test harness to test the plugin. 

<a name="generatorUsage"></a>
### Usage
```
  openapi-glue [options] <OpenApi specification>
```
or if you don't have `openapi-glue` installed:
```
  npx github:seriousme/fastify-openapi-glue <OpenApi specification>
```
This will generate a project based on the provided OpenApi specification.
Any existing files in the project folder will be overwritten!
See the [generator examples](#examples) section for a demo.
<a name="generatorOptions"></a>
### Options
```

  -p <name>                   The name of the project to generate
  --projectName=<name>        [default: generated-javascript-project]

  -b <dir> --baseDir=<dir>    Directory to generate the project in.
                              This directory must already exist.
                              [default: "."]

The following options are only useful for testing the openapi-glue plugin:
  -c --checksumOnly           Don't generate the project on disk but
                              return checksums only.
  -l --localPlugin            Use a local path to the plugin.
```
See the [generator example](#generatorExamples) section for a demo.


<a name="examples"></a>
## Examples
Clone this repository and run `npm i` 

<a name="pluginExamples"></a>
### Plugin
Executing `npm start` will start fastify on localhost port 3000 with the
routes extracted from the [petstore example](examples/petstore/petstore-openapi.v3.json) and the [accompanying serviceHandlers definition](examples/petstore/service.js)

* http://localhost:3000/v2/pet/24 will return a pet as specified in service.js
* http://localhost:3000/v2/pet/myPet will return a fastify validation error:

```json
{
  "statusCode": 400,
  "error": "Bad Request",
  "message": "params.petId should be integer"
}
```

* http://localhost:3000/v2/pet/findByStatus?status=available&status=pending will return
  the following error:

```json
{
  "statusCode": 500,
  "error": "Internal Server Error",
  "message": "Operation findPetsByStatus not implemented"
}
```

* http://localhost:3000/v2/pet/0 will return the following error:

```json
{
  "statusCode": 500,
  "error": "Internal Server Error",
  "message":"\"name\" is required!"
}
```

as the pet returned by service.js does not match the response schema.

<a name="generatorExamples"></a>
### Generator
The folder [examples/generated-javascript-project](examples/generated-javascript-project) contains the result of running `openapi-glue -l --baseDir=examples examples/petstore/petstore-swagger.v2.yaml`. The generated code can be started using `npm start` in `examples/generated-javascript-project` (you will need to run `npm i` in the generated folder first)
<a name="Notes"></a>
## Notes
- the plugin ignores information in a v3 specification under `server/url` as there could be multiple values here, use the `prefix` [option](#pluginOptions) if you need to prefix your routes. See the [servers documentation](docs/servers.md) for more details.
- fastify only supports `application/json` and `text/plain` out of the box. The default charset is `utf-8`.  If you need to support different content types, you can use the fastify `addContentTypeParser` API.
- fastify will by default coerce types, e.g when you expect a number a string like `"1"` will also pass validation, this can be reconfigured, see [Validation and Serialization](https://fastify.dev/docs/latest/Reference/Validation-and-Serialization/).
- the plugin aims to follow fastify and does not compensate for features that are possible according to the OpenApi specification but not possible in standard fastify (without plugins). This will keep the plugin lightweight and maintainable. E.g. Fastify does not support cookie validation, while OpenApi v3 does.
- in some cases however, the plugin may be able to provide you with data which could be used to enhance OpenApi support within your own Fastify application. Here is one possible way to perform [cookie validation](docs/cookieValidationHowTo.md)  yourself.
- if you have special needs on querystring handling (e.g. arrays, objects etc) then fastify supports a [custom querystring parser](https://fastify.dev/docs/latest/Reference/Server/#querystringparser). You might need to pass the AJV option `coerceTypes: 'array'` as an option to Fastify.
- the plugin is an ECMAscript Module (aka ESM). If you are using Typescript then make sure that you have read: https://www.typescriptlang.org/docs/handbook/esm-node.html to avoid any confusion.
- If you want to use a specification that consists of multiple files then please check out the page on [subschemas](docs/subSchemas.md) 
- Fastify uses AJV strict mode in validating schemas. If you get an error like `....due to error strict mode: unknown keyword: "..."` then please check out the page on [AJV strict mode](docs/AJVstrictMode.md) 
- Fastify does not support `multipart/form-data` out of the box. If you want to use it then have a look at: [@fastify/multipart](https://github.com/fastify/fastify-multipart).
- Fastify uses AJV with JSON schema draft-07 out of the box. If you want to use JSON schema draft-2020-12 features in your OpenApi 3.1+ schema then have a look at [using JSONschema 2020](docs/schema2020.md) 

<a name="Contributing"></a>
## Contributing
- contributions are always welcome. 
- if you plan on submitting new features then please create an issue first to discuss and avoid disappointments.
- main development is done on the master branch therefore PRs to that branch are preferred.
- please make sure you have run `npm test` before you submit a PR.
<a name="Fastify-swaggergen"></a>
## Fastify-swaggergen
Fastify-openapi-glue is the successor to the now deprecated [fastify-swaggergen](https://github.com/seriousme/fastify-swaggergen) project.
Main difference is that it: 
- aims to support OpenApi and not just Swagger V2 (hence the name change)
- does not include fastify-swagger support anymore. If you need to show the swagger UI you can include it yourself. Removing the swagger UI clears up a number of dependencies.
<a name="license"></a>
# License
Licensed under [MIT](LICENSE.txt)
 readmeEtag: '"790cdc6a21f22b2785b26ddd6cfa0714172cf58e"' readmeLastModified: Fri, 26 Sep 2025 14:38:03 GMT repositoryId: 139273011 description: >- A plugin for the Fastify webserver to autogenerate a Fastify configuration based on a OpenApi(v2/v3) specification. created: '2018-06-30T19:02:58Z' updated: '2026-02-05T16:51:04Z' language: JavaScript archived: false stars: 245 watchers: 5 forks: 35 owner: seriousme logo: https://avatars.githubusercontent.com/u/3322396?v=4 license: MIT repoEtag: '"5409e3950b6e6a2c61fa63228358de84702e12845d0e0d2fbd0d322942d13ff1"' repoLastModified: Thu, 05 Feb 2026 16:51:04 GMT foundInMaster: true category: - Data Validators - Parsers id: c0241df58ca59d894f729b9091502762 - source: openapi3 tags repository: https://github.com/luolingchun/flask-openapi v3: true repositoryMetadata: base64Readme: >- <div align="center">
    <a href="https://luolingchun.github.io/flask-openapi3/" target="_blank">
        <img class="off-glb" src="https://raw.githubusercontent.com/luolingchun/flask-openapi3/master/docs/images/logo-text.svg" 
             width="60%" height="auto" alt="logo">
    </a>
</div>
<p align="center">
    <em>Generate REST API and OpenAPI documentation for your Flask project.</em>
</p>
<p align="center">
    <a href="https://github.com/luolingchun/flask-openapi3/actions/workflows/tests.yml" target="_blank">
        <img class="off-glb" src="https://img.shields.io/github/actions/workflow/status/luolingchun/flask-openapi3/tests.yml?branch=master" alt="test">
    </a>
    <a href="https://pypi.org/project/flask-openapi3/" target="_blank">
        <img class="off-glb" src="https://img.shields.io/pypi/v/flask-openapi3" alt="pypi">
    </a>
    <a href="https://pypistats.org/packages/flask-openapi3" target="_blank">
        <img class="off-glb" src="https://img.shields.io/pypi/dm/flask-openapi3" alt="pypistats">
    </a>
    <a href="https://pypi.org/project/flask-openapi3/" target="_blank">
        <img class="off-glb" src="https://img.shields.io/pypi/pyversions/flask-openapi3" alt="pypi versions">
    </a>
</p>

**Flask OpenAPI3** is a web API framework based on **Flask**. It uses **Pydantic** to verify data and automatic
generation of interaction documentation.

The key features are:

- **Easy to code:** Easy to use and easy to learn

- **Standard document specification:** Based on [OpenAPI Specification](https://spec.openapis.org/oas/v3.1.0)

- **Interactive OpenAPI documentation:** [Swagger](https://github.com/swagger-api/swagger-ui), [Redoc](https://github.com/Redocly/redoc), [RapiDoc](https://github.com/rapi-doc/RapiDoc), [RapiPdf](https://mrin9.github.io/RapiPdf/), [Scalar](https://github.com/scalar/scalar), [Elements](https://github.com/stoplightio/elements)

- **Data validation:** Fast data verification based on [Pydantic](https://github.com/pydantic/pydantic)

## Requirements

Python 3.10+

flask-openapi3 is dependent on the following libraries:

- [Flask](https://github.com/pallets/flask) for the web app.
- [Pydantic](https://github.com/pydantic/pydantic) for the data validation.

## Installation

```bash
pip install -U flask-openapi3[swagger]
```

or

```bash
conda install -c conda-forge flask-openapi3[swagger]
```

<details markdown="block">
<summary>Optional dependencies</summary>

- [python-email-validator](https://github.com/JoshData/python-email-validator) supports email verification.
- [python-dotenv](https://github.com/theskumar/python-dotenv#readme) enables support
  for [Environment Variables From dotenv](https://flask.palletsprojects.com/en/latest/cli/#dotenv) when running `flask`
  commands.
- [pyyaml](https://github.com/yaml/pyyaml) is used to output the OpenAPI document in yaml format.
- [asgiref](https://github.com/django/asgiref) allows views to be defined with `async def` and use `await`.
- [flask-openapi3-plugins](https://github.com/luolingchun/flask-openapi3-plugins) Provide OpenAPI UI for flask-openapi3.

To install these dependencies with flask-openapi3:

```bash
pip install flask-openapi3[yaml]
# or
pip install flask-openapi3[async]
# or
pip install flask-openapi3[dotenv]
# or
pip install flask-openapi3[email]
# or all
pip install flask-openapi3[yaml,async,dotenv,email]
# or manually
pip install pyyaml asgiref python-dotenv email-validator
# OpenAPI UI plugins
pip install -U flask-openapi3[swagger,redoc,rapidoc,rapipdf,scalar,elements]
```

</details>

## A Simple Example

Here's a simple example, further go to the [Example](https://luolingchun.github.io/flask-openapi3/latest/Example/).

```python
from pydantic import BaseModel

from flask_openapi3 import Info, Tag
from flask_openapi3 import OpenAPI

info = Info(title="book API", version="1.0.0")
app = OpenAPI(__name__, info=info)

book_tag = Tag(name="book", description="Some Book")


class BookQuery(BaseModel):
    age: int
    author: str


@app.get("/book", summary="get books", tags=[book_tag])
def get_book(query: BookQuery):
    """
    to get all books
    """
    return {
        "code": 0,
        "message": "ok",
        "data": [
            {"bid": 1, "age": query.age, "author": query.author},
            {"bid": 2, "age": query.age, "author": query.author}
        ]
    }


if __name__ == "__main__":
    app.run(debug=True)
```

<details>
<summary>Class-based API View Example</summary>

```python
from pydantic import BaseModel, Field

from flask_openapi3 import OpenAPI, Tag, Info, APIView


info = Info(title="book API", version="1.0.0")
app = OpenAPI(__name__, info=info)

api_view = APIView(url_prefix="/api/v1", view_tags=[Tag(name="book")])


class BookPath(BaseModel):
    id: int = Field(..., description="book ID")


class BookQuery(BaseModel):
    age: int | None = Field(None, description="Age")


class BookBody(BaseModel):
    age: int | None = Field(..., ge=2, le=4, description="Age")
    author: str = Field(None, min_length=2, max_length=4, description="Author")


@api_view.route("/book")
class BookListAPIView:
    a = 1

    @api_view.doc(summary="get book list")
    def get(self, query: BookQuery):
        print(self.a)
        return query.model_dump_json()

    @api_view.doc(summary="create book")
    def post(self, body: BookBody):
        """description for a created book"""
        return body.model_dump_json()


@api_view.route("/book/<id>")
class BookAPIView:
    @api_view.doc(summary="get book")
    def get(self, path: BookPath):
        print(path)
        return "get"

    @api_view.doc(summary="update book")
    def put(self, path: BookPath):
        print(path)
        return "put"

    @api_view.doc(summary="delete book", deprecated=True)
    def delete(self, path: BookPath):
        print(path)
        return "delete"


app.register_api_view(api_view)

if __name__ == "__main__":
    app.run(debug=True)
```

</details>

## API Document

Run the [simple example](https://github.com/luolingchun/flask-openapi3/blob/master/examples/simple_demo.py), and go to http://127.0.0.1:5000/openapi.

> OpenAPI UI plugins are optional dependencies that require manual installation.
>
> `pip install -U flask-openapi3[swagger,redoc,rapidoc,rapipdf,scalar,elements]`
>
> More optional ui templates goto the document
> about [UI_Templates](https://luolingchun.github.io/flask-openapi3/latest/Usage/UI_Templates/).

![openapi](https://raw.githubusercontent.com/luolingchun/flask-openapi3/master/docs/images/openapi-all.png)
 readmeEtag: '"42f07b2bab518037b7c58535d07874183c378cd8"' readmeLastModified: Sat, 18 Oct 2025 06:06:04 GMT repositoryId: 362315893 description: Generate REST API and OpenAPI documentation for your Flask project. created: '2021-04-28T02:39:57Z' updated: '2026-02-05T01:24:48Z' language: Python archived: false stars: 258 watchers: 6 forks: 45 owner: luolingchun logo: https://avatars.githubusercontent.com/u/22740403?v=4 license: MIT repoEtag: '"385754d861a161499d59c5d314c10e65ea82bc6334c5dbdbd634bf72f04ffbed"' repoLastModified: Thu, 05 Feb 2026 01:24:48 GMT foundInMaster: true category: Server Implementations id: 99911adb8c5a6cb23bdeadbe7854c4f2 oldLocations: - https://github.com/luolingchun/flask-openapi3 - source: openapi3 tags repository: https://github.com/microsoft/openapi.net.odata v3: true repositoryMetadata: base64Readme: >- WyFbbnVnZXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnVnZXQvdi9NaWNyb3NvZnQuT3BlbkFwaS5PRGF0YS5zdmcpXShodHRwczovL3d3dy5udWdldC5vcmcvcGFja2FnZXMvTWljcm9zb2Z0Lk9wZW5BcGkuT0RhdGEvKQoKIyBDb252ZXJ0IE9EYXRhIHRvIE9wZW5BUEkuTkVUCgojIyBJbnRyb2R1Y3Rpb24KClRoZSAqKk1pY3Jvc29mdC5PcGVuQVBJLk9EYXRhLlJlYWRlcioqIGxpYnJhcnkgaGVscHMgcmVwcmVzZW50IGFuIE9EYXRhIHNlcnZpY2UgbWV0YWRhdGEgYXMgYW4gT3BlbkFwaSBkZXNjcmlwdGlvbi4gSXQgY29udmVydHMgW09EYXRhXShodHRwOi8vd3d3Lm9kYXRhLm9yZykgW0NTRExdKGh0dHA6Ly9kb2NzLm9hc2lzLW9wZW4ub3JnL29kYXRhL29kYXRhLWNzZGwteG1sL3Y0LjAxL29kYXRhLWNzZGwteG1sLXY0LjAxLmh0bWwpLCB0aGUgWE1MIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBFbnRpdHkgRGF0YSBNb2RlbCAoRURNKSBkZXNjcmliaW5nIGFuIE9EYXRhIHNlcnZpY2UgaW50byBbT3BlbiBBUEldKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKSBiYXNlZCBvbiBbT3BlbkFQSS5ORVRdKGh0dHA6Ly9ha2EubXMvb3BlbmFwaSkgb2JqZWN0IG1vZGVsLgoKVGhlIGNvbnZlcnNpb24gaXMgYmFzZWQgb24gdGhlIG1hcHBpbmcgZG9jIGZyb20gW09BU0lTIE9EYXRhIE9wZW5BUEkgdjEuMF0oaHR0cHM6Ly93d3cub2FzaXMtb3Blbi5vcmcvY29tbWl0dGVlcy9kb2N1bWVudC5waHA/ZG9jdW1lbnRfaWQ9NjE4NTImd2dfYWJicmV2PW9kYXRhKSBhbmQgdXNlcyB0aGUgZm9sbG93aW5nIDoKCjEuIFtDYXBhYmlsaXRpZXMgdm9jYWJ1bGFyeSBhbm5vdGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vb2FzaXMtdGNzL29kYXRhLXZvY2FidWxhcmllcy9ibG9iL21haW4vdm9jYWJ1bGFyaWVzL09yZy5PRGF0YS5DYXBhYmlsaXRpZXMuVjEueG1sKQoyLiBbQXV0aG9yaXphdGlvbiB2b2NhYnVsYXJ5IGFubm90YXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9vYXNpcy10Y3Mvb2RhdGEtdm9jYWJ1bGFyaWVzL2Jsb2IvbWFpbi92b2NhYnVsYXJpZXMvT3JnLk9EYXRhLkF1dGhvcml6YXRpb24uVjEueG1sKQozLiBbQ29yZSB2b2NhYnVsYXJ5IGFubm90YXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9vYXNpcy10Y3Mvb2RhdGEtdm9jYWJ1bGFyaWVzL2Jsb2IvbWFpbi92b2NhYnVsYXJpZXMvT3JnLk9EYXRhLkNvcmUuVjEueG1sKQo0LiBOYXZpZ2F0aW9uIHByb3BlcnR5IHBhdGgKNS4gRWRtIG9wZXJhdGlvbiBhbmQgb3BlcmF0aW9uIGltcG9ydCBwYXRoCgojIyBPdmVydmlldwoKVGhlIGltYWdlIGJlbG93IGlzIGdlbmVyaWMgb3ZlcnZpZXcgb2YgaG93IHRoaXMgbGlicmFyeSBjYW4gY29udmVydCB0aGUgRURNIG1vZGVsIHRvIGFuIFtPcGVuQVBJLk5FVCBkb2N1bWVudF0oaHR0cHM6Ly9naXRodWIuY29tL01pY3Jvc29mdC9PcGVuQVBJLk5FVC9ibG9iL21haW4vc3JjL01pY3Jvc29mdC5PcGVuQXBpL01vZGVscy9PcGVuQXBpRG9jdW1lbnQuY3MpIG9iamVjdC4KCiFbQ29udmVydCBPRGF0YSBDU0RMIHRvIE9wZW5BUEldKGRvY3MvaW1hZ2VzL29kYXRhLTItb3BlbmFwaS5wbmcgIk1hcCAvLy8gT0RhdGEgQ1NETCAtLT4gT3BlbkFQSS5ORVQiKQoKRm9yIG1vcmUgaW5mb3JtYXRpb24gb24gdGhlIENTREwgYW5kIEVudGl0eSBEYXRhIG1vZGVsLCBwbGVhc2UgcmVmZXIgdG8gW2h0dHA6Ly93d3cub2RhdGEub3JnL2RvY3VtZW50YXRpb25dKGh0dHA6Ly93d3cub2RhdGEub3JnL2RvY3VtZW50YXRpb24pLgpGb3IgbW9yZSBpbmZvcm1hdGlvbiBhYm91dCB0aGUgT3BlbiBBUEkgb2JqZWN0IG9mIG1vZGVsLCBwbGVhc2UgcmVmZXIgdG8gW2h0dHA6Ly9naXRodWIuY29tL21pY3Jvc29mdC9PcGVuQVBJLk5FVF0oaHR0cDovL2dpdGh1Yi5jb20vbWljcm9zb2Z0L09wZW5BUEkuTkVUKQoKIyMgU2FtcGxlIGNvZGUKClRoZSBmb2xsb3dpbmcgc2FtcGxlIGNvZGUgaWxsdXN0cmF0ZXMgdGhlIHVzZSBvZiB0aGUgbGlicmFyeQoKYGBgY3NoYXJwCnB1YmxpYyBzdGF0aWMgdm9pZCBHZW5lcmF0ZU9wZW5BcGlEZXNjcmlwdGlvbigpCnsKICAgIElFZG1Nb2RlbCBtb2RlbCA9IEdldEVkbU1vZGVsKCk7CiAgICBPcGVuQXBpRG9jdW1lbnQgZG9jdW1lbnQgPSBtb2RlbC5Db252ZXJ0VG9PcGVuQXBpKCk7CiAgICB2YXIgb3V0cHV0SlNPTiA9IGRvY3VtZW50LlNlcmlhbGl6ZUFzSnNvbihPcGVuQXBpU3BlY1ZlcnNpb24uT3BlbkFwaTNfMCk7CiAgICB2YXIgb3V0cHV0WUFNTCA9IGRvY3VtZW50LlNlcmlhbGl6ZUFzWWFtbChPcGVuQXBpU3BlY1ZlcnNpb24uT3BlbkFwaTNfMCk7Cn0KCnB1YmxpYyBzdGF0aWMgSUVkbU1vZGVsIEdldEVkbU1vZGVsKCkKewogICAgLy8gbG9hZCBFRE0gbW9kZWwgaGVyZS4uLgp9CmBgYAoKT3Igd2l0aCB0aGUgY29udmVydCBzZXR0aW5nczoKCmBgYGNzaGFycApwdWJsaWMgc3RhdGljIHZvaWQgR2VuZXJhdGVPcGVuQXBpRGVzY3JpcHRpb24oKQp7CiAgICBJRWRtTW9kZWwgbW9kZWwgPSBHZXRFZG1Nb2RlbCgpOwogICAgT3BlbkFwaUNvbnZlcnRTZXR0aW5ncyBzZXR0aW5ncyA9IG5ldyBPcGVuQXBpQ29udmVydFNldHRpbmdzCiAgICB7CiAgICAgICAgLy8gY29uZmlndXJhdGlvbgogICAgfTsKICAgIE9wZW5BcGlEb2N1bWVudCBkb2N1bWVudCA9IG1vZGVsLkNvbnZlcnRUb09wZW5BcGkoc2V0dGluZ3MpOwogICAgdmFyIG91dHB1dEpTT04gPSBkb2N1bWVudC5TZXJpYWxpemVBc0pzb24oT3BlbkFwaVNwZWNWZXJzaW9uLk9wZW5BcGkzXzApOwogICAgdmFyIG91dHB1dFlBTUwgPSBkb2N1bWVudC5TZXJpYWxpemVBc1lhbWwoT3BlbkFwaVNwZWNWZXJzaW9uLk9wZW5BcGkzXzApOwp9CgpwdWJsaWMgc3RhdGljIElFZG1Nb2RlbCBHZXRFZG1Nb2RlbCgpCnsKICAgIC8vIGxvYWQgRURNIG1vZGVsIGhlcmUuLi4KfQpgYGAKClRoZSBgR2V0RWRtTW9kZWwoKWAgbWV0aG9kIGNhbiBsb2FkIGEgbW9kZWwgaW4gMyB3YXlzOgoKMS4gQ3JlYXRlIHRoZSBFZG0gbW9kZWwgZnJvbSBzY3JhdGNoLiBGb3IgZGV0YWlscyByZWZlciBbYnVpbGRpbmcgYSBiYXNpYyBtb2RlbF0oaHR0cDovL29kYXRhLmdpdGh1Yi5pby9vZGF0YS5uZXQvIzAyLTAxLWJ1aWxkLWJhc2ljLW1vZGVsKQoKMi4gTG9hZCB0aGUgRWRtIG1vZGVsIGZyb20gQ1NETCBmaWxlLiBUaGUgZm9sbG93aW5nIHNob3dzIGEgY29kZSBzYW1wbGUgdGhhdCBsb2FkcyBhIG1vZGVsIGZyb20gYSBjc2RsIGZpbGUuCgogICAgYGBgY3NoYXJwCiAgICBwdWJsaWMgc3RhdGljIElFZG1Nb2RlbCBHZXRFZG1Nb2RlbCgpCiAgICB7CiAgICAgICAgc3RyaW5nIGNzZGxGaWxlUGF0aCA9IEAiYzpcY3NkbC54bWwiOwogICAgICAgIHN0cmluZyBjc2RsID0gU3lzdGVtLklPLkZpbGUuUmVhZEFsbFRleHQoY3NkbEZpbGVQYXRoKTsKICAgICAgICBJRWRtTW9kZWwgbW9kZWwgPSBDc2RsUmVhZGVyLlBhcnNlKFhFbGVtZW50LlBhcnNlKGNzZGwpLkNyZWF0ZVJlYWRlcigpKTsKICAgICAgICByZXR1cm4gbW9kZWw7CiAgICB9CiAgICBgYGAKCjMuIENyZWF0ZSB0aGUgRWRtIG1vZGVsIHVzaW5nIFdlYiBBUEkgT0RhdGEgbW9kZWwgYnVpbGRlci4gRm9yIGRldGFpbHMgcmVmZXIgdG8gdGhlIFt3ZWIgYXBpIG1vZGVsIGJ1aWxkZXIgYXJ0aWNsZV0oaHR0cDovL29kYXRhLmdpdGh1Yi5pby9XZWJBcGkvIzAyLTAxLW1vZGVsLWJ1aWxkZXItYWJzdHJhY3QpCgoKIyMgTnVnZXQgcGFja2FnZXMKClRoZSBPcGVuQVBJLk9EYXRhLnJlYWRlciBudWdldCBwYWNrYWdlIGlzIGF0OiBbaHR0cHM6Ly93d3cubnVnZXQub3JnL3BhY2thZ2VzL01pY3Jvc29mdC5PcGVuQXBpLk9EYXRhL10oaHR0cHM6Ly93d3cubnVnZXQub3JnL3BhY2thZ2VzL01pY3Jvc29mdC5PcGVuQXBpLk9EYXRhKQoKLS0tCgojIENvbnRyaWJ1dGluZwoKVGhpcyBwcm9qZWN0IHdlbGNvbWVzIGNvbnRyaWJ1dGlvbnMgYW5kIHN1Z2dlc3Rpb25zLiAgTW9zdCBjb250cmlidXRpb25zIHJlcXVpcmUgeW91IHRvIGFncmVlIHRvIGEKQ29udHJpYnV0b3IgTGljZW5zZSBBZ3JlZW1lbnQgKENMQSkgZGVjbGFyaW5nIHRoYXQgeW91IGhhdmUgdGhlIHJpZ2h0IHRvLCBhbmQgYWN0dWFsbHkgZG8sIGdyYW50IHVzCnRoZSByaWdodHMgdG8gdXNlIHlvdXIgY29udHJpYnV0aW9uLiBGb3IgZGV0YWlscywgdmlzaXQgW2h0dHBzOi8vY2xhLm1pY3Jvc29mdC5jb21dKGh0dHBzOi8vY2xhLm1pY3Jvc29mdC5jb20pLgoKV2hlbiB5b3Ugc3VibWl0IGEgcHVsbCByZXF1ZXN0LCBhIENMQS1ib3Qgd2lsbCBhdXRvbWF0aWNhbGx5IGRldGVybWluZSB3aGV0aGVyIHlvdSBuZWVkIHRvIHByb3ZpZGUKYSBDTEEgYW5kIGRlY29yYXRlIHRoZSBQUiBhcHByb3ByaWF0ZWx5IChlLmcuLCBsYWJlbCwgY29tbWVudCkuIFNpbXBseSBmb2xsb3cgdGhlIGluc3RydWN0aW9ucwpwcm92aWRlZCBieSB0aGUgYm90LiBZb3Ugd2lsbCBvbmx5IG5lZWQgdG8gZG8gdGhpcyBvbmNlIGFjcm9zcyBhbGwgcmVwb3MgdXNpbmcgb3VyIENMQS4KCllvdSBjYW4gYWxzbyBvcGVuIGFuIGlzc3VlIGRpcmVjdGx5IG9uIHRoaXMgcmVwbyB2aWEgdGhpcyBbbGlua10oaHR0cHM6Ly9naXRodWIuY29tL21pY3Jvc29mdC9PcGVuQVBJLk5FVC5PRGF0YS9pc3N1ZXMvbmV3P2Fzc2lnbmVlcz0mbGFiZWxzPSZwcm9qZWN0cz0mdGVtcGxhdGU9YnVnX3JlcG9ydC5tZCkuCgpUaGlzIHByb2plY3QgaGFzIGFkb3B0ZWQgdGhlIFtNaWNyb3NvZnQgT3BlbiBTb3VyY2UgQ29kZSBvZiBDb25kdWN0XShodHRwczovL29wZW5zb3VyY2UubWljcm9zb2Z0LmNvbS9jb2Rlb2Zjb25kdWN0LykuCkZvciBtb3JlIGluZm9ybWF0aW9uIHNlZSB0aGUgW0NvZGUgb2YgQ29uZHVjdCBGQVFdKGh0dHBzOi8vb3BlbnNvdXJjZS5taWNyb3NvZnQuY29tL2NvZGVvZmNvbmR1Y3QvZmFxLykgb3IKY29udGFjdCBbb3BlbmNvZGVAbWljcm9zb2Z0LmNvbV0obWFpbHRvOm9wZW5jb2RlQG1pY3Jvc29mdC5jb20pIHdpdGggYW55IGFkZGl0aW9uYWwgcXVlc3Rpb25zIG9yIGNvbW1lbnRzLgo= readmeEtag: '"b0212307211649d28cf1d7aebe7eca2441190eed"' readmeLastModified: Tue, 24 Dec 2024 16:56:18 GMT repositoryId: 107572738 description: Generates OpenAPI document from OData CSDL created: '2017-10-19T16:39:13Z' updated: '2026-02-04T22:23:18Z' language: C# archived: false stars: 234 watchers: 26 forks: 65 owner: microsoft logo: https://avatars.githubusercontent.com/u/6154722?v=4 license: MIT repoEtag: '"149d66ed0ea96c1ab29d23463578c88641ea8318cf53061fb54f604d2ba9fd5d"' repoLastModified: Wed, 04 Feb 2026 22:23:18 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: a40464e098a746d489cfc2f5261d45e5 - source: openapi3 tags repository: https://github.com/oas-tools/oas-tools v3: true repositoryMetadata: base64Readme: >- IyBPQVMgVE9PTFMKCjxkaXYgYWxpZ249ImNlbnRlciI+CgpbIVtOUE1dKGh0dHBzOi8vbm9kZWkuY28vbnBtL0BvYXMtdG9vbHMvY29yZS5wbmc/Y29tcGFjdD10cnVlKV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vQG9hcy10b29scy9jb3JlKQoKIVtucG1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3YvQG9hcy10b29scy9jb3JlKQohW25vZGUtY3VycmVudF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ub2RlL3YvQG9hcy10b29scy9jb3JlKQohW25wbV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZHcvQG9hcy10b29scy9jb3JlKQpbIVtOb2RlLmpzIENJXShodHRwczovL2dpdGh1Yi5jb20vb2FzLXRvb2xzL29hcy10b29scy9hY3Rpb25zL3dvcmtmbG93cy9ub2RlanMueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vb2FzLXRvb2xzL29hcy10b29scy9hY3Rpb25zL3dvcmtmbG93cy9ub2RlanMueWFtbCkKWyFbQ29udmVudGlvbmFsIENvbW1pdHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvQ29udmVudGlvbmFsJTIwQ29tbWl0cy0xLjAuMC1ncmVlbi5zdmcpXShodHRwczovL2NvbnZlbnRpb25hbGNvbW1pdHMub3JnKQo8YnIvPgoKWyFbS25vd24gVnVsbmVyYWJpbGl0aWVzXShodHRwczovL3NueWsuaW8vdGVzdC9naXRodWIvb2FzLXRvb2xzL29hcy10b29scy9tYWluL2JhZGdlLnN2ZyldKGh0dHBzOi8vc255ay5pby90ZXN0L2dpdGh1Yi9vYXMtdG9vbHMvb2FzLXRvb2xzKQpbIVtDb3ZlcmFnZSBTdGF0dXNdKGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9vYXMtdG9vbHMvb2FzLXRvb2xzL2JhZGdlLnN2Zz9icmFuY2g9bWFpbildKGh0dHBzOi8vY292ZXJhbGxzLmlvL2dpdGh1Yi9vYXMtdG9vbHMvb2FzLXRvb2xzP2JyYW5jaD1tYWluKQo8L2Rpdj4KCj4gKipXYXJuaW5nKioKPgo+IE9BUyBUb29scyBwYWNrYWdlIGhhcyBiZWVuIHJlbmFtZWQgZnJvbSBgb2FzLXRvb2xzYCB0byBgQG9hcy10b29scy9jb3JlYAoKIyMgUXVpY2tzdGFydAoKSW5pdGlhbGl6ZSBhIG5ldyBPQVMgVG9vbHMgcHJvamVjdCBmcm9tIHRoZSBDTEkKCmBgYHNoCj4gbnB4IEBvYXMtdG9vbHMvY2xpIGluaXQKYGBgCgpJdCB3aWxsIHByb21wdCB0aGUgZm9sbG93aW5nIG1lbnUsIGFza2luZyB3aGljaCB0eXBlIG9mIHJlc291cmNlIHdpbGwgYmUgaW5pdGlhbGl6ZWQ6CgpgYGBzaAo/IFNlbGVjdCBhIHJlc291cmNlIHRvIGluaXRpYWxpemUgKFVzZSBhcnJvdyBrZXlzKQo+IFNlcnZlcgogIE1vZHVsZQogIERldmVsb3BtZW50IGVudmlyb25tZW50CiAgT3BlbkFQSSBGaWxlCmBgYApTZWxlY3QgYHNlcnZlcmAsIGFuc3dlciB0aGUgcXVlc3Rpb25zIGFuZCB5b3UnbGwgZ2V0IGEgcnVubmluZyBzZXJ2ZXIgd2l0aCBPQVMgVG9vbHMuCgpNb3JlIGluZm9ybWF0aW9uIGluIG91ciBbd2ViIHBhZ2VdKGh0dHBzOi8vb2FzLXRvb2xzLmdpdGh1Yi5pbykKCiMjIENvbnRyaWJ1dGluZyBndWlkZWxpbmVzCldlIGdyZWF0bHkgYXBwcmVjaWF0ZSBjb21tdW5pdHkgY29udHJpYnV0aW9ucywgdGhhdCdzIHdoeSB3ZSBoYXZlIGNyZWF0ZWQgYSBkb2N1bWVudCBzdGF0aW5nIGNsZWFyIGd1aWRlbGluZXMgZm9yIFtjb250cmlidXRpbmcgdG8gT0FTIFRvb2xzXShodHRwczovL2dpdGh1Yi5jb20vb2FzLXRvb2xzLy5naXRodWIvYmxvYi9tYWluL0NPTlRSSUJVVElORy5tZCkuCldlIGFsc28gZW5jb3VyYWdlIHlvdSB0byBwYXJ0aWNpcGF0ZSBpbiBkaXNjdXNzaW9ucyBhbmQgYWN0aXZhdGUgbm90aWZpY2F0aW9uIGZvciBvdXIgYW5ub3VuY2VtZW50cyBpbiB0aGUgb3JnYW5pemF0aW9uJ3MgW2Rpc2N1c3Npb25zIGRhc2hib2FyZF0oaHR0cHM6Ly9naXRodWIuY29tL29yZ3Mvb2FzLXRvb2xzL2Rpc2N1c3Npb25zKQoKIyMgTGljZW5zZQoKQ29weXJpZ2h0IDIwMTgsIFtJU0EgR3JvdXBdKGh0dHA6Ly93d3cuaXNhLnVzLmVzKSwgW1VuaXZlcnNpdHkgb2YgU2V2aWxsYV0oaHR0cDovL3d3dy51cy5lcykKCkZvciB0ZWNobmljYWwgaW5xdWlyeSBwbGVhc2UgY29udGFjdCB0byBbZW5naW5lZXJpbmcgdGVhbV0oLi9leHRyYS90ZWFtLm1kKS4KClshW0lTQSBHcm91cF0oaHR0cDovL3d3dy5pc2EudXMuZXMvMi4wL2Fzc2V0cy9pbWcvdGhlbWUvbG9nbzIucG5nKV0oaHR0cDovL3d3dy5pc2EudXMuZXMpCgpMaWNlbnNlZCB1bmRlciB0aGUgKipBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAqKiAodGhlICJbTGljZW5zZV0oLi9MSUNFTlNFKSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCBhcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCgpVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgo= readmeEtag: '"bf604680bde757c44491ca8515680c64c9e5daf4"' readmeLastModified: Tue, 07 Mar 2023 01:20:33 GMT repositoryId: 120771836 description: >- NodeJS module to manage RESTful APIs defined with OpenAPI 3.0 Specs over express servers. created: '2018-02-08T14:28:39Z' updated: '2026-01-21T21:42:51Z' language: JavaScript archived: false stars: 203 watchers: 11 forks: 60 owner: oas-tools logo: https://avatars.githubusercontent.com/u/97441539?v=4 license: Apache-2.0 repoEtag: '"615596a788fc710d5757b63eed80ffde386e450b837ce47636ae1cb5cb34dace"' repoLastModified: Wed, 21 Jan 2026 21:42:51 GMT foundInMaster: true category: - Parsers - Server id: a30dcc4f00df91ead5cba361635a68d5 oldLocations: - https://github.com/isa-group/oas-tools - source: openapi3 tags name: swagger-node-codegen homepage: https://github.com/fmvilas/swagger-node-codegen language: Node.js source_description: >- Generates a Node.js/express server, but also has a template engine for creating any templates needed. category: - Code Generators - Parsers repository: https://github.com/fmvilas/swagger-node-codegen v3: true repositoryMetadata: base64Readme: >- PGgxPlRISVMgUEFDS0FHRSBJUyBOT1QgTUFJTlRBSU5FRCBBTllNT1JFLiBJRiBZT1UgV0FOVCBUTyBNQUlOVEFJTiBJVCBEUk9QIE1FIEEgTElORSBBVCBmcmFuLm1lbmRlelthdF1oZXkuY29tLjwvaDE+Cgo8cCBhbGlnbj0iY2VudGVyIj48aW1nIHNyYz0ibG9nby5wbmciPjwvcD4KPHAgYWxpZ249ImNlbnRlciI+CiAgPHN0cm9uZz5PcGVuQVBJIE5vZGUuanM8YnI+Q29kZSBHZW5lcmF0b3I8L3N0cm9uZz4KPC9wPgo8YnI+PGJyPgpVc2UgeW91ciBBUEkgT3BlbkFQSSAzLngvU3dhZ2dlciAyIGRlZmluaXRpb24gdG8gZ2VuZXJhdGUgTm9kZS5qcyBFUzctY29tcGxpYW50IGNvZGUgZm9yIHlvdXIgQVBJLgoKVGhlIGdlbmVyYXRlZCBjb2RlIGZlYXR1cmVzOgoKKiBFUzcKKiBFU0xpbnQKKiBZQU1MIGNvbmZpZyBmaWxlCiogRXhwcmVzcwoqIE5vIHRyYW5zcGlsaW5nCgojIyBJbnN0YWxsCgpUbyB1c2UgaXQgZnJvbSB0aGUgQ0xJOgoKYGBgYmFzaApucG0gaW5zdGFsbCAtZyBzd2FnZ2VyLW5vZGUtY29kZWdlbgpgYGAKClRvIHVzZSBpdCBhcyBhIG1vZHVsZSBpbiB5b3VyIHByb2plY3Q6CgpgYGBiYXNoCm5wbSBpbnN0YWxsIC0tc2F2ZSBzd2FnZ2VyLW5vZGUtY29kZWdlbgpgYGAKCiMjIFJlcXVpcmVtZW50cwoKKiBOb2RlLmpzIHY3LjYrCgojIyBVc2FnZQoKIyMjIEZyb20gdGhlIGNvbW1hbmQtbGluZSBpbnRlcmZhY2UgKENMSSkKCmBgYGJhc2gKICBVc2FnZTogc25jIFtvcHRpb25zXSA8c3dhZ2dlckZpbGU+CgoKICBPcHRpb25zOgoKICAgIC1WLCAtLXZlcnNpb24gICAgICAgICAgICAgICAgICBvdXRwdXQgdGhlIHZlcnNpb24gbnVtYmVyCiAgICAtbywgLS1vdXRwdXQgPG91dHB1dERpcj4gICAgICAgZGlyZWN0b3J5IHdoZXJlIHRvIHB1dCB0aGUgZ2VuZXJhdGVkIGZpbGVzIChkZWZhdWx0cyB0byBjdXJyZW50IGRpcmVjdG9yeSkKICAgIC10LCAtLXRlbXBsYXRlcyA8dGVtcGxhdGVEaXI+ICBkaXJlY3Rvcnkgd2hlcmUgdGVtcGxhdGVzIGFyZSBsb2NhdGVkIChkZWZhdWx0cyB0byBpbnRlcm5hbCBub2RlanMgdGVtcGxhdGVzKQogICAgLWgsIC0taGVscCAgICAgICAgICAgICAgICAgICAgIG91dHB1dCB1c2FnZSBpbmZvcm1hdGlvbgpgYGAKCiMjIyMgRXhhbXBsZXMKClRoZSBzaG9ydGVzdCBwb3NzaWJsZSBzeW50YXg6CmBgYGJhc2gKc25jIHN3YWdnZXIueWFtbApgYGAKClNwZWNpZnkgd2hlcmUgdG8gcHV0IHRoZSBnZW5lcmF0ZWQgY29kZToKYGBgYmFzaApzbmMgc3dhZ2dlci55YW1sIC1vIC4vbXktYXBpCmBgYAoKIyMjIEFzIGEgbW9kdWxlIGluIHlvdXIgcHJvamVjdAoKYGBganMKY29uc3QgcGF0aCA9IHJlcXVpcmUoJ3BhdGgnKTsKY29uc3QgY29kZWdlbiA9IHJlcXVpcmUoJ3N3YWdnZXItbm9kZS1jb2RlZ2VuJyk7CmNvbnN0IHN3YWdnZXIgPSByZXF1aXJlKCcuL3N3YWdnZXIuanNvbicpOwoKY29kZWdlbi5nZW5lcmF0ZSh7CiAgc3dhZ2dlciwKICB0YXJnZXRfZGlyOiBwYXRoLnJlc29sdmUoX19kaXJuYW1lLCAnLi9teS1hcGknKQp9KS50aGVuKCgpID0+IHsKICBjb25zb2xlLmxvZygnRG9uZSEnKTsKfSkuY2F0Y2goZXJyID0+IHsKICBjb25zb2xlLmVycm9yKGBTb21ldGhpbmcgd2VudCB3cm9uZzogJHtlcnIubWVzc2FnZX1gKTsKfSk7CmBgYAoKVGhlIGBzd2FnZ2VyYCBwYXJhbWV0ZXIgY2FuIGJlIGVpdGhlciBKU09OIG9yIGEgcGF0aCBwb2ludGluZyB0byBhIEpTT04gb3IgWUFNTCBmaWxlLgoKYGBganMKY29uc3QgcGF0aCA9IHJlcXVpcmUoJ3BhdGgnKTsKY29uc3QgY29kZWdlbiA9IHJlcXVpcmUoJ3N3YWdnZXItbm9kZS1jb2RlZ2VuJyk7Cgpjb2RlZ2VuLmdlbmVyYXRlKHsKICBzd2FnZ2VyOiBwYXRoLnJlc29sdmUoX19kaXJuYW1lLCAnLi9zd2FnZ2VyLnltbCcpLAogIHRhcmdldF9kaXI6IHBhdGgucmVzb2x2ZShfX2Rpcm5hbWUsICcuL215LWFwaScpCn0pLnRoZW4oKCkgPT4gewogIGNvbnNvbGUubG9nKCdEb25lIScpOwp9KS5jYXRjaChlcnIgPT4gewogIGNvbnNvbGUuZXJyb3IoYFNvbWV0aGluZyB3ZW50IHdyb25nOiAke2Vyci5tZXNzYWdlfWApOwp9KTsKYGBgCiMjIyMgVXNpbmcgYXN5bmMvYXdhaXQKClRoZSBmdW5jdGlvbiBgY29kZWdlbi5nZW5lcmF0ZWAgcmV0dXJucyBhIFByb21pc2UsIHNvIGl0IG1lYW5zIHlvdSBjYW4gdXNlIGFzeW5jL2F3YWl0OgoKYGBganMKY29uc3QgcGF0aCA9IHJlcXVpcmUoJ3BhdGgnKTsKY29uc3QgY29kZWdlbiA9IHJlcXVpcmUoJ3N3YWdnZXItbm9kZS1jb2RlZ2VuJyk7Cgp0cnkgewogIGF3YWl0IGNvZGVnZW4uZ2VuZXJhdGUoewogICAgc3dhZ2dlcjogcGF0aC5yZXNvbHZlKF9fZGlybmFtZSwgJy4vc3dhZ2dlci55bWwnKSwKICAgIHRhcmdldF9kaXI6IHBhdGgucmVzb2x2ZShfX2Rpcm5hbWUsICcuL215LWFwaScpCiAgfSk7CiAgY29uc29sZS5sb2coJ0RvbmUhJyk7Cn0gY2F0Y2ggKGVycikgewogIGNvbnNvbGUuZXJyb3IoYFNvbWV0aGluZyB3ZW50IHdyb25nOiAke2Vyci5tZXNzYWdlfWApOwp9CmBgYAoKIyMgQVBJIERvY3VtZW50YXRpb24KCiMjIyBNb2R1bGVzCgo8ZGw+CjxkdD48YSBocmVmPSIjbW9kdWxlX2NvZGVnZW4iPmNvZGVnZW48L2E+PC9kdD4KPGRkPjxwPlRoaXMgbW9kdWxlIGdlbmVyYXRlcyBhIGNvZGUgc2tlbGV0b24gZm9yIGFuIEFQSSB1c2luZyBPcGVuQVBJL1N3YWdnZXIuPC9wPgo8L2RkPgo8ZHQ+PGEgaHJlZj0iI2NvZGVnZW4ubW9kdWxlX2dlbmVyYXRlIj5nZW5lcmF0ZTwvYT4g4oeSIDxjb2RlPlByb21pc2U8L2NvZGU+PC9kdD4KPGRkPjxwPkdlbmVyYXRlcyBhIGNvZGUgc2tlbGV0b24gZm9yIGFuIEFQSSBnaXZlbiBhbiBPcGVuQVBJL1N3YWdnZXIgZmlsZS48L3A+CjwvZGQ+CjwvZGw+Cgo8YSBuYW1lPSJtb2R1bGVfY29kZWdlbiI+PC9hPgoKIyMjIGNvZGVnZW4KVGhpcyBtb2R1bGUgZ2VuZXJhdGVzIGEgY29kZSBza2VsZXRvbiBmb3IgYW4gQVBJIHVzaW5nIE9wZW5BUEkvU3dhZ2dlci4KCjxhIG5hbWU9ImNvZGVnZW4ubW9kdWxlX2dlbmVyYXRlIj48L2E+CgojIyMjIGdlbmVyYXRlIOKHkiA8Y29kZT5Qcm9taXNlPC9jb2RlPgpHZW5lcmF0ZXMgYSBjb2RlIHNrZWxldG9uIGZvciBhbiBBUEkgZ2l2ZW4gYW4gT3BlbkFQSS9Td2FnZ2VyIGZpbGUuCgoKfCBQYXJhbSB8IFR5cGUgfCBEZXNjcmlwdGlvbiB8CnwgLS0tIHwgLS0tIHwgLS0tIHwKfCBjb25maWcgfCA8Y29kZT5PYmplY3Q8L2NvZGU+IHwgQ29uZmlndXJhdGlvbiBvcHRpb25zIHwKfCBjb25maWcuc3dhZ2dlciB8IDxjb2RlPk9iamVjdDwvY29kZT4gXHwgPGNvZGU+U3RyaW5nPC9jb2RlPiB8IE9wZW5BUEkvU3dhZ2dlciBKU09OIG9yIGEgc3RyaW5nIHBvaW50aW5nIHRvIGFuIE9wZW5BUEkvU3dhZ2dlciBmaWxlLiB8CnwgY29uZmlnLnRhcmdldF9kaXIgfCA8Y29kZT5TdHJpbmc8L2NvZGU+IHwgUGF0aCB0byB0aGUgZGlyZWN0b3J5IHdoZXJlIHRoZSBmaWxlcyB3aWxsIGJlIGdlbmVyYXRlZC4gfAp8IGNvbmZpZy50ZW1wbGF0ZXN8IDxjb2RlPlN0cmluZzwvY29kZT4gfCBQYXRoIHRvIHRoZSBkaXJlY3Rvcnkgd2hlcmUgY3VzdG9tIHRlbXBsYXRlcyBhcmUgKG9wdGlvbmFsKS4gfAoKCiMjIFRlbXBsYXRlcwpZb3UgY2FuIGNyZWF0ZSB5b3VyIG93biBbdGVtcGxhdGVzXSguL3RlbXBsYXRlcy9SRUFETUUubWQpLgoKIyMgQXV0aG9ycwoKKiBGcmFuIE3DqW5kZXogKFtAZm12aWxhc10oaHR0cDovL3R3aXR0ZXIuY29tL2ZtdmlsYXMpKQoqIFJpY2hhcmQgS2xvc2UgKFtAcmljaGFyZGtsb3NlXShodHRwOi8vZ2l0aHViLmNvbS9yaWNoYXJka2xvc2UpKQo= readmeEtag: '"7ecaca28f3d8531ae5c8f7b927ff1928d3bd2b55"' readmeLastModified: Tue, 17 Nov 2020 09:06:19 GMT repositoryId: 46659429 description: An OpenAPI 3.x/Swagger 2 code generator for Node.js created: '2015-11-22T12:22:22Z' updated: '2025-02-17T03:00:45Z' language: JavaScript archived: false stars: 199 watchers: 6 forks: 54 owner: fmvilas logo: https://avatars.githubusercontent.com/u/242119?v=4 license: Apache-2.0 repoEtag: '"3013a1e03a15394f0ec2e58bf3fcd3da616723d722deb67a7f46c1f391ccc076"' repoLastModified: Mon, 17 Feb 2025 03:00:45 GMT foundInMaster: true id: d432635d31e91cae35355677ce1e130a - source: openapi3 tags repository: https://github.com/himenon/openapi-typescript-code-generator v3: true repositoryMetadata: base64Readme: >- # @himenon/openapi-typescript-code-generator

[日本語](./docs/ja/README-ja.md)

This library provides TypeScript type definitions and extracted parameters from OpenAPI v3.0.x compliant specifications.
TypeScript AST is used to generate the code, which is accurately converted to TypeScript code.
Since the parameters extracted from OpenAPI can be used freely, it can be used for automatic generation of API Client and Server Side code, load balancer configuration files, etc.

## Playground

- [Playground](https://openapi-typescript-code-generator.netlify.app/)

## Installation

```bash
npm  i   -D @himenon/openapi-typescript-code-generator
# or
pnpm i   -D @himenon/openapi-typescript-code-generator
# or
yarn add -D @himenon/openapi-typescript-code-generator
```

## DEMO

- [Short DEMO](https://github.com/Himenon/openapi-typescript-code-generator-demo-project)
- [DEMO: github/rest-api-client code generate](https://github.com/Himenon/github-rest-api-client/tree/master/source)
  - https://github.com/github/rest-api-description

## Usage

The example shown here can be cloned from the [DEMO repository](https://github.com/Himenon/openapi-typescript-code-generator-demo-project) to see how it works.

### Generate typedef-only code

```ts
import * as fs from "fs";

import { CodeGenerator } from "@himenon/openapi-typescript-code-generator";

const main = () => {
  const codeGenerator = new CodeGenerator("your/openapi/spec.yml");
  const code = codeGenerator.generateTypeDefinition();
  fs.writeFileSync("client.ts", code, { encoding: "utf-8" });
};

main();
```

### Generate code containing the API Client

```ts
import * as fs from "fs";

import { CodeGenerator } from "@himenon/openapi-typescript-code-generator";
import * as Templates from "@himenon/openapi-typescript-code-generator/dist/templates";
import type * as Types from "@himenon/openapi-typescript-code-generator/dist/types";

const main = () => {
  const codeGenerator = new CodeGenerator("your/openapi/spec.yml");

  const apiClientGeneratorTemplate: Types.CodeGenerator.CustomGenerator<Templates.FunctionalApiClient.Option> = {
    generator: Templates.FunctionalApiClient.generator,
    option: {},
  };

  const code = codeGenerator.generateTypeDefinition([
    codeGenerator.getAdditionalTypeDefinitionCustomCodeGenerator(),
    apiClientGeneratorTemplate,
  ]);

  fs.writeFileSync("client.ts", code, { encoding: "utf-8" });
};

main();
```

### The variation of template code

This library provides three types of templates

```ts
import * as Templates from "@himenon/openapi-typescript-code-generator/dist/templates";

Templates.ClassApiClient.generator;
Templates.FunctionalApiClient.generator;
Templates.CurryingFunctionalApiClient.generator;
```

#### `Templates.ClassApiClient.generator`

We provide a class-based API client. Please inject the API client dependency and use it instead of `constructor`.

```ts
export interface RequestArgs {
  httpMethod: HttpMethod;
  url: string;
  headers: ObjectLike | any;
  requestBody?: ObjectLike | any;
  requestBodyEncoding?: Record<string, Encoding>;
  queryParameters?: QueryParameters | undefined;
}

export interface ApiClient<RequestOption> {
  request: <T = SuccessResponses>(requestArgs: RequestArgs, options?: RequestOption) => Promise<T>;
}

export class Client<RequestOption> {
  private baseUrl: string;
  constructor(
    private apiClient: ApiClient<RequestOption>,
    baseUrl: string,
  ) {
    this.baseUrl = baseUrl.replace(/\/$/, "");
  }

  public async createPublisherV2<RequestContentType extends RequestContentType$createPublisherV2>(
    params: Params$createPublisherV2<RequestContentType>,
    option?: RequestOption,
  ): Promise<Response$createPublisherV2$Status$200["application/json"]> {
    const url = this.baseUrl + `/create/v2/publisher/{id}`;
    const headers = {
      "Content-Type": params.headers["Content-Type"],
      Accept: "application/json",
    };
    const requestEncodings = {
      "application/x-www-form-urlencoded": {
        color: {
          style: "form",
          explode: false,
        },
      },
      "application/json": {
        color: {
          style: "form",
          explode: false,
        },
      },
    };
    return this.apiClient.request(
      {
        httpMethod: "POST",
        url,
        headers,
        requestBody: params.requestBody,
        requestBodyEncoding: requestEncodings[params.headers["Content-Type"]],
      },
      option,
    );
  }
}
```

#### `Templates.FunctionalApiClient.generator`

We also provide a function-based API client that replaces the class-based API client with `createClient`. Please inject the API client dependency and use it.

```ts
export interface RequestArgs {
  httpMethod: HttpMethod;
  url: string;
  headers: ObjectLike | any;
  requestBody?: ObjectLike | any;
  requestBodyEncoding?: Record<string, Encoding>;
  queryParameters?: QueryParameters | undefined;
}

export interface ApiClient<RequestOption> {
  request: <T = SuccessResponses>(requestArgs: RequestArgs, options?: RequestOption) => Promise<T>;
}

export const createClient = <RequestOption>(apiClient: ApiClient<RequestOption>, baseUrl: string) => {
  const _baseUrl = baseUrl.replace(/\/$/, "");
  return {
    createPublisherV2: <RequestContentType extends RequestContentType$createPublisherV2>(
      params: Params$createPublisherV2<RequestContentType>,
      option?: RequestOption,
    ): Promise<Response$createPublisherV2$Status$200["application/json"]> => {
      const url = _baseUrl + `/create/v2/publisher/{id}`;
      const headers = {
        "Content-Type": params.headers["Content-Type"],
        Accept: "application/json",
      };
      const requestEncodings = {
        "application/x-www-form-urlencoded": {
          color: {
            style: "form",
            explode: false,
          },
        },
        "application/json": {
          color: {
            style: "form",
            explode: false,
          },
        },
      };
      return apiClient.request(
        {
          httpMethod: "POST",
          url,
          headers,
          requestBody: params.requestBody,
          requestBodyEncoding: requestEncodings[params.headers["Content-Type"]],
        },
        option,
      );
    },
  };
};
```

#### `Templates.CurryingFunctionalApiClient.generator`

**Tree shaking support**

We also provide a curried function-based API client that requires injection of API client for each `operationId`. The first function argument demands `ApiClient` while the second function argument demands `RequestArgs`. The `ApiClient` interface is different from the others, as it requires `uri` as an argument.

This is designed for use cases that utilize **tree shaking**.

```ts
export interface RequestArgs {
  httpMethod: HttpMethod;
  uri: string; // <------------------ Note that the uri
  headers: ObjectLike | any;
  requestBody?: ObjectLike | any;
  requestBodyEncoding?: Record<string, Encoding>;
  queryParameters?: QueryParameters | undefined;
}
export interface ApiClient<RequestOption> {
  request: <T = SuccessResponses>(requestArgs: RequestArgs, options?: RequestOption) => Promise<T>;
}
export const createPublisherV2 =
  <RequestOption>(apiClient: ApiClient<RequestOption>) =>
  <RequestContentType extends RequestContentType$createPublisherV2>(
    params: Params$createPublisherV2<RequestContentType>,
    option?: RequestOption,
  ): Promise<Response$createPublisherV2$Status$200["application/json"]> => {
    const uri = `/create/v2/publisher/{id}`;
    const headers = {
      "Content-Type": params.headers["Content-Type"],
      Accept: "application/json",
    };
    const requestEncodings = {
      "application/x-www-form-urlencoded": {
        color: {
          style: "form",
          explode: false,
        },
      },
      "application/json": {
        color: {
          style: "form",
          explode: false,
        },
      },
    };
    return apiClient.request(
      {
        httpMethod: "POST",
        uri,
        headers,
        requestBody: params.requestBody,
        requestBodyEncoding: requestEncodings[params.headers["Content-Type"]],
      },
      option,
    );
  };
```

### Split the type definition file and the API Client implementation

```ts
import * as fs from "fs";

import { CodeGenerator } from "@himenon/openapi-typescript-code-generator";
import * as Templates from "@himenon/openapi-typescript-code-generator/dist/templates";
import type * as Types from "@himenon/openapi-typescript-code-generator/dist/types";

const main = () => {
  const codeGenerator = new CodeGenerator("your/openapi/spec.yml");

  const apiClientGeneratorTemplate: Types.CodeGenerator.CustomGenerator<Templates.FunctionalApiClient.Option> = {
    generator: Templates.FunctionalApiClient.generator,
    option: {},
  };

  const typeDefCode = codeGenerator.generateTypeDefinition();
  const apiClientCode = codeGenerator.generateCode([
    {
      generator: () => {
        return [`import { Schemas, Responses } from "./types";`];
      },
    },
    codeGenerator.getAdditionalTypeDefinitionCustomCodeGenerator(),
    apiClientGeneratorTemplate,
  ]);

  fs.writeFileSync("types.ts", typeDefCode, { encoding: "utf-8" });
  fs.writeFileSync("apiClient.ts", apiClientCode, { encoding: "utf-8" });
};

main();
```

## Create a Code Template

The examples in this section can be used in the following ways

```ts
import * as fs from "fs";

import { CodeGenerator } from "@himenon/openapi-typescript-code-generator";
import type * as Types from "@himenon/openapi-typescript-code-generator/dist/types";

/** Write the definition of the Code Template here. */
const customGenerator: Types.CodeGenerator.CustomGenerator<{}> = {
  /** .... */
};

const codeGenerator = new CodeGenerator("your/openapi/spec.yml");

const code = codeGenerator.generateCode([customGenerator]);

fs.writeFileSync("output/file/name", code, { encoding: "utf-8" });
```

### Define a text-based code template

A self-defined code generator can return an array of `string`.

```ts
import * as Types from "@himenon/openapi-typescript-code-generator/dist/types";

interface Option {
  showLog?: boolean;
}

const generator: Types.CodeGenerator.GenerateFunction<Option> = (payload: Types.CodeGenerator.Params[], option): string[] => {
  if (option && option.showLog) {
    console.log("show log message");
  }
  return ["Hello world"];
};

const customGenerator: Types.CodeGenerator.CustomGenerator<Option> = {
  generator: generator,
  option: {},
};
```

### Define using the information extracted from OpenAPI Schema

The self-defined code generator can accept parameters extracted from OpenAPI Schema.
See Type definitions for available parameters.

```ts
import * as Types from "@himenon/openapi-typescript-code-generator/dist/types";

interface Option {}

const generator: Types.CodeGenerator.GenerateFunction<Option> = (payload: Types.CodeGenerator.Params[], option): string[] => {
  return payload.map(params => {
    return `function ${params.operationId}() { console.log("${params.comment}") }`;
  });
};

const customGenerator: Types.CodeGenerator.CustomGenerator<Option> = {
  generator: generator,
  option: {},
};
```

### Define any Data Types Format

Convert a Data Type with the following `format` to any type definition.

```yaml
components:
  schemas:
    Binary:
      type: string
      format: binary
    IntOrString:
      type: string
      format: int-or-string
    AandB:
      type: string
      format: A-and-B
```

The option to convert the Data Type Format to an arbitrary type definition is defined as follows.

```ts
import { CodeGenerator, Option } from "@himenon/openapi-typescript-code-generator";
const option: Option = {
  convertOption: {
    formatConversions: [
      {
        selector: {
          format: "binary",
        },
        output: {
          type: ["Blob"],
        },
      },
      {
        selector: {
          format: "int-or-string",
        },
        output: {
          type: ["number", "string"],
        },
      },
      {
        selector: {
          format: "A-and-B",
        },
        output: {
          type: ["A", "B"],
          multiType: "allOf",
        },
      },
    ],
  },
};
const codeGenerator = new CodeGenerator(inputFilename, option);
```

The typedef generated by this will look like this

```ts
export namespace Schemas {
  export type Binary = Blob;
  export type IntOrString = number | string;
  export type AandB = A & B;
}
```

### Define a code template with TypeScript AST

You can extend your code using the API of TypeScript AST.
You can directly use the API of TypeScript AST or use the wrapper API of TypeScript AST provided by this library.

```ts
import * as Types from "@himenon/openapi-typescript-code-generator/dist/types";
import { TsGenerator } from "@himenon/openapi-typescript-code-generator/dist/api";

interface Option {}

const factory = TsGenerator.Factory.create();

const generator: Types.CodeGenerator.GenerateFunction<Option> = (
  payload: Types.CodeGenerator.Params[],
  option,
): Types.CodeGenerator.IntermediateCode[] => {
  return payload.map(params => {
    return factory.InterfaceDeclaration.create({
      export: true,
      name: params.functionName,
      members: [],
    });
  });
};

const customGenerator: Types.CodeGenerator.CustomGenerator<Option> = {
  generator: generator,
  option: {},
};
```

## API

### CodeGenerator

```ts
import { CodeGenerator } from "@himenon/openapi-typescript-code-generator";
```

#### validateOpenApiSchema

Performs validation of the input OpenAPI Schema.

#### generateTypeDefinition

Generates code that converts OpenAPI Schema to TypeScript type definitions.

#### generateCode

You can specify several of your own code generators, and the generators can use parameters extracted from OpenAPI Schema.
It internally performs the conversion of an array of `string` or `ts.Statement` as a string.

For example, creating a generator in units of file divisions increases the reusability of the generator.

#### getCodeGeneratorParamsArray

It provides parameters extracted from OpenAPI Schema.

#### getAdditionalTypeDefinitionCustomCodeGenerator

This is a type definition file for `Templates.FunctionalApiClient`. The reason it is not included in `generateTypeDefinition` is that you may not use the type definition generated by this function depending on your usage.

※ The reason it is not included in `generateTypeDefinition` is that you may not use the type definitions generated by this function depending on your application.

### TsGenerator

```ts
import { TsGenerator } from "@himenon/openapi-typescript-code-generator/dist/api";
```

This is a wrapper API for the TypeScript AST used internally.
It is subject to change without notice.

### OpenApiTools

```ts
import { OpenApiTools } from "@himenon/openapi-typescript-code-generator/dist/api";
```

#### Parser

- `OpenApiTools.Parser`

This is the API for parsing OpenAPI Schema.
It is subject to change without notice.

## Restrictions

### Directory Restrictions for Remote Reference

There is a limitation on the directory structure supported.
To simplify implementation when converting directory structures to TypeScript namespaces, Remote References using `$ref` should only be defined in the following directory structures.
If you want to extend it, please fork this repository and do it yourself.

```
spec.yml // entry file
components/
  headers/
  parameters/
  pathItems/
  requestBodies/
  responses/
  schemas/
  paths/
```

### HTTP communication restrictions for Remote Reference

`$ref: http://....` Currently not supported. We hope to support it in the future.

## Contributions

First of all, thank you for your interest.
When converting from the API specification to TypeScript code, resolving reference relationships can be particularly challenging, and there may not be enough test cases.
Adding test cases is a very powerful support for stabilizing the behavior, so please report any bugs you find that are behaving strangely.
Also, the basic design concepts of this repository can be found below. If you want to make changes that do not follow these concepts, please fork and extend them.
If your changes are in line with the design concept, please submit a pull request or issue!

## Design Concept

- Be typedef first.
- Typedefs should not contain any entities (file size should be 0 when typedefs are converted to `.js`)
- The directory structure should be mapped to the typedef structure.
- No dependency on any API client library.
- Can be extended by TypeScript AST.
- Conform to the OpenAPI specification.
- It should be a single file to maintain portability.

## Development

```bash
git clone https://github.com/Himenon/openapi-typescript-code-generator.git
cd openapi-typescript-code-generator
pnpm i
#### your change
pnpm build
pnpm run test:code:gen
pnpm run update:snapshot # if you changed
pnpm run test
```

## Useful development tools

TypeScript AST

- https://ts-ast-viewer.com

## LICENCE

[@himenon/openapi-typescript-code-generator](https://github.com/Himenon/typescript-codegen), MIT

### Reference implementation

Validation Design

- Copyright (c) 2018 Kogo Software LLC - [https://github.com/kogosoftwarellc/open-api/tree/master/packages/openapi-schema-validator#readme](openapi-schema-validator)
 readmeEtag: '"162fb539b6d489eef4e5c58714484103f827ab0c"' readmeLastModified: Mon, 04 Mar 2024 06:11:33 GMT repositoryId: 321679344 description: TypeScript code generator via OpenAPI scheme. created: '2020-12-15T13:40:09Z' updated: '2026-01-23T08:00:34Z' language: TypeScript archived: false stars: 207 watchers: 3 forks: 16 owner: Himenon logo: https://avatars.githubusercontent.com/u/6715229?v=4 license: MIT repoEtag: '"6a8d2eb15a6d1c50f592c6804a4e5ffc44d2fb63f63de663fef2d500c5ebd20e"' repoLastModified: Fri, 23 Jan 2026 08:00:34 GMT foundInMaster: true category: SDK id: fda6dd851e4578e64350df218e79cdaa - source: openapi3 tags repository: https://github.com/aress31/openapi-parser v3: true repositoryMetadata: base64Readme: >- # openapi-parser

[![Java CI with Gradle](https://github.com/aress31/openapi-parser/actions/workflows/gradle-build.yml/badge.svg)](https://github.com/aress31/openapi-parser/actions/workflows/gradle-build.yml)
<a href="https://portswigger.net/bappstore/6bf7574b632847faaaa4eb5e42f1757c"><img alt="bapp store" src="https://img.shields.io/badge/BApp-Published-orange.svg"></a>
<a href="https://www.java.com"><img alt="lang" src="https://img.shields.io/badge/Lang-Java-blue.svg"></a>
<a href="https://opensource.org/licenses/Apache-2.0"><img alt="license" src="https://img.shields.io/badge/License-Apache%202.0-red.svg"></a>
<img alt="version" src="https://img.shields.io/badge/Version-2.3-green.svg">

> [!UPDATE]
> This extension has been updated to use the latest [Burp Montoya Java API](https://portswigger.github.io/burp-extensions-montoya-api/javadoc/burp/api/montoya/package-summary.html). The extension has undergone a complete overhaul to improve both its `UI`/`UX` and performance. These changes ensure that the extension is modern and optimised for use.

## openapi-parser is a `Burp Suite` extension designed for `OpenAPI`-based `API` testing

> The `OpenAPI` Specification (`OAS`) defines a standard, programming language-agnostic interface description for `REST` `APIs`, which allows both humans and computers to discover and understand the capabilities of a service without requiring access to source code, additional documentation, or inspection of network traffic. When properly defined via `OpenAPI`, a consumer can understand and interact with the remote service with a minimal amount of implementation logic. Similar to what interface descriptions have done for lower-level programming, the `OpenAPI` Specification removes guesswork in calling a service.
>
> Use cases for machine-readable `API` definition documents include, but are not limited to: interactive documentation; code generation for documentation, clients, and servers; and automation of test cases. `OpenAPI` documents describe an `API`'s services and are represented in either `YAML` or `JSON` formats. These documents may either be produced and served statically or be generated dynamically from an application.
>
> \- [`OpenAPI` Initiative](https://github.com/OAI/`OpenAPI`-Specification)

Performing security assessment of `OpenAPI`-based `APIs` can be a tedious task due to `Burp Suite` (industry standard) lacking native `OpenAPI` parsing capabilities. A solution to this situation, is to use third-party tools (e.g. `SOAP-UI`) or to implement custom scripts (often on a per engagement basis) to handle the parsing of `OpenAPI` documents and integrate/chain the results to `Burp Suite` to use its first class scanning capabilities.

openapi-parser is an `OpenAPI` parser that aims to streamline this entire process by allowing security professionals to use `Burp Suite` as a standalone tool for security assessment of `OpenAPI`-based `APIs`.

## Features

- `OpenAPI` documents can be parsed either from a supplied file or URL. The extension can fetch `OpenAPI` documents directly from a URL using the `Send to Swagger Parser` feature under the `Target -> Site map` context menu.
- Parse `OpenAPI` documents, formerly known as the `Swagger specification`, fully compliant with `OpenAPI` 2.0/3.0 Specifications (`OAS`).
- Requests can be directly viewed/edited within the extension prior to sending them to other Burp tools.
- Requests can be sent to the `Comparer`, `Intruder`, `Organizer`, `Repeater`, `Scanner`, `Site map` and `Scope` Burp tools.
- Requests matching specific criterias (detailed in the 'Parameters' tab) can be intercepted to automatically match and replace the parsed parameters default values defined in the 'Parameters' tab. This feature allows for fine-tuning of the requests prior to sending them to other Burp tools (e.g., scanner). Edited requests can be viewed within the 'Modified Request (`OpenAPI` Parser)' tab of Burp's message editor.
- Row highlighting allowing pentesters to highlight "interesting" `API` calls and/or colour code them for reporting purposes.
- Includes an export to `CSV` feature, allowing users to easily export selected `API` requests in `CSV` format for further analysis or reporting.
- Supports both `JSON` and `YAML` formats.

## Requirements

### 1. System requirements

- Operating System: Compatible with `Linux`, `macOS`, and `Windows` operating systems.
- Java Development Kit (JDK): `Version 11` or later.
- Burp Suite Professional or Community Edition: `Version 2023.11.1.3` or later.

  > [!IMPORTANT]
  > Please note that using any version lower than `2023.3.2` may result in a [java.lang.NoSuchMethodError](https://forum.portswigger.net/thread/montoya-api-nosuchmethoderror-275048be). It is crucial to use the specified version or a more recent one to avoid this issue.

### 2. Build tool

- Gradle: `Version 8.5` or later (recommended). The [build.gradle](https://github.com/aress31/openapi-parser/blob/main/lib/build.gradle) file is provided in the project repository.

### 3. Environment variables

- Set up the `JAVA_HOME` environment variable to point to the `JDK` installation directory.

Please ensure that all system requirements, including a compatible version of `Burp Suite`, are met before building and running the project. Note that the project's external dependencies will be automatically managed and installed by `Gradle` during the build process. Adhering to the requirements will help avoid potential issues and reduce the need for opening new issues in the project repository.

## Installation

### 1. Compilation

1. Ensure you have [Gradle](https://gradle.org/) installed and configured.

2. Download the `openapi-parser` repository:

   ```bash
   git clone https://github.com/aress31/openapi-parser
   cd .\openapi-parser\
   ```

3. Build the standalone `jar`:

   ```bash
   ./gradlew fatJar
   ```

### 2. Loading the Extension Into `Burp Suite`

To install `openapi-parser` in `Burp Suite`, first go to the `Extensions` tab and click on the `Add` button. Then, select the `openapi-parser-all` jar file located in the `.\build\libs` folder to load the extension.

Alternatively, you can skip the [Compilation](#1-compilation) step entirely and download the extension directly from the [BApp Store](https://portswigger.net/bappstore/6bf7574b632847faaaa4eb5e42f1757c).

_Note: The version distributed on the [BApp Store](https://portswigger.net/bappstore/6bf7574b632847faaaa4eb5e42f1757c) might be lagging behind the version available on this repository._

## Roadmap

- [ ] Beautify the graphical user interface.
- [ ] Deep parsing of `OpenAPI` schemas to collect all nested parameters along with their example/type.
- [ ] Code simplification/refactoring.
  - [x] Use `MyHttpRequest` instead of `RequestWithMetadata`.
- [x] Enable cells editing to change `API` calls directly from the `GUI`.
- [ ] Fix the custom request editor tab to work properly with intercepted requests based on the match and replace rulesets.
- [x] Further optimise the source code.
- [ ] Implement support for authenticated testing (via user-supplied `API`-keys).
- [x] Improve the `Param` column by adding parameters type (e.g. `inquery`, `inbody`).
- [x] Improve the tables and context menus.
- [x] Increase the extension verbosity (via the bottom panel).

See [TODO](TODO.md) for additional outstanding tasks.

## Project Information

In July 2016, after posting a request for improvement on the [PortSwigger support forum](https://support.portswigger.net/customer/portal/questions/16358278-swagger-parser-and-wsdler-improvement), I decided to take the initiative and to implement a solution myself.

The extension is still in development, feedback, comments and contributions are therefore much appreciated.

## Sponsor 💖

If this extension has saved you time and hassle during a security assessment, consider showing some love by sponsoring a cup of coffee ☕ for the developer. It's the fuel that powers development, after all. Just hit that shiny `Sponsor` button at the top of the page or [click here](https://github.com/sponsors/aress31) to contribute and keep the caffeine flowing. 💸

## Reporting Issues

Did you find a bug? Well, don't just let it crawl around! Let's squash it together like a couple of bug whisperers! 🐛💪

Please report any issues on the [GitHub issues tracker](https://github.com/aress31/openapi-parser/issues). Together, we'll make this extension as reliable as a cockroach surviving a nuclear apocalypse! 🚀

## Contributing

Looking to make a splash with your mad coding skills? 💻

Awesome! Contributions are welcome and greatly appreciated. Please submit all PRs on the [GitHub pull requests tracker](https://github.com/aress31/openapi-parser/pulls). Together we can make this extension even more amazing! 🚀

## License

See [LICENSE](LICENSE).
 readmeEtag: '"32d486082518e7ab9b4321d546b059fe5dc739c1"' readmeLastModified: Wed, 03 Jan 2024 18:53:00 GMT repositoryId: 70046718 description: >- Parse OpenAPI documents into Burp Suite for automating OpenAPI-based APIs security assessments (approved by PortSwigger for inclusion in their official BApp Store). created: '2016-10-05T09:32:42Z' updated: '2026-01-12T04:15:30Z' language: Java archived: false stars: 206 watchers: 4 forks: 53 owner: aress31 logo: https://avatars.githubusercontent.com/u/11601622?v=4 license: Apache-2.0 repoEtag: '"936e7d492d724cb64c2b2f359b5a2861df30eae0a4e72fb9d8fce9cdc3f37c6e"' repoLastModified: Mon, 12 Jan 2026 04:15:30 GMT foundInMaster: true category: - Documentation - Parsers id: 3dd0465c78eeb00e652527da7f6ec81a oldLocations: - https://github.com/aress31/swurg - source: openapi3 tags repository: https://github.com/openapi-library/openapivalidators v3: true repositoryMetadata: base64Readme: >- # OpenAPI Validators

![build status](https://github.com/openapi-library/OpenAPIValidators/actions/workflows/ci.yml/badge.svg)
![style](https://img.shields.io/badge/code%20style-airbnb-ff5a5f.svg)
[![codecov](https://codecov.io/gh/openapi-library/OpenAPIValidators/branch/master/graph/badge.svg)](https://codecov.io/gh/openapi-library/OpenAPIValidators)
[![MIT License](https://img.shields.io/npm/l/openapi-validator.svg?style=flat-square)](https://github.com/openapi-library/OpenAPIValidators/blob/master/LICENSE)
[![contributions welcome](https://img.shields.io/badge/contributions-welcome-brightgreen.svg?style=flat)](https://github.com/openapi-library/OpenAPIValidators/blob/master/CONTRIBUTING.md)

Use Jest or Chai to assert that HTTP responses satisfy an OpenAPI spec.

## Problem 😕

If your server's behaviour doesn't match your API documentation, then you need to correct your server, your documentation, or both. The sooner you know the better.

## Solution 😄

These test plugins let you automatically test whether your server's behaviour and documentation match. They extend Jest and Chai to support the [OpenAPI standard](https://swagger.io/docs/specification/about/) for documenting REST APIs. In your JavaScript tests, you can simply assert `expect(responseObject).toSatisfyApiSpec()`

### [jest-openapi](https://github.com/openapi-library/OpenAPIValidators/tree/master/packages/jest-openapi#readme)

[![downloads](https://img.shields.io/npm/dm/jest-openapi)](https://www.npmjs.com/package/jest-openapi)
[![npm](https://img.shields.io/npm/v/jest-openapi.svg)](https://www.npmjs.com/package/jest-openapi)

### [Chai OpenAPI Response Validator](https://github.com/openapi-library/OpenAPIValidators/tree/master/packages/chai-openapi-response-validator#readme)

[![downloads](https://img.shields.io/npm/dm/chai-openapi-response-validator)](https://www.npmjs.com/package/chai-openapi-response-validator)
[![npm](https://img.shields.io/npm/v/chai-openapi-response-validator.svg)](https://www.npmjs.com/package/chai-openapi-response-validator)

## Contributors ✨

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://github.com/rwalle61"><img src="https://avatars1.githubusercontent.com/u/18170169?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Richard Waller</b></sub></a><br /><a href="#maintenance-rwalle61" title="Maintenance">🚧</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=rwalle61" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=rwalle61" title="Documentation">📖</a> <a href="https://github.com/openapi-library/OpenAPIValidators/pulls?q=is%3Apr+reviewed-by%3Arwalle61" title="Reviewed Pull Requests">👀</a></td>
    <td align="center"><a href="https://github.com/JonnySpruce"><img src="https://avatars3.githubusercontent.com/u/30812276?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Jonny Spruce</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=JonnySpruce" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=JonnySpruce" title="Documentation">📖</a> <a href="https://github.com/openapi-library/OpenAPIValidators/pulls?q=is%3Apr+reviewed-by%3AJonnySpruce" title="Reviewed Pull Requests">👀</a></td>
    <td align="center"><a href="https://github.com/AlexDobeck"><img src="https://avatars2.githubusercontent.com/u/10519388?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Alex Dobeck</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=AlexDobeck" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3AAlexDobeck" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://github.com/BenGu3"><img src="https://avatars2.githubusercontent.com/u/7105857?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ben Guthrie</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=BenGu3" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3ABenGu3" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://martijnvegter.com/"><img src="https://avatars3.githubusercontent.com/u/25134477?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Martijn Vegter</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=mvegter" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/ludeknovy"><img src="https://avatars1.githubusercontent.com/u/13610612?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Ludek</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=ludeknovy" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3Aludeknovy" title="Bug reports">🐛</a></td>
    <td align="center"><a href="https://github.com/tgiardina"><img src="https://avatars1.githubusercontent.com/u/37459104?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Tommy Giardina</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=tgiardina" title="Code">💻</a> <a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3Atgiardina" title="Bug reports">🐛</a></td>
  </tr>
  <tr>
    <td align="center"><a href="https://xotabu4.github.io/"><img src="https://avatars3.githubusercontent.com/u/3033972?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Oleksandr Khotemskyi</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=Xotabu4" title="Documentation">📖</a></td>
    <td align="center"><a href="https://github.com/amitkeinan9"><img src="https://avatars.githubusercontent.com/u/16577335?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Amit Keinan</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=amitkeinan9" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/DetachHead"><img src="https://avatars.githubusercontent.com/u/57028336?v=4?s=100" width="100px;" alt=""/><br /><sub><b>DetachHead</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/issues?q=author%3ADetachHead" title="Bug reports">🐛</a></td>
    <td align="center"><a href="http://karlssonkristoffer.com/"><img src="https://avatars.githubusercontent.com/u/20490202?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Kristoffer Karlsson</b></sub></a><br /><a href="https://github.com/openapi-library/OpenAPIValidators/commits?author=kristofferkarlsson93" title="Documentation">📖</a></td>
  </tr>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->
 readmeEtag: '"4d5a529f7a19b52fab62aaf17e9c5263bd765e2d"' readmeLastModified: Tue, 29 Mar 2022 10:12:21 GMT repositoryId: 199614125 description: Use Jest or Chai to assert that HTTP responses satisfy an OpenAPI spec created: '2019-07-30T08:56:01Z' updated: '2025-12-11T11:06:52Z' language: TypeScript archived: false stars: 194 watchers: 11 forks: 38 owner: openapi-library logo: https://avatars.githubusercontent.com/u/53815441?v=4 license: MIT repoEtag: '"982859aab5b874db7602c4d6d41812452f2b57b69b87a00845a5040995271994"' repoLastModified: Thu, 11 Dec 2025 11:06:52 GMT foundInMaster: true category: Testing id: a81dda804efb1a473233a16a10be8d0b - source: - openapi3 tags - openapi31 tags repository: https://github.com/redocly/openapi-sampler v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXNhbXBsZXIKClshW1RyYXZpcyBidWlsZCBzdGF0dXNdKGh0dHA6Ly9pbWcuc2hpZWxkcy5pby90cmF2aXMvUmVkb2NseS9vcGVuYXBpLXNhbXBsZXIuc3ZnP3N0eWxlPWZsYXQpXShodHRwczovL3RyYXZpcy1jaS5vcmcvUmVkb2NseS9vcGVuYXBpLXNhbXBsZXIpIFshW0NvdmVyYWdlIFN0YXR1c10oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vcmVwb3MvUmVkb2NseS9vcGVuYXBpLXNhbXBsZXIvYmFkZ2Uuc3ZnP2JyYW5jaD1tYXN0ZXImc2VydmljZT1naXRodWIpXShodHRwczovL2NvdmVyYWxscy5pby9naXRodWIvUmVkb2NseS9vcGVuYXBpLXNhbXBsZXI/YnJhbmNoPW1hc3RlcikgWyFbRGVwZW5kZW5jeSBTdGF0dXNdKGh0dHBzOi8vZGF2aWQtZG0ub3JnL1JlZG9jbHkvb3BlbmFwaS1zYW1wbGVyLnN2ZyldKGh0dHBzOi8vZGF2aWQtZG0ub3JnL1JlZG9jbHkvb3BlbmFwaS1zYW1wbGVyKSBbIVtkZXZEZXBlbmRlbmN5IFN0YXR1c10oaHR0cHM6Ly9kYXZpZC1kbS5vcmcvUmVkb2NseS9vcGVuYXBpLXNhbXBsZXIvZGV2LXN0YXR1cy5zdmcpXShodHRwczovL2RhdmlkLWRtLm9yZy9SZWRvY2x5L29wZW5hcGktc2FtcGxlciNpbmZvPWRldkRlcGVuZGVuY2llcykKClRvb2wgZm9yIGdlbmVyYXRpb24gc2FtcGxlcyBiYXNlZCBvbiBPcGVuQVBJIHBheWxvYWQvcmVzcG9uc2Ugc2NoZW1hCgojIyBGZWF0dXJlcwoKLSBEZXRlcm1pbmlzdGljIChnaXZlbiBhIHBhcnRpY3VsYXIgaW5wdXQsIHdpbGwgYWx3YXlzIHByb2R1Y2UgdGhlIHNhbWUgb3V0cHV0KQotIFN1cHBvcnRzIGNvbXBvdW5kIGtleXdvcmRzOiBgYWxsT2ZgLCBgb25lT2ZgLCBgYW55T2ZgLCBgaWYvdGhlbi9lbHNlYAotIFN1cHBvcnRzIGBhZGRpdGlvbmFsUHJvcGVydGllc2Agd2l0aCBbYHgtYWRkaXRpb25hbFByb3BlcnRpZXNOYW1lYF0oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvcmVkb2MvYmxvYi9tYXN0ZXIvZG9jcy9yZWRvYy12ZW5kb3ItZXh0ZW5zaW9ucy5tZCN4LWFkZGl0aW9uYWxwcm9wZXJ0aWVzbmFtZSkKLSBVc2VzIGBjb25zdGAsIGBleGFtcGxlc2AsIGBlbnVtYCBhbmQgYGRlZmF1bHRgIHdoZXJlIHBvc3NpYmxlIC0gaW4gdGhpcyBvcmRlcgotIEdvb2QgYXJyYXkgc3VwcG9ydDogc3VwcG9ydHMgYGNvbnRhaW5zYCwgYG1pbkl0ZW1zYCwgYG1heEl0ZW1zYCwgYW5kIHR1cGxlcyAoYGl0ZW1zYCBhcyBhbiBhcnJheSkKLSBTdXBwb3J0cyBgbWluTGVuZ3RoYCwgYG1heExlbmd0aGAsIGBtaW5gLCBgbWF4YCwgYGV4Y2x1c2l2ZU1pbmltdW1gLCBgZXhjbHVzaXZlTWF4aW11bWAsIChbbGltaXRlZF0oaHR0cHM6Ly9mYWtlcmpzLmRldi9hcGkvaGVscGVycy5odG1sI2Zyb21yZWdleHApKSBgcGF0dGVybmAKLSBTdXBwb3J0cyB0aGUgZm9sbG93aW5nIGBzdHJpbmdgIGZvcm1hdHM6CiAgLSBlbWFpbAogIC0gaWRuLWVtYWlsCiAgLSBwYXNzd29yZAogIC0gZGF0ZS10aW1lCiAgLSBkYXRlCiAgLSB0aW1lCiAgLSBpcHY0CiAgLSBpcHY2CiAgLSBob3N0bmFtZQogIC0gaWRuLWhvc3RuYW1lCiAgLSB1cmkKICAtIHVyaS1yZWZlcmVuY2UKICAtIHVyaS10ZW1wbGF0ZQogIC0gaXJpCiAgLSBpcmktcmVmZXJlbmNlCiAgLSB1dWlkCiAgLSBqc29uLXBvaW50ZXIKICAtIHJlbGF0aXZlLWpzb24tcG9pbnRlcgogIC0gcmVnZXgKLSBJbmZlcnMgc2NoZW1hIHR5cGUgYXV0b21hdGljYWxseSBmb2xsb3dpbmcgc2FtZSBydWxlcyBhcyBbanNvbi1zY2hlbWEtZmFrZXJdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL2pzb24tc2NoZW1hLWZha2VyI2luZmVycmVkLXR5cGVzKQotIFN1cHBvcnQgZm9yIGAkcmVmYCByZXNvbHZpbmcKLSBIYXMgYmFzaWMgc3VwcG9ydHMgZm9yIEpTT04gU2NoZW1hIGRyYWZ0IDcgKHRoYW5rcyB0byBbQFAwbGlwXShodHRwczovL2dpdGh1Yi5jb20vUDBsaXApIGZyb20gW0BzdG9wbGlnaHRpb10oaHR0cHM6Ly9naXRodWIuY29tL3N0b3BsaWdodGlvKSBmb3IgY29udHJpYnV0aW5nKQoKIyMgSW5zdGFsbGF0aW9uCgpJbnN0YWxsIHVzaW5nIFtucG1dKGh0dHBzOi8vZG9jcy5ucG1qcy5jb20vZ2V0dGluZy1zdGFydGVkL3doYXQtaXMtbnBtKQoKICAgIG5wbSBpbnN0YWxsIG9wZW5hcGktc2FtcGxlcgoKb3IgdXNpbmcgW3lhcm5dKGh0dHBzOi8veWFybnBrZy5jb20pCgogICAgeWFybiBhZGQgb3BlbmFwaS1zYW1wbGVyCgpUaGVuIHJlcXVpcmUgaXQgaW4geW91ciBjb2RlOgoKYGBganMKdmFyIE9wZW5BUElTYW1wbGVyID0gcmVxdWlyZSgnb3BlbmFwaS1zYW1wbGVyJyk7CmBgYAoKIyMgVXNhZ2UKIyMjIyBgT3BlbkFQSVNhbXBsZXIuc2FtcGxlKHNjaGVtYSwgW29wdGlvbnNdLCBbc3BlY10pYAotICoqc2NoZW1hKiogKF9yZXF1aXJlZF8pIC0gYG9iamVjdGAKQW4gW09wZW5BUEkgU2NoZW1hIE9iamVjdF0oaHR0cDovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8jc2NoZW1hT2JqZWN0KSBvciBhIEpTT04gU2NoZW1hIERyYWZ0IDcgZG9jdW1lbnQuCi0gKipvcHRpb25zKiogKF9vcHRpb25hbF8pIC0gYG9iamVjdGAKQXZhaWxhYmxlIG9wdGlvbnM6CiAgLSAqKnNraXBOb25SZXF1aXJlZCoqIC0gYGJvb2xlYW5gCiAgRG9uJ3QgaW5jbHVkZSBub24tcmVxdWlyZWQgb2JqZWN0IHByb3BlcnRpZXMgbm90IHNwZWNpZmllZCBpbiBbYHJlcXVpcmVkYCBwcm9wZXJ0eSBvZiB0aGUgc2NoZW1hIG9iamVjdF0oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9kYXRhLW1vZGVscy9kYXRhLXR5cGVzLyNyZXF1aXJlZCkKICAtICoqc2tpcFJlYWRPbmx5KiogLSBgYm9vbGVhbmAKICBEb24ndCBpbmNsdWRlIGByZWFkT25seWAgb2JqZWN0IHByb3BlcnRpZXMKICAtICoqc2tpcFdyaXRlT25seSoqIC0gYGJvb2xlYW5gCiAgRG9uJ3QgaW5jbHVkZSBgd3JpdGVPbmx5YCBvYmplY3QgcHJvcGVydGllcwogIC0gKipxdWlldCoqIC0gYGJvb2xlYW5gCiAgRG9uJ3QgbG9nIGNvbnNvbGUgd2FybmluZyBtZXNzYWdlcwotICoqc3BlYyoqIC0gd2hvbGUgc3BlY2lmaWNhdGlvbiB3aGVyZSB0aGUgc2NoZW1hIGlzIHRha2VuIGZyb20uIFJlcXVpcmVkIG9ubHkgd2hlbiBzY2hlbWEgbWF5IGNvbnRhaW4gYCRyZWZgLiAqKnNwZWMqKiBtdXN0IG5vdCBjb250YWluIGFueSBleHRlcm5hbCByZWZlcmVuY2VzCgojIyBFeGFtcGxlCmBgYGpzCmNvbnN0IE9wZW5BUElTYW1wbGVyID0gcmVxdWlyZSgnLicpOwpPcGVuQVBJU2FtcGxlci5zYW1wbGUoewogIHR5cGU6ICdvYmplY3QnLAogIHByb3BlcnRpZXM6IHsKICAgIGE6IHt0eXBlOiAnaW50ZWdlcicsIG1pbmltdW06IDEwfSwKICAgIGI6IHt0eXBlOiAnc3RyaW5nJywgZm9ybWF0OiAncGFzc3dvcmQnLCBtaW5MZW5ndGg6IDEwfSwKICAgIGM6IHt0eXBlOiAnYm9vbGVhbicsIHJlYWRPbmx5OiB0cnVlfQogIH0KfSwge3NraXBSZWFkT25seTogdHJ1ZX0pOwovLyB7IGE6IDEwLCBiOiAncGEkJHdvcmRfcScgfQpgYGAK readmeEtag: '"edd41323d08c6427f8d1a0dc4acab1874964b3e9"' readmeLastModified: Thu, 01 Aug 2024 06:57:47 GMT repositoryId: 59937640 description: >- :capital_abcd: Tool for generation samples based on OpenAPI(fka Swagger) payload/response schema created: '2016-05-29T10:29:32Z' updated: '2026-01-27T17:29:06Z' language: JavaScript archived: false stars: 216 watchers: 14 forks: 51 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"efcf4ea0624563c967e9f6da4d3da8e538c03d685afce41ddfec8bbf843e13fc"' repoLastModified: Tue, 27 Jan 2026 17:29:06 GMT foundInMaster: true v3_1: true category: - Documentation - Parsers id: 03703532e31126094015624abe623cf9 - source: openapi3 tags repository: https://github.com/christianhelle/apiclientcodegen v3: true id: 30d6812808d99b1b58b61c10028cb8de repositoryMetadata: base64Readme: >- [![CLI Tool](https://github.com/christianhelle/apiclientcodegen/actions/workflows/cli-tool.yml/badge.svg)](https://github.com/christianhelle/apiclientcodegen/actions/workflows/cli-tool.yml)
[![VSIX](https://github.com/christianhelle/apiclientcodegen/actions/workflows/vsix.yml/badge.svg)](https://github.com/christianhelle/apiclientcodegen/actions/workflows/vsix.yml)
[![VS Code](https://github.com/christianhelle/apiclientcodegen/actions/workflows/vscode.yml/badge.svg)](https://github.com/christianhelle/apiclientcodegen/actions/workflows/vscode.yml)
[![IntelliJ](https://github.com/christianhelle/apiclientcodegen/actions/workflows/intellij.yml/badge.svg)](https://github.com/christianhelle/apiclientcodegen/actions/workflows/intellij.yml)
![Smoke Tests](https://github.com/christianhelle/apiclientcodegen/workflows/Smoke%20Tests/badge.svg)

[![Maintainability Rating](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=sqale_rating)](https://sonarcloud.io/dashboard?id=christianhelle_apiclientcodegen)
[![Reliability Rating](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=reliability_rating)](https://sonarcloud.io/dashboard?id=christianhelle_apiclientcodegen)
[![Security Rating](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=security_rating)](https://sonarcloud.io/dashboard?id=christianhelle_apiclientcodegen)
[![Bugs](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=bugs)](https://sonarcloud.io/dashboard?id=christianhelle_apiclientcodegen)
[![Vulnerabilities](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=vulnerabilities)](https://sonarcloud.io/dashboard?id=christianhelle_apiclientcodegen)
[![Code Smells](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=code_smells)](https://sonarcloud.io/summary/new_code?id=christianhelle_apiclientcodegen)
[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_apiclientcodegen&metric=coverage)](https://sonarcloud.io/summary/new_code?id=christianhelle_apiclientcodegen)

[![Version](https://img.shields.io/visual-studio-marketplace/v/ChristianResmaHelle.ApiClientCodeGenerator2022?label=Marketplace)](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.APIClientCodeGenerator2022) 
[![Downloads](https://img.shields.io/visual-studio-marketplace/d/ChristianResmaHelle.ApiClientCodeGenerator?label=VS%202019)](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.APIClientCodeGenerator)
[![Rating](https://img.shields.io/visual-studio-marketplace/stars/ChristianResmaHelle.ApiClientCodeGenerator)](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.APIClientCodeGenerator)
[![Downloads](https://img.shields.io/visual-studio-marketplace/d/ChristianResmaHelle.APIClientCodeGenerator2022?label=VS%202022)](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.APIClientCodeGenerator2022) 
[![Rating](https://img.shields.io/visual-studio-marketplace/stars/ChristianResmaHelle.ApiClientCodeGenerator2022)](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.APIClientCodeGenerator2022)
[![NuGet](https://img.shields.io/nuget/dt/rapicgen.svg?style=flat-square&label=nuget)](http://www.nuget.org/packages/rapicgen)

[![buymeacoffee](https://img.shields.io/badge/buy%20me%20a%20coffee-donate-yellow.svg)](https://www.buymeacoffee.com/christianhelle)

# REST API Client Code Generator

A collection of Visual Studio C# custom tool code generators for Swagger / OpenAPI specification files

## Download

- **[Visual Studio 2026 (Out-of-Process Model)](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.Rapicgen)**
- **[Visual Studio 2022](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.ApiClientCodeGenerator2022)**
- **[Visual Studio 2019](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.ApiClientCodeGenerator)**
- **[Visual Studio 2017](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.ApiClientCodeGenerator2017)**
- **[Visual Studio Code](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.apiclientcodegen)**
- **[Visual Studio for Mac](#visual-studio-for-mac)**
- **[JetBrains Rider](https://plugins.jetbrains.com/plugin/28472-rest-api-client-code-generator)**

## Features

- Supports Visual Studio 2017, 2019, 2022, 2026, [Visual Studio Code](#visual-studio-code), [Visual Studio for Mac](#visual-studio-for-mac), and [JetBrains Rider](#intellij-plugin)
- Add New REST API Client to a project from an OpenAPI specification URL (e.g <https://petstore.swagger.io/v2/swagger.json>) using the following code generators:
  - [NSwag](https://github.com/RicoSuter/NSwag)
  - [OpenAPI Generator](https://github.com/OpenAPITools/openapi-generator)
  - [Swagger Codegen](https://github.com/swagger-api/swagger-codegen)
  - [Refitter](https://github.com/christianhelle/refitter)
  - [Microsoft Kiota](https://github.com/microsoft/kiota)
  - [AutoRest](https://github.com/Azure/autorest)
- Define custom namespace for the generated file
- Auto-updating of generated code file when changes are made to the OpenAPI specification JSON or YAML file
- Generate code using a configuration files using the following:
  - `.nswag` configuration files from [NSwagStudio](https://github.com/RicoSuter/NSwag/wiki/NSwagStudio) by including it in the project and using the **Generate NSwag Studio output** context menu
  - `.refitter` settings files from [Refitter](https://github.com/christianhelle/refitter) by including it in the project and using the **Generate Refitter output** context menu
  - `kiota-lock.json` configuration files from [Microsoft Kiota](https://github.com/microsoft/kiota) by including it in the project and using the **Generate Kiota output** context menu

### Custom Tools

Custom tools let you associate a tool with an item in a project and run that tool whenever the file is saved

- ***NSwagCodeGenerator*** - Generates a single file C# REST API Client using the [NSwag.CodeGeneration.CSharp](https://github.com/RSuter/NSwag/wiki/CSharpClientGenerator) [nuget package](https://www.nuget.org/packages/NSwag.CodeGeneration.CSharp/) **v14.6.3**

- ***OpenApiCodeGenerator*** - Generates a single file C# REST API Client using **[OpenAPI Generator v7.19.0](https://github.com/OpenAPITools/openapi-generator/releases/tag/v7.19.0)**.
The output file is the result of merging all the files generated using the OpenAPI Generator tool with:
`generate -g csharp --input-spec [swagger file] --output [output file] -DapiTests=false -DmodelTests=false -DpackageName=[default namespace] --skip-overwrite`. It is possible to configure the OpenAPI Generator to generate multiple files which will be placed at the same path as the OpenAPI specifications document that was used to generate code, this is done under Tools -> REST API Client Code Generator -> OpenAPI Generator and setting **Generate Multiple Files** to **true**

- ***KiotaCodeGenerator*** - Generates a single file C# REST API Client using the Microsoft project **[Kiota v1.30.0](https://learn.microsoft.com/en-us/openapi/kiota/)** generator. The output file is the result of merging all the files generated using the Kiota dotnet tool with: `generate -l CSharp -d [swagger file] -o [output file] -n [default namespace]`. It is possible to configure the OpenAPI Generator to generate multiple files which will be placed at the same path as the OpenAPI specifications document that was used to generate code, this is done under Tools -> REST API Client Code Generator -> Kiota and setting **Generate Multiple Files** to **true**

- ***SwaggerCodeGenerator*** - Generates a single file C# REST API Client using **Swagger Codegen CLI v3.0.34**.
The output file is the result of merging all the files generated using the Swagger Codegen CLI tool with:
`generate -l csharp --input-spec [swagger file] --output [output file] -DapiTests=false -DmodelTests=false -DpackageName=[default namespace] --skip-overwrite`

- ***AutoRestCodeGenerator*** - Generates a single file C# REST API Client using **AutoRest v3.0.0-beta.20210504.2** for OpenAPI v3 and **AutoRest v2.0.4417** for OpenAPI v2. The resulting file is the equivalent of using the AutoRest CLI tool with:
 `--csharp --input-file=[swagger file] --output-file=[output file] --namespace=[default namespace] --add-credentials`

- ***RefitterCodeGenerator*** - Generates a single file C# REST API Client interface for [Refit](https://github.com/reactiveui/refit) using [Refitter.Core](https://github.com/christianhelle/refitter) [nuget package](https://www.nuget.org/packages/Refitter.Core/) **v1.7.3**.
The output file contains a Refit interface generated by [Refitter](https://github.com/christianhelle/refitter) and contracts generated using [NSwag.CodeGeneration.CSharp](https://github.com/RSuter/NSwag/wiki/CSharpClientGenerator) [nuget package](https://www.nuget.org/packages/NSwag.CodeGeneration.CSharp/)

### Dependencies

The custom tool code generators piggy back on top of well known Open API client code generators like **AutoRest**, **NSwag**, **OpenAPI Generator**, **Microsoft Kiota**, **Refitter**, and **Swagger Codegen CLI**. These tools require [.NET SDK](https://dotnet.microsoft.com/download), [NPM](https://www.npmjs.com/get-npm) and the [Java Runtime Environment](https://java.com/en/download/manual.jsp) to be installed on the developers machine. Alternative Java SDK implementations such as the [OpenJDK](https://adoptopenjdk.net) works fine with this extension. By default, the path to **java.exe** is read from the **JAVA_HOME** environment variable, but is also configurable in the Settings screen. Future versions of the extension will include the [OpenJDK](https://adoptopenjdk.net) in the VSIX package

The **Swagger Codegen CLI** and **OpenAPI Generator** are distributed as JAR files and are downloaded on demand but requires the Java SDK to be installed on the machine. **AutoRest** is installed on-demand via [NPM](https://www.npmjs.com/get-npm) as a global tool and uses the latest available version. **Microsoft Kiota** is installed on-demand as a .NET Tool and requires .NET 7.0. This means that using these custom tools have an initial delay upon first time use. 

**NSwagStudio** is stand alone UI tool for editing a **.nswag** specification file for generating code. This tool is optional to install and official installation instructions are available on the [NSwag Wiki on Github](https://github.com/RicoSuter/NSwag/wiki/NSwagStudio). If **NSwagStudio** is not installed on the machine then the Visual Studio Extension will install the **NSwag CLI** as a [.NET global tool](https://www.nuget.org/packages/NSwag.ConsoleCore) from nuget.org using the latest available version. 

The **NSwag** code generator produces code that depends on the [Newtonsoft.Json](https://www.nuget.org/packages/Newtonsoft.Json/13.0.3) NuGet package

The **Refitter** code generator produces code that depends on the [Refit](https://www.nuget.org/packages/Refit/9.0.2) NuGet package

The **OpenAPI Generator** code generator produces code that depends on the following NuGet packages:

- [RestSharp](https://www.nuget.org/packages/RestSharp/112.0.0)
- [JsonSubTypes](https://www.nuget.org/packages/JsonSubTypes/2.0.1)
- [Polly](https://www.nuget.org/packages/Polly/8.6.5)
- [Newtonsoft.Json](https://www.nuget.org/packages/Newtonsoft.Json/13.0.3)

The **Microsoft Kiota** code generator produces code that depends on the following NuGet packages

- [Microsoft.Kiota.Abstractions](https://www.nuget.org/packages/Microsoft.Kiota.Abstractions)
- [Microsoft.Kiota.Http.HttpClientLibrary](https://www.nuget.org/packages/Microsoft.Kiota.Http.HttpClientLibrary)
- [Microsoft.Kiota.Serialization.Form](https://www.nuget.org/packages/Microsoft.Kiota.Serialization.Form)
- [Microsoft.Kiota.Serialization.Text](https://www.nuget.org/packages/Microsoft.Kiota.Serialization.Text)
- [Microsoft.Kiota.Serialization.Json](https://www.nuget.org/packages/Microsoft.Kiota.Serialization.Json)
- [Microsoft.Kiota.Serialization.Multipart](https://www.nuget.org/packages/Microsoft.Kiota.Serialization.Multipart)
- [Microsoft.Kiota.Authentication.Azure](https://www.nuget.org/packages/Microsoft.Kiota.Authentication.Azure)
- [Azure.Identity](https://www.nuget.org/packages/Azure.Identity)

The **Swagger Codegen CLI** code generator produces code that depends on the [RestSharp](https://www.nuget.org/packages/RestSharp/105.2.3) and [JsonSubTypes](https://www.nuget.org/packages/JsonSubTypes/1.9.0) NuGet packages

The **AutoRest** code generator produces code that depends on the [Microsoft.Rest.ClientRuntime](https://www.nuget.org/packages/Microsoft.Rest.ClientRuntime/2.3.24) and [Newtonsoft.Json](https://www.nuget.org/packages/Newtonsoft.Json/13.0.3) NuGet packages

This Visual Studio Extension will automatically add the required NuGet packages that the generated code depends on

### Using Generated Code

For detailed examples and best practices on how to use the code generated by each code generator, see the **[Generated Code Usage Guide](docs/GeneratedCodeUsage.md)**.

The guide includes:
- Sample code for each generator (NSwag, AutoRest, Swagger Codegen CLI, OpenAPI Generator, Kiota, Refitter)
- Dependency injection examples
- Authentication patterns
- Error handling
- Links to official documentation

## Screenshots

![Add - API Client from OpenAPI Specification](images/add-new-menu.png)

![Enter URL to OpenAPI Specification](images/add-new-dialog.png)

![Solution Explorer Context Menus](images/solution-explorer-context-menu.jpg)

![NSwag Studio Context Menu](images/nswagstudio-context-menu.jpg)

![Refitter Context Menu](images/refitter-command.png)

![Kiota Context Menu](images/generate-kiota-output.png)

### Settings

This extension will by default make some assumptions on the installation paths for **Java** and **NPM** but also provides option pages for configuring this. The **Swagger Codegen CLI** and the **OpenAPI Generator** JAR files are by default downloaded to the user TEMP folder but it is also possible to specify to use existing JAR files

![Options - General](images/options-general.png)

Supports customising how AutoRest generates code based on the C# generator settings that the AutoRest CLI tool provides

![Options - AutoRest](images/options-autorest.png)

Supports customising how NSwag generates code using the properties exposed by the NSwag NuGet package

![Options - NSwag](images/options-nswag.png)

Supports customising how the **.nswag** file is generated using a subset of the options available in NSwag Studio

![Options - NSwag Studio](images/options-nswagstudio.png)

Supports customising how OpenAPI Generator generates code using the additional optional properties that the OpenAPI Generator CLI tool provides

![Options - OpenAPI Generator](images/options-openapigenerator.png)

Supports customising how Refitter generates the Refit interface. This allows you to configure whether to return the type directly or wrap it inside an `IApiResponse<T>`. You can also disable generating contracts or XML doc style comments

![Options - Refitter](images/refitter-options.png)

Supports allowing Kiota to generate multiple files

![Options - Kiota](images/options-kiota.png)

This extension collects errors and tracks feature usages to a service called [Exceptionless](https://exceptionless.com) and [Azure Application Insights](https://learn.microsoft.com/en-us/azure/azure-monitor/app/app-insights-overview?WT.mc_id=DT-MVP-5004822). This is done anonymously using a support key and a generated anonymous identity based on a secure hash of username@host

![Options - Analytics](images/support-key.png)

### Visual Studio Code

![VS Code - Command Palette](images/vscode-command-palette.png)

![VS Code - C#](images/vscode-context-menu.png)

![VS Code - TypeScript](images/vscode-context-menu-typescript.png)

### MacOS

![Add - API Client from OpenAPI Specification](images/vsmac-add-new-menu.png)

![Enter URL to OpenAPI Specification](images/vsmac-add-new-dialog.png)

![Solution Explorer Context Menus](images/vsmac-generate-code.png)

![NSwag Studio Context Menu](images/vsmac-nswag-studio.png)

## Visual Studio Code

The VS Code extension provides similar functionality through a context menu on JSON and YAML files in the explorer. It executes the `rapicgen` .NET tool to generate code, so you will need to have the .NET SDK installed.

### Installation

Download the VSIX file from the [GitHub releases](https://github.com/christianhelle/apiclientcodegen/releases/latest) page, then in VS Code:
1. Go to Extensions view (Ctrl+Shift+X)
2. Click on the "..." menu in the top right corner
3. Select "Install from VSIX..."
4. Select the downloaded .vsix file

### Features

- Context menu on JSON and YAML files for generating code with various generators
- Supports all code generators: NSwag, Refitter, OpenAPI Generator, Microsoft Kiota, Swagger Codegen CLI, and AutoREST
- Configuration options for namespace and output directory

### Usage

1. Right-click on a Swagger/OpenAPI specification file (JSON or YAML) in the VS Code explorer
2. Select "REST API Client Code Generator" in the context menu
3. Choose one of the available code generators
4. The generated code will open in the editor

# Installation

The Visual Studio extension can be installed directly from Visual Studio 2019 and 2022 via the **Extensions Dialog Box**. The process is best described in the official Microsoft documentation for [Managing extensions for Visual Studio](https://docs.microsoft.com/en-us/visualstudio/ide/finding-and-using-visual-studio-extensions?view=vs-2019&WT.mc_id=DT-MVP-5004822)

## Visual Studio for Mac

This installation process for **Visual Studio for Mac** is currently a bit troublesome as the MonoDevelop Addin Repository is currently not accepting new users so I can't really register and setup my extension.

There are 2 ways of installing my extension on Visual Studio for Mac: Adding a custom extension repository or Installing the **.mpack** file directly from the Extensions Manager

### Adding a new extension repository

Here's what you need to do:

- Open the Visual Studio for Mac **Extension Manager**
- You can do this from the menu **Visual Studio** -> **Extentions**

![Open Extensions Dialog Box](images/vsmac-extensions-menu.png)

- Select the **Gallery** Tab
- Expand the Repository drop down box and select **Manage Repositories**

![Manage Repositories](images/vsmac-extensions-manage-repositories.png)

- Press on the **Add** button to add a new custom extension repository

![Manage Repositories](images/vsmac-extensions-repositories.png)

- Register an online repository at **https://christianhelle.com/vsmac/main.mrep**
- Click **OK**

![Add Repository](images/vsmac-extensions-add-repository.png)

- Now my extension repository is added to the list
- Make sure that this is enabled (indicated by a check box)

![Add Repository Dialog](images/vsmac-extensions-added-repository.png)

- You should now be able to see my extensions from the **Gallery** tab

![Added Repository](images/vsmac-extensions-gallery.png)

- By adding my extension repository you will be able to conveniently update my extension using the Visual Studio for Mac Extension Manager

![Add Repository](images/vsmac-updates.png)

![Add Repository](images/vsmac-update-install.png)

![Add Repository](images/vsmac-update-install-progress.png)

### Installing the **.mpack** file directly

Here's what you need to do:
- Download the latest **.mpack file** from the [Latest Github Release](https://github.com/christianhelle/apiclientcodegen/releases/latest)
- Now from within Visual Studio for Mac you need to launch the **Extensions Dialog Box**. You can do this from the menu **Visual Studio** -> **Extensions**

![Open Extensions Dialog Box](images/vsmac-extensions-menu.png)

- Click on the **Install from File** button

![Manually install .mpack file](images/vsmac-extensions-install.png)

- Browse to the .mpack file and select it. You will be prompted with a confirmation dialog

![Confirm .mpack file install](images/vsmac-extensions-install-confirm.png)

- Click **Install** and restart Visual Studio for Mac
- To Verify that the Add-in was installed you can re-open the Extensions Dialog Box, select the **Installed** tab and expand the **IDE Extensions**. You should be able to see the **REST API Client Code Generator**.

- Uninstalling the Add-in is done in this same dialog box by clicking the **Uninstall** button

![Manually uninstall Add-in](images/vsmac-extensions-uninstall.png)

## IntelliJ Plugin

The IntelliJ plugin provides similar functionality to the VS Code extension through context menus and commands within JetBrains IDEs. It executes the `rapicgen` .NET tool to generate code, so you will need to have the .NET SDK installed.

### IntelliJ Plugin Installation

The plugin is available on the JetBrains Marketplace and can be installed directly from your IDE:

1. Open your JetBrains IDE (IntelliJ IDEA, Rider, WebStorm, etc.)
2. Go to **File** → **Settings** (or **IntelliJ IDEA** → **Preferences** on macOS)
3. Navigate to **Plugins**
4. Search for "REST API Client Code Generator"
5. Click **Install** and restart your IDE

Alternatively, you can install it from the [JetBrains Marketplace](https://plugins.jetbrains.com/plugin/28472-rest-api-client-code-generator).

### IntelliJ Plugin Features

- Context menu on JSON and YAML files for generating code with various generators
- Supports all code generators: NSwag, Refitter, OpenAPI Generator, Microsoft Kiota, Swagger Codegen CLI, and AutoREST
- Multiple language support: C#, TypeScript, and more
- Configuration options for namespace and output directory
- Integration with JetBrains project structure

### IntelliJ Plugin Usage

1. Right-click on a Swagger/OpenAPI specification file (JSON or YAML) in the project explorer
2. Select "Generate REST API Client" from the context menu
3. Choose your preferred code generator and target language
4. Configure the namespace and output options
5. The generated code will be added to your project

### IntelliJ Plugin Screenshots

![Generate C# Client](images/intellij-generate-csharp.png)

![Generate TypeScript Client](images/intellij-generate-typescript.png)

![Generate Refitter Interface](images/intellij-generate-refitter.png)

## Cross Platform Command Line Tool

All custom tools mentioned above are also implemented in a cross platform command line application.

### Requirements

- .NET 6.0 runtime
- Java Runtime Environment
- NPM

### CLI Installation

The tool can be installed as a .NET Core global tool that you can call from the shell / command line

```bash
dotnet tool install --global rapicgen
```

or by following the instructions [here](https://www.nuget.org/packages/rapicgen) to install a specific version of tool

### CLI Usage

Since the tool is published as a .NET Core Tool, it can be launched from anywhere using any command line interface by calling **rapicgen**.
The help information is displayed when not specifying any arguments to **rapicgen**

```
Usage: rapicgen [command] [options]

Options:
  -v|--verbose       Show verbose output
  -?|-h|--help       Show help information.

Commands:
  csharp             Generate C# API clients
  jmeter             Generate Apache JMeter test plans
  openapi-generator  Generate code using OpenAPI Generator (v7.19.0).
                     See supported generators at https://openapi-generator.tech/docs/generators/
  typescript         Generate TypeScript API clients

Run 'rapicgen [command] -?|-h|--help' for more information about a command.
```

Some help information is also provided per command and can be launched by 

```
rapicgen [command name] -?
```

or

```
rapicgen [command name] [sub command name] -?
```

For example:

```
rapicgen csharp -?
``` 

will output this:

```
Generate C# API clients

Usage: rapicgen csharp [command] [options]

Options:
  -?|-h|--help  Show help information.

Commands:
  autorest      AutoRest (v3.0.0-beta.20210504.2)
  kiota         Microsoft Kiota (v1.30.0)
  nswag         NSwag (v14.6.3)
  openapi       OpenAPI Generator (v7.19.0)
  refitter      Refitter (v1.7.3)
  swagger       Swagger Codegen CLI (v3.0.34)

Run 'csharp [command] -?|-h|--help' for more information about a command.

```

and

```
rapicgen csharp autorest -?
```

will output this:

```
Generate Swagger / Open API client using AutoRest

Usage: run autorest [options] <swaggerFile> <namespace> <outputFile>

Arguments:
  swaggerFile   Path to the Swagger / Open API specification file
  namespace     Default namespace to in the generated code
  outputFile    Output filename to write the generated code to. Default is the swaggerFile .cs

Options:
  -?|-h|--help  Show help information
```

or 

```
rapicgen typescript -?
```

will output this:

```
Generate TypeScript API client

Usage: rapicgen typescript [options] <generator> <swaggerFile> <outputPath>

Arguments:
  generator         The tech stack to use for the generated client library
                    Allowed values are: Angular, Aurelia, Axios, Fetch, Inversify, JQuery, NestJS, Node, ReduxQuery,
                    Rxjs.
                    Default value is: Angular.
  swaggerFile       Path to the Swagger / Open API specification file
  outputPath        Output folder to write the generated code to
                    Default value is: typescript-generated-code.

Options:
  -nl|--no-logging  Disables Analytics and Error Reporting
  -?|-h|--help      Show help information.
```

## Usage Examples:

Let's say we have a OpenAPI Specifications document called **Swagger.json**

For starters, we can use the Swagger Petstore spec. Here's an example powershell script for downloading it

```
Invoke-WebRequest -Uri https://petstore.swagger.io/v3/swagger.json -OutFile Swagger.json
```

In case you don't have the CLI tool installed you can install it by

```
dotnet tool install --global rapicgen
```

Here's an example of how to generate code using **AutoRest**

```
rapicgen csharp autorest Swagger.json GeneratedCode ./AutoRestOutput.cs
```

Here's an example of how to generate code using **Kiota**

```
rapicgen csharp kiota Swagger.json GeneratedCode ./OpenApiOutput.cs
```

Here's an example of how to generate code using **NSwag**

```
rapicgen csharp nswag Swagger.json GeneratedCode ./NSwagOutput.cs
```

Here's an example of how to generate code using **Swagger Codegen CLI**

```
rapicgen csharp swagger Swagger.json GeneratedCode ./SwaggerOutput.cs
```

Here's an example of how to generate code using **OpenAPI Generator**

```
rapicgen csharp openapi Swagger.json GeneratedCode ./OpenApiOutput.cs
```

Here's an example of how to generate code **JMeter** test plans

```
rapicgen jmeter Swagger.json
```

Here's an example of how to generate code for **TypeScript**

```
rapicgen typescript Angular Swagger.json
```

#

For tips and tricks on software development, check out [my blog](https://christianhelle.com)

If you find this useful and feel a bit generous then feel free to [buy me a coffee ☕](https://www.buymeacoffee.com/christianhelle)

 readmeEtag: '"7140714731974e2a4cce7bea1aa13c10ea0467e5"' readmeLastModified: Tue, 27 Jan 2026 23:16:00 GMT repositoryId: 166226127 description: >- A collection of Visual Studio code generators for Swagger / OpenAPI specification files created: '2019-01-17T12:58:50Z' updated: '2026-02-05T22:37:40Z' language: C# archived: false stars: 199 watchers: 4 forks: 27 owner: christianhelle logo: https://avatars.githubusercontent.com/u/710400?v=4 license: GPL-3.0 repoEtag: '"8a315f020c72be2903af0878ab62a4087d0bd939c6b944a0a8ffee8d1de95038"' repoLastModified: Thu, 05 Feb 2026 22:37:40 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/neotoolkit/dummy v3: true repositoryMetadata: base64Readme: >- IyBkdW1teQoKWyFbQ0ktaW1nXV1bQ0ktdXJsXQpbIVtwa2ctaW1nXV1bcGtnLXVybF0KWyFbcmVwb3J0Y2FyZC1pbWddXVtyZXBvcnRjYXJkLXVybF0KWyFbY292ZXJhZ2UtaW1nXV1bY292ZXJhZ2UtdXJsXQpbIVt2ZXJzaW9uLWltZ11dW3ZlcnNpb24tdXJsXQoKPGltZyBhbGlnbj0icmlnaHQiIGFsdD0iZHVtbXkiIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL25lb3Rvb2xraXQvLmdpdGh1Yi9tYWluL2Fzc2V0cy9kdW1teV9naF9vcHQtMl9pbWdfNHgucG5nIiB3aWR0aD0iMTUwcHgiIGhlaWdodD0iMTUwcHgiPgoKUnVuIG1vY2sgc2VydmVyIGJhc2VkIG9mZiBhbiBBUEkgY29udHJhY3Qgd2l0aCBvbmUgY29tbWFuZAoKW0R1bW15XShodHRwczovL2V2cm9uZS5jb20vZHVtbXk/dXRtX3NvdXJjZT1naXRodWImdXRtX2NhbXBhaWduPWR1bW15KSBpcyBjcmVhdGVkICYgc3VwcG9ydGVkIGJ5IFtFdnJvbmVdKGh0dHBzOi8vZXZyb25lLmNvbT91dG1fc291cmNlPWdpdGh1YiZ1dG1fY2FtcGFpZ249ZHVtbXkpCgojIyBGZWF0dXJlcwotIFN1cHBvcnRzIGBPcGVuQVBJIDMueGAKCiMjIEluc3RhbGxhdGlvbgpgYGBzaGVsbApnbyBpbnN0YWxsIGdpdGh1Yi5jb20vbmVvdG9vbGtpdC9kdW1teS9jbWQvZHVtbXlAbGF0ZXN0CmBgYAoKIyMgVXNhZ2UKRHVtbXkgY2FuIGhlbHAgeW91IHJ1biBtb2NrIHNlcnZlciBiYXNlZCBvZmYgYW4gQVBJIGNvbnRyYWN0LCB3aGljaCBoZWxwcyBwZW9wbGUgc2VlIGhvdyB5b3VyIEFQSSB3aWxsIHdvcmsgYmVmb3JlIHlvdSBldmVuIGhhdmUgaXQgYnVpbHQuIFJ1biBpdCBsb2NhbGx5IHdpdGggdGhlIGBkdW1teSBzYCBjb21tYW5kIHRvIHJ1biB5b3VyIEFQSSBvbiBhIEhUVFAgc2VydmVyIHlvdSBjYW4gaW50ZXJhY3Qgd2l0aC4KYGBgc2hlbGwKZHVtbXkgcyBvcGVuYXBpLnltbApgYGAKYGBgc2hlbGwKZHVtbXkgcyBodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vbmVvdG9vbGtpdC9kdW1teS9tYWluL2V4YW1wbGVzL2RvY2tlci9vcGVuYXBpLnltbApgYGAKTW9yZSB1c2FnZSBbZXhhbXBsZXNdKGV4YW1wbGVzKQoKIyMgRG9jdW1lbnRhdGlvbgpTZWUgW3RoZXNlIGRvY3NdW3BrZy11cmxdLgoKIyMgTGljZW5zZQpbTUlUIExpY2Vuc2VdKExJQ0VOU0UpLgoKIyMgU3BvbnNvcnMKPHA+CiAgPGEgaHJlZj0iaHR0cHM6Ly9ldnJvbmUuY29tLz91dG1fc291cmNlPWdpdGh1YiZ1dG1fY2FtcGFpZ249bmVvdG9vbGtpdCI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL25lb3Rvb2xraXQvLmdpdGh1Yi9tYWluL2Fzc2V0cy9zcG9uc29yZWRfYnlfZXZyb25lLnN2ZyIKICAgICAgYWx0PSJTcG9uc29yZWQgYnkgRXZyb25lIj4KICA8L2E+CjwvcD4KCltDSS1pbWddOiBodHRwczovL2dpdGh1Yi5jb20vbmVvdG9vbGtpdC9kdW1teS93b3JrZmxvd3MvQ0kvYmFkZ2Uuc3ZnCltDSS11cmxdOiBodHRwczovL2dpdGh1Yi5jb20vbmVvdG9vbGtpdC9kdW1teS9hY3Rpb25zCltwa2ctaW1nXTogaHR0cHM6Ly9wa2cuZ28uZGV2L2JhZGdlL25lb3Rvb2xraXQvZHVtbXkKW3BrZy11cmxdOiBodHRwczovL3BrZy5nby5kZXYvZ2l0aHViLmNvbS9uZW90b29sa2l0L2R1bW15CltyZXBvcnRjYXJkLWltZ106IGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9uZW90b29sa2l0L2R1bW15CltyZXBvcnRjYXJkLXVybF06IGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvbmVvdG9vbGtpdC9kdW1teQpbY292ZXJhZ2UtaW1nXTogaHR0cHM6Ly9jb2RlY292LmlvL2doL25lb3Rvb2xraXQvZHVtbXkvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnCltjb3ZlcmFnZS11cmxdOiBodHRwczovL2NvZGVjb3YuaW8vZ2gvbmVvdG9vbGtpdC9kdW1teQpbdmVyc2lvbi1pbWddOiBodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3JlbGVhc2UvbmVvdG9vbGtpdC9kdW1teQpbdmVyc2lvbi11cmxdOiBodHRwczovL2dpdGh1Yi5jb20vbmVvdG9vbGtpdC9kdW1teS9yZWxlYXNlcwo= readmeEtag: '"f03ee0da18daed2d3289f8396f6b6742bf664ec7"' readmeLastModified: Mon, 25 Jul 2022 15:10:04 GMT repositoryId: 427261627 description: Run mock server based off an API contract with one command created: '2021-11-12T06:54:04Z' updated: '2025-07-30T10:12:50Z' language: Go archived: true stars: 177 watchers: 2 forks: 9 owner: neotoolkit logo: https://avatars.githubusercontent.com/u/94162314?v=4 license: MIT repoEtag: '"b1b433759bd38086fe0e63cca6becd0756afabf626b3da81f21d1c6c71e7f03b"' repoLastModified: Wed, 30 Jul 2025 10:12:50 GMT foundInMaster: true category: Testing id: 008c4fd437f4f20d365f1f13b08a2f6f oldLocations: - https://github.com/go-dummy/dummy - source: - openapi3 tags - openapi31 tags repository: https://github.com/redocly/openapi-starter v3: true repositoryMetadata: base64Readme: >- # OpenAPI Definition Starter

## How to use this starter

![Click use template button](https://user-images.githubusercontent.com/3975738/92927304-12e35d80-f446-11ea-9bd3-a0f8a69792d0.png)

## Working on your OpenAPI Definition

### Install

1. Install [Node JS](https://nodejs.org/).
2. Clone this repo and run `npm install` in the repo root.

### Usage

#### `npm start`
Starts the reference docs preview server.

#### `npm run build`
Bundles the definition to the dist folder.

#### `npm test`
Validates the definition.

## Contribution Guide

Below is a sample contribution guide. The tools
in the repository don't restrict you to any
specific structure. Adjust the contribution guide
to match your own structure. However, if you
don't have a structure in mind, this is a
good place to start.

Update this contribution guide if you
adjust the file/folder organization.

The `.redocly.yaml` controls settings for various
tools including the lint tool and the reference
docs engine.  Open it to find examples and
[read the docs](https://redocly.com/docs/cli/configuration/)
for more information.


### Schemas

#### Adding Schemas

1. Navigate to the `openapi/components/schemas` folder.
2. Add a file named as you wish to name the schema.
3. Define the schema.
4. Refer to the schema using the `$ref` (see example below).

##### Example Schema
This is a very simple schema example:
```yaml
type: string
description: The resource ID. Defaults to UUID v4
maxLength: 50
example: 4f6cf35x-2c4y-483z-a0a9-158621f77a21
```
This is a more complex schema example:
```yaml
type: object
properties:
  id:
    description: The customer identifier string
    readOnly: true
    allOf:
      - $ref: ./ResourceId.yaml
  websiteId:
    description: The website's ID
    allOf:
      - $ref: ./ResourceId.yaml
  paymentToken:
    type: string
    writeOnly: true
    description: |
      A write-only payment token; if supplied, it will be converted into a
      payment instrument and be set as the `defaultPaymentInstrument`. The
      value of this property will override the `defaultPaymentInstrument`
      in the case that both are supplied. The token may only be used once
      before it is expired.
  defaultPaymentInstrument:
    $ref: ./PaymentInstrument.yaml
  createdTime:
    description: The customer created time
    allOf:
      - $ref: ./ServerTimestamp.yaml
  updatedTime:
    description: The customer updated time
    allOf:
      - $ref: ./ServerTimestamp.yaml
  tags:
    description: A list of customer's tags
    readOnly: true
    type: array
    items:
      $ref: ./Tags/Tag.yaml
  revision:
    description: >
      The number of times the customer data has been modified.

      The revision is useful when analyzing webhook data to determine if the
      change takes precedence over the current representation.
    type: integer
    readOnly: true
  _links:
    type: array
    description: The links related to resource
    readOnly: true
    minItems: 3
    items:
      anyOf:
        - $ref: ./Links/SelfLink.yaml
        - $ref: ./Links/NotesLink.yaml
        - $ref: ./Links/DefaultPaymentInstrumentLink.yaml
        - $ref: ./Links/LeadSourceLink.yaml
        - $ref: ./Links/WebsiteLink.yaml
  _embedded:
    type: array
    description: >-
      Any embedded objects available that are requested by the `expand`
      querystring parameter.
    readOnly: true
    minItems: 1
    items:
      anyOf:
        - $ref: ./Embeds/LeadSourceEmbed.yaml

```

If you have an JSON example, you can convert it to JSON schema using Redocly's [JSON to JSON schema tool](https://redocly.com/tools/json-to-json-schema/).

##### Using the `$ref`

Notice in the complex example above the schema definition itself has `$ref` links to other schemas defined.

Here is a small excerpt with an example:

```yaml
defaultPaymentInstrument:
  $ref: ./PaymentInstrument.yaml
```

The value of the `$ref` is the path to the other schema definition.

You may use `$ref`s to compose schema from other existing schema to avoid duplication.

You will use `$ref`s to reference schema from your path definitions.

#### Editing Schemas

1. Navigate to the `openapi/components/schemas` folder.
2. Open the file you wish to edit.
3. Edit.

### Paths

#### Adding a Path

1. Navigate to the `openapi/paths` folder.
2. Add a new YAML file named like your URL endpoint except replacing `/` with `_` (or whichever character you prefer) and putting path parameters into curly braces like `{example}`.
3. Add the path and a ref to it inside of your `openapi.yaml` file inside of the `openapi` folder.

Example addition to the `openapi.yaml` file:
```yaml
'/customers/{id}':
  $ref: './paths/customers_{id}.yaml'
```

Here is an example of a YAML file named `customers_{id}.yaml` in the `paths` folder:

```yaml
get:
  tags:
    - Customers
  summary: Retrieve a list of customers
  operationId: GetCustomerCollection
  description: |
    You can have a markdown description here.
  parameters:
    - $ref: ../components/parameters/collectionLimit.yaml
    - $ref: ../components/parameters/collectionOffset.yaml
    - $ref: ../components/parameters/collectionFilter.yaml
    - $ref: ../components/parameters/collectionQuery.yaml
    - $ref: ../components/parameters/collectionExpand.yaml
    - $ref: ../components/parameters/collectionFields.yaml
  responses:
    '200':
      description: A list of Customers was retrieved successfully
      headers:
        Rate-Limit-Limit:
          $ref: ../components/headers/Rate-Limit-Limit.yaml
        Rate-Limit-Remaining:
          $ref: ../components/headers/Rate-Limit-Remaining.yaml
        Rate-Limit-Reset:
          $ref: ../components/headers/Rate-Limit-Reset.yaml
        Pagination-Total:
          $ref: ../components/headers/Pagination-Total.yaml
        Pagination-Limit:
          $ref: ../components/headers/Pagination-Limit.yaml
        Pagination-Offset:
          $ref: ../components/headers/Pagination-Offset.yaml
      content:
        application/json:
          schema:
            type: array
            items:
              $ref: ../components/schemas/Customer.yaml
        text/csv:
          schema:
            type: array
            items:
              $ref: ../components/schemas/Customer.yaml
    '401':
      $ref: ../components/responses/AccessForbidden.yaml
  x-code-samples:
    - lang: PHP
      source:
        $ref: ../code_samples/PHP/customers/get.php
post:
  tags:
    - Customers
  summary: Create a customer (without an ID)
  operationId: PostCustomer
  description: Another markdown description here.
  requestBody:
    $ref: ../components/requestBodies/Customer.yaml
  responses:
    '201':
      $ref: ../components/responses/Customer.yaml
    '401':
      $ref: ../components/responses/AccessForbidden.yaml
    '409':
      $ref: ../components/responses/Conflict.yaml
    '422':
      $ref: ../components/responses/InvalidDataError.yaml
  x-code-samples:
    - lang: PHP
      source:
        $ref: ../code_samples/PHP/customers/post.php
```

You'll see extensive usage of `$ref`s in this example to different types of components including schemas.

You'll also notice `$ref`s to code samples.

### Code samples

Automated code sample generations is enabled in the Redocly configuration file. Add manual code samples by the following process:

1. Navigate to the `openapi/code_samples` folder.
2. Navigate to the `<language>` (e.g. PHP) sub-folder.
3. Navigate to the `path` folder, and add ref to the code sample.

You can add languages by adding new folders at the appropriate path level.

More details inside the `code_samples` folder README.
 readmeEtag: '"a8de8cdd75c9b76475ef4aedb54a444ab2d0a64a"' readmeLastModified: Thu, 01 Aug 2024 06:35:10 GMT repositoryId: 294444776 description: OpenAPI starter repository created: '2020-09-10T15:11:31Z' updated: '2026-01-27T17:29:06Z' language: C# archived: false stars: 209 watchers: 15 forks: 131 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"51ea764c7989d56cff7799955081755027eab97b6f3e8644ab98aab0089d79d7"' repoLastModified: Tue, 27 Jan 2026 17:29:06 GMT foundInMaster: true v3_1: true category: Parsers id: 9f59fa22b947a112b53c92bbf81c95c4 - source: openapi3 tags repository: https://github.com/ccouzens/keycloak-openapi v3: true repositoryMetadata: base64Readme: >- IyBLZXljbG9hay1BZG1pbi1PcGVuQVBJLURlZmluaXRpb24KCltPcGVuQVBJIGRlZmluaXRpb25zXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikgZm9yCltLZXljbG9ha10oaHR0cHM6Ly93d3cua2V5Y2xvYWsub3JnLykncyBBZG1pbiBBUEkuCgpUaGVzZSBjYW4gYmUgdXNlZCB0byBnZW5lcmF0ZSBsaWJyYXJpZXMgZm9yIGludGVyYWN0aW5nIHdpdGggS2V5Y2xvYWsgZnJvbSBhbnkKbWFpbnN0cmVhbSBwcm9ncmFtbWluZyBsYW5nYXVnZS4KCk9wZW5BUEkgZGVmaW5pdGlvbnMgYXJlIHNvbWV0aW1lcyBrbm93biBieSB0aGVpciBwcmV2aW91cyBuYW1lIG9mIFN3YWdnZXIKc3BlY2lmaWNhdGlvbnMuCgpbS2V5Y2xvYWsgMjIuMC4wIEFkbWluIEFQSV0oLi9rZXljbG9hay8yMi4wLjAuanNvbikKCltLZXljbG9hayAxMy4wIEFkbWluIEFQSSB3aXRoIHBhdGNoZXNdKC4va2V5Y2xvYWsvMTMuMC1wYXRjaGVkLmpzb24pCgpbS2V5Y2xvYWsgMTIuMCBBZG1pbiBBUEkgd2l0aCBwYXRjaGVzXSguL2tleWNsb2FrLzEyLjAtcGF0Y2hlZC5qc29uKQoKW090aGVyIHZlcnNpb25zXSguL2tleWNsb2FrLykKCiMjIEFsdGVybmF0aXZlcwoKREFIQUcgUmVjaHRzc2VydmljZXMgQUcKW3Byb3ZpZGUgT3BlbkFQSSBkZWZpbml0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL2RhaGFnLWFnL2tleWNsb2FrLW9wZW5hcGkpIGZvcgpyZWNlbnQgdmVyc2lvbnMgb2YgS2V5Y2xvYWsgd2hpY2ggYXJlIG1vcmUgY29tcGxldGUgdGhhbiB3aGF0J3MgaGVyZS4KCiMjIEFjY291bnQgQVBJCgpLZXljbG9hayBoYXMgYW4gdW5kb2N1bWVudGVkIFtBY2NvdW50IEFQSV0oaHR0cHM6Ly9naXRodWIuY29tL2tleWNsb2FrL2tleWNsb2FrL2Jsb2IvbWFpbi9zZXJ2aWNlcy9zcmMvbWFpbi9qYXZhL29yZy9rZXljbG9hay9zZXJ2aWNlcy9yZXNvdXJjZXMvYWNjb3VudC9BY2NvdW50UmVzdFNlcnZpY2UuamF2YSkgdGhhdCBpcyB1c2VkIGJ5IHRoZSBpbnRlcm5hbCBhY2NvdW50IGNvbnNvbGUgdG8gZGlzcGxheSBhbmQgdXBkYXRlIHVzZXIgaW5mb3JtYXRpb24uIFRoaXMgY2FuIGJlIHVzZWQgZm9yIGJ1aWxkaW5nIGEgY3VzdG9tIGFjY291bnQgY29uc29sZSwgb3IgYnVpbGRpbmcgYWNjb3VudCBpbmZvcm1hdGlvbiBhbmQgZWRpdGluZyBpbnRvIGFuIGFwcGxpY2F0aW9uLiBbQHhncF0oaHR0cHM6Ly9naXRodWIuY29tL3hncCkgaGFzIGNyZWF0ZWQgYSBbR2lzdCBvZiBPcGVuQVBJIGRlZmluaXRpb25zIG9mIHRoZSBBY2NvdW50IEFQSV0oaHR0cHM6Ly9naXN0LmdpdGh1Yi5jb20veGdwLzJkNzdjYmViYzYxNjQxNjBmYWFlNmFhNzdkMTI3YTU3KS4KCiMjIEdlbmVyYWwgVXNhZ2UKClR5cGljYWxseSB0byB1c2UgS2V5Y2xvYWsncyBhZG1pbiBSZXN0IEFQSSwgeW91IGZpcnN0IGdldCBhIHRva2VuIGZyb20gYSByZWFsbS4KVGhpcyBpcyBkb25lIHVzaW5nIHRoZSBPQXV0aDIgcHJvdG9jb2wuCgpgYGBiYXNoCmJlYXJlcl90b2tlbj0kKGN1cmwgLVggUE9TVCAnaHR0cDovL2xvY2FsaG9zdDo4MDgwL2F1dGgvcmVhbG1zL21hc3Rlci9wcm90b2NvbC9vcGVuaWQtY29ubmVjdC90b2tlbicgXAogICAgIC0tZGF0YS11cmxlbmNvZGUgJ3VzZXJuYW1lPWFkbWluLXVzZXInIFwKICAgICAtLWRhdGEtdXJsZW5jb2RlICdwYXNzd29yZD1hZG1pbi1wYXNzd29yZCcgXAogICAgIC0tZGF0YS11cmxlbmNvZGUgJ2dyYW50X3R5cGU9cGFzc3dvcmQnIFwKICAgICAtLWRhdGEtdXJsZW5jb2RlICdjbGllbnRfaWQ9YWRtaW4tY2xpJyBcCiAgICAgIHwganEgLXIgJy5hY2Nlc3NfdG9rZW4nKQpgYGAKClRoZSByZWFsbSB1c2VkIGZvciB0aGUgdG9rZW4gaXMgbm90IG5lY2Vzc2FyaWx5IHRoZSByZWFsbSB1c2VkIGluIHRoZSBBUEkuCgpUaGUgZW5kcG9pbnRzIGluIHRoaXMgc3BlY2lmaWNhdGlvbiBjYW4gdGhlbiBiZSB1c2VkIHdpdGggYSBiYXNlIHVybCBvZiB0aGUgZm9ybQpgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2F1dGgvYWRtaW4vcmVhbG1zYCBhbmQgdGhlIGFib3ZlIGJlYXJlciB0b2tlbi4KClBsZWFzZSBzZWUgdGhlIFtleGFtcGxlIGFwcF0oZXhhbXBsZV9hcHAvc3JjL2luZGV4LnRzKS4KCiMjIEhlbHAgd2FudGVkCgpUaGUgZGVmaW5pdGlvbnMgYXJlIGNvbXB1dGVyIGdlbmVyYXRlZCwgYnV0Cltjb3VsZCBkbyB3aXRoIGEgaHVtYW4ncyBpbnB1dF0oaHR0cHM6Ly9naXRodWIuY29tL2Njb3V6ZW5zL2tleWNsb2FrLW9wZW5hcGkvaXNzdWVzLzEwKS4KCltQbGVhc2UgcHJvdmlkZSBhZGRpdGlvbmFsIGV4YW1wbGUgYXBwc10oaHR0cHM6Ly9naXRodWIuY29tL2Njb3V6ZW5zL2tleWNsb2FrLW9wZW5hcGkvaXNzdWVzLzE0KS4KCiMjIEFkZGluZyBuZXcgdmVyc2lvbnMKClRvIGFkZCBhIG5ldyB2ZXJzaW9uIG9mIEtleWNsb2FrIHBlcmZvcm0gdGhlc2Ugc3RlcHM6CgoxLiBNb2RpZnkgdGhlIFtNYWtlZmlsZV0oLi9NYWtlZmlsZSkgd2l0aCB0aGUKICAgW25ldyB2ZXJzaW9uXShodHRwczovL2dpdGh1Yi5jb20vY2NvdXplbnMva2V5Y2xvYWstb3BlbmFwaS9jb21taXQvNmY3MTc1MzNiMTNjZDA3YmFlM2M0YWExZjU1OTA3ZmI0MDMzODU0MCNkaWZmLTc2ZWQwNzRhOTMwNWMwNDA1NGNkZWJiOWU5YWFkMmQ4MTgwNTJiMDcwOTFkZTFmMjBjYWQwYmJhYzM0ZmZiNTIpLgoyLiBWZXJpZnkgeW91IGhhdmUgaW5zdGFsbGVkIHRoZSBbcmVxdWlyZW1lbnRzXSgjcmVxdWlyZW1lbnRzKS4KMy4gUnVuIHRoZSBjb21tYW5kIGBtYWtlYC4gSXQgd2lsbCBhdXRvbWF0aWNhbGx5OgogICAxLiBkb3dubG9hZCB0aGUgZG9jdW1lbnRhdGlvbiBpbiBIVE1MIGZvcm1hdAogICAyLiBjb21waWxlIHRoZSBbdHJhbnNmb3JtZXJdKCNrZXljbG9hay1vcGVuYXBpLXRyYW5zZm9ybWVyKQogICAzLiBjcmVhdGUgdGhlIEpTT04gZGVmaW5pdGlvbgogICA0LiBjcmVhdGUgdGhlIFlNTCBkZWZpbml0aW9uCjQuIENvbW1pdCB5b3VyIGNoYW5nZXMgYW5kIG9wZW4gYSBwdWxsIHJlcXVlc3QuCgojIyBOb3RhYmxlIENoYW5nZXMKCjIwMjEgRWFybHkgLSBSZW5hbWVkIG1hc3RlciBicmFuY2ggdG8gbWFpbi4gQW55b25lIGZldGNoaW5nIHRoZSBzY2hlbWEgZGlyZWN0bHkKZnJvbSB0aGUgYnJhbmNoIHNob3VsZCB1cGRhdGUgdGhlaXIgcmVmZXJlbmNlLgoKMjAyMCBNYXkgLSBBZGRlZCB0YWdzIHRvIG9wZXJhdGlvbnMuIFVzZXJzIG9mIE9wZW5BUEkgR2VuZXJhdG9yIGdlbmVyYXRlZApjbGllbnRzIHdpbGwgbmVlZCB0byBzd2FwIGZyb20gdXNpbmcgdGhlIGBEZWZhdWx0QXBpYCBjbGFzcyB0byBtdWx0aXBsZSBjbGFzc2VzCnN1Y2ggYXMgYFVzZXJzQXBpYCBvciBgQ2xpZW50c0FwaWAuCgojIyBFeGFtcGxlIEFwcAoKW1RoaXMgZXhhbXBsZV0oLi9leGFtcGxlX2FwcC8pIHVzZXMKW29wZW5hcGktZ2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKSB0byBidWlsZAphbiBBUEktY2xpZW50LiBUaGUgYXBwIGluIHRoZSBleGFtcGxlIGlzIHZlcnkgc2ltcGxlIGFuZCBvbmx5IGZldGNoZXMgYSBsaXN0IG9mCktleWNsb2FrLWNsaWVudHMuCgpgYGBiYXNoCiMgQ3JlYXRlIGEga2V5Y2xvYWsgaW5zdGFuY2UgdG8gcnVuIGFnYWluc3QKZG9ja2VyIGNvbnRhaW5lciBydW4gLS1ybSAtZSBLRVlDTE9BS19VU0VSPWFkbWluLXVzZXIgLWUgS0VZQ0xPQUtfUEFTU1dPUkQ9YWRtaW4tcGFzc3dvcmQgLXAgODA4MDo4MDgwIGRvY2tlci5pby9qYm9zcy9rZXljbG9hazoxMy4wLjAKIyBPcGVuIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC8gYW5kIHdhaXQgZm9yIGtleWNsb2FrIHRvIHN0YXJ0IHVwCgojIEluIGEgc2Vjb25kIHRlcm1pbmFsIHJ1bgpjZCBleGFtcGxlX2FwcApucG0gaW5zdGFsbApucG0gcnVuIGdlbmVyYXRlLWNsaWVudApucG0gcnVuIGNvbXBpbGUKbm9kZSBkaXN0L2luZGV4LmpzCiMgcHJpbnRzIG91dAojIFRoZSBkZWZhdWx0IGNsaWVudHM6CiMgICBhY2NvdW50CiMgICBhY2NvdW50LWNvbnNvbGUKIyAgIGFkbWluLWNsaQojICAgYnJva2VyCiMgICBtYXN0ZXItcmVhbG0KIyAgIHNlY3VyaXR5LWFkbWluLWNvbnNvbGUKYGBgCgpJbiBnZW5lcmFsLCBjbGllbnRzIGNhbiBiZSBnZW5lcmF0ZWQgYnkgcnVubmluZyBhIGNvbW1hbmQgc2ltaWxhciB0bzoKCmBgYGJhc2gKb3BlbmFwaS1nZW5lcmF0b3IgZ2VuZXJhdGUgLWkgJ2h0dHBzOi8vZ2l0aHViLmNvbS9jY291emVucy9rZXljbG9hay1vcGVuYXBpL3Jhdy9tYWluL2tleWNsb2FrLzIwLjAuMy5qc29uJyAtZyAndHlwZXNjcmlwdC1heGlvcycgLW8gJ3NyYy9rZXljbG9hay1jbGllbnQnCmBgYAoKb3IgdXNpbmcgZG9ja2VyLCBpZiB5b3UgY2FuJ3QgaW5zdGFsbCBgb3BlbmFwaS1nZW5lcmF0b3JgOgoKYGBgYmFzaApkb2NrZXIgcnVuIC0tcm0gLS11c2VyICQoaWQgLXUpOiQoaWQgLWcpIC12ICRQV0Q6L2xvY2FsIG9wZW5hcGl0b29scy9vcGVuYXBpLWdlbmVyYXRvci1jbGkgZ2VuZXJhdGUgLWkgJ2h0dHBzOi8vZ2l0aHViLmNvbS9jY291emVucy9rZXljbG9hay1vcGVuYXBpL3Jhdy9tYWluL2tleWNsb2FrLzIwLjAuMy5qc29uJyAtZyB0eXBlc2NyaXB0LWF4aW9zIC1vIC9sb2NhbC9zcmMva2V5Y2xvYWstY2xpZW50CmBgYAojIyBLZXljbG9hayBPcGVuQVBJIFRyYW5zZm9ybWVyCgpBbGwgdGhlIE9wZW5BUEkgZGVmaW5pdGlvbnMgYXJlIGdlbmVyYXRlZCBmcm9tIHRoZSBwdWJsaXNoZWQgSFRNTCBkb2N1bWVudGF0aW9uLgpUaGlzIHRvb2wgdHJhbnNmb3JtcyB0aGUgSFRNTCBkb2N1bWVudGF0aW9uIGludG8gT3BlbkFQSSBkZWZpbml0aW9ucy4KCkhUTUwgb2YgdGhlIGRvY3VtZW50YXRpb24gaXMgaW5wdXR0ZWQgdGhyb3VnaCBgc3RkaW5gIGFuZCB0aGUgSlNPTiBvZiB0aGUKZGVmaW5pdGlvbiBpcyBvdXRwdXR0ZWQgdGhyb3VnaCBgc3Rkb3V0YC4KCiMjIFJlcXVpcmVtZW50cwoKQnVpbGRpbmcgUmVkIEhhdCBTaW5nbGUgU2lnbi1PbiBkZWZpbml0aW9ucyBvciByZWJ1aWxkaW5nIEtleWNsb2FrJ3MgZGVmaW5pdGlvbnMKcmVxdWlyZXM6CgotIGBtYWtlYAotIFtgUnVzdGBdKGh0dHBzOi8vd3d3LnJ1c3QtbGFuZy5vcmcvdG9vbHMvaW5zdGFsbCkKLSBbYHlxYF0oaHR0cHM6Ly9naXRodWIuY29tL21pa2VmYXJhaC95cS8pCgpCdWlsZGluZyB0aGUgdHJhbnNmb3JtZXIgcmVxdWlyZXMKW2BSdXN0YF0oaHR0cHM6Ly93d3cucnVzdC1sYW5nLm9yZy90b29scy9pbnN0YWxsKS4KCkJ1aWxkaW5nIHRoZSBleGFtcGxlIGFwcCByZXF1aXJlcyBgTm9kZWAgYW5kIGBKYXZhYC4KClJ1bm5pbmcgdGhlIGtleWNsb2FrIGluc3RhbmNlIHRoYXQgcGFpcnMgd2l0aCB0aGUgZXhhbXBsZSBhcHAgcmVxdWlyZXMgYERvY2tlcmAuCgojIyBMaWNlbnNpbmcKClRoZSBPcGVuQVBJIGRlZmluaXRpb25zIGFyZSBbQXBhY2hlIDIuMCBsaWNlbnNlZF0oLi9rZXljbG9hay9MSUNFTlNFLnR4dCkuCgpUaGUgdHJhbnNmb3JtZXIgaXMgW01JVCBsaWNlbnNlZF0oa2V5Y2xvYWstb3BlbmFwaS10cmFuc2Zvcm1lci9MSUNFTlNFKS4KClRoZSBleGFtcGxlIGFwcCBoYXMgYSBbQ0MwIHdhaXZlcl0oZXhhbXBsZV9hcHAvV0FJVkVSKSAoaGFzIGhhZCBpdHMgY29weXJpZ2h0CndhaXZlZCkuCg== readmeEtag: '"b3f1a7082b583e11c50e79884ecde04aa8673c96"' readmeLastModified: Thu, 29 Feb 2024 19:59:31 GMT repositoryId: 185045075 description: OpenAPI definitions for Keycloak's Admin API created: '2019-05-05T14:54:51Z' updated: '2025-12-07T15:37:40Z' language: Rust archived: false stars: 181 watchers: 8 forks: 67 owner: ccouzens logo: https://avatars.githubusercontent.com/u/241046?v=4 repoEtag: '"78dde2ca2ab397d5b608512e29d9c8bc4172c3e1c7dacf904af5cadf6305adbf"' repoLastModified: Sun, 07 Dec 2025 15:37:40 GMT foundInMaster: true category: Server Implementations id: e7c4d0adf91d49ed856cc5a453ccfdf0 - source: openapi3 tags repository: https://github.com/pace/bricks v3: true repositoryMetadata: base64Readme: >- IyBQQUNFIEJyaWNrcyAhW0RlcHJlY2F0ZWRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZGVwcmVjYXRlZC1GRjAwMDApICFbR29sYW5nIENJXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2dvbGFuZy0wMEFERDgpCgohW10oYXJ0d29yay9QQUNFLUJyaWNrc19IZWFkZXJfTGlnaHRCYWNrZ3JvdW5kLnBuZykKCk9waW5pb25hdGVkIG1pY3Jvc2VydmljZSBraXQgdG8gaGVscCBkZXZlbG9wZXJzIHRvIGJ1aWxkIG1pY3Jvc2VydmljZXMgd2l0aCBnby4KCj4gWyFDQVVUSU9OXQo+ICMjIFByb2plY3QgU3RhdHVzOiBEZXByZWNhdGVkL0FyY2hpdmVkCj4KPiAhW0RlcHJlY2F0ZWRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZGVwcmVjYXRlZC1GRjAwMDApICFbQXJjaGl2ZWRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvYXJjaGl2ZWQtMUVDQ0YwKQo+Cj4gKipBcyBvZiBKYW51YXJ5IDIwMjYsIHRoaXMgcHJvamVjdCBoYXMgcmVhY2hlZCBpdHMgZW5kLW9mLWxpZmUgYW5kIGlzIG5vIGxvbmdlciBiZWluZyBhY3RpdmVseSBkZXZlbG9wZWQuIFdlIGFyZSB1bmFibGUgdG8gcHJvdmlkZSBzdXBwb3J0LCBzZWN1cml0eSB1cGRhdGVzLCBvciBidWcgZml4ZXMuIERldmVsb3BlcnMgbG9va2luZyBmb3Igc2ltaWxhciBmdW5jdGlvbmFsaXR5IGFyZSBlbmNvdXJhZ2VkIHRvIHNlZWsgYWN0aXZlbHkgbWFpbnRhaW5lZCBhbHRlcm5hdGl2ZXMuKioKCiMjIE9waW5pb25zCgpBIHBhY2UvYnJpY2tzIG1pY3Jvc2VydmljZSBpczoKCiogYnVpbHQgYXMgYSAqKmRvY2tlciBjb250YWluZXIqKgoqIGRlcGxveWVkIGludG8gYSAqKmt1YmVybmV0ZXMgY2x1c3RlcioqCiAgKiBzdXBwb3J0IGZvciB0aGUgKip0ZXJtaW5hdGlvbiBsb2cqKgoqIGNvbmZpZ3VyZWQgdXNpbmcgKiplbnZpcm9ubWVudCB2YXJpYWJsZXMqKiAoW1RXRUxWRS1GQUNUT1IgQVBQXShodHRwczovLzEyZmFjdG9yLm5ldC8pKQoqIG1vbml0b3JlZCB1c2luZyAqKnByb21ldGhldXMqKgoqIHJlcG9ydHMgZXJyb3JzIHRvICoqc2VudHJ5KioKKiBzYW1wbGVzIHRyYWNlcyB0byAqKnNlbnRyeSoqCiogKipsb2dzKiogdG8gc3Rkb3V0IHVzaW5nIGpzb24gZGVwbG95ZWQgKiprdWJlcm5ldGVzKiogb3RoZXJ3aXNlIGh1bWFuIHJlYWRhYmxlCiogb2ZmZXJzICoqaGVhbHRoKiogZW5kcG9pbnRzCiogYnVpbHQtaW4gcmVkYWN0aW9uIG9mIEpXVHMgYW5kIGNhcmQgc2NoZW1lcwoqIGNvbm5lY3RzIHRvIGJhY2tlbmQgc2VydmljZXMKICAqICoqcG9zdGdyZXMqKiAobG9nZ2luZywgbWV0cmljcywgdHJhY2luZywgaGVhbHRoKQogICogKipyZWRpcyoqIChsb2dnaW5nLCBtZXRyaWNzLCB0cmFjaW5nLCBoZWFsdGgpCiAgKiAqKnF1ZXVlKiogdmlhIHJlZGlzIChsb2dnaW5nLCBtZXRyaWNzLCB0cmFjaW5nLCBoZWFsdGgpCiAgKiAqKmh0dHAqKiAobG9nZ2luZywgbWV0cmljcywgdHJhY2luZywgcmV0cmllcykKICAqICoqczMqKiB2aWEgaHR0cCAobG9nZ2luZywgbWV0cmljcywgdHJhY2luZywgaGVhbHRoKQogICogKipjb3VjaGRiKiogdmlhIGh0dHAgKGxvZ2dpbmcsIG1ldHJpY3MsIHRyYWNpbmcsIHJldHJpZXMsIGhlYWx0aCkKICAqICoqZ3JwYyoqIChsb2dnaW5nLCBtZXRyaWNzLCB0cmFjaW5nLCByZXRyaWVzKQoqIHByb3ZpZGVzIHR3byBjb21tYW5kcyAqKmNvbnRyb2wqKiBhbmQgKipkYWVtb24qKgoqIHByb3ZpZGVzIGEgKipSRVNUZnVsKiogQVBJCiAgKiBjb2RlIGlzIGdlbmVyYXRlZCBmcm9tIHRoZSAqKk9wZW5BUEl2MyoqIHNwZWMKICAqIGF1dGhlbnRpY2F0ZWQgdmlhICoqT0F1dGgyKioKICAqIGVuY29kZWQgdXNpbmcgKipbanNvbjphcGldKGh0dHBzOi8vanNvbmFwaS5vcmcvKSoqCiAgKiB0aGF0IHN1cHBvcnRzICoqbG9nZ2luZyoqLCAqKnRyYWNpbmcqKiBhbmQgKiptZXRyaWNzKioKKiBvcHRpb25hbGx5IHByb3ZpZGVzIGEgKipHUlBDKiogQVBJCiAgKiBjb2RlIGlzIGdlbmVyYXRlZCBmcm9tIHRoZSAqKnByb3RvYyoqIHNwZWMKICAqIHRoYXQgc3VwcG9ydHMgKipsb2dnaW5nKiosICoqdHJhY2luZyoqIGFuZCAqKm1ldHJpY3MqKgoKIyMgSW5zdGFsbAoKICAgIGdvIGdldCBnaXRodWIuY29tL3BhY2UvYnJpY2tzL2NtZC9wYgoKIyMgVXNhZ2UKCiAgICBwYiAtaAoKIyMgQ29udHJpYnV0aW5nCiAKUmVhZCBvdXIgW2NvbnRyaWJ1dG9ycyBndWlkZV0oQ09OVFJJQlVUSU5HLm1kKS4KCiMjIFJlcXVpcmVtZW50cwoKKiBBIHdvcmtpbmcgZ28gaW5zdGFsbGF0aW9uCiogQSB3b3JraW5nIGdpdCBpbnN0YWxsYXRpb24KCiMjIFRlc3RpbmcKCiogVXNlIGBtYWtlIHRlc3RgIHRvIHRlc3Qgd2l0aG91dCBkZXBlbmRlbmNpZXMKKiBVc2UgYGRvY2tlci1jb21wb3NlIHJ1biB0ZXN0c2VydmVyIG1ha2UgaW50ZWdyYXRpb25gIHRvIHRlc3Qgd2l0aCBkZXBlbmRlbmNpZXMKKiBVc2UgYG1ha2UgdGVzdHNlcnZlcmAgdG8gc3RhcnQgYSB0ZXN0c2VydmVyIHRoYXQgd2lsbCBiZSBzdGFydGVkIHdpdGggZGVwZW5kZW5jaWVzLiBJbiBvcmRlciB0byB1cGRhdGUgdGhlIHNlcnZlciBvbmUgbmVlZCB0byBgZG9ja2VyLWNvbXBvc2UgcmVzdGFydCB0ZXN0c2VydmVyYAoKIyMgRW52aXJvbm1lbnQgdmFyaWFibGVzIGZvciB0aGUgcGIgY29tbWFuZAoKfCBWYXJpYWJsZSB8IERlc2NyaXB0aW9uIHwKfC18LXwKfCBgUEFDRV9CUklDS1NfRURJVE9SYCB8wqAgVGhlIHBhdGggdG8gdGhlIGVkaXRvciB0aGF0IHNob3VsZCBiZSB1c2VkIGZvciBvcGVuaW5nIGEgcHJvamVjdC4gRGVmYXVsdHMgdG8gYCRFRElUT1JgLiB8CnwgYFBBQ0VfQlJJQ0tTX1BBVEhgIHwgVGhlIHBhdGggd2hlcmUgbmV3IHByb2plY3Qgc2hvdWxkIGJlIGNyZWF0ZWQuIERlZmF1bHRzIHRvIGAkSE9NRS9QQUNFYC4gfAo= readmeEtag: '"4058f3e44ebf778879f5ca87c614cd80341a633c"' readmeLastModified: Wed, 14 Jan 2026 10:38:05 GMT repositoryId: 166778187 description: A standard library for microservices. created: '2019-01-21T08:41:32Z' updated: '2026-01-28T15:13:40Z' language: Go archived: false stars: 177 watchers: 8 forks: 20 owner: pace logo: https://avatars.githubusercontent.com/u/16001479?v=4 license: MIT repoEtag: '"ff95f44cc67fbc04a5ed3af8b2bce5e7c3ede1f337f5d57fdce321c25244c49d"' repoLastModified: Wed, 28 Jan 2026 15:13:40 GMT foundInMaster: true category: Server id: e6dd06a50e0916bde709fbfdeae306a5 - source: openapi3 tags repository: https://github.com/lezhnev74/openapi-psr7-validator v3: true repositoryMetadata: base64Readme: >- [![Latest Stable Version](https://poser.pugx.org/lezhnev74/openapi-psr7-validator/v/stable)](https://packagist.org/packages/lezhnev74/openapi-psr7-validator)
[![Build Status](https://travis-ci.org/lezhnev74/openapi-psr7-validator.svg?branch=master)](https://travis-ci.org/lezhnev74/openapi-psr7-validator)
[![License](https://poser.pugx.org/lezhnev74/openapi-psr7-validator/license)](https://packagist.org/packages/lezhnev74/openapi-psr7-validator)
![contributions welcome](https://img.shields.io/badge/contributions-welcome-brightgreen.svg)

# NOTICE - THE PACKAGE HAS BEEN CONTRIBUTED TO THE PHP LEAGUE

Go to https://github.com/thephpleague/openapi-psr7-validator

This package is here for existing users only.

# OpenAPI PSR-7 Message (HTTP Request/Response) Validator
This package can validate PSR-7 messages against OpenAPI (3.0.x) specifications 
expressed in YAML or JSON. 

![](image.jpg)

## Installation
```
composer require lezhnev74/openapi-psr7-validator
```

## OpenAPI (OAS) Terms
There are some specific terms that are used in the package. These terms come 
from OpenAPI:
- `specification` - an OpenAPI document describing an API, expressed in JSON or YAML file
- `data` - actual thing that we validate against a specification, including body and metadata
- `schema` - the part of the specification that describes the body of the request / response
- `keyword` - properties that are used to describe the instance are called key
words, or schema keywords
- `path` - a relative path to an individual endpoint
- `operation` - a method that we apply on the path (like `get /password`)
- `response` - described response (includes status code, content types etc)


## How To Validate

### ServerRequest Message
You can validate `\Psr\Http\Message\ServerRequestInterface` instance like this:

```php
$yamlFile = "api.yaml";
$jsonFile = "api.json";

$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getServerRequestValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYaml(file_get_contents($yamlFile))->getServerRequestValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromJson(file_get_contents($jsonFile))->getServerRequestValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromJsonFile($jsonFile)->getServerRequestValidator();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getServerRequestValidator();

$match = $validator->validate($request);
```

As a result you would get and `OperationAddress $match` which has matched the given request. If you already know
the operation which should match your request (i.e you have routing in your project), you can use 
`RouterRequestValidator`

```php
$address = new \OpenAPIValidation\PSR7\OperationAddress('/some/operation', 'post');

$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getRoutedRequestValidator();

$validator->validate($address, $request);
```

This would simplify validation a lot and give you more performance.

### Request Message
You can validate `\Psr\Http\Message\RequestInterface` instance like this:

```php
$yamlFile = "api.yaml";
$jsonFile = "api.json";

$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getRequestValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYaml(file_get_contents($yamlFile))->getRequestValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromJson(file_get_contents($jsonFile))->getRequestValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromJsonFile($jsonFile)->getRequestValidator();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getRequestValidator();

$match = $validator->validate($request);
```

### Response Message
Validation of `\Psr\Http\Message\ResponseInterface` is a bit more complicated
. Because you need not only YAML file and Response itself, but also you need 
to know which operation this response belongs to (in terms of OpenAPI).

Example:

```php
$yamlFile = "api.yaml";
$jsonFile = "api.json";

$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getResponseValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYaml(file_get_contents($yamlFile))->getResponseValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromJson(file_get_contents($jsonFile))->getResponseValidator();
#or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromJsonFile($jsonFile)->getResponseValidator();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromSchema($schema)->getResponseValidator();

$operation = new \OpenAPIValidation\PSR7\OperationAddress('/password/gen', 'get') ;

$validator->validate($operation, $request);
```

### Reuse Schema After Validation

`\OpenAPIValidation\PSR7\ValidatorBuilder` reads and compiles schema in memory as instance of `\cebe\openapi\spec\OpenApi`. Validators use this instance to perform validation logic. You can reuse this instance after the validation like this:

```php
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getServerRequestValidator();
# or
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)->fromYamlFile($yamlFile)->getResponseValidator();

/** @var \cebe\openapi\spec\OpenApi */
$openApi = $validator->getSchema();
```

### Request Message
`\Psr\Http\Message\RequestInterface` validation is not implemented. 

### PSR-15 Middleware
PSR-15 middleware can be used like this:

```php
$yamlFile = 'api.yaml';
$jsonFile = 'api.json';

$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYamlFile($yamlFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYaml(file_get_contents($yamlFile))->getValidationMiddleware();
#or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJsonFile($jsonFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJson(file_get_contents($jsonFile))->getValidationMiddleware();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \OpenAPIValidation\PSR7\ValidationMiddlewareBuilder)->fromSchema($schema)->getValidationMiddleware();
```

### SlimFramework Middleware
Slim framework uses slightly different middleware interface, so here is an 
adapter which you can use like this:

```php
$yamlFile = 'api.yaml';
$jsonFile = 'api.json';

$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYamlFile($yamlFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromYaml(file_get_contents($yamlFile))->getValidationMiddleware();
#or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJsonFile($jsonFile)->getValidationMiddleware();
#or
$psr15Middleware = (new \OpenAPIValidation\PSR15\ValidationMiddlewareBuilder)->fromJson(file_get_contents($jsonFile))->getValidationMiddleware();
#or
$schema = new \cebe\openapi\spec\OpenApi(); // generate schema object by hand
$validator = (new \OpenAPIValidation\PSR7\ValidationMiddlewareBuilder)->fromSchema($schema)->getValidationMiddleware();

$slimMiddleware = new \OpenAPIValidation\PSR15\SlimAdapter($psr15Middleware);

/** @var \Slim\App $app */
$app->add($slimMiddleware);
```

### Caching Layer / PSR-6 Support
PSR-7 Validator has a built-in caching layer (based on [PSR-6](https://www.php-fig.org/psr/psr-6/) interfaces) which saves time on parsing OpenAPI specs. It is optional.
You enable caching if you pass a configured Cache Pool Object to the static constructor like this:
```php
// Configure a PSR-6 Cache Pool
$cachePool = new ArrayCachePool();

// Pass it as a 2nd argument
$validator = (new \OpenAPIValidation\PSR7\ValidatorBuilder)
    ->fromYamlFile($yamlFile)
    ->setCache($cachePool)
    ->getResponseValidator();
# or
\OpenAPIValidation\PSR15\ValidationMiddleware::fromYamlFile($yamlFile, $cachePool);
```

You can use `->setCache($pool, $ttl)` call for both PSR-7 and PSR-15 builder in order to set 
[proper expiration ttl in seconds (or explicit `null`)](https://www.php-fig.org/psr/psr-6/#definitions)

If you want take control over the cache key for schema item, or your cache does not support cache key generation by itself
you can `->overrideCacheKey('my_custom_key')` to ensure cache uses key you want.

### Standalone OpenAPI Validator
The package contains a standalone validator which can validate any data 
against an OpenAPI schema like this:

```php
$spec = <<<SPEC
schema:
  type: string
  enum:
  - a
  - b
SPEC;
$data = "c";

$spec   = cebe\openapi\Reader::readFromYaml($spec);
# (optional) reference resolving
$spec->resolveReferences(new ReferenceContext($spec, "/"));
$schema = new cebe\openapi\spec\Schema($spec->schema);

try {
    (new \OpenAPIValidation\Schema\SchemaValidator())->validate($data, $schema);
} catch(\OpenAPIValidation\Schema\Exception\KeywordMismatch $e) {
    // you can evaluate failure details
    // $e->keyword() == "enum"
    // $e->data() == "c"
    // $e->dataBreadCrumb()->buildChain() -- only for nested data
}
```

## Custom Type Formats
As you know, OpenAPI allows you to add formats to types:

```yaml
schema:
  type: string
  format: binary
```

This package contains a bunch of built-in format validators:
- `string` type:
    - `byte`
    - `date`
    - `date-time`
    - `email`
    - `hostname`
    - `ipv4`
    - `ipv6`
    - `uri`
    - `uuid` (uuid4)
- `number` type
    - `float`
    - `double`

You can also add your own formats. Like this:
```php
# A format validator must be a callable
# It must return bool value (true if format matched the data, false otherwise)

# A callable class:
$customFormat = new class()
{
    function __invoke($value): bool
    {
        return $value === "good value";
    }
};

# Or just a closure:
$customFormat = function ($value): bool {
    return $value === "good value";
};

# Register your callable like this before validating your data
\OpenAPIValidation\Schema\TypeFormats\FormatsContainer::registerFormat('string', 'custom', $customFormat);
```

## Exceptions
The package throws a list of various exceptions which you can catch and handle. There are some of them:
- Schema related:
    - `\OpenAPIValidation\Schema\Exception\KeywordMismatch` - Indicates that data was not matched against a schema's keyword
        - `\OpenAPIValidation\Schema\Exception\TypeMismatch` - Validation for `type` keyword failed against a given data. For example `type:string` and value is `12`
        - `\OpenAPIValidation\Schema\Exception\FormatMismatch` - data mismatched a given type format. For example `type: string, format: email` won't match `not-email`.
- PSR7 Messages related:
    - `\OpenAPIValidation\PSR7\Exception\NoContentType` - HTTP message(request/response) contains no Content-Type header. General HTTP errors.
    - `\OpenAPIValidation\PSR7\Exception\NoPath` - path is not found in the spec
    - `\OpenAPIValidation\PSR7\Exception\NoOperation` - operation os not found in the path
    - `\OpenAPIValidation\PSR7\Exception\NoResponseCode` - response code not found under the operation in the spec
    - Validation exceptions (check parent exception for possible root causes):
        - `\OpenAPIValidation\PSR7\Exception\ValidationFailed` - generic exception for failed PSR-7 message
        - `\OpenAPIValidation\PSR7\Exception\Validation\InvalidBody` - body does not match schema
        - `\OpenAPIValidation\PSR7\Exception\Validation\InvalidCookies` - cookies does not match schema or missing required cookie
        - `\OpenAPIValidation\PSR7\Exception\Validation\InvalidHeaders` - header does not match schema or missing required header
        - `\OpenAPIValidation\PSR7\Exception\Validation\InvalidPath` - path does not match pattern or pattern values does not match schema
        - `\OpenAPIValidation\PSR7\Exception\Validation\InvalidQueryArgs` - query args does not match schema or missing required argument
        - `\OpenAPIValidation\PSR7\Exception\Validation\InvalidSecurity` - request does not match security schema or invalid security headers
    - Request related:
        - `\OpenAPIValidation\PSR7\Exception\MultipleOperationsMismatchForRequest` - request matched multiple operations in the spec, 
        but validation failed for all of them.

## Testing
You can run the tests with:

```
vendor/bin/phpunit
```

## Contribution Guide
Feel free to open an Issue or add a Pull request. 
There is a certain code style that this package follows: [doctrine/coding-standard](https://www.doctrine-project.org/projects/doctrine-coding-standard/en/latest/reference/index.html#introduction).

To conform to this style please use a git hook, shipped with this package at `.githooks/pre-commit`.

How to use it:
1. Clone the package locally and navigate to the folder
2. Create a symlink to the hook like this: `ln -s -f ../../.githooks/pre-commit .git/hooks/pre-commit`
3. Add execution rights: `chmod +x .git/hooks/pre-commit`
4. Now commit any new changes and the code will be checked and formatted accordingly.
5. If there are any issues with your code, check the log here: `.phpcs-report.txt`

## Credits
People:
- [Dmitry Lezhnev](https://github.com/lezhnev74)
- [Carsten Brandt](https://github.com/cebe)
- [Samuel Nela](https://github.com/samnela)
- [Pavel Batanov](https://github.com/scaytrase)
- [Christopher L Bray](https://github.com/brayniverse)
- [David Pauli](https://github.com/dpauli)
- [Jason Judge](https://github.com/judgej)
- [Yannick Chenot](https://github.com/osteel)
- [TarasBK](https://github.com/TarasBK)
- [Jason B. Standing](https://github.com/jasonbstanding)
- [Dmytro Demchyna](https://github.com/dmytro-demchyna)
- [Will Chambers](https://github.com/willchambers99)
- [Ignacio](https://github.com/imefisto)
- A big thank you to [Henrik Karlström](https://github.com/hkarlstrom) who kind of inspired me to work on this package. 

Resources:
- Icons made by Freepik, licensed by CC 3.0 BY
- [cebe/php-openapi](https://github.com/cebe/php-openapi) package for Reading OpenAPI files
- [slim3-psr15](https://github.com/bnf/slim3-psr15) package for Slim middleware adapter
 
## License
The MIT License (MIT). Please see `License.md` file for more information.

## TODO
- [ ] Support Discriminator Object (note: apparently, this is not so straightforward, as discriminator can point to any external scheme)
 readmeEtag: '"50ae034d1007fb5d4f98a0969a7d466e429e89e3"' readmeLastModified: Thu, 19 Sep 2019 09:02:21 GMT repositoryId: 184420719 description: >- It validates PSR-7 messages (HTTP request/response) against OpenAPI specifications created: '2019-05-01T13:32:30Z' updated: '2026-01-16T08:25:28Z' language: PHP archived: false stars: 168 watchers: 12 forks: 10 owner: lezhnev74 logo: https://avatars.githubusercontent.com/u/10206110?v=4 license: MIT repoEtag: '"f255b1e256393f6eee28a3cdb0e5977060b65f0f8e1048bd61f75e54f7f08292"' repoLastModified: Fri, 16 Jan 2026 08:25:28 GMT foundInMaster: true id: 827ca67e948986acbbe7941175945bd7 - source: openapi3 tags repository: https://github.com/swagger-api/swagger-inflector v3: true repositoryMetadata: base64Readme: >- # Swagger Inflector

[![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.swagger/swagger-inflector/badge.svg?style=plastic)](https://maven-badges.herokuapp.com/maven-central/io.swagger/swagger-inflector)

----

**NOTE:** If you're looking for `swagger-inflector` 1.X and Swagger/OpenApi 2.0, please refer to [v1 branch](https://github.com/swagger-api/swagger-inflector/tree/v1)

----

This project uses the Swagger Specification to drive an API implementation.  Rather than a typical top-down or bottom-up swagger integration, the Inflector uses the swagger specification as a DSL for the REST API.  The spec drives the creation of routes and controllers automatically, matching methods and method signatures from the implementation.  This brings a similar integration approach to the JVM as [swagger-node](https://github.com/swagger-api/swagger-node) brings to the javascript world.

To allow for an iterative development, the framework will mock responses for any unimplemented methods, based on the specification.  That means you can ship your API to your consumers for review immediately as you build it out.

You have full control over the mapping of controllers to classes and methods as well as models.

## Quick start!

Run this command to start in a hurry.  It will create a project named `my-project`
```
curl -L https://raw.githubusercontent.com/swagger-api/swagger-inflector/master/setup.sh | project=my-project bash
```

This will download everything you need to start editing and running a swagger-inflector based project.  See the output of the command for instructions.

### Components

Inflector uses the following libraries:

 - swagger models for the swagger definition
 - Jackson for JSON processing
 - Jersey 2.x for REST
 - Minimum Java 8

### Integration

Inflector will create routes and add them to Jersey.  You simply need to register the Inflector application in your webapp and it should be compatible with your existing deployment, whether with web.xml, spring, dropwizard, etc.

To add inflector via `web.xml`:

```xml
<servlet>
  <servlet-name>swagger-inflector</servlet-name>
  <servlet-class>org.glassfish.jersey.servlet.ServletContainer</servlet-class>
  <init-param>
    <param-name>javax.ws.rs.Application</param-name>
    <param-value>io.swagger.oas.inflector.OpenAPIInflector</param-value>
  </init-param>
  <load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
  <servlet-name>swagger-inflector</servlet-name>
  <url-pattern>/*</url-pattern>
</servlet-mapping>
```

This simply adds the `SwaggerInflector` application to Jersey.

### Configuration

Inflector uses a single yaml file for configuration.  The default file is `inflector.yaml` but it can be overridden by setting a system property when starting the JVM:

```
-Dconfig=/path/to/config
```

The configuration supports the following:

```yaml
# mode (development | staging | production).  Default is development, and this value will be overridden by a system property
# -Denvironment=production for example
environment: development

# configure your default controller package for method discovery
controllerPackage: io.swagger.oas.sample.controllers

# configure the default model package for model discovery
modelPackage: io.swagger.oas.sample.models

# the path to the swagger definition (Note! this can be overridden with -DswaggerUrl as a system property
swaggerUrl: openapi.yaml

# specific mappings for models, used to locate models in the `#/definitions/${model}`
modelMappings:
  User: io.swagger.oas.sample.models.User

# HTTP response code when required parameters are missing
invalidRequestCode: 400

#Allows to configure the exposed spec (values in example are the defaults)
exposedSpecOptions:
  parseOptions:
    resolve: false
    resolveFully: false
  useOriginalNotParsed: false
  hideInflectorExtensions: true
  mergeRootPath: true

```

### Locating the controller class

The actual controller class for each method is located via the first of the following mechanisms:
- a x-swagger-router-controller extension at the method level can specify the specific controller class
- each tag associated with the method is assembled into the classnames "&lt;controllerPackage&gt;.&lt;Tag&gt;" or 
"&lt;controllerPackage&gt;.&lt;Tag&gt;Controller", the first of these classes that is found by Class.forName(...) will be used
- an optional &lt;controllerClass&gt; configuration parameter is appended to &lt;controllerPackage&gt; 
- as a last resort a class named &lt;controllerPackage&gt;.Default is used

By default the class is loaded directly with Class.forName(...).newInstance() - but you can override class creation
by providing a custom ControllerFactory to the inflector configuration (for example if you want your controllers to be 
loaded by a DI framework).

### Locating the target method

When locating methods, the `operationId` is used as the method name for lookup via reflection.  If not specified, there is logic for generation of a method name.

Once a method is matched via name, the parameter types will be compared to ensure we have the right model.  In all methods, only java objects are supported--primitives currently will not match (this allows for proper nulls).

You can override a model mapping by setting a vendor extension in the swagger yaml:

```yaml
# uses method name, look for controllerPackage in the configuration
paths:
  /test1:
    get:
      x-swagger-router-controller: SampleController
      operationId: getTest1
      parameters:
        - name: name
          in: query
          type: string
      responses:
        200:
          description: Success!
```

From the configuration example above, this will look for the following class:

```
class: io.swagger.sample.controllers.SampleController
```

with the following method:

```
method: public Object getTest1(
    RequestContext,
    java.lang.String name)
```

#### Complex inputs

When there are complex inputs, such as the example below:

```yaml
paths:
  /test2:
    post:
      x-swagger-router-controller: SampleController
      operationId: addUser
      requestBody:
          content:
            "application/json":
              schema:
                $ref: '#/definitions/User'
      parameters:
        - name: name
          in: query
          type: string
      responses:
        200:
          description: Success!
```

the Inflector will do the following:

 - Look in vendor extensions for the models to see if a mapping exists.  If so, it will attempt to load it via the classloader

 ```yaml
   Address:
    x-swagger-router-model: io.swagger.test.models.Address
    properties:
      street:
        type: string
        example: 12345 El Monte Road
      city:
        type: string
        example: Los Altos Hills
      state:
        type: string
        example: CA
      zip:
        type: string
        example: '94022'
 ```

 - Look in the configuration for a mapping between `User` and a concrete class definition.  If the definition exists AND the class can be loaded, the method will look like such:

 ```
 public ResponseContext addUser (
    RequestContext context,             // request context
    io.swagger.sample.models.User user, // user being added
    java.lang.String name)              // the `name` query param
 ```

 - If the definition does not exist, the `modelPackage` from the configuration will be used to attempt to load the class:

 ```


 ```

 If the definition can be loaded it will be used as the method signature

 - If no model can be loaded, it is the developer's job to unwrap the input and parse it on their own.  This requires `Content-Type`-specific processing.  Inflector will then look for the following method:

 ```
 public ResponseContext addUser (
    RequestContext context,             // request context
    JsonNode user,                      // a Json tree representing the user
    java.lang.String name)              // the `name` query param
 ```

 - If no method can be found, a mock response will be returned based on the swagger definition.  For complex objects, if an `example` exists, we will use that.  Otherwise, it will be constructed.


The RequestWrapper and ResponseContext contain information about headers (in and outbound), content-type and acceptable response types.

#### Outputs

Your controllers can return null (void response), an object (entity), or a `ResponseContext`, which allows you to send specific error codes, headers, and an optional entity.

For example, if you want to return a `Pet` from a controller:

```java
    public ResponseContext getPet(RequestContext request, java.lang.Integer petId) {
        // do your magic to fetch a pet...
        Pet pet = complexBusinessLogic.getPetById(petId);

        return new ResponseContext()
                .status(Status.OK)
                .entity(pet);
    }
```

and the Inflector will return a `200` response code, marshalling the `Pet` object into the appropriate content type.

If you do not implement your controller, the Inflector will generate sample data based on your model definitions.  It will honor any examples that you have in the definitions, assuming they are compatible with the schema you declared.  For example, this definition:

```yaml
properties:
  street:
    type: "string"
    example: "12345 El Monte Blvd"
  city:
    type: "string"
    example: "Los Altos Hills"
  state:
    type: "string"
    example: "CA"
    minLength: 2
    maxLength: 2
  zip:
    type: "string"
    example: "94022"
xml:
  name: "address"
```

Will produce this example for a `Accept:application/json`:

```json
{
  "street" : "12345 El Monte Blvd",
  "city" : "Los Altos Hills",
  "state" : "CA",
  "zip" : "94022"
}
```

and `application/yaml`:

```yaml
street: "12345 El Monte Blvd"
city: "Los Altos Hills"
state: "CA"
zip: "94022"
```

and `application/xml`:

```xml
<address>
  <street>12345 El Monte Blvd</street>
  <city>Los Altos Hills</city>
  <state>CA</state>
  <zip>94022</zip>
</address>
```

#### Payload validation

Since your inbound and outbound payloads are defined with the Swagger schema, Inflector 
can validate them at runtime.  Just enable payload validations in your inflector config:

```yaml
validatePayloads: true
```

And at start-up, Inflector will read the schema and attach the relevant section of it 
to the operation.  For example, a post operation that has this as the schema definition:

```json
{
  "Category": {
    "required": [
      "id"
    ],
    "properties": {
      "id": {
        "type": "integer",
        "format": "int64"
      },
      "name": {
        "type": "string"
      }
    },
    "xml": {
      "name": "Category"
    }
  }
}
```

Will fail if the incoming body looks like this:

```json
{
  "name": "Tony"
}
```
because the required field `id` is missing.

The same goes for responses generated by the server.  Any response code that you send will 
be validated against it's corresponding schema.

You can choose to enable this in development,
staging, or production.

#### Content type negotiation

There is a pluggable framework for handling different content types.  You can register any processor by the following:

```java
EntityProcessor myProcessor = new MyEntityProcessor();  // implements EntityProcessor
EntityProcessorFactory.addProcessor(myProcessor);
```

#### Development Lifecycle

There are three modes that the Inflector supports, as configured by the `environment` attribute in the inflector config:

 - **development**.  In this mode, mock responses will be sent for controllers which are not implemented.  The intention
   is to allow you to quickly iterate on the implementation of the design.  In addition, missing model implementations
   are tolerated and supported.

 - **staging**.  Warning messages will be logged when starting the service for any missing controller, method, or model.

 - **production**.  The expectation is all methods and declared (manually mapped) models exist.  If they don't, it'll throw
   nasty errors and the server will not start.

In development mode, there is a `/debug.json` page which shows implementation details of the inflector service.

If your Swagger Description is unparsable, the server will throw ugly errors on startup and the `debug.json` page will
   give indications as to why.

#### Samples
The samples are a being refactor to support the new inflector.

You will soon find samples for the inflector project in the [Swagger-Samples](https://github.com/swagger-api/swagger-samples) repository.  The inflector projects start with `inflector-`

#### Running tests

If running Java 8, you will need to run a variant that has backported fix 8157236. Azul Zulu is confirmed to work (https://github.com/jmockit/jmockit1/issues/710).

If running with Java 9 or later, you will need to either:
- Pass `-Djdk.attach.allowAttachSelf=true` to the VM.
- Configure the test execution JVM to start with the "-javaagent:<proper path>/jmockit.1.x.jar" initialization parameter. It can be specified in the build script file for tools such as Maven or Gradle, or in a "Run/Debug Configuration" for IntelliJ IDEA or Eclipse.

## Security contact

Please disclose any security-related issues or vulnerabilities by emailing [security@swagger.io](mailto:security@swagger.io), instead of using the public issue tracker.
 readmeEtag: '"e526a9e92382e438d0a6e9de2a6473b04affe96d"' readmeLastModified: Mon, 08 Apr 2024 07:01:04 GMT repositoryId: 39236124 description: null created: '2015-07-17T05:14:14Z' updated: '2026-01-15T10:51:05Z' language: Java archived: false stars: 167 watchers: 21 forks: 85 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"af63bb418c22dfcb1ded96b5239bf99ba4e31a668d3dbd0f6c2290b9920d39ba"' repoLastModified: Thu, 15 Jan 2026 10:51:05 GMT foundInMaster: true category: Parsers id: 73bbfe691003c37a641d8864645f2425 - source: openapi3 tags repository: https://github.com/sunrise-php/http-router v3: true repositoryMetadata: base64Readme: >- IyBBIHBvd2VyZnVsIHNvbHV0aW9uIGFzIHRoZSBmb3VuZGF0aW9uIG9mIHlvdXIgcHJvamVjdAoKIyMgUmVzb3VyY2VzCgotIFtEb2N1bWVudGF0aW9uXShodHRwczovL2Rldi5zdW5yaXNlLXN0dWRpby5pby9kb2NzL3BhY2thZ2VzL3N1bnJpc2UvaHR0cC1yb3V0ZXIvKQo= readmeEtag: '"a55ac755dbefedd1dbd70b4c674bd30e25475fe0"' readmeLastModified: Mon, 10 Mar 2025 13:57:11 GMT repositoryId: 161002423 description: A powerful solution as the foundation of your project. created: '2018-12-09T03:59:48Z' updated: '2025-12-03T20:28:21Z' language: PHP archived: false stars: 162 watchers: 4 forks: 10 owner: sunrise-php logo: https://avatars.githubusercontent.com/u/41580063?v=4 license: MIT repoEtag: '"07e1d63b96197c7566db644d95026049afb4ef7c526b4d94dc832f18140b05e2"' repoLastModified: Wed, 03 Dec 2025 20:28:21 GMT foundInMaster: true category: Data Validators id: d37760d7bb1c259409b90020129dcf91 - source: openapi3 tags repository: https://github.com/netwo-io/apistos v3: true id: 644bd457afd213be930b970bd9f35da0 repositoryMetadata: base64Readme: >- # Apistos &emsp; [![Documentation]][docs.rs] [![Latest Version]][crates.io] [![Build Status]][build] [![Deps Status]][deps.rs]

[docs.rs]: https://docs.rs/apistos/

[crates.io]: https://crates.io/crates/apistos

[build]: https://github.com/netwo-io/apistos/actions/workflows/build.yaml?branch=main

[Documentation]: https://img.shields.io/docsrs/apistos

[Latest Version]: https://img.shields.io/crates/v/apistos.svg

[Build Status]: https://github.com/netwo-io/apistos/actions/workflows/build.yaml/badge.svg?branch=main

[deps.rs]: https://deps.rs/crate/apistos

[Deps Status]: https://deps.rs/crate/apistos/latest/status.svg

[OASv3.md]: https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.3.md

An OpenAPI documentation tool exposing [OAS 3.0][OASv3.md] models as well as an actix-web wrapper similar
to [paperclip](https://github.com/paperclip-rs/paperclip).

**Apistos** is composed of these crates:

- [`apistos`](./apistos): [actix-web](https://github.com/actix/actix-web) wrapper to generate an OpenAPI v3.0.3
  documentation file
- [`apistos-core`](./apistos-core): A set of traits and common models around [OpenAPI v3.0.3][OASv3.md]
- [`apistos-gen`](./apistos-gen): macro utilities to generate [OpenAPI v3.0.3][OASv3.md] documentation from Rust models
- [`apistos-models`](./apistos-models): [OpenAPI v3.0.3][OASv3.md] models
  with [`Schema`](https://docs.rs/schemars/latest/schemars/schema/enum.Schema.html) based
  on [schemars](https://github.com/GREsau/schemars) definition
- [`apistos-plugins`](./apistos-plugins): traits and utilities to extend apistos
- [`apistos-rapidoc`](./apistos-rapidoc): bridge between Apistos and [RapiDoc](https://rapidocweb.com/) for actix.
- [`apistos-plugins`](./apistos-redoc): bridge between Apistos and [Redoc](https://redocly.com/redoc/) for actix.
- [`apistos-scalar`](./apistos-scalar): bridge between Apistos and [Scalar](https://scalar.com/) for actix.
- [`apistos-shuttle`](./apistos-shuttle): allows you to run an actix-web server documented with Apistos
  on [Shuttle](https://www.shuttle.rs/).
- [`apistos-swagger-ui`](./apistos-swagger-ui): bridge between Apistos
  and [Swagger UI](https://swagger.io/tools/swagger-ui/) for actix

Check out our [example project](examples/petstore).

### What does Apistos means

Apistos (pronounced **_/a.p.i.stos/_**) is a word play between Héphaïstos (Ἥφαιστος, grec god of blacksmiths,
carpenters, craftsmen, metallurgy ... which can also be considered by some as the god of technology) and API (pronounced
**_/a.p.i/_** in French).

## Apistos

- [Installation](#installation)
- [Usage example](#usage-example)
- [Feature flags](#feature-flags)
- [Alternatives](#alternatives)
- [About us](#about-us)

### Installation

```toml
[dependencies]
#schemars = "0.8"
# sadly we currently rely on a fork to fix multiple flatten for enums, related PR can be found here: https://github.com/GREsau/schemars/pull/264
schemars = { package = "apistos-schemars", version = "0.8" }
apistos = "0.6"
```

### Usage example

Wrap your regular actix-web app using apistos types.

Most of these types are drop-in types for actix-web one's.

```rust
use std::fmt::Display;
use actix_web::{App, HttpServer, ResponseError};
use actix_web::http::StatusCode;
use actix_web::middleware::Logger;
use actix_web::web::Json;
use apistos::actix::CreatedJson;
use apistos::api_operation;
use apistos::ApiComponent;
use apistos::ApiErrorComponent;
use apistos::app::OpenApiWrapper;
use apistos::spec::Spec;
use apistos::web::{post, resource, scope};
use apistos_models::info::Info;
use core::fmt::Formatter;
use schemars::JsonSchema;
use serde::{Deserialize, Serialize};
use std::error::Error;
use std::net::Ipv4Addr;

#[derive(Serialize, Deserialize, Debug, Clone, JsonSchema, ApiComponent)]
pub struct Test {
  pub test: String
}

#[derive(Serialize, Deserialize, Debug, Clone, ApiErrorComponent)]
#[openapi_error(
  status(code = 403),
  status(code = 404),
  status(code = 405, description = "Invalid input"),
  status(code = 409)
)]
pub enum ErrorResponse {
  MethodNotAllowed(String),
  NotFound(String),
  Conflict(String),
  Unauthorized(String),
}

impl Display for ErrorResponse {
  fn fmt(&self, _f: &mut Formatter<'_>) -> std::fmt::Result {
    todo!()
  }
}

impl ResponseError for ErrorResponse {
  fn status_code(&self) -> StatusCode {
    todo!()
  }
}

#[api_operation(
  tag = "pet",
  summary = "Add a new pet to the store",
  description = r###"Add a new pet to the store
    Plop"###,
  error_code = 405
)]
pub(crate) async fn test(
  body: Json<Test>,
) -> Result<CreatedJson<Test>, ErrorResponse> {
  Ok(CreatedJson(body.0))
}

#[actix_web::main]
async fn main() -> Result<(), impl Error> {
  HttpServer::new(move || {
    let spec = Spec {
      info: Info {
        title: "An API".to_string(),
        version: "1.0.0".to_string(),
        ..Default::default()
      },
      ..Default::default()
    };

    App::new()
        .document(spec)
        .wrap(Logger::default())
        .service(scope("/test")
            .service(
              resource("")
                  .route(post().to(test))
            )
        )
        .build("/openapi.json")
  })
      .bind((Ipv4Addr::UNSPECIFIED, 8080))?
      .run()
      .await
}
```

For a complete example, see [the sample petstore](https://github.com/netwo-io/apistos/tree/main/examples/petstore).

### Feature flags

| name               | description                                                              | extra dependencies                                              |
|--------------------|--------------------------------------------------------------------------|-----------------------------------------------------------------|
| `query` (default)  | Enables documenting `actix_web::web::Query`                              |                                                                 |
| `actix` (default)  | Enables documenting types from `actix`                                   |                                                                 |
| `lab_query`        | Enables documenting `actix_web_lab::extract::Query`                      | [`actix-web-lab`](https://crates.io/crates/actix-web-lab)       |
| `garde`            | Enables input validation through `garde`                                 | [`garde`](https://crates.io/crates/garde)                       |
| `actix-session`    | Enables documenting types from `actix-session`                           | [`actix-session`](https://crates.io/crates/actix-session)       |
| `actix-web-grants` | Enables support for `actix-web-grants`                                   | [`actix-web-grants`](https://crates.io/crates/actix-web-grants) |
| `rapidoc`          | Enables RapiDoc to expose the generated openapi file                     |                                                                 |
| `redoc`            | Enables Redoc to expose the generated openapi file                       |                                                                 |
| `swagger-ui`       | Enables Swagger UI to expose the generated openapi file                  |                                                                 |
| `qs_query`         | Enables documenting types from `serde_qs`                                | [`serde_qs`](https://crates.io/crates/serde-qs)                 |
| `chrono`           | Enables documenting types from `chrono`                                  | [`chrono`](https://crates.io/crates/chrono)                     |
| `multipart`        | Enables documenting types from `actix-multipart`                         | [`actix-multipart`](https://crates.io/crates/actix-multipart)   |
| `rust_decimal`     | Enables documenting types from `rust_decimal`                            | [`rust_decimal`](https://crates.io/crates/rust-decimal)         |
| `uuid`             | Enables documenting types from `uuid`                                    | [`uuid`](https://crates.io/crates/uuid)                         |
| `url`              | Enables documenting types from `url`                                     | [`url`](https://crates.io/crates/url)                           |
| `extras`           | Enables `chrono`, `multipart`, `rust_decimal`, `uuid` and `url` features | All from previous features                                      |

### What's next

- Handle schema for errors using ApiErrorComponent derive macro

### Alternatives

| Crate                                             | Key differences                                                                                                                                                                                                                                                                                                                               |
|---------------------------------------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| [`paperclip`](https://crates.io/crates/paperclip) | Paperclip is similar to this project but generates Swagger v2 documentation. Paperclip also provides a tool to generate rust code from a Swagger v2 document.                                                                                                                                                                                 |
| [`utoipa`](https://crates.io/crates/utoipa)       | Utoipa-actix integration rely on [actix web macros](https://docs.rs/actix-web-macros/latest/actix_web_macros/) for routing definition. At first, we planned on relying on utoipa for OAS types and schema derivation but for now [utoipa doesn't support generic struct the way we intended to](https://github.com/juhaku/utoipa/issues/703). |
| [`okapi`](https://crates.io/crates/okapi)         | Pretty similar, based on schemars as well (and maintained by the founder of schemars) but not integrated with actix.                                                                                                                                                                                                                          |

### Articles ###

- announcement article can be found [on medium](https://medium.com/netwo/documenting-api-for-actix-web-b575adb841a1). It
  acts as a tutorial for Apistos.

### About us

apistos is provided by [Netwo](https://www.netwo.io).

We use this crate for our internal needs and therefore are committed to its maintenance, however we cannot provide any
additional guaranty. Use it at your own risks.

While we won't invest in any feature we don't need, we are open to accept any pull request you might propose.

We are a France based full-remote company operating in the telecom industry. If you are interested in learning more,
feel free to visit [our career page](https://www.netwo.io/carriere).
 readmeEtag: '"ff5595f5f96dad75f7984f9b02ebc1a36bfe3af3"' readmeLastModified: Wed, 11 Jun 2025 13:23:17 GMT repositoryId: 691100358 description: Actix-web wrapper for automatic OpenAPI 3.0 documentation generation. created: '2023-09-13T13:54:51Z' updated: '2026-02-03T08:13:17Z' language: Rust archived: false stars: 207 watchers: 4 forks: 11 owner: netwo-io logo: https://avatars.githubusercontent.com/u/64151169?v=4 license: MIT repoEtag: '"240adb7d2334452a94469b10a6f3f836291096b90ee6aa26b2b4ffbcda2eafa7"' repoLastModified: Tue, 03 Feb 2026 08:13:17 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/aaronchen2k/deeptest v3: true id: aba6967d5bd400d92f9d80a990915b27 repositoryMetadata: base64Readme: >- PGltZyBzcmM9Inhkb2MvaW1hZ2UvbG9nby5wbmciIHdpZHRoPSIyMDAiIC8+CgpEZWVwVGVzdCBpcyBDb250aW51ZSBUZXN0aW5nIFBsYXRmb3JtIHdyaXR0ZW4gaW4gR28gd2l0aCBJcmlzLCBHb3JtLCBNeVNRTCBhbmQgVnVlMywgUHJvdG9idWYsIFdlYlNvY2tldC4KCipQbGVhc2UgcmVmZXIgdG8gW05HVGVzdGluZ10oaHR0cHM6Ly9naXRodWIuY29tL2Fhcm9uY2hlbjJrL25ndGVzdGluZy1wbGF0Zm9ybSkgdG8gZmluZCB0aGUgcHJldmlvdXMgamF2YSBwcm9qZWN0LioKCiMjIyBVc2VyIEd1aWRlCmh0dHA6Ly9kb2MuZGVlcHRlc3QuY29tCgojIyMgUXVpY2sgU3RhcnQKaHR0cDovL2RvYy5kZWVwdGVzdC5jb20vc2VydmVyLmh0bWwKCiMjIyBMaWNlbnNlcwpBbGwgc291cmNlIGNvZGUgaXMgbGljZW5zZWQgdW5kZXIgdGhlIFtHUEx2MyBMaWNlbnNlXShMSUNFTlNFLm1kKS4KCiMjIFRoYW5rcwpUaGFuayB5b3UgW0pldEJyYWluc10oaHR0cHM6Ly93d3cuamV0YnJhaW5zLmNvbSkgZm9yIHByb3ZpZGluZyBmcmVlIG9wZW4gc291cmNlIGxpY2Vuc2VzLgo= readmeEtag: '"9410f3c68f21cda7faf6fc0bb6d84759ed16e7b6"' readmeLastModified: Fri, 16 Aug 2024 09:26:22 GMT repositoryId: 122702750 description: API Management and Testing Platform created: '2018-02-24T04:23:08Z' updated: '2025-09-05T02:08:45Z' language: Go archived: false stars: 153 watchers: 18 forks: 67 owner: aaronchen2k logo: https://avatars.githubusercontent.com/u/3222874?v=4 license: GPL-3.0 repoEtag: '"8f6930939c505b3a45e9a4d45d1626fd1a228f4a5c9117c47e0ae58243bf0c77"' repoLastModified: Fri, 05 Sep 2025 02:08:45 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/deeptest-com/deeptest - source: openapi3 tags repository: https://github.com/tyrchen/quenya v3: true repositoryMetadata: base64Readme: >- # Quenya

__Disclaimer: Quenya is under active development and is at its early stage. Please DO NOT use it in prod environment. Use with cautions.__

Quenya is a framework to build high-quality REST API applications based on extended [OpenAPI spec](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.1.0.md). For the Quenya extension, see [here](docs/quenya_extension.md). With the OAPI spec, Quenya can generate high-quality code for many parts of the API pipeline:

- Preprocessors:
  - [x] request validator: validate the request params
  - [ ] auth handler: process authentication for the API endpoints
  - [ ] access controller: process authorization for the API endpoints
- API handlers:
  - [x] fake API handler to generate a fake response for mocking purpose
  - [ ] gRPC handler to act as a proxy between your client and your gRPC server (require extended OpenAPI grammar)
- Postprocessors:
  - [x] response validator to validate the response body and headers (for dev/testing purpose)

Quenya will also generate property testing, it will use `Plug.Test` and `StreamData` to build tests. Requests (url, query, request headers and request body) will be generated and then sent to generated `Router`, then it will use the response schema to validate the result. Currently the testing only covers happy path.

Quenya will also provide a set of modules, plugs, test helpers to help you build REST APIs easily.

## How to use Quenya

### Install CLI

First of all, install Quenya CLI:

```bash
$ mix archive.install hex quenya_installer
Resolving Hex dependencies...
Dependency resolution completed:
New:
  quenya_installer 0.3.0
* Getting quenya_installer (Hex package)

20:22:15.605 [info]  erl_tar: removed leading '/' from member names

All dependencies are up to date
Compiling 5 files (.ex)
Generated quenya_installer app
Generated archive "quenya_installer-0.3.0.ez" with MIX_ENV=prod
Are you sure you want to install "quenya_installer-0.3.0.ez"? [Yn]
* creating /Users/tchen/.mix/archives/quenya_installer-0.3.0
```

### Generate APP from an existing OAPI spec

Once you finished installing quenya CLI, you can build a API app with quenya:

```bash
$ cd /tmp
$ curl https://raw.githubusercontent.com/tyrchen/quenya/master/parser/test/fixture/petstore.yml > petstore.yml
$ mix quenya.new petstore.yml petstore
* creating petstore/config/config.exs
* creating petstore/config/dev.exs
* creating petstore/config/prod.exs
* creating petstore/config/staging.exs
* creating petstore/config/test.exs
* creating petstore/lib/petstore/application.ex
* creating petstore/lib/petstore.ex
* creating petstore/mix.exs
* creating petstore/README.md
* creating petstore/.formatter.exs
* creating petstore/.gitignore
* creating petstore/test/test_helper.exs

Fetch and install dependencies? [Yn]
* running mix deps.get
* running mix deps.compile

We are almost there! The following steps are missing:

    $ cd petstore

You can run your app inside IEx (Interactive Elixir) as:

    $ iex -S mix


```

This will create a new elixir app, copy your spec file (or spec folder) to `priv/spec/main.yml`, and generate API code based on the spec.

### Running the app

Now you can run the app:

```bash
$ cd petstore/
$ mix compile.quenya # this command will generate/regenerate code on /gen and /test/gen folders
$ iex -S mix
Erlang/OTP 23 [erts-11.1.3] [source] [64-bit] [smp:16:16] [ds:16:16:10] [async-threads:1] [hipe] [dtrace]

Compiling 44 files (.ex)
Generated petstore app
Interactive Elixir (1.11.2) - press Ctrl+C to exit (type h() ENTER for help)
```

Just run a few commands without writing even a single line of code, you have an API app ready to use. Try open `http://localhost:4000/swagger`. You will see an API playground with standard Swagger UI:

![](docs/images/swagger.jpg)

It's great but nothing special. Now, try to invoke one of the APIs, say `GET /pet/findByStatus`:

![](docs/images/swagger_call.jpg)

Amazing! Don't believe what you saw? Try with this command:

```bash
curl -X POST "http://localhost:4000/pet" -H  "accept: application/json" -H  "Content-Type: application/json" -d "{\"name\":\"doggie\",\"photoUrls\":[\"bad url\"]}" -i
HTTP/1.1 400 Bad Request
cache-control: max-age=0, private, must-revalidate
content-length: 33
date: Mon, 30 Nov 2020 04:45:37 GMT
server: Cowboy

Expected to be a valid image_uri.
```

According to [petstore.yml](test/fixture/petstore.yml), request body must be a Pet type, and `name` / `photoUrls` are required. `photoUrls` shall be an array of string, with format as `image_url` (an extended format by quenya). Quenya will validate requests by its schema so here we need a valid url. Let's correct this:

```bash
$ curl -X POST "http://localhost:4000/pet" -H  "accept: application/json" -H  "Content-Type: application/json" -d "{\"name\":\"doggie\",\"photoUrls\":[\"https://source.unsplash.com/random\"]}" -i
HTTP/1.1 200 OK
cache-control: max-age=0, private, must-revalidate
content-length: 376
content-type: application/json; charset=utf-8
date: Mon, 30 Nov 2020 04:51:03 GMT
server: Cowboy

{"category":{"id":683,"name":"Dtlir6vgkz6UeAwK5q4._9--A.--._V_mjp.K--3T.0-e_.7-_qfRmfu"},"id":928,"name":"758Yhl_jx_Rt_fi5fz_JtE_k__JY2J__Tt9Y1","photoUrls":["https://source.unsplash.com/random/400x400","https://source.unsplash.com/random/400x400"],"status":"sold","tags":[{"id":480,"name":"iusto"},{"id":64,"name":"error"},{"id":658,"name":"modi"},{"id":313,"name":"nihil"}]}
```

### Running the tests

Quenya generates property tests for all your API endpoints based on OAPI spec, so before coding your own API handler into the repo, you'd like to be more test-driven, try `mix test` now:

```bash
$ mix test
Compiling 42 files (.ex)
Generated petstore app
....................

Finished in 2.7 seconds
20 properties, 2 failures
```

Note these tests covers all success cases. In future, we will try to cover all failed cases in Quenya.

### How much code Quenya generated for you?

If you have `tokei` installed, you can have a basic idea on how much code Quenya generated for you:

```bash
$ tokei gen test
-------------------------------------------------------------------------------
 Language            Files        Lines         Code     Comments       Blanks
-------------------------------------------------------------------------------
 Elixir                 83         8152         7060            0         1092
-------------------------------------------------------------------------------
 Total                  83         8152         7060            0         1092
-------------------------------------------------------------------------------
```

That's 8k LoC for the petstore spec. The more APIs you defined, the more Quenya will do for you. Once we have most of the parts of Quenya built, this number will be much bigger.

### What's under the hood?

Now you have a basic feeling on what's going on. By default, Quenya will generate an API router based on API spec, with a convenient swagger UI. For each route defined in the spec, Quenya will generate a Plug for it. And a Plug is a pipeline which will execute in this order:

- preprocessors: any Plug to be executed before the actual route handler. Here, RequestValidator Plug will help to validate request params against the schema.
- handlers: handlers for the route. This is what you shall put your actual API logic, but for mocking purpose, Quenya generates a fake handler which meets the response schema. In future, Quenya will support gRPC handler which will be very useful if what you need is a grpc proxy (think [grpc-gateway](https://github.com/grpc-ecosystem/grpc-gateway)).
- postprocessors: any Plug to be executed before sending the response. Quenya can generate a ResponseValidator if you need it. It's good for dev/staging purpose. By default it won't generate it.

Quenya consists of 3 parts:

1. quenya_installer: help with Quenya project generation (the CLI you just used).
2. quenya_builder: a code generator to generate API implementation based on extended OpenAPI v3 spec. Every time you run `mix compile`, Quenya will rebuild the spec to code (need improvement here).
3. quenya: a library consist of utility functions, tests and a playground to play with API or API stub.

## What's the generated code?

If you look at the `gen` folder in the newly generated app, you'll find all your routes and routers are organized by `operationId`:

```bash
$ tree -L 1
.
├── Petstore.Gen.ApiRouter.ex
├── Petstore.Gen.Router.ex
├── addPet
├── createUser
├── createUsersWithArrayInput
├── createUsersWithListInput
├── deleteOrder
├── deletePet
├── deleteUser
├── findPetsByStatus
├── findPetsByTags
├── getInventory
├── getOrderById
├── getPetById
├── getUserByName
├── loginUser
├── logoutUser
├── placeOrder
├── updatePet
├── updatePetWithForm
├── updateUser
└── uploadFile

20 directories, 2 files
```

The main router will serve swagger and forward the path (extracted from the spec) to the API router:

```elixir
defmodule Petstore.Gen.Router do
  @moduledoc false
  use Plug.Router
  use Plug.ErrorHandler
  require Logger
  alias Quenya.Plug.SwaggerPlug
  plug Plug.Logger, log: :info
  plug Plug.Static, at: "/public", from: {:quenya, "priv/swagger"}

  plug :match
  plug Plug.Parsers, parsers: [:json], pass: ["application/json"], json_decoder: Jason
  plug :dispatch

  def handle_errors(conn, %{kind: _kind, reason: %{message: msg}, stack: _stack}) do
    Plug.Conn.send_resp(conn, conn.status, msg)
  end

  def handle_errors(conn, %{kind: kind, reason: reason, stack: stack}) do
    Logger.warn(
      "Internal error:\n kind: #{inspect(kind)}\n reason: #{inspect(reason)}\n stack: #{
        inspect(stack)
      }"
    )

    Plug.Conn.send_resp(conn, conn.status, "Internal server error")
  end

  get("/swagger/main.json", to: SwaggerPlug, init_opts: [app: :petstore])
  get("/swagger", to: SwaggerPlug, init_opts: [spec: "/swagger/main.json"])
  forward "/", to: Petstore.Gen.ApiRouter, init_opts: []
end
```

The API router contains code for all routes, for example:

```elixir
put("/user/:username",
    to: RoutePlug,
    init_opts: [
      preprocessors: [Petstore.Gen.UpdateUser.RequestValidator],
      postprocessors: [],
      handlers: [Petstore.Gen.UpdateUser.FakeHandler]
    ]
  )
```

When a `PUT /user/:username` request kicks in, it will be handled by `Quenya.Plug.RoutePlug`, and it will run `preprocessors`, `handlers` and `postprocessors` in the right order.

## Why Quenya?

I've given a topic [Building next-gen APIs](docs/building-next-gen-apis.pdf) in 10/2020. The original idea is: why don't I build a code generator to generate API code that we don't need to write repeatedly? I always hold this tenet that everything could be generated should be generated.

Building a high-quality HTTP API app is non-trivial. Good APIs have these traits:

For API users:

- Easy to learn and intuitive to use (the app provides full-fledged and good quality docs / playground)
- Hard to misuse (API is type-safety and provides proper error responses)
- Powerful enough to drive business requirements (flexible, performant)
- Easy to evolve as the products grow
- Opinionated (don't make me think)

For developers:

- Easy to read and maintain existing code
- Easy to write new APIs / extend existing APIs
- Easy to generate code based on API spec (client SDKs, test cases, and even server implementation)

API implementation is just a small part of the API lifecycle, we need API design, mocking, testing, simulating, documentation, deployment, etc.

![](docs/images/api.png)

Quenya tries to help you start with the API spec, iterate it without writing the code, while at the same time various teams can play with the mocking server based on the spec to nail down what is actually needed. We believe this is the best approach to improve productivity.

## Why not GraphQL or other solutions?

See the above slides and you'll see why.
 readmeEtag: '"3f7fdbde8450a026158c1a5bbbe80f499359b0f7"' readmeLastModified: Thu, 17 Dec 2020 06:37:01 GMT repositoryId: 316165371 description: >- Quenya is a framework to build high-quality REST API applications based on extended OpenAPI spec created: '2020-11-26T08:20:54Z' updated: '2025-01-20T10:19:02Z' language: Elixir archived: false stars: 146 watchers: 4 forks: 14 owner: tyrchen logo: https://avatars.githubusercontent.com/u/473888?v=4 license: MIT repoEtag: '"bcbe57ec38f47cf99e16792392f918715928c8828f793dddc277feed8623dfd3"' repoLastModified: Mon, 20 Jan 2025 10:19:02 GMT foundInMaster: true category: Server Implementations id: c5ab6433c14ab86f1eea6e5d29b5e111 - source: - openapi3 tags - openapi31 tags repository: https://github.com/oai/learn.openapis.org v3: true v3_1: true id: bc6087c180d696bd2a572b177febd681 repositoryMetadata: base64Readme: >- IyBIb21lCgo8aW1nIGFsdD0iT3BlbkFQSSBJbml0aWF0aXZlIiBzcmM9ImFzc2V0cy9pbWFnZXMvT3BlbkFQSV9Mb2dvX1BhbnRvbmUtMS5wbmciIHdpZHRoPSIxMDAlIiBoZWlnaHQ9ImF1dG8iPgoKQXJlIHlvdSBuZXcgdG8gdGhlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbj8KClJlYWQgb3VyIFtHZXR0aW5nIFN0YXJ0ZWRdKGh0dHBzOi8vbGVhcm4ub3BlbmFwaXMub3JnLykgcGFnZSBmaXJzdC4KCjxwPk9yaWdpbmFsIHNpdGUgY3VzdG9taXplZCBiYXNlZCBvbiBhIHRoZW1lIGJ5IFBhdHJpY2sgTWFyc2NlaWxsLCBkaXN0cmlidXRlZCBieSBhbiA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vcG1hcnNjZWlsbC9qdXN0LXRoZS1kb2NzL3RyZWUvbWFzdGVyL0xJQ0VOU0UudHh0Ij5NSVQgbGljZW5zZS48L2E+PC9wPgoKIyMgRGV2ZWxvcG1lbnQKClRvIHJ1biB0aGlzIHdlYnNpdGUgbG9jYWxseSB5b3UnbGwgbmVlZCB0byB1c2UgSmVreWxsLgoKUGxlYXNlIGZvbGxvdyB0aGUgb2ZmaWNpYWwgW1F1aWNrc3RhcnRdKGh0dHBzOi8vamVreWxscmIuY29tL2RvY3MvKSBndWlkZSBmb3IgbW9yZSBpbmZvcm1hdGlvbiBpZiB5b3UgYXJlIHN0YXJ0aW5nIGZyb20gc2NyYXRjaCwgYW5kIHRoZW4gZm9sbG93IHRoZSBpbnN0cnVjdGlvbnMgaW4gdGhlIFtDb250cmlidXRpbmcgR3VpZGVdKC4vQ09OVFJJQlVUSU5HLm1kKS4K readmeEtag: '"b5219351d8369a2617c1d7eadc12b8d80b737f68"' readmeLastModified: Tue, 04 Jun 2024 11:56:58 GMT repositoryId: 285621457 description: OpenAPI - Getting started, and the specification explained created: '2020-08-06T16:40:47Z' updated: '2026-01-30T16:44:10Z' language: Markdown archived: false stars: 200 watchers: 19 forks: 101 owner: OAI logo: https://avatars.githubusercontent.com/u/16343502?v=4 license: CC-BY-4.0 repoEtag: '"1e688c2346f972df8fc0082d8b0d5df57931113270874385346daf3473658300"' repoLastModified: Fri, 30 Jan 2026 16:44:10 GMT category: - Parsers - Server Implementations oldLocations: - https://github.com/oai/documentation foundInMaster: true - source: openapi3 tags name: angular-swagger-ui homepage: https://github.com/Orange-OpenSource/angular-swagger-ui language: AngularJS source_description: An angularJS implementation of Swagger UI category: - User Interfaces - Parsers repository: https://github.com/orange-opensource/angular-swagger-ui v3: true repositoryMetadata: base64Readme: >- # angular-swagger-ui

`angular-swagger-ui` is an angularJS implementation of OpenAPI UI

[OpenAPI](https://www.openapis.org) (aka [Swagger](https://swagger.io)) helps you documenting your RESTful API.

OpenAPI UI helps developers discovering your RESTful API by providing an online documentation with an integrated API explorer.

### Warning 
> By default, only OpenAPI 2.0 is supported.
To handle OpenAPI 3.0.0 please add module `openapi3-converter` see [Enable OpenAPI 3.0.0](#enable-openapi-300).
To handle OpenAPI 1.2 please add module `swagger1-converter` see [Enable OpenAPI 1.2](#enable-openapi-12).
To handle authorization please add module `swagger-auth` see [Enable authorization](#enable-authorization)
To handle YAML please add module `swagger-yaml-parser` see [Enable YAML](#enable-yaml)

## Demo

A sample app using `angular-swagger-ui` is available here:

http://orange-opensource.github.io/angular-swagger-ui

## Quick Start

### Install

`npm install angular-swagger-ui`

### Dependencies

1. [angularJS](https://angularjs.org)
2. [bootstrap CSS](http://getbootstrap.com)
3. [angular-ui-bootstrap](https://angular-ui.github.io/bootstrap/) (required only if using [Authorization](#enable-authorization))

## License

All code in this repository is covered by the [MIT license](http://opensource.org/licenses/MIT).
See LICENSE file for copyright details.

## Getting Started

Include `angular-swagger-ui` as a dependency into your application

As some properties of OpenAPI specifications can be formatted as HTML:

* You **SHOULD** include `ngSanitize` as a dependency into your application (avoids JS injection) if OpenAPI specifications are loaded from **untrusted** sources (see `dist/index.html` as an example)
* You **CAN** add `trusted-sources="true"` as directive parameter (avoids embedding `ngSanitize`) if OpenAPI specifications are loaded from **trusted** sources (see `src/index.html` as an example)
* You **MUST** at least choose one of the two previous solutions

```html
<script type="text/javascript">
	// If directive has parameter trusted-sources="true"
	angular.module('yourApp', ['swaggerUi']);
	...
	// OR if you choosed to use "ngSanitize"
	angular.module('yourApp', ['ngSanitize', 'swaggerUi']);
	...
</script>
```
Create an HTML element in your angularJS application's template or in your HTML page
```html
<div swagger-ui url="URLToYourOpenAPISpecification" api-explorer="true"></div>
```
Add `swagger-ui.min.js` and `angular.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<!-- if you choosed to use "ngSanitize" -->
 	<script src="yourPathToAngularSanitize/angular-sanitize.min.js"></script>
</body>
```
Add `swagger-ui.min.css` and `bootstrap.min.css` to the head of the HTML page.
```html
<body>
	<head>
		...
		<link rel="stylesheet" href="yourPathToBootstrapCSS/bootstrap.min.css">
		<link rel="stylesheet" href="yourPathToAngularSwaggerUI/dist/css/swagger-ui.min.css">
  	</head>
</body>
```

## Parameters

#### API explorer
Display or not API explorer, default is `false`
```html
<div swagger-ui url="URLToYourOpenAPISpecification" api-explorer="true/false"></div>
```

#### OpenAPI specification loading indicator
`yourScopeVariable` will be assigned to `true` or `false` depending on OpenAPI specification loading status
```html
<div ng-show="yourScopeVariable">loading ...</div>
<div swagger-ui url="URLToYourOpenAPISpecification" loading="yourScopeVariable"></div>
```

#### Error handler
Define an error handler to catch errors, if none defined `console.error` is used
```html
<div swagger-ui url="URLToYourOpenAPISpecification" error-handler="yourErrorHandler"></div>
```
```js
$scope.yourErrorHandler = function(/*String or Object*/ message, /*Integer*/ code){
	
}
```

#### Permalinks
Allows having a URL direct access to a group of operations or to an operation and making it unfolded at startup
```html
<div swagger-ui url="URLToYourOpenAPISpecification" permalinks="true/false"></div>
```

#### Download
Display or not a link to download swagger file. 

```html
<!-- display link with url label -->
<div swagger-ui url="URLToYourOpenAPISpecification" download></div>

<!-- display link with specific key enter in swaggerTranslatorProvider -->
<div swagger-ui url="URLToYourOpenAPISpecification" download="downloadLabel"></div>
```

#### OpenAPI validator
Disable OpenAPI validator or define a custom OpenAPI validator.
If parameter not defined, the validator will be 'http://online.swagger.io/validator'
```html
<div swagger-ui url="URLToYourOpenAPISpecification" validator-url="false or URL"></div>
```

#### Parser type
OpenAPI specification parser is chosen depending on the `Content-Type` of the specification response. If host serving your OpenAPI specification does not send `Content-Type: application/json` then you can force the parser to JSON:
```html
<div swagger-ui url="URLToYourOpenAPISpecification" parser="json"></div>
```

#### Template URL
Define a custom template to be used by OpenAPIUI
```html
<div swagger-ui url="URLToYourOpenAPISpecification" template-url="yourTemplatePath"></div>
```

#### Inherited properties
Allows displaying inherited properties of polymorphic models
```html
<div swagger-ui url="URLToYourOpenAPISpecification" show-inherited-properties="true/false"></div>
```

#### Input type and input
##### Render an OpenAPI specification from JSON object
```html
<div swagger-ui input-type="json" input="yourJsonObject"></div>
```

##### Render an OpenAPI specification from YAML string
Make sure to use module `swagger-yaml-parser`, see [Enable YAML](#enable-yaml)
```html
<div swagger-ui input-type="yaml" input="yourYamlString"></div>
```

##### Render an OpenAPI specification from URL (same behavior as using "url" parameter)
```html
<div swagger-ui input-type="url" input="yourURL"></div>
```

## i18n

#### Built-in languages
`angular-swagger-ui` is available in english and french, english is used by default

To use french, add `fr.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/i18/fr.min.js"></script>
</body>
```
Set language to french at startup
```html
<script type="text/javascript">
	angular
		.module('yourApp', ['swaggerUi'])
		.config(function(swaggerTranslatorProvider) {
			swaggerTranslatorProvider.setLanguage('fr');
		});
	...
</script>
```
Set language to french at runtime
```html
<script type="text/javascript">
	angular
		.module('yourApp', ['swaggerUi'])
		.controller('yourController', function(swaggerTranslator) {
			swaggerTranslator.useLanguage('fr');
		});
	...
</script>
```

#### Add languages
You can add your own languages, see `src/scripts/i18n/en.js` to find the keys you have to override
```html
<script type="text/javascript">
	angular
		.module('yourApp', ['swaggerUi'])
		.config(function(swaggerTranslatorProvider) {
			swaggerTranslatorProvider.addTranslations('yourLangId', {
				key: 'value'
				...
			});
			swaggerTranslatorProvider.setLanguage('yourLangId');
		});
	...
</script>
```

#### Internationalize your app
You can also use `swaggerTranslator` to internationalize your app by using a service, a directive or a filter
```html
<body>
 	...
 	<div swagger-translate="yourKey" swagger-translate-value="yourParam"></div>
 	<div ng-bind="yourDynamicKey|swaggerTranslate:yourDynamicParam"></div>
 	...
	<script type="text/javascript">
		angular
			.module('yourApp', ['swaggerUi'])
			.config(function(swaggerTranslatorProvider) {
				swaggerTranslatorProvider.addTranslations('en', {
					yourKey: 'blablabla {{propertyNameOfYourParam}}'
					...
				});
			})
			.controller('yourController', function(swaggerTranslator) {
				var localizedMessage = swaggerTranslator.translate('yourKey', yourParam);
			});
		...
	</script>
</body>
```

## Customization

#### Enable OpenAPI 3.0.0
See [OpenAPI 3.0.0 spec](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md).
Add `openapi3-converter.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/openapi3-converter.min.js"></script>
</body>
```

#### Enable authorization
`oauth` is not implemented, only `basic` and `API key` authorizations are implemented.
Add `swagger-auth.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger-auth.min.js"></script><!-- without angular-ui-bootstrap modal embedded -->
 	OR
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger-auth-ui-boostrap-modal.min.js"></script><!-- angular-ui-bootstrap modal embedded -->
 	...
	<script type="text/javascript">
		angular
			.module('yourApp', ['swaggerUi', 'swaggerUiAuthorization'])
			// what is below is required for oauth2 flows 'implicit' and 'accessCode' (ie. authorizationCode)
			// what is below can also be used to initialize apiKey or Basic authorizations
      .config(function(swaggerUiAuthProvider) {
          swaggerUiAuthProvider.configuration({
              // required for oauth2 flow 'implicit' and 'accessCode' (ie. authorizationCode)
             	redirectUrl: 'yourPathToAngularSwaggerUI/oauth2-redirect.html' 
              // optional
              yourSecurityName: {
              	apiKey: 'yourApiKeyValue' // optional, can be used to initialize api key value
              },
              // optional
              yourSecurityName: {
              	login: 'yourLogin', // optional, can be used to initialize basic login
              	password: 'yourPassword' // optional, can be used to initialize basic password
              },
              // optional
              yourSecurityName: {
              	clientId: 'yourClientId', // optional, can be used to initialize oauth2 credentials
              	clientSecret: 'yourClientSecret', // optional, can be used to initialize oauth2 credentials
              	login: 'yourLogin', // optional, can be used to initialize oauth2 credentials
              	password: 'yourPassword', // optional, can be used to initialize oauth2 credentials
              	scopeSeparator: 'scopeSeparator', // optional, can be used to configure oauth2 scopes separator, default value is space
              	// optional, can be used to configure oauth2 additional query params to tokenUrl and authorizationUrl
              	queryParams: {
              		'yourQueryParamName': 'yourQueryParamValue'
              		...
              	}, 
              },
          });
      })
			...
	</script>
</body>
```

#### Enable OpenAPI [aka Swagger] 1.2
See [OpenAPI 1.2 spec](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/1.2.md).
Add `swagger1-converter.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger1-converter.min.js"></script>
</body>
```

#### Enable OpenAPI external references
See [OpenAPI 2.0 spec](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#relative-schema-file-example).
Add `swagger-external-references.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger-external-references.min.js"></script>
</body>
```

#### Enable XML formatter on API explorer responses
Add `swagger-xml-formatter.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger-xml-formatter.min.js"></script>
</body>
```

#### Enable YAML
Add [js-yaml library](https://cdnjs.com/libraries/js-yaml).
Add `swagger-yaml-parser.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToJsYaml/js-yaml.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger-yaml-parser.min.js"></script>
</body>
```

#### Enable markdown
Add [marked library](https://cdnjs.com/libraries/marked).
Add `swagger-markdown.min.js` at the end of the body
```html
<body>
 	...
 	<script src="yourPathToAngularJS/angular.min.js"></script>
 	<script src="yourPathToMarked/marked.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/swagger-ui.min.js"></script>
 	<script src="yourPathToAngularSwaggerUI/dist/scripts/modules/swagger-markdown.min.js"></script>
</body>
```

#### Writing your own modules
Modifying `angular-swagger-ui` can be achieved by writing your own modules. As an example your can have a look at the ones in `src/scripts/modules`.
A module is an object (can be a service) having a function `execute` which must return a promise.

You can make your module modifying behaviours at different phases:

* `BEFORE_LOAD`: allows modifying OpenAPI specification request before it is sent
* `BEFORE_PARSE`: allows modifying OpenAPI specification after it has been loaded
* `PARSE`: allows adding an OpenAPI parser for content types other than JSON
* `BEFORE_DISPLAY`: allows modifying internal parsed OpenAPI specification before it is displayed
* `BEFORE_EXPLORER_LOAD`: allows modifying API explorer request before it is sent
* `AFTER_EXPLORER_LOAD`: allows modifying API explorer response before it is displayed

```js
angular
	.module('myApp', ['swaggerUi'])
	.service('myModule', function($q) {

		this.execute = function(data) {
			var deferred = $q.defer();
			// if nothing done: call deferred.resolve(false);
			// if success: call deferred.resolve(true);
			// if error: call deferred.reject({message: 'error message', code: 'error_code'});
			return deferred.promise;
		}

	})
	.run(function(swaggerModules, myModule){
		// default priority is 1
		// higher is the priority, sooner the module is executed at the specified phase
		swaggerModules.add(swaggerModules.BEFORE_LOAD, myModule, priority);
	})
	...

```
 readmeEtag: '"4f2c75d8482865d9e7949580790c5bec602563fe"' readmeLastModified: Mon, 05 Oct 2020 09:54:38 GMT repositoryId: 31470522 description: An angularJS implementation of Swagger UI created: '2015-02-28T17:44:11Z' updated: '2024-11-29T17:18:55Z' language: JavaScript archived: true stars: 136 watchers: 20 forks: 71 owner: Orange-OpenSource logo: https://avatars.githubusercontent.com/u/1506386?v=4 license: MIT repoEtag: '"d8bb2db1085ee990aa302386a8c358b6f83056d91ac5be11b35385bf28efe9ab"' repoLastModified: Fri, 29 Nov 2024 17:18:55 GMT foundInMaster: true id: 4db6178272e1a2c2ca2eef009acc2dd1 - source: openapi3 tags repository: https://github.com/sourcemeta/awesome-jsonschema v3: true repositoryMetadata: base64Readme: >- <!-- ATTENTION! This file is auto-generated. Do not edit this file directly.
Instead, edit `data.yaml` and re-generate this file as per the README's
instructions -->

# Awesome JSON Schema [![Awesome](https://cdn.rawgit.com/sindresorhus/awesome/d7305f38d29fed78fa85652e3a63e154dd8e8829/media/badge.svg)](https://github.com/sindresorhus/awesome)

[<img src="logo.svg" align="right" width="100">](https://json-schema.org)

[![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg?style=flat-square)](http://makeapullrequest.com)
[![Join Slack](https://img.shields.io/badge/Slack-Join%20Slack-blue.svg?style=flat-square)](https://json-schema.slack.com)
[![OpenCollective](https://img.shields.io/opencollective/all/json-schema?label=OpenCollective&style=flat-square)](https://opencollective.com/json-schema)

> A curated list of awesome JSON Schema resources, tutorials, tools, and more.

JSON Schema is a JSON-based format to annotate and validate JSON documents with
a vibrant community. JSON Schema is defined by a set of IETF specifications and
it is the industry-standard for defining the structure and meaning of JSON
documents.

---

**Would you like to promote your company or product here? [Sponsor us on
GitHub](https://github.com/sponsors/sourcemeta)**

---

## Contents

- [Getting Started](#getting-started)
- [Courses](#courses)
- [Development Tools](#development-tools)
- [Books](#books)
- [Registries](#registries)
- [Articles](#articles)
- [Related Specifications](#related-specifications)
- [Videos](#videos)
- [Papers](#papers)
- [Libraries](#libraries)

## Getting Started

- [Learn JSON Schema](https://www.learnjsonschema.com?utm_source=awesome-jsonschema) - A comprehensive JSON Schema documentation website covering all specification versions.
- [JSON Schema Tour](https://tour.json-schema.org?utm_source=awesome-jsonschema) - An interactive tutorial to learn JSON Schema step by step.

## Courses

- [Master JSON Schema for OpenAPI](https://www.sourcemeta.com/courses/jsonschema-for-openapi/?utm_source=awesome-jsonschema) - A comprehensive 9+ hour video course teaching advanced JSON Schema techniques for API design, covering dynamic references, unevaluated properties, schema composition, testing, linting, and deployment to registries.

## Development Tools

- [AlterSchema](https://alterschema.sourcemeta.com?utm_source=awesome-jsonschema) - Convert a JSON Schema definition between specification versions.
- [JSON Schema CLI](https://github.com/sourcemeta/jsonschema?utm_source=awesome-jsonschema) - A comprehensive command-line tool for working with JSON Schema supporting formatting, linting, testing, bundling, and validation across all JSON Schema versions.
- [JSONBuddy](https://www.json-buddy.com?utm_source=awesome-jsonschema) - A JSON editor and validator desktop application for Windows.
- [Sourcemeta Studio](https://github.com/sourcemeta/studio?utm_source=awesome-jsonschema) - A Visual Studio Code extension providing professional JSON Schema tooling with real-time linting, automatic formatting, and metaschema validation.

## Books

- (2024) [Unifying Business, Data, and Code: Designing Data Products with JSON Schema](https://www.oreilly.com/library/view/unifying-business-data/9781098144999/?utm_source=awesome-jsonschema) - Covers topics such as writing your own JSON Schema vocabularies, understanding JSON Schema annotations, and hosting your own JSON Schema registries. More importantly, our book teaches you a methodology for effective data management.
- (2021) [API by Design](https://smizell.gumroad.com/l/apibydesign?utm_source=awesome-jsonschema) - Introduces an approach to measure API complexity by analyzing entropy in JSON Schema definitions.
- (2017) [JSON at Work](https://www.oreilly.com/library/view/json-at-work/9781491982389/?utm_source=awesome-jsonschema) - A comprehensive overview of the JSON ecosystem, including JSON Schema.
- (2014) [Using JSON Schema](https://books.apple.com/us/book/using-json-schema/id903248630?utm_source=awesome-jsonschema) - Learn and Apply JSON Schema by Example, with JavaScript (Node.js) and Python Programs.

## Registries

- [Apicurio Registry](https://www.apicur.io/registry/?utm_source=awesome-jsonschema) - A runtime server system for storing and managing API designs and schemas including OpenAPI, AsyncAPI, Avro, and JSON Schema with configurable content rules for evolution control.
- [Sourcemeta One](https://one.sourcemeta.com?utm_source=awesome-jsonschema) - A self-hosted JSON Schema microservice that transforms Git repositories into searchable, discoverable schema catalogs with a web explorer, editor integration, schema health checks, and a rich HTTP API.
- [Sourcemeta Schemas](https://schemas.sourcemeta.com?utm_source=awesome-jsonschema) - A public free instance of Sourcemeta One re-offering various open source schema collections.

## Articles

- (2023) [LLMs for Schema Augmentation](https://michael.mior.ca/blog/llms-for-schema-augmentation/?utm_source=awesome-jsonschema) - Large language models can be exploited to automatically augment JSON Schemas with useful information.
- (2022) [What is &quot;Modern&quot; JSON Schema?](https://modern-json-schema.com/what-is-modern-json-schema?utm_source=awesome-jsonschema) - How &quot;modern&quot; JSON Schema, meaning 2019-09, 2020-12, and later, can solve far more problems than &quot;classical&quot; JSON Schema (draft-07 and earlier).
- (2022) [Using Dynamic References to Support Generic Types](https://json-schema.org/blog/posts/dynamicref-and-generics?utm_source=awesome-jsonschema) - Representing generic types from strongly-typed programming languages with JSON Schema.
- (2022) [JSON Schema is a constraint system](https://modern-json-schema.com/json-schema-is-a-constraint-system?utm_source=awesome-jsonschema) - Or: Why object-oriented programming is a mis-matched mental model for JSON Schema.
- (2022) [Forming new relationships: Contributing to Open source](https://www.crossref.org/blog/forming-new-relationships-contributing-to-open-source/?utm_source=awesome-jsonschema) - A description of adopting and extending JSON Forms, a JSON Schema based form rendering library, to support VueJS and Vuetify..
- (2022) [Docs! Docs! Docs!](https://brandur.org/nanoglyphs/031-api-docs?utm_source=awesome-jsonschema) - An overview of API documentation generation using Hyper Schema and OpenAPI.
- (2021) [Understanding JSON Schema compatibility](https://yokota.blog/2021/03/29/understanding-json-schema-compatibility/?utm_source=awesome-jsonschema) - A study of open, closed, and partially-open content models when evolving schemas in a backward, forward, or fully compatible manners.
- (2021) [Understanding API Complexity Through Schema Entropy](https://smizell.com/posts/2021/05/understanding-api-complexity-through-schema-entropy/?utm_source=awesome-jsonschema) - A look at API complexity by analyzing entropy of JSON Schema documents.
- (2021) [Robustness, Tolerance, and JSON Schema](https://smizell.com/posts/2021/07/robustness-tolerance-and-json-schema/?utm_source=awesome-jsonschema) - Applying the Robustness Principle and increasing schema entropy when making changes to an API.
- (2021) [Modeling a file system with JSON Schema](https://json-schema.org/learn/file-system.html?utm_source=awesome-jsonschema) - This example shows a possible JSON Schema representation of file system mount points as represented in an /etc/fstab file.
- (2021) [JSON Schema bundling finally formalised](https://json-schema.org/blog/posts/bundling-json-schema-compound-documents?utm_source=awesome-jsonschema) - A close look at bundling and de-referencing JSON Schema documents.
- (2020) [Using the JSON Schema standard for scientific applications?](https://cerfacs.fr/coop/json-schema-for-sci-apps?utm_source=awesome-jsonschema) - A discussion on how to use JSON Schema to validate input, add precise documentation, auto-fill missing parts, and create graphical user interfaces in the context of scientific applications.
- (2020) [Project Cambria: Translate your data with lenses](https://www.inkandswitch.com/cambria/?utm_source=awesome-jsonschema) - A proposal for an isolated software layer that translates data between JSON Schema definitions on demand. Translation logic is defined by composing bidirectional lenses, a kind of data transformation that can run both forward and backward.
- (2020) [JSON Schema, Schema.org, JSON-LD: What&#x27;s the Difference?](https://dashjoin.medium.com/json-schema-schema-org-json-ld-whats-the-difference-e30d7315686a?utm_source=awesome-jsonschema) - An overview of the difference between JSON Schema and JSON-LD, their relationship and the role of Schema.org.
- (2020) [Building a No-Code JSON Schema Form Builder with ReactJS](https://www.ginkgobioworks.com/2020/10/08/building-a-no-code-json-schema-form-builder/?utm_source=awesome-jsonschema) - An update on the additional features implemented in the react-json-schema-form-builder open-source JSON Schema form builder project.
- (2020) [Azure Pipelines autocomplete in PyCharm, IntelliJ, WebStorm, CLion, and Rider](https://tonybaloney.github.io/posts/azure-pipelines-autocomplete-in-pycharm.html?utm_source=awesome-jsonschema) - Setting up PyCharm, IntelliJ, WebStorm, CLion and Rider to have auto-complete, syntax highlighting and validation support of Azure Pipelines workflows.
- (2019) [Saved by the Schema: Using JSON Schema to Document, Test, and Debug APIs](https://blog.heroku.com/json-schema-document-debug-apis?utm_source=awesome-jsonschema) - Learn how Heroku uses JSON Schema to test and document their Platform API, and how it helped them uncover an unexpected bug, rooted in the way the Oj gem parses Big Decimals.
- (2018) [JSON Schema Validator, Generator &amp; Editor Guide](https://stoplight.io/json-guide/?utm_source=awesome-jsonschema) - How the JSON and JSON Schema standards are defined and how to put them to use in your code and in your APIs.
- (2018) [JSON Schema Validation &amp; Expressive Query Syntax in MongoDB 3.6](https://www.sitepoint.com/json-schema-validation-expressive-query-syntax-in-mongodb-3-6/?utm_source=awesome-jsonschema) - An in-depth discussion about using JSON Schema to define collection validation on MongoDB.
- (2018) [Definitions for filtering properties in JSON schema](https://gist.github.com/LucianBuzzo/5ff9106ce3ae12fb58e4def572b9344d?utm_source=awesome-jsonschema) - An introduction to the problem of using JSON Schema for data filtering and querying purposes.
- (2014) [Introducing SchemaVer for semantic versioning of schemas](https://snowplowanalytics.com/blog/2014/05/13/introducing-schemaver-for-semantic-versioning-of-schemas/#feedback?utm_source=awesome-jsonschema) - A proposal for a versioning convention for JSON Schema definitions based on SemVer.

## Related Specifications

- [Agent2Agent Protocol (A2A)](https://a2a-protocol.org?utm_source=awesome-jsonschema) - An open protocol by Google enabling communication and interoperability between agentic applications. A2A uses JSON-RPC 2.0 over HTTP and JSON Schema for defining Agent Cards and message structures.
- [AsyncAPI](https://www.asyncapi.com?utm_source=awesome-jsonschema) - AsyncAPI is an open source initiative that seeks to improve the current state of Event-Driven Architectures (EDA). The AsyncAPI specification supports data modeling using JSON Schema.
- [JSON Schema in RDF](https://www.w3.org/2019/wot/json-schema?utm_source=awesome-jsonschema) - This document introduces an RDF vocabulary for JSON Schema definitions. This vocabulary provides a stable namespace IRI for JSON Schema keywords, as well as simple axioms, defined against schema.org&#x27;s meta-model.
- [Model Context Protocol (MCP)](https://modelcontextprotocol.io?utm_source=awesome-jsonschema) - An open standard by Anthropic that enables AI systems like large language models to integrate with external tools and data sources. MCP uses JSON-RPC 2.0 and JSON Schema for defining server capabilities.
- [OpenAPI](https://www.openapis.org?utm_source=awesome-jsonschema) - The OpenAPI Specification embeds and extends JSON Schema for defining API requests and responses.
- [RAML](https://raml.org?utm_source=awesome-jsonschema) - The RAML specification supports modeling API data using JSON Schema.
- [REST API Linked Data Keywords](https://www.ietf.org/archive/id/draft-polli-restapi-ld-keywords-07.html?utm_source=awesome-jsonschema) - An Internet-Draft proposing JSON Schema keywords to attach semantic information to OpenAPI and JSON Schema documents, enabling contract-first API design with RDF type information and JSON-LD context.
- [Semantic Definition Format (SDF)](https://datatracker.ietf.org/doc/draft-ietf-asdf-sdf/?utm_source=awesome-jsonschema) - An IETF specification for modeling Internet of Things devices and their interactions through Properties, Actions, and Events. SDF uses JSON to represent definitions and incorporates JSON Schema for data validation.
- [W3C Web of Things](https://www.w3.org/WoT/?utm_source=awesome-jsonschema) - The Web of Things (WoT) seeks to counter the fragmentation of the IoT by using and extending existing, standardized Web technologies. WoT models data using JSON Schema.

## Videos

For more video content, check out the [official JSON Schema YouTube channel](https://www.youtube.com/channel/UCrl3fjmHSp6FhJUKcEJhisA) and the [JSON Schema Conference](https://conference.json-schema.org) website.

- (2021) [So you think you understand JSON Schema? - Ben Hutton](https://www.youtube.com/watch?v=vMG0NCDifI0?utm_source=awesome-jsonschema) - In this session you&#x27;ll learn some key fundamentals, intracacies that even catch out the experienced, and how to develop your own interoperable JSON Schema Vocabulary.
- (2021) [Maintaining JSON Schemas at Scale - Jason Desrosiers](https://www.youtube.com/watch?v=GjJpRsVffg0?utm_source=awesome-jsonschema) - In this talk, we will cover the keywords, concepts, and patterns for maintaining large schemas or large collections of schemas. We&#x27;ll tackle problems like how to breakdown and organize schemas, how to bundle schemas, how to make large schemas efficient, how to get better error messages from large schemas, and more.
- (2021) [JSON Schema Validation in Postman](https://www.youtube.com/watch?v=8BfshV5n6ac?utm_source=awesome-jsonschema) - An tutorial of performing JSON Schema validation in Postman in API tests.
- (2021) [Configuring Umbraco on .NET Core - JSON Schema](https://www.youtube.com/watch?v=rpUg-oySw8g?utm_source=awesome-jsonschema) - Configuring Umbraco on .NET Core with JSON Schema-powered autocompletions for appsettings.json using SchemaStore.
- (2021) [API Storytelling with Ben Hutton](https://www.youtube.com/watch?v=4xbA82lo_lc?utm_source=awesome-jsonschema) - This episode we sit down with Ben Hutton, the lead behind the JSON Schema community to talk about vocabularies and other ways we validate and annotate how we tell stories we tell in the API space.
- (2020) [What is JSON Schema](https://www.youtube.com/watch?v=kK-_gL7Vsc0?utm_source=awesome-jsonschema) - A basic introduction to JSON Schema showing how to auto-generate JSON Schema document from an existing JSON document.
- (2019) [What is a JSON Schema? Generate, Modify, and Understand a JSON Schema: Example](https://www.youtube.com/watch?v=hGXxXyJmaUo?utm_source=awesome-jsonschema) - An in-depth introduction to JSON Schema including auto-generating JSON Schema documents using QuickType.io.
- (2019) [JSON Schema Validation: How to Validate JSON Schema with Postman?](https://www.youtube.com/watch?v=X072eKtOIio?utm_source=awesome-jsonschema) - An introduction to JSON Schema and how to use it in Postman.
- (2017) [JSON Schema - the Good, the Bad and the Ugly](https://vimeo.com/222489288?utm_source=awesome-jsonschema) - With JSON Schema you are required to define your data and the available tooling to do so is not really great (yet). This presentation provides you with criteria to judge whether JSON Schema could be beneficial for your project or is just too much overhead.

## Papers

- (2025) [Blaze: Compiling JSON Schema for 10x Faster Validation](https://arxiv.org/abs/2503.02770?utm_source=awesome-jsonschema) - This paper introduces Blaze, a JSON Schema validator compiles complex schemas to an efficient representation in seconds to minutes, adding minimal overhead at build time. Blaze incorporates several unique optimizations to reduce the validation time by an average of approximately 10x compared existing validators on a variety of datasets. In some cases, Blaze achieves a reduction in validation time of multiple orders of magnitude compared to the next fastest validator. We also demonstrate that several popular validators produce incorrect results in some cases, while Blaze maintains strict adherence to the JSON Schema specification..
- (2025) [Elimination of annotation dependencies in validation for Modern JSON Schema](https://arxiv.org/abs/2503.11288?utm_source=awesome-jsonschema) - This paper proves that the elimination of annotation dependent keywords cannot, in general, avoid an exponential increase of the schema dimension. We provide an algorithm to eliminate these keywords that, despite the theoretical lower bound, behaves quite well in practice, as we verify with an extensive set of experiments..
- (2025) [JSONSchemaBench: A Rigorous Benchmark of Structured Outputs for Language Models](https://arxiv.org/abs/2501.10868?utm_source=awesome-jsonschema) - This paper introduces JSONSchemaBench, a benchmark for constrained decoding comprising 10K real-world JSON schemas that encompass a wide range of constraints with varying complexity. We pair the benchmark with the existing official JSON Schema Test Suite and evaluate six state-of-the-art constrained decoding frameworks, including Guidance, Outlines, Llamacpp, XGrammar, OpenAI, and Gemini. Through extensive experiments, we gain insights into the capabilities and limitations of constrained decoding on structured generation with real-world JSON schemas..
- (2024) [Validation of Modern JSON Schema: Formalization and Complexity](https://arxiv.org/abs/2307.10034?utm_source=awesome-jsonschema) - In this paper, we give the first formal description of Modern JSON Schema, which we consider a central contribution of the work that we present here. We then prove that its data validation problem is PSPACE-complete. We prove that the origin of the problem lies in dynamic references, and not in annotation-dependent validation. We study the schema and data complexities, showing that the problem is PSPACE-complete with respect to the schema size even with a fixed instance, but is in PTIME when the schema is fixed and only the instance size is allowed to vary. Finally, we run experiments that show that there are families of schemas where the difference in asymptotic complexity between dynamic and static references is extremely visible, even with small schemas..
- (2023) [An Analysis of Defects in Public JSON Schemas](https://www.cri.minesparis.psl.eu/classement/doc/A-794.pdf?utm_source=awesome-jsonschema) - Analysis of common defects found in publicly available schemas leading to recommend changes to the spec.
- (2023) [Comprehending Semantic Types in JSON Data with Graph Neural Networks](https://arxiv.org/abs/2307.12807?utm_source=awesome-jsonschema) - Graph neural networks for semantic type detection in JSON.
- (2023) [JSONoid: Distributed JSON Schema Discovery](https://github.com/dataunitylab/jsonoid-discovery?utm_source=awesome-jsonschema) - A tool for distributed JSON schema discovery including many properties of the data.
- (2023) [JSONoid: Monoid-based Enrichment for Configurable and Scalable Data-Driven Schema Discovery](https://arxiv.org/abs/2307.03113?utm_source=awesome-jsonschema) - Meaningful schema information for semi-structured data.
- (2022) [Implicit JSON Schema Versioning Triggered by Temporal Updates to JSON-Based Big Data in the τJSchema Framework](https://link.springer.com/chapter/10.1007/978-3-031-07969-6_3?utm_source=awesome-jsonschema) - This paper proposes an approach for handling implicit schema changes triggered by temporal updates of JSON-based Big Data. More precisely, when a user specifies a temporal JSON update operation that modifies a snapshot JSON component assigning a valid-time timestamp to its new value, the execution of such an operation requires the JSON component to become temporal, which is for all intents a schema change. Thus, a new version of the τJSchema temporal characteristics document is generated, with the addition of a new valid-time characteristic. New versions of the temporal JSON schema and of the temporal JSON document are also accordingly created.
- (2022) [JSON BinPack: A space-efficient schema-driven and schema-less binary serialization specification based on JSON Schema](https://www.jviotti.com/assets/dissertation.pdf?utm_source=awesome-jsonschema) - A survey and benchmark of JSON-compatible binary serialization specifications followed by the introduction of JSON BinPack, a novel protocol-independent schema-driven and schema-less binary serialization specification that is strictly-compatible with JSON and takes advantage of JSON Schema formal definitions to produce bit-strings that are space-efficient in comparison to every considered alternative serialization specification.
- (2022) [Machine actionable metadata models](https://pmc.ncbi.nlm.nih.gov/articles/PMC9525592/?utm_source=awesome-jsonschema) - This paper discussed the use of JSON Schema to define human and machine readable metadata models.
- (2022) [Negation-Closure for JSON Schema](https://arxiv.org/abs/2202.13434?utm_source=awesome-jsonschema) - Examines how JSON Schema handles negation, demonstrates that the language lacks negation closure, explores how recent schema drafts address this limitation, and proposes enrichments to the language. Includes an algebraic reformulation of JSON Schema and a prototype system for generating schema witnesses.
- (2022) [The Usage of Negation in Real-World JSON Schema Documents](http://bilioso.isti.cnr.it/sebd2022/preprint/SEBD_2022_Camera-ready_PDF_8.pdf?utm_source=awesome-jsonschema) - Many software tools, but also formal frameworks for working with JSON Schema, do not fully support negation. This motivates us to study whether negation is actually used in practice, for which aims, and whether it could, in principle, be replaced by simpler operators. We have collected a large corpus of 80k open source JSON Schema documents. We perform a systematic analysis, quantify usage patterns of negation, and also qualitatively analyze schemas. We show that negation is indeed used, albeit infrequently, following a stable set of patterns.
- (2022) [Validating Streaming JSON Documents with Learned VPAs](https://arxiv.org/abs/2211.08891?utm_source=awesome-jsonschema) - This paper presents a new streaming algorithm to validate JSON documents against a set of constraints given as a JSON schema. It proves that there always exists a visibly pushdown automaton (VPA) that accepts the same set of JSON documents as a JSON schema.
- (2022) [Witness Generation for JSON Schema](https://arxiv.org/abs/2202.12849?utm_source=awesome-jsonschema) - JSON Schema is an important, evolving standard schema language for families of JSON documents. It is based on a complex combination of structural and Boolean assertions, and features negation and recursion. The static analysis of JSON Schema documents comprises practically relevant problems, including schema satisfiability, inclusion, and equivalence. These three problems can be reduced to witness generation: given a schema, generate an element of the schema, if it exists, and report failure otherwise.
- (2021) [Deriving Semantics-Aware Fuzzers from Web API Schemas](https://arxiv.org/abs/2112.10328?utm_source=awesome-jsonschema) - Discusses JSON Schema canonicalization and JSON Schema instance derivation in the context of property-based testing of APIs.
- (2021) [Enhancing JSON Schema Discovery by Uncovering Hidden Data](https://ceur-ws.org/Vol-2971/paper11.pdf?utm_source=awesome-jsonschema) - Enhancing discovered JSON Schemas by disambiguating data and metadata.
- (2021) [Fast Discovery of Nested Dependencies on JSON Data](https://arxiv.org/abs/2111.10398?utm_source=awesome-jsonschema) - Efficient dependency mining algorithms for non-relational data.
- (2021) [Not Elimination and Witness Generation for JSON Schema](https://arxiv.org/abs/2104.14828?utm_source=awesome-jsonschema) - In this paper, we present an algebraic characterization of JSON Schema, obtained by adding opportune operators, and by mirroring existing ones. We present then algebra-based approaches for dealing with not-elimination and witness generation problems, which play a central role as they lead to solutions for the other mentioned complex problems.
- (2021) [TILT: A GDPR-Aligned Transparency Information Language and Toolkit for Practical Privacy Engineering](https://dl.acm.org/doi/10.1145/3442188.3445925?utm_source=awesome-jsonschema) - We present TILT, a transparency information language and toolkit explicitly designed to represent and process transparency information in line with the requirements of the GDPR and allowing for a more automated and adaptive use of such information than established, legalese data protection policies do.
- (2020) [Challenges in Checking JSON Schema Containment over Evolving Real-World Schemas](https://link.springer.com/chapter/10.1007/978-3-030-65847-2_20?utm_source=awesome-jsonschema) - This paper presents the results of an empirical study of the first generation of tools for checking JSON Schema containment which is applied to a diverse collection of over 230 real-world schemas and their altogether 1k historic versions.
- (2020) [JSON Schema Inference Approaches](https://link.springer.com/chapter/10.1007/978-3-030-65847-2_16?utm_source=awesome-jsonschema) - In the context of document NoSQL databases, namely those assuming the JSON data format, this paper focuses on several representatives of the existing inference approaches and provide their thorough comparison.
- (2020) [Type Safety with JSON Subschema](https://arxiv.org/abs/1911.12651?utm_source=awesome-jsonschema) - Deciding whether one schema is a subschema of another is non-trivial because of the richness of the JSON Schema specification language. Given a pair of schemas, our approach first canonicalizes and simplifies both schemas, then decides the subschema question on the canonical forms, dispatching simpler subschema queries to type-specific checkers.
- (2019) [What Are Real JSON Schemas Like?](https://link.springer.com/chapter/10.1007/978-3-030-34146-6_9?utm_source=awesome-jsonschema) - A first empirical analysis of a curated collection of real-world JSON Schemas. Knowing what real JSON Schemas are like (to borrow from a title of a related study on DTDs) helps practitioners and researchers in making realistic assumptions when building tools for JSON Schema processing.
- (2018) [An Approach for Schema Extraction of JSON and Extended JSON Document Collections](https://ieeexplore.ieee.org/abstract/document/8424731?utm_source=awesome-jsonschema) - This paper presents an approach that extracts a schema from a JSON or Extended JSON document collection stored in a NoSQL document-oriented database or other document repository. Aggregation operations are considered in order to obtain a schema for each distinct structure in the collection, and a hierarchical data structure is proposed to group these schemas in order to generate a global schema in JSON Schema format.
- (2018) [Top-Down Model-Driven Engineering of Web Services from Extended OpenAPI Models](https://ieeexplore.ieee.org/abstract/document/9000020?utm_source=awesome-jsonschema) - Shows how OpenAPI can be extended to add implementation details inside models. These extensions link services to assemblies of components that describe computations. Hence a top-down development process that keeps model and implementation aligned.
- (2017) [Definition of REST web services with JSON schema](https://dl.acm.org/doi/abs/10.1002/spe.2466?utm_source=awesome-jsonschema) - The aim of this article is to demonstrate how JSON Schema, and particularly the JSON Hyper Schema extension, is suitable to describe JSON-based web services that follow the REST architectural pattern.
- (2017) [Example-Driven Web API Specification Discovery](https://link.springer.com/chapter/10.1007/978-3-319-61482-3_16?utm_source=awesome-jsonschema) - In this paper we present an example-driven discovery process that generates model-based OpenAPI specifications for REST Web APIs by using API call examples. A tool implementing our approach and a community-driven repository for the discovered APIs are also presented.
- (2017) [Schema Inference for Massive JSON Datasets](https://hal.archives-ouvertes.fr/hal-01491765?utm_source=awesome-jsonschema) - Recent years have seen the widespread use of JSON as a data format to represent massive data collections. JSON data collections are usually schemaless. While this ensures several advantages, the absence of schema information has important negative consequences: the correctness of complex queries and programs cannot be statically checked, users cannot rely on schema information to quickly figure out structural properties that could speed up the formulation of correct queries, and many schema-based optimizations are not possible. In this paper we deal with the problem of inferring a schema from massive JSON data sets.
- (2016) [Foundations of JSON Schema](https://dl.acm.org/doi/abs/10.1145/2872427.2883029?utm_source=awesome-jsonschema) - In this paper we provide the first formal definition of syntax and semantics for JSON Schema and use it to show that implementing this layer on top of JSON is feasible in practice.
- (2016) [τJSchema: A Framework for Managing Temporal JSON-Based NoSQL Databases](https://link.springer.com/chapter/10.1007/978-3-319-44406-2_13?utm_source=awesome-jsonschema) - This paper proposes a framework called Temporal JSON Schema (τJSchema), inspired by the τXSchema framework defined for XML data. τJSchema allows defining a temporal JSON schema from a conventional JSON schema and a set of temporal logical and physical characteristics. Our framework guarantees logical and physical data independence for temporal schemas and provides a low-impact solution since it requires neither modifications of existing JSON documents, nor extensions to the JSON format, the JSON Schema language, and all related tools and languages.
- (2015) [Schema extraction and structural outlier detection for JSON-based nosql data stores](https://dl.gi.de/handle/20.500.12116/2420?utm_source=awesome-jsonschema) - Rather than designing the schema up front, extracting a schema in hindsight can be seen as a reverse-engineering step. Based on the extracted schema information, we propose set of similarity measures that capture the degree of heterogeneity of JSON data and which reveal structural outliers in the data.
- (2014) [Jsongen: a quickcheck based library for testing JSON web services](https://dl.acm.org/doi/abs/10.1145/2633448.2633454?utm_source=awesome-jsonschema) - This article describes a systematic approach to testing behavioural aspects of Web Services that communicate using the JSON data format. To generate random JSON data for populating tests we have developed a new library, jsongen, which given a characterisation of the JSON data as a JSON schema, (i) automatically derives a QuickCheck generator which can generate an infinite number of JSON values that validate against the schema, and (ii) provides a generic QuickCheck state machine which is capable of following the (hyper)links documented in the JSON schema, to automatically explore the web service.
- (2012) [User profile integration made easy: model-driven extraction and transformation of social network schemas](https://dl.acm.org/doi/abs/10.1145/2187980.2188227?utm_source=awesome-jsonschema) - This paper presents, firstly, a semi-automatic approach to extract schema information from instance data. Secondly, transformations of the derived schemas to different technical spaces are utilized, thereby allowing, amongst other benefits, the application of established integration tools and methods. Finally, as a case study, schemas are derived for Facebook, Google+, and LinkedIn.

## Libraries

The JSON Schema website includes an extensive list of implementations and
related libraries: https://json-schema.org/implementations.html. Check out the
[Bowtie](https://bowtie.report) project that measures compliance of JSON Schema
implementations to help users select properly compliant and maintained
libraries.

---

> Special thanks to [@kinlane](https://github.com/kinlane) for curating the
initial version of this list.
 readmeEtag: '"6ca767f9af081e4ad86ee194e851d2cd53f3ae66"' readmeLastModified: Fri, 16 Jan 2026 14:58:25 GMT repositoryId: 404698785 description: >- A curated list of awesome JSON Schema resources, tutorials, tools, and more created: '2021-09-09T11:39:06Z' updated: '2026-02-05T19:41:53Z' language: Handlebars archived: false stars: 160 watchers: 10 forks: 4 owner: sourcemeta logo: https://avatars.githubusercontent.com/u/74150453?v=4 license: CC0-1.0 repoEtag: '"216b1d45dae95ee85faf2bd876d9cb465aeee099e61cbcb0a6c6de1ac056d44e"' repoLastModified: Thu, 05 Feb 2026 19:41:53 GMT foundInMaster: true category: Converters id: 40df0f4f4e0e95da0b9f4aeb27567a36 oldLocations: - https://github.com/jviotti/awesome-jsonschema - source: openapi3 tags repository: https://github.com/cebe/yii2-openapi v3: true repositoryMetadata: base64Readme: >- # yii2-openapi

>
> **This repository has been moved to <https://github.com/php-openapi/yii2-openapi>.**  
> **Please use the new package `php-openapi/yii2-openapi` instead.**
>

REST API application generator for Yii2, openapi 3.0 YAML -> Yii2.

Base on [Gii, the Yii Framework Code Generator](https://www.yiiframework.com/extension/yiisoft/yii2-gii).

[![Latest Stable Version](https://poser.pugx.org/cebe/yii2-openapi/v/stable)](https://packagist.org/packages/cebe/yii2-openapi)
[![Latest Alpha Version](https://poser.pugx.org/cebe/yii2-openapi/v/unstable)](https://packagist.org/packages/cebe/yii2-openapi)
[![Total Downloads](https://poser.pugx.org/cebe/yii2-openapi/downloads)](https://packagist.org/packages/cebe/yii2-openapi)
[![License](https://poser.pugx.org/cebe/yii2-openapi/license)](https://packagist.org/packages/cebe/yii2-openapi)
![yii2-openapi](https://github.com/cebe/yii2-openapi/workflows/yii2-openapi/badge.svg?branch=wip)

## TLDR; what is this?

A code generator for OpenAPI and Yii Framework based PHP API application.

Input: [OpenAPI 3.0 YAML or JSON](https://github.com/OAI/OpenAPI-Specification#the-openapi-specification) (via [cebe/php-openapi](https://github.com/cebe/php-openapi))

Output: Yii Framework Application with Controllers, Models, database schema


## Features

Currently available features:

- Generate Path mappings, **Controllers** and Actions **for API Endpoints**. CRUD Endpoints are ready-to-use, other Endpoints are generated as abstract functions that need to be implemented
- Generate **Models** and validation based on OpenAPI Schema
- Generate **Database Schema** from OpenAPI Schema
- Generates **Database Migrations** for schema changes
- Provide **Dummy Data** via Faker for development

## Requirements

- PHP 7.1 or higher (works fine with PHP 8)


## Install

    composer require php-openapi/yii2-openapi:^2.0@beta

## Usage

You can use this package in your existing application or start a new project using the
[yii2-app-api](https://github.com/cebe/yii2-app-api) application template.
For usage of the template, see instructions in the template repo readme.

In your existing Yii application config (works for console as well as web):

```php
<?php
$config = [
    // ... this is your application config ...
];

if (YII_ENV_DEV) {
    // enable Gii module
    $config['bootstrap'][] = 'gii';
    $config['modules']['gii'] = [
        'class' => \yii\gii\Module::class,
        'generators' => [
            // add ApiGenerator to Gii module
            'api' => \cebe\yii2openapi\generator\ApiGenerator::class,

            // --------- OR ---------
            // to disable generation of migrations files or with default config change
            'api' => [
              'class' => \cebe\yii2openapi\generator\ApiGenerator::class,
              'generateMigrations' => false, # this config can also be applied in CLI command
            ],
        ],
    ];
}

return $config;
```

To use the web generator, open `index.php?r=gii` and select the `REST API Generator`.

On console you can run the generator with `./yii gii/api --openApiPath=@app/openapi.yaml`. Where `@app/openapi.yaml` should be the absolute path to your OpenAPI spec file. This can be JSON as well as YAML (see also [cebe/php-openapi](https://github.com/cebe/php-openapi/) for supported formats).

Run `./yii gii/api --help` for all options. Example: Disable generation of migrations files `./yii gii/api --generateMigrations=0`

See [Petstore example](https://github.com/OAI/OpenAPI-Specification/blob/main/examples/v3.0/petstore.yaml) for example OpenAPI spec.


## OpenAPI extensions

This library understands the following extensions to the OpenAPI spec:

### `x-faker`

You may specify custom PHP code for generating fake data for a property:

```yaml
    Post:
      properties:
        id:
          type: integer
        tags:
          type: array
          items:
            type: string
          example: ['one', 'two']
          x-faker: "$faker->randomElements(['one', 'two', 'three', 'four'])"
```

To avoid generating faker code for particular model attribute, use value `false`:

```yaml
    Post:
      properties:
        age:
          type: integer
          x-faker: false
```

Using in reference with `allOf`:

```yaml
    Invoice:
      type: object
      required:
        - id
      properties:
        id:
          type: integer

    Order:
      type: object
      required:
        - id
      properties:
        id:
          type: integer
        invoice:
          allOf:
            - $ref: '#/components/schemas/Invoice'
            - x-faker: false
```

### `x-table`

Specify the table name for a Schema that defines a model which is stored in the database.
You can generate non-db model based on \yii\base\Model without migrations by setting `x-table: false`

### `x-pk`

Explicitly specify primary key name for table, if it is different from "id" 

```yaml
    Post:
      x-table: posts
      x-pk: uid
      properties:
        uid:
           type: integer
        title:
           type: string
```

### `x-db-type`

Explicitly specify the database type for a column. (MUST contain only real DB type! (`json`, `jsonb`, `uuid`, `varchar` etc.)).
If `x-db-type` is set to `false`, property will be processed as virtual;
It will be added in model as public property, but skipped for migrations generation.

Example values of `x-db-type` are:

 - `false` (boolean false)
 - as string and its value can be like:
     - text
     - text[]
     - INTEGER PRIMARY KEY AUTO_INCREMENT
     - decimal(12,4)
     - json
     - varchar
     - VARCHAR
     - SMALLINT UNSIGNED ZEROFILL
     - MEDIUMINT(10) UNSIGNED ZEROFILL COMMENT "comment" (note the double quotes here)

Such values are not allowed:
   - `int null default null after low_price` (null and default will be handled by `nullable` and `default` keys respectively)
   - MEDIUMINT(10) UNSIGNED ZEROFILL NULL DEFAULT '7' COMMENT 'comment' AFTER `seti`, ADD INDEX `t` (`w`)

If `enum` and `x-db-type` both are provided then for database column schema (migrations), only `x-db-type` will be considered ignoring `enum`.

### `x-indexes`

Specify table indexes

```yaml
    Post:
      x-table: posts
      x-indexes:
          - 'visible,publish_date'
          - 'unique:title' #for unique attributes also unique validation check will be added
          - 'gist:metadata' #for postgres will generate index using GIST index type
      properties:
        id:
           type: integer
           x-db-type: INTEGER PRIMARY KEY AUTO_INCREMENT
        title:
           type: string
        visible:
            type: boolean
        publish_date:
            type: string
            format: date
        metadata:
           type: object
           x-db-type: JSON
           default: '{}' 
```

### `x-db-default-expression`

Ability to provide default value by database expression

```yaml
created_at:
  readOnly: true
  type: string
  format: datetime
  x-db-type: datetime
  nullable: false
  x-db-default-expression: current_timestamp()
```

Note: If both `default` and `x-db-default-expression` are present then `default` will be considered.

```yaml
created_at:
  readOnly: true
  type: string
  format: datetime
  x-db-type: datetime
  nullable: false
  x-db-default-expression: current_timestamp() # this will be ignored
  default: "2011-11-11" # this will be considered
```

Also see: https://dev.mysql.com/doc/refman/8.0/en/data-type-defaults.html

### `x-fk-on-delete`

Allow to set foreign key constraint in migrations for ON DELETE event of row in database table. Example:

```yaml
  components:
    schemas:
      User:
        type: object
        description: x on-x (update|delete) foreign key constraint
        properties:
          id:
            type: integer
          name:
            type: string
      Post:
        type: object
        description: x on-x (update|delete) foreign key constraint
        properties:
          id:
            type: integer
          title:
            type: string
          user:
            allOf:
              - $ref: '#/components/schemas/User'
              - x-fk-on-update: CASCADE
          user_2:
            allOf:
              - $ref: '#/components/schemas/User'
              - x-fk-on-update: CASCADE
              - x-fk-on-delete: SET NULL
          user_3:
            allOf:
              - $ref: '#/components/schemas/User'
              - x-fk-on-delete: SET NULL
          user_4:
            $ref: '#/components/schemas/User' # without any constraints
```

### `x-fk-on-update`

Allow to set foreign key constraint in migrations for ON UPDATE event of row in database table. For example, see above section for `x-fk-on-delete`.

### `x-fk-column-name`

Provide custom database table column name in case of relationship column. This will not reflect in models relations, faker etc. Example:

```yaml
  components:
    schemas:
      Webhook:
        type: object
        description: example for x-fk-column-name
        properties:
          id:
            type: integer
          name:
            type: string
          user:
            $ref: '../openapi.yaml#/components/schemas/User' # this will automatically create `user_id` column
          redelivery_of:
            allOf:
              - $ref: '../openapi.yaml#/components/schemas/Delivery'
              # this will automatically create `redelivery_of_id` column, but to avoid that use below extension
              - x-fk-column-name: redelivery_of # this will create `redelivery_of` column instead of `redelivery_of_id`
```

## Many-to-Many relation definition

There are two ways for define many-to-many relations:

### Simple many-to-many without junction model

   - property name for many-to-many relation should be equal lower-cased, pluralized related schema name
     
   - referenced schema should contains mirrored reference to current schema
     
   - migration for junction table can be generated automatically - table name should be [pluralized, lower-cased
 schema_name1]2[pluralized, lower-cased schema name2], in alphabetical order;
 For example, for schemas Post and Tag - table should be posts2tags, for schemas Post and Attachement - table should
  be attachments2posts
  
```
Post:
  properties:
  ...
    tags:
      type: array
      items:
        $ref: '#/components/schemas/Tag'

Tag:
  properties:
  ...
    posts:
      type: array
      items:
        $ref: '#/components/schemas/Post'
```
  
### Many-to-many with junction model

This way allowed creating multiple many-to-many relations between to models 

- define junction schema with all necessary attributes. There are only one important requirement - the junction
 schema name
 must be started with prefix 'junction_' (This prefix will be used internally only and
 will be trimmed before table and model generation)
 
```
# Model TeamMembers with table team_members will be generated with columns team_id, user_id and role
junction_TeamMembers:
   team:
      $ref: '#/components/schemas/Team'
   user:
      $ref: '#/components/schemas/User'
   role:
     type: string
```
- Both many-to-many related schemas must have properties with reference to "junction_*" schema. These properties will be
 used as relation names 

```
Team:
  properties:
  ...
     team_members:
       type: array
       items:
         $ref: '#/components/schemas/junction_TeamMembers'

User:
  properties:
  ...
    memberships: #You absolutely free with naming for relationship attributes
      type: array
      items:
        $ref: '#/components/schemas/junction_TeamMembers'
```
  
 - see both examples here [tests/specs/many2many.yaml](tests/specs/many2many.yaml)
 

## Handling of `NOT NULL` constraints

`NOT NULL` in DB migrations is determined by `nullable` and `required` properties of the OpenAPI schema.
e.g. attribute = 'my_property'.

- If you define attribute neither "required" nor via "nullable", then it is by default `NULL`:

```yaml
  ExampleSchema:
    properties:
      my_property:
        type: string
```

- If you define attribute in "required", then it is `NOT NULL`

```yaml
  ExampleSchema:
    required:
     - my_property
    properties:
      my_property:
        type: string
```

- If you define attribute via "nullable", then it overrides "required", e.g. allow `NULL` in this case:

```yaml
  ExampleSchema:
    required:
      - my_property
    properties:
      my_property:
        type: string
        nullable: true
```

- If you define attribute via "nullable", then it overrides "required", e.g. `NOT NULL` in this case:

```yaml
  test_table:
    required:
    properties:
      my_property:
        type: string
        nullable: false
```

## Handling of `enum` (#enum)
It works on all 3 DB: MySQL, MariaDb and PgSQL.

 ```yaml
  test_table:
    properties:
      my_property:
        enum:
          - one
          - two
          - three
```

Note: Changes in enum values are not very simple. For Mysql and Mariadb, migrations will be generated but in many cases custom modification in it are required. For Pgsql migrations for change in enum values will not be generated. It should be handled manually.

It will be ignored for database column schema (migrations) if `x-db-type` is provided.

## Handling of `numeric` (#numeric, #MariaDb)

precision-default = 10
scale-default = 2

- You can define attribute like "numeric(precision,scale)":
 ```yaml
  test_table:
    properties:
      my_property:
        x-db-type: decimal(12,4)
```
DB-Result = decimal(12,4)

- You can define attribute like "numeric(precision)" with default scale-default = 2:
 ```yaml
  test_table:
    properties:
      my_property:
        x-db-type: decimal(12)
```
DB-Result = decimal(12,2)

- You can define attribute like "numeric" with precision-default = 10 and scale-default = 2:
 ```yaml
  test_table:
    properties:
      my_property:
        x-db-type: decimal
```
DB-Result = decimal(10,2)

## Handling of `timestamp` database column data type

If field is defined as

```yaml
created_at:
    type: string
    format: date-time # or datetime
    example: '2020-03-14T21:42:17Z'
    readOnly: true
```

then database type selected will be `timestamp`. This is by design. If `datetime` data type is needed, use `x-db-type` as

```yaml
created_at:
    type: string
    format: date-time # or datetime
    example: '2020-03-14T21:42:17Z'
    x-db-type: datetime
    readOnly: true
```

## Assumptions

When generating code from an OpenAPI description there are many possible ways to achive a fitting result.
Thus there are some assumptions and limitations that are currently applied to make this work.
Here is a (possibly incomplete) list:

- The current implementation works best with OpenAPI description that follows the [JSON:API](https://jsonapi.org/) guidelines.
  - The request and response format/schema is currently not extracted from OpenAPI schema and may need to be adjusted manually if it does not follow JSON:API
- column/field/property with name `id` is considered as Primary Key by this library and it is automatically handled by DB/Yii; so remove it from validation `rules()`
  - other fields can currently be used as primary keys using the `x-pk` OpenAPI extension (see below) but it may not be work correctly in all cases, please report bugs if you find them.

Other things to keep in mind:

### Adding columns to existing tables

When adding new fields in the API models, new migrations will be generated to add these fields to the table.
For a project that is already in production, it should be considered to adjust the generated migration to add default
values for existing data records.

One case where this is important is the addition of a new column with `NOT NULL` contraint, which does not provide a default value.
Such a migration will fail when the table is not empty:

```php
$this->addColumn('{{%company}}', 'name', $this->string(128)->notNull());
```

Fails on a PostgreSQL database with

> add column name string(128) NOT NULL to table {{%company}} ...Exception: SQLSTATE[23502]: Not null violation: 7 ERROR:  column "name" contains null values

The solution would be to create the column, allowing NULL, set the value to a default and add the null constraint later.

```php
$this->addColumn('{{%company}}', 'name', $this->string(128)->null());
$this->update('{{%company}}', ['name' => 'No name']);
$this->alterColumn('{{%company}}', 'name', $this->string(128)->notNull());
```

## Screenshots

Gii Generator Form:

![Gii Generator Form](doc/screenshot-form.png)

Generated files:

![Gii Generated Files](doc/screenshot-files.png)


# Development

To contribute or play around, steps to set up this project up locally are in [CONTRIBUTING.md](./CONTRIBUTING.md).


# Support

**Need help with your API project?**

Professional support, consulting as well as software development services are available:

https://www.cebe.cc/en/contact

Development of this library is sponsored by [cebe.:cloud: "Your Professional Deployment Platform"](https://cebe.cloud).
 readmeEtag: '"f488dc2f1263f780818dd46f0c1631c496339331"' readmeLastModified: Tue, 04 Jun 2024 15:27:09 GMT repositoryId: 154310166 description: REST API application generator for Yii2, openapi 3.0 YAML -> Yii2 created: '2018-10-23T10:39:18Z' updated: '2025-08-05T16:56:41Z' language: PHP archived: true stars: 128 watchers: 8 forks: 23 owner: cebe logo: https://avatars.githubusercontent.com/u/189796?v=4 license: MIT repoEtag: '"796aa4d6912dd45488b4d821582c436536fbea35a876ae96fc39c71201688f6c"' repoLastModified: Tue, 05 Aug 2025 16:56:41 GMT foundInMaster: true category: Parsers id: 17efb27f7e36f9a426c34f2a74c84a78 - source: openapi3 tags repository: https://github.com/parvez3019/go-swagger3 v3: true repositoryMetadata: base64Readme: >- ### Demo

Click here for [Youtube Demo Link](https://www.youtube.com/watch?v=GLM9c5j8g7I)


# go-swagger3

Generate [OpenAPI Specification](https://swagger.io/specification) v3 file with comments in Go.

### Table of content

- [1. Install](#1-install)
- [2. Documentation Generation](#2-documentation-generation)
- [3. Usage](#3-usage)
    - [Service Description](#service-description)
    - [Handler functions](#handler-functions)
    - [Title And Description](#title-and-description)
    - [Parameter](#parameter)
    - [Header](#header)
    - [Header Parameters](#header-parameters)
    - [Response](#response)
    - [Resource & Tag](#resource--tag)
    - [Route](#route)
    - [Enums](#enums)
- [4. Security](#4-security)
- [5. Limitations](#5-limitations)
- [6. References](#6-references)

## 1. Install

```
go install github.com/parvez3019/go-swagger3@latest
```


### 2. Documentation Generation
#### Using binary
Go to the folder where is main.go in

``` shell
// go.mod and main file are in the same directory
go-swagger3 --module-path . --output oas.json --schema-without-pkg --generate-yaml true

// go.mod and main file are in the different directory
go-swagger3 --module-path . --main-file-path ./cmd/xxx/main.go --output oas.json --schema-without-pkg --generate-yaml true

// in case you get 'command not found: go-swagger3' error, please export add GOPATH/bin to PATH
export PATH="$HOME/go/bin:$PATH"

Notes - 
- Pass schema-without-pkg flag as true if you want to generate schemas without package names
- Pass generate-yaml as trus if you want to generate yaml spec file instead of json

```

#### Using docker
``` shell
// go.mod and main file are in the same directory
docker run -t --rm -v $(pwd):/app -w /app parvez3019/go-swagger3:latest --module-path . --output oas.json --schema-without-pkg --generate-yaml true

// go.mod and main file are in the different directory
docker run -t --rm -v $(pwd):/app -w /app parvez3019/go-swagger3:latest --module-path . --main-file-path ./cmd/xxx/main.go --output oas.json --schema-without-pkg --generate-yaml true

Notes - 
- Pass schema-without-pkg flag as true if you want to generate schemas without package names
- Pass generate-yaml as trus if you want to generate yaml spec file instead of json

```




## 3. Usage

You can document your service by placing annotations inside your godoc at various places in your code.

### Service Description

The service description comments can be located in any of your .go files. They provide general information about the service you are documenting.

``` go
// @Version 1.0.0
// @Title Backend API
// @Description API usually works as expected. But sometimes its not true.
// @ContactName Parvez
// @ContactEmail abce@email.com
// @ContactURL http://someurl.oxox
// @TermsOfServiceUrl http://someurl.oxox
// @LicenseName MIT
// @LicenseURL https://en.wikipedia.org/wiki/MIT_License
// @Server http://www.fake.com Server-1
// @Server http://www.fake2.com Server-2
// @Security AuthorizationHeader read write
// @SecurityScheme AuthorizationHeader http bearer Input your token
```

### Handler Functions

By adding comments to your handler func godoc, you can document individual actions as well as their input and output.

``` go
type User struct {
  ID   uint64 `json:"id" example:"100" description:"User identity"`
  Name string `json:"name" example:"Parvez"` 
}

type UsersResponse struct {
  Data []Users `json:"users" example:"[{\"id\":100, \"name\":\"Parvez\"}]"`
}

type Error struct {
  Code string `json:"code"`
  Msg  string `json:"msg" skip:"true"`
  // use skip:"true" if you want to skip the field in the documentation spec
}

type ErrorResponse struct {
  ErrorInfo Error `json:"error"`
}

// RequestHeaders represents the model for header params
// @HeaderParameters RequestHeaders
type RequestHeaders struct {
    Authorization  string  `json:"Authorization"`
}

// @Title Get user list of a group.
// @Description Get users related to a specific group.
// @Header model.RequestHeaders
// @Param  groupID  path  int  true  "Id of a specific group." "120"
// @Success  200  object  UsersResponse  "UsersResponse JSON"
// @Failure  400  object  ErrorResponse  "ErrorResponse JSON"
// @Resource users
// @Route /api/group/{groupID}/users [get]
func GetGroupUsers() {
  // ...
}

// @Title Get user list of a group.
// @Description Create a new user.
// @Param  user  body  User  true  "Info of a user."
// @Success  200  object  User           "UsersResponse JSON"
// @Failure  400  object  ErrorResponse  "ErrorResponse JSON"
// @Resource users
// @Route /api/user [post]
func PostUser() {
  // ...
}
```

#### Title And Description
```
@Title {title}
@Title Get user list of a group.

@Description {description}.
@Description Get users related to a specific group.
```
- {title}: The title of the route.
- {description}: The description of the route.

#### Parameter
```
@Param  {name}    {in}  {goType}  {required}  {description}             {example}
@Param  groupID   path  int       true        "Id of a specific group." "120"
```
- {name}: The parameter name.
- {in}: The parameter is in `path`, `query`, `form`, `header`, `cookie`, `body` or `file`.
- {goType}: The type in go code. This will be ignored when {in} is `file`.
- {required}: `true`, `false`, `required` or `optional`. 
- {description}: The description of the parameter. Must be quoted.
- {example}: **Optional** example of this parameter. Must be quoted.

One can also override example for an object with `override-example` key in struct
eg -
``` go
type Request struct {
    version  model.Version `"json:"Version" override-example:"11.0.0"`
}
```

#### Header 
```
@Header          {goType}
@HeaderParameters   model.RequestHeaders
```
- Header query param for endpoints, parses the query param from the model

#### Header Parameters
```
@Param              {goType}
@HeaderParameters   RequestHeaders
```

- {goType}: The type in go code. This will be ignored when {in} is `file`.
- Parses parameters from the type and keep it up component section for reference

#### Response
``` json
@Success  {status}  {jsonType}  {goType}       {description}
@Success  200       object      UsersResponse  "UsersResponse JSON"

@Failure  {status}  {jsonType}  {goType}       {description}
@Failure  400       object      ErrorResponse  "ErrorResponse JSON"
```
- {status}: The HTTP status code.
- {jsonType}: The value can be `object` or `array`. 
- {goType}: The type in go code.
- {description}: The description of the response. Must be quoted.

#### Resource & Tag
``` json
@Resource {resource}
@Resource users

@Tag {tag}
@tag xxx
```

- {resource}, {tag}: Tag of the route.

#### Route

``` json
@Route {path}    {method}
@Route /api/user [post]
```

- {path}: The URL path.
- {method}: The HTTP Method. Must be put in brackets.

#### Enums

- To generate enums create type structs for enum field with comma-separated values as follows:
- Create struct type fields with @Enum Tag
- Example as follows-

``` go
// @Enum CountriesEnum
type CountriesEnum struct {
    // Create the field name with same as struct name
    CountriesEnum string `enum:"india,china,mexico,japan" example:"india"`
}
```

##### How to add reference of Enum on types

``` go
type Request struct {
  Name string `json:"name" example:"Parvez"` 
  Country string `json:"country" $ref:"CountriesEnum"`
}

```

#### Response body
You need to provide swagger fields as reflect tags in your structure. In example:
```go
type Filter struct {
	Rating       int    `json:"rating"`
	Type         string `json:"type"`
	Distance     int64  `json:"distance", minimum:"1", maximum:"100"`
	DistrictCode string `json:"district_code", pattern:"[a-z]{2}\\d[a-z]\\s\\d[a-z]{2}"`
}
```

Available fields:
- type
- format
- required
- properties
- description
- items
- example
- deprecated (bool)
- ref
- enum
- title
- maximum (float64)
- exclusiveMaximum (bool)
- minimum (float64)
- exclusiveMinimum (bool)
- maxLength (uint)
- minLength (uint)
- pattern
- maxItems (uint)
- minItems (uint)
- uniqueItems (bool)
- maxProperties (uint)
- minProperties (uint)
- additionalProperties (bool)
- nullable (bool)
- readOnly (bool)
- writeOnly (bool)

### 4. Security

If authorization is required, you must define security schemes and then apply those to the API. A scheme is defined
using `@SecurityScheme [name] [type] [parameters]` and applied by
adding `@Security [scheme-name] [scope1] [scope2] [...]`.

All examples in this section use `MyApiAuth` as the name. This name can be anything you chose; multiple named schemes
are supported. Each scheme must have its own name, except for OAuth2 schemes - OAuth2 supports multiple schemes by the
same name.

A number of different types is supported, they all have different parameters:

|Type|Description|Parameters|Example|
|---|---|---|---|
|HTTP|A HTTP Authentication scheme using the `Authorization` header|scheme: any [HTTP Authentication scheme](https://www.iana.org/assignments/http-authschemes/http-authschemes.xhtml)|`@SecurityScheme MyApiAuth basic`|
|APIKey|Authorization by passing an API Key along with the request|in: Location of the API Key, options are `header`, `query` and `cookie`. name: The name of the field where the API Key must be set|`@SecurityScheme MyApiAuth apiKey header X-MyCustomHeader`|
|OpenIdConnect|Delegating security to a known OpenId server|url: The URL of the OpenId server|`@SecurityScheme MyApiAuth openIdConnect https://example.com/.well-known/openid-configuration`|
|OAuth2AuthCode|Using the "Authentication Code" flow of OAuth2|authorizationUrl, tokenUrl|`@SecurityScheme MyApiAuth oauth2AuthCode /oauth/authorize /oauth/token`| 
|OAuth2Implicit|Using the "Implicit" flow of OAuth2|authorizationUrl|`@SecurityScheme MyApiAuth oauth2Implicit /oauth/authorize| 
|OAuth2ResourceOwnerCredentials|Using the "Resource Owner Credentials" flow of OAuth2|authorizationUrl|`@SecurityScheme MyApiAuth oauth2ResourceOwnerCredentials /oauth/token| 
|OAuth2ClientCredentials|Using the "Client Credentials" flow of OAuth2|authorizationUrl|`@SecurityScheme MyApiAuth oauth2ClientCredentials /oauth/token| 

Any text that is present after the last parameter wil be used as the description. For
instance `@SecurityScheme MyApiAuth basic Login with your admin credentials`.

Once all security schemes have been defined, they must be configured. This is done with the `@Security` comment.
Depending on the `type` of the scheme, scopes (see below) may be supported. *At the moment, it is only possible to
configure security for the entire service*.

``` go
// @Security MyApiAuth read_user write_user
```

#### Scopes

For OAuth2 security schemes, it is possible to define scopes using
the `@SecurityScope [schema-name] [scope-code] [scope-description]` comment.

``` go
// @SecurityScope MyApiAuth read_user Read a user from the system
// @SecurityScope MyApiAuth write_user Write a user to the system
```

### 5. Limitations

- Only support go module.
- Anonymous struct field is not supported.

### 6. References

- The project is based on the following repositories -
- [yvasiyarov/swagger](https://github.com/yvasiyarov/swagger)
- [uudashr/go-module](https://github.com/uudashr/go-module)
- [mikunalpha/goas](https://github.com/mikunalpha/goas)
 readmeEtag: '"5c182058c40ce228bfd9e012db5c1e96501930e0"' readmeLastModified: Wed, 19 Jun 2024 06:12:07 GMT repositoryId: 373055212 description: Swagger 3.0 implementation for go created: '2021-06-02T05:54:10Z' updated: '2026-02-03T01:00:04Z' language: Go archived: false stars: 147 watchers: 5 forks: 25 owner: parvez3019 logo: https://avatars.githubusercontent.com/u/10362597?v=4 license: MIT repoEtag: '"f5da95b7f9387b64959a834a78f33301449e7af80d5429d494781c4dc24d8a99"' repoLastModified: Tue, 03 Feb 2026 01:00:04 GMT foundInMaster: true category: Server Implementations id: 00ef51b7bb2f326af2de90739df6907b - source: openapi3 tags repository: https://github.com/swagger-api/swagger-converter v3: true repositoryMetadata: base64Readme: >- UHJvdmlkZXMgYSBjb252ZXJ0ZXIgYmV0d2VlbiBPcGVuQVBJIDIuMCBhbmQgMy4wIHNwZWNpZmljYXRpb25zIGFjY2Vzc2libGUgdmlhIEFQSXMgYW5kIHdlYiBVSS4gCgpJdCBpcyBhdmFpbGFibGUgb25saW5lIGF0IGh0dHBzOi8vY29udmVydGVyLnN3YWdnZXIuaW8vIGFuZCBhcyBhIFtkb2NrZXIgaW1hZ2VdKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9zd2FnZ2VyYXBpL3N3YWdnZXItY29udmVydGVyKSBlLmcuOgoKYGBgCmRvY2tlciBwdWxsIHN3YWdnZXJhcGkvc3dhZ2dlci1jb252ZXJ0ZXI6djEuMC42LVNOQVBTSE9UCmRvY2tlciBydW4gLWl0IC1wIDgwODA6ODA4MCAtLW5hbWUgc3dhZ2dlci1jb252ZXJ0ZXIgc3dhZ2dlcmFwaS9zd2FnZ2VyLWNvbnZlcnRlcjp2MS4wLjYtU05BUFNIT1QKYGBgCgpXZWIgVUkgaXMgcmVhY2hhYmxlIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9pbmRleC5odG1sIGFuZCBPcGVuQVBJIHNwZWMgYXQgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS9vcGVuYXBpLmpzb24KCiMjIFNlY3VyaXR5IGNvbnRhY3QKClBsZWFzZSBkaXNjbG9zZSBhbnkgc2VjdXJpdHktcmVsYXRlZCBpc3N1ZXMgb3IgdnVsbmVyYWJpbGl0aWVzIGJ5IGVtYWlsaW5nIFtzZWN1cml0eUBzd2FnZ2VyLmlvXShtYWlsdG86c2VjdXJpdHlAc3dhZ2dlci5pbyksIGluc3RlYWQgb2YgdXNpbmcgdGhlIHB1YmxpYyBpc3N1ZSB0cmFja2VyLgoKIyMgTGljZW5zZQoKYGBgCkNvcHlyaWdodCAyMDE5IFNtYXJ0QmVhciBTb2Z0d2FyZQoKTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7CnlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0IFthcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wXShodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjApCgpVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCmRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCldJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgpTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCmxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgpgYGAK readmeEtag: '"1926fae77184799da27abea97b2530f40c480474"' readmeLastModified: Tue, 14 May 2024 09:01:26 GMT repositoryId: 87553967 description: OpenAPI/Swagger 2.0 to OpenAPI 3.0 Converter WebService created: '2017-04-07T14:18:38Z' updated: '2026-02-04T13:55:18Z' language: Shell archived: false stars: 149 watchers: 11 forks: 34 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"61466a84d5e02adaa7a011c6913cb3b505ed0a0c8533c7cf4978f89fcb392d5d"' repoLastModified: Wed, 04 Feb 2026 13:55:18 GMT foundInMaster: true category: - Code Generators - Parsers id: 97edfdde2e669f6ea8e322d590966d62 - source: openapi3 tags repository: https://github.com/oai/tools.openapis.org v3: true id: cf25f8e3e1410d3084950f72c6067f4e repositoryMetadata: base64Readme: >- # OpenAPI Tooling

This project is provided by the OpenAPI Initiative as a means to centralize ecosystem information on OpenAPI-related tooling. It leverages open-source projects that have gone before to provide a consolidated list of tooling.

The project is split into two features:

- A list of tooling merged from sources across the interwebs that users can grab and slice and dice as they see fit.
- A website that allows users to search and inspect the tooling data first-hand.

Each is expanded upon in the sections below.

## Roll Call

The following projects are being leveraged to provide the majority of the source information.

| Name          | Source                                           | Description                                                                                                                                        |
| ------------- | ------------------------------------------------ | -------------------------------------------------------------------------------------------------------------------------------------------------- |
| OpenAPI.Tools | https://github.com/apisyouwonthate/openapi.tools | APIs You Won't Hate efforts to create uber list of tooling.                                                                                        |
| APIs.guru     | https://github.com/apis-guru/awesome-openapi3    | Repository/site based on tagged repositories in Github.<br>This repository reuses the build approach rather than pulling the list from the source. |

## How Can You Help?

This project is designed to continue the work of APIs.guru and collect data based on repositories tagged with a topic.

If you want your project included in the tooling list tag your project with one-or-more of the following topics:

- `swagger` or `openapi2` (For Swagger/OpenAPI 2.0 support).
- `openapi3` (For OpenAPI 3.0 support).
- `openapi31` (For OpenAPI 3.1 support).

If you aren't familiar with topics in GitHub please follow [this guide](https://docs.github.com/en/repositories/managing-your-repositorys-settings-and-features/customizing-your-repository/classifying-your-repository-with-topics) to add them to your repository.

> **_Note: Collection of the `swagger`/`openapi2` topics is not currently implemented - see dependencies described in this [issue](https://github.com/OAI/Tooling/issues/19)._**

## Tooling List

The tooling list is built in largely the same format as the majority of projects that have blazed a trail in tooling before (which of course this project takes full advantage of).

In order to bring this together in a sensible way a Gulp-based process has been implemented. Gulp was chosen given the relative ease with which functions can be implemented to massage the data stream and to ensure the build is not closely coupled to a (commercial) CI tool. There are a couple of principles around the design worth stating:

- The transform functions that massage the data are abstracted away from Gulp to enable the build to "lift-and-shift" to a new build tool as required.
- Pipes between functions are always formatted as YAML to allow for simple dumping of the data for humans appraisal.
- The source data collection is written as independent packages referenced by metadata to allow new sources to be "slotted" in.

Note that if better tools are identified for the build then Gulp should be easy to change.

### Environment Variables

Access to the GitHub API is required to run the build. Access is supported through [basic authentication](https://docs.github.com/en/rest/guides/getting-started-with-the-rest-api#authentication) using a GitHub username and a [personal access token](https://github.com/settings/tokens/new) as environment variables.

The following variables are therefore required to run the build:

| Name                     | Description                                                                                                                                                                           |
| ------------------------ | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- |
| GH_API_USERNAME          | GitHub username to access the GitHub API                                                                                                                                              |
| GH_API_TOKEN             | OAuth/Personal Access Token to access the GitHub API                                                                                                                                  |
| GH_API_CONCURRENCY_TOKEN | Number of simultaneous connections to the GitHub API. Recommended value is 2.<br>Values greater than 2 appear to result in connections being throttled and the API returning a `403`. |

> **You must export these before running either of the data collection builds.**

We've used custom environment variables for GitHub API access rather than default GitHub variables provided by Actions. This provides both a separation-of-concerns between access controls and the build mechanism and enables higher rate limits.

> Note: We plan to introduce `dotenv` to help with the setting of environment variables.

### Full Build

The full build takes the following approach:

- Retrieve each tooling source, including the existing list at [`src/_data/tools.yaml`](src/_data/tools.yaml).
- Combine source data based on repository name.
- Normalise property names across sources using simple statistics (Sørensen–Dice, Damerau–Levenshtein distance).
- Get repository metadata from GitHub.
- Categorise the tools using Bayesian statistics.
- Write to [`src/_data/tools.yaml`](src/_data/tools.yaml).

Currently this build is scheduled using [GitHub Actions](.github/workflows/full.yaml) and runs once a week on Sunday.

The schedule will be reviewed as we collect data to see if executing it with greater frequency would be beneficial.

To run the full build locally:

```bash
yarn install

GH_API_USERNAME=<username> GH_API_TOKEN=<personal-access-token> GH_AP_CONCURRENCY_LIMIT=2
export GH_API_USERNAME GH_API_TOKEN GH_AP_CONCURRENCY_LIMIT
yarn run build:data:full
```

### Metadata Update

The goal of the metadata update is to provide consistent repository metadata without sourcing new tooling:

- Read the existing list at [`src/_data/tools.yaml`](src/_data/tools.yaml).
- Get repository metadata from GitHub.
- Write to [`src/_data/tools.yaml`](src/_data/tools.yaml).

Currently this build is scheduled using [GitHub Actions](.github/workflows/metadata.yaml) and runs every day.

The scheduled will be reviewed as we collect data to see if executing it with greater frequency would be beneficial.

To run the metadata build locally:

```bash
# If you haven't done this already
yarn install

# If you haven't done this already
GH_API_USERNAME=<username> GH_API_TOKEN=<personal-access-token> GH_AP_CONCURRENCY_LIMIT=2

# If you haven't done this already
export GH_API_USERNAME GH_API_TOKEN GH_AP_CONCURRENCY_LIMIT

yarn run build:metadata
```

### Testing Locally

To test locally you can clone or fork the Tooling repository and create yourself a `.env` file that meets your needs. For example:

```text
export GH_API_USERNAME=<your GitHub username>
export GH_API_TOKEN=<your GitHub personal access token>
export GH_API_CONCURRENCY_LIMIT=2
export TOOLING_REPOSITORY_OWNER=<your GitHub organisation or username>
export TOOLING_REPOSITORY_REPO_NAME=Tooling
```

With this in-hand there's a bunch of options to send into either `yarn run build:full` or `yarn run build:metadata`:

- `--metadata`: If you don't want to run everything you can change the configuration that drives the build (more on this below).
- `--env-file`: Supply an alternative .env file as described above.
- `--output-dir`: Change where you write the `tools.yaml` file.
- `--dry-run`: Don't do destructive things like closing issues (actually this is all it does right now).

#### Configuration File

The build is driven by a configuration file, the default being [`gulpfile.js/metadata.json`](gulpfile.js/metadata.json) which is validated at the start of the build using the JSON Schema found in [`validate-metadata.js`](lib/data/transform/validate-metadata.js).

The purpose of the configuration file is to define what data sources are collected. It contains an array of objects, each with two mandatory properties:

- `title`: The name of data processor. This is recorded in `tools.yaml` to identify which processor picked the data up.
- `processor`: Path to a JavaScript library that implements the data collection logic.

Other properties specific to a particular data source can be defined as required.

These data processors collect data and pass it into the Gulp pipeline for processing. That's all they do - everything else is done downstream in the JavaScript libraries found in the [`transform`](lib/data/transform/) directory.

If you want to test only a subset of data sources in isolation you can create your own configuration file. For example, if you are testing the GitHub issue-sourced data processor you can define just this in the configuration file - you'd only get those tools in the resultant `tools.yaml` file.

You should also consider whether to test with a subset of master data from `src/_data/tools.yaml` (as it is voluminous). You can edit your configuration file to point somewhere else e.g.:

```json
{
  "title": "master",
  "url": "<Path to your alternative tools.yaml file>",
  "processor": "../processors/master-processor.js"
}
```

## Website

The website is a static site built from the tooling data. It is exposed by GitHub Pages and can be found [here](https://tools.openapis.org/).

The design of the site is intentionally "lean", and provides the tooling list by category (the categorisation being done as described [above](#full-build)).

### Build

The site uses the [eleventy](https://www.11ty.dev/) site generator and is rebuilt after each [full](#full-build) and [metadata](#metadata-update) build, using the newly-updated data at [`src/_data/tools.yaml`](src/_data/tools.yaml).

To run the site build locally:

```bash
yarn install
yarn run build:site
```

> Note the build uses an environment variable `HOSTED_AT` to allow the site to be deployed to an alternative root URI and therefore amend the "Home" button. This is for the benefit of GitHub Pages, where the site is deployed to `/Tooling`. Unless you need to move URL then just leave this unset.

### Running locally

If you want to run the site locally it's just damn simple:

```bash
yarn install
yarn run serve
```

The development server is set to reload on change. Now isn't that convenient.

## Contributing

Please refer to the [Contributing Guide](CONTRIBUTING.md)
 readmeEtag: '"a9fe682e156cd528675b0389f2ed6b3667ec63ab"' readmeLastModified: Fri, 07 Nov 2025 10:41:30 GMT repositoryId: 448083351 description: >- A collection of open-source and commercial tools for creating your APIs with OpenAPI - Sourced from and published for the community created: '2022-01-14T19:24:14Z' updated: '2026-02-05T05:06:46Z' language: JavaScript archived: false stars: 159 watchers: 10 forks: 66 owner: OAI logo: https://avatars.githubusercontent.com/u/16343502?v=4 repoEtag: '"53b79419ca300ee714bd4a00ffa957cbc59f3d60a3ef71353d16931a7fca4954"' repoLastModified: Thu, 05 Feb 2026 05:06:46 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/oai/tooling - source: openapi3 tags repository: https://github.com/networknt/light-rest-4j v3: true repositoryMetadata: base64Readme: >- QSBSRVNUZnVsIEFQSSBvciBzZXJ2aWNlIGZyYW1ld29yayBidWlsdCBvbiB0b3Agb2YgbGlnaHQtNGoKCltTdGFjayBPdmVyZmxvd10oaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9xdWVzdGlvbnMvdGFnZ2VkL2xpZ2h0LTRqKSB8CltHb29nbGUgR3JvdXBdKGh0dHBzOi8vZ3JvdXBzLmdvb2dsZS5jb20vZm9ydW0vIyFmb3J1bS9saWdodC00aikgfApbR2l0dGVyIENoYXRdKGh0dHBzOi8vZ2l0dGVyLmltL25ldHdvcmtudC9saWdodC1yZXN0LTRqKSB8CltTdWJyZWRkaXRdKGh0dHBzOi8vd3d3LnJlZGRpdC5jb20vci9saWdodGFwaS8pIHwKW1lvdXR1YmUgQ2hhbm5lbF0oaHR0cHM6Ly93d3cueW91dHViZS5jb20vY2hhbm5lbC9VQ0hDUk1XSlZYdzhpQjd6S3hGNTVCeXcpIHwKW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vc3R5bGUvbGlnaHQtcmVzdC00ai8pIHwKW0NvbnRyaWJ1dGlvbiBHdWlkZV0oaHR0cHM6Ly9kb2MubmV0d29ya250LmNvbS9jb250cmlidXRlLykgfAoKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvbmV0d29ya250L2xpZ2h0LXJlc3QtNGouc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvbmV0d29ya250L2xpZ2h0LXJlc3QtNGopIFshW2NvZGVjb3YuaW9dKGh0dHBzOi8vY29kZWNvdi5pby9naXRodWIvbmV0d29ya250L2xpZ2h0LXJlc3QtNGovY292ZXJhZ2Uuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL2NvZGVjb3YuaW8vZ2l0aHViL25ldHdvcmtudC9saWdodC1yZXN0LTRqP2JyYW5jaD1tYXN0ZXIpCgpUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgbWlkZGxld2FyZSBoYW5kbGVycyB0aGF0IHdvcmsgd2l0aCBlaXRoZXIgU3dhZ2dlciAyLjAgb3IgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbi4gSWYgeW91IGFyZSBzdGFydGluZyBhIGJyYW5kIG5ldyBwcm9qZWN0LCBpdCBpcyBoaWdobHkgcmVjb21tZW5kZWQgdXNpbmcgdGhlIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gZm9yIHlvdXIgZGVzaWduLgoKVG8gZ2V0IHN0YXJ0ZWQgeW91ciBmaXJzdCBwcm9qZWN0LCBwbGVhc2UgdmlzaXQgdGhlIFtnZXR0aW5nIHN0YXJ0ZWRdKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vZ2V0dGluZy1zdGFydGVkL2xpZ2h0LXJlc3QtNGovKSB0byB3YWxrIHRocm91Z2ggdGhlIHByb2Nlc3MuCgpGb3IgZG9jdW1lbnRhdGlvbiBvbiB0aGUgaGFuZGxlcnMsIHBsZWFzZSB2aXNpdCB0aGUgW3JlZmVyZW5jZSBkb2N1bWVudF0oaHR0cHM6Ly9kb2MubmV0d29ya250LmNvbS9zdHlsZS9saWdodC1yZXN0LTRqLykuCgpUaGVyZSBhcmUgYWxzbyBudW1lcmljIFt0dXRvcmlhbHNdKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vdHV0b3JpYWwvcmVzdC8pIHRvIHNob3cgdXNlcnMgaG93IHRvIGJ1aWxkIFJlc3RmdWwgc2VydmljZXMuCg== readmeEtag: '"db3d10c58fce6adb30f08d263d0b14a71b36bf06"' readmeLastModified: Thu, 15 Aug 2024 15:56:25 GMT repositoryId: 79723681 description: >- A RESTful framework built on top of light-4j with both Swagger 2.0 and OpenAPI 3.0 supports created: '2017-01-22T14:48:44Z' updated: '2026-02-03T01:08:22Z' language: Java archived: false stars: 121 watchers: 24 forks: 31 owner: networknt logo: https://avatars.githubusercontent.com/u/8740739?v=4 license: Apache-2.0 repoEtag: '"e3268566df742075105c8ff1de47d70d344edcc35f991af569ba2a7de66fc210"' repoLastModified: Tue, 03 Feb 2026 01:08:22 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 5066b11d537986b50ebc1d3f7a7d721f - source: openapi3 tags repository: https://github.com/tobilg/api2html v3: true repositoryMetadata: base64Readme: >- IyBhcGkyaHRtbApBIENMSSB0b29sIHRvIHRyYW5zZm9ybSBTd2FnZ2VyL09wZW5BUEkvQXN5bmNBUEkgZG9jcyB0byBiZWF1dGlmdWwgSFRNTCBwYWdlcyB2aWEgW1NoaW5zXShodHRwczovL2dpdGh1Yi5jb20vbWVybWFkZS9zaGlucykvW1dpZGRlcnNoaW5zXShodHRwczovL2dpdGh1Yi5jb20vbWVybWFkZS93aWRkZXJzaGlucykuCgpZb3UgY2FuIGZpbmQgYW4gZXhhbXBsZSBnZW5lcmF0ZWQgcGFnZSBhdCBbaHR0cDovL3RvYmlsZy5naXRodWIuaW8vYXBpMmh0bWwvcGV0c3RvcmUvXShodHRwOi8vdG9iaWxnLmdpdGh1Yi5pby9hcGkyaHRtbC9wZXRzdG9yZS8pLgoKIyMgSW5zdGFsbGF0aW9uCgpUbyBpbnN0YWxsIGBhcGkyaHRtbGAgZ2xvYmFsbHksIHVzZQoKYGBgYmFzaAokIG5wbSBpIGFwaTJodG1sIC1nCmBgYAoKWW91IGNhbiBhbHNvIGluc3RhbGwgaXQgdG8gdXNlIGFzIGBkZXZEZXBlbmRlbmNpZXNgLCBhbmQgdXNlIGl0IGxvY2FsbHkgdmlhIGEgYG5wbSBydW5gIHRhc2sgaW4geW91ciBgcGFja2FnZS5qc29uYDoKCmBgYGJhc2gKJCBucG0gaSBhcGkyaHRtbCAtLXNhdmUtZGV2CmBgYAoKVXNhZ2UgaW4gYHBhY2thZ2UuanNvbmA6CgpgYGBqYXZhc2NyaXB0CnsKICAic2NyaXB0cyI6IHsKICAgICJhcGktZG9jcyI6ICJub2RlX21vZHVsZXMvLmJpbi9hcGkyaHRtbCAtbyBkb2NzL2FwaS5odG1sIC1sIHNoZWxsLGphdmFzY3JpcHQtLW5vZGVqcyBkb2NzL29wZW5hcGkvYXBpLnltbCIKICB9Cn0KYGBgCgojIyBVc2FnZQoKIyMjIEF2YWlsYWJsZSBjb21tYW5kcwoKYGBgYmFzaAokIGFwaTJodG1sIC0taGVscCAKVXNhZ2U6IGFwaTJodG1sIFtvcHRpb25zXSA8c291cmNlUGF0aD4KICAKICBPcHRpb25zOgogICAgLVYsIC0tdmVyc2lvbiAgICAgICAgICAgICAgICAgICBvdXRwdXQgdGhlIHZlcnNpb24gbnVtYmVyCiAgICAtciwgLS1yZXNvbHZlIDxzb3VyY2U+ICAgICAgICAgIHJlc29sdmUgZXh0ZXJuYWwgZGVwZW5kZW5jaWVzLCBzb3VyY2Ugc2hvdWxkIGJlIGEgdXJsIG9yIGEgcGF0aAogICAgLW8sIC0tb3V0IDxvdXRwdXRQYXRoPiAgICAgICAgICBvdXRwdXQgcGF0aCBmb3IgdGhlIHJlc3VsdGluZyBIVE1MIGRvY3VtZW50CiAgICAtdCwgLS10aGVtZSA8dGhlbWVOYW1lPiAgICAgICAgIHRoZW1lIHRvIHVzZSAoc2VlIGh0dHBzOi8vaGlnaGxpZ2h0anMub3JnL3N0YXRpYy9kZW1vLyBmb3IgYSBsaXN0KQogICAgLWMsIC0tY3VzdG9tTG9nbyA8bG9nb1BhdGg+ICAgICB1c2UgY3VzdG9tIGxvZ28gYXQgdGhlIHJlc3BlY3RpdmUgcGF0aAogICAgLXUsIC0tY3VzdG9tTG9nb1VybCA8bG9nb1VSTD4gICB1cmwgZm9yIHRoZSBjdXN0b20gbG9nbyB0byBwb2ludCB0bwogICAgLUMsIC0tY3VzdG9tQ3NzICAgICAgICAgICAgICAgICB1c2UgY3VzdG9tIGNzcwogICAgLVAsIC0tY3VzdG9tQ3NzUGF0aCA8Y3NzUGF0aD4gICB1c2UgY3VzdG9tIGNzcyBmaWxlCiAgICAtaSwgLS1pbmNsdWRlcyA8aW5jbHVkZXNMaXN0PiAgIGNvbW1hLXNlcGFyYXRlZCBsaXN0IG9mIGZpbGVzIHRvIGluY2x1ZGUKICAgIC1sLCAtLWxhbmd1YWdlcyA8bGFuZ3VhZ2VMaXN0PiAgY29tbWEtc2VwYXJhdGVkIGxpc3Qgb2YgbGFuZ3VhZ2VzIHRvIHVzZSBmb3IgdGhlIGxhbmd1YWdlIHRhYnMgKG91dCBvZiBzaGVsbCwgaHR0cCwgamF2YXNjcmlwdCwgamF2YXNjcmlwdC0tbm9kZWpzLCBydWJ5LCBweXRob24sIGphdmEsIGdvKQogICAgLU4sIC0tbm9Db2RlU2FtcGxlcyAgICAgICAgICAgICBvbWl0IGFsbCBjb2RlIHNhbXBsZXMgKG92ZXJyaWRlcyAtLWxhbmd1YWdlcykKICAgIC1zLCAtLXNlYXJjaCAgICAgICAgICAgICAgICAgICAgZW5hYmxlIHNlYXJjaAogICAgLVMsIC0tc3VtbWFyeSAgICAgICAgICAgICAgICAgICB1c2Ugc3VtbWFyeSBpbnN0ZWFkIG9mIG9wZXJhdGlvbklkIGZvciBUT0MKICAgIC1iLCAtLW9taXRCb2R5ICAgICAgICAgICAgICAgICAgT21pdCB0b3AtbGV2ZWwgZmFrZSBib2R5IHBhcmFtZXRlciBvYmplY3QKICAgIC1SLCAtLXJhdyAgICAgICAgICAgICAgICAgICAgICAgU2hvdyByYXcgc2NoZW1hcyBpbiBzYW1wbGVzLCBub3QgZXhhbXBsZSB2YWx1ZXMKICAgIC1oLCAtLWhlbHAgICAgICAgICAgICAgICAgICAgICAgb3V0cHV0IHVzYWdlIGluZm9ybWF0aW9uCmBgYAoKIyMjIFVzYWdlIGV4YW1wbGVzCgojIyMjIFJlbmRlciBPcGVuQVBJIHYzIGZpbGUgYXMgSFRNTAoKVGhpcyB3aWxsIHJlbmRlciB0aGUgYGFwaS55bWxgIGZpbGUgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5IGFzIGBteWFwaS5odG1sYCBmaWxlIGluIHRoZSBjdXJyZW50IGRpcmVjdG9yeS4KCmBgYGJhc2gKJCBhcGkyaHRtbCAtbyBteWFwaS5odG1sIG15YXBpLnltbApgYGAKCiMjIyMgVXNlIGN1c3RvbSBsb2dvCgpUaGlzIHdpbGwgcmVuZGVyIHRoZSBgYXBpLnltbGAgZmlsZSBpbiB0aGUgY3VycmVudCBkaXJlY3RvcnkgYXMgYG15YXBpLmh0bWxgIGZpbGUgaW4gdGhlIHNhbWUgZGlyZWN0b3J5LCBhbmQgdXNlIHRoZSBjdXN0b20gbG9nbyBgbXlsb2dvLnBuZ2AuCgpgYGBiYXNoCiQgYXBpMmh0bWwgLW8gbXlhcGkuaHRtbCAtYyBteWxvZ28ucG5nIG15YXBpLnltbApgYGAKCiMjIyMgRGVmaW5lIHdoaWNoIGxhbmd1YWdlIGV4YW1wbGVzIHNob3VsZCBiZSBnZW5lcmF0ZWQKClRoaXMgd2lsbCByZW5kZXIgdGhlIGBhcGkueW1sYCBmaWxlIGluIHRoZSBjdXJyZW50IGRpcmVjdG9yeSBhcyBgbXlhcGkuaHRtbGAgZmlsZSBpbiB0aGUgc2FtZSBkaXJlY3RvcnksIGFuZCB1c2UgYGdvYCBhbmQgYGphdmFzY3JpcHRgIGV4YW1wbGVzLgoKYGBgYmFzaAokIGFwaTJodG1sIC1vIG15YXBpLmh0bWwgLWwgZ28samF2YXNjcmlwdCBteWFwaS55bWwKYGBgCgojIyMjIFVzZSBkaWZmZXJlbnQgc3ludGF4IGhpZ2hsaWdodGVyCgpUaGlzIHdpbGwgcmVuZGVyIHRoZSBgYXBpLnltbGAgZmlsZSBpbiB0aGUgY3VycmVudCBkaXJlY3RvcnkgYXMgYG15YXBpLmh0bWxgIGZpbGUgaW4gdGhlIHNhbWUgZGlyZWN0b3J5LCBhbmQgdXNlIGBnb2AgYW5kIGBqYXZhc2NyaXB0YCBleGFtcGxlcywgYXMgd2VsbCBhcyBhIGRpZmZlcmVudCBzeW50YXggaGlnbGlnaHRlciBmcm9tIFtoaWdobGlnaHQuanNdKGh0dHBzOi8vaGlnaGxpZ2h0anMub3JnL3N0YXRpYy9kZW1vLykuCgpgYGBiYXNoCiQgYXBpMmh0bWwgLW8gbXlhcGkuaHRtbCAtbCBnbyxqYXZhc2NyaXB0IC10IGFydGEgbXlhcGkueW1sCmBgYAoKIyMjIyBSZXNvbHZlIGV4dGVybmFsIGRlcGVuZGVuY2llcwoKSWYgeW91IGFkZCByZWZzIHRvIGV4dGVybmFsIGZpbGVzIGluIHlvdXIgc291cmNlIGZpbGUsIHlvdSBjYW4gZW5hYmxlIHRoZW0gYnkgdXNpbmcgYC1yIDxzb3VyY2U+YC4gVGhlIGZvbGxvd2luZyBjb21tYW5kIHdpbGwgcmVzb2x2ZSBhbGwgeW91ciByZWxhdGl2ZSBpbXBvcnRzIGZyb20gdGhlIGN1cnJlbnQgZGlyZWN0b3J5LgoKYGBgYmFzaAokIGFwaTJodG1sIC1vIG15YXBpLmh0bWwgLXIgLi8gbXlhcGkueW1sCmBgYAo= readmeEtag: '"2c38b17a93ff4ff6c74a44a2c25018e74ef8d708"' readmeLastModified: Thu, 21 Sep 2023 07:04:29 GMT repositoryId: 133497107 description: >- A CLI tool to transform Swagger/OpenAPI/AsyncAPI docs to beautiful HTML pages via Shins/Widdershins. created: '2018-05-15T10:03:41Z' updated: '2025-05-15T12:10:26Z' language: JavaScript archived: false stars: 120 watchers: 3 forks: 25 owner: tobilg logo: https://avatars.githubusercontent.com/u/1841985?v=4 license: MIT repoEtag: '"9e9df3db44ce06492837435b0ed42ce00749d240b60cc3dbf0f816d8fe0f74d0"' repoLastModified: Thu, 15 May 2025 12:10:26 GMT foundInMaster: true category: - Documentation - Parsers id: 90756f66a4bf98b61166ca0c0cde5583 - source: openapi3 tags repository: https://github.com/snok/drf-openapi-tester v3: true repositoryMetadata: base64Readme: >- [![PyPI](https://img.shields.io/pypi/v/drf-openapi-tester.svg)](https://pypi.org/project/drf-openapi-tester/)
[![Coverage](https://codecov.io/gh/snok/drf-openapi-tester/branch/master/graph/badge.svg)](https://codecov.io/gh/snok/drf-openapi-tester)
[![Python versions](https://img.shields.io/badge/Python-3.7%2B-blue)](https://pypi.org/project/drf-openapi-tester/)
[![Django versions](https://img.shields.io/badge/Django-3.0%2B-blue)](https://pypi.org/project/drf-openapi-tester/)


# DRF OpenAPI Tester

This is a test utility to validate DRF Test Responses against OpenAPI 2 and 3 schema. It has built-in support for:

- OpenAPI 2/3 yaml or json schema files.
- OpenAPI 2 schemas created with [drf-yasg](https://github.com/axnsan12/drf-yasg).
- OpenAPI 3 schemas created with [drf-spectacular](https://github.com/tfranzel/drf-spectacular).

> This project is no longer maintained. If you want to use it, consider taking a look at [this fork](https://github.com/maticardenas/drf-openapi-tester/) which might still be.

## Installation

```shell script
pip install drf-openapi-tester
```

## Usage

Instantiate one or more instances of `SchemaTester`:

```python
from openapi_tester import SchemaTester

schema_tester = SchemaTester()
```

If you are using either [drf-yasg](https://github.com/axnsan12/drf-yasg)
or [drf-spectacular](https://github.com/tfranzel/drf-spectacular) this will be auto-detected, and the schema will be
loaded by the `SchemaTester` automatically.

If you are using schema files, you will need to pass the file path:

```python
from openapi_tester import SchemaTester

# path should be a string
schema_tester = SchemaTester(schema_file_path="./schemas/publishedSpecs.yaml")
```

Once you've instantiated a tester, you can use it to test responses:

```python
from openapi_tester.schema_tester import SchemaTester

schema_tester = SchemaTester()


def test_response_documentation(client):
    response = client.get('api/v1/test/1')
    assert response.status_code == 200
    schema_tester.validate_response(response=response)
```

If you are using the Django testing framework, you can create a base `APITestCase` that incorporates schema validation:

```python
from rest_framework.response import Response
from rest_framework.test import APITestCase

from openapi_tester.schema_tester import SchemaTester

schema_tester = SchemaTester()


class BaseAPITestCase(APITestCase):
    """ Base test class for api views including schema validation """

    @staticmethod
    def assertResponse(response: Response, **kwargs) -> None:
        """ helper to run validate_response and pass kwargs to it """
        schema_tester.validate_response(response=response, **kwargs)
```

Then use it in a test file:

```python
from shared.testing import BaseAPITestCase


class MyAPITests(BaseAPITestCase):
    def test_some_view(self):
        response = self.client.get("...")
        self.assertResponse(response)
```

## Options

You can pass options either globally, when instantiating a `SchemaTester`, or locally, when
invoking `validate_response`:

```python
from openapi_tester import SchemaTester, is_camel_case
from tests.utils import my_uuid_4_validator

schema_test_with_case_validation = SchemaTester(
    case_tester=is_camel_case,
    ignore_case=["IP"],
    validators=[my_uuid_4_validator]
)

```

Or

```python
from openapi_tester import SchemaTester, is_camel_case
from tests.utils import my_uuid_4_validator

schema_tester = SchemaTester()


def my_test(client):
    response = client.get('api/v1/test/1')
    assert response.status_code == 200
    schema_tester.validate_response(
        response=response,
        case_tester=is_camel_case,
        ignore_case=["IP"],
        validators=[my_uuid_4_validator]
    )
```

### case_tester

The case tester argument takes a callable that is used to validate the key case of both schemas and responses. If
nothing is passed, case validation is skipped.

The library currently has 4 built-in case testers:

- `is_pascal_case`
- `is_snake_case`
- `is_camel_case`
- `is_kebab_case`

You can use one of these, or your own.

### ignore_case

List of keys to ignore when testing key case. This setting only applies when case_tester is not `None`.

### validators

List of custom validators. A validator is a function that receives two parameters: schema_section and data, and returns
either an error message or `None`, e.g.:

```python
from typing import Any, Optional
from uuid import UUID


def my_uuid_4_validator(schema_section: dict, data: Any) -> Optional[str]:
    schema_format = schema_section.get("format")
    if schema_format == "uuid4":
        try:
            result = UUID(data, version=4)
            if not str(result) == str(data):
                return f"Expected uuid4, but received {data}"
        except ValueError:
            return f"Expected uuid4, but received {data}"
    return None
```

### field_key_map

You can pass an optional dictionary that maps custom url parameter names into values, for situations where this cannot be
inferred by the DRF `EndpointEnumerator`. A concrete use case for this option is when
the [django i18n locale prefixes](https://docs.djangoproject.com/en/3.1/topics/i18n/translation/#language-prefix-in-url-patterns).

```python
from openapi_tester import SchemaTester

schema_tester = SchemaTester(field_key_map={
  "language": "en",
})
```

## Schema Validation

When the SchemaTester loads a schema, it parses it using an
[OpenAPI spec validator](https://github.com/p1c2u/openapi-spec-validator). This validates the schema.
In case of issues with the schema itself, the validator will raise the appropriate error.

## Django testing client

The library includes an `OpenAPIClient`, which extends Django REST framework's
[`APIClient` class](https://www.django-rest-framework.org/api-guide/testing/#apiclient).
If you wish to validate each response against OpenAPI schema when writing
unit tests - `OpenAPIClient` is what you need!

To use `OpenAPIClient` simply pass `SchemaTester` instance that should be used
to validate responses and then use it like regular Django testing client:

```python
schema_tester = SchemaTester()
client = OpenAPIClient(schema_tester=schema_tester)
response = client.get('/api/v1/tests/123/')
```

To force all developers working on the project to use `OpenAPIClient` simply
override the `client` fixture (when using `pytest` with `pytest-django`):

```python
from pytest_django.lazy_django import skip_if_no_django

from openapi_tester.schema_tester import SchemaTester


@pytest.fixture
def schema_tester():
    return SchemaTester()


@pytest.fixture
def client(schema_tester):
    skip_if_no_django()

    from openapi_tester.clients import OpenAPIClient

    return OpenAPIClient(schema_tester=schema_tester)
```

If you are using plain Django test framework, we suggest to create custom
test case implementation and use it instead of original Django one:

```python
import functools

from django.test.testcases import SimpleTestCase
from openapi_tester.clients import OpenAPIClient
from openapi_tester.schema_tester import SchemaTester

schema_tester = SchemaTester()


class MySimpleTestCase(SimpleTestCase):
    client_class = OpenAPIClient
    # or use `functools.partial` when you want to provide custom
    # ``SchemaTester`` instance:
    # client_class = functools.partial(OpenAPIClient, schema_tester=schema_tester)
```

This will ensure you all newly implemented views will be validated against
the OpenAPI schema.

## Known Issues

* We are using [prance](https://github.com/jfinkhaeuser/prance) as a schema resolver, and it has some issues with the
  resolution of (very) complex OpenAPI 2.0 schemas. If you encounter
  issues, [please document them here](https://github.com/snok/drf-openapi-tester/issues/205).

## Contributing

Contributions are welcome. Please see the [contributing guide](https://github.com/snok/.github/blob/main/CONTRIBUTING.md)
 readmeEtag: '"84e7f0f953f52e829df14422c6ac971d44864a73"' readmeLastModified: Sun, 19 Nov 2023 17:47:20 GMT repositoryId: 232892108 description: Test utility for validating OpenAPI documentation created: '2020-01-09T19:49:31Z' updated: '2025-11-28T11:56:50Z' language: Python archived: true stars: 118 watchers: 6 forks: 25 owner: snok logo: https://avatars.githubusercontent.com/u/64945977?v=4 license: BSD-3-Clause repoEtag: '"c3d1a71648e126f6e6bfc987391ff600f03a37ddef576648868364d52bd06ac7"' repoLastModified: Fri, 28 Nov 2025 11:56:50 GMT foundInMaster: true category: Data Validators id: ed3a0abc85391a21e6ab59d06a1270d1 - source: openapi3 tags repository: https://github.com/sphinx-contrib/openapi v3: true repositoryMetadata: base64Readme: >- PT09PT09PT09PT09PT09PT09PT09CnNwaGlueGNvbnRyaWItb3BlbmFwaQo9PT09PT09PT09PT09PT09PT09PT0KCioqc3BoaW54Y29udHJpYi1vcGVuYXBpKiogaXMgYSBgU3BoaW54YF8gZXh0ZW5zaW9uIHRvIGdlbmVyYXRlIEFQSXMgZG9jcyBmcm9tCmBPcGVuQVBJYF8gKGZrYSBTd2FnZ2VyKSBzcGVjLiBJdCBkZXBlbmRzIG9uIGBzcGhpbnhjb250cmliLWh0dHBkb21haW5gXyB0aGF0CnByb3ZpZGVzIGFuIEhUVFAgZG9tYWluIGZvciBkZXNjcmliaW5nIFJFU1RmdWwgSFRUUCBBUElzLCBzbyB3ZSBkb24ndCBuZWVkIHRvCnJlaW52ZW50IHRoZSB3aGVlbC4KCi4uIGNvZGU6OiBiYXNoCgogICAkIHB5dGhvbjMgLW0gcGlwIGluc3RhbGwgc3BoaW54Y29udHJpYi1vcGVuYXBpCgoKVXNhZ2UKPT09PT0KClBhc3MgYGBzcGhpbnhjb250cmliLW9wZW5hcGlgYCB0byBgYGV4dGVuc2lvbnNgYCBsaXN0IGluICBTcGhpbngncyBgYGNvbmYucHlgYAoKLi4gY29kZTo6IHB5dGhvbgoKICAgZXh0ZW5zaW9ucyA9IFsKICAgICAgLi4uCiAgICAgICdzcGhpbnhjb250cmliLm9wZW5hcGknLAogICBdCgphbmQgZmVlbCBmcmVlIHRvIHVzZSB0aGUgYGBvcGVuYXBpYGAgZGlyZWN0aXZlIHRvIHJlbmRlciBPcGVuQVBJIHNwZWNzCgouLiBjb2RlOjogcmVzdHJ1Y3R1cmVkdGV4dAoKICAgLi4gb3BlbmFwaTo6IHBhdGgvdG8vb3BlbmFwaS55bWwKCgpMaW5rcwo9PT09PQoKKiBEb2N1bWVudGF0aW9uOiBodHRwczovL3NwaGlueGNvbnRyaWItb3BlbmFwaS5yZWFkdGhlZG9jcy5vcmcvCiogU291cmNlOiBodHRwczovL2dpdGh1Yi5jb20vc3BoaW54LWNvbnRyaWIvb3BlbmFwaQoqIEJ1Z3M6IGh0dHBzOi8vZ2l0aHViLmNvbS9zcGhpbngtY29udHJpYi9vcGVuYXBpL2lzc3VlcwoKCi4uIF9TcGhpbng6IGh0dHBzOi8vd3d3LnNwaGlueC1kb2Mub3JnL2VuL21hc3Rlci8KLi4gX09wZW5BUEk6IGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uCi4uIF9zcGhpbnhjb250cmliLWh0dHBkb21haW46IGh0dHBzOi8vc3BoaW54Y29udHJpYi1odHRwZG9tYWluLnJlYWR0aGVkb2NzLmlvLwo= readmeEtag: '"6dfb1836efc473494f87ce2831c9edf633954831"' readmeLastModified: Tue, 13 Feb 2024 10:44:53 GMT repositoryId: 68995825 description: OpenAPI (fka Swagger) spec renderer for Sphinx. created: '2016-09-23T06:28:32Z' updated: '2026-02-04T09:46:16Z' language: Python archived: false stars: 122 watchers: 9 forks: 87 owner: sphinx-contrib logo: https://avatars.githubusercontent.com/u/31936682?v=4 license: BSD-2-Clause repoEtag: '"16294a70226b0ec9898421e99201d1e32bd6471e4f2ab93461d6377bb76d0b93"' repoLastModified: Wed, 04 Feb 2026 09:46:16 GMT foundInMaster: true category: - Converters - Parsers id: 5d08f10c77ed85b8c8c49382bb5bef3a - source: openapi3 tags repository: https://github.com/moclojer/moclojer v3: true id: 5f88120dcd8e4bcd38a9215d0e1ec08b repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://github.com/moclojer/moclojer">
    <picture>
      <source media="(prefers-color-scheme: dark)" srcset="https://github.com/moclojer/moclojer/raw/main/docs/assets/logo.png">
      <img src="https://github.com/moclojer/moclojer/raw/main/docs/assets/logo.png" height="128">
    </picture>
    <h1 align="center">moclojer</h1>
  </a>
</p>

<p align="center">
  <a aria-label="CLAs Agree" href="https://cla-assistant.io/moclojer/moclojer" target="_blank">
    <img src="https://img.shields.io/badge/CLAs-ff009e.svg?style=for-the-badge&labelColor=000000&label=agree">
  </a>
  <a aria-label="License" href="https://github.com/moclojer/moclojer/blob/main/LICENSE">
    <img alt="" src="https://img.shields.io/badge/MIT-ff009e.svg?style=for-the-badge&labelColor=000000&label=LICENSE">
  </a>
  <a aria-label="Join the community on GitHub" href="https://github.com/moclojer/moclojer/discussions">
    <img alt="" src="https://img.shields.io/badge/Join%20the%20community-ff009e.svg?style=for-the-badge&labelColor=000000&label=Discussion">
  </a>

  <img src="https://github.com/moclojer/moclojer/actions/workflows/linter.yml/badge.svg?branch=main" alt="linter">

  <a aria-label="CI Tests" href="https://github.com/moclojer/moclojer/actions/workflows/tests.yml">
    <img src="https://github.com/moclojer/moclojer/actions/workflows/tests.yml/badge.svg?branch=main" alt="tests">
  </a>
</p>

<p align="center">
  <a href="https://www.producthunt.com/posts/moclojer?utm_source=badge-featured&utm_medium=badge&utm_souce=badge-moclojer" target="_blank"><img src="https://api.producthunt.com/widgets/embed-image/v1/featured.svg?post_id=449961&theme=neutral" alt="moclojer - &#0032;Simple&#0032;and&#0032;efficient&#0032;HTTP&#0032;mock&#0032;server&#0032;with&#0032;easy&#0032;spec | Product Hunt" style="width: 250px; height: 54px;" width="250" height="54" /></a>
</p>

Simple and efficient HTTP mock server with specification written in `yaml`, `edn`, `OpenAPI` or `Postman Collection`.

> 💾 Download the `.jar` file with the latest version of moclojer to test on your computer [here](https://github.com/moclojer/moclojer/releases/latest).

[**📖 See the complete documentation for moclojer here**](https://docs.moclojer.com/), if you want to contribute (or complement) the documentation, it is [here](https://github.com/moclojer/moclojer/tree/main/docs).

**`YAML` example**

```yaml
# This mock register route: GET /hello/:username
- endpoint:
    # Note: the method could be omitted because GET is the default
    method: GET
    path: /hello/:username
    response:
      # Note: the status could be omitted because 200 is the default
      status: 200
      headers:
        Content-Type: application/json
      # Note: the body will receive the value passed in the url using the
      # :username placeholder
      body: >
        {
          "hello": "{{path-params.username}}!"
        }
```

**Postman Collection Support**

Moclojer can directly use Postman Collection v2.1 exports as mock specifications. Simply export your Postman collection and use it:

```sh
# Using environment variable
CONFIG=my-collection.json moclojer

# Using CLI parameter
moclojer --config my-collection.json
```

**How to export from Postman:**

1. Open your Postman collection
2. Click the three dots menu → Export
3. Select "Collection v2.1" format
4. Save the JSON file
5. Use it directly with moclojer!

The converter automatically:
- Detects Postman Collection format
- Extracts HTTP methods, paths, and response examples
- Processes nested folders
- Converts path variables (`:id`, `:username`, etc.)
- Includes headers and status codes from examples

**WebSocket Support**

Moclojer also supports WebSocket connections with a simple configuration approach:

```yaml
# WebSocket echo server
- websocket:
    path: /ws/echo
    on-connect:
      # Message sent when client connects
      response: '{"status": "connected", "message": "Welcome to WebSocket Echo!"}'
    on-message:
      # Simple echo for "ping" message
      - pattern: "ping"
        response: "pong"
      # Echo back any JSON content with an "echo" field
      - pattern: '{"echo": "{{json-params.echo}}"}'
        response: '{"echoed": "{{json-params.echo}}"}'
```

You can test the WebSocket connection using tools like `websocat`:

```sh
websocat "ws://localhost:8000/ws/echo" --text
```

## docker

* **image:** `ghcr.io/moclojer/moclojer:latest`
* **port _(default)_:** `8000`_, if you want to change the port set the environment variable `PORT`_

```sh
docker run -it \
  -p 8000:8000 -v $(pwd)/moclojer.yml:/app/moclojer.yml \
  ghcr.io/moclojer/moclojer:latest
```

**We have two versions available:**

* `dev`: main branch docker image
* `latest`: latest stable version image

## manual installation

We distribute via the `.jar` file, you need to have Java installed on your operating system.

```sh
bash < <(curl -s https://raw.githubusercontent.com/moclojer/moclojer/main/install.sh)
```

> If you are using Linux you maybe need `sudo`.

## CLI Usage

* `clj -M:run [OPTIONS]`
* `java -jar moclojer.jar [OPTIONS]`
* `moclojer_Linux [OPTIONS]`

### Options

parameter | description
--- | ---
`-c, --config` | Config path or the CONFIG environment variable. \[**default:** `~/.config/moclojer.yml`\]
`-m, --mocks` | OpenAPI v3 mocks path or the MOCKS environment variable.
`-f, --format` | Output and logging format. Either `println` or `json`.
`-h, --help` | Show help information
`-v, --version` | Show version information

**sentry:** set environment var `SENTRY_DSN` ([sentry doc](https://docs.sentry.io/platforms/node/guides/azure-functions/configuration/options/#dsn)), automatic send backtrace to <sentry.io>

> **Config** uses `XDG_CONFIG_HOME` to fetch the default moclojer configuration file, if you want to set a different directory you must use the `-c` or environment variable `CONFIG`

## 💻 dev environment

moclojer is written in **Clojure**, to run the commands below we assume you have clojure _installed_ on your operating system.

**run:**

```sh
clj -M:run
```

**test:**

```sh
clj -M:test
```

> _if you want to run a specific test:_ `clj -M:test -n com.moclojer.external-body.excel-test`

**`moclojer.jar` generate:**

```sh
clj -A:dev -M --report stderr -m com.moclojer.build
```

## AI-assisted development

This project includes an [`AGENTS.md`](AGENTS.md) file with guidelines for AI coding assistants (Claude Code, Cursor, GitHub Copilot, etc.). It documents architecture decisions, code patterns, extension points, and common pitfalls to help both humans and AI tools understand and contribute to the codebase effectively.

> **Windows users:** This project uses symlinks for AI tool config files. Git on Windows disables symlinks by default. Enable them with `git config --global core.symlinks true` before cloning (may require Developer Mode or admin privileges).

## framework integrations

We distribute the library via [Clojars](https://clojars.org/com.moclojer/moclojer).

### Clojure CLI/deps.edn

```clojure
com.moclojer/moclojer {:mvn/version "0.3.1"}
```

### Leiningen/Boot

```clojure
[com.moclojer/moclojer "0.3.1"]
```

### [`git`](https://clojure.org/guides/deps_and_cli#_using_git_libraries) in `deps.edn`

```edn
{:deps
 {com.moclojer/moclojer {:git/url "https://github.com/moclojer/moclojer.git"
                         :git/tag "v0.3.1"
                         :git/sha "c4ca0f2cfcfbe47de6eb0c601b26106190e20793"}}}
```

### example of use

```clj
(ns my-app.core
  (:require [com.moclojer.adapters :as adapters]
            [com.moclojer.server :as server]))

(def *router
  "create a router from a config map"
  (adapters/generate-routes
   [{:endpoint
     {:method "GET"
      :path "/example"
      :response {:status 200
                 :headers {:Content-Type "application/json"}
                 :body {:id 123}}}}]))

(defn -main
  "start the server"
  [& args]
  (server/start-server! *router))
```
 readmeEtag: '"8fa47901a7666c9457bd80ed5078b8b6abe4003e"' readmeLastModified: Sun, 21 Dec 2025 21:18:07 GMT repositoryId: 462693523 description: >- Simple and efficient HTTP and Webscoket mock server with specification in yaml, edn or OpenAPI created: '2022-02-23T10:49:57Z' updated: '2025-12-27T05:49:28Z' language: Clojure archived: false stars: 130 watchers: 1 forks: 13 owner: moclojer logo: https://avatars.githubusercontent.com/u/103039458?v=4 license: MIT repoEtag: '"4592bd6fd9478845931439a4fea8473b735ca781a006ae233251c8b06827136c"' repoLastModified: Sat, 27 Dec 2025 05:49:28 GMT category: Testing foundInMaster: true oldLocations: - https://github.com/avelino/moclojer - source: openapi3 tags repository: https://github.com/kuimono/openapi-schema-pydantic v3: true repositoryMetadata: base64Readme: >- # openapi-schema-pydantic

[![PyPI](https://img.shields.io/pypi/v/openapi-schema-pydantic)](https://pypi.org/project/openapi-schema-pydantic/)
[![PyPI - License](https://img.shields.io/pypi/l/openapi-schema-pydantic)](https://github.com/kuimono/openapi-schema-pydantic/blob/master/LICENSE)

OpenAPI (v3) specification schema as [Pydantic](https://github.com/samuelcolvin/pydantic) classes.

The naming of the classes follows the schema in 
[OpenAPI specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.1.0.md#schema).

## Installation

`pip install openapi-schema-pydantic`

## Try me

```python
from openapi_schema_pydantic import OpenAPI, Info, PathItem, Operation, Response

# Construct OpenAPI by pydantic objects
open_api = OpenAPI(
    info=Info(
        title="My own API",
        version="v0.0.1",
    ),
    paths={
        "/ping": PathItem(
            get=Operation(
                responses={
                    "200": Response(
                        description="pong"
                    )
                }
            )
        )
    },
)
print(open_api.json(by_alias=True, exclude_none=True, indent=2))
```

Result:

```json
{
  "openapi": "3.1.0",
  "info": {
    "title": "My own API",
    "version": "v0.0.1"
  },
  "servers": [
    {
      "url": "/"
    }
  ],
  "paths": {
    "/ping": {
      "get": {
        "responses": {
          "200": {
            "description": "pong"
          }
        },
        "deprecated": false
      }
    }
  }
}
```

## Take advantage of Pydantic

Pydantic is a great tool, allow you to use object / dict / mixed data for for input.

The following examples give the same OpenAPI result as above:

```python
from openapi_schema_pydantic import OpenAPI, PathItem, Response

# Construct OpenAPI from dict
open_api = OpenAPI.parse_obj({
    "info": {"title": "My own API", "version": "v0.0.1"},
    "paths": {
        "/ping": {
            "get": {"responses": {"200": {"description": "pong"}}}
        }
    },
})

# Construct OpenAPI with mix of dict/object
open_api = OpenAPI.parse_obj({
    "info": {"title": "My own API", "version": "v0.0.1"},
    "paths": {
        "/ping": PathItem(
            get={"responses": {"200": Response(description="pong")}}
        )
    },
})
```

## Use Pydantic classes as schema

- The [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md#schemaObject)
  in OpenAPI has definitions and tweaks in JSON Schema, which is hard to comprehend and define a good data class
- Pydantic already has a good way to [create JSON schema](https://pydantic-docs.helpmanual.io/usage/schema/),
  let's not re-invent the wheel
  
The approach to deal with this:

1. Use `PydanticSchema` objects to represent the `Schema` in `OpenAPI` object
2. Invoke `construct_open_api_with_schema_class` to resolve the JSON schemas and references

```python
from pydantic import BaseModel, Field

from openapi_schema_pydantic import OpenAPI
from openapi_schema_pydantic.util import PydanticSchema, construct_open_api_with_schema_class

def construct_base_open_api() -> OpenAPI:
    return OpenAPI.parse_obj({
        "info": {"title": "My own API", "version": "v0.0.1"},
        "paths": {
            "/ping": {
                "post": {
                    "requestBody": {"content": {"application/json": {
                        "schema": PydanticSchema(schema_class=PingRequest)
                    }}},
                    "responses": {"200": {
                        "description": "pong",
                        "content": {"application/json": {
                            "schema": PydanticSchema(schema_class=PingResponse)
                        }},
                    }},
                }
            }
        },
    })

class PingRequest(BaseModel):
    """Ping Request"""
    req_foo: str = Field(description="foo value of the request")
    req_bar: str = Field(description="bar value of the request")

class PingResponse(BaseModel):
    """Ping response"""
    resp_foo: str = Field(description="foo value of the response")
    resp_bar: str = Field(description="bar value of the response")

open_api = construct_base_open_api()
open_api = construct_open_api_with_schema_class(open_api)

# print the result openapi.json
print(open_api.json(by_alias=True, exclude_none=True, indent=2))
```

Result:

```json
{
  "openapi": "3.1.0",
  "info": {
    "title": "My own API",
    "version": "v0.0.1"
  },
  "servers": [
    {
      "url": "/"
    }
  ],
  "paths": {
    "/ping": {
      "post": {
        "requestBody": {
          "content": {
            "application/json": {
              "schema": {
                "$ref": "#/components/schemas/PingRequest"
              }
            }
          },
          "required": false
        },
        "responses": {
          "200": {
            "description": "pong",
            "content": {
              "application/json": {
                "schema": {
                  "$ref": "#/components/schemas/PingResponse"
                }
              }
            }
          }
        },
        "deprecated": false
      }
    }
  },
  "components": {
    "schemas": {
      "PingRequest": {
        "title": "PingRequest",
        "required": [
          "req_foo",
          "req_bar"
        ],
        "type": "object",
        "properties": {
          "req_foo": {
            "title": "Req Foo",
            "type": "string",
            "description": "foo value of the request"
          },
          "req_bar": {
            "title": "Req Bar",
            "type": "string",
            "description": "bar value of the request"
          }
        },
        "description": "Ping Request"
      },
      "PingResponse": {
        "title": "PingResponse",
        "required": [
          "resp_foo",
          "resp_bar"
        ],
        "type": "object",
        "properties": {
          "resp_foo": {
            "title": "Resp Foo",
            "type": "string",
            "description": "foo value of the response"
          },
          "resp_bar": {
            "title": "Resp Bar",
            "type": "string",
            "description": "bar value of the response"
          }
        },
        "description": "Ping response"
      }
    }
  }
}
```

## Notes

### Use of OpenAPI.json() / OpenAPI.dict()

When using `OpenAPI.json()` / `OpenAPI.dict()` function,
arguments `by_alias=True, exclude_none=True` has to be in place.
Otherwise the result json will not fit the OpenAPI standard.

```python
# OK
open_api.json(by_alias=True, exclude_none=True, indent=2)

# Not good
open_api.json(indent=2)
```

More info about field alias:

| OpenAPI version | Field alias info |
| --------------- | ---------------- |
| 3.1.0 | [here](https://github.com/kuimono/openapi-schema-pydantic/blob/master/openapi_schema_pydantic/v3/v3_1_0/README.md#alias) |
| 3.0.3 | [here](https://github.com/kuimono/openapi-schema-pydantic/blob/master/openapi_schema_pydantic/v3/v3_0_3/README.md#alias) |

### Non-pydantic schema types

Some schema types are not implemented as pydantic classes.
Please refer to the following for more info:

| OpenAPI version | Non-pydantic schema type info |
| --------------- | ----------------------------- |
| 3.1.0 | [here](https://github.com/kuimono/openapi-schema-pydantic/blob/master/openapi_schema_pydantic/v3/v3_1_0/README.md#non-pydantic-schema-types) |
| 3.0.3 | [here](https://github.com/kuimono/openapi-schema-pydantic/blob/master/openapi_schema_pydantic/v3/v3_0_3/README.md#non-pydantic-schema-types) |

### Use OpenAPI 3.0.3 instead of 3.1.0

Some UI renderings (e.g. Swagger) still do not support OpenAPI 3.1.0.
It is allowed to use the old 3.0.3 version by importing from different paths:

```python
from openapi_schema_pydantic.v3.v3_0_3 import OpenAPI, ...
from openapi_schema_pydantic.v3.v3_0_3.util import PydanticSchema, construct_open_api_with_schema_class
```

## License

[MIT License](https://github.com/kuimono/openapi-schema-pydantic/blob/master/LICENSE)
 readmeEtag: '"2769cce17ec836e639e2a438a513b737787e5d5f"' readmeLastModified: Wed, 29 Jun 2022 12:42:14 GMT repositoryId: 263836606 description: 'OpenAPI (v3) specification schema as pydantic class ' created: '2020-05-14T06:42:49Z' updated: '2026-01-20T18:10:47Z' language: Python archived: false stars: 122 watchers: 3 forks: 22 owner: kuimono logo: https://avatars.githubusercontent.com/u/16016416?v=4 license: MIT repoEtag: '"d2c5ac2dd2b8e8a2316a9e85956f773bc7a58880f2e5f6c371a67c9595f8eae3"' repoLastModified: Tue, 20 Jan 2026 18:10:47 GMT foundInMaster: true category: Parsers id: db4e005af8e10dd51fade90218a598f9 - source: openapi3 tags repository: https://github.com/digitalocean/openapi v3: true repositoryMetadata: base64Readme: >- IyBEaWdpdGFsT2NlYW4gT3BlbkFQSSBTcGVjaWZpY2F0aW9uCiFbU3BlYyBNYWluXShodHRwczovL2dpdGh1Yi5jb20vZGlnaXRhbG9jZWFuL2FwaXYyLW9wZW5hcGkvd29ya2Zsb3dzL1NwZWMlMjBNYWluL2JhZGdlLnN2ZykgIVtTdGF0dXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvU3RhdHVzLUVhcmx5JTIwQXZhaWxhYmlsaXR5LWJsdWUpCgpUaGUgT3BlbkFQSSB2MyBzcGVjaWZpY2F0aW9uIGZvciBbRGlnaXRhbE9jZWFuJ3MgcHVibGljIEFQSSB2Ml0oaHR0cHM6Ly9kb2NzLmRpZ2l0YWxvY2Vhbi5jb20vcmVmZXJlbmNlL2FwaS9kaWdpdGFsb2NlYW4vKS4KCiMjIFdoYXQgaXMgT3BlbkFQST8KCkZyb20gdGhlIFtPcGVuQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLyk6Cgo+IFRoZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gKE9BUykgZGVmaW5lcyBhIHN0YW5kYXJkLCBsYW5ndWFnZS1hZ25vc3RpYyBpbnRlcmZhY2UgdG8gUkVTVGZ1bCBBUElzIHdoaWNoIGFsbG93cyBib3RoIGh1bWFucyBhbmQgY29tcHV0ZXJzIHRvIGRpc2NvdmVyIGFuZCB1bmRlcnN0YW5kIHRoZSBjYXBhYmlsaXRpZXMgb2YgdGhlIHNlcnZpY2Ugd2l0aG91dCBhY2Nlc3MgdG8gc291cmNlIGNvZGUsIGRvY3VtZW50YXRpb24sIG9yIHRocm91Z2ggbmV0d29yayB0cmFmZmljIGluc3BlY3Rpb24uIFdoZW4gcHJvcGVybHkgZGVmaW5lZCwgYSBjb25zdW1lciBjYW4gdW5kZXJzdGFuZCBhbmQgaW50ZXJhY3Qgd2l0aCB0aGUgcmVtb3RlIHNlcnZpY2Ugd2l0aCBhIG1pbmltYWwgYW1vdW50IG9mIGltcGxlbWVudGF0aW9uIGxvZ2ljLgoKPiBBbiBPcGVuQVBJIGRlZmluaXRpb24gY2FuIHRoZW4gYmUgdXNlZCBieSBkb2N1bWVudGF0aW9uIGdlbmVyYXRpb24gdG9vbHMgdG8gZGlzcGxheSB0aGUgQVBJLCBjb2RlIGdlbmVyYXRpb24gdG9vbHMgdG8gZ2VuZXJhdGUgc2VydmVycyBhbmQgY2xpZW50cyBpbiB2YXJpb3VzIHByb2dyYW1taW5nIGxhbmd1YWdlcywgdGVzdGluZyB0b29scywgYW5kIG1hbnkgb3RoZXIgdXNlIGNhc2VzLgoKIyMgUHJvamVjdCBTdGF0dXMKClRoZSBEaWdpdGFsT2NlYW4gT3BlbkFQSSBTcGVjaWZpY2F0aW9uIGlzIGN1cnJlbnRseSBpbiAqKkVhcmx5IEF2YWlsYWJpbGl0eSoqLiBXaGlsZSB0aGUgc3BlY2lmaWNhdGlvbiBzaG91bGQgYmUgYWNjdXJhdGUsIGl0IGlzIHVuZGVyIGFjdGl2ZSBkZXZlbG9wbWVudC4gVGhlIHN0cnVjdHVyZSBvZiB0aGlzIHJlcG9zaXRvcnkgbWF5IGNvbnRpbnVlIHRvIGV2b2x2ZS4gSWYgeW91IGVuY291bnRlciBhbnkgaW5hY2N1cmFjaWVzIG9yIGhhdmUgZmVlZGJhY2sgb24gaG93IGl0IGNhbiBiZXR0ZXIgc3VpdGUgeW91ciB1c2UgY2FzZSwgcGxlYXNlIFtvcGVuIGFuIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vZGlnaXRhbG9jZWFuL2FwaXYyLW9wZW5hcGkvaXNzdWVzL25ldykgdG8gbGV0IHVzIGtub3cuCgojIyBTcGVjaWZpY2F0aW9uCgpUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgdGhlIHNvdXJjZSBmaWxlcyB1c2VkIHRvIGNvbXBpbGUgdGhlIHNwZWNpZmljYXRpb24uIE9uIGVhY2ggbWVyZ2UgdG8gYG1haW5gLCBhIGJ1bmRsZWQgdmVyc2lvbiBpcyBnZW5lcmF0ZWQgY29udGFpbmluZyB0aGUgZW50aXJlIHNwZWNpZmljYXRpb24uIEEgZG9jdW1lbnRhdGlvbiBwcmV2aWV3IHVzaW5nIFtSZWRvY10oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvcmVkb2MpIGlzIGFsc28gcHJvdmlkZWQuCgpbIVtTcGVjIERvd25sb2FkXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0Rvd25sb2FkLU9wZW5BUEklMjB2MyUyMFNwZWMtYmx1ZT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289ZGlnaXRhbG9jZWFuKV0oaHR0cHM6Ly9hcGktZW5naW5lZXJpbmcubnljMy5kaWdpdGFsb2NlYW5zcGFjZXMuY29tL3NwZWMtY2kvRGlnaXRhbE9jZWFuLXB1YmxpYy52Mi55YW1sKSBbIVtEb2NzIFByZXZpZXddKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUHJldmlldy1PcGVuQVBJJTIwRG9jdW1lbnRhdGlvbi1ibHVlP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1kaWdpdGFsb2NlYW4pXShodHRwczovL2FwaS1lbmdpbmVlcmluZy5ueWMzLmRpZ2l0YWxvY2VhbnNwYWNlcy5jb20vc3BlYy1jaS9yZWRvYy1pbmRleC5odG1sKQoKIyMjIyBQb3N0bWFuIENvbGxlY3Rpb24KCkluIG9yZGVyIHRvIGdlbmVyYXRlIGEgY29sbGVjdGlvbiB0aGF0IG1heSBiZSBpbXBvcnRlZCB0byBQb3N0bWFuLCBydW46CgpgYGBzaAptYWtlIGNvbGxlY3Rpb24KYGBgCgpUaGUgcmVzdWx0cyBjYW4gYmUgZm91bmQgaW4gYHRlc3RzL3Bvc3RtYW4uanNvbmAuCgojIyBEZXZlbG9wbWVudAoKVG8gZ2VuZXJhdGUgYSBidW5kbGVkIHZlcnNpb24gb2YgdGhlIHNwZWNpZmljYXRpb24gbG9jYWxseSwgcnVuOgoKYGBgc2gKbWFrZSBidW5kbGUKYGBgCgpUbyBwcmV2aWV3IHRoZSBkb2N1bWVudGF0aW9uIGxvY2FsbHksIHJ1bjoKCmBgYHNoCm1ha2UgcHJldmlldwpgYGAKClRoZSBkb2N1bWVudGF0aW9uIHdpbGwgYmUgYXZhaWxhYmxlIGF0OiBgaHR0cDovLzEyNy4wLjAuMTo4MDgwYC4KCkZvciBtb3JlIGRldGFpbHMgb24gb3VyIGRldmVsb3BtZW50IHByb2Nlc3MgYW5kIHRoZSBzdHJ1Y3R1cmUgb2YgdGhpcyByZXBvc2l0b3J5LCBzZWUgW0NPTlRSSUJVVElORy5tZF0oL0NPTlRSSUJVVElORy5tZCkuCgojIyBMaWNlbnNlCgpUaGlzIHNwZWNpZmljYXRpb24gaXMgbGljZW5zZWQgdW5kZXIgdGhlIFtBcGFjaGUgTGljZW5zZSAyLjBdKC9MSUNFTlNFKS4K readmeEtag: '"95ccb6f1f17c68cb42bd90853f6ed06c98171e7c"' readmeLastModified: Tue, 05 Aug 2025 13:52:58 GMT repositoryId: 264196717 description: The OpenAPI v3 specification for DigitalOcean's public API. created: '2020-05-15T13:10:36Z' updated: '2026-02-02T10:43:33Z' language: JavaScript archived: false stars: 133 watchers: 8 forks: 133 owner: digitalocean logo: https://avatars.githubusercontent.com/u/4650108?v=4 license: Apache-2.0 repoEtag: '"7f2a0b71567c7f22670fc274496cf1889e9f74353fce51b2f86f7a2ddf3cd2ea"' repoLastModified: Mon, 02 Feb 2026 10:43:33 GMT foundInMaster: true category: - Testing - Parsers id: 496d74396e4ed66d5b8c2865dbbe953f - source: openapi3 tags repository: https://github.com/hummingbird-me/api-docs v3: true repositoryMetadata: base64Readme: >- IyBBUEkgRG9jcwpbIVtBUEkgRG9jcyBEZXBsb3ltZW50XShodHRwczovL2dpdGh1Yi5jb20vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3Mvd29ya2Zsb3dzL0FQSSUyMERvY3MlMjBEZXBsb3ltZW50L2JhZGdlLnN2ZyldKGh0dHBzOi8vaHVtbWluZ2JpcmQtbWUuZ2l0aHViLmlvL2FwaS1kb2NzKQoKS2l0c3UgaXMgYSBtb2Rlcm4gYW5pbWUgZGlzY292ZXJ5IHBsYXRmb3JtIHRoYXQgaGVscHMgeW91IHRyYWNrIHRoZSBhbmltZSB5b3UncmUgd2F0Y2hpbmcsIGRpc2NvdmVyIG5ldyBhbmltZSBhbmQgc29jaWFsaXplIHdpdGggb3RoZXIgZmFucy4KCi0tLQoqKjxwIGFsaWduPSJjZW50ZXIiPlRoaXMgaXMgb3VyIFthcGkgZG9jc10gcmVwb3NpdG9yeS48YnIgLz5DaGVjayBvdXQgdGhlIFt0b29sc10sIFt3ZWJdLCBbbW9iaWxlXSBhbmQgW3NlcnZlcl0gcmVwb3NpdG9yaWVzLjwvcD4qKgoKW3Rvb2xzXTpodHRwczovL2dpdGh1Yi5jb20vaHVtbWluZ2JpcmQtbWUva2l0c3UtdG9vbHMKW3dlYl06aHR0cHM6Ly9naXRodWIuY29tL2h1bW1pbmdiaXJkLW1lL2tpdHN1LXdlYgpbc2VydmVyXTpodHRwczovL2dpdGh1Yi5jb20vaHVtbWluZ2JpcmQtbWUva2l0c3Utc2VydmVyClttb2JpbGVdOmh0dHBzOi8vZ2l0aHViLmNvbS9odW1taW5nYmlyZC1tZS9raXRzdS1tb2JpbGUKW2FwaSBkb2NzXTpodHRwczovL2tpdHN1LmRvY3MuYXBpYXJ5LmlvCgotLS0KCiMjIERvY3VtZW50YXRpb24KCkxlZ2FjeSBkb2N1bWVudGF0aW9uIGlzIGF2YWlsYWJsZSBvbiBBcGlhcnk6IGh0dHBzOi8va2l0c3UuZG9jcy5hcGlhcnkuaW8vCgpUaGUgQVBJIGRvY3VtZW50YXRpb24gaXMgY3VycmVudGx5IGJlaW5nIGNvbnZlcnRlZCBmcm9tIEFQSSBCbHVlcHJpbnQgdG8gT3BlbkFQSSAzLiBZb3UgY2FuIHZpZXcgdGhlIHByb2dyZXNzIGF0OiBodHRwczovL2h1bW1pbmdiaXJkLW1lLmdpdGh1Yi5pby9hcGktZG9jcyBhbmQgaHR0cHM6Ly9naXRodWIuY29tL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2lzc3Vlcy83MQoKIyMgQ29udHJpYnV0aW5nCgpUaGUgZG9jdW1lbnRhdGlvbiBpcyB3cml0dGVuIHdpdGggW09wZW5BUEkgM11bM10gYW5kIFtZQU1MIDEuMl1bNF0uIFRyaXZpYWwgY2hhbmdlcyBjYW4gYmUgYWRkZWQgdXNpbmcgR2l0SHViJ3MgYnVpbHQtaW4gZWRpdG9yLiBGb3IgYmlnZ2VyIGNoYW5nZXMgaXQgaXMgYWR2aXNlZCB0byBmb2xsb3cgdGhlIHN0ZXBzIGJlbG93LgoKSWYgeW91IGhhdmUgYW55IHF1ZXN0aW9ucyB5b3UgY2FuOgotIEFzayB1cyBpbiBgI2FwaWAgb24gb3VyIFtEaXNjb3JkIHNlcnZlcl1bN10KLSBQaW5nIFtAd29waWFuXSwgW0BtYXR0aGV3ZGlhc10gb3IgW0BudWNrXSBvbiBLaXRzdS4KCiMjIyBGaWxlIFN0cnVjdHVyZQoKVGhlIE9wZW5BUEkgc2NoZW1hIGlzIHNwbGl0IGludG8gbXVsdGlwbGUgZGlyZWN0b3JpZXMuIFdoaWxlIGl0IGxvb2tzIGNvbXBsaWNhdGVkLCBpdCByZWR1Y2VzIHRoZSBhbW91bnQgb2YgcmVwZWF0ZWQgcGFyYW1ldGVycyBhbmQgcmVzb3VyY2UgYXR0cmlidXRlcy9yZWxhdGlvbnNoaXBzIHRvIG5lYXIgemVyby4KClJlc291cmNlLCByZWxhdGlvbnNoaXAgYW5kIGZpbHRlciBpbmhlcml0ZW5jZSBtYXRjaGVzIHRoZSBtb2RlbCBkZWZpbml0aW9ucyBvbiB0aGUgc2VydmVyIHJlcG8uCgpgYGB5YW1sCmFwaQrilJzilIDilIDilIBwYXJhbWV0ZXJzICAjIFF1ZXJ5IFBhcmFtZXRlcnMK4pSCICAg4pSU4pSA4pSA4pSAcGF0aCAgICAjIFBhdGggUGFyYW1ldGVycwrilIIK4pSc4pSA4pSA4pSAcGF0aHMgICAgICAgIyBFbmRwb2ludHMsIGdyb3VwZWQgYnkgdGFnR3JvdXBzCuKUggrilJzilIDilIDilIByZXNvdXJjZXMgICAjIFRvcC1sZXZlbCBKU09OOkFQSSByZXNvdXJjZSBzdHJ1Y3R1cmVzIChvbmUsIG1hbnksIGluY2x1ZGVkKQrilIIgICDilJzilIDilIDilIBhbmltZQrilIIgICDilJTilIDilIDilIBlcnJvciAgICMgSlNPTjpBUEkgZXJyb3IgcmVzcG9uc2VzCuKUggrilJTilIDilIDilIBzY2hlbWFzICAgICAgICAgICAgICAgICAgICAgICAjIERhdGEgTW9kZWxzCiAgICDilJzilIDilIDilIBlbnVtcwogICAg4pSCCiAgICDilJzilIDilIDilIBmaWx0ZXJzICAgICAgICAgICAgICAgICAgICMgUmV1c2FibGUgSlNPTjpBUEkgZmlsdGVyIHBhcmFtZXRlcnMKICAgIOKUgiAgIOKUnOKUgOKUgOKUgHJlc291cmNlICAgICAgICAgICAgICAjIFJvb3QgZmlsdGVyIHBhcmFtZXRlciBmb3IgZWFjaCByZXNvdXJjZQogICAg4pSCICAg4pSU4pSA4pSA4pSAcmVzb3VyY2VGaWx0ZXJzICAgICAgICMgR3JvdXBlZCBmaWx0ZXJzIChpbmhlcml0YW5jZSkKICAgIOKUggogICAg4pSc4pSA4pSA4pSAbGlua3MgICAgICAgICAgICAgICAgICAgICAjIEpTT046QVBJIGxpbmsgcHJvcGVydGllcwogICAg4pSCCiAgICDilJzilIDilIDilIBtZXRhICAgICAgICAgICAgICAgICAgICAgICMgSlNPTjpBUEkgbWV0YSBwcm9wZXJ0aWVzCiAgICDilIIKICAgIOKUlOKUgOKUgOKUgHJlc291cmNlcyAgICAgICAgICAgICAgICAgIyBKU09OOkFQSSByZXNvdXJjZXMKICAgICAgICDilJzilIDilIDilIByZWxhdGlvbnNoaXBzICAgICAgICAgIyBKU09OOkFQSSByZWxhdGlvbnNoaXBzCiAgICAgICAg4pSU4pSA4pSA4pSAcmVzb3VyY2VSZWxhdGlvbnNoaXBzICMgR3JvdXBlZCByZWxhdGlvbnNoaXBzIChpbmhlcml0YW5jZSkKYGBgCgojIyMgQmFzZSBGaWxlcwoKVGhlc2UgYXJlIGRlZmluZWQgYXMgc2VwZXJhdGUgcmVzb3VyY2VzIGluIHRoZSBzZXJ2ZXIgZGVmaW5pdGlvbnMgYW5kIGFyZSBpbmhlcml0ZWQgYnkgYWxsIG90aGVyIGRlZmluaXRpb25zLiBDaGVjayBgYW5pbWVgIGFuZCBgbWVkaWFgIGluIGBhcGkvc2NoZW1hcy9yZXNvdXJjZXNgIGFuZCBgYXBpL3NjaGVtYXMvZmlsdGVycy9yZXNvdXJjZUZpbHRlcnNgIHRvIHNlZSBob3cgaW5oZXJpdGFuY2UgaXMgYXBwbGllZC4KCiMjIyMgUmVzb3VyY2VzCgpBbGwgcmVzb3VyY2VzIGluaGVyaXQgZnJvbSBgYXBpL3NjaGVtYXMvcmVzb3VyY2VzL2Jhc2UueW1sYAoKSW4gYWRkaXRpb246CgotIFNsdWdnYWJsZSByZXNvdXJjZXMgaW5oZXJpdCBmcm9tIGBhcGkvc2NoZW1hcy9yZXNvdXJjZXMvYmFzZV9zbHVnZ2FibGUueW1sYAotIEVwaXNvZGljIHJlc291cmNlcyBpbmhlcml0IGZyb20gYGFwaS9zY2hlbWFzL3Jlc291cmNlcy9iYXNlX2VwaXNvZGljLnltbGAKCiMjIyMgUmVzb3VyY2UgUmVsYXRpb25zaGlwcwoKLSBFcGlzb2RpYyByZXNvdXJjZXMgaW5oZXJpdCBmcm9tIGBhcGkvc2NoZW1hcy9yZXNvdXJjZXMvcmVzb3VyY2VSZWxhdGlvbnNoaXBzL2Jhc2VfZXBpc29kaWMueW1sYAoKIyMjIyBGaWx0ZXJzCgpBbGwgcmVzb3VyY2VzIGluaGVyaXQgZnJvbSBgYXBpL3NjaGVtYXMvZmlsdGVycy9yZXNvdXJjZUZpbHRlcnMvYmFzZS55bWxgCgpJbiBhZGRpaW9uOgoKLSBTbHVnZ2FibGUgcmVzb3VyY2VzIGluaGVyaXQgZnJvbSBgYXBpL3NjaGVtYXMvZmlsdGVycy9yZXNvdXJjZUZpbHRlcnMvYmFzZV9zbHVnZ2FibGUueW1sYAotIEVwaXNvZGljIHJlc291cmNlcyBpbmhlcml0IGZyb20gYGFwaS9zY2hlbWFzL2ZpbHRlcnMvcmVzb3VyY2VGaWx0ZXJzL2Jhc2VfZXBpc29kaWMueW1sYAoKIyMjIFJlcXVpcmVtZW50cwoKLSBbZ2l0XShodHRwczovL2dpdC1zY20uY29tKSA+IGAyLjAuMGAKLSBbbm9kZS5qc10oaHR0cHM6Ly9ub2RlanMub3JnKSA+IGAxMi4wLjBgCi0gW3lhcm5dKGh0dHBzOi8veWFybnBrZy5jb20pID4gYDEuMjIuMGAKCiMjIyBEZXZlbG9wbWVudAoKMS4gW0ZvcmsgdGhlIHJlcG9dWzVdCgoyLiBDbG9uZSB0aGUgcmVwbyAoYGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20ve3lvdXJ1c2VybmFtZX0vYXBpLWRvY3MuZ2l0YCkKCjMuIENyZWF0ZSBhIG5ldyBicmFuY2ggKGBnaXQgY2hlY2tvdXQgLWIgaW1wcm92ZS1kb2NzYCkKCjQuIEluc3RhbGwgZGVwZW5kZW5jaWVzIChgeWFybmApCgo1LiBNYWtlIHRoZSBhcHByb3ByaWF0ZSBjaGFuZ2VzIGluIHRoZSBzb3VyY2UgZmlsZXMKICAtIENoZWNrIHRoZSBjaGFuZ2VzIGxvY2FsbHkgd2l0aCBgeWFybiBzdGFydGAKCjYuIENvbW1pdCB5b3VyIGNoYW5nZXMgKGBnaXQgY29tbWl0IC1hbSAnZmVhdDogaW1wcm92ZSBkb2NzJ2ApCgo3LiBQdXNoIHRvIHlvdXIgYnJhbmNoIChgZ2l0IHB1c2ggb3JpZ2luIGltcHJvdmUtZG9jc2ApCgo4LiBbQ3JlYXRlIGEgUHVsbCBSZXF1ZXN0XVs2XQoKIyMgQ29udHJpYnV0b3JzCgpbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy8wKV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvMClbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy8xKV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvMSlbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy8yKV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvMilbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy8zKV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvMylbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy80KV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvNClbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy81KV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvNSlbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy82KV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvNilbIVtdKGh0dHBzOi8vc291cmNlcmVyLmlvL2ZhbWUvd29waWFuL2h1bW1pbmdiaXJkLW1lL2FwaS1kb2NzL2ltYWdlcy83KV0oaHR0cHM6Ly9zb3VyY2VyZXIuaW8vZmFtZS93b3BpYW4vaHVtbWluZ2JpcmQtbWUvYXBpLWRvY3MvbGlua3MvNykKClswXTpodHRwczovL2dpdGh1Yi5jb20vaHVtbWluZ2JpcmQtbWUvaHVtbWluZ2JpcmQtc2VydmVyClsxXTpodHRwczovL2dpdGh1Yi5jb20vaHVtbWluZ2JpcmQtbWUvaHVtbWluZ2JpcmQtY2xpZW50ClszXTpodHRwOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMwpbNF06aHR0cHM6Ly95YW1sLm9yZy9zcGVjLzEuMi9zcGVjLmh0bWwKWzVdOmh0dHBzOi8vaGVscC5naXRodWIuY29tL2FydGljbGVzL2ZvcmstYS1yZXBvLyNmb3JrLWFuLWV4YW1wbGUtcmVwb3NpdG9yeQpbNl06aHR0cHM6Ly9oZWxwLmdpdGh1Yi5jb20vYXJ0aWNsZXMvY3JlYXRpbmctYS1wdWxsLXJlcXVlc3QvI2NyZWF0aW5nLXRoZS1wdWxsLXJlcXVlc3QKWzddOmh0dHBzOi8vaW52aXRlLmdnL2tpdHN1CgpbQHdvcGlhbl06aHR0cHM6Ly9raXRzdS5pby91c2Vycy93b3BpYW4KW0BtYXR0aGV3ZGlhc106aHR0cHM6Ly9raXRzdS5pby91c2Vycy9tYXR0aGV3ZGlhcwpbQG51Y2tdOmh0dHBzOi8va2l0c3UuaW8vdXNlcnMvbnVjawo= readmeEtag: '"ddb20eabf757e8b276878335d2c2c6b9c4190c81"' readmeLastModified: Sat, 28 Oct 2023 01:18:47 GMT repositoryId: 92471764 description: 🤖 API documentation for Kitsu created: '2017-05-26T04:21:01Z' updated: '2026-01-28T18:20:54Z' language: YAML archived: false stars: 118 watchers: 10 forks: 207 owner: hummingbird-me logo: https://avatars.githubusercontent.com/u/7648832?v=4 repoEtag: '"5ae1fe2186fe888045b8f792ac90afdaacabe66239a2af6a2533b6477bac0a22"' repoLastModified: Wed, 28 Jan 2026 18:20:54 GMT foundInMaster: true category: Low-level Tooling id: 76ad1690754ba6c1e408622d1ed39e6c - source: openapi3 tags repository: https://github.com/oapigen/actix-swagger v3: true repositoryMetadata: base64Readme: >- IyBhY3RpeCBzd2FnZ2VyCgojIyBVc2FnZQoKPiBOb3QgZm9yIHByb2R1Y3Rpb24gdXNlIHlldAoKYGBgYmFzaAojIEFkZCBjYXJnbyBjb21tYW5kIHRvIHNpbXBsaWZ5IHVzYWdlCmNhcmdvIGluc3RhbGwgY2FyZ28tc3dhZ2cKCiMgQWRkIHN1cHBvcnQgbGlicmFyeSB0byB5b3VyIHByb2plY3QgKHZpYSBjYXJnby1lZGl0IG9yIG1hbnVhbCkKY2FyZ28gYWRkIGFjdGl4LXN3YWdnZXIKCiMgR2VuZXJhdGUgeW91ciBjb2RlIHdpdGggY2FyZ28gc3ViY29tbWFuZApjYXJnbyBzd2FnZyAuL29wZW5hcGkueWFtbCAtLW91dC1maWxlIC4vc3JjL2FwaS5ycwoKIyBGb3JtYXQgZmlsZSBhZnRlcgpydXN0Zm10IC4vc3JjL2FwaS5ycwpgYGAKCiMjIERldmVsb3BtZW50CgpJdCB1c2VzIFtpbnN0YV0oaHR0cHM6Ly9naXRodWIuY29tL21pdHN1aGlrby9pbnN0YSkgZm9yIHNuYXBzaG90IHRlc3RpbmcuCgpJbnN0YWxsIGBjYXJnby1pbnN0YWAgdG8gYmV0dGVyIHJldmlldyBleHBlcmllbmNlLgoKIyMjIE1lbWJlcnMKCi0gYGNhcmdvLWFjdGl4YCDigJQgc3VwcG9ydCBsaWJyYXJ5LCBjb250ZW50cyB0eXBlZCByZXNwb25zZSBuYW1lZCBgQW5zd2VyYCBhbmQgY3VzdG9tIGBNZXRob2RgIGFuZCBgQ29udGVudFR5cGVgIHRoYXQgc3VwcG9ydHMgaW4gc3dhZ2cKLSBgc3dhZ2dgIOKAlCBsaWJyYXJ5IHRoYXQgdHJhbnNmb3JtcyBvcGVuYXBpMyAoeWFtbHxqc29uKSBzcGVjIHRvIHJ1c3QgY29kZQotIGBjYXJnby1zd2FnZ2Ag4oCUIHNhbWUgYXMgYHN3YWdnYCBidXQgZm9yIGNsaQotIGBkZW1vYCDigJQgY2hlY2tzIHRoYXQgZ2VuZXJhdGVkIGNvZGUgaXMgY29tcGlsZXMKCiMjIyBEZW1vCgpgYGBiYXNoCiMgdG8gY29udmVydCAuL2RlbW8vb3BlbmFwaS55YW1sIHRvIC4vZGVtby9zcmMvbGliLnJzCiMgZm9ybWF0IC4vZGVtby9zcmMvbGliLnJzCiMgYW5kIGNoZWNrIGp1c3QgcnVuCmNhcmdvIGluc3RhbGwgY2FyZ28tbWFrZQpjYXJnbyBtYWtlCmBgYAo= readmeEtag: '"822b2abdf32e5903eb852ed8facd9a106ed97d47"' readmeLastModified: Tue, 15 Nov 2022 12:51:21 GMT repositoryId: 235156358 description: Swagger code generator for actix-web framework created: '2020-01-20T17:21:33Z' updated: '2025-06-03T09:55:56Z' language: Rust archived: true stars: 112 watchers: 7 forks: 8 owner: oapigen logo: https://avatars.githubusercontent.com/u/44164641?v=4 license: MIT repoEtag: '"6eb6253b6dcbd0f9d62a48b8c5c2dce97991ee63c99378d1708ebedd462c6902"' repoLastModified: Tue, 03 Jun 2025 09:55:56 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 2ef5d9b209ad59a8276d793ce09da95e oldLocations: - https://github.com/openapigen/actix-swagger - source: openapi3 tags repository: https://github.com/nasruddin/spring-boot-3-jwt-auth v3: true repositoryMetadata: base64Readme: >- IyBzcHJpbmctYm9vdC0zLWp3dC1hdXRoCjprZXk6IFNhbXBsZSBTcHJpbmcgYm9vdCAzIGFwcGxpY2F0aW9uIGZvciBBdXRoZW50aWNhdGlvbiBhbmQgQXV0aG9yaXphdGlvbgoKIyMgRmVhdHVyZXMKKiBDdXN0b21pemFibGUgaGVhZGVyKFgtQXV0aC1Ub2tlbikgdG8gcGFzcyBBdXRoIHRva2VuLgoqIEpXVCBmb3IgdG9rZW4gY3JlYXRpb24gYW5kIHZhbGlkYXRpb24uCiogUm9sZSBiYXNlZCBhdXRob3JpemF0aW9uLgoqIERldmljZSBiYXNlZCBhdXRoLgoqIEN1c3RvbSBWYWxpZGF0b3JzCiogU3ByaW5nIGRvYy4KCgojIyBSdW5uaW5nIHRoZSBzYW1wbGUgYXBwCmBgYAptdm4gc3ByaW5nLWJvb3Q6cnVuCmBgYAoKIyMgUmVnaXN0ZXJpbmcgYSBVc2VyCmBgYApjdXJsIC1YIFBPU1QgImh0dHA6Ly9sb2NhbGhvc3Q6OTAwMC9hcGkvYXV0aC9yZWdpc3RlciIgLUggICJhY2NlcHQ6ICovKiIgLUggICJDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24iIC1kICJ7XCJ1c2VybmFtZVwiOlwibmFzcnVkZGluXCIsXCJwYXNzd29yZFwiOlwicEBzc3cwMGRcIixcImRldmljZVwiOlwid2ViXCIsXCJlbWFpbFwiOlwibmFzcnVkZGluQGdtYWlsLmNvbVwifSIKYGBgCgpgYGAKewogICAgImlkIjoyLAogICAgInVzZXJuYW1lIjoibmFzcnVkZGluIiwKICAgICJwYXNzd29yZCI6IiQyYSQxMCRVM0NSNFQxR293ZDUwUS4weUsvVXVPaC5YV1Z4MEJZSWU3QmlBbXltWFouTVlQVXRVNUYuZSIsCiAgICAiZW1haWwiOiJuYXNydWRkaW5AZ21haWwuY29tIiwKICAgICJsYXN0UGFzc3dvcmRSZXNldCI6IjIwMjMtMDktMTRUMDg6NDE6MTAuMDgwKzAwOjAwIiwKICAgICJhdXRob3JpdGllcyI6IkFETUlOIgp9CmBgYApIMi1jb25zb2xlIGNhbiBiZSBhY2Nlc3NlZCBhdCA8aHR0cDovL2xvY2FsaG9zdDo5MDAwL2FwaS9oMi1jb25zb2xlPgohW0pXVCBEZWNvZGVkXShodHRwczovL2dpdGh1Yi5jb20vTmFzcnVkZGluL3NwcmluZy1ib290LWp3dC1hdXRoL2Jsb2IvcG9tLXVwZGF0ZS9pbWFnZXMvaDItY29uc29sZS5wbmc/cmF3PXRydWUpCgojIyBMb2dpbiBhIFVzZXIgLyBGZXRjaCBUb2tlbgpgYGAKY3VybCAtWCBQT1NUICJodHRwOi8vbG9jYWxob3N0OjkwMDAvYXBpL2F1dGgiIC1IICAiYWNjZXB0OiAqLyoiIC1IICAiQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uIiAtZCAie1widXNlcm5hbWVcIjpcIm5hc3J1ZGRpblwiLFwicGFzc3dvcmRcIjpcInBAc3N3MDBkXCIsXCJkZXZpY2VcIjpcIndlYlwifSIKYGBgCmBgYAp7InRva2VuIjoiZXlKaGJHY2lPaUpJVXpVeE1pSjkuZXlKemRXSWlPaUp1WVhOeWRXUmthVzRpTENKaGRXUnBaVzVqWlNJNkluZGxZaUlzSW1OeVpXRjBaV1FpT2pFMk9UUTJPREUyT0RFM01EVXNJbVY0Y0NJNk1UWTVOVEk0TmpRNE1YMC5NeWR3SVd6TjNTZ0N2QjhjWW96S2NSMnRITUNNNW5ySVBYVUJ0eDRvODJvdDF0YUxfTlFNNVRSSFo0eU9jOXVVY1pGcnoxWFFBTF9mRE5YQUl3bVp4dyJ9CmBgYAoKIVtKV1QgRGVjb2RlZF0oaHR0cHM6Ly9naXRodWIuY29tL05hc3J1ZGRpbi9zcHJpbmctYm9vdC0zLWp3dC1hdXRoL2Jsb2IvbWFzdGVyL2ltYWdlcy9kZWNvZGVkLWp3dC5wbmc/cmF3PXRydWUpCgojIyBBY2Nlc3NpbmcgVXNlci9Qcm90ZWN0ZWQgQVBJCgpXaXRob3V0IHNldHRpbmcgWC1BVVRILVRPS0VOCmBgYApjdXJsIC1YIEdFVCAiaHR0cDovL2xvY2FsaG9zdDo5MDAwL2FwaS91c2VyL25hc3J1ZGRpbiIgLUggICJhY2NlcHQ6ICovKiIKYGBgCmBgYAp7CiAgICAidGltZXN0YW1wIjoiMjAyMy0wOS0xNFQwODo1NzowOC40MDMrMDA6MDAiLAogICAgInN0YXR1cyI6NDAxLAogICAgImVycm9yIjoiVW5hdXRob3JpemVkIiwKICAgICJwYXRoIjoiL2FwaS91c2VyL25hc3J1ZGRpbiIKfSAKYGBgCgpXaXRoIHNldHRpbmcgWC1BVVRILVRPS0VOCmBgYApjdXJsIC1YIEdFVCAiaHR0cDovL2xvY2FsaG9zdDo5MDAwL2FwaS91c2Vycy9uYXNydWRkaW4iIC1IICAiYWNjZXB0OiAqLyoiIC1IICAiWC1BdXRoLVRva2VuOiBleUpoYkdjaU9pSklVelV4TWlKOS5leUp6ZFdJaU9pSnVZWE55ZFdSa2FXNGlMQ0poZFdScFpXNWpaU0k2SW5kbFlpSXNJbU55WldGMFpXUWlPakUyT1RRMk9ERTJPREUzTURVc0ltVjRjQ0k2TVRZNU5USTROalE0TVgwLk15ZHdJV3pOM1NnQ3ZCOGNZb3pLY1IydEhNQ001bnJJUFhVQnR4NG84Mm90MXRhTF9OUU01VFJIWjR5T2M5dVVjWkZyejFYUUFMX2ZETlhBSXdtWnh3IgpgYGAKYGBgCnsKICAgICJpZCI6MSwKICAgICJ1c2VybmFtZSI6Im5hc3J1ZGRpbiIsCiAgICAicGFzc3dvcmQiOiIkMmEkMTAkZHE2dUZsZWh0ZXRzZkk2Z2xMa0EuT2Flb0lFdTVQUHFJVk5aSERNQ0NpRWVqOGIvMHZoV2EiLCJlbWFpbCI6Im5hc3J1ZGRpbkBnbWFpbC5jb20iLAogICAgImxhc3RQYXNzd29yZFJlc2V0IjoiMjAyMy0wOS0xNFQwODo0MjozNy43NTgrMDA6MDAiLAogICAgImF1dGhvcml0aWVzIjoiQURNSU4iCn0gCmBgYAoKIyMgQWRtaW4gQVBJCmBgYApjdXJsIC1YIEdFVCAiaHR0cDovL2xvY2FsaG9zdDo5MDAwL2FwaS9hZG1pbiIgLUggICJhY2NlcHQ6ICovKiIgLUggICJYLUF1dGgtVG9rZW46IGV5SmhiR2NpT2lKSVV6VXhNaUo5LmV5SnpkV0lpT2lKdVlYTnlkV1JrYVc0aUxDSmhkV1JwWlc1alpTSTZJbmRsWWlJc0ltTnlaV0YwWldRaU9qRTJPVFEyT0RFMk9ERTNNRFVzSW1WNGNDSTZNVFk1TlRJNE5qUTRNWDAuTXlkd0lXek4zU2dDdkI4Y1lvektjUjJ0SE1DTTVucklQWFVCdHg0bzgyb3QxdGFMX05RTTVUUkhaNHlPYzl1VWNaRnJ6MVhRQUxfZkROWEFJd21aeHciCmBgYApgYGAKOk8KYGBgCgojIyBPcGVuQVBJIFN3YWdnZXIKMS4gU3dhZ2dlciBjYW4gYmUgYWNjZXNzZWQgYXQgPGh0dHA6Ly9sb2NhbGhvc3Q6OTAwMC9hcGkvc3dhZ2dlci11aS9pbmRleC5odG1sPgoKIVtTd2FnZ2VyXShodHRwczovL2dpdGh1Yi5jb20vTmFzcnVkZGluL3NwcmluZy1ib290LWp3dC1hdXRoL2Jsb2IvcG9tLXVwZGF0ZS9pbWFnZXMvc3dhZ2dlci5wbmc/cmF3PXRydWUpCgoyLiBBUEkgRG9jcyBjYW4gYmUgYWNjZXNzZWQgYXQgPGh0dHA6Ly9sb2NhbGhvc3Q6OTAwMC9hcGkvYXBpLWRvY3M+CiFbQVBJIERvY3NdKGh0dHBzOi8vZ2l0aHViLmNvbS9OYXNydWRkaW4vc3ByaW5nLWJvb3Qtand0LWF1dGgvYmxvYi9wb20tdXBkYXRlL2ltYWdlcy9vcGVuLWFwaS5wbmc/cmF3PXRydWUpCg== readmeEtag: '"8f2664385224ec7eb4d159ef52779751a747eb3e"' readmeLastModified: Thu, 28 Sep 2023 19:47:26 GMT repositoryId: 51244983 description: >- :key: Sample Spring boot application secured using JWT auth in custom header(X-Auth-Token). created: '2016-02-07T11:17:36Z' updated: '2025-12-17T04:18:19Z' language: Java archived: false stars: 107 watchers: 5 forks: 35 owner: Nasruddin logo: https://avatars.githubusercontent.com/u/2290767?v=4 license: MIT repoEtag: '"ea21748c4e48ce8d503cb7fe1e64eceb8d14e6c57631ebe4211b9b933344e56c"' repoLastModified: Wed, 17 Dec 2025 04:18:19 GMT foundInMaster: true category: Server Implementations id: bb56c80ba4c7463e50ddc2959f3113f1 oldLocations: - https://github.com/nasruddin/spring-boot-jwt-auth - source: openapi3 tags repository: https://github.com/contiamo/openapi-generator-go v3: true repositoryMetadata: base64Readme: >- b3BlbmFwaS1nZW5lcmF0b3ItZ28KPT09PT09PT09PT09PT09PT09PT0KCiFbdGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL2NvbnRpYW1vL29wZW5hcGktZ2VuZXJhdG9yLWdvL3dvcmtmbG93cy90ZXN0L2JhZGdlLnN2ZykKCkdlbmVyYXRlIGdvIGNvZGUgZnJvbSBhbiBvcGVuYXBpIHNwZWMhCgpgYGBiYXNoCiMgZ2VuZXJhdGUgc3RydWN0cywgZW51bXMgYW5kIGEgcm91dGVyIGZyb20gYSBzcGVjIGZpbGUKb3BlbmFwaS1nZW5lcmF0b3ItZ28gZ2VuZXJhdGUgLS1zcGVjIC4vZXhhbXBsZS9hcGkueWFtbCAtLW91dHB1dCAuL2V4YW1wbGUvZ2VuZXJhdGVkCiMgZ2VuZXJhdGUgbW9kZWxzIGFuZCByb3V0ZXIgaW5kZXBlbmRlbnRseQpvcGVuYXBpLWdlbmVyYXRvci1nbyBnZW5lcmF0ZSBtb2RlbHMgLS1zcGVjIC4vZXhhbXBsZS9hcGkueWFtbCAtLW91dHB1dCAuL2V4YW1wbGUvbW9kZWxzIC0tcGFja2FnZS1uYW1lIG1vZGVscwpvcGVuYXBpLWdlbmVyYXRvci1nbyBnZW5lcmF0ZSByb3V0ZXIgLS1zcGVjIC4vZXhhbXBsZS9hcGkueWFtbCAtLW91dHB1dCAuL2V4YW1wbGUvcm91dGVyIC0tcGFja2FnZS1uYW1lIHJvdXRlcgpgYGAKCiMjIFdoeQpUaGUgcGxhdGZvcm0gdGVhbSBhdCBDb250aWFtbyBoYXMgYSBzY2hlbWEgZmlyc3QgZGV2ZWxvcG1lbnQgZmxvdy4gVGhpcyBtZWFucyB3ZSB3cml0ZSBhbmQgcmV2aWV3IHRoZSBPcGVuQVBJIHNwZWMgYmVmb3JlIHdlIHdyaXRlIHRoZSBpbXBsZW1lbnRhdGlvbi4KClRvIG1ha2UgdGhpcyBlYXNpZXIgd2Ugc3RhcnRlZCB1c2luZyAob3IgdHJpZWQgdG8gdXNlKSB2YXJpb3VzIE9wZW5BUEkgY29kZSBnZW5lcmF0b3JzIHRvIHJlZHVjZSBidXN5IHdvcmsgYW5kIGVuc3VyZSBwcmVjaXNlIGNvbnNpc3RlbnQgaW1wbGVtZW50YXRpb25zLiBVbmZvcnR1bmF0ZWx5LCB3ZSBuZXZlciByZWFsbHkgZW5qb3llZCB0aGUgb3V0cHV0IGZyb20gdGhlc2UgZ2VuZXJhdG9ycyBhbmQgaGF2ZSBjcmVhdGVkIG91ciBvd24gZ2VuZXJhdG9yIHRoYXQgbWF0Y2hlcyBvdXIgcHJlZmVycmVkIHN0eWxlIGFuZCBjb252ZW50aW9ucy4KCiMjIFdoYXQKVGhlIGdlbmVyYXRvciBjb25zaXN0cyBvZiB0d28gcGFydHMKCjEuIGdlbmVyYXRvciBmb3IgdGhlIEhUVFAgc2VydmVyIG11eCB1c2luZyBbZ28tY2hpXShodHRwczovL2dpdGh1Yi5jb20vZ28tY2hpL2NoaSksIGFuZAoyLiBhIG1vZGVsIGdlbmVyYXRvciBmb3IgdGhlIHJlcXVlc3QgYW5kIHJlc3BvbnNlIG9iamVjdHMuCgojIyMgVGhlIHJvdXRlciBnZW5lcmF0b3IKV2UgY2hvc2UgYGdvLWNoaWAgYmVjYXVzZSBpdCBpcyBmYXN0LCBzdXBwb3J0cyB0aGUgc3RhbmRhcmQgbGlicmFyeSBIYW5kbGVyIGludGVyZmFjZSwgYW5kIGlzIGVhc3kgdG8gYWRkIG1pZGRsZXdhcmVzIHVzaW5nIGByLlVzZWAuCgpUaGUgZ2VuZXJhdG9yIHdpbGwgY3JlYXRlIGEgbWV0aG9kIGBOZXdSb3V0ZXJgIGFuZCBoYW5kbGVyIGludGVyZmFjZXMgdGhhdCB5b3UgbXVzdCB0aGVuIGltcGxlbWVudCBhbmQgcGFzcyB0byBgTmV3Um91dGVyYC4gSXQgaXMgaW1wb3J0YW50IHRvIG5vdGUgdGhhdCBpdCBkb2VzIG5vdCBnZW5lcmF0ZSBhbnkgaGFuZGxlciBsb2dpYywganVzdCB0aGUgcmVxdWlyZWQgaW50ZXJmYWNlcy4gIEhhbmRsZXJzIGNhbiBiZSBncm91cGVkIGludG8gc3BlY2lmaWMgaW50ZXJmYWNlcyBieSBhZGRpbmcgYHgtaGFuZGxlci1ncm91cDogPE5hbWU+YCB0byB5b3VyIHNjaGVtYS4gVGhpcyBpcyBvbmUgb2YgdGhlIG9ubHkgc2NoZW1hIGV4dGVuc2lvbnMgd2UgaGF2ZSBhZGRlZCB0byBpbXByb3ZlIG91ciBjb2RlIGdlbmVyYXRpb24uIFRoZSBoYW5kbGVyIGludGVyZmFjZSB3aWxsIGJlIGNhbGxlZCBgTmFtZUhhbmRsZXJgIGFuZCB3aWxsIGhhdmUgYSBtZXRob2QgbWF0Y2hpbmcgdGhlIGBvcGVyYXRpb25JZGAgZnJvbSB0aGUgc3BlYy4KCkZvciBleGFtcGxlLCBlYWNoIG9mIG91ciBwcm9qZWN0cyB3aWxsIGltcGxlbWVudCBhIHByb2plY3Qgc3BlY2lmaWMgYE5ld1JvdXRlcmAgdGhhdCBkb2VzIHRoZSBoYW5kbGVyIGNvbmZpZ3VyYXRpb24gYW5kICBpbml0aWFsaXphdGlvbiBvZiB0aGUgZ2VuZXJhdGVkIHJvdXRlci4gSW4gdGhlIG5leHQgZXhhbXBsZSwgYG9wZW5hcGkuTmV3Um91dGVyYCBpcyBnZW5lcmF0ZWQgdXNpbmcKCmBgYHNoCm9wZW5hcGktZ2VuZXJhdG9yLWdvIGdlbmVyYXRlIHJvdXRlciAtLXNwZWMgLi9leGFtcGxlL2FwaS55YW1sIC0tb3V0cHV0IC4vZXhhbXBsZS9yb3V0ZXIgLS1wYWNrYWdlLW5hbWUgb3BlbmFwaQpgYGAKVGhlIHByb2plY3Qgc3BlY2lmaWMgcm91dGVyIGluaXRpYWxpemF0aW9uOgpgYGBnbwpmdW5jIE5ld1JvdXRlcihjdHggY29udGV4dC5Db250ZXh0LCBkYiAqc3FsLkRCLCBjZmcgY29uZmlnLkNvbmZpZykgKGh0dHAuSGFuZGxlciwgZXJyb3IpIHsKICAgIC8vIGRvIHNldHVwIHN0dWZmCgogICAgLy8gYW5kIHByb2JhYmx5IHNvbWUgbW9yZSBzZXR1cCBmb3IgZWFjaCBvZiB0aGUgaGFuZGxlciBpbXBsZW1lbnRhdGlvbnMKICAgIHVzZXJIYW5kbGVyIDo9IGhhbmRsZXJzLk5ld1VzZXJIYW5kbGVyKGN0eCwgZGIpCiAgICByZXNvdXJjZXNIYW5kbGVyIDo9IGhhbmRsZXJzLk5ld1Jlc291cmNlKGN0eCwgZGIpCgogICAgLy8gb3BlbmFwaS1nZW5lcmF0b3ItZ28gZ2VuZXJhdGUgcm91dGVyIC0tc3BlYyAuL2V4YW1wbGUvYXBpLnlhbWwgLS1vdXRwdXQgLi9leGFtcGxlL3JvdXRlciAtLXBhY2thZ2UtbmFtZSBvcGVuYXBpCiAgICBhcGlSb3V0ZXIgOj0gb3BlbmFwaS5OZXdSb3V0ZXIoCiAgICAgICAgLy8gdGhlIFVzZXJzSGFuZGxlciBpbnRlcmZhY2UgY29udGFpbnMgYSBtZXRob2QgZm9yIGVuZHBvaW50IHdpdGgKICAgICAgICAvLyB4LWhhbmRsZXItZ3JvdXA6IFVzZXJzCiAgICAgICAgdXNlckhhbmRsZXIsCgogICAgICAgIC8vIFJlc291cmNlc0hhbmRsZXIgaW50ZXJhY2UgY29udGFpbnMgYSBtZXRob2QgZm9yIGVhY2ggZW5kcG9pbnQgd2l0aAogICAgICAgIC8vIHgtaGFuZGxlci1ncm91cDogUmVzb3VyY2VzCiAgICAgICAgcmVzb3VyY2VzSGFuZGxlciwKICAgICkKCiAgICByb290LlJvdXRlKCIvIiwgZnVuYyhhcGkgY2hpLlJvdXRlcikgewogICAgICAgIC8vIG5vdyAgaW5pdCBhbmQgZGQgeW91ciBtaWRkbGV3YXJlcwoJCXIgOj0gbWlkZGxld2FyZXMuV2l0aFJlY292ZXJ5KG9zLlN0ZGVyciwgY2ZnLkRlYnVnKQoJCXQgOj0gbWlkZGxld2FyZXMuV2l0aFRyYWNpbmcoY29uZmlnLkFwcGxpY2F0aW9uTmFtZSwgbmlsLCBtaWRkbGV3YXJlcy5DaGlSb3V0ZU5hbWUpCgkJbCA6PSBtaWRkbGV3YXJlcy5XaXRoTG9nZ2luZyhjb25maWcuQXBwbGljYXRpb25OYW1lKQoJCW0gOj0gbWlkZGxld2FyZXMuV2l0aE1ldHJpY3MoY29uZmlnLkFwcGxpY2F0aW9uTmFtZSwgbmlsKQoJCWEgOj0gYXV0aG9yaXphdGlvbi5OZXdNaWRkbGV3YXJlKGNmZy5BdXRob3JpemF0aW9uLkhlYWRlck5hbWUsIHB1YmxpY0tleSkKCgkJYXBpLlVzZShyLldyYXBIYW5kbGVyKQoJCWFwaS5Vc2UodC5XcmFwSGFuZGxlcikKCQlhcGkuVXNlKGwuV3JhcEhhbmRsZXIpCgkJYXBpLlVzZShtLldyYXBIYW5kbGVyKQoJCWFwaS5Vc2UoYS5XcmFwSGFuZGxlcikKCiAgICAgICAgLy8gbW91bnQgdGhlIGdlbmVyYXRlIHJvdXRlcgoJCWFwaS5Nb3VudCgiLyIsIGFwaVJvdXRlcikKCX0pCgoJcmV0dXJuIHJvb3QsIG5pbAoKfQpgYGAKCiMjIyBUaGUgbW9kZWwgZ2VuZXJhdG9yClRoZSBtb2RlbCBnZW5lcmF0b3IgaXMgYSB3b3JrIGluIHByb2dyZXNzLCBidXQgY292ZXJzIHRoZSBtb3N0IGNvbW1vbiBjYXNlcyB3ZSBuZWVkIGluIGEgUkVTVCBBUEkuCgpZb3UgY2FuIGZpbmQgdmFyaW91cyBleGFtcGxlcyBvZiB0aGUgd2hhdCBpcyBzdXBwb3J0ZWQgYW5kIHRoZSBjb3JyZXNwb25kaW5nIG91dHB1dCBpbiBvdXIgW3Rlc3QgZml4dHVyZXNdKC4vcGtnL2dlbmVyYXRvcnMvbW9kZWxzL3Rlc3RkYXRhL2Nhc2VzKS4KCk91ciBnZW5lcmF0b3IgZGlmZmVycyBmcm9tIHRoZSBvZmZpY2lhbCBPcGVuQVBJIGdlbmVyYXRvciB0b29scyBieSBhbHNvIHByb3ZpZGluZyBnZXR0ZXJzIGZvciBtYW55IGZpZWxkcywgd2hpY2ggbWFrZXMgaXQgZWFzaWVyIHRvIGRlZmluZSBhbmQgd29yayB3aXRoIGludGVyZmFjZXMgb2YgdGhlIG1vZGVscy4KCldlIGFsc28gcHJvdmlkZSBiZXR0ZXIgc3VwcG9ydCBmb3IgOgoqIFtlbnVtc10oLi9wa2cvZ2VuZXJhdG9ycy9tb2RlbHMvdGVzdGRhdGEvY2FzZXMvZW51bXMvZXhwZWN0ZWQvbW9kZWxfZmlsdGVyX3R5cGUuZ28pLCBlbnVtIHN1cHBvcnQgYWxzbyBpbmNsdWRlcyBzb21lIHV0aWxpdHkgbWV0aG9kcyB0aGF0IG1ha2UgdmFsaWRhdGluZyB0aGUgZW51bXMgbXVjaCBlYXNpZXIsCiAgKiBOb3cgaW5jbHVkaW5nIHN1cHBvcnQgZm9yIHN0cmluZywgaW50ZWdlciwgYW5kIG51bWJlciAoaS5lLiBkZWNpbWFsKSBlbnVtcy4KKiBbYXJyYXlzXSguL3BrZy9nZW5lcmF0b3JzL21vZGVscy90ZXN0ZGF0YS9jYXNlcy90eXBlZF9hcnJheXMvZXhwZWN0ZWQvbW9kZWxfZm9vLmdvKSwKKiBbYWxsT2ZdKC4vcGtnL2dlbmVyYXRvcnMvbW9kZWxzL3Rlc3RkYXRhL2Nhc2VzL2FsbG9mMS9leHBlY3RlZC9tb2RlbF9mb28uZ28pLAoqIGFuZCBbb25lT2ZdKC4vcGtnL2dlbmVyYXRvcnMvbW9kZWxzL3Rlc3RkYXRhL2Nhc2VzL29uZW9mL2V4cGVjdGVkL21vZGVsX2Zvby5nbykKdGhhdCBmZWVsIG1vcmUgbmF0dXJhbCBpbiBHby4gQ3JlYXRpbmcgYXMgbWFueSBzdHJvbmcgdHlwZXMgYXMgaXMgcG9zc2libGUgYW5kIHVzaW5nIGBpbnRlcmZhY2V7fWAgb3RoZXJ3aXNlLiBUaGVzZSBjYXNlcyBvZnRlbiBmYWlsZWQgb3IgZ2VuZXJhdGVkIG5vbi1jb21waWxhYmxlIGNvZGUgd2l0aCB0aGUgb2ZmaWNpYWwgZ2VuZXJhdG9yLgo= readmeEtag: '"3882a5622cc5f1a56c9fee78df8398ac95f0683f"' readmeLastModified: Mon, 17 Jun 2024 11:52:29 GMT repositoryId: 312311416 description: >- An opinionated OpenAPI v3 code generator for Go. Use this to generate API models and router scaffolding. created: '2020-11-12T15:04:52Z' updated: '2026-01-09T23:48:51Z' language: Go archived: false stars: 112 watchers: 7 forks: 12 owner: contiamo logo: https://avatars.githubusercontent.com/u/862286?v=4 license: MIT repoEtag: '"d1b381f48ce963735383a73cd9242300a5e8c1e64b51d9b4972736fbdb3e11c9"' repoLastModified: Fri, 09 Jan 2026 23:48:51 GMT foundInMaster: true category: Server Implementations id: eef90532aca60fba2399f8da8d25e5f2 - source: openapi3 tags repository: https://github.com/playpauseandstop/rororo v3: true repositoryMetadata: base64Readme: >- PT09PT09CnJvcm9ybwo9PT09PT0KCi4uIGltYWdlOjogaHR0cHM6Ly9naXRodWIuY29tL3BsYXlwYXVzZWFuZHN0b3Avcm9yb3JvL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmcKICAgIDp0YXJnZXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9wbGF5cGF1c2VhbmRzdG9wL3Jvcm9yby9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWwKICAgIDphbHQ6IENJIFdvcmtmbG93CgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcHJlLS1jb21taXQtZW5hYmxlZC1icmlnaHRncmVlbj9sb2dvPXByZS1jb21taXQmbG9nb0NvbG9yPXdoaXRlCiAgICA6dGFyZ2V0OiBodHRwczovL2dpdGh1Yi5jb20vcHJlLWNvbW1pdC9wcmUtY29tbWl0CiAgICA6YWx0OiBwcmUtY29tbWl0CgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvY29kZSUyMHN0eWxlLWJsYWNrLTAwMDAwMC5zdmcKICAgIDp0YXJnZXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9wc2YvYmxhY2sKICAgIDphbHQ6IGJsYWNrCgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS92L3Jvcm9yby5zdmcKICAgIDp0YXJnZXQ6IGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9yb3Jvcm8vCiAgICA6YWx0OiBMYXRlc3QgVmVyc2lvbgoKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvcHl2ZXJzaW9ucy9yb3Jvcm8uc3ZnCiAgICA6dGFyZ2V0OiBodHRwczovL3B5cGkub3JnL3Byb2plY3Qvcm9yb3JvLwogICAgOmFsdDogUHl0aG9uIHZlcnNpb25zCgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9sL3Jvcm9yby5zdmcKICAgIDp0YXJnZXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9wbGF5cGF1c2VhbmRzdG9wL3Jvcm9yby9ibG9iL21haW4vTElDRU5TRQogICAgOmFsdDogQlNEIExpY2Vuc2UKCi4uIGltYWdlOjogaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vcmVwb3MvcGxheXBhdXNlYW5kc3RvcC9yb3Jvcm8vYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluJnNlcnZpY2U9Z2l0aHViCiAgICA6dGFyZ2V0OiBodHRwczovL2NvdmVyYWxscy5pby9naXRodWIvcGxheXBhdXNlYW5kc3RvcC9yb3Jvcm8KICAgIDphbHQ6IENvdmVyYWdlCgouLiBpbWFnZTo6IGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL3Jvcm9yby9iYWRnZS8/dmVyc2lvbj1sYXRlc3QKICAgIDp0YXJnZXQ6IGh0dHBzOi8vcm9yb3JvLnJlYWR0aGVkb2NzLmlvLwogICAgOmFsdDogRG9jdW1lbnRhdGlvbgoKSW1wbGVtZW50IGBhaW9odHRwLndlYmBfIGBPcGVuQVBJIDNgXyBzZXJ2ZXIgYXBwbGljYXRpb25zIHdpdGggc2NoZW1hIGZpcnN0CmFwcHJvYWNoLgoKQXMgd2VsbCBhcyBidW5jaCBvdGhlciB1dGlsaXRpZXMgdG8gYnVpbGQgZWZmZWN0aXZlIHNlcnZlciBhcHBsaWNhdGlvbnMgd2l0aApgUHl0aG9uYF8gMyAmIGBhaW9odHRwLndlYmBfLgoKKiBXb3JrcyBvbiBgUHl0aG9uYF8gMy44KwoqIFdvcmtzIHdpdGggYGFpb2h0dHAud2ViYF8gMy44LjErCiogQlNEIGxpY2Vuc2VkCiogU291cmNlLCBpc3N1ZXMsIGFuZCBwdWxsIHJlcXVlc3RzIGBvbiBHaXRIdWIKICA8aHR0cHM6Ly9naXRodWIuY29tL3BsYXlwYXVzZWFuZHN0b3Avcm9yb3JvPmBfCgouLiBfYE9wZW5BUEkgM2A6IGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMwouLiBfYGFpb2h0dHAud2ViYDogaHR0cHM6Ly9kb2NzLmFpb2h0dHAub3JnL2VuL3N0YWJsZS93ZWIuaHRtbAouLiBfYFB5dGhvbmA6IGh0dHBzOi8vd3d3LnB5dGhvbi5vcmcvCgpRdWljayBTdGFydAo9PT09PT09PT09PQoKKnJvcm9ybyogcmVsaWVzIG9uIHZhbGlkIE9wZW5BUEkgMyBzY2hlbWEgZmlsZSAoYm90aCBKU09OIG9yIFlBTUwgZm9ybWF0cwpzdXBwb3J0ZWQpLgoKRXhhbXBsZSBiZWxvdywgaWxsdXN0cmF0ZXMgb24gaG93IHRvIGhhbmRsZSBvcGVyYXRpb24gYGBoZWxsb193b3JsZGBgIGZyb20KYG9wZW5hcGkueWFtbGBfIHNjaGVtYSBmaWxlLgoKLi4gY29kZS1ibG9jazo6IHB5dGhvbgoKICAgIGZyb20gcGF0aGxpYiBpbXBvcnQgUGF0aAoKICAgIGZyb20gYWlvaHR0cCBpbXBvcnQgd2ViCiAgICBmcm9tIHJvcm9ybyBpbXBvcnQgKAogICAgICAgIG9wZW5hcGlfY29udGV4dCwKICAgICAgICBPcGVyYXRpb25UYWJsZURlZiwKICAgICAgICBzZXR1cF9vcGVuYXBpLAogICAgKQoKCiAgICBvcGVyYXRpb25zID0gT3BlcmF0aW9uVGFibGVEZWYoKQoKCiAgICBAb3BlcmF0aW9ucy5yZWdpc3RlcgogICAgYXN5bmMgZGVmIGhlbGxvX3dvcmxkKHJlcXVlc3Q6IHdlYi5SZXF1ZXN0KSAtPiB3ZWIuUmVzcG9uc2U6CiAgICAgICAgd2l0aCBvcGVuYXBpX2NvbnRleHQocmVxdWVzdCkgYXMgY29udGV4dDoKICAgICAgICAgICAgbmFtZSA9IGNvbnRleHQucGFyYW1ldGVycy5xdWVyeS5nZXQoIm5hbWUiLCAid29ybGQiKQogICAgICAgICAgICBlbWFpbCA9IGNvbnRleHQucGFyYW1ldGVycy5xdWVyeS5nZXQoCiAgICAgICAgICAgICAgICAiZW1haWwiLCAid29ybGRAZXhhbXBsZS5jb20iCiAgICAgICAgICAgICkKICAgICAgICAgICAgcmV0dXJuIHdlYi5qc29uX3Jlc3BvbnNlKAogICAgICAgICAgICAgICAgeyJtZXNzYWdlIjogZiJIZWxsbywge25hbWV9ISIsICJlbWFpbCI6IGVtYWlsfQogICAgICAgICAgICApCgoKICAgIGRlZiBjcmVhdGVfYXBwKGFyZ3Y6IGxpc3Rbc3RyXSA9IE5vbmUpIC0+IHdlYi5BcHBsaWNhdGlvbjoKICAgICAgICByZXR1cm4gc2V0dXBfb3BlbmFwaSgKICAgICAgICAgICAgd2ViLkFwcGxpY2F0aW9uKCksCiAgICAgICAgICAgIFBhdGgoX19maWxlX18pLnBhcmVudCAvICJvcGVuYXBpLnlhbWwiLAogICAgICAgICAgICBvcGVyYXRpb25zLAogICAgICAgICAgICBzZXJ2ZXJfdXJsPSIvYXBpIiwKICAgICAgICApCgouLiBfYG9wZW5hcGkueWFtbGA6IGh0dHBzOi8vZ2l0aHViLmNvbS9wbGF5cGF1c2VhbmRzdG9wL3Jvcm9yby9ibG9iL21haW4vdGVzdHMvcm9yb3JvL29wZW5hcGkueWFtbAoKU2NoZW1hIEZpcnN0IEFwcHJvYWNoCi0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKVW5saWtlIG90aGVyIHBvcHVsYXIgUHl0aG9uIE9wZW5BUEkgMyBzb2x1dGlvbnMsIHN1Y2ggYXMKYERqYW5nbyBSRVNUIEZyYW1ld29ya2BfLCBgRmFzdEFQSWBfLCAgYGZsYXNrLWFwaXNwZWNgXywgb3IgYGFpb2h0dHAtYXBpc3BlY2BfCipyb3Jvcm8qICoqcmVxdWlyZXMqKiB5b3UgdG8gcHJvdmlkZSB2YWxpZCBgT3BlbkFQSSAzYF8gc2NoZW1hIGZpcnN0LiBUaGlzCm1ha2VzICpyb3Jvcm8qIHNpbWlsYXIgdG8gYGNvbm5leGlvbmBfLCBgcHlyYW1pZF9vcGVuYXBpM2BfIGFuZCBvdGhlciBzY2hlbWEKZmlyc3QgbGlicmFyaWVzLgoKLi4gX2BEamFuZ28gUkVTVCBGcmFtZXdvcmtgOiBodHRwczovL3d3dy5kamFuZ28tcmVzdC1mcmFtZXdvcmsub3JnCi4uIF9gRmFzdEFQSWA6IGh0dHBzOi8vZmFzdGFwaS50aWFuZ29sby5jb20KLi4gX2BmbGFzay1hcGlzcGVjYDogaHR0cHM6Ly9mbGFzay1hcGlzcGVjLnJlYWR0aGVkb2NzLmlvCi4uIF9gYWlvaHR0cC1hcGlzcGVjYDogaHR0cHM6Ly9haW9odHRwLWFwaXNwZWMucmVhZHRoZWRvY3MuaW8KLi4gX2Bjb25uZXhpb25gOiBodHRwczovL2Nvbm5leGlvbi5yZWFkdGhlZG9jcy5pbwouLiBfYHB5cmFtaWRfb3BlbmFwaTNgOiBodHRwczovL2dpdGh1Yi5jb20vUHlsb25zL3B5cmFtaWRfb3BlbmFwaTMKCkNsYXNzIEJhc2VkIFZpZXdzCi0tLS0tLS0tLS0tLS0tLS0tCgoqcm9yb3JvKiBzdXBwb3J0cyBgY2xhc3MgYmFzZWQgdmlld3NgXyBhcyB3ZWxsLiBgVG9kby1CYWNrZW5kYF8gZXhhbXBsZQppbGx1c3RyYXRlcyBob3cgdG8gdXNlIGNsYXNzIGJhc2VkIHZpZXdzIGZvciBPcGVuQVBJIDMgc2VydmVycy4KCkluIHNuaXBwZXQgYmVsb3csICpyb3Jvcm8qIGV4cGVjdHMgdGhhdCBPcGVuQVBJIDMgc2NoZW1hIGNvbnRhaW5zIG9wZXJhdGlvbiBJRApgYFVzZXJWaWV3LmdldGBgLAoKLi4gY29kZS1ibG9jazo6IHB5dGhvbgoKICAgIEBvcGVyYXRpb25zLnJlZ2lzdGVyCiAgICBjbGFzcyBVc2VyVmlldyh3ZWIuVmlldyk6CiAgICAgICAgYXN5bmMgZGVmIGdldChzZWxmKSAtPiB3ZWIuUmVzcG9uc2U6IC4uLgoKCi4uIF9gY2xhc3MgYmFzZWQgdmlld3NgOiBodHRwczovL2RvY3MuYWlvaHR0cC5vcmcvZW4vc3RhYmxlL3dlYl9xdWlja3N0YXJ0Lmh0bWwjYWlvaHR0cC13ZWItY2xhc3MtYmFzZWQtdmlld3MKLi4gX2BUb2RvLUJhY2tlbmRgOiBodHRwczovL2dpdGh1Yi5jb20vcGxheXBhdXNlYW5kc3RvcC9yb3Jvcm8vdHJlZS9tYWluL2V4YW1wbGVzL3RvZG9iYWNrZW5kCgpNb3JlIEV4YW1wbGVzCi0tLS0tLS0tLS0tLS0KCkNoZWNrIGBleGFtcGxlc2BfIGZvbGRlciB0byBzZWUgb3RoZXIgZXhhbXBsZXMgb24gaG93IHRvIGJ1aWxkIGFpb2h0dHAud2ViCk9wZW5BUEkgMyBzZXJ2ZXIgYXBwbGljYXRpb25zLgoKLi4gX2BleGFtcGxlc2A6IGh0dHBzOi8vZ2l0aHViLmNvbS9wbGF5cGF1c2VhbmRzdG9wL3Jvcm9yby90cmVlL21haW4vZXhhbXBsZXMK readmeEtag: '"251bf3809845d785731edd01b21d86f83b382645"' readmeLastModified: Fri, 16 Aug 2024 19:22:28 GMT repositoryId: 6331465 description: >- Implement aiohttp.web OpenAPI 3 server applications with schema first approach. created: '2012-10-22T08:08:52Z' updated: '2025-12-07T18:11:19Z' language: Python archived: false stars: 106 watchers: 4 forks: 10 owner: playpauseandstop logo: https://avatars.githubusercontent.com/u/21325?v=4 license: BSD-3-Clause repoEtag: '"a9ff6601e2e615eb8d9d8328683b10f8c5a3120978ef8c61091ebef123daae22"' repoLastModified: Sun, 07 Dec 2025 18:11:19 GMT foundInMaster: true category: Server Implementations id: bccba2fe3a44fa5e2681bdcd3e986ba7 - source: - openapi3 tags - openapi31 tags repository: https://github.com/python-openapi/openapi-schema-validator v3: true id: 3e4ac3612c445ce7087449f67b501d73 repositoryMetadata: base64Readme: >- KioqKioqKioqKioqKioqKioqKioqKioqCm9wZW5hcGktc2NoZW1hLXZhbGlkYXRvcgoqKioqKioqKioqKioqKioqKioqKioqKioKCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3Yvb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yLnN2ZwogICAgIDp0YXJnZXQ6IGh0dHBzOi8vcHlwaS5weXRob24ub3JnL3B5cGkvb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yCi4uIGltYWdlOjogaHR0cHM6Ly90cmF2aXMtY2kub3JnL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvci5zdmc/YnJhbmNoPW1hc3RlcgogICAgIDp0YXJnZXQ6IGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9weXRob24tb3BlbmFwaS9vcGVuYXBpLXNjaGVtYS12YWxpZGF0b3IKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yL21hc3Rlci5zdmc/c3R5bGU9ZmxhdAogICAgIDp0YXJnZXQ6IGh0dHBzOi8vY29kZWNvdi5pby9naXRodWIvcHl0aG9uLW9wZW5hcGkvb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yP2JyYW5jaD1tYXN0ZXIKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvcHl2ZXJzaW9ucy9vcGVuYXBpLXNjaGVtYS12YWxpZGF0b3Iuc3ZnCiAgICAgOnRhcmdldDogaHR0cHM6Ly9weXBpLnB5dGhvbi5vcmcvcHlwaS9vcGVuYXBpLXNjaGVtYS12YWxpZGF0b3IKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvZm9ybWF0L29wZW5hcGktc2NoZW1hLXZhbGlkYXRvci5zdmcKICAgICA6dGFyZ2V0OiBodHRwczovL3B5cGkucHl0aG9uLm9yZy9weXBpL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvcgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9zdGF0dXMvb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yLnN2ZwogICAgIDp0YXJnZXQ6IGh0dHBzOi8vcHlwaS5weXRob24ub3JnL3B5cGkvb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yCgpBYm91dAojIyMjIwoKT3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yIGlzIGEgUHl0aG9uIGxpYnJhcnkgdGhhdCB2YWxpZGF0ZXMgc2NoZW1hIGFnYWluc3Q6CgoqIGBPcGVuQVBJIFNjaGVtYSBTcGVjaWZpY2F0aW9uIHYzLjAgPGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kI3NjaGVtYU9iamVjdD5gX18gd2hpY2ggaXMgYW4gZXh0ZW5kZWQgc3Vic2V0IG9mIHRoZSBgSlNPTiBTY2hlbWEgU3BlY2lmaWNhdGlvbiBXcmlnaHQgRHJhZnQgMDAgPGh0dHA6Ly9qc29uLXNjaGVtYS5vcmcvPmBfXy4KKiBgT3BlbkFQSSBTY2hlbWEgU3BlY2lmaWNhdGlvbiB2My4xIDxodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjEuMC5tZCNzY2hlbWFPYmplY3Q+YF9fIHdoaWNoIGlzIGFuIGV4dGVuZGVkIHN1cGVyc2V0IG9mIHRoZSBgSlNPTiBTY2hlbWEgU3BlY2lmaWNhdGlvbiBEcmFmdCAyMDIwLTEyIDxodHRwOi8vanNvbi1zY2hlbWEub3JnLz5gX18uCgoKRG9jdW1lbnRhdGlvbgojIyMjIyMjIyMjIyMjCgpDaGVjayBkb2N1bWVudGF0aW9uIHRvIHNlZSBtb3JlIGRldGFpbHMgYWJvdXQgdGhlIGZlYXR1cmVzLiBBbGwgZG9jdW1lbnRhdGlvbiBpcyBpbiB0aGUgImRvY3MiIGRpcmVjdG9yeSBhbmQgb25saW5lIGF0IGBvcGVuYXBpLXNjaGVtYS12YWxpZGF0b3IucmVhZHRoZWRvY3MuaW8gPGh0dHBzOi8vb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yLnJlYWR0aGVkb2NzLmlvPmBfXwoKCkluc3RhbGxhdGlvbgojIyMjIyMjIyMjIyMKClJlY29tbWVuZGVkIHdheSAodmlhIHBpcCk6CgouLiBjb2RlLWJsb2NrOjogY29uc29sZQoKICAgcGlwIGluc3RhbGwgb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yCgpBbHRlcm5hdGl2ZWx5IHlvdSBjYW4gZG93bmxvYWQgdGhlIGNvZGUgYW5kIGluc3RhbGwgZnJvbSB0aGUgcmVwb3NpdG9yeToKCi4uIGNvZGUtYmxvY2s6OiBjb25zb2xlCgogICBwaXAgaW5zdGFsbCAtZSBnaXQraHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvci5naXQjZWdnPW9wZW5hcGlfc2NoZW1hX3ZhbGlkYXRvcgoKClVzYWdlCiMjIyMjCgpUbyB2YWxpZGF0ZSBhbiBPcGVuQVBJIHYzLjEgc2NoZW1hOgoKLi4gY29kZS1ibG9jazo6IHB5dGhvbgoKICAgZnJvbSBvcGVuYXBpX3NjaGVtYV92YWxpZGF0b3IgaW1wb3J0IHZhbGlkYXRlCgogICAjIEEgc2FtcGxlIHNjaGVtYQogICBzY2hlbWEgPSB7CiAgICAgICAidHlwZSI6ICJvYmplY3QiLAogICAgICAgInJlcXVpcmVkIjogWwogICAgICAgICAgIm5hbWUiCiAgICAgICBdLAogICAgICAgInByb3BlcnRpZXMiOiB7CiAgICAgICAgICAgIm5hbWUiOiB7CiAgICAgICAgICAgICAgICJ0eXBlIjogInN0cmluZyIKICAgICAgICAgICB9LAogICAgICAgICAgICJhZ2UiOiB7CiAgICAgICAgICAgICAgICJ0eXBlIjogWyJpbnRlZ2VyIiwgIm51bGwiXSwKICAgICAgICAgICAgICAgImZvcm1hdCI6ICJpbnQzMiIsCiAgICAgICAgICAgICAgICJtaW5pbXVtIjogMCwKICAgICAgICAgICB9LAogICAgICAgICAgICJiaXJ0aC1kYXRlIjogewogICAgICAgICAgICAgICAidHlwZSI6ICJzdHJpbmciLAogICAgICAgICAgICAgICAiZm9ybWF0IjogImRhdGUiLAogICAgICAgICAgIH0sCiAgICAgICAgICAgImFkZHJlc3MiOiB7CiAgICAgICAgICAgICAgICAidHlwZSI6ICdhcnJheScsCiAgICAgICAgICAgICAgICAicHJlZml4SXRlbXMiOiBbCiAgICAgICAgICAgICAgICAgICAgeyAidHlwZSI6ICJudW1iZXIiIH0sCiAgICAgICAgICAgICAgICAgICAgeyAidHlwZSI6ICJzdHJpbmciIH0sCiAgICAgICAgICAgICAgICAgICAgeyAiZW51bSI6IFsiU3RyZWV0IiwgIkF2ZW51ZSIsICJCb3VsZXZhcmQiXSB9LAogICAgICAgICAgICAgICAgICAgIHsgImVudW0iOiBbIk5XIiwgIk5FIiwgIlNXIiwgIlNFIl0gfQogICAgICAgICAgICAgICAgXSwKICAgICAgICAgICAgICAgICJpdGVtcyI6IEZhbHNlLAogICAgICAgICAgICB9CiAgICAgICB9LAogICAgICAgImFkZGl0aW9uYWxQcm9wZXJ0aWVzIjogRmFsc2UsCiAgIH0KCiAgICMgSWYgbm8gZXhjZXB0aW9uIGlzIHJhaXNlZCBieSB2YWxpZGF0ZSgpLCB0aGUgaW5zdGFuY2UgaXMgdmFsaWQuCiAgIHZhbGlkYXRlKHsibmFtZSI6ICJKb2huIiwgImFnZSI6IDIzLCAiYWRkcmVzcyI6IFsxNjAwLCAiUGVubnN5bHZhbmlhIiwgIkF2ZW51ZSJdfSwgc2NoZW1hKQoKICAgdmFsaWRhdGUoeyJuYW1lIjogIkpvaG4iLCAiY2l0eSI6ICJMb25kb24ifSwgc2NoZW1hKQoKICAgVHJhY2ViYWNrIChtb3N0IHJlY2VudCBjYWxsIGxhc3QpOgogICAgICAgLi4uCiAgIFZhbGlkYXRpb25FcnJvcjogQWRkaXRpb25hbCBwcm9wZXJ0aWVzIGFyZSBub3QgYWxsb3dlZCAoJ2NpdHknIHdhcyB1bmV4cGVjdGVkKQoKQnkgZGVmYXVsdCwgdGhlIGxhdGVzdCBPcGVuQVBJIHNjaGVtYSBzeW50YXggaXMgZXhwZWN0ZWQuCgpGb3IgbW9yZSBkZXRhaWxzIHJlYWQgYWJvdXQgYFZhbGlkYXRpb24gPGh0dHBzOi8vb3BlbmFwaS1zY2hlbWEtdmFsaWRhdG9yLnJlYWR0aGVkb2NzLmlvL2VuL2xhdGVzdC92YWxpZGF0aW9uLmh0bWw+YF9fLgoKUmVsYXRlZCBwcm9qZWN0cwojIyMjIyMjIyMjIyMjIyMjCiogYG9wZW5hcGktY29yZSA8aHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktY29yZT5gX18KICAgUHl0aG9uIGxpYnJhcnkgdGhhdCBhZGRzIGNsaWVudC1zaWRlIGFuZCBzZXJ2ZXItc2lkZSBzdXBwb3J0IGZvciB0aGUgT3BlbkFQSS4KKiBgb3BlbmFwaS1zcGVjLXZhbGlkYXRvciA8aHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1vcGVuYXBpL29wZW5hcGktc3BlYy12YWxpZGF0b3I+YF9fCiAgIFB5dGhvbiBsaWJyYXJ5IHRoYXQgdmFsaWRhdGVzIE9wZW5BUEkgU3BlY3MgYWdhaW5zdCB0aGUgT3BlbkFQSSAyLjAgKGFrYSBTd2FnZ2VyKSBhbmQgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbgo= readmeEtag: '"5d8b66ff8d4ebe6abb9e4bf725860d6ec4e1defe"' readmeLastModified: Sat, 13 Jul 2024 06:54:27 GMT repositoryId: 245119225 description: >- OpenAPI schema validator is a Python library that validates schema against OpenAPI Schema Specification v3.0 and v3.1 created: '2020-03-05T09:23:39Z' updated: '2026-02-04T03:21:21Z' language: Python archived: false stars: 121 watchers: 3 forks: 35 owner: python-openapi logo: https://avatars.githubusercontent.com/u/126442889?v=4 license: BSD-3-Clause repoEtag: '"162561e0dce076129857be41a992c4eeb7b0956a5ab1a91dad029a4c2b981375"' repoLastModified: Wed, 04 Feb 2026 03:21:21 GMT category: Parsers oldLocations: - https://github.com/p1c2u/openapi-schema-validator foundInMaster: true v3_1: true - source: openapi3 tags name: openapi4j category: - Data Validators - Schema Validators - Parsers repository: https://github.com/openapi4j/openapi4j language: Java source_description: >- Parse Description Document, validate API requests and responses using OpenAPI 3.x. v2: false v3: true repositoryMetadata: base64Readme: >- WyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5jb20vb3BlbmFwaTRqL29wZW5hcGk0ai5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLmNvbS9vcGVuYXBpNGovb3BlbmFwaTRqKQpbIVtNYWludGFpbmFiaWxpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1vcmcub3BlbmFwaTRqJTNBb3BlbmFwaTRqJm1ldHJpYz1zcWFsZV9yYXRpbmcpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPW9yZy5vcGVuYXBpNGolM0FvcGVuYXBpNGopClshW1NlY3VyaXR5IFJhdGluZ10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9b3JnLm9wZW5hcGk0aiUzQW9wZW5hcGk0aiZtZXRyaWM9c2VjdXJpdHlfcmF0aW5nKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1vcmcub3BlbmFwaTRqJTNBb3BlbmFwaTRqKQpbIVtDb3ZlcmFnZV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9b3JnLm9wZW5hcGk0aiUzQW9wZW5hcGk0aiZtZXRyaWM9Y292ZXJhZ2UpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPW9yZy5vcGVuYXBpNGolM0FvcGVuYXBpNGopCgpfX1RoaXMgcmVwb3NpdG9yeSBpcyBub3cgYXJjaGl2ZWQuIEkgZG9uJ3QgaGF2ZSBlbm91Z2ggc3BhcmUgdGltZSB0byBtYWludGFpbiB0aGlzIHByb2plY3QgKHdlbGwgYWN0dWFsbHkgcmV2YW1wKSBhbmQgZm9sbG93IE9BSSBzcGVjcy4gVGhpcyBwcm9qZWN0IGRlc2VydmVzIG11Y2ggbW9yZSB0aGF0IEkgY2FuIGdpdmUgdG8gc291cmNlIGNvZGUgYW5kIGZvbGxvd2VycyB0byBwcm92aWRlIGFwcHJvcHJpYXRlIG91dHB1dC5fXwoKIyBPcGVuQVBJIGZvciBqYXZhIHByb2plY3QgaG9tZQoKVGhpcyBpcyB0aGUgaG9tZSBwYWdlIG9mIHRoZSBvcGVuYXBpNGogcHJvamVjdCBmb3IgSmF2YSAoSmFrYXJ0YSBvciBKVk0gcGxhdGZvcm0gaW4gZ2VuZXJhbCkuCgpvcGVuYXBpNGogaXMgYSBzdWl0ZSBvZiB0b29scywgaW5jbHVkaW5nIHRoZSBmb2xsb3dpbmcgOgoqIFtPcGVuIEFQSSBzcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMy5tZCkgcGFyc2VyIGFuZCB2YWxpZGF0b3IuCiogT3BlbiBBUEkgW1NjaGVtYSBPYmplY3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4zLm1kI3NjaGVtYU9iamVjdCkgdmFsaWRhdG9yLgoqIFtKU09OIHJlZmVyZW5jZV0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL2RyYWZ0LXBicnlhbi16eXAtanNvbi1yZWYtMDMpIGltcGxlbWVudGF0aW9uLgoqIFJlcXVlc3QvcmVzcG9uc2UgdmFsaWRhdG9yIGFnYWluc3Qgb3BlcmF0aW9uLgoqIEZvciBpbnRlcm5hbCB1c2Ugb25seSwgcGVyZm9ybWFuY2UgcHJvamVjdCByZXBvcnRzIHNvbWUgbnVtYmVycyB0byAnbWFudWFsbHknIGNoZWNrIGFueSBpbXByb3ZlbWVudHMgb3IgcmVncmVzc2lvbnMgYmV0d2VlbiB2ZXJzaW9ucy4KCiMjIE1vZHVsZXMKCiogW1BhcnNlcl0ob3BlbmFwaS1wYXJzZXIpIGFsbG93cyB0aGUgKGRlLSlzZXJpYWxpemF0aW9uIGFuZCBtYW5pcHVsYXRpb24gb2YgdGhlIHNjaGVtYSBhbmQgaXRzIHZhbGlkYXRpb24uCiogW1NjaGVtYSB2YWxpZGF0b3JdKG9wZW5hcGktc2NoZW1hLXZhbGlkYXRvcikgYWxsb3dzIHRoZSB2YWxpZGF0aW9uIG9mIGRhdGEgYWdhaW5zdCBhIGdpdmVuIHNjaGVtYS4KKiBbUmVxdWVzdCB2YWxpZGF0b3JdKG9wZW5hcGktb3BlcmF0aW9uLXZhbGlkYXRvcikgaXMgaGlnaCBsZXZlbCBtb2R1bGUgdG8gbWFuYWdlIHZhbGlkYXRpb24gZm9yIHJlcXVlc3RzIGFuZC9vciByZXNwb25zZXMgYWdhaW5zdCBvcGVyYXRpb25zLiBNb3JlIGRldGFpbHMgaW4gdGhlIHJlbGF0ZWQgcHJvamVjdC4KKiBbUmVxdWVzdCBhZGFwdGVyc10ob3BlbmFwaS1vcGVyYXRpb24tYWRhcHRlcnMpIGlzIHRoZSByZXBvc2l0b3J5IG9mIHNwZWNpZmljIGFkYXB0ZXJzIHRvIHdyYXAgcmVxdWVzdHMgYW5kIHJlc3BvbnNlcy4KCiMjIERvY3VtZW50YXRpb24KClRoZSBkb2N1bWVudGF0aW9uIGZvciBhbGwgbW9kdWxlcyBpcyBhdmFpbGFibGUgW2hlcmVdKGh0dHBzOi8vb3BlbmFwaTRqLmdpdGh1Yi5pby9vcGVuYXBpNGovKS4KCiMjIFZlcnNpb25pbmcgYW5kIGNvbXBhdGliaWxpdHkKCkFsbCBtb2R1bGVzIGZvbGxvdyB0aGUgW1NlbWFudGljIFZlcnNpb25pbmcgMi4wLjBdKGh0dHBzOi8vc2VtdmVyLm9yZykgYW5kIGFyZSBhbGlnbmVkIG9uIGVhY2ggcmVsZWFzZSBldmVuIHRoZXJlJ3Mgbm8gY2hhbmdlcy4KCmBgYHhtbAo8ZGVwZW5kZW5jeT4KICAgIDxncm91cElkPm9yZy5vcGVuYXBpNGo8L2dyb3VwSWQ+CiAgICA8YXJ0aWZhY3RJZD5vcGVuYXBpLVttb2R1bGVdPC9hcnRpZmFjdElkPgo8L2RlcGVuZGVuY3k+CmBgYAoKWyFbUmVsZWFzZSB2ZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL25leHVzL3Ivb3JnLm9wZW5hcGk0ai9vcGVuYXBpLW9wZXJhdGlvbi12YWxpZGF0b3I/c3R5bGU9Zm9yLXRoZS1iYWRnZSZjb2xvcj1ibHVlJmxhYmVsPVJlbGVhc2Umc2VydmVyPWh0dHBzJTNBJTJGJTJGb3NzLnNvbmF0eXBlLm9yZyldKGh0dHBzOi8vc2VhcmNoLm1hdmVuLm9yZy9zZWFyY2g/cT1nOm9yZy5vcGVuYXBpNGopClshW1NuYXBzaG90IHZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbmV4dXMvcy9vcmcub3BlbmFwaTRqL29wZW5hcGktb3BlcmF0aW9uLXZhbGlkYXRvcj9zdHlsZT1mb3ItdGhlLWJhZGdlJmNvbG9yPWJsdWUmbGFiZWw9U25hcHNob3Qmc2VydmVyPWh0dHBzJTNBJTJGJTJGb3NzLnNvbmF0eXBlLm9yZyldKGh0dHBzOi8vb3NzLnNvbmF0eXBlLm9yZy9jb250ZW50L3JlcG9zaXRvcmllcy9zbmFwc2hvdHMvb3JnL29wZW5hcGk0ai8pCgpTbmFwc2hvdCBpcyBhdmFpbGFibGUgZm9yIGxhdGVzdCB2YWxpZCBjb21taXQgb24gJ21hc3RlcicgYnJhbmNoLgoKIyMgUGVyZm9ybWFuY2UKCkNoZWNrIFtoZXJlXShodHRwczovL3d3dy5vcGVuYXBpNGoub3JnL3BlcmYtdmFsdWVzLmh0bWwpIGZvciBzb21lIHZhbHVlcy4KCiMjIE5hdGl2ZSBjb21waWxhdGlvbiAoR3JhYWxWTSkKCkZyb20gdmVyc2lvbiAwLjcsIHRoZSB0b29sc2V0IGlzIGZ1bGx5IGNvbXBsaWFudCB3aXRoIG5hdGl2ZSBjb21waWxhdGlvbiAoQU9UKS4KVGhpcyB3YXMgdGVzdGVkIHdpdGggR3JhYWxWTSAxOS4zLjEuCk5vIGZ1cnRoZXIgY29uZmlndXJhdGlvbiBvciBkaXJlY3RpdmUgaXMgbmVlZGVkIHRvIGluY2x1ZGUgdGhlIG1vZHVsZXMgaWYgYXZhaWxhYmxlIG9uIGNsYXNzcGF0aC4KCmBgYHNoZWxsIHNjcmlwdApuYXRpdmUtaW1hZ2UgLUg6K1JlcG9ydEV4Y2VwdGlvblN0YWNrVHJhY2VzIC0tbm8tZmFsbGJhY2sgLWphciB5b3VyLWFwcC5qYXIKYGBgCgpGWUksIHRlc3RpbmcgcnVucyBtYWRlIGRvbid0IHNob3cgbXVjaCBwZXJmb3JtYW5jZSBpbXByb3ZlbWVudHMgYnV0IHBhcnNlciBtb2R1bGUuCgojIyBTdXBwb3J0ZWQgdmVyc2lvbnMKClRoZSBtb2R1bGVzIGN1cnJlbnRseSBzdXBwb3J0IHRoZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gKE9BUykgdmVyc2lvbiAzLjAueC4KCk9BSSAzLjEuMCBoYXMgYmVlbiByZWxlYXNlZCBhcyBjYW5kaWRhdGUuICAKVGhlcmUncyB0b28gbXVjaCBjaGFuZ2VzIHRvbyBrZWVwIGNvZGUgb24gc2FtZSBiYXNpcyBhbmQga2VlcCBhIGZhaXJseSBsb3cgbGV2ZWwgb2YgY29tcGxleGl0eS4gIApBcyBhIGNvbnNlcXVlbmNlLCBPQUkgMy4xLnggc3VwcG9ydCB3aWxsIGJlIG1hZGUgaW4gYSB2ZXJzaW9uIDIgb2Ygb3BlbmFwaTRqLiAgCgpBcyBteSB0aW1lIGlzIHZlcnkgbGltaXRlZCwgdmVyc2lvbiAxIHNob3VsZCBiZSBjb25zaWRlcmVkIGFzIGZyZWV6ZWQgZm9yIG5vdy4KClNlZSByZWxhdGVkIHByb2plY3RzIGZvciBsaW1pdGF0aW9ucyBhbmQgaXNzdWVzLgoKIyMgQ29udHJpYnV0aW5nCgpSZXBvcnRpbmcgaXNzdWVzLCBtYWtpbmcgY29tbWVudHMsIC4uLiBBbnkgaGVscCBpcyB3ZWxjb21lICEKCldlIGFjY2VwdCBQdWxsIFJlcXVlc3RzIHZpYSBHaXRIdWIuIFRoZXJlIGFyZSBzb21lIGd1aWRlbGluZXMgd2hpY2ggd2lsbCBtYWtlIGFwcGx5aW5nIFBScyBlYXNpZXIgZm9yIHVzIDoKCiogUmVzcGVjdCB0aGUgY29kZSBzdHlsZSBhbmQgaW5kZW50YXRpb24uIC5lZGl0b3Jjb25maWcgZmlsZSBpcyBwcm92aWRlZCB0byBub3QgYmUgd29ycmllZCBhYm91dCB0aGlzLgoqIENyZWF0ZSBtaW5pbWFsIGRpZmZzIC0gZGlzYWJsZSBvbiBzYXZlIGFjdGlvbnMgbGlrZSByZWZvcm1hdCBzb3VyY2UgY29kZSBvciBvcmdhbml6ZSBpbXBvcnRzLiBJZiB5b3UgZmVlbCB0aGUgc291cmNlIGNvZGUgc2hvdWxkIGJlIHJlZm9ybWF0dGVkIGNyZWF0ZSBhIHNlcGFyYXRlIFBSIGZvciB0aGlzIGNoYW5nZS4KKiBQcm92aWRlIEpVbml0IHRlc3RzIGZvciB5b3VyIGNoYW5nZXMgYW5kIG1ha2Ugc3VyZSB5b3VyIGNoYW5nZXMgZG9uJ3QgYnJlYWsgYW55dGhpbmcgYnkgcnVubmluZyBgZ3JhZGxldyBjbGVhbiBjaGVja2AuCiogUHJvdmlkZSBhIHNlbGYgZXhwbGFuYXRvcnkgYnV0IGJyaWVmIGNvbW1pdCBtZXNzYWdlIHdpdGggaXNzdWUgcmVmZXJlbmNlIGlmIGFueSwgYXMgaXQgd2lsbCBiZSByZXBvcnRlZCBkaXJlY3RseSBmb3IgcmVsZWFzZSBjaGFuZ2Vsb2cuCgojIyBMaWNlbnNlCgpvcGVuYXBpNGogYW5kIGFsbCB0aGUgbW9kdWxlcyBhcmUgcmVsZWFzZWQgdW5kZXIgdGhlIEFwYWNoZSAyLjAgbGljZW5zZS4gU2VlIFtMSUNFTlNFXShodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaTRqL29wZW5hcGk0ai9ibG9iL21hc3Rlci9MSUNFTlNFLm1kKSBmb3IgZGV0YWlscy4K readmeEtag: '"0af446f681826fbe20696adc11289e18a75b18ff"' readmeLastModified: Fri, 09 Jul 2021 15:33:59 GMT repositoryId: 218804581 description: OpenAPI 3 parser, JSON schema and request validator. created: '2019-10-31T15:59:35Z' updated: '2026-01-31T08:37:16Z' language: Java archived: true stars: 109 watchers: 4 forks: 40 owner: openapi4j logo: https://avatars.githubusercontent.com/u/39759775?v=4 license: Apache-2.0 repoEtag: '"d0dfbaec93724584a3b5e20c365ee4ba933c374732a235dc06689f4b167c72ff"' repoLastModified: Sat, 31 Jan 2026 08:37:16 GMT foundInMaster: true id: c5b32630ea7cd375c61c41563e2a2f09 - source: openapi3 tags repository: https://github.com/davidmoten/openapi-to-plantuml v3: true repositoryMetadata: base64Readme: >- # openapi-to-plantuml
<a href="https://github.com/davidmoten/openapi-to-plantuml/actions/workflows/ci.yml"><img src="https://github.com/davidmoten/openapi-to-plantuml/actions/workflows/ci.yml/badge.svg"/></a><br/>
[![codecov](https://codecov.io/gh/davidmoten/openapi-to-plantuml/branch/main/graph/badge.svg)](https://codecov.io/gh/davidmoten/openapi-to-plantuml)<br/>
[![Maven Central](https://img.shields.io/maven-central/v/com.github.davidmoten/openapi-to-plantuml)](https://central.sonatype.com/artifact/com.github.davidmoten/openapi-to-plantuml)<br/>

Java library and Maven plugin to generate a [PlantUML](https://plantuml.com) Class Diagram (supplemented with path information) from an OpenAPI 3.0 definition (YAML or JSON). 

Try it online [here](https://openapi-to-puml.davidmoten.org/prod/site/index.html).

**Features**

* Maven plugin to produce diagrams in multiple formats
* Gradle usage example
* One consolidated diagram or diagram-per-operation options
  
Supports all features of OpenAPI 3.0 including

* Mixed types (*oneOf*, *anyOf*, *allOf*) 
* Multiple levels of indirection (`$ref` -> `$ref` for example) 
* Multiple response content types
* Can produce diagram-per-path (great for large APIs)

**Status:** Released to Maven Central 

**Limitations**

* Only handles internal references ("#/*"), that is the yaml file should be self-contained. As of 0.1.9 external refs are displayed but not exploded. Full external reference support is being considered.

**TODO**
* Model *Callbacks*
* Model *additionalProperties*

[openapi-example.yml](src/test/resources/openapi-example.yml): 

<img style="background-color:white" src="openapi-to-plantuml/src/docs/openapi-example.svg"/>

## How to build

```bash
mvn clean install
```

## Getting started

### Online diagram generator (SINGLE style only)
The easiest thing to do is to use the [online diagram generator](https://openapi-to-puml.davidmoten.org/prod/site/index.html). If your definition is large then PlantText may refuse to render it (the Plant UML text is encoded and added to the PlantText url which can get too long). 

To trial quickly, clone this repo and add your yaml file to `src/test/resources/demo/` directory and run `mvn test`. The generated image will be saved in `target/demos`.

### Maven Plugin
Use the Maven plugin:

```xml
<plugin>
    <groupId>com.github.davidmoten</groupId>
    <artifactId>openapi-to-plantuml-maven-plugin</artifactId>
    <version>VERSION_HERE</version>
    <executions>
        <execution>
            <id>generate-split</id>
            <goals>
                <goal>generate</goal>
            </goals>
            <configuration>
                <style>SPLIT</style> <!-- one diagram per operation -->
                <input>src/main/openapi/openapi.yml</input>
                <formats>
                    <format>PNG</format>
                    <format>SVG</format>
                </formats>
            </configuration>
        </execution>
    </executions>
</plugin>
```
| Parameter | Type          | Required | Default                          | Description |
|-----------|---------------|----------|----------------------------------|-------------|
| `style`   | `Style`       | No       | `SPLIT`                          | The style of the generated diagrams. `SINGLE` creates one diagram for all operations, while `SPLIT` creates a separate diagram per operation (recommended for large OpenAPI specs). |
| `input`   | `File`        | Yes      | –                                | The source OpenAPI file to generate diagrams from. Can include relative references to other files. |
| `formats` | `List<String>`| No       | PNG, SVG                         | Output formats. Possible values include: `PUML`, `EPS`, `EPS_TEXT`, `ATXT`, `UTXT`, `XMI_STANDARD`, `XMI_STAR`, `XMI_ARGO`, `SCXML`, `GRAPHML`, `PDF`, `MJPEG`, `ANIMATED_GIF`, `HTML`, `HTML5`, `VDX`, `LATEX`, `LATEX_NO_PREAMBLE`, `BASE64`, `BRAILLE_PNG`, `PREPROC`, `DEBUG`, `PNG`, `RAW`, `SVG` |
| `output`  | `File`        | No       | `target/diagrams/class-diagram.png` (for `SINGLE`) or `target/diagrams` (for `SPLIT`) | The output file (for `SINGLE`) or output directory (for `SPLIT`). |

### Standalone jar

An alternative to the maven plugin is to use the standalone *jar-with-dependencies* artifact from [Maven Central](https://search.maven.org/search?q=g:com.github.davidmoten%20AND%20a:openapi-to-plantuml) (or from the `target` directory if you have built the project locally):

```bash
java -jar openapi-to-plantuml-VERSION_HERE-jar-with-dependencies.jar single openapi.yaml PNG openapi.png
```
Large diagrams are better with SVG format (rendering is sharp at every zoom level) so just replace `PNG` in the above command with `SVG`. If you still want a PNG and you find it 
is getting cut off then you need to increase the plantuml image size limit like below. Increase the limit till your diagram fits. You may need to increase memory using this JVM arg `-Xmx1024m`.

```bash
java -DPLANTUML_LIMIT_SIZE=8192 -jar openapi-to-plantuml-VERSION_HERE-jar-with-dependencies.jar openapi.yaml PNG openapi.png
```
### Gradle
You can configure the diagram generator from Gradle in `build.gradle` like this:
```groovy
configurations {
    diagrams
}

dependencies {
    diagrams 'com.github.davidmoten:openapi-to-plantuml:VERSION_HERE:jar-with-dependencies'
}

task diagrams(type: JavaExec) {
    mainClass = 'com.github.davidmoten.oas3.puml.ConverterMain'
    classpath = configurations.diagrams
    args = ['split','openapi.yaml','PNG', 'build/diagrams']
}

// include this to ensure runs with `gradle build`
build.dependsOn diagrams
```
Then call
```bash
gradle diagrams
```

### How to produce a diagram per service method

```bash
java -jar openapi-to-plantuml-VERSION_HERE-jar-with-dependencies.jar split openapi.yaml PNG diagrams
```
The above call will write a diagram per method into the `diagrams` directory (and will create it if it doesn't exist). 

As an example, do `mvn clean test` on the project and look in `target/unqork.svg` directory.

## Usage

```java
import com.github.davidmoten.oas3.puml.Converter;

String puml = Converter.openApiToPuml(openapi);
```

## System properties
* `max.enum.entries` defaults to 12. If an enum has more than the max then it will be limited to that number and an entry `...` will appear at the end of the enums.

## Examples

Unit test examples are [here](openapi-to-plantuml/src/docs/generated/examples.md).

[petstore-expanded.yml](openapi-to-plantuml/src/test/resources/inputs/petstore-expanded.yml):

<img src="openapi-to-plantuml/src/docs/tests/petstore-expanded.puml.svg"/>

[bookstore.yml](openapi-to-plantuml/src/test/resources/demos/bookstore.yml):

<img src="openapi-to-plantuml/src/docs/demos/bookstore.svg"/>

[ecommerce.yml](openapi-to-plantuml/src/test/resources/demos/ecommerce.yml)

<img src="openapi-to-plantuml/src/docs/demos/ecommerce.svg"/>

[news.yml](openapi-to-plantuml/src/test/resources/demos/news.yml):

<img src="openapi-to-plantuml/src/docs/demos/news.svg"/>

[strava.yml](openapi-to-plantuml/src/test/resources/demos/strava.yml):

<img src="openapi-to-plantuml/src/docs/demos/strava.svg"/>

[twitter.yml](openapi-to-plantuml/src/test/resources/demos/twitter.yml):

<img src="openapi-to-plantuml/src/docs/demos/twitter.svg"/>

[ebay.yml](openapi-to-plantuml/src/test/resources/demos/ebay.yml):

<img src="openapi-to-plantuml/src/docs/demos/ebay.svg"/>

 readmeEtag: '"0098afee878ec5c4c314b98e13ff339c3d69b1be"' readmeLastModified: Tue, 22 Jul 2025 22:51:28 GMT repositoryId: 339604008 description: >- Converts OpenAPI 3.0 definitions to Plant UML text for visualisation of your API. created: '2021-02-17T03:47:43Z' updated: '2026-02-02T23:43:30Z' language: Java archived: false stars: 113 watchers: 4 forks: 22 owner: davidmoten logo: https://avatars.githubusercontent.com/u/318187?v=4 repoEtag: '"45b61bb985493c7e67c57870dff08c778d527c65da2cd43b1db89704710d005e"' repoLastModified: Mon, 02 Feb 2026 23:43:30 GMT foundInMaster: true category: SDK id: d4f9d833f5a48b843c52f5ec6f582976 - source: openapi3 tags repository: https://github.com/oatpp/oatpp-swagger v3: true repositoryMetadata: base64Readme: >- IyBvYXRwcC1zd2FnZ2VyIFshW29hdHBwIGJ1aWxkIHN0YXR1c10oaHR0cHM6Ly9kZXYuYXp1cmUuY29tL2xnYW56enpvL2xnYW56enpvL19hcGlzL2J1aWxkL3N0YXR1cy9vYXRwcC5vYXRwcC1zd2FnZ2VyKV0oaHR0cHM6Ly9kZXYuYXp1cmUuY29tL2xnYW56enpvL2xnYW56enpvL19idWlsZD9kZWZpbml0aW9uSWQ9MikKU3dhZ2dlciBVSSBmb3Igb2F0cHAgc2VydmljZXMKClJlYWQgbW9yZToKLSBbQWJvdXQgb2F0cHBdKGh0dHBzOi8vb2F0cHAuaW8vKSAgCi0gW1doYXQgaXMgU3dhZ2dlciBVSV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3Rvb2xzL3N3YWdnZXItdWkvKQotIFtFbmRwb2ludCBhbm5vdGF0aW9uIGFuZCBBUEkgZG9jdW1lbnRhdGlvbiBpbiBvYXRwcF0oaHR0cHM6Ly9vYXRwcC5pby9kb2NzL2NvbXBvbmVudHMvYXBpLWNvbnRyb2xsZXIvI2VuZHBvaW50LWFubm90YXRpb24tYW5kLWFwaS1kb2N1bWVudGF0aW9uKS4KCiMjIEV4YW1wbGUKCkZvciBmdWxsIGV4YW1wbGUgcHJvamVjdCBzZWU6IFtFeGFtcGxlIENSVUQtQVBJIHByb2plY3Qgd2l0aCBTd2FnZ2VyIFVJXShodHRwczovL2dpdGh1Yi5jb20vb2F0cHAvZXhhbXBsZS1jcnVkKQoKIyMgQnJpZWYKCi0gVXNlIGBgYG9hdHBwOjpzd2FnZ2VyOjpDb250cm9sbGVyYGBgIHdpdGggYGBgb2F0cHA6OndlYjo6c2VydmVyOjpIdHRwQ29ubmVjdGlvbkhhbmRsZXJgYGAKLSBVc2UgYGBgb2F0cHA6OnN3YWdnZXI6OkFzeW5jQ29udHJvbGxlcmBgYCB3aXRoIGBgYG9hdHBwOjp3ZWI6OnNlcnZlcjo6QXN5bmNIdHRwQ29ubmVjdGlvbkhhbmRsZXJgYGAKCi0gU3dhZ2dlciBVSSBsb2NhdGlvbiAtIGBgYGh0dHA6Ly9sb2NhbGhvc3Q6PFBPUlQ+L3N3YWdnZXIvdWlgYGAKLSBPcGVuQXBpIDMuMC4wIHNwZWNpZmljYXRpb24gbG9jYXRpb24gLSBgYGBodHRwOi8vbG9jYWxob3N0OjxQT1JUPi9hcGktZG9jcy9vYXMtMy4wLjAuanNvbmBgYAoKSWYgeW91IGFyZSB1c2luZyBgYGBvYXRwcDo6d2ViOjpzZXJ2ZXI6OmFwaTo6QXBpQ29udHJvbGxlcmBgYCBtb3N0IHBhcnRzIG9mIHlvdXIgZW5kcG9pbnRzIGFyZSBkb2N1bWVudGVkIGF1dG9tYXRpY2FsbHkgbGlrZToKCi0gRW5kcG9pbnQgbmFtZQotIFBhcmFtZXRlcnMKLSBSZXF1ZXN0IEJvZHkKCllvdSBtYXkgYWRkIG1vcmUgaW5mb3JtYXRpb24gdG8geW91ciBlbmRwb2ludCBsaWtlIGZvbGxvd3M6CgpgYGBjKysKRU5EUE9JTlRfSU5GTyhjcmVhdGVVc2VyKSB7CiAgaW5mby0+c3VtbWFyeSA9ICJDcmVhdGUgbmV3IFVzZXIiOwogIGluZm8tPmFkZENvbnN1bWVzPFVzZXJEdG8+KCJhcHBsaWNhdGlvbi9qc29uIik7CiAgaW5mby0+YWRkUmVzcG9uc2U8VXNlckR0bz4oU3RhdHVzOjpDT0RFXzIwMCwgImFwcGxpY2F0aW9uL2pzb24iKTsKICBpbmZvLT5hZGRSZXNwb25zZShTdGF0dXM6OkNPREVfNTAwKTsKfQpFTkRQT0lOVCgiUE9TVCIsICJkZW1vL2FwaS91c2VycyIsIGNyZWF0ZVVzZXIsCiAgICAgICAgIEJPRFlfRFRPKFVzZXJEdG8sIHVzZXJEdG8pKSB7CiAgcmV0dXJuIGNyZWF0ZUR0b1Jlc3BvbnNlKFN0YXR1czo6Q09ERV8yMDAsIG1fZGF0YWJhc2UtPmNyZWF0ZVVzZXIodXNlckR0bykpOwp9CmBgYAoKIyMjIEhvdyB0byBhZGQgU3dhZ2dlciBVSSB0byB5b3VyIHByb2plY3QKCjEpIEFkZCBgYGBvYXRwcDo6c3dhZ2dlcjo6RG9jdW1lbnRJbmZvYGBgIGFuZCBgYGBvYXRwcDo6c3dhZ2dlcjo6UmVzb3VyY2VzYGBgIGNvbXBvbmVudHMgdG8geW91ciBBcHBDb21wb25lbnRzOgoKYGBgYysrCi8qKgogKiAgR2VuZXJhbCBBUEkgZG9jcyBpbmZvCiAqLwpPQVRQUF9DUkVBVEVfQ09NUE9ORU5UKHN0ZDo6c2hhcmVkX3B0cjxvYXRwcDo6c3dhZ2dlcjo6RG9jdW1lbnRJbmZvPiwgc3dhZ2dlckRvY3VtZW50SW5mbykoW10gewoKICBvYXRwcDo6c3dhZ2dlcjo6RG9jdW1lbnRJbmZvOjpCdWlsZGVyIGJ1aWxkZXI7CgogIGJ1aWxkZXIKICAuc2V0VGl0bGUoIlVzZXIgZW50aXR5IHNlcnZpY2UiKQogIC5zZXREZXNjcmlwdGlvbigiQ1JVRCBBUEkgRXhhbXBsZSBwcm9qZWN0IHdpdGggc3dhZ2dlciBkb2NzIikKICAuc2V0VmVyc2lvbigiMS4wIikKICAuc2V0Q29udGFjdE5hbWUoIkl2YW4gT3ZzeWFub2Noa2EiKQogIC5zZXRDb250YWN0VXJsKCJodHRwczovL29hdHBwLmlvLyIpCgogIC5zZXRMaWNlbnNlTmFtZSgiQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wIikKICAuc2V0TGljZW5zZVVybCgiaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wIikKCiAgLmFkZFNlcnZlcigiaHR0cDovL2xvY2FsaG9zdDo4MDAwIiwgInNlcnZlciBvbiBsb2NhbGhvc3QiKTsKCiAgLy8gV2hlbiB5b3UgYXJlIHVzaW5nIHRoZSBBVVRIRU5USUNBVElPTigpIEVuZHBvaW50LU1hY3JvIHlvdSBtdXN0IGFkZCBhbiBTZWN1cml0eVNjaGVtZSBvYmplY3QgKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLyNzZWN1cml0eVNjaGVtZU9iamVjdCkKICAvLyBGb3IgYmFzaWMtYXV0aGVudGljYXRpb24geW91IGNhbiB1c2UgdGhlIGRlZmF1bHQgQmFzaWMtQXV0aG9yaXphdGlvbi1TZWN1cml0eS1TY2hlbWUgbGlrZSB0aGlzCiAgLy8gRm9yIG1vcmUgY29tcGxleCBhdXRoZW50aWNhdGlvbiBzY2hlbWVzIHlvdSBjYW4gdXNlIHRoZSBvYXRwcDo6c3dhZ2dlcjo6RG9jdW1lbnRJbmZvOjpTZWN1cml0eVNjaGVtZUJ1aWxkZXIgYnVpbGRlcgogIC8vIERvbid0IGZvcmdldCB0byBhZGQgaW5mby0+YWRkU2VjdXJpdHlSZXF1aXJlbWVudCgiYmFzaWNfYXV0aCIpIHRvIHlvdXIgRU5EUE9JTlRfSU5GTygpIE1hY3JvIQogIC5hZGRTZWN1cml0eVNjaGVtZSgiYmFzaWNfYXV0aCIsIG9hdHBwOjpzd2FnZ2VyOjpEb2N1bWVudEluZm86OlNlY3VyaXR5U2NoZW1lQnVpbGRlcjo6RGVmYXVsdEJhc2ljQXV0aG9yaXphdGlvblNlY3VyaXR5U2NoZW1lKCkpOwoKICByZXR1cm4gYnVpbGRlci5idWlsZCgpOwoKfSgpKTsKCgovKioKICogIFN3YWdnZXItVWkgUmVzb3VyY2VzICg8b2F0cHAtZXhhbXBsZXM+L2xpYi9vYXRwcC1zd2FnZ2VyL3JlcykKICovCk9BVFBQX0NSRUFURV9DT01QT05FTlQoc3RkOjpzaGFyZWRfcHRyPG9hdHBwOjpzd2FnZ2VyOjpSZXNvdXJjZXM+LCBzd2FnZ2VyUmVzb3VyY2VzKShbXSB7CiAgLy8gTWFrZSBzdXJlIHRvIHNwZWNpZnkgY29ycmVjdCBmdWxsIHBhdGggdG8gb2F0cHAtc3dhZ2dlci9yZXMgZm9sZGVyICEhIQogIHJldHVybiBvYXRwcDo6c3dhZ2dlcjo6UmVzb3VyY2VzOjpsb2FkUmVzb3VyY2VzKCI8WU9VUi1QQVRILVRPLVJFUE8+L2xpYi9vYXRwcC1zd2FnZ2VyL3JlcyIpOwp9KCkpOwoKYGBgCgoyKSBDcmVhdGUgYGBgb2F0cHA6OnN3YWdnZXI6OkNvbnRyb2xsZXJgYGAgd2l0aCBsaXN0IG9mIGVuZHBvaW50cyB5b3Ugd2hhbnQgdG8gZG9jdW1lbnQgYW5kIGFkZCBpdCB0byByb3V0ZXI6CgpgYGBjKysKYXV0byBzd2FnZ2VyQ29udHJvbGxlciA9IG9hdHBwOjpzd2FnZ2VyOjpDb250cm9sbGVyOjpjcmVhdGVTaGFyZWQoPGxpc3Qtb2YtZW5kcG9pbnRzLXRvLWRvY3VtZW50Pik7CnN3YWdnZXJDb250cm9sbGVyLT5hZGRFbmRwb2ludHNUb1JvdXRlcihyb3V0ZXIpOwpgYGAKCjMpIGNtYWtlIDogIAoKQWRkIHRoZSBmb2xsb3dpbmcgbGluZXMgdG8geW91ciBDTWFrZUxpc3RzLnR4dCBwcm9qZWN0IGZpbGUKYGBgCgpmaW5kX3BhY2thZ2Uob2F0cHAgMS4zLjAgUkVRVUlSRUQpCmlmKG9hdHBwX0ZPVU5EKQogIG1lc3NhZ2UoU1RBVFVTICJGb3VuZCBvYXRwcCB2ZXJzaW9uOiAke29hdHBwX1ZFUlNJT05fU1RSSU5HfSIpCmVsc2UoKQogIG1lc3NhZ2UoRkFUQUxfRVJST1IgIkNvdWxkIG5vdCBmaW5kIG9hdHBwIikKZW5kaWYoKQoKZmluZF9wYWNrYWdlKG9hdHBwLXN3YWdnZXIgIDEuMy4wIFJFUVVJUkVEKQppZihvYXRwcC1zd2FnZ2VyX0ZPVU5EKQogIG1lc3NhZ2UoU1RBVFVTICJGb3VuZCBvYXRwcC1zd2FnZ2VyIHZlcnNpb246ICR7b2F0cHAtc3dhZ2dlcl9WRVJTSU9OX1NUUklOR30iKQplbHNlKCkKICBtZXNzYWdlKEZBVEFMX0VSUk9SICJDb3VsZCBub3QgZmluZCBvYXRwcC1zd2FnZ2VyIikKZW5kaWYoKQoKaW5jbHVkZV9kaXJlY3Rvcmllcygke29hdHBwX0lOQ0xVREVfRElSU30pCmluY2x1ZGVfZGlyZWN0b3JpZXMoJHtvYXRwcC1zd2FnZ2VyX0lOQ0xVREVfRElSU30pCgphZGRfZGVmaW5pdGlvbnMoIAogIC1ET0FUUFBfU1dBR0dFUl9SRVNfUEFUSD0iJHtPQVRQUF9CQVNFX0RJUn0vYmluL29hdHBwLXN3YWdnZXIvcmVzIgopCgp0YXJnZXRfbGlua19saWJyYXJpZXMgKHByb2plY3QgUFVCTElDCiAgIC8vIGFkZF95b3VyIGxpYnJhcmllcyBoZXJlCiAgIFBVQkxJQyBvYXRwcDo6b2F0cHAKICAgUFVCTElDIG9hdHBwOjpvYXRwcC1zd2FnZ2VyCikKYGBgCgojIyMgQ3VzdG9taXNlIFN3YWdnZXIgVUkgUGF0aHMKClRvIGN1c3RvbWlzZSBzd2FnZ2VyIFVJIGVuZHBvaW50cyBwYXRocyBhZGQgdGhlIGZvbGxvd2luZyBjb21wb25lbnQ6CgpgYGBjKysKICAvKioKICAgKiAgU3dhZ2dlciBDb250cm9sbGVyIFBhdGhzCiAgICovCiAgT0FUUFBfQ1JFQVRFX0NPTVBPTkVOVChzdGQ6OnNoYXJlZF9wdHI8b2F0cHA6OnN3YWdnZXI6OkNvbnRyb2xsZXJQYXRocz4sIGNvbnRyb2xsZXJQYXRocykoW10gewogICAgYXV0byBwYXRocyA9IHN0ZDo6bWFrZV9zaGFyZWQ8b2F0cHA6OnN3YWdnZXI6OkNvbnRyb2xsZXJQYXRocz4oKTsKICAgIHBhdGhzLT5hcGlKc29uID0gImN1c3RvbS9wYXRoL2Zvci9hcGkuanNvbiI7ICAgICAgIC8vIGRlZmF1bHQgaXMgImFwaS1kb2NzL29hcy0zLjAuMC5qc29uIgogICAgcGF0aHMtPnVpID0gIm15L2N1c3RvbS9wYXRoL3N3YWdnZXItdWkiOyAgICAgICAgICAgLy8gZGVmYXVsdCBpcyAic3dhZ2dlci91aSIKICAgIHBhdGhzLT51aVJlc291cmNlcyA9ICJteS9jdXN0b20vcGF0aC97ZmlsZW5hbWV9IjsgIC8vIGRlZmF1bHQgaXMgInN3YWdnZXIve2ZpbGVuYW1lfSIKICAgIHJldHVybiBwYXRoczsKICB9KCkpOwpgYGAKCioqTk9URToqKiBgcGF0aHMtPnVpYCBhbmQgYHBhdGhzLT51aVJlc291cmNlc2AgTVVTVCBoYXZlIHRoZSBzYW1lIGJhc2UgcGF0aCAtIGFzIHNob3duIGFib3ZlLgoKKipEb25lISoqCg== readmeEtag: '"273b24fa1dc9545d778a601ef94b676b6e4d0998"' readmeLastModified: Tue, 04 Jun 2024 23:13:23 GMT repositoryId: 142623136 description: OpenApi 3.0.0 docs + Swagger UI for oatpp services created: '2018-07-27T20:59:35Z' updated: '2026-01-20T02:10:10Z' language: C++ archived: false stars: 106 watchers: 7 forks: 66 owner: oatpp logo: https://avatars.githubusercontent.com/u/37167448?v=4 license: Apache-2.0 repoEtag: '"caa9d9bb72938fc9afcf30eb5b78fc2f28981ed3dd6cdf227c660fae413bce72"' repoLastModified: Tue, 20 Jan 2026 02:10:10 GMT foundInMaster: true category: Low-level Tooling id: e2b5ed322cb3283e7ad1edd0b0634fc4 - source: openapi3 tags repository: https://github.com/cdwv/oas3-api-snippet-enricher v3: true repositoryMetadata: base64Readme: >- IyBFbnJpY2ggeW91ciBPcGVuQVBJIDMuMCBzY2hlbWEgd2l0aCBleGFtcGxlcwoKVGhhbmtzIHRvIHRoZSB3b25kZXJmdWwgW3N3YWdnZXItc25pcHBldF0oaHR0cHM6Ly9naXRodWIuY29tL0VyaWtXaXR0ZXJuL3N3YWdnZXItc25pcHBldCkgbW9kdWxlIHlvdSBjYW4gbm93IHNpbXBseSBlbnJpY2ggeW91ciBPcGVuQVBJIHNjaGVtYSB3aXRoIGNvZGUgc2FtcGxlcy4gSXQncyBhcyBlYXN5IGFzIDEuMi4zLgoKMS4gYGBucG0gaW5zdGFsbCBzbmlwcGV0LWVucmljaGVyLWNsaWBgCjIuIGBgLi9ub2RlX21vZHVsZXMvLmJpbi9zbmlwcGV0LWVucmljaGVyLWNsaSAtLWlucHV0PXlvdXJfb2FzLmpzb25gYAoKIyMgRXhhbXBsZSBVc2FnZQoKRW5yaWNoIHlvdXIgT0FTIDMuMCBTY2hlbWEKYGBgCi4vbm9kZV9tb2R1bGVzLy5iaW4vc25pcHBldC1lbnJpY2hlci1jbGkgLS1pbnB1dD1vcGVuYXBpLmpzb24gPiBvcGVuYXBpLXdpdGgtZXhhbXBsZXMuanNvbgpgYGAKQWx0ZXJuYXRpdmVseSB5b3UgY2FuIHBvaW50IGl0IHRvIGEgWUFNTC1mb3JtYXR0ZWQgc3BlYzoKYGBgCmN1cmwgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFzdGVyL2V4YW1wbGVzL3YzLjAvcGV0c3RvcmUueWFtbCAtLW91dHB1dCBwZXRzdG9yZS55YW1sCi4vbm9kZV9tb2R1bGVzLy5iaW4vc25pcHBldC1lbnJpY2hlci1jbGkgLS1pbnB1dD1wZXRzdG9yZS55YW1sID4gb3BlbmFwaS13aXRoLWV4YW1wbGVzLmpzb24KYGBgCgpVc2UgdGFyZ2V0cyBvcHRpb25zIHRvIHNwZWNpZmljIGxhbmd1YWdlczoKYGBgCi4vbm9kZV9tb2R1bGVzLy5iaW4vc25pcHBldC1lbnJpY2hlci1jbGkgLS10YXJnZXRzPSJub2RlX3JlcXVlc3Qsc2hlbGxfY3VybCIgLS1pbnB1dD1vcGVuYXBpLmpzb24gPiBvcGVuYXBpLXdpdGgtZXhhbXBsZXMuanNvbgpgYGAKClVzZSBbUmVEb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jLykgdG8gYnVpbGQgYmVhdXRpZnVsIEFQSSBkb2M6CmBgYApyZWRvYy1jbGkgYnVuZGxlIG9wZW5hcGktd2l0aC1leGFtcGxlcy5qc29uCmBgYAoKZW5qb3kuCgohW1JlRG9jIEFQSSBkb2N1bWVudGF0aW9uIHdpdGggY29kZSBzYW1wbGVzXShpbWFnZS5wbmcpCgpDb250cmlidXRpbmcKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCkNvbnRyaWJ1dGlvbnMgYXJlIG1vc3Qgd2VsY29tZSEKCgpMaWNlbnNlCj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CgpNSVQKCk1haW50YWluZXJzCj09PT09PT09PT09CgpbPGltZyB3aWR0aD0iMzAwIiB0aXRsZT0iQ29kZXdhdmUuZXUiIHNyYz0iY2R3di1sb2dvLW5ldy5zdmciPl0oaHR0cDovL2NvZGV3YXZlLmV1KQoKUHJvamVjdCBpcyBjdXJyZW50bHkgbWFpbnRhaW5lZCwgaW4gb3VyIHNwYXJlIHRpbWUsIGJ5IFtjb2Rld2F2ZS5ldV0oaHR0cDovL2NvZGV3YXZlLmV1KSBhbmQgYSBncm93aW5nIG51bWJlciBvZiBDb250cmlidXRvcnMhCg== readmeEtag: '"2e480b8b5b9a3c8516169de7cef40e16dade1c06"' readmeLastModified: Mon, 11 Dec 2023 12:08:59 GMT repositoryId: 220618185 description: Enrich your OpenAPI 3.0 JSON with code samples created: '2019-11-09T09:13:16Z' updated: '2025-12-19T14:23:25Z' language: JavaScript archived: false stars: 97 watchers: 8 forks: 14 owner: cdwv logo: https://avatars.githubusercontent.com/u/6742687?v=4 license: MIT repoEtag: '"07f7ab831d4772605944d76b1543ab16c3be38090dabb5f8cb226bfbdcf2de06"' repoLastModified: Fri, 19 Dec 2025 14:23:25 GMT foundInMaster: true category: - Documentation - Parsers id: 839bb72e7d7b73d50edcb2723923e0e2 - source: openapi3 tags repository: https://github.com/koumoul-dev/vue-openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHZpZXdlciBjb21wb25lbnQgZm9yIFZ1ZUpTCgpUaGlzIFtWdWUuanNdKGh0dHBzOi8vdnVlanMub3JnLykgY29tcG9uZW50IGlzIGRlc2lnbmVkIHRvIGVhc2lseSBicm93c2UgYW5kIHRlc3QgYSBSRVNUIEFQSSBkZXNjcmliZWQKd2l0aCB0aGUgW09wZW5BUEkgMy4wIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKSAoZm9ybWVybHkga25vd24gYXMgU3dhZ2dlciBTcGVjaWZpY2F0aW9uKS4gVGhpcyBjb21wb25lbnQgZm9sbG93cyBHb29nbGUgW01hdGVyaWFsIERlc2lnbl0oaHR0cHM6Ly9tYXRlcmlhbC5nb29nbGUuY29tLykgcHJpbmNpcGxlcyBhbmQgcmVsaWVzIG9uIHRoZSBbVnVlIE1hdGVyaWFsXShodHRwczovL2dpdGh1Yi5jb20vbWFyY29zbW91cmEvdnVlLW1hdGVyaWFsKSBmcmFtZXdvcmsuIEl0IGFsc28gcmVsaWVzIG9uIFt2dWUtcmVzb3VyY2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9wYWdla2l0L3Z1ZS1yZXNvdXJjZSkgdG8gcGVyZm9ybSBBUEkgcmVxdWVzdHMuCgpTZWUgaXQgaW4gYWN0aW9uIDoKICogaHR0cHM6Ly9rb3Vtb3VsLmNvbS9vcGVuYXBpLXZpZXdlci8KCiMjIEluc3RhbGwKCm5wbSA6CmBgYGNvbnNvbGUKJCBucG0gaW5zdGFsbCAtLXNhdmUgdnVlLW9wZW5hcGkKYGBgCgojIyBVc2FnZQpgYGBqcwppbXBvcnQgVnVlIGZyb20gJ3Z1ZScKaW1wb3J0IFZ1ZU1hdGVyaWFsIGZyb20gJ3Z1ZS1tYXRlcmlhbCcKaW1wb3J0IE9wZW5BcGkgZnJvbSAndnVlLW9wZW5hcGknCmltcG9ydCAndnVlLW1hdGVyaWFsL2Rpc3QvdnVlLW1hdGVyaWFsLmNzcycKaW1wb3J0IFZ1ZVJlc291cmNlIGZyb20gJ3Z1ZS1yZXNvdXJjZScKCmltcG9ydCBqc29uQXBpIGZyb20gJy4vc3dhZ2dlci5qc29uJwoKVnVlLnVzZShWdWVNYXRlcmlhbCkKVnVlLnVzZShWdWVSZXNvdXJjZSkKCm5ldyBWdWUoewogIGVsOiAnI2FwcCcsCiAgdGVtcGxhdGU6ICc8b3Blbi1hcGkgdi1pZj0ianNvbkFwaSIgOmFwaT0ianNvbkFwaSIgbWQtdGhlbWU9IlwnZGVmYXVsdFwnIiA6cXVlcnktcGFyYW1zPSJxdWVyeVBhcmFtcyIgOmhlYWRlcnM9ImhlYWRlcnMiPjwvb3Blbi1hcGk+JywKICBkYXRhOiAoKSA9PiAoewogICAganNvbkFwaToganNvbkFwaSwKICAgIHF1ZXJ5UGFyYW1zOiB7CiAgICAgIHVzZXJJZDogJ2pvaG5fZG9lJwogICAgfSwKICAgIGhlYWRlcnM6IHsKICAgICAgYXBpX2tleTogJ215X2FwaV9rZXknCiAgICB9CiAgfSksCiAgY29tcG9uZW50czogewogICAgT3BlbkFwaQogIH0KfSkKYGBgCgojIyBEZXZlbG9wCgpSdW4gd2VicGFjayBpbiB3YXRjaCBtb2RlOgoKICAgIG5wbSBydW4gZGV2CgpUaGVuIG9wZW4gdGVzdC9pbmRleC5odG1sIGluIHlvdXIgYnJvd3Nlci4KClRvIHN3aXRjaCBiZXR3ZWVuIGV4YW1wbGVzLCBtb2RpZnkgdGhlIGltcG9ydCAianNvbkFwaSIgaW4gdGVzdC9hcHAuanMuCgojIyBMaWNlbnNlCgpbTUlUIExpY2Vuc2VdKGxpY2Vuc2UubWQpCgojIyBSZXNvdXJjZXMKCiogW0F3ZXNvbWUgT3BlbkFwaSAzXShodHRwczovL2dpdGh1Yi5jb20vTWVybWFkZS9hd2Vzb21lLW9wZW5hcGkzKSBsaXN0cyB2YXJpb3VzIHByb2plY3RzIHJlbGF0ZWQgdG8gT3BlbkFwaSAzLjAueAoqIFtzd2FnZ2VyMm9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL3N3YWdnZXIyb3BlbmFwaSkgbGV0cyB5b3UgY29udmVydCBTd2FnZ2VyIDIuMCBkZWZpbml0aW9ucyBpbnRvIE9wZW5BcGkgMy4wLngKCiMjIFNpbWlsYXIgcHJvamVjdHMKClRoaXMgcHJvamVjdCBoYXMgYmVlbiBpbnNwaXJlZCBieSB0aGUgZm9sbG93aW5nIHByb2plY3RzIDoKCiAqIFthbmd1bGFyLXN3YWdnZXItdWldKGh0dHBzOi8vZ2l0aHViLmNvbS9PcmFuZ2UtT3BlblNvdXJjZS9hbmd1bGFyLXN3YWdnZXItdWkpCiAqIFthbmd1bGFyLXN3YWdnZXItdWktbWF0ZXJpYWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9kYXJvc2gvYW5ndWxhci1zd2FnZ2VyLXVpLW1hdGVyaWFsKQo= readmeEtag: '"3533626f90ee3a6fc8cb8614676b12dcd9d4f1b7"' readmeLastModified: Wed, 06 Dec 2023 10:39:08 GMT repositoryId: 75237921 description: OpenAPI viewer component for VueJS created: '2016-11-30T23:56:46Z' updated: '2025-07-21T13:35:41Z' language: Vue archived: false stars: 95 watchers: 7 forks: 19 owner: koumoul-dev logo: https://avatars.githubusercontent.com/u/16051219?v=4 license: MIT repoEtag: '"5a35eb99ace8d18a67da7cf6ff6013ba75c0c26f1806f462e0f77d0b6220ed69"' repoLastModified: Mon, 21 Jul 2025 13:35:41 GMT foundInMaster: true category: Parsers id: 016162b2c3c0b0a72ac73ea4566276c8 - source: openapi3 tags repository: https://github.com/data-fair/openapi-viewer v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHZpZXdlcgoKVGhpcyBzZXJ2aWNlIGlzIGRlc2lnbmVkIHRvIGVhc2lseSBicm93c2UgYW5kIHRlc3QgYSBSRVNUIEFQSSBkZXNjcmliZWQgd2l0aCB0aGUgW09wZW5BUEkgMy4wIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKS4KCiMjIFNwb25zb3JzCgp8IHwgQ2xpY2sgW2hlcmUgdG8gc3VwcG9ydCB0aGUgZGV2ZWxvcG1lbnQgb2YgdGhpcyBwcm9qZWN0XShodHRwczovL2dpdGh1Yi5jb20vc3BvbnNvcnMva291bW91bC1kZXYpLiB8CnwtfC18CnwgIVtLb3Vtb3VsIGxvZ29dKGh0dHBzOi8va291bW91bC5jb20vc3RhdGljL2xvZ28tc2xvZ2FuLnBuZykgfCBbS291bW91bF0oaHR0cHM6Ly9rb3Vtb3VsLmNvbSkgZGV2ZWxvcHMgdGhlIERhdGEgRmFpciBlY29zeXN0ZW0gYW5kIGhvc3RzIGl0IGFzIGFuIG9ubGluZSBzZXJ2aWNlLiB8CnwgIVtEYXdpenogbG9nb10oaHR0cHM6Ly9kYXdpenouZnIvbG9nby1EYXdpenotYWxsLWFib3V0LXlvdXItZGF0YS1ob21lLnBuZykgfCBbRGF3aXp6XShodHRwczovL2Rhd2l6ei5mcikgdXNlcyB0aGUgRGF0YSBGYWlyIGVjb3N5c3RlbSBpbnNpZGUgaXRzIHBsYXRmb3JtIGFuZCBzdXBwb3J0cyBpdHMgZGV2ZWxvcG1lbnQuIHwKCiMjIFBhcmFtZXRlcnMKCiMjIyBRdWVyeSBQYXJhbWV0ZXJzCgotICoqZHJhd2VyTG9jYXRpb24qKiA6IFRoZSBsb2NhdGlvbiBvZiB0aGUgbmF2aWdhdGlvbi1kcmF3ZXIuIENhbiBiZSBgbGVmdGAgb3IgYHJpZ2h0YC4gRGVmYXVsdHMgdG8gYGxlZnRgLgotICoqdXJsVHlwZSoqIDogVGhlIHR5cGUgb2YgdGhlIFVSTCBkZWZpbmVkIGluIHRoZSBlbnZpcm9ubWVudCB2YXJpYWJsZSBgQUxMT1dFRF9VUkxTYC4KLSAqKm9wZXJhdGlvbioqIDogVGhlIGBvcGVyYXRpb25JZGAgb2YgdGhlIG9wZXJhdGlvbiB0byBuYXZpZ2F0ZSB0by4gW1JlZmVyZW5jZSA6IE9wZW5BUEkgMy4xIE9wZXJhdGlvbiBPYmplY3RdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCNvcGVyYXRpb24tb2JqZWN0KQoKPiDimqDvuI8gKipEZXByZWNhdGVkKio6Cj4KPiAtIH5+Kip1cmwqKiA6IFRoZSBsb2NhdGlvbiBvZiB0aGUgQVBJIGRvY3VtZW50YXRpb24gZmlsZSB0byBsb2FkLCBpbiBPcGVuQVBJIHYzIEpTT04gZm9ybWF0Ln5+ICpSZXBsYWNlZCBieSB0aGUgYHVybFR5cGVgIHF1ZXJ5IHBhcmFtZXRlciBpbnN0ZWFkIGFuZCB0aGUgYEFMTE9XRURfVVJMU2AgZW52aXJvbm1lbnQgdmFyaWFibGUuKgo+IC0gfn4qKmhpZGUtdG9vbGJhcioqIDogYHRydWVgIG9yIGBmYWxzZWAgdG8gaGlkZSB0aGUgdG9vbGJhciAodXNlZnVsIGZvciBpZnJhbWUgaW50ZWdyYXRpb24pLiBEZWZhdWx0cyB0byBgZmFsc2VgLn5+CgojIyMgRW52aXJvbm1lbnQgVmFyaWFibGVzCgotICoqVVNFX1NJTVBMRV9ESVJFQ1RPUlkqKiA6IEEgYm9vbGVhbiB0byBlbmFibGUgaW50ZWdyYXRpb24gd2l0aCAqKltTaW1wbGVEaXJlY3RvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9kYXRhLWZhaXIvc2ltcGxlLWRpcmVjdG9yeSkqKi4gRGVmYXVsdHMgdG8gYGZhbHNlYC4KCi0gKipERUZBVUxUX1VSTCoqIDogVGhlIGRlZmF1bHQgVVJMIHRvIGxvYWQgd2hlbiBubyBgdXJsVHlwZWAgcXVlcnkgcGFyYW1ldGVyIGlzIHByb3ZpZGVkLgoKLSAqKkFMTE9XRURfVVJMUyoqIDogQSBKU09OIG9iamVjdCBjb250YWluaW5nIGEgbGlzdCBvZiBhbGxvd2VkIFVSTHMuIFRoZSBrZXlzIGFyZSB1c2VkIGFzIHRoZSBgdXJsVHlwZWAgcXVlcnkgcGFyYW1ldGVyLiBUaGUgdmFsdWVzIGFyZSB0aGUgVVJMcyB0byB0aGUgT3BlbkFQSSBmaWxlcy4gVGhlc2UgVVJMcyBjYW4gY29udGFpbiBwbGFjZWhvbGRlcnMgZW5jbG9zZWQgaW4gY3VybHkgYnJhY2VzIHt9LCB3aGljaCB3aWxsIGJlIHJlcGxhY2VkIGJ5IHRoZSBjb3JyZXNwb25kaW5nIHF1ZXJ5IHBhcmFtZXRlcnMuIFRoZSBwbGFjZWhvbGRlcnMgYXJlIGRlZmluZWQgYnkgdGhlIGtleXMgb2YgdGhlIHF1ZXJ5IHBhcmFtZXRlcnMuIERlZmF1bHRzIHRvIGFuIGVtcHR5IG9iamVjdC4KCj4gKkV4YW1wbGVzIG9mIEFMTE9XRURfVVJMUyA6Kgo+Cj4gYGBganNvbgo+IHsKPiAgICJleGFtcGxlWWFtbCI6ICJodHRwczovL2V4YW1wbGUxLmNvbS9vcGVuYXBpLnlhbWwiLAo+ICAgImV4YW1wbGVXaXRoVXJsVGVtcGxhdGUiOiAiaHR0cHM6Ly9leGFtcGxlMi5jb20ve2lkfS9vcGVuYXBpLmpzb24iCj4gfQo+IGBgYAoKIyMgSW50ZWdyYXRpb24gd2l0aCBTaW1wbGVEaXJlY3RvcnkKCkJ5IGRlZmF1bHQsIHRoZSB2aWV3ZXIgcnVucyBpbiAqKnN0YW5kYWxvbmUgbW9kZSoqIGFuZCBkb2VzIG5vdCBpbnRlZ3JhdGUgd2l0aCAqKltTaW1wbGVEaXJlY3RvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9kYXRhLWZhaXIvc2ltcGxlLWRpcmVjdG9yeSkqKi4gIApIb3dldmVyLCB5b3UgY2FuIGVuYWJsZSBpbnRlZ3JhdGlvbiB0byBiZW5lZml0IGZyb206CgotICoqVGhlbWluZyoqOiBBdXRvbWF0aWNhbGx5IGFkYXB0cyB0byB0aGUgcGxhdGZvcm0ncyBsb29rIGFuZCBmZWVsLgotICoqTGFuZ3VhZ2UqKjogVXNlcyB0aGUgc2FtZSBsYW5ndWFnZSBzZXR0aW5ncyBhcyB0aGUgZW52aXJvbm1lbnQuCgpUbyBlbmFibGUgaW50ZWdyYXRpb24gd2l0aCBTaW1wbGVEaXJlY3RvcnksIHNldCB0aGUgZW52aXJvbm1lbnQgdmFyaWFibGU6CgpgYGBiYXNoClVTRV9TSU1QTEVfRElSRUNUT1JZPXRydWUKYGBgCgojIyBEZXZlbG9wZXJzCgpUYWtlIGEgbG9vayBhdCB0aGUgW2NvbnRyaWJ1dGlvbiBndWlkZWxpbmVzXSguL0NPTlRSSUJVVElORy5tZCkuCg== readmeEtag: '"23e945a14de7ab424221132f20647e5cd88768ae"' readmeLastModified: Thu, 20 Mar 2025 13:59:20 GMT repositoryId: 84982324 description: Browse and test a REST API described with the OpenAPI 3.0 Specification created: '2017-03-14T18:12:56Z' updated: '2025-11-10T14:55:00Z' language: Vue archived: false stars: 94 watchers: 7 forks: 12 owner: data-fair logo: https://avatars.githubusercontent.com/u/83776320?v=4 license: MIT repoEtag: '"708f964483858dba587b1b25b95c60f36e7156854a0ea4a76035a3efc35bd12f"' repoLastModified: Mon, 10 Nov 2025 14:55:00 GMT foundInMaster: true category: - Documentation - Parsers id: c84aa3074b4ed8af1ecf41bafb7964d4 oldLocations: - https://github.com/koumoul-dev/openapi-viewer - source: openapi3 tags repository: https://github.com/jellyfin/jellyfin-sdk-kotlin v3: true repositoryMetadata: base64Readme: >- PCEtLSBtYXJrZG93bmxpbnQtZGlzYWJsZSBNRDAzMyBuby1pbmxpbmUtaHRtbCAtLT4KPGgxIGFsaWduPSJjZW50ZXIiPkplbGx5ZmluIEtvdGxpbiBTREs8L2gxPgo8aDMgYWxpZ249ImNlbnRlciI+UGFydCBvZiB0aGUgPGEgaHJlZj0iaHR0cHM6Ly9qZWxseWZpbi5vcmcvIj5KZWxseWZpbiBQcm9qZWN0PC9hPjwvaDM+CgotLS0KCjxwIGFsaWduPSJjZW50ZXIiPgo8aW1nIGFsdD0iTG9nbyBCYW5uZXIiIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2plbGx5ZmluL2plbGx5ZmluLXV4L21hc3Rlci9icmFuZGluZy9TVkcvYmFubmVyLWxvZ28tc29saWQuc3ZnP3Nhbml0aXplPXRydWUiLz4KPGJyLz4KPGJyLz4KPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2plbGx5ZmluL2plbGx5ZmluLXNkay1rb3RsaW4iPgo8aW1nIGFsdD0iTEdQTCAzLjAgbGljZW5zZSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL2plbGx5ZmluL2plbGx5ZmluLXNkay1rb3RsaW4uc3ZnIi8+CjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2plbGx5ZmluL2plbGx5ZmluLXNkay1rb3RsaW4vcmVsZWFzZXMiPgo8aW1nIGFsdD0iQ3VycmVudCBSZWxlYXNlIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3JlbGVhc2UvamVsbHlmaW4vamVsbHlmaW4tc2RrLWtvdGxpbi5zdmciLz4KPC9hPgo8YSBocmVmPSJodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9b3JnLmplbGx5ZmluLnNkayI+CjxpbWcgYWx0PSJNYXZlbiBDZW50cmFsIFJlbGVhc2UiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3Yvb3JnLmplbGx5ZmluLnNkay9qZWxseWZpbi1jb3JlLnN2ZyIvPgo8L2E+Cjxici8+CjxhIGhyZWY9Imh0dHBzOi8vb3BlbmNvbGxlY3RpdmUuY29tL2plbGx5ZmluIj4KPGltZyBhbHQ9IkRvbmF0ZSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL29wZW5jb2xsZWN0aXZlL2FsbC9qZWxseWZpbi5zdmc/bGFiZWw9YmFja2VycyIvPgo8L2E+CjxhIGhyZWY9Imh0dHBzOi8vbWF0cml4LnRvLyMvK2plbGx5ZmluLWFuZHJvaWQtZGV2Om1hdHJpeC5vcmciPgo8aW1nIGFsdD0iQ2hhdCBvbiBNYXRyaXgiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXRyaXgvamVsbHlmaW4tYW5kcm9pZC1kZXY6bWF0cml4Lm9yZy5zdmc/bG9nbz1tYXRyaXgiLz4KPC9hPgo8YSBocmVmPSJodHRwczovL3d3dy5yZWRkaXQuY29tL3IvamVsbHlmaW4iPgo8aW1nIGFsdD0iSm9pbiBvdXIgU3VicmVkZGl0IiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcmVkZGl0LXIlMkZqZWxseWZpbi0lMjNGRjU3MDAuc3ZnIi8+CjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2plbGx5ZmluL2plbGx5ZmluLXNkay1rb3RsaW4vcmVsZWFzZXMuYXRvbSI+CjxpbWcgYWx0PSJSZWxlYXNlIFJTUyBGZWVkIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcnNzLXJlbGVhc2VzLWZmYTUwMD9sb2dvPXJzcyIgLz4KPC9hPgo8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vamVsbHlmaW4vamVsbHlmaW4tc2RrLWtvdGxpbi9jb21taXRzL21hc3Rlci5hdG9tIj4KPGltZyBhbHQ9Ik1hc3RlciBDb21taXRzIFJTUyBGZWVkIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcnNzLWNvbW1pdHMtZmZhNTAwP2xvZ289cnNzIiAvPgo8L2E+CjwvcD4KCi0tLQoKVGhlIEtvdGxpbiBTREsgZm9yIEplbGx5ZmluIGltcGxlbWVudHMgdGhlIEplbGx5ZmluIEFQSSB0byBlYXNpbHkgYWNjZXNzIHNlcnZlcnMuIEl0IGlzIGN1cnJlbnRseSBhdmFpbGFibGUKZm9yIHRoZSBKVk0gYW5kIEFuZHJvaWQuIERldmVsb3BlciBkb2N1bWVudGF0aW9uIGlzIGF2YWlsYWJsZSBhdCBba290bGluLXNkay5qZWxseWZpbi5vcmddLgoKW2tvdGxpbi1zZGsuamVsbHlmaW4ub3JnXTogaHR0cHM6Ly9rb3RsaW4tc2RrLmplbGx5ZmluLm9yZy9ndWlkZS9nZXR0aW5nLXN0YXJ0ZWQuaHRtbAoKIyMgQ29udHJpYnV0aW5nCgpXZSB3ZWxjb21lIGNvbnRyaWJ1dGlvbnMgdG8gdGhlIFNESy4gT3BlbiBhbiBpc3N1ZSBvciBhc2sgaW4gb3VyIG9mZmljaWFsIGNoYXRzIGlmIHlvdSBwbGFuIHRvIG1ha2UgYmlnZ2VyIGNoYW5nZXMuCgpUbyB2YWxpZGF0ZSBiaW5hcnkgY29tcGF0aWJpbGl0eSB3ZSB1c2UgdGhlIFtCaW5hcnkgY29tcGF0aWJpbGl0eSB2YWxpZGF0b3JdIHRvb2wgZnJvbSB0aGUgS290bGluIHRlYW0uIFdoZW4gY3JlYXRpbmcKcHVsbCByZXF1ZXN0cyB0aGUgYXBpIGZpbGVzIG5lZWQgdG8gYmUgdXBkYXRlZC4gVXNlIHRoZSBgYXBpRHVtcGAgR3JhZGxlIHRhc2sgdG8gZ2VuZXJhdGUgdGhlIGFwaSBmaWxlcy4gQWRkIHRoZSBjaGFuZ2VzCmZyb20gdGhpcyBjb21tYW5kIHRvIGEgc2VwYXJhdGUgY29tbWl0IHRvIG1ha2UgdGhlIHJldmlldyBwcm9jZXNzIGVhc2llci4KCltCaW5hcnkgY29tcGF0aWJpbGl0eSB2YWxpZGF0b3JdOiBodHRwczovL2dpdGh1Yi5jb20vS290bGluL2JpbmFyeS1jb21wYXRpYmlsaXR5LXZhbGlkYXRvcgoKIyMgVGVzdGluZwoKVGhlIFNESyBpbmNsdWRlcyB0d28gZXhhbXBsZSBwcm9qZWN0cywgdGhlIGtvdGxpbi1jbGkgYW5kIGphdmEtY2xpLCB0byB0ZXN0IHZhcmlvdXMgbGFyZ2VyIGZ1bmN0aW9ucyBsaWtlIHNlcnZlcgpkaXNjb3ZlcnkuIEJlc2lkZXMgdGhhdCB3ZSB1c2UgdW5pdCB0ZXN0cyB0byB0ZXN0IHNtYWxsZXIgY29tcG9uZW50cywgdGhlc2UgY2FuIGJlIGV4ZWN1dGVkIHdpdGggdGhlIGBhbGxUZXN0c2AgR3JhZGxlCnRhc2suIFdlIHJlY29tbWVuZCBhZGRpbmcgbmV3IHRlc3RzIGZvciBjaGFuZ2VzIHRvIHRoZQpjb2RlLgoKIyMjIFRlc3RpbmcgaW4gYXBwCgpJdCBpcyBhbHNvIHBvc3NpYmxlIHRvIHRlc3QgYSBuZXcgdmVyc2lvbiBvZiB0aGUgU0RLIGluIHlvdXIgb3duIGFwcC4gVXNlIHRoZSBgcHVibGlzaFRvTWF2ZW5Mb2NhbGAgR3JhZGxlIHRhc2sgdG8KcHVibGlzaCB0aGUgU0RLIHRvIHlvdXIgbG9jYWwgc3lzdGVtLCBhZnRlcndhcmRzIHlvdSBjYW4gYWRkIGBtYXZlbkxvY2FsKClgIGFzIHJlcG9zaXRvcnkgYW5kIHVzZSB0aGUgYGxhdGVzdC1TTkFQU0hPVGAKdmVyc2lvbiBmb3IgdGhlIFNESy4gVGhpcyBwcm9jZXNzIGlzIHNpbXBsaWZpZWQgaW4gb3VyIG9mZmljaWFsIGFwcHMgYnkgYWRkaW5nIGFuIG9wdGlvbiB0byB0aGUgYGdyYWRsZS5wcm9wZXJ0aWVzYApmaWxlLgo= readmeEtag: '"925febc521a5d268edef39455564ae08f6a8166c"' readmeLastModified: Mon, 05 Aug 2024 06:33:58 GMT repositoryId: 162836730 description: Kotlin SDK for Jellyfin, supporting Android and JVM targets created: '2018-12-22T19:14:06Z' updated: '2026-01-22T16:25:38Z' language: Kotlin archived: false stars: 105 watchers: 5 forks: 55 owner: jellyfin logo: https://avatars.githubusercontent.com/u/45698031?v=4 license: LGPL-3.0 repoEtag: '"63a4372670c7eb2383cfd7a1e8e9625d83810e11e8fce91fae11d67727c292f8"' repoLastModified: Thu, 22 Jan 2026 16:25:38 GMT foundInMaster: true category: Testing id: f51a5c8eb1a78ed2479cabdb83d894d5 - source: openapi3 tags repository: https://github.com/openapi-contrib/openapi3-generator v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPk9wZW5BUEkgMyBHZW5lcmF0b3I8L2gxPgo8cCBhbGlnbj0iY2VudGVyIj4KICBVc2UgeW91ciBBUEkgT3BlbkFQSSAzIGRlZmluaXRpb24gdG8gZ2VuZXJhdGUgY29kZSwgZG9jdW1lbnRhdGlvbiwgYW5kIGxpdGVyYWxseSBhbnl0aGluZyB5b3UgbmVlZC4KPC9wPgoKIyMgSW5zdGFsbAoKVG8gdXNlIGl0IGZyb20gdGhlIENMSToKCmBgYGJhc2gKbnBtIGluc3RhbGwgLWcgb3BlbmFwaTMtZ2VuZXJhdG9yCmBgYAoKIyMgUmVxdWlyZW1lbnRzCgoqIE5vZGUuanMgdjcuNisKCiMjIFVzYWdlCgojIyMgRnJvbSB0aGUgY29tbWFuZC1saW5lIGludGVyZmFjZSAoQ0xJKQoKYGBgYmFzaAogIFVzYWdlOiBvZyBbb3B0aW9uc10gPG9wZW5hcGlGaWxlT3JVUkw+IDx0ZW1wbGF0ZT4KCgogIE9wdGlvbnM6CgogICAgLVYsIC0tdmVyc2lvbiAgICAgICAgICAgICAgICAgIG91dHB1dCB0aGUgdmVyc2lvbiBudW1iZXIKICAgIC1vLCAtLW91dHB1dCA8b3V0cHV0RGlyPiAgICAgICBkaXJlY3Rvcnkgd2hlcmUgdG8gcHV0IHRoZSBnZW5lcmF0ZWQgZmlsZXMgKGRlZmF1bHRzIHRvIGN1cnJlbnQgZGlyZWN0b3J5KQogICAgLXQsIC0tdGVtcGxhdGVzIDx0ZW1wbGF0ZURpcj4gIGRpcmVjdG9yeSB3aGVyZSB0ZW1wbGF0ZXMgYXJlIGxvY2F0ZWQgKGRlZmF1bHRzIHRvIGludGVybmFsIHRlbXBsYXRlcyBkaXJlY3RvcnkpCiAgICAtYiwgLS1iYXNlZGlyIDxiYXNlRGlyPiAgICAgICAgZGlyZWN0b3J5IHRvIHVzZSBhcyB0aGUgYmFzZSB3aGVuIHJlc29sdmluZyBsb2NhbCBmaWxlIHJlZmVyZW5jZXMgKGRlZmF1bHRzIHRvIE9wZW5BUEkgZmlsZSBkaXJlY3RvcnkpCiAgICAtaCwgLS1oZWxwICAgICAgICAgICAgICAgICAgICAgb3V0cHV0IHVzYWdlIGluZm9ybWF0aW9uCmBgYAoKIyMjIyBFeGFtcGxlcwoKVGhlIHNob3J0ZXN0IHBvc3NpYmxlIHN5bnRheDoKYGBgYmFzaApvZyBvcGVuYXBpLnlhbWwgbWFya2Rvd24KYGBgCgpTcGVjaWZ5IHdoZXJlIHRvIHB1dCB0aGUgZ2VuZXJhdGVkIGNvZGU6CmBgYGJhc2gKb2cgLW8gLi9teS1kb2NzIG9wZW5hcGkueWFtbCBtYXJrZG93bgpgYGAKCiMjIFRlbXBsYXRlcwoKIyMjIENyZWF0aW5nIHlvdXIgb3duIHRlbXBsYXRlcwpUZW1wbGF0ZXMgYXJlIHRoZSBzb3VyY2VzIHdoZXJlIHRoZSByZXN1bHQgd2lsbCBiZSBnZW5lcmF0ZWQgZnJvbS4gVGhlcmUgYXJlIGFscmVhZHkgc29tZSB0ZW1wbGF0ZXMKeW91IGNhbiB1c2UgdG8gZ2VuZXJhdGUgY29kZSBhbmQgZG9jdW1lbnRhdGlvbi4KClRoZSBmaWxlcyBpbiB5b3VyIHRlbXBsYXRlIGNhbiBiZSBvZiB0aGUgZm9sbG93aW5nIHR5cGVzOgoxLiBTdGF0aWM6IFRoaXMga2luZCBvZiBmaWxlcyB3aWxsIGJlIHNpbXBseSBjb3BpZWQgdG8gdGhlIG91dHB1dCBkaXJlY3RvcnkuCjIuIFRlbXBsYXRlczogVGhpcyBraW5kIG9mIGZpbGVzIHdpbGwgYmUgY29tcGlsZWQgdXNpbmcgW0hhbmRsZWJhcnNdKGh0dHA6Ly9oYW5kbGViYXJzanMuY29tLyksIGFuZCBjb3BpZWQgdG8gdGhlIG91dHB1dCBkaXJlY3RvcnkuCjMuIFBhdGggdGVtcGxhdGVzOiBUaGlzIGtpbmQgb2YgZmlsZXMgd2lsbCBiZSBjb21waWxlZCB1c2luZyBbSGFuZGxlYmFyc10oaHR0cDovL2hhbmRsZWJhcnNqcy5jb20vKSwgYnV0IGl0IHdpbGwgZ2VuZXJhdGUgb25lIGZpbGUgcGVyIE9wZW5BUEkgcGF0aC4KCkFzc3VtaW5nIHdlIGhhdmUgdGhlIGZvbGxvd2luZyBPcGVuQVBJIFNwZWM6CmBgYHlhbWwKb3BlbmFwaTogIjMuMC4wIgppbmZvOgogIHZlcnNpb246IDEuMC4wCiAgdGl0bGU6IE9wZW5BUEkgUGV0c3RvcmUKICBsaWNlbnNlOgogICAgbmFtZTogTUlUCnNlcnZlcnM6CiAgLSB1cmw6IGh0dHA6Ly9wZXRzdG9yZS5vcGVuYXBpLmlvL3YxCnBhdGhzOgogIC9wZXQ6CiAgICBnZXQ6Li4uCiAgICBwb3N0Oi4uLgogIC9wZXQve3BldElkfToKICAgIGdldDouLi4KICAvdXNlci9sb2dpbjoKICAgIHBvc3Q6Li4uCiAgL3VzZXIve3VzZXJuYW1lfToKICAgIGdldDouLi4KICAgIHB1dDouLi4KICAgIGRlbGV0ZTouLi4KLi4uCmBgYApBbmQgc29tZSB0ZW1wbGF0ZSBmaWxlcyBsaWtlIHRoaXM6CmBgYAp8LSBpbmRleC5qcyAgICAgICAgICAgIC8vIFRoaXMgZmlsZSBjb250YWlucyBzdGF0aWMgY29kZSwgZS5nLiBzdGFydGluZyBhIHdlYnNlcnZlciBhbmQgaW5jbHVkaW5nIC4vYXBpL2luZGV4LmpzCnwrIGFwaS8KIHwtIGluZGV4LmpzLmhicyAgICAgICAvLyBUaGlzIGlzIGEgc3RhdGljIHRlbXBsYXRlLCBpdCBjb250YWlucyBwbGFjZWhvbGRlcnMgdGhhdCB3aWxsIGJlIGZpbGxlZCBpbiwgZS5nLiBpbmNsdWRlcyBmb3IgZWFjaCBmaWxlIGluIHJvdXRlcwogfCsgcm91dGVzLwogIHwtICQkcGF0aCQkLnJvdXRlLmpzLmhicyAgICAgIC8vIFRoaXMgZmlsZSB3aWxsIGJlIGdlbmVyYXRlZCBmb3IgZWFjaCBvcGVyYXRpb24gYW5kIGNvbnRhaW5zIHNrZWxldG9uIGNvZGUgZm9yIGVhY2ggbWV0aG9kIGZvciBhbiBvcGVyYXRpb24uCiAgfCsgJCRwYXRoJCQvICAgICAgICAgICAgICAgICAgLy8gVGhpcyBmb2xkZXIgd2lsbCBhbHNvIGJlIGdlbmVyYXRlZCBmb3IgZWFjaCBvcGVyYXRpb24uCiAgICB8LSByb3V0ZS5qcy5oYnMgICAgICAgICAgICAgLy8gVGhpcyBpcyBhbm90aGVyIGV4YW1wbGUgb2YgYW4gb3BlcmF0aW9uIGZpbGUuCmBgYApUaGUgZmlyc3QgaW1wb3J0YW50IHRoaW5nIHRvIG5vdGljZSBoZXJlIGlzIHRoZSB2YXJpYWJsZSBub3RhdGlvbiBpbiBgJCRwYXRoJCQucm91dGUuanMuaGJzYC4gSXQgd2lsbCBiZSByZXBsYWNlZCBieSB0aGUgbmFtZSBvZiB0aGUgcGF0aC4KClRoaXMgZXhhbXBsZSBhbHNvIHNob3dzIGAkJHBhdGgkJGAgdXNlZCBpbiBhIGZvbGRlciBuYW1lIC0gdGhlIGdlbmVyYXRlZCBmb2xkZXIgbmFtZXMgaGVyZSB3aWxsIHJlcGxhY2UgJCRwYXRoJCQgd2l0aAp0aGUgbmFtZSBvZiB0aGUgcGF0aCAoaW4ga2ViYWItY2FzZSkuCgpJbiB0aGlzIGV4YW1wbGUgdGhlIGdlbmVyYXRlZCBkaXJlY3Rvcnkgc3RydWN0dXJlIHdpbGwgYmUgbGlrZSB0aGlzOgpgYGAKfC0gaW5kZXguanMgICAgICAgICAgICAvLyBUaGlzIGZpbGUgc3RpbGwgY29udGFpbnMgc3RhdGljIGNvZGUgbGlrZSBiZWZvcmUuCnwrIGFwaS8KIHwtIGluZGV4LmpzICAgICAgICAgICAvLyBUaGlzIGZpbGUgd2lsbCBub3cgZS5nLiBoYXZlIGluY2x1ZGVkIHRoZSB0d28gZmlsZXMgaW4gcm91dGVzLgogfCsgcm91dGVzLwogIHwtIHBldC5yb3V0ZS5qcyAgICAgIC8vIFRoaXMgZmlsZSBjb250YWlucyB0aGUgY29kZSBmb3IgbWV0aG9kcyBvbiBwZXRzLgogIHwgICAgICAgICAgICAgICAgICAgIC8vIChlLmcuIGdldFBldCwgcG9zdFBldCwgZ2V0UGV0QnlQZXRJZCkuCiAgfC0gdXNlci5yb3V0ZS5qcyAgICAgLy8gVGhpcyBmaWxlIHdpbGwgY29udGFpbiB0aGUgY29kZSBmb3IgbWV0aG9kcyBvbiB1c2Vycy4KICB8ICAgICAgICAgICAgICAgICAgICAvLyAoZS5nLiBwb3N0VXNlckxvZ2luLCBnZXRVc2VyQnlVc2VybmFtZSwgcHV0VXNlckJ5VXNlcm5hbWUsIGRlbGV0ZVVzZXJCeVVzZXJuYW1lKS4KICB8KyBwZXQvCiAgIHwgLSByb3V0ZS5qcyAgICAgICAgLy8gdGhpcyBmaWxlIGFsc28gY29udGFpbnMgdGhlIGNvZGUgZm9yIG1ldGhvZHMgb24gcGV0cy4KICB8KyB1c2VyLwogICB8IC0gcm91dGUuanMgICAgICAgIC8vIHRoaXMgZmlsZSBhbHNvIGNvbnRhaW5zIHRoZSBjb2RlIGZvciBtZXRob2RzIG9uIHVzZXJzLgpgYGAKCiMjIyBUZW1wbGF0ZSBmaWxlIGV4dGVuc2lvbnMKWW91IGNhbiAob3B0aW9uYWxseSkgbmFtZSB5b3VyIHRlbXBsYXRlIGZpbGVzIHdpdGggYC5oYnNgIGV4dGVuc2lvbnMsIHdoaWNoIHdpbGwgYmUgcmVtb3ZlZCB3aGVuIHdyaXRpbmcgdGhlIGdlbmVyYXRlZApmaWxlLiBlLmcuIGBpbmRleC5qcy5oYnNgIHdyaXRlcyBgaW5kZXguanNgLiBgaW5kZXguanNgIHdvdWxkIGFsc28gd3JpdGUgdG8gYGluZGV4LmpzYCwgaWYgeW91IHByZWZlciB0byBvbWl0IHRoZSBoYnMKZXh0ZW5zaW9uLgoKVGhlIG9ubHkgY2FzZSB3aGVyZSB0aGUgYC5oYnNgIGV4dGVuc2lvbiBpc24ndCBvcHRpb25hbCB3b3VsZCBiZSBpZiB5b3UgYXJlIHdyaXRpbmcgaGFuZGxlYmFycyB0ZW1wbGF0ZXMgd2l0aCB0aGUKdGVtcGxhdGVzLiBJbiB0aGF0IGNhc2UgdGhlIHRoZSB0ZW1wbGF0ZSB3b3VsZCBuZWVkIHRoZSBleHRlbnNpb24gYC5oYnMuaGJzYC4gYHVzZXJ0cGwuaGJzLmhic2Agd3JpdGVzIGB1c2VydHBsLmhic2AKKGJ1dCBgdXNlcnRwbC5oYnNgIGFzIGEgc291cmNlIHdvdWxkIHdyaXRlIGB1c2VydHBsYCB3aXRoIG5vIGV4dGVuc2lvbikuCgojIyMgVGVtcGxhdGUgZmlsZSBjb250ZW50ClRoZSBnZW5lcmF0b3IgcGFzc2VzIHRoZSBPcGVuQVBJIHNwZWMgdG8gdGVtcGxhdGUgZmlsZXMsIHNvIGFsbCBpbmZvcm1hdGlvbiBzaG91bGQgYmUgYXZhaWxhYmxlIHRoZXJlLgpJbiBhZGRpdGlvbiB0byB0aGF0LCB0aGUgY29kZSBnZW5lcmF0b3IgYWRkcyBhIGJpdCBbbW9yZSBkYXRhXSgjZGF0YS1wYXNzZWQtdG8taGFuZGxlYmFycy10ZW1wbGF0ZXMpIHRoYXQgY2FuIGJlIGhlbHBmdWwuCgojIyMjIEV4YW1wbGVzOgojIyMjIyBEeW5hbWljYWxseSByZXF1aXJlIGZpbGVzIGluIEphdmFTY3JpcHQKYGBgbXVzdGFjaGUKe3sjZWFjaCBAcm9vdC5vcGVuYXBpLmVuZHBvaW50c319CmNvbnN0IHt7Ln19ID0gcmVxdWlyZSgnLi9yb3V0ZXMve3sufX0ucm91dGUuanMnKQp7ey9lYWNofX0KYGBgCndpbGwgcHJvZHVjZSAodXNpbmcgdGhlIE9BUyBTcGVjIGV4YW1wbGUgZnJvbSBhYm92ZSk6CmBgYGpzCmNvbnN0IHBldCA9IHJlcXVpcmUoJy4vcm91dGVzL3BldC5yb3V0ZS5qcycpCmNvbnN0IHVzZXIgPSByZXF1aXJlKCcuL3JvdXRlcy91c2VyLnJvdXRlLmpzJykKYGBgCgojIyMgRGF0YSBwYXNzZWQgdG8gSGFuZGxlYmFycyB0ZW1wbGF0ZXMKfCBQYXJhbSB8IFR5cGUgfCBEZXNjcmlwdGlvbiB8CnwgLS0tIHwgLS0tIHwgLS0tIHwKfG9wZW5hcGl8b2JqZWN0fFRoZSBPcGVuQVBJIHNwZWMufAp8b3BlbmFwaS5lbmRwb2ludHN8IG9iamVjdCB8IEFsbCBmaXJzdCBsZXZlbCBlbmRwb2ludHMgKGUuZyAgYHBldGAgYW5kIGB1c2VyYCkgfAoKIyMjIEN1c3RvbSBoYW5kbGViYXJzIGhlbHBlcnMKSWYgeW91ciB0ZW1wbGF0ZSBuZWVkcyBIYW5kbGViYXJzIGhlbHBlcnMsIHlvdSBjYW4gZGVmaW5lIHRoZW0gaW4gYSBkaXJlY3RvcnkgY2FsbGVkIGAuaGVscGVyc2AgaW5zaWRlIHlvdXIgdGVtcGxhdGUuCgpDaGVjayBvdXQgc29tZSBleGFtcGxlcyBpbiB0aGUgW21hcmtkb3duXSguL3RlbXBsYXRlcy9tYXJrZG93bi8uaGVscGVycykgdGVtcGxhdGUuCgojIyMgVXNpbmcgaGFuZGxlYmFycyBwYXJ0aWFscwpJZiB5b3Ugd2FudCB0byB1c2UgcGFydGlhbHMgaW4geW91ciB0ZW1wbGF0ZSwgZGVmaW5lIHRoZW0gaW4gYSBkaXJlY3RvcnkgY2FsbGVkIGAucGFydGlhbHNgIGluc2lkZSB5b3VyIHRlbXBsYXRlLgoKQ2hlY2sgb3V0IHNvbWUgZXhhbXBsZXMgaW4gdGhlIFttYXJrZG93bl0oLi90ZW1wbGF0ZXMvbWFya2Rvd24vLnBhcnRpYWxzKSB0ZW1wbGF0ZS4KCj4gVGhlIG5hbWUgb2YgdGhlIHBhcnRpYWwgd2lsbCBiZSBvYnRhaW5lZCBmcm9tIHRoZSBmaWxlIG5hbWUsIGNvbnZlcnRlZCB0byBjYW1lbCBjYXNlLiBTbywgZm9yIGluc3RhbmNlLCBpZiB0aGUgZmlsZSBuYW1lIGlzIGBteS1wYXJ0aWFsLmpzYCwgeW91IGNhbiB1c2UgdGhlIHBhcnRpYWwgd2l0aCBge3s+IG15UGFydGlhbH19YC4KCiMjIEF1dGhvcnMKCiogRnJhbiBNw6luZGV6IChbQGZtdmlsYXNdKGh0dHA6Ly90d2l0dGVyLmNvbS9mbXZpbGFzKSkKKiBSaWNoYXJkIEtsb3NlIChbQHJpY2hhcmRrbG9zZV0oaHR0cDovL2dpdGh1Yi5jb20vcmljaGFyZGtsb3NlKSkK readmeEtag: '"212d73f5503dc5958a886b77ddcca13207b5f3fd"' readmeLastModified: Thu, 11 Jun 2020 10:46:34 GMT repositoryId: 142300673 description: >- Use your API OpenAPI 3 definition to generate code, documentation, and literally anything you need. created: '2018-07-25T13:02:15Z' updated: '2026-01-19T09:17:34Z' language: JavaScript archived: false stars: 89 watchers: 2 forks: 26 owner: openapi-contrib logo: https://avatars.githubusercontent.com/u/49447320?v=4 license: Apache-2.0 repoEtag: '"9a3c6c80ea1a3c248e04475892112ac9929db2d5fb4677dcb1558e041e0afceb"' repoLastModified: Mon, 19 Jan 2026 09:17:34 GMT foundInMaster: true category: - Documentation - Parsers id: e0d1c54d6f4f90b4425f772e2c721064 - source: openapi3 tags repository: https://github.com/christianhelle/httpgenerator v3: true id: 57615d15d5539b58b50687ba6bff2905 repositoryMetadata: base64Readme: >- <!--toc:start-->

- [HTTP File Generator](#http-file-generator)
  - [Installation](#installation)
  - [Usage](#usage) - [Error Logging, Telemetry, and Privacy](#error-logging-telemetry-and-privacy) - [Visual Studio 2022 Extension](#visual-studio-2022-extension)
  <!--toc:end-->

#

[![Build](https://github.com/christianhelle/httpgenerator/actions/workflows/build.yml/badge.svg)](https://github.com/christianhelle/httpgenerator/actions/workflows/build.yml)
[![Smoke Tests](https://github.com/christianhelle/httpgenerator/actions/workflows/smoke-tests.yml/badge.svg)](https://github.com/christianhelle/httpgenerator/actions/workflows/smoke-tests.yml)
[![NuGet](https://img.shields.io/nuget/v/httpgenerator?color=blue)](https://www.nuget.org/packages/httpgenerator)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=christianhelle_httpgenerator&metric=alert_status)](https://sonarcloud.io/summary/new_code?id=christianhelle_httpgenerator)
[![codecov](https://codecov.io/gh/christianhelle/httpgenerator/graph/badge.svg?token=YeSFnn0bH6)](https://codecov.io/gh/christianhelle/httpgenerator)

# HTTP File Generator

Generate .http files from OpenAPI specifications

`.http` files were made popular by the Visual Studio Code extension [REST Client](https://marketplace.visualstudio.com/items?itemName=humao.rest-client), which then was adopted by JetBrains IDE's, and later on [Visual Studio 2022](https://marketplace.visualstudio.com/items?itemName=MadsKristensen.RestClient)

## Installation

This is tool is distrubuted as a .NET Tool on NuGet.org

To install, simply use the following command

```bash
dotnet tool install --global httpgenerator
```

## Usage

```text
USAGE:
    httpgenerator [URL or input file] [OPTIONS]

EXAMPLES:
    httpgenerator ./openapi.json
    httpgenerator ./openapi.json --output ./
    httpgenerator ./openapi.json --output-type onefile
    httpgenerator https://petstore.swagger.io/v2/swagger.json
    httpgenerator https://petstore3.swagger.io/api/v3/openapi.json --base-url https://petstore3.swagger.io
    httpgenerator ./openapi.json --authorization-header Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9c
    httpgenerator ./openapi.json --azure-scope [Some Application ID URI]/.default
    httpgenerator ./openapi.json --generate-intellij-tests
    httpgenerator ./openapi.json --custom-header X-Custom-Header: Value --custom-header X-Another-Header: AnotherValue

ARGUMENTS:
    [URL or input file]    URL or file path to OpenAPI Specification file

OPTIONS:
                                                                DEFAULT
    -h, --help                                                                       Prints help information
    -v, --version                                                                    Prints version information
    -o, --output <OUTPUT>                                       ./                   Output directory
        --no-logging                                                                 Don't log errors or collect telemetry
        --skip-validation                                                            Skip validation of OpenAPI Specification file
        --authorization-header <HEADER>                                              Authorization header to use for all requests
        --load-authorization-header-from-environment                                 Load the authorization header from an environment variable or define it in the
                                                                                     .http file. You can use --authorization-header-variable-name to specify the
                                                                                     environment variable name
        --authorization-header-variable-name <VARIABLE-NAME>    authorization        Name of the environment variable to load the authorization header from
        --content-type <CONTENT-TYPE>                           application/json     Default Content-Type header to use for all requests
        --base-url <BASE-URL>                                                        Default Base URL to use for all requests. Use this if the OpenAPI spec doesn't
                                                                                     explicitly specify a server URL
        --output-type <OUTPUT-TYPE>                             OneRequestPerFile    OneRequestPerFile generates one .http file per request. OneFile generates a
                                                                                     single .http file for all requests. OneFilePerTag generates one .http file per
                                                                                     first tag associated with each request
        --azure-scope <SCOPE>                                                        Azure Entra ID Scope to use for retrieving Access Token for Authorization header
        --azure-tenant-id <TENANT-ID>                                                Azure Entra ID Tenant ID to use for retrieving Access Token for Authorization
                                                                                     header
        --timeout <SECONDS>                                     120                  Timeout (in seconds) for writing files to disk
        --generate-intellij-tests                                                    Generate IntelliJ tests that assert whether the response status code is 200
        --custom-header                                                              Add custom HTTP headers to the generated request
        --skip-headers                                                               Don't generate header parameters in the files
```

Running the following:

```sh
httpgenerator https://petstore.swagger.io/v2/swagger.json
```

Outputs the following:

![CLI Output](https://github.com/christianhelle/httpgenerator/raw/main/images/httpgenerator-output.png)

Which will produce the following files:

```sh
-rw-r--r-- 1 christian 197121  593 Dec 10 10:44 DeleteOrder.http
-rw-r--r-- 1 christian 197121  231 Dec 10 10:44 DeletePet.http
-rw-r--r-- 1 christian 197121  358 Dec 10 10:44 DeleteUser.http
-rw-r--r-- 1 christian 197121  432 Dec 10 10:44 GetFindPetsByStatus.http
-rw-r--r-- 1 christian 197121  504 Dec 10 10:44 GetFindPetsByTags.http
-rw-r--r-- 1 christian 197121  371 Dec 10 10:44 GetInventory.http
-rw-r--r-- 1 christian 197121  247 Dec 10 10:44 GetLoginUser.http
-rw-r--r-- 1 christian 197121  291 Dec 10 10:44 GetLogoutUser.http
-rw-r--r-- 1 christian 197121  540 Dec 10 10:44 GetOrderById.http
-rw-r--r-- 1 christian 197121  275 Dec 10 10:44 GetPetById.http
-rw-r--r-- 1 christian 197121  245 Dec 10 10:44 GetUserByName.http
-rw-r--r-- 1 christian 197121  513 Dec 10 10:44 PostAddPet.http
-rw-r--r-- 1 christian 197121  521 Dec 10 10:44 PostCreateUser.http
-rw-r--r-- 1 christian 197121  610 Dec 10 10:44 PostCreateUsersWithListInput.http
-rw-r--r-- 1 christian 197121  464 Dec 10 10:44 PostPlaceOrder.http
-rw-r--r-- 1 christian 197121  299 Dec 10 10:44 PostUpdatePetWithForm.http
-rw-r--r-- 1 christian 197121  274 Dec 10 10:44 PostUploadFile.http
-rw-r--r-- 1 christian 197121  513 Dec 10 10:44 PutUpdatePet.http
-rw-r--r-- 1 christian 197121  541 Dec 10 10:44 PutUpdateUser.http
```

In this example, the contents of `PostAddPet.http` looks like this:

```sh
@contentType = application/json

#############################################
### Request: POST /pet
### Summary: Add a new pet to the store
### Description: Add a new pet to the store
#############################################

POST https://petstore3.swagger.io/api/v3/pet
Content-Type: {{contentType}}

{
  "id": 0,
  "name": "name",
  "category": {
    "id": 0,
    "name": "name"
  },
  "photoUrls": [
    ""
  ],
  "tags": [
    {
      "id": 0,
      "name": "name"
    }
  ],
  "status": "available"
}
```

and the contents of `GetPetById.http` looks like this:

```sh
@contentType = application/json

#######################################
### Request: GET /pet/{petId}
### Summary: Find pet by ID
### Description: Returns a single pet
#######################################

### Path Parameter: ID of pet to return
@petId = 0

GET https://petstore3.swagger.io/api/v3/pet/{{petId}}
Content-Type: {{contentType}}
```

with the `--generate-intellij-tests` option, the output looks like this:

```sh
@contentType = application/json

#######################################
### Request: GET /pet/{petId}
### Summary: Find pet by ID
### Description: Returns a single pet
#######################################

### Path Parameter: ID of pet to return
@petId = 1

GET https://petstore3.swagger.io/api/v3/pet/{{petId}}
Content-Type: {{contentType}}

> {%
    client.test("Request executed successfully", function() {
        client.assert(
            response.status === 200,
            "Response status is not 200");
    });
%}
```

Here's an advanced example of generating `.http` files for a REST API hosted on Microsoft Azure that uses the Microsoft Entra ID service as an STS. For this example, I use PowerShell and Azure CLI to retrieve an access token for the user I'm currently logged in with.

```powershell
az account get-access-token --scope [Some Application ID URI]/.default `
| ConvertFrom-Json `
| %{
    httpgenerator `
        https://api.example.com/swagger/v1/swagger.json `
        --authorization-header ("Bearer " + $_.accessToken) `
        --base-url https://api.example.com `
        --output ./HttpFiles
}
```

You can also use the `--azure-scope` and `azure-tenant-id` arguments internally use `DefaultAzureCredentials` from the `Microsoft.Extensions.Azure` NuGet package to retrieve an access token for the specified `scope`.

```powershell
httpgenerator `
  https://api.example.com/swagger/v1/swagger.json `
  --azure-scope [Some Application ID URI]/.default `
  --base-url https://api.example.com `
  --output ./HttpFiles
```

### Error Logging, Telemetry, and Privacy

This tool collects errors and tracks features usages to service called [Exceptionless](https://exceptionless.com/)

By default, error logging and telemetry collection is enabled but it is possible to **opt-out** by using the `--no-logging` CLI argument.

User tracking is done anonymously using the **Support key** shown when running the tool and a generated anonymous identity based on a secure hash algorithm of username@host.

```sh
HTTP File Generator v0.1.5
Support key: mbmbqvd
```

The support key is just the first 7 characters of the generated anonymous identity

![Exceptionless](https://github.com/christianhelle/httpgenerator/raw/main/images/exceptionless-overview.png)

![Exceptionless](https://github.com/christianhelle/httpgenerator/raw/main/images/exceptionless-exception.png)

The `--authorization-header` value is **`[REDACTED]`** and the same goes for all personal identifiable information like the IP address, machine name, and file system folders

![Exceptionless](https://github.com/christianhelle/httpgenerator/raw/main/images/exceptionless-environment.png)

It's important to know that no **support key** will be generated if you opt-out from telemetry collection and that the Exceptionless SDK will be completely disabled.

### Visual Studio 2022 Extension

This tool is also available as a [Visual Studio 2022 extension](https://marketplace.visualstudio.com/items?itemName=ChristianResmaHelle.HttpGenerator)

From the **Tools** menu select **Generate .http files**

![Tools menu](https://github.com/christianhelle/httpgenerator/blob/main/images/vsix_tools.png?raw=true)

This opens the main dialog which has similar input fields as the CLI tool

![Main dialog](https://github.com/christianhelle/httpgenerator/blob/main/images/vsix_httpgenerator_dialog.png?raw=true)

You can acquire an Azure Entra ID access token by clicking on the `...` button beside the **Authorization Headers** input field

![Acquire Azure Entra ID access token](https://github.com/christianhelle/httpgenerator/blob/main/images/vsix_azure_entra_id.png?raw=true)

By default, the **Output folder** is pre-filled with the path of the currently active C# Project in the Solution Explorer, suffixed with **\HttpFiles**

![Solution explorer](https://github.com/christianhelle/httpgenerator/blob/main/images/vsix_solution_explorer.png?raw=true)

Once the .http files are generated you can easily open and inspect them

![.http file](https://github.com/christianhelle/httpgenerator/blob/main/images/vsix_http_file.png?raw=true)

#

For tips and tricks on software development, check out [my blog](https://christianhelle.com)

If you find this useful and feel a bit generous then feel free to [buy me a coffee ☕](https://www.buymeacoffee.com/christianhelle)
 readmeEtag: '"a01bc34fac36e05f5e424f329a3c885998e2c26b"' readmeLastModified: Fri, 19 Sep 2025 22:16:36 GMT repositoryId: 714590129 description: Generate .http files from OpenAPI (Swagger) specifications created: '2023-11-05T10:19:39Z' updated: '2026-02-01T04:10:58Z' language: C# archived: false stars: 121 watchers: 1 forks: 11 owner: christianhelle logo: https://avatars.githubusercontent.com/u/710400?v=4 license: MIT repoEtag: '"faa0bc0d70e9f2e4fcb60e153ef117d9e4130d4475c94b35d25c67c5801fc788"' repoLastModified: Sun, 01 Feb 2026 04:10:58 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/hvuhsg/goapi v3: true id: 40d913554d015468454ac2a32e279fc9 repositoryMetadata: base64Readme: >- # GoAPI

<p align="center">
    <em>A Fast and Easy-to-use Web Framework for Building APIs in Go</em>
</p>
<p align="center">
	<a href="https://github.com/hvuhsg/GoAPI/actions/workflows/tests.yml" target="_blank">
		<img src="https://github.com/hvuhsg/GoAPI/actions/workflows/tests.yml/badge.svg?branch=main" alt="tests status">
	</a>
	<a href="https://www.standwithus.com/" target="_blank">
		<img src="https://badge.yehoyada.com" alt="Bring Them Home">
	</a>
</p>

---

**Documentation**: <a href="https://github.com/hvuhsg/GoAPI/README.md" target="_blank">https://github.com/hvuhsg/GoAPI/README.md</a>

**Source Code**: <a href="https://github.com/hvuhsg/GoAPI" target="_blank">https://github.com/hvuhsg/GoAPI</a>

---


GoAPI is a web framework written in Go that is inspired by FastAPI. It allows you to quickly build and deploy RESTful APIs with minimal effort. The framework comes with built-in validators that can be used to validate and auto-build the API schema with OpenAPI format (version 3). This means you can focus on the main logic of your application while the framework handles the validation and API documentation generation.  


The key features are:
- **automatic docs** [---->](#api-documentation)
- **extensible validators system** [---->](#validation)
- **high level syntax** [---->](#usage)
- **middlewares** support [---->](#middlewares)
- **native handlers** support [---->](#native-handlers)
- **SSL/TLS** support [---->](#https-support)
- **ngrok tunnel** built in [---->](#ngrok-tunnel)

## Quick Start
### Install
To install GoAPI, you need to have Go version 1.13 or higher installed on your system. Once you have installed Go, you can use the following command to install GoAPI:

```sh
go get github.com/hvuhsg/GoAPI
```

### Usage

To use GoAPI, you need to import it in your Go code and create a new instance of the goapi.App object:

```go
package main

import "github.com/hvuhsg/goapi"

func main() {
	app := goapi.GoAPI("small", "1.0v")
}
```

Once you have created the app instance, you can add routes to it by calling the Path() method and specifying the route path. You can also set the HTTP methods that the route supports using the Methods() method. Here is an example of a route that support GET method:

```go
package main

import (
	"github.com/hvuhsg/goapi"
	"github.com/hvuhsg/goapi/request"
	"github.com/hvuhsg/goapi/responses"
	"github.com/hvuhsg/goapi/validators"
)

func main() {
	app := goapi.GoAPI("small", "1.0v")

	echo := app.Path("/echo")
	echo.Methods(goapi.GET)
	echo.Description("echo a back")
	echo.Parameter("a", goapi.QUERY, validators.VRequired{}, validators.VIsInt{})
	echo.Action(func(request *request.Request) responses.Response {
		return responses.NewJSONResponse(responses.Json{"a": request.GetInt("a")}, 200)
	})

	app.Run("127.0.0.1", 8080)
}
```

In the above example, we have created a route at `/echo` that supports GET method. We have also added a description for the route and specified that it expects an integer parameter called "a". Finally, we have specified the action that should be performed when the route is accessed. In this case, we are returning a JSON object that contains a "a" key with the value of a as an integer.

You can add as many routes as you like to the app instance, and each route can have its own unique set of parameters and validators.

## Examples

To help you get started with using GoAPI, we have provided some examples in the examples directory of the repository. These examples demonstrate various use cases of the framework and how to use its features.

The examples included are:  
- [**smallest**](/examples/smallest/) The smallest example of ready to run api.
- [**math_api**](/examples/math_api/) Simple use of methods, parameters and validators.  
- [**external_handler**](/examples/external_handler/) Use of external handler (FileServer) int the app.
- [**tls**](/examples/tls/) Running the app with HTTPS support.
- [**ngrok_tunnel**](/examples/ngrok_tunnel/) Run your app over ngrok tunnel without any external dependencies.  

To run the examples, navigate to the examples directory and run the following command:

```sh
go run <example_name>/main.go
```

This will start the example server `127.0.0.1:8080` and you can visit the example endpoints in your web browser or via curl.
To see all of the endpoints tou can visit `127.0.0.1:8080/docs` to see the auto generated interactive docs.

Feel free to use these examples as a starting point for your own projects and modify them as needed.

## Validation
GoAPI comes with built-in validators that can be used to validate input data automatically. In the above example, we used the `VIsInt` validator to ensure that the "timestamp" parameter is an integer. We also used the `VRange` validator to ensure that the "a" and "b" parameters falls within a specified range.

If the input data fails validation, the framework will automatically return an error to the client. This means you can focus on the main logic of your application, and the framework will handle the validation for you.

Here is a validator for example
<details>
<summary>See validator</summary>

```go
// VRange implementaion for refernce
type VRange struct {
	Min float64
	Max float64
}

func (v VRange) UpdateOpenAPISchema(schema *openapi3.Schema) {
	schema.Min = &v.Min
	schema.Max = &v.Max
}
func (v VRange) Validate(r *request.Request, paramName string) error {
	vr := VIsFloat{}
	err := vr.Validate(r, paramName)
	if err != nil {
		return err
	}

	fVal := r.GetFloat(paramName)

	if fVal < v.Min || fVal > v.Max {
		return fmt.Errorf("parameter %s must be between %f and %f", paramName, v.Min, v.Max)
	}

	return nil
}
```
</details>

## Middlewares
GoAPI supports middlewares, middlewares can be defind in the app level or in the view level, middlewares in the app level are applied to all views.

here is an example for logging middleware, that logs requests in the format that nginx uses.
<details>
<summary>See middleware</summary>

```go
type LoggingMiddleware struct{}

func (LoggingMiddleware) Apply(next AppHandler) AppHandler {
	return func(request *request.Request) responses.Response {
		response := next(request)

		scheme := "http"
		if request.HTTPRequest.TLS != nil {
			scheme = "https"
		}
		fullURL := fmt.Sprintf("%s://%s%s", scheme, request.HTTPRequest.Host, request.HTTPRequest.URL.String())

		method := request.HTTPRequest.Method
		path := request.HTTPRequest.URL.Path
		responseSize := len(response.ToBytes())
		remoteAddr := request.HTTPRequest.RemoteAddr
		date := time.Now().Format("2006-01-02 15:04:05")
		userAgent := request.HTTPRequest.UserAgent()
		statusCode := response.StatusCode()

		log.Printf("%s - - [%s] \"%s %s %s\" %d %d \"%s\" \"%s\"\n", remoteAddr, date, method, path, request.HTTPRequest.Proto, statusCode, responseSize, fullURL, userAgent)
		return response
	}
}
```
</details>

To apply middlewares simply pass it to the Middlewares method for the app or for the view.

```go
package main

import (
	"net/http"

	"github.com/hvuhsg/goapi"
	"github.com/hvuhsg/goapi/middlewares"
)

func main() {
	app := goapi.GoAPI("external handler", "1.0v")
	app.Middlewares(middlewares.LoggingMiddleware{})
}
```

## Native handlers
To allow the usage of native handlers we added a simple way to include them in the app, simply pass the native Handler into the Include method of the app.

Note that native handlers will not be shown in the automatic docs, and currently will not be affected by the middlewares.

here is an example:
```go
package main

import (
	"net/http"

	"github.com/hvuhsg/goapi"
)

func main() {
	app := goapi.GoAPI("external handler", "1.0v")

	// Serve files from GoAPI app using the Include method that allow us to include external handlers in the app
	app.Include("/", http.FileServer(http.Dir(".")))

	app.Run("127.0.0.1", 8080)
}
```

## API Documentation
GoAPI can automatically generate API documentation in OpenAPI format (version 3). This makes it easy to share your API with others and integrate it with other tools that support OpenAPI.

To generate the API documentation, you can simply visit the "/docs" endpoint in your web browser. This will display a user-friendly interface that allows you to view the API schema and test the API endpoints.
For the JSON schema you can visit "/openapi.json".  


![Swagger UI](/docs/images/openapi_closed.png)
![Swagger route open](/docs/images/openapi_open.png)

## HTTPS Support
GoAPI can also serve the api over https, this is not recommanded for production.
We recommand using Nginx or other types of reverse proxy to handle the SSL/TLS security.

To run the application over TLS just use the **RunTLS** method of the app.

```go
package main

import (
	"github.com/hvuhsg/goapi"
	"github.com/hvuhsg/goapi/request"
	"github.com/hvuhsg/goapi/responses"
)

func main() {
	app := goapi.GoAPI("TLS example", "1.0v")

	root := app.Path("/")
	root.Methods(goapi.GET)
	root.Description("simple route")
	root.Action(func(request *request.Request) responses.Response {
		return responses.NewHTMLResponse("<h1>HTML Over HTTPS</h1>", 200)
	})

	app.RunTLS("127.0.0.1", 8080, "./server.crt", "./server.key")
}
```

## Ngrok Tunnel
GoAPI support seamless and simple use of ngrok tunnels during the development of the server.
<details>
<summary>Code example</summary>

```go
package main

import (
	"log"
	"os"

	"github.com/hvuhsg/goapi"
	"github.com/hvuhsg/goapi/request"
	"github.com/hvuhsg/goapi/responses"
)

func main() {
	app := goapi.GoAPI("ngrok tunnel", "1.0v")

	echo := app.Path("/")
	echo.Methods(goapi.GET)
	echo.Description("home page")
	echo.Action(func(request *request.Request) responses.Response {
		return responses.NewHTMLResponse("<h1>Served over ngrok tunnel</h1>", 200)
	})

	ngrokToken := os.Getenv("NGROK_TOKEN")
	if ngrokToken == "" {
		log.Println("Enviroment variable NGROK_TOKEN is required")
		os.Exit(1)
	}

	app.RunNgrok(ngrokToken)
}
```
</details>
 readmeEtag: '"ec8f2f7f9ec7ed89691552cba2ae5a4cdc94210f"' readmeLastModified: Wed, 10 Apr 2024 15:44:02 GMT repositoryId: 601809698 description: GoAPI - A Fast and Easy-to-use Web Framework for Building APIs in Go created: '2023-02-14T21:41:22Z' updated: '2025-11-23T00:06:14Z' language: Go archived: false stars: 87 watchers: 1 forks: 4 owner: hvuhsg logo: https://avatars.githubusercontent.com/u/34660046?v=4 license: MIT repoEtag: '"3fc96727e5107b5be240d912ef019b7822e0c26878dec05df88192eb38b6567c"' repoLastModified: Sun, 23 Nov 2025 00:06:14 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/pylons/pyramid_openapi3 v3: true repositoryMetadata: base64Readme: >- # pyramid_openapi3

## Validate [Pyramid](https://trypyramid.com) views against [OpenAPI 3.0/3.1](https://swagger.io/specification/) documents

<p align="center">
  <img alt="Pyramid and OpenAPI logos"
       height="200"
       src="https://github.com/Pylons/pyramid_openapi3/blob/main/header.jpg?raw=true">
</p>

<p align="center">
  <a href="https://github.com/Pylons/pyramid_openapi3/actions/workflows/ci.yml">
    <img alt="CI for pyramid_openapi3 (main branch)"
         src="https://github.com/Pylons/pyramid_openapi3/actions/workflows/ci.yml/badge.svg">
  </a>
  <img alt="Test coverage (main branch)"
       src="https://img.shields.io/badge/tests_coverage-100%25-brightgreen.svg">
  <img alt="Test coverage (main branch)"
       src="https://img.shields.io/badge/types_coverage-100%25-brightgreen.svg">
  <a href="https://pypi.org/project/pyramid_openapi3/">
    <img alt="latest version of pyramid_openapi3 on PyPI"
         src="https://img.shields.io/pypi/v/pyramid_openapi3.svg">
  </a>
  <a href="https://pypi.org/project/pyramid_openapi3/">
    <img alt="Supported Python versions"
         src="https://img.shields.io/pypi/pyversions/pyramid_openapi3.svg">
  </a>
  <a href="https://github.com/Pylons/pyramid_openapi3/blob/main/LICENSE">
    <img alt="License: MIT"
         src="https://img.shields.io/badge/License-MIT-yellow.svg">
  </a>
  <a href="https://github.com/Pylons/pyramid_openapi3/graphs/contributors">
    <img alt="Built by these great folks!"
         src="https://img.shields.io/github/contributors/Pylons/pyramid_openapi3.svg">
  </a>
</p>

## Peace of Mind

The reason this package exists is to give you peace of mind when providing a RESTful API. Instead of chasing down preventable bugs and saying sorry to consumers, you can focus on more important things in life.

- Your **API documentation is never out-of-date**, since it is generated out of the API document that you write.
- The documentation comes with **_try-it-out_ examples** for every endpoint in your API. You don't have to provide (and maintain) `curl` commands to showcase how your API works. Users can try it themselves, right in their browsers.
- Your **API document is always valid**, since your Pyramid app won't even start if the document does not comply with the OpenAPI 3.0 specification.
- Automatic request **payload validation and sanitization**. Your views do not require any code for validation and input sanitation. Your view code only deals with business logic. Tons of tests never need to be written since every request, and its payload, is validated against your API document before it reaches your view code.
- Your API **responses always match your API document**. Every response from your view is validated against your document and a `500 Internal Server Error` is returned if the response does not exactly match what your document says the output of a certain API endpoint should be. This decreases the effects of [Hyrum's Law](https://www.hyrumslaw.com).
- **A single source of truth**. Because of the checks outlined above, you can be sure that whatever your API document says is in fact what is going on in reality. You have a single source of truth to consult when asking an API related question, such as "Remind me again, which fields are returned by the endpoint `/user/info`?".
- Based on [Pyramid](https://trypyramid.com), a **mature Python Web framework**. Companies such as Mozilla, Yelp, RollBar and SurveyMonkey [trust Pyramid](https://trypyramid.com/community-powered-by-pyramid.html), and the new [pypi.org](https://github.com/pypa/warehouse) runs on Pyramid, too. Pyramid is thoroughly [tested](https://github.com/Pylons/Pyramid/actions?query=workflow%3A%22Build+and+test%22) and [documented](https://docs.pylonsproject.org/projects/pyramid/en/latest/), providing flexibility, performance, and a large ecosystem of [high-quality add-ons](https://trypyramid.com/extending-pyramid.html).

<p align="center">
<a href="https://www.youtube.com/watch?v=P0zNxrDO0sE&amp;t=1061" title="Building Robust APIs" rel="nofollow" class="rich-diff-level-one"><img src="https://user-images.githubusercontent.com/311580/97364772-6d246a80-189c-11eb-84f2-a0ad23236003.png" alt="Building Robust APIs" style="max-width:100%;"></a>
</p>

## Features

- Validates your API document (for example, `openapi.yaml` or `openapi.json`) against the OpenAPI 3.0 specification using the [openapi-spec-validator](https://github.com/p1c2u/openapi-spec-validator).
- Generates and serves the [Swagger try-it-out documentation](https://swagger.io/tools/swagger-ui/) for your API.
- Validates incoming requests _and_ outgoing responses against your API document using [openapi-core](https://github.com/p1c2u/openapi-core).

## Getting started

1. Declare `pyramid_openapi3` as a dependency in your Pyramid project.

2. Include the following lines:

    ```python
    config.include("pyramid_openapi3")
    config.pyramid_openapi3_spec('openapi.yaml', route='/api/v1/openapi.yaml')
    config.pyramid_openapi3_add_explorer(route='/api/v1/')
    ```

3. Use the `openapi` [view predicate](https://docs.pylonsproject.org/projects/pyramid/en/latest/narr/viewconfig.html#view-configuration-parameters) to enable request/response validation:

    ```python
    @view_config(route_name="foobar", openapi=True, renderer='json')
    def myview(request):
        return request.openapi_validated.parameters
    ```

For requests, `request.openapi_validated` is available with two fields: `parameters` and `body`.
For responses, if the payload does not match the API document, an exception is raised.

## Advanced configuration

### Relative File References in Spec

A feature introduced in OpenAPI3 is the ability to use `$ref` links to external files (<https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#reference-object>).

To use this, you must ensure that you have all of your spec files in a given directory (ensure that you do not have any code in this directory as all the files in it are exposed as static files), then **replace** the `pyramid_openapi3_spec` call that you did in [Getting Started](#getting-started) with the following:

```python
config.pyramid_openapi3_spec_directory('path/to/openapi.yaml', route='/api/v1/spec')
```

Some notes:

- Do not set the `route` of your `pyramid_openapi3_spec_directory` to the same value as the `route` of `pyramid_openapi3_add_explorer`.
- The `route` that you set for `pyramid_openapi3_spec_directory` should not contain any file extensions, as this becomes the root for all of the files in your specified `filepath`.
- You cannot use `pyramid_openapi3_spec_directory` and `pyramid_openapi3_spec` in the same app.

### Endpoints / Request / Response Validation

Provided with `pyramid_openapi3` are a few validation features:

- incoming request validation (i.e., what a client sends to your app)
- outgoing response validation (i.e., what your app sends to a client)
- endpoint validation (i.e., your app registers routes for all defined API endpoints)

These features are enabled as a default, but you can disable them if you need to:

```python
config.registry.settings["pyramid_openapi3.enable_endpoint_validation"] = False
config.registry.settings["pyramid_openapi3.enable_request_validation"] = False
config.registry.settings["pyramid_openapi3.enable_response_validation"] = False
```

> [!WARNING]
> Disabling request validation will result in `request.openapi_validated` no longer being available to use.

### Register Pyramid's Routes

You can register routes in your pyramid application.
First, write the `x-pyramid-route-name` extension in the PathItem of the OpenAPI schema.

```yaml
paths:
  /foo:
    x-pyramid-route-name: foo_route
    get:
      responses:
        200:
          description: GET foo
```

Then put the config directive `pyramid_openapi3_register_routes` in the app_factory of your application.

```python
config.pyramid_openapi3_register_routes()
```

This is equal to manually writing the following:

```python
config.add_route("foo_route", pattern="/foo")
```

The `pyramid_openapi3_register_routes()` method supports setting a factory and route prefix as well. See the source for details.

### Specify protocol and port for getting the OpenAPI 3 spec file

Sometimes, it is necessary to specify the protocol and port to access the openapi3 spec file. This can be configured using the `proto_port` optional parameter to the the `pyramid_openapi3_add_explorer` function:

```python
config.pyramid_openapi3_add_explorer(proto_port=('https', 443))
```

### CSP nonce

If a Content Security Policy (CSP) is used in your Pyramid application, you can pass a nonce to the OpenAPI explorer UI by setting the `csp_nonce` request parameter:

```python
def inject_csp_header_tween(request):
    nonce = secrets.token_urlsafe(16)
    request.csp_nonce = nonce
    response = handler(request)
    response.headers["Content-Security-Policy"] = f"script-src 'self' 'nonce-{nonce}'"
    return response
```


## Demo / Examples

There are three examples provided with this package:

- A fairly simple [single-file app providing a Hello World API](https://github.com/Pylons/pyramid_openapi3/tree/main/examples/singlefile).
- A slightly more [built-out app providing a TODO app API](https://github.com/Pylons/pyramid_openapi3/tree/main/examples/todoapp).
- Another TODO app API, defined using a [YAML spec split into multiple files](https://github.com/Pylons/pyramid_openapi3/tree/main/examples/splitfile).

All examples come with tests that exhibit pyramid_openapi's error handling and validation capabilities.

A **fully built-out app**, with 100% test coverage, providing a [RealWorld.io](https://realworld.io) API is available at [niteoweb/pyramid-realworld-example-app](https://github.com/niteoweb/pyramid-realworld-example-app). It is a Heroku-deployable Pyramid app that provides an API for a Medium.com-like social app. You are encouraged to use it as a scaffold for your next project.

## Design defense

The authors of pyramid_openapi3 believe that the approach of validating a manually-written API document is superior to the approach of generating the API document from Python code. Here are the reasons:

1. Both generation and validation against a document are lossy processes. The underlying libraries running the generation/validation will always have something missing. Either a feature from the latest OpenAPI specification, or an implementation bug. Having to fork the underlying library in order to generate the part of your API document that might only be needed for the frontend is unfortunate.

    Validation on the other hand allows one to skip parts of validation that are not supported yet, and not block a team from shipping the document.

2. The validation approach does sacrifice DRY-ness, and one has to write the API document and then the (view) code in Pyramid. It feels a bit redundant at first. However, this provides a clear separation between the intent and the implementation.

3. The generation approach has the drawback of having to write Python code even for parts of the API document that the Pyramid backend does not handle, as it might be handled by a different system, or be specific only to documentation or only to the client side of the API. This bloats your Pyramid codebase with code that does not belong there.

## Running tests

You need to have [poetry](https://python-poetry.org/) and Python 3.10 & 3.12 installed on your machine. All `Makefile` commands assume you have the Poetry environment activated, i.e. `poetry shell`.

Alternatively, if you use [nix](https://nix.dev/tutorials/declarative-and-reproducible-developer-environments), run `nix-shell` to drop into a shell that has everything prepared for development.

Then you can run:

```shell
make tests
```

## Related packages

These packages tackle the same problem-space:

- [pyramid_oas3](https://github.com/kazuki/pyramid-oas3) seems to do things very similarly to pyramid_openapi3, but the documentation is not in English and we sadly can't fully understand what it does by just reading the code.
- [pyramid_swagger](https://github.com/striglia/pyramid_swagger) does a similar
  thing, but for Swagger 2.0 documents.
- [connexion](https://github.com/zalando/connexion) takes the same "write spec first, code second" approach as pyramid_openapi3, but is based on Flask.
- [bottle-swagger](https://github.com/ampedandwired/bottle-swagger) takes the same "write spec first, code second" approach too, but is based on Bottle.
- [pyramid_apispec](https://github.com/ergo/pyramid_apispec) uses generation with
  help of apispec and the marshmallow validation library. See above [why we prefer validation instead of generation](#design-defense).

## Deprecation policy

We do our best to follow the rules below.

- Support the latest few releases of Python, currently Python 3.10 through 3.12.
- Support the latest few releases of Pyramid, currently 1.10.7 through 2.0.2.
- Support the latest few releases of `openapi-core`, currently just 0.19.0.
- See `poetry.lock` for a frozen-in-time known-good-set of all dependencies.

## Use in the wild

A couple of projects that use pyramid_openapi3 in production:

- [Pareto Security Team Dashboard API](https://dash.paretosecurity.com/api/v1) - Team Dashboard for Pareto Security macOS security app.
- [SEO Domain Finder API](https://app.seodomainfinder.com/api/v1) - A tool for finding expired domains with SEO value.
- [Rankalyzer.io](https://app.rankalyzer.io/api/v1) - Monitor SEO changes and strategies of competitors to improve your search traffic.
- [Kafkai API](https://app.kafkai.com/api/v1) - User control panel for Kafkai text generation service.
- [Open on-chain data API](https://tradingstrategy.ai/api/explorer/) - Decentralised exchange and blockchain trading data open API.
- [Mayet RX](https://app.mayetrx.com/api/v1) - Vendor management system for pharma/medical clinical trials.
 readmeEtag: '"5256a27f27fdde8d1e3d3395ac05249aecd92536"' readmeLastModified: Tue, 29 Jul 2025 09:25:25 GMT repositoryId: 138299589 description: Pyramid addon for OpenAPI3 validation of requests and responses. created: '2018-06-22T12:20:57Z' updated: '2026-01-13T20:00:49Z' language: Python archived: false stars: 84 watchers: 12 forks: 47 owner: Pylons logo: https://avatars.githubusercontent.com/u/452227?v=4 license: MIT repoEtag: '"087af9a5c0806a23fee52f5f2cafb83d8c21e22d0a4f041b941de6e4c4c56ce9"' repoLastModified: Tue, 13 Jan 2026 20:00:49 GMT foundInMaster: true category: Testing id: 68586d443740aa8b483e2facfd87c65e - source: openapi3 tags repository: https://github.com/theoomoregbee/sails-hook-swagger-generator v3: true repositoryMetadata: base64Readme: >- # Swagger Generator Sails Hook
[![Travis](https://img.shields.io/travis/theoomoregbee/sails-hook-swagger-generator.svg)](https://travis-ci.org/theoomoregbee/sails-hook-swagger-generator)
[![npm](https://img.shields.io/npm/v/sails-hook-swagger-generator.svg)](https://www.npmjs.com/package/sails-hook-swagger-generator)
[![npm](https://img.shields.io/npm/l/sails-hook-swagger-generator.svg)]()
[![npm](https://img.shields.io/npm/dt/sails-hook-swagger-generator.svg)](https://www.npmjs.com/package/sails-hook-swagger-generator)
[![semantic-release](https://img.shields.io/badge/%20%20%F0%9F%93%A6%F0%9F%9A%80-semantic--release-e10079.svg)](https://github.com/semantic-release/semantic-release)


This helps to create swagger documentation json which is based entirely on Swagger/OpenAPI specification (see [here](https://swagger.io/specification/)).
The hook produces specification based upon OAS 3.0.

![](screenshot/swagger-doc.gif)


## Installation
```sh
$ npm install sails-hook-swagger-generator --save
```


## Demo
Copy the content of [generatedSwagger](test/fixtures/generatedSwagger.json) and paste it in [Swagger Online Editor](https://editor.swagger.io/).


## Usage
Simply by lifting your sails app `sails lift`, after lifting or starting the app,there should be `swagger.json` within **./swagger** folder.

> make sure ./swagger folder is already existing.

Check the **./swagger/swagger.json** for generated swagger documentation json, then head to [Swagger Editor](https://editor.swagger.io/).

## Generated Output

By default, the Swagger Generator Sails Hook generates:
1. Full automatic documentation for all Sails Blueprint routes;
1. Documentation for all Sails
   [actions2](https://sailsjs.com/documentation/concepts/actions-and-controllers#?actions-2)
   actions with routes configured in `config/routes.js`; and
1. Listing of all routes configured in `config/routes.js` (full details cannot be inferred
   for custom routes without additional information being provided - see below).
1. Creation of default tags for paths based upon Sails Model and Controller `globalId`'s.

## Use with [Swagger UI](https://github.com/swagger-api/swagger-ui)

See [#28](https://github.com/theoomoregbee/sails-hook-swagger-generator/issues/28)

## Adding/Customising Generated Output

Documentation detail and customisation of most aspects of the generated Swagger can be achieved by adding:
1. Top-level configuration to `config/swaggergenerator.js`. This provides direct JSON
   used as the template for the output Swagger/OpenAPI.
1. Objects with the key `swagger` to custom route configuration, controller files, action
   functions, model definitions and model attribute definitions. The `swagger` element must be of type
   [SwaggerActionAttribute](./lib/interfaces.ts#L55) for actions (based on OpenApi.Operation)
   or [SwaggerModelSchemaAttribute](./lib/interfaces.ts#L66) for model schemas (based on OpenApi.UpdatedSchema).
1. JSDoc ([swagger-jsdoc](https://github.com/Surnet/swagger-jsdoc)) `@swagger` comments
   to controller/action files and model files.

Top-level Swagger/OpenAPI definitions for `tags` and `components` may be added in all `swagger` objects
above and in all JSDoc `@swagger` documentation comments. This enables the definition of top-level elements.

See below for details.

## Configurations

It comes with some default settings which can be overridden by creating `config/swaggergenerator.js`:
```javascript
module.exports['swagger-generator'] = {
    disabled: false,
    swaggerJsonPath: './swagger/swagger.json',
    swagger: {
        openapi: '3.0.0',
        info: {
            title: 'Swagger Json',
            description: 'This is a generated swagger json for your sails project',
            termsOfService: 'http://example.com/terms',
            contact: {name: 'Theophilus Omoregbee', url: 'http://github.com/theo4u', email: 'theo4u@ymail.com'},
            license: {name: 'Apache 2.0', url: 'http://www.apache.org/licenses/LICENSE-2.0.html'},
            version: '1.0.0'
        },
        servers: [
            { url: 'http://localhost:1337/' }
        ],
        externalDocs: {url: 'https://theoomoregbee.github.io/'}
    },
    defaults: {
        responses: {
            '200': { description: 'The requested resource' },
            '404': { description: 'Resource not found' },
            '500': { description: 'Internal server error' }
        }
    },
    excludeDeprecatedPutBlueprintRoutes: true,
    includeRoute: function(routeInfo) { return true; },
    updateBlueprintActionTemplates: function(blueprintActionTemplates) { ... },
    postProcess: function(specifications) { ... }
};
```

Notes on the use of configuration:

* `disabled` attribute is used to disable the module (e.g you may want to disable it on production).
* `swaggerJsonPath` where to generate the `swagger.json` file to; defaults to `sails.config.appPath + '/swagger/swagger.json'`
   and output file will not be written if empty/null/undefined (see `postProcess` below for alternate save mechanism).
* `swagger` object is template for the Swagger/OpenAPI output. It defaults to the minimal content above.
   Check Swagger/OpenAPI specification for more, in case you want to extend it.
   Generally, this hook provides sensible defaults for as much as possible but you may
   override them in this location or in any of the mechanisms explained below.
* `defaults` object should contain the `responses` element; defaults to the above if not specified.
* `excludeDeprecatedPutBlueprintRoutes` should
  [deprecated](https://sailsjs.com/documentation/reference/blueprint-api/update#?notes) `PUT` blueprint
  routes be excluded from generated Swagger output; defaults to `true`.
* `includeRoute` function used to filter routes to be included in generated Swagger output; see advanced section below.
* `updateBlueprintActionTemplates` allows customisation of the templates used to generate Swagger for blueprints; see advanced section below.
* `postProcess` allows an alternate mechanism for saving and/or modification of the generated Swagger output before it is written to
   the output file; see advanced section below.


## Custom Route Configuration

Documentation detail and customisation of most aspects of the generated Swagger for
[custom routes](https://sailsjs.com/documentation/concepts/routes/custom-routes) may be achieved by:

1. Adding an object with the key `swagger` (must be of type [SwaggerActionAttribute](./lib/interfaces.ts#L55) for actions, based on OpenApi.Operation) to individual route configurations in `config/routes.js`.
1. Adding an object with the key `swagger` (must be of type [SwaggerControllerAttribute](./lib/interfaces.ts#L76)) to the exports of a controller file, standalone action file or actions2 file.
1. Adding an object with the key `swagger` (must be of type [SwaggerModelAttribute](./lib/interfaces.ts#L87)) to the exports of a model file.
1. Adding JSDoc `@swagger` comments to Sails
  [model files](https://sailsjs.com/documentation/concepts/models-and-orm/models),
  [controller files](https://sailsjs.com/documentation/concepts/actions-and-controllers#?controllers),
  [standalone action files](https://sailsjs.com/documentation/concepts/actions-and-controllers#?standalone-actions) or
  [actions2 files](https://sailsjs.com/documentation/concepts/actions-and-controllers#?actions-2); specifically:
    - JSDoc `@swagger` documentation under the `/{actionName}` path for the route (controllers/actions), or
    - JSDoc `@swagger` documentation under the `/{blueprintAction}` path for the route (models), or
    - JSDoc `@swagger` documentation under `tags` and `components` paths for adding to the top-level Swagger/OpenAPI definitions.


### Custom Route Configuration in `config/routes.js`

If you want to add extra configuration to a route, it can be done via the `config/routes.js`, since Sails uses different [route targets](https://sailsjs.com/documentation/concepts/routes/custom-routes#?route-target), we can leverage the route object target to extend/override our swagger configuration by adding an object with a key `swagger`.

For example, in `config/routes.js`:
```javascript
{
  'post /user/login': {
    controller: 'UserController',
    action: 'login',
    swagger: {
      summary: 'Authentication',
      description: 'This is for authentication of any user',
      tags: [ 'Tag Name' ],
      requestBody: {
        content: {
          'application/json': {
            schema: {
              properties: {
                email: { type: 'string' },
                password: { type: 'string', format: 'password' }
              },
              required: [ 'email', 'password' ],
            }
          }
        }
      },
      parameters: [{
        in: 'query',
        name: 'firstName',
        required: true,
        schema: { type: 'string' },
        description: 'This is a custom required parameter'
      }],
      responses: {
        '200': {
          description: 'The requested resource',
            content: {
              'application/json': {
                schema: {
                  type: 'array',
                  items: { '$ref': '#/components/schemas/someDataType' },
                },
              },
            },
        },
        '404': { description: 'Resource not found' },
        '500': { description: 'Internal server error' }
      }
    }
  }
}
```

### Custom Route Configuration in Controller or Action files

Documentation detail and customisation of most aspects of the generated Swagger may be added to
[controller files](https://sailsjs.com/documentation/concepts/actions-and-controllers#?controllers),
[standalone action files](https://sailsjs.com/documentation/concepts/actions-and-controllers#?standalone-actions) or
[actions2 files](https://sailsjs.com/documentation/concepts/actions-and-controllers#?actions-2) as follows:

1. Adding an object with the key `swagger` added to a controller file action function.
1. Adding an object with the key `swagger` to the exports of a controller file, standalone
   action file or actions2 file:
   - For controller files, actions are referenced by adding objects keyed on `swagger.actions.{actionName}`
     name. See [UserController.js](./api/controllers/UserController.js);
   - For standalone action or actions2 files, placing content in the
     `swagger.actions.{actionFileName|actions2FileName}` object. See
     [actions2.js](./api/controllers/subdir/actions2.js)
     ***Note**: `actionFileName|actions2FileName` must correspond to the filename*;
   - For all controller/action files, adding per-action documentation to be
     applied to **all** actions using the key `swagger.actions.allActions`
     e.g. use this to apply common tags to all actions for a controller.
   - Adding documentation under `tags` and `components` elements for adding to the
     top-level Swagger/OpenAPI definitions. See example in either
     [UserController.js](./api/controllers/UserController.js)
     or [actions2.js](./api/controllers/subdir/actions2.js).
1. Adding JSDoc `@swagger` comments to controller file, standalone action file or actions2 file:
   - JSDoc `@swagger` documentation under the `/{actionName}` path for the controller file actions,
   - JSDoc `@swagger` documentation under the `/{actionFileName|actions2FileName}` path for standalone action or actions2 files,
   - JSDoc `@swagger` documentation under the `/allActions` path to be applied to **all** actions for the controller, or
   - JSDoc `@swagger` documentation under `tags` and `components` paths for adding to the top-level Swagger/OpenAPI definitions.

An `exclude` property, set to `true`, may be added to any `swagger` element or `@swagger` JSDoc
action documentation to exclude that action from the generated Swagger. See example in
[NomodelController.js](api/controllers/NomodelController.js).

The Swagger definition for each action is merged in the order above to form the final
definition, with `config/routes.js` taking highest precendence and **earlier** definitions
above taking precedence over later.

#### For actions2 files:
1. Inputs are parsed to generate parameter documentation.
2. Exits are parsed to generate response documentation.
3. Both may be customised/overridden by specifying parameters and/or responses in the `swagger` object
   in actions2 file.
4. Inputs may also add an object with the key `meta.swagger` to document the attributes
   Swagger/OpenAPI **schema** associated with the input value. See example in
   [actions2.js](./api/controllers/subdir/actions2.js).
5. Inputs may be excluded from the generated Swagger by setting `meta.swagger.exclude` to `true`.
6. Inputs may specify *where* the input should be included within the generated Swagger using the key
   `meta.swagger.in`. The values `query`/`header`/`path`/`cookie` may be used to produce Swagger operation
   parameters and the value `body` may be used to produce `requestBody` schema properties
   (valid for PUT/POST/PATCH operations only).

For example, for a route configured as:
```javascript
module.exports.routes = {
    '/api/v1/auth/tokens': 'AuthController.tokens',
};
```

The `tokens` action might be documented in a Controller `api/controllers/AuthController.js` as follows:
```javascript
function tokens(req, res) {
    ...
}

module.exports = {
    tokens: tokens,
    swagger: {
      actions: {
        tokens: {
            tags: [ 'Auth' ],
            description: 'Route description...'
        }
      }
      tags: [
             {
                name: 'Auth',
                description: 'Module description ...',
             }
        ],
      components: {
        ...
      }
    }
};
```

Or, alternately using JSDoc:
```javascript
/**
 * @swagger
 *
 * /tokens:
 *   description: Route description...
 *   tags:
 *     - Auth
 * tags:
 *   - name: Auth
 *     description: Module description...
 */
function tokens(req, res) {
    ...
}

module.exports = {
    tokens: tokens
};
```

## Blueprint Route Configuration

Documentation detail and customisation of most aspects of the generated Swagger for
[blueprint routes](https://sailsjs.com/documentation/concepts/blueprints/blueprint-routes) may be achieved by:

1. Adding an object with the key `swagger` to individual models e.g. `api/models/modelName.js`:
   - Adding documentation to the model's Swagger **schema** using the key `swagger.modelSchema`
     e.g. use this to apply detailed documentation via the `description` field;
   - In additon to the model's schema, the key `swagger.modelSchema` may be used to specify
     tag names (as a `string[]`) to be assigned all blueprint actions for the model.
     This is a *non-standard* convenience function i.e. in Swagger/OpenAPI you need to
     explicitly add tags to **each/every** OpenAPI.Operation;
   - Adding per-action documentation by adding objects keyed on
     `swagger.actions.{blueprintAction}` name;
   - Adding action documentation to **all** actions using the key `swagger.actions.allActions`
     e.g. use this to apply common `externalDocs` to all blueprint actions for the model; or
   - Adding documentation under `swagger.tags` and `swagger.components` elements for adding to the top-level Swagger/OpenAPI definitions.
1. Adding documentation-specific fields to model attributes (supports `description`, `moreInfoUrl` and `example`).
   Note that applicable Sails [attributes](https://sailsjs.com/documentation/concepts/models-and-orm/attributes),
   [automigrations](https://sailsjs.com/documentation/concepts/models-and-orm/attributes#?automigrations) and
   [validations](https://sailsjs.com/documentation/concepts/models-and-orm/validations) are also parsed.
1. Adding an object with the key `meta.swagger` to individual model attributes to document
   the attributes Swagger/OpenAPI **schema**. See example in [Pet.js](./api/models/Pet.js).
1. Adding JSDoc `@swagger` comments to model files:
   - JSDoc `@swagger` documentation under the `/{globalId}` to add documentation to
     the model's Swagger **schema** (or tags as noted above),
   - JSDoc `@swagger` documentation under the `/{blueprintAction}` to add per-action
     documentation for the
     [model blueprint actions](https://sailsjs.com/documentation/concepts/blueprints/blueprint-actions),
   - JSDoc `@swagger` documentation under the `/allActions` path to be applied to **all**
     blueprint actions for the model, or
   - JSDoc `@swagger` documentation under `tags` and `components` paths for adding to the
     top-level Swagger/OpenAPI definitions.

An `exclude` property, set to `true`, may be added to any `swagger` element of `@swagger` JSDoc
action documentation to exclude the model *completely* (exclude the **schema**) or a specific
blueprint action from the generated Swagger. See example in [OldPet.js](api/models/OldPet.js).

Individual model attributes may be excluded from the generated Swagger by setting
`meta.swagger.exclude` to `true`. See example in [Pet.js](api/models/Pet.js).

OpenAPI 3 specifies the ***Any Type*** by the absence of the `type` property in a schema;
this may be achieved by setting a model attribute's `meta.swagger.type` value to `null`.
See example in [User.js](api/models/User.js).

The Swagger definition for each action is merged in the order above to form the final
definition, with `config/routes.js` taking highest precendence and **earlier** definitions
above taking precedence over later.

For example, in a model `api/models/User.js`:

```javascript
/**
 * @swagger
 *
 * /User:
 *   tags:
 *     - Tag Name
 * /findone:
 *   externalDocs:
 *     url: https://docs.com/here
 */
module.exports = {
  attributes: {
    uid: {
      type: 'string',
      example: '012345',
      description: 'A unique identifier',
    }
  },
  swagger: {
    actions: {
      create: { ... },
    },
    modelSchema: { ... },
    tags: [...]
    components: {...}
  }
};
```

Note that following parameters are added to the `components/parameters` if they are not
provided in `config/swaggergenerator.js` (expressed as OpenAPI references):

```javascript
[
  { $ref: '#/components/parameters/WhereQueryParam' },
  { $ref: '#/components/parameters/LimitQueryParam' },
  { $ref: '#/components/parameters/SkipQueryParam' },
  { $ref: '#/components/parameters/SortQueryParam' },
  { $ref: '#/components/parameters/SelectQueryParam' },
  { $ref: '#/components/parameters/PopulateQueryParam' },
]
```

Note that when generating Swagger/OpenAPI documentation for blueprint routes, the hook also
generates:

1. Schemas for **models**, which may be referenced using the form `{ $ref: '#/components/schemas/modelName' }`.
2. Parameters for model **primary keys**, which may be referenced using the form `{ $ref: '#/components/parameters/ModelPKParam-modelName' }`.

These may be re-used (referenced) if/as applicable within custom route documentation.


## Handling Top-Level Swagger Defintions (Tags and Components)

You are able to add to the top-level Swagger/OpenAPI definitions for `tags` and `components` in all `swagger` objects
detailed above and in all JSDoc `@swagger` documention comments.

All `swagger` objects may contain the elements `tags` and `components`(*except the ones specified in `config.routes.js*) e.g.

```javascript
{
  tags: [
    {
      name: 'Test Module',
      description: 'Module description ...',
      externalDocs: { url: 'https://docs.com/test' }
    }
  ],
  components: {
    schemas: {
      test: { ... }
    }
  }
}
```

Similarly, JSDoc `@swagger` tags may define `tags` and `components`:

```javascript
/**
 * @swagger
 *
 * tags:
 *   - name: Test Module
 *     description: |
 *       Module description
 *       (continued).
 *
 *       Another paragraph.
 *
 *     externalDocs:
 *       url: https://docs.com/test
 *       description: Refer to these docs
 *
 * components:
 *   schemas:
 *     test:
 *       ...
 */
```

### Tags Handling

Tags are added to the top-level Swagger/OpenAPI definitions as follows:
1. If a tags with the specified name **does not** exist, it is added.
1. Where a tag with the specified name **does** exist, elements _of that tag_ that do not exist are added
   e.g. `description` and `externalDocs` elements.

Note that a final *clean-up* phase is run after processing, which performs the following:
1. Removal of unreferenced tags; and
2. Creation of tags referenced but are not defined.

### Component Element Handling

Elements of components are added to the top-level Swagger/OpenAPI definitions as follows:
1. Elements of the component definition reference (schemas, parameters, etc) are added where
   they **do not exist**.
1. Existing elements are **not** overwritten or merged.

For example, the element `components.schemas.pet` will be added as part of a merge process,
but the contents of multiple definitions of `pet` **will not** be merged.

The following elements (from the OpenAPI 3 specification) are handled:
```javascript
let componentDefinitionReference = {
    // Reusable schemas (data models)
    schemas: {},
    // Reusable path, query, header and cookie parameters
    parameters: {},
    // Security scheme definitions (see Authentication)
    securitySchemes: {},
    // Reusable request bodies
    requestBodies: {},
    // Reusable responses, such as 401 Unauthorized or 400 Bad Request
    responses: {},
    // Reusable response headers
    headers: {},
    // Reusable examples
    examples: {},
    // Reusable links
    links: {},
    // Reusable callbacks
    callbacks: {},
};
```


## Advanced Filtering/Processing of Generated Swagger

Three mechanisms are provided to enable advancing filtering of the Swagger generation process:
1. An `includeRoute()` function used to filter routes to be included in generated Swagger output.
1. An `updateBlueprintActionTemplates()` function allows customisation of the templates used to generate Swagger for blueprints.
1. A `postProcess()` function allows an alternate mechanism for saving and/or modification of the generated Swagger output before it is written to the output file.

Each is configured in `config/swaggergenerator.js`.

### Route Information

This hook parses all routes, custom and blueprint, before commencing the generation of the Swagger output.
Each route is described by a `SwaggerRouteInfo` object
(see defintion [here](./lib/interfaces.ts#L184)):

```typescript
export interface SwaggerRouteInfo {
  middlewareType: MiddlewareType; //< one of action|blueprint

  verb: HTTPMethodVerb; //< one of all|get|post|put|patch|delete
  path: string; //< full Sails URL as per sails.getUrlFor() including prefix
  variables: string[]; //< list of ALL variables extracted from path e.g. `/pet/:id` --> `id`
  optionalVariables: string[]; //< list of optional variables from path e.g. `/pet/:id?`

  action: string; //< either blueprint action (e.g. 'find') or action identity (e.g. 'subdir/reporting/run')
  actionType: ActionType; //< one of blueprint|shortcutBlueprint|controller|standalone|actions2|function
  actions2Machine?: Sails.Actions2Machine; //< for actionType === 'actions2', details of the action2 machine

  model?: SwaggerSailsModel; //< reference to Sails Model (blueprints only)
  associationAliases?: string[]; //< association attribute names (relevant blueprint routes only)

  defaultTagName?: string; //< default tag name for route, if any, based on Sails Model or Controller

  swagger?: SwaggerActionAttribute; //< per-route Swagger (OpenApi Operation)
}
```

Other interfaces for models, `swagger` elements etc may be found in [interfaces.ts](./lib/interfaces.ts).


### Route Filtering

The `includeRoute(routeInfo): boolean` function may be used to select which routes are included in the generated Swagger output.

For example:
```javascript
module.exports['swagger-generator'] = {
  includeRoute: (routeInfo) => {
    let c = routeInfo.controller;
    if(!c) return true;
    if(c.toLowerCase().startsWith('user')) return true;
    return false;
  }
}
```

### Customising Blueprint Action Templates

The templates used for generating Swagger for each Sails blueprint action route may be
customised / modified / added to using the `updateBlueprintActionTemplates` config option
e.g. to support custom blueprint actions/routes.

For example:
```javascript
module.exports['swagger-generator'] = {
  updateBlueprintActionTemplates: function(blueprintActionTemplates) {
    blueprintActionTemplates.search = { ... };
    return blueprintActionTemplates;
  }
}
```

The `blueprintActionTemplates` object contains keys of the blueprint **action names**
and values as per the following example (refer to the
[source code](lib/type-formatter.js#L70) for the default templates):

```javascript
let blueprintActionTemplates = {
  findone: {
    summary: 'Get {globalId} (find one)',
    description: 'Look up the **{globalId}** record with the specified ID.',
    externalDocs: {
      url: 'https://sailsjs.com/documentation/reference/blueprint-api/find-one',
      description: 'See https://sailsjs.com/documentation/reference/blueprint-api/find-one'
    },
    parameters: [
      'primaryKeyPathParameter', // special case; filtered and substituted during generation phase
      { $ref: '#/components/parameters/LimitQueryParam' },
    ],
    resultDescription: 'Responds with a single **{globalId}** record as a JSON dictionary',
    notFoundDescription: 'Response denoting **{globalId}** record with specified ID **NOT** found',
    // if functions, each called with (blueprintActionTemplate, routeInfo, pathEntry)
    modifiers: ['addSelectQueryParam', exampleModifierFunctionRef],
  },
  ...
};
```

Note that:
1. For summary and description strings the value `{globalId}` is replaced with the applicable Sails model value.
1. Parameters values are Swagger definitions, with the exception of the *special* string value
   `primaryKeyPathParameter`, which may be used to include a reference to a model's primary key.
1. Modifiers are used to apply custom changes to the generated Swagger, noting that:
   - String values are predefined in `generatePaths()` (refer to the [source code](lib/generators.js#L246));
     valid  modifiers are:
     - `addPopulateQueryParam`
     - `addSelectQueryParam`
     - `addOmitQueryParam`
     - `addModelBodyParam`
     - `addModelBodyParamUpdate`
     - `addResultOfArrayOfModels`
     - `addAssociationPathParam`
     - `addAssociationFKPathParam`
     - `addAssociationResultOfArray`
     - `addResultOfModel`
     - `addResultNotFound`
     - `addResultValidationError`
     - `addFksBodyParam`
     - `addShortCutBlueprintRouteNote`
   - Functions are called as `func(blueprintActionTemplate, routeInfo, pathEntry, tags, components)`
     where
     - `blueprintActionTemplate` the blueprint action template (see above) to which the modifier relates
     - `routeInfo` the route information object (see above) for which the Swagger is being generated
     - `pathEntry` the generated Swagger path entry to be modified
     - `tags` the generated Swagger **tag** definitions to be modified/extended
     - `components` the generated Swagger **component** definitions to be modified/extended

### Post-processing Generated Swagger Output

The final generated Swagger output may be post-processed before it is written to
the output file using a post-processing function specified as the `postProcess` config option.

For situations where saving the generated swagger documentation JSON to a file is
not desired/appropriate, the `postProcess` config option may be used to specify
an alternate save mechanism.

Note that if `swaggerJsonPath` config option is empty/null/undefined the output file will not be written.

For example:
```javascript
module.exports['swagger-generator'] = {
  postProcess: function(specifications) {
    let sch = specifications.components.schemas;
    Object.keys(sch).map(k => {
      sch[k].description = sck[k].description.toUpperCase();
    });
  }
}
```


## Testing

* Clone this repository

* Install all development dependencies

```sh
 npm install
```
* Then run test

```sh
npm test
```

## Contribute

Fork this repo and push in your ideas.
Do not forget to add a bit of test(s) of what value you adding.
* stick to conventional commit message [here](https://conventionalcommits.org/) or read more [angular commit](https://github.com/angular/angular/blob/master/CONTRIBUTING.md#commit) pattern
* While developing, you can run the below command to start nodemon watch too run linting and testing on `.ts` changes

```sh
npm run dev
```

## Changelog

See the different releases [here](https://github.com/theo4u/sails-hook-swagger-generator/releases)

## License

MIT License (MIT)
 readmeEtag: '"4f2cdccc179372dc67c604bdcbe65a6dd9ecc6dd"' readmeLastModified: Mon, 05 Jul 2021 16:21:27 GMT repositoryId: 99695966 description: >- A tool to help generate Swagger specification documentation based on OAS 3.0 for Sails APIs created: '2017-08-08T13:27:48Z' updated: '2025-08-27T12:13:43Z' language: TypeScript archived: false stars: 82 watchers: 7 forks: 33 owner: theoomoregbee logo: https://avatars.githubusercontent.com/u/10440327?v=4 license: MIT repoEtag: '"86b54448585ead86a5801237e631fa10e4df37275e1ff90d73d6201ff22705d7"' repoLastModified: Wed, 27 Aug 2025 12:13:43 GMT foundInMaster: true category: Parsers id: 735d3b9a4efacd3a1909f6a1b08a9b7a - source: openapi3 tags repository: https://github.com/landgraf-dev/openapi-delphi-generator v3: true id: c46c02d7c3412c4eb8f626910aefa33f repositoryMetadata: base64Readme: >- # OpenAPI Client Generator for Delphi

Generate Delphi client SDKs for any REST API defined with the OpenAPI specification. 

This generator can read an OpenAPI document (from local file or URL) of a REST API and generate Delphi classes that you can use to invoke such REST API endpoints in a friendly way.

The generator creates the DTO classes (representing the JSON definitions used by the API) and the interfaces with methods you can call (representing the endpoints to invoke).

## Usage

[Download the binaries](https://github.com/landgraf-dev/openapi-delphi-generator/releases) and decompress in a folder in your disk. The generator is a single command-line exe file named `OpenApiDelphiGen.exe`. Execute it passing the proper command-line parameters. The tool provides a help file showing you all the available options.

### Generating the client

The following example generates client files for the popular [Swagger PetStore](https://petstore.swagger.io) example API.

```shell
OpenApiDelphiGen -i:"https://petstore.swagger.io/v2/swagger.json" -o:"C:\PetStore" -n:PetStore -m:MultipleClientsFromFirstTagAndOperationId
```

The `-i` parameter indicates the OpenAPI document to import. The `-o` parameter provides the output folder where the Delphi files should be generated. The `-n` parameter is the name of the API, used to prefix unit and class names. Finally, `-m` parameter indicates how the service classes and interfaces should be organized.

The generator will create the following units:

<img width="408" alt="image" src="https://user-images.githubusercontent.com/10242580/181278336-6ec72270-ee32-416b-bb69-8e90b88e5b07.png">

`PetStoreClient` is the main unit with the service classes and interfaces. `PetStoreDTOs` contains all the DTO classes, and `PetStoreJson` holds code for serializing/deserializing DTOs to and from JSON.

### Using the client

To use the client, just add the generated units to your uses classes. Instantiate the client and use its methods. Note that the generated code depends on the units provided in the [Dist](Dist) folder, so you must also add them to your project.

Once you instantiate the client, just call a method from a service to invoke an endpoint. The following example will create a new pet in the server by invoking the `POST /Pets`  endpoint:

```delphi
uses {...}, PetStoreClient, PetStoreDtos;

procedure TPetStoreClientTests.CreateAndGetPet;
const
  PetId = 61341;
  CategoryId = 61341;
  TagId = 61341;
var
  Pet: TPet;
  Tag: TTag;
  Client: IPetStoreClient;
begin
  Client := TPetStoreClient.Create;

  // Create the pet
  Pet := TPet.Create;
  try
    Pet.Id := PetId;
    Pet.Name := 'Josephine';
    Pet.Status := 'available';

    Pet.Category := TCategory.Create;
    Pet.Category.Id := CategoryId;
    Pet.Category.Name := 'Terrier Dogs';

    Pet.Tags := TTagList.Create;
    Tag := TTag.Create;
    Tag.Id := TagId;
    Tag.Name := 'Terrier';
    Pet.Tags.Add(Tag);

    Pet.PhotoUrls.Add('http://dummy.com/dog.png');
    Client.Pet.AddPet(Pet);
  finally
    Pet.Free;
  end;
```

Another example that will retrieve and then delete a pet:

```delphi
  Client := TPetStoreClient.Create;
  Pet := Client.Pet.GetPetById(PetId);
  try
    Check(Pet <> nil, Format('Pet %d not found', [PetId]));
    CheckEquals(PetId, Pet.Id);
    CheckEquals('Josephine', Pet.Name);
    CheckEquals('available', Pet.Status);
  finally
    Pet.Free;
  end;

  // Delete the newly created pet
  Client.Pet.DeletePet('special-key', PetId);
```

You must create and destroy all DTO classes yourself. The classes and lists owned by the DTO classes are destroyed automatically.

### Authentication

You can set the `Config.AccessToken` property to authenticate with the API:

```delphi
Client.Config.AccessToken := '<access token>';
```

It will set the `Authorization` HTTP header with the token value prefixed by `Bearer `.

### Client compatibility

Current the generated code doesn't require any dependency on 3rd party units, working with a plain Delphi install. The code is currently using `THttpClient` class from unit `System.Net.HttpClient` to perform the HTTP requests, so the client will only compile on Delphi versions that provide such class.

Alternatively, you can force the client to use Indy components to perform HTTP requests. For that, can use the following code snippet.

```delphi
uses {...}, OpenApiRest, OpenApiIndy;

begin
  // At the beginning of your application, set the DefaultRequestFactory
  DefaultRequestFactory := TIndyRestRequestFactory.Create;
end;
```

You can intercept the moment where a new `TIdHTTP` client is created - in case you want to setup specific properties like proxy or IO handlers. For that you can use a code like this:

```delphi
uses {...}, OpenApiRest, OpenApiIndy, IdHTTP;

procedure TMainDataModule.IndyClientCreated(Client: TIdHTTP);
begin
  // Set extra Client properties here
end;

procedure TMainDataModule.DataModuleCreate(Sender: TObject);
var
  Factory: TIndyRestRequestFactory;
begin
  // At the beginning of your application, set the DefaultRequestFactory
  Factory := TIndyRestRequestFactory.Create;
  DefaultRequestFactory := Factory;
  Factory.OnClientCreated := IndyClientCreated;
end;
```

## Compiling and running tests

You can compile and run the test project also without any dependencies. The tests were created using DUnit for maximum compatibility with several Delphi versions. 

## Compiling the generator tool

The generator tool itself uses 3rd party libraries. You don't need them to use it, you can simply download the executable and use it directly. In case you want to compile and debug the tool, you will need the following libraries:

* [VSoft.CommandLineParser](https://github.com/VSoftTechnologies/VSoft.CommandLineParser)
* [TMS BIZ](https://www.tmssoftware.com/site/tmsbizintro.asp)

Note that TMS BIZ is a commercial library. You can use the trial version for development (in case you want to contribute with the project).

## License

This project is [fair-code](http://faircode.io) distributed under [**Apache 2.0 with Commons Clause**](LICENSE) license.
 readmeEtag: '"db514f84ca82654bd88f2578cda0cc39d251d183"' readmeLastModified: Tue, 30 Jan 2024 18:38:25 GMT repositoryId: 517490524 description: >- Generate Delphi client SDKs for any REST API defined with the OpenAPI specification. created: '2022-07-25T02:23:12Z' updated: '2026-01-23T08:40:28Z' language: Pascal archived: false stars: 89 watchers: 10 forks: 18 owner: landgraf-dev logo: https://avatars.githubusercontent.com/u/52084575?v=4 license: NOASSERTION repoEtag: '"e6b934d241dfb3fa768d662ca8a58be9eaa6471701032ab2d37629d56f243a21"' repoLastModified: Fri, 23 Jan 2026 08:40:28 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/jimschubert/intellij-openapi-generator v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIEdlbmVyYXRvciBwbHVnaW4gZm9yIEludGVsbGlKIElERXMKCiFbQXNzZW1ibGUgUGx1Z2luXShodHRwczovL2dpdGh1Yi5jb20vamltc2NodWJlcnQvaW50ZWxsaWotb3BlbmFwaS1nZW5lcmF0b3Ivd29ya2Zsb3dzL0Fzc2VtYmxlJTIwUGx1Z2luL2JhZGdlLnN2ZykKCjwhLS0gUGx1Z2luIGRlc2NyaXB0aW9uIC0tPgpUaGUgT3BlbkFQSSBHZW5lcmF0b3IgcGx1Z2luIGFsbG93cyB5b3UgdG8gZ2VuZXJhdGUgY2xpZW50LCBzZXJ2ZXIsIG9yIGRvY3VtZW50YXRpb24gY29kZSBmcm9tIHlvdXIgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyB3aXRoaW4geW91ciBJREUuCjwhLS0gUGx1Z2luIGRlc2NyaXB0aW9uIGVuZCAtLT4KClRoaXMgcGx1Z2luIGFsbG93cyBmb3IgYSBzb3J0IG9mIF9zcGVjXyBkcml2ZW4gZGV2ZWxvcG1lbnQuCgpXaGlsZSBkZXNpZ25pbmcgeW91ciBBUEksIHlvdSBjYW4gZ2VuZXJhdGUgb25lIG9yIG1vcmUgY2xpZW50cyBmcm9tIHdpdGhpbiB0aGUgSURFIHRvIGV2YWx1YXRlIHlvdXIgY2hhbmdlcy4KCkFsc28gaW5zdGFsbCBbaW50ZWxsaWotc3dhZ2dlcl0oaHR0cHM6Ly9wbHVnaW5zLmpldGJyYWlucy5jb20vcGx1Z2luLzgzNDcpIGZvciBhIHNlYW1sZXNzIFN3YWdnZXIvT3BlbkFQSSBlZGl0b3IgZXhwZXJpZW5jZS4KCiMgQnVpbGRpbmcKCmBgYAouL2dyYWRsZXcgY2xlYW4gYnVpbGRQbHVnaW4KYGBgCgojIFJ1bm5pbmcgaW4gSW50ZWxsaUoKCkZpcnN0LCB5b3UnbGwgbmVlZCB0byBbc2V0dXAgYSBkZXZlbG9wbWVudCBlbnZpcm9ubWVudF0oaHR0cDovL3d3dy5qZXRicmFpbnMub3JnL2ludGVsbGlqL3Nkay9kb2NzL2Jhc2ljcy9nZXR0aW5nX3N0YXJ0ZWQvc2V0dGluZ191cF9lbnZpcm9ubWVudC5odG1sKS4KClRoZW46CgpgYGAKLi9ncmFkbGV3IHJ1bklkZQpgYGAKCk5PVEU6IFdoZW4gcnVubmluZyBsb2NhbGx5LCBpZiBjaGFuZ2VzIGRvbid0IGFwcGVhciBpbiB0aGUgc2FuZGJveCBpbnN0YW5jZSB5b3UgbWF5IGhhdmUgZm9yZ290dGVuIHRvIHVwZGF0ZSB0aGUgcGx1Z2luIHZlcnNpb24gaW4gYGdyYWRsZS5wcm9wZXJ0aWVzYC4KCiMgVXNhZ2UKCk9wZW4gYSBTcGVjaWZpY2F0aW9uIFlBTUwgb3IgSlNPTiBmaWxlIGluIEludGVsbGlKLiBTdXBwb3J0ZWQgc3BlY2lmaWNhdGlvbnMgYXJlOiBPcGVuQVBJIDIuMC9PcGVuQVBJIDMuMC4gV2l0aCB0aGUgZG9jdW1lbnQgYWN0aXZlLCBuYXZpZ2F0ZSB0byBDb2RlIC0+IE9wZW5BUEkgLT4gR2VuZXJhdGUgZnJvbSBEb2N1bWVudC4KCkNob29zZSB5b3VyIGRlc2lyZWQgZ2VuZXJhdG9yIGFuZCBmaWxsIG91dCBvcHRpb25zLiBUaGVuLCBjaG9vc2UgYW4gb3V0cHV0IGRpcmVjdG9yeSBhbmQgZ2VuZXJhdGUuCgojIyBMaWNlbnNlCgpUaGlzIHByb2plY3QgaXMgbGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSAyLjAgbGljZW5zZS4gU2VlIFsuL0xJQ0VOU0VdKExJQ0VOU0UpIGZvciBtb3JlIGRldGFpbHMuCgpTd2FnZ2Vywq4gaXMgYSByZWdpc3RlcmVkIHRyYWRlbWFyayBvZiBTbWFydEJlYXIgU29mdHdhcmUsIEluYy4KClRoaXMgcHJvamVjdCBpcyBub3QgbWFpbnRhaW5lZCBieSwgb3IgYXNzb2NpYXRlZCB3aXRoLCBTbWFydEJlYXIgU29mdHdhcmUsIEluYy4KVGhpcyBwcm9qZWN0IGlzIG5vdCBtYWludGFpbmVkIGJ5LCBvciBhc3NvY2lhdGVkIHdpdGgsIE9wZW5BUEkgSW5pdGlhdGl2ZSAoT0FJKSBpbiBhbnl3YXkuCg== readmeEtag: '"c19f0115de2fcf02ed9589470f9238a7556b4008"' readmeLastModified: Thu, 02 Jul 2020 02:27:56 GMT repositoryId: 61175252 description: Intellij Plugin for openapi-generator created: '2016-06-15T03:38:07Z' updated: '2025-08-15T07:38:32Z' language: Kotlin archived: false stars: 78 watchers: 4 forks: 10 owner: jimschubert logo: https://avatars.githubusercontent.com/u/109659?v=4 license: Apache-2.0 repoEtag: '"8d4ff19c7d3a073bd37020b4867215f5608c159e774902bbf75090b5fa1041b6"' repoLastModified: Fri, 15 Aug 2025 07:38:32 GMT foundInMaster: true category: - SDK - Parsers id: 6f1cc2a930b656d771fb7dcdb2527d95 - source: - openapi3 tags - openapi31 tags repository: https://github.com/enzonotario/vitepress-openapi v3: true v3_1: true id: 6b0b55df8cbd772582e7b1db14781e00 repositoryMetadata: base64Readme: >- IyB2aXRlcHJlc3Mtb3BlbmFwaQoKR2VuZXJhdGUgVml0ZVByZXNzIEFQSSBEb2N1bWVudGF0aW9uIGZyb20gT3BlbkFQSSBTcGVjaWZpY2F0aW9uLgoKIyMgRG9jdW1lbnRhdGlvbgoKVG8gZ2V0IHN0YXJ0ZWQsIGNoZWNrIG91dCB0aGUgW2RvY3VtZW50YXRpb25dKGh0dHBzOi8vdml0ZXByZXNzLW9wZW5hcGkudmVyY2VsLmFwcC8pLgoKIyMgQ29udHJpYnV0b3JzCgo8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vZW56b25vdGFyaW8vdml0ZXByZXNzLW9wZW5hcGkvZ3JhcGhzL2NvbnRyaWJ1dG9ycyI+CiAgPGltZyBzcmM9Imh0dHBzOi8vY29udHJpYi5yb2Nrcy9pbWFnZT9yZXBvPWVuem9ub3RhcmlvL3ZpdGVwcmVzcy1vcGVuYXBpIiAvPgo8L2E+CgpNYWRlIHdpdGggW2NvbnRyaWIucm9ja3NdKGh0dHBzOi8vY29udHJpYi5yb2NrcykuCgojIyBMaWNlbnNlCgpbTUlUIExpY2Vuc2VdKC4vTElDRU5TRSkgwqkgMjAyMy1wcmVzZW50IFtFbnpvIE5vdGFyaW9dKGh0dHBzOi8vZ2l0aHViLmNvbS9lbnpvbm90YXJpbykuCg== readmeEtag: '"7d9450d92c42307a858bb5f2c5c52d93c19d850f"' readmeLastModified: Mon, 30 Jun 2025 01:16:25 GMT repositoryId: 735578513 description: Generate VitePress API Docs from OpenAPI specifications created: '2023-12-25T12:25:21Z' updated: '2026-02-06T02:40:07Z' language: TypeScript archived: false stars: 212 watchers: 2 forks: 37 owner: enzonotario logo: https://avatars.githubusercontent.com/u/10469299?v=4 license: MIT repoEtag: '"f6cdfd6cdac17cff220fc4c7255811c2e21b27ebca8e4bdf8f828aae6cc5bf74"' repoLastModified: Fri, 06 Feb 2026 02:40:07 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/enzonotario/vitepress-theme-openapi - source: - openapi3 tags - openapi31 tags repository: https://github.com/swagger-api/apidom v3: true v3_1: true id: 1f412460f4d4f6b023e9a61816b07932 repositoryMetadata: base64Readme: >- [comment]: <> (SPDX-FileCopyrightText: Copyright &#40;c&#41; 2015 refractproject)
[comment]: <> (SPDX-License-Identifier: MIT)

[comment]: <> (SPDX-FileCopyrightText: Copyright &#40;c&#41; 2015 Apiary Inc.)
[comment]: <> (SPDX-License-Identifier: MIT)

# ApiDOM

[![Build Status](https://github.com/swagger-api/apidom/actions/workflows/build.yml/badge.svg)](https://github.com/swagger-api/apidom/actions)
[![Dependabot enabled](https://badgen.net/badge/icon/dependabot?icon=dependabot&label)](https://docs.github.com/en/code-security/supply-chain-security/keeping-your-dependencies-updated-automatically)


The purpose of ApiDOM is to provide a single, unifying structure for describing APIs across
API description language and serialization formats. There currently exists several API description languages one can choose
when defining an API, from OpenAPI, RAML, API Blueprint or others.
There are also many serialization formats such as XML, YAML or JSON. Without a way to parse these formats
to the same structure, developers are required to handle each format one-by-one, each in a different
way and each translating to their internal domain model. This is tedious, time-consuming,
and requires each maintainer to stay in step with every format they support.

ApiDOM solves this complex problem in a simple way. It allows parsers to parse to a single structure
and allows tool builders to consume one structure for all formats.

## Table of Contents

- [Getting started](#getting-started)
  - [Prerequisites](#prerequisites)
  - [Installation](#installation)
  - [Usage](#usage)
  - [ApiDOM Playground](#apidom-playground)
- [Development](#development)
  - [Prerequisites](#prerequisites)
  - [Setting up](#setting-up)
  - [Setting up via docker](#setting-up-via-docker)
  - [Setting up via GitHub Codespaces](#setting-up-via-github-codespaces)
  - [npm scripts](#npm-scripts)
  - [Build artifacts](#build-artifacts)
  - [Using this monorepo as a local dev dependency](#using-this-monorepo-as-a-local-dev-dependency)
- [Contributing](#contributing)
- [Documentation](#documentation)
  - [What is an Element ?](#what-is-an-element-)
  - [As a way to annotate JSON](#as-a-way-to-annotate-json)
  - [As a unifying structure](#as-a-unifying-structure)
  - [As a queryable structure](#as-a-queryable-structure)
  - [ApiDOM stages](#apidom-stages)
- [ApiDOM packages](#apidom-packages)
  - [ApiDOM AST](#apidom-ast)
  - [ApiDOM Core](#apidom-core)
  - [Retrieving ApiDOM Elements](#retrieving-apidom-elements)
  - [ApiDOM Namespaces](#apidom-namespaces)
  - [ApiDOM Parsers](#apidom-parsers)
  - [Advanced ApiDOM Manipulations](#advanced-apidom-manipulations)
  - [ApiDOM Language Service](#apidom-language-service)
- [License](#license)
- [Software Bill Of Materials (SBOM)](#software-bill-of-materials-sbom)

## Getting started

### Prerequisites

These prerequisites are required both for installing ApiDOM as a npm package and local development setup.

- [node-gyp](https://www.npmjs.com/package/node-gyp) `>=10` with [Python 3.x](https://www.python.org/downloads/)
- [GLIBC](https://www.gnu.org/software/libc/) `>=2.29`
- [GCC compiler](https://gcc.gnu.org/)
- [emscripten](https://emscripten.org/docs/getting_started/downloads.html) or [docker](https://www.docker.com/) needs to be installed, we recommend going with a docker option

### Installation

Assuming [prerequisites](#prerequisites) are already installed, ApiDOM npm packages are installable and works with `Node.js >=12.20.0 <=22`.

You can install ApiDOM packages using [npm CLI](https://docs.npmjs.com/cli):

```sh
 $ npm install @swagger-api/apidom-ast
 $ npm install @swagger-api/apidom-converter
 $ npm install @swagger-api/apidom-core
 $ npm install @swagger-api/apidom-error
 $ npm install @swagger-api/apidom-json-path
 $ npm install @swagger-api/apidom-json-pointer
 $ npm install @swagger-api/apidom-json-pointer-relative
 $ npm install @swagger-api/apidom-logging
 $ npm install @swagger-api/apidom-ls
 $ npm install @swagger-api/apidom-ns-api-design-systems
 $ npm install @swagger-api/apidom-ns-arazzo-1
 $ npm install @swagger-api/apidom-ns-asyncapi-2
 $ npm install @swagger-api/apidom-ns-json-schema-2019-09
 $ npm install @swagger-api/apidom-ns-json-schema-2020-12
 $ npm install @swagger-api/apidom-ns-json-schema-draft-4
 $ npm install @swagger-api/apidom-ns-json-schema-draft-6
 $ npm install @swagger-api/apidom-ns-json-schema-draft-7
 $ npm install @swagger-api/apidom-ns-openapi-2
 $ npm install @swagger-api/apidom-ns-openapi-3-0
 $ npm install @swagger-api/apidom-ns-openapi-3-1
 $ npm install @swagger-api/apidom-parser
 $ npm install @swagger-api/apidom-parser-adapter-api-design-systems-json
 $ npm install @swagger-api/apidom-parser-adapter-api-design-systems-yaml
 $ npm install @swagger-api/apidom-parser-adapter-arazzo-json-1
 $ npm install @swagger-api/apidom-parser-adapter-arazzo-yaml-1
 $ npm install @swagger-api/apidom-parser-adapter-asyncapi-json-2
 $ npm install @swagger-api/apidom-parser-adapter-asyncapi-yaml-2
 $ npm install @swagger-api/apidom-parser-adapter-asyncapi-json-3
 $ npm install @swagger-api/apidom-parser-adapter-asyncapi-yaml-3
 $ npm install @swagger-api/apidom-parser-adapter-json
 $ npm install @swagger-api/apidom-parser-adapter-json-schema-json-2020-12
 $ npm install @swagger-api/apidom-parser-adapter-json-schema-yaml-2020-12
 $ npm install @swagger-api/apidom-parser-adapter-openapi-json-2
 $ npm install @swagger-api/apidom-parser-adapter-openapi-json-3-0
 $ npm install @swagger-api/apidom-parser-adapter-openapi-json-3-1
 $ npm install @swagger-api/apidom-parser-adapter-openapi-yaml-2
 $ npm install @swagger-api/apidom-parser-adapter-openapi-yaml-3-0
 $ npm install @swagger-api/apidom-parser-adapter-openapi-yaml-3-1
 $ npm install @swagger-api/apidom-parser-adapter-yaml-1-2
 $ npm install @swagger-api/apidom-reference
```

### Usage

Every package of the monorepo has an associated README file demonstrating its purpose and containing
usage examples. An overview of available packages is provided in the [ApiDOM packages](#apidom-packages) section.

### ApiDOM Playground

ApiDOM Playground is a React application that runs in a browser and can visually demonstrate capabilities
of the ApiDOM. ApiDOM Playground is build and deployed whenever the new commit lands on `main` branch.

![image](https://user-images.githubusercontent.com/193286/145010522-5d85e34b-8d28-4a07-9ee2-b28807a013cd.png)

ApiDOM Playground is available at [https://swagger-api.github.io/apidom/](https://swagger-api.github.io/apidom/)

## Development

This is a monorepo for all ApiDOM packages. All the code is written in [TypeScript](https://www.typescriptlang.org/).
All the information necessary for working with monorepo can be found in this [article](https://vladimirgorej.com/blog/things-i-have-learned-maintaining-javascript-monorepo-with-lerna/).

Assuming [prerequisites](#prerequisites) are already installed, [Node.js](https://nodejs.org/) `>=22.14.0` and `npm >=10.9.2`
are the minimum required versions that this repo runs on, but we recommend using the latest version of Node.js@22.

### Setting up

Run the following commands to setup the repository for local development:

```shell
 $ git clone https://github.com/swagger-api/apidom.git
 $ cd apidom
 $ npm i
 $ npm run build
```

### Setting up via docker

There are situations, when satisfying all the **prerequisites** of this repository on you local
development machine is just not possible. In that case, you can use **docker** to get around it.
Repository directory is mounted as volume inside a running container called `apidom-dev`.
That way you can edit code locally on your development machine and run **npm scripts**
inside the `apidom-dev` docker container.

**Build the ApiDOM docker image:**

```sh
 $ git clone https://github.com/swagger-api/apidom.git
 $ cd apidom
 $ docker-compose up
```

**Install dependencies and build ApiDOM inside the docker container:**

```sh
$ docker exec -it apidom-dev npm i --verbose
$ docker exec -it apidom-dev npm run build
```

**Run npm scripts inside the docker container:**

```sh
$ docker exec -it apidom-dev npm run test
```

> Note: monorepo needs to be build in order for monorepo package topology to work correctly.

### Setting up via GitHub Codespaces

This repository is [configured](https://github.com/swagger-api/apidom/tree/main/.devcontainer) to work with [GitHub Codespaces](https://github.com/codespaces).
Create a new codespace by picking this repository from the list of available repositories.
Once the codespace is created, run following commands inside the codespace terminal:

```sh
 $ npm i
 $ npm run build
```

### npm scripts

Some npm scripts run in parallel. Default maximum parallelization is set `2`. This is due to the fact
that our `CI` runs on GitHub Actions which uses GitHub hosted runners with [2-core CPUs](https://docs.github.com/en/actions/using-github-hosted-runners/about-github-hosted-runners#supported-runners-and-hardware-resources).
If you have computer with more than 2 CPU cores, you can speed running npm scripts by
creating an environment variable called `CPU_CORES` and assign it a number of your CPU cores.

Assuming 4 CPU cores are available:

```sh
  $ export CPU_CORES=4
  $ npm run build
```

`build` scripts now runs much faster than before.

**Build artifacts**

```sh
 $ npm run build
```

**Test**

You must first **build the artifacts** before running tests.

```sh
 $ npm run test
```

**Lint**

```sh
 $ npm run lint
```

**Check TypeScript types**

```sh
 $ npm run typescript:check-types
```

**Generate TypeScript types**

```sh
 $ npm run typescript:declaration
```

**Clean**

```sh
 $ npm run clean
```

### Build artifacts

All the packages have identical build system and expose build artifacts in identical way.
After building artifacts, every package will contain five (5) additional directories.
All the build artifacts are polymorphic - they can run in different environments like [Web Browser](https://en.wikipedia.org/wiki/Web_browser), [Node.js](https://nodejs.org/) or [Web Worker](https://developer.mozilla.org/en-US/docs/Web/API/Web_Workers_API).

***.cjs**

These files are generated inside `src/` directory.
Contain ES5 compatible code with [CommonJS](https://en.wikipedia.org/wiki/CommonJS) style imports.
These build fragments are ideal for legacy [Node.js](https://nodejs.org/) and similar environments.

***.mjs**

These files are generated inside `src/` directory.
Contain ES5 compatible code with [ES6 imports](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Statements/import).
These build fragments are ideal for modern [Node.js](https://nodejs.org/),
bundling with [Webpack](https://webpack.js.org/) or similar bundlers.

**dist/**

This directory contains bundled build fragments that use [UMD](https://github.com/umdjs/umd) modules.
They're ideal for browser usage. The fragments are both in minified and un-minified form.

**types/**

TypeScript types generated from the source code.

### Using this monorepo as a local dev dependency

For using this monorepo as a local dev dependency for `dependent project`,
following commands needs to be issued inside the monorepo directory after
it has been cloned to a local filesystem:

```sh
 $ npm i
 $ npm run build
 $ npm run link
```
This will install the dependencies, built the monorepo and link all it's packages to
global `node_modules`.

#### Usage in `dependent project`

Now that we have monorepo packages globally linked we can use them in `dependent project`.
Let's say `dependent project` needs to directly use following packages:

- @swagger-api/apidom-ast
- @swagger-api/apidom-core

Issuing following command from inside the `dependent project` will link these packages:

```sh
 $ npm link @swagger-api/apidom-ast @swagger-api/apidom-core
```

If more packages (or all of them) need to be used in `dependent project`, they need to be explicitly
enumerated using above command and separated by single empty space.

Notice that we link packages using single `npm link` command. This is necessary
because of how `npm link` works internally. Always use single `npm link` command with
multiple package names as argument.

**Don't ever do this!**

```sh
 $ npm link @swagger-api/apidom-ast
 $ npm link @swagger-api/apidom-core
```

> Setting up npm script in `dependent project` can help keep things DRY.

#### Cleaning up

##### Dependent project

The best way to unlink monorepo packages from `dependent project` is to run following command
inside the `dependent project`:

```shell
 $ npm i
```

Running `npm i` will remove the links to monorepo packages and install the packages from npm registry.

> Note: running `npm unlink <package-name>` in `dependent project` will remove the link to monorepo package,
but will leave the `dependent project` node_modules in corrupted state as there is no version of the package
installed anymore. Running `npm i` is always a prefered way to restore your node_modules to original state.


##### ApiDOM monorepo

It is not necessary to unlink monorepo packages from global `node_modules`. But if you
want to keep your global `node_modules` tidy you can run the following command in monorepo directory:

```shell
 $ npm run unlink
```

Running above npm script will unlink all monorepo packages from global `node_modules`.

If you want to just unlink particular monorepo packages, you have to enumerate them explicitly:

```shell
 $ npm unlink --global @swagger-api/apidom-ast @swagger-api/apidom-core
```

## Contributing

This project uses [swagger-api](https://github.com/swagger-api) GitHub organizations contributing guide.
You can obtain copy of this contributing guide at [https://github.com/swagger-api/.github/blob/master/CONTRIBUTING.md](https://github.com/swagger-api/.github/blob/master/CONTRIBUTING.md).
Read our contributing guide to learn about our development process, how to propose bugfixes and improvements,
and how to build and test your changes to ApiDOM.

## Documentation

If there is one thing API description languages have taught us, it is that a single contract provides
the best and fastest way to design and iterate on an API. Developers building the API can move independently
as they progress towards the defined contract found in the OpenAPI or RAML document.
Conversely, API consumers can build tools for consuming the API based on the API definition document.

This same pattern has proven to be just as valuable for building API description languages and tooling.
ApiDOM is the contract for producing and consuming the many API description languages and serialization formats
and allows everyone to move quickly and independently.

### What is an Element ?

ApiDOM is made up of many small elements that have a rich semantic meaning given their value and context.
An element is an individual piece that makes up an API, and can range from defining a resource to providing
an example of an HTTP request.

The ApiDOM defines elements to be used for:

Describing an API
Describing data structures used within that API
Describing parse results when parsing API-related documents
These elements also seek to provide a way to decouple APIs and their semantics from the implementation details.

The structure of an ApiDOM is recursive by nature. When looking for specific elements,
it is best to traverse the ApiDOM tree to look for a match. Querying the ApiDOM tree will
decouple our code from specific API description language. Also, it decouples our code from the
specific structure of these documents as long as they are semantically equivalent.

### As a way to annotate JSON

ApiDOM provides the ability to take a normal JSON structure and add a layer on top of it for the purpose
of annotating and adding semantic data. Instead of creating an entirely different structure to describe the data,
ApiDOM's approach is to expand the existing structure (we call it "refracting" a structure).
Here is an example to show our point.

Take the following simple JSON object.

```json
{
  "name": "John Doe",
  "email": "john@example.com"
}
```

Using ApiDOM, we can expand this out and add some human-readable titles and descriptions.

```json
{
  "element": "object",
  "content": [
    {
      "element": "member",
      "meta": {
        "title": "Name",
        "description": "Name of a person"
      },
      "content": {
        "key": {
          "element": "string",
          "content": "name"
        },
        "value": {
          "element": "string",
          "content": "John Doe"
        }
      }
    },
    {
      "element": "member",
      "meta": {
        "title": "Email",
        "description": "Email address for the person"
      },
      "content": {
        "key": {
          "element": "string",
          "content": "email"
        },
        "value": {
          "element": "string",
          "content": "john@example.com"
        }
      }
    }
  ]
}
```

We added some semantic data to the existing data, but we did so while retaining the semantic structure of the data
with the object and string elements. **This means there is no semantic difference in the ApiDOM structure and
the original JSON structure**. It also means we can add extra semantics on top of these structural ones.

### As a unifying structure

You may have noticed the similarities between the JSON example above and XML.
XML has elements, attributes, and content. It would be a good question to ask if we simply turned JSON into XML.

ApiDOM is actually meant to provide these cross-format similarities. It means that a JSON structure
may be refracted and converted to XML. It also means an XML document may be converted into ApiDOM.
This also goes for YAML, HTML, CSV, and many other formats. ApiDOM is a way to use refracting to unify these structures.

Let's look at another example, this time refacting XML with ApiDOM.

```xml
<person name="John Doe" email="john@example.com"></person>
```

This example in refracted form would look like the following snippet. Notice that we're using attributes in resulting ApiDOM structure.

```json
{
  "element": "person",
  "attributes": {
    "name": {
      "element": "string",
      "content": "John Doe"
    },
    "email": {
      "element": "string",
      "content": "john@example.com"
    }
  }
}
```

Since we can go back and forth between JSON, YAML, XML, and other formats, we are now able to use same toolset across the different formats.
That means we could use XSLT to transform JSON documents.

### As a queryable structure

ApiDOM is meant to free us from the structure of our documents, similar to how XML does with things
like XPATH or the DOM. It means we can now query JSON documents as if there was an underlying DOM,
which decouples our SDK from our structure and our structure from our data.

### ApiDOM stages

There are three stages to ApiDOM

- Parse stage
- Refract stage
- Generate stage


#### Parse stage

The parse stage takes JSON string and produces ApiDOM structure using the base ApiDOM namespace. There are two phases of parsing:

- Lexical Analysis phase
- Syntactic Analysis phase


##### Lexical Analysis phase

Lexical Analysis will take a JSON string and turn it into a stream of tokens. tree-sitter / web-tree-sitter is used
as an underlying lexical analyzer.

##### Syntactic Analysis

Syntactic Analysis will take a stream of tokens and turn it into an ApiDOM representation.
CST produced by lexical analysis is syntactically analyzed, and ApiDOM structure using base (generic) ApiDOM namespace is produced.
Syntactic analysis can further be direct or indirect. JSON parser has both direct and indirect syntactical analyzers,
but YAML parser only has an indirect one.

###### Direct Syntactical analysis

This analysis directly turns tree-sitter CST into ApiDOM. Single traversal is required, which makes it super performant,
and it's the default analysis used.

###### Indirect Syntactic analysis

This analysis turns trees-sitter CST into JSON AST representation. Then JSON AST is turned into ApiDOM.
Two traversals are required, which makes the indirect analysis less performant than the direct one.
Though less performant, having JSON AST representation allows us to do further complex analysis.

#### Refract stage

The refract stage takes a generic ApiDOM structure (base namespace) and traverses through it, adding, updating,
and removing nodes as it goes along and turning it into semantic ApiDOM structure (like OpenAPI or AsyncAPI).
This is by far the most complex part of ApiDOM. This is where plugins operate.
If plugins are used, additional traversal is currently needed.

#### Generate stage

We can generate JSON/YAML documents from the ApiDOM structure.
It doesn't matter if the original document was originally defined in JSON or YAML.
Generated JSON/YAML documented will have exactly the same semantic information as the original one,
but the style information from the original document is not preserved (white spaces/comments, etc..).

```mermaid
flowchart TD
  %% Input node
  In["JSON/YAML string"]

  %%--- Parse Stage and internals
  subgraph Parse_Stage["Parse Stage"]
    P1["Lexical Analysis<br/>JSON/YAML → CST"]
    SA["Syntactic Analysis"]
    SA1["Direct<br/>*CST → Generic ApiDOM*"]
    SA2["Indirect<br/>*CST → JSON AST → Generic ApiDOM*"]

    P1 --> SA
    SA --> SA1
    SA --> SA2
  end

  %%--- Refract Stage
  subgraph Refract_Stage["Refract Stage"]
    R["Generic ApiDOM + plugins → Semantic ApiDOM"]
  end

  %%--- Generate Stage
  subgraph Generate_Stage["Generate Stage"]
    G["Generate JSON/YAML output<br/>(style info lost)"]
  end

  %% Output node
  Out["JSON/YAML string"]

  %%--- Connections
  In --> Parse_Stage
  Parse_Stage --> Refract_Stage
  Refract_Stage --> Generate_Stage
  Generate_Stage --> Out
```

---

Having said that, this is how JSON OpenAPI 3.1 document gets transformed into ApiDOM:


**with direct syntactic analysis (requires 2 traversals)**
```
JSON string -> tree-sitter CST ->  generic ApiDOM -> OpenAPI 3.1 ApiDOM
```

**with indirect syntactic analysis (requires 3 traversals)**
```
JSON string -> tree-sitter CST -> JSON AST -> generic ApiDOM -> OpenAPI 3.1 ApiDOM
```

**with direct syntactic analysis and additional plugins (requires 3 traversal)**
```
JSON string -> tree-sitter CST -> generic ApiDOM -> OpenAPI 3.1 ApiDOM -> plugins -> OpenAPI 3.1 ApiDOM
```
---

This very closely reflects how [Babel](https://github.com/babel/babel) works ([Babel Plugin Handbook](https://github.com/jamiebuilds/babel-handbook/blob/master/translations/en/plugin-handbook.md)).
Their transform phase is our refract phase. The only difference is that when plugins are involved, our transform phase
requires 2 traversals instead of a single one. We can find a way in the future how to fold these 2 traversals into a single one.

## ApiDOM packages

### ApiDOM AST

[@swagger-api/apidom-ast](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ast) contains tools necessary for [syntactic analysis](#syntactic-analysis), which will take a stream of tokens and turn it into an AST representation. The AST represents the structure of input string in a way that makes it easier to work with. The package contains AST nodes for JSON and YAML 1.2 formats and comes with its own traversal algorithm convenient for traversing CST or AST.

### ApiDOM Core

[@swagger-api/apidom-core](https://github.com/swagger-api/apidom/tree/main/packages/apidom-core) is a package that contains tools for manipulating ApiDOM structures. It provides a base ApiDOM namespace, predicates for all primitive and base namespace elements, and algorithms for transforming and handling ApiDOM structures. The available algorithms allow for transcluding, merging, copying, traversing and transforming ApiDOM structures into its different forms.

### Retrieving ApiDOM Elements

It is possible to evaluate [JSON Pointer](https://github.com/swagger-api/apidom/tree/main/packages/apidom-json-pointer), [Relative JSON Pointer](https://github.com/swagger-api/apidom/tree/main/packages/apidom-json-pointer-relative) and [JSONPath expressions](https://github.com/swagger-api/apidom/tree/main/packages/apidom-json-path) against ApiDOM structures. A successful evaluation returns the corresponding ApiDOM Element at the specified location.

### ApiDOM Namespaces

The namespace packages provide ApiDOM namespaces for specifications, consisiting of a number of elements implemented on top of primitive ones. The refractor layer in each namespace allows to transform JavaScript structures or generic ApiDOM into elements from that specific namespace. The elements can be traversed using the `visit` function provided by [apidom-core](https://github.com/swagger-api/apidom/tree/main/packages/apidom-core) package.

Available namespaces:

- [API Design Systems](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-api-design-systems)
- [Arazzo 1.0.1](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-arazzo-1)
- [AsyncAPI 2.x.y](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-asyncapi-2)
- [JSON Schema Draft 4/5](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-json-schema-draft-4)
- [JSON Schema Draft 6](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-json-schema-draft-6)
- [JSON Schema Draft 7](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-json-schema-draft-7)
- [JSON Schema 2019-09](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-json-schema-2019-09)
- [JSON Schema 2020-12](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-json-schema-2020-12)
- [OpenAPI 2.0](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-openapi-2)
- [OpenAPI 3.0.x](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-openapi-3-0)
- [OpenAPI 3.1.0](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ns-openapi-3-1)

### ApiDOM Parsers

The parser packages provide parser adapters for specifications in JSON and YAML formats, allowing to transform them into generic ApiDOM or specific ApiDOM namespace.

The base parser adapters support [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-json) and [YAML 1.2](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-yaml-1-2) formats. They are responsible for [lexical and syntactic analysis](#parse-stage), transforming the provided string into a generic ApiDOM structure. These base adapters are extended in parser adapters for specifications, enabling parsing of definitions directly into the ApiDOM namespaces for those specifications.

Available parser adapters for ApiDOM namespaces:

- API Design Systems - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-api-design-systems-json) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-api-design-systems-yaml)
- Arazzo 1.0.1 - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-arazzo-json-1) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-arazzo-yaml-1)
- AsyncAPI 2.x.y - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-asyncapi-json-2) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-asyncapi-yaml-2)
- AsyncAPI 3.x.y - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-asyncapi-json-3) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-asyncapi-yaml-3)
- JSON Schema 2020-12 - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-json-schema-json-2020-12) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-json-schema-yaml-2020-12)
- OpenAPI 2.0 - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-openapi-json-2) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-openapi-yaml-2)
- OpenAPI 3.0.x - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-openapi-json-3-0) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-openapi-yaml-3-0)
- OpenAPI 3.1.0 - [JSON](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-openapi-json-3-1) / [YAML](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser-adapter-openapi-yaml-3-1)

[@swagger-api/apidom-parser](https://github.com/swagger-api/apidom/tree/main/packages/apidom-parser) consumes any parser adapter with compatible shape and provides a unified API for parsing. It contains logic of mapping a source string to appropriate media type and a source string with media type to appropriate ApiDOM namespace.

### Advanced ApiDOM manipulations

[@swagger-api/apidom-reference](https://github.com/swagger-api/apidom/tree/main/packages/apidom-reference) provides algorithms for semantic ApiDOM manipulations. It contains components enabling parsing, resolving, dereferencing and bundling.

The [parse component](https://github.com/swagger-api/apidom/tree/main/packages/apidom-reference#parse-component) implements the default parser plugins, enabling parsing for files located on local filesystems and on the internet. It can parse provided definitions into generic ApiDOM structure or into one of the ApiDOM namespaces.

Available parse strategies:

- JSON format
- YAML 1.2 format
- Binary format
- ApiDOM - JSON
- API Design Systems - JSON / YAML
- Arazzo 1.0.1 - JSON / YAML
- AsyncAPI 2.x.y - JSON / YAML
- OpenAPI 2.0 - JSON / YAML
- OpenAPI 3.0.x - JSON / YAML
- OpenAPI 3.1.0 - JSON / YAML

The [resolve component](https://github.com/swagger-api/apidom/tree/main/packages/apidom-reference#resolve-component) provides file resolution and external resolution sub-components. File resolution contains plugins allowing to read files located on local filesystem and on the internet, and provide their content. External resolution resolves external dependencies of a document using a specific external resolution strategy, providing a set of resolved references as a result.

Available external resolution strategies:

- ApiDOM
- AsyncAPI 2.x.y
- OpenAPI 2.0
- OpenAPI 3.0.x
- OpenAPI 3.1.0

The [dereference component](https://github.com/swagger-api/apidom/tree/main/packages/apidom-reference#dereference-component) transcludes referencing elements (internal or external) with referenced elements, using a specific dereference strategy.

Available dereference strategies:

- ApiDOM
- AsyncAPI 2.x.y
- OpenAPI 2.0
- OpenAPI 3.0.x
- OpenAPI 3.1.0

The [bundle component](https://github.com/swagger-api/apidom/tree/main/packages/apidom-reference#bundle-component) allows to package up resources spread across multiple files in a single file (Compound Document) using a specific bundle strategy.

Available bundle strategies:

- OpenAPI 3.1.0

### ApiDOM Language Service

[ApiDOM Language Service](https://github.com/swagger-api/apidom/tree/main/packages/apidom-ls) provides language processing for ApiDOM supported languages for editing experience. It adheres to the LSP Protocol and can therefore be used via LSP Server wrapper, providing editing capabilities to various editors and IDEs. It provides validation, documentation, definitions, completion, semantic highlighting and dereferencing for supported specifications.

ApiDOM Language Service supports:

- AsyncAPI 2.x.y
- OpenAPI 2.0
- OpenAPI 3.0.x
- OpenAPI 3.1.0

## License

ApiDOM is licensed under [Apache 2.0 license](https://github.com/swagger-api/apidom/blob/main/LICENSES/Apache-2.0.txt).
ApiDOM comes with an explicit [NOTICE](https://github.com/swagger-api/apidom/blob/main/NOTICE) file
containing additional legal notices and information.

This project uses [REUSE specification](https://reuse.software/spec/) that defines a standardized method
for declaring copyright and licensing for software projects.

## Software Bill Of Materials (SBOM)

Software Bill Of materials is available in this repository [dependency graph](https://github.com/swagger-api/apidom/network/dependencies).
Click on `Export SBOM` button to download the SBOM in [SPDX format](https://spdx.dev/).

 readmeEtag: '"68a59077b704640e9a7db8926d1fa525f5facf8d"' readmeLastModified: Mon, 01 Dec 2025 14:07:09 GMT repositoryId: 249737066 description: Semantic parser for API specifications created: '2020-03-24T14:55:51Z' updated: '2026-02-03T14:04:06Z' language: TypeScript archived: false stars: 93 watchers: 6 forks: 20 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 repoEtag: '"4819013eedf96bd3cb38ae666ff9346faabcd92b7d01d8060475f9f324302d74"' repoLastModified: Tue, 03 Feb 2026 14:04:06 GMT category: Parsers foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/stoplightio/spectral-owasp-ruleset v3: true v3_1: true id: a98ab8aaed507481a9bf8be2c5b38a8f repositoryMetadata: base64Readme: >- IyBTcGVjdHJhbCBPV0FTUCBBUEkgU2VjdXJpdHkKClshW05QTSBEb3dubG9hZHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2R3L0BzdG9wbGlnaHQvc3BlY3RyYWwtb3dhc3AtcnVsZXNldD9jb2xvcj1ibHVlKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQHN0b3BsaWdodC9zcGVjdHJhbC1vd2FzcC1ydWxlc2V0KSBbIVtTdG9wbGlnaHQgRm9yZXN0XShodHRwczovL2ltZy5zaGllbGRzLmlvL2Vjb2xvZ2kvdHJlZXMvc3RvcGxpZ2h0aW5jKV1bc3RvcGxpZ2h0X2ZvcmVzdF0KClNjYW4gYW4gW09wZW5BUEldKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCkgZG9jdW1lbnQgdG8gZGV0ZWN0IHNlY3VyaXR5IGlzc3Vlcy4gQXMgT3BlbkFQSSBpcyBvbmx5IGRlc2NyaWJpbmcgdGhlIHN1cmZhY2UgbGV2ZWwgb2YgdGhlIEFQSSBpdCBjYW5ub3Qgc2VlIHdoYXQgaXMgaGFwcGVuaW5nIGluIHlvdXIgY29kZSwgYnV0IGl0IGNhbiBzcG90IG9idmlvdXMgaXNzdWVzIGFuZCBvdXRkYXRlZCBzdGFuZGFyZHMgYmVpbmcgdXNlZC4KCnYyLnggb2YgdGhpcyBydWxlc2V0IGlzIGJhc2VkIG9uIHRoZSBbT1dBU1AgQVBJIFNlY3VyaXR5IFRvcCAxMCAyMDIzIGVkaXRpb25dKGh0dHBzOi8vb3dhc3Aub3JnL0FQSS1TZWN1cml0eS9lZGl0aW9ucy8yMDIzL2VuLzB4MDAtaGVhZGVyLyksIGJ1dCBpZiB5b3Ugd291bGQgbGlrZSB0byB1c2UgdGhlIFsyMDE5IGVkaXRpb25dKGh0dHBzOi8vb3dhc3Aub3JnL0FQSS1TZWN1cml0eS9lZGl0aW9ucy8yMDE5L2VuLzB4MDAtaGVhZGVyLykgcGxlYXNlIHVzZSB2MS54LgoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCm5wbSBpbnN0YWxsIC0tc2F2ZSAtRCBAc3RvcGxpZ2h0L3NwZWN0cmFsLW93YXNwLXJ1bGVzZXRAXjIuMApucG0gaW5zdGFsbCAtLXNhdmUgLUQgQHN0b3BsaWdodC9zcGVjdHJhbC1jbGkKYGBgCgojIyBVc2FnZQoKQ3JlYXRlIGEgbG9jYWwgcnVsZXNldCB0aGF0IGV4dGVuZHMgdGhlIHJ1bGVzZXQuIEluIGl0cyBtb3N0IGJhc2ljIGZvcm0gdGhpcyBqdXN0IHRlbGxzIFNwZWN0cmFsIHdoYXQgcnVsZXNldCB5b3Ugd2FudCB0byB1c2UsIGJ1dCBpdCB3aWxsIGFsbG93IHlvdSB0byBjdXN0b21pc2UgdGhpbmdzLCBhZGQgeW91ciBvd24gcnVsZXMsIHR1cm4gYml0cyBvZmYgaWYgaXRzIGNhdXNpbmcgdHJvdWJsZS4KCmBgYApjZCB+L3NyYy88eW91ci1hcGk+CgplY2hvICdleHRlbmRzOiBbIkBzdG9wbGlnaHQvc3BlY3RyYWwtb3dhc3AtcnVsZXNldCJdJyA+IC5zcGVjdHJhbC55YW1sCmBgYAoKX0lmIHlvdSdyZSB1c2luZyBWUyBDb2RlIG9yIFN0b3BsaWdodCBTdHVkaW8gdGhlbiB0aGUgTlBNIG1vZHVsZXMgd2lsbCBub3QgYmUgYXZhaWxhYmxlLiBJbnN0ZWFkIHlvdSBjYW4gdXNlIHRoZSBDRE4gaG9zdGVkIHZlcnNpb246XwoKYGBgCmVjaG8gJ2V4dGVuZHM6IFsiaHR0cHM6Ly91bnBrZy5jb20vQHN0b3BsaWdodC9zcGVjdHJhbC1vd2FzcC1ydWxlc2V0L2Rpc3QvcnVsZXNldC5tanMiXScgPiAuc3BlY3RyYWwueWFtbApgYGAKCl8qKk5vdGU6KiogWW91IG5lZWQgdG8gdXNlIHRoZSBmdWxsIFVSTCB3aXRoIENETiBob3N0ZWQgcnVsZXNldHMgYmVjYXVzZSBTcGVjdHJhbCBbY2Fubm90IGZvbGxvdyByZWRpcmVjdHMgdGhyb3VnaCBleHRlbmRzXShodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vc3BlY3RyYWwvaXNzdWVzLzIyNjYpLl8KCk5leHQsIHVzZSBTcGVjdHJhbCBDTEkgdG8gbGludCBhZ2FpbnN0IHlvdXIgT3BlbkFQSSBkZXNjcmlwdGlvbi4gRG9uJ3QgaGF2ZSBhbnkgT3BlbkFQST8gW1JlY29yZCBzb21lIEhUVFAgdHJhZmZpYyB0byBtYWtlIE9wZW5BUEldKGh0dHBzOi8vYXBpc3lvdXdvbnRoYXRlLmNvbS9ibG9nL2NyZWF0aW5nLW9wZW5hcGktZnJvbS1odHRwLXRyYWZmaWMpIGFuZCB0aGVuIHlvdSBjYW4gc3dpdGNoIHRvIEFQSSBEZXNpZ24tRmlyc3QgZ29pbmcgZm9yd2FyZHMuCgpgYGAKc3BlY3RyYWwgbGludCBhcGkvb3BlbmFwaS55YW1sCmBgYAoKWW91IHNob3VsZCBzZWUgc29tZSBvdXRwdXQgbGlrZSB0aGlzOgoKYGBgCi9Vc2Vycy9waGlsL3NyYy9wcm90ZWN0LWVhcnRoLWFwaS9hcGkvb3BlbmFwaS55YW1sCiAgNDo1ICAgICAgICBlcnJvciAgICBvd2FzcDphcGk4OjIwMjMtaW52ZW50b3J5LWFjY2VzcyAgICAgICAgICAgIERlY2xhcmUgaW50ZW5kZWQgYXVkaWVuY2Ugb2YgZXZlcnkgc2VydmVyIGJ5IGRlZmluaW5nIHNlcnZlcnNbMF0ueC1pbnRlcm5hbCBhcyB0cnVlL2ZhbHNlLiAgc2VydmVyc1swXQogIDQ6MTAgICAgICAgZXJyb3IgICAgb3dhc3A6YXBpODoyMDIzLW5vLXNlcnZlci1odHRwICAgICAgICAgICAgICBTZXJ2ZXIgVVJMcyBtdXN0IG5vdCB1c2UgaHR0cDovLy4gaHR0cHM6Ly8gaXMgaGlnaGx5IHJlY29tbWVuZGVkLiAgICAgICAgICAgICAgICAgICAgICAgIHNlcnZlcnNbMF0udXJsCiAgNDU6MTUgICAgICAgIGVycm9yICBvd2FzcDphcGk0OjIwMjMtcmF0ZS1saW1pdCAgICAgICAgICAgICAgICAgIEFsbCAyWFggYW5kIDRYWCByZXNwb25zZXMgc2hvdWxkIGRlZmluZSByYXRlIGxpbWl0aW5nIGhlYWRlcnMuICBwYXRocy4vdXBsb2FkLnBvc3QucmVzcG9uc2VzWzIwMV0KICA0NzoxNSAgICAgICAgZXJyb3IgIG93YXNwOmFwaTQ6MjAyMy1yYXRlLWxpbWl0ICAgICAgICAgICAgICAgICAgQWxsIDJYWCBhbmQgNFhYIHJlc3BvbnNlcyBzaG91bGQgZGVmaW5lIHJhdGUgbGltaXRpbmcgaGVhZGVycy4gIHBhdGhzLi91cGxvYWQucG9zdC5yZXNwb25zZXNbNDAxXQogIDkzOjE2ICBpbmZvcm1hdGlvbiAgb3dhc3A6YXBpMjoyMDIzLXJlYWQtcmVzdHJpY3RlZCAgICAgICAgICAgICBUaGlzIG9wZXJhdGlvbiBpcyBub3QgcHJvdGVjdGVkIGJ5IGFueSBzZWN1cml0eSBzY2hlbWUuICBwYXRocy4vc2l0ZXMuZ2V0LnNlY3VyaXR5CiAyMTA6MTYgIGluZm9ybWF0aW9uICBvd2FzcDphcGkyOjIwMjMtcmVhZC1yZXN0cmljdGVkICAgICAgICAgICAgIFRoaXMgb3BlcmF0aW9uIGlzIG5vdCBwcm90ZWN0ZWQgYnkgYW55IHNlY3VyaXR5IHNjaGVtZS4gIHBhdGhzLi9zcGVjaWVzLmdldC5zZWN1cml0eQpgYGAKCk5vdyB5b3UgaGF2ZSBzb21lIHRoaW5ncyB0byB3b3JrIG9uIGZvciB5b3VyIEFQSS4gVGhhbmtmdWxseSB0aGVzZSBhcmUgb25seSBhdCB0aGUgYHdhcm5pbmdgIGFuZCBgaW5mb3JtYXRpb25gIHNldmVyaXR5LCBhbmQgdGhhdCBpcyBub3QgZ29pbmcgdG8gW2ZhaWwgY29udGludW91cyBpbnRlZ3JhdGlvbl0oaHR0cHM6Ly9kb2NzLnN0b3BsaWdodC5pby9kb2NzL3NwZWN0cmFsL1pHOWpPakV4TlRNeU9UQXgtY29udGludW91cy1pbnRlZ3JhdGlvbikgKHVubGVzcyBbeW91IHdhbnQgdGhlbSB0b10oaHR0cHM6Ly9kb2NzLnN0b3BsaWdodC5pby9kb2NzL3NwZWN0cmFsL1pHOWpPakkxTVRnMS1zcGVjdHJhbC1jbGkjZXJyb3ItcmVzdWx0cykpLgoKVGhlcmUgYXJlIFthIGJ1bmNoIG9mIG90aGVyIHJ1bGVzZXRzXShodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vc3BlY3RyYWwtcnVsZXNldHMpIG9yIFtTdG9wbGlnaHQgQVBJIFN0eWxlYm9va10oaHR0cDovL2FwaXN0eWxlYm9vay5zdG9wbGlnaHQuaW8pIHlvdSBjYW4gdXNlLCBvciB1c2UgZm9yIGluc3BpcmF0aW9uIGZvciB5b3VyIG93biBydWxlc2V0cyBhbmQgQVBJIFN0eWxlIEd1aWRlcy4KCiMjIPCfjokgVGhhbmtzCgotIFtBbmRyemVqXShodHRwczovL2dpdGh1Yi5jb20vamVyenluKSAtIEdyZWF0IHJ1bGVzIGNvbnRyaWJ1dGVkIHRvIHRoZSBBZGlkYXMgc3R5bGUgZ3VpZGUuCi0gW1JvYmVydG8gUG9sbGldKGh0dHBzOi8vZ2l0aHViLmNvbS9pb2dnc3RyZWFtKSAtIENyZWF0ZWQgbG90cyBvZiBleGNlbGxlbnQgU3BlY3RyYWwgcnVsZXMgZm9yIFtBUEkgT0FTIENoZWNrZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9pdGFsaWEvYXBpLW9hcy1jaGVja2VyLykgd2hpY2ggYWxpZ25lZCB3aXRoIHRoZSBPV0FTUCBBUEkgcnVsZXMuCgojIyDwn5OcIExpY2Vuc2UKClRoaXMgcmVwb3NpdG9yeSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIGxpY2Vuc2UuCgojIyDwn4yyIFNwb25zb3IKCklmIHlvdSB3b3VsZCBsaWtlIHRvIHRoYW5rIHVzIGZvciBjcmVhdGluZyBTcGVjdHJhbCwgd2UgYXNrIHRoYXQgeW91IFsqKmJ1eSB0aGUgd29ybGQgYSB0cmVlKipdW3N0b3BsaWdodF9mb3Jlc3RdLgoKW3N0b3BsaWdodF9mb3Jlc3RdOiBodHRwczovL2Vjb2xvZ2kuY29tL3N0b3BsaWdodGluYwo= readmeEtag: '"0d92d6543e532164348d50cc4b5837198a046ca8"' readmeLastModified: Tue, 12 Mar 2024 15:49:11 GMT repositoryId: 528967945 description: >- Improve the security of your API by detecting common vulnerabilities as defined by OWASP and enforced with Spectral. created: '2022-08-25T18:14:13Z' updated: '2026-02-04T01:01:12Z' language: TypeScript archived: false stars: 84 watchers: 3 forks: 19 owner: stoplightio logo: https://avatars.githubusercontent.com/u/10767217?v=4 repoEtag: '"cef1fd181398cbb14e9ff4a8aab3f02e313f5697337c4dc000179bd4e47e0783"' repoLastModified: Wed, 04 Feb 2026 01:01:12 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/skellla/fastify-oas v3: true repositoryMetadata: base64Readme: >- # fastify-oas

**NOTE**: OpenAPI 3 support finally [landed](https://github.com/fastify/fastify-swagger/pull/333) in "official" [fastify-swagger](https://github.com/fastify/fastify-swagger) module. Consider using it instead of this one, since it has better support for $ref in schemas.

This plugin is deprecated and no longer mainteined. Feel free to fork it and publish if needed.

<div align="center">
  <img src="https://gitlab.com/m03geek/fastify-oas/raw/master/logo.png" alt="fastify-oas logo"/>
</div>

[![NPM Version](https://img.shields.io/npm/v/fastify-oas.svg)](https://www.npmjs.com/package/fastify-oas)
[![Downloads Count](https://img.shields.io/npm/dm/fastify-oas.svg)](https://www.npmjs.com/package/fastify-oas)
[![Vunerabilities Count](https://snyk.io/test/npm/fastify-oas/badge.svg)](https://www.npmjs.com/package/fastify-oas)
[![Build Status](https://gitlab.com/m03geek/fastify-oas/badges/master/pipeline.svg)](https://gitlab.com/m03geek/fastify-oas/commits/master)
[![Coverage Status](https://gitlab.com/m03geek/fastify-oas/badges/master/coverage.svg)](https://gitlab.com/m03geek/fastify-oas/commits/master)
[![License](https://img.shields.io/npm/l/fastify-oas.svg)](https://gitlab.com/m03geek/fastify-oas/blob/master/LICENSE)

[OpenAPI 3.0+ (OAS3)](https://swagger.io/docs/specification/about/) documentation generator for Fastify.
It uses the schemas you declare in your routes to generate an OpenAPI (swagger) compliant doc.

This plugin based on [fastify-swagger](https://github.com/fastify/fastify-swagger/) that generates swagger 2.0 docs.

This plugin designed in such way to be compatible with it's predcessor and in most cases if you already use `fastify-swagger` you may just replace it with current plugin and it should work.

## ToC
- [fastify-oas](#fastify-oas)
  - [ToC](#toc)
  - [Fastify support](#fastify-support)
  - [Installation](#installation)
  - [Features and requirements](#features-and-requirements)
  - [Usage](#usage)
    - [Docs](#docs)
    - [Plugin options](#plugin-options)
    - [Additional schema options](#additional-schema-options)
    - [OpenAPI](#openapi)
    - [Swagger 2.0](#swagger-20)
  - [UI](#ui)
  - [Development](#development)
  - [See also](#see-also)
  - [License](#license)

## Fastify support

- v0.X.X - v1.X.X - supports fastify v1.X.X
- v2.X.X - will support fastify v2.X.X*

<span style="color:red; font-size:2em;">
Currently in Fastify v2.12.0 there's regression bug that breakes this exenstion for many users. So for now this extension doesn't support fastify 2.12.0. Cause it was cause by fastify minor change in order to preven issues this library will throw an error when you'll try to use it with that fastify version.
In order to use it, you can either lock your fastify at 2.11.0 or fastify-oas at 2.5.0 (but there are no gaurantee that it will work correctly).
</span>


## Installation

```sh
npm i fastify-oas --save
```

<sub>[Back to top](#toc)</sub>

## Features and requirements

* Supports OpenAPI 3+.
* Supports [fastify-swagger](https://github.com/fastify/fastify-swagger/) module configs.
* Supports swagger 2.0 fastify schemas.
* Supports fastify named schemas convertion to swaagger/openapi models.

--- 

* Requires fastify `>=1.9.0`.
* Node.js `>=8.9.0`.

NOTE: If you need to generate fastify routes from your swagger document - please refer to plugins in [See also](#see-also) like fastify-swaggergen or fastify-openapi-glue.

<sub>[Back to top](#toc)</sub>

## Usage

Add it to your project like regular fastify plugin. Use `register` method and pass it swagger options, then call the api constructor.

```js
const fastify = require('fastify');
const oas = require('fastify-oas');
const app = fastify();

app.register(oas, {
  routePrefix: '/documentation',
  swagger: {
    info: {
      title: 'Test openapi',
      description: 'testing the fastify swagger api',
      version: '0.1.0',
    },
    externalDocs: {
      url: 'https://swagger.io',
      description: 'Find more info here',
    },
    consumes: ['application/json'],
    produces: ['application/json'],
  },
  exposeRoute: true
});

// put some routes and schemas

app.ready(err => {
  if (err) throw err;
  app.oas();
});
```

Please note, the schema format for Fastify routes is [JSONSchema](https://github.com/fastify/fastify/blob/v2.13.0/docs/Routes.md#routes-option) and you may encounter some differences in the format for route spec vs. output OpenAPI spec.  
This plugin includes handling around a few of these differences.

One such case is the `example` or `examples` keywords:
```js
fastify.route({
  method: 'POST',
  url: '/',
  schema: {
    body: {
      type: 'object',
      description: 'an object',
      examples: [
          {
            name: 'Object Sample',
            summary: 'an example',
            value: {a: 'payload'},
          }
      ],
      properties: {
        a: {type: 'string', description: 'your payload'}
      }
    }
  },
  handler: // ...
})
```
Which produces a spec similar to:
```json
{
  ... 

  "content": {
    "application/json": {
      "examples": {
        "Object Sample": {
          "summary": "an example",
          "value": {
            "a": "payload"
          }
        }
      },
      "schema": {
        "type": "object",
        "properties": {
          "a": {
            "type": "string",
            "description": "your payload"
          }
        }
      }
    }
  }
}
```
(in this case, the name property is extracted as the examples key)

<sub>[Back to top](#toc)</sub>

### Docs

See [Docs](/docs/README.md) for more details on the TypeScript types that you may use when working with OpenAPI spec.

### Plugin options

| parameter      | type    | description                                                                                                                                | default          |
| -------------- | ------- | ------------------------------------------------------------------------------------------------------------------------------------------ | ---------------- |
| `routePrefix`  | String  | Documentation endpoint                                                                                                                     | `/documentation` |
| `exposeRoute`  | Boolean|Object** | If `true` the plugin will expose the documentation with the following apis: `/<routePrefix>`, `/<routePrefix>/json`, `/<routePrefix>/yaml` | `false`          |
| `addModels`    | Boolean | If `true` adds fastify schemas as openapi models*                                                                                          | `false`          |
| `openapi`      | String  | Openapi version                                                                                                                            | '3.0.0'          |
| `yaml`         | Boolean | If `true` returns yaml instead of json                                                                                                     | `false`          |
| `hideUntagged` | Boolean | If `true` remove routes without tags in schema from resulting swagger file                                                                 | `false`          |
| `swagger`      | Object  | Swagger object except paths                                                                                                                | `{}`             |

Note (*): Fastify-oas plugin gather all schemas, so you should ensure that all of them under current and nested scopes have unique names.
Note (**): see [Expose route options](/docs/interfaces/fastifyoas.exposeoptions.md)
<sub>[Back to top](#toc)</sub>

### Additional schema options

In order to remove some endpoints from Swagger/OpenAPI document you may add `{hide: true}` option to route schema.

```js
const fastify = require('fastify')()
fastify.get('/some-secrete-route/:id', {
  schema: {
    hide: true,
    params: {
      type: 'object',
      properties: {
        id: {
          type: 'string',
          description: 'user id'
        }
      }
    },
    response: {
      201: {
        description: 'Successful response',
        type: 'object',
        properties: {
          hello: { type: 'string' }
        }
      }
    },
  }
}, (req, reply) => {})

```

<sub>[Back to top](#toc)</sub>

### OpenAPI

Unlike regular OpenAPI spec you'll still need some options from swagger 2.0.

```js
const fastify = require('fastify');
const oas = require('fastify-oas');
const app = fastify();

app.register(oas, {
  routePrefix: '/documentation',
  swagger: {
    info: {
      title: 'Test openapi',
      description: 'testing the fastify swagger api',
      version: '0.1.0',
    },
    externalDocs: {
      url: 'https://swagger.io',
      description: 'Find more info here',
    },
    consumes: ['application/json'], // app-wide default media-type
    produces: ['application/json'], // app-wide default media-type
    servers: [{
      url: 'http://api.example.com/v1',
      description: 'Optional server description, e.g. Main (production) server',
    }],
    components: {
      // see https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#componentsObject for more options
      securitySchemes: {
        BasicAuth: {
          type: 'http',
          scheme: 'basic',
        },
      },
    }, 
  },
});
```

<sub>[Back to top](#toc)</sub>

### Swagger 2.0

This will not generate swagger 2.0 docs. It will generate openapi 3.0 docs, but from swagger 2.0 (and fastify-swagger) compatible configuration.
It will allow easily migrate from fastify-swagger.

The example below is taken from fastify-swagger repo to show the differences .

```js
const fastify = require('fastify')()

// before: fastify.register(require('fastify-swagger'), {
fastify.register(require('fastify-oas'), { // after
  routePrefix: '/documentation',
  swagger: {
    info: {
      title: 'Test swagger',
      description: 'testing the fastify swagger api',
      version: '0.1.0'
    },
    externalDocs: {
      url: 'https://swagger.io',
      description: 'Find more info here'
    },
    host: 'localhost',
    schemes: ['http'],
    consumes: ['application/json'],
    produces: ['application/json'],
    tags: [
      { name: 'user', description: 'User related end-points' },
      { name: 'code', description: 'Code related end-points' }
    ],
    securityDefinitions: {
      apiKey: {
        type: 'apiKey',
        name: 'apiKey',
        in: 'header'
      }
    }
  }
})

fastify.put('/some-route/:id', {
  schema: {
    description: 'post some data',
    tags: ['user', 'code'],
    summary: 'qwerty',
    params: {
      type: 'object',
      properties: {
        id: {
          type: 'string',
          description: 'user id'
        }
      }
    },
    body: {
      type: 'object',
      properties: {
        hello: { type: 'string' },
        obj: {
          type: 'object',
          properties: {
            some: { type: 'string' }
          }
        }
      }
    },
    response: {
      201: {
        description: 'Successful response',
        type: 'object',
        properties: {
          hello: { type: 'string' }
        }
      }
    },
    security: [
      {
        "api_key": []
      }
    ]
  }
}, (req, reply) => {})

fastify.ready(err => {
  if (err) throw err
  fastify.oas()
})
```

<sub>[Back to top](#toc)</sub>

## UI

Swagger UI is available via `/<routePrefix>/index.html`. By default it's `/documentation/index.html`.

ReDoc UI is available via `/<routePrefix>/docs.html`. By default it's `/documentation/docs.html`.

<sub>[Back to top](#toc)</sub>

## Development

In order to start development run:

```sh
npm i
npm run prepare
```

So that [swagger-ui](https://github.com/swagger-api/swagger-ui) static folder will be generated for you.

<sub>[Back to top](#toc)</sub>

## See also

* [fastify-swagger](https://github.com/fastify/fastify-swagger) - swagger 2.0 docs generation plugin.
* [fastify-swaggergen](https://github.com/seriousme/fastify-swaggergen) - fastify routes generation from swagger 2.0 docs.
* [fastify-openapi-glue](https://github.com/seriousme/fastify-openapi-glue) - fastify-swaggergen successor, generates fastify routes from swagger 2.0 and openapi 3.0 docs (just like this module, but in opposite direction).

<sub>[Back to top](#toc)</sub>

## License

Licensed under [MIT](./LICENSE).

<sub>[Back to top](#toc)</sub>
 readmeEtag: '"383de813919532cd88e4767385f02dfc62359d26"' readmeLastModified: Thu, 03 Nov 2022 12:15:58 GMT repositoryId: 142213233 description: Fastify OpenAPI plugin. created: '2018-07-24T21:09:52Z' updated: '2025-03-16T00:16:44Z' language: JavaScript archived: false stars: 73 watchers: 4 forks: 20 owner: SkeLLLa logo: https://avatars.githubusercontent.com/u/2273103?v=4 license: MIT repoEtag: '"cb0636c33ed6f76660d0ff9ff746aeeb84bee2549aeafdc807519c0ef464f76f"' repoLastModified: Sun, 16 Mar 2025 00:16:44 GMT foundInMaster: true id: 0bda0b3ad75df5840c48691fe579a850 - source: openapi3 tags repository: https://github.com/manchenkoff/openapi3-parser v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFBhcnNlcgoKWyFbUHlQSSAtIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS92L29wZW5hcGkzLXBhcnNlcildKGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9vcGVuYXBpMy1wYXJzZXIvKQpbIVtQeVBJIC0gRG93bmxvYWRzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvZG0vb3BlbmFwaTMtcGFyc2VyKV0oaHR0cHM6Ly9jbGlja3B5LmNsaWNraG91c2UuY29tL2Rhc2hib2FyZC9vcGVuYXBpMy1wYXJzZXIpClshW1B5UEkgLSBQeXRob24gVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3B5dmVyc2lvbnMvb3BlbmFwaTMtcGFyc2VyKV0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L29wZW5hcGkzLXBhcnNlci8pClshW1B5UEkgLSBGb3JtYXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9mb3JtYXQvb3BlbmFwaTMtcGFyc2VyKV0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L29wZW5hcGkzLXBhcnNlci8pClshW1B5UEkgLSBMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvbC9vcGVuYXBpMy1wYXJzZXIpXShsaWNlbnNlLnR4dCkKCkEgc2ltcGxlIHBhY2thZ2UgdG8gcGFyc2UgeW91ciBPcGVuQVBJIDMgZG9jdW1lbnRzIGludG8gUHl0aG9uIG9iamVjdCB0byB3b3JrIHdpdGguCgpTdXBwb3J0ZWQgdmVyc2lvbnM6Cgp8IFZlcnNpb24gfCBTdGF0dXMgICAgICAgICB8CnwgLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0tIHwKfCAyLjAgICAgIHwgRGVwcmVjYXRlZCAgICAgfAp8IDMuMCAgICAgfCAqKlN1cHBvcnRlZCoqICB8CnwgMy4xICAgICB8IEluIGRldmVsb3BtZW50IHwKCiMjIEhvdyB0byBpbnN0YWxsCgpUbyBpbnN0YWxsIHBhY2thZ2UgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZAoKYGBgCnBpcCBpbnN0YWxsIG9wZW5hcGkzLXBhcnNlcgpgYGAKCiMjIEhvdyB0byB1c2UKCkV4YW1wbGUgb2YgcGFyc2VyIHVzYWdlCgpgYGAKPj4+IGZyb20gb3BlbmFwaV9wYXJzZXIgaW1wb3J0IHBhcnNlCj4+PiBjb250ZW50ID0gcGFyc2UoJ3N3YWdnZXIueW1sJykKPj4+IHByaW50KGNvbnRlbnQpCmBgYAoKR2V0IGFwcGxpY2F0aW9uIHNlcnZlcnMKCmBgYHB5dGhvbgpmcm9tIG9wZW5hcGlfcGFyc2VyIGltcG9ydCBwYXJzZQoKc3BlY2lmaWNhdGlvbiA9IHBhcnNlKCdkYXRhL3N3YWdnZXIueW1sJykKCnByaW50KCJBcHBsaWNhdGlvbiBzZXJ2ZXJzIikKCmZvciBzZXJ2ZXIgaW4gc3BlY2lmaWNhdGlvbi5zZXJ2ZXJzOgogICAgcHJpbnQoZiJ7c2VydmVyLmRlc2NyaXB0aW9ufSAtIHtzZXJ2ZXIudXJsfSIpCgojIE91dHB1dAojCiMgPj4gQXBwbGljYXRpb24gc2VydmVycwojID4+IHByb2R1Y3Rpb24gLSBodHRwczovL3VzZXJzLmFwcAojID4+IHN0YWdpbmcgLSBodHRwOi8vc3RhZ2UudXNlcnMuYXBwCiMgPj4gZGV2ZWxvcG1lbnQgLSBodHRwOi8vdXNlcnMubG9jYWwKYGBgCgpHZXQgbGlzdCBvZiBhcHBsaWNhdGlvbiBVUkxzCgpgYGBweXRob24KZnJvbSBvcGVuYXBpX3BhcnNlciBpbXBvcnQgcGFyc2UKCnNwZWNpZmljYXRpb24gPSBwYXJzZSgndGVzdHMvZGF0YS9zd2FnZ2VyLnltbCcpCgp1cmxzID0gW3gudXJsIGZvciB4IGluIHNwZWNpZmljYXRpb24ucGF0aHNdCgpwcmludCh1cmxzKQoKIyBPdXRwdXQKIwojID4+IFsnL3VzZXJzJywgJy91c2Vycy97dXVpZH0nXQpgYGAKCkdldCBvcGVyYXRpb24gd2l0aCBzdXBwb3J0ZWQgSFRUUCBtZXRob2RzCgpgYGBweXRob24KZnJvbSBvcGVuYXBpX3BhcnNlciBpbXBvcnQgcGFyc2UKCnNwZWNpZmljYXRpb24gPSBwYXJzZSgndGVzdHMvZGF0YS9zd2FnZ2VyLnltbCcpCgpmb3IgcGF0aCBpbiBzcGVjaWZpY2F0aW9uLnBhdGhzOgogICAgc3VwcG9ydGVkX21ldGhvZHMgPSAnLCcuam9pbihbeC5tZXRob2QudmFsdWUgZm9yIHggaW4gcGF0aC5vcGVyYXRpb25zXSkKCiAgICBwcmludChmIk9wZXJhdGlvbjoge3BhdGgudXJsfSwgbWV0aG9kczoge3N1cHBvcnRlZF9tZXRob2RzfSIpCgojIE91dHB1dAojCiMgPj4gT3BlcmF0aW9uOiAvdXNlcnMsIG1ldGhvZHM6IGdldCxwb3N0CiMgPj4gT3BlcmF0aW9uOiAvdXNlcnMve3V1aWR9LCBtZXRob2RzOiBnZXQscHV0CmBgYAo= readmeEtag: '"6597b342af7ba10b333a6781f10ca6b41c064ded"' readmeLastModified: Fri, 25 Apr 2025 11:01:49 GMT repositoryId: 290637240 description: >- OpenAPI 3 parser to use a specification inside of the code in your projects created: '2020-08-27T00:47:00Z' updated: '2025-12-30T17:42:57Z' language: Python archived: false stars: 85 watchers: 1 forks: 41 owner: manchenkoff logo: https://avatars.githubusercontent.com/u/6690063?v=4 license: MIT repoEtag: '"4fe06daedaa79bb3969746cb7c71c99110f3378dd57e5e74e39fb2fb3215d1b8"' repoLastModified: Tue, 30 Dec 2025 17:42:57 GMT foundInMaster: true category: Parsers id: 074facec95becd6c9d361eb75f2d1d2f - source: openapi3 tags repository: https://github.com/meteran/gnext v3: true repositoryMetadata: base64Readme: >- IyBnTmV4dCBXZWIgRnJhbWV3b3JrCgpbIVtHbyBSZXBvcnQgQ2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vbWV0ZXJhbi9nbmV4dCldKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9tZXRlcmFuL2duZXh0KQpbIVtHb0RvY10oaHR0cHM6Ly9wa2cuZ28uZGV2L2JhZGdlL2dpdGh1Yi5jb20vbWV0ZXJhbi9nbmV4dD9zdGF0dXMuc3ZnKV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2dpdGh1Yi5jb20vbWV0ZXJhbi9nbmV4dD90YWI9ZG9jKQpbIVtSZWxlYXNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9yZWxlYXNlL21ldGVyYW4vZ25leHQuc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9naXRodWIuY29tL21ldGVyYW4vZ25leHQvcmVsZWFzZXMpCgpnTmV4dCBpcyBhIEdvbGFuZyBBUEktZm9jdXNlZCBmcmFtZXdvcmsgZXh0ZW5kaW5nIFtHaW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9naW4tZ29uaWMvZ2luKS4gT2ZmZXJzIHRoZSBBUEkKc3RydWN0dXJpbmcsIGF1dG9tYXRlcyB2YWxpZGF0aW9uIGFuZCBnZW5lcmF0ZXMgZG9jdW1lbnRhdGlvbi4gSXQncyBjb21wYXRpYmxlIHdpdGggdGhlIGV4aXN0aW5nIEdpbiBoYW5kbGVycyBhbmQgR2luCm1pZGRsZXdhcmVzLiBEZXNpZ25lZCB0byBzaW1wbGlmeSBhbmQgYm9vc3QgZGV2ZWxvcG1lbnQgb2YgSlNPTiBBUElzLiBZb3UgY2FuIGxlYXZlIGdlbmVyaWMgYW5kIGJvcmluZyBzdHVmZiB0byBnTmV4dAphbmQgcHVyZWx5IGZvY3VzIG9uIHRoZSBidXNpbmVzcyBsb2dpYy4KCiMjIENvbnRlbnRzCgotIFtnTmV4dCBXZWIgRnJhbWV3b3JrXSgjZ25leHQtd2ViLWZyYW1ld29yaykKICAgIC0gW0NvbnRlbnRzXSgjY29udGVudHMpCiAgICAtIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCiAgICAtIFtRdWljayBzdGFydF0oI3F1aWNrLXN0YXJ0KQotIFtEb2N1bWVudGF0aW9uXShodHRwczovL21ldGVyYW4uZ2l0aHViLmlvL2duZXh0L2RvY3VtZW50YXRpb24vc2l0ZS8pCgojIyBJbnN0YWxsYXRpb24KCllvdSBjYW4gZG93bmxvYWQgZ05leHQgYW5kIGluc3RhbGwgaXQgaW4geW91ciBwcm9qZWN0IGJ5IHJ1bm5pbmc6CgpgYGBzaGVsbApnbyBnZXQgLXUgZ2l0aHViLmNvbS9tZXRlcmFuL2duZXh0CmBgYAoKIyMgUXVpY2sgc3RhcnQKClRoaXMgdHV0b3JpYWwgYXNzdW1lcywgdGhhdCB5b3UgYWxyZWFkeSBoYXZlIEdvbGFuZyBpbnN0YWxsYXRpb24gYW5kIGJhc2ljIGtub3dsZWRnZSBhYm91dCBob3cgdG8gYnVpbGQgYW5kIHJ1biBHbwpwcm9ncmFtcy4gSWYgdGhpcyBpcyB5b3VyIGZpcnN0IGhpdCB3aXRoIEdvLCBhbmQgeW91IGZlZWwgeW91IGhhdmUgbm8gaWRlYSB3aGF0IGlzIGhhcHBlbmluZyBoZXJlLCBwbGVhc2UgcmVhZCBob3cKdG8gW2dldCBzdGFydGVkIHdpdGggR29dKGh0dHBzOi8vZ28uZGV2L2RvYy90dXRvcmlhbC9nZXR0aW5nLXN0YXJ0ZWQpLgoKT2ssIHNvIGxldCdzIGNyZWF0ZSBhIHByb2plY3Q6CgpgYGBzaGVsbApta2RpciBnbmV4dC1leGFtcGxlCmNkIGduZXh0LWV4YW1wbGUKZ28gbW9kIGluaXQgZXhhbXBsZS5jb20vZ25leHQKZ28gZ2V0IGdpdGh1Yi5jb20vbWV0ZXJhbi9nbmV4dApgYGAKCkNyZWF0ZSBhIGZpbGUgYGV4YW1wbGUuZ29gIGFuZCBmaWxsIGl0IHVwIHdpdGggdGhlIGZvbGxvd2luZyBjb2RlOgoKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgImdpdGh1Yi5jb20vbWV0ZXJhbi9nbmV4dCIKCmZ1bmMgbWFpbigpIHsKCXIgOj0gZ25leHQuUm91dGVyKCkKCglyLkdFVCgiL2V4YW1wbGUiLCBmdW5jKCkgc3RyaW5nIHsKCQlyZXR1cm4gIkhlbGxvIFdvcmxkISIKCX0pCgoJXyA9IHIuUnVuKCkKfQpgYGAKClJ1biBpdDoKCmBgYHNoZWxsCmdvIHJ1biBleGFtcGxlCmBgYAoKTm93IHlvdSBjYW4gdmlzaXQgdGhpcyBsaW5rIGluIHlvdXIgYnJvd3NlcjogaHR0cDovL2xvY2FsaG9zdDo4MDgwL2V4YW1wbGUKClllcywgeWVzLi4uIG9mIGNvdXJzZSBpdCB3b3JrcywgYnV0IHRoYXQncyBib3JpbmcuLi4gTGV0J3Mgb3BlbiB0aGlzIHBhZ2U6IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9kb2NzCgpXaG9hLCB0aGF0IHdhcyBhbWF6aW5nLCAuLi5idXQgbm90IHZlcnkgdXNlZnVsLgoKTGV0J3MgdHJ5IHNvbWUgcmVhbCBleGFtcGxlLiBXaXRoIHJlcXVlc3QgYW5kIHJlc3BvbnNlLiBXZSBjYW4gbW9kaWZ5IG91ciBoYW5kbGVyIHRvIHVzZSBzdHJ1Y3R1cmVzOgoKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgImdpdGh1Yi5jb20vbWV0ZXJhbi9nbmV4dCIKCmZ1bmMgbWFpbigpIHsKCXIgOj0gZ25leHQuUm91dGVyKCkKCglyLlBPU1QoIi9leGFtcGxlIiwgaGFuZGxlcikKCV8gPSByLlJ1bigpCn0KCnR5cGUgTXlSZXF1ZXN0IHN0cnVjdCB7CglJZCAgIGludCAgICBganNvbjoiaWQiIGJpbmRpbmc6InJlcXVpcmVkImAKCU5hbWUgc3RyaW5nIGBqc29uOiJuYW1lImAKfQoKdHlwZSBNeVJlc3BvbnNlIHN0cnVjdCB7CglSZXN1bHQgc3RyaW5nIGBqc29uOiJyZXN1bHQiYAp9CgpmdW5jIGhhbmRsZXIocmVxICpNeVJlcXVlc3QpICpNeVJlc3BvbnNlIHsKCXJldHVybiAmTXlSZXNwb25zZXtSZXN1bHQ6IHJlcS5OYW1lfQp9CmBgYAoKUmVzdGFydCB0aGUgc2VydmVyIGFuZCB2aXNpdCB0aGUgZG9jcyBwYWdlLiBZb3UgY2FuIHNlZSB0aGF0IHJlcXVlc3QgYW5kIHJlc3BvbnNlIG9mIGBQT1NUIC9leGFtcGxlYCBlbmRwb2ludCBhcmUKZG9jdW1lbnRlZC4gVGhhdCdzIHRoZSByZWFsIHBvd2VyIQoKVGhlIFBPU1QgcmVxdWVzdCB3aXRob3V0IHJlcXVpcmVkIGBpZGAgbm93IGZhaWxzIHdpdGggdGhlIHZhbGlkYXRpb24gZXJyb3I6CgpgYGBzaGVsbApjdXJsIC0tcmVxdWVzdCBQT1NUIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9leGFtcGxlIC0tZGF0YSAneyJuYW1lIjogInNvbWUgbmFtZSJ9JwpgYGAKCmdpdmVzIG91dHB1dDoKCmBgYGpzb24KewogICJtZXNzYWdlIjogInZhbGlkYXRpb24gZXJyb3IiLAogICJkZXRhaWxzIjogWwogICAgImZpZWxkIHZhbGlkYXRpb24gZm9yICdpZCcgZmFpbGVkIG9uIHRoZSAncmVxdWlyZWQnIHRhZyB3aXRoIHZhbHVlICcnIgogIF0sCiAgInN1Y2Nlc3MiOiBmYWxzZQp9CmBgYAoKdGhlIHZhbGlkIHJlcXVlc3Q6CgpgYGBzaGVsbApjdXJsIC0tcmVxdWVzdCBQT1NUIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9leGFtcGxlIC0tZGF0YSAneyJuYW1lIjogInNvbWUgbmFtZSIsICJpZCI6IDR9JwpgYGAKCmdpdmVzIHVzIHRoZSBleHBlY3RlZCByZXNwb25zZToKCmBgYGpzb24KewogICJyZXN1bHQiOiAic29tZSBuYW1lIgp9CmBgYAoKQ29uZ3JhdHVsYXRpb25zISBOb3cgeW91IGFyZSBwcmVwYXJlZCBmb3IgdGhlIGZhc3QgZm9yd2FyZGluZyBkZXZlbG9wbWVudCBvZiB5b3VyIGdyZWF0IEFQSS4KCg== readmeEtag: '"05889fcd99495fbaa5b5c9b75bd40d37d5e74146"' readmeLastModified: Mon, 11 Sep 2023 11:17:40 GMT repositoryId: 434389055 description: >- Web Framework extension for Gin. Offers the API structuring, automates validation and generates documentation. It's fully compatible with the current Gin usages and Gin's middlewares. created: '2021-12-02T22:07:56Z' updated: '2025-10-07T01:51:12Z' language: Go archived: false stars: 72 watchers: 6 forks: 5 owner: meteran logo: https://avatars.githubusercontent.com/u/2293924?v=4 license: MIT repoEtag: '"6e909a9c4c656232ada96ffd386bac56ed8d2e18a268252a4dc91dec95dd1ee6"' repoLastModified: Tue, 07 Oct 2025 01:51:12 GMT foundInMaster: true category: - SDK - Server Implementations id: 7e0dad785526ee6f46eb80765465f803 - source: - openapi3 tags - openapi31 tags repository: https://github.com/pb33f/libopenapi-validator v3: true id: b77b0d150f7c5b07a939f34b5a16c59d repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+DQoJPGltZyBzcmM9ImxpYm9wZW5hcGktbG9nby5wbmciIGFsdD0ibGlib3BlbmFwaSIgaGVpZ2h0PSIzMDBweCIgd2lkdGg9IjQ1MHB4Ii8+DQo8L3A+DQoNCiMgRW50ZXJwcmlzZSBncmFkZSBPcGVuQVBJIHZhbGlkYXRpb24gdG9vbHMgZm9yIGdvbGFuZy4NCg0KIVtQaXBlbGluZV0oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL2xpYm9wZW5hcGktdmFsaWRhdG9yL3dvcmtmbG93cy9CdWlsZC9iYWRnZS5zdmcpDQpbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvcGIzM2YvbGlib3BlbmFwaS12YWxpZGF0b3IvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnPyldKGh0dHBzOi8vY29kZWNvdi5pby9naC9wYjMzZi9saWJvcGVuYXBpLXZhbGlkYXRvcikNClshW2Rpc2NvcmRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZGlzY29yZC85MjMyNTgzNjM1NDA4MTU5MTIpXShodHRwczovL2Rpc2NvcmQuZ2cveDdWQUNWdUVHUCkNClshW0RvY3NdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZ29kb2MtcmVmZXJlbmNlLTVmYWZkNyldKGh0dHBzOi8vcGtnLmdvLmRldi9naXRodWIuY29tL3BiMzNmL2xpYm9wZW5hcGktdmFsaWRhdG9yKQ0KDQpBIHZhbGlkYXRpb24gbW9kdWxlIGZvciBbbGlib3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL3BiMzNmL2xpYm9wZW5hcGkpLg0KDQpgbGlib3BlbmFwaS12YWxpZGF0b3JgIHdpbGwgdmFsaWRhdGUgdGhlIGZvbGxvd2luZyBlbGVtZW50cyBhZ2FpbnN0IGFuIE9wZW5BUEkgMysgc3BlY2lmaWNhdGlvbg0KDQotICpodHRwLlJlcXVlc3QqIC0gVmFsaWRhdGVzIHRoZSByZXF1ZXN0IGFnYWluc3QgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbg0KLSAqaHR0cC5SZXNwb25zZSogLSBWYWxpZGF0ZXMgdGhlIHJlc3BvbnNlIGFnYWluc3QgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbg0KLSAqbGlib3BlbmFwaS5Eb2N1bWVudCogLSBWYWxpZGF0ZXMgdGhlIE9wZW5BUEkgZG9jdW1lbnQgYWdhaW5zdCB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uDQotICpiYXNlLlNjaGVtYSogLSBWYWxpZGF0ZXMgYSBzY2hlbWEgYWdhaW5zdCBhIEpTT04gb3IgWUFNTCBibG9iIC8gdW5tYXJzaGFsbGVkIG9iamVjdA0KDQrwn5GJ8J+RiSBbQ2hlY2sgb3V0IHRoZSBmdWxsIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS92YWxpZGF0aW9uLykg8J+RiPCfkYgNCg0KLS0tDQoNCiMjIEluc3RhbGxhdGlvbg0KDQpgYGBiYXNoDQpnbyBnZXQgZ2l0aHViLmNvbS9wYjMzZi9saWJvcGVuYXBpLXZhbGlkYXRvcg0KYGBgDQoNCiMjIFZhbGlkYXRlIE9wZW5BUEkgRG9jdW1lbnQNCg0KYGBgYmFzaA0KZ28gcnVuIGdpdGh1Yi5jb20vcGIzM2YvbGlib3BlbmFwaS12YWxpZGF0b3IvY21kL3ZhbGlkYXRlQGxhdGVzdCBbLS1yZWdleGVuZ2luZV0gWy0teWFtbDJqc29uXSA8ZmlsZT4NCmBgYA0KDQojIyMgT3B0aW9ucw0KDQojIyMjIC0tcmVnZXhlbmdpbmUNCvCflI0gRXhhbXBsZTogVXNlIGEgY3VzdG9tIHJlZ2V4IGVuZ2luZS9mbGFnIChlLmcuLCBlY21hc2NyaXB0KQ0KYGBgYmFzaA0KZ28gcnVuIGdpdGh1Yi5jb20vcGIzM2YvbGlib3BlbmFwaS12YWxpZGF0b3IvY21kL3ZhbGlkYXRlQGxhdGVzdCAtLXJlZ2V4ZW5naW5lPWVjbWFzY3JpcHQgPGZpbGU+DQpgYGANCvCflKcgU3VwcG9ydGVkICoqLS1yZWdleGVuZ2luZSoqIGZsYWdzL3ZhbHVlcyAo4oS577iPIERlZmF1bHQ6IHJlMikNCi0gbm9uZQ0KLSBpZ25vcmVjYXNlDQotIG11bHRpbGluZQ0KLSBleHBsaWNpdGNhcHR1cmUNCi0gY29tcGlsZWQNCi0gc2luZ2xlbGluZQ0KLSBpZ25vcmVwYXR0ZXJud2hpdGVzcGFjZQ0KLSByaWdodHRvbGVmdA0KLSBkZWJ1Zw0KLSBlY21hc2NyaXB0DQotIHJlMg0KLSB1bmljb2RlDQoNCiMjIyMgLS15YW1sMmpzb24NCvCflI0gQ29udmVydCBZQU1MIGZpbGVzIHRvIEpTT04gYmVmb3JlIHZhbGlkYXRpb24gKOKEue+4jyBEZWZhdWx0OiBmYWxzZSkNCg0KW2xpYm9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9wYjMzZi9saWJvcGVuYXBpL2Jsb2IvbWFpbi9kYXRhbW9kZWwvc3BlY19pbmZvLmdvI0wxMTUpIHBhc3NlcyBgbWFwW2ludGVyZmFjZXt9XWludGVyZmFjZXt9YCBzdHJ1Y3R1cmVzIGZvciBkZWVwbHkgbmVzdGVkIG9iamVjdHMgb3IgY29tcGxleCBtYXBwaW5ncyBpbiB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLCB3aGljaCBhcmUgbm90IGFsbG93ZWQgaW4gSlNPTi4NClRoZXNlIHN0cnVjdHVyZXMgY2Fubm90IGJlIHByb3Blcmx5IGNvbnZlcnRlZCB0byBKU09OIGJ5IGxpYm9wZW5hcGkgYW5kIGNhbm5vdCBiZSB2YWxpZGF0ZWQgYnkganNvbnNjaGVtYSwgcmVzdWx0aW5nIGluIGFtYmlndW91cyBlcnJvcnMuDQoNClRoaXMgZmxhZyBhbGxvd3MgcHJlLWNvbnZlcnRpbmcgZnJvbSBZQU1MIHRvIEpTT04gdG8gYnlwYXNzIHRoaXMgbGltaXRhdGlvbiBvZiB0aGUgbGlib3BlbmFwaS4NCg0KKipXaGVuIGRvZXMgdGhpcyBoYXBwZW4/KioNCi0gT3BlbkFQSSBzcGVjcyB3aXRoIGRlZXBseSBuZXN0ZWQgc2NoZW1hIGRlZmluaXRpb25zDQotIENvbXBsZXggYGFsbE9mYCwgYG9uZU9mYCwgb3IgYGFueU9mYCBzdHJ1Y3R1cmVzIHdpdGggbXVsdGlwbGUgbGV2ZWxzDQotIFNwZWNpZmljYXRpb25zIHdpdGggaW50cmljYXRlIG9iamVjdCBtYXBwaW5ncyBpbiBleGFtcGxlcyBvciBzY2hlbWEgcHJvcGVydGllcw0KDQpFbmFibGluZyB0aGlzIGZsYWcgcHJlLWNvbnZlcnRzIHRoZSBZQU1MIGRvY3VtZW50IGZyb20gWUFNTCB0byBKU09OLCBlbnN1cmluZyBhIGNsZWFuIEpTT04gc3RydWN0dXJlIGJlZm9yZSB2YWxpZGF0aW9uLg0KDQpFeGFtcGxlOg0KYGBgYmFzaA0KZ28gcnVuIGdpdGh1Yi5jb20vcGIzM2YvbGlib3BlbmFwaS12YWxpZGF0b3IvY21kL3ZhbGlkYXRlQGxhdGVzdCAtLXlhbWwyanNvbiA8ZmlsZT4NCmBgYA0KDQojIyBEb2N1bWVudGF0aW9uDQoNCi0gW1RoZSBzdHJ1Y3R1cmUgb2YgdGhlIHZhbGlkYXRvcl0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3ZhbGlkYXRpb24vI3RoZS1zdHJ1Y3R1cmUtb2YtdGhlLXZhbGlkYXRvcikNCiAgLSBbVmFsaWRhdGlvbiBlcnJvcnNdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS92YWxpZGF0aW9uLyN2YWxpZGF0aW9uLWVycm9ycykNCiAgLSBbU2NoZW1hIGVycm9yc10oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3ZhbGlkYXRpb24vI3NjaGVtYS1lcnJvcnMpDQogIC0gW0hpZ2gtbGV2ZWwgdmFsaWRhdGlvbl0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3ZhbGlkYXRpb24vI2hpZ2gtbGV2ZWwtdmFsaWRhdGlvbikNCi0gW1ZhbGlkYXRpbmcgaHR0cC5SZXF1ZXN0XShodHRwczovL3BiMzNmLmlvL2xpYm9wZW5hcGkvdmFsaWRhdGlvbi8jdmFsaWRhdGluZy1odHRwcmVxdWVzdCkNCi0gW1ZhbGlkYXRpbmcgaHR0cC5SZXF1ZXN0IGFuZCBodHRwLlJlc3BvbnNlXShodHRwczovL3BiMzNmLmlvL2xpYm9wZW5hcGkvdmFsaWRhdGlvbi8jdmFsaWRhdGluZy1odHRwcmVxdWVzdC1hbmQtaHR0cHJlc3BvbnNlKQ0KLSBbVmFsaWRhdGluZyBqdXN0IGh0dHAuUmVzcG9uc2VdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS92YWxpZGF0aW9uLyN2YWxpZGF0aW5nLWp1c3QtaHR0cHJlc3BvbnNlKQ0KLSBbVmFsaWRhdGluZyBIVFRQIFBhcmFtZXRlcnNdKGh0dHBzOi8vcGIzM2YuaW8vbGlib3BlbmFwaS92YWxpZGF0aW9uLyN2YWxpZGF0aW5nLWh0dHAtcGFyYW1ldGVycykNCi0gW1ZhbGlkYXRpbmcgYW4gT3BlbkFQSSBkb2N1bWVudF0oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3ZhbGlkYXRpb24vI3ZhbGlkYXRpbmctYW4tb3BlbmFwaS1kb2N1bWVudCkNCi0gW1ZhbGlkYXRpbmcgU2NoZW1hc10oaHR0cHM6Ly9wYjMzZi5pby9saWJvcGVuYXBpL3ZhbGlkYXRpb24vI3ZhbGlkYXRpbmctc2NoZW1hcykNCg0KW2xpYm9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9wYjMzZi9saWJvcGVuYXBpKSBhbmQgW2xpYm9wZW5hcGktdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20vcGIzM2YvbGlib3BlbmFwaS12YWxpZGF0b3IpIGFyZQ0KcHJvZHVjdHMgb2YgUHJpbmNlc3MgQmVlZiBIZWF2eSBJbmR1c3RyaWVzLCBMTEMNCg== readmeEtag: '"e416418107bd5dc5e2d6da22eddcef8c73b72270"' readmeLastModified: Tue, 02 Dec 2025 11:11:50 GMT repositoryId: 621782398 description: >- OpenAPI validation extension for libopenapi, validate http requests and responses as well as schemas created: '2023-03-31T11:28:11Z' updated: '2026-02-05T01:14:36Z' language: Go archived: false stars: 129 watchers: 2 forks: 40 owner: pb33f logo: https://avatars.githubusercontent.com/u/104016643?v=4 license: MIT repoEtag: '"c52c89d4cd86f1d0347ab4cbcf4416d842d0053e82611837206dcb6e8291c3fc"' repoLastModified: Thu, 05 Feb 2026 01:14:36 GMT category: Parsers foundInMaster: true v3_1: true - source: openapi3 tags repository: https://github.com/vitalybibikov/azureextensions.swashbuckle v3: true repositoryMetadata: base64Readme: >- 
# AzureExtensions.Swashbuckle v4.0.0

### (Searching for collaborators!)

OpenAPI 2/3 implementation based on Swashbuckle(Swagger) tooling for API's built with Azure Functions 

This product aims to easily provide Swagger and Swagger UI of APIs created in Azure Functions using isolated worker model

------------------------------
4.0.4
- Updated to latest dependencies

------------------------------

4.0.3
by @MikeHookTransparity
Added ClientSecret, UseBasicAuthenticationWithAccessCodeGrant

------------------------------
4.0.2
- Updated to latest Swagger 6.5.1
- Fixes & Improvements
- Added symbols package

------------------------------
4.0.1

- Fixed some minor issues
- Updated to new language features
- Added exception handling and nullability
- Doocumenation fixes

------------------------------
4.0.0-beta
- just remebering what the heck is going om here
- Updated to v4 Functions
- Updated to .NET 8
- Updated to isolated worker model (from now on it is going to be the only one that is supported, as inprocess is going to be deprecated)
- Updated to UI v5.17.3
- Updated to Swagger 5.6.5
- Updated docs
- Considering removing support of NewtonJson

  https://www.nuget.org/packages/AzureExtensions.Swashbuckle/4.0.0-beta
------------------------------
3.3.1-beta

- #64 Support for authorization configuration
- #60 Consolidated extensions and added one to support .net 5
- Updated docs
- Updated js/html/css libs
- Some classed made public to support 3-party IoC.
- Fixed several issues, related to versioning and XML comments.
- Updated to UI v3.37.2
- Updated to Swagger 5.6.3
- Updated documentation
- Ability to create multiple versions of documents, example added.
- Added examples of a custom filter, improved test application

https://www.nuget.org/packages/AzureExtensions.Swashbuckle/4.0.0-beta


------------------------------
3.1.6

https://www.nuget.org/packages/AzureExtensions.Swashbuckle/3.1.6

Fixed #8, #9

Updated to UI v3.25.1

Updated to Swagger 5.4.1

Fixed base url for Swagger UI

**Breaking:**

Option and DocumentOption renamed to SwaggerDocOptions and SwaggerDocument respectivly
and moved to AzureFunctions.Extensions.Swashbuckle.Settings namespace

**Properties renamed:**

PrepandOperationWithRoutePrefix => PrependOperationWithRoutePrefix

AddCodeParamater => AddCodeParameter

**Properties added:**

Added ability to configure SwaggerGen via ConfigureSwaggerGen

Added ability to override default url to Swagger json document (in case of reverse proxy/gateway/ingress) are used.


**Size:**

All the resources are places in zip archive in order to decrease result dll size by 338% (from 1.594kb to 472kb)



------------------------------
3.0.0
- Updated to v3 Functions
- Updated to 5.0.0 Swashbuckle.AspNetCore nugets
- Merged PRs to fix issues related to RequestBodyType and Ignore attribute
- application/json is a default media type.



# Sample

https://github.com/vitalybibikov/azure-functions-extensions-swashbuckle/tree/master/src/AzureFunctions.Extensions.Swashbuckle/TestFunction

# Update

Version 3.0.0


# Getting Started

1. Install the standard Nuget package into your Azure Functions application.

```
Package Manager : Install-Package AzureExtensions.Swashbuckle
CLI : dotnet add package AzureExtensions.Swashbuckle
```

2. Add Program.cs class on your Functions project.

!!! Now you need to specify in option the RoutePrefix.

            opts.RoutePrefix = "api";


```csharp
using System;
using System.Collections.Generic;
using Microsoft.Extensions.Hosting;
using System.Reflection;
using AzureFunctions.Extensions.Swashbuckle.Settings;
using Microsoft.Extensions.DependencyInjection;
using Microsoft.OpenApi;
using Microsoft.OpenApi.Models;
using AzureFunctions.Extensions.Swashbuckle;
using Swashbuckle.AspNetCore.SwaggerGen;

var host = new HostBuilder()
    .ConfigureServices((hostContext, services) =>
    {
        //Register the extension
        services.AddSwashBuckle(opts =>
        {
            // If you want to add Newtonsoft support insert next line
            // opts.AddNewtonsoftSupport = true;
            opts.RoutePrefix = "api";
            opts.SpecVersion = OpenApiSpecVersion.OpenApi3_0;
            opts.AddCodeParameter = true;
            opts.PrependOperationWithRoutePrefix = true;
            opts.XmlPath = "TestFunction.xml";
            opts.Documents = new[]
            {
                new SwaggerDocument
                {
                    Name = "v1",
                    Title = "Swagger document",
                    Description = "Swagger test document",
                    Version = "v2"
                },
                new SwaggerDocument
                {
                    Name = "v2",
                    Title = "Swagger document 2",
                    Description = "Swagger test document 2",
                    Version = "v2"
                }
            };
            opts.Title = "Swagger Test";
            //opts.OverridenPathToSwaggerJson = new Uri("http://localhost:7071/api/Swagger/json");
            opts.ConfigureSwaggerGen = x =>
            {
                //custom operation example
                x.CustomOperationIds(apiDesc => apiDesc.TryGetMethodInfo(out MethodInfo methodInfo)
                    ? methodInfo.Name
                    : new Guid().ToString());

                //custom filter example
                //x.DocumentFilter<RemoveSchemasFilter>();

                //oauth2
                x.AddSecurityDefinition("oauth2",
                    new OpenApiSecurityScheme
                    {
                        Type = SecuritySchemeType.OAuth2,
                        Flows = new OpenApiOAuthFlows
                        {
                            Implicit = new OpenApiOAuthFlow
                            {
                                AuthorizationUrl = new Uri("https://your.idserver.net/connect/authorize"),
                                Scopes = new Dictionary<string, string>
                                {
                                    { "api.read", "Access read operations" },
                                    { "api.write", "Access write operations" }
                                }
                            }
                        }
                    });
            };

            // set up your client ID if your API is protected
            opts.ClientId = "your.client.id";
            opts.OAuth2RedirectPath = "http://localhost:7071/api/swagger/oauth2-redirect";
        });
    })
    .Build();

host.Run();

```

3. Add swagger and swagger ui endpoint functions on your project.

```csharp
    public class SwaggerController
    {
        private readonly ISwashBuckleClient swashBuckleClient;

        public SwaggerController(ISwashBuckleClient swashBuckleClient)
        {
            this.swashBuckleClient = swashBuckleClient;
        }

        [SwaggerIgnore]
        [Function("SwaggerJson")]
        public async Task<HttpResponseData> SwaggerJson(
            [HttpTrigger(AuthorizationLevel.Function, "get", Route = "Swagger/json")]
            HttpRequestData  req)
        {
            return await this.swashBuckleClient.CreateSwaggerJsonDocumentResponse(req);
        }

        [SwaggerIgnore]
        [Function("SwaggerYaml")]
        public async Task<HttpResponseData> SwaggerYaml(
            [HttpTrigger(AuthorizationLevel.Function, "get", Route = "Swagger/yaml")]
            HttpRequestData req)
        {
            return await this.swashBuckleClient.CreateSwaggerYamlDocumentResponse(req);
        }

        [SwaggerIgnore]
        [Function("SwaggerUi")]
        public async Task<HttpResponseData> SwaggerUi(
            [HttpTrigger(AuthorizationLevel.Function, "get", Route = "Swagger/ui")]
            HttpRequestData req)
        {
            return await this.swashBuckleClient.CreateSwaggerUIResponse(req, "swagger/json");
        }

        /// <summary>
        /// This is only needed for OAuth2 client. This redirecting document is normally served
        /// as a static content. Functions don't provide this out of the box, so we serve it here.
        /// Don't forget to set OAuth2RedirectPath configuration option to reflect this route.
        /// </summary>
        /// <param name="req"></param>
        /// <param name="swashBuckleClient"></param>
        /// <returns></returns>
        [SwaggerIgnore]
        [Function("SwaggerOAuth2Redirect")]
        public async Task<HttpResponseData> SwaggerOAuth2Redirect(
            [HttpTrigger(AuthorizationLevel.Function, "get", Route = "swagger/oauth2-redirect")]
            HttpRequestData req)
        {
            return await this.swashBuckleClient.CreateSwaggerOAuth2RedirectResponse(req);
        }
    }
```

4. Open Swagger UI URL in your browser.

If you does not changed api route prefix. Swagger UI URL is https://hostname/api/swagger/ui .

## Options

### Include Xml document file

AzureFunctions.Extensions.Swashbuckle can include xml document file.

1. Change your functions project's GenerateDocumentationFile option to enable.

            builder.AddSwashBuckle(Assembly.GetExecutingAssembly(), opts =>
            {
                opts.XmlPath = "TestFunction.xml";
            });

2. Add configration setting this extensions on your functions project's local.settings.json

```json
  "SwaggerDocOptions": {
    "XmlPath": "TestFunction.xml"
  }
```

Alternatively you can add this section to your host.json

```json
  "extensions": {
    "Swashbuckle": {
      "XmlPath": "TestFunction.xml"
    }
  }
```
 readmeEtag: '"fe74cad82aaf5703f9e4fe987fce5c3db2ecae25"' readmeLastModified: Wed, 28 Aug 2024 11:18:57 GMT repositoryId: 230613540 description: This extension enriches Azure Functions with Swagger/ Open API support created: '2019-12-28T13:27:18Z' updated: '2026-02-04T09:11:08Z' language: C# archived: false stars: 71 watchers: 4 forks: 56 owner: vitalybibikov logo: https://avatars.githubusercontent.com/u/7008739?v=4 license: MIT repoEtag: '"bdb25a0fb9778471bb0854c037bd85da3cbca15991a584eb6d32bf093e1ed9d8"' repoLastModified: Wed, 04 Feb 2026 09:11:08 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: ee4e8f0b8483cc304f05e07164cded87 - source: openapi3 tags repository: https://github.com/eazybytes/openapi v3: true id: a9c0bde4fbedfc45a7eed13c72f3e382 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNpZmljYXRpb24gJiBTd2FnZ2VyIFRvb2xzIC0gWmVybyBUbyBNYXN0ZXIKClshW0ltYWdlXShodHRwczovL2dpdGh1Yi5jb20vZWF6eWJ5dGVzL29wZW5hcGkvYmxvYi9tYWluL29wZW5hcGkucG5nICJPcGVuQVBJIFNwZWNpZmljYXRpb24gJmFtcDsgU3dhZ2dlciBUb29scyAtIFplcm8gVG8gTWFzdGVyIildKGh0dHBzOi8vd3d3LnVkZW15LmNvbS9jb3Vyc2Uvb3BlbmFwaS1zcGVjaWZpY2F0aW9uLXN3YWdnZXItdG9vbHMtemVyby10by1tYXN0ZXIvP3JlZmVycmFsQ29kZT1GMDAyQTlBNzk5REIxMjZDRDE4OSkKCioqJ09wZW5BUEkgU3BlY2lmaWNhdGlvbiAmIFN3YWdnZXIgVG9vbHMgLSBaZXJvIFRvIE1hc3RlcicqKiBjb3Vyc2Ugd2lsbCBoZWxwIGluIHVuZGVyc3RhbmRpbmcgYWJvdXQgT3BlbiBBUEkgU3BlY2lmaWNhdGlvbiBhbmQgaG93IHRvIGRlc2NyaWJlLCBkb2N1bWVudCBBUElzIHVzaW5nIE9wZW5BUEkgJiBTd2FnZ2VyIHRvb2xzLgoKIyMgVG9waWNzIGNvdmVyZWQgaW4gdGhlIGNvdXJzZQoKMSkgRGVzaWduaW5nIEFQSXMgd2l0aCBPcGVuQVBJIFNwZWNpZmljYXRpb25zIGFuZCBTd2FnZ2VyIFRvb2xzCjIpIERlc2NyaWJpbmcsIERvY3VtZW50aW5nIEFQSXMgZGV0YWlscyB1c2luZyBPcGVuQVBJIFNwZWNpZmljYXRpb25zCjMpIEhpc3Rvcnkgb2YgT3BlbkFQSSAmIFN3YWdnZXIgYW5kIHJlbGF0aW9uIGJldHdlZW4gdGhlbQo0KSBEZXRhaWxzIGFib3V0IFN3YWdnZXIgdG9vbHMgbGlrZSBTd2FnZ2VyIEVkaXRvciwgU3dhZ2dlciBVSSxTd2FnZ2VySHViLFN3YWdnZXJIdWIgRXhwbG9yZSxTd2FnZ2VyIENvZGVnZW4gZXRjLgo1KSBIb3cgdG8gZ2V0IHN0YXJ0ZWQgd2l0aCBPcGVuQVBJIGluIGNvZGUgZmlyc3QgJiBEZXNpZ24gZmlyc3Qgc2NlbmFyaW9zCjYpIEhvdyB0byB3cml0ZSBhIHZhbGlkIE9wZW5BUEkgZG9jdW1lbnQgdXNpbmcgWUFNTCBzeW50YXgKNykgV3JpdGluZyByZS11c2FibGUgY29udGVudCBpbnNpZGUgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiB3aXRoIGNvbXBvbmVudHMKOCkgRGF0YSB0eXBlcyBzdXBwb3J0ZWQgYnkgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiAmIHRoZWlyIGRldGFpbHMKOSkgSW5oZXJpdGFuY2UgJiBwb2x5bW9ycGhpc20gaW5zaWRlIE9wZW5BUEkgd2l0aCBrZXl3b3JkcyBvbmVPZiwgYW55T2YsIGFsbE9mIGFuZCBub3QKMTAgKSBIb3cgdG8gZGVzY3JpYmUgQVBJcyBzZWN1cml0eSBpbnNpZGUgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbgoxMSkgSG93IHRvIG1vY2sgQVBJcyB3aXRoIE9wZW4gQVBJIHNwZWNpZmljYXRpb24gJiBQcmlzbSBtb2NrIHNlcnZlcgoxMikgSG93IHRvIGdlbmVyYXRlIGNsaWVudCBjb2RlICYgc2VydmVyIHN0dWJzIHVzaW5nIE9wZW5BUEkgZm9yIHZhcmlvdXMgcG9wdWxhciBwcm9ncmFtbWluZyBsYW5ndWFnZXMgJiBmcmFtZXdvcmtzCjEzKSBEZXBsb3lpbmcgJiBIb3N0aW5nIE9wZW4gQVBJIHNwZWNpZmljYXRpb24gaW5zaWRlIGEgR2l0SHViIHBhZ2UgYWxvbmcgd2l0aCBTd2FnZ2VyIFVJCjE0KSBQcm92aWRpbmcgZXhhbXBsZXMgZGF0YSBmb3IgdGhlIEFQSXMgaW5zaWRlIE9wZW4gQVBJIHNwZWNpZmljYXRpb24KMTUpIEFkdmFudGFnZXMgb2YgdXNpbmcgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiAKMTYpIFByb3ZpZGluZyBiZXR0ZXIgZG9jdW1lbnRhdGlvbiB1c2luZyBDb21tb25NYXJrIHN5bnRheCBpbnNpZGUgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiAKCiMjIFByZS1yZXF1aXNpdGUgZm9yIHRoZSBjb3Vyc2UKCi0gQmFzaWMga25vd2xlZGdlIG9uIEFQSXMKLSBJbnRlcmVzdCB0byBsZWFybiBhbmQgZXhwbG9yZSBhYm91dCBPcGVuQVBJICYgU3dhZ2dlciB0b29scwoKIyBJbXBvcnRhbnQgTGlua3MKCi0gT3BlbiBBUEkgV2Vic2l0ZSAtIGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZwotIFN3YWdnZXIgV2Vic2l0ZSAtIGh0dHBzOi8vc3dhZ2dlci5pbwotIFN3YWdnZXIgRWRpdG9yIC0gaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pbwotIFN3YWdnZXJIdWIgRXhwbG9yZSAtIGh0dHBzOi8vZXhwbG9yZS5zd2FnZ2VyaHViLmNvbS8KLSBPcGVuQVBJIE1hcCAtIGh0dHBzOi8vb3BlbmFwaS1tYXAuYXBpaGFuZHltYW4uaW8vCi0gT3BlbkFQSSAmIFNwcmluZ0Jvb3QgbGlicmFyeSAtIGh0dHBzOi8vc3ByaW5nZG9jLm9yZwotIExpc3Qgb2YgT3BlbkFQSSB0b29scyAtIGh0dHBzOi8vb3BlbmFwaS50b29scy8KLSBQcmlzbSBNb2NrIHNlcnZlciAtIGh0dHBzOi8vc3RvcGxpZ2h0LmlvL29wZW4tc291cmNlL3ByaXNtCi0gU1dBUEkgVGhlIFN0YXIgV2FycyBBUEkgLSBodHRwczovL3N3YXBpLmRldi8KLSBSRVFSRVMgTW9jayBBUElzIC0gaHR0cHM6Ly9yZXFyZXMuaW4K readmeEtag: '"27af9e6a7f448b9773e72076bee46be955120e2c"' readmeLastModified: Mon, 15 Jul 2024 11:00:32 GMT repositoryId: 512335029 description: 'OpenAPI Specification & Swagger Tools : Zero To Master - Code Examples' created: '2022-07-10T03:25:03Z' updated: '2026-02-02T01:50:10Z' language: Java archived: false stars: 81 watchers: 5 forks: 130 owner: eazybytes logo: https://avatars.githubusercontent.com/u/79041235?v=4 repoEtag: '"2daed25b3a120dbc53e92efe168c78966e9b89a5adb5a57fee4c0f98a550cabd"' repoLastModified: Mon, 02 Feb 2026 01:50:10 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/drwpow/openapi-fetch v3: true id: eedc4905504decd63f3a047810e71311 repositoryMetadata: base64Readme: >- IyDwn46+IG9wZW5hcGktZmV0Y2gKCiMjIyDwn5qaIFRoaXMgcmVwbyBoYXMgbW92ZWQhCgpJdCBoYXMgYmVlbiBtZXJnZWQgd2l0aCBbb3BlbmFwaS10eXBlc2NyaXB0XShodHRwczovL2dpdGh1Yi5jb20vZHJ3cG93L29wZW5hcGktdHlwZXNjcmlwdCkuIFBsZWFzZSBzdWJtaXQgaXNzdWVzIGFuZCBjb2RlIHRoZXJlLgoKLSDwn5al77iPIFtDb2RlXShodHRwczovL2dpdGh1Yi5jb20vZHJ3cG93L29wZW5hcGktdHlwZXNjcmlwdC90cmVlL21haW4vcGFja2FnZXMvb3BlbmFwaS1mZXRjaCkKLSDwn5OWIFtEb2NzXShodHRwczovL29wZW5hcGktdHMucGFnZXMuZGV2KQo= readmeEtag: '"fd89503619966750f4a3e6280eeb339a55994fc9"' readmeLastModified: Mon, 22 May 2023 16:30:51 GMT repositoryId: 612091894 description: >- Ultra-fast fetching for TypeScript generated automatically from your OpenAPI schema created: '2023-03-10T07:18:20Z' updated: '2025-08-27T12:33:36Z' language: null archived: true stars: 67 watchers: 1 forks: 14 owner: drwpow logo: https://avatars.githubusercontent.com/u/1369770?v=4 license: MIT repoEtag: '"d72c8ccff1f88e37a568610c6babfd48b06d3bedfcdc8a8420b5ec6b030aee31"' repoLastModified: Wed, 27 Aug 2025 12:33:36 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/wol-soft/php-json-schema-model-generator v3: true repositoryMetadata: base64Readme: >- [![Latest Version](https://img.shields.io/packagist/v/wol-soft/php-json-schema-model-generator.svg)](https://packagist.org/packages/wol-soft/php-json-schema-model-generator)
[![Minimum PHP Version](https://img.shields.io/badge/php-%3E%3D%208.0-8892BF.svg)](https://php.net/)
[![Maintainability](https://qlty.sh/gh/wol-soft/projects/php-json-schema-model-generator/maintainability.svg)](https://qlty.sh/gh/wol-soft/projects/php-json-schema-model-generator)
[![Build Status](https://github.com/wol-soft/php-json-schema-model-generator/actions/workflows/main.yml/badge.svg)](https://github.com/wol-soft/php-json-schema-model-generator/actions/workflows/main.yml)
[![Coverage Status](https://coveralls.io/repos/github/wol-soft/php-json-schema-model-generator/badge.svg?branch=master)](https://coveralls.io/github/wol-soft/php-json-schema-model-generator?branch=master)
[![MIT License](https://img.shields.io/packagist/l/wol-soft/php-json-schema-model-generator.svg)](https://github.com/wol-soft/php-json-schema-model-generator/blob/master/LICENSE)
[![Documentation Status](https://readthedocs.org/projects/php-json-schema-model-generator/badge/?version=latest)](https://php-json-schema-model-generator.readthedocs.io/en/latest/?badge=latest)

# php-json-schema-model-generator
Generates PHP model classes from JSON-Schema files including validation and providing a fluent auto completion for the generated classes.

## Table of Contents ##

* [Motivation](#Motivation)
* [Requirements](#Requirements)
* [Installation](#Installation)
* [Basic usage](#Basic-usage)
* [Configuring using the GeneratorConfiguration](#Configuring-using-the-GeneratorConfiguration)
* [Examples](#Examples)
* [How the heck does this work?](#How-the-heck-does-this-work)
* [Tests](#Tests)
* [Docs](#Docs)

## Motivation ##

Simple example from a PHP application: you define and document an API with swagger annotations and JSON-Schema models. Now you want to use models in your controller actions instead of manually accessing the request data (eg. array stuff). Additionally your schema already defines the validation rules for the models. Why duplicate this rules into your manually written code? Instead you can set up a middleware which instantiates models generated with this library and feed the model with the request data. Now you have a validated model which you can use in your controller action. With full auto completion when working with nested objects. Yay!

## Requirements ##

- Requires at least PHP 8.0
- Requires the PHP extensions ext-json and ext-mbstring

## Installation ##

The recommended way to install php-json-schema-model-generator is through [Composer](http://getcomposer.org):
```
$ composer require --dev wol-soft/php-json-schema-model-generator
$ composer require wol-soft/php-json-schema-model-generator-production
```

To avoid adding all dependencies of the php-json-schema-model-generator to your production dependencies it's recommended to add the library as a dev-dependency and include the [wol-soft/php-json-schema-model-generator-production](https://github.com/wol-soft/php-json-schema-model-generator-production) library. The production library provides all classes to run the generated code. Generating the classes should either be a step done in the development environment or as a build step of your application (which is the recommended workflow).

## Basic usage ##

Check out the [docs](https://php-json-schema-model-generator.readthedocs.io/en/latest/) for more details.

The base object for generating models is the *ModelGenerator*. After you have created a Generator you can use the object to generate your model classes without any further configuration:

```php
(new ModelGenerator())
    ->generateModels(new RecursiveDirectoryProvider(__DIR__ . '/schema'), __DIR__ . '/result');
```
The first parameter of the *generateModels* method must be a class implementing the *SchemaProviderInterface*. The provider fetches the JSON schema files and provides them for the generator. The following providers are available:

Provider | Description
--- | ---
RecursiveDirectoryProvider | Fetches all *.json files from the given source directory. Each file must contain a JSON Schema object definition on the top level
OpenAPIv3Provider | Fetches all objects defined in the `#/components/schemas section` of an Open API v3 spec file

The second parameter must point to an existing and empty directory (you may use the `generateModelDirectory` helper method to create your destination directory). This directory will contain the generated PHP classes after the generator is finished.

As an optional parameter you can set up a *GeneratorConfiguration* object (check out the docs for all available options) to configure your Generator and/or use the method *generateModelDirectory* to generate your model directory (will generate the directory if it doesn't exist; if it exists, all contained files and folders will be removed for a clean generation process):

```php
$generator = new ModelGenerator(
    (new GeneratorConfiguration())
        ->setNamespacePrefix('MyApp\Model')
        ->setImmutable(false)
);

$generator
    ->generateModelDirectory(__DIR__ . '/result')
    ->generateModels(new RecursiveDirectoryProvider(__DIR__ . '/schema'), __DIR__ . '/result');
```

The generator will check the given source directory recursive and convert all found *.json files to models. All JSON-Schema files inside the source directory must provide a schema of an object.

## Examples ##

The directory `./tests/manual` contains some easy examples which show the usage. After installing the dependencies of the library via `composer update` you can execute `php ./tests/manual/test.php` to generate the examples and play around with some JSON-Schema files to explore the library.

Let's have a look into an easy example. We create a simple model for a person with a name and an optional age. Our resulting JSON-Schema:
```json
{
  "$id": "Person",
  "type": "object",
  "properties": {
    "name": {
      "type": "string"
    },
    "age": {
      "type": "integer",
      "minimum": 0
    }
  },
  "required": [
    "name"
  ]
}
```

After generating a class with this JSON-Schema our class with the name `Person` will provide the following interface:
```php
// the constructor takes an array with data which is validated and applied to the model
public function __construct(array $modelData);

// the method getRawModelDataInput always delivers the raw input which was provided on instantiation
public function getRawModelDataInput(): array;

// getters to fetch the validated properties. Age is nullable as it's not required
public function getName(): string;
public function getAge(): ?int;

// setters to change the values of the model after instantiation (only generated if immutability is disabled)
public function setName(string $name): Person;
public function setAge(?int $age): Person;
```

Now let's have a look at the behaviour of the generated model:
```php
// Throws an exception as the required name isn't provided.
// Exception: 'Missing required value for name'
$person = new Person([]);

// Throws an exception as the name provides an invalid value.
// Exception: 'Invalid type for name. Requires string, got int'
$person = new Person(['name' => 12]);

// Throws an exception as the age contains an invalid value due to the minimum definition.
// Exception: 'Value for age must not be smaller than 0'
$person = new Person(['name' => 'Albert', 'age' => -1]);

// A valid example as the age isn't required
$person = new Person(['name' => 'Albert']);
$person->getName(); // returns 'Albert'
$person->getAge(); // returns NULL
$person->getRawModelDataInput(); // returns ['name' => 'Albert']

// If setters are generated the setters also perform validations.
// Exception: 'Value for age must not be smaller than 0'
$person->setAge(-10);
```

More complex exception messages eg. from a [allOf](https://json-schema.org/understanding-json-schema/reference/combining.html#allof) composition may look like:
```
Invalid value for Animal declined by composition constraint.
  Requires to match 3 composition elements but matched 1 elements.
  - Composition element #1: Failed
    * Value for age must not be smaller than 0
  - Composition element #2: Valid
  - Composition element #3: Failed
    * Value for legs must not be smaller than 2
    * Value for legs must be a multiple of 2
```

## How the heck does this work? ##

The class generation process basically splits up into three to four steps:

- Scan the given source directory to find all *.json files which should be processed.
- Loop over all schemas which should be generated. This is the main step of the class generation. Now each schema is parsed and a Schema model class which holds the properties for the generated model is populated. All validation rules defined in the JSON-Schema are translated into plain PHP code. After the model is finished a RenderJob is generated and added to the RenderQueue. If a JSON-Schema contains nested objects or references multiple RenderJobs may be added to the RenderQueue for a given schema file.
- If post processors are defined for the generation process the post processors will be applied.
- After all schema files have been parsed without an error the RenderQueue will be worked off. All previous added RenderJobs will be executed and the PHP classes will be saved to the filesystem at the given destination directory.

## Tests ##

The library is tested via [PHPUnit](https://phpunit.de/).

After installing the dependencies of the library via `composer update` you can execute the tests with `./vendor/bin/phpunit` (Linux) or `vendor\bin\phpunit.bat` (Windows). The test names are optimized for the usage of the `--testdox` output. Most tests are atomic integration tests which will set up a JSON-Schema file and generate a class from the schema and test the behaviour of the generated class afterwards.

During the execution the tests will create a directory PHPModelGeneratorTest in tmp where JSON-Schema files and PHP classes will be written to.

If a test which creates a PHP class from a JSON-Schema fails the JSON-Schema and the generated class(es) will be dumped to the directory `./failed-classes`

## Docs ##

The [docs](https://php-json-schema-model-generator.readthedocs.io/en/latest/) for the library is generated with [Sphinx](https://www.sphinx-doc.org/en/master/).

To generate the documentation install Sphinx, enter the docs directory and execute `make html` (Linux) or `make.bat html` (Windows). The generated documentation will be available in the directory `./docs/build`.

The documentation hosted at [Read the Docs](https://readthedocs.org/) is updated on each push.
 readmeEtag: '"0f8067a91d7dccfee81b8b348ef3e12e8be14156"' readmeLastModified: Fri, 19 Sep 2025 15:18:08 GMT repositoryId: 130233015 description: >- Creates (immutable) PHP model classes from JSON-Schema files including all validation rules as PHP code created: '2018-04-19T15:03:37Z' updated: '2026-01-16T14:19:40Z' language: PHP archived: false stars: 81 watchers: 2 forks: 18 owner: wol-soft logo: https://avatars.githubusercontent.com/u/14750468?v=4 license: MIT repoEtag: '"94c0a9d09ed14dda6f5eae214cd891cfe7b950c5746353152b39171c408cf119"' repoLastModified: Fri, 16 Jan 2026 14:19:40 GMT foundInMaster: true id: 66096cd4d636503d8f7fc479dbf415f6 - source: openapi3 tags repository: https://github.com/wy-z/requests-openapi v3: true repositoryMetadata: base64Readme: >- IyByZXF1ZXN0cy1vcGVuYXBpCgpbIVtpbWFnZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3YvcmVxdWVzdHMtb3BlbmFwaS5zdmcpXShodHRwczovL3B5cGkub3JnL3Byb2plY3QvcmVxdWVzdHMtb3BlbmFwaS8pClshW2ltYWdlXShodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvbC9yZXF1ZXN0cy1vcGVuYXBpLnN2ZyldKGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9yZXF1ZXN0cy1vcGVuYXBpLykKWyFbaW1hZ2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9weXZlcnNpb25zL3JlcXVlc3RzLW9wZW5hcGkuc3ZnKV0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L3JlcXVlc3RzLW9wZW5hcGkvKQpbIVtpbWFnZV0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3d5LXovcmVxdWVzdHMtb3BlbmFwaS9tYXN0ZXIvdGVzdHMvY292ZXJhZ2UtYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3d5LXovcmVxdWVzdHMtb3BlbmFwaSkKCkEgbGlnaHR3ZWlnaHQgYnV0IHBvd2VyZnVsIGFuZCBlYXN5LXRvLXVzZSBQeXRob24gY2xpZW50IGxpYnJhcnkgZm9yIE9wZW5BUEkgdjMuCgojIyBLZXkgRmVhdHVyZXMKCi0gKipMaWdodHdlaWdodCBEZXNpZ24qKjogRXhwZXJpZW5jZSBhIG1pbmltYWxpc3RpYyBpbnRlcmZhY2UuCi0gKipGb2N1cyBvbiBFc3NlbnRpYWxzKio6IEhlbHBzIHlvdSBzaW1wbGlmeSB0aGUgaGFuZGxpbmcgb2YgUGF0aHMsIFBhcmFtZXRlcnMsIEhlYWRlcnMsIENvb2tpZXMsIGV0Yy4sIHdoaWxlIGluaGVyaXRpbmcgYWxsIHRoZSBjYXBhYmlsaXRpZXMgb2YgUmVxdWVzdHMuCi0gKipUZXN0aW5nIGFuZCBJbnRlZ3JhdGlvbiBNYWRlIEVhc3kqKjogV2hldGhlciB5b3UncmUgcnVubmluZyB0ZXN0cyBvciBpbnRlZ3JhdGluZyB3aXRoIG90aGVyIHN5c3RlbXMsIHRoZSBjbGllbnQgc2ltcGxpZmllcyB0aGVzZSB0YXNrcy4KCiMjIFVzYWdlCgpgYGBweXRob24KaW1wb3J0IHJlcXVlc3RzX29wZW5hcGkKCiMgbG9hZCBzcGVjIGZyb20gdXJsCmMgPSByZXF1ZXN0c19vcGVuYXBpLkNsaWVudCgpLmxvYWRfc3BlY19mcm9tX3VybCgiaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFzdGVyL2V4YW1wbGVzL3YzLjAvcGV0c3RvcmUueWFtbCIpCiMgb3IgbG9hZCBmcm9tIGZpbGUKYyA9IHJlcXVlc3RzX29wZW5hcGkuQ2xpZW50KCkubG9hZF9zcGVjX2Zyb21fZmlsZSgib3BlbmFwaS5qc29uIikKIyBzZXQgc2VydmVyCmMuc2V0X3NlcnZlcihyZXF1ZXN0c19vcGVuYXBpLlNlcnZlcih1cmw9Imh0dHBzOi8vZmFrZS5jb20vYXBpIikpCgojIGN1c3RvbSBzZXNzaW9uIGZvciBhdXRoIG9yIG90aGVycwpjLnJlcXVlc3RvciAjIGEgaW5zdGFuY2Ugb2YgcmVxdWVzdHMuU2Vzc2lvbiwgc2VlIGh0dHBzOi8vcmVxdWVzdHMucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0L3VzZXIvYWR2YW5jZWQvI3Nlc3Npb24tb2JqZWN0cwojIHNldCB1cGRhdGUgdG9rZW4KYy5yZXF1ZXN0b3IuaGVhZGVycy51cGRhdGUoeyJBdXRob3JpemF0aW9uIjogInRva2VuIn0pCgojIGNhbGwgYXBpIGJ5IG9wZXJhdGlvbiBpZApyZXNwID0gYy5saXN0UGV0cygpICMgcmVzcDogcmVxdWVzdHMuUmVzcG9uc2UKcmVzcC5qc29uKCkKIyBnZXQgYnkgcGF0aCBpZApyZXNwID0gYy5zaG93UGV0QnlJZChwZXRJZD0xKQpyZXNwLmpzb24oKQojIHBvc3QKcmVzcCA9IGMuY3JlYXRlUGV0cyhqc29uPXt9KQpyZXNwLmpzb24oKQoKIwojIEFkdmFuY2VkIFVzYWdlCiMKCiMgc2V0IHJlcSBvcHRpb25zLCAncmVxX29wdHMnIHBhcmFtIHRvIGN1c3RvbSByZXF1ZXN0IG9wdGlvbnMKcmVxdWVzdHNfb3BlbmFwaS5DbGllbnQocmVxX29wdHM9eyJ0aW1lb3V0IjogNjB9KS5sb2FkX3NwZWNfZnJvbV9maWxlKCJ4eCIpCgojIHBhcmFtZXRlcnMgc3RhcnRzIHdpdGggJ18nIG9yIG5vdCBmb3VuZCBpbiBvcGVuYXBpIHNwZWMsIHdpbGwgYmUgcGFzc2VkIHRocm91Z2ggdG8gdGhlIHJlcXVlc3RpbmcKYy5jcmVhdGVQZXRzKGpzb249e30sIF9oZWFkZXJzPXt9LCBfcGFyYW1zPXt9LCBfY29va2llcz17fSkKCiMgcGFyYW1ldGVycwojIGluOiBjb29raWUsIG5hbWU6IGNzcmZ0b2tlbgpjLmNyZWF0ZVBldHMoY3NyZnRva2VuPSIqKioiKQojIGluOiBoZWFkZXIsIG5hbWU6IHgtZm9vCmMuY3JlYXRlUGV0cygqKnsieC1mb28iOiAiKioqIn0pCiMgaW46IHBhdGgsIG5hbWU6IHVzZXJJZApjLmdldFVzZXIodXNlcklkPTEpCiMgaW46IHF1ZXJ5LCBuYW1lOiBvZmZzZXQKYy5saXN0VXNlcnMob2Zmc2V0PTEpCgojIGh0dHAgYm9keSwganVzdCBsaWtlIHJlcXVlc3RzLlNlc3Npb24KYy5jcmVhdGVQZXRzKGpzb249e30pIG9yIGMuY3JlYXRlUGV0cyhkYXRhPXt9KQpgYGAKCiMjIEluc3RhbGxhdGlvbgoKYGBgCnBpcCBpbnN0YWxsIHJlcXVlc3RzLW9wZW5hcGkKYGBgCgojIyBMaWNlbnNlCgpNSVQK readmeEtag: '"77db283aa6864fae56676707aac289039c9b1e0b"' readmeLastModified: Wed, 07 Feb 2024 16:35:18 GMT repositoryId: 188546010 description: >- A lightweight but powerful and easy-to-use Python client library for OpenAPI v3. created: '2019-05-25T09:22:34Z' updated: '2025-11-29T13:57:33Z' language: Python archived: false stars: 65 watchers: 3 forks: 8 owner: wy-z logo: https://avatars.githubusercontent.com/u/4213483?v=4 license: MIT repoEtag: '"302b9cffd30daa6c8f7aa629c5e9d684fa48ca31248e061933011b6951dfb79e"' repoLastModified: Sat, 29 Nov 2025 13:57:33 GMT foundInMaster: true category: - Data Validators - Parsers id: bfbe253fe4519d1b8b32b8770b5622e5 - source: openapi3 tags repository: https://github.com/fsprojects/openapitypeprovider v3: true repositoryMetadata: base64Readme: >- <img src="https://github.com/Dzoukr/OpenAPITypeProvider/raw/master/logo.jpg" alt="drawing" width="100px"/>


# OpenAPI Type Provider
*Erased netstandard2.0 type provider for web API providers.*

## Why to use this type provider?
If you write F# backend for some application providing JSON API, you probably want to document this using [OpenAPI v3 specification](https://www.openapis.org/) (previously called Swagger). This documentation can be created basically two ways:

1) Generate it from code (code first approach)
2) Write it manually (document first approach)

This type provider is focused on second option when you already got existing documentation (e.g. from frontend developer) and you want to be 100% sure that your API follows it correctly, which mostly means two things: Validation of requests payload is in correct form (as described in API documentation) and creating responses. Both things can be quite tedious and error-prone. It is common in web API development that after some time that server behavior is not what is written in documentation. This type provider is here to help with that.

## Goals of OpenAPITypeProvider
When I started to think about writing this type provider, I set few goals:

1. Netstandard2.0 support
2. Erased type
3. Tightly connected to Newtonsoft.Json
4. Based on the latest OpenAPI specification (no support for Swagger)

## Version 2.0.0 change
In versions < 2.0, the Schemas were created based on simplified logic: If the schema is the same (having same structure), it is considered to be the same schema no matter name you use (actually the first parsed named is used for all others). This approach had good intentions - to minimize amount of created Schemas - however shown to be wrong for complex scenarios. This could easily lead to situation where change in one Schema breaks your other Schema, even if they were not directly linked using `$ref`.

Since version **2.0.0** the Schemas are created based on logic:
1. Root inline schemas are always created as separated one
2. Root `$ref` schemas are created as separated one but with structure "copied" from referenced schema
3. Nested inline schemas are created as separated one
4. Nested `$ref` schemas are linked to referenced schema

## Instalation

First install [NuGet package](https://www.nuget.org/packages/OpenAPITypeProvider/)

    Install-Package OpenAPITypeProvider

or using Paket

    nuget OpenAPITypeProvider

## How to use

First open correct namespace and create basic type based on your YAML documentation

```fsharp
open OpenAPITypeProvider

type PetStore = OpenAPIV3Provider<"PetStore.yaml">
```

Now you can use defined Schemas in your specification like F# types.

### Parsing from JSON

Each Schema type can be created from JSON string using static method `Parse`.

```fsharp
let json = """{"name":"Roman"}"""
let pet = PetStore.Schemas.NewPet.Parse(json)
let name = pet.Name // contains Roman
let tag = pet.Tag // contains Option value None
```

Sometimes you need to use parse JSON with custom date format.

```fsharp
let json = """{"date1":"31. 12. 2018 12:34:56","date2":"31. 12. 2017 12:34:56"}"""
let customDateFormat = "dd. MM. yyyy HH:mm:ss"
let twoDates = PetStore.Schemas.TwoDates.Parse(json, customDateFormat)
```

Method `Parse` throws an exception in case JSON does not fit to definition:

```fsharp
// fails with exception that property 'name' is not present,
// but should be based on Schema definition
let json = """{"notExistingProperty":"Roman"}"""
let pet = PetStore.Schemas.NewPet.Parse(json) 

// fails with exception that property 'name' not convertible
// to string value
let json = """{"name":123456}"""
let pet = PetStore.Schemas.NewPet.Parse(json) 
```

### Instantiation of Schema types

Schema types has constructors based on definition so you can instantiate them as you need.

```fsharp
let pet = new PetStore.Schemas.NewPet(name = "Roman")
```

### Converting to JSON

Each Schema instance has method `ToJson` with few overloads.

```fsharp
let pet = new PetStore.Schemas.NewPet(name = "Roman")
pet.ToJson() // returns '{"name":"Roman"}' - no indenting
pet.ToJson(Newtonsoft.Json.Formatting.Indented) // return json with fancy formatting
```

### Converting to JSON with custom serializer

Each `ToJson` method has overload with supporting `JsonSerializerSettings` as parameter.

```fsharp
let pet = new PetStore.Schemas.NewPet(name = "Roman")

let settings = JsonSerializerSettings()
settings.NullValueHandling <- NullValueHandling.Include

pet.ToJson(settings, Formatting.None) // returns '{"name":"Roman","tag":null}'
```

### Converting to Newtonsoft JToken

If you need `JToken` instead of string with JSON, method `ToJToken` is here for you.

```fsharp
let pet = new PetStore.Schemas.NewPet(name = "Roman")
let jtoken = pet.ToJToken()
```

Again, you can customize how to handle optional values.

```fsharp
let pet = new PetStore.Schemas.NewPet(name = "Roman")
let jtoken = pet.ToJToken(NullValueHandling.Include) // this now contains JNull value inside JObject
```

### Simple values

By specification you are allowed to have Schema types not an objects, but simple values like strings or integers. This type provider supports them as well.

```yaml
SimpleString:
    type: string    
    
SimpleArray:
    type: array
    items:
        type: string
```

```fsharp
let simpleString = PetStore.Schemas.SimpleString("ABC")
simpleString.Value // contains "ABC"

let simpleArray = PetStore.Schemas.SimpleArray(["A";"B"])
simpleArray.Values // contains List<string> ["A";"B"]
```

### Requests & ResponseBodies

Using Schema types directly is quite handy and straightforward, but it doesn't say anything about routes, requests and responses. If you want to be 100% sure that you are fullfilling specification, go for `Parse` on Requests and `ToJson` / `ToJToken` methods on ResponseBodies.

```fsharp
let petStoreAPI = new PetStore() // Note! Instance of PetStore type is needed here.
let pet = new PetStore.Schemas.NewPet("Roman")

// this route returns NewPet schema by definition so ToJson allows only NewPet schema as parameter
petStoreAPI.Paths.``/pets/{id}``.Get.Responses.``200``.``application/json``.ToJson(pet)

// this route expects NewPet schema by definition so Parse method returns NewPet
let parsedPet = petStoreAPI.Paths.``/pets``.Post.RequestBody.``application/json``.Parse(jsonFromRequest)
```

In case you have any doubts, you can always have a look [at unit tests](https://github.com/Dzoukr/OpenAPITypeProvider/blob/master/tests/OpenAPITypeProvider.Tests/BasicTests.fs)

## Known issues & Limitations
1. No support for `OneOf` and `ManyOf` since they are basically union types which is quite difficult (or maybe impossible) to generate from type provider

## Contribution

You are more than welcome to send a [pull request](https://github.com/Dzoukr/OpenAPITypeProvider/pulls) if you find some bug or missing functionality.
 readmeEtag: '"b64c6755beae2e67b78991440ea9c3ed33b596db"' readmeLastModified: Mon, 28 Jan 2019 20:18:17 GMT repositoryId: 118037484 description: F# type provider for Open API specification created: '2018-01-18T21:00:25Z' updated: '2025-02-18T13:30:07Z' language: F# archived: false stars: 65 watchers: 9 forks: 5 owner: fsprojects logo: https://avatars.githubusercontent.com/u/6001315?v=4 repoEtag: '"acf47739a5b9cd7b0d8c84b9b658f4ad9cae873bf442cc55c4d7c5861b549df0"' repoLastModified: Tue, 18 Feb 2025 13:30:07 GMT foundInMaster: true category: Parsers id: 6b6a608ac17ac38db46546688326462f - source: openapi3 tags repository: https://github.com/marcomuellner/openapi-python-generator v3: true id: f16c0d05a415eb485dad3d30fa66ea5c repositoryMetadata: base64Readme: >- IyBPcGVuYXBpIFB5dGhvbiBHZW5lcmF0b3IKClshW1B5UEldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS92L29wZW5hcGktcHl0aG9uLWdlbmVyYXRvci5zdmcpXVtweXBpX10KWyFbU3RhdHVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvc3RhdHVzL29wZW5hcGktcHl0aG9uLWdlbmVyYXRvci5zdmcpXVtzdGF0dXNdClshW1B5dGhvbiBWZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvcHl2ZXJzaW9ucy9vcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3IpXVtweXRob24gdmVyc2lvbl0KWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL2wvb3BlbmFwaS1weXRob24tZ2VuZXJhdG9yKV1bbGljZW5zZV0KClshW10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9zdGF0aWMvdjE/bGFiZWw9ZG9jdW1lbnRhdGlvbiZtZXNzYWdlPWVuYWJsZWQmY29sb3I9PENPTE9SPildW2RvY3VtZW50YXRpb25dClshW1Rlc3RzXShodHRwczovL2dpdGh1Yi5jb20vTWFyY29NdWVsbG5lci9vcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3Ivd29ya2Zsb3dzL1Rlc3RzL2JhZGdlLnN2ZyldW3Rlc3RzXQpbIVtDb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvTWFyY29NdWVsbG5lci9vcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3IvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnKV1bY29kZWNvdl0KClshW3ByZS1jb21taXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcHJlLS1jb21taXQtZW5hYmxlZC1icmlnaHRncmVlbj9sb2dvPXByZS1jb21taXQmbG9nb0NvbG9yPXdoaXRlKV1bcHJlLWNvbW1pdF0KWyFbQmxhY2tdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvY29kZSUyMHN0eWxlLWJsYWNrLTAwMDAwMC5zdmcpXVtibGFja10KCltweXBpX106IGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9vcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3IvCltzdGF0dXNdOiBodHRwczovL3B5cGkub3JnL3Byb2plY3Qvb3BlbmFwaS1weXRob24tZ2VuZXJhdG9yLwpbcHl0aG9uIHZlcnNpb25dOiBodHRwczovL3B5cGkub3JnL3Byb2plY3Qvb3BlbmFwaS1weXRob24tZ2VuZXJhdG9yCltkb2N1bWVudGF0aW9uXTogaHR0cHM6Ly9tYXJjb211ZWxsbmVyLmdpdGh1Yi5pby9vcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3IvClt0ZXN0c106IGh0dHBzOi8vZ2l0aHViLmNvbS9NYXJjb011ZWxsbmVyL29wZW5hcGktcHl0aG9uLWdlbmVyYXRvci9hY3Rpb25zP3dvcmtmbG93PVRlc3RzCltjb2RlY292XTogaHR0cHM6Ly9hcHAuY29kZWNvdi5pby9naC9NYXJjb011ZWxsbmVyL29wZW5hcGktcHl0aG9uLWdlbmVyYXRvcgpbcHJlLWNvbW1pdF06IGh0dHBzOi8vZ2l0aHViLmNvbS9wcmUtY29tbWl0L3ByZS1jb21taXQKW2JsYWNrXTogaHR0cHM6Ly9naXRodWIuY29tL3BzZi9ibGFjawoKIVtdKGxvZ28ucG5nKQoKLS0tCl9fRG9jdW1lbnRhdGlvbjpfXyBbaGVyZV1bZG9jdW1lbnRhdGlvbl0KCi0tLQoKIyMgRmVhdHVyZXMKCi0gX19FYXNlIG9mIHVzZV9fLiBQcm92aWRlIGlucHV0LCBvdXRwdXQgYW5kIHRoZSBsaWJyYXJ5LCBhbmQgdGhlIGdlbmVyYXRvciB3aWxsIGRvIHRoZSByZXN0LgotIF9fVHlwZSBzYWZldHkgYW5kIHR5cGUgaGludGluZy5fXyBfX09wZW5BUEkgcHl0aG9uIGdlbmVyYXRvcl9fIG1ha2VzIGhlYXZ5IHVzZSBvZiBweWRhbnRpYyBtb2RlbHMgdG8gcHJvdmlkZSB0eXBlLXNhZmUgZGF0YSBzdHJ1Y3R1cmVzLgotIF9fU3VwcG9ydCBmb3IgbXVsdGlwbGUgcmVzdCBmcmFtZXdvcmtzLl9fIF9fT3BlbkFQSSBweXRob24gZ2VuZXJhdG9yX18gY3VycmVudGx5IHN1cHBvcnRzIHRoZSBmb2xsb3dpbmc6CiAgICAtIFtodHRweF0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2h0dHB4LykKICAgIC0gW3JlcXVlc3RzXShodHRwczovL3B5cGkub3JnL3Byb2plY3QvcmVxdWVzdHMvKQogICAgLSBbYWlvaHR0cF0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2Fpb2h0dHAvKQotIF9fQXN5bmMgYW5kIHN5bmMgY29kZSBnZW5lcmF0aW9uIHN1cHBvcnRfXywgZGVwZW5kaW5nIG9uIHRoZSBmcmFtZXdvcmsuIEl0IHdpbGwgYXV0b21hdGljYWxseSBjcmVhdGUgYm90aCBmb3IgZnJhbWV3b3JrcyB0aGF0IHN1cHBvcnQgYm90aC4KLSBfX0Vhc2lseSBleHRlbmRhYmxlIHVzaW5nIEppbmphMiB0ZW1wbGF0ZXNfXy4gVGhlIGNvZGUgaXMgZGVzaWduZWQgdG8gYmUgZWFzaWx5IGV4dGVuZGFibGUgYW5kIHNob3VsZCBzdXBwb3J0IGV2ZW4gbW9yZSBsYW5ndWFnZXMgYW5kIGZyYW1ld29ya3MgaW4gdGhlIGZ1dHVyZS4KLSBfX0Z1bGx5IHRlc3RlZF9fLiBFdmVyeSBnZW5lcmF0ZWQgY29kZSBpcyBhdXRvbWF0aWNhbGx5IHRlc3RlZCBhZ2FpbnN0IHRoZSBPcGVuQVBJIHNwZWMgYW5kIHdlIGhhdmUgMTAwJSBjb3ZlcmFnZS4KLSBfX1VzYWdlIGFzIENMSSBvciBhcyBsaWJyYXJ5X18uCgojIyBSZXF1aXJlbWVudHMKCi0gUHl0aG9uIDMuNysKCiMjIEluc3RhbGxhdGlvbgoKWW91IGNhbiBpbnN0YWxsIF9PcGVuYXBpIFB5dGhvbiBHZW5lcmF0b3JfIHZpYSBbcGlwXSBmcm9tIFtQeVBJXToKCmBgYGNvbnNvbGUKJCBwaXAgaW5zdGFsbCBvcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3IKYGBgCgojIyBVc2FnZQoKUGxlYXNlIHNlZSB0aGUgW1F1aWNrIHN0YXJ0IHBhZ2VdIGZvciBkZXRhaWxzLgoKIyMgUm9hZG1hcAoKLSBTdXBwb3J0IGZvciBhbGwgY29tbW9ubHkgdXNlZCBodHRwIGxpYnJhcmllcyBpbiB0aGUgcHl0aG9uIGVjb3N5c3RlbSAofn5yZXF1ZXN0c35+LCB1cmxsaWIsIC4uLikKLSBTdXBwb3J0IGZvciBtdWx0aXBsZSBsYW5ndWFnZXMKLSBTdXBwb3J0IGZvciBtdWx0aXBsZSBhdXRoZW50aWNhdGlvbiBzY2hlbWVzCi0gU3VwcG9ydCBjdXN0b20gdGhlbWVzCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHZlcnkgd2VsY29tZS4KVG8gbGVhcm4gbW9yZSwgc2VlIHRoZSBbQ29udHJpYnV0b3IgR3VpZGVdLgoKIyMgTGljZW5zZQoKRGlzdHJpYnV0ZWQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbTUlUIGxpY2Vuc2VdW2xpY2Vuc2VdLApfT3BlbmFwaSBQeXRob24gR2VuZXJhdG9yXyBpcyBmcmVlIGFuZCBvcGVuIHNvdXJjZSBzb2Z0d2FyZS4KCiMjIElzc3VlcwoKSWYgeW91IGVuY291bnRlciBhbnkgcHJvYmxlbXMsCnBsZWFzZSBbZmlsZSBhbiBpc3N1ZV0gYWxvbmcgd2l0aCBhIGRldGFpbGVkIGRlc2NyaXB0aW9uLgoKIyMgQ3JlZGl0cwoKU3BlY2lhbCB0aGFua3MgdG8gdGhlIHBlZXBzIGZyb20gW29wZW5hcGktc2NoZW1hLXB5ZGFudGljXShodHRwczovL2dpdGh1Yi5jb20va3VpbW9uby9vcGVuYXBpLXNjaGVtYS1weWRhbnRpYyksCndoaWNoIGFscmVhZHkgZGlkIGEgbG90IG9mIHRoZSBsZWd3b3JrIGJ5IHByb3ZpZGluZyBhIHB5ZGFudGljIHNjaGVtYSBmb3IgdGhlIE9wZW5BUEkgMy4wLjArIHNwZWNpZmljYXRpb24uCgpUaGlzIHByb2plY3Qgd2FzIGdlbmVyYXRlZCBmcm9tIFtAY2pvbG93aWN6XSdzIFtIeXBlcm1vZGVybiBQeXRob24gQ29va2llY3V0dGVyXSB0ZW1wbGF0ZS4KCltAY2pvbG93aWN6XTogaHR0cHM6Ly9naXRodWIuY29tL2Nqb2xvd2ljegpbcHlwaV06IGh0dHBzOi8vcHlwaS5vcmcvCltoeXBlcm1vZGVybiBweXRob24gY29va2llY3V0dGVyXTogaHR0cHM6Ly9naXRodWIuY29tL2Nqb2xvd2ljei9jb29raWVjdXR0ZXItaHlwZXJtb2Rlcm4tcHl0aG9uCltmaWxlIGFuIGlzc3VlXTogaHR0cHM6Ly9naXRodWIuY29tL01hcmNvTXVlbGxuZXIvb3BlbmFwaS1weXRob24tZ2VuZXJhdG9yL2lzc3VlcwpbcGlwXTogaHR0cHM6Ly9waXAucHlwYS5pby8KCjwhLS0gZ2l0aHViLW9ubHkgLS0+CgpbbGljZW5zZV06IGh0dHBzOi8vZ2l0aHViLmNvbS9NYXJjb011ZWxsbmVyL29wZW5hcGktcHl0aG9uLWdlbmVyYXRvci9ibG9iL21haW4vTElDRU5TRQpbY29udHJpYnV0b3IgZ3VpZGVdOiBodHRwczovL2dpdGh1Yi5jb20vTWFyY29NdWVsbG5lci9vcGVuYXBpLXB5dGhvbi1nZW5lcmF0b3IvYmxvYi9tYWluL0NPTlRSSUJVVElORy5tZApbUXVpY2sgc3RhcnQgcGFnZV06IGh0dHBzOi8vbWFyY29tdWVsbG5lci5naXRodWIuaW8vb3BlbmFwaS1weXRob24tZ2VuZXJhdG9yL3F1aWNrX3N0YXJ0Lwo= readmeEtag: '"413113cdd192d35685a4bdae3fb50e3743c87d8e"' readmeLastModified: Fri, 15 Mar 2024 21:55:10 GMT repositoryId: 506151512 description: 'A client generator from openapi for python. ' created: '2022-06-22T07:56:40Z' updated: '2026-01-29T10:16:28Z' language: Python archived: false stars: 92 watchers: 3 forks: 39 owner: MarcoMuellner logo: https://avatars.githubusercontent.com/u/13886384?v=4 license: MIT repoEtag: '"f83e0d2428eab10edae1396b7d97fb28b4a2d657ffd2b0e5459fbea501acfb78"' repoLastModified: Thu, 29 Jan 2026 10:16:28 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/apis-guru/google-discovery-to-swagger v3: true id: f6231c27c4bcb653f01628db4e528f60 repositoryMetadata: base64Readme: >- IyBnb29nbGUtZGlzY292ZXJ5LXRvLXN3YWdnZXIKClNjcmlwdCBmb3IgY29udmVydGluZyBbR29vZ2xlIERpc2NvdmVyeSBmb3JtYXRdKGh0dHBzOi8vZGV2ZWxvcGVycy5nb29nbGUuY29tL2Rpc2NvdmVyeS92MS9yZWZlcmVuY2UvYXBpcykgaW50byBbT3BlbkFQSSAzLjBdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKQoKIyMgVXNhZ2UKCklmIHlvdSBuZWVkIHRvIGNvbnZlcnQgc29tZSBBUElzIGZyb20gR29vZ2xlIERpc2NvdmVyeSBzZXJ2aWNlIEkgaGF2ZSBnb29kIG5ld3MgZm9yIHlvdSwgaXQgaXMgYWxyZWFkeSBbZG9uZV0oaHR0cHM6Ly9naXRodWIuY29tL0FQSXMtZ3VydS9vcGVuYXBpLWRpcmVjdG9yeS90cmVlL21hc3Rlci9BUElzL2dvb2dsZWFwaXMuY29tKSA6c21pbGU6CgpUaGlzIGxpYnJhcnkgd2FzIGRldmVsb3BlZCBvZiBhbm90aGVyIHByb2plY3Q6IFtBUElzLmd1cnVdKGh0dHBzOi8vZ2l0aHViLmNvbS9BUElzLWd1cnUvb3BlbmFwaS1kaXJlY3RvcnkpIC0gV2lraXBlZGlhIGZvciBXZWIgQVBJcy4KICAKKipJTVBPUlRBTlQqKjogZG9uJ3QgdXNlIEdpdGh1YiBSQVcgbGlua3MgYmVjYXVzZSB0aGV5IGFyZSBzdWJqZWN0IHRvIGNoYW5nZS4KSW5zdGVhZCB5b3UgY2FuIGZpbmQgZGlyZWN0IGxpbmtzIGZvciBhbnkgcGFydGljdWxhciBBUEkgaW4gW3RoaXMgbGlzdF0oaHR0cHM6Ly9hcGkuYXBpcy5ndXJ1L3YyL2xpc3QuanNvbikuCkFsbCBPcGVuQVBJIGRvY3VtZW50cyBhcmUgdXBkYXRlZCwgdmFsaWRhdGVkIGFuZCBmaXhlZCBvbiBhIHdlZWtseSBiYXNpcy4KCklmIHlvdSBoYXZlIG5vbi1Hb29nbGUgQVBJIGRvY3VtZW50IGluIHRoaXMgZm9ybWF0IHBsZWFzZSBjb25zaWRlciBhZGRpbmcgaXQgdG8gdGhlIGNvbGxlY3Rpb246IGp1c3Qgb3BlbiBhbiBbaXNzdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9BUElzLWd1cnUvb3BlbmFwaS1kaXJlY3RvcnkvaXNzdWVzL25ldykgd2l0aCBsaW5rIHRvIGl0LgoKSWYgaXQgaXMgYW4gKippbnRlcm5hbCoqIEFQSSB5b3UgY2FuIHVzZSBbYXBpLXNwZWMtY29udmVydGVyXShodHRwczovL2dpdGh1Yi5jb20vbHVjeWJvdC9hcGktc3BlYy1jb252ZXJ0ZXIpIHRvb2wgdG8gZG8gY29udmVyc2lvbiBmb3IgeW91LgoKIyMgQ3JlZGl0cwoqIFtJdmFuIEdvbmNoYXJvdl0oaHR0cHM6Ly9naXRodWIuY29tL0l2YW5Hb25jaGFyb3YvKQoqIFtNaWtlIFJhbHBoc29uXShodHRwczovL2dpdGh1Yi5jb20vTWlrZVJhbHBoc29uLykKCiMjIExpY2Vuc2UKCk1JVAo= readmeEtag: '"85bbb68a69ef685ca9e08d6316f9ebd8d06fc658"' readmeLastModified: Mon, 20 Feb 2023 15:02:12 GMT repositoryId: 31059694 description: Script for converting Google Discovery format into OpenAPI (swagger) 3.0 created: '2015-02-20T10:13:13Z' updated: '2026-01-06T04:16:31Z' language: JavaScript archived: false stars: 69 watchers: 3 forks: 22 owner: APIs-guru logo: https://avatars.githubusercontent.com/u/10975548?v=4 license: MIT repoEtag: '"54fd5a8498c5901be55fde56684588239c452e93e4b55f2deee0bb5301c1c87b"' repoLastModified: Tue, 06 Jan 2026 04:16:31 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/a-h/rest v3: true id: e20761e913a0567699fb6cea88173185 repositoryMetadata: base64Readme: >- IyBSRVNUCgpEb2N1bWVudCBhIFJFU1QgQVBJIHdpdGggYW4gT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbi4KCiogQ29kZSwgbm90IGNvbmZpZ3VyYXRpb24uCiogTm8gbWFnaWMgY29tbWVudHMsIHRhZ3MsIG9yIGRlY29yYXRvcnMuCiogVXNlIHdpdGggb3Igd2l0aG91dCBhIEdvIHdlYiBmcmFtZXdvcmsuCiogUG9wdWxhdGVzIHNjaGVtYSBhdXRvbWF0aWNhbGx5IHVzaW5nIHJlZmxlY3Rpb24uCgojIyBXaHkgd291bGQgSSB3YW50IHRvIHVzZSB0aGlzPwoKKiBBZGQgT3BlbkFQSSBkb2N1bWVudGF0aW9uIHRvIGFuIEFQSS4KICAqIENyZWF0ZSBhIGBzd2FnZ2VyLmpzb25gIG9yIGBzd2FnZ2VyLnlhbWxgIGZpbGUuCiogU2VydmUgdGhlIFN3YWdnZXIgVUkgdG8gY3VzdG9tZXJzLgoKIyMgRXhhbXBsZXMKClNlZSB0aGUgWy4vZXhhbXBsZXNdKC4vZXhhbXBsZXMpIGRpcmVjdG9yeSBmb3IgY29tcGxldGUgZXhhbXBsZXMuCgojIyMgQ3JlYXRlIGFuIE9wZW5BUEkgMy4wIChzd2FnZ2VyKSBmaWxlCgpgYGBnbwovLyBDb25maWd1cmUgdGhlIG1vZGVscy4KYXBpIDo9IHJlc3QuTmV3QVBJKCJtZXNzYWdlcyIpCmFwaS5TdHJpcFBrZ1BhdGhzID0gW11zdHJpbmd7ImdpdGh1Yi5jb20vYS1oL3Jlc3QvZXhhbXBsZSIsICJnaXRodWIuY29tL2EtaC9yZXNwb25kIn0KCmFwaS5SZWdpc3Rlck1vZGVsKHJlc3QuTW9kZWxPZltyZXNwb25kLkVycm9yXSgpLCByZXN0LldpdGhEZXNjcmlwdGlvbigiU3RhbmRhcmQgSlNPTiBlcnJvciIpLCBmdW5jKHMgKm9wZW5hcGkzLlNjaGVtYSkgewogIHN0YXR1cyA6PSBzLlByb3BlcnRpZXNbInN0YXR1c0NvZGUiXQogIHN0YXR1cy5WYWx1ZS5XaXRoTWluKDEwMCkuV2l0aE1heCg2MDApCn0pCgphcGkuR2V0KCIvdG9waWMve2lkfSIpLgogIEhhc1BhdGhQYXJhbWV0ZXIoImlkIiwgcmVzdC5QYXRoUGFyYW17CiAgICBEZXNjcmlwdGlvbjogImlkIG9mIHRoZSB0b3BpYyIsCiAgICBSZWdleHA6ICAgICAgYFxkK2AsCiAgfSkuCiAgSGFzUmVzcG9uc2VNb2RlbChodHRwLlN0YXR1c09LLCByZXN0Lk1vZGVsT2ZbbW9kZWxzLlRvcGljXSgpKS4KICBIYXNSZXNwb25zZU1vZGVsKGh0dHAuU3RhdHVzSW50ZXJuYWxTZXJ2ZXJFcnJvciwgcmVzdC5Nb2RlbE9mW3Jlc3BvbmQuRXJyb3JdKCkpCgovLyBDcmVhdGUgdGhlIHNwZWNpZmljYXRpb24uCnNwZWMsIGVyciA6PSBhcGkuU3BlYygpCmlmIGVyciAhPSBuaWwgewogIGxvZy5GYXRhbGYoImZhaWxlZCB0byBjcmVhdGUgc3BlYzogJXYiLCBlcnIpCn0KCi8vIFdyaXRlIHRvIHN0ZG91dC4KZW5jIDo9IGpzb24uTmV3RW5jb2Rlcihvcy5TdGRvdXQpCmVuYy5TZXRJbmRlbnQoIiIsICIgIikKZW5jLkVuY29kZShzcGVjKQpgYGAKCiMjIyBTZXJ2ZSBBUEkgZG9jdW1lbnRhdGlvbiBhbG9uZ3NpZGUgeW91ciBBUEkKCmBgYGdvCi8vIENyZWF0ZSByb3V0ZXMuCnJvdXRlciA6PSBodHRwLk5ld1NlcnZlTXV4KCkKcm91dGVyLkhhbmRsZSgiL3RvcGljcyIsICZnZXQuSGFuZGxlcnt9KQpyb3V0ZXIuSGFuZGxlKCIvdG9waWMiLCAmcG9zdC5IYW5kbGVye30pCgphcGkgOj0gcmVzdC5OZXdBUEkoIm1lc3NhZ2VzIikKYXBpLlN0cmlwUGtnUGF0aHMgPSBbXXN0cmluZ3siZ2l0aHViLmNvbS9hLWgvcmVzdC9leGFtcGxlIiwgImdpdGh1Yi5jb20vYS1oL3Jlc3BvbmQifQoKLy8gUmVnaXN0ZXIgdGhlIGVycm9yIHR5cGUgd2l0aCBjdXN0b21pc2F0aW9ucy4KYXBpLlJlZ2lzdGVyTW9kZWwocmVzdC5Nb2RlbE9mW3Jlc3BvbmQuRXJyb3JdKCksIHJlc3QuV2l0aERlc2NyaXB0aW9uKCJTdGFuZGFyZCBKU09OIGVycm9yIiksIGZ1bmMocyAqb3BlbmFwaTMuU2NoZW1hKSB7CiAgc3RhdHVzIDo9IHMuUHJvcGVydGllc1sic3RhdHVzQ29kZSJdCiAgc3RhdHVzLlZhbHVlLldpdGhNaW4oMTAwKS5XaXRoTWF4KDYwMCkKfSkKCmFwaS5HZXQoIi90b3BpY3MiKS4KICBIYXNSZXNwb25zZU1vZGVsKGh0dHAuU3RhdHVzT0ssIHJlc3QuTW9kZWxPZltnZXQuVG9waWNzR2V0UmVzcG9uc2VdKCkpLgogIEhhc1Jlc3BvbnNlTW9kZWwoaHR0cC5TdGF0dXNJbnRlcm5hbFNlcnZlckVycm9yLCByZXN0Lk1vZGVsT2ZbcmVzcG9uZC5FcnJvcl0oKSkKCmFwaS5Qb3N0KCIvdG9waWMiKS4KICBIYXNSZXF1ZXN0TW9kZWwocmVzdC5Nb2RlbE9mW3Bvc3QuVG9waWNQb3N0UmVxdWVzdF0oKSkuCiAgSGFzUmVzcG9uc2VNb2RlbChodHRwLlN0YXR1c09LLCByZXN0Lk1vZGVsT2ZbcG9zdC5Ub3BpY1Bvc3RSZXNwb25zZV0oKSkuCiAgSGFzUmVzcG9uc2VNb2RlbChodHRwLlN0YXR1c0ludGVybmFsU2VydmVyRXJyb3IsIHJlc3QuTW9kZWxPZltyZXNwb25kLkVycm9yXSgpKQoKLy8gQ3JlYXRlIHRoZSBzcGVjLgpzcGVjLCBlcnIgOj0gYXBpLlNwZWMoKQppZiBlcnIgIT0gbmlsIHsKICBsb2cuRmF0YWxmKCJmYWlsZWQgdG8gY3JlYXRlIHNwZWM6ICV2IiwgZXJyKQp9CgovLyBBcHBseSBhbnkgZ2xvYmFsIGN1c3RvbWlzYXRpb24uCnNwZWMuSW5mby5WZXJzaW9uID0gInYxLjAuMC4iCnNwZWMuSW5mby5EZXNjcmlwdGlvbiA9ICJNZXNzYWdlcyBBUEkiCgovLyBBdHRhY2ggdGhlIFN3YWdnZXIgVUkgaGFuZGxlciB0byB5b3VyIHJvdXRlci4KdWksIGVyciA6PSBzd2FnZ2VydWkuTmV3KHNwZWMpCmlmIGVyciAhPSBuaWwgewogIGxvZy5GYXRhbGYoImZhaWxlZCB0byBjcmVhdGUgc3dhZ2dlciBVSSBoYW5kbGVyOiAldiIsIGVycikKfQpyb3V0ZXIuSGFuZGxlKCIvc3dhZ2dlci11aSIsIHVpKQpyb3V0ZXIuSGFuZGxlKCIvc3dhZ2dlci11aS8iLCB1aSkKCi8vIEFuZCBzdGFydCBsaXN0ZW5pbmcuCmZtdC5QcmludGxuKCJMaXN0ZW5pbmcgb24gOjgwODAuLi4iKQpmbXQuUHJpbnRsbigiVmlzaXQgaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkgdG8gc2VlIEFQSSBkZWZpbml0aW9ucyIpCmZtdC5QcmludGxuKCJMaXN0ZW5pbmcgb24gOjgwODAuLi4iKQpodHRwLkxpc3RlbkFuZFNlcnZlKCI6ODA4MCIsIHJvdXRlcikKYGBgCgojIyBUYXNrcwoKIyMjIHRlc3QKCmBgYApnbyB0ZXN0IC4vLi4uCmBgYAoKIyMjIHJ1bi1leGFtcGxlCgpEaXI6IC4vZXhhbXBsZXMvc3RkbGliCgpgYGAKZ28gcnVuIG1haW4uZ28KYGBgCg== readmeEtag: '"ff454f37f0e4ee23fab1100a8417417d22a9cc88"' readmeLastModified: Sat, 04 May 2024 11:35:46 GMT repositoryId: 599199253 description: Generate OpenAPI 3.0 specifications from Go code. created: '2023-02-08T16:53:48Z' updated: '2025-12-12T13:16:31Z' language: Go archived: false stars: 74 watchers: 2 forks: 29 owner: a-h logo: https://avatars.githubusercontent.com/u/1029947?v=4 license: MIT repoEtag: '"7ebe203c2f0a52b0a831d2546eb18ecdbb47bfd98db89b7b364e4341a7436bdc"' repoLastModified: Fri, 12 Dec 2025 13:16:31 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/hh-h/aiohttp-swagger3 v3: true repositoryMetadata: base64Readme: >- YWlvaHR0cC1zd2FnZ2VyMwo9PT09PT09PT09PT09PT09CgouLiBpbWFnZTo6IGh0dHBzOi8vZ2l0aHViLmNvbS9oaC1oL2Fpb2h0dHAtc3dhZ2dlcjMvYWN0aW9ucy93b3JrZmxvd3MvY2kueWFtbC9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcgogICA6dGFyZ2V0OiBodHRwczovL2dpdGh1Yi5jb20vaGgtaC9haW9odHRwLXN3YWdnZXIzL2FjdGlvbnMKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvaGgtaC9haW9odHRwLXN3YWdnZXIzL21hc3Rlci5zdmc/c3R5bGU9ZmxhdAogICA6dGFyZ2V0OiBodHRwczovL2NvZGVjb3YuaW8vZ2l0aHViL2hoLWgvYWlvaHR0cC1zd2FnZ2VyMz9icmFuY2g9bWFzdGVyCi4uIGltYWdlOjogaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL3B5L2Fpb2h0dHAtc3dhZ2dlcjMuc3ZnCiAgIDp0YXJnZXQ6IGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9weS9haW9odHRwLXN3YWdnZXIzCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9weXRob24tMy43JTJCLWJyaWdodGdyZWVuLnN2ZwogICA6dGFyZ2V0OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3B5dGhvbi0zLjclMkItYnJpZ2h0Z3JlZW4uc3ZnCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlJTIwc3R5bGUtYmxhY2stYmxhY2suc3ZnCiAgIDp0YXJnZXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9hbWJ2L2JsYWNrCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL2wvYWlvaHR0cC1zd2FnZ2VyMy5zdmcKICAgOnRhcmdldDogaHR0cHM6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKQWJvdXQKPT09PT0KClBhY2thZ2UgZm9yIGRpc3BsYXlpbmcgc3dhZ2dlciBkb2NzIHZpYSBkaWZmZXJlbnQgVUkgYmFja2VuZHMgYW5kCm9wdGlvbmFsbHkgdmFsaWRhdGluZy9wYXJzaW5nIGFpb2h0dHAgcmVxdWVzdHMgdXNpbmcgc3dhZ2dlcgpzcGVjaWZpY2F0aW9uIDMuMCwga25vd24gYXMgT3BlbkFQSTMuCgpTdXBwb3J0ZWQgVUkgYmFja2VuZHMKPT09PT09PT09PT09PT09PT09PT09CgpNdWx0aXBsZSBVSSBiYWNrZW5kcyBjYW4gYmUgdXNlZCBvciBVSSBiYWNrZW5kIGNhbiBiZSBkaXNhYmxlZCBhdCBhbGwgaWYgb25seSBuZWVkZWQKdmFsaWRhdGlvbiB3aXRob3V0IGJlaW5nIGFibGUgdG8gdmlldyBkb2N1bWVudGF0aW9uLgoKLSBTd2FnZ2VyIFVJIC0gaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkKLSBSZURvYyAtIGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jCi0gUmFwaURvYyAtIGh0dHBzOi8vZ2l0aHViLmNvbS9tcmluOS9SYXBpRG9jCgpEb2N1bWVudGF0aW9uCj09PT09PT09PT09PT0KCmh0dHBzOi8vYWlvaHR0cC1zd2FnZ2VyMy5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvCgpEaXNhYmxlIHZhbGlkYXRpb24KPT09PT09PT09PT09PT09PT09Cgp8IFBhc3MgYGB2YWxpZGF0ZT1GYWxzZWBgIHRvIGBgU3dhZ2dlckRvY3NgYC9gYFN3YWdnZXJGaWxlYGAgY2xhc3MsIHRoZSBkZWZhdWx0IGlzIGBgVHJ1ZWBgCnwgQWxzbywgc29tZXRpbWVzIHZhbGlkYXRpb24gaGFzIHRvIGJlIGRpc2FibGVkIGZvciBhIHJvdXRlLAogIHRvIGRvIHRoaXMgeW91IGhhdmUgdG8gcGFzcyBgYHZhbGlkYXRlPUZhbHNlYGAgZHVyaW5nIHRoZSBpbml0aWFsaXphdGlvbiBvZiB0aGUgcm91dGUuCnwgZXguIGBgd2ViLnBvc3QoIi9yb3V0ZSIsIGhhbmRsZXIsIHZhbGlkYXRlPUZhbHNlKWBgLCB0aGUgZGVmYXVsdCBpcyBgYFRydWVgYAoKUmVxdWlyZW1lbnRzCj09PT09PT09PT09PQoKLSBweXRob24gPj0gMy45Ci0gYWlvaHR0cCA+PSAzLjguMAotIHB5eWFtbCA+PSA1LjQKLSBhdHRycyA+PSAxOS4zLjAKLSBweXRob24tZmFzdGpzb25zY2hlbWEgPj0gMi4xNS4wCi0gcmZjMzMzOS12YWxpZGF0b3IgPj0gMC4xLjQKCkxpbWl0YXRpb25zCj09PT09PT09PT09CgotICBvbmx5IGFwcGxpY2F0aW9uL2pzb24gYW5kIGFwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZCBzdXBwb3J0ZWQKICAgZm9yIG5vdywgYnV0IHlvdSBjYW4gY3JlYXRlIG93bgogICBgaGFuZGxlciA8aHR0cHM6Ly9naXRodWIuY29tL2hoLWgvYWlvaHR0cC1zd2FnZ2VyMy90cmVlL21hc3Rlci9leGFtcGxlcy9jdXN0b21faGFuZGxlcj5gX18KLSAgaGVhZGVyL3F1ZXJ5IHBhcmFtZXRlcnMgb25seSBzdXBwb3J0ZWQgc2ltcGxlL2Zvcm0gYXJyYXkKICAgc2VyaWFsaXphdGlvbiwgZS5nLiAxLDIsMyw0Ci0gIHNlZSBUT0RPIGJlbG93CgpJbnN0YWxsYXRpb24KPT09PT09PT09PT09CgpgYHBpcCBpbnN0YWxsIGFpb2h0dHAtc3dhZ2dlcjNgYAoKRXhhbXBsZQo9PT09PT09CgouLiBjb2RlOjogcHl0aG9uCgogICAgZnJvbSBhaW9odHRwIGltcG9ydCB3ZWIKICAgIGZyb20gYWlvaHR0cF9zd2FnZ2VyMyBpbXBvcnQgU3dhZ2dlckRvY3MsIFN3YWdnZXJJbmZvLCBTd2FnZ2VyVWlTZXR0aW5ncwoKICAgIGFzeW5jIGRlZiBnZXRfb25lX3BldChyZXF1ZXN0OiB3ZWIuUmVxdWVzdCwgcGV0X2lkOiBpbnQpIC0+IHdlYi5SZXNwb25zZToKICAgICAgICAiIiIKICAgICAgICBPcHRpb25hbCByb3V0ZSBkZXNjcmlwdGlvbgogICAgICAgIC0tLQogICAgICAgIHN1bW1hcnk6IEluZm8gZm9yIGEgc3BlY2lmaWMgcGV0CiAgICAgICAgdGFnczoKICAgICAgICAgIC0gcGV0cwogICAgICAgIHBhcmFtZXRlcnM6CiAgICAgICAgICAtIG5hbWU6IHBldF9pZAogICAgICAgICAgICBpbjogcGF0aAogICAgICAgICAgICByZXF1aXJlZDogdHJ1ZQogICAgICAgICAgICBkZXNjcmlwdGlvbjogVGhlIGlkIG9mIHRoZSBwZXQgdG8gcmV0cmlldmUKICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgICAgICAgICAgICBmb3JtYXQ6IGludDMyCiAgICAgICAgcmVzcG9uc2VzOgogICAgICAgICAgJzIwMCc6CiAgICAgICAgICAgIGRlc2NyaXB0aW9uOiBFeHBlY3RlZCByZXNwb25zZSB0byBhIHZhbGlkIHJlcXVlc3QKICAgICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgICBhcHBsaWNhdGlvbi9qc29uOgogICAgICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgICAgICAkcmVmOiAiIy9jb21wb25lbnRzL3NjaGVtYXMvUGV0IgogICAgICAgICIiIgogICAgICAgIGlmIHBldF9pZCBub3QgaW4gcmVxdWVzdC5hcHBbJ3N0b3JhZ2UnXToKICAgICAgICAgICAgcmFpc2Ugd2ViLkhUVFBOb3RGb3VuZCgpCiAgICAgICAgcmV0dXJuIHdlYi5qc29uX3Jlc3BvbnNlKHJlcXVlc3QuYXBwWydzdG9yYWdlJ11bcGV0X2lkXSkKCiAgICBkZWYgbWFpbigpOgogICAgICAgIGFwcCA9IHdlYi5BcHBsaWNhdGlvbigpCiAgICAgICAgc3dhZ2dlciA9IFN3YWdnZXJEb2NzKAogICAgICAgICAgICBhcHAsCiAgICAgICAgICAgIHN3YWdnZXJfdWlfc2V0dGluZ3M9U3dhZ2dlclVpU2V0dGluZ3MocGF0aD0iL2RvY3MvIiksCiAgICAgICAgICAgIGluZm89U3dhZ2dlckluZm8oCiAgICAgICAgICAgICAgICB0aXRsZT0iU3dhZ2dlciBQZXRzdG9yZSIsCiAgICAgICAgICAgICAgICB2ZXJzaW9uPSIxLjAuMCIsCiAgICAgICAgICAgICksCiAgICAgICAgICAgIGNvbXBvbmVudHM9ImNvbXBvbmVudHMueWFtbCIKICAgICAgICApCiAgICAgICAgc3dhZ2dlci5hZGRfcm91dGVzKFsKICAgICAgICAgICAgd2ViLmdldCgiL3BldHMve3BldF9pZH0iLCBnZXRfb25lX3BldCksCiAgICAgICAgXSkKICAgICAgICBhcHBbJ3N0b3JhZ2UnXSA9IHt9CiAgICAgICAgd2ViLnJ1bl9hcHAoYXBwKQoKTW9yZSBgZXhhbXBsZXMgPGh0dHBzOi8vZ2l0aHViLmNvbS9oaC1oL2Fpb2h0dHAtc3dhZ2dlcjMvdHJlZS9tYXN0ZXIvZXhhbXBsZXM+YF8KCkhvdyBpdCBoZWxwcwo9PT09PT09PT09PT0KCi4uIGltYWdlOjogaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2hoLWgvYWlvaHR0cC1zd2FnZ2VyMy9tYXN0ZXIvZG9jcy9fc3RhdGljL2NvbXBhcmlzb24ucG5nCgpGZWF0dXJlcwo9PT09PT09PQoKLSBhcHBsaWNhdGlvbi9qc29uCi0gYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkIChleGNlcHQgYXJyYXkgYW5kIG9iamVjdCkKLSBpdGVtcwotIHByb3BlcnRpZXMKLSBwYXR0ZXJuCi0gcmVxdWlyZWQKLSBlbnVtCi0gbWluaW11bSwgbWF4aW11bQotIGV4Y2x1c2l2ZU1pbmltdW0sIGV4Y2x1c2l2ZU1heGltdW0KLSBtaW5MZW5ndGgsIG1heExlbmd0aAotIG1pbkl0ZW1zLCBtYXhJdGVtcwotIHVuaXF1ZUl0ZW1zCi0gbWluUHJvcGVydGllcywgbWF4UHJvcGVydGllcwotIGRlZmF1bHQgKG9ubHkgcHJpbWl0aXZlcykKLSBhZGRpdGlvbmFsUHJvcGVydGllcwotIG51bGxhYmxlCi0gcmVhZE9ubHkKLSBhbGxPZiwgb25lT2YsIGFueU9mCi0gc3RyaW5nIGZvcm1hdHM6IGRhdGUsIGRhdGUtdGltZSwgYnl0ZSwgZW1haWwsIHV1aWQsIGhvc3RuYW1lLCBpcHY0LCBpcHY2Ci0gY3VzdG9tIHN0cmluZyBmb3JtYXQgdmFsaWRhdG9ycwoKVE9ETyAocmFpc2UgYW4gaXNzdWUgaWYgbmVlZGVkKQo9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CgotIG11bHRpcGxlT2YKLSBub3QKLSBhbGxvd0VtcHR5VmFsdWUKLSBDb21tb24gUGFyYW1ldGVycyBmb3IgQWxsIE1ldGhvZHMgb2YgYSBQYXRoIChzcGVjIGZpbGUgb25seSkKLSBtb3JlIHNlcmlhbGl6YXRpb24gbWV0aG9kcywgc2VlOiBodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL3NlcmlhbGl6YXRpb24vCi0gZW5jb2RpbmcKLSBmb3JtIGRhdGEgc2VyaWFsaXphdGlvbiAoYXJyYXksIG9iamVjdCkKLSBkZWZhdWx0IChhcnJheSwgb2JqZWN0KQo= readmeEtag: '"953fe68918e4be2145780da7762aeaba9d487c29"' readmeLastModified: Tue, 11 Feb 2025 10:48:58 GMT repositoryId: 157923095 description: >- Library for swagger documentation browsing and validating aiohttp requests using swagger specification 3.0 created: '2018-11-16T21:26:37Z' updated: '2025-11-05T06:40:04Z' language: Python archived: false stars: 61 watchers: 1 forks: 21 owner: hh-h logo: https://avatars.githubusercontent.com/u/8166963?v=4 license: Apache-2.0 repoEtag: '"8c3d60c8d4021fba0b025d2000661580d431462503a3aab3774f888a8593676a"' repoLastModified: Wed, 05 Nov 2025 06:40:04 GMT foundInMaster: true category: Server Implementations id: f215ffe0267c42662e80c69062c4948d - source: openapi3 tags repository: https://github.com/italia/api-oas-checker v3: true repositoryMetadata: base64Readme: >- IyBDaGVja2VyIHBlciBBUEkgY29uZm9ybWkgYWwgTW9kZWxsbyBkaSBJbnRlcm9wZXJhYmlsaXTDoAoKWyFbSm9pbiB0aGUgI2FwaSBjaGFubmVsXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1NsYWNrLSUyM2FwaS1ibHVlLnN2Zz9sb2dvPXNsYWNrKV0oaHR0cHM6Ly9kZXZlbG9wZXJzaXRhbGlhLnNsYWNrLmNvbS9tZXNzYWdlcy9DREtCWVRHNzQpClshW0FQSSBvbiBmb3J1bS5pdGFsaWEuaXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvRm9ydW0taW50ZXJvcGVyYWJpbGl0JUMzJUEwLWJsdWUuc3ZnKV0oaHR0cHM6Ly9mb3J1bS5pdGFsaWEuaXQvYy9waWFuby10cmllbm5hbGUvaW50ZXJvcGVyYWJpbGl0YSkKWyFbUkVBRE1FIGluIEVuZ2xpc2hdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUmVhZG1lLUVuZ2xpc2gtZGFya2dyZWVuLnN2ZyldKFJFQURNRS5lbi5tZCkKCvCfkqEgUXVlc3RvIHJlcG9zaXRvcnkgY29udGllbmUgdW4gY2hlY2tlciBpbi1icm93c2VyIGNoZSB2ZXJpZmljYSBhbGN1bmUgZGVsbGUgcmVnb2xlIHBlciBsZSBBUEkgUkVTVCBpbmRpY2F0ZSBuZWwgTW9kZWxsbyBkaSBJbnRlcm9wZXJhYmlsaXTDoC4KCvCfl4LvuI8gSSBwcm9nZXR0aSBhc3NvY2lhdGkgc29ubyBpbmRpY2F0aSBuZWxsJ1tBUEkgU3RhcnRlciBLaXRdKGh0dHBzOi8vZ2l0aHViLmNvbS90ZWFtZGlnaXRhbGUvYXBpLXN0YXJ0ZXIta2l0KS4KCvCfkajwn4+74oCN8J+SuyBMJ2FwcGxpY2F6aW9uZSBvbi1saW5lIHByb250YSBhbGwndXNvIMOoIGRpc3BvbmliaWxlIFtxdWldKGh0dHBzOi8vaXRhbGlhLmdpdGh1Yi5pby9hcGktb2FzLWNoZWNrZXIvKS4KCiMjIPCfk4sgUmVxdWlzaXRpCgotIE5vZGUuanMgMTYrICh2ZWRpIGAubnZtcmNgKQotIFlhcm4gKGNvbnNpZ2xpYXRvKSBvIG5wbQotIERvY2tlciAob3B6aW9uYWxlLCBwZXIgZXNlZ3VpcmUgaW4gY29udGFpbmVyKQoKIyMg8J+UjSBFc2VndWlyZSBpbCBjaGVjayBkZWxsZSBBUEkKCklsIG1vZG8gcGnDuSBzZW1wbGljZSBwZXIgY29udHJvbGxhcmUgdW4nQVBJIMOoIGRpIHV0aWxpenphcmUgcXVlc3RvIGNoZWNrZXIsIGluc2VyZW5kbyBpbCBjb250ZW51dG8gZGVsbCdBUEkgZSBzZWxlemlvbmFuZG8gdW4gc2V0IGRpIHJlZ29sZSAoZGkgZGVmYXVsdDogX0l0YWxpYW4gR3VpZGVsaW5lcyBGdWxsXykuIENsaWNjYW5kbyBzdSAiQ2hlY2siIHNhcsOgIHBvc3NpYmlsZSBlc2FtaW5hcmUgdHV0dGkgZ2xpIGVycm9yaSwgd2FybmluZywgaW5mbyBlIGhpbnQgcmlsZXZhdGkgZGEgU3BlY3RyYWwuCgrwn5OMICoqUGVyIGxhIHB1YmJsaWNhemlvbmUgZGkgdW5hIEFQSSBzdWwgQ2F0YWxvZ28gUERORCoqLCBlc2VndWlyZSBsJ09BUyBDaGVja2VyIGNvbiBpbCBwcm9maWxvICoqX0l0YWxpYW4gR3VpZGVsaW5lcyBGdWxsXyoqIGUgdmVyaWZpY2FyZSBjaGUgbG8gX3lhbWxfIHByZXNlbnRpICoqMCBlcnJvcmkqKiBlZCBhdXNwaWNhYmlsbWVudGUgKiowIHdhcm5pbmdzKiouCgpJbiBhbHRlcm5hdGl2YSwgw6ggcG9zc2liaWxlIGZhcmUgaWwgY2hlY2sgZGVsbGUgQVBJIHRyYW1pdGUgSURFLCBDTEkgZSBHaXRIdWIgQWN0aW9uOiBzaSByaW1hbmRhIGFsIHNlZ3VlbnRlIFtSRUFETUVdKGh0dHBzOi8vZ2l0aHViLmNvbS9pdGFsaWEvYXBpLW9hcy1jaGVja2VyLXJ1bGVzL2Jsb2IvbWFpbi9SRUFETUUubWQpIGRlbCByZXBvIFthcGktb2FzLWNoZWNrZXItcnVsZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9pdGFsaWEvYXBpLW9hcy1jaGVja2VyLXJ1bGVzKSBwZXIgdHV0dGUgbGUgaW5mb3JtYXppb25pLgoKIyMg8J+TpiBSZWdvbGUKCkxlIHJlZ29sZSBjaGUgaWwgY2hlY2tlciB1dGlsaXp6YSBzb25vIGdlc3RpdGUgaW4gdW4gcmVwb3NpdG9yeSBkZWRpY2F0bzogW2FwaS1vYXMtY2hlY2tlci1ydWxlc10oaHR0cHM6Ly9naXRodWIuY29tL2l0YWxpYS9hcGktb2FzLWNoZWNrZXItcnVsZXMpLgoKSSBydWxlc2V0IGRpc3BvbmliaWxpIHNvbm8gc2NhcmljYWJpbGkgZGFsbGUgW3JlbGVhc2UgZGVsIHJlcG9zaXRvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9pdGFsaWEvYXBpLW9hcy1jaGVja2VyLXJ1bGVzL3JlbGVhc2VzL2xhdGVzdCk6Ci0gW3NwZWN0cmFsLnltbF0oaHR0cHM6Ly9naXRodWIuY29tL2l0YWxpYS9hcGktb2FzLWNoZWNrZXItcnVsZXMvcmVsZWFzZXMvbGF0ZXN0L2Rvd25sb2FkL3NwZWN0cmFsLnltbCksIG8gX0l0YWxpYW4gR3VpZGVsaW5lcyBGdWxsXywgcXVlbGxlIGRpIGRlZmF1bHQKLSBbc3BlY3RyYWwtZ2VuZXJpYy55bWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9pdGFsaWEvYXBpLW9hcy1jaGVja2VyLXJ1bGVzL3JlbGVhc2VzL2xhdGVzdC9kb3dubG9hZC9zcGVjdHJhbC1nZW5lcmljLnltbCksIG8gX0Jlc3QgUHJhY3RpY2VzIE9ubHlfCi0gW3NwZWN0cmFsLXNlY3VyaXR5LnltbF0oaHR0cHM6Ly9naXRodWIuY29tL2l0YWxpYS9hcGktb2FzLWNoZWNrZXItcnVsZXMvcmVsZWFzZXMvbGF0ZXN0L2Rvd25sb2FkL3NwZWN0cmFsLXNlY3VyaXR5LnltbCksIG8gX0V4dHJhIFNlY3VyaXR5IENoZWNrc18KLSBbc3BlY3RyYWwtZnVsbC55bWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9pdGFsaWEvYXBpLW9hcy1jaGVja2VyLXJ1bGVzL3JlbGVhc2VzL2xhdGVzdC9kb3dubG9hZC9zcGVjdHJhbC1mdWxsLnltbCksIG8gX0l0YWxpYW4gR3VpZGVsaW5lcyBGdWxsICsgRXh0cmEgU2VjdXJpdHkgQ2hlY2tzXwoKIyMg8J+agCBBdnZpYXJlIGxhIHdlYiBhcHAgaW4gbG9jYWxlCgpRdWVzdGEgd2ViIGFwcCDDqCBiYXNhdGEgc3VsbGEgbGlicmVyaWEgUmVhY3QgZSB1c2EgV2VicGFjayBwZXIgZ2VuZXJhcmUgaWwgYnVuZGxlIGRlbGwnYXBwbGljYXppb25lIGNvbiBpbCBzdXBwb3J0byBkaSBCYWJlbCBwZXIgdHJhbnNwaWxhcmUgaWwgY29kaWNlIEphdmFTY3JpcHQuCgpQZXIgYXZ2aWFyZSBsJ2FwcGxpY2F6aW9uZToKCmBgYGJhc2gKJCB5YXJuCiQgeWFybiBzdGFydApgYGAKCkluIGFsdGVybmF0aXZhOgoKYGBgYmFzaAokIGRvY2tlci1jb21wb3NlIHVwIC0tYnVpbGQgc3RhcnQKYGBgCgplIGFsIHRlcm1pbmUgZGVsbGEgY29tcGlsYXppb25lIGNvbGxlZ2Fyc2kgYSBodHRwOi8vbG9jYWxob3N0OjMwMDAKCiMjIPCfp6ogVGVzdAoKUGVyIGVzZWd1aXJlIGkgdGVzdDoKCmBgYGJhc2gKJCB5YXJuIHRlc3QKYGBgCgojIyDwn5OaIERvY3VtZW50YXppb25lIGUgSW50ZWdyYXppb25pCgotICoqTW9kZWxsbyBkaSBJbnRlcm9wZXJhYmlsaXTDoCoqOiBbTGluZWUgZ3VpZGEgdWZmaWNpYWxpXShodHRwczovL2RvY3MuaXRhbGlhLml0L2l0YWxpYS9waWFuby10cmllbm5hbGUtaWN0L2xnLW1vZGVsbG9pbnRlcm9wZXJhYmlsaXRhLWRvY3MpCi0gKipSZWdvbGUgY29tcGxldGUqKjogUmVwb3NpdG9yeSBbYXBpLW9hcy1jaGVja2VyLXJ1bGVzXShodHRwczovL2dpdGh1Yi5jb20vaXRhbGlhL2FwaS1vYXMtY2hlY2tlci1ydWxlcykKLSAqKkdpdEh1YiBBY3Rpb24qKjogRXNlbXBpbyBkaSBbY29uZmlndXJhemlvbmUgcGVyIENJL0NEXShodHRwczovL2dpdGh1Yi5jb20vaXRhbGlhL2FwaS1vYXMtY2hlY2tlci1ydWxlcy9ibG9iL21haW4vZG9jcy9yZXNvdXJjZXMvZ2l0aHViLWFjdGlvbi55bWwpCi0gKipBUEkgU3RhcnRlciBLaXQqKjogW1Byb2dldHRpIGNvcnJlbGF0aV0oaHR0cHM6Ly9naXRodWIuY29tL3RlYW1kaWdpdGFsZS9hcGktc3RhcnRlci1raXQpCgojIyDinI3wn4+7IENvbnRyaWJ1dGkKCkdyYXppZSBhIFBhb2xvIEZhbG9tbywKRnJhbmNlc2NvIE1hcmludWNjaSwKR2l1c2VwcGUgRGUgTWFyY28sCkFuZHJlYSBNaXN1cmFjYSwKU2ltb25lIEVzcG9zaXRvLApSb2NjbyBBZmZpbml0bwplIFZpbmNlbnpvIENoaWFuZXNlIHBlciBpIHN1Z2dlcmltZW50aSBlZCBpIGNvbnRyaWJ1dGkhCgrimpnvuI8gSWwgY2hlY2tlciDDqCBiYXNhdG8gc3UgW1NwZWN0cmFsXShodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vc3BlY3RyYWwpLgoKPiAjIyMgUGVyY2jDqSBTcGVjdHJhbD8g8J+klAo+IExvIGFiYmlhbW8gcHJlZmVyaXRvIHJpc3BldHRvIGFkIGFsdHJpIHNvZnR3YXJlIHBlcmNow6kKbm9uIHJpY2hpZWRlIGwndXRpbGl6em8gZGkgZGF0YWJhc2UgbyBjb21wb25lbnRpIHNlcnZlciBhIGN1aSBpbnZpYXJlIGkgdHVvaSBkb2N1bWVudGkgT3BlbkFQSSAoT0FTIENoZWNrZXIgw6ggdW5hIHBhZ2luYSBzdGF0aWNhIGRlcGxveWF0YSBzdSBHaXRIdWIgUGFnZXMpIGUgcGVyY2jDqSBsYSBtYWdnaW9yIHBhcnRlIGRlbGxlIHJlZ29sZSDDqCBkZXNjcml0dGEgdHJhbWl0ZSBmaWxlIHN0YXRpY2kgKGUuZy4gWUFNTCk6CnRyYW5uZSBpbiBjYXNpIHNwZWNpZmljaSAoZXMuIHNldCBkaSByZWdvbGUgcGVyIGxhIHNlY3VyaXR5KSwgbm9uIMOoIG5lY2Vzc2FyaW8gcXVpbmRpIGVzZWd1aXJlIGNvZGljZSBKYXZhU2NyaXB0LiBJbm9sdHJlLCBnbGkgdXRlbnRpIHBvc3Nvbm8gc2VtcHJlIGxpbWl0YXJzaSBhZCBpbXBvcnRhcmUgbGUgc29sZSByZWdvbGUgc3RhdGljaGUuCj4KPiBMZSBhbHRlcm5hdGl2ZSB2YWx1dGF0ZSwgdWd1YWxtZW50ZSB2YWxpZGUsIHNvbm86Cj4gLSBbWmFsbHldKGh0dHBzOi8vZ2l0aHViLmNvbS96YWxhbmRvL3phbGx5KSBoYSBiaXNvZ25vIGRpIHVuIGRhdGFiYXNlIGUgbm9uIMOoIHBvc3NpYmlsZSBmYXJuZSB1biB3ZWJwYWNrYWdlOwo+IC0gW1NwZWNjeV0oaHR0cHM6Ly9naXRodWIuY29tL3dld29yay9zcGVjY3kpIHBhcmUgc3VwcG9ydGFyZSBzb2xvIHJlZ29sZSBpbiBKYXZhU2NyaXB0LCBtZW50cmUgcXVlc3RvIGNoZWNrZXIgdXRpbGl6emEgcGVyIGxvIHBpw7kgZGVpIGZpbGUgWUFNTCBzdGF0aWNpLgo= readmeEtag: '"ea499b9abd9854b72d3f4a4a19d54783cd8a86f6"' readmeLastModified: Mon, 01 Dec 2025 14:50:56 GMT repositoryId: 244353016 description: An OpenAPI 3 checker based on spectral. created: '2020-03-02T11:23:30Z' updated: '2026-02-05T15:01:48Z' language: JavaScript archived: false stars: 68 watchers: 7 forks: 27 owner: italia logo: https://avatars.githubusercontent.com/u/15377824?v=4 license: NOASSERTION repoEtag: '"4052731b21dff0bbe161b1e2122d55336a87657f32f1e07f02fd3cca46e31ffd"' repoLastModified: Thu, 05 Feb 2026 15:01:48 GMT foundInMaster: true category: Description Validators id: d0d29c360728cda0cf8a4698f3c79f4b - source: openapi3 tags repository: https://github.com/sideko-inc/sideko v3: true id: 233ad37dec563fa65e5320d96629990b repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiA8YSBocmVmPSJodHRwczovL3NpZGVrby5kZXYiPgo8aW1nIHNyYz0iaHR0cHM6Ly9zdG9yYWdlLmdvb2dsZWFwaXMuY29tL3NpZGVrby5hcHBzcG90LmNvbS9wdWJsaWNfYXNzZXRzL3dlYnNpdGVfYXNzZXRzL2xvZ28tc3ltYm9sLnN2ZyIgaGVpZ2h0PSI5NiI+CjxoMyBhbGlnbj0iY2VudGVyIj5TaWRla288L2gzPgo8L2E+CjwvcD4KPHAgYWxpZ249ImNlbnRlciI+CiBBdXRvbWF0ZSB5b3VyIEFQSSBXb3JrIGFuZCBBY2NlbGVyYXRlIEludGVncmF0aW9ucwo8L3A+CjxwIGFsaWduPSJjZW50ZXIiPgo8YSBocmVmPSJodHRwczovL3NpZGVrby5kZXYiPjxzdHJvbmc+V2Vic2l0ZTwvc3Ryb25nPjwvYT4gwrcKPGEgaHJlZj0iaHR0cHM6Ly9kb2NzLnNpZGVrby5kZXYiPjxzdHJvbmc+RG9jdW1lbnRhdGlvbjwvc3Ryb25nPjwvYT4KPC9wPgoKIyMgT3BlbkFQSSAtPiBEb3plbnMgb2YgVG9vbHMKLSDwn5qAICoqU0RLcyoqIHdpdGggdHlwZXMsIHRlc3RzLCBhdXRoZW50aWNhdGlvbiwgYW5kIGNvZGUgc25pcHBldHMKLSDwn5SEICoqTW9jayBTZXJ2ZXJzKiogdGhhdCBtaW1pYyBBUEkgYmVoYXZpb3IKLSDwn5OaICoqQVBJIERvY3VtZW50YXRpb24qKiB0aGF0IHN0YXlzIHVwIHRvIGRhdGUgYXV0b21hdGljYWxseQotIPCfkrsgKipDTEkgVG9vbHMqKiBmb3Igc2ltcGxlIEFQSSBpbnRlcmFjdGlvbnMKCiMjIEluc3RhbGxhdGlvbgoKCmBgYGJhc2gKIyDwn42PIG1hY09TCmJyZXcgaW5zdGFsbCBzaWRla28taW5jL3RhcC9zaWRla28KCiMg8J+QjSBweXRob24gLSBweXBpCnBpcCBpbnN0YWxsIHNpZGVrby1weQoKIyDwn5OmIGpzIC0gbnBtCm5wbSBpbnN0YWxsIC1nIEBzaWRla28vY2xpCgojIOKaoSBjdXJsCmN1cmwgLWZzU0wgaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL1NpZGVrby1JbmMvc2lkZWtvL21haW4vaW5zdGFsbC5zaCB8IHNoCmBgYAoKCiMjIFF1aWNrc3RhcnQgLSBQcmVtaXVtIFNES3MgaW4gU2Vjb25kcwpgYGBiYXNoCiMgYXV0aGVudGljYXRlICh1c2VzIG5hdGl2ZSBrZXljaGFpbiB0byBzdG9yZSBjcmVkZW50aWFscyBvbiBob3N0KQpzaWRla28gbG9naW4KCiMgaW50ZXJhY3RpdmVseSBjcmVhdGUgc2RrcwpzaWRla28gc2RrIGluaXQKYGBgCgojIyBMb2NhbCBBdXRoZW50aWNhdGlvbgpUaGlzIENMSSB1dGlsaXplcyB5b3VyIGNvbXB1dGVyJ3MgbmF0aXZlIHNlY3VyZSBzdG9yZToKLSBtYWNPUzogS2V5Y2hhaW4KLSBMaW51eDoga2V5dXRpbHMKLSBXaW5kb3dzOiBDcmVkZW50aWFsIE1hbmFnZXIKClRvIGF1dGhlbnRpY2F0ZSBsb2NhbGx5IHJ1bjogYGBgc2lkZWtvIGxvZ2luYGBgCgoKIyMgQXV0aGVudGljYXRpbmcgaW4gQ0kvQ0QKU2V0IGBTSURFS09fQVBJX0tFWWAgZW52aXJvbm1lbnQgdmFyaWFibGUgaW4gQ0kvQ0QuCjEuIFdpdGggeW91ciBvd24gQVBJIGtleSBydW46IGBzaWRla28gYWNjb3VudCBnZXQtbXktYXBpLWtleWAKMi4gV2l0aCBhIHNlcnZpY2UgYWNjb3VudCBbc2VlIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vZG9jcy5zaWRla28uZGV2L3NpZGVrby1wbGF0Zm9ybS9zZXJ2aWNlLWFjY291bnRzKQoKIyBSZXNvdXJjZXMKLSBb8J+TmCBDb21wbGV0ZSBDTEkgUmVmZXJlbmNlIERvY3NdKC4vZG9jcy9DTEkubWQpCi0gW/Cfk50gUmVwb3J0IGFuIElzc3VlXShodHRwczovL2dpdGh1Yi5jb20vU2lkZWtvLUluYy9zaWRla28vaXNzdWVzL25ldz90ZW1wbGF0ZT1nZW5lcmF0aW9uLWJ1Zy5tZCkKLSBb4pqW77iPIExpY2Vuc2VdKC4vTElDRU5TRSkKLSBTaGVsbCBhdXRvY29tcGxldGU6IGBzaWRla28gY29uZmlnIGF1dG9jb21wbGV0ZSAtLXNoZWxsICh5b3VyIHNoZWxsKWAgKG11c3QgcnVuIHRoaXMgY29tbWFuZCB3aXRoIGVhY2ggU2lkZWtvIENMSSB1cGRhdGUpCgotLS0KCjwvZGV0YWlscz4KCjxkZXRhaWxzPgo8c3VtbWFyeT5MZWdhY3kgQ0xJIEluc3RhbGxhdGlvbnM8L3N1bW1hcnk+CgpgYGBiYXNoCiMgdmlhIGN1cmwKY3VybCAtZnNTTCBodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vU2lkZWtvLUluYy9zaWRla28vdjAuMTAuMi9pbnN0YWxsLnNoIHwgc2gKCiMgdmlhIHBpcApwaXAgaW5zdGFsbCBzaWRla28tcHk9PTAuMTAuMgpgYGAKPC9kZXRhaWxzPgo= readmeEtag: '"2d7f5bbe89e89fd3c19054f60df47a445384868f"' readmeLastModified: Tue, 19 Aug 2025 20:45:41 GMT repositoryId: 699872484 description: Generate SDKs and docs for your API created: '2023-10-03T14:04:44Z' updated: '2025-09-21T02:32:08Z' language: Rust archived: false stars: 88 watchers: 5 forks: 3 owner: Sideko-Inc logo: https://avatars.githubusercontent.com/u/122366043?v=4 license: NOASSERTION repoEtag: '"0f04d5aca02f9d54a99ce8b354a103b03665967362c5f6e560f06e103e0bddb3"' repoLastModified: Sun, 21 Sep 2025 02:32:08 GMT category: SDK foundInMaster: true oldLocations: - https://github.com/sideko-inc/sdk-generator - source: openapi3 tags repository: https://github.com/apis-guru/aws2openapi v3: true repositoryMetadata: base64Readme: >- IyBhd3Myb3BlbmFwaQpBbWF6b24gV2ViIFNlcnZpY2VzIEFQSSBkZXNjcmlwdGlvbiB0byBPcGVuQVBJIDMuMCBzcGVjaWZpY2F0aW9uCgojIyBXb3JrIGluIHByb2dyZXNzIC0gYmV0YSBxdWFsaXR5CgojIyMgSGFuZGxlcyBhbGwgY3VycmVudCAodjIpIEFXUyBqc29uIGFuZCB4bWwgc3BlY2lmaWNhdGlvbnMKClJlc3VsdGFudCBPcGVuQXBpIHNwZWNpZmljYXRpb25zIHBhc3MgW211bHRpcGxlXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9zY2hlbWFzL3YzLjAvc2NoZW1hLnlhbWwpIFt2YWxpZGF0b3JzXShodHRwczovL2dpdGh1Yi5jb20vTWVybWFkZS9vYXMta2l0KQoKIVtzY3JlZW5zaG90XShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vQVBJcy1ndXJ1L2F3czJvcGVuYXBpL21haW4vZG9jcy9zY3JlZW5zaG90LnBuZykKClRoZSByZXN1bHRzIG9mIHRoaXMgY29udmVydGVyIGNhbiBiZSBmb3VuZCBbaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL0FQSXMtZ3VydS9vcGVuYXBpLWRpcmVjdG9yeS90cmVlL21hc3Rlci9BUElzL2FtYXpvbmF3cy5jb20pCgojIyMgVE9ETyAqKmhlbHAgd2FudGVkKioKCiogfn5GaXggaW5wdXQgaGVhZGVyIHBhcmFtZXRlciBzZWxlY3Rpdml0eX5+Ciogfn5Qcm9jZXNzIHByb3RvY29sOmVjMn5+Ciogfn5BdXRoZW50aWNhdGlvbn5+IFlvdSB3aWxsIGhhdmUgdG8gY2FsY3VsYXRlIEhNQUMgaGVhZGVycyBtYW51YWxseQoqIH5+cGFnaW5hdGlvbn5+Ciogfn5FeGFtcGxlc35+Ciogfn5XYWl0ZXJzfn4gKGFzIHZlbmRvciBleHRlbnNpb24pCiogVGVzdCBhZ2FpbnN0IGxpdmUgZW5kcG9pbnRzCiogVmVyaWZ5IG91dHB1dCBoZWFkZXIgbWFwcGluZ3MKKiBUZXN0IHBhdGgtd2l0aC1mcmFnbWVudCBoYWNrIHdvcmtzCiogVGVzdCBwYXRocyB3aXRoIGhhcmRjb2RlZCA/cGFyYW1ldGVycyB3b3JrCiogVmFsaWRhdGUgeG1sLWhhbmRsaW5nIGtleXdvcmQgdHJhbnNsYXRpb25zCg== readmeEtag: '"461c15eda2266d399b10a99de5e02ebbf93edbb6"' readmeLastModified: Wed, 15 Feb 2023 18:06:55 GMT repositoryId: 68278564 description: Amazon Web Services API description to OpenAPI 3.0 definition created: '2016-09-15T08:36:59Z' updated: '2025-08-07T22:28:43Z' language: JavaScript archived: false stars: 61 watchers: 3 forks: 12 owner: APIs-guru logo: https://avatars.githubusercontent.com/u/10975548?v=4 license: MIT repoEtag: '"03b8dacc07fc5075f1a8458ac9d889f2ebc3b24570c44095944cc578dfd78fe9"' repoLastModified: Thu, 07 Aug 2025 22:28:43 GMT foundInMaster: true category: Parsers id: 0309debf319b1f557d23ca065812fbc7 - source: openapi3 tags repository: https://github.com/cocreators-ee/apity v3: true id: 000c3a69f9b14fbece85d34193b80799 repositoryMetadata: base64Readme: >- IyDwn5OY77iPIGFwaXR5IC0gVHlwZWQgQVBJIGNsaWVudCBmb3IgU3ZlbHRlIGFuZCBTdmVsdGVLaXQKCkEgdHlwZWQgZmV0Y2ggY2xpZW50IGZvciBbb3BlbmFwaS10eXBlc2NyaXB0XShodHRwczovL2dpdGh1Yi5jb20vZHJ3cG93L29wZW5hcGktdHlwZXNjcmlwdCkgY29tcGF0aWJsZSB3aXRoIFN2ZWx0ZUtpdCdzIGN1c3RvbSBgZmV0Y2hgCgojIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKbnBtIGluc3RhbGwgQGNvY3JlYXRvcnMtZWUvYXBpdHkKYGBgCgpPcgoKYGBgYmFzaApwbnBtIGFkZCBAY29jcmVhdG9ycy1lZS9hcGl0eQpgYGAKCiMjIEZlYXR1cmVzCgotIFt4XSBTdXBwb3J0IG9mIEpTT04gcmVxdWVzdCBhbmQgcmVzcG9uc2VzIGZyb20gW09wZW5BUEkgMy4wXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbikKLSBbeF0gU3VwcG9ydCBvZiBgeyNhd2FpdH1gIHN5bnRheCBpbiBTdmVsdGUgdGVtcGxhdGVzCi0gW3hdIENvbXBhdGliaWxpdHkgd2l0aCBTdmVsdGVLaXQncyBgZmV0Y2hgIGluIGBsb2FkYCBmdW5jdGlvbnMKLSBbeF0gUmVxdWVzdCByZWxvYWRpbmcKLSBbeF0gQ29uZmlndXJhdGlvbiBvZiBkZWZhdWx0IFtmZXRjaCBvcHRpb25zXShodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9lbi1VUy9kb2NzL1dlYi9BUEkvZmV0Y2gjcGFyYW1ldGVycykKCk9uIHRoZSByb2FkbWFwOgoKLSBbIF0gQ2FjaGluZyBvZiBzdWJzZXF1ZW50IHJlcXVlc3RzIHdpdGggdGhlIHNhbWUgVVJMIGFuZCBwYXJhbWV0ZXJzCgojIyMgTGl2ZSBkZW1vCgpBcGl0eSBoYXMgYSBbbGl2ZSBkZW1vXShodHRwczovL2FwaXR5LWRlbW8udmVyY2VsLmFwcC8pIHdpdGggY29kZSBzYW1wbGVzLgpBbHNvIHlvdSBjYW4gcmVhZCBhbiBpbnRyb2R1Y3Rpb25hcnkgW2Jsb2cgcG9zdF0oaHR0cHM6Ly9kZXYudG8vZmJqb3JuL2EtdHlwZWQtaHR0cC1jbGllbnQtZm9yLXN2ZWx0ZWtpdC04OGIpLgoKIyMgVXNhZ2UKCiMjIyBHZW5lcmF0ZSB0eXBlc2NyaXB0IGRlZmluaXRpb24gZnJvbSBzY2hlbWEKCkJlZm9yZSB3b3JraW5nIHdpdGggdGhlIGxpYnJhcnksIHlvdSBuZWVkIHRvIGdlbmVyYXRlIGFuIEFQSSBzcGVjIHVzaW5nIFtvcGVuYXBpLXR5cGVzY3JpcHRdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29wZW5hcGktdHlwZXNjcmlwdCk6CgpgYGBiYXNoCm5weCBvcGVuYXBpLXR5cGVzY3JpcHQgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uIC0tb3V0cHV0IHNyYy9wZXRzdG9yZS50cwoK8J+agCBodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92My9vcGVuYXBpLmpzb24g4oaSIGZpbGU6Li9zcmMvcGV0c3RvcmUudHMgWzg3MG1zXQpgYGAKCiMjIyBVc2luZyBBcGl0eQoKQ29uZmlndXJlIEFwaXR5IGluc3RhbmNlIGFuZCBnZW5lcmF0ZSBmdW5jdGlvbnMgZm9yIG1ha2luZyBBUEkgY2FsbHM6CgpgYGB0cwovLyBGaWxlOiBhcGkudHMKCmltcG9ydCB7IEFwaXR5IH0gZnJvbSAnQGNvY3JlYXRvcnMtZWUvYXBpdHknCmltcG9ydCB0eXBlIHsgcGF0aHMgfSBmcm9tICdzcmMvcGV0c3RvcmUnCgpjb25zdCBhcGl0eSA9IEFwaXR5LmZvcjxwYXRocz4oKQoKLy8gZ2xvYmFsIGNvbmZpZ3VyYXRpb24KYXBpdHkuY29uZmlndXJlKHsKICAvLyBCYXNlIFVSTCB0byB5b3VyIEFQSQogIGJhc2VVcmw6ICdodHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjInLAogIC8vIFJlcXVlc3RJbml0IG9wdGlvbnMsIGUuZy4gZGVmYXVsdCBoZWFkZXJzCiAgaW5pdDogewogICAgLy8gbW9kZTogJ2NvcnMnCiAgICAvLyBoZWFkZXJzOiB7fQogIH0sCn0pCgovLyBjcmVhdGUgZmV0Y2ggb3BlcmF0aW9ucwpleHBvcnQgY29uc3QgZmluZFBldHNCeVN0YXR1cyA9IGFwaXR5CiAgLnBhdGgoJy9wZXQvZmluZEJ5U3RhdHVzJykKICAubWV0aG9kKCdnZXQnKQogIC5jcmVhdGUoKQpleHBvcnQgY29uc3QgYWRkUGV0ID0gYXBpdHkucGF0aCgnL3BldCcpLm1ldGhvZCgncG9zdCcpLmNyZWF0ZSgpCmBgYAoKRWFjaCBBUEkgY2FsbCBpcyByZXByZXNlbnRlZCBhcyBhIHJlcXVlc3Qgb2JqZWN0IHRoYXQgaGFzIHRoZSBmb2xsb3dpbmcgcHJvcGVydGllczoKCmBgYHR5cGVzY3JpcHQKdHlwZSBBcGlSZXF1ZXN0PFIgPSBhbnk+ID0gewogIC8vIFN2ZWx0ZSBzdG9yZSBjb250YWluaW5nIHRoZSByZXNwb25zZSBvZiB0aGUgQVBJIGNhbGwuCiAgcmVhZG9ubHkgcmVzcDogV3JpdGFibGU8QXBpUmVzcG9uc2U8Uj4gfCB1bmRlZmluZWQ+CgogIC8vIFN2ZWx0ZSBzdG9yZSB0aGF0IGNvbnRhaW5zIGEgcHJvbWlzZSBmb3IgYW4gQVBJIGNhbGwuCiAgLy8gSWYgeW91IHJlbG9hZCB0aGUgcmVxdWVzdCB1c2luZyByZWxvYWQoKSBmdW5jdGlvbiwgdGhpcyBzdG9yZSB3aWxsIGJlIHVwZGF0ZWQuCiAgcmVhZG9ubHkgcmVhZHk6IFdyaXRhYmxlPHVuZGVmaW5lZCB8IFByb21pc2U8QXBpUmVzcG9uc2U8Uj4+PgoKICAvLyBGdW5jdGlvbiB0aGF0IHJlbG9hZHMgdGhlIHJlcXVlc3Qgd2l0aCB0aGUgc2FtZSBwYXJhbWV0ZXJzLgogIHJlbG9hZDogKCkgPT4gUHJvbWlzZTxBcGlSZXNwb25zZTxSPj4KCiAgLy8gUHJvbWlzZSBmb3IgdGhlIEFQSSBjYWxsLgogIC8vIFVzZWZ1bCBmb3Igc2VydmVyIGNvZGUgYW5kIHBsYWNlcyB3aGVyZSB5b3UgY2FuJ3QgdXNlIHRoZSBgcmVhZHlgIHN0b3JlLgogIHJlc3VsdDogUHJvbWlzZTxBcGlSZXNwb25zZTxSPj4KfQpgYGAKCkVhY2ggcmVzcG9uc2UgaXMgYSBTdmVsdGUgc3RvcmUgcmV0dXJuaW5nIGVpdGhlciBhbiBgdW5kZWZpbmVkYCwgb3IgdGhlIGZvbGxvd2luZyBvYmplY3Q6CgpgYGB0cwp0eXBlIFN1Y2Nlc3NmdWxSZXNwPFI+ID0gewogIG9rOiB0cnVlCiAgLy8gVHlwZWQgb2JqZWN0IGZvciBhIHN1Y2Nlc3NmdWwgcmVxdWVzdC4gQnVpbHQgZnJvbSB0aGUgT3BlbkFQSSBzcGVjCiAgZGF0YTogUgogIC8vIEhUVFAgc3RhdHVzIGNvZGUKICBzdGF0dXM6IG51bWJlcgp9Cgp0eXBlIEZhaWxlZFJlc3AgPSB7CiAgb2s6IGZhbHNlCiAgZGF0YTogYW55CiAgLy8gSFRUUCBzdGF0dXMgY29kZQogIHN0YXR1czogbnVtYmVyCn0KCnR5cGUgQXBpUmVzcG9uc2U8Uj4gPSBTdWNjZXNzZnVsUmVzcDxSPiB8IEZhaWxlZFJlc3AKYGBgCgojIyMgRXJyb3IgaGFuZGxpbmcKClRoZXJlIGFyZSBjZXJ0YWluIGNvbmRpdGlvbnMgdW5kZXIgd2hpY2ggYW4gQVBJIHJlcXVlc3QgY291bGQgdGhyb3cgYW4gZXhjZXB0aW9uIHdpdGhvdXQKYWN0dWFsbHkgcmVhY2hpbmcgdGhlIGRlc2lyZWQgc2VydmVyLCBmb3IgZXhhbXBsZSwgdW5wcmVkaWN0YWJsZSBuZXR3b3JrIGlzc3Vlcy4gRm9yIHN1Y2gKY2FzZXMsIHRoZSBhcGkgcmVzcG9uc2Ugd2lsbCBjb250YWluIGEgc3RhdHVzIHNldCB0byBhIG5lZ2F0aXZlIG51bWJlciwgaW5kaWNhdGluZyB0aGF0CmFuIGV4Y2VwdGlvbiB3YXMgdGhyb3duLgoKYGBganMKewogIG9rOiBmYWxzZSwKICBzdGF0dXM6IC0xLAogIGRhdGE6IHVuZGVmaW5lZCwKfQpgYGAKCiMjIyBVc2luZyBBcGl0eSB3aXRoIGF3YWl0IHN5bnRheCBpbiB0ZW1wbGF0ZXMKCkFzc3VtaW5nIHlvdSd2ZSBjcmVhdGVkIGFuIGBzcmMvYXBpLnRzYCBmcm9tIFt1c2luZyBBcGl0eV0oI3VzaW5nLWFwaXR5KSBzZWN0aW9uOgoKYGBgc3ZlbHRlCjxzY3JpcHQgbGFuZz0idHMiPgogIGltcG9ydCB7IGZpbmRQZXRCeVN0YXR1cyB9IGZyb20gJ3NyYy9hcGkudHMnCiAgY29uc3QgcmVxdWVzdCA9IGZpbmRQZXRCeVN0YXR1cyh7IHN0YXR1czogJ3NvbGQnIH0pCiAgY29uc3QgcGV0c1JlYWR5ID0gcmVxdWVzdC5yZWFkeQo8L3NjcmlwdD4KCjxkaXY+CiAgeyNhd2FpdCAkcGV0c1JlYWR5fQogICAgPHA+TG9hZGluZy4uPC9wPgogIHs6dGhlbiByZXNwfQogICAgeyNpZiByZXNwLm9rfQogICAgICB7I2VhY2ggcmVzcC5kYXRhIGFzIHBldH0KICAgICAgICA8cD57cGV0Lm5hbWV9PC9wPgogICAgICB7L2VhY2h9CiAgICB7OmVsc2V9CiAgICAgIDxwPkVycm9yIHdoaWxlIGxvYWRpbmcgcGV0czwvcD4KICAgIHsvaWZ9CiAgey9hd2FpdH0KCiAgPGJ1dHRvbiBvbjpjbGljaz17KCkgPT4ge3JlcXVlc3QucmVsb2FkKCl9fT4KICAgIFJlbG9hZCBwZXRzCiAgPC9idXR0b24+CjwvZGl2PgpgYGAKCiMjIyBTdWJzY3JpYmluZyB0byByZXNwb25zZSBzdG9yZQoKQXNzdW1pbmcgeW91J3ZlIGNyZWF0ZWQgYW4gYHNyYy9hcGkudHNgIGZyb20gW3VzaW5nIEFwaXR5XSgjdXNpbmctYXBpdHkpIHNlY3Rpb246CgpgYGBzdmVsdGUKPHNjcmlwdCBsYW5nPSJ0cyI+CiAgaW1wb3J0IHsgZmluZFBldEJ5U3RhdHVzIH0gZnJvbSAnc3JjL2FwaS50cycKICBjb25zdCByZXF1ZXN0ID0gZmluZFBldEJ5U3RhdHVzKHsgc3RhdHVzOiAnc29sZCcgfSkKICBsZXQgbmFtZXMgPSBbXQoKICByZXF1ZXN0LnJlc3Auc3Vic2NyaWJlKHJlc3AgPT4gewogICAgaWYgKHJlc3Aub2spIHsKICAgICAgbmFtZXMgPSByZXNwLmRhdGEubWFwKHBldCA9PiBwZXQubmFtZSkKICAgIH0KICB9KQo8L3NjcmlwdD4KCjxkaXY+CiAgeyNlYWNoIG5hbWVzIGFzIG5hbWV9CiAgICA8cD57bmFtZX08L3A+CiAgey9lYWNofQo8L2Rpdj4KYGBgCgojIyMgVXNpbmcgaW4gbG9hZCBmdW5jdGlvbnMKCkZldGNoIG9wZXJhdGlvbnMgc3VwcG9ydCBTdmVsdGVLaXQncyBbbG9hZF0oaHR0cHM6Ly9raXQuc3ZlbHRlLmRldi9kb2NzL2xvYWQjbWFraW5nLWZldGNoLXJlcXVlc3RzKSBmdW5jdGlvbiBmcm9tIGArcGFnZS50c2AgYW5kIGArcGFnZS5zZXJ2ZXIudHNgLgoKQXNzdW1pbmcgeW91J3ZlIGNyZWF0ZWQgYW4gYHNyYy9hcGkudHNgIGZyb20gW3VzaW5nIEFwaXR5XSgjdXNpbmctYXBpdHkpIHNlY3Rpb246CgpgYGB0cwppbXBvcnQgeyBmaW5kUGV0QnlTdGF0dXMgfSBmcm9tICdzcmMvYXBpLnRzJwoKZXhwb3J0IGFzeW5jIGZ1bmN0aW9uIGxvYWQoeyBmZXRjaCB9KSB7CiAgY29uc3QgcmVxdWVzdCA9IGZpbmRQZXRCeVN0YXR1cyh7IHN0YXR1czogJ3NvbGQnIH0pCiAgY29uc3QgcmVzcCA9IGF3YWl0IHJlcXVlc3QucmVzdWx0CiAgaWYgKHJlc3Aub2spIHsKICAgIHJldHVybiB7IHBldHM6IHJlc3AuZGF0YSwgZXJyb3I6IHVuZGVmaW5lZCB9CiAgfSBlbHNlIHsKICAgIHJldHVybiB7IHBldHM6IFtdLCBlcnJvcjogJ0ZhaWxlZCB0byBsb2FkIHBldHMnIH0KICB9Cn0KYGBgCgojIEZpbmFuY2lhbCBzdXBwb3J0CgpUaGlzIHByb2plY3QgaGFzIGJlZW4gbWFkZSBwb3NzaWJsZSB0aGFua3MgdG8gW0NvY3JlYXRvcnNdKGh0dHBzOi8vY29jcmVhdG9ycy5lZSkuIFlvdSBjYW4gaGVscCB1cyBjb250aW51ZSBvdXIgb3BlbiBzb3VyY2Ugd29yayBieSBzdXBwb3J0aW5nIHVzIG9uIFtCdXkgbWUgYSBjb2ZmZWVdKGh0dHBzOi8vd3d3LmJ1eW1lYWNvZmZlZS5jb20vY29jcmVhdG9ycykuCgpbIVsiQnV5IE1lIEEgQ29mZmVlIl0oaHR0cHM6Ly93d3cuYnV5bWVhY29mZmVlLmNvbS9hc3NldHMvaW1nL2N1c3RvbV9pbWFnZXMvb3JhbmdlX2ltZy5wbmcpXShodHRwczovL3d3dy5idXltZWFjb2ZmZWUuY29tL2NvY3JlYXRvcnMpCg== readmeEtag: '"54532d3f9c27bf2ccea5fa6a5ecf300e4257def0"' readmeLastModified: Tue, 03 Oct 2023 11:45:20 GMT repositoryId: 591235475 description: A typed fetch client for openapi-typescript for use with SvelteKit created: '2023-01-20T08:55:39Z' updated: '2026-01-02T19:01:15Z' language: TypeScript archived: false stars: 59 watchers: 0 forks: 3 owner: cocreators-ee logo: https://avatars.githubusercontent.com/u/25827530?v=4 license: MIT repoEtag: '"113fd2c4e5ab36674702ac5edf42151fe6f706a43c86dd74d1ecdcf223a2c33d"' repoLastModified: Fri, 02 Jan 2026 19:01:15 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/meeshkan/hmt v3: true repositoryMetadata: base64Readme: >- # HMT

[![CircleCI](https://circleci.com/gh/meeshkan/hmt.svg?style=shield)](https://circleci.com/gh/meeshkan/hmt)
[![PyPI](https://img.shields.io/pypi/dm/hmt.svg)](https://pypi.org/project/hmt/)
[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://black.readthedocs.io/)
[![License: MIT](https://img.shields.io/badge/License-MIT-green.svg)](https://opensource.org/licenses/MIT)

The HTTP Mocking Toolkit (HMT) is a tool that mocks HTTP APIs for use in sandboxes as well as for automated and exploratory testing. It uses a combination of API definitions, recorded traffic and code in order to make crafting mocks as enjoyable as possible.

[Chat with us on Gitter](https://gitter.im/Meeshkan/community) to let us know about questions, problems or ideas!

## What's in this document

- [Installation](#installation)
- [Getting started with HMT](#getting-started-with-hmt)
  - [Tutorial](#tutorial)
- [Collect recordings of API traffic](#collect-recordings-of-api-traffic)
- [Build a HMT spec from recordings](#build-a-hmt-spec-from-recordings)
  - [Building modes](#building-modes)
- [Mock server traffic using a HMT spec](#mock-server-traffic-using-a-hmt-spec)
- [Development](#development)
  - [Getting started](#getting-started)
  - [Tests](#tests)
  - [Type-checking](#type-checking)
  - [Automated builds](#automated-builds)
  - [Publishing HMT as a PyPi package](#publishing-hmt-as-a-pypi-package)
- [Contributing](#contributing)
  - [Code of Conduct](#code-of-conduct)

## Installation
Install via [pip](https://pip.pypa.io/en/stable/installing/) (requires **Python 3.6+**):

```bash
pip install hmt
```

macOS users can install HMT with [Homebrew](https://brew.sh/):

```sh
brew tap meeshkan/tap
brew install hmt
```

Debian and Ubuntu users can install HMT with `apt`:

```sh
echo "deb [trusted=yes] https://dl.bintray.com/meeshkan/apt all main" | tee -a /etc/apt/sources.list
apt-get -qq update && apt-get install hmt
```


## Getting started with HMT

The basic HMT flow is **collect, build and mock.**
1. First, **collect** data from recorded server traffic and/or OpenAPI specs.
1. Then, **build** a schema that unifies these various data sources.
1. Finally, use this schema to create a **mock** server of an API.

### Tutorial

The quickest way to get an overview of HMT is to complete our interactive tutorial. It walks you through the collect, build, and mock flow - while also covering the concepts necessary for development.

_Note: This tutorial has been tested on Python 3.6, 3.7, and 3.8._

After installing HMT, you can begin the tutorial by invoking from the command line:

```bash
$ hmt tutorial
```

Once you've run this, you should see:

```bash
    __              __ 
   / /_  ____ ___  / /_
  / __ \/ __ `__ \/ __/
 / / / / / / / / / /_
/_/ /_/_/ /_/ /_/\__/


The tutorial!!
Press ENTER to continue...
```

If not, it's probably our fault. Please let us know by [filing an issue on this repo](https://github.com/meeshkan/hmt/issues).

## Collect recordings of API traffic

Let's look at how to build a HMT spec. First, you have to **collect** recordings of server traffic and/or OpenAPI server specs.

To record API traffic, the HMT CLI provides a `record` mode that captures API traffic using a proxy.

```bash
$ hmt record
```

This command starts HMT as a reverse proxy on the default port of `8000` and creates two directories: `logs` and `specs`. 

With [curl](https://curl.haxx.se/), for example, you can use HMT as a proxy like so:

```bash
$ curl http://localhost:8000/http://api.example.com
```

By default, the recording proxy treats the path as the target URL. It then writes a [`.jsonl`](https://jsonlines.org) file containing logs of all server traffic to the `logs` directory.  All logs are created in the [`http-types`](https://github.com/meeshkan/http-types) format. This is because HMT's `build` tool expects all recordings to be represented in a `.jsonl` file containing recordings represented in the `http-types` format.

For more information about recording, including direct file writing and kafka streaming, see the [recording documentation](./docs/RECORD.md).

## Build a HMT spec from recordings

Using the HMT CLI, you can **build** an OpenAPI schema from a single `.jsonl` file, in addition to any existing OpenAPI specs that describe how your service works.

```bash
$ hmt build --input-file path/to/recordings.jsonl 
```

_Note: The input file should be in [JSON Lines](http://jsonlines.org/) format and every line should be in [http-types](https://github.com/meeshkan/http-types) JSON format. For an example input file, see [recordings.jsonl](./resources/recordings.jsonl)._

Optionally, you can also specify an output directory using the `--out` flag followed by the path to this directory. By default, HMT will build the new OpenAPI specifications in the `specs` directory. 

Use dash (`--input-file -`) to read from standard input:

```bash
$ hmt build --input-file - < recordings.jsonl
```
### Building modes
You can use a mode flag to indicate how the OpenAPI spec should be built, for example:

```bash
hmt build --input-file path/to/recordings.jsonl --mode gen
```

Supported modes are:
* gen [default] - infer a schema from the recorded data
* replay - replay the recorded data based on exact matching

For more information about building, including mixing together the two modes and editing the created OpenAPI schema, see the [building documentation](./docs/BUILD.md).

## Mock server traffic using a HMT spec

You can use an OpenAPI spec, such as the one created with `hmt build`, to create a **mock** server using HMT.

```bash
$ hmt mock path/to/dir/
```

_Note: You can specify a path to the directory your OpenAPI spec is in or a path to one specific file._

For more information about mocking, including adding custom middleware and modifying the mocking schema JIT via an admin API, see the [mocking documentation](./docs/MOCK.md).

## Development

Here are some useful tips for building and running HMT from source. 

If you run into any issues, please [reach out to our team on Gitter](https://gitter.im/meeshkan/community).

### Getting started

1. Clone this repository: `git clone https://github.com/meeshkan/hmt`
1. Create a virtual environment: `python3 -m venv .venv && source .venv/bin/activate`
1. Install dependencies: `pip install --upgrade -e '.[dev]'`
1. Install [`pre-commit`](https://pre-commit.com/) hooks to automatically format code as a git hook: `pre-commit install`

### Tests

Run all checks:

```bash
$ python setup.py test
```

#### `pytest`

Run [tests/](https://github.com/meeshkan/hmt/tree/master/tests/) with `pytest`:

```bash
pytest
# or
python setup.py test
```

Configuration for `pytest` is found in [pytest.ini](https://github.com/meeshkan/hmt/tree/master/pytest.ini).

#### Formatting

Formatting is checked by the above mentioned `python setup.py test` command.

To fix formatting:

```sh
$ python setup.py format
```

#### `flake8`

Run style checks:

```bash
$ flake8 .
```

#### `pyright`

You can run type-checking by installing [pyright](https://github.com/microsoft/pyright) globally:

```bash
$ npm -i -g pyright
```

And then running:

```bash
$ pyright --lib
$ # or
$ python setup.py typecheck
```

Using the [Pyright extension](https://marketplace.visualstudio.com/items?itemName=ms-pyright.pyright) is recommended for development in VS Code.

### Automated builds

Configuration for CircleCI [build pipeline](https://app.circleci.com/pipelines/github/meeshkan/hmt) can be found in [.circleci/config.yml](./.circleci/config.yml).

### Publishing HMT as a PyPi package

To publish HMT as a PyPi package, complete the following steps:

1. Bump the version in [setup.py](./setup.py) if the version is the same as in the published [package](https://pypi.org/project/hmt/). Commit and push.
1. Run `python setup.py test` to check that everything works
1. To build and upload the package, run `python setup.py upload`. Insert PyPI credentials to upload the package to `PyPI`. The command will also run `git tag` to tag the commit as a release and push the tags to remote.

> To see what the different commands do, see `Command` classes in [setup.py](https://github.com/meeshkan/hmt/tree/master/setup.py).

## Contributing

Thanks for your interest in contributing! Please take a look at our [development guide](#development) for notes on how to develop the package locally.  A great way to start contributing is to [file an issue](https://github.com/meeshkan/hmt/issues/new/choose) or [make a pull request](https://github.com/meeshkan/hmt/pulls).

### Code of Conduct

Please note that this project is governed by the [Meeshkan Community Code of Conduct](https://github.com/meeshkan/code-of-conduct). By participating, you agree to abide by its terms.
 readmeEtag: '"6631662563d52c609f223dd4be9a0670d14646d2"' readmeLastModified: Mon, 07 Sep 2020 09:00:41 GMT repositoryId: 235120275 description: HTTP Mocking Toolkit created: '2020-01-20T14:27:44Z' updated: '2025-02-23T07:16:39Z' language: Python archived: true stars: 54 watchers: 3 forks: 7 owner: meeshkan logo: https://avatars.githubusercontent.com/u/32298527?v=4 license: MIT repoEtag: '"3b556621df8928502a1e2370749be1f9de55a81f8f5b75d6d558ce586ef06c08"' repoLastModified: Sun, 23 Feb 2025 07:16:39 GMT foundInMaster: true category: Testing id: 0a7406562ab364da1ea6a15dee9c91ac - source: openapi3 tags repository: https://github.com/yayoc/swagger-to-mock v3: true repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyLXRvLW1vY2sgWyFbdHJhdmlzLWNpXShodHRwczovL3RyYXZpcy1jaS5vcmcveWF5b2Mvc3dhZ2dlci10by1tb2NrLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL3lheW9jL3N3YWdnZXItdG8tbW9jaykgWyFbR3JlZW5rZWVwZXIgYmFkZ2VdKGh0dHBzOi8vYmFkZ2VzLmdyZWVua2VlcGVyLmlvL3lheW9jL3N3YWdnZXItdG8tbW9jay5zdmcpXShodHRwczovL2dyZWVua2VlcGVyLmlvLykKCk1vY2sgZGF0YSBnZW5lcmF0b3IgQ0xJIGZvciBTd2FnZ2VyMyAoT3BlbkFQSSAzKQoKIyMgSW5zdGFsbAoKYGBgc2hlbGwKbnBtIGkgLWcgc3dhZ2dlci10by1tb2NrCmBgYAoKIyMgR2VuZXJhdGUgbW9jayBkYXRhCgpgYGBzaGVsbApzd2FnZ2VyLXRvLW1vY2sgPFlPVVIgU1dBR0dFUiBGSUxFPgpgYGAKCndpbGwgZ2VuZXJhdGUgKipKU09OIGZpbGUqKiBwZXIgZWFjaCBBUEkgcmVzcG9uc2UuICAKSlNPTiBkYXRhIHZhbHVlcyBzaG91bGQgYmUgZXhhbXBsZSB2YWx1ZXMgb24geW91ciBzd2FnZ2VyIGlmIHlvdSBzcGVjaWZpZWQgZXhhbXBsZXMuICAKT3RoZXJ3aXNlLCBgc3dhZ2dlci10by1tb2NrYCBmb2xsb3dzIFtkYXRhIHR5cGUgcnVsZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS95YXlvYy9zd2FnZ2VyLXRvLW1vY2svI2RhdGEtdHlwZS1zdXBwb3J0KSBhbmQgZ2VuZXJhdGUgYXJiaXRyYXJ5IHZhbHVlcy4gIApJZiB0aGVyZSBpcyBubyBydWxlIGxpa2UgYGZvcm1hdGAsIHZhbHVlcyBzaG91bGQgYmUgYmVsb3cuCgpgYGB5YW1sCnN0cmluZzogIiIKbnVtYmVyOiAwCmludGVnZXI6IDAKYm9vbGVhbjogdHJ1ZQphcnJheTogW10Kb2JqZWN0OiB7fQpgYGAKCiMjIEV4YW1wbGUKCklmIHdlIHBhc3MgW2FuIGV4YW1wbGUgWUFNTCBmaWxlXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9leGFtcGxlcy92My4wL3BldHN0b3JlLWV4cGFuZGVkLnlhbWwpCgpgYGB5YW1sCnJlc3BvbnNlczoKICAnMjAwJzoKICAgIGRlc2NyaXB0aW9uOiBwZXQgcmVzcG9uc2UKICAgIGNvbnRlbnQ6CiAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgc2NoZW1hOgogICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgIGl0ZW1zOgogICAgICAgICAgICAkcmVmOiAnIy9jb21wb25lbnRzL3NjaGVtYXMvUGV0JwpgYGAKCmBzd2FnZ2VyLXRvLW1vY2tgIHdpbGwgZ2VuZXJhdGUgZmlsZSBuYW1lZCBgcGV0c19nZXRfMjAwLmpzb25gIGFuZCB0aGUgYm9keSBzaG91bGQgYmUgYmVsb3cKCmBgYGpzb24KWwogIHsKICAgICJuYW1lIjogIiIsCiAgICAidGFnIjogIiIsCiAgICAiaWQiOiAwCiAgfQpdCmBgYAoKIyMgRmlsZSBOYW1lCgpOYW1pbmcgSlNPTiBmaWxlIHdpbGwgZm9sbG93IHRoZSBmb3JtYXQgYmVsb3cuIApgJHtBUElfUEFUSH1fJHtIVFRQX01FVEhPRH1fJHtSRVNQT05TRV9TVEFUVVN9Lmpzb25gCgojIyBEYXRhIFR5cGUgU3VwcG9ydCBbSW4gUHJvZ3Jlc3NdCgpgc3dhZ2dlci10by1tb2NrYCB3aWxsIGZvbGxvdyBydWxlcyBiYXNlZCBvbiBbT3BlbkFQSSAzIHNwZWNpZmljYXRpb25dKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vZGF0YS1tb2RlbHMvZGF0YS10eXBlcy8pIGZvciBlYWNoIGRhdGEgdHlwZSwgSWYgdGhlIGV4YW1wbGUgdmFsdWUgaXMgbm90IHNwZWNpZmllZC4KCiMjIyBNaXhlZCBUeXBlcwoKSWYgeW91IHNwZWNpZnkgYG9uZU9mYCBvciBgYW55T2ZgLCBUaGUgdmFsdWUgc2hvdWxkIGJlIGF0IHRoZSB0b3AgdHlwZS4KCiMjIyBOdW1iZXJzCgotIGZvcm1hdAotIE1pbmltdW0gYW5kIE1heGltdW0KLSBNdWx0aXBsZXMKCiMjIyBTdHJpbmcKCi0gZm9ybWF0Ci0gcGF0dGVybgoKIyMjIEJvb2xlYW4KCnZhbHVlIHNob3VsZCBiZSBgdHJ1ZWAgb3IgYGZhbHNlYC4KCiMjIyBOdWxsCgojIyMgQXJyYXlzCgotIE1peGVkLVR5cGUgQXJyYXlzCi0gQXJyYXkgTGVuZ3RoCgojIyMgT2JqZWN0cwoKLSBGcmVlLUZvcm0gT2JqZWN0CgojIyMgRmlsZXMKCiMjIyBBbnkgVHlwZQo= readmeEtag: '"63c3bf0392868b1724f0a23bfcbf6925ab8d4f9e"' readmeLastModified: Tue, 29 Jun 2021 12:34:25 GMT repositoryId: 138289588 description: Mock data generator CLI for Swagger3 (OpenAPI 3) created: '2018-06-22T10:26:57Z' updated: '2025-08-08T16:31:19Z' language: TypeScript archived: false stars: 53 watchers: 2 forks: 8 owner: yayoc logo: https://avatars.githubusercontent.com/u/6383334?v=4 repoEtag: '"4cf84c6724f20d599c469c1a5dd492633fe3e9ee989d9042eecca63d4f7939d8"' repoLastModified: Fri, 08 Aug 2025 16:31:19 GMT foundInMaster: true category: Parsers id: 10c19215807691859c470cb694614c33 - source: openapi3 tags repository: https://github.com/smartbear/swaggerhub-cli v3: true repositoryMetadata: base64Readme: >- SwaggerHub CLI
==============

[![NPM](https://img.shields.io/npm/v/swaggerhub-cli.svg)](https://www.npmjs.com/package/swaggerhub-cli)

The SwaggerHub CLI enables teams to build automation and workflows around SwaggerHub. Teams can use it in places like their CI/CD pipeline to create new APIs, create and update API versions, and mark API versions as published and default among other features. Every team has their own workflow, and the SwaggerHub CLI can help teams build the workflow that fits their needs.

<!-- toc -->
* [Requirements](#requirements)
* [Installation](#installation)
* [Setup](#setup)
* [Additional configuration for SwaggerHub On-Premise](#additional-configuration-for-swaggerhub-on-premise)
* [Usage](#usage)
* [Commands](#commands)
* [Plugins](#plugins)
* [Contributing](#contributing)
<!-- tocstop -->
# Requirements
Node.js 20.x or later.

# Installation
```sh-session
$ npm i -g swaggerhub-cli
```
# Setup
The SwaggerHub CLI can be configured through environment variables or through the [`swaggerhub configure`](#swaggerhub-configure) command. The CLI will look for the following environment variables.

* `SWAGGERHUB_API_KEY` (required) – **Important: keep this key secure.** This is the SwaggerHub API key the CLI will use for authentication. You can find your API key on the [user settings page](https://app.swaggerhub.com/settings/apiKey) in SwaggerHub.
* `SWAGGERHUB_URL` (optional, default is `https://api.swaggerhub.com`) – Customers with on-premise installations need to point this to their on-premise API, which is `http(s)://{swaggerhub-host}/v1` (do not append a backslash).

Alernatively, you can use the `swaggerhub configure` command to create a configuration file for the CLI to use. This command will walk you through the steps to set up the necessary configurations.

```sh-session
$ swaggerhub configure
? SwaggerHub URL: https://api.swaggerhub.com
? API Key: <your-api-key>
```

Environment variables take precedence over the configuration file created by this command.

# Additional configuration for SwaggerHub On-Premise

If your SwaggerHub On-Premise instance uses a **self-signed or privately signed SSL certificate**, there are additional steps required to make the SwaggerHub CLI trust this certificate.

By default, Node.js rejects self-signed or privately signed SSL certificates because their root CA is not known. You will see an error like this in the CLI output:

```
FetchError: request to https://... failed, reason: self signed certificate
```

The solution is to use the [`NODE_EXTRA_CA_CERTS`](https://nodejs.org/api/cli.html#cli_node_extra_ca_certs_file) environment variable to specify custom trusted certificates for Node.js.

Start by creating a .pem file containing your custom trusted certificates in the PEM format.
* If the certificate is _self-signed_ (so that it is its own CA), include the certificate itself.
* If the certificate is _signed by a private CA_, include the CA root and any intermediate certificates, in any order. Blank lines are allowed, but optional, between individual certificates.

```
-----BEGIN CERTIFICATE-----
CA root certificate
-----END CERTIFICATE-----

-----BEGIN CERTIFICATE-----
Intermediate certificate 1
-----END CERTIFICATE-----

-----BEGIN CERTIFICATE-----
Intermediate certificate 2
-----END CERTIFICATE-----
```

Specify the path to this PEM file in the `NODE_EXTRA_CA_CERTS` environment variable.

macOS/*nix/bash examples:
```sh-session
export NODE_EXTRA_CA_CERTS=~/Work/extra-ca-certs.pem   # '~' means the home folder of the logged-in user

export NODE_EXTRA_CA_CERTS=$HOME/.ssh/extra-ca-certs.pem

export NODE_EXTRA_CA_CERTS=/Users/username/Work/extra-ca-certs.pem
```

Windows examples:
```
:: Both forward and backslashes are OK
set NODE_EXTRA_CA_CERTS=C:\Work\extra-ca-certs.pem
set NODE_EXTRA_CA_CERTS=C:/Work/extra-ca-certs.pem

:: You can also define the path itself using environment variables
set NODE_EXTRA_CA_CERTS=%USERPROFILE%\extra-ca-certs.pem
```

# Usage
```sh-session
$ swaggerhub COMMAND
running command...
$ swaggerhub (-v|--version|version)
swaggerhub/0.1.2 darwin-x64 node-v12.13.0
$ swaggerhub --help [COMMAND]
USAGE
  $ swaggerhub COMMAND
...
```
# Commands
<!-- commands -->
* [`swaggerhub api:create OWNER/API_NAME/[VERSION]`](#swaggerhub-apicreate)
* [`swaggerhub api:delete OWNER/API_NAME/[VERSION]`](#swaggerhub-apidelete)
* [`swaggerhub api:get OWNER/API_NAME/[VERSION]`](#swaggerhub-apiget)
* [`swaggerhub api:publish OWNER/API_NAME/VERSION`](#swaggerhub-apipublish)
* [`swaggerhub api:setdefault OWNER/API_NAME/VERSION`](#swaggerhub-apisetdefault)
* [`swaggerhub api:unpublish OWNER/API_NAME/VERSION`](#swaggerhub-apiunpublish)
* [`swaggerhub api:update OWNER/API_NAME/[VERSION]`](#swaggerhub-apiupdate)
* [`swaggerhub api:validate OWNER/API_NAME/[VERSION]`](#swaggerhub-apivalidate)
* [`swaggerhub api:validate:download-rules OWNER`](#swaggerhub-apivalidatedownload)
* [`swaggerhub api:validate:local`](#swaggerhub-apivalidatelocal)
* [`swaggerhub configure`](#swaggerhub-configure)
* [`swaggerhub domain:create OWNER/DOMAIN_NAME/[VERSION]`](#swaggerhub-domaincreate)
* [`swaggerhub domain:delete OWNER/DOMAIN_NAME/[VERSION]`](#swaggerhub-domaindelete)
* [`swaggerhub domain:get OWNER/DOMAIN_NAME/[VERSION]`](#swaggerhub-domainget)
* [`swaggerhub domain:publish OWNER/DOMAIN_NAME/VERSION`](#swaggerhub-domainpublish)
* [`swaggerhub domain:setdefault OWNER/DOMAIN_NAME/VERSION`](#swaggerhub-domainsetdefault)
* [`swaggerhub domain:unpublish OWNER/DOMAIN_NAME/VERSION`](#swaggerhub-domainunpublish)
* [`swaggerhub domain:update OWNER/DOMAIN_NAME/[VERSION]`](#swaggerhub-domainupdate)
* [`swaggerhub help [COMMAND]`](#swaggerhub-help)
* [`swaggerhub integration:create OWNER/API_NAME/[VERSION]`](#swaggerhub-integrationcreate)
* [`swaggerhub integration:delete OWNER/API_NAME/VERSION/INTEGRATION_ID`](#swaggerhub-integrationdelete)
* [`swaggerhub integration:execute OWNER/API_NAME/VERSION/INTEGRATION_ID`](#swaggerhub-integrationexecute)
* [`swaggerhub integration:get OWNER/API_NAME/VERSION/INTEGRATION_ID`](#swaggerhub-integrationget)
* [`swaggerhub integration:list OWNER/API_NAME/[VERSION]`](#swaggerhub-integrationlist)
* [`swaggerhub integration:update OWNER/API_NAME/VERSION/INTEGRATION_ID`](#swaggerhub-integrationupdate)
* [`swaggerhub plugins`](#swaggerhub-plugins)
* [`swaggerhub plugins:add PLUGIN`](#swaggerhub-pluginsadd)
* [`swaggerhub plugins:inspect PLUGIN...`](#swaggerhub-pluginsinspect)
* [`swaggerhub plugins:install PLUGIN`](#swaggerhub-pluginsinstall)
* [`swaggerhub plugins:link PATH`](#swaggerhub-pluginslink)
* [`swaggerhub plugins:remove [PLUGIN]`](#swaggerhub-pluginsremove)
* [`swaggerhub plugins:reset`](#swaggerhub-pluginsreset)
* [`swaggerhub plugins:uninstall [PLUGIN]`](#swaggerhub-pluginsuninstall)
* [`swaggerhub plugins:unlink [PLUGIN]`](#swaggerhub-pluginsunlink)
* [`swaggerhub plugins:update`](#swaggerhub-pluginsupdate)
* [`swaggerhub project:api:add OWNER/PROJECT_NAME API`](#swaggerhub-projectapiadd)
* [`swaggerhub project:api:remove OWNER/PROJECT_NAME API`](#swaggerhub-projectapiremove)
* [`swaggerhub project:create OWNER/PROJECT_NAME`](#swaggerhub-projectcreate)
* [`swaggerhub project:delete OWNER/PROJECT_NAME`](#swaggerhub-projectdelete)
* [`swaggerhub project:domain:add OWNER/PROJECT_NAME DOMAIN`](#swaggerhub-projectdomainadd)
* [`swaggerhub project:domain:remove OWNER/PROJECT_NAME DOMAIN`](#swaggerhub-projectdomainremove)
* [`swaggerhub project:get OWNER/PROJECT_NAME`](#swaggerhub-projectget)
* [`swaggerhub project:list [OWNER]`](#swaggerhub-projectlist)
* [`swaggerhub project:member:list OWNER/PROJECT_NAME`](#swaggerhub-projectmemberlist)
* [`swaggerhub spectral:upload OWNER/RULESET_NAME directory`](#swaggerhub-spectralupload)
* [`swaggerhub spectral:download OWNER/RULESET_NAME directory`](#swaggerhub-spectraldownload)

## `swaggerhub api:create`

creates a new API / API version from a YAML/JSON file

```
USAGE
  $ swaggerhub api:create OWNER/API_NAME/[VERSION] [--visibility public|private] (--published publish|unpublish
    -f <value>) [--setdefault ] [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to create on SwaggerHub

FLAGS
  -f, --file=<value>         (required) file location of API to create
  -h, --help                 Show CLI help.
      --published=<option>   [default: unpublish] sets the lifecycle setting of the API version
                             <options: publish|unpublish>
      --setdefault           sets API version to be the default
      --visibility=<option>  [default: private] visibility of API in SwaggerHub
                             <options: public|private>

DESCRIPTION
  creates a new API / API version from a YAML/JSON file
  The API version from the file will be used unless the version is specified in the command argument.
  An error will occur if the API version already exists.


EXAMPLES
  $ swaggerhub api:create organization/api/1.0.0 --file api.yaml --visibility public

  $ swaggerhub api:create organization/api --file api.yaml

  $ swaggerhub api:create organization/api/1.0.0 --published=publish --file api.json

  $ swaggerhub api:create organization/api/1.0.0 --setdefault --file api.json

  $ swaggerhub api:create organization/api/1.0.0 --published=publish --setdefault --file api.json
```

_See code: [src/commands/api/create.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/create.js)_

## `swaggerhub api:delete`

delete an API or API version

```
USAGE
  $ swaggerhub api:delete OWNER/API_NAME/[VERSION] [-f] [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to delete on SwaggerHub

FLAGS
  -f, --force  delete API without prompting for confirmation
  -h, --help   Show CLI help.

DESCRIPTION
  delete an API or API version


EXAMPLES
  $ swaggerhub api:delete organization/api/1.0.0

  $ swaggerhub api:delete organization/api

  $ swaggerhub api:delete organization/api --force
```

_See code: [src/commands/api/delete.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/delete.js)_

## `swaggerhub api:get`

fetches an API definition

```
USAGE
  $ swaggerhub api:get OWNER/API_NAME/[VERSION] [-j] [-r] [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to fetch from Swaggerhub

FLAGS
  -h, --help      Show CLI help.
  -j, --json      returns the API in JSON format.
  -r, --resolved  gets the resolved API definition (supported in v1.25+).

DESCRIPTION
  fetches an API definition
  When VERSION is not included in the argument, the default version will be returned.
  Returns the API in YAML format by default.


EXAMPLES
  $ swaggerhub api:get organization/api

  $ swaggerhub api:get organization/api/1.0.0 --json
```

_See code: [src/commands/api/get.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/get.js)_

## `swaggerhub api:publish`

publish an API version

```
USAGE
  $ swaggerhub api:publish OWNER/API_NAME/VERSION [-f] [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION  API to publish on Swaggerhub

FLAGS
  -f, --force  publish API without prompting for confirmation
  -h, --help   Show CLI help.

DESCRIPTION
  publish an API version

EXAMPLES
  $ swaggerhub api:publish organization/api/1.0.0

  $ swaggerhub api:publish organization/api/1.0.0 --force
```

_See code: [src/commands/api/publish.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/publish.js)_

## `swaggerhub api:setdefault`

set the default version of an API

```
USAGE
  $ swaggerhub api:setdefault OWNER/API_NAME/VERSION [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION  API to set as default on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  set the default version of an API

EXAMPLES
  $ swaggerhub api:setdefault organization/api/2.0.0
```

_See code: [src/commands/api/setdefault.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/setdefault.js)_

## `swaggerhub api:unpublish`

unpublish an API version

```
USAGE
  $ swaggerhub api:unpublish OWNER/API_NAME/VERSION [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION  API identifier

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  unpublish an API version

EXAMPLES
  $ swaggerhub api:unpublish organization/api/1.0.0
```

_See code: [src/commands/api/unpublish.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/unpublish.js)_

## `swaggerhub api:update`

update an API

```
USAGE
  $ swaggerhub api:update OWNER/API_NAME/[VERSION] [-f <value>] [--visibility public|private] [--published
    publish|unpublish] [--setdefault] [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to update on SwaggerHub

FLAGS
  -f, --file=<value>         file location of API to update
  -h, --help                 Show CLI help.
      --published=<option>   sets the lifecycle setting of the API version
                             <options: publish|unpublish>
      --setdefault           sets API version to be the default
      --visibility=<option>  visibility of API in SwaggerHub
                             <options: public|private>

DESCRIPTION
  update an API
  The API version from the file will be used unless the version is specified in the command argument.
  When no file is specified then the default API version will be updated.
  The API visibility can be changed by using visibility flag.


EXAMPLES
  $ swaggerhub api:update organization/api --file api.yaml

  $ swaggerhub api:update organization/api/1.0.0 --file api.json

  $ swaggerhub api:update organization/api/1.0.0 --published=publish --file api.json

  $ swaggerhub api:update organization/api/1.0.0 --setdefault --file api.json

  $ swaggerhub api:update organization/api/1.0.0 --published=unpublish --setdefault --file api.json

  $ swaggerhub api:update organization/api/1.0.0 --visibility=private
```

_See code: [src/commands/api/update.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/update.js)_

## `swaggerhub api:validate`

Get validation result for an API version

```
USAGE
  $ swaggerhub api:validate OWNER/API_NAME/[VERSION] [-c] [-j] [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to fetch validation errors for from Swaggerhub

FLAGS
  -c, --fail-on-critical  Exit with error code 1 if there are critical standardization errors present
  -h, --help              Show CLI help.
  -j, --json              Print output in JSON instead of table format

DESCRIPTION
  Get validation result for an API version
  When VERSION is not included in the argument, the default version will be validated.
  An error will occur if the API version does not exist.
  If the flag `-c` or `--failOnCritical` is used and there are standardization
  errors with `Critical` severity present, the command will exit with error code `1`.


EXAMPLES
  $ swaggerhub api:validate organization/api/1.0.0

  $ swaggerhub api:validate -c -j organization/api/1.0.0

  $ swaggerhub api:validate --fail-on-critical --json organization/api
```

_See code: [src/commands/api/validate/index.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/validate/index.js)_

## `swaggerhub api:validate:download-rules`

Get existing SwaggerHub's organization standardization ruleset.

```
USAGE
  $ swaggerhub api:validate:download-rules OWNER [-s] [-d] [-h]

ARGUMENTS
  OWNER  Which organization standardization rules to fetch from SwaggerHub

FLAGS
  -d, --include-disabled-rules  Includes disabled rules in fetched organization's ruleset
  -h, --help                    Show CLI help.
  -s, --include-system-rules    Includes system rules in fetched organization's ruleset

DESCRIPTION
  Get existing SwaggerHub's organization standardization ruleset.
  Requires organization name argument. An error will occur if provided organization doesn't exist
  or your account is not permitted to access that organization's settings.
  If the flag `-s` or `--include-system-rules` is used, the returned ruleset will also include SwaggerHub system rules.
  If the flag `-d` or `--include-disabled-rules` is used, the returned ruleset will also include disabled custom rules

EXAMPLES
  $ swaggerhub api:validate:download-rules myOrg -s

  $ swaggerhub api:validate:download-rules myOrg --include-disabled-rules -s
```

_See code: [src/commands/api/validate/download-rules.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/validate/download-rules.js)_

## `swaggerhub api:validate:local`

Runs a scan against a local API definition using the organization's standardization configuration on SwaggerHub.

```
USAGE
  $ swaggerhub api:validate:local -f <value> -o <value> [-c] [-j] [-h]

FLAGS
  -c, --fail-on-critical      Exit with error code 1 if there are critical standardization errors present
  -f, --file=<value>          (required) Path of API definition file to run scan against
  -h, --help                  Show CLI help.
  -j, --json                  Print output in JSON instead of table format
  -o, --organization=<value>  (required) Which organization's standardization settings to use for linting the target
                              definition

DESCRIPTION
  Runs a scan against a local API definition using the organization's standardization configuration on SwaggerHub.
  If the flag `-c` or `--failOnCritical` is used and there are standardization
  errors with `Critical` severity present, the command will exit with error code `1`.


EXAMPLES
  $ swaggerhub api:validate:local -o myOrg -f ./my-api.yaml -c -j 

  $ swaggerhub api:validate:local --organization myOrg --file ./my-api/json --fail-on-critical --json
```

_See code: [src/commands/api/validate/local.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/api/validate/local.js)_

## `swaggerhub configure`

configure application settings

```
USAGE
  $ swaggerhub configure [-h]

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  configure application settings
  Enter the SwaggerHub URL - default is https://api.swaggerhub.com
  Customers with on-premise installations need to point this to their on-premise API, which is
  http(s)://{swaggerhub-host}/v1
  Enter the API Key - this can be retrieved from https://app.swaggerhub.com/settings/apiKey
  You can set these as environment variables: SWAGGERHUB_URL, SWAGGERHUB_API_KEY. These take priority over config
  settings.
```

_See code: [src/commands/configure.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/configure.js)_

## `swaggerhub domain:create`

creates a new domain / domain version from a YAML/JSON file

```
USAGE
  $ swaggerhub domain:create OWNER/DOMAIN_NAME/[VERSION] [--visibility public|private] (--published
    publish|unpublish -f <value>) [--setdefault ] [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/[VERSION]  Domain to create on SwaggerHub

FLAGS
  -f, --file=<value>         (required) file location of domain to create
  -h, --help                 Show CLI help.
      --published=<option>   [default: unpublish] sets the lifecycle setting of the domain version
                             <options: publish|unpublish>
      --setdefault           sets domain version to be the default
      --visibility=<option>  [default: private] visibility of domain in SwaggerHub
                             <options: public|private>

DESCRIPTION
  creates a new domain / domain version from a YAML/JSON file
  The domain version from the file will be used unless the version is specified in the command argument.
  An error will occur if the domain version already exists.


EXAMPLES
  $ swaggerhub domain:create organization/domain/1.0.0 --file domain.yaml --visibility public

  $ swaggerhub domain:create organization/domain --file domain.yaml

  $ swaggerhub domain:create organization/domain/1.0.0 --publish --file domain.json

  $ swaggerhub domain:create organization/domain/1.0.0 --setdefault --file domain.json

  $ swaggerhub domain:create organization/domain/1.0.0 --publish --setdefault --file domain.json
```

_See code: [src/commands/domain/create.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/create.js)_

## `swaggerhub domain:delete`

delete a domain or domain version

```
USAGE
  $ swaggerhub domain:delete OWNER/DOMAIN_NAME/[VERSION] [-f] [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/[VERSION]  Domain to delete on SwaggerHub

FLAGS
  -f, --force  delete domain without prompting for confirmation
  -h, --help   Show CLI help.

DESCRIPTION
  delete a domain or domain version


EXAMPLES
  $ swaggerhub domain:delete organization/domain/1.0.0

  $ swaggerhub domain:delete organization/domain

  $ swaggerhub domain:delete organization/domain --force
```

_See code: [src/commands/domain/delete.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/delete.js)_

## `swaggerhub domain:get`

fetches a domain definition

```
USAGE
  $ swaggerhub domain:get OWNER/DOMAIN_NAME/[VERSION] [-j] [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/[VERSION]  Domain to fetch from SwaggerHub

FLAGS
  -h, --help  Show CLI help.
  -j, --json  returns the domain in JSON format.

DESCRIPTION
  fetches a domain definition
  When VERSION is not included in the argument, the default version will be returned.
  Returns the domain in YAML format by default.


EXAMPLES
  $ swaggerhub domain:get organization/domain

  $ swaggerhub domain:get organization/domain/1.0.0 --json
```

_See code: [src/commands/domain/get.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/get.js)_

## `swaggerhub domain:publish`

publish a domain version

```
USAGE
  $ swaggerhub domain:publish OWNER/DOMAIN_NAME/VERSION [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/VERSION  Domain to publish on SwaggerHub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  publish a domain version

EXAMPLES
  $ swaggerhub domain:publish organization/domain/1.0.0
```

_See code: [src/commands/domain/publish.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/publish.js)_

## `swaggerhub domain:setdefault`

set the default version of a domain

```
USAGE
  $ swaggerhub domain:setdefault OWNER/DOMAIN_NAME/VERSION [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/VERSION  Domain to set as default on SwaggerHub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  set the default version of a domain

EXAMPLES
  $ swaggerhub domain:setdefault organization/domain/2.0.0
```

_See code: [src/commands/domain/setdefault.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/setdefault.js)_

## `swaggerhub domain:unpublish`

unpublish a domain version

```
USAGE
  $ swaggerhub domain:unpublish OWNER/DOMAIN_NAME/VERSION [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/VERSION  Domain to unpublish on SwaggerHub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  unpublish a domain version

EXAMPLES
  $ swaggerhub domain:unpublish organization/domain/1.0.0
```

_See code: [src/commands/domain/unpublish.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/unpublish.js)_

## `swaggerhub domain:update`

update a domain

```
USAGE
  $ swaggerhub domain:update OWNER/DOMAIN_NAME/[VERSION] [-f <value>] [--visibility public|private] [--published
    publish|unpublish] [--setdefault] [-h]

ARGUMENTS
  OWNER/DOMAIN_NAME/[VERSION]  Domain to update on SwaggerHub

FLAGS
  -f, --file=<value>         file location of domain to update
  -h, --help                 Show CLI help.
      --published=<option>   sets the lifecycle setting of the domain version
                             <options: publish|unpublish>
      --setdefault           sets domain version to be the default
      --visibility=<option>  visibility of domain in SwaggerHub
                             <options: public|private>

DESCRIPTION
  update a domain
  The domain version from the file will be used unless the version is specified in the command argument.
  When no file is specified then the default domain version will be updated.
  The domain visibility can be changed by using visibility flag.


EXAMPLES
  $ swaggerhub domain:update organization/domain --file domain.yaml

  $ swaggerhub domain:update organization/domain/1.0.0 --file domain.json

  $ swaggerhub domain:update organization/domain/1.0.0 --published=publish --file domain.json

  $ swaggerhub domain:update organization/domain/1.0.0 --setdefault --file domain.json

  $ swaggerhub domain:update organization/domain/1.0.0 --published=unpublish --setdefault --file domain.json

  $ swaggerhub domain:update organization/domain/1.0.0 --visibility=private
```

_See code: [src/commands/domain/update.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/domain/update.js)_

## `swaggerhub help`

Display help for swaggerhub.

```
USAGE
  $ swaggerhub help [COMMAND...] [-n]

ARGUMENTS
  COMMAND...  Command to show help for.

FLAGS
  -n, --nested-commands  Include all nested commands in the output.

DESCRIPTION
  Display help for swaggerhub.
```

_See code: [@oclif/plugin-help](https://github.com/oclif/plugin-help/blob/v6.0.22/src/commands/help.ts)_

## `swaggerhub integration:create`

creates a new API integration from a JSON configuration file.

```
USAGE
  $ swaggerhub integration:create OWNER/API_NAME/[VERSION] -f <value> [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to add integration to on SwaggerHub

FLAGS
  -f, --file=<value>  (required) location of integration configuration file
  -h, --help          Show CLI help.

DESCRIPTION
  creates a new API integration from a JSON configuration file.
  See the documentation for configuration files:
  https://github.com/SmartBear/swaggerhub-cli/tree/main/examples/integrations
  When VERSION is not included in the argument, the integration will be added to be default API version.


EXAMPLES
  $ swaggerhub integration:create organization/api/1.0.0 --file config.json
```

_See code: [src/commands/integration/create.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/integration/create.js)_

## `swaggerhub integration:delete`

deletes the integration from the given API.

```
USAGE
  $ swaggerhub integration:delete OWNER/API_NAME/VERSION/INTEGRATION_ID [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION/INTEGRATION_ID  Integration to delete for given API on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  deletes the integration from the given API.

EXAMPLES
  $ swaggerhub integration:delete organization/api/1.0.0/503c2db6-448a-4678-a310-f465429e9704
```

_See code: [src/commands/integration/delete.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/integration/delete.js)_

## `swaggerhub integration:execute`

executes an integration for the given API.

```
USAGE
  $ swaggerhub integration:execute OWNER/API_NAME/VERSION/INTEGRATION_ID [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION/INTEGRATION_ID  Integration to execute for given API on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  executes an integration for the given API.

EXAMPLES
  $ swaggerhub integration:execute organization/api/1.0.0/503c2db6-448a-4678-a310-f465429e9704
```

_See code: [src/commands/integration/execute.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/integration/execute.js)_

## `swaggerhub integration:get`

retieves an integration for the given API.

```
USAGE
  $ swaggerhub integration:get OWNER/API_NAME/VERSION/INTEGRATION_ID [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION/INTEGRATION_ID  Integration to fetch for given API on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  retieves an integration for the given API.

EXAMPLES
  $ swaggerhub integration:get organization/api/1.0.0/503c2db6-448a-4678-a310-f465429e9704
```

_See code: [src/commands/integration/get.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/integration/get.js)_

## `swaggerhub integration:list`

list integrations on an API.

```
USAGE
  $ swaggerhub integration:list OWNER/API_NAME/[VERSION] [-h]

ARGUMENTS
  OWNER/API_NAME/[VERSION]  API to list integrations for on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  list integrations on an API.

EXAMPLES
  $ swaggerhub integration:list organization/api/1.0.0
```

_See code: [src/commands/integration/list.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/integration/list.js)_

## `swaggerhub integration:update`

update the configuration of an API integration.

```
USAGE
  $ swaggerhub integration:update OWNER/API_NAME/VERSION/INTEGRATION_ID -f <value> [-h]

ARGUMENTS
  OWNER/API_NAME/VERSION/INTEGRATION_ID  Integration to update for given API on Swaggerhub

FLAGS
  -f, --file=<value>  (required) location of integration configuration file
  -h, --help          Show CLI help.

DESCRIPTION
  update the configuration of an API integration.

EXAMPLES
  $ swaggerhub integration:update organization/api/1.0.0/503c2db6-448a-4678-abcd-0123456789abc --file config.json
```

_See code: [src/commands/integration/update.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/integration/update.js)_

## `swaggerhub plugins`

List installed plugins.

```
USAGE
  $ swaggerhub plugins [--json] [--core]

FLAGS
  --core  Show core plugins.

GLOBAL FLAGS
  --json  Format output as json.

DESCRIPTION
  List installed plugins.

EXAMPLES
  $ swaggerhub plugins
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/index.ts)_

## `swaggerhub plugins:add`

Installs a plugin into swaggerhub.

```
USAGE
  $ swaggerhub plugins:add PLUGIN... [--json] [-f] [-h] [-s | -v]

ARGUMENTS
  PLUGIN...  Plugin to install.

FLAGS
  -f, --force    Force npm to fetch remote resources even if a local copy exists on disk.
  -h, --help     Show CLI help.
  -s, --silent   Silences npm output.
  -v, --verbose  Show verbose npm output.

GLOBAL FLAGS
  --json  Format output as json.

DESCRIPTION
  Installs a plugin into swaggerhub.

  Uses npm to install plugins.

  Installation of a user-installed plugin will override a core plugin.

  Use the SWAGGERHUB_NPM_LOG_LEVEL environment variable to set the npm loglevel.
  Use the SWAGGERHUB_NPM_REGISTRY environment variable to set the npm registry.

ALIASES
  $ swaggerhub plugins:add

EXAMPLES
  Install a plugin from npm registry.

    $ swaggerhub plugins:add myplugin

  Install a plugin from a github url.

    $ swaggerhub plugins:add https://github.com/someuser/someplugin

  Install a plugin from a github slug.

    $ swaggerhub plugins:add someuser/someplugin
```

## `swaggerhub plugins:inspect`

Displays installation properties of a plugin.

```
USAGE
  $ swaggerhub plugins:inspect PLUGIN...

ARGUMENTS
  PLUGIN...  [default: .] Plugin to inspect.

FLAGS
  -h, --help     Show CLI help.
  -v, --verbose

GLOBAL FLAGS
  --json  Format output as json.

DESCRIPTION
  Displays installation properties of a plugin.

EXAMPLES
  $ swaggerhub plugins:inspect myplugin
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/inspect.ts)_

## `swaggerhub plugins:install`

Installs a plugin into swaggerhub.

```
USAGE
  $ swaggerhub plugins:install PLUGIN... [--json] [-f] [-h] [-s | -v]

ARGUMENTS
  PLUGIN...  Plugin to install.

FLAGS
  -f, --force    Force npm to fetch remote resources even if a local copy exists on disk.
  -h, --help     Show CLI help.
  -s, --silent   Silences npm output.
  -v, --verbose  Show verbose npm output.

GLOBAL FLAGS
  --json  Format output as json.

DESCRIPTION
  Installs a plugin into swaggerhub.

  Uses npm to install plugins.

  Installation of a user-installed plugin will override a core plugin.

  Use the SWAGGERHUB_NPM_LOG_LEVEL environment variable to set the npm loglevel.
  Use the SWAGGERHUB_NPM_REGISTRY environment variable to set the npm registry.

ALIASES
  $ swaggerhub plugins:add

EXAMPLES
  Install a plugin from npm registry.

    $ swaggerhub plugins:install myplugin

  Install a plugin from a github url.

    $ swaggerhub plugins:install https://github.com/someuser/someplugin

  Install a plugin from a github slug.

    $ swaggerhub plugins:install someuser/someplugin
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/install.ts)_

## `swaggerhub plugins:link`

Links a plugin into the CLI for development.

```
USAGE
  $ swaggerhub plugins:link PATH [-h] [--install] [-v]

ARGUMENTS
  PATH  [default: .] path to plugin

FLAGS
  -h, --help          Show CLI help.
  -v, --verbose
      --[no-]install  Install dependencies after linking the plugin.

DESCRIPTION
  Links a plugin into the CLI for development.

  Installation of a linked plugin will override a user-installed or core plugin.

  e.g. If you have a user-installed or core plugin that has a 'hello' command, installing a linked plugin with a 'hello'
  command will override the user-installed or core plugin implementation. This is useful for development work.


EXAMPLES
  $ swaggerhub plugins:link myplugin
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/link.ts)_

## `swaggerhub plugins:remove`

Removes a plugin from the CLI.

```
USAGE
  $ swaggerhub plugins:remove [PLUGIN...] [-h] [-v]

ARGUMENTS
  PLUGIN...  plugin to uninstall

FLAGS
  -h, --help     Show CLI help.
  -v, --verbose

DESCRIPTION
  Removes a plugin from the CLI.

ALIASES
  $ swaggerhub plugins:unlink
  $ swaggerhub plugins:remove

EXAMPLES
  $ swaggerhub plugins:remove myplugin
```

## `swaggerhub plugins:reset`

Remove all user-installed and linked plugins.

```
USAGE
  $ swaggerhub plugins:reset [--hard] [--reinstall]

FLAGS
  --hard       Delete node_modules and package manager related files in addition to uninstalling plugins.
  --reinstall  Reinstall all plugins after uninstalling.
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/reset.ts)_

## `swaggerhub plugins:uninstall`

Removes a plugin from the CLI.

```
USAGE
  $ swaggerhub plugins:uninstall [PLUGIN...] [-h] [-v]

ARGUMENTS
  PLUGIN...  plugin to uninstall

FLAGS
  -h, --help     Show CLI help.
  -v, --verbose

DESCRIPTION
  Removes a plugin from the CLI.

ALIASES
  $ swaggerhub plugins:unlink
  $ swaggerhub plugins:remove

EXAMPLES
  $ swaggerhub plugins:uninstall myplugin
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/uninstall.ts)_

## `swaggerhub plugins:unlink`

Removes a plugin from the CLI.

```
USAGE
  $ swaggerhub plugins:unlink [PLUGIN...] [-h] [-v]

ARGUMENTS
  PLUGIN...  plugin to uninstall

FLAGS
  -h, --help     Show CLI help.
  -v, --verbose

DESCRIPTION
  Removes a plugin from the CLI.

ALIASES
  $ swaggerhub plugins:unlink
  $ swaggerhub plugins:remove

EXAMPLES
  $ swaggerhub plugins:unlink myplugin
```

## `swaggerhub plugins:update`

Update installed plugins.

```
USAGE
  $ swaggerhub plugins:update [-h] [-v]

FLAGS
  -h, --help     Show CLI help.
  -v, --verbose

DESCRIPTION
  Update installed plugins.
```

_See code: [@oclif/plugin-plugins](https://github.com/oclif/plugin-plugins/blob/v5.4.25/src/commands/plugins/update.ts)_

## `swaggerhub project:api:add`

Adds an API to an existing project.

```
USAGE
  $ swaggerhub project:api:add OWNER/PROJECT_NAME API [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The project to add the API to on Swaggerhub
  API                 The name of the API on Swaggerhub to add

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Adds an API to an existing project.

EXAMPLES
  $ swaggerhub project:api:add organization/project_name my_api
```

_See code: [src/commands/project/api/add.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/api/add.js)_

## `swaggerhub project:api:remove`

Removes an API from a project in SwaggerHub.

```
USAGE
  $ swaggerhub project:api:remove OWNER/PROJECT_NAME API [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The project to remove the API from on Swaggerhub
  API                 The name of the API on Swaggerhub to remove

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Removes an API from a project in SwaggerHub.

EXAMPLES
  $ swaggerhub project:api:remove organization/project_name my_api
```

_See code: [src/commands/project/api/remove.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/api/remove.js)_

## `swaggerhub project:create`

Creates a new project in SwaggerHub.

```
USAGE
  $ swaggerhub project:create OWNER/PROJECT_NAME [--description <value>] [-a <value>] [-d <value>] [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The new project to create on Swaggerhub

FLAGS
  -a, --apis=<value>         Comma separated list of api names to include in project
  -d, --domains=<value>      Comma separated list of domain names to include in project
  -h, --help                 Show CLI help.
      --description=<value>  Description of project

DESCRIPTION
  Creates a new project in SwaggerHub.

EXAMPLES
  $ swaggerhub project:create organization/new_project_name --description "project description"

  $ swaggerhub project:create organization/new_project_name -a "testapi1,testapi2"

  $ swaggerhub project:create organization/new_project_name --apis "testapi1,testapi2"

  $ swaggerhub project:create organization/new_project_name -d "testdomain3,testdomain4"

  $ swaggerhub project:create organization/new_project_name --domains "testdomain3,testdomain4"

  $ swaggerhub project:create organization/new_project_name -a "testapi1" -d "testdomain3" --description "description"
```

_See code: [src/commands/project/create.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/create.js)_

## `swaggerhub project:delete`

Deletes a project from SwaggerHub.

```
USAGE
  $ swaggerhub project:delete OWNER/PROJECT_NAME [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The project to delete on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Deletes a project from SwaggerHub.

EXAMPLES
  $ swaggerhub project:delete organization/project_name
```

_See code: [src/commands/project/delete.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/delete.js)_

## `swaggerhub project:domain:add`

Adds a domain to an existing project.

```
USAGE
  $ swaggerhub project:domain:add OWNER/PROJECT_NAME DOMAIN [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The project to add the domain to on Swaggerhub
  DOMAIN              The name of the domain on Swaggerhub to add

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Adds a domain to an existing project.

EXAMPLES
  $ swaggerhub project:domain:add organization/project_name my_domain
```

_See code: [src/commands/project/domain/add.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/domain/add.js)_

## `swaggerhub project:domain:remove`

Removes a domain from a project in SwaggerHub.

```
USAGE
  $ swaggerhub project:domain:remove OWNER/PROJECT_NAME DOMAIN [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The project to remove the domain from on Swaggerhub
  DOMAIN              The name of the domain on Swaggerhub to remove

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Removes a domain from a project in SwaggerHub.

EXAMPLES
  $ swaggerhub project:domain:remove organization/project_name my_domain
```

_See code: [src/commands/project/domain/remove.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/domain/remove.js)_

## `swaggerhub project:get`

Retrieves the details for a project.

```
USAGE
  $ swaggerhub project:get OWNER/PROJECT_NAME [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  The project to get details for on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Retrieves the details for a project.

EXAMPLES
  $ swaggerhub project:get organization/project_name
```

_See code: [src/commands/project/get.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/get.js)_

## `swaggerhub project:list`

list projects

```
USAGE
  $ swaggerhub project:list [OWNER] [-h]

ARGUMENTS
  OWNER  The organization to list projects for on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  list projects

EXAMPLES
  $ swaggerhub project:list

  $ swaggerhub project:list organization
```

_See code: [src/commands/project/list.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/list.js)_

## `swaggerhub project:member:list`

list members of a project

```
USAGE
  $ swaggerhub project:member:list OWNER/PROJECT_NAME [-h]

ARGUMENTS
  OWNER/PROJECT_NAME  Project to list members of on Swaggerhub

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  list members of a project

EXAMPLES
  $ swaggerhub project:member:list organisation/project_name
```

_See code: [src/commands/project/member/list.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/project/member/list.js)_

## `swaggerhub spectral:upload`

Create or update organization's Spectral ruleset

```
USAGE
  $ swaggerhub spectral:upload OWNER/RULESET_NAME directory [-h]

ARGUMENTS
  OWNER/RULESET_NAME            The Spectral ruleset details for SwaggerHub organization
  directory                     Relative path to directory with ruleset files

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Create or update organization's Spectral ruleset

EXAMPLES
  $ swaggerhub spectral:upload my_organization/my_api_ruleset rules
```

_See code: [src/commands/spectral/upload.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/spectral/upload.js)_


## `swaggerhub spectral:download`

Fetch organization's Spectral ruleset

```
USAGE
  $ swaggerhub spectral:download OWNER/RULESET_NAME directory [-h]

ARGUMENTS
  OWNER/RULESET_NAME            The Spectral ruleset details for SwaggerHub organization
  directory                     Relative path to directory the ruleset files should be saved to

FLAGS
  -h, --help  Show CLI help.

DESCRIPTION
  Fetch organization's Spectral ruleset

EXAMPLES
  $ swaggerhub spectral:download my_organization/my_api_ruleset rules
```

_See code: [src/commands/spectral/download.js](https://github.com/SmartBear/swaggerhub-cli/blob/v0.9.1/src/commands/spectral/download.js)_
<!-- commandsstop -->

# Plugins

The SwaggerHub CLI supports plugins via the [oclif plugin](https://oclif.io/docs/plugins) infrastructure.

To install a plugin

```sh-session
$ swaggerhub plugins:install <github-url>
```

To list other options related to plugins

```sh-session
$ swaggerhub plugins --help
```

An example plugin used for fetching popular [JSON Schema](https://json-schema.org/) files, can be found here:  [https://github.com/ponelat/swaggerhub-cli-plugin-schema](https://github.com/ponelat/swaggerhub-cli-plugin-schema)

Example usage

```sh-session
$ swaggerhub plugins:install https://github.com/ponelat/swaggerhub-cli-plugin-schema
$ swaggerhub schema:list
  angular-cli-json
  ansible
  apple-app-site-association
  appsscript-json
  #...
$ swaggerhub schema:get ansible
  {
    "description": "Auto-Generated JSON Schema for Ansible-stable 2.9 (https://github.com/shaded-enmity/ansible-schema-generator)",
    "title": "Ansible 2.9",
    "$schema": "http://json-schema.org/draft-04/schema#",
    "type": "array",
  # ...
```

# Contributing
<!-- contributing -->
The SwaggerHub CLI is currently in an active development phase—we will not be accepting Pull Requests at this time. If you’ve found any bugs or typos, or have a feature requests or general feedback you’d like to share, please open an [issue](https://github.com/SmartBear/swaggerhub-cli/issues) and let us know.
<!-- contributingstop -->
 readmeEtag: '"5259ac3d47763307d68b1dd2ebadf2b873a5c454"' readmeLastModified: Thu, 25 Sep 2025 11:18:33 GMT repositoryId: 261799480 description: SwaggerHub CLI created: '2020-05-06T15:21:34Z' updated: '2026-01-23T14:15:57Z' language: JavaScript archived: false stars: 54 watchers: 31 forks: 16 owner: SmartBear logo: https://avatars.githubusercontent.com/u/1644671?v=4 license: Apache-2.0 repoEtag: '"871c2631d0c1f262ff80c0b30cfc7d2341e04ceb820af2a2a39305e50b6ec331"' repoLastModified: Fri, 23 Jan 2026 14:15:57 GMT foundInMaster: true category: Code Generators id: 4921a5a6c94697cd438f25deccf95ab6 - source: openapi3 tags repository: https://github.com/mermade/openapi-lint-vscode v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWxpbnQgUkVBRE1FCgohW1Zpc3VhbCBTdHVkaW8gTWFya2V0cGxhY2UgRG93bmxvYWRzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3Zpc3VhbC1zdHVkaW8tbWFya2V0cGxhY2UvZC9tZXJtYWRlLm9wZW5hcGktbGludCkgIVtWaXN1YWwgU3R1ZGlvIE1hcmtldHBsYWNlIFN0YXJzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3Zpc3VhbC1zdHVkaW8tbWFya2V0cGxhY2Uvc3RhcnMvbWVybWFkZS5vcGVuYXBpLWxpbnQpCgpUaGlzIGV4dGVuc2lvbiBjYW4gYmUgdXNlZCB0byB2YWxpZGF0ZSBhbmQgW2xpbnRdKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0xpbnRfKHNvZnR3YXJlKSkgT3BlbkFQSSAzLjAueCBkb2N1bWVudHMsIGFuZCBjb252ZXJ0IGJldHdlZW4gT3BlbkFQSSAyLjAgYW5kIDMuMC4wLiBJdCBwcm92aWRlcyAxMSBWaXN1YWwgU3R1ZGlvIENvZGUgY29tbWFuZHM6CgoqIE9wZW5BUEkgVmFsaWRhdGUgLSB3aGljaCBmdWxseSB2YWxpZGF0ZXMgeW91ciBPcGVuQVBJIGRvY3VtZW50IGFnYWluc3QgdGhlIFtzcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMi5tZCkKKiBPcGVuQVBJIExpbnQgLSB3aGljaCBhZGRpdGlvbmFsbHkgYXBwbGllcyBhIGxpZ2h0bHktb3BpbmlvbmF0ZWQgc2V0IG9mIGJlc3QtcHJhY3RpY2VzCiogT3BlbkFQSSBSZXNvbHZlIGFuZCBWYWxpZGF0ZSAtIHdoaWNoIHB1bGxzIGluIGV4dGVybmFsIGAkcmVmYHMgYmVmb3JlIHZhbGlkYXRpbmcKKiBPcGVuQVBJIFJlc29sdmUgYW5kIExpbnQgLSB3aGljaCBwdWxscyBpbiBleHRlcm5hbCBgJHJlZmBzIGJlZm9yZSBsaW50aW5nCiogT3BlbkFQSSBSZXNvbHZlIChCdW5kbGUpIC0gd2hpY2ggcmVzb2x2ZXMgZXh0ZXJuYWwgYCRyZWZgcyB0byBhIG5ldyB3aW5kb3cKKiBPcGVuQVBJIDIuMCB0byAzLjAuMCAoWUFNTCkgLSB3aGljaCBjb252ZXJ0cyB0aGUgY3VycmVudCBPcGVuQVBJIDIuMCBkb2N1bWVudCB0byAzLjAuMCBpbiBZQU1MCiogT3BlbkFQSSAyLjAgdG8gMy4wLjAgKEpTT04pIC0gd2hpY2ggY29udmVydHMgdGhlIGN1cnJlbnQgT3BlbkFQSSAyLjAgZG9jdW1lbnQgdG8gMy4wLjAgaW4gSlNPTgoqIE9wZW5BUEkgMi4wIHRvIDMuMC4wIChZQU1ML1Jlc29sdmVkKSAtIGxpa2UgdGhlIGFib3ZlIGJ1dCByZXNvbHZpbmcgZXh0ZXJuYWwgYCRyZWZgcwoqIE9wZW5BUEkgMi4wIHRvIDMuMC4wIChKU09OL1Jlc29sdmVkKSAtIGxpa2UgdGhlIGFib3ZlIGJ1dCByZXNvbHZpbmcgZXh0ZXJuYWwgYCRyZWZgcwoqIE9wZW5BUEkgdG8gSlNPTiAtIHdoaWNoIHNpbXBseSB0cmFuc2xhdGVzIHRoZSBjdXJyZW50IGRvY3VtZW50IHRvIEpTT04KKiBPcGVuQVBJIHRvIFlBTUwgLSB3aGljaCBzaW1wbHkgdHJhbnNsYXRlcyB0aGUgY3VycmVudCBkb2N1bWVudCB0byBZQU1MCgpUaGUgdmFsaWRhdG9yIGVuZm9yY2VzIHNlbWFudGljIHJlc3RyaWN0aW9ucyB3aGljaCBjYW5ub3QgYmUgZXhwcmVzc2VkIGJ5IHRoZSBKU09OIFNjaGVtYSB3aGljaCBkcml2ZXMgdGhlIGludGVsbGlzZW5zZSBmZWF0dXJlcywgc28gaXQgd2lsbCBwaWNrIHVwIG9uIG1hbnkgbW9yZSBlcnJvcnMuCgpUaGUgbGludGVyIGFwcGxpZXMgYSBsaWdodGx5IG9waW5pb25hdGVkIHNldCBvZiAnYmVzdCBwcmFjdGljZXMnIHRvIHlvdXIgQVBJIGRvY3VtZW50LCBtYWtpbmcgaXQgY29tcGxldGUgYW5kIGNvbXByZWhlbnNpdmUsIGFzIG9wcG9zZWQgdG8gbWVyZWx5IG1pbmltYWxseSB2YWxpZC4KClRoZSBkZWZhdWx0IGxpbnRlciBydWxlcyBhcmUgZG9jdW1lbnRlZCBbaGVyZV0oaHR0cHM6Ly9tZXJtYWRlLmdpdGh1Yi5pby9vYXMta2l0L2RlZmF1bHQtcnVsZXMuaHRtbCkuIFRoZSBsaW50ZXIgW0RTTCBydWxlcyBmb3JtYXRdKGh0dHBzOi8vbWVybWFkZS5naXRodWIuaW8vb2FzLWtpdC9saW50ZXItcnVsZXMuaHRtbCkgaXMgYWxzbyBkb2N1bWVudGVkLgoKIyMgRmVhdHVyZXMKCiogVmFsaWRhdGlvbiB1c2luZyBbb2FzLXZhbGlkYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL01lcm1hZGUvb2FzLWtpdC90cmVlL21hc3Rlci9wYWNrYWdlcy9vYXMtdmFsaWRhdG9yKSBmcm9tIFtPQVMtS2l0XShodHRwczovL21lcm1hZGUuZ2l0aHViLmlvL29hcy1raXQvKQoqIExpbnRpbmcgdXNpbmcgW29hcy1saW50ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL29hcy1raXQvdHJlZS9tYXN0ZXIvcGFja2FnZXMvb2FzLWxpbnRlcikgYWxzbyBmcm9tIE9BUy1LaXQKKiBSZXNvbHV0aW9uIHVzaW5nIFtvYXMtcmVzb2x2ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL29hcy1raXQvdHJlZS9tYXN0ZXIvcGFja2FnZXMvb2FzLXJlc29sdmVyKSBhbHNvIGZyb20gT0FTLUtpdAoqIENvbnZlcnNpb24gdXNpbmcgW3N3YWdnZXIyb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL01lcm1hZGUvb2FzLWtpdC90cmVlL21hc3Rlci9wYWNrYWdlcy9zd2FnZ2VyMm9wZW5hcGkpIGFsc28gZnJvbSBPQVMtS2l0CiogT0FTIHYzIGludGVsbGlzZW5zZSBmb3IgZmlsZXMgbmFtZWQgYCpvcGVuYXBpLmpzb25gLCBgKm9wZW5hcGkueWFtbGAsIGAqb3BlbmFwaS55bWxgLCBgKm9hczMuanNvbmAsIGAqb2FzMy55YW1sYCwgYCpvYXMzLnltbGAKKiBPQVMgdjIgaW50ZWxsaXNlbnNlIGZvciBmaWxlcyBuYW1lZCBgKnN3YWdnZXIuanNvbmAsIGAqc3dhZ2dlci55YW1sYCwgYCpzd2FnZ2VyLnltbGAsIGAqb2FzMi5qc29uYCwgYCpvYXMyLnlhbWxgLCBgKm9hczIueW1sYAoqIENvbnZlcnNpb24gYmV0d2VlbiBKU09OIGFuZCBZQU1MCiogU25pcHBldHMgKG1pbmltYWwgdmFsaWQgb3BlbmFwaS9zd2FnZ2VyIEpTT04gYW5kIFlBTUwgZG9jdW1lbnRzKQoqIEFzeW5jQVBJIHYyLjAueCBpbnRlbGxpc2Vuc2UgZm9yIGZpbGVzIG5hbWVkIGAqYXN5bmNhcGkuanNvbmAsIGAqYXN5bmNhcGkueWFtbGAsIGAqYXN5bmNhcGkueW1sYAoqIEFzeW5jQVBJIHYxLjIueCBpbnRlbGxpc2Vuc2UgZm9yIGZpbGVzIG5hbWVkIGAqYXN5bmNhcGkxLmpzb25gLCBgKmFzeW5jYXBpMS55YW1sYCwgYCphc3luY2FwaTEueW1sYAoKIyMgQ29taW5nIHNvb24KCiogTW9yZSBzbmlwcGV0cwoqIEludGVncmF0aW9uIHdpdGggW0FQSXMuZ3VydSBPcGVuQVBJIGRpcmVjdG9yeV0oaHR0cHM6Ly9naXRodWIuY29tL2FwaXMtZ3VydS9vcGVuYXBpLWRpcmVjdG9yeSkKKiBJbnRlZ3JhdGlvbiB3aXRoIFtBUElzLmd1cnUgQXN5bmNBUEkgZGlyZWN0b3J5XShodHRwczovL2dpdGh1Yi5jb20vYXBpcy1ndXJ1L2FzeW5jYXBpLWRpcmVjdG9yeSkKKiBFeHRyYWN0IGRlZmF1bHQgbGludGVyIHJ1bGVzIHRvIG5ldyB0YWIgdG8gYWxsb3cgc2F2aW5nIGFzIG92ZXJyaWRlcwoKIyMgU2NyZWVuc2hvdHMKCiFbc2NyZWVuc2hvdF0oLi9pbWFnZXMvdnNjb2RlLWxpbnQucG5nKQoKIyMgUmVxdWlyZW1lbnRzCgpUaGUgZXh0ZW5zaW9uIHNob3VsZCB3b3JrIGFzLWlzCgojIyBFeHRlbnNpb24gU2V0dGluZ3MKClRoZSBgb25TYXZlQWN0aW9uYCBjYW4gYmUgc2V0IHRvIG9uZSBvZiBgbm9uZWAsIGB2YWxpZGF0ZWAsIGByZXNvbHZlQW5kVmFsaWRhdGVgICh0aGUgZGVmYXVsdCksIGBsaW50YCBvciBgcmVzb2x2ZUFuZExpbnRgLgoKIyMgS25vd24gSXNzdWVzCgoqIEp1bXAtdG8tZXJyb3IgZnVuY3Rpb25hbGl0eSBub3QgaW1wbGVtZW50ZWQgeWV0IChhbGwgc2hvdyBhcyBsaW5lIDEpCiogTm8gd2F5IHRvIG92ZXJyaWRlL2FtZW5kIGxpbnRlciBydWxlcyB5ZXQKCiMjIENyZWRpdHMKCiogSWNvbiBiYXNlZCB1cG9uIFtJY29uczhdKGh0dHA6Ly9pY29uczguY29tLykKCiMjIFJlbGVhc2UgTm90ZXMKClNlZSB0aGUgW0NIQU5HRUxPR10oLi9DSEFOR0VMT0cubWQpCg== readmeEtag: '"e34f67b2987c46133e75a44a24bedd5c73575ceb"' readmeLastModified: Wed, 12 Aug 2020 13:36:40 GMT repositoryId: 139170191 description: >- OpenAPI 2.0/3.0.x intellisense, validator, linter, converter and resolver extension for Visual Studio Code created: '2018-06-29T16:05:52Z' updated: '2025-07-06T20:31:52Z' language: JavaScript archived: false stars: 52 watchers: 3 forks: 8 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"628cdafcd9832c909187eb4ffb30934252957d5bef2356bdf7aa9f868032e356"' repoLastModified: Sun, 06 Jul 2025 20:31:52 GMT foundInMaster: true name: VSCode/openapi-lint category: - Text Editors - Parsers language: Node.js link: https://marketplace.visualstudio.com/items?itemName=mermade.openapi-lint source_description: OpenAPI 2.0/3.0.x intellisense, validator and linter for Visual Studio Code v2: true id: ab48d5ce77b0c9f808ee6c600d40888c - source: openapi3 tags repository: https://github.com/iyobo/amala v3: true repositoryMetadata: base64Readme: >- IyBBbWFsYQohW2FtYWxhMl0oaHR0cHM6Ly9naXRodWIuY29tL2l5b2JvL2FtYWxhL2Fzc2V0cy81ODA0MjQ2L2FjYzY4YTUyLTExMTAtNGU4NC04NGIzLTc1MjliMWYwZDM1NCkKCgoqKkFtYWxhKiogaXMgYSBkZWNvcmF0b3ItYmFzZWQgQVBJIGZyYW1ld29yayBmb3IgS29hSlMgdjIrIGFuZCBUeXBlc2NyaXB0LgpBbWFsYSBpcyBmYXN0LCBsaWdodCwgc2VsZi1kb2N1bWVudGluZyBhbmQgRG9ja2VyLXJlYWR5OyBQZXJmZWN0IGZvciBtaWNyb3NlcnZpY2VzLgoKLSBEZWZpbmUgeW91ciBSRVNUIEFQSSBlbmRwb2ludHMgdXNpbmcgRVM4IF9jbGFzc2VzXyBhbmQgX2RlY29yYXRvcnNfLgotIEluamVjdCBhcmd1bWVudHMgaW50byB5b3VyIGVuZHBvaW50IGhhbmRsZXJzLCBlZmZlY3RpdmVseSB0dXJuaW5nIHlvdXIgY29udHJvbGxlciBlbmRwb2ludHMgaW50byBzdGFuZGFsb25lLCB0ZXN0YWJsZSBzZXJ2aWNlIGVuZHBvaW50cy4KLSBDbGVhbiwgbGlnaHQgYW5kIEZBU1QgZW5kcG9pbnRzLiBQb3dlcmVkIGJ5IEtvYS4KLSBObyBmdXJ0aGVyIG1hZ2ljIHBhc3QgZGVjb3JhdG9ycy4gRnVsbCBhY2Nlc3MgdG8gdW5kZXJseWluZyBLb2EgYXBwLgotIFByb2plY3QgY3JlYXRvciBjb21lcyB3aXRoIGZ1bGx5IGNvbmZpZ3VyZWQgRG9ja2VyIGFuZCBEb2NrZXItY29tcG9zZSBzZXR0aW5ncyBmb3IgcXVpY2sgY29udGFpbmVyaXphdGlvbi4KLSBJbi1idWlsdCBPcGVuQVBJIHNwZWMgZXhwb3J0ZXIgYW5kIFN3YWdnZXIgVUkhCgpUaGlzIGxlYWRzIHRvIGNsZWFuLCBzZWxmLWRvY3VtZW50aW5nIEFQSSBlbmRwb2ludHMgYW5kIG1ha2VzIGl0IHNvIHlvdSBjYW4gcmUtdXNlIHRob3NlIHNlcnZpY2UgZW5kcG9pbnRzIGVsc2V3aGVyZS4KSXQgYWxzbyBtYWtlcyB5b3VyIGVuZHBvaW50IGVuZHBvaW50cyBlYXNpZXIgdG8gdGVzdC4KCiMjIFN1cHBvcnRpbmcgQW1hbGEKCioqQW1hbGEqKiBpcyBhbiBNSVQtbGljZW5zZWQgb3BlbiBzb3VyY2UgcHJvamVjdCB3aXRoIGl0cyBvbmdvaW5nIGRldmVsb3BtZW50IG1hZGUgcG9zc2libGUgZW50aXJlbHkgYnkKY29tbXVuaXR5IHN1cHBvcnQuIElmIEFtYWxhIGlzIGhlbHBpbmcgeW91IGJ1aWxkCmF3ZXNvbWUgQVBJcywgcGxlYXNlIGNvbnNpZGVyIDxhIGhyZWY9Imh0dHBzOi8vd3d3LnBhdHJlb24uY29tL2JlUGF0cm9uP3U9MTk2NjE5MzkiIGRhdGEtcGF0cmVvbi13aWRnZXQtdHlwZT0iYmVjb21lLXBhdHJvbi1idXR0b24iPkJlY29taW5nIGEgUGF0cm9uPC9hPi4KCklmIHlvdSB3b3VsZCBsaWtlIHRvIGNvbnRyaWJ1dGUgaW4gb3RoZXIgd2F5cywgUHVsbCByZXF1ZXN0cyBhcmUgYWxzbyB3ZWxjb21lIQoKIyMgR2V0dGluZyBzdGFydGVkCllvdSBtYXkgY3JlYXRlIGFuIEFtYWxhIHByb2plY3Qgd2l0aCBhbnkgb2YgdGhlIGZvbGxvd2luZzoKLSBgbnBtIGluaXQgYW1hbGEtYXBwIDxwcm9qZWN0X25hbWU+YAotIGBucG0gY3JlYXRlIGFtYWxhLWFwcCA8cHJvamVjdF9uYW1lPmAKLSBgeWFybiBjcmVhdGUgYW1hbGEtYXBwIDxwcm9qZWN0X25hbWU+YAoKRm9yIG5leHQgc3RlcHMsIHNlZSBvZmZpY2lhbCBkb2NzIGF0IGh0dHBzOi8vYW1hbGFqcy5jb20uCg== readmeEtag: '"b62622a8ea918e0d8d16ac28e4440e2885eb1b63"' readmeLastModified: Tue, 09 Jan 2024 14:46:39 GMT repositoryId: 195160672 description: >- NodeJs Framework for creating REST API endpoints with Typescript decorators. Supports API versioning, OpenAPI3 and docker. Powered by Koa 2+ and Nigerian food (amala+ewedu) created: '2019-07-04T03:11:00Z' updated: '2024-08-02T07:23:52Z' language: TypeScript archived: false stars: 51 watchers: 3 forks: 7 owner: iyobo logo: https://avatars.githubusercontent.com/u/5804246?v=4 license: MIT repoEtag: '"c8c81a05d80a555bc4a6adea08394be40ea6351663f6971979e419d6df375075"' repoLastModified: Fri, 02 Aug 2024 07:23:52 GMT foundInMaster: true category: Server Implementations id: 2907bc65572d999425a855e40ee714ed - source: openapi3 tags repository: https://github.com/openapitools/openapi-petstore v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHBldHN0b3JlCgpUaGlzIGlzIGFuIGltcGxlbWVudGF0aW9uIG9mIHRoZSBPcGVuQVBJIHBldCBzdG9yZSBiYXNlZCBvbiBTcHJpbmctQm9vdC4KCiMjIE92ZXJ2aWV3CgpTdGFydCB5b3VyIHNlcnZlciBhcyBhIHNpbXBsZSBTcHJpbmctQm9vdCBhcHBsaWNhdGlvbgpgYGAKbXZuIHNwcmluZy1ib290OnJ1bgpgYGAKT3IgcGFja2FnZSBpdCB0aGVuIHJ1biBpdCBhcyBhIEphdmEgYXBwbGljYXRpb24KYGBgCm12biBwYWNrYWdlCmphdmEgLWphciB0YXJnZXQvb3BlbmFwaS1wZXRzdG9yZS17VkVSU0lPTn0uamFyCmBgYAoKWW91IGNhbiB2aWV3IHRoZSBhcGkgZG9jdW1lbnRhdGlvbiBpbiBzd2FnZ2VyLXVpIGJ5IHBvaW50aW5nIHRvICAKaHR0cDovL2xvY2FsaG9zdDo4MDgwLwoKIyMgRG9ja2VyCgpUbyBzdGFydCB0aGUgc2VydmVyIHZpYSBkb2NrZXIsIHBsZWFzZSBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kczoKYGBgc2gKZG9ja2VyIHB1bGwgb3BlbmFwaXRvb2xzL29wZW5hcGktcGV0c3RvcmUKZG9ja2VyIHJ1biAtZCAtZSBPUEVOQVBJX0JBU0VfUEFUSD0vdjMgLXAgODA6ODA4MCBvcGVuYXBpdG9vbHMvb3BlbmFwaS1wZXRzdG9yZQpgYGAKClJlZjogaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL29wZW5hcGl0b29scy9vcGVuYXBpLXBldHN0b3JlLwoKIyMgU2VjdXJpdHkKCiMjIyBBUEkga2V5ClVzZSBgc3BlY2lhbC1rZXlgIGZvciBlbmRwb2ludHMgcHJvdGVjdGVkIGJ5IHRoZSBBUEkga2V5CgojIyMgT0F1dGgyCkJ5IGRlZmF1bHQgdGhlIHNlcnZlciBzdXBwb3J0cyB0aGUgaW1wbGljaXQgYW5kIHRoZSBwYXNzd29yZCBmbG93IChldmVuIHRob3VnaCBvbmx5IHRoZSBpbXBsaWNpdCBmbG93IGlzIGRlc2NyaWJlZCBpbiB0aGUgT0FJIHNwZWMpClRoZSBkZWZhdWx0IGNyZWRlbnRpYWxzIGFyZToKKiBjbGllbnQtaWQ6IHNhbXBsZS1jbGllbnQtaWQKKiBjbGllbnQtc2VjcmV0OiBzZWNyZXQKKiB1c2VybmFtZTogdXNlcgoqIHBhc3N3b3JkOiB1c2VyCgojIyBDb25maWd1cmF0aW9uCgpTcHJpbmcgcGFyYW1ldGVycyBpbiBhcHBsaWNhdGlvbi5wcm9wZXJ0aWVzOgoqIFNlcnZlciBwb3J0IDogYHNlcnZlci5wb3J0YCAoZGVmYXVsdD04MDgwKQoqIEFQSSBiYXNlIHBhdGggOiBgb3BlbmFwaS5vcGVuQVBJUGV0c3RvcmUuYmFzZS1wYXRoYCAoZGVmYXVsdD0vdjMpLiBJbiB0aGUgZG9ja2VyIGltYWdlIHRoZSBiYXNlIHBhdGggY2FuIGFsc28gYmUgc2V0IHdpdGggdGhlIGBPUEVOQVBJX0JBU0VfUEFUSGAgZW52aXJvbm1lbnQgdmFyaWFibGUuCgpFbnZpcm9ubWVudCB2YXJpYWJsZXM6CiogYERJU0FCTEVfQVBJX0tFWWAgOiBpZiBzZXQgdG8gIjEiLCB0aGUgc2VydmVyIHdpbGwgbm90IGNoZWNrIHRoZSBhcGkga2V5IGZvciB0aGUgcmVsZXZhbnQgZW5kcG9pbnRzLgoqIGBESVNBQkxFX09BVVRIYCA6IGlmIHNldCB0byAiMSIsIHRoZSBzZXJ2ZXIgd2lsbCBub3QgY2hlY2sgZm9yIGFuIE9BdXRoMiBhY2Nlc3MgdG9rZW4uCgojIyBMaWNlbnNlCgpbQXBhY2hlIDIuMCBMaWNlbnNlXShodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAp readmeEtag: '"31d448ef5211e24c2722fa848d2623050d3a9609"' readmeLastModified: Mon, 13 Aug 2018 06:24:53 GMT repositoryId: 136197966 description: The pet store sample created: '2018-06-05T15:27:38Z' updated: '2025-12-05T08:03:57Z' language: Java archived: false stars: 52 watchers: 5 forks: 45 owner: OpenAPITools logo: https://avatars.githubusercontent.com/u/37325267?v=4 license: Apache-2.0 repoEtag: '"7d8d76a3380223d39e35569bc167dd7331b6857e77bc652d390968f4c3adedad"' repoLastModified: Fri, 05 Dec 2025 08:03:57 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 25fdf5fe94f828502b6c15037763b24c - source: openapi3 tags name: Gen homepage: https://github.com/wzshiming/gen language: Go source_description: Generate OpenAPI 3, client, and route based on golang source code. category: Code Generators repository: https://github.com/wzshiming/gen v3: true repositoryMetadata: base64Readme: >- IyBHZW4gLSBUb29scyBmb3IgZ2VuZXJhdGluZyBzb3VyY2UgY29kZSBmb3IgbWljcm9zZXJ2aWNlcwoKSnVzdCB3cml0ZSBub3JtYWwgZnVuY3Rpb25zLCBhbmQgR2VuIGdlbmVyYXRlcyBlZmZpY2llbnQgcm91dGluZyBzb3VyY2UgY29kZSBhbmQgZG9jdW1lbnRhdGlvbiBmb3IgaXQKQmVjYXVzZSB0aGUgc291cmNlIGNvZGUgaXMgZ2VuZXJhdGVkLCBub25lIG9mIHRoaXMgYWZmZWN0cyBydW50aW1lIHBlcmZvcm1hbmNlICAKVGhlIGRpZmZlcmVuY2VzIGNhdXNlZCBieSBlYWNoIGNoYW5nZSBpbiB0aGUgdG9vbCBhcmUgc2hvd24gZGlyZWN0bHkgaW4gdGhlIGdlbmVyYXRlZCBzb3VyY2UgY29kZSAgCmdlbmVyYXRpbmcgY2xpZW50cyBpcyBhbHNvIHN1cHBvcnRlZCAgCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy93enNoaW1pbmcvZ2VuLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL3d6c2hpbWluZy9nZW4pClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS93enNoaW1pbmcvZ2VuKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL3d6c2hpbWluZy9nZW4pClshW0dpdEh1YiBsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL3d6c2hpbWluZy9nZW4uc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3d6c2hpbWluZy9nZW4vYmxvYi9tYXN0ZXIvTElDRU5TRSkKCi0gW0VuZ2xpc2hdKGh0dHBzOi8vZ2l0aHViLmNvbS93enNoaW1pbmcvZ2VuL2Jsb2IvbWFzdGVyL1JFQURNRS5tZCkKLSBb566A5L2T5Lit5paHXShodHRwczovL2dpdGh1Yi5jb20vd3pzaGltaW5nL2dlbi9ibG9iL21hc3Rlci9SRUFETUVfY24ubWQpCgojIyBFeGFtcGxlcwoKJyMnIGlzIHRoZSBhbm5vdGF0aW9uLCB0aGUgYW5ub3RhdGlvbiBpcyB0aGUgZ29sYW5nIHRhZyBzeW50YXgsIHRoZSBvbmx5IGRpZmZlcmVuY2UgaGVyZSBpcyAnIycgd3JhcHMgbm90ICdgJy4KCmBgYCBnb2xhbmcKLy8gSXRlbVNlcnZpY2UgI3BhdGg6Ii9pdGVtLyIjCnR5cGUgSXRlbVNlcnZpY2Ugc3RydWN0IHt9CgovLyBDcmVhdGUgYSBJdGVtICNyb3V0ZToiUE9TVCAvIiMKZnVuYyAocyAqSXRlbVNlcnZpY2UpIENyZWF0ZShpdGVtICpJdGVtKSAoZXJyIGVycm9yKSB7fQoKLy8gVXBkYXRlIHRoZSBJdGVtICNyb3V0ZToiUFVUIC97aXRlbV9pZH0iIwpmdW5jIChzICpJdGVtU2VydmljZSkgVXBkYXRlKGl0ZW1JRCBpbnQgLyogI25hbWU6Iml0ZW1faWQiIyAqLywgaXRlbSAqSXRlbSkgKGVyciBlcnJvcikge30KCi8vIERlbGV0ZSB0aGUgSXRlbSAjcm91dGU6IkRFTEVURSAve2l0ZW1faWR9IiMKZnVuYyAocyAqSXRlbVNlcnZpY2UpIERlbGV0ZShpdGVtSUQgaW50IC8qICNuYW1lOiJpdGVtX2lkIiMgKi8pIChlcnIgZXJyb3IpIHt9CgovLyBHZXQgdGhlIEl0ZW0gI3JvdXRlOiJHRVQgL3tpdGVtX2lkfSIjCmZ1bmMgKHMgKkl0ZW1TZXJ2aWNlKSBHZXQoaXRlbUlEIGludCAvKiAjbmFtZToiaXRlbV9pZCIjICovKSAoaXRlbSAqSXRlbVdpdGhJRCwgZXJyIGVycm9yKSB7fQoKLy8gTGlzdCBvZiB0aGUgSXRlbSAjcm91dGU6IkdFVCAvIiMKZnVuYyAocyAqSXRlbVNlcnZpY2UpIExpc3Qob2Zmc2V0LCBsaW1pdCBpbnQpIChpdGVtcyBbXSpJdGVtV2l0aElELCBlcnIgZXJyb3IpIHt9CmBgYAoKMS4gSW5zdGFsbCBnZW4gdG9vbCBgZ28gZ2V0IC12IGdpdGh1Yi5jb20vd3pzaGltaW5nL2dlbi9jbWQvZ2VuYAoyLiBBZGQgZ2VuIHRvb2wgdG8gJFBBVEgKMy4gU3RhcnQgaXQgYGdlbiBydW4gZ2l0aHViLmNvbS93enNoaW1pbmcvZ2VuLWV4YW1wbGVzL3NlcnZpY2UvLi4uYAo0LiBPcGVuIFtodHRwOi8vMTI3LjAuMC4xOjgwODAvc3dhZ2dlci8/dXJsPS4vb3BlbmFwaS5qc29uI10oaHR0cDovLzEyNy4wLjAuMTo4MDgwL3N3YWdnZXIvP3VybD0uL29wZW5hcGkuanNvbiMpIHdpdGggeW91ciBicm93c2VyCgpbRXhhbXBsZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS93enNoaW1pbmcvZ2VuLWV4YW1wbGVzLykgIAoKT3IgdHJ5IHRvIHF1aWNrbHkgYnVpbGQgc2VydmljZXMgZnJvbSBzY3JhdGNoCgoxLiBNYWtlIGEgZGlyZWN0b3J5IGBta2RpciAtcCAkKGdvIGVudiBHT1BBVEgpL3NyYy9nZW50ZXN0YAoyLiBDaGFuZ2UgZGlyZWN0b3J5IGBjZCAkKGdvIGVudiBHT1BBVEgpL3NyYy9nZW50ZXN0L2AKMy4gRGVmaW5lIG1vZGVscwpgYGAgc2hlbGwKY2F0ID4gbW9kZWxzLmdvIDw8RU9GCnBhY2thZ2UgZ2VudGVzdAp0eXBlIEdlbnRlc3Qgc3RydWN0IHsKICAgIE5hbWUgc3RyaW5nIFxganNvbjoibmFtZSJcYAogICAgQWdlICBpbnQgICAgXGBqc29uOiJhZ2UiXGAKfQpFT0YKYGBgCjQuIEdlbmVyYXRlZCBmcm9tIENSVUQgdGVtcGxhdGUgYGdlbiBjcnVkIC10IG1vY2sgLW4gR2VudGVzdGAKNS4gU3RhcnQgaXQgYEdPMTExTU9EVUxFPW9mZiBnZW4gcnVuIGdlbnRlc3RgCgojIyBTdXBwb3J0ZWQKCi0gW1hdIEdlbmVyYXRlIGRvY3VtZW50YXRpb24KICAtIFtYXSBbT3BlbkFQSSAzXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3R5bGUtR3VpZGUpCiAgLSBbWF0gW1N3YWdnZXJVSV0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkpCiAgLSBbWF0gW1JlRG9jXShodHRwczovL2dpdGh1Yi5jb20vUmViaWxseS9SZURvYykKLSBbWF0gUkVTVGZ1bAogIC0gW1hdIEdlbmVyYXRlIEdvIHJvdXRlcgogICAgLSBbWF0gU2VjdXJpdHkKICAgICAgLSBbWF0gYXBpS2V5CiAgICAgIC0gW1hdIGh0dHAKICAgICAgICAtIFtYXSBiYXNpYwogICAgICAgIC0gWyBdIGJlYXJlcgogICAgICAtIFsgXSBvYXV0aDIKICAgICAgLSBbIF0gb3BlbklkQ29ubmV0CiAgICAtIFtYXSBDb250ZW50CiAgICAgIC0gW1hdIFF1ZXJ5CiAgICAgIC0gW1hdIFBhdGgKICAgICAgLSBbWF0gSGVhZGVyCiAgICAgIC0gW1hdIENvb2tpZQogICAgICAtIFtYXSBCb2R5CiAgICAgICAgLSBbWF0gSlNPTgogICAgICAgIC0gWyBdIFhNTAogICAgICAgIC0gWyBdIEZvcm1kYXRhCiAgICAgICAgICAtIFtYXSBGaWxlCiAgICAgICAgICAtIFsgXSBWYWx1ZQogICAgICAgIC0gWyBdIFVSTEVuY29kZQogIC0gW1hdIEdlbmVyYXRlIEdvIGNsaWVudAogICAgLSBbWF0gU2VjdXJpdHkKICAgICAgLSBbWF0gYXBpS2V5CiAgICAgIC0gW1hdIGh0dHAKICAgICAgICAtIFtYXSBiYXNpYwogICAgICAgIC0gW1hdIGJlYXJlcgogICAgICAtIFsgXSBvYXV0aDIKICAgICAgLSBbIF0gb3BlbklkQ29ubmV0CiAgICAtIFtYXSBDb250ZW50CiAgICAgIC0gW1hdIFF1ZXJ5CiAgICAgIC0gW1hdIFBhdGgKICAgICAgLSBbWF0gSGVhZGVyCiAgICAgIC0gW1hdIENvb2tpZQogICAgICAtIFtYXSBCb2R5CiAgICAgICAgLSBbWF0gSlNPTgogICAgICAgIC0gW1hdIFhNTAogICAgICAgIC0gW1hdIEZvcm1kYXRhCiAgICAgICAgICAtIFtYXSBGaWxlCiAgICAgICAgICAtIFtYXSBWYWx1ZQogICAgICAgIC0gWyBdIFVSTEVuY29kZQogIC0gT3RoZXIgQ2xpZW50CiAgICAtIFtPcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IpCiAgICAtIFtzd2FnZ2VyLWFwaS9zd2FnZ2VyLWpzXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1qcykKICAgIC0gW3N3YWdnZXItYXBpL3N3YWdnZXItY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29kZWdlbi90cmVlLzMuMC4wKQotIFsgXSBnUlBDICYgUHJvdG8zCgojIyBMaWNlbnNlCgpQb3VjaCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSBbTElDRU5TRV0oaHR0cHM6Ly9naXRodWIuY29tL3d6c2hpbWluZy9nZW4vYmxvYi9tYXN0ZXIvTElDRU5TRSkgZm9yIHRoZSBmdWxsIGxpY2Vuc2UgdGV4dC4K readmeEtag: '"689ac134bbeafe8e6c678104343f9b1901004fd3"' readmeLastModified: Mon, 13 Dec 2021 05:31:34 GMT repositoryId: 138471634 description: >- Gen generates efficient web routing source code and documentation from annotations created: '2018-06-24T10:07:22Z' updated: '2024-11-15T22:55:13Z' language: Go archived: false stars: 50 watchers: 5 forks: 6 owner: wzshiming logo: https://avatars.githubusercontent.com/u/6565744?v=4 license: MIT repoEtag: '"7e78df966689048f53818f59c1ac16667319583b84c3b20e50e35da16530ecb6"' repoLastModified: Fri, 15 Nov 2024 22:55:13 GMT foundInMaster: true id: 94aefe764a91d556fe6001241c0796f8 - source: openapi3 tags repository: https://github.com/sanjayvacharya/sleeplessinslc v3: true repositoryMetadata: base64Readme: >- IyBzbGVlcGxlc3NpbnNsYwpEZW1vbnN0cmF0aW9uIGFuZCBUdXRvcmlhbCBjb2RlIGZyb20gdGhlIEJMT0cgaHR0cDovL3NsZWVwbGVzc2luc2xjLmJsb2dzcG90LmNvbS4gSWYgeW91IGNsb25lIHRoaXMgcmVwbywgZm9yayB0aGlzIHJlcG8sIHNoYXJlIGEgc3RhciB0byBzaG93IGhvdyBhd2Vzb21lIGl0IGlzISBDaGVlcnMhCg== readmeEtag: '"a0791fcbf318c272bf621109ee8decf5097bad50"' readmeLastModified: Sun, 26 Jul 2020 19:06:11 GMT repositoryId: 35256352 description: 'Demo code from ' created: '2015-05-08T03:21:12Z' updated: '2024-08-24T01:36:19Z' language: Java archived: false stars: 50 watchers: 2 forks: 61 owner: sanjayvacharya logo: https://avatars.githubusercontent.com/u/605398?v=4 repoEtag: '"8169caa1efcb9baedbb66a48d092c88b19f23c1a9fcc60b25fb8cf0d01f70783"' repoLastModified: Sat, 24 Aug 2024 01:36:19 GMT foundInMaster: true category: - SDK - Server Implementations id: cab64553444e97be76afb26aa96e229a - source: openapi3 tags repository: https://github.com/rots/n26-api v3: true repositoryMetadata: base64Readme: >- IyBuMjYtYXBpClVub2ZmaWNpYWwgTjI2IEFQSSBkb2N1bWVudGF0aW9uCgpJdCBpcyBhIGNvbGxlY3Rpb24gb2Ygb2JzZXJ2ZWQgQVBJIGNhbGxzIGFuZCBtZXRob2RzLiBObyBndWFyYW50ZWVzIGFyZSBwcm92aWRlZCB0aGF0IHRoZXkgYXJlIGNvbXBsZXRlLCBjb3JyZWN0IG9yIHRoYXQgdGhleSBhcmUgc3RpbGwgZnVuY3Rpb25pbmcgYXMgc3BlY2lmaWVkLgoKVGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBpcyBwcm92aWRlZCBpbiB0aGUgYG9wZW5hcGkueW1sYCBmaWxlLgpGb3IgdGVzdGluZyB0aGUgc3BlYywgaXQgaXMgYWxzbyBwdWJsaXNoZWQgYXQgaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcy9Sb3RzL04yNgoKCiMgQ2xpZW50IHdyYXBwZXIgZ2VuZXJhdGlvbgoKVGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBhbGxvd3MgZm9yIGdlbmVyYXRpbmcgY2xpZW50IHdyYXBwZXJzIHRvIGFjY2VzcyB0aGUgZGF0YS4KCkV4YW1wbGU6CgpgYGAKZG9ja2VyIHJ1biAtLXJtIC12ICR7UFdEfTovbG9jYWwga29yY2hhc2Evc3dhZ2dlci1jb2RlZ2VuLW9wZW5hcGkgZ2VuZXJhdGUgLWkgL2xvY2FsL29wZW5hcGkueW1sIC1sIGphdmEgLW8gL2xvY2FsL291dC9qYXZhCmBgYAoKIyBTZWN1cml0eSB3YXJuaW5nIQoKRG8gbm90IHR5cGUgaW4geW91ciBiYW5rIGNyZWRlbnRpYWxzIHRvIGFuIHVudHJ1c3RlZCBzb2Z0d2FyZSEKT25jZSB5b3UgaGF2ZSBnaXZlbiBmdWxsIGFjY2VzcyB0byBhIHRoaXJkIHBhcnR5LCB0aGV5IGNhbiBpbmRlZmluaXRlbHkgImtlZXAgdGhlIGJhbmsgQVBJIHNlc3Npb24gYWxpdmUiIGJ5IHJlZnJlc2hpbmcgdGhlaXIgYWNjZXNzIHRva2VuIGFuZCBtYWtlIHRyYW5zYWN0aW9ucyBvbiB5b3VyIGJlaGFsZiAoYW5kIGNoYW5nZSBhbnkgZGV0YWlscyBvbiB0aGUgYmFua2luZyBhcHApLgpOMjYgaGF2ZW4ndCBlbmFibGVkIGEgd2F5IHRvIHJldm9rZSBhY2Nlc3Mgb2YgdGhpcmQgcGFydHkgYXBwcyAoYXMgZmFyIGFzIEkga25vdywgeW91IGNhbid0IHNlY3VyZWx5IHNpZ24gaW4gdG8gdGhlIGJhbmsgd2l0aG91dCBnaXZpbmcgeW91ciBwYXNzd29yZCBpbiBwbGFpbiB0ZXh0IHRvIHRoZSB0aGlyZCBwYXJ0eSkuCgpJZiB5b3UgdGhpbmsgdGhhdCB5b3VyIGNyZWRlbnRpYWxzIG1heSBoYXZlIGJlZW4gY29tcHJvbWlzZWQsIGltbWlkaWF0ZWx5IGNoYW5nZSB5b3VyIGxvZ2luIHBhc3N3b3JkIGluIHRoZSBvZmZpY2lhbCBOMjYgYXBwIGFuZCBub3RpZnkgdGhlIGJhbmsgb2YgYW55IHJlbGV2YW50IGluZm9ybWF0aW9uLgoKIyBPdGhlciBrbm93biBzb2Z0d2FyZSBmb3IgTjI2CgoqIGh0dHBzOi8vZ2l0aHViLmNvbS9ndWl0bXovbjI2IEdvIEFQSQoqIGh0dHBzOi8vZ2l0aHViLmNvbS9QaWVycmlja1AvbjI2IEpTT04gQVBJCgo= readmeEtag: '"87a9f22df22926c417642603db1ab7ee253d8872"' readmeLastModified: Tue, 01 May 2018 11:05:25 GMT repositoryId: 131702971 description: Unofficial N26 Bank API documentation created: '2018-05-01T10:55:00Z' updated: '2025-08-20T20:23:43Z' language: null archived: false stars: 50 watchers: 5 forks: 4 owner: Rots logo: https://avatars.githubusercontent.com/u/362397?v=4 license: MIT repoEtag: '"3f722be8cd390b950e5f87942d84b0c56eff1ec1a22c62c6874f06d36bfe6d43"' repoLastModified: Wed, 20 Aug 2025 20:23:43 GMT foundInMaster: true category: Server Implementations id: 1ea5bf53ca4a549ad53a8eda42392826 - source: - openapi3 tags - openapi31 tags repository: https://github.com/sv-tools/openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHYzLjEgU3BlY2lmaWNhdGlvbgoKWyFbQ29kZSBBbmFseXNpc10oaHR0cHM6Ly9naXRodWIuY29tL3N2LXRvb2xzL29wZW5hcGkvYWN0aW9ucy93b3JrZmxvd3MvY29kZS55YW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9zdi10b29scy9vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2NvZGUueWFtbCkKWyFbR28gUmVmZXJlbmNlXShodHRwczovL3BrZy5nby5kZXYvYmFkZ2UvZ2l0aHViLmNvbS9zdi10b29scy9vcGVuYXBpLnN2ZyldKGh0dHBzOi8vcGtnLmdvLmRldi9naXRodWIuY29tL3N2LXRvb2xzL29wZW5hcGkpClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9zdi10b29scy9vcGVuYXBpL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj0wWFZPVERSMUNXKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL3N2LXRvb2xzL29wZW5hcGkpClshW0dpdEh1YiB0YWcgKGxhdGVzdCBTZW1WZXIpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3RhZy9zdi10b29scy9vcGVuYXBpP3N0eWxlPWZsYXQpXShodHRwczovL2dpdGh1Yi5jb20vc3YtdG9vbHMvb3BlbmFwaS9yZWxlYXNlcykKClRoZSBpbXBsZW1lbnRhdGlvbiBvZiBPcGVuQVBJIHYzLjEgU3BlY2lmaWNhdGlvbiBmb3IgR28gdXNpbmcgZ2VuZXJpY3MuCgpgYGBzaGVsbApnbyBnZXQgZ2l0aHViLmNvbS9zdi10b29scy9vcGVuYXBpCmBgYAoKIyMgU3VwcG9ydGVkIEdvIHZlcnNpb25zCgoqIHYxLjI1CiogdjEuMjQKKiB2MS4yMwoqIHYxLjIyCgojIyBWZXJzaW9ucwoKKiB2MCAtICoqRGVwcmVjYXRlZCoqLiBUaGUgaW5pdGlhbCB2ZXJzaW9uIHdpdGggdGhlIGZ1bGwgaW1wbGVtZW50YXRpb24gb2YgdGhlIHYzLjEgU3BlY2lmaWNhdGlvbiB1c2luZyBnZW5lcmljcy4gU2VlIHRoZSBgdjBgIGJyYW5jaC4KKiB2MSAtIFRoZSBjdXJyZW50IHZlcnNpb24gd2l0aCB0aGUgaW4tcGxhY2UgdmFsaWRhdGlvbiBvZiB0aGUgc3BlY2lmaWNhdGlvbi4KICAqIFRoZSBtaW5pbXVtIHZlcnNpb24gb2YgR28gaXMgYHYxLjIyYC4KICAqIEV2ZXJ5dGhpbmcgaGFzIGJlZW4gbW92ZWQgdG8gdGhlIHJvb3QgZm9sZGVyLiBTbywgdGhlIGltcG9ydCBwYXRoIGlzIGBnaXRodWIuY29tL3N2LXRvb2xzL29wZW5hcGlgLgogICogQWRkZWQgYFZhbGlkYXRvcmAgc3RydWN0IGZvciB2YWxpZGF0aW9uIG9mIHRoZSBzcGVjaWZpY2F0aW9uIGFuZCB0aGUgZGF0YS4KICAgICogYFZhbGlkYXRvci5WYWxpZGF0ZVNwZWMoKWAgbWV0aG9kIHZhbGlkYXRlcyB0aGUgc3BlY2lmaWNhdGlvbi4KICAgICogYFZhbGlkYXRvci5WYWxpZGF0ZURhdGEoKWAgbWV0aG9kIHZhbGlkYXRlcyB0aGUgZGF0YS4KICAgICogYFZhbGlkYXRvci5WYWxpZGF0ZURhdGFBc0pTT04oKWAgbWV0aG9kIHZhbGlkYXRlcyB0aGUgZGF0YSBieSBjb252ZXJ0aW5nIGl0IGludG8gYG1hcFtzdHJpbmddYW55YCB0eXBlIGZpcnN0IHVzaW5nIGBqc29uLk1hcnNoYWxgIGFuZCBganNvbi5Vbm1hcnNoYWxgLgogICAgICAqKldBUk5JTkcqKjogdGhlIGZ1bmN0aW9uIGlzIHNsb3cgZHVlIHRvIGRvdWJsZSBjb252ZXJzaW9uLgogICogQWRkZWQgYFBhcnNlT2JqZWN0YCBmdW5jdGlvbiB0byBjcmVhdGUgYFNjaGVtYUJ1aWxkZXJgIGJ5IHBhcnNpbmcgYW4gb2JqZWN0LgogICAgVGhlIGZ1bmN0aW9uIHN1cHBvcnRzIGBqc29uYCwgYHlhbWxgIGFuZCBgb3BlbmFwaWAgZmllbGQgdGFncyBmb3IgdGhlIHN0cnVjdHMuCiAgKiBVc2UgT3BlbkFQSSBgdjMuMS4xYCBieSBkZWZhdWx0LgoKIyMgRmVhdHVyZXMKCiogVGhlIG9mZmljaWFsIHYzLjAgYW5kIHYzLjEgW2V4YW1wbGVzXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi90cmVlL21haW4vZXhhbXBsZXMpIGFyZSB0ZXN0ZWQuCiAgSW4gbW9zdCBjYXNlcywgdGhlIHYzLjAgc3BlY2lmaWNhdGlvbiBjYW4gYmUgY29udmVydGVkIHRvIHYzLjEgYnkgY2hhbmdpbmcgb25seSB0aGUgdmVyc2lvbiBwYXJhbWV0ZXIuCgogIGBgYGRpZmYKICBAQCAtMSw0ICsxLDQgQEAKICAtb3BlbmFwaTogIjMuMC4wIgogICtvcGVuYXBpOiAiMy4xLjAiCiAgYGBgCgoqKk5PVEUqKjogVGhlIGRlc2NyaXB0aW9ucyBvZiBtb3N0IHN0cnVjdHVyZXMgYW5kIHRoZWlyIGZpZWxkcyBhcmUgdGFrZW4gZnJvbSB0aGUgb2ZmaWNpYWwgZG9jdW1lbnRhdGlvbnMuCgojIyBMaW5rcwoKKiBPcGVuQVBJIFNwZWNpZmljYXRpb246IDxodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbj4gYW5kIDxodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4xLjA+CiogSlNPTiBTY2hlbWE6IDxodHRwczovL2pzb24tc2NoZW1hLm9yZy91bmRlcnN0YW5kaW5nLWpzb24tc2NoZW1hL2luZGV4Lmh0bWw+IGFuZCA8aHR0cHM6Ly9qc29uLXNjaGVtYS5vcmcvZHJhZnQvMjAyMC0xMi9qc29uLXNjaGVtYS1jb3JlLmh0bWw+CiogVGhlIGxpc3Qgb2YgbW9zdCBwb3B1bGFyIGFsdGVybmF0aXZlczogPGh0dHBzOi8vdG9vbHMub3BlbmFwaXMub3JnPgoKIyMgTGljZW5zZQoKTUlUIGxpY2Vuc2VkLiBTZWUgdGhlIGJ1bmRsZWQgW0xJQ0VOU0VdKExJQ0VOU0UpIGZpbGUgZm9yIG1vcmUgZGV0YWlscy4K readmeEtag: '"f7775bf70dd41e2e11fbdd58845c266446c7f206"' readmeLastModified: Tue, 26 Aug 2025 18:29:28 GMT repositoryId: 439764250 description: OpenAPI v3.1 Spec implementation in Go with generics created: '2021-12-19T03:11:04Z' updated: '2025-11-10T13:26:50Z' language: Go archived: false stars: 56 watchers: 3 forks: 11 owner: sv-tools logo: https://avatars.githubusercontent.com/u/59230487?v=4 license: MIT repoEtag: '"69c3515437b993b51bdb05701b82353f59fefb52724f41c22e555f847645dc92"' repoLastModified: Mon, 10 Nov 2025 13:26:50 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 3d4e8c0ed6ec76b10233ee8d28bd125e name: openapi homepage: https://github.com/sv-tools/openapi language: Go source_description: OpenAPI v3.1 Spec implementation in Go with generics v3_1: true - source: openapi3 tags repository: https://github.com/javalin/javalin-openapi v3: true id: 91ce32879876ec8e68965162b84f61ec repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFBsdWdpbiBbIVtDSV0oaHR0cHM6Ly9naXRodWIuY29tL2phdmFsaW4vamF2YWxpbi1vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2dyYWRsZS55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2phdmFsaW4vamF2YWxpbi1vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2dyYWRsZS55bWwpICFbTWF2ZW4gQ2VudHJhbF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3YvaW8uamF2YWxpbi5jb21tdW5pdHkub3BlbmFwaS9vcGVuYXBpLWFubm90YXRpb24tcHJvY2Vzc29yP2xhYmVsPU1hdmVuJTIwQ2VudHJhbCkgWyFbVmVyc2lvbiAvIFNuYXBzaG90XShodHRwczovL21hdmVuLnJlcG9zaWxpdGUuY29tL2FwaS9iYWRnZS9sYXRlc3Qvc25hcHNob3RzL2lvL2phdmFsaW4vY29tbXVuaXR5L29wZW5hcGkvamF2YWxpbi1vcGVuYXBpLXBsdWdpbj9jb2xvcj1BOTdCRkYmbmFtZT1TbmFwc2hvdCldKGh0dHBzOi8vbWF2ZW4ucmVwb3NpbGl0ZS5jb20vIy9zbmFwc2hvdHMvaW8vamF2YWxpbi9jb21tdW5pdHkvb3BlbmFwaSkKQ29tcGlsZS10aW1lIE9wZW5BUEkgaW50ZWdyYXRpb24gZm9yIEphdmFsaW4gNi54IGVjb3N5c3RlbS4KVGhpcyBpcyBhIG5ldyBwbHVnaW4gdGhhdCByZXBsYWNlcyBbb2xkIGJ1aWx0LWluIE9wZW5BcGkgbW9kdWxlXShodHRwczovL2dpdGh1Yi5jb20vamF2YWxpbi9qYXZhbGluL3RyZWUvamF2YWxpbi00eC9qYXZhbGluLW9wZW5hcGkpLCAKdGhlIEFQSSBsb29rcyBxdWl0ZSB0aGUgc2FtZSBkZXNwaXRlIHNvbWUgbWlub3IgY2hhbmdlcy4KCiFbUHJldmlld10oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vNDIzNTcyMi8xMjI5ODIxNjItZDIzNDRmODAtZDM5YS0xMWViLTlhOTMtZTUyYjliN2I3YjUzLnBuZykKCiMjIyBIb3cgdG8gdXNlCgoqIFtXaWtpIC8gSW5zdGFsbGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vamF2YWxpbi9qYXZhbGluLW9wZW5hcGkvd2lraS8xLi1JbnN0YWxsYXRpb24pCiogW1dpa2kgLyBTZXR1cF0oaHR0cHM6Ly9naXRodWIuY29tL2phdmFsaW4vamF2YWxpbi1vcGVuYXBpL3dpa2kvMi4tU2V0dXApCiogW1dpa2kgLyBGZWF0dXJlc10oaHR0cHM6Ly9naXRodWIuY29tL2phdmFsaW4vamF2YWxpbi1vcGVuYXBpL3dpa2kvMy4tRmVhdHVyZXMpCgojIyMgTm90ZXMKKiBSZWZsZWN0aW9uIGZyZWUsIGRvZXMgbm90IHBlcmZvcm0gYW55IGV4dHJhIG9wZXJhdGlvbnMgYXQgcnVudGltZQoqIFVzZXMgYEBPcGVuQXBpYCB0byBzaW1wbGlmeSBtaWdyYXRpb24gZnJvbSBidW5kbGVkIE9wZW5BcGkgaW1wbGVtZW50YXRpb24KKiBTdXBwb3J0cyBKYXZhIDExKyAoYWxzbyAxNiBhbmQgYW55IGZ1cnRoZXIgcmVsZWFzZXMpIGFuZCBLb3RsaW4gKHRocm91Z2ggW0thcHRdKGh0dHBzOi8va290bGlubGFuZy5vcmcvZG9jcy9rYXB0Lmh0bWwpKQoqIFVzZXMgaW50ZXJuYWwgV2ViSmFyIGhhbmRsZXIgdGhhdCB3b3JrcyB3aXRoIGAvKmAgcm91dGUgb3V0IG9mIHRoZSBib3gKKiBQcm92aWRlcyBiZXR0ZXIgcHJvamVjdGlvbiBvZiBPcGVuQVBJIHNwZWNpZmljYXRpb24KKiBTY2hlbWEgdmFsaWRhdGlvbiB0aHJvdWdoIFN3YWdnZXIgY29yZSBtb2R1bGUKCiMjIyBPdGhlciBleGFtcGxlcwoqIFtUZXN0IG1vZHVsZV0oaHR0cHM6Ly9naXRodWIuY29tL2phdmFsaW4vamF2YWxpbi1vcGVuYXBpL2Jsb2IvbWFpbi9leGFtcGxlcy9qYXZhbGluLWdyYWRsZS1rb3RsaW4vc3JjL21haW4vamF2YS9pby9qYXZhbGluL29wZW5hcGkvcGx1Z2luL3Rlc3QvSmF2YWxpblRlc3QuamF2YSkgLSBgSmF2YWxpblRlc3RgIHNob3dzIGhvdyB0aGlzIHBsdWdpbiB3b3JrIGluIEphdmEgY29kZWJhc2UgdXNpbmcgdmFyaW91cyBmZWF0dXJlcwoqIFtSZXBvc2lsaXRlXShodHRwczovL2dpdGh1Yi5jb20vZHppa295c2svcmVwb3NpbGl0ZSkgLSByZWFsIHdvcmxkIGFwcCB1c2luZyBKYXZhbGluIGFuZCBPcGVuQXBpIGludGVncmF0aW9uCgojIyMgUmVwb3NpdG9yeSBzdHJ1Y3R1cmUKCiMjIyMgVW5pdmVyc2FsIG1vZHVsZXMKCnwgTW9kdWxlICAgICAgICAgICAgICAgICAgICAgICAgIHwgRGVzY3JpcHRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfDotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfDotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8IGBvcGVuYXBpLWFubm90YXRpb24tcHJvY2Vzc29yYCB8IENvbXBpbGUtdGltZSBhbm5vdGF0aW9uIHByb2Nlc3Nvciwgc2hvdWxkIGdlbmVyYXRlIGAvb3BlbmFwaS1wbHVnaW4vb3BlbmFwaS5qc29uYCByZXNvdXJjZSB8CnwgYG9wZW5hcGktc3BlY2lmaWNhdGlvbmAgICAgICAgIHwgQW5ub3RhdGlvbnMgJiBjbGFzc2VzIHVzZWQgdG8gZGVzY3JpYmUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgb3BlbmFwaS10ZXN0YCAgICAgICAgICAgICAgICAgfCBFeGFtcGxlIEphdmFsaW4gYXBwbGljYXRpb24gdGhhdCB1c2VzIE9wZW5BcGkgcGx1Z2luIGluIEdyYWRsZSAmIE1hdmVuICAgICAgICAgICAgICAgICAgICAgfAoKIyMjIyBKYXZhbGluIHBsdWdpbnMKCnwgUGx1Z2luICAgICAgICAgICAgICAgICAgIHwgRGVzY3JpcHRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfDotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfDotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8IGBqYXZhbGluLW9wZW5hcGktcGx1Z2luYCB8IExvYWRzIGAvb3BlbmFwaS1wbHVnaW4vb3BlbmFwaS5qc29uYCByZXNvdXJjZSBhbmQgc2VydmVzIG1haW4gT3BlbkFwaSBlbmRwb2ludCB8CnwgYGphdmFsaW4tc3dhZ2dlci1wbHVnaW5gIHwgU2VydmVzIFN3YWdnZXIgVUkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgamF2YWxpbi1yZWRvYy1wbHVnaW5gICAgfCBTZXJ2ZXMgUmVEb2MgVUkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAoKIyMjIyBCcmFuY2hlcwoKfCBCcmFuY2ggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCBKYXZhbGluIHZlcnNpb24gfCBPcGVuQXBpIFZlcnNpb24gfCBKYXZhIFZlcnNpb24gfAp8Oi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS18CnwgW21haW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9qYXZhbGluL2phdmFsaW4tb3BlbmFwaS90cmVlL21haW4pIHwgNi54ICAgICAgICAgICAgIHwgNi54ICAgICAgICAgICAgIHwgSkRLMTEgICAgICAgIHwKfCBbNS54XShodHRwczovL2dpdGh1Yi5jb20vamF2YWxpbi9qYXZhbGluLW9wZW5hcGkvdHJlZS81LngpICAgfCA1LnggICAgICAgICAgICAgfCA1LnggICAgICAgICAgICAgfCBKREsxMSAgICAgICAgfAp8IFs0LnhdKGh0dHBzOi8vZ2l0aHViLmNvbS9qYXZhbGluL2phdmFsaW4tb3BlbmFwaS90cmVlLzQueCkgICB8IDQueCAgICAgICAgICAgICB8IDEueCAgICAgICAgICAgICB8IEpESzggICAgICAgICB8Cg== readmeEtag: '"fc0e52845b3780f38287e9aa78acc1b14220a87d"' readmeLastModified: Sun, 21 Jul 2024 14:21:45 GMT repositoryId: 364314288 description: >- Annotation processor for compile-time OpenAPI & JsonSchema, with out-of-the-box support for Javalin 5.x & 6.x, Swagger & ReDoc created: '2021-05-04T16:10:52Z' updated: '2026-01-20T11:46:27Z' language: Kotlin archived: false stars: 60 watchers: 5 forks: 22 owner: javalin logo: https://avatars.githubusercontent.com/u/28701086?v=4 license: Apache-2.0 repoEtag: '"d513a9de4a69650a5f138d9951da0aa95a83c63f8ae7096fe853cfa8e09826f2"' repoLastModified: Tue, 20 Jan 2026 11:46:27 GMT category: SDK foundInMaster: true oldLocations: - https://github.com/dzikoysk/javalin-openapi - https://github.com/reposilite-playground/javalin-openapi - source: openapi3 tags repository: https://github.com/juliacomputing/openapi.jl v3: true id: daf8fe2008e1ce6b16d45cb87d8bbb50 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9KdWxpYUNvbXB1dGluZy9PcGVuQVBJLmpsL3dvcmtmbG93cy9DSS9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vSnVsaWFDb21wdXRpbmcvT3BlbkFQSS5qbC9hY3Rpb25zP3F1ZXJ5PXdvcmtmbG93JTNBQ0krYnJhbmNoJTNBbWFpbikKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL0p1bGlhQ29tcHV0aW5nL09wZW5BUEkuamwvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPWlaZUZMN0pzMGwpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvSnVsaWFDb21wdXRpbmcvT3BlbkFQSS5qbCkKClRoaXMgaXMgdGhlIEp1bGlhIGxpYnJhcnkgbmVlZGVkIGFsb25nIHdpdGggY29kZSBnZW5lcmF0ZWQgYnkgdGhlIFtPcGVuQVBJIGdlbmVyYXRvcl0oaHR0cHM6Ly9vcGVuYXBpLWdlbmVyYXRvci50ZWNoLykgdG8gaGVscCBkZWZpbmUsIHByb2R1Y2UgYW5kIGNvbnN1bWUgT3BlbkFQSSBpbnRlcmZhY2VzLgoKWyFbXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2RvY3MtbGF0ZXN0LWJsdWUuc3ZnKV0oaHR0cHM6Ly9KdWxpYUNvbXB1dGluZy5naXRodWIuaW8vT3BlbkFQSS5qbCkKCiMjIFF1aWNrIEd1aWRlCgotIENyZWF0ZSBhbiBBUEkgc3BlY2lmaWNhdGlvbi4gQ2hlY2sgb3V0IFtPcGVuQVBJLVNwZWNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKSBmb3Igc3BlY2lmaWNhdGlvbiBzeW50YXggYW5kIGV4YW1wbGVzLgotIFVzZSBbaW5zdHJ1Y3Rpb25zXShodHRwczovL29wZW5hcGktZ2VuZXJhdG9yLnRlY2gvZG9jcy9nZW5lcmF0b3JzKSBwcm92aWRlZCBmb3IgdGhlIEp1bGlhIE9wZW5BUEkgY29kZSBnZW5lcmF0b3IgcGx1Z2luIHRvIGdlbmVyYXRlIEp1bGlhIGNvZGUuCi0gQ2xpZW50OgogICAgLSBVc2UgdGhlIGdlbmVyYXRlZCBjbGllbnQgaW4gSnVsaWEgZGlyZWN0bHkgdG8gaW52b2tlIEFQSXMKLSBTZXJ2ZXI6CiAgICAtIFByb3ZpZGUgY29kZSB0byBoYW5kbGUgQVBJIGludm9jYXRpb25zIG9uIHRoZSBzZXJ2ZXIgc2lkZSBieSBpbXBsZW1lbnRpbmcgdGhlIEp1bGlhIG1ldGhvZHMgY29ycmVzcG9uZGluZyB0byBBUEkgc3R1YnMuCiAgICAtIFN0YXJ0IGEgc2VydmVyIHVzaW5nIEhUVFAuamwgYW5kIHJlZ2lzdGVyIHRoZSBnZW5lcmF0ZWQgcmVxdWVzdCBoYW5kbGVycy4KCiMjIEV4YW1wbGVzCgpUaGUgUGV0c3RvcmUgaXMgYSBjb21tb24gZXhhbXBsZSB0aGF0IG1vc3QgT3BlbkFQSSBpbXBsZW1lbnRhdGlvbnMgdXNlIHRvIHRlc3QgYW5kIGRlbW9uc3RyYXRlLiBDbGllbnRzIGFuZCBzZXJ2ZXJzIGdlbmVyYXRlZCBmcm9tIGJvdGggdmVyc2lvbiAyIGFuZCAzIHNwZWNpZmljYXRpb25zIGFyZSBpbmNsdWRlZCBpbiB0aGlzIHJlcG8uCgotIFBldHN0b3JlIHYyOgogICAgLSBDbGllbnQ6IFtkb2NzXSh0ZXN0L2NsaWVudC9wZXRzdG9yZV92Mi9wZXRzdG9yZS9SRUFETUUubWQpLCBbaW1wbGVtZW50YXRpb25dKHRlc3QvY2xpZW50L3BldHN0b3JlX3YyKQogICAgLSBTZXJ2ZXI6IFtkb2NzXSh0ZXN0L3NlcnZlci9wZXRzdG9yZV92Mi9wZXRzdG9yZS9SRUFETUUubWQpLCBbaW1wbGVtZW50YXRpb25dKHRlc3Qvc2VydmVyL3BldHN0b3JlX3YyKQotIFBldHN0b3JlIHYzOgogICAgLSBDbGllbnQ6IFtkb2NzXSh0ZXN0L2NsaWVudC9wZXRzdG9yZV92My9wZXRzdG9yZS9SRUFETUUubWQpLCBbaW1wbGVtZW50YXRpb25dKHRlc3QvY2xpZW50L3BldHN0b3JlX3YzKQogICAgLSBTZXJ2ZXI6IFtkb2NzXSh0ZXN0L3NlcnZlci9wZXRzdG9yZV92My9wZXRzdG9yZS9SRUFETUUubWQpLCBbaW1wbGVtZW50YXRpb25dKHRlc3Qvc2VydmVyL3BldHN0b3JlX3YzKQo= readmeEtag: '"267b6741436f2490b6fe88033bf5824947d7b9d0"' readmeLastModified: Fri, 26 Jul 2024 07:24:14 GMT repositoryId: 535564203 description: OpenAPI helper and code generator for Julia created: '2022-09-12T07:55:03Z' updated: '2026-01-23T12:46:48Z' language: Julia archived: false stars: 56 watchers: 24 forks: 13 owner: JuliaComputing logo: https://avatars.githubusercontent.com/u/10258468?v=4 license: NOASSERTION repoEtag: '"c2ca58afecb35cd129c1dd8dfda62c0eaaad8a33387a61ca771d3e55e13bc715"' repoLastModified: Fri, 23 Jan 2026 12:46:48 GMT foundInMaster: true category: Parsers - source: openapi3 tags repository: https://github.com/mossaabfrifita/spring-boot-4-security-7-jwt v3: true id: 8f25bb636638af5ab1f70138e6f2fa7c repositoryMetadata: base64Readme: >- # 🔐 Spring Security JWT Authentication & Authorization

A comprehensive Spring Boot application demonstrating JWT-based authentication and role-based authorization with refresh token support. This project showcases modern security practices using Spring Security 7 and Spring Boot 4.

## ✨ Features

- 🔑 **JWT Authentication** - Secure token-based authentication
- 🔄 **Refresh Token Support** - Long-lived refresh tokens stored in database
- 👥 **User Registration & Login** - Complete authentication flow
- 🛡️ **Role-Based Access Control (RBAC)** - Fine-grained authorization with roles and privileges
- 🍪 **HTTP-Only Cookies** - Secure token storage in cookies
- 📝 **OpenAPI/Swagger Documentation** - Interactive API documentation
- ✅ **Password Validation** - Strong password requirements
- 🚫 **Custom Error Handling** - Customized access denied and unauthorized handlers
- 🔒 **Spring Security Integration** - Full Spring Security configuration

## 🛠️ Tech Stack

- **Java 17**
- **Spring Boot 4.0.1**
- **Spring Security 7**
- **Spring Data JPA**
- **PostgreSQL** - Database
- **JWT (jjwt 0.11.5)** - JSON Web Token implementation
- **Maven** - Build tool
- **Lombok** - Boilerplate reduction
- **OpenAPI/SpringDoc** - API documentation
- **Bean Validation** - Request validation

## 📋 Prerequisites

Before you begin, ensure you have the following installed:

- **JDK 17** or higher
- **Maven 3.6+**
- **PostgreSQL 12+**
- **Git** (for cloning the repository)

## 🚀 Getting Started

### 1. Clone the Repository

```bash
git clone https://github.com/yourusername/security.git
cd security
```

### 2. Database Setup

Create a PostgreSQL database:

```sql
CREATE DATABASE db_security;
```

### 3. Configuration

Update the `application.yml` file with your database credentials:

```yaml
spring:
  datasource:
    url: jdbc:postgresql://localhost:5432/db_security
    username: postgres
    password: ${POSTGRES_PASSWORD}  # Set this environment variable
```

**Important:** Set the `POSTGRES_PASSWORD` environment variable with your PostgreSQL password:

**Windows (PowerShell):**
```powershell
$env:POSTGRES_PASSWORD="your_password"
```

**Windows (CMD):**
```cmd
set POSTGRES_PASSWORD=your_password
```

**Linux/Mac:**
```bash
export POSTGRES_PASSWORD=your_password
```

### 4. Build the Project

```bash
mvn clean install
```

### 5. Run the Application

```bash
mvn spring-boot:run
```

Or using the Maven wrapper:

```bash
./mvnw spring-boot:run
```

The application will start on **http://localhost:8086**

## 📚 API Documentation

Once the application is running, access the Swagger UI for interactive API documentation:

```
http://localhost:8086/swagger-ui.html
```

## 🔌 API Endpoints

### Authentication Endpoints

| Method | Endpoint | Description | Auth Required |
|--------|----------|-------------|---------------|
| `POST` | `/api/v1/auth/register` | Register a new user | ❌ |
| `POST` | `/api/v1/auth/authenticate` | Login and get JWT tokens | ❌ |
| `POST` | `/api/v1/auth/refresh-token` | Refresh access token | ❌ |
| `POST` | `/api/v1/auth/refresh-token-cookie` | Refresh token via cookie | ❌ |
| `POST` | `/api/v1/auth/logout` | Logout and invalidate tokens | ❌ |
| `GET` | `/api/v1/auth/info` | Get authentication info | ❌ |

### Authorization Endpoints

| Method | Endpoint | Description | Required Role | Required Privilege |
|--------|----------|-------------|---------------|-------------------|
| `GET` | `/api/v1/admin/resource` | Admin read resource | `ADMIN` | `READ_PRIVILEGE` |
| `DELETE` | `/api/v1/admin/resource` | Admin delete resource | `ADMIN` | `DELETE_PRIVILEGE` |
| `POST` | `/api/v1/user/resource` | User create resource | `ADMIN` or `USER` | `WRITE_PRIVILEGE` |
| `PUT` | `/api/v1/user/resource` | User update resource | `ADMIN` or `USER` | `UPDATE_PRIVILEGE` |

## 💡 Usage Examples

### Register a New User

```bash
curl -X POST http://localhost:8086/api/v1/auth/register \
  -H "Content-Type: application/json" \
  -d '{
    "firstname": "John",
    "lastname": "Doe",
    "email": "john.doe@example.com",
    "password": "SecurePass123!"
  }'
```

### Authenticate (Login)

```bash
curl -X POST http://localhost:8086/api/v1/auth/authenticate \
  -H "Content-Type: application/json" \
  -d '{
    "email": "john.doe@example.com",
    "password": "SecurePass123!"
  }'
```

**Response:**
```json
{
  "accessToken": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9...",
  "refreshToken": "eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9...",
  "tokenType": "Bearer"
}
```

Tokens are also set as HTTP-only cookies automatically.

### Access Protected Resource

```bash
curl -X GET http://localhost:8086/api/v1/admin/resource \
  -H "Authorization: Bearer YOUR_ACCESS_TOKEN"
```

Or if using cookies (set automatically during login):

```bash
curl -X GET http://localhost:8086/api/v1/admin/resource \
  --cookie "jwt-cookie=YOUR_JWT_TOKEN"
```

### Refresh Token

```bash
curl -X POST http://localhost:8086/api/v1/auth/refresh-token \
  -H "Content-Type: application/json" \
  -d '{
    "refreshToken": "YOUR_REFRESH_TOKEN"
  }'
```

## 🏗️ Project Structure

```
src/
├── main/
│   ├── java/fr/mossaab/security/
│   │   ├── config/              # Security configuration classes
│   │   │   ├── SecurityConfiguration.java
│   │   │   ├── ApplicationSecurityConfig.java
│   │   │   ├── JwtAuthenticationFilter.java
│   │   │   ├── CustomAccessDeniedHandler.java
│   │   │   └── Http401UnauthorizedEntryPoint.java
│   │   ├── controller/          # REST controllers
│   │   │   ├── AuthenticationController.java
│   │   │   └── AuthorizationController.java
│   │   ├── entities/            # JPA entities
│   │   │   ├── User.java
│   │   │   └── RefreshToken.java
│   │   ├── enums/               # Enumerations
│   │   │   ├── Role.java
│   │   │   ├── Privilege.java
│   │   │   └── TokenType.java
│   │   ├── service/             # Business logic
│   │   │   ├── AuthenticationService.java
│   │   │   ├── JwtService.java
│   │   │   ├── RefreshTokenService.java
│   │   │   └── impl/            # Service implementations
│   │   ├── repository/          # Data access layer
│   │   │   ├── UserRepository.java
│   │   │   └── RefreshTokenRepository.java
│   │   ├── payload/             # DTOs
│   │   │   ├── request/
│   │   │   └── response/
│   │   ├── validation/          # Custom validators
│   │   ├── exception/           # Custom exceptions
│   │   └── handlers/            # Exception handlers
│   └── resources/
│       └── application.yml      # Application configuration
└── test/                        # Test files
```

## 🔒 Security Features

### Roles and Privileges

The application implements a role-based access control system:

**Roles:**
- `ADMIN` - Full access with all privileges
- `USER` - Limited access with read and write privileges

**Privileges:**
- `READ_PRIVILEGE` - Read access
- `WRITE_PRIVILEGE` - Create access
- `UPDATE_PRIVILEGE` - Update access
- `DELETE_PRIVILEGE` - Delete access

### Token Configuration

- **Access Token Expiration:** 15 minutes (900,000 ms)
- **Refresh Token Expiration:** 15 days (1,296,000,000 ms)
- **Token Storage:** HTTP-only cookies for enhanced security

### Password Requirements

The application enforces strong password validation. Ensure your password meets the requirements defined in the `StrongPassword` validator.

## ⚙️ Configuration

### JWT Configuration

Configure JWT settings in `application.yml`:

```yaml
application:
  security:
    jwt:
      secret-key: YOUR_SECRET_KEY  # Change this in production!
      expiration: 900000  # 15 minutes
      cookie-name: jwt-cookie
      refresh-token:
        expiration: 1296000000  # 15 days
        cookie-name: refresh-jwt-cookie
```

**⚠️ Security Note:** Always change the `secret-key` in production environments!

## 🧪 Testing

Run the test suite:

```bash
mvn test
```

## 📝 License

This project is open source and available under the [MIT License](LICENSE).

## 🤝 Contributing

Contributions, issues, and feature requests are welcome! Feel free to check the [issues page](https://github.com/yourusername/security/issues).

1. Fork the repository
2. Create your feature branch (`git checkout -b feature/AmazingFeature`)
3. Commit your changes (`git commit -m 'Add some AmazingFeature'`)
4. Push to the branch (`git push origin feature/AmazingFeature`)
5. Open a Pull Request


---

⭐ If you find this project helpful, please consider giving it a star!
 readmeEtag: '"791b78a6131227df851b67b9fb290bd992741fa5"' readmeLastModified: Sat, 24 Jan 2026 20:13:38 GMT repositoryId: 659193077 description: >- Spring Boot 4 & Spring Security 7 : Token Based Authentication example with JWT, Authorization, Spring Data & PostgreSQL created: '2023-06-27T10:15:47Z' updated: '2026-01-31T06:29:29Z' language: Java archived: false stars: 65 watchers: 2 forks: 20 owner: MossaabFrifita logo: https://avatars.githubusercontent.com/u/34568592?v=4 repoEtag: '"84691f59f5a22b6222b62045282fcbc0351b04d6ee23a0dc60b0856b199bcb02"' repoLastModified: Sat, 31 Jan 2026 06:29:29 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/mossaabfrifita/spring-boot-3-security-6-jwt - source: - openapi3 tags - openapi31 tags repository: https://github.com/swaggerexpert/swagger-editor-validate v3: true v3_1: true id: cd216d2979394961a0ed7aaffafe46a4 repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIEVkaXRvciBWYWxpZGF0ZSBHaXRodWIgQWN0aW9uCgpUaGlzIEdpdEh1YiBBY3Rpb24gdmFsaWRhdGVzIE9wZW5BUEkgKE9BUykgZGVmaW5pdGlvbiBmaWxlIHVzaW5nIFtTd2FnZ2VyIEVkaXRvcl0oaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby8pLgoKIVt0ZXN0XShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS8xOTMyODYvMTEwMjQ2MTQ4LWI3YWVhNTgwLTdmNjYtMTFlYi05YmQ3LTk0MGVjZTFmYzZiMC5wbmcpCgpJdCdzIGhhbmR5IGZvciB1c2UtY2FzZXMgd2hlbiBPQVMgZGVmaW5pdGlvbiBpcyBtYWludGFpbmVkIG1hbnVhbGx5IGFuZCBpcyBjaGVja2VkIHdpdGhpbiB0aGUgZ2l0LgpJZiBtb2RpZmljYXRpb25zIGFyZSBtYWRlIHRvIHRoZSBPQVMgZGVmaW5pdGlvbiB5b3Ugd2FudCB0byBtYWtlIHN1cmUgdGhhdCB0aGVyZSBhcmUgbm8gZXJyb3JzCmludHJvZHVjZWQgYnkgbW9kaWZpY2F0aW9ucy4gRXJyb3JzIHRoYXQgd291bGQgYXBwZWFyIGluIFN3YWdnZXIgRWRpdG9yIHdoZW4gdGhlIE9BUyBkZWZpbml0aW9uCndvdWxkIGJlIHBhc3RlZCBpbnNpZGUgaXQuCgohW2ltYWdlXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS8xOTMyODYvMTEwMjQ0NjE4LWRjZWJlNTgwLTdmNWYtMTFlYi04ZGQ4LWNiMzFmNDk5NzYxZS5wbmcpCgpJZiB5b3UncmUgaW50ZXJlc3RlZCBhYm91dCB0ZWNobmljYWwgZGVzaWduIGFuZCBldm9sdXRpb24gb2YgdGhpcyBHaXRIdWIgQWN0aW9uIHBsZWFzZQpyZWFkIG91ciBbcmVsZWFzZSBhcnRpY2xlXShodHRwczovL3ZsYWRpbWlyZ29yZWouY29tL2Jsb2cvaG93LXRvLXZhbGlkYXRlLW9wZW5hcGktZGVmaW5pdGlvbnMtaW4tc3dhZ2dlci1lZGl0b3ItdXNpbmctZ2l0aHViLWFjdGlvbnMvKS4KCiMjIElucHV0cwoKIyMjIGBzd2FnZ2VyLWVkaXRvci11cmxgCgoqKk9wdGlvbmFsKiogRGVmaW5lcyBVUkwgb2YgW3N3YWdnZXItZWRpdG9yXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9zd2FnZ2VyLWVkaXRvcikgd2hlcmUgZGVmaW5pdGlvbgpmaWxlIGlzIGdvaW5nIHRvIGJlIHZhbGlkYXRlZC4gRGVmYXVsdCBgaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby9gLgoKIyMjIGBkZWZpbml0aW9uLWZpbGVgCgoqKlJlcXVpcmVkKiogRGVmaW5lcyBwYXRoIG9mIFtPQVNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKSBkZWZpbml0aW9uIGZpbGUgdGhhdCBleGlzdHMKYXMgYSBwaHlzaWNhbCBmaWxlIGluIHlvdXIgcmVwb3NpdG9yeS4KCiMjIyBgaWdub3JlLWVycm9yYAoKKipPcHRpb25hbCoqIERlZmluZXMgcGF0aCB0byBKYXZhU2NyaXB0IGZpbGUgY29udGFpbmluZyBwcmVkaWNhdGUgZm9yIGRldGVybWluaW5nIGlmIHRoZSBlcnJvciBzaG91bGQgYmUgaWdub3JlZCBvciBub3QuCgoKIyMjIGBkZWZhdWx0LXRpbWVvdXRgCgoqKk9wdGlvbmFsKiogRGVmaW5lcyBtYXhpbXVtIHRpbWUgaW4gbWlsbGlzZWNvbmRzIGEgc2NyaXB0IHdhaXRzIGZvciBjZXJ0YWluIGFjdGlvbnMgb3IgZXZlbnRzIHRvIG9jY3VyLgoKCiMjIEV4YW1wbGUgdXNhZ2UKClRoZXJlIGFyZSB0d28gbWFqb3IgdXNlLWNhc2VzIG9mIGhvdyB0byB1c2UgdGhpcyBHaXRIdWIgQWN0aW9uLgoKIyMjIFB1YmxpYyB1c2UtY2FzZQoKSWYgeW91IGhhdmUgYWNjZXNzIHRvIHRoZSBpbnRlcm5ldCBhbmQgZG9uJ3QgbWluZCB0aGF0IHRoaXMgYWN0aW9uIHNlbmRzIHlvdXIgT0FTIGRlZmluaXRpb24KdG8gaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby8gZm9yIHZhbGlkYXRpb24uCgpgYGB5YW1sCm9uOiBbcHVzaF0KCmpvYnM6CiAgdGVzdF9zd2FnZ2VyX2VkaXRvcl92YWxpZGF0b3JfcmVtb3RlOgogICAgcnVucy1vbjogdWJ1bnR1LWxhdGVzdAogICAgbmFtZTogU3dhZ2dlciBFZGl0b3IgVmFsaWRhdG9yIFJlbW90ZQoKICAgIHN0ZXBzOgogICAgICAtIHVzZXM6IGFjdGlvbnMvY2hlY2tvdXRAdjQKICAgICAgLSBuYW1lOiBWYWxpZGF0ZSBPcGVuQVBJIGRlZmluaXRpb24KICAgICAgICB1c2VzOiBzd2FnZ2VyZXhwZXJ0L3N3YWdnZXItZWRpdG9yLXZhbGlkYXRlQHYxCiAgICAgICAgd2l0aDoKICAgICAgICAgIGRlZmluaXRpb24tZmlsZTogZXhhbXBsZXMvb3BlbmFwaS0yLTAueWFtbApgYGAKCiMjIyBQcml2YXRlIHVzZS1jYXNlCgpJZiB5b3Ugd2FudCB0byBtYWludGFpbiBjb21wbGV0ZSBwcml2YWN5IGFuZCB5b3VyIE9BUyBkZWZpbml0aW9uIG1heSBjb250YWluCnNlbnNpdGl2ZSBpbmZvcm1hdGlvbiB1c2UgdGhlIGZvbGxvd2luZyB3b3JrZmxvdy4gVGhlIHdvcmtmbG93IHVzZXMgc3dhZ2dlci1lZGl0b3IKZG9ja2VyIGltYWdlIHRoYXQgcnVucyBhcyBzZXJ2aWNlIG9mIHRoZSB3b3JrZmxvdy4KCmBgYHlhbWwKb246IFtwdXNoXQoKam9iczoKICB0ZXN0X3N3YWdnZXJfZWRpdG9yX3ZhbGlkYXRvcl9zZXJ2aWNlOgogICAgcnVucy1vbjogdWJ1bnR1LWxhdGVzdAogICAgbmFtZTogU3dhZ2dlciBFZGl0b3IgVmFsaWRhdG9yIFNlcnZpY2UKCiAgICAjIFNlcnZpY2UgY29udGFpbmVycyB0byBydW4gd2l0aCBgcnVubmVyLWpvYmAKICAgIHNlcnZpY2VzOgogICAgICAjIExhYmVsIHVzZWQgdG8gYWNjZXNzIHRoZSBzZXJ2aWNlIGNvbnRhaW5lcgogICAgICBzd2FnZ2VyLWVkaXRvcjoKICAgICAgICAjIERvY2tlciBIdWIgaW1hZ2UKICAgICAgICBpbWFnZTogc3dhZ2dlcmFwaS9zd2FnZ2VyLWVkaXRvcgogICAgICAgIHBvcnRzOgogICAgICAgICAgIyBNYXBzIHBvcnQgODA4MCBvbiBzZXJ2aWNlIGNvbnRhaW5lciB0byB0aGUgaG9zdCA4MAogICAgICAgICAgLSA4MDo4MDgwCgogICAgc3RlcHM6CiAgICAgIC0gdXNlczogYWN0aW9ucy9jaGVja291dEB2NAogICAgICAtIG5hbWU6IFZhbGlkYXRlIE9wZW5BUEkgZGVmaW5pdGlvbgogICAgICAgIHVzZXM6IHN3YWdnZXJleHBlcnQvc3dhZ2dlci1lZGl0b3ItdmFsaWRhdGVAdjEKICAgICAgICB3aXRoOgogICAgICAgICAgc3dhZ2dlci1lZGl0b3ItdXJsOiBodHRwOi8vbG9jYWxob3N0LwogICAgICAgICAgZGVmaW5pdGlvbi1maWxlOiBleGFtcGxlcy9vcGVuYXBpLTItMC55YW1sCiAgICAgICAgICBkZWZhdWx0LXRpbWVvdXQ6IDIwMDAwCmBgYAo= readmeEtag: '"25280be4d237a47125cf5817a64d7df75fc251b2"' readmeLastModified: Thu, 19 Jun 2025 05:06:53 GMT repositoryId: 344901461 description: >- This GitHub Actions validates OpenAPI (OAS) definition file using Swagger Editor. created: '2021-03-05T18:37:29Z' updated: '2025-12-01T04:29:37Z' language: JavaScript archived: false stars: 55 watchers: 2 forks: 15 owner: swaggerexpert logo: https://avatars.githubusercontent.com/u/172408630?v=4 license: BSD-3-Clause repoEtag: '"b4e26e0a89201113a0d45e206caa5ad96885c8414df93e187c372090a9299cc9"' repoLastModified: Mon, 01 Dec 2025 04:29:37 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/jdkandersson/openalchemy v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQWxjaGVteQoKIVtDb2RlIFF1YWxpdHkgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vamRrYW5kZXJzc29uL09wZW5BbGNoZW15L3dvcmtmbG93cy9Db2RlJTIwcXVhbGl0eSUyMGNoZWNrcy9iYWRnZS5zdmcpCiFbQXp1cmUgRGV2T3BzIGNvdmVyYWdlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2F6dXJlLWRldm9wcy9jb3ZlcmFnZS9hbmRlcnNzb25wdWJsaWMvYW5kZXJzc29ucHVibGljLzEpClshW0RvY3VtZW50YXRpb24gU3RhdHVzXShodHRwczovL3JlYWR0aGVkb2NzLm9yZy9wcm9qZWN0cy9vcGVuYXBpLXNxbGFsY2hlbXkvYmFkZ2UvP3ZlcnNpb249bGF0ZXN0KV0oaHR0cHM6Ly9vcGVuYXBpLXNxbGFsY2hlbXkucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0Lz9iYWRnZT1sYXRlc3QpCiFbQ29kZSBDbGltYXRlIG1haW50YWluYWJpbGl0eV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9jb2RlY2xpbWF0ZS9tYWludGFpbmFiaWxpdHkvamRrYW5kZXJzc29uL09wZW5BbGNoZW15KQohW0NvZGUgQ2xpbWF0ZSB0ZWNobmljYWwgZGVidF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9jb2RlY2xpbWF0ZS90ZWNoLWRlYnQvamRrYW5kZXJzc29uL09wZW5BbGNoZW15KQohW0xHVE0gR3JhZGVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbGd0bS9ncmFkZS9weXRob24vZ2l0aHViL2pka2FuZGVyc3Nvbi9PcGVuQWxjaGVteSkKClRyYW5zbGF0ZXMgYW4gT3BlbkFQSSBzY2hlbWEgdG8gU1FMQWxjaGVteSBtb2RlbHMuCgpTdXBwb3J0cyBPcGVuQVBJIDMuMCBhbmQgMy4xLgoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCnB5dGhvbiAtbSBwaXAgaW5zdGFsbCBPcGVuQWxjaGVteQojIFRvIGJlIGFibGUgdG8gbG9hZCBZQU1MIGZpbGUKcHl0aG9uIC1tIHBpcCBpbnN0YWxsIE9wZW5BbGNoZW15W3lhbWxdCmBgYAoKIyMgRXhhbXBsZQoKRm9yIGV4YW1wbGUsIGdpdmVuIHRoZSBmb2xsb3dpbmcgT3BlbkFQSSBzcGVjaWZpY2F0aW9uOgoKYGBgeWFtbAojIC4vZXhhbXBsZXMvc2ltcGxlL2V4YW1wbGUtc3BlYy55bWwKb3BlbmFwaTogIjMuMC4wIgoKaW5mbzoKICB0aXRsZTogVGVzdCBTY2hlbWEKICBkZXNjcmlwdGlvbjogQVBJIHRvIGlsbHVzdHJhdGUgT3BlbkFsY2hlbXkgTVZQLgogIHZlcnNpb246ICIwLjEiCgpwYXRoczoKICAvZW1wbG95ZWU6CiAgICBnZXQ6CiAgICAgIHN1bW1hcnk6IFVzZWQgdG8gcmV0cmlldmUgYWxsIGVtcGxveWVlcy4KICAgICAgcmVzcG9uc2VzOgogICAgICAgIDIwMDoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBSZXR1cm4gYWxsIGVtcGxveWVlcyBmcm9tIHRoZSBkYXRhYmFzZS4KICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgICAgICAgIGl0ZW1zOgogICAgICAgICAgICAgICAgICAiJHJlZiI6ICIjL2NvbXBvbmVudHMvc2NoZW1hcy9FbXBsb3llZSIKCmNvbXBvbmVudHM6CiAgc2NoZW1hczoKICAgIEVtcGxveWVlOgogICAgICBkZXNjcmlwdGlvbjogUGVyc29uIHRoYXQgd29ya3MgZm9yIGEgY29tcGFueS4KICAgICAgdHlwZTogb2JqZWN0CiAgICAgIHgtdGFibGVuYW1lOiBlbXBsb3llZQogICAgICBwcm9wZXJ0aWVzOgogICAgICAgIGlkOgogICAgICAgICAgdHlwZTogaW50ZWdlcgogICAgICAgICAgZGVzY3JpcHRpb246IFVuaXF1ZSBpZGVudGlmaWVyIGZvciB0aGUgZW1wbG95ZWUuCiAgICAgICAgICBleGFtcGxlOiAwCiAgICAgICAgICB4LXByaW1hcnkta2V5OiB0cnVlCiAgICAgICAgICB4LWF1dG9pbmNyZW1lbnQ6IHRydWUKICAgICAgICBuYW1lOgogICAgICAgICAgdHlwZTogc3RyaW5nCiAgICAgICAgICBkZXNjcmlwdGlvbjogVGhlIG5hbWUgb2YgdGhlIGVtcGxveWVlLgogICAgICAgICAgZXhhbXBsZTogRGF2aWQgQW5kZXJzc29uCiAgICAgICAgICB4LWluZGV4OiB0cnVlCiAgICAgICAgZGl2aXNpb246CiAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgIGRlc2NyaXB0aW9uOiBUaGUgcGFydCBvZiB0aGUgY29tcGFueSB0aGUgZW1wbG95ZWUgd29ya3MgaW4uCiAgICAgICAgICBleGFtcGxlOiBFbmdpbmVlcmluZwogICAgICAgICAgeC1pbmRleDogdHJ1ZQogICAgICAgIHNhbGFyeToKICAgICAgICAgIHR5cGU6IG51bWJlcgogICAgICAgICAgZGVzY3JpcHRpb246IFRoZSBhbW91bnQgb2YgbW9uZXkgdGhlIGVtcGxveWVlIGlzIHBhaWQuCiAgICAgICAgICBleGFtcGxlOiAxMDAwMDAwLjAwCiAgICAgIHJlcXVpcmVkOgogICAgICAgIC0gaWQKICAgICAgICAtIG5hbWUKICAgICAgICAtIGRpdmlzaW9uCmBgYAoKVGhlIFNRTEFMY2hlbXkgbW9kZWxzIGZpbGUgdGhlbiBiZWNvbWVzOgoKYGBgcHl0aG9uCiMgbW9kZWxzLnB5CmZyb20gb3Blbl9hbGNoZW15IGltcG9ydCBpbml0X3lhbWwKCmluaXRfeWFtbCgiLi9leGFtcGxlcy9zaW1wbGUvZXhhbXBsZS1zcGVjLnltbCIpCmBgYAoKVGhlIF9CYXNlXyBhbmQgX0VtcGxveWVlXyBvYmplY3RzIGNhbiBiZSBhY2Nlc3NlZDoKCmBgYHB5dGhvbgpmcm9tIG9wZW5fYWxjaGVteS5tb2RlbHMgaW1wb3J0IEJhc2UKZnJvbSBvcGVuX2FsY2hlbXkubW9kZWxzIGltcG9ydCBFbXBsb3llZQpgYGAKCldpdGggdGhlIF9tb2RlbHNfZmlsZW5hbWVfIHBhcmFtZXRlciBhIGZpbGUgaXMgYXV0byBnZW5lcmF0ZWQgd2l0aCB0eXBlIGhpbnRzCmZvciB0aGUgU1FMQWxjaGVteSBtb2RlbHMgYXQgdGhlIHNwZWNpZmllZCBsb2NhdGlvbiwgZm9yIGV4YW1wbGU6Clt0eXBlIGhpbnRlZCBtb2RlbHMgZXhhbXBsZV0oZXhhbXBsZXMvc2ltcGxlL21vZGVsc19hdXRvLnB5KS4gVGhpcyBhZGRzIHN1cHBvcnQKZm9yIElERSBhdXRvIGNvbXBsZXRlLCBmb3IgZXhhbXBsZSBmb3IgdGhlIG1vZGVsIGluaXRpYWxpemF0aW9uOgoKIVthdXRvY29tcGxldGUgaW5pdF0oZXhhbXBsZXMvc2ltcGxlL21vZGVsc19hdXRvY29tcGxldGVfaW5pdC5wbmcpCgphbmQgZm9yIHByb3BlcnRpZXMgYW5kIG1ldGhvZHMgYXZhaWxhYmxlIG9uIGFuIGluc3RhbmNlOgoKIVthdXRvY29tcGxldGUgaW5zdGFuY2VdKGV4YW1wbGVzL3NpbXBsZS9tb2RlbHNfYXV0b2NvbXBsZXRlX2luc3RhbmNlLnBuZykKCkFuIGV4dGVuc2l2ZSBzZXQgb2YgZXhhbXBsZXMgd2l0aCBhIHJhbmdlIG9mIGZlYXR1cmVzIGlzIGhlcmU6CgpbZXhhbXBsZXMgZm9yIG1haW4gZmVhdHVyZXNdKGV4YW1wbGVzKQoKQW4gZXhhbXBsZSBBUEkgaGFzIGJlZW4gZGVmaW5lZCB1c2luZyBjb25uZXhpb24gYW5kIEZsYXNrIGhlcmU6CgpbZXhhbXBsZSBjb25uZXhpb24gYXBwXShleGFtcGxlcy9hcHApCgojIyBEb2N1bWVudGF0aW9uCgpbUmVhZCB0aGUgRG9jc10oaHR0cHM6Ly9vcGVuYXBpLXNxbGFsY2hlbXkucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0LykKCiMjIEJ1eSBtZSBhIGNvZmZlZQoKWyFbQnV5IE1lIEEgQ29mZmVlXShodHRwczovL2Nkbi5idXltZWFjb2ZmZWUuY29tL2J1dHRvbnMvdjIvZGVmYXVsdC15ZWxsb3cucG5nKV0oaHR0cHM6Ly93d3cuYnV5bWVhY29mZmVlLmNvbS9qZGthbmRlcnNzb24pCgojIyBGZWF0dXJlcwoKLSBpbml0aWFsaXppbmcgZnJvbSBKU09OLAotIGluaXRpYWxpemluZyBmcm9tIFlBTUwsCi0gYnVpbGQgYSBwYWNrYWdlIHdpdGggdGhlIG1vZGVscyBmb3IgZGlzdHJpYnV0aW9uLCBwYWNrYWdlZCBhcyBzZGlzdCBvciB3aGVlbCwKLSBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlIGEgbW9kZWxzIGZpbGUsCi0gYGludGVnZXJgICgzMiBhbmQgNjQgYml0KSwKLSBgbnVtYmVyYCAoZmxvYXQgb25seSksCi0gYGJvb2xlYW5gLAotIGBzdHJpbmdgLAotIGBwYXNzd29yZGAsCi0gYGJ5dGVgLAotIGBiaW5hcnlgLAotIGBkYXRlYCwKLSBgZGF0ZS10aW1lYCwKLSBnZW5lcmljIEpTT04gZGF0YSwKLSBgJHJlZmAgcmVmZXJlbmNlcyBmb3IgY29sdW1ucyBhbmQgbW9kZWxzLAotIHJlbW90ZSBgJHJlZmAgdG8gb3RoZXIgZmlsZXMgb24gdGhlIHNhbWUgZmlsZSBzeXN0ZW0KICAoX25vdCBzdXBwb3J0ZWQgb24gV2luZG93c18pLAotIHJlbW90ZSBgJHJlZmAgdG8gb3RoZXIgZmlsZXMgYXQgYSBVUkwsCi0gcHJpbWFyeSBrZXlzLAotIGF1dG8gaW5jcmVtZW50aW5nLAotIGluZGV4ZXMsCi0gY29tcG9zaXRlIGluZGV4ZXMsCi0gdW5pcXVlIGNvbnN0cmFpbnRzLAotIGNvbXBvc2l0ZSB1bmlxdWUgY29uc3RyYWludHMsCi0gY29sdW1uIG51bGxhYmlsaXR5LAotIGZvcmVpZ24ga2V5cywKLSBkZWZhdWx0IHZhbHVlcyBmb3IgY29sdW1ucyAoYm90aCBhcHBsaWNhdGlvbiBhbmQgZGF0YWJhc2Ugc2lkZSksCi0gbWFueSB0byBvbmUgcmVsYXRpb25zaGlwcywKLSBvbmUgdG8gb25lIHJlbGF0aW9uc2hpcHMsCi0gb25lIHRvIG1hbnkgcmVsYXRpb25zaGlwcywKLSBtYW55IHRvIG1hbnkgcmVsYXRpb25zaGlwcywKLSBtYW55IHRvIG1hbnkgcmVsYXRpb25zaGlwcyB3aXRoIGN1c3RvbSBhc3NvY2lhdGlvbiB0YWJsZXMsCi0gY3VzdG9tIGZvcmVpZ24ga2V5cyBmb3IgcmVsYXRpb25zaGlwcywKLSBiYWNrIHJlZmVyZW5jZXMgZm9yIHJlbGF0aW9uc2hpcHMsCi0gYGFsbE9mYCBpbmhlcml0YW5jZSBmb3IgY29sdW1ucyBhbmQgbW9kZWxzLAotIGpvaW5lZCBhbmQgc2luZ2xlIHRhYmxlIGluaGVyaXRhbmNlLAotIGBmcm9tX3N0cmAgbW9kZWwgbWV0aG9kcyB0byBjb25zdHJ1Y3QgZnJvbSBKU09OIHN0cmluZywKLSBgZnJvbV9kaWN0YCBtb2RlbCBtZXRob2RzIHRvIGNvbnN0cnVjdCBmcm9tIGRpY3Rpb25hcmllcywKLSBgdG9fc3RyYCBtb2RlbCBtZXRob2RzIHRvIGNvbnZlcnQgaW5zdGFuY2VzIHRvIEpTT04gc3RyaW5nLAotIGBfX3N0cl9fYCBtb2RlbCBtZXRob2RzIHRvIHN1cHBvcnQgdGhlIHB5dGhvbiBgc3RyYCBmdW5jdGlvbiwKLSBgX19yZXByX19gIG1vZGVsIG1ldGhvZHMgdG8gc3VwcG9ydCB0aGUgcHl0aG9uIGByZXByYCBmdW5jdGlvbiwKLSBgdG9fZGljdGAgbW9kZWwgbWV0aG9kcyB0byBjb252ZXJ0IGluc3RhbmNlcyB0byBkaWN0aW9uYXJpZXMsCi0gYHJlYWRPbmx5YCBhbmQgYHdyaXRlT25seWAgZm9yIGluZmx1ZW5jZSB0aGUgY29udmVyc2lvbiB0byBhbmQgZnJvbQogIGRpY3Rpb25hcmllcywKLSBleHBvc2luZyBjcmVhdGVkIG1vZGVscyB1bmRlciBgb3Blbl9hbGNoZW15Lm1vZGVsc2AgcmVtb3ZpbmcgdGhlIG5lZWQgZm9yCiAgYG1vZGVscy5weWAgZmlsZXMsCi0gYWJpbGl0eSB0byBtaXggaW4gYXJiaXRyYXJ5IGNsYXNzZXMgaW50byBhIG1vZGVsLAotIGNhbiB1c2UgdGhlIHNob3J0IGB4LWAgcHJlZml4IG9yIGEgbmFtZXNwYWNlZCBgeC1vcGVuLWFsY2hlbXktYCBwcmVmaXggZm9yCiAgZXh0ZW5zaW9uIHByb3BlcnRpZXMgYW5kCi0gZ3JvdXBpbmcgbW9kZWxzIGludG8gc2NoZW1hcy4KCiMjIENvbnRyaWJ1dGluZwoKRm9yayBhbmQgY2hlY2tvdXQgdGhlIHJlcG9zaXRvcnkuIFRvIGluc3RhbGw6CgpgYGBiYXNoCnBvZXRyeSBpbnN0YWxsCmBgYAoKVG8gcnVuIHRlc3RzOgoKYGBgYmFzaApwb2V0cnkgcnVuIHB5dGVzdApgYGAKCk1ha2UgeW91ciBjaGFuZ2VzIGFuZCByYWlzZSBhIHB1bGwgcmVxdWVzdC4KCiMjIENvbXBpbGluZyBEb2NzCgpgYGBiYXNoCnBvZXRyeSBzaGVsbApjZCBkb2NzCm1ha2UgaHRtbApgYGAKClRoaXMgY3JlYXRlcyB0aGUgYGluZGV4Lmh0bWxgIGZpbGUgaW4gYGRvY3MvYnVpbGQvaHRtbC9pbmRleC5odG1sYC4KCiMjIFJlbGVhc2UgQ29tbWFuZHMKCmBgYGJhc2gKcm0gLXIgZGlzdC8qCnBvZXRyeSBidWlsZApwb2V0cnkgcHVibGlzaApgYGAK readmeEtag: '"df3d513e4cc2ca3a4eced9cd6fda5b84ebf03d02"' readmeLastModified: Wed, 05 Oct 2022 01:05:41 GMT repositoryId: 201582900 description: Define SQLAlchemy models using the OpenAPI specification. created: '2019-08-10T05:09:36Z' updated: '2026-01-05T00:30:16Z' language: Python archived: false stars: 55 watchers: 1 forks: 13 owner: jdkandersson logo: https://avatars.githubusercontent.com/u/51036209?v=4 license: Apache-2.0 repoEtag: '"1d18d4b7e41f1f56f872d48eec5e48737cfcb876dc725387ac41df50740ca2d1"' repoLastModified: Mon, 05 Jan 2026 00:30:16 GMT foundInMaster: true category: Parsers id: 59cf3e6ba29a432331242eb475acac2f - source: openapi3 tags repository: https://github.com/piomin/sample-quarkus-applications v3: true repositoryMetadata: base64Readme: >- IyBHdWlkZSB0byBRdWFya3VzIERlbW8gUHJvamVjdCBbIVtUd2l0dGVyXShodHRwczovL2ltZy5zaGllbGRzLmlvL3R3aXR0ZXIvZm9sbG93L3Bpb3RyX21pbmtvd3NraS5zdmc/c3R5bGU9c29jaWFsJmxvZ289dHdpdHRlciZsYWJlbD1Gb2xsb3clMjBNZSldKGh0dHBzOi8vdHdpdHRlci5jb20vcGlvdHJfbWlua293c2tpKQoKWyFbQ2lyY2xlQ0ldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL3Bpb21pbi9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMuc3ZnP3N0eWxlPXN2ZyldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL3Bpb21pbi9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMpCgpbIVtTb25hckNsb3VkXShodHRwczovL3NvbmFyY2xvdWQuaW8vaW1hZ2VzL3Byb2plY3RfYmFkZ2VzL3NvbmFyY2xvdWQtYmxhY2suc3ZnKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1waW9taW5fc2FtcGxlLXF1YXJrdXMtYXBwbGljYXRpb25zKQpbIVtCdWdzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1waW9taW5fc2FtcGxlLXF1YXJrdXMtYXBwbGljYXRpb25zJm1ldHJpYz1idWdzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1waW9taW5fc2FtcGxlLXF1YXJrdXMtYXBwbGljYXRpb25zKQpbIVtDb3ZlcmFnZV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9cGlvbWluX3NhbXBsZS1xdWFya3VzLWFwcGxpY2F0aW9ucyZtZXRyaWM9Y292ZXJhZ2UpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPXBpb21pbl9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMpClshW0xpbmVzIG9mIENvZGVdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PXBpb21pbl9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMmbWV0cmljPW5jbG9jKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1waW9taW5fc2FtcGxlLXF1YXJrdXMtYXBwbGljYXRpb25zKQoKSW4gdGhpcyBwcm9qZWN0IEknbSBkZW1vbnN0cmF0aW5nIHRoZSBtb3N0IGludGVyZXN0aW5nIGZlYXR1cmVzIG9mIFtRdWFya3VzXShodHRwczovL3F1YXJrdXMuaW8vKSBmb3IgYnVpbGRpbmcgYXBwbGljYXRpb25zIGluIEtvdGxpbi4KCiMjIEdldHRpbmcgU3RhcnRlZCAKSGVyZSdzIGEgZnVsbCBsaXN0IG9mIGF2YWlsYWJsZSBleGFtcGxlczoKMS4gVXNpbmcgUXVhcmt1cyBmb3IgYnVpbGRpbmcgUkVTVCBhcHBsaWNhdGlvbiB0aGF0IGNvbm5lY3RzIHRvIEgyIGRhdGFiYXNlIHVzaW5nIEhpYmVybmF0ZSBPUk0uIFRoZSBleGFtcGxlIGlzIGF2YWlsYWJsZSBpbiB0aGUgbW9kdWxlIFtlbXBsb3llZS1zZXJ2aWNlXShodHRwczovL2dpdGh1Yi5jb20vcGlvbWluL3NhbXBsZS1xdWFya3VzLWFwcGxpY2F0aW9ucy90cmVlL21hc3Rlci9lbXBsb3llZS1zZXJ2aWNlKS4gQSBkZXRhaWxlZCBndWlkZSBtYXkgYmUgZm91bmQgaW4gdGhlIGZvbGxvd2luZyBhcnRpY2xlOiBbR3VpZGUgdG8gUXVhcmt1cyB3aXRoIEtvdGxpbl0oaHR0cHM6Ly9waW90cm1pbmtvd3NraS5jb20vMjAyMC8wOC8wOS9ndWlkZS10by1xdWFya3VzLXdpdGgta290bGluLykKMi4gVXNpbmcgUXVhcmt1cyBLdWJlcm5ldGVzIGV4dGVuc2lvbnMgdG8gZGVwbG95IGFwcGxpY2F0aW9uIGVhc2lseSBvbiBLdWJlcm5ldGVzLiBUaGUgZXhhbXBsZSBpcyBhdmFpbGFibGUgaW4gdGhlIG1vZHVsZSBbZW1wbG95ZWUtc2VydmljZV0oaHR0cHM6Ly9naXRodWIuY29tL3Bpb21pbi9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMvdHJlZS9tYXN0ZXIvZW1wbG95ZWUtc2VydmljZSkuIEEgZGV0YWlsZWQgZ3VpZGUgbWF5IGJlIGZvdW5kIGluIHRoZSBmb2xsb3dpbmcgYXJ0aWNsZTogW0d1aWRlIHRvIFF1YXJrdXMgb24gS3ViZXJuZXRlc10oaHR0cHM6Ly9waW90cm1pbmtvd3NraS5jb20vMjAyMC8wOC8xMC9ndWlkZS10by1xdWFya3VzLW9uLWt1YmVybmV0ZXMvKQozLiBVc2luZyBRdWFya3VzIE9BdXRoMiBleHRlbnNpb24gdG8gcHJvdmlkZSBSQkFDIGF1dGhvcml6YXRpb24gYmFzZWQgb24gaW50ZWdyYXRpb24gd2l0aCBLZXljbG9hay4gVGhlIGV4YW1wbGUgaXMgYXZhaWxhYmxlIGluIHRoZSBtb2R1bGUgW2VtcGxveWVlLXNlY3VyZS1zZXJ2aWNlXShodHRwczovL2dpdGh1Yi5jb20vcGlvbWluL3NhbXBsZS1xdWFya3VzLWFwcGxpY2F0aW9ucy90cmVlL21hc3Rlci9lbXBsb3llZS1zZWN1cmUtc2VydmljZSkuIEEgZGV0YWlsZWQgZ3VpZGUgbWF5IGJlIGZvdW5kIGluIHRoZSBmb2xsb3dpbmcgYXJ0aWNsZTogW1F1YXJrdXMgT0F1dGgyIGFuZCBzZWN1cml0eSB3aXRoIEtleWNsb2FrXShodHRwczovL3Bpb3RybWlua293c2tpLmNvbS8yMDIwLzA5LzE2L3F1YXJrdXMtb2F1dGgyLWFuZC1zZWN1cml0eS13aXRoLWtleWNsb2FrLykKNC4gVXNpbmcgUXVhcmt1cyB3aXRoIFNtYWxsUnllIEdyYXBoIGV4dGVuc2lvbiB0byBHcmFwaFFMIEFQSSBhbmQgaW50ZWdyYXRpb24gd2l0aCBhIGRhdGFiYXNlIHdpdGggUGFuYWNoZS4gVGhlIGV4YW1wbGUgaXMgYXZhaWxhYmxlIGluIHRoZSBtb2R1bGUgW3NhbXBsZS1hcHAtZ3JhcGhxbF0oaHR0cHM6Ly9naXRodWIuY29tL3Bpb21pbi9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMvdHJlZS9tYXN0ZXIvc2FtcGxlLWFwcC1ncmFwaHFsKS4gQSBkZXRhaWxlZCBndWlkZSBtYXkgYmUgZm91bmQgaW4gdGhlIGZvbGxvd2luZyBhcnRpY2xlOiBbQW4gQWR2YW5jZWQgR3JhcGhRTCB3aXRoIFF1YXJrdXNdKGh0dHBzOi8vcGlvdHJtaW5rb3dza2kuY29tLzIwMjEvMDQvMTQvYWR2YW5jZWQtZ3JhcGhxbC13aXRoLXF1YXJrdXMvKQo1LiBVc2luZyBRdWFya3VzIEZ1bnF5IEhUVFAgYW5kIEF6dXJlIEV4dGVuc2lvbnMgdG8gYnVpbGQgYW5kIHJ1biBzZXJ2ZXJsZXNzIGFwcHMgb24gQXp1cmUgRnVuY3Rpb25zLiBUaGUgZXhhbXBsZSBpcyBhdmFpbGFibGUgaW4gdGhlIG1vZHVsZSBbYWNjb3VudC1mdW5jdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL3Bpb21pbi9zYW1wbGUtcXVhcmt1cy1hcHBsaWNhdGlvbnMvdHJlZS9tYXN0ZXIvYWNjb3VudC1mdW5jdGlvbikuIEEgZGV0YWlsZWQgZ3VpZGUgbWF5IGJlIGZvdW5kIGluIHRoZSBmb2xsb3dpbmcgYXJ0aWNsZTogW1NlcnZlcmxlc3Mgb24gQXp1cmUgRnVuY3Rpb24gd2l0aCBRdWFya3VzXShodHRwczovL3Bpb3RybWlua293c2tpLmNvbS8yMDI0LzAxLzE5L3NlcnZlcmxlc3Mtb24tYXp1cmUtd2l0aC1zcHJpbmctY2xvdWQtZnVuY3Rpb24vKQ== readmeEtag: '"86e7361daebe88c2832f2145cb13edd6ee7ca185"' readmeLastModified: Sat, 10 Aug 2024 02:29:45 GMT repositoryId: 286198180 description: Example application built using Quarkus framework created: '2020-08-09T08:34:19Z' updated: '2025-11-29T20:46:29Z' language: Java archived: false stars: 48 watchers: 1 forks: 25 owner: piomin logo: https://avatars.githubusercontent.com/u/2892502?v=4 repoEtag: '"3e4acc741c70aa41e82a0f0892068e530381e102760e07d5035752ade64ad582"' repoLastModified: Sat, 29 Nov 2025 20:46:29 GMT foundInMaster: true category: SDK id: dd8f9923c6579fbc72bcd8fc5a69a30c - source: openapi3 tags repository: https://github.com/egeucak/api-doc-gpt v3: true id: 1b63b5474bf136c577ee3b41fafb3685 repositoryMetadata: base64Readme: >- IyBBUEkgRG9jIEdQVAoKVGhpcyBpcyBhbiBBSSB0b29sIHRoYXQgaGVscHMgeW91IGRpc2NvdmVyIHlvdXIgQVBJIGRvY3VtZW50YXRpb24uIEl0IGlzIGJhc2VkIG9uIHRoZSBbR1BULTMuNV0oaHR0cHM6Ly9vcGVuYWkuY29tL2Jsb2cvYmV0dGVyLWxhbmd1YWdlLW1vZGVscy8pIGxhbmd1YWdlIG1vZGVsIGZyb20gT3BlbkFJLgpZb3UgY2FuIHVzZSB0aGlzIHdpdGggZWl0aGVyIGRpcmVjdGx5IGNvbm5lY3RpbmcgdG8gYSBGYXN0QXBpIGFwcCBvciBieSBwYXNzaW5nIGluIGFuIGBvcGVuYXBpLmpzb25gIGZpbGUuCgojIyBTZXR1cAoKSW5zdGFsbCB0aGUgZGVwZW5kZW5jaWVzCgpgYGBiYXNoCnBpcCBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQKYGBgCgojIyBIb3cgdG8gdXNlIHdpdGggRmFzdEFwaQoKMS4gUnVuIHlvdXIgYmFja2VuZCBhcHAKCmBgYGJhc2gKdXZpY29ybiBtYWluOmFwcCAtLXJlbG9hZApgYGAKCjIuIFJ1biB0aGUgc2NyaXB0CgpgYGBiYXNoCnB5dGhvbiBhcGlfbWFzdGVyLnB5IC0tb3BlbmFpLWtleSA8eW91ci1vcGVuYWkta2V5PiAtLXRhcmdldCA8eW91ci1mYXN0YXBpLWFwcD4gLS1iYXNlLXVybCA8eW91ci1iYXNlLXVybD4KYGBgCgojIyBIb3cgdG8gdXNlIHdpdGggb3BlbmFwaS5qc29uCgpgYGBiYXNoCnB5dGhvbiBhcGlfbWFzdGVyLnB5IC0tb3BlbmFpLWtleSA8eW91ci1vcGVuYWkta2V5PiAtLW9wZW5hcGktanNvbiA8b3BlbmFwaS1qc29uPiAtLWJhc2UtdXJsIDx5b3VyLWJhc2UtdXJsPgpgYGAKCiMjIFJ1bm5pbmcgZXhhbXBsZQoKYGBgYmFzaApweXRob24gYXBpX21hc3Rlci5weSAtLW9wZW5haS1rZXk9PHlvdXItb3BlbmFpLWtleT4gLS1vcGVuYXBpLWpzb249Li9leGFtcGxlL29wZW5hcGkuanNvbiAtLWJhc2UtdXJsPSJodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92MyIKYGBgCgpFbmpveSBpbnRlcmFjdGluZyB3aXRoIHlvdXIgQVBJIGRvY3VtZW50YXRpb24KIVtzaG93Y2FzZV0oLi9zaG93Y2FzZS5wbmcpCgojIFdpdGggR1BULTQKClRoaXMgYWxzbyB3b3JrcyB3aXRoIEdQVC00LiBZb3UganVzdCBuZWVkIHRvIHBhc3MgcGFyYW1ldGVyIGAtLW1vZGVsLW5hbWU9Z3B0LTRgIHdoaWxlIHJ1bm5pbmcgdGhlIHNjcmlwdC4KCiMjIEhvdyBpdCB3b3JrcwoKVGhpcyB3b3JrcyBieSBjcmVhdGluZyBvcGVuIGFwaSBkb2N1bWVudGF0aW9uIGZyb20geW91ciBmYXN0YXBpIGFwcCBhbmQgdGhlbiB1c2luZyB0aGUgR1BULTMuNSBsYW5ndWFnZSBtb2RlbCB0byBhbmFseXplIGRvY3VtZW50YXRpb24gZm9yIHlvdXIgQVBJLgoKYGBgbWVybWFpZApzZXF1ZW5jZURpYWdyYW0KICAgIFVzZXItLT4+K1Byb2Nlc3NvcjogR2l2ZSBtZSBhIGxpc3Qgb2YgaXRlbXMuIEkgb25seSBuZWVkIDUgb2YgdGhlbQogICAgUHJvY2Vzc29yLS0+PitBSTogUFJPTVBUOiBHaXZlIG1lIGEgbGlzdCBvZiBpdGVtcy4gSSBvbmx5IG5lZWQgNSBvZiB0aGVtCiAgICBBSS0tPj4rUHJvY2Vzc29yOiBDTUQ6IEdFVCAvaXRlbXM/ZmlsdGVyPTUKICAgIFByb2Nlc3Nvci0tPj4rQUk6IENNRF9SRVNQOiBbeyJuYW1lIjogImN1Y3VtYmVyIn0sIHsibmFtZSI6ICJjZWxlcnkifS4uLl0KICAgIEFJLS0+PitQcm9jZXNzb3I6IE9VVDogSXRlbXMgYXJlIGN1Y3VtYmVyLCBjZWxlcnkuLi4KICAgIFByb2Nlc3Nvci0tPj4rVXNlcjogSXRlbXMgYXJlIGN1Y3VtYmVyLCBjZWxlcnkuLi4KYGBgCgpUaGlzIGFsc28gc3VwcG9ydHMgcmVhY3QuIE1vcmUgZGV0YWlsIGNhbiBiZSBmb3VuZCBhdDogaHR0cHM6Ly9hcnhpdi5vcmcvYWJzLzIyMTAuMDM2MjkKCiMjIENvbnN0cmFpbnRzCgotIFRva2VuIHNpemUK readmeEtag: '"1fbcc22a62df0c9f992b6601ffeed6e01ca6517c"' readmeLastModified: Mon, 27 Mar 2023 15:02:29 GMT repositoryId: 609666398 description: ChatGPT for Openapi created: '2023-03-04T21:34:21Z' updated: '2025-10-28T17:06:10Z' language: Python archived: false stars: 45 watchers: 2 forks: 6 owner: egeucak logo: https://avatars.githubusercontent.com/u/15627365?v=4 license: MIT repoEtag: '"2d0c0eb79a093e4227441e62003aeefb71da8432133cd532b83161b2170c2e17"' repoLastModified: Tue, 28 Oct 2025 17:06:10 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/juzibot/wecom-openapi v3: true repositoryMetadata: base64Readme: >- PGltZyBzcmM9Imh0dHBzOi8vd3djZG4ud2VpeGluLnFxLmNvbS9ub2RlL3dld29yay9pbWFnZXMvbG9nby5jNzY4Yzc1NmFiLnBuZyIgd2lkdGg9IjMwMCI+CjxpbWcgc3JjPSJodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvd3AtY29udGVudC91cGxvYWRzL3NpdGVzLzMvMjAxOC8wMi9PcGVuQVBJX0xvZ29fUGFudG9uZS0xLnBuZyIgd2lkdGg9IjMwMCI+CjxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci5pby93b3JkcHJlc3MvaW1hZ2VzL2Fzc2V0cy9TV1UtbG9nby1jbHIucG5nIiB3aWR0aD0iMzAwIj4KCiMKClvnroDkvZPkuK3mloddKC4vUkVBRE1FX1pILm1kKQoKIyMgRGVzY3JpcHRpb24KClRoaXMgcHJvamVjdCBhaW1zIHRvIHRyYW5zbGF0ZSB0aGUgV2VDb20gZG9jdW1lbnRhdGlvbiBpbnRvIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiwgc28gdGhhdCB5b3UgY2FuIGdlbmVyYXRlIGNsaWVudCBjb2RlIGluIHZhcmlvdXMgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2VzLiBDdXJyZW50bHksIHdlIGFyZSB1c2luZyB2ZXJzaW9uIDMueCBvZiBPcGVuQVBJIFNwZWNpZmljYXRpb24uCgojIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKJCBucG0gaW5zdGFsbApgYGAKCiMjIFJ1bm5pbmcgdGhlIGFwcAoKV2hlbiB0aGUgYXBwIGlzIHJ1bm5pbmcsIHRoZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gZmlsZSBgb3BlbmFwaS55YW1sYCB3aWxsIGJlIGdlbmVyYXRlZC4gWW91IGNhbiBvcGVuIGBodHRwOi8vbG9jYWxob3N0OjMwMDAvb3BlbmFwaWAgdG8gdmlldyBTd2FnZ2VyIFVJLgoKYGBgYmFzaAokIG5wbSBydW4gc3RhcnQKYGBgCgojIyBHZW5lcmF0aW5nIGNsaWVudCBjb2RlCgpbc3dhZ2dlci1jb2RlZ2VuXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1jb2RlZ2VuKSBpcyBhIHZlcnkgcG9wdWxhciB0b29sIHRvIGdlbmVyYXRlIGNvZGUgZnJvbSBPcGVuQVBJIFNwZWNpZmljYXRpb24uIEJ1dCBpZiB5b3UgYXJlIHVzaW5nIEdvbGFuZywgW2dvLXN3YWdnZV0oaHR0cHM6Ly9naXRodWIuY29tL2dvLXN3YWdnZXIvZ28tc3dhZ2dlcikgaXMgcmVjb21tZW5kZWQuCgojIyMgRXhhbXBsZQoKW2dvLXN3YWdnZV0oaHR0cHM6Ly9naXRodWIuY29tL2dvLXN3YWdnZXIvZ28tc3dhZ2dlcikgaXMgYmFzZWQgb24gT3BlbkFQSSBTcGVjaWZpY2F0aW9uIDIuMCwgc28geW91Cm5lZWQgdG8gZG93bmdyYWRlIHRoZSB2ZXJzaW9uLgoKYGBgYmFzaAokIG5wbSBpbnN0YWxsIC1nIGFwaS1zcGVjLWNvbnZlcnRlcgokIGFwaS1zcGVjLWNvbnZlcnRlciAtLWZyb209b3BlbmFwaV8zIC0tdG89c3dhZ2dlcl8yIC0tc3ludGF4PXlhbWwgLS1vcmRlcj1hbHBoYSAuL29wZW5hcGkueWFtbCA+IHN3YWdnZXIueWFtbApgYGAKClRoZW4gZ2VuZXJhdGUgR29sYW5nIGNvZGUgZnJvbSBgc3dhZ2dlci55YW1sYCBmaWxlLgoKYGBgCiQgYnJldyB0YXAgZ28tc3dhZ2dlci9nby1zd2FnZ2VyCgokIGJyZXcgaW5zdGFsbCBnby1zd2FnZ2VyCgokIG1rZGlyIHdlY29tLWFwaSAmJiBjZCB3ZWNvbS1hcGkKCiMgTk9URTogeW91IG5lZWQgcnVuIGdvIG1vZCBpbml0IHRvIGNyZWF0ZSBhIGdvLm1vZCBmaWxlCiQgZ28gbW9kIGluaXQgd2Vjb20tYXBpCgokIHN3YWdnZXIgZ2VuZXJhdGUgY2xpZW50IC1mIHN3YWdnZXIueWFtbCAtdCB3ZWNvbS1hcGkKYGBgCgojIyBEZXYKClRoaXMgcHJvamVjdCBpcyBiYXNlZCBvbiBTd2FnZ2VyIGludGVncmF0aW9uIG9mIE5lc3RKUy4gUmVmZXIgdG8gW09wZW5BUEkoTmVzdEpTKV0oaHR0cHM6Ly9kb2NzLm5lc3Rqcy5jb20vb3BlbmFwaS9pbnRyb2R1Y3Rpb24pIGZvciBkZXRhaWwuCgojIyMgQVBJIERlZmluaXRpb24gaW4gY29udHJvbGxlciBmaWxlcwoKYGBgCm5lc3QgZyAtLW5vLXNwZWMgY29udHJvbGxlciBkZXBhcnRtZW50CmBgYAoKIyMjIENyZWF0ZSBEVE8gYW5kIFJlc3BvbnNlIHN0cnVjdHVyZQoKVXN1YWxseSwgd2UgZGVjbGFyZSBEVE9zIGFuZCBSZXNwb25zZXMgaW4gZGlmZmVyZW50IGNsYXNzZXMuCgpgYGAKbmVzdCBnIC0tbm8tc3BlYyBjbGFzcyBkZXBhcnRtZW50L0RlcGFydG1lbnQKYGBgCgojIyMgQVBJIFByb3BlcnRpZXMKCmBgYAogIEBBcGlQcm9wZXJ0eSh7CiAgICByZXF1aXJlZDogZmFsc2UsCiAgICB0eXBlOiAnc3RyaW5nJywKICAgIGlzQXJyYXk6IHRydWUsCiAgICBtYXhJdGVtczogMTAwLAogICAgZXhhbXBsZTogJ1siYWJlbCJdJywKICAgIGRlc2NyaXB0aW9uOiAndXNlciBpZCBsaXN0LCBtYXggbGVuZ3RoIGlzIDEwMCcsCiAgfSkKICB1c2VyaWRfbGlzdDogc3RyaW5nW107CmBgYAoKLSBgcmVxdWlyZWRgLCBgdHlwZWAgaXMgcmVxdWlyZWQuCi0gYGRlc2NyaXB0aW9uYCwgYGV4YW1wbGVgIHJlY29tbWVuZGVk44CCCgpgYGAKICBAQXBpT3BlcmF0aW9uKHsKICAgIG9wZXJhdGlvbklkOiAnZ2V0R3JvdXBDaGF0JywKICAgIHN1bW1hcnk6ICdHZXQgdXNlciBncm91cCBkZXRhaWwnLAogICAgZGVzY3JpcHRpb246ICdHZXQgdXNlciBncm91cCBkZXRhaWwnLAogICAgZXh0ZXJuYWxEb2NzOiB7CiAgICAgIHVybDogJ2h0dHBzOi8vZGV2ZWxvcGVyLndvcmsud2VpeGluLnFxLmNvbS9kb2N1bWVudC9wYXRoLzkyMTIyJywKICAgIH0sCiAgfSkKYGBgCgotIGBvcGVyYXRpb25JZGAgUmVxdWlyZWQsIFVuaXF1ZSwgSXMgZnVuY3Rpb24gbmFtZSBpbiBnZW5lcnRhdGlvbiBjb2RlLgotIGBzdW1tYXJ5YCBSZXF1aXJlZAotIGBleHRlcm5hbERvY3NgIExpbmsgdG8gV2VDb20gb2ZmaWNpYWwgZG9jdW1lbnQuCgojIyBTd2FnZ2VyIFVJIFRyeSBpdCBvdXQKCiMjIyBBdXRob3JpemUKCkZpcnN0LCB5b3UgbmVlZCB0byBnZXQgYW4gYGFjY2Vzc190b2tlbmAgdG8gYXV0aG9yaXplIHRoZSBhcGkgY2FsbHMuCiFbXSguL3NjcmVlbnNob3QvYXV0aG9yaXplX3N0ZXAucG5nKQoKIyMjIFByb3h5CgpCcm93c2VyIGhhcyBDT1JTIGlzc3VlLCBzbyB3ZSBjYW5ub3QgYWNjZXNzIHd4LXdvcmsgZGlyZWN0bHkuIFNvIHdlIG1hZGUgYSBwcm94eSBmb3IgeW91LiBUbyBhY2Nlc3MsIHVzZSBgaHR0cDovL2xvY2FsaG9zdDozMDAwL2NnaS1iaW5gLgohW10oLi9zY3JlZW5zaG90L3Byb3h5X3N0ZXAucG5nKQo= readmeEtag: '"3ab97d0b553ac8683a7785a2d4d1a16c7d3ce104"' readmeLastModified: Fri, 11 Nov 2022 03:36:15 GMT repositoryId: 457707390 description: Swagger OpenAPI of WeCom. 企业微信的 Swagger 文档。 created: '2022-02-10T09:13:32Z' updated: '2026-01-26T05:37:13Z' language: TypeScript archived: false stars: 47 watchers: 3 forks: 3 owner: juzibot logo: https://avatars.githubusercontent.com/u/36260234?v=4 repoEtag: '"3c64520692ee4cff13718905dac94ad39c8e453c945d184c606b296f542b20f6"' repoLastModified: Mon, 26 Jan 2026 05:37:13 GMT foundInMaster: true category: Parsers id: 34d44163975586f318085cb7dc4b1ba6 - source: openapi3 tags repository: https://github.com/redocly/museum-openapi-example v3: true id: 1908c5d265dd6cda6a2bb2946e0fe880 repositoryMetadata: base64Readme: >- IyBSZWRvY2x5IE11c2V1bSBBUEkgRXhhbXBsZQoKQW4gaW1hZ2luYXJ5LCBidXQgZGVsaWdodGZ1bCBNdXNldW0gQVBJIGZvciBpbnRlcmFjdGluZyB3aXRoIG11c2V1bSBzZXJ2aWNlcyBhbmQgaW5mb3JtYXRpb24uCkJ1aWx0IGJ5IFJlZG9jbHkgZm9yIGVkdWNhdGlvbmFsIHB1cnBvc2VzLgoKPiBbIU5PVEVdICAKPiBUaGlzIE9wZW5BUEkgZGVzY3JpcHRpb24gdXNlcyB0aGUgW09wZW5BUEkgMy4xLjAgc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMS4wKS4KCiMjIE92ZXJ2aWV3CgpJbnRyb2R1Y2luZyB0aGUgIk11c2V1bSBBUEkiLCB3aGljaCBvZmZlcnMgYSBzZXQgb2YgZW5kcG9pbnRzIHRvIGludGVyYWN0IHdpdGggYSBtdXNldW0ncyBzZXJ2aWNlcy0tc3VjaCBhcyByZXRyaWV2aW5nIG11c2V1bSBob3VycywgbWFuYWdpbmcgc3BlY2lhbCBldmVudHMsIGFuZCBwdXJjaGFzaW5nIHRpY2tldHMuCgpOZXcgZnVuY3Rpb25hbGl0eSBtYXkgYmUgYWRkZWQgdG8gdGhlIE11c2V1bSBBUEkgaW4gdGhlIGZ1dHVyZS4KSXMgdGhlcmUgYW4gZXhhbXBsZSB5b3UnZCBsaWtlIHRvIHNlZT8gUGxlYXNlIFtvcGVuIGFuIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9tdXNldW0tb3BlbmFwaS1leGFtcGxlL2lzc3Vlcy9uZXcpLgoKIyMgRmVhdHVyZXMKCj4gWyFOT1RFXSAgCj4gUmVtaW5kZXIgdGhhdCB0aGVzZSBhcmUgZmljdGlvbmFsIGV4YW1wbGVzIGZvciBsZWFybmluZyBwdXJwb3Nlcy4KPiBUaGUgY29udGVudHMgb2YgdGhpcyByZXBvc2l0b3J5IGFyZSBpbGx1c3RyYXRpb25zIHRvIHVzZSB3aXRoIHlvdXIgT3BlbkFQSSBzdHVkeSBvciB0b29scyBleHBsb3JhdGlvbi4KCiMjIyBPcGVuQVBJCgotIGBvcGVuYXBpLnlhbWxgIGNvbnRhaW5zIHRoZSBNdXNldW0gQVBJLiBJdCBpcyBhbiBPcGVuQVBJIDMuMS4wIGRlc2NyaXB0aW9uLgoKVGhlIE11c2V1bSBBUEkgaGFzIHRoZSBmb2xsb3dpbmcgY29yZSBmZWF0dXJlcyAoc29ydGVkIGJ5IHRhZ3MpOgotIE9wZXJhdGlvbnMKICAtIEdldCBtdXNldW0gb3BlcmF0aW9uYWwgaG91cnMKLSBTcGVjaWFsIGV2ZW50cwogIC0gR2V0IHNwZWNpYWwgZXZlbnQgZGF0YQogIC0gTGlzdCBzcGVjaWFsIGV2ZW50cwogIC0gQ3JlYXRlIGFuZCB1cGRhdGUgYSBzcGVjaWFsIGV2ZW50CiAgLSBEZWxldGUgYSBzcGVjaWFsIGV2ZW50Ci0gVGlja2V0cwogIC0gUHVyY2hhc2UgbXVzZXVtIHRpY2tldHMgZm9yIGdlbmVyYWwgZW50cnkgb3Igc3BlY2lhbCBldmVudHMKICAtIEdldCBzY2FubmFibGUgUVIgY29kZSBmb3IgbXVzZXVtIHRpY2tldAoKIyMjIEFyYXp6bwoKLSBgYXJhenpvL211c2V1bS10aWNrZXRzLmFyYXp6by55YW1sYCBpcyBhbiBBcmF6em8gMS4wLjAgZGVzY3JpcHRpb24gb2YgdXNpbmcgdGhlIE11c2V1bSBPcGVuQVBJIHNvdXJjZSB0byBkZXNjcmliZSBhbiBBUEkgc2VxdWVuY2UuCi0gYGFyYXp6by9tdXNldW0tYXBpLmFyYXp6by55YW1sYCBpcyBhbiBBcmF6em8gMS4wLjAgZGVzY3JpcHRpb24gdXNpbmcgYm90aCB0aGUgTXVzZXVtIEFQSSBhbmQgYW5vdGhlciBBcmF6em8gZmlsZSB0byBkZXNjcmliZSBhIHNlcmllcyBvZiBtdWx0aS1zdGVwIEFQSSB3b3JrZmxvd3MgYW5kIHBhc3NpbmcgdGhlIHZhcmlhYmxlcyBiZXR3ZWVuIHRoZW0uCgojIyBHZXR0aW5nIHN0YXJ0ZWQKCjEuIENsb25lIHRoaXMgcmVwby4KMi4gT3BlbiB0aGUgcmVwbyBpbiB5b3VyIElERS4KMy4gUnVuIGBucG0gaW5zdGFsbGAgdG8gaW5zdGFsbCB0aGUgUmVkb2NseSBDTEkuCgojIyMgV29ya2luZyB3aXRoIHRoZSBPcGVuQVBJIGRlc2NyaXB0aW9uCgpXZSBlbmNvdXJhZ2UgeW91IHRvIGV4cGxvcmUgdGhlIG11c2V1bSdzIE9wZW5BUEkgZGVzY3JpcHRpb24gYW5kIG1ha2UgY2hhbmdlcy4KVHJ5IGV4cGVyaW1lbnRpbmcgd2l0aCB0aGUgZm9sbG93aW5nIGFwcHJvYWNoZXM6CgoqKlByZXZpZXcgdGhlIE11c2V1bSBPcGVuQVBJIGV4YW1wbGUncyBBUEkgZG9jcyBhbmQgb2JzZXJ2ZSB5b3VyIGNoYW5nZXMgdmlzdWFsbHkuKioKCi0gUnVuIGBucG0gcnVuIHByZXZpZXdgIHRvIHByZXZpZXcgdGhlIGRvY3VtZW50YXRpb24uCi0gTmF2aWdhdGUgdG8gdGhlICoqTGlzdCBzcGVjaWFsIGV2ZW50cyoqIG9wZXJhdGlvbiBpbiB0aGUgcHJldmlldy4KLSBXaXRoIHRoZSBwcmV2aWV3IHJ1bm5pbmcuLi4KICAtIEdvIHRvIHRoZSBfb3BlbmFwaS55YW1sXyBmaWxlIGluIHlvdXIgSURFLgogIC0gU2VhcmNoIGZvciBgbGlzdFNwZWNpYWxFdmVudHNgIHRvIGZpbmQgdGhlIG1hdGNoaW5nIGBvcGVyYXRpb25JZGAuCiAgLSBSZXBsYWNlIHRoZSBgZGVzY3JpcHRpb25gIGZpZWxkIHdpdGggdGhlIHNuaXBwZXQgYmVsb3c6CgpgYGB5YW1sCiAgICBkZXNjcmlwdGlvbjogfC0KICAgICAgICBSZXR1cm4gYSBsaXN0IG9mIF91cGNvbWluZ18gc3BlY2lhbCBldmVudHMgYXQgdGhlIG11c2V1bS4KICAgICAgICAgICAgCiAgICAgICAgU2VlIG9uZSB5b3UgbGlrZT8gVXNlIHRoaXMgQVBJIHRvIFtidXkgYSB0aWNrZXRdKC8jdGFnL1RpY2tldHMvb3BlcmF0aW9uL2J1eU11c2V1bVRpY2tldHMpLiAgCmBgYAoKU2VlIHRoZSB1cGRhdGVkIGRlc2NyaXB0aW9uPyBUaGlzIGlzIGEgZ3JlYXQgd2F5IHRvIHByZXZpZXcgaG93IGVuZC11c2VycyBvZiB5b3VyIGRvY3Mgd2lsbCBleHBlcmllbmNlIHlvdXIgY2hhbmdlcy4KCioqTGludCB5b3VyIGNoYW5nZXMgdG8gdGhlIE9wZW5BUEkgZGVzY3JpcHRpb24gdXNpbmcgW1JlZG9jbHkgQ0xJXShodHRwczovL3JlZG9jbHkuY29tL2RvY3MvY2xpLykuKioKCi0gT3BlbiB0aGUgbXVzZXVtJ3MgX29wZW5hcGkueWFtbF8gZmlsZSBpbiB5b3VyIElERS4gCi0gQWRkIHRoZSBmb2xsb3dpbmcgc25pcHBldCB0byBgcGF0aHNgIGFib3ZlIHRoZSAvbXVzZXVtLWhvdXJzIG9wZXJhdGlvbjoKCmBgYHlhbWwKICAvZXhhbXBsZToKICAgIGdldDogCiAgICAgIHN1bW1hcnk6IEV4YW1wbGUgU3VtbWFyeQogICAgICBkZXNjcmlwdGlvbjogRXhhbXBsZSBkZXNjcmlwdGlvbgogICAgICByZXNwb25zZXM6IAogICAgICAgICcyMDAnOgogICAgICAgICAgZGVzY3JpcHRpb246IFN1Y2Nlc3MKICAgICAgICAnNDAwJzogCiAgICAgICAgICBkZXNjcmlwdGlvbjogQmFkIFJlcXVlc3QKYGBgCgotIFJ1biBgbnBtIHJ1biBsaW50YCBpbiB5b3VyIHRlcm1pbmFsLiBTZWUgdGhlIGVycm9ycz8gCgpUaGUgbGludGluZyBiZWhhdmlvciBpcyBjb250cm9sbGVkIGJ5IHRoZSBfcmVkb2NseS55YW1sXyBjb25maWd1cmF0aW9uIGZpbGUuClRoZSBsaW50ZXIgaXMgY29uZmlndXJlZCB0byB1c2UgUmVkb2NseSdzIFtyZWNvbW1lbmRlZCBydWxlc2V0XShodHRwczovL3JlZG9jbHkuY29tL2RvY3MvY2xpL3J1bGVzL3JlY29tbWVuZGVkLyNyZWNvbW1lbmRlZC1ydWxlc2V0KSwgd2hpY2ggYXJlIGJ1aWx0IGludG8gdGhlIENMSS4KSG93ZXZlciwgd2UgYWxzbyBhZGRlZCBhIFtjb25maWd1cmFibGUgcnVsZV0oaHR0cHM6Ly9yZWRvY2x5LmNvbS9kb2NzL2NsaS9ydWxlcy9jb25maWd1cmFibGUtcnVsZXMvKSBmb3IgZW5mb3JjaW5nIHNlbnRlbmNlIGNhc2luZyBvbiBvcGVyYXRpb24gc3VtbWFyaWVzLgoKKipMaW50IGFuIEFyYXp6byBkZXNjcmlwdGlvbiB1c2luZyBbUmVkb2NseSBDTEldKGh0dHBzOi8vcmVkb2NseS5jb20vZG9jcy9jbGkvKS4qKgoKLSBSZWRvY2x5IENMSSBjYW4gYWxzbyBsaW50IGZvcm1hdHMgb3RoZXIgdGhhbiBPcGVuQVBJLCBzdWNoIGFzIHRoZSBBcmF6em8gZm9ybWF0LgotIFJ1biBgbnBtIHJ1biBsaW50IGFyYXp6by9tdXNldW0tYXBpLmFyYXp6by55YW1sYCB0byBsaW50IGFuIGV4YW1wbGUgQXJhenpvIGRlc2NyaXB0aW9uLgo= readmeEtag: '"c9c914daa47efa61b11958a22b1523a89d69d58d"' readmeLastModified: Thu, 25 Jul 2024 14:17:47 GMT repositoryId: 713070411 description: 'An example OpenAPI description for an imaginary Museum API. ' created: '2023-11-01T19:22:54Z' updated: '2025-12-28T12:55:56Z' language: null archived: false stars: 66 watchers: 11 forks: 40 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"7c41d16f735128e58652f7de733c97419a545e9f56dce3eda1eb5f2797742a3b"' repoLastModified: Sun, 28 Dec 2025 12:55:56 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/alexferl/echo-boilerplate v3: true id: d15d98fca8fb5baef974da4b37117de6 repositoryMetadata: base64Readme: >- # echo-boilerplate [![Go Report Card](https://goreportcard.com/badge/github.com/alexferl/echo-boilerplate)](https://goreportcard.com/report/github.com/alexferl/echo-boilerplate) [![codecov](https://codecov.io/gh/alexferl/echo-boilerplate/branch/master/graph/badge.svg)](https://codecov.io/gh/alexferl/echo-boilerplate)

A Go 1.22+ boilerplate app using the minimalist [echo](https://github.com/labstack/echo) framework and with
authentication, authorization and request/response validation.

> **Note**: I use this as a starting point for personal projects, it can and will change without notice.

## Features
- [JWT](https://jwt.io/) for authentication with access and [refresh](https://auth0.com/blog/refresh-tokens-what-are-they-and-when-to-use-them/) tokens.
 The access token can be sent in the [Authorization](https://developer.mozilla.org/en-US/docs/Web/HTTP/Headers/Authorization) header or
 as a [cookie](https://developer.mozilla.org/en-US/docs/Web/HTTP/Cookies). See [echo-jwt](https://github.com/alexferl/echo-jwt).
- [Casbin](https://casbin.io/) for authorization using RBAC. See [echo-casbin](https://github.com/alexferl/echo-casbin).
- [OpenAPI](https://www.openapis.org/) for request and response validation. See [echo-openapi](https://github.com/alexferl/echo-openapi).

## Requirements
Before getting started, install the following:

Required:
- [pre-commit](https://pre-commit.com/#install)
- [MongoDB](https://www.mongodb.com/docs/manual/installation/#mongodb-installation-tutorials)

Optional:

- [gofumpt](https://pkg.go.dev/mvdan.cc/gofumpt) (needed to run `make fmt`)
- [redocly-cli](https://redocly.com/docs/cli/installation/) (needed to run `make openapi-lint`)

## Using
Setup the dev environment first:
```shell
make dev
```
>**Note**: An RSA private key will be generated in the current folder to sign and verify the JSON web tokens.

### Creating the superuser
Launch the superuser cmd with `go run ./cmd/superuser --password <your password>`. You can change the default values
with the following flags: `--email`, `--name` and `--username`. You can view all the other settings with `--help`.

### Building & Running locally
```shell
make run
```
### Using the API
#### Login
Request:
```shell
curl --request POST \
  --url http://localhost:1323/auth/login \
  --header 'Content-Type: application/json' \
  --data '{
	"email": "super@example.com",
	"password": "<your password>"
}'
```
Response:
```json
{
  "access_token": "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9...",
  "expires_in": 3600,
  "refresh_token": "eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9...",
  "token_type": "Bearer"
}
```
**Note**: The `access_token` only lasts 60 minutes by default, this is as designed. A client
(like an [SPA](https://en.wikipedia.org/wiki/Single-page_application) or a mobile application) would have an interceptor
to catch the 401 responses, send the `refresh_token` to the `/auth/refresh` endpoint to get new access and refresh tokens and
then retry the previous request with the new `access_token` which should then succeed. The duration of the `access_token`
can be modified with `--jwt-access-token-expiry` and the `refresh_token` with `--jwt-refresh-token-expiry`.

#### Get currently authenticated user
Request:

Using the `Authorization` header:
```shell
curl --request GET \
  --url http://localhost:1323/me \
  --header 'Authorization: Bearer eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9...'
```

Using the cookie (the cookie is sent automatically with web browsers, HTTPie and some other clients):
```shell
curl --request GET \
  --url http://localhost:1323/me \
  --cookie access_token=eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXVCJ9...
```

Response:
```json
{
  "id": "cdhgh0dfclscplnrcuag",
  "username": "super",
  "email": "super@example.com",
  "name": "Super",
  "bio": "",
  "created_at": "2022-11-03T00:17:05.837Z",
  "updated_at": null
}
```

### OpenAPI docs
You can see the OpenAPI docs by running the app and navigating to `http://localhost:1323/docs` or by
opening [assets/index.html](docs/index.html) in your web browser.

### Repository layout
```
.
├── casbin    <--- model and policy files for Casbin
├── cmd       <--- entrypoints
├── config    <--- config structs and defaults are specified here
├── configs   <--- config files, for configs that rarely change, but should override the defaults
├── data      <--- base mapper, database helpers
├── docs      <--- generated documentation from OpenAPI schema
├── handlers  <--- HTTP handlers (aka controllers, endpoints etc.) that interacts with the services
├── mappers   <--- mapper layer that the services use to insert/retrieve models from the database
├── models    <--- structs defining the various resources
├── openapi   <--- OpenAPI schema files
├── server    <--- glues handlers/services/mappers
├── services  <--- service layer that interacts with the mappers
├── testing   <--- testing helpers
└── util      <--- general helpers
```

### Usage
```shell
go run ./cmd/server --help
```

```
Usage of ./echo-boilerplate:
      --app-name string                                The name of the application. (default "app")
      --base-url string                                Base URL where the app will be served (default "http://localhost:1323")
      --casbin-model string                            Casbin model file (default "./casbin/model.conf")
      --casbin-policy string                           Casbin policy file (default "./casbin/policy.csv")
      --cookies-domain string                          Cookies domain
      --cookies-enabled                                Send cookies with authentication requests
      --csrf-cookie-domain string                      CSRF cookie domain
      --csrf-cookie-name string                        CSRF cookie name (default "csrf_token")
      --csrf-enabled                                   CSRF enabled
      --csrf-header-name string                        CSRF header name (default "X-CSRF-Token")
      --csrf-secret-key string                         CSRF secret used to hash the token
      --env-name string                                The environment of the application. Used to load the right configs file. (default "local")
      --http-bind-address ip                           The IP address to listen at. (default 127.0.0.1)
      --http-bind-port uint                            The port to listen at. (default 1323)
      --http-cors-allow-credentials                    Tells browsers whether to expose the response to frontend JavaScript code when the request's credentials mode (Request.credentials) is 'include'.
      --http-cors-allow-headers strings                Indicate which HTTP headers can be used during an actual request.
      --http-cors-allow-methods strings                Indicates which HTTP methods are allowed for cross-origin requests. (default [GET,HEAD,PUT,PATCH,POST,DELETE])
      --http-cors-allow-origins strings                Indicates whether the response can be shared with requesting code from the given origin. (default [*])
      --http-cors-enabled                              Enable cross-origin resource sharing.
      --http-cors-expose-headers strings               Indicates which headers can be exposed as part of the response by listing their name.
      --http-cors-max-age int                          Indicates how long the results of a preflight request can be cached.
      --http-graceful-timeout duration                 Timeout for graceful shutdown. (default 30s)
      --http-log-requests                              Controls the logging of HTTP requests (default true)
      --http-tls-cert-file string                      TLS certificate file
      --http-tls-key-file string                       TLS key file
      --jwt-access-token-cookie-name string            JWT access token cookie name (default "access_token")
      --jwt-access-token-expiry duration               JWT access token expiry (default 1h0m0s)
      --jwt-issuer string                              JWT issuer (default "http://localhost:1323")
      --jwt-private-key string                         JWT private key file path (default "./private-key.pem")
      --jwt-refresh-token-cookie-name string           JWT refresh token cookie name (default "refresh_token")
      --jwt-refresh-token-expiry duration              JWT refresh token expiry (default 720h0m0s)
      --log-level string                               The granularity of log outputs. Valid levels: 'PANIC', 'FATAL', 'ERROR', 'WARN', 'INFO', 'DEBUG', 'TRACE', 'DISABLED' (default "INFO")
      --log-output string                              The output to write to. 'stdout' means log to stdout, 'stderr' means log to stderr. (default "stdout")
      --log-writer string                              The log writer. Valid writers are: 'console' and 'json'. (default "console")
      --mongodb-app-name string                        MongoDB app name
      --mongodb-connect-timeout-ms duration            MongoDB connect timeout ms (default 10s)
      --mongodb-password string                        MongoDB password
      --mongodb-replica-set string                     MongoDB replica set
      --mongodb-server-selection-timeout-ms duration   MongoDB server selection timeout ms (default 10s)
      --mongodb-socket-timeout-ms duration             MongoDB socket timeout ms (default 30s)
      --mongodb-uri string                             MongoDB URI (default "mongodb://localhost:27017")
      --mongodb-username string                        MongoDB username
      --oauth2-google-client-id string                 OAuth2 Google client id
      --oauth2-google-client-secret string             OAuth2 Google client secret
      --oauth2-providers strings                       OAuth2 providers
      --openapi-schema string                          OpenAPI schema file (default "./openapi/openapi.yaml")
```

### Docker
#### Build
```shell
make docker-build
```

#### Run
```shell
make docker-run
```

#### Passing args
CLI:
```shell
docker run -p 1323:1323 --rm echo-boilerplate --env-name prod
```

Environment variables:
```shell
docker run -p 1323:1323 -e "APP_ENV_NAME=prod" --rm echo-boilerplate
```
 readmeEtag: '"6174ccf8149a1263cc989f9972cc2c3f11fcad87"' readmeLastModified: Sun, 07 Apr 2024 22:53:32 GMT repositoryId: 161701098 description: >- Boilerplate for the Echo framework with authentication, authorization and request/response validation. created: '2018-12-13T22:16:36Z' updated: '2026-01-14T13:58:51Z' language: Go archived: true stars: 48 watchers: 2 forks: 15 owner: alexferl logo: https://avatars.githubusercontent.com/u/3533424?v=4 license: MIT repoEtag: '"b91e3b9a5bcbd3d1477758e9adeed89799942f3b78401b3afb1f42d1f6c4699c"' repoLastModified: Wed, 14 Jan 2026 13:58:51 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/atlassian/oas3-chow-chow v3: true repositoryMetadata: base64Readme: >- IyBvYXMzLWNob3ctY2hvdwoKPiBSZXF1ZXN0IGFuZCByZXNwb25zZSB2YWxpZGF0b3IgYWdhaW5zdCBPcGVuQVBJIFNwZWNpZmljYXRpb24KClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2F0bGFzc2lhbi9vYXMzLWNob3ctY2hvdy5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9hdGxhc3NpYW4vb2FzMy1jaG93LWNob3cpClshW25wbV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vYXMzLWNob3ctY2hvdy5zdmc/c3R5bGU9ZmxhdCldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29hczMtY2hvdy1jaG93KQoKIyMgTm90ZXMKCklmIHlvdSBhcmUgbG9va2luZyBmb3IgZnJhbWV3b3JrIHNwZWNpZmljIG1pZGRsZXdhcmUsIHlvdSBtaWdodCB3YW50IHRvIGxvb2sgYXQgZm9sbG93aW5nIGxpYnJhcmllcyB0aGF0IHVzZSBvYXMzLWNob3ctY2hvdyB1bmRlciB0aGUgaG9vZC4KCltrb2Etb2FzM10oaHR0cHM6Ly9naXRodWIuY29tL2F0bGFzc2lhbi9rb2Etb2FzMykKW29wZW5hcGkzLW1pZGRsZXdhcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9uYXVndHVyL29wZW5hcGkzLW1pZGRsZXdhcmUpCgojIyBJbnN0YWxsYXRpb24KTm90ZTogU3RhcnRpbmcgZnJvbSB2ZXJzaW9uIDUsIGl0J3MgdHJhbnNmZXJyZWQgdG8gYmUgdW5kZXIgYEBhdGxhc3NpYW5gIHNjb3BlLgoKYGBgYmFzaAokIHlhcm4gYWRkIEBhdGxhc3NpYW4vb2FzMy1jaG93LWNob3cKJCAjIE9yCiQgbnBtIGkgQGF0bGFzc2lhbi9vYXMzLWNob3ctY2hvdwpgYGAKCiMjIFVzYWdlCgpgYGB0eXBlc2NyaXB0CmltcG9ydCBDaG93Q2hvdyBmcm9tICJAYXRsYXNzaWFuL29hczMtY2hvdy1jaG93IjsKaW1wb3J0ICogYXMgZnMgZnJvbSAiZnMiOwppbXBvcnQgKiBhcyB5YW1sIGZyb20gImpzLXlhbWwiOwoKdmFyIGRvYyA9IHlhbWwuc2FmZUxvYWQoZnMucmVhZEZpbGVTeW5jKCIuL29wZW5hcGkueW1sIiwgInV0ZjgiKSk7CmNvbnN0IGNob3cgPSBDaG93Q2hvdy5jcmVhdGUoZG9jKTsKCi8vIEZvciBVUkw6IC86cGF0aFBhcmFtL2luZm8/YXJyUGFyYW09eCZhcnJQYXJhbT15Jm90aGVyPXoKY2hvdy52YWxpZGF0ZVJlcXVlc3RCeVBhdGgoCiAgLy8gdXJsLnBhdGhuYW1lLAogICIvYm9va3MvaW5mbyIsCiAgIlBPU1QiLCB7CiAgICBwYXRoOiB7IHBhdGhQYXJhbTogImJvb2tzIiB9LAogICAgLy8gcXVlcnk6IHF1ZXJ5c3RyaW5nLnBhcnNlKHVybC5zZWFyY2guc3Vic3RyKDEpKSwKICAgIHF1ZXJ5OiB7IGFyclBhcmFtOiBbIngiLCAieSJdLCBvdGhlcjogInoiIH0sCiAgICAvLyBoZWFkZXI6IHJlcS5oZWFkZXJzLAogICAgaGVhZGVyOiB7ICJDb250ZW50LVR5cGUiOiAiYXBwbGljYXRpb24vanNvbiIgfSwKICAgIGJvZHk6IHsgYTogMSwgYjogMiB9LAogIH0KKTsKY2hvdy52YWxpZGF0ZVJlc3BvbnNlQnlQYXRoKCIvYm9va3MvaW5mbyIsICJQT1NUIiwgewogIGhlYWRlcjogeyAiQ29udGVudC1UeXBlIjogImFwcGxpY2F0aW9uL2pzb24iIH0sCiAgYm9keTogewogICAgbmFtZTogImEgbmljZSBib29rIiwKICAgIGF1dGhvcjogIm1lIG1lIG1lIgogIH0KfSk7CmBgYAoKIyMgQ29uZmlnCgpZb3UgY291bGQgb3B0aW9uYWxseSBwcm92aWRlIGNvbmZpZ3MgdG8gdGhlIGNvbnN0cnVjdG9yCmBgYHR5cGVzY3JpcHQKY29uc3QgY2hvdyA9IENob3dDaG93LmNyZWF0ZShkb2MsIHsKICBoZWFkZXJBanZPcHRpb25zOiB7fSwKICBjb29raWVBanZPcHRpb25zOiB7fSwKICBwYXRoQWp2T3B0aW9uczogeyBjb2VyY2VUeXBlczogdHJ1ZSB9LAogIHF1ZXJ5QWp2T3B0aW9uczogeyBjb2VyY2VUeXBlczogJ2FycmF5JyB9LAogIHJlcXVlc3RCb2R5QWp2T3B0aW9uczoge30sCiAgcmVzcG9uc2VCb2R5QWp2T3B0aW9uczoge30sCn0pOwpgYGAKCiogKipoZWFkZXJBanZPcHRpb25zKio6IEFqdiBvcHRpb25zIHRoYXQgcGFzcyB0byBoZWFkZXIgYWp2IGluc3RhbmNlCiogKipjb29raWVBanZPcHRpb25zKio6IEFqdiBvcHRpb25zIHRoYXQgcGFzcyB0byBjb29raWUgYWp2IGluc3RhbmNlCiogKipwYXRoQWp2T3B0aW9ucyoqOiBBanYgb3B0aW9ucyB0aGF0IHBhc3MgdG8gcGF0aCBhanYgaW5zdGFuY2UsIGRlZmF1bHQgYHsgY29lcmNlVHlwZXM6IHRydWUgfWAKKiAqKnF1ZXJ5QWp2T3B0aW9ucyoqOiBBanYgb3B0aW9ucyB0aGF0IHBhc3MgdG8gcXVlcnkgYWp2IGluc3RhbmNlLCBkZWZhdWx0IGB7IGNvZXJjZVR5cGVzOiAnYXJyYXknIH1gCiogKipyZXF1ZXN0Qm9keUFqdk9wdGlvbnMqKjogQWp2IG9wdGlvbnMgdGhhdCBwYXNzIHRvIHJlcXVlc3QgYm9keSBhanYgaW5zdGFuY2UKKiAqKnJlc3BvbnNlQm9keUFqdk9wdGlvbnMqKjogQWp2IG9wdGlvbnMgdGhhdCBwYXNzIHRvIHJlc3BvbnNlIGJvZHkgYWp2IGluc3RhbmNlCgojIyBDb250cmlidXRvcnMKClB1bGwgcmVxdWVzdHMsIGlzc3VlcyBhbmQgY29tbWVudHMgd2VsY29tZS4gRm9yIHB1bGwgcmVxdWVzdHM6CgoqIEFkZCB0ZXN0cyBmb3IgbmV3IGZlYXR1cmVzIGFuZCBidWcgZml4ZXMKKiBGb2xsb3cgdGhlIGV4aXN0aW5nIHN0eWxlCiogU2VwYXJhdGUgdW5yZWxhdGVkIGNoYW5nZXMgaW50byBtdWx0aXBsZSBwdWxsIHJlcXVlc3RzCiogU2VlIHRoZSBleGlzdGluZyBpc3N1ZXMgZm9yIHRoaW5ncyB0byBzdGFydCBjb250cmlidXRpbmcuCgpGb3IgYmlnZ2VyIGNoYW5nZXMsIG1ha2Ugc3VyZSB5b3Ugc3RhcnQgYSBkaXNjdXNzaW9uIGZpcnN0IGJ5IGNyZWF0aW5nIGFuIGlzc3VlIGFuZCBleHBsYWluaW5nIHRoZSBpbnRlbmRlZCBjaGFuZ2UuCgpBdGxhc3NpYW4gcmVxdWlyZXMgY29udHJpYnV0b3JzIHRvIHNpZ24gYSBDb250cmlidXRvciBMaWNlbnNlIEFncmVlbWVudCwga25vd24gYXMgYSBDTEEuIFRoaXMgc2VydmVzIGFzIGEgcmVjb3JkIHN0YXRpbmcgdGhhdCB0aGUgY29udHJpYnV0b3IgaXMgZW50aXRsZWQgdG8gY29udHJpYnV0ZSB0aGUgY29kZS9kb2N1bWVudGF0aW9uL3RyYW5zbGF0aW9uIHRvIHRoZSBwcm9qZWN0IGFuZCBpcyB3aWxsaW5nIHRvIGhhdmUgaXQgdXNlZCBpbiBkaXN0cmlidXRpb25zIGFuZCBkZXJpdmF0aXZlIHdvcmtzIChvciBpcyB3aWxsaW5nIHRvIHRyYW5zZmVyIG93bmVyc2hpcCkuCgpQcmlvciB0byBhY2NlcHRpbmcgeW91ciBjb250cmlidXRpb25zIHdlIGFzayB0aGF0IHlvdSBwbGVhc2UgZm9sbG93IHRoZSBhcHByb3ByaWF0ZSBsaW5rIGJlbG93IHRvIGRpZ2l0YWxseSBzaWduIHRoZSBDTEEuIFRoZSBDb3Jwb3JhdGUgQ0xBIGlzIGZvciB0aG9zZSB3aG8gYXJlIGNvbnRyaWJ1dGluZyBhcyBhIG1lbWJlciBvZiBhbiBvcmdhbml6YXRpb24gYW5kIHRoZSBpbmRpdmlkdWFsIENMQSBpcyBmb3IgdGhvc2UgY29udHJpYnV0aW5nIGFzIGFuIGluZGl2aWR1YWwuCgoqIFtDTEEgZm9yIGNvcnBvcmF0ZSBjb250cmlidXRvcnNdKGh0dHBzOi8vbmEyLmRvY3VzaWduLm5ldC9NZW1iZXIvUG93ZXJGb3JtU2lnbmluZy5hc3B4P1Bvd2VyRm9ybUlkPWUxYzE3YzY2LWNhNGQtNGFhYi1hOTUzLTJjMjMxYWY0YTIwYikKKiBbQ0xBIGZvciBpbmRpdmlkdWFsc10oaHR0cHM6Ly9uYTIuZG9jdXNpZ24ubmV0L01lbWJlci9Qb3dlckZvcm1TaWduaW5nLmFzcHg/UG93ZXJGb3JtSWQ9M2Y5NGZiZGMtMmZiZS00NmFjLWIxNGMtNWQxNTI3MDBhZTVkKQo= readmeEtag: '"fbe104e559edcaec87bb5d1b87b2a92b715efe59"' readmeLastModified: Mon, 22 Sep 2025 01:28:24 GMT repositoryId: 122282653 description: Request and response validator against OpenAPI Specification 3 created: '2018-02-21T02:16:51Z' updated: '2025-09-22T01:28:28Z' language: TypeScript archived: false stars: 44 watchers: 1 forks: 24 owner: atlassian logo: https://avatars.githubusercontent.com/u/168166?v=4 license: NOASSERTION repoEtag: '"fd323215b2ab8033ba934188e7107c11056ee989799a9fb236bdfff54e04465b"' repoLastModified: Mon, 22 Sep 2025 01:28:28 GMT foundInMaster: true id: 70270b46839e1466d6a425a8da6ded65 - source: openapi3 tags repository: https://github.com/rpstreef/openapi-tf-example v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHdpdGggQVdTIEFQSSBHYXRld2F5LCBMYW1iZGEsIENvZ25pdG8sIFNOUyBhbmQgQ2xvdWRXYXRjaCBsb2dzCgpUaGlzIHJlcG8gb25seSBkZXBsb3lzIHRoZSBpbmZyYXN0cnVjdHVyZSB2aWEgVGVycmFmb3JtLiBUaGUgc291cmNlIGNvZGUsIFt0aGlzXShodHRwczovL2dpdGh1Yi5jb20vcnBzdHJlZWYvb3BlbmFwaS1ub2RlLWV4YW1wbGUpIHJlcG8sIHdpbGwgYmUgZGVwbG95ZWQgYXV0b21hdGljYWxseSB2aWEgQVdTIENvZGVQaXBlbGluZS4gQnkgZGVmYXVsdCwgaXQncyBjb25maWd1cmVkIHRvIGF1dG9tYXRpY2FsbHkgZGVwbG95IGF0IGV2ZXJ5IHB1c2ggdG8gdGhlIG1hc3RlciBicmFuY2guCgpDaGVjayB0aGUgY29tcGFuaW9uIGFydGljbGVzIHNlcmllcyAnX09wZW5BUElfJyBvbiBbZGV2LnRvXShodHRwczovL2Rldi50by9yb2xmc3RyZWVma2Vyay9vcGVuYXBpLXdpdGgtdGVycmFmb3JtLW9uLWF3cy1hcGktZ2F0ZXdheS0xN2plKS4KCiMgR2V0IHN0YXJ0ZWQKCiMjIFRoZSBlc3NlbnRpYWxzCgotIERvd25sb2FkIFRlcnJhZm9ybSB2MC4xMi54IFtoZXJlXShodHRwczovL3d3dy50ZXJyYWZvcm0uaW8vZG93bmxvYWRzLmh0bWwpCi0gWW91IHdpbGwgbmVlZCBOb2RlIHYxMi54IGZyb20gW2hlcmVdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi9kb3dubG9hZC8pCi0gR2l0LCB0byBjbG9uZSB0aGlzIFJlcG8sIGZyb20gW2hlcmVdKGh0dHBzOi8vZ2l0LXNjbS5jb20vZG93bmxvYWRzKQotIENyZWF0ZSBhIGZyZWUgQVdTIGFjY291bnQgKHJlcXVpcmVzIGNyZWRpdCBjYXJkKSBbaGVyZV0oaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS8pCi0gRmluYWxseSwgZG93bmxvYWQgdGhlIFtBV1MgQ0xJIHRvb2xdKGh0dHBzOi8vYXdzLmFtYXpvbi5jb20vY2xpLykgCi0gU2V0dXAgeW91ciBBV1MgbG9jYWwgcHJvZmlsZSwgc2VlIFt0aGlzXShodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vY2xpL2xhdGVzdC91c2VyZ3VpZGUvY2xpLWNvbmZpZ3VyZS1wcm9maWxlcy5odG1sKSBndWlkZSBob3cgaXQncyBkb25lLgotIE1hbnVhbGx5IHNldHVwIGFuIEFXUyBTMyBCdWNrZXQgZm9yIFRlcnJhZm9ybSBzdGF0ZSBzdG9yYWdlLgoKIyMgVG8gZ2V0IHRoZSBBUEkgcnVubmluZwpJZiB5b3UgbWVldCBhbGwgdGhlIHByZS1yZXF1aXNpdGVzLCBkbyB0aGUgZm9sbG93aW5nCgotIEluIHlvdXIgQVdTIGRldmVsb3BtZW50IGFjY291bnQgY3JlYXRlIHRoZSBTMyBidWNrZXQgZm9yIHlvdXIgVGVycmFmb3JtIHN0YXRlIGZpbGVzLgogIC0gT3B0aW9uYWxseSwgZW5jcnlwdCB0aGUgUzMgYnVja2V0IGFuZCBlbmFibGUgdmVyc2lvbmluZyBzdWNoIHRoYXQgeW91IGNhbiBkbyBhIHJvbGxiYWNrLgotIGBgYGdpdCBjbG9uZWBgYCB0aGlzIHJlcG8uCi0gQ2hhbmdlIHlvdXIgQVdTIGNyZWRlbnRpYWxzIHByb2ZpbGUgbmFtZSBpbiB0aGVzZSBmaWxlczogCiAgLSBgYGAuL2Vudi9kZXYvcmVtb3RlLWJhY2tlbmQudGZgYGAKICAtIGBgYC4vZW52L2Rldi9kZXYudGZ2YXJzYGBgCi0gUnVuIGBgYCBucG0gaW5zdGFsbCBgYGAgYW5kIHRoZW4gZXhlY3V0ZSBgYGAgbnBtIHJ1biBkZXYtaW5pdCBgYGAsIHRoaXMgd2lsbDoKICAtIEluaXRpYWxpemUgdGhlIFRlcnJhZm9ybSBwcm9qZWN0IGZvciB0aGUgJ2RldicgZW52aXJvbm1lbnQsIGFuZCBzeW5jaHJvbml6ZSB0aGUgc3RhdGUgd2l0aCB0aGUgY2xvdWQgc3RvcmVkIC50ZnN0YXRlIGZpbGUuCiAgLSBJZiB5b3UgcnVuIGl0IGEgc2Vjb25kIHRpbWUsIGl0IHdpbGwgZmFpbCBvbiB0aGUgd29ya3NwYWNlIGNyZWF0aW9uLCB0aGlzIGlzIG5vdCBhbiBpc3N1ZSAodGhlIHdvcmtzcGFjZSBhbHJlYWR5IGV4aXN0cykKLSBSdW4gYGBgbnBtIHJ1biBkZXYtaW5mcmFgYGAgdG8gcHJlcGFyZSB0aGUgZGVwbG95bWVudCB0byB5b3VyIEFXUyBhY2NvdW50LgogIC0gTm90ZTogdGhpcyBbcmVwb10oaHR0cHM6Ly9naXRodWIuY29tL3Jwc3RyZWVmL29wZW5hcGktbm9kZS1leGFtcGxlKSBjb250YWlucyB0aGUgbm9kZSBzb3VyY2UgY29kZSB0aGF0IHRoZSBDb2RlUGlwZWxpbmUgd2lsbCBkZXBsb3kgdG8gdGhlIEFXUyBMYW1iZGEgYW5kIExhbWJkYS1sYXllci4KICAtIENvbmZpcm0gd2l0aCBgYGB5ZXNgYGAgdG8gZGVwbG95LCBhbnl0aGluZyBlbHNlIHdpbGwgY2FuY2VsIHRoZSBkZXBsb3ltZW50CgpTZWUgbXkgZnVsbCBndWlkZSBvbiBkZXYudG8gZm9yIG1vcmUgaW5mb3JtYXRpb24gYWJvdXQgdGhpcyBwcm9qZWN0Cgpwcy4gSSdtIGF3YXJlIHRoZSBndWxwIGZpbGUgZG9lc24ndCBleGl0IG5pY2VseSwgaSdtIG5vdCBhbiBleHBlcnQuIEFueSBwdWxsIHJlcXVlc3RzIG9yIGlzc3VlIHJlcG9ydHMgd2l0aCBwb2ludGVycyBpcyBkZWZpbml0ZWx5IGFwcHJlY2lhdGVkLgoKIyMgVlMgQ29kZSBwbHVnaW5zIHVzZWQKCi0gW1N0YW5kYXJkSlNdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1jaGVueHNhbi52c2NvZGUtc3RhbmRhcmRqcykKLSBbVGVycmFmb3JtXShodHRwczovL21hcmtldHBsYWNlLnZpc3VhbHN0dWRpby5jb20vaXRlbXM/aXRlbU5hbWU9bWF1dmUudGVycmFmb3JtKQotIFtPcGVuQVBJIEVkaXRvcl0oaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPTQyQ3J1bmNoLnZzY29kZS1vcGVuYXBpKQotIFtPcGVuQVBJIERlc2lnbmVyXShodHRwczovL21hcmtldHBsYWNlLnZpc3VhbHN0dWRpby5jb20vaXRlbXM/aXRlbU5hbWU9cGhpbG9zb3dhZmZsZS5vcGVuYXBpLWRlc2lnbmVyKQoKIyMgUnVubmluZyBjb3N0cwoKVGhlcmUgYXJlIG5vIGNvc3RzIGFzc29jaWF0ZWQgd2l0aCBkZXBsb3lpbmcgYW55IG9mIHRoaXMgb24gQVdTLCB0aGVyZSBpcyBbRnJlZSBUaWVyXShodHRwczovL2F3cy5hbWF6b24uY29tL2ZyZWUpIGNvdmVyYWdlIGZvciBsaW1pdGVkIGZyZWUgdXNlLgoKVGhlIGZvbGxvd2luZyBzZXJ2aWNlcyBhcmUgZGVwbG95ZWQgd2l0aCBUZXJyYWZvcm07Ci0gQVdTIENvZ25pdG8KLSBBV1MgQVBJIEdhdGV3YXkKLSBBV1MgTGFtYmRhICYgTGFtYmRhIExheWVyCi0gQVdTIElBTQotIChBZGRlZCkgQ2xvdWRXYXRjaCBBbGFybXMsIGNvc3RzIHdpbGwgYmUgaW5jdXJyZWQgZm9yIGVuYWJsaW5nIERldGFpbGVkIE1vbml0b3JpbmcgZm9yIEFQSSBHYXRld2F5ICghKQotIChBZGRlZCkgQVdTIENvZGVQaXBlbGluZSwgYW5kIENvZGVCdWlsZCB3aXRoIEdpdGh1YiBhcyBzb3VyY2UgcmVwb3NpdG9yeS4gVGhlcmUncyBhIGZyZWUgdGllciBmb3I6CiAgLSBbQ29kZUJ1aWxkXShodHRwczovL2F3cy5hbWF6b24uY29tL2NvZGVidWlsZC9wcmljaW5nLyksIDEwMCBidWlsZCBtaW51dGVzIG9mIGBgYGJ1aWxkLmdlbmVyYWwxLnNtYWxsYGBgIHBlciBtb250aC4KICAtIFtBV1MgQ29kZVBpcGVsaW5lXShodHRwczovL2F3cy5hbWF6b24uY29tL2NvZGVwaXBlbGluZS9wcmljaW5nLyk6IDEgZnJlZSBwaXBlbGluZSBhY3RpdmUgcGVyIG1vbnRoLiA= readmeEtag: '"e1dbe8e92d6a3be3e4dc1937bec8d38439974d7c"' readmeLastModified: Mon, 23 Nov 2020 07:58:54 GMT repositoryId: 236130695 description: >- Example of how you can use OpenAPI with AWS API Gateway, Also includes integrations with AWSLambda, AWS Cognito, AWS SNS and CloudWatch logs created: '2020-01-25T05:40:11Z' updated: '2025-10-15T14:03:30Z' language: HCL archived: false stars: 44 watchers: 3 forks: 19 owner: rpstreef logo: https://avatars.githubusercontent.com/u/15830262?v=4 license: Apache-2.0 repoEtag: '"72c39d98af809f185e0686403730bf94c73369f33e5d7376cfbd0ace20ce35ec"' repoLastModified: Wed, 15 Oct 2025 14:03:30 GMT foundInMaster: true category: - Description Validators - Server Implementations id: d88e9e79cc86cd11423f309b56d3accc - source: openapi3 tags repository: https://github.com/benirave/aspnetcore.scalar v3: true id: 2dbfc9923a0675a6a8269f3d0af9b9da repositoryMetadata: base64Readme: >- WyFbTnVnZXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnVnZXQvdi9hc3BuZXRjb3JlLnNjYWxhcildKGh0dHBzOi8vd3d3Lm51Z2V0Lm9yZy9wYWNrYWdlcy9Bc3BOZXRDb3JlLlNjYWxhcikKCiMgLk5FVCBTY2FsYXIgQVBJIFJlZmVyZW5jZQoKQWRkIHRoZSBbU2NhbGFyIEFQSSBSZWZlcmVuY2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9zY2FsYXIvc2NhbGFyP3RhYiUyNTNEcmVhZG1lLW92LWZpbGUjd2l0aC1uZXh0anMpIHRvIGFueSBvZiB5b3VyIC5ORVQgYXBwbGljYXRpb25zLgoKPiBkb3RuZXQgYWRkIHBhY2thZ2UgQXNwTmV0Q29yZS5TY2FsYXIgLS12ZXJzaW9uIDEuMi4wCgojIyBVc2FnZQoKVG8gaW5jb3Jwb3JhdGUgdGhlIFNjYWxhciBVSSwgc2ltcGx5IGV4cG9zZSBhbiBPcGVuQVBJIHNjaGVtYSBieSB1c2luZyBbU3dhc2hidWNrbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9kb21haW5kcml2ZW5kZXYvU3dhc2hidWNrbGUuV2ViQXBpKSBvciBbTlN3YWddKGh0dHBzOi8vZ2l0aHViLmNvbS9SaWNvU3V0ZXIvTlN3YWcpLCBhbmQgdGhlbiBpbnZva2UgdGhlICoqYXBwLlVzZVNjYWxhcigpKiouCgpDb2RlIGV4YW1wbGUuOgoKYGBgY3NoYXJwCnVzaW5nIEFzcE5ldENvcmUuU2NhbGFyOwoKdmFyIGJ1aWxkZXIgPSBXZWJBcHBsaWNhdGlvbi5DcmVhdGVCdWlsZGVyKGFyZ3MpOwoKLy8gQWRkIHN3YWdnZXIgc2NoZW1hIGdlbmVyYXRvcgpidWlsZGVyLlNlcnZpY2VzLkFkZEVuZHBvaW50c0FwaUV4cGxvcmVyKCk7CmJ1aWxkZXIuU2VydmljZXMuQWRkU3dhZ2dlckdlbigpOwoKdmFyIGFwcCA9IGJ1aWxkZXIuQnVpbGQoKTsKCi8vIEFkZCBzd2FnZ2VyIG1pZGRsZXdhcmUKYXBwLlVzZVN3YWdnZXIoKTsKCi8vIEFkZCBTY2FsYXIgVUkKYXBwLlVzZVNjYWxhcihvcHRpb25zID0+CnsKICAgIG9wdGlvbnMuVXNlVGhlbWUoVGhlbWUuU29sYXJpemVkKTsKfSk7CgpgYGAKCkV4cGxvcmUgYSB3b3JraW5nIGV4YW1wbGUgW2hlcmVdKC4vZXhhbXBsZS8pLgoKIyMjIEN1c3RvbWl6YXRpb24gT3B0aW9ucwoKYEluamVjdFN0eWxlc2hlZXRgCgotICoqRGVzY3JpcHRpb24qKjogSW5qZWN0cyBhIHN0eWxlc2hlZXQgbGluayBpbnRvIHRoZSBoZWFkIGNvbnRlbnQgb2YgdGhlIFNjYWxhck9wdGlvbnMuCi0gKipQYXJhbWV0ZXJzKio6CiAgLSBgcGF0aGAgKHN0cmluZyk6IFRoZSBwYXRoIHRvIHRoZSBzdHlsZXNoZWV0IGZpbGUuCiAgLSBgbWVkaWFgIChzdHJpbmcsIG9wdGlvbmFsKTogVGhlIG1lZGlhIHR5cGUgZm9yIHdoaWNoIHRoZSBzdHlsZXNoZWV0IGlzIGludGVuZGVkLiBEZWZhdWx0IHZhbHVlIGlzICJzY3JlZW4iLgoKYFVzZVNwZWNVcmxgCgotICoqRGVzY3JpcHRpb24qKjogU2V0cyB0aGUgVVJMIGZvciB0aGUgc3BlY2lmaWNhdGlvbiBwYWdlLgotICoqUGFyYW1ldGVycyoqOgogIC0gYHVybGAgKHN0cmluZyk6IFRoZSBVUkwgb2YgdGhlIHNwZWNpZmljYXRpb24gcGFnZS4KCmBVc2VDZG5gCgotICoqRGVzY3JpcHRpb24qKjogU2V0cyB0aGUgQ0ROIFVSTCBmb3IgdGhlIGFzc2V0LgotICoqRGVmYXVsdCoqOiBodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvbnBtL0BzY2FsYXIvYXBpLXJlZmVyZW5jZQotICoqUGFyYW1ldGVycyoqOgogICAgLSBgdXJsYCAoc3RyaW5nKTogVGhlIFVSTCBvZiB0aGUgY2RuIGFzc2V0LgoKYFVzZVByb3h5YAoKLSAqKkRlc2NyaXB0aW9uKio6IFNldHMgdGhlIFByb3h5IFVSTCB0byB1c2UuCi0gKipQYXJhbWV0ZXJzKio6CiAgICAtIGB1cmxgIChzdHJpbmcpOiBUaGUgVVJMIG9mIHRoZSBwcm94eSBzZXJ2ZXIuCgoKYFVzZUxheW91dGAKCi0gKipEZXNjcmlwdGlvbioqOiBTZXRzIHRoZSBsYXlvdXQgZm9yIHRoZSBTY2FsYXJPcHRpb25zLiBZb3UgY2FuIHVzZSAnTW9kZXJuJyBhbmQgQ2xhc3NpYy4KLSAqKkRlZmF1bHQqKjogTW9kZXJuCi0gKipQYXJhbWV0ZXJzKio6CiAgLSBgbGF5b3V0YCAoVGhlbWUpOiBBbiBlbnVtIHJlcHJlc2VudGluZyB0aGUgbGF5b3V0IHRvIHVzZS4KCmBVc2VUaGVtZWAKCi0gKipEZXNjcmlwdGlvbioqOiBTZXRzIHRoZSB0aGVtZSBmb3IgdGhlIFNjYWxhck9wdGlvbnMuCi0gKipQYXJhbWV0ZXJzKio6CiAgLSBgdGhlbWVgIChUaGVtZSk6IEFuIGVudW0gcmVwcmVzZW50aW5nIHRoZSB0aGVtZSB0byB1c2UuCgpgSGlkZVNpZGViYXJgCgotICoqRGVzY3JpcHRpb24qKjogSGlkZXMgdGhlIHNpZGViYXIgaW4gdGhlIFNjYWxhck9wdGlvbnMuCi0gKipQYXJhbWV0ZXJzKio6IE5vbmUuCgpgVXNlU2VhcmNoQXNIb3RLZXlgCgotICoqRGVzY3JpcHRpb24qKjogU2V0cyB0aGUgc2VhcmNoIGhvdGtleSBmb3IgdGhlIFNjYWxhck9wdGlvbnMuCi0gKipQYXJhbWV0ZXJzKio6CiAgLSBgaG90S2V5YCAoY2hhcik6IFRoZSBjaGFyYWN0ZXIgcmVwcmVzZW50aW5nIHRoZSBob3RrZXkgZm9yIHNlYXJjaC4KCmBBZGRBZGRpdGlvbmFsSXRlbWAKCi0gKipEZXNjcmlwdGlvbioqOiBBZGRzIGFuIGFkZGl0aW9uYWwgaXRlbSB0byB0aGUgU2NhbGFyT3B0aW9ucyBjb25maWd1cmF0aW9uLgotICoqUGFyYW1ldGVycyoqOgogIC0gYGtleWAgKHN0cmluZyk6IFRoZSBrZXkgb2YgdGhlIGFkZGl0aW9uYWwgaXRlbS4KICAtIGB2YWx1ZWAgKG9iamVjdCk6IFRoZSB2YWx1ZSBvZiB0aGUgYWRkaXRpb25hbCBpdGVtLgoKVGhlc2UgY3VzdG9taXphdGlvbiBvcHRpb25zIHByb3ZpZGUgZGV2ZWxvcGVycyB3aXRoIGZsZXhpYmlsaXR5IGluIGNvbmZpZ3VyaW5nIHRoZSBTY2FsYXIgbGlicmFyeSB0byBzdWl0IHRoZWlyIHNwZWNpZmljIG5lZWRzIGFuZCBwcmVmZXJlbmNlcy4gRGV2ZWxvcGVycyBjYW4gdXRpbGl6ZSB0aGVzZSBvcHRpb25zIHRvIGVuaGFuY2UgdGhlIGZ1bmN0aW9uYWxpdHkgYW5kIGFwcGVhcmFuY2Ugb2YgdGhlaXIgYXBwbGljYXRpb25zIHNlYW1sZXNzbHkuCgojIyBDb21wYXRpYmlsaXR5Cgp8IEFzcE5ldENvcmUuU2NhbGFyIHwgU2NhbGFyIHwKfC0tLS0tLS0tLS0tLS0tLS0tLS18Oi0tLS0tLTp8CnwgMS4wLjAgICAgICAgICAgICAgfCAxLjE3LjEyIHwKfCAxLjEuMCAgICAgICAgICAgICB8IDEuMTkuMiB8CnwgMS4xLjEgICAgICAgICAgICAgfCAxLjIyLjQ4IHwKfCAxLjEuMiAgICAgICAgICAgICB8IDEuMjMuNSB8CnwgMS4xLjMgICAgICAgICAgICAgfCAxLjI0LjcyIHwKfCAxLjEuNCAgICAgICAgICAgICB8IDEuMjQuNzUgfAp8IDEuMS41ICAgICAgICAgICAgIHwgMS4yNS42IHwKfCAxLjEuNiAgICAgICAgICAgICB8IDEuMjUuOSB8CnwgMS4xLjcgICAgICAgICAgICAgfCAxLjI1LjE3IHwKfCAxLjEuOCAgICAgICAgICAgICB8IDEuMjUuNjMgfAp8IDEuMi4wICAgICAgICAgICAgIHwgMS4zNC42IHwKClRoaXMgcHJvamVjdCB3YXMgYmFzZWQgb24gW1N3YXNoYnVja2xlIFJlZG9jXShodHRwczovL2dpdGh1Yi5jb20vZG9tYWluZHJpdmVuZGV2L1N3YXNoYnVja2xlLkFzcE5ldENvcmUvdHJlZS9tYXN0ZXIvc3JjL1N3YXNoYnVja2xlLkFzcE5ldENvcmUuUmVEb2MpLgo= readmeEtag: '"f0e2fda3d3a514db0effefa8a13372590fe479b8"' readmeLastModified: Mon, 01 Sep 2025 19:37:00 GMT repositoryId: 720201668 description: Scalar API reference from Swagger/OpenAPI files for .NET🔥 created: '2023-11-17T20:04:30Z' updated: '2025-10-10T01:39:16Z' language: C# archived: false stars: 49 watchers: 1 forks: 3 owner: benirave logo: https://avatars.githubusercontent.com/u/23312114?v=4 license: MIT repoEtag: '"6845b5c024ad62cff267e92feb30d38ec4f2617987a97d8c4638b4e0ef95b22b"' repoLastModified: Fri, 10 Oct 2025 01:39:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/konfig-sdks/openapi-examples v3: true id: b6d4b2fc8b2568e91841ef29462ff558 repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KCiFbQmFubmVyXSguL2Jhbm5lci5wbmcpCgpbIVtMaWNlbnNlOiBNSVRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTGljZW5zZS1NSVQteWVsbG93LnN2ZyldKGh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUKQoKPC9kaXY+Cg== readmeEtag: '"732401a7dda3a77bdf43a476c7662df9902220bd"' readmeLastModified: Sun, 11 Aug 2024 00:27:02 GMT repositoryId: 748795976 description: >- The highest quality collection of up-to-date OpenAPI specifications for public APIs on the internet. This dataset also includes descriptions, categories, uptime metrics, and media assets for every API. created: '2024-01-26T19:30:01Z' updated: '2026-01-22T15:46:38Z' language: null archived: false stars: 65 watchers: 1 forks: 9 owner: konfig-sdks logo: https://avatars.githubusercontent.com/u/156027835?v=4 repoEtag: '"bfddbb481a0ae2cd1631317711e171d93421bd02066debfd607a6c64d9800e5a"' repoLastModified: Thu, 22 Jan 2026 15:46:38 GMT category: - Parsers - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/jaredce/serverless-openapi-documenter v3: true id: 73f90a4d2a5d00e00db21917b3fb11e5 repositoryMetadata: base64Readme: >- # OpenAPI Generator for serverless

<p>
  <a href="https://www.serverless.com">
    <img src="http://public.serverless.com/badges/v3.svg">
  </a>
  <a href="https://www.npmjs.com/package/serverless-openapi-documenter">
    <img src="https://img.shields.io/npm/v/serverless-openapi-documenter.svg?style=flat-square">
  </a>
  <a href="https://github.com/JaredCE/serverless-openapi-documenter/actions/workflows/node.yml">
    <img src="https://github.com/JaredCE/serverless-openapi-documenter/actions/workflows/node.yml/badge.svg">
  </a>
</p>

This will generate an OpenAPI V3 (up to v3.0.4) file for you from your serverless file. It can optionally generate a [Postman Collection V2](https://github.com/postmanlabs/openapi-to-postman) or (as of 0.0.120) [Bruno Collection](https://docs.usebruno.com/) from the OpenAPI file. This currently works for `http` and `httpApi` configurations.

Originally based off of: https://github.com/temando/serverless-openapi-documentation

## Install

This plugin works for Serverless (2.x, 3.x and 4.x) and only supports node.js 20 and up.

To add this plugin to your package.json:

**Using npm:**

```bash
npm install --save-dev serverless-openapi-documenter
```

Next you need to add the plugin to the `plugins` section of your `serverless.yml` file.

```yml
plugins:
  - serverless-openapi-documenter
```

> Note: Add this plugin _after_ `serverless-offline` to prevent issues with `String.replaceAll` being overridden incorrectly.

## Adding documentation to serverless

To Run: `serverless openapi generate -o openapi.json -f json -a 3.0.4 -p postman.json`

Options:

```
--output                -o  What filename the OpenAPI Description should output under. Default: openapi.json
--format                -f  Whether to output the OpenAPI Description as json or yaml. Default: json
--indent                -i  File indentation in spaces. Default: 2
--openApiVersion        -a  OpenAPI version to generate for. Default: 3.0.0
--postmanCollection     -p  Will generate a postman collection (from the generated OpenAPI Description), in json only, if passed in. Default: postman.json
--brunoCollection       -b  Will generate a Bruno collection (from the generated OpenAPI Description), in json only, if passed in. Default: bruno.json
--validationWarn        -w  Warn about validation errors only.  Will write the OpenAPI file if generation is successful.  Default: false
```

### README Highlighted Reading

#### Security Details

- [Security](#securityschemes)
- [Security on All Operations](#security-on-each-operation)
- [Security Per Operation](#security)

#### Model Details

- [Models](#models)
- [Notes on Schemas](#notes-on-schemas)
- [Request Schema Validators](#serverless-request-schema-validators)

#### Response Headers

- [CORS](#cors)
- [OWASP Secure Headers](#owasp)

#### Validation

- [Validation](#validator)

### OpenAPI Mapping

| OpenAPI field                                             | Serverless field                                                                                                                                      |
| --------------------------------------------------------- | ----------------------------------------------------------------------------------------------------------------------------------------------------- |
| info.title                                                | `custom.documentation.title` OR service                                                                                                               |
| info.description                                          | `custom.documentation.description` OR blank string                                                                                                    |
| info.version                                              | `custom.documentation.version` OR random v4 uuid if not provided                                                                                      |
| info.termsOfService                                       | `custom.documentation.termsOfService`                                                                                                                 |
| info.contact                                              | custom.documentation.contact                                                                                                                          |
| info.contact.name                                         | `custom.documentation.contact.name` OR blank string                                                                                                   |
| info.contact.url                                          | `custom.documentation.contact.url` if provided                                                                                                        |
| info.contact.x-                                           | `custom.documentation.contact.x-` extended specifications provided                                                                                    |
| info.license                                              | custom.documentation.license                                                                                                                          |
| info.license.name                                         | `custom.documentation.license.name` OR blank string                                                                                                   |
| info.license.url                                          | `custom.documentation.license.url` if provided                                                                                                        |
| info.license.x-                                           | `custom.documentation.license.x-` if extended specifications provided provided                                                                        |
| externalDocs.description                                  | `custom.documentation.externalDocumentation.description `                                                                                             |
| externalDocs.url                                          | `custom.documentation.externalDocumentation.url`                                                                                                      |
| x-tagGroups                                               | `custom.documentation.x-tagGroups` if provided                                                                                                        |
| security                                                  | `custom.documentation.security`                                                                                                                       |
| servers[].description                                     | custom.documentation.servers.description                                                                                                              |
| servers[].url                                             | custom.documentation.servers.url                                                                                                                      |
| servers[].variables                                       | custom.documentation.servers.variables                                                                                                                |
| tags[].name                                               | `custom.documentation.tags.name`                                                                                                                      |
| tags[].description                                        | `custom.documentation.tags.description`                                                                                                               |
| tags[].externalDocs.url                                   | `custom.documentation.tags.externalDocumentation.url`                                                                                                 |
| tags[].externalDocs.description                           | `custom.documentation.tags.externalDocumentation.description`                                                                                         |
| tags[].externalDocs.x-                                    | `custom.documentation.tags.externalDocumentation.x-` if extended specifications provided                                                              |
| path[path]                                                | functions.functions.events.[http OR httpApi].path                                                                                                     |
| path[path].servers[].description                          | functions.functions.servers.description                                                                                                               |
| path[path].servers[].url                                  | functions.functions.servers.url                                                                                                                       |
| path[path].[operation]                                    | functions.functions.[http OR httpApi].method                                                                                                          |
| path[path].[operation].summary                            | functions.functions.[http OR httpApi].documentation.summary                                                                                           |
| path[path].[operation].description                        | functions.functions.[http OR httpApi].documentation.description                                                                                       |
| path[path].[operation].operationId                        | functions.functions.[http OR httpApi].documentation.operationId OR functionName                                                                       |
| path[path].[operation].deprecated                         | functions.functions.[http OR httpApi].documentation.deprecated                                                                                        |
| path[path].[operation].externalDocs.description           | functions.functions.[http OR httpApi].documentation.externalDocumentation.description                                                                 |
| path[path].[operation].externalDocs.url                   | functions.functions.[http OR httpApi].documentation.externalDocumentation.url                                                                         |
| path[path].[operation].servers[].description              | functions.functions.[http OR httpApi].documentation.servers.description                                                                               |
| path[path].[operation].servers[].url                      | functions.functions.[http OR httpApi].documentation.servers.url                                                                                       |
| path[path].[operation].security                           | functions.functions.[http OR httpApi].documentation.security                                                                                          |
| path[path].[operation].deprecated                         | functions.functions.[http OR httpApi].documentation.deprecated                                                                                        |
| path[path].[operation].parameters                         | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params                                                                  |
| path[path].[operation].parameters.name                    | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.name                                                             |
| path[path].[operation].parameters.in                      | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params                                                                  |
| path[path].[operation].parameters.description             | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.description                                                      |
| path[path].[operation].parameters.required                | functions.functions.[http OR httpApi].documentation.[query/cookie/header]Params.required                                                              |
| path[path].[operation].parameters.deprecated              | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.deprecated                                                       |
| path[path].[operation].parameters.allowEmptyValue         | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.allowEmptyValue                                                  |
| path[path].[operation].parameters.style                   | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.style                                                            |
| path[path].[operation].parameters.explode                 | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.explode                                                          |
| path[path].[operation].parameters.allowReserved           | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.allowReserved                                                    |
| path[path].[operation].parameters.schema                  | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.schema                                                           |
| path[path].[operation].parameters.example                 | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.example                                                          |
| path[path].[operation].parameters.examples                | functions.functions.[http OR httpApi].documentation.[path/query/cookie/header]Params.examples                                                         |
| path[path].[operation].requestBody                        | functions.functions.[http OR httpApi].documentation.requestBody                                                                                       |
| path[path].[operation].requestBody.description            | functions.functions.[http OR httpApi].documentation.requestBody.description                                                                           |
| path[path].[operation].requestBody.required               | functions.functions.[http OR httpApi].documentation.requestBody.required                                                                              |
| path[path].[operation].requestBody.content                | functions.functions.[http OR httpApi].documentation.requestModels[contentType].name Links to custom.documentation.models.name                         |
| path[path].[operation].responses                          | functions.functions.[http OR httpApi].documentation.methodResponses                                                                                   |
| path[path].[operation].responses.[statusCode]             | functions.functions.[http OR httpApi].documentation.methodResponses[statusCode]                                                                       |
| path[path].[operation].responses.[statusCode].description | functions.functions.[http OR httpApi].documentation.methodResponses[statusCode].responseBody.description                                              |
| path[path].[operation].responses.[statusCode].content     | functions.functions.[http OR httpApi].documentation.methodResponses[statusCode].responseModels[contentType] Links to custom.documentation.models.name |

### Configuration

To configure this plugin to generate a valid OpenAPI Description, there are two places you'll need to modify in your `serverless.yml` file, the `custom` variables section and the `http/httpApi` event section for each given function in your service.

The `custom` section of your `serverless.yml` can be configured as below:

```yml
custom:
  documentation:
    version: "1"
    title: "My API"
    description: "This is my API"
    termsOfService: https://google.com
    externalDocumentation:
      url: https://google.com
      description: A link to google
    servers:
      url: https://example.com:{port}/
      description: The server
      variables:
        port:
          enum:
            - 4000
            - 3000
          default: 3000
          description: The port the server operates on
    tags:
      - name: tag1
        description: this is a tag
        externalDocumentation:
          url: https://npmjs.com
          description: A link to npm
    models: {}
```

Mostly everything here is optional. A version from a UUID will be generated for you if you don't specify one, title will be the name of your service if you don't specify one. You will need to specify the `documentation` top object.

#### termsOfService

Must be in the format of a url if included.

#### Contact

You can provide an optional contact object such as:

```yml
custom:
  documentation:
    contact:
      name: John
      url: https://example.com
      email: John@example.com
```

These fields are optional, though `url` needs to in the form of a URL and `email` needs to be in the format of an email address (ed: what that might be, I'm not 100% sure... go read the email RFC(s)).

This can be extended using the `^x-` specification extension.

#### License

You can provide an optional license object such as:

```yml
custom:
  documentation:
    license:
      name: Apache 2.0
      url: https://www.apache.org/licenses/LICENSE-2.0.html
```

Name is required but `url` is optional and must be in the format of a url.

This can be extended using the `^x-` specification extension.

#### Extended Fields

You can also add extended fields to the documentation object:

```yml
custom:
  documentation:
    x-other-field: This is an extended field
```

These fields must have `x-` before them, otherwise they will be ignored:

```yml
custom:
  documentation:
    other-field: This is an extended field
```

`other-field` here will not make it to the generated OpenAPI schema.

Currently extended specification fields defined under the `documentation` tag will sit under the OpenAPI `info` object e.g.

```yml
custom:
  documentation:
    title: myService
    x-other-field: This is an extended field
```

converts to:

```json
{
  "info": {
    "title": "myService",
    "x-other-field": "This is an extended field"
  }
}
```

An exception to this is Redocly `x-tagGroups`. If defined, they will sit at the root level of the OpenAPI specification, e.g.

```yml
custom:
  documentation:
    title: myService
    x-other-field: This is an extended field
    x-tagGroups:
      - name: Customers
        tags:
          - Customers
```

converts to:

```json
{
  "info": {
    "title": "myService",
    "x-other-field": "This is an extended field"
  },
  "x-tagGroups": [
    {
      "name": "Customers",
      "tags": ["Customers"]
    }
  ]
}
```

#### Moving documentation to a separate file

These configurations can be quite verbose; you can separate it out into it's own file, such as `serverless.doc.yml` as below:

```yml
custom:
  documentation: ${file(serverless.doc.yml):documentation}

functions:
  myFunc:
    events:
      - http:
          path: getStuff
          method: get
          documentation: ${file(serverless.doc.yml):endpoints.myFunc}
```

For more info on `serverless.yml` syntax, see their docs.

#### securitySchemes

You can provide optional Security Schemes:

```yml
custom:
  documentation:
    securitySchemes:
      my_api_key:
        type: apiKey
        name: api_key
        in: header
```

It accepts all available Security Schemes and follows the specification: https://spec.openapis.org/oas/v3.0.4#security-scheme-object

This can be extended using the `^x-` specification extension.

#### Security on each operation

To apply an overall security scheme to all of your operations without having to add the documentation to each one, you can write it like:

```yml
custom:
  documentation:
    securitySchemes:
      my_api_key:
        type: apiKey
        name: api_key
        in: header
    security:
      - my_api_key: []
```

This will apply the requirement of each operation requiring your `my_api_key` security scheme, [you can override this](#security).

#### Models

There are two ways to write the Models. Models contain additional information that you can use to define schemas for endpoints. You must define the _content type_ for each schema that you provide in the models.

The first way of writing the model is:
_required_ directives for the models section are as follow:

- `name`: the name of the schema
- `description`: a description of the schema
- `contentType`: the content type of the described request/response (ie. `application/json` or `application/xml`).
- `schema`: The JSON Schema ([website](http://json-schema.org/)) that describes the model. You can either use inline `YAML` to define these or use either an external file schema that serverless will resolve (as below), or a reference to an externally hosted schema that will be attempted to be resolved.

```yml
custom:
  documentation:
    models:
      - name: "ErrorResponse"
        description: "This is an error"
        contentType: "application/json"
        schema: ${file(models/ErrorResponse.json)}
      - name: "PutDocumentResponse"
        description: "PUT Document response model (external reference example)"
        contentType: "application/json"
        schema: ${file(models/PutDocumentResponse.json)}
      - name: "PutDocumentRequest"
        description: "PUT Document request model (inline example)"
        contentType: "application/json"
        schema:
          $schema: "http://json-schema.org/draft-04/schema#"
          type: object
          properties:
            SomeObject:
              type: "object"
              properties:
                SomeAttribute:
                  type: "string"
        examples:
          - name: someObjectInlineExample
            summary: an example of a request
            description: a longer string than the summary
            value: { SomeObject: { SomeAttribute: "attribute" } }
          - name: someObjectExternalExample
            summary: an example of a request external
            description: a longer string than the summary
            externalValue: https://example.com/external.json
```

The Second way of writing the models:

- `name`: the name of the schema
- `description`: a description of the schema
- `content`: an Object made up of the contentType and the schema, as shown below

```yml
custom:
  documentation:
    models:
      - name: "ErrorResponse"
        description: "This is an error"
        content:
          application/json:
            schema: ${file(models/ErrorResponse.json)}
      - name: "PutDocumentResponse"
        description: "PUT Document response model (external reference example)"
        content:
          application/json:
            schema: ${file(models/PutDocumentResponse.json)}
      - name: "PutDocumentRequest"
        description: "PUT Document request model (inline example)"
        content:
          application/json:
            schema:
              $schema: "http://json-schema.org/draft-04/schema#"
              type: object
              properties:
                SomeObject:
                  type: "object"
                  properties:
                    SomeAttribute:
                      type: "string"
            examples:
              - name: someObjectInlineExample
                summary: an example of a request
                description: a longer string than the summary
                value: { SomeObject: { SomeAttribute: "attribute" } }
              - name: someObjectExternalExample
                summary: an example of a request external
                description: a longer string than the summary
                externalValue: https://example.com/external.json
```

##### Model re-use

Through the magic of YAML, you can re-use models:

```yml
custom:
  documentation:
    ...
    models:
      - name: "ErrorResponse"
        description: "This is an error"
        content:
          application/json:
            schema: &ErrorItem
              type: object
              properties:
                message:
                  type: string
                code:
                  type: integer

      - name: "PutDocumentResponse"
        description: "PUT Document response model (external reference example)"
        content:
          application/json:
            schema:
              type: array
              items: *ErrorItem
```

`&ErrorItem` in the above example creates a node anchor (&ErrorItem) to the `ErrorResponse` schema which then can be used in the `PutDocumentResponse` schema via the reference (\*ErrorItem). The node anchor needs to be declared first before it can be used elsewhere via the reference, swapping the above example around would result in an error.

##### ModelsList - Backwards compatibility

It was brought to my attention that an older plugin version allowed the use of `modelsList`. As of 0.0.60, you can continue to use `modelsList` as well as using `models`, however `modelsList` now has to be nested within the `documentation` section. You can write `modelsList` the same way as any of the two styles for [Models](#Models).

```
custom:
  documentation:
    ...
    modelsList:
      - name: "ErrorResponse"
        description: "This is an error"
        content:
          application/json:
            schema:
              type: string
```

##### Serverless Request Schema Validators

As of 0.0.64, you can now make use of [Request Schema Validators](https://www.serverless.com/framework/docs/providers/aws/events/apigateway#request-schema-validators). This allows you to define Request models via the `apiGateway` settings:

```yml
provider:
  ...
  apiGateway:
    request:
      schemas:
        post-create-model:
          name: PostCreateModel
          schema: ${file(api_schema/post_add_schema.json)}
          description: "A Model validation for adding posts"
```

which are then used like:

```yml
functions:
  create:
    handler: posts.create
    events:
      - http:
          path: posts/create
          method: post
          request:
            schemas:
              application/json: post-create-model
          documentation: ...
```

The generator will match to the model within the `apiGateway` settings model list. If you are using the `apiGateway` to define models, please do not re-use any names that you might define in the [`models`](#models) list.

You can also skip writing a `requestBody` and `requestModels` if you have defined a `request` property in your event.

If you're not using `apiGateway`, you can still make use of `request` by writing in the other styles that serverless accepts for Request Schema Validators:

```yml
functions:
  create:
    handler: posts.create
    events:
      - http:
          path: posts/create
          method: post
          request:
            schemas:
              application/json:
                schema: ${file(create_request.json)}
                name: PostCreateModel
                description: "Validation model for Creating Posts"
```

or

```yml
functions:
  create:
    handler: posts.create
    events:
      - http:
          path: posts/create
          method: post
          request:
            schemas:
              application/json: ${file(create_request.json)}
```

#### Functions

To define the documentation for a given function event, you need to create a `documentation` attribute for your `http` or `httpApi` event in your `serverless.yml` file.

The `documentation` section of the event configuration can contain the following attributes:

- `summary`: A short description of the method
- `description`: A detailed description of the method
- `tags`: An array of tags for this event
- `deprecated`: Boolean indicator that indicates clients should migrate away from this function
- `requestBody`: Contains description of the request
  - `description`: A description of the request body
  - `required`: Whether the request body is required, defaults to false
- `requestModels`: A list of models to describe the request bodies (see [requestModels](#requestmodels) below)
- `queryParams`: A list of query parameters (see [queryParams](#queryparams) below)
- `pathParams`: A list of path parameters (see [pathParams](#pathparams) below)
- `cookieParams`: A list of cookie parameters (see [cookieParams](#cookieparams) below)
- `headerParams`: A list of headers (see [headerParams](#headerparams---request-headers) below)
- `security`: The security requirement to apply (see [security](#security) below)
- `methodResponses`: An array of response models and applicable status codes
  - `statusCode`: Applicable http status code (ie. 200/404/500 etc.)
  - `responseBody`: Contains description of the response
    - `description`: A description of the body response
  - `responseHeaders`: A list of response headers (see [responseHeaders](#responseheaders) below)
  - `responseModels`: A list of models to describe the request bodies (see [responseModels](#responsemodels) below) for each `Content-Type`

If you don't want a `http` or `httpApi` event to be documented, you can leave off the `documentation` object. The configuration schema will only check that you have specified a `methodResponses` on the `documentation` event, previously the plugin would cause serverless to warn or error (depending on your `configValidationMode`) if you had not supplied a `documentation` on an event.

```yml
functions:
  createUser:
    handler: handler.create
    events:
      - http:
        path: create
        method: post
        cors: true
        summary:
        documentation:
          summary: "Create User"
          description: "Creates a user and then sends a generated password email"
          tags:
            - tag1
          externalDocumentation:
            url: https://bing.com
            description: A link to bing
          requestBody:
            description: "A user information object"
          requestModels:
            application/json: "PutDocumentRequest"
          pathParams:
            - name: "username"
              description: "The username for a user to create"
              schema:
                type: "string"
                pattern: "^[-a-z0-9_]+$"
          queryParams:
            - name: "membershipType"
              description: "The user's Membership Type"
              schema:
                type: "string"
                enum:
                  - "premium"
                  - "standard"
          cookieParams:
            - name: "SessionId"
              description: "A Session ID variable"
              schema:
                type: "string"
          headerParams:
            name: "Content-Type"
            description: "The content type"
            schema:
              type: "string"
          methodResponses:
            - statusCode: 201
              responseBody:
                description: "A user object along with generated API Keys"
              responseModels:
                application/json: "PutDocumentResponse"
              responseHeaders:
                X-Rate-Limit-Limit:
                  description: The number of allowed requests in the current period
                  schema:
                    type: integer
            - statusCode: 500
              responseBody:
                description: "An error message when creating a new user"
              responseModels:
                application/json: "ErrorResponse"
```

#### `queryParams`

Query parameters can be described as follow:

- `name`: the name of the query variable
- `description`: a description of the query variable
- `required`: whether the query parameter is mandatory (boolean)
- `schema`: JSON schema (inline, file or externally hosted)

```yml
queryParams:
  - name: "filter"
    description: "The filter parameter"
    required: true
    schema:
      type: "string"
```

#### `pathParams`

Path parameters can be described as follow:

- `name`: the name of the path parameter
- `description`: a description of the path parameter
- `schema`: JSON schema (inline, file or externally hosted)

```yml
pathParams:
  - name: "usernameId"
    description: "The usernameId parameter"
    schema:
      type: "string"
```

#### `cookieParams`

Cookie parameters can be described as follow:

- `name`: the name of the cookie parameter
- `description`: a description of the cookie parameter
- `required`: whether the cookie parameter is mandatory (boolean)
- `schema`: JSON schema (inline, file or externally hosted)

```yml
cookieParams:
  - name: "sessionId"
    description: "The sessionId parameter"
    required: true
    schema:
      type: "string"
```

#### `headerParams` - Request Headers

Request Headers can be described as follow:

- `name`: the name of the header
- `description`: a description of the header
- `required`: whether the header is mandatory (boolean)
- `schema`: JSON schema (inline, file or externally hosted)

```yml
headerParams:
  - name: "Content-Type"
    description: "The content type"
    required: true
    schema:
      type: "string"
```

#### `security`

The `security` property allows you to specify the [Security Scheme](#securityschemes) to apply to the HTTP Request. If you have applied an `security` ([see Security on each operation](#security-on-each-operation)) then you can either leave this field off, or to override it with a different scheme you can write it like:

```yml
custom:
  documentation:
    securitySchemes:
      my_api_key:
        type: apiKey
        name: api_key
        in: header
      petstore_auth:
        type: oauth2
        flows:
          implicit:
            authorizationUrl: https://example.com/api/oauth/dialog
            scopes:
              write:pets: modify pets in your account
              read:pets: read your pets
    security:
      - my_api_key: []

functions:
  getData:
    events:
      - http:
          documentation:
            security:
              - petstore_auth:
                  - write:pets
                  - read:pets
```

If you have specified an `security` at the document root, but this HTTP Request should not apply any security schemes, you should set security to be an array with an empty object:

```yml
custom:
  documentation:
    securitySchemes:
      my_api_key:
        type: apiKey
        name: api_key
        in: header
    security:
      - my_api_key: []

functions:
  getData:
    events:
      - http:
          documentation:
            security:
              - {}
```

##### private

If you use the [private](https://www.serverless.com/framework/docs/providers/aws/events/apigateway#setting-api-keys-for-your-rest-api) property on your event:

```yml
functions:
  getData:
    events:
      - http:
          path: /
          method: get
          private: true
```

It will automatically setup an apiKey security scheme of `x-api-key` attached to that method. You don't need to add this to the [Security Scheme](#securityschemes) in the main documentation. If you have already added a Security Scheme of an `apiKey` with a name of `x-api-key`, it will associate with that key.

```yml
custom:
  documentation:
    securitySchemes:
      my_api_key:
        type: apiKey
        name: x-api-key
        in: header
    security:
      - my_api_key: []

functions:
  getData:
    events:
      - http:
          path: /
          method: get
          private: true
          documentation: ...
```

Will set the Security Scheme to `my_api_key` for that operation.

#### `requestModels`

The `requestModels` property allows you to define models for the HTTP Request of the function event. You can define a different model for each different `Content-Type`. You can define a reference to the relevant request model named in the `models` section of your configuration (see [Defining Models](#models) section).

```yml
requestModels:
  application/json: "CreateRequest"
  application/xml: "CreateRequestXML"
```

#### `methodResponses`

`methodResponses` is a mandatory property and should include the `responseBody` and `description` properties.

You can define the response schemas by defining properties for your function event.

For an example of a `methodResponses` configuration for an event see below:

```yml
methodResponse:
  - statusCode: 200
    responseBody:
      description: Success
    responseModels:
      application/json: "CreateResponse"
      application/xml: "CreateResponseXML"
    links:
      getDataLink:
        operation: getData
        description: The id created here can be used to get Data
        parameters:
          contentId: $response.body#/id
    responseHeaders:
      X-Rate-Limit-Limit:
        description: The number of allowed requests in the current period
        schema:
          type: integer
      X-Rate-Limit-Remaining:
        description: The number of remaining requests in the current period
        schema:
          type: integer
      X-Rate-Limit-Reset:
        description: The number of seconds left in the current period
        schema:
          type: integer
```

##### `responseModels`

The `responseModels` property allows you to define models for the HTTP Response of the function event. You can define a different model for each different `Content-Type`. You can define a reference to the relevant response model named in the `models` section of your configuration (see [Defining Models](#models) section).

```yml
responseModels:
  application/json: "CreateResponse"
  application/xml: "CreateResponseXML"
```

##### `links`

The `links` property allows you to define how operations are linked to each other:

```yml
links:
  linkName:
    operation: getContent
    description: The contentId created here can be used to get content
    parameters:
      contentId: $response.body#/contentId
```

Where we are specifying operation, this should map to the function name:

```yml
functions:
  createContent:
    events:
      - httpApi:
          path: /
          method: POST
          documentation: ...
  getContent:
    events:
      - http:
          path: /{contentId}
          method: POST
          documentation: ...
```

If our example link was attached to the **createContent** function, and we wanted the `contentId` that was created to be used on the **getContent** function in the `contentId` parameter, we'd specify the `operation` property as **getContent**. If however, you had specified an operationId in the documentation to override the automatically created one:

```yml
getContent:
  events:
    - http:
        path: /{contentId}
        method: POST
        documentation:
          operationId: getMyContent
```

You can refer to the `operationId` that you created.

You can read more about [links](https://swagger.io/docs/specification/links/) on the swagger.io site and in the [OpenAPI](https://spec.openapis.org/oas/v3.0.4#link-object) specification. They don't seem widely supported just yet, but perhaps they'll improve your documentation.

##### `responseHeaders`

The `responseHeaders` property allows you to define the headers expected in a HTTP Response of the function event. This should only contain a description and a schema, which must be a JSON schema (inline, file or externally hosted).

```yml
responseHeaders:
  X-Rate-Limit-Limit:
    description: The number of allowed requests in the current period
    schema:
      type: integer
```

###### CORS

You can automatically generate CORS response headers by setting `cors` at the function level. Serverless allows you to modify how CORS is setup, so you can have the default options with `cors: true`, or you can modify the settings as shown in the [serverless documentation for CORS](https://www.serverless.com/framework/docs/providers/aws/events/apigateway#enabling-cors).

The generator will interpret your settings for CORS and automatically add the response headers. If for whatever reason you wish to override these, you can set them via the above `responseHeaders` setting and it'll apply your overrides.

##### OWASP

You can make use of the [OWASP Secure Headers](https://owasp.org/www-project-secure-headers/#x-permitted-cross-domain-policies) to generate response headers. These are a selection of response headers with default values that OWASP recommends returning with your response to help secure your application.

The OWASP Secure Headers Project contains a set of recommended headers to return with recommended values, when generating the documentation, the generator will attempt to get the latest version of this document and apply the latest recommendations. If you do not allow outside connections, it will default to a version of recommendations from **2025-08-17 15:23:47 UTC**.

Like CORS, if you have already set any of the OWASP Secure headers via `responseHeaders`, it will not overwrite them.

To make use of OWASP Secure Headers, you can use the following:

###### All OWASP Secure Headers

```yml
methodResponse:
  - statusCode: 200
    responseBody:
      description: Success
    responseModels:
      application/json: "CreateResponse"
    owasp: true
```

This will use the full set of OWASP Secure Headers and their recommended values. Some of these might not be appropriate for your application.

###### Subset of OWASP Secure Headers

```yml
methodResponse:
  - statusCode: 200
    responseBody:
      description: Success
    responseModels:
      application/json: "CreateResponse"
    owasp:
      cacheControl: true
      referrerPolicy: true
```

This will set only the `cacheControl` and `referrerPolicy` response header with the default recommendations.

The full list of OWASP Secure Headers you can set are:

- cacheControl - Cache-Control,
- clearSiteData - Clear-Site-Data,
- contentSecurityPolicy - Content-Security-Policy,
- crossOriginEmbedderPolicy - Cross-Origin-Embedder-Policy,
- crossOriginOpenerPolicy - Cross-Origin-Opener-Policy,
- crossOriginResourcePolicy - Cross-Origin-Resource-Policy,
- permissionsPolicy - Permissions-Policy,
- referrerPolicy - Referrer-Policy,
- strictTransportSecurity - Strict-Transport-Security,
- xContentTypeOptions - X-Content-Type-Options,
- xFrameOptions - X-Frame-Options,
- xPermittedCrossDomainPolicies - X-Permitted-Cross-Domain-Policies
- xDNSPrefetchControl - X-DNS-Prefetch-Control

You should note that `Pragma` has been [deprecated by owasp](https://owasp.org/www-project-secure-headers/#pragma), this plugin will issue a warning when you are still using Pragma and might drop support.

###### Subset of OWASP Secure Headers with user defined values

If you wish to override the OWASP Secure Headers, you can write your `methodResponse` like:

```yml
methodResponse:
  - statusCode: 200
    responseBody:
      description: Success
    responseModels:
      application/json: "CreateResponse"
    owasp:
      cacheControl:
        value: no-store
```

This will set the `Cache-Control` Response Header to have a value of "no-store" rather than any value the OWASP Secure Headers Project recommends.

## Validator

Validation for the OpenAPI Description is now (as of 0.0.90) done by [Redocly](https://redocly.com/). This is a slightly less opinionated validator for an OpenAPI Description, it should result in less errors around "YAML Anchors". It's also a maintained library, and has support for OpenAPI 3.1.0 which I hope to be able to support very soon.

I am making use of https://www.npmjs.com/package/@redocly/openapi-core, which I have been warned is likely to change. If you notice anything going wrong with validation of your OpenAPI Description, feel free to open an issue here. I make active use of this library, so will hopefully come across those issues too.

### Rules

I have configured the validator to use these Rules:

- [struct](https://redocly.com/docs/cli/rules/oas/struct)
- [path-parameters-defined](https://redocly.com/docs/cli/rules/oas/path-parameters-defined)
- [operation-2xx-response](https://redocly.com/docs/cli/rules/oas/operation-2xx-response)
- [operation-4xx-response](https://redocly.com/docs/cli/rules/oas/operation-4xx-response)
- [operation-operationId-unique](https://redocly.com/docs/cli/rules/oas/operation-operationId-unique)
- [path-declaration-must-exist](https://redocly.com/docs/cli/rules/oas/path-declaration-must-exist)

However, you can configure your own rules from the [ruleset available on the Redocly site](https://redocly.com/docs/cli/rules/built-in-rules/). To do this, you will need to create a `redocly.json` file within an `options` folder. The file should look like:

```json
{
  "struct": "error",
  "path-parameters-defined": "error",
  "operation-2xx-response": "error",
  "operation-4xx-response": "error",
  "operation-operationId-unique": "error",
  "path-declaration-must-exist": "error"
}
```

Since rules can be set to "warn", you no longer are required to tell the plugin to ignore errors with the `--validationWarn` flag.

## Example configuration

Please view the example [serverless.yml](test/serverless-tests/best/serverless.yml).

## Notes on schemas

Schemas can be either: inline, in file or externally hosted. If they're inline or in file, the plugin will attempt to normalise the schema to [OpenAPI 3.0.X specification](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.0.md#schemaObject).

If they exist as an external reference, for instance:

```yaml
schema: https://raw.githubusercontent.com/SchemaStore/schemastore/master/src/schemas/json/bettercodehub.json
```

We use the plugin [JSON Schema $Ref Parser](https://apitools.dev/json-schema-ref-parser/) to attempt to parse and resolve the references. There are limitations to this. Consider the schema:

```json
{
  "$schema": "https://json-schema.org/draft-04/schema",
  "title": "Reusable Definitions",
  "type": "object",
  "id": "https://raw.githubusercontent.com/json-editor/json-editor/master/tests/fixtures/definitions.json",
  "definitions": {
    "address": {
      "title": "Address",
      "type": "object",
      "properties": {
        "street_address": { "type": "string" },
        "city": { "type": "string" },
        "state": { "type": "string" }
      },
      "required": ["street_address"]
    },
    "link": { "$refs": "./properties.json#/properties/title" }
  },
  "properties": {
    "address": { "$refs": "#/definitions/address" }
  }
}
```

Where the definition "link" refers to a schema held in a directory that the resolver does not know about, we will not be able to fully resolve the schema which will likely cause errors in validation of the OpenAPI 3.0.X Description.

Because of the dependency we use to parse externally linked schemas, we can supply our own options to resolve schemas that are more difficult than a straight forward example.

You can create your own options file: https://apitools.dev/json-schema-ref-parser/docs/options.html to pass into the dependency that contains it's own resolver to allow you to resolve references that might be in hard to reach places. In your main project folder, you should have a folder called `options` with a file called `ref-parser.js` that looks like:

```js
"use strict";

// options from: https://apitools.dev/json-schema-ref-parser/docs/options.html

module.exports = {
  continueOnError: true, // Don't throw on the first error
  parse: {
    json: false, // Disable the JSON parser
    yaml: {
      allowEmpty: false, // Don't allow empty YAML files
    },
    text: {
      canParse: [".txt", ".html"], // Parse .txt and .html files as plain text (strings)
      encoding: "utf16", // Use UTF-16 encoding
    },
  },
  resolve: {
    file: false, // Don't resolve local file references
    http: {
      timeout: 2000, // 2 second timeout
      withCredentials: true, // Include auth credentials when resolving HTTP references
    },
  },
  dereference: {
    circular: false, // Don't allow circular $refs
    excludedPathMatcher: (
      path // Skip dereferencing content under any 'example' key
    ) => path.includes("/example/"),
  },
};
```

If you don't supply this file, it will use the default options.

## License

MIT
 readmeEtag: '"15d5bcc0a6d02636e7eb65dc4ae1eb6b6375414a"' readmeLastModified: Mon, 15 Dec 2025 12:25:20 GMT repositoryId: 495583813 description: >- Serverless plugin to export your config as OpenAPI v3 Documentation and Postman Collection V2 created: '2022-05-23T21:49:03Z' updated: '2026-01-29T13:47:07Z' language: JavaScript archived: false stars: 46 watchers: 2 forks: 18 owner: JaredCE logo: https://avatars.githubusercontent.com/u/770400?v=4 license: MIT repoEtag: '"6b5e00d06a1ee3af079dffe3831d2af86470a91aa4b5946b0aec1b78c5320d58"' repoLastModified: Thu, 29 Jan 2026 13:47:07 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/vuongtlt13/full-stack-fastapi-mysql v3: true repositoryMetadata: base64Readme: >- IyBGdWxsIFN0YWNrIEZhc3RBUEkgVGVtcGxhdGUKCiMjIyBOb3RlOiBUaGlzIHByb2plY3QgYmFzZWQgb24gdGhlIHByb2plY3QgaHR0cHM6Ly9naXRodWIuY29tL2Zhc3RhcGkvZnVsbC1zdGFjay1mYXN0YXBpLXRlbXBsYXRlLCBidXQgdXNlICoqTXlTUUwqKiBpbnN0ZWFkIFBvc3RncmVTUUwKCiMjIERldmVsb3BtZW50CiMjIyAxLiBTdGFydCBpbmZyYQotIEluIGBiYWNrZW5kYCBmb2xkZXIsIHJ1biBjb21tYW5kCmBgYGJhc2gKICAgIG1ha2Ugc3RhcnQtaW5mcmEKYGBgCiMjIyAyLiBTdGFydCBCYWNrZW5kCi0gSW4gYGJhY2tlbmRgIGZvbGRlciwgc2V0dXAgYC52ZW52YCB1c2luZyBgdXZgCi0gcnVuIGFsZW1iaWMgZm9yIGNyZWF0aW5nIG1pZ3JhdGlvbiBmaWxlIGJ5IHVzaW5nIGNvbW1hbmQKYGBgYmFzaAogICAgbWFrZSBtaWdyYXRpb24tZ2VuZXJhdGUgaW5pdGlhbGl6ZV9tb2RlbHMKYGBgCi0gQXBwbHkgdGhlIG1pZ3JhdGlvbiBhbmQgc2VlZCBmb3IgaW5pdGlhbCB1c2VyCmBgYGJhc2gKICBtYWtlIG1pZ3JhdGUtdXAgJiYgbWFrZSBkYi1zZWVkCmBgYAoKLSBTdGFydCBmYXN0YXBpIGFwcGxpY2F0aW9uCmBgYGJhc2gKICAgIGZhc3RhcGkgZGV2IGFwcC9tYWluLnB5CmBgYAoKIyMjIDMuIFN0YXJ0IEZyb250ZW5kCi0gSW4gYGZyb250ZW5kYCBmb2xkZXIsIGluc3RhbGwgZGVwZW5kZW5jaWVzCmBgYGJhc2gKICAgIG5wbSBpbnN0YWxsCmBgYAotIFJ1biBmcm9udGVuZCBhcHAKYGBgYmFzaAogICAgbnBtIHJ1biBkZXYKYGBgCgojIyBUZWNobm9sb2d5IFN0YWNrIGFuZCBGZWF0dXJlcwoKLSDimqEgWyoqRmFzdEFQSSoqXShodHRwczovL2Zhc3RhcGkudGlhbmdvbG8uY29tKSBmb3IgdGhlIFB5dGhvbiBiYWNrZW5kIEFQSS4KICAgIC0g8J+nsCBbU1FMTW9kZWxdKGh0dHBzOi8vc3FsbW9kZWwudGlhbmdvbG8uY29tKSBmb3IgdGhlIFB5dGhvbiBTUUwgZGF0YWJhc2UgaW50ZXJhY3Rpb25zIChPUk0pLgogICAgLSDwn5SNIFtQeWRhbnRpY10oaHR0cHM6Ly9kb2NzLnB5ZGFudGljLmRldiksIHVzZWQgYnkgRmFzdEFQSSwgZm9yIHRoZSBkYXRhIHZhbGlkYXRpb24gYW5kIHNldHRpbmdzIG1hbmFnZW1lbnQuCiAgICAtIPCfkr4gW015U1FMXShodHRwczovL3d3dy5teXNxbC5jb20vKSBhcyB0aGUgU1FMIGRhdGFiYXNlLgotIPCfmoAgW1JlYWN0XShodHRwczovL3JlYWN0LmRldikgZm9yIHRoZSBmcm9udGVuZC4KICAgIC0g8J+SgyBVc2luZyBUeXBlU2NyaXB0LCBob29rcywgVml0ZSwgYW5kIG90aGVyIHBhcnRzIG9mIGEgbW9kZXJuIGZyb250ZW5kIHN0YWNrLgogICAgLSDwn46oIFtDaGFrcmEgVUldKGh0dHBzOi8vY2hha3JhLXVpLmNvbSkgZm9yIHRoZSBmcm9udGVuZCBjb21wb25lbnRzLgogICAgLSDwn6SWIEFuIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkIGZyb250ZW5kIGNsaWVudC4KICAgIC0g8J+nqiBbUGxheXdyaWdodF0oaHR0cHM6Ly9wbGF5d3JpZ2h0LmRldikgZm9yIEVuZC10by1FbmQgdGVzdGluZy4KICAgIC0g8J+mhyBEYXJrIG1vZGUgc3VwcG9ydC4KLSDwn5CLIFtEb2NrZXIgQ29tcG9zZV0oaHR0cHM6Ly93d3cuZG9ja2VyLmNvbSkgZm9yIGRldmVsb3BtZW50IGFuZCBwcm9kdWN0aW9uLgotIPCflJIgU2VjdXJlIHBhc3N3b3JkIGhhc2hpbmcgYnkgZGVmYXVsdC4KLSDwn5SRIEpXVCAoSlNPTiBXZWIgVG9rZW4pIGF1dGhlbnRpY2F0aW9uLgotIPCfk6sgRW1haWwgYmFzZWQgcGFzc3dvcmQgcmVjb3ZlcnkuCi0g4pyFIFRlc3RzIHdpdGggW1B5dGVzdF0oaHR0cHM6Ly9weXRlc3Qub3JnKS4KLSDwn5OeIFtUcmFlZmlrXShodHRwczovL3RyYWVmaWsuaW8pIGFzIGEgcmV2ZXJzZSBwcm94eSAvIGxvYWQgYmFsYW5jZXIuCi0g8J+aoiBEZXBsb3ltZW50IGluc3RydWN0aW9ucyB1c2luZyBEb2NrZXIgQ29tcG9zZSwgaW5jbHVkaW5nIGhvdyB0byBzZXQgdXAgYSBmcm9udGVuZCBUcmFlZmlrIHByb3h5IHRvIGhhbmRsZSBhdXRvbWF0aWMgSFRUUFMgY2VydGlmaWNhdGVzLgotIPCfj60gQ0kgKGNvbnRpbnVvdXMgaW50ZWdyYXRpb24pIGFuZCBDRCAoY29udGludW91cyBkZXBsb3ltZW50KSBiYXNlZCBvbiBHaXRIdWIgQWN0aW9ucy4KCiMjIyBEYXNoYm9hcmQgTG9naW4KClshW0FQSSBkb2NzXShpbWcvbG9naW4ucG5nKV0oaHR0cHM6Ly9naXRodWIuY29tL2Zhc3RhcGkvZnVsbC1zdGFjay1mYXN0YXBpLXRlbXBsYXRlKQoKIyMjIERhc2hib2FyZCAtIEFkbWluCgpbIVtBUEkgZG9jc10oaW1nL2Rhc2hib2FyZC5wbmcpXShodHRwczovL2dpdGh1Yi5jb20vZmFzdGFwaS9mdWxsLXN0YWNrLWZhc3RhcGktdGVtcGxhdGUpCgojIyMgRGFzaGJvYXJkIC0gQ3JlYXRlIFVzZXIKClshW0FQSSBkb2NzXShpbWcvZGFzaGJvYXJkLWNyZWF0ZS5wbmcpXShodHRwczovL2dpdGh1Yi5jb20vZmFzdGFwaS9mdWxsLXN0YWNrLWZhc3RhcGktdGVtcGxhdGUpCgojIyMgRGFzaGJvYXJkIC0gSXRlbXMKClshW0FQSSBkb2NzXShpbWcvZGFzaGJvYXJkLWl0ZW1zLnBuZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9mYXN0YXBpL2Z1bGwtc3RhY2stZmFzdGFwaS10ZW1wbGF0ZSkKCiMjIyBEYXNoYm9hcmQgLSBVc2VyIFNldHRpbmdzCgpbIVtBUEkgZG9jc10oaW1nL2Rhc2hib2FyZC11c2VyLXNldHRpbmdzLnBuZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9mYXN0YXBpL2Z1bGwtc3RhY2stZmFzdGFwaS10ZW1wbGF0ZSkKCiMjIyBEYXNoYm9hcmQgLSBEYXJrIE1vZGUKClshW0FQSSBkb2NzXShpbWcvZGFzaGJvYXJkLWRhcmsucG5nKV0oaHR0cHM6Ly9naXRodWIuY29tL2Zhc3RhcGkvZnVsbC1zdGFjay1mYXN0YXBpLXRlbXBsYXRlKQoKIyMjIEludGVyYWN0aXZlIEFQSSBEb2N1bWVudGF0aW9uCgpbIVtBUEkgZG9jc10oaW1nL2RvY3MucG5nKV0oaHR0cHM6Ly9naXRodWIuY29tL2Zhc3RhcGkvZnVsbC1zdGFjay1mYXN0YXBpLXRlbXBsYXRlKQoKIyMgTGljZW5zZQoKVGhlIEZ1bGwgU3RhY2sgRmFzdEFQSSBUZW1wbGF0ZSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIE1JVCBsaWNlbnNlLgo= readmeEtag: '"fa33fb118d4d5124f26a4c4ad03995e2186c0bb6"' readmeLastModified: Mon, 24 Nov 2025 04:26:02 GMT repositoryId: 479628739 description: >- Full stack, modern web application generator. Using FastAPI, MySQL as database, Docker, automatic HTTPS and more. created: '2022-04-09T05:32:37Z' updated: '2026-02-04T18:23:56Z' language: TypeScript archived: false stars: 47 watchers: 2 forks: 14 owner: vuongtlt13 logo: https://avatars.githubusercontent.com/u/14292473?v=4 license: MIT repoEtag: '"2fabb46313d2153cf8bb9c3deba43db0236c9f7aa236026426eeb32eff5bd40b"' repoLastModified: Wed, 04 Feb 2026 18:23:56 GMT foundInMaster: true category: Server Implementations id: 288909aadff60d6d617f878d02cb018c - source: openapi3 tags repository: https://github.com/backbase/backbase-openapi-tools v3: true repositoryMetadata: base64Readme: >- ![Build](https://github.com/Backbase/backbase-openapi-tools/workflows/BOAT/badge.svg)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=com.backbase.oss%3Abackbase-openapi-tools&metric=alert_status)](https://sonarcloud.io/dashboard?id=com.backbase.oss%3Abackbase-openapi-tools)
[![Ask DeepWiki](https://deepwiki.com/badge.svg)](https://deepwiki.com/Backbase/backbase-openapi-tools)
# Backbase OpenApi Tools 

The Backbase Open API Tools is a collection of tools created to work efficiently with OpenAPI

It currently consists of

* Create Diff Report between 2 OpenAPI versions of the same spec (Based on https://github.com/quen2404/openapi-diff)
* Decompose Transformer to remove Composed Schemas from OpenAPI specs to aid in code generators
* Case Transformer to view how your API looks like when going from Camel Case to Snake Case  (transforms examples too)
* [Code Generator](boat-maven-plugin/README.md) based on [openapi-generator.tech](https://openapi-generator.tech/) with optimized templates and fixes.
* Lint mojo based on Zalando Zally and Backbase API

# Release Notes
BOAT is still under development and subject to change.
## 0.17.74
* Swift5: Removed deprecated initializer from model objects. The initializer is now internal and only accessible through the Builder pattern, preventing breaking code from deprecation warnings.

## 0.17.73
Added code related to generate webhook endpoints with different goal
To add prehook and posthook request mappings for all the requests in spec yaml using webhooks.
Created a new Mojo, added codegen and related templates under boat-webhooks.

## 0.17.70
Fix: Examples of responses defined in components/responses were not inlined by the ExamplesProcessor. 

## 0.17.69
Update query parameter handling in Swift5 template (#1177)

## 0.17.68
0.17.67 release publishing failed, 0.17.68 is the public release.

## 0.17.67
Update swift5 template to add support for exploded GET parameters (Swift only).

## 0.17.66
* Add support params of openapi-generator `name-mappings` and `enum-name-mappings`.
Check `Property & Enum Name Mappings (new)` in [boat-maven-plugin](boat-maven-plugin/README.md)
* fixed release process

## 0.17.62
* Fixed reactive generation of code to not include extra Flux<> to return parameter
## 0.17.61
* Added new `excludes` parameter into `bundle` execution to exclude certain files (default: `**/lib/**`) to avoid generating externally downloaded spec files.
## 0.17.60
* Allow references to parent directory of the spec itself, to be able to refer to common examples in a common lib.
* Fix tests... petshop example moved.
## 0.17.59
* Fix bug where java-spring generator puts `@Valid` annoation on return type of Api methods, cause service to blame
  client for invalid generated response with a 400 status.
## 0.17.58

## 0.17.57

## 0.17.56
* Added new `flattenOutput` parameter into `bundle` execution to generate the bundled APIs directly into `output` directory, even though the respective `input` files are located within a subdirectory in the `input`.
## 0.17.55
* Enhanced `boat-swift5` generator to cater for nested freeformObjects. This issue was identified in `ContentServicesApi`
## 0.17.54 and later
Starting from 0.17.54 the release notes are available at https://github.com/Backbase/backbase-openapi-tools/releases
## 0.17.53
* Added 2 Lint rules `B007U` and `B009U` for Unified Backbase API specs:
  * `B007U` checks whether paths do not contain `client-api`, `service-api` and `integration-api` prefixes. Any prefix is not allowed for Unified Backbase API.
  * `B009U` checks whether paths do not contain any version number. Any version number is not allowed for Unified Backbase API.
* These rules are ignored by default, but if you redefine the list of ignored rules in your project, then you need to add these two rules to your list.
## 0.17.52
* Lint rule `B014` fix reference link to component examples
## 0.17.46
* boat-scaffold
  * Enhanced ISO8601 Date Formatting with Fractional Seconds Support for Swift template.
    * The `OpenISO8601DateFormatter` template now supports parsing and formatting ISO8601 dates with none, one, or multiple fractional seconds in Swift 5. This enhancement provides greater flexibility and precision when working with date and time values, accommodating various use cases that require different levels of fractional second accuracy.
## 0.17.36
* Lint rule `B014` doesn't throw a null exception when parsing a string array property in a schema.  
## 0.17.35
* ISSUE: #776 add new lint rule `B014` - to validate if examples contain all defined properties in the schema
* BOAT now supports multiple access control permissions within the tag `x-BbAccessControls` in the OpenAPI spec. 
It contains two parameters `description` which describes the relationship between the permissions and `permissions` 
which is an array of permissions with tags `resource`, `function` and `privilege`. 
* Improve generated client Class Name
* Fix Jakarta native client generation options
## 0.17.30
* Angular 17
## 0.17.29
* Fixed Spring deprecations
## 0.17.24
* boat-spring
  * Fix [Inconsistent @Component for RestTemplate](https://github.com/Backbase/backbase-openapi-tools/issues/661)
## 0.17.23
* *Boat Engine*
  * Added support for byte array examples
* *Boat Scaffold*
  * Removed tag import in order to allow BOAT objects to have the name 'tag'
## 0.17.22
* boat-maven-plugin
  * Issue 211: Fix typo in changelogRenderer parameter in DiffMojo.java (#652) @talbot
## 0.17.21
* boat-spring, boat-java
  * Issue 649: missing BigDecimalCustomSerializer.java (#650) @walaniam
* BOAT Android (#637) @charbelmkh
## 0.17.18
* boat-spring
  * Fix [Regression on putting valid annotation on FQCN,https://github.com/Backbase/backbase-openapi-tools/issues/619]
  * Fix [Broken pojo field validations, https://github.com/Backbase/backbase-openapi-tools/issues/633]
  * Fix Add `@NotNull` for required properties.
  * Fix initiation of `required` property that has `additionalProperties` (generated as Map) when using `containerDefaultToNull=true`
  * Fix [`createApiComponent` option in springtemplate generator is ignored](https://github.com/Backbase/backbase-openapi-tools/issues/630)
  * Fix https://github.com/Backbase/backbase-openapi-tools/issues/635
  * Fix [List of Maps of Strings generated code does not compile. ,https://github.com/Backbase/backbase-openapi-tools/issues/647]
## 0.17.16
* BOAT Angular Generator
  * added `@angular/common` as peer dependency in the generated package.json files
  * replace usages of `isListContainer` with `isArray` in api.service template
## 0.17.15
* boat-spring
  * fix: incorrectly used enum param on collection items (see `collectionDataTypeParam.mustache`)
## 0.17.14
* boat-spring
  * added validation constraints on primitive collection items types in request params (see `collectionDataTypeParam.mustache`)
  * added validation constraints on primitive map value types (see `mapDataType.mustache`) 
## 0.17.13
* boat-spring
  * Fix: generate validation constraints on primitive collection items types (updates in pojo.mustache and new collectionDataType.mustache)
## 0.17.12
* BoatJavaCodeGen, BoatSpringCodeGen
  * Fix: Always generate collection initializer when array is required in the schema (even if containerDefaultToNull=true)
  * Fix: [validated pojo referenced class causes invalid code to be generated](https://github.com/Backbase/backbase-openapi-tools/issues/615)
  * Fix: boat-java pojo template to support `useWithModifier` to generate `withSomething` methods; fixes referencing 'common' classes generated using boat-spring templates.
## 0.17.11
* BoatJavaCodeGen, BoatSpringCodeGen
  * Fix: always generate collection initializer when array is required in the schema (even if containerDefaultToNull=true)
## 0.17.10
* Boat maven plugin
  * Fix: When using Multipart, generate with `@RequestPart` instead of `@RequestParam`
## 0.17.9
* Boat Maven plugin
  * Fix: [containerDefaultToNull=true is not backward compatible](https://github.com/Backbase/backbase-openapi-tools/issues/604)
## 0.17.8
* Boat Spring generator
  * Fix: [Serialization / deserialization issues of JsonSubTypes due to incorrect usage of JsonIgnoreProperties](https://github.com/Backbase/backbase-openapi-tools/issues/601)
## 0.17.7
* Boat maven plugin
  * Fix: Added 'containerDefaultToNull' default option to the MOJO generator
## 0.17.6
* Boat engine
  * Fix: Processing of a single string type example on media-type.
## 0.17.4
* Boat Angular generator
  * New format for Angular mocks, which are now export an array with responses.
## 0.17.3
* Boat Angular generator
  * Map the `Set` type to `Array` by default to avoid breaking changes vs clients generated with 0.16.x. 
## 0.17.2
* Boat Angular generator
  * Remove leading comment from typescript files so that there are fewer differences between files when regenerating with a new spec version (info moved to README)
  * Remove licence field from package.json
## 0.17.1
* Boat Angular generator
  * Added artifact information to generated package.json
## 0.17.0
* General
  * Removed RAML Support
  * Removed custom JavaScript client
  * Update OpenAPI Tools to 6.2.1
* Boat Java Generator
  * Jakarta EE 9 compatibility
  * `spring-mvc` library is removed because it is not supported by OpenAPI Tools anymore
  * Use of `Set` for unique items is now enabled by default as OpenAPI Generator fixed their implementation
## 0.16.15
* Boat engine
  * Fix: Processing of a single string type example on media-type.
* Boat Angular generator
  * New format for Angular mocks, which are now export an array with responses.
## 0.16.13
* Boat Angular generator
  * `@angular/common` added as a peer dependency in the generated package.json files
## 0.16.12
* Boat Angular generator
  * Remove leading comment from typescript files so that there are fewer differences between files when regenerating with a new spec version (info moved to README)
  * Remove licence field from package.json
## 0.16.11
* Boat Angular generator
  * Set `removeComments: false` in generated tsconfig.json files to retain `/*#__PURE__*/` annotation in compiled JS.
## 0.16.10
* Boat Angular generator
  * Remove types from generated mock providers due to a TypeScript which prevents `/*#__PURE__*/` annotation from working
## 0.16.9
* Boat Angular generator
  * Use of `/*#__PURE__*/` for `createMocks` function call in templates to enable tree-shaking
## 0.16.6
* Boat Angular generator
  * Use `^` instead of `>=` for `devDependencies` in the generated project, so project will be built using correct target version of ng-packagr & the Angular CLI
  * Declare some extra `devDependencies` at explicit versions to fix `npm install` issues in the generated project when using `npm` v8
## 0.16.5
* Boat Angular generator
  * Allow any format for spec versions.
## 0.16.4
* Boat Angular generator
  * Add ability to bake the spec version used in the package.json file.
## 0.16.3
* *Maven Plugin*
  * Do not treat spaces as delimiters in configuration properties which take a comma-separated list of values.
    Fixes an issue that prevented the use of tag names containing spaces in the `apisToGenerate` property.
## 0.16.2
* Boat Angular generator
  * Specify minimum Angular-related package versions when Angular v13 is used
## 0.16.1
* Boat Java generator
  * Java 17 compatibility update
## 0.16.0
* Boat Angular generator
  * Use `>=` for main peer dependency to improve forward compatibility
## 0.15.8
* *Boat Scaffold*
  * Makes sure to not autogenerate examples from schema if examples are already present
## 0.15.6
* *Maven Plugin*
  * Add example of use the additional properties like createApiComponent
* *Boat Angular generator*
  * Generate the `ng-package.json` file for use with the `ng-package` CLI
## 0.15.5
* *Boat Lint*
 * Bumped version of Zally Linting Libraries that contain several vulnerability fixes
## 0.15.4
* *Boat Angular generator*
  * Enable partial Ivy compilation for libraries.
## 0.15.3
* *Boat Scaffold*
 * A fix for mapping of Map for query params (in client code). 
## 0.15.2
* *Maven Plugin*
  * Fixed handling of absolute file paths for linting OpenAPI specs
  * Changed boat-bay dateLibrary for upload spec .
## 0.15.1
* *Maven Plugin*
  * Made `boat:radio` goal properties w.r.t boat-bay server unique.
## 0.15.0
* *Maven Plugin*
  * Added new goal `boat:radio`; see the description in the [plugin **documentation**](boat-maven-plugin/README.md#boatradio).
## 0.14.12
* *Boat Scaffold*
  * References to /examples/foo now are also dereferenced
## 0.14.10
* *Boat Scaffold*
  * Makes sure to URLDecode paths while dereferencing examples
## 0.14.9
* *Boat Scaffold*
  * Resolve references to other path operations' examples

## 0.14.8
* *Boat Marina*
  * Removes flatObjects as they are no longer needed
  * response.message was wrongfully escaped, escaping in the docs template instead
  * We no longer need to generate model aliases in boat marina
## 0.14.7
* *Boat Marina*
  * Exports flatObjects as a model to prevent errors with references
## 0.14.6 
* *Maven Plugin*
  * Added `boat:ship` mojo to automagically zip specs found in `src/main/resources` and attach it to the project reactor so `mvn install` or `mvn deploy` will also include a zip archive of the project with with `api` classifier
  * Changed `boat:bundle`  `includes` parameter to an array of patterns to allow recursive patters to discover OpenAPIs 
* *Boat Scaffold*
  * Fixed bug in typescript generator where only one of the overloaded function signatures per deprecated endpoint was marked as deprecated.


## 0.14.5
* *Boat Marina*
  * Added `boat-markers` directory in target to keep track of successful and unsuccessful actions on OpenAPI specs

## 0.14.4

* *Boat Marina*
  * Added a now BOAT Scaffold template called Marina, as that is where the models hang out. The Boat Marina template outputs a single JSON file that is used to offer a rich user interface built from the specs including search, page per operation and many more features!

* *Maven Plugin*
    * The boat:doc goal now recursively search OpenAPI specs in a directory to generate docs for each found spec.  

## 0.14.3

* *Maven Plugin*
  * Added new goal `boat:transform`; see the description in the [plugin documentation](boat-maven-plugin/README.md#boattransform).

## 0.14.2
* *Angular Generator*
  * Added support for Angular version ranges in peer dependencies
  
## 0.14.1
* *Angular Generator*
  * Added support for Angular 11
  
## 0.14.0
* *Angular Generator*
  * Simplify Angular generator options by removing the unused ones (withInterfaces,providedInRoot)
  * fix Mock is not generated if Http status equals to 201 (SDK-2388)

## 0.13.0
* *Lint*
  * Added rules. 
    * Check plurals on paths resource names. 
    * Check prefix for paths should contain version.
  * Enabled rules.
    * Use Standard HTTP Status Codes.

## 0.12.0
* *General*
  * Improved code quality
  * Added more unit tests
  * Added more realistic example projects in boat-maven-plugin
  * Added help:describe support for boat-maven-plugin
  * Fixed building in Windows 
* *Boat Docs*
  * Render response examples from response instead of schema object
* *Lint*
  * Added additional rules. 
    * Check x-icon value in the info block.
    * Check info block description.
    * Check tags allowed.
    * Check info block title.
    * Check prefix for paths. i.e. "client-api", "service-api", "integration-api"
  * Ignore Zalando Ruleset by default in boat-maven-plugin
  
## 0.11.4
* *Java Templates*
  * Correct the import and use of @Qualifier in ApiClient template
  * Only change base type when date useSetForUniqueNames is set true
    
* *Boat Docs*
  * Handle examples for MediaTypes without Schemas (such as text/csv)
  
## 0.11.3

* *Angular Generator*
  * Apply the correct return type when multiple responses are present
  * Generate mocks for examples defined in dereferenced schemas
  
## 0.11.1

* *Java Generator - boat-spring library*
  * Fixed reactive spring templates
  * Avoid importing `HttpServletResponse` when using reactive
* *Angular Generator*
  * Handle empty bodies properly in Mock generation
  * Update foundation-ang to latest version

## 0.11.0

* *Maven Plugin*
   * Added `removeExtensions` mojo parameter to `boat:bundle` to filter out the given vendor extensions from bundle.
   * Added `includes` mojo parameter to `boat:bundle` as a glob pattern selecting the specification files (defaults to `*.yaml`).
   * Added `apisToGenerate` mojo parameter to `boat:generate`
   * Set the default of `httpUserAgent` to `${project.artifactId}-${project.version}`.

* *Java Generator - resttemplate library*
  * Added `useWithModifiers` option to use the `with` prefix for POJO modifiers (defaults to `false`).
  * added `useSetForUniqueItems` to map arrays containing `uniqueItems` to `Set` (defaults to `false`).
  * Added `useClassLevelBeanValidation` option (defaults to `false`).
  * Added `useJacksonConversion` to use Jackson for parameters conversion instead of `toString` (defaults to `false`).
  * Added `restTemplateBeanName` to qualify the autowired RestTemplate bean.

* *Angular Generator*
  * Added an Angular client generator for version 10 and up. 
    The generator template is inherited from the standard one at [openapi-generator.tech](https://openapi-generator.tech/), with the addition of mock responses and a several fixes, among which:
    * Handling of reserved typescript words
    * Added support for Typescript and Javascript for escaping of strings in generators
    * Escaping of model properties when not using `camelCase`
    * Support for multiple `MediaTypes`
  * To enable mocks generation, set the `withMocks` option to `true`  

## 0.10.0
* *Maven Plugin*
   * `boat:lint` mojo will generate an HTML report based on API Guidelines 
   * `boat:docs` mojo will generate HTML documentation from OpenAPI showing multiple examples and requests as well as Custom Annotations
* General Bug Fixes
* Linting Rule Engine extended with reserved word linting
* **NOTE**: The lint rules are still in development. The documentation is still in the works. 

## 0.9.0
* *Maven Plugin*
  * Added `version` parameter to `bundle` goal.
  * Added `bundleSpecs` parameter to `generate` goal to automatically bundle specs into single file
* Modernised BOAT Terminal
* Improved BOAT:Docs Templates
* Properly dereference examples

## 0.8.0
* Improved styling HTML docs
* preview BOAT:QUAY linting mojo for linting OpenAPI specs.
* Avoid circular references when derefenencing OpenAPI specs

## 0.7.0
* Render multiple requests and examples in boat-docs
* Created HTML templates for boat-docs
* Pretty Print JSON Examples
* Added boat:doc mojo for generating beautiful HTML2 docs

* * Spring Generator*
  * Restored `HttpServletRequest` parameter (regression).
* Added boat:yard to create static website based on a collection of specs

## 0.6.0
* simple fix to check for null value in openApi.getComponents().getSchemas()
* ability to resolve references like #/components/schemas/myObject/items or #/components/schemas/myObject/properties/embeddedObject
* simple fix to avoid npe in StaticHtml2Generation escaping response message.

## 0.5.0

* Add DereferenceComponentsPropertiesTransformer (that does a bit extra)
* Fix recursive referencing in UnAliasTransformer

## 0.4.0
* Added bundle skip
* Changed numbering scheme

## 0.3.0

* *Maven Plugin*
  * Added `bundle.skip` parameter to `bundle` goal (defaults to false).

* *HTML2 Generator*
  * Removes examples
  * Adds title of API to the left navigation
  * Removes unnecessary spaces in the docs
  * Fixes item focus on left navigation
  * Updates Json Schema Ref Parser library
  * Updates Json schema view library
  * Adds support for allOf with Json schema merge all of https://github.com/mokkabonna/json-schema-merge-allof
  * Fixes header x- params being escaped. eg X-Total-Count to XMinusTotalMunisCount
  * Fixes markdown in description not being escaped and breaking javascript.
  * Fixes missing references to extended simple types (set `unAlias` option to true).
  * Fixes missing references because confusion over whether to reference name or classname.
  * Moved the code generation into a separate module to be used by other BOAT components.
  * Cleaning up dependencies
  * Added boat:bundle mojo to bundle fragments into a single spec.
  * boat:bundle unaliases the spec. 

* *Spring Generator*
  * Added `useWithModifiers` to use the `with` prefix for POJO modifiers (defaults to `false`; for compatibility with the old RAML generator must be set to `true`).
  * Fixed x-abstract extension (not generated)
  * Reset the defaults of the options added in 0.2.7 to avoid breaking changes.
    - useLombokAnnotations: false
    - openApiNullable: true
    - useSetForUniqueItems: false

## 0.2.7

* *Spring Generator*
  * added in-container validation, e.g. `List<@Size(max = 36) String>` (see [JSR-380 - Container element constraints](https://beanvalidation.org/2.0/spec/#constraintdeclarationvalidationprocess-containerelementconstraints)).
  * added vendor extensions: `x-abstract`, `x-implements`.
  * added `useLombokAnnotations` option (defaults to `true`)
  * added `openApiNullable` option (taken from 5.0,  breaking change, defaults to `false`, set to `true` if not ready).
  * added `useSetForUniqueItems` to map arrays with `uniqueItems` to `Set` instead of `List` (breaking change, defaults to `true`, set to `false` if not ready).
  * added `additionalDependencies` to be used in `spring-boot/pom.mustache` template.
  * formatted method parameters.

* *Maven Plugin*
  * added `addTestCompileSourceRoot` which adds the output directory to the project as a test source root.
  * added `apiNameSuffix` to customise the name of the API interface.
  * corrected `generatorName` property to point to `openapi.generator.maven.plugin.generatorName`.
  * fixed the code generated for properties of type `Map` in model.
  * refactored `GenerateMojo` so `mvn boat:generate -Dcodegen.configHelp -Dopenapi.generator.maven.plugin.generatorName=spring` works correctly.
  * test the generated code in the integration test phase

## 0.2.6
* Ensure RAML traits that are converted to OAS extensions are all using lower case. 

## 0.2.5 
* Fixed a bug how duplicate names are generated if RAML source has duplicate names for references. The parent resource name is now prepended to the schema name without removing the last character of the parent resource name
* Fixed a bug when in RAML resources were inline references instead of global type references for Request Bodies causing Response Schemas being referenced as Request Bodies

## 0.2.4 - Breaking Change!
* **Changed how operationIds are generated**. The previous implementation ended up generating very long and confusing names. 
    The improved generator greatly improves the names of operationId when converting from RAML to OAS3
* Default version of OpenAPI is now **3.0.3**
* Generated STUBS and Clients must be refactored to use the new names! It should not affect the names of Schemas converted from RAML. 


## 0.2.3
* Use RAML Display Name as Summary on Http Operations when converting to OAS3
* Also include integration-spec and artifacts ending on specs as default for conversion using `export-dep`
* Fix HTML2 Titles

## 0.2.2
* Fixed enum conversion. Empty enums are now set to null again when converting from raml to OpenAPI
* Added more robust code gen mojos


## 0.2.1
* Improved Open API Diff
* Sonar Fixes 

## 0.2.0
* Created new Code Generation Mojos with opinionated settings for
** Java Client with Spring WebClient (Reactive)
** Java Server Stubs for WebFlux (Reactive)
** Java Client with Spring Rest Template (Non Reactive)
** Java Server Stubs for Spring Rest Controller (Non Reactive)
** Improved Java Client API's to better cope with reserved words
* Export Dependencies will now traverse through the artifact to find all raml specs
* Improved RAML 2 Open API conversion
* Upgraded OpenAPI Diff library to more current version
* Mojo's can now break the build by setting `continueOnError` to false

## 0.1.9
* Improved how services are named after base url conversion was introduced.

## 0.1.8
* Reversed normalization of schema names as that causes stack overflow errors. 
* Fixed Base URL Conversion from RAML to OpenAPI
* Specify schema type when adding additional properties in Maven plugin using `additionalPropertiesType` configuration option


## 0.1.7
* Added configurable flag to add HttpServletRequest parameters to codegen'd server stubs.
* Extract inline examples from the obtained OpenAPI spec and put them under '<output-dir>/examples/' as json files.
* Changed the normalization of Schema Names to ensure existing casing is not lost

## 0.1.6
* Added documentation on boat-maven-plugin
* Upgraded YAML Libraries to improve output of YAML files
* Use standardized swagger YAML output
* Added Bean Validator in Code Generator
* Changed Open API Loader to correctly resolve references from reading input location instead of string

## 0.1.5

* Upgraded openapi-generator to 4.3.0
* Fixed java doc in the Java templates to allow usage in Java 11 projects
* Rename variable name `accept` to `acceptMediaType` in Java templates to allow OpenAPI Specs with parameters called `accept`


## 0.1.4

* Fixed template for HTML2 generator
* Include conversion of api.raml files found in dependencies

## 0.1.3  

* Added Code Generator Mojo from on [openapi-generator.tech](https://openapi-generator.tech/) with custom templates for Java, JavaSpring and HTML2
* Renamed `export` to `export-dep` mojo for converting RAML specs to oas from dependencies
* Added `export` mojo for converting RAML specs from input file
* Added Normaliser transformer for transforming examples names to be used in Java code generation  as example names cannot have special characters.
* Improve Title and Descriptions of converted RAML specs
* Always wrap examples in example object
* Many code improvements to be not ashamed of Sonar Reports.  


# Build & Install

```shell script
mvn install
```

## Maven Plugin Usage

The following command will generate `index.html` file in the specified output folder that contains API endpoints description.  
 
```bash
mvn boat:generate@generate-docs
```

## Generate API interfaces

Configuration
```
<build>
  <plugins>
    <plugin>
      <groupId>com.backbase.oss</groupId>
      <artifactId>boat-maven-plugin</artifactId>
      <version>${boat-maven-plugin.version}</version>
      <executions>
        <execution>
          <id>generate-api-code</id>
          <goals>
            <goal>generate</goal>
          </goals>
          <phase>generate-sources</phase>
          <configuration>
            <inputSpec>${project.basedir}/src/main/resources/api.yaml</inputSpec>
            <output>${project.build.directory}/generated-sources/api</output>
            <generatorName>spring</generatorName>
            [...]
            <configOptions>
              <library>spring-boot</library>
              <apiPackage>com.example.my.service.api.interfaces</apiPackage>
              <modelPackage>com.example.my.service.models</modelPackage>
              <hideGenerationTimestamp>true</hideGenerationTimestamp>
              <dateLibrary>java8</dateLibrary>
              <interfaceOnly>true</interfaceOnly>
              <skipDefaultInterface>true</skipDefaultInterface>
              <useBeanValidation>true</useBeanValidation>
              <useTags>true</useTags>
              <java8>true</java8>
              <useOptional>false</useOptional>
              [...]
            </configOptions>
          </configuration>
        </execution>
      </executions>
    </plugin>
 </plugins>
</build>
```

A comprehensive list of the Configuration options can be found below.

| Option | Property | Description |
|--------|----------|-------------|
| `verbose` |  `openapi.generator.maven.plugin.verbose` | verbose mode (`false` by default)
| `inputSpec` |  `openapi.generator.maven.plugin.inputSpec` | OpenAPI Spec file path
| `language` |  `openapi.generator.maven.plugin.language` | target generation language (deprecated, replaced by `generatorName` as values here don't represent only 'language' any longer)
| `generatorName` |  `openapi.generator.maven.plugin.generatorName` | target generator name
| `output` |  `openapi.generator.maven.plugin.output` | target output path (default is `${project.build.directory}/generated-sources/openapi`. Can also be set globally through the `openapi.generator.maven.plugin.output` property)
| `gitHost` | `openapi.generator.maven.plugin.gitHost` | The git host, e.g. gitlab.com
| `gitUserId` |  `openapi.generator.maven.plugin.gitUserId` | sets git information of the project
| `gitRepoId` | `openapi.generator.maven.plugin.gitRepoId` | sets the repo ID (e.g. openapi-generator)
| `templateDirectory` |  `openapi.generator.maven.plugin.templateDirectory` | directory with mustache templates
| `templateResourcePath` |  `openapi.generator.maven.plugin.templateResourcePath` | directory with mustache templates via resource path. This option will overwrite any option defined in `templateDirectory`.
| `engine` | `openapi.generator.maven.plugin.engine` | The name of templating engine to use, "mustache" (default) or "handlebars" (beta)
| `auth` |  `openapi.generator.maven.plugin.auth` | adds authorization headers when fetching the OpenAPI definitions remotely. Pass in a URL-encoded string of `name:header` with a comma separating multiple values
| `configurationFile` |  `openapi.generator.maven.plugin.configurationFile` | Path to separate json configuration file. File content should be in a json format {"optionKey":"optionValue", "optionKey1":"optionValue1"...} Supported options can be different for each language. Run `config-help -g {generator name}` command for language specific config options
| `skipOverwrite` |  `openapi.generator.maven.plugin.skipOverwrite` | Specifies if the existing files should be overwritten during the generation. (`false` by default)
| `apiPackage` |  `openapi.generator.maven.plugin.apiPackage` | the package to use for generated api objects/classes
| `modelPackage` |  `openapi.generator.maven.plugin.modelPackage` | the package to use for generated model objects/classes
| `invokerPackage` |  `openapi.generator.maven.plugin.invokerPackage` | the package to use for the generated invoker objects
| `packageName` | `openapi.generator.maven.plugin.packageName` | the default package name to use for the generated objects
| `groupId` | `openapi.generator.maven.plugin.groupId`  | sets project information in generated pom.xml/build.gradle or other build script. Language-specific conversions occur in non-jvm generators
| `artifactId` |  `openapi.generator.maven.plugin.artifactId` | sets project information in generated pom.xml/build.gradle or other build script. Language-specific conversions occur in non-jvm generators
| `artifactVersion` |  `openapi.generator.maven.plugin.artifactVersion` | sets project information in generated pom.xml/build.gradle or other build script. Language-specific conversions occur in non-jvm generators
| `library` |  `openapi.generator.maven.plugin.library` | library template (sub-template)
| `modelNamePrefix` |  `openapi.generator.maven.plugin.modelNamePrefix` | Sets the prefix for model classes and enums
| `modelNameSuffix` |  `openapi.generator.maven.plugin.modelNameSuffix` | Sets the suffix for model classes and enums
| `ignoreFileOverride` |  `openapi.generator.maven.plugin.ignoreFileOverride` | specifies the full path to a `.openapi-generator-ignore` used for pattern based overrides of generated outputs
| `httpUserAgent` | `openapi.generator.maven.plugin.httpUserAgent` | Sets custom User-Agent header value
| `removeOperationIdPrefix` |  `openapi.generator.maven.plugin.removeOperationIdPrefix` | remove operationId prefix (e.g. user_getName => getName)
| `logToStderr` |  `openapi.generator.maven.plugin.logToStderr` | write all log messages (not just errors) to STDOUT
| `enablePostProcessFile` |  `openapi.generator.maven.plugin.` | enable file post-processing hook
| `skipValidateSpec` |  `openapi.generator.maven.plugin.skipValidateSpec` | Whether or not to skip validating the input spec prior to generation. By default, invalid specifications will result in an error.
| `strictSpec` |  `openapi.generator.maven.plugin.strictSpec` | Whether or not to treat an input document strictly against the spec. 'MUST' and 'SHALL' wording in OpenAPI spec is strictly adhered to. e.g. when false, no fixes will be applied to documents which pass validation but don't follow the spec.
| `generateAliasAsModel` |  `openapi.generator.maven.plugin.generateAliasAsModel` | generate alias (array, map) as model
| `configOptions` |  N/A | a **map** of language-specific parameters. To show a full list of generator-specified parameters (options), please use `configHelp` (explained below)
| `instantiationTypes` |  `openapi.generator.maven.plugin.instantiationTypes` | sets instantiation type mappings in the format of type=instantiatedType,type=instantiatedType. For example (in Java): `array=ArrayList,map=HashMap`. In other words array types will get instantiated as ArrayList in generated code. You can also have multiple occurrences of this option
| `importMappings` |  `openapi.generator.maven.plugin.importMappings` | specifies mappings between a given class and the import that should be used for that class in the format of type=import,type=import. You can also have multiple occurrences of this option
| `typeMappings` |  `openapi.generator.maven.plugin.typeMappings` | sets mappings between OpenAPI spec types and generated code types in the format of OpenAPIType=generatedType,OpenAPIType=generatedType. For example: `array=List,map=Map,string=String`. You can also have multiple occurrences of this option
| `languageSpecificPrimitives` |  `openapi.generator.maven.plugin.languageSpecificPrimitives` | specifies additional language specific primitive types in the format of type1,type2,type3,type3. For example: `String,boolean,Boolean,Double`. You can also have multiple occurrences of this option
| `additionalProperties` |  `openapi.generator.maven.plugin.additionalProperties` | sets additional properties that can be referenced by the mustache templates in the format of name=value,name=value. You can also have multiple occurrences of this option
| `serverVariableOverrides` | `openapi.generator.maven.plugin.serverVariableOverrides` | A map of server variable overrides for specs that support server URL templating
| `reservedWordsMappings` |  `openapi.generator.maven.plugin.reservedWordsMappings` | specifies how a reserved name should be escaped to. Otherwise, the default `_<name>` is used. For example `id=identifier`. You can also have multiple occurrences of this option
| `generateApis` |  `openapi.generator.maven.plugin.generateApis` | generate the apis (`true` by default). Specific apis may be defined as a CSV via `apisToGenerate`.
| `apisToGenerate` |  `openapi.generator.maven.plugin.apisToGenerate` | A comma separated list of apis to generate.  All apis is the default.
| `generateModels` |  `openapi.generator.maven.plugin.generateModels` | generate the models (`true` by default). Specific models may be defined as a CSV via `modelsToGenerate`.
| `modelsToGenerate` |  `openapi.generator.maven.plugin.modelsToGenerate` | A comma separated list of models to generate.  All models is the default.
| `generateSupportingFiles` |  `openapi.generator.maven.plugin.generateSupportingFiles` | generate the supporting files (`true` by default)
| `supportingFilesToGenerate` |  `openapi.generator.maven.plugin.supportingFilesToGenerate` | A comma separated list of supporting files to generate.  All files is the default.
| `generateModelTests` |  `openapi.generator.maven.plugin.generateModelTests` | generate the model tests (`true` by default. Only available if `generateModels` is `true`)
| `generateModelDocumentation` |  `openapi.generator.maven.plugin.generateModelDocumentation` | generate the model documentation (`true` by default. Only available if `generateModels` is `true`)
| `generateApiTests` |  `openapi.generator.maven.plugin.generateApiTests` | generate the api tests (`true` by default. Only available if `generateApis` is `true`)
| `generateApiDocumentation` |  `openapi.generator.maven.plugin.generateApiDocumentation` | generate the api documentation (`true` by default. Only available if `generateApis` is `true`)
| `withXml` |  `openapi.generator.maven.plugin.withXml` | enable XML annotations inside the generated models and API (only works with Java `language` and libraries that provide support for JSON and XML)
| `skip` |  `codegen.skip` | skip code generation (`false` by default. Can also be set globally through the `codegen.skip` property)
| `skipIfSpecIsUnchanged` |  `codegen.skipIfSpecIsUnchanged` | Skip the execution if the source file is older than the output folder (`false` by default. Can also be set globally through the `codegen.skipIfSpecIsUnchanged` property)
| `addCompileSourceRoot` |  `openapi.generator.maven.plugin.addCompileSourceRoot` | Add the output directory to the project as a source root, so that the generated java types are compiled and included in the project artifact (`true` by default). Mutually exclusive with `addTestCompileSourceRoot`.
| `addTestCompileSourceRoot` |  `openapi.generator.maven.plugin.addTestCompileSourceRoot` | Add the output directory to the project as a test source root, so that the generated java types are compiled only for the test classpath of the project (`false` by default). Mutually exclusive with `addCompileSourceRoot`.
| `environmentVariables` | N/A | A **map** of items conceptually similar to "environment variables" or "system properties". These are merged into a map of global settings available to all aspects of the generation flow. Use this map for any options documented elsewhere as `systemProperties`.
| `configHelp` |  `codegen.configHelp` | dumps the configuration help for the specified library (generates no sources)

For the `spring` generator, the additional configuration options are:

| Option | Description |
|--------|-------------|
| `sortParamsByRequiredFlag` | Sort method arguments to place required parameters before optional parameters. (Default: true) |
| `sortModelPropertiesByRequiredFlag` | Sort model properties to place required parameters before optional parameters. (Default: true) |
| `ensureUniqueParams` | Whether to ensure parameter names are unique in an operation (rename parameters that are not). (Default: true) |
| `allowUnicodeIdentifiers` | boolean, toggles whether unicode identifiers are allowed in names or not, default is false (Default: false) |
| `prependFormOrBodyParameters` | Add form or body parameters to the beginning of the parameter list. (Default: false) |
| `modelPackage` | package for generated models (Default: org.openapitools.model) |
| `apiPackage` | package for generated api classes (Default: org.openapitools.api) |
| `invokerPackage` | root package for generated code (Default: org.openapitools.api) |
| `groupId` | groupId in generated pom.xml (Default: org.openapitools) |
| `artifactId` | artifactId in generated pom.xml. This also becomes part of the generated library's filename (Default: openapi-spring) |
| `artifactVersion` | artifact version in generated pom.xml. This also becomes part of the generated library's filename (Default: 1.0.0) |
| `artifactUrl` | artifact URL in generated pom.xml (Default: https://github.com/openapitools/openapi-generator) |
| `artifactDescription` | artifact description in generated pom.xml (Default: OpenAPI Java) |
| `scmConnection` | SCM connection in generated pom.xml (Default: scm:git:git@github.com:openapitools/openapi-generator.git) |
| `scmDeveloperConnection` | SCM developer connection in generated pom.xml (Default: scm:git:git@github.com:openapitools/openapi-generator.git) |
| `scmUrl` | SCM URL in generated pom.xml (Default: https://github.com/openapitools/openapi-generator) |
| `developerName` | developer name in generated pom.xml (Default: OpenAPI-Generator Contributors) |
| `developerEmail` | developer email in generated pom.xml (Default: team@openapitools.org) |
| `developerOrganization` | developer organization in generated pom.xml (Default: OpenAPITools.org) |
| `developerOrganizationUrl` | developer organization URL in generated pom.xml (Default: http://openapitools.org) |
| `licenseName` | The name of the license (Default: Unlicense) |
| `licenseUrl` | The URL of the license (Default: http://unlicense.org) |
| `sourceFolder` | source folder for generated code (Default: src/main/java) |
| `serializableModel` | boolean - toggle "implements Serializable" for generated models (Default: false) |
| `bigDecimalAsString` | Treat BigDecimal values as Strings to avoid precision loss. (Default: false) |
| `fullJavaUtil` | whether to use fully qualified name for classes under java.util. This option only works for Java API client (Default: false) |
| `hideGenerationTimestamp` | Hides the generation timestamp when files are generated. (Default: false) |
| `withXml` | whether to include support for application/xml content type and include XML annotations in the model (works with libraries that provide support for JSON and XML) (Default: false) |
| `dateLibrary` | Option. Date library to use (Default: threetenbp)<br>joda - Joda (for legacy app only)<br>legacy - Legacy java.util.Date (if you really have a good reason not to use threetenbp<br>java8-localdatetime - Java 8 using LocalDateTime (for legacy app only)<br>java8 - Java 8 native JSR310 (preferred for jdk 1.8+) - note: this also sets "java8" to true<br>threetenbp - Backport of JSR310 (preferred for jdk < 1.8) |
| `java8` | Option. Use Java8 classes instead of third party equivalents (Default: false)<br>true - Use Java 8 classes such as Base64. Use java8 default interface when a responseWrapper is used<br>false - Various third party libraries as needed |
| `disableHtmlEscaping` | Disable HTML escaping of JSON strings when using gson (needed to avoid problems with byte[] fields) (Default: false) |
| `booleanGetterPrefix` | Set booleanGetterPrefix (Default: get) |
| `additionalModelTypeAnnotations` | Additional annotations for model type(class level annotations) |
| `parentGroupId` | parent groupId in generated pom N.B. parentGroupId, parentArtifactId and parentVersion must all be specified for any of them to take effect |
| `parentArtifactId` | parent artifactId in generated pom N.B. parentGroupId, parentArtifactId and parentVersion must all be specified for any of them to take effect |
| `parentVersion` | parent version in generated pom N.B. parentGroupId, parentArtifactId and parentVersion must all be specified for any of them to take effect |
| `snapshotVersion` | Uses a SNAPSHOT version. true - Use a SnapShot Versionfalse - Use a Release Version |
| `title` | server title name or client service name (Default: OpenAPI Spring) |
| `configPackage` | configuration package for generated code (Default: org.openapitools.configuration) |
| `basePackage` | base package (invokerPackage) for generated code (Default: org.openapitools) |
| `interfaceOnly` | Whether to generate only API interface stubs without the server files. (Default: false) |
| `delegatePattern` | Whether to generate the server files using the delegate pattern (Default: false) |
| `singleContentTypes` | Whether to select only one produces/consumes content-type by operation. (Default: false) |
| `skipDefaultInterface` | Whether to generate default implementations for java8 interfaces (Default: false) |
| `async` | use async Callable controllers (Default: false) |
| `reactive` | wrap responses in Mono/Flux Reactor types (spring-boot only) (Default: false) |
| `responseWrapper` | wrap the responses in given type (Future, Callable, CompletableFuture,ListenableFuture, DeferredResult, HystrixCommand, RxObservable, RxSingle or fully qualified type) |
| `virtualService` | Generates the virtual service. For more details refer - https://github.com/elan-venture/virtualan/wiki (Default: false) |
| `useTags` | use tags for creating interface and controller classnames (Default: false) |
| `useBeanValidation` | Use BeanValidation API annotations (Default: true) |
| `performBeanValidation` | Use Bean Validation Impl. to perform BeanValidation (Default: false) |
| `useClassLevelBeanValidation` | Adds @Validated annotation to API interfaces (Default: false) |
| `useLombokAnnotations` | Use Lombok annotations to generate properties accessors and `hashCode`/`equals` methods (Default: false) |
| `addServletRequest` | Adds ServletRequest objects to API method definitions (Default: false) |
| `addBindingResult` | Adds BindingResult to Api method definitions' request bodies if UseBeanValidation true, for this to be effective you must configure UseBeanValidation, this is not done automatically (Default: false)|
| `implicitHeaders` | Skip header parameters in the generated API methods using @ApiImplicitParams annotation. (Default: false) |
| `swaggerDocketConfig` | Generate Spring OpenAPI Docket configuration class. (Default: false) |
| `apiFirst` | Generate the API from the OAI spec at server compile time (API first approach) (Default: false) |
| `useOptional` | Use Optional container for optional parameters (Default: false) |
| `hateoas` | Use Spring HATEOAS library to allow adding HATEOAS links (Default: false) |
| `returnSuccessCode` | Generated server returns 2xx code (Default: false) |
| `unhandledException` | Declare operation methods to throw a generic exception and allow unhandled exceptions (useful for Spring `@ControllerAdvice` directives). (Default: false) |
| `library` | library template (sub-template) (Default: spring-boot)<br>spring-boot - Spring-boot Server application using the SpringFox integration.<br>spring-mvc - Spring-MVC Server application using the SpringFox integration.<br>spring-cloud - Spring-Cloud-Feign client with Spring-Boot auto-configured settings.|
 readmeEtag: '"0cd3191a9e367f7d07060950c04a4da24c14da6f"' readmeLastModified: Thu, 18 Dec 2025 15:09:26 GMT repositoryId: 241074287 description: Backbase Open Api Tools (boat) helps manage large OpenAPI projects created: '2020-02-17T10:00:41Z' updated: '2026-01-21T17:47:30Z' language: Mustache archived: false stars: 47 watchers: 11 forks: 30 owner: Backbase logo: https://avatars.githubusercontent.com/u/1914334?v=4 license: NOASSERTION repoEtag: '"ecde275b45b330be67a3605d97e55244f62ffaee76e621a43bca7861686a845f"' repoLastModified: Wed, 21 Jan 2026 17:47:30 GMT foundInMaster: true category: SDK id: fd5b893e6ddf5af8821536a61ecb1d4e - source: openapi3 tags repository: https://github.com/kstasik/schema-tools v3: true repositoryMetadata: base64Readme: >- # Schema Tools

[![build](https://github.com/kstasik/schema-tools/workflows/build/badge.svg)](https://github.com/kstasik/schema-tools/actions)
[![tests](https://github.com/kstasik/schema-tools/workflows/test/badge.svg)](https://github.com/kstasik/schema-tools/actions)
[![crates-cli](https://img.shields.io/crates/v/schematools-cli)](https://crates.io/crates/schematools-cli)
[![crates-lib](https://img.shields.io/crates/v/schematools)](https://crates.io/crates/schematools)

# Introduction

Just another approach to openapi/jsonschema code generator. It's a home project written in `Rust`, simple all-in-one console tool with features like:

- openapi/json schema validation
- schema preprocessing
    - dereference
    - merge-allof
    - patch (apply/generate json-patch)
    - name
    - merge-openapi
- [Tera](https://github.com/Keats/tera) (jinja2) **code generator** with custom templates support

It is designed to speed up development of microservices heavly using json objects (json schemas on api level as well events).

Main differences in approach between other solutions like `openapi-generator`:

- more robust template language like jinja2
- language specific logic moved to templates including type mapping, reserved words
- wrapping of mixed types without discriminator
- one tool for clients, servers, json-schema queue consumers and processing of openapi
- relatively small binary used on every build of project
- codegen executed per microservice approach (not as a separate, generic client library)
- json-schema registry support, TODO: shared models - create one model for shared structures in different clients/servers to avoid mapping same structures

# Examples

Install `CLI` tool:

```
cargo install schematools-cli@0.21.0
```

## json-schema struct codegen

```
schematools-cli chain -vvv -c 'registry add templates https://github.com/kstasik/schema-tools-templates --rev 411f5207001637c19380046a21991b9ddba2bfe9' \
   -c 'process dereference https://raw.githubusercontent.com/kstasik/schema-tools/refs/heads/master/crates/schematools/resources/test/json-schemas/01-simple.json --skip-root-internal-references --create-internal-references' \
   -c 'process merge-all-of - --leave-invalid-properties' \
   -c 'codegen json-schema - --base-name Person --template templates::rust/_common/ --template templates::rust/model/ --format "rustfmt --edition 2021" --target-dir schemas/ -o namespace=people -o skipValidate=~true'
```

## openapi client codegen

```
schematools-cli \
   chain -vvv -c 'registry add templates https://github.com/kstasik/schema-tools-templates --rev 411f5207001637c19380046a21991b9ddba2bfe9' \
   -c 'process dereference https://raw.githubusercontent.com/kstasik/schema-tools/refs/heads/master/crates/schematools/resources/test/openapi/01-simple.yaml --skip-root-internal-references --create-internal-references' \
   -c 'process merge-all-of - --leave-invalid-properties' \
   -c 'codegen openapi - --template templates::rust/_common/ --template templates::rust/client/ --format "rustfmt --edition 2021" --target-dir src/clients/ -o apm=tracing -o name=simple -o namespace=simple -o skipValidate=~true'
```

## debugging json processing stages

You can use the `output` command multiple times to inspect how the JSON/YAML file appears at different stages of the process.

```
   -c 'output --to-file formatted.json -o json"' \
```

## using local templates / overwriting original templates

Create a `local-templates` directory and link it as the last used directory. Maintain the original registry naming if you wish to overwrite the templates.

```
   --template templates::rust/_common/ --template templates::rust/client/ --template local-templates/
```

# General rules

- All commands support yaml and json files.
- Use help to get list of available arguments `schema-tools process --help`
- `-v`, `-vv`, `-vvv`, `-vvvv` verbosity levels

# Validate

To validate openapi specification:

```
schematools validate openapi openapi.yaml
```

To validate json schema definition:

```
schematools validate openapi schema.yaml
```

Both commands return non-zero exit code in case of failure. Error reporting is not very clear but it shows the place where json schema is not met. TODO: resolve this [issue](https://github.com/Stranger6667/jsonschema-rs/issues?q=is%3Aissue+is%3Aopen+error)

## Process

Common CLI arguments:

```
<file>                      Path to json/yaml file with openapi specification
-o, --output <output>       Returned format [default: json] [possible values: json,  yaml]
--to-file <to-file>         Path of output file, default output to stdout
```

### Naming

If your openapi specification follows `RESTFUL` openapi rules you can create missing json-schema titles or try to rename operationId of existing endpoint:

```
schematools process name schema.yaml
```

Additional options are:
```
--overwrite                  Should overwrite existing titles
--overwrite-ambiguous         Should overwrite ambiguous titles
--resource-method-version    Reverts order of operationId generator to resource+method+version
```

### Dereferencing

To replace all occurrences of `$ref` in openapi you may type:

```
schematools process dereference schema.yaml
```

You should hardly ever perform full dereference of openapi. It partially dereference schema you may use following options:

- **skip-references** - is useful if you are using registry of common schemas. In many cases you don't want to dereference such `$ref`s
- **skip-root-internal-references** - is useful to skip `/components.*` openapi schema references which are very often stored in root openapi file.
- **create-internal-references** - saves space and dereferences each pointer only once, all next occurrences are replaced by pointer to first pointer

```
 --create-internal-references             Creates internal references if refs where pointing to same place
--skip-root-internal-references           Leaves internal references intact in root schema file
--skip-references <skip-references>...    List of hostnames to skip dereference
```

### Merge all of

To merge `allOf`s into objects type:

```
schematools process merge-all-of openapi.yaml
```

It's useful to perform such thing before code generation taking into account that json schema is more representation of validation not data structure itself. In many languages unions are a complicated thing but if you are using allOfs to extract common parts of structs it's a functionality which may be very helpful for you.

### Patch

If openapi you received seems broken you may fix it and create [json-patch](http://jsonpatch.com/) file:

```
schematools process patch <file> create <original-file> 
```

Then you can apply such patch to original openapi file during processing:

```
schematools process patch <file> apply <patch-file> 
```

### Merge openapi and bump

If you microservice is split to more than one service (and is exposed under same ingress) you may find it useful to create one openapi definition:

```
schematools process merge-openapi <file> --with <with>
```

Some useful options which may be needed for versioning merged openapi:

```
--add-version <add-version>    Should add info.x-version- attribute to openapi specification
--retag <retag>                Should change tags of all endpoints of merged openapi
```

To bump merged openapi version you may use this command:

```
schematools process bump-openapi <file> --original <previous-version-file>
```

It should correctly change version of openapi according to all sub-openapi semversions.

## Codegen openapi

Code generation itself is performed by processing templates directory. Before it is done all data from openapi/json-schema files has to be extracted and processed. There are two ways of performing codegen:

- `schematools codegen json-schema json-schema.json [...]` is used to process json schema file. It needs additional attribute `--base-name <base-name>` if title of json schema is missing. One json-schema doesnt mean that result of such codegeneration run will be exactly one struct/object/class - in case of complex json schemas it will be many models.
- `schematools codegen openapi openapi.json [...]` is used to process openapi specification - endpoints and models extraction.

Simple usage:

```
schema-tools codegen openapi openapi.json --template templates/  --target-dir pkg/client/
```

- `openapi.json` - openapi specification file
- `--template templates/` - directory with jinja2 files
- `--target-dir pkg/client/` - where code should be generated

### Codegen options

- `--nested-arrays-as-models` - some languages allow to create `Vec<HashMap<Vec<HashMap>>>>` / `[][][]int` inline types, some may need to create wrapping types for such cases
- `--optional-and-nullable-as-models` - openapi allows to create two levels of "nullability", some languages doesnt distinguish between null and undefined. This option wrap all occurrences of nullable and optional fields in separate types
- `--wrappers` - option to wrap mixed types (oneOf) to custom objects with custom deserialization logic
- `-o <options>` - option to pass options (string or json) to all templates files ex. `-o 'name=ordersClient' -o 'usedEndpoints=["/orders", "/orders/{id}/items"]'`
- `--format` - executes language formatter after code generation ex. `--format "gofmt -w"`

### Codegen templates

Codegen templates directory is targeted using `--template templates/` option. All simple files from this directory are copied to `--target-dir` beside `.j2` templates. The most important of each `.j2` is it's header (first line of file).

Example of models.j2:

```
{# type=models,filename=models.go #}

/* options: {{ options | json_encode(pretty=true) }} */
/* models: {{ models | json_encode(pretty=true) }} */
```

Example of endpoints.j2:

```
{# type=endpoints,filename=endpoints.go #}

/* options: {{ options | json_encode(pretty=true) }} */
/* models: {{ endpoints | json_encode(pretty=true) }} */
```

*Header* decides how to treat template file, how to and when generate files. Header options:

- `type=?` - possible values: `endpoints`, `models`
- `filename=?` - target filepath to create. May be mixed with options, ex. `filename=clients/%options.name%/endpoints.go`
- `if=foo:bar` - condition when to use template file. Should be mixed with options ex. `if=%options.type%:server`

For more information how to write template files please refer to [Tera docs](https://tera.netlify.app/docs/). To get list of additional filters we created please visit [filters.rs](https://github.com/kstasik/schema-tools/blob/master/src/codegen/filters.rs).

### Codegen template inheritance

Codegen allows to defined multiple `--template` options.

```
schematools codegen openapi.json --template dir1/ --template2 dir2/ --target-dir output/
```

Files from all directories are loaded one by one and in case of conflicts they are overwritten. There is also option to point to registry which currently may be only a **git repository**:

```
schematools codegen openapi.json --template REGISTRY::dir1/ --template2 dir2/ --target-dir output/
```

### Codegen ready to use templates

TODO: push codegen templates to the separate repo and write there an example

## Chain

This is the whole point of this tool. It wraps all existing functionalities together and adds global application context to openapi processing.

```
schematools chain -vvvv \
   -c 'process merge-all-of --leave-invalid-properties https://domain.com/openapi/orders/api.yaml' \
   -c 'process name - --resource-method-version --overwrite' \
   -c 'validate openapi - --continue-on-error' \
   -c 'codegen openapi - \
        --template codegen/client/ \
        --format "gofmt -w" \
        --target-dir pkg/client/ \
        -o namespace=orders \
        -o clientName=OrdersClient'
```

All commands take same arguments as they were executed separately. The only difference is that the first execution has to take real schema file as `-f` argument. The next executions should take `-` to use previously generated schema file.

```
schematools chain -vvvv \
   -c 'process merge-all-of --leave-invalid-properties specifications/api.yaml' \
   -c 'process name - --resource-method-version --overwrite' \
   -c 'validate openapi - ' 
   -c 'codegen openapi - \
        --template codegen/server/ \
        --format "gofmt -w" \
        --target-dir internal/http/ \
        -o namespace=myservice'
```

There is an option to dump processed schema to a file during chaining using `output` command:

```
schematools chain -vvvv \
   -c 'process merge-all-of --leave-invalid-properties specifications/api.yaml' \
   -c 'process name - --resource-method-version --overwrite' \
   -c 'validate openapi - ' \
   -c 'output --to-file=test.json -o json' \
   -c 'codegen openapi - \
        --template codegen/server/ \
        --format "gofmt -w" \
        --target-dir internal/http/ \
        -o namespace=myservice'
```

### Registry

There is an option to treat a separate git repository as source of templates:

```
schematools chain -vvvv \
   -c 'registry add common git://github.com/kstasik/schema-tools --tag v0.0.1' \
   -c 'process merge-all-of --leave-invalid-properties clients/client1.yaml' \
   -c 'process name - --resource-method-version --overwrite' \
   -c 'validate openapi - --continue-on-error' \
   -c 'codegen openapi - --template common::resources/openapi/ --target-dir pkg/client1/ -o namespace=client1 -o clientName=Client1'
```

To target such registry you simply use: `--template REGISTRY_NAME::path/`

## Example of usage

This example shows openapi http server with two external openapi client dependencies:

```
schematools chain -vv \
  # 0. Register external repository with templates and fix it to tag
  -c 'registry add default https://codegen-templates/templates.git --tag v0.5.0' \
  # 1. Load local openapi specification from file and dereference
  -c 'process dereference spec/api.yaml --skip-root-internal-references --create-internal-references' \
  # 1. Convert allOf to structs
  -c 'process merge-all-of - --leave-invalid-properties' \
  # 1. Overwrite titles of schemas and operationIds of endpoints in openapi
  -c 'process name - --overwrite --resource-method-version' \
  # 1. Perform validation of our openapi specification - interrupt build on error
  -c 'validate openapi - ' \
  # 1. Create models and routers
  -c 'codegen openapi - --template default::rust-actix-server/ --format "rustfmt --edition 2018" --target-dir src/app/ -o name=ShippingApp'  \
  \
  # 2. Load remote openapi definition of external service
  -c 'process dereference https://schemas.com/openapi/orders/v0.1.0.json --skip-root-internal-references' \
  # 2. Convert allOf to structs
  -c 'process merge-all-of - --leave-invalid-properties' \
  # 2. Overwrite titles of schemas and operationIds of endpoints in openapi because it follow restful standards
  -c 'process name - --overwrite --resource-method-version' \
  # 2. Patch openapi specification because it has an error and we don't want to wait for a fix to be published by other project
  -c 'patch - apply specs/fixes/orders.yaml' \
  # 2. Validate openapi definition but continue on failure because it's an external client not owned by project
  -c 'validate openapi - --continue-on-error' \
  # 2. Create client
  -c "codegen openapi - --optional-and-nullable-as-models --template default::rust-reqwest-http/ --format 'rustfmt --edition 2018' \
    -o 'usedEndpoints=~[\"ordersListV3\",\"ordersCreateV3\"]' \
    --target-dir src/clients/ -o name=OrdersClient" \
  \
  # 3. Load remote openapi definition of external service
  -c 'process dereference https://schemas.com/openapi/users/v0.1.0.json --skip-root-internal-references' \
  # 3. Convert allOf to structs
  -c 'process merge-all-of - --leave-invalid-properties' \
  # 3. Overwrite titles of schemas and operationIds of endpoints in openapi because it follow restful standards
  -c 'process name - --overwrite --resource-method-version' \
  # 3. Validate openapi definition but continue on failure because it's an external client not owned by project
  -c 'validate openapi - --continue-on-error' \
  # 3. Create client
  -c "codegen openapi - --optional-and-nullable-as-models --template default::rust-reqwest-http/ --format 'rustfmt --edition 2018' \
    -o 'usedEndpoints=~[\"usersListV3\",\"usersCreateV3\"]' \
    --target-dir src/clients/ -o name=UsersClient"
```
 readmeEtag: '"298bcf0416c32261ce2fa9a8daa99d4a2d4ea35b"' readmeLastModified: Sun, 15 Jun 2025 17:56:48 GMT repositoryId: 319082686 description: >- Set of tools for codegen, preprocessing and validation of json-schema and openapi spec created: '2020-12-06T16:48:56Z' updated: '2025-11-18T09:09:31Z' language: Rust archived: false stars: 48 watchers: 7 forks: 15 owner: kstasik logo: https://avatars.githubusercontent.com/u/676617?v=4 license: MIT repoEtag: '"8835e1cdc6c0cf56ed72289282bb972c25041b0edf4314b943a79f49e626a69b"' repoLastModified: Tue, 18 Nov 2025 09:09:31 GMT category: Parsers foundInMaster: true id: 152cddaa4f413487d9d5bab9d4eb78cb - source: openapi3 tags repository: https://github.com/alidehbansiahkarbon/openapiclientwizard v3: true id: 8ebb675de44da3df54a6c6b891575bd3 repositoryMetadata: base64Readme: >- PGltZyB3aWR0aD0iMjAwIiBoZWlnaHQ9IjIwMCIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vQWxpRGVoYmFuc2lhaGthcmJvbi9PcGVuQVBJQ2xpZW50V2l6YXJkL2Fzc2V0cy81NjAxNjA4LzllYWIwOTM3LTkwZTQtNDZhZS1iY2UyLTI5YTI0YzAyYmQ5ZCIgYWx0PSIiLz4gCgojIE9wZW5BUElDbGllbnRXaXphcmQoYmV0YS0gbm8gcHJvZHVjdGlvbiByZWFkeSkKCldlbGNvbWUgdG8gKipPcGVuQVBJQ2xpZW50V2l6YXJkKiog4oCTIHRoZSB1bHRpbWF0ZSBwbHVnaW4gZm9yIFJBRCBTdHVkaW8gdGhhdCByZXZvbHV0aW9uaXplcyB0aGUgd2F5IERlbHBoaSBkZXZlbG9wZXJzIGltcGxlbWVudCBSRVNUIEFQSSBjbGllbnRzLgoKIyMgT3ZlcnZpZXcKCkFzIERlbHBoaSBkZXZlbG9wZXJzLCB3ZSBvZnRlbiBmYWNlIHRoZSBjaGFsbGVuZ2Ugb2YgaW1wbGVtZW50aW5nIFJFU1QgQVBJIGNsaWVudHMuIEFQSSBkb2N1bWVudGF0aW9uIHJhcmVseSBpbmNsdWRlcyBEZWxwaGkgY29kZSBzbmlwcGV0cywgZm9yY2luZyB1cyB0byByZWFkIHRoZSBkb2N1bWVudGF0aW9uIGFuZCBpbXBsZW1lbnQgZWFjaCBtZXRob2QgLSBHRVQsIFBPU1QsIFBVVCwgREVMRVRFIC0gbWFudWFsbHkuIEV2ZW4gd2l0aCBTd2FnZ2VyIG9yIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMsIGl0J3MgYSB0aW1lLWNvbnN1bWluZyBwcm9jZXNzLiBXaXRob3V0IHRoZXNlIHNwZWNpZmljYXRpb25zLCB3ZSByZXNvcnQgdG8gdG9vbHMgbGlrZSBEZWxwaGkncyBSRVNUIERlYnVnZ2VyIG9yIFBvc3RtYW4sIHRlc3RpbmcgZWFjaCBtZXRob2QgaW5kaXZpZHVhbGx5IGJlZm9yZSBjb2RpbmcgdGhlbSBpbnRvIG91ciBwcm9qZWN0cy4KCioqT3BlbkFQSUNsaWVudFdpemFyZCoqIGNoYW5nZXMgYWxsIHRoYXQuIFRoaXMgcG93ZXJmdWwgcGx1Z2luIHJlYWRzIFN3YWdnZXIsIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMsIG9yIFBvc3RtYW4gY29sbGVjdGlvbiBhbmQgZ2VuZXJhdGVzIGEgZnVsbHkgaW1wbGVtZW50ZWQgY2xpZW50LXNpZGUgc2FtcGxlKFZDTCkgcHJvamVjdCBpbiBEZWxwaGksIHJlYWR5IHRvIGNvbXBpbGUsIGluIGp1c3QgMi0zIHNlY29uZHMhCgojIyBJREUgVmVyc2lvbiBzdXBwb3J0Ci0gKipTdXBwb3J0cyBmcm9tIERlbHBoaSAxMC4xIEJlcmxpbiB0byAxMi54IEF0aGVucyoqCgojIyBGZWF0dXJlcwoKLSAqKlN1cHBvcnRzIFN3YWdnZXIgU3BlY2lmaWNhdGlvbnM6KiogSGFuZGxlcyBKU09OIGZvcm1hdCBlZmZvcnRsZXNzbHkuCi0gKipTdXBwb3J0cyBPcGVuQVBJIFNwZWNpZmljYXRpb25zOioqIEhhbmRsZXMgYm90aCBKU09OIGFuZCBZQU1MIGZvcm1hdHMgZWZmb3J0bGVzc2x5LgotICoqUG9zdG1hbiBDb2xsZWN0aW9uIENvbnZlcnNpb246KiogQ29udmVydHMgUG9zdG1hbiBjb2xsZWN0aW9ucyBpbnRvIGNvbXBsZXRlIERlbHBoaSBjbGllbnQgcHJvamVjdHMuCi0gKipSYXBpZCBQcm9qZWN0IEdlbmVyYXRpb246KiogR2VuZXJhdGVzIGZ1bGx5IGltcGxlbWVudGVkIGNsaWVudC1zaWRlIHByb2plY3RzIGluIHNlY29uZHMsIHNhdmluZyB5b3UgYSBodWdlIHRpbWUod2Vla3Mgb2YgZGV2ZWxvcG1lbnQpLgoKCiMjIEJlbmVmaXRzCgotICoqU2F2ZSBUaW1lOioqIFJlZHVjZXMgd2Vla3Mgb2Ygd29yayB0byBtZXJlIHNlY29uZHMuCi0gKipJbmNyZWFzZSBQcm9kdWN0aXZpdHk6KiogRm9jdXMgb24gYnVpbGRpbmcgYW1hemluZyBhcHBsaWNhdGlvbnMgaW5zdGVhZCBvZiBzcGVuZGluZyB0aW1lIG9uIHJlcGV0aXRpdmUgY29kaW5nIHRhc2tzLgotICoqU2ltcGxpZnkgRGV2ZWxvcG1lbnQ6KiogQXV0b21hdGUgdGhlIGNyZWF0aW9uIG9mIERlbHBoaSBjbGllbnQtc2lkZSBwcm9qZWN0cyBmcm9tIEFQSSBzcGVjaWZpY2F0aW9ucy4KCiMjIEdldHRpbmcgU3RhcnRlZAoKMS4gKipEb3dubG9hZCBhbmQgSW5zdGFsbDoqKiBGb2xsb3cgdGhlIGluc3RhbGxhdGlvbiBpbnN0cnVjdGlvbnMgaW4gdGhlIHJlcG9zaXRvcnkuCjIuICoqUnVuIE9wZW5BUElDbGllbnRXaXphcmQ6KiogT3BlbiB5b3VyIFJBRCBTdHVkaW8gSURFIGFuZCBydW4gdGhlIHBsdWdpbi4KMy4gKipDcmVhdGUgYSBuZXcgcHJvamVjdCoqIGluIFJBRCBTdHVkaW8gYnkgZm9sbG93aW5nIHRoaXMgcGF0aDogRmlsZS9OZXcvT3RoZXI6CgoKIVtpbWFnZV0oaHR0cHM6Ly9naXRodWIuY29tL0FsaURlaGJhbnNpYWhrYXJib24vT3BlbkFQSUNsaWVudFdpemFyZC9hc3NldHMvNTYwMTYwOC9mMGRmZWFlNy01ZDFhLTQ5YjYtYjk3MC00N2UyNWQzYjM5NDQpCgo0LiAqKlNldHVwIHRoZSBwcm9qZWN0J3MgcHJlZmVyZW5jZXMqKgoKCiFbaW1hZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS91c2VyLWF0dGFjaG1lbnRzL2Fzc2V0cy84YjhkMWMzYi1hNTRiLTQzNzgtYWFhZC04MDI4NTJlYjc0YjApCgogICAKNi4gKipHZW5lcmF0ZSBZb3VyIFByb2plY3Q6KiogUHJvdmlkZSB5b3VyIFN3YWdnZXIgb3IgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLCBvciBQb3N0bWFuIGNvbGxlY3Rpb24sIGFuZCBsZXQgdGhlIHBsdWdpbiBkbyB0aGUgcmVzdC4KCiFbaW1hZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9BbGlEZWhiYW5zaWFoa2FyYm9uL09wZW5BUElDbGllbnRXaXphcmQvYXNzZXRzLzU2MDE2MDgvNDliYTFlMGQtOGI0YS00ZjA4LTg5YTgtODVkYjBiY2ZmOWJmKQoKCgojIyBSZW1hcmtzCioqVG8gZG93bmxvYWQsIHlvdSBtdXN0IHByb3ZpZGUgdGhlIGRpcmVjdCBVUkwgb2YgdGhlIGZpbGU7IHJlZGlyZWN0aW9uIGlzIG5vdCBzdXBwb3J0ZWQuKioKKipZb3UgY2Fubm90IHVzZSB0aGUgZmlyc3QgVVJMIHRvIGRvd25sb2FkIHRoZSBKU09OIHNwZWNpZmljYXRpb24sIGJ1dCB0aGUgc2Vjb25kIG9uZSBpcyBhY2NlcHRhYmxlLioqCgoxLSBodHRwczovL2dpdGh1Yi5jb20vYWRld2cvSUNBUi9ibG9iL0FERS0xL3VybC1zY2hlbWVzL3JlcHJvZHVjdGlvblVSTFNjaGVtZS5qc29uIOKdjAoKMi0gaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2FkZXdnL0lDQVIvQURFLTEvdXJsLXNjaGVtZXMvcmVwcm9kdWN0aW9uVVJMU2NoZW1lLmpzb24g4pyFCgoKIyMgRGVtbyBWaWRlbwoqKldhdGNoIGEgc2hvcnQgdmlkZW8gW2hlcmVdKGh0dHBzOi8veW91dHUuYmUvN0I3blNISXNWNjQpOioqIHRvIGxlYXJuIGhvdyB0byB1c2UgdGhlIHBsdWctaW4gKG9yIGNsaWNrIG9uIHRoZSBiZWxvdyBpbWFnZfCfkYfwn5GHKQoKPGEgaHJlZj0iaHR0cHM6Ly95b3V0dS5iZS83QjduU0hJc1Y2NCIgdGFyZ2V0PSJfYmxhbmsiPjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vQWxpRGVoYmFuc2lhaGthcmJvbi9PcGVuQVBJQ2xpZW50V2l6YXJkL2Fzc2V0cy81NjAxNjA4LzkwMTVjYTQzLTNkM2EtNGRmYS04NDM2LTFiYmJhN2FjNmZkZCIgd2lkdGggPSAiMzAwIiBoZWlnaHQgPSAiMzAwIiAvPjwvYT4KCiMjIFN1cHBvcnQKCklmIHlvdSBmaW5kIE9wZW5BUElDbGllbnRXaXphcmQgdXNlZnVsLCBwbGVhc2UgZ2l2ZSBpdCBhICoqc3RhcioqLgoKWW91ciBzdXBwb3J0IGlzIGdyZWF0bHkgYXBwcmVjaWF0ZWQhCgpZb3UgY2FuIGFsc28gc3VwcG9ydCBteSB3b3JrIGJ5IGJ1eWluZyBtZSBhIGNvZmZlZS4gVGhpcyBoZWxwcyBtZSBjb250aW51ZSBkZXZlbG9waW5nIHRoaXMgcGx1Z2luIGFuZCBvdGhlciB0b29scyBsaWtlIFtDaGF0R1BUV2l6YXJkXShodHRwczovL2dpdGh1Yi5jb20vQWxpRGVoYmFuc2lhaGthcmJvbi9DaGF0R1BUV2l6YXJkKSBhbmQgW0Vhc3lEQk1pZ3JhdG9yXShodHRwczovL2dpdGh1Yi5jb20vQWxpRGVoYmFuc2lhaGthcmJvbi9FYXN5REJNaWdyYXRvcikuCgo8YSBocmVmPSJodHRwczovL3d3dy5idXltZWFjb2ZmZWUuY29tL2FkZWhiYW5yIiB0YXJnZXQ9Il9ibGFuayI+PGltZyBzcmM9Imh0dHBzOi8vY2RuLmJ1eW1lYWNvZmZlZS5jb20vYnV0dG9ucy9kZWZhdWx0LW9yYW5nZS5wbmciIGFsdD0iQnV5IE1lIEEgQ29mZmVlIiBoZWlnaHQ9IjQxIiB3aWR0aD0iMTc0Ij48L2E+CgojIyMgQXNrIGZvciBkaXJlY3Qgc3VwcG9ydCBvciBzcGVjaWZpYyBpbXBsZW1lbnRhdGlvbgoKT3BlbkFQSUNsaWVudFdpemFyZCBpcyBwcm92aWRlZCBhcyBpcy4gSWYgeW91IG5lZWQgYW55IHNwZWNpZmljIGltcGxlbWVudGF0aW9uIGNvbnRhY3QgbWUgdmlhIFtlbWFpbF0oYWRlaGJhbkBnbWFpbC5jb20pLCBwbGVhc2UuCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWUhIApJZiB5b3UgaGF2ZSBzdWdnZXN0aW9ucyBmb3IgaW1wcm92ZW1lbnRzIG9yIGZpbmQgYW55IGlzc3VlcywgcGxlYXNlIG9wZW4gYW4gaXNzdWUgb3Igc3VibWl0IGEgcHVsbCByZXF1ZXN0LgoKIyMgTGljZW5zZQpUaGlzIHByb2plY3QgaXMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiAKU2VlIHRoZSBMSUNFTlNFIGZpbGUgZm9yIGRldGFpbHMuCgojIyBEZXBlbmRlbmNpZXMKVGhpcyBwbHVnaW4gdXRpbGl6ZXMgdGhlIE5lc2xpYiBsaWJyYXJ5IHRvIHBhcnNlIFlhbWwgZm9ybWF0LCB3aGljaCBpcyBpbmNsdWRlZCBpbiB0aGUgcmVwb3NpdG9yeSBmb3IgY29udmVuaWVuY2UuIAoKTm8gYWN0aW9uIGlzIHJlcXVpcmVkIG9uIHlvdXIgcGFydC4gCgpIb3dldmVyLCB5b3UgY2FuIGZpbmQgdGhlIGxhdGVzdCBzb3VyY2UgW2hlcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9uZXNsaWIvTmVzbGliLkpzb24pLgoKLS0tCgpUcmFuc2Zvcm0gdGhlIHdheSB5b3UgZGV2ZWxvcCBSRVNUIEFQSSBjbGllbnRzIGluIERlbHBoaSB3aXRoICoqT3BlbkFQSUNsaWVudFdpemFyZCoqLiAKCkRvd25sb2FkIGl0IHRvZGF5IGFuZCBzdHJlYW1saW5lIHlvdXIgd29ya2Zsb3chCgo8aHI+CjxwIGFsaWduPSJjZW50ZXIiPgo8aW1nIHNyYz0iaHR0cHM6Ly9pMC53cC5jb20vYmxvZ3MuZW1iYXJjYWRlcm8uY29tL3dwLWNvbnRlbnQvdXBsb2Fkcy8yMDIyLzExL2Rsb2dvbmV3LTU1ODI3NDAucG5nP3Jlc2l6ZT0yNTQlMkMyNDImc3NsPTEiIGFsdD0iRGVscGhpIj4KPC9wPgo8aDUgYWxpZ249ImNlbnRlciI+Ck1hZGUgd2l0aCA6aGVhcnQ6IG9uIERlbHBoaQo8L2g1Pgo= readmeEtag: '"f25aed280079a9f45bd4b6decd9534447425add8"' readmeLastModified: Tue, 03 Sep 2024 20:56:51 GMT repositoryId: 812935191 description: >- OpenAPIClientWizard – the ultimate plugin for RAD Studio that revolutionizes the way Delphi developers implement REST API clients. created: '2024-06-10T07:29:10Z' updated: '2026-01-13T13:04:12Z' language: Pascal archived: false stars: 51 watchers: 8 forks: 6 owner: AliDehbansiahkarbon logo: https://avatars.githubusercontent.com/u/5601608?v=4 license: MIT repoEtag: '"b0c869463b73cde3922e2dcd761f40af733cd7c205e264003c30808ad6c9d739"' repoLastModified: Tue, 13 Jan 2026 13:04:12 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/networknt/light-codegen v3: true repositoryMetadata: base64Readme: >- QSBjb2RlIGdlbmVyYXRvciBiYXNlZCBvbiBbcm9ja2VyXShodHRwczovL2dpdGh1Yi5jb20vZml6emVkL3JvY2tlcikgdGhhdCBjYW4gYmUgdXNlZCBhcyBhIGNvbW1hbmQgbGluZSB1dGlsaXR5IG9yIHdlYiBzZXJ2aWNlLgoKW1N0YWNrIE92ZXJmbG93XShodHRwczovL3N0YWNrb3ZlcmZsb3cuY29tL3F1ZXN0aW9ucy90YWdnZWQvbGlnaHQtNGopIHwKW0dvb2dsZSBHcm91cF0oaHR0cHM6Ly9ncm91cHMuZ29vZ2xlLmNvbS9mb3J1bS8jIWZvcnVtL2xpZ2h0LTRqKSB8CltHaXR0ZXIgQ2hhdF0oaHR0cHM6Ly9naXR0ZXIuaW0vbmV0d29ya250L2xpZ2h0LWNvZGVnZW4pIHwKW1N1YnJlZGRpdF0oaHR0cHM6Ly93d3cucmVkZGl0LmNvbS9yL2xpZ2h0YXBpLykgfApbWW91dHViZSBDaGFubmVsXShodHRwczovL3d3dy55b3V0dWJlLmNvbS9jaGFubmVsL1VDSENSTVdKVlh3OGlCN3pLeEY1NUJ5dykgfApbRG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9kb2MubmV0d29ya250LmNvbS90b29sL2xpZ2h0LWNvZGVnZW4vKSB8CltDb250cmlidXRpb24gR3VpZGVdKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vY29udHJpYnV0ZS8pIHwKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL25ldHdvcmtudC9saWdodC1jb2RlZ2VuLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL25ldHdvcmtudC9saWdodC1jb2RlZ2VuKQoKVGhlIExpZ2h0IHBsYXRmb3JtIGNvbnRhaW5zIHNldmVyYWwgZnJhbWV3b3JrcyB0byBidWlsZCBkaWZmZXJlbnQgc3R5bGVzIG9mIEFQSXMgb3IgbWljcm9zZXJ2aWNlcyBiYXNlZCBvbiB0aGUgY29udHJhY3Qgc3BlY2lmaWNhdGlvbnMuIEl0IGVuY291cmFnZXMgdXNlcnMgdG8gY3JlYXRlIHNlcnZpY2UgY29udHJhY3Qgc3BlY2lmaWNhdGlvbiBiZWZvcmUgY29kaW5nLCBhbmQgaXQgd291bGQgYmUgbmljZSBpZiB3ZSBjYW4gc2NhZmZvbGQgYSBwcm9qZWN0IGJhc2VkIG9uIHRoZSBkZXNpZ25lZCBzcGVjaWZpY2F0aW9uLiBGdXJ0aGVyLCBpdCB3b3VsZCBiZSBldmVuIHN3ZWV0ZXIgaWYgd2UgY291bGQgdXRpbGl6ZSB0aGUgc3BlY2lmaWNhdGlvbiBkdXJpbmcgcnVudGltZSB0byB2ZXJpZnkgdGhlIHNlY3VyaXR5IGFuZCB2YWxpZGF0ZSBpbnB1dCByZXF1ZXN0L291dCByZXNwb25zZSBiYXNlZCBvbiB0aGUgc2FtZSBzcGVjaWZpY2F0aW9uLgoKQmVmb3JlIHdlIGJ1aWx0IG91ciBnZW5lcmF0b3IsIHdlIGxvb2tlZCBhdCB0aGUgb3Blbi1zb3VyY2UgY29tbXVuaXR5IGFuZCBjb3VsZG4ndCBmaW5kIGFueXRoaW5nIHN1aXRhYmxlIGZvciBvdXIgbmVlZHMuIFNvIHdlIGNyZWF0ZWQgb25lIG92ZXIgYSB3ZWVrZW5kLCBhbmQgaXQgdHVybmVkIG91dCB2ZXJ5IHN1Y2Nlc3NmdWwuIElmIHlvdSB3YW50IHRvIGtub3cgbW9yZSBhYm91dCBpdCwgcGxlYXNlIGNoZWNrIG91dCB0aGUgW2xpZ2h0LWNvZGVnZW4gZG9jdW1lbnRdKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vdG9vbC9saWdodC1jb2RlZ2VuLykuCgpUaGUgZ2VuZXJhdG9yIGlzIHZlcnkgZWFzeSB0byB1c2UsIGFuZCB0aGVyZSBhcmUgbnVtZXJpYyB3YXlzIHRvIHVzZSBpdC4gQ29tbWFuZC1saW5lLCBEb2NrZXIsIFNjcmlwdGluZywgYW5kIFtXZWIgVUldKGh0dHBzOi8vY29kZWdlbi5saWdodGFwaS5uZXQpLiBUbyBnZXQgc3RhcnRlZCwgcGxlYXNlIGZvbGxvdyB0aGUgW2dldHRpbmctc3RhcnRlZF0oaHR0cHM6Ly9kb2MubmV0d29ya250LmNvbS9nZXR0aW5nLXN0YXJ0ZWQvbGlnaHQtY29kZWdlbi8pLgoKSWYgeW91IHdhbnQgdG8ga25vdyBhbGwgdGhlIG9wdGlvbnMgd2l0aCB0aGUgbGlnaHQtY29kZWdlbiwgeW91IGNhbiBmb2xsb3cgdGhlIHR1dG9yaWFscyBmb3IgZWFjaCBnZW5lcmF0b3IgYXQgW2xpZ2h0LWNvZGVnZW4gdHV0b3JpYWxdKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vdHV0b3JpYWwvZ2VuZXJhdG9yLykuCgpBbHNvLCB0aGVyZSBhcmUgc28gbWFueSBbb3RoZXIgdHV0b3JpYWxzXShodHRwczovL2RvYy5uZXR3b3JrbnQuY29tL3R1dG9yaWFsLykgdGhhdCBhcmUgdXNpbmcgbGlnaHQtY29kZWdlbiB0byBzY2FmZm9sZCBwcm9qZWN0cywgYW5kIHRoZXNlIGNhbiBiZSBmb3VuZCBpbiB0aGUgdHV0b3JpYWwgc2VjdGlvbi4K readmeEtag: '"3e249f4adf8a02097ffe857091c3fd6de2561e9c"' readmeLastModified: Sat, 22 Jun 2024 15:21:05 GMT repositoryId: 89109684 description: >- A code generator based on rocker that can be used as an utility or web service created: '2017-04-23T01:25:59Z' updated: '2026-02-03T00:39:44Z' language: Java archived: false stars: 46 watchers: 20 forks: 24 owner: networknt logo: https://avatars.githubusercontent.com/u/8740739?v=4 license: Apache-2.0 repoEtag: '"f2aff630c0694ac423bf76b10371b4f6e4489be11da3cacc4a9f3de25ceaf7bc"' repoLastModified: Tue, 03 Feb 2026 00:39:44 GMT foundInMaster: true category: - Code Generators - Parsers id: d81400a71f884e180d82047ee75eafea - source: - openapi3 tags - openapi31 tags name: openapi-processor-spring category: Server language: Java link: https://docs.openapiprocessor.io/spring repository: https://github.com/openapi-processor/openapi-processor-spring source_description: >- Generates java interfaces & model classes for Spring Boot (annotation based, MVC & WebFlux) from an openapi.yaml. Provides type mapping capabilities to adjust the generated code. Gradle support. v2: false v3: true repositoryMetadata: base64Readme: >- WyFbXVtiYWRnZS1jaV1dW3dvcmtmbG93LWNpXQpbIVtdW3NvbmFyLXRlY2RlYnRdXVtzb25hcl0KWyFbXVtiYWRnZS1jZW50cmFsXV1bb2FwLWNlbnRyYWxdCgohW29wZW5hcGktcHJvY2Vzc29yLXNwcmluZyBsb2dvXShpbWFnZXMvb3BlbmFwaS1wcm9jZXNzb3Itc3ByaW5nQDEyODB4MjAwLnBuZykKCiMgb3BlbmFwaS1wcm9jZXNzb3Itc3ByaW5nCgphbiBbT3BlbkFQSV1bb3BlbmFwaV0gaW50ZXJmYWNlIG9ubHkgJiBtb2RlbCBqYXZhIGNvZGUgZ2VuZXJhdG9yIGZvciBbU3ByaW5nIEJvb3RdW3NwcmluZ2Jvb3RdLgogCgojIGRvY3VtZW50YXRpb24KClNlZSBbaGVyZV1bb2FwLWRvY3NdLgoKIyBzbmFwc2hvdCByZXBvc2l0b3J5Cgp0byB1c2Ugc25hcHNob3QgdmVyc2lvbnMgYWRkIGBodHRwczovL2NlbnRyYWwuc29uYXR5cGUuY29tL3JlcG9zaXRvcnkvbWF2ZW4tc25hcHNob3RzYCBhcyBtYXZlbiByZXBvc2l0b3J5IHRvIHlvdXIgYnVpbGQgZmlsZS4KCgpbb2FwLWNlbnRyYWxdOiBodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9aW8ub3BlbmFwaXByb2Nlc3NvcgpbYmFkZ2UtY2VudHJhbF06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2lvLm9wZW5hcGlwcm9jZXNzb3Ivb3BlbmFwaS1wcm9jZXNzb3Itc3ByaW5nP2xhYmVsPU1hdmVuJTIwQ2VudHJhbApbYmFkZ2UtbGljZW5zZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTGljZW5zZS1BcGFjaGUlMjAyLjAtYmx1ZS5zdmc/bGFiZWxDb2xvcj0zMTNBNDIKW2JhZGdlLWNpXTogaHR0cHM6Ly9naXRodWIuY29tL29wZW5hcGktcHJvY2Vzc29yL29wZW5hcGktcHJvY2Vzc29yLXNwcmluZy93b3JrZmxvd3MvYnVpbGQvYmFkZ2Uuc3ZnCltvYXAtbGljZW5zZV06IGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYXBpLXByb2Nlc3Nvci9vcGVuYXBpLXByb2Nlc3Nvci1zcHJpbmcvYmxvYi9tYXN0ZXIvTElDRU5TRQpbd29ya2Zsb3ctY2ldOiBodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1wcm9jZXNzb3Ivb3BlbmFwaS1wcm9jZXNzb3Itc3ByaW5nL2FjdGlvbnM/cXVlcnk9d29ya2Zsb3clM0FidWlsZApbc29uYXItY292ZXJhZ2VdOiBodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1vcGVuYXBpLXByb2Nlc3Nvcl9vcGVuYXBpLXByb2Nlc3Nvci1zcHJpbmcmbWV0cmljPWNvdmVyYWdlCltzb25hci10ZWNkZWJ0XTogaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9b3BlbmFwaS1wcm9jZXNzb3Jfb3BlbmFwaS1wcm9jZXNzb3Itc3ByaW5nJm1ldHJpYz1zcWFsZV9pbmRleApbc29uYXJdOiBodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPW9wZW5hcGktcHJvY2Vzc29yX29wZW5hcGktcHJvY2Vzc29yLXNwcmluZwpbb2FwLWRvY3NdOiBodHRwczovL2RvY3Mub3BlbmFwaXByb2Nlc3Nvci5pbwpbb3BlbmFwaV06IGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8KW3NwcmluZ2Jvb3RdOiBodHRwczovL3NwcmluZy5pby9wcm9qZWN0cy9zcHJpbmctYm9vdAo= readmeEtag: '"4a95bb31365ca4b7430f2bf90bf3458a3d98a74c"' readmeLastModified: Thu, 03 Jul 2025 06:41:44 GMT repositoryId: 196863956 description: >- an OpenAPI 3.2, 3.1 & 3.0 interface & model java code generator for Spring Boot created: '2019-07-14T17:16:49Z' updated: '2026-01-31T17:39:33Z' language: Kotlin archived: false stars: 49 watchers: 2 forks: 11 owner: openapi-processor logo: https://avatars.githubusercontent.com/u/66728774?v=4 license: Apache-2.0 repoEtag: '"7e68f50d41eeddf04804ea934b63300d6c8247ce2cec58da415abf3d02c357d4"' repoLastModified: Sat, 31 Jan 2026 17:39:33 GMT foundInMaster: true id: 544de493e48dbce482dd3fcfc87ad568 v3_1: true - source: openapi3 tags repository: https://github.com/oas-tools/oas-generator v3: true repositoryMetadata: base64Readme: >- PiAqKldhcm5pbmcqKgo+Cj4gVGhpcyBwYWNrYWdlIGlzIG5vdyBpbmNsdWRlZCBpbiBbT0FTIFRvb2xzIENMSV0oaHR0cHM6Ly9naXRodWIuY29tL29hcy10b29scy9vYXMtY2xpKS4gQ2hlY2sgdGhlIFtkb2NzXShodHRwczovL29hcy10b29scy5naXRodWIuaW8vZG9jcy9jbGkpIGZvciBtb3JlIGluZm8uCgojIG9hcy1nZW5lcmF0b3IKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2lzYS1ncm91cC9vYXMtZ2VuZXJhdG9yLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2lzYS1ncm91cC9vYXMtZ2VuZXJhdG9yKQpbIVtkZXBlbmRlbmNpZXMgU3RhdHVzXShodHRwczovL2RhdmlkLWRtLm9yZy9pc2EtZ3JvdXAvb2FzLWdlbmVyYXRvci9zdGF0dXMuc3ZnKV0oaHR0cHM6Ly9kYXZpZC1kbS5vcmcvaXNhLWdyb3VwL29hcy1nZW5lcmF0b3IpClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9pc2EtZ3JvdXAvb2FzLWdlbmVyYXRvci9icmFuY2gvbWFzdGVyL2dyYXBoL2JhZGdlLnN2ZyldKGh0dHBzOi8vY29kZWNvdi5pby9naC9pc2EtZ3JvdXAvb2FzLWdlbmVyYXRvcikKWyFbS25vd24gVnVsbmVyYWJpbGl0aWVzXShodHRwczovL3NueWsuaW8vdGVzdC9ucG0vb2FzLWdlbmVyYXRvci9iYWRnZS5zdmcpXShodHRwczovL3NueWsuaW8vdGVzdC9ucG0vb2FzLWdlbmVyYXRvcikKWyFbR3JlZW5rZWVwZXIgYmFkZ2VdKGh0dHBzOi8vYmFkZ2VzLmdyZWVua2VlcGVyLmlvL2lzYS1ncm91cC9vYXMtZ2VuZXJhdG9yLnN2ZyldKGh0dHBzOi8vZ3JlZW5rZWVwZXIuaW8vKQoKWyFbTlBNXShodHRwczovL25vZGVpLmNvL25wbS9vYXMtZ2VuZXJhdG9yLnBuZz9kb3dubG9hZHM9dHJ1ZSZkb3dubG9hZFJhbms9dHJ1ZSZzdGFycz10cnVlKV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vb2FzLWdlbmVyYXRvci8pCgpUaGlzIG1vZHVsZSBwcm92aWRlcyBOb2RlSlMgUkVTVGZ1bGwgQVBJcyBzY2FmZm9sZGluZyBiYXNlZCBPcGVuQVBJIDMueCBzcGVjcyB1c2luZyBbb2FzLXRvb2xzXShodHRwczovL2dpdGh1Yi5jb20vaXNhLWdyb3VwL29hcy10b29scykgYW5kIFtleHByZXNzXShodHRwczovL2dpdGh1Yi5jb20vZXhwcmVzc2pzL2V4cHJlc3MpLiBJdCBpcyBjb21wYXRpYmxlIHdpdGggc2VydmVycyBnZW5lcmF0ZWQgd2l0aCB0aGUgc3dhZ2dlci10b29sIHN1aXRlIHRvIGhlbHAgaW4gdGhlIHRyYW5zaXRpb24gZnJvbSBwcmUtZXhpc3Rpbmcgc2VydmVycyBiYXNlZCBvbiAyLnggc3BlY3MuCgpXZSBoYXZlIGEgMiBtaW4uIHR1dG9yaWFsOgoKPGEgaHJlZj0iaHR0cHM6Ly95b3V0dS5iZS9PWmh0b1hQMkdpNCIgYWx0PSJvYXMtZ2VuZXJhdG9yIGludHJvZHVjdGlvbiAodjIuMC42KSAtIENsaWNrIHRvIFdhdGNoISI+PGltZyBzcmM9Imh0dHBzOi8vaS5pbWd1ci5jb20vZmNxYVdDcS5qcGciIGFsaWduPSJjZW50ZXIiIHdpZHRoPSIzMDAiIGFsdD0ib2FzLWdlbmVyYXRvciBpbnRyb2R1Y3Rpb24gKHYyLjAuNikgLSBDbGljayB0byBXYXRjaCEiPjwvYT4KCgojIyAxXC4gSW5zdGFsbCBvYXMtZ2VuZXJhdG9yCgpgYGBiYXNoCm5wbSBpbnN0YWxsIG9hcy1nZW5lcmF0b3IgLWcKYGBgCgojIyAyXC4gVXNlIG9hcy1nZW5lcmF0b3IgY29tbWFuZAoKIyMjIDIuMVwuIEhlbHAKCmBgYGJhc2gKJCBvYXMtZ2VuZXJhdG9yIC0taGVscAoKVXNhZ2U6ICBvYXMtZ2VuZXJhdG9yIDxPQVMgdjMgZmlsZSBpbiBZQU1MIG9yIEpTT04+CgogIE9wdGlvbnM6CgogICAgLW4sIC0tcHJvamVjdE5hbWUgPHByb2plY3ROYW1lPiAgTmFtZSBmb3IgdGhlIGdlbmVyYXRlZCBmb2xkZXIKICAgIC16LCAtLWdlbmVyYXRlWmlwICAgICAgICAgICAgICAgIEluZGljYXRlIHdoZXRoZXIgdGhlIGdlbmVyYXRlZCBmb2xkZXIgbXVzdCBiZSBkZWxldGVkIGFmdGVyIGNvbXByZXNzaW9uCiAgICAtaCwgLS1oZWxwICAgICAgICAgICAgICAgICAgICAgICBvdXRwdXQgdXNhZ2UgaW5mb3JtYXRpb24KYGBgCgojIyMgMi4yIEV4YW1wbGVzCgpHZW5lcmF0ZSBhIE5vZGVKUyBwcm9qZWN0OiBUaGUgZm9sbG93aW5nIGNvbW1hbmQgd2lsIGNyZWF0ZSB0aGUgY29kZSBvZiBhIE5vZGVKUyBhcHBsaWNhdGlvbiBiYXNlZCBvbiB0aGUgT3BlbkFQSSBTcGVjIDMueCBgZG9ja2VyLWVuZ2luZS55YW1sYCBhbmQgcGxhY2UgaXRzIGNvZGUgaW4gYSBmb2xkZXIgYnkgdGhlIG5hbWUgYGdlbmVyYXRlZFNlcnZlcmAuCgpgYGBiYXNoCiQgb2FzLWdlbmVyYXRvciBzcGVjcy9kb2NrZXItZW5naW5lLnlhbWwgLW4gZ2VuZXJhdGVkU2VydmVyCmBgYAoKT25jZSB0aGUgZm9sZGVyIGlzIGNyZWF0ZWQgbG9jYXRlIGludG8gaXQuIEluc3RhbGwgZGVwZW5kZW5jaWVzIGFuZCBydW4gaXQgYnkgZG9pbmcgbnBtIHN0YXJ0OgoKYGBgYmFzaAokIGNkIGdlbmVyYXRlZFNlcnZlcgokIG5wbSBzdGFydApgYGAKCk5vdyB5b3VyIGFwcCBpcyBydW5uaW5nIG9uIHBvcnQgODA4MC4KWW91IGNhbiB0cnkgdGhlIHVybCBbaHR0cDovL2xvY2FsaG9zdDo4MDgwL3YxLjMzL3ZvbHVtZXNdKGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC92MS4zMy92b2x1bWVzKQoKSW4gdGhlIGZvbGRlciBgY29udHJvbGxlcnNgIHRoZXJlIHdpbGwgYmUgYSBzZXQgb2YganMgZmlsZXMgd2l0aCB0aGUgdGVtcGxhdGVzIGZvciBlYWNoIG1ldGhvZCBzcGVjaWZpZWQgaW4gdGhlIHNwZWMgdGhhdCBzaG91bGQgYmUgY29tcGxldGVkIHRvIGhhdmUgYSBmdWxseSB3b3JraW5nIEFQSS4KCkhhcHB5IGNvZGluZyEgCgoKIyMgTGljZW5zZQoKQ29weXJpZ2h0IDIwMTgsIFtJU0EgR3JvdXBdKGh0dHA6Ly93d3cuaXNhLnVzLmVzKSwgW1VuaXZlcnNpdHkgb2YgU2V2aWxsYV0oaHR0cDovL3d3dy51cy5lcykKCkZvciB0ZWNobmljYWwgaW5xdWlyeSBwbGVhc2UgY29udGFjdCB0byBbZW5naW5lZXJpbmcgdGVhbV0oLi9leHRyYS90ZWFtLm1kKS4KClshW0lTQSBHcm91cF0oaHR0cDovL3d3dy5pc2EudXMuZXMvMi4wL2Fzc2V0cy9pbWcvdGhlbWUvbG9nbzIucG5nKV0oaHR0cDovL3d3dy5pc2EudXMuZXMpCgpMaWNlbnNlZCB1bmRlciB0aGUgKipBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAqKiAodGhlICJbTGljZW5zZV0oLi9MSUNFTlNFKSIpOyB5b3UgbWF5IG5vdCB1c2UgdGhpcyBmaWxlIGV4Y2VwdCBpbiBjb21wbGlhbmNlIHdpdGggdGhlIExpY2Vuc2UuIFlvdSBtYXkgb2J0YWluIGEgY29weSBvZiB0aGUgTGljZW5zZSBhdCBhcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCgpVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlIGRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsIFdJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLiBTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kIGxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgo= readmeEtag: '"21e0d301092f58c02c880708a88bd38cd13d9555"' readmeLastModified: Mon, 15 Aug 2022 00:25:20 GMT repositoryId: 134678864 description: >- NodeJS RESTful APIs scaffolding based OpenAPI 3.x specs using oas-tools and express. created: '2018-05-24T07:35:23Z' updated: '2025-08-27T12:16:51Z' language: JavaScript archived: true stars: 42 watchers: 7 forks: 4 owner: oas-tools logo: https://avatars.githubusercontent.com/u/97441539?v=4 license: Apache-2.0 repoEtag: '"3d9649d36eaa89ff9c5fc5171d565e1bd6383dd261bd90c590234f7e3c3e8630"' repoLastModified: Wed, 27 Aug 2025 12:16:51 GMT foundInMaster: true category: - Server - Parsers id: 18da66b6dce72c18901effa0c36ae85a oldLocations: - https://github.com/isa-group/oas-generator - source: openapi3 tags repository: https://github.com/hpoul/openapi_dart v3: true repositoryMetadata: base64Readme: >- IyBEYXJ0IE9wZW5BUEkgQ29kZSBHZW5lcmF0b3IKCkRhcnQgaW1wbGVtZW50YXRpb24gdG8gZ2VuZXJhdGUgKGRhcnQpIHNlcnZlcgpzdHVicyBhbmQgY2xpZW50IGxpYnJhcmllcy4KClNlZSBbcGFja2FnZXMvb3BlbmFwaV9jb2RlX2J1aWxkZXIvUkVBRE1FLm1kXShwYWNrYWdlcy9vcGVuYXBpX2NvZGVfYnVpbGRlci9SRUFETUUubWQpIGZvciBkZXRhaWxzLgo= readmeEtag: '"0b2f1170acd3a87d380fb9559d68595bf4ec8fba"' readmeLastModified: Sun, 26 May 2024 10:39:04 GMT repositoryId: 270302676 description: >- Dart implementation to generate (dart) server stubs and client libraries from OpenAPI 3.0 yaml files. created: '2020-06-07T12:41:17Z' updated: '2026-01-14T07:13:42Z' language: Dart archived: false stars: 42 watchers: 2 forks: 25 owner: hpoul logo: https://avatars.githubusercontent.com/u/313066?v=4 repoEtag: '"9541f5a241ec4cb3bf3208e69d5d839f0c7cae88b9308494c7c498f46cbba237"' repoLastModified: Wed, 14 Jan 2026 07:13:42 GMT foundInMaster: true category: Code Generators id: ced3cf309aea10ea14f4d94579214646 - source: openapi3 tags name: go-openapi homepage: https://github.com/nasa9084/go-openapi language: Go source_description: Golang struct model for OpenAPI 3.x. category: - Low-level Tooling - Parsers repository: https://github.com/nasa9084/go-openapi v3: true repositoryMetadata: base64Readme: >- T3BlbkFQSSBTcGVjaWZpY2F0aW9uIG9iamVjdCBtb2RlbAo9PT0KClshW0dvRG9jXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL25hc2E5MDg0L2dvLW9wZW5hcGk/c3RhdHVzLnN2ZyldKGh0dHBzOi8vZ29kb2Mub3JnL2dpdGh1Yi5jb20vbmFzYTkwODQvZ28tb3BlbmFwaSkKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvbmFzYTkwODQvZ28tb3BlbmFwaS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9uYXNhOTA4NC9nby1vcGVuYXBpKQpbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvbmFzYTkwODQvZ28tb3BlbmFwaS9icmFuY2gvbWFzdGVyL2dyYXBoL2JhZGdlLnN2ZyldKGh0dHBzOi8vY29kZWNvdi5pby9naC9uYXNhOTA4NC9nby1vcGVuYXBpKQoKLS0tCgoqKlRoaXMgcGFja2FnZSBpcyBzdGlsbCB1bmRlciBkZXZlbG9wbWVudCwgc28gdGhlIEFQSSB3aWxsIGJlIGNoYW5nZWQgd2l0aG91dCBhbnkgbm90aWZpY2F0aW9uKioKCiMjIE92ZXJ2aWV3CgpUaGlzIGlzIGFuIGltcGxlbWVudGF0aW9uIG9mIFtPcGVuQVBJIFNwZWNpZmljYXRpb24gMy4wXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikgb2JqZWN0IG1vZGVsIHdpdGggc29tZSB1c2FibGUgZnVuY3Rpb25zLgoKIyMgU3lub3BzaXMKCmBgYCBnbwpwYWNrYWdlIG1haW4KCmltcG9ydCAoCiAgICAiZm10IgoKICAgICJnaXRodWIuY29tL25hc2E5MDg0L2dvLW9wZW5hcGkiCikKCmZ1bmMgbWFpbigpIHsKICAgIGRvYywgXyA6PSBvcGVuYXBpLkxvYWRGaWxlKCJwYXRoL3RvL3NwZWMiKQogICAgZm10LlByaW50KGRvYy5WZXJzaW9uKQp9CmBgYAoKIyMgU3RhdHVzCgoqIFt4XSBNb2RlbCBkZWZpbml0aW9uCiogW3hdIExvYWQgT3BlbkFQSSAzLjAgc3BlYyBmaWxlCiogWyBdIFJlc29sdmUgUmVmZXJlbmNlIG9iamVjdAogICogW3hdIFJlc29sdmUgIy9jb21wb25lbnQgcmVmZXJlbmNlCiAgKiBbIF0gUmVzb2x2ZSBvdGhlciBmaWxlIHJlZmVyZW5jZQoqIFsgXSBWYWxpZGF0aW9uCiAgKiBbeF0gVmFsaWRhdGUgc3BlYyB2YWx1ZXMKICAgICogWyBdIHRlc3QgZm9yIHZhbGlkYXRpb24KICAgICAgKiBbeF0gRG9jdW1lbnQKICAgICAgKiBbeF0gSW5mbwogICAgICAqIFt4XSBDb250YWN0CiAgICAgICogW3hdIExpY2Vuc2UKICAgICAgKiBbeF0gU2VydmVyCiAgICAgICogW3hdIFNlcnZlclZhcmlhYmxlCiAgICAgICogW3hdIFBhdGhzCiAgICAgICogW3hdIFBhdGhJdGVtCiAgICAgICogW3hdIE9wZXJhdGlvbgogICAgICAqIFt4XSBQYXJhbWV0ZXIKICAgICAgKiBbeF0gUmVxdWVzdEJvZHkKICAgICAgKiBbeF0gUmVzcG9uc2VzCiAgICAgICogW3hdIFJlc3BvbnNlCiAgICAgICogW3hdIENhbGxiYWNrcwogICAgICAqIFt4XSBDYWxsYmFjawogICAgICAqIFsgXSBTY2hlbWEKICAgICAgKiBbeF0gRXhhbXBsZQogICAgICAqIFsgXSBNZWRpYVR5cGUKICAgICAgKiBbIF0gSGVhZGVyCiAgICAgICogWyBdIExpbmsKICAgICAgKiBbIF0gRW5jb2RpbmcKICAgICAgKiBbeF0gRGlzY3JpbWluYXRvcgogICAgICAqIFt4XSBYTUwKICAgICAgKiBbeF0gQ29tcG9uZW50cwogICAgICAqIFt4XSBTZWN1cml0eVNjaGVtZQogICAgICAqIFt4XSBPQXV0aEZsb3dzCiAgICAgICogW3hdIE9BdXRoRmxvdwogICAgICAqIFsgXSBTZWN1cml0eVJlcXVpcmVtZW50CiAgICAgICogW3hdIFRhZwogICAgICAqIFt4XSBFeHRlcm5hbERvY3VtZW50YXRpb24KICAqIFsgXSBWYWxpZGF0ZSBIVFRQIFJlcXVlc3QKICAqIFsgXSBWYWxpZGF0ZSBIVFRQIFJlc3BvbnNlCg== readmeEtag: '"41aa7a6b31dade11ea34606947980743473497e8"' readmeLastModified: Wed, 19 Jun 2024 11:38:10 GMT repositoryId: 113154608 description: OpenAPI Specification (OAS) 3.0 implementation for Go created: '2017-12-05T08:30:55Z' updated: '2026-01-13T04:15:15Z' language: Go archived: false stars: 37 watchers: 1 forks: 3 owner: nasa9084 logo: https://avatars.githubusercontent.com/u/11725486?v=4 license: MIT repoEtag: '"6908356c7ee35f5998a048cfdc93b6d6f27509fc7edf9871ba32d1b0636a76c3"' repoLastModified: Tue, 13 Jan 2026 04:15:15 GMT foundInMaster: true id: 7b277653d8ebb25b494d8eda45d7b35e - source: openapi3 tags repository: https://github.com/quantmind/aio-openapi v3: true repositoryMetadata: base64Readme: >- IyBhaW8tb3BlbmFwaQoKWyFbUHlQSSB2ZXJzaW9uXShodHRwczovL2JhZGdlLmZ1cnkuaW8vcHkvYWlvLW9wZW5hcGkuc3ZnKV0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL3B5L2Fpby1vcGVuYXBpKQpbIVtQeXRob24gdmVyc2lvbnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9weXZlcnNpb25zL2Fpby1vcGVuYXBpLnN2ZyldKGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9haW8tb3BlbmFwaSkKWyFbQnVpbGRdKGh0dHBzOi8vZ2l0aHViLmNvbS9xdWFudG1pbmQvYWlvLW9wZW5hcGkvd29ya2Zsb3dzL2J1aWxkL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9xdWFudG1pbmQvYWlvLW9wZW5hcGkvYWN0aW9ucz9xdWVyeT13b3JrZmxvdyUzQWJ1aWxkKQpbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2l0aHViL3F1YW50bWluZC9haW8tb3BlbmFwaS9icmFuY2gvbWFpbi9ncmFwaC9iYWRnZS5zdmc/dG9rZW49WFYyR0Q5NDZRSSldKGh0dHBzOi8vY29kZWNvdi5pby9naXRodWIvcXVhbnRtaW5kL2Fpby1vcGVuYXBpKQpbIVtEb2N1bWVudGF0aW9uIFN0YXR1c10oaHR0cHM6Ly9yZWFkdGhlZG9jcy5vcmcvcHJvamVjdHMvYWlvLW9wZW5hcGkvYmFkZ2UvP3ZlcnNpb249bGF0ZXN0KV0oaHR0cHM6Ly9haW8tb3BlbmFwaS5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvP2JhZGdlPWxhdGVzdCkKWyFbRG93bmxvYWRzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvZGQvYWlvLW9wZW5hcGkuc3ZnKV0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2Fpby1vcGVuYXBpLykKCkFzeW5jaHJvbm91cyB3ZWIgbWlkZGxld2FyZSBmb3IgW2Fpb2h0dHBdW10gYW5kIHNlcnZpbmcgUmVzdCBBUElzIHdpdGggW09wZW5BUEldW10gdiAzCnNwZWNpZmljYXRpb24gYW5kIHdpdGggb3B0aW9uYWwgW1Bvc3RncmVTcWxdW10gZGF0YWJhc2UgYmluZGluZ3MuCgpTZWUgdGhlIFt0dXRvcmlhbF0oaHR0cHM6Ly9haW8tb3BlbmFwaS5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvdHV0b3JpYWwuaHRtbCkgZm9yIGEgcXVpY2sgaW50cm9kdWN0aW9uLgoKClthaW9odHRwXTogaHR0cHM6Ly9haW9odHRwLnJlYWR0aGVkb2NzLmlvL2VuL3N0YWJsZS8KW29wZW5hcGldOiBodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvCltwb3N0Z3Jlc3FsXTogaHR0cHM6Ly93d3cucG9zdGdyZXNxbC5vcmcvCltzcWxhbGNoZW15XTogaHR0cHM6Ly93d3cuc3FsYWxjaGVteS5vcmcvCltjbGlja106IGh0dHBzOi8vZ2l0aHViLmNvbS9wYWxsZXRzL2NsaWNrClthbGVtYmljXTogaHR0cDovL2FsZW1iaWMuenp6Y29tcHV0aW5nLmNvbS9lbi9sYXRlc3QvClthc3luY3BnXTogaHR0cHM6Ly9naXRodWIuY29tL01hZ2ljU3RhY2svYXN5bmNwZwo= readmeEtag: '"47846cd96146992f3d69208b90bf08036431f616"' readmeLastModified: Thu, 31 Aug 2023 20:48:01 GMT repositoryId: 133532636 description: >- A python module for building OpenAPI compliant asynchronous Rest Servers. Auto documentation, serialization and validation in the same unified API. created: '2018-05-15T15:00:21Z' updated: '2024-11-23T06:34:06Z' language: Python archived: true stars: 37 watchers: 3 forks: 3 owner: quantmind logo: https://avatars.githubusercontent.com/u/144099?v=4 license: BSD-3-Clause repoEtag: '"b852b6459f3cfd98740779dc511dd6e6bbc6281593e3ea2cd403b080f7fac684"' repoLastModified: Sat, 23 Nov 2024 06:34:06 GMT foundInMaster: true category: Server Implementations id: feb4cad12aee75a89800812a52852fdc - source: openapi3 tags repository: https://github.com/apidevtools/openapi-schemas v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNpZmljYXRpb24gU2NoZW1hcwoKWyFbQ3Jvc3MtUGxhdGZvcm0gQ29tcGF0aWJpbGl0eV0oaHR0cHM6Ly9hcGl0b29scy5kZXYvaW1nL2JhZGdlcy9vcy1iYWRnZXMuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL0FQSURldlRvb2xzL29wZW5hcGktc2NoZW1hcy9hY3Rpb25zKQpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9BUElEZXZUb29scy9vcGVuYXBpLXNjaGVtYXMvd29ya2Zsb3dzL0NJLUNEL2JhZGdlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly9naXRodWIuY29tL0FQSURldlRvb2xzL29wZW5hcGktc2NoZW1hcy9hY3Rpb25zKQoKWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvQVBJRGV2VG9vbHMvb3BlbmFwaS1zY2hlbWFzL2JhZGdlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL0FQSURldlRvb2xzL29wZW5hcGktc2NoZW1hcykKWyFbRGVwZW5kZW5jaWVzXShodHRwczovL2RhdmlkLWRtLm9yZy9BUElEZXZUb29scy9vcGVuYXBpLXNjaGVtYXMuc3ZnKV0oaHR0cHM6Ly9kYXZpZC1kbS5vcmcvQVBJRGV2VG9vbHMvb3BlbmFwaS1zY2hlbWFzKQoKWyFbbnBtXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L0BhcGlkZXZ0b29scy9vcGVuYXBpLXNjaGVtYXMuc3ZnKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQGFwaWRldnRvb2xzL29wZW5hcGktc2NoZW1hcykKWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vbC9AYXBpZGV2dG9vbHMvb3BlbmFwaS1zY2hlbWFzLnN2ZyldKExJQ0VOU0UpClshW0J1eSB1cyBhIHRyZWVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvVHJlZXdhcmUtJUYwJTlGJThDJUIzLWxpZ2h0Z3JlZW4pXShodHRwczovL3BsYW50LnRyZWV3YXJlLmVhcnRoL0FQSURldlRvb2xzL29wZW5hcGktc2NoZW1hcykKCgoKVGhpcyBwYWNrYWdlIGNvbnRhaW5zIFsqKnRoZSBvZmZpY2lhbCBKU09OIFNjaGVtYXMqKl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vdHJlZS9tYXN0ZXIvc2NoZW1hcykgZm9yIGV2ZXJ5IHZlcnNpb24gb2YgU3dhZ2dlci9PcGVuQVBJIFNwZWNpZmljYXRpb246Cgp8IFZlcnNpb24gfCBTY2hlbWEgfCBEb2NzCnwtLS0tLS0tLS18LS0tLS0tLS18LS0tLS0tLQp8IFN3YWdnZXIgMS4yICAgfCBbdjEuMiBzY2hlbWFdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL3RyZWUvbWFzdGVyL3NjaGVtYXMvdjEuMikgICAgICAgICAgICAgICB8IFt2MS4yIGRvY3NdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzEuMi5tZCkKfCBTd2FnZ2VyIDIuMCAgIHwgW3YyLjAgc2NoZW1hXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9zY2hlbWFzL3YyLjAvc2NoZW1hLmpzb24pICAgfCBbdjIuMCBkb2NzXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8yLjAubWQpCnwgT3BlbkFQSSAzLjAueCB8IFt2My4wLnggc2NoZW1hXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9zY2hlbWFzL3YzLjAvc2NoZW1hLmpzb24pIHwgW3YzLjAuMyBkb2NzXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMy5tZCkKfCBPcGVuQVBJIDMuMS54IHwgW3YzLjEueCBzY2hlbWFdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3NjaGVtYXMvdjMuMS9zY2hlbWEuanNvbikgfCBbdjMuMS4wIGRvY3NdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMS4wLm1kKQoKCkFsbCBzY2hlbWFzIGFyZSBrZXB0IHVwLXRvLWRhdGUgd2l0aCB0aGUgbGF0ZXN0IG9mZmljaWFsIGRlZmluaXRpb25zIHZpYSBhbiBhdXRvbWF0ZWQgQ0kvQ0Qgam9iLiDwn6SW8J+TpgoKCgpJbnN0YWxsYXRpb24KLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KWW91IGNhbiBpbnN0YWxsIE9wZW5BUEkgU2NoZW1hcyB2aWEgW25wbV0oaHR0cHM6Ly9kb2NzLm5wbWpzLmNvbS9hYm91dC1ucG0vKS4KCmBgYGJhc2gKbnBtIGluc3RhbGwgQGFwaWRldnRvb2xzL29wZW5hcGktc2NoZW1hcwpgYGAKCgoKVXNhZ2UKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KClRoZSBkZWZhdWx0IGV4cG9ydCBjb250YWlucyBhbGwgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIHZlcnNpb25zOgoKYGBgamF2YXNjcmlwdApjb25zdCBvcGVuYXBpID0gcmVxdWlyZSgiQGFwaWRldnRvb2xzL29wZW5hcGktc2NoZW1hcyIpOwoKY29uc29sZS5sb2cob3BlbmFwaS52MSk7ICAgIC8vIHsgJHNjaGVtYSwgaWQsIHByb3BlcnRpZXMsIGRlZmluaXRpb25zLCAuLi4gfQpjb25zb2xlLmxvZyhvcGVuYXBpLnYyKTsgICAgLy8geyAkc2NoZW1hLCBpZCwgcHJvcGVydGllcywgZGVmaW5pdGlvbnMsIC4uLiB9CmNvbnNvbGUubG9nKG9wZW5hcGkudjMpOyAgICAvLyB7ICRzY2hlbWEsIGlkLCBwcm9wZXJ0aWVzLCBkZWZpbml0aW9ucywgLi4uIH0KY29uc29sZS5sb2cob3BlbmFwaS52MzEpOyAgICAvLyB7ICRzY2hlbWEsIGlkLCBwcm9wZXJ0aWVzLCBkZWZpbml0aW9ucywgLi4uIH0KYGBgCgpPciB5b3UgY2FuIGltcG9ydCB0aGUgc3BlY2lmaWMgdmVyc2lvbihzKSB0aGF0IHlvdSBuZWVkOgoKYGBgamF2YXNjcmlwdApjb25zdCB7IG9wZW5hcGlWMSwgb3BlbmFwaVYyLCBvcGVuYXBpVjMsIG9wZW5hcGlWMzEgfSA9IHJlcXVpcmUoIkBhcGlkZXZ0b29scy9vcGVuYXBpLXNjaGVtYXMiKTsKCmNvbnNvbGUubG9nKG9wZW5hcGlWMSk7ICAgIC8vIHsgJHNjaGVtYSwgaWQsIHByb3BlcnRpZXMsIGRlZmluaXRpb25zLCAuLi4gfQpjb25zb2xlLmxvZyhvcGVuYXBpVjIpOyAgICAvLyB7ICRzY2hlbWEsIGlkLCBwcm9wZXJ0aWVzLCBkZWZpbml0aW9ucywgLi4uIH0KY29uc29sZS5sb2cob3BlbmFwaVYzKTsgICAgLy8geyAkc2NoZW1hLCBpZCwgcHJvcGVydGllcywgZGVmaW5pdGlvbnMsIC4uLiB9CmNvbnNvbGUubG9nKG9wZW5hcGlWMzEpOyAgICAvLyB7ICRzY2hlbWEsIGlkLCBwcm9wZXJ0aWVzLCBkZWZpbml0aW9ucywgLi4uIH0KYGBgCgpZb3UgY2FuIHVzZSBhIEpTT04gU2NoZW1hIHZhbGlkYXRvciBzdWNoIGFzIFtaLVNjaGVtYV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvei1zY2hlbWEpIG9yIFtBSlZdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL2FqdikgdG8gdmFsaWRhdGUgT3BlbkFQSSBkZWZpbml0aW9ucyBhZ2FpbnN0IHRoZSBzcGVjaWZpY2F0aW9uLgoKYGBgamF2YXNjcmlwdApjb25zdCB7IG9wZW5hcGlWMzEgfSA9IHJlcXVpcmUoIkBhcGlkZXZ0b29scy9vcGVuYXBpLXNjaGVtYXMiKTsKY29uc3QgWlNjaGVtYSA9IHJlcXVpcmUoInotc2NoZW1hIik7CgovLyBDcmVhdGUgYSBaU2NoZW1hIHZhbGlkYXRvcgpsZXQgdmFsaWRhdG9yID0gbmV3IFpTY2hlbWEoKTsKCi8vIFZhbGlkYXRlIGFuIE9wZW5BUEkgZGVmaW5pdGlvbiBhZ2FpbnN0IHRoZSBPcGVuQVBJIHYzLjAgc3BlY2lmaWNhdGlvbgp2YWxpZGF0b3IudmFsaWRhdGUob3BlbmFwaURlZmluaXRpb24sIG9wZW5hcGlWMzEpOwpgYGAKCgoKQ29udHJpYnV0aW5nCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCkNvbnRyaWJ1dGlvbnMsIGVuaGFuY2VtZW50cywgYW5kIGJ1Zy1maXhlcyBhcmUgd2VsY29tZSEgIFtPcGVuIGFuIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vQVBJRGV2VG9vbHMvb3BlbmFwaS1zY2hlbWFzL2lzc3Vlcykgb24gR2l0SHViIGFuZCBbc3VibWl0IGEgcHVsbCByZXF1ZXN0XShodHRwczovL2dpdGh1Yi5jb20vQVBJRGV2VG9vbHMvb3BlbmFwaS1zY2hlbWFzL3B1bGxzKS4KCiMjIyMgQnVpbGRpbmcKVG8gYnVpbGQgdGhlIHByb2plY3QgbG9jYWxseSBvbiB5b3VyIGNvbXB1dGVyOgoKMS4gX19DbG9uZSB0aGlzIHJlcG9fXzxicj4KYGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vQVBJRGV2VG9vbHMvb3BlbmFwaS1zY2hlbWFzLmdpdGAKCjIuIF9fSW5zdGFsbCBkZXBlbmRlbmNpZXNfXzxicj4KYG5wbSBpbnN0YWxsYAoKMy4gX19CdWlsZCB0aGUgY29kZV9fPGJyPgpgbnBtIHJ1biBidWlsZGAKCjQuIF9fUnVuIHRoZSB0ZXN0c19fPGJyPgpgbnBtIHRlc3RgCgoKCkxpY2Vuc2UKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KT3BlbkFQSSBTY2hlbWFzIGlzIDEwMCUgZnJlZSBhbmQgb3Blbi1zb3VyY2UsIHVuZGVyIHRoZSBbTUlUIGxpY2Vuc2VdKExJQ0VOU0UpLiBVc2UgaXQgaG93ZXZlciB5b3Ugd2FudC4KClRoaXMgcGFja2FnZSBpcyBbVHJlZXdhcmVdKGh0dHA6Ly90cmVld2FyZS5lYXJ0aCkuIElmIHlvdSB1c2UgaXQgaW4gcHJvZHVjdGlvbiwgdGhlbiB3ZSBhc2sgdGhhdCB5b3UgWyoqYnV5IHRoZSB3b3JsZCBhIHRyZWUqKl0oaHR0cHM6Ly9wbGFudC50cmVld2FyZS5lYXJ0aC9BUElEZXZUb29scy9vcGVuYXBpLXNjaGVtYXMpIHRvIHRoYW5rIHVzIGZvciBvdXIgd29yay4gQnkgY29udHJpYnV0aW5nIHRvIHRoZSBUcmVld2FyZSBmb3Jlc3QgeW914oCZbGwgYmUgY3JlYXRpbmcgZW1wbG95bWVudCBmb3IgbG9jYWwgZmFtaWxpZXMgYW5kIHJlc3RvcmluZyB3aWxkbGlmZSBoYWJpdGF0cy4KCgoKQmlnIFRoYW5rcyBUbwotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpUaGFua3MgdG8gdGhlc2UgYXdlc29tZSBjb21wYW5pZXMgZm9yIHRoZWlyIHN1cHBvcnQgb2YgT3BlbiBTb3VyY2UgZGV2ZWxvcGVycyDinaQKClshW0dpdEh1Yl0oaHR0cHM6Ly9hcGl0b29scy5kZXYvaW1nL2JhZGdlcy9naXRodWIuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL29wZW4tc291cmNlKQpbIVtOUE1dKGh0dHBzOi8vYXBpdG9vbHMuZGV2L2ltZy9iYWRnZXMvbnBtLnN2ZyldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS8pClshW0NvdmVyYWxsc10oaHR0cHM6Ly9hcGl0b29scy5kZXYvaW1nL2JhZGdlcy9jb3ZlcmFsbHMuc3ZnKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8pClshW1RyYXZpcyBDSV0oaHR0cHM6Ly9hcGl0b29scy5kZXYvaW1nL2JhZGdlcy90cmF2aXMtY2kuc3ZnKV0oaHR0cHM6Ly90cmF2aXMtY2kuY29tKQpbIVtTYXVjZUxhYnNdKGh0dHBzOi8vYXBpdG9vbHMuZGV2L2ltZy9iYWRnZXMvc2F1Y2UtbGFicy5zdmcpXShodHRwczovL3NhdWNlbGFicy5jb20pCg== readmeEtag: '"630689d883f00bf566fa29ef5b93e65f2cdee68c"' readmeLastModified: Sat, 01 May 2021 15:44:27 GMT repositoryId: 193236806 description: JSON Schemas for every version of the OpenAPI Specification created: '2019-06-22T13:26:54Z' updated: '2025-11-01T05:42:28Z' language: JavaScript archived: false stars: 45 watchers: 2 forks: 2 owner: APIDevTools logo: https://avatars.githubusercontent.com/u/43750074?v=4 license: MIT repoEtag: '"0c0a5efeeed8cbf96bf38b01ebfd7a2e94aa00ecf1d8ce6db99a8a77b862427c"' repoLastModified: Sat, 01 Nov 2025 05:42:28 GMT foundInMaster: true category: Parsers id: 1dbb49e9733b078e9e04a157e4d9dc56 - source: openapi3 tags repository: https://github.com/vymalo/keycloak-webhook v3: true id: ed4d4928e6bf8f5d23dd3eeccc015ab0 repositoryMetadata: base64Readme: >- # Keycloak Webhook Plugin

A modular Keycloak event listener plugin that triggers webhooks whenever specific events (like login, registration, or
logout) occur in Keycloak. This project leverages a multi-module design so you can choose which transport provider (
HTTP,
AMQP, or Syslog) to deploy based on your needs.

| Keycloak Version | Plugin Version |
|------------------|----------------|
| 21               | ✅ 0.10.0-rc.1  |
| 22               | ✅ 0.10.0-rc.1  |
| 23               | ✅ 0.10.0-rc.1  |
| 24               | ✅ 0.10.0-rc.1  |
| 25               | ✅ 0.10.0-rc.1  |
| 26               | ✅ 0.10.0-rc.1  |

---

## 1. What It Is

The Keycloak Webhook Plugin consists of four modules:

- **Core Module (`keycloak-webhook-provider-core`)**  
  Contains common SPI interfaces, shared models, and helper utilities.

- **AMQP Provider (`keycloak-webhook-provider-amqp`)**  
  Implements webhook notifications over AMQP (e.g., RabbitMQ). If the AMQP dependency is present on the classpath, this
  provider is loaded automatically.

- **HTTP Provider (`keycloak-webhook-provider-http`)**  
  Implements webhook notifications over HTTP. This provider uses OpenAPI-generated clients to ensure compliance with the
  target API.

- **Syslog Provider (`keycloak-webhook-provider-syslog`)**  
  Implements webhook notifications over Syslog (TCP/UDP). Supports RFC 3164 and RFC 5424 message formats.

Keycloak uses Java's `ServiceLoader` mechanism to conditionally load these providers at runtime if their JARs (and
dependencies) are available.

---

## 2. How to Use It

### Downloading the Plugins

Download the latest release artifacts (shaded JARs) from the GitHub releases page. For example, using `curl`:

```bash
# Replace <version> with the desired release version.

VERSION=<version>; curl -L -o keycloak-webhook-provider-core.jar https://github.com/vymalo/keycloak-webhook/releases/download/v${VERSION}/keycloak-webhook-provider-core-${VERSION}-all.jar; \
curl -L -o keycloak-webhook-provider-amqp.jar https://github.com/vymalo/keycloak-webhook/releases/download/v${VERSION}/keycloak-webhook-provider-amqp-${VERSION}-all.jar; \
curl -L -o keycloak-webhook-provider-http.jar https://github.com/vymalo/keycloak-webhook/releases/download/v${VERSION}/keycloak-webhook-provider-http-${VERSION}-all.jar; \
curl -L -o keycloak-webhook-provider-syslog.jar https://github.com/vymalo/keycloak-webhook/releases/download/v${VERSION}/keycloak-webhook-provider-syslog-${VERSION}-all.jar; \
```

### a. Docker

When running Keycloak in Docker, mount the downloaded JARs into Keycloak's providers directory. For example, in your
`docker-compose.yaml`:

```yaml
services:
  keycloak:
    image: quay.io/keycloak/keycloak:26.4.0
    ports:
      - "9100:9100"
    environment:
      # HTTP Provider Configuration
      WEBHOOK_HTTP_BASE_PATH: "http://prism:4010"
      WEBHOOK_HTTP_AUTH_USERNAME: "admin"
      WEBHOOK_HTTP_AUTH_PASSWORD: "password"
      # AMQP Provider Configuration
      WEBHOOK_AMQP_HOST: rabbitmq
      WEBHOOK_AMQP_USERNAME: username
      WEBHOOK_AMQP_PASSWORD: password
      WEBHOOK_AMQP_PORT: 5672
      WEBHOOK_AMQP_VHOST: "/"
      WEBHOOK_AMQP_EXCHANGE: keycloak
      WEBHOOK_AMQP_SSL: "no"
      # Syslog Provider Configuration
      WEBHOOK_SYSLOG_PROTOCOL: udp
      WEBHOOK_SYSLOG_HOSTNAME: keycloak
      WEBHOOK_SYSLOG_APP_NAME: Keycloak
      WEBHOOK_SYSLOG_FACILITY: USER
      WEBHOOK_SYSLOG_SEVERITY: INFORMATIONAL
      WEBHOOK_SYSLOG_SERVER_HOSTNAME: syslog-ng
      WEBHOOK_SYSLOG_SERVER_PORT: 5514
      WEBHOOK_SYSLOG_MESSAGE_FORMAT: RFC_5425
      # Keycloak Admin Credentials
      KEYCLOAK_ADMIN: admin
      KEYCLOAK_ADMIN_PASSWORD: password
      KC_HTTP_PORT: 9100
      KC_METRICS_ENABLED: "true"
      KC_LOG_CONSOLE_COLOR: "true"
      KC_HEALTH_ENABLED: "true"
    entrypoint: /bin/sh
    command:
      - -c
      - |
        set -ex
        # Copy all plugin JARs from the mounted volume into Keycloak's providers folder
        cp /tmp/plugins/*.jar /opt/keycloak/providers
        /opt/keycloak/bin/kc.sh start-dev --import-realm
    volumes:
      - ./plugins:/tmp/plugins:ro # Place your downloaded JARs in this folder
      - ./.docker/keycloak-config/:/opt/keycloak/data/import/:ro
```

### b. Kubernetes

In Kubernetes, you can use an init container to download the plugin JARs from GitHub artifacts and copy them into
Keycloak's providers folder. For example:

```yaml
apiVersion: apps/v1
kind: Deployment
metadata:
  name: keycloak
spec:
  replicas: 1
  selector:
    matchLabels:
      app: keycloak
  template:
    metadata:
      labels:
        app: keycloak
    spec:
      volumes:
        - name: providers-volume
          emptyDir: { }
      initContainers:
        - name: download-plugins
          image: curlimages/curl:8.1.2
          command:
            - sh
            - -c
            - |
              mkdir -p /plugins
              # Download the plugins from GitHub releases (update the URLs accordingly)
              curl -L -o /plugins/keycloak-webhook-provider-core.jar https://github.com/vymalo/keycloak-webhook/releases/download/v<version>/keycloak-webhook-provider-core-<version>-all.jar
              curl -L -o /plugins/keycloak-webhook-provider-amqp.jar https://github.com/vymalo/keycloak-webhook/releases/download/v<version>/keycloak-webhook-provider-amqp-<version>-all.jar
              curl -L -o /plugins/keycloak-webhook-provider-http.jar https://github.com/vymalo/keycloak-webhook/releases/download/v<version>/keycloak-webhook-provider-http-<version>-all.jar
              curl -L -o /plugins/keycloak-webhook-provider-syslog.jar https://github.com/vymalo/keycloak-webhook/releases/download/v<version>/keycloak-webhook-provider-syslog-<version>-all.jar
              cp /plugins/*.jar /providers/
          volumeMounts:
            - name: providers-volume
              mountPath: /providers
      containers:
        - name: keycloak
          image: quay.io/keycloak/keycloak:26.4.0
          env:
            - name: WEBHOOK_HTTP_BASE_PATH
              value: "http://prism:4010"
            - name: WEBHOOK_HTTP_AUTH_USERNAME
              value: "admin"
            - name: WEBHOOK_HTTP_AUTH_PASSWORD
              value: "password"
            - name: WEBHOOK_AMQP_HOST
              value: "rabbitmq"
            - name: WEBHOOK_AMQP_USERNAME
              value: "username"
            - name: WEBHOOK_AMQP_PASSWORD
              value: "password"
            - name: WEBHOOK_AMQP_PORT
              value: "5672"
            - name: WEBHOOK_AMQP_VHOST
              value: "/"
            - name: WEBHOOK_AMQP_EXCHANGE
              value: "keycloak"
            - name: WEBHOOK_AMQP_SSL
              value: "no"
            - name: WEBHOOK_SYSLOG_PROTOCOL
              value: "udp"
            - name: WEBHOOK_SYSLOG_SERVER_HOSTNAME
              value: "syslog-ng"
            - name: WEBHOOK_SYSLOG_SERVER_PORT
              value: "5514"
          volumeMounts:
            - name: providers-volume
              mountPath: /opt/keycloak/providers
```

---

## 3. Environment Variables

### HTTP Provider

- **`WEBHOOK_HTTP_BASE_PATH`**  
  Coma-separated list of endpoint URLs where webhook requests are sent.

- **`WEBHOOK_HTTP_AUTH_USERNAME` (optional)**  
  Basic auth username.

- **`WEBHOOK_HTTP_AUTH_PASSWORD` (optional)**  
  Basic auth password.

### AMQP Provider

- **`WEBHOOK_AMQP_HOST`**  
  RabbitMQ server hostname.

- **`WEBHOOK_AMQP_USERNAME`**  
  Username for RabbitMQ.

- **`WEBHOOK_AMQP_PASSWORD`**  
  Password for RabbitMQ.

- **`WEBHOOK_AMQP_PORT`**  
  Port for RabbitMQ.

- **`WEBHOOK_AMQP_VHOST` (optional)**  
  Virtual host for RabbitMQ.

- **`WEBHOOK_AMQP_EXCHANGE`**  
  Exchange name for RabbitMQ.

- **`WEBHOOK_AMQP_SSL` (optional)**  
  `"yes"` or `"no"`, indicating if SSL is enabled.

### Syslog Provider

- **`WEBHOOK_SYSLOG_PROTOCOL`**  
  `"TCP"` or `"UDP"` protocol for Syslog communication.

- **`WEBHOOK_SYSLOG_HOSTNAME`**  
  Hostname of the Keycloak instance.

- **`WEBHOOK_SYSLOG_APP_NAME`**  
  Application name for Syslog messages.

- **`WEBHOOK_SYSLOG_FACILITY`**  
  Syslog facility (e.g., USER, DAEMON, AUTH).

- **`WEBHOOK_SYSLOG_SEVERITY`**  
  Syslog severity level (e.g., INFORMATIONAL, WARNING, ERROR).

- **`WEBHOOK_SYSLOG_SERVER_HOSTNAME`**  
  Hostname of the Syslog server.

- **`WEBHOOK_SYSLOG_SERVER_PORT`**  
  Port of the Syslog server.

- **`WEBHOOK_SYSLOG_MESSAGE_FORMAT`**  
  `"RFC_3164"`, `"RFC_5424"` or `"RFC_5425"` message format.

- **`WEBHOOK_EVENTS_TAKEN` (optional)**  
  A comma-separated list of Keycloak events (e.g., `"LOGIN,REGISTER,LOGOUT"`) that should trigger webhooks. If not
  specified, all events are processed.

---

## 4. Architecture

The architecture of the Keycloak Webhook Plugin is illustrated using a Mermaid diagram below:

```mermaid
graph TD
    A["Keycloak (Event Source)"]
    B["ServiceLoader (SPI)"]
    C[Core Module]
    D[HTTP Provider]
    E[AMQP Provider]
    F[Syslog Provider]
    G[External HTTP Server]
    H[RabbitMQ Broker]
    I[Syslog Server]
    A --> B
    B --> C
    C --> D
    C --> E
    C --> F
    D --> G
    E --> H
    F --> I
```

- **Core Module:**  
  Provides common interfaces, models, and utilities.

- **Provider Modules:**  
  Implement specific webhook delivery mechanisms (HTTP, AMQP, or Syslog) and are conditionally loaded if their JARs are
  present.

- **ServiceLoader:**  
  Uses Java's SPI to discover and load the providers.

- **External Systems:**  
  Webhook notifications are sent to an HTTP server, published to a RabbitMQ broker, or forwarded to a Syslog server.

---

## 5. Contribute

We welcome contributions! To get started:

1. **Fork the Repository:**  
   Create your own fork of the project on GitHub.

2. **Set Up Your Development Environment:**

- Clone your fork locally.
- Ensure you have JDK 17 and Gradle installed.
- Build the project using:
  ```bash
  ./gradlew clean shadow
  ```

3. **Follow Code Conventions:**

- Keep the code style consistent with the existing modules.
- Write tests where applicable.
- Update the README and documentation if your changes require it.

4. **Submit a Pull Request:**  
   Open a pull request with your proposed changes. Please include a detailed description and reference any related
   issues.

5. **Join Discussions:**  
   Use GitHub issues to discuss ideas, report bugs, or ask for help.

---

This modular and flexible design allows you to deploy only the providers you need while keeping the project maintainable
and extensible. Happy coding!
 readmeEtag: '"4267215ab32a08545cffb81def25c6687abff42e"' readmeLastModified: Sun, 16 Nov 2025 08:42:39 GMT repositoryId: 628094839 description: Event-based Webhook plugin for Keycloak created: '2023-04-14T22:47:31Z' updated: '2026-01-31T13:25:51Z' language: Kotlin archived: false stars: 95 watchers: 1 forks: 30 owner: vymalo logo: https://avatars.githubusercontent.com/u/128943481?v=4 license: MIT repoEtag: '"09767b061ee6445dea21007b3c70685da400ef955c0994e20f826bd2c3c764c7"' repoLastModified: Sat, 31 Jan 2026 13:25:51 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/namsor/namsor-python-sdk2 v3: true repositoryMetadata: base64Readme: >- # openapi-client
NamSor API v2 : enpoints to process personal names (gender, cultural origin or ethnicity) in all alphabets or languages. By default, enpoints use 1 unit per name (ex. Gender), but Ethnicity classification uses 10 to 20 units per name depending on taxonomy. Use GET methods for small tests, but prefer POST methods for higher throughput (batch processing of up to 100 names at a time). Need something you can't find here? We have many more features coming soon. Let us know, we'll do our best to add it! 

This Python package is automatically generated by the [OpenAPI Generator](https://openapi-generator.tech) project:

- API version: 2.0.27
- Package version: 2.0.27
- Build package: org.openapitools.codegen.languages.PythonClientCodegen
For more information, please visit [http://www.namsor.com/](http://www.namsor.com/)

## Requirements.

Python &gt;&#x3D;3.7

## Migration from other generators like python and python-legacy

### Changes
1. This generator uses spec case for all (object) property names and parameter names.
    - So if the spec has a property name like camelCase, it will use camelCase rather than camel_case
    - So you will need to update how you input and read properties to use spec case
2. Endpoint parameters are stored in dictionaries to prevent collisions (explanation below)
    - So you will need to update how you pass data in to endpoints
3. Endpoint responses now include the original response, the deserialized response body, and (todo)the deserialized headers
    - So you will need to update your code to use response.body to access deserialized data
4. All validated data is instantiated in an instance that subclasses all validated Schema classes and Decimal/str/list/tuple/frozendict/NoneClass/BoolClass/bytes/io.FileIO
    - This means that you can use isinstance to check if a payload validated against a schema class
    - This means that no data will be of type None/True/False
        - ingested None will subclass NoneClass
        - ingested True will subclass BoolClass
        - ingested False will subclass BoolClass
        - So if you need to check is True/False/None, instead use instance.is_true_oapg()/.is_false_oapg()/.is_none_oapg()
5. All validated class instances are immutable except for ones based on io.File
    - This is because if properties were changed after validation, that validation would no longer apply
    - So no changing values or property values after a class has been instantiated
6. String + Number types with formats
    - String type data is stored as a string and if you need to access types based on its format like date,
    date-time, uuid, number etc then you will need to use accessor functions on the instance
    - type string + format: See .as_date_oapg, .as_datetime_oapg, .as_decimal_oapg, .as_uuid_oapg
    - type number + format: See .as_float_oapg, .as_int_oapg
    - this was done because openapi/json-schema defines constraints. string data may be type string with no format
    keyword in one schema, and include a format constraint in another schema
    - So if you need to access a string format based type, use as_date_oapg/as_datetime_oapg/as_decimal_oapg/as_uuid_oapg
    - So if you need to access a number format based type, use as_int_oapg/as_float_oapg
7. Property access on AnyType(type unset) or object(dict) schemas
    - Only required keys with valid python names are properties like .someProp and have type hints
    - All optional keys may not exist, so properties are not defined for them
    - One can access optional values with dict_instance['optionalProp'] and KeyError will be raised if it does not exist
    - Use get_item_oapg if you need a way to always get a value whether or not the key exists
        - If the key does not exist, schemas.unset is returned from calling dict_instance.get_item_oapg('optionalProp')
        - All required and optional keys have type hints for this method, and @typing.overload is used
        - A type hint is also generated for additionalProperties accessed using this method
    - So you will need to update you code to use some_instance['optionalProp'] to access optional property
    and additionalProperty values
8. The location of the api classes has changed
    - Api classes are located in your_package.apis.tags.some_api
    - This change was made to eliminate redundant code generation
    - Legacy generators generated the same endpoint twice if it had > 1 tag on it
    - This generator defines an endpoint in one class, then inherits that class to generate
      apis by tags and by paths
    - This change reduces code and allows quicker run time if you use the path apis
        - path apis are at your_package.apis.paths.some_path
    - Those apis will only load their needed models, which is less to load than all of the resources needed in a tag api
    - So you will need to update your import paths to the api classes

### Why are Oapg and _oapg used in class and method names?
Classes can have arbitrarily named properties set on them
Endpoints can have arbitrary operationId method names set
For those reasons, I use the prefix Oapg and _oapg to greatly reduce the likelihood of collisions
on protected + public classes/methods.
oapg stands for OpenApi Python Generator.

### Object property spec case
This was done because when payloads are ingested, they can be validated against N number of schemas.
If the input signature used a different property name then that has mutated the payload.
So SchemaA and SchemaB must both see the camelCase spec named variable.
Also it is possible to send in two properties, named camelCase and camel_case in the same payload.
That use case should be support so spec case is used.

### Parameter spec case
Parameters can be included in different locations including:
- query
- path
- header
- cookie

Any of those parameters could use the same parameter names, so if every parameter
was included as an endpoint parameter in a function signature, they would collide.
For that reason, each of those inputs have been separated out into separate typed dictionaries:
- query_params
- path_params
- header_params
- cookie_params

So when updating your code, you will need to pass endpoint parameters in using those
dictionaries.

### Endpoint responses
Endpoint responses have been enriched to now include more information.
Any response reom an endpoint will now include the following properties:
response: urllib3.HTTPResponse
body: typing.Union[Unset, Schema]
headers: typing.Union[Unset, TODO]
Note: response header deserialization has not yet been added


## Installation & Usage
### pip install

If the python package is hosted on a repository, you can install directly using:

```sh
pip install git+https://github.com/namsor/namsor-python-sdk2.git
```
(you may need to run `pip` with root permission: `sudo pip install git+https://github.com/namsor/namsor-python-sdk2.git`)

Then import the package:
```python
import openapi_client
```

### Setuptools

Install via [Setuptools](http://pypi.python.org/pypi/setuptools).

```sh
python setup.py install --user
```
(or `sudo python setup.py install` to install the package for all users)

Then import the package:
```python
import openapi_client
```

## Getting Started

Please follow the [installation procedure](#installation--usage) and then run the following:

```python

import time
import openapi_client
from pprint import pprint
from openapi_client.apis.tags import admin_api
from openapi_client.model.api_classifier_taxonomy_out import APIClassifierTaxonomyOut
from openapi_client.model.api_classifiers_status_out import APIClassifiersStatusOut
from openapi_client.model.api_key_out import APIKeyOut
from openapi_client.model.api_period_usage_out import APIPeriodUsageOut
from openapi_client.model.api_services_out import APIServicesOut
from openapi_client.model.api_usage_aggregated_out import APIUsageAggregatedOut
from openapi_client.model.api_usage_history_out import APIUsageHistoryOut
from openapi_client.model.region_out import RegionOut
from openapi_client.model.software_version_out import SoftwareVersionOut
# Defining the host is optional and defaults to https://v2.namsor.com/NamSorAPIv2
# See configuration.py for a list of all supported configuration parameters.
configuration = openapi_client.Configuration(
    host = "https://v2.namsor.com/NamSorAPIv2"
)

# The client must configure the authentication and authorization parameters
# in accordance with the API server security policy.
# Examples for each auth method are provided below, use the example that
# satisfies your auth use case.

# Configure API key authorization: api_key
configuration.api_key['api_key'] = 'YOUR_API_KEY'

# Uncomment below to setup prefix (e.g. Bearer) for API key, if needed
# configuration.api_key_prefix['api_key'] = 'Bearer'

# Enter a context with an instance of the API client
with openapi_client.ApiClient(configuration) as api_client:
    # Create an instance of the API class
    api_instance = admin_api.AdminApi(api_client)
    source = "source_example" # str | 
anonymized = True # bool | 
token = "token_example" # str | 

    try:
        # Activate/deactivate anonymization for a source.
        api_response = api_instance.anonymize(sourceanonymizedtoken)
        pprint(api_response)
    except openapi_client.ApiException as e:
        print("Exception when calling AdminApi->anonymize: %s\n" % e)
```

## Documentation for API Endpoints

All URIs are relative to *https://v2.namsor.com/NamSorAPIv2*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AdminApi* | [**anonymize**](docs/apis/tags/AdminApi.md#anonymize) | **get** /api2/json/anonymize/{source}/{anonymized}/{token} | Activate/deactivate anonymization for a source.
*AdminApi* | [**anonymize1**](docs/apis/tags/AdminApi.md#anonymize1) | **get** /api2/json/anonymize/{source}/{anonymized} | Activate/deactivate anonymization for a source.
*AdminApi* | [**api_key_info**](docs/apis/tags/AdminApi.md#api_key_info) | **get** /api2/json/apiKeyInfo | Read API Key info.
*AdminApi* | [**api_status**](docs/apis/tags/AdminApi.md#api_status) | **get** /api2/json/apiStatus | Prints the current status of the classifiers. A classifier name in apiStatus corresponds to a service name in apiServices.
*AdminApi* | [**api_usage**](docs/apis/tags/AdminApi.md#api_usage) | **get** /api2/json/apiUsage | Print current API usage.
*AdminApi* | [**api_usage_history**](docs/apis/tags/AdminApi.md#api_usage_history) | **get** /api2/json/apiUsageHistory | Print historical API usage.
*AdminApi* | [**api_usage_history_aggregate**](docs/apis/tags/AdminApi.md#api_usage_history_aggregate) | **get** /api2/json/apiUsageHistoryAggregate | Print historical API usage (in an aggregated view, by service, by day/hour/min).
*AdminApi* | [**available_services**](docs/apis/tags/AdminApi.md#available_services) | **get** /api2/json/apiServices | List of classification services and usage cost in Units per classification (default is 1&#x3D;ONE Unit). Some API endpoints (ex. Corridor) combine multiple classifiers.
*AdminApi* | [**disable**](docs/apis/tags/AdminApi.md#disable) | **get** /api2/json/disable/{source}/{disabled} | Activate/deactivate an API Key.
*AdminApi* | [**learnable**](docs/apis/tags/AdminApi.md#learnable) | **get** /api2/json/learnable/{source}/{learnable}/{token} | Activate/deactivate learning from a source.
*AdminApi* | [**learnable1**](docs/apis/tags/AdminApi.md#learnable1) | **get** /api2/json/learnable/{source}/{learnable} | Activate/deactivate learning from a source.
*AdminApi* | [**regions**](docs/apis/tags/AdminApi.md#regions) | **get** /api2/json/regions | Print basic source statistics.
*AdminApi* | [**software_version**](docs/apis/tags/AdminApi.md#software_version) | **get** /api2/json/softwareVersion | Get the current software version
*AdminApi* | [**taxonomy_classes**](docs/apis/tags/AdminApi.md#taxonomy_classes) | **get** /api2/json/taxonomyClasses/{classifierName} | Print the taxonomy classes valid for the given classifier.
*ChineseApi* | [**chinese_name_candidates**](docs/apis/tags/ChineseApi.md#chinese_name_candidates) | **get** /api2/json/chineseNameCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming
*ChineseApi* | [**chinese_name_candidates_batch**](docs/apis/tags/ChineseApi.md#chinese_name_candidates_batch) | **post** /api2/json/chineseNameCandidatesBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**chinese_name_candidates_gender_batch**](docs/apis/tags/ChineseApi.md#chinese_name_candidates_gender_batch) | **post** /api2/json/chineseNameCandidatesGenderBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname) ex. Wang Xiaoming.
*ChineseApi* | [**chinese_name_gender_candidates**](docs/apis/tags/ChineseApi.md#chinese_name_gender_candidates) | **get** /api2/json/chineseNameGenderCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin}/{knownGender} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming - having a known gender (&#x27;male&#x27; or &#x27;female&#x27;)
*ChineseApi* | [**chinese_name_match**](docs/apis/tags/ChineseApi.md#chinese_name_match) | **get** /api2/json/chineseNameMatch/{chineseSurnameLatin}/{chineseGivenNameLatin}/{chineseName} | Return a score for matching Chinese name ex. 王晓明 with a romanized name ex. Wang Xiaoming
*ChineseApi* | [**chinese_name_match_batch**](docs/apis/tags/ChineseApi.md#chinese_name_match_batch) | **post** /api2/json/chineseNameMatchBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**gender_chinese_name**](docs/apis/tags/ChineseApi.md#gender_chinese_name) | **get** /api2/json/genderChineseName/{chineseName} | Infer the likely gender of a Chinese full name ex. 王晓明
*ChineseApi* | [**gender_chinese_name_batch**](docs/apis/tags/ChineseApi.md#gender_chinese_name_batch) | **post** /api2/json/genderChineseNameBatch | Infer the likely gender of up to 100 full names ex. 王晓明
*ChineseApi* | [**gender_chinese_name_pinyin**](docs/apis/tags/ChineseApi.md#gender_chinese_name_pinyin) | **get** /api2/json/genderChineseNamePinyin/{chineseSurnameLatin}/{chineseGivenNameLatin} | Infer the likely gender of a Chinese name in LATIN (Pinyin).
*ChineseApi* | [**gender_chinese_name_pinyin_batch**](docs/apis/tags/ChineseApi.md#gender_chinese_name_pinyin_batch) | **post** /api2/json/genderChineseNamePinyinBatch | Infer the likely gender of up to 100 Chinese names in LATIN (Pinyin).
*ChineseApi* | [**parse_chinese_name**](docs/apis/tags/ChineseApi.md#parse_chinese_name) | **get** /api2/json/parseChineseName/{chineseName} | Infer the likely first/last name structure of a name, ex. 王晓明 -&gt; 王(surname) 晓明(given name)
*ChineseApi* | [**parse_chinese_name_batch**](docs/apis/tags/ChineseApi.md#parse_chinese_name_batch) | **post** /api2/json/parseChineseNameBatch | Infer the likely first/last name structure of a name, ex. 王晓明 -&gt; 王(surname) 晓明(given name).
*ChineseApi* | [**pinyin_chinese_name**](docs/apis/tags/ChineseApi.md#pinyin_chinese_name) | **get** /api2/json/pinyinChineseName/{chineseName} | Romanize the Chinese name to Pinyin, ex. 王晓明 -&gt; Wang (surname) Xiaoming (given name)
*ChineseApi* | [**pinyin_chinese_name_batch**](docs/apis/tags/ChineseApi.md#pinyin_chinese_name_batch) | **post** /api2/json/pinyinChineseNameBatch | Romanize a list of Chinese name to Pinyin, ex. 王晓明 -&gt; Wang (surname) Xiaoming (given name).
*GeneralApi* | [**name_type**](docs/apis/tags/GeneralApi.md#name_type) | **get** /api2/json/nameType/{properNoun} | Infer the likely type of a proper noun (personal name, brand name, place name etc.)
*GeneralApi* | [**name_type_batch**](docs/apis/tags/GeneralApi.md#name_type_batch) | **post** /api2/json/nameTypeBatch | Infer the likely common type of up to 100 proper nouns (personal name, brand name, place name etc.)
*GeneralApi* | [**name_type_geo**](docs/apis/tags/GeneralApi.md#name_type_geo) | **get** /api2/json/nameTypeGeo/{properNoun}/{countryIso2} | Infer the likely type of a proper noun (personal name, brand name, place name etc.)
*GeneralApi* | [**name_type_geo_batch**](docs/apis/tags/GeneralApi.md#name_type_geo_batch) | **post** /api2/json/nameTypeGeoBatch | Infer the likely common type of up to 100 proper nouns (personal name, brand name, place name etc.)
*IndianApi* | [**caste_indian_batch**](docs/apis/tags/IndianApi.md#caste_indian_batch) | **post** /api2/json/casteIndianBatch | [USES 10 UNITS PER NAME] Infer the likely Indian name caste of up to 100 personal Indian Hindu names. 
*IndianApi* | [**castegroup_indian**](docs/apis/tags/IndianApi.md#castegroup_indian) | **get** /api2/json/castegroupIndian/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of a first / last name.
*IndianApi* | [**castegroup_indian_batch**](docs/apis/tags/IndianApi.md#castegroup_indian_batch) | **post** /api2/json/castegroupIndianBatch | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of up to 100 personal first / last names. 
*IndianApi* | [**castegroup_indian_full**](docs/apis/tags/IndianApi.md#castegroup_indian_full) | **get** /api2/json/castegroupIndianFull/{subDivisionIso31662}/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of a personal full name.
*IndianApi* | [**castegroup_indian_full_batch**](docs/apis/tags/IndianApi.md#castegroup_indian_full_batch) | **post** /api2/json/castegroupIndianFullBatch | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of up to 100 personal full names. 
*IndianApi* | [**castegroup_indian_hindu**](docs/apis/tags/IndianApi.md#castegroup_indian_hindu) | **get** /api2/json/casteIndian/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely Indian name caste of a personal Hindu name.
*IndianApi* | [**religion**](docs/apis/tags/IndianApi.md#religion) | **get** /api2/json/religionIndianFull/{subDivisionIso31662}/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal Indian full name, provided the Indian state or Union territory (NB/ this can be inferred using the subclassification endpoint).
*IndianApi* | [**religion1**](docs/apis/tags/IndianApi.md#religion1) | **get** /api2/json/religionIndian/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal Indian first/last name, provided the Indian state or Union territory (NB/ this can be inferred using the subclassification endpoint).
*IndianApi* | [**religion_indian_batch**](docs/apis/tags/IndianApi.md#religion_indian_batch) | **post** /api2/json/religionIndianBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal first/last Indian names, provided the subclassification at State or Union territory level (NB/ can be inferred using the subclassification endpoint).
*IndianApi* | [**religion_indian_full_batch**](docs/apis/tags/IndianApi.md#religion_indian_full_batch) | **post** /api2/json/religionIndianFullBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal full Indian names, provided the subclassification at State or Union territory level (NB/ can be inferred using the subclassification endpoint).
*IndianApi* | [**subclassification_indian**](docs/apis/tags/IndianApi.md#subclassification_indian) | **get** /api2/json/subclassificationIndian/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on the name.
*IndianApi* | [**subclassification_indian_batch**](docs/apis/tags/IndianApi.md#subclassification_indian_batch) | **post** /api2/json/subclassificationIndianBatch | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on a list of up to 100 names.
*IndianApi* | [**subclassification_indian_full**](docs/apis/tags/IndianApi.md#subclassification_indian_full) | **get** /api2/json/subclassificationIndianFull/{fullName} | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on the name.
*IndianApi* | [**subclassification_indian_full_batch**](docs/apis/tags/IndianApi.md#subclassification_indian_full_batch) | **post** /api2/json/subclassificationIndianFullBatch | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on a list of up to 100 names.
*JapaneseApi* | [**gender_japanese_name_full**](docs/apis/tags/JapaneseApi.md#gender_japanese_name_full) | **get** /api2/json/genderJapaneseNameFull/{japaneseName} | Infer the likely gender of a Japanese full name ex. 王晓明
*JapaneseApi* | [**gender_japanese_name_full_batch**](docs/apis/tags/JapaneseApi.md#gender_japanese_name_full_batch) | **post** /api2/json/genderJapaneseNameFullBatch | Infer the likely gender of up to 100 full names
*JapaneseApi* | [**gender_japanese_name_pinyin**](docs/apis/tags/JapaneseApi.md#gender_japanese_name_pinyin) | **get** /api2/json/genderJapaneseName/{japaneseSurname}/{japaneseGivenName} | Infer the likely gender of a Japanese name in LATIN (Pinyin).
*JapaneseApi* | [**gender_japanese_name_pinyin_batch**](docs/apis/tags/JapaneseApi.md#gender_japanese_name_pinyin_batch) | **post** /api2/json/genderJapaneseNameBatch | Infer the likely gender of up to 100 Japanese names in LATIN (Pinyin).
*JapaneseApi* | [**japanese_name_gender_kanji_candidates_batch**](docs/apis/tags/JapaneseApi.md#japanese_name_gender_kanji_candidates_batch) | **post** /api2/json/japaneseNameGenderKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName &#x3D; japaneseGivenName; lastName&#x3D;japaneseSurname) with KNOWN gender, ex. Yamamoto Sanae
*JapaneseApi* | [**japanese_name_kanji_candidates**](docs/apis/tags/JapaneseApi.md#japanese_name_kanji_candidates) | **get** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**japanese_name_kanji_candidates1**](docs/apis/tags/JapaneseApi.md#japanese_name_kanji_candidates1) | **get** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{knownGender} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae - and a known gender.
*JapaneseApi* | [**japanese_name_kanji_candidates_batch**](docs/apis/tags/JapaneseApi.md#japanese_name_kanji_candidates_batch) | **post** /api2/json/japaneseNameKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName &#x3D; japaneseGivenName; lastName&#x3D;japaneseSurname), ex. Yamamoto Sanae
*JapaneseApi* | [**japanese_name_latin_candidates**](docs/apis/tags/JapaneseApi.md#japanese_name_latin_candidates) | **get** /api2/json/japaneseNameLatinCandidates/{japaneseSurnameKanji}/{japaneseGivenNameKanji} | Romanize japanese name, based on the name in Kanji.
*JapaneseApi* | [**japanese_name_latin_candidates_batch**](docs/apis/tags/JapaneseApi.md#japanese_name_latin_candidates_batch) | **post** /api2/json/japaneseNameLatinCandidatesBatch | Romanize japanese names, based on the name in KANJI
*JapaneseApi* | [**japanese_name_match**](docs/apis/tags/JapaneseApi.md#japanese_name_match) | **get** /api2/json/japaneseNameMatch/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | Return a score for matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**japanese_name_match_batch**](docs/apis/tags/JapaneseApi.md#japanese_name_match_batch) | **post** /api2/json/japaneseNameMatchBatch | Return a score for matching a list of Japanese names in KANJI ex. 山本 早苗 with romanized names ex. Yamamoto Sanae
*JapaneseApi* | [**japanese_name_match_feedback_loop**](docs/apis/tags/JapaneseApi.md#japanese_name_match_feedback_loop) | **get** /api2/json/japaneseNameMatchFeedbackLoop/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | [CREDITS 1 UNIT] Feedback loop to better perform matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**parse_japanese_name**](docs/apis/tags/JapaneseApi.md#parse_japanese_name) | **get** /api2/json/parseJapaneseName/{japaneseName} | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae
*JapaneseApi* | [**parse_japanese_name_batch**](docs/apis/tags/JapaneseApi.md#parse_japanese_name_batch) | **post** /api2/json/parseJapaneseNameBatch | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae 
*PersonalApi* | [**corridor**](docs/apis/tags/PersonalApi.md#corridor) | **get** /api2/json/corridor/{countryIso2From}/{firstNameFrom}/{lastNameFrom}/{countryIso2To}/{firstNameTo}/{lastNameTo} | [USES 20 UNITS PER NAME COUPLE] Infer several classifications for a cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**corridor_batch**](docs/apis/tags/PersonalApi.md#corridor_batch) | **post** /api2/json/corridorBatch | [USES 20 UNITS PER NAME PAIR] Infer several classifications for up to 100 cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**country**](docs/apis/tags/PersonalApi.md#country) | **get** /api2/json/country/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely country of residence of a personal full name, or one surname. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**country_batch**](docs/apis/tags/PersonalApi.md#country_batch) | **post** /api2/json/countryBatch | [USES 10 UNITS PER NAME] Infer the likely country of residence of up to 100 personal full names, or surnames. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**diaspora**](docs/apis/tags/PersonalApi.md#diaspora) | **get** /api2/json/diaspora/{countryIso2}/{firstName}/{lastName} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**diaspora_batch**](docs/apis/tags/PersonalApi.md#diaspora_batch) | **post** /api2/json/diasporaBatch | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**gender**](docs/apis/tags/PersonalApi.md#gender) | **get** /api2/json/gender/{firstName} | Infer the likely gender of a just a fiven name, assuming default &#x27;US&#x27; local context. Please use preferably full names and local geographic context for better accuracy.
*PersonalApi* | [**gender1**](docs/apis/tags/PersonalApi.md#gender1) | **get** /api2/json/gender/{firstName}/{lastName} | Infer the likely gender of a name.
*PersonalApi* | [**gender_batch**](docs/apis/tags/PersonalApi.md#gender_batch) | **post** /api2/json/genderBatch | Infer the likely gender of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**gender_full**](docs/apis/tags/PersonalApi.md#gender_full) | **get** /api2/json/genderFull/{fullName} | Infer the likely gender of a full name, ex. John H. Smith
*PersonalApi* | [**gender_full_batch**](docs/apis/tags/PersonalApi.md#gender_full_batch) | **post** /api2/json/genderFullBatch | Infer the likely gender of up to 100 full names, detecting automatically the cultural context.
*PersonalApi* | [**gender_full_geo**](docs/apis/tags/PersonalApi.md#gender_full_geo) | **get** /api2/json/genderFullGeo/{fullName}/{countryIso2} | Infer the likely gender of a full name, given a local context (ISO2 country code).
*PersonalApi* | [**gender_full_geo_batch**](docs/apis/tags/PersonalApi.md#gender_full_geo_batch) | **post** /api2/json/genderFullGeoBatch | Infer the likely gender of up to 100 full names, with a given cultural context (country ISO2 code).
*PersonalApi* | [**gender_geo**](docs/apis/tags/PersonalApi.md#gender_geo) | **get** /api2/json/genderGeo/{firstName}/{lastName}/{countryIso2} | Infer the likely gender of a name, given a local context (ISO2 country code).
*PersonalApi* | [**gender_geo_batch**](docs/apis/tags/PersonalApi.md#gender_geo_batch) | **post** /api2/json/genderGeoBatch | Infer the likely gender of up to 100 names, each given a local context (ISO2 country code).
*PersonalApi* | [**origin**](docs/apis/tags/PersonalApi.md#origin) | **get** /api2/json/origin/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely country of origin of a personal name. Assumes names as they are in the country of origin. For US, CA, AU, NZ and other melting-pots : use &#x27;diaspora&#x27; instead.
*PersonalApi* | [**origin_batch**](docs/apis/tags/PersonalApi.md#origin_batch) | **post** /api2/json/originBatch | [USES 10 UNITS PER NAME] Infer the likely country of origin of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**parse_name**](docs/apis/tags/PersonalApi.md#parse_name) | **get** /api2/json/parseName/{nameFull} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. 
*PersonalApi* | [**parse_name_batch**](docs/apis/tags/PersonalApi.md#parse_name_batch) | **post** /api2/json/parseNameBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John.
*PersonalApi* | [**parse_name_geo**](docs/apis/tags/PersonalApi.md#parse_name_geo) | **get** /api2/json/parseName/{nameFull}/{countryIso2} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. For better accuracy, provide a geographic context.
*PersonalApi* | [**parse_name_geo_batch**](docs/apis/tags/PersonalApi.md#parse_name_geo_batch) | **post** /api2/json/parseNameGeoBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. Giving a local context improves precision. 
*PersonalApi* | [**religion2**](docs/apis/tags/PersonalApi.md#religion2) | **get** /api2/json/religion/{countryIso2}/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal first/last name. NB: only for INDIA (as of current version).
*PersonalApi* | [**religion_batch**](docs/apis/tags/PersonalApi.md#religion_batch) | **post** /api2/json/religionBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal first/last names. NB: only for India as of currently.
*PersonalApi* | [**religion_full**](docs/apis/tags/PersonalApi.md#religion_full) | **get** /api2/json/religionFull/{countryIso2}/{subDivisionIso31662}/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal full name. NB: only for INDIA (as of current version).
*PersonalApi* | [**religion_full_batch**](docs/apis/tags/PersonalApi.md#religion_full_batch) | **post** /api2/json/religionFullBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal full names. NB: only for India as of currently.
*PersonalApi* | [**subclassification**](docs/apis/tags/PersonalApi.md#subclassification) | **get** /api2/json/subclassification/{countryIso2}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely origin of a name at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#x27;IN&#x27;).
*PersonalApi* | [**subclassification_batch**](docs/apis/tags/PersonalApi.md#subclassification_batch) | **post** /api2/json/subclassificationBatch | [USES 10 UNITS PER NAME] Infer the likely origin of a list of up to 100 names at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#x27;IN&#x27;).
*PersonalApi* | [**subclassification_full**](docs/apis/tags/PersonalApi.md#subclassification_full) | **get** /api2/json/subclassificationFull/{countryIso2}/{fullName} | [USES 10 UNITS PER NAME] Infer the likely origin of a name at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#x27;IN&#x27;).
*PersonalApi* | [**subclassification_full_batch**](docs/apis/tags/PersonalApi.md#subclassification_full_batch) | **post** /api2/json/subclassificationFullBatch | [USES 10 UNITS PER NAME] Infer the likely origin of a list of up to 100 names at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#x27;IN&#x27;).
*PersonalApi* | [**us_race_ethnicity**](docs/apis/tags/PersonalApi.md#us_race_ethnicity) | **get** /api2/json/usRaceEthnicity/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer a US resident&#x27;s likely race/ethnicity according to US Census taxonomy W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**us_race_ethnicity_batch**](docs/apis/tags/PersonalApi.md#us_race_ethnicity_batch) | **post** /api2/json/usRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#x27;s likely race/ethnicity according to US Census taxonomy. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**us_race_ethnicity_zip5**](docs/apis/tags/PersonalApi.md#us_race_ethnicity_zip5) | **get** /api2/json/usRaceEthnicityZIP5/{firstName}/{lastName}/{zip5Code} | [USES 10 UNITS PER NAME] Infer a US resident&#x27;s likely race/ethnicity according to US Census taxonomy, using (optional) ZIP5 code info. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**us_zip_race_ethnicity_batch**](docs/apis/tags/PersonalApi.md#us_zip_race_ethnicity_batch) | **post** /api2/json/usZipRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#x27;s likely race/ethnicity according to US Census taxonomy, with (optional) ZIP code. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*SocialApi* | [**phone_code**](docs/apis/tags/SocialApi.md#phone_code) | **get** /api2/json/phoneCode/{firstName}/{lastName}/{phoneNumber} | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, given a personal name and formatted / unformatted phone number.
*SocialApi* | [**phone_code_batch**](docs/apis/tags/SocialApi.md#phone_code_batch) | **post** /api2/json/phoneCodeBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, detecting automatically the local context given a name and formatted / unformatted phone number.
*SocialApi* | [**phone_code_geo**](docs/apis/tags/SocialApi.md#phone_code_geo) | **get** /api2/json/phoneCodeGeo/{firstName}/{lastName}/{phoneNumber}/{countryIso2} | [USES 11 UNITS PER NAME] Infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).
*SocialApi* | [**phone_code_geo_batch**](docs/apis/tags/SocialApi.md#phone_code_geo_batch) | **post** /api2/json/phoneCodeGeoBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, with a local context (ISO2 country of residence).
*SocialApi* | [**phone_code_geo_feedback_loop**](docs/apis/tags/SocialApi.md#phone_code_geo_feedback_loop) | **get** /api2/json/phoneCodeGeoFeedbackLoop/{firstName}/{lastName}/{phoneNumber}/{phoneNumberE164}/{countryIso2} | [CREDITS 1 UNIT] Feedback loop to better infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).

## Documentation For Models

 - [APIBillingPeriodUsageOut](docs/models/APIBillingPeriodUsageOut.md)
 - [APIClassifierOut](docs/models/APIClassifierOut.md)
 - [APIClassifierTaxonomyOut](docs/models/APIClassifierTaxonomyOut.md)
 - [APIClassifiersStatusOut](docs/models/APIClassifiersStatusOut.md)
 - [APICounterV2Out](docs/models/APICounterV2Out.md)
 - [APIKeyOut](docs/models/APIKeyOut.md)
 - [APIPeriodUsageOut](docs/models/APIPeriodUsageOut.md)
 - [APIPlanSubscriptionOut](docs/models/APIPlanSubscriptionOut.md)
 - [APIServiceOut](docs/models/APIServiceOut.md)
 - [APIServicesOut](docs/models/APIServicesOut.md)
 - [APIUsageAggregatedOut](docs/models/APIUsageAggregatedOut.md)
 - [APIUsageHistoryOut](docs/models/APIUsageHistoryOut.md)
 - [BatchCorridorIn](docs/models/BatchCorridorIn.md)
 - [BatchCorridorOut](docs/models/BatchCorridorOut.md)
 - [BatchFirstLastNameCasteOut](docs/models/BatchFirstLastNameCasteOut.md)
 - [BatchFirstLastNameCastegroupOut](docs/models/BatchFirstLastNameCastegroupOut.md)
 - [BatchFirstLastNameDiasporaedOut](docs/models/BatchFirstLastNameDiasporaedOut.md)
 - [BatchFirstLastNameGenderIn](docs/models/BatchFirstLastNameGenderIn.md)
 - [BatchFirstLastNameGenderedOut](docs/models/BatchFirstLastNameGenderedOut.md)
 - [BatchFirstLastNameGeoIn](docs/models/BatchFirstLastNameGeoIn.md)
 - [BatchFirstLastNameGeoSubclassificationOut](docs/models/BatchFirstLastNameGeoSubclassificationOut.md)
 - [BatchFirstLastNameGeoSubdivisionIn](docs/models/BatchFirstLastNameGeoSubdivisionIn.md)
 - [BatchFirstLastNameGeoZippedIn](docs/models/BatchFirstLastNameGeoZippedIn.md)
 - [BatchFirstLastNameIn](docs/models/BatchFirstLastNameIn.md)
 - [BatchFirstLastNameOriginedOut](docs/models/BatchFirstLastNameOriginedOut.md)
 - [BatchFirstLastNamePhoneCodedOut](docs/models/BatchFirstLastNamePhoneCodedOut.md)
 - [BatchFirstLastNamePhoneNumberGeoIn](docs/models/BatchFirstLastNamePhoneNumberGeoIn.md)
 - [BatchFirstLastNamePhoneNumberIn](docs/models/BatchFirstLastNamePhoneNumberIn.md)
 - [BatchFirstLastNameReligionedOut](docs/models/BatchFirstLastNameReligionedOut.md)
 - [BatchFirstLastNameSubdivisionIn](docs/models/BatchFirstLastNameSubdivisionIn.md)
 - [BatchFirstLastNameUSRaceEthnicityOut](docs/models/BatchFirstLastNameUSRaceEthnicityOut.md)
 - [BatchMatchPersonalFirstLastNameIn](docs/models/BatchMatchPersonalFirstLastNameIn.md)
 - [BatchNameGeoIn](docs/models/BatchNameGeoIn.md)
 - [BatchNameIn](docs/models/BatchNameIn.md)
 - [BatchNameMatchCandidatesOut](docs/models/BatchNameMatchCandidatesOut.md)
 - [BatchNameMatchedOut](docs/models/BatchNameMatchedOut.md)
 - [BatchPersonalNameCastegroupOut](docs/models/BatchPersonalNameCastegroupOut.md)
 - [BatchPersonalNameGenderedOut](docs/models/BatchPersonalNameGenderedOut.md)
 - [BatchPersonalNameGeoIn](docs/models/BatchPersonalNameGeoIn.md)
 - [BatchPersonalNameGeoOut](docs/models/BatchPersonalNameGeoOut.md)
 - [BatchPersonalNameGeoSubclassificationOut](docs/models/BatchPersonalNameGeoSubclassificationOut.md)
 - [BatchPersonalNameGeoSubdivisionIn](docs/models/BatchPersonalNameGeoSubdivisionIn.md)
 - [BatchPersonalNameIn](docs/models/BatchPersonalNameIn.md)
 - [BatchPersonalNameParsedOut](docs/models/BatchPersonalNameParsedOut.md)
 - [BatchPersonalNameReligionedOut](docs/models/BatchPersonalNameReligionedOut.md)
 - [BatchPersonalNameSubdivisionIn](docs/models/BatchPersonalNameSubdivisionIn.md)
 - [BatchProperNounCategorizedOut](docs/models/BatchProperNounCategorizedOut.md)
 - [CorridorIn](docs/models/CorridorIn.md)
 - [CorridorOut](docs/models/CorridorOut.md)
 - [FeedbackLoopOut](docs/models/FeedbackLoopOut.md)
 - [FirstLastNameCasteOut](docs/models/FirstLastNameCasteOut.md)
 - [FirstLastNameCastegroupOut](docs/models/FirstLastNameCastegroupOut.md)
 - [FirstLastNameDiasporaedOut](docs/models/FirstLastNameDiasporaedOut.md)
 - [FirstLastNameGenderIn](docs/models/FirstLastNameGenderIn.md)
 - [FirstLastNameGenderedOut](docs/models/FirstLastNameGenderedOut.md)
 - [FirstLastNameGeoIn](docs/models/FirstLastNameGeoIn.md)
 - [FirstLastNameGeoSubclassificationOut](docs/models/FirstLastNameGeoSubclassificationOut.md)
 - [FirstLastNameGeoSubdivisionIn](docs/models/FirstLastNameGeoSubdivisionIn.md)
 - [FirstLastNameGeoZippedIn](docs/models/FirstLastNameGeoZippedIn.md)
 - [FirstLastNameIn](docs/models/FirstLastNameIn.md)
 - [FirstLastNameOriginedOut](docs/models/FirstLastNameOriginedOut.md)
 - [FirstLastNameOut](docs/models/FirstLastNameOut.md)
 - [FirstLastNamePhoneCodedOut](docs/models/FirstLastNamePhoneCodedOut.md)
 - [FirstLastNamePhoneNumberGeoIn](docs/models/FirstLastNamePhoneNumberGeoIn.md)
 - [FirstLastNamePhoneNumberIn](docs/models/FirstLastNamePhoneNumberIn.md)
 - [FirstLastNameReligionedOut](docs/models/FirstLastNameReligionedOut.md)
 - [FirstLastNameSubdivisionIn](docs/models/FirstLastNameSubdivisionIn.md)
 - [FirstLastNameUSRaceEthnicityOut](docs/models/FirstLastNameUSRaceEthnicityOut.md)
 - [MatchPersonalFirstLastNameIn](docs/models/MatchPersonalFirstLastNameIn.md)
 - [NameGeoIn](docs/models/NameGeoIn.md)
 - [NameIn](docs/models/NameIn.md)
 - [NameMatchCandidateOut](docs/models/NameMatchCandidateOut.md)
 - [NameMatchCandidatesOut](docs/models/NameMatchCandidatesOut.md)
 - [NameMatchedOut](docs/models/NameMatchedOut.md)
 - [PersonalNameCastegroupOut](docs/models/PersonalNameCastegroupOut.md)
 - [PersonalNameGenderedOut](docs/models/PersonalNameGenderedOut.md)
 - [PersonalNameGeoIn](docs/models/PersonalNameGeoIn.md)
 - [PersonalNameGeoOut](docs/models/PersonalNameGeoOut.md)
 - [PersonalNameGeoSubclassificationOut](docs/models/PersonalNameGeoSubclassificationOut.md)
 - [PersonalNameGeoSubdivisionIn](docs/models/PersonalNameGeoSubdivisionIn.md)
 - [PersonalNameIn](docs/models/PersonalNameIn.md)
 - [PersonalNameParsedOut](docs/models/PersonalNameParsedOut.md)
 - [PersonalNameReligionedOut](docs/models/PersonalNameReligionedOut.md)
 - [PersonalNameSubdivisionIn](docs/models/PersonalNameSubdivisionIn.md)
 - [ProperNounCategorizedOut](docs/models/ProperNounCategorizedOut.md)
 - [RegionISO](docs/models/RegionISO.md)
 - [RegionOut](docs/models/RegionOut.md)
 - [ReligionStatOut](docs/models/ReligionStatOut.md)
 - [SoftwareVersionOut](docs/models/SoftwareVersionOut.md)

## Documentation For Authorization

Authentication schemes defined for the API:
<a id="api_key"></a>
### api_key

- **Type**: API key
- **API key parameter name**: X-API-KEY
- **Location**: HTTP header


## Author

contact@namsor.com
contact@namsor.com
contact@namsor.com
contact@namsor.com
contact@namsor.com
contact@namsor.com
contact@namsor.com

## Notes for Large OpenAPI documents
If the OpenAPI document is large, imports in openapi_client.apis and openapi_client.models may fail with a
RecursionError indicating the maximum recursion limit has been exceeded. In that case, there are a couple of solutions:

Solution 1:
Use specific imports for apis and models like:
- `from openapi_client.apis.default_api import DefaultApi`
- `from openapi_client.model.pet import Pet`

Solution 1:
Before importing the package, adjust the maximum recursion limit as shown below:
```
import sys
sys.setrecursionlimit(1500)
import openapi_client
from openapi_client.apis import *
from openapi_client.models import *
```
 readmeEtag: '"f1e8481de55c3dbef80e7be35ac335d84669b148"' readmeLastModified: Wed, 23 Aug 2023 08:02:50 GMT repositoryId: 165632249 description: >- NamSor API v2 Python SDK - classify personal names accurately by gender, country of origin, or ethnicity. created: '2019-01-14T09:22:11Z' updated: '2026-01-16T13:28:18Z' language: Python archived: false stars: 37 watchers: 5 forks: 8 owner: namsor logo: https://avatars.githubusercontent.com/u/6951565?v=4 license: AGPL-3.0 repoEtag: '"7f72a8ab6c2aa7e7ee74f957beba5aedd5a921cf579da27da52fe98057ac6f0b"' repoLastModified: Fri, 16 Jan 2026 13:28:18 GMT foundInMaster: true category: - Server Implementations - Description Validators id: 78fde2a8b60c871d4f8c2b5f7bb08775 - source: openapi3 tags repository: https://github.com/rexagon/opg v3: true repositoryMetadata: base64Readme: >- <p align="center">
    <h3 align="center">opg</h3>
    <p align="center">Rust OpenAPI 3.0 docs generator</p>
    <p align="center">
        <a href="/LICENSE">
            <img alt="GitHub" src="https://img.shields.io/github/license/Rexagon/opg" />
        </a>
        <a href="https://github.com/Rexagon/opg/actions?query=workflow%3Amaster">
            <img alt="GitHub Workflow Status" src="https://img.shields.io/github/workflow/status/Rexagon/opg/master" />
        </a>
        <a href="https://crates.io/crates/opg">
            <img alt="Crates.io Version" src="https://img.shields.io/crates/v/opg" />
        </a>
        <a href="https://docs.rs/opg">
            <img alt="Docs.rs" src="https://docs.rs/opg/badge.svg" />
        </a>
    </p>
</p>

#### Example:
> Or see more [here](https://github.com/Rexagon/opg/tree/master/test_suite/tests)

```rust
use opg::*;
use serde::{Serialize, Deserialize};

#[derive(Serialize, Deserialize, OpgModel)]
#[serde(rename_all = "camelCase")]
#[opg("Simple enum")]
enum SimpleEnum {
    Test,
    Another,
    Yay,
}

#[derive(Serialize, Deserialize, OpgModel)]
#[opg("newtype string", format = "id", example = "abcd0001")]
struct NewType(String);

#[derive(Serialize, Deserialize, OpgModel)]
struct SimpleStruct {
    first_field: i32,
    #[opg("Field description")]
    second: String,
}

#[derive(Serialize, Deserialize, OpgModel)]
#[serde(rename_all = "kebab-case")]
enum ExternallyTaggedEnum {
    Test(String),
    AnotherTest(String, #[opg("Second")] String),
}

#[derive(Serialize, Deserialize, OpgModel)]
#[serde(untagged)]
enum UntaggedEnum {
    First {
        value: NewType,
    },
    #[opg("Variant description")]
    Second {
        #[opg("Inlined struct", inline)]
        another: SimpleStruct,
    },
}

#[derive(Serialize, Deserialize, OpgModel)]
#[serde(tag = "tag", rename_all = "lowercase")]
enum InternallyTaggedEnum {
    First(SimpleStruct),
    Second { field: String },
}

#[derive(Serialize, Deserialize, OpgModel)]
#[serde(tag = "tag", content = "content", rename_all = "lowercase")]
enum AdjacentlyTaggedEnum {
    First(String),
    Second(NewType, NewType),
}

#[derive(Serialize, Deserialize, OpgModel)]
#[serde(rename_all = "camelCase")]
struct TypeChangedStruct {
    #[serde(with = "chrono::naive::serde::ts_milliseconds")]
    #[opg("UTC timestamp in milliseconds", integer, format = "int64")]
    pub timestamp: chrono::NaiveDateTime,
}

#[derive(Serialize, Deserialize, OpgModel)]
struct StructWithComplexObjects {
    #[serde(skip_serializing_if = "Option::is_none")]
    #[opg(optional)]
    super_optional: Option<Option<String>>,
    field: Option<String>,
    boxed: Box<Option<i32>>,
}

#[derive(Serialize, OpgModel)]
struct GenericStructWithRef<'a, T> {
    message: &'a str,
    test: T,
}

#[derive(Serialize, OpgModel)]
struct SuperResponse {
    simple_enum: SimpleEnum,
    #[serde(rename = "new_type")]
    newtype: NewType,
    externally_tagged_enum: ExternallyTaggedEnum,
    untagged_enum: UntaggedEnum,
    internally_tagged_enum: InternallyTaggedEnum,
    adjacently_tagged_enum: AdjacentlyTaggedEnum,
    type_changed_struct: TypeChangedStruct,
    struct_with_complex_objects: StructWithComplexObjects,
}

#[test]
fn print_api() {
    let test = describe_api! {
        info: {
            title: "My super API",
            version: "0.0.0",
        },
        tags: {internal, admin("Super admin methods")},
        servers: {
            "https://my.super.server.com/v1",
        },
        security_schemes: {
            (http "bearerAuth"): {
                scheme: Bearer,
                bearer_format: "JWT",
            },
        },
        paths: {
            ("hello" / "world" / { paramTest: String }): {
                summary: "Some test group of requests",
                description: "Another test description",
                parameters: {
                    (header "x-request-id"): {
                        description: "Test",
                        required: true,
                    },
                },
                GET: {
                    tags: {internal},
                    summary: "Small summary",
                    description: "Small description",
                    deprecated: true,
                    parameters: {
                        (query someParam: u32): {
                            description: "Test",
                        }
                    },
                    200: String,
                    418 ("Optional response description"): String
                },
                POST: {
                    tags: {admin},
                    security: {"bearerAuth"},
                    body: {
                        description: "Some interesting description",
                        schema: GenericStructWithRef<'static, i64>,
                        required: true,
                    },
                    200: SuperResponse,
                    callbacks: {
                        callbackUrl: {
                            ("callbackUrl"): {
                                POST: {
                                    200: std::vec::Vec<String>,
                                }
                            }
                        }
                    }
                }
            }
        }
    };

    println!("{}", serde_yaml::to_string(&test).unwrap());
}
```

<details><summary><b>Result:</b></summary>
<p>

```yaml
---
openapi: 3.0.3
info:
  title: My super API
  version: 0.0.0
tags:
  - name: admin
    description: Super admin methods
  - name: internal
servers:
  - url: "https://my.super.server.com/v1"
paths:
  "/hello/world/{paramTest}":
    summary: Some test group of requests
    description: Another test description
    get:
      tags:
        - internal
      summary: Small summary
      description: Small description
      deprecated: true
      responses:
        200:
          description: OK
          content:
            application/json:
              schema:
                type: string
        418:
          description: Optional response description
          content:
            application/json:
              schema:
                type: string
      parameters:
        - name: someParam
          description: Test
          in: query
          schema:
            type: integer
            format: uint32
    post:
      tags:
        - admin
      security:
        - bearerAuth: []
      requestBody:
        required: true
        description: Some interesting description
        content:
          application/json:
            schema:
              $ref: "#/components/schemas/GenericStructWithRef"
      responses:
        200:
          description: OK
          content:
            application/json:
              schema:
                $ref: "#/components/schemas/SuperResponse"
      callbacks:
        callbackUrl:
          /callbackUrl:
            post:
              responses:
                200:
                  description: OK
                  content:
                    application/json:
                      schema:
                        type: array
                        items:
                          type: string
    parameters:
      - name: paramTest
        in: path
        required: true
        schema:
          type: string
      - name: x-request-id
        description: Test
        in: header
        required: true
        schema:
          type: string
components:
  schemas:
    AdjacentlyTaggedEnum:
      type: object
      properties:
        content:
          oneOf:
            - type: string
            - type: array
              items:
                oneOf:
                  - $ref: "#/components/schemas/NewType"
                  - $ref: "#/components/schemas/NewType"
        tag:
          description: AdjacentlyTaggedEnum type variant
          type: string
          enum:
            - first
            - second
          example: first
      required:
        - tag
        - content
    ExternallyTaggedEnum:
      type: object
      additionalProperties:
        oneOf:
          - type: string
          - type: array
            items:
              oneOf:
                - type: string
                - description: Second
                  type: string
    GenericStructWithRef:
      type: object
      properties:
        message:
          type: string
        test:
          type: integer
          format: int64
      required:
        - message
        - test
    InternallyTaggedEnum:
      oneOf:
        - type: object
          properties:
            first_field:
              type: integer
              format: int32
            second:
              description: Field description
              type: string
            tag:
              description: InternallyTaggedEnum type variant
              type: string
              enum:
                - first
              example: first
          required:
            - first_field
            - second
            - tag
        - type: object
          properties:
            field:
              type: string
            tag:
              description: InternallyTaggedEnum type variant
              type: string
              enum:
                - second
              example: second
          required:
            - field
            - tag
    NewType:
      description: newtype string
      type: string
      format: id
      example: abcd0001
    SimpleEnum:
      description: Simple enum
      type: string
      enum:
        - test
        - another
        - yay
      example: test
    StructWithComplexObjects:
      type: object
      properties:
        boxed:
          nullable: true
          type: integer
          format: int32
        field:
          nullable: true
          type: string
        super_optional:
          nullable: true
          type: string
      required:
        - field
        - boxed
    SuperResponse:
      type: object
      properties:
        adjacently_tagged_enum:
          $ref: "#/components/schemas/AdjacentlyTaggedEnum"
        externally_tagged_enum:
          $ref: "#/components/schemas/ExternallyTaggedEnum"
        internally_tagged_enum:
          $ref: "#/components/schemas/InternallyTaggedEnum"
        new_type:
          $ref: "#/components/schemas/NewType"
        simple_enum:
          $ref: "#/components/schemas/SimpleEnum"
        struct_with_complex_objects:
          $ref: "#/components/schemas/StructWithComplexObjects"
        type_changed_struct:
          $ref: "#/components/schemas/TypeChangedStruct"
        untagged_enum:
          $ref: "#/components/schemas/UntaggedEnum"
      required:
        - simple_enum
        - new_type
        - externally_tagged_enum
        - untagged_enum
        - internally_tagged_enum
        - adjacently_tagged_enum
        - type_changed_struct
        - struct_with_complex_objects
    TypeChangedStruct:
      type: object
      properties:
        timestamp:
          description: UTC timestamp in milliseconds
          type: integer
          format: int64
      required:
        - timestamp
    UntaggedEnum:
      oneOf:
        - type: object
          properties:
            value:
              $ref: "#/components/schemas/NewType"
          required:
            - value
        - description: Variant description
          type: object
          properties:
            another:
              description: Inlined struct
              type: object
              properties:
                first_field:
                  type: integer
                  format: int32
                second:
                  description: Field description
                  type: string
              required:
                - first_field
                - second
          required:
            - another
  securitySchemes:
    bearerAuth:
      type: http
      scheme: bearer
      bearerFormat: JWT
```
</p>
</details>
 readmeEtag: '"93ada7b30d97fd956334f7b2152c7c9a3a52bc79"' readmeLastModified: Wed, 05 Apr 2023 13:48:24 GMT repositoryId: 273586856 description: Rust OpenAPI 3.0 docs generator created: '2020-06-19T21:08:55Z' updated: '2025-06-03T09:55:56Z' language: Rust archived: false stars: 36 watchers: 2 forks: 8 owner: Rexagon logo: https://avatars.githubusercontent.com/u/7278479?v=4 license: Apache-2.0 repoEtag: '"c9a2185cf3173111e0c9b7c6e22e605d6a58d0ff55381cc98e9fa15ca515594a"' repoLastModified: Tue, 03 Jun 2025 09:55:56 GMT foundInMaster: true category: Parsers id: 06fd1d157b70af7d99fc62cfd47a1723 - source: openapi3 tags repository: https://github.com/frameio/rolodex v3: true repositoryMetadata: base64Readme: >- IyBSb2xvZGV4CgpbIVtoZXgucG0gdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9oZXhwbS92L3JvbG9kZXguc3ZnKV0oaHR0cHM6Ly9oZXgucG0vcGFja2FnZXMvcm9sb2RleCkgWyFbQ2lyY2xlQ0ldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL0ZyYW1laW8vcm9sb2RleC5zdmc/c3R5bGU9c3ZnKV0oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvRnJhbWVpby9yb2xvZGV4KQoKUm9sb2RleCBnZW5lcmF0ZXMgZG9jdW1lbnRhdGlvbiBmb3IgeW91ciBQaG9lbml4IEFQSS4KClNpbXBseSBhbm5vdGF0ZSB5b3VyIFBob2VuaXggY29udHJvbGxlciBhY3Rpb24gZnVuY3Rpb25zIHdpdGggYEBkb2NgIG1ldGFkYXRhLCBhbmQgUm9sb2RleCB3aWxsIHR1cm4gdGhlc2UgZGVzY3JpcHRpb25zIGludG8gdmFsaWQgZG9jdW1lbnRhdGlvbiBmb3IgYW55IHBsYXRmb3JtLgoKQ3VycmVudGx5IHN1cHBvcnRzOgotIFtPcGVuQVBJIDMuMF0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKQoKIyMgRGlzY2xhaW1lcgoKUm9sb2RleCBpcyBjdXJyZW50bHkgdW5kZXIgYWN0aXZlIGRldmVsb3BtZW50ISBUaGUgQVBJIGlzIGEgd29yayBpbiBwcm9ncmVzcyBhcyB3ZSBoZWFkIHRvd2FyZHMgdjEuMC4KCiMjIERvY3VtZW50YXRpb24KClNlZSBbaHR0cHM6Ly9oZXhkb2NzLnBtL3JvbG9kZXhdKGh0dHBzOi8vaGV4ZG9jcy5wbS9yb2xvZGV4L1JvbG9kZXguaHRtbCkKCiMjIEluc3RhbGxhdGlvbgoKUm9sb2RleCBpcyBbYXZhaWxhYmxlIGluIEhleF0oaHR0cHM6Ly9oZXgucG0vcGFja2FnZXMvcm9sb2RleCkuIEFkZCBpdCB0byB5b3VyCmRlcHMgaW4gYG1peC5leHNgOgoKYGBgZWxpeGlyCmRlZiBkZXBzIGRvCiAgWwogICAgezpyb2xvZGV4LCAifj4gMC4xMC4wIn0KICBdCmVuZApgYGAK readmeEtag: '"3f157d5eb96f9c9fe916623cc7b67602c4262e0f"' readmeLastModified: Tue, 14 Apr 2020 15:29:38 GMT repositoryId: 164505694 description: 📇API Documentation Generator for Phoenix created: '2019-01-07T22:18:10Z' updated: '2026-02-03T18:15:03Z' language: Elixir archived: false stars: 38 watchers: 19 forks: 2 owner: Frameio logo: https://avatars.githubusercontent.com/u/8882715?v=4 license: MIT repoEtag: '"6194425ed3c5a35ae282827a1907ac8ebdf068094de1628882d85e319139626e"' repoLastModified: Tue, 03 Feb 2026 18:15:03 GMT foundInMaster: true category: Server id: b5c2d746eddf512431874f73431f26ba - source: openapi3 tags repository: https://github.com/bezkoder/spring-boot-swagger-3-example v3: true id: 2dcf393fa42baff255226528de79cac5 repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIDMgYW5kIFNwcmluZyBCb290IGV4YW1wbGUgKHdpdGggT3BlbkFQSSAzKQoKRG9jdW1lbnQgUkVTVCBBUEkgd2l0aCBTd2FnZ2VyIDMgaW4gU3ByaW5nIEJvb3QgZXhhbXBsZSAoZm9sbG93IE9wZW5BUEkgMyBzcGVjaWZpY2F0aW9uKS4gWW91IHdpbGwgYWxzbyBrbm93IHNldmVyYWwgd2F5cyB0byBjb25maWd1cmUgU3dhZ2dlciBBUEkgZGVzY3JpcHRpb24gYW5kIHJlc3BvbnNlLgoKRm9yIG1vcmUgZGV0YWlsLCBwbGVhc2UgdmlzaXQ6Cj4gW1NwcmluZyBCb290IHdpdGggU3dhZ2dlciAzIGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1zd2FnZ2VyLTMvKQoKWW91IGNhbiBhbHNvIGFwcGx5IHRoZSBjb2RlIGVhc2lseSBvbiBmb2xsb3dpbmcgUHJvamVjdHM6Cj4gW1NwcmluZyBCb290IDMgUkVTVCBBUEkgZXhhbXBsZV0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL3NwcmluZy1ib290LTMtcmVzdC1hcGkvKQoKPiBbU3ByaW5nIEJvb3QgUmVzdCBBUEkgd2l0aCBIMl0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL3NwcmluZy1ib290LWpwYS1oMi1leGFtcGxlLykKCj4gW1NwcmluZyBCb290IFJlc3QgQVBJIHdpdGggTXlTUUxdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1qcGEtY3J1ZC1yZXN0LWFwaS8pCgo+IFtTcHJpbmcgQm9vdCBSZXN0IEFQSSB3aXRoIFBvc3RncmVTUUxdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1wb3N0Z3Jlc3FsLWV4YW1wbGUvKQoKPiBbU3ByaW5nIEJvb3QgUmVzdCBBUEkgd2l0aCBTUUwgU2VydmVyXShodHRwczovL3d3dy5iZXprb2Rlci5jb20vc3ByaW5nLWJvb3Qtc3FsLXNlcnZlci8pCgo+IFtTcHJpbmcgQm9vdCBSZXN0IEFQSSB3aXRoIE1vbmdvREJdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1tb25nb2RiLWNydWQvKQoKPiBbU3ByaW5nIEJvb3QgUmVzdCBBUEkgd2l0aCBDYXNzYW5kcmFdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1jYXNzYW5kcmEtY3J1ZC8pCgo+IFtTcHJpbmcgQm9vdCBSZXN0IEFQSSB3aXRoIE9yYWNsZV0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL3NwcmluZy1ib290LWhpYmVybmF0ZS1vcmFjbGUvKQoKRnJvbnQtZW5kIHRoYXQgd29ya3Mgd2VsbCB3aXRoIHRoaXMgQmFjay1lbmQKPiBbQW5ndWxhciA4IENsaWVudF0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2FuZ3VsYXItY3J1ZC1hcHAvKSAvIFtBbmd1bGFyIDEwIENsaWVudF0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2FuZ3VsYXItMTAtY3J1ZC1hcHAvKSAvIFtBbmd1bGFyIDExIENsaWVudF0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2FuZ3VsYXItMTEtY3J1ZC1hcHAvKSAvIFtBbmd1bGFyIDEyIENsaWVudF0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2FuZ3VsYXItMTItY3J1ZC1hcHAvKSAvIFtBbmd1bGFyIDEzIENsaWVudF0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2FuZ3VsYXItMTMtY3J1ZC1leGFtcGxlLykgLyBbQW5ndWxhciAxNCBDbGllbnRdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9hbmd1bGFyLTE0LWNydWQtZXhhbXBsZS8pIC8gW0FuZ3VsYXIgMTUgQ2xpZW50XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vYW5ndWxhci0xNS1jcnVkLWV4YW1wbGUvKQoKPiBbVnVlIDIgQ2xpZW50XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vdnVlLWpzLWNydWQtYXBwLykgLyBbVnVlIDMgQ2xpZW50XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vdnVlLTMtY3J1ZC8pIC8gW1Z1ZXRpZnkgQ2xpZW50XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vdnVldGlmeS1kYXRhLXRhYmxlLWV4YW1wbGUvKQoKPiBbUmVhY3QgQ2xpZW50XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vcmVhY3QtaG9va3MtY3J1ZC1heGlvcy1hcGkvKSAvIFtSZWFjdCBSZWR1eCBDbGllbnRdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9yZWR1eC10b29sa2l0LWNydWQtcmVhY3QtaG9va3MvKQoKTW9yZSBQcmFjdGljZToKPiBbU3ByaW5nIEJvb3QgRmlsZSB1cGxvYWQgZXhhbXBsZSB3aXRoIE11bHRpcGFydCBGaWxlXShodHRwczovL3d3dy5iZXprb2Rlci5jb20vc3ByaW5nLWJvb3QtZmlsZS11cGxvYWQvKQoKPiBbU3ByaW5nIEJvb3QgUGFnaW5hdGlvbiAmIEZpbHRlciBleGFtcGxlIHwgU3ByaW5nIEpQQSwgUGFnZWFibGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1wYWdpbmF0aW9uLWZpbHRlci1qcGEtcGFnZWFibGUvKQoKPiBbU3ByaW5nIERhdGEgSlBBIFNvcnQvT3JkZXIgYnkgbXVsdGlwbGUgQ29sdW1ucyB8IFNwcmluZyBCb290XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vc3ByaW5nLWRhdGEtc29ydC1tdWx0aXBsZS1jb2x1bW5zLykKCj4gW1NwcmluZyBCb290IFJlcG9zaXRvcnkgVW5pdCBUZXN0IHdpdGggQERhdGFKcGFUZXN0XShodHRwczovL3d3dy5iZXprb2Rlci5jb20vc3ByaW5nLWJvb3QtdW5pdC10ZXN0LWpwYS1yZXBvLWRhdGFqcGF0ZXN0LykKCj4gW0RlcGxveSBTcHJpbmcgQm9vdCBBcHAgb24gQVdTIOKAkyBFbGFzdGljIEJlYW5zdGFsa10oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2RlcGxveS1zcHJpbmctYm9vdC1hd3MtZWIvKQoKRXhjZXB0aW9uIEhhbmRsaW5nOgo+IFtTcHJpbmcgQm9vdCBAQ29udHJvbGxlckFkdmljZSAmIEBFeGNlcHRpb25IYW5kbGVyIGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1jb250cm9sbGVyYWR2aWNlLWV4Y2VwdGlvbmhhbmRsZXIvKQoKPiBbQFJlc3RDb250cm9sbGVyQWR2aWNlIGV4YW1wbGUgaW4gU3ByaW5nIEJvb3RdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1yZXN0Y29udHJvbGxlcmFkdmljZS8pCgpTZWN1cml0eToKPiBbU3ByaW5nIEJvb3QgKyBTcHJpbmcgU2VjdXJpdHkgSldUIEF1dGhlbnRpY2F0aW9uICYgQXV0aG9yaXphdGlvbl0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL3NwcmluZy1ib290LWp3dC1hdXRoZW50aWNhdGlvbi8pCgpGdWxsc3RhY2s6Cj4gW1Z1ZSArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC12dWUtanMtY3J1ZC1leGFtcGxlLykKCj4gW0FuZ3VsYXIgOCArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9hbmd1bGFyLXNwcmluZy1ib290LWNydWQvKQoKPiBbQW5ndWxhciAxMCArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9hbmd1bGFyLTEwLXNwcmluZy1ib290LWNydWQvKQoKPiBbQW5ndWxhciAxMSArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9hbmd1bGFyLTExLXNwcmluZy1ib290LWNydWQvKQoKPiBbQW5ndWxhciAxMiArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9hbmd1bGFyLTEyLXNwcmluZy1ib290LWNydWQvKQoKPiBbQW5ndWxhciAxMyArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1hbmd1bGFyLTEzLWNydWQvKQoKPiBbQW5ndWxhciAxNCArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1hbmd1bGFyLTE0LWNydWQvKQoKPiBbQW5ndWxhciAxNSArIFNwcmluZyBCb290IGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1hbmd1bGFyLTE1LWNydWQvKQoKPiBbUmVhY3QgKyBTcHJpbmcgQm9vdCArIE15U1FMIGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9yZWFjdC1zcHJpbmctYm9vdC1jcnVkLykKCj4gW1JlYWN0ICsgU3ByaW5nIEJvb3QgKyBQb3N0Z3JlU1FMIGV4YW1wbGVdKGh0dHBzOi8vd3d3LmJlemtvZGVyLmNvbS9zcHJpbmctYm9vdC1yZWFjdC1wb3N0Z3Jlc3FsLykKClJ1biBib3RoIEJhY2stZW5kICYgRnJvbnQtZW5kIGluIG9uZSBwbGFjZToKPiBbSW50ZWdyYXRlIEFuZ3VsYXIgd2l0aCBTcHJpbmcgQm9vdCBSZXN0IEFQSV0oaHR0cHM6Ly93d3cuYmV6a29kZXIuY29tL2ludGVncmF0ZS1hbmd1bGFyLXNwcmluZy1ib290LykKCj4gW0ludGVncmF0ZSBSZWFjdC5qcyB3aXRoIFNwcmluZyBCb290IFJlc3QgQVBJXShodHRwczovL3d3dy5iZXprb2Rlci5jb20vaW50ZWdyYXRlLXJlYWN0anMtc3ByaW5nLWJvb3QvKQoKPiBbSW50ZWdyYXRlIFZ1ZS5qcyB3aXRoIFNwcmluZyBCb290IFJlc3QgQVBJXShodHRwczovL3d3dy5iZXprb2Rlci5jb20vaW50ZWdyYXRlLXZ1ZS1zcHJpbmctYm9vdC8pCgojIyBSdW4gU3ByaW5nIEJvb3QgYXBwbGljYXRpb24KYGBgCm12biBzcHJpbmctYm9vdDpydW4KYGBgCgo= readmeEtag: '"b53f18bde83eb6f6547ef70b47e6267d2a12e859"' readmeLastModified: Wed, 15 Mar 2023 03:35:06 GMT repositoryId: 614175956 description: >- Spring Boot and Swagger 3 example - configuration for API description / response example - Swagger annotations with OpenAPI 3 created: '2023-03-15T03:33:57Z' updated: '2025-10-10T10:56:48Z' language: Java archived: false stars: 36 watchers: 1 forks: 38 owner: bezkoder logo: https://avatars.githubusercontent.com/u/52996966?v=4 repoEtag: '"c6206eb3bb391fe040672e4fa2c4f111c6d0d158c969ad04afb158f138030a91"' repoLastModified: Fri, 10 Oct 2025 10:56:48 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cxxxr/apispec v3: true repositoryMetadata: base64Readme: >- IyBhcGlzcGVjCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9mdWthbWFjaGkvYXBpc3BlYy5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9mdWthbWFjaGkvYXBpc3BlYykKWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9mdWthbWFjaGkvYXBpc3BlYy9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vY292ZXJhbGxzLmlvL3IvZnVrYW1hY2hpL2FwaXNwZWMpCgpBIENvbW1vbiBMaXNwIGxpYnJhcnkgZm9yIGhhbmRsaW5nIFdlYiBBUEkgc3BlY2lmaWNhdGlvbnMuIFRoaXMgYWxsb3dzIHRvIHZhbGlkYXRlIGFuZCBwYXJzZSBIVFRQIHJlcXVlc3QgaGVhZGVycywgcGFyYW1ldGVycyBhbmQgYm9kaWVzIHdpdGggT3BlbkFQSTMgc3BlY2lmaWNhdGlvbi4KCiMjIFdhcm5pbmcKClRoaXMgc29mdHdhcmUgaXMgc3RpbGwgQUxQSEEgcXVhbGl0eS4gVGhlIEFQSXMgd2lsbCBiZSBsaWtlbHkgdG8gY2hhbmdlLgoKIyMgUHJlcmVxdWlzaXRlCgoqIFtsaWJ5YW1sXShodHRwOi8vcHl5YW1sLm9yZy93aWtpL0xpYllBTUwpIGZvciBsb2FkaW5nIE9wZW5BUEkgc3BlYyBmaWxlcy4KCiMjIFVzYWdlCgojIyMgTG9hZGluZyBzcGVjaWZpY2F0aW9uIGZpbGUKCmBgYGNvbW1vbi1saXNwCihkZWZ2YXIgKnNwZWMqCiAgKGFwaXNwZWM6bG9hZC1mcm9tLWZpbGUgI1AiZG9jcy9hcGkueWFtbCIpKQoKKGFwaXNwZWM6c3BlYy12ZXJzaW9uICpzcGVjKikKOz0+ICIzLjAuMiIKYGBgCgojIyMgR2V0dGluZyB0aGUgb3BlcmF0aW9uCgpgYGBjb21tb24tbGlzcAooZGVmdmFyICpyb3V0ZXIqIChhcGlzcGVjOnNwZWMtcm91dGVyICpzcGVjKikpCgooYXBpc3BlYzpmaW5kLXJvdXRlICpyb3V0ZXIqIDpHRVQgIi91c2Vycy8xMiIpCjs9PiAjPEFQSVNQRUMvQ0xBU1NFUy9PUEVSQVRJT046T1BFUkFUSU9OIHsxMDAzRERCMDczfT4KYGBgCgojIyMgUGFyc2luZyBhbmQgVmFsaWRhdGluZyBIVFRQIHJlcXVlc3RzCgpgYGBjb21tb24tbGlzcAooaW1wb3J0ICcobGFjay5yZXF1ZXN0OnJlcXVlc3QtcXVlcnktcGFyYW1ldGVycwogICAgICAgICAgbGFjay5yZXF1ZXN0OnJlcXVlc3QtYm9keS1wYXJhbWV0ZXJzCiAgICAgICAgICBsYWNrLnJlcXVlc3Q6cmVxdWVzdC1jb29raWVzCiAgICAgICAgICBhcGlzcGVjOnJlcXVlc3QtcGF0aC1wYXJhbWV0ZXJzKSkKCjs7IENsYWNrIGFwcGxpY2F0aW9uCihkZWZ2YXIgKmFwcCoKICAobGFtYmRhIChlbnYpCiAgICAobXVsdGlwbGUtdmFsdWUtYmluZCAob3BlcmF0aW9uIHBhdGgtcGFyYW1ldGVycykKICAgICAgICAoYXBpc3BlYzpmaW5kLXJvdXRlIChzcGVjLXJvdXRlciAqc3BlYyopCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZ2V0ZiBlbnYgOnJlcXVlc3QtbWV0aG9kKQogICAgICAgICAgICAgICAgICAgICAgICAgICAgKGdldGYgZW52IDpwYXRoLWluZm8pKQogICAgICA7OyBHZXR0aW5nIExhY2suUmVxdWVzdAogICAgICAobGV0ICgocmVxdWVzdCAoYXBpc3BlYzp2YWxpZGF0ZS1yZXF1ZXN0IG9wZXJhdGlvbiBlbnYKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICA6cGF0aC1wYXJhbWV0ZXJzIHBhdGgtcGFyYW1ldGVycykpKQogICAgICAgIDs7IFdyaXRlIHRoZSBtYWluIGFwcGxpY2F0aW9uIGhlcmUuCgogICAgICAgIDs7IEFjY2Vzc29ycyBmb3IgZ2V0dGluZyBlYWNoIHBhcmFtZXRlcnMuCiAgICAgICAgKHJlcXVlc3QtcXVlcnktcGFyYW1ldGVycyByZXF1ZXN0KSAgOz0+IFF1ZXJ5IHBhcmFtZXRlcnMgKGFsaXN0KQogICAgICAgIChyZXF1ZXN0LWJvZHktcGFyYW1ldGVycyByZXF1ZXN0KSAgIDs9PiBCb2R5IHBhcmFtZXRlcnMgKGFsaXN0KQogICAgICAgIChyZXF1ZXN0LXBhdGgtcGFyYW1ldGVycyByZXF1ZXN0KSAgIDs9PiBQYXRoIHBhcmFtZXRlcnMgKGFsaXN0KQogICAgICAgIChyZXF1ZXN0LWNvb2tpZXMpICAgICAgICAgICAgICAgICAgIDs9PiBDb29raWUgcGFyYW1ldGVycyAoYWxpc3QpCgogICAgICAgICkpKSkKCjs7IFN0YXJ0IHRoZSBzZXJ2ZXIKKGNsYWNrOmNsYWNrdXAgKmFwcCopCmBgYAoKIyMjIFZhbGlkYXRpbmcgYW5kIEVuY29kaW5nIEhUVFAgcmVzcG9uc2VzCgpgYGBjb21tb24tbGlzcAooaW1wb3J0ICdsYWNrLnJlc3BvbnNlOm1ha2UtcmVzcG9uc2UpCgooYXBpc3BlYzp2YWxpZGF0ZS1yZXNwb25zZSBvcGVyYXRpb24KICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1ha2UtcmVzcG9uc2UgMjAwCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICcoOmNvbnRlbnQtdHlwZSAiYXBwbGljYXRpb24vanNvbiIpCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICcoKCJpZCIgLiAzKQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICgibmFtZSIgLiAi5Yid6Z+z44Of44KvIikKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAoImlzX2FkbWluIiAuIG5pbCkpKSkKOz0+ICgyMDAgKDpDT05URU5ULVRZUEUgImFwcGxpY2F0aW9uL2pzb24iKSAoIntcImlkXCI6MyxcIm5hbWVcIjpcIuWInemfs+ODn+OCr1wiLFwiaXNfYWRtaW5cIjpmYWxzZX0iKSkKYGBgCgojIyMgQ3VzdG9tIEVuY29kZXIgZm9yIHN0YW5kYXJkIG9iamVjdHMKCmBgYGNvbW1vbi1saXNwCihpbXBvcnQgJ2xhY2sucmVzcG9uc2U6bWFrZS1yZXNwb25zZSkKCjs7IEN1c3RvbSBjbGFzcwooZGVmY2xhc3MgdXNlciAoKQogICgoaWQgOmluaXRhcmcgOmlkKQogICAobmFtZSA6aW5pdGFyZyA6bmFtZSkKICAgKGlzLWFkbWluIDppbml0YXJnIDppcy1hZG1pbikpKQoKOzsgRGVmaW5lIEFQSVNQRUM6RU5DT0RFLU9CSkVDVCBmb3IgdGhlIGNsYXNzCihkZWZtZXRob2QgYXBpc3BlYzplbmNvZGUtb2JqZWN0ICgodXNlciB1c2VyKSkKICBgKCgiaWQiIC4gLChzbG90LXZhbHVlIHVzZXIgJ2lkKSkKICAgICgibmFtZSIgLiAsKHNsb3QtdmFsdWUgdXNlciAnbmFtZSkpCiAgICAoImlzX2FkbWluIiAuICwoc2xvdC12YWx1ZSB1c2VyICdpcy1hZG1pbikpKSkKCihkZWZ2YXIgKnl1a2FyaSoKICAobWFrZS1pbnN0YW5jZSAndXNlcgogICAgICAgICAgICAgICAgIDppZCAxNAogICAgICAgICAgICAgICAgIDpuYW1lICLntZDmnIjjgobjgYvjgooiCiAgICAgICAgICAgICAgICAgOmlzLWFkbWluIG5pbCkpCgooYXBpc3BlYzp2YWxpZGF0ZS1yZXNwb25zZSBvcGVyYXRpb24KICAgICAgICAgICAgICAgICAgICAgICAgICAgKG1ha2UtcmVzcG9uc2UgMjAwCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICcoOmNvbnRlbnQtdHlwZSAiYXBwbGljYXRpb24vanNvbiIpCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICp5dWthcmkqKSkKOz0+ICgyMDAgKDpDT05URU5ULVRZUEUgImFwcGxpY2F0aW9uL2pzb24iKSAoIntcImlkXCI6MTQsXCJuYW1lXCI6XCLntZDmnIjjgobjgYvjgopcIixcImlzX2FkbWluXCI6ZmFsc2V9IikpCmBgYAoKIyMgRXhhbXBsZXMKClNlZSBbZXhhbXBsZXMvXShleGFtcGxlcy8pLgoKIyMgU2VlIEFsc28KCiogW09wZW5BUEkgU3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24pCiogW0xhY2tdKGh0dHBzOi8vZ2l0aHViLmNvbS9mdWthbWFjaGkvbGFjaykKCiMjIEF1dGhvcgoKKiBFaXRhcm8gRnVrYW1hY2hpIChlLmFycm93c0BnbWFpbC5jb20pCgojIyBDb3B5cmlnaHQKCkNvcHlyaWdodCAoYykgMjAxOSBFaXRhcm8gRnVrYW1hY2hpIChlLmFycm93c0BnbWFpbC5jb20pCgojIyBMaWNlbnNlCgpMaWNlbnNlZCB1bmRlciB0aGUgQlNEIDMtQ2xhdXNlIExpY2Vuc2UuCg== readmeEtag: '"6d45557234331003c7fd2f1c0e77d3f996e3e5c8"' readmeLastModified: Wed, 17 Jan 2024 07:44:28 GMT repositoryId: 200196107 description: A Common Lisp library for handling Web API requests and responses. created: '2019-08-02T08:23:34Z' updated: '2025-12-21T20:20:57Z' language: Common Lisp archived: false stars: 36 watchers: 4 forks: 8 owner: cxxxr logo: https://avatars.githubusercontent.com/u/13656378?v=4 repoEtag: '"ca18c1701592cfc8fc2cc162bd0625674248f355c0782811b3375aa28d7e1490"' repoLastModified: Sun, 21 Dec 2025 20:20:57 GMT foundInMaster: true category: Data Validators id: 6c1ee68ee37441a94526d023a6d888e6 - source: openapi3 tags repository: https://github.com/dolmen-go/openapi-preprocessor v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXByZXByb2Nlc3NvcgoKYG9wZW5hcGktcHJlcHJvY2Vzc29yYCBpcyBhbiBwcm9jZXNzaW5nIHRvb2wgdGhhdCBnaXZlcyBmbGV4aWJpbGl0eSB0byBBUEkgZG9jdW1lbnRhdGlvbiBhdXRob3JzIGZvciB3cml0aW5nIE9wZW5BUEkgMi4wLzMueCBzcGVjaWZpY2F0aW9ucy4KClshW1RyYXZpcy1DSV0oaHR0cHM6Ly9hcGkudHJhdmlzLWNpLm9yZy9kb2xtZW4tZ28vb3BlbmFwaS1wcmVwcm9jZXNzb3Iuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvZG9sbWVuLWdvL29wZW5hcGktcHJlcHJvY2Vzc29yKQpbIVtDb2RlY292XShodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvZG9sbWVuLWdvL29wZW5hcGktcHJlcHJvY2Vzc29yL21hc3Rlci5zdmcpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvZG9sbWVuLWdvL29wZW5hcGktcHJlcHJvY2Vzc29yL2JyYW5jaC9tYXN0ZXIpClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS9kb2xtZW4tZ28vb3BlbmFwaS1wcmVwcm9jZXNzb3IpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20vZG9sbWVuLWdvL29wZW5hcGktcHJlcHJvY2Vzc29yKQoKIyMgVXNlcyBDYXNlcwoKLSBBdXRob3IgeW91ciBPcGVuQVBJIHNwZWMgaW4gWUFNTCBidXQgcHVibGlzaCBhcyBKU09OLgotIFNwbGl0IHlvdXIgT3BlbkFQSSBzcGVjIHNvdXJjZSBpbiBtdWx0aXBsZSBmaWxlcyBmb3IgYXV0aG9yaW5nLCBidXQgcHVibGlzaCBhIHNpbmdsZSBmaWxlLgotIEJ1aWxkIG11bHRpcGxlIHNwZWNzIGZyb20gc2hhcmVkIHBhcnRzLgotIE1lcmdlIHNwZWMgZ2VuZXJhdGVkIGZyb20gc291cmNlIGNvZGUgd2l0aCB5b3VyIGFkZGl0aW9uYWwgY29udGVudCBjcmVhdGVkIGJ5IGhhbmQuCi0gVXNlIGFkdmFuY2VkIGlubGluaW5nIChgJGlubGluZWAsIGAkbWVyZ2VgKSB0byByZW1vdmUgZHVwbGljYXRpb24gKHNvdXJjZSBvZiBpbmNvbnNpc3RlbmNpZXMpLgotIFVzZSBhZHZhbmNlZCBpbmxpbmluZyAoYCRpbmxpbmVgLCBgJG1lcmdlYCkgdG8gcHJvZHVjZSBjb21wbGV4IHNjaGVtYXMgdGhhdCBzaGFyZSBzdWJzZXQgb2YgcHJvcGVydGllcy4KLSBEZXJpdmF0ZSBhIHNwZWMgdG8gYnVpbGQgYSBuZXcgc3BlYyB3aXRoIGFsdGVyZWQgc2VydmVycyBzZXR0aW5ncyBmb3IgbG9jYWxob3N0L3N0YWdpbmcvcHJlcHJvZCBlbnZpcm9ubWVudHMuCi0gRGVyaXZhdGUgYSBzcGVjIGZvciBpbnRlcm9wZXJhYmlsaXR5IHdpdGggcHJvY2Vzc2luZyB0b29scyAoZXg6IGFwcGx5IGEgcGF0Y2ggZm9yIGRvd25ncmFkaW5nIHRoZSBPcGVuQVBJCiAgdmVyc2lvbiBmb3IgYSBjb2RlIGdlbmVyYXRpb24gdG9vbCB0aGF0IGRvZXNuJ3Qgc3VwcG9ydCB0aGUgbGF0ZXN0IHN0YW5kYXJkKS4KLSBEZXJpdmF0ZSBhIHNwZWMgZnJvbSBhIHBhcnRuZXIgdG8gZml4IGludGVyb3BlcmFiaWxpdHkgaXNzdWVzOiBlbmFibGUgdGhlIG1haW50ZW5hbmNlIG9mIGEgc2V0IG9mIHBhdGNoZXMKICB0byBhcHBseSB0byB0aGUgcHVibGljIHNwZWMgcmVsZWFzZXMgYmVmb3JlIHVzaW5nIHRoZW0gaW50ZXJuYWxseS4KLSAqU3VibWl0IHlvdXJzLi4uKgoKIyMgRmVhdHVyZXMKCi0gRXZlcnkgdmFsaWQgT3BlbkFQSSAyLjAvMy54IHNwZWNpZmljYXRpb24gaXMgYSB2YWxpZCBpbnB1dCAoc28geW91IGNhbiBlYXNpbHkgc3RhcnQgcmVmYWN0b3JpbmcgZ3JhZHVhbGx5IGZyb20gYW4gZXhpc3Rpbmcgc3BlYykKLSBBbGxvd3MgdG8gYnVpbGQgYSBzcGVjIGZyb20gbXVsdGlwbGUgZmlsZXM7IHByb2R1Y2VzIGEgc2luZ2xlIG91dHB1dCBmaWxlCi0gWUFNTCBvciBKU09OIGlucHV0Ci0gUHJvZHVjZXMgYW4gT3BlbkFQSSB3aXRoIG1heGltdW0gY29tcGF0aWJpbGl0eSB3aXRoIGNvbnN1bW1pbmcgdG9vbHM6CiAgLSBzaW1wbGlmaWVzIGNvbXBsZXggcGFydHMgb2YgdGhlIHNwZWMgbm90IHN1cHBvcnRlZCBieSBhbGwgdG9vbHMKICAtIEpTT04gb3V0cHV0Ci0gQWRkcyBhIGZldyBrZXl3b3JkcyAoYCRpbmxpbmVgLCBgJG1lcmdlYCkgdGhhdCBhbGxvdyB0byBhdm9pZCBkdXBsaWNhdGlvbiBvZiBjb250ZW50IGFuZCBlYXNlIHRoZSB3cml0aW5nIG9mIGNvbnNpc3RlbnQgZG9jdW1lbnRhdGlvbgotIFJlbW92ZXMgdW51c2VkIGdsb2JhbCBzY2hlbWFzICh1bmRlciBgL2NvbXBvbmVudHMvc2NoZW1hc2ApLCBwYXJhbWV0ZXJzICh1bmRlciBgL2NvbXBvbmVudHMvcGFyYW1ldGVyc2ApIGFuZCByZXNwb25zZXMgKHVuZGVyIGAvY29tcG9uZW50cy9yZXNwb25zZXNgKS4gVGhpcyByZWR1Y2VzIHJpc2sgb2YgbGVha2luZyB3b3JrIGluIHByb2dyZXNzIG9yIGludGVybmFsIGRldGFpbHMuCgojIyBJbnN0YWxsCgojIyMgSW5zdGFsbCBmcm9tIHNvdXJjZQoKQSBbR28gMS4yMysgZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnRdKGh0dHBzOi8vZ28uZGV2L2RvYy9pbnN0YWxsI2luc3RhbGwpIGlzIHJlcXVpcmVkLgoKQnVpbGQgYG9wZW5hcGktcHJlcHJvY2Vzc29yYCBiaW5hcnkgYW5kIGluc3RhbGwgaW4gYCRHT1BBVEgvYmluYDoKCiAgICAkIG1ha2UgaW5zdGFsbAoKIyMgVXNhZ2UKCiAgICBvcGVuYXBpLXByZXByb2Nlc3NvciBbPG9wdGlvbj4uLi5dIDxmaWxlPgoKIyMgS2V5d29yZHMKCiMjIyBgJHJlZmAKCiAgICB7ICIkcmVmIjogIjxmaWxlPiIgfQogICAgeyAiJHJlZiI6ICI8ZmlsZT4jPHBvaW50ZXI+IiB9CiAgICB7ICIkcmVmIjogIiM8cG9pbnRlcj4iIH0KCmAkcmVmYCBpcyBsaWtlIGluIE9wZW5BUEksIGJ1dCBpdCBjYW4gcmVmZXJlbmNlIGNvbnRlbnQgaW4gZXh0ZXJuYWwgZmlsZXMgdXNpbmcgcmVsYXRpdmUgVVJMcyBhcyB3ZWxsIGFzIGludHJhLWRvY3VtZW50LiBUaGUgcmVmZXJlbmNlZCBwYXJ0IG9mIHRoZSBwb2ludGVkIGRvY3VtZW50IGlzIGluamVjdGVkIGludG8gdGhlIG91dHB1dCBkb2N1bWVudC4KClJlc3RyaWN0aW9uczoKLSBKU09OIHBvaW50ZXIgbG9jYXRpb24gaW4gdGhlIG91dHB1dCBkb2N1bWVudCB3aWxsIGJlIHRoZSBzYW1lIGxvY2F0aW9uIGFzIGluIHRoZSByZWYgbGluay4gRXhhbXBsZTogYHsiJHJlZiI6ICJleHRlcm5hbC55bWwjL2NvbXBvbmVudHMvcGFyYW1ldGVycy9JZCJ9YCB3aWxsIGltcG9ydCB0aGUgY29udGVudCB0byBgL2NvbXBvbmVudHMvcGFyYW1ldGVycy9JZGAuIFRoaXMgaW1wbGllcyB0aGF0IHBhcnRpYWwgZmlsZXMgc2hvdWxkIGhhdmUgdGhlIHNhbWUgbGF5b3V0IGFzIGEgZnVsbCBzcGVjICh0aGlzIGlzIGEgZmVhdHVyZSBhcyBpdCBlbmZvcmNlcyByZWFkYWJpbGl0eSBvZiBwYXJ0aWFscykuCi0gb3RoZXIgcHJvcGVydGllcyBhbG9uZyBgJHJlZmAgYXJlIG5vdCBhbGxvd2VkIGFzIHRoZSBzZW1hbnRpY3MgaW4gSlNPTiBTY2hlbWEgYW5kIFN3YWdnZXIvT3BlbkFQSSBoYXMgZXZvbHZlZCBhbmQgdGhlIHN1cHBvcnQgaW4gY29uc3VtaW5nIHRvb2xzIG1heSB2YXJ5LiBVc2UgYCRtZXJnZWAgaW5zdGVhZCB0aGF0IGhhcyBhIHN0cmljdCBiZWhhdmlvdXIgaW4gdGhpcyB0b29sLgoKIyMjIGAkaW5saW5lYAoKICAgIHsgIiRpbmxpbmUiOiAiPGZpbGU+Izxwb2ludGVyPiJ9CgogICAgewogICAgICAgICIkaW5saW5lIjogIjxmaWxlPiM8cG9pbnRlcj4iLAogICAgICAgICJwb2ludGVyMSI6IDx2YWx1ZT4sIC8vIE92ZXJyaWRlcyB2YWx1ZSBhdCA8ZmlsZT4jPHBvaW50ZXI+L3BvaW50ZXIxCiAgICAgICAgInBvaW50ZXIyL3NsYXNoIjogPHZhbHVlPiAvLyBPdmVycmlkZXMgdmFsdWUgZGVlcGx5IGF0IDxmaWxlPiM8cG9pbnRlcj4vcG9pbnRlci9zbGFzaAogICAgfQoKYCRpbmxpbmVgIGlzIGFuIE9wZW5BUEkgZXh0ZW5zaW9uIGFsbG93aW5nIHRvIGluamVjdCBhIGNvcHkgb2YgYW5vdGhlciBwYXJ0IG9mIGEgZG9jdW1lbnQgaW4gcGxhY2UuIEtleXMgYWxvbmcgdGhlIGAkaW5saW5lYCBrZXl3b3JkIGFyZSBKU09OIHBvaW50ZXJzICh3aXRoIHRoZSBsZWFkaW5nIGAvYCByZW1vdmVkKSBhbGxvd2luZyB0byBvdmVycmlkZSBzb21lIHBhcnRzIG9mIHRoZSBpbmxpbmVkIGNvbnRlbnQuCgpJZiB0aGUgdGFyZ2V0IG9mIGAkaW5saW5lYCBpcyBhIGAkcmVmYCBhbmQgYCRpbmxpbmVgIGhhcyBvdmVycmlkZXMsIHRoZSBsaW5rIGlzIGRlcmVmZXJlbmNlZCByZWN1cnNpdmVseSBiZWZvcmUgaW5saW5pbmcuCgpOb3RlOiBkZWVwIGlubGluaW5nIChpbmxpbmluZyBhIG5vZGUgd2hpY2ggaXRzZWxmIHVzZSBgJGlubGluZWAgaW4gaXRzIHRyZWUpIG1pZ2h0IHdvcmssIGJ1dCB3aWxsIHByb2JhYmx5IG5vdCAoc2VlIFtpc3N1ZSAjNl0oaHR0cHM6Ly9naXRodWIuY29tL2RvbG1lbi1nby9vcGVuYXBpLXByZXByb2Nlc3Nvci9pc3N1ZXMvNikgYXMgYW4gZXhhbXBsZSkuIFVzZSBpbnN0ZWFkIGAkbWVyZ2VgIHdoaWNoIHN1cHBvcnRzIGl0LgoKIyMjIGAkbWVyZ2VgCgogICAgewogICAgICAgICIkbWVyZ2UiOiAiPGZpbGU+Izxwb2ludGVyPiIsCiAgICAgICAgImtleSI6IDx2YWx1ZT4sCiAgICAgICAgImtleS9zbGFzaCI6IDx2YWx1ZT4gLy8gT3ZlcnJpZGVzIHZhbHVlIGF0IDxmaWxlPiM8cG9pbnRlcj4va2V5fjFzbGFzaAogICAgfQoKICAgIHsKICAgICAgICAiJG1lcmdlIjogWwogICAgICAgICAgICAiPGZpbGUxPiM8cG9pbnRlcjE+IiwKICAgICAgICAgICAgIjxmaWxlMj4jPHBvaW50ZXIyPiIgLy8gT3ZlcnJpZGVzIGtleXMgZnJvbSA8ZmlsZTE+Izxwb2ludGVyMT4KICAgICAgICBdCiAgICAgICAgImtleSI6IDx2YWx1ZT4sCiAgICAgICAgImtleS9zbGFzaCI6IDx2YWx1ZT4gLy8gT3ZlcnJpZGVzIHZhbHVlIGF0IDxmaWxlMj4jPHBvaW50ZXIyPi9rZXl+MXNsYXNoCiAgICB9CgoKYCRtZXJnZWAgaXMgYW4gT3BlbkFQSSBleHRlbnNpb24gYWxsb3dpbmcgdG8gY29weSBhIG5vZGUsIG92ZXJyaWRpbmcgc29tZSBrZXlzLiBUaGlzIGlzIGEga2luZCBvZiBpbmxpbmVkICpgJHJlZmAgd2l0aCBrZXlzIG92ZXJyaWRlcyouCgojIyBFeGFtcGxlcwoKU2VlIHRoZSBbdGVzdHN1aXRlXShodHRwczovL2dpdGh1Yi5jb20vZG9sbWVuLWdvL29wZW5hcGktcHJlcHJvY2Vzc29yL3RyZWUvbWFzdGVyL3Rlc3RkYXRhKS4KClJ1bm5pbmcgYSBiYXNpYyBleGFtcGxlOgoKICAgICQgbWFrZQogICAgJCAuL29wZW5hcGktcHJlcHJvY2Vzc29yIHRlc3RkYXRhLzEwLXJlZi1leHQvaW5wdXQueW1sCgojIyBMaWNlbnNlCgpDb3B5cmlnaHQgMjAxOC0yMDIyIE9saXZpZXIgTWVuZ3XDqQoKTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7CnlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0CgogICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKClVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuClNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCg== readmeEtag: '"785ab3381e81ad7814c3313f61b1a48abd845ff8"' readmeLastModified: Tue, 21 Oct 2025 17:01:43 GMT repositoryId: 141041148 description: An authoring tool for OpenAPI specifications created: '2018-07-15T16:16:48Z' updated: '2025-10-21T17:02:05Z' language: Go archived: false stars: 38 watchers: 3 forks: 5 owner: dolmen-go logo: https://avatars.githubusercontent.com/u/22580372?v=4 license: Apache-2.0 repoEtag: '"78a861fdc5c2c1ba013bd1aa6e0e8d4e7f9054bfb99725c127295ee6d3240d4e"' repoLastModified: Tue, 21 Oct 2025 17:02:05 GMT foundInMaster: true category: Parsers id: 084230f40fdffa9865facd7e88ca7499 - source: openapi3 tags repository: https://github.com/data-fair/data-fair v3: true repositoryMetadata: base64Readme: >- IyA8aW1nIGFsdD0iRGF0YSBGQUlSIGxvZ28iIHNyYz0iaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L2doL2RhdGEtZmFpci9kYXRhLWZhaXJAbWFzdGVyL3B1YmxpYy9hc3NldHMvbG9nby5zdmciIHdpZHRoPSI0MCI+IERhdGEgRkFJUgoKKkZpbmRhYmxlLCBBY2Nlc3NpYmxlLCBJbnRlcm9wZXJhYmxlIGFuZCBSZXVzYWJsZSBEYXRhKgoKW1Zpc2l0IGRvY3VtZW50YXRpb24gd2Vic2l0ZV0oaHR0cHM6Ly9kYXRhLWZhaXIuZ2l0aHViLmlvLzMvKQoKIVtdKGRvYy9zdGF0aWMvZGF0YS1mYWlyLmdpZikKCiMjIFNwb25zb3JzCgp8IHwgQ2xpY2sgW2hlcmUgdG8gc3VwcG9ydCB0aGUgZGV2ZWxvcG1lbnQgb2YgdGhpcyBwcm9qZWN0XShodHRwczovL2dpdGh1Yi5jb20vc3BvbnNvcnMva291bW91bC1kZXYpLiB8CnwtfC18CnwgWzxpbWcgYWx0PSJLb3Vtb3VsIGxvZ28iIHNyYz0iaHR0cHM6Ly9rb3Vtb3VsLmNvbS9zdGF0aWMvbG9nby1zbG9nYW4ucG5nIiBoZWlnaHQ9IjQwIj5dKGh0dHBzOi8va291bW91bC5jb20pIHwgW0tvdW1vdWxdKGh0dHBzOi8va291bW91bC5jb20pIGRldmVsb3BzIHRoZSBEYXRhIEZhaXIgZWNvc3lzdGVtIGFuZCBob3N0cyBpdCBhcyBhbiBvbmxpbmUgc2VydmljZS4gfAp8IFs8aW1nIGFsdD0iRGF3aXp6IGxvZ28iIHNyYz0iaHR0cHM6Ly9kYXdpenouZnIvbG9nby1EYXdpenotYWxsLWFib3V0LXlvdXItZGF0YS1ob21lLnBuZyIgaGVpZ2h0PSI0MCI+XShodHRwczovL2Rhd2l6ei5mcikgfCBbRGF3aXp6XShodHRwczovL2Rhd2l6ei5mcikgdXNlcyBEYXRhIEZhaXIgaW5zaWRlIGl0cyBwbGF0Zm9ybSBhbmQgc3VwcG9ydHMgaXRzIGRldmVsb3BtZW50LiB8CgojIyBEZXZlbG9wZXJzCgpUYWtlIGEgbG9vayBhdCB0aGUgW2NvbnRyaWJ1dGlvbiBndWlkZWxpbmVzXSguL0NPTlRSSUJVVElORy5tZCku readmeEtag: '"371ffe787bc5809f23aff5dfc3933f5feea445cb"' readmeLastModified: Tue, 13 Aug 2024 13:20:53 GMT repositoryId: 111959381 description: >- Findable, Accessible, Interoperable and Reusable Data. A complete open-source solution for your open and private data needs. French only for the time being, internationalization coming soon. created: '2017-11-24T22:19:00Z' updated: '2026-02-05T11:02:28Z' language: JavaScript archived: false stars: 42 watchers: 3 forks: 9 owner: data-fair logo: https://avatars.githubusercontent.com/u/83776320?v=4 license: AGPL-3.0 repoEtag: '"0167a1fe3358d9119742f26ae000c6282b29161b34a7851d1bfccdaf771e43f7"' repoLastModified: Thu, 05 Feb 2026 11:02:28 GMT foundInMaster: true category: - Documentation - Server Implementations id: 384e501654823f177688cd3302ea80ee - source: openapi3 tags repository: https://github.com/luoyunchong/igeekfan.aspnetcore.rapidoc v3: true repositoryMetadata: base64Readme: >- <div align="center">
<h1 align="center"> <img alt="MrinDoc logo" src="docs/images/logo.png" width="40px" />  IGeekFan.AspNetCore.RapiDoc </h1>

**[RapiDoc](https://github.com/mrin9/RapiDoc)** Custom Element for Open-API spec viewing ，Support .NET Core3.1 、.NET Standard2.0、.NET5.0、.NET6.0。

[![.NET IDE Rider](https://img.shields.io/static/v1?style=float&logo=rider&label=Rider&message=jetbrains&color=red)](https://www.jetbrains.com/rider/)
[![GitHub license](https://img.shields.io/badge/license-MIT-blue.svg)](https://raw.githubusercontent.com/luoyunchong/IGeekFan.AspNetCore.RapiDoc/master/LICENSE)
<p>
    <span>English</span> |  
    <a href="README.zh-CN.md">中文</a>
</p>
</div>

An API document that implements swagger 2.0 and OpenAPI 3.0. I integrate it into aspnetcore。

## Features
- Supports Swagger 2.0 and OpenAPI 3.0
- Works with any framework or with no framework
- Allows making API calls. And More....
- For more features, please refer to **[RapiDoc](https://github.com/mrin9/RapiDoc) README**

## Nuget Packages

Package name                              | Version                     | Downloads
------------------------------------------|-----------------------------|-------------
`IGeekFan.AspNetCore.RapiDoc` | [![NuGet](https://img.shields.io/nuget/v/IGeekFan.AspNetCore.RapiDoc.svg?style=flat-square&label=nuget&color=fedcba)](https://www.nuget.org/packages/IGeekFan.AspNetCore.RapiDoc/) | ![downloads](https://img.shields.io/nuget/dt/IGeekFan.AspNetCore.RapiDoc.svg)
`IGeekFan.AspNetCore.RapiDoc.Extra` | [![NuGet](https://img.shields.io/nuget/v/IGeekFan.AspNetCore.RapiDoc.Extra.svg?style=flat-square&label=nuget)](https://www.nuget.org/packages/IGeekFan.AspNetCore.RapiDoc.Extra/) | ![downloads](https://img.shields.io/nuget/dt/IGeekFan.AspNetCore.RapiDoc.Extra.svg)

 
## Dependencies
### [RapiDoc](https://github.com/mrin9/RapiDoc)
- rapidoc^(version) [https://www.npmjs.com/package/rapidoc](https://www.npmjs.com/package/rapidoc)
### [Swashbuckle.AspNetCore](https://github.com/domaindrivendev/Swashbuckle.AspNetCore)
- Swashbuckle.AspNetCore.Swagger
- Swashbuckle.AspNetCore.SwaggerGen

## Demo
- [Basic](https://github.com/luoyunchong/IGeekFan.AspNetCore.RapiDoc/blob/master/test/Basic)
- [RapiDocDemo](https://github.com/luoyunchong/IGeekFan.AspNetCore.RapiDoc/blob/master/test/RapiDocDemo)
- [OAuth2Integration](https://github.com/luoyunchong/IGeekFan.AspNetCore.RapiDoc/blob/master/test/WebSites/OAuth2Integration)
- [ASPNET Core 6](https://github.com/luoyunchong/IGeekFan.AspNetCore.RapiDoc/blob/master/test/AspNetCore6_RapiDemo)
## 📚 QuickStart

### 🚀 Install Package
use Swashbuckle.AspNetCore.Swagger Components

1.Install the standard Nuget package into your ASP.NET Core application.

```
Package Manager : 

Install-Package Swashbuckle.AspNetCore.Swagger
Install-Package Swashbuckle.AspNetCore.SwaggerGen
Install-Package IGeekFan.AspNetCore.RapiDoc

OR

CLI :

dotnet add package Swashbuckle.AspNetCore.Swagger
dotnet add package Swashbuckle.AspNetCore.SwaggerGen
dotnet add package IGeekFan.AspNetCore.RapiDoc
```

2.In the ConfigureServices method of Startup.cs, register the Swagger generator, defining one or more Swagger documents.

```
using Microsoft.AspNetCore.Mvc.Controllers
using Microsoft.OpenApi.Models;
using Swashbuckle.AspNetCore.SwaggerGen;
using IGeekFan.AspNetCore.RapiDoc;
```
### 🚁 ConfigureServices

3.Services Configure
```
   services.AddSwaggerGen(c =>
    {
        c.SwaggerDoc("v1",new OpenApiInfo{Title = "API V1",Version = "v1"});
        var filePath = Path.Combine(System.AppContext.BaseDirectory,$"{typeof(Startup).Assembly.GetName().Name}.xml");
        c.IncludeXmlComments(filePath, true);
    });
```

### 💪 Configure
4. Middleware Configure
```
app.UseSwagger();

app.UseRapiDocUI(c =>
{
    c.RoutePrefix = ""; // serve the UI at root
    c.SwaggerEndpoint("/v1/api-docs", "V1 Docs");
    //https://mrin9.github.io/RapiDoc/api.html
    //This Config Higher priority
    c.GenericRapiConfig = new GenericRapiConfig()
    {
        RenderStyle = "read",
        Theme = "light",//light | dark
        SchemaStyle = "table"////tree | table
    };
});

app.UseEndpoints(endpoints =>
{
    endpoints.MapControllers();
    endpoints.MapSwagger("{documentName}/api-docs");
});
```
[https://mrin9.github.io/RapiDoc/api.html](https://mrin9.github.io/RapiDoc/api.html) GenericRapiConfig Configuration items refer to this document

### 🔎 Views
Run Project，Open WebSite https://localhost:5001/index.html#/home

![docs/images/view.png](docs/images/view.png)

5.More Configure

To add comments to a document, right-click on the project - properties - generate

![](https://pic.downk.cc/item/5f34161d14195aa59413f0fc.jpg)

In AddSwaggerGen Methods You should add this methond

```
c.IncludeXmlComments(Path.Combine(AppContext.BaseDirectory, "SwaggerDemo.xml"),true);
```
The last parameter is set to true to enable comments on the controller



### NSwag.AspNetCore

Another way

Please refer to the table of contents [test/WebSites/NSwag.Swagger.RapiDoc](https://github.com/luoyunchong/IGeekFan.AspNetCore.RapiDoc/tree/master/test/WebSites/NSwag.Swagger.RapiDoc)

```
Package Manager : 

Install-Package IGeekFan.AspNetCore.RapiDoc

OR

CLI :

dotnet add package NSwag.AspNetCore
```

```
public void ConfigureServices(IServiceCollection services)
 {
    // 其它Service
     services.AddOpenApiDocument();
 }
```

```
public void Configure(IApplicationBuilder app, IWebHostEnvironment env)
{
        // 其它 Use
      app.UseOpenApi();
      app.UseRapiDocUI(c =>
     {
           c.RoutePrefix = "";
           c.SwaggerEndpoint("/swagger/v1/swagger.json");
           //https://mrin9.github.io/RapiDoc/api.html
           //This Config Higher priority
           c.GenericRapiConfig = new GenericRapiConfig()
           {
                RenderStyle = "read",
                Theme = "light",//light | dark
                SchemaStyle = "table"////tree | table
           };
      });
}
```
Every Things is Ok. Now， You can visist RapiDoc



### IGeekFan.AspNetCore.RapiDoc.Extra
There is only one class，Display labels on methods through 'filter'

CLI 
```
dotnet add package IGeekFan.AspNetCore.RapiDoc.Extra
```

in AddSwaggerGen services add `RapiDocLableOperationFilter` Filter、

Your need add this namespace`IGeekFan.AspNetCore.RapiDoc.Extra`

```diff
builder.Services.AddSwaggerGen(c =>
{
+   c.OperationFilter<RapiDocLableOperationFilter>();
    var filePath = Path.Combine(System.AppContext.BaseDirectory, $"{typeof(Program).Assembly.GetName().Name}.xml");
    c.IncludeXmlComments(filePath, true);
    c.SwaggerDoc("v1", new OpenApiInfo { Title = "AspNetCore6_RapiDemo", Version = "v1" });
});
```


In Your Controller，You can add this attribute to the method

```diff
+   [RapiDocLabel("Core Api")]
+   [RapiDocLabel("Test",RapiDocColor.RED)]
    [HttpGet(Name = "GetWeatherForecast")]
    public IEnumerable<WeatherForecast> Get()
    {
        return null;
    }
```
Views

![docs/images/light-badges.png](docs/images/light-badges.png)

- In appsettings.json 

```csharp
builder.Services.Configure<RapiDocOptions>(c =>{
    builder.Configuration.Bind("RapiDoc", c);
});
```

aoosettings.json
```json
 "RapiDoc": {
    "RoutePrefix": "swagger",
    "DocumentTitle": "ASPNET CORE 6 RAPI DOC",
    "GenericRapiConfig": {
      "Theme": "dark"
    }
  }
```
Use Middleware Config Higher priority, All GenericRapiConfig can configuration in you appsettings.json,Please refer to this documents [https://mrin9.github.io/RapiDoc/api.html](https://mrin9.github.io/RapiDoc/api.html)
```csharp
 app.UseRapiDocUI(c =>
    {
        //This Config Higher priority
        c.GenericRapiConfig = new GenericRapiConfig()
        {
            RenderStyle= "read",//read | view | focused
            Theme="light",//light | dark
            SchemaStyle= "table"//tree | table
        };

    });
```

### More Configure

- [https://github.com/domaindrivendev/Swashbuckle.AspNetCore](https://github.com/domaindrivendev/Swashbuckle.AspNetCore)
- [https://mrin9.github.io/RapiDoc/api.html](https://mrin9.github.io/RapiDoc/api.html)

 readmeEtag: '"4b762d10a1a847aa413b673b6d8043b50af83d96"' readmeLastModified: Fri, 03 May 2024 06:20:10 GMT repositoryId: 352583663 description: RapiDoc Custom Element for Open-API spec viewing +asp.net core created: '2021-03-29T09:17:57Z' updated: '2025-08-10T13:23:55Z' language: C# archived: false stars: 35 watchers: 1 forks: 2 owner: luoyunchong logo: https://avatars.githubusercontent.com/u/18613266?v=4 license: MIT repoEtag: '"7bdead23824fb94a1e3d28983fcf0a8595d5acfa04fd13c55623ce0ac59e5760"' repoLastModified: Sun, 10 Aug 2025 13:23:55 GMT foundInMaster: true category: SDK id: 35f12ae192b1843f5eb2aa583dc33f45 - source: - openapi3 tags - openapi31 tags repository: https://github.com/shapehq/framna-docs v3: true v3_1: true id: 8e493592c82829f4d1bb7891c4ac356c repositoryMetadata: base64Readme: >- <div align="center">
<img width="200" src="https://github.com/shapehq/framna-docs/raw/main/logo.png" alt="Framna Docs logo" />
</div>

<div align="center">
<h3>👋 Welcome to Framna Docs</h3>
<h4>Self-hosted web portal that centralizes OpenAPI documentation and facilitates spec-driven development, built with GitHub-based authorization.</h4>
</div>

<div align="center">
<a href="https://github.com/shapehq/framna-docs/actions/workflows/build.yml"><img src="https://github.com/shapehq/framna-docs/actions/workflows/build.yml/badge.svg"></a>
<a href="https://github.com/shapehq/framna-docs/actions/workflows/run-unit-tests.yml"><img src="https://github.com/shapehq/framna-docs/actions/workflows/run-unit-tests.yml/badge.svg"></a>
<a href="https://github.com/shapehq/framna-docs/actions/workflows/test-sql-queries.yml"><img src="https://github.com/shapehq/framna-docs/actions/workflows/test-sql-queries.yml/badge.svg"></a>
<a href="https://github.com/shapehq/framna-docs/actions/workflows/lint.yml"><img src="https://github.com/shapehq/framna-docs/actions/workflows/lint.yml/badge.svg"></a>
<a href="https://github.com/shapehq/framna-docs/actions/workflows/build-docker-image"><img src="https://github.com/shapehq/framna-docs/actions/workflows/build-docker-image.yml/badge.svg"></a>
</div>

---

<div align="center">
<a href="#-getting-started">🚀 Getting Started</a>&nbsp;&nbsp;&nbsp;&nbsp;
<a href="#-how-does-it-work">👨‍🔧 How does it work?</a>&nbsp;&nbsp;&nbsp;&nbsp;
<a href="#-how-can-i-contribute">👩‍💻 How can I contribute?</a>&nbsp;&nbsp;&nbsp;&nbsp;
<a href="https://github.com/shapehq/framna-docs/wiki">📖 Wiki</a>
</div>

<hr />

Framna Docs makes managing and previewing OpenAPI documentation a breeze, streamlining spec-driven development. With GitHub-based authorization, you can easily control who accesses your docs. Framna Docs comments on pull requests that tweak your OpenAPI specs, giving you preview URLs to ensure every update is well-reviewed.

<div align="center">
<img width="600" src="https://github.com/shapehq/framna-docs/raw/main/wiki/home.png?raw=true" alt="Screenshot of Framna Docs"/>
</div>

## 🚀 Getting Started

Please refer to the following articles in [the wiki](https://github.com/shapehq/framna-docs/wiki) to get started with Framna Docs.

- [Adding Documentation to Framna Docs](https://github.com/shapehq/framna-docs/wiki/Adding-Documentation-to-Framna-Docs)
- [Browsing Documentation](https://github.com/shapehq/framna-docs/wiki/Browsing-Documentation)
- [Updating Documentation](https://github.com/shapehq/framna-docs/wiki/Updating-Documentation)
- [Deploying Framna Docs](https://github.com/shapehq/framna-docs/wiki/Deploying-Framna-Docs)

### Install the OpenAPI diff tool locally

Framna Docs relies on the [`oasdiff`](https://github.com/oasdiff/oasdiff) CLI when comparing specifications.  

On MacOS you can install with homebrew:

```bash
brew tap oasdiff/homebrew-oasdiff
brew install oasdiff
```

## 👨‍🔧 How does it work?

Framna Docs uses [OpenAPI specifications](https://swagger.io) from GitHub repositories. Users log in with their GitHub account to access documentation for projects they have access to. A repository only needs an OpenAPI spec to be recognized by Framna Docs, but customization is possible with a .framna-docs.yml file. Here's an example:

<img width="650" src="https://github.com/shapehq/framna-docs/raw/main/wiki/example-openapi-repository-with-config.png?raw=true"/>

Framna Docs supports spec-driven development by requiring OpenAPI specs in GitHub repos, ensuring version control and peer review. When a pull request is opened, Framna Docs comments with links to preview the documentation:

<img width="760" src="https://github.com/shapehq/framna-docs/raw/main/wiki/pr-comment.png?raw=true"/>

Learn more from the [Adding Documentation](https://github.com/shapehq/framna-docs/wiki/Adding-Documentation-to-Framna-Docs), [Browsing Documentation](https://github.com/shapehq/framna-docs/wiki/Browsing-Documentation), and [Updating Documentation](https://github.com/shapehq/framna-docs/wiki/Updating-Documentation) articles in the wiki.

## 👩‍💻 How can I contribute?

Pull requests with bugfixes and new features are much appreciated. We are happy to review PRs and merge them once they are ready, as long as they contain changes that fit within the vision of Framna Docs.

Clone the repository and consult the articles on [running Framna Docs locally](https://github.com/shapehq/framna-docs/wiki/Running-Framna-Docs-Locally) and [contributing](https://github.com/shapehq/framna-docs/wiki/Contributing) to get started contributing changes the project.

```bash
git clone git@github.com:shapehq/framna-docs.git
```

### 🔀 Git Workflow

Two following long-lived branches exist:

* **main**: Stable/release branch meant for deployment to the production environment.
* **develop**: Branch meant for deployment to a staging environment.

**Do's 👍**

- Features are branched off from `develop` and merged back in using a PR when ready. Rebase or merge `develop` in to keep the feature branch up to date. Squash merge the feature branch into `develop`.
- `develop` is merged into `main` whenever a new release is made. Only regular merge commits are allowed in this case. You do not need to bring develop up to date with `main` before merging.
- A hotfix is applied by branching out from `main`. The hotfix branch _must_ be merged into both `main` and `develop`.

**Don'ts 🙅‍♂️**

- Never squash merge `develop` into `main`.

## ❤️ The Product of a Shape Weekend

Before we became Framna, our company was known as Shape. Every year, we held "Shape Weekend", a three-day hackathon where all employees came together to build exciting new products. In 2023, one of those teams, passionate about documentation and spec-driven development, built Framna Docs. We have used it daily ever since.

---

Framna Docs is built with ❤️ by [Framna](https://framna.com/) in Denmark. Oh, and [we are hiring](https://framna.com/careers) 🤗
 readmeEtag: '"dda1ecea634317f2453f752915413f8c7b576c3e"' readmeLastModified: Mon, 08 Dec 2025 08:01:45 GMT repositoryId: 702539097 description: >- 📖❤️ Self-hosted web portal that centralizes OpenAPI documentation and facilitates spec-driven development, built with GitHub-based authorization. created: '2023-10-09T14:04:55Z' updated: '2026-02-01T09:39:50Z' language: TypeScript archived: false stars: 44 watchers: 1 forks: 2 owner: shapehq logo: https://avatars.githubusercontent.com/u/818802?v=4 license: MIT repoEtag: '"870faba9e401d760d32d1957ffb606a83b777c37dee73368c465305247873ab1"' repoLastModified: Sun, 01 Feb 2026 09:39:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/contentjet/openapi-ui v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFVJCgpUaGlzIGlzIGEgUmVhY3QgYmFzZWQgc2luZ2xlIHBhZ2UgYXBwIHdoaWNoIHJlbmRlcnMgZG9jdW1lbnRhdGlvbiBmcm9tIGEgdmFsaWQgT3BlbkFQSSAzLjAuMC1SQzAgZG9jdW1lbnQuCgojIE5PVEUKClRoaXMgcHJvamVjdCB3YXMgZXhwZXJpbWVudGFsIGFuZCBpcyBub3QgYmVpbmcgYWN0aXZlbHkgbWFpbnRhaW5lZC4KCiMjIEdldHRpbmcgc3RhcnRlZAoKIyMjIyBQcm9qZWN0IHNldHVwCgpJbnN0YWxsIHByb2plY3QgZGVwZW5kZW5jaWVzOgoKYGBgCm5wbSBpbnN0YWxsCmBgYAoKIyMjIyBSdW4gZGV2ZWxvcG1lbnQgc2VydmVyCgpgYGAKbnBtIHN0YXJ0CmBgYAoKIyMjIyBSdW4gbGludGVyCgpgYGAKbnBtIHJ1biBsaW50CmBgYAoKIyMjIyBDcmVhdGUgcHJvZHVjdGlvbiBidWlsZAoKYGBgCm5wbSBydW4gYnVpbGQKYGBgCg== readmeEtag: '"096c9725fbaffffd629a454da4dc657fefc07dd4"' readmeLastModified: Thu, 15 Aug 2019 06:16:12 GMT repositoryId: 84547405 description: React based OpenAPI 3.0+ documentation generator created: '2017-03-10T10:16:22Z' updated: '2024-10-03T09:19:57Z' language: JavaScript archived: false stars: 33 watchers: 1 forks: 3 owner: contentjet logo: https://avatars.githubusercontent.com/u/21168640?v=4 repoEtag: '"9b597ef219dfcbfdf5c3b0609722f3587270938ae9df8d3e0efaf46cbbcafd4d"' repoLastModified: Thu, 03 Oct 2024 09:19:57 GMT foundInMaster: true category: - Description Validators - Server Implementations id: 28b84909b56d604ecd1db7ae362fec50 - source: openapi3 tags repository: https://github.com/anupsaund/vertx-auto-swagger v3: true repositoryMetadata: base64Readme: >- IyB2ZXJ0eC1hdXRvLXN3YWdnZXIKCj4gVEw7RFIgLSBIb3cgdG8gZ2V0IGphdmEgVmVydC54IHRvIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGUgYSBPcGVuIEFQSSB2MyBzcGVjIChBLmsuYS4gU3dhZ2dlcikgYW5kIHNlcnZlIGl0IHRvIFN3YWdnZXIgVUksIHNlcnZlZCB0aHJvdWdoIFZlcnQueC4KCiMjIyBNb3RpdmF0aW9uCkkgbmVlZGVkIGEgd2F5IHRvIGdlbmVyYXRlIGEgc3dhZ2dlciBzcGVjIGZyb20gSmF2YSBjb2RlLCBpbnN0ZWFkIG9mIGhhdmluZyB0byBoYW5kIGNyYWZ0IGEgc3dhZ2dlciBKU09OIGZpbGUuCgpJIGFsc28gd2FudGVkIHRvIHNlcnZlIG91dCB0aGUgc3BlYyBmcm9tIFZlcnQueCBpbnRvIFN3YWdnZXIgVUkgc28gdGhhdCBpdCBjb3VsZCBiZSB1c2VkIGJ5IGFsbC4KClRoZXJlIGlzIGEgbW9yZSBkZXRhaWxlZCB3cml0ZSB1cCBhdmFpbGFibGUgb24gbXkgYmxvZyBwb3N0OiAKaHR0cDovL2FudXBzYXVuZC5jb20vaG93LXRvLWdlbmVyYXRlLW9wZW5hcGktMy0wLXN3YWdnZXItc3BlYy1mcm9tLXZlcnR4LWphdmEtYW5kLXNlcnZlLWl0LXZpYS1zd2FnZ2VyLXVpCgojIyMgV2hhdCBkb2VzIGl0IGRvPwoKMS4gIFJlYWQgSmF2YSBBbm5vdGF0aW9ucyBhbmQgbWFwIHRoZW0gaW50byBhIG9wZW5BUEkgc3BlYy4KMi4gIFNlcnZlIHRoZSBvcGVuQVBJIHNwZWMgb3V0IG9uIGFuIGVuZCBwb2ludC4KMy4gIFNlcnZlIGEgZGlzdHJpYnV0YWJsZSB2ZXJzaW9uIG9mIFN3YWdnZXJVSSB3aGljaCBwcmVzZW50cyB0aGUgc3dhZ2dlciBzcGVjIGZyb20gcG9pbnQgMi4KCgojIyMgU2NyZWVuc2hvdCBvZiBTd2FnZ2VyIFVJCiFbU3dhZ2dlclVJXShVSS5QTkcgIlN3YWdnZXJVSSIpCgojIyMgV2hhdCBkb2VzIGEgdHlwaWNhbCBKYXZhIGFubm90YXRpb24gbG9vayBsaWtlPwoKYGBgamF2YQpAT3BlcmF0aW9uKHN1bW1hcnkgPSAiRmluZCBwcm9kdWN0cyBieSBJRCIsIG1ldGhvZCA9ICJHRVQiLCBvcGVyYXRpb25JZCA9ICJwcm9kdWN0Lzpwcm9kdWN0SWQiLAogICAgdGFncyA9IHsKICAgICAgIlByb2R1Y3QiCiAgICB9LAogICAgcGFyYW1ldGVycyA9IHsKICAgICAgQFBhcmFtZXRlcihpbiA9IFBhcmFtZXRlckluLlBBVEgsIG5hbWUgPSAicHJvZHVjdElkIiwKICAgICAgICByZXF1aXJlZCA9IHRydWUsIGRlc2NyaXB0aW9uID0gIlRoZSB1bmlxdWUgSUQgYmVsb25naW5nIHRvIHRoZSBwcm9kdWN0Iiwgc2NoZW1hID0gQFNjaGVtYSh0eXBlID0gInN0cmluZyIpKQogICAgfSwKICAgIHJlc3BvbnNlcyA9IHsKICAgICAgQEFwaVJlc3BvbnNlKHJlc3BvbnNlQ29kZSA9ICIyMDAiLCBkZXNjcmlwdGlvbiA9ICJPSyIsCiAgICAgICAgY29udGVudCA9IEBDb250ZW50KAogICAgICAgICAgbWVkaWFUeXBlID0gImFwcGxpY2F0aW9uL2pzb24iLAogICAgICAgICAgZW5jb2RpbmcgPSBARW5jb2RpbmcoY29udGVudFR5cGUgPSAiYXBwbGljYXRpb24vanNvbiIpLAogICAgICAgICAgc2NoZW1hID0gQFNjaGVtYShuYW1lID0gInByb2R1Y3QiLCBleGFtcGxlID0KICAgICAgICAgICAgInsiICsKICAgICAgICAgICAgICAiJ19pZCc6J2FiYycsIiArCiAgICAgICAgICAgICAgIid0aXRsZSc6J1JlZCBUcnVjaycsIiArCiAgICAgICAgICAgICAgIidpbWFnZV91cmwnOidodHRwczovL2ltYWdlcy5wZXhlbHMuY29tL3Bob3Rvcy8xMTEyNTk3L3BleGVscy1waG90by0xMTEyNTk3LmpwZWcnLCIgKwogICAgICAgICAgICAgICInZnJvbV9kYXRlJzonMjAxOC0wOC0zMCcsIiArCiAgICAgICAgICAgICAgIid0b19kYXRlJzonMjAxOS0wOC0zMCcsIiArCiAgICAgICAgICAgICAgIidwcmljZSc6JzEyNS4wMCcsIiArCiAgICAgICAgICAgICAgIidlbmFibGVkJzp0cnVlIiArCiAgICAgICAgICAgICAgIn0iLAogICAgICAgICAgICBpbXBsZW1lbnRhdGlvbiA9IFByb2R1Y3QuY2xhc3MpCiAgICAgICAgKQogICAgICApLAogICAgICBAQXBpUmVzcG9uc2UocmVzcG9uc2VDb2RlID0gIjQwNCIsIGRlc2NyaXB0aW9uID0gIk5vdCBmb3VuZC4iKSwKICAgICAgQEFwaVJlc3BvbnNlKHJlc3BvbnNlQ29kZSA9ICI1MDAiLCBkZXNjcmlwdGlvbiA9ICJJbnRlcm5hbCBTZXJ2ZXIgRXJyb3IuIikKICAgIH0KICApCmBgYAoKIyMgSG93IHRvIGdldCBpdCBydW5uaW5nCgo+IERlcGVuZGFuY2llczogTWF2ZW4sIEpBVkEgYW5kIGEgSkFWQSBJREUgaXMgaGVscGZ1bC4KCjEuIENsb25lIHRoZSByZXBvc2l0b3J5IGFuZCB1c2UgTWF2ZW4gdG8gaW5zdGFsbCBkZXBlbmRhbmNpZXMuCjEuIEluIEludGVsbGlqIHNldCB1cCBhIGNvbmZpZyB0byBydW4gYSBKYXZhIEFwcGxpY2F0aW9uIHdpdGggdGhlIGZvbGxvd2luZyBzZXR0aW5ncy4KCmBgYCAKICBNYWluIENsYXNzOiBpby52ZXJ0eC5jb3JlLkxhdW5jaGVyCiAgVk0gT3B0aW9uczogPHVwIHRvIHlvdSwgb3IgbGVhdmUgYmxhbms+CiAgUHJvZ3JhbSBBcmd1bWVudHM6IHJ1biBpby52ZXJ0eC5WZXJ0eEF1dG9Td2FnZ2VyLk1haW5WZXJ0aWNsZQpgYGAKCjMuIEFmdGVyIHRoYSBhcHBsaWNhdGlvbiBoYXMgbGF1bmNoZWQsIGdvIHRvIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9kb2MvaW5kZXguaHRtbAoKIyMjIyBTcGVjaWFsIENyZWRpdApHb2VzIHRvIENocmlzdG9zIEthcmF0emFzIGZvciBjcmVhdGluZyB0aGUgZ2VuZXJhdG9yIGNsYXNzIHdoaWNoIGhhcyBiZWVuIHVzZWQgYW5kIGVuaGFuY2VkIGZvciB0aGlzIHJlc3Bvc2l0b3J5OgogCkhpcyBvcmlnaW5hbCByZXBvc2l0b3J5IGlzIGF2YWlsYWJsZSBhdDogaHR0cHM6Ly9naXRodWIuY29tL2NrYXJhdHphcy92ZXJ0eC1vcGVuYXBpLXNwZWMtZ2VuZXJhdG9yCgo= readmeEtag: '"b8f5574724322b3a9f82670fee3c87e13b84bb50"' readmeLastModified: Wed, 23 Feb 2022 05:43:43 GMT repositoryId: 147231389 description: Java Micro Service API Generation created: '2018-09-03T16:50:50Z' updated: '2024-12-20T07:55:56Z' language: Java archived: false stars: 33 watchers: 4 forks: 15 owner: anupsaund logo: https://avatars.githubusercontent.com/u/6728822?v=4 license: MIT repoEtag: '"a07e82fe65046a5699d893963f706fa94f7f62285d35818a58ab5662ae4aad06"' repoLastModified: Fri, 20 Dec 2024 07:55:56 GMT foundInMaster: true category: Parsers id: d2fbd0151824a8918e13f184d83e8083 - source: openapi3 tags repository: https://github.com/ryan-mars/workers-queue-demo v3: true repositoryMetadata: base64Readme: >- IyBEdXJhYmxlIE9iamVjdHMgUkVTVCBNZXNzYWdlIFF1ZXVlIEV4YW1wbGUKCj4gTm90ZTogWW91IG11c3QgdXNlIFt3cmFuZ2xlcl0oaHR0cHM6Ly9kZXZlbG9wZXJzLmNsb3VkZmxhcmUuY29tL3dvcmtlcnMvY2xpLXdyYW5nbGVyL2luc3RhbGwtdXBkYXRlKSAxLjE5LjMgb3IgbmV3ZXIgdG8gZGVwbG95IHRoaXMgcHJvamVjdC4KCkFuIGVkZ2UgbWVzc2FnZSBxdWV1ZSBzZXJ2aWNlIHRoYXQgcnVucyBvbiBDbG91ZGZsYXJlIFdvcmtlcnMgdXNpbmcgRHVyYWJsZSBPYmplY3RzIGFuZCBLVi4gSXQgaXMgaW1wbGVtZW50ZWQgYXMgYSBzaW1wbGUgUkVTVCBBUEkuCgpUaGlzIGRlbW8gdXNlczoKCi0gW0Nsb3VkZmxhcmUgV29ya2Vyc10oaHR0cHM6Ly9kZXZlbG9wZXJzLmNsb3VkZmxhcmUuY29tL3dvcmtlcnMvbGVhcm5pbmcvaG93LXdvcmtlcnMtd29ya3MpCi0gW0R1cmFibGUgT2JqZWN0c10oaHR0cHM6Ly9kZXZlbG9wZXJzLmNsb3VkZmxhcmUuY29tL3dvcmtlcnMvbGVhcm5pbmcvdXNpbmctZHVyYWJsZS1vYmplY3RzKQotIFtLVl0oaHR0cHM6Ly9kZXZlbG9wZXJzLmNsb3VkZmxhcmUuY29tL3dvcmtlcnMvcnVudGltZS1hcGlzL2t2KQotIFtNaW5pZmxhcmUgdjJdKGh0dHBzOi8vdjIubWluaWZsYXJlLmRldikKLSBbVHlwZVNjcmlwdF0oaHR0cHM6Ly93d3cudHlwZXNjcmlwdGxhbmcub3JnKQotIFtKZXN0XShodHRwczovL2plc3Rqcy5pbykKLSBbRVMgTW9kdWxlc10oaHR0cHM6Ly9kZXZlbG9wZXIubW96aWxsYS5vcmcvZW4tVVMvZG9jcy9XZWIvSmF2YVNjcmlwdC9HdWlkZS9Nb2R1bGVzKQotIFtlc2J1aWxkXShodHRwczovL2VzYnVpbGQuZ2l0aHViLmlvKQotIFtXcmFuZ2xlcl0oaHR0cHM6Ly9kZXZlbG9wZXJzLmNsb3VkZmxhcmUuY29tL3dvcmtlcnMvY2xpLXdyYW5nbGVyL2luc3RhbGwtdXBkYXRlKQoKIyMgUnVubmluZyB0aGUgRGVtbwoKLSBgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9yeWFuLW1hcnMvd29ya2Vycy1xdWV1ZS1kZW1vLmdpdGAKLSBJbnN0YWxsIFt3cmFuZ2xlcl0oaHR0cHM6Ly9kZXZlbG9wZXJzLmNsb3VkZmxhcmUuY29tL3dvcmtlcnMvY2xpLXdyYW5nbGVyL2luc3RhbGwtdXBkYXRlKQotIGBjZCB3b3JrZXJzLXF1ZXVlLWRlbW9gCi0gYG5wbSBpbnN0YWxsYAotIGB3cmFuZ2xlciBsb2dpbmAKLSBgd3JhbmdsZXIga3Y6bmFtZXNwYWNlIGNyZWF0ZSBRVUVVRVNfS1ZgCiAgWW91IHNob3VsZCBzZWUgc29tZXRoaW5nIGxpa2U6CgpgYGAK8J+MgCAgQ3JlYXRpbmcgbmFtZXNwYWNlIHdpdGggdGl0bGUgIndvcmtlcnMtcXVldWUtZGVtby1RVUVVRVNfS1YiCuKcqCAgU3VjY2VzcyEKQWRkIHRoZSBmb2xsb3dpbmcgdG8geW91ciBjb25maWd1cmF0aW9uIGZpbGUgaW4geW91ciBrdl9uYW1lc3BhY2VzIGFycmF5Ogp7IGJpbmRpbmcgPSAiUVVFVUVTX0tWIiwgaWQgPSAiYmFmMDg1YzIzYy4uLiIgfQpgYGAKCi0gQWRkIHRoZSBLViBjb25maWcgdG8gYHdyYW5nbGVyLnRvbWxgCi0gYHdyYW5nbGVyIHB1Ymxpc2hgIHdpbGwgYnVpbGQsIHJ1biB0ZXN0cywgYW5kIGRlcGxveSB0byBDbG91ZGZsYXJlCgpZb3Ugc2hvdWxkIHNlZSBzb21ldGhpbmcgbGlrZToKCmBgYArinKggIEJ1aWxkIGNvbXBsZXRlZCBzdWNjZXNzZnVsbHkhYArinKggIFN1Y2Nlc3NmdWxseSBwdWJsaXNoZWQgeW91ciBzY3JpcHQgdG8KIGh0dHBzOi8vd29ya2Vycy1xdWV1ZS1kZW1vLnlvdTEyMzQud29ya2Vycy5kZXYKYGBgCgo+IE5vdGUgdGhlIFVSTCDwn5GG8J+PuwoKQ29weS9wYXN0ZSB0aGUgY29udGVudHMgb2YgYHNjaGVtYS55YW1sYCBpbnRvIFtTd2FnZ2VyIEVkaXRvcl0oaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby8pLiBDaGFuZ2UgdGhlIGBzZXJ2ZXJzYCBrZXkgdG8gbWF0Y2ggeW91ciBkZXBsb3llZCBVUkwuCgpIYXZlIGZ1biEKCiMjIFdoZXJlIHRvIGZpbmQgdGhpbmdzCgotIFJFU1QgQVBJIHNwZWMgaXMgaW4gYHNjaGVtYS55YW1sYC4KLSBXb3JrZXIgY29kZSBpcyBpbiBgc3JjL2AuIFRoZSBEdXJhYmxlIE9iamVjdCBgUXVldWVgIGNsYXNzIGlzIGluIGBzcmMvcXVldWUudHNgLCBhbmQgdGhlIFdvcmtlciBzY3JpcHQgaXMgaW4gYHNyYy9pbmRleC50c2AuCi0gZXNidWlsZCBpcyBjb25maWd1cmVkIHRvIG91dHB1dCBhIGJ1bmRsZWQgRVMgTW9kdWxlIHRvIGBkaXN0L2luZGV4Lm1qc2AuCi0gVW5pdCB0ZXN0cyBpbiBgc3JjL2luZGV4LnNwZWMudHNgLCB3aGljaCB3aWxsIHJ1biBhcyBwYXJ0IG9mIGB3cmFuZ2xlciBidWlsZGAuIFRvIHJ1biB0ZXN0cyBvbiB0aGVpciBvd24gdXNlIGBucG0gdGVzdGAuCg== readmeEtag: '"f9c76c7d6c392322cbda83405447e0547810b5f3"' readmeLastModified: Sun, 21 Nov 2021 06:08:47 GMT repositoryId: 430176829 description: >- Edge message queue service on Cloudflare Workers using Durable Objects and KV. It provides a simple REST API. created: '2021-11-20T18:14:55Z' updated: '2025-10-16T08:08:15Z' language: TypeScript archived: false stars: 38 watchers: 2 forks: 3 owner: ryan-mars logo: https://avatars.githubusercontent.com/u/824046?v=4 license: MIT repoEtag: '"22d242be1f2c69b4e8ff36b87539890fd826434db3daad065e5852b4e6ead135"' repoLastModified: Thu, 16 Oct 2025 08:08:15 GMT foundInMaster: true category: - Documentation - Server Implementations id: 1f14d7ec72292757f719d521c8fd5f54 - source: openapi3 tags repository: https://github.com/tfranzel/drf-spectacular-sidecar v3: true repositoryMetadata: base64Readme: >- PT09PT09PT09PT09PT09PT09PT09PT0KZHJmLXNwZWN0YWN1bGFyLXNpZGVjYXIKPT09PT09PT09PT09PT09PT09PT09PT0KCnxweXBpLXZlcnNpb258IHxweXBpLWRsfAoKU2VydmUgc2VsZi1jb250YWluZWQgZGlzdHJpYnV0aW9uIGJ1aWxkcyBvZiBgU3dhZ2dlciBVSWBfIGFuZCBgUmVkb2NgXyB3aXRoIGBEamFuZ29gXyBlaXRoZXIgdmlhIGBydW5zZXJ2ZXJgXyBvciBgY29sbGVjdHN0YXRpY2BfLgoKVGhpcyBEamFuZ28gYXBwIGlzIGFuIG9wdGlvbmFsIGFkZGl0aW9uIHRvIGBkcmYtc3BlY3RhY3VsYXJgXywgYnV0IGRvZXMgbm90IGRlcGVuZCBvbiBpdC4gSXQgbWF5IGFsc28gYmUgdXNlZCBpbmRlcGVuZGVudGx5LgoKKiBgU3dhZ2dlciBVSWBfIHZlcnNpb24gYGA1LjMxLjBgYCAoYG5wbSA8aHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvc3dhZ2dlci11aS1kaXN0PmBfXykKKiBgUmVkb2NgXyB2ZXJzaW9uIGBgMi41LjJgYCAoYG5wbSA8aHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvcmVkb2M+YF9fKQoKVGhpcyBpcyBhIHNlbGYtdXBkYXRpbmcgYW5kIHNlbGYtcHVibGlzaGluZyByZXBvc2l0b3J5IHRoYXQgbG9va3MgZm9yIHVwZGF0ZXMgb24gdGhlIDFzdCBvZiBldmVyeSBtb250aC4KVGhlIGRpc3RyaWJ1dGlvbiBmaWxlcyBhcmUgc291cmNlZCBmcm9tIG5wbSB2aWEgYGpzZGVsaXZyYF8sIHZhbGlkYXRlZCwgcGFja2FnZWQgYW5kIHVwbG9hZGVkIHRvIGBQeVBJYF8uCgpJbnN0YWxsYXRpb24KLS0tLS0tLS0tLS0tCgouLiBjb2RlOjogYmFzaAoKICAgICQgcGlwIGluc3RhbGwgZHJmLXNwZWN0YWN1bGFyLXNpZGVjYXIKClRoZSBwYWNrYWdlIG5lZWRzIHRvIGJlIHJlZ2lzdGVyZWQgdG8gYWxsb3cgRGphbmdvIHRvIGRpc2NvdmVyIHRoZSBzdGF0aWMgZmlsZXMuCgouLiBjb2RlOjogcHl0aG9uCgogICAgSU5TVEFMTEVEX0FQUFMgPSBbCiAgICAgICAgIyBBTEwgWU9VUiBBUFBTCiAgICAgICAgJ2RyZl9zcGVjdGFjdWxhcl9zaWRlY2FyJywKICAgIF0KClJlcXVpcmVtZW50cwotLS0tLS0tLS0tLS0KCkRqYW5nbyA+PSAyLjIKCkxpY2Vuc2VzCi0tLS0tLS0tCgpQcm92aWRlZCBieSBgVC4gRnJhbnplbCA8aHR0cHM6Ly9naXRodWIuY29tL3RmcmFuemVsPmBfLiBgTGljZW5zZWQgdW5kZXIgMy1DbGF1c2UgQlNEIDxodHRwczovL2dpdGh1Yi5jb20vdGZyYW56ZWwvZHJmLXNwZWN0YWN1bGFyLXNpZGVjYXIvYmxvYi9tYXN0ZXIvTElDRU5TRT5gXy4KClRoaXMgcGFja2FnZSBpbmNsdWRlcyBkaXN0cmlidXRpb24gYnVpbGRzIG9mCgoqIGBTd2FnZ2VyIFVJYF86IFRoZSBgb3JpZ2luYWwgbGljZW5zZSAoQXBhY2hlIDIuMCkgPGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLXVpL2Jsb2IvbWFzdGVyL0xJQ0VOU0U+YF8gYW5kIGNvcHlyaWdodCBhcHBseSB0byB0aG9zZSBmaWxlcy4KKiBgUmVkb2NgXzogVGhlIGBvcmlnaW5hbCBsaWNlbnNlIChNSVQpIDxodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9yZWRvYy9ibG9iL21hc3Rlci9MSUNFTlNFPmBfIGFuZCBjb3B5cmlnaHQgYXBwbHkgdG8gdGhvc2UgZmlsZXMuCgoKLi4gfHB5cGktdmVyc2lvbnwgaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvdi9kcmYtc3BlY3RhY3VsYXItc2lkZWNhci5zdmcKICAgOnRhcmdldDogaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2RyZi1zcGVjdGFjdWxhci1zaWRlY2FyLwouLiB8cHlwaS1kbHwgaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL3B5cGkvZG0vZHJmLXNwZWN0YWN1bGFyLXNpZGVjYXIKICAgOnRhcmdldDogaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2RyZi1zcGVjdGFjdWxhci1zaWRlY2FyLwoKLi4gX1B5UEk6IGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9kcmYtc3BlY3RhY3VsYXItc2lkZWNhci8KLi4gX2pzZGVsaXZyOiBodHRwczovL3d3dy5qc2RlbGl2ci5jb20vCi4uIF9EamFuZ286IGh0dHBzOi8vd3d3LmRqYW5nb3Byb2plY3QuY29tLwouLiBfZHJmLXNwZWN0YWN1bGFyOiBodHRwczovL2dpdGh1Yi5jb20vdGZyYW56ZWwvZHJmLXNwZWN0YWN1bGFyCi4uIF9SZWRvYzogaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvcmVkb2MKLi4gX1N3YWdnZXIgVUk6IGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLXVpCi4uIF9jb2xsZWN0c3RhdGljOiBodHRwczovL2RvY3MuZGphbmdvcHJvamVjdC5jb20vZW4vMy4yL3JlZi9jb250cmliL3N0YXRpY2ZpbGVzLyNjb2xsZWN0c3RhdGljCi4uIF9ydW5zZXJ2ZXI6IGh0dHBzOi8vZG9jcy5kamFuZ29wcm9qZWN0LmNvbS9lbi8zLjIvcmVmL2NvbnRyaWIvc3RhdGljZmlsZXMvI3J1bnNlcnZlcgo= readmeEtag: '"45ee5d750fd05b791465879df4cd4df938129de0"' readmeLastModified: Thu, 01 Jan 2026 11:27:07 GMT repositoryId: 410115575 description: >- Serve self-contained distribution builds of Swagger UI and Redoc with Django. created: '2021-09-24T22:17:14Z' updated: '2026-01-13T18:15:48Z' language: Python archived: false stars: 46 watchers: 1 forks: 6 owner: tfranzel logo: https://avatars.githubusercontent.com/u/13507857?v=4 license: BSD-3-Clause repoEtag: '"daf8f9de5c7d4734e00ff9cc7d70c6275e83a27349614086d6a3418e42c71a5f"' repoLastModified: Tue, 13 Jan 2026 18:15:48 GMT foundInMaster: true category: - Converters - Server Implementations id: efb63b84a6a36939fb408835301f3b03 - source: openapi3 tags repository: https://github.com/maximiliankoestler/hcloud-openapi v3: true repositoryMetadata: base64Readme: >- # Unofficial OpenAPI Description for the Hetzner Cloud API

![](https://github.com/MaximilianKoestler/hcloud-openapi/workflows/CI%20Build%20and%20OpenAPI%20Spec%20Generation/badge.svg)

This is an attempt to improve Hetzner's own OpenAPI specification to make it usable in code generation.

When this project was started, Hetzner had not published their own OpenAPI specification for their cloud API, so I decided to build my own based on the HTML documentation on their website.
By now, Hetzner has actually published an [OpenAPI spec for their Cloud API](https://docs.hetzner.cloud/cloud.spec.json) and recently also [the Hetzner API](https://docs.hetzner.cloud/hetzner.spec.json) However, I think this project still adds some value.

While Hetzner now appears to generate their documentation website from the OpenAPI spec, the spec is not very useful for automatic code generation.
This project aims to convert the official specs to an improved version with the following main features:

- A single file describing all API endpoints from multiple source documents
- Definition of common components that are reused throughout the schema
- API-friendly tag names
- Unique `operationId`s

As an added benefit, mainly from use of common components, the improved spec is much smaller (1.162 MB vs 2.902 MB) than the original.

## Further Reading

- [Link to the generated openAPI document](openapi/hcloud.json)
- [Official API documentation](https://docs.hetzner.cloud/)
- [OpenAPI spec document provided by Hetzner for the Hetzner Cloud API](https://docs.hetzner.cloud/cloud.spec.json)
- [OpenAPI spec document provided by Hetzner for the Hetzner API](https://docs.hetzner.cloud/hetzner.spec.json)
- [OpenAPI standard](https://swagger.io/specification/)

## Use Cases

This API description is currently being used by the following projects:

- [hcloud-rust](https://github.com/HenningHolmDE/hcloud-rust)
  [![Crates.io](https://img.shields.io/crates/v/hcloud.svg)](https://crates.io/crates/hcloud)
  [![Documentation](https://docs.rs/hcloud/badge.svg)](https://docs.rs/hcloud/)

## Supported Endpoints

- actions
- certificates
- datacenters
- firewalls
- floating_ips
- images
- isos
- load_balancer_types
- load_balancers
- locations
- networks
- placement_groups
- pricing
- primary_ips
- server_types
- servers
- ssh_keys
- storage_box_types
- storage_boxes
- volumes
- zones

```
# Main hcloud API

/actions (GET)
/actions/{id} (GET)
/certificates (GET, POST)
/certificates/{id} (DELETE, GET, PUT)
/certificates/{id}/actions (GET)
/certificates/{id}/actions/{action_id} (GET)
/certificates/{id}/actions/retry (POST)
/certificates/actions (GET)
/certificates/actions/{id} (GET)
/datacenters (GET)
/datacenters/{id} (GET)
/firewalls (GET, POST)
/firewalls/{id} (DELETE, GET, PUT)
/firewalls/{id}/actions (GET)
/firewalls/{id}/actions/{action_id} (GET)
/firewalls/{id}/actions/apply_to_resources (POST)
/firewalls/{id}/actions/remove_from_resources (POST)
/firewalls/{id}/actions/set_rules (POST)
/firewalls/actions (GET)
/firewalls/actions/{id} (GET)
/floating_ips (GET, POST)
/floating_ips/{id} (DELETE, GET, PUT)
/floating_ips/{id}/actions (GET)
/floating_ips/{id}/actions/{action_id} (GET)
/floating_ips/{id}/actions/assign (POST)
/floating_ips/{id}/actions/change_dns_ptr (POST)
/floating_ips/{id}/actions/change_protection (POST)
/floating_ips/{id}/actions/unassign (POST)
/floating_ips/actions (GET)
/floating_ips/actions/{id} (GET)
/images (GET)
/images/{id} (DELETE, GET, PUT)
/images/{id}/actions (GET)
/images/{id}/actions/{action_id} (GET)
/images/{id}/actions/change_protection (POST)
/images/actions (GET)
/images/actions/{id} (GET)
/isos (GET)
/isos/{id} (GET)
/load_balancer_types (GET)
/load_balancer_types/{id} (GET)
/load_balancers (GET, POST)
/load_balancers/{id} (DELETE, GET, PUT)
/load_balancers/{id}/actions (GET)
/load_balancers/{id}/actions/{action_id} (GET)
/load_balancers/{id}/actions/add_service (POST)
/load_balancers/{id}/actions/add_target (POST)
/load_balancers/{id}/actions/attach_to_network (POST)
/load_balancers/{id}/actions/change_algorithm (POST)
/load_balancers/{id}/actions/change_dns_ptr (POST)
/load_balancers/{id}/actions/change_protection (POST)
/load_balancers/{id}/actions/change_type (POST)
/load_balancers/{id}/actions/delete_service (POST)
/load_balancers/{id}/actions/detach_from_network (POST)
/load_balancers/{id}/actions/disable_public_interface (POST)
/load_balancers/{id}/actions/enable_public_interface (POST)
/load_balancers/{id}/actions/remove_target (POST)
/load_balancers/{id}/actions/update_service (POST)
/load_balancers/{id}/metrics (GET)
/load_balancers/actions (GET)
/load_balancers/actions/{id} (GET)
/locations (GET)
/locations/{id} (GET)
/networks (GET, POST)
/networks/{id} (DELETE, GET, PUT)
/networks/{id}/actions (GET)
/networks/{id}/actions/{action_id} (GET)
/networks/{id}/actions/add_route (POST)
/networks/{id}/actions/add_subnet (POST)
/networks/{id}/actions/change_ip_range (POST)
/networks/{id}/actions/change_protection (POST)
/networks/{id}/actions/delete_route (POST)
/networks/{id}/actions/delete_subnet (POST)
/networks/actions (GET)
/networks/actions/{id} (GET)
/placement_groups (GET, POST)
/placement_groups/{id} (DELETE, GET, PUT)
/pricing (GET)
/primary_ips (GET, POST)
/primary_ips/{id} (DELETE, GET, PUT)
/primary_ips/{id}/actions (GET)
/primary_ips/{id}/actions/{action_id} (GET)
/primary_ips/{id}/actions/assign (POST)
/primary_ips/{id}/actions/change_dns_ptr (POST)
/primary_ips/{id}/actions/change_protection (POST)
/primary_ips/{id}/actions/unassign (POST)
/primary_ips/actions (GET)
/primary_ips/actions/{id} (GET)
/server_types (GET)
/server_types/{id} (GET)
/servers (GET, POST)
/servers/{id} (DELETE, GET, PUT)
/servers/{id}/actions (GET)
/servers/{id}/actions/{action_id} (GET)
/servers/{id}/actions/add_to_placement_group (POST)
/servers/{id}/actions/attach_iso (POST)
/servers/{id}/actions/attach_to_network (POST)
/servers/{id}/actions/change_alias_ips (POST)
/servers/{id}/actions/change_dns_ptr (POST)
/servers/{id}/actions/change_protection (POST)
/servers/{id}/actions/change_type (POST)
/servers/{id}/actions/create_image (POST)
/servers/{id}/actions/detach_from_network (POST)
/servers/{id}/actions/detach_iso (POST)
/servers/{id}/actions/disable_backup (POST)
/servers/{id}/actions/disable_rescue (POST)
/servers/{id}/actions/enable_backup (POST)
/servers/{id}/actions/enable_rescue (POST)
/servers/{id}/actions/poweroff (POST)
/servers/{id}/actions/poweron (POST)
/servers/{id}/actions/reboot (POST)
/servers/{id}/actions/rebuild (POST)
/servers/{id}/actions/remove_from_placement_group (POST)
/servers/{id}/actions/request_console (POST)
/servers/{id}/actions/reset (POST)
/servers/{id}/actions/reset_password (POST)
/servers/{id}/actions/shutdown (POST)
/servers/{id}/metrics (GET)
/servers/actions (GET)
/servers/actions/{id} (GET)
/ssh_keys (GET, POST)
/ssh_keys/{id} (DELETE, GET, PUT)
/volumes (GET, POST)
/volumes/{id} (DELETE, GET, PUT)
/volumes/{id}/actions (GET)
/volumes/{id}/actions/{action_id} (GET)
/volumes/{id}/actions/attach (POST)
/volumes/{id}/actions/change_protection (POST)
/volumes/{id}/actions/detach (POST)
/volumes/{id}/actions/resize (POST)
/volumes/actions (GET)
/volumes/actions/{id} (GET)
/zones (GET, POST)
/zones/actions (GET)
/zones/actions/{id} (GET)
/zones/{id_or_name} (DELETE, GET, PUT)
/zones/{id_or_name}/actions (GET)
/zones/{id_or_name}/actions/change_primary_nameservers (POST)
/zones/{id_or_name}/actions/change_protection (POST)
/zones/{id_or_name}/actions/change_ttl (POST)
/zones/{id_or_name}/actions/import_zonefile (POST)
/zones/{id_or_name}/actions/{action_id} (GET)
/zones/{id_or_name}/rrsets (GET, POST)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type} (DELETE, GET, PUT)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type}/actions/add_records (POST)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type}/actions/change_protection (POST)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type}/actions/change_ttl (POST)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type}/actions/remove_records (POST)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type}/actions/set_records (POST)
/zones/{id_or_name}/rrsets/{rr_name}/{rr_type}/actions/update_records (POST)
/zones/{id_or_name}/zonefile (GET)

# Storage Boxes

/storage_box_types (GET)
/storage_box_types/{id} (GET)
/storage_boxes (GET, POST)
/storage_boxes/actions (GET)
/storage_boxes/actions/{id} (GET)
/storage_boxes/{id} (DELETE, GET, PUT)
/storage_boxes/{id}/actions (GET)
/storage_boxes/{id}/actions/change_protection (POST)
/storage_boxes/{id}/actions/change_type (POST)
/storage_boxes/{id}/actions/disable_snapshot_plan (POST)
/storage_boxes/{id}/actions/enable_snapshot_plan (POST)
/storage_boxes/{id}/actions/reset_password (POST)
/storage_boxes/{id}/actions/rollback_snapshot (POST)
/storage_boxes/{id}/actions/update_access_settings (POST)
/storage_boxes/{id}/actions/{action_id} (GET)
/storage_boxes/{id}/folders (GET)
/storage_boxes/{id}/snapshots (GET, POST)
/storage_boxes/{id}/snapshots/{snapshot_id} (DELETE, GET, PUT)
/storage_boxes/{id}/subaccounts (GET, POST)
/storage_boxes/{id}/subaccounts/{subaccount_id} (DELETE, GET, PUT)
/storage_boxes/{id}/subaccounts/{subaccount_id}/actions/change_home_directory (POST)
/storage_boxes/{id}/subaccounts/{subaccount_id}/actions/reset_subaccount_password (POST)
/storage_boxes/{id}/subaccounts/{subaccount_id}/actions/update_access_settings (POST)
```

## Quick Start

```
npm run convert -- --output openapi/hcloud.json
```

## Usage with OpenAPI Generator

[OpenAPI Generator on GitHub](https://github.com/OpenAPITools/openapi-generator)

### Validate

```
java -jar <path>/openapi-generator-cli.jar validate --input-spec openapi/hcloud.json
```

### Generate

```
java -jar <path>/openapi-generator-cli.jar generate --input-spec openapi/hcloud.json --generator-name <name> --output <path>
```
 readmeEtag: '"b962530eb9ecab391ceb782e33931d3550798a09"' readmeLastModified: Tue, 16 Dec 2025 22:08:10 GMT repositoryId: 266785930 description: >- This is the unofficial OpenAPI description of the Hetzner Cloud API. It allows automatic code generation for the hcloud API. created: '2020-05-25T13:26:36Z' updated: '2026-02-05T13:17:15Z' language: TypeScript archived: false stars: 35 watchers: 3 forks: 5 owner: MaximilianKoestler logo: https://avatars.githubusercontent.com/u/6814304?v=4 license: MIT repoEtag: '"93710ea79283e7cde25177cdfeef0a438b42d581c9508a3e14b023a783c426d9"' repoLastModified: Thu, 05 Feb 2026 13:17:15 GMT foundInMaster: true category: Testing id: faad796cc6069f33f77380c65a4256f3 - source: openapi3 tags repository: https://github.com/kota65535/openapi-merger v3: true id: 6e88483ef2d9b10f9f1f1268e7d4834b repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLW1lcmdlcgoKWWV0IGFub3RoZXIgQ0xJIHRvb2wgZm9yIG1lcmdpbmcgbXVsdGlwbGUgT3BlbkFQSSBmaWxlcyBpbnRvIGEgc2luZ2xlIGZpbGUuCgohW2J1aWxkIHdvcmtmbG93XShodHRwczovL2dpdGh1Yi5jb20va290YTY1NTM1L29wZW5hcGktbWVyZ2VyL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnltbC9iYWRnZS5zdmcpClshW05QTV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vb3BlbmFwaS1tZXJnZXIucG5nKV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vb3BlbmFwaS1tZXJnZXIvKQoKIyMgRmVhdHVyZXMKLSBTaW1pbGFyIHRvIFtzd2FnZ2VyLW1lcmdlcl0oaHR0cHM6Ly9naXRodWIuY29tL1dpbmRvbVovc3dhZ2dlci1tZXJnZXIpCi0gQ29udmVydCBhbGwgcmVtb3RlICYgVVJMIHJlZnMgaW50byBsb2NhbCByZWZzCiAgLSBUaGlzIGlzIHVzZWZ1bCBvbiBjb2RlIGdlbmVyYXRpb24sIGJlY2F1c2UgaXQgaGVscHMgW29wZW5hcGktZ2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKQogICAgdG8gb3V0cHV0IHVuaXF1ZSBtb2RlbCBjbGFzc2VzIHdpdGhvdXQgZHVwbGljYXRpb24uCi0gYCRpbmNsdWRlYCBrZXl3b3JkOiBzYW1lIGFzIGAkcmVmYCwgZXhjZXB0IGl0IG1lcmdlcyB0aGUgb2JqZWN0IHdpdGggc2libGluZyBlbGVtZW50cy4gKGAkcmVmYCBpZ25vcmVzIHRoZW0pCgojIyBVc2FnZQoKYGBgc2gKJCBucG0gaW5zdGFsbCAtZyBvcGVuYXBpLW1lcmdlcgokIG9wZW5hcGktbWVyZ2VyIC1pIG9wZW5hcGkueWFtbCAtbyBtZXJnZWQueWFtbApgYGAKCiMjICRpbmNsdWRlIGtleXdvcmQKCm9wZW5hcGktbWVyZ2VyIGludHJvZHVjZXMgdGhlIHNwZWNpYWwga2V5d29yZCBgJGluY2x1ZGVgLgpJdCBoYXMgc2ltaWxhciBzeW50YXggYXMgYCRyZWZgLCB3aGljaCB0YWtlcyBKU09OIHJlZmVyZW5jZSBhcyBpdHMgdmFsdWUuCgpgYGB5YW1sCiRpbmNsdWRlOiAncmVmZXJlbmNlIHRvIGNvbnRlbnQnCmBgYAoKVGhlIGJpZ2dlc3QgZGlmZmVyZW5jZSBpcyB0aGF0IGAkaW5jbHVkZWAgcmVwbGFjZXMgaXRzZWxmIGRpcmVjdGx5IGJ5IHRoZSByZWZlcmVuY2VkIGNvbnRlbnQsIGFsbG93aW5nIHRvIG1lcmdlIGl0cyBzaWJsaW5nIGVsZW1lbnRzLgoKCiMjIyBNZXJnZSBvYmplY3RzICYgYXJyYXlzCgpJZiBgJGluY2x1ZGVgIGlzIHVzZWQgaW4gYW4gb2JqZWN0IGFuZCB0aGVuIHJlZmVyZW5jZWQgY29udGVudCBpcyBhbiBvYmplY3QgdG9vLCB0aGV5IGFyZSBtZXJnZWQuCgotIG1haW4ueW1sCmBgYHlhbWwKb2JqZWN0OgogICRpbmNsdWRlOiBvYmplY3QueW1sCiAga2V5MzogdmFsMwpgYGAKCi0gb2JqZWN0LnltbApgYGB5YW1sCmtleTE6IHZhbDEKa2V5MjogdmFsMgpgYGAKCi0gcmVzdWx0cyBpbjoKYGBgeWFtbApvYmplY3Q6CiAga2V5MTogdmFsMQogIGtleTI6IHZhbDIKICBrZXkzOiB2YWwzCmBgYCAKCkFycmF5cyBnbyBpbiB0aGUgc2FtZSBtYW5uZXIuCgotIG1haW4ueW1sCmBgYHlhbWwKYXJyYXk6CiAgLSAkaW5jbHVkZTogYXJyYXkueW1sCiAgLSB2YWwzCmBgYAoKLSBhcnJheS55bWwKYGBgeWFtbAotIHZhbDEKLSB2YWwyCmBgYAoKLSByZXN1bHRzIGluOgpgYGB5YW1sCmFycmF5OgogIC0gdmFsMQogIC0gdmFsMgogIC0gdmFsMwpgYGAgCgpJZiB5b3Ugd2FudCBub3QgdG8gbWVyZ2UgYXJyYXlzLCB1c2UgYCRpbmNsdWRlYCBpbiBhIG5lc3RlZCBhcnJheS4KCi0gbWFpbi55bWwKYGBgeWFtbAphcnJheToKICAtIC0gJGluY2x1ZGU6IGFycmF5LnltbAogIC0gdmFsMwpgYGAKCi0gYXJyYXkueW1sCmBgYHlhbWwKLSB2YWwxCi0gdmFsMgpgYGAKCi0gcmVzdWx0cyBpbjoKYGBgeWFtbAphcnJheToKICAtIC0gdmFsMQogICAgLSB2YWwyCiAgLSB2YWwzCmBgYCAKCiMjIyBNdWx0aXBsZSAkaW5jbHVkZSBhdCBzYW1lIHBsYWNlCgpgJGluY2x1ZGVgIGNhbiBiZSB1c2VkIG11bHRpcGxlIHRpbWVzIGluIHRoZSBzYW1lIHBsYWNlIGJ5IGFwcGVuZGluZyBgI2Agd2l0aCBzb21lIElELCBhdm9pZGluZyBrZXkgZHVwbGljYXRpb24uCgpgYGB5YW1sCiRpbmNsdWRlI2ZvbzogLi9mb28ueW1sCiRpbmNsdWRlI2JhcjogLi9iYXIueW1sCmBgYAoKCiMjIyBLZXkgbW9kaWZpY2F0aW9uICYgRmlsdGVyaW5nCgpgJGluY2x1ZGVgIGlzIGNhcGFibGUgb2YgbW9kaWZpY2F0aW9uIGFuZCBmaWx0ZXJpbmcgb2YgdGhlIGtleXMgb2YgdGhlIHJlZmVyZW5jZWQgY29udGVudC4KVGhpcyBpcyB1c2VmdWwgd2hlbiB5b3Ugd2FudCB0byBhZ2dyZWdhdGUgbXVsdGlwbGUgT3BlbkFQSSBkb2N1bWVudHMgb2YgYmFja2VuZCBzZXJ2aWNlcyBpbnRvIG9uZSBmb3IgQVBJIEdhdGV3YXkuCgpUbyB1dGlsaXplIHRoaXMgZnVuY3Rpb24sIGEgY29uZmlndXJhdGlvbiBmaWxlIHNob3VsZCBiZSBnaXZlbiBieSBgLWNgIG9wdGlvbi4KVGhlIGNvbmZpZ3VyYXRpb24gZmlsZSBpcyBsaWtlIGZvbGxvd2luZzoKCmBgYHlhbWwKaW5jbHVkZToKICAjICdmb28nIGNsYXNzLCB3aGljaCBhZGQgJy92MScgcHJlZml4IHRvIGVhY2gga2V5CiAgZm9vOgogICAgcHJlZml4OiAvdjEKICAjICdiYXInIGNsYXNzLCB3aGljaCBzZWxlY3RzIG9ubHkga2V5cyBtYXRjaGluZyB0byByZWdleCAKICAjIGhlcmUgZXhjbHVkaW5nIHBhdGhzIHRoYXQgYmVnaW5zICdpbnRlcm5hbCcKICBiYXI6IAogICAgZmlsdGVyOiBeKD8hL2ludGVybmFsKS4qCmBgYAoKVXNlIGRlZmluZWQgY2xhc3MgYXMgZm9sbG93aW5nOgotIG1haW4ueW1sCmBgYHlhbWwKIyB1c2luZyBmb28gY2xhc3MKJGluY2x1ZGUuZm9vOiBwYXRocy55bWwgCiMgdXNpbmcgYmFyIGNsYXNzCiRpbmNsdWRlLmJhcjogcGF0aHMueW1sCmBgYAoKLSBwYXRocy55bWwKYGBgeWFtbAovdXNlcnM6CiAgcG9zdDoKICAgIC4uLgoKL3VzZXJzL3tpZH06CiAgZ2V0OgogICAgLi4uCgovaW50ZXJuYWwvcGV0czoKICBwb3N0OgogICAgLi4uCmBgYAoKLSByZXN1bHRzIGluOgpgYGB5YW1sCiMgZnJvbSAkaW5jbHVkZS5mb28KL3YxL3VzZXJzOgogIHBvc3Q6CiAgICAuLi4KCi92MS91c2Vycy97aWR9OgogIGdldDoKICAgIC4uLgoKL3YxL2ludGVybmFsL3BldHM6CiAgcG9zdDoKICAgIC4uLgoKIyBmcm9tICRpbmNsdWRlLmJhcgovdXNlcnM6CiAgcG9zdDoKICAgIC4uLgoKL3VzZXJzL3tpZH06CiAgZ2V0OgogICAgLi4uCmBgYAoKWW91IGNhbiBzdGlsbCB1c2UgYCNgIG5vdGF0aW9uIHRvIGF2b2lkIGtleSBjb25mbGljdHMgbGlrZSBiZWxvdy4gCgpgYGB5YW1sCiRpbmNsdWRlI2EuZm9vOiBwYXRoczEueW1sCiRpbmNsdWRlI2IuZm9vOiBwYXRoczIueW1sCmBgYAo= readmeEtag: '"2f6bd1fba948ab091412152a8319c37185f35c2b"' readmeLastModified: Fri, 15 Mar 2024 09:27:33 GMT repositoryId: 275802208 description: Merges multiple OpenAPI document files into a single file. created: '2020-06-29T12:02:45Z' updated: '2025-09-22T13:43:19Z' language: JavaScript archived: false stars: 34 watchers: 1 forks: 9 owner: kota65535 logo: https://avatars.githubusercontent.com/u/10975834?v=4 license: MIT repoEtag: '"c02a50b5e96f20477c35b790ba25002532891702ceb3bdce532f3695bfdfecf2"' repoLastModified: Mon, 22 Sep 2025 13:43:19 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/openapitools/sbt-openapi-generator v3: true repositoryMetadata: base64Readme: >- # sbt-openapi-generator

A Sbt plugin to support the OpenAPI generator project.

# Usage

Add to your `project/plugins.sbt`:

```sbt
addSbtPlugin("org.openapitools" % "sbt-openapi-generator" % "7.19.0")
```

# Configuration

Configuration based on project module is recommended way to separate specifications by modules.

You must define one of the settings `openApiInputSpec` or `openApiConfigFile` to able run plugin to generate.

Settings will be picked up from `openApiConfigFile` first if defined and then will be overwritten with module specified
settings if provided

With the next example module `generated` will be defined as:

```sbt
lazy val generated = project.in(file("generated"))
  .settings(
    openApiInputSpec := "openapi.yaml",
    openApiConfigFile := "config.yaml"
  )
```

There is a helpers to have boolean settings more readable. Instead of `Some(true)` it possible to do next:
```sbt
    openApiValidateSpec := SettingDisabled,
    openApiGenerateModelTests := SettingEnabled,
```
# Execution

To print all available languages use
```shell script
sbt openApiGenerators
```

To run template generation process
```shell script
sbt openApiGenerate
```
or per defined module
```shell script
sbt generated/openApiGenerate
```

# Settings


| Setting  | Type   | Description              |
|----------|--------|--------------------------|
| openApiGeneratorName *| `String` | The name of the generator which will handle codegen. (see \"openApiGenerators\" task)<br> Required. Can be provided as `generatorName` option of `openApiConfigFile` json config |
| openApiInputSpec *| `String` | The Open API 2.0/3.x specification location (file or url)<br> Required. Can be provided as `inputSpec` option of `openApiConfigFile` json config |
| openApiOutputDir| `String` | The output target directory into which code will be generated |
| openApiConfigFile **| `String` | Path to json configuration file<br> This setting is required with `generatorName` and `inputSpec` settings provided if sbt settings `openApiGeneratorName` and `openApiInputSpec` are absent |
| openApiAdditionalProperties | `Map[String, String]` | Sets additional properties that can be referenced by the mustache templates in the format of name=value,name=value. You can also have multiple occurrences of this option |
| openApiGlobalProperties | `Map[String, String]` |Sets specified system properties |
| openApiVerbose | `Option[Boolean]` | The verbosity of generation |
| openApiValidateSpec | `Option[Boolean]` | Whether or not an input specification should be validated upon generation |
| openApiTemplateDir | `String` | The template directory holding a custom template |
| openApiAuth | `String` | Adds authorization headers when fetching the OpenAPI definitions remotely. Pass in a URL-encoded string of name:header with a comma separating multiple values |
| openApiSkipOverwrite | `Option[Boolean]` | Specifies if the existing files should be overwritten during the generation |
| openApiPackageName | `String` | Package for generated classes (where supported) |
| openApiApiPackage | `String` | Package for generated api classes |
| openApiModelPackage | `String` | Package for generated models |
| openApiModelNamePrefix | `String` | Prefix that will be prepended to all model names |
| openApiModelNameSuffix | `String` | Suffix that will be appended to all model names |
| openApiInstantiationTypes | `Map[String, String]` | Sets instantiation type mappings |
| openApiTypeMappings | `Map[String, String]` | Sets mappings between OpenAPI spec types and generated code types |
| openApiServerVariables | `Map[String, String]` | Sets server variable for server URL template substitution, in the format of name=value,name=value. You can also have multiple occurrences of this option |
| openApiLanguageSpecificPrimitives | `List[String]` | Specifies additional language specific primitive types in the format of type1,type2,type3,type3. For example: String,boolean,Boolean,Double |
| openApiImportMappings | `Map[String, String]` | Specifies mappings between a given class and the import that should be used for that class |
| openApiInvokerPackage | `String` | Root package for generated code |
| openApiGroupId | `String` | groupId in generated pom.xml/build.sbt |
| openApiId | `String` | artifactId in generated pom.xml/build.sbt. This also becomes part of the generated library's filename |
| openApiLibrary | `String` | library template (sub-template) |
| openApiGitHost | `String` |Git host, e.g. gitlab.com |
| openApiGitUserId | `String` | Git user ID, e.g. openapitools |
| openApiGitRepoId | `String` | Git repo ID, e.g. openapi-generator |
| openApiReleaseNote | `String` | Release note, default to 'Minor update' |
| openApiHttpUserAgent | `String` | HTTP user agent, e.g. codegen_csharp_api_client, default to 'OpenAPI-Generator/{packageVersion}}/{language}' |
| openApiReservedWordsMappings | `Map[String, String]` | ]("Specifies how a reserved name should be escaped to |
| openApiIgnoreFileOverride | `String` | Specifies an override location for the .openapi-generator-ignore file. Most useful on initial generation. |
| openApiRemoveOperationIdPrefix | `Option[Boolean]` | Remove prefix of operationId, e.g. config_getId => getId |
| openApiApiFilesConstrainedTo | `List[String]` | Defines which API-related files should be generated. This allows you to create a subset of generated files (or none at all) |
| openApiModelFilesConstrainedTo | `List[String]` | Defines which model-related files should be generated. This allows you to create a subset of generated files (or none at all) |
| openApiSupportingFilesConstrainedTo | `List[String]` | Defines which supporting files should be generated. This allows you to create a subset of generated files (or none at all |
| openApiGenerateModelTests | `Option[Boolean]` | Specifies that model tests are to be generated |
| openApiGenerateModelDocumentation | `Option[Boolean]` | Defines whether or not model-related _documentation_ files should be generated |
| openApiGenerateApiTests | `Option[Boolean]` | Specifies that api tests are to be generated |
| openApiGenerateApiDocumentation | `Option[Boolean]` | Defines whether or not api-related _documentation_ files should be generated |
| openApiWithXml | `Option[Boolean]` | A special-case setting which configures some generators with XML support. In some cases, this forces json OR xml, so the default here is false |
| openApiLogToStderr | `Option[Boolean]` | To write all log messages (not just errors) to STDOUT |
| openApiEnablePostProcessFile | `Option[Boolean]` | Enable post-processing file using environment variables | \
| openApiSkipValidateSpec | `Option[Boolean]` | To skip spec validation. When true, we will skip the default behavior of validating a spec before generation |
| openApiGenerateAliasAsModel | `Option[Boolean]` | Generate model implementation for aliases to map and array schemas |
| openApiGenerateMetadata | `Option[Boolean]` | Generate metadata files used by OpenAPI Generator. This includes `.openapi-generator-ignore` and any files within `.openapi-generator`. |

# Examples

Please see [an sbt-test configuration](src/sbt-test) for examples of using the plugin.
Do not run those examples directly, please copy them to separate place first.

# Contribution and Tests

Write plugin integration tests under [src/sbt-test](src/sbt-test)

Execute next command to run tests:

```shell script
sbt scripted
```

More information about how to write and execute tests [is here](https://www.scala-sbt.org/1.x/docs/Testing-sbt-plugins.html)
 readmeEtag: '"187154c00bfc91b3037e2cf22099350c2ada490d"' readmeLastModified: Wed, 21 Jan 2026 08:29:48 GMT repositoryId: 244148241 description: null created: '2020-03-01T12:49:22Z' updated: '2026-01-21T08:29:52Z' language: Scala archived: false stars: 37 watchers: 3 forks: 41 owner: OpenAPITools logo: https://avatars.githubusercontent.com/u/37325267?v=4 license: NOASSERTION repoEtag: '"434e981fc4179d911b37b0850b59a2e2a7ad0b4cd50c5a7311e5a4cc5889a44d"' repoLastModified: Wed, 21 Jan 2026 08:29:52 GMT foundInMaster: true category: Description Validators id: b442d7afc94f3b9d9e3009c0936c0e9c - source: - openapi3 tags - openapi31 tags repository: https://github.com/redocly/redocly-cli-cookbook v3: true v3_1: true id: 2d252e6d912a2772d88fd6f022fff74f repositoryMetadata: base64Readme: >- IyBSZWRvY2x5IENMSSBDb29rYm9vawoKQSBjb21tdW5pdHkgY29sbGVjdGlvbiBvZiBydWxlc2V0cywgY29uZmlndXJhdGlvbiwgY3VzdG9tIHBsdWdpbnMgYW5kIG90aGVyIGFkZGl0aW9ucyBmb3IgW1JlZG9jbHkgQ0xJXShodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9yZWRvY2x5LWNsaSkuIFdlIGtub3cgb3VyIHVzZXJzIGhhdmUgc29tZSBncmVhdCB0aXBzLCBleGFtcGxlcywgYW5kIGNvZGUgdG8gc2hhcmUsIGFuZCB0aGlzIGlzIHRoZSBwbGFjZSB0byBkbyBqdXN0IHRoYXQuIFdlIHdvdWxkIGxvdmUgdG8gaGF2ZSB5b3VyIFtjb250cmlidXRpb25zXSgjY29udHJpYnV0aW5nKSBoZXJlIHRvbyEKCj4gWyFJTVBPUlRBTlRdCj4gUmVkb2NseSBhcmUgdGhlIHJlcG9zaXRvcnkgbWFpbnRhaW5lcnMsIGJ1dCB3ZSBjYW4ndCB0aG9yb3VnaGx5IHRlc3QgZXZlcnl0aGluZyBoZXJlLiBQbGVhc2UgYnJvd3NlLCBzaGFyZSwgYW5kIHVzZSB3aGF0IHlvdSBmaW5kIGF0IHlvdXIgb3duIHJpc2suCgpJZiB5b3UncmUgbmV3IHRvIFJlZG9jbHkgQ0xJLCBzdGFydCB3aXRoIHRoZSBbZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9yZWRvY2x5LmNvbS9kb2NzL2NsaS8pIHRvIGdldCB1cCBhbmQgcnVubmluZywgdGhlbiBjb21lIGJhY2sgaGVyZSB0byBwaWNrIG91dCBhbnkgZWxlbWVudHMgeW91IHdvdWxkIGxpa2UgdG8gcmUtdXNlIHlvdXJzZWxmLiBUbyBrZWVwIHVwIHdpdGggbmV3IGRldmVsb3BtZW50cywgZWl0aGVyIHN1YnNjcmliZSB0byB0aGUgcHJvamVjdCByZXBvc2l0b3J5LCBvciBbc2lnbiB1cCBmb3IgdGhlIFJlZG9jbHkgcHJvZHVjdCBuZXdzbGV0dGVyXShodHRwczovL3JlZG9jbHkuY29tL3Byb2R1Y3QtdXBkYXRlcy8pLgoKIyMgVXNhZ2UKClVzZSB0aGUgY29udGVudCBoZXJlIGFzIGEgc3RhcnRpbmcgcG9pbnQgZm9yIHlvdXIgb3duIHdvcmsuCgoxLiBUYWtlIGEgbG9vayBhdCB3aGF0J3MgYXZhaWxhYmxlIGluIGVhY2ggY2F0ZWdvcnksIGFuZCBwaWNrIGFueSB0aGF0IHlvdSB0aGluayBhcHBseSB0byB5b3VyIHNpdHVhdGlvbi4KCjIuIEVhY2ggc2VjdGlvbiBsaW5rcyB0byB0aGUgZG9jdW1lbnRhdGlvbiBmb3IgdGhhdCBmZWF0dXJlLCBpbmNhc2UgeW91IG5lZWQgYW4gaW50cm9kdWN0aW9uIG9yIHJlZnJlc2hlci4KCjMuIENvcHkgYW5kIHBhc3RlIHRoZSBleGFtcGxlcyB5b3Ugd2FudCB0byB1c2UgaW50byB5b3VyIG93biBzZXR1cCwgdGhlbiBlZGl0IHRoZW0gdG8gZml0IHlvdXIgb3duIG5lZWRzLgoKSWYgeW91IGNvbWUgdXAgd2l0aCBzb21ldGhpbmcgbmV3LCBwbGVhc2UgY29uc2lkZXIgc2hhcmluZyBpdCBoZXJlIGJ5IG9wZW5pbmcgYSBwdWxsIHJlcXVlc3QuCgojIyBDYXRlZ29yaWVzCgojIyMgUnVsZXNldHMKCkNvbWJpbmUgZXhpc3RpbmcgW2J1aWx0LWluIHJ1bGVzXShodHRwczovL3JlZG9jbHkuY29tL2RvY3MvY2xpL3J1bGVzL2J1aWx0LWluLXJ1bGVzLykgaW4gd2F5cyB0aGF0IHNlcnZlIGEgc3BlY2lmaWMgcHVycG9zZSwgYW5kIG1ha2UgYSBbcmVzdWFibGUgcnVsZXNldF0oaHR0cHM6Ly9yZWRvY2x5LmNvbS9kb2NzL2NsaS9ndWlkZXMvY29uZmlndXJlLXJ1bGVzLyNjcmVhdGUtYS1yZXVzYWJsZS1ydWxlc2V0KS4KCi0gW1NwZWMtY29tcGxpYW50IHJ1bGVzZXRdKHJ1bGVzZXRzL3NwZWMtY29tcGxpYW50LykKLSBbU3BvdCBjb21tb24gbWlzdGFrZXNdKHJ1bGVzZXRzL2NvbW1vbi1taXN0YWtlcykKLSBbU2VjdXJpdHkgcnVsZXNldF0ocnVsZXNldHMvc2VjdXJpdHkpIGFkZHMgc29tZSBkZWZlbnNpdmUgcnVsZXMgdG8geW91ciBkZXNjcmlwdGlvbi4KCiMjIyBDb25maWd1cmFibGUgcnVsZXMKClRoZXJlIGFyZSBzb21lIGZhbnRhc3RpYyBleGFtcGxlcyBvZiBbY29uZmlndXJhYmxlIHJ1bGVzXShodHRwczovL3JlZG9jbHkuY29tL2RvY3MvY2xpL3J1bGVzL2NvbmZpZ3VyYWJsZS1ydWxlcy8pIGluIHRoZSB3aWxkLCB3ZSBob3BlIHRoZSBsaXN0IGhlcmUgaW5zcGlyZXMgeW91IHRvIHNoYXJlIG1vcmUgb2YgeW91ciBvd24hCgotIFtCYW4gY2VydGFpbiB3b3JkcyBmcm9tIGRlc2NyaXB0aW9uc10oY29uZmlndXJhYmxlLXJ1bGVzL2Rlc2NyaXB0aW9uLWJhbm5lZC13b3Jkcy8pCi0gW1JlcXVpcmUgYGl0ZW1zYCBmaWVsZCBmb3Igc2NoZW1hcyBvZiB0eXBlIGBhcnJheWBdKGNvbmZpZ3VyYWJsZS1ydWxlcy9yZXF1aXJlZC1pdGVtcy1mb3ItYXJyYXktc2NoZW1hcy8pCi0gW0Vuc3VyZSBzZW50ZW5jZSBjYXNlIGluIG9wZXJhdGlvbiBzdW1tYXJpZXNdKGNvbmZpZ3VyYWJsZS1ydWxlcy9vcGVyYXRpb24tc3VtbWFyeS1zZW50ZW5jZS1jYXNlKQotIFtgUE9TVGAgU0hPVUxEIGRlZmluZSBgcmVxdWVzdEJvZHlgIHNjaGVtYV0oY29uZmlndXJhYmxlLXJ1bGVzL29wZXJhdGlvbi1wb3N0LXNob3VsZC1kZWZpbmUtcmVxdWVzdC1ib2R5LykKLSBbYEdFVGAgU0hPVUxEIE5PVCBkZWZpbmUgYHJlcXVlc3RCb2R5YCBzY2hlbWFdKGNvbmZpZ3VyYWJsZS1ydWxlcy9vcGVyYXRpb24tZ2V0LXNob3VsZC1ub3QtZGVmaW5lLXJlcXVlc3RCb2R5LykKLSBbYERFTEVURWAgU0hPVUxEIE5PVCBkZWZpbmUgYHJlcXVlc3RCb2R5YCBzY2hlbWFdKGNvbmZpZ3VyYWJsZS1ydWxlcy9vcGVyYXRpb24tZGVsZXRlLXNob3VsZC1ub3QtZGVmaW5lLXJlcXVlc3RCb2R5LykKLSBbSW5mbyBzZWN0aW9uIG11c3QgaGF2ZSBhIGRlc2NyaXB0aW9uXShjb25maWd1cmFibGUtcnVsZXMvaW5mby1kZXNjcmlwdGlvbikKLSBbTm8gYDxzY3JpcHQ+YCB0YWdzIGluIGRlc2NyaXB0aW9uc10oY29uZmlndXJhYmxlLXJ1bGVzL25vLXNjcmlwdCkKLSBbUGF0aHMgc2hvdWxkIG5vdCBtYXRjaCBhIHBhdHRlcm5dKGNvbmZpZ3VyYWJsZS1ydWxlcy9wYXRoLWV4Y2x1ZGVzLXBhdHRlcm4vKQotIFtBUEkgaGVhbHRoY2hlY2sgcnVsZXNdKGNvbmZpZ3VyYWJsZS1ydWxlcy9hcGktaGVhbHRoLykKLSBbU3RyaW5nIHNjaGVtYXMgbGVuZ3RoIGRlZmluZWRdKGNvbmZpZ3VyYWJsZS1ydWxlcy9zdHJpbmctc2NoZW1hcy1sZW5ndGgtZGVmaW5lZC8pCi0gW0pTT04gU2NoZW1hIG1pc2NvbmZpZ3VyYXRpb25zXShjb25maWd1cmFibGUtcnVsZXMvanNvbi1zY2hlbWEtbWlzY29uZmlndXJhdGlvbnMvKQotIFtBenVyZSBBUElNIHVuc3VwcG9ydGVkIGtleXdvcmRzXShjb25maWd1cmFibGUtcnVsZXMvYXp1cmUtYXBpbS11bnN1cHBvcnRlZC1rZXl3b3JkLykKCiMjIyBDdXN0b20gcGx1Z2lucwoKVGhlIFtjdXN0b20gcGx1Z2luXShodHRwczovL3JlZG9jbHkuY29tL2RvY3MvY2xpL2N1c3RvbS1wbHVnaW5zLykgaXMgdGhlIHVsdGltYXRlIGluIGV4dGVuc2liaWxpdHksIGJ1dCBpdCdzIGFuIGFkdmFuY2VkIGZlYXR1cmUuIFRyeSB0aGVzZSBwbHVnaW5zIGZvciBpbnNwaXJhdGlvbiBhbmQgdG8gZ2V0IHlvdSBzdGFydGVkLiBSYXRoZXIgdGhhbiBpbmNsdWRpbmcgdGhlIHdob2xlIHBsdWdpbiwgdGhlcmUgYXJlIGFsc28gc2VjdGlvbnMgZm9yIGluZGl2aWR1YWwgcnVsZXMgYW5kIGRlY29yYXRvcnMgZnVydGhlciBkb3duLgoKLSBbU29ydGluZyBwbHVnaW5dKC4vY3VzdG9tLXBsdWdpbnMvc29ydGluZykgLSBydWxlcyB0byBjaGVjayBzb3J0IG9yZGVyIGFuZCBkZWNvcmF0b3JzIHRvIHRyYW5zZm9ybSBhbiBBUEkgZGVzY3JpcHRpb24gaW50byB0aGUgY29ycmVjdCBvcmRlci4gSW5jbHVkZXMgc29ydGluZyBmb3IgdGFncywgbWV0aG9kcywgcHJvcGVydGllcyBhbmQgZW51bSB2YWx1ZXMuCgojIyMjIERlY29yYXRvcnMgKGZvciBjdXN0b20gcGx1Z2lucykKCi0gW1RhZyBzb3J0aW5nXSguL2N1c3RvbS1wbHVnaW4tZGVjb3JhdG9ycy90YWctc29ydGluZykgLSBwdXQgeW91ciB0YWdzIGxpc3QgaW4gYWxwaGFiZXRpY2FsIG9yZGVyLgotIFtTdWJzdGl0dXRlIGRhdGV0aW1lIHBsYWNlaG9sZGVycyBpbiBhbiBBUEkgZGVzY3JpcHRpb25dKC4vY3VzdG9tLXBsdWdpbi1kZWNvcmF0b3JzL3VwZGF0ZS1leGFtcGxlLWRhdGVzKSAtIHVwZGF0ZSBkYXRlcyBpbiBleGFtcGxlcyB0byB0aGUgY3VycmVudCBkYXRlLgotIFtPcGVuQUkgaXNDb25zZXF1ZW50aWFsXSguL2N1c3RvbS1wbHVnaW4tZGVjb3JhdG9ycy9vcGVuYWktaXMtY29uc2VxdWVudGlhbCkgLSBhZGQgYHgtb3BlbmFpLWlzQ29uc2VxdWVudGlhbDogdHJ1ZWAgc3BlY2lmaWNhdGlvbiBleHRlbnNpb24gdG8gR0VUIG9wZXJhdGlvbnMuCi0gW1JlbW92ZSBleHRlbnNpb25zXSguL2N1c3RvbS1wbHVnaW4tZGVjb3JhdG9ycy9yZW1vdmUtZXh0ZW5zaW9ucykgLSByZW1vdmUgYW55IGdpdmVuIFtPcGVuQVBJIEV4dGVuc2lvbnNdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCNzcGVjaWZpY2F0aW9uLWV4dGVuc2lvbnMpIGZyb20gYW4gT3BlbkFQSSBkb2N1bWVudC4KLSBbUmVtb3ZlIHVudXNlZCB0YWdzXSguL2N1c3RvbS1wbHVnaW4tZGVjb3JhdG9ycy9yZW1vdmUtdW51c2VkLXRhZ3MpIC0gcmVtb3ZlIHRhZ3MgdGhhdCBhcmUgZGVjbGFyZWQgYnV0IG5vdCB1c2VkIGJ5IGFueSBvcGVyYXRpb25zLgotIFtBenVyZSBBUElNXSguL2N1c3RvbS1wbHVnaW4tZGVjb3JhdG9ycy9henVyZS1hcGltKSAtIHJlbW92ZSBmZWF0dXJlcyB1bnN1cHBvcnRlZCBieSBBenVyZSBBUElNIHN1Y2ggYXMgZXhhbXBsZXMuCi0gW1N3YXAgc3VtbWFyeSBhbmQgZGVzY3JpcHRpb25dKC4vY3VzdG9tLXBsdWdpbi1kZWNvcmF0b3JzL3N3YXAtc3VtbWFyeS1kZXNjcmlwdGlvbikgLSBzd2FwIHRoZSBjb250ZW50cyBvZiBzdW1tYXJ5IGFuZCBkZXNjcmlwdGlvbiBmaWVsZHMgaWYgdGhleSBhcmUgdGhlIHdyb25nIHdheSByb3VuZC4KCiMjIyMgUnVsZXMgKGZvciBjdXN0b20gcGx1Z2lucykKCi0gW1ZhbGlkYXRlIE1hcmtkb3duXSguL2N1c3RvbS1wbHVnaW4tcnVsZXMvbWFya2Rvd24tdmFsaWRhdG9yKSAtIGNoZWNrIE1hcmtkb3duIGluIGRlc2NyaXB0aW9uIGZpZWxkcyBpcyB2YWxpZC4KLSBbQ2hlY2sgY29kZSBzYW1wbGVzXSguL2N1c3RvbS1wbHVnaW4tcnVsZXMvY29kZS1zYW1wbGUtY2hlY2tzKSAtIGNoZWNrIHRoYXQgYW4gZXhwZWN0ZWQgbGlzdCBvZiBjb2RlIHNhbXBsZXMgaXMgcHJlc2VudCBpbiBgeC1jb2RlLXNhbXBsZXNgIGZvciBldmVyeSBvcGVyYXRpb24uCgojIyMgTWlzY2VsbGFuZW91cyAoaW5jbHVkaW5nIHRpcHMgYW5kIHRyaWNrcykKClNoYXJlIGFueXRoaW5nIHRoYXQgZGlkbid0IGZpdCB0aGUgZXhpc3RpbmcgY2F0ZWdvcmllcyBoZXJlLgoKLSBbU2NyaXB0IHRvIHJlLW9yZGVyIGFuIEFQSSBkZXNjcmlwdGlvbl0oLi9taXNjZWxsYW5lb3VzL3Jlb3JkZXItYnVuZGxlZC1kZXNjcmlwdGlvbi1wcm9wZXJ0aWVzLykgYW5kIHB1dCB0aGUgdG9wLWxldmVsIHByb3BlcnRpZXMgaW4gYSBwYXJ0aWN1bGFyIG9yZGVyLgoKIyMgQ29udHJpYnV0aW5nCgpQbGVhc2Ugc2hhcmUgeW91ciBiZXN0IFJlZG9jbHkgQ0xJIHVzYWdlIHdpdGggdXMhIEVhY2ggaXRlbSBzaG91bGQgYmUgc2hhcmVkIGluIGl0cyBvd24gcHVsbCByZXF1ZXN0LCBmb2xsb3dpbmcgdGhlIGV4aXN0aW5nIGRpcmVjdG9yeSBzdHJ1Y3R1cmUgYW5kIGluY2x1ZGluZyB0aGUgW1JFQURNRSB0ZW1wbGF0ZV0ocmVhZG1lLXRlbXBsYXRlLm1kKSBjb3BpZWQgaW50byBlYWNoIGZvbGRlci4gRnVsbCBpbnN0cnVjdGlvbnMgYXJlIGluIHRoZSBbQ09OVFJJQlVUSU5HIGZpbGVdKENPTlRSSUJVVElORy5tZCkuCgojIyBSZXF1ZXN0cwoKSWYgdGhlcmUncyBzb21ldGhpbmcgeW91IHRoaW5rIHNob3VsZCBiZSBpbiB0aGUgY29sbGVjdGlvbiBhbmQgaXQgaXNuJ3QsIGxldCB1cyBrbm93ISBPcGVuIGFuIGlzc3VlLCBhbmQgZGVzY3JpYmUgdGhlIHByb2JsZW0geW91J2QgbGlrZSB0byBzZWUgc29sdmVkIHdpdGggUmVkb2NseSBDTEkuIFdlIGNhbid0IG1ha2UgcHJvbWlzZXMsIGJ1dCB3ZSBhcmUgcHJldHR5IHN1cmUgc29tZW9uZSBvdXQgdGhlcmUgd2lsbCBrbm93IHRoZSBhbnN3ZXIuCg== readmeEtag: '"0edeb5c62c6fee81a5914cc4cd2adf6e16c21dd2"' readmeLastModified: Wed, 23 Oct 2024 07:48:19 GMT repositoryId: 697834648 description: >- A community-created collection of configuration, plugins and techniques for getting the best from Redocly CLI in every situation. created: '2023-09-28T15:12:02Z' updated: '2026-02-03T10:46:30Z' language: JavaScript archived: false stars: 43 watchers: 13 forks: 9 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 license: MIT repoEtag: '"c5280db260dacb8b4123544856c791092e1471658cceebe9cb3c6d98560a5807"' repoLastModified: Tue, 03 Feb 2026 10:46:30 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/franeklubi/clubi v3: true repositoryMetadata: base64Readme: >- IyBDbHViaQoKIVtsb2dvLnBuZ10oLi9hc3NldHMvbG9nby5wbmcpCgojIyMgQ2x1YmkgaXMgYSBncm91cC1vcmllbnRlZCBzb2NpYWwgbWVkaWEgcGxhdGZvcm0gd3JpdHRlbiBpbiBMYXJhdmVsLgoKVmlzaXQgdXMgYXQ6IGh0dHBzOi8vY2x1YmkuZnJhbmVrbHViaS5jb20vCgotLS0KKiBbRmVhdHVyZXNdKCNmZWF0dXJlcykKKiBbRG9jdW1lbnRhdGlvbl0oI2RvY3VtZW50YXRpb24pCiogW0NvbnRyaWJ1dGluZ10oI2NvbnRyaWJ1dGluZykKLS0tCgohW3Jlc3BvbnNpdmUucG5nXSguL2Fzc2V0cy9yZXNwb25zaXZlLnBuZykKKlNob3djYXNlIG9mIHRoZSByZXNwb25zaXZlIGRlc2lnbioKCiMgRmVhdHVyZXMKKiBvcGVuIGFwaSB3aXRoIHRva2VuIGF1dGggdGhhdCdzIHByZXBhcmVkIGZvciBwb3RlbnRpYWwgYXBwIGRldmVsb3BtZW50IChnbyB0byBbZG9jdW1lbnRhdGlvbl0oI2RvY3VtZW50YXRpb24pIHRvIGxlYXJuIG1vcmUpCiogYWxsb3dzIGZvciBlYXN5IChwdWJsaWMgb3IgcHJpdmF0ZSkgZ3JvdXAgY3JlYXRpb24gYW5kIG1hbmFnZW1lbnQKKiBkaXNwbGF5cyBub3RpZmljYXRpb25zLCBzdWNoIGFzOgogICAgLSBuZXcgcG9zdCBpbiBZb3VyIGdyb3VwCiAgICAtIHNvbWVvbmUgaGFzIGxpa2VkIFlvdXIgcG9zdC9jb21tZW50CiAgICAtIHNvbWVvbmUgaGFzIGpvaW5lZCBZb3VyIGdyb3VwCiAgICAtIHNvbWVvbmUgaGFzIGxlZnQgYSBjb21tZW50IG9uIFlvdXIgcG9zdAoqIGRpc3Rpbmd1aXNoZXMgYmV0d2VlbiB1c2VycywgZ3JvdXAgYWRtaW5zLCBhbmQgc2l0ZSBhZG1pbnMKKiBpbXBsZW1lbnRzIHVzZXItc3BlY2lmaWMgZmVlZCBjb250YWluaW5nIGxhdGVzdCBwb3N0cyBmcm9tIGFsbCBncm91cHMgdGhlIHVzZXIgaGFzIGpvaW5lZAoqIGltcGxlbWVudHMgaW52aXRhdGlvbiBzeXN0ZW0gd2l0aCBzcGVjaWZpZWQgY29udHJvbHMgc3VjaCBhczogbmVjZXNzYXJ5IGFkbWluIGFwcHJvdmFsIGZvciBpbnZpdGF0aW9ucyB0byBwcml2YXRlIGdyb3VwcwoqIHJhbmtzIDEwIG1vc3QgcG9wdWxhciBncm91cHMgb24gc2l0ZQoqIGFsbG93cyBmb3Igc2VhcmNoaW5nIGdyb3VwcyBieSBuYW1lIG9yIHRoZWlyIGlkIHN0cmluZwoqIHVzZXJzIGFyZSBhYmxlIHRvIGF0dGFjaCBwaWN0dXJlcyB0byBwb3N0cy9jb21tZW50cyBhbmQgYWxzbyBsaWtlIHRoZW0KKiBjb250YWlucyBiYWRnZXMgbmV4dCB0byB1c2VybmFtZSBpbmRpY2F0aW5nIHVzZXIncyByZWxldmFuY2U6CiAgICAtIHBvc3QgYXV0aG9yCiAgICAtIGdyb3VwIGFkbWluCiAgICAtIHNpdGUgYWRtaW4KKiBpbXBsZW1lbnRzICdMb2FkIG1vcmUnIGZ1bmN0aW9uYWxpdHkgZm9yIHBvc3RzIGFuZCBjb21tZW50cwoqIHVzZXMgcmVzcG9uc2l2ZSBkZXNpZ24KCiMgRG9jdW1lbnRhdGlvbgoKRm9yIEFQSSBkb2N1bWVudGF0aW9uLCB2aXNpdCB0aGlzIGxpbms6IFtnZW5lcmF0ZWQgc3dhZ2dlciBkb2NdKGh0dHBzOi8vZ2VuZXJhdG9yLnN3YWdnZXIuaW8vP3VybD1odHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vZnJhbmVrbHViaS9jbHViaS9tYXN0ZXIvZG9jcy9vcGVuYXBpLnlhbWwpCgojIENvbnRyaWJ1dGluZwoKQ2x1Ymkgd2VsY29tZXMgY29udHJpYnV0aW9ucyBvZiBhbnkga2luZCA6KQoKVG8gc3RhcnQgY29udHJpYnV0aW5nIGNsb25lIHRoZSByZXBvc2l0b3J5IGFuZCBydW4gYC4vZGV2X3NldHVwLnNoYCB0byBoZWxwIGdldCBZb3Ugc3RhcnRlZC4gVGhpcyBzY3JpcHQgd2lsbDoKKiBpbnN0YWxsIGNvbXBvc2VyIGFuZCBucG0gcGFja2FnZXMKKiBjb21waWxlIGpzIGFzc2V0cwoqIGNvcHkgdGhlIC5lbnYuZXhhbXBsZSB0byAuZW52CiogZ2VuZXJhdGUgc2VjdXJlIGtleSB3aXRoIGFydGlzYW4KKiBvcGVuIC5lbnYgaW4gdmltIGZvciBZb3UgdG8gbW9kaWZ5IGl0Cg== readmeEtag: '"02f9329c336de4dfe43fca97896180f7ea8bd184"' readmeLastModified: Tue, 06 Dec 2022 19:49:21 GMT repositoryId: 280255962 description: A group-oriented social media platform written in Laravel and Vue created: '2020-07-16T20:52:54Z' updated: '2025-09-16T04:31:06Z' language: PHP archived: false stars: 32 watchers: 4 forks: 2 owner: franeklubi logo: https://avatars.githubusercontent.com/u/33394849?v=4 license: MIT repoEtag: '"24aa7ae35aa8d7da4d2915bdfd2080695555ac93f40ccde703b130516151d7d2"' repoLastModified: Tue, 16 Sep 2025 04:31:06 GMT foundInMaster: true category: - Testing - Server Implementations id: 45fe333e65955184af9db3099d28c59b - source: openapi3 tags repository: https://github.com/namuan/http-rider v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9uYW11YW4vaHR0cC1yaWRlci9yYXcvbWFzdGVyL3Jlc291cmNlcy9pbWFnZXMvaHR0cHJpZGVyLXJlYWRtZS5wbmciIHdpZHRoPSIxMjhweCIvPgo8L3A+CjxoMSBhbGlnbj0iY2VudGVyIj5IdHRwUmlkZXIgOjogU2ltcGxlIGFuZCBQb3dlcmZ1bCBDcm9zcy1QbGF0Zm9ybSBBUEkgQ2xpZW50PC9oMT4KClshW01hY09TIFJlbGVhc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3YvcmVsZWFzZS9uYW11YW4vaHR0cC1yaWRlci1vc3g/bGFiZWw9bWFjb3MpXShodHRwczovL2dpdGh1Yi5jb20vbmFtdWFuL2h0dHAtcmlkZXItb3N4L3JlbGVhc2VzL2xhdGVzdCkgWyFbV2luZG93cyBSZWxlYXNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3JlbGVhc2UvbmFtdWFuL2h0dHAtcmlkZXItd2luP2xhYmVsPXdpbmRvd3MpXShodHRwczovL2dpdGh1Yi5jb20vbmFtdWFuL2h0dHAtcmlkZXItd2luL3JlbGVhc2VzL2xhdGVzdCkgWyFbTGludXggUmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvdi9yZWxlYXNlL25hbXVhbi9odHRwLXJpZGVyP2xhYmVsPUxpbnV4KV0oaHR0cHM6Ly9naXRodWIuY29tL25hbXVhbi9odHRwLXJpZGVyL3JlbGVhc2VzL2xhdGVzdCkgWyFbR2l0SHViIGxpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvbmFtdWFuL2h0dHAtcmlkZXIuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL25hbXVhbi9odHRwLXJpZGVyL2Jsb2IvbWFzdGVyL0xJQ0VOU0UpIFshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kuY29tL25hbXVhbi9odHRwLXJpZGVyLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL25hbXVhbi9odHRwLXJpZGVyKSBbIVtHaXRIdWIgbGFzdCBjb21taXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhc3QtY29tbWl0L25hbXVhbi9odHRwLXJpZGVyKV0oaHR0cHM6Ly9naXRodWIuY29tL25hbXVhbi9odHRwLXJpZGVyL2NvbW1pdHMvbWFzdGVyKSBbIVtHaXRIdWIgY29tbWl0IGFjdGl2aXR5IHRoZSBwYXN0IHdlZWssIDQgd2Vla3MsIHllYXJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2NvbW1pdC1hY3Rpdml0eS95L25hbXVhbi9odHRwLXJpZGVyKV0oaHR0cHM6Ly9naXRodWIuY29tL25hbXVhbi9odHRwLXJpZGVyL2NvbW1pdHMvbWFzdGVyKSBbIVtUd2l0dGVyIEZvbGxvd10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90d2l0dGVyL2ZvbGxvdy9kZXNrcmlkZXJzX3R3dC5zdmc/c3R5bGU9c29jaWFsJmxhYmVsPUZvbGxvdyldKGh0dHBzOi8vdHdpdHRlci5jb20vZGVza3JpZGVyc190d3QpIFshW0dpdHRlcl0oaHR0cHM6Ly9iYWRnZXMuZ2l0dGVyLmltL2h0dHAtcmlkZXIvY29tbXVuaXR5LnN2ZyldKGh0dHBzOi8vZ2l0dGVyLmltL2h0dHAtcmlkZXIvY29tbXVuaXR5P3V0bV9zb3VyY2U9YmFkZ2UmdXRtX21lZGl1bT1iYWRnZSZ1dG1fY2FtcGFpZ249cHItYmFkZ2UpCgpbIVtIdHRwUmlkZXIgSW50cm9dKGh0dHBzOi8vaW1nLnlvdXR1YmUuY29tL3ZpL3JXbXZ3VnV1TjZJLzAuanBnKV0oaHR0cHM6Ly93d3cueW91dHViZS5jb20vd2F0Y2g/dj1yV212d1Z1dU42SSkKCiMjIyBEb3dubG9hZHMKCkRvd25sb2FkcyBhcmUgY3VycmVudGx5IGF2YWlsYWJsZSBmb3IgTWFjT1MgYW5kIFdpbmRvd3MgaG93ZXZlciBpdCBpcyBzaW1wbGUgdG8gcnVuIGZyb20gc291cmNlIGlmIHlvdSBwcmVmZXIgdGhhdC4KClNlZSBbaHR0cHM6Ly93d3cuaHR0cHJpZGVyLmNvbS9kb2NzL2dldHRpbmctc3RhcnRlZC9pbnN0YWxsYXRpb24vXShodHRwczovL25hbXVhbi5naXRodWIuaW8vaHR0cC1yaWRlci1kb2NzL2RvY3MvZ2V0dGluZy1zdGFydGVkL2luc3RhbGxhdGlvbi8pIGZvciBpbnN0cnVjdGlvbnMgZm9yIGRpZmZlcmVudCBvcGVyYXRpbmcgc3lzdGVtcy4KCiMjIyBJbnN0YWxsYXRpb24gZnJvbSBTb3VyY2UKClRvIGluc3RhbGwgSHR0cFJpZGVyIGZyb20gc291cmNlOgoKMS4gKipDbG9uZSB0aGUgcmVwb3NpdG9yeSBvciBkb3dubG9hZCB0aGUgWklQIGZpbGU6KioKCiAgIGBgYGJhc2gKICAgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9uYW11YW4vaHR0cC1yaWRlci5naXQKICAgY2QgaHR0cC1yaWRlcgogICBgYGAKCjIuICoqUnVuIHRoZSBpbnN0YWxsYXRpb24gc2NyaXB0IChtYWNPUyk6KioKCiAgIGBgYGJhc2gKICAgLi9pbnN0YWxsLmNvbW1hbmQKICAgYGBgCgogICBUaGlzIHNjcmlwdCB3aWxsOgoKICAgLSBJbnN0YWxsIGB1dmAgKGlmIG5vdCBhbHJlYWR5IGluc3RhbGxlZCkKICAgLSBTZXQgdXAgdGhlIHZpcnR1YWwgZW52aXJvbm1lbnQgYW5kIGRlcGVuZGVuY2llcwogICAtIEluc3RhbGwgdGhlIGFwcGxpY2F0aW9uIGluIHlvdXIgYH4vQXBwbGljYXRpb25zYCBmb2xkZXIKCjMuICoqQWx0ZXJuYXRpdmUgbWFudWFsIGluc3RhbGxhdGlvbjoqKgogICBgYGBiYXNoCiAgIG1ha2UgaW5zdGFsbCAgIyBJbnN0YWxsIGRlcGVuZGVuY2llcyBhbmQgcHJlLWNvbW1pdCBob29rcwogICBtYWtlIHJ1biAgICAgICMgUnVuIHRoZSBhcHBsaWNhdGlvbgogICBgYGAKCiMjIyBGZWF0dXJlcwoK8J+agCBGZWF0dXJlIGNvbXBsZXRlIHwg8J+ZiCBFeHBlcmltZW50YWwgfCDwn5OdIEluIGRldmVsb3BtZW50IHwg4piB77iP77iPIFBsYW5uZWQKCi0tLQoK8J+agCBNYWtpbmcgYSBzaW1wbGUgcmVxdWVzdCAtIFtEZW1vXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PWZ3OGpNalFwZnk4KQoK8J+agCBBZGRpbmcgcXVlcnkgcGFyYW0sIGhlYWRlciwgcmVxdWVzdCBib2R5Cgrwn5qAIFJ1bm5pbmcgbXVsdGlwbGUgQVBJcy9SZWFycmFuZ2luZyBBUElzCgrwn5qAIFRhZ3MvTGFiZWxzIGFuZCBTZWFyY2ggYW5kIGZpbHRlcmluZwoK8J+agCBIdHRwIHJlcXVlc3RzIGhpc3RvcnkKCvCfmoAgUHJvamVjdHMgc3VwcG9ydAoK8J+agCBFbnZpcm9ubWVudHMgYW5kIHVzaW5nIGVudmlyb25tZW50IHZhcmlhYmxlcwoK8J+agCBJbXBvcnRpbmcvRXhwb3J0aW5nIGVudmlyb25tZW50cwoK8J+agCBVc2luZyBGYWtlIG9yIHJhbmRvbSBkYXRhIGdlbmVyYXRvcnMKCvCfmoAgVXNpbmcgdmFyaWFibGVzIGJldHdlZW4gQVBJIHJlcXVlc3RzCgrwn5qAIEFzc2VydGlvbnMKCvCfmoAgTW9ja2luZyByZXNwb25zZXMKCioqRXhwb3J0ZXJzKioKCvCfmoAgRXhwb3J0IFtQbGFudFVNTF0oaHR0cHM6Ly90d2l0dGVyLmNvbS9wbGFudHVtbCkgU2VxdWVuY2UgZGlhZ3JhbSAtIFtEZW1vXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PTRBc3JfNGlPeFVNKQoK8J+agCBFeHBvcnQgW1Nsb3cgQ29va2VyXShodHRwczovL2dpdGh1Yi5jb20vYnVveWFudGlvL3Nsb3dfY29va2VyKSBwZXJmb3JtYW5jZSB0ZXN0cwoK8J+agCBFeHBvcnQgW0xvY3VzdC5pb10oaHR0cHM6Ly9sb2N1c3QuaW8pIHBlcmZvcm1hbmNlIHRlc3RzIC0gW0RlbW9dKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9N3pxY1ltWklkVnMpCgrwn5qAIEV4cG9ydCBQeXRob24gY29kZSB1c2luZyByZXF1ZXN0cyBtb2R1bGUKCvCfmoAgRXhwb3J0IE1hcmtkb3duIGZvciBkb2N1bWVudGluZyBBUEkgcmVxdWVzdHMvcmVzcG9uc2VzCgrwn5qAIEV4cG9ydCBbQXBpY2tsaV0oaHR0cHM6Ly9naXRodWIuY29tL2FwaWNrbGkvYXBpY2tsaSkgZnVuY3Rpb25hbCB0ZXN0cyBbRGVtb10oaHR0cHM6Ly9kZXNrcmlkZXJzLmRldi9nZW5lcmF0aW5nLWJkZC10ZXN0cy13aXRoLWh0dHByaWRlci8pCgrwn5qAIEV4cG9ydCBjdXJsIHJlcXVlc3RzCgrwn5qAIEV4cG9ydCBbTWVybWFpZEpTXShodHRwczovL21lcm1haWRqcy5naXRodWIuaW8vKSBTZXF1ZW5jZSBkaWFncmFtCgrwn5qAIEV4cG9ydCBbUnVuc2NvcGVdKGh0dHBzOi8vd3d3LnJ1bnNjb3BlLmNvbSkgbW9uaXRvcmluZyB0ZXN0cyBbRGVtb10oaHR0cHM6Ly9kZXNrcmlkZXJzLmRldi9nZW5lcmF0aW5nLXJ1bnNjb3BlLW1vbml0b3JpbmctdGVzdHMvKQoK8J+agCBFeHBvcnQgW09wZW5BcGkgVjNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kKSBkb2N1bWVudAoK8J+agCBFeHBvcnQgW1Jlc3RBc3N1cmVkXShodHRwOi8vcmVzdC1hc3N1cmVkLmlvKSBmdW5jdGlvbmFsIHRlc3RzCgoqKkltcG9ydGVycyoqCgrwn5qAIEltcG9ydCBbT3BlbkFwaSBWM10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjAubWQpIGRvY3VtZW50Cgrwn5qAIEltcG9ydCBQb3N0bWFuIGNvbGxlY3Rpb25zCgojIyMgRGV2ZWxvcG1lbnQKClJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQgdG8gc2V0LXVwIHZpcnR1YWwgZW52aXJvbm1lbnQgYW5kIGluc3RhbGwgcHJlLWNvbW1pdCBob29rcwoKYGBgCiQgbWFrZSBpbnN0YWxsCmBgYAoKVGhlbiBgbWFrZSBydW5gIHNob3VsZCBzdGFydHVwIHRoZSBhcHBsaWNhdGlvbi4KCmBgYAokIG1ha2UgcnVuCmBgYAoKWW91IGNhbiBydW4gYG1ha2UgaGVscGAgb3IganVzdCBgbWFrZWAgdG8gZGlzcGxheSBsaXN0IG9mIGF2YWlsYWJsZSBjb21tYW5kcy4K readmeEtag: '"6bdfa73890ee7843e263e4417cd54a05b7ff12a3"' readmeLastModified: Sun, 21 Sep 2025 08:12:39 GMT repositoryId: 196688038 description: Simple and Powerful desktop client for working with JSON APIs created: '2019-07-13T06:41:18Z' updated: '2025-12-20T21:19:51Z' language: Python archived: false stars: 31 watchers: 1 forks: 8 owner: namuan logo: https://avatars.githubusercontent.com/u/575441?v=4 license: MIT repoEtag: '"2d62e63e3d0373f6b9b5e7aa81c06a02b72cd884d1ed20eb30219212a2c542e1"' repoLastModified: Sat, 20 Dec 2025 21:19:51 GMT foundInMaster: true category: SDK id: f0ed41f04fc11f264ffb1b4221371aec - source: openapi3 tags repository: https://github.com/hei-school/hei-admin-api v3: true repositoryMetadata: base64Readme: >- VGhpcyBwcm9qZWN0IGJvdGggc3BlY2lmaWVzIHRoZQpbSEVJIEFkbWluIEFQSV0oaHR0cHM6Ly9wZXRzdG9yZS5zd2FnZ2VyLmlvLz91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2hlaS1zY2hvb2wvaGVpLWFkbWluLWFwaS9kZXYvZG9jL2FwaS55bWwpCmFuZCBpbXBsZW1lbnRzIGl0IGluIEphdmEuCgpbUmVsZWFzZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9oZWktc2Nob29sL2hlaS1hZG1pbi1hcGkvcmVsZWFzZXMpIGFyZSBwdWJsaXNoZWQgW2hlcmVdKGh0dHBzOi8vZ2FsbGVyeS5lY3IuYXdzL3E2aTZ5NW80L2hlaS1hZG1pbi1hcGkpIGFzIERvY2tlciBpbWFnZXMuIEZlZWwgZnJlZSB0byB1c2UgdGhlbS4KCldlIHdlbGNvbWUgW2NvbnRyaWJ1dGlvbnNdKGh0dHBzOi8vZ2l0aHViLmNvbS9oZWktc2Nob29sL2hlaS1hZG1pbi1hcGkvYmxvYi9kZXYvQ09OVFJJQlVUSU5HLm1kKS4K readmeEtag: '"acfd1751707c93d0284459681d3566f9b375948c"' readmeLastModified: Fri, 16 Aug 2024 08:38:47 GMT repositoryId: 393620909 description: RESTful API for managing a school created: '2021-08-07T08:06:01Z' updated: '2026-02-05T12:35:07Z' language: Java archived: false stars: 30 watchers: 0 forks: 126 owner: hei-school logo: https://avatars.githubusercontent.com/u/88576095?v=4 license: MIT repoEtag: '"a316e0908f5671e125395b77f8b436e70869ebdf30f39136eaa748556fde2dc1"' repoLastModified: Thu, 05 Feb 2026 12:35:07 GMT foundInMaster: true category: Code Generators id: bf98240ea893afa32efc180fb73dd5b6 - source: openapi3 tags repository: https://github.com/outofcoffee/vertx-oas v3: true repositoryMetadata: base64Readme: >- IyB2ZXJ0eC1vYXMKCkdlbmVyYXRlcyBhIFN3YWdnZXIvT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZyb20gYSBWZXJ0LnggV2ViIFJvdXRlci4KCiMjIFVzYWdlCgpBc3N1bWluZyB5b3UncmUgdXNpbmcgVmVydC54IFdlYiwgdXNlIGl0IHdpdGggeW91ciBgUm91dGVyYCBhcyBmb2xsb3dzOgoKYGBgamF2YQovLyB5b3VyIG5vcm1hbCBWZXJ0LnggV2ViIFJvdXRlciB3aXRoIHBhdGhzIGV0Yy4KUm91dGVyIHJvdXRlciA9IFJvdXRlci5yb3V0ZXIodmVydHgpOwpyb3V0ZXIucG9zdCgiL3VzZXJzIikuaGFuZGxlciggcm91dGluZ0NvbnRleHQgLT4geyAvKiBldGMuLi4gKi8gfSk7CnJvdXRlci5nZXQoIi91c2Vycy86dXNlcklkIikuaGFuZGxlciggcm91dGluZ0NvbnRleHQgLT4geyAvKiBldGMuLi4gKi8gfSk7CgovLyBwdWJsaXNoIHRoZSBTd2FnZ2VyL09wZW5BUEkgc3BlY2lmaWNhdGlvbiB0byBhIFVSTApSb3V0ZXJTcGVjR2VuZXJhdG9yLnB1Ymxpc2hBcGlEb2NzKHJvdXRlciwgIi9hcGkvc3BlYyIpOwpgYGAKCkluIHRoaXMgZXhhbXBsZSwgdGhlIHNwZWNpZmljYXRpb24gaXMgcHVibGlzaGVkIHRvIGAvYXBpL3NwZWNgLgoKWW91IGNhbiBvYnRhaW4gWUFNTCBvciBKU09OIHZlcnNpb25zIG9mIHRoZSBzcGVjaWZpY2F0aW9uIGJ5IGFkZGluZyB0aGUgYXBwcm9wcmlhdGUgZmlsZSBleHRlbnNpb24uCgotLS0KCkZvciBleGFtcGxlLCBmZXRjaGluZyBgL2FwaS9zcGVjLnlhbWxgIHdvdWxkIHByb2R1Y2U6CgpgYGB5YW1sCm9wZW5hcGk6ICIzLjAuMSIKaW5mbzoKICB0aXRsZTogIlZlcnQueCBBUElzIgogIGRlc2NyaXB0aW9uOiAiVGhpcyBzcGVjaWZpY2F0aW9uIHdhcyBnZW5lcmF0ZWQgZnJvbSBhIFZlcnQueCBXZWIgUm91dGVyLiIKcGF0aHM6CiAgL3VzZXJzOgogICAgcG9zdDoKICAgICAgcGFyYW1ldGVyczogW10KICAvdXNlcnMve3VzZXJJZH06CiAgICBnZXQ6CiAgICAgIHBhcmFtZXRlcnM6CiAgICAgIC0gbmFtZTogInVzZXJJZCIKICAgICAgICByZXF1aXJlZDogdHJ1ZQogICAgICAgIGFsbG93RW1wdHlWYWx1ZTogZmFsc2UKYGBgCgojIyBMaW1pdGF0aW9ucwoKLSBEb2Vzbid0IHVuZGVyc3RhbmQgcmVnZXggcGF0aHMKLSBEb2Vzbid0IGtub3cgdGhlIHR5cGUgb2YgcGF0aCBwYXJhbWV0ZXJzCi0gRG9lc24ndCBrbm93IGFib3V0IHJlcXVlc3Qgb3IgcmVzcG9uc2UgYm9keSBtb2RlbHMKCiMjIFVzZSBpbiB5b3VyIHByb2plY3QKCiMjIyBNYXZlbgoKQWRkIHJlcG9zaXRvcnk6CgpgYGB4bWwKPHByb2plY3Q+Ci4uLgogIDxyZXBvc2l0b3JpZXM+CiAgICA8cmVwb3NpdG9yeT4KICAgICAgPGlkPmdhdGVoaWxsc29mdHdhcmUtc25hcHNob3RzPC9pZD4KICAgICAgPG5hbWU+R2F0ZWhpbGwgU29mdHdhcmUgU25hcHNob3RzPC9uYW1lPgogICAgICA8dXJsPmh0dHBzOi8vczMtZXUtd2VzdC0xLmFtYXpvbmF3cy5jb20vZ2F0ZWhpbGxzb2Z0d2FyZS1tYXZlbi9zbmFwc2hvdHM8L3VybD4KICAgIDwvcmVwb3NpdG9yeT4KICAgIDxyZXBvc2l0b3J5PgogICAgICA8aWQ+c29uYXR5cGUtc25hcHNob3RzPC9pZD4KICAgICAgPG5hbWU+U29uYXR5cGUgT1NTIFNuYXBzaG90czwvbmFtZT4KICAgICAgPHVybD5odHRwczovL29zcy5zb25hdHlwZS5vcmcvY29udGVudC9yZXBvc2l0b3JpZXMvc25hcHNob3RzPC91cmw+CiAgICA8L3JlcG9zaXRvcnk+CiAgPC9yZXBvc2l0b3JpZXM+Ci4uLgo8L3Byb2plY3Q+CmBgYAogICAgCkFkZCBkZXBlbmRlbmN5OgoKYGBgeG1sCjxkZXBlbmRlbmNpZXM+CiAgICAuLi4KICAgIDxkZXBlbmRlbmN5PgogICAgICAgIDxncm91cElkPmNvbS5nYXRlaGlsbC52ZXJ0eC1vYXM8L2dyb3VwSWQ+CiAgICAgICAgPGFydGlmYWN0SWQ+dmVydHgtb2FzPC9hcnRpZmFjdElkPgogICAgICAgIDx2ZXJzaW9uPjEuMC4xPC92ZXJzaW9uPgogICAgPC9kZXBlbmRlbmN5PgogICAgLi4uCjwvZGVwZW5kZW5jaWVzPgpgYGAKCiMjIyBHcmFkbGUKCkFkZCByZXBvc2l0b3J5OgoKICAgIHJlcG9zaXRvcmllcyB7CiAgICAgICAgbWF2ZW4gewogICAgICAgICAgICB1cmwgJ2h0dHBzOi8vczMtZXUtd2VzdC0xLmFtYXpvbmF3cy5jb20vZ2F0ZWhpbGxzb2Z0d2FyZS1tYXZlbi9zbmFwc2hvdHMnCiAgICAgICAgfQogICAgICAgIG1hdmVuIHsKICAgICAgICAgICAgdXJsICdodHRwczovL29zcy5zb25hdHlwZS5vcmcvY29udGVudC9yZXBvc2l0b3JpZXMvc25hcHNob3RzJwogICAgICAgIH0KICAgIH0KICAgIApBZGQgZGVwZW5kZW5jeToKCiAgICBjb21waWxlICdjb20uZ2F0ZWhpbGwudmVydHgtb2FzOnZlcnR4LW9hczoxLjAuMScgIAoKIyMgQnVpbGQKCklmIHlvdSdkIGxpa2UgdG8gYnVpbGQgdGhlIGNvZGUgbG9jYWxseSwgZm9sbG93IHRoZXNlIGluc3RydWN0aW9ucy4KCiMjIyBQcmVyZXF1aXNpdGVzCgotIEpESzgKCiMjIyBDb21waWxlIGFuZCB0ZXN0CgogICAgLi9ncmFkbGV3IGNsZWFuIGJ1aWxkCgojIyMgUHVibGlzaAoKUHVibGlzaCB0byBsb2NhbCBNYXZlbiByZXBvc2l0b3J5OgoKICAgIC4vZ3JhZGxldyBwdWJsaXNoVG9NYXZlbkxvY2FsCgpQdWJsaXNoIHRvIHJlbW90ZSBNYXZlbiByZXBvc2l0b3J5OgoKICAgIC4vZ3JhZGxldyBwdWJsaXNoCgojIyBDb250cmlidXRpbmcKCiogUHVsbCByZXF1ZXN0cyBhcmUgd2VsY29tZS4KKiBQbGVhc2UgcnVuIGBrdGxpbnRgIG9uIHlvdXIgYnJhbmNoLgoKIyMgQXV0aG9yCgpQZXRlIENvcm5pc2ggKG91dG9mY29mZmVlQGdtYWlsLmNvbSkK readmeEtag: '"c0adb9fa3d4ca39583c24d35776d56936c0cee2a"' readmeLastModified: Sun, 04 Mar 2018 17:57:03 GMT repositoryId: 118284639 description: Generates a Swagger/OpenAPI specification from a Vert.x Web Router. created: '2018-01-20T22:08:13Z' updated: '2025-03-03T08:13:08Z' language: Kotlin archived: false stars: 30 watchers: 5 forks: 4 owner: outofcoffee logo: https://avatars.githubusercontent.com/u/1248633?v=4 repoEtag: '"da23116d879cd0d9fe64dfc8d5cde0c2e240f1609bb96baed9530144d6bc0f51"' repoLastModified: Mon, 03 Mar 2025 08:13:08 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 0b55bd32d15d8b2237f4f34d7fc0335e - source: openapi3 tags repository: https://github.com/mermade/openapi-extract v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWV4dHJhY3QKCiFbY2ldKGh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL29wZW5hcGktZXh0cmFjdC93b3JrZmxvd3MvY2kvYmFkZ2Uuc3ZnKQoKRXh0cmFjdCBwYXRocywgb3BlcmF0aW9ucywgcGFyYW1ldGVycywgc2NoZW1hcyBldGMgZnJvbSBPcGVuQVBJL1N3YWdnZXIgZGVmaW5pdGlvbnMuCgpXb3JrcyB3aXRoIE9wZW5BUEkvU3dhZ2dlciAyLjAgYW5kIDMueCBkZWZpbml0aW9ucy4KCmBgYApVc2FnZTogb3BlbmFwaS1leHRyYWN0IFtvcHRpb25zXSB7aW5maWxlfSBbe291dGZpbGV9XQoKT3B0aW9uczoKICAtaCwgLS1oZWxwICAgICAgICAgICAgIFNob3cgaGVscCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbYm9vbGVhbl0KICAtLXZlcnNpb24gICAgICAgICAgICAgIFNob3cgdmVyc2lvbiBudW1iZXIgICAgICAgICAgICAgICAgICAgICAgICAgICBbYm9vbGVhbl0KICAtLW9wZW5haSAgICAgICAgICAgICAgIG1ha2UgdGhlIGRlZmluaXRpb24gT3BlbkFJIGNvbXBsaWFudCAgICAgICAgICBbYm9vbGVhbl0KICAtLXNlcnZlciAgICAgICAgICAgICAgIGluY2x1ZGUgc2VydmVyIGluZm9ybWF0aW9uICAgICAgICAgICAgICAgICAgICBbYm9vbGVhbl0KICAtLXNoYXJkICAgICAgICAgICAgICAgIHNoYXJkIHRoZSBpbnB1dCB0byBhbiBvdXRwdXQgZGlyZWN0b3J5ICAgICAgICAgW3N0cmluZ10KICAtcCwgLS1wYXRoICAgICAgICAgICAgIHRoZSBwYXRoIHRvIGV4dHJhY3QgICAgICAgICAgICAgICAgICAgICAgICAgICAgW3N0cmluZ10KICAtbywgLS1vcGVyYXRpb25pZCAgICAgIHRoZSBvcGVyYXRpb25JZHMgdG8gZXh0cmFjdCAgICAgICAgICAgICAgICAgICAgIFthcnJheV0KICAtbSwgLS1tZXRob2QgICAgICAgICAgIHRoZSBtZXRob2QgdG8gZXh0cmFjdCBmb3IgdGhlIGdpdmVuIHBhdGggICAgICAgW3N0cmluZ10KICAtaSwgLS1pbmZvICAgICAgICAgICAgIGNvcHkgZnVsbCBpbmZvIG9iamVjdCwgb3RoZXJ3aXNlIG1pbmltYWwgICAgICBbYm9vbGVhbl0KICAtZCwgLS1yZW1vdmVEb2NzICAgICAgIHJlbW92ZSBhbGwgZXh0ZXJuYWxEb2NzIHByb3BlcnRpZXMgICAgICAgICAgICBbYm9vbGVhbl0KICAtciwgLS1yZW1vdmVFeGFtcGxlcyAgIHJlbW92ZSBhbGwgZXhhbXBsZS9leGFtcGxlcyBwcm9wZXJ0aWVzICAgICAgICBbYm9vbGVhbl0KICAteCwgLS1yZW1vdmVFeHRlbnNpb25zIHJlbW92ZSBhbGwgeC0gZXh0ZW5zaW9uIHByb3BlcnRpZXMgICAgICAgICAgICBbYm9vbGVhbl0KICAtcywgLS1zZWN1cml0eSAgICAgICAgIGluY2x1ZGUgc2VjdXJpdHkgaW5mb3JtYXRpb24gICAgICAgICAgICAgICAgICBbYm9vbGVhbl0KICAtdiwgLS12ZXJib3NlICAgICAgICAgIGluY3JlYXNlIHZlcmJvc2l0eSAgICAgICAgICAgICAgICAgICAgICAgICAgICBbYm9vbGVhbl0KYGBgCgpvcgoKYGBgamF2YXNjcmlwdApjb25zdCBvcGVuYXBpRXh0cmFjdG9yID0gcmVxdWlyZSgnb3BlbmFwaS1leHRyYWN0Jyk7CmNvbnN0IG9wdGlvbnMgPSB7fTsKLy8gb3B0aW9ucy5wYXRoID0gJy4uLic7Ci8vIG9wdGlvbnMubWV0aG9kID0gJy4uLic7Ci8vIG9wdGlvbnMub3BlcmF0aW9uaWQgPSBbJy4uLiddOwpjb25zdCByZXMgPSBvcGVuYXBpRXh0cmFjdG9yLmV4dHJhY3Qob2JqLCBvcHRpb25zKTsKCmNvbnN0IG1hcCA9IG9wZW5hcGlFeHRyYWN0b3Iuc2hhcmQob2JqLCBvcHRpb25zKTsKYGBgCgpUaGUgYG9wdGlvbnNgIG9iamVjdCB0YWtlcyB0aGUgc2FtZSB2YWx1ZXMgYXMgdGhlIENMSSwgZm9yIHRoZXNlIGtleXMgYW5kIGRlZmF1bHQgdmFsdWVzOgoKKiAgIHBhdGggPSAnJwoqICAgbWV0aG9kID0gJycKKiAgIGluZm8gPSBmYWxzZQoqICAgb3BlbmFpID0gZmFsc2UKKiAgIHJlbW92ZURvY3MgPSBmYWxzZQoqICAgcmVtb3ZlRXhhbXBsZXMgPSBmYWxzZQoqICAgcmVtb3ZlRXh0ZW5zaW9ucyA9IGZhbHNlCiogICBzZXJ2ZXIgPSBmYWxzZQoqICAgc2VjdXJpdHkgPSBmYWxzZQoqICAgb3BlcmF0aW9uaWQgPSBbXQoKIyMgT3BlbkFJIGNvbXBsaWFudCBtb2RlCgpUaGlzIG9wdGlvbiB0dXJucyBvbiB0aGUgZm9sbG93aW5nIHJ1bGVzOgoKMS4gVGhlIGBkZXNjcmlwdGlvbmAgcHJvcGVydGllcyBtdXN0IGhhdmUgYSBtYXhpbXVtIGxlbmd0aCBvZiAzMDAgY2hhcmFjdGVycwo= readmeEtag: '"5f353389f7717a81f71aa1c92d7ed45243239dd2"' readmeLastModified: Thu, 20 Apr 2023 05:29:05 GMT repositoryId: 118493814 description: Extract single paths/operations from OpenAPI definitions created: '2018-01-22T17:56:20Z' updated: '2026-01-06T07:35:58Z' language: JavaScript archived: false stars: 34 watchers: 2 forks: 8 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"88cf457ba6d17c7140dd53af6f04db4103e18b247eacf63b67c1edc08fd8733a"' repoLastModified: Tue, 06 Jan 2026 07:35:58 GMT foundInMaster: true category: Parsers id: 8345bfb16b8aa0dc647fa5b2ab90b100 - source: openapi3 tags repository: https://github.com/mrtimofey/vue-admin v3: true repositoryMetadata: base64Readme: >- IyMgV2hhdCBpcyBpdD8KClRoaXMgaXMgdGhlIGxhc3QgcGFja2FnZSB5b3Ugd2lsbCBldmVyIG5lZWQgdG8gYnVpbGQgeW91ciBjdXN0b20gZGFzaGJvYXJkIG9yIGFkbWluaXN0cmF0aXZlIHBhbmVsLiBJdCBpbmNsdWRlczoKKiBBZG1pbkxURSAob25seSBzdHlsZXMpCiogVnVlLmpzIGJhc2VkIHJlYWR5LXRvLXVzZSBvdXQgb2YgdGhlIGJveCBTUEEKKiBTaW1wbGUgYnV0IGZsZXhpYmxlIGN1c3RvbWl6YXRpb24gc3lzdGVtCiogT3BlbkFQSSAzIChmb3JtZXJseSBrbm93biBhcyBTd2FnZ2VyKSBzcGVjaWZpY2F0aW9uIGZvciBhIGNvbXBhdGlibGUgc2VydmVyIEFQSSBpbXBsZW1lbnRhdGlvbgoqIEJpZyB2YXJpZXR5IG9mIGJ1aWx0LWluIGZpZWxkIHR5cGVzCiogQmlnIHZhcmlldHkgb2YgYnVpbHQtaW4gZGF0YSBkaXNwbGF5IGZvcm1hdCB0eXBlcwoqIFR5cGUtc2FmZSBieSBkZXNpZ24KKiBTdWl0YWJsZSBmb3IgYSBmYXN0IHByb3RvdHlwaW5nCgpbRGVtbyB3aXRoIExhcmF2ZWwgNS42IGJhY2tlbmRdKGh0dHA6Ly9hZG1pbi1kZW1vLnNoaXQtZnJlZS5zcGFjZSkKCltEb2N1bWVudGF0aW9uXShodHRwczovL21yLXRpbW9mZXkuZ2l0Ym9va3MuaW8vdnVlLWFkbWluKQoKIyMgSG93IGRvZXMgaXQgd29yaz8KClRoZSBwYWNrYWdlIGNvbnRhaW5zIGFsbCB0aGUgbmVjZXNzYXJ5IGZpbGVzIGZvciBidWlsZCBhbmQgZGV2ZWxvcG1lbnQuIFRoZXJlIGlzIG5vIHByZWJ1aWx0IHZlcnNpb24gc2luY2UKYHZ1ZS1hZG1pbi1mcm9udGAgYWltcyB0byBwcm92aWRlIGEgZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnQgd2hlcmUgeW91IGNhbiBjdXN0b21pemUgYWxtb3N0IGV2ZXJ5dGhpbmcgaW5jbHVkaW5nCmJ1aWxkLXRpbWUgc3R1ZmYuIEJ1dCB5b3UgY2FuIHVzZSB0aGlzIGp1c3QgYXMtaXMgYnkgZXhlY3V0aW5nIGEgYnVpbGQgY29tbWFuZCBkZXNjcmliZWQgYmVsbG93LgoKQWxsIHRoZSB0aGluZ3MgcmVsYXRlZCB0byB5b3VyIHByb2plY3QgZGF0YSBzdHJ1Y3R1cmUgYW5kIGRhdGEgbWFuaXB1bGF0aW9uIGNhcGFiaWxpdGllcyBzaG91bGQgYmUgcHJvdmlkZWQKYnkgeW91ciBiYWNrZW5kIFJFU1QgQVBJIGVuZHBvaW50cwpbZGVzY3JpYmVkIGluIE9wZW5BUEkgMyBmb3JtYXRdKGh0dHBzOi8vZ2l0aHViLmNvbS9tclRpbW9mZXkvdnVlLWFkbWluL2Jsb2IvbWFzdGVyL2RvY3Mvc3dhZ2dlci55bWwpCgojIyBXaGF0IGFib3V0IHNlcnZlciBzb2x1dGlvbnM/CgpPbmx5IFtQSFA3LjEvTGFyYXZlbCBiYXNlZCBzb2x1dGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL21yVGltb2ZleS9sYXJhdmVsLWFkbWluLWFwaSkgZXhpc3RzIGF0IHRoZSBtb21lbnQuCk1ha2luZyBtb3JlIG9mIHRoZW0gd2lsbCBiZSB2ZXJ5IGFwcHJlY2lhdGVkLgpbUmVhZCB0aGUgZG9jc10oaHR0cHM6Ly9tci10aW1vZmV5LmdpdGJvb2tzLmlvL3Z1ZS1hZG1pbi9zZXJ2ZXItYXBpLXByZXJlcXVpc2l0aWVzLmh0bWwpIGZvciBtb3JlIGluZm8uCgojIyBIb3cgdG8gdXNlPwoKSW5zdGFsbDoKYGBgYmFzaApucG0gaSAtUyB2dWUtYWRtaW4tZnJvbnQgY3Jvc3MtZW52IHJpbXJhZiB3ZWJwYWNrLW5hbm8KYGBgCgpBZGQgdGhpcyB0byBwYWNrYWdlLmpzb24gc2NyaXB0czoKYGBganNvbgp7CgkiYWRtaW46ZGV2IjogIm5vZGUgbm9kZV9tb2R1bGVzL3Z1ZS1hZG1pbi1mcm9udC9pbmRleC5qcyIsCgkiYWRtaW46YnVpbGQiOiAicmltcmFmIHB1YmxpYy9hZG1pbi1kaXN0ICYmIGNyb3NzLWVudiBOT0RFX0VOVj1wcm9kdWN0aW9uIHdwIC0tY29uZmlnIG5vZGVfbW9kdWxlcy92dWUtYWRtaW4tZnJvbnQvd2VicGFjay5jb25maWcuanMiCn0KYGBgCgpVc2UgYG5wbSBydW4gYWRtaW46ZGV2YCB0byBzdGFydCBkZXYgc2VydmVyLCBgbnBtIHJ1biBhZG1pbjpidWlsZGAgdG8gbWFrZSBhIHN0YXRpYyBidWlsZCB3aXRoaW4gYHB1YmxpYy9hZG1pbi1kaXN0YCBkaXJlY3RvcnkuCgpNb3JlIGluZm8gW2hlcmVdKGh0dHBzOi8vbXItdGltb2ZleS5naXRib29rcy5pby92dWUtYWRtaW4vcXVpY2stc3RhcnQuaHRtbCkuCg== readmeEtag: '"e61e7c35756e2a6624a0f2f5f6fd4e5cdab04dae"' readmeLastModified: Wed, 07 Oct 2020 08:58:30 GMT repositoryId: 112570086 description: Vue.js based administrative interface created: '2017-11-30T05:46:35Z' updated: '2023-08-18T22:29:38Z' language: Vue archived: false stars: 28 watchers: 2 forks: 4 owner: mrTimofey logo: https://avatars.githubusercontent.com/u/7373909?v=4 license: MIT repoEtag: '"70798c5bfcbd1e15f95f9aeee3da26aedd6f9932def36247b9635f51a60d3758"' repoLastModified: Fri, 18 Aug 2023 22:29:38 GMT foundInMaster: true category: - Documentation - Server Implementations id: 74974a7100769636130f55d931ed814f - source: openapi3 tags repository: https://github.com/redocly/json-to-json-schema v3: true repositoryMetadata: base64Readme: >- IyBqc29uLXRvLWpzb24tc2NoZW1hCgpbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZW4ubmV0L25wbS92L0ByZWRvY2x5L2pzb24tdG8tanNvbi1zY2hlbWEpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9AcmVkb2NseS9qc29uLXRvLWpzb24tc2NoZW1hKSBbIVtidW5kbGUgc2l6ZV0oaHR0cHM6Ly9iYWRnZW4ubmV0L2J1bmRsZXBob2JpYS9taW4vQHJlZG9jbHkvanNvbi10by1qc29uLXNjaGVtYSldKGh0dHBzOi8vYnVuZGxlcGhvYmlhLmNvbS9wYWNrYWdlL0ByZWRvY2x5L2pzb24tdG8tanNvbi1zY2hlbWEpIFshW2RlcGVuZGVuY3kgc2l6ZV0oaHR0cHM6Ly9iYWRnZW4ubmV0L2J1bmRsZXBob2JpYS9kZXBlbmRlbmN5LWNvdW50L0ByZWRvY2x5L2pzb24tdG8tanNvbi1zY2hlbWEpXShodHRwczovL2J1bmRsZXBob2JpYS5jb20vcGFja2FnZS9AcmVkb2NseS9qc29uLXRvLWpzb24tc2NoZW1hKSBbIVtDb3ZlcmFnZSBTdGF0dXNdKGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9SZWRvY2x5L2pzb24tdG8tanNvbi1zY2hlbWEvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL1JlZG9jbHkvanNvbi10by1qc29uLXNjaGVtYT9icmFuY2g9bWFpbikKCkNvbnZlcnQgSlNPTiBleGFtcGxlcyBpbnRvIEpTT04gc2NoZW1hLgoKU3VwcG9ydHMgSlNPTiBTY2hlbWEgYGRyYWZ0LTA1YCB1c2VkIGluIFN3YWdnZXIgMi4wIGFuZCBPcGVuQVBJIDMuMCBhbmQgbmV3IGRyYWZ0IGBkcmFmdC0yMDIwLTEyYCB1c2VkIGluIE9wZW5BUEkgMy4xLgoKIyMgVXNhZ2UKCmBgYGpzCmltcG9ydCB7IGNvbnZlcnQsIGZvcm1hdCB9IGZyb20gJ0ByZWRvY2x5L2pzb24tdG8tanNvbi1zY2hlbWEnOwoKY29uc3QgZXhhbXBsZSA9IHsgZmlyc3RuYW1lOiAnSm9obicsIHN1cm5hbWU6ICdEb2UnLCBiaXJ0aGRheTogJzE5OTAtMDItMjMnIH07Cgpjb25zdCBzY2hlbWEgPSBjb252ZXJ0KGV4YW1wbGUsIHsKICB0YXJnZXQ6ICdkcmFmdC0wNS1vYXMnLCAvLyBvciBkcmFmdC0yMDIwLTEyCiAgaW5jbHVkZUV4YW1wbGVzOiBmYWxzZSwKICBkaXNhYmxlQWRkaXRpb25hbFByb3BlcnRpZXM6IGZhbHNlLAogIGluZmVyUmVxdWlyZWQ6IGZhbHNlLAp9KTsKYGBgCgpPdXRwdXQ6CgpgYGB5YW1sCmZpcnN0bmFtZToKICB0eXBlOiBzdHJpbmcKc3VybmFtZToKICB0eXBlOiBzdHJpbmcKYmlydGhkYXk6CiAgdHlwZTogc3RyaW5nCiAgZm9ybWF0OiBkYXRlCmBgYAoKIyMgU2ltaWxhciB0b29scwoKaHR0cHM6Ly9naXRodWIuY29tL21vaHNlbjEvanNvbi10by1qc29uLXNjaGVtYQo= readmeEtag: '"e67a573719c6836f2f476fc15cd6c16a2afe4c2b"' readmeLastModified: Tue, 09 Jul 2024 12:53:52 GMT repositoryId: 470442887 description: >- Convert JSON examples into JSON schema (supports Swagger 2, OpenAPI 3 and 3.1) created: '2022-03-16T05:21:04Z' updated: '2026-01-27T17:29:06Z' language: TypeScript archived: false stars: 31 watchers: 11 forks: 6 owner: Redocly logo: https://avatars.githubusercontent.com/u/32099856?v=4 repoEtag: '"4196ca454760694e7750485a426b1a4a399d2455e280b8d1bd4d95c9fb88463f"' repoLastModified: Tue, 27 Jan 2026 17:29:06 GMT foundInMaster: true category: - Data Validators - Parsers id: b3e82f9d6dce83866b919d7eb14c19bc - source: openapi3 tags repository: https://github.com/aerokube/selenium-openapi v3: true repositoryMetadata: base64Readme: >- IyBTZWxlbml1bSBPcGVuQVBJIFNwZWMKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBTZWxlbml1bSBPcGVuQVBJIDMuMCAoZXgtU3dhZ2dlcikgc3BlYy4KCiMjIFVzaW5nIFN3YWdnZXIgRWRpdG9yCgoxKSBMYXVuY2ggU3dhZ2dlciBFZGl0b3IgRG9ja2VyIGNvbnRhaW5lcjoKYGBgCiQgZG9ja2VyIHJ1biAtZCAtLW5hbWUgc3dhZ2dlci1lZGl0b3IgLXAgODA4MDo4MDgwIHN3YWdnZXJhcGkvc3dhZ2dlci1lZGl0b3IKYGBgCgoyKSBPcGVuIGBodHRwOi8vbG9jYWxob3N0OjgwODAvYCBpbiBicm93c2VyLgozKSBDb3B5IGFuZCBwYXN0ZSBgc2VsZW5pdW0ueWFtbGAgY29udGVudHMgdG8gZWRpdG9yLgo= readmeEtag: '"56432a639227e29ac74f08e20891709988efc3f6"' readmeLastModified: Sat, 03 Sep 2022 17:21:07 GMT repositoryId: 185360868 description: The missing Selenium OpenAPI spec created: '2019-05-07T08:50:02Z' updated: '2023-01-29T06:21:22Z' language: Shell archived: false stars: 27 watchers: 3 forks: 9 owner: aerokube logo: https://avatars.githubusercontent.com/u/26328913?v=4 license: Apache-2.0 repoEtag: '"19e2bfd15e8f4f4a96bbaf2da874672cdf53c1179eac5b8eb6706e96061dc475"' repoLastModified: Sun, 29 Jan 2023 06:21:22 GMT foundInMaster: true category: - Code Generators - Server Implementations id: f034b9fb8c5913474475f9f688e93960 - source: openapi3 tags repository: https://github.com/mockoon/mock-samples v3: true repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL21vY2tvb24uY29tIiBhbHQ9Im1vY2tvb24gbG9nbyI+CiAgICA8aW1nIHdpZHRoPSIyMDAiIGhlaWdodD0iMjAwIiBzcmM9Imh0dHBzOi8vbW9ja29vbi5jb20vaW1hZ2VzL2xvZ28tc3F1YXJlLXdlYnNpdGUucG5nIj4KICA8L2E+CiAgPGJyPgogIDxhIGhyZWY9Imh0dHBzOi8vbW9ja29vbi5jb20vZG93bmxvYWQvIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9Eb3dubG9hZCUyMGFwcC1Hby1ncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUmY29sb3JCPTE5OTdjNiIvPjwvYT4KICA8YSBocmVmPSJodHRwczovL21vY2tvb24uY29tLyI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvV2Vic2l0ZS1Hby1ncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUmY29sb3JCPTE5OTdjNiIvPjwvYT4KICA8YSBocmVmPSJodHRwczovL21vY2tvb24uY29tL25ld3NsZXR0ZXIvIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9OZXdzbGV0dGVyLVN1YnNjcmliZS1ncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUiLz48L2E+CiAgPGJyPgogIDxicj4KICA8aDE+TW9ja29vbidzIG1vY2sgc2FtcGxlczwvaDE+CjwvZGl2PgoKV2VsY29tZSB0byBNb2Nrb29uJ3MgbW9jayBBUEkgc2FtcGxlcyByZXBvc2l0b3J5LiBZb3Ugd2lsbCBmaW5kIGluIHRoZSBbYGFwaXNgXSgvbW9jay1hcGlzKSBmb2xkZXIgbWFueSBtb2NrIHNhbXBsZXMgZnJvbSBwb3B1bGFyIEFQSXMgKE9wZW4gV2VhdGhlciwgR2lwaHksIGV0Yy4pLiBXZSB3aWxsIGJlIGFkZGluZyBtb3JlIHNhbXBsZXMgZnJvbSB0aW1lIHRvIHRpbWUuCgpZb3Ugd2lsbCBmaW5kIHRoZSBmdWxsIGxpc3Qgb2Ygc2FtcGxlIG1vY2sgQVBJcyBvbiBvdXIgd2Vic2l0ZSB0b286IGh0dHBzOi8vbW9ja29vbi5jb20vbW9jay1zYW1wbGVzLwoKLS0tCgojIyBIb3cgdG8gb3BlbiB0aGUgc2FtcGxlcyBpbiBNb2Nrb29uCgpUbyBvcGVuIHRoZSBzYW1wbGVzIGluIE1vY2tvb24gYXBwbGljYXRpb24sIHlvdSBoYXZlIHR3byBwb3NzaWJpbGl0aWVzOgoKMS4gRG93bmxvYWQgdGhlIHNhbXBsZSBKU09OIGZpbGUgYW5kIGNsaWNrIG9uIHRoZSBvcGVuIGJ1dHRvbjoKCiAgICFbb3BlbiBlbnZpcm9ubWVudF0oL2RvY3Mvb3Blbi1lbnZpcm9ubWVudC5wbmcpCgoxLiBDb3B5IHRoZSBzYW1wbGUgZmlsZSBjb250ZW50IGZyb20gdGhpcyByZXBvc2l0b3J5IHRvIHRoZSBjbGlwYm9hcmQgYW5kIG9wZW4gdGhlIGBGaWxlYCBtZW51IGFuZCBjaG9vc2UgYE5ldyBlbnZpcm9ubWVudCBmcm9tIGNsaXBib2FyZGA6CgogICAhW25ldyBlbnZpcm9ubWVudCBmcm9tIGNsaXBib2FyZF0oL2RvY3MvbmV3LWVudmlyb25tZW50LWZyb20tY2xpcGJvYXJkLnBuZykKCkluIGJvdGggY2FzZXMsIGEgbmV3IGVudmlyb25tZW50IHdpbGwgYmUgY3JlYXRlZCBhbmQgcmVhZHkgdG8gdXNlIGFmdGVyIHJ1bm5pbmcgaXQgYnkgcHJlc3NpbmcgdGhlIGdyZWVuICJwbGF5IiBidXR0b24uCgohW1J1biB0aGUgbW9jayBBUEldKC9kb2NzL3J1bi1hcGkucG5nKQoKIyMgSG93IHRvIHVzZSB0aGUgc2FtcGxlcyB3aXRoIE1vY2tvb24gQ0xJCgpUbyBydW4gdGhlIG1vY2sgc2FtcGxlcyB3aXRoIHRoZSBbQ0xJXShodHRwczovL2dpdGh1Yi5jb20vbW9ja29vbi9jbGkpIHlvdSBjYW4gZGlyYWN0bHkgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB1c2luZyB0aGUgc2FtcGxlIHJhdyBmaWxlIEdpdEh1YiBVUkw6CgpgbW9ja29vbi1jbGkgc3RhcnQgLS1kYXRhIGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9tb2Nrb29uL21vY2stc2FtcGxlcy9tYWluL21vY2stYXBpcy9kYXRhL2dpcGh5Y29tLmpzb25gCgojIyBTdXBwb3J0L2ZlZWRiYWNrCgpZb3UgY2FuIGRpc2N1c3MgYWxsIHRoaW5ncyByZWxhdGVkIHRvIE1vY2tvb24sIGFuZCBhc2sgZm9yIGhlbHAsIG9uIHRoZSBbb2ZmaWNpYWwgY29tbXVuaXR5XShodHRwczovL2dpdGh1Yi5jb20vbW9ja29vbi9tb2Nrb29uL2Rpc2N1c3Npb25zKS4gSXQncyBhbHNvIGEgZ29vZCBwbGFjZSB0byBkaXNjdXNzIGJ1Z3MgYW5kIGZlYXR1cmUgcmVxdWVzdHMgYmVmb3JlIG9wZW5pbmcgYW4gaXNzdWUgb24gdGhpcyByZXBvc2l0b3J5LgoKIyMgQ29udHJpYnV0aW5nCgpJZiB5b3UgYXJlIGludGVyZXN0ZWQgaW4gY29udHJpYnV0aW5nIHRvIE1vY2tvb24gd2Vic2l0ZSBhbmQgZG9jdW1lbnRhdGlvbiwgcGxlYXNlIHRha2UgYSBsb29rIGF0IHRoZSBbY29udHJpYnV0aW5nIGd1aWRlbGluZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tb2Nrb29uL21vY2tvb24uY29tL2Jsb2IvbWFpbi9DT05UUklCVVRJTkcubWQpLgoKUGxlYXNlIGFsc28gdGFrZSBhIGxvb2sgYXQgb3VyIFtDb2RlIG9mIENvbmR1Y3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9tb2Nrb29uLy5naXRodWIvYmxvYi9tYWluL0NPREVfT0ZfQ09ORFVDVC5tZCkuCgojIyBEb2N1bWVudGF0aW9uCgpZb3Ugd2lsbCBmaW5kIE1vY2tvb24ncyBbZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9tb2Nrb29uLmNvbS9kb2NzL2xhdGVzdC8pIG9uIHRoZSBvZmZpY2lhbCB3ZWJzaXRlLiBJdCBjb3ZlcnMgTW9ja29vbidzIG1vc3QgY29tcGxleCBmZWF0dXJlcy4gRmVlbCBmcmVlIHRvIGNvbnRyaWJ1dGUgb3IgYXNrIGZvciBuZXcgdG9waWNzIHRvIGJlIGNvdmVyZWQuCgojIyBSb2FkbWFwCgpJZiB5b3Ugd2FudCB0byBrbm93IHdoYXQgd2lsbCBiZSBjb21pbmcgaW4gdGhlIG5leHQgcmVsZWFzZSB5b3UgY2FuIGNoZWNrIHRoZSBnbG9iYWwgW1JvYWRtYXBdKGh0dHBzOi8vbW9ja29vbi5jb20vcHVibGljLXJvYWRtYXAvKSBvciBbc3Vic2NyaWJlIHRvIG91ciBuZXdzbGV0dGVyXShodHRwczovL21vY2tvb24uY29tL25ld3NsZXR0ZXIvKS4K readmeEtag: '"8ed035cd01844de1f302bbb2a131cbbd51791b33"' readmeLastModified: Mon, 29 Sep 2025 14:56:39 GMT repositoryId: 310602836 description: 'Start using Mockoon in no time with popular APIs mock samples. ' created: '2020-11-06T13:20:14Z' updated: '2026-01-07T08:12:51Z' language: TypeScript archived: false stars: 33 watchers: 2 forks: 23 owner: mockoon logo: https://avatars.githubusercontent.com/u/49429147?v=4 license: MIT repoEtag: '"85253cadab0ebe35459c4303917fedb0eb67b60d129f964379cfb0340b1969b5"' repoLastModified: Wed, 07 Jan 2026 08:12:51 GMT foundInMaster: true category: - Documentation - Testing id: af9630261912faa3f1476d3277e03d2a - source: openapi3 tags repository: https://github.com/rafaelcaricio/sticker v3: true repositoryMetadata: base64Readme: >- .. image:: https://github.com/rafaelcaricio/sticker/raw/master/docs/images/sticker_logo.png
    :align: center
    :alt: Sticker
    :target: https://github.com/rafaelcaricio/sticker

|

.. image:: https://img.shields.io/pypi/v/sticker.svg
    :target: https://pypi.python.org/pypi/sticker

.. image:: https://img.shields.io/pypi/l/sticker.svg
    :target: https://pypi.python.org/pypi/sticker

.. image:: https://img.shields.io/pypi/pyversions/sticker.svg
    :target: https://pypi.python.org/pypi/sticker

.. image:: https://img.shields.io/github/contributors/rafaelcaricio/sticker.svg
    :target: https://github.com/rafaelcaricio/sticker/graphs/contributors

Write boilerplate-free Python functions and use them as your API handlers.
Sticker allows you to choose Flask, bottle.py, Sanic, or Tornado as your
application runtime.

**Highlights**:

* Community created and maintained
* Support for `OpenAPI 3.0 <https://swagger.io/specification/>`_
* Multi-framework support: `Flask <http://flask.pocoo.org/>`_, `bottle.py <https://github.com/bottlepy/bottle>`_, `Sanic <https://github.com/channelcat/sanic>`_, and `Tornado <http://www.tornadoweb.org/en/stable/>`_
* Support for **pure Python handlers** (no boilerplate code)

It's Easy to Write
==================

You need a little bit of Python.

.. code-block:: python

    def say_hello(params):
        return {"contents": "Hello World!"}

Plus bits of your API description.

.. code-block:: YAML

    openapi: "3.0.0"
    paths:
      /:
        get:
          operationId: hello.say_hello

Now the fun part, you choose which web framework you want to use.

Run with Flask:

.. code-block:: python

    from sticker import FlaskAPI
    api = FlaskAPI('hello.yml')
    api.get_app(__name__).run()

Run with Bottle.py:

.. code-block:: python

    from sticker import BottleAPI
    api = BottleAPI('hello.yml')
    api.run()

Run with Sanic:

.. code-block:: python

    from sticker import SanicAPI
    api = SanicAPI('hello.yml')
    api.get_app(__name__).run()

Run with Tornado:

.. code-block:: python

    from sticker import TornadoAPI
    import tornado.ioloop
    api = TornadoAPI('hello.yml')
    api.get_app().listen(8888)
    tornado.ioloop.IOLoop.current().start()

The framework setup, validation, types conversion, and mocking is handled at runtime by Sticker.

✨

Installation
============

Sticker is published at PyPI, so you can use ``pip`` to install:

.. code-block:: bash

    $ pip install sticker

Requirements
============

Sticker was developed for **Python >=3.6** and **OpenAPI 3.0**. Support for Python 2.7 is not present nor planned for this project.

Documentation
=============

Sticker is a flexible metaframework for Web API development and execution. The OpenAPI 3.0 standard is used as
description format for Sticker powered APIs. You provide the API specification and choose one of the
Sticker's runtimes to have a webserver up and running.

In this document we will describe a few different ways to write code that works well with Sticker.

Pure Python Handlers
--------------------

Sticker supports the use of pure Python functions as handlers. Your code will be free of any framework
specific boilerplate code, including Sticker's itself. This allows you to swap between different frameworks
as you wish. Sticker will take care of putting together your code, your API, and the framework you choose.

.. code-block:: python

    def myhandler(params):
        return {
            "content": f"Hello {params.get("name", "World")}!",
            "status": 200
        }

Writing tests for pure Python handles is easy and also
free of boilerplate code.

.. code-block:: python

    def test_myhandler():
        params = {
            "name": "John Doe"
        }
        response = myhandler(params)
        assert response["content"] == "Hello John Doe!"

As you could see in the example above, no imports from Sticker were necessary to define the API handler function.
This is only possible because Sticker expects your handlers to follow a code convention.

Anatomy Of An API Handler Function
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

Write this part.

Responses
^^^^^^^^^

API handlers are expected to return a Python dictionary (``dict``) object. The returned dictionary defines how a response
will look like. All keys in the dictionary are optional. The expected keys are described in the table bellow.

=========== ======================== ===========
Key         Type                     Description
=========== ======================== ===========
content     str                      Body of HTTP request. No treatment/parsing of this value is done. The value is passed directly to the chosen framework.
json        Union[dict, List[dict]]  JSON value to be used in the body of the request. This is a shortcut to having the header "Content-Type: application/json" and serializing this value using the most common way done by the chosen framework.
file        Union[IO[AnyStr], str]   Data to be returned as byte stream. This is a shortcut for having the header "Content-Type: application/octet-stream". Uses the most common way to stream files with the chosen framework.
redirect    str                      The path or full URL to be redirected. This is a shortcut for having the header "Location:" with HTTP status `301`.
status      int                      The HTTP status code to be used in the response. This value overrides any shortcut default status code.
headers     Dict[str, str]           The HTTP headers to be used in the response. This value is merged with the shortcut values with priority.
=========== ======================== ===========


We have exposed here some examples of using different configurations of the ``dict`` we've defined above to describe the
HTTP response of API handlers. The actual HTTP response value generated will vary depending on the framework chosen as
runtime. The examples are a minimal illustration of what to expect to be the HTTP response.

The "content" key can be used when it's desired to return a "Hello world!" string with status ``200``.

.. code-block:: python

    def say_hello(params):
        return {"content": "Hello world!"}

Results in the HTTP response similar to:

.. code-block::

    HTTP/1.1 200 OK
    Content-Type: text/plain

    Hello world!

The "json" key can be used when desired to return an JSON response with status ``201``.

.. code-block:: python

    def create(params):
        data = {
            "id": "uhHuehuE",
            "value": "something"
        }
        return {"json": data, "status": 201}

The HTTP response generated will be similar to:

.. code-block::

    HTTP/1.1 201 Created
    Content-Type: application/json

    {"id":"uhHuehuE","value":"something"}

The "file" key is used to return file contents.

.. code-block:: python

    def homepage(params):
        return {
            "file": open('templates/home.html', 'r'),
            "headers": {
                "Content-Type": "text/html"
            }
        }

The HTTP response will be similar to:

.. code-block::

    HTTP/1.1 200 OK
    Content-Type: text/html

    <html><title>My homepage</title><body><h1>Welcome!</h1></body></html>

When necessary to redirect request, the "redirect" key can be used.

.. code-block:: python

    def old_endpoint(params):
        return {'redirect': '/new-path'}

The HTTP response will be similar to:

.. code-block::

    HTTP/1.1 301 Moved Permanently
    Location: https://example.com/new-path

The usage of keys "status" and "headers" were shown in the previous examples. The "status" and "headers" keys, when set,
override the values set by default when using the shortcut keys ("json", "file", and "redirect").

Error Handling
--------------

Sticker expects you to define the error format to be returned by your API. A error handler is configurable,
and called every time validation for the endpoint fails.

.. code-block:: python

    def error_handler(error):
        return {
            "content": {
                "error": error["message"]
            },
            "headers": {
                "Content-Type": "application/json"
            },
            "status_code": 400
        }

Developing
==========

We follow `Semantic Versioning <https://semver.org/spec/v2.0.0.html>`_.

Contributing
============

Sticker is developed under the `Apache 2.0 license <https://github.com/rafaelcaricio/sticker/blob/master/LICENSE>`_
and is publicly available to everyone. We are happy to accept contributions.

How to Contribute
-----------------

#. Check for open issues or open a fresh issue to start a discussion around a feature idea or a bug. There is a `Good First Issue`_ tag for issues that should be ideal for people who are not very familiar with the codebase yet.
#. Fork `the repository`_ on GitHub to start making your changes to the **master** branch (or branch off of it).
#. Write a test which shows that the bug was fixed or that the feature works as expected.
#. Send a pull request and bug the maintainer until it gets merged and published. :) Make sure to add yourself to AUTHORS_.

.. _`the repository`: https://github.com/rafaelcaricio/sticker
.. _AUTHORS: https://github.com/rafaelcaricio/sticker/blob/master/AUTHORS.rst
.. _Good First Issue: https://github.com/rafaelcaricio/sticker/issues?q=is%3Aissue+is%3Aopen+label%3A%22good+first+issue%22
 readmeEtag: '"32a10c263ca8bfd42a12eb6312b770f0514a8a67"' readmeLastModified: Sat, 01 May 2021 22:48:47 GMT repositoryId: 126081831 description: >- Sticker is a powerful yet boilerplate-free alternative to writing your web API. created: '2018-03-20T21:00:32Z' updated: '2023-01-27T23:43:25Z' language: Python archived: true stars: 26 watchers: 2 forks: 1 owner: rafaelcaricio logo: https://avatars.githubusercontent.com/u/107613?v=4 license: Apache-2.0 repoEtag: '"531fb023aa42fa3cb2ae6f50d9761c795897303c41418f09be9b823d90311c3d"' repoLastModified: Fri, 27 Jan 2023 23:43:25 GMT foundInMaster: true category: Server Implementations id: cb1a6cacac08f3879554ade8788535c8 - source: openapi3 tags repository: https://github.com/chanced/openapi v3: true id: 7ee990b4d1feb380f92d50101672d759 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpIC0gQW4gT3BlbkFQSSAzLnggbGlicmFyeSBmb3IgZ28KClshW0dvIFJlZmVyZW5jZV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2JhZGdlL2dpdGh1Yi5jb20vY2hhbmNlZC9vcGVuYXBpLnN2ZyldKGh0dHBzOi8vcGtnLmdvLmRldi9naXRodWIuY29tL2NoYW5jZWQvb3BlbmFwaSkKWyFbTGF0ZXN0IFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3YvdGFnL2NoYW5jZWQvb3BlbmFwaS5zdmc/c29ydD1zZW12ZXImc3R5bGU9ZmxhdC1zcXVhcmUmbGFiZWw9dmVyc2lvbiZjb2xvcj1ibHVlKV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvdi90YWcvY2hhbmNlZC9vcGVuYXBpLnN2Zz9zb3J0PXNlbXZlciZzdHlsZT1mbGF0LXNxdWFyZSZsYWJlbD12ZXJzaW9uJmNvbG9yPWJsdWUpCiFbQnVpbGQgU3RhdHVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi93b3JrZmxvdy9zdGF0dXMvY2hhbmNlZC9vcGVuYXBpL0J1aWxkP3N0eWxlPWZsYXQtc3F1YXJlKQoKb3BlbmFwaSBpcyBhIGxpYnJhcnkgZm9yIHBhcnNpbmcgYW5kIHZhbGlkYXRpbmcgT3BlbkFQSQpbMy4xXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4xLjApLApbMy4wXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4wLjMpLiBUaGUgaW50ZW50IG9mIHRoZSBsaWJyYXJ5IGlzIHRvCm9mZmVyIGJ1aWxkaW5nIGJsb2NrcyBmb3IgY29kZSBhbmQgZG9jdW1lbnRhdGlvbiBnZW5lcmF0aW9uLgoKOndhcm5pbmc6IFRoaXMgbGlicmFyeSBpcyB1bmRlciBhY3RpdmUgZGV2ZWxvcG1lbnQ7IHRoZXJlIG1heSBiZSBicmVha2luZyBjaGFuZ2VzIGFuZCBidWdzLgoKIyMgRmVhdHVyZXMKCi0gICBgJHJlZmAgcmVzb2x1dGlvbgotICAgQWxsIGtleXMgcmV0YWluIHRoZWlyIG9yZGVyIGZyb20gdGhlIG1hcmt1cCB1c2luZyBzbGljZXMgb2Yga2V5L3ZhbHVlcyB3aGljaAogICAgYWlkcyBpbiBjb2RlIGdlbmVyYXRpb24uCi0gICBWYWxpZGF0aW9uIChbc2VlIHRoZSB2YWxpZGF0aW9uIHNlY3Rpb25dKCN2YWxpZGF0aW9uKSkKLSAgIEFsbCBub24tcHJpbWl0aXZlIG5vZGVzIGhhdmUgYW4gYWJzb2x1dGUgJiByZWxhdGl2ZSBsb2NhdGlvbgotICAgU3RyaW5ncyBhcmUgW3RleHQuVGV4dF0oaHR0cHM6Ly9naXRodWIuY29tL2NoYW5jZWQvY2Fwcykgd2hpY2ggaGFzIGNhc2UKICAgIGNvbnZlcnNpb25zIGFuZCBgc3RyaW5nc2AgZnVuY3Rpb25zIGFzIG1ldGhvZHMuCi0gICBFeHRlbnNpb25zLCB1bmtub3duIEpTT04gU2NoZW1hIGtleXdvcmRzLCBleGFtcGxlcywgYW5kIGEgZmV3IG90aGVyIGZpZWxkcwogICAgYXJlIGluc3RhbmNlcyBvZiBbanNvbnguUmF3TWVzc2FnZV0oaHR0cHM6Ly9naXRodWIuY29tL2NoYW5jZWQvanNvbngpIHdoaWNoCiAgICBjb21lcyB3aXRoIGEgZmV3IGhlbHBlciBtZXRob2RzLgotICAgU3VwcG9ydHMgYm90aCBKU09OIGFuZCBZQU1MCgojIyBVc2FnZQoKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgKAogICAgImdpdGh1Yi5jb20vY2hhbmNlZC9vcGVuYXBpIgogICAgImdpdGh1Yi5jb20vY2hhbmNlZC91cmkiCiAgICAiZ2l0aHViLmNvbS9zYW50aG9zaC10ZWt1cmkvanNvbnNjaGVtYS92NSIKICAgICJlbWJlZCIKICAgICJpbyIKICAgICJwYXRoL2ZpbGVwYXRoIgogICAgImxvZyIKKQoKLy9nbzplbWJlZCBzcGVjCnZhciBzcGVjRmlsZXMgZW1iZWQuRlMKCmZ1bmMgbWFpbigpIHsKICAgIGN0eCA6PSBjb250ZXh0LkJhY2tncm91bmQoKQoKICAgIGMsIGVyciA6PSBvcGVuYXBpLlNldHVwQ29tcGlsZXIoanNvbnNjaGVtYS5OZXdDb21waWxlcigpKSAvLyBhZGRpbmcgc2NoZW1hIGZpbGVzCiAgICBpZiBlcnIgIT0gbmlsIHsKICAgICAgICBsb2cuRmF0YWwoZXJyKQogICAgfQogICAgdiwgZXJyIDo9IG9wZW5hcGkuTmV3VmFsaWRhdG9yKGMpCiAgICBpZiBlcnIgIT0gbmlsIHsKICAgICAgICBsb2cuRmF0YWwoZXJyKQogICAgfQoKICAgIGZuIDo9IGZ1bmMoXyBjb250ZXh0LkNvbnRleHQsIHVyaSB1cmkuVVJJLCBraW5kIG9wZW5hcGkuS2luZCkgKG9wZW5hcGkuS2luZCwgW11ieXRlLCBlcnJvcil7CiAgICAgICAgZiwgZXJyIDo9IHNwZWNGaWxlcy5PcGVuKGZwKQogICAgICAgIGlmIGVyciAhPSBuaWwgewogICAgICAgICAgICBsb2cuRmF0YWwoZXJyKQogICAgICAgIH0KICAgICAgICAvLyB5b3UgY2FuIHJldHVybiBlaXRoZXIgSlNPTiBvciBZQU1MCiAgICAgICAgZCwgZXJyIDo9IGlvLlJlYWRBbGwoZikKICAgICAgICBpZiBlcnIgIT0gbmlsewogICAgICAgICAgICBsb2cuZmF0YWwoZXJyKQogICAgICAgIH0KICAgICAgICAvLyB1c2UgdGhlIHVyaSBvciB0aGUgZGF0YSB0byBkZXRlcm1pbmUgdGhlIEtpbmQKICAgICAgICByZXR1cm4gb3BlbmFwaS5LaW5kRG9jdW1lbnQsIGQsIG5pbAogICAgfQogICAgLy8geW91IGNhbiBMb2FkIGVpdGhlciBKU09OIG9yIFlBTUwKICAgIC8vIExvYWQgdmFsaWRhdGVzIHRoZSBEb2N1bWVudCBhcyB3ZWxsLgogICAgZG9jLCBlcnIgOj0gb3BlbmFwaS5Mb2FkKGN0eCwgInNwZWMvb3BlbmFwaS55YW1sIiwgdiwgZm4pCiAgICBpZiBlcnIgIT0gbmlsewogICAgICAgIGxvZy5GYXRhbChlcnIpCiAgICB9CiAgICBfID0gZG9jIC8vICpvcGVuYXBpLkRvY3VtZW50Cn0KYGBgCgojIyBWYWxpZGF0aW9uCgpUaGUgc3RhbmRhcmQgdmFsaWRhdG9yIChgU3RkVmFsaWRhdG9yYCkgY3VycmVudGx5IHZhbGlkYXRlcyBPcGVuQVBJIGRvY3VtZW50cwp3aXRoIEpTT04gU2NoZW1hLiBQZXIgT3BlbkFQSSdzIGRvY3VtZW50YXRpb24sIHRoaXMgbWF5IG5vdCBiZSBlbm91Z2ggdG8KcHJvcGVybHkgZW5jYXBzdWxhdGUgYWxsIHRoZSBudWFuY2VzIG9mIGEgc3BlY2lmaWNhdGlvbi4gSG93ZXZlciwgSlNPTiBTY2hlbWEgaXMKYWJsZSB0byBzdWNjZXNzZnVsbHkgdmFsaWRhdGUgdGhlIGN1cnJlbnQgT3BlbkFQSSAzLjEgU3BlY2lmaWNhdGlvbiB0ZXN0IHN1aXRlLgoKVmFsaWRhdGlvbiBpcyBhbiBhcmVhIHRoYXQgc3RpbGwgbmVlZHMgd29yay4gSWYgeW91IGRvIGZpbmQgY2FzZXMgd2hlcmUgdGhlCmN1cnJlbnQgdmFsaWRhdG9yIGlzIG5vdCBzdWZmaWNpZW50LCBwbGVhc2Ugb3BlbiBhbiBpc3N1ZSBzbyB0aGF0IHRoZSBsaWJyYXJ5CmNhbiBiZSB1cGRhdGVkIHdpdGggcHJvcGVyIGNvdmVyYWdlIG9mIHRoYXQgY2FzZS4KCiMjIERlcGVuZGVuY2llcwoKfCBEZXBlbmRlbmN5ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IFVzYWdlICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAp8IFtnaXRodWIuY29tL3NhbnRob3NoLXRla3VyaS9qc29uc2NoZW1hL3Y1XShodHRwczovL2dpdGh1Yi5jb20vc2FudGhvc2gtdGVrdXJpL2pzb25zY2hlbWEvdjUpIHwgdXNlZCBpbiB0aGUgYFN0ZFZhbGlkYXRvcmAgdG8gdmFsaWRhdGUgT3BlbkFQSSBkb2N1bWVudHMgJiBjb21wb25lbnRzICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBbZ2l0aHViLmNvbS9jaGFuY2VkL2NhcHMvdGV4dF0oaHR0cHM6Ly9naXRodWIuY29tL2NoYW5jZWQvY2FwcykgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IHVzZWQgZm9yIGFsbCBzdHJpbmcgZmllbGRzIHRvIHByb3ZpZGUgY2FzZSBjb252ZXJzaW9uIGFuZCBmdW5jdGlvbnMgZnJvbSBgc3RyaW5nc2AgYXMgbWV0aG9kcyB8CnwgW2dpdGh1Yi5jb20vY2hhbmNlZC9qc29ucG9pbnRlcl0oaHR0cHM6Ly9naXRodWIuY29tL2NoYW5jZWQvanNvbnBvaW50ZXIpICAgICAgICAgICAgICAgICAgICAgfCByZWxhdGl2ZSBsb2NhdGlvbnMgb2YgYWxsIG5vbi1zY2FsYXIgbm9kZXMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IFtnaXRodWIuY29tL3RpZHdhbGwvZ2pzb25dKGh0dHBzOi8vZ2l0aHViLmNvbS90aWR3YWxsL2dqc29uKSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgSlNPTiBwYXJzaW5nICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBbZ2l0aHViLmNvbS9jaGFuY2VkL2pzb254XShodHRwczovL2dpdGh1Yi5jb20vY2hhbmNlZC9qc29ueCkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IHJhdyBKU09OIHR5cGUgYW5kIGEgdG9vbGtpdCBmb3IganNvbiB0eXBlIGRldGVjdGlvbiAmIHBhcnNpbmcgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgW2dpdGh1Yi5jb20vY2hhbmNlZC9tYXBzXShodHRwczovL2dpdGh1Yi5jb20vY2hhbmNlZC9tYXBzKSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCBzbWFsbCB1dGlsaXR5IHVzZWQgdG8gc29ydCBgRXh0ZW5zaW9uc2AgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IFtnaXRodWIuY29tL2NoYW5jZWQvdHJhbnNjb2RlXShodHRwczovL2dpdGh1Yi5jb20vY2hhbmNlZC90cmFuc2NvZGUpICAgICAgICAgICAgICAgICAgICAgICAgIHwgdXNlZCB0byB0cmFuc2Zvcm0gWUFNTCBpbnRvIEpTT04gYW5kIHZpY2UgdmVyc2EgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBbZ2l0aHViLmNvbS9jaGFuY2VkL3VyaV0oaHR0cHM6Ly9naXRodWIuY29tL2NoYW5jZWQvdXJpKSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IHVzZWQgdG8gcmVwcmVzZW50IFVSSXMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgW2dpdGh1Yi5jb20vTWFzdGVybWluZHMvc2VtdmVyXShodHRwczovL2dpdGh1Yi5jb20vTWFzdGVybWluZHMvc2VtdmVyKSAgICAgICAgICAgICAgICAgICAgICAgfCBvcGVuYXBpIGZpZWxkIGFuZCB2ZXJzaW9uIGRldGVjdGlvbiBvZiBPcGVuQVBJIGRvY3VtZW50cyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IFtnb3BrZy5pbi95YW1sLnYzXShodHRwczovL2dvcGtnLmluL3lhbWwudjMpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgbmVlZGVkIHRvIHNhdGlzZnkgYHlhbWwuTWFyc2hhbGVyYCBhbmQgYHlhbWwuVW5tYXJzaGFsZXJgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBbZ2l0aHViLmNvbS9nb29nbGUvZ28tY21wXShodHRwczovL2dpdGh1Yi5jb20vZ29vZ2xlL2dvLWNtcCkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IHRlc3RpbmcgcHVycG9zZXMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CgojIyBDb250cmlidXRpb25zCgpQbGVhc2UgZmVlbCBmcmVlIHRvIG9wZW4gdXAgYW4gaXNzdWUgb3IgY3JlYXRlIGEgcHVsbCByZXF1ZXN0IGlmIHlvdSBmaW5kIGlzc3VlcwpvciBpZiB0aGVyZSBhcmUgZmVhdHVyZXMgeW91J2QgbGlrZSB0byBzZWUgYWRkZWQuCgojIyBMaWNlbnNlCgpNSVQK readmeEtag: '"148e14ce9c3d302f82a384f2ae0020f85d5818b9"' readmeLastModified: Sat, 01 Oct 2022 17:55:19 GMT repositoryId: 433235623 description: an OpenAPI 3.x library for go created: '2021-11-30T00:04:28Z' updated: '2024-05-30T08:21:32Z' language: Go archived: false stars: 26 watchers: 2 forks: 2 owner: chanced logo: https://avatars.githubusercontent.com/u/1073892?v=4 license: MIT repoEtag: '"2e82edb867610e6548c3e4cea028224669b4bd5e26423059bdf980db5f227828"' repoLastModified: Thu, 30 May 2024 08:21:32 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mermade/openapi3-examples v3: true repositoryMetadata: repositoryId: 84428973 description: Passing and failing test cases for OpenAPI 3.x validators created: '2017-03-09T10:24:25Z' updated: '2025-11-25T12:52:24Z' language: null archived: false stars: 28 watchers: 4 forks: 15 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: Apache-2.0 repoEtag: '"b90b2fdb527e641a11e0ca17e933015f66e65c60d0e212353329a18b1259d3ec"' repoLastModified: Tue, 25 Nov 2025 12:52:24 GMT foundInMaster: true id: a3fce9bef41c3b8feb3992adbd1f694e - source: openapi3 tags repository: https://github.com/typeable/comparest v3: true repositoryMetadata: base64Readme: >- # compaREST

[![Hackage](https://img.shields.io/hackage/v/compaREST.svg?logo=haskell)](https://hackage.haskell.org/package/compaREST)
[![MIT license](https://img.shields.io/badge/license-MIT-blue.svg)](LICENSE)

Compatibility checker for OpenAPI

## Using compaREST in Github Actions

Add the following step to your  Github Actions workflow:

```yaml
- uses: typeable/comparest
    if: ${{ github.event_name == 'pull_request' }}
    with:
      old: old-openapi.yaml
      new: new-openapi.yaml
```

The `new` and `old` values should be paths to your OpenAPI specifications you want to compare.

You will get something like this in your pull requests:

![](docs/img/github-action-report.png)

For more detail please see our [integration guide](docs/Integration_guide.md).

## An example

### Your situation

You are developing a very important server with a REST API. You have clients who use your API that you do not control. Say, you are also developing a mobile app that uses your API and you can't force someone to update to the latest version. (Or you prefer not to for UX reasons.)

You have recently released version 1.0 and things are going great: user are downloading your app, servers are processing requests.

You describe your API in a file `api-1.0.0.yaml`:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Swagger Petstore
  license:
    name: MIT
servers:
  - url: https://example.com
paths:
  /pets:
    get:
      parameters:
        - name: limit
          in: query
          required: false
          schema:
            type: integer
            maximum: 20
      responses:
        "200":
          description: ""
          headers:
            x-next:
              schema:
                type: string
          content:
            application/json:
              schema:
                $ref: "#/components/schemas/Pets"
    post:
      requestBody:
        content:
          application/json:
            schema:
              $ref: "#/components/schemas/Pet"
      responses:
        "201":
          description: ""
components:
  schemas:
    Pet:
      type: object
      required:
        - id
        - name
      properties:
        id:
          type: integer
        name:
          type: string
          minLength: 3
          maxLength: 10
    Pets:
      type: array
      items:
        $ref: "#/components/schemas/Pet"
```

### Evolving your product

Enthused over your initial success you hurry to release a new and improved version of your API and mobile app.

After a round of very intense programming you take a look at your new `api-1.1.0.yaml`:

```yaml
openapi: "3.0.0"
info:
  version: 1.1.0
  title: Swagger Petstore
  license:
    name: MIT
servers:
  - url: https://example.com
paths:
  /pets:
    get:
      parameters:
        - name: limit
          in: query
          required: false
          schema:
            type: integer
            maximum: 30
      responses:
        "200":
          description: ""
          headers:
            x-next:
              schema:
                type: string
          content:
            application/json:
              schema:
                $ref: "#/components/schemas/Pets"
    post:
      requestBody:
        content:
          application/json:
            schema:
              $ref: "#/components/schemas/Pet"
      responses:
        "201":
          description: ""
components:
  schemas:
    Pet:
      type: object
      required:
        - id
        - name
      properties:
        id:
          type: integer
        name:
          type: string
          minLength: 1
          maxLength: 15
        weight:
          type: integer
    Pets:
      type: array
      items:
        $ref: "#/components/schemas/Pet"
```

Looking at the very large and complex API description, you grow more and more concerned that your old mobile app might stop working when you update the server. But the spec is too large and too complex to reasonably assess this manually.

### Assessing compatibility automatically

Luckily, you have access to compaREST which can programmatically analyze your APIs and determine what, if anything, breaks compatibility and what doesn't.

You can call it, passing the API your client will be aware of, and the API your server will serve like so:

```bash
docker run --rm -v $(pwd):/data:rw typeable/comparest --client /data/api-1.0.0.yaml --server /data/api-1.1.0.yaml --output /data/report.md
```

Running this command will output a file `report.md`, containing the compatibility report between the two APIs:

> # Summary
>
> | [❌ Breaking changes](#breaking-changes) | [⚠️ Non-breaking changes](#non-breaking-changes) |
> |------------------------------------------|--------------------------------------------------|
> | 5                                        | 6                                                |
>
> # <span id="breaking-changes"></span>❌ Breaking changes
>
> ## **GET** /pets
>
> ### ⬅️☁️ JSON Response – 200
>
> #### `$[*].name(String)`
>
> 1.  Maximum length of the string changed from 10 to 15.
>
> 2.  Minimum length of the string changed from 3 to 1.
>
> ## **POST** /pets
>
> ### ➡️☁️ JSON Request
>
> #### `$.weight`
>
> 1.  Values are now limited to the following types:
>
>     -   Number
>
> 2.  The property was previously implicitly described by the catch-all
>     "additional properties" case. It is now explicitly defined.
>
> #### `$.weight(Number)`
>
> Value is now a multiple of 1.0.
>
> # <span id="non-breaking-changes"></span>⚠️ Non-breaking changes
>
> ## **GET** /pets
>
> ### Parameter limit
>
> #### JSON Schema
>
> ##### `$(Number)`
>
> Upper bound changed from 20.0 inclusive to 30.0 inclusive.
>
> ### ⬅️☁️ JSON Response – 200
>
> #### `$[*].weight`
>
> 1.  Values are now limited to the following types:
>
>     -   Number
>
> 2.  The property was previously implicitly described by the catch-all
>     "additional properties" case. It is now explicitly defined.
>
> #### `$[*].weight(Number)`
>
> Value is now a multiple of 1.0.
>
> ## **POST** /pets
>
> ### ➡️☁️ JSON Request
>
> #### `$.name(String)`
>
> 1.  Maximum length of the string changed from 10 to 15.
>
> 2.  Minimum length of the string changed from 3 to 1.

You now know exactly in what situations and in what way your 1.0 version of the app will break if you deploy your 1.1 version of the server.

### Additional formats

You can also produce a self-contained HTML report that you can open in your browser by simply omitting the file extension of the output file:

```bash
docker run --rm -v $(pwd):/data:rw typeable/comparest --client /data/api-1.0.0.yaml --server /data/api-1.1.0.yaml --output /data/report
```

## CLI docs

For more detail please see our [user guide](docs/User_guide.md).

```
Usage: comparest (-c|--client ARG) (-s|--server ARG)
                 [--silent | --only-breaking | --all] [-o|--output ARG]
                 [--folding-block-quotes-style | --header-style]
                 [--signal-exit-code]
  A tool to check compatibility between two OpenAPI specifications.

  Usage examples

      Compare files old.yaml with new.yaml and output the resulting report to
      stdout:

          comparest -c old.yaml -s new.yaml

      Only output breaking changes and write a styled HTML report to file
      report.html:

          comparest -c old.yaml -s new.yaml --only-breaking -o report

      Don't output anything, only fail if there are breaking changes:

          comparest -c old.json -s new.json --silent

      Write full report suitable for embedding into a GitHub comment to
      report.html:

          comparest -c old.json -s new.json --folding-block-quotes-style -o report.html

Available options:
  -h,--help                Show this help text
  -c,--client ARG          A path to the file containing the specification that
                           will be used for the client of the API. Can be either
                           a YAML or JSON file.
  -s,--server ARG          A path to the file containing the specification that
                           will be used for the server of the API. Can be either
                           a YAML or JSON file.
  --silent                 Silence all output. Only makes sense in combination
                           with --signal-exit-code.
  --only-breaking          Only report breaking changes in the output.
  --all                    Report both incompatible and compatible changes.
                           Compatible changes will not trigger a failure exit
                           code.
  -o,--output ARG          The file path where the output should be written. If
                           the option is omitted the result will be written to
                           stdout.

                           The file extension is used to determine the type of
                           the output file.

                           Supports many formats such as markdown, html, rtf,
                           doc, txt, rst, and many more.

                           Leave out the extension to produce a self-contained
                           HTML report with styling.
  --folding-block-quotes-style
                           The report tree is structured using summary/detail
                           HTML elements and indented using block quotes. This
                           style renders well on GitHub.Intended for HTML output
                           format. Markdown has rendering bugs on GitHub.
  --header-style           The report tree is structured using increasing levels
                           of headers.
  --signal-exit-code       Signal API compatibility with the exit code.

                           Exit with 0 if there are no breaking changes.
                           Exit with 1 if there are breaking changes.
                           Exit with 2 if could not determine compatibility.
```
 readmeEtag: '"9cadc60d8914e12b5fe7f44ec8cec9782c6dd8a9"' readmeLastModified: Mon, 29 Jul 2024 18:37:22 GMT repositoryId: 340356702 description: Compatibility checker for OpenAPI created: '2021-02-19T12:03:06Z' updated: '2025-12-01T22:53:41Z' language: Haskell archived: false stars: 26 watchers: 1 forks: 2 owner: typeable logo: https://avatars.githubusercontent.com/u/22911778?v=4 license: MIT repoEtag: '"e35d7d116b5e6744f6ecb8748dc21b7ad482767c2d0e7d87e48dac75356ec14a"' repoLastModified: Mon, 01 Dec 2025 22:53:41 GMT foundInMaster: true category: Parsers id: 7658feea5b221bdef8d18a4d78f0ae7f - source: openapi3 tags repository: https://github.com/chukmunnlee/caddy-openapi v3: true repositoryMetadata: base64Readme: >- # caddy-openapi

This middleware validates HTTP request and response against a OpenAPI V3 Specification file

## Installation

Build caddy with caddy-openapi, run `make`. This will build for Linux, Windows and OSX.

You can also build with `xcaddy`
```
xcaddy build \
    --with github.com/chukmunnlee/caddy-openapi
```

Tested with `go version go1.22.3 linux/amd64` on `Linux <name> 6.5.0-35-generic #35~22.04.1-Ubuntu SMP PREEMPT_DYNAMIC <timestamp> x86_64 x86_64 x86_64 GNU/Linux`

## Usage

### Caddyfile

Load `examples/customer/customer.yaml` file with defaults

```
:8080 {
  route /api {
    openapi ./examples/customer/customer.yaml
  }
}
```

One with all the options

```
:8080 {
  route /api {
    openapi {
      spec ./examples/customer/customer.yaml
      fall_through
      log_error
    }
  }
}
```

Reports any errors as a `{openapi.error}` [placeholder](https://caddyserver.com/docs/caddyfile/concepts#placeholders) which can be used in other [directives](https://caddyserver.com/docs/caddyfile/directives) like [`respond`](https://caddyserver.com/docs/caddyfile/directives/respond)

| Fields                   | Description |
|--------------------------|-------------|
| `spec <oas_file>`        | The OpenAPI3 YAML file. This attribute is a mandatory |
| `policy_bundle <bundle>` | [OPA](https://www.openpolicyagent.org/) policy bundle created with `opa build`. |
| `fall_through`           | Toggles fall through when the request does do match the provided OpenAPI spec. Default is `false` |
| `validate_servers`       | Enable server validation. Accepts `true`, `false` or just the directive which enables validation. Default is `true`. |
| `log_error`              | Toggles error logging. Default is `false` |
| `check`                  | Enable validation of the request parameters; include one or more of the following directives in the body:`req_params`, `req_body` and `resp_body`. `resp_body` only validates `application/json` payload. Note that validating the request body will implicitly set `req_params` |

Errors are reported in the following [placeholders](https://caddyserver.com/docs/caddyfile/concepts#placeholders). You can use them in other [directives](https://caddyserver.com/docs/caddyfile/directives) like [`respond`](https://caddyserver.com/docs/caddyfile/directives/respond)

| Placeholders             | Description |
|--------------------------|-------------|
| `openapi.error`          | Description of the error |
| `openapi.status_code`    | Suggested status code |
| `openapi.response_error` | Resonse error |

## Example

The following example validates all request, including query string as well as payloads, to `localhost:8080/api` 
against the `./examples/customer/customer.yaml` file.  Any non compliant request will be logged to Caddy's console. 
Respond to the client with the error `{openapi.error}`.

```
:8080 {

  @api {
    path /api/*
  }

  reverse_proxy @api {
    to localhost:3000
  }

  route @api {
    openapi {
      spec ./examples/customer/customer.yaml 
      policy_bundle ./examples/policy/bundle.tar.gz
      check {
        req_body 
        resp_body 
      }
      validate_servers
      log_error 
    }
  }

  handle_errors {
    respond @api "Resource: {http.request.orig_uri}. Error: {openapi.error}" {openapi.status_code}  {
      close
    }
  }
}
```

Try out the `customer.yaml` API by running the accompanying node application.

## Using OpenPolicyAgent

You can enforce policies on routes by adding the `x-policy` field to either the [OpenAPI3 document](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#schema) level, or the [path item](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#pathItemObject) level or or the [operation](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.1.0.md#operationObject) level. 

If a `x-policy` field is added at the
- *OpenAPI3 document* then the policy will be applied to all path
- *Path item* then the policy will be applied to all methods specified for that path eg `POST`, `GET` to `/api/v1/customer`
- *Operation* then the policy will only be applied to that operation eg. `GET/api/v1/customer`
`x-policy` attribute nested deeper into the 

The 'deeper' a `x-policy` field, the higher its precedence. Since `policy_bundle` is optional, no `x-policy` will be evaluated if no bundle are loaded.

Assume the following OPA policy file
```
package authz

default allow = false

allow {
  lower(input.method) = "get"
  array.slice(input.path, 0, 2) = [ "api", "customer" ]
  to_number(input.pathParams.custId) >= 100
}
```
has been bundled as `bundle.tar.gz`. Load it with `policy_bundle`

The following OpenAPI3 fragment show how you can evaluate `authz.allow` on all `GET /api/customer/`
```
paths:
  /api/customer/{custId}:
    get:
      description: Get customer
      operationId: getCustomer
      x-policy: authz.allow
      parameters:
      - name: custId
        in: path
        required: true
        schema:
           type: number
```

The HTTP request are converted into `input` according to the following table

| Fields                   | Description |
|--------------------------|-------------|
| `input.scheme`           | HTTP or HTTPS |
| `input.host`             | Host and port number |
| `input.method`           | HTTP method  |
| `input.path`             | Array of path elements eg. `/api/customer/123` is converted to `[ 'api', 'customer', '123' ]` |
| `input.remoteAddr`       | Host and port number of the client |
| `input.queryString`      | If a query string is present, the query string will be destructed into a map under `queryString` root. Example `?offset=10&limit=10` will be converted to the following keys: `input.queryString.offset` and `input.queryString.limit`. Query parameters with multiple value will have an array as its value. `queryString` will not be present if the request do not contain any query params |
| `input.pathParams`       | Like query string but a map of matched path parameters from the OpenAPI3 spec where parameter type is `in: path`. See above example |
| `input.headers`          | Map of all the request headers |
| `input.body`             | Access to the request's body. Only supports `application/json` content type. **Not implemented yet** |

Assume all values are string

This plugin currently can only work with policies/rules that return true or false.
 readmeEtag: '"a0da3c3c39baa725bb64411929b5a519a0991567"' readmeLastModified: Wed, 05 Jun 2024 00:13:19 GMT repositoryId: 288679362 description: >- A Caddy module to validate HTTP request and response against a OpenAPI spec (V3) file created: '2020-08-19T08:42:14Z' updated: '2025-12-21T22:10:43Z' language: Go archived: false stars: 28 watchers: 1 forks: 8 owner: chukmunnlee logo: https://avatars.githubusercontent.com/u/5455699?v=4 license: Apache-2.0 repoEtag: '"5ab37dae2b1ab4ec1adaf6df8a5dca1e25bc90a4cc3e1fddbbe328bdfeb585df"' repoLastModified: Sun, 21 Dec 2025 22:10:43 GMT foundInMaster: true category: Parsers id: 522bdf41bdbd2bfe5d70f1b36466741d - source: openapi3 tags repository: https://github.com/stackql/stackql-provider-registry v3: true id: ab1223f46183efa011bd7c836c344820 repositoryMetadata: base64Readme: >- PCEtLSBsYW5ndWFnZTogbGFuZy1ub25lIC0tPgpbIVtHaXRIdWIgQWN0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL3N0YWNrcWwvc3RhY2txbC1wcm92aWRlci1yZWdpc3RyeS9hY3Rpb25zL3dvcmtmbG93cy9tYWluLnltbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vc3RhY2txbC9zdGFja3FsLXByb3ZpZGVyLXJlZ2lzdHJ5L2FjdGlvbnMvd29ya2Zsb3dzL21haW4ueW1sKQohW0xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2Uvc3RhY2txbC9zdGFja3FsKQoKIyBTdGFja1FMIFByb3ZpZGVyIFJlZ2lzdHJ5CgpBIHJlcG9zaXRvcnkgb2YgYHByb3ZpZGVyYCBpbnRlcmZhY2UgZG9jdW1lbnRzIHN1cHBvcnRpbmcgW3N0YWNrcWxdKGh0dHBzOi8vc3RhY2txbC5pby8pIChbc3RhY2txbCBHaXRIdWIgcmVwb10oaHR0cHM6Ly9naXRodWIuY29tL3N0YWNrcWwvc3RhY2txbCkpLiAKCiMjIENvbnRleHQKCgpTdGFja1FMIHByb3ZpZGVyIGludGVyZmFjZSBkb2N1bWVudHMgaW5mb3JtIHRoZSBzdGFja3FsIGFwcGxpY2F0aW9uIG9uIGhvdyB0byBpbnRlcmFjdCB3aXRoIGEgZ2l2ZW4gcHJvdmlkZXIgKGxpa2UgYGF3c2AsIGBhenVyZWAsIGBnb29nbGVgLCBldGMpLCBpbmNsdWRpbmcgd2hhdCBtZXRob2RzIGFyZSBhdmFpbGFibGUgaW4gdGhlIHByb3ZpZGVyIGFuZCBob3cgdG8gaW52b2tlIHRoZXNlIHVzaW5nIFNRTCBzZW1hbnRpY3MuICBQcm92aWRlciBpbnRlcmZhY2UgZG9jdW1lbnRzIGFyZSBgeWFtbGAgZm9ybWF0dGVkLCBPcGVuQVBJIHNwZWNpZmljYXRpb25zIHdpdGggZXh0ZW5zaW9ucy4gIAoKVGhlIGRvY3VtZW50cyBhcmUgdmVyc2lvbmVkIHBlciBwcm92aWRlciBpbiB0aGlzIHJlcG9zaXRvcnksIGFuZCBidWlsdCBhcyBzaWduZWQgYW5kIGNvbXByZXNzZWQgYXMgcGFja2FnZWQgYXJ0aWZhY3RzLiAgVGhlIHBhY2thZ2VkIGFydGlmYWN0cyBhcmUgcmVnaXN0ZXJlZCBhbmQgcHVibGlzaGVkIHRvIHRoZSBTdGFja1FMIFByb3ZpZGVyIFJlZ2lzdHJ5IEFydGlmYWN0IFJlcG9zaXRvcnkgaW4gQVdTIFMzLiAgVGhlIHByb3ZpZGVyIHJlZ2lzdHJ5IEFQSSBpcyBhIFtEZW5vIERlcGxveV0oaHR0cHM6Ly9kZW5vLmNvbS9kZXBsb3kpIGFwcGxpY2F0aW9uIHRoYXQgc2VydmVzIHRoZSBwcm92aWRlciBpbnRlcmZhY2UgZG9jdW1lbnRzIHRvIHRoZSBzdGFja3FsIGFwcGxpY2F0aW9uIHVzaW5nIHRoZSBgUkVHSVNUUlkgTElTVGAgYW5kIGBSRUdJU1RSWSBQVUxMYCBjb21tYW5kcy4gIAoKVGhlIGZvbGxvd2luZyBkaWFncmFtIHNob3dzIHRoZSBjb250ZXh0IG9mIHRoZSBwcm92aWRlciByZWdpc3RyeTogIAoKYGBgbWVybWFpZApDNENvbnRleHQKICAgIFN5c3RlbV9FeHQoZ2l0aHViX3JlcG8sICJzdGFja3FsLXByb3ZpZGVyLXJlZ2lzdHJ5IiwgIkdpdEh1YiBSZXBvc2l0b3J5IikKICAgIFN5c3RlbV9FeHQoZ2l0aHViX2FjdGlvbnMsICJCdWlsZCBhbmQgRGVwbG95IiwgIkdpdEh1YiBBY3Rpb25zIikKICAgIFN5c3RlbURiKGFydGlmYWN0X3JlcG8sICJBcnRpZmFjdCBSZXBvc2l0b3J5IiwgIkFXUyBTMyIpCiAgICBTeXN0ZW0oZGVub19yZWdpc3RyeSwgIlByb3ZpZGVyIFJlZ2lzdHJ5IEFQSSIsICJEZW5vIERlcGxveSIpCiAgICBTeXN0ZW0oc3RhY2txbCwgIlN0YWNrUUwgQXBwbGljYXRpb24iLCAic3RhY2txbCIpCgogICAgUmVsKGdpdGh1Yl9yZXBvLCBnaXRodWJfYWN0aW9ucywgInRyaWdnZXJzLi4uIikKICAgIFJlbChnaXRodWJfYWN0aW9ucywgYXJ0aWZhY3RfcmVwbywgInJlZ2lzdGVycyBhbmQgcHVzaGVzIHRvLi4uIiwgInNpZ25lZCB0Z3ogcGFja2FnZSIpCiAgICBSZWwoZ2l0aHViX2FjdGlvbnMsIGRlbm9fcmVnaXN0cnksICJwdXNoZXMgdG8uLi4iLCAic2lnbmVkIHRneiBwYWNrYWdlIikKICAgIFJlbChzdGFja3FsLCBkZW5vX3JlZ2lzdHJ5LCAibGlzdCBhbmQgcHVsbHMgcmVnaXN0cnkgZG9jcyBmcm9tLi4uIiwgIlJFR0lTVFJZIExJU1QgfCBSRUdJU1RSWSBQVUxMIikKICAgIFVwZGF0ZUxheW91dENvbmZpZygkYzRTaGFwZUluUm93PSIzIiwgJGM0Qm91bmRhcnlJblJvdz0iMCIpCiAgICBVcGRhdGVSZWxTdHlsZShnaXRodWJfcmVwbywgZ2l0aHViX2FjdGlvbnMsICRvZmZzZXRZPSIxMCIsICRvZmZzZXRYPSItMjAiKQogICAgVXBkYXRlUmVsU3R5bGUoZ2l0aHViX2FjdGlvbnMsIGFydGlmYWN0X3JlcG8sICRvZmZzZXRZPSI0NCIsICRvZmZzZXRYPSItNTUiKQogICAgVXBkYXRlUmVsU3R5bGUoZ2l0aHViX2FjdGlvbnMsIGRlbm9fcmVnaXN0cnksICRvZmZzZXRZPSItMTgiLCAkb2Zmc2V0WD0iLTEzMCIpCiAgICBVcGRhdGVSZWxTdHlsZShzdGFja3FsLCBkZW5vX3JlZ2lzdHJ5LCAkb2Zmc2V0WT0iNDAiLCAkb2Zmc2V0WD0iLTQwIikKYGBgCgpUaGUgcHVibGljIFN0YWNrUUwgUHJvdmlkZXIgUmVnaXN0cnkgaXMgZGlzdHJpYnV0ZWQgdmlhIFtEZW5vIERlcGxveV0oaHR0cHM6Ly9kZW5vLmNvbS9kZXBsb3kpLCB1c2luZyB0aGUgZm9sbG93aW5nIGVuZHBvaW50czogIAoKfCBFbmRwb2ludCB8IERlc2NyaXB0aW9uIHwKfCAtLS0gfCAtLS0gfAp8IFtyZWdpc3RyeS5zdGFja3FsLmFwcF0oaHR0cHM6Ly9yZWdpc3RyeS5zdGFja3FsLmFwcC9waW5nKSB8IFByb2R1Y3Rpb24gcmVnaXN0cnkgKGJ1aWx0IGZyb20gYG1haW5gKSB8CnwgW3JlZ2lzdHJ5LWRldi5zdGFja3FsLmFwcF0oaHR0cHM6Ly9yZWdpc3RyeS5zdGFja3FsLmFwcC9waW5nKSB8IERldmVsb3BtZW50IHJlZ2lzdHJ5IChidWlsdCBmcm9tIGBkZXZlbG9wYCkgfAoKIyMgQ29udHJpYnV0aW5nCgpQbGVhc2Ugc2VlIFsuZ2l0aHViL0NPTlRSSUJVVElORy5tZF0oLy5naXRodWIvQ09OVFJJQlVUSU5HLm1kKS4KCiMjIERldmVsb3BpbmcgYSBQcm92aWRlcgoKU3RhY2tRTCBwcm92aWRlcnMgYXJlIGdlbmVyYXRlZCBmcm9tIHByb3ZpZGVyIE9wZW5BUEkgb3IgU3dhZ2dlciBzcGVjaWZpY2F0aW9ucyAoZWl0aGVyIHN1cHBsaWVkIGJ5IHRoZSBwcm92aWRlciBvciBjb25zdGl0dXRlZCB0aHJvdWdoIG90aGVyIHNjcmlwdHMgLSBmb3IgZXhhbXBsZSwgW2dvb2dsZS1kaXNjb3ZlcnktdG8tb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL3N0YWNrcWwvZ29vZ2xlLWRpc2NvdmVyeS10by1vcGVuYXBpKSBvciBbc3RhY2txbC1henVyZS1vcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vc3RhY2txbC9zdGFja3FsLWF6dXJlLW9wZW5hcGkpKS4gIAoKT25jZSB5b3UgaGF2ZSBhbiBPcGVuQVBJIHNwZWNpZmljYXRpb24sIHlvdSBjYW4gdXNlIHRoZSBbb3BlbmFwaXNhdXJ1c10oaHR0cHM6Ly9naXRodWIuY29tL3N0YWNrcWwvb3BlbmFwaXNhdXJ1cykgdXRpbGl0eSBwcm9qZWN0IHRvIGdlbmVyYXRlIGEgU3RhY2tRTCBwcm92aWRlciBkb2N1bWVudC4gIAoKIyMgQnVpbGQgYW5kIERlcGxveW1lbnQgV29ya2Zsb3cKClRoZSBwcm92aWRlciByZWdpc3RyeSBpcyBidWlsdCBhbmQgZGVwbG95ZWQgdXNpbmcgR2l0SHViIEFjdGlvbnMuICBQcm92aWRlciBkb2N1bWVudHMgYXJlIHZhbGlkYXRlZCBhbmQgdGVzdGVkIGluIHdvcmtmbG93IHN0ZXBzIGFuZCB0aGVuIHBhY2thZ2VkIGFuZCBzdG9yZWQgaW4gdGhlIGFydGlmYWN0IHJlcG9zaXRvcnkuICBUaGUgbW9zdCByZWNlbnQgcGFja2FnZWQgdmVyc2lvbnMgYXJlIHB1Ymxpc2hlZCB0byB0aGUgcmVnaXN0cnkgQVBJIChhIFtEZW5vIERlcGxveV0oaHR0cHM6Ly9kZW5vLmNvbS9kZXBsb3kpIGFwcGxpY2F0aW9uKSwgd2hlcmUgdGhleSBhcmUgYXZhaWxhYmxlIGZyb20gdGhlIGBzdGFja3FsYCBhcHBsaWNhdGlvbiB1c2luZyBgUkVHSVNUUlkgTElTVGAgb3IgYFJFR0lTVFJZIFBVTExgLiAgU2VlIFtkb2NzL2J1aWxkLWFuZC1kZXBsb3ltZW50Lm1kXShkb2NzL2J1aWxkLWFuZC1kZXBsb3ltZW50Lm1kKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4gIAoKIyMgVGVzdGluZyBhIFByb3ZpZGVyIHVzaW5nIHRoZSBgZGV2YCBSZWdpc3RyeQoKVXNlIHRoZSBmb2xsb3dpbmcgc3RlcHMgdG8gdGVzdCBhIHByb3ZpZGVyIHVzaW5nIHRoZSBgZGV2YCByZWdpc3RyeTogIAoKYGBgYmFzaApleHBvcnQgREVWX1JFRz0ieyBcInVybFwiOiBcImh0dHBzOi8vcmVnaXN0cnktZGV2LnN0YWNrcWwuYXBwL3Byb3ZpZGVyc1wiIH0iCi4vc3RhY2txbCAtLXJlZ2lzdHJ5PSIke0RFVl9SRUd9IiBzaGVsbApgYGA= readmeEtag: '"49bf6971ba5d86cf1233c839a45ebb41f6c1e236"' readmeLastModified: Wed, 31 Jul 2024 13:53:02 GMT repositoryId: 441087132 description: >- Registry for cloud and SaaS providers for StackQL, generated from extensions to the providers OpenAPI3 specification created: '2021-12-23T06:39:01Z' updated: '2026-01-20T11:01:25Z' language: Go archived: false stars: 27 watchers: 2 forks: 3 owner: stackql logo: https://avatars.githubusercontent.com/u/95105302?v=4 repoEtag: '"51ad0a86fd86249b196e17af83ef8dee35e31f31f16185a159d62f1d53a8dd43"' repoLastModified: Tue, 20 Jan 2026 11:01:25 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/shipengine/shipengine-openapi v3: true repositoryMetadata: base64Readme: >- IVtPcGVuQVBJIExvZ29dKGh0dHBzOi8vc2hpcGVuZ2luZS5naXRodWIuaW8vaW1nL29wZW5hcGktbG9nby5wbmcpIFNoaXBFbmdpbmXihKIgT3BlbkFQSSBEZWZpbml0aW9uCj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KClRoaXMgcmVwbyBjb250YWlucyB0aGUgb2ZmaWNpYWwgW09wZW5BUEkgMy4wXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMi5tZCkgZGVmaW5pdGlvbnMgZm9yIHRoZSBbU2hpcEVuZ2luZSBBUEldKGh0dHBzOi8vc2hpcGVuZ2luZS5jb20pLiAgWW91IGNhbiB1c2UgdGhlc2UgZGVmaW5pdGlvbnMgd2l0aCBhbnkgb2YgY291bnRsZXNzIFtPcGVuQVBJIHRvb2xzXShodHRwczovL29wZW5hcGkudG9vbHMvKSB0byBnZW5lcmF0ZSBzYW1wbGUgY29kZSwgdGVzdHMsIG1vY2sgc2VydmVycywgZXRjLgoKCldoaWNoIGZpbGUgdG8gdXNlCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCkRlcGVuZGluZyBvbiB5b3VyIHByZWZlcmVuY2VzIGFuZC9vciB0b29saW5nLCB5b3UgbWF5IGNob29zZSB0byB1c2Ugb25lIG9yIG1vcmUgb2YgdGhlIGZvbGxvd2luZyBmaWxlczoKCnxQYXRoICAgICAgICAgICAgICAgIHxEZXNjcmlwdGlvbgp8Oi0tLS0tLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCnxbYG9wZW5hcGkueWFtbGBdKG9wZW5hcGkueWFtbCkgICAgICAgIHxUaGUgZW50aXJlIFNoaXBFbmdpbmUgT3BlbkFQSSBkZWZpbml0aW9uLCBpbiBhIHNpbmdsZSBZQU1MIGZpbGUuICBUaGlzIGZpbGUgdXNlcyBbYCRyZWZgIHBvaW50ZXJzXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMi5tZCNyZWZlcmVuY2Utb2JqZWN0KSB0byByZWR1Y2UgZHVwbGljYXRpb24gYW5kIGtlZXAgdGhlIGZpbGUgc21hbGwgYW5kIGZhaXJseSBodW1hbi1yZWFkYWJsZS48YnI+PGJyPiBTb21lIFtPcGVuQVBJIHRvb2xzXShodHRwczovL29wZW5hcGkudG9vbHMvKSBkb24ndCBzdXBwb3J0IFlBTUwgb3IgZG9uJ3Qgc3VwcG9ydCBgJHJlZmAgcG9pbnRlcnMsIHNvIHlvdSBtYXkgbmVlZCB0byB1c2UgdGhlIFtgb3BlbmFwaS5qc29uYCBmaWxlXShvcGVuYXBpLmpzb24pIGluc3RlYWQuCnxbYG9wZW5hcGkuanNvbmBdKG9wZW5hcGkuanNvbikgICAgICAgIHxUaGUgZW50aXJlIFNoaXBFbmdpbmUgT3BlbkFQSSBkZWZpbml0aW9uLCBpbiBhIHNpbmdsZSBKU09OIGZpbGUuICBUaGlzIGZpbGUgKipkb2VzIG5vdCoqIGNvbnRhaW4gYW55IGAkcmVmYCBwb2ludGVycywgd2hpY2ggbWVhbnMgaXQgc2hvdWxkIHdvcmsgd2l0aCBhbnkgW09wZW5BUEkgdG9vbF0oaHR0cHM6Ly9vcGVuYXBpLnRvb2xzLykuCgoKT3RoZXIgQVBJIERlZmluaXRpb24gRm9ybWF0cwotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyMgIVtSZWRvYyBMb2dvXShodHRwczovL3NoaXBlbmdpbmUuZ2l0aHViLmlvL2ltZy9yZWRvYy1sb2dvLXNtYWxsLnBuZykKVmlldyB0aGUgU2hpcEVuZ2luZSBBUEkgZGVmaW5pdGlvbiBbb25saW5lIGluIHlvdXIgYnJvd3Nlcl0oaHR0cHM6Ly9zaGlwZW5naW5lLmdpdGh1Yi5pby9zaGlwZW5naW5lLW9wZW5hcGkvKS4gVGhpcyB3ZWIgcGFnZSBpcyBnZW5lcmF0ZWQgZnJvbSB0aGUgW09wZW5BUEkgZGVmaW5pdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL1NoaXBFbmdpbmUvc2hpcGVuZ2luZS1vcGVuYXBpKSB1c2luZyBbUmVEb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKS4KCgojIyMgIVtQb3N0bWFuIExvZ29dKGh0dHBzOi8vc2hpcGVuZ2luZS5naXRodWIuaW8vaW1nL3Bvc3RtYW4tbG9nby1zbWFsbC5wbmcpIFBvc3RtYW4KVGhlIG9mZmljaWFsIFtQb3N0bWFuIHJlZmVyZW5jZSBjb2xsZWN0aW9uXShodHRwczovL2RvY3VtZW50ZXIuZ2V0cG9zdG1hbi5jb20vdmlldy8zMDUyMDQvU1c3VzVWNm8pIGZvciBTaGlwRW5naW5lLiAgSnVzdCBpbXBvcnQgaXQgaW50byBbUG9zdG1hbl0oaHR0cHM6Ly9nZXRwb3N0bWFuLmNvbSkgYW5kIGltbWVkaWF0ZWx5IGJlZ2luIGludGVyYWN0aW5nIHdpdGggdGhlIFNoaXBFbmdpbmUgQVBJLiAKCgpOZXcgdG8gU2hpcEVuZ2luZT8gRG93bmxvYWQgb3VyIFt3YWxrdGhyb3VnaCBjb2xsZWN0aW9uIGluc3RlYWRdKGh0dHBzOi8vZG9jdW1lbnRlci5nZXRwb3N0bWFuLmNvbS92aWV3LzMwNTIwNC9TVzdYYkE2VikuCgoKIyMjICFbSlNPTiBTY2hlbWEgTG9nb10oaHR0cHM6Ly9zaGlwZW5naW5lLmdpdGh1Yi5pby9pbWcvanNvbi1zY2hlbWEtbG9nby1zbWFsbC5wbmcpIEpTT04gU2NoZW1hCklmIHlvdSBuZWVkIHRvIHZhbGlkYXRlIEFQSSByZXF1ZXN0cyBhbmQgcmVzcG9uc2VzLCB0aGVuIHlvdSBtYXkgd2FudCB0byB1c2UgW291ciBKU09OIFNjaGVtYSBkZWZpbml0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL1NoaXBFbmdpbmUvc2hpcGVuZ2luZS1qc29uLXNjaGVtYSkuCg== readmeEtag: '"905646882c040ce5f7cbff386973219b5fc0b66b"' readmeLastModified: Tue, 13 Aug 2024 01:33:22 GMT repositoryId: 155577069 description: The official OpenAPI 3.0 definitions for ShipEngine™ created: '2018-10-31T15:06:25Z' updated: '2026-02-06T01:39:28Z' language: null archived: false stars: 26 watchers: 22 forks: 16 owner: ShipEngine logo: https://avatars.githubusercontent.com/u/42709242?v=4 repoEtag: '"12edf10ff8a52c48f2e6ccf999eda36b6cb155ac9c1250ed2c6fde5b94933181"' repoLastModified: Fri, 06 Feb 2026 01:39:28 GMT foundInMaster: true category: Parsers id: ff407dc64fade9a4275ec5d32f20aab5 - source: openapi3 tags repository: https://github.com/contractual-dev/contractual v3: true id: 626dd00f9d9912930334e688a8dffb7a repositoryMetadata: base64Readme: >- <h1 align="center">Contractual</h1>

Contractual is a tool for managing API and data schemas as structured contracts. It ensures that schemas
are defined, versioned, and enforced across teams, whether for REST APIs, event-driven systems, or structured data
exchanges.

Common use cases include: \
🔹 Keeping API Contracts in Sync Between Backend and Frontend \
🔹 Generating Type-Safe Clients and Server Contracts \
🔹 Preventing Breaking Changes and Detecting Schema Drift \
🔹 Ensuring Consistency Between Backend and Data Teams \
🔹 Generating Language-Specific Types from a Shared Contract

By treating schemas as first-class entities, Contractual eliminates uncertainty at integration points, enabling backend,
frontend, and data engineering teams to maintain predictable and enforceable APIs and structured data across the entire
stack.

> Initially built for the **Node.js and TypeScript ecosystem**, Contractual is planned to support additional
> languages.

## 🚀 In Practice

### Install Contractual

To get started, install the Contractual CLI globally:

```bash
npm i -g @contractual/cli
```

### Initialize Your Project

Run the `init` command to scaffold a new project:

```bash
contractual init
```

This command creates the following project structure:

```
frontend/          # Your frontend application
server/            # Your server application
contractual/       # Contractual files
├── api.tsp        # TypeSpec API definition
├── specs/         # OpenAPI auto-generated specs
```

> Contractual works seamlessly with **monorepos**, **monoliths**, and distributed repositories.

### Define Your API

Write your API definition in the `api.tsp` file. For example:

```tsp
import "@typespec/http";
import "@typespec/openapi";
import "@typespec/openapi3";

using TypeSpec.Http;

@service({
  title: "Petstore API",
})
namespace PetstoreAPI;

model Pet {
  id: string;
  name: string;
}

@route("/pet")
@post
op addPet(@body body: Pet): Pet;
```

> You can experiment and validate your API definitions [using the TypeSpec playground](https://typespec.io/playground/)

### Manage API Changes

#### Save the Current State of Your API

Run the `spec graduate` command to save the current state of your OpenAPI spec:

```bash
contractual spec graduate
```

This will generate a new OpenAPI (3.1.0) YAML file with versioning, enabling to track API changes over time. The
updated structure will look like this:

```
contractual/
├── api.tsp                  # TypeSpec API definition
├── specs/                   # OpenAPI auto-generated specs
│   ├── openapi-v1.0.0.yaml
client/                      # Generated API clients
server/                      # Server contracts
e2e/                         # Type-safe API-driven tests
```

> You can track API evolution and changes easily with clear, versioned OpenAPI specs.

Here’s a quick video showing how this works:

<div align="center">
  <img src="spec-graduate.gif" />
</div>

### Generate Contracts

Run the `contract generate` command to generate type-safe clients, server contracts, and updated OpenAPI specs:

```bash
contractual contract generate
```

This command creates:

- **Type-safe client libraries** [using **ts-rest**](https://ts-rest.com), integrated with **Zod** for runtime
  validation.
- **Server contracts** for frameworks like **Express**, **Fastify**, and **NestJS**.
- **Updated OpenAPI specs**.

Here’s a short video showing contract generation in action:

<div align="center">
  <img src="contract-generate.gif" />
</div>
```

## 🔍 Why Contractual?

Maintaining the consistency of schemas across various services presents significant challenges. As systems evolve,
type-definitions and schemas drift, unnoticed breaking changes occur, and different teams find it challenging to
synchronize. APIs, event schemas, and structured data formats often become disconnected from their original intent,
leading to brittle integrations, manual fixes, and unexpected failures.

**Some of the biggest pain points teams face include:**

- **Schema Drift & Misalignment:** APIs and data contracts become inconsistent across teams, leading to mismatches, broken integrations, and regressions.

- **Untracked Changes & Breaking Updates:** Without tracking modifications, updates can unexpectedly break consumers, causing downtime and costly debugging.

- **Scattered Schemas & Code Maintenance:** Outdated documentation and manually managed type definitions create unreliable integrations and make maintaining entity models error-prone.

## 🔑 The Contract-First Approach
Most teams take a **code-first** approach to API development, where schemas are generated after implementation. This often results in **misalignment between services, outdated documentation, and accidental breaking changes.** Backend teams define APIs, frontend teams consume them, and data engineers rely on structured data formats—all of which can drift over time when schemas are an afterthought.

A **contract-first** approach flips this process: schemas are designed before any implementation begins, ensuring that API structures, event definitions, and data formats remain stable and predictable. This approach allows teams to:

- Define schemas upfront and enforce them as the single source of truth.

- Track changes and prevent breaking updates before they impact consumers.

- Generate type-safe clients and server contracts in multiple languages, reducing friction between teams.

## 📘 Roadmap

Want to contribute? Check out the alpha version [Roadmap](https://github.com/contractual-dev/contractual/issues/8) and
join the journey! 🚀

## ❤️ Join the Community

Contractual is open-source, and we’re looking for contributors to help shape its future, if you’re interested in
collaborating, please reach out.

📩 **Feedback or Questions?** Reach out
via [GitHub Discussions](https://github.com/contractual-dev/contractual/discussions).

## 🔒 License

Licensed under [MIT](LICENSE).
 readmeEtag: '"eeff71ed86a8bbd6bfc24ab55e6eafbf68fc7332"' readmeLastModified: Fri, 07 Feb 2025 15:37:24 GMT repositoryId: 917350111 description: >- Contractual is a tool for managing API and data schemas as structured contracts. It ensures that schemas are defined, versioned, and enforced across teams, whether for REST APIs, event-driven systems, or structured data exchanges. created: '2025-01-15T20:18:28Z' updated: '2026-01-10T18:10:07Z' language: TypeScript archived: false stars: 33 watchers: 3 forks: 0 owner: contractual-dev logo: https://avatars.githubusercontent.com/u/195186508?v=4 license: MIT repoEtag: '"98056c624066e87ff7a2951dc0a4dd193eda9202e2d8e1b86b3cb6a4306ef421"' repoLastModified: Sat, 10 Jan 2026 18:10:07 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/curvednebula/apibake-js v3: true id: bf23379a010e0d69a29b65a9831316fd repositoryMetadata: base64Readme: >- IyBBcGlCYWtlCgpDb252ZXJ0IE9wZW5BUEkgc3BlYyB0byBQREYuIFN1cHBvcnRzIE9wZW5BUEkgMy4wLjArIGpzb24gYW5kIHlhbWwuCgojIyBRdWljayBTdGFydAoKTm9kZS5qcyAxNisgcmVxdWlyZWQuCgpgYGAKbnBtIGluc3RhbGwgLWcgYXBpYmFrZQoKYXBpYmFrZSA8b3BlbmFwaS5qc29ufC55YW1sfGZvbGRlci1uYW1lPiBbPGZpbGUtb3ItZm9sZGVyMj4gPGZpbGUtb3ItZm9sZGVyMz4gLi4uXSBbPG9wdGlvbnM+XQpgYGAKCioqT3B0aW9uczoqKgoKYGBgCiAtLW91dCA8c3RyaW5nPjogT3V0cHV0IFBERiBmaWxlIG5hbWUuCiAtLXRpdGxlIDxzdHJpbmc+OiBEb2N1bWVudCB0aXRsZS4KIC0tc3VidGl0bGUgPHN0cmluZz46IERvY3VtZW50IHN1YiB0aXRsZS4KIC0tc2VwYXJhdGUtc2NoZW1hczogV2hlbiBtdWx0aXBsZSBBUEkgZmlsZXMgcGFyc2VkLCBjcmVhdGUgc2VwYXJhdGUgc2NoZW1hcyBzZWN0aW9uIGZvciBlYWNoLgogLS1mb290ZXIgPHN0cmluZz46IERlZmluZXMgY29udGVudCBvZiBjb21tb24gcGFnZSBmb290ZXIuIE9wdGlvbnM6ICJwYWdlLW51bWJlciIuIFRvIHR1cm4gb2ZmOiAiIi4KIC0tY29uZmlnIDxzdHJpbmc+OiBQYXRoIHRvIGFwaWJha2UtY29uZmlnLmpzb24uIFNlZSAtLWV4cG9ydC1jb25maWcuCiAtLWV4cG9ydC1jb25maWc6IFNhdmUgZGVmYXVsdCBjb25maWcgaW50byBqc29uIGZpbGUgZm9yIGVkaXRpbmcuCiAtaDogU2hvdyB0aGlzIGhlbHAuCmBgYAoKIyMgRXhhbXBsZXMKClNwZWNpZnkgdGl0bGUgYW5kIHN1YnRpdGxlIGZvciB5b3VyIFBERjoKCmBgYAphcGliYWtlIG9wZW5hcGkuanNvbiAtLXRpdGxlICdSRVNUIEFQSSBTcGVjJyAtLXN1YnRpdGxlICdjcmVhdGVkIHdpdGggQXBpQmFrZScKYGBgCgpDb21iaW5lIHNldmVyYWwgT3BlbkFQSSBzcGVjcyBpbnRvIG9uZSBQREY6CgpgYGAKYXBpYmFrZSBhcGkxLmpzb24gYXBpMi55YW1sIC0tdGl0bGUgJ1JFU1QgQVBJIFNwZWMnCmFwaWJha2UgZGlyL3dpdGgvb3BlbmFwaS1zcGVjcyAtLXRpdGxlICdSRVNUIEFQSSBTcGVjJwpgYGAKCiMjIEN1c3RvbSBjb25maWc6IGZvbnRzLCBjb2xvcnMsIHBhZ2UgbWFyZ2lucy4KClRvIG1vZGlmeSBkZWZhdWx0IGFwaWJha2UgY29uZmlnIC0gZmlyc3QgZXhwb3J0IGl0IGludG8gYSBmaWxlOgoKYGBgCmFwaWJha2UgLS1leHBvcnQtY29uZmlnCmBgYAoKTW9kaWZ5IGFwaWJha2UtY29uZmlnLmpzb24gYW5kIHRlbGwgYXBpYmFrZSB0byB1c2UgaXQ6CgpgYGAKYXBpYmFrZSBvcGVuYXBpLmpzb24gLS1jb25maWcgYXBpYmFrZS1jb25maWcuanNvbgpgYGAKClBERiBkZWZhdWx0IGZvbnRzIGNhbiBiZSBzcGVjaWZpZWQgYnkgdGhlaXIgbmFtZXM6CgogLSBDb3VyaWVyCiAtIENvdXJpZXItQm9sZAogLSBDb3VyaWVyLU9ibGlxdWUKIC0gQ291cmllci1Cb2xkT2JsaXF1ZQogLSBIZWx2ZXRpY2EKIC0gSGVsdmV0aWNhLUJvbGQKIC0gSGVsdmV0aWNhLU9ibGlxdWUKIC0gSGVsdmV0aWNhLUJvbGRPYmxpcXVlCiAtIFN5bWJvbAogLSBUaW1lcy1Sb21hbgogLSBUaW1lcy1Cb2xkCiAtIFRpbWVzLUl0YWxpYwogLSBUaW1lcy1Cb2xkSXRhbGljCiAtIFphcGZEaW5nYmF0cwoKQWx0ZXJuYXRpdmVseSwgZXh0ZXJuYWwgZm9udCBmaWxlcyBjYW4gYmUgc3BlY2lmaWVkLiBTdXBwb3J0ZWQgZm9udCBmb3JtYXRzOiBUcnVlVHlwZSAoLnR0ZiksIE9wZW5UeXBlICgub3RmKSwgV09GRiwgV09GRjIsIFRydWVUeXBlIENvbGxlY3Rpb24gKC50dGMpLCBhbmQgRGF0YWZvcmsgVHJ1ZVR5cGUgKC5kZm9udCkgZm9udHMuCgpFeGFtcGxlOgoKYGBgCnsKICAiZm9udCI6IHsKICAgICJiYXNlU2l6ZSI6IDEwLAogICAgIm1haW4iOiB7CiAgICAgICJub3JtIjogewogICAgICAgICJmYWNlIjogImZvbnRzL1JvYm90by1SZWd1bGFyLnR0ZiIKICAgICAgfSwKICAgICAgImJvbGQiOiB7CiAgICAgICAgImZhY2UiOiAiZm9udHMvUm9ib3RvLUJvbGQudHRmIgogICAgICB9LAogICAgICAiaXRhbGljIjogewogICAgICAgICJmYWNlIjogImZvbnRzL1JvYm90by1JdGFsaWMudHRmIgogICAgICB9CiAgICB9LAogICAgIm1vbm8iOiB7CiAgICAgICJub3JtIjogewogICAgICAgICJmYWNlIjogIkNvdXJpZXIiCiAgICAgIH0sCiAgICAgICJib2xkIjogewogICAgICAgICJmYWNlIjogIkNvdXJpZXItQm9sZCIKICAgICAgfSwKICAgICAgIml0YWxpYyI6IHsKICAgICAgICAiZmFjZSI6ICJDb3VyaWVyLU9ibGlxdWUiCiAgICAgIH0KICAgIH0KICB9LAogIC4uLgp9CmBgYAoKTm90ZTogaWYgZm9udCBmaWxlIGlzIGEgY29sbGVjdGlvbiAoLnR0YykgLSB0aGVuIGZvbnQgc3R5bGUgbXVzdCBiZSBhbHNvIHNwZWNpZmllZC4gRXhhbXBsZToKCmBgYAogImZvbnQiOiB7CiAgICAibWFpbiI6IHsKICAgICAgIm5vcm0iOiB7CiAgICAgICAgImZhY2UiOiAiZm9udHMvUm9ib3RvLnR0YyIsCiAgICAgICAgInN0eWxlIjogIlJvYm90by1SZWd1bGFyIgogICAgICB9LAogICAgICAiYm9sZCI6IHsKICAgICAgICAiZmFjZSI6ICJmb250cy9Sb2JvdG8udHRjIiwKICAgICAgICAic3R5bGUiOiAiUm9ib3RvLUJvbGQiCiAgICAgIH0sCiAgICB9LAogICAgLi4uCiB9CmBgYAoKIyBNSVQgTGljZW5zZQoKQ29weXJpZ2h0IChjKSAyMDIzIEN1cnZlZE5lYnVsYS5jb20KClBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhIGNvcHkKb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzICh0aGUgIlNvZnR3YXJlIiksIHRvIGRlYWwKaW4gdGhlIFNvZnR3YXJlIHdpdGhvdXQgcmVzdHJpY3Rpb24sIGluY2x1ZGluZyB3aXRob3V0IGxpbWl0YXRpb24gdGhlIHJpZ2h0cwp0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsCmNvcGllcyBvZiB0aGUgU29mdHdhcmUsIGFuZCB0byBwZXJtaXQgcGVyc29ucyB0byB3aG9tIHRoZSBTb2Z0d2FyZSBpcwpmdXJuaXNoZWQgdG8gZG8gc28sIHN1YmplY3QgdG8gdGhlIGZvbGxvd2luZyBjb25kaXRpb25zOgoKVGhlIGFib3ZlIGNvcHlyaWdodCBub3RpY2UgYW5kIHRoaXMgcGVybWlzc2lvbiBub3RpY2Ugc2hhbGwgYmUgaW5jbHVkZWQgaW4gYWxsCmNvcGllcyBvciBzdWJzdGFudGlhbCBwb3J0aW9ucyBvZiB0aGUgU29mdHdhcmUuCgpUSEUgU09GVFdBUkUgSVMgUFJPVklERUQgIkFTIElTIiwgV0lUSE9VVCBXQVJSQU5UWSBPRiBBTlkgS0lORCwgRVhQUkVTUyBPUgpJTVBMSUVELCBJTkNMVURJTkcgQlVUIE5PVCBMSU1JVEVEIFRPIFRIRSBXQVJSQU5USUVTIE9GIE1FUkNIQU5UQUJJTElUWSwKRklUTkVTUyBGT1IgQSBQQVJUSUNVTEFSIFBVUlBPU0UgQU5EIE5PTklORlJJTkdFTUVOVC4gSU4gTk8gRVZFTlQgU0hBTEwgVEhFCkFVVEhPUlMgT1IgQ09QWVJJR0hUIEhPTERFUlMgQkUgTElBQkxFIEZPUiBBTlkgQ0xBSU0sIERBTUFHRVMgT1IgT1RIRVIKTElBQklMSVRZLCBXSEVUSEVSIElOIEFOIEFDVElPTiBPRiBDT05UUkFDVCwgVE9SVCBPUiBPVEhFUldJU0UsIEFSSVNJTkcgRlJPTSwKT1VUIE9GIE9SIElOIENPTk5FQ1RJT04gV0lUSCBUSEUgU09GVFdBUkUgT1IgVEhFIFVTRSBPUiBPVEhFUiBERUFMSU5HUyBJTiBUSEUKU09GVFdBUkUuCg== readmeEtag: '"99e8b092bf3a384ed53c1dddb9bfeac2ed13457d"' readmeLastModified: Mon, 20 May 2024 19:20:55 GMT repositoryId: 683452181 description: Convert OpenAPI to PDF. created: '2023-08-26T16:09:46Z' updated: '2025-11-12T14:43:40Z' language: TypeScript archived: false stars: 33 watchers: 2 forks: 4 owner: curvednebula logo: https://avatars.githubusercontent.com/u/1443238?v=4 repoEtag: '"5c5df2ea54914d26cdc7acdaf11af486fbdf0f79157efc49d6c804d9eaedff04"' repoLastModified: Wed, 12 Nov 2025 14:43:40 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/liveramp/reslang v3: true repositoryMetadata: base64Readme: >- # Reslang: API definitions made easy

Reslang is a simple language for describing resource-oriented APIs & turning them into Swagger. It produces Swagger/OpenAPI which is fully aligned with [LiveRamp's API standards](https://liveramp.atlassian.net/wiki/spaces/CI/pages/1014498273/RFC+API-3+LiveRamp+API+Standards).

It can also generate an [AsyncAPI specification](https://www.asyncapi.com/), describing events, from the same spec. Recently we have added the ability to generate [JSON schema](https://json-schema.org/) from the Reslang spec too.

[Release notes](./docs/releases.md)

Reslang is licensed under [Apache V2](https://www.apache.org/licenses/LICENSE-2.0)

## Docs

| Topic                                                    | Description                                                                                  |
| -------------------------------------------------------- | -------------------------------------------------------------------------------------------- |
| [Why reslang?](./docs/why.md)                            | Why do we need this tool and what does it offer?                                             |
| [Intro tutorial](./docs/intro.md)                        | Describes making a toy API for manipulating files and directories.                           |
| [The API paradigm](./docs/paradigm.md)                   | What is the Reslang paradigm and how does it relate to REST, OO and RPC                      |
| [Event intro tutorial](./docs/intro-events.md)           | Describes a simple example of how to generate AsyncAPI to describe events                    |
| [Sections](./docs/sections.md)                           | This explains how to reorder / group the presentation of your resources                      |
| [Direct2Dist API](./docs/direct2dist-explanation.md)     | This is a more complex example which recreates the entire Direct2Dist API.                   |
| [MULTI verbs](./docs/multi.md)                           | Describes MULTI- verbs such as MULTIPOST, MULTIPATCH etc                                     |
| [Reference manual](./docs/reference.md)                  | This describes the syntax and features in more depth.                                        |
| [Server blocks](./docs/server-blocks.md)                 | This explains how to define servers for REST and events APIs                                 |
| [JSON schema generation](./docs/jsonschema.md)           | This explains how to generate JSON schema definitions from Reslang code                      |
| [Internal System Design](./docs/ReslangSystemDesign.pdf) | Explains how Reslang is structured as a typescript application, and how the transpiler works |
| [VSCode Syntax highlighting](./vscode/README.md)         | Explains how to install VSCode syntax highlighting. Hint search extensions for Reslang!      |

## Installation

### For local use

1. ensure node & yarn are installed

(This is the node.js stack - please see here for installation instructions: [node.js](https://nodejs.org/en/download/) & [yarn](https://classic.yarnpkg.com/en/docs/install/#mac-stable))

2. clone the reslang repo
3. yarn install
4. ./install-reslang

Note that step 4 installs it as a command line tool using "yarn link"

## Running

Test it out by typing:

    reslang

This should bring up the options.

<details>
  <summary>Click to see Installation options</summary>

## Running in Docker (for internal LiveRamp users only)

Individuals who do not want to build Reslang from scratch are free to use the `reslang-docker` script which provides convenient, but limited, functionality with a reslang container.
This script outputs the generated swagger to STDOUT and requires an absolute path to function.

```
    bash ./reslang-docker.sh $PWD/models/distribution
```

</details>

## Creating & viewing the Swagger / OpenAPI

To create swagger, you first create a reslang file. Then you simply ask the reslang program to turn this into swagger.

Note that the models directory has a set of example definitions.

The following copies the swagger to the clipboard and opens the ReDoc browser for you.

    reslang models/simple-resource --open

If you want just want to print to stdout use:

    reslang models/simple-resource --stdout

## Viewing Swagger / OpenAPI in Swagger editor

If you want to use the Swagger editor, add --web to the cmd line:

    reslang models/simple-resource --open --web

Then paste the clipboard into the left window pane of the editor.

## Creating a graphical view

Reslang can generate dotviz output, which provides a nice graphical view of the resources.

The following command will copy the dotviz output to the clipboard.

    reslang models/simple-resource --open --diagram main

This will open your browser at a nice graphviz online editor. Paste the clipboard into the editor and you will get your graphical view of your API - a resource diagram.

## Viewing events via AsyncAPI

Reslang can generate AsyncAPI, describing the events a REST API can generate. It also provides an "event" keyword which you can use to describe adhoc events.

    reslang models/eventing --open --events

If instead you wish to view the events in the AsyncAPI Playgrounder for editing, please add the --web option.

    reslang models/eventing --open --events --web

Note that the AsyncAPI spec has been copied to the clipboard - you will need to paste it into the opened editor, on the left.

## Pretty Printing - Stripping out the comments for review

Reslang can produce a nice pretty-printed, stripped down version of the Reslang in html. It is often easier to review this form, as it removes the comments and error structures.

    reslang models/resources --stripped --open

Will open a browser on the stripped down file. If you just want plain text, add --plain

## Requesting new features

New features and enhancements that involve changing LiveRamp's API standards
will not be considered until the [API standards proposal process][standards process] is complete.

Other enhancements should be requested by opening a GitHub Issue using the
appropriate issue template or by opening a Pull Request and having your changes
reviewed by the API Squad.

[standards process]: https://docs.google.com/document/d/1X3fcFYLdsRzsQeX89zdmCOoisNsD6X-wcrtYcb8MkGA/edit
 readmeEtag: '"341f1bfef251d4848c3ed009ef4013654b3c046a"' readmeLastModified: Mon, 01 Apr 2024 08:11:04 GMT repositoryId: 195102746 description: >- A language for describing resource-oriented APIs & turning them into Swagger or resource diagrams. Oriented around the concepts we want to expose in the APIs. created: '2019-07-03T17:51:56Z' updated: '2025-10-22T07:29:24Z' language: TypeScript archived: false stars: 24 watchers: 75 forks: 6 owner: LiveRamp logo: https://avatars.githubusercontent.com/u/2374529?v=4 license: Apache-2.0 repoEtag: '"1e4ad56a91d9c594ceecdd97e4228e912553ca9f801447d3c85e81eaaa42303a"' repoLastModified: Wed, 22 Oct 2025 07:29:24 GMT foundInMaster: true category: - Documentation - Server Implementations id: 9b8ed2b3568ac6aa6c9cbbd70e4574b4 - source: openapi3 tags repository: https://github.com/eikek/sbt-openapi-schema v3: true repositoryMetadata: base64Readme: >- # SBT OpenApi Schema Codegen

[![CI](https://github.com/eikek/sbt-openapi-schema/actions/workflows/ci.yml/badge.svg)](https://github.com/eikek/sbt-openapi-schema/actions/workflows/ci.yml)
[![Scala Steward badge](https://img.shields.io/badge/Scala_Steward-helping-blue.svg?style=flat-square&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA4AAAAQCAMAAAARSr4IAAAAVFBMVEUAAACHjojlOy5NWlrKzcYRKjGFjIbp293YycuLa3pYY2LSqql4f3pCUFTgSjNodYRmcXUsPD/NTTbjRS+2jomhgnzNc223cGvZS0HaSD0XLjbaSjElhIr+AAAAAXRSTlMAQObYZgAAAHlJREFUCNdNyosOwyAIhWHAQS1Vt7a77/3fcxxdmv0xwmckutAR1nkm4ggbyEcg/wWmlGLDAA3oL50xi6fk5ffZ3E2E3QfZDCcCN2YtbEWZt+Drc6u6rlqv7Uk0LdKqqr5rk2UCRXOk0vmQKGfc94nOJyQjouF9H/wCc9gECEYfONoAAAAASUVORK5CYII=)](https://scala-steward.org)
[![License](https://img.shields.io/github/license/eikek/sbt-openapi-schema.svg?style=flat-square&color=steelblue)](https://github.com/eikek/sbt-openapi-schema/blob/master/LICENSE.txt)


This is an sbt plugin to generate Scala or Elm code given an openapi
3.x specification. Unlike other codegen tools, this focuses only on
the `#/components/schema` section. Also, it generates immutable
classes and optionally the corresponding JSON conversions.

- Scala: `case class`es are generated and JSON conversion via circe.
- Elm: records are generated and constructors for "empty" values. It
  works only for objects. JSON conversion is generated using Elm's
  default encoding support and the
  [json-decode-pipeline](https://github.com/NoRedInk/elm-json-decode-pipeline)
  module for decoding.
- JSON support is optional.

The implementation is based on the
[swagger-parser](https://github.com/swagger-api/swagger-parser)
project.

It is possible to customize the code generation.

## Usage

Add this plugin to your build in `project/plugins.sbt`:

```
addSbtPlugin("com.github.eikek" % "sbt-openapi-schema" % "x.y.z")
```

Please check the git tags or maven central for the current version.
Then enable the plugin in some project:

```
enablePlugins(OpenApiSchema)
```

There are two required settings: `openapiSpec` and
`openapiTargetLanguage`. The first defines the openapi.yml file and
the other is a constant from the `Language` object:

```scala
import com.github.eikek.sbt.openapi._

project.
  enablePlugins(OpenApiSchema).
  settings(
    openapiTargetLanguage := Language.Scala
    openapiSpec := (Compile/resourceDirectory).value/"test.yml"
  )
```

The sources are automatically generated when you run `compile`. The
task `openapiCodegen` can be used to run the generation independent
from the `compile` task.

## Configuration

The configuration is specific to the target language. There exists a
separate configuration object for Scala and Elm.

The key `openapiScalaConfig` defines some configuration to customize
the code generation.

For Scala, it looks like this:
```scala
case class ScalaConfig(
    mapping: CustomMapping = CustomMapping.none,
    json: ScalaJson = ScalaJson.none
) {

  def withJson(json: ScalaJson): ScalaConfig =
    copy(json = json)

  def addMapping(cm: CustomMapping): ScalaConfig =
    copy(mapping = mapping.andThen(cm))

  def setMapping(cm: CustomMapping): ScalaConfig =
    copy(mapping = cm)
}
```

By default, no JSON support is added to the generated classes. This
can be changed via:

```
openapiScalaConfig := ScalaConfig().withJson(ScalaJson.circeSemiauto)
```

This generates the encoder and decoder using
[circe](https://github.com/circe/circe). Note, that this plugin
doesn't change your `libraryDependencies` setting. So you need to add
the circe dependencies yourself.

The `CustomMapping` class allows to change the class names or use
different types (for example, you might want to change `LocalDate` to
`Date`).

It looks like this:
```scala
trait CustomMapping { self =>

  def changeType(td: TypeDef): TypeDef

  def changeSource(src: SourceFile): SourceFile

  def andThen(cm: CustomMapping): CustomMapping = new CustomMapping {
    def changeType(td: TypeDef): TypeDef =
      cm.changeType(self.changeType(td))

    def changeSource(src: SourceFile): SourceFile =
      cm.changeSource(self.changeSource(src))
  }
}
```

There are convenient constructors in its companion object.

It allows to use different types via `changeType` or change the source
file. Here is a `build.sbt` example snippet:

```scala
import com.github.eikek.sbt.openapi._

val CirceVersion            = "0.14.1"
libraryDependencies ++= Seq(
  "io.circe" %% "circe-generic" % CirceVersion,
  "io.circe" %% "circe-parser"  % CirceVersion
)

openapiSpec := (Compile/resourceDirectory).value/"test.yml"
openapiTargetLanguage := Language.Scala
Compile/openapiScalaConfig := ScalaConfig()
    .withJson(ScalaJson.circeSemiauto)
    .addMapping(CustomMapping.forType({ case TypeDef("LocalDateTime", _) =>
      TypeDef("Timestamp", Imports("com.mypackage.Timestamp"))
    }))
    .addMapping(CustomMapping.forName({ case s => s + "Dto" }))

enablePlugins(OpenApiSchema)
```

It adds circe JSON support and changes the name of all classes by
appending the suffix "Dto". It also changes the type used for local
dates to be `com.mypackage.Timestamp`.


## Elm

There is some experimental support for generating Elm data structures
and corresponding JSON conversion functions. When using the
`decodePipeline` json variant, you need to install these packages:

```
elm install elm/json
elm install NoRedInk/elm-json-decode-pipeline
```

The default output path for elm sources is `target/elm-src`. So in
your `elm.json` file, add this directory to the `source-directories`
list along with the main source dir. It may look something like this:

```json
{
    "type": "application",
    "source-directories": [
        "modules/webapp/target/elm-src",
        "modules/webapp/src/main/elm"
    ],
    "elm-version": "0.19.0",
    "dependencies": {
        "direct": {
            "NoRedInk/elm-json-decode-pipeline": "1.0.0",
            "elm/browser": "1.0.1",
            "elm/core": "1.0.2",
            "elm/html": "1.0.0",
            "elm/json": "1.1.3"
        },
        "indirect": {
            "elm/time": "1.0.0",
            "elm/url": "1.0.0",
            "elm/virtual-dom": "1.0.2"
        }
    },
    "test-dependencies": {
        "direct": {},
        "indirect": {}
    }
}
```

It always generates type aliases for records.

While source files for scala are added to sbt's `sourceGenerators` so
that they get compiled with your sources, the elm source files are not
added anywhere, because there is no support for Elm in sbt. However,
in the `build.sbt` file, you can tell sbt to generate the files before
compiling your elm app. This can be configured to run during resource
generation. Example:

``` scala

// Put resulting js file into the webjar location
def compileElm(logger: Logger, wd: File, outBase: File, artifact: String, version: String): Seq[File] = {
  logger.info("Compile elm files ...")
  val target = outBase/"META-INF"/"resources"/"webjars"/artifact/version/"my-app.js"
  val proc = Process(Seq("elm", "make", "--output", target.toString) ++ Seq(wd/"src"/"main"/"elm"/"Main.elm").map(_.toString), Some(wd))
  val out = proc.!!
  logger.info(out)
  Seq(target)
}

val webapp = project.in(file("webapp")).
  enablePlugins(OpenApiSchema).
  settings(
    openapiTargetLanguage := Language.Elm,
    openapiPackage := Pkg("Api.Model"),
    openapiSpec := (Compile/resourceDirectory).value/"openapi.yml",
    openapiElmConfig := ElmConfig().withJson(ElmJson.decodePipeline),
    Compile/resourceGenerators += (Def.task {
      openapiCodegen.value // generate api model files
      compileElm(streams.value.log
        , (Compile/baseDirectory).value
        , (Compile/resourceManaged).value
        , name.value
        , version.value)
    }).taskValue,
    watchSources += Watched.WatchSource(
      (Compile/sourceDirectory).value/"elm"
        , FileFilter.globFilter("*.elm")
        , HiddenFileFilter
    )
  )
```

This example assumes a `elm.json` project file in the source root.

## Discriminator Support

OpenAPI 3.0 enables to introduce subtyping on generated schemas by using [discriminators](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#discriminatorObject).

Two of these are currently supported only in Scala : `oneOf` and `allOf`.   

#### Setup

In order to provide JSON conversion for these discriminators with Circe, we need to make use of [circe-generic-extras](https://github.com/circe/circe-generic-extras)

An example build.sbt using the plugin would look like the following:
```scala
import com.github.eikek.sbt.openapi._

libraryDependencies ++= Seq(
  "io.circe" %% "circe-generic-extras" % "0.11.1",
  "io.circe" %% "circe-core" % "0.11.1",
  "io.circe" %% "circe-generic" % "0.11.1",
  "io.circe" %% "circe-parser" % "0.11.1"
)

openapiSpec := (Compile/resourceDirectory).value/"test.yml"
openapiTargetLanguage := Language.Scala
openapiScalaConfig := ScalaConfig().
  withJson(ScalaJson.circeSemiautoExtra).
  addMapping(CustomMapping.forName({ case s => s + "Dto" }))

enablePlugins(OpenApiSchema)
```


#### Handle `allOf` keywords in Scala

Here is an example OpenAPI spec and the resulting Scala models with JSON conversions

```yaml
components:
  schemas:
    Pet:
      type: object
      discriminator:
        propertyName: petType
      properties:
        name:
          type: string
        petType:
          type: string
      required:
      - name
      - petType
    Cat:  ## "Cat" will be used as the discriminator value
      description: A representation of a cat
      allOf:
      - $ref: '#/components/schemas/Pet'
      - type: object
        properties:
          huntingSkill:
            type: string
            description: The measured skill for hunting
        required:
        - huntingSkill
    Dog:  ## "Dog" will be used as the discriminator value
      description: A representation of a dog
      allOf:
      - $ref: '#/components/schemas/Pet'
      - type: object
        properties:
          packSize:
            type: integer
            format: int32
            description: the size of the pack the dog is from
        required:
        - packSize
```

```scala
import io.circe._
import io.circe.generic.extras.semiauto._
import io.circe.generic.extras.Configuration

sealed trait PetDto {
  val name: String
}
object PetDto {
  implicit val customConfig: Configuration = Configuration.default.withDefaults.withDiscriminator("petType")

  case class Cat (
    huntingSkill: String, name: String
  ) extends PetDto

  case class Dog (
    packSize: Int, name: String
  ) extends PetDto

  object Cat {
    implicit val customConfig: Configuration = Configuration.default.withDefaults.withDiscriminator("petType")
    private implicit val jsonDecoder: Decoder[Cat] = deriveDecoder[Cat]
    private implicit val jsonEncoder: Encoder[Cat] = deriveEncoder[Cat]
  }

  object Dog {
    implicit val customConfig: Configuration = Configuration.default.withDefaults.withDiscriminator("petType")
    private implicit val jsonDecoder: Decoder[Dog] = deriveDecoder[Dog]
    private implicit val jsonEncoder: Encoder[Dog] = deriveEncoder[Dog]
  }

  implicit val jsonDecoder: Decoder[PetDto] = deriveDecoder[PetDto]
  implicit val jsonEncoder: Encoder[PetDto] = deriveEncoder[PetDto]
}

```

Notes about the above example:
- The internal schemas ("Dog" and "Cat") have private encoder/decoders so that they are only encoded and decoded as the trait interface. If you try to decode as a Dog or Cat type, the circe-generic-extras doesn't include the discriminant type
- The mapping functionality (adding "Dto") is only used on the sealed trait since the discriminant type uses the name of the inner case classes ("Dog" and "Cat").

#### Handle `oneOf` keywords in Scala

Another way of transform composed schemas into `sealed trait` hierarchies is to use `oneOf`.

```yaml 
Pet:
  type: object
  discriminator:
    propertyName: petType
  oneOf:
    - $ref: '#/components/schemas/Cat'
    - $ref: '#/components/schemas/Dog'
Cat:  ## "Cat" will be used as the discriminator value
  description: A representation of a cat
  properties:
    huntingSkill:
      type: string
      description: The measured skill for hunting
      enum:
        - clueless
        - lazy
        - adventurous
        - aggressive
    name:
      type: string
    petType:
      type: string
  required:
    - huntingSkill
    - name
    - petType
Dog:  ## "Dog" will be used as the discriminator value
  description: A representation of a dog
  properties:
    packSize:
      type: integer
      format: int32
      description: the size of the pack the dog is from
      default: 0
      minimum: 0
    name:
      type: string
    petType:
      type: string
  required:
    - packSize
    - name
    - petType
```

```scala
import io.circe._
import io.circe.generic.extras.semiauto._
import io.circe.generic.extras.Configuration

sealed trait PetDto {
} 
object PetDto {
  implicit val customConfig: Configuration = Configuration.default.withDefaults.withDiscriminator("petType")
  case class Cat (
    huntingSkill: String, name: String, petType: String
  ) extends PetDto
  case class Dog (
    packSize: Int, name: String, petType: String
  ) extends PetDto
  object Cat {
    implicit val customConfig: Configuration = Configuration.default.withDefaults.withDiscriminator("petType")
    private implicit val jsonDecoder: Decoder[Cat] = deriveDecoder[Cat]
    private implicit val jsonEncoder: Encoder[Cat] = deriveEncoder[Cat]
  }
  object Dog {
    implicit val customConfig: Configuration = Configuration.default.withDefaults.withDiscriminator("petType")
    private implicit val jsonDecoder: Decoder[Dog] = deriveDecoder[Dog]
    private implicit val jsonEncoder: Encoder[Dog] = deriveEncoder[Dog]
  }
  implicit val jsonDecoder: Decoder[PetDto] = deriveDecoder[PetDto]
  implicit val jsonEncoder: Encoder[PetDto] = deriveEncoder[PetDto]
} 
```

Unlike `allOf`, `oneOf` doesn't permit subschemas to inherit fields from their parent. This kind of relation fits well to algebraic data types encodings in Scala.

## Static Documentation

The plugin can run the [swagger codegen
tool](https://github.com/swagger-api/swagger-codegen) or
[redoc](https://github.com/Redocly/redoc) to produce a static HTML
page of the OpenAPI specification file.

Define which generator to use via:

``` scala
openapiStaticGen := OpenApiDocGenerator.Redoc //or
openapiStaticGen := OpenApiDocGenerator.Swagger
```

Note that nodejs (the `npx` command) is required for redoc! The
default is swagger.

Then use the `openapiStaticDoc` task to generate the documentation
from your openapi specification.


Additionally, there is also a task that runs [`openapi-cli
lint`](https://redoc.ly/docs/cli/) against your specification file.
This also requires to have nodejs installed.



## Credits

First, thank you all who reported issues! It follows a list of
contributions in form of code. If you find yourself missing, please
let me know or open a PR.

- @xela85 for adding `oneOf` keywords support (#42)
- @mhertogs for adding support for discriminators (#8)
 readmeEtag: '"6e5a4e6bd94d6f320f02b9b00de4e8ed3e3b5cba"' readmeLastModified: Tue, 23 Jul 2024 06:13:55 GMT repositoryId: 178213765 description: Generate schema sources for Scala, Java and Elm from an openapi 3.0 spec. created: '2019-03-28T13:53:17Z' updated: '2026-02-02T21:15:52Z' language: Scala archived: false stars: 26 watchers: 2 forks: 8 owner: eikek logo: https://avatars.githubusercontent.com/u/701128?v=4 license: MIT repoEtag: '"dcba4248f9059d7b6f26651a3c9f04a95330ec84e88e16578f5edd011538de99"' repoLastModified: Mon, 02 Feb 2026 21:15:52 GMT foundInMaster: true category: Parsers id: c55aff4f35aaeecc6904953f3eb67cb6 - source: openapi3 tags repository: https://github.com/eropple/fastify-openapi3 v3: true repositoryMetadata: base64Readme: >- # `@eropple/fastify-openapi3` #
_Because I just can't stop making OpenAPI libraries, I guess._

[![NPM version](https://img.shields.io/npm/v/@eropple/fastify-openapi3)](https://www.npmjs.com/package/@eropple/fastify-openapi3) [![CI](https://github.com/eropple/fastify-openapi3/actions/workflows/ci.yaml/badge.svg)](https://github.com/eropple/fastify-openapi3/actions/workflows/ci.yaml)

## What is it? ##
This is a library to help you generate [OpenAPI 3.1](https://spec.openapis.org/oas/v3.1.0)-compliant (or 3.0.3 if you do a little work on your own) specs from your [Fastify](https://www.fastify.io/) app. Others exist, but to my mind don't scratch the itch that the best OAS tooling does: making it faster and easier to create correct specs and valid API clients from those specs. Because of my [own](https://github.com/modern-project/modern-ruby) [background](https://github.com/eropple/nestjs-openapi3) in building OpenAPI libraries, and my growing appreciation for Fastify, I decided to take a crack at it.

This library presupposes that you use [`@sinclair/typebox`](https://github.com/sinclairzx81/typebox) to define the JSON schema used in your requests, and from that JSON Schema derives types. (Ergonomics for non-TypeScript users is specifically out-of-scope.) It will walk all your routes, determine your schema, and extract and deduplicate those schemas to present a relatively clean and easy-to-use OpenAPI document. It'll then also serve JSON and YAML versions of your specification, as well as host an interactive API explorer with try-it-out features courtesy of [Rapidoc](https://mrin9.github.io/RapiDoc/) or [Scalar](https://scalar.com).

**Fair warning:** This library is in Early Access(tm) and while the functionality that's here does work, there's some functionality that _doesn't_ exist. The stuff that stands out to me personally can be found in [TODO.md](https://github.com/eropple/fastify-openapi3/blob/main/TODO.md), including a short list of things this plugin _won't_ do.

## Usage ##

First, install it, etc. etc.:

```bash
npm install @eropple/fastify-openapi3
pnpm add @eropple/fastify-openapi3
yarn add @eropple/fastify-openapi3
```

Once you've installed it--well, you'd best go do some things to set it up, huh? There's a manual test (originally added to smoke out issues with Rapidoc serving) in [`examples/start-server.ts`], which can also be directly invoked from the repository with `npm run demo`. Below are the important bits from that demo:

```ts
import Fastify, { FastifyInstance } from 'fastify';
import { Static, Type } from '@sinclair/typebox';

import OAS3Plugin, { OAS3PluginOptions, schemaType, oas3PluginAjv } from '../src/index.js';
```

Your imports. (Obviously, in your project, the last import will be from `"@eropple/fastify-openapi3"`.)

```ts
const fastifyOpts: FastifyServerOptions = {
  logger: { level: 'error' },
  ajv: {
    plugins: [oas3PluginAjv],
  }
}

const fastify = Fastify(fastifyOpts);
await fastify.register(OAS3Plugin, { ...pluginOpts });
```

Register the OAS3 plugin. This plugin uses the Fastify logger and can be pretty chatty on `debug`, so bear that in mind. `pluginOpts` is visible in that file for an example, but it's also commented exhaustively for your IntellSensing pleasure while you're writing it.

```ts
const PingResponse = schemaType('PingResponse', Type.Object({ pong: Type.Boolean() }));
type PingResponse = Static<typeof PingResponse>;
```

Your schema. `schemaType` takes a string as a name, which _must be unique for your entire project_, as well as a `@sinclair/typebox` `Type` (which you can then use as a TypeScript type by doing `Static<typeof T>`, it's awesome). This is now a `TaggedSchema`, which can be used anywhere a normal JSON Schema object can be used within Fastify and will handle validation as you would expect.

If you use a `TaggedSchema` within another schema, the OAS3 plugin is smart enough to extract it into its own OpenAPI `#/components/schemas/YourTypeHere` entry, so your generated clients will also only have the minimal set of model classes, etc. to worry about. Ditto having them in arrays and so on. I've tried to make this as simple to deal with as possible; if it acts in ways you don't expect, _please file an issue_.

And now let's make a route:

```ts
  await fastify.register(async (fastify: FastifyInstance) => {
    fastify.route<{ Reply: PingResponse }>({
      url: '/ping',
      method: 'GET',
      schema: {
        response: {
          200: PingResponse,
        },
      },
      oas: {
        operationId: 'pingPingPingAndDefinitelyNotPong',
        summary: "a ping to the server",
        description: "This ping to the server lets you know that it has not been eaten by a grue.",
        deprecated: false,
        tags: ['meta'],
        vendorPrefixedFields: {
          "x-my-vendor-field": true,
        },
      },
      handler: async (req, reply) => {
        return { pong: true };
      }
    });
  }, { prefix: '/api' });
```

You don't have to put yours inside a prefixed route, but I like to, so, well, there you go.

If you do a `npm run demo`, you'll get a UI that looks like the following:

![a docs screenshot](https://i.imgur.com/iOPApmq.png)

And there you go.

## Autowired Security ##

This plugin includes an autowired security system that automatically handles authentication based on your OpenAPI security schemes. Define your security schemes once, and the plugin will automatically validate credentials on routes that use them.

### Basic Setup ###

```ts
await fastify.register(OAS3Plugin, {
  openapiInfo: { title: 'My API', version: '1.0.0' },
  autowiredSecurity: {
    securitySchemes: {
      ApiKey: {
        type: 'apiKey',
        in: 'header',
        name: 'X-Api-Key',
        fn: (apiKey, request) => {
          return apiKey === 'secret' ? { ok: true } : { ok: false, code: 401 };
        },
      },
      BearerAuth: {
        type: 'http',
        scheme: 'bearer',
        fn: (token, request) => {
          // Validate JWT or other bearer token
          return isValidToken(token) ? { ok: true } : { ok: false, code: 401 };
        },
      },
    },
  },
});

// Use security on routes
fastify.get('/protected', {
  oas: {
    security: { ApiKey: [] },
  },
}, async (req, reply) => {
  return { data: 'secret stuff' };
});
```

### Handler Return Values ###

Security handlers return `{ ok: true }` for success, or `{ ok: false, code: 401 | 403 }` for failure:
- Return `401` (Unauthorized) when credentials are missing or invalid
- Return `403` (Forbidden) when credentials are valid but access is denied

### AND/OR Security Logic ###

OpenAPI supports combining security requirements:

```ts
// OR logic: either scheme can grant access
oas: {
  security: [{ ApiKey: [] }, { BearerAuth: [] }],
}

// AND logic: both schemes must pass
oas: {
  security: { ApiKey: [], BearerAuth: [] },
}
```

### Optional Credentials with `passNullIfNoneProvided` ###

By default, missing credentials return 401 immediately. Set `passNullIfNoneProvided: true` to receive `null` in your handler instead, allowing you to implement optional authentication:

```ts
securitySchemes: {
  OptionalApiKey: {
    type: 'apiKey',
    in: 'header',
    name: 'X-Api-Key',
    passNullIfNoneProvided: true,
    fn: (apiKey, request) => {
      if (apiKey === null) {
        // No key provided - allow anonymous access
        return { ok: true };
      }
      // Key provided - validate it
      return apiKey === 'secret' ? { ok: true } : { ok: false, code: 401 };
    },
  },
}
```

### Accessing Request Body with `requiresParsedBody` ###

Sometimes authentication decisions depend on the request body (e.g., request signing, body-based authorization). Set `requiresParsedBody: true` to receive the parsed body in your handler:

```ts
securitySchemes: {
  BodyAwareAuth: {
    type: 'apiKey',
    in: 'header',
    name: 'X-Api-Key',
    requiresParsedBody: true,
    fn: (apiKey, request, context) => {
      // context.body contains the parsed request body
      const body = context?.body as { resourceId?: string };

      // Check if user has access to the requested resource
      if (!userCanAccessResource(apiKey, body?.resourceId)) {
        return { ok: false, code: 403 };
      }
      return { ok: true };
    },
  },
}
```

When `requiresParsedBody` is not set or is `false`, the `context` parameter will be `undefined`.

### Raw Body Access for Signature Validation ###

For webhook signature validation or HMAC verification, you need access to the raw (unparsed) request body. Use the [`fastify-raw-body`](https://github.com/Eomm/fastify-raw-body) plugin alongside `requiresParsedBody`:

```ts
import fastifyRawBody from 'fastify-raw-body';

// Register fastify-raw-body BEFORE the OAS plugin
await fastify.register(fastifyRawBody, {
  global: false,  // Only on routes that need it
  runFirst: true, // Get raw body before any transforms
});

await fastify.register(OAS3Plugin, {
  // ... other options
  autowiredSecurity: {
    securitySchemes: {
      WebhookSignature: {
        type: 'apiKey',
        in: 'header',
        name: 'X-Signature',
        requiresParsedBody: true,
        fn: (signature, request, context) => {
          // request.rawBody contains the raw string/buffer
          const expectedSig = crypto
            .createHmac('sha256', secret)
            .update(request.rawBody as string)
            .digest('hex');

          if (signature !== expectedSig) {
            return { ok: false, code: 401 };
          }
          return { ok: true };
        },
      },
    },
  },
});

// Enable rawBody on specific routes
fastify.post('/webhook', {
  config: { rawBody: true },
  oas: { security: { WebhookSignature: [] } },
}, handler);
```

### Supported Security Schemes ###

- **API Key** (`type: 'apiKey'`): Validates keys from headers or cookies
- **HTTP Basic** (`type: 'http', scheme: 'basic'`): Validates username/password credentials
- **HTTP Bearer** (`type: 'http', scheme: 'bearer'`): Validates bearer tokens

## Contributing ##
Issues and PRs welcome! Constructive criticism on how to improve the library would be awesome, even as I use it in my own stuff and figure out where to go from there, too.

**Before you start in on a PR, however**, please do me a solid and drop an issue so we can discuss the approach. Thanks!
 readmeEtag: '"69e0ac3b9a0770d1bad22d0f56419b89267d0f1a"' readmeLastModified: Thu, 15 Jan 2026 21:48:04 GMT repositoryId: 459027206 description: Developer-friendly OpenAPI3 tooling for Fastify that's easy to use. created: '2022-02-14T05:24:29Z' updated: '2026-02-05T01:38:39Z' language: TypeScript archived: false stars: 27 watchers: 3 forks: 3 owner: eropple logo: https://avatars.githubusercontent.com/u/109262?v=4 repoEtag: '"2158c66f92ca102f416544877880fe6c5a116173fa19f9e75ff8b9ab0e1244b7"' repoLastModified: Thu, 05 Feb 2026 01:38:39 GMT foundInMaster: true category: - Server - Parsers id: 69aabd9435e6e1468c8a137c700edcaf - source: - openapi3 tags - openapi31 tags repository: https://github.com/janbuchar/payload-oapi v3: true v3_1: true id: a75ea2a00fbf40d2b02b06915b51c5f4 repositoryMetadata: base64Readme: >- IyBQYXlsb2FkIE9wZW5BUEkgUGx1Z2luCgpbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL3BheWxvYWQtb2FwaS5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9wYXlsb2FkLW9hcGkpCgpBdXRvZ2VuZXJhdGUgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZyb20geW91ciBQYXlsb2FkIENNUyBpbnN0YW5jZSBhbmQgdXNlIGl0IGZvciBkb2N1bWVudGF0aW9uIG9yIHRvIGdlbmVyYXRlIGNsaWVudCBTREtzLgoKIyBSb2FkbWFwCgotIFt4XSBDb21wbGV0ZSBkZXNjcmlwdGlvbiBvZiBjb2xsZWN0aW9uIENSVUQgZW5kcG9pbnRzCi0gW3hdIENvbXBsZXRlIGRlc2NyaXB0aW9uIG9mIGdsb2JhbHMgQ1JVRCBlbmRwb2ludHMKLSBbeF0gSW50ZWdyYXRlZCBTd2FnZ2VyIFVJIGFuZCBSYXBpZG9jCi0gW3hdIEF1dGhlbnRpY2F0aW9uIGVuZHBvaW50cyBhbmQgc3BlY2lmaWNhdGlvbgotIFt4XSBQcmVmZXJlbmNlcyBlbmRwb2ludHMKLSBbeF0gU3VwcG9ydCBQYXlsb2FkIENNUyAzLngKLSBbeF0gU3VwcG9ydCBnZW5lcmF0aW5nIGJvdGggT3BlbkFQSSAzLjAgYW5kIDMuMQotIFsgXSBDdXN0b20gZW5kcG9pbnRzCgojIEluc3RhbGxhdGlvbgoKWW91IGNhbiBpbnN0YWxsIHRoZSBwbHVnaW4gdXNpbmcgeW91ciBwcmVmZXJyZWQgcGFja2FnZSBtYW5hZ2VyOgoKLSBgcG5wbSBhZGQgcGF5bG9hZC1vYXBpYAotIGBucG0gaW5zdGFsbCBwYXlsb2FkLW9hcGlgCi0gYHlhcm4gYWRkIHBheWxvYWQtb2FwaWAKCiMgU2V0dXAKCiMjIDEuIEFkZCB0aGUgT3BlbkFQSSBjb3JlIHBsdWdpbgoKVG8gYWRkIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gZW5kcG9pbnQgdG8geW91ciBQYXlsb2FkIGFwcCwgc2ltcGx5IGltcG9ydCB0aGUgYG9wZW5hcGlgIHBsdWdpbiBhbmQgYWRkIGl0IHRvIHlvdXIgcGF5bG9hZCBjb25maWd1cmF0aW9uOgoKYGBgdHlwZXNjcmlwdAppbXBvcnQgeyBvcGVuYXBpIH0gZnJvbSAncGF5bG9hZC1vYXBpJwoKYnVpbGRDb25maWcoewogIHBsdWdpbnM6IFsKICAgIG9wZW5hcGkoeyBvcGVuYXBpVmVyc2lvbjogJzMuMCcsIG1ldGFkYXRhOiB7IHRpdGxlOiAnRGV2IEFQSScsIHZlcnNpb246ICcwLjAuMScgfSB9KSwKICBdLAogIC8vIC4uLgp9KQpgYGAKCiMjIDIuIEFkZCBhIGRvY3VtZW50YXRpb24gVUkgcGx1Z2luIChvcHRpb25hbCkKClRvIHByb3ZpZGUgYSB1c2VyIGludGVyZmFjZSBmb3IgeW91ciBBUEkgZG9jdW1lbnRhdGlvbiwgeW91IGNhbiBhZGQgb25lIG9mIHRoZSBmb2xsb3dpbmcgcGx1Z2luczoKCi0gW2BzY2FsYXJgXShodHRwczovL2dpdGh1Yi5jb20vc2NhbGFyL3NjYWxhcikKLSBbYHN3YWdnZXJVSWBdKGh0dHBzOi8vc3dhZ2dlci5pby90b29scy9zd2FnZ2VyLXVpLykKLSBbYHJhcGlkb2NgXShodHRwczovL21yaW45LmdpdGh1Yi5pby9SYXBpRG9jLykKLSBbYHJlZG9jYF0oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvcmVkb2MpCgpFeGFtcGxlIHVzYWdlOgoKYGBgdHlwZXNjcmlwdAppbXBvcnQgeyBvcGVuYXBpLCBzY2FsYXIsIHN3YWdnZXJVSSwgcmFwaWRvYywgcmVkb2MgfSBmcm9tICdwYXlsb2FkLW9hcGknCgovLyBDaG9vc2Ugb25lIGRvY3VtZW50YXRpb24gVUkgcGx1Z2lucyBhcyBuZWVkZWQKYnVpbGRDb25maWcoewogIHBsdWdpbnM6IFsKICAgIG9wZW5hcGkoLyogLi4uICovKSwKICAgIC8vIFVuY29tbWVudCB0aGUgVUkgeW91IHdhbnQgdG8gdXNlOgogICAgc2NhbGFyKHsgLyogLi4ub3B0aW9ucyAqLyB9KSwKICAgIC8vIHN3YWdnZXJVSSh7IC8qIC4uLm9wdGlvbnMgKi8gfSksCiAgICAvLyByYXBpZG9jKHsgLyogLi4ub3B0aW9ucyAqLyB9KSwKICAgIC8vIHJlZG9jKHsgLyogLi4ub3B0aW9ucyAqLyB9KSwKICBdLAogIC8vIC4uLgp9KQpgYGAKCiMgVXNhZ2UKClVubGVzcyB5b3UgY29uZmlndXJlZCBpdCBvdGhlcndpc2UsIHlvdXIgc3BlYyB3aWxsIGJlIGFjY2Vzc2libGUgdmlhIDxodHRwczovL3lvdXItcGF5bG9hZC5jb20vYXBpL29wZW5hcGkuanNvbj4uIElmIHlvdQphZGRlZCBhIGRvY3VtZW50YXRpb24gVUksIHRoYXQgd2lsbCBiZSBhY2Nlc3NpYmxlIHZpYSA8aHR0cHM6Ly95b3VyLXBheWxvYWQuY29tL2FwaS9kb2NzPiAodGhpcyBpcyBhbHNvIGNvbmZpZ3VyYWJsZSkuCg== readmeEtag: '"28915de6673eeecdd68f6fd6c4045c0dee7a2936"' readmeLastModified: Tue, 27 May 2025 14:53:12 GMT repositoryId: 618796772 description: A Payload CMS plugin for OpenAPI (3.0, 3.1) specification generation created: '2023-03-25T11:34:02Z' updated: '2026-02-06T00:52:31Z' language: TypeScript archived: false stars: 114 watchers: 2 forks: 24 owner: janbuchar logo: https://avatars.githubusercontent.com/u/461515?v=4 license: MIT repoEtag: '"350588071a807d885387742852a2df9de9e7be9c69e1fbe40bc8a47a3820a9fb"' repoLastModified: Fri, 06 Feb 2026 00:52:31 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/teyras/payload-oapi - source: openapi3 tags repository: https://github.com/microprofile-extensions/openapi-ext v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQXBpIEV4dGVuc2lvbnMKClshW2J1aWxkX3N0YXR1c10oaHR0cHM6Ly9hcHAudHJhdmlzLWNpLmNvbS9taWNyb3Byb2ZpbGUtZXh0ZW5zaW9ucy9vcGVuYXBpLWV4dC5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2FwcC50cmF2aXMtY2kuY29tL2dpdGh1Yi9taWNyb3Byb2ZpbGUtZXh0ZW5zaW9ucy9vcGVuYXBpLWV4dCkgWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLUFwYWNoZSUyMDItYmx1ZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbWljcm9wcm9maWxlLWV4dGVuc2lvbnMvb3BlbmFwaS1leHQvYmxvYi9tYWluL0xJQ0VOU0UpCl9fX19fX19fX19fCkhlcmUgeW91IHdpbGwgZmluZCBzb21lIGV4dHJhIHRvb2xzIGZvciBPcGVuIEFQSQoKKiBbT3BlbkFwaSBVSV0oaHR0cHM6Ly9naXRodWIuY29tL21pY3JvcHJvZmlsZS1leHRlbnNpb25zL29wZW5hcGktZXh0L2Jsb2IvbWFpbi9vcGVuYXBpLXVpL1JFQURNRS5tZCkKCiMjIEV4YW1wbGVzCgpBbHNvIGxvb2sgYXQgdGhlIFtleGFtcGxlIGFwcGxpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vbWljcm9wcm9maWxlLWV4dGVuc2lvbnMvb3BlbmFwaS1leHQvYmxvYi9tYWluL29wZW5hcGktZXhhbXBsZXMvYmFzaWMtZXhhbXBsZS9SRUFETUUubWQpIHRvIHNlZSBob3cgdGhpcyBpcyB1c2VkLgpbSGVsaWRvbiBleGFtcGxlXShodHRwczovL2dpdGh1Yi5jb20vbWljcm9wcm9maWxlLWV4dGVuc2lvbnMvb3BlbmFwaS1leHQvYmxvYi9tYWluL29wZW5hcGktZXhhbXBsZXMvaGVsaWRvbi1iYXNpYy1leGFtcGxlL1JFQURNRS5tZCkgZGVtb25zdHJhdGUgT3BlbkFwaSB1c2FnZSB3aXRoIEhlbGlkb24gTWljcm9Qcm9maWxlLgo= readmeEtag: '"310bf9c993b51c13e1dc837e3d10c7c41e32aba4"' readmeLastModified: Mon, 01 May 2023 10:37:19 GMT repositoryId: 149434703 description: Some extensions for MicroProfile OpenAPI created: '2018-09-19T10:39:23Z' updated: '2025-01-21T03:51:15Z' language: Java archived: false stars: 23 watchers: 2 forks: 14 owner: microprofile-extensions logo: https://avatars.githubusercontent.com/u/43223776?v=4 license: Apache-2.0 repoEtag: '"25a18a618cac59c72b0a6ea99d878b956d5899d090eacd5e09d42b7af135abbd"' repoLastModified: Tue, 21 Jan 2025 03:51:15 GMT foundInMaster: true category: - SDK - Parsers id: a6ea8a703b31ae1a40d92590739ef8a4 - source: openapi3 tags repository: https://github.com/riconijeboer/laravel-to-swagger v3: true repositoryMetadata: base64Readme: >- # Laravel to Swagger

<a href="https://packagist.org/packages/riconijeboer/laravel-to-swagger" target="_blank" title="Packagist version">
    <img src="https://img.shields.io/packagist/v/riconijeboer/laravel-to-swagger" alt="Packagist version">
</a>
<img src="https://img.shields.io/packagist/dm/riconijeboer/laravel-to-swagger" alt="Packagist downloads">

This package aims to bring the easiest path to creating a Swagger / OpenApi 3 config for your Laravel API's.

## Installation

For local only installation go [here](#local-only-installation)

1. `composer require riconijeboer/laravel-to-swagger`
1. `php artisan vendor:publish --provider="RicoNijeboer\Swagger\SwaggerServiceProvider"`
    - This will publish the package's config-file and migrations

### Local Only installation

1. `composer require riconijeboer/laravel-to-swagger --dev`
1. `php artisan vendor:publish --provider="RicoNijeboer\Swagger\SwaggerServiceProvider"`
    - This will publish the package's config-file and migrations
1. Finally, you should prevent the Swagger package from being [auto-discovered](https://laravel.com/docs/8.x/packages#package-discovery) by adding the following to your `composer.json` file:
   ```
   "extra": {
       "laravel": {
           "dont-discover": [
               "riconijeboer/laravel-to-swagger"
           ]
       }
   }
   ```
1. Now that auto-discovery is disabled you should manually register the service provider in the `register` method of your `App\Providers\AppServiceProvider` class.
   In the example we only enable the package when we are running in a `local` environment.
   ```php
   /**
    * Register any application services.
    *
    * @return void
    */
   public function register()
   {
       if ($this->app->environment('local')) {
           $this->app->register(\RicoNijeboer\Swagger\SwaggerServiceProvider::class);
       }
   }
   ```

### Requirements

- **PHP**: 7.4.x or 8.0.x
- **Laravel**: v6 / v7 / v8

### Updating

When changes are made that impact existing users, I will make sure that they are documented in the [Changelog](#changelog).

These will contain changes like database columns added / removed / renamed. Or functional changes that need action within your code.

#### v2.1.x to v2.2.x

- When updating from v2.1 to v2.2 add a nullable `route_domain` string-column to the `swagger_batches` domain.   
  `$table->string('route_domain')->nullable();`

## Usage

### Registering the Redoc Documentation route.

Reading a Swagger config is an acquired taste. To display the config in a more user-friendly way I've used [Redoc](https://github.com/Redocly/redoc). To register the documentation route you can simply
add the code below to your `routes/web.php` file or within a ServiceProvider.

```php
use RicoNijeboer\Swagger\Swagger;

Swagger::routes();
```

#### Customizing the /docs Url

```php
use RicoNijeboer\Swagger\Http\Routing\RouteRegistrar;
use RicoNijeboer\Swagger\Swagger;

Swagger::routes(fn (RouteRegistrar $routes) => $routes->forDocumentation('/different-url/docs'));
```

#### Customizing the routing group

```php
use RicoNijeboer\Swagger\Swagger;

Swagger::routes(null, [
    'prefix' => 'swagger', // This will do Route::group(['prefix' => 'swagger']) under the hood.
]);
```

#### Using Swagger UI instead of Redoc

You can disable Redoc and use Swagger UI instead by passing `false` as the second parameter to the `forDocumentation` method.

```php
use RicoNijeboer\Swagger\Http\Routing\RouteRegistrar;
use RicoNijeboer\Swagger\Swagger;

Swagger::routes(fn (RouteRegistrar $routes) => $routes->forDocumentation('/docs', false));
```

### Registering routes for your Swagger config

To add routes to your Swagger config you want to add the `\RicoNijeboer\Swagger\Http\Middleware\SwaggerReader` middleware. Which is aliased to both `swagger_reader` and `openapi_reader`.

```php
Route::middleware('swagger_reader')->get('products', [ProductController::class,'index']);
```

#### Tagging / Grouping routes

To group multiple routes in Swagger you add a tag to the path. Using the package you may register a tag through adding a middleware to your route and supplying it with the desired tags as shown below.

> Keep in mind that the `swagger_tag` (`\RicoNijeboer\Swagger\Http\Middleware\SwaggerTag`) middleware is only going to tag your request once the batch has been stored.
> If a batch has not been created it will continue and do nothing, no questions asked.

```php
// Using the SwaggerReader middleware
Route::middleware('swagger_reader:tag-one,tag-two')->get('products', [ProductController::class,'index']);

// Using the SwaggerTag middleware
Route::middleware('swagger_tag:tag-one,tag-two')->get('products', [ProductController::class,'index']);
```

## Configuration

### Database connection

You may want your database entries from the Laravel to Swagger package in a separate database or to have an extra prefix. To do this, add an extra connection to your Laravel project and just simply
set the `swagger.database.connection` config value.

For example, if I want all parts of the Swagger database to be in a connection I've named `swagger`, I'd set the config like below.

```php
return [
    //...
    'database' => [
        'connection' => 'swagger',
    ],
    //...
];
```

### Evaluation delay, configure how often your endpoints are re-evaluated

By default, the package only updates the configuration of your route every 12 hours, within a couple restraints. This does keep into account the response codes you are sending back to your users.

If you want to change this to for example 24 hours, you'd set the `swagger.evaluation-delay` config value to the time in seconds.

```php
return [
    //...
    'evaluation-delay' => 24 * 60 * 60, // Or 86400
    //...
];
```

### Swagger Meta Information

Swagger allows you to provide the users with a bit of info about your API; a title, a description and a version.

In the config there is a `swagger.info` array where you can add your API's title, description and version, as shown below.

```php
return [
    //...
    'info'             => [
        'title'       => 'Laravel to Swagger',
        'description' => null,
        'version'     => '2.1.2',
    ],
    //...
];
```

### Swagger Servers

Swagger allows you to show the user servers that you can access the API from. A basic setup would be the one below;

- You have one API where your users can access live data and one where they can access some demo information.

```php
return [
    //...
    'servers'          => [
        [
            'url'         => 'http://api.example.com/v1',
            'description' => null,
        ],
        [
            'url'         => 'http://demo-api.example.com/v1',
            'description' => 'The demo environment description',
        ],
    ],
    //...
];
```

#### Server templating

Swagger also supports [Server templating](https://swagger.io/docs/specification/api-host-and-base-path/). You are able to add variables to your server which gives you the possibility to obfuscate the
urls a bit.

An example from their documentation is a server where you have `https://{customerId}.saas-app.com:{port}/v2` as the URL. This is translated to their Yaml as shown below. They define the url,
containing its variables, and describe what the variables do and what their default values are.

Laravel to Swagger supports basically the same format as the Yaml file, below the Yaml you can see how you translate it to the Swagger config. I've added very light validation to this, so if the
Swagger UI / Redoc UI breaks try to first ensure that you've correctly configured the config according to the [Swagger docs](https://swagger.io/docs/specification/api-host-and-base-path/)

**Swagger Yaml**

```yaml
servers:
  - url: https://{customerId}.saas-app.com:{port}/v2
    variables:
      customerId:
        default: demo
        description: Customer ID assigned by the service provider
      port:
        enum:
          - '443'
          - '8443'
        default: '443'
```

**Laravel to Swagger Config**

```php
return [
    //...
    'servers'          => [
        [
            'url'       => 'https://{customerId}.saas-app.com:{port}/v2',
            'variables' => [
                'customerId' => [
                    'default'     => 'demo',
                    'description' => 'Customer ID assigned by the service provider',
                ],
                'port'       => [
                    'enum'    => [
                        '443',
                        '8443',
                    ],
                    'default' => '443',
                ],
            ],
        ],
    ],
    //...
];
```

### Redoc Version

Let's say you run into a bug within the Redoc that the team behind Redoc has fixed in a newer version, and I have not updated the package yet to have the fixed version used. I have added a config
value (`swagger.redoc.version`), so you don't have to mess around with things like published views or anything; You can simply change the version based on
their [releases](https://github.com/Redocly/redoc/releases), at the time of writing this documentation the latest version is `v2.0.0-rc.53`.

```php
return [
    //...
    'redoc' => [
        'version' => 'v2.0.0-rc.53',
        //...
    ],
    //...
];
```

### Grouping Tags
![redoc-only](https://img.shields.io/badge/Redoc%20Only-red.svg)

If you are using Redoc you are able to group your tags by using the config.
In the example below a group called "User Management" is created containing the tags "Users", "Admin" and "API keys".

> Tags that are not grouped by you will be added under a default group, <br>
  the name of which can be changed using the config (`swagger.redoc.default-group`)

```php
return [
    //...
    'redoc' => [
        //...
        'default-group' => null,
        'tag-groups' => [
            [
                'name' => 'User Management',
                'tags' => [ 'Users', 'Admin', 'API keys' ],
            ],
        ],
    ],
    //...
];
```

### Adding a logo
![redoc-only](https://img.shields.io/badge/Redoc%20Only-red.svg)

You can add a logo to the sidebar in your Redoc visualization by adding a full-url to the `swagger.info.logo.url`.

> The logo is only displayed when you use the Redoc documentation route.

```php
return [
    //...
    'info' => [
        //...
        'logo' => [
            'url' => 'https://picsum.photos/300/200',
        ],
    ],
    //...
];
```

## Testing

```
composer test
```

## Credits

- [Rico Nijeboer](https://github.com/RicoNijeboer/)
- [All Contributors](https://github.com/RicoNijeboer/laravel-to-swagger/graphs/contributors)

## License

The MIT License (MIT). Please see [License File](LICENSE.md) for more information.

## Changelog

All notable changes can be found in the [CHANGELOG.md](CHANGELOG.md).
 readmeEtag: '"f64ef8febcfdad081fee6d050fdd1c1de5ec2b0f"' readmeLastModified: Tue, 15 Feb 2022 07:38:07 GMT repositoryId: 350056876 description: >- This package aims to bring you the easiest way to create a Swagger / OpenApi 3 config for your Laravel API's. created: '2021-03-21T16:37:23Z' updated: '2025-02-22T10:32:09Z' language: PHP archived: true stars: 22 watchers: 1 forks: 0 owner: RicoNijeboer logo: https://avatars.githubusercontent.com/u/26306300?v=4 license: MIT repoEtag: '"429ce68b3d44c5f504d147137e368735025a5acf59aa9d6be33608ab264c47d8"' repoLastModified: Sat, 22 Feb 2025 10:32:09 GMT foundInMaster: true category: Server Implementations id: 79cd08ad7f4391e56675bdf9ed635f89 - source: openapi3 tags repository: https://github.com/cdimascio/kotlin-spring-mvc-template v3: true repositoryMetadata: base64Readme: >- IyBrb3RsaW4tc3ByaW5nLW12Yy10ZW1wbGF0ZQoKU3ByaW5nIDUgTVZDIHRlbXBsYXRlIHdpdGggKktvdGxpbiogYW5kICpPcGVuQVBJIDMuMCouIChBbHNvIFN1cHBvcnRzICpTd2FnZ2VyIDIuMCopCgpGZWF0dXJlcyBhdXRvbWF0aWMgcmVxdWVzdC9yZXNwb25zZSB2YWxpZGF0aW9uIGFuZCBpbnRlcmFjdGl2ZSBBUEkgZG9jCgo8cCBhbGlnbj0ibGVmdCI+CjxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vY2RpbWFzY2lvL2tvdGxpbi1zcHJpbmctbXZjLXRlbXBsYXRlL21hc3Rlci9hc3NldHMvc3ByaW5nLW12Yy1vcGVuYXBpLnBuZyIgd2lkdGg9IjYwMCIvPgo8L3A+CgoqKkZlYXR1cmVzKio6CgotIEF1dG9tYXRpYyByZXF1ZXN0IGFuZCByZXNwb25zZSB2YWxpZGF0aW9uIHZpYSBhdGxhc3NpYW4gW09wZW5BcGkgMy4wXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2Fib3V0LykKLSBBdXRvbWF0aWMgZG9jdW1lbnRhdGlvbiBnZW5lcmF0aW9uIHZpYSBbU3dhZ2dlciBVSV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3Rvb2xzL3N3YWdnZXItdWkvKQotIDEyIGZhY3RvciBjb21wbGlhbnQgY29uZmlndXJhdGlvbiB2aWEgW2phdmEtZG90ZW52XShodHRwczovL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2phdmEtZG90ZW52KQotIEF1dG9tYXRpYyBjb2RlIGZvcm1hdHRpbmcgYW5kIGxpbnRpbmcgdmlhIFtrdGxpbnRdKGh0dHBzOi8va3RsaW50LmdpdGh1Yi5pbykKLSBGdWxseSBvcGVyYXRpb25hbCBzYW1wbGUgUkVTVCBBUEkKLSBBUEkgaW50ZWdyYXRpb24gdGVzdHMgd2l0aCBbVW5pcmVzdF0oaHR0cDovL3VuaXJlc3QuaW8vamF2YS5odG1sKSBSRVNUIGNsaWVudAotIFNpbXBsZSBIVFRQIGVycm9ycyB2aWEgW2phcGktZXJyb3JzXShodHRwczovL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2phcGktZXJyb3JzKQotIERvY2tlcmZpbGUKCiMjIFNldHVwCgotIENsb25lIHRoaXMgcmVwbwotIGNvcHkgYC5lbnYudGVtcGxhdGVgIHRvIGBzcmMvbWFpbi9yZXNvdXJjZXMvLmVudmAKCgoKIyMgQnVpbGQKCmBgYHNoZWxsCi4vZ3JhZGxldyBidWlsZApgYGAKCiMjIEZvcm1hdC9MaW50IAoKYGBgc2hlbGwKLi9ncmFkbGV3IGxpbnRLb3RsaW4gIyBzZWUgbGludCBlcnJvcnMKLi9ncmFkbGV3IGZvcm1hdEtvdGxpbiAjIGF0dGVtcHQgdG8gYXV0b21hdGljYWxseSBmaXggbGludCBlcnJvcnMKYGBgCgojIyBSdW4KYGBgc2hlbGwKLi9ncmFkbGV3IGJvb3RSdW4KYGBgCgojIyBUZXN0CgpgYGBzaGVsbAouL2dyYWRsZXcgdGVzdApgYGAKCiMjIFBhY2thZ2UgLyBEaXN0CgpgYGBzaGVsbAouL2dyYWRsZXcgamFyCgojIFJ1biB0aGUgZGlzdApqYXZhIC1qYXIgLi9idWlsZC9saWJzL2V4YW1wbGUtc2VydmljZS0xLjAuMC5qYXIgIApgYGAKCiMjIFRyeSBJdAoKTmF2aWdhdGUgdG8gW2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGlfZXhwbG9yZXIvaW5kZXguaHRtbF0oaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaV9leHBsb3Jlci9pbmRleC5odG1sKQoKIyMgVmFsaWRhdGUgRXhhbXBsZQoKCmBgYHNoZWxsCmN1cmwgLVggUE9TVCAiaHR0cDovL2xvY2FsaG9zdDo4MDgwL3VzZXJzIiAtSCAiYWNjZXB0OiBhcHBsaWNhdGlvbi9qc29uIiAtSCAiQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uIiAtZCAie1wibWVcIjpcImNhcm1pbmVcIn0iCgp7CiAgImVycm9yIjogIiBPYmplY3QgaW5zdGFuY2UgaGFzIHByb3BlcnRpZXMgd2hpY2ggYXJlIG5vdCBhbGxvd2VkIGJ5IHRoZSBzY2hlbWE6IFtcIm1lXCJdIiwKICAiY29kZSI6IDQwMAp9CmBgYAojIyBJbnRlcmFjdGl2ZSBBcGkgZG9jCgohW10oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2NkaW1hc2Npby9rb3RsaW4tc3ByaW5nLW12Yy10ZW1wbGF0ZS9tYXN0ZXIvYXNzZXRzL3N3YWdnZXItdWkucG5nKQoKIyMgTGljZW5zZQpBcGFjaGUgMi4wCg== readmeEtag: '"6f5c449cd0dd9eb5a7267b2983326b53e82bd1a1"' readmeLastModified: Fri, 15 Feb 2019 02:10:19 GMT repositoryId: 151765325 description: >- 12-factor compliant Spring MVC Kotlin template. Features automatic request/response validation and interactive API doc created: '2018-10-05T18:57:20Z' updated: '2024-11-19T12:58:03Z' language: Kotlin archived: false stars: 23 watchers: 1 forks: 8 owner: cdimascio logo: https://avatars.githubusercontent.com/u/4706618?v=4 repoEtag: '"3e5a241f7d4c3e243a6cd4b0a4f811d51fb10e343e5d41c0ec0b23e4fc8dc7d8"' repoLastModified: Tue, 19 Nov 2024 12:58:03 GMT foundInMaster: true category: - Data Validators - Server Implementations id: f8dfb37ecbaed15a1f658a4017e9e785 - source: openapi3 tags repository: https://github.com/postgrest/postgrest-openapi v3: true id: 587979cc8238ff762a0d0bea8a052e41 repositoryMetadata: base64Readme: >- IyBQb3N0Z1JFU1QgT3BlbkFQSQoKU1FMIGZ1bmN0aW9ucyB0byBidWlsZCB0aGUgT3BlbkFQSSBvdXRwdXQgb2YgYSBQb3N0Z1JFU1QgaW5zdGFuY2UuCgojIyBSb2FkbWFwCgotIFRoZSBmaXJzdCBzdGVwIGluIHRoZSByb2FkbWFwIGlzIHRvIG1pZ3JhdGUgdGhlIE9wZW5BUEkgc3BlYyBmcm9tIHRoZSBQb3N0Z1JFU1QgY29yZSByZXBvc2l0b3J5ICh2ZXJzaW9uIDIuMCB0byAzLjEpOgogIC0gW3hdIEluZm8gb2JqZWN0CiAgLSBbeF0gU2VydmVyIG9iamVjdCAocmVwbGFjZXMgaG9zdCwgYmFzZVBhdGggYW5kIHNjaGVtZXMgZnJvbSBPQVMgMi4wKQogIC0gW3hdIENvbXBvbmVudHMgb2JqZWN0CiAgICAtIFt4XSBTY2hlbWFzIChkZWZpbml0aW9ucyBpbiBPQVMgMi4wKQogICAgLSBbeF0gU2VjdXJpdHkgc2NoZW1lIChzZWN1cml0eSBkZWZpbml0aW9ucyBpbiBPQVMgMi4wKQogICAgLSBbeF0gUGFyYW1ldGVycwogIC0gW3hdIFBhdGhzIG9iamVjdAogICAgLSBbeF0gVGFibGVzIGFuZCBWaWV3cwogICAgICAtIFt4XSBHRVQKICAgICAgLSBbeF0gUE9TVAogICAgICAtIFt4XSBQQVRDSAogICAgICAtIFt4XSBERUxFVEUKICAgIC0gW3hdIEZ1bmN0aW9ucwogICAgICAtIFt4XSBHRVQKICAgICAgLSBbeF0gUE9TVAogIC0gWyBdIEV4dGVybmFsIERvY3VtZW50YXRpb24gT2JqZWN0CiAgLSBbIF0gSGFuZGxlIHJlbGV2YW50IE9wZW5BUEkgZWxlbWVudHMgYWNjb3JkaW5nIHRvIHVzZXIgcGVybWlzc2lvbnMKLSBUaGUgbmV4dCBzdGVwIGlzIHRvIGZpeCB0aGUgaXNzdWVzIHRhZ2dlZCB3aXRoIGBPcGVuQVBJYCBpbiB0aGUgY29yZSByZXBvLgoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCm1ha2UgJiYgc3VkbyBtYWtlIGluc3RhbGwKYGBgCgojIyBEZXZlbG9wbWVudAoKQ2hlY2sgdGhlIGNvZGUgZG9jdW1lbnRhdGlvbiBhdCBbL2RvY3NdKC9kb2NzKSB0byBmYW1pbGlhcml6ZSB5b3Vyc2VsZiB3aXRoIHRoZSBwcm9qZWN0IHN0cnVjdHVyZSBhbmQgZnVuY3Rpb24gdXNhZ2UuCgpGb3IgdGVzdGluZyBvbiB5b3VyIGxvY2FsIGRhdGFiYXNlOgoKYGBgYmFzaAojIHRoaXMgd2lsbCBsb2FkIGZpeHR1cmVzIGluIGEgY29udHJpYl9yZWdyZXNzaW9uIGRiIG9uIHlvdXIgbG9jYWwgcG9zdGdyZXMKbWFrZSBmaXh0dXJlcwoKIyBydW4gdGhlIHRlc3RzLCB0aGV5IGNhbiBiZSBydW4gcmVwZWF0ZWRseQptYWtlIGluc3RhbGxjaGVjawoKIyB0byBjbGVhbiB0aGUgZml4dHVyZXMgeW91IGNhbiB1c2UKbWFrZSBjbGVhbgpgYGAKCkZvciBhbiBpc29sYXRlZCBhbmQgcmVwcm9kdWNpYmxlIGVudmlyb21lbnQgeW91IGNhbiB1c2UgW05peF0oaHR0cHM6Ly9uaXhvcy5vcmcvZG93bmxvYWQuaHRtbCkuCgpgYGBiYXNoCiMgdG8gcnVuIHRlc3RzCm5peC1zaGVsbCAtLXJ1biAid2l0aC1wZy0xNSBtYWtlIGluc3RhbGxjaGVjayIKCiMgdG8gaW50ZXJhY3Qgd2l0aCB0aGUgbG9jYWwgZGF0YWJhc2Ugd2l0aCBmaXh0dXJlcyBsb2FkZWQKbml4LXNoZWxsIC0tcnVuICJ3aXRoLXBnLTE1IHBzcWwgY29udHJpYl9yZWdyZXNzaW9uIgoKIyB5b3UgY2FuIGNob29zZSB0aGUgcGcgdmVyc2lvbgpuaXgtc2hlbGwgLS1ydW4gIndpdGgtcGctMTMgbWFrZSBpbnN0YWxsY2hlY2siCmBgYAoKRm9yIHRob3NlIHdobyBpbnNpc3Qgb24gRG9ja2VyOgpgYGBiYXNoCiMgVG8gYnVpbGQgYSBkb2NrZXIgaW1hZ2UgYW5kIHJ1biB0aGUgdGVzdHMgaW4gaXQKbWFrZSBkb2NrZXItYnVpbGQtdGVzdAoKIyBUbyBidWlsZCBhIGRvY2tlciBpbWFnZSBmb3IgYWN0dWFsIHVzZQptYWtlIGRvY2tlci1idWlsZApgYGAKCiMjIFJlZmVyZW5jZXMKCi0gW09wZW5BUEkgMyBTcGVjaWZpY2F0aW9uIERvY3VtZW50YXRpb25dKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCk6IFRoZSBvZmZpY2lhbCBkb2N1bWVudGF0aW9uIG9mIHRoZSBzcGVjLgotIFtPcGVuQVBJIFNwZWNpZmljYXRpb24gRXhwbGFpbmVkXShodHRwczovL2xlYXJuLm9wZW5hcGlzLm9yZy9zcGVjaWZpY2F0aW9uLyk6IEludHJvZHVjdG9yeSBleHBsYW5hdGlvbiBvZiB0aGUgc3BlYy4KLSBbT3BlbkFQSSBHdWlkZV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9hYm91dC8pOiBEZXRhaWxlZCBleHBsYW5hdGlvbiBmb3IgZWFjaCBjb25jZXB0IG9mIHRoZSBzcGVjLCB1c2VmdWwgdG8gYnVpbGQgaXQgZnJvbSBzY3JhdGNoLgotIFtPcGVuQVBJIFZpc3VhbCBNYXBdKGh0dHA6Ly9vcGVuYXBpLW1hcC5hcGloYW5keW1hbi5pby8/dmVyc2lvbj0zLjApOiBWaXN1YWwgcmVwcmVzZW50YXRpb24gb2YgdGhlIHNwZWMgdXNpbmcgYW4gaW50ZXJhY3RpdmUgR1VJIHRvIG5hdmlnYXRlIHRocm91Z2ggaXRzIGNvbXBvbmVudHMuCg== readmeEtag: '"c01c958911e1dcde69044c23a149a1e65108d47b"' readmeLastModified: Mon, 11 Nov 2024 22:01:59 GMT repositoryId: 638743792 description: OpenAPI output generated in SQL for a PostgREST instance created: '2023-05-10T02:33:03Z' updated: '2025-11-05T14:01:19Z' language: PLpgSQL archived: false stars: 25 watchers: 5 forks: 6 owner: PostgREST logo: https://avatars.githubusercontent.com/u/15115011?v=4 license: MIT repoEtag: '"2c473f33f6322840e96775801e62e1ba82e77911cb029f1136902d2cd675f94c"' repoLastModified: Wed, 05 Nov 2025 14:01:19 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/azcodingaccount/igomokugame v3: true id: b51a5e8608a4db43e47c31fe151fe98c repositoryMetadata: base64Readme: >- IyBpR29tb2t1R2FtZQoKPGhyPgoKIyMg5LuL57uN8J+TmAoK4oCLCQlpR29tb2t1R2FtZeaYr+S4gOS4quWfuuS6jioqVnVlMyoq44CBKipTcHJpbmdCb290Myoq77yMKipFbGVtZW50LVBsdXMqKuOAgSoqV2ViU29ja2V0KirjgIEqKlNwcmluZ1Rhc2sqKuOAgSoqRGF0YVYqKuOAgSoq5LqU5a2Q5qOLQUnnrpfms5UqKuetieWunueOsOeahOS4gOasvuWcqOe6v+S6lOWtkOaji+a4uOaIj+OAguaUr+aMgSoq5o6S6KGM5qacKirjgIEqKuWlveWPiyoq44CBKirkurrmnLrlr7nmiJgqKuOAgSoq5Lq65py65LqS5YqoKirjgIEqKuiBlOacuuWvueaImCoq44CBKirmlbDmja7lj6/op4bljJbliIbmnpAqKuetieWKn+iDveOAgumhueebruS9v+eUqOacgOaWsOaKgOacr+W8gOWPke+8jOW8gOWPkei/h+eoi+inhOiMg++8jOmAu+i+keS4peiwqOOAggoKQuermemhueebruS7i+e7jeWcsOWdgO+8miBodHRwczovL2IyMy50di9waDRPejVsCgojIyDlnKjnur/pooTop4jwn5GACgotIOWJjeWPsO+8mmh0dHA6Ly9nYW1lLmJ1Z2Rlc2lnbmVyLmNuCQnotKblj7dgZGVtbzFgOmBkZW1vMWDjgIJgZGVtbzJgOmBkZW1vMmDjgIJgZGVtbzNgOmBkZW1vM2Ag77yM5oKo5Lmf5Y+v5Lul6Ieq6KGM5rOo5YaM6LSm5Y+3Ci0g5ZCO5Y+w77yaaHR0cDovL2dhbWUuYnVnZGVzaWduZXIuY24vYWRtaW4gICAg6LSm5Y+377yaYGFkbWluYCDlr4bnoIHvvJpgYWRtaW5gCgrihLnvuI8g5a+55LqO55So5oi356uv77yM5oKo5Y+v6IO96ZyA6KaBKipDdHJsK+a7mui9rioq5bCG5bGP5bmV57yp5pS+5Yiw5LiA5a6a5bC65a+477yI5LiA6Iis5pivODAl77yJCgrihLnvuI8g56ys5LiA5qyh6K6/6Zeu6K+36ICQ5b+D562J5b6F5rWP6KeI5Zmo5LuO5pyN5Yqh5Zmo5ouJ5Y+W6LWE5rqQCgojIyDms6jmhI/wn5SSCgrnlLHkuo7mnKzpobnnm67kuLvopoHkuLrmvJTnpLrpobnnm67vvIzlpoLpnIDms6jlhozotKblj7fvvIzor7fpgb/lhY3kvb/nlKjkuKrkurrluLjnlKjnlKjmiLflkI3kuI7lr4bnoIHvvIzku6XlhY3lm6DmlbDmja7ms4TpnLLooqvpu5HlrqLojrflj5bov5vooYwqKuaSnuW6k+aUu+WHuyoqCgojIyDku6PnoIHku5PlupPwn4yfCgotIEdpdGVl77yaaHR0cHM6Ly9naXRlZS5jb20vQWxiZXJ0X2hhbi9pLWdvbW9rdS1nYW1lCi0gR2l0SHVi77yaaHR0cHM6Ly9naXRodWIuY29tL0FaQ29kaW5nQWNjb3VudC9pR29tb2t1R2FtZQoKIyMg6aG555uu5Lqu54K58J+SoQoKMS4g5oqA5pyv5paw44CC6YeH55So5pyA5paw54mI5LyB5Lia5Li75rWB5Y2V5L2T5bqU55So5byA5Y+R5oqA5pyv44CCCjIuIOW6lOeUqOaWsOOAguaOoue0okFJ6K+t6Z+z44CB5LqS5Yqo5Lqk5Y+L5LiO5qOL57G75ri45oiP57uT5ZCI55qE5paw5Y+v6IO944CCCjMuIOeul+azleaWsOOAguWunueOsOe7j+WFuOeul+azleW5tuWKoOWFpeWOn+WIm+mDqOWIhuOAggo0LiDmnInlub/luqbjgILlrozmlbRDUlVE44CBQUnnrpfms5XjgIHmlbDmja7lrp7ml7bpgJrkv6HjgIHmlbDmja7lj6/op4bljJblupTnlKjnrYkKCiMjIOmhueebruaKgOacr+W6lOeUqPCfm6DvuI8KCjEuIOacrOmhueebrumHh+eUqGBWdWUzYCtgU3ByaW5nQm9vdDNg5Li65Li76KaB5byA5Y+R5oqA5pyv44CCCgoyLiDkvb/nlKhgU3ByaW5nIFRhc2tg5oqA5pyv5a6e546w5a6a5pe25Lu75YqhLOiHquWKqOWFs+mXreaIv+mXtOOAggoKMy4g5L2/55SoYFdlYlNvY2tldGDmioDmnK/lrp7njrDnlKjmiLfogYrlpKnkuI7ogZTmnLrlr7nmiJjnmoTlrp7njrDjgIIKCjQuIOS9v+eUqGDljZrlvIjmoJFgK2DmnoHlpKfmnoHlsI/mkJzntKJgK2BBbHBoYS1CZXRh5Ymq5p6dYCtg6K+E5Lyw5Ye95pWwYOWunueOsOS6uuacuuWvueaImAoKNS4g5L2/55SoYEVjaGFydHNg5ZKMYERhdGFWaWV3YOWunueOsOaVsOaNruWPr+inhuWMlgoKNi4gYEhUVFDnn63ova7or6Jg5a6e546w5pWw5o2u5Y+v6KeG5YyW5Lit5pWw5o2u55qE5pu05pawCgo3LiDkvb/nlKjliY3nq6/nmoRgRXhjZWxg5YyF5a6e546w5pWw5o2u55qE5a+85YWl5a+85Ye6Cgo4LiDkvb/nlKhgYW5pbWF0ZS5jc3Ng5a6e546w5ri45oiP57uT5p2f5pe25Yqo55S755qE5pKt5pS+Cgo5LiDkvb/nlKhgVFRTYOaKgOacr+WunueOsOaWh+acrOi9rOivremfs+WujOaIkEFJ5LiO55So5oi35LqS5Yqo55qE5Yqf6IO9CgoxMC4g5Z+65LqOYGtuaWZlNGpg6YG15b6qYE9wZW5BUEkzYOazqOino+inhOiMg+iHquWKqOeUn+aIkOaOpeWPo+aWh+ahowoKMTEuIOS9v+eUqGBJUDJSZWdpb25g5a6e546w5qC55o2uSVDojrflj5bnlKjmiLflnLDlnYAKCiMjIOW/q+mAn+W8gOWniyDwn5qACgotICoq5ouJ5Y+W6aG555uuKiogKOaCqOmcgOimgeWFiOWuieijhUdpdCkKCmBgYGJhc2gKIyBHaXRlZQpnaXQgcHVsbCBodHRwczovL2dpdGVlLmNvbS9BbGJlcnRfaGFuL2ktZ29tb2t1LWdhbWUuZ2l0CiMgR2l0SHViCmdpdCBwdWxsIGh0dHBzOi8vZ2l0aHViLmNvbS9BWkNvZGluZ0FjY291bnQvaUdvbW9rdUdhbWUKYGBgCgotICoq6L+Q6KGM5YmN56uv6aG555uuKioKCmBgYGJhc2gKY2Qg5YmN56uv6aG555uu55uu5b2VCnBucG0gaQkgICAjIOWuieijheS+nei1lgpwbnBtIGRldiAgICMg6L+Q6KGM56iL5bqPCQpgYGAKCi0gKirov5DooYzmlbDmja7lupPohJrmnKwqKgoKaW5pdC5zcWzmlofku7blnKggIGAvaUdvbW9rdUdhbWUtU2VydmVyL3NyYy9tYWluL3Jlc291cmNlcy9zcWxg55uu5b2V5LiL44CCCgox77ya5Y+z6ZSu5L2/55SoaWRlYei/kOihjAoKMu+8muaJvuWIsOS4gOasvuaVsOaNruW6k+euoeeQhuW3peWFt++8jOWmgk5hdmljYXQg77yMRGF0YUdyaXDnrYnvvIzlr7zlhaVzcWzmlofku7bvvIzov5DooYzjgIIKCjPvvJrkvb/nlKjlkb3ku6TooYzov5DooYwKCmBgYGJhc2gKbXlzcWwgLXUgdXNlcm5hbWUgLXAgPCAuL2luaXQuc3FsCSMgc3Fs5paH5Lu255qE55u45a+55oiW57ud5a+56Lev5b6ECmBgYAoKLSAqKui/kOihjOWQjuerr+mhueebrioqCgox77ya5Y+z6ZSu5L2/55SoaWRlYeaJk+W8gOOAgueCueWHu3J1buWNs+WPr+i/kOihjAoKMu+8muS9v+eUqOWRveS7pOihjOi/kOihjAoKYGBgYmFzaAogY2Qg5YmN56uv6aG555uu55uu5b2VCiBtdm4gY2xlYW4gcGFja2FnZQkjIOaJk+WMhemhueebrgogY2QgdGFyZ2V0IAogamF2YSAtamFyIGphcuWMheWQjSAgIyDov5DooYzpobnnm64KYGBgCgoqKumhueebrumDqOe9sioqCgrpobnnm67pg6jnvbLor7fnp7vmraXljZrlrqLmlofnq6DvvJpbRG9ja2Vy6YOo572ySmF2YemhueebrueahOatpemqpCBdKGh0dHBzOi8vYmxvZy5idWdkZXNpZ25lci5jbi9kb2NrZXLpg6jnvbJqYXZh6aG555uu55qE5q2l6aqkLykKCiMjIOmhueebrumDqOWIhueVjOmdoumihOiniPCfkYHvuI8KCi0g5Li76aG1CgohW2ltYWdlLTIwMjMxMjI2MTYwMDQzODYzXShodHRwczovL215LXBpY3R1cmUtYmVkMS0xMzIxMTAwMjAxLmNvcy5hcC1iZWlqaW5nLm15cWNsb3VkLmNvbS9teXBpY3R1cmVzL2ltYWdlLTIwMjMxMjI2MTYwMDQzODYzLnBuZykKCi0g5Lq65py65a+55oiY6aG1CgohW2ltYWdlLTIwMjMxMjI2MTkxNjA0Mjc2XShodHRwczovL215LXBpY3R1cmUtYmVkMS0xMzIxMTAwMjAxLmNvcy5hcC1iZWlqaW5nLm15cWNsb3VkLmNvbS9teXBpY3R1cmVzL2ltYWdlLTIwMjMxMjI2MTkxNjA0Mjc2LnBuZykKCgoKLSDlkI7lj7DpppbpobUKCiFbaW1hZ2UtMjAyMzEyMjYxNjAzMDA0MTVdKGh0dHBzOi8vbXktcGljdHVyZS1iZWQxLTEzMjExMDAyMDEuY29zLmFwLWJlaWppbmcubXlxY2xvdWQuY29tL215cGljdHVyZXMvaW1hZ2UtMjAyMzEyMjYxNjAzMDA0MTUucG5nKQoKCgotIOWPr+inhuWMluWkp+WxjwoKIVtpbWFnZS0yMDIzMTIyNjE2MDQzNTc1Ml0oaHR0cHM6Ly9teS1waWN0dXJlLWJlZDEtMTMyMTEwMDIwMS5jb3MuYXAtYmVpamluZy5teXFjbG91ZC5jb20vbXlwaWN0dXJlcy9pbWFnZS0yMDIzMTIyNjE2MDQzNTc1Mi5wbmcpCgoKCi0g5rWB6YeP5YiG5p6Q6aG1CgohW2ltYWdlLTIwMjMxMjI2MTYwNTA3MTM0XShodHRwczovL215LXBpY3R1cmUtYmVkMS0xMzIxMTAwMjAxLmNvcy5hcC1iZWlqaW5nLm15cWNsb3VkLmNvbS9teXBpY3R1cmVzL2ltYWdlLTIwMjMxMjI2MTYwNTA3MTM0LnBuZykKCiMjIOm4o+iwovCfjLkKCjEuIFRUU+aWh+acrOi9rOa8lOiusuaKgOacryAKCuKAiwkJQUnlmInnhLbvvJrmqKHlnovkvZzogIXvvJpgWHrkuZTluIxgCgrigIsJCUFJ5LiB55yf77ya6a2U5aGU56S+5Yy66aG555uu5Zyw5Z2A77yaaHR0cHM6Ly9tb2RlbHNjb3BlLmNuL3N0dWRpb3MvTWlEZDFFeWUvRFotQmVydC1WSVRTMi9zdW1tYXJ5CgrigIsJ5LqM6ICF5Z+65LqOQmVydC1WSVRTMuaooeWei++8jEdpdGh1YumhueebruWcsOWdgO+8miAgaHR0cHM6Ly9naXRodWIuY29tL2Zpc2hhdWRpby9CZXJ0LVZJVFMyCgoyLiBEYXRhVuaVsOaNruWPr+inhuWMluaKgOacrwoKICAg5a6Y572R77yaaHR0cDovL2RhdGF2LmppYW1pbmdoaS5jb20vZ3VpZGUvCgozLiBLbmlmZTRqCgrigIsJCUdpdGh1YumhueebruWcsOWdgO+8mmh0dHBzOi8vZ2l0aHViLmNvbS94aWFveW1pbi9rbmlmZTRqCgoyLiBJUDJSZWdpb24KCiAgIEdpdGh1YumhueebruWcsOWdgO+8mmh0dHBzOi8vZ2l0aHViLmNvbS9saW9uc291bDIwMTQvaXAycmVnaW9uCgozLiDkupTlrZDmo4tBSeeul+azleWPguiAg+WNmuWuogoKICAgW+OAkOeul+azleOAkeaegeWkp+aegeWwj86xLc6y5Ymq5p6d566X5rOVIC0g55+l5LmOICh6aGlodS5jb20pXShodHRwczovL3podWFubGFuLnpoaWh1LmNvbS9wLzY1MjE5NDQ2KQoKIyMg5o2Q6LWg8J+NtQoK5aaC5p6c5oKo6K6k5Li66L+Z5Liq6aG555uu5a+55oKo5pyJ5biu5Yqp77yM5Y+v5Lul6YCa6L+H5LiL6Z2i5pa55byP5pSv5oyB5oiRCgotIFN0YXLjgIFGb3Jr44CBV2F0Y2gg5LiA6ZSu5LiJ6L+eIPCfjJ8KLSDpgJrov4flvq7kv6HvvIzmlK/ku5jlrp3or7fmiJHllp3mna/lpbbojLYg4p2kCgp8ICAgICAgICAgICAgICAgICAgICAgICAgICAgICDlvq7kv6EgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgICAgICAgICAgICAgICAgICAgICAgICAgICAg5pSv5LuY5a6dICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCA6LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLTogfCA6LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLTogfAp8IDxpbWcgc3JjPSJodHRwczovL215LXBpY3R1cmUtYmVkMS0xMzIxMTAwMjAxLmNvcy5hcC1iZWlqaW5nLm15cWNsb3VkLmNvbS9teXBpY3R1cmVzL3dlY2hhdC5qcGciIGFsdD0id2VjaGF0IiBzdHlsZT0iem9vbTogMTUlO21hcmdpbjowIGF1dG8iIC8+IHwgPGltZyBzcmM9Imh0dHBzOi8vbXktcGljdHVyZS1iZWQxLTEzMjExMDAyMDEuY29zLmFwLWJlaWppbmcubXlxY2xvdWQuY29tL215cGljdHVyZXMvYWxpcGF5LmpwZyIgYWx0PSJhbGlwYXkiIHN0eWxlPSJ6b29tOjE1JTttYXJnaW46MCBhdXRvIiAvPiB8CgoKCg== readmeEtag: '"ce42366b098d6304b2e5db990453941e7e2dcbcb"' readmeLastModified: Sat, 25 May 2024 14:11:27 GMT repositoryId: 735775072 description: 一个基于vue3、springboot3、websocket、alpha-beta cut的五子棋游戏 created: '2023-12-26T03:35:56Z' updated: '2026-01-26T10:45:13Z' language: Java archived: false stars: 31 watchers: 3 forks: 3 owner: AZCodingAccount logo: https://avatars.githubusercontent.com/u/145426558?v=4 license: MIT repoEtag: '"cdf00b891a2c444c9f500bbf6ecb7fa77d200ba7cee81e79e8104cb7bd3eacdb"' repoLastModified: Mon, 26 Jan 2026 10:45:13 GMT category: Code Generators foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/apiaddicts/apigen.springboot v3: true v3_1: true id: fd840bafced1deba1ac3d162e2c56928 repositoryMetadata: base64Readme: >- 
# 🛠️ Apigen ![Release](https://img.shields.io/badge/release-2.0.2-purple) ![Swagger](https://img.shields.io/badge/-openapi-%23Clojure?style=flat&logo=swagger&logoColor=white) ![Java](https://img.shields.io/badge/java-%23ED8B00.svg?style=flat&logo=openjdk&logoColor=white) ![Spring](https://img.shields.io/badge/spring-%236DB33F.svg?style=flat&logo=spring&logoColor=white) [![License: LGPL v3](https://img.shields.io/badge/license-LGPL_v3-blue.svg)](https://www.gnu.org/licenses/lgpl-3.0)

Welcome to **apigen.springboot**, made in spain ![Spain](https://raw.githubusercontent.com/stevenrskelton/flag-icon/master/png/16/country-4x3/es.png "Spain"), the opensource project in Java that allows you to generate an archetype of the springboot framework using the openapi file as a mapping tool between the openapi definition and the database. Click maven to see available mvnrepository dependencies.

<a title="mvnrepository" href="https://mvnrepository.com/artifact/org.apiaddicts.apitools.apigen/apigen">
    <img width="150"  src="https://blog.irontec.com/wp-content/uploads/2019/12/1280px-Maven_logo.svg_-300x76.png">
</a>

### This repository is intended for :octocat: **community** use, it can be modified and adapted without commercial use. If you need a version, support or help for your **enterprise** or project, please contact us 📧 devrel@apiaddicts.org

[![Twitter](https://img.shields.io/badge/Twitter-%23000000.svg?style=for-the-badge&logo=x&logoColor=white)](https://twitter.com/APIAddicts)
[![Discord](https://img.shields.io/badge/Discord-%235865F2.svg?style=for-the-badge&logo=discord&logoColor=white)](https://discord.gg/ZdbGqMBYy8)
[![LinkedIn](https://img.shields.io/badge/linkedin-%230077B5.svg?style=for-the-badge&logo=linkedin&logoColor=white)](https://www.linkedin.com/company/apiaddicts/)
[![Facebook](https://img.shields.io/badge/Facebook-%231877F2.svg?style=for-the-badge&logo=Facebook&logoColor=white)](https://www.facebook.com/apiaddicts)
[![YouTube](https://img.shields.io/badge/YouTube-%23FF0000.svg?style=for-the-badge&logo=YouTube&logoColor=white)](https://www.youtube.com/@APIAddictslmaoo)

# 🙌 Join the **Apigen** Adopters list
📢 If Apigen is part of your organization's toolkit, we kindly encourage you to include your company's name in our Adopters list. 🙏 This not only significantly boosts the project's visibility and reputation but also represents a small yet impactful way to give back to the project.

| Organization  | Description of Use / Referenc |
|---|---|
|  [CloudAppi](https://cloudappi.net/)  | Apification and generation of microservices |
| [Acciona](https://www.acciona.com/)  | Generation of microservices |
| [Madrid Digital](https://www.comunidad.madrid/servicios/sede-electronica/madrid-digital/)  | Generation of microservices  |
| [Apiquality](https://apiquality.io/)  | Generation of microservices  |

# 👩🏽‍💻  Contribute to ApiAddicts

We're an inclusive and open community, welcoming you to join our effort to enhance ApiAddicts, and we're excited to prioritize tasks based on community input, inviting you to review and collaborate through our GitHub issue tracker.

Feel free to drop by and greet us on our GitHub discussion or Discord chat. You can also show your support by giving us some GitHub stars ⭐️, or by following us on Twitter, LinkedIn, and subscribing to our YouTube channel! 🚀

[!["Buy Me A Coffee"](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/apiaddicts)


# 📑 Getting started

## 🖥️ CLI

```bash
java -jar generator-cli.jar generate -f openapi.yaml -o ./code

java -jar generator-cli.jar help
```

## 🐋 Docker compose

```yaml
version: "3.3"
services:
  apigen:
    image: "apiaddicts/apitools-apigen:2.0.2"
    ports:
      - "8080:8080"
```

```bash
curl -X POST "http://localhost:8080/generator/file" \
  -F "file=@openapi.yaml"
```

## Project

### Scheme
    x-apigen-project:
      name: string
      description: string
      version: string
      java-properties:
        group-id: string
        artifact-id: string

### Definition
- `x-apigen-project`: Section where project information is defined
  - `name`: Project name
  - `description`: Brief project description
  - `version`: Project version
  - `java-properties`: Section where specific Java project information is defined
    - `group-id`: Initial package name where the project resides. If it consists of multiple words, they will be separated by `.`
    - `artifact-id`: Identifier name for the project
    - `base-package`: Name of the base package to use in the generated code; if not specified, it is obtained from the `group-id` and `artifact-id`
  - `standard-response-operations`: Section where standard apigen response transformation operations are defined (optional)
    - `<jsonpatch operation node>`: Node that complies with the json-patch standard, allowing optional operations to be declared on all elements of an array

### Samples
    x-apigen-project:
      name: Colors
      description: This would be the Colors project
      version: 1.0.0
      java-properties:
        group-id: the.group
        artifact-id: app

## Models

### Scheme
    x-apigen-models:
      <model_name>:
        relational-persistence:
          table: string
        attributes:
          - name: string
            type: string [ENUM[Array, String, Boolean, Double, Float, BigDecimal, Integer, Long, BigInteger, LocalDate, LocalDateTime, ZonedDateTime, OffsetDateTime, Instant, ComposedID] o <model_name>]
            items-type: string [<model_name>]
            relational-persistence:
              column: string
              primary-key: boolean
              foreign-column: string
              intermediate-table: string
              owner: boolean
            validations:
              - type: string [ENUM[NotNull, Size, Min, Max, Email, NotEmpty, NotBlank, Positive, PositiveOrZero, Negative, NegativeOrZero, Past, PastOrPresent, Future, FutureOrPresent, Pattern, Digits, DecimalMin, DecimalMax]]
                min: integer
                max: integer
                regex: string
                value: integer | string
                integer: integer
                fraction: integer
                inclusive: boolean   

### Definition
- `x-apigen-models`: Section where project models are defined
  - `<model_name>`: Name of the model
    - `relational-persistence`: Section where all aspects related to the model's relational persistence are indicated
      - `table`: Name of the table in the database
    - `attributes`: Section that contains the model's attributes
      - `name`: Name of the attribute
      - `type`: Type of the attribute, supported types are: [Array, String, Boolean, Double, Float, Integer, Long, LocalDate, OffsetDateTime, ComposedID], or in the case of another model, the name of that model
      - `items-type`: In the case of the `type` being an `Array`, this field should be defined with the name of the referenced model
      - `attributes`: Similar to the `attributes` of the model, only used when the `type` is ComposedID to indicate the attributes that make up the identifier
      - `relational-persistence`: Section where all aspects related to the relational persistence of the model's attribute are indicated
        - `column`: Name of the column in the database
        - `columns`: Relationship of columns from the current table to another related table when dealing with a model with a composite identifier (each entry is defined as key: value where the key is the name in the current table and the value is the name in the related table)
        - `primary-key`: Indicates if it is the primary key, if not specified, the default value will be `false`
        - `foreign-column`: Indicates the name of the column in the related table
        - `foreign-columns`: Relationship of columns in the related table to the current table when dealing with a model with a composite identifier (each entry is defined as key: value where the key is the name in the related table and the value is the name in the current table)
        - `intermediate-table`: Name of the intermediate table in those attributes that represent a many-to-many relationship
        - `owner`: Indicates if this part of the relationship is the owner, necessary in one-to-one or one-to-many relationships
        - `sequence`: Section where the optional generator to use for the primary key can be indicated
      - `validations`: Section where attribute validations are defined
        - `type`: The supported types for validation are as follows: [NotNull, Size, Min, Max, Email, NotEmpty, NotBlank, Positive, PositiveOrZero, Negative, NegativeOrZero, Past, PastOrPresent, Future, FutureOrPresent, Pattern, Digits, DecimalMin, DecimalMax]
        - `min`: Value used when the `type` field is `Size`
        - `max`: Value used when the `type` field is `Size`
        - `regex`: Value used when the `type` field is `Pattern`
        - `value`: Value used when the `type` field is `Min`, `Max`, `DecimalMin`, or `DecimalMax`
        - `integer`: Value used when the `type` field is `Digits`, representing the maximum number of digits in the integer part of the number
        - `fraction`: Value used when the `type` field is `Digits`, representing the maximum number of digits in the decimal part of the number
        - `inclusive`: Value used when the `type` field is `DecimalMin` or `DecimalMax`

### Samples
    x-apigen-models:
      Color:
        relational-persistence:
          table: colors
        attributes:
          - name: id
            type: Long
            relational-persistence:
              column: id
              primary-key: true
            validations:
              - type: NotNull
          - name: name
            type: String
            relational-persistence:
              column: name
            validations:
              - type: Pattern
                regex: [A-Z]+
      Composed:
        relational-persistence:
          table: composed
        attributres:
          - name: id
            type: ComposedID
            validations:
               - type: NotNull
            attributes:
              - name: one
                type: String
                relational-persistence:
                  column: c_one
                validations:
                 - type: NotNull
              - name: two
                type: Integer      
                relational-persistence:
                  column: c_two      
                validations:
                 - type: NotNull

## Path expansion

### Scheme
    paths:
      <path>:
        ...
        x-apigen-binding:
          model: string

### Definition
- `x-apigen-binding`: Section where the connection between the endpoint and the model we will use is defined
  - `model`: In this field, we can specify the name of the model.
  - `child-model`: In this field, we will put the name of the child model. Only necessary for self-managed endpoints of parent-child type entities.
  - `child-parent-relation-property`: In this field, we will put the name of the property in the child that relates to the parent. Only necessary for self-managed endpoints of parent-child type entities.

### Sample
    paths:
      /colors:
        x-apigen-binding:
          model: Color

## Extension of the schema section of the requestBody

### Scheme

    schemas:
      <schema>:
        x-apigen-mapping:
          model: string
      type: object
      properties:
        <prop_name>:
          type: string
          x-apigen-mapping:
            model: string
            field: string
        <prop_name_2>:
          x-apigen-mapping:
            model: string

### Definition

- `x-apigen-mapping`: Section where all mapping data between resources and models are defined. Each attribute of an input resource, including the input resource itself, can have this section defined. If it is not defined, it is assumed to have the default values indicated in each section.
  - `model`: Name of the model that the resource or attribute represents. If not specified at the resource level, it is not considered a standard input resource.
  - `field`: Name of the model attribute to which the resource attribute will be mapped. By default, if nothing is specified, it will be mapped to an attribute with the same name if it exists; otherwise, it will be ignored. There are two specific cases in which it is mandatory to define this field:
    - When the name of the attribute in the resource and in the model are different:
      ```yaml
      nombre:
        type: string
        x-apigen-mapping:
          field: primerNombre
      ```
    - When the attribute in the resource is an abbreviation of the identifier of a nested model:
      ```yaml
      color:
        type: string
        x-apigen-mapping:
          model: color
          field: valorCromatico.id
      ```

## Expansion of the response outline section

### Preconditions

The Apigen extension for OpenAPI forces us to take into account a series of conditions so that endpoint responses comply with the standard.

#### Have a defined standard response

````yaml
  schemas:
    standard_response_result:
      properties:
        result:
          type: object
          properties:
            status:
              type: boolean
            http_code:
              type: integer
            errors:
              type: array
              items:
                $ref: '#/components/schemas/standard_error'
            info:
              type: string
            trace_id:
              type: string
            num_elements:
              type: integer
          required:
            - status
            - http_code
            - trace_id
    standard_error:
      type: object
      properties:
        code:
          type: integer
        message:
          type: string
````

#### Have a single standard collection response defined for each model

````yaml
  schemas:
    standard_response_collection_<model>:
      allOf:
        - $ref: "#/components/schemas/standard_response_result"
        - type: object
          properties:
            data:
              type: object
              properties:
                <model_plural>:
                  type: array
                  items:
                    $ref: "#/components/schemas/<model_resource>"
````

#### Have a single simple standard response defined for each model

````yaml
  schemas:
    standard_response_<model>:
      allOf:
        - $ref: "#/components/schemas/standard_response_result"
        - type: object
          properties:
            data:
              $ref: "#/components/schemas/<model_resource>"
````

### Scheme

    schemas:
      <schema>:
        x-apigen-mapping:
          model: string
      type: object
      properties:
        <prop_name>:
          type: string
          x-apigen-mapping:
            field: string

### Definition
- `x-apigen-mapping`: Section where all mapping data between resources and models are defined. Each output resource must have this section defined.
  - `model`: Name of the model that the resource represents.
  - `field`: Name of the model attribute to which the resource attribute will be mapped.

### Sample

    color:
      x-apigen-mapping:
        model: Color
      type: object
      properties:
        json_id:
          type: integer
          x-apigen-mapping:
            field: id
        name:
          type: string
        forms:
          $ref: "#/components/schemas/forms"


## 💛 Sponsors
<p align="center">
	<a href="https://apiaddicts.org/">
    	<img src="https://apiaddicts.cloudappi.net/web/image/4248/LOGOCloudappi2020Versiones-01.png" alt="cloudappi" width="150"/>
        <img src="https://www.comunidad.madrid/sites/default/files/styles/block_teaser_image/public/img/logos-simbolos/logo_centrado_md.png?itok=4rTUhmcj" alt="md" width="150"/>
        <img src="https://apiquality.io/wp-content/uploads/2022/09/cropped-logo-apiquality-principal-1-170x70.png" height = "75">
        <img src="https://apiaddicts-web.s3.eu-west-1.amazonaws.com/wp-content/uploads/2022/03/17155736/cropped-APIAddicts-logotipo_rojo.png" height = "75">
	</a>
</p> readmeEtag: '"31288d1b58f708375566235e5058fa977dc4ce06"' readmeLastModified: Tue, 03 Feb 2026 11:19:57 GMT repositoryId: 317847354 description: >- Apigen allow generate springboot archetipe using openapi file as mapping tool between the openapi definition and the database. created: '2020-12-02T11:56:38Z' updated: '2026-02-03T12:04:45Z' language: Java archived: false stars: 29 watchers: 2 forks: 3 owner: apiaddicts logo: https://avatars.githubusercontent.com/u/31730093?v=4 license: LGPL-3.0 repoEtag: '"8ca4a3599e24471c8b93f836c7e0386376bb526f81b450c35ee50026d5d4e24c"' repoLastModified: Tue, 03 Feb 2026 12:04:45 GMT category: SDK foundInMaster: true oldLocations: - https://github.com/apiaddicts/apigen - source: - openapi3 tags - openapi31 tags repository: https://github.com/fatal1ty/openapify v3: true v3_1: true id: 2e5e94dab0e4592660585f4aecadb841 repositoryMetadata: base64Readme: >- # openapify

###### Framework agnostic OpenAPI Specification generation for code lovers

[![Build Status](https://github.com/Fatal1ty/openapify/workflows/tests/badge.svg)](https://github.com/Fatal1ty/openapify/actions)
[![Latest Version](https://img.shields.io/pypi/v/openapify.svg)](https://pypi.python.org/pypi/openapify)
[![Python Version](https://img.shields.io/pypi/pyversions/openapify.svg)](https://pypi.python.org/pypi/openapify)
[![License](https://img.shields.io/badge/License-Apache%202.0-blue.svg)](https://opensource.org/licenses/Apache-2.0)

---

This library is designed for code-first people who don't want to bother diving
into the details
of [OpenAPI Specification](https://spec.openapis.org/oas/v3.1.0), but who
instead want to use advantages of Python typing system, IDE code-completion and
static type checkers to continuously build the API documentation and keep it
always up to date.

Openapify is based on the idea of applying decorators on route handlers. Any
web-framework has a routing system that let us link a route to a handler
(a high-level function or a class method). By using decorators, we can add
information about requests, responses and other details that will then be used
to create an entire OpenAPI document.

> [!WARNING]\
> This library is currently in pre-release stage and may have backward
> incompatible changes prior to version 1.0. Please use caution when using this
> library in production environments and be sure to thoroughly test any updates
> before upgrading to a new version.

Table of contents
--------------------------------------------------------------------------------

* [Installation](#installation)
* [Quickstart](#quickstart)
* [Building the OpenAPI Document](#building-the-openapi-document)
* [Integration with web-frameworks](#integration-with-web-frameworks)
    * [aiohttp](#aiohttp)
    * [Writing your own integration](#writing-your-own-integration)
* [Decorators](#decorators)
    * [Generic operation info](#generic-operation-info)
    * [Request](#request)
    * [Response](#response)
    * [Security requirements](#security-requirements)
* [Plugins](#plugins)
  * [`schema_helper`](#schema_helper)
  * [`media_type_helper`](#media_type_helper)

Installation
--------------------------------------------------------------------------------

Use pip to install:

```shell
$ pip install openapify
```

Quickstart
--------------------------------------------------------------------------------

> [!NOTE]\
> In the following example, we will intentionally demonstrate the process of
> creating an OpenAPI document without being tied to a specific web-framework.
> However, this process may be easier on a supported web-framework.
> See [Integration with web-frameworks](#integration-with-web-frameworks) for
> more info.

Let's see how to build an OpenAPI document with openapify. Suppose we are
writing an app for a bookstore that return a list of new books. Here we have a
dataclass model `Book` that would be used as a response model in a real-life
scenario. A function `get_new_books` is our handler.

```python
from dataclasses import dataclass

@dataclass
class Book:
    title: str
    author: str
    year: int

def get_new_books(...):
    ...
```

Now we want to say that our handler returns a json serialized list of books
limited by the optional `count` parameter. We use `request_schema`
and `response_schema` decorators accordingly:

```python
from openapify import request_schema, response_schema

@request_schema(query_params={"count": int})
@response_schema(list[Book])
def get_new_books(...):
    ...
```

And now we need to collect all the route definitions and pass them to the
`build_spec` function. This function returns an object that has `to_yaml`
method.

```python
from openapify import build_spec
from openapify.core.models import RouteDef

routes = [RouteDef("/books", "get", get_new_books)]
spec = build_spec(routes)
print(spec.to_yaml())
```

As a result, we will get the following OpenAPI document which can be rendered
using tools such as Swagger UI:

```yaml
openapi: 3.1.0
info:
  title: API
  version: 1.0.0
paths:
  /books:
    get:
      parameters:
      - name: count
        in: query
        schema:
          type: integer
      responses:
        '200':
          description: OK
          content:
            application/json:
              schema:
                type: array
                items:
                  $ref: '#/components/schemas/Book'
components:
  schemas:
    Book:
      type: object
      title: Book
      properties:
        title:
          type: string
        author:
          type: string
        year:
          type: integer
      additionalProperties: false
      required:
      - title
      - author
      - year
```

Building the OpenAPI Document
--------------------------------------------------------------------------------
The final goal of this library is to build
the [OpenAPI Document](https://spec.openapis.org/oas/v3.1.0#openapi-document)
for your web-application. This document consists of common information about
the application, such as a title and version, and specific information that
outlines the functionalities of the API.

Since openapify is now based
on [apispec](https://github.com/marshmallow-code/apispec) library, the OpenAPI
document is presented by `APISpec` class for the convenience of using the
existing ecosystem of plugins. However, openapify has its own
subclass `OpenAPIDocument` which makes it easier to add some common fields,
such as an array
of [Server](https://spec.openapis.org/oas/v3.1.0#server-object) objects or
array of
common [Security Scheme](https://spec.openapis.org/oas/v3.1.0#security-scheme-object)
objects.

To build the document, there is `build_spec` function. The very basic document
can be created by calling it with an empty list of route definitions, leaving
all the parameters with their default values.
```python
from openapify import build_spec

print(build_spec([]).to_yaml())
```

As a result, we will get the following document:

```yaml
openapi: 3.1.0
info:
  title: API
  version: 1.0.0
paths: {}
```

We can change the common document attributes either by passing them
to `build_spec`:

```python
from openapify import build_spec
from openapify.core.openapi.models import HTTPSecurityScheme

build_spec(
    routes=[],
    title="My Bookstore API",
    version="1.1.0",
    openapi_version="3.1.0",
    servers=["http://127.0.0.1"],
    security_schemes={"basic_auth": HTTPSecurityScheme()}
)
```

or using a prepared `OpenAPIDocument` object:

```python
from openapify import OpenAPIDocument, build_spec
from openapify.core.openapi.models import HTTPSecurityScheme

spec = OpenAPIDocument(
    title="My Bookstore API",
    version="1.1.0",
    openapi_version="3.1.0",
    servers=["http://127.0.0.1"],
    security_schemes={"basic_auth": HTTPSecurityScheme()},
)
build_spec([], spec)
```

To add meaning to our document, we can
add [Path](https://spec.openapis.org/oas/v3.1.0#paths-object),
[Component](https://spec.openapis.org/oas/v3.1.0#components-object)
and other OpenAPI objects by applying [decorators](#decorators) on our route
handlers and constructing route definitions that will be passed to the builder.
A single complete route definition presented by `RouteDef` class can look like
this:

```python
from openapify.core.models import RouteDef
from openapify.core.openapi.models import Parameter, ParameterLocation

def get_book_by_id_handler(...):
    ...

RouteDef(
    path="/book/{id}",
    method="get",
    handler=get_book_by_id_handler,
    summary="Getting the book",
    description="Getting the book by id",
    parameters=[
        Parameter(
            name="id",
            location=ParameterLocation.PATH,
            required=True,
            schema={"type": "integer"},
        )
    ],
    tags=["book"],
)
```

As will be shown further, optional
arguments `summary`, `description`, `parameters` and `tags` can be overridden
or extended by `operation_docs` and `request_schema` decorators.

The creating of these route definitions can be automated and adapted to a
specific web-framework, and openapify has built-in support for a few of them.
See [Integration with web-frameworks](#integration-with-web-frameworks) for
details.

Integration with web-frameworks
--------------------------------------------------------------------------------

There is built-in support for a few web-frameworks, which makes creating the
documentation even easier and more fun. Any other frameworks can be integrated
with a little effort. If you are ready to take on this, you are very welcome to
create a [pull request](https://github.com/Fatal1ty/openapify/pulls).

### aiohttp

The documentation for [aiohttp](https://github.com/aio-libs/aiohttp)
web-application can be built in three ways:

* Using an already existing [`aiohttp.web.Application`](https://docs.aiohttp.org/en/stable/web_reference.html#application) object
* Using a set of [`aiohttp.web.RouteDef`](https://docs.aiohttp.org/en/stable/web_reference.html#aiohttp.web.RouteDef) objects
* Using a set of objects implementing [`AioHttpRouteDef`](https://github.com/Fatal1ty/openapify/blob/2bbf2e99c06b31fa2f1465f2ebc118884ce2bb95/openapify/ext/web/aiohttp.py#L43-L46) protocol

All we need is to pass either an application, or a set of route defs to
modified `build_spec` function. See the example:
```python
from aiohttp import web
from openapify import request_schema, response_schema
from openapify.ext.web.aiohttp import build_spec

routes = web.RouteTableDef()

@response_schema(str, media_type="text/plain")
@routes.post("/")
async def hello(request):
    return web.Response(text="Hello, world")

app = web.Application()
app.add_routes(routes)

print(build_spec(app).to_yaml())
```
As a result, we will get the following document:

```yaml
openapi: 3.1.0
info:
  title: API
  version: 1.0.0
paths:
  /:
    post:
      responses:
        '200':
          description: OK
          content:
            text/plain:
              schema:
                type: string
```

### Writing your own integration

🚧 To be described

Decorators
--------------------------------------------------------------------------------

Openapify has several decorators that embed necessary specific information for
later use when building the OpenAPI document. In general, decorators will
define the information that will be included in
the [Operation Object](https://spec.openapis.org/oas/v3.1.0#operation-object)
which describes a single API operation on a path. We will look at what each
decorator parameter is responsible for and how it is reflected in the final
document.

### Generic operation info

Decorator `operation_docs` adds generic information about the Operation object,
which includes summary, description, tags, external documentation and
deprecation marker.

```python
from openapify import operation_docs
```

#### summary

An optional, string summary, intended to apply to the operation. This affects
the value of
the [`summary`](https://spec.openapis.org/oas/v3.1.0#operation-object) field of
the Operation object.

| Possible types | Examples              |
|----------------|-----------------------|
| `str`          | `"Getting new books"` |

#### description

An optional, string description, intended to apply to the
operation. [CommonMark syntax](https://spec.commonmark.org) MAY be used for
rich text representation. This affects the value of
the [`description`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object.

| Possible types | Examples                    |
|----------------|-----------------------------|
| `str`          | `"Returns a list of books"` |

#### tags

A list of tags for API documentation control. Tags can be used for logical
grouping of operations by resources or any other qualifier. This affects the
value of the [`tags`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object.

| Possible types  | Examples   |
|-----------------|------------|
| `Sequence[str]` | `["book"]` |

#### operation_id

Unique string used to identify the operation. This affects the
value of
the [`operationId`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object.

| Possible types | Examples   |
|----------------|------------|
| `str`          | `getBooks` |

#### external_docs

Additional external documentation for the operation. It can be a single url or
(url, description) pair. This affects the value of
the [`summary`](https://spec.openapis.org/oas/v3.1.0#operation-object) field of
the Operation object.

| Possible types    | Examples                                                                  |
|-------------------|---------------------------------------------------------------------------|
| `str`             | `"https://example.org/docs/books"`                                        |
| `Tuple[str, str]` | `("https://example.org/docs/books", "External documentation for /books")` |

#### deprecated

Declares the operation to be deprecated. Consumers SHOULD refrain from usage
of the declared operation. Default value is false. This affects the value of
the [`deprecated`](https://spec.openapis.org/oas/v3.1.0#operation-object) field
of the Operation object.

| Possible types | Examples                       |
|----------------|--------------------------------|
| `bool`         | <code lang="python">True</pre> |

### Request

Decorator `request_schema` adds information about the operation requests.
Request can have a body, query parameters, headers and cookies.

```python
from openapify import request_schema
```

#### body

A request body can be described entirely by one `body` parameter of type `Body`
or partially by separate `body_*` parameters (see below).

In the first case it is `openapify.core.models.Body` object that has all the
separate `body_*` parameters inside. This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object.

In the second case it is the request body Python data type for which the JSON
Schema will be built. This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`schema`](https://spec.openapis.org/oas/v3.1.0#media-type-object) field of
Media Type object inside
the value
of [`content`](https://spec.openapis.org/oas/v3.1.0#request-body-object) field
of Request Body object.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Type</code> </td>
<td>

```python
Book
```

</td>
</tr>
<tr>
<td> <code>Body</code> </td>
<td>

```python
Body(
    value_type=Book,
    media_type="application/json",
    required=True,
    description="A book",
    example={
        "title": "Anna Karenina",
        "author": "Leo Tolstoy",
        "year": 1877,
    },
)
```

</td>
</tr>
</table>

#### media_type

A media type
or [media type range](https://www.rfc-editor.org/rfc/rfc7231#appendix-D) of the
request body. This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the key
of [`content`](https://spec.openapis.org/oas/v3.1.0#request-body-object) field
of Request Body object.

The default value is `"application/json"`.

| Possible types | Examples            |
|----------------|---------------------|
| `str`          | `"application/xml"` |

#### body_required

Determines if the request body is required in the request. Defaults to false.
This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`required`](https://spec.openapis.org/oas/v3.1.0#request-body-object)
field of Request Body object.

| Possible types | Examples |
|----------------|----------|
| `bool`         | `True`   |

#### body_description

A brief description of the request body. This could contain examples of
use. [CommonMark syntax](https://spec.commonmark.org) MAY be used for rich text
representation. This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`description`](https://spec.openapis.org/oas/v3.1.0#request-body-object)
field of Request Body object.

| Possible types | Examples   |
|----------------|------------|
| `str`          | `"A book"` |

#### body_example

Example of the request body. The example object SHOULD be in the correct format
as specified by the media type. This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`example`](https://spec.openapis.org/oas/v3.1.0#media-type-object) field
of
Media Type object inside
the value
of [`content`](https://spec.openapis.org/oas/v3.1.0#request-body-object) field
of Request Body object.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Any</code> </td>
<td>

```python
{
    "title": "Anna Karenina",
    "author": "Leo Tolstoy",
    "year": 1877,
}
```

</td>
</tr>
</table>

#### body_examples

Examples of the request body. Each example object SHOULD match the media type
and specified schema if present. This affects the value of
the [`requestBody`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`examples`](https://spec.openapis.org/oas/v3.1.0#media-type-object) field
of
Media Type object inside
the value
of [`content`](https://spec.openapis.org/oas/v3.1.0#request-body-object) field
of Request Body object.

The values of this dictionary could be either examples themselves,
or `openapify.core.openapi.models.Example` objects. In the latter case,
extended information about examples, such as a summary and description, can be
added to the [Example](https://spec.openapis.org/oas/v3.1.0#example-object)
object.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, Any]</code> </td>
<td>

```python
{
    "Anna Karenina": {
        "title": "Anna Karenina",
        "author": "Leo Tolstoy",
        "year": 1877,
    }
}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, Example]</code> </td>
<td>

```python
{
    "Anna Karenina": Example(
        value={
            "title": "Anna Karenina",
            "author": "Leo Tolstoy",
            "year": 1877,
        },
        summary="The book 'Anna Karenina'",
    )
}
```

</td>
</tr>
</table>

#### query_params

Dictionary of query parameters applicable for the operation, where the key is
the parameter name and the value can be either a Python data type or
a `QueryParam` object.

In the first case it is the Python data type for the query parameter for which
the JSON Schema will be built. This affects the value of
the [`parameters`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`schema`](https://spec.openapis.org/oas/v3.1.0#parameter-object) field of
Parameter object.

In the second case it is `openapify.core.models.QueryParam` object that can
have extended information about the parameter, such as a default value,
deprecation marker, examples etc.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, Type]</code> </td>
<td>

```python
{"count": int}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, QueryParam]</code> </td>
<td>

```python
{
    "count": QueryParam(
        value_type=int,
        default=10,
        required=True,
        description="Limits the number of books returned",
        deprecated=False,
        allowEmptyValue=False,
        example=42,
    )
}
```

</td>
</tr>
</table>

### path_params

Dictionary of path parameters applicable for the operation, where the key is
the parameter name and the value can be either a Python data type or
a `PathParam` object.

In the first case it is the Python data type for the path parameter for which
the JSON Schema will be built. This affects the value of
the [`parameters`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`schema`](https://spec.openapis.org/oas/v3.1.0#parameter-object) field of
Parameter object.

In the second case it is `openapify.core.models.PathParam` object that can
have extended information about the parameter, such as a description, examples.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, Type]</code> </td>
<td>

```python
{"id": UUID}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, PathParam]</code> </td>
<td>

```python
{
    "id": PathParam(
        value_type=UUID,
        description="ID of the book",
        example="eab9d66d-4317-464a-a995-510bd6e2148f",
    )
}
```

</td>
</tr>
</table>

#### headers

Dictionary of request headers applicable for the operation, where the key is
the header name and the value can be either a string or a `Header` object.

In the first case it is the header description. This affects the value of
the [`parameters`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`description`](https://spec.openapis.org/oas/v3.1.0#parameter-object)
field of Parameter object.

In the second case it is `openapify.core.models.Header` object that can have
extended information about the header, such as a description, deprecation
marker, examples etc.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, str]</code> </td>
<td>

```python
{"X-Requested-With": "Information about the creation of the request"}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, Header]</code> </td>
<td>

```python
{
    "X-Requested-With": Header(
        description="Information about the creation of the request",
        required=True,
        value_type=str,
        deprecated=False,
        allowEmptyValue=False,
        example="XMLHttpRequest",
    )
}
```

</td>
</tr>
</table>

#### cookies

Dictionary of request cookies applicable for the operation, where the key is
the cookie name and the value can be either a string or a `Cookie` object.

In the first case it is the cookie description. This affects the value of
the [`parameters`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`description`](https://spec.openapis.org/oas/v3.1.0#parameter-object)
field of Parameter object.

In the second case it is `openapify.core.models.Cookie` object that can have
extended information about the cookie, such as a description, deprecation
marker, examples etc.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, str]</code> </td>
<td>

```python
{"__ga": "A randomly generated number as a client ID"}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, Cookie]</code> </td>
<td>

```python
{
    "__ga": Cookie(
        description="A randomly generated number as a client ID",
        required=True,
        value_type=str,
        deprecated=False,
        allowEmptyValue=False,
        example="1.2.345678901.2345678901",
    )
}
```

</td>
</tr>
</table>

### Response

Decorator `response_schema` describes a single response from the API Operation.
Response can have an HTTP code, body and headers. If the Operation supports
more than one response, then the decorator must be applied multiple times to
cover each of them.

```python
from openapify import response_schema
```

#### body

A Python data type for the response body for which
the JSON Schema will be built. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`schema`](https://spec.openapis.org/oas/v3.1.0#media-type-object) field of
Media Type object inside the value
of [`content`](https://spec.openapis.org/oas/v3.1.0#response-object) field
of Response object.

| Possible types | Examples |
|----------------|----------|
| `Type`         | `Book`   |

#### http_code

An HTTP code of the response. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely, the patterned key in
the [Responses](https://spec.openapis.org/oas/v3.1.0#responses-object) object.

| Possible types | Examples |
|----------------|----------|
| `str`          | `"200"`  |
| `int`          | `400`    |

#### media_type

A media type
or [media type range](https://www.rfc-editor.org/rfc/rfc7231#appendix-D) of the
response body. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely, the key
of [`content`](https://spec.openapis.org/oas/v3.1.0#response-object) field of
Response object.

The default value is `"application/json"`.

| Possible types | Examples            |
|----------------|---------------------|
| `str`          | `"application/xml"` |

####

#### description

A description of the response. [CommonMark syntax](https://spec.commonmark.org)
MAY be used for rich text representation. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`description`](https://spec.openapis.org/oas/v3.1.0#response-object) field
of Response object.


| Possible types | Examples                |
|----------------|-------------------------|
| `str`          | `"Invalid ID Supplied"` |

#### headers

Dictionary of response headers applicable for the operation, where the key is
the header name and the value can be either a string or a `Header` object.

In the first case it is the header description. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`description`](https://spec.openapis.org/oas/v3.1.0#header-object)
field of Header object.

In the second case it is `openapify.core.models.Header` object that can have
extended information about the header, such as a description, deprecation
marker, examples etc.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, str]</code> </td>
<td>

```python
{"Content-Location": "An alternate location for the returned data"}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, Header]</code> </td>
<td>

```python
{
    "Content-Location": Header(
        description="An alternate location for the returned data",
        example="/index.htm",
    )
}
```

</td>
</tr>
</table>

#### example

Example of the response body. The example object SHOULD be in the correct format
as specified by the media type. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`example`](https://spec.openapis.org/oas/v3.1.0#media-type-object) field
of Media Type object inside the value
of [`content`](https://spec.openapis.org/oas/v3.1.0#response-object) field of
Response object.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Any</code> </td>
<td>

```python
{
    "title": "Anna Karenina",
    "author": "Leo Tolstoy",
    "year": 1877,
}
```

</td>
</tr>
</table>

#### examples

Examples of the response body. Each example object SHOULD match the media type
and specified schema if present. This affects the value of
the [`responses`](https://spec.openapis.org/oas/v3.1.0#operation-object)
field of the Operation object, or more precisely,
the [`examples`](https://spec.openapis.org/oas/v3.1.0#media-type-object) field
of Media Type object inside the value
of [`content`](https://spec.openapis.org/oas/v3.1.0#response-object) field of
Response object.

The values of this dictionary could be either examples themselves,
or `openapify.core.openapi.models.Example` objects. In the latter case,
extended information about examples, such as a summary and description, can be
added to the [Example](https://spec.openapis.org/oas/v3.1.0#example-object)
object.

<table>
<tr>
<th>Possible types</th>
<th>Examples</th>
</tr>
<tr>
<td> <code>Mapping[str, Any]</code> </td>
<td>

```python
{
    "Anna Karenina": {
        "title": "Anna Karenina",
        "author": "Leo Tolstoy",
        "year": 1877,
    }
}
```

</td>
</tr>
<tr>
<td> <code>Mapping[str, Example]</code> </td>
<td>

```python
{
    "Anna Karenina": Example(
        value={
            "title": "Anna Karenina",
            "author": "Leo Tolstoy",
            "year": 1877,
        },
        summary="The book 'Anna Karenina'",
    )
}
```

</td>
</tr>
</table>

### Security requirements

Decorator `security_requirements`
declares [security mechanisms](https://spec.openapis.org/oas/v3.1.0#securityRequirementObject)
that can be used for the operation.

```python
from openapify import security_requirements
```

This decorator takes one or more `SecurityRequirement` mappings, where the key
is the requirement name and the value is `SecurityScheme` object. There are
classes for
each [security scheme](https://spec.openapis.org/oas/v3.1.0#security-scheme-object)
which can be imported as follows:

```python
from openapify.core.openapi.models import (
    APIKeySecurityScheme,
    HTTPSecurityScheme,
    OAuth2SecurityScheme,
    OpenIDConnectSecurityScheme,
)
```

For example, to add authorization by token, you can write something like this:

```python
from openapify import security_requirements
from openapify.core.openapi.models import (
    APIKeySecurityScheme,
    SecuritySchemeAPIKeyLocation,
)

XAuthTokenSecurityRequirement = {
    "x-auth-token": APIKeySecurityScheme(
        name="X-Auh-Token",
        location=SecuritySchemeAPIKeyLocation.HEADER,
    )
}

@security_requirements(XAuthTokenSecurityRequirement)
def secure_operation():
    ...
```

And the generated specification document will look like this:

```yaml
openapi: 3.1.0
info:
  title: API
  version: 1.0.0
paths:
  /secure_path:
    get:
      security:
      - x-auth-token: []
components:
  securitySchemes:
    x-auth-token:
      type: apiKey
      name: X-Auh-Token
      in: header
```

Plugins
--------------------------------------------------------------------------------

Some aspects of creating an OpenAPI document can be changed using plugins.
There is `openapify.plugins.BasePlugin` base class, which has all the methods
available for definition. If you want to write a plugin that, for example, will
only generate schema for request parameters, then it will be enough for you to
define only one appropriate method, and leave the rest non-implemented.
Plugin system works by going through all registered plugins and calling
the appropriate method. If such a method raises `NotImplementedError` or
returns `None`, it is assumed that this plugin doesn't provide the necessary
functionality. Iteration stops at the first plugin that returned something
other than `None`.

Plugins are registered via the `plugins` argument of the `build_spec` function:

```python
from openapify import BasePlugin, build_spec


class MyPlugin1(BasePlugin):
    def schema_helper(...):
        # return something meaningful here, see the following chapters
        ...


build_spec(..., plugins=[MyPlugin1()])
```

### `schema_helper`

OpenAPI [Schema](https://spec.openapis.org/oas/v3.1.0#schemaObject) object
is built from python types stored in the `value_type` attribute of the
following openapify dataclasses defined in `openapify.core.models`:
* `Body`
* `Cookie`
* `Header`
* `QueryParam`

Out of the box, the schema is generated by using
[`mashumaro`](https://github.com/Fatal1ty/mashumaro) library (see the note
below), but support for third-party entity schema generators can be achieved
through `schema_helper` method. For example, here's what a plugin for pydantic
models might look like:

```python
from typing import Any

from openapify import BasePlugin
from openapify.core.models import Body, Cookie, Header, QueryParam
from pydantic import BaseModel


class PydanticSchemaPlugin(BasePlugin):
    def schema_helper(
        self,
        obj: Body | Cookie | Header | QueryParam,
        name: str | None = None,
    ) -> dict[str, Any] | None:
        if issubclass(obj.value_type, BaseModel):
            schema = obj.value_type.model_json_schema(
                ref_template="#/components/schemas/{model}"
            )
            self.spec.components.schemas.update(schema.pop("$defs", {}))
            return schema
```

> [!NOTE]\
> The [`BaseSchemaPlugin`](https://github.com/Fatal1ty/openapify/blob/master/openapify/core/base_plugins.py#L41-L64),
> which is enabled by default and has the lowest priority, is responsible for
> generating the schema. This plugin utilizes the mashumaro library for schema
> generation, which in turn incorporates its own [plugin system](https://github.com/Fatal1ty/mashumaro?tab=readme-ov-file#json-schema-plugins),
> enabling customization of JSON Schema generation and support for additional
> data types. For more nuanced modifications, particularly within nested data
> models, you can employ `BaseSchemaPlugin` with a specified list of mashumaro JSON
> Schema plugins. This approach allows for finer control over schema generation
> when needed:
> ```python
> from mashumaro.jsonschema.plugins import DocstringDescriptionPlugin
>
> from openapify import build_spec
> from openapify.core.base_plugins import BaseSchemaPlugin
>
> spec = build_spec(
>     routes=[...],
>     plugins=[
>         BaseSchemaPlugin(
>             plugins=[
>                 DocstringDescriptionPlugin(),
>             ]
>         ),
>     ],
> )
> ```

### media_type_helper

A media type is used in OpenAPI Request
[Body](https://spec.openapis.org/oas/v3.1.0#request-body-object) and
[Response](https://spec.openapis.org/oas/v3.1.0#response-object) objects.
By default, `application/octet-stream` is applied for `bytes` or `bytearray`
types, and `application/json` is applied otherwise. You can support more media
types or override existing ones with `media_type_helper` method.

Let's imagine that you have an API route that returns PNG images as the body.
You can have a separate model class representing images, but the more common
case is to use `typing.Annotated` wrapper for bytes. Here's what a plugin for
`image/png` media type might look like:

```python
from typing import Annotated, Any, Dict, Optional

from openapify import BasePlugin, build_spec, response_schema
from openapify.core.models import Body, RouteDef

ImagePNG = Annotated[bytes, "PNG"]


class ImagePNGPlugin(BasePlugin):
    def media_type_helper(
        self, body: Body, schema: Dict[str, Any]
    ) -> Optional[str]:
        if body.value_type is ImagePNG:
            return "image/png"


@response_schema(body=ImagePNG)
def foo():
    ...


routes = [RouteDef("/foo", "get", foo)]
spec = build_spec(routes, plugins=[ImagePNGPlugin()])
print(spec.to_yaml())
```

The resulting document will contain `image/png` content in the response:
```yaml
openapi: 3.1.0
info:
  title: API
  version: 1.0.0
paths:
  /foo:
    get:
      responses:
        '200':
          description: OK
          content:
            image/png:
              schema: {}
```
 readmeEtag: '"43fecb544f81b9b8c6ab4a1d89041b9081cde12c"' readmeLastModified: Mon, 08 Sep 2025 11:50:42 GMT repositoryId: 625681832 description: Framework agnostic OpenAPI Specification generation for code lovers created: '2023-04-09T21:19:19Z' updated: '2026-01-28T04:26:22Z' language: Python archived: false stars: 28 watchers: 1 forks: 0 owner: Fatal1ty logo: https://avatars.githubusercontent.com/u/323962?v=4 license: Apache-2.0 repoEtag: '"c6519917d28f2ce13be99903be843d5be357e264cea83b1e60e1c5e9e4d28152"' repoLastModified: Wed, 28 Jan 2026 04:26:22 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/skyffel/airbyte-connector-generator-poc v3: true id: 860d1acedb05489453d89925436dff89 repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KCTxwPjxpbWcgd2lkdGg9IjE2MHB4IiBzcmM9Imh0dHBzOi8vZnJhbWVydXNlcmNvbnRlbnQuY29tL2ltYWdlcy9SaE9DUVMwZTJ4OTRXWnEyZDhocGh0N0lOUlEucG5nIiAvPjwvcD4KCTxwPm1vdmluZyB5b3VyIGRhdGEgZnJvbSBhIHRvIGIsIG9uZSBzaG92ZWwgYXQgYSB0aW1lLjwvcD4KCTxwIGFsaWduPSJjZW50ZXIiPgoJCTxhIGhyZWY9IiMta2V5LWZlYXR1cmVzIj5GZWF0dXJlczwvYT4g4oCiCgkJPGEgaHJlZj0iIy1pbnN0YWxsYXRpb24iPkluc3RhbGxhdGlvbjwvYT4g4oCiCgkJPGEgaHJlZj0iIy1ob3ctdG8tdXNlIj5Ib3cgdG8gdXNlPC9hPgogICAJIDwvcD4KPC9kaXY+CgpUaGlzIGlzIGEgcHJvb2Ygb2YgY29uY2VwdCB0byBnZW5lcmF0ZSBBaXJieXRlIGxvdy1jb2RlIFlBTUwgY29ubmVjdG9ycyBmcm9tIEFQSSBkb2N1bWVudGF0aW9uLiBXZSB3YW50IHRoaXMgdG8gc2VydmUgYXMgaW5zcGlyYXRpb24gdG8gd2hhdCBjYW4gYmUgZG9uZSB3aXRoIExMTXMuIEhlcmUncyBob3cgaXQgd29ya3M6CgotIFNwZWNpZnkgYSBnb2FsLCBlLmcgIl9GZXRjaCBhbGwgcGFnZXMgcG9zdHNfIgotIFByb3ZpZGUgb25lIG9yIG1vcmUgbGlua3MgdG8gZG9jdW1lbnRhdGlvbiwgZS5nIGZvciBOb3Rpb246IFtBUEkgSW50cm9dKGh0dHBzOi8vZGV2ZWxvcGVycy5ub3Rpb24uY29tL3JlZmVyZW5jZS9pbnRybyksIFtBUEkgVmVyc2lvbmluZ10oaHR0cHM6Ly9kZXZlbG9wZXJzLm5vdGlvbi5jb20vcmVmZXJlbmNlL3ZlcnNpb25pbmcpIGFuZCBbU2VhcmNoIEVuZHBvaW50XShodHRwczovL2RldmVsb3BlcnMubm90aW9uLmNvbS9yZWZlcmVuY2UvcG9zdC1zZWFyY2gpCgpUaGlzIHdpbGwgZ2VuZXJhdGUgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW4gYW5kIGEgQWlyYnl0ZSBsb3ctY29kZSBjb25uZWN0b3IuIFRyeSBpdCEKCiMjIPCfk73vuI8gRGVtbwoKIyMjIE5vdGlvbgpodHRwczovL2dpdGh1Yi5jb20vc2t5ZmZlbC9haXJieXRlLWNvbm5lY3Rvci1nZW5lcmF0b3ItcG9jL2Fzc2V0cy8zMTM0ODk1L2Y3ZTAzZjZkLTYwZDktNDRiNi04OGVlLTNiYzBlNGJjZDMzOQoKIyMjIFBlcnBsZXhpdHkKaHR0cHM6Ly9naXRodWIuY29tL3NreWZmZWwvYWlyYnl0ZS1jb25uZWN0b3ItZ2VuZXJhdG9yLXBvYy9hc3NldHMvMjU2MjI0MTIvZTI5MjJhYWEtNGYxOS00NjA4LThmYmEtZTFlMDhjMjAxMDMzCgoKIyMg4pyFIEtleSBmZWF0dXJlcwoKLSBDby1waWxvdCBmb3IgZ2VuZXJhdGluZyBFVEwgY29kZSBmb3IgSFRUUCBBUElzCi0gU3VwcG9ydCBBaXJieXRlIGxvdy1jb2RlIFlBTUwgY29ubmVjdG9ycwotIFNjcmFwZXMgQVBJIGRvY3VtZW50YXRpb24KLSBQcm9kdWNlcyBPcGVuQVBJIHNwZWNpZmljYXRpb25zCgojIyDwn5OmIEluc3RhbGxhdGlvbgoKMS4gQ2xvbmUgdGhlIHJlcG9zaXRvcnkKCiAgIGBgYGJhc2gKICAgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9za3lmZmVsL2FpcmJ5dGUtY29ubmVjdG9yLWdlbmVyYXRvci1wb2MKICAgY2QgYWlyYnl0ZS1jb25uZWN0b3ItZ2VuZXJhdG9yLXBvYwogICBgYGAKCjIuIFNldHVwIHZpcnR1YWwgZW52aXJvbm1lbnQKCiAgIGBgYGJhc2gKICAgIyBVc2luZyB2aXJ0dWFsZW52CiAgIHZpcnR1YWxlbnYgZW52CiAgIHNvdXJjZSBlbnYvYmluL2FjdGl2YXRlCgogICAjIE9yIHVzaW5nIHZlbnYKICAgcHl0aG9uMyAtbSB2ZW52IGVudgogICBzb3VyY2UgZW52L2Jpbi9hY3RpdmF0ZQoKICAgIyBPciB1c2luZyBwb2V0cnkKICAgcG9ldHJ5IHNoZWxsCiAgIGBgYAoKMy4gSW5zdGFsbCByZXF1aXJlZCBwYWNrYWdlcwoKICAgYGBgYmFzaAogICBwb2V0cnkgaW5zdGFsbAogICBwbGF5d3JpZ2h0IGluc3RhbGwKICAgYGBgCgojIyDwn5qAIEhvdyB0byB1c2UKCkdlbmVyYXRlcyBhbiBBaXJieXRlIGxvdy1jb2RlIFlBTUwgY29ubmVjdG9yIHVzaW5nIHRoZSBBUEkgZG9jdW1lbnRhdGlvbiBwcm92aWRlZCB2aWEgVVJMcy4KCj4gU2V0IGBERUJVRz10cnVlYCBpbiBgLmVudmAgdG8gZW5hYmxlIGxvZ3MKCmBgYGJhc2gKc2t5ZmZlbCAtLWdvYWwgIjxNWSBFVEwgR09BTD4iIC0tdXJscyAiPFVSTCBET0MgMT4iIC0tdXJscyAiPFVSTCBET0MgMj4iCmBgYAoKSGVyZSB3ZSBnZW5lcmF0ZSBhIGNvbm5lY3RvciBmb3IgZXh0cmFjdGluZyBhbGwgYmxvZyBwb3N0cyBmcm9tIHRoZSBEZXBhcnRtZW50IG9mIEp1c3RpY2UuCgpgYGBiYXNoCnNreWZmZWwgXAogICAgLS1nb2FsICJleHRyYWN0IGFsbCBibG9nIGVudHJpZXMgZnJvbSBkZXBhcnRtZW50IG9mIGp1c3RpY2UiIFwKICAgIC0tdXJscyBodHRwczovL3d3dy5qdXN0aWNlLmdvdi9kZXZlbG9wZXIvYXBpLWRvY3VtZW50YXRpb24vYXBpX3YxCmBgYAoKIyMjIPCfk6UgSW1wb3J0IHRvIEFpcmJ5dGUKCkFmdGVyIGdlbmVyYXRpbmcgdGhlIGNvbm5lY3RvciwgeW91IG5lZWQgdG8gaW1wb3J0IGl0IHRvIEFpcmJ5dGUuIEV2ZW50dWFsbHkgdGhleSBtaWdodCBleHBvc2UgYW4gQVBJIHRvIGRvIHRoaXMgcHJvZ3JhbWF0aWNhbGx5IPCfpJ4gVW50aWwgdGhlbiwgaGVyZSdzIGhvdzoKCjEuIEdvIHRvIHlvdXIgQWlyYnl0ZSB3b3Jrc3BhY2UKMi4gQ2xpY2sgb24gQnVpbGRlciBbQkVUQV0gaW4gdGhlIG1lbnUKCiAgIDxpbWcgd2lkdGg9IjIwMHB4IiBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9za3lmZmVsL2FpcmJ5dGUtY29ubmVjdG9yLWdlbmVyYXRvci1wb2MvYXNzZXRzLzI1NjIyNDEyLzRiN2NlMTgyLTAzYjQtNDhkNy1hOTlhLWJlYWQyODdmZjI5NyIgLz4KCjMuIENsaWNrIOKAnE5ldyBjdXN0b20gY29ubmVjdG9y4oCdIGluIHRoZSB1cHBlciByaWdodCBjb3JuZXIKCiAgIDxpbWcgd2lkdGg9IjIwMHB4IiBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9za3lmZmVsL2FpcmJ5dGUtY29ubmVjdG9yLWdlbmVyYXRvci1wb2MvYXNzZXRzLzI1NjIyNDEyL2Y4OWQwNjYwLTFkYzEtNGYzNy1iNDZjLWMyMmE5NGU3Y2VlMCIgLz4KCjQuIENsaWNrIOKAnEltcG9ydCBhIFlBTUzigJ0gYW5kIHNlbGVjdCB0aGUgZ2VuZXJhdGVkIGBhaXJieXRlX2Nvbm5lY3Rvci55YW1sYAoKICAgPGltZyB3aWR0aD0iMjAwcHgiIHNyYz0iaHR0cHM6Ly9naXRodWIuY29tL3NreWZmZWwvYWlyYnl0ZS1jb25uZWN0b3ItZ2VuZXJhdG9yLXBvYy9hc3NldHMvMjU2MjI0MTIvZGMyMTAyNDAtYjIzZC00N2IwLWEwMjQtMjZlNzA4MzRiMjhhIiAvPgoKNS4gRmlsbCBpbiB0aGUgdGVzdCB2YWx1ZXMgYW5kIHJ1biB0aGUgdGVzdAo2LiBQcmVzcyDigJxQdWJsaXNoIHRvIHdvcmtzcGFjZeKAnQo= readmeEtag: '"cb5b34a23d3565e46223887f3bfb4f1e8756ad04"' readmeLastModified: Mon, 11 Mar 2024 15:30:53 GMT repositoryId: 761669694 description: >- proof of concept to generate Airbyte low-code YAML connectors from API documentation created: '2024-02-22T09:11:10Z' updated: '2025-10-19T23:12:03Z' language: Python archived: false stars: 27 watchers: 1 forks: 2 owner: skyffel logo: https://avatars.githubusercontent.com/u/160840259?v=4 license: NOASSERTION repoEtag: '"ae21bac233536dd5f55c7e3e02fa4b8bb7eaacaaf041d34c26a24c58e2e9c4e3"' repoLastModified: Sun, 19 Oct 2025 23:12:03 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/outline/openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNpZmljYXRpb24KClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBbT3BlbkFQSSBzcGVjaWZpY2F0aW9uc10oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKSBmb3IgT3V0bGluZSdzIEFQSS4KCiMjIyBEZXZlbG9wbWVudAoKSXQgaXMgcmVjb21tZW5kZWQgdG8gW2VkaXQgdGhlIHNwZWNpZmljYXRpb24gdXNpbmcgdGhlIFN3YWdnZXIgZWRpdG9yLF0oaHR0cDovL2VkaXRvci5zd2FnZ2VyLmlvLz91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL291dGxpbmUvb3BlbmFwaS9tYWluL3NwZWMzLnltbCkgd2hpY2ggd2lsbCB2YWxpZGF0ZSB0aGUgc3BlYyBhcyB5b3UgdHlwZS4gSWYgc3VnZ2VzdGluZyBjaGFuZ2VzIG9yIGZpeGVzIHRvIHRoaXMgcmVwb3NpdG9yeSBwbGVhc2UgbWFrZSBzdXJlIHRvIGluY2x1ZGUgYm90aCB0aGUganNvbiBhbmQgeW1sIHZlcnNpb25zIGluIHRoZSBQUi4gCgoKIyMjIERlcGxveW1lbnQKCkNoYW5nZXMgdG8gdGhpcyByZXBvc2l0b3J5IGFwcGVhciBhdCBbZ2V0b3V0bGluZS5jb20vZGV2ZWxvcGVyc10oaHR0cHM6Ly9nZXRvdXRsaW5lLmNvbS9kZXZlbG9wZXJzKSBhdXRvbWF0aWNhbGx5IHdoZW4gdGhlIFttYXJrZXRpbmcgd2Vic2l0ZV0oaHR0cHM6Ly9naXRodWIuY29tL291dGxpbmUvd2Vic2l0ZSkgaXMgZGVwbG95ZWQuCg== readmeEtag: '"aea3c806ec00242ad22a0e442f652ffb918d249c"' readmeLastModified: Sat, 06 Jul 2024 01:43:53 GMT repositoryId: 272337808 description: OpenAPI specification for the Outline API created: '2020-06-15T03:58:23Z' updated: '2026-02-01T14:54:17Z' language: JavaScript archived: false stars: 28 watchers: 3 forks: 11 owner: outline logo: https://avatars.githubusercontent.com/u/1765001?v=4 license: BSD-3-Clause repoEtag: '"f3fe2a62ab4a73465b9f8153a735e7c4fa39fc613ecf75c425f00717b248e69b"' repoLastModified: Sun, 01 Feb 2026 14:54:17 GMT foundInMaster: true category: - Converters - Parsers id: 859f9bdad31e9a5c13169e55d6f37e73 - source: openapi3 tags repository: https://github.com/yhnavein/swaggie v3: true id: 4dabb31099f545ae69a12182a15f3739 repositoryMetadata: base64Readme: >- <div style="text-align: center; margin: 0px auto 30px; max-width: 600px">
  <img src="./swaggie.svg" alt="Swaggie logo">
</div>

# Swaggie

![npm latest version](https://img.shields.io/npm/v/swaggie)
![NodeCI](https://github.com/yhnavein/swaggie/workflows/NodeCI/badge.svg)
![Test Coverage](https://img.shields.io/badge/test_coverage-98%25-brightgreen)
![npm downloads](https://img.shields.io/npm/dw/swaggie.svg)
![npm install size](https://packagephobia.now.sh/badge?p=swaggie)

Generate Typescript code from an OpenAPI 3.0 document, so that accessing REST API resources from the client code is less error-prone, static-typed and just easier to use long-term.

You can take a look at the [Examples section](#example) down below.

Project is based and inspired by [OpenApi Client](https://github.com/mikestead/openapi-client).

### Features

- Generate TypeScript code from OpenAPI 3.0 spec
- Supports `fetch`, `axios`, `xior`, `SWR + axios`, `Angular 1`, `Angular 2+` templates. It's flexible.
- Possible to create your own template that works with your existing codebase
- It's a dev tool that generates code, so it's not a runtime dependency
- Support for `allOf`, `oneOf`, `anyOf` and `$ref` in schemas
- Support for different types of enums
- Support for different content types
- JSDoc comments for generated code
- Small library size and very small and tree-shakable output that is all placed in one file
- OpenAPI 3.1 is partially supported (mostly enums, more to come)

## Install

In your project

    npm install swaggie --save-dev

Or globally to run CLI from anywhere

    npm install swaggie -g

## OpenAPI versions

Swaggie from version 1.0 supports OpenAPI 3.0 (and some features of 3.1). Swagger or OpenAPI 2.0 documents are not supported anymore, but you have few options how to deal with it:

- **(preferred)** From your backend server generate OpenAPI 3.0 spec instead of version 2 (samples are updated to use OpenAPI 3.0)
- Convert your OpenAPI 2.0 spec to 3.0 using [swagger2openapi](https://www.npmjs.com/package/swagger2openapi) tool (or something similar)
- If you can't do that for any reason, you can stick to `Swaggie v0.x`. But upgrade is suggested

Please note that OpenAPI 3.0 is a major spec upgrade and it's possible that there will be some breaking changes in the generated code.
I have tried my best to minimize the impact, but it was not possible to avoid it completely.

More info about breaking changes can be found in the [Releases](https://github.com/yhnavein/swaggie/releases).

### CLI

```
Usage: swaggie [options]

Options:

  -V, --version             output the version number
  -c, --config <path>       The path to the configuration JSON file. You can do all the set up there instead of parameters in the CLI
  -s, --src <url|path>      The url or path to the Open API spec file
  -o, --out <filePath>      The path to the file where the API would be generated. Use stdout if left empty
  -b, --baseUrl <string>    Base URL that will be used as a default value in the clients (default: "")
  -t, --template <string>   Template used for generating API client. Default: "axios"
  --preferAny               Use "any" type instead of "unknown" (default: false)
  --skipDeprecated          Skip deprecated operations. When enabled, deprecated operations will be skipped from the generated code (default: false)
  --servicePrefix <string>  Prefix for service names. Useful when you have multiple APIs and you want to avoid name collisions (default: "")
  --allowDots <bool>        Determines if dots should be used for serialization object properties
  --arrayFormat <format>    Determines how arrays should be serialized (choices: "indices", "repeat", "brackets")
  -h, --help                display help for command
```

Sample CLI usage using Swagger's Pet Store:

```bash
swaggie -s https://petstore3.swagger.io/api/v3/openapi.json -o ./client/petstore.ts
```

`swaggie` outputs TypeScript that is somehow formatted, but it's far from perfect. You can adjust the generated code by prettifying output using your preferred beautify tool using your repo's styling guidelines. For example involving `prettier` looks like this:

```bash
swaggie -s $URL -o ./client/petstore.ts && prettier ./client/petstore.ts --write`
```

And this can be easily automated (in the `npm` scripts for example)

### Configuration File

Instead of providing all required flags from the command line you can alternatively create a new JSON file where you can fill up all settings.

Sample configuration looks like this:

```json
{
  "$schema": "https://raw.githubusercontent.com/yhnavein/swaggie/master/schema.json",
  "out": "./src/client/petstore.ts",
  "src": "https://petstore3.swagger.io/api/v3/openapi.json",
  "template": "axios",
  "baseUrl": "/api",
  "preferAny": true,
  "servicePrefix": "",
  "dateFormat": "Date", // "string" | "Date"
  "queryParamsSerialization": {
    "arrayFormat": "repeat", // "repeat" | "brackets" | "indices"
    "allowDots": true
  }
}
```

### Templates

The following templates are bundled with Swaggie:

```
axios       Default template. Recommended for React / Vue / similar frameworks. Uses axios
xior        Lightweight and modern alternative to axios. Uses [xior](https://github.com/suhaotian/xior#intro)
swr-axios   SWR for GET requests with axios as backend
tsq-xior    TanStack Query for GET requests with xior as backend
fetch       Barebone fetch API. Recommended for React / Vue / similar frameworks
ng1         Angular 1 client (this is for the old one)
ng2         Angular 2+ client (uses HttpClient, InjectionTokens, etc)
```

If you want to use your own template, you can use the path to your template for the `-t` parameter:

```bash
swaggie -s https://petstore3.swagger.io/api/v3/openapi.json -o ./client/petstore --template ./my-swaggie-template/
```

## Usage – Integrating into your project

Let's assume that you have a [PetStore API](http://petstore.swagger.io/) as your REST API and you are developing a client app written in TypeScript that will consume this API.

Instead of writing any code by hand for fetching particular resources, we will let Swaggie do it for us.

### Query Parameters Serialization

When it comes to use of query parameters then you might need to adjust the way these parameters will be serialized, as backend server you are using expects them to be in a specific format. Thankfully in Swaggie you can specify how they should be handled. If you won't provide any configuration, then Swaggie will use the defaults values expected in the ASP.NET Core world.

For your convenience there are few config examples to achieve different serialization formats for an object `{ "a": { "b": 1 }, "c": [2, 3] }`:

| Expected Format         | allowDots | arrayFormat |
| ----------------------- | --------- | ----------- |
| `?a.b=1&c=2&c=3`        | `true`    | `repeat`    |
| `?a.b=1&c[]=2&c[]=3`    | `true`    | `brackets`  |
| `?a.b=1&c[0]=2&c[1]=3`  | `true`    | `indices`   |
| `?a[b]=1&c=2&c=3`       | `false`   | `repeat`    |
| `?a[b]=1&c[]=2&c[]=3`   | `false`   | `brackets`  |
| `?a[b]=1&c[0]=2&c[1]=3` | `false`   | `indices`   |

Once you know what your backend expects, you can adjust the configuration file accordingly: (below are default values)

```json
{
  "queryParamsSerialization": {
    "arrayFormat": "repeat",
    "allowDots": true
  }
}
```

### Code Quality

> Please note that it's **recommended** to pipe Swaggie command to some prettifier like `prettier`, `biome` or `dprint` to make the generated code look not only nice, but also persistent.
> Because Swaggie relies on a templating engine, whitespaces are generally a mess, so they may change between versions.

**Suggested prettiers**

[prettier](https://prettier.io/) - the most popular one

```bash
prettier ./FILE_PATH.ts --write
```

[biome](https://biomejs.dev) - the super fast one

```bash
biome check ./FILE_PATH.ts --apply-unsafe
```

You are not limited to any of these, but in our examples we will use Prettier. Please remember that these tools needs to be installed first and they need a config file in your project.

### Example

Let's run `swaggie` against PetStore API and see what will happen:

```bash
swaggie -s https://petstore3.swagger.io/api/v3/openapi.json -o ./api/petstore.ts && prettier ./api/petstore.ts --write
```

```typescript
// ./api/petstore.ts

import Axios, { AxiosPromise } from 'axios';
const axios = Axios.create({
  baseURL: '/api',
  paramsSerializer: (params) =>
    encodeParams(params, null, {
      allowDots: true,
      arrayFormat: 'repeat',
    }),
});

/** [...] **/

export const petClient = {
  /**
   * @param petId
   */
  getPetById(petId: number): AxiosPromise<Pet> {
    let url = `/pet/${encodeURIComponent(`${petId}`)}`;

    return axios.request<Pet>({
      url: url,
      method: 'GET',
    });
  },

  // ... and other methods ...
};
```

When we have that we can write some domain code and use this auto-generated classes:

```typescript
// app.ts

import { petClient } from './api/petClient';

petClient.getPetById(123).then((pet) => console.log('Pet: ', pet));
```

If Petstore owners decide to remove method we use, then after running `swaggie` again it will no longer be present in the `petClient` class. This will result in the build error, which is very much appreciated at this stage.

Without this approach, the error would be spotted by our end-user and he/she would not appreciate it at all!

## Other features

### Parameter adjustment

In some cases you might want to adjust the parameters globally but without touching the OpenAPI spec. For example, the spec may explicitly define some of parameters as `required`, but you will handle them in the interceptor. In such case, you don't want to define them is every single method. This is where this feature comes in handy.

Example (in the config file):

```json
{
  "modifiers": {
    "parameters": {
      "clientId": "ignore",
      "orgId": "optional",
      "country": "required"
    }
  }
}
```

This will ensure that `clientId` parameters will never appear in any of the generated methods, and `orgId` will be optional (regardless of what the spec says).
You can also use `required` value to enforce the parameter to be required, _(but in this case, realistically it would be better to just fix the spec)_.

## Server config

You might wonder how to set up server to fully utilize Swaggie's features. For that I've added a `samples/` folder with sample configurations.

[ASP.NET Core + Nswag](./samples/dotnetcore/nswag/README.md)

[ASP.NET Core + Swashbuckle](./samples/dotnetcore/swashbuckle/README.md)

Server is not necessary to use Swaggie. Swaggie cares only about the JSON/yaml file with the Open API spec, but for your development purpose you might want to have a server that can serve this file automatically from the actual endpoints.

## Using Swaggie programmatically

```javascript
const swaggie = require('swaggie');
swaggie
  .genCode({
    src: 'https://petstore3.swagger.io/api/v3/openapi.json',
    out: './api/petstore.ts',
  })
  .then(complete, error);

function complete(spec) {
  console.info('Service generation complete');
}

function error(e) {
  console.error(e.toString());
}
```

## Notes

| Supported                                                                      | Not supported                                   |
| ------------------------------------------------------------------------------ | ----------------------------------------------- |
| OpenAPI 3                                                                      | Swagger, Open API 2.0                           |
| `allOf`, `oneOf`, `anyOf`, `$ref` to schemas                                   | `not`                                           |
| Spec formats: `JSON`, `YAML`                                                   | Very complex query params                       |
| Extensions: `x-position`, `x-name`, `x-enumNames`, `x-enum-varnames`           | Multiple response types (only one will be used) |
| Content types: `JSON`, `text`, `multipart/form-data`                           | Multiple request types (only one will be used)  |
| Content types: `application/x-www-form-urlencoded`, `application/octet-stream` | References to other spec files                  |
| Different types of enum definitions (+ OpenAPI 3.1 support for enums)          | OpenAPI callbacks                               |
| Paths inheritance, comments (descriptions)                                     | OpenAPI webhooks                                |
| Getting documents from remote locations or as path reference (local file)      |                                                 |
| Grouping endpoints by tags + handle gracefully duplicate operation ids         |                                                 |

## Used by

<div style="display: flex; gap: 1rem;">
<a href="https://www.britishcouncil.org"><img alt="British Council" src="https://upload.wikimedia.org/wikipedia/commons/thumb/e/e1/BritishCouncil.png/320px-BritishCouncil.png" style="height: 50px;" /></a>
<a href="https://kpmg.com/"><img alt="KPMG" src="https://upload.wikimedia.org/wikipedia/commons/thumb/3/31/KPMG.svg/320px-KPMG.svg.png" style="height: 50px;" /></a>
<a href="https://klarna.com/"><img alt="Klarna" src="https://upload.wikimedia.org/wikipedia/commons/4/40/Klarna_Payment_Badge.svg" style="height: 50px;" /></a>
</div>
 readmeEtag: '"f93660b15104019b13a8c6c53d17639900d9ed0c"' readmeLastModified: Mon, 18 Aug 2025 22:21:54 GMT repositoryId: 120893123 description: Tool for generating TypeScript client code for given Swagger API endpoints created: '2018-02-09T10:37:44Z' updated: '2026-01-04T21:44:46Z' language: TypeScript archived: false stars: 24 watchers: 2 forks: 9 owner: yhnavein logo: https://avatars.githubusercontent.com/u/516709?v=4 license: MIT repoEtag: '"921d9be24f4405c0b5819aaa7c9d13f31e7c060f8de1e62c40be1f94e7af216e"' repoLastModified: Sun, 04 Jan 2026 21:44:46 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mnf-group/openapimux v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJTXV4Ck9wZW5BUElNdXggaXMgYSAic2NoZW1hLWZpcnN0IiBIVFRQIHJvdXRlci4gSXQgdGFrZXMgb25lIG9yIG11bHRpcGxlCltPcGVuQVBJICAoU3dhZ2dlcildKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykgc2NoZW1hIGZpbGVzIGFzIGFuIGlucHV0CmFuZCB0aGVuIG1hdGNoZXMsIHZhbGlkYXRlcyBhbmQgaGFuZGxlcyBhbGwgaW5jb21pbmcgSFRUUCBiYXNlZCBvbiB0aGVzZSBzY2hlbWFzLgpVbmRlciB0aGUgaG9vZCwgaXQgdXNlcyBba2luLW9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9nZXRraW4va2luLW9wZW5hcGkvKSBmb3IKT3BlbkFQSSBzY2hlbWEgcGFyc2luZyBhbmQgdmFsaWRhdGlvbi4KCiMgTW90aXZhdGlvbgpPcGVuQVBJIG9mZmVycyBhIGdyZWF0IHdheSBvZiBkb2N1bWVudGluZyBBUEkuIEhvd2V2ZXIsIG5vbmUgb2YKW2V4aXN0aW5nIGdvIHJvdXRlcnNdKGh0dHBzOi8vZ2l0aHViLmNvbS9hdmVsaW5vL2F3ZXNvbWUtZ28jcm91dGVycykgb2ZmZXJzCiJzY2hlbWEgZmlyc3QiIGFwcHJvYWNoLiBJbiBlYWNoIHJvdXRlciwgeW91IG5lZWQgdG8gaW5pdGlhbGl6ZSBhIGxpc3Qgb2YgYXZhaWxhYmxlCnJvdXRlcyBhbmQgdGhlbiBkbyB0aGUgcmVxdWVzdCB2YWxpZGF0aW9uIG1hbnVhbGx5LiBPcGVuQVBJTXV4IGZpbGxzIHRoaXMgZ2FwIGJ5CmFsbG93aW5nIHRvIGluaXRpYWxpemUgYSByb3V0ZXIgZGlyZWN0bHkgZnJvbSB0aGUgT3BlbkFQSSBzY2hlbWEgZGVmaW5pdGlvbiBmaWxlLgoKIyBGZWF0dXJlcwoqIFdvcmtzIHdpdGggYm90aCBPcGVuQVBJIDMuMCBhbmQgT3BlbkFQSSAyLjAgKGFrYSBTd2FnZ2VyKS4gQXMgd2VsbCBhcyBib3RoIGpzb24gYW5kIHlhbWwgc2NoZW1hcwoqIE11bHRpcGxlIE9wZW5BUEkgc2NoZW1hIGZpbGVzIGNhbiBiZSB1c2VkIGF0IHRoZSBzYW1lIHJvdXRlciB0byBzdXBwb3J0IEFQSSB2ZXJzaW9uaW5nCiogSW1wbGVtZW50cyBgaHR0cC5IYW5kbGVyYCBpbnRlcmZhY2UsIHNvIGl0IGlzIGNvbXBhdGlibGUgd2l0aCB0aGUgc3RhbmRhcmQgaHR0cC5TZXJ2ZU11eAoqIFN1cHBvcnRzIGdsb2JhbCBsZXZlbCBgaHR0cC5IYW5kbGVyYCBtaWRkbGV3YXJlcywgc28gaXQgaXMgY29tcGF0aWJsZSB3aXRoIHRoaXJkLXBhcnR5IG1pZGRsZXdhcmVzCiogU3VwcG9ydHMgY3VzdG9tIGVycm9yIGhhbmRsZXIgZm9yIG1vcmUgY29udHJvbAoKIyBSb3V0aW5nCiogYG9wZXJhdGlvbklkYCBhdHRyaWJ1dGUgb2YgYW4gT3BlbkFQSSBwYXRoIGlzIHVzZWQgdG8gcmVzb2x2ZSBpdCB0byBhbiBhcHByb3ByaWF0ZSBoYW5kbGVyCiogT3BlbkFQSU11eCBjYW4gZW5jYXBzdWxhdGUgb25lIG9yIG1vcmUgc3dhZ2dlciByb3V0ZXJzLiBFYWNoIHJvdXRlciBjb3VsZApiZSBjcmVhdGVkIGZyb20gYW4gT3BlbkFQSSBzY2hlbWEgZmlsZSBvciBkaXJlY3RseSBhcyBhIHN3YWdnZXIgb2JqZWN0CiogVG8gaGFuZGxlIG11bHRpcGxlIHZlcnNpb25zLCB1c2UgdGhlIGBzZXJ2ZXJzLnVybGAgYXR0cmlidXRlIGluIE9wZW5BUEkgc2NoZW1hLiBFZwogYGBgeWFtbAogc2VydmVyczoKICAtIHVybDogIi92MS4yIgpgYGAKKiBXaGVuIGZpbmRpbmcgYSBtYXRjaGluZyByb3V0ZSwgcm91dGVycyB3aXRoIGBzZXJ2ZXJzYCBhdHRyaWJ1dGUgc2V0IHRha2UgcHJpb3JpdHkKCiMgSW5zdGFsbApgZ28gZ2V0IC11IGdpdGh1Yi5jb20vTU5GR3JvdXAvb3BlbmFwaW11eGAKCiMgRnVsbCBFeGFtcGxlCkFzc3VtaW5nIGBvcGVuYXBpLnlhbWxgIGhhcyB0aGUgZm9sbG93aW5nIHNjaGVtYQpgYGB5YW1sCm9wZW5hcGk6IDMuMC4wCgpwYXRoczoKICAvZm9vOgogICAgZ2V0OgogICAgICBvcGVyYXRpb25JZDogZ2V0Rm9vCmBgYAoKSXQgd2lsbCBjcmVhdGUgYW5kIHN0YXJ0IGEgc2VydmVyIG9uIDgwODAKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgKAoJImZtdCIKCSJsb2ciCgkibmV0L2h0dHAiCgoJImdpdGh1Yi5jb20vZ28tY2hpL2NoaS9taWRkbGV3YXJlIgopCgp0eXBlIGZvb0hhbmRsZXIgc3RydWN0e30KCmZ1bmMgKGYgZm9vSGFuZGxlcikgU2VydmVIVFRQKHcgaHR0cC5SZXNwb25zZVdyaXRlciwgciAqaHR0cC5SZXF1ZXN0KSB7CglmbXQuRnByaW50KHcsICJIZWxsbyIpCn0KCmZ1bmMgbWFpbigpIHsKCXIsIGVyciA6PSBOZXdSb3V0ZXIoIi4vb3BlbmFwaS55YW1sIikKCWlmIGVyciAhPSBuaWwgewoJCXBhbmljKGVycikKCX0KCglyLlVzZUhhbmRsZXJzKG1hcFtzdHJpbmddaHR0cC5IYW5kbGVyewoJCSJnZXRGb28iOiBmb29IYW5kbGVye30sCgl9KQoKCXIuVXNlTWlkZGxld2FyZSgKCQltaWRkbGV3YXJlLlJlY292ZXJlciwKCQltaWRkbGV3YXJlLlJlcXVlc3RJRCwKCQltaWRkbGV3YXJlLkRlZmF1bHRDb21wcmVzcywKCSkKCglyLkVycm9ySGFuZGxlciA9IGZ1bmModyBodHRwLlJlc3BvbnNlV3JpdGVyLCByICpodHRwLlJlcXVlc3QsIGRhdGEgc3RyaW5nLCBjb2RlIGludCkgewoJCXcuV3JpdGVIZWFkZXIoY29kZSkKCQlpZiBjb2RlID09IGh0dHAuU3RhdHVzSW50ZXJuYWxTZXJ2ZXJFcnJvciB7CgkJCWZtdC5QcmludGxuKCJGYXRhbDoiLCBkYXRhKQoJCQl3LldyaXRlKFtdYnl0ZSgiT29wcyIpKQoJCX0gZWxzZSB7CgkJCXcuV3JpdGUoW11ieXRlKGRhdGEpKQoJCX0KCX0KCglsb2cuRmF0YWwoaHR0cC5MaXN0ZW5BbmRTZXJ2ZSgiOjgwODAiLCByKSkKfQpgYGAK readmeEtag: '"b2584cc991f2bbdf819484e0b0bb0dc68fe11eb3"' readmeLastModified: Mon, 11 Oct 2021 22:26:40 GMT repositoryId: 193996898 description: Open API router in go created: '2019-06-27T00:45:48Z' updated: '2023-08-14T14:37:21Z' language: Go archived: false stars: 22 watchers: 4 forks: 3 owner: mnf-group logo: https://avatars.githubusercontent.com/u/40686352?v=4 license: MIT repoEtag: '"ae2ba003e9a06dab47032d5b7fcad6177bb18eb04c1c7e059975169876901bd7"' repoLastModified: Mon, 14 Aug 2023 14:37:21 GMT foundInMaster: true category: - Data Validators - Parsers id: c89db4b421338959f10cb6d1be0f8c7b - source: openapi3 tags repository: https://github.com/datazoode/flapi v3: true id: 1cf545e15d0ee2560d0090440fcc77b2 repositoryMetadata: base64Readme: >- # flAPI: Instant SQL based APIs

flAPI is a powerful service that automatically generates read-only APIs for datasets by utilizing SQL templates. Built on top of [DuckDB](https://duckdb.org/) and leveraging its SQL engine and extension ecosystem, flAPI offers a seamless way to connect to various data sources and expose them as RESTful APIs.

![overview of flAPI](https://i.imgur.com/m7UVZlR.png)

## ⚡ Features

- **Automatic API Generation**: Create APIs for your datasets without coding
- **MCP (Model Context Protocol) Support**: Declarative creation of AI tools alongside REST endpoints
- **Multiple Data Sources**: Connect to [BigQuery](https://github.com/hafenkran/duckdb-bigquery), SAP ERP & BW (via [ERPL](https://github.com/datazoode/erpl)), Parquet, [Iceberg](https://github.com/duckdb/duckdb_iceberg), [Postgres](https://github.com/duckdb/postgres_scanner), [MySQL](https://github.com/duckdb/duckdb_mysql), and more
- **SQL Templates**: Use Mustache-like syntax for dynamic queries
- **Caching**: DuckLake-backed cache with full refresh and incremental sync
- **Security**: Implement row-level and column-level security with ease
- **Easy deployment**: Deploy flAPI with a single binary file

## 🛠 Quick Start
The easiest way to get started with flAPI is to use the pre-built docker image.

#### 1. Pull the docker image from the Github Container Registry:

```bash
> docker pull ghcr.io/datazoode/flapi:latest
```


The image is pretty small and mainly contains the flAPI binary which is statically linked against [DuckDB v1.4.3](https://github.com/duckdb/duckdb/releases/tag/v1.4.3). Details about the docker image can be found in the [Dockerfile](./docker/development/Dockerfile).

#### 2. Run flAPI:
Once you have downloaded the binary, you can run flAPI by executing the following command:

```
> docker run -it --rm -p 8080:8080 -p 8081:8081 -v $(pwd)/examples/:/config ghcr.io/datazoode/flapi -c /config/flapi.yaml
```

The different arguments in this docker command are:
- `-it --rm`: Run the container in interactive mode and remove it after the process has finished
- `-p 8080:8080`: Exposes port 8080 of the container to the host, this makes the REST API available at `http://localhost:8080`
- `-p 8081:8081`: Exposes port 8081 for the MCP server (when enabled)
- `-v $(pwd)/examples/:/config`: This mounts the local `examples` directory to the `/config` directory in the container, this is where the flAPI configuration file
is expected to be found.
- `ghcr.io/datazoode/flapi`: The docker image to use
- `-c /config/flapi.yaml`: This is an argument to the flAPI application which tells it to use the `flapi.yaml` file in the `/config` directory as the configuration file.

#### 2.1 Enable MCP Support:
To enable MCP support, you can either:

**Option A: Use the command line flag**
```
> docker run -it --rm -p 8080:8080 -p 8081:8081 -v $(pwd)/examples/:/config ghcr.io/datazoode/flapi -c /config/flapi.yaml --enable-mcp
```

**Option B: Configure in flapi.yaml**
```yaml
mcp:
  enabled: true
  port: 8081
  # ... other MCP configuration
```

#### 3.1 Test the API server:
If everything is set up correctly, you should be able to access the API at the URL specified in the configuration file.

```bash
> curl 'http://localhost:8080/'


         ___
     ___( o)>   Welcome to
     \ <_. )    flAPI
      `---'    

    Fast and Flexible API Framework
    powered by DuckDB
```

#### 3.2 Get an overview of the available endpoints:
The flAPI server creates embedded Swagger UI at which provides an overview of the available endpoints and allows you to test them. It can be found at

[`> http://localhost:8080/doc`](http://localhost:8080/doc)

You should see the familiar Swagger UI page:

![flAPI Swagger UI](https://i.imgur.com/HqjHMlA.png)

The raw yaml [Swagger 2.0](https://swagger.io/specification/) is also available at [`http://localhost:8080/doc.yaml`](http://localhost:8080/doc.yaml)

#### 3.3 Test the MCP server:
If MCP is enabled, you can test the MCP server as well:

```bash
# Check MCP server health
> curl 'http://localhost:8081/mcp/health'

{"status":"healthy","server":"flapi-mcp-server","version":"0.3.0","protocol_version":"2024-11-05","tools_count":0}

# Initialize MCP connection
> curl -X POST http://localhost:8081/mcp/jsonrpc \
  -H "Content-Type: application/json" \
  -d '{"jsonrpc": "2.0", "id": 1, "method": "initialize"}'

# List available tools
> curl -X POST http://localhost:8081/mcp/jsonrpc \
  -H "Content-Type: application/json" \
  -d '{"jsonrpc": "2.0", "id": 2, "method": "tools/list"}'
```

## 🤖 MCP (Model Context Protocol) Support

flAPI now supports the **Model Context Protocol (MCP)** in a **unified configuration approach**. Every flAPI instance automatically runs both a REST API server and an MCP server concurrently, allowing you to create AI tools alongside your REST endpoints using the same configuration files and SQL templates.

### Key Features

- **Unified Configuration**: Single YAML files can define REST endpoints, MCP tools, and MCP resources
- **Automatic Detection**: Configuration type is determined by presence of `url-path` (REST), `mcp-tool` (MCP tool), or `mcp-resource` (MCP resource)
- **Shared Components**: MCP tools and resources use the same SQL templates, parameter validation, authentication, and caching as REST endpoints
- **Concurrent Servers**: REST API (port 8080) and MCP server (port 8081) run simultaneously
- **Declarative Definition**: Define everything using YAML configuration with SQL templatestocol
- **Tool Discovery**: Automatic tool discovery and schema generation
- **Security Integration**: Reuse existing authentication, rate limiting, and caching features

### MCP Endpoints

- `POST /mcp/jsonrpc` - Main JSON-RPC endpoint for tool calls
- `GET /mcp/health` - Health check endpoint

### Unified Configuration

**MCP is now automatically enabled** - no separate configuration needed! Every flAPI instance runs both REST API and MCP servers concurrently.

Configuration files can define multiple entity types:

#### REST Endpoint + MCP Tool (Unified)

```yaml
# Single configuration file serves as BOTH REST endpoint AND MCP tool
url-path: /customers/                    # Makes this a REST endpoint
mcp-tool:                                # Also makes this an MCP tool
  name: get_customers
  description: Retrieve customer information by ID
  result-mime-type: application/json

request:
  - field-name: id
    field-in: query
    description: Customer ID
    required: false
    validators:
      - type: int
        min: 1
        max: 1000000
        preventSqlInjection: true

template-source: customers.sql
connection: [customers-parquet]

rate-limit:
  enabled: true
  max: 100
  interval: 60

auth:
  enabled: true
  type: basic
  users:
    - username: admin
      password: secret
      roles: [admin]
```

#### MCP Resource Only

```yaml
# MCP Resource example
mcp-resource:
  name: customer_schema
  description: Customer database schema definition
  mime-type: application/json

template-source: customer-schema.sql
connection: [customers-parquet]
```

### Using MCP Tools

Once MCP is enabled, you can interact with tools using JSON-RPC 2.0:

```bash
# Check MCP server health
curl 'http://localhost:8081/mcp/health'

# Initialize MCP connection
curl -X POST http://localhost:8081/mcp/jsonrpc \
  -H "Content-Type: application/json" \
  -d '{"jsonrpc": "2.0", "id": 1, "method": "initialize"}'

# List available tools (discovered from unified configuration)
curl -X POST http://localhost:8081/mcp/jsonrpc \
  -H "Content-Type: application/json" \
  -d '{"jsonrpc": "2.0", "id": 2, "method": "tools/list"}'

# Call a tool (same SQL template used for both REST and MCP)
curl -X POST http://localhost:8081/mcp/jsonrpc \
  -H "Content-Type: application/json" \
  -d '{"jsonrpc": "2.0", "id": 3, "method": "tools/call", "params": {"name": "get_customers", "arguments": {"id": "123"}}}'
```

## 🎓 Example

Here's a simple example of how to create an API endpoint using flAPI:

### 1. Create a basic flAPI configuration
flAPI uses the popular [YAML](https://yaml.org/) format to configure the API endpoints. A basic configuration file looks like this:

```yaml
project_name: example-flapi-project
project_description: An example flAPI project demonstrating various configuration options
template:
  path: './sqls'            # The path where SQL templates and API endpoint configurations are stored
  environment-whitelist:    # Optional: List of regular expressions for whitelisting envvars which are available in the templates
    - '^FLAPI_.*'

duckdb:                     # Configuration of the DuckDB embedded into flAPI
  db_path: ./flapi_cache.db # Optional: remove or comment out for in-memory database, we use this store also as cache
  access_mode: READ_WRITE   # See the https://duckdb.org/docs/configuration/overview) for more details
  threads: 8
  max_memory: 8GB
  default_order: DESC

connections:                # A YAML map of database connection configurations, a API endpoint needs to reference one of these connections
   bigquery-lakehouse: 
                            # SQL commands to initialize the connection (e.g., e.g. installing, loading and configuring the BQ a DuckDB extension)
      init: |
         INSTALL 'bigquery' FROM 'http://storage.googleapis.com/hafenkran';
         LOAD 'bigquery';
      properties:           # A YAML map of connection-specific properties (accessible in templates via {{ context.conn.property_name }})
         project_id: 'my-project-id'

   customers-parquet: 
      properties:
         path: './data/customers.parquet'

heartbeat:
  enabled: true            # The eartbeat worker is a background thread which can can be used to periodically trigger endpionts
  worker-interval: 10      # The interval in seconds at which the heartbeat worker will trigger endpoints

enforce-https:
  enabled: false           # Whether to force HTTPS for the API connections, we strongly recommend to use a reverse proxy to do SSL termination
  # ssl-cert-file: './ssl/cert.pem'
  # ssl-key-file: './ssl/key.pem'

```

After that ensure that the template path (`./sqls` in this example) exists.

### 1. Define your API endpoint (`./sqls/customers.yaml`):
Each endpoint is at least defined by a YAML file and a corresponding SQL template in the template path. For our example we will create the file `./sqls/customers.yaml`:

```yaml
url-path: /customers/      # The URL path at which the endpoint will be available

request:                  # The request configuration for the endpoint, this defines the parameters that can be used in the query
  - field-name: id
    field-in: query       # The location of the parameter, other options are 'path', 'query' and 'body'
    description: Customer ID # A description of the parameter, this is used in the auto-generated API documentation
    required: false       # Whether the parameter is required
    validators:           # A list of validators that will be applied to the parameter
      - type: int
        min: 1
        max: 1000000
        preventSqlInjection: true

template-source: customers.sql # The path to the SQL template that will be used to generate the endpoint
connection: 
  - customers-parquet          # The connection that will be used to execute the query

rate-limit:
  enabled: true           # Whether rate limiting is enabled for the endpoint
  max: 100                # The maximum number of requests per interval
  interval: 60            # The interval in seconds
  
auth:
  enabled: true           # Whether authentication is enabled for the endpoint
  type: basic             # The type of authentication, other options are 'basic' and 'bearer'
  users:                  # The users that are allowed to access the endpoint
    - username: admin
      password: secret
      roles: [admin]
    - username: user
      password: password
      roles: [read]

heartbeat:
  enabled: true           # Whether the heartbeat worker if enabled will trigger the endpoint periodically
  params:                 # A YAML map of parameters that will be passed by the heartbeat worker to the endpoint
    id: 123

```
There are many more configuration options available, see the [full documentation](link-to-your-docs) for more details.

### 2. Configure the endpoints SQL template (`./sqls/customers.sql`):
After the creation of the YAML endpoint configuration we need to connect the SQL template which connects the enpoint to the data connection.
The template files use the [Mustache templating language](https://mustache.github.io/) to dynamically generate the SQL query. 

```sql
SELECT * FROM '{{{conn.path}}}'
WHERE 1=1
{{#params.id}}
  AND c_custkey = {{{ params.id }}}
{{/params.id}}
```

The above template uses the `path` parameter defined in the connection configuration to directly query a local parquet file. If the id parameter is 
provided, it will be used to filter the results.

### 3. Send a request:
To test the endpoint and see if everything worked, we can use curl. We should also provide the correct basic auth credentials (`admin:secret` in this case). To make the JSON result easier to read, we pipe the output to [`jq`](https://jqlang.github.io/jq/).

```bash
> curl -X GET -u admin:secret "http://localhost:8080/customers?id=123" | jq .

{
  "next": "",
  "total_count": 1,
  "data": [
    {
      "c_mktsegment": "BUILDING",
      "c_acctbal": 5897.82999999999992724,
      "c_phone": "15-817-151-1168",
      "c_address": "YsOnaaER8MkvK5cpf4VSlq",
      "c_nationkey": 5,
      "c_name": "Customer#000000123",
      "c_comment": "ependencies. regular, ironic requests are fluffily regu",
      "c_custkey": 123
    }
  ]
}

```

## ⁉️ DuckLake-backed caching (current implementation)
flAPI uses the DuckDB DuckLake extension to provide modern, snapshot-based caching. You write the SQL to define the cached table, and flAPI manages schemas, snapshots, retention, scheduling, and audit logs.

### Quick start: Full refresh cache
1) Configure DuckLake globally (alias is `cache` by default):
```yaml
ducklake:
  enabled: true
  alias: cache
  metadata-path: ./examples/data/cache.ducklake
  data-path: ./examples/data/cache.ducklake
  data-inlining-row-limit: 10  # Enable data inlining for small changes (optional)
  retention:
    max-snapshot-age: 14d
  compaction:
    enabled: false
  scheduler:
    enabled: true
```

2) Add cache block to your endpoint (no `primary-key`/`cursor` → full refresh):
```yaml
url-path: /publicis
template-source: publicis.sql
connection: [bigquery-lakehouse]

cache:
  enabled: true
  table: publicis_cache
  schema: analytics
  schedule: 5m
  retention:
    max_snapshot_age: 14d
  template_file: publicis/publicis_cache.sql
```

3) Write the cache SQL template (CTAS):
```sql
-- publicis/publicis_cache.sql
CREATE OR REPLACE TABLE {{cache.catalog}}.{{cache.schema}}.{{cache.table}} AS
SELECT
  p.country,
  p.product_category,
  p.campaign_type,
  p.channel,
  sum(p.clicks) AS clicks
FROM bigquery_scan('{{{conn.project_id}}}.landing__publicis.kaercher_union_all') AS p
GROUP BY 1, 2, 3, 4;
```

4) Query from the cache in your main SQL:
```sql
-- publicis.sql
SELECT
  p.country,
  p.product_category,
  p.campaign_type,
  p.channel,
  p.clicks
FROM {{cache.catalog}}.{{cache.schema}}.{{cache.table}} AS p
WHERE 1=1
```

Notes:
- The cache schema (`cache.analytics`) is created automatically if missing.
- Regular GET requests never refresh the cache. Refreshes happen on warmup, on schedule, or via the manual API.
- **Data Inlining**: When `data-inlining-row-limit` is configured, small cache changes (≤ specified row limit) are written directly to DuckLake metadata instead of creating separate Parquet files. This improves performance for small incremental updates.

#### Data inlining (optional, for small changes)

DuckLake supports writing very small inserts directly into the metadata catalog instead of creating a Parquet file for every micro-batch. This is called "Data Inlining" and can significantly speed up small, frequent updates.

- **Enable globally**: configure once under the top-level `ducklake` block:

  ```yaml
  ducklake:
    enabled: true
    alias: cache
    metadata_path: ./examples/data/cache.ducklake
    data_path: ./examples/data/cache.ducklake
    data_inlining_row_limit: 10  # inline inserts up to 10 rows
  ```

- **Behavior**:
  - Inserts with rows ≤ `data-inlining-row-limit` are inlined into the catalog metadata.
  - Larger inserts automatically fall back to normal Parquet file writes.
  - Inlining applies to all caches (global setting), no per-endpoint toggle.

- **Manual flush (optional)**: you can flush inlined data to Parquet files at any time using DuckLake’s function. Assuming your DuckLake alias is `cache`:

  ```sql
  -- Flush all inlined data in the catalog
  CALL ducklake_flush_inlined_data('cache');

  -- Flush only a specific schema
  CALL ducklake_flush_inlined_data('cache', schema_name => 'analytics');

  -- Flush only a specific table (default schema "main")
  CALL ducklake_flush_inlined_data('cache', table_name => 'events_cache');

  -- Flush a specific table in a specific schema
  CALL ducklake_flush_inlined_data('cache', schema_name => 'analytics', table_name => 'events_cache');
  ```

- **Notes**:
  - This feature is provided by DuckLake and is currently marked experimental upstream. See the DuckLake docs for details: [Data Inlining](https://ducklake.select/docs/stable/duckdb/advanced_features/data_inlining.html).
  - If you don’t set `data_inlining_row_limit`, flAPI won’t enable inlining and DuckLake will use regular Parquet writes.

### Advanced: Incremental append and merge
The engine infers sync mode from your YAML:
- No `primary-key`, no `cursor` → full refresh (CTAS)
- With `cursor` only → incremental append
- With `primary-key` + `cursor` → incremental merge (upsert)

Example YAMLs:
```yaml
# Incremental append
cache:
  enabled: true
  table: events_cache
  schema: analytics
  schedule: 10m
  cursor:
    column: created_at
    type: timestamp
  template-file: events/events_cache.sql

# Incremental merge (upsert)
cache:
  enabled: true
  table: customers_cache
  schema: analytics
  schedule: 15m
  primary-key: [id]
  cursor:
    column: updated_at
    type: timestamp
  template_file: customers/customers_cache.sql
```

Cache template variables available to your SQL:
- `{{cache.catalog}}`, `{{cache.schema}}`, `{{cache.table}}`, `{{cache.schedule}}`
- `{{cache.snapshotId}}`, `{{cache.snapshotTimestamp}}` (current)
- `{{cache.previousSnapshotId}}`, `{{cache.previousSnapshotTimestamp}}` (previous)
- `{{cache.cursorColumn}}`, `{{cache.cursorType}}`
- `{{cache.primaryKeys}}`
- `{{params.cacheMode}}` is available with values `full`, `append`, or `merge`

Incremental append example:
```sql
-- events/events_cache.sql
INSERT INTO {{cache.catalog}}.{{cache.schema}}.{{cache.table}}
SELECT *
FROM source_events
WHERE {{#cache.previousSnapshotTimestamp}} event_time > TIMESTAMP '{{cache.previousSnapshotTimestamp}}' {{/cache.previousSnapshotTimestamp}}
```

Incremental merge example:
```sql
-- customers/customers_cache.sql
MERGE INTO {{cache.catalog}}.{{cache.schema}}.{{cache.table}} AS t
USING (
  SELECT * FROM source_customers
  WHERE {{#cache.previousSnapshotTimestamp}} updated_at > TIMESTAMP '{{cache.previousSnapshotTimestamp}}' {{/cache.previousSnapshotTimestamp}}
) AS s
ON t.id = s.id
WHEN MATCHED THEN UPDATE SET
  name = s.name,
  email = s.email,
  updated_at = s.updated_at
WHEN NOT MATCHED THEN INSERT (*) VALUES (s.*);
```

### When does the cache refresh?
- Startup warmup: flAPI refreshes caches for endpoints with cache enabled.
- Scheduled refresh: controlled by `cache.schedule` on each endpoint (e.g., `5m`).
- Manual refresh: call the refresh API (see below).
- Regular GET requests do not refresh the cache.

### Audit, retention, compaction, and control APIs
flAPI maintains an audit table inside DuckLake at `cache.audit.sync_events` and provides control endpoints:

- Manual refresh:
```bash
curl -X POST "http://localhost:8080/api/v1/_config/endpoints/publicis/cache/refresh"
```

- Audit logs (endpoint-specific and global):
```bash
curl "http://localhost:8080/api/v1/_config/endpoints/publicis/cache/audit"
curl "http://localhost:8080/api/v1/_config/cache/audit"
```

- Garbage collection (retention):
Retention can be configured per endpoint under `cache.retention`:
```yaml
cache:
  retention:
    max-snapshot-age: 7d     # time-based retention
    # keep-last-snapshots: 3 # version-based retention (subject to DuckLake support)
```
The system applies retention after each refresh and you can also trigger GC manually:
```bash
curl -X POST "http://localhost:8080/api/v1/_config/endpoints/publicis/cache/gc"
```

- Compaction:
If enabled in global `ducklake.scheduler`, periodic file merging is performed via DuckLake `ducklake_merge_adjacent_files`.

### Template authoring guide (reference)
Use these variables inside your cache templates and main queries:

- Identification
  - `{{cache.catalog}}` → usually `cache`
  - `{{cache.schema}}` → e.g., `analytics` (auto-created if missing)
  - `{{cache.table}}` → your cache table name

- Mode and scheduling
  - `{{params.cacheMode}}` → `full` | `append` | `merge`
  - `{{cache.schedule}}` → if set in YAML

- Snapshots
  - `{{cache.snapshotId}}`, `{{cache.snapshotTimestamp}}`
  - `{{cache.previousSnapshotId}}`, `{{cache.previousSnapshotTimestamp}}`

- Incremental hints
  - `{{cache.cursorColumn}}`, `{{cache.cursorType}}`
  - `{{cache.primaryKeys}}` → comma-separated list, e.g., `id,tenant_id`

Authoring tips:
- Full refresh: use `CREATE OR REPLACE TABLE ... AS SELECT ...`.
- Append: `INSERT INTO cache.table SELECT ... WHERE event_time > previousSnapshotTimestamp`.
- Merge: `MERGE INTO cache.table USING (SELECT ...) ON pk ...`.
- Do not create schemas in templates; flAPI does that automatically.

### Troubleshooting
- Cache refresh happens on every request: by design this is disabled. Ensure you’re not calling the manual refresh endpoint from a client and that your logs show scheduled or warmup refreshes only.
- Schema not found: verify `cache.schema` is set; flAPI will auto-create it.
- Retention errors: use time-based `max-snapshot-age` first. Version-based retention depends on DuckLake support.

## 🧩 YAML includes and environment variables
flAPI extends plain YAML with lightweight include and environment-variable features so you can keep configurations modular and environment-aware.

### Environment variables
- Write environment variables as `{{env.VAR_NAME}}` anywhere in your YAML.
- Only variables that match the whitelist in your root config are substituted:
  ```yaml
  template:
    path: './sqls'
    environment-whitelist:
      - '^FLAPI_.*'     # allow all variables starting with FLAPI_
      - '^PROJECT_.*'   # optional additional prefixes
  ```
- If the whitelist is empty or omitted, all environment variables are allowed.

Examples:
```yaml
# Substitute inside strings
project-name: "${{env.PROJECT_NAME}}"

# Build include paths dynamically
template:
  path: "{{env.CONFIG_DIR}}/sqls"
```

### Include syntax
You can splice content from another YAML file directly into the current document.

- Basic include: `{{include from path/to/file.yaml}}`
- Section include: `{{include:top_level_key from path/to/file.yaml}}` includes only that key
- Conditional include: append `if <condition>` to either form

Conditions supported:
- `true` or `false`
- `env.VAR_NAME` (include if the variable exists and is non-empty)
- `!env.VAR_NAME` (include if the variable is missing or empty)

Examples:
```yaml
# Include another YAML file relative to this file
{{include from common/settings.yaml}}

# Include only a section (top-level key) from a file
{{include:connections from shared/connections.yaml}}

# Conditional include based on an environment variable
{{include from overrides/dev.yaml if env.FLAPI_ENV}}

# Use env var in the include path
{{include from {{env.CONFIG_DIR}}/secrets.yaml}}
```

Resolution rules and behavior:
- Paths are resolved relative to the current file first; absolute paths are supported.
- Includes inside YAML comments are ignored (e.g., lines starting with `#`).
- Includes are expanded before the YAML is parsed.
- Includes do not recurse: include directives within included files are not processed further.
- Circular includes are guarded against within a single expansion pass; avoid cycles.

Tips:
- Prefer section includes (`{{include:...}}`) to avoid unintentionally overwriting unrelated keys.
- Keep shared blocks in small files (e.g., `connections.yaml`, `auth.yaml`) and include them where needed.



## 🏭 Building from source
The source code of flAPI is written in C++ and closely resembles the [DuckDB build process](https://duckdb.org/docs/dev/building/overview). A good documentation of the build process is the GitHub action in [`build.yaml`](.github/workflows/build.yaml). In essecence a few prerequisites need to be met:
In essecence a few prerequisites need to be met:

- Install the dependencies: `sudo apt-get install -y build-essential cmake ninja-build`
- Checkout the repository and submodules: `git clone --recurse-submodules https://github.com/datazoode/flapi.git`
- Build the project: `make release`

The build process will download and build DuckDB v1.1.2 and install the vcpkg package manager. We depend on the following vcpkg ports:

- [`argparse`](https://github.com/p-ranav/argparse) - Command line argument parser
- [`crow`](https://github.com/CrowCpp/Crow) - Our REST-Web framework and JSON handling
- [`yaml-cpp`](https://github.com/jbeder/yaml-cpp) - YAML parser
- [`jwt-cpp`](https://github.com/Thalhammer/jwt-cpp) - JSON Web Token library
- [`openssl`](https://github.com/openssl/openssl) - Crypto library
- [`catch2`](https://github.com/catchorg/Catch2) - Testing framework

**Note**: MCP support is built-in and doesn't require additional dependencies beyond what's already included.

## 📚 Documentation

For more detailed information, check out our [full documentation](docs/):

- **[Reference Documentation Map](docs/REFERENCE_MAP.md)** - Quick navigation guide for all docs
- **[Configuration Reference](docs/CONFIG_REFERENCE.md)** - Complete flapi.yaml configuration options
- **[MCP Reference](docs/MCP_REFERENCE.md)** - Model Context Protocol specification and implementation
- **[Config Service API Reference](docs/CONFIG_SERVICE_API_REFERENCE.md)** - REST API for runtime configuration
- **[CLI Reference](docs/CLI_REFERENCE.md)** - Server executable command-line options
- **[Cloud Storage Guide](docs/CLOUD_STORAGE_GUIDE.md)** - Using cloud storage backends (S3, GCS, Azure)
- **[Architecture & Design](docs/spec/)** - System architecture, design decisions, and component documentation

## 🤝 Contributing

We welcome contributions! Please see our [Contributing Guide](CONTRIBUTING.md) for more details.

## 📄 License

flAPI is licensed under the Apache License 2.0. See the [LICENSE](LICENSE) file for more details.

## 🙋‍♀️ Support

If you have any questions or need help, please [open an issue](https://github.com/yourusername/flapi/issues) or join our [community chat](link-to-your-chat).

---
 readmeEtag: '"0ed289390b9a92677f6fc0a168faa97988d4b14a"' readmeLastModified: Sun, 25 Jan 2026 15:30:32 GMT repositoryId: 864915169 description: >- API Framework heavily relying on the power of DuckDB and DuckDB extensions. Ready to build performant and cost-efficient APIs on top of BigQuery or Snowflake for AI Agents and Data Apps created: '2024-09-29T14:06:01Z' updated: '2026-02-01T17:44:15Z' language: C++ archived: false stars: 65 watchers: 3 forks: 3 owner: DataZooDE logo: https://avatars.githubusercontent.com/u/136052936?v=4 license: Apache-2.0 repoEtag: '"74bbbe8627af85a241acd2f6f8e9de5698c9e238a5824088277526858e6ea064"' repoLastModified: Sun, 01 Feb 2026 17:44:15 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/juststeveking/go-api v3: true repositoryMetadata: base64Readme: >- IyBHbyBBUEkKCkEgc2ltcGxlIEdvIEFQSSBmb2xsb3dpbmcgY29uY2VwdHMgb2YgRG9tYWluIERyaXZlbiBEZXNpZ24gZm9yIGVkdWNhdGlvbmFsIHB1cnBvc2VzLgoKT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiBjYW4gYmUgZm91bmQgW2hlcmVdKGRvY3MvcmVmZXJlbmNlL29wZW5hcGkudjEueWFtbCkKCiMjIEluc3RhbGxhdGlvbgoKQ2xvbmUgdGhlIHJlcG8KCmBgYGJhc2gKJCBnaXQgY2xvbmUgZ2l0QGdpdGh1Yi5jb206SnVzdFN0ZXZlS2luZy9nby1hcGkuZ2l0CmBgYAoKYGBgYmFzaAokIGNkIGdvLWFwaQpgYGAKCiMjIFJ1bm5pbmcKCk9uY2UgaW5zdGFsbGVkLCBzaW1wbHkgc3BpbiB1cCB0aGUgZG9ja2VyIGNvbnRhaW5lcjoKCmBgYGJhc2gKJCBkb2NrZXItY29tcG9zZSB1cCAtZCAtLWJ1aWxkCmBgYAoKVGhpcyB3aWxsIGdpdmUgeW91IGEgc2luZ2xlIGVuZHBvaW50IGZvciBub3cgdW5kZXI6IGBodHRwOi8vbG9jYWxob3N0OjgwODAvcGluZ2Agd2hpY2ggaXMgYSBoZWFsdGhjaGVjayBVUkwsIGFuZCB3aWxsIHJldHVybiB0aGUgZm9sbG93aW5nOgoKYGBganNvbgp7CiAgICAibWVzc2FnZSI6ICJTZXJ2aWNlIE9ubGluZSIKfQpgYGAK readmeEtag: '"69eb1ee01846f68945f716e4100d11270378bf27"' readmeLastModified: Mon, 20 Jul 2020 10:51:45 GMT repositoryId: 279888235 description: >- A simple Go API following concepts of Domain Driven Design for educational purposes. created: '2020-07-15T14:18:30Z' updated: '2025-10-14T07:57:32Z' language: Go archived: false stars: 23 watchers: 0 forks: 3 owner: JustSteveKing logo: https://avatars.githubusercontent.com/u/6368379?v=4 repoEtag: '"a1c17e0ae2491ddf21b04a00197af227306ed77b770dbe999a10fefbb15bbf22"' repoLastModified: Tue, 14 Oct 2025 07:57:32 GMT foundInMaster: true category: - Code Generators - Server Implementations id: a5ef021c7ce2024aa60ecd3895fd1304 - source: openapi3 tags repository: https://github.com/simonireilly/compeller v3: true repositoryMetadata: base64Readme: >- IyBDb21wZWxsZXIKCkEgc3Ryb25nIHR5cGVzY3JpcHQgYmluZGluZyBmb3IgeW91ciBPcGVuQVBJIFNjaGVtYSB0aGF0IGRvZXNuJ3QgbmVlZCBnZW5lcmF0aW9uLCBhbmQgaXNuJ3QgcHJlc2NyaXB0aXZlLgoKIVtob3cgdG8gZ2V0IHN0YXJ0ZWQgd2l0aCBjb21wZWxsZXJdKC4vYXNzZXRzL3VzYWdlLmdpZikKCgotIFtDb21wZWxsZXJdKCNjb21wZWxsZXIpCiAgLSBb8J+aqCBBbHBoYSBzb2Z0d2FyZSDwn5qoXSgjLWFscGhhLXNvZnR3YXJlLSkKICAtIFtBYm91dF0oI2Fib3V0KQogIC0gW0dldCBzdGFydGVkXSgjZ2V0LXN0YXJ0ZWQpCiAgLSBb8J+bo++4jyBSb2FkIG1hcF0oI++4jy1yb2FkLW1hcCkKICAgIC0gW1VzYWdlXSgjdXNhZ2UpCiAgLSBbU2hvdWxkZXJzXSgjc2hvdWxkZXJzKQoKIyMg8J+aqCBBbHBoYSBzb2Z0d2FyZSDwn5qoCgpDb21wZWxsZXIgaXMgaW4gYWxwaGEsIHNvIGl0J3MgQVBJIG1pZ2h0IGNoYW5nZSwgbWF5YmUgeW91IGhhdmUgc29tZSB0aG91Z2h0cz8KCiMjIEFib3V0CgpDb21wZWxsZXIgdHJpZXMgdG8gaW5mZXIgeW91ciBPcGVuQVBJIHZhbGlkYXRpb25zIGFuZCByZXNwb25zZXMsIGZyb20gYSB0eXBlZCBPcGVuQVBJIHNwZWNpZmljYXRpb24uCgojIyBHZXQgc3RhcnRlZAoKWW91IGNhbiB1c2UgdGhlIENMSSB0byBzdGFydCBhIG5ldyBwcm9qZWN0LCBnZW5lcmF0aW5nIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbi4KCmBgYGJhc2gKbnB4IGNvbXBlbGxlckBhbHBoYSBuZXcKYGBgCgojIyDwn5uj77iPIFJvYWQgbWFwCgotIFt4XSBTdXBwb3J0IGZvciByZXF1ZXN0IGJvZHkgdmFsaWRhdGlvbiB0byB0eXBlIGd1YXJkIChhanYpCi0gW3hdIFN1cHBvcnQgZm9yIGhlYWRlciByZXNwb25zZSB0eXBlcwotIFsgXSBTdXBwb3J0IGZvciByZXNwb25zZSB0eXBlIG1hcHBpbmcKLSBbIF0gU3VwcG9ydCBmb3IgcGF0aCB2YWxpZGF0aW9uCi0gWyBdIFN1cHBvcnQgaGVhZGVyIHZhbGlkYXRpb24KCiMjIyBVc2FnZQoKQ3JlYXRlIGEgU2NoZW1hIHNwZWNpZmljYXRpb24gZm9yIGFuIEFQSSBNb2RlbCBsaWtlOgoKYGBgdHMKLy8gLi9leGFtcGxlcy9zdGFuZGFsb25lL29wZW5hcGkvc2NoZW1hcy92ZXJzaW9uLnNjaGVtYS50cwoKaW1wb3J0IHsgRnJvbVNjaGVtYSB9IGZyb20gJ2pzb24tc2NoZW1hLXRvLXRzJzsKCmV4cG9ydCBjb25zdCBWZXJzaW9uU2NoZW1hID0gewogIHR5cGU6ICdvYmplY3QnLAogIHJlcXVpcmVkOiBbJ3ZlcnNpb24nXSwKICBhZGRpdGlvbmFsUHJvcGVydGllczogZmFsc2UsCiAgcHJvcGVydGllczogewogICAgdmVyc2lvbjogewogICAgICB0eXBlOiAnc3RyaW5nJywKICAgIH0sCiAgfSwKfSBhcyBjb25zdDsKCmV4cG9ydCB0eXBlIFZlcnNpb24gPSBGcm9tU2NoZW1hPHR5cGVvZiBWZXJzaW9uU2NoZW1hPjsKCmBgYAoKTmV4dCwgYmluZCB0aGUgbW9kZWwgaW50byBhbiBPcGVuQVBJIHNwZWNpZmljYXRpb24gb2JqZWN0LgoKYGBgdHMKLy8gLi9leGFtcGxlcy9zdGFuZGFsb25lL29wZW5hcGkvc3BlYy50cwoKaW1wb3J0IHsgVmVyc2lvblNjaGVtYSB9IGZyb20gJy4vc2NoZW1hcy92ZXJzaW9uLnNjaGVtYSc7CgpleHBvcnQgY29uc3QgT3BlbkFQSVNwZWNpZmljYXRpb24gPSB7CiAgaW5mbzogewogICAgdGl0bGU6ICdOZXcgQVBJIGdlbmVyYXRlZCB3aXRoIGNvbXBlbGxlcicsCiAgICB2ZXJzaW9uOiAnMS4wLjAnLAogIH0sCiAgb3BlbmFwaTogJzMuMS4wJywKICBwYXRoczogewogICAgJ3YxL3ZlcnNpb24nOiB7CiAgICAgIGdldDogewogICAgICAgIHJlc3BvbnNlczogewogICAgICAgICAgJzIwMCc6IHsKICAgICAgICAgICAgZGVzY3JpcHRpb246ICdHZXQgdGhlIGN1cnJlbnQgQVBJIHZlcnNpb24nLAogICAgICAgICAgICBjb250ZW50OiB7CiAgICAgICAgICAgICAgJ2FwcGxpY2F0aW9uL2pzb24nOiB7CiAgICAgICAgICAgICAgICBzY2hlbWE6IFZlcnNpb25TY2hlbWEsCiAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgfSwKICAgICAgICAgIH0sCiAgICAgICAgfSwKICAgICAgfSwKICAgIH0sCiAgfSwKfTsKCmBgYAoKV2l0aCBjb21wZWxsZXIgeW91IGNhbiBjb21waWxlIHRoaXMgaW50byBhIHR5cGVkIHJlcXVlc3QgYW5kIHJlc3BvbnNlIGhhbmRsZXIgbGlrZToKCmBgYHRzCmltcG9ydCB7IE9wZW5BUElTcGVjaWZpY2F0aW9uIH0gZnJvbSAnLi9zcGVjJzsKCi8vIFRoZSBhcGkgaXMgaW5mZXJyZWQsIGFuZCB2YWxpZGF0aW9ucyBhcmUgZ2VuZXJhdGVkCmNvbnN0IGFwaSA9IGNvbXBlbGxlcihPcGVuQVBJU3BlY2lmaWNhdGlvbik7CgovLyBUaGVzZSByb3V0ZXMgYW5kIG1ldGhvZHMgYXJlIGVuZm9yY2VkCmNvbnN0IHsgcmVzcG9uc2UgfSA9IGFwaSgnL3YxL3ZlcnNpb24nLCAnZ2V0Jyk7CgovLyBUaGUgcmVzcG9uc2UgY29kZSBhbmQgYm9keSBzY2hlbWEgYXJlIGJvdW5kCmNvbnN0IHJlc3AgPSByZXNwb25zZSgnMjAwJywgeyBuYW1lOiAnVHlwZS1zYWZlIHJlcGx5JyB9KTsKYGBgCgpTZWUgWy4vZXhhbXBsZXNdKC4vZXhhbXBsZXMpCgojIyBTaG91bGRlcnMKCkNvbXBlbGwgaXMgYnVpbHQgb24gdG9wIG9mIHNvbWUgZ3JlYXQgbGlicmFyaWVzLCBhdCBpdCdzIGNvcmUgaXQgcmVsaWVzIG9uOgoKLSBbb3BlbmFwaTMtdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tZXRhZGV2cHJvL29wZW5hcGkzLXRzKQotIFtqc29uLXNjaGVtYS10by10c10oaHR0cHM6Ly9naXRodWIuY29tL1Rob21hc0FyaWJhcnQvanNvbi1zY2hlbWEtdG8tdHMpCgpJdCBhbHNvIGxheWVycyBvbiByZXF1ZXN0IGFuZCByZXNwb25zZSB2YWxpZGF0aW9uIHBvd2VyZWQgYnk6CgotIFthanZdKGh0dHBzOi8vZ2l0aHViLmNvbS9hanYtdmFsaWRhdG9yL2FqdikKCkknbSBncmF0ZWZ1bCBmb3IgdGhlc2UgbGlicmFyaWVzCg== readmeEtag: '"0c64ee60e30f8b960c24d00b92244b4f9157b5ab"' readmeLastModified: Mon, 31 Jan 2022 21:08:10 GMT repositoryId: 452000471 description: >- A strong typescript binding for your OpenAPI Schema that doesn't need generation and is not prescriptive in coding style created: '2022-01-25T18:44:08Z' updated: '2024-08-13T03:40:21Z' language: TypeScript archived: false stars: 22 watchers: 4 forks: 1 owner: simonireilly logo: https://avatars.githubusercontent.com/u/30017294?v=4 license: MIT repoEtag: '"a80f043e73215cd0b38c8d823ddc3854efd1652fec839da5cae15a6f4502ac17"' repoLastModified: Tue, 13 Aug 2024 03:40:21 GMT foundInMaster: true category: Parsers id: c2573db36bc9c133fcb66e2714f9a37c - source: openapi3 tags repository: https://github.com/zweidenker/openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy96d2VpZGVua2VyL09wZW5BUEkuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvendlaWRlbmtlci9PcGVuQVBJKQo= readmeEtag: '"79163d291a392028dcd86abd2b0143eb4f734b1b"' readmeLastModified: Thu, 21 Jan 2021 16:35:59 GMT repositoryId: 118589193 description: A pharo implementation of OpenAPI 3.0.1 created: '2018-01-23T09:38:10Z' updated: '2023-08-13T00:57:19Z' language: Smalltalk archived: false stars: 22 watchers: 5 forks: 7 owner: zweidenker logo: https://avatars.githubusercontent.com/u/20300491?v=4 license: MIT repoEtag: '"9b21a980252330951eb3210abff815894c4378b2bb2fc85e0aed793ed7ae5ec5"' repoLastModified: Sun, 13 Aug 2023 00:57:19 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 1dbc4782ef0fd6319683bd896d58b471 - source: - openapi3 tags - openapi31 tags repository: https://github.com/mnahkies/openapi-code-generator v3: true repositoryMetadata: base64Readme: >- IyBAbmFoa2llcy9vcGVuYXBpLWNvZGUtZ2VuZXJhdG9yCgpbIVtDSS9DRF0oaHR0cHM6Ly9naXRodWIuY29tL21uYWhraWVzL29wZW5hcGktY29kZS1nZW5lcmF0b3IvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9tbmFoa2llcy9vcGVuYXBpLWNvZGUtZ2VuZXJhdG9yL2FjdGlvbnM/cXVlcnk9YnJhbmNoJTNBbWFpbitldmVudCUzQXB1c2gpClshW25wbV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9AbmFoa2llcy9vcGVuYXBpLWNvZGUtZ2VuZXJhdG9yLnN2ZyldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL0BuYWhraWVzL29wZW5hcGktY29kZS1nZW5lcmF0b3IpCgpgQG5haGtpZXMvb3BlbmFwaS1jb2RlLWdlbmVyYXRvcmAgaXMgYSBDTEkgdG9vbCB0aGF0IGFpbXMgdG8gZ2VuZXJhdGUgaGlnaCBxdWFsaXR5IHR5cGVzY3JpcHQgY2xpZW50IFNESydzLAphbmQgQVBJIHNlcnZlciBzY2FmZm9sZGluZyAocm91dGluZywgdmFsaWRhdGlvbiwgc2VyaWFsaXphdGlvbikgZnJvbSBhcGkgc3BlY2lmaWNhdGlvbnMuCgpDdXJyZW50bHksIFtPcGVuQVBJIDMuMC54XShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4wLjQuaHRtbCksIFtPcGVuQVBJIDMuMS54XShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4xLjEuaHRtbCksCmFuZCBbVHlwZVNwZWNdKGh0dHBzOi8vdHlwZXNwZWMuaW8vKSBhcmUgc3VwcG9ydGVkIGFzIGlucHV0IHNwZWNpZmljYXRpb25zLgoKV2l0aCB0eXBlc2NyaXB0IHRlbXBsYXRlcyBmb3IgW2tvYV0oaHR0cHM6Ly9vcGVuYXBpLWNvZGUtZ2VuZXJhdG9yLm5haGtpZXMuY28ubnovZ3VpZGVzL3NlcnZlci10ZW1wbGF0ZXMvdHlwZXNjcmlwdC1rb2EpLCBbZXhwcmVzc10oaHR0cHM6Ly9vcGVuYXBpLWNvZGUtZ2VuZXJhdG9yLm5haGtpZXMuY28ubnovZ3VpZGVzL3NlcnZlci10ZW1wbGF0ZXMvdHlwZXNjcmlwdC1leHByZXNzKSwgW2ZldGNoXShodHRwczovL29wZW5hcGktY29kZS1nZW5lcmF0b3IubmFoa2llcy5jby5uei9ndWlkZXMvY2xpZW50LXRlbXBsYXRlcy90eXBlc2NyaXB0LWZldGNoKSwgW2F4aW9zXShodHRwczovL29wZW5hcGktY29kZS1nZW5lcmF0b3IubmFoa2llcy5jby5uei9ndWlkZXMvY2xpZW50LXRlbXBsYXRlcy90eXBlc2NyaXB0LWF4aW9zKSwgYW5kIFthbmd1bGFyXShodHRwczovL29wZW5hcGktY29kZS1nZW5lcmF0b3IubmFoa2llcy5jby5uei9ndWlkZXMvY2xpZW50LXRlbXBsYXRlcy90eXBlc2NyaXB0LWFuZ3VsYXIpIGN1cnJlbnRseSBhdmFpbGFibGUuCgpUaGUgW2ZldGNoXShodHRwczovL29wZW5hcGktY29kZS1nZW5lcmF0b3IubmFoa2llcy5jby5uei9ndWlkZXMvY2xpZW50LXRlbXBsYXRlcy90eXBlc2NyaXB0LWZldGNoKSBhbmQgW2F4aW9zXShodHRwczovL29wZW5hcGktY29kZS1nZW5lcmF0b3IubmFoa2llcy5jby5uei9ndWlkZXMvY2xpZW50LXRlbXBsYXRlcy90eXBlc2NyaXB0LWF4aW9zKSB0ZW1wbGF0ZXMgd29yayBncmVhdCBpbiBjb25qdW5jdGlvbiB3aXRoIFtyZWFjdC1xdWVyeV0oaHR0cHM6Ly90YW5zdGFjay5jb20vcXVlcnkvbGF0ZXN0KQoKPiBcWyFUSVBdCj4gVHJ5IG91dCBvdXIgW2ludGVyYWN0aXZlIHBsYXlncm91bmQgaGVyZV0oaHR0cHM6Ly9vcGVuYXBpLWNvZGUtZ2VuZXJhdG9yLm5haGtpZXMuY28ubnovcGxheWdyb3VuZCkKCjwhLS0gdG9jIC0tPgoKKiBbRG9jdW1lbnRhdGlvbl0oI2RvY3VtZW50YXRpb24pCiogW1Byb2plY3QgU3RydWN0dXJlXSgjcHJvamVjdC1zdHJ1Y3R1cmUpCiogW0NvbnRyaWJ1dGluZ10oI2NvbnRyaWJ1dGluZykKKiBbTGljZW5zZV0oI2xpY2Vuc2UpCgo8IS0tIHRvY3N0b3AgLS0+CgojIyBEb2N1bWVudGF0aW9uCgpWaXNpdCBodHRwczovL29wZW5hcGktY29kZS1nZW5lcmF0b3IubmFoa2llcy5jby5uei8gZm9yIGRldGFpbGVkIGRvY3VtZW50YXRpb24gaW5jbHVkaW5nCnF1aWNrIHN0YXJ0IGd1aWRlcyBhbmQgcmVmZXJlbmNlIG1hdGVyaWFsLgoKVGhlIGRvY3VtZW50YXRpb24gaXMgYnVpbHQgdXNpbmcgTmV4dEpTIC8gTmV4dHJhLCBhbmQgZGVwbG95ZWQgdXNpbmcgR2l0aHViIHBhZ2VzLgpZb3UgY2FuIGNvbnRyaWJ1dGUgdG8gaXQgaW4gWy4vcGFja2FnZXMvZG9jdW1lbnRhdGlvbl0oLi9wYWNrYWdlcy9kb2N1bWVudGF0aW9uKQoKIyMgUHJvamVjdCBTdHJ1Y3R1cmUKClRoZSByZXBvc2l0b3J5IGlzIHN0cnVjdHVyZWQgYXMgYSBtb25vIHJlcG8gb2Ygc2V2ZXJhbCBucG0gcGFja2FnZXMgdGhhdCB3b3JrIHRvZ2V0aGVyIHVuZGVyIFsuL3BhY2thZ2VzXSguL3BhY2thZ2VzKToKCiogW29wZW5hcGktY29kZS1nZW5lcmF0b3JdKC4vcGFja2FnZXMvb3BlbmFwaS1jb2RlLWdlbmVyYXRvcikKKiBbdHlwZXNjcmlwdC1heGlvcy1ydW50aW1lXSguL3BhY2thZ2VzL3R5cGVzY3JpcHQtYXhpb3MtcnVudGltZSkKKiBbdHlwZXNjcmlwdC1leHByZXNzLXJ1bnRpbWVdKC4vcGFja2FnZXMvdHlwZXNjcmlwdC1leHByZXNzLXJ1bnRpbWUpCiogW3R5cGVzY3JpcHQtZmV0Y2gtcnVudGltZV0oLi9wYWNrYWdlcy90eXBlc2NyaXB0LWZldGNoLXJ1bnRpbWUpCiogW3R5cGVzY3JpcHQta29hLXJ1bnRpbWVdKC4vcGFja2FnZXMvdHlwZXNjcmlwdC1rb2EtcnVudGltZSkKClRoZSBgb3BlbmFwaS1jb2RlLWdlbmVyYXRvcmAgcGFja2FnZSBpcyB0aGUgbWFpbiBwYWNrYWdlLCB3aGlsc3QgdGhlIG90aGVycyBhcmUgc3VwcG9ydGluZyBwYWNrYWdlcyB1c2VkIGF0IHJ1bnRpbWUgYnkKdGhlIGNvZGUgb3V0cHV0IGJ5IHNvbWUgb2YgdGhlIHRlbXBsYXRlcy4KCkludGVncmF0aW9uIHRlc3QgZGVmaW5pdGlvbnMgbGl2ZSBpbiBbLi9pbnRlZ3JhdGlvbi10ZXN0cy1kZWZpbml0aW9uc10oLi9pbnRlZ3JhdGlvbi10ZXN0cy1kZWZpbml0aW9ucykgYW5kIHRoZSBnZW5lcmF0ZWQKY29kZSBvdXRwdXQgdG8gWy4vaW50ZWdyYXRpb24tdGVzdHNdKC4vaW50ZWdyYXRpb24tdGVzdHMpCgpTY3JpcHRzIHRvIHJlZnJlc2ggdGhlIHRlc3QgZGF0YSBsaXZlIGluIFsuL3NjcmlwdHNdKC4vc2NyaXB0cykKCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW5nIGd1aWRlbGluZXMgY2FuIGJlIGZvdW5kIGluIFsuL0NPTlRSSUJVVElORy5tZF0oLi9DT05UUklCVVRJTkcubWQpLgoKIyMgTGljZW5zZQoKTUlUIExpY2Vuc2VkLCBzZWUgWy4vTElDRU5TRV0oLi9MSUNFTlNFKQo= readmeEtag: '"1d2ed0cd775c5ab5ed6b4f9d7a1eaf2f1aeba583"' readmeLastModified: Sun, 02 Nov 2025 13:00:54 GMT repositoryId: 283031380 description: >- A code generation tool for openapi 3 / 3.1 specifications written in typescript, primarily aimed at generating typescript clients and server stubs. Other target languages may be added in future. created: '2020-07-27T22:27:40Z' updated: '2026-02-04T12:58:14Z' language: TypeScript archived: false stars: 34 watchers: 4 forks: 5 owner: mnahkies logo: https://avatars.githubusercontent.com/u/2555533?v=4 license: MIT repoEtag: '"8f014148c4febd6403bda05a03da4e6d57f7c4db197fcbb2ec443124511cdcaf"' repoLastModified: Wed, 04 Feb 2026 12:58:14 GMT foundInMaster: true category: Parsers id: f48e00de5360ad43542a200fc9b5769b v3_1: true - source: openapi3 tags repository: https://github.com/dkackman/chia-api v3: true repositoryMetadata: base64Readme: >- IyBjaGlhLWFwaQoKWyFbT3BlbkFQSSBMaW50ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9ka2Fja21hbi9jaGlhLWFwaS9hY3Rpb25zL3dvcmtmbG93cy9saW50ZXIueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZGthY2ttYW4vY2hpYS1hcGkvYWN0aW9ucy93b3JrZmxvd3MvbGludGVyLnlhbWwpCgpBbiB1bm9mZmljYWwgW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIGRlc2NyaXB0aW9uIG9mIFtjaGlhXShodHRwczovL2dpdGh1Yi5jb20vQ2hpYS1OZXR3b3JrL2NoaWEtYmxvY2tjaGFpbikgUlBDIGludGVyZmFjZXMuCgpWaWV3IGN1cnJlbnQgc3BlYyBiYXNlZCBvbiBjaGlhIDIuNS4wOgoKLSBbSW4gdGhlIHN3YWdnZXIgVUldKGh0dHBzOi8vZGthY2ttYW4uZ2l0aHViLmlvL2NoaWEtYXBpLz91cmxzLnByaW1hcnlOYW1lPUZ1bGwlMjBOb2RlKQotIFtBcyBhIHJlZG9jIGdlbmVyYXRlZCBzaXRlXShodHRwczovL2RrYWNrbWFuLmdpdGh1Yi5pby9jaGlhLWFwaS9yZWRvYy8pCi0gW0FzIGEgc3dhZ2dlci1jb2RlZ2VuIGdlbmVyYXRlZCBzaXRlXShodHRwczovL2RrYWNrbWFuLmdpdGh1Yi5pby9jaGlhLWFwaS9zdGF0aWMvKQoKX19fCgpfY2hpYSBhbmQgaXRzIGxvZ28gYXJlIHRoZSByZWdpc3RlcmVkIHRyYWRlbWFyayBvciB0cmFkZW1hcmsgb2YgW0NoaWEgTmV0d29yaywgSW5jXShodHRwczovL2NoaWEubmV0KS4gaW4gdGhlIFVuaXRlZCBTdGF0ZXMgYW5kIHdvcmxkd2lkZS5fCg== readmeEtag: '"e4bfe01c7e75933fb5d725934e998f18a61c4861"' readmeLastModified: Sat, 11 Jan 2025 02:05:45 GMT repositoryId: 453839137 description: Swagger for chia https://dkackman.github.io/chia-api/ created: '2022-01-31T00:32:10Z' updated: '2025-03-19T21:03:36Z' language: Mustache archived: false stars: 22 watchers: 1 forks: 5 owner: dkackman logo: https://avatars.githubusercontent.com/u/5160233?v=4 license: Apache-2.0 repoEtag: '"6e7a458e227ca08a9d814d1bfe1de07df1143da5074882677cf46a99d80fb70b"' repoLastModified: Wed, 19 Mar 2025 21:03:36 GMT foundInMaster: true category: Server id: af8d79e99ec00e2f41229885f84248da - source: openapi3 tags repository: https://github.com/postman-open-technologies/bingchain v3: true id: 7775903803c68ddd46ca381ffe39c75b repositoryMetadata: base64Readme: >- IyDwn5W177iP8J+UlyBCaW5nQ2hhaW4KClRoaXMgaXMgYW4gZXZvbHV0aW9uIG9mIFtsYW5nY2hhaW4tbWluaV0oaHR0cHM6Ly9naXRodWIuY29tL0NvbGluRWJlcmhhcmR0L2xhbmdjaGFpbi1taW5pKSwgYSB2ZXJ5IHNpbXBsZSByZS1pbXBsZW1lbnRhdGlvbiBvZiBbTGFuZ0NoYWluXShodHRwczovL2dpdGh1Yi5jb20vaHdjaGFzZTE3L2xhbmdjaGFpbiksIGluIH4zNTAgbGluZXMgb2YgY29yZSBjb2RlLiBJbiBlc3NlbmNlLCBpdCBpcyBhIG11bHRpLW1vZGVsIExMTS1wb3dlcmVkIGNoYXQgYXBwbGljYXRpb24gdGhhdCBpcyBhYmxlIHRvIHVzZSB0b29scyAoTWljcm9zb2Z0ICoqQmluZyoqIHNlYXJjaCwgVVJMIHJldHJpZXZhbCwgQVBJIHBsdWdpbiBpbnN0YWxsYXRpb24sIEFQSSBjYWxscywgYSBKYXZhc2NyaXB0IHNhbmRib3gsIEpzRmlkZGxlIGNyZWF0aW9uLCBpbWFnZSBhbmQgdmlkZW8gcHJldmlldywgYW5kIGEgc2NpZW50aWZpYyBjYWxjdWxhdG9yLCBhcyB3ZWxsIGFzIG1ldGEtdG9vbHMgc3VjaCBhcyBgbGlzdGAsIGBkaXNhYmxlYCwgYHJlc2V0YCBhbmQgYGRlYnVnYCkgaW4gb3JkZXIgdG8gYnVpbGQgYSAqKmNoYWluKiogb2YgdGhvdWdodCB0byBob2xkIGNvbnZlcnNhdGlvbnMgYW5kIGFuc3dlciBxdWVzdGlvbnMuCgpIZXJlJ3MgYW4gZXhhbXBsZToKCn5+fgpROiBXaGF0IGlzIHRoZSB3b3JsZCByZWNvcmQgZm9yIHNvbHZpbmcgYSBydWJpa3MgY3ViZT8KVGhlIHdvcmxkIHJlY29yZCBmb3Igc29sdmluZyBhIFJ1YmlrJ3MgQ3ViZSBpcyA0LjY5IHNlY29uZHMsIGhlbGQgYnkgWWloZW5nIFdhbmcgKENoaW5hKS4KUTogQ2FuIGEgcm9ib3Qgc29sdmUgaXQgZmFzdGVyPwpUaGUgZmFzdGVzdCB0aW1lIGEgcm9ib3QgaGFzIHNvbHZlZCBhIFJ1YmlrJ3MgQ3ViZSBpcyAwLjYzNyBzZWNvbmRzLgpROiBXaG8gbWFkZSB0aGlzIHJvYm90PwpJbmZpbmVvbiBjcmVhdGVkIHRoZSByb2JvdCB0aGF0IHNvbHZlZCBhIFJ1YmlrJ3MgQ3ViZSBpbiAwLjYzNyBzZWNvbmRzLgpROiBXaGF0IHRpbWUgd291bGQgYW4gYXZlcmFnZSBodW1hbiBleHBlY3QgZm9yIHNvbHZpbmc/Ckl0IHRha2VzIHRoZSBhdmVyYWdlIHBlcnNvbiBhYm91dCB0aHJlZSBob3VycyB0byBzb2x2ZSBhIFJ1YmlrJ3MgY3ViZSBmb3IgdGhlIGZpcnN0IHRpbWUuCn5+fgoKVGhpcyBpcyBub3QgaW50ZW5kZWQgdG8gYmUgYSByZXBsYWNlbWVudCBmb3IgTGFuZ0NoYWluLCB3aGljaCBoYXMgbWFueSBhbHRlcm5hdGl2ZSBhbmQgY29tcG9zYWJsZSBidWlsZGluZyBibG9ja3MsIGluc3RlYWQgaXQgd2FzIGJ1aWx0IHRvIGRlbW9uc3RyYXRlIHRoZSBwb3dlciBvZiBhc3NlbWJsaW5nIGEgc2V0IG9mIHRvb2xzIChzdWNoIGFzIEFQSSBjYWxsaW5nIGFuZCBKYXZhc2NyaXB0IGV4ZWN1dGlvbikuIElmIHlvdSdyZSBpbnRlcmVzdGVkIGluIGhvdyBMYW5nQ2hhaW4sIGFuZCBzaW1pbGFyIHRvb2xzIHdvcmssIHRoaXMgaXMgYSB2ZXJ5IGdvb2Qgc3RhcnRpbmcgcG9pbnQuCgojIyBSdW5uaW5nIC8gZGV2ZWxvcGluZwoKSW5zdGFsbCBkZXBlbmRlbmNpZXMsIGFuZCBydW4gKHdpdGggbm9kZSA+PSB2MTgpOgoKfn5+CiUgbnBtIGluc3RhbGwKfn5+CgpUbyBkaXNwbGF5IHZpZGVvcyBpbiB0aGUgdGVybWluYWwsIHlvdSB3aWxsIG5lZWQgdG8gaW5zdGFsbCBgZmZtcGVnYC4KCllvdSdsbCBuZWVkIHRvIGhhdmUgYW4gT3BlbkFJIEFQSSBrZXksIGFuZCBvcHRpb25hbGx5IGEgQmluZyBTZWFyY2ggQVBJIGtleS4gVGhlc2UgY2FuIGJlIHN1cHBsaWVkIHRvIHRoZSBhcHBsaWNhdGlvbiB2aWEgYSBgLmVudmAgZmlsZToKCmBgYHNoZWxsCk9QRU5BSV9BUElfS0VZPSIuLi4iCkJJTkdfQVBJX0tFWT0iLi4uIgpNT0RFTD1ncHQtNApUT0tFTl9MSU1JVD0zMjc2OApURU1QRVJBVFVSRT0wLjI1ClJFU1BPTlNFX0xJTUlUPTUxMgpQT1JUPTEzMzcKR1VJPTEKI0xBTkc9VWtyYWluaWFuCiNERUJVRz0yCiNTRUVEX1FVRVJJRVM9MQojUFJPTVBUX09WRVJSSURFPVJpZGRsZSBtZSB0aGlzISAke3F1ZXN0aW9ufQpgYGAKCllvdSBjYW4gYWxzbyBzZXQgYFBST1ZJREVSPWFudGhyb3BpY2AgKHdpdGggYSByZWxldmFudCBgQU5USFJPUElDX0FQSV9LRVlgLCBgTU9ERUxgIGFuZCBgVE9LRU5fTElNSVRgKSB0byB1c2UgYW4gYWx0ZXJuYXRpdmUgTExNL0FQSSBwcm92aWRlci4KClNldCB0aGUgdG9rZW4gbGltaXQgdG8gdGhlIGFkdmVydGlzZWQgbGltaXQgb2YgdGhlIG1vZGVsIHlvdSBhcmUgdXNpbmcsIHNvIDMyNzY4IGZvciBgZ3B0LTRgLCA0MDk2IGZvciBgdGV4dC1kYXZpbmNpLTAwM2AgYW5kIDIwNDggZm9yIGB0ZXh0LWN1cmllLTAwMWAuCgpUaGUgY2xldmVyIHBhcnQgaXMgdGhlIGRlZmF1bHQgaW5pdGlhbCBwcm9tcHQsIHdoaWNoIGlzIGhlbGQgaW4gW2Bwcm9tcHQudHh0YF0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3Bvc3RtYW4tb3Blbi10ZWNobm9sb2dpZXMvYmluZ2NoYWluL21haW4vcHJvbXB0LnR4dCksIHVubGVzcyBvdmVycmlkZGVuIGJ5IHRoZSBgUFJPTVBUX09WRVJSSURFYCBlbnZpcm9ubWVudCB2YXJpYWJsZS4KCkV4YW1wbGUgcHJvbXB0cyBhbmQgcmVzcG9uc2VzIHRvIHNob3cgaG93IHRoZSB2YXJpb3VzIGJ1aWx0LWluIHRvb2xzIHdvcmsgY2FuIGJlIGZvdW5kIGluIHRoZSBbYGV4YW1wbGVzYF0oaHR0cHM6Ly9naXRodWIuY29tL3Bvc3RtYW4tb3Blbi10ZWNobm9sb2dpZXMvYmluZ2NoYWluL3RyZWUvbWFpbi9leGFtcGxlcykgZGlyZWN0b3J5LiBUaGUgdG9vbHMgdGhlbXNlbHZlcyBhcmUgZGVmaW5lZCBpbiBbYGxpYi90b29scy5tanNgXShodHRwczovL2dpdGh1Yi5jb20vcG9zdG1hbi1vcGVuLXRlY2hub2xvZ2llcy9iaW5nY2hhaW4vdHJlZS9tYWluL2xpYi90b29scy5tanMpLCBpbmNsdWRpbmcgdGhlIGBkZXNjcmlwdGlvbmAgcHJvcGVydGllcyB3aGljaCBhY3QgYXMgZnVydGhlciBwcm9tcHRzIHRvIHRoZSBMTE0gdG8gc3VnZ2VzdCB3aGVuIGFuZCBob3cgdGhlIHRvb2xzIHNob3VsZCBiZSB1c2VkLgoKVGhlcmUgYXJlIGEgZmV3IEphdmFzY3JpcHQgYW5kIENTUyBmaWxlcyBzY2F0dGVyZWQgYWJvdXQgZnJvbSBbanNmaWRkbGUubmV0XShodHRwczovL2pzZmlkZGxlLm5ldC8pIHRvIG1ha2UgdGhlIGBzYXZldGV4dGAsIGBzYXZlaHRtbGAgYW5kIGBzYXZlY29kZWAgdG9vbHMgd29yayBsb2NhbGx5LgoKKipOb3RlKio6IHRvIGVuYWJsZSB0aGUgSmF2YXNjcmlwdCBzYW5kYm94LCB5b3UgbXVzdCBwYXNzIHRoZSBvcHRpb24gYC0tZXhwZXJpbWVudGFsLXZtLW1vZHVsZXNgIHRvIE5vZGUuanMuIFRoZSBpbmNsdWRlZCBgZ28uc2hgIHNjcmlwdCBzZXRzIHRoZSBOb2RlLmpzIHJlY29tbWVuZGVkIG9wdGlvbnMuCgojIyBTdGFydC11cAoKVGhlIGFwcGxpY2F0aW9uIHdpbGwgZGlzcGxheSB0aGUgYnVpbHQtaW4gdG9vbHMgYXMgaXQgaW5pdGlhbGlzZXMgdGhlbS4gVG9vbCBuYW1lcyBmb2xsb3dlZCBieSBbMV0gYXJlIGRpc2FibGVkIGJ5IGRlZmF1bHQgZm9yIHNlY3VyaXR5IHJlYXNvbnMgKGkuZS4gdGhleSBtYXkgYWNjZXNzIGZpbGVzIG9uIHlvdXIgbG9jYWwgZmlsZXN5c3RlbSBvciB5b3VyIGVudmlyb25tZW50IHZhcmlhYmxlcykuIFlvdSBjYW4gZW5hYmxlIHRoZW0gYnkgdHlwaW5nIGBlbmFibGUgW3Rvb2xuYW1lXWAgYXQgdGhlIHByb21wdC4gVG9vbCBuYW1lcyBmb2xsb3dlZCBieSBbMl0gYXJlIGRpc2FibGVkIGJlY3Vhc2UgeW91IGRvIG5vdCBoYXZlIHRoZSByZXF1aXNpdGUgQVBJIGtleSBpbiB5b3VyIGVudmlyb25tZW50IG9yIHlvdXIgdmVyc2lvbiBvZiBOb2RlLmpzIGRvZXMgbm90IHN1cHBvcnQgdGhlIHJlcXVpcmVkIGZlYXR1cmVzLgoKIyMgRXhhbXBsZSBkaWFsb2d1ZQoKWW91IGNhbiBub3cgcnVuIHRoZSBjaGFpbjoKCmBgYHJlcGwKJSAuL2dvLnNoCkhvdyBjYW4gSSBoZWxwPyA+IHdoYXQgd2FzIHRoZSBuYW1lIG9mIHRoZSBmaXJzdCB3b21hbiBpbiBzcGFjZT8KYGBgCgoqIEkgbmVlZCB0byBzZWFyY2ggZm9yIHRoZSBuYW1lIG9mIHRoZSBmaXJzdCB3b21hbiBpbiBzcGFjZS4KKiAqQWN0aW9uKjogYHNlYXJjaGAKKiAqQWN0aW9uIElucHV0KjogYGZpcnN0IHdvbWFuIGluIHNwYWNlIG5hbWVgCgpDYWxsaW5nIGBzZWFyY2hgIHdpdGggYGZpcnN0IHdvbWFuIGluIHNwYWNlIG5hbWVgCgoxLiAqKlZhbGVudGluYSBUZXJlc2hrb3ZhIC0gRmlyc3QgV29tYW4gaW4gU3BhY2UgLSBCaW9ncmFwaHkqKgoyLiAqKlZhbGVudGluYSBUZXJlc2hrb3ZhOiBGaXJzdCBXb21hbiBpbiBTcGFjZSB8IFNwYWNlKioKMy4gKipUaGUgRmlyc3QgV29tYW4gaW4gU3BhY2U6IFZhbGVudGluYSBUZXJlc2hrb3ZhIC0gVGhvdWdodENvKioKCiogKlRob3VnaHQqOiBJIG5vdyBrbm93IHRoZSBmaW5hbCBhbnN3ZXIuCiogKkZpbmFsIEFuc3dlcio6IFRoZSBuYW1lIG9mIHRoZSBmaXJzdCB3b21hbiBpbiBzcGFjZSBpcyBWYWxlbnRpbmEgVGVyZXNoa292YS4KKiAqKlRoZSBuYW1lIG9mIHRoZSBmaXJzdCB3b21hbiBpbiBzcGFjZSBpcyBWYWxlbnRpbmEgVGVyZXNoa292YS4qKgoKIyMjIEV4aXRpbmcgdGhlIGNoYWluIC8gdmkgbW9kZQoKKiBZb3UgY2FuIHVzZSBgdmlgL2B2aW1gLWxpa2UgY29tbWFuZHMgdG8gZXhpdCwgc3VjaCBhcyBgOnFgIG9yIHlvdSBjYW4gQ3RybC1DIHR3aWNlIHRvIGV4aXQuCiogWW91IGNhbiB1c2UgYDpzZXRgIHRvIHF1ZXJ5IGFsbCBlbnZpcm9ubWVudCB2YXJpYWJsZXMgb3IgYDpzZXQgW3ZhcmlhYmxlXT1bdmFsdWVdYCB0byB0ZW1wb3JhcmlseSBhbWVuZCB0aGUgY3VycmVudCBlbnZpcm9ubWVudC4KCiMjIEF1dGhvcnMKCiogW01pa2UgUmFscGhzb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9NaWtlUmFscGhzb24pCiogW0diYWRleWJvaCBCZWxsb10oaHR0cHM6Ly9naXRodWIuY29tL0diYWhkZXlib2gpCiogW0NvbGluIEViZXJoYXJkdF0oaHR0cHM6Ly9naXRodWIuY29tL0NvbGluRWJlcmhhcmR0KQoKIyMgRnV0dXJlIHdvcmsgcGxhbm5lZAoKKiBJZGVhcyBhbmQgUFJzIGdyYXRlZnVsbHkgcmVjZWl2ZWQuCg== readmeEtag: '"212c190a1f5a844903068a3e6c9644f508923de8"' readmeLastModified: Fri, 19 May 2023 10:39:01 GMT repositoryId: 636541324 description: Capable Langchain/AutoGPT alternative in ~350 lines of core code created: '2023-05-05T04:49:20Z' updated: '2025-11-26T00:14:37Z' language: JavaScript archived: false stars: 20 watchers: 5 forks: 2 owner: postman-open-technologies logo: https://avatars.githubusercontent.com/u/79494470?v=4 license: MIT repoEtag: '"e41bd45fa6dc65acd0aba7e7f9514007f41c1f9704a3fe066aed536aab9dffef"' repoLastModified: Wed, 26 Nov 2025 00:14:37 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cloudedbats/cloudedbats_wurb_2020 v3: true repositoryMetadata: base64Readme: >- #  CloudedBats - WURB 2020

Welcome to CloudedBats-WURB, the DIY bat detector.

**Note: This is a new version of the WURB bat detector.
The old version can still be found here: 
https://github.com/cloudedbats/cloudedbats_wurb**

![WURB-2020](images/CloudedBats-WURB.jpg?raw=true  "CloudedBats WURB 2020.")
CloudedBats - WURB 2020. CloudedBats.org / [CC-BY](https://creativecommons.org/licenses/by/3.0/)

## Introduction

This is what you need to build your own bat detector; an ultrasonic USB microphone, a Raspberry Pi computer 
with SD card, power supply and this CloudedBats-WURB 2020 open source software. Optional parts that 
are recommended are USB memory sticks, an USB GPS receiver and a Power bank for single night sessions.
For the v0.9.0 release you will also need headphones with a 3.5 mm jack for audio feedback.

## Functional features, from the earlier CloudedBats-WURB:

- Support for USB ultrasonic microphones from Pettersson and Dodotronic 
  (tested with Pettersson u256, u384, M500 (the 500 kHz variant), M500-384 (384 kHz) and 
  Dodotronic UltraMic 192K, 250K, 384K BLE).
- Support for USB GPS receivers (tested with "Navilock NL-602U", "G-STAR IV, BU-353S4" 
  and "GPS/GLONASS U-blox7").
- File storage alternatives are; internally (on the SD card), USB memory or USB disc.
- Recording modes are full spectrum or time expanded "wav" files.
- Sound detection algorithm to avoid empty files. Buffer before the detected sound is always on.
- Time and position from attached GPS receiver is used in filenames and for the schedulers calculations.
- Scheduler for start and stop related to timestamps, sunset, dawn, etc. that are calculated from GPS time and position.
- Manually controlled via switches or a computer mouse.
- Log files for performed actions and errors.
- This version was configured via settings files stored on the USB memory stick.

## Added features in CloudedBats-WURB 2020:

Release v0.8:

- The detector acts as a WiFi Hotspot/AccessPoint and contains a web server. 
- Web pages are used for configuration and control. Adaptive layout for both smartphones and computers.
- Automatic detection of connected Ultrasonic microphones (valid for Pettersson and Dodotronic).
- Automatic detection of connected USB GPS Receiver.
- Sound peaks in kHz and dBFS are presented in live log and in file names.
- Automatic check where recorded files should be stored. On USB memory (multiple can be used) or 
  internally if no USB are found. Automatic failover to internal storage when USB memory is full.
- Possible to run the detectors user interface over internet, and to download files via SFTP.

Release v0.9:

- Reorganized user interface with a more intuitive detector mode selector.
- Audio feedback via the 3.5 mm audio jack for active monitoring with adjustable pitch level.
- New alternatives for GPS sources with fallback alternatives suitable for passive monitoring.
- Two configurable settings: "user default settings" and "start-up settings".
- Basic control via switches or a computer mouse (reintroduced, was missing in v0.8).
- Configurable host and port for internet traffic via environment variables.
- Possible to experiment and use other sound cards that are compatible with ALSA on Raspberry Pi.
  Environment variables are used to specify sound card name and sampling frequency.
- User manual for basic usage.

## Other characteristics

- The CloudedBats software is developed as open source software. Free to use under the terms of the MIT license.
- The software is developed on top of modern open source code libraries and frameworks.
- Inexpensive hardware can be used. Most of it is available where they are selling electronic things, except for the ultrasonic microphone.
- The Raspberry Pi computer offers many options for enhancements, like adding cameras and different types of sensors.
- Software developers can fork the code from GitHub to adjust the detector code to match their own needs.

## Drawbacks

- Higher power consumption compared to other passive monitoring systems. 
- You must put together the system yourself.
- The code is delivered "as-is" with no warranties at all. However, most software developers using Linux / Python 
  should understand the code and system setup to correct errors if I can't help.

## User manuals:
- User manual for basic usage:
  https://github.com/cloudedbats/cloudedbats_wurb_2020/blob/master/docs/user_manual_basic.md
- The user manual for advanced usage is not written yet. Please contact me if there are questions.

## Software notes, for developers:

This is valid for the CloudedBats-WURB 2020 version.

- Software completely rewritten. Python 3.7+ needed.
- Raspbian Buster Lite used for Raspberry Pi 4 support.
- RaspAP (https://raspap.com/) used to run the detector as a WiFi Hotspot/AccessPoint.
- The software design is inspired by micro services.
- The REST API is specified as OpenAPI 3.0.
- Asynchronous (async/await) programming both in frontend and backend.
- WebSockets are used for fast client updates.
- Stylesheets for responsive web pages are based on Bulma (https://bulma.io).
- FastAPI and uvicorn (https://www.uvicorn.org) are used to run the asynchronous web services.
- Pythons new asyncio library is used to replace all threads in the backend services. 
- Sounddevice and pyaudio have previously been used for communication with audio devices, but
  they are now replaced by alsaaudio (https://larsimmisch.github.io/pyalsaaudio/).

## Hardware

Any Raspberry Pi model with WiFi. (RPi Zero W works if auto detection and 
audio feedback is turned off). Raspberry Pi 4 is recommended if you want high speed transfer
of recorded files over internet.

Most SD cards seems to work but faster ones are recommended.
Personally I mostly use U3 (UHS Speed Class U3), for example Toshiba Exceria Pro.

**Note:** For some strange reason both M500 and M500-384 have problems if connected directly to a 
Raspberry Pi 4 at startup (RPi3B+ works fine). Workarounds are to use an extra USB 2.0 Hub (passive is ok), 
or attach the M500 or M500-384 microphone after startup.

## Installation

Note: This installation guide is updated to match the Ansible
installation guide that is used for more automated installations.
If you are familiar with Ansible, then please have a look here:
https://github.com/cloudedbats/ansible-playbooks

### Main workflow, overview

The main steps during the installation are:

- Install the Raspberry Pi Operating system on an SD card, with some basic configurations.
- Move the SD card to a Raspberry Pi computer and power it up.
- The Raspberry Pi should now be connected to your local WiFi network, or via an optional Ethernet cable.
- Login to the Raspberry Pi with SSH by using a terminal window.
- Make some basic configurations.
- Install needed Linux modules.
- Install the CloudedBats WURB-2020 software.
- Restart the Raspberry Pi.
- Add extra optional features like RaspAP, etc.

### Install the Raspberry Pi OS

Use the **Raspberry Pi Imager** to install the Raspberry Pi operating system.

- Download and install the Raspberry Pi Imager from here:
https://www.raspberrypi.com/software/
- Start the Raspberry Pi Imager.
- Select the operating system **"Raspberry Pi OS Lite"**, use the 32-bit version. (The "Desktop" version will not work, it must be the **"Lite"** version.)
- Attach an SD card and select it in the Raspberry Pi Imager.
- Click the “settings” button (the cogwheel).

Then make these setting, to be used as an example. It will also work with an
Ethernet cable connected to the Raspberry Pi, and then the WiFi (wireless LAN) part can be omitted.

Note that the username must be **pi** in this version of the WURB bat detector.

- Hostname: wurb99
- Enable SSH
- Use password authentication
- Username: pi
- Password: my-wurb99-password
- Configure wireless LAN
  - SSID: my-home-wifi
  - Password: my-home-wifi-password
  - Wireless LAN country: SE
- Locale settings
  - Time zone: Europe/Stockholm
  - Keyboard layout: se

Finally write to the SD card. When finished remove the SD card.

If you have many detectors, then be sure that you use different hostnames.
Otherwise there will be problems when they are connected to the same network
if more than one is identified as, for example, "wurb99.local".

### Start the Raspberry Pi

Insert the SD card into the Raspberry Pi and power it up.

The Raspberry Pi should now be connected to your local network, either via 
WiFi or an Ethernet cable depending on your configuration above.

Start a terminal window and connect with SSH.
For Windows users Putty is a well known alternative.

    ssh pi@wurb99.local

It is also possible to attach a screen and keyboard/mouse directly to
the Raspberry Pi instead of using ssh. 
No graphical user interface will be available, only command line mode.

### Basic configurations

Update the Raspberry Pi software.

    sudo apt update
    sudo apt upgrade

Make some configurations.
Most of them are already made when running
Raspberry Pi Imager above, but the last two must
be done here. Later modification can be done
whenever you want by running raspi-config.

    sudo raspi-config

Check or change this (example for Swedish users):

- System Options - Hostname: wurb99 
- System Options - Password: my-wurb99-password
- Localization Options - Timezone: Europe - Stockholm

Mandatory:

- System Options - Network at boot: No
- Advanced Options - Expand Filesystem.

A reboot is needed to expand the file system.

    sudo reboot

Hint: WiFi networks can be changed via raspi-config.
Another alternative is to edit the wpa_supplicant.conf file
and change SSID and password for the network.
Even a list of networks can be added in this file if the
detector is used at different places.

    sudo nano /etc/wpa_supplicant/wpa_supplicant.conf

### Install Linux packages

Install needed software packages.

    sudo apt install git python3-venv python3-dev libatlas-base-dev udevil

To make it work with the "Bookworm" release of the Raspberry Pi OS these 
installations must be done. Note that "numpy" and "scipy" now is removed from
the "requirements.txt" file used by "pip install". The command for creating
virtual environments for python is also modified to accept numpy and 
scipy when they are not created inside the virtual environment. 
The package "pyalsaaudio" is not available in the 64 bits version, 
and we have to stay with the 32 bits version of the operating system.

    sudo apt install pulseaudio python3-numpy python3-scipy

### Pettersson M500 (500kHz)

The Pettersson M500 microphone (the 500kHz version) differ in communication
format compared to M500-384, u256, and u384. Some udev rules must be setup
to allow the detector to communicate with it.

    # Create a file with udev rules:
    sudo nano /etc/udev/rules.d/pettersson_m500_batmic.rules
    
    # Add this row in the new file. 
    SUBSYSTEM=="usb", ENV{DEVTYPE}=="usb_device", MODE="0664", GROUP="pi"

### Install the CloudedBats WURB-2020 software

Clone the software from the GitHub repository and install Python related packages.

    git clone https://github.com/cloudedbats/cloudedbats_wurb_2020.git
    cd cloudedbats_wurb_2020/
    python -m venv --system-site-packages venv
    source venv/bin/activate
    pip install -r requirements.txt 

Note: When installing packages most of them are pre-prepared in something
called "wheels".
The main source for wheels for Raspberry Pi can be found here,
with an example for SciPy:
https://piwheels.org/project/scipy/
If that build has failed, then it is possible to use an older version.
The installation process may find a working version automatically with a
lot of warnings during the installation process.
Then it is possible to edit the requirements.txt file to speed up the
installation process. In the SciPy case the row can be changed
from "scipy" to "scipy<1.9.0" if there are problems with version 1.9.0.

### Run the detector software as a service

Install the service and enable it. Enable means that the service will
start automatically when the Raspberry Pi is started.

    sudo cp wurb_2020.service /etc/systemd/system/wurb_2020.service
    sudo systemctl daemon-reload
    sudo systemctl enable wurb_2020.service
    sudo systemctl start wurb_2020.service

### Adjust sound level

Set the headphone volume used for audio feedback.

    amixer set 'PCM' 100%
    # Other useful commands if there are problems:
    aplay -L
    amixer scontrols
    alsamixer -c0

### And finally, restart the detector

    sudo reboot

### RaspAP, optional installation

If you want to use the detector away from computer networks, then
the RaspAP software can help. Read more here: https://raspap.com

Note that the built-in WiFi module will be used by RaspAP to provide a WiFi
access point. That means that the previously used WiFi connection to
the local wireless network will not work when RaspAP is installed.
When internet access is needed you have to connect an Ethernet cable.
Another option is to use a 4G/LTE modem, like the HUAWEI E3372, and then
the detector also can be used as a WiFi access point for internet access.

Download and install RaspAp.

    curl -sL https://install.raspap.com | bash

    # It is ok to select "Y" on all steps during the installation.
    # OpenVPN and Wireguard are not used, you can press "N" for them.

Restart the detector.

    sudo reboot

Connect to the wifi network named raspi-webgui. Password: ChangeMe.

Start a web browser and go to: http://10.3.141.1.

If the detector is using an Ethernet cable the web address "http://wurb99.local/"
will work from any computer in the network (if the hostname is wurb99).

Login for the administration page: Username: **admin**, password: **secret**.

Change the defaults settings from:

- Hotspot-Basic-SSID (wifi name): raspi-webgui
- Hotspot-Security-PSK (wifi password): ChangeMe
- Authentication-Username (admin username): admin
- Authentication-Password (admin password): secret

Change to (example for Swedish users):

- Hotspot-Basic-SSID (wifi name): wifi4bats
- Hotspot-Security-PSK (wifi password): chiroptera
- Authentication-Username (admin username): batman
- Authentication-Password (admin password): chiroptera
- Hotspot-Advanced-Country Code: Sweden

## Run the detector

- Connect the ultrasonic microphone to the detector (can also be done after startup).
- Connect USB memory stick(s) (optional, or can be done after startup).
- Start the Raspberry Pi. It will start automatically when power is added.
- Connect a computer or mobile phone to the WiFi network called "wifi4bats". Password: chiroptera.
- Open a web browser and go to http://10.3.141.1:8000 (http://wurb99:8000 will work if not using RaspAP.)
- Check "Geographic location" and "Settings".
- Press "Set time".
- Select "Mode", for example "Recording - Auto detection".
- Connect headphones (3.5 mm jacket on the Raspberry Pi) and listen to the bats.
- Record some bats.
- Press "Stop recording".

There are different ways to turn the Raspberry Pi detector off (power off only is not
recommended since there is a small risk of corrupted SD card or USB memory sticks):

- Select "Mode": "Detector - Power off..." and press the "Shutdown" button.
- Go to the WiFi administration page at http://10.3.141.1. Select "System" and press "Shutdown".
- Attach an USB Computer mouse and hold down both the left end right button for 5 sec.
- Connect a physical button to GPIO pin #37 and pin #39 (but be careful, connecting the wrong GPIO 
  pins can destroy your Raspberry Pi). Then press that button for a few seconds.

More than one USB memory stick can be used. They will be filled up with wave files
in alphabetic order. When the last USB memory stick is full, then it will continue
on the SD card. It will stop when there is less than 0.5 GB left on the SD card.

If files are stored on the SD card then they can be downloaded and/or removed by
using for example FileZilla. Connect with SFTP to http://10.3.141.1 
with user "pi" and password "chiroptera".
Path to files on the SD card: "/home/pi/wurb_recordings".

It is also possible to download files by using FileZilla, or similar, from the USB
memory stick during an ongoing recording session. The path to the USB memory is then
"/media/pi/<your-usb-memory-stick-name>".

Note: Some mobile phones complains if the "wifi4bats" network not is
connected to internet. If that happens, then tell it to forget the network,
connect to i again and answer the question if you want to connect to it
anyway.

## The MIT license

This software is released under the MIT license that means that you are free to use it as you want;
use it, share it, cut it into pieces, extend it, or even sell it for money.
But you are not allowed to remove the MIT license clause
(just change my name to yours in the copyright row for modified code files),
and it comes "as-is" with no warranties at all.

## Contact

Arnold Andreasson, Sweden.

info@cloudedbats.org
 readmeEtag: '"55170a91d0eadc6be1eba1b1416e91cb13990471"' readmeLastModified: Mon, 23 Oct 2023 19:21:41 GMT repositoryId: 247984856 description: CloudedBats-WURB. The DIY bat detector. created: '2020-03-17T14:03:00Z' updated: '2026-01-22T16:38:38Z' language: Python archived: false stars: 24 watchers: 4 forks: 3 owner: cloudedbats logo: https://avatars.githubusercontent.com/u/16818499?v=4 license: MIT repoEtag: '"cd4bb2d66a0793d4072f1d20cbcc4992ef621a7db80a1fbfc380a5225fe46b04"' repoLastModified: Thu, 22 Jan 2026 16:38:38 GMT foundInMaster: true category: Server Implementations id: 2acd0e2152c96fbe7f024c2c46ddd40c - source: openapi3 tags repository: https://github.com/httptoolkit/openapi-directory-js v3: true repositoryMetadata: base64Readme: >- IyBBbiBPcGVuQVBJIERpcmVjdG9yeSBmb3IgSlMgWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vaHR0cHRvb2xraXQvb3BlbmFwaS1kaXJlY3RvcnktanMvd29ya2Zsb3dzL0NJL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9odHRwdG9vbGtpdC9vcGVuYXBpLWRpcmVjdG9yeS1qcy9hY3Rpb25zKSBbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL29wZW5hcGktZGlyZWN0b3J5LnN2ZyldKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9vcGVuYXBpLWRpcmVjdG9yeSkKCj4gX1BhcnQgb2YgW0hUVFAgVG9vbGtpdF0oaHR0cHM6Ly9odHRwdG9vbGtpdC5jb20pOiBwb3dlcmZ1bCB0b29scyBmb3IgYnVpbGRpbmcsIHRlc3RpbmcgJiBkZWJ1Z2dpbmcgSFRUUChTKV8KClRoaXMgcmVwbyBidWlsZHMgJiBidW5kbGVzIHRoZSBbT3BlbkFQSSBEaXJlY3RvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9BUElzLWd1cnUvb3BlbmFwaS1kaXJlY3RvcnkpLCBzbyB5b3UgY2FuIGVhc2lseSBmaW5kLCByZXF1aXJlIGFuZCB1c2UgYW55IE9wZW5BUEkgc3BlYyBmcm9tIHRoZSBkaXJlY3RvcnkgaW4geW91ciBKUyBwcm9qZWN0cy4KCkl0IHByb3ZpZGVzIGZpbGVzIHRoYXQgY2FuIGJlIGluZGl2aWR1YWxseSByZXF1aXJlZCBvciByZW1vdGVseSBkb3dubG9hZGVkICh2aWEgaHR0cHM6Ly91bnBrZy5jb20vb3BlbmFwaS1kaXJlY3RvcnkvKSBmb3IgZXZlcnkgQVBJIGluIHRoZSBjb2xsZWN0aW9uLCBhbmQgYW4gaW5kZXggdG8gcXVpY2tseSBmaW5kIHRoZSByZWxldmFudCBPcGVuQVBJIHNwZWMgZm9yIGEgZ2l2ZW4gVVJMLgoKQWxsIHNwZWNzIGFyZToKCiogUHJlLXBhcnNlZCBhbmQgZXhwb3NlZCBhcyBKYXZhU2NyaXB0IG9iamVjdHMgKG5vdCBZQU1MIHN0cmluZ3MpLgoqIENvbnZlcnRlZCB0byBPcGVuQVBJIHYzLgoqIFByZS1idW5kbGVkIHdpdGggYWxsIGV4dGVybmFsICRyZWZzLgoKVGhhdCBtZWFucyB5b3UgY2FuIGltcG9ydCB0aGVtLCBhbmQgaW1tZWRpYXRlbHkgJiBjb25zaXN0ZW50bHkgc3RhcnQgdXNpbmcgdGhlbS4KCiMjIEhvdyB0byB1c2UgaXQKCkZpcnN0IHVwLCBpbnN0YWxsIGl0IHdpdGg6CgpgYGBiYXNoCm5wbSBpbnN0YWxsIG9wZW5hcGktZGlyZWN0b3J5CmBgYAoKQWxsIE9wZW5BUEkgc3BlY3MgY2FuIGJlIG5vdyByZXF1aXJlZCB3aXRoOgoKYGBganMKY29uc3Qgc3BlYyA9IHJlcXVpcmUoJ29wZW5hcGktZGlyZWN0b3J5L2FwaS88c3BlYy1pZD4uanNvbicpOwpgYGAKCihvciByZWFkIGZyb20gYGh0dHBzOi8vdW5wa2cuY29tL29wZW5hcGktZGlyZWN0b3J5L2FwaS88c3BlYy1pZD4uanNvbmApCgpUaGUgZWFzaWVzdCB3YXkgdG8gb2J0YWluIGEgc3BlYyBpZCBpcyB0byB1c2UgdGhlIGluZGV4LiBZb3UgY2FuIGxvb2sgdXAgYSBVUkwgaW4gdGhlIGluZGV4IHdpdGg6CgpgYGBqcwpjb25zdCB7IGZpbmRBcGkgfSA9IHJlcXVpcmUoJ29wZW5hcGktZGlyZWN0b3J5Jyk7CgpmaW5kQXBpKCd3aWtpbWVkaWEub3JnL2FwaS9yZXN0X3YxL2ZlZWQvYXZhaWxhYmlsaXR5Jyk7CmBgYAoKYGZpbmRBcGlgIHRha2VzIGEgVVJMIChob3N0IGFuZCBwYXRoLCBfd2l0aG91dF8gdGhlIHByb3RvY29sKSB3aXRoaW4gYW55IEFQSSwgYW5kIHdpbGwgcmV0dXJuIGVpdGhlcjoKCiogVW5kZWZpbmVkLCBpZiB0aGVyZSBpcyBubyBtYXRjaGluZyBBUElzLgoqIEEgc3RyaW5nIHNwZWMgaWQsIGlmIHRoZXJlIGlzIGV4YWN0bHkgb25lIEFQSSB0aGF0J3MgcmVsZXZhbnQgdG8gdGhhdCBVUkwuCiogQSBsaXN0IG9mIHNwZWMgaWRzLCBpbiByYXJlIGNhc2VzIHdoZXJlIG11bHRpcGxlIHNwZWNzIG1heSBjb3ZlciB0aGUgc2FtZSBVUkwuCgpBbHRlcm5hdGl2ZWx5IGlmIHlvdSBrbm93IGluIGFkdmFuY2Ugd2hpY2ggc3BlYyB5b3Ugd2FudCB5b3UgY2FuIHJlcXVpcmUgaXQgZGlyZWN0bHkuIFRoZSBpZCBmb3IgZXZlcnkgc3BlYyBpbiB0aGUgZGlyZWN0b3J5IGlzIG1hZGUgdXAgb2YgdGhlIHByb3ZpZGVyIG5hbWUsIGZvbGxvd2VkIGJ5IGEgc2xhc2ggYW5kIHRoZSBzZXJ2aWNlIG5hbWUgaWYgYSBzZXJ2aWNlIG5hbWUgZXhpc3RzLiBTb21lIGV4YW1wbGUgaWRzOgoKKiBgeGtjZC5jb21gIChwcm92aWRlciBpcyB4a2NkLmNvbSwgbm8gc2VydmljZSBuYW1lKQoqIGBhbWF6b25hd3MuY29tL2FjbWAgKHByb3ZpZGVyIGlzIGFtYXpvbmF3cy5jb20sIHNlcnZpY2UgbmFtZSBpcyBhY20pLgoKWW91IGNhbiBmaW5kIHRoZSBwcm92aWRlciBhbmQgc2VydmljZSBuYW1lIGluIHRoZSBzcGVjIGl0c2VsZiAodW5kZXIgYGluZm9gLCBgeC1wcm92aWRlck5hbWVgIGFuZCBgeC1zZXJ2aWNlTmFtZWApLCBhbmQgeW91IGNhbiBicm93c2UgdGhlIHJhdyBzcGVjcyBkaXJlY3RseSBhdCBodHRwczovL2dpdGh1Yi5jb20vQVBJcy1ndXJ1L29wZW5hcGktZGlyZWN0b3J5LgoKIyMgTGljZW5zZQoKVGhpcyByZXBvL25wbSBtb2R1bGUgaXMgbGljZW5zZWQgYXMgTUlULgoKVGhlIGxpY2Vuc2UgZm9yIEFQSSBkZWZpbml0aW9ucyB2YXJpZXMgYnkgc3BlYywgc2VlIGh0dHBzOi8vZ2l0aHViLmNvbS9BUElzLWd1cnUvb3BlbmFwaS1kaXJlY3RvcnkjbGljZW5zZXMgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgpJbiBnZW5lcmFsIGl0J3MgdmVyeSBsaWtlbHkgdGhhdCB5b3VyIHVzZSBvZiBhbnkgQVBJIGRlZmluaXRpb24gaXMgY292ZXJlZCBlaXRoZXIgYnkgQ0MwIChmb3Igc3BlY3Mgc3VibWl0dGVkIGRpcmVjdGx5IHRvIHRoZSBkaXJlY3RvcnkpLCB0aGUgc3BlYydzIG93biBsaWNlbnNlIChjaGVjayBgaW5mby5saWNlbnNlYCkgb3IgYnkgRmFpciBVc2UgcHJvdmlzaW9ucyB3aGVuIGNvbW11bmljYXRpbmcgd2l0aCB0aGUgY29ycmVzcG9uZGluZyBzZXJ2aWNlLiBUaGlzIGlzIG5vdCBmb3JtYWwgbGVnYWwgYWR2aWNlIHRob3VnaCwgaXRzIHlvdXIgcmVzcG9uc2liaWxpdHkgdG8gY29uZmlybSB0aGlzIGZvciB5b3Vyc2VsZiBmb3IgdGhlIHNwZWNzIHlvdSdyZSB1c2luZy4= readmeEtag: '"5124ae309ffc8962b2afc7add3307aab28133a19"' readmeLastModified: Thu, 15 Aug 2024 13:41:53 GMT repositoryId: 170318480 description: >- Building & bundling https://github.com/APIs-guru/openapi-directory for easy use from JS created: '2019-02-12T12:59:30Z' updated: '2026-01-14T19:39:20Z' language: TypeScript archived: false stars: 26 watchers: 2 forks: 6 owner: httptoolkit logo: https://avatars.githubusercontent.com/u/39777515?v=4 license: MIT repoEtag: '"3badaca7c181b85bcb5ddefb4ddd75705c0e149801c0d03f8195e0dc858328d8"' repoLastModified: Wed, 14 Jan 2026 19:39:20 GMT foundInMaster: true category: - Testing - Parsers id: 0b4f0ec3b8efec08b4c9de26e8119588 - source: openapi3 tags repository: https://github.com/lornajane/openapi-overlays-js v3: true id: 23a1b93a676dfb84bfc1d9adb30b4875 repositoryMetadata: base64Readme: >- IyBPdmVybGF5cyBmb3IgT3BlbkFQSQoKVGhpcyBOb2RlSlMgbGlicmFyeSBpcyBhbiBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgaW4tcHJvZ3Jlc3MgW09wZW5BUEkgT3ZlcmxheXMgU3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PdmVybGF5LVNwZWNpZmljYXRpb24vYmxvYi9tYWluL3ZlcnNpb25zLzEuMC4wLm1kKS4KCiMjIEFib3V0IE92ZXJsYXlzCgpPdmVybGF5cyBhcmUgYSB3YXkgdG8gZXh0ZW5kIG9yIGVuaGFuY2UgYW4gZXhpc3RpbmcgW09wZW5BUEkgZGVzY3JpcHRpb25dKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIGJ5IGFkZGluZywgdXBkYXRpbmcgb3IgcmVtb3ZpbmcgZmllbGRzLiBGb3IgZXhhbXBsZToKCiogVGhlIE9wZW5BUEkgZGVzY3JpcHRpb24gbGl2ZXMgaW4gdGhlIGNvZGViYXNlLCBidXQgeW91ciB0ZWNoIHdyaXRlcnMgbmVlZCB0byBhZGQgZXhhbXBsZXMgYW5kIGluZm9ybWF0aW9uIGJlZm9yZSBkb2N1bWVudGF0aW9uIGlzIHByb2R1Y2VkLgoqIFlvdXIgb3JnYW5pc2F0aW9uIHVzZXMgT3BlbkFQSSBidXQgaXQncyBnZW5lcmF0ZWQgZnJvbSBjb2RlLCBhbmQgbmVlZHMgYW1lbmRtZW50cyBvciBhZGRpdGlvbnMgdG8gYmUgdXNlZnVsIHRvIHVzZXJzLgoqIFlvdSdyZSB1c2luZyBvdGhlciB0b29scyBzdWNoIGFzIEFQSSBjbGllbnQgZ2VuZXJhdG9ycywgdGhhdCBjYW4gbWFrZSB1c2Ugb2YgYWRkaXRpb25hbCBjb250ZXh0IGFkZGVkIHRvIHRoZSBBUEkgZGVzY3JpcHRpb24sIHN1Y2ggYXMgdGFncyB0byByZW1vdmUgY2VydGFpbiBlbmRwb2ludHMsIHdoaWNoIGFyZW4ndCBpbiB0aGUgbWFpbiBBUEkgZGVzY3JpcHRpb24gZmlsZS4KCioqQnkgdXNpbmcgT3ZlcmxheXMgdG8gZGVzY3JpYmUgYW5kIGFwcGx5IHRoZSBjaGFuZ2VzLCB3aGVuIHRoZSBBUEkgZGVzY3JpcHRpb24gaXMgdXBkYXRlZCwgdGhlIHNhbWUgY2hhbmdlcyBjYW4gaW5zdGFudGx5IGJlIHJlLWFwcGxpZWQuKioKCiMjIEluc3RhbGxhdGlvbgoKSW5zdGFsbCB0aGUgYG92ZXJsYXlqc2AgY29tbWFuZCBmcm9tIFtucG1dKGh0dHBzOi8vbnBtanMuY29tKSB3aXRoIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYHRleHQKbnBtIGluc3RhbGwgLWcgb3BlbmFwaS1vdmVybGF5cy1qcwpgYGAKClRoZSBgLWdgIHN3aXRjaCBpbnN0YWxscyBpdCBnbG9iYWxseSBzbyB5b3UgY2FuIHVzZSB0aGUgY29tbWFuZCBmcm9tIGFueXdoZXJlIG9uIHlvdXIgc3lzdGVtLgoKPiAqKlByby10aXA6KiogdXNlIGBub2RlIGluZGV4LmpzYCBmcm9tIHRoZSByb290IG9mIHRoZSBwcm9qZWN0IGlmIHlvdSdyZSB1c2luZyB0aGUgYmxlZWRpbmcgZWRnZSBvZiB0aGUgcHJvamVjdCBvciB3b3JraW5nIG9uIGEgYnJhbmNoLgoKIyMgVXNhZ2UKCkV4YW1wbGU6IGBvdmVybGF5anMgLS1vcGVuYXBpIG9wZW5hcGkueW1sIC0tb3ZlcmxheSBhZGQtc3BhcmtsZS55YW1sYAoKVXNlIGBvdmVybGF5anMgLS1oZWxwYCB0byBzZWUgdGhlIHVzYWdlIGluZm9ybWF0aW9uLgoKU2VlIFtkb2NzL2V4YW1wbGVzXShodHRwczovL2dpdGh1Yi5jb20vbG9ybmFqYW5lL29wZW5hcGktb3ZlcmxheXMtanMvYmxvYi9tYWluL2RvY3MvZXhhbXBsZXMvaW5kZXgubWQpIGZvciBtb3JlIGV4YW1wbGVzLgoKIyMgUHJvamVjdCBzdGF0dXMKCjp3YXJuaW5nOiBQcm9qZWN0IHN0YXR1czogYWxwaGEKClRoaXMgcHJvamVjdCBpcyB2ZXJ5IG11Y2ggYXQgYW4gYWxwaGEgc3RhZ2UgYnV0IGZlZWRiYWNrLCBjb21tZW50cyBhbmQgcXVlc3Rpb25zIGFyZSBhbGwgdmVyeSB3ZWxjb21lIGFzIGlzc3VlcyBvbiB0aGlzIHJlcG9zaXRvcnkgLSBhbmQgcHVsbCByZXF1ZXN0cyBhcmUgYWxzbyBncmF0ZWZ1bGx5IHJlY2VpdmVkLiBJdCB3b3VsZCBiZSBleGNlbGxlbnQgdG8gaGVhciB3aGF0IHByb2JsZW0gdGhpcyB0b29sIGNhbiBzb2x2ZSwgYW5kIGhvdyB5b3UgZ2V0IG9uIHdpdGggaXQuIEJvbnVzIHBvaW50cyBpZiB5b3UgY2FuIHNoYXJlIGV4YW1wbGVzICh3b3JraW5nIG9yIGJyb2tlbikgb2Ygd2hhdCB5b3UgdHJpZWQuCg== readmeEtag: '"ce407838371cebf7ca08a8e42486a447ce1acf30"' readmeLastModified: Tue, 20 May 2025 19:39:48 GMT repositoryId: 591983470 description: Apply overlays to OpenAPI descriptions created: '2023-01-22T15:11:54Z' updated: '2026-01-27T18:39:10Z' language: JavaScript archived: false stars: 26 watchers: 3 forks: 7 owner: lornajane logo: https://avatars.githubusercontent.com/u/172607?v=4 license: Apache-2.0 repoEtag: '"cd11161a36dcb9ddf3f66c7fa5d290845e2f56af3b52b43039202de0c97b19b1"' repoLastModified: Tue, 27 Jan 2026 18:39:10 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mikekonan/go-oas3 v3: true repositoryMetadata: base64Readme: >- # Go OpenAPI v3 Server Code Generator

A powerful, modern code generator that creates clean server boilerplate from OpenAPI v3 specifications. Built with Go 1.22+ and the latest dependencies, featuring enhanced error handling and extensive customization options.

## Table of Contents

- [Quick Start](#quick-start)
- [Key Ideas](#key-ideas)
- [Installation](#installation)
- [Usage](#usage)
- [Complete Workflow Example](#complete-workflow-example)
- [OpenAPI Features](#openapi-features)
- [Extensions Reference](#extensions-reference)
- [Limitations](#limitations)
- [Questions or Feature Requests](#questions-or-feature-requests)
- [Contributing](#contributing)

## Quick Start

### Prerequisites
- **Go 1.22+** (updated for latest language features)
- Valid OpenAPI 3.0+ specification file
- Basic understanding of Go modules

### Installation

```bash
go install github.com/mikekonan/go-oas3@latest
```

### Basic Usage

```bash
# Generate from local file
go-oas3 -swagger-addr swagger.yaml -package myapi -path ./generated

# Generate from remote URL
go-oas3 -swagger-addr https://example.com/api/swagger.yaml -package myapi -path ./generated
```

## Key Ideas

- **Request Parsing**: Stubs handle all request parsing logic automatically
- **Type Safety**: Response builders ensure you can only respond according to your specification  
- **Validation**: Built-in validation for all request parameters and bodies
- **Security**: Automatic security middleware generation from OpenAPI security schemes

**Note:** Path stub generation relies on the **first tag** from your paths. While tags are not required, they are **strongly recommended** for better organization:
- **With tags**: Creates separate service interfaces per tag (e.g., `UserService`, `OrderService`)
- **Without tags**: All operations are grouped under a single `DefaultService` interface

Example with tags:
```yaml
paths:
  /users:
    get:
      tags: [users]  # Creates UserService interface
  /orders:
    post:
      tags: [orders]  # Creates OrderService interface
```
## Usage

### Command Line Arguments

| Flag | Type | Description | Default |
|------|------|-------------|---------|
| `-swagger-addr` | string | Path or URL to OpenAPI specification | `swagger.yaml` |
| `-package` | string | **Required.** Go package name for generated code | - |
| `-path` | string | **Required.** Output directory for generated files | - |
| `-componentsPackage` | string | Package name for components (if different from main) | Same as `-package` |
| `-componentsPath` | string | Path for components (if different from main) | Same as `-path` |
| `-authorization` | string | Headers for remote swagger files (`key1:value1,key2:value2`) | - |
| `-prioritize-x-go-type` | bool | Prioritize `x-go-type` over schema properties | `false` |
| `-pass-raw-request` | bool | Pass raw HTTP request to handler functions | `false` |

### Examples

```bash
# Basic local generation
go-oas3 -swagger-addr swagger.yaml -package api -path ./generated

# Remote file with custom components
go-oas3 \
  -swagger-addr https://petstore3.swagger.io/api/v3/openapi.json \
  -package petstore \
  -path ./api \
  -componentsPackage models \
  -componentsPath ./api/models

# With authorization headers
go-oas3 \
  -swagger-addr https://api.example.com/swagger.yaml \
  -package myapi \
  -path ./generated \
  -authorization "X-API-Key:secret,Authorization:Bearer token"
```

## Complete Workflow Example

Here's a complete example from an OpenAPI spec to a running server:

### 1. Create OpenAPI Specification (`api.yaml`)
```yaml
openapi: 3.0.0
info:
  title: User API
  version: 1.0.0
paths:
  /users/{id}:
    get:
      tags: [users]
      parameters:
        - name: id
          in: path
          required: true
          schema:
            type: string
            x-go-type: github.com/google/uuid.UUID
      responses:
        '200':
          description: User found
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/User'
        '404':
          description: User not found
components:
  schemas:
    User:
      type: object
      required: [id, email]
      properties:
        id:
          type: string
          format: uuid
        email:
          type: string
          format: email
        name:
          type: string
          x-go-omitempty: true
```

### 2. Generate Code
```bash
go-oas3 -swagger-addr api.yaml -package userapi -path ./generated
```

### 3. Implement Handlers
```go
package main

import (
    "context"
    "net/http"
    "github.com/go-chi/chi/v5"
    userapi "./generated"
)

type UsersService struct{}

func (s *UsersService) GetUsersId(ctx context.Context, request userapi.GetUsersIdRequestObject) userapi.GetUsersIdResponseObject {
    // Your business logic here
    user := userapi.User{
        Id:    request.Id,
        Email: "user@example.com",
        Name:  "John Doe",
    }
    
    return userapi.GetUsersId200JSONResponse{
        Body: user,
    }
}

func main() {
    service := &UsersService{}
    
    r := chi.NewRouter()
    userapi.HandlerFromMux(service, r)
    
    http.ListenAndServe(":8080", r)
}
```

### 4. Set Up Your Project
```bash
# Initialize Go module
go mod init myproject

# Add dependencies and run
go mod tidy
go run main.go
```

### 5. Test Your API
```bash
# Test the endpoint
curl http://localhost:8080/users/550e8400-e29b-41d4-a716-446655440000

# Response:
# {
#   "id": "550e8400-e29b-41d4-a716-446655440000",
#   "email": "user@example.com", 
#   "name": "John Doe"
# }
```

The generated boilerplate includes:
- **Type-safe handlers** - Request/response objects with proper Go types
- **Automatic validation** - Built-in validation for all parameters and request bodies
- **Router integration** - Ready-to-use chi router setup
- **Error handling** - Structured error responses matching your OpenAPI spec

## OpenAPI Features

### Required Fields
Required fields in path, query, headers, and components are supported.

### Security
Security schemes for HTTP and API key (header/cookie) are supported.

### Cookie
Response header `Set-Cookie` is supported. Cookies in requests are supported via security schemes only.

### Validation
Type validation supports the following data types:
- **string**: minLength, maxLength
- **number**, **integer**: minimum, maximum, exclusiveMinimum, exclusiveMaximum

### Custom Types
The generator supports several OpenAPI types for components:

| OpenAPI Type | Go Type |
|---|---|
| uuid | [github.com/google/uuid.UUID](https://github.com/google/uuid) |
| iso4217-currency-code | [github.com/mikekonan/go-types/v2/currency.Code](https://github.com/mikekonan/go-types) |
| iso3166-alpha-2 | [github.com/mikekonan/go-types/v2/country.Alpha2Code](https://github.com/mikekonan/go-types) |
| iso3166-alpha-3 | [github.com/mikekonan/go-types/v2/country.Alpha3Code](https://github.com/mikekonan/go-types) |

## Extensions Reference

The generator supports powerful OpenAPI extensions to customize Go code generation:

### Core Type Extensions

#### `x-go-type` - Custom Go Types
Specify custom Go types for any schema:

```yaml
# Use encoding/json.RawMessage for flexible JSON
metadata:
  type: object
  x-go-type: encoding/json.RawMessage

# Use third-party types
user_id:
  type: string
  x-go-type: github.com/google/uuid.UUID

# Use custom domain types
amount:
  type: string
  x-go-type: github.com/shopspring/decimal.Decimal
```

#### `x-go-type-string-parse` - Custom Parsing
For string parameters that need custom parsing:

```yaml
# Custom UUID parsing from string
user_id:
  type: string
  x-go-type: github.com/google/uuid.UUID
  x-go-type-string-parse: github.com/google/uuid.Parse

# Custom time parsing
created_at:
  type: string
  x-go-type: time.Time
  x-go-type-string-parse: github.com/spf13/cast.ToTimeE
```

### Field Modifiers

#### `x-go-pointer` - Force Pointer Types
```yaml
# Make field a pointer (useful for optional fields)
optional_amount:
  type: integer
  x-go-pointer: true
  # Generates: OptionalAmount *int `json:"optional_amount,omitempty"`
```

#### `x-go-omitempty` - JSON Omitempty Tag
```yaml
# Add omitempty to JSON tag
description:
  type: string
  x-go-omitempty: true
  # Generates: Description string `json:"description,omitempty"`
```

#### `x-go-string-trimmable` - Auto-trim Strings
```yaml
# Automatically trim whitespace before validation
title:
  type: string
  minLength: 1
  x-go-string-trimmable: true
  # Trims spaces before checking minLength
```

### Validation Extensions

#### `x-go-regex` - Regex Validation
```yaml
# Add regex validation to parameters
phone:
  type: string
  x-go-regex: ^\+?[1-9]\d{1,14}$
  # Generates validation code with regexp.MustCompile
```

#### `x-go-skip-validation` - Disable Validation
```yaml
# Skip validation for performance-critical paths
large_payload:
  type: object
  properties:
    data: 
      type: string
  x-go-skip-validation: true
```

### Security Extensions

#### `x-go-skip-security-check` - Skip Security Validation
```yaml
# For operations that parse auth but don't enforce it
paths:
  /health:
    get:
      x-go-skip-security-check: true
      security:
        - ApiKeyAuth: []
      # Parses auth header but doesn't fail on missing/invalid auth
```

### Advanced Map Types

#### `x-go-map-type` - Custom Map Types
```yaml
# Custom map with typed keys/values
metadata:
  type: object
  additionalProperties:
    type: string
  x-go-map-type: map[github.com/google/uuid.UUID]string
  # Generates: map[uuid.UUID]string instead of map[string]string
```

### Complete Extension Example

```yaml
components:
  schemas:
    User:
      type: object
      required: [id, email]
      properties:
        id:
          type: string
          x-go-type: github.com/google/uuid.UUID
        email:
          type: string
          format: email
          x-go-regex: ^[a-zA-Z0-9._%+-]+@[a-zA-Z0-9.-]+\.[a-zA-Z]{2,}$
        profile:
          type: object
          x-go-pointer: true
          x-go-omitempty: true
          properties:
            bio:
              type: string
              x-go-string-trimmable: true
        metadata:
          type: object
          additionalProperties:
            type: string
          x-go-type: map[string]interface{}
      x-go-skip-validation: false

  parameters:
    UserID:
      name: user_id
      in: path
      required: true
      schema:
        type: string
        x-go-type: github.com/google/uuid.UUID
        x-go-type-string-parse: github.com/google/uuid.Parse
```

## Limitations

### Known Limitations

#### Inline Schema Responses
The generator currently has limited support for inline schemas in responses. For example:

```yaml
# ❌ Not fully supported - may cause compilation issues
responses:
  '200':
    content:
      application/json:
        schema:
          type: object
          properties:
            message: 
              type: string

# ✅ Recommended - use $ref to components
responses:
  '200':
    content:
      application/json:
        schema:
          $ref: '#/components/schemas/SuccessResponse'
```

**Workaround**: Define all response schemas in `components/schemas` and reference them using `$ref`.

#### Anonymous Types
- Anonymous slice elements and map entries have limited support
- Complex nested anonymous types may not generate correctly

**Workaround**: Define explicit component schemas for complex types.

### Best Practices

1. **Always use $ref for schemas** - Avoid inline type definitions
2. **Define reusable components** - Create schemas in `components/schemas`
3. **Use meaningful names** - Component names become Go type names
4. **Test generated code** - Always compile and test after generation


## Questions or Feature Requests

Have a question or need some functionality? Feel free to open an issue or submit a pull request.

## Contributing

Go OpenAPI v3 server code generator uses [github.com/dave/jennifer](https://github.com/dave/jennifer) for code generation. 
Using [github.com/aloder/tojen](https://github.com/aloder/tojen) is the suggested way to generate Jennifer code.

 readmeEtag: '"b8182949e4b0c36cbae298126e2d5ba07374de57"' readmeLastModified: Thu, 06 Nov 2025 10:25:50 GMT repositoryId: 315667251 description: Open API v3 server code generator created: '2020-11-24T15:03:56Z' updated: '2026-01-29T11:12:28Z' language: Go archived: false stars: 21 watchers: 2 forks: 9 owner: mikekonan logo: https://avatars.githubusercontent.com/u/36304777?v=4 license: MIT repoEtag: '"b4f1324c65c12c5c2c457d5bc7822f85be1f3de3d9b4447812e3c6635cfa8b32"' repoLastModified: Thu, 29 Jan 2026 11:12:28 GMT foundInMaster: true category: Parsers id: 15bb9572a91297aa6595fb0450194d91 - source: openapi3 tags repository: https://github.com/uderline/openapi-php-attributes v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFBIUCBBdHRyaWJ1dGVzIEdlbmVyYXRvcgoKVGhpcyBDTEkgVG9vbCBpcyBhYmxlIHRvIGdlbmVyYXRlIGFuIE9wZW5BcGkgSlNPTiBmaWxlIGRlc2NyaXB0aW9uIGFjY29yZGluZyB0byBQSFAgYXR0cmlidXRlcyBjb250YWluZWQgaW4geW91ciBmaWxlcy4KCiMjIOKaoO+4jyBNaXNzaW5nIHNvbWV0aGluZyA/Ckp1c3Qgb3BlbiBhbiBpc3N1ZSBzYXlpbmcgd2hhdCdzIG1pc3NpbmcgISBGZWVsIGZyZWUgdG8gb3BlbiBhIFBSIGJ1dCB3ZSByZWNvbW1lbmQgb3BlbmluZyBhbiBpc3N1ZSBiZWZvcmVoYW5kLgoKIyMgV2hlcmUgdG8gc3RhcnQgPwoxIC0gSW5zdGFsbCB0aGUgcGFja2FnZSBvcGVuYXBpLXBocC1hdHRyaWJ1dGVzLWdlbmVyYXRvciB3aXRoIGNvbXBvc2VyLgoKYGBgYmFzaApjb21wb3NlciByZXF1aXJlIHVkZXJsaW5lL29wZW5hcGktcGhwLWF0dHJpYnV0ZXMKYGBgCgoyIC0gRGVzY3JpYmUgeW91ciBBUEkgYnkgZm9sbG93aW5nIHRoaXMgZG9jdW1lbnRhdGlvbjogaHR0cHM6Ly91ZGVybGluZS5naXRodWIuaW8vb3BlbmFwaS1waHAtYXR0cmlidXRlcy8KCjMgLSBHZW5lcmF0ZSB0aGUgSlNPTiBmaWxlOgpgYGBiYXNoCnBocCAuL3ZlbmRvci9iaW4vb3BhZyAvc3JjL2ZpbGVzL3Byb2plY3Qgb3BlbmFwaS5qc29uCmBgYAoKQSBuZXcgZmlsZSBjYWxsZWQgYG9wZW5hcGkuanNvbmAgaGFzIGJlZW4gZ2VuZXJhdGVkICEKCiMjIEV4YW1wbGUKYGBgcGhwCiNbQ29udHJvbGxlcl0KY2xhc3MgQ29udHJvbGxlciB7CiAgICAjWwogICAgICAgIEdFVCgiL3BhdGgve2lkfSIsIFsiVGFnMSIsICJUYWcyIl0sICJEZXNjcmlwdGlvbiBvZiB0aGUgbWV0aG9kIiksCiAgICAgICAgUHJvcGVydHkoVHlwZTo6U1RSSU5HLCAicHJvcDEiLCBkZXNjcmlwdGlvbjogIlByb3BlcnR5IGRlc2NyaXB0aW9uIiwgZW51bTogQmFja2VkRW51bTo6Y2xhc3MpLAogICAgICAgIFByb3BlcnR5KFR5cGU6OklOVCwgInByb3AyIiwgZXhhbXBsZTogMSksCiAgICAgICAgUHJvcGVydHkoVHlwZTo6Qk9PTEVBTiwgInByb3AzIiksCiAgICAgICAgUHJvcGVydHkoVHlwZTo6UkVGLCAicHJvcDQiLCByZWY6IFJlZlNjaGVtYTo6Y2xhc3MpCiAgICAgICAgUmVzcG9uc2UocmVmOiBTY2hlbWFOYW1lOjpjbGFzcywgZGVzY3JpcHRpb246ICJSZXNwb25zZSBkZXNjcmlwdGlvbiIpCiAgICBdCiAgICBwdWJsaWMgZnVuY3Rpb24gZ2V0KCNbUGFyYW1ldGVyKCJQYXJhbWV0ZXIgZGVzY3JpcHRpb24iKV0gaW50ICRpZCk6IEpzb25SZXNwb25zZSB7CiAgICAgICAgLy8gLi4uCiAgICB9Cn0KCmVudW0gQmFja2VkRW51bTogc3RyaW5nCnsKICAgIGNhc2UgVkFMMTogInZhbDEiOwogICAgY2FzZSBWQUwyOiAidmFsMiI7Cn0KCiNbCiAgICBTY2hlbWEsCiAgICBQcm9wZXJ0eShUeXBlOjpTVFJJTkcsICJQcm9wZXJ0eSAxIiksCiAgICBQcm9wZXJ0eShUeXBlOjpJTlQsICJQcm9wZXJ0eSAyIiksCl0KY2xhc3MgUmVmU2NoZW1hCnsKICAgIHB1YmxpYyBzdHJpbmcgJHByb3BlcnR5MTsKICAgIHB1YmxpYyBpbnQgJHByb3BlcnR5MjsKfQpgYGAKCldpbGwgZ2VuZXJhdGUKYGBganNvbgp7CiAgICAicGF0aHMiOiB7CiAgICAgICAgIi9wYXRoL3tpZH0iOiB7CiAgICAgICAgICAgICJwb3N0IjogewogICAgICAgICAgICAgICAgInRhZ3MiOiBbCiAgICAgICAgICAgICAgICAgICAgIlRhZzEiLAogICAgICAgICAgICAgICAgICAgICJUYWcyIgogICAgICAgICAgICAgICAgXSwKICAgICAgICAgICAgICAgICJzdW1tYXJ5IjogIkRlc2NyaXB0aW9uIG9mIHRoZSBtZXRob2QiLAogICAgICAgICAgICAgICAgInBhcmFtZXRlcnMiOiBbCiAgICAgICAgICAgICAgICAgICAgewogICAgICAgICAgICAgICAgICAgICAgICAibmFtZSI6ICJpZCIsCiAgICAgICAgICAgICAgICAgICAgICAgICJpbiI6ICJwYXRoIiwKICAgICAgICAgICAgICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIlBhcmFtZXRlciBkZXNjcmlwdGlvbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICJyZXF1aXJlZCI6IHRydWUsCiAgICAgICAgICAgICAgICAgICAgICAgICJzY2hlbWEiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAidHlwZSI6ICJpbnRlZ2VyIgogICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgXSwKICAgICAgICAgICAgICAgICJyZXF1ZXN0Qm9keSI6IHsKICAgICAgICAgICAgICAgICAgICAiY29udGVudCI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgImFwcGxpY2F0aW9uL2pzb24iOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAic2NoZW1hIjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ0eXBlIjogIm9iamVjdCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgInByb3BlcnRpZXMiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJwcm9wMSI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ0eXBlIjogInN0cmluZyIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiZGVzY3JpcHRpb24iOiAiUHJvcGVydHkgZGVzY3JpcHRpb24iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgImVudW0iOiBbCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgInZhbDEiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ2YWwyIgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgXQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAicHJvcDIiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAidHlwZSI6ICJpbnRlZ2VyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJkZXNjcmlwdGlvbiI6ICIiCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJwcm9wMyI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ0eXBlIjogImJvb2xlYW4iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIiIKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgInByb3A0IjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIiRyZWYiOiAiIy9jb21wb25lbnRzL3NjaGVtYXMvUmVmU2NoZW1hIgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgICAgICJyZXNwb25zZXMiOiB7CiAgICAgICAgICAgICAgICAgICAgIjIwMCI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIlJlc3BvbnNlIGRlc2NyaXB0aW9uIiwKICAgICAgICAgICAgICAgICAgICAgICAgImNvbnRlbnQiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAiYXBwbGljYXRpb24vanNvbiI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAic2NoZW1hIjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiJHJlZiI6ICIjL2NvbXBvbmVudHMvc2NoZW1hcy9EdW1teUNvbXBvbmVudCIKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9Cn0KYGBgCg== readmeEtag: '"496c83d0500db6d82730fb9964a6eae3d6662ee4"' readmeLastModified: Mon, 25 Mar 2024 20:17:29 GMT repositoryId: 374416484 description: >- Automatically render your OpenApi 3 file describing your PHP API using attributes created: '2021-06-06T16:59:48Z' updated: '2024-05-03T14:58:45Z' language: PHP archived: false stars: 21 watchers: 3 forks: 8 owner: uderline logo: https://avatars.githubusercontent.com/u/10776690?v=4 license: MIT repoEtag: '"3adb88f7edc335ac394f5ca64f9b0d1e3ab2b0e91fe3c61583d2566b14c21bb2"' repoLastModified: Fri, 03 May 2024 14:58:45 GMT foundInMaster: true category: Parsers id: c9db84b60f30668d6e75ceb8a14d8e62 - source: openapi3 tags repository: https://github.com/ben-jamin-chen/kafka-streams-redis-statestore v3: true id: 574d5c6f9e5705994875220ae1875d15 repositoryMetadata: base64Readme: >- # Kafka Streams (2.6.0) with a Redis-backed State Store
By default, Kafka Streams utilizes the RocksDB storage engine for persistent state stores. One glaring issue you may have noticed is the biggest delay occurs when Kafka Streams is in a rebalancing state, where it internally rebuilds these state stores from the change-log topics. This is especially a common problem if you use [Kubernetes](https://kubernetes.io) and your environment is designed to scale up (new pods) or down (nodes are removed). While it’s in a rebalancing state, all involved consumer processing is blocked, so what this means is the application won’t be able to service any requests until the process becomes unblocked. This isn’t ideal especially in a production environment or from a practicality standpoint. The overall time with this delay seems to correspond with how large your dataset is in your topics, so if you're developing under a continuous integration environment this poses a major challenge since every deployment will require some amount of upfront time to rebalance.

Anyways, this is the primary motivation for switching over to a custom state store. We can leverage a permanent storage solution like Redis such that in any new deployment or application restart, we can avoid these upfront rebalancing delay.

## What You Need

* Java 14
* Maven 3.6.0+
* Docker 19+
* Redis 5+

## Getting Started
First, install a local instance of Redis (if you haven't already). If you're running on Mac OS X, you can use [Homebrew](https://brew.sh) via the command:

```zsh
$  brew install redis
```

For Windows, there's a couple of ways to do this, but I generally like to deploy and run [Redis](https://hub.docker.com/_/redis) in [Docker](https://docs.docker.com/docker-for-windows/install).

```pwsh
>  docker run --name some-redis -p 6379 -d redis
```

The easiest way to start the Redis server is just executing the `redis-server` command without any arguments.

```zsh
$  redis-server
```

You can connect to the Redis instance (with default configurations) using the `redis-cli` command:

```zsh
$  redis-cli -h localhost -p 6379
```

Next, we need to launch the various Confluent services (i.e. Schema Registry, Broker, ZooKeeper) locally by running the `docker-compose up -d` CLI command where the [docker-compose.yml](https://github.com/bchen04/kafka-streams-redis-statestore/blob/master/docker-compose.yml) file is. Typically, you can create a stack file (in the form of a YAML file) to define your applications. You can also run `docker-compose ps` to check the status of the stack. Notice, the endpoints from within the containers on your host machine.

| Name | From within containers | From host machine |
| ------------- | ------------- | ------------- |
| Kafka Broker | broker:9092 | localhost:9092 |
| Schema Registry  | http://schema-registry:8081 | http://localhost:8081 |
| ZooKeeper | zookeeper:2181 | localhost:2181 |

> Note: you can run `docker-compose down` to stop all services and containers.

As part of this sample, I've retrofitted the average aggregate example from [Confluent's Kafka Tutorials](https://kafka-tutorials.confluent.io/aggregating-average/kstreams.html) into this project. The API will calculate and return a running average rating for a given movie identifier. This should demonstrate how to build a basic API service on top of an aggregation result.

Notice in the `~/src/main/avro` directory, we have all our Avro schema files for the stream of `ratings` and `countsum`. For your convenience, the classes were already generated under the `~/src/main/java/io/confluent/demo` directory, but feel free to tinker with them and recompile the schemas if needed. The Avro classes can be programmatically generated using `Maven` or by manually invoking the [schema compiler](https://avro.apache.org/docs/1.10.0/gettingstartedjava.html#Compiling+the+schema). 

So before building and running the project, open a new terminal and run the following commands to generate your input and output topics.

```zsh
$  docker-compose exec broker kafka-topics --create --bootstrap-server \
   localhost:9092 --replication-factor 1 --partitions 1 --topic ratings

$  docker-compose exec broker kafka-topics --create --bootstrap-server \
   localhost:9092 --replication-factor 1 --partitions 1 --topic rating-averages
```

Next, we will need to produce some data onto the input topic.

```zsh
$  docker exec -i schema-registry /usr/bin/kafka-avro-console-producer --topic ratings --broker-list broker:9092\
    --property "parse.key=false"\
    --property "key.separator=:"\
    --property value.schema="$(< src/main/avro/rating.avsc)"
 ```
 
Paste in the following `json` data when prompted and be sure to press enter twice to actually submit it.

```json
{"movie_id":362,"rating":10}
{"movie_id":362,"rating":8}
 ```

Optionally, you can also see the consumer results on the output topic by running this command on a new terminal window:

```zsh
$  docker exec -it broker /usr/bin/kafka-console-consumer --topic rating-averages --bootstrap-server broker:9092 \
    --property "print.key=true"\
    --property "key.deserializer=org.apache.kafka.common.serialization.LongDeserializer" \
    --property "value.deserializer=org.apache.kafka.common.serialization.DoubleDeserializer" \
    --from-beginning
```

## Build and Run the Sample

You can import the code straight into your preferred IDE or run the sample using the following command (in the root project folder).

```zsh
$  mvn spring-boot:run
```

After the application runs, from the `redis-cli`, observe that a new [Redis stream](https://redis.io/topics/streams-intro) got created using the `KEYS` command:

```zsh
localhost:6379> keys *
1) "rating-averages-stream"
```

To query the stream, you can use the `XRANGE` command where each entry returned is an array of the ID and the list of field-value pairs. The `-` and `+` represent the smallest and the greatest ID possible. If you used the same input data from above, it should return something similar like this below:

```zsh
localhost:6379> xrange rating-averages-stream - +
1) 1) "1605043614011-0"
   2) 1) "362"
      2) "9.0"
```

Finally, navigate to [http://localhost:7001/swagger-ui/index.html?configUrl=/api-docs/swagger-config](http://localhost:7001/swagger-ui/index.html?configUrl=/api-docs/swagger-config) in your web browser to access the Swagger UI. You can enter `362` as the `movieId` and it should return the same value from above.

```json
{
  "movieId": 362,
  "rating": 9
}
```

You'll find that the `store.read(...)` method in the controller is able to query into this Redis-backed state store.

> Note: keep in mind the various [states](https://kafka.apache.org/26/javadoc/org/apache/kafka/streams/KafkaStreams.State.html) of the client. When a Kafka Streams instance is in `RUNNING` state, it allows for inspection of the stream's metadata using methods like `queryMetadataForKey()`. While it is in `REBALANCING` state, the REST service cannot immediately answer requests until the state stores are fully rebuilt.

## Troubleshooting

* In certain conditions, you may need to do a complete application reset. You can delete the application’s local state directory where the application instance was run. In this project, Kafka Streams persists local states under the `~/data` folder.
 readmeEtag: '"ee740369c0ba2ffe584b7045371f503358c5048d"' readmeLastModified: Wed, 11 Nov 2020 18:21:16 GMT repositoryId: 289421697 description: >- A demo project demonstrating how to read and write data into a Redis-backed state store using Kafka Streams. created: '2020-08-22T04:54:11Z' updated: '2026-01-19T07:49:11Z' language: Java archived: false stars: 24 watchers: 1 forks: 6 owner: ben-jamin-chen logo: https://avatars.githubusercontent.com/u/41641688?v=4 license: MIT repoEtag: '"0b5c336194be21fcc7a2df86631826d1828fac3ec239283a603288d7682ce11b"' repoLastModified: Mon, 19 Jan 2026 07:49:11 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/technocreatives/openapi-eller v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyB3aWR0aD0iMjU2IiBoZWlnaHQ9IjI1NiIgc3JjPSJodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS8yNzkwOTkvMzk2NzcyNjQtMzk2Zjk1NjgtNTE3OC0xMWU4LTlhZmMtYjg0NWZkZDIyMThmLnBuZyIgYWx0PSJFbGxlciBsb2dvIi8+CjwvcD4KClshW0FjdGlvbnMgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vdGVjaG5vY3JlYXRpdmVzL29wZW5hcGktZWxsZXIvd29ya2Zsb3dzL25vZGVqcy1idWlsZC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vdGVjaG5vY3JlYXRpdmVzL29wZW5hcGktZWxsZXIvYWN0aW9ucykgWyFbTlBNIHZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3Yvb3BlbmFwaS1lbGxlci5zdmc/c3R5bGU9ZmxhdCldKGh0dHBzOi8vd3d3Lm5wbWpzLm9yZy9wYWNrYWdlL29wZW5hcGktZWxsZXIpCgojIE9wZW5BUEkgR2VuZXJhdG9yLCBlbGxlcj8KCkdlbmVyYXRlIE9wZW5BUEkgdjMgY2xpZW50cyBhbmQgc2VydmVycyBmcm9tIHRoZSBjb21tYW5kIGxpbmUgd2l0aCBub3RoaW5nIG1vcmUKdGhhbiBOb2RlLmpzLgoKCkp1c3QgcnVuOgoKYGBgCm5wbSBpIC1nIG9wZW5hcGktZWxsZXIKYGBgCgpTZWUgYG9wZW5hcGktZWxsZXIgLS1oZWxwYCBmb3IgdXNhZ2UgZGV0YWlscy4KCi0gW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vdGVjaG5vY3JlYXRpdmVzLmdpdGh1Yi5pby9vcGVuYXBpLWVsbGVyKQoKLS0tCgpMb29raW5nIGZvciBhbiBlYXN5IHdheSB0byBnZW5lcmF0ZSBhIG1vY2sgc2VydmVyIGZyb20gYW4gT3BlbkFQSSB2MyBzcGVjPyBXZSd2ZSBnb3QgeW91IGNvdmVyZWQuCgpUcnkgW29wZW5hcGktbW9jay1lbGxlcl0oaHR0cHM6Ly9naXRodWIuY29tL3RlY2hub2NyZWF0aXZlcy9vcGVuYXBpLW1vY2stZWxsZXIpIHRvZGF5IQoKLS0tCgoqKlBMRUFTRSBOT1RFOiBUaGlzIGNvZGViYXNlIGlzIHN0aWxsIGEgd29yay1pbi1wcm9ncmVzcywgYnV0IGl0IGRvZXMgcHJvZHVjZSBwcm9kdWN0aW9uLWdyYWRlIGNvZGUKZm9yIHRob3NlIHRhcmdldHMgbGlzdGVkIGFzIHN1cHBvcnRlZC4gQmVoYXZpb3VyIGlzIHN1YmplY3QgdG8gY2hhbmdlIGJldHdlZW4gdmFyaWFudHMgdW50aWwgMS4wLjAuKioKCiMjIEZlYXR1cmVzCgotIFN1cHBvcnRzKiB0aGUgZnVsbCBPcGVuQVBJIHYzIHNwZWNpZmljYXRpb24KLSBVc2VzIGFuIGludGVyY2VwdG9yIHBhdHRlcm4gZm9yIGhhbmRsaW5nIHNlY3VyaXR5IHNjaGVtYXMgaW4gY2xpZW50cwotIE9BdXRoIDIgY2xpZW50cyBjb21wbHkgd2l0aCBbUkZDNjc0OV0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzY3NDkpIGFuZCAKICBbUkZDNjc1MF0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzY3NTApIChCZWFyZXIgVG9rZW4gVXNhZ2UpCi0gVGFyZ2V0cyBjYW4gYmUgY29uZmlndXJlZCB3aXRoIGEgSlNPTiBvciBZQU1MIGZpbGUgZm9yIHNpbXBsZSwgcmVwcm9kdWNhYmxlIGdlbmVyYXRpb25zCi0gRWFzaWx5IGV4dGVuc2libGUgSGFuZGxlYmFycyB0ZW1wbGF0ZXMgZm9yIGNvcmUgc3RydWN0dXJlIG9mIGZpbGVzLCB3aXRoIFR5cGVTY3JpcHQgCiAgdGFyZ2V0LXNwZWNpZmljIGNvZGUgZm9yIGhhbmRsaW5nIHdpdGggcG9pbnR5IGJpdHMKCiMjIFN1cHBvcnRlZCB0YXJnZXRzCgotIENsaWVudHM6CiAgLSBLb3RsaW4gKEFuZHJvaWQpCiAgLSBTd2lmdCAoaU9TKQogIC0gVHlwZVNjcmlwdAotIFNlcnZlcnM6CiAgLSBBU1AuTkVUIChNVkMgRnJhbWV3b3JrIDQuNSkKClRoZXJlIGFyZSBvdGhlciB0YXJnZXRzIGluIHRoZSB0cmVlLCB0aG91Z2ggdGhleSBhcmUgYSB3b3JrLWluLXByb2dyZXNzLgoKIyMgUm9hZG1hcCB0byAwLjQKCi0gWyBdIEdlbmVyYXRpbmcgcGxhdGZvcm0tY29uZm9ybWFudCBBUEkgZG9jdW1lbnRhdGlvbgotIFsgXSBIYW5kbGUgcmV0dXJuaW5nIGhlYWRlcnMsIHN0YXR1cyBjb2RlcyBhbmQgcmF3IHJlc3BvbnNlIG9iamVjdHMgd2hlcmUgbmVjZXNzYXJ5Ci0gWyBdIEhhbmRsZSBtYW5kYXRvcnkgY29uZmlndXJhdGlvbiBmb3IgdGFyZ2V0cwotIFsgXSBHZW5lcmF0ZSBkb2N1bWVudGF0aW9uIGZvciB0YXJnZXQgY29uZmlndXJhdGlvbgoKIyMgVXNlcnMKCi0gVGhlIFRlY2hubyBDcmVhdGl2ZXMKCiMjIENvbnRyaWJ1dGluZwoKV2UgaGFwcGlseSBhY2NlcHQgY29udHJpYnV0aW9ucyEgV2Ugc2ltcGx5IGFzayB0aGF0IHlvdSBwbGVhc2UgbWFrZSBzdXJlIHRoYXQgYW55IGRlcGVuZGVuY2llcyAKb2YgeW91ciB0YXJnZXRzIHVzZSBhIHBlcm1pc3NpdmUgbGljZW5zZSBjb21wYXRpYmxlIHdpdGggdGhlIElTQyBsaWNlbnNlICh3aGljaCBtZWFucyBubyBBR1BMIG9yIApHUEwgZGVwZW5kZW5jaWVzLCB1bmZvcnR1bmF0ZWx5LikKCklmIHlvdSdyZSB1bnN1cmUsIG9wZW4gYW4gaXNzdWUgYW5kIHdlIGNhbiBoZWxwIHlvdSBvdXQhCgojIyBMaWNlbnNlCgpJU0MgbGljZW5zZSAtIHNlZSBMSUNFTlNFIGZpbGUuCgpBbnkgY29kZSBvdXRwdXR0ZWQgYnkgdGhpcyBnZW5lcmF0b3IgaXMgdGhlIGxpY2Vuc2Ugb2YgeW91ciBjaG9pY2UuCg== readmeEtag: '"14a1bcf12a26b2a7e23f2ea775a520b7ac613910"' readmeLastModified: Tue, 12 May 2020 08:48:11 GMT repositoryId: 130574986 description: Generate OpenAPI v3 clients and servers from the command line created: '2018-04-22T14:25:35Z' updated: '2024-06-10T18:55:28Z' language: TypeScript archived: false stars: 20 watchers: 6 forks: 4 owner: technocreatives logo: https://avatars.githubusercontent.com/u/4668783?v=4 license: ISC repoEtag: '"94511f5aa63dc43c3f5d92bf1f4771106c493e7ff596d6b95e5e9fc8f2e6ba08"' repoLastModified: Mon, 10 Jun 2024 18:55:28 GMT foundInMaster: true category: - Server - Parsers id: 9ab6e634f0f3553238502aa27b3e5a45 - source: openapi3 tags repository: https://github.com/vanmoof/gopenapi v3: true repositoryMetadata: base64Readme: >- PSBHT3BlbkFQSQoKaW1hZ2U6aHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvVmFuTW9vZi9nb3BlbmFwaS5zdmc/c3R5bGU9c3ZnJmNpcmNsZS10b2tlbj0zYWY2MjY4YjJjOGRhMjBjMjI2MzIyNjE4ODJkMzU4ZTMwMjdjMDQ1WyJDaXJjbGVDSSIsIGxpbms9Imh0dHBzOi8vY2lyY2xlY2kuY29tL2doL1Zhbk1vb2YvZ29wZW5hcGkiXQoKQW4gT3BlbkFQSSB1dGlsaXR5IGZvciBHby4KVGhpcyBwcm9qZWN0IGFpbXMgdG8gYnJpbmcgc3VwcG9ydCBvZiBPcGVuQVBJIHYzLgoKPT0gVXNhZ2UKCmBgYGJhc2gKJCBnb3BlbmFwaSBbY29tbWFuZF0gW2FyZ10KYGBgCgo9PT0gR2VuZXJhdGluZyBTcGVjaWZpY2F0aW9ucyBGcm9tIENvZGUKCmBgYGJhc2gKZ29wZW5hcGkgZ2VuZXJhdGUgc3BlYyBbb3B0aW9uYWwgcGF0aF0gW2ZsYWdzXQpgYGAKCj09PT0gQXJncwoKYGBgYmFzaApbb3B0aW9uYWwgcGF0aF0gICBPcHRpb25hbGx5IHNwZWNpZnkgdGhlIGRpcmVjdG9yeSBpbiB3aGljaCB0byBzZWFyY2guIEFjY2VwdHMgYWJzb2x1dGUgcGF0aHMuIFJlbGF0aXZlIHBhdGhzIGFyZSByZWxhdGl2ZSB0byB0aGUgY3VycmVudCBkaXJlY3RvcnkuIChkZWZhdWx0ICIuIikKYGBgCgo9PT09IEZsYWdzCgpgYGBiYXNoCi1mLCAtLWZvcm1hdCBzdHJpbmcgICBUaGUgZm9ybWF0IG9mIHRoZSBvdXRwdXQuIE1heSBiZSBqc29uIG9yIHlhbWwgKGRlZmF1bHQgImpzb24iKQotbywgLS1vdXRwdXQgc3RyaW5nICAgV2hlcmUgdGhlIG91dHB1dCBzaG91bGQgYmUgZGlyZWN0ZWQuIE1heSBiZSAnLScgKHN0ZG91dCkgb3IgYSBwYXRoIHRvIGEgZmlsZSAoZGVmYXVsdCAiLSIpCmBgYAoKPT09PSBGb3JtYXQKCkNvZGUgaXMgYW5ub3RhdGVkIHdpdGggZGlmZmVyZW50IHR5cGVzIG9mIGNvbW1lbnRzIHRoYXQgaGVscCBnZW5lcmF0ZSB0aGUgc3BlYy4KClRoZSBjb21tZW50IGNvbnRhaW5zIGEga2V5d29yZCB0aGF0IHNwZWNpZmllcyB0aGUgdHlwZSBvZiB0aGUgT3BlbkFQSSBlbGVtZW50LgoKVGhlIGNvbnRlbnQgb2YgdGhlIGNvbW1lbnQgc2hvdWxkIGJlIGEgdmFsaWQgWUFNTCBPcGVuQVBJIGVsZW1lbnQKCj09PT09IEluZm8KCkJlZ2luIGEgY29tbWVudCB3aXRoIGBnb3BlbmFwaTppbmZvYCBhbmQgZm9sbG93IHVwIHdpdGggYSBZQU1MIHJlcHJlc2VudGF0aW9uIG9mIHRoZSBPcGVuQVBJIEluZm8gZWxlbWVudC4KClRoaXMgZWxlbWVudCBpcyB0aGVuIHNldCB0byB0aGUgYGluZm9gIHByb3BlcnR5IG9mIHRoZSBzcGVjaWZpY2F0aW9uLgoKYGBgZ28KcGFja2FnZSBtYWluCgovKgpnb3BlbmFwaTppbmZvCnRpdGxlOiBUaGUgQXBwIE5hbWUKdmVyc2lvbjogMS4wCmRlc2NyaXB0aW9uOiB8LQogIFRoZSBhcHAgZGVzY3JpcHRpb24KY29udGFjdDoKICBuYW1lOiBKaW1ib2IgSm9uZXMKICB1cmw6IGh0dHBzOi8vam9uZXMuY29tCiAgZW1haWw6IGppbWJvYkBqb25lcy5jb20KbGljZW5zZToKICBuYW1lOiBBcGFjaGUgMi4wCiAgdXJsOiBodHRwczovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wLmh0bWwKKi8KZnVuYyBtYWluKCkgewp9CmBgYAoKPT09PT0gUGF0aAoKQmVnaW4gYSBjb21tZW50IHdpdGggYGdvcGVuYXBpOnBhdGhgIGFuZCBmb2xsb3cgdXAgd2l0aCBhIFlBTUwgcmVwcmVzZW50YXRpb24gb2YgdGhlIE9wZW5BUEkgUGF0aEl0ZW0gZWxlbWVudC4KClRoaXMgZWxlbWVudCBpcyB0aGVuIGFwcGVuZGVkIHRvIHRoZSBgcGF0aHNgIHByb3BlcnR5IG9mIHRoZSBzcGVjaWZpY2F0aW9uLgoKYGBgZ28KcGFja2FnZSBtYWluCgovKgpnb3BlbmFwaTpwYXRoCi9waW5nOgogIGdldDoKICAgIHJlc3BvbnNlczoKICAgICAgMjAwOgogICAgICAgIGRlc2NyaXB0aW9uOiB8LQogICAgICAgICAgVGhlIGRlZmF1bHQgcmVzcG9uc2Ugb2YgInBpbmciCiAgICAgICAgY29udGVudDoKICAgICAgICAgIHRleHQvcGxhaW46CiAgICAgICAgICAgIGV4YW1wbGU6IHBvbmcKKi8KZnVuYyBDb250cm9sbGVyRnVuYygpIHsKfQpgYGAKCj09PT09IE9iamVjdCBTY2hlbWEKCkFubm90YXRlIGEgc3RydWN0IHdpdGggYSBgZ29wZW5hcGk6b2JqZWN0U2NoZW1hYC4KClRoZSBnZW5lcmF0ZWQgT2JqZWN0U2NoZW1hIGVsZW1lbnQgd2lsbCBiZSBhcHBlbmRlZCB0byB0aGUgYGNvbXBvbmVudHMuc2NoZW1hc2AgcHJvcGVydHkgb2YgdGhlIHNwZWNpZmljYXRpb24uCgpgYGBnbwovL2dvcGVuYXBpOm9iamVjdFNjaGVtYQp0eXBlIFJvb3RNb2RlbCBzdHJ1Y3QgewoJSW50RmllbGQgICAgaW50NjQgIGBqc29uOiJpbnRGaWVsZCJgCglTdHJpbmdGaWVsZCBzdHJpbmcgYGpzb246InN0cmluZ0ZpZWxkImAKfQoKLy8gVGhpcyBzdHJ1Y3Qgd2lsbCBiZSBpZ25vcmVkCnR5cGUgSWdub3JlZE1vZGVsIHN0cnVjdCB7Cn0KCi8vZ29wZW5hcGk6b2JqZWN0U2NoZW1hCnR5cGUgQWxpYXNlZE1vZGVscyBbXSpBbGlhc2VkTW9kZWwgLy8gVGhpcyBhbGlhcyB3aWxsIGFwcGVhciBhcyBhIHNjaGVtYSB0b28KCi8vZ29wZW5hcGk6b2JqZWN0U2NoZW1hCnR5cGUgQWxpYXNlZE1vZGVsIHN0cnVjdCB7CglJZ25vcmVkRmllbGQgc3RyaW5nIGBqc29uOiItImAgLy8gVGhpcyBmaWVsZCB3aWxsIGJlIGlnbm9yZWQKCVRpbWVGaWVsZCAgICB0aW1lLlRpbWUKfQoKYGBgCgo9PT09PSBQYXJhbWV0ZXIKCkFubm90YXRlIGEgYGNvbnN0YCBvciBhIGB2YXJgIHdpdGggYSBgZ29wZW5hcGk6cGFyYW1ldGVyYC4KClRoZSBhbm5vdGF0ZWQgZmllbGQgd2lsbCBiZSBhcHBlbmRlZCB0byB0aGUgYGNvbXBvbmVudHMucGFyYW1ldGVyc2AgcHJvcGVydHkgb2YgdGhlIHNwZWNpZmljYXRpb24uCgpgYGBnbwovKgpnb3BlbmFwaTpwYXJhbWV0ZXIKaW46IHBhdGgKcmVxdWlyZWQ6IHRydWUKY29udGVudDoKICB0ZXh0L3BsYWluOgogICAgZXhhbXBsZTogMzAKKi8KY29uc3QgTGltaXQgPSAibGltaXQiCmBgYAoKVGhlIG5hbWUgb2YgdGhlIGZpZWxkIChgTGltaXRgKSB3aWxsIGJlIHRoZSBwYXJhbWV0ZXIgaWRlbnRpZmllciBhbmQgdGhlIHZhbHVlIG9mIHRoZSBmaWVsZCAoYGxpbWl0YCkgd2lsbCBiZSB0aGUgbmFtZSBvZiB0aGUgcGFyYW1ldGVyLg== readmeEtag: '"d002052719c18775f586320321423b3be79c70a5"' readmeLastModified: Tue, 15 Oct 2019 10:03:39 GMT repositoryId: 209779146 description: An OpenAPI v3 utility for Go created: '2019-09-20T11:53:43Z' updated: '2025-03-06T13:03:14Z' language: Go archived: false stars: 19 watchers: 5 forks: 1 owner: VanMoof logo: https://avatars.githubusercontent.com/u/26412800?v=4 license: MIT repoEtag: '"77ec456a852f3eef49f4466aa83ce459be68eb1623d93dc5db000c22c9c684bc"' repoLastModified: Thu, 06 Mar 2025 13:03:14 GMT foundInMaster: true category: Parsers id: 87176f9790da6975ca8309ca566efc2d - source: openapi3 tags repository: https://github.com/speakeasy-api/openapi-reference-documentation v3: true id: 529feb5341d9e218d083e115c280ae49 repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8cGljdHVyZT4KICAgIDxzb3VyY2UgbWVkaWE9IihwcmVmZXJzLWNvbG9yLXNjaGVtZTogZGFyaykiIHNyY3NldD0iLi9pbWFnZXMvbG9nb3Mvb3BlbmFwaS1kYXJrLnN2ZyI+CiAgICA8aW1nIGFsdD0iT3BlbkFQSSBsb2dvIiBzcmM9Ii4vaW1hZ2VzL2xvZ29zL29wZW5hcGktbGlnaHQuc3ZnIiB3aWR0aD0iOTAlIj4KICA8L3BpY3R1cmU+CjwvZGl2PgoKPGJyPjxicj4KCjxkaXYgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9zcGVha2Vhc3lhcGkuZGV2LyI+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9jdXN0b20taWNvbi1iYWRnZXMuZGVtb2xhYi5jb20vYmFkZ2UvLUJ1aWx0JTIwQnklMjBTcGVha2Vhc3ktMjEyMDE1P3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nb0NvbG9yPUZCRTMzMSZsb2dvPXNwZWFrZWFzeSZsYWJlbENvbG9yPTU0NTQ1NCIgLz48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQiPgogICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTGljZW5zZS1NSVQtYmx1ZS5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSIgc3R5bGU9IndpZHRoOiAxMDBweDsgaGVpZ2h0OiAyOHB4OyIgLz48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3NwZWFrZWFzeS1hcGkvb3BlbmFwaS1yZWZlcmVuY2UtZG9jdW1lbnRhdGlvbi9pc3N1ZXMiPgogICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy9zcGVha2Vhc3ktYXBpL29wZW5hcGktcmVmZXJlbmNlLWRvY3VtZW50YXRpb24/c3R5bGU9Zm9yLXRoZS1iYWRnZSIgc3R5bGU9IndpZHRoOiAxMDBweDsgaGVpZ2h0OiAyOHB4OyIgLz48L2E+CjwvZGl2PgoKPGJyIC8+CjxiciAvPgoKPGgxIGFsaWduPSJjZW50ZXIiPlRoZSBPcGVuQVBJIFJlZmVyZW5jZTwvaDE+CjxkaXYgYWxpZ249ImNlbnRlciI+CkFuIGV4YW1wbGUtcmljaCwgZWFzeSB0byB1bmRlcnN0YW5kIHJlZmVyZW5jZSBmb3IgT3BlbkFQSS4KPC9kaXY+CjwvYnI+Cgo8cCBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL3d3dy5zcGVha2Vhc3lhcGkuZGV2L29wZW5hcGkiPiBMaXZlIFJlZmVyZW5jZTwvYT4KPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vc3BlYWtlYXN5LWFwaS9vcGVuYXBpLXJlZmVyZW5jZS1kb2N1bWVudGF0aW9uL2lzc3Vlcy9uZXc/YXNzaWduZWVzPSZsYWJlbHM9YnVnJnByb2plY3RzPSZ0ZW1wbGF0ZT1idWdfcmVwb3J0Lm1kIj5SZXBvcnQgSXNzdWU8L2E+CiAgICDCtwogICAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3NwZWFrZWFzeS1hcGkvb3BlbmFwaS1yZWZlcmVuY2UtZG9jdW1lbnRhdGlvbi9pc3N1ZXMvbmV3P2Fzc2lnbmVlcz0mbGFiZWxzPWVuaGFuY2VtZW50JnByb2plY3RzPSZ0ZW1wbGF0ZT1mZWF0dXJlX3JlcXVlc3QubWQiPlJlcXVlc3QgRmVhdHVyZTwvYT4KICAgIMK3CiAgPGEgaHJlZj0iaHR0cHM6Ly9qb2luLnNsYWNrLmNvbS90L3NwZWFrZWFzeS1kZXYvc2hhcmVkX2ludml0ZS96dC0xY3diM2ZseHotbFM1U3laeEFzRl8zTk9xNXhjOENqdyI+Sm9pbiBPdXIgU2xhY2s8L2E+CiAgCiAgPC9wPgoKIyMgV2hhdCBpcyB0aGlzIHJlZmVyZW5jZSBmb3I/CgpPcGVuQVBJIGlzIGEgc3BlY2lmaWNhdGlvbiBmb3IgZGVzY3JpYmluZyBSRVNUZnVsIEFQSXMuIEZ1bmN0aW9uYWxseSwgaXQgc2VydmVzIGFzIGEgc291cmNlIG9mIHRydXRoIGZvciBBUEkgdG9vbGluZywgZG9jdW1lbnRhdGlvbiwgYW5kIHRlc3RpbmcuIEFzIHN1Y2gsIGl0IGlzIGVzc2VudGlhbCBmb3IgZGV2ZWxvcGVycyB0byB1bmRlcnN0YW5kIHRoZSBzcGVjaWZpY2F0aW9uLgoKVGhpcyByZWZlcmVuY2UgaXMgaW50ZW5kZWQgdG8gYmUgY29tcGxlbWVudGFyeSBvZiB0aGUgW29mZmljaWFsIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBSZWZlcmVuY2VdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCkuIEl0IGlzIGZvY3VzZWQgb24gYmVpbmcgZGV2LWZyaWVuZGx5OiBleGFtcGxlLWJhc2VkIHdpdGggdGhvcm91Z2ggZXhwbGFuYXRpb25zLCBwbHVzIEFJIHNlYXJjaCEKCklmIHlvdSB3b3VsZCBsaWtlIHRvIGNvbnRyaWJ1dGUgdG8gdGhpcyByZWZlcmVuY2UsIHBsZWFzZSBzZWUgdGhlIFtjb250cmlidXRpbmcgZ3VpZGVdKENPTlRSSUJVVElORy5tZCkuCgojIyBXaGF0IGlzIFNwZWFrZWFzeQoKQXQgU3BlYWtlYXN5IHdlIGFyZSBidWlsZGluZyB0aGUgdG9vbGNoYWluIHRvIG1ha2UgQVBJcyBlYXN5IHRvIGNyZWF0ZSBhbmQgY29uc3VtZS4KClRvZGF5LCBvdXIgcGxhdGZvcm0gYWxsb3dzIHlvdSB0byBjcmVhdGUgYSByb2J1c3QgZGV2ZWxvcGVyIGV4cGVyaWVuY2UgZm9yIHlvdXIgQVBJIHVzaW5nIGFuIE9wZW5BUEkgc3BlYzoKCi0gW1NES3MgaW4gOSBsYW5ndWVzXShodHRwczovL3d3dy5zcGVha2Vhc3lhcGkuZGV2L2RvY3MvY3JlYXRlLWNsaWVudC1zZGtzKTogVHlwZVNjcmlwdCwgUHl0aG9uLCBHbywgSmF2YSwgQyMsIFJ1YnksIFBIUCwgU3dpZnQgYW5kIFVuaXR5Ci0gW1RlcnJhZm9ybSBQcm92aWRlcnNdKGh0dHBzOi8vd3d3LnNwZWFrZWFzeWFwaS5kZXYvZG9jcy9jcmVhdGUtdGVycmFmb3JtKQoKRHVlIHRvIHRoZSBuYXR1cmUgb2Ygb3VyIHdvcmssIHdlIHNwZW5kIGEgbG90IG9mIHRpbWUgd29ya2luZyB3aXRoIE9wZW5BUEkgYW5kIGhhdmUgYmVjb21lIGV4cGVydHMgb24gaG93IHRoZSBzcGVjIGltcGFjdHMgZG93bnN0cmVhbSB3b3JrZmxvd3MuIFRoaXMgcmVmZXJlbmNlIGlzIGEgd2F5IGZvciB1cyB0byBnaXZlIGJhY2sgdG8gdGhlIGNvbW11bml0eSBhbmQgcGFzcyBvbiBzb21lIG9mIHRoZSBsZWFybmluZyB0aGF0IHdlJ3ZlIGRvbmUgYWxvbmcgdGhlIHdheS4KCiMjIENvbnRyaWJ1dGluZwoKV2hpbGUgd2UgaGF2ZSBhIGxvdCBvZiBpbnRlcm5hbCBvcGluaW9ucyBvbiBob3cgdG8gd3JpdGUgT3BlbkFQSSwgd2UgdmFsdWUgb3BpbmlvbnMgZnJvbSB0aGUgY29tbXVuaXR5LiBXZSB3b3VsZCBsb3ZlIGZvciBjb250cmlidXRpb25zIHRvIHRoaXMgcmVmZXJlbmNlIHRvIGNvbWUgZnJvbSBhIHdpZGUgcmFuZ2Ugb2YgcGVvcGxlLCB3aXRoIGRpZmZlcmVudCBiYWNrZ3JvdW5kcyBhbmQgZXhwZXJpZW5jZXMuCgpUaGlzIE9wZW5BUEkgUmVmZXJlbmNlIGlzIE9wZW4gU291cmNlIHVuZGVyIHRoZSBbTUlUXShMSUNFTlNFKSwgYW5kIGlzIHRoZSBbY29weXJpZ2h0IG9mIGl0cyBjb250cmlidXRvcnNdKE5PVElDRSkuIElmIHlvdSB3b3VsZCBsaWtlIHRvIGNvbnRyaWJ1dGUgdG8gdGhlIHNvZnR3YXJlLCB5b3UgbXVzdDoKCjEuIFJlYWQgdGhlIERldmVsb3BlciBDZXJ0aWZpY2F0ZSBvZiBPcmlnaW4gVmVyc2lvbiAxLjEgKGh0dHBzOi8vZGV2ZWxvcGVyY2VydGlmaWNhdGUub3JnLykKMi4gU2lnbmluZyBhbGwgY29tbWl0cyB0byB0aGUgT3BlbkFQSSBSZWZlcmVuY2UgcHJvamVjdC4KClRoaXMgZW5zdXJlcyB0aGF0IHVzZXJzLCBkaXN0cmlidXRvcnMsIGFuZCBvdGhlciBjb250cmlidXRvcnMgY2FuIHJlbHkgb24gYWxsIHRoZSBzb2Z0d2FyZSByZWxhdGVkIHRvIFNwZWFrZWFzeSBiZWluZyBjb250cmlidXRlZCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIFtMaWNlbnNlXShMSUNFTlNFKS4gTm8gY29udHJpYnV0aW9ucyB3aWxsIGJlIGFjY2VwdGVkIHdpdGhvdXQgZm9sbG93aW5nIHRoaXMgcHJvY2Vzcy4KCkFmdGVyd2FyZHMsIG5hdmlnYXRlIHRvIHRoZSBbY29udHJpYnV0aW5nIGd1aWRlXShDT05UUklCVVRJTkcubWQpIHRvIGdldCBzdGFydGVkLgoKIyMgTGljZW5zZQoKVGhpcyByZXBvc2l0b3J5IGNvbnRhaW5zIFNwZWFrZWFzeSdzIE9wZW5BUEkgUmVmZXJlbmNlLCBjb3ZlcmVkIHVuZGVyIHRoZSBbTUlUXShMSUNFTlNFKSwgZXhjZXB0IHdoZXJlIG5vdGVkIChhbnkgRGF5dG9uYSBsb2dvcyBvciB0cmFkZW1hcmtzIGFyZSBub3QgY292ZXJlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIGFuZCBzaG91bGQgYmUgZXhwbGljaXRseSBub3RlZCBieSBhIExJQ0VOU0UgZmlsZS4pCgpTcGVha2Vhc3kgaXMgYSBwcm9kdWN0IHRoYXQgbWFrZXMgdXNlIG9mIHRoaXMgb3BlbiBzb3VyY2UgZG9jdW1lbnRhdGlvbi4gSXQgaXMgZGlzdHJpYnV0ZWQgdW5kZXIgb3VyIGNvbW1lcmNpYWwgdGVybXMuCgpPdGhlcnMgYXJlIGFsbG93ZWQgdG8gbWFrZSB0aGVpciBvd24gZGlzdHJpYnV0aW9uIG9mIHRoaXMgZG9jdW1lbnRhdGlvbiwgYnV0IHRoZXkgY2Fubm90IHVzZSBhbnkgb2YgdGhlIFNwZWFrZWFzeSB0cmFkZW1hcmtzLCBjbG91ZCBzZXJ2aWNlcywgZXRjLgoKIyMgQ29kZSBvZiBDb25kdWN0CgpUaGlzIHByb2plY3QgaGFzIGFkYXB0ZWQgdGhlIENvZGUgb2YgQ29uZHVjdCBmcm9tIHRoZSBbQ29udHJpYnV0b3IgQ292ZW5hbnRdKGh0dHBzOi8vd3d3LmNvbnRyaWJ1dG9yLWNvdmVuYW50Lm9yZy8pLiBGb3IgbW9yZSBpbmZvcm1hdGlvbiBzZWUgdGhlIFtDb2RlIG9mIENvbmR1Y3RdKENPREVfT0ZfQ09ORFVDVC5tZCkuCgojIyBRdWVzdGlvbnMKCkZvciBtb3JlIGluZm9ybWF0aW9uIG9uIGhvdyB0byB1c2UgYW5kIGRldmVsb3AgU3BlYWtlYXN5LCB0YWxrIHRvIHVzIG9uCltTbGFja10oaHR0cHM6Ly9qb2luLnNsYWNrLmNvbS90L3NwZWFrZWFzeS1kZXYvc2hhcmVkX2ludml0ZS96dC0xY3diM2ZseHotbFM1U3laeEFzRl8zTk9xNXhjOENqdykuCg== readmeEtag: '"5d53ec1c7efd99d73fe1abf3b311807aae3081d14f10a6890440a829e1895692"' readmeLastModified: Tue, 09 Apr 2024 11:51:00 GMT repositoryId: 648239100 description: Best in class OpenAPI reference documentation created: '2023-06-01T14:08:03Z' updated: '2025-03-27T18:07:57Z' language: MDX archived: false stars: 21 watchers: 12 forks: 5 owner: speakeasy-api logo: https://avatars.githubusercontent.com/u/91446104?v=4 repoEtag: '"a217844bfc1b98b48e1942c5503bc1702f0360d9a561f696252abae39c36a350"' repoLastModified: Thu, 27 Mar 2025 18:07:57 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/erasys/openapi-php v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXBocAoKWyFbTGF0ZXN0IFZlcnNpb24gb24gUGFja2FnaXN0XVtpY28tdmVyc2lvbl1dW2xpbmstcmVsZWFzZXNdClshW1NvZnR3YXJlIExpY2Vuc2VdW2ljby1saWNlbnNlXV0oTElDRU5TRSkKWyFbQnVpbGQgU3RhdHVzXVtpY28tdHJhdmlzXV1bbGluay10cmF2aXNdClshW0NvdmVyYWdlIFN0YXR1c11baWNvLWNvdmVyYWdlXV1bbGluay1jb3ZlcmFnZV0KWyFbUXVhbGl0eSBTY29yZV1baWNvLXNjcnV0aW5pemVyXV1bbGluay1zY3J1dGluaXplcl0KWyFbVG90YWwgRG93bmxvYWRzXVtpY28tZG93bmxvYWRzXV1bbGluay1kb3dubG9hZHNdCgpPcGVuIEFQSSAzLjAgYnVpbGRlciBhbmQgdmFsaWRhdGlvbiBsaWJyYXJ5IGZvciBQSFAgdGhhdCBoZWxwcyB5b3Ugd3JpdGUgdmFsaWQgc3BlY3MuCgpbUFNSLTFdOiBodHRwczovL2dpdGh1Yi5jb20vcGhwLWZpZy9maWctc3RhbmRhcmRzL2Jsb2IvbWFzdGVyL2FjY2VwdGVkL1BTUi0xLWJhc2ljLWNvZGluZy1zdGFuZGFyZC5tZApbUFNSLTJdOiBodHRwczovL2dpdGh1Yi5jb20vcGhwLWZpZy9maWctc3RhbmRhcmRzL2Jsb2IvbWFzdGVyL2FjY2VwdGVkL1BTUi0yLWNvZGluZy1zdHlsZS1ndWlkZS5tZApbUFNSLTRdOiBodHRwczovL2dpdGh1Yi5jb20vcGhwLWZpZy9maWctc3RhbmRhcmRzL2Jsb2IvbWFzdGVyL2FjY2VwdGVkL1BTUi00LWF1dG9sb2FkZXIubWQKCgpUaGlzIHByb2plY3QgaXMgY29tcGxpYW50IHdpdGggW1BTUi0xXSwgW1BTUi0yXSBhbmQgW1BTUi00XS4KSWYgeW91IG5vdGljZSBjb21wbGlhbmNlIG92ZXJzaWdodHMsIHBsZWFzZSBzZW5kIGEgcGF0Y2ggdmlhIHB1bGwgcmVxdWVzdC4KCiMjIEZlYXR1cmVzCgotIEZ1bGx5IGRvY3VtZW50ZWQgb2JqZWN0LW9yaWVudGVkIHJlcHJlc2VudGF0aW9uIG9mIHRoZSAKW09wZW4gQVBJIDMuMCtdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL3RyZWUvbWFzdGVyL3ZlcnNpb25zKSBzcGVjaWZpY2F0aW9uIHdpdGggaGVscGVyIG1ldGhvZHMKdG8gd3JpdGUgdmFsaWQgZG9jdW1lbnRzLgotIFN1cHBvcnRzIElsbHVtaW5hdGUgKExhcmF2ZWwpIFtgSnNvbmFibGVgXShodHRwczovL2dpdGh1Yi5jb20vaWxsdW1pbmF0ZS9jb250cmFjdHMvYmxvYi92NS40LjAvU3VwcG9ydC9Kc29uYWJsZS5waHApIAphbmQgW2BBcnJheWFibGVgXShodHRwczovL2dpdGh1Yi5jb20vaWxsdW1pbmF0ZS9jb250cmFjdHMvYmxvYi92NS40LjAvU3VwcG9ydC9BcnJheWFibGUucGhwKS4KLSBHZW5lcmF0ZXMgYW4gc3BlY2lmaWNhdGlvbiBpbiBwbGFpbiBQSFAgYXJyYXlzLCBwbGFpbiBvYmplY3RzLCBKU09OIG9yIFlBTUwuCi0gVmFsaWRhdGVzIE9wZW4gQVBJIGRvY3VtZW50cyBhZ2FpbnN0IHRoZSBPcGVuIEFQSSAzLjAueCBKU09OIFNjaGVtYS4KCgojIyBJbnN0YWxsCgpWaWEgQ29tcG9zZXIKCmBgYCBiYXNoCiQgY29tcG9zZXIgcmVxdWlyZSBlcmFzeXMvb3BlbmFwaS1waHAKYGBgCgpWaWEgR2l0CgpgYGAgYmFzaAokIGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vZXJhc3lzL29wZW5hcGktcGhwLmdpdApgYGAKCiMjIFVzYWdlCgpCYXNpYyBleGFtcGxlOgoKYGBgcGhwCjw/cGhwCgp1c2UgZXJhc3lzXE9wZW5BcGlcU3BlY1x2MyBhcyBPQVN2MzsKCiRkb2MgPSBuZXcgT0FTdjNcRG9jdW1lbnQoCiAgICBuZXcgT0FTdjNcSW5mbygnTXkgQVBJJywgJzEuMC4wJywgJ015IEFQSSBkZXNjcmlwdGlvbicpLAogICAgWwogICAgICAgICcvZm9vL2JhcicgPT4gbmV3IE9BU3YzXFBhdGhJdGVtKAogICAgICAgICAgICBbCiAgICAgICAgICAgICAgICAnZ2V0JyA9PiBuZXcgT0FTdjNcT3BlcmF0aW9uKAogICAgICAgICAgICAgICAgICAgIFsKICAgICAgICAgICAgICAgICAgICAgICAgJzIwMCcgPT4gbmV3IE9BU3YzXFJlc3BvbnNlKCdTdWNjZXNzZnVsIHJlc3BvbnNlLicpLAogICAgICAgICAgICAgICAgICAgICAgICAnZGVmYXVsdCcgPT4gbmV3IE9BU3YzXFJlc3BvbnNlKCdEZWZhdWx0IGVycm9yIHJlc3BvbnNlLicpLAogICAgICAgICAgICAgICAgICAgIF0KICAgICAgICAgICAgICAgICksCiAgICAgICAgICAgIF0KICAgICAgICApLAogICAgXQopOwoKJHlhbWwgPSAkZG9jLT50b1lhbWwoKTsKJGpzb24gPSAkZG9jLT50b0pzb24oKTsKJGFyciAgPSAkZG9jLT50b0FycmF5KCk7CiRvYmogID0gJGRvYy0+dG9PYmplY3QoKTsKCmBgYAoKIyMgVGVzdGluZwoKYGBgIGJhc2gKJCBjb21wb3NlciB0ZXN0CmBgYAoKb3IKCmBgYCBiYXNoCiQgdmVuZG9yL2Jpbi9waHB1bml0CiQgdmVuZG9yL2Jpbi9waHBjcwpgYGAKCiMjIENvbnRyaWJ1dGluZwoKUGxlYXNlIHNlZSBbQ09OVFJJQlVUSU5HXShodHRwczovL2dpdGh1Yi5jb20vZXJhc3lzL29wZW5hcGktcGhwL2Jsb2IvbWFzdGVyL0NPTlRSSUJVVElORy5tZCkgZm9yIGRldGFpbHMuCgojIyBMaWNlbnNlCgpUaGUgTUlUIExpY2Vuc2UgKE1JVCkuClBsZWFzZSBzZWUgW0xpY2Vuc2UgRmlsZV0oaHR0cHM6Ly9naXRodWIuY29tL2VyYXN5cy9vcGVuYXBpLXBocC9ibG9iL21hc3Rlci9MSUNFTlNFKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCgpbaWNvLXZlcnNpb25dOiBodHRwczovL2ltZy5zaGllbGRzLmlvL3BhY2thZ2lzdC92L2VyYXN5cy9vcGVuYXBpLXBocC5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUKW2ljby1saWNlbnNlXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1icmlnaHRncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUKW2ljby10cmF2aXNdOiBodHRwczovL2ltZy5zaGllbGRzLmlvL3RyYXZpcy9lcmFzeXMvb3BlbmFwaS1waHAvbWFzdGVyLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZQpbaWNvLWRvd25sb2Fkc106IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L2R0L2VyYXN5cy9vcGVuYXBpLXBocC5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUKW2ljby1jb3ZlcmFnZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vc2NydXRpbml6ZXIvY292ZXJhZ2UvZy9lcmFzeXMvb3BlbmFwaS1waHAuc3ZnP3N0eWxlPWZsYXQtc3F1YXJlCltpY28tc2NydXRpbml6ZXJdOiBodHRwczovL2ltZy5zaGllbGRzLmlvL3NjcnV0aW5pemVyL2cvZXJhc3lzL29wZW5hcGktcGhwLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZQpbbGluay1yZWxlYXNlc106IGh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9lcmFzeXMvb3BlbmFwaS1waHAKW2xpbmstdHJhdmlzXTogaHR0cHM6Ly90cmF2aXMtY2kub3JnL2VyYXN5cy9vcGVuYXBpLXBocApbbGluay1kb3dubG9hZHNdOiBodHRwczovL3BhY2thZ2lzdC5vcmcvcGFja2FnZXMvZXJhc3lzL29wZW5hcGktcGhwCltsaW5rLWNvdmVyYWdlXTogaHR0cHM6Ly9zY3J1dGluaXplci1jaS5jb20vZy9lcmFzeXMvb3BlbmFwaS1waHAvY29kZS1zdHJ1Y3R1cmUKW2xpbmstc2NydXRpbml6ZXJdOiBodHRwczovL3NjcnV0aW5pemVyLWNpLmNvbS9nL2VyYXN5cy9vcGVuYXBpLXBocAo= readmeEtag: '"a68dd99cb7a4677ca15321a9279bf5bc88ee1cc5"' readmeLastModified: Fri, 03 Nov 2023 09:41:25 GMT repositoryId: 129788604 description: >- 📚Swagger / Open API 3.0 builder and validation library for PHP that helps you write valid specs. created: '2018-04-16T18:32:26Z' updated: '2024-10-19T14:24:07Z' language: PHP archived: false stars: 20 watchers: 12 forks: 13 owner: erasys logo: https://avatars.githubusercontent.com/u/1845621?v=4 license: MIT repoEtag: '"7f5b5958acda76b1e2bc9d930edd2df5e587c548f5b9aba33217efbd670438b7"' repoLastModified: Sat, 19 Oct 2024 14:24:07 GMT foundInMaster: true category: Data Validators id: acce8eb5c93c964fa251e3fed927c64d - source: openapi3 tags repository: https://github.com/hunyadi/pyopenapi v3: true id: 9a608e628c0c1af0db91578fd6850e9f repositoryMetadata: base64Readme: >- # Generate an OpenAPI specification from a Python class

*PyOpenAPI* produces an OpenAPI specification in JSON, YAML or HTML format with endpoint definitions extracted from member functions of a strongly-typed Python class.

## Features

* supports standard and asynchronous functions (`async def`)
* maps function name prefixes such as `get_` or `create_` to HTTP GET, POST, PUT, DELETE, PATCH
* handles both simple and composite types (`int`, `str`, `Enum`, `@dataclass`)
* handles generic types (`list[T]`, `dict[K, V]`, `Optional[T]`, `Union[T1, T2, T3]`)
* maps Python positional-only and keyword-only arguments (of simple types) to path and query parameters, respectively
* maps composite types to HTTP request body
* supports user-defined routes, request and response samples with decorator `@webmethod`
* extracts description text from class and function doc-strings (`__doc__`)
* recognizes parameter description text given in reStructuredText doc-string format (`:param name: ...`)
* converts exceptions declared in doc-strings into HTTP 4xx and 5xx responses (e.g. `:raises TypeError:`)
* recursively converts composite types into JSON schemas
* groups frequently used composite types into a separate section and re-uses them with `$ref`
* displays generated OpenAPI specification in HTML with [ReDoc](https://github.com/Redocly/redoc)

## Live examples

* [Endpoint definition in Python](https://hunyadi.github.io/pyopenapi/examples/python/index.html)
* [Generated OpenAPI specification in JSON](https://hunyadi.github.io/pyopenapi/examples/json/index.html)
* [Generated OpenAPI specification in YAML](https://hunyadi.github.io/pyopenapi/examples/yaml/index.html)
* [Generated OpenAPI specification in HTML with ReDoc](https://hunyadi.github.io/pyopenapi/examples/index.html)

## User guide

### The specification object

In order to generate an [OpenAPI specification document](https://spec.openapis.org/oas/latest.html), you should first construct a `Specification` object, which encapsulates the formal definition:

```python
specification = Specification(
    MyEndpoint,
    Options(
        server=Server(url="http://example.com/api"),
        info=Info(
            title="Example specification",
            version="1.0",
            description=description,
        ),
        default_security_scheme=SecuritySchemeHTTP(
            "Authenticates a request by verifying a JWT (JSON Web Token) passed in the `Authorization` HTTP header.",
            "bearer",
            "JWT",
        ),
        extra_types=[ExampleType, UnreferencedType],
        error_wrapper=True,
    ),
)
```

The first argument to `Specification` is a Python class (`type`) whose methods will be inspected and converted into OpenAPI endpoint operations. The second argument is additional options that fine-tune how the specification is generated.

### Defining endpoint operations

Let's take a look at the definition of a simple endpoint called `JobManagement`:

```python
class JobManagement:
    def create_job(self, items: list[URL]) -> uuid.UUID:
        ...

    def get_job(self, job_id: uuid.UUID, /, format: Format) -> Job:
        ...

    def remove_job(self, job_id: uuid.UUID, /) -> None:
        ...

    def update_job(self, job_id: uuid.UUID, /, job: Job) -> None:
        ...
```

The name of each method begins with a prefix such as `create`, `get`, `remove` or `update`, each of which maps to an HTTP verb, e.g. `POST`, `GET`, `DELETE` or `PATCH`. The rest of the function name serves as an identifier, e.g. `job`. The `self` argument to the function is ignored. Other arguments indicate what path and query [parameter objects](https://spec.openapis.org/oas/latest.html#parameter-object), and what [HTTP request body](https://spec.openapis.org/oas/latest.html#request-body-object) the operation accepts.

### Function signatures for operations

Function signatures for operations must have full type annotation, including parameter types and return type.

Python [positional-only arguments](https://peps.python.org/pep-0570/) map to path parameters. Python positional-or-keyword arguments map to query parameters if they are of a simple type (e.g. `int` or `str`). If a composite type (e.g. a class, a list or a union) occurs in the Python parameter list, it is treated as the definition of the HTTP request body. Only one composite type may appear in the parameter list. The return type of the function is treated as the HTTP response body. If the function returns `None`, it corresponds to an HTTP response with no payload (i.e. a `Content-Length` of 0).

The JSON schema for the HTTP request and response body is generated with the library [json_strong_typing](https://github.com/hunyadi/strong_typing), and is automatically embedded in the OpenAPI specification document.

### User-defined operation path

By default, the library constructs the operation path from the Python function name and positional-only parameters. However, it is possible to supply a custom path (route) using the `@webmethod` decorator:

```python
@webmethod(
    route="/person/name/{family}/{given}",
)
def get_person_by_name(self, family: str, given: str, /) -> Person:
    ...
```

The custom path must have placeholders for all positional-only parameters in the function signature, and vice versa.

### Documenting operations

Use Python ReST (ReStructured Text) doc-strings to attach documentation to operations:

```python
def get_job(self, job_id: uuid.UUID, /, format: Format) -> Job:
    """
    Query status information about a job.

    :param job_id: Unique identifier for the job to query.
    :returns: Status information about the job.
    :raises NotFoundError: The job does not exist.
    :raises ValidationError: The input is malformed.
    """
    ...
```

Fields such as `param` and `returns` help document path and query parameters, HTTP request and response body. The field `raises` helps document error responses by identifying the exact return type when an error occurs. The Python type for `returns` and `raises` is translated to a JSON schema and embedded in the OpenAPI specification document.

### Request and response examples

OpenAPI supports specifying [examples](https://spec.openapis.org/oas/latest.html#example-object) for the HTTP request and response body of endpoint operations. This is supported via the `@webmethod` decorator:

```python
    @webmethod(
        route="/member/name/{family}/{given}",
        response_examples=[
            Student("Szörnyeteg", "Lajos"),
            Student("Ló", "Szerafin"),
            Student("Bruckner", "Szigfrid"),
            Student("Nagy", "Zoárd"),
            Teacher("Mikka", "Makka", "Négyszögletű Kerek Erdő"),
            Teacher("Vacska", "Mati", "Négyszögletű Kerek Erdő"),
        ],
    )
    def get_member_by_name(self, family: str, given: str, /) -> Union[Student, Teacher]:
        ...
```

A response example may be an exception or error class (a type that derives from `Exception`). These are usually shown under an HTTP status code of 4xx or 5xx.

The Python objects in `request_examples` and `response_examples` are translated to JSON with the library [json_strong_typing](https://github.com/hunyadi/strong_typing).

### Mapping function name prefixes to HTTP verbs

The following table identifies which function name prefixes map to which HTTP verbs:

| Prefix | HTTP verb   |
| ------ | ----------- |
| create | POST        |
| delete | REMOVE      |
| do     | GET or POST |
| get    | GET         |
| post   | POST        |
| put    | POST        |
| remove | REMOVE      |
| set    | PUT         |
| update | PATCH       |

If the function signature conflicts with the HTTP verb (e.g. a function name starts with `get` but has a composite type in the parameter list, which maps to a non-empty HTTP request body), the HTTP verb is automatically adjusted.

### Associating HTTP status codes with response types

By default, the library associates success responses with HTTP status code 200, and error responses with HTTP status code 500. However, it is possible to associate any Python type with any HTTP status code:

```python
specification = Specification(
    MyEndpoint,
    Options(
        server=Server(url="http://example.com/api"),
        info=Info(
            title="Example specification",
            version="1.0",
            description=description,
        ),
        success_responses={
            Student: HTTPStatus.CREATED,
            Teacher: HTTPStatus.ACCEPTED,
        },
        error_responses={
            AuthenticationError: HTTPStatus.UNAUTHORIZED,
            BadRequestError: 400,
            InternalServerError: 500,
            NotFoundError: HTTPStatus.NOT_FOUND,
            ValidationError: "400",
        },
        error_wrapper=True,
    ),
)
```

The arguments `success_responses` and `error_responses` take a dictionary that maps types to status codes. Status codes may be integers (e.g. `400`), strings (e.g. `"400"` or `"4xx"`) or [HTTPStatus](https://docs.python.org/3/library/http.html#http.HTTPStatus) enumeration values. The string representation of the status code must be valid as per the OpenAPI specification.
 readmeEtag: '"f896afc900eeb4fcf79e22c548548d79c479c6f3"' readmeLastModified: Mon, 07 Apr 2025 09:44:51 GMT repositoryId: 429913601 description: Generate an OpenAPI specification from a Python class definition created: '2021-11-19T19:34:39Z' updated: '2025-10-09T09:33:33Z' language: Python archived: false stars: 23 watchers: 1 forks: 2 owner: hunyadi logo: https://avatars.githubusercontent.com/u/7794010?v=4 license: MIT repoEtag: '"5256f9dee04a2642768147892c9d4a4b553061eb73eacbf876b7252612db040f"' repoLastModified: Thu, 09 Oct 2025 09:33:33 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/antonk52/swaggerlint v3: true repositoryMetadata: base64Readme: >- # Swaggerlint

`Swaggerlint` helps you to have a consistent API style by linting your swagger / OpenAPI Scheme.

<p align="center"><img src="https://user-images.githubusercontent.com/5817809/72013495-0b443700-326f-11ea-9549-84dce1ec861e.png" width="750" alt="npm command"></p>

## Installation

Install it in your project

```sh
npm install swaggerlint
```

Install it globally

```sh
npm install --global swaggerlint
```

## Usage

### CLI

You can lint your swagger scheme by path

```sh
swaggerlint /path/to/swagger.json
```

Or by providing a URL

```sh
swaggerlint https://...
```

#### Config flag

`swaggerlint` will automatically search up the directory tree for a `swaggerlint.config.js` file. Or you can specify it explicitly

```sh
swaggerlint --config /path/to/swaggerlint.config.js
```

### Nodejs

```js
const {swaggerlint} = require('swaggerlint');
const config = require('./configs/swaggerlint.config.js');
const swaggerScheme = require('./swagger.json');

const result = swaggerlint(swaggerScheme, config);

console.log(result); // an array or errors

/**
 * [{
 *   name: 'string', // rule name
 *   msg: 'string', // message from the rule checker
 *   location: ['path', 'to', 'error'] // what caused an error
 * }]
 */
```

### Docker image

If you do not have nodejs installed you can use the [swaggerlint docker image](https://hub.docker.com/r/antonk52/alpine-swaggerlint).

## Config

```js
// swaggerlint.config.js
module.exports = {
    rules: {
        'object-prop-casing': ['camel'],
        'properties-for-object-type': true,
        'latin-definitions-only': true,
    },
};
```

## Rules

You can set any rule value to `false` to disable it or to `true` to enable and set its setting to default value.

<!-- GENERATED_START(id:rulestable;hash:78f3b724d8ef667a4bcef20d6c92d22e) This is generated content, do not modify by hand, to regenerate run "npm run updateDocs" -->

| rule name                                                                                | description                                                                    | default                                     |
| ---------------------------------------------------------------------------------------- | ------------------------------------------------------------------------------ | ------------------------------------------- |
| [`expressive-path-summary`](./src/rules/expressive-path-summary/readme.md)               | Enforces an intentional path summary                                           |                                             |
| [`latin-definitions-only`](./src/rules/latin-definitions-only/readme.md)                 | Bans non Latin characters usage in definition names                            | ["placeholder_to_be_removed",{"ignore":[]}] |
| [`no-empty-object-type`](./src/rules/no-empty-object-type/readme.md)                     | Object types have to have their properties specified explicitly                |                                             |
| [`no-external-refs`](./src/rules/no-external-refs/readme.md)                             | Forbids the usage of external ReferenceObjects                                 |                                             |
| [`no-inline-enums`](./src/rules/no-inline-enums/readme.md)                               | Enums must be in `DefinitionsObject` or `ComponentsObject`                     |                                             |
| [`no-ref-properties`](./src/rules/no-ref-properties/readme.md)                           | Disallows to have additional properties in Reference objects                   |                                             |
| [`no-single-allof`](./src/rules/no-single-allof/readme.md)                               | Object types should not have a redundant single `allOf` property               |                                             |
| [`no-trailing-slash`](./src/rules/no-trailing-slash/readme.md)                           | URLs must NOT end with a slash                                                 |                                             |
| [`object-prop-casing`](./src/rules/object-prop-casing/readme.md)                         | Casing for your object property names                                          | ["camel"]                                   |
| [`only-valid-mime-types`](./src/rules/only-valid-mime-types/readme.md)                   | Checks mime types against known from [`mime-db`](https://npm.im/mime-db)       |                                             |
| [`parameter-casing`](./src/rules/parameter-casing/readme.md)                             | Casing for your parameters                                                     | ["camel",{"header":"kebab"}]                |
| [`path-param-required-field`](./src/rules/path-param-required-field/readme.md)           | Helps to keep consistently set optional `required` property in path parameters |                                             |
| [`required-operation-tags`](./src/rules/required-operation-tags/readme.md)               | All operations must have tags                                                  |                                             |
| [`required-parameter-description`](./src/rules/required-parameter-description/readme.md) | All parameters must have description                                           |                                             |
| [`required-tag-description`](./src/rules/required-tag-description/readme.md)             | All tags must have description                                                 |                                             |

<!-- GENERATED_END(id:rulestable) -->

## Documentation

-   [How to write a rule](./docs/how-to-write-a-rule.md)

### Acknowledgments

This tool has been inspired by already existing swagger validation checkers:

-   [api lint](https://github.com/danielgtaylor/apilint)
-   [speccy](https://github.com/wework/speccy)
-   [zally](https://github.com/zalando/zally)
-   [openapi-validator](https://github.com/IBM/openapi-validator)
 readmeEtag: '"43e61a83ca19a938628f274362a0f9069a40ac9e"' readmeLastModified: Fri, 10 Mar 2023 13:40:44 GMT repositoryId: 227476747 description: Keep your API consistent created: '2019-12-11T23:03:15Z' updated: '2023-08-22T16:29:55Z' language: TypeScript archived: false stars: 20 watchers: 1 forks: 1 owner: antonk52 logo: https://avatars.githubusercontent.com/u/5817809?v=4 repoEtag: '"7e1db7f844552fb527e7a935c474680eea29615a4abb57ba4804ad7fed36720e"' repoLastModified: Tue, 22 Aug 2023 16:29:55 GMT foundInMaster: true category: Description Validators id: ab3e4e31e5204b176458f2b5fc9126e7 - source: openapi3 tags repository: https://github.com/mockoon/mockoon.com v3: true repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL21vY2tvb24uY29tIiBhbHQ9Im1vY2tvb24gbG9nbyI+CiAgICA8aW1nIHdpZHRoPSIyMDAiIGhlaWdodD0iMjAwIiBzcmM9Imh0dHBzOi8vbW9ja29vbi5jb20vaW1hZ2VzL2xvZ28tc3F1YXJlLXdlYnNpdGUucG5nIj4KICA8L2E+CiAgPGJyPgogIDxhIGhyZWY9Imh0dHBzOi8vbW9ja29vbi5jb20vZG93bmxvYWQvIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9Eb3dubG9hZCUyMGFwcC1Hby1ncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUmY29sb3JCPTE5OTdjNiIvPjwvYT4KICA8YSBocmVmPSJodHRwczovL21vY2tvb24uY29tLyI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvV2Vic2l0ZS1Hby1ncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUmY29sb3JCPTE5OTdjNiIvPjwvYT4KICA8YSBocmVmPSJodHRwczovL21vY2tvb24uY29tL25ld3NsZXR0ZXIvIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9OZXdzbGV0dGVyLVN1YnNjcmliZS1ncmVlbi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUiLz48L2E+CiAgPGJyPgogIDxicj4KICA8aDE+TW9ja29vbidzIHdlYnNpdGUgcmVwb3NpdG9yeTwvaDE+CjwvZGl2PgoKVGhpcyByZXBvc2l0b3J5IGlzIGhvbWUgb2YgTW9ja29vbidzIGRvY3VtZW50YXRpb24sIHR1dG9yaWFscyBhbmQgYmxvZyBwb3N0cy4KCi0tLQoKIyMgU3VwcG9ydC9mZWVkYmFjawoKWW91IGNhbiBkaXNjdXNzIGFsbCB0aGluZ3MgcmVsYXRlZCB0byBNb2Nrb29uLCBhbmQgYXNrIGZvciBoZWxwLCBvbiB0aGUgW29mZmljaWFsIGNvbW11bml0eV0oaHR0cHM6Ly9naXRodWIuY29tL21vY2tvb24vbW9ja29vbi9kaXNjdXNzaW9ucykuIEl0J3MgYWxzbyBhIGdvb2QgcGxhY2UgdG8gZGlzY3VzcyBidWdzIGFuZCBmZWF0dXJlIHJlcXVlc3RzIGJlZm9yZSBvcGVuaW5nIGFuIGlzc3VlIG9uIHRoaXMgcmVwb3NpdG9yeS4KCiMjIENvbnRyaWJ1dGluZwoKSWYgeW91IGFyZSBpbnRlcmVzdGVkIGluIGNvbnRyaWJ1dGluZyB0byBNb2Nrb29uIHdlYnNpdGUgYW5kIGRvY3VtZW50YXRpb24sIHBsZWFzZSB0YWtlIGEgbG9vayBhdCB0aGUgW2NvbnRyaWJ1dGluZyBndWlkZWxpbmVzXShodHRwczovL2dpdGh1Yi5jb20vbW9ja29vbi9tb2Nrb29uLmNvbS9ibG9iL21haW4vQ09OVFJJQlVUSU5HLm1kKS4KClBsZWFzZSBhbHNvIHRha2UgYSBsb29rIGF0IG91ciBbQ29kZSBvZiBDb25kdWN0XShodHRwczovL2dpdGh1Yi5jb20vbW9ja29vbi8uZ2l0aHViL2Jsb2IvbWFpbi9DT0RFX09GX0NPTkRVQ1QubWQpLgoKIyMgRG9jdW1lbnRhdGlvbgoKWW91IHdpbGwgZmluZCBNb2Nrb29uJ3MgW2RvY3VtZW50YXRpb25dKGh0dHBzOi8vbW9ja29vbi5jb20vZG9jcy9sYXRlc3QvKSBvbiB0aGUgb2ZmaWNpYWwgd2Vic2l0ZS4gSXQgY292ZXJzIE1vY2tvb24ncyBtb3N0IGNvbXBsZXggZmVhdHVyZXMuIEZlZWwgZnJlZSB0byBjb250cmlidXRlIG9yIGFzayBmb3IgbmV3IHRvcGljcyB0byBiZSBjb3ZlcmVkLgoKIyMgUm9hZG1hcAoKSWYgeW91IHdhbnQgdG8ga25vdyB3aGF0IHdpbGwgYmUgY29taW5nIGluIHRoZSBuZXh0IHJlbGVhc2UgeW91IGNhbiBjaGVjayB0aGUgZ2xvYmFsIFtSb2FkbWFwXShodHRwczovL21vY2tvb24uY29tL3B1YmxpYy1yb2FkbWFwLykgb3IgW3N1YnNjcmliZSB0byBvdXIgbmV3c2xldHRlcl0oaHR0cHM6Ly9tb2Nrb29uLmNvbS9uZXdzbGV0dGVyLykuCg== readmeEtag: '"e2d0f3d2c5ac5f13bc35498c9d5136cf56c2ce77"' readmeLastModified: Tue, 09 Sep 2025 20:50:44 GMT repositoryId: 98350934 description: Mockoon application official website repository and documentation created: '2017-07-25T21:19:22Z' updated: '2026-02-05T16:38:15Z' language: TypeScript archived: false stars: 24 watchers: 1 forks: 18 owner: mockoon logo: https://avatars.githubusercontent.com/u/49429147?v=4 license: MIT repoEtag: '"d8c641a5b88556822827bc8d5ce0052cb7223e16bf4c347c7090045f7928ba41"' repoLastModified: Thu, 05 Feb 2026 16:38:15 GMT foundInMaster: true category: Testing id: 84d3391fa9deedcd6398666ba504eb78 - source: openapi3 tags repository: https://github.com/orsinium-labs/valdo v3: true id: 65eda01d79d1e4a40435ef8f687e6ff2 repositoryMetadata: base64Readme: >- IyB2YWxkbwoKWyBb8J+TmiBkb2NzXShodHRwczovL3BrZy5nby5kZXYvZ2l0aHViLmNvbS9vcnNpbml1bS1sYWJzL3ZhbGRvL3ZhbGRvKSBdIFsgW/CfkJkgZ2l0aHViXShodHRwczovL2dpdGh1Yi5jb20vb3JzaW5pdW0tbGFicy92YWxkbykgXQoKR28gcGFja2FnZSBmb3IgdmFsaWRhdGluZyBKU09OLiBDYW4gZ2VuZXJhdGUgW0pTT04gU2NoZW1hXShodHRwczovL2pzb24tc2NoZW1hLm9yZy9vdmVydmlldy93aGF0LWlzLWpzb25zY2hlbWEpICgxMDAlIGNvbXBhdGlibGUgd2l0aCBbT3BlbkFQSV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKSksIHByb2R1Y2VzIHVzZXItZnJpZW5kbHkgZXJyb3JzLCBzdXBwb3J0cyB0cmFuc2xhdGlvbnMuCgpZb3UgY291bGQgd3JpdGUgT3BlbkFQSSBkb2N1bWVudGF0aW9uIGJ5IGhhbmQgKHdoaWNoIGlzIHZlcnkgcGFpbmZ1bCkgYW5kIHRoZW4gdXNlIGl0IHRvIHZhbGlkYXRlIHVzZXIgaW5wdXQgaW4geW91ciBIVFRQIHNlcnZpY2UsIGJ1dCB0aGVuIGVycm9yIG1lc3NhZ2VzIGFyZSB2ZXJ5IGNvbmZ1c2luZywgbm90IHVzZXItZnJpZW5kbHksIGFuZCBvbmx5IGluIEVuZ2xpc2guIE9yIHlvdSBjb3VsZCB3cml0ZSBpbnB1dCB2YWxpZGF0aW9uIGJ5IGhhbmQgYW5kIHRoZW4gbWFpbnRhaW4gdGhlIE9wZW5BUEkgZG9jdW1lbnRhdGlvbiBzZXBhcmF0ZWx5IGJ1dCB0aGVuIHRoZSB0d28gd2lsbCBldmVudHVhbGx5IGRyaWZ0IGFuZCB5b3VyIGRvY3VtZW50YXRpb24gd2lsbCBiZSBhIGxpZS4gVmFsZG8gc29sdmVzIGFsbCB0aGVzZSBwcm9ibGVtczogd3JpdGUgdmFsaWRhdGlvbiBvbmNlIHVzaW5nIGEgcmVhbCBwcm9ncmFtbWluZyBsYW5ndWFnZSwgdXNlIGl0IGV2ZXJ5d2hlcmUuCgpGZWF0dXJlczoKCiogTWVjaGFuaXNtIHRvIHRyYW5zbGF0ZSBlcnJvciBtZXNzYWdlcy4KKiBPdXQtb2YtdGhlLWJveCB0cmFuc2xhdGlvbnMgZm9yIHNvbWUgbGFuZ3VhZ2VzLgoqIFN1cHBvcnRzIHRoZSBsYXRlc3QgSlNPTiBTY2hlbWEgc3BlY2lmaWNhdGlvbiAoMjAyMC0xMikuCiogUHVyZSBHby4KKiBObyBjb2RlIGdlbmVyYXRpb24sIG5vIHJlZmxlY3Rpb24sIG5vIHVuc2FmZSBjb2RlLgoqIFVzZXItZnJpZW5kbHkgZXJyb3IgbWVzc2FnZXMuCiogQ29uY3VycmVuY3ktc2FmZSwgbm8gZ2xvYmFsIHN0YXRlLgoqIFN0cmljdCBieSBkZWZhdWx0LCB3aXRob3V0IGltcGxpY2l0IHR5cGUgY2FzdGluZy4KKiBUeXBlLXNhZmUsIHRoYW5rcyB0byBnZW5lcmljcy4KCiMjIEluc3RhbGxhdGlvbgoKYGBgYmFzaApnbyBnZXQgZ2l0aHViLmNvbS9vcnNpbml1bS1sYWJzL3ZhbGRvCmBgYAoKIyMgVXNhZ2UKCmBgYGdvCnZhbGlkYXRvciA6PSB2YWxkby5PYmplY3QoCiAgICB2YWxkby5Qcm9wZXJ0eSgibmFtZSIsIHZhbGRvLlN0cmluZyh2YWxkby5NaW5MZW4oMSkpKSwKICAgIHZhbGRvLlByb3BlcnR5KCJhZG1pbiIsIHZhbGRvLkJvb2woKSksCikKCi8vIHZhbGlkYXRlIEpTT04KaW5wdXQgOj0gW11ieXRlKGB7Im5hbWUiOiAiYXJhZ29ybiIsICJhZG1pbiI6IHRydWV9YCkKZXJyIDo9IHZhbGRvLlZhbGlkYXRlKHZhbGlkYXRvciwgcmF3KQoKLy8gdmFsaWRhdGUgYW5kIHVubWFyc2hhbCBKU09OCnR5cGUgVXNlciBzdHJ1Y3QgewogICAgTmFtZSAgc3RyaW5nIGBqc29uOiJuYW1lImAKICAgIEFkbWluIGJvb2wgICBganNvbjoiYWRtaW4iYAp9CnVzZXIsIGVyciA6PSB2YWxkby5Vbm1hcnNoYWxbVXNlcl0odmFsaWRhdG9yLCBpbnB1dCkKCi8vIGdlbmVyYXRlIEpTT04gU2NoZW1hCnNjaGVtYSA6PSB2YWxkby5TY2hlbWEodmFsaWRhdG9yKQpgYGAKClNlZSBbZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9wa2cuZ28uZGV2L2dpdGh1Yi5jb20vb3JzaW5pdW0tbGFicy92YWxkby92YWxkbykgZm9yIG1vcmUuCg== readmeEtag: '"095879191918ac2ed7ba8727b83b8c39a19a1b6b"' readmeLastModified: Fri, 31 Oct 2025 03:42:26 GMT repositoryId: 809004064 description: >- ✅ Go package for validating JSON. It's the first Go package that can generate JSON Schema (OpeanAPI-compatible), produces user-friendly errors, and supports translations. created: '2024-06-01T12:09:15Z' updated: '2025-11-20T15:38:11Z' language: Go archived: false stars: 19 watchers: 2 forks: 0 owner: orsinium-labs logo: https://avatars.githubusercontent.com/u/62876761?v=4 license: MIT repoEtag: '"c019b157235240de45bca145da09d171d199b37ee85ec1a81aeace7faad28e8f"' repoLastModified: Thu, 20 Nov 2025 15:38:11 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/lusingander/topi v3: true repositoryMetadata: base64Readme: >- IyB0b3BpCgpUZXJtaW5hbCBPcGVuQVBJIGRvY3VtZW50YXRpb24gdmlld2VyIPCfkJAKCiMjIEFib3V0Cgo+IHRvcGkgaXMgc3RpbGwgdW5kZXIgZGV2ZWxvcG1lbnQuLi4g8J+QkAoKdG9waSBpcyB0aGUgZG9jdW1lbnRhdGlvbiB2aWV3ZXIgZm9yIE9wZW5BUEkgdjMgZGVmaW5pdGlvbnMgaW4gdGhlIHRlcm1pbmFsLgoKPGltZyBzcmM9Ii4vaW1nL2ltYWdlLmdpZiIgd2lkdGg9ODAwPgoKKFRoaXMgaW1hZ2Ugc2hvdyBodHRwczovL2dpdGh1Yi5jb20vZ2l0aHViL3Jlc3QtYXBpLWRlc2NyaXB0aW9uKQoKIyMgSW5zdGFsbGF0aW9uCgpgJCBnbyBpbnN0YWxsIGdpdGh1Yi5jb20vbHVzaW5nYW5kZXIvdG9waUBsYXRlc3RgCgoocmVxdWlyZSBHbyAxLjE4KykKCiMjIFVzYWdlCgpgJCB0b3BpIDxwYXRoPmAKCj4gYHBhdGhgIGNhbiBiZSBsb2NhbCBmaWxlIHBhdGggb3IgcmVtb3RlIFVSTC4KCiMjIyBLZXliaW5kaW5ncwoKIyMjIyBDb21tb24KCmNvbW1vbiBrZXliaW5kaW5ncyBmb3IgYWxsIHBhZ2VzCgp8S2V5fERlc2NyaXB0aW9ufAp8LXwtfAp8PGtiZD5CYWNrc3BhY2U8L2tiZD58YmFjayB0byBwZXJ2IHBhZ2V8Cnw8a2JkPkN0cmwrYzwva2JkPnxxdWl0fAp8PGtiZD4/PC9rYmQ+fHNob3cgaGVscCBwYWdlfAoKIyMjIyBMaXN0IHBhZ2UKCmtleWJpbmRpbmdzIGZvciBsaXN0LXN5bGUgcGFnZXMgCgp8S2V5fERlc2NyaXB0aW9ufAp8LXwtfAp8PGtiZD5qPC9rYmQ+fGN1cnNvciBkb3dufAp8PGtiZD5rPC9rYmQ+fGN1cnNvciB1cHwKfDxrYmQ+Zjwva2JkPiA8a2JkPmw8L2tiZD58bmV4dCBwYWdlfAp8PGtiZD5iPC9rYmQ+IDxrYmQ+aDwva2JkPnxwcmV2IHBhZ2V8Cnw8a2JkPmc8L2tiZD58Z28gdG8gc3RhcnR8Cnw8a2JkPkc8L2tiZD58Z28gdG8gZW5kfAp8PGtiZD4vPC9rYmQ+fEVudGVyIGZpbHRlcmluZyBtb2RlfAp8PGtiZD5FbnRlcjwva2JkPnwoZGVmYXVsdCkgc2VsZWN0IGl0ZW0sIChmaWx0ZXJpbmcpIGFwcGx5IGZpbHRlcnwKfDxrYmQ+RXNjPC9rYmQ+fChmaWx0ZXJpbmcpIGNhbmNlbCBmaWx0ZXIsIChmaWx0ZXIgYXBwbGllZCkgcmVtb3ZlIGZpbHRlcnwKCiMjIyMgRG9jdW1lbnQgcGFnZQoKa2V5YmluZGluZ3MgZm9yIGRvY3VtZW50IHBhZ2VzIAoKfEtleXxEZXNjcmlwdGlvbnwKfC18LXwKfDxrYmQ+ajwva2JkPnxwYWdlIGRvd24gb25lIGxpbmV8Cnw8a2JkPms8L2tiZD58cGFnZSB1cCBvbmUgbGluZXwKfDxrYmQ+Zjwva2JkPnxwYWdlIGRvd258Cnw8a2JkPmI8L2tiZD58cGFnZSB1cHwKfDxrYmQ+ZDwva2JkPnxoYWxmIHBhZ2UgZG93bnwKfDxrYmQ+dTwva2JkPnxoYWxmIHBhZ2UgdXB8Cnw8a2JkPlRhYjwva2JkPnxzZWxlY3QgbGlua3wKfDxrYmQ+eDwva2JkPnxvcGVuIHNlbGVjdGluZyBsaW5rfAoKc3BlY2lmaWMgdG8gdGhlIGNyZWRpdHMgcGFnZQoKfEtleXxEZXNjcmlwdGlvbnwKfC18LXwKfDxrYmQ+dDwva2JkPnx0b2dnbGUgY3JlZGl0cyBsaXN0fAoKIyMgTGljZW5zZQoKTUlUCg== readmeEtag: '"ea4e07f1cf4fd5b52dc4b092a72088a0f71a95bf"' readmeLastModified: Sun, 26 Jun 2022 07:45:32 GMT repositoryId: 463991716 description: Terminal OpenAPI documentation viewer 🐐 created: '2022-02-26T23:31:07Z' updated: '2024-11-11T01:46:26Z' language: Go archived: false stars: 19 watchers: 1 forks: 0 owner: lusingander logo: https://avatars.githubusercontent.com/u/31386431?v=4 license: MIT repoEtag: '"4ba262133c5670d5e0d72db5d0eba0368ee12fd04b22e191b9269e81273340bd"' repoLastModified: Mon, 11 Nov 2024 01:46:26 GMT foundInMaster: true category: Documentation id: 236a30079db63d08dcedfb42b08f615d - source: openapi3 tags repository: https://github.com/mikekonan/go-types v3: true repositoryMetadata: base64Readme: >- WyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL21pa2Vrb25hbi9nby10eXBlcyldKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMpIFshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kuY29tL21pa2Vrb25hbi9nby10eXBlcy5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL3RyYXZpcy1jaS5jb20vbWlrZWtvbmFuL2dvLXR5cGVzKSBbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvbWlrZWtvbmFuL2dvLXR5cGVzL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj04M1EwNE9XNEkxKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL21pa2Vrb25hbi9nby10eXBlcykKIyBnby10eXBlcwpUaGlzIGxpYnJhcnkgaGFzIGJlZW4gY3JlYXRlZCB3aXRoIHRoZSBwdXJwb3NlIHRvIGZhY2lsaXRhdGUgdGhlIHN0b3JlLCB2YWxpZGF0aW9uLCBhbmQgdHJhbnNmZXIgb2YgR28gSVNPLTMxNjYvSVNPLTQyMTcvdGltZXpvbmVzL2VtYWlscy9VUkwgdHlwZXMuIFRoZXJlIGlzIGEgW29wZW5hcGkzIHNwZWNdKGh0dHBzOi8vZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvYmxvYi9tYWluL3N3YWdnZXIueWFtbCkgb2YgdGhhdCB0eXBlIGFuZCBtYWtlIHlvdSBhYmxlIHRvIGluY2x1ZGUgaXQgaW50byB5b3VyIHNwZWMuIEFsbCB0eXBlcyBoYXMgb3duIG96em8uVmFsaWRhdGUsIGpzb24uVW5tYXJzaGFsZXIsIFN0cmluZ2VyIGFuZCBkcml2ZXIuVmFsdWVyIGltcGxlbWVudGF0aW9ucy4KCiMgSW5zdGFsbGF0aW9uCmBgYGJhc2gKZ28gZ2V0IGdpdGh1Yi5jb20vbWlrZWtvbmFuL2dvLXR5cGVzL3YyCmBgYAojIFVzYWdlOgoKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgKAoJImVuY29kaW5nL2pzb24iCgkiZm10IgoJImxvZyIKCgl2YWxpZGF0aW9uICJnaXRodWIuY29tL2dvLW96em8vb3p6by12YWxpZGF0aW9uL3Y0IgoJImdpdGh1Yi5jb20vbWlrZWtvbmFuL2dvLXR5cGVzL3YyL2NvdW50cnkiCgkiZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvdjIvY291bnRyeS9hbHBoYTIiCgkiZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvdjIvY291bnRyeS9hbHBoYTMiCgkiZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvdjIvbGFuZ3VhZ2UiCgkiZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvdjIvY291bnRyeS9uYW1lIgoJImdpdGh1Yi5jb20vbWlrZWtvbmFuL2dvLXR5cGVzL3YyL2N1cnJlbmN5IgoJImdpdGh1Yi5jb20vbWlrZWtvbmFuL2dvLXR5cGVzL3YyL2N1cnJlbmN5L2NvZGUiCgkiZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvdjIvcGhvbmUiCgkiZ2l0aHViLmNvbS9taWtla29uYW4vZ28tdHlwZXMvdjIvcG9zdGFsX2NvZGUiCikKCi8vIDEuIHVzZSBpbiB5b3VyIHN0cnVjdHMKdHlwZSBVc2VyIHN0cnVjdCB7CglOYW1lICAgICAgICAgICAgc3RyaW5nICAgICAgICAgICAgICAgIGBqc29uOiJuYW1lIiBkYjoibmFtZSJgCglDb3VudHJ5ICAgICAgICAgY291bnRyeS5BbHBoYTJDb2RlICAgIGBqc29uOiJjb3VudHJ5IiBkYjoiY291bnRyeSJgCglDdXJyZW5jeSAgICAgICAgY3VycmVuY3kuQ29kZSAgICAgICAgIGBqc29uOiJjdXJyZW5jeSIgZGI6ImN1cnJlbmN5ImAKCUxhbmd1YWdlICAgICAgICBsYW5ndWFnZS5BbHBoYTJDb2RlICAgYGpzb246Imxhbmd1YWdlIiBkYjoibGFuZ3VhZ2UiYAoJUGhvbmUgICAgICAgICAgIHBob25lLk51bWJlciAgICAgICAgICBganNvbjoicGhvbmUiIGRiOiJwaG9uZSJgCglDb3VudHJ5RGlhbENvZGUgcGhvbmUuRGlhbENvZGUgICAgICAgIGBqc29uOiJkaWFsQ29kZSIgZGI6ImRpYWxDb2RlImAKCVBvc3RhbENvZGUgICAgICBwb3N0YWxjb2RlLlBvc3RhbENvZGUgYGpzb246InBvc3RhbENvZGUiIGRiOiJwb3N0YWxDb2RlImAKfQoKZnVuYyBtYWluKCkgewoJLy8gMi4gdXNlIGluIHlvdXIgd2lyZQoJdXNlciA6PSBVc2Vye30KCV8gPSBqc29uLlVubWFyc2hhbChbXWJ5dGUoYHsibmFtZSI6Im5hbWUiLCAiY291bnRyeSI6ICJDQSIsICJjdXJyZW5jeSI6ICJDQUQiLCAibGFuZ3VhZ2UiOiAiZnIiLCAicGhvbmUiOiAiMTIzNDU2Nzg5IiwgImRpYWxDb2RlIjogIjEifWApLCAmdXNlcikKCgkvLyAzLiBjaGVjayBpcyBzZXQKCXVzZXIuQ291bnRyeS5Jc1NldCgpCgl1c2VyLkN1cnJlbmN5LklzU2V0KCkKCXVzZXIuTGFuZ3VhZ2UuSXNTZXQoKQoKCS8vIDQuIHZhbGlkYXRlIHVzaW5nIG96em8tdmFsaWRhdGlvbgoJaWYgZXJyIDo9IHZhbGlkYXRpb24uVmFsaWRhdGVTdHJ1Y3QoJnVzZXIsIHZhbGlkYXRpb24uRmllbGQoJnVzZXIuQ291bnRyeSksIHZhbGlkYXRpb24uRmllbGQoJnVzZXIuQ3VycmVuY3kpKTsgZXJyICE9IG5pbCB7CiAgICAgICAgICAgIGxvZy5GYXRhbChlcnIpCgl9CgoJLy8gNS4gbG9va3VwIGJ5IGFscGhhMiwgYWxwaGEzLCBjb3VudHJ5IG5hbWUKCWlmIHVzZXJDb3VudHJ5LCBvayA6PSBjb3VudHJ5LkJ5QWxwaGEyQ29kZSh1c2VyLkNvdW50cnkpOyBvayB7CiAgICAgICAgICAgIGZtdC5QcmludGYoImNvdW50cnkgbmFtZSAtICclcycsIGFscGhhLTIgLSAnJXMnLCBhbHBoYS0zIC0gJyVzJyIsIHVzZXJDb3VudHJ5Lk5hbWUoKSwgdXNlckNvdW50cnkuQWxwaGEyQ29kZSgpLCB1c2VyQ291bnRyeS5BbHBoYTNDb2RlKCkpCgl9CgoJLy8gNi4gbG9va3VwIGJ5IDIgYW5kIDMgY2hhciBjb2RlcywgbGFuZ3VhZ2UgbmFtZQoJaWYgdXNlckxhbmd1YWdlLCBvayA6PSBsYW5ndWFnZS5CeUFscGhhMkNvZGUodXNlci5MYW5ndWFnZSk7IG9rIHsKICAgICAgICAgICAgZm10LlByaW50ZigibGFuZ3VhZ2UgbmFtZSAtICclcycsIGFscGhhLTIgLSAnJXMnLCBhbHBoYS0zIC0gJyVzJyIsIHVzZXJMYW5ndWFnZS5OYW1lKCksIHVzZXJMYW5ndWFnZS5BbHBoYTJDb2RlKCksIHVzZXJMYW5ndWFnZS5BbHBoYTNDb2RlKCkpCgl9CgoJLy8gNy4gbG9va3VwIGJ5IGNvdW50cnkgZGlhbCBjb2RlCglpZiBwaG9uZUNvdW50cmllcywgb2sgOj0gcGhvbmUuQ291bnRyaWVzQnlEaWFsQ29kZSh1c2VyLkNvdW50cnlEaWFsQ29kZSk7IG9rIHsKICAgICAgICAgICAgZm9yIF8sIHBob25lQ291bnRyeSA6PSByYW5nZSBwaG9uZUNvdW50cmllcyB7CiAgICAgICAgICAgICAgICBmbXQuUHJpbnRmKCJjb3VudHJ5IGJ5IGRpYWwgY29kZSAtICclcyciLCBwaG9uZUNvdW50cnkpCiAgICAgICAgICAgIH0KCX0KCgkvLyA4LiBsb29rdXAgYnkgY291bnRyeQoJaWYgZGlhbENvZGUsIG9rIDo9IHBob25lLkRpYWxCeUFscGhhMkNvZGUodXNlci5Db3VudHJ5KTsgb2sgewogICAgICAgICAgICBmbXQuUHJpbnRmKCInJXMnIGRpYWwgY29kZSBpcyAnJXMnIiwgdXNlci5Db3VudHJ5LCBkaWFsQ29kZSkKICAgICAgICB9CgoJLy8gOS4gbG9va3VwIGJ5IGN1cnJlbmN5IGNvZGUKCWlmIHVzZXJDdXJyZW5jeSwgb2sgOj0gY3VycmVuY3kuQnlDb2RlKHVzZXIuQ3VycmVuY3kpOyBvayB7CiAgICAgICAgICAgIGZtdC5QcmludGYoImN1cnJlbmN5IG5hbWUgLSAnJXMnLCBjb2RlIC0gJyVzJywgbnVtYmVyIC0gJyVzJywgY291bnRyaWVzIC0gJyVzJywgZGVjaW1hbCBwbGFjZXMgLSAnJWQnIiwKICAgICAgICAgICAgICAgIHVzZXJDdXJyZW5jeS5DdXJyZW5jeSgpLCB1c2VyQ3VycmVuY3kuQ29kZSgpLCB1c2VyQ3VycmVuY3kuTnVtYmVyKCksIHVzZXJDdXJyZW5jeS5Db3VudHJpZXMoKSwgdXNlckN1cnJlbmN5LkRlY2ltYWxQbGFjZXMoKSkKCX0KCgkvLyAxMC4gc3RvcmUgaW4gZGIKCWZtdC5QcmludGxuKHVzZXIuQ291bnRyeS5WYWx1ZSgpKSAgLy9wcmludHMgJ0NBJwoJZm10LlByaW50bG4odXNlci5DdXJyZW5jeS5WYWx1ZSgpKSAvL3ByaW50cyAnQ0FOJwoJZm10LlByaW50bG4odXNlci5MYW5ndWFnZS5WYWx1ZSgpKSAvL3ByaW50cyAnZnInCgoJLy8gMTEuIHVzZSBzcGVjaWZpYyBjb3VudHJ5IGNvbnN0YW50cwoJZm10LlByaW50bG4oY291bnRyeS5DYW5hZGEuQWxwaGEyQ29kZSgpKQoJZm10LlByaW50bG4oIm5hbWU6IiwgbmFtZS5DYW5hZGEpCglmbXQuUHJpbnRsbigiYWxwaGEtMjoiLCBhbHBoYTIuQ0EpCglmbXQuUHJpbnRsbigiYWxwaGEtMzoiLCBhbHBoYTMuQ0FOKQoKCS8vIDEyLiB1c2Ugc3BlY2lmaWMgY3VycmVuY3kgY29kZXMKCWZtdC5QcmludGxuKGNvZGUuQ0FEKQp9CmBgYAoKIyMgTGlua3M6Ci0gQ3VycmVuY3kgQ29kZXMgWyhJU08gNDIxNyldKGh0dHBzOi8vd3d3LmN1cnJlbmN5LWlzby5vcmcvZW4vaG9tZS90YWJsZXMvdGFibGUtYTEuaHRtbCkgLSBbd2lraV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvSVNPXzQyMTcpLCBbZGF0YSBzb3VyY2VdKGh0dHBzOi8vd3d3LnNpeC1ncm91cC5jb20vZGFtL2Rvd25sb2FkL2ZpbmFuY2lhbC1pbmZvcm1hdGlvbi9kYXRhLWNlbnRlci9pc28tY3VycnJlbmN5L2xpc3RzL2xpc3Qtb25lLnhtbCkKLSBDb3VudHJ5IENvZGVzIFsoSVNPIDMxNjYpXShodHRwczovL3d3dy5pc28ub3JnL2lzby0zMTY2LWNvdW50cnktY29kZXMuaHRtbCkgLSBbd2lraV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvSVNPXzMxNjYtMikKLSBVUkwoaW5jbHVkaW5nIEh0dHBVUkwpIFsoc3RhbmRhcmQpXShodHRwczovL3VybC5zcGVjLndoYXR3Zy5vcmcvKSAtIFt3aWtpXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9VUkwpCi0gRW1haWwgWyhwYXJ0IG9mIFJGQzUzMjIpXShodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjNTMyMikgLSBbd2lraV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvRW1haWxfYWRkcmVzcykKLSBUaW1lem9uZSBbKFJGQzY1NTcgSUFOQSB0aW1lem9uZXMpXShodHRwczovL3d3dy5pYW5hLm9yZy90aW1lLXpvbmVzKSAtIFt3aWtpXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9UaW1lX3pvbmUpCi0gTGFuZ3VhZ2VzIFsoSVNPIDYzOS0xKV0oaHR0cHM6Ly93d3cuaXNvLm9yZy9zdGFuZGFyZC8yMjEwOS5odG1sKSAtIFt3aWtpXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9MaXN0X29mX0lTT182MzktMV9jb2Rlcyk7IFsoSVNPIDYzOS0yKV0oaHR0cHM6Ly93d3cuaXNvLm9yZy9zdGFuZGFyZC80NzY3Lmh0bWwpIC0gW3dpa2ldKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0xpc3Rfb2ZfSVNPXzYzOS0yX2NvZGVzKSwgW2RhdGEgc291cmNlXShodHRwczovL2RhdGFodWIuaW8vY29yZS9sYW5ndWFnZS1jb2Rlcy9yL2xhbmd1YWdlLWNvZGVzLTNiMi5qc29uKQotIERpYWwgQ29kZXMgWyhFLjE2NCldKGh0dHBzOi8vd3d3Lml0dS5pbnQvcmVjL1QtUkVDLUUuMTY0LTIwMTIwMy1JIVN1cDYvZW4pIC0gW3dpa2ldKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0UuMTY0KSwgW2RhdGEgc291cmNlXShodHRwczovL2RhdGFodWIuaW8vY29yZS9jb3VudHJ5LWNvZGVzL3IvY291bnRyeS1jb2Rlcy5qc29uKQotIEJDUDQ3IGxhbmd1YWdlIHRhZ3MgWyh3aWtpKV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvSUVURl9sYW5ndWFnZV90YWcpLCBbcmZjXShodHRwczovL3d3dy5yZmMtZWRpdG9yLm9yZy9pbmZvL2JjcDQ3KQo= readmeEtag: '"ac383a2beb748745027f0ecef63e948c2f69d7e1"' readmeLastModified: Fri, 26 Jul 2024 20:27:42 GMT repositoryId: 360145760 description: >- Library providing opanapi3 and Go types for store/validation and transfer of ISO-4217, ISO-3166, and other types. created: '2021-04-21T11:34:25Z' updated: '2026-01-29T10:01:52Z' language: Go archived: false stars: 23 watchers: 1 forks: 13 owner: mikekonan logo: https://avatars.githubusercontent.com/u/36304777?v=4 license: MIT repoEtag: '"48da50d77071f5a4512ff3a68ee1ae0a6346493b493735c5a8075c7f24807d9d"' repoLastModified: Thu, 29 Jan 2026 10:01:52 GMT foundInMaster: true category: Low-level Tooling id: fe04f57bdbf5a56f74b29a4a912ef4c2 - source: - openapi3 tags - openapi31 tags repository: https://github.com/pb33f/doctor v3: true id: 74c08eebdc1d3c5abe3120f4b374bb0a repositoryMetadata: base64Readme: >- IyBUaGUgRG9jdG9yCgojIyBEciwgV2hvPwoKIyMjIEp1c3QgdGhlIGRvY3Rvci4KClRoZSBkb2N0b3IgcHJvdmlkZXMgc3VwZXItY2hhcmdlZCB2ZXJzaW9ucyBvZiBwYjMzZiBsaWJyYXJpZXMgYW5kIHRvb2xzLiBPZmZlcmluZyBjYXBhYmlsaXRpZXMgbm90IGF2YWlsYWJsZSBpbiB0aGUgT1NTLgoKLS0tCgpUaGlzIGlzIGEgQlVTTC0xLjEgbGljZW5zZWQgcHJvamVjdC4gWW91IGNhbiBmaW5kIHRoZSBsaWNlbnNlIGluIHRoZSBMSUNFTlNFIGZpbGUuCgpJZiB5b3UncmUgdXNpbmcgaXQgZm9yIGludGVybmFsIG9yIG5vbi1jb21tZXJjaWFsIC8gb3BlbiBzb3VyY2UgcHVycG9zZXMsIHlvdSBjYW4gdXNlIGl0IGZvciBmcmVlLiAKCklmIHlvdSdyZSB1c2luZyBpdCBmb3IgY29tbWVyY2lhbCBwdXJwb3NlcywgeW91IG5lZWQgdG8gW2NvbnRhY3QgdXMgYXQgc2FsZXNAcGIzM2YuaW9dKG1haWx0bzpzYWxlc0BwYjMzZi5pbykuCgpUaGlzIGxpYnJhcnkgY29tZXMgd2l0aCBubyBkb2N1bWVudGF0aW9uIGFuZCBubyBzdXBwb3J0IGZvciBmcmVlIHVzZXJzLgoKLS0t readmeEtag: '"236afb7ba0daa364b724772c5033540bf8a11d2c"' readmeLastModified: Wed, 09 Jul 2025 15:47:42 GMT repositoryId: 736423336 description: >- The Doctor. Just the doctor, no more, no less. The OpenAPI all in one suite. created: '2023-12-27T21:41:40Z' updated: '2026-02-04T15:20:41Z' language: Go archived: false stars: 37 watchers: 1 forks: 7 owner: pb33f logo: https://avatars.githubusercontent.com/u/104016643?v=4 license: NOASSERTION repoEtag: '"e418d1f6e8c217b175c7014c78ca2ff147f395c6e2af0286bb33c196308bd9e4"' repoLastModified: Wed, 04 Feb 2026 15:20:41 GMT category: Server Implementations foundInMaster: true v3_1: true - source: openapi3 tags repository: https://github.com/rpstreef/aws-sam-node-example v3: true repositoryMetadata: base64Readme: IyBPcGVuQVBJIE5vZGUgY29kZSBleGFtcGxlCgo= readmeEtag: '"e9ba7d0270059b5e2bb339f36a56857576323f15"' readmeLastModified: Thu, 07 Jan 2021 02:46:27 GMT repositoryId: 261105398 description: AWS SAM NodeJS project example created: '2020-05-04T07:17:35Z' updated: '2024-08-13T03:40:19Z' language: JavaScript archived: false stars: 19 watchers: 1 forks: 13 owner: rpstreef logo: https://avatars.githubusercontent.com/u/15830262?v=4 license: Apache-2.0 repoEtag: '"9b9773ccb637fe8a0b02a037c05a47320d66b92aa3e44b60ada50fc4632e3259"' repoLastModified: Tue, 13 Aug 2024 03:40:19 GMT foundInMaster: true category: Parsers id: ad0e4c9c11802319236236fb57f3c0be - source: openapi3 tags repository: https://github.com/lrstanley/entrest v3: true id: 71bac1f375dfb3f22aec8de61c4aab9f repositoryMetadata: base64Readme: >- <!-- template:define:options
{
  "nodescription": true
}
-->
![logo](https://liam.sh/-/gh/svg/lrstanley/entrest?layout=left&icon=logos%3Aopenapi&icon.height=60&bg=geometric&bgcolor=rgba%2810%2C+10%2C+10%2C+1%29)

<!-- template:begin:header -->
<!-- do not edit anything in this "template" block, its auto-generated -->

<p align="center">
  <a href="https://github.com/lrstanley/entrest/tags">
    <img title="Latest Semver Tag" src="https://img.shields.io/github/v/tag/lrstanley/entrest?style=flat-square">
  </a>
  <a href="https://github.com/lrstanley/entrest/commits/master">
    <img title="Last commit" src="https://img.shields.io/github/last-commit/lrstanley/entrest?style=flat-square">
  </a>




  <a href="https://github.com/lrstanley/entrest/actions?query=workflow%3Atest+event%3Apush">
    <img title="GitHub Workflow Status (test @ master)" src="https://img.shields.io/github/actions/workflow/status/lrstanley/entrest/test.yml?branch=master&label=test&style=flat-square">
  </a>




  <a href="https://codecov.io/gh/lrstanley/entrest">
    <img title="Code Coverage" src="https://img.shields.io/codecov/c/github/lrstanley/entrest/master?style=flat-square">
  </a>

  <a href="https://pkg.go.dev/github.com/lrstanley/entrest">
    <img title="Go Documentation" src="https://pkg.go.dev/badge/github.com/lrstanley/entrest?style=flat-square">
  </a>
  <a href="https://goreportcard.com/report/github.com/lrstanley/entrest">
    <img title="Go Report Card" src="https://goreportcard.com/badge/github.com/lrstanley/entrest?style=flat-square">
  </a>
</p>
<p align="center">
  <a href="https://github.com/lrstanley/entrest/issues?q=is:open+is:issue+label:bug">
    <img title="Bug reports" src="https://img.shields.io/github/issues/lrstanley/entrest/bug?label=issues&style=flat-square">
  </a>
  <a href="https://github.com/lrstanley/entrest/issues?q=is:open+is:issue+label:enhancement">
    <img title="Feature requests" src="https://img.shields.io/github/issues/lrstanley/entrest/enhancement?label=feature%20requests&style=flat-square">
  </a>
  <a href="https://github.com/lrstanley/entrest/pulls">
    <img title="Open Pull Requests" src="https://img.shields.io/github/issues-pr/lrstanley/entrest?label=prs&style=flat-square">
  </a>
  <a href="https://github.com/lrstanley/entrest/discussions/new?category=q-a">
    <img title="Ask a Question" src="https://img.shields.io/badge/support-ask_a_question!-blue?style=flat-square">
  </a>
  <a href="https://liam.sh/chat"><img src="https://img.shields.io/badge/discord-bytecord-blue.svg?style=flat-square" title="Discord Chat"></a>
</p>
<!-- template:end:header -->

<!-- template:begin:toc -->
<!-- do not edit anything in this "template" block, its auto-generated -->
## :link: Table of Contents

  - [Features](#sparkles-features)
  - [Usage](#gear-usage)
  - [Support &amp; Assistance](#raising_hand_man-support--assistance)
  - [Contributing](#handshake-contributing)
  - [License](#balance_scale-license)
<!-- template:end:toc -->

## :sparkles: Features

**entrest** is an [EntGo](https://entgo.io/) extension for generating compliant OpenAPI
specs and an HTTP handler implementation that matches that spec. It expands upon the
approach used by [entoas](https://github.com/ent/contrib/tree/master/entoas#entoas),
with additional functionality, and pairs the generated specification with a
fully-functional HTTP handler implementation.

- :sparkles: Generates OpenAPI specs for your EntGo schema.
- :sparkles: Generates a fully functional HTTP handler implementation that matches the OpenAPI spec.
- :sparkles: Supports automatic pagination (where applicable).
- :sparkles: Supports advanced filtering (using query parameters, `AND`/`OR` predicates, etc).
- :sparkles: Supports eager-loading edges, so you don't have to make additional calls unnecessarily.
- :sparkles: Supports various forms of sorting.
- :sparkles: And more!

---

## :gear: Usage

Take a look at the [official documentation](https://lrstanley.github.io/entrest/) for guides, examples, and more.

<!-- template:begin:goget -->
<!-- do not edit anything in this "template" block, its auto-generated -->
```console
go get -u github.com/lrstanley/entrest@latest
```
<!-- template:end:goget -->

---

<!-- template:begin:support -->
<!-- do not edit anything in this "template" block, its auto-generated -->
## :raising_hand_man: Support & Assistance

* :heart: Please review the [Code of Conduct](.github/CODE_OF_CONDUCT.md) for
     guidelines on ensuring everyone has the best experience interacting with
     the community.
* :raising_hand_man: Take a look at the [support](.github/SUPPORT.md) document on
     guidelines for tips on how to ask the right questions.
* :lady_beetle: For all features/bugs/issues/questions/etc, [head over here](https://github.com/lrstanley/entrest/issues/new/choose).
<!-- template:end:support -->

<!-- template:begin:contributing -->
<!-- do not edit anything in this "template" block, its auto-generated -->
## :handshake: Contributing

* :heart: Please review the [Code of Conduct](.github/CODE_OF_CONDUCT.md) for guidelines
     on ensuring everyone has the best experience interacting with the
    community.
* :clipboard: Please review the [contributing](.github/CONTRIBUTING.md) doc for submitting
     issues/a guide on submitting pull requests and helping out.
* :old_key: For anything security related, please review this repositories [security policy](https://github.com/lrstanley/entrest/security/policy).
<!-- template:end:contributing -->

<!-- template:begin:license -->
<!-- do not edit anything in this "template" block, its auto-generated -->
## :balance_scale: License

```
MIT License

Copyright (c) 2024 Liam Stanley <liam@liam.sh>

Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all
copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
SOFTWARE.
```

_Also located [here](LICENSE)_
<!-- template:end:license -->
 readmeEtag: '"76dde167d103063d415dfecb3815b11a4eca99a0"' readmeLastModified: Sun, 25 Jan 2026 07:10:42 GMT repositoryId: 809436745 description: >- Extension that generates a compliant OpenAPI spec and server implementation created: '2024-06-02T17:24:14Z' updated: '2026-02-01T09:47:11Z' language: Go archived: false stars: 40 watchers: 2 forks: 8 owner: lrstanley logo: https://avatars.githubusercontent.com/u/1847365?v=4 license: MIT repoEtag: '"f5a0abd03a2874ccd04bd0e230fc5d94fe4aff4ba821afc892b922d9b52807a7"' repoLastModified: Sun, 01 Feb 2026 09:47:11 GMT category: Documentation foundInMaster: true - source: openapi3 tags repository: https://github.com/edo1z/rust-axum-sqlx-sample v3: true repositoryMetadata: base64Readme: >- IyBydXN0LWF4dW0tc3FseC1zYW1wbGUKCiMjIEluc3RhbGwKCmBgYHNoZWxsCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vd2ViM3RlbjAvcnVzdC1heHVtLXNxbHgtMS5naXQKY2QgcnVzdC1heHVtLXNxbHgtMS9sb2NhbApkb2NrZXItY29tcG9zZSB1cCAtZApzaCBzY3JpcHRzL2V4ZWNfaW5pdF9kYgpgYGAKCiMjIE1pZ3JhdGlvbgoKLSDlj4Lnhac6IGh0dHBzOi8vZ2l0aHViLmNvbS9sYXVuY2hiYWRnZS9zcWx4L3RyZWUvbWFzdGVyL3NxbHgtY2xpCgpgYGBzaGVsbApjZCBsb2NhbC9zY3JpcHRzCnNoIG1pZ3JhdGUgYWRkIC1yIHVzZXJzCnNoIG1pZ3JhdGUgcnVuCnNoIG1pZ3JhdGUgcmV2ZXJ0CmBgYAoKIyMgU3dhZ2dlcgojIyMgc3dhZ2dlci11aeOBrlVSTAotIGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMQoK readmeEtag: '"d7b06b401cbbc36c5f51365937fa9fe6fa2002e4"' readmeLastModified: Tue, 19 Sep 2023 21:37:10 GMT repositoryId: 450315160 description: Rust Axum+SQLx Sample created: '2022-01-21T01:33:05Z' updated: '2025-05-30T13:45:08Z' language: Rust archived: false stars: 20 watchers: 1 forks: 3 owner: edo1z logo: https://avatars.githubusercontent.com/u/89882017?v=4 license: MIT repoEtag: '"b92114c0502a6c1ddfa937f1fcedc5754e0a82e1bc9b98af8e562d6f65ee639b"' repoLastModified: Fri, 30 May 2025 13:45:08 GMT foundInMaster: true category: Code Generators id: 3bcd05d3559d8a12b7772c87d59d8c84 oldLocations: - https://github.com/net3i/rust-axum-sqlx-sample - source: openapi3 tags repository: https://github.com/igor-baiborodine/campsite-booking v3: true id: d2c07b19920cc8af4f5cbac91985559f repositoryMetadata: base64Readme: >- # Campsite Booking API (Java)  
![Master Branch](https://github.com/igor-baiborodine/campsite-booking/workflows/Build%20Master%20Branch/badge.svg)
[![Sonarcloud Status](https://sonarcloud.io/api/project_badges/measure?project=igor-baiborodine_campsite-booking&metric=alert_status)](https://sonarcloud.io/dashboard?id=igor-baiborodine_campsite-booking) 
[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=igor-baiborodine_campsite-booking&metric=coverage)](https://sonarcloud.io/summary/new_code?id=igor-baiborodine_campsite-booking)

🔥 A Go-based implementation is available in the [campsite-booking-go](https://github.com/igor-baiborodine/campsite-booking-go) repository. 

### A RESTful web service that manages campsite bookings. 

Read these articles to get more insights:
* [Campsite Booking API: Revisited](https://www.kiroule.com/article/campsite-booking-api-revisited/)
* [Campsite Booking API: Revisited 2](https://www.kiroule.com/article/campsite-booking-api-revisited-2/)
* [Campsite Booking API: Revisited 3](https://www.kiroule.com/article/campsite-booking-api-revisited-3/)

<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
**Table of Contents**

- [Technical Task](#technical-task)
  - [Booking Constraints](#booking-constraints)
  - [System Requirements](#system-requirements)
- [Implementation Details](#implementation-details)
- [Up & Running](#up--running)
  - [Run with Maven](#run-with-maven)
  - [Executable JAR](#executable-jar)
  - [Docker](#docker)
- [Tests](#tests)
  - [Maven](#maven)
  - [Swagger UI](#swagger-ui)
  - [Concurrent Tests](#concurrent-tests)
    - [Bookings Creation](#bookings-creation)
    - [Booking Update](#booking-update)
  - [Basic Load Testing](#basic-load-testing)
- [Continuous Integration](#continuous-integration)
  - [Build Master Branch](#build-master-branch)
  - [Build on Pull Request](#build-on-pull-request)
  - [Generate README TOC](#generate-readme-toc)
  - [Perform Release](#perform-release)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

## Technical Task

### Booking Constraints
* The campsite can be reserved for max 3 days.
* The campsite can be reserved minimum 1 day(s) ahead of arrival and up to 1 month in advance.
* Reservations can be cancelled anytime.
* For sake of simplicity assume the check-in & check-out time is 12:00 AM.

### System Requirements
* The users will need to find out when the campsite is available. So the system should expose an API to provide information of the
availability of the campsite for a given date range with the default being 1 month.
* Provide an end point for reserving the campsite. The user will provide his/her email & full name at the time of reserving the campsite
along with intended arrival date and departure date. Return a unique booking identifier back to the caller if the reservation is successful.
* The unique booking identifier can be used to modify or cancel the reservation later on. Provide appropriate end point(s) to allow
modification/cancellation of an existing reservation.
* Due to the popularity of the campsite, there is a high likelihood of multiple users attempting to reserve the campsite for the same/overlapping
date(s). Demonstrate with appropriate test cases that the system can gracefully handle concurrent requests to reserve the campsite.
* Provide appropriate error messages to the caller to indicate the error cases.
* The system should be able to handle large volume of requests for getting the campsite availability.
* There are no restrictions on how reservations are stored as long as system constraints are not violated.

## Implementation Details

This project is implemented using an API-first(or contract-first) approach along with Maven's
multi-module project structure. You can read
more [here](https://swagger.io/resources/articles/adopting-an-api-first-approach/) on the API-first
development approach.

## Up & Running
### Run with Maven
```bash
$ git clone https://github.com/igor-baiborodine/campsite-booking.git
$ cd campsite-booking
$ mvn clean install -DskipTests -DskipITs
$ mvn spring-boot:run -Dspring-boot.run.profiles=in-memory-db -f campsite-booking-service/pom.xml
```
The Swagger UI is available at `http://localhost:8080/swagger-ui.html`.

### Executable JAR
```bash
$ git clone https://github.com/igor-baiborodine/campsite-booking.git
$ cd campsite-booking
$ mvn clean install -DskipTests -DskipITs
$ mvn package spring-boot:repackage -DskipTests -DskipITs -f campsite-booking-service/pom.xml
$ java -jar -Dspring.profiles.active=in-memory-db campsite-booking-service/target/campsite-booking-service-<version>.jar
```
The Swagger UI is available at `http://localhost:8080/swagger-ui.html`.

### Docker
```bash
$ git clone https://github.com/igor-baiborodine/campsite-booking.git
$ cd campsite-booking
$ docker build --rm --file container/Dockerfile --tag campsite-booking-service .
$ docker run -e "SPRING_PROFILES_ACTIVE=in-memory-db" --name campsite-booking-service -d campsite-booking-service
$ docker logs -f campsite-booking-service 
```

The Swagger UI is available at `http://<container-ip>:8080/swagger-ui.html`. To get the container IP
address, execute the following command:

```console
$ docker inspect -f '{{range .NetworkSettings.Networks}}{{.IPAddress}}{{end}}' campsite-booking-service
```

Via the host machine on port 80:
```console
$ docker run -e "SPRING_PROFILES_ACTIVE=in-memory-db" --name campsite-booking-service -p 80:8080 -d campsite-booking-service
```
The Swagger UI is available at `http://localhost:80/swagger-ui.html` or `http://host-ip:80/swagger-ui.html`.

... or with an [image from Docker Hub](https://hub.docker.com/r/ibaiborodine/campsite-booking):
```console
$ docker run -e "SPRING_PROFILES_ACTIVE=in-memory-db" --name campsite-booking-service -p 80:8080 -d ibaiborodine/campsite-booking-service
```
... or with in-memory DB [docker-compose](../container/docker-compose.yml):
```console
$ docker compose -f container/docker-compose.yml up -d
```
... or with MySQL [docker-compose](../container/campsite-booking-service-mysql/docker-compose.yml):
```console
$ docker compose -f container/campsite-booking-service-mysql/docker-compose.yml up -d
```
 
## Tests

### Maven
* Run only unit tests:
```bash
$ mvn clean test
```
* Run only integration tests:
```bash
$ mvn clean failsafe:integration-test
```
* Run unit and integration tests:
```bash
$ mvn clean verify
```
* Run SonarCloud analysis, including test coverage, code smells, vulnerabilities, etc.:
```bash
$ mvn clean verify sonar:sonar -Dsonar.login=<SONAR_TOKEN>
```

### Swagger UI
The API can be tested via the Swagger UI:

![Swagger UI Main View](/readme/swagger-main-view.png)

For example, to add a new booking, expand the `POST` operation. Then click on the `Try it out`, add
the payload below to the `Request Body` text area, and click on the `Execute`:

```json
{
  "campsiteId": 1,
  "email": "John Smith",
  "fullName": "john.smith@email.com",
  "startDate": "2023-11-19",
  "endDate": "2023-11-21",
  "active": true
}
```
![Swagger UI Add Booking 1](/readme/swagger-add-booking-1.png)

If the operation is successful, you will get the following response:

![Swagger UI Add Booking 1](/readme/swagger-add-booking-2.png)

### Concurrent Tests

Start an instance of the Campsite Booking API via Docker Compose either
in [the in-memory-db](../container/docker-compose.yml) or
in [mysql](../container/campsite-booking-service-mysql/docker-compose.yml) profile.

```bash
$ docker-compose.yml up -d
```

#### Bookings Creation

Execute the [concurrent-create-bookings.sh](../script/test/concurrent-create-bookings.sh) script to
simulate concurrent booking creation for the same booking dates:

```bash
$ ./script/test/concurrent-create-bookings.sh 2023-11-16 2023-11-17 http:/localhost:80
```
The response should be as follows after formatting, i.e., only one booking was created:
```json
[
  {
    "uuid": "4da1818c-2d9e-4efe-b59d-38915d6bc5d3",
    "version": 0,
    "campsiteId": 1,
    "email": "john.smith.2@email.com",
    "fullName": "John Smith 2",
    "startDate": "2023-11-16",
    "endDate": "2023-11-17",
    "active": true
  },
  {
    "status": 400,
    "message": "No vacant dates available from 2023-11-16 to 2023-11-17",
    "timestamp": "2023-11-12T20:31:23.751+00:00",
    "subErrors": []
  },
  {
    "status": 400,
    "message": "No vacant dates available from 2023-11-16 to 2023-11-17",
    "timestamp": "2023-11-12T20:31:23.756+00:00",
    "subErrors": []
  }
]
```

#### Booking Update

Execute the [concurrent-update-booking.sh](../script/test/concurrent-update-booking.sh) script to
simulate concurrent updates for the same booking:

```bash
$ ./script/test/concurrent-update-booking.sh 2023-11-15 2023-11-16 http:/localhost:80
```
The response should be as follows after formatting, i.e., only one booking was updated:
```json
[
  {
    "uuid": "ea10008b-c60e-41f9-97bb-313e9502e7f4",
    "version": 1,
    "campsiteId": 3,
    "email": "john.smith.1@email.com",
    "fullName": "John Smith 1",
    "startDate": "2023-11-15",
    "endDate": "2023-11-16",
    "active": true
  },
  {
    "status": 409,
    "message": "Optimistic locking error: com.kiroule.campsitebooking.repository.entity.BookingEntity with id 1 was updated by another transaction",
    "timestamp": "2023-11-12T20:29:55.008+00:00",
    "subErrors": []
  }
]
```

### Basic Load Testing 
Basic load testing for retrieving vacant dates can be performed with the ApacheBench by executing the following command:
```Bash
$ docker-compose.yml up -d
$ ab -n 10000 -c 100 -k http://localhost:80/api/v2/booking/vacant-dates
```
* **-n 10000** is the number of requests to make
* **-c 100** is the number of concurrent requests to make at a time
* **-k** sends the **KeepAlive** header, which asks the web server to not shut down the connection after each request is done, but to instead keep reusing it

Result:
```
Benchmarking localhost (be patient)
Completed 1000 requests
Completed 2000 requests
Completed 3000 requests
Completed 4000 requests
Completed 5000 requests
Completed 6000 requests
Completed 7000 requests
Completed 8000 requests
Completed 9000 requests
Completed 10000 requests
Finished 10000 requests

Server Software:        
Server Hostname:        localhost
Server Port:            80

Document Path:          /api/v2/booking/vacant-dates
Document Length:        159 bytes

Concurrency Level:      100
Time taken for tests:   2.134 seconds
Complete requests:      10000
Failed requests:        0
Non-2xx responses:      10000
Keep-Alive requests:    0
Total transferred:      2720000 bytes
HTML transferred:       1590000 bytes
Requests per second:    4685.95 [#/sec] (mean)
Time per request:       21.340 [ms] (mean)
Time per request:       0.213 [ms] (mean, across all concurrent requests)
Transfer rate:          1244.70 [Kbytes/sec] received

Connection Times (ms)
              min  mean[+/-sd] median   max
Connect:        0    1   0.8      1       7
Processing:     2   20   9.6     19      88
Waiting:        1   19   9.0     18      87
Total:          2   21   9.3     19      88

Percentage of the requests served within a certain time (ms)
  50%     19
  66%     23
  75%     25
  80%     27
  90%     32
  95%     38
  98%     47
  99%     55
 100%     88 (longest request)
```

## Continuous Integration

Continuous integration is implemented using GitHub Actions, and it includes
the `Build Master Branch`, `Build on Pull Request`, `Generate README TOC`, and `Perform Release`
workflows:

![GitHub Actions](/readme/github-actions.png)

### Build Master Branch

This workflow is executed automatically on any commit to the `master` branch and consists of
the `SonarCloud Scan` and `Snapshot Publishing` jobs:

![Build Master Branch Workflow](/readme/github-actions-build-master-branch.png)

### Build on Pull Request

This workflow is executed automatically on any pull request and consists of
the `Unit & SonarCloud Scan` job:

![Build on Pull Request Workflow](/readme/github-actions-build-on-pull-request.png)

### Generate README TOC

This workflow is executed automatically on any update of the `readme/README.md` file pushed to
the `master` branch and consists of the `Generate TOC` job:

![Generate README TOC Workflow](/readme/github-actions-generate-readme-toc.png)

### Perform Release
This workflow is executed manually and consists of the `Maven Release` and `Docker Image` jobs:

![Perform Release Workflow](/readme/github-actions-perform-release.png)

The `Release Version` parameter value should be provided before executing this  workflow:

![Perform Release Workflow](/readme/github-actions-perform-release-parameter.png)
 readmeEtag: '"31ed882b5511c5fd30c0828e941f8b76391fbb96"' readmeLastModified: Mon, 23 Sep 2024 00:10:46 GMT repositoryId: 154585856 description: An example REST API service built using Java and Spring Boot 3. created: '2018-10-25T00:16:09Z' updated: '2025-07-04T21:13:47Z' language: Java archived: false stars: 18 watchers: 1 forks: 12 owner: igor-baiborodine logo: https://avatars.githubusercontent.com/u/1027701?v=4 repoEtag: '"068fdc9282ecef950afa23d8ad515930bc90cb37c1702116e23776ea67a76ce2"' repoLastModified: Fri, 04 Jul 2025 21:13:47 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mikuhuyo/file-service v3: true id: 22c13f95118be85ee3165ce5f382e8ad repositoryMetadata: base64Readme: >- IyDmlofku7bmnI3liqEKClshW0dpdEh1YiBsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL21pa3VodXlvL2ZpbGUtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9maWxlLXNlcnZpY2UvYmxvYi9tYXN0ZXIvTElDRU5TRSkKWyFbR2l0SHViIGlzc3Vlc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvaXNzdWVzL21pa3VodXlvL2ZpbGUtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9maWxlLXNlcnZpY2UvaXNzdWVzKQpbIVtHaXRIdWIgc3RhcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3N0YXJzL21pa3VodXlvL2ZpbGUtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9maWxlLXNlcnZpY2Uvc3RhcmdhemVycykKWyFbR2l0SHViIGZvcmtzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9mb3Jrcy9taWt1aHV5by9maWxlLXNlcnZpY2UpXShodHRwczovL2dpdGh1Yi5jb20vbWlrdWh1eW8vZmlsZS1zZXJ2aWNlL25ldHdvcmspCiFbSmF2YSB2ZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0pkay0xMS15ZWxsb3cpCiFbU3ByaW5nQm9vdCB2ZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1NwcmluZ0Jvb3QtMi4zLjEyLlJFTEVBU0UtYnJpZ2h0Z3JlZW4pCgo+IGh0dHA6Ly8xMjcuMC4wLjE6NTEwMDAvc3dhZ2dlci11aS5odG1sCgojIyDlkK/liqgKCmBgYHlhbWwKdmVyc2lvbjogIjMuNyIKc2VydmljZXM6CiAgbWluaW86CiAgICBpbWFnZTogbWluaW8vbWluaW86UkVMRUFTRS4yMDIyLTA4LTAyVDIzLTU5LTE2WgogICAgcHJpdmlsZWdlZDogdHJ1ZQogICAgY29udGFpbmVyX25hbWU6IG1pbmlvCiAgICBwb3J0czoKICAgICAgLSAiOTAwMDo5MDAwIgogICAgICAtICI5MDAxOjkwMDEiCiAgICB2b2x1bWVzOgogICAgICAtIC9yb290L2RvY2tlci9taW5pby9kYXRhMTovZGF0YTEKICAgICAgLSAvcm9vdC9kb2NrZXIvbWluaW8vZGF0YTI6L2RhdGEyCiAgICBjb21tYW5kOiBzZXJ2ZXIgLS1jb25zb2xlLWFkZHJlc3MgIjo5MDAxIiBodHRwOi8vbWluaW8vZGF0YXsxLi4uMn0KICAgIGVudmlyb25tZW50OgogICAgICAtIE1JTklPX1JPT1RfVVNFUj1yb290CiAgICAgIC0gTUlOSU9fUk9PVF9QQVNTV09SRD15dWVsaW1pbnZjQG91dGxvb2suY29tCiAgICAgICMtIE1JTklPX0FDQ0VTU19LRVk9QUtJQUlPU0ZPRE5ON0VYQU1QTEUKICAgICAgIy0gTUlOSU9fU0VDUkVUX0tFWT13SmFsclhVdG5GRU1JL0s3TURFTkcvYlB4UmZpQ1lFWEFNUExFS0VZCiAgICBoZWFsdGhjaGVjazoKICAgICAgdGVzdDogWyJDTUQiLCAiY3VybCIsICItZiIsICJodHRwOi8vbG9jYWxob3N0OjkwMDAvbWluaW8vaGVhbHRoL2xpdmUiXQogICAgICBpbnRlcnZhbDogMzBzCiAgICAgIHRpbWVvdXQ6IDIwcwogICAgICByZXRyaWVzOiAzCmBgYAoKPiBkb2NrZXItY29tcG9zZSB1cCAtZAoKIyMg54m55Yir6bij6LCiCgojIyMg5YWz5rOo6ICFCgpbIVtTdGFyZ2F6ZXJzIHJlcG8gcm9zdGVyIGZvciBAbWlrdWh1eW8vZmlsZS1zZXJ2aWNlXShodHRwczovL3JlcG9yb3N0ZXIuY29tL3N0YXJzL21pa3VodXlvL2ZpbGUtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9maWxlLXNlcnZpY2Uvc3RhcmdhemVycykKCiMjIyDmlLbol4/ogIUKClshW0ZvcmtlcnMgcmVwbyByb3N0ZXIgZm9yIEBtaWt1aHV5by9maWxlLXNlcnZpY2VdKGh0dHBzOi8vcmVwb3Jvc3Rlci5jb20vZm9ya3MvbWlrdWh1eW8vZmlsZS1zZXJ2aWNlKV0oaHR0cHM6Ly9naXRodWIuY29tL21pa3VodXlvL2ZpbGUtc2VydmljZS9uZXR3b3JrL21lbWJlcnMpCgojIyDor7fov5nkuKpi5Zad5p2v5rC0PwoKIVtBbGlwYXldKC4vaW1hZ2UvYWxpcGF5cy5wbmcpCgotLS0KCiFbV2VDaGF0UGF5XSguL2ltYWdlL3dlY2hhdHMucG5nKQo= readmeEtag: '"c8a92ce1d18dad63b05f91592df245a5eec41253"' readmeLastModified: Mon, 27 Feb 2023 13:11:35 GMT repositoryId: 400815425 description: 单纯的文件服务, 目前仅仅集成了minio(人在电信外包差点没了) created: '2021-08-28T14:40:30Z' updated: '2025-07-28T17:05:26Z' language: Java archived: false stars: 19 watchers: 1 forks: 6 owner: mikuhuyo logo: https://avatars.githubusercontent.com/u/42843191?v=4 repoEtag: '"4032751b7a8c0a2209e20da4d6633cd1d7a63ebcdeeccf2cd58ba97c5f7ffe95"' repoLastModified: Mon, 28 Jul 2025 17:05:26 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/openehr/specifications-its-rest v3: true id: 975080575406c455891722c24cae7045 repositoryMetadata: base64Readme: >- IyBzcGVjaWZpY2F0aW9ucy1JVFMtUkVTVCAtIG9wZW5FSFIgUkVTVCBBUEkgU3BlY2lmaWNhdGlvbnMKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyB0aGUgZG9jdW1lbnRzIGFuZCBzb3VyY2VzIChpbiBodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4wLjNbT3BlbkFQSSAzLjBdIGZvcm1hdCkgb2YgdGhlIG9wZW5FSFIgUkVTVCBBUEkgc3BlY2lmaWNhdGlvbnMuClRoZSBsYXRlc3QgcHVibGlzaGVkIGZvcm0gY2FuIGJlIGFjY2Vzc2VkIGF0IGh0dHBzOi8vc3BlY2lmaWNhdGlvbnMub3BlbmVoci5vcmcvcmVsZWFzZXMvSVRTLVJFU1QvbGF0ZXN0W0lUUy1SRVNUIHNwZWNpZmljYXRpb25zIGluZGV4IHBhZ2VdLgo= readmeEtag: '"4c8cf33553f29e93cdebed4245936615305d036c"' readmeLastModified: Tue, 12 Mar 2024 09:28:48 GMT repositoryId: 40630650 description: openEHR REST API Specifications created: '2015-08-13T00:20:48Z' updated: '2025-12-17T16:28:16Z' language: PHP archived: false stars: 18 watchers: 19 forks: 18 owner: openEHR logo: https://avatars.githubusercontent.com/u/1481762?v=4 license: Apache-2.0 repoEtag: '"6f791c6a80e3e1e29c9b4cb17705b5eda92273c1414d0cceafc35942de5a394c"' repoLastModified: Wed, 17 Dec 2025 16:28:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/yinzhenyu-su/api-typing v3: true id: 79905bfab17d50136d875818f7c1066e repositoryMetadata: base64Readme: >- IyBhcGktdHlwaW5nCgpbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZW4ubmV0L25wbS92L2FwaS10eXBpbmcpXShodHRwczovL25wbS5pbS9hcGktdHlwaW5nKQoKIyMgIVthcGktdHlwaW5nXShodHRwczovL2dpdGh1Yi5jb20veWluemhlbnl1LXN1L2FwaS10eXBpbmcvYmxvYi9tYWluL2FwaS10eXBpbmcuZ2lmP3Jhdz10cnVlKQoKLS0tCgpFbmdsaXNoIHwgW+eugOS9k+S4reaWh10oLi9SRUFETUUuemgtSGFucy5tZCkgfCBbRGVlcFdpa2ldKGh0dHBzOi8vZGVlcHdpa2kuY29tL3lpbnpoZW55dS1zdS9hcGktdHlwaW5nKQoKIyMgQSBzdHJvbmcgdHlwZS1oaW50IGh0dHAgY2xpZW50IGZyYW13b3JrKGJhc2VkIG9uIGF4aW9zKS4KCiAgICBEZWFyIGZyb250LWVuZCBjb2xsZWFndWVzLCBhcmUgeW91IGZlZCB1cCB3aXRoIHRoZSBpbnRlcmZhY2Ugd2l0aCB0aGUgYmFjay1lbmQgZXZlcnkgZGF5LCBhZGp1c3RpbmcgdGhlIGludGVyZmFjZSBwYXRoIGFuZCBwYXJhbWV0ZXIgdHJhbnNmZXIgZm9ybWF0PyBUaXJlZCBvZiB2aWV3aW5nIGJhY2tlbmQgZG9jdW1lbnRhdGlvbj8gQXJlIHlvdSB0aXJlZCBvZiBjaGVja2luZyBhbmQgc3luY2hyb25pemluZyB0aGUgaW50ZXJmYWNlIGZpZWxkIGNoYW5nZXMgYWZ0ZXIgZWFjaCB1cGRhdGU/IEl0J3MgdGltZSB0byBtYWtlIHRoZXNlIGpvYnMgYSBsaXR0bGUgZWFzaWVyLgoKIyMgSW5zdGFsbAoKYGBgYmFzaApwbnBtIGkgYXBpLXR5cGluZwpgYGAKCiMjIFVzYWdlCgoxLiAgc2V0IGEgc2NyaXB0IHRvIHJ1biBgZ2V0LXR5cGVzYCBpbiBgcGFja2FnZS5qc29uYAoKICAgIGBgYGpzb24KICAgIHsKICAgICAgInNjcmlwdHMiOiB7CiAgICAgICAgImdldC10eXBlcyI6ICJnZXQtdHlwZXMgXCJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYWluL2V4YW1wbGVzL3YzLjAvcGV0c3RvcmUtZXhwYW5kZWQuanNvblwiIFwiLi9hcGktdHlwaW5nLW1ldGEuZC50c1wiIgogICAgICB9CiAgICB9CiAgICBgYGAKCiAgICB0aGUgZmlyc3QgYXJndW1lbnQgaXMgYSBvcGVuYXBpIGRlZmluaXRpb24gZmlsZSwganNvbiBmb3JtYXQuCiAgICB0aGUgc2Vjb25kIGFyZ3VtZW50IGlzIGF1dG8tZ2VuZXJhdGVkIHR5cGUgZGVmaW5ldGlvbiBmaWxlIG5hbWUgd2l0aCBkZWZhdWx0IHZhbHVlIGBhcGktdHlwaW5nLW1ldGEuZC50c2AKCjIuICBydW4KCiAgICBgYGBiYXNoCiAgICBwbnBtIHJ1biBnZXQtdHlwZXMKICAgIGBgYAoKICAgIHRoZSB0eXBlIGRlZmluaXRpb24gZmlsZSB3b3VsZCBiZSBnZW5lcmF0ZWQgaW4geW91ciBwcm9qZWN0IHJvb3QuCgozLiAgYWZ0ZXIgYGFwaS10eXBpbmctbWV0YS5kLnRzYCBnZW5ncmF0ZWQgaW4geW91ciBwcm9qZWN0IHJvb3QuIG1ha2Ugc3VyZSB5b3VyIGB0c2NvbmZpZy5qc29uYCBpbmNsdWRlIHRoaXMgZGVjbGFyZS4KCiAgICBgYGBqc29uCiAgICB7CiAgICAgICJpbmNsdWRlIjogWyJhcGktdHlwaW5nLW1ldGEuZC50cyJdCiAgICB9CiAgICBgYGAKCjQuICBpbXBvcnQgYW5kIGVuam95IGl0IQoKICAgIGBgYHRzCiAgICBpbXBvcnQgeyBjcmVhdGVIVFRQQ2xpZW50IH0gZnJvbSAiYXBpLXR5cGluZyIKICAgIGNyZWF0ZUhUVFBDbGllbnQoeyBiYXNlVVJMOiAieW91ciBiYXNlVVJMIiB9KQogICAgICAucG9zdCgiY2hvb3NlL3VybC93aXRoL2hpbnQiLCB7CiAgICAgICAgLyoqCiAgICAgICAgICogcmVxdWVzdCBmaWVsZCB3aXRoIHR5cGUtaGludAogICAgICAgICAqLwogICAgICB9KQogICAgICAudGhlbigoZCkgPT4gewogICAgICAgIC8qKgogICAgICAgICAqIHJlc3BvbnNlIGZpZWxkIHdpdGggdHlwZS1oaW50CiAgICAgICAgICovCiAgICAgIH0pCiAgICBgYGAK readmeEtag: '"fcfb4df58df26db70be3dd9afc2a7d534469a0da"' readmeLastModified: Sun, 27 Apr 2025 05:08:00 GMT repositoryId: 530271572 description: Axios based HTTP client with type hint created: '2022-08-29T15:06:26Z' updated: '2026-01-17T02:39:06Z' language: TypeScript archived: false stars: 18 watchers: 1 forks: 0 owner: yinzhenyu-su logo: https://avatars.githubusercontent.com/u/16461374?v=4 license: MIT repoEtag: '"ad51ad4541f5c189570249c3d2e1bfb623cb9a26aa8d233fda7ffe36f6febec7"' repoLastModified: Sat, 17 Jan 2026 02:39:06 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/felipementel/deploy.cachorro.api v3: true id: 0812df37e319298aadaf68398b59b2d5 repositoryMetadata: base64Readme: >- # DEPLOY.Cachorro.Api

![Imagem projeto api de cachorro](./docs/imgreadme1.png)

[![Quality gate](https://sonarcloud.io/api/project_badges/quality_gate?project=felipementel_DEPLOY.Cachorro.Api)](https://sonarcloud.io/summary/new_code?id=felipementel_DEPLOY.Cachorro.Api)
![actions.](https://github.com/felipementel/DEPLOY.Cachorro.Api/actions/workflows/push-image.yaml/badge.svg?branch=main)
<img alt="gitleaks badge" src="https://img.shields.io/badge/protected%20by-gitleaks-blue">
# Configuração local do Git

```
git config --local --list
```

```
git config --local user.name "Felipe Augusto"
```

```
git config --local user.email felipementel@hotmail.com
```

```

git fetch origin --prune

```

Projeto educacional, criado e mantido através do canal DEPLOY no YouTube.

> Para criar a imagem, a partir do diretório root da aplicação (pasta que contem o arquivo sln)

# pre-requisitos para executar o projeto

1. Docker
2. Um conta no Azure para criar recursos

> Para executar o projeto local:

```
dotnet run --project ./src/DEPLOY.Cachorro.Api/DEPLOY.Cachorro.Api.csproj ASPNETCORE_ENVIRONMENT=aaa ConnectionStrings:ApplicationInsights="bbb" ConnectionStrings:DefaultConnection="ccc" ConnectionStrings:AppConfiguration="ddd"
ApplicationInsights:ApiKey="eee" KeyVault:VaultUri="fff"
```

> Para executar o projeto local, utilizando docker

```
docker container run --rm -p 8088:80 \
-e ASPNETCORE_ENVIRONMENT=aaa \
-e ConnectionStrings__ApplicationInsights="bbb" \
-e ConnectionStrings__DefaultConnection="ccc" \
-e ConnectionStrings__AppConfiguration="ddd" \
-e ApplicationInsights__ApiKey="eee" \
-e KeyVault__VaultUri="fff" \
-e AzureAd__Instance="https://login.microsoftonline.com" \
-e AzureAd__ClientId="ggg" \
-e AzureAd__TenantId="hhh" \
felipementel/cachorro.api:latest
```

```
aaa = Development|Production
bbb = ConnectionString do Application Insights
ccc = Connection String (Azure SQL Database ou https://hub.docker.com/_/microsoft-azure-sql-edge)
ddd = Connection string do App Configuration
eee = ApiKey gerada dentro do Application Insights
fff = URI do Azure KeyVault
ggg = Identificador do cliente criado no Entra ID
hhh = Identificador do Tenant
```

Comando para subir o banco de dados local:

```
docker run --cap-add SYS_PTRACE -e 'ACCEPT_EULA=1' -e 'MSSQL_SA_PASSWORD=Abcd1234%' -e 'MSSQL_PID=Developer' -p 1433:1433 --name azuresqledge -d mcr.microsoft.com/azure-sql-edge:2.0.0
```

Caso queira testar com outra tag, utilize os comandos abaixo para obter a lista de tags

```
https://mcr.microsoft.com/v2/azure-sql-edge/tags/list
```

# Testes de unidade

Tecnologia: XUnit

0. Pre requisito
   Será necessário instalar os dois pacotes abaixo para ter sucesso ao executar os comandos descritos nesse arquivo.

```
dotnet tool install --global dotnet-reportgenerator-globaltool
```

```
dotnet tool install --global dotnet-coverage
```

1. Como Executar:
   1.1 A partir da pasta src execute o comando:

```
dotnet test
```

2. Geração de relatório de testes

   1.1 A partir da pasta src execute o comando:

```
dotnet test --collect:"XPlat Code Coverage" --logger "console;verbosity=detailed" --results-directory ..\TestResults\XPlatCodeCoverage\
```

e depois execute:

```
reportgenerator -reports:..\TestResults\XPlatCodeCoverage\**\coverage.cobertura.xml  -targetdir:..\TestResults\XPlatCodeCoverage\CoverageReport -reporttypes:"Html;SonarQube;JsonSummary;Badges" -classfilters:"-*.Migrations.*" -verbosity:Verbose -title:Cachorro.API -tag:canal-deploy
```

ou

```
$var = (Get-Date).ToString("yyyyMMdd-HHmmss"); dotnet-coverage collect "dotnet test" -f xml -o "..\TestResults\DotnetCoverageCollect\$var\coverage.cobertura.xml"
```

e depois execute:

```
reportgenerator -reports:..\TestResults\DotnetCoverageCollect\**\coverage.cobertura.xml  -targetdir:..\TestResults\DotnetCoverageCollect\CoverageReport -reporttypes:"Html;SonarQube;JsonSummary;Badges" -verbosity:Verbose -title:Cachorro.API -tag:canal-deploy
```

1.2 A partir da pasta src execute o comando:

# Ambiente Local

## SonarQube

1. Comandos para submeter o código ao sonarqube do Container

```
dotnet sonarscanner begin /k:"CachorroAPI" /d:sonar.host.url="http://localhost:9044" /d:sonar.token="xxxxx" /d:sonar.cs.vscoveragexml.reportsPaths=coverage.xml /d:sonar.exclusions="**/Migrations/**"
```

```
dotnet restore .\src\DEPLOY.Cachorro.Api\DEPLOY.Cachorro.Api.csproj
```

```
dotnet build .\src\DEPLOY.Cachorro.Api\DEPLOY.Cachorro.Api.csproj --no-incremental
```

```
dotnet-coverage collect 'dotnet test ./src/' -f xml -o 'coverage.xml'
```

```
dotnet-sonarscanner end /d:sonar.token="xxxxx"
```

---

# EntityFramework Commands

```

dotnet tool install --global dotnet-ef

```

```

dotnet ef migrations add InitDatabaseAPI -s DEPLOY.Cachorro.Api -p DEPLOY.Cachorro.Infra.Repository -c DEPLOY.Cachorro.Infra.Repository.CachorroDbContext --output-dir Migrations/API -v

```

```

dotnet ef database update InitDatabaseAPI --startup-project DEPLOY.Cachorro.Api --project DEPLOY.Cachorro.Infra.Repository --context DEPLOY.Cachorro.Infra.Repository.CachorroDbContext --verbose

```

Connection String

```

Data Source=127.0.0.1,1433;Initial Catalog=Cachorro;User Id=sa;Password=Abcd1234%;Integrated Security=False;MultipleActiveResultSets=True;TrustServerCertificate=true;

```

# Azure Entra ID

```

https://learn.microsoft.com/en-us/entra/identity-platform/v2-overview
```

```
https://learn.microsoft.com/en-us/entra/identity-platform/quickstart-register-app
```

```
https://learn.microsoft.com/en-us/entra/identity-platform/scopes-oidc
```

```

https://learn.microsoft.com/en-us/entra/identity-platform/scenario-protected-web-api-app-configuration?tabs=aspnetcore#bearer-token

```

```

https://learn.microsoft.com/en-us/entra/identity-platform/scenario-web-app-sign-user-app-configuration?tabs=aspnet

```

<br/>
<br/>
<br/>
<br/>
<br/>

## Link de documentações citadas durante a criaçao do projeto

```
https://learn.microsoft.com/en-us/dotnet/core/tools/dotnet
```

```

https://learn.microsoft.com/en-us/aspnet/core/fundamentals/logging/?view=aspnetcore-7.0

```

```
https://learn.microsoft.com/en-us/dotnet/core/compatibility/containers/8.0/app-user

```

```

https://learn.microsoft.com/en-us/aspnet/core/security/app-secrets?view=aspnetcore-7.0&tabs=windows

```

### Github

```
https://docs.github.com/en/actions

```

```
https://github.com/danielpalme/ReportGenerator
```

```
https://github.com/marketplace/actions/reportgenerator
```

```
https://github.com/marketplace/actions/reportgenerator
```

### ILogger

```

https://learn.microsoft.com/en-us/aspnet/core/fundamentals/logging/?view=aspnetcore-7.0

```

```

https://learn.microsoft.com/en-us/dotnet/core/tools/dotnet-test

```

```

https://learn.microsoft.com/en-us/azure/azure-monitor/app/ilogger?tabs=dotnet6

```

### Application Insights

```

https://learn.microsoft.com/pt-br/azure/azure-monitor/app/asp-net-core?tabs=netcorenew%2Cnetcore6

```

Live Stream / Live Metrics

```

https://learn.microsoft.com/en-us/azure/azure-monitor/app/live-stream?tabs=dotnet6

```

### Key Vault

```

https://learn.microsoft.com/pt-br/azure/key-vault/general/basic-concepts

```

```
https://learn.microsoft.com/en-us/dotnet/azure/sdk/dependency-injection?tabs=web-app-builder
```

```
https://learn.microsoft.com/pt-br/aspnet/core/security/key-vault-configuration?view=aspnetcore-3.1
```

### Configuration

```

https://learn.microsoft.com/en-us/aspnet/core/fundamentals/configuration/?view=aspnetcore-7.0

```

### SonarQube

```

https://docs.sonarsource.com/sonarcloud/advanced-setup/analysis-parameters/

```

### Azure App Configuration

```

https://learn.microsoft.com/en-us/azure/azure-app-configuration/quickstart-aspnet-core-app?tabs=core6x
```

```
https://learn.microsoft.com/en-us/azure/azure-app-configuration/howto-feature-filters-aspnet-core
```

```
https://learn.microsoft.com/en-us/azure/azure-app-configuration/enable-dynamic-configuration-aspnet-core?tabs=core6x

```

### Docker Commands

```

docker container rm -f $(docker container ls -a -q)

```

```

docker volume rm $(docker volume ls -q)

```

```

docker image rm -f $(docker image ls -a -q)

```

```

docker system prune

```

```

docker image ls

```

```

docker build -f ./src/DEPLOY.Cachorro.Api/Dockerfile -t felipementel/cachorro.api:local ./src

```

```

docker tag 430f5d9f4229 crcanaldeploydev.azurecr.io/cachorro.api:latest

```

```

docker push crcanaldeploydev.azurecr.io/cachorro.api:latest

```

```

docker push ghcr.io/felipementel/cachorro.api:latest

```

```

docker container run -p 8088:80 crcanaldeploydev.azurecr.io/cachorro.api:latest

```

```

docker container run --rm --name sonar-canal-deploy -p 9044:9000 sonarqube:10.3.0-community

```

### Comandos Azure Key Vault

```

az keyvault secret show --name CachorroConnectionString --vault-name kv-canaldeploy-dev

```

```

@Microsoft.KeyVault(SecretUri=https://kv-canaldeploy-dev.vault.azure.net/secrets/ConnectionStrings--ApplicationInsights/)

```

```

@Microsoft.KeyVault(VaultName=kv-canaldeploy-dev;SecretName=ConnectionStrings--ApplicationInsights)

```

### Comandos Azure Container Instance

```

az container logs --resource-group rg-canaldeploy-dev --name api-cachorro

```

```

az container exec -g g-canaldeploy-dev --name cachorro.api --exec-command "/bin/bash"

```

#git
#github
#github-actions
#sonar-qube
#wsl-2
#docker
#docker-compose
#docker-hub
#donet
#ef-core
#entity-framework-core
#x-unit
#postman
#azure
#azure-application-ingishts
#azure-sql-database
#azure-app-configuration
#azure-container-registry
#azure-container-instance
#azure-container-apps
#azure-key-vault
 readmeEtag: '"b8c9dc24197c5c3f12f5047f2084de6b78fcc2e3"' readmeLastModified: Thu, 17 Oct 2024 12:04:05 GMT repositoryId: 682787791 description: API criada no canal DEPLOY para fins de estudo created: '2023-08-24T23:30:51Z' updated: '2026-01-13T15:30:24Z' language: C# archived: false stars: 24 watchers: 2 forks: 4 owner: felipementel logo: https://avatars.githubusercontent.com/u/14238257?v=4 license: Apache-2.0 repoEtag: '"c76a2c1f502a72d203a4135cfd50f1d3fc6fe431a556054e687f33a80be12f72"' repoLastModified: Tue, 13 Jan 2026 15:30:24 GMT category: Code Generators foundInMaster: true - source: openapi3 tags repository: https://github.com/wblondel/tgtg-api-wrapper v3: true repositoryMetadata: base64Readme: >- # tgtg-api-wrapper
tgtg-api-wrapper is a wrapper around the [TooGoodToGo](https://toogoodtogo.com/) API.

## Work in progress
Currently redacting the API structure in OpenAPI 3 standard.
A Python wrapper will be written next.

## List of API servers

The list of TooGoodToGo's API servers can be found at this address: https://meta.apptoogoodtogo.com/env/v1/list.json

As of 20 February 2024, the servers are:

| Name                     	| Public URL                                      	| "Hidden" URL                                                 	|
|--------------------------	|-------------------------------------------------	|--------------------------------------------------------------	|
| Production (Nearest)     	| https://apptoogoodtogo.com                      	| https://nearest-prod-backend-public.pr-be-shared.tgtg.ninja  	|
| Production (EU)          	| https://euw1.apptoogoodtogo.com                 	| https://prod-backend-public.prod-eks-1.tgtg.ninja            	|
| Production (US)          	| https://use1.apptoogoodtogo.com                 	| https://prod-backend-public.pr-be-use1.tgtg.ninja            	|
| Staging (Nearest)        	| https://backend-staging.apptoogoodtogo.com      	| https://nearest-staging-backend-public.dev-shared.tgtg.ninja 	|
| Staging (EU)             	| https://backend-staging-euw1.apptoogoodtogo.com 	| https://staging-backend-public.dev-euw1.tgtg.ninja           	|
| Staging (US)             	| https://backend-staging-use1.apptoogoodtogo.com 	| https://staging-backend-public.dev-use1.tgtg.ninja           	|
| Gold (EU) (Requires VPN) 	|                                                 	| https://gold-backend-internal.dev-euw1.tgtg.ninja            	|
| Gold (US) (Requires VPN) 	|                                                 	| https://gold-backend-internal.dev-use1.tgtg.ninja            	|
| Partner (EU)              | https://backend-partner-euw1.apptoogoodtogo.com   |                                                               |
| Partner (US)              | https://backend-partner-use1.apptoogoodtogo.com   |                                                               |
| QA (EU)                   | https://backend-qa-euw1.apptoogoodtogo.com        |                                                               |
| QA (US)                   | https://backend-qa-use1.apptoogoodtogo.com        |                                                               |
| Test1 (EU)               	| https://backend-test1.apptoogoodtogo.com        	| https://test1-backend-public.dev-euw1.tgtg.ninja             	|
| Test1 (US)               	| https://backend-test1-use1.apptoogoodtogo.com   	| https://test1-backend-public.dev-use1.tgtg.ninja             	|
| Test2 (EU)               	| https://backend-test2.apptoogoodtogo.com        	| https://test2-backend-public.dev-euw1.tgtg.ninja             	|
| Test2 (US)               	| https://backend-test2-use1.apptoogoodtogo.com   	| https://test2-backend-public.dev-use1.tgtg.ninja             	|
| Test3 (EU)               	| https://backend-test3.apptoogoodtogo.com        	| https://test3-backend-public.dev-euw1.tgtg.ninja             	|
| Test3 (US)               	| https://backend-test3-use1.apptoogoodtogo.com   	| https://test3-backend-public.dev-use1.tgtg.ninja             	|
| Test4 (EU)               	| https://backend-test4.apptoogoodtogo.com        	| https://test4-backend-public.dev-euw1.tgtg.ninja             	|
| Test4 (US)               	| https://backend-test4-use1.apptoogoodtogo.com   	| https://test4-backend-public.dev-use1.tgtg.ninja             	|
| Test5 (EU)               	| https://backend-test5.apptoogoodtogo.com        	| https://test5-backend-public.dev-euw1.tgtg.ninja             	|
| Test5 (US)               	| https://backend-test5-use1.apptoogoodtogo.com   	| https://test5-backend-public.dev-use1.tgtg.ninja             	|
| Test6 (EU)               	| https://backend-test6.apptoogoodtogo.com        	| https://test6-backend-public.dev-euw1.tgtg.ninja             	|
| Test6 (US)               	| https://backend-test6-use1.apptoogoodtogo.com   	| https://test6-backend-public.dev-use1.tgtg.ninja             	|
| Test7 (EU)               	| https://backend-test7.apptoogoodtogo.com        	| https://test7-backend-public.dev-euw1.tgtg.ninja             	|
| Test7 (US)               	| https://backend-test7-use1.apptoogoodtogo.com   	| https://test7-backend-public.dev-use1.tgtg.ninja             	|
| Test8 (EU)               	| https://backend-test8.apptoogoodtogo.com        	| https://test8-backend-public.dev-euw1.tgtg.ninja             	|
| Test8 (US)               	| https://backend-test8-use1.apptoogoodtogo.com   	| https://test8-backend-public.dev-use1.tgtg.ninja             	|
| Test9 (EU)               	| https://backend-test9.apptoogoodtogo.com        	| https://test9-backend-public.dev-euw1.tgtg.ninja             	|
| Test9 (US)               	| https://backend-test9-use1.apptoogoodtogo.com   	| https://test9-backend-public.dev-use1.tgtg.ninja             	|
| Test10 (EU)               | https://backend-test10.apptoogoodtogo.com        	| https://test10-backend-public.dev-euw1.tgtg.ninja             |
| Test10 (US)               | https://backend-test10-use1.apptoogoodtogo.com   	| https://test10-backend-public.dev-use1.tgtg.ninja             |
| Test11 (EU)               | https://backend-test11.apptoogoodtogo.com        	| https://test11-backend-public.dev-euw1.tgtg.ninja             |
| Test11 (US)               | https://backend-test11-use1.apptoogoodtogo.com   	| https://test11-backend-public.dev-use1.tgtg.ninja             |
| Test12 (EU)               | https://backend-test12.apptoogoodtogo.com        	| https://test12-backend-public.dev-euw1.tgtg.ninja             |
| Test12 (US)               | https://backend-test12-use1.apptoogoodtogo.com   	| https://test12-backend-public.dev-use1.tgtg.ninja             |
| Test13 (EU)               | https://backend-test13.apptoogoodtogo.com        	| https://test13-backend-public.dev-euw1.tgtg.ninja             |
| Test13 (US)               | https://backend-test13-use1.apptoogoodtogo.com   	| https://test13-backend-public.dev-use1.tgtg.ninja             |
| Test14 (EU)               | https://backend-test14.apptoogoodtogo.com        	| https://test14-backend-public.dev-euw1.tgtg.ninja             |
| Test14 (US)               | https://backend-test14-use1.apptoogoodtogo.com   	| https://test14-backend-public.dev-use1.tgtg.ninja             |
| Test15 (EU)               | https://backend-test15.apptoogoodtogo.com        	| https://test15-backend-public.dev-euw1.tgtg.ninja             |
| Test15 (US)               | https://backend-test15-use1.apptoogoodtogo.com   	| https://test15-backend-public.dev-use1.tgtg.ninja             |
| Test16 (EU)               | https://backend-test16.apptoogoodtogo.com        	| https://test16-backend-public.dev-euw1.tgtg.ninja             |
| Test16 (US)               | https://backend-test16-use1.apptoogoodtogo.com   	| https://test16-backend-public.dev-use1.tgtg.ninja             |
| Test17 (EU)               | https://backend-test17.apptoogoodtogo.com        	| https://test17-backend-public.dev-euw1.tgtg.ninja             |
| Test17 (US)               | https://backend-test17-use1.apptoogoodtogo.com   	| https://test17-backend-public.dev-use1.tgtg.ninja             |
| Test18 (EU)               | https://backend-test18.apptoogoodtogo.com        	| https://test18-backend-public.dev-euw1.tgtg.ninja            	|
| Test18 (US)              	| https://backend-test18-use1.apptoogoodtogo.com   	| https://test18-backend-public.dev-use1.tgtg.ninja             |
| Test19 (EU)              	| https://backend-test19.apptoogoodtogo.com        	| https://test19-backend-public.dev-euw1.tgtg.ninja             |
| Test19 (US)               | https://backend-test19-use1.apptoogoodtogo.com   	| https://test19-backend-public.dev-use1.tgtg.ninja             |
| Test20 (EU)               | https://backend-test20.apptoogoodtogo.com        	| https://test20-backend-public.dev-euw1.tgtg.ninja             |
| Test20 (US)               | https://backend-test20-use1.apptoogoodtogo.com   	| https://test20-backend-public.dev-use1.tgtg.ninja             |
| Test21 (EU)               | https://backend-test21.apptoogoodtogo.com        	| https://test21-backend-public.dev-euw1.tgtg.ninja             |
| Test21 (US)               | https://backend-test21-use1.apptoogoodtogo.com   	| https://test21-backend-public.dev-use1.tgtg.ninja             |
| Test22 (EU)               | https://backend-test22.apptoogoodtogo.com        	| https://test22-backend-public.dev-euw1.tgtg.ninja             |
| Test22 (US)               | https://backend-test22-use1.apptoogoodtogo.com   	| https://test22-backend-public.dev-use1.tgtg.ninja             |
| Test23 (EU)               | https://backend-test23.apptoogoodtogo.com        	| https://test23-backend-public.dev-euw1.tgtg.ninja             |
| Test23 (US)               | https://backend-test23-use1.apptoogoodtogo.com   	| https://test23-backend-public.dev-use1.tgtg.ninja             |
| Test24 (EU)               | https://backend-test24.apptoogoodtogo.com        	| https://test24-backend-public.dev-euw1.tgtg.ninja             |
| Test24 (US)               | https://backend-test24-use1.apptoogoodtogo.com   	| https://test24-backend-public.dev-use1.tgtg.ninja             |
| Test25 (EU)               | https://backend-test25.apptoogoodtogo.com        	| https://test25-backend-public.dev-euw1.tgtg.ninja             |
| Test25 (US)               | https://backend-test25-use1.apptoogoodtogo.com   	| https://test25-backend-public.dev-use1.tgtg.ninja             |
| Test26 (EU)               | https://backend-test26.apptoogoodtogo.com        	| https://test26-backend-public.dev-euw1.tgtg.ninja             |
| Test26 (US)               | https://backend-test26-use1.apptoogoodtogo.com   	| https://test26-backend-public.dev-use1.tgtg.ninja             |
| Test27 (EU)               | https://backend-test27.apptoogoodtogo.com        	| https://test27-backend-public.dev-euw1.tgtg.ninja             |
| Test27 (US)               | https://backend-test27-use1.apptoogoodtogo.com   	| https://test27-backend-public.dev-use1.tgtg.ninja             |
| Test28 (EU)               | https://backend-test28.apptoogoodtogo.com        	| https://test28-backend-public.dev-euw1.tgtg.ninja             |
| Test28 (US)               | https://backend-test28-use1.apptoogoodtogo.com   	| https://test28-backend-public.dev-use1.tgtg.ninja             |
| Test29 (EU)               | https://backend-test29.apptoogoodtogo.com        	| https://test29-backend-public.dev-euw1.tgtg.ninja             |
| Test29 (US)               | https://backend-test29-use1.apptoogoodtogo.com   	| https://test29-backend-public.dev-use1.tgtg.ninja             |
| Test30 (EU)               | https://backend-test30.apptoogoodtogo.com        	| https://test30-backend-public.dev-euw1.tgtg.ninja             |
| Test30 (US)               | https://backend-test30-use1.apptoogoodtogo.com   	| https://test30-backend-public.dev-use1.tgtg.ninja             |
| Pentest (EU)              | https://backend-pentest.apptoogoodtogo.com        |                                                               |
| Pentest (US)              | https://backend-pentest-use1.apptoogoodtogo.com   |                                                               |
| Bounty (EU)               | https://backend-bounty.apptoogoodtogo.com         |                                                               |
| Bounty (US)               | https://backend-bounty-use1.apptoogoodtogo.com    |                                                               |
| Partner (EU)              | https://backend-partner.apptoogoodtogo.com        |                                                               |
| Partner (US)              | https://backend-partner-use1.apptoogoodtogo.com   |                                                               |
| QA (EU)                   | https://backend-qa.apptoogoodtogo.com             |                                                               |
| QA (US)                   | https://backend-qa-use1.apptoogoodtogo.com        |                                                               |



The base path for the API is `/api/`.

The documentation seems to be at `/api/api-docs` but is only accessible through their VPN.

A GET request to `/web` throws a 403 Forbidden error with the message "Invalid CSRF token".

A GET request to `/logout` redirects (302) to `/login?logout`, which throws a 404 Not Found error.

A GET request to `/ping` returns `pong` :D

A GET request to `/api/` on a public URL redirects (302) to `/api/api-docs` on the corresponding "Hidden" server.

The APIs seem to be built with Spring Boot (clue: the `Whitelabel Error Page`)
 readmeEtag: '"ed7b0bc073189ec9dc5b179944913f2d527b0d3f"' readmeLastModified: Tue, 20 Feb 2024 10:45:29 GMT repositoryId: 242966436 description: Attempt to reverse-engineer the TooGoodToGo API created: '2020-02-25T09:57:17Z' updated: '2025-12-21T21:16:46Z' language: Java archived: false stars: 19 watchers: 1 forks: 3 owner: wblondel logo: https://avatars.githubusercontent.com/u/7508531?v=4 repoEtag: '"91b83f5ba93d6e47b560a2b75e85a6350abcfd0bc970eb2b7ae843cce8196888"' repoLastModified: Sun, 21 Dec 2025 21:16:46 GMT foundInMaster: true category: Server Implementations id: 85d23379e17d8db6ffff513aa15140c5 - source: openapi3 tags name: openapi homepage: https://github.com/wzshiming/openapi language: Go source_description: OpenAPI 3 Specification for golang category: - Low-level Tooling - Parsers repository: https://github.com/wzshiming/openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJCgpbIVtHb0RvY10oaHR0cHM6Ly9nb2RvYy5vcmcvZ2l0aHViLmNvbS93enNoaW1pbmcvb3BlbmFwaT9zdGF0dXMuc3ZnKV0oaHR0cHM6Ly9nb2RvYy5vcmcvZ2l0aHViLmNvbS93enNoaW1pbmcvb3BlbmFwaSkKWyFbR2l0SHViIGxpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2Uvd3pzaGltaW5nL29wZW5hcGkuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3d6c2hpbWluZy9vcGVuYXBpL2Jsb2IvbWFzdGVyL0xJQ0VOU0UpCgpUaGUgb2JqZWN0IG1vZGVsIGZvciBbT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGRvY3VtZW50c10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24pCgpVSSBtb3ZlIHRvIFtnaXRodWIuY29tL3d6c2hpbWluZy9vcGVuYXBpdWldKGh0dHBzOi8vZ2l0aHViLmNvbS93enNoaW1pbmcvb3BlbmFwaXVpKQoKIyMgTGljZW5zZQoKUG91Y2ggaXMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgW0xJQ0VOU0VdKGh0dHBzOi8vZ2l0aHViLmNvbS93enNoaW1pbmcvb3BlbmFwaS9ibG9iL21hc3Rlci9MSUNFTlNFKSBmb3IgdGhlIGZ1bGwgbGljZW5zZSB0ZXh0Lgo= readmeEtag: '"ebf44953ba48498ad777498d0330c0eae5ced41c"' readmeLastModified: Fri, 03 Jul 2020 17:16:32 GMT repositoryId: 138125024 description: OpenAPI 3 Specification for golang created: '2018-06-21T05:49:32Z' updated: '2023-09-08T17:41:57Z' language: Go archived: false stars: 17 watchers: 1 forks: 0 owner: wzshiming logo: https://avatars.githubusercontent.com/u/6565744?v=4 license: MIT repoEtag: '"b7291bded4c0746cdfa4322beb734d10720985f257c006a4c04f70160f105fda"' repoLastModified: Fri, 08 Sep 2023 17:41:57 GMT foundInMaster: true id: 4cfd529550011a9abf3d56ff76ff174b - source: openapi3 tags repository: https://github.com/tavis-software/tavis.openapi v3: true repositoryMetadata: base64Readme: >- IyBUYXZpcy5PcGVuQVBJCgoqKk5vdGU6IFRoaXMgbGlicmFyeSBoYXMgYmVlbiBzdXBlcmNlZGVkIGJ5IFtNaWNyb3NvZnQuT3BlbkFQSS5ORVRdKGh0dHBzOi8vZ2l0aHViLmNvbS9NaWNyb3NvZnQvT3BlbkFQSS5ORVQpLiAgSSBkb24ndCBleHBlY3QgYW55IGZ1cnRoZXIgZGV2ZWxvcG1lbnQgb24gdGhpcyBwcm9qZWN0LioqCgpUaGlzIGxpYnJhcnkgaXMgYSBwYXJzZXIgZm9yIHRoZSBbT3BlbkFQSSBTcGVjaWZpY2F0aW9uXShodHRwczovL29wZW5hcGlzLm9yZy8pLiAgVGhlIG1vZGVsIGlzIGJhc2VkIGFyb3VuZCBPcGVuQVBJIDMuMCBzcGVjaWZpY2F0aW9uLgoKIyMgU2ltcGxlIEV4YW1wbGUKCmBgYGNzaGFycAogICAgICAgICAgICB2YXIgcGFyc2luZ0NvbnRleHQgPSBPcGVuQXBpUGFyc2VyLlBhcnNlKEAiCiAgICAgICAgICAgICAgICAgICAgb3BlbmFwaTogMy4wLjAKICAgICAgICAgICAgICAgICAgICBpbmZvOgogICAgICAgICAgICAgICAgICAgICAgICB0aXRsZTogQSBzaW1wbGUgaW5saW5lIGV4YW1wbGUKICAgICAgICAgICAgICAgICAgICAgICAgdmVyc2lvbjogMS4wLjAKICAgICAgICAgICAgICAgICAgICBwYXRoczoKICAgICAgICAgICAgICAgICAgICAgIC9hcGkvaG9tZToKICAgICAgICAgICAgICAgICAgICAgICAgZ2V0OgogICAgICAgICAgICAgICAgICAgICAgICAgIHJlc3BvbnNlczoKICAgICAgICAgICAgICAgICAgICAgICAgICAgIDIwMDoKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb246IEEgaG9tZSBkb2N1bWVudAogICAgICAgICAgICAgICAgICAgICIpOwoKICAgICAgICAgICAgQXNzZXJ0LkVxdWFsKCIzLjAuMCIsIHBhcnNpbmdDb250ZXh0Lk9wZW5BcGlEb2MuVmVyc2lvbik7CiAgICAgICAgICAgIEFzc2VydC5FcXVhbCgwLCBwYXJzaW5nQ29udGV4dC5QYXJzaW5nRXJyb3JzLkNvdW50KCkpOwpgYGAKCiMjIEdvYWxzCgotIEltcG9ydCBPcGVuQVBJIFYzIGRlZmluaXRpb25zIGluIGJvdGggWUFNTCBhbmQgSlNPTiBmb3JtYXRzLgotIEV4cG9ydCBPcGVuQVBJIGRlZmluaXRpb24gaW4gWUFNTCBmb3JtYXQKLSBJbXBvcnQgT3BlbkFQSSBWMiBkZWZpbml0aW9ucwotIFByb3ZpZGUgY29tcHJlaGVuc2l2ZSBzeW50YXggYW5kIHNlbWFudGljIGVycm9yIHJlcG9ydGluZyAKLSBFbmFibGUgY29uc3RydWN0aW5nIG9mIE9wZW5BUEkgZGVzY3JpcHRpb25zIHZpYSBhIGRvY3VtZW50IG9iamVjdCBtb2RlbAo= readmeEtag: '"ec945c4a56aa3941d5db5e51a3f812ef03184da0"' readmeLastModified: Fri, 19 Jan 2018 19:15:32 GMT repositoryId: 69284017 description: Parser for OpenAPI Specification created: '2016-09-26T19:13:14Z' updated: '2023-09-27T14:30:38Z' language: JavaScript archived: true stars: 17 watchers: 2 forks: 3 owner: tavis-software logo: https://avatars.githubusercontent.com/u/1613247?v=4 license: Apache-2.0 repoEtag: '"78621f56672cdc7b56901d109ec1d8a540dc63b2b616793853c966751ca8b0cb"' repoLastModified: Wed, 27 Sep 2023 14:30:38 GMT foundInMaster: true category: Parsers id: a804baa6bc67699b99d15abe7827381b - source: - openapi3 tags - openapi31 tags repository: https://github.com/isa-group/idl v3: true v3_1: true id: d1ffb665bb9f1281ca8a6c9e385b648e repositoryMetadata: base64Readme: >- # IDL: Inter-parameter Dependency Language

Inter-Parameter Dependency Language (IDL) is a textual domain-specific language for the specification of dependencies among input parameters in web APIs.
IDL is designed to express the seven types of inter-parameter dependencies identified on real-world APIs, including (**Requires**, **Or**, **OnlyOne**, **AllOrNone**, **ZeroOrOne**, **Arithmetic/Relational**, **Complex**). For more details about IDL projects and to explore its applications, please visit the **[IDL website](https://isa-group.github.io/IDL/)**.

## Index:

1. [Catalogue of inter-parameter dependencies](#catalogue-of-inter-parameter-dependencies)
2. [How to use IDL?](#how-to-use-idl)
3. [IDL grammar](#idl-grammar)
4. [IDL Editor](#idl-editor)


## Catalogue of inter-parameter dependencies:

* **Requires:** This type of dependency is expressed as **“IF predicate THEN predicate;”**, where the first predicate is the condition and the second is the consequence. 
  * Example: **[PayPal API - Create invoice](https://developer.paypal.com/docs/api/invoicing/v1/#invoices_create)**

        IF custom.label THEN custom.amount;
        IF minimum_amount_due THEN allow_partial_payment==true;

* **Or:** This type of dependency is expressed using the keyword **“Or”** followed by a list of two or more predicates placed inside parentheses: **“Or(predicate, predicate [, ...]);”**. The dependency is satisfied if at least one of the predicates evaluates to true.
  * Example: **[Reddit API - Upload subreddit image](https://www.reddit.com/dev/api#POST_api_upload_sr_img)**

        IF upload_type=='img' THEN name;
        Or(header, upload_type);

* **OnlyOne:** These dependencies are specified using the keyword **“OnlyOne”** followed by a list of two or more predicates placed inside parentheses: **“OnlyOne(predicate, predicate [, ...]);”**. The dependency is satisfied if one, and only one of the predicates evaluates to true.
  * Example: **[Stripe API - Create coupon](https://stripe.com/docs/api/coupons/create)**
  
        OnlyOne(amount_off, percent_off);
        IF amount_off THEN currency;
        AllOrNone(duration=='repeating', duration_in_months);

* **AllOrNone:** This type of dependency is specified using the keyword **“AllOrNone”** followed by a list of two or more predicates placed inside parentheses: **“AllOrNone(predicate, predicate [, ...]);”**. The dependency is satisfied if either all the predicates evaluate to true, or all of them evaluate to false.
  * Example: **[Napster API - Get favorites](https://developer.prod.napster.com/#member-favorites)**

        AllOrNone(rights, filter=='track'|'album');

* **ZeroOrOne:** These dependencies are specified using the keyword **“ZeroOrOne”** followed by a list of two or more predicates placed inside parentheses: **“ZeroOrOne(predicate, predicate [, ...]);”**. The dependency is satisfied if none or at most one of the predicates evaluates to true.
  * Example: **[Github API - Get User Repos](https://docs.github.com/en/rest/repos/repos#list-repositories-for-the-authenticated-user)**

        ZeroOrOne(type, visibility);
        ZeroOrOne(type, affiliation);

* **Arithmetic/Relational:** Relational dependencies are specified as pairs of parameters joined by any of the following relational operators: ==, !=, <=, <, >= or >. Arithmetic dependencies relate two or more parameters using the operators +, - , *, / followed by a final comparison using a relational operator.
  * Example: **[Shopify API - Get price rules](https://shopify.dev/api/admin-rest/2022-10/resources/pricerule#index-2020-01)**

        created_at_min <= created_at_max;
        updated_at_min <= updated_at_max;
        starts_at_min <= starts_at_max;
        ends_at_min <= ends_at_max;
  
* **Complex:** These dependencies are specified as a combination of the previous ones. As an exception to this rule, predicates cannot include Requires dependencies to avoid over-complicated specifications (such dependencies can be expressed in simpler ways).

### More IDL examples
If you want to see more examples of IDL specifications from real APIs, check [here](es.us.isa.interparamdep/resources/expressiveness_evaluation).

## How to use IDL?

IDL dependencies can be straightforwardly specified in OAS specifications using **IDL4OAS**, an OAS extension. 
You just need to include the keyword `x-dependencies` at the operation level, and include the list of dependencies as an array. See this example from the search operation from
the Google Maps Places API extended with IDL4OAS.

             paths:
               /search:
                 get:
                 parameters:
                   - name: radius [...]
                   - name: rankby [...]
                   - name: keyword [...]
                   - name: name [...]
                   - name: type [...]
                   - name: minprice [...]
                   - name: maxprice [...]
                   - [...]
                   [...]
                 x-dependencies:
                   - ZeroOrOne(radius, rankby=='distance');
                   - IF rankby=='distance' THEN keyword OR name OR type;
                   - maxprice >= minprice;

## IDL grammar
The following section presents a simplified version of the IDL grammar. The full version is available as a part of the implementation of IDL can be found [here](https://github.com/isa-group/IDL/blob/master/es.us.isa.interparamdep/src/es/us/isa/interparamdep/InterparameterDependenciesLanguage.xtext).

            Model:
                Dependency*;
            Dependency:
                RelationalDependency | ArithmeticDependency |
                ConditionalDependency | PredefinedDependency;
            RelationalDependency:
                Param RelationalOperator Param;
            ArithmeticDependency:
                Operation RelationalOperator DOUBLE;
            Operation:
                Param OperationContinuation |
                '(' Operation ')' OperationContinuation?;
            OperationContinuation:
                ArithmeticOperator (Param | Operation);
            ConditionalDependency:
                'IF' Predicate 'THEN' Predicate;
            Predicate:
                Clause ClauseContinuation?;
            Clause:
                (Term | RelationalDependency | ArithmeticDependency
                | PredefinedDependency) | 'NOT'? '(' Predicate ')';
            Term:
                'NOT'? (Param | ParamValueRelation);
            Param:
                ID | '[' ID ']';
            ParamValueRelation:
                Param '==' STRING('|'STRING)* |
                Param 'LIKE' PATTERN_STRING | Param '==' BOOLEAN |
                Param RelationalOperator DOUBLE;
            ClauseContinuation:
                ('AND' | 'OR') Predicate;
            PredefinedDependency:
                'NOT'? ('Or' | 'OnlyOne' | 'AllOrNone' |
                'ZeroOrOne') '(' Clause (',' Clause)+ ')';
            RelationalOperator:
                '<' | '>' | '<=' | '>=' | '==' | '!=';
            ArithmeticOperator:
                '+' | '-' | '*' | '/';

## IDL Editor:

The IDL is implemented using Xtext, a popular framework for the development of programming languages and DSLs. Xtext takes a grammar as input and generates a complete set of tools as output, including a linker, a compiler, a parser and a fully-fledged editor supporting features such as code completion, type checking, syntax coloring and validation as can be observed in the figure below. For instructions on how to generate the Xtext artifacts and test the DSL, refer to [this tutorial](https://www.eclipse.org/Xtext/documentation/102_domainmodelwalkthrough.html).

![IDL Editor](https://raw.githubusercontent.com/isa-group/IDL/IDLWebsite/assets/images/IDLEditor.png)

### Things to notice
- The project containing the grammar definition and all language-specific components (parser, lexer, linker, validation, etc.) is ```'es.us.isa.interparamdep'```.
- For instructions on how to generate the Xtext artifacts and test the DSL, refer to [this tutorial](https://www.eclipse.org/Xtext/documentation/102_domainmodelwalkthrough.html).


 readmeEtag: '"e8bc0f87fa5ef755da5206762212626fd82ff119"' readmeLastModified: Sun, 17 Sep 2023 16:47:13 GMT repositoryId: 211313406 description: 'IDL: Inter-parameter Dependency Language' created: '2019-09-27T12:22:42Z' updated: '2026-01-06T11:19:48Z' language: Java archived: false stars: 20 watchers: 9 forks: 0 owner: isa-group logo: https://avatars.githubusercontent.com/u/2708867?v=4 license: GPL-3.0 repoEtag: '"26ccab84a3601be697aabafb42439b65bbb81fc3aa2ebdd962a44d0e8d0992f3"' repoLastModified: Tue, 06 Jan 2026 11:19:48 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/elixir-cloud-aai/foca v3: true id: 1b552721e4edb94864dd3f1fad63ef9f repositoryMetadata: base64Readme: >- # ![FOCA_logo][foca-logo] &ensp;_Develop Flask microservices quickly!_

[![License][badge-license]][badge-url-license]
[![Build_status][badge-build-status]][badge-url-build-status]
[![Docs][badge-docs]][badge-url-docs]
[![Coverage][badge-coverage]][badge-url-coverage]
[![GitHub_tag][badge-github-tag]][badge-url-github-tag]
[![PyPI_release][badge-pypi]][badge-url-pypi]

**FOCA** (**F**lask-**O**penAPI-**C**onnexion-**A**rchetype) is an opinionated
archetype that enables fast development of [OpenAPI][res-openapi]-based HTTP
API microservices in [Flask][res-flask], leveraging the excellent
[Connexion][res-connexion] framework.

FOCA reduces the required boilerplate code to fire up your app to a bare
minimum and allows you to _**focus on your application logic**_. It also avoids
unnecessary code repetition and introduces cross-service consistency when
developing multiple applications. Simply write a configuration file, pass
it to FOCA and you're good to go!

Currently supported features:

* Manage app configuration
* Handle exceptions
* Register OpenAPI 2.x/3.x specifications
* Protect endpoints via [JWT][res-jwt] validation
* Register [MongoDB][res-mongo-db] collections
* Run asynchronous tasks via [RabbitMQ][res-rabbitmq] & [Celery][res-celery]
* [CORS][res-cors] support

Check the [API docs][badge-url-docs] for further details.

## Table of Contents

* [Usage](#usage)
* [Configuration](#configuration)
  * [Configuring OpenAPI specifications](#configuring-openapi-specifications)
  * [Configuring MongoDB](#configuring-mongodb)
  * [Configuring exceptions](#configuring-exceptions)
  * [Configuring asynchronous tasks](#configuring-asynchronous-tasks)
  * [Configuring logging](#configuring-logging)
  * [Configuring security](#configuring-security)
  * [Configuring the server](#configuring-the-server)
  * [Custom configuration](#custom-configuration)
  * [Accessing configuration parameters](#accessing-configuration-parameters)
* [Utilities](#utilities)
  * [Database utilities](#database-utilities)
  * [Logging utitlies](#logging-utilities)
  * [Miscellaneous utilities](#miscellaneous-utilities)
  * [Access Control utilities](#access-control-utilities)
* [Contributing](#contributing)
* [Versioning](#versioning)
* [License](#license)
* [Contact](#contact)

## Usage

(1) Install the [FOCA package][badge-url-pypi] with `pip`:

```bash
pip install foca
```

(2) Create a [configuration](#configuration) file.

(3) Import the FOCA class and pass your config file:

```python
from foca import Foca

foca = Foca(config_file="path/to/my/app/config.yaml")
app = foca.create_app()  # returns a Connexion app instance
```

(4) Start your [Flask][res-flask] app as usual.

![Hint][img-hint] _**Check out the [Petstore example application][example]
shipped with this repository to see FOCA in action!**_

## Configuration

![Hint][img-hint] _**In order to get you started writing your own app
configuration, you can copy the [**annotated template**][config-template]
shipped with this repository and modify it.**_

In order to use FOCA functionalities, you must create a [YAML][res-yaml]
configuration file that includes keyword sections reserved by FOCA. The
following top-level sections are interpreted by FOCA (exhaustive; links are
provided to the corresponding sections in this documentation, as well as to the
corresponding models in the [API docuementation][badge-url-docs]):

* [`api`](#configuring-openapi-specifications) / [model][docs-models-api]
* [`db`](#configuring-mongodb) / [model][docs-models-db]
* [`exceptions`](#configuring-exceptions) / [model][docs-models-exceptions]
* [`jobs`](#configuring-asynchronous-tasks) / [model][docs-models-jobs]
* [`log`](#configuring-logging) / [model][docs-models-log]
* [`security`](#configuring-security) / [model][docs-models-security]
* [`server`](#configuring-the-server) / [model][docs-models-server]

**_Any values passed to reserved keywords are automatically validated_**, and a
corresponding informative exception will be raised whenever a value does not
adhere to the corresponding model as described in the [API
documentation][docs-models]. If you do _not_ want to make use of a specific
FOCA functionality, simply omit the corresponding section.

### Configuring OpenAPI specifications

The `api` section is used to specify any [OpenAPI][res-openapi] specifications
consumed as part of your application. Essentially, FOCA adds a wrapper around
[Connexion][res-connexion], which validates requests/responses and can serve
the specifications as well as a [Swagger][res-swagger]-based user interface to
explore the API. FOCA supports multiple specification files (versions
Swagger/OpenAPI 2.x, OpenAPI 3.x and mixed) and multiple fragments thereof, and
it adds additional features that allow easy modification of specifications on
the fly. In particular, links to routers and security definitions can be added
to each specified endpoint.

_Example:_

```yaml
api:
  specs:
    - path:
        - path/to/openapi/specs.yaml
        - path/to/openapi/additions.yaml
      add_operation_fields:
        x-openapi-router-controller: myapi.controllers
      add_security_fields:
        x-bearerInfoFunc: app.validate_token
      disable_auth: False
      connexion:
        strict_validation: True
        validate_responses: True
        options:
          swagger_ui: True
          serve_spec: True
    - path:
        - path/to/openapi/other_specs.yaml
```

> In this example, the configuration file lists two separate specifications.
> The first is a composite one that FOCA will compile from two files,
> `path/to/openapi/specs.yaml` and `path/to/openapi/additions.yaml`. It comes
> with a range of different explicitly specified parameters to further
> customize the specification itself (classes/functions implementing
> controllers and token validation are linked to each endpoint via
> `add_operation_fields`; `x-openapi-router-controller` and `x-bearerInfoFunc`
> can be used to link controller functions/classes and authorization validation
> functions to endpoints, respectively. Furthermore, a flag to disable the need
> for passing authorization tokens and several [Connexion][res-connexion]
> options are explicitly set for this specification. In contrast, only the path
> to a single file is specified for the second specification, implying default
> values for all other options.  
>  
> Further support for validating authorization can also be added to
> specifications via the `add_security_fields` parameter under `specs` (not
> shown here). Cf. the [API model][docs-models-api] for this and other options,
> as well as further details.

### Configuring MongoDB

FOCA can register one or more [MongoDB][res-mongo-db] databases and/or
collections for you. To use that functionality, simply include the top-level
`db` keyword section in your configuration file and tune its behavior through
the available parameters.

_Example:_

```yaml
db:
  host: mongodb
  port: 27017
  dbs:
    myDb:
      collections:
        myCollection:
          indexes:
            - keys:
                id: 1
              options:
                'unique': True
        mySecondCollection:
          indexes:
            - keys:
                other_id: 1
    myOtherDb:
      collections:
        myThirdCollection:
          indexes:
            - keys:
                third_id: 1
```

> In this example, two databases (`myDb` and `myOtherDb`) are configured, the
> former with two and the latter with one collection (`myCollection`,
> `mySecondCollection` and `myThirdCollection`, respectively). FOCA will
> automatically register and initialize these databases and collections for you
> and add convenient clients to the app instance (accessible as children of
> `current_app.config.foca` in an [application
> context][res-flask-app-context]). The collections would be indexed by keys
> `id`, `other_id` and `third_id`, respectively. Out of these, only `id`
> will be required to be unique.  
>  
> Cf. the [API model][docs-models-db] for further options and details.

### Configuring exceptions

FOCA provides a convenient, configurable exception handler and a simple way
of adding new exceptions to be used with that handler. To use it, specify a
top-level `exceptions` section in the app configuration file.

_Example:_

```yaml
exceptions:
  required_members: [['msg'], ['status']]
  status_member: ['status']
  exceptions: my_app.exceptions.exceptions
  logging: one_line
```

> This example configuration would attach the exceptions defined in the
> `my_app.exceptions.exceptions` dictionary to the exception handler. The
> exception handler ensures that every exception in that dictionary defines
> at least members `msg` and `status`. Out of these, `status` will be used
> to inform the status code for the error response. Exceptions processed via
> FOCA's exception handler will be automatically logged, if requested. In this
> case, the handler is configured to log all errors verbosely (including any
> traceback information, if applicable) on a single line (other rendering
> options are also supported).
>  
> You may further configure optional members, a list of `public members` (to be
> included in error responses) and `private members` (only visible in logs).
> Cf. the [API model][docs-models-exceptions] for further options and details.

### Configuring asynchronous tasks

FOCA offers limited support for running asynchronous tasks via the
[RabbitMQ][res-rabbitmq] broker and [Celery][res-celery]. To make use of it,
include the `jobs` top-level section in the app configuration file.

_Example:_

```yaml
jobs:
  host: rabbitmq
  port: 5672
  backend: 'rpc://'
  include:
    - my_app.tasks.my_task_1
    - my_app.tasks.my_task_2
```

> This config attaches the `rabbitmq` broker host running on port `5672` to
> FOCA and registers the tasks found in modules `my_task_1` and `my_task_2`.  
>  
> Cf. the [API model][docs-models-jobs] for further details.  

The `foca.Foca` class provides a method `.create_celery_app()` that you can
use in your Celery worker entry point to crate a Celery app, like so:

```py
from foca import Foca

foca = Foca(config="my_app/config.yaml")
my_celery_app = foca.create_celery_app()
```

### Configuring logging

FOCA allows you to specify a YAML-based logging configuration to control your
application's logging behavior in an effort to provide a single configuration
file for every application. To use it, simply add a `log` top-level section in
your app configuration file.

_Example:_

```yaml
log:
  version: 1
  disable_existing_loggers: False
  formatters:
    standard:
      class: logging.Formatter
      style: "{"
      format: "[{asctime}: {levelname:<8}] {message} [{name}]"
  handlers:
    console:
      class: logging.StreamHandler
      level: 20
      formatter: standard
      stream: ext://sys.stderr
  root:
    level: 10
    handlers: [console]
```

> The logging configuration is simply passed on to Python's `logging' package,
> and so it has to conform with that [package's
> requirements][res-python-logging]. See [here][res-python-logging-how-to] for
> more info.

### Configuring security

FOCA offers some convenience functionalities for securing your app.
Specifically, it allows you to configure the validation of [JSON Web
Token (JWT)][res-jwt]-based authorization, a [Casbin][res-casbin]-based access
control model, and the use of [cross-origin resource sharing (CORS)][res-cors].
To make use of them, include the `security` top-level section in your app
configuration, as well as the desired sublevel section(s):

```yaml
security:
  auth:
    algorithms:
      - RS256
    allow_expired: False
    validation_methods:
      - userinfo
      - public_key
    validation_checks: any
  access_control:
    api_specs: 'path/to/your/access/control/specs'
    api_controllers: 'path/to/your/access/control/spec/controllers'
    api_route: '/route/to/access_control_api'
    db_name: access_control_db_name
    collection_name: access_control_collection_name
    model: access_control_model_definition
  cors:
    enabled: True
```

> In this example, the validation of JWT Bearer tokens would make use of the
> `RS256` algorithm, would not allow expired tokens and would grant access to
> a protected endpoint if `any` of the two listed validation methods (via the
> identity provider's `/userinfo` endpoint or its JSON Web Key (JWK) public
> key. Furthermore, the application created with this config would provide
> an access control model `model`. Corresponding permissions could be accessed
> and altered by a user with admin permissions via the dedicated endpoints
> defined in the `api_specs`, operationalized by the controllers in
> `api_controllers` and hosted at `api_route`. Permissions will be stored in
> collection `collection_name` of a dedicated MongoDB database `db_name`.
> Finally, CORS would be enabled for this application.
>  
> Cf. the [API model][docs-models-security] for further options and details.

**Note:** A detailed explaination of the access control implementation can be
found [here][docs-access-control].

### Configuring the server

FOCA allows you to pass certain basic configuration options to your Flask
application. To modify defaults, include the top-level `server` keyword section
in your app configuration file:

```yaml
server:
  host: '0.0.0.0'
  port: 8080
  debug: True
  environment: development
  use_reloader: False
```

> This config would create an application server hosting a Flask `development`
> environment at `0.0.0.0:8080`, Flask's debugger switched on, and its reloader
> off.
>  
> Cf. the [API model][docs-models-server] for further options and details.

### Custom configuration

If you would like FOCA to validate your custom app configuration (e.g.,
parameters required for individual controllers, you can provide a path, in
dot notation, to a [`pydantic`][res-pydantic] `BaseModel`-derived model. FOCA
then tries to instantiate the model class with any custom parameters listed
under keyword section `custom`.

Suppose you have a model like the following defined in module
`my_app.custom_config`:

```py
from pydantic import BaseModel


class CustomConfig(BaseModel):
    my_param: int = 5
```

And you have, in your app configuration file `my_app/config.yaml`, the
following section:

```console
custom:
  my_param: 10
```

You can then have FOCA validate your custom configuration against the
`CustomConfig` class by including it in the `Foca()` call like so:

```py
from foca import Foca

foca = Foca(
  config="my_app/config.yaml",
  custom_config_model="my_app.custom_config.CustomConfig",
)
my_app = foca.create_app()
```

We recommend that, when defining your `pydantic` model, that you supply
default values wherever possible. In this way, the custom configuration
parameters will always be available, even if not explicitly listed in the app
configuration (like with the FOCA-specific parameters).

> Note that there is tooling available to automatically generate `pydantic`
> models from different file formats like JSON Schema etc. See here for the
> [datamodel-code-generator][res-datamodel-code-generator] project.

Apart from the reserved keyword sections listed above, you are free to include
any other sections and parameters in your app configuration file. FOCA will
simply attach these to your application instance as described
[above](#configuration) and shown [below](#accessing-configuration-parameters).
Note, however, that any such parameters need to be _manually_ validated. The
same is true if you include a `custom` section but do _not_ provide a
validation model class via the `custom_config_model` parameter when
instantiating `Foca`.

_Example:_

```yaml
my_custom_param: 'some_value'

my_custom_param_section:
  another_custom_param: 3
  my_custom_list_param:
    - 1
    - 2
    - 3
```

### Accessing configuration parameters

Once the application is created using `foca()`, one can easily access any
configuration parameters from within the [application
context][res-flask-app-context] through `current_app.config.foca like so:

```python
from flask import current_app

app_config = current_app.config.foca

db = app_config.db
api = app_config.api
server = app_config.server
exceptions = app_config.exceptions
security = app_config.security
jobs = app_config.jobs
log = app_config.log
app_specific_param = current_app.config['app_specific_param']
```

_Outside of the application context_, configuration parameters are available
via `app.config.foca` in a similar way.

### More examples

Apart from the [annotated template][config-template], you can also check
out the [configuration file][config-petstore] of the [Petstore app][example]
for another example.

![Hint][img-hint] _**Or why not explore [apps that already use
FOCA][res-using-foca]?**_

## Utilities

FOCA provides some functions that may be useful for several applications.
Simply import them if you want to use them.

### Database utilities

FOCA provides the following general-purpose MongoDB controllers:

* Fetch latest object given the db `collection`:

```python
from foca.utils.db import find_one_latest

latest_object = find_one_latest("your_db_collection_instance")
```

* Fetch latest object identifier (`id`) given the db `collection`:

```python
from foca.utils.db import find_id_latest

latest_object_id = find_id_latest("your_db_collection_instance")
```

### Logging utilities

FOCA provides a decorator that can be used on any route to automatically log
any requests and/or responses passing through that route:

```python
from foca.utils.logging import log_traffic

@log_traffic(log_request=True, log_response=True, log_level=20)
def your_controller():
    pass
```

> The above decorater will log both requests and responses with the specified
> logging level (`20`, or `INFO`).

### Miscellaneous utilities

* Generate a random object from a given character set:

```python
import string

from foca.utils.misc import generate_id

obj_id = generate_id(charset=string.digits, length=6)
```

> The above function processes and returns a random `obj_id` of length `6`
> consisting of only digits (`string.digits`).

### Access Control utilities

FOCA provides a decorator that can be used on any route to automatically
validate request on the basis of permission rules.

```python
from foca.security.access_control.register_access_control import (
    check_permissions
)

@check_permissions
def your_controller():
    pass
```

## Contributing

This project is a community effort and lives off your contributions, be it in
the form of bug reports, feature requests, discussions, or fixes and other code
changes. Please refer to our organization's [contributing
guidelines][res-elixir-cloud-contributing] if you are interested to contribute.
Please mind the [code of conduct][res-elixir-cloud-coc] for all interactions
with the community.

## Versioning

The project adopts [semantic versioning][res-semver]. Currently the service
is in beta stage, so the API may change without further notice.

## License

This project is covered by the [Apache License 2.0][license-apache] also
[shipped with this repository][license].

## Contact

The project is a collaborative effort under the umbrella of [ELIXIR Cloud &
AAI][org-elixir-cloud]. Follow the link to get in touch with us via chat or
email. Please mention the name of this service for any inquiry, proposal,
question etc.

![Logo_banner][img-logo-banner]

[badge-build-status]: <https://github.com/elixir-cloud-aai/foca/actions/workflows/checks.yml/badge.svg>
[badge-coverage]: <https://codecov.io/gh/elixir-cloud-aai/foca/branch/dev/graph/badge.svg?branch=dev>
[badge-docs]: <https://readthedocs.org/projects/foca/badge/>
[badge-github-tag]: <https://img.shields.io/github/v/tag/elixir-cloud-aai/foca?color=C39BD3>
[badge-license]: <https://img.shields.io/badge/license-Apache%202.0-blue.svg>
[badge-pypi]: <https://img.shields.io/pypi/v/foca.svg?style=flat&color=C39BD3>
[badge-url-build-status]: <https://github.com/elixir-cloud-aai/foca/actions/workflows/checks.yml>
[badge-url-coverage]: <https://codecov.io/gh/elixir-cloud-aai/foca?branch=dev>
[badge-url-docs]: <https://foca.readthedocs.io/en/latest/>
[badge-url-github-tag]: <https://github.com/elixir-cloud-aai/foca/releases>
[badge-url-license]: <http://www.apache.org/licenses/LICENSE-2.0>
[badge-url-pypi]: <https://pypi.python.org/pypi/foca>
[config-template]: templates/config.yaml
[config-petstore]: examples/petstore/config.yaml
[docs-access-control]: docs/access_control/README.md
[docs-models]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html>
[docs-models-api]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.APIConfig>
[docs-models-db]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.DBConfig>
[docs-models-exceptions]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.ExceptionConfig>
[docs-models-jobs]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.JobsConfig>
[docs-models-log]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.LogConfig>
[docs-models-security]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.SecurityConfig>
[docs-models-server]: <https://foca.readthedocs.io/en/latest/modules/foca.models.html#foca.models.config.ServerConfig>
[example]: examples/petstore/README.md
[foca-logo]: images/foca_logo_192px.png
[img-hint]: images/hint.svg
[img-logo-banner]: images/logo-banner.svg
[license]: LICENSE
[license-apache]: <https://www.apache.org/licenses/LICENSE-2.0>
[org-elixir-cloud]: <https://github.com/elixir-cloud-aai/elixir-cloud-aai>
[res-casbin]: <https://casbin.org/>
[res-celery]: <http://docs.celeryproject.org/>
[res-connexion]: <https://github.com/zalando/connexion>
[res-cors]: <https://flask-cors.readthedocs.io/en/latest/>
[res-elixir-cloud-coc]: <https://github.com/elixir-cloud-aai/elixir-cloud-aai/blob/dev/CODE_OF_CONDUCT.md>
[res-elixir-cloud-contributing]: <https://github.com/elixir-cloud-aai/elixir-cloud-aai/blob/dev/CONTRIBUTING.md>
[res-flask]: <http://flask.pocoo.org/>
[res-flask-app-context]: <https://flask.palletsprojects.com/en/1.1.x/appcontext/>
[res-jwt]: <https://jwt.io>
[res-mongo-db]: <https://www.mongodb.com/>
[res-openapi]: <https://www.openapis.org/>
[res-pydantic]: <https://pydantic-docs.helpmanual.io/>
[res-rabbitmq]: <https://www.rabbitmq.com/>
[res-semver]: <https://semver.org/>
[res-swagger]: <https://swagger.io/tools/swagger-ui/>
[res-using-foca]: <https://github.com/elixir-cloud-aai/foca/network/dependents>
[res-yaml]: <https://yaml.org/>
 readmeEtag: '"49341f02d708e9732db76eb3f2d2ca068f40ca86"' readmeLastModified: Thu, 11 Jul 2024 10:51:55 GMT repositoryId: 238483669 description: >- Opinionated Flask microservice archetype for quick OpenAPI-based microservice development created: '2020-02-05T15:30:51Z' updated: '2026-01-30T05:13:52Z' language: Python archived: false stars: 17 watchers: 26 forks: 13 owner: elixir-cloud-aai logo: https://avatars.githubusercontent.com/u/60704687?v=4 license: Apache-2.0 repoEtag: '"d7bd0983800339e902f97ab2fca717b7eb96ecc1d2fabf49216fa299d95cd5f5"' repoLastModified: Fri, 30 Jan 2026 05:13:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/spaceavocado/apidoc v3: true repositoryMetadata: base64Readme: >- # APIDoc
> Current version: beta-0.3.3

<br>
<p align="center">
    <img alt="apidoc" src="https://github.com/spaceavocado/apidoc/raw/master/assets/apidoc.png" width="240">
</p>

<p align="center">
  Generate RESTful API documentation from GO source files into the <a href="https://swagger.io/specification/">OpenAPI v3.0.2</a> specification (formal Swagger 2.0 Specification).
</p>

<p align="center">
<a href="https://travis-ci.org/spaceavocado/apidoc.svg?branch=master"><img alt="Travis Status" src="https://travis-ci.org/spaceavocado/apidoc.svg?branch=master"></a> <a href="https://codecov.io/gh/spaceavocado/apidoc">
  <img src="https://codecov.io/gh/spaceavocado/apidoc/branch/master/graph/badge.svg" />
</a> <a href="https://goreportcard.com/badge/github.com/spaceavocado/apidoc"><img alt="Go Report Card" src="https://goreportcard.com/badge/github.com/spaceavocado/apidoc"></a> <a href="https://www.codacy.com/app/davidhorak/apidoc?utm_source=github.com&amp;utm_medium=referral&amp;utm_content=spaceavocado/apidoc&amp;utm_campaign=Badge_Grade"><img src="https://api.codacy.com/project/badge/Grade/ed701d5e42f3426d832de973906ba19b"/></a> <a href="https://app.fossa.io/projects/git%2Bgithub.com%2Fspaceavocado%2Fapidoc?ref=badge_shield" alt="FOSSA Status"><img src="https://app.fossa.io/api/projects/git%2Bgithub.com%2Fspaceavocado%2Fapidoc.svg?type=shield"/></a> <a href="https://godoc.org/github.com/spaceavocado/apidoc"><img alt="Go Doc" src="https://godoc.org/github.com/spaceavocado/apidoc?status.svg"></a>
</p>

## Table of Contents
- [APIDoc](#apidoc)
  - [Table of Contents](#table-of-contents)
  - [Summary](#summary)
  - [Examples](#examples)
  - [Getting Started](#getting-started)
- [API Annotation in Comments](#api-annotation-in-comments)
  - [Main Section](#main-section)
    - [Supported Tags](#supported-tags)
    - [Example](#example)
  - [An Endpoint](#an-endpoint)
    - [Supported Tags](#supported-tags-1)
    - [Param Tag](#param-tag)
      - [name](#name)
      - [in](#in)
      - [type](#type)
    - [Body Tag](#body-tag)
      - [reference](#reference)
    - [Wrapper Tag](#wrapper-tag)
      - [Example](#example-1)
    - [Response Tag](#response-tag)
      - [code](#code)
      - [type](#type-1)
      - [reference](#reference-1)
    - [Path Tag](#path-tag)
      - [path](#path)
      - [method](#method)
    - [Example Endpoint Annotation](#example-endpoint-annotation)
  - [gorilla/mux Handler Functions](#gorillamux-handler-functions)
    - [Notes](#notes)
  - [gorilla/mux Subrouter](#gorillamux-subrouter)
    - [Subrouter Annotation](#subrouter-annotation)
      - [Example](#example-2)
  - [Mime Types Annotation](#mime-types-annotation)
  - [Struct Annotation](#struct-annotation)
  - [Data Types Conversion](#data-types-conversion)
- [Tips](#tips)
  - [Annotation over Multiple Lines](#annotation-over-multiple-lines)
  - [Array References](#array-references)
  - [And Endpoint With Many Decralarions](#and-endpoint-with-many-decralarions)
    - [Example](#example-3)
- [APIDoc CLI](#apidoc-cli)
- [About the Project](#about-the-project)
- [Contributing](#contributing)
  - [Pull Request Process](#pull-request-process)
- [License](#license)

## Summary
APIDoc extracts the API documentation annotation from your GO source files, recursively resoles struct references, and it generates the YAML [OpenAPI v3.0.2](https://swagger.io/specification/) spec. file, which could be tested in the [Swagger Editor](https://editor.swagger.io/) and quickly integrated with [Swagger UI](https://swagger.io/tools/swagger-ui/download/).

The generator is also able to read [gorilla/mux](https://github.com/gorilla/mux) **Handler** and **HandlerFunc** func signature to automatically generate the `@router` tag, and `@param` tag/s. [See gorilla/mux Handler Functions](#gorillamux-handler-functions).

> Note: It does not generate/support all OpenAPI v3.0.2 blocks, the tool handles just a subset required by our internal needs, it might be extended if there will be high demand on extension, or if anyone would like to contribute.

## Examples
[Basic API Project](https://github.com/spaceavocado/apidoc/tree/master/example)

## Getting Started
1. Add annotation into your API source code files, [See API Annotation in Comments](#api-annotation-in-comments).
   
2. Get the APIDoc by using:
    ```sh
    go get -u github.com/spaceavocado/apidoc
    ```
    > This will create a binary in your **$GOPATH/bin** folder called `apidoc` (Mac/Unix) or `apidoc.exe` (Windows).

3. Now you can run `apidoc` command from your shell.
   > Make sure that **$GOPATH/bin** is in your Environment Variables:
    - **Linux**:
      ```sh
      export PATH=$PATH:$GOPATH/bin
      ```
      *Note*: assumption that your have correctly set $GOPATH env. variable.

    - **Windows**: "Control Panel" > "System" > "Edit the system environment variables" > "Advanced" > "Environment Variables" > "Path" > "Edit". and add the directory.

4. Run `apidoc` in the your project's root folder. This will extract and process your annotation and it generates the output YAML file.
    ```sh
    apidoc -m main.go -e handler -o docs/api
    ```
    *Note*: the example shows the default flag values, for more details, [See APIDoc CLI](#apidoc-cli).

5. Preview the documentation in the [Swagger Editor](https://editor.swagger.io/), i.e. put the openapi.yaml content into the editor.

# API Annotation in Comments
An API annotation is represented as an code comment starting with **@** symbol followed by documentation tag and it value or parameters. Examples:

* `// @title Example RESTful API`, tag **title** with value "Example RESTful API"
* `// @server api.domain.com/v3 Production`, tag **server** with params: "api.domain.com/v3", "Production" 

The the value of the tag or the last parameter of the tag captures the multiline comments, example:
```go
// @desc Lorem ipsum dolor sit amet, consectetur adipiscing
// elit. Nullam rhoncus magna nunc, in faucibus metus pulvinar
// et. Mauris pellentesque 
```
> The value of desc tag is captured as: "Lorem ipsum dolor sit amet, consectetur adipiscing elit. Nullam rhoncus magna nunc, in faucibus metus pulvinar et. Mauris pellentesque"
```go
// @server api.domain.com/v3 Lorem ipsum dolor sit amet, consectetur
// elit. Nullam rhoncus magna nunc.
```
> The last parameter, **description** of the server tag is captured as: "Lorem ipsum dolor sit amet, consectetur elit. Nullam rhoncus magna nunc."
  
## Main Section
This section should be located in the **main** file, passed to the [APIDoc CLI](#apidoc-cli) in **-m** flag (defaults to **main.go**).

### Supported Tags
> Note: **()** within **Annotation** indicates an annotation parameter captured by the generator.

| Annotation                 | Description                                                                                             | OpenAPI Spec.                                                      | Example                                                      |
| -------------------------- | ------------------------------------------------------------------------------------------------------- | ------------------------------------------------------------------ | ------------------------------------------------------------ |
| title (value)              | **REQUIRED**. The title of the application.                                                             | https://swagger.io/specification/#infoObject field: title          | // @title Example RESTful API                                |
| desc (value)               | A short description of the application.                                                                 | https://swagger.io/specification/#infoObject field: description    | // @desc An example of publicly accessible API               |
| ver (value)                | **REQUIRED**. The version of the OpenAPI document                                                       | https://swagger.io/specification/#infoObject field: version        | // @ver 1.0.0                                                |
| terms (value)              | A URL to the Terms of Service for the API.                                                              | https://swagger.io/specification/#infoObject field: termsOfService | // @terms https://domain.com/api/terms                       |
| contact.name (value)       | The identifying name of the contact person/organization.                                                | https://swagger.io/specification/#contactObject field: name        | // @contact.name API Support                                 |
| contact.url (value)        | The URL pointing to the contact information.                                                            | https://swagger.io/specification/#contactObject field: url         | // @contact.url https://domain.com/api/support               |
| contact.email (value)      | The email address of the contact person/organization.                                                   | https://swagger.io/specification/#contactObject field: email       | // @contact.email support@domain.com                         |
| lic.name (value)           | The license name used for the API.                                                                      | https://swagger.io/specification/#licenseObject field: name        | // @lic.name Apache 2.0                                      |
| lic.url (value)            | A URL to the license used for the API.                                                                  | https://swagger.io/specification/#licenseObject field: url         | // @lic.url https://www.apache.org/licenses/LICENSE-2.0.html |
| server (url) (description) | An object representing a Server <br><br>Note: There might be many @server tags within the main section. | https://swagger.io/specification/#serverObject                     | // @server api.domain.com/v3 Production                      |

### Example
```go
// @title An example authentication API
// @desc Publicly accessible authentication REST API.
// @terms https://domain.com/docs/api/terms
//
// @contact.name API Support
// @contact.url https://domain.com/contact
// @contact.email support@domain.com
//
// @lic.name Apache 2.0
// @lic.url https://www.apache.org/licenses/LICENSE-2.0.html
//
// @ver 1.0
// @server https://auth.domain.com/v3 Production API
// @server https://auth.dev.domain.com/v3 Development API
func main() {}
```

## An Endpoint
An endpoint is being considered as a API comment annotation block found within any file located inside the **endpoints** root folder, passed to the [APIDoc CLI](#apidoc-cli) in **-e** flag (defaults to **./**).
> For better performance is highly recommended to pass the endpoints root folder as a flag to the CLI to avoid unnecessary file processing.

### Supported Tags
> Note: **()** within **Annotation** indicates an annotation parameter captured by the generator.

| Annotation                                                 | Description                                                                                                                                                                                   | OpenAPI Spec.                                                        | Example                                                                                               |
| ---------------------------------------------------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | -------------------------------------------------------------------- | ----------------------------------------------------------------------------------------------------- |
| summary (value)                                            | A short summary of what the operation does.                                                                                                                                                   | https://swagger.io/specification/#operationObject field: summary     | @sumamry Lorem ipsum dolor                                                                            |
| desc (value)                                               | A verbose explanation of the operation behavior.                                                                                                                                              | https://swagger.io/specification/#operationObject field: description | @desc Lorem ipsum dolor sit amet, consectetur adipiscing eli                                          |
| id (value)                                                 | Unique string used to identify the operation. The id MUST be unique among all operations described in the API. The operationId value is case-sensitive.                                       | https://swagger.io/specification/#operationObject field: operationId | // @id login                                                                                          |
| tag (array)                                                | A list of tags for API documentation control. Tags can be used for logical grouping of operations by resources or any other qualifier.<br><br>Note: Comma separated value, or a single value. | https://swagger.io/specification/#operationObject field: tags        | // @tag User<br><br>// @tag User, Authentication                                                      |
| accept (array)                                             | A list of accepted request content mime types<br><br>[See Mime Types Annotation](#mime-types-annotation)                                                                                      | n/a                                                                  | // @accept json<br><br>// @accept application/json, application/x-www-form-urlencoded                 |
| produce (array)                                            | **REQUIRED**. A list of supported content response mime types<br><br>[See Mime Types Annotation](#mime-types-annotation)                                                                      | n/a                                                                  | // @produce json<br><br>// @produce application/json, application/x-www-form-urlencoded               |
| param (name) (in) {(type)} (required) (description)        | Describes a single operation parameter. <br><br>[See Param Tag](#param-tag)<br><br>Note: There might be many @param tags within the endpoint block.                                           | https://swagger.io/specification/#parameterObject                    | // @param token path {string} true Security Token                                                     |
| body (reference)                                           | Describes a single request body.<br><br>[See Body Tag](#body-tag)                                                                                                                             | https://swagger.io/specification/#requestBodyObject                  | //@body model.Login                                                                                   |
| swrap (reference) (field pointer)                          | Success object wrapper. If this tag is set, any success response object is being wrapped with this object on the desired **pointer field**<br><br>[See Wrapper Tag](#wrapper-tag)             | n/a                                                                  | // @swrap response.Success                                                                            |
| success (code) {(type)} (reference or empty) (description) | **REQUIRED**. Describes a single success response from an API Operation.<br><br>[See Response Tag](#response-tag)                                                                             | https://swagger.io/specification/#responseObject                     | // @success 200 {object} response.Success OK<br><br>// @success 200 {string} OK                       |
| fwrap (reference) (field pointer)                          | Failure object wrapper. If this tag is set, any failure response object is being wrapped with this object on the desired **pointer field**<br><br>[See Wrapper Tag](#wrapper-tag)             | n/a                                                                  | // @fwrap response.Error                                                                              |
| failure (code) {(type)} (reference or empty) (description) | Describes a single failure response from an API Operation.<br><br>[See Response Tag](#response-tag)                                                                                           | https://swagger.io/specification/#responseObject                     | // @failure 401 {object} response.AuthError Unauthorized<br><br>// @failure 401 {string} Unauthorized |
| subrouter (value)                                          | Name of the subrouter used for this endpoint. <br><br>[See gorilla/mux Subrouter](#gorillamux-subrouter)                                                                                      | n/a                                                                  | // @subrouter user [post]                                                                             |
| router (path) [(method)]                                   | **REQUIRED**. Describes the operations available on a single path, i.e. endpoint URL<br><br>[See Path Tag](#path-tag)                                                                                       | https://swagger.io/specification/#pathItemObject                     | // @router /login [post]                                                                              |

### Param Tag
> *Annotation:* param (name) (in) {(type)} (required) (description)

This section contains detailed information about the **param** tag
#### name 
* The name of the parameter. Parameter names are case sensitive.
* If `in` is "path", the name field MUST correspond to the associated router url parameter, e.g.:
   `// @param token path {string} true Security token` must match with `// @router /auth/{token} [get]`

#### in 
* The location of the parameter. Possible values are "query", "header", "path" or "cookie".

#### type
* the outer "{}" are used just as visual separators of type field, i.e. their are not required.
  E.g. `// @param token path {string} true Security token` = `// @param token path string true Security token`
* Param tag does not support struct type, just base go types, [See Type Mapping](#type-mapping)
* **Array annotation**: `{[]string}`, `{[]int}`, etc.

### Body Tag
> *Annotation:* body (reference)

#### reference
* go structure used as request model, example: `// @body model.Login`
  
  ```go
  // Login struct located in "model" package,
  // referenced in the go file where the endpoint
  // annotation is set
  type Login struct {
    // User's email address
    Username string `json:"username"`
    Password string
  }
  ```
  will be resolved as:
  ```yaml
  schema:
    type: object
    parameters:
      username:
        type: string
        description: User's email address
      password:
        type: string
  ```
* The reference structure is being resolved recursively, i.e. it might contain fields referencing other go struct.
* [See Struct Annotation](#struct-annotation) for more details.

### Wrapper Tag
> *Annotation:* swrap (reference) (pointer field), fwrap (reference) (field pointer)

If this tag is set, any success/failure response object is being wrapped with this object on the desired pointer field. This is useful if you are using a generic API response object with various data field.

#### Example
```go
type APISuccess struct {
  Status string `json:"status"`
  Data interface{} `json:"data"`
}
type Profile struct {
  Firstname int `json:"firstname"`
  Lastname string `json:"lastname"`
}
```

`// @swrap APISuccess data` and `// @success 200 {object} Profile OK` will produce:
```yaml
"200":
  description: OK
  content:
    application/json:
      schema:
        type: object
        properties:
          status:
            type: string
          data:
            schema:
              $ref: "#/components/schemas/Profile"
```

* The reference structure is being resolved recursively, i.e. it might contain fields referencing other go struct.
* [See Struct Annotation](#struct-annotation) for more details.

### Response Tag
> *Annotation:* success (code) {(type)} (reference or string) (description), failure (code) {(type)} (reference or string) (description)

#### code
* Any valid HTTP reponse code

#### type
* The outer "{}" are used just as visual separators of type field, i.e. their are not required.
  E.g. `// @success 200 {string} string OK` = `// @success 200 string string OK`
* Supported: object, string
  * if set to object, the next parameter expected is **reference**
  * if set to string, next parameter is skipped (reference)

#### reference
* Reference response go struct
* **Array annotation**: `{[]response.Car}`, etc.
* The reference structure is being resolved recursively, i.e. it might contain fields referencing other go struct.
* [See Struct Annotation](#struct-annotation) for more details.

### Path Tag
> *Annotation:* router (path) [(method)]

#### path 
* It might contain the "path" parameters, it this format: `/login/{token}` where token is the name of param defined in the [Param Tag](#patam-tag).

#### method
* HTTP method
* The outer "[]" are used just as visual separators of method field, i.e. their are not required.
  E.g. `// @router /login [post]` = `// @router /login post`
* It might contain an array of methods, e.g.: `[post, put]`

### Example Endpoint Annotation
```go
// Login request
// @summary Login user
// @desc Authentication request
// @id login
// @tag Authentication
// @accept json, application/x-www-form-urlencoded
// @produce json
// @body model.Login
// @swrap response.Data data
// @success 200 {object} request.Token OK
// @failure 401 {object} response.Error Unauthorized.
// See error code and error message for more details.
// @failure 500 {string} Internal Server Error
// @router /login [post]
func handler(w http.ResponseWriter, r *http.Request) {}

// @summary Activate user
// @desc User activation via the token request.
// @id activate
// @tag Registration
// @param token path {string} true Activation token
// @produce json
// @success 200 {string} OK
// @failure 500 {string} Internal Server Error
// @router /registration/activate/{token} [get]
func handler(w http.ResponseWriter, r *http.Request) {}

// Handlers with passed gorilla mux router
//
// In this example the @router tag is being resolved
// from the HandleFunc or Handle gorilla *mux.Router func.
// @param tags ara also resolved from the func path string.
//
// These tags will be extracted automatically:
// @router /person/{id} [get]
// @param id path {string} true
func Handlers(r *mux.Router) {
  // GetPerson handler
  // @summary Person
  // @desc Get person by ID.
  // @id person
  // @tag Person
  // @produce json
  // @success 200 {object} Person OK
  // @failure 500 {string} Internal Server Error
  r.HandleFunc("/person/{id:[0-9]+}", GetPerson).Methods("GET")
}
```

## gorilla/mux Handler Functions
`@router` tag and  `@param` tags could automatically resolved if the endpoint annotation is placed above the [gorilla/mux](https://github.com/gorilla/mux) Handler or HandlerFunc function:
```go
// GetPerson handler
// @summary Person
// @desc Get person by ID.
// @id person
// @tag Person
// @produce json
// @success 200 {object} Person OK
// @failure 500 {string} Internal Server Error
r.HandleFunc("/person/{id:[0-9]+}", GetPerson).Methods("GET")
```
Automatically resolved tags will be be:
* `@router /person/{id} [get]`
* `@param id path {string} true`

### Notes
* This process is skipped if the endpoint annotation contains the `@router` tag
* If the endpoint annotation contains a `@param` tag found in the func path parameter, it is not being overwritten by the this process nor double annotated.
* Please [see gorilla/mux Subrouter](#gorillamux-subrouter) section for more information how to work with gorilla/mux subrouters.

## gorilla/mux Subrouter
An endpoint can use `subrouter` tag to connect a endpoint with the subrouter to resolve the final endpoint URL.

### Subrouter Annotation
| Annotation       | Description                   | Example             |
| ---------------- | ----------------------------- | ------------------- |
| router (name)    | Name of the subrouter.        | // @router account  |
| subrouter (name) | Name of the parent subrouter. | // @subrouter admin |

The annotation must be placed above the **gorilla.mux Subrouter** method anywhere within the Main file or in any file within the endpoints root folder.

#### Example
```go
// @router admin
r.PathPrefix("/admin").Subrouter()

// @router user
// @subrouter admin
r.PathPrefix("/user").Subrouter()

// @summary List of users
// @produce json
// @success 200 {object} Person OK
// @subrouter user
r.HandleFunc("/list", GetPersonList).Methods("GET")
```
> The URL resolved for the "List of users" endpoint will be `/admin/user/list`

## Mime Types Annotation
| Mime Type                         | Annotation                              |
| --------------------------------- | --------------------------------------- |
| text/plain                        | text, text/plain                        |
| text/html                         | html, text/html                         |
| text/xml                          | xml, text/xml                           |
| application/json                  | json, application/json                  |
| application/x-www-form-urlencoded | form, application/x-www-form-urlencoded |
| multipart/form-data               | multipart, multipart/form-data          |
| application/vnd.api+json          | json-api, application/vnd.api+json      |
| application/x-json-stream         | json-stream, application/x-json-stream  |
| application/octet-stream          | octet-stream, application/octet-stream  |
| image/png                         | png, image/png                          |
| image/jpeg                        | jpeg, image/jpeg                        |
| image/jpeg                        | jpg,  image/jpeg                        |
| image/gif                         | gif,  image/gif                         |

## Struct Annotation
```go
type Profile struct {
  // User's email
  Username string `json:"username" required:"true"`
  Status UserStatus `apitype:"int"`
}
type UserStatus int
```

* Comment above the field is being captured as field "description"
* `json:"x"` overrides the field name
* `apitype:"x"` overrides the field type
* `required:"true"` marks the field as required

## Data Types Conversion
Go types are being converted into OpenAPI accepted format

| go Type | Converted Type |
| ------- | -------------- |
| byte    | integer        |
| rune    | integer        |
| int     | integer        |
| int8    | integer        |
| int16   | integer        |
| int32   | integer        |
| int64   | integer        |
| uint    | integer        |
| uint8   | integer        |
| uint16  | integer        |
| uint32  | integer        |
| uint64  | integer        |
| uintptr | integer        |
| float32 | number         |
| float64 | number         |
| bool    | boolean        |

# Tips

## Annotation over Multiple Lines
Last parameter of any tag is could spread over multiple lines, e.g.:
```go
// @desc Lorem ipsum dolor sit amet, consectetur adipiscing
// elit. Nullam rhoncus magna nunc, in faucibus metus pulvinar
// et. Mauris pellentesque enim justo

// @failure 500 {string} Lorem ipsum dolor sit amet, consectetur
// adipiscing elit. Nullam rhoncus magna nunc, in faucibus metus
// pulvinar et. Mauris pellentesque enim justo.
```

## Array References
Use **[]** annotation in from of the reference.
```go
// @body []request.Person

// @success 200 {object} []response.Person OK
```

## And Endpoint With Many Decralarions
An endpoint, with the same URL could be declared separately with different methods, and the generator will properly group them.
### Example
```go
// @summary Get User
// @param id path {int} true User ID
// @produce text
// @success 200 {string} OK
// @router /user/{id} [post]

// ...

// @summary Delete User
// @param id path {int} true User ID
// @produce text
// @success 200 {string} OK
// @router /user/{id} [delete]
```
will produce
```yaml
/user/{id}:
  get:
    summary: Get User
    parameters:
    - name: id
      description: User ID
      in: path
      required: true
      schema:
        type: integer
    responses:
      "200":
        description: OK
        content:
          plain/text:
            schema:
              type: string
  delete:
    summary: Delete User
    parameters:
    - name: id
      description: User ID
      in: path
      required: true
      schema:
        type: integer
    responses:
      "200":
        description: OK
        content:
          plain/text:
            schema:
              type: string
```

# APIDoc CLI
Run the `apidoc -h` to all flags and commands:
```console
API Documentation Generator

Usage:
   [flags]
   [command]

Available Commands:
  help        Help about any command
  version     Show the APIDoc version

Flags:
  -e, --endpoints string   Root endpoints folder (default "./")
  -h, --help               Help for this command
  -m, --main string        Main API documentation file (default "main.go")
  -o, --output string      Documentation output folder (default "docs/api")
  -v, --verbose            Show generation warnings

Use " [command] --help" for more information about a command.
```

# About the Project
This project was inspired by [swaggo/swag](https://github.com/swaggo/swag/), designed mainly to handle our API documentation needs, i.e. add support for response wrappers, generate OpenAPI v3.X documentation. Any feedback, contribution to this project is welcomed.

The project is in a beta phase, therefore there might be major changes in near future, the annotation should stay the same, though.

# Contributing
When contributing to this repository, please first discuss the change you wish to make via issue, email, or any other method with the owners of this repository before making a change. 

## Pull Request Process
1. Fork it
2. Create your feature branch (`git checkout -b ft/new-feature-name`)
3. Commit your changes (`git commit -am 'Add some feature'`)
4. Push to the branch (`git push origin ft/new-feature-name`)
5. Create new Pull Request

> Please make an issue first if the change is likely to increase.

# License
APIDoc is released under the Apache 2.0 license. See [LICENSE.txt](https://github.com/spaceavocado/apidoc/blob/master/LICENSE.txt)

 readmeEtag: '"407860f005d432f04e671a9e1ddb89f05aa66be2"' readmeLastModified: Wed, 13 Mar 2019 21:55:10 GMT repositoryId: 168361295 description: >- Generate RESTful API documentation from GO source files into the OpenAPI v3.0.2 specification (formal Swagger 2.0 Specification). created: '2019-01-30T15:00:04Z' updated: '2023-10-13T20:53:04Z' language: Go archived: false stars: 17 watchers: 2 forks: 5 owner: spaceavocado logo: https://avatars.githubusercontent.com/u/47183501?v=4 license: Apache-2.0 repoEtag: '"3ef215bc75cc39a9969043252cad25d37021c12086b6ed8558d9fe559f015430"' repoLastModified: Fri, 13 Oct 2023 20:53:04 GMT foundInMaster: true category: Parsers id: 5dcebefc091c51707c14ac3e9f3253e5 - source: openapi3 tags repository: https://github.com/go-courier/oas v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWMgQnVpbGRlcgoKWyFbR29Eb2MgV2lkZ2V0XShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2dvLWNvdXJpZXIvb2FzP3N0YXR1cy5zdmcpXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2dvLWNvdXJpZXIvb2FzKQpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9nby1jb3VyaWVyL29hcy5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9nby1jb3VyaWVyL29hcykKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2dvLWNvdXJpZXIvb2FzL2JyYW5jaC9tYXN0ZXIvZ3JhcGgvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2dvLWNvdXJpZXIvb2FzKQpbIVtHbyBSZXBvcnQgQ2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vZ28tY291cmllci9vYXMpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20vZ28tY291cmllci9vYXMpCgoKW09wZW5BUEkgU3BlY10oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24pIDMuMC4zIGJ1aWxkZXIgZm9yIEdvbGFuZwoKIyMgVXNhZ2UKCmBgYGJhc2gKZ28gZ2V0IGdpdGh1Yi5jb20vZ28tY291cmllci9vYXMKYGBgCgo= readmeEtag: '"cad75c7f933591c657627a9abae8032d1d4fb8f3"' readmeLastModified: Wed, 24 Feb 2021 03:01:47 GMT repositoryId: 116089474 description: OpenAPI Spec builder in go created: '2018-01-03T04:01:46Z' updated: '2023-10-26T06:26:25Z' language: Go archived: false stars: 17 watchers: 4 forks: 3 owner: go-courier logo: https://avatars.githubusercontent.com/u/37426719?v=4 license: MIT repoEtag: '"b4e0e89e3b5e46ef080647421cb6cb12fff772bab9ee6da6cdaa80362df705d3"' repoLastModified: Thu, 26 Oct 2023 06:26:25 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: e6691a9306efe4c41d1db1f42c020391 - source: openapi3 tags repository: https://github.com/stefanbratanov/sofia-supermarkets-api v3: true repositoryMetadata: base64Readme: >- IyBzb2ZpYS1zdXBlcm1hcmtldHMtYXBpCgpbIVtidWlsZF0oaHR0cHM6Ly9naXRodWIuY29tL1N0ZWZhbkJyYXRhbm92L3NvZmlhLXN1cGVybWFya2V0cy1hcGkvYWN0aW9ucy93b3JrZmxvd3MvYnVpbGQueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9TdGVmYW5CcmF0YW5vdi9zb2ZpYS1zdXBlcm1hcmtldHMtYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnltbCkKWyFbUXVhbGl0eSBHYXRlIFN0YXR1c10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9U3RlZmFuQnJhdGFub3Zfc29maWEtc3VwZXJtYXJrZXRzLWFwaSZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9U3RlZmFuQnJhdGFub3Zfc29maWEtc3VwZXJtYXJrZXRzLWFwaSkKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL1N0ZWZhbkJyYXRhbm92L3NvZmlhLXN1cGVybWFya2V0cy1hcGkvYnJhbmNoL21hc3Rlci9ncmFwaC9iYWRnZS5zdmc/dG9rZW49M1YzVEhJWTZBWCldKGh0dHBzOi8vY29kZWNvdi5pby9naC9TdGVmYW5CcmF0YW5vdi9zb2ZpYS1zdXBlcm1hcmtldHMtYXBpKQoKQVBJINC30LAg0LjQt9Cy0LvQuNGH0LDQvdC1INC90LAg0LjQvdGE0L7RgNC80LDRhtC40Y8g0LfQsCDQv9GA0L7QtNGD0LrRgtC4INC+0YIg0YHRg9C/0LXRgNC80LDRgNC60LXRgtC4INCyINCh0L7RhNC40Y8uCgpbIVtTd2FnZ2VyIFVJXShodHRwczovL3ZhbGlkYXRvci5zd2FnZ2VyLmlvL3ZhbGlkYXRvcj91cmw9aHR0cHM6Ly9hcGkubmFvZmVydGEubmV0L3YzL2FwaS1kb2NzKV0oaHR0cHM6Ly9hcGkubmFvZmVydGEubmV0L3N3YWdnZXItdWkuaHRtbCkKCiMjINCh0LDQudGC0L7QstC1LCDQuNC30L/QvtC70LfQstCw0YnQuCDRgtC+0LLQsCBBUEkKCi0gW9CQ0LvQutC+0YXQvtC7INC90LAg0L7RhNC10YDRgtCwXShodHRwczovL25hb2ZlcnRhLm5ldC8pCgojIyDQn9C+0LTQtNGK0YDQttCw0L3QuCDRgdGD0L/QtdGA0LzQsNGA0LrQtdGC0LgKCi0gW3hdIEJpbGxhCgo+INCf0L7Qu9GD0YfQsNCy0LAg0LjQvdGE0L7RgNC80LDRhtC40Y8g0LfQsCDQv9GA0L7QtNGD0LrRgtC40YLQtSDQvtGCIGh0dHBzOi8vc3NiYmlsbGEuc2l0ZS8uINCY0LfQvtCx0YDQsNC20LXQvdC40Y/RgtCwINGB0LUg0LjQt9Cy0LvQuNGH0LDRgiDQstGK0Lcg0L7RgdC90L7QstCwCj4g0L3QsAo+INC/0YrRgNCy0LjRjyDRgNC10LfRg9C70YLQsNGCINCyINGC0YrRgNGB0LXQvdC10YLQviDQvdCwINC40LfQvtCx0YDQsNC20LXQvdC40Y8g0LIgR29vZ2xlCgotIFt4XSBGYW50YXN0aWNvCgo+INCY0LfRgtC10LPQu9GPIFBERiDQsdGA0L7RiNGD0YDQuNGC0LUg0L7RgiBodHRwczovL3d3dy5mYW50YXN0aWNvLmJnL3NwZWNpYWwtb2ZmZXJzINC4INCw0L3QsNC70LjQt9C40YDQsCDQuNC90YTQvtGA0LzQsNGG0LjRj9GC0LAg0LfQsAo+INC/0YDQvtC00YPQutGC0LjRgtC1INCyINGC0Y/RhS4g0JjQt9C+0LHRgNCw0LbQtdC90LjRj9GC0LAg0YHQtSDQuNC30LLQu9C40YfQsNGCINCy0YrQtyDQvtGB0L3QvtCy0LAg0L3QsCDQv9GK0YDQstC40Y8g0YDQtdC30YPQu9GC0LDRgiDQsiDRgtGK0YDRgdC10L3QtdGC0L4g0L3QsAo+INC40LfQvtCx0YDQsNC20LXQvdC40Y8KPiDQsiBHb29nbGUuCgotIFt4XSBLYXVmbGFuZAoKPiDQn9C+0LvRg9GH0LDQstCwINC40L3RhNC+0YDQvNCw0YbQuNGPINC30LAg0L/RgNC+0LTRg9C60YLQuNGC0LUg0L7RgiBodHRwczovL3d3dy5rYXVmbGFuZC5iZy8uCgotIFt4XSBMaWRsCgo+INCf0L7Qu9GD0YfQsNCy0LAg0LjQvdGE0L7RgNC80LDRhtC40Y8g0LfQsCDQv9GA0L7QtNGD0LrRgtC40YLQtSDQvtGCIGh0dHBzOi8vd3d3LmxpZGwuYmcvLgoKLSBbeF0gVC1NYXJrZXQKCj4g0J/QvtC70YPRh9Cw0LLQsCDQuNC90YTQvtGA0LzQsNGG0LjRjyDQt9CwINC/0YDQvtC00YPQutGC0LjRgtC1INC+0YIgaHR0cHM6Ly90bWFya2V0b25saW5lLmJnLy4KCi0gWyBdIE1FVFJPCi0gWyBdIEthbSBNYXJrZXQKLSBbIF0gQ0JBCi0gWyBdIFByb01hcmtldAotIFsgXSBIaXQgTWF4CgojIyBUZWNoIFN0YWNrCgoqIEtvdGxpbgoqIEdyYWRsZQoqIFNwcmluZyBCb290CgojIyBDb2RlIFN0eWxlCgpLb3RsaW4gY29kZSBjb252ZW50aW9ucywgYmFzZWQgb24gW2t0Zm10XShodHRwczovL2dpdGh1Yi5jb20vZmFjZWJvb2sva3RmbXQpCmFuZCBbZ29vZ2xlLWphdmEtZm9ybWF0XShodHRwczovL2dpdGh1Yi5jb20vZ29vZ2xlL2dvb2dsZS1qYXZhLWZvcm1hdCkgYXJlIHVzZWQgaW4gdGhpcyBwcm9qZWN0LiBUbwpyZWZvcm1hdCBjb2RlLCBydW46CgpgYGAKLi9ncmFkbGV3IHNwb3RsZXNzQXBwbHkKYGBgCgojIyBSdW5uaW5nIExvY2FsbHkKCkluc3RhbGwgSkRLIDE3IG9yIGFib3ZlIGZyb20gW2hlcmVdKGh0dHBzOi8vamRrLmphdmEubmV0LykuCgoqICoqVW5peCoqCgpgYGAKLi9ncmFkbGV3IGJvb3RSdW4KYGBgCgoqICoqV2luZG93cyoqCgpgYGAKZ3JhZGxldy5iYXQgYm9vdFJ1bgpgYGAKClRoZSBmb2xsb3dpbmcgZW52aXJvbm1lbnQgdmFyaWFibGVzIG5lZWQgdG8gYmUgc2V0IHByaW9yIHRvIHJ1bm5pbmc6CgoqIERCX1VSTAoqIERCX1VTRVJOQU1FCiogREJfUEFTU1dPUkQKKiBHT09HTEVfQVBJX0tFWQoqIENMT1VESU5BUllfQVBJX1NFQ1JFVAoKTmF2aWdhdGUgdG8gPGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpLmh0bWw+IHRvIGNoZWNrIHRoZSBBUEkgZG9jdW1lbnRhdGlvbiBhbmQKdGVzdCB0aGUgZW5kcG9pbnRzLgoKIVtBUEkgRG9jdW1lbnRhdGlvbl0oaW1hZ2VzL3N3YWdnZXItdWkucG5nKQo= readmeEtag: '"008584ea68aebd1b1f7673db8853862c70c939a5"' readmeLastModified: Thu, 01 Aug 2024 20:36:54 GMT repositoryId: 346132785 description: API за извличане на информация за продукти от супермаркети в София created: '2021-03-09T20:16:47Z' updated: '2026-01-22T07:32:56Z' language: Kotlin archived: false stars: 24 watchers: 2 forks: 7 owner: StefanBratanov logo: https://avatars.githubusercontent.com/u/14827647?v=4 license: Apache-2.0 repoEtag: '"88df90a2419a2b94bdb5e46af695f3d907af7c7cd419ec83356d7b982bf8a30f"' repoLastModified: Thu, 22 Jan 2026 07:32:56 GMT foundInMaster: true category: SDK id: 04dc35e7b9c33ef61294aaf0e176d771 - source: openapi3 tags repository: https://github.com/serafin-labs/serafin v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+PGltZyBzcmM9Imh0dHBzOi8vc2VyYWZpbi1sYWJzLmdpdGh1Yi5pby9pbWFnZXMvbG9nby1zZXJhZmluLXdpdGgtdGV4dC0xMDgwLnBuZyIgd2lkdGg9IjMwMCIvPjwvcD4KCioqU2VyYWZpbioqIGlzIGFuICpBUEkgZnJhbWV3b3JrKiBkZXNpZ25lZCB0byBxdWlja2x5IHNldCB1cCBhIHJvYnVzdCAqKnNlbGYtZGVzY3JpcHRpdmUgUkVTVCBBUEkqKiB3cml0dGVuIGluICpub2RlSlMvVHlwZXNjcmlwdCouCgpJdCBpcyBiYXNlZCBvbiAqKk9wZW4gQVBJIDMqKiwgKipKU09OIFNjaGVtYSoqIGFuZCAqKkdyYXBoUUwqKiBzdGFuZGFyZHMuCgpbIVtHcmVlbmtlZXBlciBiYWRnZV0oaHR0cHM6Ly9iYWRnZXMuZ3JlZW5rZWVwZXIuaW8vc2VyYWZpbi1sYWJzL3NlcmFmaW4uc3ZnKV0oaHR0cHM6Ly9ncmVlbmtlZXBlci5pby8pClshW0RlcGVuZGVuY2llc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kYXZpZC9zZXJhZmluLWxhYnMvc2VyYWZpbi5zdmcpXShodHRwczovL2RhdmlkLWRtLm9yZy9zZXJhZmluLWxhYnMvc2VyYWZpbikKWyFbTWFpbnRhaW5hYmlsaXR5XShodHRwczovL2FwaS5jb2RlY2xpbWF0ZS5jb20vdjEvYmFkZ2VzL2JlYmExNjFhZTBlNWY0ZjY5Yzc5L21haW50YWluYWJpbGl0eSldKGh0dHBzOi8vY29kZWNsaW1hdGUuY29tL2dpdGh1Yi9zZXJhZmluLWxhYnMvc2VyYWZpbi9tYWludGFpbmFiaWxpdHkpClshW1Rlc3QgQ292ZXJhZ2VdKGh0dHBzOi8vYXBpLmNvZGVjbGltYXRlLmNvbS92MS9iYWRnZXMvYmViYTE2MWFlMGU1ZjRmNjljNzkvdGVzdF9jb3ZlcmFnZSldKGh0dHBzOi8vY29kZWNsaW1hdGUuY29tL2dpdGh1Yi9zZXJhZmluLWxhYnMvc2VyYWZpbi90ZXN0X2NvdmVyYWdlKQoKIyMgSW5zdGFsbGF0aW9uClRoZXJlJ3Mgbm8gbnBtIHBhY2thZ2UgeWV0ISBXZSBhcmUgY2xvc2UgdG8gdGhlIGFscGhhIHJlbGVhc2UgYW5kIHdlIHdpbGwgcHJvZHVjZSBwYWNrYWdlcyBmb3IgdGhpcyB2ZXJzaW9uLgpJZiB5b3Ugd2FudCB0byB0ZXN0ICoqc2VyYWZpbioqIHlvdSBjYW4gY2xvbmUgdGhlIHJlcG8gYW5kIHJ1biBpdCBsb2NhbGx5IG9yIHlvdSBjYW4gaW5jbHVkZSBhIGRpcmVjdCBnaXQgcmVmZXJlbmNlIHRvIHlvdXIgYGBgcGFja2FnZS5qc29uYGBgIDoKCmBgYGpzb24KIkBzZXJhZmluL2FwaSI6ICJnaXQrc3NoOi8vZ2l0QGdpdGh1Yi5jb20vc2VyYWZpbi1mcmFtZXdvcmsvc2VyYWZpbi5naXQiCmBgYAoKIyMgQ29uY2VwdHMKCklmIHlvdSB3YW50IHRvIGtub3cgbW9yZSBhYm91dCBTZXJhZmluIGNvbmNlcHRzIGFuZCBmZWF0dXJlcywgZ28gdG8gb3VyIFtvdmVydmlldyBkb2N1bWVudF0oLi9taXNjL2RvYy9PVkVSVklFVy5tZCkKCiMjIEdldHRpbmcgc3RhcnRlZAoKSWYgeW91IGp1c3Qgd2FudCB0byBnZXQgc3RhcnRlZCBhbmQgd3JpdGUgc29tZSBjb2RlLCBnbyB0byBvdXIgW3dhbGt0aHJvdWdoIGRvY3VtZW50XSguL21pc2MvZG9jL1dBTEtUSFJPVUdILm1kKQoKIyMgV2hhdCBkb2VzIGl0IGxvb2sgbGlrZSA/CgpBIHZlcnkgc2ltcGxlIGV4YW1wbGUgbG9va3MgbGlrZSB0aGF0IDoKCmBgYHR5cGVzY3JpcHQKaW1wb3J0ICogYXMgZXhwcmVzcyBmcm9tICdleHByZXNzJzsKaW1wb3J0ICogYXMgYm9keVBhcnNlciBmcm9tICdib2R5LXBhcnNlcic7CmltcG9ydCB7IFNjaGVtYUJ1aWxkZXIgfSBmcm9tICdAc2VyYWZpbi9zY2hlbWEtYnVpbGRlcic7CmltcG9ydCB7IEFwaSwgUGlwZWxpbmVTb3VyY2VJbk1lbW9yeSwgUmVzdFRyYW5zcG9ydCB9IGZyb20gJ0BzZXJhZmluL2FwaSc7CgovLyBleHByZXNzIGluaXRpYWxpemF0aW9uCmxldCBhcHAgPSBleHByZXNzKCk7CmFwcC51c2UoYm9keVBhcnNlci5qc29uKCkpOwoKLy8gRGVjbGFyZSBvdXIgQXBpIHdpdGggaXRzIGdlbmVyYWwgaW5mb3JtYXRpb24KbGV0IGFwaSA9IG5ldyBBcGkoYXBwLCB7CiAgICAib3BlbmFwaSI6ICIzLjAuMCIsCiAgICAiaW5mbyI6IHsKICAgICAgICAidmVyc2lvbiI6ICIxLjAuMCIsCiAgICAgICAgInRpdGxlIjogIkFuIEFQSSIKICAgIH0sCiAgICBwYXRoczoge30KfSk7CmFwaS5jb25maWd1cmUobmV3IFJlc3RUcmFuc3BvcnQoKSk7CgovLyBEZWNsYXJlIGEgU2NoZW1hIGZvciBvdXIgImVudGl0eSIKbGV0IGFNb2RlbFNjaGVtYSA9IFNjaGVtYUJ1aWxkZXIuZW1wdHlTY2hlbWEoKS5hZGRTdHJpbmcoImlkIikuYWRkU3RyaW5nKCJkYXRhIik7CgovLyBEZWZpbmUgdGhlIHBpcGVsaW5lLCBpdCBzdG9yZXMgZGF0YSBpbnRvIG1lbW9yeSBkaXJlY3RseQpsZXQgYVBpcGVsaW5lID0gKG5ldyBQaXBlbGluZVNvdXJjZUluTWVtb3J5KGFNb2RlbFNjaGVtYSkpCiAgLy8ucGlwZSguLi4pIC8vIEFkZCBhIHBpcGVsaW5lIHRvIGV4dGVuZCB0aGUgYmVoYXZpb3IKCi8vIFVzZSB0aGUgcGlwZWxpbmUgaW4gdGhlIGFwaS4gSXQgd2lsbCBhZGQgYWxsIHRoZSByb3V0ZXMgYW5kIGNvbXB1dGUgT3BlbiBBcGkgc3BlYwphcGkudXNlKGFQaXBlbGluZSwgIm1vZGVsIik7CgovLyBTdGFydCB0aGUgc2VydmVyCmFwcC5saXN0ZW4ocHJvY2Vzcy5lbnYuUE9SVCB8fCA4MCk7CmBgYAoKV2l0aCB0aGlzIGJhc2ljIGV4YW1wbGUgeW91IG5vdyBoYXZlIHRoZSBmb2xsb3dpbmcgZW5kcG9pbnRzOgoKLSBHRVQgL2FwaS5qc29uIHdoaWNoIGNvbnRhaW5zIE9wZW4gQXBpIHNwZWMgZm9yIHRoaXMgQVBJCi0gR0VUIC9tb2RlbHMKLSBQT1NUIC9tb2RlbHMKLSBHRVQgL21vZGVscy86aWQKLSBQVVQgL21vZGVscy86aWQKLSBQQVRDSCAvbW9kZWxzLzppZAotIERFTEVURSAvbW9kZWxzLzppZAoKVGhlIGltcG9ydGFudCBwb2ludCBpcyB0aGF0IHRoZSAqKkFwaSoqIHJlYWN0IHRvIHRoZSAqKnBpcGVsaW5lKiogYmVoYXZpb3VyLiBXaGVuIHlvdSBkZWZpbmUgbmV3IGNvbnN0cmFpbnRzIG9uIHlvdXIgKipzY2hlbWEqKiBvciBuZXcgb3B0aW9ucyBpbiBhICoqcGlwZWxpbmUqKiwgdGhlICoqQXBpKiogd2lsbCByZWFjdCBhY2NvcmRpbmdseS4KCklmIHlvdSB3YW50IHRvIHNlZSBtb3JlIGNvbXBsZXggZXhhbXBsZXMsIHRha2UgYSBsb29rIGF0IHRoZSBgYGBzcmMvZXhhbXBsZWBgYCBmb2xkZXIuCgoKIyMgQ29udHJpYnV0aW5nCgpUaGUgcHJvamVjdCBpbnRlcmVzdHMgeW91ID8gUmVhZCBvdXIgW2NvbnRyaWJ1dGVyIGd1aWRlXSguL0NPTlRSSUJVVElORy5tZCkgc28geW91IGNhbiBnZXQgaW52b2x2ZWQuCgoKCgoKCg== readmeEtag: '"a63119ec4a801e2768bbe1efa68063d92b3eb7cf"' readmeLastModified: Thu, 09 May 2019 22:34:30 GMT repositoryId: 109129356 description: An API framework in Typescript/Node.js with OpenApi 3 & GraphQL created: '2017-11-01T12:35:17Z' updated: '2024-11-15T11:01:48Z' language: TypeScript archived: false stars: 17 watchers: 3 forks: 0 owner: serafin-labs logo: https://avatars.githubusercontent.com/u/33131861?v=4 license: MIT repoEtag: '"d9b04ddf97b11de385b58366e761eb7d0ec7ff8942b0c7d72eebe94a0c0e4691"' repoLastModified: Fri, 15 Nov 2024 11:01:48 GMT foundInMaster: true category: - Server - Parsers id: ab96ee28815dc64b9a94c061ff5169f1 - source: openapi3 tags repository: https://github.com/louis-lau/duckyapi v3: true repositoryMetadata: base64Readme: >- X19OT1RFOl9fIFRoaXMgcHJvamVjdCBpcyBzdGlsbCBhIHdvcmsgaW4gcHJvZ3Jlc3MuIFlvdSBjYW4gYWxyZWFkeSBjaGVjayBpdCBvdXQgaWYgeW91J3JlIGN1cmlvdXMuIFRoZSBBUEkgbWF5IGNoYW5nZSBpbiB0aGUgZnV0dXJlLgoKIyBEdWNreUFQSQoKQVBJIHRoYXQgaW50ZXJhY3RzIHdpdGggdGhlIFtXaWxkRHVja10oaHR0cHM6Ly9naXRodWIuY29tL25vZGVtYWlsZXIvd2lsZGR1Y2spIEFQSS4gTW9zdGx5IGJ1aWx0IGFzIGEgYmFja2VuZCB0byBbRHVja3lQYW5lbF0oaHR0cHM6Ly9naXRodWIuY29tL2xvdWlzLWxhdS9EdWNreVBhbmVsKS4KCgohW1dlIG5lZWQgdG8gZ28gZGVlcGVyXShkb2NzL2ltYWdlcy9kZWVwZXIuanBnKQoKIyMgV2h5PwpJbiBXaWxkRHVjayBhIHVzZXIgaXMgYSBzaW5nbGUgRW1haWwgQWNjb3VudCwgdXNpbmcgdGhlIGFwaSBhcyBhbiBlbmQtdXNlciB5b3UgY2FuIGFkZCBhZGRyZXNzIGFsaWFzZXMgdG8gdGhhdCBpbmJveC4gWW91IGNhbiBub3QgYWRkIGV4dHJhIGVtYWlsIGFjY291bnRzIG9yIG1hbmFnZSBkb21haW4gbGV2ZWwgZnVuY3Rpb25hbGl0eSBsaWtlIERLSU0uIFRoZSBhaW0gb2YgRHVja3lBUEkgaXMgdG8gb2ZmZXIgYW4gZW5kLXVzZXIgQVBJIHRoYXQgYWxsb3dzIGNvbXBsZXRlIG1hbmFnZW1lbnQgb2YgZG9tYWlucyBhbmQgZW1haWwgYWNjb3VudHMgd2l0aGluIHRob3NlIGRvbWFpbnMuCgojIyBIb3c/CkR1Y2t5QVBJIHN0b3JlcyBpdHMgdXNlcnMgaW4gTW9uZ29EQiwgdGhpcyBjYW4gYmUgdGhlIHNhbWUgaW5zdGFuY2UgdGhhdCB5b3UncmUgYWxyZWFkeSB1c2luZyBmb3IgV2lsZER1Y2suIEVhY2ggdXNlciBvd25zIGEgbGlzdCBvZiBkb21haW5zLCBncmFudGluZyBwZXJtaXNzaW9uIHRvIG1hbmFnZSBka2ltIG9yIGFkZC9lZGl0IGVtYWlsIGFjY291bnRzIGFuZCBmb3J3YXJkZXJzIHVuZGVyIHRoYXQgZG9tYWluLiBFYWNoIHVzZXIgY2FuIGJlIGFzc2lnbmVkIGEgcGFja2FnZSwgY29udGFpbmluZyBxdW90YXMgYW5kIGxpbWl0cyBmb3IgdGhlIHVzZXIuIEN1cnJlbnRseSBub3RoaW5nIGhhcHBlbnMgd2hlbiBxdW90YSBpcyBleGNlZWRlZCwgdGhpcyBtYXkgY2hhbmdlIGluIHRoZSBmdXR1cmUuCgpMaWtlIFdpbGREdWNrLCB0aGlzIGFwcGxpY2F0aW9uIGRvZXMgbm90IGRlcGVuZCBvbiBtZW1vcnkgZm9yIGFueXRoaW5nLiBVc2VycyBldGMgYXJlIHN0b3JlZCBpbiBtb25nb2RiLCBxdWV1ZSBtYW5hZ2VtZW50IGlzIGRvbmUgaW4gcmVkaXMuIFRoaXMgYXBwbGljYXRpb24gaXMgc3RhdGVsZXNzLgoKIyMgRmVhdHVyZXMKU2VlIFtEdWNreVBhbmVsIGZlYXR1cmVzXShodHRwczovL2dpdGh1Yi5jb20vbG91aXMtbGF1L0R1Y2t5UGFuZWwvYmxvYi9tYXN0ZXIvUkVBRE1FLm1kI2N1cnJlbnQtZmVhdHVyZXMpLgoKIyMgRGVwZW5kZW5jaWVzCiogTm9kZS5qcwoqIE1vbmdvREIKKiBSZWRpcwoqIFdpbGREdWNrCgojIyBJbnN0YWxsYXRpb24KYGBgYmFzaAokIGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vbG91aXMtbGF1L0R1Y2t5QVBJLmdpdAokIGNkIER1Y2t5QVBJCiQgbnBtIGluc3RhbGwKYGBgCgojIyBDb25maWd1cmF0aW9uCkNvcHkgYGNvbmZpZy9leGFtcGxlLmVudmAgdG8gYGNvbmZpZy9wcm9kdWN0aW9uLmVudmAgb3IgYGNvbmZpZy9kZXZlbG9wbWVudC5lbnZgIGRlcGVuZGluZyBvbiB5b3VyIGVudmlyb25tZW50LiBZb3UgbXVzdCBjaGFuZ2UgdGhlIGNvbmZpZ3VyYXRpb24gZm9yIHRoZSBhcHBsaWNhdGlvbiB0byBzdGFydC4gSWYgeW91J3ZlIG1pc2NvbmZpZ3VyZWQgc29tZXRoaW5nIHRoZSBhcHBsaWNhdGlvbiBzaG91bGQgdGVsbCB5b3Ugb24gc3RhcnQuCgojIyBVc2FnZQpgYGBiYXNoCiQgbnBtIHJ1biBjbGVhbgokIG5wbSBydW4gYnVpbGQKJCBucG0gc3RhcnQKCiMgQ3JlYXRlIHlvdXIgZmlyc3QgYWRtaW4gdXNlciwgYWRtaW4gdXNlcnMgYXJlCiMgb25seSBtZWFudCBmb3IgYWRkaW5nIGFuZCB1cGRhdGluZyB1c2Vycy9wYWNrYWdlcwokIG5vZGUgZGlzdC9jbGkgY3JlYXRlLWFkbWluIDx1c2VybmFtZT4KIyBDcmVhdGUgYW4gYXBpIGtleSBmb3IgeW91ciBhZG1pbiB1c2VyCiQgbm9kZSBkaXN0L2NsaSBjcmVhdGUtYXBpa2V5IDx1c2VybmFtZT4gPGtleU5hbWU+CgojIEFkZCBhIG5vcm1hbCB1c2VyIHVzaW5nIHRoZSBhcGksIGJlIHN1cmUgdG8gcmVwbGFjZQojIHRoZSBhY2Nlc3MgdG9rZW4gd2l0aCB0aGUgb25lIHlvdSBqdXN0IGdvdCBmcm9tIGNyZWF0ZS1hcGlrZXkKY3VybCAtWCBQT1NUICJodHRwOi8vbG9jYWxob3N0OjMwMDAvdXNlcnMiIFwKLUggIkF1dGhvcml6YXRpb246IEJlYXJlciBZT1VSLUFDQ0VTUy1UT0tFTi1IRVJFIiBcCi1IICJDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24iIFwKLWQgJ3sidXNlcm5hbWUiOiJqb2huZG9lIiwgInBhc3N3b3JkIjoic3VwZXJzZWNyZXQifScKCiMgTm93IHVzZSB0aGUgbm9ybWFsIHVzZXIgdG8gbG9nIGluIHRvIER1Y2t5UGFuZWwsCiMgb3IgcmVxdWVzdCBhbiBhY2Nlc3MgdG9rZW4gZnJvbSB0aGUgL2F1dGhlbnRpY2F0aW9uIGVuZHBvaW50CmBgYArwn5GGIEluc3RlYWQgb2YgdXNpbmcgY3VybCB5b3UgY2FuIGFsc28gZXhlY3V0ZSB0aGlzIHJlcXVlc3QgZnJvbSBbbG9jYWxob3N0OjMwMDAvc3dhZ2dlcl0oaHR0cDovL2xvY2FsaG9zdDozMDAwL3N3YWdnZXIpCgojIyBBUEkgZG9jdW1lbnRhdGlvbgpBUEkgZG9jdW1lbnRhdGlvbiB3aXRoIGNvZGUgZXhhbXBsZXMgaXMgYXZhaWxhYmxlIG9uIFtsb3Vpcy1sYXUuZ2l0aHViLmlvL0R1Y2t5QVBJXShodHRwczovL2xvdWlzLWxhdS5naXRodWIuaW8vRHVja3lBUEkpLgoKWW91IGNhbiBhbHNvIHZpc2l0IFtsb2NhbGhvc3Q6MzAwMC9zd2FnZ2VyXShodHRwOi8vbG9jYWxob3N0OjMwMDAvc3dhZ2dlcikgdG8gdHJ5IHRoZSBhcGkgb3V0IGxpdmUgaW4geW91ciBicm93c2VyLiBNdWNoIG5pY2UgdGhhbiB1c2luZyBjdXJsIQoKIyMgSW50ZWdyYXRlZCBEdWNreVBhbmVsCkR1Y2t5QXBpIGNhbiBzZXJ2ZSBEdWNreVBhbmVsIG9uIGl0cyBpbnRlZ3JhdGVkIHNlcnZlci4gSnVzdCBvcGVuIHlvdXIgY29uZmlndXJhdGlvbiBhbmQgc2V0IGBTRVJWRV9EVUNLWVBBTkVMYCB0byBgdHJ1ZWAuIFRoZW4gc2V0IGEgY3VzdG9tIGBCQVNFX1VSTGAgZm9yIHRoZSBhcGksIGZvciBleGFtcGxlIGAvYXBpYC4KCkR1Y2t5cGFuZWwgd2lsbCBub3cgYmUgbGl2ZSBhdCBbbG9jYWxob3N0OjMwMDBdKGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMCksIGFuZCBEdWNreUFwaSBhdCBbbG9jYWxob3N0OjMwMDAvYXBpXShodHRwOi8vbG9jYWxob3N0OjMwMDAvYXBpKS4KCiMjIFRhc2sgcXVldWUKQW55IGNyZWF0ZWQgYmFja2dyb3VuZCB0YXNrcyBhbmQgdGhlaXIgcHJvZ3Jlc3MgY2FuIGJlIHZpZXdlZCBvbiBbbG9jYWxob3N0OjMwMDAvcXVldWVzXShodHRwOi8vbG9jYWxob3N0OjMwMDAvcXVldWVzKSB3aXRoIGJhc2ljYXV0aCBpZiB5b3UndmUgZW5hYmxlZCB0aGlzIGluIHRoZSBjb25maWd1cmF0aW9uLiBSZW1vdmluZyBhIGRvbWFpbiBvciBzdXNwZW5kaW5nIGEgdXNlciB3aWxsIHRyaWdnZXIgYSBiYWNrZ3JvdW5kIHRhc2sgdG8gZXhlY3V0ZSBtYXNzIGNoYW5nZXMu readmeEtag: '"1910d5ed1d3bd579279e36aa22d64567a8a46a4a"' readmeLastModified: Fri, 30 Apr 2021 14:53:40 GMT repositoryId: 197958401 description: Domain admin API on top of the WildDuck API. Back-end for DuckyPanel. created: '2019-07-20T16:55:26Z' updated: '2024-12-03T11:53:02Z' language: TypeScript archived: false stars: 17 watchers: 3 forks: 15 owner: louis-lau logo: https://avatars.githubusercontent.com/u/1346804?v=4 license: AGPL-3.0 repoEtag: '"cadcf6fd47126c040af260b89cfe32d220d8d9c3df0d5f1e48eafaae71ae819c"' repoLastModified: Tue, 03 Dec 2024 11:53:02 GMT foundInMaster: true category: Server Implementations id: d8f0bcb887ddf1e1f6c1bcb9d08314e5 - source: openapi3 tags repository: https://github.com/ben-jamin-chen/springboot-kafka-streams-rest-api v3: true id: 0c2d44cef1118bbebe90119d0b6e9879 repositoryMetadata: base64Readme: >- IyBTcHJpbmcgQm9vdCAoMi4zLjMpIFJFU1RmdWwgQVBJIHdpdGggS2Fma2EgU3RyZWFtcyAoMi42LjApCgpXaGlsZSBsb29raW5nIHRocm91Z2ggdGhlIEthZmthIFR1dG9yaWFscyB0byBzZWUgaG93IEkgY291bGQgc2V0dXAgYSBTcHJpbmcgQm9vdCBBUEkgcHJvamVjdCB3aXRoIEthZmthIFN0cmVhbXMsIEkgZm91bmQgaXQgc3RyYW5nZSB0aGF0IHRoZXJlIHdhc24ndCBhIGNvbXBsZXRlIG9yIG1vcmUgaW5mb3JtYXRpdmUgZXhhbXBsZSBvbiBob3cgdGhpcyBjb3VsZCBiZSBhY2hpZXZlZC4gTW9zdCB1c2UgY2FzZXMgZGVtb25zdHJhdGVkIGhvdyB0byBjb21wdXRlIGFnZ3JlZ2F0aW9ucyBhbmQgaG93IHRvIGJ1aWxkIHNpbXBsZSB0b3BvbG9naWVzLCBidXQgaXQgd2FzIGRpZmZpY3VsdCB0byBmaW5kIGEgY29uY3JldGUgZXhhbXBsZSBvbiBob3cgdG8gYnVpbGQgYW4gQVBJIHNlcnZpY2UgdGhhdCBjb3VsZCBxdWVyeSBpbnRvIHRoZXNlIG1hdGVyaWFsaXplZCBuYW1lIHN0b3Jlcy4gQW55d2F5cywgSSB0aG91Z2h0IEnigJlkIGNyZWF0ZSBteSBvd24gdXNpbmcgYSBtb3JlIHJlY2VudCB2ZXJzaW9uIG9mIFNwcmluZyBCb290IHdpdGggSmF2YSAxNC4KCiMjIFdoYXQgWW91IE5lZWQKCiogSmF2YSAxNAoqIE1hdmVuIDMuNi4wKwoqIERvY2tlciAxOSsKCiMjIEdldHRpbmcgU3RhcnRlZApXZSBuZWVkIHRvIGZpcnN0IGxhdW5jaCB0aGUgQ29uZmx1ZW50IHNlcnZpY2VzIChpLmUuIFNjaGVtYSBSZWdpc3RyeSwgQnJva2VyLCBab29LZWVwZXIpIGxvY2FsbHkgYnkgcnVubmluZyB0aGUgYGRvY2tlci1jb21wb3NlIHVwIC1kYCBDTEkgY29tbWFuZCB3aGVyZSB0aGUgW2RvY2tlci1jb21wb3NlLnltbF0oaHR0cHM6Ly9naXRodWIuY29tL2JjaGVuMDQvc3ByaW5nYm9vdC1rYWZrYS1zdHJlYW1zLXJlc3QtYXBpL2Jsb2IvbWFzdGVyL2RvY2tlci1jb21wb3NlLnltbCkgZmlsZSBpcy4gVHlwaWNhbGx5LCB5b3UgY2FuIGNyZWF0ZSBhIHN0YWNrIGZpbGUgKGluIHRoZSBmb3JtIG9mIGEgWUFNTCBmaWxlKSB0byBkZWZpbmUgeW91ciBhcHBsaWNhdGlvbnMuIFlvdSBjYW4gYWxzbyBydW4gYGRvY2tlci1jb21wb3NlIHBzYCB0byBjaGVjayB0aGUgc3RhdHVzIG9mIHRoZSBzdGFjay4gTm90aWNlLCB0aGUgZW5kcG9pbnRzIGZyb20gd2l0aGluIHRoZSBjb250YWluZXJzIG9uIHlvdXIgaG9zdCBtYWNoaW5lLgoKfCBOYW1lIHwgRnJvbSB3aXRoaW4gY29udGFpbmVycyB8IEZyb20gaG9zdCBtYWNoaW5lIHwKfCAtLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0gfAp8IEthZmthIEJyb2tlciB8IGJyb2tlcjo5MDkyIHwgbG9jYWxob3N0OjkwOTIgfAp8IFNjaGVtYSBSZWdpc3RyeSAgfCBodHRwOi8vc2NoZW1hLXJlZ2lzdHJ5OjgwODEgfCBodHRwOi8vbG9jYWxob3N0OjgwODEgfAp8IFpvb0tlZXBlciB8IHpvb2tlZXBlcjoyMTgxIHwgbG9jYWxob3N0OjIxODEgfAoKPiBOb3RlOiB5b3UgY2FuIHJ1biBgZG9ja2VyLWNvbXBvc2UgZG93bmAgdG8gc3RvcCBhbGwgc2VydmljZXMgYW5kIGNvbnRhaW5lcnMuCgpBcyBwYXJ0IG9mIHRoaXMgc2FtcGxlLCBJJ3ZlIHJldHJvZml0dGVkIHRoZSBhdmVyYWdlIGFnZ3JlZ2F0ZSBleGFtcGxlIGZyb20gW0NvbmZsdWVudCdzIEthZmthIFR1dG9yaWFsc10oaHR0cHM6Ly9rYWZrYS10dXRvcmlhbHMuY29uZmx1ZW50LmlvL2FnZ3JlZ2F0aW5nLWF2ZXJhZ2Uva3N0cmVhbXMuaHRtbCkgaW50byB0aGlzIHByb2plY3QuIFRoZSBBUEkgd2lsbCBjYWxjdWxhdGUgYW5kIHJldHVybiBhIHJ1bm5pbmcgYXZlcmFnZSByYXRpbmcgZm9yIGEgZ2l2ZW4gbW92aWUgaWRlbnRpZmllci4gVGhpcyBzaG91bGQgZGVtb25zdHJhdGUgaG93IHRvIGJ1aWxkIGEgYmFzaWMgQVBJIHNlcnZpY2Ugb24gdG9wIG9mIGFuIGFnZ3JlZ2F0aW9uIHJlc3VsdC4KCk5vdGljZSBpbiB0aGUgYH4vc3JjL21haW4vYXZyb2AgZGlyZWN0b3J5LCB3ZSBoYXZlIGFsbCBvdXIgQXZybyBzY2hlbWEgZmlsZXMgZm9yIHRoZSBzdHJlYW0gb2YgYHJhdGluZ3NgIGFuZCBgY291bnRzdW1gLiBGb3IgeW91ciBjb252ZW5pZW5jZSwgdGhlIGNsYXNzZXMgd2VyZSBhbHJlYWR5IGdlbmVyYXRlZCB1bmRlciB0aGUgYH4vc3JjL21haW4vamF2YS9pby9jb25mbHVlbnQvZGVtb2AgZGlyZWN0b3J5LCBidXQgZmVlbCBmcmVlIHRvIHRpbmtlciB3aXRoIHRoZW0gYW5kIHJlY29tcGlsZSB0aGUgc2NoZW1hcyBpZiBuZWVkZWQuIFRoZSBBdnJvIGNsYXNzZXMgY2FuIGJlIHByb2dyYW1tYXRpY2FsbHkgZ2VuZXJhdGVkIHVzaW5nIGBNYXZlbmAgb3IgYnkgbWFudWFsbHkgaW52b2tpbmcgdGhlIFtzY2hlbWEgY29tcGlsZXJdKGh0dHBzOi8vYXZyby5hcGFjaGUub3JnL2RvY3MvMS4xMC4wL2dldHRpbmdzdGFydGVkamF2YS5odG1sI0NvbXBpbGluZyt0aGUrc2NoZW1hKS4gCgpTbyBiZWZvcmUgYnVpbGRpbmcgYW5kIHJ1bm5pbmcgdGhlIHByb2plY3QsIG9wZW4gYSBuZXcgdGVybWluYWwgYW5kIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmRzIHRvIGdlbmVyYXRlIHlvdXIgaW5wdXQgYW5kIG91dHB1dCB0b3BpY3MuCgpgYGB6c2gKJCAgZG9ja2VyLWNvbXBvc2UgZXhlYyBicm9rZXIga2Fma2EtdG9waWNzIC0tY3JlYXRlIC0tYm9vdHN0cmFwLXNlcnZlciBcCiAgIGxvY2FsaG9zdDo5MDkyIC0tcmVwbGljYXRpb24tZmFjdG9yIDEgLS1wYXJ0aXRpb25zIDEgLS10b3BpYyByYXRpbmdzCgokICBkb2NrZXItY29tcG9zZSBleGVjIGJyb2tlciBrYWZrYS10b3BpY3MgLS1jcmVhdGUgLS1ib290c3RyYXAtc2VydmVyIFwKICAgbG9jYWxob3N0OjkwOTIgLS1yZXBsaWNhdGlvbi1mYWN0b3IgMSAtLXBhcnRpdGlvbnMgMSAtLXRvcGljIHJhdGluZy1hdmVyYWdlcwpgYGAKCk5leHQsIHdlIHdpbGwgbmVlZCB0byBwcm9kdWNlIHNvbWUgZGF0YSBvbnRvIHRoZSBpbnB1dCB0b3BpYy4KCmBgYHpzaAokICBkb2NrZXIgZXhlYyAtaSBzY2hlbWEtcmVnaXN0cnkgL3Vzci9iaW4va2Fma2EtYXZyby1jb25zb2xlLXByb2R1Y2VyIC0tdG9waWMgcmF0aW5ncyAtLWJyb2tlci1saXN0IGJyb2tlcjo5MDkyXAogICAgLS1wcm9wZXJ0eSAicGFyc2Uua2V5PWZhbHNlIlwKICAgIC0tcHJvcGVydHkgImtleS5zZXBhcmF0b3I9OiJcCiAgICAtLXByb3BlcnR5IHZhbHVlLnNjaGVtYT0iJCg8IHNyYy9tYWluL2F2cm8vcmF0aW5nLmF2c2MpIgogYGBgCiAKUGFzdGUgaW4gdGhlIGZvbGxvd2luZyBganNvbmAgZGF0YSB3aGVuIHByb21wdGVkIGFuZCBiZSBzdXJlIHRvIHByZXNzIGVudGVyIHR3aWNlIHRvIGFjdHVhbGx5IHN1Ym1pdCBpdC4KCmBgYGpzb24KeyJtb3ZpZV9pZCI6MzYyLCJyYXRpbmciOjEwfQp7Im1vdmllX2lkIjozNjIsInJhdGluZyI6OH0KIGBgYAoKT3B0aW9uYWxseSwgeW91IGNhbiBhbHNvIHNlZSB0aGUgY29uc3VtZXIgcmVzdWx0cyBvbiB0aGUgb3V0cHV0IHRvcGljIGJ5IHJ1bm5pbmcgdGhpcyBjb21tYW5kIG9uIGEgbmV3IHRlcm1pbmFsIHdpbmRvdzoKCmBgYHpzaAokICBkb2NrZXIgZXhlYyAtaXQgYnJva2VyIC91c3IvYmluL2thZmthLWNvbnNvbGUtY29uc3VtZXIgLS10b3BpYyByYXRpbmctYXZlcmFnZXMgLS1ib290c3RyYXAtc2VydmVyIGJyb2tlcjo5MDkyIFwKICAgIC0tcHJvcGVydHkgInByaW50LmtleT10cnVlIlwKICAgIC0tcHJvcGVydHkgImtleS5kZXNlcmlhbGl6ZXI9b3JnLmFwYWNoZS5rYWZrYS5jb21tb24uc2VyaWFsaXphdGlvbi5Mb25nRGVzZXJpYWxpemVyIiBcCiAgICAtLXByb3BlcnR5ICJ2YWx1ZS5kZXNlcmlhbGl6ZXI9b3JnLmFwYWNoZS5rYWZrYS5jb21tb24uc2VyaWFsaXphdGlvbi5Eb3VibGVEZXNlcmlhbGl6ZXIiIFwKICAgIC0tZnJvbS1iZWdpbm5pbmcKYGBgCgojIyBCdWlsZCBhbmQgUnVuIHRoZSBTYW1wbGUKCllvdSBjYW4gaW1wb3J0IHRoZSBjb2RlIHN0cmFpZ2h0IGludG8geW91ciBwcmVmZXJyZWQgSURFIG9yIHJ1biB0aGUgc2FtcGxlIHVzaW5nIHRoZSBmb2xsb3dpbmcgY29tbWFuZCAoaW4gdGhlIHJvb3QgcHJvamVjdCBmb2xkZXIpLgoKYGBgenNoCiQgIG12biBzcHJpbmctYm9vdDpydW4KYGBgCkFmdGVyIHRoZSBhcHBsaWNhdGlvbiBydW5zLCBuYXZpZ2F0ZSB0byBbaHR0cDovL2xvY2FsaG9zdDo3MDAxL3N3YWdnZXItdWkvaW5kZXguaHRtbD9jb25maWdVcmw9L2FwaS1kb2NzL3N3YWdnZXItY29uZmlnXShodHRwOi8vbG9jYWxob3N0OjcwMDEvc3dhZ2dlci11aS9pbmRleC5odG1sP2NvbmZpZ1VybD0vYXBpLWRvY3Mvc3dhZ2dlci1jb25maWcpIGluIHlvdXIgd2ViIGJyb3dzZXIgdG8gYWNjZXNzIHRoZSBTd2FnZ2VyIFVJLiBJZiB5b3UgdXNlZCB0aGUgc2FtZSBzYW1wbGUgZGF0YSBmcm9tIGFib3ZlLCB5b3UgY2FuIGVudGVyIGAzNjJgIGFzIHRoZSBgbW92aWVJZGAgYW5kIGl0IHNob3VsZCByZXR1cm4gc29tZXRoaW5nIHNpbWlsYXIgbGlrZSB0aGlzIGJlbG93OgoKYGBganNvbgp7CiAgIm1vdmllSWQiOiAzNjIsCiAgInJhdGluZyI6IDkKfQpgYGAKCj4gTm90ZToga2VlcCBpbiBtaW5kIHRoZSB2YXJpb3VzIFtzdGF0ZXNdKGh0dHBzOi8va2Fma2EuYXBhY2hlLm9yZy8yNS9qYXZhZG9jL29yZy9hcGFjaGUva2Fma2Evc3RyZWFtcy9LYWZrYVN0cmVhbXMuU3RhdGUuaHRtbCkgb2YgdGhlIGNsaWVudC4gV2hlbiBhIEthZmthIFN0cmVhbXMgaW5zdGFuY2UgaXMgaW4gYFJVTk5JTkdgIHN0YXRlLCBpdCBhbGxvd3MgZm9yIGluc3BlY3Rpb24gb2YgdGhlIHN0cmVhbSdzIG1ldGFkYXRhIHVzaW5nIG1ldGhvZHMgbGlrZSBgcXVlcnlNZXRhZGF0YUZvcktleSgpYC4gV2hpbGUgaXQgaXMgaW4gYFJFQkFMQU5DSU5HYCBzdGF0ZSwgdGhlIFJFU1Qgc2VydmljZSBjYW5ub3QgaW1tZWRpYXRlbHkgYW5zd2VyIHJlcXVlc3RzIHVudGlsIHRoZSBzdGF0ZSBzdG9yZXMgYXJlIGZ1bGx5IHJlYnVpbHQuCgojIyBUcm91Ymxlc2hvb3RpbmcKCiogSW4gY2VydGFpbiBjb25kaXRpb25zLCB5b3UgbWF5IG5lZWQgdG8gZG8gYSBjb21wbGV0ZSBhcHBsaWNhdGlvbiByZXNldC4gWW91IGNhbiBkZWxldGUgdGhlIGFwcGxpY2F0aW9u4oCZcyBsb2NhbCBzdGF0ZSBkaXJlY3Rvcnkgd2hlcmUgdGhlIGFwcGxpY2F0aW9uIGluc3RhbmNlIHdhcyBydW4uIEluIHRoaXMgcHJvamVjdCwgS2Fma2EgU3RyZWFtcyBwZXJzaXN0cyBsb2NhbCBzdGF0ZXMgdW5kZXIgdGhlIGB+L2RhdGFgIGZvbGRlci4K readmeEtag: '"8c67934d12533f0829b86012ce9b09bb6fc938d6"' readmeLastModified: Sat, 15 Aug 2020 18:19:50 GMT repositoryId: 281567320 description: >- A sample RESTful API with Kafka Streams (2.6.0) using Spring Boot (2.3.3) and Java 14. created: '2020-07-22T03:43:45Z' updated: '2024-09-08T15:02:38Z' language: Java archived: false stars: 16 watchers: 3 forks: 9 owner: ben-jamin-chen logo: https://avatars.githubusercontent.com/u/41641688?v=4 license: MIT repoEtag: '"4da566a4962d385eb3f39c6a05026390e501984be947d5130094ea7be27c573d"' repoLastModified: Sun, 08 Sep 2024 15:02:38 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/oai/projects v3: true repositoryMetadata: base64Readme: >- # OpenAPI Initiative (OAI) Projects
This is a repo for moving forward a variety of projects within the OpenAPI community. In an effort to be more transparent and help involve the community in a variety of ways we will be managing most projects within this repository. This README should provide you with everything you need to get started working on an existing project, or suggest a new project.

## How It All Works
We are just getting started with this effort to external project work, so this process will evolve. Here are the moving parts for managing projects using this repository:

- **README** - The README is the "home page" for all the project being managed and will be kept as an overview of everything that is happening.
- **Issues** - We are using issues for the catch all basket for any comments, questions, or feedback, and projects can have multiple issues.
- **Projects** - Once an idea or suggestion for a new project is set into motion it will be given a formal project for managing all tasks.
- **Pages** - Ongoing projects may also have a dedicated markdown page that provides access to any related documents and artifacts.
- **Folder** - Projects may also be able to have their own folders within the repository for organizing any code or related artifacts.

Feel free to suggest a new project by submitting an issue, or get involved in a specific project by jumping in on the issues or the discussions for each area of work.

<!-- dynamic content -->
## Active Projects
These are the projects currently being moved forward in some capacity. Each project has a listing of to-do, in progress, and done tasks, as well as link to any open issues or discussions.

Project|Description|
|---|---|
|[**Member Showcase & Engagement**](https://github.com/OAI/Projects/projects/2)|Continually improving how we showcase and engage with the members of the OAI, and increase their participation within the OAS community.|
|[**Profile & Engage with API Providers**](https://github.com/OAI/Projects/projects/3)|Work to identify, profile, and build relationships with API providers who have implemented the OpenAPI specification, and publish profile to a central database.|
|[**Profile and Engage with Open Source Tooling**](https://github.com/OAI/Projects/projects/4)|Establish an official directory of open source tooling that uses OAS, and actively work to establish and build relationships with each tooling provider to get them more involved in the community.|
|[**Business Sector Showcase & Engagement**](https://github.com/OAI/Projects/projects/5)|Work to profile business sectors that are putting OpenAPI to work, then engage, and build relationships with individuals or organizations, while helping stimulate OAI special interest groups in these areas.|
|[**Quantify the Scope of the OpenAPI Community**](https://github.com/OAI/Projects/projects/6)|Work to establish the size and scope of the OpenAPI community and then track on the growth over time.|
|[**Strengthen Multi-Specification Relationships**](https://github.com/OAI/Projects/projects/7)|Ongoing work to help strengthen the relationships between the OAI and other API specifications like AsyncAPI, GraphQL, JSON Schema, and gRPC.|
|[**Curate and Publish API Articles, Podcasts, Videos**](https://github.com/OAI/Projects/projects/8)|Work to discover, curate, and then showcase and syndicate the existing articles, podcasts, and videos that exist about OpenAPI.|
|[**Additional OAI Leadership**](https://github.com/OAI/Projects/projects/9)|Work to define the roles for 3 additional leadership within the OAI, and help them be successful in work over a year.|
|[**JSON Schema Documentation Update**](https://github.com/OAI/Projects/projects/10)|This is work to invest in the updating of JSON Schema documentation, helping invest in supporting specifications.|
|[**OpenAPI Search Engine Optimization**](https://github.com/OAI/Projects/projects/11)|This is ongoing work to help improve the search engine optimization for the OAI and OAS, helping increase it's presence.|
|[**Workflow SIG**](https://github.com/OAI/Projects/projects/13)|This is a group to move forward the workflow and scenario conversation.|
|[**Security SIG**](https://github.com/OAI/Projects/projects/14)|This is a group to move forward the security discussion.|
|[**SLA SIG**](https://github.com/OAI/Projects/projects/15)|This is a group to move forward the SLA conversation.|
|[**Travel SIG**](https://github.com/OAI/Projects/projects/16)|This is a group to move the travel conversation forward.|
|[**Overlays**](https://github.com/OAI/Projects/projects/17)|This is to track on all the moving parts of the overlays conversation.|
|[**Healthcare SIG**](https://github.com/OAI/Projects/projects/18)|This is a project to setup a working group to focus on the healthcare industry.|
|[**Pitch Deck**](https://github.com/OAI/Projects/projects/19)|Move forward the work around a formal pitch deck to represent what OpenAPI is to individual onboarding with the spec or the OAI as an organization.|
|[**API Specifications Conference**](https://github.com/OAI/Projects/projects/20)|Keep track of some of the goings on around ASC.|
|[**Office Hours**](https://github.com/OAI/Projects/projects/22)|This is a project to offer ongoing weekly office hours where anyone can come and learn about the OAI, OAS, and how it all works.|
|[**SIG Planning**](https://github.com/OAI/Projects/projects/23)|This is a project to track on the details across the SIGs. Each SIG has own project board, but this is used to plan across the SIGs.|
|[**New Member Identification & Engagement**](https://github.com/OAI/Projects/projects/24)|Work to identify potential new members to the OAI and reach out to them to help start a conversation and make them aware of the benefits of being a member.|
|[**Business Governance Board (BGB)**](https://github.com/OAI/Projects/projects/25)|This is used to manage tasks for the BGB, helping Neal, Isabel, and Kin be more organized.|
|[**Extension Directory**](https://github.com/OAI/Projects/projects/26)|This is a project to move forward an official OAI extension registry, cataloging how the specification is being extended.|
|[**Blog Process**](https://github.com/OAI/Projects/projects/27)|How do you submit a story for inclusion on the blog.|

## Incubator Projects

Project|Description|
|---|---|
<!-- dynamic content -->

As projects are complete they will be closed, with some projects living on forever and issues and discussions being used to drive ongoing work.

## Active Participants
These are the individuals who are currently part of moving projects forward as part of the OAI, and here to answer any questions.

- [Kin Lane](https://github.com/kinlane) - Representing as co-chair of the BGB Board.
- [Chris Tsai](https://github.com/grizzicle) - Identifying & engaging new members to grow the OAI.
- [Chris Wood](https://github.com/SensibleWood) - Engaging with open source tooling and curate content.
- [Phil Sturgeon](https://github.com/philsturgeon) - Engaging with APIs, open source tooling and curate content.

If you'd like to be added please submit a Github issue stating how you'd like to help and we'll add you to the list of participants.
 readmeEtag: '"f619bc60a700df6dc6174ecd2fbb7071385653dc"' readmeLastModified: Wed, 05 Jan 2022 23:07:05 GMT repositoryId: 366089329 description: >- All of the open projects occurring within the Open API Initiative (OAI) community. created: '2021-05-10T15:25:36Z' updated: '2024-10-11T10:57:50Z' language: JavaScript archived: false stars: 16 watchers: 9 forks: 11 owner: OAI logo: https://avatars.githubusercontent.com/u/16343502?v=4 license: Apache-2.0 repoEtag: '"2ef014308f098ee88eaca0c31530de1507ef7d8ea9a9a8b1f85993d11b5443c0"' repoLastModified: Fri, 11 Oct 2024 10:57:50 GMT foundInMaster: true category: Server id: dd0109543e2834f7f902f1f438b6b874 - source: openapi3 tags repository: https://github.com/philosowaffle/vs-openapi-designer v3: true repositoryMetadata: base64Readme: >- IyB2cy1vcGVuYXBpLWRlc2lnbmVyIFtQcmV2aWV3XQoKIFZTIENvZGUgZXh0ZW5zaW9uIGZvciBwcmV2aWV3aW5nIFtPcGVuQXBpIFNjaGVtYSdzXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikgd2l0aGluIFZTIENvZGUuCgogRmluZCBpdCBpbiB0aGUgW1ZTIENvZGUgTWFya2V0UGxhY2VdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1waGlsb3Nvd2FmZmxlLm9wZW5hcGktZGVzaWduZXIpLgoKIyMgRmVhdHVyZXMKCi0gUHJldmlldyBPcGVuQXBpIHNwZWNzIGluIHNpZGUgcGFuZWwgaW4gVlMgQ29kZSBvciBpbiBCcm93c2VyCi0gU3dhZ2dlciAyLjAgU3BlYyBwcmV2aWV3aW5nIGFuZCB2YWxpZGF0aW9uCiAgLSBZQU1ML0pTT04KICAtIFNpbmdsZSBhbmQgTXVsdGktRmlsZSAoYm90aCBsb2NhbCBhbmQgcmVtb3RlIHJlZmVyZW5jZXMpCi0gT3BlbkFwaSAzLjAgU3BlYyBwcmV2aWV3aW5nIGFuZCB2YWxpZGF0aW9uCiAgLSBZQU1ML0pTT04KICAtIFNpbmdsZSBhbmQgTXVsdGktRmlsZSAoYm90aCBsb2NhbCBhbmQgcmVtb3RlIHJlZmVyZW5jZXMpCgojIyBVc2FnZQoKLSBPcGVuIHRoZSByb290IG9mIHlvdXIgc2NoZW1hCi0gYGN0cmwtc2hmdC1wYCA+IGBPcGVuQXBpIERlc2lnbmVyOiBQcmV2aWV3YAotIE90aGVyIEF2YWlsYWJsZSBDb21tYW5kcwogIC0gYE9wZW5BcGkgRGVzaWduZXI6IFByZXZpZXdgIC0gb3BlbnMgcHJldmlldyB1c2luZyB5b3VyIGRlZmF1bHQgdmlldyBwcmVmZXJlbmNlCiAgLSBgT3BlbkFwaSBEZXNpZ25lcjogUHJldmlldyBJbiBTaWRlIFBhbmVsYCAtIG9wZW4gcHJldmlldyBpbiBzaWRlIHBhbmVsCiAgLSBgT3BlbkFwaSBEZXNpZ25lcjogUHJldmlldyBJbiBCcm93c2VyYCAtIG9wZW4gcHJlaXZldyBpbiBicm93c2VyCiAgLSBgT3BlbkFwaSBEZXNpZ25lcjogQ29tcGlsZSBTY2hlbWFgIC0gQ29tcGlsZXMgYSB1bmlmaWVkIHNjaGVtYSBhbmQgZGVyZWZlcmVuY2VzIGFsbCBgJHJlZnNgIGludG8gYSBzaW5nbGUgZmlsZQoKIyMgUmVxdWlyZW1lbnRzCgotIE5vbmUKCiMjIEV4dGVuc2lvbiBTZXR0aW5ncwoKVGhpcyBleHRlbnNpb24gY29udHJpYnV0ZXMgdGhlIGZvbGxvd2luZyBzZXR0aW5nczoKCi0gYG9wZW5BcGlEZXNpZ25lci5kZWZhdWx0UG9ydGA6IGRlZmF1bHQgcG9ydCBmb3Igc2VydmluZyB0aGUgU3dhZ2dlciBVSSwgZGVmYXVsdCBgOTAwNWAKLSBgb3BlbkFwaURlc2lnbmVyLnByZXZpZXdJbkJyb3dzZXJgOiB3aGV0aGVyIHRvIG9wZW4gcHJldmlldyBpbiB0aGUgQnJvd3NlciBvciBpbiBWUyBDb2RlIHNpZGUgcGFuZWwsIGRlZmF1bHQgYGZhbHNlYAoKIyMgS25vd24gSXNzdWVzCgpTZWUga25vd24gaXNzdWVzIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vcGhpbG9zb3dhZmZsZS92cy1vcGVuYXBpLWRlc2lnbmVyL2lzc3VlcykuICBJZiB5b3VyIGlzc3VlIGlzIG5vdCBhbHJlYWR5IGxpc3RlZCB0aGVyZSBwbGVhc2UgbG9nIGEgbmV3IG9uZS4KCiMjIFJlbGVhc2UgTm90ZXMKClNlZSBbQ0hBTkdFTE9HXShodHRwczovL2dpdGh1Yi5jb20vcGhpbG9zb3dhZmZsZS92cy1vcGVuYXBpLWRlc2lnbmVyL2Jsb2IvbWFzdGVyL0NIQU5HRUxPRy5tZCkuCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKIyMgQ29udHJpYnV0aW5nCgojIyMgRGV2ZWxvcG1lbnQKCi0gY2QgZGlyLCBgbnBtIGluc3RhbGxgIHRvIGluc3RhbGwgbmVlZGVkIGRlcGVuZGVuY2llcwotIEY1IHRvIGxhdW5jaCBleHRlbnNpb24gYW5kIHZhbGlkYXRlCi0gYG5wbSBpbnN0YWxsIC1nIHZzY2VgIEZvciBwYWNrYWdpbmcgYW5kIHB1Ymxpc2hpbmcKICAgIC0gYHZzY2UgcGFja2FnZWAgLSBUbyBidWlsZCBwcmUtcmVsZWFzZSBwYWNha2FnZXMKICAgIC0gVG8gc2lkZSBsb2FkIGluIFZTIENvZGUgYGN0cmwtc2hmdC1wYCwgYEluc3RhbGwgRnJvbSBWU0lYYCwgbG9jYXRlIG91dHB1dCBmcm9tIGBwYWNrYWdlYAo= readmeEtag: '"7a32ffd2f503223348c02e00cf78ee0ea520dace"' readmeLastModified: Sun, 14 Apr 2019 20:00:46 GMT repositoryId: 142328036 description: OpenApi Designer Extension for VS Code created: '2018-07-25T16:45:17Z' updated: '2025-07-18T14:03:25Z' language: JavaScript archived: false stars: 17 watchers: 2 forks: 3 owner: philosowaffle logo: https://avatars.githubusercontent.com/u/7275041?v=4 license: GPL-3.0 repoEtag: '"486481da297745247f841f8554ee153311a80649ed102f5375f0a090b0f28277"' repoLastModified: Fri, 18 Jul 2025 14:03:25 GMT foundInMaster: true category: Documentation id: bbbe73cd6a16928706867d54bc485989 - source: - openapi3 tags - openapi31 tags name: aiopenapi3 homepage: https://github.com/commonism/aiopenapi3 language: Python source_description: >- OpenAPI 3, Python3.7+ client & validator with automatic data validation & serialization, sync or asyncio. category: - Client Implementations - Server Implementations repository: https://github.com/commonism/aiopenapi3 repositoryMetadata: base64Readme: >- IyBhaW9wZW5hcGkzCgpBIFB5dGhvbiBbT3BlbkFQSSAzIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjAuMy5tZCkgY2xpZW50IGFuZCB2YWxpZGF0b3IgZm9yIFB5dGhvbiAzLgoKWyFbVGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL2NvbW1vbmlzbS9haW9wZW5hcGkzL2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVjb3YueW1sL2JhZGdlLnN2Zz9ldmVudD1wdXNoJmJyYW5jaD1tYXN0ZXIpXShodHRwczovL2dpdGh1Yi5jb20vY29tbW9uaXNtL2Fpb3BlbmFwaTMvYWN0aW9ucz9xdWVyeT13b3JrZmxvdyUzQUNvZGVjb3YrZXZlbnQlM0FwdXNoK2JyYW5jaCUzQW1hc3RlcikKWyFbcHJlLWNvbW1pdC5jaSBzdGF0dXNdKGh0dHBzOi8vcmVzdWx0cy5wcmUtY29tbWl0LmNpL2JhZGdlL2dpdGh1Yi9jb21tb25pc20vYWlvcGVuYXBpMy9tYXN0ZXIuc3ZnKV0oaHR0cHM6Ly9yZXN1bHRzLnByZS1jb21taXQuY2kvbGF0ZXN0L2dpdGh1Yi9jb21tb25pc20vYWlvcGVuYXBpMy9tYXN0ZXIpClshW0NvdmVyYWdlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvY29tbW9uaXNtL2Fpb3BlbmFwaTMpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvY29tbW9uaXNtL2Fpb3BlbmFwaTMpClshW1N1cHBvcnRlZCBQeXRob24gdmVyc2lvbnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9weXZlcnNpb25zL2Fpb3BlbmFwaTMuc3ZnKV0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2Fpb3BlbmFwaTMpClshW0RvY3VtZW50YXRpb24gU3RhdHVzXShodHRwczovL3JlYWR0aGVkb2NzLm9yZy9wcm9qZWN0cy9haW9wZW5hcGkzL2JhZGdlLz92ZXJzaW9uPWxhdGVzdCldKGh0dHBzOi8vYWlvcGVuYXBpMy5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvP2JhZGdlPWxhdGVzdCkKCgpUaGlzIHByb2plY3QgaXMgYSBmb3JrIG9mIFtEb3J0aHUvb3BlbmFwaTNdKGh0dHBzOi8vZ2l0aHViLmNvbS9Eb3J0aHUvb3BlbmFwaTMvKS4KCiMjIEZlYXR1cmVzCiAgKiBpbXBsZW1lbnRzIOKApgogICAgKiBTd2FnZ2VyIDIuMAogICAgKiBPcGVuQVBJIDMuMC4zCiAgICAqIE9wZW5BUEkgMy4xLjAKICAqIGRlc2NyaXB0aW9uIGRvY3VtZW50IHBhcnNpbmcgdmlhIFtweWRhbnRpY10oaHR0cHM6Ly9naXRodWIuY29tL3NhbXVlbGNvbHZpbi9weWRhbnRpYykKICAgICogcmVjdXJzaXZlIHNjaGVtYXMgKEEuYSAtPiBBKQogICogcmVxdWVzdCBib2R5IG1vZGVsIGNyZWF0aW9uIHZpYSBweWRhbnRpYwogICAgKiBweWRhbnRpYyBjb21wYXRpYmxlICJmb3JtYXQiLXR5cGUgY29lcmNpb24gKGUuZy4gZGF0ZXRpbWUuaW50ZXJ2YWwpCiAgICAqIGFkZGl0aW9uYWxQcm9wZXJ0aWVzIChsaW1pdGVkIHRvIHN0cmluZy10by1hbnkgZGljdGlvbmFyaWVzIHdpdGhvdXQgcHJvcGVydGllcykKICAqIHJlc3BvbnNlIGJvZHkgJiBoZWFkZXIgcGFyc2luZyB2aWEgcHlkYW50aWMKICAqIGJsb2NraW5nIGFuZCBub25ibG9ja2luZyAoYXN5bmNpbykgaW50ZXJmYWNlIHZpYSBbaHR0cHhdKGh0dHBzOi8vd3d3LnB5dGhvbi1odHRweC5vcmcvKQogICAgKiBTT0NLUzUgdmlhIGh0dHB4X3NvY2tzCiAgKiB0ZXN0cyB3aXRoIHB5dGVzdCAmIFtmYXN0YXBpXShodHRwczovL2Zhc3RhcGkudGlhbmdvbG8uY29tLykKICAqIHByb3ZpZGluZyBhY2Nlc3MgdG8gbWV0aG9kcyBhbmQgYXJndW1lbnRzIHZpYSB0aGUgc2FkIHNtaWxleSAuXy4gaW50ZXJmYWNlCiAgKiBQbHVnaW4gSW50ZXJmYWNlL2FwaSB0byBtb2RpZnkgZGVzY3JpcHRpb24gZG9jdW1lbnRzL3JlcXVlc3RzL3Jlc3BvbnNlcyB0byBhZGFwdCB0byBub24gY29tcGxpYW50IHNlcnZpY2VzCiAgKiBZQU1MIHR5cGUgY29lcmNpb24gaGludHMgZm9yIG5vdCB3ZWxsIGZvcm1hdHRlZCBkZXNjcmlwdGlvbiBkb2N1bWVudHMKICAqIERlc2NyaXB0aW9uIERvY3VtZW50IGRlcGVuZGVuY3kgZG93bmxvYWRzICh1c2luZyB0aGUgV2ViTG9hZGVyKQogICAgKiBsb2dnaW5nCiAgICAgICogYGV4cG9ydCBBSU9QRU5BUEkzX0xPR0dJTkdfSEFORExFUlM9ZGVidWdgIHRvIGdldCAvdG1wL2Fpb3BlbmFwaTMtZGVidWcubG9nCgoKIyMgRG9jdW1lbnRhdGlvbgpbQVBJIERvY3VtZW50YXRpb25dKGh0dHBzOi8vYWlvcGVuYXBpMy5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvKQoKCiMjIFJ1bm5pbmcgVGVzdHMKClRoaXMgcHJvamVjdCBpbmNsdWRlcyBhIHRlc3Qgc3VpdGUsIHJ1biB2aWEgYGBweXRlc3RgYC4gIFRvIHJ1biB0aGUgdGVzdCBzdWl0ZSwKZW5zdXJlIHRoYXQgeW91J3ZlIGluc3RhbGxlZCB0aGUgZGVwZW5kZW5jaWVzIGFuZCB0aGVuIHJ1biBgYHB5dGVzdGBgIGluIHRoZSByb290Cm9mIHRoaXMgcHJvamVjdC4KCmBgYHNoZWxsClBZVEhPTlBBVEg9LiBweXRlc3QgLS1jb3Y9Li8gLS1jb3YtcmVwb3J0PXhtbCAuCmBgYAo= readmeEtag: '"091767230f77346d3ea0eb78ac7448ca039d6745"' readmeLastModified: Wed, 07 Aug 2024 07:24:30 GMT repositoryId: 443839740 description: 'Python OpenAPI3 client/validator w\ {a,}syncio ' created: '2022-01-02T18:23:26Z' updated: '2026-02-04T07:32:28Z' language: Python archived: false stars: 18 watchers: 2 forks: 9 owner: commonism logo: https://avatars.githubusercontent.com/u/164513?v=4 license: BSD-3-Clause repoEtag: '"3a9c8e27e0872123a93c9a93ffab6b42cc8bf98dbcab7f3bcb795d7cd18f1d2b"' repoLastModified: Wed, 04 Feb 2026 07:32:28 GMT foundInMaster: true id: f7fe65d08545a563279e1526fe797380 v3: true v3_1: true - source: openapi3 tags repository: https://github.com/dgouyette/play-api-refiner v3: true repositoryMetadata: base64Readme: >- WyFbQnVpbGQgc3RhdHVzXShodHRwczovL2FwaS50cmF2aXMtY2kuY29tL2Rnb3V5ZXR0ZS9wbGF5LWFwaS1yZWZpbmVkLnN2Zz9icmFuY2g9ZGV2ZWxvcCldKGh0dHBzOi8vdHJhdmlzLWNpLmNvbS9kZ291eWV0dGUvcGxheS1hcGktcmVmaW5lZD9icmFuY2g9ZGV2ZWxvcCkKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2Rnb3V5ZXR0ZS9wbGF5LWFwaS1yZWZpbmVkL2JyYW5jaC9kZXZlbG9wL2dyYXBoL2JhZGdlLnN2ZyldKGh0dHBzOi8vY29kZWNvdi5pby9naC9kZ291eWV0dGUvcGxheS1hcGktcmVmaW5lZCkKWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby86bGljZW5zZS1NSVQtYmx1ZS5zdmcpXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkKCgojIFBsYXkgQVBJIFJlZmluZWQKCkEgdGlueSBsaWJyYXJ5IHRvIGdlbmVyYXRlIE9wZW5BUEkgIGRvY3VtZW50YXRpb24gYW5kIGpzb24gc2NoZW1hIGJhc2VkIG9uIFJlZmluZWQncyBjYXNlIGNsYXNzZXMuCgoKIyMgU2V0dXAKCkluIG9yZGVyIHRvIHVzZSBwbGF5LWpzb24tb3JnLmRnb3V5ZXR0ZS5yZWZpbmVkIHlvdSBuZWVkIHRvIGFkZCB0aGUgZm9sbG93aW5nIGxpbmVzIHRvIHlvdXIgYGJ1aWxkLnNidGA6CgpgYGBzY2FsYQpyZXNvbHZlcnMgKz0gUmVzb2x2ZXIuYmludHJheVJlcG8oImRnb3V5ZXR0ZSIsICJtYXZlbiIpCgpsaWJyYXJ5RGVwZW5kZW5jaWVzICs9ICJvcmcuZGdvdXlldHRlIiAlJSAiYXBpLXJlZmluZXIiICUgIjEuMC4wLU0xIgpgYGAKCiMjIFVzYWdlCgojIyMgUGxheSBKc29uIFJlZmluZWQgIGVycm9ycyAKClRvIHJldHVybiBqc29uIGVycm9ycyBiYXNlZCBvbiBvcmcuZGdvdXlldHRlLnJlZmluZWQgY2xhc3NlcyB5b3UgbXVzdCBpbXBvcnQgOgoKYGBgYHNjYWxhCmltcG9ydCBvcmcub3JnLmRnb3V5ZXR0ZS5qc29uLlJlZmluZWRSdW50aW1lVmFsaWRhdG9yLl8KYGBgYAoKIyMjIEpzb24gU2NoZW1hIGdlbmVyYXRpb24KCmBgYHNjYWxhCgpjYXNlIGNsYXNzIENvbGxlY3Rpb25TdHJpbmdOb25FbXB0eShlIDogTGlzdFtTdHJpbmddIFJlZmluZWQgTm9uRW1wdHkpCkpzb25TY2hlbWEuYXNKc1ZhbHVlW0NvbGxlY3Rpb25TdHJpbmdOb25FbXB0eV0gLy8gPT4geyJlIjp7Im1pbkxlbmd0aCI6MSwidHlwZSI6ImFycmF5IiwiaXRlbXMiOnsidHlwZSI6InN0cmluZyJ9fX0KCmBgYAoKIyMjIEpzb24gU2NoZW1hIG9uIENsaWVudCBFcnJvciAKCmBgYHNjYWxhCmNhc2UgY2xhc3MgTG9naW5EVE8oZW1haWwgIDogU3RyaW5nIFJlZmluZWQgTm9uRW1wdHksIHBhc3N3b3JkIDogU3RyaW5nIFJlZmluZWQgTm9uRW1wdHkpCgoKQFNpbmdsZXRvbgpjbGFzcyBIb21lQ29udHJvbGxlciBASW5qZWN0KCkoYnA6IEJvZHlQYXJzZXJXaXRoSnNvblNjaGVtYSkgZXh0ZW5kcyBBYnN0cmFjdENvbnRyb2xsZXIoY2MpIHsKICBpbXBsaWNpdCB2YWwgbG9naW5TY2hlbWEgPSBKc29uU2NoZW1hLmFzSnNWYWx1ZVtMb2dpbkRUT10KICBpbXBsaWNpdCB2YWwgZHRvU2NoZW1hID0gSnNvblNjaGVtYS5hc0pzVmFsdWVbU2ltcGxlRFRPXQogIAogIGRlZiBsb2dpbigpOiBBY3Rpb25bTG9naW5EVE9dID0gQWN0aW9uKGJwLmpzb25SZWZpbmVkKExvZ2luRFRPLmZtdCwgbG9naW5TY2hlbWEpKSB7CiAgICBpbXBsaWNpdCByZXF1ZXN0ID0+CiAgICBPawogIH0KfQpgYGAgCgpXaXRoIGEgYmFkIGNsaWVudCByZXF1ZXN0IDogCgpgY3VybCAtWCBQT1NUIGh0dHA6Ly9sb2NhbGhvc3Q6OTAwMC9sb2dpbiAtZCAneyJlbWFpbCIgOiIiLCJwYXNzd29yZCIgOiIiIH0nYAoKWW91IHdpbGwgaGF2ZSB0aGlzIGVycm9yIAoKYGBganNvbgp7CiAgICAib2JqLmVtYWlsIjogWwogICAgICAgIHsKICAgICAgICAgICAgIm1zZyI6IFsKICAgICAgICAgICAgICAgICJQcmVkaWNhdGUgaXNFbXB0eSgpIGRpZCBub3QgZmFpbC4iCiAgICAgICAgICAgIF0sCiAgICAgICAgICAgICJhcmdzIjogW10KICAgICAgICB9CiAgICBdLAogICAgIl9zY2hlbWEiOiB7CiAgICAgICAgInBhc3N3b3JkIjogewogICAgICAgICAgICAibWluTGVuZ3RoIjogMSwKICAgICAgICAgICAgInR5cGUiOiAic3RyaW5nIgogICAgICAgIH0sCiAgICAgICAgImVtYWlsIjogewogICAgICAgICAgICAibWluTGVuZ3RoIjogMSwKICAgICAgICAgICAgInR5cGUiOiAic3RyaW5nIgogICAgICAgIH0KICAgIH0sCiAgICAib2JqLnBhc3N3b3JkIjogWwogICAgICAgIHsKICAgICAgICAgICAgIm1zZyI6IFsKICAgICAgICAgICAgICAgICJQcmVkaWNhdGUgaXNFbXB0eSgpIGRpZCBub3QgZmFpbC4iCiAgICAgICAgICAgIF0sCiAgICAgICAgICAgICJhcmdzIjogW10KICAgICAgICB9CiAgICBdCn0KYGBgCgojIyBHZW5lcmF0ZSBPcGVuQVBJIChTd2FnZ2VyKSBEb2N1bWVudGF0aW9uCgpPcGVuQVBJQ29udHJvbGxlciA6IApgYGBgc2NhbGEKZGVmIGpzb246IEFjdGlvbltVbml0XSA9IEFjdGlvbihwYXJzZS5lbXB0eSkgewogICAgaW1wbGljaXQgIHJlcSA9PgogICAgIE9rKE9wZW5BUEkuZnJvbVJvdXRlc0ZpbGUoImNvbmYvcm91dGVzIikpCiAgfQpgYGBgCgpjb25mL3JvdXRlcwpgYGAKUE9TVCAgICAgICAgL2xvZ2luICAgICAgICAgICAgICAgICAgICAgICAgICAgY29udHJvbGxlcnMuSG9tZUNvbnRyb2xsZXIubG9naW4KR0VUICAgICAgICAgL2RvYy9qc29uICAgICAgICAgICAgICAgICAgICAgICAgY29udHJvbGxlcnMuT3BlbkFQSUNvbnRyb2xsZXIuanNvbgpgYGAKCiFbT3BlbkFQSSBEb2N1bWVudGF0aW9uXShvcGVuQVBJLnBuZykKCg== readmeEtag: '"2a3a7bda5cbb93243decfc59c607390854ff1ae5"' readmeLastModified: Wed, 16 Oct 2019 07:59:29 GMT repositoryId: 206240618 description: null created: '2019-09-04T05:43:38Z' updated: '2022-07-07T15:57:50Z' language: Scala archived: false stars: 16 watchers: 3 forks: 0 owner: dgouyette logo: https://avatars.githubusercontent.com/u/274061?v=4 repoEtag: '"469629d5ffa28ca34f5eb3c3b9cfb7a1596f2080ce5cc5ac41558723ce96016d"' repoLastModified: Thu, 07 Jul 2022 15:57:50 GMT foundInMaster: true category: Data Validators id: df6e428f59101c20eaeb110a9e268e9b - source: openapi3 tags repository: https://github.com/alekshura/sourceapi v3: true repositoryMetadata: base64Readme: >- # <img src="/Compentio.Assets/Logo.png" align="left" width="50"> SourceApi

[![NuGet](http://img.shields.io/nuget/v/Compentio.SourceApi.svg)](https://www.nuget.org/packages/Compentio.SourceApi)
![Nuget](https://img.shields.io/nuget/dt/Compentio.SourceApi)
![GitHub](https://img.shields.io/github/license/alekshura/SourceApi)
![GitHub top language](https://img.shields.io/github/languages/top/alekshura/SourceApi)

# Introduction
Two different approaches for Web API implementation often used by development teams: `code first` and `API first`.
During the `code first` approach Web API Controllers are implemented, Swagger/Open API libraries are added to the application and finnaly it deployed.
Open API libraries actually adds a middlewares to application which serve Open API definitions, 
Swagger UI thus the clients can use this definitions to generate code or use test API using UI.    

In second, `API First` approach, API defined in `json` or `yaml` files using [Open API standard](https://swagger.io/specification/) first and after that
server or client code, interfaces or DTO's are generated in application.

This approach is technology agnostic: API can be disigned independently from technologies used in cloud native architecture with a wide tech stack, then 
shared between defferent teams (`Java`, `.NET`, `NodeJS`, `Frontend`, ect.) and ech team can decide about what has to be done with it: to generate only DTO's, or 
create base abstract controllers, with routes, documentation and DTO's or to generate client code for consume the API.

`SourceApi` is a code generator that helps to use `API First` approach for .NET Core 3+ (also .NET 5+) Web API applications:
during design time in Visual Studio IDE it generates abstract base Controllers classes and DTO's that can be used by developer to implement the target functionality.

It is based on [Source Generators Additional File Transaformation](https://github.com/dotnet/roslyn/blob/main/docs/features/source-generators.cookbook.md#additional-file-transformation) feature
where it is possible to be able to transform an external non-C# file into an equivalent C# representation.

Rico Suter's [NSwag](https://github.com/RicoSuter/NSwag) is used underhood:
- [CSharpControllerGenerator](https://github.com/RicoSuter/NSwag/wiki/CSharpControllerGenerator) is used for abstract base controllers code generation
- [CSharpClientGenerator](https://github.com/RicoSuter/NSwag/wiki/CSharpClientGenerator) used for DTO's only code generation mode. 

# Installation
Install using nuget package manager:

```console
Install-Package Compentio.SourceApi
```

or `.NET CLI`:

```console
dotnet add package Compentio.SourceApi
```

# How to use
In basic and most simple scenario: add Open API definition file or files (`*.json` and `*.yaml` formats are supported) to you project as `AdditionalFiles`:

>```xml
><ItemGroup>
>   <AdditionalFiles Include="OpenApi/Pets.json">
>     <CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
>   </AdditionalFiles>
>   <AdditionalFiles Include="OpenApi/Users.yaml">
>     <CopyToOutputDirectory>PreserveNewest</CopyToOutputDirectory>
>   </AdditionalFiles>
> </ItemGroup> 

or in file properties in Visual Studio:

<img src="/Compentio.Assets/AdditionalFile.png" align="center" width="95%">

You will see generated abstract controllers in 
> Dependencies -> Analyzers -> Compentio.SourceApi > Compentio.SourceApi.Generator.

<p align="center">
  <img src="/Compentio.Assets/GeneratedFiles.png" align="center" width="50%">
</p>

>! For now use one Open API file per controller. [Open API Tags](https://swagger.io/docs/specification/grouping-operations-with-tags/) are not supported.

Now you can use these base controllers during implementation of you Web API. The DTO's are generated, routes are defined, documentation for the API methods also used from 
Open API definitions:

```cs
[ApiController]
[ApiConventionType(typeof(DefaultApiConventions))]
public class StoreController : StoreControllerBase
{
	/// <inheritdoc />
	public override Task<IActionResult> DeleteOrder([BindRequired] long orderId)
	{
		throw new NotImplementedException();
	}

	/// <inheritdoc />
	public override Task<ActionResult<IDictionary<string, int>>> GetInventory()
	{
		throw new NotImplementedException();
	}

	/// <inheritdoc />
	public override Task<ActionResult<Order>> GetOrderById([BindRequired] long orderId)
	{
		throw new NotImplementedException();
	}

	/// <inheritdoc />
	public override Task<ActionResult<Order>> PlaceOrder([BindRequired, FromBody] Order body)
	{
		throw new NotImplementedException();
	}
}
```

>! You need to add `<inheritdoc />` tag to show documentation from base class in Swagger.

And example of generated code for base controller class:

```cs

//----------------------
// <auto-generated>
//     Generated using the NSwag toolchain v13.13.2.0 (NJsonSchema v10.5.2.0 (Newtonsoft.Json v12.0.0.2)) (http://NSwag.org)
// </auto-generated>
//----------------------
#pragma warning disable 108 // Disable "CS0108 '{derivedDto}.ToJson()' hides inherited member '{dtoBase}.ToJson()'. Use the new keyword if hiding was intended."

#pragma warning disable 114 // Disable "CS0114 '{derivedDto}.RaisePropertyChanged(String)' hides inherited member 'dtoBase.RaisePropertyChanged(String)'. To make the current member override that implementation, add the override keyword. Otherwise add the new keyword."

#pragma warning disable 472 // Disable "CS0472 The result of the expression is always 'false' since a value of type 'Int32' is never equal to 'null' of type 'Int32?'

#pragma warning disable 1573 // Disable "CS1573 Parameter '...' has no matching param tag in the XML comment for ...

#pragma warning disable 1591 // Disable "CS1591 Missing XML comment for publicly visible type or member ..."

#pragma warning disable 8073 // Disable "CS8073 The result of the expression is always 'false' since a value of type 'T' is never equal to 'null' of type 'T?'"

#pragma warning disable 3016 // Disable "CS3016 Arrays as attribute arguments is not CLS-compliant"

namespace Compentio.SourceApi.WebExample.Controllers
{
    using System = global::System;

    [System.CodeDom.Compiler.GeneratedCode("NSwag", "13.13.2.0 (NJsonSchema v10.5.2.0 (Newtonsoft.Json v12.0.0.2))")]
    [Microsoft.AspNetCore.Mvc.Route("api/v1")]
    public abstract class StoreControllerBase : Microsoft.AspNetCore.Mvc.ControllerBase
    {
        /// <summary>Returns pet inventories by status</summary>
        /// <returns>successful operation</returns>
        [Microsoft.AspNetCore.Mvc.HttpGet, Microsoft.AspNetCore.Mvc.Route("store/inventory")]
        public abstract System.Threading.Tasks.Task<Microsoft.AspNetCore.Mvc.ActionResult<System.Collections.Generic.IDictionary<string, int>>> GetInventory();
        /// <summary>Place an order for a pet</summary>
        /// <param name = "body">order placed for purchasing the pet</param>
        /// <returns>successful operation</returns>
        [Microsoft.AspNetCore.Mvc.HttpPost, Microsoft.AspNetCore.Mvc.Route("store/order")]
        public abstract System.Threading.Tasks.Task<Microsoft.AspNetCore.Mvc.ActionResult<Order>> PlaceOrder([Microsoft.AspNetCore.Mvc.FromBody][Microsoft.AspNetCore.Mvc.ModelBinding.BindRequired] Order body);
        /// <summary>Find purchase order by ID</summary>
        /// <param name = "orderId">ID of pet that needs to be fetched</param>
        /// <returns>successful operation</returns>
        [Microsoft.AspNetCore.Mvc.HttpGet, Microsoft.AspNetCore.Mvc.Route("store/order/{orderId}")]
        public abstract System.Threading.Tasks.Task<Microsoft.AspNetCore.Mvc.ActionResult<Order>> GetOrderById([Microsoft.AspNetCore.Mvc.ModelBinding.BindRequired] long orderId);
        /// <summary>Delete purchase order by ID</summary>
        /// <param name = "orderId">ID of the order that needs to be deleted</param>
        [Microsoft.AspNetCore.Mvc.HttpDelete, Microsoft.AspNetCore.Mvc.Route("store/order/{orderId}")]
        public abstract System.Threading.Tasks.Task<Microsoft.AspNetCore.Mvc.IActionResult> DeleteOrder([Microsoft.AspNetCore.Mvc.ModelBinding.BindRequired] long orderId);
    }

    [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "10.5.2.0 (Newtonsoft.Json v12.0.0.2)")]
    public partial class Order
    {
        [Newtonsoft.Json.JsonProperty("id", Required = Newtonsoft.Json.Required.DisallowNull, NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore)]
        public long Id { get; set; }

        [Newtonsoft.Json.JsonProperty("petId", Required = Newtonsoft.Json.Required.DisallowNull, NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore)]
        public long PetId { get; set; }

        [Newtonsoft.Json.JsonProperty("quantity", Required = Newtonsoft.Json.Required.DisallowNull, NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore)]
        public int Quantity { get; set; }

        [Newtonsoft.Json.JsonProperty("shipDate", Required = Newtonsoft.Json.Required.DisallowNull, NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore)]
        public System.DateTimeOffset ShipDate { get; set; }

        /// <summary>Order Status</summary>
        [Newtonsoft.Json.JsonProperty("status", Required = Newtonsoft.Json.Required.DisallowNull, NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore)]
        [Newtonsoft.Json.JsonConverter(typeof(Newtonsoft.Json.Converters.StringEnumConverter))]
        public OrderStatus Status { get; set; }

        [Newtonsoft.Json.JsonProperty("complete", Required = Newtonsoft.Json.Required.DisallowNull, NullValueHandling = Newtonsoft.Json.NullValueHandling.Ignore)]
        public bool Complete { get; set; } = false;
    }

    [System.CodeDom.Compiler.GeneratedCode("NJsonSchema", "10.5.2.0 (Newtonsoft.Json v12.0.0.2)")]
    public enum OrderStatus
    {
        [System.Runtime.Serialization.EnumMember(Value = @"placed")]
        Placed = 0,
        [System.Runtime.Serialization.EnumMember(Value = @"approved")]
        Approved = 1,
        [System.Runtime.Serialization.EnumMember(Value = @"delivered")]
        Delivered = 2,
    }
}
#pragma warning restore 1591
#pragma warning restore 1573
#pragma warning restore 472
#pragma warning restore 114
#pragma warning restore 108
#pragma warning restore 3016

```
> The default namespace here is concatenation of you project name and directory of Open API definitions.
> To change namespace see `Configuration`   

# Configuration

To customize the generated code and override defaults `SourceApi` 
[consumes MSBuild properties and metadata](https://github.com/dotnet/roslyn/blob/main/docs/features/source-generators.cookbook.md#consume-msbuild-properties-and-metadata).

Two configuration properies you can dafine in `*.cproj` file to customize SourceApi generator:
- SourceApi_GeneratorNamespace - you can define namespace for generated classes
- SourceApi_GenerateOnlyContracts - you can generate only DTO's without Controller base classes.

>In a case if properties are not added, default values are used: `SourceApi_GenerateOnlyContracts` is set to `False` and for `SourceApi_GeneratorNamespace` 
>concatenation of you project name and directory of Open API definitions is used

For global configuration (used for all added Open API files) define these parameters in `<PropertyGroup> section:

```xml
<PropertyGroup>
   <TargetFramework>net5.0</TargetFramework>
   <GenerateDocumentationFile>true</GenerateDocumentationFile>
   <NoWarn>$(NoWarn);1591</NoWarn>
   ...
   <SourceApi_GeneratorNamespace>Compentio.SourceApi.WebExample.Controllers</SourceApi_GeneratorNamespace>
   <SourceApi_GenerateOnlyContracts>false</SourceApi_GenerateOnlyContracts>
</PropertyGroup>
 ```


You can also define these parameters per file in `<ItemGroup>`: 

```xml
<ItemGroup>
   <AdditionalFiles Include="OpenApi\Pets.yaml"/>
   <AdditionalFiles Include="OpenApi\Store.yaml" SourceApi_GeneratorNamespace="Compentio.SourceApi.WebExample.WebApi"/>
   <AdditionalFiles Include="OpenApi\Users.yaml" SourceApi_GeneratorNamespace="Compentio.SourceApi.WebExample.WebApi" SourceApi_GenerateOnlyContracts = "true"/>
 </ItemGroup>
 <Import Project="..\Compentio.SourceApi\Generator.props" />
```

Here in a case global configuration exists, for `Pets.yaml` global config is used, for `Store.yaml` namespace is overriden and for `Users.yaml` 
`SourceApi_GeneratorNamespace` and `SourceApi_GenerateOnlyContracts` are overriden.

 readmeEtag: '"ded6d51ffe7f2a935d0d442a4c1eae2ff8f63b20"' readmeLastModified: Thu, 02 Dec 2021 10:31:06 GMT repositoryId: 420145950 description: 'API first Open API code generator based on json or yaml definitions. ' created: '2021-10-22T15:16:23Z' updated: '2025-10-08T00:55:40Z' language: C# archived: false stars: 15 watchers: 2 forks: 0 owner: alekshura logo: https://avatars.githubusercontent.com/u/4995545?v=4 license: MIT repoEtag: '"4901f8d71fb048a0f052e59a264c8a479a08303aea30b7af1e589a88d3b46abd"' repoLastModified: Wed, 08 Oct 2025 00:55:40 GMT foundInMaster: true category: SDK id: 482e24e836e1e54d828cc864a71c9b8e - source: openapi3 tags repository: https://github.com/briankinney/airtable-swagger v3: true repositoryMetadata: base64Readme: >- YWlydGFibGUtc3dhZ2dlcgo9PT0KCkdlbmVyYXRlIE9BUyAzLjAgQVBJIGRlZmluaXRpb24gZnJvbSBBaXJ0YWJsZSBzY2hlbWEgb2JqZWN0CgpCYWNrZ3JvdW5kCi0tLQoKRXZlcnkgQWlydGFibGUgYmFzZSBoYXMgYW4gYXV0by1nZW5lcmF0ZWQgZG9jdW1lbnRhdGlvbiBwYWdlIHRoYXQgaXMgbGFyZ2VseSBkcml2ZW4gYnkgYW4gb2JqZWN0IHRoYXQgZGVzY3JpYmVzIHRoZQpzY2hlbWEgb2YgdGhlIGJhc2UuIFRoaXMgdXRpbGl0eSB3aWxsIGNvbnZlcnQgdGhlIHNjaGVtYSBvYmplY3QgaW50byBhbiBPQVMgMy4wIChzd2FnZ2VyKSBkb2N1bWVudCBzbyB5b3UgY2FuIHRha2UKYWR2YW50YWdlIG9mIHRvb2xzIHRoYXQgbWFrZSB1c2Ugb2YgdGhpcyBBUEkgZGVmaW5pdGlvbiBmb3JtYXQuCgoKRmlyZWZveCBBZGRvbgotLS0KCiMjIyBTZXR1cAoKW0Rvd25sb2FkIGxpbmtdKGh0dHBzOi8vYWRkb25zLm1vemlsbGEub3JnL2VuLVVTL2ZpcmVmb3gvYWRkb24vYWlydGFibGUtc3dhZ2dlci1nZW5lcmF0b3IvKQoKIyMjIFVzYWdlCgoxLiBPcGVuIHlvdXIgQWlydGFibGUgZG9jdW1lbnRhdGlvbiBwYWdlCjIuIENsaWNrIHRoaXMgaWNvbiBpbiB5b3VyIFVSTCBiYXIgdG8gb3BlbiB0aGUgc2lkZSBwYW5lbAoKICAgICFbVXJsIGJhciBpY29uXShwbHVnaW4vaWNvbnMvaWNvbnM4LXVzZXItbWFudWFsLTQ4LnBuZykKMy4gQ2xpY2sgdGhlIGJ1dHRvbi4gQW4gT0FTIGpzb24gYmxvYiBzaG91bGQgYXBwZWFyIGluIHRoZSB0ZXh0IGFyZWEgYWJvdmUKCkNvbW1hbmQgTGluZSBUb29sCi0tLQoKIyMjIFNldHVwCgpJbnN0YWxsIGZyb20gc291cmNlIHVzaW5nIG5wbQpgYGBiYXNoCnlvdXItY29tcHV0ZXI6L3BhdGgvdG8vYWlydGFibGUtc3dhZ2dlciQgbnBtIGkgLWcgLgpgYGAKCiMjIyBVc2FnZQoKMS4gVmlldyB0aGUgc291cmNlIG9mIHRoZSBBaXJ0YWJsZSBkb2N1bWVudGF0aW9uIHBhZ2UgZ2VuZXJhdGVkIGZvciB5b3VyIGJhc2UKMi4gRmluZCB0aGUgbGluZSBjb250YWluaW5nIGB3aW5kb3cubG9jYXRpb24gPSB7Li4ufWAKMy4gQ29weSB0aGUgdmFsdWUgb2YgYHdpbmRvdy5sb2NhdGlvbmAgdG8gYSBmaWxlCjQuIFJ1biB0aGUgdG9vbDoKICAgIGBgYGJhc2gKICAgIGFpcnRhYmxlLXN3YWdnZXIgL3BhdGgvdG8vYWlydGFibGUtc2NoZW1hLmpzb24KICAgIGBgYAo1LiBVc2UgdGhlIG91dHB1dCB3aXRoIHlvdXIgZmF2b3JpdGUgc3dhZ2dlciB0b29s readmeEtag: '"6cc029e73b65b39ce49fe4b13665497a3f8525d0"' readmeLastModified: Tue, 07 Nov 2023 23:23:33 GMT repositoryId: 183827168 description: Build OAS 3.0 API Definitions from Airtable Schema Object created: '2019-04-27T22:26:25Z' updated: '2025-03-31T20:22:56Z' language: JavaScript archived: false stars: 16 watchers: 2 forks: 3 owner: briankinney logo: https://avatars.githubusercontent.com/u/6935747?v=4 license: ISC repoEtag: '"73da60f24437e201bd56a83297c2103979e21553a5cde68370d3643a63ca8866"' repoLastModified: Mon, 31 Mar 2025 20:22:56 GMT foundInMaster: true category: Parsers id: 5f05bc19eecf433b1626fbc38eeea108 - source: openapi3 tags repository: https://github.com/greenstand/treetracker-query-api v3: true repositoryMetadata: base64Readme: >- IyBUcmVldHJhY2tlciBRdWVyeSBBUEkKClRoaXMgQVBJIGV4cG9zZXMgYSBSRVNUZnVsIGludGVyZmFjZSB0byBxdWVyeSB0aGUgdHJlZXRyYWNrZXIgZGF0YSwgY2FwdHVyZSwgcGxhbnRlciBhbmQgb3JnYW5pemF0aW9uIGFuZCBvdGhlcnMuCgpPbmUgb2YgcXVlcnkgYXBpJ3MgY2xpZW50IGlzIG91ciB3ZWIgbWFwIGFwcDogaHR0cHM6Ly9naXRodWIuY29tL0dyZWVuc3RhbmQvdHJlZXRyYWNrZXItd2ViLW1hcC1jbGllbnQvCgojIERldmVsb3BtZW50IHRvb2xraXQKClRoaXMgcmVwb3NpdG9yeSB3YXMgY3JlYXRlZCBmcm9tIEdyZWVuc3RhbmQncyB0ZW1wbGF0ZSBmb3IgbWljcm9zZXJ2aWNlIHByb2plY3RzLiBUaGlzIG1lYW5zIGl0IGNvbWVzIHdpdGggbWFueSBkZXZlbG9wbWVudCB0b29scyB0aGF0IHdlIHVzZSBmb3IgZGV2ZWxvcG1lbnQgYW5kIGRlcGxveW1lbnQuIEFzIGEgY29udHJpYnV0b3IgdG8gdGhpcyByZXBvc2l0b3J5LCB5b3Ugc2hvdWxkIGxlYXJuIGFuZCB1c2UgdGhlc2UgdG9vbHMuIFRoZXkgYXJlIG91dGxpbmVkIGJlbG93LgoKLSBDb252ZW50aW9uYWwgQ29tbWl0cwotIGh1c2t5Ci0gcHJldHRpZXIgLyBsaW50Ci0gZ2l0aHViIGFjdGlvbnMKLSBKZXN0Ci0gVHlwZVNjcmlwdAoKIyBHZXR0aW5nIFN0YXJ0ZWQKCiMjIFByb2plY3QgU2V0dXAKCk9wZW4gdGVybWluYWwgYW5kIG5hdmlnYXRlIHRvIGEgZm9sZGVyIHRvIGluc3RhbGwgdGhpcyBwcm9qZWN0OgoKYGBgCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vR3JlZW5zdGFuZC90cmVldHJhY2tlci1yZXBvc2l0b3J5LW5hbWUuZ2l0CgpgYGAKCkluc3RhbGwgYWxsIG5lY2Vzc2FyeSBkZXBlbmRlbmNpZXM6CgpgYGAKbnBtIGNpCmBgYAoKUnVuIHRoZSBzZXJ2ZXIgd2l0aCBkYXRhYmFzZSBzZXR0aW5nczoKCmBgYApEQVRBQkFTRV9VUkw9Wy4uLl0gbnBtIHJ1biBkZXYKYGBgCgpQbGVhc2Ugam9pbiBvdXIgc2xhY2sgY2hhbm5lbCB0byBnZXQgaGVscCB3aXRoIHNldHRpbmcgdXAgdGhlIGRhdGFiYXNlLgoKIyBXb3JrZmxvdyB3aXRoIEdpdGh1YgoKW2NoZWNrIG91dCBoZXJlXShodHRwczovL2dpdGh1Yi5jb20vR3JlZW5zdGFuZC90cmVldHJhY2tlci13ZWItbWFwLWNsaWVudCN3b3JrZmxvdy13aXRoLWdpdGh1YikKCiMgRGV2ZWxvcG1lbnQgU3BlY2lmaWNhdGlvbgoKLSBFdmVyeSBlbmRwb2ludCBzaG91bGQgaGF2ZSBhIGUyZSB0ZXN0IHRvIGNvdmVyIHRoZSBtYWluIHVzZSBjYXNlcy4KCi0gRm9yIGVkZ2UgY2FzZXMsIHdlIGNhbiB1c2UgdW5pdCB0ZXN0cyB0byB0ZXN0IHRoZSBlZGdlIGNhc2VzLCBkb24ndCBuZWVkIHRvIHVzZSBlMmUgdGVzdCB0byBjb3ZlciBhbGwgY2FzZXMsIGUyZSBqdXN0IGNvdmVyIG1haW4gd29ya2Zsb3cuCgotIENsYXNzIG5hbWUgc2hvdWxkIGJlIGNhcGl0YWxpemVkLgoKLSBEbyBub3Qgd3JpdGUgU1FMIGRpcmVjdGx5IGluIGByb3V0ZXJgIGFuZCBgbW9kZWxgIGZpbGVzLCB0aGVyZSBpcyBhIGZ1bmN0aW9uIGNhbGxlZCBgZGVsZWdhdGVSZXBvc2l0b3J5YCBjYW4gaGVscCB0byBzaW1wbGlmeSBzb21lIHNpbXBsZSBjYXNlczsKCi0gUGxlYXNlIHVzZSBgbG9nbGV2ZWxgIHRvIHJlcGxhY2UgYGNvbnNvbGUubG9nYCwgYW5kIGFsd2F5cyB1c2UgYXBwcm9wcmlhdGUgbG9nIGxldmVsIHRvIGxvZy4KCiMgQXJjaGl0ZWN0dXJlIG9mIHRoaXMgcHJvamVjdAoKVGhpcyBwcm9qZWN0IHVzZSBtdWx0aXBsZSBsYXllciBzdHJ1Y3R1cmUgdG8gYnVpbGQgdGhlIHdob2xlIHN5c3RlbS4gVGhlIGFyY2hpdGVjdHVyZSBmb2xsb3dzIHNvbWUgcHJpbmNpcGxlcyBvZiBbRG9tYWluIERyaXZlbiBEZXNpZ25dKGh0dHBzOi8vZW4ud2lraXBlZGlhLm9yZy93aWtpL0RvbWFpbi1kcml2ZW5fZGVzaWduKS4gQW5kIGluc3BpcmVkIGJ5IHRoaXMgYXJ0aWNsZTogaHR0cHM6Ly9tZWRpdW0uY29tL3Nwb3RsaWdodC1vbi1qYXZhc2NyaXB0L2RvbWFpbi1kcml2ZW4tZGVzaWduLWZvci1qYXZhc2NyaXB0LWRldmVsb3BlcnMtOWZjM2Y2ODE5MzFhCgpXZSBjb21wb3NlL2J1aWxkIHRoZSBtb2RlbCBpbnN0YW5jZSBpbiBhIGZ1bmN0aW9uYWwgc3R5bGUsIHRoZW4gZXhlY3V0ZSB0aGUgYnVzaW5lc3MuCgotICoqUHJvdG9jb2wgbGF5ZXIqKgoKSSB0aGluayB3ZSBjYW4gYWxzbyBjYWxsIGl0IHRoZSBBcHBsaWNhdGlvbiBMYXllciwgYSB0ZXJtIGluIHRoZSBEREQsIHRoZSBlbnRyeSBvZiB0aGlzIHByb2plY3QuCgpUaGlzIG1pY3Jvc2VydmljZSBvZmZlcnMgUkVTVEZ1bCBBUEkgaW50ZXJmYWNlIGJhc2VkIG9uIEhUVFAgcHJvdG9jb2wuIFdlIHVzZSBFeHByZXNzIHRvIGhhbmRsZSBhbGwgSFRUUCByZXF1ZXN0cy4KClRoZSBFeHByZXNzLXJvdXRlcnMgd29yayBsaWtlIHRoZSBjb250cm9sbGVyIHJvbGUgaW4gTVZDLCB0aGV5IHJlY2VpdmUgdGhlIHJlcXVlc3RzIGFuZCBwYXJhbWV0ZXJzIGZyb20gY2xpZW50LCBhbmQgdHJhbnNsYXRlIGl0IGFuZCBkaXNwYXRjaCB0YXNrcyB0byBhcHByb3ByaWF0ZSBidXNpbmVzcyBtb2RlbHMuIFRoZW4gcmVjZWl2ZSB0aGUgcmVzdWx0IGZyb20gdGhlbSwgdHJhbnNsYXRlIHRvIHRoZSAndmlldycsIHRoZSBKU09OIHJlc3BvbnNlLCB0byBjbGllbnQuCgotICoqTW9kZWwgbGF5ZXIqKgoKVGhlIGJ1c2luZXNzIG1vZGVsLCBtb3N0IG9mIHRoZSBidXNpbmVzcyBsb2dpYyBpcyBoZXJlLiBXZSBhcmUgY29uc2lkZXJpbmcgcHV0IG1vc3Qgb2YgdGhlIGJ1c2luZXNzIGxvZ2ljIGluIHRoZSBtb2RlbCBsYXllci4gU28gaXQgc2hvdWxkIGJlIHRoZSB0aGlja2VzdCBsYXllciwgaW5jbHVkaW5nIGFsbCB0aGUgYnVzaW5lc3MgbG9naWMsIGFuZCBidWlsZCB1cCB0aGUgcmVsYXRpb25zaGlwIGJldHdlZW4gbW9kZWxzKGJ1c2luZXNzKS4KCi0gKipJbmZyYXN0cnVjdHVyZSBsYXllcioqCgogIC0gKipSZXBvc2l0b3J5IGxheWVyKioKCiAgICBSZXBvc2l0b3J5IGlzIHJlc3BvbnNpYmxlIGZvciBjb21tdW5pY2F0aW5nIHdpdGggdGhlIHJlYWwgZGF0YWJhc2UuCgogICAgQWxsIHRoZSBTUUwgc3RhdGVtZW50cyBzaG91bGQgYmUgaGVyZS4KCiAgLSAqKk90aGVycyoqCgogICAgRm9yIGV4YW1wbGUsIHRoZSBNZXNzYWdlIFF1ZXVlLgoKIyBBYm91dCB0aGUgZG9jdW1lbnRhdGlvbi9zcGVjaWZpY2F0aW9uCgpXZSB1c2UgT3BlbkFQSSAzLjAgdG8gZG9jdW1lbnQgdGhlIEFQSS4KCllvdSBjYW4gY29weSB0aGUgeWFtbCBmaWxlIGFuZCBpbXBvcnQgdG8gc3dhZ2dlci11aSB0byBzZWUgdGhlIEFQSS4KCiMgSG93IHRvIHRlc3QKCiMjIFVuaXQgdGVzdAoKVG8gcnVuIHRoZSB1bml0IHRlc3RzOgoKYGBgCm5wbSBydW4gdGVzdC11bml0CmBgYAoKIyMgRW5kIHRvIEVuZCB0ZXN0CgpBbGwgdGhlIGVuZCB0byBlbmQgdGVzdHMgYXJlIGxvY2F0ZWQgdW5kZXIgZm9sZGVyIGBfX3Rlc3RzX18vZTJlYCwgdGhlIHRlc3Qgd2lsbCBydW4gYWdhaW5zdCB0aGUgZGV2IGRhdGFiYXNlLgoKVG8gcnVuIHRoZSBpbnRlZ3JhdGlvbiB0ZXN0OgoKUnVuIHRlc3RzOgoKYGBgCm5wbSBydW4gdGVzdC1lMmUKYGBgCgoqKklmIGVycm9yczoqKgoKYGBgCgpEQVRBQkFTRV9VUkwgaXMgdW5kZWZpbmVkCgpvcgoKeyAiY29kZSI6IDUwMCwibWVzc2FnZSI6ICJVbmtub3duIGVycm9yIChzZWxmIHNpZ25lZCBjZXJ0aWZpY2F0ZSBpbiBjZXJ0aWZpY2F0ZSBjaGFpbikiIH0KYGBgCgoqKkZvbGxvd3MgdGhvc2Ugc3RlcHM6KioKCjEtIEdvIHRvIHRoZSAuZW52IGZpbGUsIGNvcHkgdGhlIERBVEFCQVNFX1VSTCB3aXRoIGl0cyB2YWx1ZS4KCjItIEFkZCBpdCB3aXRoIE5PREVfVExTX1JFSkVDVF9VTkFVVEhPUklaRUQ9JzAnICwgbnBtIHJ1biB0ZXN0LWUyZSwgYW5kIHJ1biB0aGUgdGVzdHMuCgpGb3IgZXhhbXBsZToKCmBgYAoKREFUQUJBU0VfVVJMPVt0aGUgbGluayBwcm92aWRlZF0gTk9ERV9UTFNfUkVKRUNUX1VOQVVUSE9SSVpFRD0nMCcgbnBtIHJ1biB0ZXN0LWUyZQpgYGAKCi4KLgouCi4KLgouCi4K readmeEtag: '"5388e765dcc1d00f3a990488d39db21fd04ff112"' readmeLastModified: Mon, 08 Jul 2024 00:52:41 GMT repositoryId: 431701346 description: >- To fetch Greenstand map data for client's like web map, wallet app, and so on. created: '2021-11-25T03:25:18Z' updated: '2025-10-06T09:12:35Z' language: TypeScript archived: false stars: 16 watchers: 9 forks: 60 owner: Greenstand logo: https://avatars.githubusercontent.com/u/25363578?v=4 license: GPL-3.0 repoEtag: '"f8917d5ccef34e3f184191ea9a435e1a90b22182397f2265f33854d1303fc949"' repoLastModified: Mon, 06 Oct 2025 09:12:35 GMT foundInMaster: true category: Server Implementations id: 7d2ec66434cc6bbf5bad52bd1d32e0f7 - source: - openapi3 tags - openapi31 tags repository: https://github.com/ts-oas/ts-oas v3: true v3_1: true id: 8d7ac7c8151dcbc595ddbbcc548ccdad repositoryMetadata: base64Readme: >- # Typescript OpenAPI Spec Generator

[![NPM version](https://img.shields.io/npm/v/ts-oas.svg)](https://www.npmjs.com/package/ts-oas)
![GitHub License](https://img.shields.io/github/license/ts-oas/ts-oas)
![NPM Unpacked Size](https://img.shields.io/npm/unpacked-size/ts-oas)

Automatically generate OpenAPI (formerly Swagger) specifications from opinionated Typescript types. Supports OpenAPI **v3.1** and **v3.0**. Requires interfaces/types in a specific format.

## Benefits

-   **Write once, use many.** Typescript provides a fluent way to declare API specifications. With `ts-oas`, you can use the generated specs for documentation, input validation (e.g. with AJV), and serialization — while still reusing the original types in your business logic, unit tests, and more.
-   **Automation first.** Simply write a script or use the CLI to regenerate specs accordingly after any type changes.
-   **Framework-agnostic.** Works seamlessly with any server-side framework, unlike some other tools.

## Features

-   Both [Programmatic](#a-quick-example) and [Command line](#cli) support.
-   Reference schemas and components. Generate schema references that correspond to their Typescript type references.
-   Supports JSDoc annotations. With both pre-defined and custom keywords, metadata can be included in every schema object.
-   Schema processor function for any custom post-processing (if JSDoc annotations aren't enough).
-   Generate json-schemas separately.
-   Typescript 4 and 5 compliant.

## Install

```
npm i ts-oas
```

## Getting Started

Firstly, We need types for each API, compatible with the following format:

```ts
type Api = {
    path: string;
    method: HTTPMethod;
    body?: Record<string, any>;
    params?: Record<string, any>;
    query?: Record<string, any>;
    responses: Partial<Record<HttpStatusCode, any>>;
    security?: Record<string, string[]>[];
};
```

### A quick example

We have `interfaces.ts` where our API types are present:

```ts
import { ApiMapper } from "ts-oas";
// ApiMapper only validates the type structure at compile time,
// it's optional, omit it if you need dependency-free interface files.

export type GetBarAPI = {
    path: "/foo/bar/{id}";
    method: "GET";
    params: {
        id: number;
    };
    query: {
        from_date: Date;
    };
    responses: {
        /** @contentType application/json */
        "200": Bar;
        "404": { success: false };
    };
};

/**
 * Sample description.
 * @summary Add a Bar
 */
export type AddBarAPI = ApiMapper<{
    path: "/foo/bar";
    method: "POST";
    body: Bar;
    responses: {
        /** No content */
        "201": never;
    };
}>;

export type Bar = {
    /**
     * Description for barName.
     * @minLength 10
     */
    barName: string;
    barType: "one" | "two";
};
```

In `script.ts` file:

```ts
import { createProgram, TsOAS } from "ts-oas";
import { resolve } from "path";

// create a Typescript program. or any generic ts program can be used.
const tsProgram = createProgram(["interfaces.ts"], { strictNullChecks: true }, resolve());

// initiate the OAS generator.
const tsoas = new TsOAS(tsProgram, { ref: true });

// get the complete OAS. determine type names (Regex/exact name) to be considered for specs.
const specObject = tsoas.getOpenApiSpec([/API$/]); // all types that ends with "API"

// log results:
console.log(JSON.stringify(specObject, null, 4));

// or write into a ts file:
// writeFileSync("./schema.ts", `const spec = ${inspect(specObject, { depth: null })};\n`);
```

Run the above script.

<details><summary>Expected output</summary>

```json
{
    "openapi": "3.1.0",
    "info": {
        "title": "OpenAPI specification",
        "version": "1.0.0"
    },
    "components": {
        "schemas": {
            "Bar": {
                "type": "object",
                "properties": {
                    "barName": {
                        "description": "Description for barName.",
                        "minLength": 10,
                        "type": "string"
                    },
                    "barType": {
                        "enum": ["one", "two"],
                        "type": "string"
                    }
                },
                "required": ["barName", "barType"]
            }
        }
    },
    "paths": {
        "/foo/bar/{id}": {
            "get": {
                "operationId": "GetBarAPI",
                "parameters": [
                    {
                        "name": "id",
                        "in": "path",
                        "required": true,
                        "schema": {
                            "type": "number"
                        }
                    },
                    {
                        "name": "from_date",
                        "in": "query",
                        "required": true,
                        "schema": {
                            "type": "string",
                            "format": "date-time"
                        }
                    }
                ],
                "responses": {
                    "200": {
                        "description": "",
                        "content": {
                            "application/json": {
                                "schema": {
                                    "$ref": "#/components/schemas/Bar"
                                }
                            }
                        }
                    },
                    "404": {
                        "description": "",
                        "content": {
                            "*/*": {
                                "schema": {
                                    "type": "object",
                                    "properties": {
                                        "success": {
                                            "type": "boolean",
                                            "enum": [false]
                                        }
                                    },
                                    "required": ["success"]
                                }
                            }
                        }
                    }
                }
            }
        },
        "/foo/bar": {
            "post": {
                "operationId": "AddBarAPI",
                "description": "Sample description.",
                "summary": "Add a Bar",
                "requestBody": {
                    "content": {
                        "*/*": {
                            "schema": {
                                "$ref": "#/components/schemas/Bar"
                            }
                        }
                    }
                },
                "responses": {
                    "201": {
                        "description": "No content"
                    }
                }
            }
        }
    }
}
```

</details>

### Get json-schemas separately

Schemas with any format can be generated by:

```ts
const schema = tsoas.getSchemas(["Bar"]);
console.log(schema);
```

<details><summary>Expected output</summary>

```
{
  Bar: {
    type: 'object',
    properties: {
      barName: {
        description: 'Description for barName.',
        minLength: 10,
        type: 'string'
      },
      barType: { enum: [ 'one', 'two' ], type: 'string' }
    },
    required: [ 'barName', 'barType' ]
  }
}
```

</details>

## CLI

Command line tool is designed to behave just like the programmatic way. Once it has been installed, CLI can be executed using `npx ts-oas`, or just `ts-oas` if installed globally.

```
Usage: ts-oas <file-paths> <type-names> [options]

<file-paths> : Comma-separated list of relative .ts file paths which contain
types.
<type-names> : Comma-separated list of type names (Regex/exact name) to be
considered in files.

Options:
  -c, --tsconfig-file  Path to a JSON tsconfig file.                    [string]
  -p, --options-file   Path to a JSON file containing 'ts-oas' Options. Refer to
                       the documentation.                               [string]
  -s, --spec-file      Path to a JSON file containing additional OpenAPI
                       specifications.                                  [string]
  -e, --schema-only    Only generates pure schemas from given types.
                       ('spec-file' will be ignored.)                  [boolean]
  -o, --output         Path to a JSON file that will be used to write the
                       output. Will create the file if not existed.     [string]
  -h, --help           Show help                                       [boolean]
  -v, --version        Show version number                             [boolean]

Examples:
  ts-oas ./interfaces/sample.ts myApi,mySecondApi

```

## Documentations

### JSDoc annotations

| Keyword                                                        | Fields  | Examples                                                                                                     |
| -------------------------------------------------------------- | ------- | ------------------------------------------------------------------------------------------------------------ |
| @default                                                       | any     | `@default foo` `@default 3` `@default ["a", "b", "c"]`                                                       |
| @format                                                        | strings | `@format email`                                                                                              |
| @items                                                         | arrays  | `@items.minimum 1` `@items.format email` `@items {"type":"integer", "minimum":0}` `@default ["a", "b", "c"]` |
| @$ref                                                          | any     | `@ref http://my-schema.org`                                                                                  |
| @title                                                         | any     | `@title foo`                                                                                                 |
| @minimum<br>@maximum<br>@exclusiveMinimum<br>@exclusiveMaximum | numbers | `@minimum 10` `@maximum 100`                                                                                 |
| @minLength<br>@maxLength                                       | strings | `@minLength 10` `@maxLength 100`                                                                             |
| @minItems<br>@maxItems                                         | arrays  | `@minItems 10` `@maxItems 100`                                                                               |
| @minProperties<br>@maxProperties                               | objects | `@minProperties 10` `@maxProperties 100`                                                                     |
| @additionalProperties                                          | objects | `@additionalProperties`                                                                                      |
| @ignore                                                        | any     | `@ignore`                                                                                                    |
| @pattern                                                       | strings | `@pattern /^[0-9a-z]+$/g`                                                                                    |
| @example                                                       | any     | `@example foo` `@example {"abc":true}` `@example require('./examples.ts').myExampleConst`                    |
| @examples                                                      | any     | `@example ["foo", "bar"]` `@example require('./examples.ts').myExampleArrayConst`                            |

#### Special keywords for root of API types

| Keyword           | Description                                                |
| ----------------- | ---------------------------------------------------------- |
| @summary          | Provides a brief summary of the API endpoint               |
| @operationId      | Sets a unique identifier for the operation                 |
| @tags             | Assigns tags to group related operations (comma-separated) |
| @deprecated       | Marks the operation as deprecated                          |
| @ignore           | Excludes the API type from the generated specification     |
| @body.description | Adds a description for the request body                    |
| @body.contentType | Sets the content-type for the request body                 |

<details><summary>Example</summary>

```ts
/**
 * Sample description.
 * @summary Summary of Endpoint
 * @operationId addBar
 * @tags foos,bars
 * @deprecated
 * @ignore
 * @body.description Description for body of request.
 * @body.contentType application/json
 */
export type AddBarAPI = ApiMapper<{
    path: "/foo/bar";
    method: "POST";
    body: Bar;
    responses: {
        "201": {};
    };
}>;
```

</details>

#### Special keywords for response items

| Keyword      | Description                            |
| ------------ | -------------------------------------- |
| @contentType | Sets the content-type for the response |

<details><summary>Example</summary>

```ts
    ...
    responses: {
        /**
        * Description for response 200.
        * @contentType application/json
        */
        "200": { success: true };
    };
```

</details>

### Options

#### `ref`

> _default: false_

Defines references for schemas based on their type references.

#### `titles`

> _default: false_

Provides a `title` field in each schema, filled with its corresponding field name or type name.

#### `ignoreRequired`

> _default: false_

Ignores the `required` field in all schemas.

#### `ignoreErrors`

> _default: false_

Ignores errors in Typescript files. May introduce wrong schemas.

#### `uniqueNames`

> _default: false_

Replaces every type name with a unique hash to avoid duplication issues.

#### `tsNodeRegister`

> _default: false_

Uses `ts-node/register` as a runtime argument, enabling direct execution of TypeScript on Node.js without precompiling.

#### `nullableKeyword`

> _default: true_

Provides `nullable: true` for nullable fields; otherwise, set `type: "null"`.

#### `defaultContentType`

> _default: "\*/\*"_

Sets the default content type for all operations. This can be overridden case-by-case (see the annotations section).

#### `defaultNumberType`

> _default: "number"_

Sets the default schema type for number values, which can be overridden case-by-case (see the annotations section).

#### `customKeywords`

A list of custom keywords to consider in annotations.

#### `customKeywordPrefix`

> _default: "x-"_

The prefix added to all `customKeywords`.

#### `customOperationProperties`

> _default: false_

Whether to consider custom operation properties in the root of API types.
If true, avoid using `ApiMapper`, as it will override these properties.

#### `schemaProcessor`

A function that runs over each generated schema.

## Inspirations

`ts-oas` is highly inspired by [typescript-json-schema](https://github.com/YousefED/typescript-json-schema). While using the so-called library, it took lots of workarounds to create compatible OpenAPI v3.0 specs. For example, modifying output schemas enforced us to use schema-walker tools which added lots of overhead in our scripts (Despite of compatible OpenAPI schemas in `ts-oas`, we added a schema-processor custom function as an option as well).

## Contributing

Contributions of any kind are welcome!

<details><summary>TODOs</summary>

-   [x] CLI
-   [ ] Support for request and response header specs
-   [ ] More docs and examples
-   [ ] Complete tests
 readmeEtag: '"4b2d24c2972d4c623a6ad3a31e86d7da81a0c41d"' readmeLastModified: Thu, 25 Sep 2025 06:15:28 GMT repositoryId: 559674094 description: >- Automatically generate OpenAPI specifications from opinionated Typescript types. created: '2022-10-30T20:36:46Z' updated: '2026-01-12T09:31:20Z' language: TypeScript archived: false stars: 43 watchers: 2 forks: 6 owner: ts-oas logo: https://avatars.githubusercontent.com/u/117396202?v=4 license: MIT repoEtag: '"88b43d9daebc7eae5dae189636d0f7caec2b9646e04adaf8d518532c908d7b22"' repoLastModified: Mon, 12 Jan 2026 09:31:20 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/marcelthole/openapi-merge v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIE1lcmdlCgoKWyFbVGVzdCBTdGF0dXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXJjZWx0aG9sZS9vcGVuYXBpLW1lcmdlL3dvcmtmbG93cy9UZXN0cy9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbWFyY2VsdGhvbGUvb3BlbmFwaS1tZXJnZS9hY3Rpb25zKQpbIVtEb2NrZXIgQnVpbGQgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vbWFyY2VsdGhvbGUvb3BlbmFwaS1tZXJnZS93b3JrZmxvd3MvRG9ja2VyLUJ1aWxkL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXJjZWx0aG9sZS9vcGVuYXBpLW1lcmdlL2FjdGlvbnMpClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9tYXJjZWx0aG9sZS9vcGVuYXBpLW1lcmdlL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj1kZmZWYmhxeHZnKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL21hcmNlbHRob2xlL29wZW5hcGktbWVyZ2UpClshW011dGF0aW9uIHRlc3RpbmcgYmFkZ2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZW5kcG9pbnQ/c3R5bGU9ZmxhdCZ1cmw9aHR0cHMlM0ElMkYlMkZiYWRnZS1hcGkuc3RyeWtlci1tdXRhdG9yLmlvJTJGZ2l0aHViLmNvbSUyRm1hcmNlbHRob2xlJTJGb3BlbmFwaS1tZXJnZSUyRm1haW4pXShodHRwczovL2Rhc2hib2FyZC5zdHJ5a2VyLW11dGF0b3IuaW8vcmVwb3J0cy9naXRodWIuY29tL21hcmNlbHRob2xlL29wZW5hcGktbWVyZ2UvbWFpbikKWyFbTGF0ZXN0IFN0YWJsZSBWZXJzaW9uXShodHRwczovL3Bvc2VyLnB1Z3gub3JnL21hcmNlbHRob2xlL29wZW5hcGktbWVyZ2UvdildKC8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9tYXJjZWx0aG9sZS9vcGVuYXBpLW1lcmdlKQpbIVtMaWNlbnNlXShodHRwczovL3Bvc2VyLnB1Z3gub3JnL21hcmNlbHRob2xlL29wZW5hcGktbWVyZ2UvbGljZW5zZSldKC8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9tYXJjZWx0aG9sZS9vcGVuYXBpLW1lcmdlKQoKClJlYWQgbXVsdGlwbGUgT3BlbkFQSSAzLjAueCBZQU1MIGFuZCBKU09OIGZpbGVzIGFuZCBtZXJnZSB0aGVtIGludG8gb25lIGxhcmdlIGZpbGUuICAKVGhpcyBhcHBsaWNhdGlvbiBpcyBidWlsZCBvbiBbY2ViZS9waHAtb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL2NlYmUvcGhwLW9wZW5hcGkpIAoKIyBJbnN0YWxsYXRpb24KYGBgCmNvbXBvc2VyIHJlcXVpcmUgbWFyY2VsdGhvbGUvb3BlbmFwaS1tZXJnZQpgYGAKCiMgVXNhZ2UKIyMgQ0xJCmBgYAokIHZlbmRvci9iaW4vb3BlbmFwaS1tZXJnZSAtLWhlbHAKClVzYWdlOgogICAgb3BlbmFwaS1tZXJnZSBiYXNlZmlsZS55bWwgYWRkaXRpb25hbEZpbGVBLnltbCBhZGRpdGlvbmFsRmlsZUIueW1sIFsuLi5dICA+IGNvbWJpbmVkLnltbAoKYGBgCgojIyMgQXJndW1lbnRzCnwgQXJndW1lbnQgfCBNZWFuaW5nIHwKfCAtLS0gfCAtLS0gIHwKfCAtLW1hdGNoWz1NQVRDSF0gfCBVc2UgYSBSZWdFeCBwYXR0ZXJuIHRvIGRldGVybWluZSB0aGUgYWRkaXRpb25hbEZpbGVzLiBJZiB0aGlzIG9wdGlvbiBpcyBzZXQgdGhlIGFkZGl0aW9uYWxGaWxlcyBjb3VsZCBiZSBvbWl0dGVkIChtdWx0aXBsZSB2YWx1ZXMgYWxsb3dlZCkgfAp8IC0tcmVzb2x2ZS1yZWZlcmVuY2VzWz1SRVNPTFZFLVJFRkVSRU5DRVNdIHwgUmVzb2x2ZSB0aGUgIiRyZWZzIiBpbiB0aGUgZ2l2ZW4gZmlsZXMgW2RlZmF1bHQ6IHRydWVdIHwKfCAtbywgLS1vdXRwdXRmaWxlWz1PVVRQVVRGSUxFXSB8IERlZmluZXMgdGhlIG91dHB1dCBmaWxlIGZvciB0aGUgcmVzdWx0LiBEZWZhdWx0cyB0aGUgcmVzdWx0IHdpbGwgcHJpbnRlZCB0byBzdGRvdXQgfAoKCiMjIERvY2tlcgpSdW4gdGhlIGBvcGVuYXBpLW1lcmdlYCBjb21tYW5kIHdpdGhpbiBhIGRvY2tlciBjb250YWluZXIKYGBgCmRvY2tlciBwdWxsIGdoY3IuaW8vbWFyY2VsdGhvbGUvb3BlbmFwaS1tZXJnZQpkb2NrZXIgcnVuIC12ICRQV0Q6L2FwcCAtLXJtIGdoY3IuaW8vbWFyY2VsdGhvbGUvb3BlbmFwaS1tZXJnZSBbYXJndW1lbnRzXQpgYGAKCkJ1aWxkIHRoZSBpbWFnZSBsb2NhbGx5IGZyb20gdGhlIHNvdXJjZWNvZGU6CmBgYApkb2NrZXIgYnVpbGQgLS1idWlsZC1hcmcgQ09NUE9TRVJfUkVRVUlSRV9WRVJTSU9OPTx2ZXJzaW9uPiAtLW5vLWNhY2hlIC10IG1hcmNlbHRob2xlL29wZW5hcGktbWVyZ2U6ZGV2IGRvY2tlcgpkb2NrZXIgcnVuIC12ICRQV0Q6L2FwcCAtLXJtIG1hcmNlbHRob2xlL29wZW5hcGktbWVyZ2U6ZGV2IFthcmd1bWVudHNdCmBgYAoKIyMgT3V0cHV0Zm9ybWF0ClRoZSBvdXRwdXQgZm9ybWF0IGlzIGRldGVybWluZWQgYnkgdGhlIGJhc2VmaWxlIGV4dGVuc2lvbi4K readmeEtag: '"f38a618bcdaad980cdeebe190ddf0b31e15dde62"' readmeLastModified: Tue, 06 Aug 2024 10:46:53 GMT repositoryId: 306759712 description: null created: '2020-10-23T22:26:31Z' updated: '2025-10-06T09:08:30Z' language: PHP archived: false stars: 18 watchers: 1 forks: 6 owner: marcelthole logo: https://avatars.githubusercontent.com/u/850125?v=4 license: MIT repoEtag: '"cad5d7d983a9c7c14c21e2d5520cd20f5b96abc2e80de5bbf9c63043c06a432f"' repoLastModified: Mon, 06 Oct 2025 09:08:30 GMT foundInMaster: true category: Description Validators id: d4f30650d1fe678296b5e48ab1cf97bb - source: openapi3 tags repository: https://github.com/yukihirop/r2-oas v3: true repositoryMetadata: base64Readme: >- [![Gem Version](https://badge.fury.io/rb/r2-oas.svg)](https://badge.fury.io/rb/r2-oas)
[![Build Status](https://travis-ci.org/yukihirop/r2-oas.svg?branch=master)](https://travis-ci.org/yukihirop/r2-oas)
[![Coverage Status](https://coveralls.io/repos/github/yukihirop/r2-oas/badge.svg)](https://coveralls.io/github/yukihirop/r2-oas)
[![Maintainability](https://api.codeclimate.com/v1/badges/f8c3846f350bb412fd63/maintainability)](https://codeclimate.com/github/yukihirop/r2-oas/maintainability)

<p align="center">
	<img alt="logo" width="196" src="https://raw.githubusercontent.com/yukihirop/r2-oas/master/docs/assets/logo.png">
</p>

<h1 align="center" style="font-family: sans-serif; font-size: 50px; font-weight: 700; background: -webkit-linear-gradient(90deg, #CC342D 20%, #E63D36 60%, #FF5C55 100%); -webkit-background-clip: text; background-clip: text; -webkit-text-fill-color: transparent; color: transparent;">R2-OAS</h1>

<p align="center">
	<a href="https://yukihirop.github.io/r2-oas/" target="_blank">📚 Documentation</a>
</p>

Generate api document (OpenAPI) side only from `Rails` routing.

Provides a rake command to help `generate`, `view`, and `edit` OpenAPI documents.

```bash
bundle exec rake routes:oas:init    # r2-oas initialize
bundle exec rake routes:oas:docs    # generate oas_docs
bundle exec rake routes:oas:ui      # view at swagger ui
bundle exec rake routes:oas:editor  # edit at swagger editor
bundle exec rake routes:oas:monitor # monitor oas_docs and analyze
bundle exec rake routes:oas:build   # build oas_docs from src
bundle exec rake routes:oas:clean   # clean unused components
bundle exec rake routes:oas:analyze # analyze oas_docs and generae src
bundle exec rake routes:oas:deploy  # deploy oas_docs to deploy_docs
```

## 💎 Installation

Add this line to your application's Gemfile:

```ruby
group :development do
  gem 'r2-oas'
end
```

And then execute:

    $ bundle

Or install it yourself as:

    $ gem install r2-oas

## 🔦 Requirements

If you want to view with `Swagger UI` or edit with `Swagger Editor`, This gem needs the following:

- [`swaggerapi/swagger-ui:latest` docker image](https://hub.docker.com/r/swaggerapi/swagger-ui/)
- [`swaggerapi/swagger-editor:latest` docker image](https://hub.docker.com/r/swaggerapi/swagger-editor/)
- [`chromedriver`](http://chromedriver.chromium.org/downloads)

If you do not have it download as below.

```
$ docker pull swaggerapi/swagger-editor:latest
$ docker pull swaggerapi/swagger-ui:latest
$ brew install chromedriver
```

## 🚀 Tutorial

After requiring a gem and Configure `Rakefile` in your rails project

```rb
R2OAS.load_tasks
```

```bash
$ bundle exec rake routes:oas:init
      create	oas_docs
      create	oas_docs/.paths
      create	oas_docs/plugins/helpers
      create	oas_docs/tasks/helpers
      create	oas_docs/plugins/.gitkeep
      create	oas_docs/plugins/helpers/.gitkeep
      create	oas_docs/tasks/.gitkeep
      create	oas_docs/tasks/helpers/.gitkeep
$ bundle exec rake routes:oas:docs
$ bundle exec rake routes:oas:editor
```

#### Generate docs

https://github.com/user-attachments/assets/0d094a92-dbcd-49f0-9ef0-e1809f0710a9

#### Edit docs

https://github.com/user-attachments/assets/e10a695c-1d18-4da6-8827-28cb3d0b9e3a

## Usage

You can execute the following command in the root directory of rails.  
The following are examples of typical command usage.

Full docs are available at https://yukihirop.github.io/r2-oas

### Initialize

Initialize r2-oas.

```bash
$ bundle exec rake routes:oas:init
      create	oas_docs
      create	oas_docs/.paths
      create	oas_docs/plugins/helpers
      create	oas_docs/tasks/helpers
      create	oas_docs/plugins/.gitkeep
      create	oas_docs/plugins/helpers/.gitkeep
      create	oas_docs/tasks/.gitkeep
      create	oas_docs/tasks/helpers/.gitkeep
```

### Generate

Generate docs.

```bash
$ bundle exec rake routes:oas:docs                                                       # Generate docs
$ PATHS_FILE="oas_docs/schema/paths/api/v1/task.yml" bundle exec rake routes:oas:docs    # Generate docs by specify unit paths
```

### Editor

Start Swagger editor.

```bash
$ bundle exec rake routes:oas:editor                                                     # Start swagger editor
$ PATHS_FILE="oas_docs/schema/paths/api/v1/task.yml" bundle exec rake routes:oas:editor  # Start swagger editor by specify unit paths
```

### UI

Start swagger ui.

```bash
$ bundle exec rake routes:oas:ui                                                         # Start swagger ui
$ PATHS_FILE="oas_docs/schema/paths/api/v1/task.yml" bundle exec rake routes:oas:ui      # Start swagger ui by specify unit paths
```

### Build

Build docs.  
`Plugin is applied`

```bash
$ bundle exec rake routes:oas:build
```

### Analyze

Analyze docs.  
Reads OpenAPI format document and divides it into several parts to generate a source file

```bash
$ OAS_FILE="~/Desktop/swagger.yml" bundle exec rake routes:oas:analyze
```

## 📚 Documents

Full docs are available at https://yukihirop.github.io/r2-oas

## ❤️ Support Rails Version

- Rails 7.2.2.2
- Rails (>= 8.x)

## ❤️ Support Ruby Version

- Ruby 3.2.6
- Ruby 3.3.6
- Ruby (>= 3.4.x)

## ❤️ Support Rouging

- Rails Engine Routing
- Rails Normal Routing

## ❤️ Support OpenAPI Schema

Full docs are available at https://yukihirop.github.io/r2-oas/#/schema/3.0.0

## ❗️ Convention over Configuration (CoC)

- `tag name` represents `controller name` and determines `paths file name`.
  - For example, If `controller name` is `Api::V1::UsersController`, `tag_name` is `api/v1/user`, then `paths file name` is `api/v1/user.yml`

- `_` of `components/{schemas,requestBodies, ...} name` convert `/` when save file.
  - For example, If `components/schemas name` is `Api_V1_User`, `components/schemas file name` is `api/v1/user.yml`.
  - `_` is supposed to be used to express `namespace`.
  - format is `Namespace1_Namespace2_Model`.

- `.` of `components/{schemas,requestBodies, ...} name` convert `/` when saving the file.
  - For example, If `components/schemas name` is `api.v1.User`, `components/schemas file name` is `api/v1/user.yml`.
  - `.` is supposed to be used to express `namespace`.
  - format is `namespace1.namespace2.Model`.

## ⚙ Configure

All settings are `optional`

Full docs are available at https://yukihirop.github.io/r2-oas/#/setting/configure

## RBS, Steep

```bash
# Errors here can prevent type information from showing when hovering in the editor
bundle exec steep stats
# You can check the error details with this command
bundle exec steep check
```

#### steep rake task

```bash
bundle exec rake steep:ignore:dig | pbcopy
# =>
#
# ignore 'RBS::DuplicatedMethodDefinition'
# ignore 'Ruby::ArgumentTypeMismatch'
```

```bash
bundle exec rake steep:ignore:file | pbcopy
# =>
#
# ignore 'lib/r2-oas.rb'
# ignore 'lib/r2-oas/app_configuration.rb'
# ignore 'lib/r2-oas/app_configuration/deprecation.rb'
```

```bash
bundle exec rake steep:ignore:preview
# =>
# 📄 lib/r2-oas/schema/monitor.rb
#   Line 50: if current_time - last_check_time >= interval_to_save_edited_tmp_schema
#               →           if current_time - last_check_time >= interval_to_save_edited_tmp_schema # steep:ignore Ruby::NoMethod
#   Line 62: analyzer.analyze_docs
#               →         analyzer.analyze_docs # steep:ignore Ruby::NoMethod
#   Line 66: YAML.load_file(doc_save_file_path) || @after_schema_data
#               →         YAML.load_file(doc_save_file_path) || @after_schema_data # steep:ignore Ruby::UnknownConstant
```

```bash
bundle exec rake steep:ignore:auto
# =>
# Analyzing steep check errors...
# ✅ Modified: lib/r2-oas/app_configuration/deprecation.rb (2 lines)
# ✅ Modified: lib/r2-oas/plugin/base.rb (3 lines)
# ✅ Modified: lib/r2-oas/schema/squeezer.rb (2 lines)
```

## Bundle and Rspec with multiple ruby ​​versions

#### Bundle

```bash
/bin/bash devscript/all_support_ruby.sh bundle
.
.
.
===== Bundle install for All Support Ruby Result =====
ruby-3.2.6: 0
ruby-3.3.6: 0
ruby-3.4.7: 0
======================================================
```

If specify ruby version `3.2.6` and `3.3.6`

```bash
/bin/bash devscript/all_support_ruby.sh bundle 3.2.6 3.3.6
.
.
.
===== Bundle install for All Support Ruby Result =====
ruby-3.2.6: 0
ruby-3.3.6: 0
======================================================
```

#### Rspec

```bash
/bin/bash devscript/all_support_ruby.sh rspec
.
.
.
===== Rspec for All Support Ruby Result =====
ruby-3.2.6: 0
ruby-3.3.6: 0
ruby-3.4.7: 0
=============================================
```

If specify ruby version `3.2.6` and `3.3.6`

```bash
/bin/bash devscript/all_support_ruby.sh rspec 3.2.6 3.3.6
.
.
.
===== Rspec for All Support Ruby Result =====
ruby-3.2.6: 0
ruby-3.3.6: 0
=============================================
```

## 🔩 CORS

Use [rack-cors](https://github.com/cyu/rack-cors) to enable CORS.

```ruby
require 'rack/cors'
use Rack::Cors do
  allow do
    origins '*'
    resource '*', headers: :any, methods: [ :get, :post, :put, :delete, :options ]
  end
end
```

Alternatively, you can set CORS headers in a `before` block.

```ruby
before do
  header['Access-Control-Allow-Origin'] = '*'
  header['Access-Control-Request-Method'] = '*'
end
```

## 📝 License

The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).

## 🤝 Contributing

1. Fork it ( http://github.com/yukihirop/r2-oas/fork )
2. Create your feature branch (`git checkout -b my-new-feature`)
3. Commit your changes (`git commit -am 'Add some feature'`)
4. Push to the branch (`git push origin my-new-feature`)
5. Create new Pull Request
 readmeEtag: '"b085ebc730bf84831c6de8439577a5326c93b9b9"' readmeLastModified: Fri, 31 Oct 2025 15:28:04 GMT repositoryId: 175621234 description: >- r2-oas allows generation of OpenAPI Specification (V3) from config/routes.rb in Rails application. https://deepwiki.com/yukihirop/r2-oas created: '2019-03-14T12:51:19Z' updated: '2025-10-31T15:28:09Z' language: Ruby archived: false stars: 15 watchers: 1 forks: 4 owner: yukihirop logo: https://avatars.githubusercontent.com/u/11146767?v=4 license: MIT repoEtag: '"4fe3c959af3f3d52be57fa0a9bd2519d979acf4441c9a581cb8deab0c8ac0840"' repoLastModified: Fri, 31 Oct 2025 15:28:09 GMT foundInMaster: true category: Data Validators id: c0d7926604df1b7fdea354fdb2b100b9 - source: openapi3 tags repository: https://github.com/vymalo/keycloak-phone-number v3: true id: cdc1a6a78dba21b420011984b42b5d12 repositoryMetadata: base64Readme: >- # Keycloak Phone Number Login Plugin

This plugin helps you to login by phone number.

| Keycloak Version | Plugin Version |
|------------------|----------------|
| 21               | ✅ 1.1.0        |
| 22               | ✅ 26.1.3       |
| 23               | ✅ 26.1.3       |
| 24               | ✅ 26.1.3       |
| 25               | ✅ 26.1.3       |
| 26               | ✅ 26.1.3       |

![Example](./docs/screenshot.png)

## 1. What It Is

The **Keycloak Phone Number Login Plugin** is a custom extension for Keycloak that enables authentication via SMS-based
phone number login. Instead of (or in addition to) traditional username/password logins, this plugin allows end-users to
authenticate using their phone number. The process includes:

- **Phone Number Entry:** Users enter their phone number.
- **Confirmation:** The system displays a formatted version of the number for user confirmation.
- **User Lookup/Creation:** The plugin checks if the user exists (creating one if not).
- **SMS TAN Sending:** A Transaction Authentication Number (TAN) is generated and sent via an external SMS API.
- **TAN Validation:** The user inputs the TAN to complete authentication.
- **Profile Update:** On successful TAN verification, users can update their profile information.

This modular plugin uses a multi-step flow and is designed with separation of concerns in mind, using a dedicated
service layer for SMS and phone number processing.

## New Feature: Dual Authentication Support for SMS Service

The `SmsService` has been enhanced to support both **Basic Authentication** and **OAuth2 Client Credentials Grant** for secure communication with the SMS provider. The plugin prioritizes OAuth2 if the required environment variables are set; otherwise, it falls back to Basic Auth.

## 2. How to Use It

### Downloading the Plugin

You can download the latest plugin JAR from the GitHub artifacts using a `curl` command. For example:

```bash
curl -L -o keycloak-phonenumber-login.jar https://github.com/vymalo/keycloak-phone-number/releases/download/v<version>/keycloak-phonenumber-login-<version>.jar
```

### Mounting into Keycloak

Once you have the JAR, mount it into the Keycloak server. For example, when running Keycloak in Docker, you can mount
the plugin as a volume:

#### Docker Example

```bash
docker run -d \
  --name keycloak \
  -p 8080:8080 \
  -e KEYCLOAK_ADMIN=admin \
  -e KEYCLOAK_ADMIN_PASSWORD=password \
  -e SMS_API_URL=http://your-sms-api-url \
  -e SMS_API_COUNTRY_PATTERN='cm|de|fr' \
  -e SMS_API_AUTH_USERNAME=someuser \
  -e SMS_API_AUTH_PASSWORD=somepassword \
  -e OAUTH2_CLIENT_ID=some-client-id \
  -e OAUTH2_CLIENT_SECRET=some-client-secret \
  -e OAUTH2_TOKEN_ENDPOINT=http://token-mock:8080/token \
  -v /path/to/keycloak-phonenumber-login.jar:/opt/keycloak/providers/keycloak-phonenumber-login.jar \
  quay.io/keycloak/keycloak:26.1.2 start-dev
```

#### Kubernetes Example

```yaml
apiVersion: apps/v1
kind: Deployment
metadata:
  name: keycloak
spec:
  replicas: 1
  selector:
    matchLabels:
      app: keycloak
  template:
    metadata:
      labels:
        app: keycloak
    spec:
      volumes:
        - name: plugin-volume
          emptyDir: {}
      initContainers:
        - name: download-plugin
          image: curlimages/curl:8.1.2
          env:
            - name: VERSION
              value: 26.1.3
          command:
            - sh
            - -c
            - |
              curl -L -o /plugin/keycloak-phonenumber-login.jar https://github.com/vymalo/keycloak-phone-number/releases/download/v$VERSION/keycloak-phonenumber-login-$VERSION.jar
          volumeMounts:
            - name: plugin-volume
              mountPath: /plugin
      containers:
        - name: keycloak
          image: quay.io/keycloak/keycloak:26.1.2
          ports:
            - containerPort: 8080
          env:
            - name: KEYCLOAK_ADMIN
              value: "admin"
            - name: KEYCLOAK_ADMIN_PASSWORD
              value: "password"
            - name: SMS_API_URL
              value: "http://your-sms-api-url"
            - name: SMS_API_COUNTRY_PATTERN
              value: "cm|de|fr"
            - name: SMS_API_AUTH_USERNAME
              value: "someuser"
            - name: SMS_API_AUTH_PASSWORD
              value: "somepassword"
            - name: OAUTH2_CLIENT_ID
              value: "some-client-id"
            - name: OAUTH2_CLIENT_SECRET
              value: "some-client-secret"
            - name: OAUTH2_TOKEN_ENDPOINT
              value: "http://token-mock:8080/token"
          volumeMounts:
            - name: plugin-volume
              mountPath: /opt/keycloak/providers
```

*Tip:* For Kubernetes, it's more common to use a sidecar or init container to download the plugin from GitHub and copy
it into the appropriate directory.

---

## 3. Environment Variables

The following environment variables are used by the plugin:

- **KEYCLOAK_ADMIN**: The administrator username for Keycloak.  
- **KEYCLOAK_ADMIN_PASSWORD**: The administrator password for Keycloak.  
- **KC_LOG_CONSOLE_COLOR**: Enables colored logging in the console (set to `'true'` or `'false'`).  
- **KC_HTTP_PORT**: The HTTP port on which Keycloak runs.  
- **SMS_API_URL**: The base URL of the SMS API service.  
- **SMS_API_COUNTRY_PATTERN**: A regex pattern to match supported phone number country codes.
- **SMS_API_AUTH_USERNAME**: The basic auth username for the SMS API.
- **SMS_API_AUTH_PASSWORD**: The basic auth password for the SMS API.
- **OAUTH2_CLIENT_ID**: The client ID for OAuth2 authentication with the SMS provider.  
- **OAUTH2_CLIENT_SECRET**: The client secret for OAuth2 authentication with the SMS provider.  
- **OAUTH2_TOKEN_ENDPOINT**: The URL of the token endpoint for OAuth2 authentication (e.g., `http://token-mock:8080/token` for testing, or `http://your-auth-server/oauth/token` in production).


Configure these variables in your deployment (Docker, Kubernetes, etc.) as shown in the examples above.  

---

## 4. Architecture

The plugin is built using a multi-module Maven structure with the following layers:

- **Core Module:**  
  Contains shared constants, utilities, and models (e.g., phone number helpers, country codes).

- **Authenticator Module:**  
  Implements custom Keycloak authenticators that control the multi-step authentication flow:
    - **PhoneNumberGetNumber:** Collects user phone numbers.
    - **PhoneNumberConfirmNumber:** Displays the confirmed phone number.
    - **PhoneNumberChooseUser:** Looks up or creates users based on phone number.
    - **PhoneNumberSendTan:** Sends a TAN via an SMS service.
    - **PhoneNumberValidateTan:** Validates the TAN entered by the user.
    - **PhoneNumberUpdateUser:** Prompts users to update their profile post-authentication.

- **Service Module:**  
  Encapsulates business logic for phone number processing and SMS operations:
    - **PhoneNumberService:** Validates and formats phone numbers.
    - **SmsService:** Handles interactions with the external SMS API (using an OpenAPI-generated client).

- **OpenAPI Client Module:**  
  Contains the generated client code for the SMS API integration, ensuring a robust and type-safe communication with the
  external service.

Dependency injection (via CDI) is used to wire components together, making the system more modular, testable, and
maintainable.

---

## 5. Contribute

We welcome contributions! Here’s how you can get involved:

### Reporting Issues

- Use the GitHub issues page to report bugs, suggest enhancements, or request features.
- Ensure your issue includes detailed reproduction steps and relevant logs.

### Contributing Code

- **Fork the Repository:**  
  Create a fork, then clone it locally.
- **Branching Model:**
    - Use feature branches for new features or fixes.
    - Ensure your branch name is descriptive (e.g., `feature/sms-service-refactor`).
- **Coding Standards:**  
  Follow the existing coding conventions.
    - Write unit tests for your changes.
    - Run `mvn clean install` and ensure all tests pass.
- **Pull Requests:**  
  Submit a pull request with a clear description of your changes, referencing any issues addressed.

### Documentation & Feedback

- Update documentation as needed with your changes.
- Provide feedback on areas for improvement or additional features.

---

*For more details or questions, please contact [dev@ssegning.com](mailto:dev@ssegning.com).* readmeEtag: '"0820e74afb4a959bfd759b9e73db6af3d15c6971"' readmeLastModified: Sat, 22 Mar 2025 07:56:06 GMT repositoryId: 290233843 description: Keycloak plugin for logins using phone number created: '2020-08-25T14:13:59Z' updated: '2026-02-04T03:31:59Z' language: Java archived: false stars: 44 watchers: 1 forks: 12 owner: vymalo logo: https://avatars.githubusercontent.com/u/128943481?v=4 license: MIT repoEtag: '"f279a177a3b5293635781f356f262b73605256784351663a4d556bef1d1a49d8"' repoLastModified: Wed, 04 Feb 2026 03:31:59 GMT category: Server Implementations oldLocations: - https://github.com/bayamsell/keycloak-phone-number foundInMaster: true - source: openapi3 tags repository: https://github.com/0xsirsaif/frappeapi v3: true id: 13092aaf1a2505f819c5f57374f67408 repositoryMetadata: base64Readme: >- # FrappeAPI

Better APIs for Frappe.

FrappeAPI brings FastAPI-style routing and validation to the Frappe Framework. Define endpoints with type hints, get automatic validation and documentation.

## Installation

```bash
pip install frappeapi
```

## Quick Start

```python
from frappeapi import FrappeAPI

app = FrappeAPI()

@app.get()
def hello(name: str = "World"):
    return {"message": f"Hello, {name}!"}
```

## Examples

### Path Parameters

Enable FastAPI-style paths for cleaner URLs:

```python
from frappeapi import FrappeAPI

app = FrappeAPI(fastapi_path_format=True)

@app.get("/items/{item_id}")
def get_item(item_id: str):
    return {"id": item_id}

# GET /api/items/abc123
# Response: {"id": "abc123"}
```

Multiple path parameters:

```python
@app.get("/users/{user_id}/orders/{order_id}")
def get_user_order(user_id: str, order_id: int):
    return {"user_id": user_id, "order_id": order_id}

# GET /api/users/john/orders/42
# Response: {"user_id": "john", "order_id": 42}
```

Combine path and query parameters:

```python
@app.get("/products/{category}")
def list_products(
    category: str,           # Path parameter
    sort_by: str = "name",   # Query parameter
    limit: int = 10          # Query parameter
):
    return {"category": category, "sort_by": sort_by, "limit": limit}

# GET /api/products/electronics?sort_by=price&limit=20
```

### Query Parameters

Automatic type parsing:

```python
@app.get()
def get_product_details(
    product_id: int,
    unit_price: float,
    in_stock: bool
):
    return {
        "product_id": product_id,  # "123" -> 123
        "unit_price": unit_price,  # "9.99" -> 9.99
        "in_stock": in_stock       # "true" -> True
    }
```

Optional parameters with defaults:

```python
@app.get()
def list_products(
    category: str = "all",
    page: int = 1,
    search: str | None = None
):
    return {"category": category, "page": page, "search": search}
```

Enum parameters:

```python
from enum import Enum

class OrderStatus(str, Enum):
    pending = "pending"
    processing = "processing"
    completed = "completed"

@app.get()
def list_orders(status: OrderStatus = OrderStatus.pending):
    return {"status": status}
```

List parameters:

```python
from frappeapi import Query

@app.get()
def search_products(
    tags: List[str] = Query(default=[]),
    categories: List[int] = Query(default=[])
):
    return {"tags": tags, "categories": categories}

# GET ?tags=electronics&tags=sale&categories=1&categories=2
# Response: {"tags": ["electronics", "sale"], "categories": [1, 2]}
```

Aliased parameters:

```python
from typing import Annotated
from frappeapi import Query

@app.get()
def search_items(
    search_text: Annotated[str, Query(alias="q")] = "",
    page_number: Annotated[int, Query(alias="p")] = 1
):
    return {"search": search_text, "page": page_number}

# GET ?q=laptop&p=2
```

Query parameter models:

```python
from pydantic import BaseModel, Field

class ProductFilter(BaseModel):
    search: str | None = None
    category: str = "all"
    min_price: float = Field(0, ge=0)
    in_stock: bool = True

@app.get()
def filter_products(filters: Annotated[ProductFilter, Query()]):
    return filters
```

### Request Body

Single model:

```python
from pydantic import BaseModel, Field

class Item(BaseModel):
    name: str = Field(..., min_length=1, max_length=50)
    description: str | None = None
    price: float = Field(..., gt=0)

@app.post()
def create_item(item: Item):
    return item
```

Multiple body parameters:

```python
class User(BaseModel):
    username: str
    email: str

class Item(BaseModel):
    name: str
    price: float

@app.post()
def create_user_item(user: User, item: Item):
    return {"user": user, "item": item}

# Request body:
# {
#     "user": {"username": "john", "email": "john@example.com"},
#     "item": {"name": "Laptop", "price": 999.99}
# }
```

Nested models:

```python
from pydantic import HttpUrl

class Image(BaseModel):
    url: HttpUrl
    name: str

class Product(BaseModel):
    name: str
    price: float
    images: List[Image]

@app.post()
def create_product(product: Product):
    return product
```

### Form Data

```python
from typing import Annotated
from frappeapi import Form

@app.post()
def create_user_profile(
    username: Annotated[str, Form()],
    email: Annotated[str, Form()],
    bio: Annotated[str | None, Form()] = None
):
    return {"username": username, "email": email, "bio": bio}
```

### File Uploads

Small files (in-memory):

```python
from typing import Annotated
from frappeapi import File, Form

@app.post()
def upload_document(
    file: Annotated[bytes, File()],
    description: Annotated[str | None, Form()] = None
):
    return {"file_size": len(file), "description": description}
```

Large files (streamed):

```python
from frappeapi import UploadFile

@app.post()
def upload_large_file(file: UploadFile):
    return {
        "filename": file.filename,
        "content_type": file.content_type
    }
```

### Response Models

Filter response data:

```python
class UserResponse(BaseModel):
    id: int
    username: str
    email: str

@app.get(response_model=UserResponse)
def get_user(user_id: int):
    return {
        "id": user_id,
        "username": "john_doe",
        "email": "john@example.com",
        "password": "secret"  # Filtered out
    }
```

List responses:

```python
class Product(BaseModel):
    id: int
    name: str
    price: float

@app.get(response_model=List[Product])
def list_products():
    return [
        {"id": 1, "name": "Laptop", "price": 999.99},
        {"id": 2, "name": "Mouse", "price": 24.99}
    ]
```

### Error Handling

Raise HTTP exceptions:

```python
from frappeapi.exceptions import HTTPException

@app.get()
def get_item(item_id: int):
    if item_id < 0:
        raise HTTPException(status_code=400, detail="Item ID must be positive")
    return {"id": item_id}
```

Custom exception handlers:

```python
from frappeapi import JSONResponse, Request

class ItemNotFound(Exception):
    def __init__(self, item_id: int):
        self.item_id = item_id

@app.exception_handler(ItemNotFound)
def item_not_found_handler(request: Request, exc: ItemNotFound):
    return JSONResponse(
        status_code=404,
        content={"error": "ITEM_NOT_FOUND", "detail": f"Item {exc.item_id} not found"}
    )
```

Override validation error handler:

```python
from frappeapi.exceptions import RequestValidationError

@app.exception_handler(RequestValidationError)
def validation_error_handler(request: Request, exc: RequestValidationError):
    return JSONResponse(
        status_code=422,
        content={
            "error": "VALIDATION_ERROR",
            "details": [{"field": e["loc"][-1], "message": e["msg"]} for e in exc.errors()]
        }
    )
```

### Header Parameters

```python
from typing import Annotated
from frappeapi import Header

@app.get()
def get_user_info(
    user_agent: Annotated[str, Header()],
    x_custom_header: Annotated[str, Header()]
):
    return {"user_agent": user_agent, "custom_header": x_custom_header}

# Headers: User-Agent, X-Custom-Header (hyphen converted to underscore)
```

### Field Validation

String validation:

```python
class Product(BaseModel):
    name: str = Field(min_length=3, max_length=50)
    sku: str = Field(pattern="^[A-Z]{2}-[0-9]{4}$")  # Format: XX-0000
```

Numeric validation:

```python
class Order(BaseModel):
    quantity: int = Field(gt=0, le=100)
    unit_price: float = Field(gt=0)
    discount_percent: float = Field(ge=0, le=100)
```

## Version Compatibility

FrappeAPI automatically detects your Frappe version:

| Frappe Version | Support |
|---------------|---------|
| v14.x | Stable |
| v15.x | Stable |
| v16.x | Beta |

Check detected version:

```python
import frappeapi
print(frappeapi.get_detected_frappe_version())  # Returns: 14, 15, or 16
```

## Documentation

FrappeAPI follows FastAPI's interface. For detailed information, see [FastAPI's documentation](https://fastapi.tiangolo.com/).

## Roadmap

### Frappe Versions

- [x] Frappe V14 support
- [x] Frappe V15 support
- [x] Frappe V16 support (develop branch)

### Methods

- [x] `app.get(...)`
- [x] `app.post(...)`
- [x] `app.put(...)`
- [x] `app.patch(...)`
- [x] `app.delete(...)`

### Query Parameters

- [x] Automatic type parsing based on type hints
- [x] Required parameters (`needy: str`, `needy: str = ...`)
- [x] Optional parameters with defaults (`skip: int = 0`)
- [x] Optional parameters without defaults (`limit: int | None = None`)
- [x] Enum support
- [x] Boolean parameters
- [x] List parameters (`?q=foo&q=bar`)
- [x] Aliased parameters (`Query(alias="query")`)
- [x] Query parameter models
- [x] Automatic documentation generation

### Body Parameters

- [x] Pydantic model body (`item: Item`)
- [x] Multiple body parameters
- [x] Singular values with `Body()`
- [x] Embed body parameter
- [x] Nested models
- [x] Automatic type parsing

### Header Parameters

- [x] Basic header support
- [ ] Header parameter models
- [ ] Duplicate headers
- [ ] Forbid extra headers

### Cookie Parameters

- [ ] Cookie parameter support

### Form Data

- [x] Form fields with `Form()`
- [x] Multiple form fields
- [x] Form data as Pydantic model
- [ ] Forbid extra form fields

### File Uploads

- [x] Small files with `File()`
- [x] Large files with `UploadFile`
- [ ] Optional file uploads
- [ ] Multiple file uploads

### Error Handling

- [x] HTTPException
- [x] RequestValidationError
- [x] ResponseValidationError
- [x] Custom exception handlers
- [x] Override default handlers
- [ ] Frappe transaction management

### Response Models

- [x] `response_model` parameter
- [x] Return type annotations
- [x] Output filtering
- [x] `response_model` takes precedence over return type

### Validation

- [x] String validations (`min_length`, `max_length`, `pattern`)
- [x] Numeric validations (`gt`, `ge`, `lt`, `le`)
- [x] Metadata (`title`, `description`, `deprecated`)
- [x] `include_in_schema`

### Planned

- [ ] Rate limiting
- [ ] Dependencies
- [ ] Middleware
- [ ] Debugging capabilities
- [ ] Dotted path parameters

## Related

- [PR #23135](https://github.com/frappe/frappe/pull/23135): Type hints for API functions
- [PR #22300](https://github.com/frappe/frappe/pull/22300): Enhanced `frappe.whitelist()`
- [PR #19029](https://github.com/frappe/frappe/pull/19029): Type safety improvements
- [Issue #14905](https://github.com/frappe/frappe/issues/14905): API documentation discussion
 readmeEtag: '"d081126ee5cb104ed843358dd2707482a7344ec8"' readmeLastModified: Sat, 24 Jan 2026 21:40:41 GMT repositoryId: 825093329 description: FastAPI for Frappe created: '2024-07-06T18:48:18Z' updated: '2026-01-24T21:40:46Z' language: Python archived: false stars: 57 watchers: 4 forks: 4 owner: 0xsirsaif logo: https://avatars.githubusercontent.com/u/55336614?v=4 repoEtag: '"c6d5e63caccc357e7a7529e6ba0766540dbca610532573ec8a8b79129381881e"' repoLastModified: Sat, 24 Jan 2026 21:40:46 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/0xsirsaif/frappe-api - source: openapi3 tags repository: https://github.com/180protocol/codaptor v3: true repositoryMetadata: base64Readme: >- <img src=".github/Codaptor-logo.png" alt="Codaptor Logo" width="40%"/>

Codaptor is an open source project designed to address the needs of teams building decentralized applications
on [Corda](https://github.com/corda/corda) or integrating CorDapps with other systems. Corda is a great engine
for decentralized applications, but in order to communicate with a Corda node, teams have to develop bespoke
integrations using its Java client libraries. This comes with a steep learning curve, and adds complexity for
for teams working outside Java ecosystem, e.g. Node.js, .Net, or Python.

Codaptor solves this problem by automatically creating REST API for any CorDapp running on a Corda node. There are
many tools that understand REST APIs in every technology stack, and teams can pick and choose what works for them.
Codaptor also allows to decouple API users from the underlying Corda node in order to improve the availability,
reliability, and flexibility of the overall system.

## Features

* Zero-configuration deployment option as an embedded Corda service -- great for development and integration testing
* Generation of [OpenAPI 3](https://github.com/OAI/OpenAPI-Specification) JSON spec for the API endpoint based 
  on actual flows and contract state classes of available CorDapps
* Embedded [Swagger UI](https://swagger.io/tools/swagger-ui/) for exploring the API in your browser
* Docker-friendly standalone deployment option configurable via environment variables and compatible with
  Kubernetes secrets management
* Synchronous or asynchronous execution of Corda flows via the API
* Comprehensive API for node vault queries
* Flexible API security based on [PAC4J](https://www.pac4j.org/) allowing such authorization models as API keys,
  OpenID Connect, SAML 2, or just about anything else -- great for integrating managed services like AWS Cognito
* Full support for SSL out of the box
* Extensible architecture allowing bespoke features to be added as simply as dropping a JAR file into a directory

## Versions and artifacts

At the moment Codaptor is a pre-1.0 technology preview made available to the community to gather feedback and identify
areas for improvement. It is not considered production-ready yet.

Latest Codaptor release is [0.4.1](https://github.com/180Protocol/codaptor/releases/tag/0.4.1). You can download
standalone bundle archive from the release page on GitHub. From version 0.4.1 cordaptor releases are tagged with the corda platform version they are built under. For ex, the current 0.4.1 version is tagged as `0.4.1-corda4.7`. All Codaptor modules are 
[available in Maven Central](https://search.maven.org/search?q=g:tech.b180.cordaptor%20AND%20v:0.4.1-corda4.7).
Embedded CorDapp bundle can be added using coordinates `tech.b180.cordaptor:cordaptor-bundle-rest-embedded:0.4.1-corda4.7` (
see [getting started guide](./docs/getting-started.md))

Branch [0.1.x](https://github.com/b180tech/cordaptor/tree/0.1.x) was created for bug fixes, and subsequent
releases of the 0.x.x are going to be bugfix only. Snapshot versions of all artifacts (`0.4.1-corda4.7-SNAPSHOT`)
are available from [OSS Sonatype Cordaptor](https://oss.sonatype.org/#nexus-search;gav~tech.b180.cordaptor)
Maven repository if you do not want to wait for the release. Published snapshot versions pass all automated tests.

All new functionality is developed in `master` branch, and the next major milestone release is going to be 0.5.0.
Snapshot versions of all artifacts (`0.4.1-corda4.7-SNAPSHOT`) are available from
[OSS Sonatype Cordaptor](https://oss.sonatype.org/#nexus-search;gav~tech.b180.cordaptor)
Maven repository if you do not want to wait for the release. Published snapshot versions pass all automated tests.

## Getting started

Codaptor is designed from the ground up to be unobstructed, so there is **no code or configuration required**!
Simply download the embedded CorDapp bundle JAR file from the
[latest release page](https://github.com/b180tech/cordaptor/releases) and drop it into `cordapps` 
directory of your Corda node. Restart the node and fire up your browser to access the Swagger UI.

Read more in [Getting started](./docs/getting-started.md) guide about other ways to get immediately
productive with Codaptor.

## Next steps

* Learn how to [get started](./docs/getting-started.md) with Codaptor
* Learn more about [using](./docs/how-to-use.md) Codaptor
* Read about how Codaptor fits into your [architecture](./docs/architecture.md)
* Understand how to [configure](./docs/configuration.md) Codaptor
* Learn how to [create extensions](./docs/extensions.md) for Codaptor
* Stay tuned to updates in the [development blog](https://medium.com/b180tech)
* If you are developing for the Codaptor [developer guide](./docs/developer-guide.md)

## Getting support

We offer community support for Codaptor in `#cordaptor` channel on [Cordaledger Slack](https://slack.corda.net/).
The development team members can often be seen hanging out there.

If something isn't working, feel free to file a bug report through
[GitHub Issues](https://github.com/b180tech/cordaptor/issues). Although, we ask you to search for
related messages in the above Slack channel and among existing reported issues on GitHub first
to avoid creating duplicates.

For bespoke feature development, custom integrations and extensions, or commercial support enquiries, please email 
to [management@bond180.com](mailto:management@bond180.com?subject=Cordaptor).

## Contributing

Codaptor is an open-source project and contributions are welcome!

## License

[GNU Affero General Public License version 3 or later](./LICENSE)

SPDX:AGPL-3.0-or-later

Copyright (C) 2020 Bond180 Limited

**Important notice**: for the avoidance of doubt in the interpretation of the license terms,
the copyright holders commit to treat the following uses of Codaptor as 'aggregate' as opposed to 'modified versions':
1. Deploying embedded Codaptor bundle JAR file into a Corda node, regardless of whether it is a file
distributed as a binary or built from the source code, as long as the source code of all modules in the bundle
remains unmodified.
2. Annotating application code with annotation types provided by Codaptor in order to fine-tune the behaviour
of Codaptor components interacting with the application code.
3. Creating extensions for Codaptor using it's published microkernel's and modules' APIs, where the
extensions' code is assembled into separate JAR files and made available for Codaptor microkernel
to dynamically discover at runtime. For clarity's sake, code constituting published APIs must be appropriately
annotated, see [Extending Cordaptor](./docs/extensions.md) for further details.
4. Including Codaptor as a component of a broader application architecture where other components interact with it
using network communication protocols regardless of how Codaptor is deployed and configured.

The intent of using AGPL is to protect the interests of the Codaptor user community and ensure any bug fixes
and important new features developed by some users become available to everyone else. It is not the intent of
using AGPL to force disclose of any proprietary application code relying on Codaptor.
 readmeEtag: '"d2e9e89471e0f2d0b4e1e640d59978a8ce286929"' readmeLastModified: Thu, 21 Jul 2022 11:16:00 GMT repositoryId: 311944546 description: >- Instantly add Corda to any tech stack and improve resiliency of your architecture created: '2020-11-11T11:02:29Z' updated: '2025-04-27T20:09:21Z' language: Kotlin archived: false stars: 14 watchers: 2 forks: 5 owner: 180Protocol logo: https://avatars.githubusercontent.com/u/74298369?v=4 license: AGPL-3.0 repoEtag: '"6b4bca34f9036c3150d56b3349ec2dcb87f07a78387d9bf47232b39bc1b2d453"' repoLastModified: Sun, 27 Apr 2025 20:09:21 GMT foundInMaster: true category: - Documentation - Server Implementations id: f46a7821f80968f7a52b8d1d69aabac9 - source: openapi3 tags repository: https://github.com/nijens/openapi-bundle v3: true repositoryMetadata: base64Readme: >- # OpenAPI bundle

[![Latest version on Packagist][ico-version]][link-version]
[![Software License][ico-license]][link-license]
[![Build Status][ico-build]][link-build]
[![Code Quality][ico-code-quality]][link-code-quality]

Helps you create a REST API from your OpenAPI specification.

This bundle supports a design-first methodology for creating an API with Symfony by providing the following tools:

* [Loading the path items and operations of an OpenAPI specification as routes](#routing)
* [Validation of the request to those routes](#validation-of-the-request)
* [Deserialization of a validated JSON request body into an object](#deserialize-a-json-request-body)
* [OpenAPI-based serialization context for the Symfony Serializer](#openapi-based-serialization-context-for-the-symfony-serializer)
* [Exception handling](#exception-handling)

## Installation

### Applications that use Symfony Flex
Open a command console, enter your project directory and execute:

```console
composer require nijens/openapi-bundle
```

### Applications that don't use Symfony Flex

#### Step 1: Download the Bundle

Open a command console, enter your project directory and execute the
following command to download the latest stable version of this bundle:

```console
composer require nijens/openapi-bundle
```

This command requires you to have Composer installed globally, as explained
in the [installation chapter](https://getcomposer.org/doc/00-intro.md)
of the Composer documentation.

#### Step 2: Enable the Bundle

Then, enable the bundle by adding it to the list of registered bundles
in the `src/Kernel.php` file of your project:

```php
<?php
// src/Kernel.php

// ...
class Kernel extends BaseKernel
{
    public function registerBundles(): iterable
    {
        return [
            // ...
            new Nijens\OpenapiBundle\NijensOpenapiBundle(),
        ];
    }

    // ...
}
```

## Usage
Before starting with the implementation of the bundle, you should take the time to design your API according
to the OpenAPI specification.

The following resources can help you with designing the specification:
* [OpenAPI specification version 3](https://swagger.io/specification)
* [Swagger specification editor](https://editor.swagger.io)

### Routing
This bundle provides a route loader that loads [path items](https://swagger.io/specification/#pathItemObject)
and [operations](https://swagger.io/specification/#operationObject) from your OpenAPI document.

You load your OpenAPI document by configuring it in the routing of your application:

```yaml
# app/config/routes.yml

api:
    prefix: /api
    resource: ../openapi.yaml # or ../openapi.json
    type: openapi
    name_prefix: "api_"
```

Within the OpenAPI document we will use the `x-openapi-bundle` specification extension to add additional configuration
to the operations defined in the document.

#### Configuring a controller for a route
A Symfony controller for a route is configured by adding the `controller` property to the `x-openapi-bundle`
specification extension within an operation within your OpenAPI document.

```yaml
paths:
  /pets/{uuid}:
    put:
      x-openapi-bundle:
        controller: 'Nijens\OpenapiBundle\Controller\PetController::put'
      requestBody:
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/Pet'
      responses:
        '200':
          description: 'Returns the stored pet.'
```

<details>
<summary>Example of an OpenAPI document in JSON format</summary>

```json
{
  "paths": {
    "/pets/{uuid}": {
      "put": {
        "x-openapi-bundle": {
          "controller": "Nijens\\OpenapiBundle\\Controller\\PetController::put"
        },
        "requestBody": {
          "content": {
            "application/json": {
              "schema": {
                "$ref": "#/components/schemas/Pet"
              }
            }
          }
        },
        "responses": {
          "200": {
            "description": "Returns the stored pet."
          }
        }
      }
    }
  }
}
```

</details>

The value of the `controller` property is the same as you would normally add to a [Symfony route](https://symfony.com/doc/current/routing.html#creating-routes).

#### Using the operationId of an operation as the name of the Symfony route
Within an OpenAPI document, you can give each operation an
[operationId](https://spec.openapis.org/oas/latest.html#fixed-fields-7) to better identify it.
To use the `operationId` as the name for a loaded Symfony route, add the following bundle configuration:

```yaml
# config/packages/nijens_openapi.yaml
nijens_openapi:
    routing:
        operation_id_as_route_name: true
```

Using the `operationId` for your routes gives you more control over the API route names and allows you to better use
them with a `UrlGenerator`.

### Validation of the request
By default, the [deprecated validation component](docs/validation/deprecated-validation-component.md) is enabled.
To enable the improved validation component, add the following YAML configuration.

```yaml
# config/packages/nijens_openapi.yaml
nijens_openapi:
    exception_handling:
        enabled: true

    validation:
        enabled: true
```

It is strongly advised to also enable the improved exception handling component, as it will convert the details of
the validation exceptions into proper JSON responses.

The validation component comes with validation for the following parts of a request:

* **Content-type**: Based on the configured content types configured in the `requestBody` property of an operation
* **Query parameters**: Validates the query parameters configured of the operation and path item.
  *Note that this type of validation is experimental as it might be missing validation of certain query parameter types.*
* **JSON request body**: Based on the JSON schema in the `requestBody` property of an operation

#### Learn more

* [Activate query parameter request validation](docs/validation/activate-query-parameter-request-validation.md)
* [Create your custom request validator](docs/validation/create-your-custom-request-validator.md)
* Content-type validation explained (coming soon™)
* JSON request body validation explained (coming soon™)
* Query parameter validation explained (coming soon™)

### Deserialize a JSON request body

Adding the `deserializationObject` property to the `x-openapi-bundle` specification extension of an operation activates
the request body deserialization.

When the request body is successfully validated against the JSON schema within your OpenAPI document,
it will deserialize the request body into the configured deserialization object.

The deserialized object is injected into the controller based on:

1. The type hint of the argument in the controller method.

2. The `#[DeserializedObject]` parameter attribute. (supported since PHP 8.0)

   This method is the recommended way, as it supports argument resolving for both array deserialization
   and mixed argument types.

3. The `deserializationObjectArgumentName` property that can be added to the `x-openapi-bundle`
   specification extension.

#### Learn more

* [How to use a single controller with the Symfony Messenger component](docs/deserialization/how-to-use-a-single-controller-with-the-symfony-messenger-component.md)

### OpenAPI-based serialization context for the Symfony Serializer
⚠ _**Please note:** This feature is still experimental. The API might change in a future minor version._

The `SerializationContextBuilder` helps you with creating a serialization context for the Symfony Serializer.
It allows you to easily create a JSON response from an object or entity based on your OpenAPI specification.

The following example shows how to use the serialization context builder by leveraging the request attributes added
by the routing.

```php
<?php

use Nijens\OpenapiBundle\Routing\RouteContext;
use Nijens\OpenapiBundle\Serialization\SerializationContextBuilderInterface;
use Symfony\Component\HttpFoundation\JsonResponse;
use Symfony\Component\HttpFoundation\Request;
use Symfony\Component\Serializer\SerializerInterface;

class ExampleController
{
    public function __invoke(
        Request $request,
        SerializerInterface $serializer,
        SerializationContextBuilderInterface $serializationContextBuilder
    ): JsonResponse {
        $pet = new Pet();

        $serializationContext = $serializationContextBuilder->getContextForSchemaObject(
            'Pet',
            $request->attributes->get(RouteContext::REQUEST_ATTRIBUTE)[RouteContext::RESOURCE]
        );

        return JsonResponse::fromJsonString(
            $serializer->serialize($pet, 'json', $serializationContext)
        );
    }
}
```

### Exception handling
By default, the [previous exception handling component](docs/previous-exception-handling.md) is enabled.
To enable the new exception handling component, add the following YAML configuration.

```yaml
# config/packages/nijens_openapi.yaml
nijens_openapi:
    exception_handling:
        enabled: true
```

The new exception handling component uses the [Problem Details JSON Object](https://datatracker.ietf.org/doc/html/rfc7807#section-3)
format to turn an exception (or `Throwable`) into a clear error response.

If you want to implement your own exception handling? Change `enabled` to `false`. It will disable the
exception handling component of the bundle.

#### Customizing the Problem Details JSON Object response of an exception
Through the exception handling configuration of the bundle, you can modify the response status code and
problem JSON response body of any `Throwable`. See the following example for more information.

```yaml
# config/packages/nijens_openapi.yaml
nijens_openapi:
    exception_handling:
        enabled: true
        exceptions:
            InvalidArgumentException:               # The fully qualified classname of the exception.
                status_code: 400                    # Modify the response status code of
                                                    # the exception response.

                type_uri: https://example.com/error # Add a unique type URI to the Problem Details.
                                                    # This could be a URL to additional documentation
                                                    # about the error.

                title: The request was invalid.     # Add a clear human-readable title property
                                                    # to the Problem Details.

                add_instance_uri: true              # Add the current route as instance_uri property
                                                    # to the Problem Details.
```

To help you include the Problem Details JSON object in your OpenAPI document, we provide an
[OpenAPI template](src/Resources/specifications/openapi_problemdetails.yaml) with schemas
for the specific Problem Details JSON objects this bundle creates.

## Credits and acknowledgements

* Author: [Niels Nijens][link-author]

Also, see the list of [contributors][link-contributors] who participated in this project.

## License
The OpenAPI bundle is licensed under the MIT License. Please see the [LICENSE file][link-license] for details.

[ico-version]: https://img.shields.io/packagist/v/nijens/openapi-bundle.svg
[ico-pre-release-version]: https://img.shields.io/packagist/vpre/nijens/openapi-bundle.svg
[ico-license]: https://img.shields.io/badge/license-MIT-brightgreen.svg
[ico-build]: https://github.com/nijens/openapi-bundle/actions/workflows/continuous-integration.yaml/badge.svg
[ico-code-quality]: https://scrutinizer-ci.com/g/nijens/openapi-bundle/badges/quality-score.png?b=main

[link-version]: https://packagist.org/packages/nijens/openapi-bundle
[link-license]: LICENSE
[link-build]: https://github.com/nijens/openapi-bundle/actions/workflows/continuous-integration.yaml
[link-coverage]: https://coveralls.io/r/nijens/openapi-bundle?branch=master
[link-code-quality]: https://scrutinizer-ci.com/g/nijens/openapi-bundle/?branch=master
[link-author]: https://github.com/niels-nijens
[link-contributors]: https://github.com/nijens/openapi-bundle/contributors
 readmeEtag: '"5b8d3a9794523acde50cfa484177df781071a6a3"' readmeLastModified: Wed, 15 May 2024 14:57:05 GMT repositoryId: 134330854 description: Helps you create a REST API from your OpenAPI specification. created: '2018-05-21T22:16:39Z' updated: '2026-01-20T14:41:16Z' language: PHP archived: false stars: 18 watchers: 4 forks: 5 owner: nijens logo: https://avatars.githubusercontent.com/u/22254531?v=4 license: MIT repoEtag: '"0254bee988a5a638051a8feb4f46bce43c94f280a6a1daf2c4d0b9e5b4f4590b"' repoLastModified: Tue, 20 Jan 2026 14:41:16 GMT foundInMaster: true category: - Data Validators - Parsers id: 2081e3bef767fdcec32955dc3c047f98 - source: - openapi3 tags - openapi31 tags repository: https://github.com/apiaddicts/openapi2postman v3: true v3_1: true id: 582d0c64ef558eda3703792449bdfa30 repositoryMetadata: base64Readme: >- # 🛠️ Openapi2Postman ![Release](https://img.shields.io/badge/release-2.0.x-purple) ![OpenApi](https://img.shields.io/badge/-openapi-%23Clojure?style=flat&logo=openapiinitiative&logoColor=white) ![Postman](https://img.shields.io/badge/postman-FF6C37?style=flat&logo=postman&logoColor=white) [![NPM](https://img.shields.io/badge/npm-%23CB3837.svg?style=flat&logo=npm&logoColor=white)](https://www.npmjs.com/package/openapi2postman) [![License: LGPL v3](https://img.shields.io/badge/license-LGPL_v3-blue.svg)](https://www.gnu.org/licenses/lgpl-3.0)

**Openapi2Postman** creates automatic tests from Openapi 3.0 using postman format. Also, it creates environments files, depending of configuration.
Those collections can be importend in postan application. The tests includes tests for 2xx, 4xx... and tests to validate output formats.

<p align="center">
	<a href="https://apiaddicts.org/">
	  <img src="https://github.com/apiaddicts/openapi2postman/raw/master/imgs/openapi2postman.svg" height = '100'>
	</a>
</p>

### This repository is intended for :octocat: **community** use, it can be modified and adapted without commercial use. If you need a version, support or help for your **enterprise** or project, please contact us 📧 devrel@apiaddicts.org

[![Twitter](https://img.shields.io/badge/Twitter-%23000000.svg?style=for-the-badge&logo=x&logoColor=white)](https://twitter.com/APIAddicts)
[![Discord](https://img.shields.io/badge/Discord-%235865F2.svg?style=for-the-badge&logo=discord&logoColor=white)](https://discord.gg/ZdbGqMBYy8)
[![LinkedIn](https://img.shields.io/badge/linkedin-%230077B5.svg?style=for-the-badge&logo=linkedin&logoColor=white)](https://www.linkedin.com/company/apiaddicts/)
[![Facebook](https://img.shields.io/badge/Facebook-%231877F2.svg?style=for-the-badge&logo=Facebook&logoColor=white)](https://www.facebook.com/apiaddicts)
[![YouTube](https://img.shields.io/badge/YouTube-%23FF0000.svg?style=for-the-badge&logo=YouTube&logoColor=white)](https://www.youtube.com/@APIAddictslmaoo)

# 🙌 Join the **Openapi2postman** Adopters list

📢 If Openapi2postman is part of your organization's toolkit, we kindly encourage you to include your company's name in our Adopters list. 🙏 This not only significantly boosts the project's visibility and reputation but also represents a small yet impactful way to give back to the project.

| Organization                                                                              | Description of Use / Referenc               |
| ----------------------------------------------------------------------------------------- | ------------------------------------------- |
| [CloudAppi](https://cloudappi.net/)                                                       | Apification and generation of microservices |
| [Madrid Digital](https://www.comunidad.madrid/servicios/sede-electronica/madrid-digital/) | Generation of microservices                 |
| [APIquality](https://apiquality.io/)  | Common contributor  |

# 👩🏽‍💻 Contribute to ApiAddicts

We're an inclusive and open community, welcoming you to join our effort to enhance ApiAddicts, and we're excited to prioritize tasks based on community input, inviting you to review and collaborate through our GitHub issue tracker.

Feel free to drop by and greet us on our GitHub discussion or Discord chat. You can also show your support by giving us some GitHub stars ⭐️, or by following us on Twitter, LinkedIn, and subscribing to our YouTube channel! 🚀

[!["Buy Me A Coffee"](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/apiaddicts)

# 📑 Getting started

[![NPM](https://img.shields.io/badge/openapi2postman-%23CB3837.svg?style=for-the-badge&logo=npm&logoColor=white)](https://www.npmjs.com/package/openapi2postman)

### Install package

```
npm i openapi2postman -g
```

### First steps

#### `Arguments`

| argv            | a   | description       |
| --------------- | --- | ----------------- |
| --file          | -f  | path openapi file |
| --configuration | -c  | path config file  |

#### `Normal` usage

```
o2p -c .\example\o2p_config_file.json -f .\example\petstore.yaml
```

#### `Default` config

```
o2p  -f .\example\petstore.yaml
```

If you do not use the -c argument, the following default configuration will be used. The output path `target_folder` is `./out`.

```
  api_name: filename,
  is_inline: false,
  schema_is_inline: false,
  schema_pretty_print: true,
  generate_oneOf_anyOf:false,
  minimal_endpoints:false,
  environments:[
    {
      name : "DEV",
      postman_collection_name: "%api_name%_DEV",
      postman_environment_name: "%api_name%_DEV",
      target_folder: "./out",
      has_scopes: false,
      application_token: false,
      number_of_scopes: 0,
      microcks_headers: false
      read_only: false,
      validate_schema:false,
	    host_server_pattern:"%dev%",
    }
  ]
```

## Table of content

- [Structure and submodules](#structure-and-submodules)
- [Build and run](#build-and-run)
  - [Prerequisites](#prerequisites)
  - [Get the source code for the first time](#get-the-source-code-for-the-first-time)
  - [Run](#run)
  - [Get the latest changes](#get-the-latest-changes)
- [Contribute](#contribute)
  - [IDE support](#ide-support)
    - [IntelliJ IDEA](#intellij-idea)
    - [Eclipse](#eclipse)
- [Documentation](#documentation)
- [Advanced Functionality](#advanced-functionality)
- [Additional resources](#additional-resources)

## Structure and submodules

- _[docs](docs)_ - This module contents the guides to configurate and run the product. Documents Pdfs by CloudAppi.
- _[example](example)_ - Integration and system tests for SoapUI.
- _[src](src)_ – Source code
- _[test](soapui-maven-plugin-tester)_ - Testing folder

## Build and run

### Prerequisites

You need node v10 or later.

### run

Please review the o2p_config_file.json before to execute the command
Execute
node index.js --configuration o2p_config_file.json --file example/swagger_provincias.yml

The output files are the following (in example folder):

- SWAGGER_API_TestSuite_DEV.postman_collection.json
- SWAGGER_API_TestSuiteEnv_DEV.postman_environment.json
- SWAGGER_API_TestSuite_PROD.postman_collection.json
- SWAGGER_API_TestSuiteEnv_PROD.postman_environment.json

In postman:

- import generated postman collection file in postman
- import generated postman environment file in postman
- update variables in the envirnoment to test all cases

### Configuration file

Collections and environments to generate can be configured using a JSON configuration file. Collection and environment name can be specified, as well as the target folder for resulting files, the authorizations collection to be used and some other characteristics. This configuration file has an specification and examples that can be read on the following document:  
[Configuration file openapi2postman](./docs/openapi2postman-guia.pdf). This document is only available in spanish by now. it will be soon translated to english.

an example configuration file is included in file structure.

### Workspace configuration

Node.js and npm package manager are required to run the tool. Its adviced to use preconfigured installation tools provided by the manufacturer:
[https://nodejs.org/es/download/](https://nodejs.org/es/download/)

## 💛 Sponsors

<p align="center">
	<a href="https://apiaddicts.org/">
    	<img src="https://apiaddicts.cloudappi.net/web/image/4248/LOGOCloudappi2020Versiones-01.png" alt="cloudappi" width="150"/>
        <img src="https://www.comunidad.madrid/sites/default/files/styles/block_teaser_image/public/img/logos-simbolos/logo_centrado_md.png?itok=4rTUhmcj" alt="md" width="150"/>
        <img src="https://apiaddicts-web.s3.eu-west-1.amazonaws.com/wp-content/uploads/2022/03/17155736/cropped-APIAddicts-logotipo_rojo.png" height = "75">
	</a>
</p>
 readmeEtag: '"d18c0813d610b703eaf2b5932294408a653a349c"' readmeLastModified: Mon, 30 Dec 2024 17:38:54 GMT repositoryId: 259690576 description: >- Project open source financed by Madrid Digital (Spain) and CloudAPPi S.L and promoted by APIAddicts. With this project, you can generate all tests automatically using a Swagger document, and you can add it to CI using newman. created: '2020-04-28T16:21:37Z' updated: '2025-12-31T15:47:07Z' language: JavaScript archived: false stars: 21 watchers: 8 forks: 7 owner: apiaddicts logo: https://avatars.githubusercontent.com/u/31730093?v=4 license: NOASSERTION repoEtag: '"5f37bb074b8f11be1606df69d26273d6ad62894a3f379205cceff1ea8d5055fe"' repoLastModified: Wed, 31 Dec 2025 15:47:07 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mermade/openapi-webconverter v3: true repositoryMetadata: base64Readme: >- IyBPbmxpbmUgU3dhZ2dlciAyLjAgdG8gT3BlbkFQSSAzLjAueCBjb252ZXJ0ZXIKCltDb252ZXJ0XShodHRwczovL21lcm1hZGUub3JnLnVrL29wZW5hcGktY29udmVydGVyKSBTd2FnZ2VyIDIuMCBkZWZpbml0aW9ucyBvciBbdmFsaWRhdGVdKGh0dHBzOi8vbWVybWFkZS5vcmcudWsvb3BlbmFwaS1jb252ZXJ0ZXIpIE9wZW5BUEkgMy4wLnggZGVmaW5pdGlvbnMKCkNvbnZlcnNpb25zIGFuZCB2YWxpZGF0aW9ucyBhcmUgcGVyZm9ybWVkIGJ5IFtPQVMtS2l0XShodHRwczovL2dpdGh1Yi5jb20vbWVybWFkZS9vYXMta2l0KQoKWW91IG1heSBhbHNvIHVzZSB0aGUgW0FQSV0oaHR0cHM6Ly9yZWRvY2x5LmdpdGh1Yi5pby9yZWRvYy8/dXJsPWh0dHBzOi8vbWVybWFkZS5vcmcudWsvb3BlbmFwaS1jb252ZXJ0ZXIvY29udHJhY3Qvc3dhZ2dlci5qc29uKSB0byBjb252ZXJ0IFN3YWdnZXIgMi4wIGRlZmluaXRpb25zIG9yIHZhbGlkYXRlIGEgMy4wLnggZGVmaW5pdGlvbgo8cD4KPHVsPgogICA8bGk+PGk+UE9TVDwvaT4gPGI+L2FwaS92MS9jb252ZXJ0PC9iPiAtIHdpdGggYSBzb3VyY2Ugb3IgZmlsZW5hbWUgcGFyYW1ldGVyLCBvcHRpb25hbGx5IHdpdGggdmFsaWRhdGU9b248L2xpPgogICA8bGk+PGk+UE9TVDwvaT4gPGI+L2FwaS92MS92YWxpZGF0ZTwvYj4gLSB3aXRoIGEgc291cmNlIG9yIGZpbGVuYW1lIHBhcmFtZXRlcjwvbGk+CiAgIDxsaT48aT5HRVQ8L2k+IDxhIGhyZWY9Imh0dHBzOi8vbWVybWFkZS5vcmcudWsvb3BlbmFwaS1jb252ZXJ0ZXIvYXBpL3YxL3N0YXR1cyI+L2FwaS92MS9zdGF0dXM8L2E+PC9saT4KICAgPGxpPjxpPkdFVDwvaT4gPGI+L2FwaS92MS9jb252ZXJ0P3VybD0uLi48L2I+IG9wdGlvbmFsbHkgd2l0aCBhIDxiPnZhbGlkYXRlPC9iPiBwYXJhbWV0ZXI8L2xpPgogICA8bGk+PGk+R0VUPC9pPiA8Yj4vYXBpL3YxL3ZhbGlkYXRlP3VybD0uLi48L2I+PC9saT4KICAgPGxpPjxpPkdFVDwvaT4gPGI+L2FwaS92MS9iYWRnZT91cmw9Li4uPC9iPiByZXR1cm5zIGEgcmVkaXJlY3QgdG8gYW4gU1ZHIGJhZGdlPC9saT4KPC91bD4KCg== readmeEtag: '"970c4b85bc245c7750c2a7fbe4768c08a8acfe50"' readmeLastModified: Tue, 04 Apr 2023 05:17:26 GMT repositoryId: 83992765 description: Mermade Swagger 2.0 to OpenAPI 3.0.0 converter front-end created: '2017-03-05T18:33:45Z' updated: '2025-02-12T06:50:03Z' language: JavaScript archived: false stars: 15 watchers: 1 forks: 8 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"79d06f940af2369a0e752d74fae187f361867dc0b5ac2a463e475b81e10176c9"' repoLastModified: Wed, 12 Feb 2025 06:50:03 GMT foundInMaster: true category: - Converters - Parsers id: 54a547cbd73c5b26b11a1f473a6d1e15 - source: openapi3 tags repository: https://github.com/vaclavnovotny/nswag.examples v3: true repositoryMetadata: base64Readme: >- ![Build and Publish](https://github.com/vaclavnovotny/NSwag.Examples/workflows/Build%20and%20Publish/badge.svg) ![Nuget](https://img.shields.io/nuget/v/NSwag.Examples?color=blue)
# Response and Request Examples for NSwag<!-- omit from toc -->
This library allows you to programmatically define swagger examples in your NSWag application. Example discovery occurs at start of application and uses reflection. 

### Overview:<!-- omit from toc -->
- [Setup](#setup)
  - [Install package](#install-package)
  - [Setup Startup.cs](#setup-startupcs)
- [Define examples for your types](#define-examples-for-your-types)
  - [Request Parameters](#request-parameters)
  - [Request Body Parameters](#request-body-parameters)
  - [Response Body](#response-body)
- [Use dependency injection](#use-dependency-injection)
- [Multiple examples](#multiple-examples)
- [Naming the examples](#naming-the-examples)
- [Set example for specific endpoint](#set-example-for-specific-endpoint)
- [Polymorphism](#polymorphism)
- [Support](#support)


# Setup

## Install package

```csharp
Install-Package NSwag.Examples
```

## Setup Startup.cs

Add services needed for example definition using `AddExampleProviders()` and provide assemblies where are your examples located.
```csharp
public void ConfigureServices(IServiceCollection services)
{
    services.AddControllers();

    services.AddExampleProviders(typeof(CityExample).Assembly);
    services.AddOpenApiDocument((settings, provider) =>
    {
        settings.AddExamples(provider);
    });
}
```

# Define examples for your types

Create class and implement interface IExampleProvider<T> where T is the type you want to define example for.
```csharp
public class CityExample : IExampleProvider<City>
{
    public City GetExample()
    {
        return new City()
        {
            Id = 5,
            Name = "Brno",
            People = new List<Person>()
            {
                new Person() {Id = 1, FirstName = "Henry", LastName = "Cavill"},
                new Person() {Id = 2, FirstName = "John", LastName = "Doe"}
            }
        };
    }
}

```
## Request Parameters
For request other than body parameters like query, route or header parameters, simply define the example using `ExampleAnnotation` atribute and set `ExampleType` to `Request` or `Both`. 
```csharp
[ExampleAnnotation(Name = "Search text 'inspektor'", ExampleType = ExampleType.Request)]
public class PersonTextExample1 : IExampleProvider<string>
{
    public string GetExample() => "inspektor";
}
```
After that, you also need decorate endpoint which examples you wish to be showed

```csharp
[HttpGet]
[EndpointSpecificExample(typeof(PersonTextExample1), ParameterName = "searchText", ExampleType = ExampleType.Request)]
public IActionResult GetPeople([FromQuery] int? minAge = null, [FromQuery] string searchText = null)
{
    ...
}
```

## Request Body Parameters

For request body parameters there is nothing you need to worry about, just mark your parameter `[FromBody]`.
```csharp
[HttpPost]
public async Task<IActionResult> CreatePerson([FromBody, BindRequired] Person person)
{
    // create person logic
    return Ok();
}
```
Result in swagger:
![Image of request body](https://github.com/vaclavnovotny/images/blob/main/requestExample.JPG)

## Response Body

For response body types you need to decorate your method with `[ProducesResponseType]`
```csharp
[HttpGet("{id}")]
[ProducesResponseType(StatusCodes.Status200OK, Type = typeof(Person))]
public async Task<IActionResult> GetPerson([FromRoute]int id)
{
    return Ok(new Person());
}
```

Result in swagger:
![Image of request body](https://github.com/vaclavnovotny/images/blob/main/responseExampleSingle.JPG)

# Use dependency injection

You can also use dependency injection in constructor of your example provider class.

Constructor and `GetExample` method gets called when operation processors are executed - when swagger specification is being generated which is during first request on swagger.
```csharp
public class PersonExample : IExampleProvider<Person>
{
    private readonly IPersonNameGenerator _nameGenerator;
    private readonly Random _random;

    public PersonExample(IPersonNameGenerator nameGenerator)
    {
        _nameGenerator = nameGenerator;
        _random = new Random(); 
    }

    public Person GetExample()
    {
        return new Person()
        {
            Id = _random.Next(1, 100),
            FirstName = _nameGenerator.GenerateRandomFirstName(),
            LastName = _nameGenerator.GenerateRandomLastName()
        };
    }
}
```

You can also combine examples by reusing `IExampleProvider<T>`. You can directly inject desired example providers into constructor and use them within your example provider.
```csharp
public class PragueExample : IExampleProvider<City>
{
    private readonly IEnumerable<IExampleProvider<Person>> _peopleExamples;

    public PragueExample(IEnumerable<IExampleProvider<Person>> peopleExamples) {
        _peopleExamples = peopleExamples;
    }
    
    public City GetExample() {
        return new City {
            Id = 420,
            Name = "Prague",
            People = _peopleExamples.Select(x => x.GetExample()).ToList()
        };
    }
}
```

# Multiple examples
To define multiple examples for request or response body, simply create more classes implementing `IExampleProvider<T>` multiple times.
Here, to define multiple examples of type `City`, we create two classes that implement interface `IExampleProvider<City>`.
```csharp
public class BrnoExample : IExampleProvider<City>
{
    public City GetExample()
    {
        return new City {
            Id = 5,
            Name = "Brno",
            People = new List<Person> {
                new Person {Id = 1, FirstName = "Henry", LastName = "Cavill"},
                new Person {Id = 2, FirstName = "John", LastName = "Doe"}
            }
        };
    }
}

public class PragueExample : IExampleProvider<City>
{
    public City GetExample() {
        return new City {
            Id = 420,
            Name = "Prague",
            People = new List<Person> {
                new Person {Id = 1, FirstName = "Henry", LastName = "Cavill"},
                new Person {Id = 2, FirstName = "John", LastName = "Doe"}
            }
        };
    }
}
```
Swagger then contains dropdown with these options:
![Multiple examples](https://github.com/vaclavnovotny/images/blob/main/multipleexamples.jpg)

# Naming the examples
You can name examples so they do not show up in Swagger as Example 1, Example 2, etc.
Simply annotate your example with `ExampleAnnotation` attribute and set `Name` property.

```csharp
[ExampleAnnotation(Name = "Brno")]
public class BrnoExample : IExampleProvider<City>
```
Swagger then shows these names in dropdown:
![Named examples](https://github.com/vaclavnovotny/images/blob/main/namingexamples.jpg)

# Restricting examples to request or response content
Examples can be annotated to restrict their usage to either a request or a response body.  This is useful when the same type may be used on both requests and responses, and you want to restrict examples without specifying specific examples.
```csharp
[ExampleAnnotation(ExampleType = ExampleType.Request)]
public class RequestExample : IExampleProvider<City> {...}

[ExampleAnnotation(ExampleType = ExampleType.Response)]
public class ResponseExample : IExampleProvider<City> {...}
```

# Set example for specific endpoint
It is possible to set specific implementation of `IExampleProvider<T>` to desired API endpoint. This may be handy when multiple endpoints return same type or if primitive type is returned but you need to specify specific value.
To set specific example, simply annotate controllers method with `EndpointSpecificExample` attribute and provide type of the example class.

```csharp
[HttpGet("{id}/age")]
[ProducesResponseType(StatusCodes.Status200OK, Type = typeof(int))]
[EndpointSpecificExample(typeof(PersonSpecificAgeExample))] // <-----
public IActionResult GetPersonAge([FromRoute] int id) {...}

public class PersonSpecificAgeExample : IExampleProvider<int>
{
    public int GetExample() => 69;
}
```

# Specify that an example is specifically for a request or response
When setting a specific implementation of `IExampleProvider<T>` for a desired API endpoint, it may be desired to specify specifically whether this example applies to the request or response.

```csharp
[HttpPost("age")]
[ProducesResponseType(StatusCodes.Status200OK, Type = typeof(int))]
[EndpointSpecificExample(typeof(PersonSpecificAge69Example), ExampleType = ExampleType.Response)] // <----- Prevents example from being used in the request
[EndpointSpecificExample(typeof(PersonSpecificAge50Example), ExampleType = ExampleType.Request)] // <----- Prevents example from being used in the response
public IActionResult GetPersonAge([FromBody] int id) {...}

// Included on response
public class PersonSpecificAge69Example : IExampleProvider<int>
{
    public int GetExample() => 69;
}

// Included on request
public class PersonSpecificAge50Example : IExampleProvider<int>
{
    public int GetExample() => 50;
}
```

# Specify that an example is specifically for a specific response code
When setting a specific implementation of `IExampleProvider<T>` for a desired API endpoint, it may be desired to specify specifically whether this example applies to the request or response.

```csharp
[HttpGet("{id}/age")]
[ProducesResponseType(StatusCodes.Status200OK, Type = typeof(int))]
[EndpointSpecificExample(typeof(PersonSpecificAgeExample), ExampleType = ExampleType.Response, ResponseStatusCode = StatusCodes.Status200OK)] // <----- Only shown for 200 responses
[EndpointSpecificExample(typeof(BadRequestExample), ExampleType = ExampleType.Response, ResponseStatusCode = StatusCodes.Status400BadRequest)] // <----- Only shown for 400 responses
public IActionResult GetPersonAge([FromRoute] int id) {...}

public class PersonSpecificAgeExample : IExampleProvider<int>
{
    public int GetExample() => 69;
}

public class BadRequestExample : IExampleProvider<string>
{
    public int GetExample() => "Oops! Invalid id format error.";
}
```

# Set multiple examples for specific endpoint
It is possible to set multiple specific implementations of `IExampleProvider<T>` to desired API endpoint. This may be handy when multiple endpoints return same type or if primitive type is returned but you need to specify specific value.
To set specific example, simply annotate controllers method with `EndpointSpecificExample` attribute and provide type of the example class.

```csharp
[HttpGet("{id}/age")]
[ProducesResponseType(StatusCodes.Status200OK, Type = typeof(int))]
[EndpointSpecificExample(typeof(PersonSpecificAge69Example), typeof(PersonSpecificAge50Example))] // <-----
public IActionResult GetPersonAge([FromRoute] int id) {...}

// Included
public class PersonSpecificAge69Example : IExampleProvider<int>
{
    public int GetExample() => 69;
}

// Included
public class PersonSpecificAge50Example : IExampleProvider<int>
{
    public int GetExample() => 50;
}

// Excluded
public class PersonSpecificAge35Example : IExampleProvider<int>
{
    public int GetExample() => 35;
}
```

# Polymorphism
To define specific examples of abstract classes that are returned or received in controller, simply implement `IExampleProvider<T>` where `T` is the type of abstract class. 
In example below, we have abstract class `Animal` which is returned as collection from `GetAnimals` endpoint. To specify examples for each implementation of `Animal` type, we create classes that implement `IExampleProvider<Animal>`.
```csharp
[HttpGet]
[ProducesResponseType(StatusCodes.Status200OK, Type = typeof(IEnumerable<Animal>))]
public IActionResult GetAnimals() {...}

public abstract class Animal {...}

public class Monkey : Animal
{
    public int PoopsThrown { get; set; }
}

public class Sloth : Animal
{
    public uint YawnsCount { get; set; }
}

[ExampleAnnotation(Name = "Monkey")]
public class MonkeyExample : IExampleProvider<Animal>
{
    public Animal GetExample() => new Monkey { Age = 5, Name = "Harambe", PawnCount = 4, PoopsThrown = 18 };
}

[ExampleAnnotation(Name = "Sloth")]
public class SlothExample : IExampleProvider<Animal>
{
    public Animal GetExample() => new Sloth { Age = 18, Name = "Vence", PawnCount = 4, YawnsCount = 158 };
}
```
Swagger then shows all examples in response body with dropdown:
![Polymorphism single item response body](https://github.com/vaclavnovotny/images/blob/main/polymorphism3.jpg)

Or in case of enumerable in response body:
![Polymorphism in response body](https://github.com/vaclavnovotny/images/blob/main/polymorphism.jpg)

Or in case of request body, Swagger shows dropdown of all defined examples:
![Polymorphism in request body](https://github.com/vaclavnovotny/images/blob/main/polymorphism2.jpg)

_Note:_ Set your JsonSerializer settings [TypeNameHandling](https://www.newtonsoft.com/json/help/html/serializetypenamehandling.htm) to see discriminator. For example:

```csharp
services.AddOpenApiDocument(
    (settings, provider) =>
    {
        settings.SerializerSettings = new JsonSerializerSettings() { TypeNameHandling = TypeNameHandling.Auto };
        settings.AddExamples(provider);
    });
```


# Support
I personally use this NuGet in my projects, so I will keep this repository up-to-date. Any ideas for extending functionalities are welcome, so create an issue with proposal. 

### Did I save you some hours?<!-- omit from toc -->
[![ko-fi](https://www.ko-fi.com/img/githubbutton_sm.svg)](https://ko-fi.com/U7U72G1A2)
 readmeEtag: '"8cba263a77f9875a5583d6efad6f978de2162493"' readmeLastModified: Sat, 18 May 2024 11:11:36 GMT repositoryId: 303455072 description: >- NSwag processor to programmatically define strongly-typed examples for response and request parameters. created: '2020-10-12T16:48:05Z' updated: '2025-10-06T09:08:23Z' language: C# archived: false stars: 19 watchers: 2 forks: 13 owner: vaclavnovotny logo: https://avatars.githubusercontent.com/u/17833494?v=4 license: MIT repoEtag: '"b33e0c5d214f3d2c3c13fc6c3f56afbcb6c2c5e582c4dac461829d45eec45bc1"' repoLastModified: Mon, 06 Oct 2025 09:08:23 GMT foundInMaster: true id: 2baf0af965838620471c5542f55d29e8 - source: openapi3 tags repository: https://github.com/env0/serverless-openapi-typescript v3: true repositoryMetadata: base64Readme: >- # Serverless OpenAPI TypeScript Plugin

Generates [**OpenAPI 3.1.0**](https://github.com/OAI/OpenAPI-Specification/blob/3.1.0/versions/3.1.0.md) documentation from serverless configuration files **including fully generated request/response models from TypeScript**.  

This is an extension plugin to the most excellent [@conqa/serverless-openapi-documentation](https://www.npmjs.com/package/@conqa/serverless-openapi-documentation).  

Partilly works with [ReDoc](https://github.com/Rebilly/ReDoc) as they're still working on OpenAPI 3.1.0 support.  
Works well with [Stoplight.io](https://stoplight.io/)    

> **Why OpenAPI 3.1.0 and not 3.0.0?**  
> Because types generated from TypeScript are complex and require at least json-schema draft07 to be described properly.  
> OpenAPI 3.1.0 is the first version to be fully compatible with JSON Schema - 3.0.0 is just an "extended subset" of draft05

---

- [Usage](#usage)
    - [Options](#options)
    - [Configuration](#configuration)
        - [Models](#models)
        - [Functions](#functions)
        - [`queryParams`](#queryparams)
        - [`pathParams`](#pathparams)
        - [`cookieParams`](#cookieparams)
        - [`requestModels`](#requestmodels)
        - [`methodResponses`](#methodresponses-and-responsemodels)
        - [`webhooks`](#webhooks)
- [Install](#install)

---

## Usage

This plugin requires additional configuration to use, see the "[Configuration](#configuration)" section for how to configure the plugin to generate documentation.

Below are the commandline options to run the generator:

```bash
serverless openapi generate [options]
```

### Options

```bash
Plugin: ServerlessOpenAPIDocumentation
openapi generate  ...................... Generate OpenAPI v3 Documentation
    --output / -o ...................... Output file location [default: openapi.yml|json]
    --format / -f ...................... OpenAPI file format (yml|json) [default: yml]
    --indent / -i ...................... File indentation in spaces [default: 2]
    --help / -h   ...................... Help
```

### Configuration

To configure this plugin to generate valid OpenAPI documentation there are two places you'll need to modify in your `serverless.yml` file, the `custom` variables section and the `http` event section for each given function in your service.

This plugin is compatible with the same documentation configuration structure in [@conqa/serverless-openapi-documentation](https://www.npmjs.com/package/@conqa/serverless-openapi-documentation) and needs to be run beside it.

The `custom` section of your `serverless.yml` can be configured as below:

```yml
custom:
  documentation:
    version: '1'
    title: 'My API'
    description: 'This is my API'
    # https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#securitySchemeObject
    securitySchemes: {}
    # https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#security-requirement-object
    security: {}
    models: {}
    typescriptApiPath: 'api.d.ts'
    tsconfigPath: 'tsconfig.json'
```

These configurations can be quite verbose; you can separate it out into it's own file, such as `serverless.doc.yml` as below:

```yml
custom:
  documentation: ${file(serverless.doc.yml):documentation}

functions:
  myFunc:
    events:
      - http:
          path: getStuff
          method: get
          documentation: ${file(serverless.doc.yml):endpoints.myFunc}
```

For more info on `serverless.yml` syntax, see their docs.

#### Models

These will be generated for you from TypeScript.  
For information on how to specify these yourself, see [Models](https://github.com/conqa/serverless-openapi-documentation#models) on the base plugin page.    

#### Functions

To define the documentation for a given function event, you need to create a `documentation` attribute for your http event in your `serverless.yml` file.  

This plugin will assert the `documentation` attribute is set on every non `private: true` function with an `http` event.  

If you wish a function to not be listed on your documentation, you must explicitly set `documentation: ~` to disable docs on that function.    

The `documentation` section of the event configuration can contain the following attributes:

* `summary`: a short description of the method
* `description`: a detailed description of the method
* `tags`: an array of tags for this event
* `deprecated`: boolean indicator that indicates clients should migrate away from this function
* `requestBody`: contains description of the request
    * `description`: a description of the request body
* `requestModels`: a list of models to describe the request bodies (see [requestModels](#requestmodels) below) - **these will be autogenerated for you from TypeScript**
* `queryParams`: a list of query parameters (see [queryParams](#queryparams) below) - **these _can_ be autogenerated for you from TypeScript**
* `pathParams`: a list of path parameters (see [pathParams](#pathparams) below) - **these _can_ be autogenerated for you from TypeScript**
* `cookieParams`: a list of cookie parameters (see [cookieParams](#cookieparams) below)
* `methodResponses`: an array of response models and applicable status codes (see [methodResponses](#methodresponses-and-responsemodels)) - **these _will_ be autogenerated for you from TypeScript**
* `webhooks`: an object with all the webhooks with descriptions (see [webhooks](#webhooks)) - **these _will_ be autogenerated for you from TypeScript**

```yml
functions:
  createUser:
    handler: "handler.create"
    events:
      - http:
        path: "create"
        method: "post"
        documentation:
          summary: "Create User"
          description: "Creates a user and then sends a generated password email"
          requestBody:
            description: "A user information object"
          requestModels:
            application/json: "PutDocumentRequest"
          pathParams:
            - name: "username"
              description: "The username for a user to create"
              schema:
                type: "string"
                pattern: "^[-a-z0-9_]+$"
          queryParams:
            - name: "membershipType"
              description: "The user's Membership Type"
              schema:
                type: "string"
                enum:
                  - "premium"
                  - "standard"
          cookieParams:
            - name: "SessionId"
              description: "A Session ID variable"
              schema:
                type: "string"
          methodResponses:
            - statusCode: 201
              responseBody:
                description: "A user object along with generated API Keys"
              responseModels:
                application/json: "PutDocumentResponse"
            - statusCode: 500
              responseBody:
                description: "An error message when creating a new user"
              responseModels:
                application/json: "ErrorResponse"
```

#### `queryParams`

Query parameters can be described as follow:

* `name`: the name of the query variable
* `description`: a description of the query variable
* `required`: whether the query parameter is mandatory (boolean)
* `schema`: JSON schema (inline or file) **or a string representing fully qualified TypeScript type to generate schema from**

> Note: If schema is not specified one will be autogenerated for you as `{ "type": "string" }` if you specify `querystrings` in your [Serverless request parameters](https://www.serverless.com/framework/docs/providers/aws/events/apigateway#request-parameters)

##### Explicit schema example
```yml
queryParams:
  - name: "filter"
    description: "The filter parameter"
    required: true
    schema:
      type: "string"
```

##### Autogenerated Schema from Serverless config
```yml
functions:
  foo:
    events:
      - http:
          request:
            parameters:
              querystrings:
                filter: true

# This will be autogenerated for you based on your Serverless config above! No need to write it yourself:                
# queryParams:
#  - name: "filter"
#    required: true 
#    schema:
#      type: "string"
```

##### Generated Schema from TypeScript type
```typescript
export namespace MyApi {
    export namespace FooBar {
        export namespace Request {
            export namespace QueryParams {
                // You may use JSdoc to set JSON Schema attributes! 
                /** @description The filter parameter **/
                export type Filter = 'hello' | 'world';
            }
        }
    }
}
```

```yml
functions:
  fooBar:
    events:
      - http:
          documentation:
            queryParams:
              - name: "filter"
                schema: "MyApi.FooBar.Request.QueryParams.Filter"
          request:
            parameters:
              querystrings:
                filter: false

# This will be autogenerated for you based from TypeScript! No need to write it yourself                
# queryParams:
#  - name: "filter"
#    required: false
#    description: "The filter parameter"  
#    schema:
#      type: string
#      enum:
#        - hello
#        - world
```

#### `pathParams`

Path parameters can be described as follow:

* `name`: the name of the query variable
* `description`: a description of the query variable
* `schema`: JSON schema (inline or file) **or a string representing fully qualified TypeScript type to generate schema from**

##### Explicit schema example

```yml
pathParams:
  - name: "usernameId"
    description: "The usernameId parameter"
    schema:
      type: "string"
```

##### Autogenerated Schema from Serverless config
```yml
functions:
  foo:
    events:
      - http:
          request:
            parameters:
              pahts:
                usernameId: true

# This will be autogenerated for you based on your Serverless config above! No need to write it yourself:                
# queryParams:
#  - name: "filter"
#    required: true 
#    schema:
#      type: "string"
```

##### Generated Schema from TypeScript type
```typescript
export namespace MyApi {
    export namespace FooBar {
        export namespace Request {
            export namespace PathParams {
                // You may use JSdoc to set JSON Schema attributes! 
                /** @description The usernameId parameter **/
                export type UsernameId = string;
            }
        }
    }
}
```

```yml
functions:
  fooBar:
    events:
      - http:
          documentation:
            paths:
              - name: "usernameId"
                schema: "MyApi.FooBar.Request.QueryParams.Filter"
          request:
            parameters:
              paths:
                usernameId: true

# This will be autogenerated for you based from TypeScript! No need to write it yourself                
# queryParams:
#  - name: "filter"
#    required: true 
#    description: "The usernameId parameter"
#    schema:
#      type: string
```

#### `cookieParams`

TypeScript generation for these isn't supported.  
For information on how to specify these yourself, see [Cookie Parameters](https://github.com/conqa/serverless-openapi-documentation#cookieparams) on the base plugin page.

#### `requestModels`

The `requestModels` property allows you to define models for the HTTP Request of the function event.  

These will be autogenerated for you from TypeScript **by convention**.  
The plugin will autogenerate a request model for every `put`, `patch` or `post` method based on the TypeScript type found by convention.    
The look strategy will be by:  
`${serverless.custom.documentation.apiNamespace}.${upper-case-name-of-function}.Request.Body`. Note that any underscores (`_`) or hyphens (`-`) in function names will be excluded in the resolved type (e.g. the Serverless function `my_get-function` will resolve to `MyGetFunction` in Typescript)  

For example, the following function:  
```yaml
functions:
  myFunction: 
    events:
      - http:
        documentation:
          summary: "Does a lot for you"
```

Will look for the resolve type as its request model:  
```typescript
// api.d.ts

export namespace MyApi {
    export namespace MyFunction {
        export namespace Request {
            export type Body = {
                id: string,
                name: string,
                age: number
                // ...
            }
        }
    }
}
```

#### `methodResponses` and `responseModels`

These will be autogenerated for you from TypeScript **by convention**.  
The plugin will autogenerate a response model for every `put`, `patch`, `post` and `get` method based on the TypeScript type found by convention.  
The response will use the generated type as body, and will have a 200 status code.  
`delete` method will be generated with an empty body and 204 status code.  
Only `application/json` response types are supported.  

The look strategy will be by:  
`${serverless.custom.documentation.apiNamespace}.${upper-case-name-of-function}.Response`

For example, the following function:
```yaml
functions:
  myFunction: 
    events:
      - http:
        documentation:
          summary: "Does a lot for you"
```

Will look for the resolve type as its request model:
```typescript
// api.d.ts

export namespace MyApi {
  export namespace MyFunction {
    export type Response = {
      id: string,
      name: string,
      age: number
      // ...
    }
  }
}
```

##### `responseHeaders` and `requestHeaders`

The `responseHeaders/requestHeaders` section of the configuration allows you to define the HTTP headers for the function event.    
TypeScript generation for these isn't supported.    
For information on how to specify these yourself, see [Response Headers](https://github.com/conqa/serverless-openapi-documentation#responseheaders-and-requestheaders) on the base plugin page.  

##### `customTags`

In some cases where a single serverless.yml contains endpoints which can be grouped differently,
you can define custom tags as following:

```yaml
custom:
  documentation:
    title: 'Project'
    description: DummyDescription

    apiNamespace: ProjectApi
    tags:
       - name: FooBarTitle
         description: FooBarDescription
       - name: BazTitle
         description: BazDescription
```

```yaml
functions:
  createFunc:
    handler: handler.create
    events:
      - http:
          documentation:
            summary: "Create Function"
            tag: FooBarTitle
```

Endpoints that are not attached to a custom tag, are still attached to the title ( which is the default tag ).

#### `webhooks`
OpenAPI have an option to add your application `webhooks`, while this feature isn't supported by `serverless`.

For those the plugin will look for the webhooks under `custom.documentation.webhooks`.

For example

```yaml
custom:
  documentation:
    apiNamespace: MyApi
    webhooks:
      WebhookName:
        post:
          requestBody:
            description: |
              This is a request body description
          responses:
            200:
              description: |
                This is a expected response description
```

this will generate the next OpenAPI file

```yaml
components:
  schemas:
    MyApi.Webhooks.WebhookName:
      type: object
webhooks:
  WebhookName:
    post:
      requestBody:
        description: |
          This is a request body description
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/MyApi.Webhooks.WebhookName'
      responses:
        '200':
          description: |
            This is a expected response description
```

With the next `api.d.ts`:

```typescript
export namespace MyApi {
  export namespace Webhooks {
    export type WebhookName = {
      id: string,
      name: string,
      age: number
      // ...
    }
  }
}
```

You can also add Webhook tags by using `webhookTags` keyword it will auto add tag to all webhooks
```yaml
custom:
  documentation:
    apiNamespace: MyApi
    webhookTags:
      - name: ProjectWebhooks
        description: This is the description for ProjectWebhooks
    webhooks:
      WebhookName:
        post:
          requestBody:
            description: |
              This is a request body description
          responses:
            200:
              description: |
                This is a expected response description
```


## Install

This plugin is **an extension**.  
It means you must install and enable `@conqa/serverless-openapi-documentation` too.

To add these plugins to your package.json:

**Using npm:**
```bash
npm install @conqa/serverless-openapi-documentation --save-dev
npm install serverless-openapi-typescript --save-dev
```

**Using Yarn:**
```bash
yarn add @conqa/serverless-openapi-documentation --dev
yarn add serverless-openapi-typescript --dev
```

Next you need to add the plugin to the `plugins` section of your `serverless.yml` file.

```yml
plugins:
  - @conqa/serverless-openapi-documentation
  - serverless-openapi-typescript
  # NOTE: Order is important here - this plugin must be added after the base @conqa/serverless-openapi-documentation plugin
  # This plugin asserts that - you will get a validation message if you get the ordering wrong 
```

> Note: Add this plugin _after_ `serverless-offline` to prevent issues with `String.replaceAll` being overridden incorrectly.

## License

MIT
 readmeEtag: '"805379935366bacdc6beb284e7684d85a00c68c2"' readmeLastModified: Mon, 27 May 2024 12:03:39 GMT repositoryId: 400071702 description: >- Serverless plugin to generate OpenAPI 3.1.0 documentation including models from serverless configuration and TypeScript created: '2021-08-26T06:57:36Z' updated: '2024-05-27T12:03:43Z' language: TypeScript archived: false stars: 14 watchers: 8 forks: 7 owner: env0 logo: https://avatars.githubusercontent.com/u/46656519?v=4 repoEtag: '"11f9e7469e230d9f4d07c8dc839930723b6de6bfedde1243ed7c84a68b1b01ff"' repoLastModified: Mon, 27 May 2024 12:03:43 GMT foundInMaster: true category: - Code Generators - Parsers id: 1f0ebfad51f6d4b3b4f19310e8beaae8 - source: openapi3 tags repository: https://github.com/superfaceai/openapi-linter v3: true id: 86b53702824389b1522bcbb79dcb880b repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIExpbnRlcgoKWyFbR2l0SHViIFdvcmtmbG93IFN0YXR1c10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvd29ya2Zsb3cvc3RhdHVzL3N1cGVyZmFjZWFpL29wZW5hcGktbGludGVyL0NJKV0oaHR0cHM6Ly9naXRodWIuY29tL3N1cGVyZmFjZWFpL29wZW5hcGktbGludGVyL2FjdGlvbnMvd29ya2Zsb3dzL21haW4ueW1sKQpbIVtucG1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3YvQHN1cGVyZmFjZWFpL29wZW5hcGktbGludGVyKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQHN1cGVyZmFjZWFpL29wZW5hcGktbGludGVyKQpbIVtsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9sL0BzdXBlcmZhY2VhaS9vcGVuYXBpLWxpbnRlcildKExJQ0VOU0UpClshW0NMSSBidWlsdCB3aXRoIG9jbGlmXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NsaS1vY2xpZi1icmlnaHRncmVlbi5zdmcpXShodHRwczovL29jbGlmLmlvKQpbIVtHaXRIdWIgRGlzY3Vzc2lvbnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2Rpc2N1c3Npb25zL3N1cGVyZmFjZWFpLy5naXRodWI/bG9nbz1naXRodWImbG9nb0NvbG9yPWZmZildKGh0dHBzOi8vZ2l0aHViLmNvbS9vcmdzL3N1cGVyZmFjZWFpL2Rpc2N1c3Npb25zKQoKPiBJcyB5b3VyIE9wZW5BUEkgU3BlYyByZWFkeSBmb3IgU0RLIGdlbmVyYXRvcnM/CgpPcGVuQVBJIExpbnRlciBpcyBhIENMSSBhbmQgYSBOb2RlLmpzIGxpYnJhcnkgdG8gdmFsaWRhdGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLgpJdCBpcyBiYXNlZCBvbiBbU3BlY3RyYWxdIGJ5IFN0b3BsaWdodCB3aXRoIFtPcGVuQVBJIHJ1bGVzXShodHRwczovL21ldGEuc3RvcGxpZ2h0LmlvL2RvY3Mvc3BlY3RyYWwvNGRlYzI0NDYxZjNhZi1vcGVuLWFwaS1ydWxlcykgd2l0aCBhZGRpdGlvbmFsIHJ1bGVzLiBUaGUgZ29hbCBvZiBMaW50ZXIgaXMgdG8gY2hlY2sgd2hldGhlciB0aGUgc3BlYyBjb250YWlucyBlbm91Z2ggaW5mb3JtYXRpb24gdG8gZ2VuZXJhdGUgaGlnaCBxdWFsaXR5LCB3ZWxsIGRvY3VtZW50ZWQgU0RLLgpXZSB1c2UgT3BlbkFQSSBMaW50ZXIgaW4gW1N1cGVyZmFjZSBJbnRlZ3JhdGlvbiBEZXNpZ25lcl1bZGVzaWduZXJdLCBidXQgYW55IGNsaWVudCBjb2RlIGdlbmVyYXRvciBiZW5lZml0cyBmcm9tIHdlbGwgd3JpdHRlbiBPcGVuQVBJIHNwZWNzLgoKIyMgU2V0dXAKCkluc3RhbGwgZnJvbSBucG0gZ2xvYmFsbHk6CgpgYGBzaGVsbApucG0gaSAtZyBAc3VwZXJmYWNlYWkvb3BlbmFwaS1saW50ZXIKYGBgCgpOb3cgeW91IGNhbiB1c2UgdGhlIGxpbnRlciB3aXRoIGNvbW1hbmRzIGBvcGVuYXBpLWxpbnRlcmAgb3IgYG9hbGAuCgpBbHRlcm5hdGl2ZWx5IHlvdSBjYW4gdXNlIHRoZSBsaW50ZXIgd2l0aG91dCBpbnN0YWxsYXRpb24gd2l0aCBgbnB4YDoKCmBgYApucHggQHN1cGVyZmFjZWFpL29wZW5hcGktbGludGVyIGxpbnQgPGZpbGUgb3IgVVJMPgpgYGAKCiMjIENMSSBjb21tYW5kcwoKICA8IS0tIGNvbW1hbmRzIC0tPgoKLSBbYG9wZW5hcGktbGludGVyIGxpbnQgU1BFQ0lGSUNBVElPTlBBVEhgXSgjb3BlbmFwaS1saW50ZXItbGludC1zcGVjaWZpY2F0aW9ucGF0aCkKCiMjIyBgb3BlbmFwaS1saW50ZXIgbGludCBTUEVDSUZJQ0FUSU9OUEFUSGAKCkxpbnRzIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiB1c2luZyB0aHJlZSBkaWZmZXJlbnQgcGFyc2Vycy92YWxpZGF0b3JzLgoKYGBgClVTQUdFCiAgJCBvcGVuYXBpLWxpbnRlciBsaW50IFtTUEVDSUZJQ0FUSU9OUEFUSF0gWy1mIHlhbWx8anNvbl0gWy1lIGVycm9yfHdhcm5pbmd8YW55XQoKQVJHVU1FTlRTCiAgU1BFQ0lGSUNBVElPTlBBVEggIFBhdGggb3IgVVJMIHRvIHNwZWNpZmljYXRpb24gZmlsZQoKRkxBR1MKICAtZSwgLS10aHJvd09uPTxvcHRpb24+ICAgICBbZGVmYXVsdDogZXJyb3JdIE9uIHdoaWNoIGtpbmQgb2Ygc2V2ZXJ0eSBlcnJvciBzaG91bGQgYmUgdGhyb3duLiBXaGVuIHNldCB0byB3YXJuaW5nCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgY29tbWFuZCB3aWxsIHRocm93IHdoZW4gdGhlcmUgaXMgYSByZXN1bHQgd2l0aCAid2FybmluZyIgb3IgImVycm9yIiBzZXZlcml0eS4KICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8b3B0aW9uczogZXJyb3J8d2FybmluZ3xhbnk+CiAgLWYsIC0tZmlsZUZvcm1hdD08b3B0aW9uPiAgRm9ybWF0IG9mIHNwZWNpZmljYXRpb24uIE11c3QgYmUgInlhbWwiIG9yICJqc29uIiB3aGVuIGRlZmluZWQKICAgICAgICAgICAgICAgICAgICAgICAgICAgICA8b3B0aW9uczogeWFtbHxqc29uPgoKREVTQ1JJUFRJT04KICBMaW50cyBPcGVuQVBJIHNwZWNpZmljYXRpb24gdXNpbmcgdGhyZWUgZGlmZmVyZW50IHBhcnNlcnMvdmFsaWRhdG9ycy4KCkVYQU1QTEVTCiAgJCBvYWwgbGludCBodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1wZXRzdG9yZS9ibG9iL21hc3Rlci9zcmMvbWFpbi9yZXNvdXJjZXMvb3BlbmFwaS55YW1sCgogICQgb2FsIGxpbnQgZXhhbXBsZXMvcGV0c3RvcmUueWFtbAoKICAkIG9hbCBsaW50IGV4YW1wbGVzL3BldHN0b3JlLnlhbWwgLWUgYW55CgogICQgb2FsIGxpbnQgZXhhbXBsZXMvcGV0c3RvcmUueWFtbCAtZiB5YW1sCmBgYAoKPCEtLSBjb21tYW5kc3N0b3AgLS0+CgojIyBVc2FnZSBpbiBjb2RlCgpJbnN0YWxsIHRoZSBsaW50ZXIgYXMgYSBkZXBlbmRlbmN5IGludG8geW91ciBwcm9qZWN0OgoKYGBgc2hlbGwKbnBtIGkgQHN1cGVyZmFjZWFpL29wZW5hcGktbGludGVyCmBgYAoKVXNlIHRoZSBgbGludGAgZnVuY3Rpb246Cgo8IS0tIFRPRE86IEFkZCBleGFtcGxlIHdoYXQncyBpbiBsaW50UmVzdWx0IC0tPgoKYGBgdHMKaW1wb3J0IHsgbGludCB9IGZyb20gJ29wZW5hcGktbGludGVyJzsKCi8vIEdldCBzcGVjaWZpY2F0aW9uIGFzIHN0cmluZwpjb25zdCBzcGVjaWZpY2F0aW9uID0gYXdhaXQgZnMucmVhZEZpbGUocGF0aFRvU3BlYywgeyBlbmNvZGluZzogJ3V0Zi04JyB9KTsKCi8vIFBhc3Mgc3BlY2lmaWNhdGlvbiwgaXRzIGV4dGVuc2lvbiAoInlhbWwiIG9yICJqc29uIikgYW5kIG5hbWUKY29uc3QgbGludFJlc3VsdCA9IGF3YWl0IGxpbnQoc3BlY2lmaWNhdGlvbiwgJ3lhbWwnLCAnbXktc3BlYy1uYW1lJyk7CgovL0RvIHNvbWV0aGluZyB3aXRoIHJlc3VsdApjb25zb2xlLmxvZyhsaW50UmVzdWx0KTsKYGBgCgo8IS0tIFRPRE8KIyMgQWRkZWQgcnVsZXMgKFdJUCkKCiogRWFjaCBvcGVyYXRpb24gaGFzIHRvIGRlZmluZSBhdCBsZWFzdCBvbmUgc3VjY2VzcyByZXNwb25zZQoqIEVhY2ggb3BlcmF0aW9uIGhhcyB0byBkZWZpbmUgYXQgbGVhc3Qgb25lIGVycm9yIG9yIGRlZmF1bHQgcmVzcG9uc2UKLS0+CgojIyBSZWxhdGVkIHByb2plY3RzCgotIFtTcGVjdHJhbF0sIGdlbmVyaWMgbGludGVyIGZvciBKU09OIGFuZCBZQU1MIHRoaXMgcHJvamVjdCBpcyBiYXNlZCBvbgotIFtPcGVuQVBJIEVuZm9yY2VyXShodHRwczovL29wZW5hcGktZW5mb3JjZXIuY29tLykKLSBbT3BlbkFQSSBWYWxpZGF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9JQk0vb3BlbmFwaS12YWxpZGF0b3IpCi0gW09BUyBUb29sc10oaHR0cHM6Ly9vYXMtdG9vbHMuZ2l0aHViLmlvLykKLSBbUmVkb2NseSBDTEldKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jbHktY2xpKQotIFtDaGVycnlib21iXShodHRwczovL2dpdGh1Yi5jb20vYmxzdC1zZWN1cml0eS9jaGVycnlib21iKSAoUnVzdCkKLSBhbmQgW21hbnkgbW9yZV0oaHR0cHM6Ly9vcGVuYXBpLnRvb2xzLykKCiMjIE1haW50YWluZXJzCgotIFtKYWt1YiBWYWNla10oaHR0cHM6Ly9naXRodWIuY29tL0pha3ViLVZhY2VrKQotIFtSYWRlayBLeXNlbMO9XShodHRwczovL2dpdGh1Yi5jb20va3lzZWx5KQoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBbTUlUIGxpY2Vuc2VdKExJQ0VOU0UpLgoKwqkgMjAyMyBTdXBlcmZhY2Ugcy5yLm8uCgpbc3BlY3RyYWxdOiBodHRwczovL3N0b3BsaWdodC5pby9vcGVuLXNvdXJjZS9zcGVjdHJhbApbZGVzaWduZXJdOiBodHRwczovL3N1cGVyZmFjZS5haS9kZXNpZ25lcgo= readmeEtag: '"dcc032fec32455a2fa8825288981763ef91c0e33"' readmeLastModified: Wed, 19 Jul 2023 11:32:46 GMT repositoryId: 530604809 description: Is your OpenAPI Spec ready for SDK generators? created: '2022-08-30T10:24:31Z' updated: '2025-05-13T07:44:23Z' language: TypeScript archived: false stars: 15 watchers: 5 forks: 0 owner: superfaceai logo: https://avatars.githubusercontent.com/u/57067369?v=4 license: MIT repoEtag: '"543970980a5b0907b8638a470ff840dc8d9fb04a204a6dcff9b4297cfba77800"' repoLastModified: Tue, 13 May 2025 07:44:23 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/ndsev/zswag v3: true id: 9ce9cc067be88c2fb444f0cda133e0ff repositoryMetadata: base64Readme: >- # Zswag

[![CI](https://github.com/ndsev/zswag/actions/workflows/build-deploy.yml/badge.svg)](https://github.com/ndsev/zswag/actions/workflows/build-deploy.yml)
[![codecov](https://codecov.io/github/ndsev/zswag/graph/badge.svg?token=5DTX2M8IDE)](https://codecov.io/github/ndsev/zswag)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=ndsev_zswag&metric=alert_status)](https://sonarcloud.io/summary/new_code?id=ndsev_zswag)
[![Release](https://img.shields.io/github/release/ndsev/zswag)](https://GitHub.com/ndsev/zswag/releases/)
[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=ndsev_zswag&metric=coverage)](https://sonarcloud.io/summary/new_code?id=ndsev_zswag)
[![License](https://img.shields.io/github/license/ndsev/zswag)](https://github.com/ndsev/zswag/blob/master/LICENSE)

zswag is a set of libraries for using/hosting zserio services through OpenAPI.

**Table of Contents:**

  * [Components](#components)
  * [Setup](#setup)
    + [For Python Users](#for-python-users)
    + [For C++ Users](#for-c-users)
      - [Offline/Disconnected Builds](#offlinedisconnected-builds)
  * [CI/CD and Release Process](#cicd-and-release-process)
    + [Continuous Integration](#continuous-integration)
    + [Release Process](#release-process)
    + [Development Snapshots](#development-snapshots)
    + [Version Validation](#version-validation)
  * [OpenAPI Generator CLI](#openapi-generator-cli)
    + [Generator Usage Example](#generator-usage-example)
    + [Documentation extraction](#documentation-extraction)
  * [Server Component (Python)](#server-component)
  * [Using the Python Client](#using-the-python-client)
  * [C++ Client](#c-client)
  * [Client Environment Settings](#client-environment-settings)
  * [HTTP Proxies and Authentication](#persistent-http-headers-proxy-cookie-and-authentication)
  * [Swagger User Interface](#swagger-user-interface)
  * [Client Result Code Handling](#client-result-code-handling)
  * [OpenAPI Options Interoperability](#openapi-options-interoperability)
    + [HTTP method](#http-method)
    + [Request Body](#request-body)
    + [URL Blob Parameter](#url-blob-parameter)
    + [URL Scalar Parameter](#url-scalar-parameter)
    + [URL Array Parameter](#url-array-parameter)
    + [URL Compound Parameter](#url-compound-parameter)
    + [Server URL Base Path](#server-url-base-path)
    + [Authentication Schemes](#authentication-schemes)

## Components

The zswag repository contains two main libraries which provide
OpenAPI layers for zserio Python and C++ clients. For Python, there
is even a generic zserio OpenAPI server layer.

The following UML diagram provides a more in-depth overview:

![Component Overview](doc/zswag-architecture.png)

Here are some brief descriptions of the main components:

* `zswagcl` is a C++ Library which exposes the zserio OpenAPI service client `OAClient`
  as well as the more generic `OpenApiClient` and `OpenApiConfig` classes.
  The latter two are reused for the Python client library.
* `zswag` is a Python Library which provides both a zserio Python service client
  (`OAClient`) as well as a zserio-OpenAPI server layer based on Flask/Connexion
  (`OAServer`). It also contains the command-line tool `zswag.gen`, which can be
  used to generate an OpenAPI specification from a zserio Python service class.
* `pyzswagcl` is a binding library which exposes the C++-based OpenApi
  parsing/request functionality to Python. **Please consider it "internal".**
* `httpcl` is a wrapper around [cpp-httplib](https://github.com/yhirose/cpp-httplib),
  HTTP request configuration and OS secret storage abilities based on
  the [keychain](https://github.com/hrantzsch/keychain) library.
  
## Setup

### For Python Users

Simply run `pip install zswag`. **Note: This only works with ...**

* 64-bit Python 3.10-3.13, `pip --version` >= 19.3
* Supported platforms: Linux (x86_64), macOS (x86_64 and arm64), Windows (x64)

**Notes:**
* On Windows, make sure that you have the *Microsoft Visual C++ Redistributable Binaries* installed. You can find the x64 installer here: https://aka.ms/vs/16/release/vc_redist.x64.exe
* zswag for Python 3.10 is not supported on Apple Silicon (arm64) because no compatible GitHub Actions runner is available.
  However, this is typically not an issue, as macOS includes more recent Python versions by default.

### For C++ Users

Using CMake, you can ...

* 🌟run tests.
* 🌟build the zswag wheels for a custom Python version.
* 🌟[integrate the C++ client into a C++ project.](#c-client)

Dependencies are managed via CMake's `FetchContent` mechanism. Make sure you have a recent version of CMake (>= 3.22.3) installed.

The basic setup follows the usual CMake configure/build steps:
```bash
mkdir build && cd build
cmake ..
cmake --build .
```

**Note:** The Python environment used for configuration will be used
to build the resulting wheels. After building, you will find the Python
wheels under `build/bin/wheel`.

**To run tests**, just execute CTest at the top of the build directory:
```bash
cd build && ctest --verbose
```

#### Offline/Disconnected Builds

For environments without internet access or for reproducible builds, zswag supports offline builds using CMake's FetchContent mechanism.

**Offline Build Process**

For offline builds, you can pre-fetch all dependencies while online and then build without network access:

```bash
# First, fetch all dependencies while online
mkdir build && cd build
cmake -DFETCHCONTENT_FULLY_DISCONNECTED=OFF ..
# This will download all dependencies

# Then build offline
cmake -DFETCHCONTENT_FULLY_DISCONNECTED=ON ..
cmake --build .
```

The `FETCHCONTENT_FULLY_DISCONNECTED=ON` option tells CMake to use only the pre-fetched dependencies and never attempt network access.

**Local Development with Custom Dependencies**

For development, you can override specific dependencies with local sources:
```bash
mkdir build && cd build
cmake -DFETCHCONTENT_SOURCE_DIR_SPDLOG=/path/to/local/spdlog ..
cmake --build .
```

Available override variables:
- `FETCHCONTENT_SOURCE_DIR_ZLIB` - zlib compression library
- `FETCHCONTENT_SOURCE_DIR_SPDLOG` - spdlog logging library  
- `FETCHCONTENT_SOURCE_DIR_YAML_CPP` - yaml-cpp parsing library
- `FETCHCONTENT_SOURCE_DIR_STX` - stx utility library
- `FETCHCONTENT_SOURCE_DIR_SPEEDYJ` - speedyj JSON library
- `FETCHCONTENT_SOURCE_DIR_HTTPLIB` - cpp-httplib HTTP library
- `FETCHCONTENT_SOURCE_DIR_OPENSSL` - OpenSSL cryptography library
- `FETCHCONTENT_SOURCE_DIR_PYBIND11` - pybind11 (when `ZSWAG_BUILD_WHEELS=ON`)
- `FETCHCONTENT_SOURCE_DIR_PYTHON_CMAKE_WHEEL` - python-cmake-wheel (when `ZSWAG_BUILD_WHEELS=ON`)
- `FETCHCONTENT_SOURCE_DIR_ZSERIO_CMAKE_HELPER` - zserio build helpers
- `FETCHCONTENT_SOURCE_DIR_KEYCHAIN` - keychain library (when `ZSWAG_KEYCHAIN_SUPPORT=ON`)
- `FETCHCONTENT_SOURCE_DIR_CATCH2` - Catch2 testing framework (when `ZSWAG_ENABLE_TESTING=ON`)

**Build Options**

Common build configuration options:
```bash
# Minimal build (no wheels, no keychain, no tests)
cmake -DZSWAG_BUILD_WHEELS=OFF -DZSWAG_KEYCHAIN_SUPPORT=OFF -DZSWAG_ENABLE_TESTING=OFF ..

# Offline build with custom spdlog
cmake -DFETCHCONTENT_FULLY_DISCONNECTED=ON -DFETCHCONTENT_SOURCE_DIR_SPDLOG=/path/to/spdlog ..

# Development build with wheels enabled
cmake -DZSWAG_BUILD_WHEELS=ON -DZSWAG_ENABLE_TESTING=ON ..
```

#### Code Coverage

[![codecov](https://codecov.io/gh/ndsev/zswag/branch/master/graph/badge.svg)](https://codecov.io/gh/ndsev/zswag)

zswag includes comprehensive C++ test coverage analysis for the `httpcl` and `zswagcl` libraries. Coverage is automatically collected in CI and reported to [Codecov](https://codecov.io/gh/ndsev/zswag).

**📊 [View HTML Coverage Report](https://ndsev.github.io/zswag/coverage/)** - Browsable coverage report hosted on GitHub Pages

**Local Coverage Analysis**

To generate coverage reports locally, you'll need:
- GCC or Clang compiler
- `lcov` and `genhtml` tools (install with `sudo apt-get install lcov` on Ubuntu/Debian)

Build with coverage enabled:
```bash
mkdir build && cd build
cmake -DCMAKE_BUILD_TYPE=Debug \
      -DZSWAG_ENABLE_COVERAGE=ON \
      -DZSWAG_ENABLE_TESTING=ON \
      -DZSWAG_BUILD_WHEELS=OFF \
      -DZSWAG_KEYCHAIN_SUPPORT=OFF ..
cmake --build .
```

**Note:** The flags `-DZSWAG_BUILD_WHEELS=OFF` and `-DZSWAG_KEYCHAIN_SUPPORT=OFF` disable features that aren't needed for coverage analysis and avoid requiring Python development headers or system keychain libraries.

Generate coverage reports:
```bash
# Run tests to generate coverage data
ctest --output-on-failure

# Generate HTML coverage report
cmake --build . --target coverage-report
```

The HTML coverage report will be available at `build/coverage/html/index.html`.

**Available Coverage Targets:**
- `coverage-clean` - Remove all coverage data
- `coverage-report` - Generate coverage report from existing test runs
- `coverage` - Clean, run tests, and generate report (all-in-one)

**Coverage Configuration:**
- **Goal:** 70%+ line coverage (initial), near 100% line and branch coverage (ultimate)
- **Scope:** Coverage is tracked only for library source files (`libs/httpcl`, `libs/zswagcl`)
- **Not included:** Test code, dependencies, generated code (zserio)

**Troubleshooting Coverage Builds:**

If you get "gcov not found" warnings:
```bash
# Check if versioned gcov exists
which gcov-13 gcov-12 gcov-11

# Create symlink (example for gcov-13)
sudo ln -s /usr/bin/gcov-13 /usr/bin/gcov

# Or install gcc package
sudo apt-get install gcc
```

If you want to build coverage **with** wheel support (requires Python development headers):
```bash
cmake -DCMAKE_BUILD_TYPE=Debug \
      -DZSWAG_ENABLE_COVERAGE=ON \
      -DZSWAG_ENABLE_TESTING=ON \
      -DZSWAG_BUILD_WHEELS=ON \
      -DZSWAG_KEYCHAIN_SUPPORT=OFF ..
```
Note: This requires `python3-dev` package (Ubuntu/Debian) or equivalent on your system.

## CI/CD and Release Process

### Continuous Integration

The project uses GitHub Actions for automated building and deployment:

- **Platforms**: Linux (x86_64), macOS (Intel x86_64 and Apple Silicon arm64), Windows (x64)
- **Python versions**: 3.10, 3.11, 3.12, 3.13
- **Triggers**: Pull requests, pushes to main branch, and version tags

### Release Process

Releases are automated through the CI/CD pipeline:

1. **Update version**: Modify `ZSWAG_VERSION` in `CMakeLists.txt`
2. **Create release tag**: Tag the commit with `v{version}` (e.g., `v1.7.2`)
3. **Automatic deployment**: The CI pipeline will:
   - Validate that the tag version matches the CMake version
   - Build wheels for all supported platforms
   - Deploy to PyPI automatically

### Development Snapshots

Pushes to the main branch automatically create development releases:
- Version format: `{base_version}.dev{commit_count}` (e.g., `1.7.2.dev3`)
- Automatically deployed to PyPI for testing

### Version Validation

The build process ensures version consistency:
- Git tags must match the version in `CMakeLists.txt`
- Mismatched versions will cause the build to fail
- This prevents accidental deployment of incorrect versions

## OpenAPI Generator CLI

After installing `zswag` via pip as [described above](#for-python-users),
you can run `python -m zswag.gen`, a CLI to generate OpenAPI YAML files.
The CLI offers the following options

```
usage: Zserio OpenApi YAML Generator [-h] -s service-identifier -i
                                     zserio-or-python-path
                                     [-r zserio-src-root-dir]
                                     [-p top-level-package] [-c tags [tags ...]]
                                     [-o output] [-b BASE_CONFIG_YAML]

optional arguments:
  -h, --help
        show this help message and exit
  -s service-identifier, --service service-identifier

        Fully qualified zserio service identifier.

        Example:
            -s my.package.ServiceClass

  -i zserio-or-python-path, --input zserio-or-python-path

        Can be either ...
        (A) Path to a zserio .zs file. Must be either a top-
            level entrypoint (e.g. all.zs), or a subpackage
            (e.g. services/myservice.zs) in conjunction with
            a "--zserio-source-root|-r <dir>" argument.
        (B) Path to parent dir of a zserio Python package.

        Examples:
            -i path/to/schema/main.zs         (A)
            -i path/to/python/package/parent  (B)

  -r zserio-src-root-dir, --zserio-source-root zserio-src-root-dir

        When -i specifies a zs file (Option A), indicate the
        directory for the zserio -src directory argument. If
        not specified, the parent directory of the zs file
        will be used.

  -p top-level-package, --package top-level-package

        When -i specifies a zs file (Option A), indicate
        that a specific top-level zserio package name
        should be used.

        Examples:
            -p zserio_pkg_name

  -c tags [tags ...], --config tags [tags ...]

        Configuration tags for a specific or all methods.
        The argument syntax follows this pattern:

           [(service-method-name):](comma-separated-tags)

        Note: The -c argument may be applied multiple times.
        The `comma-separated-tags` must be a list of tags
        which indicate OpenApi method generator preferences.
        The following tags are supported:

        get|put|post|delete : HTTP method tags
                query|path| : Parameter location tags
                header|body
                  flat|blob : Flatten request object,
                              or pass it as whole blob.
          (param-specifier) : Specify parameter name, format
                              and location for a specific
                              request-part. See below.
            security=(name) : Set a particular security
                              scheme to be used. The scheme
                              details must be provided through
                              the --base-config-yaml.
         path=(method-path) : Set a particular method path.
                              May contain placeholders for
                              path params.

        A (param-specifier) tag has the following schema:

            (field?name=...
                  &in=[path|body|query|header]
                  &format=[binary|base64|hex]
                  [&style=...]
                  [&explode=...])

        Examples:

          Expose all methods as POST, but `getLayerByTileId`
          as GET with flat path-parameters:

            `-c post getLayerByTileId:get,flat,path`

          For myMethod, put the whole request blob into the a
          query "data" parameter as base64:

            `-c myMethod:*?name=data&in=query&format=base64`

          For myMethod, set the "AwesomeAuth" auth scheme:

            `-c myMethod:security=AwesomeAuth`

          For myMethod, provide the path and place myField
          explicitely in a path placeholder:

            `-c 'myMethod:path=/my-method/{param},...
                 myField?name=param&in=path&format=string'`

        Note:
            * The HTTP-method defaults to `post`.
            * The parameter 'in' defaults to `query` for
              `get`, `body` otherwise.
            * If a method uses a parameter specifier, the
              `flat`, `body`, `query`, `path`, `header` and
              `body`-tags are ignored.
            * The `flat` tag is only meaningful in conjunction
              with `query` or `path`.
            * An unspecific tag list (no service-method-name)
              affects the defaults only for following, not
              preceding specialized tag assignments.

  -o output, --output output

        Output file path. If not specified, the output will be
        written to stdout.

  -b BASE_CONFIG_YAML, --base-config-yaml BASE_CONFIG_YAML

        Base configuration file. Can be used to fully or partially
        substitute --config arguments, and to provide additional
        OpenAPI information. The YAML file must look like this:

          method: # Optional method tags dictionary
            <method-name|*>: <list of config tags>
          securitySchemes: ... # Optional OpenAPI securitySchemes
          info: ...            # Optional OpenAPI info section
          servers: ...         # Optional OpenAPI servers section
          security: ...        # Optional OpenAPI global security
```

### Generator Usage example

Let's consider the following zserio service saved under `myapp/services.zs`:

```
package services;

struct Request {
  int32 value;
};

struct Response {
  int32 value;
};

service MyService {
  Response myApi(Request);
};
```

An OpenAPI file `api.yaml` for `MyService` can now be
created with the following `zswag.gen` invocation:

```bash
cd myapp
python -m zswag.gen -s services.MyService -i services.zs -o api.yaml
```

You can further customize the generation using `-c` configuration
arguments. For example, `-c get,flat,path` will recursively "flatten"
the zserio request object into it's compound scalar fields using
[x-zserio-request-part](#url-scalar-parameter) for all methods.
If you want to change OpenAPI parameters only for one particular
method, you can prefix the tag config argument with the method
name (`-c methodName:tags...`).

### Documentation Extraction

When invoking `zswag.gen` with `-i zserio-file` an attempt
will be made to populate the service/method/request/response
descriptions with doc-strings that are extracted from the zserio
sources.

For structs and services, the documentation is expected to be
enclosed by `/*! .... !*/` markers preceding the declaration:

```C
/*!
### My Markdown Struct Doc
I choose to __highlight__ this word.
!*/

struct MyStruct {
    ...
};
```

For service methods, a single-line doc-string is parsed which
immediately precedes the declaration:

```C
/** This method is documented. */
ReturnType myMethod(ArgumentType);
```

## Server Component

The `OAServer` component gives you the power to marry a zserio-generated app
server class with a user-written app controller and a fitting OpenAPI specification.
It is based on [Flask](https://flask.palletsprojects.com/en/1.1.x/) and
[Connexion](https://connexion.readthedocs.io/en/latest/).

**Implementation choice regarding HTTP response codes:** The server as implemented
here will return HTTP code `400` (Bad Request) when the user request could not
be parsed, and `500` (Internal Server Error) when a different exception occurred while
generating the response/running the user's controller implementation.

### Integration Example

We consider the same `myapp` directory with a `services.zs` zserio file
as already used in the [OpenAPI Generator Example](#generator-usage-example).

**Note:**

* `myapp` must be available as a module (it must be
possible to `import myapp`). 
* We recommend to run the zserio Python generator invocation
  inside the `myapp` module's `__init__.py`, like this:

```py
import zserio
from os.path import dirname, abspath

working_dir = dirname(abspath(__file__))
zserio.generate(
  zs_dir=working_dir,
  main_zs_file="services.zs",
  gen_dir=working_dir)
```

A server script like `myapp/server.py` might then look as follows:

```py
import zswag
import myapp.controller as controller
from myapp import working_dir

# This import only works after zserio generation.
import services.api as services

app = zswag.OAServer(
  controller_module=controller,
  service_type=services.MyService.Service,
  yaml_path=working_dir+"/api.yaml",
  zs_pkg_path=working_dir)

if __name__ == "__main__":
    app.run()
```

The server script above references two important components:
* An **OpenAPI file** (`myapp/api.yaml`): Upon startup, `OAServer`
  will output an error message if this file does not exist. The
  error message already contains the correct command to
  invoke the [OpenAPI Generator CLI](#openapi-generator-cli)
  to generate `myapp/api.yaml`.
* A **controller module** (`myapp/controller.py`): This file provides
  the actual implementations for your service endpoints.
  
For the current example, `controller.py` might look as follows:

```py
import services.api as services

# Written by you
def my_api(request: services.Request):
    return services.Response(request.value * 42)
```

## Using the Python Client

The generic Python client talks to any zserio service that is running
via HTTP/REST, and provides an OpenAPI specification of it's interface.

### Integration Example

As an example, consider a Python module called `myapp` which has the
same `myapp/__init__.py` and `myapp/services.zs` zserio definition as
[previously mentioned](#generator-usage-example). We consider
that the server is providing its OpenAPI spec under `localhost:5000/openapi.json`.

In this setting, a client `myapp/client.py` might look as follows:

```python
from zswag import OAClient
import services.api as services

openapi_url = "http://localhost:5000/openapi.json"

# The client reads per-method HTTP details from the OpenAPI URL.
# You can also pass a local file by setting the `is_local_file` argument
# of the OAClient constructor.
client = services.MyService.Client(OAClient(openapi_url))

# This will trigger an HTTP request under the hood.
client.my_api(services.Request(1))
```

As you can see, an instance of `OAClient` is passed into the constructor
for zserio to use as the service client's transport implementation.

**Note:** While connecting, the client will also use ...
1. [Persistent HTTP configuration](#persistent-http-headers-proxy-cookie-and-authentication).
2. Additional HTTP query/header/cookie/proxy/basic-auth configs passed
   into the `OAClient` constructor using an instance of `zswag.HTTPConfig`.
   For example:
   
   ```python
   from zswag import OAClient, HTTPConfig
   import services.api as services
   config = HTTPConfig() \
       .header(key="X-My-Header", val="value") \  # Can be specified 
       .cookie(key="MyCookie", val="value")    \  # multiple times.
       .query(key="MyCookie", val="value")     \  # 
       .proxy(host="localhost", port=5050, user="john", pw="doe") \
       .basic_auth(user="john", pw="doe") \
       .bearer("bearer-token") \
       .api_key("token")
   
   client = services.MyService.Client(
       OAClient("http://localhost:8080/openapi.", config=config))
   
   # Alternative when specifying api-key or bearer
   client = services.MyService.Client(
       OAClient("http://localhost:8080/openapi.", api_key="token", bearer="token"))
   ```
   
   **Note:** The additional `config` will only enrich, not overwrite the
   default persistent configuration. If you would like to prevent persistent
   config from being considered at all, set `HTTP_SETTINGS_FILE` to empty,
   e.g. via `os.environ['HTTP_SETTINGS_FILE']=''`

## C++ Client

The generic C++ client talks to any zserio service that is running
via HTTP/REST, and provides an OpenAPI specification of its interface.
When using the C++ `OAClient` with your zserio schema, make sure
that the flags [`-withTypeInfoCode` and `-withReflectionCode`](http://zserio.org/doc/ZserioUserGuide.html#zserio-command-line-interface) are passed to the zserio C++ emitter.

### Integration Example

As an example, we consider the `myapp` directory which contains a `services.zs`
zserio definition as [previously mentioned](#generator-usage-example).

We assume that zswag is added to `myapp` as a [Git submodule](https://git-scm.com/book/en/v2/Git-Tools-Submodules)
under `myapp/zswag`.

Next to `myapp/services.zs`, we place a `myapp/CMakeLists.txt` which describes our project:

```cmake
project(myapp)

# If you are not interested in building zswag Python
# wheels, you can set the following option:
# set(ZSWAG_BUILD_WHEELS OFF)

# If your compilation environment does not provide
# libsecret, the following switch will disable keychain integration:
# set(ZSWAG_KEYCHAIN_SUPPORT OFF)

# Optional: For offline/disconnected builds, you can
# predefine dependency sources using FETCHCONTENT_SOURCE_DIR_*
# variables (see README offline builds section for details)

# This is how C++ will know about the zswag lib
# and its dependencies, such as zserio.
if (NOT TARGET zswag)
        FetchContent_Declare(zswag
                GIT_REPOSITORY "https://github.com/ndsev/zswag.git"
                GIT_TAG        "v1.6.7"
                GIT_SHALLOW    ON)
        FetchContent_MakeAvailable(zswag)
endif()

find_package(OpenSSL CONFIG REQUIRED)
target_link_libraries(httplib INTERFACE OpenSSL::SSL)

# This command is provided by zswag to easily create
# a CMake C++ reflection library from zserio code.
add_zserio_library(${PROJECT_NAME}-zserio-cpp
  WITH_REFLECTION
  ROOT "${CMAKE_CURRENT_SOURCE_DIR}"
  ENTRY services.zs
  TOP_LEVEL_PKG myapp_services)

# We create a myapp client executable which links to
# the generated zserio C++ library and the zswag client
# library.
add_executable(${PROJECT_NAME} client.cpp)

# Make sure to link to the `zswagcl` target
target_link_libraries(${PROJECT_NAME}
    ${PROJECT_NAME}-zserio-cpp zswagcl)
```

**Note:** OpenSSL is assumed to be installed or built using the `lib` (not `lib64`) directory name.

The `add_executable` command above references the file `myapp/client.cpp`,
which contains the code to actually use the zswag C++ client.

```cpp
#include "zswagcl/oaclient.hpp"
#include <iostream>
#include "myapp_services/services/MyService.h"

using namespace zswagcl;
using namespace httpcl;
namespace MyService = myapp_services::services::MyService;

int main (int argc, char* argv[])
{
    // Assume that the server provides its OpenAPI definition here
    auto openApiUrl = "http://localhost:5000/openapi.json";
    
    // Create an HTTP client to be used by our OpenAPI client
    auto httpClient = std::make_unique<HttpLibHttpClient>();
    
    // Fetch the OpenAPI configuration using the HTTP client
    auto openApiConfig = fetchOpenAPIConfig(openApiUrl, *httpClient);
    
    // Create a Zserio reflection-based OpenAPI client that
    // uses the OpenAPI configuration we just retrieved.
    auto openApiClient = OAClient(openApiConfig, std::move(httpClient));
        
    // Create a MyService client based on the OpenApi-Client
    // implementation of the zserio::IServiceClient interface.
    auto myServiceClient = MyService::Client(openApiClient);
    
    // Create the request object
    auto request = myapp_services::services::Request(2);

    // Invoke the REST endpoint. Mind that your method-
    // name from the schema is appended with a "...Method" suffix.
    auto response = myServiceClient.myApiMethod(request);
    
    // Print the response
    std::cout << "Got " << response.getValue() << std::endl;
}
```

**Note:** While connecting, `HttpLibHttpClient` will also use ...
1. [Persistent HTTP configuration](#persistent-http-headers-proxy-cookie-and-authentication).
2. Additional HTTP query/header/cookie/proxy/basic-auth configs passed
   into the `OAClient` constructor using an instance of `httpcl::Config`.
   You can include this class via `#include "httpcl/http-settings.hpp"`.
   The additional `Config` will only enrich, not overwrite the
   default persistent configuration. If you would like to prevent persistent
   config from being considered at all, set `HTTP_SETTINGS_FILE` to empty,
   e.g. via `setenv`.

## Client Environment Settings

Both the Python and C++ Clients can be configured using the following
environment variables:

<!-- --8<-- [start:env] -->

| Variable Name | Details   |
| ------------- | --------- |
| `HTTP_SETTINGS_FILE` | Path to settings file for HTTP proxies and authentication, see [next section](#persistent-http-headers-proxy-cookie-and-authentication) |
| `HTTP_LOG_LEVEL` | Verbosity level for console/log output. Set to `debug` for detailed output. |
| `HTTP_LOG_FILE` | Logfile-path (including filename) to redirect console output. The log will rotate with three files (`HTTP_LOG_FILE`, `HTTP_LOG_FILE-1`, `HTTP_LOG_FILE-2`). |
| `HTTP_LOG_FILE_MAXSIZE` | Maximum size of the logfile, in bytes. Defaults to 1GB. |
| `HTTP_TIMEOUT` | Timeout for HTTP requests (connection+transfer) in seconds. Defaults to 60s. |
| `HTTP_SSL_STRICT` | Set to any nonempty value for strict SSL certificate validation. |

<!-- --8<-- [end:env] -->

## Persistent HTTP Headers, Proxy, Cookie and Authentication

Both the Python `OAClient` and C++ `HttpLibHttpClient` read a YAML file
stored under a path which is given by the `HTTP_SETTINGS_FILE` environment
variable.

<!-- --8<-- [start:settings] -->

### HTTP Settings File Format 

The YAML file contains a list of HTTP-related configs that are
applied to HTTP requests based on a regular expression which is matched
against the requested URL.

For example, the following entry would match all requests due to the `*`
url-match-pattern for the `scope` field:

```yaml
http-settings:
  # Under http-settings, a list of settings is defined for specific URL scopes.
  - scope: *     # URL scope - e.g. https://*.nds.live/* or *.google.com.
    basic-auth:  # Basic auth credentials for matching requests.
      user: johndoe
      keychain: keychain-service-string
      password: cleartext-password  # alternative to keychain
    proxy:      # Proxy settings for matching requests.
      host: localhost
      port: 8888
      user: test
      keychain: ...
      password: cleartext-password  # alternative to keychain
    cookies:    # Additional Cookies for matching requests.
      key: value
    headers:    # Additional Headers for matching requests.
      key: value
    query:      # Additional Query parameters for matching requests.
      key: value
    api-key: value  # API Key as required by OpenAPI config - see description below.
    oauth2:
      # REQUIRED fields
      clientId: my-client-id                               # REQUIRED: OAuth2 client identifier

      # REQUIRED if useForSpecFetch=true (default), OPTIONAL otherwise
      tokenUrl: https://issuer.example.com/oauth/token     # Token endpoint URL (see precedence rules below)

      # Client secret (choose one method)
      clientSecretKeychain: keychain-service-string        # RECOMMENDED: Load secret from OS keychain
      clientSecret: cleartext-secret                       # DISCOURAGED: Cleartext secret (use keychain instead)

      # OPTIONAL fields (with defaults/precedence)
      refreshUrl: https://issuer.example.com/oauth/token   # Optional override; defaults to refreshUrl from OpenAPI, then tokenUrl
      audience: https://api.example.com/                   # Optional: audience parameter (required by some providers)
      scope: ["orders.read", ...]                          # Optional: scope override; defaults to OpenAPI spec's per-operation scopes
      useForSpecFetch: true                                # Optional: acquire token before fetching OpenAPI spec (default: true)
      tokenEndpointAuth:                                   # Optional: token endpoint authentication method
        method: rfc6749-client-secret-basic                # Options: rfc6749-client-secret-basic (default), rfc5849-oauth1-signature
        nonceLength: 16                                    # For rfc5849-oauth1-signature: nonce length (8-64, default: 16)
```

**Note:** For `proxy` configs, the credentials are optional.

### OAuth2 Configuration: Required vs Optional Fields

**Important:** Zswag only supports the **OAuth2 `clientCredentials` flow**. Other flows (`authorizationCode`, `implicit`, `password`) are not supported.

**Field Requirements:**

| Field | Required? | Notes |
|-------|-----------|-------|
| `clientId` | ✅ Always | OAuth2 client identifier |
| `tokenUrl` | ⚠️ Conditional | **REQUIRED** when `useForSpecFetch: true` (default)<br>**OPTIONAL** when `useForSpecFetch: false` (defaults to OpenAPI spec) |
| `clientSecret` / `clientSecretKeychain` | ⚠️ Conditional | **REQUIRED** for confidential clients<br>**OPTIONAL** for public clients (if omitted, client_id is sent in request body) |
| `refreshUrl` | ❌ Optional | Defaults to `refreshUrl` from OpenAPI spec, then `tokenUrl` |
| `scope` | ❌ Optional | Defaults to scopes from OpenAPI spec's per-operation `security` requirements |
| `audience` | ❌ Optional | Only required by some OAuth2 providers |
| `useForSpecFetch` | ❌ Optional | Default: `true` (acquire token before fetching OpenAPI spec) |
| `tokenEndpointAuth` | ❌ Optional | Default: `rfc6749-client-secret-basic` |

**Precedence Rules (http-settings.yaml vs OpenAPI spec):**

When both http-settings.yaml and the OpenAPI specification provide values, the following precedence applies:

1. **`tokenUrl`**: http-settings.yaml `tokenUrl` **overrides** OpenAPI spec's `flows.clientCredentials.tokenUrl`
2. **`refreshUrl`**: http-settings.yaml `refreshUrl` **overrides** OpenAPI spec's `flows.clientCredentials.refreshUrl`
3. **`scope`**: http-settings.yaml `scope` **overrides** OpenAPI spec's per-operation `security` scopes

**Common Scenarios:**

| Scenario | `useForSpecFetch` | `tokenUrl` in http-settings | `tokenUrl` in OpenAPI spec | Result |
|----------|-------------------|----------------------------|---------------------------|--------|
| **Protected OpenAPI spec** | `true` (default) | ✅ Required | Used as fallback | http-settings value used |
| **Public OpenAPI spec** | `false` | ❌ Optional | ✅ Required in spec | OpenAPI spec value used |
| **Override spec settings** | `true` or `false` | ✅ Provided | Any | http-settings value **always wins** |

### OAuth2 Token Endpoint Authentication Methods

The `tokenEndpointAuth` field controls how the client authenticates when requesting OAuth2 tokens. Two methods are supported:

**`rfc6749-client-secret-basic` (default):** HTTP Basic Authentication (RFC 6749 Section 2.3.1)
- Both `clientId` and `clientSecret` are sent in the `Authorization: Basic` header
- Works with most OAuth2 providers

**`rfc5849-oauth1-signature`:** OAuth 1.0 HMAC-SHA256 request signing (RFC 5849)
- `clientId` is sent as `oauth_consumer_key` in both header and body
- `clientSecret` is used only for HMAC-SHA256 signature computation (never transmitted)
- Required by some providers that use OAuth 1.0 signature-based token authentication
- Provides enhanced security through cryptographic request signing
- **Note:** Only HMAC-SHA256 signature method is supported

### OAuth2-Authenticated OpenAPI Spec Fetching

By default, when OAuth2 is configured, zswag will acquire an OAuth2 access token **before** fetching the OpenAPI specification. This solves the "chicken-and-egg" problem where the OpenAPI spec endpoint itself requires authentication.

**How it works:**

1. **With `useForSpecFetch: true` (default):**
   - Client acquires OAuth2 token using configured authentication method
   - Token is included as `Authorization: Bearer <token>` header when fetching OpenAPI spec
   - OpenAPI spec fetch succeeds even if endpoint requires authentication
   - Same token is cached and reused for subsequent API calls

2. **With `useForSpecFetch: false`:**
   - OpenAPI spec is fetched without OAuth2 token (plain HTTP GET)
   - OAuth2 token acquisition is deferred until first API method call
   - Use this when the OpenAPI spec endpoint is public (doesn't require authentication)

**Configuration:**

```yaml
- scope: https://api.example.com/*
  oauth2:
    clientId: your-client-id
    clientSecret: your-client-secret
    tokenUrl: https://api.example.com/oauth/token
    useForSpecFetch: true  # Default: true. Set to false if spec is public.
```

**When to use `useForSpecFetch: false`:**
- OpenAPI spec endpoint is publicly accessible
- Avoids unnecessary token acquisition if only the spec is needed
- Improves performance by deferring OAuth2 flow until first API call

**When to keep `useForSpecFetch: true` (default):**
- OpenAPI spec endpoint requires authentication
- Service returns 401/403 when fetching spec without credentials
- Most secure option (ensures token is available from the start)

**Debugging OAuth2 Issues:**

To troubleshoot OAuth2 authentication problems, enable detailed logging:

```bash
export HTTP_LOG_LEVEL=debug   # Shows OAuth2 flow (token acquisition, cache hits/misses)
export HTTP_LOG_LEVEL=trace   # Shows additional details (request/response bodies, signatures)
```

The logs will show:
- Token endpoint authentication method being used
- Token request/response status
- Token cache hit/miss/expired events
- OAuth2 configuration status for OpenAPI spec fetch
- Whether OAuth2 token is being used for spec fetch

### Testing OAuth 1.0 Signature with Your Service

To verify OAuth 1.0 signature authentication with your service:

**1. Install zswag:**

For official releases:
```bash
pip install zswag
```

For custom builds or development snapshots:
```bash
pip install /path/to/pyzswagcl-*.whl /path/to/zswag-*.whl
```

**2. Create `http-settings.yaml`** with your service details:
```yaml
- scope: https://your-api.example.com/*
  oauth2:
    clientId: your-client-id
    clientSecret: your-client-secret
    tokenUrl: https://your-api.example.com/oauth/token
    tokenEndpointAuth:
      method: rfc5849-oauth1-signature
      nonceLength: 16
```

**3. Create a test script** (`test_oauth1.py`):
```python
import os
from zswag import OAClient

# Point to your http-settings file
os.environ['HTTP_SETTINGS_FILE'] = '/path/to/http-settings.yaml'

# Create client with your OpenAPI spec
client = OAClient("https://your-api.example.com/openapi.json")

# Import your generated zserio service
import your_service.api as api

# Create service client and make a test call
service = api.YourService.Client(client)
request = api.YourRequest(...)
response = service.your_method(request)

print(f"Success! Response: {response}")
```

**4. Run the test:**
```bash
python test_oauth1.py
```

**Verification:** Check your server logs to confirm OAuth 1.0 HMAC-SHA256 signatures are being validated correctly and the token endpoint receives properly signed requests.

The **`api-key`** setting will be applied under the correct
cookie/header/query parameter, if the service
you are connecting to uses an [OpenAPI `apiKey` auth scheme](#authentication-schemes).

Passwords can be stored in clear text by setting a `password` field instead
of the `keychain` field. Keychain entries can be made with different tools
on each platform:

* [Linux `secret-tool`](https://www.marian-dan.ro/blog/storing-secrets-using-secret-tool) 
* [macOS `add-generic-password`](https://www.netmeister.org/blog/keychain-passwords.html)
* [Windows `cmdkey`](https://www.scriptinglibrary.com/languages/powershell/how-to-manage-secrets-and-passwords-with-credentialmanager-and-powershell/)

<!-- --8<-- [end:settings] -->

## Client Result Code Handling

Both clients (Python and C++) will treat any HTTP response code other than `200` as an error since zserio services are expected to return a parsable response object. The client will throw an exception with a descriptive message if the response code is not `200`.

In case applications want to utilize for example the `204 (No Content)` response code, they have to catch the exception and handle it accordingly.

## Swagger User Interface 

If you have installed `pip install "connexion[swagger-ui]"`, you can view
API docs of your service under `[/prefix]/ui`.

## OpenAPI Options Interoperability

The Server, Clients and Generator offer various degrees of freedom
regarding the OpenAPI YAML file. The following sections detail which
components support which aspects of OpenAPI. The difference in compliance
is mostly due to limited development scopes. If you are missing a particular
OpenAPI feature for a particular component, feel free to create an issue!

**Note:** For all options that are not supported by `zswag.gen`, you
will need to manually edit the OpenAPI YAML file to achieve the desired
configuration. You will also need to edit the file manually to fill in
meta-info (provider name, service version, etc.).

### HTTP method

To change the **HTTP method**, the desired method name is placed 
as the key under the method path, such as in the following example:
```yaml
paths:
  /methodName:
    {get|post|put|delete}:
      ...
```

#### Component Support

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `get` `post` `put` `delete` | ✔️ | ✔️ | ✔️ | ✔️ |
| `patch` | ❌️ | ❌️ | ❌️ | ❌️ |

**Note:** Patch is unsupported, because the required semantics of
a partial object update cannot be realized in the zserio transport
layer interface.

### Request Body

A server can instruct clients to transmit their zserio request object in the
request body when using HTTP `post`, `put` or `delete`.
This is done by setting the OpenAPI `requestBody/content` to
`application/x-zserio-object`:

```yaml
requestBody:
  content:
    application/x-zserio-object:
      schema:
        type: string
```

#### Component Support

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `application/x-zserio-object` | ✔️ | ✔️ | ✔️ | ✔️ |

### URL Blob Parameter

Zswag tools support an additional OpenAPI method parameter
field called `x-zserio-request-part`. Through this field,
a service provider can express that a certain request parameter
only contains a part of, or the whole zserio request object.
When parameter contains the whole request object, `x-zserio-request-part`
should be set to an asterisk (`*`):

```yaml
parameters:
- description: ''
  in: query|path|header
  name: parameterName
  required: true
  x-zserio-request-part: "*"
  schema:
    format: string|byte|base64|base64url|hex|binary
```

About the `format` specifier value:
* Both `string` and `binary` result in a raw URL-encoded string buffer.
* Both `byte` and `base64` result in a standard Base64-encoded value.
  The `base64url` option indicates URL-safe Base64 format.
* The `hex` encoding produces a hexadecimal encoding of the request blob.

**Note:** When a parameter is passed with `in=path`, its value
**must not be empty**. This holds true for strings and bytes,
but also for arrays (see below).

#### Component Support

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `x-zserio-request-part: *` | ✔️ | ✔️ | ✔️ | ✔️ |
| `format: string` | ✔️ | ✔️ | ✔️ | ✔️ |
| `format: byte` | ✔️ | ✔️ | ✔️ | ✔️ |
| `format: hex` | ✔️ | ✔️ | ✔️ | ✔️ |

### URL Scalar Parameter

Using `x-zserio-request-part`, it is also possible to transfer
only a single scalar (nested) member of the request object:

```yaml
parameters:
- description: ''
  in: query|path|header
  name: parameterName
  required: true
  x-zserio-request-part: "[parent.]*member"
  schema:
    format: string|byte|base64|base64url|hex|binary
```

In this case, `x-zserio-request-part` should point to a scalar type,
such as `uint8`, `float32`, `string` etc.

The `format` value effect remains as explained above. A small
difference exists for integer types: Their hexadecimal representation
will be the natural numeric one, not binary.

#### Component Support

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `x-zserio-request-part: <[parent.]*member>` | ✔️ | ✔️ | ✔️ | ✔️ |

### URL Array Parameter

The `x-zserio-request-part` may also point to an array member of
the zserio request struct, like so:

```yaml
parameters:
- description: ''
  in: query|path|header
  style: form|simple|label|matrix
  explode: true|false
  name: parameterName
  required: true
  x-zserio-request-part: "[parent.]*array_member"
  schema:
    format: string|byte|base64|base64url|hex|binary
```

In this case, `x-zserio-request-part` should point to an array of
scalar types. The array will be encoded according
to the [format, style and explode](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.1.0.md#parameter-object)
specifiers.

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `x-zserio-request-part: <[parent.]*array_member>`  | ✔️ | ✔️ | ✔️ | ✔️ |
| `style: simple` | ✔️ | ✔️ | ✔️ | ✔️ |
| `style: form` | ✔️ | ✔️ | ✔️ | ✔️ |
| `style: label` | ✔️ | ✔️ | ❌ | ✔️ |
| `style: matrix` | ✔️ | ✔️ | ❌ | ✔️ |
| `explode: true` | ✔️ | ✔️ | ✔️ | ✔️ |
| `explode: false` | ✔️ | ✔️ | ✔️ | ✔️ |

### URL Compound Parameter

In this case, `x-zserio-request-part` points to a zserio compound struct
instead of a field with a scalar value. **This is currently not supported.**

#### Component Support

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `x-zserio-request-part: <[parent.]*compound_member>`  | ❌️ | ❌️ | ❌️ | ❌️ |

### Server URL Base Path

OpenAPI allows for a `servers` field in the spec that lists URL path prefixes
under which the specified API may be reached. The OpenAPI clients
looks into this list to determine a URL base path from
the first entry in this list. A sample entry might look as follows:

```
servers:
- http://unused-host-information/path/to/my/api
``` 

The OpenAPI client will then call methods with your specified host
and port, but prefix the `/path/to/my/api` string. 

#### Component Support

| Feature            | C++ Client | Python Client | OAServer | zswag.gen |
| ------------------ | ---------- | ------------- | -------- | --------- |
| `servers`  | ✔️ | ✔️ | ✔️ | ✔️ |

### Authentication Schemes

To facilitate the communication of authentication needs for the whole or parts
of a service, OpenAPI allows for `securitySchemes` and `security` fields in the spec.
Please refer to the relevant parts of the [OpenAPI 3 specification](https://swagger.io/docs/specification/authentication/) for some
examples on how to integrate these fields into your spec.

#### When Security Schemes Are Applied

**Important:** Security schemes (including OAuth2) are **only applied when explicitly declared** in the OpenAPI specification. Zswag clients respect the security requirements defined in the spec according to the [OpenAPI 3.0 Security Requirement specification](https://spec.openapis.org/oas/v3.0.3#security-requirement-object).

**Security Configuration Levels:**

1. **Global Security** - Applied to all endpoints by default (root-level `security` field):
   ```yaml
   components:
     securitySchemes:
       HeaderAuth:
         type: apiKey
         in: header
         name: X-Generic-Token

   security:
     - HeaderAuth: []  # Applied to all endpoints by default

   paths:
     /methodWithGlobalAuth:
       get:
         # Uses global HeaderAuth
   ```

2. **Per-Endpoint Security** - Override global security for specific operations:
   ```yaml
   paths:
     /protected:
       post:
         security:
           - oauth2: [read, write]  # Overrides global security
     /admin:
       post:
         security:
           - oauth2: [admin]  # Different scopes for admin endpoint
   ```

3. **No Authentication** - Explicitly disable security for public endpoints:
   ```yaml
   paths:
     /public:
       get:
         security: []  # Explicitly no authentication required (overrides global)
   ```

**Precedence Rule:** Per-operation `security` settings **override** global `security` settings. If an operation specifies its own security requirements (including `security: []`), the global security configuration is ignored for that operation.

**Complete Working Examples:**

- **OAuth2 with per-endpoint security**: See [`libs/zswagcl/test/testdata/oauth2-openapi.yaml`](libs/zswagcl/test/testdata/oauth2-openapi.yaml) which demonstrates different OAuth2 scopes per endpoint and public endpoints without authentication.
- **Global security with overrides**: See [`libs/zswag/test/calc/api.yaml`](libs/zswag/test/calc/api.yaml) which shows global `HeaderAuth` security with per-endpoint overrides and explicit no-auth declarations.

Zswag currently understands the following authentication schemes:

* **HTTP Basic Authorization:** If a called endpoint requires HTTP basic auth,
  zswag will verify that the HTTP config contains basic-auth credentials.
  If there are none, zswag will throw a descriptive runtime error.
* **HTTP Bearer Authorization:** If a called endpoint requires HTTP bearer auth,
  zswag will verify that the HTTP config contains a header with the
  key name `Authorization` and the value `Bearer <token>`, *case-sensitive*.
* **API-Key Cookie:** If a called endpoint requires a Cookie API-Key,
  zswag will either apply [the `api-key` setting](#persistent-http-headers-proxy-cookie-and-authentication), or verify that the
  HTTP config contains a cookie with the required name, *case-sensitive*.
* **API-Key Query Parameter:** If a called endpoint requires a Query API-Key,
  zswag will either apply the `api-key` setting, or verify that the
  HTTP config contains a query key-value pair with the required name, *case-sensitive*.
* **API-Key Header:** If a called endpoint requires an API-Key Header,
  zswag will either apply the `api-key` setting, or verify that the
  HTTP config contains a header key-value pair with the required name, *case-sensitive*.
* **OAuth2 Client Credentials:** If a called endpoint requires OAuth2 authentication,
  zswag will **automatically acquire, cache, and refresh** access tokens from the configured
  OAuth2 token endpoint. The client handles the entire OAuth2 client credentials flow
  transparently, including token expiry and refresh. **Note:** Only the `clientCredentials`
  flow is supported. See the [OAuth2 configuration section](#persistent-http-headers-proxy-cookie-and-authentication)
  for detailed setup instructions.

**Note**: If you don't want to pass your Basic-Auth/Bearer/Query/Cookie/Header
credential through your [persistent config](#persistent-http-headers-proxy-cookie-and-authentication),
you can pass a `httpcl::Config`/[`HTTPConfig`](#using-the-python-client) object to the `OAClient`/[`OAClient`](#using-the-python-client).
constructor in C++/Python with the relevant detail.

#### Component Support

| Feature                                                                                | C++ Client | Python Client | OAServer | zswag.gen |
|----------------------------------------------------------------------------------------| ---------- | ------------- | -------- | --------- |
| `HTTP Basic-Auth` `HTTP Bearer-Auth` `Cookie API-Key` `Header API-Key` `Query API-Key` | ✔️ | ✔️ | ✔️(**) | ✔️ |
| `OAuth2[clientCredentials]`                                                            | ✔️ | ✔️ | ✔️(**) | ✔️ |
| `OpenID Connect` `OAuth2[authorizationCode]` `OAuth2[implicit]` `OAuth2[password]`     | ❌️ | ❌️ | ✔️(**) | ❌️ |

**(\*\*)**: The server support for all authentication schemes depends on your
configuration of the WSGI server (Apache/Nginx/...) which wraps the zswag Flask app.
 readmeEtag: '"48ae6026d8bc0786bccf7d12d0aa96fc649a2476"' readmeLastModified: Thu, 04 Dec 2025 17:19:26 GMT repositoryId: 253472682 description: Access/Serve zserio services via REST/OpenAPI 🌍. created: '2020-04-06T11:07:06Z' updated: '2025-12-05T11:20:42Z' language: C++ archived: false stars: 15 watchers: 5 forks: 4 owner: ndsev logo: https://avatars.githubusercontent.com/u/38692822?v=4 license: BSD-3-Clause repoEtag: '"fced908ee3fc615ac4b2fde8ab723b5cf2ceaef25da9feb02246ae8ca784dbb6"' repoLastModified: Fri, 05 Dec 2025 11:20:42 GMT foundInMaster: true oldLocations: - https://github.com/klebert-engineering/zswag - source: openapi3 tags name: openapi-data-validator category: - Data Validators - Parsers repository: https://github.com/authress-engineering/openapi-data-validator.js language: - Node.js - Javascript source_description: >- Validate API requests against an OpenAPI schema. Lightweight, focused, and integrates with any framework link: https://github.com/Rhosys/openapi-data-validator.js/blob/main/README.md v2: false v3: true repositoryMetadata: base64Readme: >- IyMgT3BlbkFQSSBEYXRhIFZhbGlkYXRvcgpMaWdodHdlaWdodCBPcGVuQVBJIGNvbXBsZXRlIHJlcXVlc3QgbW9kZWwgdmFsaWRhdG9yLiBGYXN0LCB1bm9waW5pb25hdGVkLCBmdWxsIGZlYXR1cmVkIHZhbGlkYXRvciBmb3IgQVBJIHJlcXVlc3RzIHRoYXQgdXRpbGl6ZSBPcGVuQVBJIGRvY3MgZm9yIEFQSSBkb2N1bWVudGF0aW9uLgoKVGhpcyBpcyBhbiBvcGVuIHNvdXJjZSBwcm9qZWN0IG1hbmFnZWQgYnkgdGhlIFtBdXRocmVzcyBFbmdpbmVlcmluZyB0ZWFtXShodHRwczovL2F1dGhyZXNzLmlvKS4KClshW0F1dGhyZXNzIEVuZ2luZWVyaW5nXShodHRwczovL2ltZy5zaGllbGRzLmlvL3N0YXRpYy92MT9sYWJlbD1BdXRocmVzcytFbmdpbmVlcmluZyZtZXNzYWdlPU9wZW5BUEklMjBEYXRhJTIwVmFsaWRhdG9yJmNvbG9yPSUyM0ZCQUYwQiZsb2dvPWFuZHJvaWRhdXRvJmxvZ29Db2xvcj0lMjNGQkFGMEIpXShodHRwczovL2F1dGhyZXNzLmlvKSBbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL29wZW5hcGktZGF0YS12YWxpZGF0b3Iuc3ZnKV0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL29wZW5hcGktZGF0YS12YWxpZGF0b3IpCgojIyBVc2FnZQpJdCBpcyBzaW1wbGUsIGFuZCB0aGF0J3MgYWxsIHRoZXJlIGlzIHRvIGl0IQoKYGBgc2gKbnBtIGluc3RhbGwgb3BlbmFwaS1kYXRhLXZhbGlkYXRvciAtLXNhdmUKYGBgCgpgYGBqcwpjb25zdCB7IE9wZW5BcGlWYWxpZGF0b3IgfSA9IHJlcXVpcmUoJ29wZW5hcGktZGF0YS12YWxpZGF0b3InKTsKY29uc3Qgc3BlYyA9IHJlcXVpcmUoJy4vb3BlbmFwaS5qc29uJyk7Cgpjb25zdCBvcGVuQXBpVmFsaWRhdG9yID0gbmV3IE9wZW5BcGlWYWxpZGF0b3IoeyBhcGlTcGVjOiBzcGVjIH0pOwpjb25zdCB2YWxpZGF0b3IgPSBvcGVuQXBpVmFsaWRhdG9yLmNyZWF0ZVZhbGlkYXRvcigpOwoKLy8gQ29uZmlndXJlIHRoaXMgdG8gdGhlIGNsaWVudCdzIHJlcXVlc3QuIEl0IHdpbGwgcmVzb2x2ZSB0aGUgZXhwZWN0ZWQgc2NoZW1hIGluIHRoZSBzcGVjIHVzaW5nIHRoZSBtZXRob2QgYW5kIHJvdXRlIGRlZmluZWQsIGFuZCB2YWxpZGF0ZSB0aGUgcmVxdWVzdCBwYXJhbWV0ZXJzLgpjb25zdCBuZXdSZXF1ZXN0ID0gewogIG1ldGhvZDogJ0dFVCcsCiAgLy8gTWF0Y2hlZCBvcGVuYXBpIHNwZWNpZmljYXRpb24gZ2VuZXJpYyByb3V0ZSwgdGhpcyBzaG91bGQgYmUgdGhlIGdlbmVyaWMgYHBhdGhgIGZyb20gdGhlIHNwZWMsIHN1Y2ggYXMgYC9yZXNvdXJjZXMve3Jlc291cmNlSWR9L2AsIGl0IG11c3QgbWF0Y2ggb25lIG9mIHRoZW0gZXhhY3RseS4KICByb3V0ZTogcmVxdWVzdC5yb3V0ZQoKICBoZWFkZXJzOiB7IEF1dGhvcml6YXRpb246ICdCZWFyZXIgVG9rZW4nIH0sCiAgCiAgLy8gUXVlcnkgc3RyaW5nIHBhcmFtZXRlcnMgZnJvbSB0aGUgcmVxdWVzdAogIHF1ZXJ5OiB7IGxpbWl0OiAxMCB9LAogIAogIC8vIEJvZHkgYWxyZWFkeSBwYXJzZWQgdG8gSlNPTgogIGJvZHk6IHsgZmllbGQ6IHRydWUgfSwKCiAgLy8gUGF0aCBwYXJhbWV0ZXJzCiAgcGF0aDogeyB1c2VyOiAndXNlcklkJyB9Cn07CmF3YWl0IHZhbGlkYXRvcihuZXdSZXF1ZXN0KTsKYGBgCgojIyBDb21waWxlIHZhbGlkYXRvcgpGb3IgaW1wcm92ZWQgcHJvY2Vzc2luZyBzcGVlZCB0aGUgdmFsaWRhdG9yIGNhbiBiZSBwcmUtY29tcGlsZWQgZnJvbSB0aGUgc3BlYwoKYGBganMKY29uc3Qgb3BlbkFwaVZhbGlkYXRvciA9IG5ldyBPcGVuQXBpVmFsaWRhdG9yKHsgYXBpU3BlYzogc3BlYywgY29tcGlsZWRGaWxlUGF0aDogJy4vY29tcGlsZWRWYWxpZGF0b3IuanNvbicgfSk7CmF3YWl0IG9wZW5BcGlWYWxpZGF0b3IuY29tcGlsZVZhbGlkYXRvcigpOwovLyBMYXRlcgpjb25zdCB2YWxpZGF0b3IgPSBhd2FpdCBvcGVuQXBpVmFsaWRhdG9yLmxvYWRWYWxpZGF0aW9uKCk7CgovLyAuLi4KYXdhaXQgdmFsaWRhdG9yKHJlcXVlc3QpOwpgYGAKCkNoZWNrb3V0IHRoZSBmdWxsOiBbQXN5bmMgZXhhbXBsZV0oLi9kb2NzL2FzeW5jLWV4YW1wbGUubWQpCgojIyBGQVFzCgojIyMjIFdoeSBub3QganVzdCB1c2UgQUpWCkFKViBpcyB0aGUgYmVzdCwgYnV0IHRoZXJlIGFyZSBzb21lIHRoaW5ncyB0aGF0IGp1c3QgYXJlIHZlcnkgT3BlbkFQSSBzcGVjaWZpYyB0aGF0IGRvbid0IG1ha2Ugc2Vuc2UgdG8gYmUgaW4gdGhlIHZhbGlkYXRvci4gRG9uJ3QgbmVlZCB0aGVtPyBHcmVhdCwgZ28gdXNlIEFKVi4KCiogVG9wIGxldmVsIGRlZmluZWQgUGF0aCBwYXJhbWV0ZXJzIC0gQUpWIGRvZXNuJ3QgdW5kZXJzdGFuZAoqIElubGluZSByZXF1ZXN0IGJvZHkgZGVmaW5pdGlvbnMsIEFKViBkb2Vzbid0IHVuZGVyc3RhbmQgc2NoZW1hIGRlZmluZWQgaW4gdGhlIG1ldGhvZCwgaXQgaGFzIHRvIGJlIGluIGEgY29tcG9uZW50CiogQm9keSBDb250ZW50LVR5cGUgdmFsaWRhdGlvbiAtIFJlcXVlc3QgYm9kaWVzIHdpdGggbXVsdGlwbGUgY29udGVudCB0eXBlcyBhbGxvd2VkCg== readmeEtag: '"83712693f8165c9b29fdb87cddc5f5574e655383"' readmeLastModified: Tue, 31 Oct 2023 18:33:44 GMT repositoryId: 373612289 description: OpenAPI Request Validation for API for any framework created: '2021-06-03T18:58:54Z' updated: '2025-10-14T20:46:58Z' language: JavaScript archived: false stars: 16 watchers: 2 forks: 7 owner: Authress-Engineering logo: https://avatars.githubusercontent.com/u/35577654?v=4 license: Apache-2.0 repoEtag: '"db2f1d20c6091427db4324d3cf5218fd94236fcb349720020c59bb8580e571e8"' repoLastModified: Tue, 14 Oct 2025 20:46:58 GMT foundInMaster: true id: c64a20817064b7b24a1f079a7b7b5363 oldLocations: - https://github.com/rhosys/openapi-data-validator.js - source: openapi3 tags repository: https://github.com/sake92/openapi4s v3: true id: 81d3023aafb15ff4a725e961e8da13d9 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpNHMKCk9wZW5BcGkgZ2VuZXJhdG9ycyBmb3IgU2NhbGEuCgpIZXJlIGlzIGEgc21hbGwgdmlkZW8gZGVtbzogaHR0cHM6Ly95b3V0dS5iZS9rZjB2R3JsS05iOAoKIyMgRmVhdHVyZXMgYW5kIEJlbmVmaXRzCi0gaW5jcmVtZW50YWwgZ2VuZXJhdG9yCiAgLSBkb2Vzbid0IHRvdWNoIHRoZSBjb2RlIHRoYXQgeW91IGFkZGVkIG1hbnVhbGx5CiAgLSBvbmx5IGFkZHMgbmV3IHByb3BlcnRpZXMvbWV0aG9kcy9jbGFzc2VzCi0gbGVuaWVudCBwYXJzZXIrZ2VuZXJhdG9yCiAgLSBpZiBzb21ldGhpbmcgaXMgbm90IHN1cHBvcnRlZCBpdCB3aWxsIHN0aWxsIChtb3N0bHkpIHdvcmsKICAtIHlvdSBjYW4gYWRhcHQgeW91ciBvcGVuYXBpIHNwZWMgdG8gd29yayBncmFkdWFsbHkKCiMjIExpbWl0YXRpb25zCi0gSlNPTiBvbmx5Ci0gb25seSAqbmFtZWQqIGVudGl0aWVzLCBubyBhbm9ueW1vdXMgb2JqZWN0cwoKLS0tCgojIyBHZW5lcmF0b3JzCgojIyMgU2hhcmFmIGJhY2tlbmQKU3VwcG9ydHMgYWxtb3N0IGFsbCBmZWF0dXJlczoKLSBjb250cm9sbGVycwotIGRpc2NyaW1pbmF0ZWQgbW9kZWxzIChzZWFsZWQgdHJhaXRzKQotIGVudW1zIChzY2FsYTMgc2luZ2xldG9uIGVudW1zKQotIHZhbGlkYXRpb25zCi0gcXVlcnkgcGFyYW1zCgojIyMgSHR0cDRzIGJhY2tlbmQKU3VwcG9ydHMgc29tZSBmZWF0dXJlczoKLSByb3V0ZXMgKGNvbnRyb2xsZXJzKQotIGRpc2NyaW1pbmF0ZWQgbW9kZWxzIChzZWFsZWQgdHJhaXRzKQotIGVudW1zIChzY2FsYTMgc2luZ2xldG9uIGVudW1zKQoKVE9ETzogcXVlcnkgcGFyYW1zLCB2YWxpZGF0aW9uLi4gIApDb250cmlidXRpb25zIHdlbGNvbWUhCgotLS0KCiMjIFVzYWdlCgojIyMgTWlsbCBwbHVnaW4KClNlZSBodHRwczovL2dpdGh1Yi5jb20vc2FrZTkyL21pbGwtb3BlbmFwaTRzCgojIyMgQ0xJCgpZb3UgY2FuIHVzZSBgb3BlbmFwaTRzLWNsaWAgd2l0aCBDb3Vyc2llciBsYXVuY2hlciB0byBnZW5lcmF0ZSB5b3VyIHNvdXJjZXM6CgpgYGBzaGVsbApjcyBsYXVuY2ggYmEuc2FrZTo6b3BlbmFwaTRzLWNsaTowLjYuMSAtTSBiYS5zYWtlLm9wZW5hcGk0cy5jbGkuT3BlbkFwaTRzTWFpbiAtLSBcCiAgLS1nZW5lcmF0b3Igc2hhcmFmIFwKICAtLXVybCBvcGVuYXBpLmpzb24gXAogIC0tYmFzZUZvbGRlciBzcmMgXAogIC0tYmFzZVBhY2thZ2UgY29tLmV4YW1wbGUKYGBgCgoK readmeEtag: '"0a0b438e170e4aae69680d729ffb066a6838a44f"' readmeLastModified: Wed, 16 Jul 2025 13:36:32 GMT repositoryId: 906544996 description: openapi4s created: '2024-12-21T07:37:30Z' updated: '2026-01-31T14:22:19Z' language: Scala archived: false stars: 26 watchers: 3 forks: 0 owner: sake92 logo: https://avatars.githubusercontent.com/u/12010694?v=4 license: Apache-2.0 repoEtag: '"c922c8197c2d17ca7f00200a2ded3c878596282cd6950eb29dbb9bf6148d3884"' repoLastModified: Sat, 31 Jan 2026 14:22:19 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/yapily/yapily-sdk-python v3: true id: d79b8bc72d6746f1ce7697d3680dfeb3 repositoryMetadata: base64Readme: >- IyBZYXBpbHkgUHl0aG9uIFNESwpbIVtHaXRIdWIgdmVyc2lvbl0oaHR0cHM6Ly9kMjVsY2lwemlqMTdkLmNsb3VkZnJvbnQubmV0L2JhZGdlLnN2Zz9pZD1naCZ0eXBlPTYmdj0meDI9MCldKGh0dHA6Ly9iYWRnZS5mdXJ5LmlvL2doL2JvZW5uZW1hbm4lMkZiYWRnZXMpCgpUaGlzIFNESyBsaWJyYXJ5IHdhcyBnZW5lcmF0ZWQgdXNpbmcgW09wZW5BcGkgR2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKS4gVGhlIFNESyBjYW4gYmUgdXNlZCBhcyBhIG1vZHVsZSBpbiB5b3VyIGNvZGUgYW5kIHRoZSBleGFtcGxlcyBkZW1vbnN0cmF0ZSBob3cgdG8gY29ubmVjdCB0byBmaW5hbmNpYWwgaW5zdGl0dXRpb25zIGludGVncmF0ZWQgd2l0aCBZYXBpbHkuCgojIyBSZXF1aXJlbWVudHMKClRvIGNvbm5lY3QgdG8gdGhlIFlhcGlseSBBUEksIHlvdSB3aWxsIG5lZWQgdG8gcmVnaXN0ZXIgeW91ciAKYXBwbGljYXRpb24gYXQgW2h0dHBzOi8vZGFzaGJvYXJkLnlhcGlseS5jb21dKCkuCgpUaGVzZSBhcHBsaWNhdGlvbiBjcmVkZW50aWFscyBjYW4gdGhlbiBiZSB1c2VkIHRvIGF1dGhvcmlzZSBhbGwKeW91ciBBUEkgcmVxdWVzdHMuCgojIyBJbnN0YWxsYXRpb24KClRoZSBTREsgaXMgY3VycmVudGx5IGF2YWlsYWJsZSBpbiB0aGUgWWFwaWx5IGdpdGh1YiByZXBvc2l0b3J5IGFuZCAKY2FuIGJlIGluY2x1ZGVkIGluIHlvdXIgcHJvamVjdCAKYnkgYWRkaW5nIGl0IHRvIHlvdXIgZGVwZW5kZW5jaWVzCgojIyMjIHBpcCBpbnN0YWxsCgpwaXAzIGluc3RhbGwgZ2l0K2h0dHBzOi8vZ2l0aHViLmNvbS95YXBpbHkveWFwaWx5LXNkay1weXRob24uZ2l0I3N1YmRpcmVjdG9yeT1zZGsKCiMjIyMgcGlwIHVwZ3JhZGUKCnBpcDMgaW5zdGFsbCAtLXVwZ3JhZGUgZ2l0K2h0dHBzOi8vZ2l0aHViLmNvbS95YXBpbHkveWFwaWx5LXNkay1weXRob24uZ2l0I3N1YmRpcmVjdG9yeT1zZGsKCiMjIFVzYWdlCgpTYW1wbGUgdXNhZ2Ugb2YgdGhlIFNESyBjYW4gYmUgc2VlbiBpbiB0aGUgYGV4YW1wbGVzYCBmb2xkZXIuCgotIFJldHJpZXZlIGEgbGlzdCBvZiBhdmFpbGFibGUgZmluYW5jaWFsIGluc3RpdHV0aW9ucyB0byBjb25uZWN0IHRvCgpgYGBweXRob24KY29uZmlndXJhdGlvbiA9IENvbmZpZ3VyYXRpb24oKQpjb25maWd1cmF0aW9uLnVzZXJuYW1lID0gY29uc3RhbnRzLkFQUExJQ0FUSU9OX0lECmNvbmZpZ3VyYXRpb24ucGFzc3dvcmQgPSBjb25zdGFudHMuQVBQTElDQVRJT05fU0VDUkVUCgphcGlDbGllbnQgPSBBcGlDbGllbnQoY29uZmlndXJhdGlvbikKaW5zdGl0dXRpb25zQXBpID0gSW5zdGl0dXRpb25zQXBpKGFwaUNsaWVudCkKaW5zdGl0dXRpb25zID0gaW5zdGl0dXRpb25zQXBpLmdldF9pbnN0aXR1dGlvbnNfdXNpbmdfZ2V0KCkKYGBgCgotIENyZWF0aW5nIHVzZXJzIGFuZCByZXRyaWV2aW5nIHVzZXJzIGZvciB5b3VyIGFwcGxpY2F0aW9uIHJlZ2lzdGVyZWQgaW4gdGhlIFlhcGlseSBEYXNoYm9hcmQKYGBgcHl0aG9uCmFwcGxpY2F0aW9uX3VzZXIgPSBOZXdBcHBsaWNhdGlvblVzZXIoYXBwbGljYXRpb25fdXNlcl9pZD1hcHBfdXNlcl9pZCkKdXNlcl9hcGkgPSBBcHBsaWNhdGlvblVzZXJzQXBpKGFwaUNsaWVudCkKdXNlcl9hcGkuYWRkX3VzZXJfdXNpbmdfcG9zdChhcHBsaWNhdGlvbl91c2VyKQpgYGAKCi0gQ3JlYXRlIGFuIGF1dGhvcmlzYXRpb24gVVJMIGZvciB5b3VyIHVzZXJzIHRvIHVzZSB0byBsb2cgaW50byB0aGVpciBpbnN0aXR1dGlvbgoKYGBgcHl0aG9uCmFjY291bnRfYXV0aG9yaXNhdGlvbl9yZXF1ZXN0ID0gQWNjb3VudEF1dGhvcmlzYXRpb25SZXF1ZXN0KAogICAgYXBwbGljYXRpb25fdXNlcl9pZD1jb25zdGFudHMuQVBQTElDQVRJT05fVVNFUl9JRCwgCiAgICBpbnN0aXR1dGlvbl9pZD1jb25zdGFudHMuSU5TVElUVVRJT05fSUQsCiAgICBjYWxsYmFjaz0nJywKICAgIG9uZV90aW1lX3Rva2VuPScnCikKCnJlc3BvbnNlID0gYWNjb3VudHNfYXBpLmluaXRpYXRlX2FjY291bnRfcmVxdWVzdF91c2luZ19wb3N0KGFjY291bnRfYXV0aF9yZXF1ZXN0PWFjY291bnRfYXV0aG9yaXNhdGlvbl9yZXF1ZXN0KQpyZWRpcmVjdF91cmwgPSByZXNwb25zZS5kYXRhLmF1dGhvcmlzYXRpb25fdXJsCmBgYAogCi0gT2J0YWluaW5nIGEgdmFsaWQgY29uc2VudCBmb3IgZmluYW5jaWFsIGRhdGEKCmBgYHB5dGhvbgpkZWYgZmlsdGVyQnlTdGF0dXMoY29uc2VudCk6CiAgICBpZiAoY29uc2VudC5zdGF0dXMgPT0gIkFVVEhPUklaRUQiKToKICAgICAgICByZXR1cm4gVHJ1ZQogICAgZWxzZToKICAgICAgICByZXR1cm4gRmFsc2UKCmNvbnNlbnRzID0gQ29uc2VudHNBcGkoYXBpQ2xpZW50KS5nZXRfY29uc2VudHNfdXNpbmdfZ2V0KAogICAgZmlsdGVyX2FwcGxpY2F0aW9uX3VzZXJfaWQ9W2NvbnN0YW50cy5BUFBMSUNBVElPTl9VU0VSX0lEXSwKICAgIGZpbHRlcl9pbnN0aXR1dGlvbj1bY29uc3RhbnRzLklOU1RJVFVUSU9OX0lEXQopLmRhdGEKCmF1dGhvcmlzZWRfY29uc2VudHMgPSBsaXN0KGZpbHRlcihmaWx0ZXJCeVN0YXR1cywgY29uc2VudHMpKQpjb25zZW50ID0gYXV0aG9yaXNlZF9jb25zZW50c1swXQpjb25zZW50X3Rva2VuID0gY29uc2VudC5jb25zZW50X3Rva2VuCmBgYAoKLSBSZXR1cm5pbmcgdXNlciBhY2NvdW50IGRldGFpbHMKYGBgcHl0aG9uCmFjY291bnRzQXBpID0gQWNjb3VudHNBcGkoYXBpQ2xpZW50KQphY2NvdW50cyA9IGFjY291bnRzQXBpLmdldF9hY2NvdW50c191c2luZ19nZXQoY29uc2VudF90b2tlbikKYGBgCgotIFJldHVybmluZyB1c2VyIHRyYW5zYWN0aW9uIGRldGFpbHMKCmBgYHB5dGhvbgp0cmFuc2FjdGlvbnNBcGkgPSBUcmFuc2FjdGlvbnNBcGkoYXBpQ2xpZW50KQp0cmFuc2FjdGlvbnMgPSB0cmFuc2FjdGlvbnNBcGkuZ2V0X3RyYW5zYWN0aW9uc191c2luZ19nZXQoY29uc2VudF90b2tlbiwgYWNjb3VudHMuZGF0YVswXS5faWQpCmBgYAoKLSBSZXR1cm5pbmcgdXNlciBpZGVudGl0eSBkZXRhaWxzCmBgYHB5dGhvbgppbnN0aXR1dGlvbnNfYXBpID0gSW5zdGl0dXRpb25zQXBpKGFwaUNsaWVudCkKZmVhdHVyZXMgPSBpbnN0aXR1dGlvbnNfYXBpLmdldF9pbnN0aXR1dGlvbl91c2luZ19nZXQoY29uc3RhbnRzLklOU1RJVFVUSU9OX0lEKS5mZWF0dXJlcwppZiAoIklERU5USVRZIiBpbiBmZWF0dXJlcyk6CiAgICBpZGVudGl0eV9hcGkgPSBJZGVudGl0eUFwaShhcGlDbGllbnQpCiAgICBpZGVudGl0eSA9IGlkZW50aXR5X2FwaS5nZXRfaWRlbnRpdHlfdXNpbmdfZ2V0KGNvbnNlbnRfdG9rZW4pCmBgYAoKIyMgRnVydGhlciBpbmZvcm1hdGlvbgoKRm9yIG1vcmUgaW5mb3JtYXRpb24gb24gaG93IHRvIGdldCBjb25uZWN0ZWQsIHZpc2l0IHRoZSBbWWFwaWx5IEtub3dsZWRnZSBCYXNlXShodHRwczovL2tiLnlhcGlseS5jb20pLgo= readmeEtag: '"6679845fa0b0b0596176a83fe0f33c26cbbbd044"' readmeLastModified: Fri, 25 Nov 2022 12:42:30 GMT repositoryId: 133962026 description: >- Python SDK generated against the Yapily API. This SDK can be used to connect to Open Banking entities. created: '2018-05-18T14:02:57Z' updated: '2026-02-01T22:44:14Z' language: Python archived: false stars: 15 watchers: 2 forks: 9 owner: yapily logo: https://avatars.githubusercontent.com/u/36482918?v=4 license: MIT repoEtag: '"cf92ad5ca07e35d792bcd8fa4a2d92582d7bb3457e27f9874d5213e559510a44"' repoLastModified: Sun, 01 Feb 2026 22:44:14 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/ckaratzas/tus-server-implementation v3: true repositoryMetadata: base64Readme: >- IyBUVVMgU2VydmVyIEphdmEgSW1wbGVtZW50YXRpb24KQW4gaW1wbGVtZW50YXRpb24gb2YgdGhlIFR1cyBSZXN1bWFibGUgVXBsb2FkIHByb3RvY29sIFtodHRwczovL3R1cy5pby9wcm90b2NvbHMvcmVzdW1hYmxlLXVwbG9hZC5odG1sXSBpbiBqYXZhLiBWZXJ0eC1XZWIgaXMgdXNlZCBmb3IgdGhlIGh0dHAgc3RhY2sgcGFydAphbmQgcmVkaXMgYXMgdGhlIGJhY2tlbmQgZm9yIHVwbG9hZCBpbmZvcm1hdGlvbiBtYW5hZ2VtZW50LgpFeHRlbnNpb25zIHN1cHBvcnRlZCBhcmUgY3JlYXRpb24sY2hlY2tzdW0sdGVybWluYXRpb24sY29uY2F0ZW5hdGlvbi4gVGhlIHB1cnBvc2Ugb2YgdGhpcyByZXBvc2l0b3J5IGlzIHRvIHByb3ZpZGUgYSBUdXMgcHJvdG9jb2wgaW1wbGVtZW50YXRpb24gYWdub3N0aWMKb2YgdGhlIHVuZGVybHlpbmcgc3RvcmFnZSBwcm92aWRlciB0aHVzIGFsbG93aW5nIGltcGxlbWVudG9ycyB0byBmb2N1cyBvbiB0aGVpciBidXNpbmVzcyBsb2dpYyBhbmQgc3BlY2lmaWMgbmVlZHMuCkFuIGlzc3VlIHRoYXQgaXMgbm90IGEgY29uY2VybiBvZiB0aGUgcHJvdG9jb2wgYnV0IGNvdWxkIGJlIGFuIGV4dGVuc2lvbiBpcyB0aGUgaGFuZGxpbmcgb2YgdGhlIGxvY2tzIGR1cmluZyBhIHBhdGNoIG9wZXJhdGlvbi4gU28gd2hlbiBhIHBhdGNoIGlzIGluaXRpYXRlZAphIGxvY2sgaXMgYWNxdWlyZWQgKGZvciBtb3N0IG9idmlvdXMgY2FzZXMpIHRvIGVuc3VyZSBjb25zaXN0ZW5jeS4gSWYgdGhlIHByb2Nlc3MgdHJpZ2dlcmVkIGZyb20gcGF0Y2ggZmFpbHMgYW5kIGxvY2sgaXMgbm90IHJlbGVhc2VkIHRoZXJlIGlzIGEgCnBoYW50b20gbG9jayByZW1haW5pbmcuIFRoaXMgZWRnZSBjYXNlIGNvdWxkIGJlIG1pdGlnYXRlZCBieSBlaXRoZXIgbWFraW5nIHRoZSBzZXJ2ZXIgc3RpY2t5IGFuZCBrZWVwaW5nIHRoZSBsb2NrcyBpbi1wcm9jZXNzIG9yIGJ5IHBlcmhhcHMgaXNzdWluZyBhIApyZWxlYXNlIHJlcXVlc3Qgd2l0aCBhIGxvY2sgdG9rZW4gb2J0YWluZWQgYnkgdGhlIGluaXRpYXRvciBvZiB0aGUgdXBsb2FkLgpObyBhdXRoZW50aWNhdGlvbiB2YWx2ZXMgYXJlIGltcGxlbWVudGVkIGhlcmUgYWxzby4KCiMgSW5zdHJ1Y3Rpb25zClRvIGp1c3QgYnVpbGQgdGhlIHR1cy1zZXJ2ZXItaW1wbGVtZW50YXRpb24ganVzdDoKCmBgYGNkIDxyb290PmBgYAoKYGBgbXZuIGNsZWFuIGluc3RhbGxgYGAKClRvIGJ1aWxkIHRoZSBkb2NrZXIgaW1hZ2UgZm9yIHRoZSB0dXMtc2VydmVyLWltcGxlbWVudGF0aW9uIGp1c3QgKGRvY2tlciBlZGdlIHJlbGVhc2UgW2h0dHBzOi8vZG9jcy5kb2NrZXIuY29tL2VkZ2UvXSBpcyByZXF1aXJlZCEpOgoKYGBgY2QgPHJvb3Q+YGBgCgpgYGBkb2NrZXIgYnVpbGQgLXQgdHVzX3NlcnZlciAuYGBgCgpJbiBvcmRlciB0byBydW4gdHVzLXNlcnZlci1pbXBsZW1lbnRhdGlvbiBhbG9uZyB3aXRoIGFsbCBkZXBlbmRlbmNpZXMganVzdCAoYXNzdW1pbmcgeW91IGFscmVhZHkgaGF2ZSB0aGUgZG9ja2VyIGltYWdlIGJ1aWx0IGZyb20gYWJvdmUpOgoKYGBgY2QgPHJvb3Q+YGBgCgpgYGBkb2NrZXItY29tcG9zZSAtZiB0dXMtc2VydmVyLnltbCB1cGBgYAoKSW4gb3JkZXIgdG8gc2V0dXAgdGhlIHR1cy1zZXJ2ZXItaW1wbGVtZW50YXRpb24gZm9yIGRldmVsb3BtZW50IGZyb20geW91ciBmYXZvcml0ZSBJREUgcmVkaXMgcnVubmluZyBpcyByZXF1aXJlZDoKCmBgYGNkIDxyb290PmBgYAoKYGBgZG9ja2VyLWNvbXBvc2UgLWYgcmVkaXMueW1sIHVwYGBgCgpBZnRlciB0aGF0IGluIG9yZGVyIHRvIHJ1biB0aGUgdHVzLXNlcnZlci1pbXBsZW1lbnRhdGlvbiBmcm9tIGluc2lkZSB5b3VyIGZhdm9yaXRlIElERSBqdXN0OgpSdW4gdGhlIGNvbS50dXMub3NzLnNlcnZlci5hcHBsaWNhdGlvbi5BcHBsaWNhdGlvbiBtYWluIGNsYXNzIAp3aXRoIHByb2dyYW0gYXJndW1lbnRzOgogLWMgPHJvb3Q+L2NvbmZpZ3VyYXRpb24vIC1iIHR1cy1zZXJ2ZXItYmVhbnMueG1sCmFuZCBWTSBwYXJhbWV0ZXJzOiAtRGxvZ2dpbmcuY29uZmlnPWZpbGU6PHJvb3Q+L2NvbmZpZ3VyYXRpb24vbG9nYmFjay54bWwKUmVkaXMgbXVzdCBiZSBsaXZlIGFsc28gKHNlZSBhYm92ZSBob3cgdG8gcnVuIGl0KQoKIyBUZXN0aW5nCkluIHRoZSB0ZXN0IGZvbGRlcjoKVGhlcmUgaXMgYSB2ZXJ5IHNpbXBsZSB1cGxvYWQgdGVzdCB0aGF0IHVzZXMgdHVzLWphdmEtY2xpZW50IFtodHRwczovL2dpdGh1Yi5jb20vdHVzL3R1cy1qYXZhLWNsaWVudF0uIApUaGVyZSBhcmUgYWxzbyBjdXJsIHRlc3RzIHRoYXQgdGVzdCBzaW1wbGUgdXBsb2FkIGFuZCBwYXJ0aWFsIHVwbG9hZHMgKGNvbmNhdGVuYXRpb24gZXh0ZW5zaW9uKQo= readmeEtag: '"b8fbdc615a000a14734483efc3465bad6c840924"' readmeLastModified: Sun, 04 Feb 2018 21:09:29 GMT repositoryId: 118587778 description: >- A java implementation of https://tus.io/protocols/resumable-upload.html protocol with vertx-web. created: '2018-01-23T09:26:44Z' updated: '2024-03-22T12:02:19Z' language: Java archived: false stars: 13 watchers: 2 forks: 1 owner: ckaratzas logo: https://avatars.githubusercontent.com/u/35725079?v=4 license: MIT repoEtag: '"d9b007a599bad97d6beafb8f54f134575587edad500b5e9d3a401f9e3be98a4e"' repoLastModified: Fri, 22 Mar 2024 12:02:19 GMT foundInMaster: true category: Server Implementations id: e81e01372ba901858d7fe6cb28c44c60 - source: openapi3 tags repository: https://github.com/arbs-io/api-studio-visualstudio v3: true id: 48eafce82bf409d993a951b9780d6826 repositoryMetadata: base64Readme: >- ![Visual Studio Marketplace Version](https://img.shields.io/visual-studio-marketplace/v/AndrewButson.ApiStudio)
![Visual Studio Marketplace Installs](https://img.shields.io/visual-studio-marketplace/i/AndrewButson.ApiStudio)
![Visual Studio Marketplace Rating](https://img.shields.io/visual-studio-marketplace/r/AndrewButson.ApiStudio)
[![GitHub issues](https://img.shields.io/github/issues/arbs-io/api-studio-visualstudio.svg)](https://github.com/arbs-io/api-studio-visualstudio/issues)

[![Maintainability Rating](https://sonarcloud.io/api/project_badges/measure?project=arbs-io_api-studio-visualstudio&metric=sqale_rating)](https://sonarcloud.io/summary/new_code?id=arbs-io_api-studio-visualstudio)
[![Security Rating](https://sonarcloud.io/api/project_badges/measure?project=arbs-io_api-studio-visualstudio&metric=security_rating)](https://sonarcloud.io/summary/new_code?id=arbs-io_api-studio-visualstudio)
[![Reliability Rating](https://sonarcloud.io/api/project_badges/measure?project=arbs-io_api-studio-visualstudio&metric=reliability_rating)](https://sonarcloud.io/summary/new_code?id=arbs-io_api-studio-visualstudio)
[![Bugs](https://sonarcloud.io/api/project_badges/measure?project=arbs-io_api-studio-visualstudio&metric=bugs)](https://sonarcloud.io/summary/new_code?id=arbs-io_api-studio-visualstudio)
[![Vulnerabilities](https://sonarcloud.io/api/project_badges/measure?project=arbs-io_api-studio-visualstudio&metric=vulnerabilities)](https://sonarcloud.io/summary/new_code?id=arbs-io_api-studio-visualstudio)

## **What is Api Studio?**

Api Studio is a productivity extension for Microsoft Visual Studio. The extension bridges the role of architect and developer, providing rapid prototyping and promoting industry best practices. In addition, Api Studio provides templating to guarantee a consistent look and feel regardless of organisation or project size.

## **Visual Studio Extension**

Api Studio Extension enhances Visual Studio 2022+ by providing a designer, toolbox, explorer and code generation engine to create and manage Api Studio assets. Promoting An API-first approach means that your APIs are treated as "first-class citizens for any development project." Api Studio provides architects and developers with a solution to quickly design and collaborate, providing the blueprint for solution implementation. In addition, the extension will enable teams to quickly mock solution endpoints, proofing usability before investing in development.

## **How does Api Studio work?**

Api Studio contains a visual studio designer, domain-specific language (DSL) and code generation engine to provide an exceptional experience for building modern resource-based solutions.

**Download** [Api Studio Extension for Visual Studio](https://marketplace.visualstudio.com/items?itemName=AndrewButson.ApiStudio) from visual studio marketplace

## **Features**

The extension provides the following features.

- <span style="color:#267F00;">**Designer**</span>: The Visual Studio Api Studio Designer is dedicated to editing Api Studio DSL artefacts used by code generated directly in the Visual Studio environment. To open a .ApiStudio file, the designer opens in the central docking area within visual studio. If there is no such file, you may create it using the Add --> New Item --> ApiStudio.

- <span style="color:#B200FF;">**Toolbox**</span>: The Toolbox window displays controls that you can add to Visual Studio projects. Choose View > Toolbox from the menu bar or press Ctrl+Alt+X to open Toolbox. You can drag and drop different controls onto the surface of the Api Studio designer and resize and position the controls. Toolbox appears in conjunction with Api Studio designer views. Toolbox displays only those controls that the current designer can use. You can search within Toolbox to further filter the items that appear.

- <span style="color:#0094FF;">**Api Studio Explorer**</span>: You can use the Explorer tool window to create & manage your solutions and projects and to view & interact with your code.

- <span style="color:#FFD800;">**Property**</span>: The Visual Studio Properties window is a property browser for Api Studio components. The Properties window list metadata at design time for the currently selected object in any other window in the integrated development environment (IDE).

![api-studio-visualstudio-overview-highlighted__1.png](images/api-studio-visualstudio-overview-highlighted.png)

### **Visual Studio Projects**

Api Studio provides a visual studio project containing all assets to get started. The template offers a complete but lightweight solution.

### Available Project Templates

|                                          Projects                                           | Language | Library        | Framework | Version | Status | Comments                                                                     |
| :-----------------------------------------------------------------------------------------: | -------- | -------------- | --------- | ------- | :----: | ---------------------------------------------------------------------------- |
| ![csharp-azurefunction-dotnet6](images/api-studio.codegen.csharp-azurefunction-dotnet6.png) | csharp   | azure_function | net6.0    | 1.0.0   |   ✅   |
|    ![csharp-minimalapi-dotnet6](images/api-studio.codegen.csharp-minimalapi-dotnet6.png)    | csharp   | minimum_api    | net6.0    | 1.0.0   |   ✅   | net6.0 openapi specification limitations, anticipated full support in net7.0 |

### Template Standards

Api Studio provides templated standards for header, response codes and security models. Allowing enterprise standards to be shared between development teams. The options are available within visual studios options Tools --> Options --> ApiStudio. The options panel provide Import/Export functionality.

![api-studio-visualstudio-options.png](images/api-studio-visualstudio-options.png)

### Api Studio Designer

The api-studio files are located in the `"\ApiStudio"` folder within the project. To open the designer double click the file, or right click and select `"Open"`

![api-studio-visualstudio-project.png](images/api-studio-visualstudio-project.png)

## **Getting Started**

Creating a new project is simple. Visual Studio "Create Project", filter using project type "Api Studio".

![api-studio-project-template-filter.png](images/api-studio-project-template-filter.png)

## **What's next**

This extension currently supports RESTful-based APIs, but extensions to allow support for gRpc, WebHooks and WebSocket interfaces soon.

## **How can I help?**

If you enjoy using the extension, please give it a rating on the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=AndrewButson.ApiStudio).

Should you encounter bugs or if you have feature requests, head on over to the [GitHub repo](https://github.com/arbs-io/api-studio-visualstudio) to open an issue if one doesn't already exist.
Pull requests are also very welcome since I can't always get around to fixing all bugs myself.

This is a personal passion project, so my time is limited.

Another way to help out is to [sponsor me on GitHub](https://github.com/sponsors/arbs-io).

## **Copyright**

Copyright 2023 Api Studio, Inc.

Licensed under the MIT License
 readmeEtag: '"8e6b2f42fc70b9ab01ac2e06e7fdd87d58cf39b4"' readmeLastModified: Mon, 13 Feb 2023 21:09:35 GMT repositoryId: 448539420 description: >- Visual studio plug-in to create clean RESTful APIs. The plug-in provides a quick and easy ways to scaffold HTTP end-points following RFC and best practice. created: '2022-01-16T11:52:38Z' updated: '2025-05-28T04:28:35Z' language: C# archived: false stars: 12 watchers: 1 forks: 9 owner: arbs-io logo: https://avatars.githubusercontent.com/u/71968891?v=4 license: MIT repoEtag: '"df0baf455fc8d3542afb4ff7215f24da83a5966d5b9bc273741d4e58c5357871"' repoLastModified: Wed, 28 May 2025 04:28:35 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/bob-cd/wendy v3: true id: fb84178d96d3cb087e847830caeb1ba4 repositoryMetadata: base64Readme: >- IyBXZW5keQoKIyMgQm9iJ3MgcmVmZXJlbmNlIENMSSBhbmQgY29tbWFuZCBjZW50cmUKCiMjIFN0YXR1cwpBbHBoYQoKW0RvY3NdKGh0dHBzOi8vYm9iLWNkLmdpdGh1Yi5pby9jbGkvI3dlbmR5KQoKIyMgTGljZW5zZQpXZW5keSBsaWtlIEJvYiBpcyBbRnJlZV0oaHR0cHM6Ly93d3cuZ251Lm9yZy9waGlsb3NvcGh5L2ZyZWUtc3cuZW4uaHRtbCkgYW5kIE9wZW4gU291cmNlIGFuZCBhbHdheXMgd2lsbCBiZS4gTGljZW5zZWQgZnVsbHkgdW5kZXIgW01JVF0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQpCg== readmeEtag: '"c3d09bf4bf3b0a0b14d9cd412b6fa950aa110ce4"' readmeLastModified: Sat, 04 Jan 2025 08:29:36 GMT repositoryId: 174202758 description: Bob's command centre and the reference CLI/TUI created: '2019-03-06T18:57:09Z' updated: '2026-02-05T08:21:05Z' language: Go archived: false stars: 14 watchers: 8 forks: 6 owner: bob-cd logo: https://avatars.githubusercontent.com/u/32014218?v=4 license: MIT repoEtag: '"52b124cd93dcbad7751c1a273aa4bdec787da60b3383b4b7b03ee92808539cd7"' repoLastModified: Thu, 05 Feb 2026 08:21:05 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/pglass/py-openapi-schema-to-json-schema v3: true repositoryMetadata: base64Readme: >- Overview
--------

[![Build Status](https://travis-ci.org/pglass/py-openapi-schema-to-json-schema.svg?branch=master)](https://travis-ci.org/pglass/py-openapi-schema-to-json-schema)
[![PyPI](https://img.shields.io/pypi/v/py-openapi-schema-to-json-schema.svg)](https://pypi.org/project/py-openapi-schema-to-json-schema/)
[![PyPI - Python Version](https://img.shields.io/pypi/pyversions/py-openapi-schema-to-json-schema.svg)](https://pypi.org/project/py-openapi-schema-to-json-schema/)


**This is a straight Python port of the MIT-licensed
[mikunn/openapi-schema-to-json-schema](https://github.com/mikunn/openapi-schema-to-json-schema)
([v2.1.0](https://github.com/mikunn/openapi-schema-to-json-schema/tree/v2.1.0))**.
This port is similarly MIT-licensed.

It converts from [OpenAPI 3.0](
https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md) to
[JSON Schema Draft 4](http://json-schema.org/specification-links.html#draft-4).

## Why?

OpenAPI 3 Schemas and JSON Schemas are mostly similar. However, JSON Schema
validators are unaware of the differences between the two formats. This means
that validating request/response JSON using a standard JSON Schema validator
with OpenAPI 3 Schemas will result in incorrect validations in certain common
cases.

One way to solve this problem is to translate the OpenAPI 3 schema to JSON
Schema, which is the purpose of this library.

See [here](https://github.com/mikunn/openapi-schema-to-json-schema/tree/v2.1.0#why)
for more rationale, as well as [Phil Sturgeon's blog post](
https://philsturgeon.uk/api/2018/03/30/openapi-and-json-schema-divergence/)
about the problem.

## Features

* converts OpenAPI 3.0 Schema Object to JSON Schema Draft 4
* deletes `nullable` and adds `"null"` to `type` array if `nullable` is `true` and `type` is present
* adds `{"type": "null"}` to `oneOf` or `anyOf` array if `nullable` is `true` and `type` is _not_ present
* supports deep structures with nested `allOf`s etc.
* removes [OpenAPI specific
  properties](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#fixed-fields-20)
such as `discriminator`, `deprecated` etc. unless specified otherwise
* optionally supports `patternProperties` with `x-patternProperties` in the
  Schema Object

**NOTE**: `$ref`s are not dereferenced. You will need another library to
read the spec and follow `$ref` fields.


## Installation

```bash
$ pip install py-openapi-schema-to-json-schema
```


## Usage

```python
import json
from openapi_schema_to_json_schema import to_json_schema

openapi_schema = {
    "type": "object",
    "properties": {
        "name": {
            "type": "string",
            "nullable": True,
        }
    },
    "x-patternProperties": {
        "^[a-z]+$": {
            "type": "number",
        }
    }
}

options = {"supportPatternProperties": True}
converted = to_json_schema(openapi_schema, options)

print(json.dumps(converted, indent=2))
```

This outputs the following JSON schema. This shows the conversion of
`nullable: True` to `type: ["string", "null"]`, and the enablement of
unsupported JSON Schema features using OpenAPI extension fields
(`x-patternProperties` -> `patternProperties`).

```json
{
  "patternProperties": {
    "^[a-z]+$": {
      "type": "number"
    }
  },
  "properties": {
    "name": {
      "type": [
        "string",
        "null"
      ]
    }
  },
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object"
}
```

### Options

The `to_json_schema` function accepts an `options` dictionary as the second
argument.

```python
# Defaults
options = {
    'cloneSchema': True,
    'dateToDateTime': False,
    'supportPatternProperties': False,
    'keepNotSupported': [],
    'patternPropertiesHandler':
        openapi_schema_to_json_schema.patternPropertiesHandler,
    'removeReadOnly': False,
    'removeWriteOnly': True,
}
```

#### `cloneSchema` (bool)

If set to `False`, converts the provided schema in place. If `True`, clones the
schema using `copy.deepcopy`. Defaults to `True`.

#### `dateToDateTime` (bool)

This is `False` by default and leaves `date` format as is. If set to `True`,
sets `format: 'date'` to `format: 'date-time'`.

For example

```python
import json
from openapi_schema_to_json_schema import to_json_schema

schema = {
  'type': 'string',
  'format': 'date',
}

converted = to_json_schema(schema, {'dateToDateTime': True})

print(json.dumps(converted, indent=2))
```

prints

```json
{
  "format": "date-time",
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "string"
}
```

#### `keepNotSupported` (list)

By default, the following fields are removed from the result schema:
`nullable`, `discriminator`, `readOnly`, `writeOnly`, `xml`, `externalDocs`,
`example` and `deprecated` as they are not supported by JSON Schema Draft 4.
Provide a list of the ones you want to keep (as strings) and they won't be
removed.

#### `removeReadOnly` (bool)

If set to `True`, will remove properties set as `readOnly`. If the property is
set as `required`, it will be removed from the `required` list as well. The
property will be removed even if `readOnly` is set to be kept with
`keepNotSupported`.

#### `removeWriteOnly` (bool)

Similar to `removeReadOnly`, but for `writeOnly` properties.

#### `supportPatternProperties` (bool)

If set to `True` and `x-patternProperties` property is present, change
`x-patternProperties` to `patternProperties` and call
`patternPropertiesHandler`. If `patternPropertiesHandler` is not defined, call
the default handler. See `patternPropertiesHandler` for more information.

#### `patternPropertiesHandler` (function)

Provide a function to handle pattern properties and set
`supportPatternProperties` to take effect. The function takes the schema where
`x-patternProperties` is defined on the root level. At this point
`x-patternProperties` is changed to `patternProperties`. It must return the
modified schema.

If the handler is not provided, the default handler is used. If
`additionalProperties` is set and is an object, the default handler sets it to
false if the `additionalProperties` object has deep equality with a pattern
object inside `patternProperties`. This is because we might want to define
`additionalProperties` in OpenAPI spec file, but want to validate against a
pattern. The pattern would turn out to be useless if `additionalProperties` of
the same structure were allowed. Create you own handler to override this
functionality.

See `tests/to_jsonschema/test_pattern_properties.py` for examples of this.


Credits
-------

- [mikunn](https://github.com/mikunn) for the [original
openapi-schema-to-json-schema](https://github.com/mikunn/openapi-schema-to-json-schema)
- [Phil Sturgeon](https://github.com/philsturgeon) for his great 
[blog post](https://philsturgeon.uk/api/2018/03/30/openapi-and-json-schema-divergence/)
about the issue (and his [reverse implementation](https://github.com/wework/json-schema-to-openapi-schema))
 readmeEtag: '"ed09d8d2c80770ebf18b14fa5917a334d1f42bed"' readmeLastModified: Sat, 25 Jul 2020 05:42:26 GMT repositoryId: 138369054 description: Python port of https://github.com/mikunn/openapi-schema-to-json-schema created: '2018-06-23T03:46:02Z' updated: '2025-07-01T12:59:50Z' language: Python archived: false stars: 14 watchers: 2 forks: 2 owner: pglass logo: https://avatars.githubusercontent.com/u/1077740?v=4 license: MIT repoEtag: '"034cb7b0c383d47cafaa405d3b9fc0239c71c174a3ba10b639970aba835b5718"' repoLastModified: Tue, 01 Jul 2025 12:59:50 GMT foundInMaster: true category: - Converters - Parsers id: c66a109574c000814bbca0c6afab6254 - source: openapi3 tags repository: https://github.com/python-lapidary/lapidary v3: true id: 6c315a3d79c2bbf8ff1d1567f057f64b repositoryMetadata: base64Readme: >- IyBMYXBpZGFyeQoKWyFbdGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1sYXBpZGFyeS9sYXBpZGFyeS9hY3Rpb25zL3dvcmtmbG93cy90ZXN0X3B1Ymxpc2gueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vcHl0aG9uLWxhcGlkYXJ5L2xhcGlkYXJ5L2FjdGlvbnMvd29ya2Zsb3dzL3Rlc3RfcHVibGlzaC55YW1sKQoKUHl0aG9uIEhlbHBlciBmb3IgV2ViIEFQSSBjbGllbnRzLgoKIyMgV2h5CgpXZWIgQVBJIGNsaWVudHMgZm9sbG93IGEgcmVsYXRpdmVseSBzbWFsbCBzZXQgb2YgcGF0dGVybnMgYW5kIGltcGxlbWVudGluZyB0aGVtIGlzIHJhdGhlciByZXBldGl0aXZlIHRhc2suClByZXBhcmUgcmVxdWVzdCwgbWFrZSB0aGUgY2FsbCwgaGFuZGxlIHJlc3BvbnNlIHN0YXR1cywgZGVzZXJpYWxpemUgdGhlIGJvZHkKVHlwaWNhbCBleGFtcGxlcyBzaG93IGhvdyB0byB1c2UgYHJlcXVlc3QuZ2V0KClgIG9yIHNpbWlsYXIgbWV0aG9kLCBidXQgdGhpcyBpcyBhbiBhbnRpLXBhdHRlcm4uIFRoZXNlIGNhbGxzIHNob3VsZCBiZSBlbmNhcHN1bGF0ZWQgYXMgbW9kdWxlIGZ1bmN0aW9ucyBvciBtZXRob2RzLgoKIyMgSG93CgpMYXBpZGFyeSBpcyBhIGxpYnJhcnkgdGhhdCBwcm92aWRlcyBkZWNvcmF0b3JzIGFuZCBhbm5vdGF0aW9ucyBmb3IgZGVzY3JpYmluZyBXZWIgQVBJcyBpbiBhIHdheSBzaW1pbGFyIHRvIE9wZW5BUEkuCkluIGZhY3QgW2xhcGlkYXJ5IHJlbmRlcl0oaHR0cHM6Ly9naXRodWIuY29tL3B5dGhvbi1sYXBpZGFyeS9sYXBpZGFyeS1yZW5kZXIvKSBjYW4gY29udmVydCBtdWNoIG9mIE9wZW5BUEkgMy4wIHRvIExhcGlkYXJ5IGNvZGUuCgpBdCBydW50aW1lLCB0aGUgbGlicmFyeSBpbnRlcnByZXRzIHVzZXItcHJvdmlkZWQgZnVuY3Rpb24gZGVjbGFyYXRpb25zIGFuZCBtYWtlcyB0aGVtIGJlaGF2ZSBhcyBzcGVjaWZpZWQuCklmIGEgZnVuY3Rpb24gYWNjZXB0cyBwYXJhbWV0ZXIgb2YgdHlwZSBgWGAgYW5kIHJldHVybnMgYFlgLCBMYXBpZGFyeSB3aWxsIHRyeSB0byBjb252ZXJ0IGBYYCB0byBIVFRQIHJlcXVlc3QgYW5kIHRoZSByZXNwb25zZSBiYWNrIHRvIGBZYC4KCkNoZWNrIHRoZSBbZXhhbXBsZV0oaHR0cHM6Ly9weXRob24tbGFwaWRhcnkuZ2l0aHViLmlvL2xhcGlkYXJ5LyN1c2FnZSkK readmeEtag: '"f82617a4dbba47b4cc1895194c323704b1d64f93"' readmeLastModified: Tue, 23 Sep 2025 20:47:49 GMT repositoryId: 530807671 description: Write Web API clients using annotations in python created: '2022-08-30T19:50:12Z' updated: '2026-01-12T01:12:40Z' language: Python archived: false stars: 16 watchers: 1 forks: 0 owner: python-lapidary logo: https://avatars.githubusercontent.com/u/111730600?v=4 license: MIT repoEtag: '"83f80db413f685096cd4704d68e1c68c236c7cf09548dadc553735760052afc2"' repoLastModified: Mon, 12 Jan 2026 01:12:40 GMT category: - Parsers - SDK foundInMaster: true oldLocations: - https://github.com/lapis-project/lapis - source: openapi3 tags repository: https://github.com/yapily/yapily-openapi v3: true repositoryMetadata: base64Readme: >- IyBZYXBpbHktb3BlbmFwaTMtc3BlY2lmaWNhdGlvbnMKCiFbU2NoZW1hIFZhbGlkYXRvcl0oaHR0cHM6Ly92YWxpZGF0b3Iuc3dhZ2dlci5pby92YWxpZGF0b3I/dXJsPWh0dHBzOi8vb3BlbmFwaS55YXBpbHkuY29tL29wZW5hcGkuanNvbikKCllhcGlseSB1c2VzIHRoZSBgT3BlbkFQSSAzLjAuMWAgc3BlY2lmaWNhdGlvbiB0byBzY2hlbWF0aXplIG91ciBbZG9jc10oaHR0cHM6Ly9kb2NzLnlhcGlseS5jb20vKSBhbmQgZ2VuZXJhdGUgb3VyIHN1cHBvcnRlZCBjbGllbnQgbGlicmFyaWVzLiBUaGlzIHByb3ZpZGVzIGEgY29uc2lzdGVudCBkZXZlbG9wZXIgZXhwZXJpZW5jZSBhY3Jvc3Mgb3VyIGV4dGVybmFsIGludGVyZmFjZXMuCgotIFtZYXBpbHktb3BlbmFwaTMtc3BlY2lmaWNhdGlvbnNdKCN5YXBpbHktb3BlbmFwaTMtc3BlY2lmaWNhdGlvbnMpCiAgLSBbSW5zdGFsbGluZyB0aGUgSmF2YSBkZXBlbmRlbmNpZXNdKCNpbnN0YWxsaW5nLXRoZS1qYXZhLWRlcGVuZGVuY2llcykKICAtIFtKYXZhIHN1cHBvcnRdKCNqYXZhLXN1cHBvcnQpCiAgLSBbTm9kZSBzdXBwb3J0XSgjbm9kZS1zdXBwb3J0KQogIC0gW1B5dGhvbiBzdXBwb3J0XSgjcHl0aG9uLXN1cHBvcnQpCiAgLSBbS25vd24gaXNzdWVzIHdpdGggb3BlbmFwaS1nZW5lcmF0b3JdKCNrbm93bi1pc3N1ZXMtd2l0aC1vcGVuYXBpLWdlbmVyYXRvcikKICAtIFtFeGFtcGxlc10oI2V4YW1wbGVzKQoKIyMgSW5zdGFsbGluZyB0aGUgSmF2YSBkZXBlbmRlbmNpZXMKClRoZSBPcGVuQVBJIGdlbmVyYXRvciBpcyBhIG1hdmVuIHBsdWdpbiB1c2VkIHRvIGdlbmVyYXRlIHNlcnZlciBhbmQgY2xpZW50IGxpYnJhcmllcyBmcm9tIGEgZ2l2ZW4gT0FTIGZpbGUuCgpGaXJzdCByZXF1aXJlbWVudCB0byB1c2UgdGhlIE9wZW5BUEkgZ2VuZXJhdG9yIGlzIHRvIGluc3RhbGwgSmF2YS4gCgpUaGUgZm9sbG93aW5nIFt0dXRvcmlhbF0oaHR0cHM6Ly9jbG91ZC5nb29nbGUuY29tL2phdmEvZG9jcy9zZXR1cCNpbnN0YWxsX2FfamRrX2phdmFfZGV2ZWxvcG1lbnRfa2l0KSBwcm92aWRlcyBhbiBleHRlbnNpdmUgc3RlcC1ieS1zdGVwIGd1aWRlIG9uIGhvdyB0byBzZXQgdXAgYW5kIGNvbmZpZ3VyZSB5b3VyIGRldmVsb3BtZW50IGVudmlyb25tZW50LgoKIyMgSmF2YSBzdXBwb3J0CgpQbGVhc2UgcmVmZXIgdG8gdGhpcyBbZG9jdW1lbnRdKC9tYWluLy4uL2RvY3MvamF2YS5tZCkgZm9yIEphdmEuCgojIyBOb2RlIHN1cHBvcnQKClBsZWFzZSByZWZlciB0byB0aGlzIFtkb2N1bWVudF0oL21haW4vLi4vZG9jcy9ub2RlLm1kKSBmb3IgTm9kZS4KCiMjIFB5dGhvbiBzdXBwb3J0CgpQbGVhc2UgcmVmZXIgdG8gdGhpcyBbZG9jdW1lbnRdKC9tYWluLy4uL2RvY3MvcHl0aG9uLm1kKSBmb3IgUHl0aG9uLgoKIyMgS25vd24gaXNzdWVzIHdpdGggb3BlbmFwaS1nZW5lcmF0b3IKClRoZSBbb3BlbmFwaS1nZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IpIGdlbmVyYXRpb24gc3R5bGVzIHZhcnkgZGVwZW5kaW5nIG9uIHRoZSBjaG9zZW4gbGFuZ3VhZ2U6CgotIFdlIHJlZ2FyZCB0aGUgZXh0ZW5zaW9uIG9mIGVudW1lcmF0aW9ucyBpbiB0aGUgWWFwaWx5IEFQSSB0byBiZSBhIG5vbi1icmVha2luZyBjaGFuZ2UsIGFuZCB3ZSBtYXkgYWRkIG5ldyB2YWx1ZXMgd2l0aG91dCBhZHZhbmNlIHdhcm5pbmcuIFBsZWFzZSBub3RlIHRoYXQgc29tZSBPcGVuQVBJIGdlbmVyYXRvcnMgYXV0b21hdGljYWxseSBlbXBsb3kgZW51bSB2YWxpZGF0aW9uIGJ5IGRlZmF1bHQsIGhvd2V2ZXIgd2UgZXhwZWN0IGNsaWVudHMgdG8gZGlzYWJsZSBzdHJpY3QgZW51bSB2YWxpZGF0aW9uIGZvciByZXNwb25zZXMgaW4geW91ciBnZW5lcmF0ZWQgbGlicmFyaWVzLiBPdGhlcndpc2UsIHlvdXIgY2xpZW50cyB3aWxsIGV4cGVyaWVuY2UgZXJyb3JzIHdoZW4gbmV3IGVudW0gdmFsdWVzIGFyZSBhZGRlZC4KCiMjIEV4YW1wbGVzCgpGb3IgYSBiZXR0ZXIgdmlldyBvZiB0aGUgaW1wbGVtZW50YXRpb24gcHJvY2VzcywgY2hlY2sgb3V0IG91ciBbZXhhbXBsZXNdKC9tYWluLy4uL2V4YW1wbGVzL1JFQURNRS5tZCkgc2VjdGlvbiB3aGljaCBwcmVzZW50cyBhIHN0cmF0ZWd5IGZvciBpbXBsZW1lbnRpbmcgZWFzaWx5IHdpdGggWWFwaWx5IGluIHlvdXIgbGFuZ3VhZ2Ugb2YgY2hvaWNlLgo= readmeEtag: '"cd917f5a69109701b406eb7f7ba6252573c6f9a8"' readmeLastModified: Mon, 05 Aug 2024 11:31:19 GMT repositoryId: 446895316 description: >- Yapily’s OpenAPI Specification. Use this complete definition of our API to generate client libraries in the language of your choice. created: '2022-01-11T16:20:08Z' updated: '2026-01-21T10:13:00Z' language: null archived: false stars: 14 watchers: 3 forks: 2 owner: yapily logo: https://avatars.githubusercontent.com/u/36482918?v=4 repoEtag: '"cca5d5b711490f5f62bf89f9551b43d5f5fb860410605dcd3a8765f33a8d060f"' repoLastModified: Wed, 21 Jan 2026 10:13:00 GMT foundInMaster: true category: - SDK - Parsers id: 480ed47307537545205cc96bd408fb48 - source: openapi3 tags repository: https://github.com/mkbeh/fastapi-admin-panel v3: true repositoryMetadata: base64Readme: >- IyBGYXN0QVBJIGFuZCBQb3N0Z3JlU1FMIC0gQWRtaW4gUGFuZWwKCgpbIVtQeXRob24gMy45XShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3B5dGhvbi0zLjktYmx1ZS5zdmcpXShodHRwczovL3d3dy5weXRob24ub3JnL2Rvd25sb2Fkcy9yZWxlYXNlL3B5dGhvbi0zOTAvKQohW1BsYXRmb3JtXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3BsYXRmb3JtLWxpbnV4LWdyZWVuLnN2ZykKWyFbR2l0SHViIGxpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvTmFlcmVlbi9TdHJhcERvd24uanMuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL05hZXJlZW4vU3RyYXBEb3duLmpzL2Jsb2IvbWFzdGVyL0xJQ0VOU0UpCgoKQmFja2VuZCBzdGFjayB1c2luZyBQeXRob24gLCBpbmNsdWRpbmcgaW50ZXJhY3RpdmUgQVBJIGRvY3VtZW50YXRpb24gYW5kIG1hbnkgdXNlZnVsIGZlYXR1cmVzIG91dCBvZiB0aGUgYm94LgoKCiMjIEludGVyYWN0aXZlIEFQSSBkb2N1bWVudGF0aW9uCgohW1N3YWdnZXJdKGltZy9zd2FnZ2VyX2RvY3MucG5nKQoKCiMjIEFsdGVybmF0aXZlIEFQSSBkb2N1bWVudGF0aW9uCgohW1JlZG9jXShpbWcvcmVkb2NfZG9jcy5wbmcpCgoKIyMgRmVhdHVyZXMKCiogRnVsbCAqKkRvY2tlcioqIGludGVncmF0aW9uIChEb2NrZXIgYmFzZWQpLgoKKiAqKkRvY2tlciBDb21wb3NlKiogaW50ZWdyYXRpb24uCgoqICoqUHJvZHVjdGlvbiByZWFkeSoqIFB5dGhvbiB3ZWIgc2VydmVyIHVzaW5nIFV2aWNvcm4uCgoqIFB5dGhvbiAqKkZhc3RBUEkqKiBiYWNrZW5kLgoKKiAqKlNlY3VyZSBwYXNzd29yZCoqIGhhc2hpbmcgYnkgZGVmYXVsdC4KCiogKipKV1QgdG9rZW4qKiBhdXRoZW50aWNhdGlvbi4KCiogKipTUUxBbGNoZW15KiogbW9kZWxzLgoKKiAqKkFsZW1iaWMqKiBtaWdyYXRpb25zLgoKKiAqKkNPUlMqKiAoQ3Jvc3MgT3JpZ2luIFJlc291cmNlIFNoYXJpbmcpLgoKKiBSRVNUIGJhY2tlbmQgdGVzdHMgYmFzZWQgb24gKipQeXRlc3QqKiwgaW50ZWdyYXRlZCB3aXRoIERvY2tlciwgc28geW91IGNhbiB0ZXN0IHRoZSBmdWxsIEFQSSBpbnRlcmFjdGlvbiwgaW5kZXBlbmRlbnQgb24gdGhlIGRhdGFiYXNlLiBBcyBpdCBydW5zIGluIERvY2tlciwgaXQgY2FuIGJ1aWxkIGEgbmV3IGRhdGEgc3RvcmUgZnJvbSBzY3JhdGNoIGVhY2ggdGltZS4KCiogKipTZWN1cmUgZG9jcyoqIGJlaGluZCBiYXNpYyBodHRwIGF1dGguCgoqICoqU2VudHJ5KiogaW50ZWdyYXRpb24gKGFwcGxpY2F0aW9uIG1vbml0b3JpbmcgYW5kIGVycm9yIHRyYWNraW5nKS4KCiogKipTZW5kR3JpZCoqIGludGVncmF0aW9uIChlbWFpbCBkZWxpdmVyeSBzZXJ2aWNlKS4KCiogKipFbWFpbCBjb25maXJtYXRpb24qKiB3aGVuIHJlZ2lzdGVyaW5nIGEgdXNlci4KCiogSW50ZWdyYXRpb24gd2l0aCAqKlNvY2lhbCBOZXR3b3JrcyoqIChWa29udGFrZSwgRmFjZWJvb2ssIEdtYWlsKS4KCiogQmVhdXR5IG1peGlucyBmb3IgKipBc3luY2hyb25vdXMgU1FMQWxjaGVteSoqLgoKCiMjIEhvdyB0byB1c2UgaXQKCiMjIyBHZW5lcmF0ZSBwYXNzd29yZHMKCllvdSB3aWxsIGJlIGFza2VkIHRvIHByb3ZpZGUgcGFzc3dvcmRzIGFuZCBzZWNyZXQga2V5cyBmb3Igc2V2ZXJhbCBjb21wb25lbnRzLiBPcGVuIGFub3RoZXIgdGVybWluYWwgYW5kIHJ1bjoKCmBgYApvcGVuc3NsIHJhbmQgLWhleCAzMgojIE91dHB1dHMgc29tZXRoaW5nIGxpa2U6IDk5ZDNiMWYwMWFhNjM5ZTRhNzZmNGZjMjgxZmM4MzQ3NDdhNTQzNzIwYmE0YzhhODY0OGJhNzU1YWVmOWJlN2YKYGBgCgojIyMgRW52aXJvbm1lbnQgdmFyaWFibGVzCgpUaGUgaW5wdXQgdmFyaWFibGVzLCB3aXRoIHRoZWlyIGRlZmF1bHQgdmFsdWVzOgoKKiBgUFJPSkVDVF9OQU1FYDogVGhlIG5hbWUgb2YgdGhlIHByb2plY3QKCiogYFZFUlNJT05gOiBQcm9qZWN0IHZlcnNpb24KCiogYFNFUlZFUl9ET01BSU5gOiBUaGUgZG9tYWluIGluIHdoZXJlIHRvIGRlcGxveSB0aGUgcHJvamVjdC4gQnkgZGVmYXVsdCwgYmFzZWQgb24gdGhlIGBodHRwOi8vbG9jYWxob3N0OjgwMDBgLgoKKiBgRklSU1RfU1VQRVJVU0VSX0xPR0lOYDogVGhlIGZpcnN0IHN1cGVydXNlciBnZW5lcmF0ZWQsIHdpdGggaXQgeW91IHdpbGwgYmUgYWJsZSB0byBjcmVhdGUgbW9yZSB1c2VycywgZXRjLgoKKiBgRklSU1RfU1VQRVJVU0VSX1BBU1NXT1JEYDogRmlyc3Qgc3VwZXJ1c2VyIHBhc3N3b3JkLiBVc2UgdGhlIG1ldGhvZCBhYm92ZSB0byBnZW5lcmF0ZSBpdC4KCiogYERPQ1NfQURNSU5fVVNFUk5BTUVgOiBVc2VybmFtZSBmb3IgYmFzaWMgaHR0cCBhdXRoIG9mIHNlY3VyZSBkb2NzLgoKKiBgRE9DU19BRE1JTl9QQVNTV09SRGA6IFBhc3N3b3JkIGZvciBiYXNpYyBodHRwIGF1dGggb2Ygc2VjdXJlIGRvY3MuCgoqIGBFTlZgOiBFbnZpcm9ubWVudCBtb2RlLCBsaWtlIERFViBvciBQUk9ELgoKKiBgU0VOVFJZX0RTTmA6IEtleSBVUkwgKERTTikgb2YgU2VudHJ5LCBmb3IgbGl2ZSBlcnJvciByZXBvcnRpbmcuCgoqIGBBVVRIX1NFQ1JFVF9LRVlgOiBCYWNrZW5kIHNlcnZlciBzZWNyZXQga2V5LiBVc2UgdGhlIG1ldGhvZCBhYm92ZSB0byBnZW5lcmF0ZSBpdC4KCiogYEVNQUlMX1NFTkRfTU9ERWA6IFNlbmQgZW1haWxzIHZpYSBwb3N0IHNlcnZpY2UuIEJ5IGRlZmF1bHQgb2ZmIGFuZCBzaG93IG1lc3NhZ2UgY29udGV4dCBvbiBvdXRwdXQuCgoqIGBTRU5EX0dSSURfS0VZYDogS2V5IG9mIFNlbmRHcmlkLCBmb3Igc2VuZGluZyBlbWFpbHMuCgpPdGhlciBlbnZpcm9ubWVudCB2YXJpYWJsZXMgY291bGQgYmUgZm91bmQgaW4gYGVudi50ZW1wbGF0ZWAgb2YgYGJhY2tlbmRgIGRpcmVjdG9yeS4KCgojIyMgSG93IHRvIGRlcGxveQoKKiBDcmVhdGUgYC5lbnZgIGZpbGUgaW4gYGJhY2tlbmRgIGRpcmVjdG9yeSB3aXRoIHNldHRpbmdzIGZyb20gYC5lbnYudGVtcGxhdGVgLgoKKiBSdW4gY29tbWFuZCBgZG9ja2VyLWNvbXBvc2UgdXAgLS1idWlsZCAtZGAuCgoqIENoZWNrIFVSTCBgaHR0cDovLzEyNy4wLjAuMTo4MDAwYC4gQnkgZGVmYXVsdCBmb3Igc2VjdXJlIGRvY3MgdXNlIGNyZWRlbnRpYWxzOiBgYWRtaW46YWRtaW5gLgo= readmeEtag: '"6dd34f0197934fb88471aebc275fa0ede61ffcc6"' readmeLastModified: Thu, 18 Nov 2021 13:52:20 GMT repositoryId: 380564891 description: >- Backend application. Using FastAPI, PostgreSQL as database, Docker, SQLAlchemy and more. created: '2021-06-26T17:53:29Z' updated: '2025-04-13T11:51:10Z' language: HTML archived: false stars: 14 watchers: 1 forks: 5 owner: mkbeh logo: https://avatars.githubusercontent.com/u/26373902?v=4 license: MIT repoEtag: '"d7e2c90c7b5250d71f22d9eabe203d87610b916c8c194d9af0111d1c528f202f"' repoLastModified: Sun, 13 Apr 2025 11:51:10 GMT foundInMaster: true category: Server Implementations id: e236f6dafcb1ee24117db986ab0f4617 - source: openapi3 tags repository: https://github.com/bcgov/nr-arch-exploration v3: true id: 3f615b0e3b3fa8fcbfb93455f2b21739 repositoryMetadata: base64Readme: >- ClshW0NvbnRyaWJ1dG9yc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvY29udHJpYnV0b3JzL2JjZ292L2lpdC1hcmNoKV0oLy4uLy4uL2dyYXBocy9jb250cmlidXRvcnMpClshW0ZvcmtzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9mb3Jrcy9iY2dvdi9paXQtYXJjaCldKC8uLi8uLi9uZXR3b3JrL21lbWJlcnMpClshW1N0YXJnYXplcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3N0YXJzL2JjZ292L2lpdC1hcmNoKV0oLy4uLy4uL3N0YXJnYXplcnMpClshW0lzc3Vlc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvaXNzdWVzL2JjZ292L2lpdC1hcmNoKV0oLy4uLy4uL2lzc3VlcykKWyFbTUlUIExpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvYmNnb3YvaWl0LWFyY2guc3ZnKV0oL0xJQ0VOU0UubWQpCiFbTGlmZWN5Y2xlOkV4cGVyaW1lbnRhbF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWZlY3ljbGUtRXhwZXJpbWVudGFsLTMzOTk5OSkKIyBuci1hcmNoLWV4cGxvcmF0aW9uClRoaXMgY29kZSByZXBvIGNvbnRhaW5zIHZhcmlvdXMgZXhhbXBsZXMgZnJvbSBOUklEUyBBcmNoaXRlY3R1cmUgdGVhbQoxLiBBUElzCjIuIFVJcwozLiBNZXRhYmFzZSBEZXBsb3ltZW50IFNwZWNpZmljIHRvIExlZ2FjeSBPcmFjbGUgQ29ubmVjdGlvbiBvdmVyIGVuY3J5cHRlZCBsaXN0ZW5lcnMKClNjaGVtYXNweQpodHRwczovL2JjZ292LmdpdGh1Yi5pby9uci1hcmNoLWV4cGxvcmF0aW9uL3NjaGVtYXNweS9pbmRleC5odG1sCg== readmeEtag: '"51cae3045923ec4f524179bfc985e4147fcadce4"' readmeLastModified: Fri, 19 Jan 2024 00:33:14 GMT repositoryId: 467760840 description: >- This repo contains various examples on different APIs and UIs and Metabase deployment specific to Oracle connection created: '2022-03-09T03:17:27Z' updated: '2024-01-30T17:29:37Z' language: Shell archived: true stars: 13 watchers: 4 forks: 0 owner: bcgov logo: https://avatars.githubusercontent.com/u/916280?v=4 license: Apache-2.0 repoEtag: '"230106c995aa6972fe535330c3a6d1675780546c1a1f51a10ba55370ed97cbbf"' repoLastModified: Tue, 30 Jan 2024 17:29:37 GMT category: Documentation foundInMaster: true oldLocations: - https://github.com/bcgov/nr-arch-templates - source: openapi3 tags repository: https://github.com/tuwilof/tomograph v3: true id: d31c98262a36d04a86611f50ddbeb2b4 repositoryMetadata: base64Readme: >- # Tomograph 

Convert API Blueprint, Swagger and OpenAPI to minimal routes with JSON Schema. For ease of use and creation of new tools.

Will look like

  ```json
  [
    {
      "path": "/sessions",
      "method": "POST",
      "content-type": "application/json",
      "requests": [{
        "$schema": "http://json-schema.org/draft-04/schema#",
        "type": "object",
        "properties": {
          "login": {
            "type": "string"
          },
          "password": {
            "type": "string"
          },
          "captcha": {
            "type": "string"
          }
        },
        "required": [
          "login",
          "password"
        ]
      }],
      "responses": [
        {
          "status": "401",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "429",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "201",
          "content-type": "application/json",
          "body": {
            "$schema": "http://json-schema.org/draft-04/schema#",
            "type": "object",
            "properties": {
              "confirmation": {
                "type": "object",
                "properties": {
                  "id": {
                    "type": "string"
                  },
                  "type": {
                    "type": "string"
                  },
                  "operation": {
                    "type": "string"
                  }
                },
                "required": [
                  "id",
                  "type",
                  "operation"
                ]
              },
              "captcha": {
                "type": "string"
              },
              "captcha_does_not_match": {
                "type": "boolean"
              }
            }
          }
        }
      ]
    }
  ]
  ```

## Installation

Then add this line to your application's Gemfile:

```ruby
gem 'tomograph'
```

After that execute:

```bash
$ bundle
```

Or install the gem by yourself:

```bash
$ gem install tomograph
```

## Usage

### In code

#### OpenAPI 2.0

Also Swagger

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(openapi2_json_path: '/path/to/doc.json')
```

#### OpenAPI 3.0

Also OpenAPI

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(openapi3_yaml_path: '/path/to/doc.yaml')
```

#### API Blueprint

First you need to install [drafter](https://github.com/apiaryio/drafter).
Works after conversion from API Blueprint to API Elements (in YAML file) with Drafter.

That is, I mean that you first need to do this

```bash
drafter doc.apib -o doc.yaml
```

and then

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(drafter_yaml_path: '/path/to/doc.yaml')
```

#### Tomograph

To use additional features of the pre-converted

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(tomogram_json_path: '/path/to/doc.json')
```

#### prefix
Default: `''`

You can specify API prefix and path to the spec using one of the possible formats:

```ruby
Tomograph::Tomogram.new(prefix: '/api/v2', drafter_yaml_path: '/path/to/doc.yaml')
```

```ruby
Tomograph::Tomogram.new(prefix: '/api/v2', tomogram_json_path: '/path/to/doc.json')
```

#### to_json
Use `to_json` for converting to JSON, example from API Blueprint:

```ruby
tomogram.to_json
```

<details>
  <summary>Example input</summary>

  ```apib
  FORMAT: 1A
  HOST: http://test.local
  
  # project
  
  # Group project
  
  Project
  
  ## Authentication [/sessions]
  
  ### Sign In [POST]
  
  + Request (application/json)
  
      + Attributes
       + login (string, required)
       + password (string, required)
       + captcha (string, optional)
  
  + Response 401 (application/json)
  
  + Response 429 (application/json)
  
  + Response 201 (application/json)
  
      + Attributes
       + confirmation (Confirmation, optional)
       + captcha (string, optional)
       + captcha_does_not_match (boolean, optional)
  
  
  # Data Structures
  
  ## Confirmation (object)
    + id (string, required)
    + type (string, required)
    + operation (string, required)
  ```
</details>

<details>
  <summary>Example output</summary>

  ```json
  [
    {
      "path": "/sessions",
      "method": "POST",
      "content-type": "application/json",
      "requests": [{
        "$schema": "http://json-schema.org/draft-04/schema#",
        "type": "object",
        "properties": {
          "login": {
            "type": "string"
          },
          "password": {
            "type": "string"
          },
          "captcha": {
            "type": "string"
          }
        },
        "required": [
          "login",
          "password"
        ]
      }],
      "responses": [
        {
          "status": "401",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "429",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "201",
          "content-type": "application/json",
          "body": {
            "$schema": "http://json-schema.org/draft-04/schema#",
            "type": "object",
            "properties": {
              "confirmation": {
                "type": "object",
                "properties": {
                  "id": {
                    "type": "string"
                  },
                  "type": {
                    "type": "string"
                  },
                  "operation": {
                    "type": "string"
                  }
                },
                "required": [
                  "id",
                  "type",
                  "operation"
                ]
              },
              "captcha": {
                "type": "string"
              },
              "captcha_does_not_match": {
                "type": "boolean"
              }
            }
          }
        }
      ]
    }
  ]
  ```
</details> 

#### to_a
```ruby
tomogram.to_a
```

#### find_request
```ruby
request = tomogram.find_request(method: 'GET', path: '/status/1?qwe=rty')
```

#### find_request_with_content_type
```ruby
request = tomogram.find_request_with_content_type(method: 'GET', path: '/status/1?qwe=rty', content_type: 'application/json')
```

#### `find_responses`
```ruby
responses = request.find_responses(status: '200')
```

#### prefix_match?
This may be useful if you specify a prefix.

```ruby
tomogram.prefix_match?('http://local/api/v2/users')
```

#### to_resources
Maps resources for API Blueprint with possible requests.

Example output:

```ruby
{
  '/sessions' => ['POST /sessions']
}
```

### Command line tool

CLI allows you to convert files from API Blueprint (API Elements), Swagger and OpenAPI to JSON Schema.

Run CLI with `-h` to get detailed help:

```bash
tomograph -h
```

To specify the handler version use the `-d` flag:

#### OpenAPI 2.0
```bash
tomograph -d openapi2 openapi2.json tomogram.json
```

#### OpenAPI 3.0
```bash
tomograph -d openapi3 openapi3.yaml doc.json
```

#### API Blueprint
```bash
tomograph -d 4 apielemetns.yaml doc.json
```

#### exclude-description

Exclude "description" keys from json-schemas.

```bash
tomograph -d 4 apielemetns.yaml doc.json --exclude-description
```

#### split

Split output into files by method. Output in dir path.

```bash
tomograph -d 4 --split apielemetns.yaml jsons/
```

## License

The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).

[Sponsored by FunBox](https://funbox.ru)
 readmeEtag: '"1ce89b494851b667504536ff9080f94dd945f5bb"' readmeLastModified: Wed, 07 Aug 2024 20:54:22 GMT repositoryId: 81070217 description: >- Convert API Blueprint, Swagger and OpenAPI to JSON Schema and search through it created: '2017-02-06T09:27:24Z' updated: '2024-12-08T11:20:22Z' language: Ruby archived: false stars: 12 watchers: 4 forks: 6 owner: tuwilof logo: https://avatars.githubusercontent.com/u/5956924?v=4 license: MIT repoEtag: '"4459bd954b385607ad7ce411a12a8f5ca4fe627a9dfa4718208d8f47e1462c3b"' repoLastModified: Sun, 08 Dec 2024 11:20:22 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/prince-chrismc/user-management v3: true repositoryMetadata: base64Readme: >- IyBVc2VyIE1hbmFnZW1lbnQgWyFbRG9jc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9BUEklMjBEb2N1bWVudGF0aW9uLW1haW4tYmx1ZSldKGh0dHBzOi8vcHJpbmNlLWNocmlzbWMuZ2l0aHViLmlvL3VzZXItbWFuYWdlbWVudC8pCgpBbiBvcGVuLXNvdXJjZSBhcHBsaWNhdGlvbiBkZWxpdmVyaW5nIGEgcmVzcG9uc2l2ZSB1c2VyIG1hbmFnZW1lbnQgZXhwZXJpZW5jZS4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vcHJpbmNlLWNocmlzbWMvdXNlci1tYW5hZ2VtZW50L21haW4vZG9jcy9TY3JlZW5jYXN0LTIwMjAtMDctMTgtMjMwNzU0LmdpZiI+CjwvcD4KCiMjIDptaWNyb3Njb3BlOiBUZWNobmljYWwgT3ZlcnZpZXcKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBhIGRpc3RyaWJ1dGVkIGNsb3VkIG5hdGl2ZSBhcHBsaWNhdGlvbiBjb21wcmlzZWQgb2YgdHdvIGNvbXBvbmVudHM6CgoqIFtGcm9udC1lbmRdKGZyb250ZW5kLykgLSBQcmltYXJ5IHBvaW50IG9mIGludGVyYWN0aW9uIHdpdGggdXNlcnMuIEl0IGlzIGEgW1JlYWN0XShodHRwczovL3JlYWN0anMub3JnLykgY2xpZW50IHNpZGUgYXBwbGljYXRpb24gYnVpbGQgdXNpbmcgW3dlYnBhY2tdKGh0dHBzOi8vd2VicGFjay5qcy5vcmcpIGZvciBfZmFzdF8gc3VyZmluZy4KKiBbQmFjay1lbmRdKGJhY2tlbmQvKSAtIENlbnRyYWxpemVkIGluLW1lbW9yeSBkYXRhIHN0b3JhZ2UgY29udGFpbmluZyB0aGUgaW5mb3JtYXRpb24gb2YgYWxsIGtub3duIHVzZXJzLiBXcml0dGVuIGluIFtDKytdKGh0dHBzOi8vaXNvY3BwLm9yZy8pIGZvciBmbGV4aWJpbGl0eSBhbmQgc2NhbGFiaWxpdHkgaXQgbGV2ZXJhZ2VzIG1hbnkgT3Blbi1Tb3VyY2UgdGVjaG5vbG9naWVzIFtsaXN0ZWQgaGVyZV0oYmFja2VuZC9jb25hbi5sb2NrKS4KCiMjIDpyb2NrZXQ6IENsb3VkIERlcGxveW1lbnRzCgo+IDppbmZvcm1hdGlvbl9zb3VyY2U6IFRoZSBnb2FsIGZvciB0aGlzIHByb2plY3QgaXMgdG8gc3BhbiBtdWx0aXBsZSBwdWJsaWMgOmNsb3VkOiBjbG91ZHMgYWxsIHRoZSB3aGlsZSBjb3N0aW5nIHRoZSBsZWFzdCBhbW91bnQgcG9zc2libGUKCjpub19lbnRyeV9zaWduOiBDdXJyZW50bHkgb2ZmbGluZSEKCiMjIDptYW5fc2NpZW50aXN0OiBEZXZlbG9wbWVudCBTdW1tYXJ5Cgp8IENvbXBvbmVudCB8IEJ1aWxkIHwgQ292ZXJhZ2UgfAp8IC0tLSB8IC0tLSB8IC0tLSB8CnwgQmFjay1lbmQgfCBbIVtDKysgQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS9wcmluY2UtY2hyaXNtYy91c2VyLW1hbmFnZW1lbnQvd29ya2Zsb3dzL0MrKyUyMENJL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9wcmluY2UtY2hyaXNtYy91c2VyLW1hbmFnZW1lbnQvYWN0aW9ucz9xdWVyeT13b3JrZmxvdyUzQSUyMkMlMkIlMkIrQ0klMjIpfCBbIVtjb2RlY292XShodHRwczovL2ltZy5zaGllbGRzLmlvL2NvZGVjb3YvYy9naXRodWIvcHJpbmNlLWNocmlzbWMvdXNlci1tYW5hZ2VtZW50KV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL3ByaW5jZS1jaHJpc21jL3VzZXItbWFuYWdlbWVudCkgfAp8IEZyb250LWVuZCB8IFshW05vZGUuanMgQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS9wcmluY2UtY2hyaXNtYy91c2VyLW1hbmFnZW1lbnQvd29ya2Zsb3dzL05vZGUuanMlMjBDSS9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vcHJpbmNlLWNocmlzbWMvdXNlci1tYW5hZ2VtZW50L2FjdGlvbnM/cXVlcnk9d29ya2Zsb3clM0ElMjJOb2RlLmpzK0NJJTIyKSB8IFshW2NvZGVjb3ZdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY29kZWNvdi9jL2dpdGh1Yi9wcmluY2UtY2hyaXNtYy91c2VyLW1hbmFnZW1lbnQpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvcHJpbmNlLWNocmlzbWMvdXNlci1tYW5hZ2VtZW50KSB8Cg== readmeEtag: '"83f9102bdf5aa0ca38fbea1ecab79f6dd9118655"' readmeLastModified: Wed, 06 Sep 2023 18:52:51 GMT repositoryId: 245055601 description: >- An open-source application delivering a responsive user management experience. created: '2020-03-05T03:00:58Z' updated: '2025-03-30T06:54:47Z' language: C++ archived: false stars: 13 watchers: 2 forks: 1 owner: prince-chrismc logo: https://avatars.githubusercontent.com/u/16867443?v=4 license: MIT repoEtag: '"4579d60cde6fb6f58b753c4bc6698eb7d9560fb6616b4638e41128a93303ea6f"' repoLastModified: Sun, 30 Mar 2025 06:54:47 GMT foundInMaster: true category: Server id: becccc9cc767bca2f946cf466bd59707 - source: openapi3 tags repository: https://github.com/networknt/openapi-bundler v3: true repositoryMetadata: base64Readme: >- # openapi-bundler
A utility that merges multiple OpenAPI files into a single file with all external
references resolved to local component references.

## Why this OpenAPI Bundler

OpenAPI is becoming the de-facto REST API specification, and a lot people have adopted it
to write their RESTful API spec.

For small APIs, it is acceptable to have a single OpenAPI definition file, however for a large scale API
project, it is hard to manage multiple API specifications together.

For an organization with too
many APIs, it is very natural to define sharable object definitions in separate files
in order to avoid duplications and foster consistency across an organization. With multiple files inter-connected together to form
one API specification, the directory structure and external references are too complicated
to manage.

Luckily, there are a lot of tools like editors, parsers and bundlers in the market that
support multiple files. These tools can bundle multiple YAML files together to create
a final version in JSON or YAML format and de-reference external dependencies.

As our light-rest-4j framework encourages a contract driven design approach, the specification
should be created before coding starts. Actually, if the spec is ready, you can use
[light-codegen](https://github.com/networknt/light-codegen) to generate the project.

The generator for [light-rest-4j](https://github.com/networknt/light-rest-4j)
generates also the object model (POJO) from the openapi.json where objects are defined in the specification. In
addition, light-rest-4j requires the final version of the openapi.json to be included
into the service code to do runtime schema validation as well as OAuth 2.0 scope
verification at runtime.

This requires that the final version of openapi.json is self-contained and all models
should be defined in the components section instead of being de-referenced inline.

Existing bundlers such as [swagger-cli](https://github.com/BigstickCarpet/swagger-cli)
cannot handle our specification files correctly, therefore the need arises to create a generally re-usable bundler for OpenAPI definitions.

The tool validates bundled API specifications and offers also independent validation functionality.

## Features

### Remote Reference

This bundler can resolve all remote references in openapi.yaml which is the main file
to be processed. If the reference is an object, it will resolve its internal references
first and then move it into components in the generated openapi.json. At the same time
the external reference in openapi.yaml will be changed to local reference with #/components/{key}

If the remote reference is not an object, it will be resolved inline in the generated
openapi.json file.

If you have separate reference files, you must place these files into the same folder your openapi.yaml, or relative to that folder. For example, if you have common
folder that contains all the common OpenAPI files, you might need to copy the common
folder into your folder that contains openapi.yaml for your API.

### Local Reference

If the reference is an object in components, it will resolve all the remote references
in definitions.

If the reference is not an object, an error will occur and the process will exit.

## Usage

The bundler assumes that the input file is openapi.yaml and all the remote reference files
are in the right path.

### Use it as Java utility

```
java -jar target/openapi-bundler.jar <operation> <arguments....>

Operations are:
  bundle
  validate

Arguments are:
  --dir, -d : The input directory where the YAML files can be found for bundling | validation. Mandatory parameter
  --file, -f : The name of the YAML file to be bundled or validated. Default: openapi.yaml

  --outputFile, -of: The name of the bundled and validated OpenAPI file. Default: openapi.bundled
  --outputDir, -od : The output directory of the bundled and validated file. Default: same as input directory specified in <dir>
  --outputFormat, -o : TThe output format for the bundled file: YAML | JSON | both. Default: YAML

  -debug : to view debug output

# General usage:
  Bundle: 
    java -jar openapi-bundler.jar bundle -d <myFolder> -f <input file> -o <json|yaml|both> -of <output file name> -od <output folder>

  Validate:
    java -jar openapi-bundler.jar bundle -d <myFolder> -f <input file>

# To view debug messages during the bundling or valdiation process, use the utility with -debug
  java -debug -jar openapi-bundler.jar bundle -d <myFolder> -f <input file> -o <json|yaml|both> 

# To view debug messages during the bundling or valdiation process, use the utility with -debug
  java -debug -jar openapi-bundler.jar validate -d <myFolder> -f <input file> 

# Simplified bundler call, with default values, uses openapi.yaml as input and json as output format
java -jar openapi-bundler.jar  bundle -d <myFolder>


```

To view usage help you can use the following command:
```
Command:
java -jar openapi-bundler.jar  -h

Usage: [options] operation: The operation to be performed. 
       Supported operations: bundle |validate. Must be specified

  Options:
    --dir, -d
      The input directory where the YAML files can be found. Must be specified
    --file, -f
      The name of the YAML file to be bundled. Default = openapi.yaml
    --output, -o
      The output format for the bundled file: yaml | json | both. Default = json
      Default: json
    --help, -h      
    -debug
      Debug mode
      Default: false
```
### Sample files
openapi.yaml
```
openapi: 3.0.0

paths: 
  /test:
    get:
      operationId: test
      responses:
        '200':
          content:
            application/json:
              schema:
                type: array
                items:
                  $ref: 'common.yaml#/MyObj'
```

common.yaml
```
MyObj:
  type: object
  properties:
    testCode:
      type: string
      readOnly: true
    name:
      type: string
      readOnly: true
```
### Use it in an IDE

Another way to run the bundler is from an IDE. Just set the folder of the openapi.yaml file as a program argument and you can easily debug into it.

### Use Docker

There is a Docker [image](https://hub.docker.com/r/networknt/openapi-bundler/) that is
published to Docker Hub.

Here is the command line to call it.

```
TBD
```


With above command line, you can easily build a script to call it as part of your DevOps
flow.
 readmeEtag: '"61f411de5c7e566d9c7922477d174ecb6a947bea"' readmeLastModified: Thu, 13 Jun 2024 13:02:01 GMT repositoryId: 153839607 description: >- A utility that merges multiple OpenAPI specification files into a single file with all external references resolved to local reference. created: '2018-10-19T20:42:58Z' updated: '2026-01-29T19:33:38Z' language: Java archived: false stars: 13 watchers: 17 forks: 6 owner: networknt logo: https://avatars.githubusercontent.com/u/8740739?v=4 license: Apache-2.0 repoEtag: '"b5f4853a94b93b0bb6a0757d5a1fc4f596ea3afab6598624314cfce7525413ce"' repoLastModified: Thu, 29 Jan 2026 19:33:38 GMT foundInMaster: true category: Parsers id: 9e7d370d54bfa85bddd37c511827340b - source: openapi3 tags repository: https://github.com/namsor/namsor-r-sdk2 v3: true repositoryMetadata: base64Readme: >- # R API client for namsor

NOTE : this SDK will still work but is no longer maintained. Please check one of the alternative SDKs : Java, Python or Golang.

NamSor API v2 : enpoints to process personal names (gender, cultural origin or ethnicity) in all alphabets or languages. Use GET methods for small tests, but prefer POST methods for higher throughput (batch processing of up to 100 names at a time). Need something you can't find here? We have many more features coming soon. Let us know, we'll do our best to add it! 

## Overview
This API client was generated by the [OpenAPI Generator](https://openapi-generator.tech) project. By using the [OpenAPI spec](https://openapis.org) from a remote server, you can easily generate an API client.

- API version: 2.0.10
- Package version: 2.1.0
- Build package: org.openapitools.codegen.languages.RClientCodegen
For more information, please visit [http://www.namsor.com/](http://www.namsor.com/)

## Installation

### Prerequisites

Install the dependencies

```R
install.packages("jsonlite")
install.packages("httr")
install.packages("base64enc")
```

### Build the package

```sh
git clone https://github.com/namsor/namsor-r-sdk2
cd namsor-r-sdk2
R CMD build .
R CMD check namsor_2.1.0.tar.gz
R CMD INSTALL namsor_2.1.0.tar.gz
```

### Install the package

```R
install.packages("namsor")
```

To install directly from Github, use `devtools`:
```R
install.packages("devtools")
library(devtools)
install_github("namsor/namsor-r-sdk2")
```

### Usage
```R
library(namsor)

var.first.name <- 'John' # character | 
var.last.name <- 'Smith' # character | 
var.country.iso2 <- 'US' # character | 

#Infer the likely gender of a name, given a local context (ISO2 country code).
api.instance <- PersonalApi$new()
# Configure API key authorization: api_key 
api.instance$apiClient$apiKeys['X-API-KEY'] <- '83ad28e3b3f92de1494181624864bec6';
result <- api.instance$GenderGeo(var.first.name, var.last.name, var.country.iso2)
dput(result)
print(result)
```

### Usage to process a CSV file
```R

#prepare workspace
if(!'devtools' %in% installed.packages()){
  install.packages(devtools)
}

if(!'readr' %in% installed.packages()){
  install.packages("readr")
}

if(!'namsor' %in% installed.packages()){
  install_github("namsor/namsor-r-sdk2")
}

#call packages
library(devtools)
library(namsor)
library(readr)

#call api
httr::set_config(httr::config(ssl_verifypeer = 0L))
api.instance <- PersonalApi$new()
api.instance$apiClient$apiKeys['X-API-KEY'] <- '83ad28e3b3f92de1494181624864bec6';

#call function
extractor <- function(names){
  
### Function calls api to assign gender by attributes ###
  
  #Input:
  
  #names - data.frame with 4 columns:
  # 1. id(char)
  # 2.first name(char)
  # 3. second name(char)
  # 4. country(char)
  
  
  #Output:
  #names - data.frame with 5 columns:
  # 1. id(char)
  # 2.first name(char)
  # 3. second name(char)
  # 4. country(char)
  # 5. gender(char)
  


## SPLIT TO LIST
names_list <- split(names, 
                    factor(rep(seq(ceiling(nrow(names)/100)), 
                               each = 100)[1:nrow(names)]))

## ADD ID
names_list <- lapply(names_list, function(x){
  data.frame('new_id' = seq(nrow(x)), 
             'id' = x[, 1],
             'first_name' = x[, 2], 
             'last_name' = x[, 3], 
             'country' = x[, 4])
})

#create r_gender
r_gender <- lapply(names_list, function(x){
  apply(x, 1, function(y){PersonalNameGeoIn$new(as.character(y[1]),y[3], y[4])})
})
#combine
r_gender <- lapply(r_gender, function(x){c(x)})
#create var gender
var_gender <- lapply(r_gender, function(x){
  BatchPersonalNameGeoIn$new(personalNames = x)})
#call api
api.instance <- PersonalApi$new()
api.instance$apiClient$apiKeys['X-API-KEY'] <- '83ad28e3b3f92de1494181624864bec6';
#get result
result_gender <- lapply(var_gender,
                        function(x){
                          api.instance$GenderFullGeoBatch(
                            batch.personal.name.geo.in=x)})
#extract gender
gender <- lapply(result_gender, function(x){lapply(x$personalNames, 
                                                   function(y){y$likelyGender})})
  
#add gender
names$gender <- unlist(gender)

#return
return(names)
}

#load small data
df1 <- read.csv('example2_csv.csv', stringsAsFactors = FALSE)

#convert to char
df1[, 1] <- sapply(df1[, 1], as.character)

#call function
df1_with_names <- extractor(df1)

#load big data
df2 <- data.frame(read_delim("example1_pipe_input.txt", 
                             "|", escape_double = FALSE, trim_ws = TRUE, 
                             col_names = FALSE))
#all char
str(df2)

#call function
df2_with_names <- extractor(df2)

```

## Documentation for API Endpoints

All URIs are relative to *https://v2.namsor.com/NamSorAPIv2*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AdminApi* | [**AddCredits**](docs/AdminApi.md#AddCredits) | **GET** /api2/json/addCredits/{apiKey}/{usageCredits}/{userMessage} | Add usage credits to an API Key.
*AdminApi* | [**Anonymize**](docs/AdminApi.md#Anonymize) | **GET** /api2/json/anonymize/{source}/{anonymized} | Activate/deactivate anonymization for a source.
*AdminApi* | [**ApiStatus**](docs/AdminApi.md#ApiStatus) | **GET** /api2/json/apiStatus | Prints the current status of the classifiers.
*AdminApi* | [**ApiUsage**](docs/AdminApi.md#ApiUsage) | **GET** /api2/json/apiUsage | Print current API usage.
*AdminApi* | [**ApiUsageHistory**](docs/AdminApi.md#ApiUsageHistory) | **GET** /api2/json/apiUsageHistory | Print historical API usage.
*AdminApi* | [**ApiUsageHistoryAggregate**](docs/AdminApi.md#ApiUsageHistoryAggregate) | **GET** /api2/json/apiUsageHistoryAggregate | Print historical API usage (in an aggregated view, by service, by day/hour/min).
*AdminApi* | [**AvailablePlans**](docs/AdminApi.md#AvailablePlans) | **GET** /api2/json/availablePlans | List all available plans in the default currency (usd).
*AdminApi* | [**AvailablePlans1**](docs/AdminApi.md#AvailablePlans1) | **GET** /api2/json/availablePlans/{token} | List all available plans in the user's preferred currency.
*AdminApi* | [**AvailableServices**](docs/AdminApi.md#AvailableServices) | **GET** /api2/json/apiServices | List of API services and usage cost in Units (default is 1=ONE Unit).
*AdminApi* | [**BillingCurrencies**](docs/AdminApi.md#BillingCurrencies) | **GET** /api2/json/billingCurrencies | List possible currency options for billing (USD, EUR, GBP, ...)
*AdminApi* | [**BillingHistory**](docs/AdminApi.md#BillingHistory) | **GET** /api2/json/billingHistory/{token} | Read the history billing information (invoices paid via Stripe or manually).
*AdminApi* | [**BillingInfo**](docs/AdminApi.md#BillingInfo) | **GET** /api2/json/billingInfo/{token} | Read the billing information (company name, address, phone, vat ID)
*AdminApi* | [**Charge**](docs/AdminApi.md#Charge) | **POST** /api2/json/charge | Create a Stripe Customer, based on a payment card token (from secure StripeJS) and email.
*AdminApi* | [**CorporateKey**](docs/AdminApi.md#CorporateKey) | **GET** /api2/json/corporateKey/{apiKey}/{corporate} | Setting an API Key to a corporate status.
*AdminApi* | [**DebugLevel**](docs/AdminApi.md#DebugLevel) | **GET** /api2/json/debugLevel/{logger}/{level} | Update debug level for a classifier
*AdminApi* | [**Flush**](docs/AdminApi.md#Flush) | **GET** /api2/json/flush | Flush counters.
*AdminApi* | [**InvalidateCache**](docs/AdminApi.md#InvalidateCache) | **GET** /api2/json/invalidateCache | Invalidate system caches.
*AdminApi* | [**Learnable**](docs/AdminApi.md#Learnable) | **GET** /api2/json/learnable/{source}/{learnable} | Activate/deactivate learning from a source.
*AdminApi* | [**NamsorCounter**](docs/AdminApi.md#NamsorCounter) | **GET** /api2/json/namsorCounter | Get the overall API counter
*AdminApi* | [**PaymentInfo**](docs/AdminApi.md#PaymentInfo) | **GET** /api2/json/paymentInfo/{token} | Get the Stripe payment information associated with the current google auth session token.
*AdminApi* | [**ProcureKey**](docs/AdminApi.md#ProcureKey) | **GET** /api2/json/procureKey/{token} | Procure an API Key (sent via Email), based on an auth token. Keep your API Key secret.
*AdminApi* | [**RedeployUI**](docs/AdminApi.md#RedeployUI) | **GET** /api2/json/redeployUI | Redeploy UI from current dev branch.
*AdminApi* | [**RedeployUI1**](docs/AdminApi.md#RedeployUI1) | **GET** /api2/json/redeployUI/{live} | Redeploy UI from current dev branch.
*AdminApi* | [**RemoveUserAccount**](docs/AdminApi.md#RemoveUserAccount) | **GET** /api2/json/removeUserAccount/{token} | Remove the user account.
*AdminApi* | [**RemoveUserAccountOnBehalf**](docs/AdminApi.md#RemoveUserAccountOnBehalf) | **GET** /api2/json/removeUserAccountOnBehalf/{apiKey} | Remove (on behalf) a user account.
*AdminApi* | [**Shutdown**](docs/AdminApi.md#Shutdown) | **GET** /api2/json/shutdown | Stop learning and shutdown system.
*AdminApi* | [**SoftwareVersion**](docs/AdminApi.md#SoftwareVersion) | **GET** /api2/json/softwareVersion | Get the current software version
*AdminApi* | [**SourceStats**](docs/AdminApi.md#SourceStats) | **GET** /api2/json/sourceStats/{source} | Print basic source statistics.
*AdminApi* | [**Stats**](docs/AdminApi.md#Stats) | **GET** /api2/json/stats | Print basic system statistics.
*AdminApi* | [**StripeConnect**](docs/AdminApi.md#StripeConnect) | **GET** /api2/json/stripeConnect | Connects a Stripe Account.
*AdminApi* | [**SubscribePlan**](docs/AdminApi.md#SubscribePlan) | **GET** /api2/json/subscribePlan/{planName}/{token} | Subscribe to a give API plan, using the user's preferred or default currency.
*AdminApi* | [**SubscribePlanOnBehalf**](docs/AdminApi.md#SubscribePlanOnBehalf) | **GET** /api2/json/subscribePlanOnBehalf/{planName}/{apiKey} | Subscribe to a give API plan, using the user's preferred or default currency (admin only).
*AdminApi* | [**TaxonomyClasses**](docs/AdminApi.md#TaxonomyClasses) | **GET** /api2/json/taxonomyClasses/{classifierName} | Print the taxonomy classes valid for the given classifier.
*AdminApi* | [**UpdateBillingInfo**](docs/AdminApi.md#UpdateBillingInfo) | **POST** /api2/json/updateBillingInfo/{token} | Sets or update the billing information (company name, address, phone, vat ID)
*AdminApi* | [**UpdateLimit**](docs/AdminApi.md#UpdateLimit) | **GET** /api2/json/updateLimit/{usageLimit}/{hardOrSoft}/{token} | Modifies the hard/soft limit on the API plan's overages (default is 0$ soft limit).
*AdminApi* | [**UpdatePaymentDefault**](docs/AdminApi.md#UpdatePaymentDefault) | **GET** /api2/json/updatePaymentDefault/{defautSourceId}/{token} | Update the default Stripe card associated with the current google auth session token.
*AdminApi* | [**UserInfo**](docs/AdminApi.md#UserInfo) | **GET** /api2/json/userInfo/{token} | Get the user profile associated with the current google auth session token.
*AdminApi* | [**VerifyEmail**](docs/AdminApi.md#VerifyEmail) | **GET** /api2/json/verifyEmail/{emailToken} | Verifies an email, based on token sent to that email
*AdminApi* | [**VerifyRemoveEmail**](docs/AdminApi.md#VerifyRemoveEmail) | **GET** /api2/json/verifyRemoveEmail/{emailToken} | Verifies an email, based on token sent to that email
*AdminApi* | [**Vet**](docs/AdminApi.md#Vet) | **GET** /api2/json/vetting/{source}/{vetted} | Vetting of a source.
*ChineseApi* | [**ChineseNameCandidates**](docs/ChineseApi.md#ChineseNameCandidates) | **GET** /api2/json/chineseNameCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming
*ChineseApi* | [**ChineseNameCandidatesBatch**](docs/ChineseApi.md#ChineseNameCandidatesBatch) | **POST** /api2/json/chineseNameCandidatesBatch | Identify Chinese name candidates, based on the romanized name (firstName = chineseGivenName; lastName=chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**ChineseNameCandidatesGenderBatch**](docs/ChineseApi.md#ChineseNameCandidatesGenderBatch) | **POST** /api2/json/chineseNameCandidatesGenderBatch | Identify Chinese name candidates, based on the romanized name (firstName = chineseGivenName; lastName=chineseSurname) ex. Wang Xiaoming.
*ChineseApi* | [**ChineseNameGenderCandidates**](docs/ChineseApi.md#ChineseNameGenderCandidates) | **GET** /api2/json/chineseNameGenderCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin}/{knownGender} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming - having a known gender ('male' or 'female')
*ChineseApi* | [**ChineseNameMatch**](docs/ChineseApi.md#ChineseNameMatch) | **GET** /api2/json/chineseNameMatch/{chineseSurnameLatin}/{chineseGivenNameLatin}/{chineseName} | Return a score for matching Chinese name ex. 王晓明 with a romanized name ex. Wang Xiaoming
*ChineseApi* | [**ChineseNameMatchBatch**](docs/ChineseApi.md#ChineseNameMatchBatch) | **POST** /api2/json/chineseNameMatchBatch | Identify Chinese name candidates, based on the romanized name (firstName = chineseGivenName; lastName=chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**GenderChineseName**](docs/ChineseApi.md#GenderChineseName) | **GET** /api2/json/genderChineseName/{chineseName} | Infer the likely gender of a Chinese full name ex. 王晓明
*ChineseApi* | [**GenderChineseNameBatch**](docs/ChineseApi.md#GenderChineseNameBatch) | **POST** /api2/json/genderChineseNameBatch | Infer the likely gender of up to 100 full names ex. 王晓明
*ChineseApi* | [**GenderChineseNamePinyin**](docs/ChineseApi.md#GenderChineseNamePinyin) | **GET** /api2/json/genderChineseNamePinyin/{chineseSurnameLatin}/{chineseGivenNameLatin} | Infer the likely gender of a Chinese name in LATIN (Pinyin).
*ChineseApi* | [**GenderChineseNamePinyinBatch**](docs/ChineseApi.md#GenderChineseNamePinyinBatch) | **POST** /api2/json/genderChineseNamePinyinBatch | Infer the likely gender of up to 100 Chinese names in LATIN (Pinyin).
*ChineseApi* | [**ParseChineseName**](docs/ChineseApi.md#ParseChineseName) | **GET** /api2/json/parseChineseName/{chineseName} | Infer the likely first/last name structure of a name, ex. 王晓明 -> 王(surname) 晓明(given name)
*ChineseApi* | [**ParseChineseNameBatch**](docs/ChineseApi.md#ParseChineseNameBatch) | **POST** /api2/json/parseChineseNameBatch | Infer the likely first/last name structure of a name, ex. 王晓明 -> 王(surname) 晓明(given name).
*ChineseApi* | [**PinyinChineseName**](docs/ChineseApi.md#PinyinChineseName) | **GET** /api2/json/pinyinChineseName/{chineseName} | Romanize the Chinese name to Pinyin, ex. 王晓明 -> Wang (surname) Xiaoming (given name)
*ChineseApi* | [**PinyinChineseNameBatch**](docs/ChineseApi.md#PinyinChineseNameBatch) | **POST** /api2/json/pinyinChineseNameBatch | Romanize a list of Chinese name to Pinyin, ex. 王晓明 -> Wang (surname) Xiaoming (given name).
*JapaneseApi* | [**GenderJapaneseNameFull**](docs/JapaneseApi.md#GenderJapaneseNameFull) | **GET** /api2/json/genderJapaneseNameFull/{japaneseName} | Infer the likely gender of a Japanese full name ex. 王晓明
*JapaneseApi* | [**GenderJapaneseNameFullBatch**](docs/JapaneseApi.md#GenderJapaneseNameFullBatch) | **POST** /api2/json/genderJapaneseNameFullBatch | Infer the likely gender of up to 100 full names
*JapaneseApi* | [**GenderJapaneseNamePinyin**](docs/JapaneseApi.md#GenderJapaneseNamePinyin) | **GET** /api2/json/genderJapaneseName/{japaneseSurname}/{japaneseGivenName} | Infer the likely gender of a Japanese name in LATIN (Pinyin).
*JapaneseApi* | [**GenderJapaneseNamePinyinBatch**](docs/JapaneseApi.md#GenderJapaneseNamePinyinBatch) | **POST** /api2/json/genderJapaneseNameBatch | Infer the likely gender of up to 100 Japanese names in LATIN (Pinyin).
*JapaneseApi* | [**JapaneseNameKanjiCandidates**](docs/JapaneseApi.md#JapaneseNameKanjiCandidates) | **GET** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameKanjiCandidatesBatch**](docs/JapaneseApi.md#JapaneseNameKanjiCandidatesBatch) | **POST** /api2/json/japaneseNameKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName = japaneseGivenName; lastName=japaneseSurname), ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameLatinCandidates**](docs/JapaneseApi.md#JapaneseNameLatinCandidates) | **GET** /api2/json/japaneseNameLatinCandidates/{japaneseSurnameKanji}/{japaneseGivenNameKanji} | Romanize japanese name, based on the name in Kanji.
*JapaneseApi* | [**JapaneseNameLatinCandidatesBatch**](docs/JapaneseApi.md#JapaneseNameLatinCandidatesBatch) | **POST** /api2/json/japaneseNameLatinCandidatesBatch | Romanize japanese names, based on the name in KANJI
*JapaneseApi* | [**JapaneseNameMatch**](docs/JapaneseApi.md#JapaneseNameMatch) | **GET** /api2/json/japaneseNameMatch/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | Return a score for matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameMatchBatch**](docs/JapaneseApi.md#JapaneseNameMatchBatch) | **POST** /api2/json/japaneseNameMatchBatch | Return a score for matching a list of Japanese names in KANJI ex. 山本 早苗 with romanized names ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameMatchFeedbackLoop**](docs/JapaneseApi.md#JapaneseNameMatchFeedbackLoop) | **GET** /api2/json/japaneseNameMatchFeedbackLoop/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | [CREDITS 1 UNIT] Feedback loop to better perform matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**ParseJapaneseName**](docs/JapaneseApi.md#ParseJapaneseName) | **GET** /api2/json/parseJapaneseName/{japaneseName} | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae
*JapaneseApi* | [**ParseJapaneseNameBatch**](docs/JapaneseApi.md#ParseJapaneseNameBatch) | **POST** /api2/json/parseJapaneseNameBatch | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae 
*PersonalApi* | [**Country**](docs/PersonalApi.md#Country) | **GET** /api2/json/country/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely country of residence of a personal full name, or one surname. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**CountryBatch**](docs/PersonalApi.md#CountryBatch) | **POST** /api2/json/countryBatch | [USES 10 UNITS PER NAME] Infer the likely country of residence of up to 100 personal full names, or surnames. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**Diaspora**](docs/PersonalApi.md#Diaspora) | **GET** /api2/json/diaspora/{countryIso2}/{firstName}/{lastName} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**DiasporaBatch**](docs/PersonalApi.md#DiasporaBatch) | **POST** /api2/json/diasporaBatch | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**Gender**](docs/PersonalApi.md#Gender) | **GET** /api2/json/gender/{firstName}/{lastName} | Infer the likely gender of a name.
*PersonalApi* | [**GenderBatch**](docs/PersonalApi.md#GenderBatch) | **POST** /api2/json/genderBatch | Infer the likely gender of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**GenderFull**](docs/PersonalApi.md#GenderFull) | **GET** /api2/json/genderFull/{fullName} | Infer the likely gender of a full name, ex. John H. Smith
*PersonalApi* | [**GenderFullBatch**](docs/PersonalApi.md#GenderFullBatch) | **POST** /api2/json/genderFullBatch | Infer the likely gender of up to 100 full names, detecting automatically the cultural context.
*PersonalApi* | [**GenderFullGeo**](docs/PersonalApi.md#GenderFullGeo) | **GET** /api2/json/genderFullGeo/{fullName}/{countryIso2} | Infer the likely gender of a full name, given a local context (ISO2 country code).
*PersonalApi* | [**GenderFullGeoBatch**](docs/PersonalApi.md#GenderFullGeoBatch) | **POST** /api2/json/genderFullGeoBatch | Infer the likely gender of up to 100 full names, with a given cultural context (country ISO2 code).
*PersonalApi* | [**GenderGeo**](docs/PersonalApi.md#GenderGeo) | **GET** /api2/json/genderGeo/{firstName}/{lastName}/{countryIso2} | Infer the likely gender of a name, given a local context (ISO2 country code).
*PersonalApi* | [**GenderGeoBatch**](docs/PersonalApi.md#GenderGeoBatch) | **POST** /api2/json/genderGeoBatch | Infer the likely gender of up to 100 names, each given a local context (ISO2 country code).
*PersonalApi* | [**Origin**](docs/PersonalApi.md#Origin) | **GET** /api2/json/origin/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely country of origin of a personal name. Assumes names as they are in the country of origin. For US, CA, AU, NZ and other melting-pots : use 'diaspora' instead.
*PersonalApi* | [**OriginBatch**](docs/PersonalApi.md#OriginBatch) | **POST** /api2/json/originBatch | [USES 10 UNITS PER NAME] Infer the likely country of origin of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**ParseName**](docs/PersonalApi.md#ParseName) | **GET** /api2/json/parseName/{nameFull} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. 
*PersonalApi* | [**ParseNameBatch**](docs/PersonalApi.md#ParseNameBatch) | **POST** /api2/json/parseNameBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John.
*PersonalApi* | [**ParseNameGeo**](docs/PersonalApi.md#ParseNameGeo) | **GET** /api2/json/parseName/{nameFull}/{countryIso2} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. For better accuracy, provide a geographic context.
*PersonalApi* | [**ParseNameGeoBatch**](docs/PersonalApi.md#ParseNameGeoBatch) | **POST** /api2/json/parseNameGeoBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. Giving a local context improves precision. 
*PersonalApi* | [**ParsedGenderBatch**](docs/PersonalApi.md#ParsedGenderBatch) | **POST** /api2/json/parsedGenderBatch | Infer the likely gender of up to 100 fully parsed names, detecting automatically the cultural context.
*PersonalApi* | [**ParsedGenderGeoBatch**](docs/PersonalApi.md#ParsedGenderGeoBatch) | **POST** /api2/json/parsedGenderGeoBatch | Infer the likely gender of up to 100 fully parsed names, detecting automatically the cultural context.
*PersonalApi* | [**UsRaceEthnicity**](docs/PersonalApi.md#UsRaceEthnicity) | **GET** /api2/json/usRaceEthnicity/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer a US resident's likely race/ethnicity according to US Census taxonomy W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino).
*PersonalApi* | [**UsRaceEthnicityBatch**](docs/PersonalApi.md#UsRaceEthnicityBatch) | **POST** /api2/json/usRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident's likely race/ethnicity according to US Census taxonomy.
*PersonalApi* | [**UsRaceEthnicityZIP5**](docs/PersonalApi.md#UsRaceEthnicityZIP5) | **GET** /api2/json/usRaceEthnicityZIP5/{firstName}/{lastName}/{zip5Code} | [USES 10 UNITS PER NAME] Infer a US resident's likely race/ethnicity according to US Census taxonomy, using (optional) ZIP5 code info. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino).
*PersonalApi* | [**UsZipRaceEthnicityBatch**](docs/PersonalApi.md#UsZipRaceEthnicityBatch) | **POST** /api2/json/usZipRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident's likely race/ethnicity according to US Census taxonomy, with (optional) ZIP code.
*SocialApi* | [**PhoneCode**](docs/SocialApi.md#PhoneCode) | **GET** /api2/json/phoneCode/{firstName}/{lastName}/{phoneNumber} | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, given a personal name and formatted / unformatted phone number.
*SocialApi* | [**PhoneCodeBatch**](docs/SocialApi.md#PhoneCodeBatch) | **POST** /api2/json/phoneCodeBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, detecting automatically the local context given a name and formatted / unformatted phone number.
*SocialApi* | [**PhoneCodeGeo**](docs/SocialApi.md#PhoneCodeGeo) | **GET** /api2/json/phoneCodeGeo/{firstName}/{lastName}/{phoneNumber}/{countryIso2} | [USES 11 UNITS PER NAME] Infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).
*SocialApi* | [**PhoneCodeGeoBatch**](docs/SocialApi.md#PhoneCodeGeoBatch) | **POST** /api2/json/phoneCodeGeoBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, with a local context (ISO2 country of residence).
*SocialApi* | [**PhoneCodeGeoFeedbackLoop**](docs/SocialApi.md#PhoneCodeGeoFeedbackLoop) | **GET** /api2/json/phoneCodeGeoFeedbackLoop/{firstName}/{lastName}/{phoneNumber}/{phoneNumberE164}/{countryIso2} | [CREDITS 1 UNIT] Feedback loop to better infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).


## Documentation for Models

 - [APIBillingPeriodUsageOut](docs/APIBillingPeriodUsageOut.md)
 - [APIClassifierOut](docs/APIClassifierOut.md)
 - [APIClassifierTaxonomyOut](docs/APIClassifierTaxonomyOut.md)
 - [APIClassifiersStatusOut](docs/APIClassifiersStatusOut.md)
 - [APICounterV2Out](docs/APICounterV2Out.md)
 - [APIKeyOut](docs/APIKeyOut.md)
 - [APIPeriodUsageOut](docs/APIPeriodUsageOut.md)
 - [APIPlanOut](docs/APIPlanOut.md)
 - [APIPlanSubscriptionOut](docs/APIPlanSubscriptionOut.md)
 - [APIPlansOut](docs/APIPlansOut.md)
 - [APIServiceOut](docs/APIServiceOut.md)
 - [APIServicesOut](docs/APIServicesOut.md)
 - [APIUsageAggregatedOut](docs/APIUsageAggregatedOut.md)
 - [BatchFirstLastNameDiasporaedOut](docs/BatchFirstLastNameDiasporaedOut.md)
 - [BatchFirstLastNameGenderIn](docs/BatchFirstLastNameGenderIn.md)
 - [BatchFirstLastNameGenderedOut](docs/BatchFirstLastNameGenderedOut.md)
 - [BatchFirstLastNameGeoIn](docs/BatchFirstLastNameGeoIn.md)
 - [BatchFirstLastNameGeoZippedIn](docs/BatchFirstLastNameGeoZippedIn.md)
 - [BatchFirstLastNameIn](docs/BatchFirstLastNameIn.md)
 - [BatchFirstLastNameOriginedOut](docs/BatchFirstLastNameOriginedOut.md)
 - [BatchFirstLastNamePhoneCodedOut](docs/BatchFirstLastNamePhoneCodedOut.md)
 - [BatchFirstLastNamePhoneNumberGeoIn](docs/BatchFirstLastNamePhoneNumberGeoIn.md)
 - [BatchFirstLastNamePhoneNumberIn](docs/BatchFirstLastNamePhoneNumberIn.md)
 - [BatchFirstLastNameUSRaceEthnicityOut](docs/BatchFirstLastNameUSRaceEthnicityOut.md)
 - [BatchMatchPersonalFirstLastNameIn](docs/BatchMatchPersonalFirstLastNameIn.md)
 - [BatchNameMatchCandidatesOut](docs/BatchNameMatchCandidatesOut.md)
 - [BatchNameMatchedOut](docs/BatchNameMatchedOut.md)
 - [BatchParsedFullNameGeoIn](docs/BatchParsedFullNameGeoIn.md)
 - [BatchParsedFullNameIn](docs/BatchParsedFullNameIn.md)
 - [BatchPersonalNameGenderedOut](docs/BatchPersonalNameGenderedOut.md)
 - [BatchPersonalNameGeoIn](docs/BatchPersonalNameGeoIn.md)
 - [BatchPersonalNameGeoOut](docs/BatchPersonalNameGeoOut.md)
 - [BatchPersonalNameIn](docs/BatchPersonalNameIn.md)
 - [BatchPersonalNameParsedOut](docs/BatchPersonalNameParsedOut.md)
 - [BillingHistoryOut](docs/BillingHistoryOut.md)
 - [BillingInfoInOut](docs/BillingInfoInOut.md)
 - [CacheMetricsOut](docs/CacheMetricsOut.md)
 - [ClassifierMetricsOut](docs/ClassifierMetricsOut.md)
 - [CurrenciesOut](docs/CurrenciesOut.md)
 - [DeployUIOut](docs/DeployUIOut.md)
 - [ExpectedClassMetricsOut](docs/ExpectedClassMetricsOut.md)
 - [FeedbackLoopOut](docs/FeedbackLoopOut.md)
 - [FirstLastNameDiasporaedOut](docs/FirstLastNameDiasporaedOut.md)
 - [FirstLastNameGenderIn](docs/FirstLastNameGenderIn.md)
 - [FirstLastNameGenderedOut](docs/FirstLastNameGenderedOut.md)
 - [FirstLastNameGeoIn](docs/FirstLastNameGeoIn.md)
 - [FirstLastNameGeoZippedIn](docs/FirstLastNameGeoZippedIn.md)
 - [FirstLastNameIn](docs/FirstLastNameIn.md)
 - [FirstLastNameOriginedOut](docs/FirstLastNameOriginedOut.md)
 - [FirstLastNameOut](docs/FirstLastNameOut.md)
 - [FirstLastNamePhoneCodedOut](docs/FirstLastNamePhoneCodedOut.md)
 - [FirstLastNamePhoneNumberGeoIn](docs/FirstLastNamePhoneNumberGeoIn.md)
 - [FirstLastNamePhoneNumberIn](docs/FirstLastNamePhoneNumberIn.md)
 - [FirstLastNameUSRaceEthnicityOut](docs/FirstLastNameUSRaceEthnicityOut.md)
 - [InlineObject](docs/InlineObject.md)
 - [InvoiceItemOut](docs/InvoiceItemOut.md)
 - [InvoiceOut](docs/InvoiceOut.md)
 - [MatchPersonalFirstLastNameIn](docs/MatchPersonalFirstLastNameIn.md)
 - [NamSorCounterOut](docs/NamSorCounterOut.md)
 - [NameMatchCandidateOut](docs/NameMatchCandidateOut.md)
 - [NameMatchCandidatesOut](docs/NameMatchCandidatesOut.md)
 - [NameMatchedOut](docs/NameMatchedOut.md)
 - [ParsedFullNameGeoIn](docs/ParsedFullNameGeoIn.md)
 - [ParsedFullNameIn](docs/ParsedFullNameIn.md)
 - [PersonalNameGenderedOut](docs/PersonalNameGenderedOut.md)
 - [PersonalNameGeoIn](docs/PersonalNameGeoIn.md)
 - [PersonalNameGeoOut](docs/PersonalNameGeoOut.md)
 - [PersonalNameIn](docs/PersonalNameIn.md)
 - [PersonalNameParsedOut](docs/PersonalNameParsedOut.md)
 - [RomanizedNameOut](docs/RomanizedNameOut.md)
 - [SoftwareVersionOut](docs/SoftwareVersionOut.md)
 - [SourceDetailedMetricsOut](docs/SourceDetailedMetricsOut.md)
 - [SourceMetricsOut](docs/SourceMetricsOut.md)
 - [StripeCardOut](docs/StripeCardOut.md)
 - [StripeCustomerOut](docs/StripeCustomerOut.md)
 - [SystemMetricsOut](docs/SystemMetricsOut.md)
 - [UserInfoOut](docs/UserInfoOut.md)


## Documentation for Authorization


### api_key

- **Type**: API key
- **API key parameter name**: X-API-KEY
- **Location**: HTTP header



## Author

contact@namsor.com

 readmeEtag: '"2662eb18fb456251adc73e369600b29eda77e51c"' readmeLastModified: Mon, 15 Mar 2021 08:47:25 GMT repositoryId: 168223476 description: >- NamSor API v2 R SDK - classify personal names accurately by gender, country of origin, or ethnicity. created: '2019-01-29T20:28:30Z' updated: '2025-10-06T09:26:42Z' language: R archived: false stars: 12 watchers: 4 forks: 2 owner: namsor logo: https://avatars.githubusercontent.com/u/6951565?v=4 repoEtag: '"6585122e969d20f98cab008ca5c6414eab3fd209183bc17676935225fe8e27f7"' repoLastModified: Mon, 06 Oct 2025 09:26:42 GMT foundInMaster: true category: - Server Implementations - Description Validators id: 10347725bd389e99e9dcfbeb8212e0be - source: openapi3 tags repository: https://github.com/jnig/fastify-flux v3: true id: 2b41fe1fc019c16cbb976ee69aa351e0 repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://github.com/fluxapi/fluxapi" target="_blank" rel="noopener noreferrer">
    <img width="180" src="./logo.png" alt="Flux logo">
  </a>
</p>
<br/>
<p align="center">
  <a href="https://www.npmjs.com/package/create-fastify-flux"><img src="https://img.shields.io/npm/v/create-fastify-flux.svg" alt="npm package"></a>
  <a href="https://nodejs.org/en/about/releases/"><img src="https://img.shields.io/node/v/create-fastify-flux.svg" alt="node compatibility"></a>
  <img src="https://img.shields.io/badge/License-MIT-green.svg" alt="license" />
  <img src="https://github.com/Jnig/fastify-flux/actions/workflows/test.yml/badge.svg" />

</p>
<br/>

# FluxAPI ⚡

> Delightful API development

- 💡 Instant server start with <a href="https://github.com/evanw/esbuild" target="_blank">esbuild</a>
- 🎯 Automatic reloads and optional type checking
- ⚡️ Lightning fast http requests with <a href="https://github.com/fastify/fastify" target="_blank">fastify</a>
- 🛠️ E2E tests with [jest](https://github.com/facebook/jest)
- 📦 OpenAPI v3
- 🔋 Batteries included: A typed client SDK is automatically generated with <a href="https://github.com/acacode/swagger-typescript-api" target="_blank">swagger-typescript-api</a>

FluxAPI is a new API build tool that significantly improves the development experience. Typescript interfaces are automatically converted into JSON schema.

## Quickstart

```sh
npx create-fastify-flux@latest
```

Run the command and follow the instructions. Afterwards you will have an OpenAPI documentation at http://localhost:8080/ and a client SDK generated in the file `GeneratedApi.ts`.

## Motivation

There are great Node.js frameworks for building APIs with decorators. However using those can get very verbose. Consider the following example:

```ts
export class UpdateTodoDto {
  @IsInt()
  readonly id: number;

  @IsString()
  readonly name: string;
}

@Post('/:id')
@ApiResponse({
  type: Todo,
})
async update(@Param('id') id: number, @Body() body: UpdateTodoDto): Promise<Todo> {
  // implementation
}
```

The parameter decorators make the example slightly hard to read. The @ApiResponse type is redundant with function return type `Promise<Todo>`.

With FluxAPI the same can be achieved with the following:

```ts
interface UpdateTodo {
  id number;
  name: string;
}

@Post('/:id')
async update(id: number, body: UpdateTodo): Promise<Todo> {
  // implementation
}
```

The id parameter is matched with the path `/:id` and the body is provided to the paramter named body. The function return type is converted into an OpenApi response.

Another advantage is that you can leverage the interfaces which are provided by a database framework e.g. prisma, because it's no longer necessary to create a Dto class.

## Usage

The following code snippets are only provided as example. It's best to start a new project via
`npm create flux`.

### Define a controller

`Todo.controller.ts`

```ts
import { Controller, Delete, Get, Post, Put, Status } from '@fluxapi/common';
import {
  CreateTodo,
  ListTodoQuery,
  TodoResponse,
  UpdateTodo,
} from './Todo.schema';

@Controller('/todos', { tags: ['todos'] })
export class TodoController {
  @Get()
  async list(query: ListTodoQuery): Promise<TodoResponse[]> {
    // implementation
  }

  @Post()
  async create(body: CreateTodo): Promise<TodoResponse> {
    // implementation
  }

  @Get('/:id')
  async get(id: number): Promise<TodoResponse> {
    // implementation
  }

  @Put('/:id')
  async update(id: number, body: UpdateTodo): Promise<TodoResponse> {
    // implementation
  }

  @Delete('/:id')
  @Status(204)
  async remove(id: number): Promise<void> {
    // implementation
  }
}
```

### Define the schema

Schema files must end with `schema.ts`

`Todo.schema.ts`

```ts
interface Todo {
  id number;
  name: string;
  done: boolean;
}

export interface TodoResponse extends Todo {}

export interface CreateTodo extends Omit<TodoResponse, 'id'> {}

export interface UpdateTodo extends Partial<Omit<TodoResponse, 'id'>> {}

export interface ListTodoQuery {
  includeDone?: boolean;
}

```

### Create the server

`index.ts`

```ts
import Fastify from 'fastify';
import { FluxController, FluxOpenapi } from 'fastify-flux';
import { TodoController } from '~/Todo.controller';

const fastify = Fastify();

fastify.register(FluxOpenapi);
fastify.register(FluxController, {
  controllers: [TodoController],
});

fastify.listen(8080, '127.0.0.1', (err, address) => {
  console.log(`Server listening at ${address}`);
});
```

## Frequently Asked Questions

<details>
  <summary>How to provide additional JSON Schema properties?</summary>

You can use JSDoc to provide additional JSON Schema properties. You can also find more examples in the [ts-json-schema-generator](https://github.com/vega/ts-json-schema-generator) repository.

```ts
/**
 * @title Some title here
 * @description Some description here
 */
export interface MyObject {
  /**
   * @description Export field description
   * @default 'foobar'
   */
  name: string;
}
```

</details>

<details>
  <summary>How to configure the OpenAPI documentation?</summary>

The openapi function accepts all options from [fastify-swagger](https://github.com/fastify/fastify-swagger).

```ts
fastify.register(FluxOpenapi, {
  routePrefix: '/documentation',
  swagger: {
    info: {
      title: 'Test swagger',
      description: 'Testing the Fastify swagger API',
      version: '0.1.0',
    },
  },
});
```

</details>

<details>
  <summary>Which function parameters are mapped and how to provide additional mappings?</summary>

Mappings can be provided by parameter `name` or `type`. The following mappings are provided by default and can be extended.

```ts
const fastify = flux({
  mapping: [
    {
      name: 'query',
      mapper({ request }) {
        return request.query;
      },
    },
    {
      name: 'body',
      mapper({ request }) {
        return request.body;
      },
    },
    {
      type: 'FastifyReply',
      mapper({ reply }) {
        return reply;
      },
    },
    {
      type: 'FastifyRequest',
      mapper({ request }) {
        return request;
      },
    },
  ],
});
```

</details>

<details>
  <summary>How can I add authentication and authorization to the API?</summary>

For authentication fastify hooks can be used to identify user by a Token, JWT or anyhing else.

The `@Auth` decorator can be used to add values like roles or permissions to an endpoint. Via `reply.context.config.auth` the added value can be used in fastify hooks.

```ts
@Post()
@Auth('admin')
async create(body: CreateTodo): Promise<TodoResponse> {
}
```

```ts
fastify.addHook('onRequest', async (request, reply) => {
  const { auth } = reply.context.config;
  if (!request.user.roles.includes(auth)) {
    throw new Error();
  }
});
```

</details>

## Alternatives to consider

The following frameworks were used as inspiration.

[NestJS](https://github.com/nestjs/nest), [Ts.ED](https://github.com/tsedio/tsed), [tsoa](https://github.com/lukeautry/tsoa)
 readmeEtag: '"2040d508a5097a4618112bdc03bf93188fd3d07d"' readmeLastModified: Sat, 17 Aug 2024 02:07:38 GMT repositoryId: 439055297 description: >- Delightful API development: Focused on developer experience and performance. created: '2021-12-16T16:23:45Z' updated: '2025-11-21T08:43:31Z' language: TypeScript archived: false stars: 11 watchers: 1 forks: 1 owner: Jnig logo: https://avatars.githubusercontent.com/u/3729585?v=4 license: MIT repoEtag: '"51f2e1c135cf08c5572c7e70bb42a921809a3c9fd858a0e71987d5e1e25e4167"' repoLastModified: Fri, 21 Nov 2025 08:43:31 GMT category: - Data Validators - Parsers foundInMaster: true oldLocations: - https://github.com/fluxapi/fluxapi - source: openapi3 tags repository: https://github.com/entrepreneur-interet-general/open_api_schemas_to_markdown v3: true repositoryMetadata: base64Readme: >- WyFbU29mdHdhcmUgTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWNlbnNlLU1JVC1vcmFuZ2Uuc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9naXRodWIuY29tL2VudHJlcHJlbmV1ci1pbnRlcmV0LWdlbmVyYWwvb3Blbl9hcGlfc2NoZW1hc190b19tYXJrZG93bi9ibG9iL21hc3Rlci9MSUNFTlNFLm1kKQohW0NpcmNsZUNJXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NpcmNsZWNpL3Byb2plY3QvZ2l0aHViL2VudHJlcHJlbmV1ci1pbnRlcmV0LWdlbmVyYWwvb3Blbl9hcGlfc2NoZW1hc190b19tYXJrZG93bi5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUpCiFbUHlQSV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL29wZW5fYXBpX3NjaGVtYXNfdG9fbWFya2Rvd24uc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKQoKCiMgT3BlbkFQSSBDb21wb25lbnRzIFNjaGVtYXMgdG8gTWFya2Rvd24KVGhlIGdvYWwgb2YgdGhpcyBwYWNrYWdlIGlzIHRvIGdlbmVyYXRlIFtHaXRodWIgRmxhdm9yZWQgTWFya2Rvd25dKGh0dHBzOi8vZ2l0aHViLmdpdGh1Yi5jb20vZ2ZtLykgZG9jdW1lbnRhdGlvbiBvZiBDb21wb25lbnRzIFNjaGVtYXMgZnJvbSB0aGUgW09wZW5BUEkgMyBzcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikuCgpTYXkgeW91J3ZlIGdvdCBhIFlBTUwgZmlsZToKYGBgeW1sCm9wZW5hcGk6ICIzLjAuMCIKaW5mbzoKICB2ZXJzaW9uOiAiMCIKICB0aXRsZTogRGVtbwpwYXRoczoKICAvZGVtbzoKICAgIGdldDoKICAgICAgc3VtbWFyeTogRGVtbwogICAgICByZXNwb25zZXM6CiAgICAgICAgJzIwMCc6CiAgICAgICAgICBkZXNjcmlwdGlvbjogT0sKY29tcG9uZW50czoKICBzY2hlbWFzOgogICAgUmVzdWx0YXRIdW1haW46CiAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgb3BlcmF0aW9uX2lkOgogICAgICAgICAgdHlwZTogaW50ZWdlcgogICAgICAgICAgZm9ybWF0OiBpbnQ2NAogICAgICAgICAgZGVzY3JpcHRpb246IExlIG51bcOpcm8gdW5pcXVlIGRlIGwnb3DDqXJhdGlvbgogICAgICAgICAgZXhhbXBsZTogMTExOTkyMDM3MQogICAgICAgIGNhdGVnb3JpZV9wZXJzb25uZToKICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgZGVzY3JpcHRpb246IEluZGlxdWUgbGEgY2F0w6lnb3JpZSBkZSBwZXJzb25uZSBpbXBsaXF1w6llIGRhbnMgbGUgYmlsYW4gaHVtYWluCiAgICAgICAgICBleGFtcGxlOiBQw6pjaGV1ciBmcmFuw6dhaXMKICAgICAgICByZXN1bHRhdF9odW1haW46CiAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgIGRlc2NyaXB0aW9uOiBEZXNjcmlwdGlvbiBkdSBiaWxhbiBodW1haW4KICAgICAgICAgIGV4YW1wbGU6IFBlcnNvbm5lIHNlY291cnVlCiAgICAgICAgbm9tYnJlOgogICAgICAgICAgdHlwZTogbnVtYmVyCiAgICAgICAgICBmb3JtYXQ6IGludDMyCiAgICAgICAgICBtaW5pbXVtOiAwCiAgICAgICAgICBkZXNjcmlwdGlvbjogTm9tYnJlIGRlIHBlcnNvbm5lcyBpbXBsaXF1w6llcyBkYW5zIGNlIGJpbGFuCiAgICAgICAgICBleGFtcGxlOiAzCiAgICAgICAgZG9udF9ub21icmVfYmxlc3NlOgogICAgICAgICAgdHlwZTogbnVtYmVyCiAgICAgICAgICBmb3JtYXQ6IGludDMyCiAgICAgICAgICBtaW5pbXVtOiAwCiAgICAgICAgICBkZXNjcmlwdGlvbjogSW5kaXF1ZSBsZSBub21icmUgZGUgcGVyc29ubmVzIGJsZXNzw6llcyBkYW5zIGxlIGJpbGFuCiAgICAgICAgICBleGFtcGxlOiAxCgpgYGAKClRoZSBwYWNrYWdlIHdpbGwgcHJvZHVjZSBhIE1hcmtkb3duIGZpbGUgZnJvbSB0aGUgWUFNTCBzcGVjaWZpY2F0aW9uIGZpbGUgd2l0aCB0aGUgZm9sbG93aW5nIGNvbnRlbnQ6CmBgYG1hcmtkb3duCiMjIFJlc3VsdGF0SHVtYWluCnxGaWVsZHxUeXBlfERlc2NyaXB0aW9ufEV4YW1wbGV8UHJvcGVydGllc3wKfC0tLXwtLS18LS0tfC0tLXwtLS18CnxvcGVyYXRpb25faWR8aW50ZWdlcigkaW50NjQpfExlIG51bcOpcm8gdW5pcXVlIGRlIGwnb3DDqXJhdGlvbnwxMTE5OTIwMzcxfHwKfGNhdGVnb3JpZV9wZXJzb25uZXxzdHJpbmd8SW5kaXF1ZSBsYSBjYXTDqWdvcmllIGRlIHBlcnNvbm5lIGltcGxpcXXDqWUgZGFucyBsZSBiaWxhbiBodW1haW58UMOqY2hldXIgZnJhbsOnYWlzfHwKfHJlc3VsdGF0X2h1bWFpbnxzdHJpbmd8RGVzY3JpcHRpb24gZHUgYmlsYW4gaHVtYWlufFBlcnNvbm5lIHNlY291cnVlfHwKfG5vbWJyZXxudW1iZXIoJGludDMyKXxOb21icmUgZGUgcGVyc29ubmVzIGltcGxpcXXDqWVzIGRhbnMgY2UgYmlsYW58M3xtaW5pbXVtOiAwfAp8ZG9udF9ub21icmVfYmxlc3NlfG51bWJlcigkaW50MzIpfEluZGlxdWUgbGUgbm9tYnJlIGRlIHBlcnNvbm5lcyBibGVzc8OpZXMgZGFucyBsZSBiaWxhbnwxfG1pbmltdW06IDB8CmBgYAoKQW5kIGlmIHlvdSByZW5kZXIgaXQ6CiMjIFJlc3VsdGF0SHVtYWluCnxGaWVsZHxUeXBlfERlc2NyaXB0aW9ufEV4YW1wbGV8UHJvcGVydGllc3wKfC0tLXwtLS18LS0tfC0tLXwtLS18CnxvcGVyYXRpb25faWR8aW50ZWdlcigkaW50NjQpfExlIG51bcOpcm8gdW5pcXVlIGRlIGwnb3DDqXJhdGlvbnwxMTE5OTIwMzcxfHwKfGNhdGVnb3JpZV9wZXJzb25uZXxzdHJpbmd8SW5kaXF1ZSBsYSBjYXTDqWdvcmllIGRlIHBlcnNvbm5lIGltcGxpcXXDqWUgZGFucyBsZSBiaWxhbiBodW1haW58UMOqY2hldXIgZnJhbsOnYWlzfHwKfHJlc3VsdGF0X2h1bWFpbnxzdHJpbmd8RGVzY3JpcHRpb24gZHUgYmlsYW4gaHVtYWlufFBlcnNvbm5lIHNlY291cnVlfHwKfG5vbWJyZXxudW1iZXIoJGludDMyKXxOb21icmUgZGUgcGVyc29ubmVzIGltcGxpcXXDqWVzIGRhbnMgY2UgYmlsYW58M3xtaW5pbXVtOiAwfAp8ZG9udF9ub21icmVfYmxlc3NlfG51bWJlcigkaW50MzIpfEluZGlxdWUgbGUgbm9tYnJlIGRlIHBlcnNvbm5lcyBibGVzc8OpZXMgZGFucyBsZSBiaWxhbnwxfG1pbmltdW06IDB8CgojIyBJbnN0YWxsYXRpb24KYGBgCnBpcCBpbnN0YWxsIG9wZW4tYXBpLXNjaGVtYXMtdG8tbWFya2Rvd24KYGBgCgojIyBVc2FnZQpUaGUgcGFja2FnZSBwcm92aWRlcyBhIGNvbW1hbmQgbGluZSB0b29sLgpgYGAKJCBvYS10by1tZCAtaAp1c2FnZTogb2EtdG8tbWQgWy1oXSBpbnB1dF9maWxlcGF0aCBvdXRwdXRfZmlsZXBhdGggW2xvY2FsZV0KCnBvc2l0aW9uYWwgYXJndW1lbnRzOgogIGlucHV0X2ZpbGVwYXRoICAgVGhlIE9wZW5BUEkgMyBZQU1MIGZpbGVwYXRoCiAgb3V0cHV0X2ZpbGVwYXRoICBUaGUgZGVzaXJlZCBvdXRwdXQgZmlsZXBhdGggb2YgdGhlIE1hcmtkb3duIGZpbGUKICBsb2NhbGUgICAgICAgICAgIExvY2FsZSB0byB1c2Ugd2hlbiBnZW5lcmF0aW5nIHRoZSBNYXJrZG93biBmaWxlLiBBdmFpbGFibGU6CiAgICAgICAgICAgICAgICAgICBlbiwgZnIKCm9wdGlvbmFsIGFyZ3VtZW50czoKICAtaCwgLS1oZWxwICAgICAgIHNob3cgdGhpcyBoZWxwIG1lc3NhZ2UgYW5kIGV4aXQKYGBgCgpFeGFtcGxlOgpgYGAKb2EtdG8tbWQgb3Blbl9hcGkueW1sIGRvY3VtZW50YXRpb24ubWQKYGBgCgojIyBOb3RpY2UKVGhpcyBzb2Z0d2FyZSBpcyBhdmFpbGFibGUgdW5kZXIgdGhlIE1JVCBsaWNlbnNlIGFuZCB3YXMgZGV2ZWxvcGVkIGFzIHBhcnQgb2YgdGhlIFtFbnRyZXByZW5ldXIgZCdJbnTDqXLDqnQgR8OpbsOpcmFsIHByb2dyYW1dKGh0dHBzOi8vZW50cmVwcmVuZXVyLWludGVyZXQtZ2VuZXJhbC5ldGFsYWIuZ291di5mcikgYnkgdGhlIEZyZW5jaCBnb3Zlcm5tZW50LgoKUHJvamV0IGTDqXZlbG9wcMOpIGRhbnMgbGUgY2FkcmUgZHUgcHJvZ3JhbW1lIMKrIFtFbnRyZXByZW5ldXIgZOKAmWludMOpcsOqdCBnw6luw6lyYWxdKGh0dHBzOi8vZW50cmVwcmVuZXVyLWludGVyZXQtZ2VuZXJhbC5ldGFsYWIuZ291di5mcikgwrsuCg== readmeEtag: '"481c87507a13293778520645ced4918c19cd0a36"' readmeLastModified: Wed, 03 Oct 2018 11:27:27 GMT repositoryId: 130189009 description: Generate Markdown documentation from OpenAPI 3 Components Schemas created: '2018-04-19T09:07:13Z' updated: '2025-07-01T15:45:32Z' language: Python archived: false stars: 14 watchers: 13 forks: 1 owner: entrepreneur-interet-general logo: https://avatars.githubusercontent.com/u/25084749?v=4 license: MIT repoEtag: '"8988bd551c7d73ae3f3e7297035f47c2e977b4088a157cf310db3297965b8db9"' repoLastModified: Tue, 01 Jul 2025 15:45:32 GMT foundInMaster: true category: Parsers id: 7cdbea5c957437085535437b028e8fa2 - source: openapi3 tags repository: https://github.com/amishfaldu/swagger-docs v3: true id: 634e51a90d74b30e957b1b9220cc414b repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIERvY3MKClRoaXMgbGlicmFyeSBoZWxwcyB5b3UgdG8gYnVpbGQgc3dhZ2dlciBhcGkgZG9jcyBoYXNzbGUgZnJlZSBieSB1c2luZyBkZWNvcmF0b3JzLCByZWZsZWN0IG1ldGFkYXRhIGFwaS4KCiMjIERvY3VtZW50YXRpb24KCkZvciBtb3JlIGluZm9ybWF0aW9uIHJlbGF0ZWQgdG8gbGlicmFyeSB1c2FnZSBhbmQgYXBpIHJlZmVyZW5jZXMsIGhlYWQgb3ZlciB0byBwcm9qZWN0J3MgW2RvY3VtZW50YXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9BbWlzaEZhbGR1L3N3YWdnZXItZG9jcy93aWtpKS4KClRvIGZpbmQgYW5zd2VycyBmb3IgeW91ciBxdWVzdGlvbiwgYXNrIHF1ZXN0aW9ucywgYWRkIHN1Z2dlc3Rpb25zIG9yIGlkZWFzIHRvIGltcHJvdmUgdGhpcyBwcm9qZWN0LCBlbmdhZ2UgaW4gZ2VuZXJhbCBkaXNjdXNzaW9uIGhlYWQgb3ZlciB0byBwcm9qZWN0J3MgW2Rpc2N1c3Npb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9BbWlzaEZhbGR1L3N3YWdnZXItZG9jcy9kaXNjdXNzaW9ucykuCgojIyBDdXJyZW50IHN0YXRlIC8gcHJvYmxlbXMgZmFjZWQKClRvIHdyaXRlIHN3YWdnZXIgYXBpIGRvY3MgZm9yIHdlYiBhcHBzIGRldmVsb3BlZCB1c2luZyBub2RlLmpzIHdpdGhvdXQgdXNlIG9mIGZyYW1ld29ya3MgbGlrZSBuZXN0LmpzLCB5b3Ugd2lsbCBuZWVkIHRvIHdyaXRlIGFubm90YXRlZCBqc2RvYyBhbmQgc2Nyb2xsIHRob3VnaCBjb25mdXNpbmcgc3dhZ2dlciBkb2N1bWVudGF0aW9uIHRvIGZpbmQgcmlnaHQgc3BlY2lmaWNhdGlvbiBzeW50YXggZm9yIGFwaSByb3V0ZXMuCgojIyBQcm9ibGVtcyBzb2x2ZWQKCldoYXQgeW91IGFjaGVpdmUgYnkgdXNpbmcgdGhpcyBsaWJyYXJ5CgoxLiBRdWljayBzd2FnZ2VyIGFwaSBkb2NzLiA6d2hpdGVfY2hlY2tfbWFyazoKCjIuIEF1dG8tZ2VuZXJhdGUgYXBpIGRvY3MgZm9yIHRoZSByb3V0ZXMgeW91IHdyaXRlIGFuZCBhdHRhY2ggdG8gd2ViIGZyYW1ld29ya3MuIDp3aGl0ZV9jaGVja19tYXJrOgoKMy4gQ2hhbmdlcyBtYWRlIHRvIGFwaSBhbHNvIHJlZmxlY3RzIGluIHN3YWdnZXIgYXBpIGRvY3Mgd2l0aG91dCBtb2RpZnlpbmcgZnJ1c3RyYXRpbmcgZG9jIGNvZGUuIDp3aGl0ZV9jaGVja19tYXJrOgoKNC4gQXV0by1kZXRlY3QgcGFyYW1ldGVycywgYXV0aGVudGljYXRpb24sIHJlcXVlc3QgYm9keSwgcmVzcG9uc2UgYm9keSBmcm9tIHRoZSBhcGkgcm91dGVzIGRlZmluZWQuIDp3aGl0ZV9jaGVja19tYXJrOgoKNS4gTWluaW1hbCBkZXBlbmRlbmN5IGxpYnJhcnkgZm9yIGdlbmVyYXRpbmcgc3dhZ2dlciBhcGkgZG9jcy4gOndoaXRlX2NoZWNrX21hcms6CgojIyBJbXBsZW1lbnRhdGlvbgoKVGhpcyBsaWJyYXJ5IGltcGxlbWVudHMgZGVjb3JhdG9ycyBmb3IgZGVmaW5pbmcgYXBpIHJvdXRlcywgcmVsZWN0IG1ldGFkYXRhIGFwaXMgdG8gZ2VuZXJhdGUgc3dhZ2dlciBhcGkgZG9jcwoKIyMgUmVmZXJlbmNlcwoKRm9yIG1vcmUgaW5mb3JtYXRpb24gb24gZGVjb3JhdG9ycywgbWV0YWRhdGEgYW5kIHJlZmxlY3Rpb24KCjEuIFtEZWNvcmF0b3JzICYgbWV0YWRhdGEgcmVmbGVjdGlvbiBpbiBUeXBlc2NyaXB0XShodHRwOi8vYmxvZy53b2xrc29mdHdhcmUuY29tL2RlY29yYXRvcnMtcmVmbGVjdGlvbi1qYXZhc2NyaXB0LXR5cGVzY3JpcHQpCg== readmeEtag: '"27f6347e83ce936caa691cb0edefe219a840717e"' readmeLastModified: Sun, 11 Feb 2024 07:11:00 GMT repositoryId: 528825705 description: A node.js package used to build swagger docs hassle free created: '2022-08-25T11:45:33Z' updated: '2025-08-11T20:59:09Z' language: TypeScript archived: false stars: 13 watchers: 1 forks: 0 owner: AmishFaldu logo: https://avatars.githubusercontent.com/u/46478801?v=4 license: Apache-2.0 repoEtag: '"b7e2453dbfb5699b0fdfe13319ab72d47416293106d8a138706c5616fe2ea318"' repoLastModified: Mon, 11 Aug 2025 20:59:09 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/foxel/openapi3-typescript-codegen v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpMy10eXBlc2NyaXB0LWNvZGVnZW4KClR5cGVTY3JpcHQgY29kZSBnZW5lcmF0b3IgZm9yIE9wZW5BUEkgMy4wLjAKCkluc3BpcmVkIGJ5IHByb2JsZW1zIHdpdGggW29wZW5hcGktY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL01lcm1hZGUvb3BlbmFwaS1jb2RlZ2VuKQoKSWRlYSBpcyB0byBwcm92aWRlIG5hdGl2ZSBPcGVuQVBJIDMgZ2VuZXJhdG9yIHdpdGggYWxsIGxhbmd1YWdlLXNwZWNpZmljIGNvZGUgYmVpbmcgcHV0IGludG8gdGVtcGxhdGVzIHdpdGggaGVscCBvZiBnZW5lcmljIGhlbHBlcnMuCgpDdXJyZW50bHkgb25seSBUeXBlU2NyaXB0IGxhbmd1YWdlIGFuZCBBbmd1bGFyIGZyYW1ld29ya3MgYXJlIHN1cHBvcnRlZC4KCioqV29yayBpbiBwcm9ncmVzcyoqCgojIyBVc2FnZQoKIyMjIEluc3RhbGxpbmcKCmBucG0gaSAtZyBvcGVuYXBpMy10eXBlc2NyaXB0LWNvZGVnZW5gCgojIyMgQ0xJCgpgYGAKb3AzLWNvZGVnZW4ge29wZW5hcGktZGVmaW5pdGlvbi55YW1sfSBbb3B0aW9uc10KCk9wdGlvbnM6CiAgLW8sIC0tb3V0cHV0ICAgICBTcGVjaWZ5IG91dHB1dCBkaXJlY3RvcnkgICAgICAgICBbc3RyaW5nXSBbZGVmYXVsdDogIi4vb3V0LyJdCiAgLXQsIC0tdGVtcGxhdGVzICBTcGVjaWZ5IHRlbXBsYXRlcyBkaXJlY3RvcnkgKHBybyBvbmx5KSAgICAgICAgICAgICAgIFtzdHJpbmddCmBgYAoKIyMgTGltaXRhdGlvbnMgaW4gT3BlbkFQSSBzdXBwb3J0ICh0byBiZSByZW1vdmVkKQoKKiBzdXBwb3J0IG9ubHkgZm9yICdodHRwJyAoJ2Jhc2ljJyBhbmQgJ2JlYXJlcicpIGFuZCAnYXBpS2V5JyBTZWN1cml0eSBzY2hlbWFzLAoqIHBhcmFtZXRlcnMgZW5jb2RlciBzdXBwb3J0cyBvbmx5IHN0eWxlPSdmb3JtJyxleHBsb2RlPWZhbHNlIGVuY29kaW5nIHNjaGVtZSwKKiBvbmx5IEdFVCwgUE9TVCwgUFVULCBERUxFVEUgb3BlcmF0aW9ucyBjb2RlIGlzIGdlbmVyYXRlZCAK readmeEtag: '"273df9af006ceae91d646cf22328361b9cf1a654"' readmeLastModified: Tue, 15 Sep 2020 15:01:05 GMT repositoryId: 128030151 description: typescript codegen for OpenAPI 3 created: '2018-04-04T08:14:52Z' updated: '2026-01-18T13:28:53Z' language: TypeScript archived: false stars: 13 watchers: 1 forks: 10 owner: foxel logo: https://avatars.githubusercontent.com/u/1551926?v=4 repoEtag: '"ce2f3376d9db00d54fd1100e5befc1d08ead5f95972fed9d0947852df57f1960"' repoLastModified: Sun, 18 Jan 2026 13:28:53 GMT foundInMaster: true category: Parsers id: 742ee9c0aa686f002325d9be63c367de - source: openapi3 tags repository: https://github.com/narazaka/swagger-serializer v3: true repositoryMetadata: base64Readme: >- # Swagger::Serializer

[![Actions Status](https://github.com/Narazaka/swagger-serializer/workflows/Ruby/badge.svg)](https://github.com/Narazaka/swagger-serializer/actions)
[![Gem Version](https://badge.fury.io/rb/swagger-serializer.svg)](https://badge.fury.io/rb/swagger-serializer)

Swagger (OpenAPI 3) schema based serializer.

## Installation

Add this line to your application's Gemfile:

```ruby
gem 'swagger-serializer'
```

And then execute:

    $ bundle

Or install it yourself as:

    $ gem install swagger-serializer

## Usage

### Rails with swagger-dsl

use "[swagger-dsl](https://github.com/Narazaka/swagger-dsl)"!

Load it in initializer.

```ruby
# config/initializers/swagger_serializer.rb

if Rails.application.config.eager_load
  Swagger::DSL.current.config.lazy_define_paths = true
  Rails.application.config.after_initialize do
    Rails.application.reload_routes!
    Swagger::DSL.current.define_paths!
    Swagger::Schema.current = Swagger::Schema.new(Swagger::DSL.current.resolved)
  end
else
  Swagger::Schema.current = Swagger::Schema.new(Swagger::DSL.current)
  Swagger::Serializer::Store.current.serialize_options[:resolver] = Swagger::DSL.current.resolver
  Swagger::Serializer::Store.current.deserialize_options[:resolver] = Swagger::DSL.current.resolver
end
```

Use it in controllers.

```ruby
# app/controllers/application_controller.rb
class ApplicationController < ActionController::Base
  include Swagger::Serializer::RailsController
  extend Swagger::DSL::RailsController

  def render_ok(data, context = nil)
    render_as_schema 200, :json, data, context
  end

  def render_bad(data, context = nil)
    render_as_schema 400, :json, data, context
  end
end
```

```ruby
# app/controllers/users_controller.rb
class UsersController < ApplicationController
  swagger :index do
    render 200 do
      array! { cref! UserSerializer }
    end
  end
  def index
    render_ok User.all
  end

  swagger :show do
    params do
      path :id, schema: :integer
    end

    render 200 do
      cref! UserSerializer
    end
  end
  def show
    render_ok User.find(schema_params[:id])
  end

  swagger :create do
    body format: [:form, :json] do
      user :object do
        name :string
      end
    end
    
    render 200 do
      cref! UserSerializer
    end
    render 400 do
      additionalProperties! do
        array! { string! }
      end
    end
  end
  def create
    @user = User.new(schema_params[:user])

    if @user.save
      render_ok @user
    else
      render_bad @user.errors
    end
  end

  swagger :update do
    params do
      path :id, schema: :string
    end

    body format: [:form, :json] do
      user :object do
        name :string
      end
    end
    
    render 200 do
      cref! UserSerializer
    end
    render 400 do
      additionalProperties! do
        array! { string! }
      end
    end
  end
  def update
    @user = User.find(schema_params[:id])

    if @user.update(schema_params[:user])
      render_ok @user
    else
      render_bad @user.errors
    end
  end

  swagger :destroy do
    params do
      path :id, schema: :string
    end
  end
  def destroy
    @user.destroy!
    head :no_content
  end
end
```

Would you want to customize serialization?

```ruby
# app/serializers/base_serializer.rb
class BaseSerializer
  include Swagger::Serializer
  extend Swagger::DSL::Serializer
end
```

```ruby
# app/serializers/user_serializer.rb
class UserSerializer < BaseSerializer
  swagger do
    id :integer
    name :string, optional: true
  end

  def name
    "#{@model.name}!!!!"
  end
end
```

Now you can get `{ "id" => 42, "name" => "me!!!!" }`.

This serializer class detection uses the schema's `title` key.
If you want to use `Foo::BarSerializer`, set `Foo-Bar` to `title` key.

The key is configurable by

```ruby
# in config/initializers/swagger_serializer.rb
Swagger::Serializer::Store.current.serialize_options[:inject_key] = "my_inject_key"
Swagger::Serializer::Store.current.deserialize_options[:inject_key] = "my_inject_key"
Swagger::DSL.current.config.inject_key = "my_inject_key"
```

Sometimes model needs direct serialize.

```ruby
# app/models/application_record.rb
class ApplicationRecord < ActiveRecord::Base
  self.abstract_class = true
  
  include Swagger::Serializer::Model
end
```

Now you can get serialized result by `p User.first.serialize`.

### Rails with raw schema

Write your OpenAPI spec.

```yaml
# swagger.yml
openapi: 3.0.0
info:
  title: example api
  version: 0.1.0
components:
  schemas:
    User: &User
      title: User
      type: object
      properties:
        id:
          type: integer
        name:
          type: string
      required: [id]
paths:
  /users:
    get:
      responses:
        200:
          content:
            application/json:
              schema:
                type: array
                items: *User
  /users/{id}:
    get:
      responses:
        200:
          content:
            application/json:
              schema: *User

```

Load it in initializer.

```ruby
# config/initializers/swagger_serializer.rb

Swagger::Schema.load_file_to_current(
  __dir__ + "/../../swagger.yml",
)
```

Use it in controllers.

```ruby
# app/controllers/application_controller.rb
class ApplicationController < ActionController::Base
  include Swagger::Serializer::RailsController

  def render_ok(data, context = nil)
    render_as_schema 200, :json, data, context
  end
end
```

```ruby
# app/controllers/users_controller.rb
class UsersController < ApplicationController
  def index
    render_ok User.all
  end

  def show
    render_ok User.find(params[:id])
  end
end
```

Would you want to customize serialization?

```ruby
# app/serializers/base_serializer.rb
class BaseSerializer
  include Swagger::Serializer
end
```

```ruby
# app/serializers/user_serializer.rb
class UserSerializer < BaseSerializer
  def name
    "#{@model.name}!!!!"
  end
end
```

Now you can get `{ "id" => 42, "name" => "me!!!!" }`.

Sometimes model needs direct serialize.

```ruby
# app/models/application_record.rb
class ApplicationRecord < ActiveRecord::Base
  self.abstract_class = true
  
  include Swagger::Serializer::Model
end
```

Now you can get serialized result by `p User.first.serialize`.

### @context

Collection serialization sometimes needs context data for avoiding N+1 access or some other reason.

`render_as_schema`'s 4th parameter is the context that will be passed to serializers `@context`.

```ruby
# app/serializers/item_serializer.rb
class ItemSerializer < BaseSerializer
  swagger do
    id :integer
    rate :integer
  end

  def rate
    @context[:rates][id]
  end
end

# app/controllers/items_controller.rb
class ItemsController < ApplicationController
  swagger :index do
    render 200 do
      array! do
        cref! ItemSerializer
      end
    end
  end

  def index
    items = Item.all
    rates = ItemRate.calc_item_rates(items)
    render_ok items, { rates: rates }
  end
end
```

## Development

After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.

To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).

## Contributing

Bug reports and pull requests are welcome on GitHub at https://github.com/Narazaka/swagger-serializer.
 readmeEtag: '"c78406061fe60b821c3e0b80cbdf6a792e09da4e"' readmeLastModified: Mon, 21 Nov 2022 07:45:36 GMT repositoryId: 224905174 description: Swagger (OpenAPI 3) schema based serializer for ruby created: '2019-11-29T18:34:41Z' updated: '2025-05-13T13:09:20Z' language: Ruby archived: false stars: 13 watchers: 3 forks: 2 owner: Narazaka logo: https://avatars.githubusercontent.com/u/1712548?v=4 license: Zlib repoEtag: '"122226e232dea98a6f72e995b0d2caa80dc90ae021532f8f49afc02e922f6ea1"' repoLastModified: Tue, 13 May 2025 13:09:20 GMT foundInMaster: true category: Code Generators id: 68fbc50ba4ab0b9974f93a27abd30aea - source: openapi3 tags repository: https://github.com/lornajane/openapi-blockly-representation v3: true id: c97dc48c739fd02596e7521df5111ca3 repositoryMetadata: base64Readme: >- IyBCbG9ja2x5IHJlcHJlc2VudGF0aW9uIG9mIE9wZW5BUEkKClRoaXMgZG9lc24ndCBjb21waWxlIHRvIE9wZW5BUEkgKHB1bGwgcmVxdWVzdHMgd2VsY29tZSksIGJ1dCBJIHVzZSB0aGVzZSBibG9ja3MgYXMgYSB0ZWFjaGluZyBhaWQuIFlvdSdyZSB3ZWxjb21lIHRvIHVzZSB0aGVtIHRvby4KCiFbQSBzaW5nbGUgZW5kcG9pbnQgcmVwcmVzZW50ZWQgYnkgYmxvY2tzXSguL2ltYWdlcy9vcGVuYXBpLXNjaGVtYS1yZWYtYmxvY2tzLnBuZykKCiMjIEdldCBzdGFydGVkCgpSdW4gYG5wbSBpbnN0YWxsYC4KClRoZW4gc2VydmUgYGluZGV4Lmh0bWxgIGxvY2FsbHkuIEl0J3MgdXNlZnVsIHRvIGhhdmUgaXQgcmVsb2FkLCBzbyBJIHVzZSB0aGlzOgoKYGBgCm5weCBicm93c2VyLXN5bmMgc3RhcnQgLS1zZXJ2ZXIgLS1maWxlcyBpbmRleC5odG1sCmBgYAoKIyMjIEV4YW1wbGUgYmxvY2tzCgpUcnkgYGV4YW1wbGUuanNvbmAuCgojIyMgVGVjaG5pY2FsIG5vdGVzCgpUaGUgInNhdmUiIGZ1bmN0aW9uYWxpdHkgdXNlcyBsb2NhbCBzdG9yYWdlIHNvIHlvdSBjYW4gZW5naW5lZXIgdGhlIGJsb2NrcyB3aXRob3V0IGxvc2luZyB3aGF0IHlvdSBidWlsdCBvbiByZWxvYWQuCgojIyBVc2VkIGJ5CgpTZW5kIGEgcHVsbCByZXF1ZXN0IGlmIHlvdSB1c2UgdGhpcyBwcm9qZWN0PyBMZXQncyBzaGFyZSEKCi0gTG9ybmEgKCBbbWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9sb3JuYWphbmUvKSApIGluIG15IHRhbGsgW09wZW5BUEkgZm9yIERvY3VtZW50YXJpYW5zXShodHRwczovL25vdGkuc3QvbG9ybmFqYW5lLzRPamVtMi9vcGVuYXBpLWZvci1kb2N1bWVudGFyaWFucykK readmeEtag: '"824b101a611d17694bc05c076a3f012459386d2e"' readmeLastModified: Sat, 10 Jun 2023 19:00:45 GMT repositoryId: 637387454 description: Demo repo uses blockly to visualise structure of OpenAPI created: '2023-05-07T12:05:23Z' updated: '2025-01-17T07:12:59Z' language: HTML archived: false stars: 12 watchers: 1 forks: 0 owner: lornajane logo: https://avatars.githubusercontent.com/u/172607?v=4 license: Apache-2.0 repoEtag: '"44ca2b05dfbf8c6ce6eeda701667d798d74e6eae810fdd5be657fa8cd977bf23"' repoLastModified: Fri, 17 Jan 2025 07:12:59 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/yapily/yapily-sdk-java v3: true id: 41fae7ca06526a669b911fc4db371eb6 repositoryMetadata: base64Readme: >- # Yapily Java SDK
[![GitHub version](https://d25lcipzij17d.cloudfront.net/badge.svg?id=gh&type=6&v=2.13.1&x2=0)](http://badge.fury.io/gh/boennemann%2Fbadges)

This SDK library was generated using [OpenApi Generator](https://github.com/OpenAPITools/openapi-generator). The SDK can be used as a module in your code and the examples demonstrate how to connect to financial institutions integrated with Yapily.

## Requirements

To connect to the Yapily API, you will need to register your 
application at [https://dashboard.yapily.com]().

Specify the financial institutions you want to work with and save your application. You will then be prompted to download your application credentials. 
These application credentials can then be used to authorise your requests against the [Yapily API](https://docs.yapily.com/#get-started).

## Installation

The SDK is currently available in the Yapily maven repository and 
can be included in your project 
by adding it to your dependencies

#### Maven

Repository:

```xml
<repositories>
   <!-- other repos-->
   <repository>
       <id>yapily-repo</id>
       <name>yapily-repo</name>
       <url>http://maven.yapily.com/</url>
   </repository>
</repositories>
```

Dependency:

```xml
<dependencies>
   <!-- other dependencies -->
   <dependency>
       <groupId>yapily</groupId>
       <artifactId>yapily-sdk</artifactId>
       <version>2.13.1</version>
   </dependency>
</dependencies>
```

#### Gradle

Repository:

```groovy
repositories {
    // other repos
    maven {
        url "http://maven.yapily.com/"
    }
}
```

Dependency:

```groovy
compile group: 'yapily', name: 'yapily-sdk', version: '2.13.1'
```

#### Download JAR

The JAR can also be downloaded from a tagged release, 
or this project can be cloned/downloaded and packaged into a 
library JAR to be included in your project.

## Usage

Sample usages of the SDK can be seen in the `examples` folder.

- Configure the default api client

```java
ApiClient defaultClient = Configuration.getDefaultApiClient();
// Configure the API authentication
HttpBasicAuth basicAuth = (HttpBasicAuth) defaultClient.getAuthentication("basicAuth");
// Replace these demo constants with your application credentials
basicAuth.setUsername(APPLICATION_ID);
basicAuth.setPassword(APPLICATION_SECRET);
```

- Or configure a new one for multiple application cases 

```java
ApiClient applicationClient = new ApiClient();
// Configure the API authentication
HttpBasicAuth basicAuth = (HttpBasicAuth) applicationClient.getAuthentication("basicAuth");
// Replace these demo constants with your application credentials
basicAuth.setUsername(APPLICATION_ID);
basicAuth.setPassword(APPLICATION_SECRET);
InstitutionsApi institutionsApi = new InstitutionsApi();
institutionsApi.setApiClient(applicationClient);
```

- Retrieve a list of available financial institutions to connect to

```java
InstitutionsApi institutionsApi = new InstitutionsApi();
List<Institution> institutions = institutionsApi.getInstitutionsUsingGET().getData();
```

- Creating users and retrieving users for your application registered in the Yapily Dashboard
```java
final ApplicationUsersApi applicationUsersApi = new ApplicationUsersApi();
NewApplicationUser exampleUser = new NewApplicationUser();
exampleUser.setApplicationUserId("java-sdk@yapily.com");
applicationUsersApi.addUserUsingPOST(exampleUser);

List<ApplicationUser> users = applicationUsersApi.getUsersUsingGET();
```

- Create an authorization url in order for your users to give consent on accessing their account data. 

```java
final AccountsApi accountsApi = new AccountsApi();
AccountAuthorisationRequest accountAuthorisationRequest = new AccountAuthorisationRequest();
accountAuthorisationRequest.setApplicationUserId(applicationUserId);
accountAuthorisationRequest.setInstitutionId(institutionId);
/**
* Use the defaults
*/
accountAuthorisationRequest.setAccountRequest(new AccountRequest());
ApiResponseOfAuthorisationRequestResponse authorizationResponse = accountsApi.initiateAccountRequestUsingPOST(accountAuthorisationRequest, null, null, null);
String directUrl = authorizationResponse.getData().getAuthorisationUrl();
```

- Get the most resent Consent for the configured user and institution. Make sure to check that the Consent is AUTHORIZED otherwise fail the request
```java
Consent consent = consentsApi.getConsentsUsingGET(
        Collections.singletonList(applicationUserId),
        Collections.emptyList(),
        Collections.singletonList(institutionId),
        Collections.emptyList(),
        null,
        null,
        1,
        null).getData().stream()
        .filter(c -> c.getStatus().equals(Consent.StatusEnum.AUTHORIZED))
        .findFirst()
        .orElseThrow(() -> new RuntimeException(String.format("No valid consent token present for application user %s", applicationUserId)));

```
 
- Providing that the user has given consent and you have an AUTHORIZED Consent, obtain the user account details using the consent token

```java
String consentToken = consent.getConsentToken();
ApiListResponseOfAccount accountsResponse = accountsApi.getAccountsUsingGET(consentToken);
List<Account> accounts = accountsResponse.getData();
```

- Return user transaction details, using an account the user has given consent to.

```java
String consentToken = consent.getConsentToken();
ApiListResponseOfTransaction transactionsResponse = transactionsApi.getTransactionsUsingGET(
                                        consentToken, 
                                        accountId, 
                                        Collections.emptyList(),
                                        "1980-01-01T00:00:00.000Z",
                                        "2020-01-01T00:00:00.000Z",
                                        10,
                                        null,
                                        0,
                                        null);
List<Transaction> transactions = transactionsResponse.getData();
```

- Check that the identity feature is provided for this institution.
```java
Institution institution = institutionsApi.getInstitutionUsingGET(institutionId);
Boolean supportsIdentity = institution.getFeatures().stream()
        .anyMatch(featuresEnum -> featuresEnum != null && featuresEnum.equals(Institution.FeaturesEnum.IDENTITY));
```

- If the identity feature is supported return user identity details. This will fail if the identity feature is not supported.
```java
String consentToken = consent.getConsentToken();
IdentityApi identityApi = new IdentityApi();
ApiResponseOfIdentity identityResp = identityApi.getIdentityUsingGET(consentToken); 
Identity identity = identityResp.getData();
```

- Create a new payment request detailing what type of payment you want to issue on behalf of your user e.g. a Domestic 
payment to send £1.00 to another account using an account number and sort code:

```java
PaymentRequest paymentRequest = new PaymentRequest();

Amount amount = new Amount();
amount.setAmount(new BigDecimal(1));
amount.setCurrency("GBP");

Payee payee = new Payee();
payee.setName("John Doe");

List<AccountIdentification> payeeAccountIdentifications = new ArrayList<>();

AccountIdentification accountNumberIdentification = new AccountIdentification();
accountNumberIdentification.setType(AccountIdentification.TypeEnum.ACCOUNT_NUMBER);
accountNumberIdentification.setIdentification("70000005");
payeeAccountIdentifications.add(accountNumberIdentification);

AccountIdentification sortCodeIdentification = new AccountIdentification();
sortCodeIdentification.setType(AccountIdentification.TypeEnum.SORT_CODE);
sortCodeIdentification.setIdentification("700001");
payeeAccountIdentifications.add(sortCodeIdentification);

payee.setAccountIdentifications(payeeAccountIdentifications);

paymentRequest.setAmount(amount);
paymentRequest.setPayee(payee);
paymentRequest.setType(PaymentRequest.TypeEnum.DOMESTIC_PAYMENT);
paymentRequest.setReference("Sending £1.00 to John Doe via. Yapily");
paymentRequest.setPaymentIdempotencyId("anyUniqueStringOver18characters");
```

- Create a new payment authorisation request using the payment request to generate an authorization url that your users can use to authorize the payment:

```java
PaymentsApi paymentsApi = new PaymentsApi();
PaymentAuthorisationRequest paymentAuthorisationRequest = new PaymentAuthorisationRequest();
paymentAuthorisationRequest.setApplicationUserId(applicationUserId);
paymentAuthorisationRequest.setInstitutionId(institutionId);
paymentAuthorisationRequest.setPaymentRequest(paymentRequest);
ApiResponseOfPaymentAuthorisationRequestResponse authorizationResponse = 
                        paymentsApi.createPaymentAuthorisationUsingPOST(paymentAuthorisationRequest, null, null, null);
String url = authorizationResponse.getData().getAuthorisationUrl();
```

- Submit a payment request using the same payment request object in the authorization request and the AUTHORIZED consent given by the user:

```java
String consentToken = consent.getConsentToken();
ApiResponseOfPaymentResponse response = paymentsApi.createPaymentUsingPOST(consentToken, paymentRequest);
```

- Check the payment status:
```java
String consentToken = consent.getConsentToken();
String paymentId =  paymentResponse.getData().getId();
ApiResponseOfPaymentResponse apiResponseOfPaymentResponse = paymentsApi.getPaymentStatusUsingGET(paymentId, consentToken);
```


## Further information

For more information on how to get connected, visit the [Yapily Knowledge Base](https://kb.yapily.com).

###### [Website](https://yapily.com) | [Legal](https://yapily.com/legal-policies) | [Contact Us](mailto:info@yapily.com) 

 readmeEtag: '"47d36eb12a8b997bf64a83f71f118f6684009623"' readmeLastModified: Fri, 07 Oct 2022 11:52:19 GMT repositoryId: 148512801 description: >- Java SDK generated against the Yapily API. This SDK can be used to connect to Open Banking entities. created: '2018-09-12T16:54:43Z' updated: '2022-06-09T14:16:35Z' language: Java archived: false stars: 12 watchers: 6 forks: 3 owner: yapily logo: https://avatars.githubusercontent.com/u/36482918?v=4 license: MIT repoEtag: '"2c32f50f3c1a175ed1d9091ccccebf56acb6f36bc4b3b33dd380da5c9ddf271b"' repoLastModified: Thu, 09 Jun 2022 14:16:35 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/notethan/scorpio v3: true repositoryMetadata: base64Readme: >- # Scorpio

![Test CI Status](https://github.com/notEthan/scorpio/actions/workflows/test.yml/badge.svg?branch=stable)
[![Coverage Status](https://coveralls.io/repos/github/notEthan/scorpio/badge.svg)](https://coveralls.io/github/notEthan/scorpio)

Scorpio is a library that helps you, as a client, consume an HTTP service described by an OpenAPI document. You provide the OpenAPI description document, a little bit of configuration, and Scorpio will take that and dynamically generate an interface for you to call the service's operations and interact with its resources as an ORM.

Note: The canonical location of this README is on [RubyDoc](http://rubydoc.info/gems/scorpio/). When viewed on [Github](https://github.com/notEthan/scorpio/), it may be inconsistent with the latest released gem, and Yardoc links will not work.

## Background

To start with, you need an OpenAPI (formerly known as Swagger) document describing a service you will be consuming. v2 and v3 are both supported.[^1] This document can be written by hand or sometimes generated from other existing sources. The creation of an OpenAPI document describing your service is outside the scope of Scorpio. Here are several resources on OpenAPI:

- [OpenAPI Specification at Wikipedia](https://en.wikipedia.org/wiki/OpenAPI_Specification)
- [OpenAPI Initiative](https://www.openapis.org/) is the official web site for OpenAPI
- [OpenAPI Specification on GitHub](https://github.com/OAI/OpenAPI-Specification)
- [swagger.io](https://swagger.io/) API tooling

OpenAPI relies on the definition of schemas using the JSON schema specification, which can be learned about at https://json-schema.org/

Once you have the OpenAPI document describing the service you will consume, you can get started implementing the code that will interact with that service.

[^1]: Certain features may be missing, but Scorpio tries to make workarounds easy. Issues and pull requests regarding missing functionality are welcome.

## Pet Store (using Scorpio::ResourceBase)

Let's dive into some code, shall we? If you have learned about OpenAPI, you likely learned using the example of the Pet Store service. This README will use the same service. Its documentation is at https://petstore.swagger.io/.

Using the OpenAPI document, we can start interacting with the pet store with very little code. Here is that code, with explanations of each part in the comments.

```ruby
require 'scorpio'
# PetStore is a module to contain our pet store related classes.
# it is optional - your naming conventions are your own.
module PetStore
  # Scorpio's recommended structure is to have a base class which inherits from
  # Scorpio::ResourceBase to represent the Pet Store and all its resources.
  #
  # you configure the openapi document and other shared configuration on this class.
  class Resource < Scorpio::ResourceBase
    # set the openapi document. you'll usually want this to be a file in your local filesystem
    # (making network calls at application boot time is usually a bad idea), but for this
    # example we will do a quick-and-dirty http get.
    require 'json'
    self.openapi_document = JSON.parse(Faraday.get('https://petstore.swagger.io/v2/swagger.json').body)
  end

  # a Pet is a resource of the pet store, so inherits from PetStore::Resource
  class Pet < Resource
    # setting the tag name tells Scorpio to associate operations tagged with 'pet' with this
    # class and its instances. this lets you call operations such as addPet, updatePet, etc.
    self.tag_name = 'pet'

    # setting the schemas which represent a Pet will let scorpio return results from operation
    # calls properly instantiated as Pet instances. for example, calling getPetById will return
    # a PetStore::Pet instance since its success response refers to #/definitions/Pet.
    #
    # this works for nested structures as well, e.g. findPetsByStatus returns an array of
    # #/definitions/Pet and likewise Scorpio will return an array of PetStore::Pet instances.
    #
    # this also adds accessors for properties of the schema - in this case #id, #name, #tags, etc.
    self.represented_schemas = [openapi_document.definitions['Pet']]
  end
end
```

That should be all you need to start calling operations:

```ruby
# call the operation findPetsByStatus
# doc: https://petstore.swagger.io/#/pet/findPetsByStatus
sold_pets = PetStore::Pet.findPetsByStatus(status: 'sold')
# sold_pets is an array-like collection of PetStore::Pet instances

pet = sold_pets.sample

pet.tags.map(&:name)
# note that you have accessors on PetStore::Pet like #tags, and also that
# tags have accessors for properties 'name' and 'id' from the tags schema
# (your tag names will be different depending on what's in the pet store)
# => ["aucune"]

# compare to getPetById: https://petstore.swagger.io/#/pet/getPetById
pet == PetStore::Pet.getPetById(petId: pet['id'])
# pet is the same, retrieved using the getPetById operation

# let's name the pet after ourself
pet.name = ENV['USER']

# store the result in the pet store. note the updatePet call from the instance - our
# calls so far have been on the class PetStore::Pet, but scorpio defines instance
# methods to call operations where appropriate as well.
# updatePet: https://petstore.swagger.io/#/pet/updatePet
pet.updatePet

# check that it was saved
PetStore::Pet.getPetById(petId: pet['id']).name
# => "ethan" (unless for some reason your name is not Ethan)

# here is how errors are handled:
PetStore::Pet.getPetById(petId: 0)
# raises: Scorpio::HTTPErrors::NotFound404Error
#   Error calling operation getPetById on PetStore::Pet:
#   {"code":1,"type":"error","message":"Pet not found"}
```

Isn't that cool? You get class methods like getPetById, instance methods like updatePet, attribute accessors like #name and #tags, all dynamically generated from the OpenAPI description. You just make a few classes with a line or two of configuration in each.

## Pet Store (using Scorpio::OpenAPI classes)

You do not have to define resource classes to use Scorpio to call OpenAPI operations - the classes Scorpio uses to represent concepts from OpenAPI can be called directly. Scorpio uses [JSI](https://github.com/notEthan/jsi) classes to represent OpenAPI schemes such as the Document and its Operations.

We start by instantiating the OpenAPI document. `Scorpio::OpenAPI::Document.from_instance` returns a V2 or V3 OpenAPI Document class instance.

```ruby
require 'scorpio'
pet_store_doc = Scorpio::OpenAPI::Document.from_instance(JSON.parse(Faraday.get('https://petstore.swagger.io/v2/swagger.json').body))
# => #{<Scorpio::OpenAPI::V2::Document fragment="#"> "swagger" => "2.0", ...}
```

The OpenAPI document holds the JSON that represents it, so to get an Operation you go through the document's paths, just as it is represented in the JSON.

```ruby
# the store inventory operation will let us see what statuses there are in the store.
inventory_op = pet_store_doc.paths['/store/inventory']['get']
# => #{<JSI (Scorpio::OpenAPI::V2::Operation)>
#      "summary" => "Returns pet inventories by status",
#      "operationId" => "getInventory",
#      ...
#    }
```

Alternatively, Scorpio defines a helper `Document#operations` which behaves like an Enumerable of all the Operations in the Document. It can be subscripted with the `operationId`:

```ruby
inventory_op = pet_store_doc.operations['getInventory']
# => returns the same inventory_op as above.
```

Now that we have an operation, we can run requests from it. {Scorpio::OpenAPI::Operation#run} performs the operation by running a request. If the response is an error, a {Scorpio::HTTPError} subclass will be raised. On success, it returns the response body entity, instantiated according to the OpenAPI schema for the operation response, if specified. For more detail on how json-schema instances are represented, see the gem [JSI](https://github.com/notEthan/jsi).

```ruby
inventory = inventory_op.run
# => #{<JSI>
#      "unavailable" => 4,
#      "unloved - needs a home" => 1,
#      "available" => 2350,
#      "sold" => 5790,
#      "dog" => 1,
#    }
```

let's pick a state and find a pet. we'll go through the rest of the example in the ResourceBase section pretty much like it is up there:

```ruby
# call the operation findPetsByStatus
# doc: https://petstore.swagger.io/#/pet/findPetsByStatus
sold_pets = pet_store_doc.operations['findPetsByStatus'].run(status: 'sold')
# sold_pets is an array-like collection of JSI instances

pet = sold_pets.detect { |pet| pet.tags.any? }

pet.tags.map(&:name)
# note that you have accessors on the returned JSI like #tags, and also that
# tags have accessors for properties 'name' and 'id' from the tags schema
# (your tag names will be different depending on what's in the pet store)
# => ["aucune"]

# compare the pet from findPetsByStatus to one returned from getPetById
# doc: https://petstore.swagger.io/#/pet/getPetById
pet_by_id = pet_store_doc.operations['getPetById'].run(petId: pet['id'])

# unlike ResourceBase instances above, JSI instances have stricter
# equality and the pets returned from different operations are not
# equal, because they are in different JSON documents.
pet_by_id == pet
# => false

# let's name the pet after ourself
pet.name = ENV['USER']

# store the result in the pet store.
# updatePet: https://petstore.swagger.io/#/pet/updatePet
pet_store_doc.operations['updatePet'].run(body_object: pet)

# check that it was saved
pet_store_doc.operations['getPetById'].run(petId: pet['id']).name
# => "ethan" (unless for some reason your name is not Ethan)

# here is how errors are handled:
pet_store_doc.operations['getPetById'].run(petId: 0)
# raises: Scorpio::HTTPErrors::NotFound404Error
#   Error calling operation getPetById:
#   {"code":1,"type":"error","message":"Pet not found"}
```

### Another Example: Blog

For another example of an API that a client interacts with using Scorpio::ResourceBase, Scorpio's tests implement the Blog service. This is defined in test/blog.rb. The service uses ActiveRecord models and Sinatra to make a simple RESTful service.

Its API is described in `test/blog.openapi.yml`, defining the Article resource, several operations, and schemas. The client is set up in `test/blog_scorpio_models.rb`. The base class BlogModel defines the base_url and the api description, as well as some other optional setup done for testing. Its operations are tested in `test/scorpio_test.rb`.

## Scorpio::ResourceBase

Scorpio::ResourceBase is the main class used in abstracting on OpenAPI document. Scorpio::ResourceBase aims to represent RESTful resources in ruby classes with as little code as possible, given a service with a properly constructed OpenAPI document.

A class which subclasses Scorpio::ResourceBase directly (such as PetStore::Resource above) should generally represent the whole API - you set the openapi_document and other configuration on this class. As such, it is generally not instantiated. Its subclasses, representing resources with a tag or with schema definitions in the OpenAPI document, are what you mostly instantiate and interact with.

A model representing a resource needs to be configured, minimally, with:

- the OpenAPI document describing the API
- the schemas that represent instances of the model, if any

If the resource has HTTP operations associated with it (most, but not all resources will):

- a tag name identifying its tagged operations

When these are set, Scorpio::ResourceBase looks through the API description and dynamically sets up methods for the model:

- accessors for properties of the model defined as properties of schemas representing the resource in the description document
- API method calls on the model class and, where appropriate, on the model instance

## Scorpio::Ur

If you need a more complete representation of the HTTP request and/or response, Scorpio::OpenAPI::Operation#run_ur or Scorpio::Request#run_ur will return a representation of the request and response defined by the gem [Ur](https://github.com/notEthan/ur). See that link for more detail. Relating to the example above titled "Pet Store (using Scorpio::OpenAPI classes)", this code will return an Ur:

```ruby
inventory_op = Scorpio::OpenAPI::Document.from_instance(JSON.parse(Faraday.get('https://petstore.swagger.io/v2/swagger.json').body)).paths['/store/inventory']['get']
inventory_ur = inventory_op.run_ur
# => #{<Scorpio::Ur fragment="#"> ...}
```

### Scorpio ResourceBase pickle adapter

Scorpio provides a pickle adapter to use models with [Pickle](https://rubygems.org/gems/pickle). `require 'scorpio/pickle_adapter'`, ensure that the pickle ORM adapter is enabled, and you should be able to create models as normal with pickle.

### Google API discovery service

An initial implementation of Scorpio::ResourceBase was based on the format defined for Google's API discovery service.

For background on the Google discovery service and the API description format it defines, see:

- https://developers.google.com/discovery/
- https://developers.google.com/discovery/v1/reference/

This format is still supported indirectly, by converting from a Google API document to OpenAPI using `Scorpio::Google::RestDescription#to_openapi_document`. Example conversion looks like:

```ruby
class MyModel < Scorpio::ResourceBase
  rest_description_doc = YAML.load_file('path/to/doc.yml')
  rest_description = Scorpio::Google::RestDescription.new(rest_description_doc)
  self.openapi_document = rest_description.to_openapi_document

  # ... the remainder of your setup and model code here
end
```

## Other

The detailed, machine-interpretable description of an API provided by a properly-constructed OpenAPI document opens up numerous possibilities to automate aspects of clients and services to an API. These are planned to be implemented in Scorpio:

- constructing test objects in a manner similar to FactoryBot, allowing you to write tests that depend on a service without having to interact with an actual running instance of that service to run your tests
- rack middleware to test that outgoing HTTP responses are conformant to their response schemas
- rack middleware to test that incoming HTTP requests are conformant to their request schemas, and that the service handles bad requests appropriately (e.g. ensuring that for any bad request, the service responds with a 4xx error instead of 2xx).
- integrating with ORMs to generate HTTP responses that are conformant to the response schema corresponding to the resource corresponding to the ORM model
- generating model validations for ORMs

## License

[<img align="right" src="https://www.gnu.org/graphics/agplv3-155x51.png">](https://www.gnu.org/licenses/agpl-3.0.html)

Scorpio is licensed under the terms of the [GNU Affero General Public License version 3](https://www.gnu.org/licenses/agpl-3.0.html).

Unlike the MIT or BSD licenses more commonly used with Ruby gems, this license requires that if you modify Scorpio and propagate your changes, e.g. by including it in a web application, your modified version must be publicly available. The common path of forking on Github should satisfy this requirement.
 readmeEtag: '"519bf7dee47656ac88fa65e1d1dc5105840f42b0"' readmeLastModified: Mon, 01 Jul 2024 20:54:51 GMT repositoryId: 69611598 description: Resource-based Web API Client built on OpenAPI created: '2016-09-29T22:03:37Z' updated: '2026-02-02T21:24:52Z' language: Ruby archived: false stars: 12 watchers: 1 forks: 2 owner: notEthan logo: https://avatars.githubusercontent.com/u/133719?v=4 license: NOASSERTION repoEtag: '"cf136f0653f9141230418b52a14657d2945b2e348a99b9be4221a3865027d6d7"' repoLastModified: Mon, 02 Feb 2026 21:24:52 GMT foundInMaster: true name: Scorpio homepage: https://github.com/notEthan/Scorpio language: Ruby source_description: OpenAPI 2 and 3 implementation offering a HTTP client library category: - Client Implementations - Parsers id: f5eaaee9b2fb544cef330dcd2c4f0546 - source: openapi3 tags repository: https://github.com/avanov/openapi-type v3: true repositoryMetadata: base64Readme: >- Li4gX2JhZGdlczoKCi4uIGltYWdlOjogaHR0cHM6Ly9naXRodWIuY29tL2F2YW5vdi9vcGVuYXBpLXR5cGUvd29ya2Zsb3dzL0NJL2JhZGdlLnN2Zz9icmFuY2g9ZGV2ZWxvcAogICAgOnRhcmdldDogaHR0cHM6Ly9naXRodWIuY29tL2F2YW5vdi9vcGVuYXBpLXR5cGUvYWN0aW9ucz9xdWVyeT1icmFuY2glM0FkZXZlbG9wCgouLiBpbWFnZTo6IGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9hdmFub3Yvb3BlbmFwaS10eXBlL2JhZGdlLnN2Zz9icmFuY2g9ZGV2ZWxvcAogICAgOnRhcmdldDogaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL2F2YW5vdi9vcGVuYXBpLXR5cGU/YnJhbmNoPWRldmVsb3AKCi4uIGltYWdlOjogaHR0cHM6Ly9yZXF1aXJlcy5pby9naXRodWIvYXZhbm92L29wZW5hcGktdHlwZS9yZXF1aXJlbWVudHMuc3ZnP2JyYW5jaD1tYXN0ZXIKICAgIDp0YXJnZXQ6IGh0dHBzOi8vcmVxdWlyZXMuaW8vZ2l0aHViL2F2YW5vdi9vcGVuYXBpLXR5cGUvcmVxdWlyZW1lbnRzLz9icmFuY2g9bWFzdGVyCiAgICA6YWx0OiBSZXF1aXJlbWVudHMgU3RhdHVzCgouLiBpbWFnZTo6IGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL29wZW5hcGktdHlwZS9iYWRnZS8/dmVyc2lvbj1sYXRlc3QKICAgIDp0YXJnZXQ6IGh0dHBzOi8vb3BlbmFwaS10eXBlLnJlYWR0aGVkb2NzLmlvL2VuL2xhdGVzdC8KICAgIDphbHQ6IERvY3VtZW50YXRpb24gU3RhdHVzCgouLiBpbWFnZTo6IGh0dHA6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3Yvb3BlbmFwaS10eXBlLnN2ZwogICAgOnRhcmdldDogaHR0cHM6Ly9weXBpLnB5dGhvbi5vcmcvcHlwaS9vcGVuYXBpLXR5cGUKICAgIDphbHQ6IExhdGVzdCBQeVBJIFJlbGVhc2UKCgpPcGVuQVBJIFR5cGUKPT09PT09PT09PT09CgpPcGVuQVBJIHNwZWNpZmljYXRpb24gcmVwcmVzZW50ZWQgYXMgYSBQeXRob24gdHlwZS4gVXNlIGl0IHRvIHBhcnNlIHNwZWNpZmljYXRpb25zIHdyaXR0ZW4gaW4gSlNPTiBhbmQgWUFNTCBmb3JtYXRzLgoKLi4gY29kZTo6IGJhc2gKCiAgICBwaXAgaW5zdGFsbCBvcGVuYXBpLXR5cGUKCgouLiBjb2RlOjogcHl0aG9uCgogICAgZnJvbSBvcGVuYXBpX3R5cGUgaW1wb3J0IE9wZW5BUEksIHBhcnNlX3NwZWMsIHNlcmlhbGl6ZV9zcGVjCgoKICAgIHNwZWM6IE9wZW5BUEkgPSBwYXJzZV9zcGVjKHsKICAgICAgICAieW91ciBPcGVuQVBJIFNwZWMgYXMgUHl0aG9uIGRpY3Rpb25hcnkiOiAid2lsbCBiZSBwYXJzZWQgaW50byBhIHByb3BlciBQeXRob24gdHlwZSIKICAgIH0pCiAgICBhc3NlcnQgcGFyc2Vfc3BlYyhzZXJpYWxpemVfc3BlYyhzcGVjKSkgPT0gc3BlYwoKLi4gY29kZTo6IGJhc2gKCiAgICAkIGN1cmwgLXMgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uIHwgb3BlbmFwaS10eXBlIGNoZWNrCiAgICBTdWNjZXNzZnVsbHkgcGFyc2VkLgoKCkNvZGVnZW4KLS0tLS0tLQoKSWYgeW91IGFyZSBsb29raW5nIGZvciBhIGNvbXBsZXRlIGNsaWVudCBjb2RlIGdlbmVyYXRvciwgY29uc2lkZXIgYG9wZW5hcGktY2xpZW50LWdlbmVyYXRvciA8aHR0cHM6Ly9naXRodWIuY29tL2F2YW5vdi9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3I+YF8KdGhhdCB1c2VzIHRoaXMgbGlicmFyeSB1bmRlciB0aGUgaG9vZC4KCkNsb25pbmcgdGhpcyByZXBvCi0tLS0tLS0tLS0tLS0tLS0tCgpUaGUgcHJvcGVyIHdheSB0byBjbG9uZSB0aGlzIHJlcG8gaXM6CgouLiBjb2RlLWJsb2NrOjogYmFzaAoKICAgIGdpdCBjbG9uZSAtLXJlY3Vyc2Utc3VibW9kdWxlcyA8cmVwby11cmw+IDxsb2NhbC1wcm9qZWN0LXJvb3Q+CiAgICBjZCA8bG9jYWwtcHJvamVjdC1yb290PgoKICAgICMgZm9yIHNob3dpbmcgc3VibW9kdWxlIHN0YXR1cyB3aXRoIGBnaXQgc3RhdHVzYAogICAgZ2l0IGNvbmZpZyBzdGF0dXMuc3VibW9kdWxlc3VtbWFyeSAxCgogICAgIyBmb3IgbG9nZ2luZyBzdWJtb2R1bGUgZGlmZiB3aXRoIGBnaXQgZGlmZmAKICAgIGdpdCBjb25maWcgZGlmZi5zdWJtb2R1bGUgbG9nCgoKRG9jdW1lbnRhdGlvbgotLS0tLS0tLS0tLS0tCgpEb2N1bWVudGF0aW9uIGlzIGhvc3RlZCBvbiBSZWFkVGhlRG9jczogaHR0cHM6Ly9vcGVuYXBpLXR5cGUucmVhZHRoZWRvY3MuaW8vZW4vZGV2ZWxvcC8KCgpUZXN0IGZyYW1ld29yawotLS0tLS0tLS0tLS0tLQoKVGhlIHByb2plY3QgdXNlcyBgTml4IDxodHRwczovL25peG9zLm9yZy8+YF8gZm9yIGJvb3RzdHJhcHBpbmcgaXRzIGRldiBlbnZpcm9ubWVudC4KCllvdSBjYW4gcnVuIGV4aXN0aW5nIHRlc3Qgc3VpdGUgd2l0aAoKLi4gY29kZTo6IGJhc2gKCiAgIG5peC1zaGVsbCAtLXJ1biAibWFrZSB0ZXN0IgoKCkNoYW5nZWxvZwotLS0tLS0tLS0KClNlZSBgQ0hBTkdFTE9HIDxodHRwczovL2dpdGh1Yi5jb20vYXZhbm92L29wZW5hcGktdHlwZS9ibG9iL21hc3Rlci9DSEFOR0VMT0cucnN0PmBfLgo= readmeEtag: '"0fb8f518fd543da507fe9cca940d4537689e1dc4"' readmeLastModified: Sun, 19 Jun 2022 15:47:24 GMT repositoryId: 324717113 description: >- OpenAPI represented as a Python type. Use it to parse specifications written in JSON and YAML formats. created: '2020-12-27T08:19:27Z' updated: '2024-01-08T03:17:19Z' language: Python archived: false stars: 11 watchers: 1 forks: 3 owner: avanov logo: https://avatars.githubusercontent.com/u/601955?v=4 license: MIT repoEtag: '"38e8cf348056a5c515f2d5bfe7de4ea7175c315ce3c95cf2ea1b141e09a20d1a"' repoLastModified: Mon, 08 Jan 2024 03:17:19 GMT foundInMaster: true category: - Description Validators - Server Implementations id: f7745a6b6f7d618239019b87dd2cd1b1 - source: openapi3 tags repository: https://github.com/asseco-voice/laravel-open-api v3: true repositoryMetadata: base64Readme: >- <p align="center"><a href="https://see.asseco.com" target="_blank"><img src="https://github.com/asseco-voice/art/blob/main/evil_logo.png" width="500"></a></p>

# Laravel OpenApi generator

This package provides painless OpenApi YML generation from existing routes. 

The idea is to have as little work to do as possible in order to generate the 
API documentation, so the package will try to assume a lot of things such as
models from controller names, request and response parameter based on actual
tables etc. 

For custom inputs/outputs, options will be provided.

## Installation

Install the package through composer. It is automatically registered
as a Laravel service provider.

``composer require asseco-voice/laravel-open-api``

## Usage

Running the command ``php artisan asseco:open-api`` will generate a new `.yml`
file at ``project_root/open-api.yml``.

What is covered out-of-the-box:

- read all routes
- inferring model name from controller name
- group (tag) them automatically
- get title and description from 
- generate request and response parameters

For cases not covered by this convention, refer to [overriding defaults section](#overriding-defaults).

For additional tweaking, refer to [config](#config).

Depending on number of routes, first run may take a few seconds as it is requesting a DB schema for
each model it can find. This is cached, so every subsequent run will run much faster. 

### Simple out-of-the-box example

Given the controller:

```
class UserController extends Controller
{
    /**
     * Store a newly created resource in storage.    
     *
     * Create new User object and store it in DB.
     */
    public function store(Request $request): JsonResponse
    {
        $user = User::query()->create($request->all());

        return response()->json($user);
    }

    ...
```

- command will infer ``User`` as being the main model for the controller
- title: ``Store a newly created resource in storage.`` 
- description: ``Create new User object and store it in DB.``
- request data: ``User`` model table attributes without `id`, `created_at`, `updated_at` attributes
- response data: complete ``User`` model table attributes

## Overriding defaults

Custom cases are handled through controller and method annotations in doc blocks:

```
/**
 * My controller doc block
 *
 * @annotation random annotation
 */
class MyController extends Controller
{
    /**
     * My method doc block
     *
     * @annotation another random annotation
     */
    public function index()
    {
        ...
    }

   ...
}
```

### Groups (tags)

When talking about 'groups', we are actually talking about OpenApi 'tags'. 

By default, command will take the controller name, remove ``Controller`` from it
and split `PascalCase` with spaces (i.e. `SysUserController` results in `Sys User` group name).

- ``@group`` within a controller doc block will override default group. 
- ``@group`` within a method doc block will override default group and controller group
making it an operator with the highest precedence.

It is possible to stack multiple group annotations.

### Models

Model is used to try to automatically generate inputs and outputs for standard Laravel
CRUD functions.

**Input (request):** model DB schema with either only fillable properties or except guarded.
Fillable properties have precedence, guarded will be ignored if fillable exists.

**Output (response):** complete model DB schema without hidden fields. 

It is completely valid to have no model associated. In this case, no automatic actions
will be performed which require an existing model class.

By default, model name is extracted from controller name. To change this behavior,
you have few options:

- Map a specific controller to a specific model. See [config](#config) for details.
- Include ``@model`` tag within a controller:
   - Specifying namespaced model will use that model ``@model My\Namespaced\Model`` 
   - Not specifying the namespace will use controller's namespace ``@model Model``
   
Controller tag has higher precedence over configuration mapping. If both exist, and
controller tag fails, configuration will try to fetch the model as well. Failing on
both fronts will result in model being ``null``.

It is possible to exclude part of the model for the request:

- ``@exclude attribute1 attribute2`` is a space separated list of specified model attributes
which will not be included in request data.  

### Path parameters

By default, path parameter(s) will be set as integer (assuming most of the path parameters are model IDs).
In case you want to use UUIDs, this can be changed in ``asseco-open-api`` config file by
setting ``service_uses_uuid`` to true.

Override them by including the following in the method doc block:

- ``@path`` will override what is fetched by default. You must provide it in the following
convention ``@path name type description`` where:
   - ``name`` - parameter name.
   - ``type`` - [OpenApi data type](https://swagger.io/docs/specification/data-models/data-types/).
   - ``description`` - text which will be set as parameter description (not required, 
   empty by default, so it can be omitted). 
   
Examples:

```
@path name type
@path name type Some description
```

It is not possible to set path parameter ``required`` property. It is automatically set to true because
OpenApi doesn't support optional path parameters (even though Laravel does).

### Operation ID

By default, ``operationId`` will be generated for each route, based on the controller/HTTP method
mapping and the name of the model. If you want to override the default, you can use ``@operationId``
to provide a different suffix to the operation ID (which will replace what would be an auto-generated model).

Prepended controller/HTTP method will stay intact because of possible conflicts when using update routes
which map to both PUT and PATCH verbs. 

Example:

```
// Provided for an show() method:
@operationId suffix
// this will output getSuffix

// Provided for an update() method:
@operationId suffix
// this will output putSuffix and patchSuffix
```

### Request/response parameters

By default, request/response parameter(s) will be [extracted from model](#models).

Override them by including ``@request`` or/and `@response` in the method doc block.

Example for ``@request``, working the same for `@response`:

- ``@request`` will override what is fetched by default. You must provide it in the following
convention ``@request name type required description`` where:
   - ``name`` - parameter name.
   - ``type`` - [OpenApi data type](https://swagger.io/docs/specification/data-models/data-types/).
   - ``required`` - boolean `true/false` value indicating whether the parameter is required 
   (if omitted, will be set to `true`)
   - ``description`` - text which will be set as parameter description (if omitted, will be set to
    empty string)
   
Examples:

```
@request name type
@request name type true
@request name type false Some description
```

For multiple parameters it is also possible to adopt a different convention:

```
@request
name type required description
name type required description
name type required description
```

In case you want an arbitrary string as a request/response, it can be achieved 
by using double quotes when setting request/response parameters. This way, all 
other parameters will be ignored and only the string inside double quotes will 
be returned.

Example:

```
@response "example"
```

You can include additional input/output models alongside original model with `@requestAppend key Class` or 
`@responseAppend key Class` in your method doc block. This will append given ``Class`` properties on already existing model using the `key` as a key.

I.e. having original model ``User`` (with properties name, email) you want to append `Post` model (with properties
title, description) to it as a list of inputs.

```
@model User // <-- not needed if it is UserController or you already specified model on the controller

@requestAppend posts Post
``` 

This will result in following request:

```
{
    "name": "string"
    "email": "string"
    "posts": {
        "title": "string"
        "description": "string"
    }
}
```

#### Response specific

Responses will by default be marked as multiple (indicating collection output, not a single model)
when looking at ``GET`` request without path parameters.

- including ``@multiple true/false`` in the method doc block will override those defaults 

If the variable type is ``array``, you can provide additional property within the parenthesis (be
sure not to leave blank space between type and parenthesis) to indicate
of what type are the array values: 

```
@response attribute array[string] true Some description
```

It is also possible to directly append a pivot table to the response, even if it 
has no model associated with it.

```
@pivot table_name
```

For example, if the table was appended to the ``User`` model, the following 
response will be returned:

```
{
    "name": "string"
    "email": "string"
    "pivot": {
      "user_id": 0
      "example_id": 0
    }
}
```

## Cache

Models database schema is being cached for performance (1d TTL), 
if you modify a migration be sure to run ``php artisan asseco:open-api --bust-cache``
which will force re-caching. 

## Config

Publish the configuration with 
``php artisan vendor:publish --provider="Asseco\OpenApi\OpenApiServiceProvider"``.

Configuration requires your minimal engagement, however there are some things which
package can't assume. 

- For models outside of ``App`` namespace, be sure to include full namespace
to ``namespaces`` config key as well so that package can automatically get the 
model attributes. 
- For controllers not named after their models (in ``ModelController`` format)
remap in ``controller_model_mapping`` config key.
 readmeEtag: '"81e6662caa368de507e8192c6c8cf7d3049f803e"' readmeLastModified: Tue, 08 Aug 2023 08:15:58 GMT repositoryId: 299838936 description: OpenAPI 3.0. generator for Laravel 8 created: '2020-09-30T07:16:22Z' updated: '2024-06-25T15:23:31Z' language: PHP archived: false stars: 11 watchers: 6 forks: 2 owner: asseco-voice logo: https://avatars.githubusercontent.com/u/66362883?v=4 license: MIT repoEtag: '"1efccb2f9c9ff2f4b9f28b4f2dfec7efdaf6eb120a14c8e365e3d41ba5cad775"' repoLastModified: Tue, 25 Jun 2024 15:23:31 GMT foundInMaster: true category: - Server Implementations - Parsers id: 5c0da8cc4801fde67fbfe19cf237d93a - source: openapi3 tags repository: https://github.com/dzoukr/openapiparser v3: true repositoryMetadata: base64Readme: >- PGltZyBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9Eem91a3IvT3BlbkFQSVR5cGVQcm92aWRlci9yYXcvbWFzdGVyL2xvZ28uanBnIiBhbHQ9ImRyYXdpbmciIHdpZHRoPSIxMDBweCIvPgoKIyBPcGVuIEFQSSBGIyBQYXJzZXIKClNpbXBsZSBsaWJyYXJ5IGZvciBwYXJzaW5nIFlBTUwvSlNPTiBPcGVuIEFQSSAocHJldmlvdXNseSBjYWxsZWQgYXMgU3dhZ2dlcikgc3BlY2lmaWNhdGlvbiAodmVyc2lvbiAzLjAuMSkuIE9yaWdpbmFsbHkgcGFydCBvZiBteSB0eXBlIHByb3ZpZGVyLCBidXQgSSB3aWxsIHByb2JhYmx5IG5ldmVyIGZpbmlzaCBpdCwgc28gYXQgbGVhc3QgY29tbXVuaXR5IGNhbiB1c2UgcGFydCBvZiBpdCBmb3Igb3duIHByb2plY3RzLiA6KQoKIyMgSW5zdGFsbGF0aW9uCkZpcnN0IGluc3RhbGwgTnVHZXQgcGFja2FnZQoKICAgIEluc3RhbGwtUGFja2FnZSBPcGVuQVBJUGFyc2VyCgpvciB1c2luZyBbUGFrZXRdKGh0dHA6Ly9mc3Byb2plY3RzLmdpdGh1Yi5pby9QYWtldC9nZXR0aW5nLXN0YXJ0ZWQuaHRtbCkKCiAgICBudWdldCBPcGVuQVBJUGFyc2VyCgojIyBIb3cgdG8gdXNlCgpUeXBpY2FsbHksIHlvdSB3b3VsZCB1c2Ugc29tZSBleGlzdGluZyBzcGVjaWZpY2F0aW9uIChzZWUgW29mZmljaWFsIGV4YW1wbGVzIGZvciB2ZXJzaW9uIDNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL3RyZWUvbWFzdGVyL2V4YW1wbGVzL3YzLjApKToKCmBgYGZzaGFycApvcGVuIE9wZW5BUElQYXJzZXIuVmVyc2lvbjMuUGFyc2VyCmxldCBvcGVuQVBJID0gRG9jdW1lbnQubG9hZEZyb21ZYW1sRmlsZSAibXlTcGVjLnlhbWwiIApgYGAKClRoZW4geW91IGhhdmUgYmFzaWMgRiMgcmVjb3JkIGZpbGxlZAoKYGBgZnNoYXJwCmxldCB2ZXJzaW9uID0gb3BlbkFQSS5JbmZvLlZlcnNpb24KbGV0IHRpdGxlID0gb3BlbkFQSS5JbmZvLlRpdGxlCmxldCBwYXRocyA9IG9wZW5BUEkuUGF0aHMKLi4uCmBgYAoKUGxlYXNlIGNoZWNrIFt0ZXN0IHByb2plY3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9Eem91a3IvT3BlbkFQSVBhcnNlci90cmVlL21hc3Rlci90ZXN0cy9PcGVuQVBJUGFyc2VyLlRlc3RzKSBmb3IgbW9yZSBleGFtcGxlcy4KCiMjIExpbWl0YXRpb25zCgpOb3QgYWxsIHByb3BlcnRpZXMgZnJvbSBbMy4wLjEgc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjEubWQpIGFyZSBpbXBsZW1lbnRlZC4gRXNwZWNpYWxseSBgYW55T2ZgIGFuZCBgb25lT2ZgIFNjaGVtYSBvYmplY3RzLCB3aGljaCB3aWxsIGJlIHByb2JhYmx5IGltcGxlbWVudGVkIGluIGZ1dHVyZSB2ZXJzaW9ucy4gUGxlYXNlIGNoZWNrIFtTcGVjaWZpY2F0aW9uIHJlY29yZCBkZWZpbml0aW9uXShodHRwczovL2dpdGh1Yi5jb20vRHpvdWtyL09wZW5BUElQYXJzZXIvYmxvYi9tYXN0ZXIvc3JjL09wZW5BUElQYXJzZXIvVmVyc2lvbjMvU3BlY2lmaWNhdGlvbi5mcykgZm9yIGFscmVhZHkgaW1wbGVtZW50ZWQgcHJvcGVydGllcy4KCiMjIENvbnRyaWJ1dGlvbgoKWW91IGtub3cgdGhlIGRyaWxsLiBDb2RlICsgVGVzdHMgPSBHb29kIFBSLiBBbnkgY29udHJpYnV0aW9uIG1vcmUgdGhhbiB3ZWxjb21lIQ== readmeEtag: '"55d043ec4becc96e42b52774ffe9db94053d91fb"' readmeLastModified: Tue, 16 Oct 2018 12:57:05 GMT repositoryId: 136714502 description: Simple Open API F# Parser created: '2018-06-09T10:43:53Z' updated: '2020-11-12T16:00:37Z' language: F# archived: false stars: 11 watchers: 4 forks: 2 owner: Dzoukr logo: https://avatars.githubusercontent.com/u/851307?v=4 license: MIT repoEtag: '"e659e8372c291e4acfb2d598492a818065f943ed37c11a0cfd4dbbb20cb9e9af"' repoLastModified: Thu, 12 Nov 2020 16:00:37 GMT foundInMaster: true category: Parsers id: bb5f2819cf8a06099262d0b9aac90a9b - source: openapi3 tags repository: https://github.com/friskes/drf-spectacular-websocket v3: true id: 1a1dd69dc7629c186e776da3f959fd4d repositoryMetadata: base64Readme: >- # Extend websocket schema decorator for Django Channels

<div align="center">

| Project   |     | Status                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                        |
|-----------|:----|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| CI/CD     |     | [![Latest Release](https://github.com/Friskes/drf-spectacular-websocket/actions/workflows/publish-to-pypi.yml/badge.svg)](https://github.com/Friskes/drf-spectacular-websocket/actions/workflows/publish-to-pypi.yml)                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             |
| Quality   |     | [![Coverage](https://codecov.io/github/Friskes/drf-spectacular-websocket/graph/badge.svg?token=vKez4Pycrc)](https://codecov.io/github/Friskes/drf-spectacular-websocket)                                                                                                                                                                                                                                                                                                                               |
| Package   |     | [![PyPI - Version](https://img.shields.io/pypi/v/drf-spectacular-websocket?labelColor=202235&color=edb641&logo=python&logoColor=edb641)](https://badge.fury.io/py/drf-spectacular-websocket) ![PyPI - Support Python Versions](https://img.shields.io/pypi/pyversions/drf-spectacular-websocket?labelColor=202235&color=edb641&logo=python&logoColor=edb641) ![Project PyPI - Downloads](https://img.shields.io/pypi/dm/drf-spectacular-websocket?logo=python&label=downloads&labelColor=202235&color=edb641&logoColor=edb641)                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                  |
| Meta      |     | [![types - Mypy](https://img.shields.io/badge/types-Mypy-202235.svg?logo=python&labelColor=202235&color=edb641&logoColor=edb641)](https://github.com/python/mypy) [![License - MIT](https://img.shields.io/badge/license-MIT-202235.svg?logo=python&labelColor=202235&color=edb641&logoColor=edb641)](https://spdx.org/licenses/) [![code style - Ruff](https://img.shields.io/endpoint?url=https://raw.githubusercontent.com/astral-sh/ruff/main/assets/badge/format.json&labelColor=202235)](https://github.com/astral-sh/ruff) |

</div>

> Provides [extend_ws_schema](#About-decorator) decorator to documentation [Cunsumer](https://channels.readthedocs.io/en/latest/topics/consumers.html) methods from [channels](https://github.com/django/channels) just like it does [drf-spectacular](https://github.com/tfranzel/drf-spectacular)


## Install
1. Install package
    ```bash
    pip install drf-spectacular-websocket
    ```

2. Create sidecar static
    ```bash
    python manage.py collectstatic
    ```

3. Add app name to `INSTALLED_APPS`
    > `drf_spectacular_websocket` must be higher than `drf_spectacular`
    ```python
    INSTALLED_APPS = [
        'drf_spectacular_websocket',
        'drf_spectacular',
        'drf_spectacular_sidecar',
    ]
    ```


## Configure settings
```python
ASGI_APPLICATION = 'path.to.your.application'

# (Optional) this is default settings are automatically set by the drf_spectacular_websocket.
# You can override them in your application if necessary.
SPECTACULAR_SETTINGS = {
    'DEFAULT_GENERATOR_CLASS': 'drf_spectacular_websocket.schemas.WsSchemaGenerator',
    'SWAGGER_UI_DIST': 'SIDECAR',
    'SWAGGER_UI_FAVICON_HREF': 'SIDECAR',
    'REDOC_DIST': 'SIDECAR',
    'SWAGGER_UI_SETTINGS': {
        'connectSocket': True,  # Automatically establish a WS connection when opening swagger
        'socketMaxMessages': 8,  # Max number of messages displayed in the log window in swagger
        'socketMessagesInitialOpened': False,  # Automatically open the log window when opening swagger
    },
}
```

> drf_spectacular_websocket automatically finds websocket urls and related consumer using `ASGI_APPLICATION` setting.

## About decorator
`extend_ws_schema` decorator accepts one new `type` parameter relative to drf_spectacular `extend_schema`.
- possible values:
    - `send` - Type of interaction, [request -> response]
    - `receive` - Type of interaction, [response without request]

## Usage example

```python
from channels.generic.websocket import AsyncJsonWebsocketConsumer
from rest_framework.serializers import Serializer, CharField
from drf_spectacular_websocket.decorators import extend_ws_schema


class SomeMethodInputSerializer(Serializer):
    some_field = CharField()


class SomeMethodOutputSerializer(Serializer):
    some_field = CharField()


class SendMessageOutputSerializer(Serializer):
    some_field = CharField()


class SomeConsumer(AsyncJsonWebsocketConsumer):

    async def receive_json(self, content, **kwargs):
        some_value = content.get('some_key')
        if some_value:
            await self.some_method(some_value)

    @extend_ws_schema(
        type='send',
        summary='some_method_summary',
        description='some_method_description',
        request=SomeMethodInputSerializer,
        responses=SomeMethodOutputSerializer,
    )
    async def some_method(self, some_value):
        input_serializer = SomeMethodInputSerializer(data=some_value)
        input_serializer.is_valid(raise_exception=True)

        return_value = await some_business_logic(input_serializer.validated_data)

        output_serializer = SomeMethodOutputSerializer(data=return_value)
        output_serializer.is_valid(raise_exception=True)

        await self.send_json(output_serializer.validated_data)

    @extend_ws_schema(
        type='receive',
        summary='send_message_summary',
        description='send_message_description',
        request=None,
        responses=SendMessageOutputSerializer,
    )
    async def send_message(self, message):
        await self.send_json(message)
```

## Contributing
We would love you to contribute to `drf-spectacular-websocket`, pull requests are very welcome! Please see [CONTRIBUTING.md](https://github.com/Friskes/drf-spectacular-websocket/blob/main/CONTRIBUTING.md) for more information.

## Swagger Examples

### Send
![](https://raw.githubusercontent.com/Friskes/drf-spectacular-websocket/main/images/example_send.png)

### Receive
![](https://raw.githubusercontent.com/Friskes/drf-spectacular-websocket/main/images/example_receive.png)
 readmeEtag: '"10ac7c491ecba05706d0f84baf56dfc2d4e3a4bb"' readmeLastModified: Tue, 25 Jun 2024 20:05:21 GMT repositoryId: 789855166 description: Extend websocket schema decorator for Django Channels created: '2024-04-21T18:27:01Z' updated: '2026-02-02T07:01:11Z' language: Python archived: false stars: 16 watchers: 1 forks: 2 owner: Friskes logo: https://avatars.githubusercontent.com/u/78731609?v=4 license: MIT repoEtag: '"16d71d9e49f780fb61c3a83e7db539979ccd8ca056b905cca5a359a51f3ccee3"' repoLastModified: Mon, 02 Feb 2026 07:01:11 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/rahul-ghadge/spring-boot-cassandra-crud v3: true repositoryMetadata: base64Readme: >- # spring-boot-cassandra-crud
Spring boot CRUD (Create, Read, Update, Delete) demo application with cassandra DB - 
In this application, we have implemented CRUD (Create, Read, Update, Delete) operations using spring data and cassandra DB.  


## Prerequisites 
- Java
- [Spring Boot](https://spring.io/projects/spring-boot)
- [Maven](https://maven.apache.org/guides/index.html)
- [Cassandra](https://cassandra.apache.org/)


## Tools
- Eclipse or IntelliJ IDEA (or any preferred IDE) with embedded Gradle
- Maven (version >= 3.6.0)
- Postman (or any RESTful API testing tool)
- cqlsh (cassandra query language shell) - for monitoring stored data



###  Build and Run application
_GOTO >_ **~/absolute-path-to-directory/spring-boot-cassandra-crud**  
and try below command in terminal
> **```mvn spring-boot:run```** it will run application as spring boot application

or
> **```mvn clean install```** it will build application and create **jar** file under target directory 

Run jar file from below path with given command
> **```java -jar ~/path-to-spring-boot-cassandra-crud/target/spring-boot-cassandra-crud-0.0.1-SNAPSHOT.jar```**

Or
> run main method from `SpringBootCassandraCrudApplication.java` as spring boot application.  


||
|  ---------    |
| **_Note_** : In `SpringBootCassandraCrudApplication.java` class we have autowired SuperHero repository. <br/>If there is no record present in DB for SuperHero model class, static data is getting inserted in DB from `HelperUtil.java` class when we are starting the app for the first time.| 


---
### For API document using OpenAPI UI 

> **http://localhost:8080/swagger-ui-custom.html**

![Swagger Documentation](https://github.com/rahul-ghadge/spring-boot-cassandra-crud/blob/master/src/main/resources/static/spring-boot-cassandra-crud-Swagger.PNG?raw=true "Spring Data Cassandra Swagger")

---

### Install JDK8
Step 1: Download JDK8 from [JDK site](https://www.oracle.com/java/technologies/javase/javase-jdk8-downloads.html).  
Step 2: Install downloaded an executable file.  
Step 3: Add JDK8 path as environment variable.  



### Setup cqlsh (cassandra query language shell) - for monitoring stored data
Step 1: **Python2.7** is mandatory for cqlsh to handle user requests. Download Python2.7 from [Python site](https://www.python.org/downloads/release/python-2718/).  
Step 2: Install downloaded an executable file.  
Step 3: Add Python2.7 path as environment variable.  




### Setup Cassandra
Step 1: Download the latest version of apache-cassandra-x.xx.x from [Cassandra site](https://cassandra.apache.org/download/).  
Step 2: Unzip the compressed zip file using a compression tool to any location. Ex. c:\apache-cassandra-x.xx.x  
Step 3: Add c:\apache-cassandra-x.xx.x\bin path as environment variable.  


---

#### Start the Cassandra and cqlsh 

##### Start Cassandra
Make sure bin path is set for cassandra in environment variable.  
> `cassandra`

If no error on the console means cassandra is started and running.


##### Start cqlsh 
Make sure path is set for the python in environment variable.  
> `cqlsh`

If no error on the console means **cqlsh** is connected.


### Code Snippets
1. #### Maven Dependencies
    Need to add below dependency to enable cassandra in **pom.xml**.    
    ```
	<dependency>
	    <groupId>org.springframework.boot</groupId>
		<artifactId>spring-boot-starter-data-cassandra</artifactId>
	</dependency>
   		
   <!-- For Boilerplate code (Getters/Setters/Constructors) -->
   <dependency>
   		<groupId>org.projectlombok</groupId>
   		<artifactId>lombok</artifactId>
   		<optional>true</optional>
   </dependency>
    ```
    
    For API documentation using swagger and OpenApi UI add below dependency.
	```
	<dependency>
	    <groupId>org.springdoc</groupId>
		<artifactId>springdoc-openapi-ui</artifactId>
		<version>1.4.4</version>
	</dependency>
	```
   
2. #### Properties file
     Placed properties in **application.yml** file related to cassandra which we are reading in **CassandraConfig.java** class
     and configuring cassandra connection for Cassandra.  
     API documentation related swagger UI path is also placed here which will enable Swagger API Doc on same path.  
     **src/main/resources/application.yml**     
     ```
     spring:
       data:
         cassandra:
           contact-points: localhost
           port: 9042
           keyspace-name: simple_crud
           #username: cassandra
           #password: cassandra
           #schema-act: create_if_not_exists
     
     springdoc:
       version: 1.0.0
       swagger-ui:
         path: /swagger-ui-custom.html
     ```


3. #### Model class
    Below are the model classes which we will store in cassandra and perform CRUD operations.  
    **com.arya.cassandra.model.SuperHero.java**  
    **com.arya.cassandra.model.SuperPowers.java**  
    ```
    @Data
    @Builder
    @Table("super_hero")
    public class SuperHero implements Serializable {    
        @PrimaryKey
        private Long id;    
        private String name;    
        @Column("super_name")
        private String superName;    
        private String profession;    
        private int age;    
        @Column("super_powers")
        private SuperPowers superPowers;    
    }
   
   
   @Data
   @Builder
   @UserDefinedType("super_powers")
   public class SuperPowers implements Serializable {   
       private String strength;   
       private String durability;   
       private boolean canFly;
   }
    ```


4. #### Cassandra Configuration
   This is the most important class in this application, where all cassandra related configuration is placed 
   and using this class we are connecting to cassandra and creating **KEYSPACE** and **TABLES** also while starting the application.

   ```
   import org.springframework.beans.factory.annotation.Value;
   import org.springframework.context.annotation.Configuration;
   import org.springframework.data.cassandra.config.AbstractCassandraConfiguration;
   import org.springframework.data.cassandra.config.SchemaAction;
   import org.springframework.data.cassandra.core.cql.keyspace.CreateKeyspaceSpecification;
   import org.springframework.data.cassandra.core.cql.keyspace.DropKeyspaceSpecification;
   import org.springframework.data.cassandra.core.cql.keyspace.KeyspaceOption;
   
   import java.util.Collections;
   import java.util.List;
   
   @Configuration
   public class CassandraConfig extends AbstractCassandraConfiguration {
   
       @Value("${spring.data.cassandra.keyspace-name: simple_crud}")
       private String KEYSPACE;
   
       @Value("${spring.data.cassandra.contact-points: localhost}")
       private String CONTACT_POINT;
   
       @Value("${spring.data.cassandra.port: 9042}")
       private int PORT;
   
   
       @Override
       public String getContactPoints() {
           return CONTACT_POINT;
       }
   
       @Override
       protected int getPort() {
           return PORT;
       }
   
       @Override
       public SchemaAction getSchemaAction() {
           return SchemaAction.CREATE_IF_NOT_EXISTS;
       }
   
       @Override
       protected List<CreateKeyspaceSpecification> getKeyspaceCreations() {
           return Collections.singletonList(CreateKeyspaceSpecification.createKeyspace(KEYSPACE)
                   .ifNotExists()
                   .with(KeyspaceOption.DURABLE_WRITES, true)
                   .withSimpleReplication(3L));
       }
   
       @Override
       protected String getLocalDataCenter() {
           return "datacenter1";
       }
   
       //@Override
       //protected List<DropKeyspaceSpecification> getKeyspaceDrops() {
       //    return Collections.singletonList(DropKeyspaceSpecification.dropKeyspace(KEYSPACE));
       //}
   
       @Override
       protected String getKeyspaceName() {
           return KEYSPACE;
       }
      
       @Override
       public String[] getEntityBasePackages() {
           return new String[] {"com.arya.cassandra.model"};
       }   
   }
   ```
   
   
   
5. #### CRUD operation for Super Heroes

    In **com.arya.cassandra.controller.SuperHeroController.java** class, 
    we have exposed 5 endpoints for basic CRUD operations
    - GET All Super Heroes
    - GET by ID
    - POST to store Super Hero in DB
    - PUT to update Super Hero
    - DELETE by ID
    
    ```
    @RestController
    @RequestMapping("/super-heroes")
    public class SuperHeroController {
        
        @GetMapping("/save")
        public ResponseEntity<List<SuperHero>> save();
   
        @GetMapping
        public ResponseEntity<List<SuperHero>> findAll();
    
        @GetMapping("/{id}")
        public ResponseEntity<SuperHero> findById(@PathVariable String id);
    
        @PostMapping
        public ResponseEntity<SuperHero> save(@RequestBody SuperHero superHero);
    
        @PutMapping
        public ResponseEntity<SuperHero> update(@RequestBody SuperHero superHero);
    
        @DeleteMapping("/{id}")
        public ResponseEntity<SuperHero> delete(@PathVariable String id);
    }
    ```
   
    In **com.arya.cassandra.repository.SuperHeroRepository.java**, we are extending `CassandraRepository<Class, ID>` interface which enables CRUD related methods.
    ```
    @Repository
    public interface SuperHeroRepository extends CassandraRepository<SuperHero, Long> {
    }
    ```
   
   In **com.arya.cassandra.service.impl.SuperHeroServiceImpl.java**, we are autowiring above interface using `@Autowired` annotation and doing CRUD operation.


6. #### Query operation for SuperHero
   In **com.arya.cassandra.controller.SuperHeroQueryController.java** class Cassandra queries API Endpoints are placed.
   we are autowiring SuperHeroQueryService interface using `@Autowired` annotation and reaching to Service layer.  
   In **com.arya.cassandra.service.impl.SuperHeroQueryServiceImpl.java**, 
   we are autowiring SuperHeroQueryRepository interface using `@Autowired` annotation and reaching to DAO layer.
   In **com.arya.cassandra.repository.impl.SuperHeroQueryRepositoryImpl.java**, 
   we are autowiring `CassandraOperations` interface which enables CRUD related methods.
   ```
   @Autowired
   private CassandraOperations cassandraTemplate;
   ``` 
    


<br/>

### API Endpoints

- #### Super Hero CRUD Operations
    > **GET Mapping** http://localhost:8080/super-heroes  - Get all Super Heroes
    
    > **GET Mapping** http://localhost:8080/super-heroes/1  - Get Super Hero by ID
       
    > **POST Mapping** http://localhost:8080/super-heroes  - Add new Super Hero in DB  
    
     Request Body  
     ```
     {
         "id": 1,
         "name": "Tony",
         "superName": "Iron Man",
         "profession": "Business",
         "age": 50,            
         "superPowers": {
             "strength": "Suit",
             "durability": "Month",
             "canFly": true
         }
     }
     ```
    
    > **PUT Mapping** http://localhost:8080/super-heroes  - Update existing Super Hero for given ID 
    
     Request Body    
     ```
     {
         "id": 1,
         "name": "Tony",
         "superName": "Iron Man",
         "profession": "Business",
         "age": 50,         
         "superPowers": {
             "strength": "Only if he is in a suit",
             "durability": "Month",
             "canFly": true
         }
     }
     ```
    
    > **DELETE Mapping** http://localhost:8080/super-heroes/1  - Delete Super Hero by ID



### Output

![Alt text](https://github.com/rahul-ghadge/spring-boot-cassandra-crud/blob/master/src/main/resources/static/spring-data-cassandra-output.PNG?raw=true "Spring Data Cassandra output")
 readmeEtag: '"093c47bc7c2169b6144bafdd6fe939dd42f133d5"' readmeLastModified: Thu, 03 Sep 2020 15:30:51 GMT repositoryId: 288662036 description: >- Spring boot CRUD (Create, Read, Update, Delete) demo application with cassandra DB and API documentation using Swaager and OpenAPI UI created: '2020-08-19T07:18:27Z' updated: '2024-06-11T00:46:01Z' language: Java archived: false stars: 11 watchers: 1 forks: 18 owner: rahul-ghadge logo: https://avatars.githubusercontent.com/u/54259667?v=4 repoEtag: '"ed63dded7107d979326a286d2b4649778ab6d49b664a46217fbdc2bbe496452e"' repoLastModified: Tue, 11 Jun 2024 00:46:01 GMT foundInMaster: true category: Low-level Tooling id: 09cb31f38ecae6d0bbbeeddd376b3966 - source: openapi3 tags repository: https://github.com/privacyengineering/tira v3: true repositoryMetadata: base64Readme: >- # TIRA: An OpenAPI Extension and Toolbox for GDPR Transparency in RESTful Architectures


<p float="left" align="center">
  <img src="/docs/img/application_final.png" width="66%" /> 
</p>


Transparency – the provision of information about what personal data is collected for which purposes, how long it is stored, or to which parties it is transferred – is one of the core privacy principles underlying regulations such as the GDPR. Technical approaches for implementing transparency in practice are, however, only rarely considered. In this paper, we present a novel approach for doing so in current, RESTful application architectures and in line with prevailing agile and DevOps-driven practices. For this purpose, we introduce 1) a transparency-focused extension of OpenAPI specifications that allows individual service descriptions to be enriched with transparency-related annotations in a bottom-up fashion and 2) a set of higher-order tools for aggregating respective information across multiple, interdependent services and for coherently integrating our approach into automated CI/CD-pipelines. Together, these building blocks pave the way for providing transparency information that is more specific and at the same time better reflects the actual implementation givens within complex service architectures than current, overly broad privacy statements.


## 📚 Read our paper via [Github](https://github.com/PrivacyEngineering/tira/raw/main/Transparency_in_RESTful_Architectures_IWPE21_preprint.pdf), [arXiv](https://arxiv.org/abs/2106.06001) or [IEEEXplore](https://ieeexplore.ieee.org/document/9583685).


```
@inproceedings{gruenewald2021tira,
  title = {TIRA: An OpenAPI Extension and Toolbox for GDPR Transparency in RESTful Architectures},
  author = {Elias Grünewald and Paul Wille and Frank Pallas and Maria C. Borges and Max-R. Ulbricht},
  booktitle={2021 IEEE European Symposium on Security and Privacy Workshops (EuroS\&PW)},
  publisher = {IEEE Computer Society},
  doi = {10.1109/EuroSPW54576.2021.00039},
  year = {2021}
}
```



Please get in touch with us via [https://tu.berlin/ise/eg](https://tu.berlin/ise/eg). 

## Personal data indicators
To learn more about where personal data indicators in OpenAPI specifications may reside, see **[here](docs/PD_INDICATORS.md)**.

## Vocabulary
To learn more about our vocabulary used for the proposed OpenAPI extension, see **[here](docs/VOCABULARY.md)**.

<p float="left" align="center">
  <img src="/docs/img/x-tira-pd.png" width="45%" />
  <img src="/docs/img/thub_stepcount_schema.png" width="45%" />
</p>


## Installation

Make sure Ruby is installed.

```bash
ruby --version
```

TIRA was built using Ruby version `2.6.3`, other/newer versions should work fine, but were not tested against.
Use [rvm](https://rvm.io/) or `rbenv install --verbose 2.6.3`.


Install bundler

```bash
gem install bundler
```

Clone the repo

```bash
git clone https://github.com/PrivacyEngineering/tira.git
cd tira/
```

Install all gems via bundler

```bash
bundle install
```

Configure secrets and credentils 

```bash
bin/rails credentials:edit
```

Rails tries to open the crendentials with `$EDITOR`.
You can define an editor by setting the `EDITOR` variable explicitely, e.g.

```bash
EDITOR="nano" rails credentials:edit
```


This will create an encrypted config file and a master key, for details visit [this guide](https://edgeguides.rubyonrails.org/security.html#custom-credentials).
The configuration format used can be found in the sample configuration file in `config/credentials_example.yml`.

Database name and credentials need to be configured.
If a different database adapter than postgres is used, this must be configured in:

```
config/database.yml
```


Set up a postgres database (if you chose to not use postgres, set up a database according to your configuration).

You can use the offical docker image

```bash
docker run --name some-postgres -e POSTGRES_PASSWORD=mysecretpassword -d postgres
```

or set up a postgres database locally

```psql
create database $db_name;

create role $user_name with createdb login password 'password';

grant all privileges on database $db_name to $user_name;

```

Now run the migrations to set up the database

```bash
bin/rails db:migrate RAILS_ENV=development
```


TransparencyHub is now set up and you can start the application

```bash
rails s
```

The app is now accessible via `http://localhost:3000`


## Example of *x-tira* in an OpenAPI document
We describe a `ToothbrushEvent` that can be shared with other utilizers via the example service from our paper. 

```yaml
openapi: "3.0.0"
x-tira: 
  utilizer:
    - name: "AWS"
      non_eu_country: true
      country: "UK"
info:
  version: "1.0.2"
  description: "This service can share health data with other health data services and insurances"
  title: "Health Data Sharing Service"
servers: 
  - url: "https://health.domain.tld"
paths:
  "/{user_id}/toothbrush/share":
    parameters:
      - name: user_id
        in: "path"
        required: true
        description: "User ID of a Health Data Hub User"
        schema:
            type : integer
    get: 
      description: Get an array of Toothbrush Events for a given time interval.
      parameters:
        - name: startday
          in: "query"
          schema:
            type : string
            format: date-time
          required: true
          description: "Start date of requested interval."
        - name: endday
          in: "query"
          schema:
            type : string
            format: date-time
          description: "End date of requested interval."
      responses:
        200:
          description: "Request successful."
          content: 
            application/json:
              schema:
                $ref: '#/components/schemas/ToothbrushEvent'
components:
  schemas:
    ToothbrushEvent:
      type: "object"
      required:
        - user_id
      properties:
        seconds:
          type: "integer"
        datetime:
          type: "string"
        user_id:
          type: "integer"
      x-tira: 
        retention-time:
          volatile: true
        special_category:
          category: "Health Data"
        purposes:
          yappl:
            '{
               "id":123,
               "preference":[
                  {
                     "rule":{
                        "purpose":{
                           "permitted": [ "FitnessData Sharing", "Health Insurance Bonus Program" ],
                           "excluded": [ ... ]
                        },
                        "utilizer":{
                           "permitted": [ ... ],
                           "excluded": [ ... ]
                        },
                        "transformation": [ ... ],
                        "valid_from":"2021-06-09T00:00:00.000Z",
                        "exp_date":"0000-01-01T00:00:00.000Z"
                     }
                  }
               ]
            }'
        profiling:
          reason: "Health profile based on series of health related behaviour."
        utilizer:
          - name: "MyFitnessPal"
            non_eu_country: false
          - name: "Strava"
            non_eu_country: true
            country: "USA"
        utilizer_category:
          - name: "Health Insurance Company"
            country: "Germany"
            non_eu_country: false
            type: "Insurance Company"
            sector: "Insurance"
            sub_sector: 
              - "Health Insurance"
              - "Health Tax"
```
 readmeEtag: '"125ae8b28d26a405ac0308cd0506292fc2c53603"' readmeLastModified: Mon, 18 Jul 2022 20:28:40 GMT repositoryId: 364238719 description: >- TIRA: An OpenAPI Extension and Toolbox for GDPR Transparency in RESTful Architectures created: '2021-05-04T11:50:35Z' updated: '2024-09-17T17:36:37Z' language: Ruby archived: false stars: 11 watchers: 3 forks: 2 owner: PrivacyEngineering logo: https://avatars.githubusercontent.com/u/38961961?v=4 license: MIT repoEtag: '"ee32144269a45ec12ddd5ac1544e47d22176c375fed0bbefc722378e0e565ece"' repoLastModified: Tue, 17 Sep 2024 17:36:37 GMT foundInMaster: true category: - Server - Server Implementations id: 9ea99a70abf57321f866ed8393822337 - source: openapi3 tags repository: https://github.com/openmobilityfoundation/mds-openapi v3: true id: 56fb933212775b9d92e503c946ed7a88 repositoryMetadata: base64Readme: >- IyBNb2JpbGl0eSBEYXRhIFNwZWNpZmljYXRpb24gT3BlbkFQSQoKVGhlIFtNb2JpbGl0eSBEYXRhIFNwZWNpZmljYXRpb24gKE1EUyldKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVubW9iaWxpdHlmb3VuZGF0aW9uL21vYmlsaXR5LWRhdGEtc3BlY2lmaWNhdGlvbiksIGEgcHJvamVjdCBvZiB0aGUgT3BlbiBNb2JpbGl0eSBGb3VuZGF0aW9uIChPTUYpLCBpcyBhIGRhdGEgc3RhbmRhcmQgdG8gZW5hYmxlIHJpZ2h0LW9mLXdheSByZWd1bGF0aW9uIGFuZCB0d28td2F5IGNvbW11bmljYXRpb24gYmV0d2VlbiBtb2JpbGl0eSBjb21wYW5pZXMgYW5kIGxvY2FsIGdvdmVybm1lbnRzLgoKVGhpcyBpcyB0aGUgT3BlbkFQSSBkZXNjcmlwdGlvbiBmb3IgTURTIGRhdGEgZmVlZHMsIG1hbmFnZWQgYnkgdGhlIFtPcGVuIE1vYmlsaXR5IEZvdW5kYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVubW9iaWxpdHlmb3VuZGF0aW9uKS4KCk9ubGluZSBkb2N1bWVudGF0aW9uIGlzIGF2YWlsYWJsZSBvbiBbU3RvcGxpZ2h0XShodHRwczovL29wZW5tb2JpbGl0eWZuZC5zdG9wbGlnaHQuaW8vZG9jcy9tZHMtb3BlbmFwaSkuCgpNRFMgdmVyc2lvbnMgYXJlIG9yZ2FuaXplZCBieSBicmFuY2hlcyBzdGFydGluZyB3aXRoIGEgYnJhbmNoIGZvciBgdjIuMGAuCgojIyBQeXRob24gcHJvamVjdAoKQSBzbWFsbCBQeXRob24gcHJvamVjdCBkZWZpbmVkIGluIFtgcHlwcm9qZWN0LnRvbWxgXSguL3B5cHJvamVjdC50b21sKSBzdXBwb3J0cyB0aGUgc2NoZW1hIGRldmVsb3BtZW50IHByb2Nlc3MuCgpgYGBiYXNoCiMgaW5zdGFsbCB0aGUgcHJvamVjdCBhbmQgaXRzIGRlcGVuZGVuY2llcwpwaXAgaW5zdGFsbCAtZSAuCmBgYAoKIyMjIFNjaGVtYSB2YWxpZGF0aW9uCgpUZXN0cyB3cml0dGVuIGluIFtgcHl0ZXN0YF0oaHR0cHM6Ly9weXRlc3Qub3JnKSBjaGVjayB2YXJpb3VzIGNvbXBvbmVudHMgb2YgdGhlIHNjaGVtYXMuIFRoZXNlIHRlc3RzIGNhbiBiZSBydW4gbG9jYWxseSwgYW5kCmFsc28gcnVuIGluIEdpdEh1YiBBY3Rpb25zIG9uIGNvbW1pdHMgdG8gdGhpcyByZXBvc2l0b3J5LgoKYGBgYmFzaAojIHJ1biB0aGUgdGVzdHMgd2l0aCBweXRlc3QgZnJvbSB0aGUgcm9vdCBvZiB0aGUgcmVwbwpweXRlc3QKYGBgCgojIyMgTm90ZWJvb2tzCgpbSnVweXRlciBOb3RlYm9va3NdKGh0dHBzOi8vanVweXRlci5vcmcvKSB0byBoZWxwIHdpdGggb25lLXRpbWUgZGF0YSBjbGVhbnVwcyBhbmQgaXNzdWUgY2hlY2tpbmcuCgpgYGBiYXNoCiMgaW5zdGFsbCB0aGUgJ25vdGVib29rcycgZXh0cmFzCnBpcCBpbnN0YWxsIC1lIC5bbm90ZWJvb2tzXQpgYGAKCiMjIExpY2Vuc2UKClRoZSBNRFMgT3BlbkFQSSwgbGlrZSBNRFMgaXRzZWxmLCBpcyBsaWNlbnNlZCB1bmRlciBbQ3JlYXRpdmUgQ29tbW9ucyBBdHRyaWJ1dGlvbiA0LjAgSW50ZXJuYXRpb25hbCBQdWJsaWMgTGljZW5zZV0oLi4vTElDRU5TRSkK readmeEtag: '"73083afd08f874b78c4740e3fb1de2e4ee7ce1bc"' readmeLastModified: Tue, 23 May 2023 13:07:06 GMT repositoryId: 597157509 description: >- OpenAPI description for MDS data feeds, managed by the Open Mobility Foundation. created: '2023-02-03T18:55:31Z' updated: '2025-07-20T19:19:54Z' language: Python archived: false stars: 14 watchers: 2 forks: 6 owner: openmobilityfoundation logo: https://avatars.githubusercontent.com/u/52187191?v=4 license: NOASSERTION repoEtag: '"0260c4fe66ac1f768517fc1ad6a8d1ee754ed026cb74bade39394c41e1a431fa"' repoLastModified: Sun, 20 Jul 2025 19:19:54 GMT category: - Parsers - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/karatelabs/intellij-plugin v3: true id: 35c019960e3f4a93899b4502d156c101 repositoryMetadata: base64Readme: >- <table>
    <tr>
        <td>
            <br/><a href="https://karatelabs.io"><img src="resources/karate-logo.svg" height="45px"/></a>
        </td>
        <td>
            <h2>IntelliJ Plugin</h2>           
        </td>
        <th>
            <h3><a href="https://youtu.be/hkHQkrlH9xQ?si=c14DJfI6Y_K9Q414">:tv: <br/>&nbsp;&nbsp;Video&nbsp;&nbsp;</a></h3>
        </th>        
        <th>
            <h3><a href="https://www.karatelabs.io/pricing">:heavy_dollar_sign: <br/>&nbsp;&nbsp;Pricing&nbsp;&nbsp;</a></h3>
        </th>       
        <th>
            <h3><a href="https://plugins.jetbrains.com/plugin/19232-karate"><br/>:zap: <br/>&nbsp;Market&nbsp;<br/>place</a></h3>
        </th>
        <th>
            <h3><a href="https://github.com/karatelabs/intellij-plugin/issues">:octocat: <br/>&nbsp;&nbsp;Issues&nbsp;&nbsp;</a></h3>
        </th>        
    </tr>
</table>

<table>
    <tr>
        <th>PLUS</th>
        <th>PRO</th>
        <th>ULTIMATE</th>
        <th>Enterprise</th>
    </tr>    
    <tr>
        <td>            
            <ul>
                <li>Syntax coloring</li>                
                <li>Run Feature from editor</li>
                <li><a href="#run-from-editor">Run single Scenario / Example</a></li>
                <li><a href="#run-configurations">Run Configurations</a></li>
                <li><a href="#structure-view">Structure view</a></li>
                <li><a href="#code-formatting">Code formatting</a></li>
                <li><a href="#test-results">In-IDE test results</a></li>
            </ul>
        </td>
        <td>
            <i>&nbsp;&nbsp;&nbsp;&nbsp;(includes all in PLUS)</i>
            <ul>
                <li><a href="#auto-complete">Auto complete</a></li>                
                <li><a href="#code-folding">Code folding</a></li>
                <li><a href="#references">Jump to references</a></li>
                <li><a href="#json-re-formatting">JSON re-formatting</a></li>
                <li><a href="#run-folder">Run all tests in folder</a></li>
                <li>Run Karate Labs add-ons (e.g. <a href="https://github.com/karatelabs/karate-addons/blob/main/karate-kafka/README.md">Kafka</a>)</li>
                <li><a href="#debug">Debug Karate test</a></li>
                <li>Debug Java &amp; Karate in same session</li>
                <li><a href="#debug-karate-from-java">Java debug session stops at Karate breakpoints</a></li>
            </ul>        
        </td>
        <td>
            <i>&nbsp;&nbsp;&nbsp;&nbsp;(includes all in PRO)</i>
            <ul>
                <li><a href="#openapi-import">Import OpenAPI (or Swagger) definitions</a></li>
                <li><a href="#create-tests-from-spec">Convert API specifications to Karate tests</a></li>
                <li><a href="#spec-payload-chooser">Choose payload sub-sets from spec schemas</a></li>
                <li><a href="#explore-api-from-spec">Explore API directly from OpenAPI or Swagger specs</a></li>
                <li><a href="#create-mock-from-spec">Convert API specifications to Karate mocks</a></li>
                <li><a href="#spec-impact-analysis">Impact analyis of API changes</a></li>
            </ul>
        </td>
        <td>
            <ul>
                <li>Priority support</li>
                <li>SSO / SAML support</li>
                <li><a href="#offline-license">Offline license</a></li>
                <li><a href="https://www.karatelabs.io/contact-us">Contact us</a></li>
            </ul>        
        </td>        
    </tr>
</table>

This plugin works fully-featured on [IntelliJ Community Edition](https://www.jetbrains.com/products/compare/?product=idea&product=idea-ce). JS support is built-in and **does not** require you to be running [IntelliJ IDEA Ultimate](https://www.jetbrains.com/idea/features/).

> By using this plugin, you agree to the Karate Labs [EULA](https://karatelabs.io/eula). Data on a few user actions is collected and subject to our [Privacy Policy](https://karatelabs.io/privacy-policy). HTML reports generated by the Karate open-source library has [additional analytics](https://github.com/karatelabs/karate/blob/master/karate-core/src/test/resources/analytics.md).

## License Activation
License activation and status is unified within the IntelliJ settings UI. Go to `Settings -> Languages & Frameworks -> Karate` and you should see something like this.

<img src="resources/sign-in.jpg" height="300px"/>

<p>&nbsp;</p>

Click the `Sign In` link and you will be taken through the usual flow linked to your existing subscription details.

Once you have authenticated successfully, copy the session ID from the browser and paste it into the input-box now showing in the IntelliJ settings view. Click `Apply` to complete the sign-in.

Once signed-in you can work offline. You can always open this settings page to see how many days are left in your session.

If you need an [offline license](#offline-license) because of strict security or similar restrictions in your environment, please [contact us](https://www.karatelabs.io/contact-us). Please note that this option is available only for enterprise customers of Karate Labs.

### Session Renewal
Once you have signed-in once, you can renew or extend the session in just one-click without having to authenticate via a browser. 

Use the `Renew Session` button as shown below. If you have a valid subscription, this will work even if your session has expired.

<img src="resources/renew.jpg" height="150px"/>

<p>&nbsp;</p>

For convenience, when using the Karate plugin, you will see a notification when there are 10 or less days for your session to expire.

If you click the `Renew Now` link, you will be taken directly to the Karate Settings screen where you can do the renewal in one-click.

<img src="resources/renew-notification.jpg" height="100px"/>

## Run From Editor
You can right-click a `*.feature` file in the explorer or within the editor and run it. An IntelliJ [run-configuration](#run-configurations) will be created which you can customize later if needed.

You can also run a single `Scenario` by right-clicking on it or using the "gutter" icon. Right-clicking on the gutter-icon also brings up the option to debug instead of run.

<img src="resources/run-scenario.jpg" height="300px"/>

<p>&nbsp;</p>

You can even run a single "example" in a `Scenario Outline` by right-clicking on one of the data-rows within an `Examples` table.

<img src="resources/run-example.jpg" height="250px"/>

## Troubleshooting

If you don't see the options to run a test (described above) make sure that the Karate icon shows for open `*.feature` files in the editor and in the project-explorer. If the "Cucumber for Java" plugin is installed, it can conflict with Karate.

<img src="resources/file-icon.jpg" height="250px"/>

<p>&nbsp;</p>

To make sure that the Karate plugin can run `*.feature` files, you can go to `Settings -> Editor -> File Types -> Recognized File Types`.

<img src="resources/file-types.jpg" height="250px"/>

If you really need to mix Cucumber tests and Karate tests in the same workspace, refer to [this post on Stack Overflow](https://stackoverflow.com/a/76462277/143475).

## Run Configurations
IntelliJ [run-configurations](https://www.jetbrains.com/help/idea/run-debug-configuration.html) are useful for being able to re-run tests with specific parameters. All the typical Karate options you need are supported, including control over the JVM parameters and working directory.

<img src="resources/run-config.jpg" height="450px"/>

## Structure View
The IntelliJ [structure view](https://www.jetbrains.com/help/idea/viewing-structure-of-a-source-file.html) is supported so you can navigate large files with ease.

<img src="resources/structure.jpg" height="450px"/>

## Code Formatting
Intellij [code-formatting](https://www.jetbrains.com/help/idea/reformat-and-rearrange-code.html) shortcuts work and even JSON within doc-string blocks will be formatted correctly.

<img src="resources/reformat.jpg" height="250px"/>

## Test Results
Test results integrate into the IntelliJ test-results view. The HTML report is one-click away.

<img src="resources/test-results.jpg" height="300px"/>

<p>&nbsp;</p>

When there are test errors - you can navigate directly to the feature file and line that failed by clicking on hyperlinks in the log.

<img src="resources/test-error-link.jpg" height="300px"/>

## Auto Complete
Besides the syntax validation, you have auto-complete for the most commonly used keywords.

<img src="resources/autocomplete.jpg" height="250px"/>

## Code Folding
You can collapse sections to make it easier to deal with long tests.

<img src="resources/code-folding.jpg" height="300px"/>

## References
You can click to navigate to called files. Or hover to see context. Currently this is supported for the `read()` and `Java.type()` syntax. More coming soon, including auto-complete.

<img src="resources/reference-java.jpg" height="250px"/>

## JSON Re-formatting
Place the cursor within JSON to see options to re-format it, you will see a "bulb" icon.

<img src="resources/json-bulb.jpg" height="200px"/>

<p>&nbsp;</p>

The following options are possible (depending on context):

* Lenient - just like JS, where single-quotes are used and no quotes are needed for property keys
* Strict - Strict JSON, using double-quotes
* Make single Line - convert multi-line JSON to a single line
* Make multi-line - convert a single line of JSON to multi-line, including triple-quotes

## Run Folder

As a convenience, you can right-click and run a folder from the explorer view.

<img src="resources/run-folder.jpg" height="200px"/>

## Debug

You can set break-points on Karate feature files in debug mode. The Karate debugger can even step-back and hot-reload simple edits to your test. Note that the options for "Hot Reload" and "Step Back" may be hidden away in some versions of the IntelliJ UI.

<img src="resources/debug.jpg" height="600px"/>

### Debug Java from Karate

A Karate debug session will even stop at Java breakpoints.

### Debug Karate from Java

You can also start a normal Java debug session that uses the [Karate `Runner` Java API](https://github.com/karatelabs/karate#junit-5-parallel-execution) but still stop at Karate breakpoints. This requires you to be using Karate version 1.5.0 or greater. The two things to note are:

* You have to use the Karate `Runner` Java API like in [this example](https://github.com/karatelabs/karate-todo/blob/main/src/test/java/app/api/ApiTest.java).
* You have to use an IntelliJ Java or [JUnit](https://www.jetbrains.com/help/idea/run-debug-configuration-junit.html) Run Configuration but in `Debug` mode. Right-clicking on a JUnit class (or method) or a Java class with a `main` method is normally [how you create one](https://www.jetbrains.com/help/idea/run-debug-configuration.html).

Since the `Runner` API is a pure Java API - it does not get in the way of your flow or infrastructure. For example, your Java or JUnit code can start a server and then start your Karate test suite, and the debug session will stop at both Java and Karate breakpoints.

### Debug Gradle Projects

If you have a Gradle project, the Run Configuration that IntelliJ creates may default to a Gradle specific one and not the IntelliJ native Java Run Configuration needed to honor breakpoints in Karate feature files. The default is easy to change by going to `Settings > Build, Execution, Deployment > Build Tools > Gradle > Build and run`. Set the `Run tests using` option to `Choose per test` as shown below.

<img src="https://github.com/user-attachments/assets/3770fdf9-199f-4157-b6b6-a982709056b4" height="500"/>

Now when right-clicking on a JUnit test or Java class, you will have the option to choose the Java Run Configuration instead of the Gradle one.

<img src="https://github.com/user-attachments/assets/bb9b750c-6c54-4637-8516-bc2b9165e398" height="150"/>

## Offline License

> [!IMPORTANT]  
> The Offline License is a [paid feature for enterprises](https://www.karatelabs.io/pricing). Please make sure you know who your designated admin is before you perform this step.

On the [License Activation](#license-activation) settings page, you should see a link called `Offline License`. When you click it you should see something like this.

<img src="resources/offline-license.jpg" height="200px"/>

A unique code for your system will be shown which you can cut and paste. In the example above, it is: `DhHOFKHvd7XYTi+rQnNTJQ==`

Send that unique code to your designated admin. You will be issued a license that is tied to this unique code. To apply the license, copy all the text from the license file (which should start with: `-----BEGIN LICENSE FILE-----`) and paste it into the text-area provided. Now you should see something like this.

<img src="resources/offline-license-paste.jpg" height="500px"/>

Click [Apply] and you should a confirmation message and your license details.

## OpenAPI Import

Right-clicking on any file in the Project Explorer brings up a menu called `Karate Import` as shown below. OpenAPI or Swagger files of any version are supported.

<img src="resources/spec-import.jpg" height="300px"/>

When successful, you will see a Karate tool-window typically on the top-right of your IntelliJ layout, shown below. You can exand to the level of HTTP methods per API path such as `GET`, `POST` etc.

When you click on an HTTP method, you can preview the equivalent Karate snippet.

<img src="resources/spec-tool-window.jpg" height="450px"/>

## Create Tests From Spec

Creating a Karate test from an Open API or Swagger file is much quicker now. Just click and drag from the tree into the editor window. 

<img src="resources/spec-to-test.gif" height="450px"/>

This way you have full control over the sequence of API calls you make within a Karate test.

## Spec Payload Chooser

A challenge with complex APIs is that the schemas include all possible variations and permutations even though only some fields are commonly used. To solve this problem, you can right-click on an HTTP method in the tree. This gives you a menu to view the Request or Response when available.

<img src="resources/spec-method-payload.jpg" height="300px"/>

This brings up a dialog where you can un-select the JSON keys you need. It is very easy to choose payload sub-sets using this UI.

<img src="resources/spec-select-json.gif" height="500px" width="600"/>

Once satisfied you can cut and paste into a test or wherever you need the payload.

## Explore API From Spec

Coming soon.

## Create Mock From Spec

You can export all data within an imported schema into a plain-vanilla JS file that can be used as a Karate mock. A detailed explanation and sample can be found at this GitHub project: [karate-oas-demo](https://github.com/ptrthomas/karate-oas-demo).

Use the `Save Mock` button at the bottom of the tool-window.

## Spec Impact Analysis

You can also export a plain-vanilla JS file that dumps all API and schema information in a format designed to make impact analysis easier using your existing IDE diff features. The format is designed to flatten all metdata into single lines in a text file.

Once you have exported this file, you can compare two of these using the IntelliJ `Compare With` menu and view the diff in an intuitive manner.

<img src="resources/spec-diff.jpg" height="400px"/>

You can scroll-right and view the differences even for very large API schemas.

### Test Impact Analysis

Coming soon - the capability to show which tests and mocks are impacted due to changes in the API spec. readmeEtag: '"9407f160423616e4a9cea7872a9343c60b62f031"' readmeLastModified: Fri, 10 Jan 2025 10:21:09 GMT repositoryId: 496290531 description: Karate IntelliJ Plugin created: '2022-05-25T15:33:28Z' updated: '2026-01-29T12:28:05Z' language: null archived: false stars: 13 watchers: 3 forks: 6 owner: karatelabs logo: https://avatars.githubusercontent.com/u/91312095?v=4 repoEtag: '"5d8379951b41f093b9f1aa72c64aa135550fbb0e92447fa2068a52b5c86140f6"' repoLastModified: Thu, 29 Jan 2026 12:28:05 GMT category: Testing foundInMaster: true oldLocations: - https://github.com/karatelabs/karate-intellij-plugin - source: openapi3 tags repository: https://github.com/codewithpraveen/openapi-specification v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNpZmljYXRpb24gQ291cnNlIFJlcG8KT3BlbkFQSSAoU3dhZ2dlcikgU3BlY2lmaWNhdGlvbiBmb3IgU29mdHdhcmUgRGV2ZWxvcGVycywgYSBndWlkZSB0byBsZWFybmluZyBPcGVuQVBJIFNwZWNpZmljYXRpb24gZm9yIGhpZ2hseSBwcm9kdWN0aXZlIEFQSSBkZXNpZ24sIGRvY3VtZW50YXRpb24sIGRldmVsb3BtZW50LCAmYW1wOyB0ZXN0aW5nLiAKCiMjIFJlcG9zaXRvcnkgZm9yIHRoZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gQ291cnNlClRoaXMgcmVwbyBjb3ZlcnMgdGhlIGZvbGxvd2luZzoKKiBDb21wbGV0ZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gZm9yIENvbGxlZ2UgTWFuYWdlbWVudCBTeXN0ZW0gKENNUykgdXNlZCBpbiB0aGUgY291cnNlIHZpZGVvcy4KKiBBc3NpZ25tZW50IChyZWFkeS1tYWRlIHByb2plY3RzIHRvIHN0YXJ0IHdvcmtpbmcpCiogQXNzaWdubWVudCBTb2x1dGlvbnMKCiMjIENoZWNrIG91dCB0aGlzIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBpbiBQb3N0bWFuClshW1J1biBpbiBQb3N0bWFuXShodHRwczovL3J1bi5wc3Rtbi5pby9idXR0b24uc3ZnKV0oaHR0cHM6Ly9hcHAuZ2V0cG9zdG1hbi5jb20vcnVuLWNvbGxlY3Rpb24vMjU2ODQ5NS04N2FkMGQ2My1jMDNhLTQ5YTAtOTNlMy1mOGQ4NzIzMmJhZWI/YWN0aW9uPWNvbGxlY3Rpb24lMkZmb3JrJmNvbGxlY3Rpb24tdXJsPWVudGl0eUlkJTNEMjU2ODQ5NS04N2FkMGQ2My1jMDNhLTQ5YTAtOTNlMy1mOGQ4NzIzMmJhZWIlMjZlbnRpdHlUeXBlJTNEY29sbGVjdGlvbiUyNndvcmtzcGFjZUlkJTNEOTdiMjRiYjItNjI3Yi00ODhjLTk3OTYtOTU0YThiM2FhOTE0KQoKIyMgRW5yb2xsIGluIHRoaXMgY291cnNlIHdpdGggOTAlIGRpc2NvdW50CkNsaWNrIHRoaXMgbGluayB0byBlbnJvbGwgaW4gdGhpcyBjb3Vyc2U6Cmh0dHBzOi8vY29kZXdpdGhwcmF2ZWVuLmNvbS9vcGVuYXBpLWNvdXJzZQo= readmeEtag: '"94928c291cc14801f09f09798a5aaf74cdf689eb"' readmeLastModified: Wed, 02 Nov 2022 22:57:19 GMT repositoryId: 312802042 description: >- OpenAPI (Swagger) Specification for Software Developers, a guide to learning OpenAPI Specification for highly productive API design, documentation, development, & testing. created: '2020-11-14T11:21:51Z' updated: '2024-09-05T21:12:52Z' language: null archived: false stars: 11 watchers: 2 forks: 11 owner: CodeWithPraveen logo: https://avatars.githubusercontent.com/u/62539682?v=4 repoEtag: '"834a43e718a37104225808bfd08a3bf3b0e9963023fac55fbcfd861fc902d03b"' repoLastModified: Thu, 05 Sep 2024 21:12:52 GMT foundInMaster: true category: Testing id: 59826edff234d7fdd1f71a260e01f5d2 - source: openapi3 tags repository: https://github.com/responsibleapi/responsible v3: true id: b5dd9396be5b2f2cc6eb74770ebd2970 repositoryMetadata: base64Readme: >- IyBSZXNwb25zaWJsZUFQSQoKQSBzbWFsbCBsYW5ndWFnZSB0aGF0IGNvbXBpbGVzIHRvIE9wZW5BUEkgMy4xLjAuCgojIyBMYW5ndWFnZQoKUmVzcG9uc2libGUgaXMgYmFzZWQgb24gW0tETCAxLjAuMF0oaHR0cHM6Ly9rZGwuZGV2KS4KCi0gW0V4YW1wbGVzXShleGFtcGxlcy8pCi0gW0xhbmd1YWdlIHR1dG9yaWFsXShUVVRPUklBTC5tZCkKLSBbTGFuZ3VhZ2UgcmVmZXJlbmNlXShSRUZFUkVOQ0UubWQpCgojIyMgQ29tcGlsaW5nIHRvIE9wZW5BUEkKCk9uY2UgeW91J3ZlIGNyZWF0ZWQgeW91ciBSZXNwb25zaWJsZSBmaWxlLCB5b3UgY2FuIHVubG9jayBhbGwgb2YgdGhlIE9wZW5BUEkgdG9vbGluZzoKCmBgYHNoCmJ1bnggQHJlc3BvbnNpYmxlYXBpL2NsaSByZXNwb25zaWJsZS5rZGwgLW8gL3RtcC9vcGVuYXBpLmpzb24KYGBgCgpCZWxvdyBpcyBzb21lIG9mIHRoZSBPcGVuQVBJIHRvb2xpbmcgcHJvdmlkZWQgYnkgUmVzcG9uc2libGUgYW5kIHRoaXJkIHBhcnRpZXMuCgojIyBWYWxpZGF0aW5nIHJlcXVlc3RzCgojIyMgVHlwZXNjcmlwdAoKIyMjIyBbSG9ub10oaHR0cHM6Ly9ob25vLmRldikKClNlZSBbcGFja2FnZXMvaG9uby9SRUFETUUubWRdKHBhY2thZ2VzL2hvbm8vUkVBRE1FLm1kKQoKIyMjIEtvdGxpbgoKIyMjIyBbVmVydC54XShodHRwczovL3ZlcnR4LmlvKQoKVXNlIHRoZSBbYnVpbHQtaW4gT3BlbkFQSSByZXF1ZXN0IHZhbGlkYXRvcl0oaHR0cHM6Ly92ZXJ0eC5pby9kb2NzL3ZlcnR4LW9wZW5hcGkvamF2YS8jX3ZhbGlkYXRpb25fb2ZfcmVxdWVzdHMpLgoKIyMgR2VuZXJhdGluZyBhIGNsaWVudDoKCiMjIyBJbnN0YWxsCgpgYGBzaApicmV3IGluc3RhbGwgb3BlbmFwaS1nZW5lcmF0b3IKYGBgCgojIyMgVHlwZXNjcmlwdAoKYGBgc2gKb3BlbmFwaS1nZW5lcmF0b3IgZ2VuZXJhdGUgLWcgdHlwZXNjcmlwdC1mZXRjaCAtaSAvdG1wL29wZW5hcGkuanNvbiAtbyBnZW4vIC0tYWRkaXRpb25hbC1wcm9wZXJ0aWVzPXR5cGVzY3JpcHRUaHJlZVBsdXM9dHJ1ZSxtb2RlbFByb3BlcnR5TmFtaW5nPW9yaWdpbmFsLG51bGxTYWZlQWRkaXRpb25hbFByb3BzPXRydWUsZW51bVByb3BlcnR5TmFtaW5nPW9yaWdpbmFsLHN1cHBvcnRzRVM2PXRydWUsdXNlU2luZ2xlUmVxdWVzdFBhcmFtZXRlcj1mYWxzZQpgYGAKCiMjIyBLb3RsaW4KCmBgYHNoCm9wZW5hcGktZ2VuZXJhdG9yIGdlbmVyYXRlIC1nIGtvdGxpbiAtaSAvdG1wL29wZW5hcGkuanNvbiAtbyBnZW4vIC0tYWRkaXRpb25hbC1wcm9wZXJ0aWVzPWxpYnJhcnk9anZtLXZlcnR4CmBgYAoKIyMgVGVzdGluZwoKVGhlIGlkZWEgYmVoaW5kIHRlc3Rpbmcgd2l0aCBPcGVuQVBJIGlzIHZhbGlkYXRpbmcgc2VydmVyIHJlc3BvbnNlcyBhZ2FpbnN0IHRoZSBjb250cmFjdC4KCiMjIyBLb3RsaW4KCiMjIyMgVmVydC54CgpTZWUgaHR0cHM6Ly9naXRodWIuY29tL3Jlc3BvbnNpYmxlYXBpL3Rlc3Qta290bGluLXZlcnR4CgojIyMgUHl0aG9uCgpUbyBiZSBwdWJsaXNoZWQsIHNlZSBodHRwczovL2dpdGh1Yi5jb20vbGlzdGVuYm94L3lhbmljL2Jsb2IvbWFzdGVyL3Rlc3RzL3Jlc3BvbnNpYmxlLnB5CgojIyMgVHlwZXNjcmlwdAoKIyMjIyBIb25vCgohW1Jlc3BvbnNpYmxlIEhvbm8gdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9AcmVzcG9uc2libGVhcGkvaG9ubykKCmBgYHNoCmJ1biBpbnN0YWxsIEByZXNwb25zaWJsZWFwaS9ob25vCmBgYAoKYGBgdHlwZXNjcmlwdApjb25zdCByZXNwb25zaWJsZSA9IG5ldyBSZXNwb25zaWJsZTxrZXlvZiBIYW5kbGVycywgQXBwRW52PigKICBvcGVuQXBpSW50ZXJuYWwgYXMgb2FzMzEuT3BlbkFQSU9iamVjdCwKICBob25vLAopCgp0ZXN0KCJzaWdudXAiLCBhc3luYyAoKSA9PiB7CiAgYXdhaXQgcmVzcG9uc2libGUuY2hlY2soInNpZ251cCIsIHsKICAgIHJlcTogewogICAgICBib2R5OiB7CiAgICAgICAgbmFtZTogZ2VuU3RyKCksCiAgICAgICAgZW1haWw6IGdlbkVtYWlsKCksCiAgICAgICAgcGFzc3dvcmQ6IGdlblN0cigpLAogICAgICB9LAogICAgfSwKICAgIHN0YXR1czogMjAxLAogIH0pCn0pCmBgYAoKIyMgRnV6emluZwoKVXNlIGh0dHBzOi8vc2NoZW1hdGhlc2lzLmlvIGZvciBub3cKCmBgYHNoCmJyZXcgaW5zdGFsbCB1dgp1dnggc2NoZW1hdGhlc2lzIHJ1biAtLWNoZWNrcyBhbGwgLS1iYXNlLXVybCBodHRwOi8vbG9jYWxob3N0OjgwODAgLS13b3JrZXJzIDQwIHNyYy9tYWluL3Jlc291cmNlcy9vcGVuYXBpLmpzb24KYGBgCgpVbHRpbWF0ZWx5IHRoZSBwbGFuIGlzIHRvIGludGVncmF0ZSBmdXp6aW5nIGludG8gdGhlIHRlc3RpbmcgbGlicmFyaWVzLgoKIyMgU2VlIGFsc28KCi0gW09wZW5BUEldKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vYWJvdXQvKQotIFtBV1MgU21pdGh5XShodHRwczovL3NtaXRoeS5pby8yLjAvaW5kZXguaHRtbCkKLSBbQVBJIEJsdWVwcmludF0oaHR0cHM6Ly9hcGlibHVlcHJpbnQub3JnLykKLSBbV1NETF0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvV2ViX1NlcnZpY2VzX0Rlc2NyaXB0aW9uX0xhbmd1YWdlLykKLSBbUHJvdG9mb3JjZV0oaHR0cHM6Ly93d3cucHJvdG9mb3JjZS5pby8pCi0gW1N0b3BsaWdodF0oaHR0cHM6Ly9zdG9wbGlnaHQuaW8vKQotIFtSQU1MXShodHRwczovL3JhbWwub3JnLykK readmeEtag: '"15b5324f643b7d50daeaefca178e09f268256298"' readmeLastModified: Wed, 01 Jan 2025 12:16:55 GMT repositoryId: 451072524 description: A small language that compiles to OpenAPI created: '2022-01-23T10:39:17Z' updated: '2025-05-25T19:29:24Z' language: TypeScript archived: false stars: 10 watchers: 1 forks: 0 owner: responsibleapi logo: https://avatars.githubusercontent.com/u/119290467?v=4 repoEtag: '"488b89f8e5a82ea49d2e0522188928f16b45e5a5f3c6e3558deca80a3d38096b"' repoLastModified: Sun, 25 May 2025 19:29:24 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/meoyawn/responsible - source: openapi3 tags repository: https://github.com/scalar/sandbox v3: true id: 638257589df790fa77577129e5bdb83d repositoryMetadata: base64Readme: >- IyBTY2FsYXIgU2FuZGJveAoKQW4gZWFzeSB3YXkgdG8gc2hhcmUgT3BlbkFQSSBmaWxlcy4KCiMjIERldmVsb3BtZW50CgpgYGAKcG5wbSBpbnN0YWxsCnBucG0gZGI6bWlncmF0ZQpwbnBtIGRldgpgYGAKCiMjIFVwZGF0ZSB0aGUgc2NoZW1hCgoxLiBDaGFuZ2UgdGhlIHNjaGVtYTogZGIvc2NoZW1hLnRzCjIuIENyZWF0ZSBhIG1pZ3JhdGlvbjogYHBucG0gZGI6Z2VuZXJhdGVgCjMuIEFwcGx5IG1pZ3JhdGlvbnM6IGBwbnBtIGRiOm1pZ3JhdGVgCg== readmeEtag: '"1f42178be3a3b42e3811a2768270fb78005b13f2"' readmeLastModified: Fri, 16 Aug 2024 15:14:45 GMT repositoryId: 740729887 description: The easiest way to share OpenAPI files created: '2024-01-09T00:11:59Z' updated: '2026-02-04T15:35:30Z' language: Vue archived: false stars: 20 watchers: 1 forks: 6 owner: scalar logo: https://avatars.githubusercontent.com/u/301879?v=4 repoEtag: '"16d730c78a9a01892c77af28d432abec9b2197b3f2ad57d43836ea0ea32da691"' repoLastModified: Wed, 04 Feb 2026 15:35:30 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/javachinna/spring-boot-oauth2-jwt v3: true id: 78669f0233dab87b399d7de324a72dd0 repositoryMetadata: base64Readme: >- I1NlY3VyZSBSRVNUIEFQSSB3aXRoIE9BdXRoMiBKV1QgQXV0aGVudGljYXRpb24KCjEuIFtTcHJpbmcgQm9vdCBSRVNUIEFQSSBmb3IgVXNlciByZWdpc3RyYXRpb24gYW5kIGF1dGhlbnRpY2F0aW9uIHdpdGggT0F1dGggMi4wIFJlc291cmNlIHNlcnZlciB1c2luZyBKV1QgYXMgYmVhcmVyIHRva2VuXShodHRwczovL3d3dy5qYXZhY2hpbm5hLmNvbS9zZWN1cmUtc3ByaW5nLXJlc3QtYXBpLW9hdXRoMi1qd3QtYXV0aGVudGljYXRpb24vKQoKMi4gW0NvbmZpZ3VyZSBPcGVuQVBJIDMgU3BlYyB3aXRoIEJhc2ljICYgSldUIEF1dGhlbnRpY2F0aW9uXShodHRwczovL3d3dy5qYXZhY2hpbm5hLmNvbS9jb25maWd1cmUtb3BlbmFwaS1zcGVjLWJhc2ljLWp3dC1hdXRoZW50aWNhdGlvbi8pCjMuIFtDb25maWd1cmUgU3ByaW5nIFNlY3VyaXR5IOKAkyBTZWN1cmUgRGlmZmVyZW50IFVSTHMgRGlmZmVyZW50bHldKGh0dHBzOi8vd3d3LmphdmFjaGlubmEuY29tL3NwcmluZy1zZWN1cml0eS1zZWN1cmUtZGlmZmVyZW50LXVybHMtZGlmZmVyZW50bHkvKQo0LiBbUG9wdWxhdGUgRGF0YWJhc2UgZnJvbSBDU1YgZmlsZSBpbiAyIFNpbXBsZSBTdGVwc10oaHR0cHM6Ly93d3cuamF2YWNoaW5uYS5jb20vc3ByaW5nLWJvb3QtcmVhZC1jc3YtZmlsZS1wb3B1bGF0ZS1kYXRhYmFzZS8pCg== readmeEtag: '"bd56d9cf3f724bf5d64cc4293a5446d846090291"' readmeLastModified: Tue, 14 Feb 2023 23:41:03 GMT repositoryId: 599821532 description: Secure REST API with OAuth2 JWT Authentication created: '2023-02-10T00:20:30Z' updated: '2025-04-08T11:16:40Z' language: Java archived: false stars: 12 watchers: 1 forks: 15 owner: JavaChinna logo: https://avatars.githubusercontent.com/u/23244086?v=4 license: MIT repoEtag: '"1a9f97dc05af551067bc88274cc9df4324cec966a1069fd1393f530f65a0347a"' repoLastModified: Tue, 08 Apr 2025 11:16:40 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/kevinmmartins/python-flask-connexion-example-openapi3 v3: true repositoryMetadata: base64Readme: >- IyBCYXNpYyBwcm9qZWN0IHdpdGggRmxhc2ssIENvbm5leGlvbiBhbmQgT3BlbkFwaSAzCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9rZXZpbm1tYXJ0aW5zL3B5dGhvbi1mbGFzay1jb25uZXhpb24tZXhhbXBsZS1vcGVuYXBpMy5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9rZXZpbm1tYXJ0aW5zL3B5dGhvbi1mbGFzay1jb25uZXhpb24tZXhhbXBsZS1vcGVuYXBpMykKCkJhc2ljIFB5dGhvbiBwcm9qZWN0IHVzaW5nIENvbm5leGlvbiBhbmQgRmxhc2sKCmBgYGh0dHAKaHR0cHM6Ly9naXRodWIuY29tL3NwZWMtZmlyc3QvY29ubmV4aW9uCmBgYAoKIyMgUmVxdWlyZW1lbnRzCgoqIERvY2tlciBDb21wb3NlIDEuMjEuMisKKiBQeXRob24gMy45ICsKCiMjIFJ1biB3aXRoIERvY2tlciBDb21wb3NlCgpgYGBiYXNoCiMgYnVpbGRpbmcgdGhlIGNvbnRhaW5lcgpzdWRvIGRvY2tlci1jb21wb3NlIGJ1aWxkCgojIHN0YXJ0aW5nIHVwIGEgY29udGFpbmVyCnN1ZG8gZG9ja2VyLWNvbXBvc2UgdXAKYGBgCgojIyBCdWlsZCB0aGUgdmlydHVhbCBlbnZpcm9ubWVudAoKYGBgYmFzaApwaXAzIGluc3RhbGwgdmlydHVhbGVudgp2aXJ0dWFsZW52IC1wIHB5dGhvbjMuOSB2ZW52CnNvdXJjZSB2ZW52L2Jpbi9hY3RpdmF0ZQpwaXAzIGluc3RhbGwgLXIgcmVxdWlyZW1lbnRzLnR4dApwaXAzIGluc3RhbGwgLXIgdGVzdC1yZXF1aXJlbWVudHMudHh0CmBgYAoKIyMgTGF1bmNoIHRoZSBzZXJ2ZXIKCmBgYGJhc2gKcHl0aG9uMyAtbSBiYXNpYwpgYGAKCllvdSBzaG91bGQgc2VlIG91dHB1dCBzaW1pbGFyIHRvIHRoaXM6CgpgYGBiYXNoCmBDb25uZXhpb25NaWRkbGV3YXJlLnJ1bmAgaXMgb3B0aW1pemVkIGZvciBkZXZlbG9wbWVudC4gRm9yIHByb2R1Y3Rpb24sIHJ1biB1c2luZyBhIGRlZGljYXRlZCBBU0dJIHNlcnZlci4KSU5GTzogICAgIFN0YXJ0ZWQgc2VydmVyIHByb2Nlc3MgWzU3NTNdCklORk86ICAgICBXYWl0aW5nIGZvciBhcHBsaWNhdGlvbiBzdGFydHVwLgpJTkZPOiAgICAgQXBwbGljYXRpb24gc3RhcnR1cCBjb21wbGV0ZS4KSU5GTzogICAgIFV2aWNvcm4gcnVubmluZyBvbiBodHRwOi8vMTI3LjAuMC4xOjgwODEgKFByZXNzIENUUkwrQyB0byBxdWl0KQpgYGAKCiMjIFN3YWdnZXIgZGVmaW5pdGlvbgoKVmlldyB0aGUgZ2VuZXJhdGVkIFN3YWdnZXIgVUkgYXQgdGhpcyBVUkw6CgpgYGBodHRwCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MS92MS9iYXNpYy91aS8KYGBgCgojIyBIZWFsdGggQ2hlY2sKCkNoZWNrIHRoZSBzZXJ2ZXIgaGVhbHRoIGF0IHRoaXMgVVJMOgoKYGBgaHR0cApodHRwOi8vbG9jYWxob3N0OjgwODEvdjEvYmFzaWMvcGluZwpgYGAKCiMjIExhdW5jaCB0ZXN0cwoKYGBgYmFzaApzb3VyY2UgdmVudi9iaW4vYWN0aXZhdGUKcGlwMyBpbnN0YWxsIHRveAp0b3gKYGBgCg== readmeEtag: '"f0b9ffcee699fc9883a928f417df622ec88ded4d"' readmeLastModified: Thu, 10 Apr 2025 23:43:02 GMT repositoryId: 181561736 description: >- Basic REST project with Flask , Connexion by Zalando , OpenApi3, Docker and tox created: '2019-04-15T20:35:56Z' updated: '2025-10-07T01:30:06Z' language: Python archived: false stars: 10 watchers: 1 forks: 6 owner: kevinmmartins logo: https://avatars.githubusercontent.com/u/20428703?v=4 license: Apache-2.0 repoEtag: '"92301be75a940ce3bf50d96eb1d4d598bf03cc6d29d55ff0f16c9e31a09280c5"' repoLastModified: Tue, 07 Oct 2025 01:30:06 GMT foundInMaster: true category: - Mock - Server Implementations id: c3515799d051759b497d216efbf7d671 - source: openapi3 tags repository: https://github.com/mokkapps/openapi-angular-spring-demo v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWFuZ3VsYXItc3ByaW5nLWRlbW8KQSBkZW1vIGFwcGxpY2F0aW9uIHdoaWNoIGRlbW9uc3RyYXRlcyBPcGVuQVBJIGdlbmVyYXRvciBpbiBhIEFuZ3VsYXIgKyBTcHJpbmcgQm9vdCBhcHBsaWNhdGlvbgo= readmeEtag: '"3052e99afe77461677d14ac17f8d0413278b41ee"' readmeLastModified: Sun, 23 Feb 2020 14:05:05 GMT repositoryId: 242348703 description: >- A demo application which demonstrates OpenAPI generator in a Angular + Spring Boot application created: '2020-02-22T13:44:36Z' updated: '2023-11-05T10:51:03Z' language: Java archived: false stars: 10 watchers: 1 forks: 4 owner: Mokkapps logo: https://avatars.githubusercontent.com/u/3127210?v=4 license: MIT repoEtag: '"3fa25b7252fef84d9619aaa67d54b1ded0eb8fee455b9abb951bc799ca35a65e"' repoLastModified: Sun, 05 Nov 2023 10:51:03 GMT foundInMaster: true category: - Low-level Tooling - SDK - Server Implementations id: c0b59bc62c52bcac0736a7fc32b70d86 - source: openapi3 tags repository: https://github.com/systangotechnologies/swagger-generator-koa v3: true repositoryMetadata: base64Readme: >- # swagger-generator-koa

NPM module to generate swagger documentation for KOA APIs with minimum additional effort.

>[![Downloads](https://badgen.net/npm/dt/swagger-generator-koa)](https://www.npmjs.com/package/swagger-generator-koa) [![npm dependents](https://badgen.net/npm/dependents/swagger-generator-koa)](https://www.npmjs.com/package/swagger-generator-koa?activeTab=dependents)

## Description
This NPM module let's you validate and generate swagger (OpenAPI) documentation for your KOA APIs without putting in much extra efforts. You just need to follow the convention for your request and response objects, and the module will take care of the rest. This module will cover your controllers, API specs along with request and response object structures.


## Usage ##

Install using npm:

```bash
$ npm install --save swagger-generator-koa
```

### Koa setup `index.js` ###

```javascript
const Koa = require('koa');
const app = new Koa();
const swagger = require("swagger-generator-koa");

// Define your router here

const options = {
	title: "swagger-generator-koa",
	version: "1.0.0",
	host: "localhost:5000",
	basePath: "/",
	schemes: ["http", "https"],
	securityDefinitions: {
		Bearer: {
			description: 'Example value:- Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJpZCI6IjU5MmQwMGJhNTJjYjJjM',
			type: 'apiKey',
			name: 'Authorization',
			in: 'header'
		}
	},
	security: [{Bearer: []}],
	defaultSecurity: 'Bearer'
};


/**
 * serveSwagger must be called after defining your router.
 * @param app Koa object
 * @param endPoint Swagger path on which swagger UI display
 * @param options Swagget Options.
 * @param path.routePath path to folder in which routes files defined.
 * @param path.requestModelPath Optional parameter which is path to folder in which requestModel defined, if not given request params will not display on swagger documentation.
 * @param path.responseModelPath Optional parameter which is path to folder in which responseModel defined, if not given response objects will not display on swagger documentation.
 */
swagger.serveSwagger(app, "/swagger", options, {routePath : './src/routes/', requestModelPath: './src/requestModel', responseModelPath: './src/responseModel'});

```

### Koa router `user.js` ###

```javascript

const Router = require('koa-router');
const router = new Router();
const userController = require('../controller/user');
const {validation} = require('swagger-generator-koa');
var requestModel = require('../requestModel/users');
const BASE_URL = `/users`;

router.post(`${BASE_URL}/`, validation(requestModel[0]), userController.createUser);

router.get(`${BASE_URL}/`, userController.getUsers);

router.put(`${BASE_URL}/:userId`, userController.updateUser);

router.get(`${BASE_URL}/:userId`, userController.getUserDetails);

router.delete(`${BASE_URL}/:userId`, userController.deleteUser);

module.exports = router;

```

## Request Model `/requestModel/user.js`
  - File name for request model should be same as router file.
  - Define request model with their order of apis in router js file. For example first api in user router is create user so you need to define createUser schema with key 0.
  - Add boolean flag "excludeFromSwagger" inside requestmodel if you want to exclude any particular api from swagger documentation.
  - This Request model follows Joi module conventions, so it can also be used for request parameters validation.

```javascript
const Joi = require("@hapi/joi");
/**
 * File name for request and response model should be same as router file.
 * Define request model with their order in router js file.
 * For example first api in user router is create user so we define createUser schema with key 0.
 */
module.exports = {
    // Here 0 is the order of api in route file.
    0: {
        body: {
            firstName: Joi.string().required(),
            lastName: Joi.string().required(),
            address: Joi.string().required(),
            contact: Joi.number().required()
        },
        model: "createUser", // Name of the model
        group: "User", // Swagger tag for apis.
        description: "Create user and save details in database"
    },
    1: {
        query: {},
        path: {}, // Define for api path param here.
        header: {}, // Define if header required.
        group: "User",
        description: "Get All User"
    },
    2: {
        body: {
            firstName: Joi.string().required(),
            lastName: Joi.string().required(),
            address: Joi.string().required(),
            contact: Joi.number().required()
        },
        model: "updateUser",
        group: "User",
        description: "Update User"
    },
    3: {
        query: {},
        path: {
            userId: Joi.number().required()
        }, // Define for api path param here.
        header: {}, // Define if header required.
        model: 'getUserDetails',
        group: "User",
        description: "Get user details"
    },
    4: {
        excludeFromSwagger: false // Make it true if need to exclude apis from swagger.
    }
};
```

## Response Model `/responseModel/user.js`

 - File name for response model should be same as router file.
 - Response name should be same as model name from requestmodel. For example model name of create user api is "createUser" so key for response object will be "createUser".
 - Inside response model define responses with respect to their status code returned from apis.

```javascript

// The name of each response payload should be model name defined in Request model schema.

module.exports = {
    createUser: { // This name should be model name defined in request model.
        201: {
            message: {
                type: 'string'
            }
        },
        500: {
            internal: {
                type: 'string'
            }
        }
    },
    getUsers: {
        200: [{
            id: {
                type: 'number'
            },
            firstName: {
                type: 'string'
            },
            lastName: {
                type: 'string'
            },
            address: {
                type: 'string'
            },
            contact: {
                type: 'number'
            },
            createdAt: {
                type: 'number',
                format: 'date-time'
            },
            updatedAt: {
                type: 'number',
                format: 'date-time'
            }
        }],
        500: {
            internal: {
                type: 'string'
            }
        }
    },
    updateUser: {
        201: {
            message: {
                type: 'string'
            }
        },
        500: {
            internal: {
                type: 'string'
            }
        }
    },
    getUserDetails: {
        200: {
            id: {
                type: 'number'
            },
            firstName: {
                type: 'string'
            },
            lastName: {
                type: 'string'
            },
            address: {
                type: 'string'
            },
            contact: {
                type: 'number'
            },
            createdAt: {
                type: 'number',
                format: 'date-time'
            },
            updatedAt: {
                type: 'number',
                format: 'date-time'
            }
        },
        500: {
            internal: {
                type: 'string'
            }
        }
    },
};
```

Open `http://`<app_host>`:`<app_port>`/swagger` in your browser to view the documentation.

# Version changes

## v2.0.0

#### Added Request Parameter Validation Function

- Use `validation` function exported from this module to validate request params.

```javascript
'use strict';
const Router = require('koa-router');
const router = new Router();
const userController = require('../controller/user');
const {validation} = require('swagger-generator-koa');
var requestModel = require('../requestModel/users');
const BASE_URL = `/users`;

router.post(`${BASE_URL}/`, validation(requestModel[0]), userController.createUser);

module.exports = router;

```

## Requirements

- Node v10 or above
- KOA 2 or above

## Contributors

[Vikas Patidar](https://www.linkedin.com/in/vikas-patidar-0106/)
 readmeEtag: '"55dcea9b3ce1e6c0bb35087ded7c7786c9e2ea3a"' readmeLastModified: Fri, 06 Mar 2020 13:53:57 GMT repositoryId: 201899371 description: >- Allows you to programatically annotate your koa models with swagger info and then generate and validate your json spec file created: '2019-08-12T09:28:50Z' updated: '2024-01-30T16:14:31Z' language: TypeScript archived: false stars: 10 watchers: 3 forks: 4 owner: SystangoTechnologies logo: https://avatars.githubusercontent.com/u/24825108?v=4 license: MIT repoEtag: '"66b3ef0866da08f174a350c47a4315f4d79263822d5177b36253d605e94050f1"' repoLastModified: Tue, 30 Jan 2024 16:14:31 GMT foundInMaster: true category: - Data Validators - Parsers id: b33cb0aa622983134244f762ccab16c6 - source: openapi3 tags repository: https://github.com/ckaratzas/vertx-openapi-spec-generator v3: true repositoryMetadata: base64Readme: >- IyBWZXJ0eC1PcGVuQXBpLVNwZWMtR2VuZXJhdG9yClRoZSBwdXJwb3NlIG9mIHRoaXMgcmVwb3NpdG9yeSBpcyB0byBmYWNpbGl0YXRlIHRoZSBhdXRvbWF0aWMgZ2VuZXJhdGlvbiBvZiBPcGVuQVBJIDMgc3BlYyBmcm9tICJyZXN0LWxpa2UiIHZlcnR4IHJvdXRlcy4gVGhlIGZ1bmN0aW9uYWwgbmF0dXJlIG9mIHZlcnR4Cm1ha2VzIHRoaXMgam9iIGRpZmZpY3VsdCB0aHVzIG5vIHNlcmlvdXMgb2ZmaWNpYWwgdG9vbCBleGlzdHMgYXQgdGhlIG1vbWVudC4gSW4gb3JkZXIgdG8gbWl0aWdhdGUgdGhlIHByb2JsZW0gd2UgdHJ5IHRvIGRlc2NyaWJlIHRoZSAKZ2VuZXJhbCBhbGdvcml0aG0gYW5kIGFzc3VtcHRpb25zIG1hZGU6CgoxLiBHZW5lcmF0aW9uIG9mIE9wZW5BUEkgMyBTcGVjIGlzIGdlbmVyYXRlZCBhdCBydW50aW1lIGJ5IGludHJvc3BlY3Rpb24gb2YgdGhlIHZlcnR4IFJvdXRlciBPYmplY3Qgd2hpY2ggY29udGFpbnMgdGhlIHJvdXRlcy4gSWYgc29tZW9uZSB3aXNoZXMKdG8gcGVyZm9ybSBnZW5lcmF0aW9uIGF0IGJ1aWxkIHRpbWUsIGEgcG9zc2libGUgc29sdXRpb24gaXMgdG8gZ2VuZXJhdGUgdGhlIHJvdXRlIGluIGEgdW5pdCB0ZXN0IGFuZCBjcmVhdGUgdGhlIHNwZWMgZHVyaW5nIHRlc3QgZXhlY3V0aW9uLgoyLiBFYWNoIHZlcnR4IHJvdXRlIHRoYXQgbmVlZHMgdG8gdGFrZSBwYXJ0IGluIHRoZSBzcGVjIG11c3QgY29udGFpbiBhdCBtb3N0IG9uZSBoYW5kbGVyIHRoYXQgZGVsZWdhdGVzIHRvIGEgbWV0aG9kIHRoYXQgaXMgZGVjb3JhdGVkIHdpdGgKYW4gJ2lvLnN3YWdnZXIudjMub2FzLmFubm90YXRpb25zLk9wZXJhdGlvbicgYW5ub3RhdGlvbi4gVGhlIGhhbmRsZXIgaXRzZWxmIG11c3Qgbm90IGNsb3NlIGluIG5vdGhpbmcgYnV0IHRoZSByb3V0aW5nIGNvbnRleHQuIFRoaXMgd2lsbCBhaWQKdGhlIGludHJvc3BlY3Rpb24gb2YgdGhlIGFjdHVhbCB0eXBlIGxvY2F0ZWQgaW4gdGhlIEpWTSBzeW50aGV0aWMgY2xhc3Mgb2YgdGhlIGhhbmRsZXIgd2hpY2ggYWN0dWFsbHkgY29udGFpbnMgdGhlIGFubm90YXRlZCBtZXRob2QuClRoYXQgaW1wbGllcyB0aGF0IGEgcm91dGUgY2FuIGhhdmUgbWFueSBpbnRlcm1lZGlhdGUgaGFuZGxlcnMgZGVwZW5kaW5nIG9uIHRoZSB1c2UgY2FzZSBhbmQgc3RpbGwgZXhwcmVzcyB0aGUgb3ZlcmFsbCByZXN1bHQgdGhyb3VnaCBhbgonaW8uc3dhZ2dlci52My5vYXMuYW5ub3RhdGlvbnMuT3BlcmF0aW9uJyBhbm5vdGF0aW9uIGZyb20gdGhlIHNlbGVjdGVkIGhhbmRsZXIuCjMuIEFsbCBpby5zd2FnZ2VyLnYzLm9hcy5hbm5vdGF0aW9ucy5PcGVyYXRpb24nIGFubm90YXRpb25zIG11c3QgY29udGFpbiB0aGUgJ21ldGhvZCcgYXR0cmlidXRlIGluIG9yZGVyIHRvIGJlIGFibGUgdG8gYmUgY3Jvc3MtbWF0Y2hlZCB3aXRoIHRoZSByb3V0ZQpkZWZpbml0aW9uIChlLmcgcm91dGUuaGVhZCguLi4uKSAtPiBhbm5vdGF0aW9uIGluIG1ldGhvZCBtdXN0IGhhdmUgdGhlICdIRUFEJyB2YWx1ZSkKNC4gVGhlIGdlbmVyYXRvciBhdCB0aGUgbW9tZW50IHRyaWVzIHRvIGZvY3VzIG9uIE9wZXJhdGlvbnMgYW5kIGp1c3QgcHJvdmlkZSB0aGUgYmFzaWNzIGZvciBvdGhlciBwYXJ0cyBvZiB0aGUgc3BlYyAoSW5mbywgQ29udGFjdHMsIFNlcnZlcykgc2luY2UKbW9zdCBvZiB0aGVtIGFyZSBjb25maWd1cmF0aW9uIHNwZWNpZmljLgo1LiBNYXBwaW5ncyBiZXR3ZWVuIGFubm90YXRpb25zIGFuZCBPcGVuQVBJIDMgbW9kZWwgYXJlIG1pc3NpbmcgYnV0IHRoZSBvdmVyYWxsIGNvbmNlcHQgbWFrZXMgZXh0ZW5zaW9ucyBlYXNpbHkgYWRqdXN0ZWQuCjYuIE5PIEpBWC1SUyBhbm5vdGF0aW9ucyBtdXN0IGJlIHVzZWQuCgojIEV4YW1wbGUgQ2FzZQpBbiBhY3R1YWwgYXBwbGljYXRpb24gb2YgdGhlc2UgZnVuY3Rpb25zIGNhbiBiZSBmb3VuZCBhdDoKCmh0dHBzOi8vZ2l0aHViLmNvbS9ja2FyYXR6YXMvdHVzLXNlcnZlci1pbXBsZW1lbnRhdGlvbi9ibG9iL21hc3Rlci9zcmMvbWFpbi9qYXZhL2NvbS90dXMvb3NzL3NlcnZlci9jb3JlL1NlcnZlclZlcnRpY2xlLmphdmEgZm9yIGdlbmVyYXRpbmcgdGhlIHNwZWMKYW5kCmh0dHBzOi8vZ2l0aHViLmNvbS9ja2FyYXR6YXMvdHVzLXNlcnZlci1pbXBsZW1lbnRhdGlvbi9ibG9iL21hc3Rlci9zcmMvbWFpbi9qYXZhL2NvbS90dXMvb3NzL3NlcnZlci9jb3JlLypIYW5kbGVyLmphdmEgZm9yIHRoZSB1c2FnZSBvZiBhbm5vdGF0aW9ucyBpbiBoYW5kbGVycy4KCiMgQ3VycmVudCBTdGF0dXMKQnkgbm8gbWVhbnMgdGhlIHdob2xlIE9wZW5BUEkgMyBzcGVjIGlzIGNvdmVyZWQuIFRoZSBjdXJyZW50IGNvZGViYXNlIGNhbiBiZSBleHRlbmRlZCBiYXNlZCBvbiB0aGUgYWN0dWFsIHVzZSBjYXNlcyBhbmQgc3VwcG9ydCBtb3JlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbi4KVGhpcyByZXBvc2l0b3J5IGNhbiBiZSB1c2VkIGFzIGEgYmFzaXMgdG8gaW5zcGlyZSB0aGUgZGVzaWduIG9mIG9mZmljaWFsIHZlcnR4ICJvcGVuYXBpLWVuYWJsZWQiIHZlcnR4IHJvdXRlcyBpbiBvcmRlciB0byBtYWtlIHRoZSBpbnRyb3NwZWN0aW9uIGVhc2llciBhbmQKbW9yZSBlZmZlY3RpdmUuIAogCgo= readmeEtag: '"1e9c971aabf2a134539bc135ce13e7c0bc062014"' readmeLastModified: Fri, 16 Feb 2018 14:55:56 GMT repositoryId: 120278582 description: Runtime Generation of OpenApi 3 specification from vertx routes. created: '2018-02-05T08:50:42Z' updated: '2024-12-20T07:21:18Z' language: Java archived: false stars: 10 watchers: 1 forks: 5 owner: ckaratzas logo: https://avatars.githubusercontent.com/u/35725079?v=4 license: MIT repoEtag: '"401f941b18da3518720b203a1ab46b0eda797468e57df95faac4112b7de2a6ee"' repoLastModified: Fri, 20 Dec 2024 07:21:18 GMT foundInMaster: true category: - Server - Parsers id: a791ae1cc5890604063158dd13b67264 - source: openapi3 tags repository: https://github.com/elasticemail/elasticemail-python v3: true id: 04c0888f3dcc21c80cfe5e2ae425417f repositoryMetadata: base64Readme: >- # ElasticEmail
This API is based on the REST API architecture, allowing the user to easily manage their data with this resource-based approach.

Every API call is established on which specific request type (GET, POST, PUT, DELETE) will be used.

The API has a limit of 20 concurrent connections and a hard timeout of 600 seconds per request.

To start using this API, you will need your Access Token (available [here](https://app.elasticemail.com/marketing/settings/new/manage-api)). Remember to keep it safe. Required access levels are listed in the given request’s description.

Downloadable library clients can be found in our Github repository [here](https://github.com/ElasticEmail/elasticemail-python)

This Python package is automatically generated by the [OpenAPI Generator](https://openapi-generator.tech) project:

- API version: 4.0.0
- Package version: 4.1.4
- Generator version: 7.5.0
- Build package: org.openapitools.codegen.languages.PythonClientCodegen

## Requirements.

Python 3.7+

## Installation & Usage
### pip install

If the python package is hosted on a repository, you can install directly using:

```sh
pip install git+https://github.com/elasticemail/elasticemail-python.git
```
(you may need to run `pip` with root permission: `sudo pip install git+https://github.com/elasticemail/elasticemail-python.git`)

Then import the package:
```python
import ElasticEmail
```

### Setuptools

Install via [Setuptools](http://pypi.python.org/pypi/setuptools).

```sh
python setup.py install --user
```
(or `sudo python setup.py install` to install the package for all users)

Then import the package:
```python
import ElasticEmail
```

### Tests

Execute `pytest` to run the tests.

## Getting Started

Please follow the [installation procedure](#installation--usage) and then run the following:

```python

import ElasticEmail
from ElasticEmail.rest import ApiException
from pprint import pprint

# Defining the host is optional and defaults to https://api.elasticemail.com/v4
# See configuration.py for a list of all supported configuration parameters.
configuration = ElasticEmail.Configuration(
    host = "https://api.elasticemail.com/v4"
)

# The client must configure the authentication and authorization parameters
# in accordance with the API server security policy.
# Examples for each auth method are provided below, use the example that
# satisfies your auth use case.

# Configure API key authorization: apikey
configuration.api_key['apikey'] = os.environ["API_KEY"]

# Uncomment below to setup prefix (e.g. Bearer) for API key, if needed
# configuration.api_key_prefix['apikey'] = 'Bearer'


# Enter a context with an instance of the API client
with ElasticEmail.ApiClient(configuration) as api_client:
    # Create an instance of the API class
    api_instance = ElasticEmail.CampaignsApi(api_client)
    name = 'name_example' # str | Name of Campaign to delete

    try:
        # Delete Campaign
        api_instance.campaigns_by_name_delete(name)
    except ApiException as e:
        print("Exception when calling CampaignsApi->campaigns_by_name_delete: %s\n" % e)

```


## Examples

Function ||
------------ | ------------- 
[addCampaign](examples/functions/addCampaign.py) | [readme](examples/functions/addCampaign.md)
[addContacts](examples/functions/addContacts.py) | [readme](examples/functions/addContacts.md)
[addList](examples/functions/addList.py) | [readme](examples/functions/addList.md)
[addTemplate](examples/functions/addTemplate.py) | [readme](examples/functions/addTemplate.md)
[deleteCampaign](examples/functions/deleteCampaign.py) | [readme](examples/functions/deleteCampaign.md)
[deleteContacts](examples/functions/deleteContacts.py) | [readme](examples/functions/deleteContacts.md)
[deleteList](examples/functions/deleteList.py) | [readme](examples/functions/deleteList.md)
[deleteTemplate](examples/functions/deleteTemplate.py) | [readme](examples/functions/deleteTemplate.md)
[exportContacts](examples/functions/exportContacts.py) | [readme](examples/functions/exportContacts.md)
[loadCampaign](examples/functions/loadCampaign.py) | [readme](examples/functions/loadCampaign.md)
[loadCampaignsStats](examples/functions/loadCampaignsStats.py) | [readme](examples/functions/loadCampaignsStats.md)
[loadChannelsStats](examples/functions/loadChannelsStats.py) | [readme](examples/functions/loadChannelsStats.md)
[loadList](examples/functions/loadList.py) | [readme](examples/functions/loadList.md)
[loadStatistics](examples/functions/loadStatistics.py) | [readme](examples/functions/loadStatistics.md)
[loadTemplate](examples/functions/loadTemplate.py) | [readme](examples/functions/loadTemplate.md)
[sendBulkEmails](examples/functions/sendBulkEmails.py) | [readme](examples/functions/sendBulkEmails.md)
[sendTransactionalEmails](examples/functions/sendTransactionalEmails.py) | [readme](examples/functions/sendTransactionalEmails.md)
[updateCampaign](examples/functions/updateCampaign.py) | [readme](examples/functions/updateCampaign.md)
[uploadContacts](examples/functions/uploadContacts.py) | [readme](examples/functions/uploadContacts.md)

## Documentation for API Endpoints

All URIs are relative to *https://api.elasticemail.com/v4*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*CampaignsApi* | [**campaigns_by_name_delete**](docs/CampaignsApi.md#campaigns_by_name_delete) | **DELETE** /campaigns/{name} | Delete Campaign
*CampaignsApi* | [**campaigns_by_name_get**](docs/CampaignsApi.md#campaigns_by_name_get) | **GET** /campaigns/{name} | Load Campaign
*CampaignsApi* | [**campaigns_by_name_pause_put**](docs/CampaignsApi.md#campaigns_by_name_pause_put) | **PUT** /campaigns/{name}/pause | Pause Campaign
*CampaignsApi* | [**campaigns_by_name_put**](docs/CampaignsApi.md#campaigns_by_name_put) | **PUT** /campaigns/{name} | Update Campaign
*CampaignsApi* | [**campaigns_get**](docs/CampaignsApi.md#campaigns_get) | **GET** /campaigns | Load Campaigns
*CampaignsApi* | [**campaigns_post**](docs/CampaignsApi.md#campaigns_post) | **POST** /campaigns | Add Campaign
*ContactsApi* | [**contacts_by_email_delete**](docs/ContactsApi.md#contacts_by_email_delete) | **DELETE** /contacts/{email} | Delete Contact
*ContactsApi* | [**contacts_by_email_get**](docs/ContactsApi.md#contacts_by_email_get) | **GET** /contacts/{email} | Load Contact
*ContactsApi* | [**contacts_by_email_put**](docs/ContactsApi.md#contacts_by_email_put) | **PUT** /contacts/{email} | Update Contact
*ContactsApi* | [**contacts_delete_post**](docs/ContactsApi.md#contacts_delete_post) | **POST** /contacts/delete | Delete Contacts Bulk
*ContactsApi* | [**contacts_export_by_id_status_get**](docs/ContactsApi.md#contacts_export_by_id_status_get) | **GET** /contacts/export/{id}/status | Check Export Status
*ContactsApi* | [**contacts_export_post**](docs/ContactsApi.md#contacts_export_post) | **POST** /contacts/export | Export Contacts
*ContactsApi* | [**contacts_get**](docs/ContactsApi.md#contacts_get) | **GET** /contacts | Load Contacts
*ContactsApi* | [**contacts_import_post**](docs/ContactsApi.md#contacts_import_post) | **POST** /contacts/import | Upload Contacts
*ContactsApi* | [**contacts_post**](docs/ContactsApi.md#contacts_post) | **POST** /contacts | Add Contact
*DomainsApi* | [**domains_by_domain_delete**](docs/DomainsApi.md#domains_by_domain_delete) | **DELETE** /domains/{domain} | Delete Domain
*DomainsApi* | [**domains_by_domain_get**](docs/DomainsApi.md#domains_by_domain_get) | **GET** /domains/{domain} | Load Domain
*DomainsApi* | [**domains_by_domain_put**](docs/DomainsApi.md#domains_by_domain_put) | **PUT** /domains/{domain} | Update Domain
*DomainsApi* | [**domains_by_domain_restricted_get**](docs/DomainsApi.md#domains_by_domain_restricted_get) | **GET** /domains/{domain}/restricted | Check for domain restriction
*DomainsApi* | [**domains_by_domain_verification_put**](docs/DomainsApi.md#domains_by_domain_verification_put) | **PUT** /domains/{domain}/verification | Verify Domain
*DomainsApi* | [**domains_by_email_default_patch**](docs/DomainsApi.md#domains_by_email_default_patch) | **PATCH** /domains/{email}/default | Set Default
*DomainsApi* | [**domains_get**](docs/DomainsApi.md#domains_get) | **GET** /domains | Load Domains
*DomainsApi* | [**domains_post**](docs/DomainsApi.md#domains_post) | **POST** /domains | Add Domain
*EmailsApi* | [**emails_by_msgid_view_get**](docs/EmailsApi.md#emails_by_msgid_view_get) | **GET** /emails/{msgid}/view | View Email
*EmailsApi* | [**emails_by_transactionid_status_get**](docs/EmailsApi.md#emails_by_transactionid_status_get) | **GET** /emails/{transactionid}/status | Get Status
*EmailsApi* | [**emails_mergefile_post**](docs/EmailsApi.md#emails_mergefile_post) | **POST** /emails/mergefile | Send Bulk Emails CSV
*EmailsApi* | [**emails_post**](docs/EmailsApi.md#emails_post) | **POST** /emails | Send Bulk Emails
*EmailsApi* | [**emails_transactional_post**](docs/EmailsApi.md#emails_transactional_post) | **POST** /emails/transactional | Send Transactional Email
*EventsApi* | [**events_by_transactionid_get**](docs/EventsApi.md#events_by_transactionid_get) | **GET** /events/{transactionid} | Load Email Events
*EventsApi* | [**events_channels_by_name_export_post**](docs/EventsApi.md#events_channels_by_name_export_post) | **POST** /events/channels/{name}/export | Export Channel Events
*EventsApi* | [**events_channels_by_name_get**](docs/EventsApi.md#events_channels_by_name_get) | **GET** /events/channels/{name} | Load Channel Events
*EventsApi* | [**events_channels_export_by_id_status_get**](docs/EventsApi.md#events_channels_export_by_id_status_get) | **GET** /events/channels/export/{id}/status | Check Channel Export Status
*EventsApi* | [**events_export_by_id_status_get**](docs/EventsApi.md#events_export_by_id_status_get) | **GET** /events/export/{id}/status | Check Export Status
*EventsApi* | [**events_export_post**](docs/EventsApi.md#events_export_post) | **POST** /events/export | Export Events
*EventsApi* | [**events_get**](docs/EventsApi.md#events_get) | **GET** /events | Load Events
*FilesApi* | [**files_by_name_delete**](docs/FilesApi.md#files_by_name_delete) | **DELETE** /files/{name} | Delete File
*FilesApi* | [**files_by_name_get**](docs/FilesApi.md#files_by_name_get) | **GET** /files/{name} | Download File
*FilesApi* | [**files_by_name_info_get**](docs/FilesApi.md#files_by_name_info_get) | **GET** /files/{name}/info | Load File Details
*FilesApi* | [**files_get**](docs/FilesApi.md#files_get) | **GET** /files | List Files
*FilesApi* | [**files_post**](docs/FilesApi.md#files_post) | **POST** /files | Upload File
*InboundRouteApi* | [**inboundroute_by_id_delete**](docs/InboundRouteApi.md#inboundroute_by_id_delete) | **DELETE** /inboundroute/{id} | Delete Route
*InboundRouteApi* | [**inboundroute_by_id_get**](docs/InboundRouteApi.md#inboundroute_by_id_get) | **GET** /inboundroute/{id} | Get Route
*InboundRouteApi* | [**inboundroute_by_id_put**](docs/InboundRouteApi.md#inboundroute_by_id_put) | **PUT** /inboundroute/{id} | Update Route
*InboundRouteApi* | [**inboundroute_get**](docs/InboundRouteApi.md#inboundroute_get) | **GET** /inboundroute | Get Routes
*InboundRouteApi* | [**inboundroute_order_put**](docs/InboundRouteApi.md#inboundroute_order_put) | **PUT** /inboundroute/order | Update Sorting
*InboundRouteApi* | [**inboundroute_post**](docs/InboundRouteApi.md#inboundroute_post) | **POST** /inboundroute | Create Route
*ListsApi* | [**lists_by_listname_contacts_get**](docs/ListsApi.md#lists_by_listname_contacts_get) | **GET** /lists/{listname}/contacts | Load Contacts in List
*ListsApi* | [**lists_by_name_contacts_post**](docs/ListsApi.md#lists_by_name_contacts_post) | **POST** /lists/{name}/contacts | Add Contacts to List
*ListsApi* | [**lists_by_name_contacts_remove_post**](docs/ListsApi.md#lists_by_name_contacts_remove_post) | **POST** /lists/{name}/contacts/remove | Remove Contacts from List
*ListsApi* | [**lists_by_name_delete**](docs/ListsApi.md#lists_by_name_delete) | **DELETE** /lists/{name} | Delete List
*ListsApi* | [**lists_by_name_get**](docs/ListsApi.md#lists_by_name_get) | **GET** /lists/{name} | Load List
*ListsApi* | [**lists_by_name_put**](docs/ListsApi.md#lists_by_name_put) | **PUT** /lists/{name} | Update List
*ListsApi* | [**lists_get**](docs/ListsApi.md#lists_get) | **GET** /lists | Load Lists
*ListsApi* | [**lists_post**](docs/ListsApi.md#lists_post) | **POST** /lists | Add List
*SecurityApi* | [**security_apikeys_by_name_delete**](docs/SecurityApi.md#security_apikeys_by_name_delete) | **DELETE** /security/apikeys/{name} | Delete ApiKey
*SecurityApi* | [**security_apikeys_by_name_get**](docs/SecurityApi.md#security_apikeys_by_name_get) | **GET** /security/apikeys/{name} | Load ApiKey
*SecurityApi* | [**security_apikeys_by_name_put**](docs/SecurityApi.md#security_apikeys_by_name_put) | **PUT** /security/apikeys/{name} | Update ApiKey
*SecurityApi* | [**security_apikeys_get**](docs/SecurityApi.md#security_apikeys_get) | **GET** /security/apikeys | List ApiKeys
*SecurityApi* | [**security_apikeys_post**](docs/SecurityApi.md#security_apikeys_post) | **POST** /security/apikeys | Add ApiKey
*SecurityApi* | [**security_smtp_by_name_delete**](docs/SecurityApi.md#security_smtp_by_name_delete) | **DELETE** /security/smtp/{name} | Delete SMTP Credential
*SecurityApi* | [**security_smtp_by_name_get**](docs/SecurityApi.md#security_smtp_by_name_get) | **GET** /security/smtp/{name} | Load SMTP Credential
*SecurityApi* | [**security_smtp_by_name_put**](docs/SecurityApi.md#security_smtp_by_name_put) | **PUT** /security/smtp/{name} | Update SMTP Credential
*SecurityApi* | [**security_smtp_get**](docs/SecurityApi.md#security_smtp_get) | **GET** /security/smtp | List SMTP Credentials
*SecurityApi* | [**security_smtp_post**](docs/SecurityApi.md#security_smtp_post) | **POST** /security/smtp | Add SMTP Credential
*SegmentsApi* | [**segments_by_name_delete**](docs/SegmentsApi.md#segments_by_name_delete) | **DELETE** /segments/{name} | Delete Segment
*SegmentsApi* | [**segments_by_name_get**](docs/SegmentsApi.md#segments_by_name_get) | **GET** /segments/{name} | Load Segment
*SegmentsApi* | [**segments_by_name_put**](docs/SegmentsApi.md#segments_by_name_put) | **PUT** /segments/{name} | Update Segment
*SegmentsApi* | [**segments_get**](docs/SegmentsApi.md#segments_get) | **GET** /segments | Load Segments
*SegmentsApi* | [**segments_post**](docs/SegmentsApi.md#segments_post) | **POST** /segments | Add Segment
*StatisticsApi* | [**statistics_campaigns_by_name_get**](docs/StatisticsApi.md#statistics_campaigns_by_name_get) | **GET** /statistics/campaigns/{name} | Load Campaign Stats
*StatisticsApi* | [**statistics_campaigns_get**](docs/StatisticsApi.md#statistics_campaigns_get) | **GET** /statistics/campaigns | Load Campaigns Stats
*StatisticsApi* | [**statistics_channels_by_name_get**](docs/StatisticsApi.md#statistics_channels_by_name_get) | **GET** /statistics/channels/{name} | Load Channel Stats
*StatisticsApi* | [**statistics_channels_get**](docs/StatisticsApi.md#statistics_channels_get) | **GET** /statistics/channels | Load Channels Stats
*StatisticsApi* | [**statistics_get**](docs/StatisticsApi.md#statistics_get) | **GET** /statistics | Load Statistics
*SubAccountsApi* | [**subaccounts_by_email_credits_patch**](docs/SubAccountsApi.md#subaccounts_by_email_credits_patch) | **PATCH** /subaccounts/{email}/credits | Add, Subtract Email Credits
*SubAccountsApi* | [**subaccounts_by_email_delete**](docs/SubAccountsApi.md#subaccounts_by_email_delete) | **DELETE** /subaccounts/{email} | Delete SubAccount
*SubAccountsApi* | [**subaccounts_by_email_get**](docs/SubAccountsApi.md#subaccounts_by_email_get) | **GET** /subaccounts/{email} | Load SubAccount
*SubAccountsApi* | [**subaccounts_by_email_settings_email_put**](docs/SubAccountsApi.md#subaccounts_by_email_settings_email_put) | **PUT** /subaccounts/{email}/settings/email | Update SubAccount Email Settings
*SubAccountsApi* | [**subaccounts_get**](docs/SubAccountsApi.md#subaccounts_get) | **GET** /subaccounts | Load SubAccounts
*SubAccountsApi* | [**subaccounts_post**](docs/SubAccountsApi.md#subaccounts_post) | **POST** /subaccounts | Add SubAccount
*SuppressionsApi* | [**suppressions_bounces_get**](docs/SuppressionsApi.md#suppressions_bounces_get) | **GET** /suppressions/bounces | Get Bounce List
*SuppressionsApi* | [**suppressions_bounces_import_post**](docs/SuppressionsApi.md#suppressions_bounces_import_post) | **POST** /suppressions/bounces/import | Add Bounces Async
*SuppressionsApi* | [**suppressions_bounces_post**](docs/SuppressionsApi.md#suppressions_bounces_post) | **POST** /suppressions/bounces | Add Bounces
*SuppressionsApi* | [**suppressions_by_email_delete**](docs/SuppressionsApi.md#suppressions_by_email_delete) | **DELETE** /suppressions/{email} | Delete Suppression
*SuppressionsApi* | [**suppressions_by_email_get**](docs/SuppressionsApi.md#suppressions_by_email_get) | **GET** /suppressions/{email} | Get Suppression
*SuppressionsApi* | [**suppressions_complaints_get**](docs/SuppressionsApi.md#suppressions_complaints_get) | **GET** /suppressions/complaints | Get Complaints List
*SuppressionsApi* | [**suppressions_complaints_import_post**](docs/SuppressionsApi.md#suppressions_complaints_import_post) | **POST** /suppressions/complaints/import | Add Complaints Async
*SuppressionsApi* | [**suppressions_complaints_post**](docs/SuppressionsApi.md#suppressions_complaints_post) | **POST** /suppressions/complaints | Add Complaints
*SuppressionsApi* | [**suppressions_get**](docs/SuppressionsApi.md#suppressions_get) | **GET** /suppressions | Get Suppressions
*SuppressionsApi* | [**suppressions_unsubscribes_get**](docs/SuppressionsApi.md#suppressions_unsubscribes_get) | **GET** /suppressions/unsubscribes | Get Unsubscribes List
*SuppressionsApi* | [**suppressions_unsubscribes_import_post**](docs/SuppressionsApi.md#suppressions_unsubscribes_import_post) | **POST** /suppressions/unsubscribes/import | Add Unsubscribes Async
*SuppressionsApi* | [**suppressions_unsubscribes_post**](docs/SuppressionsApi.md#suppressions_unsubscribes_post) | **POST** /suppressions/unsubscribes | Add Unsubscribes
*TemplatesApi* | [**templates_by_name_delete**](docs/TemplatesApi.md#templates_by_name_delete) | **DELETE** /templates/{name} | Delete Template
*TemplatesApi* | [**templates_by_name_get**](docs/TemplatesApi.md#templates_by_name_get) | **GET** /templates/{name} | Load Template
*TemplatesApi* | [**templates_by_name_put**](docs/TemplatesApi.md#templates_by_name_put) | **PUT** /templates/{name} | Update Template
*TemplatesApi* | [**templates_get**](docs/TemplatesApi.md#templates_get) | **GET** /templates | Load Templates
*TemplatesApi* | [**templates_post**](docs/TemplatesApi.md#templates_post) | **POST** /templates | Add Template
*VerificationsApi* | [**verifications_by_email_delete**](docs/VerificationsApi.md#verifications_by_email_delete) | **DELETE** /verifications/{email} | Delete Email Verification Result
*VerificationsApi* | [**verifications_by_email_get**](docs/VerificationsApi.md#verifications_by_email_get) | **GET** /verifications/{email} | Get Email Verification Result
*VerificationsApi* | [**verifications_by_email_post**](docs/VerificationsApi.md#verifications_by_email_post) | **POST** /verifications/{email} | Verify Email
*VerificationsApi* | [**verifications_files_by_id_delete**](docs/VerificationsApi.md#verifications_files_by_id_delete) | **DELETE** /verifications/files/{id} | Delete File Verification Result
*VerificationsApi* | [**verifications_files_by_id_result_download_get**](docs/VerificationsApi.md#verifications_files_by_id_result_download_get) | **GET** /verifications/files/{id}/result/download | Download File Verification Result
*VerificationsApi* | [**verifications_files_by_id_result_get**](docs/VerificationsApi.md#verifications_files_by_id_result_get) | **GET** /verifications/files/{id}/result | Get Detailed File Verification Result
*VerificationsApi* | [**verifications_files_by_id_verification_post**](docs/VerificationsApi.md#verifications_files_by_id_verification_post) | **POST** /verifications/files/{id}/verification | Start verification
*VerificationsApi* | [**verifications_files_post**](docs/VerificationsApi.md#verifications_files_post) | **POST** /verifications/files | Upload File with Emails
*VerificationsApi* | [**verifications_files_result_get**](docs/VerificationsApi.md#verifications_files_result_get) | **GET** /verifications/files/result | Get Files Verification Results
*VerificationsApi* | [**verifications_get**](docs/VerificationsApi.md#verifications_get) | **GET** /verifications | Get Emails Verification Results


## Documentation For Models

 - [AccessLevel](docs/AccessLevel.md)
 - [AccountStatusEnum](docs/AccountStatusEnum.md)
 - [ApiKey](docs/ApiKey.md)
 - [ApiKeyPayload](docs/ApiKeyPayload.md)
 - [BodyContentType](docs/BodyContentType.md)
 - [BodyPart](docs/BodyPart.md)
 - [Campaign](docs/Campaign.md)
 - [CampaignOptions](docs/CampaignOptions.md)
 - [CampaignRecipient](docs/CampaignRecipient.md)
 - [CampaignStatus](docs/CampaignStatus.md)
 - [CampaignTemplate](docs/CampaignTemplate.md)
 - [CertificateValidationStatus](docs/CertificateValidationStatus.md)
 - [ChannelLogStatusSummary](docs/ChannelLogStatusSummary.md)
 - [CompressionFormat](docs/CompressionFormat.md)
 - [ConsentData](docs/ConsentData.md)
 - [ConsentTracking](docs/ConsentTracking.md)
 - [Contact](docs/Contact.md)
 - [ContactActivity](docs/ContactActivity.md)
 - [ContactPayload](docs/ContactPayload.md)
 - [ContactSource](docs/ContactSource.md)
 - [ContactStatus](docs/ContactStatus.md)
 - [ContactUpdatePayload](docs/ContactUpdatePayload.md)
 - [ContactsList](docs/ContactsList.md)
 - [DeliveryOptimizationType](docs/DeliveryOptimizationType.md)
 - [DomainData](docs/DomainData.md)
 - [DomainDetail](docs/DomainDetail.md)
 - [DomainOwner](docs/DomainOwner.md)
 - [DomainPayload](docs/DomainPayload.md)
 - [DomainUpdatePayload](docs/DomainUpdatePayload.md)
 - [EmailContent](docs/EmailContent.md)
 - [EmailData](docs/EmailData.md)
 - [EmailJobFailedStatus](docs/EmailJobFailedStatus.md)
 - [EmailJobStatus](docs/EmailJobStatus.md)
 - [EmailMessageData](docs/EmailMessageData.md)
 - [EmailPredictedValidationStatus](docs/EmailPredictedValidationStatus.md)
 - [EmailRecipient](docs/EmailRecipient.md)
 - [EmailSend](docs/EmailSend.md)
 - [EmailStatus](docs/EmailStatus.md)
 - [EmailTransactionalMessageData](docs/EmailTransactionalMessageData.md)
 - [EmailValidationResult](docs/EmailValidationResult.md)
 - [EmailValidationStatus](docs/EmailValidationStatus.md)
 - [EmailView](docs/EmailView.md)
 - [EmailsPayload](docs/EmailsPayload.md)
 - [EncodingType](docs/EncodingType.md)
 - [EventType](docs/EventType.md)
 - [EventsOrderBy](docs/EventsOrderBy.md)
 - [ExportFileFormats](docs/ExportFileFormats.md)
 - [ExportLink](docs/ExportLink.md)
 - [ExportStatus](docs/ExportStatus.md)
 - [FileInfo](docs/FileInfo.md)
 - [FilePayload](docs/FilePayload.md)
 - [FileUploadResult](docs/FileUploadResult.md)
 - [InboundPayload](docs/InboundPayload.md)
 - [InboundRoute](docs/InboundRoute.md)
 - [InboundRouteActionType](docs/InboundRouteActionType.md)
 - [InboundRouteFilterType](docs/InboundRouteFilterType.md)
 - [ListPayload](docs/ListPayload.md)
 - [ListUpdatePayload](docs/ListUpdatePayload.md)
 - [LogJobStatus](docs/LogJobStatus.md)
 - [LogStatusSummary](docs/LogStatusSummary.md)
 - [MergeEmailPayload](docs/MergeEmailPayload.md)
 - [MessageAttachment](docs/MessageAttachment.md)
 - [MessageCategory](docs/MessageCategory.md)
 - [MessageCategoryEnum](docs/MessageCategoryEnum.md)
 - [NewApiKey](docs/NewApiKey.md)
 - [NewSmtpCredentials](docs/NewSmtpCredentials.md)
 - [Options](docs/Options.md)
 - [RecipientEvent](docs/RecipientEvent.md)
 - [Segment](docs/Segment.md)
 - [SegmentPayload](docs/SegmentPayload.md)
 - [SmtpCredentials](docs/SmtpCredentials.md)
 - [SmtpCredentialsPayload](docs/SmtpCredentialsPayload.md)
 - [SortOrderItem](docs/SortOrderItem.md)
 - [SplitOptimizationType](docs/SplitOptimizationType.md)
 - [SplitOptions](docs/SplitOptions.md)
 - [SubAccountInfo](docs/SubAccountInfo.md)
 - [SubaccountEmailCreditsPayload](docs/SubaccountEmailCreditsPayload.md)
 - [SubaccountEmailSettings](docs/SubaccountEmailSettings.md)
 - [SubaccountEmailSettingsPayload](docs/SubaccountEmailSettingsPayload.md)
 - [SubaccountPayload](docs/SubaccountPayload.md)
 - [SubaccountSettingsInfo](docs/SubaccountSettingsInfo.md)
 - [SubaccountSettingsInfoPayload](docs/SubaccountSettingsInfoPayload.md)
 - [Suppression](docs/Suppression.md)
 - [Template](docs/Template.md)
 - [TemplatePayload](docs/TemplatePayload.md)
 - [TemplateScope](docs/TemplateScope.md)
 - [TemplateType](docs/TemplateType.md)
 - [TrackingType](docs/TrackingType.md)
 - [TrackingValidationStatus](docs/TrackingValidationStatus.md)
 - [TransactionalRecipient](docs/TransactionalRecipient.md)
 - [Utm](docs/Utm.md)
 - [VerificationFileResult](docs/VerificationFileResult.md)
 - [VerificationFileResultDetails](docs/VerificationFileResultDetails.md)
 - [VerificationStatus](docs/VerificationStatus.md)


<a id="documentation-for-authorization"></a>
## Documentation For Authorization


Authentication schemes defined for the API:
<a id="apikey"></a>
### apikey

- **Type**: API key
- **API key parameter name**: X-ElasticEmail-ApiKey
- **Location**: HTTP header

<a id="ApiKeyAuthCustomBranding"></a>
### ApiKeyAuthCustomBranding

- **Type**: API key
- **API key parameter name**: X-Auth-Token
- **Location**: HTTP header


## Author

support@elasticemail.com


 readmeEtag: '"5a907af533228184d0f0ca4140b9af67e6415407"' readmeLastModified: Thu, 22 May 2025 20:04:13 GMT repositoryId: 366327599 description: ElasticEmail - the Python library for the Elastic Email REST API created: '2021-05-11T09:30:41Z' updated: '2026-01-15T01:50:42Z' language: Python archived: false stars: 9 watchers: 1 forks: 3 owner: ElasticEmail logo: https://avatars.githubusercontent.com/u/26925596?v=4 repoEtag: '"225c2017c964a968d62824d23c7c5f3bd19175522021a3b535ae37bcde133c1c"' repoLastModified: Thu, 15 Jan 2026 01:50:42 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/brreg/openapi v3: true repositoryMetadata: base64Readme: >- UmVwb3NpdG9yeSBmb3IgT3BlbkFQSSBzcGVzaWZpa2Fzam9uZXIgZnJhIEJyw7hubsO4eXN1bmRyZWdpc3RyZW5lCg== readmeEtag: '"db76a039c35ee8ba3e02f5735518045a2ac6e30f"' readmeLastModified: Mon, 20 Nov 2023 12:05:42 GMT repositoryId: 145829476 description: Reposository holding the API specifications of Brønnøysundregistrene created: '2018-08-23T09:10:50Z' updated: '2025-04-10T19:20:34Z' language: null archived: false stars: 11 watchers: 8 forks: 5 owner: brreg logo: https://avatars.githubusercontent.com/u/37073025?v=4 repoEtag: '"2cf1199077dfa0856ce178289d0c1f531970c42ee33501ea6a21ee6e36bfb9dd"' repoLastModified: Thu, 10 Apr 2025 19:20:34 GMT foundInMaster: true category: Documentation id: 2c7899a38f9d43d529cf65be2bd07bd6 - source: openapi3 tags repository: https://github.com/awtkns/openapi-perf v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyBzcmM9Ii4vZG9jcy9lbi9kb2NzL2Fzc2V0cy9sb2dvLWxpZ2h0LnBuZyIgYWx0PSJPcGVuQVBJIFBlcmYgTG9nbyIgLz4KPC9wPgo8aDMgYWxpZ249ImNlbnRlciIgc3R5bGU9Im1hcmdpbi1ib3R0b206IDA7IGNvbG9yOiBibGFjayI+PHN0cm9uZz5PcGVuQVBJIFBlcmY8L3N0cm9uZz48L2gzPgo8cCBhbGlnbj0iY2VudGVyIj4KICDwn6SWPGVtPiBBdXRvbWF0aWMgT3BlbkFQSSBQZXJmb3JtYW5jZSBUZXN0aW5nIDwvZW0+8J+kljwvYnI+CjwvcD4KPHAgYWxpZ249ImNlbnRlciI+CjxpbWcgYWx0PSJUZXN0cyIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vYXd0a25zL29wZW5hcGktcGVyZi93b3JrZmxvd3MvVGVzdHMvYmFkZ2Uuc3ZnIiAvPgo8aW1nIGFsdD0iRG9jcyIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vYXd0a25zL2Zhc3RhcGktY3J1ZHJvdXRlci93b3JrZmxvd3MvZG9jcy9iYWRnZS5zdmciIC8+CjxhIGhyZWY9Imh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9vcGVuYXBpLXBlcmYiIHRhcmdldD0iX2JsYW5rIj4KICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3Yvb3BlbmFwaS1wZXJmP2NvbG9yPSUyMzM0RDA1OCZsYWJlbD1weXBpJTIwcGFja2FnZSIgYWx0PSJQYWNrYWdlIHZlcnNpb24iPgo8L2E+CjxpbWcgYWx0PVB5dGhvbiBWZXJzaW9uIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9weXZlcnNpb25zL29wZW5hcGktcGVyZj9jb2xvcj0lMjMzNEQwNTgiIC8+CjwvcD4KCi0tLQoKKipEb2N1bWVudGF0aW9uKio6IDxhIGhyZWY9Imh0dHBzOi8vb3BlbmFwaS1wZXJmLmF3dGtucy5jb20iIHRhcmdldD0iX2JsYW5rIj5odHRwczovL29wZW5hcGktcGVyZi5hd3RrbnMuY29tPC9hPgoKKipTb3VyY2UgQ29kZSoqOiA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vYXd0a25zL29wZW5hcGktcGVyZiIgdGFyZ2V0PSJfYmxhbmsiPmh0dHBzOi8vZ2l0aHViLmNvbS9hd3RrbnMvb3BlbmFwaS1wZXJmPC9hPgoKKipDb250aW5vdXMgSW50ZWdyYXRpb24qKjogPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2F3dGtucy9vcGVuYXBpLXBlcmYtYWN0aW9uIiB0YXJnZXQ9Il9ibGFuayI+aHR0cHM6Ly9naXRodWIuY29tL2F3dGtucy9vcGVuYXBpLXBlcmYtYWN0aW9uPC9hPgoKLS0tCgoKVGhpcyBwcm9qZWN0IHdpbGwgbWFrZSB1c2Ugb2YgdGhlIE9wZW5BUEkgc2NoZW1hIHRvIGJ1aWxkIGFuIGF1dG9tYXRlZCBSRVNUIEFQSSBwZXJmb3JtYW5jZSB0ZXN0aW5nIGFuZCBiZW5jaG1hcmtpbmcgdG9vbC4gT3BlbkFQSSBkZWZpbmVzIGFuIGludGVyZmFjZSBmb3IgUkVTVCBBUElzIGFsbG93aW5nIGF1dG9tYXRlZCBnZW5lcmF0aW9uIG9mIGFuIEFQSSBzY2hlbWEgd2hpY2ggcHJvdmlkZXMgaW5zaWdodHMgb24gdGhlIEFQSSdzIHVzYWdlIGFuZCBleHBlY3RlZCBpbnB1dCBhbmQgcmVzcG9uc2UgdmFsdWVzLiBVc2luZyB0aGUgT3BlbkFQSSBzY2hlbWEgd2UgaG9wZSBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlIHRlc3RzIGNvdmVyaW5nIGFsbCBlbmRwb2ludHMuIFdlIHdpbGwgdXNlIHRoZXNlIGdlbmVyYXRlZCB0ZXN0cyB0byBib3RoIHRlc3QgZm9yIGVuZHBvaW50IGNvcnJlY3RuZXNzIGFuZCB0byBnYXRoZXIgcmVsZXZhbnQgcGVyZm9ybWFuY2UgbWV0cmljcyBzdWNoIGFzIHJlc3BvbnNlIHRpbWUuIFdlIHdpbGwgZmluYWxseSBnZW5lcmF0ZSBhbiBpbmZvcm1hdGl2ZSByZXBvcnQgb24gZW5kcG9pbnQgcGVyZm9ybWFuY2UgYW5kIGNvcnJlY3RuZXNzLCBhbGxvd2luZyBkZXZlbG9wZXJzIHRvIHF1aWNrbHkgZGV0ZXJtaW5lIHBvdGVudGlhbGx5IHByb2JsZW1hdGljIGVuZHBvaW50cyBhbmQgZW5kcG9pbnRzIHRoYXQgd291bGQgYmVuZWZpdCBtb3N0IGZyb20gb3B0aW1pemF0aW9uLgoKIyMgSW5zdGFsbGF0aW9uCmBgYHB5dGhvbgpwaXAgaW5zdGFsbCBvcGVuYXBpLXBlcmYKYGBgCgojIyBVc2FnZQojIyMgVGVzdCBHZW5lcmF0aW9uClRoaXMgd2lsbCBjcmVhdGUgZ2VuZXJhdGUgcHJvcGVydHktYmFzZWQgcGVyZm9ybWFuY2UgdGVzdHMgZm9yIGFuIGVuZHBvaW50IGFuZCBzYXZlIHRoZSB0ZXN0IHNjaGVtYSB0byB5b3VyIHJlc3VsdHMgZGlyZWN0b3J5LgpgYGBweXRob24KZnJvbSBvcGVuYXBpX3BlcmYgaW1wb3J0IE9wZW5BUElQZXJmCgpvcCA9IE9wZW5BUElQZXJmKAogIGVuZHBvaW50X3VybCA9ICJodHRwOi8vbG9jYWxob3N0OjUwMDAiLAogIHJlc3VsdHNfZGlyID0gIi9wYXRoL3RvL3Jlc3VsdHMvZGlyZWN0b3J5IgopCmBgYAoKWW91IGNhbiBhbHNvIGxvYWQgZXhpc3RpbmcgdGVzdHMgZnJvbSBhIHRlc3Qgc2NoZW1hIGZpbGUgbGlrZSB0aGlzOgpgYGBweXRob24Kb3AgPSBPcGVuQVBJUGVyZigKICB0ZXN0X3NjaGVtYV9wYXRoID0gInBhdGgvdG8vdGVzdF9zY2hlbWEuanNvbiIKKQpgYGAKU2NoZW1hIGZpbGVzIGNhbiBiZSBtb2RpZmllZCB0byBjb25maWd1cmUgdGVzdCBleGVjdXRpb24uCgojIyMgVGVzdCBFeGVjdXRpb24KClRvIHJ1biB0aGVzZSB0ZXN0cywgdXNlCmBgYHB5dGhvbgpvcC5ydW4oKQpgYGAKClRoaXMgd2lsbCBnZW5lcmF0ZSBhIHJlcG9ydCBwZGYgZmlsZSBpbiB5b3VyIHJlc3VsdHMgZGlyZWN0b3J5CgpGb3IgZGV0YWlsZWQgdXNhZ2UsIHJlZmVyIHRvIG91ciBbZG9jc10oaHR0cHM6Ly9vcGVuYXBpLXBlcmYuYXd0a25zLmNvbSkK readmeEtag: '"ecb6d9116981c3bb46c63a8d76fa012749a11840"' readmeLastModified: Mon, 19 Apr 2021 05:17:15 GMT repositoryId: 342141821 description: Automatic OpenAPI Endpoint Performance Testing created: '2021-02-25T06:10:54Z' updated: '2024-01-17T19:49:28Z' language: Python archived: true stars: 10 watchers: 1 forks: 2 owner: awtkns logo: https://avatars.githubusercontent.com/u/32209255?v=4 license: MIT repoEtag: '"b7a0622afb43eaec2c86a56411d901473077f4e8806be98b414e781fb650229a"' repoLastModified: Wed, 17 Jan 2024 19:49:28 GMT foundInMaster: true category: Testing id: 5334723886325cf3863cf349634cd804 - source: openapi3 tags repository: https://github.com/mtamas7/spring-boot-3.2-authentication-openapi-archetype v3: true id: bb08e69a042c27bd064906e04e4824d5 repositoryMetadata: base64Readme: >- IyBTcHJpbmcgQm9vdCBBcmNoZXR5cGUgd2l0aCBwcmVjb25maWd1cmVkIE9wZW5BcGkgYW5kIFNwcmluZyBTZWN1cml0eQoKVGhpcyBpcyBhIG1hdmVuIGFyY2hldHlwZSBwcm9qZWN0IHRvIG1ha2UgaXQgZWFzaWVyIHRvIHN0YXJ0IGEgbmV3IHByb2plY3Qgd2l0aCBwcmVjb25maWd1cmVkIFNwcmluZyBTZWN1cml0eSAoYmFzZWQgb24gSldUIHRva2VuKSBhbmQgT3BlbkFwaSB3aGljaCBpcyBmcmVlbHkgbW9kaWZpYWJsZQoKIyMgVGVjaG5vbG9neSBzdGFjawotIFNwcmluZyBCb290IDMuMi40IChodHRwOi8vc3RhcnQuc3ByaW5nLmlvKQotIEhpYmVybmF0ZSwgU3ByaW5nIERhdGEgSlBBCi0gTGlxdWliYXNlCi0gTWFwc3RydWN0Ci0gTG9tYm9rCi0gU3dhZ2dlcgotIE1hdmVuCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCiMjIyBQcmVyZXF1aXNpdGVzCgpNdXN0IGhhdmUgTWF2ZW4gMysgYW5kIEphdmEgMTcgaW5zdGFsbGVkCgpUbyBjcmVhdGUgYSBwcm9qZWN0IGZyb20gdGhlIGFyY2hldHlwZSBtdXN0IGZpcnN0IGJlIGluc3RhbGxlZCBpbiB0aGUgbG9jYWwgcmVwb3NpdG9yeS4KCkNoZWNrb3V0IHRoZSByZXBvc2l0b3J5IGFuZCBydW4KCmBgYAptdm4gY2xlYW4gaW5zdGFsbApgYGAKCiMjIyBDcmVhdGUgYSBuZXcgcHJvamVjdCBmcm9tIHRoZSBhcmNoZXR5cGUKYGBgCm12biBhcmNoZXR5cGU6Z2VuZXJhdGUgLURhcmNoZXR5cGVHcm91cElkPWNvbS5kaXZpZGUuYnkuemVyby5hcmNoZXR5cGUgLURhcmNoZXR5cGVBcnRpZmFjdElkPXNwcmluZ2Jvb3QzLjItYXV0aGVudGljYXRpb24tb3BlbmFwaS1hcmNoZXR5cGUgLURhcmNoZXR5cGVWZXJzaW9uPTEuMC4wLVNOQVBTSE9UIC1EZ3JvdXBJZD08WU9VUiBHUk9VUCBJRCBIRVJFPiAtRGFydGlmYWN0SWQ9PFlPVVIgQVJUSUZBQ1QgSUQgSEVSRT4gLUR2ZXJzaW9uPTEuMC4wLVNOQVBTSE9UIC1EYXJ0aWZhY3RJZENhbWVsQ2FzZT08WW91ckNhbWVsQ2FzZUFydGlmYWN0SWRIZXJlPgpgYGAKClRoZSBnZW5lcmF0ZWQgcHJvamVjdCBjb250YWlucyB0aHJlZSBkaWZmZXJlbnQgbW9kdWxlcyBhbmQgb25lIGRpcmVjdG9yeToKCmBgYGFwaWBgYAoKYGBgYXBwYGBgCgpgYGBzZWN1cml0eWBgYAoKYGBgZG9ja2VyYGBgCiMjIyMgQVBJCgpUaGUgQVBJIG1vZHVsZSBjb250YWlucyB0aGUgT3BlbkFwaSB5YW1sIGZpbGUgaW4gdGhlIGBgYHNyYy9tYWluL3Jlc291cmNlc2BgYCBmb2xkZXIuIENvbnRhaW5zIGV2ZXJ5dGhpbmcgbmVlZGVkIGZvciBhdXRoZW50aWNhdGlvbiAoY29udHJvbGxlciwgcmVxdWVzdHMsIHJlc3BvbnNlcykuCgojIyMjIEFQUAoKVGhpcyBBUFAgbW9kdWxlIGlzIGEgc2tlbGV0b24gZm9yIHRoZSBzcHJpbmcgYm9vdCBhcHBsaWNhdGlvbi4gVGhlIGFwcCB1c2VzIGEgcG9zdGdyZXMgZGF0YWJhc2UgdG8gc3RvcmUgdGhlIG5lY2Vzc2FyeSB1c2VyIGluZm9ybWF0aW9uIGFuZCBsaXF1aWJhc2UgdG8gbWFuYWdlIGRhdGFiYXNlIGNoYW5nZXMuIChpbmNsdWRlcyB0aGUgZGVmYXVsdCBzZXR0aW5ncyBmb3IgdXNlciBhbmQgdG9rZW4gdGFibGUpCgpEb2NrZXIgZmlsZToKClRoZSBkb2NrZXIgZGlyZWN0b3J5IGNvbnRhaW5zIGEgYGRvY2tlci1jb21wb3NlLnltbGAuIEJlZm9yZSBzdGFydCB1c2luZyB5b3UgaGF2ZSB0byBjaGFuZ2UgdGhlIGltYWdlIG5hbWUgdG8geW91ciBvd24uIAoKUmVxdWlyZWQgZW52aXJvbm1lbnQgdmFyaWFibGVzOgoKYGBgREJfVVJMYGBgIFRoaXMgaXMgdGhlIHVybCBvZiB0aGUgcG9zdGdyZXMgc2VydmVyLiBEZWZhdWx0IHZhbHVlOiBgYGBqZGJjOnBvc3RncmVzcWw6Ly9sb2NhbGhvc3Q6NTQzMi9wb3N0Z3Jlc0RCYGBgIAoKYGBgREJfVVNFUk5BTUVgYGAgVGhpcyBpcyB0aGUgdXNlcm5hbWUgb2YgdGhlIHBvc3RncmVzIHNlcnZlci4gRGVmYXVsdCB2YWx1ZTogYGBgdXNlcm5hbWVgYGAgCgpgYGBEQl9QQVNTV09SRGBgYCBUaGlzIGlzIHRoZSBwYXNzd29yZCBvZiB0aGUgcG9zdGdyZXMgc2VydmVyLiBEZWZhdWx0IHZhbHVlOiBgYGBwYXNzd29yZGBgYCAKCiMjIyMgU0VDVVJJVFkKVGhpcyBTRUNVUklUWSBtb2R1bGUgY29udGFpbnMgYWxsIHRoZSBuZWNlc3NhcnkgcHJlY29uZmlndXJlZCBjbGFzcyB0byBtYWtlIHRoZSBhcHBsaWNhdGlvbiBzZWN1cmVkIHdpdGggSldUIHRva2VuLgoKIyMgQXV0aG9ycwoKKiAqKlRhbWFzIE1lc3phcm9zKioK readmeEtag: '"a690129629a224751957ad16206666af59947c94"' readmeLastModified: Wed, 03 Apr 2024 18:28:49 GMT repositoryId: 781624892 description: Preconfigured archetype for Spring Boot 3.2.4 with Spring Security and JWT created: '2024-04-03T18:28:37Z' updated: '2025-06-26T06:03:29Z' language: Java archived: false stars: 9 watchers: 1 forks: 0 owner: mtamas7 logo: https://avatars.githubusercontent.com/u/7900458?v=4 repoEtag: '"18deab3f816da338e28494694057849fafc7325e62690aae9da45274fb654bef"' repoLastModified: Thu, 26 Jun 2025 06:03:29 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/danicc097/openapi-go-gin-postgres-sqlc v3: true id: 3b62bf7982213c373ce1fded4ae53279 repositoryMetadata: base64Readme: >- # openapi-go-gin-postgres-sqlc

[![Go Report Card](https://goreportcard.com/badge/github.com/danicc097/openapi-go-gin-postgres-sqlc)](https://goreportcard.com/report/github.com/danicc097/openapi-go-gin-postgres-sqlc)
[![GoDoc](https://pkg.go.dev/badge/github.com/danicc097/openapi-go-gin-postgres-sqlc)](https://pkg.go.dev/github.com/danicc097/openapi-go-gin-postgres-sqlc)
[![tests](https://github.com/danicc097/openapi-go-gin-postgres-sqlc/actions/workflows/tests.yaml/badge.svg)](https://github.com/danicc097/openapi-go-gin-postgres-sqlc/actions/workflows/tests.yaml)

Example full stack app with an API-first and Database-first approach with OpenAPI v3, sqlc+xo codegen,
generated backend, frontend client and validators and an unimaginative title.

## What's this for?

An example fully featured app to showcase how an OpenAPI v3 spec and database
schema become a real single source of truth at once. Any
change to it is validated and cascades down to:

- **frontend**: autogenerated complex, customizable and fully type-safe UI forms
  based on the OpenAPI spec converted to JSON
  schema. Generated API queries (`orval`). User-friendly generated client-side
  validation
  (via `react-hook-form` and `ajv` plus customized `kin-openapi` error
  messages for backend errors). Check out the frontend home page for available links.
- **backend**: generated Gin server (custom `oapi-codegen` and post-generation).
  Request and response validation (`kin-openapi`). Generated CRUD and index queries via `xo`
  and custom queries via `sqlc` by leveraging custom `xo` template generation
  that ensures compatibility. Integrated OpenID Connect client via
  `zitadel/oidc` (and dockerized authorization server for development in
  `cmd/oidc-server` based on reusable mock server with generics in
  [oidc-server](https://github.com/danicc097/oidc-server)). App events processing
  via Server Sent Events.

  Queries generated by `xo` use `pgx` exclusively and includes
  pagination, indexes, soft delete handling, extensible queries (parameterized `where`, `having` clauses), joins and
  much more (see [xo integration
  tests](./internal/repos/postgresql/xo-templates/tests/)) that will get you
  **95%** there without resorting to ad-hoc queries. Found an edge case or want additional functionality just for
  your DB needs? Just edit `xo-templates/`, since codegen is interpreted by
  `yaegi`. Additionally, by
  using `pgx`, models can be easily re-exported as usable OpenAPI schemas via
  `openapi-go` and some magic Bash and Go utility programs that keep generated
  code in order.

  Since using DB models in all backend layers is considered an anti-pattern at
  some level of model complexity, you're not tied to generated schemas from DB
  models.
  Any struct can become a spec schema via `openapi-go` struct tags and some
  simple bash code behind to grab them, with minimal caveats. See `rest`
  package. It is even possible to reference openapi spec models in your `xo`
  generated models, as long as they don't create any circular dependencies.
  Since at the last generation step we generate all spec schemas back to Go models with `oapi-codegen` to
  a shared `models` package, this is not an issue (although you'd have to
  choose what models to use, since there'll duplication for the models
  that were already generated by `openapi-go` in the first place - ideally, use the
  originals).

Additionally, it features OpenTelemetry in both browser (automatic and
manual instrumentation) and backend services (manual instrumentation) via
Jaeger, TimescaleDB and ~~Promscale~~ (discontinued, should look at options like
`mimir`).

## Makefile alternative

You get dynamic `x` function and `x` options parameters documentation _and_
autocompletion (`complete -C project project`) for
free (from your own source itself and comments)
so they're always up to date without any repetitive work: add/remove functions
and flags at will.
Custom internal autocompletion functions for each `x` function or `x` option can be
easily setup.

![](.github/autodocs.png)

All calls to `x` functions are logged (distinguishing stdout and stderr) for easier parallel execution and nested
calls tracking:

![](.github/logging.png)

And help for any `x` function is easily searchable when the app inevitably grows
with `--x-help`:

![](.github/help-x-function.png)

## Setup

### Configuration and local development

Fill in `.env.template`'s:

```bash
find . -name ".env*.template"
```

For `dev` environment, you may use `env.ci` as a starting point,
replacing `ci` with `dev` and using `REVERSE_PROXY_API_PREFIX=""` since it will run
the backend without docker containers.

Assuming a recent Ubuntu release:

```bash

sudo apt install direnv -y
direnv allow # you can also customize direnv with .envrc.local as you would a regular .envrc, see example

cp openapi-go.code-workspace.example openapi-go.code-workspace # edit as desired

project bootstrap # dependency and tools interactive installation
project recreate-shared-services
project gen
project db.initial-data
project run.all
```

For first test run:

```bash
project test.backend.setup
project test.backend
project test.frontend
```

### Tracing, monitoring...

```bash
bin/deploy-instrumentation
```

## Notes on code generation

### `xo`

Db struct mappings can be extended with SQL column comments, joined with ` && `:
- `properties`:`<p1>,<p2>,...`
  - `private`: exclude a field from JSON.
  - `not-required`: make a schema field not required.
  - `hidden`: exclude field from OpenAPI generation.
  - `refs-ignore`: generate a field whose constraints are ignored by the referenced table,
  i.e. no joins will be generated.
  - `share-ref-constraints`: for a FK column, it will generate the same M2O and M2M join fields the ref column has.
- `type`: override the type annotation with an existing spec schema. Also allows
  complex JSON columns to be encoded and decoded thanks to `pgx`.
- `cardinality`:`<O2O|M2O|M2M>` to generate/override joins explicitly. Only O2O is inferred.
- `tags`: append literal struct tag strings.

### `rest` package

Structs in `models.spec.go` will be generated via `openapi-go` and replaced in
the spec. Make sure they don't clash with existing `models` names or db tables.

### `models` package

Combines db codegen via a custom `xo` alongside spec schemas codegen for ease of use.

### CRUD + tests generation

See `project test.create-crud-gen`'s implementation for an example workflow to create a
variety of implemented CRUD endpoints and relevant tests for a given database
table.
This is a one-off script per table.

### OpenAPI spec

#### Paths

Endpoints are exclusively managed through the spec and created/kept up to date
in their respective files aggregated by `tags` (limited to one)
via codegen and AST modifications. This ensures the server always implements the
spec.

For endpoint implementation, only method bodies are to be modified.

#### Vendor extensions

Component schemas:
  - `x-gen-struct` generates a schema from a struct and keeps it up to date,
    which can be tagged with OpenAPI fields (via openapi-go).
    Possible structs to generate from are listed in `internal/codegen/structs.gen.go`
    indexed by `x-gen-struct`.
    Packages to generate structs from are added adhoc (currently rest and models packages only)
    Note that rest package structs are generated without prefix since they
    are the base of the spec - else prefix is the pascal cased package name.
    The value of `x-gen-struct` must match the schema name, to prevent duplicates and
    confusing definitions.
    To use a yet to be generated `rest` package struct in spec `paths`, simply use the `$ref`
    without creating any schema and it will be autogenerated later if it doesn't exist
    (only for `rest` package structs, since any other should not be used as request/response in the
    first place).
  - `x-spec-rest-type`: set to true to mark rest types (i.e. request, responses, query params...) that are manually defined in the spec instead of structs in `models.spec.go`.
Path operations:
  - `x-required-scopes` defines required scopes for a request.
  - `x-required-role` defines the minimum required role for a request.


<!-- xo custom templates with cardinality, property comments for join and public model generation for embedding, schema from structs, spec sync -->

## Architecture

Simplified:

![](.github/system-diagram.png)

## Known issues

- Nested functions in `project`'s `x` functions will break automatic
  documentation for that particular function due to a bug in `declare` where the last nested function line
  number is returned instead of the parent.

## TODOs

- Meaningful project name.

- System design docs/diagrams.


 readmeEtag: '"3b0e003b311213b93483d4b95fb7b8e73953c327"' readmeLastModified: Sat, 05 Oct 2024 20:49:57 GMT repositoryId: 516988488 description: >- Go API-first and Database-first demo app with OpenAPI v3 and sqlc/xo codegen, OIDC auth, React TypeScript frontend, end to end tracing with OpenTelemetry and a smart Bash repo task manager. created: '2022-07-23T07:25:51Z' updated: '2025-10-08T00:55:46Z' language: TypeScript archived: false stars: 11 watchers: 0 forks: 4 owner: danicc097 logo: https://avatars.githubusercontent.com/u/71724149?v=4 license: Apache-2.0 repoEtag: '"8b9bc1fc5511c65df9ba42ed9523c231fb029876ef6683d174145e6f960a1087"' repoLastModified: Wed, 08 Oct 2025 00:55:46 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/restful-ma/rest-ruler v3: true id: b948369127981444200364f6a085ca8d repositoryMetadata: base64Readme: >- # RESTRuler

The RESTRuler CLI is a tool that can evaluate RESTful APIs on the basis of design rule violations. These violations are based on the design rules from Mark Massé's book [REST API Design Rulebook](https://www.oreilly.com/library/view/rest-api-design/9781449317904/).
Currently, RESTRuler can parse the following Web API description languages:

* [OpenAPI v2](https://swagger.io/specification/v2/)
* [OpenAPI v3](https://github.com/OAI/OpenAPI-Specification)

RESTRuler has been developed in the [Empirical Software Engineering Group](https://www.iste.uni-stuttgart.de/ese) at the University of Stuttgart, Germany, as a research prototype written in Java (version >=18 needed).
It is a command-line tool that takes the path or URL to an OpenAPI definition file as input and displays a list of design rule violations as output.
Optionally, a Markdown report file can be generated with additional details and improvement suggestions.

## Design Rules

Descriptions of the implemented design rules can be found in our [rule documentation](./docs/rules/README.md).

## Project Architecture

A detailed description of the project and implemented components can be found in the [architecture documentation](./docs/architecture.md).

## Evaluation

You will find the artifacts related to the empirical evaluation of RESTRuler in [this repository](https://github.com/restful-ma/rest-ruler-evaluation).

## General Usage Instructions

For a quick start, you can simply download `rest-ruler.jar` for a certain [release](https://github.com/restful-ma/rest-ruler/releases).
Then execute these commands in the same folder as the JAR file (Java version >=18 needed):

```bash
# execute JAR file to display CLI parameters
java -jar rest-ruler.jar -h

# run with an example API from https://apis.guru
java -jar rest-ruler.jar -p https://api.apis.guru/v2/specs/circleci.com/v1/openapi.yaml
```

If you want to download or clone the full repository, run these commands in the root folder of the repository to build and start the tool (Java version >=18 needed):

```bash
# create JAR file
./gradlew assemble
# execute JAR file to display CLI parameters
java -jar build/libs/rest-ruler.jar -h
# run tests
./gradlew test
# test coverage (output: ./build/reports/jacoco/test/html/index.html)
./gradlew jacocoTestReport
```

## Usage Example

```bash
# run with an example API from https://apis.guru
java -jar build/libs/rest-ruler.jar -p https://api.apis.guru/v2/specs/circleci.com/v1/openapi.yaml
```

This produces the following output:

```cli
java -jar build/libs/rest-ruler.jar -p https://api.apis.guru/v2/specs/circleci.com/v1/openapi.yaml

----------------START ANALYSIS----------------
-----------------------------------------------

Begin with the analysis of the file from: https://api.apis.guru/v2/specs/circleci.com/v1/openapi.yaml

----------------------------------------------
Aug 11, 2023 3:45:01 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 1 of 15 is now checked:
CRUD function names should not be used in URIs
[==========] 100%
Aug 11, 2023 3:45:01 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 2 of 15 is now checked:
GET must be used to retrieve a representation of a resource
[==========] 100%
Aug 11, 2023 3:45:01 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 3 of 15 is now checked:
Forward slash separator (/) must be used to indicate a hierarchical relationship
[==========] 100%
Aug 11, 2023 3:45:01 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 4 of 15 is now checked:
A verb or verb phrase should be used for controller names
[==========] 100%
Aug 11, 2023 3:45:03 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 5 of 15 is now checked:
Hyphens (-) should be used to improve the readability of URIs
[==========] 100%
Aug 11, 2023 3:45:04 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 6 of 15 is now checked:
File extensions should not be included in URIs
[==========] 100%
Aug 11, 2023 3:45:04 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 7 of 15 is now checked:
GET and POST must not be used to tunnel other request methods
[==========] 100%
Aug 11, 2023 3:45:04 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 8 of 15 is now checked:
A singular noun should be used for document names
[==========] 100%
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 9 of 15 is now checked:
Description of request should match with the type of the request.
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 10 of 15 is now checked:
401 ("Unauthorized") must be used when there is a problem with the client's credentials
[==========] 100%
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 11 of 15 is now checked:
Underscores (_) should not be used in URI
[==========] 100%
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 12 of 15 is now checked:
Content-Type must be used
[==========] 100%
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 13 of 15 is now checked:
Lowercase letters should be preferred in URI paths
[==========] 100%
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 14 of 15 is now checked:
A trailing forward slash (/) should not be included in URIs
[==========] 100%
Aug 11, 2023 3:45:08 PM cli.analyzer.RestAnalyzer runRuleViolationChecks
INFO: Rule 15 of 15 is now checked:
A plural noun should be used for collection or store names
[==========] 100%
REST API Specification Report
=============================
| Line No. | Line                                                     | Rule Violated                                                                           |
| -------- | -------------------------------------------------------- | --------------------------------------------------------------------------------------- |
| 27       | /me                                                      | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 38       | /project/{username}/{project}                            | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 38       | /project/{username}/{project}                            | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 38       | /project/{username}/{project}                            | A plural noun should be used for collection or store names                              |
| 80       | /project/{username}/{project}/build-cache                | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 80       | /project/{username}/{project}/build-cache                | A plural noun should be used for collection or store names                              |
| 80       | /project/{username}/{project}/build-cache                | A verb or verb phrase should be used for controller names                               |
| 97       | /project/{username}/{project}/checkout-key               | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 97       | /project/{username}/{project}/checkout-key               | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 97       | /project/{username}/{project}/checkout-key               | A plural noun should be used for collection or store names                              |
| 128      | /project/{username}/{project}/checkout-key/{fingerprint} | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 128      | /project/{username}/{project}/checkout-key/{fingerprint} | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 128      | /project/{username}/{project}/checkout-key/{fingerprint} | A plural noun should be used for collection or store names                              |
| 154      | /project/{username}/{project}/envvar                     | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 154      | /project/{username}/{project}/envvar                     | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 154      | /project/{username}/{project}/envvar                     | A plural noun should be used for collection or store names                              |
| 154      | /project/{username}/{project}/envvar                     | Hyphens (-) should be used to improve the readability of URIs                           |
| 170      | /project/{username}/{project}/envvar/{name}              | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 170      | /project/{username}/{project}/envvar/{name}              | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 170      | /project/{username}/{project}/envvar/{name}              | A plural noun should be used for collection or store names                              |
| 197      | /project/{username}/{project}/ssh-key                    | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 197      | /project/{username}/{project}/ssh-key                    | A plural noun should be used for collection or store names                              |
| 237      | /project/{username}/{project}/tree/{branch}              | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 237      | /project/{username}/{project}/tree/{branch}              | A plural noun should be used for collection or store names                              |
| 272      | /project/{username}/{project}/{build_num}                | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 272      | /project/{username}/{project}/{build_num}                | A plural noun should be used for collection or store names                              |
| 288      | /project/{username}/{project}/{build_num}/artifacts      | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 288      | /project/{username}/{project}/{build_num}/artifacts      | A plural noun should be used for collection or store names                              |
| 303      | /project/{username}/{project}/{build_num}/cancel         | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 303      | /project/{username}/{project}/{build_num}/cancel         | A plural noun should be used for collection or store names                              |
| 303      | /project/{username}/{project}/{build_num}/cancel         | Description of request should match with the type of the request.                       |
| 318      | /project/{username}/{project}/{build_num}/retry          | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 318      | /project/{username}/{project}/{build_num}/retry          | A plural noun should be used for collection or store names                              |
| 333      | /project/{username}/{project}/{build_num}/tests          | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 333      | /project/{username}/{project}/{build_num}/tests          | A plural noun should be used for collection or store names                              |
| 333      | /project/{username}/{project}/{build_num}/tests          | Description of request should match with the type of the request.                       |
| 350      | /projects                                                | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 362      | /recent-builds                                           | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 376      | /user/heroku-key                                         | 401 ("Unauthorized") must be used when there is a problem with the client's credentials |
| 376      | /user/heroku-key                                         | Hyphens (-) should be used to improve the readability of URIs                           |
----------------------------------------------

In total 40 rule violations were found
```

## Command Line Options
| Option                                       | Description                                                                                                               | Required |
| :------------------------------------------- | :------------------------------------------------------------------------------------------------------------------------ | :------- |
| `-p $URI_PATH`,<br>`--path $URI_PATH`        | Local path or public URL to OpenAPI definition (2.0 or higher; JSON or YAML)                                              | YES      |
| `-e`,<br>`--expertMode`                      | Interactively select the rules for the analysis                                                                           | NO       |
| `-r`,<br>`--report`                          | Generate a Markdown report file with the analysis results                                                                 | NO*      |
| `-rn $FILENAME`,<br>`--reportName $FILENAME` | Specify a custom filename for the Markdown report. If this option is selected, the above option for output is not needed. | NO*      |

*If no additional output was specified, the results will only be printed to the console.

```bash
# Run with local file and no output file
java -jar build/libs/rest-ruler.jar -p path/to/openapi/definiton.json

# Run with public URL and no output file
java -jar build/libs/rest-ruler.jar -p https://www.custom.domain.com/path/to/openapi-definiton.yaml

# Run with custom filename for Markdown report
java -jar build/libs/rest-ruler.jar -p path/to/openapi/definiton.yaml -rn custom-file-name

# Run with generated filename for Markdown report
java -jar build/libs/rest-ruler.jar -p path/to/openapi/definiton.yaml -r

# Run in expert mode
java -jar build/libs/rest-ruler.jar -p path/to/openapi/definiton.json -e
```
 readmeEtag: '"98b3fab0c1b8341b37b2ef0515f5353096d78941"' readmeLastModified: Thu, 14 Mar 2024 13:17:12 GMT repositoryId: 490623497 description: >- The RESTRuler is a tool that evaluates OpenAPI definitions (version>=2.0) using design rule violations. created: '2022-05-10T09:13:14Z' updated: '2025-07-09T18:05:25Z' language: Java archived: false stars: 13 watchers: 3 forks: 2 owner: restful-ma logo: https://avatars.githubusercontent.com/u/57215045?v=4 license: Apache-2.0 repoEtag: '"66b4cf0faeeb495394a3b48ff4d92fc5409f6a3277c65720526007e8c3c7afc0"' repoLastModified: Wed, 09 Jul 2025 18:05:25 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/stackql-labs/openapi-doc-util v3: true id: c08878259bb6e3f767425401ba3faec9 repositoryMetadata: base64Readme: >- # openapi-doc-util
> Command line utility to split OpenAPI documents into smaller, self contained, service oriented documents and prepare StackQL provider interfaces
> Command line utility to help developers prepare and submit StackQL provider specs, see [StackQL Readme](https://github.com/stackql/stackql/blob/main/README.md)

## Installation

<details>
<summary>NPM</summary>
<p>

```bash
npm i @stackql/openapi-doc-util
```

</p>
</details>

<details>
<summary>YARN</summary>
<p>

```bash
yarn add @stackql/openapi-doc-util
```

</p>
</details>

## Setup

from the package directory, run:

```bash
npm link
```

## Background

<details>
<summary>Read this section for background on the StackQL product</summary>
<p>
The StackQL utility provides a SQL interface to cloud and SaaS providers, mapping a provider to an ORM, transpiling input SQL to provider API requests, and bringing back response data as a SQL based result set.  StackQL is capable of DML operations such as `INSERT` and `DELETE` which can be used to provision or de-provision cloud resources, query operations using `SELECT` to collect, analyze, report on asset or configuration data, and lifecycle operations such as starting a VM instance using the `EXEC` command in StackQL.  

The StackQL ORM provides a logical model to work with cloud resources similar to the way databases are organized into schemas.  This object/resource hierarchy is summarized below:  

```
provider/
├─ service/
│  ├─ resource/
│  │  ├─ fields
│  │  ├─ methods
```

an example would be:

```
google/
├─ compute/
│  ├─ instances/
│  │  ├─ id, name, status, ...
│  │  ├─ select, insert, delete, start, stop, ...
```

Enabling StackQL to interact with the `google` provider using SQL semantics, for example:

Provider discovery operations such as..

```sql
SHOW RESOURCES IN google.compute;
DESCRIBE google.compute.instances;
```

Query operations such as..  

```sql
SELECT status, COUNT(*) as num_instances 
FROM google.compute.instances
WHERE project = 'myproject' and zone = 'us-west-1a'
GROUP BY status;
```

Provisioning operations such as creating a Compute Engine instance using an `INSERT` statement or deprovisioning an instance using a `DELETE` statement.  
</p>
</details>

<details>
<summary>Read this section for background on the StackQL Provider Registry</summary>
<p>
StackQL provider interfaces (such as GCP, Okta, GitHub, AWS, Azure, etc) are defined using annotations to the provider API (OpenAPI) specification, these annotations or extensions allow StackQL to map the providers resource to the desired ORM and define routes for SQL verbs such as `SELECT`, `INSERT`, `DELETE`, and `EXEC`.  
</p>
</details>

## Usage

```bash
openapi-doc-util <command> [<arguments(s)>] [<flag(s)>]
```

### Commands

__`validate`__  

Validates an OpenAPI specification document.   

__`split`__  

Splits an Open API document into smaller, self contained, service scoped documents based upon a specified service discriminator (JSONPath expression relative to each operation).  Takes an OpenAPI document as input and outputs the following structure:  

```
{output_dir}/
├─ {provider_name}/
│  ├─ {provider_version}/
│  │  ├─ services/
│  │  │  ├─ {service_name}/
│  │  │  │  ├─ {service_name}.yaml
│  │  │  ├─ .../
```
The `{service_name}.yaml` file is a self contained, OpenAPI specification document, containing only operations and components for the specified service.  

__`provider-dev`__  

Iterates through a directory of services (created using the `split` function), generates an additional document for each service `{service_name}-resources.yaml` and an index document for all services named `provider.yaml`, this is useful for developing new StackQL providers.  The output structure is as follows:  

```
{api_doc_dir}/
├─ {provider_name}/
│  ├─ {provider_version}/
│  │  ├─ services/
│  │  │  ├─ {service_name}/
│  │  │  │  ├─ {service_name}.yaml
│  │  │  │  ├─ {service_name}-resources.yaml
│  │  │  ├─ .../
│  │  ├─ provider.yaml
```
The original OpenAPI documents are not modified.  

The `provider-dev` function will infer SQL routes for operations under each resources under the `sqlVerbs` key, this should be reviewed by the developer, removing any duplicate or erroneous routes.

__`provider-build`__  

Operates on the dev directory structure generated using the `provider-dev` command, assembles a complete Open API specification per service including the StackQL resources for each service in each document under `components/x-stackQL-resources`, and outputs the following structure to a new directory:  

```
{output_dir}/
├─ {provider_name}/
│  ├─ {provider_version}/
│  │  ├─ services/
│  │  │  ├─ {service_name}/
│  │  │  │  ├─ {service_name}.yaml
│  │  │  ├─ .../
│  │  ├─ provider.yaml
```

The output from the `provider-build` command can be used to test locally (see [Test Using a Local Registry](#test-using-a-local-registry)). Once you have tested locally you can raise a pull request to [stackql/stackql-provider-registry](https://github.com/stackql/stackql-provider-registry).    


### Options

__`--svcDiscriminator`, `-s`__  *JSONPath expression* OR *svcName:servicename*

The __*service discriminator*__ option is used to determine how to split a large provider document into smaller, service scoped documents.  The option is required for the __`split`__ command and ignored otherwise.  If you do not wish to spilt the provider API spec, specify `svcName:<servicename>` for this option which will define one service in the StackQL provider with the name provided in *servicename*.  

> Example: `-s "$['x-github'].category"` would split the given provider API spec into service documents by matching the `x-github.category` value in each unique operation (combination of a path and an HTTP method) in API doc.

__`--resDiscriminator`, `-r`__  *JSONPath expression*  OR *path_tokens*

The __*resource discriminator*__ option is used to determine how to identify StackQL resources in a given provider API spec.  This option is used for the __`provider-dev`__ command and ignored otherwise.

> Example: `-r "$['x-github'].subcategory"`  would identify resources in the given provider API spec by matching the `x-github.subcategory` value in each unique operation (combination of a path and an HTTP method) in API doc.

if *path_tokens* is specified for the __`--resDiscriminator`__ option, the resource name will be derived by joining the 'meaningful' path tokens (not equivalent to the service name) with an '_'.  For instance a path of `/sites/{site_id}/service-instances` would result in a resource name of `sites_service_instances` assuming the service name was not `sites`.  

__`--methodkey`, `-m`__  *JSONPath expression*  

The __*method key*__ option determines uniquely named operations which are mapped to SQL methods in the StackQL ORM.  These methods are then mapped to default routes (SQL query and DML methods) in StackQL, the developer can override or update these mappings in the development docs which are outputted from the __`provider-dev`__  command.  This option is used for the __`provider-dev`__ command and ignored otherwise.  

> If supplied it must be a JSONPath expression relative to the operation (http path and verb), if not supplied it will default to `operationId` in the OpenAPI specification for each operation.  

__`--output`, `-o`__  *directory*  

The __*output directory*__ option specifies the root directory to output files from the  __`split`__, or the root directory to output assembled provider specs from the __`provider-build`__ command.  The default is the current working directory.  If the directory exists in either case the program will exit with an error unless the `--overwrite` option is set.

> If not supplied it will default to the current working directory

__`--overwrite`__

The __*overwrite*__ option specifies whether to overwrite files in the output directory.  The default is `false`.

__`--debug`, `-d`__  

__*debug flag*__ which can be set for additional print statements.  


## Example

This example demonstrates creating a StackQL provider for `github` and testing this provider locally using `stackql`

### Example Project Structure

The following directory structure represents the target after an API dev workspace is set up and the StackQL provider for `github` is built.

```
local-registry/
├─ ref/
│  ├─ github/
│  │  ├─ api.github.com.yaml
├─ dev/
│  ├─ github/
│  │  ├─ v0.1.0/
│  │  │  ├─ provider.yaml
│  │  │  ├─ services/
│  │  │  │  ├─ repos/
│  │  │  │  │  ├─ repos.yaml
│  │  │  │  │  ├─ repos-resources.yaml
│  │  │  │  ├─ .../
├─ src/
│  ├─ github/
│  │  ├─ v0.1.0/
│  │  │  ├─ provider.json
│  │  │  ├─ services/
│  │  │  │  ├─ repos/
│  │  │  │  │  ├─ repos-v0.1.0.json
```

`local-registry/ref/github/api.github.com.yaml` is the source OpenAPI 3 specification for the `github` api, this was sourced from [GitHub](https://github.com/github/rest-api-description/blob/main/descriptions/api.github.com/api.github.com.yaml).  

The `dev` directory contains the output of the dev docs generated by the `openapi-doc-util split` command, which will then serve as input to the `openapi-doc-util provider-dev` command.    

The `src` directory contains the output of the StackQL provider interface generated from the dev docs using `openapi-doc-util provider-build`.  


### 1. Split large spec into service specs

PowerShell:  

```PowerShell
cd local-registry
openapi-doc-util split `
-n github `
-v v0.1.0 `
-s "$['x-github'].category" `
-o .\dev `
./ref/github/api.github.com.yaml
```

Bash:  

```bash
cd local-registry
openapi-doc-util split \
-n github \
-v v0.1.0 \
-s '$["x-github"].category' \
-o ./dev \
ref/github/api.github.com.yaml
```

### 2. Create dev provider docs

PowerShell:  

```PowerShell
cd local-registry
openapi-doc-util provider-dev `
-n github `
-v v0.1.0 `
-r "$['x-github'].subcategory" `
.\dev
```

Bash:  

```bash
cd local-registry
openapi-doc-util provider-dev \
-n github \
-v v0.1.0 \
-r '$["x-github"].subcategory' \
./dev
```

### Assemble final provider docs

PowerShell:  

```PowerShell
cd local-registry
openapi-doc-util provider-build `
-n github `
-v v0.1.0 `
-o .\src `
.\dev
```

Bash:  

```bash
cd local-registry
openapi-doc-util provider-build \
-n github \
-v v0.1.0 \
-o ./src \
./dev
```

### Test Using a Local Registry

```bash
cd local-registry
PROVIDER_REGISTRY_ROOT_DIR="$(pwd)"
REG_STR='{"url": "file://'${PROVIDER_REGISTRY_ROOT_DIR}'", "localDocRoot": "'${PROVIDER_REGISTRY_ROOT_DIR}'", "verifyConfig": {"nopVerify": true}}'
AUTH_STR='{"github": { "type": "null_auth" }}'
./stackql shell --auth="${AUTH_STR}" --registry="${REG_STR}"
``` readmeEtag: '"ff4b3b99098da6030eed2df57bf5d3112c287d11"' readmeLastModified: Thu, 05 Jan 2023 19:01:12 GMT repositoryId: 487689765 description: 'DEPRECATED: Replaced by https://github.com/stackql/openapisaurus' created: '2022-05-02T02:01:26Z' updated: '2025-12-25T23:07:16Z' language: JavaScript archived: true stars: 10 watchers: 1 forks: 1 owner: stackql-labs logo: https://avatars.githubusercontent.com/u/251279874?v=4 license: MIT repoEtag: '"f1ffe60870cfde64965367220af98518248523e4a23c4f98deb8bc00b9ab94bf"' repoLastModified: Thu, 25 Dec 2025 23:07:16 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/stackql/openapi-doc-util - source: openapi3 tags repository: https://github.com/mattpolzin/jsonapi-openapi v3: true repositoryMetadata: base64Readme: >- # JSONAPI+OpenAPI
[![MIT license](http://img.shields.io/badge/license-MIT-lightgrey.svg)](http://opensource.org/licenses/MIT) [![Swift 6.0+](http://img.shields.io/badge/Swift-6.x-blue.svg)](https://swift.org) [![Build Status](https://app.bitrise.io/app/2ae0b5578e1905b8/status.svg?token=T8UAUN08e1_GnYk1z3P98g&branch=main)](https://app.bitrise.io/app/2ae0b5578e1905b8)

See parent project: https://github.com/mattpolzin/JSONAPI

The `JSONAPIOpenAPI` framework adds the ability to generate OpenAPI compliant JSON Schema documentation of a JSONAPI Document.

There is experimental support for generating `JSONAPI` Swift code from OpenAPI documentation in the JSONAPISwiftGen module. There is no formal documentation for this functionality, but it is an area of interest of mine. Reach out to me directly if you would like to know more.

See the Open API Spec here: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md

*This library has many loose ends and very little documentation. The documentation will grow as the framework becomes more complete.*

## Running and Testing
As of this writing, you need to run `swift package generate-xcodeproj` and then open that project in Xcode. Using Xcode's built-in Swift Package Manager support is currently broken for libraries like swift-syntax that require dynamic libraries from the Swift toolchain. `swift build`, `swift test`, etc. from the command line will work fine, though.

## _Experimental_ Swift Code Generation

The JSONAPISwiftGen module has experimental support for generating Swift code for `JSONAPI` models. You can dig into the source code or reach out to me for more information. This module is used by the API Test server project at [mattpolzin/jsonapi-openapi-test-server](https://github.com/mattpolzin/jsonapi-openapi-test-server) to generate models and tests.

## _Experimental_ Graphviz (DOT) Generation

The JSONAPIVizGen module has experimental support for generating Graphviz DOT files that graph out the models and relationships represented by the JSON:API/OpenAPI input. You can dig into the source code or reach out to me for more information.

## OpenAPI JSON Schema Generation
### Simple Example
You can try this out in the included Playground.

```swift
import Foundation
import JSONAPI
import OpenAPIKit
import JSONAPIOpenAPI
import Sampleable

let encoder = JSONEncoder()
encoder.outputFormatting = .prettyPrinted

//
// First describe the resource object
//
struct WidgetDescription: JSONAPI.ResourceObjectDescription {
    static var jsonType: String { return "widgets" }

    struct Attributes: JSONAPI.Attributes {
        let productName: Attribute<String>
    }

    struct Relationships: JSONAPI.Relationships {
        let subcomponents: ToManyRelationship<Widget, NoMetadata, NoLinks>
    }
}

typealias Widget = JSONAPI.ResourceObject<WidgetDescription, NoMetadata, NoLinks, String>

//
// Then make things sampleable
// This is needed because the only way to use reflection on
// your attributes and relationships structs is to create
// instances of them.
//
extension WidgetDescription.Attributes: Sampleable {
    static var sample: WidgetDescription.Attributes {
        return .init(productName: .init(value: "Fillihizzer Nob Hub"))
    }
}

extension WidgetDescription.Relationships: Sampleable {
    static var sample: WidgetDescription.Relationships {
        return .init(subcomponents: .init(ids: [.init(rawValue: "1")]))
    }
}

//
// We can create a JSON Schema for the Widget at this point
//
let widgetJSONSchema = Widget.openAPISchema(using: encoder)

//
// Describe a JSON:API response body with 1 widget and
// any number of related widgets included.
//
typealias SingleWidgetDocumentWithIncludes = Document<SingleResourceBody<Widget>, NoMetadata, NoLinks, Include1<Widget>, NoAPIDescription, BasicJSONAPIError<String>>

//
// Finally we can create a JSON Schema for the response body of a successful request
//
let jsonAPIResponseSchema = SingleWidgetDocumentWithIncludes.SuccessDocument.openAPISchema(using: encoder)

print(String(data: try! encoder.encode(jsonAPIResponseSchema), encoding: .utf8)!)

//
// Or a failed request
//
let jsonAPIResponseErrorSchema = SingleWidgetDocumentWithIncludes.ErrorDocument.openAPISchema(using: encoder)

//
// Or a schema describing the response as `oneOf` the success or error respones
//
let jsonAPIResponseFullSchema = SingleWidgetDocumentWithIncludes.openAPISchema(using: encoder)
```

The above code produces:
```json
{
  "type" : "object",
  "properties" : {
    "data" : {
      "type" : "object",
      "properties" : {
        "relationships" : {
          "type" : "object",
          "properties" : {
            "subcomponents" : {
              "type" : "object",
              "properties" : {
                "data" : {
                  "type" : "array",
                  "items" : {
                    "type" : "object",
                    "properties" : {
                      "type" : {
                        "type" : "string",
                        "enum" : [
                          "widgets"
                        ]
                      },
                      "id" : {
                        "type" : "string"
                      }
                    },
                    "required" : [
                      "id",
                      "type"
                    ]
                  }
                }
              },
              "required" : [
                "data"
              ]
            }
          },
          "required" : [
            "subcomponents"
          ]
        },
        "id" : {
          "type" : "string"
        },
        "type" : {
          "type" : "string",
          "enum" : [
            "widgets"
          ]
        },
        "attributes" : {
          "type" : "object",
          "properties" : {
            "productName" : {
              "type" : "string"
            }
          },
          "required" : [
            "productName"
          ]
        }
      },
      "required" : [
        "attributes",
        "id",
        "relationships",
        "type"
      ]
    },
    "included" : {
      "type" : "array",
      "items" : {
        "type" : "object",
        "properties" : {
          "relationships" : {
            "type" : "object",
            "properties" : {
              "subcomponents" : {
                "type" : "object",
                "properties" : {
                  "data" : {
                    "type" : "array",
                    "items" : {
                      "type" : "object",
                      "properties" : {
                        "type" : {
                          "type" : "string",
                          "enum" : [
                            "widgets"
                          ]
                        },
                        "id" : {
                          "type" : "string"
                        }
                      },
                      "required" : [
                        "type",
                        "id"
                      ]
                    }
                  }
                },
                "required" : [
                  "data"
                ]
              }
            },
            "required" : [
              "subcomponents"
            ]
          },
          "id" : {
            "type" : "string"
          },
          "type" : {
            "type" : "string",
            "enum" : [
              "widgets"
            ]
          },
          "attributes" : {
            "type" : "object",
            "properties" : {
              "productName" : {
                "type" : "string"
              }
            },
            "required" : [
              "productName"
            ]
          }
        },
        "required" : [
          "attributes",
          "id",
          "relationships",
          "type"
        ]
      },
      "uniqueItems" : true
    }
  },
  "required" : [
    "included",
    "data"
  ]
}
```
 readmeEtag: '"d570c0b49d32baee8f04999b438d9ba54abc4a05"' readmeLastModified: Mon, 15 Sep 2025 15:22:23 GMT repositoryId: 168446016 description: >- A library that adds support for generating OpenAPI compliant documentation from JSON API compliant models. created: '2019-01-31T01:58:58Z' updated: '2025-09-15T15:22:27Z' language: Swift archived: false stars: 12 watchers: 1 forks: 1 owner: mattpolzin logo: https://avatars.githubusercontent.com/u/2075353?v=4 license: MIT repoEtag: '"5a1cbd138eb356e2760e3cbae4169a8f2ceb5dfaa90e32bce69a2e058df55a8b"' repoLastModified: Mon, 15 Sep 2025 15:22:27 GMT foundInMaster: true category: Parsers id: 7e5ab9bd4ed0a9dd5317056ec3612287 - source: openapi3 tags repository: https://github.com/mchangrh/sb-openapi v3: true repositoryMetadata: base64Readme: >- IyBTcG9uc29yQmxvY2sgQVBJIERvY3VtZW50YXRpb24gaW4gT3BlbkFQSQoKQVBJIEZvbGxvd3MgW0NDIEJZLU5DLVNBIDQuMF0oaHR0cHM6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL2xpY2Vuc2VzL2J5LW5jLXNhLzQuMC8pIGFzIHBlciBbR2l0SHViIFdpa2ldKGh0dHBzOi8vZ2l0aHViLmNvbS9hamF5eXkvU3BvbnNvckJsb2NrL3dpa2kvRGF0YWJhc2UtYW5kLUFQSS1MaWNlbnNlKS4KCkRvY3VtZW50YXRpb24gaW4gTWVkaWFXaWtpIGZvcm0gY2FuIGJlIGZvdW5kIFtvbiB0aGUgU0Igd2lraV0oaHR0cHM6Ly93aWtpLnNwb25zb3IuYWpheS5hcHAvaW5kZXgucGhwL0FQSV9Eb2NzKQoKWW91IGNhbiBzZWUgaG93IHVwIHRvIGRhdGUgdGhlIGRvY3VtZW50YXRpb24gaXMgYnkgY2hlY2tpbmcgdGhlIFtzdXBwb3J0IG1hdHJpeF0oaHR0cHM6Ly9tY2hhbmcuaWN1L3NiLW1hdHJpeCk= readmeEtag: '"b05c62b6ff64d7ac2bafe329217a8af2f9aba997"' readmeLastModified: Sat, 28 Oct 2023 05:18:44 GMT repositoryId: 402975679 description: SponsorBlock API in Swagger/ OpenAPI created: '2021-09-04T05:49:13Z' updated: '2024-10-14T04:52:31Z' language: HTML archived: false stars: 10 watchers: 1 forks: 0 owner: mchangrh logo: https://avatars.githubusercontent.com/u/15132783?v=4 license: GPL-3.0 repoEtag: '"0bbaaee37e22c6eeb7dd1a33f7688d121dc2761340dbada86da0faed67b312bd"' repoLastModified: Mon, 14 Oct 2024 04:52:31 GMT foundInMaster: true category: - Documentation - Server Implementations id: 328ab8ea3691a6875fdc576eff130d15 - source: openapi3 tags repository: https://github.com/apiaryio/dredd-transactions v3: true repositoryMetadata: base64Readme: >- # Dredd Transactions

[![npm version](https://badge.fury.io/js/dredd-transactions.svg)](https://badge.fury.io/js/dredd-transactions)
[![Build Status](https://travis-ci.org/apiaryio/dredd-transactions.svg?branch=master)](https://travis-ci.org/apiaryio/dredd-transactions)
[![Build status](https://ci.appveyor.com/api/projects/status/hh8l50ssai3p4d3f/branch/master?svg=true)](https://ci.appveyor.com/project/Apiary/dredd-transactions/branch/master)
[![Known Vulnerabilities](https://snyk.io/test/npm/dredd-transactions/badge.svg)](https://snyk.io/test/npm/dredd-transactions)


Dredd Transactions library compiles *HTTP Transactions* (simple Request-Response pairs) from API description document.

> **Note:** To better understand *emphasized* terms in this documentation, please refer to the [Glossary of Terms][api-blueprint-glossary]. All data structures are described using the [MSON][mson-spec] format.

This project supersedes [Blueprint Transactions][blueprint-transactions] library.


## Features

* Inherits parameters from parent *Resource* and *Action* sections.
* Expands *URI Templates*.
* Warns on undefined placeholders in *URI Templates* (both query and path).
* Validates URI parameter types.
* Selects first *Request* and first *Response* if multiple are specified in the API description document.


### Deprecated Features

* Compiles [Transaction Name][transaction-object-spec] string (vague identifier) for each *Transaction*.
* Provides [Transaction Origin][transaction-object-spec] with pointers to [API Elements][api-elements] derived from the original API description document.

> **Note:** These features are to be superseded by whatever comes out of the proposal in [apiaryio/dredd#227](https://github.com/apiaryio/dredd/issues/227).


## Installation

```
npm install dredd-transactions
```


## Development

Dredd Transactions library is written in JavaScript (ES2015+).


## Usage

### `parse`

Parses given API description document into API Elements with options specific
to Dredd. Assumes that documents with unrecognizable format are
[API Blueprint][api-blueprint]. Turns any parser failures, including
the unexpected ones, into [API Elements][api-elements] annotations.

```javascript
const parse = require('dredd-transactions/parse');
// const { parse } = require('dredd-transactions');

parse('# My API\n...', (error, parseResult) => {
  // ...
});
```

### `compile`

Compiles *HTTP Transactions* from given [API Elements][api-elements]. *HTTP Transactions* are a backbone data structure to Dredd.

```javascript
const compile = require('dredd-transactions/compile');
// const { compile } = require('dredd-transactions');

const compileResult = compile(mediaType, apiElements, filename);
```

> **Note:** The `filename` argument is optional and about to get deprecated, see [#6][filename-deprecation]


## Data Structures

<a name="parse-result-object"></a>
### Parse Result (object)

Result of parsing.

- `mediaType`: `text/vnd.apiblueprint` (string, default, nullable) - Media type of the input format, can be empty in case of some fatal errors
- `apiElements` ([API Elements][api-elements]) - API Elements parse result

<a name="compile-result-object"></a>
### Compile Result (object)

Result of compilation. Alongside compiled [Transaction][transaction-object-spec] objects contains also errors and warnings, mainly from API description parser.

- `mediaType`: `text/vnd.apiblueprint` (string, default, nullable) - Media type of the input format, defaults to API Blueprint format. Can be empty in case of some fatal errors.
- `transactions` (array[[Transaction][transaction-object-spec]]) - Compiled _HTTP Transactions_.
- `annotations` (array[[Annotation][annotation-object-spec]]) - Errors and warnings which occurred during parsing of the API description or during compilation of transactions.

<a name="transaction-object"></a>
### Transaction (object)

Represents a single *HTTP Transaction* (Request-Response pair) and its location in the API description document. The location is provided in two forms, both **deprecated** as of now:

- `name` - String representation, both human- and machine-readable.
- `origin` - Object of references to nodes of [API Elements][api-elements] derived from the original API description document.

> **Note:** These two forms of locating HTTP Transactions are to be superseded by whatever comes out of the proposal in [apiaryio/dredd#227](https://github.com/apiaryio/dredd/issues/227).


### Properties

- request (object) - HTTP Request as described in API description document.
    - method
    - uri: `/message` (string) - Informative URI of the Request.
    - headers (array) - List of HTTP headers in their original order, with the original casing of the header name, including multiple headers of the same name.
        - (object)
            - name: `Content-Type` (string)
            - value: `text/plain` (string)
    - body: `Hello world!\n` (string)
- response (object) - Expected HTTP Response as described in API description document.
    - status: `200` (string)
    - headers (array) - List of HTTP headers in their original order, with the original casing of the header name, including multiple headers of the same name.
        - (object)
            - name: `Content-Type` (string)
            - value: `text/plain` (string)
    - body (string, optional)
    - schema (string, optional)


### Deprecated Properties

- name: `Hello world! > Retrieve Message` (string) - Transaction Name, non-deterministic breadcrumb location of the HTTP Transaction within the API description document.
- origin (object) - Object of references to nodes of [API Elements][api-elements] derived from the original API description document.
    - filename: `./api-description.apib` (string)
    - apiName: `My Api` (string)
    - resourceGroupName: `Greetings` (string)
    - resourceName: `Hello, world!` (string)
    - actionName: `Retrieve Message` (string)
    - exampleName: `First example` (string)

> **Note:** These properties are to be superseded by whatever comes out of the proposal in [apiaryio/dredd#227](https://github.com/apiaryio/dredd/issues/227).


<a name="annotation-object"></a>
### Annotation (object)

Description of an error or warning which occurred during parsing of the API description or during compilation of transactions.

#### Properties

- type (enum[string])
    - `error`
    - `warning`
- component (enum[string]) - In which component of the compilation process the annotation occurred
    - `apiDescriptionParser`
    - `parametersValidation`
    - `uriTemplateExpansion`
- message (string) - Textual annotation. This is – in most cases – a human-readable message to be displayed to user
- location (array, fixed, nullable) - Location of the annotation in the source file, represented by a single range of line and column number pairs if available, or by `null` otherwise
    - (array) - Start location
        - (number) - Line number
        - (number) - Column number
    - (array) - End location
        - (number) - Line number
        - (number) - Column number

### Deprecated Properties

- name: `Hello world! > Retrieve Message` (string) - Transaction Name, non-deterministic breadcrumb location of the relevant HTTP Transaction within the API description document.
- origin (object) - Object of references to nodes of [API Elements][api-elements] derived from the original API description document.
    - filename: `./api-description.apib` (string)
    - apiName: `My Api` (string)
    - resourceGroupName: `Greetings` (string)
    - resourceName: `Hello, world!` (string)
    - actionName: `Retrieve Message` (string)
    - exampleName: `First example` (string)

> **Note:** These properties are to be superseded by whatever comes out of the proposal in [apiaryio/dredd#227](https://github.com/apiaryio/dredd/issues/227).


[dredd]: https://dredd.org
[mson-spec]: https://github.com/apiaryio/mson
[api-elements]: http://api-elements.readthedocs.org/
[api-blueprint]: https://apiblueprint.org/
[api-blueprint-glossary]: https://github.com/apiaryio/api-blueprint/blob/master/Glossary%20of%20Terms.md
[blueprint-transactions]: https://github.com/apiaryio/blueprint-transactions/


[filename-deprecation]: https://github.com/apiaryio/dredd-transactions/issues/6
[compile-result-object-spec]: #compile-result-object
[transaction-object-spec]: #transaction-object
[annotation-object-spec]: #annotation-object
[source-map]: https://github.com/refractproject/refract-spec/blob/master/namespaces/parse-result-namespace.md#source-map-element
 readmeEtag: '"87add22e0b40b26bddb1a4f6454442e6d920ae51"' readmeLastModified: Tue, 01 Oct 2019 14:31:56 GMT repositoryId: 54039079 description: >- Compiles a list of HTTP transactions (request-response pairs) from an API description document created: '2016-03-16T14:30:38Z' updated: '2024-11-08T09:04:01Z' language: JavaScript archived: true stars: 10 watchers: 4 forks: 9 owner: apiaryio logo: https://avatars.githubusercontent.com/u/765943?v=4 license: MIT repoEtag: '"d673207324583d3ef413ba672d5df9ed1795b88d4e0075179f900fb3f0a2d4e3"' repoLastModified: Fri, 08 Nov 2024 09:04:01 GMT foundInMaster: true category: Testing id: 180ad77913c096145acf085c2f8a0be8 - source: openapi3 tags repository: https://github.com/metamug/r2 v3: true id: 6a1c275d56f9cea18e372366b6f469f3 repositoryMetadata: base64Readme: >- IyBSMgpSMiBpcyBhbiBvcGVuIHNvdXJjZSBSRVNUIFJlc291cmNlIHNlcnZlci4gSXQgY29tZXMgd2l0aCBhIHJlYWR5IHRvIHVzZSBSRVNUIEFQSS4gRGV2ZWxvcGVyIGNhbiBhZGQvZWRpdCByZXNvdXJjZXMgdG8gdGhlIHNlcnZlcgp1c2luZyBSMiBDb25zb2xlCgohW01ldGFtdWcgUmVzb3VyY2UgU2NyZWVuXShodHRwczovL21ldGFtdWcuY29tL2ltZy9yZXMtc2NyZWVuLnBuZykKCiMjIFIyIENvbnNvbGUKUjIgY29uc29sZSBjb21lcyB3aXRoIGZvbGxvd2luZyBmZWF0dXJlcy4KCiogUkVTVCBSZXNvdXJjZSBNYW5hZ2VtZW50IAoqIEhvdCBkZXBsb3ltZW50IGZvciBSRVNUIFJlc291cmNlcwoqIFJlc291cmNlIGVkaXRvciB3aXRoIGF1dG9jb21wbGV0ZSBhbmQgcXVlcnkgdGVzdGluZyAodXNpbmcgb3BlbiBzb3VyY2UgcHJvamVjdCkKKiBPcGVuIEFQSSBkb2N1bWVudGF0aW9uIGdlbmVyYXRlZC4KCiMjIEFQSSBJbnRlZ3JhdGlvbiB3aXRoIFhSZXF1ZXN0CgpBUEkgSW50ZWdyYXRpb24gd2l0aCB0aGlyZCBwYXJ0eSBzZXJ2aWNlcyBsaWtlIEFXUywgRmFjZWJvb2ssIFR3aXR0ZXIsIEZpcmViYXNlLCBQYXlQYWwsIE1haWxjaGltcCBhbmQgbW9yZS4KQ29tbXVuaWNhdGlvbiB3aXRoIG11bHRpcGxlIHNlcnZpY2VzIGluIGEgc2luZ2xlIHJlcXVlc3QgdXNpbmcgQVBJIEdhdGV3YXlzLgoKIVtNZXRhbXVnIEFQSSBJbnRlZ3JhdGlvbl0oaHR0cHM6Ly9tZXRhbXVnLmNvbS9pbWcvYXBpLWludGVncmF0aW9uMS5zdmcpCgoKIyMgb3BlbmFwaS1yZXN0LW1vZGVsCgohW10oaHR0cHM6Ly90cmF2aXMtY2kub3JnL21ldGFtdWcvb3BlbmFwaS1yZXN0LW1vZGVsLnN2Zz9icmFuY2g9b3Blbi1hcGkpIFshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9tZXRhbXVnL29wZW5hcGktcmVzdC1tb2RlbC9icmFuY2gvb3Blbi1hcGkvZ3JhcGgvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL21ldGFtdWcvb3BlbmFwaS1yZXN0LW1vZGVsKQoKIVtNZXRhbXVnIE9wZW4gQVBJIEludGVncmF0aW9uXShodHRwczovL21ldGFtdWcuY29tL2ltZy9vcGVuYXBpLXNwZWNpZmljYXRpb24uc3ZnKQoKCkNvbnZlcnQgW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIFNwZWMgZmlsZSBpbnRvIFtSZXNvdXJjZSBSZXNvdXJjZSBYTUxdKGh0dHBzOi8vbWV0YW11Zy5jb20vZG9jcy9yZXNvdXJjZS1maWxlKSBmaWxlcy4KVGhpcyBwcm9qZWN0IGFpbXMgdG8gZ2VuZXJhdGUgY29tcGF0aWJsZSByZXNvdXJjZSB4bWxzIGZvciBhIGdpdmVuIHNwZWMgZmlsZSBqc29uL3ltbC4KClRoZSBzYW1wbGUgcGV0c3RvcmUgeW1sIGZpbGUgd2lsbCBjb252ZXJ0IHRvIGEgbnVtYmVyIG9mIHJlc291cmNlIHhtbHMuCmh0dHBzOi8vZWRpdG9yLnN3YWdnZXIuaW8vCgojIyMgUkVTVCBSZXNvdXJjZSBGaWxlcwoKUkVTVCBSZXNvdXJjZSBmaWxlcyBhcmUgZGVzaWduZWQgdG8gYWRkcmVzcyBhIFJFU1QgcmVzb3VyY2VzIGluZGl2aWR1YWxseS4gRWFjaCBSZXNvdXJjZSBmaWxlCm1hcHMgdG8gaXRzIG93biBVUkkuIFRoaXMgbWFrZXMgcmVzb3VyY2VzIGVhc2lseSBpZGVudGlmaWFibGUgYW5kIG1hbmFnYWJsZS4KCmh0dHBzOi8vbWV0YW11Zy5jb20vZG9jcy9yZXNvdXJjZS1maWxlCgojIyMgRGF0YXNvdXJjZXMKCltSZWFkIHRoaXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tZXRhbXVnL1IyL2Jsb2IvZGV2ZWxvcC9kb2NzL2RhdGFzb3VyY2VzLm1kKQoKIyMjIE9wZW4gQVBJIAoKaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24KCiMjIERlcGVuZGVuY2llcwoKW01hc29uXShodHRwczovL2dpdGh1Yi5jb20vbWV0YW11Zy9tYXNvbikgdjMuNCBhdmFpbGFibGUgb24gTWF2ZW4gY2VudHJhbAo= readmeEtag: '"53382c632678bb35baba33a0a86dd5730360631c"' readmeLastModified: Sun, 27 Nov 2022 08:58:45 GMT repositoryId: 213134305 description: >- R2 is your REST API Server that can connect with your database and other APIs using simple resource configurations created: '2019-10-06T08:37:57Z' updated: '2025-02-13T05:06:18Z' language: Java archived: false stars: 10 watchers: 2 forks: 3 owner: metamug logo: https://avatars.githubusercontent.com/u/21278398?v=4 repoEtag: '"1f0aaf7a420ebb5fb136b9406b0eee83fa83a8bf8d743573146228840ae2a6c0"' repoLastModified: Thu, 13 Feb 2025 05:06:18 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/candiun/aws-openapi-lint v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIEFXUyBBUEkgR2F0ZXdheSBsaW50ZXIKClshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9ldmlsbWludC9hd3Mtb3BlbmFwaS1saW50L2JyYW5jaC9tYXN0ZXIvZ3JhcGgvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2V2aWxtaW50L2F3cy1vcGVuYXBpLWxpbnQpIFshW2xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvZXZpbG1pbnQvYXdzLW9wZW5hcGktbGludCldKGh0dHBzOi8vZ2l0aHViLmNvbS9ldmlsbWludC9hd3Mtb3BlbmFwaS1saW50KQoKQVdTLU9wZW5BUEktTGludCBpcyBhIHNpbXBsZSBPcGVuQVBJIDMgeWFtbCAvIGpzb24gc3BlYyBsaW50ZXIgZGVzaWduZWQgZm9yIGNoZWNraW5nIEFQSSBHYXRld2F5IGludGVncmF0aW9uLgoKIyMgUnVsZXMKCkl0IGNvbnRhaW5zIHJ1bGVzIGZvciBjaGVja2luZyB3aGV0aGVyOgoKLSB5b3UgaGF2ZSBhbiBhdXRob3JpemVyIG9uIE9QVElPTlMKLSBhdXRob3JpemVyIGlzIG1lbnRpb25lZCBpbiBgcmVxdWVzdFBhcmFtZXRlcnNgIGJ1dCBpcyBub3QgcHJlc2VudCBpbiBgc2VjdXJpdHlgCi0gaHR0cCB2ZXJicyBhcmUgY29uc2lzdGVudCBpbiB0aGUgcGF0aCBhbmQgaW50ZWdyYXRpb24KLSBhbGwgdXNlZCBoZWFkZXJzIGluIHBhdGggZnJvbSBhbGwgdmVyYnMgYXJlIG1lbnRpb25lZCBpbiBDT1JTIHJ1bGVzIGFuZCB2aWNlLXZlcnNhCi0gQ09SUyBydWxlcyBhbGxvdyBhbGwgdmVyYnMgbWVudGlvbmVkIGluIHRoZSBwYXRoCi0gQ09SUyBydWxlcyBhcmUgcHJlc2VudAotIGFtYXpvbiBpbnRlZ3JhdGlvbiBpcyBwcmVzZW50Ci0gcGF0aCBwYXJhbWV0ZXJzIHByZXNlbnQgaW4gYHJlcXVlc3RQYXJhbWV0ZXJzYCBhcmUgbm90IHVzZWQgaW4gdGhlIGRpcmVjdCBwYXRoIHBhcmFtZXRlcnMgYW5kIHZpY2UtdmVyc2EKCiMjIFJvYWRtYXAKCi0gWyBdIFVwZGF0ZSBSRUFETUUgd2l0aCBydWxlIG5hbWVzIGFuZCBiZWhhdmlvcgotIFtYXSBTdXBwb3J0IGpzb24gc3BlY3MKLSBbWF0gQWRkIG9wdGlvbmFsIHJ1bGUgZm9yIGNoZWNraW5nIGJhc2UgdXJsIGVxdWFsaXR5Ci0gWyBdIEFkZCBzdXBwb3J0IGZvciBjb25maWd1cmF0aW9uIHlhbWwgZmlsZQotIFsgXSBBZGQgcG9zc2liaWxpdHkgdG8gZGlzYWJsZSBydWxlIGNoZWNraW5nIG9uIHNwZWNpZmljIHBhdGhzCi0gWyBdIEFkZCBwb3NzaWJpbGl0eSB0byBkaXNhYmxlIHJ1bGVzIHBlciBwYXRoCi0gWyBdIElnbm9yZSBwYXRoLXBhcmFtcyBpZiBgaHR0cF9wcm94eWAgaW50ZWdyYXRpb24gdHlwZSB1c2VkCi0gW1hdIEFkZCBvcHRpb24gdG8gZGlzYWJsZSBydWxlcyB2aWEgQ0xJCi0gW1hdIEFkZCB3YXJuaW5nIHRocmVzaG9sZCB0byByZXR1cm4gd2l0aCBzdGF0dXMgY29kZSAwIGlmIGxpbWl0IG5vdCBleGNlZWRlZAotIFtYXSBGaXggZmxha2U4IHZpb2xhdGlvbnMKLSBbWF0gQWRkIGEgbGljZW5zZQotIFtYXSBQdWJsaXNoIHRvIFB5UEkgb3IgYWxpa2UKLSBbWF0gQ29uZmlndXJlIHByb3Blcmx5IHVwIEdpdEh1YiBhY3Rpb25zIHRvIHJ1biB0ZXN0cyBvbiBwdXNoCgojIyBJbnN0YWxsYXRpb24KCmBgYApwaXAgaW5zdGFsbCBhd3Mtb3BlbmFwaS1saW50CmBgYAoKIyMgVXNhZ2UKCmAkIGF3cy1vcGVuYXBpLWxpbnQgcGF0aC90by9zcGVjLnltbGAKCmBgYAp1c2FnZTogbWFpbi5weSBbLWhdIFstLXRyZWF0LWVycm9ycy1hcy13YXJuaW5nc10KICAgICAgICAgICAgICAgWy0td2FybmluZy10aHJlc2hvbGQgV0FSTklOR19USFJFU0hPTERdCiAgICAgICAgICAgICAgIFstLWV4Y2x1ZGUtcnVsZXMgRVhDTFVERV9SVUxFU10KICAgICAgICAgICAgICAgbGludF9maWxlCgpMaW50IE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgYmFzZWQgb24gQVdTIEFQSSBHYXRld2F5LgoKcG9zaXRpb25hbCBhcmd1bWVudHM6CiAgbGludF9maWxlICAgICAgICAgICAgIFNwZWNpZnkgcGF0aCB0byB0aGUgb3BlbmFwaSBzY2hlbWEgZmlsZS4KCm9wdGlvbmFsIGFyZ3VtZW50czoKICAtaCwgLS1oZWxwICAgICAgICAgICAgc2hvdyB0aGlzIGhlbHAgbWVzc2FnZSBhbmQgZXhpdAogIC0tdHJlYXQtZXJyb3JzLWFzLXdhcm5pbmdzCiAgICAgICAgICAgICAgICAgICAgICAgIFRyZWF0cyBlcnJvcnMgYXMgd2FybmluZ3MgKGV4aXQgY29kZSB3aWxsIGJlIDAgdW5sZXNzCiAgICAgICAgICAgICAgICAgICAgICAgIHdhcm5pbmcgdGhyZXNob2xkIGlzIHNwZWNpZmllZAogIC0td2FybmluZy10aHJlc2hvbGQgV0FSTklOR19USFJFU0hPTEQKICAgICAgICAgICAgICAgICAgICAgICAgV2FybmluZyB0aHJlc2hvbGQgd2hpY2ggd2hlbiBzdXJwYXNzZWQgcmVuZGVycyBleGl0CiAgICAgICAgICAgICAgICAgICAgICAgIGNvZGUgdG8gYmVjb21lIDEpCiAgLS1leGNsdWRlLXJ1bGVzIEVYQ0xVREVfUlVMRVMKICAgICAgICAgICAgICAgICAgICAgICAgRXhjbHVkZWQgcnVsZXMgc2VwYXJhdGVkIGJ5IGNvbW1hLgoKYGBgCg== readmeEtag: '"1237c8cca93439cf194c3812e78bff8f53a67a55"' readmeLastModified: Sun, 20 Mar 2022 15:58:07 GMT repositoryId: 224503318 description: AWS Gateway Integration linter for OpenAPI 3.0.x specs created: '2019-11-27T19:31:56Z' updated: '2025-11-07T09:46:42Z' language: Python archived: false stars: 11 watchers: 1 forks: 1 owner: candiun logo: https://avatars.githubusercontent.com/u/9850720?v=4 license: MIT repoEtag: '"6dc4df0e4f942338e4ec33477541889c39966282551ee0dc645ac35f675a21dd"' repoLastModified: Fri, 07 Nov 2025 09:46:42 GMT foundInMaster: true category: - Description Validators - Parsers id: 78a6fb78a21b043d94b196aab3759c01 oldLocations: - https://github.com/evilmint/aws-openapi-lint - source: openapi3 tags repository: https://github.com/pierre-emmanuelj/open-alldebrid v3: true repositoryMetadata: base64Readme: >- IyBPcGVuIEFsbGRlYnJpZAoKQWxsZGVicmlkIE9wZW5BUEkgR29sYW5nIHdyYXBwZXIKClRoZSBHb2xhbmcgd3JhcHBlciBpcyBnZW5lcmF0ZWQgZnJvbSBhbiBbT3BlbkFQSTMgc3BlY10oLi9hbGxkZWJyaWQueWFtbCkgYmFzZWQgb24gdGhlIG9mZmljaWFsIEFsbGRlYnJpZC5jb20gW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vZG9jcy5hbGxkZWJyaWQuY29tLyNnZW5lcmFsLWluZm9ybWF0aW9ucykKCk9wZW4gQWxsZGVicmlkIFN3YWdnZXItdWkgW1Zpc2l0IGhlcmVdKGh0dHBzOi8vcGllcnJlLWVtbWFudWVsai5naXRodWIuaW8vb3Blbi1hbGxkZWJyaWQvKQoKVGhpcyBbT3BlbkFQSSBzcGVjXSguL2FsbGRlYnJpZC55YW1sKSBjYW4gYmUgdXNlZCB0byBnZW5lcmF0ZSBjbGllbnQgbGlicmFyeSBpbiBvdGhlciBsYW5ndWFnZXMhCgojIyBPdmVydmlldwoKVGhpcyBBUEkgY2xpZW50IHdhcyBnZW5lcmF0ZWQgYnkgdGhlIFtPcGVuQVBJIEdlbmVyYXRvcl0oaHR0cHM6Ly9vcGVuYXBpLWdlbmVyYXRvci50ZWNoKSBwcm9qZWN0LgoKLSBBUEkgdmVyc2lvbjogNC4wLjAKLSBQYWNrYWdlIHZlcnNpb246IDEuMC4wCgojIyBJbnN0YWxsYXRpb24KCkluc3RhbGwgdGhlIGZvbGxvd2luZyBkZXBlbmRlbmNpZXM6CgpgYGBzaGVsbApnbyBnZXQgImdpdGh1Yi5jb20vcGllcnJlLWVtbWFudWVsSi9vcGVuLWFsbGRlYnJpZCIKYGBgCgpBZGQgdGhlIGZvbGxvd2luZyBpbXBvcnQ6CgpgYGBnb2xhbmcKaW1wb3J0ICJnaXRodWIuY29tL3BpZXJyZS1lbW1hbnVlbEovb3Blbi1hbGxkZWJyaWQvY2xpZW50IgpgYGAKCiMjIEV4YW1wbGVzCgpgYGBHb2xhbmcKcGFja2FnZSBtYWluCgppbXBvcnQgKAoJImNvbnRleHQiCgkiZm10IgoJIm9zIgoKCSJnaXRodWIuY29tL3BpZXJyZS1lbW1hbnVlbEovb3Blbi1hbGxkZWJyaWQvY2xpZW50IgopCgpmdW5jIG1haW4oKSB7CglhZ2VudCA6PSAiYWdlbnRfZXhhbXBsZSIgLy8gWW91ciBzb2Z0d2FyZSB1c2VyLWFnZW50LiAoZGVmYXVsdCB0byAib3Blbi1hbGxkZWJyaWQiKQoKCWNvbmZpZ3VyYXRpb24gOj0gY2xpZW50Lk5ld0NvbmZpZ3VyYXRpb24oKQoJY29uZmlndXJhdGlvbi5BZGREZWZhdWx0SGVhZGVyKCJBdXRob3JpemF0aW9uIiwgIkJlYXJlciBBcGlLZXlGcm9tQWxsZGVicmlkIikKCWFwaV9jbGllbnQgOj0gY2xpZW50Lk5ld0FQSUNsaWVudChjb25maWd1cmF0aW9uKQoKCXJlc3AsIHIsIGVyciA6PSBhcGlfY2xpZW50LkRlZmF1bHRBcGkuVXNlckdldChjb250ZXh0LkJhY2tncm91bmQoKSkuQWdlbnQoYWdlbnQpLkV4ZWN1dGUoKQoJaWYgZXJyICE9IG5pbCB7CgkJZm10LkZwcmludGYob3MuU3RkZXJyLCAiRXJyb3Igd2hlbiBjYWxsaW5nIGBEZWZhdWx0QXBpLlVzZXJHZXRgYDogJXZcbiIsIGVycikKCQlmbXQuRnByaW50Zihvcy5TdGRlcnIsICJGdWxsIEhUVFAgcmVzcG9uc2U6ICV2XG4iLCByKQoJfQoKCWZtdC5QcmludGYoIiUjdlxuIiwgcmVzcCkKfQpgYGAKClBvc3NpYmxlIHRvIGFkZCBCZWFyZXIgVG9rZW4gZnJvbSBjb250ZXh0CmBgYEdvbGFuZwpwYWNrYWdlIG1haW4KCmltcG9ydCAoCgkiY29udGV4dCIKCSJmbXQiCgkib3MiCgoJImdpdGh1Yi5jb20vcGllcnJlLWVtbWFudWVsSi9vcGVuLWFsbGRlYnJpZC9jbGllbnQiCikKCmZ1bmMgbWFpbigpIHsKCWFnZW50IDo9ICJhZ2VudF9leGFtcGxlIiAvLyBZb3VyIHNvZnR3YXJlIHVzZXItYWdlbnQuIChkZWZhdWx0IHRvICJvcGVuLWFsbGRlYnJpZCIpCglhdXRoIDo9IGNvbnRleHQuV2l0aFZhbHVlKAoJCWNvbnRleHQuQmFja2dyb3VuZCgpLAoJCW9wZW5hcGljbGllbnQuQ29udGV4dEFjY2Vzc1Rva2VuLCAiQXBpS2V5RnJvbUFsbGRlYnJpZCIsCgkpCgoJY29uZmlndXJhdGlvbiA6PSBjbGllbnQuTmV3Q29uZmlndXJhdGlvbigpCglhcGlfY2xpZW50IDo9IGNsaWVudC5OZXdBUElDbGllbnQoY29uZmlndXJhdGlvbikKCglyZXNwLCByLCBlcnIgOj0gYXBpX2NsaWVudC5EZWZhdWx0QXBpLlVzZXJHZXQoYXV0aCkuQWdlbnQoYWdlbnQpLkV4ZWN1dGUoKQoJaWYgZXJyICE9IG5pbCB7CgkJZm10LkZwcmludGYob3MuU3RkZXJyLCAiRXJyb3Igd2hlbiBjYWxsaW5nIGBEZWZhdWx0QXBpLlVzZXJHZXRgYDogJXZcbiIsIGVycikKCQlmbXQuRnByaW50Zihvcy5TdGRlcnIsICJGdWxsIEhUVFAgcmVzcG9uc2U6ICV2XG4iLCByKQoJfQoKCWZtdC5QcmludGYoIiUjdlxuIiwgcmVzcCkKfQpgYGAKCiMjIERvY3VtZW50YXRpb24gZm9yIEFQSSBFbmRwb2ludHMKCkZpbmQgYWxsIHRoZSBBUEkgRG9jdW1lbnRhdGlvbiBbaGVyZV0oLi9kb2NzL0RlZmF1bHRBcGkubWQpCgpTd2FnZ2VyIFVJIFtoZXJlXShodHRwczovL3BpZXJyZS1lbW1hbnVlbGouZ2l0aHViLmlvL29wZW4tYWxsZGVicmlkLykKCllvdSBjYW4gZXhlY3V0ZSBIVFRQIHJlcXVlc3QgZnJvbSB0aGUgVUkgd2l0aCB5b3VyIFtBbGxkZWJyaWQgQXBpS2V5XShodHRwczovL2FsbGRlYnJpZC5jb20vYXBpa2V5cy8pCgojIyBEb2N1bWVudGF0aW9uIEZvciBBdXRob3JpemF0aW9uCgojIyMgYmVhcmVyQXV0aAoKLSAqKlR5cGUqKjogSFRUUCBCZWFyZXIgdG9rZW4gYXV0aGVudGljYXRpb24KCkV4YW1wbGUKCmBgYGdvbGFuZwphdXRoIDo9IGNvbnRleHQuV2l0aFZhbHVlKGNvbnRleHQuQmFja2dyb3VuZCgpLCBzdy5Db250ZXh0QWNjZXNzVG9rZW4sICJBcGlLZXlGcm9tQWxsZGVicmlkIikKCmNvbmZpZ3VyYXRpb24gOj0gY2xpZW50Lk5ld0NvbmZpZ3VyYXRpb24oKQphcGlfY2xpZW50IDo9IGNsaWVudC5OZXdBUElDbGllbnQoY29uZmlndXJhdGlvbikKCnJlc3AsIHIsIGVyciA6PSBhcGlfY2xpZW50LkRlZmF1bHRBcGkuVXNlckdldChhdXRoKS5BZ2VudChhZ2VudCkuRXhlY3V0ZSgpCmBgYAoKIyMgUG93ZXJlZCBieQoKKiBbb3BlbmFwaS1nZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IpCg== readmeEtag: '"5b7ab51e71903f90a406d8669782747211e67aa1"' readmeLastModified: Wed, 19 Jan 2022 16:27:34 GMT repositoryId: 447968956 description: Alldebrid OpenAPI based Golang client wrapper created: '2022-01-14T13:00:45Z' updated: '2025-08-17T07:34:08Z' language: HTML archived: false stars: 11 watchers: 1 forks: 2 owner: pierre-emmanuelJ logo: https://avatars.githubusercontent.com/u/15922119?v=4 license: GPL-3.0 repoEtag: '"0b8d8732834b461debd0e7036f8e79f8720254513526e8190ac7d489bbd6b3d6"' repoLastModified: Sun, 17 Aug 2025 07:34:08 GMT foundInMaster: true category: Low-level Tooling id: 50de2f05b7baa8acd10b8749004070ed - source: openapi3 tags repository: https://github.com/m3-moretv/smogger v3: true repositoryMetadata: base64Readme: >- IyMg0KfRgtC+INGN0YLQvj8K0JjQvdGB0YLRgNGD0LzQtdC90YIg0LTQu9GPINCw0LLRgtC+0LzQsNGC0LjRh9C10YHQutC+0LPQviDRgdC+0LfQtNCw0L3QuNGPINC80L7Qui3RgdC10YDQstC10YDQsDog0LjQvNC40YLQsNGG0LjQuCDRgNC10LDQu9GM0L3QvtCz0L4gQVBJLArQvtC/0LjRgdCw0L3QvdC+0LPQviDQv9C+INGB0L/QtdGG0LjRhNC40LrQsNGG0LjQuCBbT3BlbiBBUEkgdjNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kKS4KCiMjINCX0LDQstC40YHQuNC80L7RgdGC0LgKLSBub2RlLmpzID49IDkKLSBPcGVuIEFQSSDRgdC/0LXRhtC40YTQuNC60LDRhtC40Y8gKHlhbWwvanNvbikKCiMjINCj0YHRgtCw0L3QvtCy0LrQsCDQuCDQt9Cw0L/Rg9GB0LoKLSBgbnBtIGkgQG0zLW1vcmV0di9zbW9nZ2VyYAotIGBucHggQG0zLW1vcmV0di9zbW9nZ2VyIC1zIC4vc3BlYy55YW1sYAotIGBucHggQG0zLW1vcmV0di9zbW9nZ2VyIC0taGVscGAg0L/QvtC60LDQttC10YIg0YHQv9C40YHQvtC6INC+0L/RhtC40LkKCtCd0LAg0L/QvtGA0YLRgyBgOjMwMDBgINC30LDQv9GD0YHRgtC40YLRgdGPIGh0dHAg0YHQtdGA0LLQtdGALCDQutC+0YLQvtGA0YvQuSDQsdGD0LTQtdGCINC/0L7Qu9C90L7RgdGC0YzRjiDRjdC80YPQu9C40YDQvtCy0LDRgtGMCtC+0L/QuNGB0LDQvdC90L7QtSDQsiDRgdC/0LXQutC1IEFQSS4g0JLRgdC1IGVuZHBvaW50cyDQsdGD0LTRg9GCINCy0L7Qt9Cy0YDQsNGJ0LDRgtGMINC80L7QtNC10LvQuCwg0L7Qv9C40YHQsNC90L3Ri9C1INCyIArRgdC/0LXRhtC40YTQuNC60LDRhtC40Lgg0YEg0YDQsNC90LTQvtC80L3Ri9C80Lgg0LfQvdCw0YfQtdC90LjRj9C80LgsINGB0LPQtdC90LXRgNC10L3QvdGL0LzQuCDRgSDQv9C+0LzQvtGJ0YzRjiBbRmFrZXIuanNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXJhay9GYWtlci5qcy8pCgoqKtCd0LDQv9GA0LjQvNC10YAg0Y3RgtCwINGB0YXQtdC80LA6KioKYGBgeWFtbAp0eXBlOiBvYmplY3QKcmVxdWlyZWQ6CiAgICAtIGlkCiAgICAtIG5hbWUKcHJvcGVydGllczoKICBpZDoKICAgIHR5cGU6IGludGVnZXIKICAgIGZvcm1hdDogaW50NjQKICBuYW1lOgogICAgdHlwZTogc3RyaW5nCiAgICBmb3JtYXQ6IG5hbWUuZmlyc3ROYW1lCiAgdGFnOgogICAgdHlwZTogc3RyaW5nCiAgICBmb3JtYXQ6IHJhbmRvbS53b3JkCmBgYAoqKtCS0LXRgNC90LXRgiDRgtCw0LrQuNC1INC00LDQvdC90YvQtToqKgpgYGBqc29uCnsKICAiaWQiOiAyMzEyMzEyMzEyMywKICAibmFtZSI6ICJQZXRyYSIsCiAgInRhZyI6ICJGb3Jlc3QiCn0KYGBgCgojIyDQmNGB0L/QvtC70YzQt9C+0LLQsNC90LjQtQoKIyMjINCk0L7RgNC80LDRgiDQtNCw0L3QvdGL0YUK0JTQu9GPINGB0L7Qt9C00LDQvdC40Y8g0LTQsNC90L3Ri9GFINC/0YDQuNCx0LvQuNC20LDQvdC90YvRhSDQuiDRgNC10LDQu9GM0L3QvtC80YMgQVBJINC40YHQv9C+0LvRjNC30YPQtdGC0YHRjyBbRmFrZXIuanNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXJhay9GYWtlci5qcy8pLArQutC+0YLQvtGA0YvQuSDQs9C10L3QtdGA0LjRgiDRgdC70YPRh9Cw0LnQvdGL0LUg0LTQsNC90L3Ri9C1INC90LAg0L7RgdC90L7QstC1INC/0L7Qu9GPIGBmb3JtYXRgLiDQndCw0L/RgNC40LzQtdGAINC00LvRjyDRgtC+0LPQviDRh9GC0L4K0LHRiyDQsiDRgdGC0YDQvtC60LUg0LHRi9C7IGVtYWlsINC90LXQvtCx0YXQvtC00LjQvNC+INCyINC/0L7Qu9C1IGB0eXBlYCDRg9C60LDQt9Cw0YLRjCBgc3RyaW5nYCwK0LAg0LIg0L/QvtC70LUgYGZvcm1hdGAg0YPQutCw0LfQsNGC0YwgYGludGVybmV0LmVtYWlsYC4KCtCf0L7Qu9C90YvQuSDRgdC/0LjRgdC+0Log0YTQvtGA0LzQsNGC0L7QsiDQvNC+0LbQvdC+INC90LDQudGC0LggW9GC0YPRgl0oaHR0cHM6Ly9yYXdnaXQuY29tL01hcmFrL2Zha2VyLmpzL21hc3Rlci9leGFtcGxlcy9icm93c2VyL2luZGV4Lmh0bWwpLgoKIyMjINCb0LjQvNC40YLRiwrQotCw0Log0LbQtSDQv9C+0LTQtNC10YDQttC40LLQsNGO0YLRgdGPINCy0YHQtSDQvtCz0YDQsNC90LjRh9C10L3QuNGPIE9wZW4gQVBJINGB0L/QtdGG0LjRhNC40LrQsNGG0LjQuDoKLSBtaW5MZW5ndGgKLSBtYXhMZW5ndGgKLSBtaW5pbXVtCi0gbWF4aW11bQotIC4uLiBlLnQuYy4KCtCd0LDQv9GA0LjQvNC10YAg0LTQu9GPINGC0L7Qs9C+INGH0YLQviDQsdGLIEFQSSDQstC+0LfQstGA0LDRidCw0LvQviDQstC+0LfRgNCw0YHRgiDQvtGCIDEwINC00L4gMTgg0LzRiyDQvNC+0LbQtdC8INC+0L/QuNGB0LDRgtGMINGC0LDQutGD0Y4g0YHRhdC10LzRgzoKYGBgeWFtbAp0eXBlOiBudW1iZXIKbWluaW11bTogMTAKbWF4aW11bTogMTgKYGBgCgrQotC+INC20LUg0YHQsNC80L7QtSDQvNC+0LbQvdC+INGB0LTQtdC70LDRgtGMINGBINC00LvQuNC90L3QvtC5INGB0YLRgNC+0LrQuCwg0LrQvtC70LvQuNGH0LXRgdGC0LLQvtC8INGN0LvQtdC80LXQvdGC0L7QsiDQsiDQvNCw0YHRgdC40LLQtQrQuNC70Lgg0LjRgdC/0L7Qu9GM0LfQvtCy0LDRgtGMIFtlbnVtXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2RhdGEtbW9kZWxzL2VudW1zLykKCtCn0YPRgtGMINCx0L7Qu9GM0YjQtSDQviDRgtC40L/QsNGFINC4INC40YUg0L7Qs9GA0LDQvdC40YfQtdC90LjRj9GFINC80L7QttC90L4g0L/RgNC+0YfQuNGC0LDRgtGMINCyINGB0L/QtdC60LUgW09wZW4gQVBJXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2RhdGEtbW9kZWxzL2RhdGEtdHlwZXMvKQoKIyMg0JjQt9C+0LHRgNCw0LbQtdC90LjRjwpTbW9nZ2VyINGD0LzQtdC10YIg0LPQtdC90LXRgNC40YLRjCDRgdGB0YvQu9C60Lgg0L3QsCDRgNCw0L3QtNC+0LzQvdGL0LUg0LjQt9C+0LHRgNCw0LbQtdC90LjRjywg0LjRgdC/0L7Qu9GM0LfRg9GPINC60LDQutC+0Lkg0LvQuNCx0L4K0LjQtyDQvtGC0LrRgNGL0YLRi9GFINGB0LXRgNCy0LjRgdC+0LIg0YDQsNC90LTQvtC80L3Ri9GFINC60LDRgNGC0LjQvdC+0LogKNC/0L4g0LTQtdGE0L7Qu9GC0YMgaHR0cHM6Ly9waWNzdW0ucGhvdG9zKS4KCtCh0LXRgNCy0LjRgSDQvNC+0LbQvdC+INC30LDQvNC10L3QuNGC0Ywg0YfQtdGA0LXQtyDQutC+0L3RhNC40LMsINC/0LXRgNC10LTQsNCyINCyINC/0LDRgNCw0LzQtdGC0YDQtSBgLWlgINGB0YHRi9C70LrRgyDRgdC10YDQstC40YEuCtCV0YHQu9C4INGB0LXRgNCy0LjRgSDQv9C+0LTQtNC10YDQttC40LLQsNC10YIg0YPQutCw0LfQsNC90LjQtSDQsiB1cmwg0YDQsNC30LXQvNC10YDRiyDQutCw0YDRgtC40L3QvtC6INC80L7QttC90L4K0YPQutCw0LfQsNGC0Ywg0Y3RgtC+INCyINGE0L7RgNC80LDRgtC1IGBodHRwczovL3BpY3N1bS5waG90b3MvPHdpZHRoPi88aGVpZ2h0Pi8/cmFuZG9tYC4KCiMjIEFUVEVOVElPTgrQodC10LnRh9Cw0YEg0LPQtdC90LXRgNGP0YLRgdGPINGC0L7Qu9GM0LrQviDQvtGC0LLQtdGC0YsgYXBwbGljYXRpb24vanNvbiDRgSDQutC+0LTQvtC8IDIwMC4g0JIg0LTQsNC70YzQvdC10LnRiNC10LwgCtGN0YLQviDQsdGD0LTQtdGCINC00L7RgNCw0LHQvtGC0LDQvdC+LgoK0KLQsNC6INC20LUg0L3QsCDQtNCw0L3QvdGL0Lkg0LzQvtC80LXQvdGCIEFQSSDQvdC1INGD0LzQtdC10YIg0YDQsNCx0L7RgtCw0YLRjCDRgSBmbG9hdC4KCiMjIENvbnRyaWJ1dG9ycwrQkiDQv9GA0L7QtdC60YLQtSDQuNGB0L/QvtC70YzQt9GD0LXRgtGB0Y8geWFybiDQuCBmbG93LgoKX1N1cHBvcnRlZCBieSBNb3JlVFYgd2l0aCDinaTvuI9fCg== readmeEtag: '"8d63019c9f0bced89b13af820eb7084ccd9275bf"' readmeLastModified: Tue, 26 Mar 2019 15:20:23 GMT repositoryId: 161340825 description: Simple mock server for Swagger (Open API v3) specification created: '2018-12-11T13:49:02Z' updated: '2025-02-06T06:02:48Z' language: TypeScript archived: false stars: 10 watchers: 3 forks: 1 owner: m3-moretv logo: https://avatars.githubusercontent.com/u/45791851?v=4 repoEtag: '"adb7956ce4629ebe4ca242934124c0f72c507bc70b138a42f59c73bc0ff718bd"' repoLastModified: Thu, 06 Feb 2025 06:02:48 GMT foundInMaster: true category: Mock id: 3dffde367a70e8d59e69bc8be4e07f27 - source: openapi3 tags repository: https://github.com/matchid-project/deces-backend v3: true repositoryMetadata: base64Readme: >- IyBBUEkgUGVyc29ubmVzIETDqWPDqWTDqWVzIDxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vbWF0Y2hJRC1wcm9qZWN0L2RlY2VzLXVpL3Jhdy9kZXYvcHVibGljL2Zhdmljb24uc3ZnIiB3aWR0aD0iMTgwIiBhbGlnbj0icmlnaHQiIC8+CgohW0J1aWxkIHN0YXR1c10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvYWN0aW9ucy93b3JrZmxvdy9zdGF0dXMvbWF0Y2hpZC1wcm9qZWN0L2RlY2VzLWJhY2tlbmQvZG9ja2VyaW1hZ2UueW1sKSBbIVtMaWNlbnNlOiBMR1BMIHYzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtTEdQTCUyMHYzLWJsdWUuc3ZnKV0oaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9sZ3BsLTMuMCkgIVtEb2NrZXIgUHVsbHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZG9ja2VyL3B1bGxzL21hdGNoaWQvZGVjZXMtYmFja2VuZD9sYWJlbD1Eb2NrZXIlMjBwdWxscykKCkFQSSBmb3IgcGVvcGxlIHdobyBkaWVkIGluIEZyYW5jZSBmcm9tIDE5NzAuCgojIyBGZWF0dXJlcwoKKiBNdWx0aXBsZSBmaWVsZHMgY2FuIGJlIHVzZWQgYXMgYSBmaWx0ZXI6CiAgKiBGaXJzdC9MYXN0IG5hbWUKICAqIEJpcnRoL0RlYXRoIGRhdGUKICAqIEJpcnRoL0RlYXRoIGxvY2F0aW9uIChjb3VudHJ5LCBkZXBhcnRtZW50LCBjaXR5LCBncHMgY29vcmRpbmF0ZSkKICAqIEFnZQogICogU2V4CiogVGhlIEFQSSBjYW4gaGFuZGxlIGNvbW1vbiBtaXN0YWtlcyBpbiBmaWVsZHMgdGhhbmtzIHRvIGZ1enp5IG1hdGNoaW5nCiogQnVsayBwcm9jZXNzIChDU1Ygb3IgYnVsayBKU09OKSBmb3IgbXVsdGlwbGUgaWRlbnRpdGllcwoqIE9wdGlvbmFsIHdlYmhvb2sgY2FsbGJhY2tzIHdpdGggYSBjaGFsbGVuZ2UgdmFsaWRhdGlvbiBlbmRwb2ludAoqIEV4cHJlc3MgZnJhbWV3b3JrIGZvciBSRVNUIEFQSQoqIE9wZW5BUEl2MyBkb2N1bWVudGF0aW9uIGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGVkIHVzaW5nCiAgW1RTT0FdKGh0dHBzOi8vZ2l0aHViLmNvbS9sdWtlYXV0cnkvdHNvYSkKKiBEb2NrZXIgaW1hZ2UgaXMgcHVibGlzaGVkIGF0IFtkb2NrZXIKICBodWJdKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9tYXRjaGlkL2RlY2VzLWJhY2tlbmQpIHVzaW5nIEdpdEh1YiBBY3Rpb25zLgoKRGV0YWlsZWQgZG9jdW1lbnRhdGlvbiBpcyBhdmFpbGFibGUgYXQgW3RoaXMgc3dhZ2dlciBwYWdlXShodHRwczovL2RlY2VzLm1hdGNoaWQuaW8vZGVjZXMvYXBpL3YxL2RvY3MpCgojIyBJbnN0YWxsYXRpb24KCkluc3RhbGwgdXNpbmcgZG9ja2VyIGNvbXBvc2U6CgpgYGBiYXNoCm1ha2UgdXAKYGBgCgpCZWZvcmUgc3RhcnRpbmcgYSBkYXRhYmFzZSBoYXMgdG8gYmUgY2hhcmdlZCBpbiBlbGFzdGljc2VhcmNoLiBQbGVhc2UgcmVmZXIgdG8KW2RhdGFwcmVwIHJlcG9zaXRvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXRjaElELXByb2plY3QvZGVjZXMtZGF0YXByZXApLgoKIyMgUHJvamVjdCByZXNvdXJjZXMKCiogW1NvdXJjZSBjb2RlXShodHRwczovL2dpdGh1Yi5jb20vbWF0Y2hpZC1wcm9qZWN0L2RlY2VzLWJhY2tlbmQpCiogW0lzc3VlIHRyYWNrZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXRjaGlkLXByb2plY3QvZGVjZXMtYmFja2VuZC9pc3N1ZXMpCgojIyBMaWNlbnNlCgpTb3VyY2UgY29kZSBoYXMgYmVlbiBwdWJsaXNoZWQgdXNpbmcgW0xHUEwgMy4wXShodHRwczovL2dpdGh1Yi5jb20vbWF0Y2hJRC1wcm9qZWN0L2RlY2VzLWJhY2tlbmQvYmxvYi9kZXYvTElDRU5DRSkuCgrCqSAyMDIwIENyaXN0aWFuIEJyb2thdGUsIEROVU0gLSBTRElUCg== readmeEtag: '"d89b6e1896ad8ee309b76840808d491d0d96f85c"' readmeLastModified: Sun, 08 Jun 2025 20:07:53 GMT repositoryId: 242171506 description: 'API Personnes décédées ' created: '2020-02-21T15:33:05Z' updated: '2026-02-05T01:52:42Z' language: TypeScript archived: false stars: 11 watchers: 2 forks: 8 owner: matchID-project logo: https://avatars.githubusercontent.com/u/32673019?v=4 license: LGPL-3.0 repoEtag: '"d1e994746879965d3324e0bda3a9d64d671b9d3ad81c345cac3668395ef0fa4f"' repoLastModified: Thu, 05 Feb 2026 01:52:42 GMT foundInMaster: true category: Server id: a1b78d0ba8dd46d8098f77bf077b6cac - source: openapi3 tags repository: https://github.com/shpoont/openapi-jsonapi-definition v3: true id: dec8a69f6989a95f7dc07c1775cb67b6 repositoryMetadata: base64Readme: >- W09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIGRlZmluaXRpb25zIG9mIFtKU09OOkFQSV0oaHR0cHM6Ly9qc29uYXBpLm9yZy8pCgojIE9wZW5BcGkgdjMuMC4wCi0gW0pTT046QVBJIHYxLjFdKGRlZmluaXRpb25zL2RyYWZ0Lm9wZW5hcGktMy4wLjAtanNvbmFwaS0xLjEueWFtbCkgKGRyYWZ0KQo= readmeEtag: '"84a95ad97ea20769698662439971465469ee01c6"' readmeLastModified: Sat, 26 Nov 2022 12:06:52 GMT repositoryId: 560845522 description: OpenAPI definitions of JSON:API created: '2022-11-02T11:57:00Z' updated: '2024-05-23T19:51:45Z' language: null archived: false stars: 9 watchers: 1 forks: 0 owner: shpoont logo: https://avatars.githubusercontent.com/u/1372784?v=4 repoEtag: '"8345bc8dd9261f1d2fdf8f41602d13b579b55d3cd7e57ea231a97c505a96f0b7"' repoLastModified: Thu, 23 May 2024 19:51:45 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/taskiq-python/aiohttp-deps v3: true id: 9930662c3938ea0a73abd4e4169f027f repositoryMetadata: base64Readme: >- [![PyPI - Python Version](https://img.shields.io/pypi/pyversions/aiohttp-deps?style=for-the-badge)](https://pypi.org/project/aiohttp-deps/)
[![PyPI](https://img.shields.io/pypi/v/aiohttp-deps?style=for-the-badge)](https://pypi.org/project/aiohttp-deps/)
[![PyPI - Downloads](https://img.shields.io/pypi/dm/aiohttp-deps?style=for-the-badge)](https://pypistats.org/packages/aiohttp-deps)

# AioHTTP deps


This project was initially created to show the abillities of [taskiq-dependencies](https://github.com/taskiq-python/taskiq-dependencies) project, which is used by [taskiq](https://github.com/taskiq-python/taskiq) to provide you with the best experience of sending distributed tasks.

This project adds [FastAPI](https://github.com/tiangolo/fastapi)-like dependency injection to your [AioHTTP](https://github.com/aio-libs/aiohttp) application and swagger documentation based on types.

To start using dependency injection, just initialize the injector.

```python
from aiohttp import web
from aiohttp_deps import init as deps_init


app = web.Application()


app.on_startup.append(deps_init)

web.run_app(app)

```


If you use mypy, then we have a custom router with propper types.


```python
from aiohttp import web
from aiohttp_deps import init as deps_init
from aiohttp_deps import Router

router = Router()


@router.get("/")
async def handler():
    return web.json_response({})


app = web.Application()

app.router.add_routes(router)

app.on_startup.append(deps_init)

web.run_app(app)

```

Also, you can nest routers with prefixes,

```python
api_router = Router()

memes_router = Router()

main_router = Router()

main_router.add_routes(api_router, prefix="/api")
main_router.add_routes(memes_router, prefix="/memes")
```

## Swagger

If you use dependencies in you handlers, we can easily generate swagger for you.
We have some limitations:
1. We don't support resolving type aliases if hint is a string.
    If you define variable like this: `myvar = int | None` and then in handler
    you'd create annotation like this: `param: "str | myvar"` it will fail.
    You need to unquote type hint in order to get it work.

We will try to fix these limitations later.

To enable swagger, just add it to your startup.

```python
from aiohttp_deps import init, setup_swagger

app = web.Application()

app.on_startup.extend([init, setup_swagger()])
```

### Responses

You can define schema for responses using dataclasses or
pydantic models. This would not affect handlers in any way,
it's only for documentation purposes, if you want to actually
validate values your handler returns, please write your own wrapper.

```python
from dataclasses import dataclass

from aiohttp import web
from pydantic import BaseModel

from aiohttp_deps import Router, openapi_response

router = Router()


@dataclass
class Success:
    data: str


class Unauthorized(BaseModel):
    why: str


@router.get("/")
@openapi_response(200, Success, content_type="application/xml")
@openapi_response(200, Success)
@openapi_response(401, Unauthorized, description="When token is not correct")
async def handler() -> web.Response:
    ...
```

This example illustrates how much you can do with this decorator. You
can have multiple content-types for a single status, or you can have different
possble statuses. This function is pretty simple and if you want to make
your own decorator for your responses, it won't be hard.


## Default dependencies

By default this library provides only two injectables. `web.Request` and `web.Application`.

```python

async def handler(app: web.Application = Depends()): ...

async def handler2(request: web.Request = Depends()): ...

```

It's super useful, because you can use these dependencies in
any other dependency. Here's a more complex example of how you can use this library.


```python
from aiohttp_deps import Router, Depends
from aiohttp import web

router = Router()


async def get_db_session(app: web.Application = Depends()):
    async with app[web.AppKey("db")] as sess:
        yield sess


class MyDAO:
    def __init__(self, session=Depends(get_db_session)):
        self.session = session

    async def get_objects(self) -> list[object]:
        return await self.session.execute("SELECT 1")


@router.get("/")
async def handler(db_session: MyDAO = Depends()):
    objs = await db_session.get_objects()
    return web.json_response({"objects": objs})

```

If you do something like this, you would never think about initializing your DAO. You can just inject it and that's it.


# Built-in dependencies

This library also provides you with some default dependencies that can help you in building the best web-service.

## Json

To parse json, create a pydantic model and add a dependency to your handler.


```python
from aiohttp import web
from pydantic import BaseModel
from aiohttp_deps import Router, Json, Depends

router = Router()


class UserInfo(BaseModel):
    name: str


@router.post("/users")
async def new_data(user: UserInfo = Depends(Json())):
    return web.json_response({"user": user.model_dump()})

```

This dependency automatically validates data and send
errors if the data doesn't orrelate with schema or body is not a valid json.

If you want to make this data optional, just mark it as optional.

```python
@router.post("/users")
async def new_data(user: Optional[UserInfo] = Depends(Json())):
    if user is None:
        return web.json_response({"user": None})
    return web.json_response({"user": user.model_dump()})

```

## Headers

You can get and validate headers using `Header` dependency.

Let's try to build simple example for authorization.

```python
from aiohttp_deps import Router, Header, Depends
from aiohttp import web

router = Router()


def decode_token(authorization: str = Depends(Header())) -> str:
    if authorization == "secret":
        # Let's pretend that here we
        # decode our token.
        return authorization
    raise web.HTTPUnauthorized()


@router.get("/secret_data")
async def new_data(token: str = Depends(decode_token)) -> web.Response:
    return web.json_response({"secret": "not a secret"})

```

As you can see, header name to parse is equal to the
name of a parameter that introduces Header dependency.

If you want to use some name that is not allowed in python, or just want to have different names, you can use alias. Like this:

```python
def decode_token(auth: str = Depends(Header(alias="Authorization"))) -> str:
```

Headers can also be parsed to types. If you want a header to be parsed as int, just add the typehint.

```python
def decode_token(meme_id: int = Depends(Header())) -> str:
```

If you want to get list of values of one header, use parameter `multiple=True`.

```python
def decode_token(meme_id: list[int] = Depends(Header(multiple=True))) -> str:

```

And, of course, you can provide this dependency with default value if the value from user cannot be parsed for some reason.

```python
def decode_token(meme_id: str = Depends(Header(default="not-a-secret"))) -> str:
```


## Queries

You can depend on `Query` to get and parse query parameters.

```python
from aiohttp_deps import Router, Query, Depends
from aiohttp import web

router = Router()


@router.get("/shop")
async def shop(item_id: str = Depends(Query())) -> web.Response:
    return web.json_response({"id": item_id})

```

the name of the parameter is the same as the name of function parameter.

The Query dependency is acually the same as the Header dependency, so everything about the `Header` dependency also applies to `Query`.

## Views

If you use views as handlers, please use View class from `aiohttp_deps`, otherwise the magic won't work.

```python
from aiohttp_deps import Router, View, Depends
from aiohttp import web

router = Router()


@router.view("/view")
class MyView(View):
    async def get(self, app: web.Application = Depends()):
        return web.json_response({"app": str(app)})

```


## Forms

Now you can easiy get and validate form data from your request.
To make the magic happen, please add `arbitrary_types_allowed` to the config of your model.


```python
import pydantic
from aiohttp_deps import Router, Depends, Form
from aiohttp import web

router = Router()


class MyForm(pydantic.BaseModel):
    id: int
    file: web.FileField

    model_config = pydantic.ConfigDict(arbitrary_types_allowed=True)


@router.post("/")
async def handler(my_form: MyForm = Depends(Form())):
    with open("my_file", "wb") as f:
        f.write(my_form.file.file.read())
    return web.json_response({"id": my_form.id})

```

## Path

If you have path variables, you can also inject them in your handler.

```python
from aiohttp_deps import Router, Path, Depends
from aiohttp import web

router = Router()


@router.get("/view/{var}")
async def my_handler(var: str = Depends(Path())):
    return web.json_response({"var": var})

```


## ExtraOpenAPI

This dependency is used to add additional swagger fields to the endpoint's swagger
that is using this dependency. It might be even indirect dependency.

You can check how this thing can be used in our [examples/swagger_auth.py](https://github.com/taskiq-python/aiohttp-deps/tree/master/examples/swagger_auth.py).


## Overriding dependencies

Sometimes for tests you don't want to calculate actual functions
and you want to pass another functions instead.

To do so, you can add "dependency_overrides" or "values_overrides" to the aplication's state.
These values should be dicts. The keys for these values can be found in `aiohttp_deps.keys` module.

Here's an example.

```python
def original_dep() -> int:
    return 1

class MyView(View):
    async def get(self, num: int = Depends(original_dep)):
        """Nothing."""
        return web.json_response({"request": num})
```

Imagine you have a handler that depends on some function,
but instead of `1` you want to have `2` in your tests.

To do it, jsut add `dependency_overrides` somewhere,
where you create your application. And make sure that keys
of that dict are actual function that are being replaced.

```python
from aiohttp_deps import VALUES_OVERRIDES_KEY

my_app[VALUES_OVERRIDES_KEY] = {original_dep: 2}
```

But `values_overrides` only overrides returned values. If you want to
override functions, you have to use `dependency_overrides`. Here's an example:

```python
from aiohttp_deps import DEPENDENCY_OVERRIDES_KEY

def replacing_function() -> int:
    return 2


my_app[DEPENDENCY_OVERRIDES_KEY] = {original_dep: replacing_function}
```

The cool point about `dependency_overrides`, is that it recalculates graph and
you can use dependencies in function that replaces the original.
 readmeEtag: '"d2a9fc86a3baf1bb870a15fa102ef06a0c0e3399"' readmeLastModified: Mon, 30 Sep 2024 23:19:15 GMT repositoryId: 630692437 description: Dependency injection for AioHTTP created: '2023-04-21T00:08:54Z' updated: '2025-08-29T11:39:06Z' language: Python archived: false stars: 11 watchers: 1 forks: 0 owner: taskiq-python logo: https://avatars.githubusercontent.com/u/109470701?v=4 license: MIT repoEtag: '"c62b18886ce8e93ef668a15ae3fdf8e05b6bbf3bd68ad41956ff48f46d2bc921"' repoLastModified: Fri, 29 Aug 2025 11:39:06 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/postman-open-technologies/openapi-linter v3: true repositoryMetadata: base64Readme: >- IyBzcGVjLWxpbnRlcgoKWyFbRGVwbG95XShodHRwczovL3d3dy5oZXJva3VjZG4uY29tL2RlcGxveS9idXR0b24uc3ZnKV0oaHR0cHM6Ly9oZXJva3UuY29tL2RlcGxveSkKWyFbRGVwbG95IHRvIEF6dXJlXShodHRwczovL2FrYS5tcy9kZXBsb3l0b2F6dXJlYnV0dG9uKV0oaHR0cHM6Ly9wb3J0YWwuYXp1cmUuY29tLyNjcmVhdGUvTWljcm9zb2Z0LlRlbXBsYXRlL3VyaS9odHRwcyUzQSUyRiUyRnJhdy5naXRodWJ1c2VyY29udGVudC5jb20lMkZwb3N0bWFuLW9wZW4tdGVjaG5vbG9naWVzJTJGb3BlbmFwaS1saW50ZXIlMkZtYWluJTJGZGVwbG95bWVudHMlMkZhenVyZS1mdW5jdGlvbnMlMkZ0ZW1wbGF0ZS5qc29uKQoKW0RlcGxveSB0byBBV1NdKGh0dHBzOi8vdXMtZWFzdC0yLmNvbnNvbGUuYXdzLmFtYXpvbi5jb20vY2xvdWRmb3JtYXRpb24vaG9tZT9yZWdpb249dXMtZWFzdC0yIy9zdGFja3MvY3JlYXRlL3RlbXBsYXRlP3N0YWNrTmFtZT1zcGVjLWxpbnRlci1hcGkmdGVtcGxhdGVVUkw9aHR0cHM6Ly9wdWJsaWMtc3BlYy1saW50ZXItYXBpLnMzLnVzLWVhc3QtMi5hbWF6b25hd3MuY29tL3RlbXBsYXRlLmpzb24pCgpBbiBBUEkgZm9yIGxpbnRpbmcgT3BlbkFQSSwgQXN5bmNBUEksIGFuZCBvdGhlciBKU09OL1lBTUwgZG9jdW1lbnRzLCBidWlsdCBvbiB0b3Agb2YgW1NwZWN0cmFsXShodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vc3BlY3RyYWwpLgoKIyMg8J+boCBVc2FnZQoKU2VlIHRoZSBwdWJsaXNoZWQgW1Bvc3RtYW4gQ29sbGVjdGlvbl0oaHR0cHM6Ly93d3cucG9zdG1hbi5jb20vcG9zdG1hbi93b3Jrc3BhY2Uvb3BlbmFwaS1saW50aW5nL2RvY3VtZW50YXRpb24vMTI5NTk1NDItOTNjZDkwZGUtOWYzYy00ZWQwLTljNTctY2M1YTc3MTJjZjE5KSBmb3IgbW9yZSBpbmZvcm1hdGlvbiBvbiBpbnRlcmFjdGluZyB3aXRoIHRoZSBzcGVjLWxpbnRlciBBUEkuCgpTcGVjdHJhbCBjb25maWcgbXVzdCBiZSBob3N0ZWQgYXQgYSBVUkwuIFRoaXMgQVBJIHN1cHBvcnRzIGNvbmZpZ3VyYXRpb24gaW4gdGhlIGZvbGxvd2luZyBmb3JtczoKCi0gSlNPTi9ZQU1MIGZpbGUKLSBTcGVjdHJhbCBKYXZhU2NyaXB0IGNvbmZpZ3VyYXRpb24gKGFscGhhLCB0eXBpY2FsbHkgYC5zcGVjdHJhbC5qc2ApCi0gU3BlY3RyYWwgVHlwZVNjcmlwdCBjb25maWd1cmF0aW9uIChhbHBoYSwgdHlwaWNhbGx5IGAuc3BlY3RyYWwudHNgKQoKIyMg8J+ZjyBUaGFua3MKClRoaXMgcHJvamVjdCB3b3VsZG4ndCBiZSBwb3NzaWJsZSB3aXRob3V0IHRoZSBlZmZvcnRzIG9mIFtTdG9wbGlnaHRdKGh0dHBzOi8vc3RvcGxpZ2h0LmlvL29wZW4tc291cmNlL3NwZWN0cmFsLykgYW5kIHRoZSBbU3BlY3RyYWwgY29udHJpYnV0b3JzXShodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vc3BlY3RyYWwvZ3JhcGhzL2NvbnRyaWJ1dG9ycykuCgojIyDwn5OEIExpY2Vuc2UKCkFwYWNoZS0yLjAK readmeEtag: '"f65e477337af57cb6ced3ff4fd8476a5b4235024"' readmeLastModified: Mon, 07 Feb 2022 18:18:16 GMT repositoryId: 431253781 description: >- An open source implementation of Spectral for applying linting rules to an OpenAPI which can be deployed to AWS, Azure, and Heroku. created: '2021-11-23T21:07:39Z' updated: '2025-11-07T09:47:19Z' language: TypeScript archived: false stars: 10 watchers: 1 forks: 3 owner: postman-open-technologies logo: https://avatars.githubusercontent.com/u/79494470?v=4 license: Apache-2.0 repoEtag: '"d3d40c8525123c670fa19ae15af70e36b551c6bfe5e088819807664251bdd496"' repoLastModified: Fri, 07 Nov 2025 09:47:19 GMT foundInMaster: true category: Description Validators id: 651ef44cefb0498286e46666d0f496cf - source: openapi3 tags repository: https://github.com/networknt/openapi-parser v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXBhcnNlcgoKQSBsaWdodC13ZWlnaHQsIGZhc3QgT3BlbkFQSSAzLjAgcGFyc2VyIGFuZCB2YWxpZGF0b3Igd2l0aCBtaW5pbXVtIHRoaXJkIHBhcnR5IGRlcGVuZGVuY2llcy4KCltTdGFjayBPdmVyZmxvd10oaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9xdWVzdGlvbnMvdGFnZ2VkL2xpZ2h0LTRqKSB8CltHb29nbGUgR3JvdXBdKGh0dHBzOi8vZ3JvdXBzLmdvb2dsZS5jb20vZm9ydW0vIyFmb3J1bS9saWdodC00aikgfApbR2l0dGVyIENoYXRdKGh0dHBzOi8vZ2l0dGVyLmltL25ldHdvcmtudC9saWdodC1yZXN0LTRqKSB8CltTdWJyZWRkaXRdKGh0dHBzOi8vd3d3LnJlZGRpdC5jb20vci9saWdodGFwaS8pIHwKW1lvdXR1YmUgQ2hhbm5lbF0oaHR0cHM6Ly93d3cueW91dHViZS5jb20vY2hhbm5lbC9VQ0hDUk1XSlZYdzhpQjd6S3hGNTVCeXcpIHwKW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vZG9jLm5ldHdvcmtudC5jb20vbGlicmFyeS9vcGVuYXBpLXBhcnNlci8pIHwKW0NvbnRyaWJ1dGlvbiBHdWlkZV0oaHR0cHM6Ly9kb2MubmV0d29ya250LmNvbS9jb250cmlidXRlLykgfAoKSXQgaXMgYmFzZWQgb24gdGhlIGdlbmVyYXRlZCBjb2RlIGZyb20gW0thaVplbiBPcGVuQVBJIFBhcnNlcl0oaHR0cHM6Ly9naXRodWIuY29tL1JlcHJlWmVuL0thaVplbi1PcGVuQXBpLVBhcnNlcikgd2l0aG91dApkZXBlbmRlbmN5IG9uIGphdmFwYXJzZXItY29yZSwgZ3VhdmEsIGNvbW1vbnMtY2xpLCBjb21tb25zLWlvLCBqYXZheC5tYWlsIGFuZCBndWljZS4KCiMjIyBMaWNlbnNlCgpUaGUgT3BlbkFQSSBQYXJzZXIgaXMgcHJvdmlkZWQgdW5kZXIgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMS4wIGZvbGxvd2luZyB0aGUgcGFyZW50IFtLYWlaZW4gT3BlbkFQSSBQYXJzZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZXByZVplbi9LYWlaZW4tT3BlbkFwaS1QYXJzZXIpCg== readmeEtag: '"c7e1dadd4187bd1056766b58e21ac7c57571fb43"' readmeLastModified: Thu, 18 Jul 2024 12:46:15 GMT repositoryId: 110491962 description: >- A light-weight, fast OpenAPI 3.0 parser and validator with minimum third party dependencies created: '2017-11-13T02:47:14Z' updated: '2026-02-03T01:08:21Z' language: Java archived: false stars: 11 watchers: 15 forks: 9 owner: networknt logo: https://avatars.githubusercontent.com/u/8740739?v=4 license: EPL-1.0 repoEtag: '"ab0c5df8b452b0123b87bbf7c9f3c68c34b8ec111c5d61f26210140928230206"' repoLastModified: Tue, 03 Feb 2026 01:08:21 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 326240eecd7b3028ddf9a6cfb4273a0d - source: openapi3 tags repository: https://github.com/ben-jamin-chen/springboot-swagger-rest-api v3: true id: a9df75197b7dc50eefc61eead6539c2b repositoryMetadata: base64Readme: >- IyBTYW1wbGUgU3ByaW5nIEJvb3QgKDIuMy4yKSBSRVNUZnVsIEFQSSB3aXRoIFN3YWdnZXIgKE9wZW5BUEkgMykKCldoaWxlIHNlYXJjaGluZyB0aHJvdWdoIEdpdEh1YiBmb3Igc29tZSBib2lsZXJwbGF0ZSBjb2RlIG9uIGhvdyB0byBzZXR1cCBhIFNwcmluZyBCb290IHByb2plY3Qgd2l0aCBTd2FnZ2VyLCBJIGZvdW5kIGl0IHF1aXRlIGRpZmZpY3VsdCB0byBmaW5kIGEgd29ya2luZyBleGFtcGxlIHdpdGggYSBtb3JlIHJlY2VudCB2ZXJzaW9uIG9mIFNwcmluZyBCb290IGFuZCBKYXZhIChpLmUuIDE0KS4gQW55d2F5cywgSSB0aG91Z2h0IEnigJlkIGNyZWF0ZSBteSBvd24gYW5kIHNoYXJlIHdpdGggZXZlcnlvbmUuIFRoaXMgaXMgZm9yIGFueW9uZSB0aGF0IG5lZWRzIHNvbWUgcXVpY2sgYm9pbGVycGxhdGUgY29kZSB0byBzZXR1cCB0aGVpciBuZXcgQVBJIHByb2plY3QuCgojIyBXaGF0IFlvdSBOZWVkCgoqIEphdmEgMTQKKiBNYXZlbiAzLjYuMCsKCiMjIEJ1aWxkIGFuZCBSdW4gdGhlIFNhbXBsZQoKWW91IGNhbiBpbXBvcnQgdGhlIGNvZGUgc3RyYWlnaHQgaW50byB5b3VyIHByZWZlcnJlZCBJREUgb3IgcnVuIHRoZSBzYW1wbGUgdXNpbmcgdGhlIGZvbGxvd2luZyBjb21tYW5kIChpbiB0aGUgcm9vdCBwcm9qZWN0IGZvbGRlcikuCgpgYGB6c2gKJCAgbXZuIHNwcmluZy1ib290OnJ1bgpgYGAKQWZ0ZXIgdGhlIGFwcGxpY2F0aW9uIHJ1bnMsIG5hdmlnYXRlIHRvIGBodHRwOi8vbG9jYWxob3N0OjcwMDEvc3dhZ2dlci11aS9pbmRleC5odG1sP2NvbmZpZ1VybD0vYXBpLWRvY3Mvc3dhZ2dlci1jb25maWdgIGluIHlvdXIgd2ViIGJyb3dzZXIgdG8gYWNjZXNzIHRoZSBTd2FnZ2VyIFVJIHBvcnRhbC4K readmeEtag: '"9a2b56a24abaa949af685f5503ecdcc4da8a4654"' readmeLastModified: Tue, 28 Jul 2020 00:13:07 GMT repositoryId: 281525992 description: >- A sample RESTful API using Spring Boot (2.3.2) and Java 14 with Swagger enabled. created: '2020-07-21T23:28:43Z' updated: '2024-08-29T06:57:42Z' language: Java archived: false stars: 9 watchers: 1 forks: 6 owner: ben-jamin-chen logo: https://avatars.githubusercontent.com/u/41641688?v=4 license: MIT repoEtag: '"fe0c0420ebe244459c1483a8df2b3a198e89906b137a3e90a430b6c43d5e8821"' repoLastModified: Thu, 29 Aug 2024 06:57:42 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cerberauth/openapi-oathkeeper v3: true id: ba2b06aa4f5864bd7545ae91e08c58a6 repositoryMetadata: base64Readme: >- # Ory Oathkeeper rules from OpenAPI


[![Join Discord](https://img.shields.io/discord/1242773130137833493?label=Discord&style=for-the-badge)](https://www.cerberauth.com/community)
[![GitHub Workflow Status](https://img.shields.io/github/actions/workflow/status/cerberauth/openapi-oathkeeper/ci.yml?branch=main&label=core%20build&style=for-the-badge)](https://github.com/cerberauth/openapi-oathkeeper/actions/workflows/ci.yml)
![Latest version](https://img.shields.io/github/v/release/cerberauth/openapi-oathkeeper?sort=semver&style=for-the-badge)
![Codecov](https://img.shields.io/codecov/c/gh/cerberauth/openapi-oathkeeper?token=BD1WPXJDAW&style=for-the-badge)
[![Go Report Card](https://goreportcard.com/badge/github.com/cerberauth/openapi-oathkeeper?style=for-the-badge)](https://goreportcard.com/report/github.com/cerberauth/openapi-oathkeeper)
[![GoDoc reference](https://img.shields.io/badge/godoc-reference-5272B4.svg?style=for-the-badge)](https://godoc.org/github.com/cerberauth/openapi-oathkeeper)

This CLI generates OathKeeper rules that enforce authentication and authorization policies for each API endpoint from an OpenAPI file.

This project automate the generation of Oathkeeper rules from an OpenAPI contract and save a lot of time especially for larger projects with many endpoints or many services by using the existing documentation provided in an OpenAPI contract. This can improve the overall security of the API and ensure that access is granted only to authorized parties. Additionally, this tool can simplify the development process by reducing the amount of manual work required to write and maintain OathKeeper rules.

## Ory Oathkeeper

If you're not yet familiar with Ory Oathkeeper, Oathkeeper is an Identity & Access Proxy (IAP) and Access Control Decision API that authorizes HTTP requests based on sets of Access Rules. You can find more information and get started with [Ory Oathkeeper](https://github.com/ory/oathkeeper).

> An Identity & Access Proxy is typically deployed in front of (think API Gateway or Service mesh) web-facing applications and is capable of authenticating and optionally authorizing access requests. The Access Control Decision API can be deployed alongside an existing API Gateway or reverse proxy.

## Installation

Below are the instructions to install on Linux, Windows, MacOS, and Docker. You can choose the installation method that best suits your needs and environment.

If none of the installation methods below work for you, you can also download the binary from the latest [release](https://github.com/cerberauth/openapi-oathkeeper/releases).

### Linux (Snap)

Install using [Snap](https://snapcraft.io/openapi-oathkeeper).

```sh
sudo snap install vulnapi
```

### MacOS (Homebrew)

Install using Homebrew.

```sh
brew tap cerberauth/openapi-oathkeeper https://github.com/cerberauth/openapi-oathkeeper
brew install $(brew --repository cerberauth/openapi-oathkeeper)/openapi-oathkeeper.rb
```

## Get Started

Provides the path to your OpenAPI contract file.

```sh
./openapi-oathkeeper generate -f ./openapi.json
```

Once you have specified these options, the tool will analyze your contract and generate OathKeeper rules that enforce the specified access policies. You can then save these rules to a file to make it read by Oathkeeper.

Here is an example Oathkeeper rules output from the [Petstore OpenAPI](./test/stub/petstore.openapi.json)

<details>
    <summary>Oathkeeper rules output</summary>

```json
[
    {
        "id": "addPet",
        "version": "",
        "description": "Add a new pet to the store",
        "match": {
            "methods": [
                "POST"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "createUser",
        "version": "",
        "description": "This can only be done by the logged in user.",
        "match": {
            "methods": [
                "POST"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "createUsersWithListInput",
        "version": "",
        "description": "Creates list of users with given input array",
        "match": {
            "methods": [
                "POST"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user/createWithList"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "deleteOrder",
        "version": "",
        "description": "For valid response try integer IDs with value < 1000. Anything above 1000 or nonintegers will generate API errors",
        "match": {
            "methods": [
                "DELETE"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/store/order/<\\d+>"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "deletePet",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "DELETE"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet/<\\d+>"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "deleteUser",
        "version": "",
        "description": "This can only be done by the logged in user.",
        "match": {
            "methods": [
                "DELETE"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user/<.+>"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "findPetsByStatus",
        "version": "",
        "description": "Multiple status values can be provided with comma separated strings",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet/findByStatus"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "findPetsByTags",
        "version": "",
        "description": "Multiple tags can be provided with comma separated strings. Use tag1, tag2, tag3 for testing.",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet/findByTags"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "getInventory",
        "version": "",
        "description": "Returns a map of status codes to quantities",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/store/inventory"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "getOrderById",
        "version": "",
        "description": "For valid response try integer IDs with value <= 5 or > 10. Other values will generate exceptions.",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/store/order/<\\d+>"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "getPetById",
        "version": "",
        "description": "Returns a single pet",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet/<\\d+>"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "getUserByName",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user/<.+>"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "loginUser",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user/login"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "logoutUser",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user/logout"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "placeOrder",
        "version": "",
        "description": "Place a new order in the store",
        "match": {
            "methods": [
                "POST"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/store/order"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "updatePet",
        "version": "",
        "description": "Update an existing pet by Id",
        "match": {
            "methods": [
                "PUT"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "updatePetWithForm",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "POST"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet/<\\d+>"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "updateUser",
        "version": "",
        "description": "This can only be done by the logged in user.",
        "match": {
            "methods": [
                "PUT"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/user/<.+>"
        },
        "authenticators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "uploadFile",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "POST"
            ],
            "url": "<(https://cerberauth\\.com/api/v3|http://swagger\\.io/api/v3)>/pet/<\\d+>/uploadImage"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "required_scope": [
                        "write:pets",
                        "read:pets"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": null,
        "errors": null,
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    }
]
```
</details>

## Configuration

As the authenticator rule may require additional information in order to make authorization and authentication working properly, additional information can be passed either by OpenAPI Extensions or configuration file.

### Configuration File

The recommended approach involves using dedicated configuration files for your Oathkeeper rules. These configuration files provide a more flexible and user-friendly way of managing your security settings.

Every Oathkeeper rule property can be configured this way. Here are the available properties:

| Field          | Type                                                                               | Key              |
|----------------|------------------------------------------------------------------------------------|------------------|
| Prefix         | string                                                                             | "prefix"         |
| ServerUrls     | []string                                                                           | "server_urls"    |
| Upstream       | [Upstream](https://www.ory.sh/docs/oathkeeper/api-access-rules#access-rule-format) | "upstream"       |
| Authenticators | Map of [Authenticators](https://www.ory.sh/docs/oathkeeper/pipeline/authn)         | "authenticators" |
| Authorizer     | [Authorization Handler](https://www.ory.sh/docs/oathkeeper/pipeline/authz)         | "authorizer"     |
| Mutators       | Array [of Mutator Handlers](https://www.ory.sh/docs/oathkeeper/pipeline/mutator)   | "mutators"       |
| Errors         | Array of [Error Handlers](https://www.ory.sh/docs/oathkeeper/pipeline/error)       | "errors"         |

Below is an example of a configuration file in YAML format:

```yaml
prefix: cerberauth

server_urls:
  - https://www.cerberauth.com/api
  - https://api.cerberauth.com/api

authenticators:
  openidconnect:
    handler: "jwt"
    config:
      target_audience:
      - https://api.cerberauth.com
```

In order to generate rules using the CLI, simply run the command in your terminal with the appropriate arguments.

```shell
./openapi-oathkeeper generate -c ./test/config/sample.yaml -f ./test/stub/sample.openapi.json
```

<details>
  <summary>Oathkeeper rules output</summary>

```json
[
    {
        "id": "cerberauth:getUserById",
        "version": "",
        "description": "",
        "match": {
            "methods": [
                "GET"
            ],
            "url": "<^(https://www\\.cerberauth\\.com/api|https://api\\.cerberauth\\.com/api)(/users/(?:[[:alnum:]]?\\x2D?=?\\??&?_?)+/?)$>"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "jwks_urls": [
                        "https://console.ory.sh/.well-known/jwks.json"
                    ],
                    "required_scope": [
                        "user:read"
                    ],
                    "target_audience": [
                        "https://api.cerberauth.com"
                    ],
                    "trusted_issuers": [
                        "https://console.ory.sh"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "errors": [
            {
                "handler": "json",
                "config": null
            }
        ],
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    },
    {
        "id": "cerberauth:updateUser",
        "version": "",
        "description": "This can only be done by the logged in user.",
        "match": {
            "methods": [
                "PUT"
            ],
            "url": "<^(https://www\\.cerberauth\\.com/api|https://api\\.cerberauth\\.com/api)(/users/(?:[[:alnum:]]?\\x2D?=?\\??&?_?)+/?)$>"
        },
        "authenticators": [
            {
                "handler": "jwt",
                "config": {
                    "jwks_urls": [
                        "https://console.ory.sh/.well-known/jwks.json"
                    ],
                    "required_scope": [
                        "user:write"
                    ],
                    "target_audience": [
                        "https://api.cerberauth.com"
                    ],
                    "trusted_issuers": [
                        "https://console.ory.sh"
                    ]
                }
            }
        ],
        "authorizer": {
            "handler": "allow",
            "config": null
        },
        "mutators": [
            {
                "handler": "noop",
                "config": null
            }
        ],
        "errors": [
            {
                "handler": "json",
                "config": null
            }
        ],
        "upstream": {
            "preserve_host": false,
            "strip_path": "",
            "url": ""
        }
    }
]
```
</details>

### OpenAPI Extension

OpenAPI Extensions serve as an extension mechanism for the OpenAPI Specification (OAS). When using OpenAPI-Oathkeeper with OpenAPI Extensions, you can embed Oathkeeper-specific rules directly within your API documentation. This integration can be beneficial when you desire a unified source of truth for both API specifications and security rules.

Here the available configurations:

| Name     | Security Schemes                  | OpenAPI Extension Name     |
|----------|-----------------------------------|----------------------------|
| JWKS URI | `oauth2`, `http`                  | `x-authenticator-jwks-uri` |
| Issuer   | `oauth2`, `http`                  | `x-authenticator-issuer`   |
| Audience | `openIdConnect`, `oauth2`, `http` | `x-authenticator-audience` |

### Example

Here's an example of the same OpenAPI contract but in JSON format

<details>
  <summary>OpenAPI example using OpenAPI Extensions</summary>

```json sample.openapi.json
{
    "openapi": "3.0.0",
    "info": {
        "title": "My API",
        "version": "1.0.0"
    },
    "servers": [
        {
            "url": "https://api.example.com",
            "description": "Production server"
        }
    ],
    "paths": {
        "/users/{id}": {
            "get": {
                "summary": "Get user by ID",
                "operationId": "getUserById",
                "parameters": [
                    {
                        "name": "id",
                        "in": "path",
                        "description": "The user id. ",
                        "required": true,
                        "schema": {
                            "type": "string"
                        }
                    }
                ],
                "responses": {
                    "200": {
                        "description": "Successful response",
                        "content": {
                            "application/json": {
                                "schema": {
                                    "type": "object",
                                    "properties": {
                                        "id": {
                                            "type": "integer"
                                        },
                                        "email": {
                                            "type": "string"
                                        }
                                    }
                                }
                            }
                        }
                    }
                },
                "security": [
                    {
                        "openidconnect": [
                            "user:read"
                        ]
                    }
                ]
            },
            "put": {
                "tags": [
                    "user"
                ],
                "summary": "Update user",
                "description": "This can only be done by the logged in user.",
                "operationId": "updateUser",
                "parameters": [
                    {
                        "name": "id",
                        "in": "path",
                        "description": "user id that need to be updated",
                        "required": true,
                        "schema": {
                            "type": "string"
                        }
                    }
                ],
                "requestBody": {
                    "description": "Update an existent user in the store",
                    "content": {
                        "application/json": {
                            "schema": {
                                "$ref": "#/components/schemas/User"
                            }
                        }
                    }
                },
                "responses": {
                    "default": {
                        "description": "successful operation"
                    }
                },
                "security": [
                    {
                        "openidconnect": [
                            "user:write"
                        ]
                    }
                ]
            }
        }
    },
    "components": {
        "schemas": {
            "User": {
                "type": "object",
                "properties": {
                    "id": {
                        "type": "integer",
                        "format": "int64",
                        "example": 10
                    },
                    "email": {
                        "type": "string",
                        "example": "john@email.com"
                    }
                }
            }
        },
        "securitySchemes": {
            "openidconnect": {
                "type": "openIdConnect",
                "openIdConnectUrl": "https://project.console.ory.sh/.well-known/openid-configuration"
            }
        }
    }
}
```
</details>

### Command line documentation

The documentation is available as markdown files in the [docs](./docs/openapi-oathkeeper.md) directory or by running `openapi-oathkeeper help`.

## Telemetry

The scanner collects anonymous usage data to help improve the tool. This data includes the number of scans performed, number of detected vulnerabilities, and the severity of vulnerabilities. No sensitive information is collected. You can opt-out of telemetry by passing the `--sqa-opt-out` flag.

## Roadmap

Please note that this tool is currently in beta stage and there may be limitations and bugs. Improvements and new features should come to make it more powerful and useful for developers. Any feedback or suggestions are greatly appreciated!

You can find the milestones and future enhancements planned for this tool on the project's [GitHub milestones page]((https://github.com/cerberauth/openapi-oathkeeper/milestones)).

## Useful Links

- [ORY Oathkeeper](https://github.com/ory/oathkeeper)
- [OpenAPI 3.x Specification](https://swagger.io/specification/)

## License

MIT © [CerberAuth](https://www.cerberauth.com)
 readmeEtag: '"65bd9ae7488bd125282634c87584310841b8fe38"' readmeLastModified: Thu, 08 May 2025 14:49:26 GMT repositoryId: 611484700 description: >- openapi-oathkeeper is a CLI for generating Ory Oathkeeper rules from an OpenAPI 3 contract and save a lot of time and effort, especially for larger projects with many endpoints or many services. created: '2023-03-08T23:18:03Z' updated: '2025-11-29T14:58:46Z' language: Go archived: false stars: 11 watchers: 2 forks: 8 owner: cerberauth logo: https://avatars.githubusercontent.com/u/127022998?v=4 license: MIT repoEtag: '"783d5d4ca1be01a73be1af2d0cdb8657e8670117d090a5b45a71eef488cbb4ba"' repoLastModified: Sat, 29 Nov 2025 14:58:46 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/overflowdigital/flask-openapi v3: true id: 51e114e3aa9b01b264af35e38341edcd repositoryMetadata: base64Readme: >- IyBGbGFzay1PcGVuQVBJCgoqKkZsYXNrLU9wZW5BUEkgaXMgbm93IGluIG1haW50ZW5hbmNlIG1vZGUuKioKClshW1B5dGhvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9weXRob24tMy44LjAtYmx1ZS5zdmcpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3B5dGhvbi0zLjguLjAtYmx1ZS5zdmcpClshW1B5UGldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS92L0ZsYXNrLU9wZW5BUEkzLVVJLnN2ZyldKGh0dHBzOi8vcHlwaS5weXRob24ub3JnL3B5cGkvRmxhc2stT3BlbkFQSTMtVUkpClshW1B5UGldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS9kbS9GbGFzay1PcGVuQVBJMy1VSS5zdmcpXShodHRwczovL3B5cGkucHl0aG9uLm9yZy9weXBpL0ZsYXNrLU9wZW5BUEkzLVVJKQpbIVtGbGFzay1PcGVuQVBJMy1VSV0oaHR0cHM6Ly9zbnlrLmlvL2Fkdmlzb3IvcHl0aG9uL0ZsYXNrLU9wZW5BUEkzLVVJL2JhZGdlLnN2ZyldKGh0dHBzOi8vc255ay5pby9hZHZpc29yL3B5dGhvbi9GbGFzay1PcGVuQVBJMy1VSSkKCkZsYXNrLU9wZW5BUEkgaXMgYSBGbGFzayBleHRlbnNpb24gdGhhdCBwcm92aWRlcyBpbnRlZ3JhdGlvbiB3aXRoIE9wZW5BUEkgdjIsIHYzLCBhbmQgdjMuMSBzcGVjaWZpY2F0aW9ucyBlYXNpbHkuCgojIyBGZWF0dXJlcwotIEVhc3kgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGRlZmluaXRpb25zIHdpdGggdGhlIGBvcGVuYXBpX3NwZWNgIGRlY29yYXRvcgotIEN1c3RvbWlzYWJsZSBTd2FnZ2VyIFVJCi0gU3VwcG9ydHMgT3BlbkFQSSB2Mi4wLCB2My4wLCBhbmQgdjMuMQotIFN1cHBvcnRzIFN3YWdnZXIgVUkgdjIsIHYzLCB2NCwgYW5kIHY1LgoKIyMgR2V0dGluZyBTdGFydGVkClRvIHF1aWNrbHkgaW5zdGFsbDoKYHBpcCBpbnN0YWxsIEZsYXNrLU9wZW5BUEkzLVVJYAoKU2VlIHRoZSBbR2V0dGluZyBTdGFydGVkXShodHRwczovL2dpdGh1Yi5jb20vb3ZlcmZsb3dkaWdpdGFsL0ZsYXNrLU9wZW5BUEkvd2lraSkgZG9jdW1lbnRhdGlvbiB0byBnZXQgaW5zdGFsbGVkIGFuZCBzZXR1cCB3aXRoIHlvdXIgRmxhc2sgYXBwbGljYXRpb24uCgojIyBDb250cmlidXRpbmcKU2VlIHRoZSBbQ29udHJpYnV0aW5nIEd1aWRlXShodHRwczovL2dpdGh1Yi5jb20vb3ZlcmZsb3dkaWdpdGFsL0ZsYXNrLU9wZW5BUEkvYmxvYi9tYWluL0NPTlRSSUJVVElORy5tZCkgZm9yIG1vcmUgaW5mb3JtYXRpb24gb24gY29udHJpYnV0aW5nIHRvIHRoZSBwcm9qZWN0LgoKIyMgU2VjdXJpdHkKV2UgdGFrZSBzZWN1cml0eSBzZXJpb3VzbHksIHBsZWFzZSByZWFkIHRoZSBbU2VjdXJpdHkgQWR2aXNvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9vdmVyZmxvd2RpZ2l0YWwvRmxhc2stT3BlbkFQSS9ibG9iL21haW4vU0VDVVJJVFkubWQpIGZvciBtb3JlIGluZm9ybWF0aW9uLgoKIyMgQ29kZSBvZiBDb25kdWN0CldlIHRha2UgemVybyB0b2xlcmFuY2Ugb24gYXR0YWNrcyBvbiBvdXIgT1NTIGNvbW11bml0eSwgcGxlYXNlIHNlZSB0aGUgW0NvZGUgb2YgQ29uZHVjdF0oaHR0cHM6Ly9naXRodWIuY29tL292ZXJmbG93ZGlnaXRhbC9GbGFzay1PcGVuQVBJL2Jsb2IvbWFpbi9DT0RFX09GX0NPTkRVQ1QubWQpIGZvciBtb3JlIGluZm9ybWF0aW9uCg== readmeEtag: '"756a6b04738a3a3b572b63ba0e94b60a5de63d18"' readmeLastModified: Wed, 14 Feb 2024 09:50:39 GMT repositoryId: 526292250 description: >- Flask-OpenAPI is a Flask extension that provides integration with OpenAPI v2, v3, and v3.1 specifications easily. created: '2022-08-18T16:35:43Z' updated: '2025-08-13T15:48:46Z' language: JavaScript archived: true stars: 9 watchers: 2 forks: 2 owner: overflowdigital logo: https://avatars.githubusercontent.com/u/93914478?v=4 license: MIT repoEtag: '"b3dc9fa930d3120bef9b8684d42b95edd47857c779cf740872fb8145741652ed"' repoLastModified: Wed, 13 Aug 2025 15:48:46 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/the-pawn-2017/r5t v3: true id: cd312bae0e99794c6fedc2c13c6f5e63 repositoryMetadata: base64Readme: >- IyByNXQKCj4g55yB5rWB54mI77ya6L+Z5piv5LiA5Liq55SoZ28g5Luj56CB55Sf5oiQc3dhZ2dlci9vcGVuQXBp5paH5qGj55qE5bqT77yM5LiN6ZyA6KaB5YaZ5rOo6YeK77yBO+i/meS4quW6k+WfuuS6jmBnby1vcGVuYXBpYO+8jOWPr+S7peW1jOWFpeWIsHdlYumhueebruS4reOAglvmlofmoaPov57mjqVdKGh0dHBzOi8vanVlamluLmNuL3Bvc3QvNzM5MDU2OTU0ODM2NzgyMjg1OCkKPiBBdXRvbWF0aWNhbGx5IGdlbmVyYXRlZCBPcGVuQVBJIDMvc3dhZ2dlciBkb2N1bWVudGF0aW9uIHZpYSBHbyBjb2RlLCB3aXRob3V0IHJlbHlpbmcgb24gY29tbWVudHMuIEl0IGNhbiBiZSBlbWJlZGRlZCBpbiB5b3VyIHdlYiBwcm9qZWN0LgoKaW5zdGFsbDogYGdvIGdldCAtdSBnaXRodWIuY29tL3RoZS1wYXduLTIwMTcvcjV0YAoKIyMgd2h5IGNyZWF0ZSB0aGlzIHJlcG8/CjEuIEkgd291bGQgbGlrZSB0byBpbXBsZW1lbnQgbW9yZSBvdGhlciBmZWF0dXJlcywgc3VjaCBhcyBzdXBwb3J0IGZvciBHSU4gYW5kIEVDSE8uCjIuIFNpbmNlIG1hbnkgb2YgbXkgcHJvamVjdHMgYWZ0ZXIgdGhhdCByZXF1aXJlIFJFU1QgQVBJIGRvY3VtZW50YXRpb24sIEknbSBtb3JlIG1vdGl2YXRlZCB0byBtYWludGFpbiBpdC4KIyMgdmVyc2lvbgp2MC41CiMjIHRvZG8KLSDinIUgYWxsIG9wZW5BUEkvc3dhZ2dlciBjb21wb25lbnRzIHN1cHBvcnQgYW5kIGxpbWl0Ci0g4pyFIFJlZ2lzdGVyaW5nIHJlcyZyZXEgbW9kZWwsbm93LGl0IGNhbiB1c2UganNvbixmb3JtLgotIOKchSBTdXBwb3J0aW5nIE9BdXRoMiAsIG9ubHkgY29kZSBhbmQgaW1wbGljaXQKLSDinIUgcmVnaXN0ZXIgbW9kZWwKLSDwn5qnIGNvbXBsZXRlIHRlc3QKLSDwn5qnIGZ1bGwgZG9jdW1lbnQgZm9yIHRoaXMgcmVwbwotIOKchSBTdXBwb3J0IG90aGVyIHdlYiBzZXJ2ZXIsbm93LGVjaG8gY2FuIHVzZSBgcjV0YCBieSBzb21lIGZ1bmN0aW9uLCBpdCdzIGluIFtgZXhhbXBsZS9lY2hvYF0oLi9leGFtcGxlL2VjaG8vZWNoby5tZCkKCiMjIHNvbWUgdXNlZnVsIGZlYXR1cmUKIyMjIDEuIGZhc3QgcGFnaW5hdGlvbgpgYGBnb2xhbmcKcyA6PSByNXQuTmV3U3BlYyhzcGVjLlRpdGxlKCJwYWdpbmF0aW9uLnlhbWwiKSkKcy5HZXQoIi90ZXN0LXBhZ2luYXRpb24iKS5QYWdlSW5RdWVyeSgicGFnZSIsIDEsICJwYWdlU2l6ZSIsIDEwKS5SZXNTdHJpbmcoaHR0cC5TdGF0dXNPSywgcmVzLkV4YW1wbGUoImhpIikpCmBgYAojIyMgMi4gZWFzeSB0byB1c2UgZm9yIE9BdXRoMgpgYGBnb2xhbmcKcyA6PSBzcGVjLk5ld1NwZWMoKQpzLlNlY3VyaXR5KAoJc2VjdXJpdHkuV2l0aE9BdXRoMkNvZGUoInppdGVhbCIsICJodHRwOi8vMTAuNDUuOC4xODk6ODA4MC9vYXV0aC92Mi9hdXRob3JpemUiLCAiaHR0cDovLzEwLjQ1LjguMTg5OjgwODAvb2F1dGgvdjIvdG9rZW4iLAoJc2VjdXJpdHkuQWRkU2NvcGUoIm9wZW5pZCIsICJPUEVOSUQgSVMgVVNJTkcgRk9SIElEIikpLAopCmBgYAojIyMgMy4gY29uY2lzZSBhbmQgcG93ZXJmdWwgQVBJLCBsaWtlIGBSZXFqc29uYCxgUmVzSnNvbmAsYFJlc1N0cmluZ2AuCmBgYGdvbGFuZwpzIDo9IHI1dC5OZXdTcGVjKHNwZWMuVGl0bGUoImV4YW1wbGUgcmVxU3RyaW5nIikpCnMuR2V0KCIvdGVzdC1yZXNTdHJpbmciKS5SZXNTdHJpbmcoaHR0cC5TdGF0dXNPSywgcmVzLkV4YW1wbGUoImhpISIpKQpgYGAKIyMgZXhhbXBsZToKZ28gW2AvdGVzdC9zcGVjX3Rlc3RgXSgvdGVzdHMvc3BlY190ZXN0LmdvKSB2aWV3IHNvbWUgZXhhbXBsZQpgYGBnb2xhbmcKdHlwZSBUZXN0IHN0cnVjdCB7CglBIHN0cmluZwoJQiBzdHJpbmcgYHZhbGlkYXRlOiJyZXF1aXJlZCJgCn0KYGBgCmBgYGdvbGFuZwoJcyA6PSBzcGVjLk5ld1NwZWMoKQoJcy5TZWN1cml0eSgKCQlzZWN1cml0eS5XaXRoT0F1dGgyQ29kZSgieml0ZWFsIiwgImh0dHA6Ly8xMC40NS44LjE4OTo4MDgwL29hdXRoL3YyL2F1dGhvcml6ZSIsICJodHRwOi8vMTAuNDUuOC4xODk6ODA4MC9vYXV0aC92Mi90b2tlbiIsCgkJCXNlY3VyaXR5LkFkZFNjb3BlKCJvcGVuaWQiLCAiT1BFTklEIElTIFVTSU5HIEZPUiBJRCIpKSwKCSkKCS8vIHRoYW4sIHlvdSBjYW4gdXNlIE9BdXRoMiBjb2RlIG1vZGUgbm93CglzLlBvc3QoIi9na2QiKS5OZWVkU2VjdXJpZnkoInppdGVhbCIsIFtdc3RyaW5neyJvcGVuaWQifSkuCgkJUmVxSlNPTihtb2RlbC5Nb2RlbE9mW1Rlc3RdKCksIHJlcS5XaXRoRXhhbXBsZShUZXN0e0E6ICJBIiwgQjogIkIifSkpLgoJCVJlc0pTT04oaHR0cC5TdGF0dXNPSywgbW9kZWwuTW9kZWxPZltUZXN0XSgpLCByZXMuV2l0aEV4YW1wbGUoVGVzdHtBOiAiQSIsIEI6ICJCIn0pKQpgYGAKIyMjIGVtYmVkIHN3YWdnZXItdWkKCmBgYGdvbGFuZwpwYWNrYWdlIG1haW4KCmltcG9ydCAoCgkibmV0L2h0dHAiCgoJImdpdGh1Yi5jb20vbGFic3RhY2svZWNoby92NCIKCSJnaXRodWIuY29tL2xhYnN0YWNrL2VjaG8vdjQvbWlkZGxld2FyZSIKCSJnaXRodWIuY29tL3RoZS1wYXduLTIwMTcvcjV0IgoJImdpdGh1Yi5jb20vdGhlLXBhd24tMjAxNy9yNXQvbW9kZWwiCgkiZ2l0aHViLmNvbS90aGUtcGF3bi0yMDE3L3I1dC9yZXEiCgkiZ2l0aHViLmNvbS90aGUtcGF3bi0yMDE3L3I1dC9yZXMiCgkiZ2l0aHViLmNvbS90aGUtcGF3bi0yMDE3L3I1dC9zZWN1cml0eSIKCSJnaXRodWIuY29tL3RoZS1wYXduLTIwMTcvcjV0L3N3YWdnZXJ1aSIKKQoKdHlwZSBUZXN0QmFzaWMgc3RydWN0IHsKCUEgc3RyaW5nCglCIHN0cmluZyBgdmFsaWRhdGU6InJlcXVpcmVkImAKfQoKZnVuYyBtYWluKCkgewoJZSA6PSBlY2hvLk5ldygpCgllLlVzZShtaWRkbGV3YXJlLkxvZ2dlcigpKQoJcyA6PSByNXQuTmV3U3BlYygpCglzLlNlY3VyaXR5KAoJCXNlY3VyaXR5Lk9BdXRoMkNvZGUoInppdGVhbCIsICJodHRwOi8vMTAuNDUuOC4xODk6ODA4MC9vYXV0aC92Mi9hdXRob3JpemUiLCAiaHR0cDovLzEwLjQ1LjguMTg5OjgwODAvb2F1dGgvdjIvdG9rZW4iLAoJCQlzZWN1cml0eS5BZGRTY29wZSgib3BlbmlkIiwgIk9QRU5JRCBJUyBVU0lORyBGT1IgSUQiKSksCgkpLgoJCVBvc3QoIi9na2QiKS5OZWVkU2VjdXJpZnkoInppdGVhbCIsIFtdc3RyaW5neyJvcGVuaWQifSkuCgkJUmVxSlNPTihtb2RlbC5Nb2RlbE9mW1Rlc3RCYXNpY10oKSwgcmVxLkV4YW1wbGUoVGVzdEJhc2lje0E6ICJBIiwgQjogIkIifSkpLgoJCVJlc0pTT04oaHR0cC5TdGF0dXNPSywgbW9kZWwuTW9kZWxPZltUZXN0QmFzaWNdKCksIHJlcy5FeGFtcGxlKFRlc3RCYXNpY3tBOiAiQSIsIEI6ICJCIn0pKQoJZS5HRVQoIi9zd2FnZ2VyLXRlc3QuanNvbiIsIGZ1bmMoYyBlY2hvLkNvbnRleHQpIGVycm9yIHsKCQlyZSwgZXJyIDo9IHN3YWdnZXJ1aS5HZW5TcGVjKHMpCgkJaWYgZXJyID09IG5pbCB7CgkJCXJldHVybiBjLkpTT05CbG9iKGh0dHAuU3RhdHVzT0ssIHJlKQoJCX0gZWxzZSB7CgkJCXJldHVybiBjLlN0cmluZyhodHRwLlN0YXR1c0ludGVybmFsU2VydmVyRXJyb3IsIGVyci5FcnJvcigpKQoJCX0KCX0pCgllLkdFVCgiL3N3YWdnZXIvKiIsIGZ1bmMoYyBlY2hvLkNvbnRleHQpIGVycm9yIHsKCQlwYXJhbVN0ciA6PSBjLlBhcmFtKCIqIikKCQlraW5kLCBjb250ZW50LCBlcnIgOj0gc3dhZ2dlcnVpLkdldFN3YWdnZXJVSUZpbGUoIi9zd2FnZ2VyLXRlc3QuanNvbiIsIHBhcmFtU3RyKQoJCWlmIGVyciA9PSBuaWwgewoJCQlyZXR1cm4gYy5CbG9iKGh0dHAuU3RhdHVzT0ssIGtpbmQsIGNvbnRlbnQpCgkJfQoJCXJldHVybiBjLlN0cmluZyhodHRwLlN0YXR1c0ludGVybmFsU2VydmVyRXJyb3IsIGVyci5FcnJvcigpKQoJfSkKCWUuU3RhcnQoIjoyMzMzIikKfQpgYGAKCgpbYGV4YW1wbGUvZWNob2BdKC4vZXhhbXBsZS9lY2hvL2VjaG8ubWQpCgoqKkNBUkVGVUxMWSBVU0UgSVQgSU4gWU9VUiBQUk9KRUNULCBCRUNBVVNFIElUIElOIERFVkVMT1BJTkcqKgoKKipJIGFtIGN1cnJlbnRseSB0ZXN0aW5nIHdpdGggbXkgb3duIHByb2plY3RzIHRvIHJlZmluZSBSNVQsIGV4cGVjdGluZyBpdCB0byBzdGFiaWxpemUgYnkgdGhlIGVuZCBvZiBBdWd1c3QuIEF0IHRoYXQgcG9pbnQsIEkgd2lsbCBtYXJrIFI1VCBhcyByZWFkeSBmb3Igb2ZmaWNpYWwgdXNlLCBtYWtpbmcgaXQgY29udmVuaWVudCBmb3IgZXZlcnlvbmUuKioKIyMgdG9vbHMgCltzd2FnZ2VyLXVpLWVkaXRdKGh0dHBzOi8vZWRpdG9yLW5leHQuc3dhZ2dlci5pby8pCj4gaW5zcGlyZWQgYnkgW2EtaC9yZXN0XShodHRwczovL2dpdGh1Yi5jb20vYS1oL3Jlc3Qp readmeEtag: '"be146c86aa5087abd5e59d29e1a018891004ea0a"' readmeLastModified: Thu, 05 Dec 2024 07:48:52 GMT repositoryId: 809563216 description: ' Automatically generated OpenAPI 3/swagger documentation via Go code, without relying on comments' created: '2024-06-03T02:29:55Z' updated: '2024-12-05T07:50:39Z' language: Go archived: false stars: 9 watchers: 2 forks: 0 owner: the-pawn-2017 logo: https://avatars.githubusercontent.com/u/115976311?v=4 license: LGPL-3.0 repoEtag: '"338cacc130307959676170362a12cfaf419e1605df6278997f1cc518bb9bfa2a"' repoLastModified: Thu, 05 Dec 2024 07:50:39 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/karatelabs/vscode-extension v3: true id: 48cbff2a15a37b74a0aec052425666f7 repositoryMetadata: base64Readme: >- <table>
    <tr>
        <td>
            <br/><a href="https://karatelabs.io"><img src="resources/karate-logo.svg" height="45px"/></a>
        </td>
        <td>
            <h2><br/>VS Code<br/>Extension</h2>           
        </td>
        <th>
            <h3><a href="https://youtu.be/nXUSnxcdljQ?si=wCh-m1qL5Rj1qQIS">:tv: <br/>&nbsp;Video&nbsp;</a></h3>
        </th>        
        <th>
            <h3><a href="https://www.karatelabs.io/pricing">:heavy_dollar_sign: <br/>Pricing</a></h3>
        </th>       
        <th>
            <h3><a href="https://marketplace.visualstudio.com/items?itemName=karatelabs.karate"><br/>:zap: <br/> Market<br/>place</a></h3>
        </th>
        <th>
            <h3><a href="https://open-vsx.org/extension/karatelabs/karate"><br/>:bulb: <br/>&nbsp;Open&nbsp;<br/>VSX</a></h3>
        </th>         
        <th>
            <h3><a href="https://github.com/karatelabs/vscode-extension/issues">:octocat: <br/>Issues</a></h3>
        </th>        
    </tr>
</table>

<table>
    <tr>
        <th>PLUS</th>
        <th>PRO</th>
        <th>Enterprise</th>
    </tr>    
    <tr>
        <td>            
            <ul>
                <li><a href="https://youtu.be/nXUSnxcdljQ?si=k-M5AYz9nGZylzD8&t=38">Syntax coloring</li>                
                <li>Run Feature from editor</li>
                <li><a href="#run-from-editor">Run single Scenario / Example</a></li>
                <li><a href="https://youtu.be/nXUSnxcdljQ?si=P2mVg47dMJAsQEE1&t=192">Launch Configurations</a></li>
                <li><a href="https://youtu.be/nXUSnxcdljQ?si=THQ97SfhnzkvLV0w&t=331">Outline view</a></li>
                <li><a href="https://youtu.be/nXUSnxcdljQ?si=pD7LjtF7hS43Q813&t=363">Code formatting</a></li>
                <li><a href="https://youtu.be/nXUSnxcdljQ?si=iyJMEdL19I5BkGxf&t=421">Test Results in IDE</a></li>
            </ul>
        </td>
        <td>
            <i>&nbsp;&nbsp;&nbsp;&nbsp;(includes all in PLUS)</i>
            <ul>
                <li><a href="https://youtu.be/IYSNjX5dPZA?si=3sxIzYbJj-7NlYNS&t=30">Auto complete</a></li>                
                <li><a href="https://youtu.be/IYSNjX5dPZA?si=bjJiFbyY_b6nFKJ_&t=104">Code folding</a></li>
                <li><a href="#references">Jump to references</a></li>
                <li><a href="https://youtu.be/IYSNjX5dPZA?si=_kWzYEVqvEuXHsQo&t=134">JSON re-formatting</a></li>
                <li><a href="https://youtu.be/IYSNjX5dPZA?si=CgrqGE0-XVUah4Ll&t=233">Run all tests in folder</a></li>
                <li>Run Karate Labs add-ons (e.g. <a href="https://github.com/karatelabs/karate-addons/blob/main/karate-kafka/README.md">Kafka</a>)</li>
                <li><a href="https://youtu.be/IYSNjX5dPZA?si=qKYmDrtX4DZxXKMZ&t=262">Debug Karate test</a></li>
                <li>Debug Java &amp; Karate in same session</li>
                <li>Java debug session stops at Karate breakpoints</li>
                <li><a href="#run-as-mock">Run as mock</a></li>
                <li>cURL import *</li>
                <li>OpenAPI support *</li>
            </ul><i>&nbsp;&nbsp;&nbsp;&nbsp;[*] coming soon</i>
        </td>
        <td>
            <ul>
                <li>Priority support</li>
                <li>SSO / SAML support</li>
                <li><a href="#offline-license">Offline license</a></li>
                <li><a href="https://www.karatelabs.io/contact-us">Contact us</a></li>
            </ul>        
        </td>        
    </tr>
</table>

> By using this plugin, you agree to the Karate Labs [EULA](https://karatelabs.io/eula). Data on a few user actions is collected and subject to our [Privacy Policy](https://karatelabs.io/privacy-policy). HTML reports generated by the Karate open-source library has [additional analytics](https://github.com/karatelabs/karate/blob/master/karate-core/src/test/resources/analytics.md).

## License Activation
License activation and status is available via a command. Go to `View -> Command Palette`, type `Karate` and you should see something like this.

<img src="resources/sign-in.jpg" height="200px"/>

<p>&nbsp;</p>

Choose the option to `Sign In / Manage License` and you will be shown a [dialog](#offline-license). Click `Sign In` to be taken through the usual flow linked to your existing subscription details.

Once you have authenticated successfully, copy the session ID from the browser and paste it into the input-box now showing in VS Code. Hit `ENTER` to complete the sign-in.

<img src="resources/sign-in-sessionid.jpg" height="150px"/>

<p>&nbsp;</p>

Once signed-in you can work offline. You can always run the `Sign In / Manage License` command to see how many days are left in your session.

If you need an [offline license](#offline-license) because of strict security or similar restrictions in your environment, please [contact us](https://www.karatelabs.io/contact-us). Please note that this option is available only for enterprise customers of Karate Labs.

## Settings

You can access the extension settings by clicking on the "gear" icon after going to "Extensions" on the nav-bar.

<img src="resources/settings-open.jpg" height="300px"/>

<p>&nbsp;</p>

Especially if you are working with Java Maven or Gradle projects, it is recommended that you have the [Language Support for Java](https://marketplace.visualstudio.com/items?itemName=redhat.java) VS Code extension installed.

Then keep the `karatelabs.karate.run.mode` setting as `vscode-java` (the default).

<img src="resources/settings-new.jpg" height="650px"/>

### Standalone
You can change the `karatelabs.karate.run.mode` setting to `standalone` if you want to use the [Karate standalone JAR](https://github.com/karatelabs/karate/wiki/Get-Started:-Other-Runtime-Options#standalone-jar). Download it and choose a local folder to keep JAR files in. Change the `karatelabs.karate.run.modeJars` setting to point to that folder, and for convenience, all JARs within that folder will be included in the runtime classpath.

Note that the `Mode Jars` setting should be a folder name. Do not include the `.jar` part. Note that the default of `.` will work fine for the common practice of keeping the Karate JAR in the "root" of your project folder.

The `karatelabs.karate.run.classpathPrefix` setting defaults to `.` which should suffice for picking up `karate-config.js` from the workspace root. You can add more classpath entries (comma-delimited) and even JAR files if you wish. For example `.,src/test/java` would also search in `src/test/java` for `karate-config.js`.

Note that VS Code has a mechanism to [customize settings per Workspace](https://code.visualstudio.com/docs/getstarted/settings#_workspace-settings) (or project folder) which can ensure your whole team uses the same settings. Here is an example of how the values of the `run.mode` and `classpathPrefix` can be written to a file in `.vscode/settings.json`. Now this file can be added to your Git repository or version control.

<img src="resources/settings-workspace.jpg" height="350px"/>

### Custom

You can also opt to run commands such as `java` or `mvn` directly with `karatelabs.karate.run.mode` set to `custom`. In this case `java` or `mvn` is expected to be installed and in the system `PATH`.

For convenience the string `${karateArgs}` will be replaced with command-line arguments generated by the extension when performing actions such as `Run` etc.

## Troubleshooting

A common source of issues is if you have another extension installed that also supports `*.feature` files.

Here below is a handy guide to a few things to look for that confirm that the Karate Labs extension is ready to use.

<img src="resources/ide-tips.jpg" height="350px"/>

<p>&nbsp;</p>

The `Java Ready` indicator should show up if you are within a Maven or Gradle project.

If the [Language Support for Java](https://marketplace.visualstudio.com/items?itemName=redhat.java) extension is not installed, you will need a Java runtime installed and [`JAVA_HOME` set in your environment](https://www.baeldung.com/java-home-on-windows-mac-os-x-linux). Having the `java` command in your [system `PATH`](https://www.java.com/en/download/help/path.html) should also be sufficient. Java 11 or higher is needed for Karate 1.4.X and Java 17 is needed for Karate 1.5.0 and above.

There is a `Karate Log` within the "Output" tab (next to "Terminal") typically at the bottom of your VS Code window. You can look at it to find the reasons for failure.

<img src="resources/karate-log.jpg" height="200px"/>

<p>&nbsp;</p>

To change the log level you can use the VS Code Command `Developer: Set Log Level` (`View --> Command Palette` and type "Log") and then select `Karate Log`. Changing the log level to `Debug` gives you more information.

<img src="resources/log-level.jpg" height="200px"/>

## Run From Editor
You can run a feature file open in the editor by using the CodeLens. The keyboard shortcut `[CTRL]` `[F5]` also works.

<img src="resources/run-feature.jpg" height="350px"/>

<p>&nbsp;</p>

You can run a single `Scenario` by using the CodeLens that appears above it.

You can even run a single "example" in a `Scenario Outline` by right-clicking on one of the data-rows within an `Examples` table.

<img src="resources/run-example.jpg" height="180px"/>

## Launch Configurations
VS Code [launch-configurations](https://code.visualstudio.com/docs/editor/debugging) are useful for being able to re-run tests with specific parameters. All the typical Karate options you need are supported, including control over the JVM parameters and working directory. The feature file (or path to search for feature files) should be the last argument.

Here is an example `.vscode/launch.json` file:

```json
{
    "version": "0.2.0",
    "configurations": [
        {
            "type": "karate",
            "name": "env-test",
            "request": "launch",
            "karateArgs": [
                "-e",
                "test",
                "${file}"
            ]
        }
    ]
}
```

The advantage is that you can set up any combination of [Karate runtime options](https://github.com/karatelabs/karate#karateoptions) via `karateArgs`. For example you can:
* [run tests in parallel](https://github.com/karatelabs/karate/tree/master/karate-netty#parallel-execution)
* [set the environment](https://github.com/karatelabs/karate/tree/master/karate-netty#karateenv)
* or [choose tags](https://github.com/karatelabs/karate/tree/master/karate-netty#tags) to include or exclude.

Now you can run tests from the [Run and Debug](https://code.visualstudio.com/docs/editor/debugging#_launch-configurations) view in VS Code and keyboard short-cuts such as `F5` will work for the currently focused file in the editor. Note that you can even Run without Debugging: `Ctrl + F5`.

<img src="resources/launch-config.jpg" height="400px"/>

<p>&nbsp;</p>

You can have multiple configurations and easily switch between them.

You can add `"noDebug": true` to the run-configuration JSON to force "Run without Debugging" for convenience.

Instead of `${file}` as the value for `feature`, you can use any valid folder path or even multiple feature files.

## Outline View
The VS Code [outline view](https://code.visualstudio.com/docs/getstarted/userinterface#_outline-view) is supported so you can navigate large files with ease.

<img src="resources/outline-view.jpg" height="400px"/>

## Code Formatting
VS Code [code-formatting](https://code.visualstudio.com/docs/editor/codebasics#_formatting) shortcuts work to format indenting of feature file elements.

## Test Results
The HTML report is one-click away.

<img src="resources/test-report-link.jpg" height="200px"/>

## Auto Complete
Besides the syntax validation, you have auto-complete for the most commonly used keywords.

<img src="resources/autocomplete.jpg" height="200px"/>

## Code Folding
You can collapse sections to make it easier to deal with long tests.

<img src="resources/code-folding.jpg" height="200px"/>

## References
Coming Soon.

## JSON Re-formatting
Place the cursor within JSON to see options to re-format it, you will see a "bulb" icon.

<img src="resources/json-bulb.jpg" height="250px"/>

<p>&nbsp;</p>

The following options are possible (depending on context):

* Lenient - just like JS, where single-quotes are used and no quotes are needed for property keys
* Strict - Strict JSON, using double-quotes
* Make single Line - convert multi-line JSON to a single line
* Make multi-line - convert a single line of JSON to multi-line, including triple-quotes

## Run Folder

As a convenience, you can right-click and run a folder from the explorer view.

<img src="resources/run-folder.jpg" height="450px"/>

## Debug

You can set break-points on Karate feature files in debug mode. The Karate debugger can even step-back and hot-reload simple edits to your test.

<img src="resources/debug.jpg" height="600px"/>

### Debug Java from Karate

A Karate debug session will even stop at Java breakpoints.

### Debug Karate from Java

You can also start a normal Java debug session that uses the Karate `Runner` Java API but still stop at Karate breakpoints. This requires you to be using Karate version 1.5.0.RC3 or greater.

## Run as mock

You can right click and run a `*.feature` file as a [Karate mock](https://github.com/karatelabs/karate/tree/master/karate-netty#server-life-cycle). You can also do this for the newer [JavaScript mocks](https://github.com/karatelabs/karate/wiki/Karate-JavaScript-Mocks), so `*.js` files are also supported. Make sure that the file you are right-clicking on is a valid mock script.

<img src="resources/run-as-mock.jpg" height="350px"/>

## Offline License

> [!IMPORTANT]  
> The Offline License is a [paid feature for enterprises](https://www.karatelabs.io/pricing). Please make sure you know who your designated admin is before you perform this step.


The dialog you see when you invoke the [`Sign In / Manage License`](#license-activation) command looks like this.

<img src="resources/license-dialog.jpg" height="100px"/>

<p>&nbsp;</p>

Click the `Offline License` button. A unique code for your system will be shown which you can cut and paste. In the example below, it is: `DhHOFKHvd7XYTi+rQnNTJQ==`

<img src="resources/offline-license-code.jpg" height="150px"/>

<p>&nbsp;</p>

Send that unique code to your designated admin. You will be issued a license that is tied to this unique code. To apply the license, copy *all* the text from the license file (which should start with: `-----BEGIN LICENSE FILE-----`) and paste it into the text-field shown above. Make sure you *clear* the text-box before pasting the license file text or just replace the initial contents. Now you should see something like this.

<img src="resources/offline-license-paste.jpg" height="150px"/>

<p>&nbsp;</p>

Click [ENTER] and should see a confirmation message with your license details.





 readmeEtag: '"85e01f93f531f4e25e435b5fa9dbef6f0fec2f91"' readmeLastModified: Mon, 12 May 2025 14:32:06 GMT repositoryId: 528727790 description: Karate Visual Studio Code Extension created: '2022-08-25T06:43:13Z' updated: '2025-05-12T14:32:14Z' language: null archived: false stars: 10 watchers: 1 forks: 2 owner: karatelabs logo: https://avatars.githubusercontent.com/u/91312095?v=4 repoEtag: '"8381edb5500e58cab1838c4d13321795d66e70a2286b6f38d9a49617c70e3e84"' repoLastModified: Mon, 12 May 2025 14:32:14 GMT category: Testing foundInMaster: true oldLocations: - https://github.com/karatelabs/karate-vscode-extension - source: openapi3 tags repository: https://github.com/ldej/go-openapi-example v3: true id: 50961b8a85a4eb26a25b492151b8b365 repositoryMetadata: base64Readme: >- IyBnby1vcGVuYXBpLWV4YW1wbGUKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyB0aGUgd29ya2luZyBleGFtcGxlcyBvZiBzZXJ2ZXIgKHN0dWJzKSBnZW5lcmF0ZWQgYnkgZm91ciBkaWZmZXJlbnQgZ2VuZXJhdG9ycywgYW5kIGltcGxlbWVudGVkIHdpdGggYmFzaWMgZnVuY3Rpb25hbGl0eSB0byBkZW1vbnN0cmF0ZSBob3cgdG8gdXNlIHRoZSBnZW5lcmF0ZWQgY29kZS4gW1JlYWQgdGhlIGJsb2cgcG9zdF0oaHR0cHM6Ly9sZGVqLm5sL3Bvc3QvZ2VuZXJhdGluZy1nby1mcm9tLW9wZW5hcGktMy8pIGZvciBtb3JlIGRldGFpbHMgYW5kIHJlc3VsdHMuCg== readmeEtag: '"890046f8e7c4050420ee3e70c3525aa8f09cf7ca"' readmeLastModified: Fri, 23 Jun 2023 05:45:53 GMT repositoryId: 647720230 description: >- Example implementations of Go servers based on generated code from OpenAPI 3 definitions created: '2023-05-31T11:34:44Z' updated: '2025-11-04T18:19:58Z' language: Go archived: false stars: 11 watchers: 3 forks: 4 owner: ldej logo: https://avatars.githubusercontent.com/u/18293312?v=4 repoEtag: '"9aa94279883e4052f8a6411c4cc7dc46082bf93e398c1f559f8861f066ce6211"' repoLastModified: Tue, 04 Nov 2025 18:19:58 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/avanov/openapi-client-generator v3: true repositoryMetadata: base64Readme: >- Li4gX2JhZGdlczoKCi4uIGltYWdlOjogaHR0cHM6Ly9naXRodWIuY29tL2F2YW5vdi9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3Ivd29ya2Zsb3dzL0NJL2JhZGdlLnN2Zz9icmFuY2g9ZGV2ZWxvcAogICAgOnRhcmdldDogaHR0cHM6Ly9naXRodWIuY29tL2F2YW5vdi9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3IvYWN0aW9ucz9xdWVyeT1icmFuY2glM0FkZXZlbG9wCgouLiBpbWFnZTo6IGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9hdmFub3Yvb3BlbmFwaS1jbGllbnQtZ2VuZXJhdG9yL2JhZGdlLnN2Zz9icmFuY2g9ZGV2ZWxvcAogICAgOnRhcmdldDogaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL2F2YW5vdi9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3I/YnJhbmNoPWRldmVsb3AKCi4uIGltYWdlOjogaHR0cHM6Ly9yZXF1aXJlcy5pby9naXRodWIvYXZhbm92L29wZW5hcGktY2xpZW50LWdlbmVyYXRvci9yZXF1aXJlbWVudHMuc3ZnP2JyYW5jaD1tYXN0ZXIKICAgIDp0YXJnZXQ6IGh0dHBzOi8vcmVxdWlyZXMuaW8vZ2l0aHViL2F2YW5vdi9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3IvcmVxdWlyZW1lbnRzLz9icmFuY2g9bWFzdGVyCiAgICA6YWx0OiBSZXF1aXJlbWVudHMgU3RhdHVzCgouLiBpbWFnZTo6IGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL29wZW5hcGktY2xpZW50LWdlbmVyYXRvci9iYWRnZS8/dmVyc2lvbj1sYXRlc3QKICAgIDp0YXJnZXQ6IGh0dHBzOi8vb3BlbmFwaS1jbGllbnQtZ2VuZXJhdG9yLnJlYWR0aGVkb2NzLmlvL2VuL2xhdGVzdC8KICAgIDphbHQ6IERvY3VtZW50YXRpb24gU3RhdHVzCgouLiBpbWFnZTo6IGh0dHA6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3Yvb3BlbmFwaS1jbGllbnQtZ2VuZXJhdG9yLnN2ZwogICAgOnRhcmdldDogaHR0cHM6Ly9weXBpLnB5dGhvbi5vcmcvcHlwaS9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3IKICAgIDphbHQ6IExhdGVzdCBQeVBJIFJlbGVhc2UKCgpPcGVuQVBJIENsaWVudCBHZW5lcmF0b3IKPT09PT09PT09PT09PT09PT09PT09PT09CgpUaGlzIENMSSB1dGlsaXR5IGFsbG93cyB5b3UgdG8gZ2VuZXJhdGUgUHl0aG9uIGNsaWVudCBwYWNrYWdlcyBmcm9tIE9wZW5BUEkgdjMgc3BlY2lmaWNhdGlvbnMuClRoZSBwcm9qZWN0IGFpbXMgYXQgc3VwcG9ydGluZyBhbnkgZ2VuZXJpYyB2YWxpZCBzcGVjaWZpY2F0aW9uLgoKRmVhdHVyZXMKLS0tLS0tLS0KCiogd29ya3Mgb24gUHl0aG9uIDMuOCBhbmQgYWJvdmU7CiogZ2VuZXJhdGVzIGNsaWVudHMgd2l0aCB0eXBlIGhpbnRzIHRvIGFpZCBzdGF0aWMgYW5hbHlzaXMgb2YgaGlnaGVyIGxldmVsIGludGVyZmFjZXMgdGhhdCB3b3VsZCB1c2UgdGhlIGNsaWVudHM7Ciogc3VwcG9ydHMgc3RyZWFtaW5nIGVuZHBvaW50cy4KCkluc3RhbGxhdGlvbgotLS0tLS0tLS0tLS0KCllvdSBjYW4gaW5zdGFsbCBpdCBmcm9tIFB5UEk6CgouLiBjb2RlLWJsb2NrOjogYmFzaAoKICAgIHBpcCBpbnN0YWxsIG9wZW5hcGktY2xpZW50LWdlbmVyYXRvcgoKQWZ0ZXJ3YXJkcywgdXNlIGEgQ0xJIHV0aWxpdHkgY2FsbGVkIGBgb3BlbmFwaS1jbGllbnQtZ2VuZXJhdG9yYGA6CgouLiBjb2RlLWJsb2NrOjogYmFzaAoKICAgICQgb3BlbmFwaS1jbGllbnQtZ2VuZXJhdG9yIC0taGVscAogICAgdXNhZ2U6IG9wZW5hcGktY2xpZW50LWdlbmVyYXRvciBbLWhdIFstVl0ge2dlbn0gLi4uCgogICAgT3BlbkFQSSBDbGllbnQgR2VuZXJhdG9yCgogICAgb3B0aW9uYWwgYXJndW1lbnRzOgogICAgICAtaCwgLS1oZWxwICAgICBzaG93IHRoaXMgaGVscCBtZXNzYWdlIGFuZCBleGl0CiAgICAgIC1WLCAtLXZlcnNpb24gIHNob3cgcHJvZ3JhbSdzIHZlcnNpb24gbnVtYmVyIGFuZCBleGl0CgogICAgc3ViLWNvbW1hbmRzOgogICAgICB2YWxpZCBzdWItY29tbWFuZHMKCiAgICAgIHtnZW59ICAgICAgICAgIGFkZGl0aW9uYWwgaGVscAogICAgICAgIGdlbiAgICAgICAgICBHZW5lcmF0ZSBjbGllbnQgZm9yIGEgcHJvdmlkZWQgc2NoZW1hIChKU09OLCBZQU1MKS4KCgpDb250cmlidXRpbmcKPT09PT09PT09PT09CgpDbG9uaW5nIHRoaXMgcmVwbwotLS0tLS0tLS0tLS0tLS0tLQoKVGhlIHByb3BlciB3YXkgdG8gY2xvbmUgdGhpcyByZXBvIGlzOgoKLi4gY29kZS1ibG9jazo6IGJhc2gKCiAgICBnaXQgY2xvbmUgLS1yZWN1cnNlLXN1Ym1vZHVsZXMgPHJlcG8tdXJsPiA8bG9jYWwtcHJvamVjdC1yb290PgogICAgY2QgPGxvY2FsLXByb2plY3Qtcm9vdD4KCiAgICAjIGZvciBzaG93aW5nIHN1Ym1vZHVsZSBzdGF0dXMgd2l0aCBgZ2l0IHN0YXR1c2AKICAgIGdpdCBjb25maWcgc3RhdHVzLnN1Ym1vZHVsZXN1bW1hcnkgMQoKICAgICMgZm9yIGxvZ2dpbmcgc3VibW9kdWxlIGRpZmYgd2l0aCBgZ2l0IGRpZmZgCiAgICBnaXQgY29uZmlnIGRpZmYuc3VibW9kdWxlIGxvZwoKClRlc3QgZnJhbWV3b3JrCi0tLS0tLS0tLS0tLS0tCgpUaGUgcHJvamVjdCB1c2VzIGBOaXggPGh0dHBzOi8vbml4b3Mub3JnLz5gXyBmb3IgYm9vdHN0cmFwcGluZyBpdHMgZGV2IGVudmlyb25tZW50LgoKWW91IGNhbiBydW4gZXhpc3RpbmcgdGVzdCBzdWl0ZSB3aXRoCgouLiBjb2RlOjoKCiAgICQgbml4LXNoZWxsIC0tcnVuICJtYWtlIHRlc3QiCgoKRG9jdW1lbnRhdGlvbgotLS0tLS0tLS0tLS0tCgpEb2N1bWVudGF0aW9uIGlzIGhvc3RlZCBvbiBSZWFkVGhlRG9jczogaHR0cHM6Ly9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3IucmVhZHRoZWRvY3MuaW8vZW4vZGV2ZWxvcC8KCgpDaGFuZ2Vsb2cKLS0tLS0tLS0tCgpTZWUgYENIQU5HRUxPRyA8aHR0cHM6Ly9naXRodWIuY29tL2F2YW5vdi9vcGVuYXBpLWNsaWVudC1nZW5lcmF0b3IvYmxvYi9tYXN0ZXIvQ0hBTkdFTE9HLnJzdD5gXy4KCkxJQ0VOU0UKLS0tLS0tLQoKTUlUCg== readmeEtag: '"3e9eece9b6a624b8f2da6308ee95ec7d48542d1e"' readmeLastModified: Sun, 19 Jun 2022 16:10:51 GMT repositoryId: 324674329 description: >- Generates Python client packages from OpenAPI v3 specifications. The project aims at supporting any generic valid specification. created: '2020-12-27T02:47:03Z' updated: '2025-01-18T00:58:50Z' language: Python archived: false stars: 9 watchers: 1 forks: 2 owner: avanov logo: https://avatars.githubusercontent.com/u/601955?v=4 license: MIT repoEtag: '"3a06bd6a8ab855cc3c9bdf5c03cd0dcd6b9365f7f5512875b4ed8d27f7930096"' repoLastModified: Sat, 18 Jan 2025 00:58:50 GMT foundInMaster: true category: - SDK - Server Implementations id: dd5e3ca78b17b6b7b556f603493bffa3 - source: openapi3 tags repository: https://github.com/technocreatives/openapi-mock-eller v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHYzIE1vY2sgU2VydmVyCgpBIHZlcnkgYWxwaGEgc2VydmVyIGZvciBnZW5lcmF0aW5nIG1vY2sgZGF0YSBmcm9tIGFuIE9wZW5BUEkgdjMgc3BlYy4KClRoZSBjb2RlYmFzZSBpcyB2ZXJ5IHNtYWxsLCBjb250cmlidXRpb25zIGFyZSB2ZXJ5IHdlbGNvbWUuIDpzbWlsZToKCiMjIFVzYWdlCgpObyBucG0gbW9kdWxlIHlldCwgc29ycnkuIFlvdSBjYW4gY2xvbmUgdGhpcyByZXBvIHRob3VnaCBhbmQgZXhlY3V0ZSBpdCBmcm9tIHdoZXJldmVyIHlvdSBsaWtlOgoKYGBgCm5vZGUgc2VydmVyLmpzIDxzcGVjLnlhbWw+IFstLXBvcnQ9ODAwMV0gWy0taG9zdD1sb2NhbGhvc3RdCmBgYAoKWW91ciBzZXJ2ZXIgaXMgbm93IHJ1bm5pbmcgb24gPGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMT4uCllvdSBjYW4gZmluZCB0aGUgcmVuZGVyZWQgZG9jdW1lbnRhdGlvbiBmb3IgeW91ciBzcGVjIGF0IDxodHRwOi8vbG9jYWxob3N0OjgwMDEvZG9jcz4sCmFuZCwgaWYgeW91IGV2ZXIgbmVlZCBpdCwgdGhlIEpTT04gc291cmNlIG9mIHlvdXIgc3BlYyBhdCA8aHR0cDovL2xvY2FsaG9zdDo4MDAxL2RvY3Mvc2NoZW1hLmpzb24+LgoKIyMgTGljZW5zZQoKSVNDIGxpY2Vuc2UgLSBzZWUgTElDRU5TRSBmaWxlLgo= readmeEtag: '"64b9ad1d887283eca5fbad3356cefec0523bc908"' readmeLastModified: Tue, 24 Jul 2018 21:32:34 GMT repositoryId: 130576221 description: From OpenAPI v3 spec to running mock server in one command created: '2018-04-22T14:39:26Z' updated: '2020-01-19T12:49:24Z' language: JavaScript archived: false stars: 8 watchers: 6 forks: 2 owner: technocreatives logo: https://avatars.githubusercontent.com/u/4668783?v=4 license: ISC repoEtag: '"3bd7905761a583802d06e994dac1ccdc4a1e435208007836d7d61cbf083a8feb"' repoLastModified: Sun, 19 Jan 2020 12:49:24 GMT foundInMaster: true category: - Server - Server Implementations id: 602cb19c468b36fcb1ce63095660ee4f - source: openapi3 tags repository: https://github.com/matelang/dinonce v3: true repositoryMetadata: base64Readme: >- IVtkaW5vbmNlX2dvcGhlcl0oZ29waGVyLnBuZykKIyBkaW5vbmNlIC0gYSBkaXN0cmlidXRlZCBub25jZSB0cmFja2VyCgpbIVttYWluIGJyYW5jaF0oaHR0cHM6Ly9naXRodWIuY29tL3dlbHRoZWUvZGlub25jZS9hY3Rpb25zL3dvcmtmbG93cy9tYWluLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vd2VsdGhlZS9kaW5vbmNlL2FjdGlvbnMvd29ya2Zsb3dzL21haW4ueW1sKQoKRm9yIG1vc3QgYmxvY2tjaGFpbiBjbGllbnRzIGl0IGlzIGVzc2VudGlhbCB0byBrZWVwIHRyYWNrIG9mIHRyYW5zYWN0aW9uIG5vbmNlcyB0aGF0IHByb3RlY3QgYWdhaW5zdCBkdXBsaWNhdGUgCnRyYW5zYWN0aW9ucyBhbmQgcmVwbGF5IGF0dGFja3MuCgpZb3UgY2FuIHJlYWQgbW9yZSBhYm91dCBub25jZXMgW2hlcmVdKGh0dHBzOi8vbWVkaXVtLmNvbS9zd2xoL2V0aGVyZXVtLXNlcmllcy11bmRlcnN0YW5kaW5nLW5vbmNlLTM4NTgxOTRiMzliZikuCgpOb3JtYWxseSwgYSBub24tcGFydGl0aW9uZWQgY2xpZW50IGFwcGxpY2F0aW9uLCBsaWtlIHlvdXIgbW9iaWxlIHdhbGxldCBhcHAgb3IgTWV0YU1hc2sgd2lsbCBlYXNpbHkgZG8gdGhpcyBmb3IgeW91LCAKYnV0IHRyYWNraW5nIG5vbmNlcyBnZXRzIHRyaWNraWVyIG9uY2UgeW91ciBhcHBsaWNhdGlvbiBuZWVkcyB0byBzcGxpdC1icmFpbiBhbmQgcnVuIGluIGEgZGlzdHJpYnV0ZWQgZmFzaGlvbi4KQ2hlY2sgTWV0YU1hc2sncyBub25jZS10cmFja2VyIGltcGxlbWVudGF0aW9uIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vTWV0YU1hc2svbm9uY2UtdHJhY2tlcikuCgpkaW5vbmNlIGlzIGEgbm9uY2UgdGlja2V0aW5nIHNlcnZpY2Ugd2hpY2ggd2UgdXNlIGF0IFt3ZWx0aGVlXShodHRwczovL3dlbHRoZWUuY29tKSB0byBwcm9jZXNzIHRyYW5zYWN0aW9ucyh0eCkgd2l0aCAKbXVsdGlwbGUgdHggZXhlY3V0b3JzLCBtYWtpbmcgc3VyZSB0aGF0IHdlIGF2b2lkOgoKKiBkb3VibGUgc3BlbmRpbmcKKiBmaWxsaW5nIGEgbmV0d29yaydzIHR4IHBvb2wgYnkgaGF2aW5nIGdhcHMgaW4gb3VyIG5vbmNlIHNlcXVlbmNlcwoKIyMgSG93IGl0IHdvcmtzPyBJbiBhIG51dHNoZWxsLgoqZGlub25jZSogaXMgZGVzaWduZWQgdG8gc3VwcG9ydCB0aWNrZXRpbmcgZm9yIG11bHRpcGxlIG5vbmNlIHNlcXVlbmNlcyBpbiBwYXJhbGxlbC4KCkFuIGlkZW50aXR5IG9uIGEgZ2l2ZW4gYmxvY2tjaGFpbiBzaG91bGQgaGF2ZSBpdCdzIG93biBzZXF1ZW5jZSBvZiBub25jZXMuCipkaW5vbmNlKiBkZWZpbmVzIHN1Y2ggYSBzZXF1ZW5jZSBhcyBhIGBsaW5lYWdlYC4KCkZvciBlYWNoIGxpbmVhZ2UgYSAqZGlub25jZSogY2xpZW50IGNhbiBnZXQgYSBsZWFzZWQgbm9uY2UgKnRpY2tldCogZm9yIGEgdHJhbnNjYXRpb24sIGFuZCBzaG91bGQgc3BlY2lmeSBhIHRpY2tldCAKKmV4dGVybmFsSWQqIHdoaWNoIHNob3VsZCBiZSB1bmlxdWUgZm9yIGEgdHJhbnNhY3Rpb24gZm9yIGEgZ2l2ZW4gbGluZWFnZS4gSW4gb3RoZXIgd29yZHMsIGl0IHNob3VsZCB1bmlxdWVseSBpZGVudGlmeSAKYSBuYXR1cmFsIHRyYW5zYWN0aW9uIGluIHRoZSBjYWxsaW5nIHN5c3RlbS4KCklmIHRoZSBvcGVyYXRpb24gc3VjY2VlZHMsICpkaW5vbmNlKiB3aWxsIHJlc2VydmUgaG9sZCBhIGxlYXNlIGZvciB0aGUgdHgncyBuZXdseSBhc3NvY2lhdGVkIG5vbmNlLgoKU2luY2UgbW9zdCB0eCBleGVjdXRvcnMgKCpkaW5vbmNlKidzIGNsaWVudHMpIG9wZXJhdGUgd2l0aCBhbiAqYXQtbGVhc3Qtb25jZSBzZW1hbnRpY3MqLCBpdCBpcyBwb3NzaWJsZSB0aGF0IHRoZSB0eCAKd2lsbDoKKiBjb21wbGV0ZSAoYS5rLmEgd2lsbCBiZSBtaW5lZCBvbiB0aGUgYmxvY2tjaGFpbikKKiB3aWxsIGZhaWwgZm9yIG5vbi10cmFuc2llbnQgcmVhc29ucwoKSWYgdGhlIHR4IGNvbXBsZXRlcyBzdWNjZXNzZnVsbHksIHRoZW4gdGhlIGNsaWVudCBpcyBleHBlY3RlZCB0byAqY2xvc2UgdGhlIHRpY2tldCosIG1hcmtpbmcgaXQgdW5sZWFzZS1hYmxlIGZvcmV2ZXIuCklmIHRoZSB0eCBmYWlscywgdGhlIGNsaWVudCBpcyBleHBlY3RlZCB0byBub3RpZnkgKmRpbm9uY2UqIHRvICpyZWxlYXNlKiB0aGUgdGlja2V0LiBJbiB0aGlzIGNhc2UsIGl0IHNob3VsZCBiZSBhc3NpZ25lZAp0byB0aGUgbmV4dCBsZWFzZSByZXF1ZXN0LCBhbmQgYmUgcmUtdXNlZCBhcyBzb29uIGFzIHBvc3NpYmxlLCB0byBhdm9pZCBmaWxsaW5nIG5vZGUgdHggcG9vbHMgb24gdGhlIGJsb2NrY2hhaW4gbmV0d29yay4KCiMjIENsaWVudCBJbnRlZ3JhdGlvbnMKZGlub25jZSBpcyBidWlsdCB1c2luZyBhIGNvbnRyYWN0IGZpcnN0IGFwcHJvYWNoIHdpdGggT3BlbkFQSSAzLjAuClRoZSBBUEkgZGVmaW5pdGlvbiBjYW4gYmUgZm91bmQgW2hlcmVdKC4vYXBpL2FwaS55YW1sKS4KCllvdSBjYW4gZ2VuZXJhdGUgYSBjbGllbnQgbGlicmFyeSBmb3IgdGhlIGxhbmd1YWdlIG9mIHlvdXIgY2hvaWNlIHVzaW5nIHRoZSAKW29wZW5hcGktZ2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKS4KCiMjIERlcGxveW1lbnQKZGlub25jZSBpcyBwYWNrYWdlZCBhcyBhIERvY2tlciBjb250YWluZXIgYW5kIHB1c2hlZCBhdXRvbWF0aWNhbGx5IHRvIApbRG9ja2VyIEh1Yl0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yZXBvc2l0b3J5L2RvY2tlci93ZWx0aGVlL2Rpbm9uY2UpLgpJZiB5b3UncmUgaW50ZXJlc3RlZCwgY2hlY2sgb3V0IHRoZSBEb2NrZXJmaWxlIHRoYXQgZ2VuZXJhdGVzIHRoZSBpbWFnZSBbaGVyZV0oLi9Eb2NrZXJmaWxlKS4KClNpbmNlIHdlIGFyZSBmYW5zIG9mIGJvdGggS3ViZXJuZXRlcyBhbmQgVGVycmFmb3JtIHdlIGhhdmUgY3JlYXRlZCBhIEhlbG0gQ2hhcnQgdG8gZWFzeSBkZXBsb3ltZW50IHRvIEt1YmVybmV0ZXMuCipUaGUgdXNlciBpcyByZXNwb25zaWJsZSB0byBjcmVhdGUgdGhlIENvbmZpZ01hcCBgZGlub25jZSBjb25maWdgKiwgd2hpY2ggc2hvdWxkIGNvbnRhaW4gYSBzaW5nbGUgYGNvbmZpZy55YW1sYCBkYXRhCmVudHJ5IGFuZCBoYXZlIHRoZSBzdHJ1Y3R1cmUgc2ltaWxhciB0byBbdGhpcyBleGFtcGxlXSguLy5jb25maWcvY29uZmlnLnlhbWwpLgoKV2UgbGlrZSB0byB0aGluZyBhYm91dCB0aGUgc2VydmljZXMgd2UgcnVuIGFzIHNlbGYtY29udGFpbmVkIFRlcnJhZm9ybSBwcm9qZWN0cyB3aXRoIGNsZWFyIGV4dGVybmFsIGRlcGVuZGVuY2llcywgCnNvIHdlJ3ZlIGNyZWF0ZWQgYSBoYW5keSBUZXJyYWZvcm0gbW9kdWxlIGRpcmVjdG9yeSB0byBoZWxwIHdpdGggZGVwbG95bWVudHMuCgpDdXJyZW50bHkgdGhlcmUgaXMganVzdCBvbmUgbW9kdWxlIHRoYXQgeW91IGNhbiB1c2UsIGNhbGxlZCBgaGVsbS1hd3MtcmRzLXBzcWxgLCB3aGljaCB3aWxsIGNyZWF0ZSBhIG1hbmFnZWQgCkFXUyBSRFMgQXVyb3JhIFBvc3RncmVTUUwgZGF0YWJhc2UsIGNyZWF0ZSBhIG5hbWVzcGFjZSBpbiBrdWJlcm5ldGVzIGFuZCBkZXBsb3kgdGhlIGFmb3JlbWVudGlvbmVkIEhlbG0gQ2hhcnQgdG8geW91ciAKY2x1c3Rlci4KClNlZSBhIHVzYWdlIFtleGFtcGxlIGhlcmVdKC4vZGVwbG95bWVudHMvdGVycmFmb3JtL2V4YW1wbGVzL2hlbG0tYXdzLXJkcy1wc3FsKS4KCiMjIEJhY2tlbmRzCipkaW5vbmNlKiBpcyBkZXNpZ25lZCB0byBzdXBwb3J0IG11bHRpcGxlIHN0b3JhZ2UgYmFja2VuZHMgYXMgbG9uZyBhcyB0aGV5IHJlc3BlY3QgdGhlIGFib3ZlIGRlc2NyaWJlZCBzZW1hbnRpY3MuCgpUaGUgaW5pdGlhbCBiYWNrZW5kIHdlIGFyZSBsYXVuY2hpbmcgaXMgUG9zdGdyZVNRTC4KSWYgeW91J2QgbGlrZSB0byBpbXBsZW1lbnQgYSBuZXcgYmFja2VuZCwgZmVlbCBmcmVlIHRvIGRvIHNvIGFuZCBvcGVuIGEgcHVsbCByZXF1ZXN0Lgo= readmeEtag: '"01dee5c171acaf189969c15b8dc8a9ab358eb2e9"' readmeLastModified: Wed, 02 Aug 2023 14:58:15 GMT repositoryId: 370347071 description: >- A blockchain nonce tracker to be used in distributed tx executors with at-least-once semantics. created: '2021-05-24T12:38:30Z' updated: '2025-08-25T14:13:25Z' language: Go archived: false stars: 9 watchers: 2 forks: 1 owner: matelang logo: https://avatars.githubusercontent.com/u/798365?v=4 license: MIT repoEtag: '"c0c96e0ce1019c64c264990450aceb7fb3de765ab1030d2d689b043fc66f3272"' repoLastModified: Mon, 25 Aug 2025 14:13:25 GMT foundInMaster: true category: - Server - Server Implementations id: 633d0faeb506fc54c7d80e636ec6ee09 oldLocations: - https://github.com/welthee/dinonce - source: openapi3 tags repository: https://github.com/theirish81/mockambo v3: true id: b458e9f0c5b8281b2d751d143f34ba29 repositoryMetadata: base64Readme: >- IyBNb2NrYW1ibwojIyBBUEkgTW9ja2luZywgcmVjb3JkaW5nIGFuZCBWYWxpZGF0aW9uCk1vY2thbWJvIGlzIGFuIEFQSSBtb2NraW5nIHNlcnZlciB0aGF0IHBhY2tzIGEgbG90IG9mIHVzZWZ1bCBmZWF0dXJlcyB0bzoKKiBnZW5lcmF0ZSBtb2NrZWQgQVBJcyBvbiB0aGUgZmx5IGJhc2VkIG9uICoqT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyoqIHdpdGhvdXQgYWRkaXRpb25hbCBjb25maWd1cmF0aW9uCiogVmFsaWRhdGUgcmVxdWVzdHMgYW5kIHJlc3BvbnNlcyBhZ2FpbnN0IHRoZSBjb250cmFjdAoqIEZha2UgZGF0YSBjb2hlcmVudGx5CiogU2NyaXB0IHRoZSBiZWhhdmlvciBvZiB0aGUgbW9ja3MgdXNpbmcgSmF2YVNjcmlwdAoqIExvYWQgZHluYW1pYyB0ZW1wbGF0ZXMgdG8gbWFrZSB0aGUgbW9ja3MgKipldmVuIG1vcmUgKmFsaXZlKioqCiogV29yayBhcyBhICoqcmV2ZXJzZSBwcm94eSoqIHJlcXVlc3QvcmVzcG9uc2UgKipyZWNvcmRlciBhbmQgcGxheWJhY2sqKgoKQWxsIHRoaXMgYnkgc2ltcGx5IGFsdGVyaW5nIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24sIGFkZGluZyB0aGUgYHgtbW9ja2FtYm9gIGV4dGVuc2lvbiBzZWN0aW9ucyB3aGVyZSBuZWNlc3NhcnkuCgpNb2NrYW1ibyBpcyBhIHBvd2VyZnVsICpTd2lzcyBBcm15IGtuaWZlKiBmb3IgdGhlIGZvbGxvd2luZyBhY3Rpdml0aWVzOgoqIERlc2lnbiBmaXJzdCBBUEkgZGV2ZWxvcG1lbnQKKiAzcmQgcGFydHkgQVBJIGV4cGxvcmF0aW9uIGFuZCBjbGllbnQgZGV2ZWxvcG1lbnQKKiBTZXJ2aWNlIGlzb2xhdGlvbiBkdXJpbmcgZGV2ZWxvcG1lbnQgYW5kIHRlc3RpbmcKKiBDb250cmFjdCB0ZXN0aW5nIG9mIGJvdGggY2xpZW50cyBhbmQgc2VydmVycwoqIE9mZmxpbmUgZGV2ZWxvcG1lbnQKKiBDSS9DRCBwaXBlbGluZSB0ZXN0aW5nCgojIyBEb2N1bWVudGF0aW9uCkZvciBmdXJ0aGVyIGluZm9ybWF0aW9uLCBkb2N1bWVudGF0aW9uIGFuZCBIb3dUb3MsIFtwbGVhc2UgcmVmZXIgdG8gdGhlIHByb2plY3Qgd2lraV0oaHR0cHM6Ly9naXRodWIuY29tL3RoZWlyaXNoODEvbW9ja2FtYm8vd2lraSku readmeEtag: '"bfe5206f90a294b18da059d8331619a243cd2e1c"' readmeLastModified: Sat, 27 Jul 2024 13:06:09 GMT repositoryId: 827215202 description: Dynamic API Mocking, recording and Validation created: '2024-07-11T08:07:25Z' updated: '2025-11-05T16:56:56Z' language: Go archived: false stars: 8 watchers: 3 forks: 0 owner: theirish81 logo: https://avatars.githubusercontent.com/u/10628601?v=4 license: MIT repoEtag: '"84082b938395936d05d25fbd0a7d8db7bcb3329587258bc545664a9bfc922baa"' repoLastModified: Wed, 05 Nov 2025 16:56:56 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/hslatman/caddy-openapi-validator v3: true repositoryMetadata: base64Readme: >- IyBDYWRkeSBPcGVuQVBJIFZhbGlkYXRvciAoV0lQKQoKQSBbQ2FkZHldKGh0dHBzOi8vY2FkZHlzZXJ2ZXIuY29tLykgbW9kdWxlIHRoYXQgdmFsaWRhdGVzIHJlcXVlc3RzIGFuZCByZXNwb25zZXMgYWdhaW5zdCBhbiBbT3BlbkFQSV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykgc3BlY2lmaWNhdGlvbi4KCiMjIERlc2NyaXB0aW9uCgpUaGUgT3BlbkFQSSBWYWxpZGF0b3IgbW9kdWxlIGlzIGEgW0NhZGR5XShodHRwczovL2NhZGR5c2VydmVyLmNvbS8pIEhUVFAgaGFuZGxlciB0aGF0IHZhbGlkYXRlcyByZXF1ZXN0cyBhbmQgcmVzcG9uc2VzIGFnYWluc3QgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uLgpUaGUgW2tpbi1vcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vZ2V0a2luL2tpbi1vcGVuYXBpKSBgb3BlbmFwaTNmaWx0ZXJgIGxpYnJhcnkgcGVyZm9ybXMgdGhlIGFjdHVhbCB2YWxpZGF0aW9uLgpUaGUgaGFuZGxlciBpbiB0aGlzIHJlcG9zaXRvcnkgaXMgYSBzbWFsbCB3cmFwcGVyIGZvciB0aGUgZnVuY3Rpb25hbGl0eSBwcm92aWRlZCBieSBgb3BlbmFwaTNmaWx0ZXJgLCB3aXRoIGJhc2ljIGNvbmZpZ3VyYXRpb24gYW5kIGludGVncmF0aW9ucyBvcHRpb25zIGZvciBDYWRkeS4gClRoZSBwcm9qZWN0IGlzIGN1cnJlbnRseSBpbiBQT0Mgc3RhZ2UsIG1lYW5pbmcgdGhhdCBtdWNoIG9mIGl0IGNhbiwgYW5kIGxpa2VseSB3aWxsLCBjaGFuZ2UuCgpUaGUgcmVxdWVzdC9yZXNwb25zZSBmbG93IGlzIGFzIGZvbGxvd3M6CgoqIFdoZW4gYSByZXF1ZXN0IGFycml2ZXMsIHRoZSBWYWxpZGF0b3Igd2lsbCBsb29rIGZvciBhIHZhbGlkIHJvdXRlIGluIHRoZSBwcm92aWRlZCBPcGVuQVBJIHNwZWNpZmljYXRpb24gYW5kIHZhbGlkYXRlIHRoZSByZXF1ZXN0IGFnYWluc3QgdGhlIHNjaGVtYS4KKiBUaGUgcmVxdWVzdCBpcyB0aGVuIHBhc3NlZCBvbiB0byB0aGUgbmV4dCBIVFRQIGhhbmRsZXIgaW4gdGhlIGNoYWluIGFuZCB0aGUgVmFsaWRhdG9yIHdpbGwgd2FpdCBmb3IgaXRzIHJlc3BvbnNlLgoqIEFmdGVyIGNhcHR1cmluZyB0aGUgcmVzcG9uc2UsIHRoZSBWYWxpZGF0b3Igd2lsbCB2YWxpZGF0ZSB0aGUgcmVzcG9uc2UgdG8gYmUgdmFsaWQuCiogSWYgbm8gZXJyb3JzIG9jY3VycmVkIGR1cmluZyB0aGUgdmFsaWRhdGlvbiwgdGhlIHJlc3BvbnNlIHdpbGwgYmUgcmV0dXJuZWQuCgojIyBVc2FnZQoKVGhlIHNpbXBsZXN0IHdheSB0byB1c2UgdGhlIE9wZW5BUEkgVmFsaWRhdG9yIEhUVFAgaGFuZGxlciBpcyBieSB1c2luZyBbeGNhZGR5XShodHRwczovL2dpdGh1Yi5jb20vY2FkZHlzZXJ2ZXIveGNhZGR5KToKCmBgYGJhc2gKJCB4Y2FkZHkgYnVpbGQgdjIuMS4xIC0td2l0aCBnaXRodWIuY29tL2hzbGF0bWFuL2NhZGR5LW9wZW5hcGktdmFsaWRhdG9yCmBgYAoKQWx0ZXJuYXRpdmVseSwgdGhlIEhUVFAgaGFuZGxlciBjYW4gYmUgaW5jbHVkZWQgYXMgYSBDYWRkeSBtb2R1bGUgYXMgZm9sbG93czoKCmBgYGdvbGFuZwppbXBvcnQgKAoJXyAiZ2l0aHViLmNvbS9oc2xhdG1hbi9jYWRkeS1vcGVuYXBpLXZhbGlkYXRvciIKKQpgYGAKCkNvbmZpZ3VyZSB0aGUgT3BlbkFQSSBWYWxpZGF0b3IgaGFuZGxlciBhcyBvbmUgb2YgdGhlIGhhbmRsZXJzIHRvIGJlIGV4ZWN1dGVkIGJ5IENhZGR5IChpbiBbSlNPTiBjb25maWddKGh0dHBzOi8vY2FkZHlzZXJ2ZXIuY29tL2RvY3MvanNvbi8pIGZvcm1hdCk6CgpgYGBqc29uCiAgICAuLi4KICAgICAgICAiaGFuZGxlIjogWwogICAgICAgICAgICB7CiAgICAgICAgICAgICAgICAiaGFuZGxlciI6ICJvcGVuYXBpX3ZhbGlkYXRvciIsCiAgICAgICAgICAgICAgICAiZmlsZXBhdGgiOiAiZXhhbXBsZXMvcGV0c3RvcmUueWFtbCIsCiAgICAgICAgICAgICAgICAidmFsaWRhdGVfcm91dGVzIjogdHJ1ZSwKICAgICAgICAgICAgICAgICJ2YWxpZGF0ZV9yZXF1ZXN0cyI6IHRydWUsCiAgICAgICAgICAgICAgICAidmFsaWRhdGVfcmVzcG9uc2VzIjogdHJ1ZSwKICAgICAgICAgICAgICAgICJ2YWxpZGF0ZV9zZXJ2ZXJzIjogdHJ1ZSwKICAgICAgICAgICAgICAgICJ2YWxpZGF0ZV9zZWN1cml0eSI6IHRydWUsCiAgICAgICAgICAgICAgICAicGF0aF9wcmVmaXhfdG9fYmVfdHJpbW1lZCI6ICIiLAogICAgICAgICAgICAgICAgImFkZGl0aW9uYWxfc2VydmVycyI6IFsiIl0sCiAgICAgICAgICAgICAgICAiZW5mb3JjZSI6IHRydWUsCiAgICAgICAgICAgICAgICAibG9nIjogdHJ1ZQogICAgICAgICAgICB9CiAgICAgICAgXQogICAgLi4uCmBgYAoKVGhlIE9wZW5BUEkgVmFsaWRhdG9yIGhhbmRsZXIgc2hvdWxkIGJlIGNhbGxlZCBiZWZvcmUgYW4gYWN0dWFsIEFQSSBpcyBjYWxsZWQuClRoZSBjb25maWd1cmF0aW9uIHNob3duIGFib3ZlIHNob3dzIHRoZSBkZWZhdWx0IHNldHRpbmdzLgpUaGUgYGZpbGVwYXRoYCBjb25maWd1cmF0aW9uIGlzIHJlcXVpcmVkOyB3aXRob3V0IGl0LCBvciB3aGVuIHBvaW50aW5nIHRvIGEgbm9uLWV4aXN0aW5nIGZpbGUsIHRoZSBtb2R1bGUgd29uJ3QgYmUgbG9hZGVkLgoKIyMgRXhhbXBsZQoKQW4gZXhhbXBsZSBvZiB0aGUgT3BlbkFQSSBWYWxpZGF0b3J5IEhUVFAgaGFuZGxlciBpbiB1c2UgY2FuIGJlIGZvdW5kIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vaHNsYXRtYW4vY2FkZHktb3BlbmFwaS12YWxpZGF0b3ItZXhhbXBsZSkuCgojIyBOb3RlcwoKVGhpcyBwcm9qZWN0IGlzIGN1cnJlbnRseSBhIHNtYWxsIFBPQyB3aXRoIHRoZSBpbnRlbnRpb24gdG8gZ3JvdyBpdCBhbG9uZyB3aXRoIG15IG90aGVyIHByb2plY3RzIHVzaW5nIEdvLCBPcGVuQVBJIGFuZCBDYWRkeS4KSSBvbmx5IHJlY2VudGx5IHN0YXJ0ZWQgdXNpbmcgQ2FkZHksIHNvIHRoZXJlIG1heSBiZSBzb21lIHJvdWdoIGVkZ2VzIHRvIGlyb24gb3V0IHdoZW4gbW9yZSB1c2UgY2FzZXMgcHJlc2VudCB0aGVtc2VsdmVzLgoKIyMgVE9ETwoKQSBzbWFsbCBhbmQgaW5jb21wbGV0ZSBsaXN0IG9mIHBvdGVudGlhbCB0aGluZ3MgdG8gaW1wbGVtZW50LCBpbXByb3ZlIGFuZCB0aGluayBhYm91dDoKCiogQWRkIG1vcmUgdGVzdHMgZm9yIHRoZSBPcGVuQVBJIFZhbGlkYXRvciBmdW5jdGlvbmFsaXR5IGFuZCBjb25maWd1cmF0aW9uLgoqIEltcHJvdmUgQ2FkZHlmaWxlIGhhbmRsaW5nIChlLmcuIGFkZCBtb3JlIHN1YmRpcmVjdGl2ZXMpLgoqIEFkZCBhbiBleGFtcGxlIHRoYXQgdXNlcyBhbiBIVFRQIHByb3h5L2ZjZ2kgY29uZmlndXJhdGlvbi4KKiBMb29rIGludG8gd2F5cyB0byBzcGVjaWZ5IHRoZSBlcnJvciBuaWNlbHksIGluc3RlYWQgb2YganVzdCBsb2dnaW5nIGl0IChlLmcuIHJldHVybiBlcnJvciBtZXNzYWdlKHMpIGluIHNwZWNpZmljIGZvcm1hdCkgYW5kL29yIGludGVncmF0ZSBwcm9wZXJseSB3aXRoIGhvdyBDYWRkeSBoYW5kbGVycyBlcnJvcnMuCiogTG9vayBpbnRvIGlmIChhbmQgaG93KSB0aGUgVmFsaWRhdG9yIGNhbiBiZSB1c2VkIG91dHNpZGUgb2YgQ2FkZHkgYXMgYW4gYWx0ZXJuYXRpdmUgKGkuZS4gYSBtb3JlIGdlbmVyaWMgbWlkZGxld2FyZSkuCg== readmeEtag: '"2bd32d69864d68d5dac1de7f81b2e492e7738bc2"' readmeLastModified: Fri, 18 Aug 2023 22:04:50 GMT repositoryId: 286259419 description: >- A Caddy HTTP handler for validating requests and responses against an OpenAPI specification created: '2020-08-09T15:02:31Z' updated: '2026-01-05T01:44:16Z' language: Go archived: false stars: 9 watchers: 1 forks: 0 owner: hslatman logo: https://avatars.githubusercontent.com/u/1219780?v=4 license: Apache-2.0 repoEtag: '"1af217d1c1f77431ddc5a470b238af761a8503e066d5a2bf713c12d3223867f9"' repoLastModified: Mon, 05 Jan 2026 01:44:16 GMT foundInMaster: true category: - Data Validators - Parsers id: 1816046ae5bbe57e73f76829ba004031 - source: openapi3 tags repository: https://github.com/stackql/stackql-registry-docs v3: true id: 9f3eecc809978e3f45543f2ae1e80852 repositoryMetadata: base64Readme: >- IyBzdGFja3FsLXJlZ2lzdHJ5LWRvY3MKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBkb2N1bWVudGF0aW9uIGZvciBTdGFja1FMIHByb3ZpZGVycywgd2hpY2ggaXMgcHVibGlzaGVkIHRvIFtyZWdpc3RyeS5zdGFja3FsLmlvXShodHRwczovL3JlZ2lzdHJ5LnN0YWNrcWwuaW8pLiAgVGhpcyB3ZWJzaXRlIGlzIGJ1aWx0IHVzaW5nIFtEb2N1c2F1cnVzIDJdKGh0dHBzOi8vZG9jdXNhdXJ1cy5pby8pLCBhIG1vZGVybiBzdGF0aWMgd2Vic2l0ZSBnZW5lcmF0b3IuICAKCiMjIEJ1aWxkIGFuZCBEZXBsb3kgU3RhdHVzCgp8IFR5cGUgfCBTdGF0dXMgfCBMaW5rIHwKfCAtLS0gfCAtLS0gfCAtLSB8CnwgR2l0SHViIEFjdGlvbnMgQnVpbGQgfCBbIVtHaXRIdWIgQWN0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL3N0YWNrcWwvc3RhY2txbC1yZWdpc3RyeS1kb2NzL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLWRvY3MueWFtbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vc3RhY2txbC9zdGFja3FsLXJlZ2lzdHJ5LWRvY3MvYWN0aW9ucy93b3JrZmxvd3MvYnVpbGQtZG9jcy55YW1sKSB8IFtzdGFja3FsLXJlZ2lzdHJ5LWRvY3MvYWN0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL3N0YWNrcWwvc3RhY2txbC1yZWdpc3RyeS1kb2NzL2FjdGlvbnMpIHwKfCBOZXRsaWZ5IERlcGxveSBCYXNlIHwgWyFbTmV0bGlmeSBTdGF0dXNdKGh0dHBzOi8vYXBpLm5ldGxpZnkuY29tL2FwaS92MS9iYWRnZXMvNzVmODM4YTktNzllYS00MWRjLTk2ZjktYTQwZGZmNTBjZmNhL2RlcGxveS1zdGF0dXMpXShodHRwczovL2FwcC5uZXRsaWZ5LmNvbS9zaXRlcy9zdGFja3FsLXJlZ2lzdHJ5LWRvY3MvZGVwbG95cykgfCBbcmVnaXN0cnkuc3RhY2txbC5pb10oaHR0cHM6Ly9yZWdpc3RyeS5zdGFja3FsLmlvKSB8CgojIyBBZGRpbmcgRG9jcyBmb3IgYSBOZXcgUHJvdmlkZXIKCkFkZGluZyBkb2NzIGZvciBhIG5ldyBwcm92aWRlciByZXF1aXJlcyBjcmVhdGluZyBhIG5ldyB3ZWIgcHJvcGVydHkgKHN1YmRvbWFpbikgYW5kIHdpbGwgZm9yY2UgYW4gdXBkYXRlIHRvIHRoZSByb290L2Jhc2Ugc2l0ZSBhbmQgYWxsIG90aGVyIHByb3ZpZGVycy4gIFRoZSBzdGVwcyBhcmU6ICAKCi0gWyBdIHVwZGF0ZSBgc2NyaXB0cy9kb2NnZW4vcHJvdmlkZXJfZGF0YS5weWAgd2l0aCBtZXRhZGF0YSBmb3IgbmV3IHByb3ZpZGVyIChub3QgYXBwbGljYWJsZSBpZiBkb2NzIGFyZSBidWlsdCBlbHNld2hlcmUpCi0gWyBdIGdlbmVyYXRlIGRvY3MgZm9yIHByb3ZpZGVyIHVzaW5nIGBjZCBzY3JpcHRzOyBzaCBkb2NnZW4uc2gge3Byb3ZpZGVyfWAgKG5vdCBhcHBsaWNhYmxlIGlmIGRvY3MgYXJlIGJ1aWx0IGVsc2V3aGVyZSkKLSBbIF0gcHVibGlzaCBkb2NzIGZvciBwcm92aWRlciB1c2luZyBgY2Qgc2NyaXB0czsgc2ggcHVibGlzaC5zaCB7cHJvdmlkZXJ9YCAobm90IGFwcGxpY2FibGUgaWYgZG9jcyBhcmUgYnVpbHQgZWxzZXdoZXJlKQotIFt4XSBhZGQgdGhlIG5ldyBwcm92aWRlciBkb2NzIHRvIGBkb2NzL3twcm92aWRlcn0tZG9jc2AgKGZvbGxvd2luZyBkaXJlY3Rvcnkgc3RydWN0dXJlIG9mIGV4aXN0aW5nIHByb3ZpZGVycykgKGRvbmUgYXV0b21hdGljYWxseSBieSBgc2NyaXB0cy9wdWJsaXNoLnNoYCkKLSBbeF0gdXBkYXRlIGZyb250bWF0dGVyIGluIHRoZSBgaW5kZXgubWRgIGF0IHRoZSByb290IG9mIHRoZSBuZXcgcHJvdmlkZXJzIGRvY3MsIHNldCBgc2x1Z2AgdG8gYC9wcm92aWRlcnMve3Byb3ZpZGVyfWAgYW5kIGBpZGAgdG8gYHtwcm92aWRlcn0tZG9jYCAoZG9uZSBhdXRvbWF0aWNhbGx5IGJ5IGBzY3JpcHRzL2RvY2dlbi5zaGApCi0gW3hdIHVwZGF0ZSBgc3RhY2txbC1wcm92aWRlci1yZWdpc3RyeS5tZHhgIGluIHRoZSByb290IG9mIHRoZSBuZXcgcHJvdmlkZXIgdG8gYWRkIHRoZSBgY3VycmVudFByb3ZpZGVyYCBwcm9wLCBlLmcuIGA8UmVnaXN0cnlQYWdlIGN1cnJlbnRQcm92aWRlcj0ib2t0YSIgLz5gIChkb25lIGF1dG9tYXRpY2FsbHkgYnkgYHNjcmlwdHMvZG9jZ2VuLnNoYCkKLSBbIF0gY3JlYXRlIGEgbmV3IG5ldGxpZnkgc2l0ZSAoYHN0YWNrcWwte3Byb3ZpZGVyfS1kb2NzYCkgLSB1c2UgYHlhcm4gYnVpbGQ6e3Byb3ZpZGVyfWAgYXMgdGhlIGBidWlsZCBjb21tYW5kYAotIFsgXSBzdG9wIGF1dG9tYXRpYyBidWlsZHMgZm9yIHNpdGUgaW4gTmV0bGlmeQotIFsgXSBhZGQgYE5FVExJRllgIHJlY29yZCBpbiBOZXRsaWZ5IEROUyAobWFwcGluZyBge3Byb3ZpZGVyfS1kb2NzLnN0YWNrcWwuaW9gIChQcmltYXJ5IERvbWFpbikgYW5kIGB7cHJvdmlkZXJ9LnN0YWNrcWwuaW9gIChEb21haW4gQWxpYXMpIHRvIGBzdGFja3FsLXtwcm92aWRlcn0tZG9jcy5uZXRsaWZ5LmFwcGApLCBzZWxlY3QgYEZvcmNlIEhUVFBTYAotIFsgXSBhZGQgR2l0SHViIEFjdGlvbnMgc2VjcmV0IGZvciBuZXRsaWZ5IHNpdGUgaWQgLSBgTkVUTElGWV9TSVRFX0lEX3tQUk9WSURFUn1gCi0gWyBdIGFkZCB0aGUgTmV0bGlmeSBkZXBsb3kgc3RhdHVzIGJhZGdlIHRvIHRoZSB0YWJsZSBpbiB0aGlzIGBSRUFETUVgIChzZWUgYWJvdmUpCi0gWyBdIHVwZGF0ZSB0aGUgYHByb3ZpZGVyc2AgYXJyYXkgaW4gYHNpZGViYXJzLmpzYCB3aXRoIHRoZSBuZXcgcHJvdmlkZXIKLSBbIF0gdXBkYXRlIHRoZSBgcHJvdmlkZXJzYCBhcnJheSBgc3JjL2NvbmZpZ3MvcHJvdmlkZXJzLnRzYCB3aXRoIHRoZSBuZXcgcHJvdmlkZXIKLSBbIF0gdXBkYXRlIGBwYWNrYWdlLmpzb25gIHdpdGggbmV3IGBzdGFydGAgYW5kIGBidWlsZGAgc2NyaXB0cwotIFsgXSB1cGRhdGUgYGFsbFByb3ZpZGVyc2AgaW4gYGNpLXNjcmlwdHMvZ2V0LXByb3ZpZGVycy10by1kZXBsb3kuanNgIHdpdGggdGhlIG5ldyBwcm92aWRlcgotIFsgXSBhZGQgdGhlIHJvb3Qgc2l0ZSByZWRpcmVjdHMgZm9yIHByb3ZpZGVyIHZhbml0eSB1cmxzIHRvIGAuZ2l0aHViL3dvcmtmbG93cy9idWlsZC1kb2NzLnlhbWxgIHRvIHRoZSBgYWRkIHJlZGlyZWN0cyB0byByb290IHNpdGVgIHN0ZXAgb2YgdGhlIGBkZXBsb3ktdG8tbmV0bGlmeWAgam9iCgojIyBVcGRhdGluZyBFeGlzdGluZyBQcm92aWRlciBEb2NzCgpUbyB1cGRhdGUgdGhlIGRvY3MgZm9yIGFuIGV4aXN0aW5nIHByb3ZpZGVyLCBlZGl0IHRoZSBjb3JyZXNwb25kaW5nIG1hcmtkb3duIGZpbGUgb3IgZmlsZXMgaW4gdGhlIGB7cHJvdmlkZXJ9LWRvY3NgIGRpcmVjdG9yeS4gIFJhaXNlIGEgcHVsbCByZXF1ZXN0IGludG8gdGhlIGBtYWluYCBicmFuY2guICBPbmNlIG1lcmdlZCwgdGhlIHVwZGF0ZWQgZG9jcyB3aWxsIGJlIGF1dG9tYXRpY2FsbHkgcHVibGlzaGVkLiAgCgojIyMgUnVubmluZyBMb2NhbGx5CgpJbnN0YWxsIHBhY2thZ2VzOiAgCgpgYGBiYXNoCnlhcm4KYGBgCgpUbyBidWlsZCBhIG1pY3Jvc2l0ZSBsb2NhbGx5LCB1c2UgYHlhcm4gYnVpbGQ6e3Byb3ZpZGVyfWAuICBGb3IgZXhhbXBsZSwgdG8gYnVpbGQgdGhlIEFXUyBtaWNyb3NpdGUgbG9jYWxseSwgcnVuOgoKYGBgCmV4cG9ydCBOT0RFX09QVElPTlM9LS1tYXhfb2xkX3NwYWNlX3NpemU9NDA5Ngp5YXJuIHJ1biBidWlsZDphd3MKYGBgCgpUbyBydW4gYSBtaWNyb3NpdGUgbG9jYWxseSwgdXNlIGB5YXJuIHN0YXJ0Ontwcm92aWRlcn1gLiAgRm9yIGV4YW1wbGUsIHRvIHJ1biB0aGUgQVdTIG1pY3Jvc2l0ZSBsb2NhbGx5LCBydW46ICAKCmBgYGJhc2gKeWFybiBzdGFydDphd3MKYGBg readmeEtag: '"20243572769d2b278acfb5b1dbec8bc9a5eaaa48"' readmeLastModified: Thu, 18 Sep 2025 17:01:24 GMT repositoryId: 566164202 description: >- StackQL provider documentation for cloud and SaaS providers available in the StackQL Provider Registry, built with Docusaurus. created: '2022-11-15T04:58:21Z' updated: '2025-09-18T17:01:28Z' language: CSS archived: false stars: 8 watchers: 3 forks: 1 owner: stackql logo: https://avatars.githubusercontent.com/u/95105302?v=4 repoEtag: '"77ab9dde95292650a48b741f201634972c8046f13c25935737ea3e0fcedcad7a"' repoLastModified: Thu, 18 Sep 2025 17:01:28 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/arno-di-loreto/supercharged-openapi v3: true repositoryMetadata: base64Readme: >- IyBTdXBlcmNoYXJnZWQgT3BlbkFQSSBTZXNzaW9uIFJlcG9zaXRvcnkgIwoKVGhpcyByZXBvc2l0b3J5IHdhcyB0aGUgb25lIEkgdXNlZCBkdXJpbmcgbXkgU3VwZXJjaGFyZ2VkIE9wZW5BUEkgc2Vzc2lvbiBkb25lIGR1cmluZyB0aGUgW01hbm5pbmcgTGl2ZSBBUEkgQ29uZmVyZW5jZV0oaHR0cHM6Ly9mcmVlY29udGVudC5tYW5uaW5nLmNvbS9saXZlbWFubmluZy1jb25mZXJlbmNlcy1hcGlzLykuCgojIFRvb2xzIHVzZWQvc2hvd24gZHVyaW5nIHRoZSBzZXNzaW9uCgotIFRoZSBzZXNzaW9uIHdhcyByZWNvcmRlZCB1c2luZyBbT0JTIFN0dWRpb10oaHR0cHM6Ly9vYnNwcm9qZWN0LmNvbSkgYW5kIGV2ZXJ5dGhpbmcgd2FzIGRvbmUgaW4gW1ZTIENvZGVdKGh0dHBzOi8vY29kZS52aXN1YWxzdHVkaW8uY29tKSAoc2VlIGJlbG93IGZvciBtb3JlIGRldGFpbHMpCi0gQXQgdGhlIGJlZ2lubmluZyBJIGRyYXcgcmFpbmJvdyBsaWtlIHJlY3RhbmdsZXMgb24gdGhlIHNjcmVlbiB1c2luZyBbU2NyZWVuQnJ1c2hdKGh0dHBzOi8vYXBwcy5hcHBsZS5jb20vdXMvYXBwL3NjcmVlbmJydXNoL2lkMTIzMzk2NTg3MT9tdD0xMikKLSBUaGUgW1JlZG9jXShodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9yZWRvYykgYW5kIFtTd2FnZ2VyIFVJXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkgcmVuZGVyaW5nIG9mIE9wZW5BUEkgZG9jdW1lbnQgd2VyZSBkb25lIHRoYW5rcyB0byB0aGUgW09wZW5BUEkgKDQyIENydW5jaCkgVlMgQ29kZSBFeHRlbnNpb25dKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT00MkNydW5jaC52c2NvZGUtb3BlbmFwaSkKLSBUaGUgbW9jayBBUEkgd2FzIGdlbmVyYXRlZCB3aXRoIFtTdG9wbGlnaHQgUHJpc21dKGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9wbGlnaHRpby9wcmlzbS8pIGJhc2VkIG9uIHRoZSBPcGVuQVBJIGRvY3VtZW50IGNvZGVkIGFsb25nIHRoZSBzZXNzaW9uCi0gVGhlICJ0cmVlIGxpa2UgT3BlbkFQSSBkb2N1bWVudGF0aW9uIiBzaG93biBpbiB0aGUgZW5kIGlzIG15IFtPcGVuQVBJIE1hcF0oaHR0cHM6Ly9vcGVuYXBpLW1hcC5hcGloYW5keW1hbi5pby8pCi0gVGhlIEdVSSBhbGxvd2luZyB0byBub3Qgd3JpdGUgT3BlbkFQSSBjb2RlIGJyaWVmbHkgc2hvd24gaW4gdGhlIGVuZCBpcyBbU3RvcGxpZ2h0IFN0dWRpb10oaHR0cHM6Ly9zdG9wbGlnaHQuaW8vc3R1ZGlvLykKCiMgVGhlIG1hZ2ljIGJlaGluZCB3aGF0IHlvdSd2ZSBzZWVuCgpCZWZvcmUgc3RhcnRpbmcgdG8gdGFsaywgSToKCi0gT3BlbiB0aGlzIHJlcG8gd2l0aCAgVlMgQ29kZQotIFN0YXJ0IGEgbGl2ZSBzZXJ2ZXIgbGlzdGVuaW5nIHRvIGNoYW5nZSAoaXQgcnVucyBvbiBodHRwOi8vbG9jYWxob3N0OjU1MDApCi0gUnVuIHRoZSB0YXNrICJSZXNldCBTdGVwIgotIFN0YXJ0IE9CUwoKVGhlbiB3aGlsZSBJJ20gdGFsa2luZyBhbmQgbW9kaWZ5aW5nIHRoZSBgbW90dS55YW1sYCBmaWxlLCBJIHJlZ3VsYXJseSBoaXQgPGtiZD5jdHJsPC9rYmQ+PGtiZD5tPC9rYmQ+IHRvOgoKLSBMb2FkIGEgbmV3IHZlcnNpb24gb2YgdGhlIGBtb3R1LnlhbWxgIGZpbGUKLSBDaGFuZ2UgdGhlIHRpdGxlIChPQlMgYnJvd3NlciBzb3VyY2Ugc2hvd2luZyBodHRwOi8vbG9jYWxob3N0OjU1MDAvKSBvbiB0b3Agb2YgdGhlIFZTIENvZGUgKE9CUyBkaXNwbGF5IGNhcHR1cmUgc291cmNlKQotIFVwZGF0ZSBteSB0b2RvIGxpc3QgKFNhZmFyaSBzaG93aW5nIGh0dHA6Ly9sb2NhbGhvc3Q6NTUwMC90b2RvLmh0bWwsIG5vdCBzaG93biB0byBhdHRlbmRlZXMpLgoKQWxsIHRoZSBmaWxlcyBhcmUgbG9hZGVkIGZyb20gdGhlIGBzdGVwc2Agc3ViLWZvbGRlcnMgdGhhdCBjb250YWlucyBgaW5kZXguaHRtbGAsIGB0b2RvLmh0bWxgIGFuZCBgbW90dS55YW1sYCBmaWxlcyBmb3IgZWFjaCBzdGVwLgoKQW5kIEkgaGVhdmlseSB1c2UgY29kZSBzbmlwcGV0cyB0byBhdm9pZCBsb29zaW5nIHRpbWUgdHlwaW5nIHRvbyBtdWNoLgoKIVtdKHJlYWRtZS9tYWdpYy5qcGcpCgojIFZTIENvZGUgQ29uZmlndXJhdGlvbgoKIyMgV2FybmluZwoKTm90ZSB0aGF0IGFsbCB0aGlzIGhhcyBiZWVuIGRvbmUgaW4gYSBydXNoLCBpdCBjb3VsZCBjZXJ0YWlubHkgYmUgYmV0dGVyIGRvbmUgYW5kIG9wdGltaXplZCwgYnV0IGl0IHdvcmtlZCDwn5iFLiBUaGUgaGFyZCBjb2RlZCBzdGVwcyBmb2xkZXIgYW5kIGh0bWwgZmlsZXMgd2VyZSBhIHBhaW4gdG8gbW9kaWZ5IChJIGhhZCB0byByZW1vdmUgc29tZSBzdGVwcyB0byBzaG9ydGVuIHRoZSBzZXNzaW9uJ3MgbGVuZ3RoKSBhbmQgdGhlIGJhc2ggc2NyaXB0aW5nIGluIGZ1bGwgb2YgY29weS9wYXN0ZS4KCiMjIFZTIENvZGUgRXh0ZW5zaW9ucwoKLSBbTGl2ZSBTZXJ2ZXJdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1yaXR3aWNrZGV5LkxpdmVTZXJ2ZXIpOiBUbyBzZXJ2ZSB0aGUgaW5kZXggYW5kIHRvZG8gZmlsZSBhbmQgbWFnaWNhbGx5IHJlbG9hZCB0aGVtIHdoZW4gdGhleSBhcmUgbW9kaWZpZWQgYnkgbmV4dCwgcHJldmlvdXMsIHJlc2V0IG91IHJlbG9hZCBzY3JpcHRzCi0gW1N5bnRod2F2ZSB4IEZsdW9yb21hY2hpbmVdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT13ZWJyZW5kZXIuc3ludGh3YXZlLXgtZmx1b3JvbWFjaGluZSk6IFRoZSBzaGlueS1uZW9uIFZTIENvZGUgdGhlbWUgdXNlZCBkdXJpbmcgdGhlIHNlc3Npb24gKEkgYWxzbyByZXVzZWQgdGhpcyB0aGVtZSdzIENTUyBmb3IgdGhlIEhUTUwgaW5kZXggYW5kIHRvZG8gZmlsZXMpLiBUaGlzIHRoZW1lIGlzIHN1cHBvc2VkIHRvIGJlIGxvYWRlZCBhdXRvbWF0aWNhbGx5IHRoYW5rcyB0byB0aGUgYC52c2NvZGUvc2V0dGluZ3MuanNvbmAgY29uZmlndXJhdGlvbiBmaWxlLgotIFtDdXN0b20gQ1NTIGFuZCBKUyBMb2FkZXJdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1iZTVpbnZpcy52c2NvZGUtY3VzdG9tLWNzcyk6IE5lZWRlZCBieSBTeW50aHdhdmUgeCBGbHVvcm9tYWNoaW5lIHRoZW1lCgojIyBTaGVsbCBzY3JpcHRzIGFuZCBWUyBDb2RlIFRhc2tzCgpUaGUgYC52c2NvZGUvdGFza3MuanNvbmAgZmlsZSBkZWZpbmVzIHRoZSBmb2xsb3dpbmcgYWN0aW9ucyAoYWxsIHJlbHlpbmcgYW5kIHNoZWxsIHNjcmlwdHMgbG9jYXRlZCBpbiBgc3RlcHNgKToKCnwgVGFzayAgICAgICAgICB8IFNjcmlwdCB0cmlnZ2VyZWQgICAgfCBEZXNjcmlwdGlvbgp8IC0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tCnwgR28gdG8gc3RlcCAgICB8IGBzdGVwcy9nby5zaCAkc3RlcGAgfCBDb3B5IGBzdGVwcy9zdGVwLXskc3RlcH1gIGNvbnRlbnQgdG8gcm9vdCBsZXZlbCAoZG9lcyBub3RoaW5nIGlmIHN0ZXAgZG9lc24ndCBleGlzdCkuIFRoZSBgJHN0ZXBgIGNhbiBiZSBlaXRoZXIgYSBudW1iZXIgb3IgdGhlIHN0ZXAncyBuYW1lIGNvbWluZyBmcm9tIHRoZSB0b2RvLmh0bWwgZmlsZXMsIGAiTW9yZSBhY2N1cmF0ZSBkYXRhIGRlc2NyaXB0aW9uICg0LzE1KSJgIGZvciBpbnN0YW5jZS4gVGhlIHRhc2tzIHNob3dzIHRoZSBsaXN0IG9mIGF2YWlsYWJsZSBzdGVwcyAoaGFyZGNvZGVkIGluIGB0YXNrcy5qc29uYCBmaWxlKQp8IE5leHQgc3RlcCAgICAgfCBgc3RlcHMvbmV4dC5zaGAgICAgIHwgQ29weSBgc3RlcHMvc3RlcC17Y3VycmVudCBzdGVwICsgMX1gIGNvbnRlbnQgdG8gcm9vdCBsZXZlbCAoZG9lcyBub3RoaW5nIGlmIHN0ZXAgZG9lc24ndCBleGlzdCkKfCBQcmV2aW91cyBzdGVwIHwgYHN0ZXBzL3ByZXZpb3VzLnNoYCB8IENvcHkgYHN0ZXBzL3N0ZXAte2N1cnJlbnQgc3RlcCAtIDF9YCBjb250ZW50IHRvIHJvb3QgbGV2ZWwgKGRvZXMgbm90aGluZyBpZiBzdGVwIGRvZXNuJ3QgZXhpc3QpCnwgUmVsb2FkIHN0ZXAgICB8IGBzdGVwcy9yZWxvYWQuc2hgICAgfCBDb3B5IGBzdGVwcy9zdGVwLXtjdXJyZW50IHN0ZXB9YCBjb250ZW50IHRvIHJvb3QgbGV2ZWwgKHVzZWZ1bCB0byBjaGVjayBtb2RpZmljYXRpb25zIGRvbmUgb24gY3VycmVudCBzdGVwKQp8IFJlc2V0IHN0ZXAgICAgfCBgc3RlcHMvcmVzZXQuc2hgICAgIHwgQ29weSBgc3RlcHMvcmVzZXRgIGNvbnRlbnQgdG8gcm9vdCBsZXZlbCAodG8gcmVzdGFydCBmcm9tIHRoZSBiZWdpbm5pbmcpCnwgQ2xlYW4gYmVmb3JlIGNvbW1pdCB8IGBzdGVwcy9jbGVhbi5zaGAgfCBSZW1vdmUgYGluZGV4LmNzc2AsIGBpbmRleC5odG1sYCwgYHRvZG8uaHRtbGAsIGBtb3R1LnlhbWxgIGZpbGVzIGZyb20gcm9vdCBmb2xkZXIKClRoZSAiY3VycmVudCBzdGVwIiBpcyB0cmFja2VkIGluIHRoZSBgc3RlcHMvY3VycmVudC50eHRgIGZpbGUgKGl0IGlzIGNyZWF0ZWQgaWYgaXQgZG9lc24ndCBleGlzdCkuCgpPcGVuIFZTIGNvbW1hbmQgcGFsZXR0ZSB3aXRoIDxrYmQ+4oyYPC9rYmQ+PGtiZD7ih6c8L2tiZD48a2JkPlA8L2tiZD4gKE1hY09zKSBvciA8a2JkPmN0cmw8L2tiZD48a2JkPuKHpzwva2JkPjxrYmQ+UDwva2JkPiAoV2luZG93cykgYW5kIGNob29zZSBgVGFza3M6IFJ1biB0YXNrYCwgdGhlbiBjaG9vc2UgdGhlIG9uZSB5b3UgbmVlZC4KCiMjIFZTIENvZGUgS2V5Ym9hcmQgc2hvcnRjdXRzCgpUYXNrcyBjYW4gYmUgYmluZCB0byBrZXlib2FyZCBzaG9ydGN1dHMuCk5vdGUgdGhhdCBhcyBpbiBWUyBDb2RlLCBrZXlib2FyZCBzaG9ydGN1dHMgY2FuIG9ubHkgYmUgZGVmaW5lZCBnbG9iYWxseSwgdGhlIGAudnNjb2RlL2tleWJpbmRpbmdzLmpzb25gIGZpbGUgY29udGVudCBzaG91bGQgYmUgY29waWVkIHRoZSBnbG9iYWwgY29uZmlndXJhdGlvbiAoc2hvdyBpdCB3aXRoICkuClRvIHNob3cgc2hvcnRjdXRzIGNvbmZpZ3VyYXRpb24gdXNlIDxrYmQ+4oyYPC9rYmQ+PGtiZD5LPC9rYmQ+PGtiZD7ijJg8L2tiZD48a2JkPlM8L2tiZD4gKE1hY09zKSBvciA8a2JkPmN0cmw8L2tiZD48a2JkPks8L2tiZD48a2JkPmN0cmw8L2tiZD48a2JkPlM8L2tiZD4gKFdpbmRvd3MpLCB0aGVuIGNsaWNrIG9uIHRoZSBmaWxlIGljb24gb24gdG9wIHJpZ2h0IGNvcm5lciB0byBzZWUgdGhlIGpzb24gY29udGVudCBvZiB0aGlzIGNvbmZpZ3VyYXRpb24uCgp8IEtleSBiaW5kaW5nICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgVGFzayB0cmlnZ2VyZWQgfAp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tfAp8IDxrYmQ+Y3RybDwva2JkPjxrYmQ+bTwva2JkPiAgICAgICAgICAgICAgIHwgTmV4dCBzdGVwICAgICAgfAp8IDxrYmQ+Y3RybDwva2JkPjxrYmQ+YWx0PC9rYmQ+PGtiZD5tPC9rYmQ+IHwgUHJldmlvdXMgc3RlcCAgfAoKIyMgVlMgQ29kZSBTbmlwcGV0cwoKSW4gb3JkZXIgdG8gYXZvaWQgbG9vc2luZyB0aW1lIHR5cGluZyB0b28gbXVjaCwgSSB1c2VkIGN1c3RvbSBtYWRlIHNuaXBwZXRzLgpUaGV5J3JlIGRlZmluZWQgaW4gYC52c2NvZGUvc3VwZXJjaGFyZ2VkLW9wZW5hcGkuY29kZS1zbmlwcGV0c2AuCgojIyBWUyBDb2RlIFdpbmRvdyBUaXRsZQoKVGhlIGAudnNjb2RlL3NldHRpbmdzLmpzb25gIHNldHMgdGhlIFZTIENvZGUgd2luZG93IHRpdGxlIHRvICJTdXBlcmNoYXJnZWQgT3BlbkFQSSIuCkkgZGlkIHRoYXQgdG8ga2VlcCBPQlMgZGlzcGxheSBjYXB0dXJlIGNyb3AgY29uZmlndXJhdGlvbiBzaW1wbGUgKGZvY3VzIG9uIGEgc3BlY2lmaWMgd2luZG93IGJhc2VkIG9uIGl0cyB0aXRsZSk= readmeEtag: '"699c51bd1d886e2b4a49e0a981a75b55b7d47fad"' readmeLastModified: Sat, 31 Jul 2021 15:03:00 GMT repositoryId: 388798843 description: >- This repository was the one I used during my Supercharged OpenAPI session done during the Manning Live API Conference. created: '2021-07-23T12:45:37Z' updated: '2022-12-09T23:32:32Z' language: HTML archived: false stars: 8 watchers: 1 forks: 0 owner: arno-di-loreto logo: https://avatars.githubusercontent.com/u/10104551?v=4 repoEtag: '"e97606745020b5271e39d84f1fc19b6be74ea0a50d9ff4394070d5d3e5037756"' repoLastModified: Fri, 09 Dec 2022 23:32:32 GMT foundInMaster: true category: Documentation id: 86a4db4ff05d1111bb32af59b722ef15 - source: openapi3 tags repository: https://github.com/unionj-cloud/unionj-generator v3: true repositoryMetadata: base64Readme: >- # unionj-generator
Unionj-generator is a collection of code generators for spring boot applications with a built-in RESTful api design tool(DSL) compatible with
[OpenAPI 3.0.0 specification](http://spec.openapis.org/oas/v3.0.3).

Including：
- backend: Using dsl or OpenAPI 3.0 json doc to generate vo, proto, controller, service and feign modules for spring boot backend
- openapi: dsl implementing [OpenAPI 3.0.0 specification](http://spec.openapis.org/oas/v3.0.3)
- ui: OpenAPI3 documentation web UI
- maven plugin: maven command line command to generate backend code

<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
### TOC

- [Installation](#installation)
- [OpenAPI3 Web UI Screenshot](#openapi3-web-ui-screenshot)
- [Usage](#usage)
  - [Demo project](#demo-project)
  - [Recommend project structure](#recommend-project-structure)
  - [DSL](#dsl)
    - [Schema](#schema)
      - [Example](#example)
      - [SchemaHelper](#schemahelper)
      - [Generic](#generic)
        - [Syntax](#syntax)
    - [Path](#path)
      - [Example](#example-1)
  - [Backend](#backend)
    - [Example](#example-2)
- [Must Know](#must-know)
- [Tutorials](#tutorials)
- [TODO](#todo)
- [Sister Project](#sister-project)
- [Community](#community)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

# Installation

```xml
<dependencies>
  <dependency>
     <groupId>io.github.unionj-cloud</groupId>
     <artifactId>unionj-generator-backend</artifactId>
     <version>1.6.7</version>
  </dependency>
  <dependency>
     <groupId>io.github.unionj-cloud</groupId>
     <artifactId>unionj-generator-openapi</artifactId>
     <version>1.6.7</version>
  </dependency>
  <dependency>
     <groupId>io.github.unionj-cloud</groupId>
     <artifactId>unionj-generator-ui</artifactId>
     <version>1.6.7</version>
  </dependency>
</dependencies>
```

```xml
<build>
  <plugins>
    <plugin>
      <groupId>io.github.unionj-cloud</groupId>
      <artifactId>unionj-generator-maven-plugin</artifactId>
      <version>1.6.7</version>
      <inherited>false</inherited>
      <executions>
        <execution>
          <id>petStore</id>
          <goals>
            <goal>codegen</goal>
          </goals>
          <inherited>false</inherited>
          <configuration>
            <docUrl>https://petstore3.swagger.io/api/v3/openapi.json</docUrl>
            <feignPkg>io.github.unionj-cloud.feign</feignPkg>
            <feignDir>${project.basedir}/cloud-petStore-feign</feignDir>
            <voPkg>io.github.unionj-cloud.vo</voPkg>
            <voDir>${project.basedir}/cloud-petStore-vo</voDir>
            <packages>
              <package>FEIGN</package>
              <package>VO</package>
            </packages>
            <serviceId>cloud-petStore</serviceId>
            <serviceBaseUrlKey>petStore.baseUrl</serviceBaseUrlKey>
          </configuration>
        </execution>
      </executions>
      <configuration>
        <parentGroupId>io.github.unionj-cloud</parentGroupId>
        <parentArtifactId>cloud-unionj</parentArtifactId>
        <parentVersion>1.0.0-SNAPSHOT</parentVersion>
      </configuration>
    </plugin>
  </plugins>
</build>
```

# OpenAPI3 Web UI Screenshot
![web-ui.png](web-ui.png)

# Usage

## Demo project

It's a simple typescript http client code download restful service project. Upload OpenAPI3 spec json file, download ts code.
It's used in our company project.

Repo: https://github.com/unionj-cloud/openapi-svc

Screenshot:
![screenshot.png](screenshot.png)

## Recommend project structure

Repo: https://github.com/unionj-cloud/unionj-generator-guide

![demo](./demo.png)

## DSL

### Schema

#### Example

```java
import static cloud.unionj.generator.OpenAPI3.dsl.Schema.schema;
import static cloud.unionj.generator.OpenAPI3.dsl.SchemaHelper.*;

public class Components {

  private static Schema sizeProperty = int32("每页条数，默认10，传-1查出全部数据");

  private static Schema currentProperty = int32("当前页，从1开始");

  private static Schema offsetProperty = int32("偏移量");

  private static Schema sortProperty = string("排序条件字符串：排序字段前使用'-'(降序)和'+'(升序)号表示排序规则，多个排序字段用','隔开",
      "+id,-create_at");

  private static Schema pageProperty = int32("当前页，从1开始");

  private static Schema limitProperty = int32("每页条数，默认10, 传-1查出全部数据", 10);

  private static Schema maxPageProperty = int32("导出结束页");

  private static Schema totalProperty = int64("总数，入参传入此参数则不再查询count，以此total为准");

  private static Schema topStatusProperty = int32("需要排在前的状态");
  
  public static Schema PageResultVO = schema(sb -> {
    // Schema type. Required.
    sb.type("object");
    // Schema title. Required. Otherwise the generator tool won't know it.
    sb.title("PageResultVO");
    // Generic as List<T>
    sb.properties("items", ListT);
    sb.properties("total", totalProperty);
    sb.properties("size", sizeProperty);
    sb.properties("current", currentProperty);
    sb.properties("searchCount", bool);
    sb.properties("pages", int32("当前分页总页数"));
    sb.properties("offset", offsetProperty);
  });
  
  public static Schema RankVO = schema(sb -> {
    sb.type("object");
    sb.title("RankVO");
    sb.description("排行榜");
    sb.properties("serialNo", int32);
    sb.properties("name", string);
    sb.properties("income", doubleNumer("累计收入"));
    sb.properties("quantity", int32("完成任务数量"));
  });
  
  public static Schema PageResultVOJobVO = generic(gb -> {
    gb.generic(PageResultVO, ref(RankVO.getTitle()));
  });
}

```



#### SchemaHelper

There are some built-in schemas in cloud.unionj.generator.OpenAPI3.dsl.SchemaHelper.

| Type          | Java                 |
| ------------- | -------------------- |
| int32         | Integer              |
| int64         | Long                 |
| string        | String               |
| bool          | Boolean              |
| floatNumber   | Float                |
| doubleNumer   | Double               |
| dateTime      | java.util.Date       |
| T             | <T>                  |
| ListT         | List<T>              |
| SetT          | Set<T>               |
| stringArray   | List<String>         |
| int32Array    | List<Integer>        |
| int64Array    | List<Long>           |
| floatArray    | List<Float>          |
| doubleArray   | List<Double>         |
| boolArray     | List<Boolean>        |
| dateTimeArray | List<java.util.Date> |
| enums         | enum                 |
| ref           | Object               |
| refArray      | List<Object>         |



#### Generic

##### Syntax

```java
// PageResultVO must has and only has one T like field, e.g. T, List<T>, Set<T>
// It will be represented as PageResultVO<RankVO>
public static Schema PageResultVOJobVO = generic(gb -> {
  gb.generic(PageResultVO, ref(RankVO.getTitle()));
});
```



### Path

#### Example

```java
import static cloud.unionj.generator.OpenAPI3.PathHelper.*;

@Test
public void TestPath() throws IOException {
  OpenAPI3 OpenAPI3 = OpenAPI3(ob -> {
    info(ib -> {
      ib.title("title");
      ib.version("v1.0.0");
    });

    server(sb -> {
      sb.url("http://unionj.cloud");
    });

    // Support GET, POST, PUT, DELETE only.
    post("/hall/onlineSurvey/list", PathConfig.builder()
         .summary("summary")
         .tags(new String[]{"tag1", "tag2"})
         .reqSchema(SearchJobPageResult)
         .respSchema(SearchJobPageResult)
         .build());

    post("/hall/offlineSurvey/update", PathConfig.builder()
         .summary("summary")
         // Second tag will be used as Proto or typescript Service name
         // If there was only one tag, the Proto or typescript Service name will be first part of endpoint
         // e.g. HallProto, HallService
         .tags(new String[]{"tag1", "HallOfflinesurvey"})
         .reqSchema(SearchJobPageResult)
         .respSchema(SearchJobPageResult)
         .build());

    post("/admin/onlineSurvey/top/update", PathConfig.builder()
         .summary("summary")
         .tags(new String[]{"tag1"})
         .parameters(new Parameter[]{
           ParameterBuilder.builder().name("id").in(Parameter.InEnum.QUERY).required(true).schema(string).build(),
           ParameterBuilder.builder().name("top").in(Parameter.InEnum.QUERY).required(true).schema(int32).build(),
         })
         .respSchema(SearchJobPageResult)
         .build());
  });
  Backend backend = BackendDocParser.parse(OpenAPI3);
  SpringbootFolderGenerator springbootFolderGenerator = new SpringbootFolderGenerator.Builder(backend).build();
  springbootFolderGenerator.generate();
}
```



## Backend

### Example

```java
package cloud.unionj.example.proto;

import org.springframework.http.ResponseEntity;
import org.springframework.web.bind.annotation.*;
import org.springframework.web.multipart.MultipartFile;
import java.util.*;
import cloud.unionj.example.vo.*;
import cloud.unionj.example.es.page.PageResult;

public interface AdminProto {

    @PostMapping("/admin/news/list")
    ResultDTO<PageResultVO<NewsVO>> postAdminNewsList(
        @RequestBody BaseSearchCondition body
    );

}
```


# Must Know
- Source code in proto package, vo package and controller package will be replaced with new code completely, 
so don't edit any source code in these packages.
- Existing source code in service package will be skipped and not be changed, so you can edit or add your custom code.

# Tutorials
- [unionj-generator快速上手-后端篇](https://www.jianshu.com/p/21c670ba90f1)

# TODO
Please reference [unionj-generator kanban](https://github.com/unionj-cloud/unionj-generator/projects/1)

# Sister Project

- [go-doudou](https://github.com/unionj-cloud/go-doudou): OpenAPI 3.0 spec based lightweight microservice framework for Go
- [pullcode](https://github.com/wubin1989/pullcode): a typescript http client code generation cli compatible with Swagger 2 and OpenAPI 3

# Community

Welcome to contribute to unionj-generator by forking it and submitting pr or issues. If you like unionj-generator, please give it a star!

Welcome to contact me from 
- facebook: [https://www.facebook.com/bin.wu.94617999/](https://www.facebook.com/bin.wu.94617999/) 
- twitter: [https://twitter.com/BINWU49205513](https://twitter.com/BINWU49205513) 
- email: 328454505@qq.com
- wechat:  
![qrcode.png](qrcode.png) 

 readmeEtag: '"c7b72b4ec2b1303b006507a91865720c2a6b2370"' readmeLastModified: Thu, 16 Nov 2023 08:08:54 GMT repositoryId: 316134895 description: OpenAPI 3.0 specification Dsl and a collection of code generators created: '2020-11-26T05:47:34Z' updated: '2023-05-08T08:29:23Z' language: Java archived: false stars: 8 watchers: 0 forks: 2 owner: unionj-cloud logo: https://avatars.githubusercontent.com/u/79033021?v=4 license: MIT repoEtag: '"2c9524964ff0b913bc1b4ba76cdb74ddcfe6154e2ae84a909e538304e7136138"' repoLastModified: Mon, 08 May 2023 08:29:23 GMT foundInMaster: true category: SDK id: 47459de27ac678ba9515f1a0dbd1fe1c - source: openapi3 tags repository: https://github.com/szasza/express-gateway-plugin-openapi3-mock-server v3: true repositoryMetadata: base64Readme: >- IyBFeHByZXNzIEdhdGV3YXkgT3BlbkFQSSAzIE1vY2sgU2VydmVyIFBsdWdpbgoKWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9leHByZXNzLWdhdGV3YXktcGx1Z2luLW9wZW5hcGkzLW1vY2stc2VydmVyLnN2ZyldKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9leHByZXNzLWdhdGV3YXktcGx1Z2luLW9wZW5hcGkzLW1vY2stc2VydmVyKQoKPiAqKkRFUFJFQ0FUSU9OIE5PVElDRSoqIEZvbGxvd2luZyB0aGUgbWFpbiBFeHByZXNzIEdhdGV3YXkncyBkZXByZWNhdGlvbiwgdGhpcyBwcm9qZWN0IGlzIG5vIGxvbmdlciBtYWludGFpbmVkLgoKIyMgUHJlcmVxdWlzaXRlcwoKUGxlYXNlIGZvbGxvdyB0aGUgaW5zdHJ1Y3Rpb25zIG9uIEV4cHJlc3MgR2F0ZXdheSdzIFtHZXR0aW5nIHN0YXJ0ZWRdKGh0dHA6Ly93d3cuZXhwcmVzcy1nYXRld2F5LmlvL2dldHRpbmctc3RhcnRlZC8pIHBhZ2UuCgpBbHNvIHBsZWFzZSB0YWtlIGEgbG9vayBhdCBob3cgcGx1Z2lucyBhcmUgW2luc3RhbGxlZCBhbmQgZW5hYmxlZF0oaHR0cDovL3d3dy5leHByZXNzLWdhdGV3YXkuaW8vZG9jcy9wbHVnaW5zLykuCgojIyBJbnN0YWxsYXRpb24KCmBlZyBwbHVnaW4gaW5zdGFsbCBleHByZXNzLWdhdGV3YXktcGx1Z2luLW9wZW5hcGkzLW1vY2stc2VydmVyYAoKIyMgQ29uZmlndXJhdGlvbgoKWW91IGNhbiB1c2UgYHNhbXBsZXMvYmFzZS1wYXRocy55bWxgIGZvciB0ZXN0aW5nIHB1cnBvc2VzLiBJZiBzbywgcGxlYXNlIGNyZWF0ZSBhIGBkZWZpbml0aW9uc2AgZm9sZGVyCmluIHRoZSBwcm9qZWN0IHJvb3QgYW5kIGNvcHkgdGhlIHNhbXBsZSBZQU1MIHRoZXJlLgoKX3N5c3RlbS5jb25maWcueW1sXwoKYGBgeWFtbAojIHNvbWUgY29uZmlnIGluIGZyb250IG9mIHRoZSBwbHVnaW5zIHNlY3Rpb24KCnBsdWdpbnM6CiAgZXhwcmVzcy1nYXRld2F5LXBsdWdpbi1vcGVuYXBpMy1tb2NrLXNlcnZlcjoKICAgIGRlZmluaXRpb25GaWxlOiAnZGVmaW5pdGlvbnMveW91cmRlZmluaXRpb24ueW1sJyAjIFRoZSBmaWxlIGNvbnRhaW5pbmcgeW91ciBBUEkncyBzcGVjaWZpY2F0aW9uCgojIHNvbWUgY29uZmlnIGFmdGVyIHRoZSBwbHVnaW5zIHNlY3Rpb24KYGBgCgpfZ2F0ZXdheS5jb25maWcueW1sXyAoQSBiYXJlYm9uZSBleGFtcGxlLCBpZiBub3RoaW5nIGVsc2UgaXMgdXNlZC4pCgpgYGB5YW1sCmh0dHA6CiAgcG9ydDogODAKYXBpRW5kcG9pbnRzOgogIGFwaToKICAgIGhvc3Q6ICcqJwpwb2xpY2llczoKICAtIG1vY2sKcGlwZWxpbmVzOgogIGFwaToKICAgIGFwaUVuZHBvaW50czoKICAgICAgICAtIGFwaQogICAgcG9saWNpZXM6CiAgICAgICAgLSBtb2NrOgogICAgICAgICAgLSBhY3Rpb24KYGBgClRoZSBhYm92ZSBjb25maWd1cmF0aW9uIGRpc2FibGVzIHRoZSBBZG1pbiBDTEkgYW5kIHdpbGwgZG8gbm90aGluZyBidXQgcHJvdmlkZSBhIGR1bW15IHJlc3BvbmRlciBiYXNlZCBvbiB5b3VyIApPcGVuQVBJIHNwZWNpZmljYXRpb24uCgojIyBGZWF0dXJlcwoKKiBFbmRwb2ludHMgd2l0aCBhcHByb3ByaWF0ZSByZXF1ZXN0IHR5cGVzIGFyZSBwYXJzZWQgYW5kIHRoZSBmaXJzdCByZXNwb25zZSBleGFtcGxlIGlzIHVzZWQgYXMgYSByZXNwb25zZSBib2R5LgoqIEhUVFAgYW5kIGFwaUtleSBhdXRoIGhlYWRlciBjaGVjayBzdXBwb3J0CgojIyMgVG8gZG8KCiogQWNjZXB0IGhlYWRlciBjaGVja3MgdG8gcmV0dXJuIGFwcHJvcHJpYXRlIHJlc3BvbnNlIGV4YW1wbGUKKiBYLSogaGVhZGVycyBmb3IgdGVzdGluZyBlcnJvciByZXNwb25zZXMKKiBYLSogaGVhZGVycyBmb3IgcmVzcG9uc2UgZXhhbXBsZSBzZWxlY3Rpb24KKiBJbnB1dCBwYXJhbWV0ZXIgdmFsaWRhdGlvbgoqIEdlbmVyYXRvcnMgZm9yIHJhbmRvbSByZXNwb25zZSBkYXRhIGJhc2VkIG9uIHByb3ZpZGVkIHNjaGVtYQo= readmeEtag: '"cde0d3406ecb9946e27261944e39d5df9bd706f7"' readmeLastModified: Fri, 29 Dec 2023 11:38:45 GMT repositoryId: 106981465 description: An OpenAPI 3 mock server plugin for Express Gateway created: '2017-10-15T03:42:06Z' updated: '2023-12-29T11:39:42Z' language: JavaScript archived: true stars: 8 watchers: 2 forks: 1 owner: Szasza logo: https://avatars.githubusercontent.com/u/911466?v=4 license: NOASSERTION repoEtag: '"f966b3cdcffba9257d86ecbd15cd20c3363cf4a8acb0c04d5fd3e4f67161cadb"' repoLastModified: Fri, 29 Dec 2023 11:39:42 GMT foundInMaster: true category: - Server - Server Implementations id: aa3aa0d74f315ede19ef0315444b9571 - source: openapi3 tags repository: https://github.com/stepci/plugin-openapi v3: true id: 0dc97fe6dc74ed788122f89831c17861 repositoryMetadata: repositoryId: 546665834 description: Step CI OpenAPI support created: '2022-10-06T12:56:04Z' updated: '2023-05-25T16:49:05Z' language: JavaScript archived: false stars: 8 watchers: 2 forks: 5 owner: stepci logo: https://avatars.githubusercontent.com/u/61350067?v=4 license: MPL-2.0 repoEtag: '"d9731c73f2fd75264e677e8a3100879d1be15e3a54aeaeccb8d3282da33bae31"' repoLastModified: Thu, 25 May 2023 16:49:05 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/xiaoyao-ye/initapi v3: true id: 7000a61d32fa3dc7f559c8b49c9c6af7 repositoryMetadata: base64Readme: >- <div align="center"><img src="https://xiaoyao-ye.github.io/blog/initApi/light.svg" /></div>

<br />
<br />

<div align="center"> A tool for generating APIs and types </div>

<br />

<p align="center">
  <a href="https://github.com/xiaoyao-Ye/initapi/blob/main/README.zh-CN.md">中文</a> | English
</p>

<p align="center">
  <a href="https://github.com/xiaoyao-Ye/initapi/stargazers"><img src="https://img.shields.io/github/stars/xiaoyao-Ye/initapi" /></a>
  <a href="https://www.npmjs.com/package/initapi"><img src="https://badgen.net/npm/v/initapi" /></a>
</p>

<br />

## Features

- Code as Document
- Standardized, unified team style, easy to maintain
- Tool generates code to reduce coding and improve efficiency
- If the backend uses Swagger or Openapi3 specifications, it can be accessed at zero cost

## Install

```bash
pnpm i initapi -D
```

## Usage

> configure `api.config.ts` or `api.config.js`

```ts
import { defineConfig } from "initapi";
// you can use the `defineConfig` helper which should provide intellisense without the need for jsdoc annotations:
export default defineConfig({
  // Path related configurations are all based on the working directory of the nodejs process
  // Supports Openapi3 and Swagger specifications, with content created based on specified JSON
  service: {
    pets: {
      url: "https://petstore.swagger.io/v2/swagger.json",
    },
    platform: {
      url: "http://127.0.0.1:3000/api-json",
      commonPrefix: "/api/v1",
    },
    // Can convert local JSON
    local: {
      url: "./assets/openapi.json",
      // API address public prefix, used to generate class names and file names. When not configured, it will automatically attempt to find the public prefix
      commonPrefix: "/mg/api",
    },
  },
  outputDir: "./src/api",
  outputType: "TypeScript",
  // ...
});
```

```json
// package.json
"scripts": {
  "api": "initapi create"
}
```

```bash
pnpm run api
```

Generate API Workflows Example

![Generate API Workflows Example](https://xiaoyao-ye.github.io/blog/initApi/workflow-en.png)

Generate API content example

![Generate API content example](https://xiaoyao-ye.github.io/blog/initApi/api_mul.png)

Generate Type content example

![Generate Type content example](https://xiaoyao-ye.github.io/blog/initApi/typings.png)

## Configure

Detailed configuration items for defineConfig

| Name          | Description                                                                                                                                                                                                           | Type                         | Required | Default                      |
| ------------- | --------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------- | ---------------------------- | -------- | ---------------------------- |
| importRequest | Import axios or axios encapsulation                                                                                                                                                                                   | `string`                     | No       | 'import axios from "axios";' |
| useRequest    | Using axios                                                                                                                                                                                                           | `string`                     | No       | 'axios.request'              |
| service       | The JSON (`swagger/openapi3`specification) file address corresponding to the API service                                                                                                                              | `object`                     | Yes      | —                            |
| outputDir     | Output file storage location                                                                                                                                                                                          | `string`                     | No       | './api'                      |
| outputType    | Output File Type - When there is no configuration, a command line interaction will pop up for selection                                                                                                               | `TypeScript` or `JavaScript` | No       | —                            |
| definition    | How to define types                                                                                                                                                                                                   | `class` or `interface`       | No       | interface                    |
| indexable     | The advantage of using index signature is that it can add any number of attributes, making `interface` or `class` more flexible; The disadvantage is that it may cause uncertainty in the value type of the attribute | `boolean`                    | No       | false                        |
| enumMode      | How to define enumeration (enum: generate enumeration type. type: generate type alias.)                                                                                                                               | `enum` or `type`             | No       | type                         |
| multipleFiles | Enable multi file mode - single file only creates API files. Multiple files will split the API into all controllers and generate corresponding files                                                                  | `boolean`                    | No       | true                         |

## Prettier

Before creating a file, by default, the data will be formatted using a prettier

When the project root directory has a. prettier file, the default configuration will be used for file formatting. If obtaining the file fails, the built-in prettier will be used to format the default configuration:

```JavaScript
const defaultOptions: prettier.Options = {
  parser: "typescript",
  printWidth: 130,
  tabWidth: 2,
  useTabs: false,
  semi: true,
  singleQuote: false,
  quoteProps: "as-needed",
  jsxSingleQuote: false,
  trailingComma: "all",
  bracketSpacing: true,
  bracketSameLine: true,
  jsxBracketSameLine: true,
  arrowParens: "avoid",
  rangeStart: 0,
  rangeEnd: Infinity,
  requirePragma: false,
  insertPragma: false,
  proseWrap: "preserve",
  htmlWhitespaceSensitivity: "ignore",
  endOfLine: "auto",
}
```

## Git commit

- 💍 test: Adding missing tests
- 🎸 feat: A new feature
- 🐛 fix: A bug fix
- 🤖 chore: Build process or auxiliary tool changes
- ✏️ docs: Documentation only changes
- 💡 refactor: A code change that neither fixes a bug or adds a feature
- 💄 style: Markup, white-space, formatting, missing semi-colons...
- 🎡 ci: CI related changes
- ⚡️ perf: A code change that improves performance

## Last

Welcome to provide feedback and contribute code.

## License

MIT
 readmeEtag: '"5477b96f996ae38dbc88551e15939a3b01215940"' readmeLastModified: Thu, 04 Jan 2024 04:01:10 GMT repositoryId: 513418465 description: A tool for creating API and typings created: '2022-07-13T07:13:34Z' updated: '2024-11-04T11:40:52Z' language: JavaScript archived: false stars: 8 watchers: 1 forks: 0 owner: xiaoyao-Ye logo: https://avatars.githubusercontent.com/u/52575104?v=4 license: MIT repoEtag: '"baaadf79bd7597d7dbdce42b106f7ea92fa9774df13b1b5109d4f5a9da08ff07"' repoLastModified: Mon, 04 Nov 2024 11:40:52 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/nicolastakashi/todoapp-openapidocuments v3: true repositoryMetadata: base64Readme: >- IyBUb2RvIEFwcCAtIE9wZW5BUEkgRG9jdW1lbnRzCgohW0NJXShodHRwczovL2dpdGh1Yi5jb20vbmljb2xhc3Rha2FzaGkvdG9kb2FwcC1vcGVuYXBpZG9jdW1lbnRzL3dvcmtmbG93cy9DSS9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcikKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBleGFtcGxlcyB0aGF0IHdlcmUgdXNlZCBpbiB0aGUgcG9zdCBJIG1hZGUgb24gbXkgTWVkaXVtIGFjY291bnQsIHNob3dpbmcgaG93IHRvIGRlc2lnbiBhbiBBUEkgdXNpbmcgYW4gQVBJLUZpcnN0IHN0cmF0ZWd5LgoKIyMjIFBvc3RzCiogRW5nbGlzaAogICogW0Rlc2lnbmluZyBSZXN0ZnVsIEFQSXMgdXNpbmcgYW4gQVBJLUZpcnN0IEFwcHJvYWNoXShodHRwczovL25pY29sYXN0YWthc2hpLm1lZGl1bS5jb20vcmVzdGZ1bC11c2luZy1hcGktZmlyc3QtY2QzMDVlNTkzMDVkKQogICogW0Rlc2lnbmluZyBSZXN0ZnVsIEFQSXMgdXNpbmcgYW4gQVBJLUZpcnN0IEFwcHJvYWNoIOKAlCBNb2NraW5nXShodHRwczovL25pY29sYXN0YWthc2hpLm1lZGl1bS5jb20vZGVzaWduaW5nLWFwaXMtdXNpbmctYW4tYXBpLWZpcnN0LWFwcHJvYWNoLW1vY2stc2VydmVyLTc4ZGRiZDk5OTNmNSkKICAqIFtEZXNpZ25pbmcgUmVzdGZ1bCBBUElzIHVzaW5nIGFuIEFQSS1GaXJzdCBBcHByb2FjaCDigJQgQ29udHJhY3QgVGVzdF0oaHR0cHM6Ly9uaWNvbGFzdGFrYXNoaS5tZWRpdW0uY29tL2Rlc2lnbmluZy1yZXN0ZnVsLWFwaXMtdXNpbmctYW4tYXBpLWZpcnN0LWFwcHJvYWNoLWNvbnRyYWN0LXRlc3QtMzRiZGU3OTYwM2RkKQoKIyMgUnVuaW5nCgpJZiB5b3Ugd2FudCB0byBsaW50IGFuZCBydW4gYSBtb2NrIHNlcnZlciwgeW91IG11c3QgdG8gaW5zdGFsbCB0aGUgZGVwZW5kZW5jaWVzIGJlZm9yZSwgYXMgeW91IGNhbiBzZWUgYmVsb3cuCgpgYGAKeWFybiAKYGBgCgojIyBMaW50aW5nIE9wZW5BUEkgRG9jdW1lbnRzCgpDaGVjayBpZiB5b3VyIEFQSXMgYXJlIGJlaW5nIGRlc2lnbmVkIGZvbGxvd2luZyB0aGUgcHJvcG9zZWQgZ3VpZGVsaW5lcyBieSB5b3VyIGNvbXBhbnksIHdoaWNoIGNvdWxkIGJlIGEgZGlmZmljdWx0IGpvYiB0byBkbyBtYW51YWxseS4KVGhlcmVieSB3ZSBjYW4gYXNrIGZvciBhIHRvb2wgbGlrZSBTcGVjdHJhbCB0byBoZWxwIHVzIGFuZCBjcmVhdGUgYW4gYXV0b25vbW91cyBwcm9jZXNzLCB0byBkbyB0aGlzIGpvYi4KClNwZWN0cmFsIGlzIGFuIG9wZW4tc291cmNlIHByb2plY3QgdGhhdCB3b3JrcyBsaWtlIGEgbGludGVyIHRvIE9wZW5BUEkgZG9jdW1lbnRzIGFuZCBoYXMgYSBsb3Qgb2YgcnVsZXMgdG8gZW5zdXJlIHZlcnkgY29tbW9uIGd1aWRlbGluZXMgaW4gdGhlIHNvZnR3YXJlIGluZHVzdHJ5IHRvIGRldmVsb3AgUkVTVEZ1bCBBUElzLgoKVG8gcnVuIFNwZWN0cmFsIHlvdSBjYW4ganVzdCBydW4gdGhlIGNvbW1hbmQgYmVsb3cuIAoKYGBgCnlhcm4gbGludApgYGAKCiMjIFJ1bm5pbmcgTW9jayBTZXJ2ZXIKCkdpdmUgYSBNb2NrIFNlcnZlciB3aXRoIGR5bmFtaWMgZGF0YSB0byB5b3VyIGNsaWVudHMgdGFzdGUgeW91ciBBUEkgRGVzaWduIGFuZCBwcm92aWRlIGZhc3QgZmVlZGJhY2sgYWJvdXQgdGhlIERldmVsb3BlciBFeHBlcmllbmNlLgoKUHJpc20gaXMgYW4gb3Blbi1zb3VyY2UgcHJvamVjdCB0aGF0IHByb3ZpZGVzIHVzIGEgbW9jayBzZXJ2ZXIgYmFzZWQgb24gdGhlIE9wZW5BUEkgZG9jdW1lbnRzIGFuZCBpdHMgc3RydWN0dXJlLgoKVG8gcnVuIFByaXNtIHlvdSBjYW4ganVzdCBydW4gdGhlIGNvbW1hbmQgYmVsb3cuCgpgYGAKeWFybiBtb2NrCmBgYAoKIyMgQ29udHJhY3QgVGVzdGluZwpFdmVuIGFmdGVyIHRoZSBBUEkgd2FzIG1hZGUgYXZhaWxhYmxlLCBpdCBrZWVwcyBldm9sdmluZywgYW5kIG9uZSBvZiB0aGUgYmlnZ2VzdCBjaGFsbGVuZ2VzIGR1cmluZyBhbiBBUEkgTGlmZS1jeWNsZSBpcyB0aGUgQVBJIGV2b2x1dGlvbiwgaW4gb3JkZXIgdG8gZGV0ZWN0IGJyZWFraW5nIGNoYW5nZXMgYW5kIGF2b2lkIGEgY29udHJhY3QgYnJlYWssIHdlIG11c3QgaGF2ZSBhIHN0cmF0ZWd5IHRvIGJlIG5vdGlmaWVkIGFzIHNvb24gYXMgcG9zc2libGUuCgpCZWZvcmUgeW91IHN0YXJ0IHRvIHRlc3QgdGhlIGNvbnRyYWN0cyB5b3UgbXVzdCBlbnN1cmUgdGhhdCB5b3UgaGF2ZSB0aGUgZG90bmV0IGluc3RhbGxlZCwgc2luY2UgdGhlIEFQSSB1c2VkIGluIHRoaXMgcmVwb3NpdG9yeSBpcyBhbiBBU1AuTkVUIFByb2plY3QuCkFmdGVyIHlvdSBpbnN0YWxsIERvdG5ldCBpbiB5b3VyIGNvbXB1dGVyLCB5b3UgY2FuIGluc3RhbGwgU3dhc2hidWNsbGUgQ0xJLCBydW5uaW5nIHRoZSBjb21tYW5kIGJlbG93LgoKYGBgYmFzaApkb3RuZXQgdG9vbCBpbnN0YWxsIFN3YXNoYnVja2xlLkFzcE5ldENvcmUuQ2xpIC1nCmBgYApPcGVuQVBJIERpZmYgaXMgYW4gb3Blbi1zb3VyY2UgcHJvamVjdCB0aGF0IGNvbXBhcmVzIHR3byBPcGVuQVBJIGRvY3VtZW50cyBhbmQgc2hvd3MgZGlmZmVyZW5jZXMgYmV0d2VlbiB0aG9zZSBmaWxlcywgYW5kIHlvdSBjYW4gc3RhcnQgdG8gdGVzdCB5b3VyIGNvbnRyYWN0cyBydW5uaW5nIHRoZSBjb21tYW5kcyBiZWxvdy4KCmBgYGJhc2gKZG90bmV0IGJ1aWxkIC4vYXBpL1RvZG9BcHAuQXBpL1RvZG9BcHAuQXBpLmNzcHJvaiAtLWNvbmZpZ3VyYXRpb24gUmVsZWFzZQpzd2FnZ2VyIHRvZmlsZSAtLXlhbWwgLS1vdXRwdXQgLi9iaW4vc3dhZ2dlci55YW1sIC4vYXBpL1RvZG9BcHAuQXBpL2Jpbi9SZWxlYXNlL25ldGNvcmVhcHAzLjEvVG9kb0FwcC5BcGkuZGxsIHYxCnlhcm4gY29udHJhY3QtdGVzdApgYGAKCg== readmeEtag: '"17ebb2ebf1125e2314608a468469a8e477f6ea38"' readmeLastModified: Sun, 25 Jul 2021 13:00:11 GMT repositoryId: 285396496 description: Examples used in the post published on Medium. created: '2020-08-05T20:25:43Z' updated: '2023-03-28T14:52:40Z' language: C# archived: false stars: 8 watchers: 1 forks: 1 owner: nicolastakashi logo: https://avatars.githubusercontent.com/u/8205546?v=4 license: MIT repoEtag: '"c77f1d7f28313c62eae668d1580b6b0106dcc0934ddad26e2528f0c6818157f3"' repoLastModified: Tue, 28 Mar 2023 14:52:40 GMT foundInMaster: true category: - Testing - Server Implementations id: 99c2b0a2ca7404d1f8fad9edb1c72c06 - source: openapi3 tags repository: https://github.com/simitgroup/openapigenerator v3: true id: 4d6b006ead52fd9f048f446604ba6382 repositoryMetadata: base64Readme: >- # Openapi Generator
An openapiv3 generator which can generate micro-services for GO Language

## Content
- [Openapi Generator](#openapi-generator)
  - [Contents](#contents)
  - [Introduction](#introduction)
  - [Quickstart](#quickstart)
  - [Dockerizing](#dockerizing)
- [Openapi Document](#openapi-document)
    - [Requirements](#requirements)
    - [Define Environment Variables](#define-environment-variables)
    - [Define Error Schema](#define-error-schema)
    - [Require 2XX and 4XX In Every Response ](#require-2XX-and-4XX-in-every-response)    
    - [Not Supported Features](#not-supported-features)
- [Develop In Generated Project](#develop-in-generated-project)     
    - [Project Overview](#project-overview) 
        - [Development Concept](#development-concept)  
        - [File Structure](#file-structure) 
        - [Schemas Or Models](#Schemas-or-models) 
        - [Routes](#routes) 
        - [Route Handles](#route-handles) 
    - [Develop Real Route Handle](#develop-real-route-handle)
        - [Transfer Route Handle Into New File](transfer-route-handle-into-new-file)
        - [Define Handle Exists In Document](define-handle-exists-in-document)
        - [Create Environment Variables](#create-environment-variables)
    - [Build Project](#build-project) 
        - [Simple Build](#simple-build) 
        - [Build For Distribution](#build-for-distribution) 
        - [Build For Docker](#build-for-docker) 
    - [Unit Test](#unit-test)
        - [Define Real Test](#define-real-test) 
        - [Execute Unit Test](#execute-unit-test)   
    - [Secure Microservice](#secure-microservice) 
        - [Secure with ApiKey](#secure-with-apikey) 
- [Todo](#todo)


## Introduction
We know openapi is cool, usually we use swagger or postman for api testing, documentation and design. However, design, develop, testing and documentation is redundant work and spend us lot of time.

This generator able to generate micro-services with below capability:
1. working gin mock server, you can modified it to become real microservice
2. unit test for all restapi
3. build in swagger-ui to allow you test api easily
4. easily generate docker image
5. build in support security scheme


## Quickstart
Make sure you api document comply to [Requirements](#requirements), then you can generate micro-service source code:
1. [Download openapigenerator](https://github.com/SIMITGROUP/openapigenerator/releases)
2. prepare openapi-v3 spec file, (tested in .yaml only)
3. Execute below command :
```bash
./openapigenerator-mac.bin --apifile="simpleapi.yaml"  --targetfolder=myproject --projectname="project1" --port="9000"  --lang="go"
```
4. You can use Visual Studio Code or others IDE to open `~/myproject`:
```bash
cd myproject
code .
```

5. Copy .env.default to .env, define appropriate value, like API keys and etc
6. Run below command and done:
```bash
make && ./project1 
```
7. Try access rest api interface:
     http://localhost:9000/doc/swagger-ui/index.html



## Dockerizing
Project come with few command help you run microservice in docker environment. Check `Makefile` to know more detail

1. create docker container in devnetwork.
```
cd myproject
make dockerremove ; make docker &&  make dockerrun
make dockershell # access docker shell 
```
2 You may edit `Makefile` if you wish to pass create docker container with more useful setting


# Openapi Document
Pure openapi not good enough for prepare complete microservices, there is additional guide line shall follow at below section. 

## Requirements
This project has below requirement:
1. api document written in openapi v3
2. api document shall save in .yaml format
3. define operationId in every api in path, with value:
    * Alphabet value which can use to define as programming function title
    * No special characters allowed, except prefix with '-' (avoid generate route handle)
    * example: `GetData`, `SaveData`, `-SaveData`
4. All element shall refere from #components/schema, example:
    * parameters
    * request body
    * response content
## Not Supported Features
1. openconnect security schemes
2. apiKey only supported at header, others area is not supported (cookie, query...)

## Define Environment Variables
We can use [Specification Extensions](#https://spec.openapis.org/oas/v3.0.3#specificationExtensions) to declare environment variable for microservices, it will add into `.env.default`.

Example:
```yaml
x-env-vars:
  MONGO_SERVER: mongodbserver
  MONGO_DB: db1
  MONGO_USER: 
  MONGO_PASS: 
```
# Define Error Schema
You can define special schema "Error", which shall include 2 field: `err_code` and `err_msg` as below example.
 http request for `post`,`put` which you may submit requestbody.
```yaml
    Error:
      type: object      
      properties:
        err_code:
          type: string
          example: "ER-MG-001"
        err_msg:
          description: A human readable error message
          type: string
          example: "Mongodb not connected"
        version:
          type: string
          example: v1
```
How it work:
1. This schema can define as error 4xx response for http request with requestbody (post, put, patch)
2. when the actual request body not compatible with schema defination, the `err_msg` will display error like below:
```json
{
  "err_code": "ERR_INPUT_VALIDATION",
  "err_msg": "Key: 'Model_Book.Bookid' Error:Field validation for 'Bookid' failed on the 'required' tag",
  "version": "v1"
}
```

# Require 2XX and 4XX In Every Response
For good practise, every http request shall define at least 1 response for 2xx and 4xx. There is hardcoded behaviour which will identity 2xx as  success response, and 4xx is failed response.



# Develop In Generated Project
After code generated, we can start over development by change the generated code. You shall prepare below dev environment:
1. Visual Studio Code
2. Go Sdk
3. Docker (optional if you play with docker)

## Project Overview
The generated project prepare under below structure:
1. written in go language
2. Running on top of [gin](#https://github.com/gin-gonic/gin) web service
3. It work as complete mock server to service micro-service as stated in openapi document.

### Development Concept
To make the micro-service perform real task, we shall perform development, which involve below scope:
1. Modify route handle to perform real task:
    * `route` is restapi request like `GET /myapi`, `POST /myapi/res1`
    * `route handle` is programming function, which will trigger when client access specific `route`. 
    * Every `route` connect to own `route handle`
    * `route` is terminology from `gin`, it have similar meaning with openapi `path`

2. Unit test (optional)
3. Prepare environment for deployment like:
    * environment variables in `.env` or `.env.docker`
    * prepre dockers images
    * prepare database for handle

On and off, depends on project requirement you need to perform modification at openapi document, and regenerate the code again, and again. It is important to know how to regenerate the code without overwrite your modification. 

Technically it has few rules:
1. Remain all file with prefix Z*.go remain unchange (openapi/Z*.go test/Z*.go)
2. During development, create new `.go` file in folder `openapi/`
3. Created file should not start with `Z`, to avoid it mixed with generated file 
4. Root level file as below guideline:
    * `go.main`: always overwrite by generator
    * `go.mod`:  always overwrite by generator
    * `Makefile`:  always overwrite by generator
    * `.env.default`: always overwrite by generator
    * `.env`: free to change    
    * `.env.docker`: free to change
    
### File Structure
Below is some explanation generated code
* `main.go`: entry point
* `go.mod`: go project setting
* `.env.default`: template for `.env`
* `Makefike`: store command for build project, you may change if you have complex requirement
* `Dockerfile`: template for generate docker image, change it if you have special need on docker
* `openapi/`: store all go program, you shall perform develop in this folder only
* `openapi/ZModel_*.go`: All openapi schema will generate as go `Model` here (structs)
* `openapi/ZRouterRegistry.go`: List all the route/path in this microservice, and connect to which route handle
* `openapi/ZRouterHandle.go`: Keep all default auto generate `route handle`, base on name defined in openapi's path's operationId. 
* `openapi/ZSecurity*.go`: security schemes setting according openapi document
* `openapi/ZServer.go`: store code to run gin server
* `api/api.yaml`: store openapi document, used by swagger-ui
* `dist/*`: project binary build into this folder

### Schemas Or Models
Openapi's schema equivalent to `model` in this project, and all model store as `openapi/ZModel_*.go`. In go, `Model` is kind of struct, suitable interface, getter/setter/validator was prepared.

The model is important cause it act as pattern of api output. Route handle comply output pattern using specific `model`.

### Routes
`Route` equivalent to rest api `request method` + `path`, example
* Get /api/service1
* Post /api/v1/service2
* Put /api
* Delete /api/crud

### Route Handles
1. `Route handle` is programming `function` like `GetStudents()` which is trigger by `route`
2. openapi document `operationId` declare name of `route handle`, and generator will help you prepare dummy route handle. 

## Develop Real Route Handle
To provide real data in micro-service, we perform 3 step below:
1. [Transfer Route Handle Into New File](transfer-route-handle-into-new-file)
2. [Define Handle Exists In Document](define-handle-exists-in-document)
3. [Create Environment Variables](#create-environment-variables)

### Transfer Route Handle Into New File
1. Create new file `openapi/routehandles.go` (or others name you like)
2. `Cut and paste` specific function from `openapi/ZRouteHandle.go` into `openapi/routehandles.go`. Let's assume `getMemoryInfo(c *gin.Context){}`
    * remain `getMemoryInfo()` in `ZRouteHandle.go` will cause duplicate function and cause error
3. Edit content of `openapi/routehandles.go` to serve real data:
```go
// auto generate by generator
package openapi

import (
	"fmt"
	"runtime"

	"github.com/gin-gonic/gin"
	// "net/http"
)


func getMemoryInfo(c *gin.Context) {

	var m runtime.MemStats
	runtime.ReadMemStats(&m)
	// For info on each, see: https://golang.org/pkg/runtime/#MemStats
	allocated := bToMb(m.TotalAlloc)
	available := bToMb(m.Sys)
	percent := int((allocated/available)*100) / 100
	allocatedstr := fmt.Sprintf("%v MB", allocated)
	availablestr := fmt.Sprintf("%v MB", available)
	percentstr := fmt.Sprintf("%v percent", percent)
	c.Header("Content-Type", "application/json")
	// 
	// type Model_MemoriesInfo struct{
	// 	Percent string `json:"percent" binding:""` //
	// 	Total string `json:"total" binding:""` //
	// 	Used string `json:"used" binding:""` //
    //  Version string `json:"version"`
	// }

	data := Model_MemoriesInfo{} //Model_MemoriesInfo defined at openapi/ZModel_MemoriesInfo.go
	data.SetTotal(percentstr)
	data.SetPercent(availablestr)
	data.SetUsed(allocatedstr)
    data.SetVersion("v1")
	c.JSON(200, data)
}

func bToMb(b uint64) uint64 {
	return b / 1024 / 1024
}

```
### Define Handle Exists In Document
1. Edit the original openapi document, add entry `x-operationId-exists`:
```yaml
# bottom of file
x-operationId-exists:
  getMemoryInfo: true
```
2. Try regenerate the source code, `getMemoryInfo()` no longer generate in `ZRouteHandle.go`

### Create Environment Variables
Lot of time, we wish to use configure microservice according deployment requirement, such as:
1. Define database server location and credentials
2. On/Off specific functions
3. Define apikey and etc


1. Define environment variable in openapi document using `x-env-vars` and regenerate the source code. 
Example in openapi document:
```yaml
# bottom of file
x-operationId-exists:
  getMemoryInfo: true
x-env-vars:
  APIVERSION: v1.1
```
Generated `.env.default` will automatically add below entry
```env
APIVERSION=v1.1
```
2. load environment variable in route handle using `godotenv.load()` and `os.Getenv("APIVERSION")`. Example of updated code:
```go

package openapi

import (
	"fmt"
	"os"
	"runtime"

	"github.com/gin-gonic/gin"
	"github.com/joho/godotenv"
	log "github.com/sirupsen/logrus"
	// "net/http"
)

func getMemoryInfo(c *gin.Context) {
	err := godotenv.Load()
	if err != nil {
		log.Fatal("Error loading .env file")
	}
	verionno := os.Getenv("APIVERSION")
	var m runtime.MemStats
	runtime.ReadMemStats(&m)
	// For info on each, see: https://golang.org/pkg/runtime/#MemStats
	allocated := bToMb(m.TotalAlloc)
	available := bToMb(m.Sys)
	percent := int((allocated/available)*100) / 100
	allocatedstr := fmt.Sprintf("%v MB", allocated)
	availablestr := fmt.Sprintf("%v MB", available)
	percentstr := fmt.Sprintf("%v percent", percent)
	c.Header("Content-Type", "application/json")
	//
	// type Model_MemoriesInfo struct{
	// 	Percent string `json:"percent" binding:""` //
	// 	Total string `json:"total" binding:""` //
	// 	Used string `json:"used" binding:""` //
	// }

	data := Model_MemoriesInfo{} //Model_MemoriesInfo defined at openapi/ZModel_MemoriesInfo.go
	data.SetTotal(percentstr)
	data.SetPercent(availablestr)
	data.SetUsed(allocatedstr)
	data.SetVersion(verionno)
	c.JSON(200, data)
}

func bToMb(b uint64) uint64 {
	return b / 1024 / 1024
}

```

3. define environment variable via:
    * edit `.env` in non-docker build
    * edit `.env.docker` in docker build
    * or, modify it in command line, `export X_API_Key=12345` 

## Build Project
We shall build the project to run the micro-service, follow below instruction.

### Simple Build
1. run below command:
```bash
make
```
2. To run the service:
Linux/Mac:
```bash
./<your-project-name>
```
Windows: double click the filename `<your-project-name>`
### Build For Distribution
If you wish to build binary for specific OS and distribute manually. Folow step
1. Run below command:
```bash
make windows #build for windows
make linux #build for linux
make mac #build for mac
make mac-arm #build for mac m1 type processor

```
2. Distrubute file in `dist/`, along with suitable `.env` file

### Build For Docker
We can build and run docker image via below step:
1. Copy `.env.default` to `.env.docker`, and define appropriate content
2. run command:
```bash
make dockerremove #remove existing docker 
make docker # create docker image only
make dockerrun # create docker container
```
or we can simplified as single row command:
```bash
make dockerremove; make docker && make dockerrun
```
3. We can change Docker image build from different source by edit `Dockerfile`. Like replace `alpine` to `ubuntu`
4. We can change docker image/container/network by edit `Makefile`

## Unit Test
Generator help you prepare reasonable structure for unit test, however you need to copy content into suitable file to prevent effort overwritten when regenerate code.
You need to install `grc` package for unit test
https://jakeholmquist.medium.com/add-some-fun-to-your-cli-with-grc-ea868df985b6


### Define Real Test
1. expand folder ./test, open any file (assume ZGet_Memory_test.go)
2. scroll down and follow comment, copy content into test/Get_Memory.go (File name shall follow comment)
```go
// copy and modify below content and put into new file Get_Memory.go (in this test folder)
/*
package test
import (
	"io"
...
```
3. after step 2, error in ZGet_Memory_test.go should disappear
4. Modify `Get_Memory.go` if neccessary to serve real content
5. repeat same step 1-4 for others file until all error disappear:
    * if request body is required, `FunctionName_RequestBody()` will fill in sample data which defined in openapi document
    * You can change whole structure of unit test as long as the original function name remain
    * Don't change any file start with Z*.go, cause it will overwritten by generator

### Execute Unit Test
Ensure microservice is activated, run below command and see the result:
```bash
make apitest
```

## Secure Microservice
We can secure restapi via define secruityScheme in openapi document. 

### Secure With Apikey
1. Add security Schemes. Below is apiKey example:
```yaml
components:
  securitySchemes:
    BasicApiKey:
      type: apiKey
      in: header
      name: X-Api-Key #environment variable X_Api_Key will prepare automatically
```
2. Define which api use it:
```yaml
paths:
  /api1:
    get:
      summary: welcome
      description: show msg undefine resource
      operationId: "welcome"
      security:
        - BasicApiKey: []  # Get /api1 require X-Api-Key defined in header
```

3. Re-generate the source code:
    * `X_Api_Key` will automatically prepare in `.env.default`
    * `openapi/ZSecurity_BasicApiKeyy.go` generated
    * unit test template use env var X_Api_Key in header
4. Update `.env` and `.env.docker` if both file exists, and restart the micro-service
5. Try the `Get /api` again with header `X-Api-Key` as:
```bash
http get localhost:<portno>/api1 X-Api-Key:<you-key-code>
```

# Todo
1. Add `x-generator-setting` to direct set project name, port number and etc suitable configuration
2. rename this project to prevent crash name with official openapi-generator
3. support more component type
4. client generators for different kind of languages
5. add some common template for
   * crud for different kind of database
   * messaging template for sms, email, push notifications
6. openapi 3.1
7. more securityschemes
8. more complete data validation

 readmeEtag: '"4edb2a5baffc244679f5881a9fb93723099c2f96"' readmeLastModified: Fri, 09 Dec 2022 13:48:50 GMT repositoryId: 537831519 description: Openapi v3 microservices generator created: '2022-09-17T14:21:57Z' updated: '2024-01-12T18:38:11Z' language: Go archived: false stars: 8 watchers: 1 forks: 1 owner: SIMITGROUP logo: https://avatars.githubusercontent.com/u/5270141?v=4 license: MIT repoEtag: '"1de7746b338a8a111caaeb6281fde19ac909f3cf7c11764619ab94e43a3410e7"' repoLastModified: Fri, 12 Jan 2024 18:38:11 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/shavo007/k6-demo v3: true repositoryMetadata: base64Readme: >- IyBUeXBlU2NyaXB0IHdpdGggazYKCiFbYmFubmVyXShhc3NldHMvdHMtanMtazYucG5nKQoKaHR0cHM6Ly9ibG9nLnNoYW5lbGVlLm5hbWUvMjAyMS8xMi8xNS9wZXJmb3JtYW5jZS10ZXN0aW5nLXdpdGgtazYvCgpUaGlzIHJlcG9zaXRvcnkgc2hvd2Nhc2VzIGs2IHdpdGggdHlwZXNjcmlwdCBhbmQgZ2VuZXJhdGluZyBib2lsZXJwbGF0ZSBzY3JpcHRzIGZyb20geW91ciBPQVMgKE9wZW4gQVBJIFNwZWMpLiBrNiByZWNlbnRseSBvdXRsaW5lZCBpbiB0aG91Z2h0d29ya3MgW3RlY2ggcmFkYXJdKGh0dHBzOi8vd3d3LnRob3VnaHR3b3Jrcy5jb20vcmFkYXIvdG9vbHM/YmxpcGlkPTIwMjAxMDA3OCkgdG9vbHMgcXVhZHJhbnQuCgohWy5naXRodWIvd29ya2Zsb3dzL3B1c2gueW1sXShodHRwczovL2dpdGh1Yi5jb20vc2hhdm8wMDcvazYtZGVtby93b3JrZmxvd3MvLmdpdGh1Yi93b3JrZmxvd3MvcHVzaC55bWwvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKQohW0dpdEh1YiB0b3AgbGFuZ3VhZ2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhbmd1YWdlcy90b3Avc2hhdm8wMDcvazYtZGVtbykKCi0gW1R5cGVTY3JpcHQgd2l0aCBrNl0oI3R5cGVzY3JpcHQtd2l0aC1rNikKICAtIFtQcmVyZXF1aXNpdGVzXSgjcHJlcmVxdWlzaXRlcykKICAtIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCiAgICAtIFtJbnN0YWxsIGRlcGVuZGVuY2llc10oI2luc3RhbGwtZGVwZW5kZW5jaWVzKQogIC0gW1VzYWdlL0V4YW1wbGVzXSgjdXNhZ2VleGFtcGxlcykKICAgIC0gW1J1bm5pbmcgdGhlIHRlc3RdKCNydW5uaW5nLXRoZS10ZXN0KQogICAgLSBbT0FTIGludGVncmF0aW9uXSgjb2FzLWludGVncmF0aW9uKQogICAgICAtIFtUZXN0IHJ1bm5pbmcgYWdhaW5zdCB0aGUgQVBJXSgjdGVzdC1ydW5uaW5nLWFnYWluc3QtdGhlLWFwaSkKICAgIC0gW09BdXRoMiBpbnRlZ3JhdGlvbl0oI29hdXRoMi1pbnRlZ3JhdGlvbikKICAgIC0gW0xvYWQgdGVzdGluZyB3aXRoIGluZmx1eGRiIGFuZCBncmFmYW5hXSgjbG9hZC10ZXN0aW5nLXdpdGgtaW5mbHV4ZGItYW5kLWdyYWZhbmEpCiAgLSBbUmVzb3VyY2VzXSgjcmVzb3VyY2VzKQogIC0gW1RPRE9dKCN0b2RvKQoKIyMgUHJlcmVxdWlzaXRlcwoKLSBbazZdKGh0dHBzOi8vazYuaW8vZG9jcy9nZXR0aW5nLXN0YXJ0ZWQvaW5zdGFsbGF0aW9uKQotIFtOb2RlSlNdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi9kb3dubG9hZC8pCi0gW1lhcm5dKGh0dHBzOi8veWFybnBrZy5jb20vZ2V0dGluZy1zdGFydGVkL2luc3RhbGwpCi0gW0RvY2tlcl0oaHR0cHM6Ly9kb2NzLmRvY2tlci5jb20vZ2V0LWRvY2tlci8pCgojIyBJbnN0YWxsYXRpb24KCiMjIyBJbnN0YWxsIGRlcGVuZGVuY2llcwoKQ2xvbmUgdGhlIGdlbmVyYXRlZCByZXBvc2l0b3J5IG9uIHlvdXIgbG9jYWwgbWFjaGluZSwgbW92ZSB0byB0aGUgcHJvamVjdCByb290IGZvbGRlciBhbmQgaW5zdGFsbCB0aGUgZGVwZW5kZW5jaWVzIGRlZmluZWQgaW4gW2BwYWNrYWdlLmpzb25gXSguL3BhY2thZ2UuanNvbikKCmBgYGJhc2gKeWFybgpgYGAKCiMjIFVzYWdlL0V4YW1wbGVzCgojIyMgUnVubmluZyB0aGUgdGVzdAoKVG8gcnVuIGEgdGVzdCB3cml0dGVuIGluIFR5cGVTY3JpcHQsIHdlIGZpcnN0IGhhdmUgdG8gdHJhbnNwaWxlIHRoZSBUeXBlU2NyaXB0IGNvZGUgaW50byBKYXZhU2NyaXB0IGFuZCBidW5kbGUgdGhlIHByb2plY3QKCmBgYGJhc2gKeWFybiBidW5kbGUKYGBgCgpUaGlzIGNvbW1hbmQgY3JlYXRlcyB0aGUgZmluYWwgdGVzdCBmaWxlcyB0byB0aGUgYC4vZGlzdGAgZm9sZGVyLgoKT25jZSB0aGF0IGlzIGRvbmUsIHdlIGNhbiBydW4gb3VyIHNjcmlwdCB0aGUgc2FtZSB3YXkgd2UgdXN1YWxseSBkbywgZm9yIGluc3RhbmNlOgoKYGBgYmFzaAprNiBydW4gZGlzdC90ZXN0MS5qcwpgYGAKCiMjIyBPQVMgaW50ZWdyYXRpb24KCj4gU2hvd2Nhc2UgZ2VuIGs2IHNjcmlwdCBmcm9tIE9BUyAoT3BlbiBBUEkgU3BlYykKCkdlbmVyYXRlIGs2cyBzY3JpcHRzIHZpYSBbb3BlbiBhcGkgZ2VuZXJhdG9yXShodHRwczovL29wZW5hcGktZ2VuZXJhdG9yLnRlY2gvZG9jcy9nZW5lcmF0b3JzLyNjbGllbnQtZ2VuZXJhdG9ycykgYW5kIHRoZSBbb2FzIGZpbGVdKG9hczMueWFtbCkuIENhbiBydW4gdGhlIENMSSB2aWEgdGhlIGRvY2tlciBpbWFnZS4KCmBgYGJhc2gKZG9ja2VyIHB1bGwgb3BlbmFwaXRvb2xzL29wZW5hcGktZ2VuZXJhdG9yLWNsaQoKZG9ja2VyIHJ1biAtLXJtIC12ICR7UFdEfTovbG9jYWwgb3BlbmFwaXRvb2xzL29wZW5hcGktZ2VuZXJhdG9yLWNsaSBnZW5lcmF0ZSBcCiAgICAtaSAvbG9jYWwvb2FzMy55YW1sIFwKICAgIC1nIGs2IFwKICAgIC1vIC9sb2NhbC9rNi1vYXMzLyBcCiAgICAtLXNraXAtdmFsaWRhdGUtc3BlYwoKYGBgCgpUaGlzIGdlbmVyYXRlcyBbc2NyaXB0LmpzXSguL2s2LW9hczMvc2NyaXB0LmpzKSBhbmQgaXMgYSBncmVhdCBzdGFydCB0byBoZWxwIHN1cHBvcnQgZGVmaW5pbmcgeW91ciBwZXJmIHRlc3QgY2FzZXMuCgoqKk5CKiogSXQgaXMgYm9pbGVycGxhdGUgc28gd2lsbCBuZWVkIHRvIGJlIHJlZmluZWQgYWZ0ZXIgZm9yIHJlLXVzZQoKVGhpcyBhdXRvLWdlbmVyYXRpb24gb2YgdGhlIGxvYWQgdGVzdCBzY3JpcHQgd2lsbCBoZWxwIHN0cmVhbWxpbmUgdGhlIEFQSSB0ZXN0aW5nIHByb2Nlc3MsIGtlZXBpbmcgb24gcGFyIHdpdGggdGhlIGxhdGVzdCBjaGFuZ2VzIHRvIHRoZWlyIEFQSXMgYW5kIHNwZWNpZmljYXRpb25zLgoKIyMjIyBUZXN0IHJ1bm5pbmcgYWdhaW5zdCB0aGUgQVBJCgpgYGBiYXNoCmRvY2tlciBydW4gLWQgLXA4MDkwOjgwODEgc2hhbmVsZWUwMDcvZ3JlZXRpbmdzLWFwaTpsYXRlc3QgI3J1biBncmVldGluZ3MgQVBJCnlhcm4gYnVuZGxlCms2IHJ1biAtZSBFTlY9bG9jYWwgZGlzdC9ncmVldGluZ3MuanMKeWFybiBodG1sICNnZW5lcmF0ZSBhIGh0bWwgcmVwb3J0CmBgYAoKIyMjIE9BdXRoMiBpbnRlZ3JhdGlvbgoKPiBTaG93Y2FzZSBnZW4gdHlwZXMgZnJvbSBvcGVuIGFwaSBnZW5lcmF0b3IgYW5kIGs2IHNjcmlwdHMgZm9yIFticGF5IEFQSV0oLi9icGF5L29hczMueWFtbCkKCi0gR2VuZXJhdGUgdHlwZXNjcmlwdCBjbGllbnQgc2RrIChmb3IgdHlwZXMpCi0gR2VuZXJhdGUgazYgc2NyaXB0cyBmcm9tIE9BUwoKYGBgYmFzaAojIGdlbiBjbGllbnQgc2RrIGZyb20gT0FTIHVzaW5nIGdlbmVyYXRvciB0eXBlc2NyaXB0CmRvY2tlciBydW4gLS1ybSAtdiAke1BXRH06L2xvY2FsIG9wZW5hcGl0b29scy9vcGVuYXBpLWdlbmVyYXRvci1jbGkgZ2VuZXJhdGUgXAogICAgLWkgL2xvY2FsL2JwYXkvb2FzMy55YW1sIFwKICAgIC1nIHR5cGVzY3JpcHQgXAogICAgLW8gL2xvY2FsL2JwYXkvYnBheS1jbGllbnQvIFwKICAgIC0tYWRkaXRpb25hbC1wcm9wZXJ0aWVzPXN1cHBvcnRzRVM2PXRydWUscGxhdGZvcm09bm9kZSBcCiAgICAtLXNraXAtdmFsaWRhdGUtc3BlYwoKIyBPUiBnZW5lcmF0ZSB0eXBlcyBmcm9tIE9BUyB2aWEgb3BlbmFwaS10eXBlc2NyaXB0Cm5weCBvcGVuYXBpLXR5cGVzY3JpcHQgYnBheS9vYXMzLnlhbWwgLS1vdXRwdXQgYnBheS90eXBlcy9zY2hlbWEudHMKbnB4IG9wZW5hcGktdHlwZXNjcmlwdCBicGF5L29hczMueWFtbCAtLW91dHB1dCBzcmMvYnBheS9zY2hlbWEudHMKCiMgZ2VuIGs2IHNjcmlwdHMKZG9ja2VyIHJ1biAtLXJtIC12ICR7UFdEfTovbG9jYWwgb3BlbmFwaXRvb2xzL29wZW5hcGktZ2VuZXJhdG9yLWNsaSBnZW5lcmF0ZSBcCiAgICAtaSAvbG9jYWwvYnBheS9vYXMzLnlhbWwgXAogICAgLWcgazYgXAogICAgLW8gL2xvY2FsL2JwYXkvazYvIFwKICAgIC0tc2tpcC12YWxpZGF0ZS1zcGVjCgojcnVuIGxvYWQgdGVzdCBhZ2FpbnN0IGJwYXkgYXBpCnlhcm4gYnVuZGxlCkNMSUVOVF9JRD1SRURBQ1RFRCBDTElFTlRfU0VDUkVUPVJFREFDVEVEIGs2IHJ1biBkaXN0L2JwYXkuanMKYGBgCgojIyMgTG9hZCB0ZXN0aW5nIHdpdGggaW5mbHV4ZGIgYW5kIGdyYWZhbmEKCmBgYGJhc2gKZG9ja2VyIGNvbXBvc2UgdXAgLWQgI3J1biB0aGUgc2VydmljZXMgaW4gdGhlIGJhY2tncm91bmQKZG9ja2VyLWNvbXBvc2UgcnVuIC12IFwKICAgICRQV0QvZGlzdDovc2NyaXB0cyBcCiAgICBrNiBydW4gLWUgRU5WPWRvY2tlciAgL3NjcmlwdHMvZ3JlZXRpbmdzLmpzCgpgYGAKCkFjY2VzcyBncmFmYW5hOiBgb3BlbiBodHRwOi8vbG9jYWxob3N0OjMwMDBgIGFuZCB2ZXJpZnkgY29ubmVjdGlvbiB0byBpbmZsdXhkYiBkYXRhc291cmNlLiBBY2Nlc3MgdGhlIHByZS1kZWZpbmVkIGRhc2hib2FyZCBhbmQgc2V0IGludGVydmFsIHRvIHRoZSBwYXN0IDUgbWlucy4gWW91IHNob3VsZCBzZWUgc2ltaWxhciBvdXRsaW5lIHRvIGJlbG93CgohW0dyYWZhbmEgZGFzaGJvYXJkXSguL2Fzc2V0cy9ncmFmYW5hLnBuZykKCiMjIFJlc291cmNlcwoKLSBbazYgYW5kIG9hc10oaHR0cHM6Ly9rNi5pby9ibG9nL2xvYWQtdGVzdGluZy15b3VyLWFwaS13aXRoLXN3YWdnZXItb3BlbmFwaS1hbmQtazYvKQotIFtrNiB2cyBqbWV0ZXJdKGh0dHBzOi8vazYuaW8vYmxvZy9rNi12cy1qbWV0ZXIvKQotIFtwcmV0dGllciBwcmUtY29tbWl0XShodHRwczovL3ByZXR0aWVyLmlvL2RvY3MvZW4vcHJlY29tbWl0Lmh0bWwpCgojIyBUT0RPCgotIHJhaXNlIGlzc3VlIG9uIHN0ZG91dCBjaGFycwotIHNob3djYXNlIGJyb3dzZXIgcmVjb3JkZXIK readmeEtag: '"66be7b89fd66b5b28af9b5bf3e10f01be38221c0"' readmeLastModified: Fri, 23 Jun 2023 07:00:22 GMT repositoryId: 360389583 description: demo perf tests using k6 and OAS (open api spec) created: '2021-04-22T04:16:45Z' updated: '2024-10-26T10:12:24Z' language: TypeScript archived: false stars: 8 watchers: 3 forks: 0 owner: shavo007 logo: https://avatars.githubusercontent.com/u/5466825?v=4 license: MIT repoEtag: '"c433602ca0885f54204a3e3ac4ae24250f6ca89495b5d51844406a709c6baf3a"' repoLastModified: Sat, 26 Oct 2024 10:12:24 GMT foundInMaster: true category: Code Generators id: 4d586216eabc1d69aa1fe6cb703eee26 - source: openapi3 tags repository: https://github.com/ikemtz/openapi-ts-generator v3: true repositoryMetadata: base64Readme: >- WyFbQnVpbGQgU3RhdHVzXShodHRwczovL2lrZW10ei52aXN1YWxzdHVkaW8uY29tL0NJJTIwQ0QvX2FwaXMvYnVpbGQvc3RhdHVzL29wZW5hcGktdHMtZ2VuZXJhdG9yP2JyYW5jaE5hbWU9bWFzdGVyKV0oaHR0cHM6Ly9pa2VtdHoudmlzdWFsc3R1ZGlvLmNvbS9DSSUyMENEL19idWlsZC9sYXRlc3Q/ZGVmaW5pdGlvbklkPTIwJmJyYW5jaE5hbWU9bWFzdGVyKSBbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL29wZW5hcGktdHMtZ2VuZXJhdG9yLnN2ZyldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29wZW5hcGktdHMtZ2VuZXJhdG9yKSBbIVtucG0gZG93bmxvYWRzXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9kdC9vcGVuYXBpLXRzLWdlbmVyYXRvcildKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29wZW5hcGktdHMtZ2VuZXJhdG9yKSBbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1pa2VtdHpfb3BlbmFwaS10cy1nZW5lcmF0b3ImbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9aWtlbXR6X29wZW5hcGktdHMtZ2VuZXJhdG9yKSBbIVtJc3N1ZXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy1yYXcvaWtlbXR6L09wZW5BcGktVFMtR2VuZXJhdG9yKV0oaHR0cHM6Ly9naXRodWIuY29tL2lrZW10ei9vcGVuYXBpLXRzLWdlbmVyYXRvci9pc3N1ZXMpIFshW0NvdmVyYWdlXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1pa2VtdHpfb3BlbmFwaS10cy1nZW5lcmF0b3ImbWV0cmljPWNvdmVyYWdlKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1pa2VtdHpfb3BlbmFwaS10cy1nZW5lcmF0b3IpIFshW0xpbmVzIG9mIENvZGVdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PWlrZW10el9vcGVuYXBpLXRzLWdlbmVyYXRvciZtZXRyaWM9bmNsb2MpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPWlrZW10el9vcGVuYXBpLXRzLWdlbmVyYXRvcikgWyFbU2VjdXJpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1pa2VtdHpfb3BlbmFwaS10cy1nZW5lcmF0b3ImbWV0cmljPXNlY3VyaXR5X3JhdGluZyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9aWtlbXR6X29wZW5hcGktdHMtZ2VuZXJhdG9yKSBbIVtDb2RlUUxdKGh0dHBzOi8vZ2l0aHViLmNvbS9pa2VtdHovb3BlbmFwaS10cy1nZW5lcmF0b3IvYWN0aW9ucy93b3JrZmxvd3MvY29kZXFsLWFuYWx5c2lzLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vaWtlbXR6L29wZW5hcGktdHMtZ2VuZXJhdG9yL2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVxbC1hbmFseXNpcy55bWwpCgojIE9wZW5BcGktVFMtR2VuZXJhdG9yCgpOUE0gcGFja2FnZSBiYXNlZCBvbiBbc3dhZ2dlci10cy1nZW5lcmF0b3JdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3N3YWdnZXItdHMtZ2VuZXJhdG9yKSB0byBnZW5lcmF0ZSB0eXBlc2NyaXB0IG1vZGVscyBmb3IgZW5kcG9pbnRzIGRvY3VtZW50ZWQgYnkgc3dhZ2dlciB1c2luZyB0aGUgcmVsYXRpdmVseSBuZXcgW09wZW5BUEkgc3BlY10oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9hYm91dC8pLgoKIyMgVHlwZVNjcmlwdCB1c2FnZSB3aXRoIGEgaG9zdGVkIE9wZW5BcGkgU3BlYyBkb2N1bWVudC4KCmBgYGphdmFzY3JpcHQKaW1wb3J0IHsgZ2VuZXJhdGVUc01vZGVscyB9IGZyb20gJ29wZW5hcGktdHMtZ2VuZXJhdG9yJzsKCmdlbmVyYXRlVHNNb2RlbHMoewogIG9wZW5BcGlKc29uVXJsOiAne1lvdXIgU3dhZ2dlciBFbnBvaW50IFVSTCBoZXJlfScsCiAgb3V0cHV0UGF0aDogJy4ve291dHB1dEZvbGRlcn0vJywKfSk7CmBgYAoKIyMgVHlwZVNjcmlwdCB1c2FnZSB3aXRoIGFuIE9wZW5BcGkgU3BlYyBkb2N1bWVudCBzdG9yZWQgb24geW91ciBsb2NhbCBjb21wdXRlci4KCmBgYGphdmFzY3JpcHQKaW1wb3J0IHsgZ2VuZXJhdGVUc01vZGVscyB9IGZyb20gJ29wZW5hcGktdHMtZ2VuZXJhdG9yJzsKCmdlbmVyYXRlVHNNb2RlbHMoewogIG9wZW5BcGlKc29uRmlsZU5hbWU6ICd7bG9jYXRpb24gYW5kIGZpbGUgbmFtZSBvZiB5b3VyIE9wZW5BcGkgZG9jdW1lbnR9JywKICBvdXRwdXRQYXRoOiAnLi97b3V0cHV0Rm9sZGVyfS8nLAp9KTsKYGBgCgojIyBKYXZhc2NyaXB0IHVzYWdlIHdpdGggYSBob3N0ZWQgT3BlbkFwaSBTcGVjIGRvY3VtZW50LgoKYGBgamF2YXNjcmlwdApjb25zdCBnZW5lcmF0b3IgPSByZXF1aXJlKCdvcGVuYXBpLXRzLWdlbmVyYXRvcicpOwoKZ2VuZXJhdG9yLmdlbmVyYXRlVHNNb2RlbHMoewogIG9wZW5BcGlKc29uVXJsOiAne1lvdXIgU3dhZ2dlciBFbnBvaW50IFVSTCBoZXJlfScsCiAgb3V0cHV0UGF0aDogJy4ve291dHB1dEZvbGRlcn0vJywKICBnZW5Bbmd1bGFyRm9ybUdyb3VwczogdHJ1ZSAvKiBTZXQgdGhpcyB0byB0cnVlIGlmIG9ubHkgaWYgeW91J3JlIGluIGFuIEFuZ3VsYXIgcHJvamVjdCovCn0pOwpgYGAKCiMjIEphdmFzY3JpcHQgdXNhZ2Ugd2l0aCBhbiBPcGVuQXBpIFNwZWMgZG9jdW1lbnQgc3RvcmVkIG9uIHlvdXIgbG9jYWwgY29tcHV0ZXIuCgpgYGBqYXZhc2NyaXB0CmNvbnN0IGdlbmVyYXRvciA9IHJlcXVpcmUoJ29wZW5hcGktdHMtZ2VuZXJhdG9yJyk7CgpnZW5lcmF0b3IuZ2VuZXJhdGVUc01vZGVscyh7CiAgb3BlbkFwaUpzb25GaWxlTmFtZTogJ3tsb2NhdGlvbiBhbmQgZmlsZSBuYW1lIG9mIHlvdXIgT3BlbkFwaSBkb2N1bWVudH0nLAogIG91dHB1dFBhdGg6ICcuL3tvdXRwdXRGb2xkZXJ9LycsCn0pOwpgYGAKCiMjIFdhbnQgdG8gaW5jbHVkZSBBbmd1bGFyIEZvcm1Hcm91cCBGYWN0b3JpZXM/CgpgYGBqYXZhc2NyaXB0CmNvbnN0IGdlbmVyYXRvciA9IHJlcXVpcmUoJ29wZW5hcGktdHMtZ2VuZXJhdG9yJyk7CgpnZW5lcmF0b3IuZ2VuZXJhdGVUc01vZGVscyh7CiAgb3BlbkFwaUpzb25GaWxlTmFtZTogJ3tsb2NhdGlvbiBhbmQgZmlsZSBuYW1lIG9mIHlvdXIgT3BlbkFwaSBkb2N1bWVudH0nLAogIG91dHB1dFBhdGg6ICcuL3tvdXRwdXRGb2xkZXJ9LycsCiAgZ2VuQW5ndWxhckZvcm1Hcm91cHM6IHRydWUKfSk7CmBgYAoKIyMgV29ya2luZyBleGFtcGxlIHdpdGggTlJTUnggYmFzZWQgc2VydmljZQoKYGBgamF2YXNjcmlwdAppbXBvcnQgeyBnZW5lcmF0ZVRzTW9kZWxzIH0gZnJvbSAnb3BlbmFwaS10cy1nZW5lcmF0b3InOwoKZ2VuZXJhdGVUc01vZGVscyh7CiAgb3BlbkFwaUpzb25Vcmw6ICdodHRwczovL2ltLXdhLWNtcG8tbnJzci5henVyZXdlYnNpdGVzLm5ldC9zd2FnZ2VyL3YxL3N3YWdnZXIuanNvbicsCiAgb3V0cHV0UGF0aDogJy4vbW9kZWxzLycsCiAgdHlwZUZpbHRlckNhbGxCYWNrOiAodmFsLCBpLCBhcnIpID0+ICF2YWwubmFtZS5lbmRzV2l0aCgnT0RhdGFFbnZlbG9wZScpLAogIHZhbHVlUHJvcGVydHlUeXBlRmlsdGVyQ2FsbEJhY2s6ICh2YWwsIGksIGFycikgPT4gIXZhbC5uYW1lLnN0YXJ0c1dpdGgoJ2NyZWF0ZWQnKSAmJiAhdmFsLm5hbWUuc3RhcnRzV2l0aCgndXBkYXRlZCcpLAp9KTsKYGBgCgpDaGVjayBvdXQgW29wZW5hcGktbWVybWFpZF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS1tZXJtYWlkKSwgaXQgd2lsbCBhdXRvLWdlbmVyYXRlIG1lcm1haWQgZW50aXR5IGRpYWdyYW1zIHVzaW5nIHlvdXIgc3dhZ2dlciBkb2NzLgo= readmeEtag: '"9d868013c8ad2230384eb0a0e72d077033d6c9c0"' readmeLastModified: Wed, 28 Feb 2024 06:22:26 GMT repositoryId: 231400862 description: >- NPM package based on swagger-ts-generator to generate typescript models for endpoints documented by swagger using the relatively new OpenAPI spec. created: '2020-01-02T14:43:45Z' updated: '2026-02-01T00:09:45Z' language: TypeScript archived: false stars: 8 watchers: 1 forks: 3 owner: ikemtz logo: https://avatars.githubusercontent.com/u/6444182?v=4 license: MIT repoEtag: '"f5885fcb7074ccda5751919eadf974126e0a5bd5ea4ec46415ce64e3c2011467"' repoLastModified: Sun, 01 Feb 2026 00:09:45 GMT foundInMaster: true category: SDK id: 6dae950c0e9f5ec81c345d076f76834e - source: openapi3 tags repository: https://github.com/meabed/swagger-to-typescript-client v3: true repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50ClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9naXRodWIuY29tL21lYWJlZC9zd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50L2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbWVhYmVkL3N3YWdnZXItdG8tdHlwZXNjcmlwdC1jbGllbnQvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sKQpbIVtOUE0gdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9zd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50LnN2ZyldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3N3YWdnZXItdG8tdHlwZXNjcmlwdC1jbGllbnQpClshW0Rvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0vc3dhZ2dlci10by10eXBlc2NyaXB0LWNsaWVudC5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9zd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50KQpbIVtVTlBLR10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9VTlBLRy1PSy0xNzlCRDcuc3ZnKV0oaHR0cHM6Ly91bnBrZy5jb20vYnJvd3NlL3N3YWdnZXItdG8tdHlwZXNjcmlwdC1jbGllbnRAbGF0ZXN0LykKCgojIyMgQSBDTEkgdG8gR2VuZXJhdGUgdHlwZXNjcmlwdCBheGlvcyBjbGllbnQgZnJvbSBzd2FnZ2VyICYgb3BlbmFwaSBmaWxlCgojIyBJbnN0YWxsYXRpb24KTlBNIApgYGBucG0KbnBtIGluc3RhbGwgLS1zYXZlIHN3YWdnZXItdG8tdHlwZXNjcmlwdC1jbGllbnQKYGBgCllhcm4KYGBgeWFybgp5YXJuIGFkZCBzd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50CmBgYAoKCgojIyBVc2FnZQpgYGAKbm9kZSAuL25vZGVfbW9kdWxlcy9zd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50L2Rpc3QvaW5kZXguanMgLS1wa2cgY2xpZW50LW5hbWUgLS1vdXRwdXQgLi9zZGsgLS1zd2FnZ2VyIC4vc3dhZ2dlci5qc29uCmBgYAoKT3IgdXNlIE5QWCAKYGBgCm5weCBzd2FnZ2VyLXRvLXR5cGVzY3JpcHQtY2xpZW50IC0tcGtnIGNsaWVudC1uYW1lIC0tb3V0cHV0IC4vc2RrIC0tc3dhZ2dlciAuL3N3YWdnZXIuanNvbgpgYGAKCiMjIEV4YW1wbGUKVEJBCg== readmeEtag: '"ab535ba3dc0fda805be6793e7c98c8c4788e1dcc"' readmeLastModified: Wed, 26 Apr 2023 12:59:24 GMT repositoryId: 412679955 description: A CLI to Generate typescript axios client from swagger & openapi file created: '2021-10-02T03:30:44Z' updated: '2025-06-25T15:57:25Z' language: TypeScript archived: false stars: 7 watchers: 1 forks: 0 owner: meabed logo: https://avatars.githubusercontent.com/u/45731?v=4 license: MIT repoEtag: '"ac542d2af5b4b2ce0ec60bf9a544146a51862ba3bec8beeb1dc100ba0da706dd"' repoLastModified: Wed, 25 Jun 2025 15:57:25 GMT foundInMaster: true category: - SDK - Parsers id: 0b0bcf6834af8b5ba21743dc059e389d - source: openapi3 tags repository: https://github.com/playtikaoss/rapidoc-spring-boot v3: true id: 83ce1dc12859550644790d9d62b58766 repositoryMetadata: base64Readme: >- PSBSYXBpZG9jIFNwcmluZyBCb290IExpYnJhcnkKOnRvYzoKCipUTERSOiogTW9kZXJuIFN3YWdnZXIgVUkgZm9yIHlvdXIgT3BlbiBBcGkgc3BlY2lmaWNhdGlvbiBpbiBTcHJpbmcgQm9vdCBzZXJ2aWNlcy4KCj09IFdobyBpcyBleHBlY3RlZCB0byB1c2UgdGhpcyBsaWJyYXJ5PwoKLSB5b3UgaGF2ZSBPcGVuQXBpIFNwZWNpZmljYXRpb24gZmlsZSBmb3IgeW91ciBBUEk7Ci0geW91IHVzZSBodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yL3RyZWUvbWFzdGVyL21vZHVsZXMvb3BlbmFwaS1nZW5lcmF0b3ItbWF2ZW4tcGx1Z2luW09wZW5BcGkgQ29kZSBHZW5lcmF0b3IgcGx1Z2luXTsKLSB5b3Ugd2FudCBTd2FnZ2VyIFVJIHRvIGJlIGJhc2VkIG9uIG9yaWdpbmFsIE9wZW5BcGkgU3BlY2lmaWNhdGlvbiBmaWxlOwotIHlvdSBsaWtlIGhvdyBodHRwczovL3JhcGlkb2N3ZWIuY29tL2luZGV4Lmh0bWxbUmFwaURvY10gVUkgbG9va3MgbGlrZTsKLSB5b3UgaGF2ZSBTcHJpbmcgQm9vdCBhcHBsaWNhdGlvbi4KCj09IEhvdyBVSSBsb29rcyBsaWtlPwoKaW1hZ2U6OnJhcGlkb2MtdWkucG5nW3dpZHRoPTgwMF0KCk9yIGhhdmUgYSBsb29rIGhlcmU6IGh0dHBzOi8vcmFwaWRvY3dlYi5jb20vZXhhbXBsZXMvcGV0c3RvcmUtZXh0ZW5kZWQuaHRtbAoKPT0gSG93IHRvIGNvbmZpZ3VyZT8KCi4gKltJbXBvcnRhbnRdKiBSZW1vdmUgU3ByaW5nRG9jIGFuZCBTd2FnZ2VyIFVJIGRlcGVuZGVuY2llcworCk1ha2Ugc3VyZSB0aGF0IHRoZXJlIGlzIG5vIGBvcmcuc3ByaW5nZG9jIDogc3ByaW5nZG9jLW9wZW5hcGktdWlgIGFuZCBgb3JnLndlYmphcnMgOiBzd2FnZ2VyLXVpYCBkZXBlbmRlbmNpZXMgb24gY2xhc3NwYXRoLgoKLiBBZGQgZGVwZW5kZW5jeSBgY29tLnBsYXl0aWthIDogcmFwaWRvYy1zcHJpbmctYm9vdGAKKwotLQoKW3NvdXJjZSx4bWxdCi0tLS0KPGRlcGVuZGVuY3k+CiAgICA8Z3JvdXBJZD5jb20ucGxheXRpa2E8L2dyb3VwSWQ+CiAgICA8YXJ0aWZhY3RJZD5yYXBpZG9jLXNwcmluZy1ib290PC9hcnRpZmFjdElkPgogICAgPHZlcnNpb24+aW5zZXJ0X3ZlcnNpb25faGVyZTwvdmVyc2lvbj4KPC9kZXBlbmRlbmN5PgotLS0tCi0tCgouIENvbmZpZ3VyZSBwcm9wZXJ0aWVzCisKCnw9PT0KfFByb3BlcnR5IHxEZXNjcmlwdGlvbiB8RGVmYXVsdCB2YWx1ZQoKfGBvcGVuYXBpLnJhcGlkb2MuZW5hYmxlZGAKfEVuYWJsZXMvZGlzYWJsZXMgYXV0b2NvbmZpZ3VyYXRpb24gZm9yIFJhcGlkb2MuCnxgdHJ1ZWAKCnxgb3BlbmFwaS5yYXBpZG9jLm9wZW4tYXBpLXNwZWMuZmlsZS1uYW1lYAp8TmFtZSBvZiB0aGUgT3BlbiBBcGkgU3BlYyBmaWxlLgp8YG9wZW5hcGkueWFtbGAKCnxgb3BlbmFwaS5yYXBpZG9jLm9wZW4tYXBpLXNwZWMucGF0aGAKfFBhdGggdG8gdGhlIE9wZW4gQXBpIFNwZWMgZmlsZSBpbnNpZGUgYC9zcmMvbWFpbi9yZXNvdXJjZXNgIGZvbGRlci4KfGAvb3BlbmFwaS9gCgp8PT09CgoKPT0gQXZhaWxhYmxlIGVuZHBvaW50cwoKTGlicmFyeSBleHBvc2VzIHRoZSBmb2xsb3dpbmcgZW5kcG9pbnRzOgoKLSBgL3YzL2FwaS1kb2NzYCAtLSBwcm92aWRlcyBPcGVuIEFwaSBTcGVjaWZpY2F0aW9uIGZpbGU7Ci0gYC9zd2FnZ2VyLXVpLmh0bWxgIC0tIHByb3ZpZGVzIFJhcGlEb2MgVUkgZm9yIHRoZSBPcGVuIEFwaSBTcGVjaWZpY2F0aW9uLgoKPT0gRGVtbyBwcm9qZWN0CgpEZW1vIHByb2plY3QgaXMgbG9jYXRlZCBpbiBtb2R1bGUgYHJhcGlkb2Mtc3ByaW5nLWJvb3QtZGVtb2AuIFN0YXJ0IGFwcGxpY2F0aW9uIGFuZCBoaXQgYGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpLmh0bWxgIGluIGJyb3dzZXIuCgoK readmeEtag: '"4c1217a2a12897a74f822b0211303c5095ef3ff6"' readmeLastModified: Tue, 07 Jun 2022 14:18:00 GMT repositoryId: 499638084 description: Rapidoc Open Api UI for Spring Boot created: '2022-06-03T20:08:47Z' updated: '2023-04-21T05:42:15Z' language: Java archived: false stars: 7 watchers: 5 forks: 1 owner: PlaytikaOSS logo: https://avatars.githubusercontent.com/u/397334?v=4 license: MIT repoEtag: '"8fb57a2d34b21895654cd8569761555145df04bfb62816f0ce90c5b8e08a3ca6"' repoLastModified: Fri, 21 Apr 2023 05:42:15 GMT category: SDK foundInMaster: true oldLocations: - https://github.com/playtika/rapidoc-spring-boot - source: openapi3 tags repository: https://github.com/birddevelper/notebookmanager v3: true repositoryMetadata: base64Readme: >- CiMjIE5vdGVCb29rIE1hbmFnZXIKIyMjIFNwcmluZyBib290IFJlc3QgQXBpK3N3YWdnZXIrTXlTUUwrRG9ja2VyIHR1dG9yaWFsCgpOb3RlQm9vayBNYW5hZ2VyIGlzIGEgc2FtcGxlIHRpbnkgYXBwbGljYXRpb24gdGhhdCBwZXJmb3JtcyBDUlVEIGpvYnMgb24gTm90ZWJvb2sgRW50aXR5IHZpYSBSZXN0ZnVsIFdlYnNlcnZpY2VzLgoKSXQgZXhwb3NlcyA1IGVuZHBvaW50IGFzIGZvbGxvd2luZyBsaXN0IDoKCi0gR0VUIC9hcGkvbm90ZWJvb2tzIChnZXQgYSBsaXN0IG9mIG5vdGVib29rKQotIFBPU1QgL2FwaS9ub3RlYm9va3MgKGNyZWF0ZSBhIG5vdGVib29rKQotIEdFVCAvYXBpL25vdGVib29rcy8xIChnZXQgb25lIG5vdGVib29rIGZyb20gdGhlIGxpc3QpCi0gUEFUQ0ggL2FwaS9ub3RlYm9va3MvMSAodXBkYXRlIHRoZSBwcmljZSBvZiBhIHNpbmdsZSBub3RlYm9vaykKLSBERUxFVEUgL2FwaS9ub3RlYm9va3MvMSAoZGVsZXRlIGEgc2luZ2xlIG5vdGVib29rKQoKIyMgSG93IHRvIHNldHVwCkl0IGlzIGEgZG9ja2VyIGJhc2VkIGFwcGxpY2F0aW9uLiBSdW5uaW5nIGJlbG93IGNvbW1hbmQgaW4gcHJvamVjdCdzIGRpcmVjdG9yeSBidWlsZHMgMiBydW5uaW5nIGNvbnRhaW5lcnMsIE5vdGVib29rU2VydmljZSBhbmQgTXlzcWwgY29udGFpbmVycyB3aXRoIGluaXRpYWwgcmVjb3JkczoKCmBgYGJhc2gKZG9ja2VyLWNvbXBvc2UgdXAKYGBgCgoKIyMgSG93IHRvIHVzZQpUaGlzIGFwcGxpY2F0aW9uIGlzIFJlc3RmdWwgYW5kIGl0IGZvbGxvd3MgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGluIEFQSSBkb2N1bWVudGF0aW9uLiBUaGFua3MgdG8gU3dhZ2dlclVJLCB5b3UgY2FuIHNlZSBlbmRwb2ludHMgZG9jdW1lbnRhdGlvbiBpbiBhIGdyYXBoaWNhbCB1c2VyIGludGVyZmFjZSBhbmQgdHJ5IHRoZWlyIGZ1bmN0aW9uYWxpdHkgYW5kIHNlZSB0aGUgcmVzcG9uc2UuIEFmdGVyIHJ1bm5pbmcgdGhlIGNvbnRhaW5lcnMsIHlvdSBjYW4gYWNjZXNzIHRoZSBhcHBsaWNhdGlvbiBsaW5rcyBhcyA6CgotIEFwcGxpY2F0aW9uIFJlc3Qgc2VydmljZXMgOiBodHRwOi8vbG9jYWxob3N0OjgwOTAvYXBpL25vdGVib29rcwotIFN3YWdnZXJVSSA6IGh0dHA6Ly9sb2NhbGhvc3Q6ODA5MC9zd2FnZ2VyLXVpLmh0bWwKCgoKIyMgVGVzdAoKVGVzdHMgYXJlIHdyaXR0ZW4gd2l0aCBIMiBpbi1tZW1vcnkgZGF0YWJhc2Ugd2l0aCBpbml0aWFsIHJlY29yZHMuCgoKIyMgTGluawoKW1RoaXMgcmVwb3NpdG9yeSBiZWxvbmdzIHRvIFNwcmluZyBib290IFJlc3QgQXBpK3N3YWdnZXIrTXlTUUwrRG9ja2VyIHR1dG9yaWFsXShodHRwczovL21zaGFlcmkuY29tL2Jsb2cvcmVzdGZ1bC1zcHJpbmctYm9vdC1hcHBsaWNhdGlvbi1zd2FnZ2VyLW15c3FsLWRvY2tlci1hLXJlYWwtd29ybGQtZXhhbXBsZS8pCg== readmeEtag: '"fa55b0751218cb016b358f16a0c5e25c0d43ca27"' readmeLastModified: Tue, 14 Feb 2023 19:20:00 GMT repositoryId: 455494383 description: >- A Real World Example for Open API Restful Spring boot Application + Swagger + MySQL + Docker created: '2022-02-04T09:48:40Z' updated: '2025-10-02T05:03:35Z' language: Java archived: false stars: 8 watchers: 1 forks: 9 owner: birddevelper logo: https://avatars.githubusercontent.com/u/20352967?v=4 repoEtag: '"b33885ee467fa2dd48c2ae2e258e4ee158cc7a0b86c9974ed106eafff64f2365"' repoLastModified: Thu, 02 Oct 2025 05:03:35 GMT foundInMaster: true category: Server Implementations id: 0ba2ea8447cfc632560b3b64c4551d19 - source: openapi3 tags repository: https://github.com/soundsmagic/ableton-react-remote v3: true id: b2e1eed86888f15f04ddcd02598181fb repositoryMetadata: base64Readme: >- QkFDS0dST1VORAoKQSBzY2hvb2wgcHJvamVjdCBmb3IgbXkgdGhlc2lzIGFyb3VuZCB0aGUgcG9zc2liaWxpdGllcy9kaWZmaWN1bHRpZXMgb2YgaW50ZWdyYXRpbmcgdXNlci1kZXZlbG9wZWQgc29sdXRpb25zIGludG8gREFXcyAoRGlnaXRhbCBBdWRpbyBXb3Jrc3RhdGlvbnMpLgpJIHN0dWRpZWQgaW50byBlbGV2ZW4gb2YgdGhlIG1vc3QgY29tbW9uIERBV3MsIHRlbiBvZiB0aGVtIG9uIHRoZSBzdXJmYWNlIHRocm91Z2ggcmVndWxhciByZXNlYXJjaCBhbmQgb25lIG9mIHRoZW0gLSBBYmxldG9uIExpdmUgLSBtb3JlIHRob3JvdWdobHkgCmJ5IGRldmVsb3BpbmcgdGhpcyBhcHBsaWNhdGlvbi4gKElmIGFueW9uZSBpcyBpbnRlcmVzdGVkIGluIG15IHRoZXNpcyByZXBvcnQgSSBjYW4gc2VuZCBpdCAtIGFsdGhvdWdoIGl0J3MgaW4gc3dlZGlzaC4uLikKCkkndmUgdXNlZCBBYmxldG9uIGZvciBsaXZlIGtleWJvYXJkcyB0aGUgcGFzdCB5ZWFycywgYW5kIG9ubHkgZ290IHNvIGZhciB3aXRoIENseXBoWCBQcm8gKyBUb3VjaE9TQy4gVGhpcyBpZGVhIG9mIGNyZWF0aW5nIGEgZnVsbHkgY3VzdG9taXplZCBmcm9udCBlbmQgZm9yIG15Cm5lZWRzIGhhcyBiZWVuIHdpdGggbXkgZm9yIGEgZmV3IHllYXJzLCBzbyBJIGRlY2lkZWQgdG8gdGFrZSB0aGlzIG9wcG9ydHVuaXR5IHRvIG1ha2UgYSBmaXJzdCBzaW1wbGUgdmVyc2lvbiBvZiBpdC4KCkVYVEVORElORyBBQkxFVE9OIExJVkUKClRvIHVzZSBhbmQgdW5kZXJzdGFuZCB0aGlzIHByb2plY3QsIHlvdSBuZWVkIHRvIGJlIHNvbWV3aGF0IGZhbWlsaWFyIHdpdGggQWJsZXRvbiBMaXZlIGFuZCBpdHMgTUlESSBSZW1vdGUgU2NyaXB0cyBmZWF0dXJlcy4gU2luY2UgZGV2ZWxvcG1lbnQgb2YgeW91ciBvd24gc2NyaXB0cwppc24ndCBvZmZpY2lhbGx5IHN1cHBvcnRlZCBieSBBYmxldG9uIChidXQgbm90IHByb2hpYml0ZWQpLCBpdCBjYW4gYmUgc29tZXdoYXQgdHJpY2t5IHRvIGZpbmQgaW5mb3JtYXRpb24gKGJ1dCBub3QgaGFyZGVyIHRoYW4gc3BlbmRpbmcgc29tZSBleHRyYSB0aW1lIGF0IHlvdXIKZmF2b3VyaXRlIHNlYXJjaCBlbmdpbmUpLiBUaGUgc2hvcnQgdmVyc2lvbiBpcyB0byBjcmVhdGUgYW4gYXJiaXRyYXJpbHkgbmFtZWQgZm9sZGVyIGluIEFibGV0b24ncyBNSURJIFJlbW90ZSBTY3JpcHRzIGRpcmVjdG9yeSwgdGhlbiBwbGFjZSB0aGUgY29udGVudHMgb2YgdGhlIAoiY3lvdy1iYWNrLWVuZCIgZm9sZGVyIGludG8gaXQsIGFsb25nIHdpdGggdGhlICJidWlsZCIgZm9sZGVyIHRoYXQgaXMgdGhlIHJlc3VsdCBvZiBidWlsZGluZyB0aGUgUmVhY3QgZnJvbnQgZW5kLiBXaGVuIHN0YXJ0aW5nIEFibGV0b24sIHRoZSBjcmVhdGVkIGZvbGRlciBzaG91bGQKYXBwZWFyIGluIHRoZSBDb250cm9sIFN1cmZhY2UgbGlzdCBvZiB0aGUgTUlESSB0YWIgaW4gUHJlZmVyZW5jZXMuCgpDSE9JQ0VTIC8gRlVUVVJFCgpPbmUgb2YgbXkgZ29hbHMgd2l0aCB0aGUgcHJvamVjdCB3YXMgdG8gbGVhcm4gUHl0aG9uIGJldHRlciwgc28gSSBjaG9zZSB0byBzdHVmZiBhbGwgYmFjayBlbmQgZnVuY3Rpb25hbGl0eSBpbnRvIHRoZSBNSURJIFJlbW90ZSBTY3JpcHQuIFRoaXMgZ29hbCBhbHNvIGhlbHBlZCBtZSB0bwpub3Qgc2h5IGF3YXkgZnJvbSB3cml0aW5nIGFsbCB0aGUgV1NHSSBzZXJ2ZXIvYXBwbGljYXRpb24gY29kZSBmcm9tIHNjcmF0Y2gsIHdoZW4gSSBmb3VuZCB0aGF0IHRoZSBQeXRob24gaW50ZXJwcmV0ZXIgdGhhdCdzIGJ1aWx0IGludG8gTGl2ZSBkb2Vzbid0IGNvbnRhaW4gdGhlIGZ1bGwgClB5dGhvbiBzdGFuZGFyZCBtb2R1bGUgbGlicmFyeSAoYW5kIGltcG9ydGluZyBtb2R1bGVzIGF0IHJ1bnRpbWUgcHJvdmVkIHRvIGJlIGEgZ3JlYXRlciBjaGFsbGVuZ2UgdGhhbiBJIGNvdWxkIGhhbmRsZSBhdCB0aGUgdGltZSksIHRoZXJlYnkgcHJldmVudGluZyBtZSBmcm9tIHVzaW5nCnJlYWR5IG1hZGUgZnJhbWV3b3Jrcy4gVGhpcyBibG9nIHNlcmllcyBoZWxwZWQgbWUgdHJlbWVuZG91c2x5IHRocm91Z2ggd3JpdGluZyB0aGUgV1NHSSBzdHVmZjogaHR0cHM6Ly9tbGV1ZS5jb20vcG9zdHMvbGV0cy1idWlsZC1vdXItb3duLWZhc3RhcGkvCgpJIHdhbnRlZCB0byB1c2UgUmVhY3Qgd2l0aCBSZWR1eCBhbmQgUlRLIFF1ZXJ5IGZvciB0aGUgZnJvbnQgZW5kLCBzaW5jZSBJIHN0YXJ0ZWQgbGVhcm5pbmcgdGhhdCBzaG9ydGx5IGJlZm9yZSB0aGlzIHByb2plY3QgYW5kIHdhbnRlZCB0aGUgcHJhY3RpY2UuCgpNeSB2aXNpb24gb2YgYSBjb21wbGV0ZSBhbmQgc3RhYmxlIHJlbW90ZSBhcHBsaWNhdGlvbiBmb3IgQWJsZXRvbiBMaXZlLCBmb2N1c2VkIG9uIGtleWJvYXJkcyBhbmQgYmFja2luZyB0cmFja3MgZm9yIGxpdmUgdXNlIGFuZCB0YWlsb3JlZCBmb3IgbXkgdXNlIGNhc2VzLCBpcyBzdGlsbApzb21ldGhpbmcgSSB3YW50IHRvIGFjaGlldmUuIEkga25ldyBlYXJseSBvbiB0aGF0IEkgd2FudGVkIHRoZSBmcm9udCBlbmQgdG8gcmVmbGVjdCBjaGFuZ2VzIGRvbmUgb24gbXkgbGFwdG9wLCB3aGljaCBJIGludGVuZGVkIHRvIGltcGxlbWVudCB3aXRoIFdlYnNvY2tldHMgaW4gYQpjb21pbmcgdmVyc2lvbi4gQWZ0ZXIgcmVzZWFyY2hpbmcgdGhlIHN1YmplY3QsIEkgcmVhbGl6ZWQgdGhhdCB0aGlzIHdhcyBzb21ldGhpbmcgSSBkaWRuJ3Qgd2FudCB0byBsZWFybiB0byB3cml0ZSBmcm9tIHNjcmF0Y2ggaW4gUHl0aG9uIGF0IHRoZSBtb21lbnQuIFRoZXJlZm9yZSwgSSdtCmNoYW5naW5nIHRoZSBhcmNoaXRlY3R1cmUgb2YgdGhlIGFwcGxpY2F0aW9uIHRvIHVzZSB0aGUgb3RoZXIgcmVzb3VyY2UgdGhhdCBoZWxwZWQgbWUgYSBncmVhdCBkZWFsIGluIHRoaXMgcHJvamVjdCAtIEFibGV0b25KUzogaHR0cHM6Ly9naXRodWIuY29tL2xlb2xhYnMvYWJsZXRvbi1qcwoKVGhlIHdlYiBzZXJ2ZXIgd2lsbCBnb2luZyBmb3J3YXJkIGJlIGEgc2VwYXJhdGUgTm9kZSBhcHBsaWNhdGlvbiwgd2hpY2ggY2FuIHRhbGsgV2Vic29ja2V0cyB3aXRoIHRoZSBmcm9udCBlbmQgYW5kIGxldCBBYmxldG9uSlMgdGFrZSBjYXJlIG9mIHRoZSBjb21tdW5pY2F0aW9uIHdpdGgKQWJsZXRvbiBMaXZlIGFuZCB0aGUgTUlESSBSZW1vdGUgU2NyaXB0cy4gU2luY2UgdGhpcyB3aWxsIGNoYW5nZSB0aGUgYXBwbGljYXRpb24gcXVpdGUgYSBiaXQsIEkgZGVjaWRlZCB0byBsZWF2ZSB0aGlzIHJlcG9zaXRvcnkgYXMgaXQgaXMgYW5kIGNvbnRpbnVlIG15IGRldmVsb3BtZW50CmluIGEgbmV3IHJlcG8uIFRoYXQgd2F5IHRoaXMgcmVwbyBjYW4gc3RhbmQgYXMgYSBmdW5jdGlvbmluZyBleGFtcGxlIGZvciBvdGhlcnMgd2FudGluZyB0byBnbyB0aGlzIHdheSBvZiBhIG5vLWZyYW1ld29yayBXU0dJIHNlcnZlci9hcHBsaWNhdGlvbi4K readmeEtag: '"4287b0216ce35638bffb2ab0801c8e4e7f1ae921"' readmeLastModified: Sat, 30 Jul 2022 15:23:46 GMT repositoryId: 454422973 description: An application for remote controlling Ableton Live from a React front end created: '2022-02-01T14:38:30Z' updated: '2023-11-06T20:47:09Z' language: TypeScript archived: false stars: 7 watchers: 0 forks: 1 owner: soundsmagic logo: https://avatars.githubusercontent.com/u/76114226?v=4 repoEtag: '"19241846e40f919830df8713db6f095b385e43a953f34fde5f7493c543f3ff1c"' repoLastModified: Mon, 06 Nov 2023 20:47:09 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/dotjoshrc/protoc-gen-openapi v3: true id: bc5611151c6d1be7144cc24f4ad9e974 repositoryMetadata: base64Readme: >- # protoc-gen-openapi

![GitHub release (latest by date)](https://img.shields.io/github/v/release/dotjoshrc/protoc-gen-openapi?style=flat-square)
[![lint status](https://img.shields.io/github/actions/workflow/status/dotjoshrc/protoc-gen-openapi/run-lint.yaml?style=flat-square&label=lint)](https://github.com/dotjoshrc/protoc-gen-openapi/actions/workflows/run-lint.yaml)
[![go report card](https://goreportcard.com/badge/github.com/dotjoshrc/protoc-gen-openapi?style=flat-square)](https://goreportcard.com/report/github.com/dotjoshrc/protoc-gen-openapi)

**Yes**, this is _another_ protoc generator for OpenAPI. I created this for a
couple
reasons...

- I wanted to learn protoc generation with a real-world problem.
- The official google one sticks to gRPC and envoy standards. My team and I use
  Twirp and other REST frameworks. _Sometimes you just want to define models and
  an API for docs._
- Others try to do too much per the spec and fail to do the most common things
  well.

_**DISCLAIMER: This will be a limited subset of the OAPI3 specification. Not
everything will make it in here. Why? Read the last bullet point above. :)**_

_Some patterns were heavily inspired
by [gnostic](https://github.com/google/gnostic)._

## Installation

```terminal
go install github.com/dotjoshrc/protoc-gen-openapi@latest
```

## Options

| Option             | Description                                                                       | Default          |
|--------------------|-----------------------------------------------------------------------------------|------------------|
| `version`          | The version of the API.                                                           | 0.0.1            |
| `title`            | The title of the API.                                                             |                  |
| `description`      | A description of the API.                                                         |                  |
| `include`          | A list of proto package names to include only. `ignore` is ran after this         |                  |
| `ignore`           | A list of proto package names to ignore delimited by pipes.                       |                  |
| `default_response` | The default response to be used.<sup>1</sup>                                      |                  |
| `content_type`     | The content type to be associated with all operations.<sup>1</sup>                | application/json |
| `json_names`       | Use the JSON names that Protobuf provides. Otherwise, proto field names are used. | false            |
| `json_out`         | Create a JSON file instead of the default YAML.                                   | false            |
| `host`             | The host to be used for all operations.<sup>1</sup>                               |                  |
| `filename`         | Specify the filename to output.                                                   | openapi.yaml     |

<sup>1</sup> _Can be overridden on a file, service, or method._

## Build Examples

Below are some basic examples on how to use this generator.

### Protoc

```bash
protoc \
  --openapi_out=. \
  --openapi_opt=version=1.0.0 \
  --openapi_opt=title="My Awesome API" \
  api/some_service.proto
```

### Buf

**buf.yaml**

```yaml
# ... other things
deps:
  - buf.build/dotjoshrc/protoc-gen-openapi
```

**buf.gen.yaml**

```yaml
plugins:
  - name: go
    out: api
    opt:
      - paths=source_relative
  - name: openapi
    strategy: all # important so all files are ran in the same generation.
    out: api
    opt:
      - title=My Awesome API
      - description=Look how awesome my API is!
      - ignore=module.v1|module.v2
      - default_response=SomeErrorObject
```

## Basic Usage Example

```protobuf
syntax = "proto3";

import "oapi/v1/field.proto";
import "oapi/v1/file.proto";
import "oapi/v1/method.proto";
import "oapi/v1/service.proto";

option (oapi.v1.file) = {
  servers {
    url: "myawesomeapi.com"
  }

  security_schemes: {
    name: "bearer_auth"
    scheme: {
      type: "http"
      scheme: "bearer"
      bearer_format: "JWT"
    }
  }

  // Since this is at the file level, it's applied to all.
  security: {
    name: "bearer_auth"
    scopes: ["read:resource"]
  }
};

service MyService {
  option (oapi.v1.service) = {
    prefix: "/v1"
    x_display_name: "My Service"
  };

  rpc CreateSomething (CreateSomethingRequest) returns (CreateSomethingResponse) {
    option (oapi.v1.method) = {
      post: "create-something"
      summary: "Create Something"
      status: 201
    };
  }
}

message CreateSomethingRequest {
  // The name of something.
  // Example: something-awesome
  string name = 1 [(oapi.v1.required) = true];
}

message CreateSomethingResponse {
  // The ID of something.
  string id = 1;
  string name = 2;
}
```

## Features

> **Note**
>
> Defining features is a work in progress. I aim to explain all that's possible
> the best I can.

<details>
<summary><h3>Server definitions</h3></summary>

You can define servers at the file, service, or method level. Each one overrides
the previous. This allows for more advanced composition.

**Example:**

```protobuf
syntax = "proto3";

import "google/protobuf/empty.proto";
import "oapi/v1/file.proto";
import "oapi/v1/method.proto";
import "oapi/v1/service.proto";


option (oapi.v1.file) = {
  servers {
    url: "myawesomeapi.com" // file-defined for all services and methods
  }
};

service MyService {
  option (oapi.v1.service) = {
    servers {
      url: "myawesomeapi2.com" // overrides file-defined
    }
  };

  rpc CreateSomething (google.protobuf.Empty) returns (google.protobuf.Empty) {
    option (oapi.v1.method) = {
      servers {
        url: "myaweseomeapi3.com" // overrides service-defined
      }
    };
  }
}
```

</details>

<details>
<summary><h3>Service Prefixes</h3></summary>

Each service can have a path prefix set for all methods to inherit. This is
useful when versioning your API or if you have a parameter that is defined for
each method route.

_**You can override the entire path in the method by starting the path out with
a `/`.**_

**Example:**

```protobuf
syntax = "proto3";

import "google/protobuf/empty.proto";
import "oapi/v1/file.proto";
import "oapi/v1/method.proto";
import "oapi/v1/service.proto";

option (oapi.v1.file) = {
  servers {
    url: "myawesomeapi.com"
  }
};

service MyService {
  option (oapi.v1.service) = {
    prefix: "/v1"
  };

  rpc CreateSomething (google.protobuf.Empty) returns (google.protobuf.Empty) {
    option (oapi.v1.method) = {
      post: "create" // becomes /v1/create
    };
  }

  rpc OverrideSomething (google.protobuf.Empty) returns (google.protobuf.Empty) {
    option (oapi.v1.method) = {
      get: "/create" // becomes /create
    };
  }
}
```

</details>

## Features In Progress

- [Enum](https://json-schema.org/understanding-json-schema/reference/generic.html#enumerated-values)
  requirements on fields

## Contributing

Coming... Right now I prefer that it's just me until I get a solid hold on
generator patterns and the package is stable. I'm fully open to any suggestions
though!
 readmeEtag: '"dd13599cc5eddf07f6852e83a8c2f2ce0c9cf5c2"' readmeLastModified: Mon, 02 Feb 2026 19:09:19 GMT repositoryId: 580132646 description: OpenAPI generation from Protobuf created: '2022-12-19T19:57:16Z' updated: '2026-02-02T19:09:23Z' language: Go archived: false stars: 9 watchers: 1 forks: 4 owner: dotjoshrc logo: https://avatars.githubusercontent.com/u/8335467?v=4 repoEtag: '"898c6a95d12055bc35148186aa21cc9eaa35d89ccd1d97ebccf1b5db87b37227"' repoLastModified: Mon, 02 Feb 2026 19:09:23 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/technicallyjosh/protoc-gen-openapi - source: openapi3 tags repository: https://github.com/rinold/steamer v3: true repositoryMetadata: base64Readme: >- IyBTdGVhbWVyCgpDb21tYW5kIGxpbmUgdG9vbCBmb3IgW1ZhcG9yXShodHRwczovL3ZhcG9yLmNvZGVzKSBzdHViICh0ZW1wbGF0ZSkgc2VydmVyIGdlbmVyYXRvciBmcm9tIHRoZSBbT3BlbkFQSV0oaHR0cHM6Ly9vcGVuYXBpcy5vcmcvKSAoU3dhZ2dlcikgc3BlY2lmaWNhdGlvbiB3cml0dGVuIGluIFN3aWZ0LgoKIyMjIFRhcmdldGVkIHN1cHBvcnRlZCB2ZXJzaW9uczoKLSBWYXBvciAzCi0gT3BlbkFQSSAzLjAuKiAoW0N1cnJlbnQgbGF0ZXN0IC0gMy4wLjEgU3BlY10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjEubWQpKQoKIyMjIENvbmNlcHQKVGhlIE9wZW5BUEkgZGVmaW5pdGlvbiBvZiBBUEkgZGVzaWduIGlzIGEgZ3JlYXQgYmFzZSB0aGF0IGNvdWxkIGFsbG93IHRvIGNyZWF0ZSBhdXRvZ2VuZXJhdGVkIGNvZGUgdGhhdDoKLSBXb3VsZCAqKmJlIHdlbGwtZG9jdW1lbnRlZCoqIC0gYXMgdGhlIE9wZW5BUEkgZGVmaW5pdGlvbiBieSBpdHNlbGYgaWRlYWxseSBzaG91bGQgYmUgYSAnZG9jdW1lbnRhdGlvbicgZm9yIGNsaWVudCBkZXZlbG9wZXJzIGl0IGNvdWxkIGJlIGF1dG9tYXRpY2FsbHkgYnJvdWdodCB0byB0aGUgU3dpZnQgY29kZSBkZWZpbml0aW9ucy4KLSBXb3VsZCAqKnNoYXJlIHRoZSBnZW5lcmF0ZWQgTW9kZWxzIGFuZCBSZXF1ZXN0L1Jlc3BvbnNlIGNvZGUqKiAgYXMgbXVjaCBhcyBwb3NzaWJsZSAqKmJldHdlZW4qKiB0aGUgUkVTVGZ1bCAqKnNlcnZlciBhbmQgY2xpZW50KiogYXBwbGljYXRpb24gd2hpY2ggYXV0b21hdGljYWxseSB3b3VsZCBlbnN1cmUgc2VyaWFsaXNhdGlvbiBvZiBkYXRhIHBhc3NlZCBiZXR3ZWVuIHNlcnZlciBhbmQgY2xpZW50IGlzIFNBTUU6CiAgLSBObyBtb3JlIHR5cG9zLWVycm9yIGRlYnVnZ2luZyBvZiBmYWlsZWQgcmVxdWVzdHMgKGUuZy4gd2hlbiBvbmUgc2lkZSBzZW5kcyAnbGljZW5jZScgaW5zdGVhZCBvZiAnbGljZW5zZScpCiAgLSBObyBtb3JlIHRyaWNreSBlcnJvcnMgb2YgZGF0YSB0eXBlcyBjb252ZXJzaW9ucyAobGlrZSBJbnQ2NCB2cyBJbnQzMiwgdGhlIERhdGUgZm9ybWF0cywgZXRjLikKICAtIFNoYXJlZCBtb2RlbHMgd291bGQgYmUgYSBTSU5HTEUgcGxhY2UgeW91IG5lZWQgdG8gY2hhbmdlIGlmIHJlcXVpcmVkCi0gV291bGQgKipiZSBleHRlbnNpYmxlKiosIGlkZWFsbHkgYWxsb3dpbmcgaW5jcmVtZW50YWwgdXBkYXRlcyBvZiBjb2RlIGZyb20gdGhlIHVwZGF0ZWQgT3BlbkFQSSBkZWZpbml0aW9uIHdpdGggbWluaW1hbCBpbXBhY3Qgb24gbWFudWFsbHkgYWRkZWQgY29kZS4KLSBXb3VsZCAqKmJlIHNpbXBsZSBhbmQgU3dpZnR5KiosIGNyZWF0ZWQgc28gdGhhdCB0aG91Z2h0cyBsaWtlICdpdCB3b3VsZCBiZSBlYXNpZXIgdG8gZGVsZXRlIGFuZCByZXdyaXRlIGl0JyBzaG91bGQgbmV2ZXIgYXBwZWFyIGluIGRldmVsb3BlciBoZWFkIHdoZW4gaGUgc2VlcyBpdC4KCiMjIyBHb2FscwotIEVuY291cmFnZSB0aGUgW0FQSSBkZXNpZ24gZmlyc3RdKGh0dHBzOi8vc3dhZ2dlcmh1Yi5jb20vYmxvZy9hcGktZGVzaWduL2Rlc2lnbi1maXJzdC1vci1jb2RlLWZpcnN0LWFwaS1kZXZlbG9wbWVudC8pIGlmIGl0IHN1aXRzIHlvdXIgbmVlZHMhCi0gTWFrZSBpbXBsZW1lbnRhdGlvbiBhbmQgcHJvdG90eXBpbmcgb2YgUkVTVGZ1bCBzZXJ2ZXJzIHVzaW5nIFZhcG9yIGV2ZW50IG1vcmUgZWFzaWVyIGJ5IG1pbmltaXNpbmcgYm9pbGVycGxhdGUgY29kaW5nCi0gTWVldCBpbmNyZWRpYmxlIGFuZCBhbWF6aW5nIFN3aWZ0IHByb2Zlc3Npb25hbHMgd2hvIHdvdWxkIGxpa2UgdG8gam9pbiBhbmQgY29udHJpYnV0ZSB0byB0aGlzIHByb2plY3QhIDopCgojIyMgRmVhdHVyZXMgLSBUQkQsIHN0YXJ0aW5nIGxpc3QgdG8gdGhpbmsgb2YuLi4KLSBHZW5lcmF0aW9uIG9mIHN0dWJiZWQgQVBJIHNlcnZlciBmcm9tIE9wZW5BUEkgZG9jdW1lbnQ6CiAgLSBNb2RlbHMKICAtIENvbnRyb2xsZXJzCiAgLSBSb3V0aW5nCiAgLSBldmVyeXRoaW5nIGVsc2UgbmVlZGVkIHRvIHJ1bgotIFVzZSBTZWN1cml0eSBTY2hlbWUgZGVmaW5pdGlvbnMgZm9yIGF1dGhvcml6YXRpb24gYW5kIGF1dGhlbnRpZmljYXRpb24gc3VwcG9ydAotIE1vY2sgcmVzcG9uc2VzIHdpdGggZGF0YSBleGFtcGxlcyBpZiBwcm92aWRlZAotIENvbW1lbnRzIC8gZG9jdW1lbnRhdGlvbiBnZW5lcmF0aW9uCi0gLi4uCgojIyMgQ29udHJpYnV0aW9uCklzIHdlbGNvbWUhIEpvaW4gYW5kIGJlIHRoZSBmaXJzdCBvbmUgY29tbWl0aW5nIHJlYWwgY29kZSEgIApZb3VyIGNhbiBjb250YWN0IG1lIHZpYSBlbWFpbDogW21paGFpbC5jaHVyYmFub3ZAZ21haWwuY29tXShtYWlsdG86bWloYWlsLmNodXJiYW5vdkBnbWFpbC5jb20pCgojIyMgRGVwZW5kZW5jaWVzIChQb3RlbnRpYWwpCi0gW1N3YWdnZXJQYXJzZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9BdHRpbGFUaGVGdW4vU3dhZ2dlclBhcnNlcikgLSBTd2lmdCBsaWJyYXJ5IGZvciBwYXJzaW5nIE9wZW5BUEkgMi4wIChmLmsuYS4gU3dhZ2dlcikgZG9jdW1lbnRzIGludG8gbmF0aXZlIHN0cnVjdHVyZXMgKE9wZW5BUEkgMy4wIHN1cHBvcnQgaXMgImFscGhhIi1yZWFkeSwgY3VycmVudGx5IG9uIFtmb3JrXShodHRwczovL2dpdGh1Yi5jb20vcmlub2xkL1N3YWdnZXJQYXJzZXIvdHJlZS9vcGVuYXBpLTMpKS4KLSBbP10gW1lhbXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9qcHNpbS9ZYW1zKSAtIEEgU3dlZXQgYW5kIFN3aWZ0eSBZQU1MIHBhcnNlci4KLSBbP10gW1N0ZW5jaWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9reWxlZi9TdGVuY2lsKSAvIFtMZWFmXShodHRwczovL2dpdGh1Yi5jb20vdmFwb3IvbGVhZikgb3IgLi4uIC0gbmVlZCB0byBjaG9vc2Ugc29tZXRoaW5nIGZvciB0ZW1wbGF0aW5nLgo= readmeEtag: '"5e38f17da8047fb317904041a4f70834be32d5ac"' readmeLastModified: Fri, 29 Jun 2018 15:23:04 GMT repositoryId: 115447708 description: Vapor simple CRUD-server generator created: '2017-12-26T19:01:47Z' updated: '2022-03-22T06:55:36Z' language: Swift archived: false stars: 7 watchers: 2 forks: 0 owner: rinold logo: https://avatars.githubusercontent.com/u/2253795?v=4 license: MIT repoEtag: '"4085534028adf1c89a842ee89916d31abb65c79c471c9bbead7f986fe29c56fa"' repoLastModified: Tue, 22 Mar 2022 06:55:36 GMT foundInMaster: true category: - Server - Parsers id: 8ad741c3a7108e9b58b9bda5e3524ad2 - source: openapi3 tags repository: https://github.com/hiromaily/go-goa v3: true id: 938d96a1d5b7e8bfa831a0217cf4c914 repositoryMetadata: base64Readme: >- IyBnby1nb2EKClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS9oaXJvbWFpbHkvZ28tZ29hKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL2hpcm9tYWlseS9nby1nb2EpClshW2NvZGViZWF0IGJhZGdlXShodHRwczovL2NvZGViZWF0LmNvL2JhZGdlcy9mMmVlMmVkMC01NTg4LTQ2ZjktYTQ3ZS1kNTA2MzNhMDY3MzkpXShodHRwczovL2NvZGViZWF0LmNvL3Byb2plY3RzL2dpdGh1Yi1jb20taGlyb21haWx5LWdvLWdvYS1tYXN0ZXIpClshW0NvZGFjeSBCYWRnZV0oaHR0cHM6Ly9hcGkuY29kYWN5LmNvbS9wcm9qZWN0L2JhZGdlL0dyYWRlL2YyMDdjYTU3ZTQ4ZTQ1NjM4OTM0MWZjNDFiYjA2OTUxKV0oaHR0cHM6Ly93d3cuY29kYWN5LmNvbS9hcHAvaGlyb21haWx5Mi9nby1nb2E/dXRtX3NvdXJjZT1naXRodWIuY29tJmFtcDt1dG1fbWVkaXVtPXJlZmVycmFsJmFtcDt1dG1fY29udGVudD1oaXJvbWFpbHkvZ28tZ29hJmFtcDt1dG1fY2FtcGFpZ249QmFkZ2VfR3JhZGUpClshW01JVCBMaWNlbnNlXShodHRwOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGljZW5zZS1NSVQtYmx1ZS5zdmc/c3R5bGU9ZmxhdCldKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9oaXJvbWFpbHkvZ28tZ29hL21hc3Rlci9MSUNFTlNFKQoKYGdvLWdvYWAgaXMgZXhhbXBsZSBvZiBbZ29hXShodHRwczovL2dpdGh1Yi5jb20vZ29hZGVzaWduL2dvYSkgZnJhbWV3b3JrIGZvciB2MyB3aXRoIEpXVCBpbXBsZW1lbnRhdGlvbi4gIApUaGUgZXhhbXBsZSBzaXRlIGNvbnNpc3RzIG9mIGZpbGUgc2VydmVyIGFuZCBBUEkgc2VydmVyLgoKIVtFeGFtcGxlIFJlc3VtZSBTaXRlXShyZXN1bWUucG5nICJleGFtcGxlIHJlc3VtZSBzaXRlIikKCgojIyBSZWZhY3RvcmluZwotIFsgXSBDbGVhbiB1cCBldmVyeXRoaW5nCi0gWyBdIEludGVncmF0ZSB0byBGaXJlYmFzZSB1c2luZyBOZXh0SlMgU1NHCi0gWyBdIFN3aXRjaCBDSSBmcm9tIFRyYXZpc0NJIHRvIEdpdGh1YiBBY3Rpb24KLSBbIF0gUmVidWlsZCBmcm9udC1lbmQgYnkgbGF0ZXN0IE5leHQuanMoUmVhY3QpIG9yIFN2ZWx0ZQotIFsgXSBSZXBsYWNlIEUyRSB0ZXN0IGJ5IFtIdXJsXShodHRwczovL2h1cmwuZGV2LykKCiMjIERpcmVjdG9yeSBTdHJ1Y3R1cmUKfCBEaXJlY3RvcnkgTmFtZSB8IERlc2NyaXB0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfDotLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwKfCBkb2NzKGFzc2V0cykgICB8IGFzc2V0cyBmb3IgZnJvbnQtZW5kICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBidWlsZCAgICAgICAgICB8IGRvY2tlci9nY3AgY29uZmlndXJhdGlvbiBmaWxlcyAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBjbWQgICAgICAgICAgICB8IG1haW4uZ28gZm9yIGNsaS9zZXJ2ZXIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBjb25maWcgICAgICAgICB8IHRvbWwgY29uZmlndXJhdGlvbiBmaWxlcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBpbnRlcm5hbCAgICAgICB8IGdvYSBkZXNpZ24gZmlsZXMsIGdlbmVyYXRlZCBmaWxlcyBhcyBzZXBhcmF0ZWQgbW9kdWxlIHwKfCBwa2cgICAgICAgICAgICB8IGdvIGZpbGVzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBzY3JpcHRzICAgICAgICB8IHNoZWxsIHNjcmlwdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCB3ZWIgICAgICAgICAgICB8IGRldmVsb3BtZW50YWwgZW52aXJvbm1lbnQgZm9yIGZyb250LWVuZCAgICAgICAgICAgICAgIHwKCiMjIyBwa2cgZGlyZWN0b3J5CnwgRGlyZWN0b3J5IE5hbWUgfCBEZXNjcmlwdGlvbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfDotLS0tLS0tLS0tLS0tLS18Oi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8IGNvbmZpZyAgICAgICAgIHwgY29uZmlnIHBrZyB0byByZWFkIHRvbWwgZmlsZSAgICAgICAgICAgICAgICAgICB8CnwgZW5jcnlwdGlvbiAgICAgfCBlbmNyeXB0aW9uIGZvciBoYXNoICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBnb2EgICAgICAgICAgICB8IGdvYSBsb2dpYyBiYXNlZCBvbiBnZW5lcmF0ZWQgZmlsZSBieSBgZ29hIGdlbmAgfAp8IGp3dHMgICAgICAgICAgIHwgand0ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgbG9nZ2VyICAgICAgICAgfCBsb2dnZXIgdXNpbmcgYHJzL3plcm9sb2dgICAgICAgICAgICAgICAgICAgICAgIHwKfCBtb2RlbCAgICAgICAgICB8IGF1dG8gZ2VuZXJhdGVkIGJ5IGBzcWxib2lsZXJgICAgICAgICAgICAgICAgICAgfAp8IG15c3FsICAgICAgICAgIHwgbXlzcWwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgcmVwb3NpdG9yeSAgICAgfCBkYXRhYmFzZSBvcGVyYXRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKCgojIyBJbnN0YWxsIGdvYQpgYGAKZ28gaW5zdGFsbCBnb2EuZGVzaWduL2dvYS92My9jbWQvZ29hQHYzCmdvIGluc3RhbGwgZ29vZ2xlLmdvbGFuZy5vcmcvcHJvdG9idWYvY21kL3Byb3RvYy1nZW4tZ29AbGF0ZXN0CmdvIGluc3RhbGwgZ29vZ2xlLmdvbGFuZy5vcmcvZ3JwYy9jbWQvcHJvdG9jLWdlbi1nby1ncnBjQGxhdGVzdApgYGAKCgojIyBEZXZlbG9wbWVudCBwcm9jZXNzCi0gYGNkIGludGVybmFsL2dvYS9zZXJ2aWNlL3Jlc3VtZWAKLSBtb2RpZnkgYGRlc2lnbmAgZmlsZXMKLSBnZW5lcmF0ZSBmaWxlcyBieSBgZ29hIGdlbmAKCgojIyBTZXJ2ZXIKIyMjIERhdGFiYXNlCi0gcnVuIE15U1FMIGZpcnN0CmBgYApkb2NrZXIgY29tcG9zZSBteXNxbCAtZApgYGAKCiMjIyBGaWxlIFNlcnZlcgpgYGAKZ28gYnVpbGQgLXYgLW8gJHtHT1BBVEh9L2Jpbi9nb2EtZmlsZS1zZXJ2ZXIgLi9jbWQvZmlsZXNlcnZlci9zZXJ2ZXIvLi4uCmdvYS1maWxlLXNlcnZlcgpgYGAKLSBhY2Nlc3MKICAtIGBodHRwOi8vbG9jYWxob3N0OjgwODAvYAogIC0gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9sb2dpbi5odG1sYCAoaGlyb2tpQGdvYS5jb20gLyBwYXNzd29yZCkKCiMjIyBSZXN1bWUgQVBJIFNlcnZlcgpgYGAKZ28gYnVpbGQgLXYgLW8gJHtHT1BBVEh9L2Jpbi9nb2Etc2VydmVyIC4vY21kL3Jlc3VtZS9zZXJ2ZXIvLi4uCmdvYS1zZXJ2ZXIgLWNvbmYgLi9jb25maWdzL3NldHRpbmdzLnRvbWwKYGBgCgojIyMgUmVzdW1lIEFQSSBMaXN0Ci0gaGVhbHRoCiAgLSBgaHR0cDovL2xvY2FsaG9zdDo4MDkwL2FwaS9oZWFsdGhgCi0gYXV0aAogIC0gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA5MC9hcGkvYXV0aC9sb2dpbmAKLSB0ZWNoCiAgLSBgaHR0cDovL2xvY2FsaG9zdDo4MDkwL2FwaS90ZWNoYAogIC0gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA5MC9hcGkvdGVjaC8xYAotIGNvbXBhbnkKICAtIGBodHRwOi8vbG9jYWxob3N0OjgwOTAvYXBpL2NvbXBhbnlgCiAgLSBgaHR0cDovL2xvY2FsaG9zdDo4MDkwL2FwaS9jb21wYW55LzFgCi0gdXNlcgogIC0gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA5MC9hcGkvdXNlcmAKICAtIGBodHRwOi8vbG9jYWxob3N0OjgwOTAvYXBpL3VzZXIvMWAKLSB1c2VyIHRlY2gKICAtIGBodHRwOi8vbG9jYWxob3N0OjgwOTAvYXBpL3VzZXIvMS9saWtldGVjaGAKICAtIGBodHRwOi8vbG9jYWxob3N0OjgwOTAvYXBpL3VzZXIvMS9kaXNsaWtldGVjaGAKLSB1c2VyIHdvcmsgaGlzdG9yeQogIC0gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA5MC9hcGkvdXNlci8xL3dvcmtoaXN0b3J5YAoKCiMjIFBlcmZvcm1hbmNlClRvIGV2ZWx1YXRlIHBlcmZvcm1hbmNlLCBbaGV5XShodHRwczovL2dpdGh1Yi5jb20vcmFreWxsL2hleSkgaGFzIGJlZW4gdXNlZC4KRXhlY3V0ZSBgTWFrZSBiZW5jaGAKCmBgYAojIE1hcmNoIDEyIDIwMTcKaGV5IC1uIDIwMDAwIC1jIDUwIC1tIEdFVCBodHRwOi8vbG9jYWxob3N0OjgwODAvYXBpL3VzZXIKClN1bW1hcnk6CiAgVG90YWw6CTQuMTA5MiBzZWNzCiAgU2xvd2VzdDoJMC4wOTM5IHNlY3MKICBGYXN0ZXN0OgkwLjAwMDIgc2VjcwogIEF2ZXJhZ2U6CTAuMDEwMSBzZWNzCiAgUmVxdWVzdHMvc2VjOgk0ODY3LjEyMjEKICBUb3RhbCBkYXRhOgk2MDAwMCBieXRlcwogIFNpemUvcmVxdWVzdDoJMyBieXRlcwoKU3RhdHVzIGNvZGUgZGlzdHJpYnV0aW9uOgogIFsyMDBdCTIwMDAwIHJlc3BvbnNlcwpgYGAKCiMjIExpY2VuY2UKW01JVF0oaHR0cHM6Ly9naXRodWIuY29tL2hpcm9tYWlseS9nby1nb2EvYmxvYi9tYXN0ZXIvTElDRU5TRSkKCiMjIEF1dGhvcgpbaGlyb21haWx5XShodHRwczovL2dpdGh1Yi5jb20vaGlyb21haWx5KQo= readmeEtag: '"457b8f913f33a69bf3c03de27ae41af5cdcea25c"' readmeLastModified: Sat, 11 Feb 2023 01:45:22 GMT repositoryId: 84661950 description: goa framework example created: '2017-03-11T16:01:20Z' updated: '2024-10-06T20:33:01Z' language: Go archived: false stars: 7 watchers: 1 forks: 2 owner: hiromaily logo: https://avatars.githubusercontent.com/u/301692?v=4 license: MIT repoEtag: '"6bb2d0627e4455ad11841644cb3b715d0d9c31a8055e28a7e56d0d5c1786951f"' repoLastModified: Sun, 06 Oct 2024 20:33:01 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/vinitshahdeo/openapi-url-resolver v3: true id: cc77996f9fb62a8bfdb6d8216191e7c3 repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KCjxpbWcgc3JjPSIuL2Fzc2V0cy9vcGVuYXBpLXVybC1yZXNvbHZlci1sb2dvLnBuZyIgaGVpZ2h0PSczMCUnIHdpZHRoPSczMCUnLz4KCjxoMT5PcGVuQVBJIFVSTCBSZXNvbHZlcjwvaDE+Cgo8cD4KPGEgaHJlZj0iaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS11cmwtcmVzb2x2ZXIiPjxpbWcgYWx0PSJucG0gb3BlbmFwaS11cmwtcmVzb2x2ZXIiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vcGVuYXBpLXVybC1yZXNvbHZlcj9sYWJlbD1vcGVuYXBpLXVybC1yZXNvbHZlciZsb2dvPW5wbSI+PC9hPgo8YSBocmVmPSJodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLXVybC1yZXNvbHZlciI+PGltZyBhbHQ9Im5wbSBidW5kbGUgc2l6ZSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2J1bmRsZXBob2JpYS9taW4vb3BlbmFwaS11cmwtcmVzb2x2ZXI/bG9nbz1ucG0iPjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS11cmwtcmVzb2x2ZXIiPjxpbWcgYWx0PSJucG0gb3BlbmFwaS11cmwtcmVzb2x2ZXIiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZG0vb3BlbmFwaS11cmwtcmVzb2x2ZXI/bG9nbz1ucG0mY29sb3I9eWVsbG93Z3JlZW4iPjwvYT4KPC9wPgoKPHA+CjxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS92aW5pdHNoYWhkZW8vb3BlbmFwaS11cmwtcmVzb2x2ZXIvYWN0aW9ucy93b3JrZmxvd3Mvbm9kZS5qcy55bWwiPjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vdmluaXRzaGFoZGVvL29wZW5hcGktdXJsLXJlc29sdmVyL2FjdGlvbnMvd29ya2Zsb3dzL25vZGUuanMueW1sL2JhZGdlLnN2ZyIgYWx0PSJHaXRIdWIgQWN0aW9ucyBmb3IgT3BlbkFQSSBVUkwgUmVzb2x2ZXIiPjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly90d2l0dGVyLmNvbS9WaW5pdF9TaGFoZGVvIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90d2l0dGVyL2ZvbGxvdy9WaW5pdF9TaGFoZGVvP3N0eWxlPXNvY2lhbCIgYWx0PSJUd2l0dGVyIEZvbGxvdyI+PC9hPgo8YSBocmVmPScuL0xJQ0VOU0UnPjxpbWcgYWx0PSJHaXRIdWIgTGljZW5zZSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL3Zpbml0c2hhaGRlby9jZWxlYi1kaXdhbGk/bGFiZWw9TGljZW5zZSZsb2dvPWdpdGh1YiI+PC9hPgo8L3A+Cgo8YnIvPgoKPC9kaXY+CgoKKipvcGVuYXBpLXVybC1yZXNvbHZlcioqIGlzIGEgKipsaWdodHdlaWdodCoqIE5QTSBwYWNrYWdlIHRoYXQgcHJvdmlkZXMgYSBzaW1wbGUgYW5kIGVmZmljaWVudCB3YXkgdG8gcmVzb2x2ZSBzZXJ2ZXIgVVJMcyBmcm9tIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMuIEl0IGFsc28gcmVtb3ZlcyBwcm90b2NvbHMgZnJvbSB0aGUgcmVzb2x2ZWQgVVJMcyBhbmQgYWxsb3dzIHlvdSB0byAqKmVhc2lseSBleHRyYWN0IGhvc3QgaW5mb3JtYXRpb24gZnJvbSBPcGVuQVBJIGRlZmluaXRpb25zKiouIFRoaXMgcGFja2FnZSBpcyBpZGVhbCBmb3IgZGV2ZWxvcGVycyB3b3JraW5nIHdpdGggQVBJcyB0aGF0IGNvbmZvcm0gdG8gdGhlIFtPcGVuQVBJIDMueCBzcGVjaWZpY2F0aW9uXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pIGFuZCBuZWVkIHRvIGV4dHJhY3QgW3NlcnZlciBpbmZvcm1hdGlvbl0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMS4wI3NlcnZlci1vYmplY3QpIHRvIG1ha2UgQVBJIGNhbGxzLiBDaGVjayBvdXQgdGhlIGJsb2cgcG9zdCBmb3IgbW9yZSBkZXRhaWxz4oCUW0hvdyB0byByZXNvbHZlIHNlcnZlciBVUkxzIGNvbnRhaW5pbmcgdmFyaWFibGVzIGluIE9wZW5BUEkgMy54IGRlZmluaXRpb25zP10oaHR0cHM6Ly9kZXYudG8vdmluaXRzaGFoZGVvL2hvdy10by1yZXNvbHZlLXNlcnZlci11cmxzLWNvbnRhaW5pbmctdmFyaWFibGVzLWluLW9wZW5hcGktM3gtZGVmaW5pdGlvbnMta2FqKQoKIyMjIEZlYXR1cmVzOgoKLSDwn5qAICoqTGlnaHR3ZWlnaHQqKiBtb2R1bGUgd2l0aCBvbmx5IDk2NSBieXRlcyBpbiBzaXplCi0g8J+OiSAqKlplcm8gZGVwZW5kZW5jaWVzKiosIG1ha2luZyBpdCBlYXN5IHRvIGluc3RhbGwgYW5kIHVzZSBpbiB5b3VyIHByb2plY3RzCi0g8J+UjSBFZmZpY2llbnQgYW5kIHNpbXBsZSB3YXkgdG8gcmVzb2x2ZSBbc2VydmVyIFVSTHNdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCNzZXJ2ZXItb2JqZWN0KSBmcm9tIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMKCiMjIPCfk6YgSW5zdGFsbGF0aW9uCgpZb3UgY2FuIGluc3RhbGwgYG9wZW5hcGktdXJsLXJlc29sdmVyYCB2aWEgW05QTV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS11cmwtcmVzb2x2ZXIpOgoKYGBgYmFzaApucG0gaW5zdGFsbCBvcGVuYXBpLXVybC1yZXNvbHZlcgpgYGAKClshW29wZW5hcGktdXJsLXJlc29sdmVyIG9uIE5QTV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vb3BlbmFwaS11cmwtcmVzb2x2ZXIucG5nKV0oaHR0cHM6Ly9ub2RlaS5jby9ucG0vb3BlbmFwaS11cmwtcmVzb2x2ZXIvKQoKIyMg8J+SuyBVc2FnZQoKVG8gdXNlIGBvcGVuYXBpLXVybC1yZXNvbHZlcmAsIHlvdSBuZWVkIHRvIHBhc3MgYW4gW09wZW5BUEkgMy54IHNwZWNpZmljYXRpb25dKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykgb2JqZWN0IHRvIHRoZSBgcmVzb2x2ZSgpYCBmdW5jdGlvbi4gVGhpcyBmdW5jdGlvbiB3aWxsIHJldHVybiBhbiBhcnJheSBvZiByZXNvbHZlZCBzZXJ2ZXIgVVJMczoKCmBgYGphdmFzY3JpcHQKY29uc3Qgb3BlbmFwaVVybFJlc29sdmVyID0gcmVxdWlyZSgnb3BlbmFwaS11cmwtcmVzb2x2ZXInKQoKY29uc3Qgc3BlYyA9IHsKICBvcGVuYXBpOiAnMy4wLjAnLAogIHNlcnZlcnM6IFsKICAgIHsKICAgICAgdXJsOiAnaHR0cHM6Ly97dXNlcm5hbWV9LmdpZ2FudGljLXNlcnZlci5jb206e3BvcnR9L3tiYXNlUGF0aH0nLAogICAgICBkZXNjcmlwdGlvbjogJ1RoZSBwcm9kdWN0aW9uIEFQSSBzZXJ2ZXInLAogICAgICB2YXJpYWJsZXM6IHsKICAgICAgICB1c2VybmFtZTogewogICAgICAgICAgZGVmYXVsdDogJ2RlbW8nLAogICAgICAgICAgZGVzY3JpcHRpb246ICd0aGlzIHZhbHVlIGlzIGFzc2lnbmVkIGJ5IHRoZSBzZXJ2aWNlIHByb3ZpZGVyLCBpbiB0aGlzIGV4YW1wbGUgYGdpZ2FudGljLXNlcnZlci5jb21gJwogICAgICAgIH0sCiAgICAgICAgcG9ydDogewogICAgICAgICAgZW51bTogWyc4NDQzJywgJzQ0MyddLAogICAgICAgICAgZGVmYXVsdDogJzg0NDMnCiAgICAgICAgfSwKICAgICAgICBiYXNlUGF0aDogewogICAgICAgICAgZGVmYXVsdDogJ3YyJwogICAgICAgIH0KICAgICAgfQogICAgfQogIF0KfQoKY29uc3QgaG9zdHMgPSBvcGVuYXBpVXJsUmVzb2x2ZXIucmVzb2x2ZShzcGVjKQoKLyoKClsKICAnZGVtby5naWdhbnRpYy1zZXJ2ZXIuY29tOjg0NDMvdjInLAogICdkZW1vLmdpZ2FudGljLXNlcnZlci5jb206NDQzL3YyJwpdCgoqLwpjb25zb2xlLmxvZyhob3N0cykKCmBgYAoKUGFzcyBgZmFsc2VgIGFzIHNlY29uZCBwYXJhbWV0ZXIgdG8gZ2V0IHRoZSBzZXJ2ZXIgVVJMcyB3aXRoIHByb3RvY29scy4KCmBgYGphdmFzY3JpcHQKY29uc3Qgc2VydmVyVXJscyA9IG9wZW5hcGlVcmxSZXNvbHZlci5yZXNvbHZlKHNwZWMsIGZhbHNlKQoKLyoKClsKICAnaHR0cHM6Ly9kZW1vLmdpZ2FudGljLXNlcnZlci5jb206ODQ0My92MicsCiAgJ2h0dHBzOi8vZGVtby5naWdhbnRpYy1zZXJ2ZXIuY29tOjQ0My92MicKXQoKKi8KY29uc29sZS5sb2coc2VydmVyVXJscykKCmBgYAoKIyMg8J+nqiBUZXN0aW5nCgpZb3UgY2FuIHRlc3QgdXNpbmcgdGhlIGJlbG93IGNvbW1hbmQgb3Igd3JpdGUgeW91ciBvd24gdGVzdHMgdXNpbmcgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgW2V4YW1wbGVzXSguL2RlZmluaXRpb25zLykuCgpgYGBiYXNoCm5wbSB0ZXN0CmBgYAoKIyMg8J+aqyBMaW1pdGF0aW9ucwoKVGhlIGJlbG93IGFyZSB0aGUga25vd24gbGltaXRhdGlvbnMsIGFuZCB0aGV5IGFyZSBub3QgaGFuZGxlZCB0byBrZWVwIGl0IGEgKipsaWdodHdlaWdodCoqIGFuZCAqKmZvY3VzZWQqKiBtb2R1bGUgdG8ganVzdCBleHRyYWN0IHRoZSBbc2VydmVyIGluZm9ybWF0aW9uXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4xLjAjc2VydmVyLW9iamVjdCkuCgotIEl0IGRvZXMgbm90IHZhbGlkYXRlIHRoZSBPcGVuQVBJIGRlZmluaXRpb24uIFlvdSBjYW4gdXNlIFtvcGVuYXBpLXNjaGVtYS12YWxpZGF0b3JdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29wZW5hcGktc2NoZW1hLXZhbGlkYXRvcikgZm9yIHZhbGlkYXRpbmcgdGhlIE9wZW5BUEkgZGVmaW5pdGlvbi4KLSBJdCBvbmx5IHdvcmtzIHdpdGggSlNPTiBmb3JtYXQuCiAgLSBVc2UgW3lhbWwtdG8tanNvbi1zY2hlbWFdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3lhbWwtdG8tanNvbi1zY2hlbWEpIHRvIGNvbnZlcnQgWUFNTCB0byBKU09OLgogIC0gVXNlIFtwb3N0bWFuLXRvLW9wZW5hcGldKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3Bvc3RtYW4tdG8tb3BlbmFwaSkgdG8gY29udmVydCBQb3N0bWFuIGNvbGxlY3Rpb24gdG8gT3BlbkFQSSAzLngKCiMjIPCfpJ0gQ29udHJpYnV0aW5nCgoqKkNvbnRyaWJ1dGlvbnMgdG8gYG9wZW5hcGktdXJsLXJlc29sdmVyYCBhcmUgbW9zdCB3ZWxjb21lISoqIAoKSWYgeW91IGZpbmQgYSBidWcgb3Igd2FudCB0byBzdWdnZXN0IGEgbmV3IGZlYXR1cmUsIHBsZWFzZSBbb3BlbiBhbiBpc3N1ZV0oaHR0cHM6Ly9naXRodWIuY29tL3Zpbml0c2hhaGRlby9vcGVuYXBpLXVybC1yZXNvbHZlci9pc3N1ZXMvbmV3KSBvbiB0aGUgR2l0SHViIHJlcG9zaXRvcnkuIElmIHlvdSB3YW50IHRvIGNvbnRyaWJ1dGUgY29kZSwgcGxlYXNlIFtmb3JrIHRoZSByZXBvc2l0b3J5XShodHRwczovL2dpdGh1Yi5jb20vdmluaXRzaGFoZGVvL29wZW5hcGktdXJsLXJlc29sdmVyL2ZvcmspLCBtYWtlIHlvdXIgY2hhbmdlcywgYW5kICoqc3VibWl0IGEgcHVsbCByZXF1ZXN0KiouIFlvdXIgY29udHJpYnV0aW9ucyBhbmQgZmVlZGJhY2sgYXJlIG1vc3Qgd2VsY29tZSEKCiMjIPCfk50gTGljZW5zZQoKW29wZW5hcGktdXJsLXJlc29sdmVyXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLXVybC1yZXNvbHZlcikgaXMgYXV0aG9yZWQgYnkgQFt2aW5pdHNoYWhkZW9dKGh0dHBzOi8vZ2l0aHViLmNvbS92aW5pdHNoYWhkZW8pIGFuZCByZWxlYXNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXSguL0xJQ0VOU0UpLgoKIyMg4p2k77iPIFN1cHBvcnQKCklmIHlvdSBmaW5kIHRoaXMgcGFja2FnZSB1c2VmdWwsIHBsZWFzZSBjb25zaWRlciBbc3RhcnJpbmcgdGhpcyByZXBvc2l0b3J5XShodHRwczovL2dpdGh1Yi5jb20vdmluaXRzaGFoZGVvL29wZW5hcGktdXJsLXJlc29sdmVyKSBvbiBHaXRIdWIgdG8gc2hvdyB5b3VyIGFwcHJlY2lhdGlvbi4gWW91IGNhbiBzdGF5IGNvbm5lY3RlZCB3aXRoIG1lIG9uIFR3aXR0ZXLigJRbQHZpbml0X3NoYWhkZW9dKGh0dHBzOi8vdHdpdHRlci5jb20vVmluaXRfU2hhaGRlbykuCgo8YSBocmVmPSJodHRwczovL3d3dy5idXltZWFjb2ZmZWUuY29tL3Zpbml0c2hhaGRlbyIgdGFyZ2V0PSJfYmxhbmsiPjxpbWcgc3JjPSJodHRwczovL2Nkbi5idXltZWFjb2ZmZWUuY29tL2J1dHRvbnMvdjIvZGVmYXVsdC15ZWxsb3cucG5nIiBhbHQ9IkJ1eSBNZSBBIENvZmZlZSIgaGVpZ2h0PSI0MHB4IiB3aWR0aD0iMTUwcHgiID48L2E+CgpUaGFuayB5b3UgZm9yIHlvdXIgc3VwcG9ydCEg8J+Zjwo= readmeEtag: '"d32f42d9c1c80df640aafda2af1bcc7e4579b025"' readmeLastModified: Thu, 27 Apr 2023 05:37:54 GMT repositoryId: 630161086 description: >- Resolve server URLs and extract hosts from OpenAPI 3.x and Swagger specifications. created: '2023-04-19T19:50:39Z' updated: '2025-01-21T01:56:10Z' language: JavaScript archived: false stars: 7 watchers: 1 forks: 0 owner: vinitshahdeo logo: https://avatars.githubusercontent.com/u/20594326?v=4 license: MIT repoEtag: '"58f50b78e2b5af90928f7cda06328b65ef8331c6ea2c6c8ab9e01faf5aecb20f"' repoLastModified: Tue, 21 Jan 2025 01:56:10 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/koshevy/oapi3codegen v3: true repositoryMetadata: base64Readme: >- # Code generation from OpenAPI 3 to TypeScript

Lightweight and simple. Can be used for *browser's* and for Node.js ecosystems.
Now supports converting from **OpenAPI 3** to **TypeScript** types (OpenAPI 2 and below are not supported).

<details>
<summary>For example, creates constructions such as (click to expand):</summary>

```typescript
/**
 * Typical 401 response
 */
export interface HttpErrorUnauthorized {
  /**
   * Error message
   */
  message: string;

  /**
   * Data appropriate to
   * [WWW-Authenticate](https://tools.ietf.org/html/rfc7235#section-3.1).
   */
  wwwAuthenticate?: {
    /**
     * Prompt to authenticate
     */
    title: string;

    /**
     * Kind of authorization user has to use
     */
    type: string;

    /**
     * Type of authority ("barrier" or etc.)
     */
    realm: string;
  };
}

export interface GetParametersMeta_response401
  extends HttpErrorUnauthorized {

  /**
   * Error message
   */
  message: string;

  /**
   * Data appropriate to
   * [WWW-Authenticate](https://tools.ietf.org/html/rfc7235#section-3.1).
   */
  wwwAuthenticate?: {
    /**
     * Prompt to authenticate
     */
    title: string;

    /**
     * Kind of authorization user has to use
     */
    type: string;

    /**
     * Type of authority ("barrier" or etc.)
     */
    realm: string;
  };
}

/**
 * Kind of request error: syntax or semantic. Syntax error means the
 * application logic error, semantics — error of data, that contragent
 * inputs.
 */
export enum HttpErrorBadRequestTypeEnum {
  Syntax = "syntax",
  Semantic = "semantic"
}

/**
 * Typical response one Bad Request Error (400)
 */
export interface HttpErrorBadRequest {
  /**
   * Common error message
   */
  message: string;

  /**
   * Kind of request error: syntax or semantic. Syntax error means the
   * application logic error, semantics — error of data, that contragent
   * inputs.
   */
  type?: HttpErrorBadRequestTypeEnum;

  /**
   * Explained description of error
   */
  description?: string;

  /**
   * Additional list of errors with JSON-pointers
   */
  errors?: Array<{
    originalMessage: string;

    message?: string;

    jsonPointer: string;
  }>;
}

/**
 * ## MetaDataView
 * MetaData helps decide what the method has to use to
 * interpret and render parameter or category of parameters.
 */
export type GetParametersMeta_response200 = Array<Category | Parameter>;

```
</details>

-----

> #### 🚦 Status of project
> - **Stabilization:** ready to use in non-production projects (see Road Map).
> - **Not supported YML (only JSON)**. *Will be supported soon. Now, you can use something like [js-yaml](https://www.npmjs.com/package/js-yaml)*
> - ***🐞 UPDATE:*** Fixed problem with infinity loop in recursive types!
>
> Please, if it possible: send me your schemas that not converts properly.

### Online demo

Work in progress! Now, see how to use.

### Road map

Work is just started, and current functionality (creating TypeScript types) —
It's just a first part of the supposed functionality.
With time, it's supposed, possibilities of this plugin will be extended by:

- *Make project well documented [in progress]*
- *YAML support [in progress]*
- *Stabilization [in progress]*
- *Playground site [in progress]*
- Plugins support
- Generation classes (now creates only interfaces) with internal validation [ *concept description is coming soon* ]
- Generation API Classes
    - Solutions for AngularX based on *Assured requests idea* with internal validation [ *concept description is coming soon* ]
    - *May be, something else...*
- **DDD-tools** (Start kit for OpenAPI3-projects):
    - Bundler for complex files structure. Need fo organization complex API-libraries.
    - [Dredd](https://www.npmjs.com/package/dredd)
    - Most populars doc-generators (at least, [Redoc](https://www.npmjs.com/package/redoc) and [SwaggerUI](https://www.npmjs.com/package/swagger-ui))
    - Convention for working with DDD OpenAPI3
- **Other languages** (is there are need for it?): Kotlin, Java, PHP, 

### Ideology

- **Data format loyalty:** it's should be as useful as it possible regardless validity of source data.
- **Integration**: solution should be able to be integrated to any system (at least, based on NPM). It's mean, should have CLI and API.

### Why not swagger-codegen?

- Uses NPM instead Java (install via npm and easy-integratable)
- Lightweight ([swagger-codegen](https://github.com/swagger-api/swagger-codegen) solution — it's about a 14mb for ready JAR)
- Supports OpenAPI3
- Above all, intended for TypeScript 

## Install

##### 1. Install TypeScript

```sh
npm install typescript@latest -g
``` 

##### 2. Install oapi3codegen global (for CLI using)

```sh
npm install oapi3codegen -g
``` 

##### 3. Install oapi3codegen for local project

```sh
npm install oapi3codegen --save
``` 

## Using via CLI

```sh
oapi3codegen --srcPath /PATH/TO/SRC/open-api-file.json --destPath /PATH/TO/DEST --separatedFiles true
```

#### CLI arguments

| CLI Argument       | Description                                                                   |
|--------------------|-------------------------------------------------------------------------------|
| **srcPath**        | Path of url of JSON file with OpenAPI3 specification                          |
| **destPath**       | Path for destination directory                                                |
| **separatedFiles** | Whether should converted types be saved in separated files, or in single file |

Also, you can set some of options for convertor's [configuration](https://github.com/koshevy/oapi3codegen/blob/master/core/config.ts#L99)
config via CLI:

| Option                          | Description                                                                   |
|---------------------------------|-------------------------------------------------------------------------------|
| **defaultContentType**          | Default content-type contains no prefixes/suffixes in type names.             |
| **implicitTypesRefReplacement** | Mode when models that refer to any models via `$ref` are replacing implicitly even if firsts have names |

## Using via API (TypeScript)

You can convert whole OpenAPI3-specification:

```typescript
import { Convertor } from 'oapi3codegen';

const convertor: Convertor = new Convertor();

/**
 * Base models of specification:
 *  - Requests bodies models
 *  - Requests params sets models
 *  - Responses models
 *
 * Converting starts from entry points and extracts
 * referred types and dependencies. It s why we need
 * to get "entry points". 
 */
const entryPoints = convertor.getOAPI3EntryPoints(context);

/**
 * Rendering each type: every entry point and each of
 * theirs related types.
 */
Convertor.renderRecursive(
    entryPoints,
    (descriptor: DataTypeDescriptor, text) => {
        // Here your code: you get text and type descriptor.
        // You can see how oapi3codegen's CLI uses this calback here:
        // https://github.com/koshevy/oapi3codegen/blob/master/cli.ts#L73
    }
);
```

And also, you can convert stand-alone JSON-schema into type descriptor,
that could be rendered:

```typescript
import { Convertor } from 'oapi3codegen';
// you need prettier to beautify result of rendering
import * as prettier from 'prettier';
// provides `_.each(...)` for our example
import * as _ from 'lodash';

const convertor: Convertor = new Convertor();

const anotherJsonSchemaObject = {
    "title": "Person",
    "description": "Information about person you have to register in your system.",
    "type": "object",
    "properties": {
        "firstName": {
            "type": "string"
        },
        "lastName": {
            "type": "string"
        },
        "age": {
            "description": "Age in years",
            "type": "integer",
            "minimum": 0
        }
    },
    "required": ["firstName", "lastName"]
};

const convertResult = convertor.convert(
    anotherJsonSchemaObject,
    {},
    'AnotherType'
);

_.each(convertResult, typeDescriptor => {
    const typeCode = prettier.format(
        typeDescriptor.render([]),
        {parser: 'typescript'}
    );

    console.log(typeCode);
});

```

And this code will print result:

```typescript
/**
 * ## Person
 * Information about person you have to register in your system.
 */
export interface AnotherType {
  firstName: string;

  lastName: string;

  /**
   * Age in years
   */
  age?: number;
}
```

## API (TypeScript)

Coming soon...

## How to extend?

Coming soon... readmeEtag: '"ff94db1290d21d23b6d798ee329fb71914334936"' readmeLastModified: Tue, 15 May 2018 07:24:08 GMT repositoryId: 131764821 description: Moved here — https://github.com/koshevy/codegena created: '2018-05-01T21:25:54Z' updated: '2023-01-28T14:51:04Z' language: TypeScript archived: true stars: 7 watchers: 1 forks: 3 owner: koshevy logo: https://avatars.githubusercontent.com/u/1203974?v=4 repoEtag: '"6ed3ccd1737db2e37bf8597269b9cef834f2f3e6378d8ce40a9ad75006571bd7"' repoLastModified: Sat, 28 Jan 2023 14:51:04 GMT foundInMaster: true category: Parsers id: f3f5958dfb657fddca8ac61f63c3d588 - source: openapi3 tags repository: https://github.com/wubin1989/pullcode v3: true id: bc40e809ab84668bedebf55e3df56f1b repositoryMetadata: base64Readme: >- # pullcode
  
Pullcode is OpenAPI (aka Swagger) specification compatible typescript http client code generation tool relying on axios. You can configure it to npm scripts to directly generate client code to specified path to easily use. Support Swagger 2 and OpenAPI 3 (aka Swagger 3) in json format.

<!-- START doctoc generated TOC please keep comment here to allow auto update -->
<!-- DON'T EDIT THIS SECTION, INSTEAD RE-RUN doctoc TO UPDATE -->
### TOC

- [Features](#features)
- [Credits](#credits)
- [Installation](#installation)
- [CLI Options](#cli-options)
- [Generation Rule](#generation-rule)
- [Usage](#usage)
- [Quickstart](#quickstart)
- [More Examples](#more-examples)
  - [Response Error Interceptor](#response-error-interceptor)
  - [Send Requests](#send-requests)
- [Sister Projects](#sister-projects)
- [License](#license)

<!-- END doctoc generated TOC please keep comment here to allow auto update -->

## Features

1. Totally typescript support. Pullcode generates service class, methods, request body and response body object types based on json formatted Swagger2/OpenAPI3 documentation.
2. Rich and useful configuration options. Pullcode wrapps axios configuration options and add more options for you to elegantly customize requests.
3. Built-in axios request and response interceptors. Request interceptor handles request url rewrite and adds Authorization header. Response interceptor extracts data attribute from raw response. Users can configure additional custom interceptors to do further process. There is no built-in request error and response error interceptors, users should pass their own implementations if necessary.
4. Framework-agnostic. Pullcode can be used with any frontend frameworks.

## Credits

* [commander.js](https://github.com/tj/commander.js)：nodejs command line tool library
* [swagger2openapi](https://github.com/Mermade/oas-kit/blob/main/packages/swagger2openapi/README.md)：convert swagger 2 to OpenAPI 3
* [vue-vben-admin](https://github.com/vbenjs/vue-vben-admin): a modern vue3 admin

## Installation

```shell
npm install --save pullcode
```

NOTE: generated code imports code from pullcode, so pullcode is not only a CLI, but also a npm package, you should use `--save` instead of `--save-dev`.

## CLI Options

```shell
➜  pullcode git:(master) ✗ pullcode -h                                           
Usage: pullcode [options]

Options:
  -o, --output <value>  code output path
  -u, --url <value>     swagger 2.0 or openapi 3.0 json api document download url
  -h, --help            display help for command
```

* output: directories will be recursively created if not exists.

## Generation Rule

Pullcode will check if `BizService.ts` file exists, if exists, skip. Other files will always be overwritten.

## Usage

1. Configure pullcode command into `scripts` attribute in package.json. For example:

```javascript
"scripts": {
  "pull": "pullcode -u https://petstore3.swagger.io/api/v3/openapi.json -o src/api"
},
```

2. Run `npm run pull`

3. Open `BizService.ts` file, fix `defaultOptions` parameters to fit your need.  

```javascript
const defaultOptions: CreateAxiosOptions = {
  requestOptions: {
    apiUrl: '', // same as baseUrl
    urlPrefix: '',
  } as RequestOptions,
}
```

- `apiUrl`: if configured proxy, `apiUrl` should be set to a prefix for matching the proxy.
- `urlPrefix`: should be confirmed with your backend colleagues if there is url prefix to match service name (if the service is behind a gateway) or servlet contxt path (if the service is based on spring boot framework).
- `transform`: set your custom axios interceptors. Normally, you just need to customize response error interceptor.
- `tokenGetter`: set auth token getter function. The function will be called in built-in request interceptor to put token into header.

For example: 

```javascript
import { merge } from 'lodash-es';
import { CreateAxiosOptions, VAxios } from '@/httputil/Axios';
import { useGlobSetting } from '/@/hooks/setting';
import { transform } from '/@/api/interceptor'
import { getToken } from '/@/utils/auth';
import { RequestOptions } from '@/types/axios';

const globSetting = useGlobSetting();

const defaultOptions: CreateAxiosOptions = {
  transform,
  requestOptions: {
    apiUrl: globSetting.apiUrl, // same as baseUrl
    urlPrefix: '/myservice',
  } as RequestOptions,
  tokenGetter: getToken as () => string
}

export class BizService extends VAxios {
  constructor(options?: Partial<CreateAxiosOptions>) {
    super(merge(defaultOptions, options || {}));
  }
}

export default BizService;
```

For other options, please read `CreateAxiosOptions` definition from source code.

4. After above fix, you can import default singleton service instance to components to send http requests.

## Quickstart

Please refer to this example app：[pullcode-quickstart](https://github.com/wubin1989/pullcode/tree/master/examples/pullcode-quickstart)

![petstore](./petstore.png)

## More Examples
### Response Error Interceptor

```javascript
import type { AxiosResponse } from 'axios';
import type { AxiosTransform } from '@/httputil/axiosTransform';
import { checkStatus } from './checkStatus';
import { useMessage } from '/@/hooks/web/useMessage';
import { useErrorLogStoreWithOut } from '/@/store/modules/errorLog';
import { useI18n } from '/@/hooks/web/useI18n';

const { createMessage, createErrorModal } = useMessage();

export const transform: AxiosTransform = {
  /**
   * @description: Response Error Interceptor
   */
  responseInterceptorsCatch: (_: AxiosResponse, error: any) => {
    const { t } = useI18n();
    const errorLogStore = useErrorLogStoreWithOut();
    errorLogStore.addAjaxErrorInfo(error);
    const { response, code, message, config } = error || {};
    const errorMessageMode = config?.requestOptions?.errorMessageMode || 'none';
    const msg: string = response?.data?.error?.message ?? '';
    const err: string = error?.toString?.() ?? '';
    let errMessage = '';

    try {
      if (code === 'ECONNABORTED' && message.indexOf('timeout') !== -1) {
        errMessage = t('sys.api.apiTimeoutMessage');
      }
      if (err?.includes('Network Error')) {
        errMessage = t('sys.api.networkExceptionMsg');
      }

      if (errMessage) {
        if (errorMessageMode === 'modal') {
          createErrorModal({ title: t('sys.api.errorTip'), content: errMessage });
        } else if (errorMessageMode === 'message') {
          createMessage.error(errMessage);
        }
        return Promise.reject(error);
      }
    } catch (error) {
      throw new Error(error as unknown as string);
    }

    checkStatus(error?.response?.status, msg, errorMessageMode);
    return Promise.reject(error);
  },
};
```

### Send Requests

```javascript
<script setup lang="ts">
import { petService } from '@/api/PetService';
import { Pet, PetStatusEnum } from '@/api/types';
import { ref } from 'vue';

let loading = ref(true);
let dataSource = ref([] as Pet[]);

petService.getPetFindByStatus({
  status: PetStatusEnum.AVAILABLE,
}).then((resp: Pet[]) => {
  dataSource.value = resp
  loading.value = false
})
</script>
```

## Sister Projects

- [go-doudou](https://github.com/unionj-cloud/go-doudou): OpenAPI 3.0 spec based lightweight microservice framework written in Go. It supports monolith service application as well. Currently, it supports RESTful service only.

## License

MIT readmeEtag: '"43d5b70ed509b8114b9fd4895334bf9babca2450"' readmeLastModified: Sun, 20 Aug 2023 17:15:06 GMT repositoryId: 509066883 description: >- pullcode is a typescript http client code generation cli compatible with Swagger 2 and OpenAPI 3 created: '2022-06-30T12:15:28Z' updated: '2025-09-24T07:48:19Z' language: TypeScript archived: false stars: 8 watchers: 1 forks: 2 owner: wubin1989 logo: https://avatars.githubusercontent.com/u/12456315?v=4 license: MIT repoEtag: '"ac61be12133dd2a96b4ff5b2f73f02d87d74eb9b2fd8c68ea4dca505d856a375"' repoLastModified: Wed, 24 Sep 2025 07:48:19 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/lzehrung/zod-express-openapi-routes v3: true id: 86218aaea66f6e330ef4798b47533734 repositoryMetadata: base64Readme: >- IyMgT3ZlcnZpZXcKCi0gRmluZCBhbiBlcmdvbm9taWMgd2F5IChtaW5pbWl6ZSBib2lsZXJwbGF0ZSBhbmQgY29kZSBkdXBsaWNhdGlvbikgdG8gZ2V0IHR5cGVkLCB6b2QtdmFsaWRhdGVkIGV4cHJlc3Mgcm91dGVzIGFuZCBhbiBPcGVuQVBJCiAgc2NoZW1hIGF0IHRoZSBzYW1lIHRpbWUKLSBNaW5pbWl6ZSBjb2RlIGR1cGxpY2F0aW9uIGFuZCBib2lsZXJwbGF0ZSBiZXR3ZWVuIGV4cHJlc3Mgcm91dGVzLCByZXF1ZXN0IHBhcmFtZXRlciB2YWxpZGF0aW9uLCBhbmQgT3BlbkFQSSBkZWZpbml0aW9ucwoKIyMgSGlnaGxpZ2h0cwoKLSBiYXNlIGBDb250cm9sbGVyYCBjbGFzcyB0aGF0IGNvbnNvbGlkYXRlcyBjcmVhdGlvbiBvZiBleHByZXNzIHJvdXRlLCB6b2QtdmFsaWRhdGVkIHJlcXVlc3QgcGFyYW1ldGVycywgT3BlbkFQSSBkZWZpbml0aW9uLCBhbmQgZ2V0dGluZyBhIHR5cGVkIGBleHByZXNzLlJvdXRlSGFuZGxlcmA6IFtzcmMvem9kLW9wZW5hcGktZXhwcmVzcy1yb3V0ZXMvem9kLWFwaS5jb250cm9sbGVyLnRzXShzcmMvem9kLW9wZW5hcGktZXhwcmVzcy1yb3V0ZXMvem9kLWFwaS5jb250cm9sbGVyLnRzKQotIGV4YW1wbGUgY29udHJvbGxlciBpbnN0YW5jZTogW3NyYy9wcm9kdWN0cy9wcm9kdWN0cy5jb250cm9sbGVyLnRzXShzcmMvcHJvZHVjdHMvcHJvZHVjdHMuY29udHJvbGxlci50cykKLSBleGFtcGxlIHJlcXVlc3QgcGFyYW1ldGVyIHpvZCBzY2hlbWFzOiBbc3JjL3Byb2R1Y3RzL2FwaS1zY2hlbWFzLnRzXShzcmMvcHJvZHVjdHMvYXBpLXNjaGVtYXMudHMpCi0gZXhhbXBsZSBtYW51YWxseS1kZWZpbmVkIGZpbGUgdXBsb2FkIHJvdXRlIHVzaW5nIFttdWx0ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9leHByZXNzanMvbXVsdGVyKSBmb3IgdXBsb2FkczogW3NyYy9wcm9kdWN0cy9wcm9kdWN0cy5jb250cm9sbGVyLnRzI0wxNDhdKHNyYy9wcm9kdWN0cy9wcm9kdWN0cy5jb250cm9sbGVyLnRzI0wxNDgpCgojIyBSdW4gdGhlIGV4YW1wbGUKCi0gY2xvbmUgdGhlIHJlcG9zaXRvcnkKLSBgbnBtIGluc3RhbGxgCi0gYG5wbSBydW4gZGV2YAotIG9wZW4gdGhlIGdlbmVyYXRlZCBPcGVuQVBJIGRvY3MgaHR0cDovL2xvY2FsaG9zdDozMjUwL2FwaS9yZWZlcmVuY2UKLSBvcGVuIHRoZSBnZW5lcmF0ZWQgYHN3YWdnZXIuanNvbmAgaHR0cDovL2xvY2FsaG9zdDozMjUwL2FwaS9zd2FnZ2VyLmpzb24KLSBnZXQgYSBzaW5nbGUgcmVzb3VyY2UgaHR0cDovL2xvY2FsaG9zdDozMjUwL2FwaS9wcm9kdWN0cy8xCi0gYW4gaW52YWxpZCBwYXRoIHBhcmFtZXRlciByZXN1bHRzIGluIHZhbGlkYXRpb24gZXJyb3JzIGh0dHA6Ly9sb2NhbGhvc3Q6MzI1MC9hcGkvcHJvZHVjdHMvYWJjCgojIyBEZXBlbmRlbmNpZXMKCi0gYEBhbmF0aW5lL3pvZC1vcGVuYXBpYDogZ2VuZXJhdGVzIHJlcXVlc3QgcGFyYW1ldGVyIE9wZW5BUEkgZGVmaW5pdGlvbnMgZnJvbSB6b2Qgc2NoZW1hcwotIGBvcGVuYXBpMy10c2A6IHVzZWQgdG8gZGVmaW5lIE9wZW5BUEkgc2NoZW1hIGluIFR5cGVTY3JpcHQKLSBgc3dhZ2dlci11aS1leHByZXNzYDogc2VydmVzIHRoZSBPcGVuQVBJIGRvY3VtZW50YXRpb24KLSBgem9kYDogdXNlZCB0byBkZWZpbmUgcmVxdWVzdCBwYXJhbWV0ZXIgdmFsaWRhdGlvbiBzY2hlbWFzCgojIyBMaW1pdGF0aW9ucwoKLSBmaWxlIHVwbG9hZCByb3V0ZXMgbXVzdCBiZSBkb2N1bWVudGVkIG1hbnVhbGx5Ci0gaGF2ZW4ndCB0ZXN0ZWQgbmVzdGVkIHJvdXRlcnMgLyByb3V0ZXMKLSBub3QgZXh0ZW5zaXZlbHkgdGVzdGVkCg== readmeEtag: '"0bd31b6e9696fb8a5820a4bcb9181fe61851f364"' readmeLastModified: Sun, 21 Apr 2024 22:36:31 GMT repositoryId: 577179642 description: >- POC for cozy API validation, typing, and OpenAPI/swagger schema generation with Zod and Express created: '2022-12-12T06:37:15Z' updated: '2025-11-26T19:13:52Z' language: TypeScript archived: false stars: 7 watchers: 1 forks: 1 owner: lzehrung logo: https://avatars.githubusercontent.com/u/924226?v=4 repoEtag: '"29b0655b8aa3ea788305cc24172a24fff37d4e718d66e58119b49abfb15702f3"' repoLastModified: Wed, 26 Nov 2025 19:13:52 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/laura-rodriguez/dotnetconf22 v3: true id: 3eb48a23918165d7000be240cf49d8a6 repositoryMetadata: base64Readme: >- IyBHcm9jZXJ5IEl0ZW1zIE1hbmFnZW1lbnQgU0RLCiAKIFRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBhbGwgdGhlIHByb2plY3RzIGJ1aWx0IGR1cmluZyBteSB0YWxrICJCdWlsZCBhbiBTREsgd2l0aCBPcGVuQVBJIGFuZCBOU3dhZyIgYXQgLk5FVENvbmYgMjAyMiAKCiAqIEdyb2NlcnkgQVBJOiBUaGlzIHByb2plY3QgY29udGFpbnMgdGhlIEFQSSBpbXBsZW1lbnRhdGlvbiBwbHVzIHRoZSBEQUwgbG9naWMgKFdlYiBBUEkvLk5FVCA3KQogKiBHcm9jZXJ5IFNESzogVGhpcyBwcm9qZWN0IGNvbnRhaW5zIHRoZSBjbGllbnQgaW1wbGVtZW50YXRpb24gZ2VuZXJhdGVkIGJ5IE5Td2FnIChDbGFzcyBMaWJyYXJ5Ly5ORVQgU3RhbmRhcmQgMi4wKQogKiBTYW1wbGUgQXBwOiBUaGlzIHByb2plY3QgdXRpbGl6ZXMgdGhlIFNESyAoQ29uc29sZSBBcHAvLk5FVCBDb3JlIDMuMSkKCiAjIyBHZXR0aW5nIFN0YXJ0ZWQKCjEuIE1ha2Ugc3VyZSB5b3UgaGF2ZSBhbGwgdGhlIC5ORVQgcnVudGltZXMgdXNlZCBieSB0aGlzIHNvbHV0aW9uIGluc3RhbGxlZAoyLiBDb25maWd1cmUgdGhlIEdyb2NlcnkgQVBJIHByb2plY3QgdG8gdXNlIFVzZXIgSldUcyBieSBydW5uaW5nIGBkb3RuZXQgdXNlci1qd3RzIGNyZWF0ZWAgCjMuIFVwZGF0ZSB0aGUgUHJvZ3JhbSBmaWxlIHdpdGggeW91ciB0b2tlbiBpbiB0aGUgQ29uc29sZSBBcHAKNC4gQ29uZmlndXJlIHlvdXIgc29sdXRpb24gdG8gcnVuIHRoZSBBUEkgYW5kIHRoZSBDb25zb2xlIEFwcCBzaW11bHRhbmVvdXNseSAKCiMjIFJlc291cmNlcwoKSGVyZSdzIGEgbGlzdCB3aXRoIGFsbCB0aGUgcmVzb3VyY2VzIHRvIGxlYXJuIG1vcmUgYWJvdXQgYWxsIHRoZSB0b3BpY3MhCgoqIFtNaW5pbWFsIEFQSXNdKGh0dHBzOi8vbGVhcm4ubWljcm9zb2Z0LmNvbS9lbi11cy9hc3BuZXQvY29yZS90dXRvcmlhbHMvbWluLXdlYi1hcGk/dmlldz1hc3BuZXRjb3JlLTcuMCZ0YWJzPXZpc3VhbC1zdHVkaW8pCiogW1VzZXIgSldUc10oaHR0cHM6Ly9sZWFybi5taWNyb3NvZnQuY29tL2VuLXVzL2FzcG5ldC9jb3JlL3NlY3VyaXR5L2F1dGhlbnRpY2F0aW9uL2p3dC1hdXRobj9wcmVzZXJ2ZS12aWV3PXRydWUmdmlldz1hc3BuZXRjb3JlLTcuMCZ0YWJzPXdpbmRvd3MpCiogW09wZW5BUEldKGh0dHBzOi8vbGVhcm4ubWljcm9zb2Z0LmNvbS9lbi11cy9hc3BuZXQvY29yZS9mdW5kYW1lbnRhbHMvbWluaW1hbC1hcGlzL29wZW5hcGk/dmlldz1hc3BuZXRjb3JlLTcuMCkKKiBbTlN3YWddKGh0dHBzOi8vbGVhcm4ubWljcm9zb2Z0LmNvbS9lbi11cy9hc3BuZXQvY29yZS90dXRvcmlhbHMvZ2V0dGluZy1zdGFydGVkLXdpdGgtbnN3YWc/dmlldz1hc3BuZXRjb3JlLTcuMCZ0YWJzPXZpc3VhbC1zdHVkaW8pCiogWy5ORVQgU3RhbmRhcmRdKGh0dHBzOi8vbGVhcm4ubWljcm9zb2Z0LmNvbS9lbi11cy9kb3RuZXQvc3RhbmRhcmQvbmV0LXN0YW5kYXJkP3RhYnM9bmV0LXN0YW5kYXJkLTItMCkKCg== readmeEtag: '"53da0b6829eff853567645f10be503bb9989a391"' readmeLastModified: Wed, 09 Nov 2022 19:04:42 GMT repositoryId: 555555610 description: Minimal API + OpenAPI Sample created: '2022-10-21T20:24:18Z' updated: '2024-08-21T07:52:52Z' language: C# archived: false stars: 7 watchers: 1 forks: 1 owner: laura-rodriguez logo: https://avatars.githubusercontent.com/u/16751130?v=4 license: GPL-3.0 repoEtag: '"59ce58355659bb5a5672a8a3a5d37c9d1d436eba2eda354664f5364b55b3f8d3"' repoLastModified: Wed, 21 Aug 2024 07:52:52 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/hirako2000/hirako-starter-kit v3: true repositoryMetadata: base64Readme: >- <p align="center">
  <a href="#"><img src="https://i.ibb.co/qY7BdkD/banner.png" /></a>
</p>

<p align="center">
  A slightly <em>opinionated</em> starter kit <br/>
  for building Node.js server.
</p>
<br/>
<br/>

<p align="center">
  <a href="#launch">Launch</a> •
  <a href="#tech">Tech</a> •
  <a href="#roadmap">Roadmap</a> •
</p>

<p align="center">
  <a href="https://travis-ci.org/hirako2000/hirako-starter-kit">
    <img src="https://travis-ci.org/hirako2000/hirako-starter-kit.svg?branch=master">
  </a>
  <a href="./License.md">
    <img src="https://img.shields.io/npm/l/slate.svg?maxAge=300">
  </a>
  <a href="https://www.codacy.com/manual/hirako2000/hirako-starter-kit">
    <img src="https://api.codacy.com/project/badge/Grade/1cca597cc64f4d9d97b2a1682ec65f7a">
  </a>
</p>

<br/>
<br/>

A start kit for building web apps using express, marko, mongoose and passport.

## Launch

```sh
git clone https://github.com/hirako2000/hirako-starter-kit.git
cd hirako-starter-kit
npm install # this will take a while depending on your internet bandwith
cp .env.example .env
npm start # Your default browser will open and load the root page
npm run start:dev # Same, but for dev, auto refresh on saving code change
```

## Design Philosophy

- Maximize productivity and ease in development
- Guarantee ease of deployment and performance in production
- Must be easy to adopt and not force you to change the way you write your code

## Another Express boilerplate

Maybe. I checked out a number of supposedly rock solid express boilerplate, many of them don't even build or start, issues raised with no activity. It happens with open source.. I can often fix the issues, but, one or more of those dissapointing surprise follow:

- Lack of decent server rendering set up, just API. I want an API and the ablity to render structued pages already, API boilerplate are fine, but I don't want two projects, especially when prototying
- Bloated with heavy front end libraries, I need performance
- Boated with poorly configured webpack bundler, I need performance
- No proper production structure and config, good for developement, but I need a production ready foundation, or at least reasonably configurable and not having to re-write my own production build pipeline
- Lack of unit tests, lack of means to unit test anything in there or my own code
- Hard to remove dependencies such as outdated or even unmaintained libraries

So yea, I built my own. Open sourcing it for the poor bears in need of a solution.

## Opinions

The boilerplate is not what one would call opinionated. Some decisions had to be made though. A totally opinionless starter kit leads to some gaps to fill.

When incorporating a design pattern, library or framework, the question is how difficult would it be to drop it or replace with better suited alternative. If the answer is `way too difficult`, then it's left off. Choices made so far can be formulated as below:

- Server side rendering is just plain more performant than SPA. Can't beat text for templating, can't beat a server processing power (if well provisioned), and it's SEO ready with no headache.
- RESTful API. It is needed, really. Express may not be the 'best' framework for API development, but it is a very well rounded, backed by the largest community and number of solid libraries. And, most node.js developers are familiar with it.
- LessCSS. Sass is superior, I know sassy. But it requires complied binaries to execute. It's OK for dev (to some degree) but it is NOT ok for devops. Any OS distribution should be able to build the project. Some native from source compilation on some distribs is a nightmare. Been there. the LessCSS processor is written in JavaScript. that's cross platform as it only needs node.js. If you want Sass, pull it yoursef.
- MarkoJS. While the community backing this framework isn't massive, it is production ready. Other templating solutions on node.js don't support streaming, and usually adppt specific syntax to get familiar with. Marko's synstax follow JS, and even support inline JavaScript.
- OpenAPI is the most evolved standard. And, let's get serious about doc, APIs need documentation, ideally generated from source. SwaggerUI and Redoc are beautiful doc templates
- Linting. Get a life. Just follow some standard and don't waste time arguing. If you don't like the standard followed in this project, jsut change the liniting and run --fix.
- Testing. I have to admit I don't know enough about front end testing to defend what I did here. My view is, testing is vital, but can be very expensive when bootstraing and asserting the surface/UI of applications. Frigile slow tests doesn't contribute to tddd. So I favour unit test covering business logic. If you need to test the dom, go ahead.
- pm2. The license is a bit obscure, but afaik if no modification is made to its source then it's fine. It is superior to `forever` and others. Just see for yourself.

## Prerequisites

- Node.js (v10+)
- MongoDB

## Tech

- Server framework with [Express](https://expressjs.com/)
- Server side templating and streaming rendering with [Marko](https://markojs.com/)
- Route structuring with [Lumie](https://github.com/Alex-Levacher/Lumie)
- Local Authentication with [Passport.js](http://www.passportjs.org/)
- Security with [Helmet](https://helmetjs.github.io/)
- CSS foundation with [Basscss](https://basscss.com/)
- [LessCSS](http://lesscss.org/) support
- Front end asset bundling and serving with [Lasso](https://github.com/lasso-js/lasso)
- Auto recompile and browser autorefresh with [browser-refresh](https://github.com/patrick-steele-idem/browser-refresh)
- Linting with [ESLint](https://github.com/eslint/eslint)
- HTTP Logging with [Morgan](https://github.com/expressjs/morgan)
- Auto generated [Open API](https://github.com/OAI/OpenAPI-Specification) and [ReDoc](https://github.com/Redocly/redoc)/[Swagger UI](https://github.com/swagger-api/swagger-ui) documentation
- Unit test context with [SuperTest](https://github.com/visionmedia/supertest) and cases with [Jest](https://github.com/facebook/jest)
- Continuous integration with [Travis CI](https://github.com/travis-ci/travis-ci)
- Production instance mgt with [pm2](https://github.com/Unitech/pm2)
- Devops deployment with [Docker Compose](https://github.com/docker/compose)

## Roadmap

- [x] Express and server side Templating/Rendering
- [x] Mongoose
- [x] supertest and Jest
- [x] Linting
- [x] Live browser refresh on js/css and template changes
- [ ] Stand up multiple routes
- [x] Lightweight CSS library bundled in
- [x] LessCSS support
- [ ] Style themes
- [x] API with structured routing
- [x] API doc
- [x] Local auth/login/signup
- [x] Auto-generate OPENAPI spec (with Swagger-ui) and Redoc
- [ ] CSRF
- [ ] i18n
- [ ] socket.io
- [x] Docker image

# HOWTO

## Add a dependency

It's all npm managed. You can add a server or browser (front end) dependency as below.

### Server

```sh
npm install --save lodash
# or
npm install --save-dev typescript
```

```javascript
const lodash = require("lodash");
// or
import lodash from "lodash";
```

### Browser

```sh
npm install --save lodash
```

Then edit `browser.json`

```json
{
  "dependencies": [
    "basscss/css/basscss.css",
    "basscss-btn/css/btn.css",
    "public/css/styles.less",
    "jquery/dist/jquery.slim.min.js",
    "lodash/path/to/lodash.js"
  ]
}
```

## Docker

You can also use docker for development. Make sure you run npm install on your development environment for linting and everything to work fine.

```sh
npm i
cp .env.example .env
```

Start the services

```sh
docker-compose up -d
```

View the logs

```sh
docker-compose logs -f
```

In case you install a npm module while developing, it should also be installed within docker container, to do this first install the module you want with simple `npm i module name`, then run it within docker container

```sh
docker-compose exec node npm i
```

If you make any changes to a project file, browser-refresh should automatically pick up and restart within docker (you can see this in the logs)

To run tests

```sh
docker-compose exec -e MONGODB_URL=mongodb://mongo:27017/noobjs_test node npm test
```

Note: Difference between exec and run, exec executes the command within the running container and run will spin up a new container to run that command. So if you only want to run the tests without docker-compose up, you may run `docker-compose run -e MONGODB_URL=mongodb://mongo:27017/my_app_test node npm test`

## License

MIT
 readmeEtag: '"dc06e87d81da1770f2c0ba8890f457c7a385ea04"' readmeLastModified: Wed, 17 Feb 2021 12:08:10 GMT repositoryId: 229742217 description: >- Hirako Starter Kit - Node.js server with Express, Marko, Mongoose, auto generated API doc created: '2019-12-23T11:49:50Z' updated: '2025-06-29T14:34:14Z' language: Marko archived: false stars: 7 watchers: 1 forks: 1 owner: hirako2000 logo: https://avatars.githubusercontent.com/u/1500712?v=4 license: MIT repoEtag: '"4dcebee413f0dc1bf091c1d1543cd3a256aef8f783ae72da3872ae5ef03281e0"' repoLastModified: Sun, 29 Jun 2025 14:34:14 GMT foundInMaster: true category: - Server - Server Implementations id: 53483f9bdb3947465e28a8eed1aeab92 - source: openapi3 tags repository: https://github.com/jayvynl/goctl-openapi v3: true id: cd58ead0537042867ae2097f082399d6 repositoryMetadata: base64Readme: >- Z29jdGwtb3BlbmFwaQo9PT0KClRoaXMgcHJvamVjdCBpcyBhIHBsdWdpbiBmb3IgW2dvY3RsXShodHRwczovL2dpdGh1Yi5jb20vemVyb21pY3JvL2dvLXplcm8vdHJlZS9tYXN0ZXIvdG9vbHMvZ29jdGwpLiBJdCdzIGFibGUgdG8gZ2VuZXJhdGUgW29wZW5hcGkgc3BlY2lmaWNhdGlvbiB2ZXJzaW9uIDNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjAuMy5tZCkgZmlsZSBmcm9tIFtnby1jdGwgYXBpXShodHRwczovL2dvLXplcm8uZGV2L2VuL2RvY3MvdHV0b3JpYWxzKSBmaWxlLgoKCiMjIyBGZWF0dXJlcwoKLSBnZW5lcmF0ZSBjb3JyZWN0IHNjaGVtYSBmb3IgYW55IGxldmVsIG9mIGVtYmVkZGVkIHN0cnVjdHVyZSB0eXBlLgotIGdlbmVyYXRlIGNvcnJlY3Qgc2NoZW1hIGZvciBjb21wbGljYXRlZCB0eXBlIGRlZmluaXRpb24gbGlrZSBgbWFwW3N0cmluZ11bXW1hcFtpbnRdW10qQXV0aG9yYC4KLSBwYXJzZSBwYXJhbWV0ZXIgY29uc3RyYWludHMgZnJvbSBbdmFsaWRhdGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1wbGF5Z3JvdW5kL3ZhbGlkYXRvcikgdGFnLgoKCiMjIyBJbnN0YWxsCgpUaGlzIHBsdWdpbidzIHZlcnNpb24gYW5kIGdvY3RsJ3MgdmVyc2lvbiBzaG91bGQgaGF2ZSB0aGUgc2FtZSBtYWpvciBhbmQgbWlub3IgdmVyc2lvbiwgaXQncyByZWNvbW1lbmRlZCB0byBpbnN0YWxsIHRoZSBtYXRjaGluZyB2ZXJzaW9uLiBJZiB2ZXJzaW9ucyBkb2Vzbid0IG1hdGNoLCBpdCBtYXkgbm90IHdvcmsgcHJvcGVybHkuCgpGb3IgZXhhbXBsZSwgaWYgeW91IHVzZSBnb2N0bCB2MS42LjMsIHRoZW4geW91IHNob3VsZCBpbnN0YWxsIHRoaXMgcGx1Z2luIHdpdGg6CgpgYGBzaGVsbApnbyBpbnN0YWxsIGdpdGh1Yi5jb20vamF5dnlubC9nb2N0bC1vcGVuYXBpQHYxLjYKYGBgCgojIyMgVXNhZ2UKCkhlbHAgbWVzc2FnZXMuCgpgYGBiYXNoClVzYWdlIGdvY3RsLW9wZW5hcGk6CiAgLWZpbGVuYW1lIHN0cmluZwogICAgICAgIG9wZW5hcGkgZmlsZSBuYW1lLCBkZWZhdWx0ICJvcGVuYXBpLmpzb24iLCAiLSIgd2lsbCBvdXRwdXQgdG8gc3Rkb3V0LgogIC1mb3JtYXQgc3RyaW5nCiAgICAgICAgc2VyaWFsaXphdGlvbiBmb3JtYXQsICJqc29uIiBvciAieWFtbCIsIGRlZmF1bHQgImpzb24iLgogIC1wcmV0dHkKICAgICAgICBwcmV0dHkgcHJpbnQgb2YganNvbi4KICAtdmVyc2lvbgogICAgICAgIHNob3cgdmVyc2lvbiBhbmQgZXhpdC4KYGBgCgpVc2FnZSBleGFtcGxlLgoKYGBgc2hlbGwKZ29jdGwgYXBpIHBsdWdpbiAtcGx1Z2luIGdvY3RsLW9wZW5hcGkgLWFwaSBleGFtcGxlLmFwaSAtZGlyIGV4YW1wbGUKYGBgCgpUYWtlIHRoZSBhcGkgZmlsZSBmcm9tIFtleGFtcGxlXShodHRwczovL2dpdGh1Yi5jb20vamF5dnlubC9nb2N0bC1vcGVuYXBpL2Jsb2IvbWFpbi9leGFtcGxlL2V4YW1wbGUuYXBpKSwgW3RoZSBnZW5lcmF0ZWQgb3BlbmFwaSBmaWxlXShodHRwczovL2dpdGh1Yi5jb20vamF5dnlubC9nb2N0bC1vcGVuYXBpL2Jsb2IvbWFpbi9leGFtcGxlL29wZW5hcGkuanNvbikgY2FuIGJlIHZpc3VhbGl6ZWQgYnkgW3N3YWdnZXIgZWRpdG9yXShodHRwczovL2VkaXRvci5zd2FnZ2VyLmlvLz91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2pheXZ5bmwvZ29jdGwtb3BlbmFwaS9tYWluL2V4YW1wbGUvb3BlbmFwaS5qc29uKS4K readmeEtag: '"adfc8248075e6dc3012076c4198e269fc8836157"' readmeLastModified: Thu, 28 Mar 2024 14:49:36 GMT repositoryId: 776884350 description: >- OpenAPI3 generator for [goctl](https://github.com/zeromicro/go-zero/tree/master/tools/goctl) created: '2024-03-24T17:56:45Z' updated: '2026-02-02T02:38:51Z' language: Go archived: false stars: 9 watchers: 1 forks: 2 owner: jayvynl logo: https://avatars.githubusercontent.com/u/34599950?v=4 license: MIT repoEtag: '"8b1373169f0ee483dfcd4ccd526577d6f5ac8c10538f5bd27ad2c09d9d8684a1"' repoLastModified: Mon, 02 Feb 2026 02:38:51 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/endava/endava-hl7fhir-openapi v3: true repositoryMetadata: base64Readme: >- ![Endava Logo](res/endava_logo_329x128.jpg)


# Endava HL7 FHIR OpenAPI Demo
Matjaz Bravc, Senior Developer



# Introduction

This repo contains demonstration code to illustrate the use of the FIHR standard for healthcare data, developed by the HL7 organisation.  The demonstration shows how to build an OpenAPI to FHIR data using .NET.  The demonstration accompanies a series of blog posts on the [Endava Engineering Blog](https://www.endava.com/en/blog/Engineering) by Matjaz, entitled "Creating EHR to HL7 FHIR Integration - The Software Developer's Guide".

Table of Contents
=================

   * [Overview of the Project](#overview-of-the-project)
   * [What is FHIR?](#what-is-fhir)
     * [Why FHIR?](#why-fhir)
     * [Summary](#summarising-fhir)
   * [Using the Demonstration Code](#using-the-demonstration-code)

# Overview of the Project

![HL7 Logo](res/hl7-fhir-340x132.png)

When adopting FHIR, a common scenario is needing to convert your existing data into the FHIR model. For this demo, we will be building a OpenAPI which maps custom EHRs into FHIR Patient and Observation resources. Other resources were not implemented (yet).
For this purpose we will use the [.NET 5](https://dotnet.microsoft.com/download/dotnet/5.0),  [Firely .NET SDK](https://fire.ly/products/firely-net-sdk/) and public FHIR server [UHN_HAPI Server](http://hapi.fhir.org) (R4) which is regularly purged and reloaded with fixed test data.

    
# What is FHIR?
![](res/hapi-fhir.png)


**FHIR** (**F**ast **H**ealthcare **I**nteroperability **R**esources) is a new and emerging standard being developed under the auspices of the Health Level Seven (HL7) organization. Pronounced as 'Fire,' it was initially developed by Graham Grieve, who insisted FHIR be **open sourced**. At its core, FHIR is intended to be the next generation of healthcare interoperability. It tries to combine the best features of HL7 Version 2 and Version 3, in which Grieve was significantly involved.

## Why FHIR?
The current predominant method for exchanging clinical data between healthcare applications is known as HL7v2 and it presents serious challenges. Organizations today spend huge amount of money per HL7v2 interface, not to mention licensing fees to implement and use integration engines. Why then we need an integration engine? That's why:

**HL7v2 is a standard but it is NOT an open standard**. You need to be a member of the HL7 organization and pay fees before using the content in any commercial fashion, and HL7v2 is an ancient standard. It was developed in the late 80s when a lot of things we take for granted now didn't exist - mobile numbers, emails, APIs, Cloud, etc.

What resulted was a lot of "I'll just do it my way," causing an explosion of HL7v2 variants. With other words: "Welcome to the jungle!"

Given the above, **FHIR** does offer many improvements over existing standards:

- It's open source: This is a big deal and the first effort in making healthcare integration more transparent and accessible. Putting it out in the open has created a significant community including developers, vendors and enterprises.
- RESTful: REST-based design brings a significant amount of benefit, namely that an API that adheres to the principles of REST does not require the client to know anything about the structure of the API. Rather, the server needs to provide whatever information the client needs to interact with the service.
- Extensible: Extensibility under the RESTful context ensures that additions can be easily tacked on to cover specific use cases without impacting the core models.
- Composable: Composability ensures that almost any request can be cobbled together using core models or resources and associated extensions.
- Good documentation: Uniquely driven by the RESTful API approach, which enforces good documentation as a byproduct.
- Support for modern web standards: XML, JSON, HTTP, Atom, OAuth, REST - these are the underlying technologies that FHIR leverages. These are battle tested and have been proven at scale and under significant security requirements.
- Human readability: HL7 3.0 had a concept of a human readable version of the document or data being shared to ensure that developers or clinicians could still read the source data to eliminate any potential of misconfiguration or coding errors. FHIR borrows this concept as well. Every resource carries a human-readable text representation using html as a fallback display option. This is particularly important for complex clinical information where many systems take a simple textual or document-based approach.

## Summarising FHIR
FHIR is still a work in progress, but it is maturing quickly. It is not an event-based (triggered) protocol. In contrast, HL7v2 pushes data based on some event or activity in the source system. If someone was admitted to the hospital, the source system could be configured to push an ADT message based on the admission. FHIR is currently still a request based protocol.

FHIR has cemented its place as the next interoperability standard for clinical data exchange. Any standard has gaps, but they are neither unexpected nor insurmountable. The Standard, the official blog of the HL7 organization, has frequent updates and news as the FHIR standard evolves.  

FHIR R4, released in 2019, includes a normative base with backwards compatibility, which aims to give developers and organizations confidence that FHIR implementations they undertake will be supported for the foreseeable future and allow developers to implement FHIR more consistently and uniformly. While FHIR Release 4 is the current published version, development on Release 5 is underway with the hope of getting even closer to interoperability.

# Using the Demonstration Code

The demonstration code is a Microsoft Visual Studio project, which you can simply open with VS and use directly.

Enjoy!

## Prerequisites
- [Visual Studio](https://www.visualstudio.com/vs/community) 2019 16.8.1 or greater 
- [.NET 5.x](https://dotnet.microsoft.com/download/dotnet/5.0)

## API documentation
![](res/api-docs.jpg)

## Swagger UI
![](res/swagger.jpg)

## Tags, technologies and sources
- [Swagger](https://swagger.io/) - interactive API documentation
- [API-Docs](https://api-docs.io/) - friendly API documentation
- [Serilog](https://serilog.net/) - flexible, structured logger
- [FluentValidation](https://fluentvalidation.net/) - a popular .NET library for building strongly-typed validation rules
- [Firely .NET SDK](https://fire.ly/products/firely-net-sdk/) - the official HL7 FHIR SDK in .NET

## Further information
- [Extra documentation for FhirClient](https://docs.fire.ly/firelynetsdk/client.html)  
- [HL7 FHIR Restful API specification](https://www.hl7.org/fhir/http.html)
- [Firely .NET SDK FHIR Community](https://chat.fhir.org/#narrow/stream/179171-dotnet)
- [List of FHIR Test servers](https://confluence.hl7.org/display/FHIR/Public+Test+Servers)

## Contributing
Please refer to [CONTRIBUTING.md](CONTRIBUTING.md).

## Trademarks
HL7®, FHIR® and the flame Design mark are the registered trademarks of Health Level Seven International.
 readmeEtag: '"6bfcd4258a0feb2da1f5f09acbab6aaf9c3b5963"' readmeLastModified: Fri, 12 Jan 2024 09:57:46 GMT repositoryId: 375084335 description: >- This repo contains demonstration code to illustrate the use of the FIHR standard for healthcare data, developed by the HL7 organisation. created: '2021-06-08T16:59:18Z' updated: '2025-08-26T00:10:18Z' language: C# archived: false stars: 8 watchers: 1 forks: 4 owner: Endava logo: https://avatars.githubusercontent.com/u/19396140?v=4 license: Apache-2.0 repoEtag: '"2e6847dfc818fb8076df7fefb99e117ba54b70cbf1016091b19dacc9e6c86a3e"' repoLastModified: Tue, 26 Aug 2025 00:10:18 GMT foundInMaster: true category: Server Implementations id: 33de045bdbfe9b67453c38a60b903a2e - source: openapi3 tags repository: https://github.com/armsnyder/openapi-language-server v3: true id: 423ee8b0307b4ead6d192490008c9fbf repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIExhbmd1YWdlIFNlcnZlcgoKQW4gT3BlbkFQSSBsYW5ndWFnZSBzZXJ2ZXIgZm9yIFtMU1AgY29tcGF0aWJsZSBjb2RlCmVkaXRvcnMuXShodHRwczovL21pY3Jvc29mdC5naXRodWIuaW8vbGFuZ3VhZ2Utc2VydmVyLXByb3RvY29sL2ltcGxlbWVudG9ycy90b29scy8pCgo+IDp3YXJuaW5nOiBUaGlzIGlzIGJldGEgc29mdHdhcmUuIE1hbnkgZmVhdHVyZXMgYXJlIHN0aWxsIG1pc3NpbmcuIFNlZQo+IFtGZWF0dXJlc10oaHR0cHM6Ly9naXRodWIuY29tL2FybXNueWRlci9vcGVuYXBpLWxhbmd1YWdlLXNlcnZlcj90YWI9cmVhZG1lLW92LWZpbGUjZmVhdHVyZXMpCj4gYmVsb3cuCgpbIVthc2NpaWNhc3RdKGh0dHBzOi8vYXNjaWluZW1hLm9yZy9hL3Y3ZXRaYjgwSGJZa0tCUVVhM2RWU2VuUHouc3ZnKV0oaHR0cHM6Ly9hc2NpaW5lbWEub3JnL2EvdjdldFpiODBIYllrS0JRVWEzZFZTZW5QeikKCiMjIEZlYXR1cmVzCgpJIGNyZWF0ZWQgdGhpcyBsYW5ndWFnZSBzZXJ2ZXIgYmVjYXVzZSBJIG1hbnVhbGx5IGVkaXQgT3BlbkFQSS9Td2FnZ2VyIGZpbGVzLAphbmQgSSBuZWVkZWQgYSBxdWljayB3YXkgdG8ganVtcCBiZXR3ZWVuIHNjaGVtYSBkZWZpbml0aW9ucyBhbmQgcmVmZXJlbmNlcy4KCkkgdXNlClt5YW1sLWxhbmd1YWdlLXNlcnZlcl0oaHR0cHM6Ly9naXRodWIuY29tL3JlZGhhdC1kZXZlbG9wZXIveWFtbC1sYW5ndWFnZS1zZXJ2ZXIpCmZvciB2YWxpZGF0aW9uIGFuZCBjb21wbGV0aW9uLCBzbyB0aGVzZSBmZWF0dXJlcyBhcmUgbm90IGEgcHJpb3JpdHkgZm9yIG1lCnJpZ2h0IG5vdy4KCiMjIyBMYW5ndWFnZSBGZWF0dXJlcwoKLSBbeF0gSnVtcCB0byBkZWZpbml0aW9uCi0gW3hdIEZpbmQgcmVmZXJlbmNlcwotIFsgXSBDb2RlIGNvbXBsZXRpb24KLSBbIF0gRGlhZ25vc3RpY3MKLSBbIF0gSG92ZXIKLSBbIF0gUmVuYW1lCi0gWyBdIERvY3VtZW50IHN5bWJvbHMKLSBbIF0gQ29kZSBhY3Rpb25zCgojIyMgT3RoZXIgRmVhdHVyZXMKCi0gW3hdIFlBTUwgZmlsZXR5cGUgc3VwcG9ydAotIFsgXSBKU09OIGZpbGV0eXBlIHN1cHBvcnQKLSBbIF0gVlNDb2RlIGV4dGVuc2lvbgoKIyMgSW5zdGFsbGF0aW9uCgojIyMgRnJvbSBHaXRIdWIgUmVsZWFzZXMgKFJlY29tbWVuZGVkKQoKRG93bmxvYWQgdGhlIGxhdGVzdCByZWxlYXNlIGZyb20gW0dpdEh1YiByZWxlYXNlc10oaHR0cHM6Ly9naXRodWIuY29tL2FybXNueWRlci9vcGVuYXBpLWxhbmd1YWdlLXNlcnZlci9yZWxlYXNlcykuCgojIyMgVXNpbmcgR28KCmBgYGJhc2gKZ28gaW5zdGFsbCBnaXRodWIuY29tL2FybXNueWRlci9vcGVuYXBpLWxhbmd1YWdlLXNlcnZlckBsYXRlc3QKYGBgCgojIyBVc2FnZQoKIyMjIE5lb3ZpbSBDb25maWd1cmF0aW9uIEV4YW1wbGUKCkFzc3VtaW5nIHlvdSBhcmUgdXNpbmcgTmVvdmltIGFuZCBoYXZlIHRoZSBpbnN0YWxsZWQgb3BlbmFwaS1sYW5ndWFnZS1zZXJ2ZXIKYmluYXJ5IGluIHlvdXIgUEFUSCwgeW91IGNhbiB1c2UgdGhlIGZvbGxvd2luZyBMdWEgY29kZSBpbiB5b3VyIE5lb3ZpbQpjb25maWd1cmF0aW9uOgoKYGBgbHVhCiAgICB2aW0uYXBpLm52aW1fY3JlYXRlX2F1dG9jbWQoJ0ZpbGVUeXBlJywgewogICAgICBwYXR0ZXJuID0gJ3lhbWwnLAogICAgICBjYWxsYmFjayA9IGZ1bmN0aW9uKCkKICAgICAgICB2aW0ubHNwLnN0YXJ0IHsKICAgICAgICAgIGNtZCA9IHsgJ29wZW5hcGktbGFuZ3VhZ2Utc2VydmVyJyB9LAogICAgICAgICAgZmlsZXR5cGVzID0geyAneWFtbCcgfSwKICAgICAgICAgIHJvb3RfZGlyID0gdmltLmZuLmdldGN3ZCgpLAogICAgICAgIH0KICAgICAgZW5kLAogICAgfSkKYGBgCgpUaGlzIGlzIGp1c3QgYSBiYXNpYyB3b3JraW5nIGV4YW1wbGUuIFlvdSB3aWxsIHByb2JhYmx5IHdhbnQgdG8gZnVydGhlcgpjdXN0b21pemUgdGhlIGNvbmZpZ3VyYXRpb24gdG8geW91ciBuZWVkcy4K readmeEtag: '"4c23fd434c0c1cae60ca0382f0b6ef35752c8f59"' readmeLastModified: Mon, 24 Jun 2024 05:06:03 GMT repositoryId: 818968969 description: OpenAPI language server for LSP compatible code editors created: '2024-06-23T11:56:13Z' updated: '2026-02-05T17:38:31Z' language: Go archived: false stars: 17 watchers: 2 forks: 0 owner: armsnyder logo: https://avatars.githubusercontent.com/u/9969202?v=4 license: MIT repoEtag: '"cc80095067da6c48fd99cabc95ce3397f6f5522ea08be7af080cf55b36d93dc2"' repoLastModified: Thu, 05 Feb 2026 17:38:31 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/phogolabs/stride v3: true repositoryMetadata: base64Readme: >- IyBzdHJpZGUKQSBjb2RlIGdlbmVyYXRvciBmb3IgT3BlbkFQSSB2My54IEJFVEEuCgoqKlRoZSB0b29sIGlzIGRlcHJhY3RlZC4gSSBhbSByZWNvbW1lbmRpbmcgdXNpbmcgR1JQQyArIFBMRVguKioKCgpbIVtEb2N1bWVudGF0aW9uXVtnb2RvYy1pbWddXVtnb2RvYy11cmxdClshW0xpY2Vuc2VdW2xpY2Vuc2UtaW1nXV1bbGljZW5zZS11cmxdClshW0J1aWxkIFN0YXR1c11bYWN0aW9uLWltZ11dW2FjdGlvbi11cmxdClshW0NvdmVyYWdlXVtjb2RlY292LWltZ11dW2NvZGVjb3YtdXJsXQpbIVtHbyBSZXBvcnQgQ2FyZF1bcmVwb3J0LWltZ11dW3JlcG9ydC11cmxdCgojIyBPdmVydmlldwoKU3RyaWRlIGlzIGEgY29tbWFuZCBsaW5lIHRvb2wgZm9yIHJhcGlkIGFwcGxpY2F0aW9uIGRldmVsb3BtZW50IHdpdGggT3BlbkFQSSB2MwpzcGVjaWZpY2F0aW9uLgoKIyMgTW90aXZhdGlvbgoKVGhlcmUgYXJlIGEgbG90IG9mIHRvb2xzIGFyb3VuZCB0aGUgT3BlbkFQSSBpbml0aWF0aXZlLiBIb3dldmVyIG5laXRoZXIgb2YgdGhlbQpnaXZlcyBjb21wcmVoZW5zaXZlIHNvbHV0aW9uIGZvciByYXBpZCBhcHBsaWNhdGlvbiBkZXZlbG9wbWVudC4gVGhlIGdvYWwgaXMKdG8gcHJvdmlkZSBhIHRvb2wgdGhhdCB5b3UgY2FuIHVzZSB0byBnZW5lcmF0ZSBwcm9kdWN0aW9uIHJlYWR5IHNlcnZlcgphcHBsaWNhdGlvbnMuCgpEb2Vzbid0IG1hdHRlciB3aGV0aGVyIHlvdSBhcmUgc3RhcnRpbmcgZnJvbSBqdXN0IGNvbmNlcHQsIHByb3RvdHlwZSBvcgpidWlsZGluZyB5b3VyIHYxIHByb2R1Y3QuIFRoZSB0b29sIHByb3ZpZGVzIGZ1bmN0aW9uYWxpdHkgZm9yIGVkaXRpbmcgYW5kCnZpZXdpbmcgb2YgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLiBJdCBnZW5lcmF0ZXMgYSBwcm9kdWN0aW9uIHJlYWR5IHNjYWZmb2xkCnByb2plY3RzLiBBbHNvIGl0IGNhbiBzdGFydCBhIG1vY2sgc2VydmVyLiBTbyB5b3UgY2FuIHF1aWNrbHkgc3RhcnQgYnVpbGRpbmcKeW91ciBmcm9udC1lbmQgYXBwIHdpdGhvdXQgd2FpdGluZyBmb3IgeW91ciBhY3R1YWwgYmFjay1lbmQuCgpUaGFua3MgdG8gdGhlIGNvbnRyaWJ1dG9ycyBvZiB0aGUgZm9sbG93aW5nIHByb2plY3RzIHRoYXQgZW1wb3dlciBgc3RyaWRlYDoKLSBba2luLW9wZW5hcGldKGh0dHBzOi8vZ2l0aHViLmNvbS9nZXRraW4va2luLW9wZW5hcGkpCi0gW3N3YWdnZXItZWRpdG9yXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1lZGl0b3IpCi0gW3N3YWdnZXItdWldKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLXVpKQoKIyMgSW50cm9kdWN0aW9uCgpJdCBwcm92aWRlcyB0aGUgZm9sbG93aW5nIHN1YmNvbW1hbmRzOgoKYGBgYmFzaApOQU1FOgogICBzdHJpZGUgLSBPcGVuQVBJIHZpZXdlciwgZWRpdG9yLCBnZW5lcmF0b3IsIHZhbGlkYXRvciBhbmQgbW9ja2VyClVTQUdFOgogICBzdHJpZGUgW2dsb2JhbCBvcHRpb25zXQpDT01NQU5EUzoKICAgICBlZGl0ICAgICAgRWRpdCBhbiBPcGVuQVBJIHNwZWNpZmljYXRpb24gaW4gdGhlIGJyb3dzZXIKICAgICB2aWV3ICAgICAgU2hvd3MgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGluIHRoZSBicm93c2VyCiAgICAgbW9jayAgICAgIFJ1bnMgYSBtb2NrIHNlcnZlciBmcm9tIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbgogICAgIGdlbmVyYXRlICBHZW5lcmF0ZXMgYSBwcm9qZWN0IGZyb20gYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uCiAgICAgdmFsaWRhdGUgIFZhbGlkYXRlcyBhbiBPcGVuQVBJIHNwZWNpZmljYXRpb24KICAgICBoZWxwLCBoICAgU2hvd3MgYSBsaXN0IG9mIGNvbW1hbmRzIG9yIGhlbHAgZm9yIG9uZSBjb21tYW5kCgpPUFRJT05TOgogICAtLXZlcnNpb24sIC12ICBwcmludHMgdGhlIHZlcnNpb24KICAgLS1oZWxwLCAtaCAgICAgc2hvd3MgaGVscApgYGAKClJpZ2h0IG5vdyBgc3RyaWRlYCBzdXBwb3J0cyBvbmx5IGBnb2xhbmdgLiBUaGVyZSBhcmUgYSBmZXcgbGltaXRhdGlvbnMgdGhhdCB0aGUKZ2VuZXJhdG9yIGRvZXMgbm90IHN1cHBvcnQgZm9yIG5vdy4gVGhlIGZvbGxvd2luZyBmZWF0dXJlcyBhcmUgbm90IHN1cHBvcnRlZDoKCi0gSW5oZXJpdGFuY2UgYW5kIFBvbHltb3JwaGlzbQotIE9uZU9mLCBBbnlPZiwgQWxsT2YgYW5kIE5vdCAodGhlcmUgYXJlIHNvbWUgbGltaXRhdGlvbnMgZHVlIHRvIHRoZSBsYW5ndWFnZSBjb25zdHJhaW50cykKLSBMaW5rcwotIENhbGxiYWNrcwotIEF1dGhlbnRpY2F0aW9uCgpOb3RlIHRoYXQgdGhlIGNvZGUgZ2VuZXJhdGVkIGJ5IHRoZSBgZ29sYW5nYCBnZW5lcmF0b3IgY29tcGlsZXMgYW5kIHJ1biBvdXQgb2YKdGhlIGJveC4gSG93ZXZlciwgdGhlIHBhY2thZ2UKW3Jlc3RpZnldKGh0dHBzOi8vZ2l0aHViLmNvbS9waG9nb2xhYnMvcmVzdGlmeSkgdGhhdCB3aWxsIHN1cHBvcnQgdGhlCmdlbmVyYXRlZCBjb2RlIGlzIHN0aWxsIGluIGRldmVsb3BtZW50IHBoYXNlLgoKIyMgUm9hZCBtYXAKCi0gW3hdIEdvbGFuZyBnZW5lcmF0b3IgKGluIHRlc3RpbmcgcGhhc2UpCi0gWyBdIEVuYWJsZSBhIG1vY2sgc2VydmVyIGZvciBnaXZlbiBPcGVuQVBJIHNwZWNpZmljYXRpb24KLSBbeF0gRG93bmxvYWQgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBmcm9tIGRpZmZlcmVudCBzb3VyY2VzIChsb2NhbCwgczMsIGdpdCBhbmQgZXRjLikKLSBbeF0gU3VwcG9ydCBmb3IgRGljdGlvbmFyaWVzLCBIYXNoIE1hcHMgYW5kIEFzc29jaWF0aXZlIEFycmF5cyBpbiBHb2xhbmcKLSBbeF0gU3VwcG9ydCBmb3IgYGFwcGxpY2F0aW9uL3htbGAgYW5kIGBhcHBsaWNhdGlvbi94LXd3dy1mb3JtLXVybGVuY29kZWRgCi0gW3hdIEltcHJvdmUgdGhlIE9wZW5BUEkgdmFsaWRhdGlvbiByZXBvcnRzCi0gWyBdIEFsbG93IGltcGxlbWVudGF0aW9uIG9mIDNyZCBwYXJ0eSBnZW5lcmF0b3JzIGluIG90aGVyIGxhbmd1YWdlcyAodmlhIEdSUEMpCgojIyBJbnN0YWxsYXRpb24KClRoZXJlIGFyZSBhIGZldyB3YXlzIHRvIGluc3RhbGwgaXQ6CgojIyMjIEdpdEh1YgoKYGBgY29uc29sZQokIGdvIGdldCAtdSBnaXRodWIuY29tL3Bob2dvbGFicy9zdHJpZGUKJCBnbyBpbnN0YWxsIGdpdGh1Yi5jb20vcGhvZ29sYWJzL3N0cmlkZS9jbWQvc3RyaWRlCmBgYAoKIyMjIyBIb21lYnJldyAoZm9yIE1hYyBPUyBYKQoKYGBgY29uc29sZQokIGJyZXcgdGFwIHBob2dvbGFicy90YXAKJCBicmV3IGluc3RhbGwgc3RyaWRlCmBgYAoKIyMgQ29udHJpYnV0aW5nCgpXZSBhcmUgb3BlbiBmb3IgYW55IGNvbnRyaWJ1dGlvbnMuIEp1c3QgZm9yayB0aGUKW3Byb2plY3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9waG9nb2xhYnMvc3RyaWRlKS4KCltyZXBvcnQtaW1nXTogaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vcGhvZ29sYWJzL3N0cmlkZQpbcmVwb3J0LXVybF06IGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9waG9nb2xhYnMvc3RyaWRlCltsb2dvLWF1dGhvci11cmxdOiBodHRwczovL3d3dy5mcmVlcGlrLmNvbS9mcmVlLXZlY3Rvci9hYnN0cmFjdC1jcm9zcy1sb2dvLXRlbXBsYXRlXzExODU5MTkuaHRtCltsb2dvLWxpY2Vuc2VdOiBodHRwOi8vY3JlYXRpdmVjb21tb25zLm9yZy9saWNlbnNlcy9ieS8zLjAvCltjb2RlY292LXVybF06IGh0dHBzOi8vY29kZWNvdi5pby9naC9waG9nb2xhYnMvc3RyaWRlCltjb2RlY292LWltZ106IGh0dHBzOi8vY29kZWNvdi5pby9naC9waG9nb2xhYnMvc3RyaWRlL2JyYW5jaC9tYXN0ZXIvZ3JhcGgvYmFkZ2Uuc3ZnClthY3Rpb24taW1nXTogaHR0cHM6Ly9naXRodWIuY29tL3Bob2dvbGFicy9zdHJpZGUvd29ya2Zsb3dzL21haW4vYmFkZ2Uuc3ZnClthY3Rpb24tdXJsXTogaHR0cHM6Ly9naXRodWIuY29tL3Bob2dvbGFicy9zdHJpZGUvYWN0aW9ucwpbZ29kb2MtdXJsXTogaHR0cHM6Ly9nb2RvYy5vcmcvZ2l0aHViLmNvbS9waG9nb2xhYnMvc3RyaWRlCltnb2RvYy1pbWddOiBodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL3Bob2dvbGFicy9zdHJpZGU/c3RhdHVzLnN2ZwpbbGljZW5zZS1pbWddOiBodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2xpY2Vuc2UtTUlULWJsdWUuc3ZnCltsaWNlbnNlLXVybF06IExJQ0VOU0UK readmeEtag: '"bc31c41068e345d5a0a949c54d888a9474c66d12"' readmeLastModified: Fri, 30 Apr 2021 13:10:03 GMT repositoryId: 181135924 description: A code generator for OpenAPI v3.x created: '2019-04-13T07:22:39Z' updated: '2023-01-28T09:14:39Z' language: Go archived: true stars: 7 watchers: 1 forks: 1 owner: phogolabs logo: https://avatars.githubusercontent.com/u/15003293?v=4 license: MIT repoEtag: '"95d169fb8d9c059c2d3a68b366870fa3c954264e00dfe9595a60f2697a3ebd17"' repoLastModified: Sat, 28 Jan 2023 09:14:39 GMT foundInMaster: true category: Server id: fc3e66dec06633cdd45ad847e80ab5c6 - source: openapi3 tags repository: https://github.com/yasuflatland-lf/spring-boot-webflux-kotlin-coroutine v3: true id: 6654b73e4bbebeac7741bb5090d17536 repositoryMetadata: base64Readme: >- IyBzcHJpbmctYm9vdC13ZWJmbHV4LWtvdGxpbi1jb3JvdXRpbmUKIVtHaXRIdWIgV29ya2Zsb3cgU3RhdHVzICh3aXRoIGJyYW5jaCldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3lhc3VmbGF0bGFuZC1sZi9zcHJpbmctYm9vdC13ZWJmbHV4LWtvdGxpbi1jb3JvdXRpbmUvc3ByaW5nLWJvb3Qtd2ViZmx1eC1rb3RsaW4tY29yb3V0aW5lLnltbD9icmFuY2g9ZGV2ZWxvcCkKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2dpdGh1Yi95YXN1ZmxhdGxhbmQtbGYvc3ByaW5nLWJvb3Qtd2ViZmx1eC1rb3RsaW4tY29yb3V0aW5lL2JyYW5jaC9kZXZlbG9wL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj1LajliMEJRWmNWKV0oaHR0cHM6Ly9jb2RlY292LmlvL2dpdGh1Yi95YXN1ZmxhdGxhbmQtbGYvc3ByaW5nLWJvb3Qtd2ViZmx1eC1rb3RsaW4tY29yb3V0aW5lKQoKQSBzYW1wbGUgb2YgU3ByaW5nIGJvb3QgV2ViRmx1eCBhbmQgS290bGluIENvcm91dGluZSB3aXRoIEhhbmRsZXIgYW5kIFJvdXRlci4gSW4gdGhpcyBzYW1wbGUsIGltcGxlbWVudGluZyBUb2RvIG1vZGVsLCBoYW5kbGVyIGFuZCByb3V0ZXIgd2l0aCBkYXRhYmFzZSAoTXlTUUwpIHRlc3RpbmcuCgojIFJlcWlyZW1lbnRzCi0gSmF2YSAyMQotIERvY2tlciA0LjM3LjIgPj0KLSBncmFkbGUgOC4xMiA+PQoKIyBHZXR0aW5nIFN0YXJ0ZWQKUGxlYXNlIG1ha2Ugc3VyZSBEb2NrZXIgaXMgdXAgYW5kIHJ1bm5pbmcuCgojIyBQcm9kdWN0aW9uCmBgYGJhc2gKJCBtYWtlIHJ1bgpgYGAKYGFwcGxpY2F0aW9uLnltbGAgd2lsbCBiZSByZWZlcnJlZCBmb3IgdGhlIGNvbmZpZ3VyYXRpb24uIFRoaXMgY29tbWFuZCBkb2VzCi0gQnVpbGQgRG9ja2VyIGltYWdlIHdpdGggdGhlIGltcGxlbWVudGF0aW9uCi0gU3BpbiB1cCBEYXRhYmFzZSBhbmQgU3ByaW5nYm9vdCBhcHBsaWNhdGlvbiAodGhpcyBhcHBsaWNhdGlvbikgYnkgYGRvY2tlci1jb21wb3NlYAoKIyMgRGV2ZWxvcG1lbnQKYGBgYmFzaAokIG1ha2UgZGV2REIKJCBtYWtlIGRldkJvb3QKYGBgCmBhcHBsaWNhdGlvbi1sb2NhbC55bWxgIHdpbGwgYmUgcmVmZXJyZWQgZm9yIHRoZSBjb25maWd1cmF0aW9uLgoKYG1ha2UgZGV2REJgIG9ubHkgc3BpbnMgdXAgRGF0YWJhc2UsIGFuZCBgbWFrZSBkZXZCb290YCBydW4gdGhpcyBhcHBsaWNhdGlvbiBieSBkZXZlbG9wbWVudCBtb2RlIHZpYSBgZ3JhZGxlYC4KCiMjIyBIb3cgdG8gcnVuIHRlc3QKYGBgYmFzaAokIGdyYWRsZSB0ZXN0IC0taW5mbwpgYGAKIyMgQ2xlYW4gdXAgZW52aXJvbm1lbnQKYGBgYmFzaAokIG1ha2UgY2xlYW4KYGBgCgpUaGlzIGNvbW1hbmQgY2xlYW5zIHVwIGRhdGFiYXNlLgoKIyBPcGVuQVBJCiMjIEhvdyB0byBhY2Nlc3MgU3dhZ2dlciBVSQpgYGBiYXNoCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC93ZWJqYXJzL3N3YWdnZXItdWkvaW5kZXguaHRtbApgYGAKCiMgVGlwcwoKIyMgTWFrZWZpbGUgY29tbWFuZHMKWW91IGNhbiBydW4gYWxsIGNvbW1hbmRzIGJ5IHVzaW5nIE1ha2VmaWxlLiBBbGwgYXZhaWxhYmxlIGNvbW1hbmRzIGlzIGRpc3BsYXllZCBieSB0eXBpbmcsCmBgYGJhc2gKJCBtYWtlIGhlbHAKYGBgCg== readmeEtag: '"911afce9e1f27db18cba49c156dbd2a2279f522d"' readmeLastModified: Sat, 01 Feb 2025 07:07:16 GMT repositoryId: 476884435 description: >- 🛠️ A sample of Spring boot WebFlux and Kotlin Coroutine with Handler and Router created: '2022-04-01T22:24:52Z' updated: '2025-03-08T09:56:46Z' language: Kotlin archived: false stars: 8 watchers: 0 forks: 1 owner: yasuflatland-lf logo: https://avatars.githubusercontent.com/u/2855663?v=4 repoEtag: '"52d9212420d596af03058cc2f952c7b103c78f3c3f666438712db85ab803bfb5"' repoLastModified: Sat, 08 Mar 2025 09:56:46 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/apideck-libraries/openapi-specs v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXNwZWNzCgpUaGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBmb3IgdGhlIEFwaWRlY2sgQVBJcy4KCjxpbWcgc3JjPSJodHRwOi8vdmFsaWRhdG9yLnN3YWdnZXIuaW8vdmFsaWRhdG9yP3VybD1odHRwczovL3NwZWNzLmFwaWRlY2suY29tL2NybS55bWwiPgo= readmeEtag: '"a365f19dadbbcd8f25a1ea0e20063f437174a2f8"' readmeLastModified: Mon, 29 Sep 2025 14:24:29 GMT repositoryId: 318430456 description: The OpenAPI specifications for the Apideck APIs. created: '2020-12-04T06:55:12Z' updated: '2026-02-05T17:22:05Z' language: null archived: false stars: 7 watchers: 4 forks: 3 owner: apideck-libraries logo: https://avatars.githubusercontent.com/u/73573473?v=4 license: MIT repoEtag: '"1a9e0435d37de62782e8b7220bb2ad74c520986f2984428a973e5f68d4807665"' repoLastModified: Thu, 05 Feb 2026 17:22:05 GMT foundInMaster: true category: - Testing - Parsers id: c977303d84eebe1aa3c23e0a1f6d548d - source: openapi3 tags repository: https://github.com/d-yoshi/redmine-openapi v3: true id: 0304cdf06000047bbe0a8072d0785b11 repositoryMetadata: base64Readme: >- IyBVbm9mZmljaWFsIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBmb3IgdGhlIFJlZG1pbmUgQVBJCgpbIVtDSV0oaHR0cHM6Ly9naXRodWIuY29tL2QteW9zaGkvcmVkbWluZS1vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZC15b3NoaS9yZWRtaW5lLW9wZW5hcGkvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sKQoKIyMgT3ZlcnZpZXcKClRoaXMgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZpbGUgd2FzIGNyZWF0ZWQgYnkgbWFudWFsIHNjcmFwaW5nIHRoZSBvZmZpY2lhbCBhcGkgZG9jcyBhbmQgZXhwZXJpbWVudGF0aW9uLgoKIyMgRG9jcwoKYGBgClJlZG1pbmU6IDUuMC54Ck9wZW5BUEk6IDMuMC4zCmBgYAoKaHR0cHM6Ly9kLXlvc2hpLmdpdGh1Yi5pby9yZWRtaW5lLW9wZW5hcGkvCgojIyBSZWZlcmVuY2UKCltSZWRtaW5lIEFQSSBPZmZpY2lhbCBEZXZlbG9wZXIgR3VpZGVdKGh0dHBzOi8vd3d3LnJlZG1pbmUub3JnL3Byb2plY3RzL3JlZG1pbmUvd2lraS9SZXN0X2FwaSkK readmeEtag: '"63703a7edc35369c81185297dd3bd2e4b61f0ee5"' readmeLastModified: Wed, 12 Jun 2024 04:15:44 GMT repositoryId: 490070672 description: Unofficial OpenAPI specification for the Redmine API created: '2022-05-08T22:36:30Z' updated: '2025-07-28T09:17:02Z' language: null archived: false stars: 12 watchers: 1 forks: 2 owner: d-yoshi logo: https://avatars.githubusercontent.com/u/22128066?v=4 license: MIT repoEtag: '"ba2f6e5ac0ef808b7b0a239ac111ad8a6b4cfe442dbe2fafdfbc5a90861bed18"' repoLastModified: Mon, 28 Jul 2025 09:17:02 GMT category: - Description Validators - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mimicry-tech/openapi_validator v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJdjNWYWxpZGF0b3IKCkEgc21hbGwgbGlicmFyeSBmb3IgdmFsaWRhdGluZyBPcGVuQVBJIHNwZWNzIGluZGVwZW5kZW50bHkgb2YgdGhlIFtTd2FnZ2VyIEVkaXRvcl0oaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pbykgLSBhbmQgaW5kZXBlbmRlbnRseSBvZiBKYXZhU2NyaXB0LgoKT3JpZ2luYWxseSBleHRyYWN0ZWQgZnJvbSBhIGJyYW5jaCBvZiBbbWltaWNyeV0oaHR0cHM6Ly9naXRodWIuY29tL21pbWljcnktdGVjaC9taW1pY3J5KSBhcyB0aGlzIG1pZ2h0IGJlIHVzZWZ1bCB0byBvdGhlcnMuCgojIyBJbnN0YWxsYXRpb24KClRoZSBwYWNrYWdlIGNhbiBiZSBpbnN0YWxsZWRieSBhZGRpbmcgYG9wZW5hcGl2M192YWxpZGF0b3JgIHRvIHlvdXIgbGlzdCBvZmRlcGVuZGVuY2llcyBpbiBgbWl4LmV4c2A6CgpgYGBlbGl4aXIKZGVmIGRlcHMgZG8KICBbCiAgICB7Om9wZW5hcGl2M192YWxpZGF0b3IsICJ+PiAwLjEuMCJ9CiAgXQplbmQKYGBgCgpEb2N1bWVudGF0aW9uIGNhbiBiZSBnZW5lcmF0ZWQgd2l0aCBbRXhEb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9lbGl4aXItbGFuZy9leF9kb2MpCmFuZCBwdWJsaXNoZWQgb24gW0hleERvY3NdKGh0dHBzOi8vaGV4ZG9jcy5wbSkuIE9uY2UgcHVibGlzaGVkLCB0aGUgZG9jcyBjYW4KYmUgZm91bmQgYXQgW2h0dHBzOi8vaGV4ZG9jcy5wbS9vcGVuYXBpX3ZhbGlkYXRvcl0oaHR0cHM6Ly9oZXhkb2NzLnBtL29wZW5hcGl2M192YWxpZGF0b3IpLgoKIyMgVXNhZ2UKClRoZSBwYWNrYWdlIHByb3ZpZGVzIHRoZSBvcGVuYXBpIHYzIHNjaGVtYSBhcyBhIHBsYWluIGVsaXhpciBzdHJ1Y3QsIHNvIHlvdSBjYW4gZWl0aGVyOgoKYGBgZWxpeGlyCm15X3NwZWNpZmljYXRpb24gPSAle30KCm15X3NwZWNpZmljYXRpb24gfD4gT3BlbkFQSXYzVmFsaWRhdG9yLnZhbGlkPygpCiMgdHJ1ZSAvIGZhbHNlCgpteV9zcGVjaWZpY2F0aW9uIHw+IE9wZW5BUEl2M1ZhbGlkYXRvci52YWxpZGF0ZSgpCiMgOm9rIHwgezplcnJvciwgW119CmBgYAoKVGhpcyB1c2VzIHRoZSBtb3N0IGV4Y2VsbGVudCBbYGV4X2pzb25fc2NoZW1hYF0oaHR0cHM6Ly9oZXgucG0vcGFja2FnZXMvZXhfanNvbl9zY2hlbWEpLCB5b3UgY2FuIGFsc28gZ3JhYiB0aGUgc2NoZW1hIGRpcmVjdGx5IGFuZCB1c2UgaXQgd2l0aCBgZXhfanNvbl9zY2hlbWFgcyBgVmFsaWRhdG9yYDoKCmBgYGVsaXhpcgphbGlhcyBPcGVuQVBJdjNWYWxpZGF0b3IuU2NoZW1hcy5PcGVuQVBJLlYzLCBhczogVjMKYWxpYXMgRXhKc29uLlNjaGVtYQphbGlhcyBFeEpzb25TY2hlbWEuVmFsaWRhdG9yCgpteV9zcGVjaWZpY2F0aW9uID0gJXsKICAib3BlbmFwaSIgPT4gIjMuMC4wIiwKICAjIC4uLgp9CgpWMy5zY2hlbWEoKSB8PiBTY2hlbWEucmVzb2x2ZSB8PiBWYWxpZGF0b3IudmFsaWQ/KG15X3NwZWNpZmljYXRpb24pCmBgYAoKCg== readmeEtag: '"b61daad631ed9b35074cb65a5a931d73228240b9"' readmeLastModified: Thu, 07 Sep 2023 22:19:04 GMT repositoryId: 385714891 description: A small wrapper for validating OpenAPIv3 schemata in plain elixir created: '2021-07-13T19:30:25Z' updated: '2023-12-18T14:58:12Z' language: Elixir archived: false stars: 6 watchers: 1 forks: 0 owner: mimicry-tech logo: https://avatars.githubusercontent.com/u/85833773?v=4 license: Apache-2.0 repoEtag: '"865e9620f963cc3c10c5149503c4d6ed6f485eb4c3827e288fd779122fca7c76"' repoLastModified: Mon, 18 Dec 2023 14:58:12 GMT foundInMaster: true category: Parsers id: fab14465507a3232925341d0ad4443d4 - source: openapi3 tags repository: https://github.com/apiboard/php-openapi v3: true id: b7ee0faa773471470fa4444a50560ed0 repositoryMetadata: base64Readme: >- IyBQSFAgT3BlbkFQSQoKT3BlbkFQSSBTcGVjaWZpY2F0aW9uIHBhcnNlciBmb3IgUEhQIDguIFN1cHBvcnRzIGJvdGggT0FTIDMuMCBhbmQgMy4xLgoKWyFbTGF0ZXN0IFZlcnNpb24gb24gUGFja2FnaXN0XShodHRwczovL2ltZy5zaGllbGRzLmlvL3BhY2thZ2lzdC92cHJlL2FwaWJvYXJkL3BocC1vcGVuYXBpLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9hcGlib2FyZC9waHAtb3BlbmFwaSkKIVtQSFAgZnJvbSBQYWNrYWdpc3RdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L3BocC12L2FwaWJvYXJkL3BocC1vcGVuYXBpP3N0eWxlPWZsYXQtc3F1YXJlKQohW0NJXShodHRwczovL2dpdGh1Yi5jb20vYXBpYm9hcmQvcGhwLW9wZW5hcGkvd29ya2Zsb3dzL0NJL2JhZGdlLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSkKCiMjIEZlYXR1cmVzCgotIFBhcnNlIE9wZW5BUEkgZmlsZXMgaW50byBhIFBIUCBvYmplY3QgdG8gaW50ZXJhY3Qgd2l0aCBpbiBjb2RlCi0gVmFsaWRhdGUgT3BlbkFQSSBmaWxlcyBhZ2FpbnN0IHRoZSBvZmZpY2lhbCBKU09OLXNjaGVtYSBkZXNjcmlwdGlvbnMKLSBSZXNvbHZlIGV4dGVybmFsIGFuZCBpbnRlcm5hbCByZWZlcmVuY2VzCgojIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKY29tcG9zZXIgcmVxdWlyZSBhcGlib2FyZC9waHAtb3BlbmFwaQpgYGAKCiMjIFVzYWdlCgpZb3UgY2FuIGludGVyYWN0IHdpdGggdGhpcyBsaWJyYXJ5IHRocm91Z2ggdGhlIGBPcGVuQVBJOjpjbGFzc2AgZGlyZWN0bHkuCgpgYGBwaHAKJG9wZW5BUEkgPSBuZXcgT3BlbkFQSSgpOwpgYGAKVGhpcyBjbGFzcyBvcHRpb25hbGx5IGFjY2VwdHMgYW4gaW1wbGVtZW50YXRpb24gb2YgYEFwaWJvYXJkXE9wZW5BUElcQ29udGVudHNcUmV0cmlldmVyOjpjbGFzc2Agd2hpY2ggd2lsbCBiZSB1c2VkIHRvIHJldHJpZXZlIHRoZSBmaWxlIGNvbnRlbnRzLiBCeSBkZWZhdWx0IHRoZSBsb2NhbCBmaWxlc3lzdGVtIHdpbGwgYmUgdXNlZCB0byByZXRyaWV2ZSBmaWxlIGNvbnRlbnRzLgoKIyMjIFBhcnNlCgpZb3UgY2FuIHBhcnNlIHRoZSBjb250ZW50cyBvZiBhIGZpbGUgYnkgcGFzc2luZyBpdHMgcGF0aCB0byBgcGFyc2UoKWAuIFRoaXMgd2lsbCBhdHRlbXB0IHRvIHJldHJpZXZlIHRoZSBmaWxlJ3MgY29udGVudHMgYW5kIHJlc29sdmUgYW55IGV4dGVybmFsIHJlZmVyZW5jZXMuCgpJdCByZXR1cm5zIGEgUEhQIG9iamVjdCB0aGF0IHJlcHJlc2VudHMgdGhlIE9BUyBkb2N1bWVudCBzdHJ1Y3R1cmUgdGhhdCBjYW4gYmUgdXNlZCBpbiBjb2RlLgpgYGBwaHAKJGRvY3VtZW50ID0gJG9wZW5BUEktPnBhcnNlKCcvcGF0aC90by9teS1vYXMuanNvbicpOwoKJGRvY3VtZW50LT5vcGVuYXBpKCk7IC8vIDMuMS4wCmBgYAoKIyMjIFZhbGlkYXRlCllvdSBjYW4gZGlyZWN0bHkgdmFsaWRhdGUgdGhlIGNvbnRlbnRzIG9mIGEgZmlsZSBhZ2FpbnN0IHRoZSBvZmZpY2lhbCBPcGVuQVBJIEpTT04tc2NoZW1hIGRlc2NyaXB0aW9ucy4gSXQgcmV0dXJucyBhbiBhcnJheSBvZiBwb3NzaWJsZSBlcnJvcnMgdGhhdCBvY2N1cmVkIGR1cmluZyB0aGUgdmFsaWRhdGlvbi4KCmBgYHBocAokZXJyb3JzID0gJG9wZW5BUEktPnZhbGlkYXRlKCcvcGF0aC90by9teS1vYXMueWFtbCcpOwpgYGAKCj4g4pqg77iPIFZhbGlkYXRpb24gZm9yIE9BUyAzLjEgZG9lcyBub3QgY2hlY2sgYW55IEpTT04gU2NoZW1hcyBpbiB5b3VyIE9wZW5BUEkgZG9jdW1lbnQgYmVjYXVzZSBpdCBhbGxvd3MgeW91IHRvIHVzZSBhbnkgSlNPTiBTY2hlbWEgZGlhbGVjdCB5b3UgY2hvb3NlIQoKIyMjIFJlc29sdmUKCllvdSBjYW4gcmVzb2x2ZSBleHRlcm5hbCBhbmQgaW50ZXJuYWwgcmVmZXJlbmNlcy4gSXQgcmV0dXJucyBhIFBIUCBvYmplY3Qgd2l0aCB0aGUgcmVzb2x2ZWQgY29udGVudHMuCgpgYGBwaHAKJGNvbnRlbnRzID0gJG9wZW5BUEktPnJlc29sdmUoJy9wYXRoL3RvL215LW9hcy5qc29uJyk7CgokZG9jdW1lbnQgPSBuZXcgQXBpYm9hcmRcT3BlbkFQSVxTdHJ1Y3R1cmVcRG9jdW1lbnQoJGNvbnRlbnRzKTsKYGBgCldoZW4gcmVzb2x2aW5nIHJlZmVyZW5jZXMgdGhlIGNvbnRlbnRzIHdpbGwgYmUgcmV0cmlldmVkIGZyb20gdGhlIGxvY2FsIGZpbGV5c3RlbSBieSBkZWZhdWx0LiBZb3UgY2FuIG92ZXJyaWRlIHRoZSB3YXkgZmlsZSBjb250ZW50cyBpcyByZXRyaWV2ZWQgYnkgcGFzc2luZyBhIGN1c3RvbSBjbGFzcyB0aGF0IGltcGxlbWVudHMgdGhlIGBBcGlib2FyZFxPcGVuQVBJXENvbnRlbnRzXFJldHJpZXZlcmAgaW50ZXJmYWNlLgoKYGBgcGhwCgokY3VzdG9tUmV0cmlldmVyID0gbmV3IE15Q3VzdG9tUmV0cmlldmVyKCk7CiRvcGVuQVBJID0gbmV3IE9wZW5BUEkoJGN1c3RvbVJldHJpZXZlcik7Cgokb3BlbkFQSS0+cmVzb2x2ZSgnL3BhdGgvdG8vbXktb2FzLmpzb24nKTsKYGBgCgpDaXJjdWxhciByZWZlcmVuY2VzIGFyZSByZXNvbHZlZCBhcyBhbiBpbnRlcm5hbCByZWZlcmVuY2UgYWZ0ZXIgcmVjdXJzaW5nIHR3aWNlLCB0aGlzIGlzIHRvIHByZXZlbnQgaW5maW5pdGUgcmVjdXJzaW9uLgoKIyMgTGljZW5zZQoKVGhlIE1JVCBMaWNlbnNlIChNSVQpLiBQbGVhc2Ugc2VlIFtMaWNlbnNlIEZpbGVdKExJQ0VOU0UpIGZvciBtb3JlIGluZm9ybWF0aW9uLgo= readmeEtag: '"f9c6a4d0ec68939a474e83dcf9f743f8f6ef2062"' readmeLastModified: Wed, 21 Feb 2024 20:55:11 GMT repositoryId: 553826449 description: OpenAPI Specification library for PHP 8. Supports both OAS 3.0 and 3.1. created: '2022-10-18T20:46:53Z' updated: '2025-11-30T14:41:51Z' language: PHP archived: false stars: 6 watchers: 2 forks: 1 owner: Apiboard logo: https://avatars.githubusercontent.com/u/110201247?v=4 license: MIT repoEtag: '"04782a4b9d90ff5b3257acc637099520c86fa5f447300abdb7fae1b50614f292"' repoLastModified: Sun, 30 Nov 2025 14:41:51 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/bump-sh/bump-ruby-cli v3: true repositoryMetadata: base64Readme: >- IyBCdW1wOjpDTEkKCioq4pqg77iPIERFUFJFQ0FUSU9OIFdBUk5JTkc6KiogVGhpcyByZXBvc2l0b3J5IGlzIG5vdyBkZXByZWNhdGVkIGluIGZhdm9yIG9mIG91ciBbbmV3IE5vZGUgYmFzZWQgQnVtcCBDTEldKGh0dHBzOi8vZ2l0aHViLmNvbS9idW1wLXNoL2NsaSkuIFBsZWFzZSBtaWdyYXRlIHRvIHYyLngrIHZlcnNpb24gb2Ygb3VyIENMSSB0byBlbmpveSBhbGwgZmVhdHVyZXMgb2YgQnVtcC5zaC4g4pqg77iPCgotIFsqKlJlYWQgdGhlIG5ldyBDTEkgYW5ub3VuY2VtZW50KipdKGh0dHBzOi8vaGVhZHdheWFwcC5jby9idW1wLWNoYW5nZWxvZy9uZXctY29tbWFuZC1saW5lLWludGVyZmFjZS1lcmEtMTk2MjI2KQotIFsqKlJlYWQgb3VyIEJ1bXAgQ0xJIGhlbHAgc2VjdGlvbioqXShodHRwczovL2hlbHAuYnVtcC5zaC9idW1wLWNsaSkKCl9JZiB5b3Ugc3RpbGwgd2FudCB0byB1c2UgdGhpcyBydWJ5IGdlbSwgcGxlYXNlIFtjb250YWN0IHVzXShtYWlsdG86aGVsbG9AYnVtcC5zaCkgdG8gZGlzY3VzcyBhbHRlcm5hdGl2ZXMgYW5kIHNoYXJlIHlvdXIgdXNlIGNhc2UuIFRoYW5rIHlvdSFfCgpUaGUgYGJ1bXAtY2xpYCBnZW0gcHJvdmlkZXMgYSBzaW1wbGUgY29tbWFuZCBsaW5lIGFjY2VzcyB0byB0aGUgQnVtcCAoaHR0cHM6Ly9idW1wLnNoKSBBUEkuCgojIyBJbnN0YWxsYXRpb24KCkFkZCB0aGlzIGxpbmUgdG8geW91ciBhcHBsaWNhdGlvbidzIEdlbWZpbGU6CgpgYGBydWJ5CmdlbSAnYnVtcC1jbGknCmBgYAoKQW5kIHRoZW4gZXhlY3V0ZToKCiAgICAkIGJ1bmRsZQoKT3IgaW5zdGFsbCBpdCB5b3Vyc2VsZiBhczoKCiAgICAkIGdlbSBpbnN0YWxsIGJ1bXAtY2xpCgojIyBVc2FnZQoKQnVtcCB0cmllcyB0byBpZGVudGlmeSB5b3VyIGZpbGUgc3BlY2lmaWNhdGlvbiBhbmQgZm9ybWF0IGF1dG9tYXRpY2FsbHkuIFlvdSBjYW4gZm9yY2UgaXQgYnkgdXNpbmcgdGhlIGAtLXNwZWNpZmljYXRpb25gIG9wdGlvbi4gSGVyZSBhcmUgdGhlIHN1cHBvcnRlZCB2YWx1ZXM6CgoqIGBvcGVuYXBpL3YyL2pzb25gCiogYG9wZW5hcGkvdjIveWFtbGAKKiBgb3BlbmFwaS92My9qc29uYAoqIGBvcGVuYXBpL3YzL3lhbWxgCiogYGFzeW5jYXBpL3YyL2pzb25gCiogYGFzeW5jYXBpL3YyL3lhbWxgCgpCb3RoIHRoZSBgLS1kb2NgIGFuZCBgLS10b2tlbmAgb3B0aW9ucyB1c2VkIGJlbG93IGNhbiBiZSBmb3VuZCBpbiB5b3VyIGRvY3VtZW50YXRpb24gc2V0dGluZ3MgcGFnZSBvbiBodHRwczovL2J1bXAuc2guIE5vdGUgdGhhdCB5b3UgY2FuIHJlcGxhY2UgYm90aCB0aGUgYC0tZG9jYCBhbmQgYC0tdG9rZW5gIG9wdGlvbiBieSBhbiBlbnZpcm9ubWVudCB2YXJpYWJsZS4gVGhpcyB3aWxsIGhlbHAgdG8ga2VlcCB0aG9zZSB2YWx1ZXMgc2VjcmV0OgoKLSBgLS1kb2NgIGNhbiBieSByZXBsYWNlZCBieSBgQlVNUF9JRGAuCi0gYC0tdG9rZW5gIGNhbiBieSByZXBsYWNlZCBieSBgQlVNUF9UT0tFTmAuCgojIyMgUHJldmlldwoKWW91IGNhbiBwcmV2aWV3IHlvdXIgZG9jdW1lbnRhdGlvbiBieSBjYWxsaW5nIHRoZSBgcHJldmlld2AgY29tbWFuZC4gQSB0ZW1wb3JhcnkgcHJldmlldyB3aWxsIGJlIGNyZWF0ZWQsIHdpdGggYSB1bmlxdWUgVVJMLiBUaGlzIHByZXZpZXcgd2lsbCBiZSBhdmFpbGFibGUgZm9yIDMwIG1pbnV0ZXMuIFlvdSBkb24ndCBuZWVkIGFueSBjcmVkZW50aWFscyB0byB1c2UgdGhpcyBjb21tYW5kLgoKUHJldmlldyBhIGRvY3VtZW50YXRpb246CgogICAgJCBidW5kbGUgZXhlYyBidW1wIHByZXZpZXcgcGF0aC90by95b3VyL2ZpbGUueW1sCgojIyMgVmFsaWRhdGUKClZhbGlkYXRlIHlvdXIgZmlsZSBhZ2FpbnN0IGl0cyBzcGVjaWZpY2F0aW9uOgoKICAgICQgYnVuZGxlIGV4ZWMgYnVtcCB2YWxpZGF0ZSBwYXRoL3RvL3lvdXIvZmlsZS55bWwgLS1kb2MgRE9DX0lEX09SX1NMVUcgLS10b2tlbiBET0NfVE9LRU4KCiMjIyBEZXBsb3kKCkRlcGxveSB0aGUgZmlsZSBhcyB0aGUgY3VycmVudCB2ZXJzaW9uIG9mIHRoZSBkb2N1bWVudGF0aW9uOgoKICAgICQgYnVuZGxlIGV4ZWMgYnVtcCBkZXBsb3kgcGF0aC90by95b3VyL2ZpbGUueW1sIC0tZG9jIERPQ19JRF9PUl9TTFVHIC0tdG9rZW4gRE9DX1RPS0VOCgpBdXRvbWF0aWNhbGx5IGNyZWF0ZSBhIGRvY3VtZW50YXRpb24gaW5zaWRlIGEgaHViIGFuZCBkZXBsb3kgaXQ6CgogICAgJCBidW5kbGUgZXhlYyBidW1wIGRlcGxveSBwYXRoL3RvL3lvdXIvZmlsZS55bWwgLS1hdXRvLWNyZWF0ZSAtLWRvYyBET0NfU0xVRyAtLWh1YiBIVUJfSURfT1JfU0xVRyAtLXRva2VuIEhVQl9UT0tFTgoKIyMgRGV2ZWxvcG1lbnQKCkFmdGVyIGNoZWNraW5nIG91dCB0aGUgcmVwbywgcnVuIGBiaW4vc2V0dXBgIHRvIGluc3RhbGwgZGVwZW5kZW5jaWVzLiBUaGVuLCBydW4gYHJha2Ugc3BlY2AgdG8gcnVuIHRoZSB0ZXN0cy4gWW91IGNhbiBhbHNvIHJ1biBgYmluL2NvbnNvbGVgIGZvciBhbiBpbnRlcmFjdGl2ZSBwcm9tcHQgdGhhdCB3aWxsIGFsbG93IHlvdSB0byBleHBlcmltZW50LgoKIyMgQ29udHJpYnV0aW5nCgpCdWcgcmVwb3J0cyBhbmQgcHVsbCByZXF1ZXN0cyBhcmUgd2VsY29tZSBvbiBHaXRIdWIgYXQgaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvYnVtcC1jbGkuIFRoaXMgcHJvamVjdCBpcyBpbnRlbmRlZCB0byBiZSBhIHNhZmUsIHdlbGNvbWluZyBzcGFjZSBmb3IgY29sbGFib3JhdGlvbiwgYW5kIGNvbnRyaWJ1dG9ycyBhcmUgZXhwZWN0ZWQgdG8gYWRoZXJlIHRvIHRoZSBbQ29udHJpYnV0b3IgQ292ZW5hbnRdKGh0dHA6Ly9jb250cmlidXRvci1jb3ZlbmFudC5vcmcpIGNvZGUgb2YgY29uZHVjdC4KCiMjIExpY2Vuc2UKClRoZSBnZW0gaXMgYXZhaWxhYmxlIGFzIG9wZW4gc291cmNlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgW01JVCBMaWNlbnNlXShodHRwOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUKS4KCiMjIENvZGUgb2YgQ29uZHVjdAoKRXZlcnlvbmUgaW50ZXJhY3RpbmcgaW4gdGhlIEJ1bXA6OkNMSSBwcm9qZWN04oCZcyBjb2RlYmFzZXMsIGlzc3VlIHRyYWNrZXJzLCBjaGF0IHJvb21zIGFuZCBtYWlsaW5nIGxpc3RzIGlzIGV4cGVjdGVkIHRvIGZvbGxvdyB0aGUgW2NvZGUgb2YgY29uZHVjdF0oaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvYnVtcC1jbGkvYmxvYi9tYXN0ZXIvQ09ERV9PRl9DT05EVUNULm1kKS4K readmeEtag: '"0af8faf032d816a1f99226bc75094cd719933f86"' readmeLastModified: Wed, 07 Jul 2021 08:40:21 GMT repositoryId: 108739848 description: >- Old bump.sh ruby CLI - Please head to our new CLI https://github.com/bump-sh/cli created: '2017-10-29T14:10:45Z' updated: '2023-01-28T07:46:06Z' language: Ruby archived: true stars: 6 watchers: 1 forks: 3 owner: bump-sh logo: https://avatars.githubusercontent.com/u/33217836?v=4 license: MIT repoEtag: '"be2760393905516a574b147e95e0c0770ac4e73a74e250b6e9c81dbd8200542e"' repoLastModified: Sat, 28 Jan 2023 07:46:06 GMT foundInMaster: true category: Server Implementations id: 36740d58dd6e9e8806db6ebe24f79fc3 - source: openapi3 tags repository: https://github.com/criteria-labs/criteria-api-tools v3: true id: 73352c1e6b52001db4a01daf9eb0c83e repositoryMetadata: repositoryId: 585077559 description: Tools for working with the JSON Schema and OpenAPI specifications. created: '2023-01-04T09:03:35Z' updated: '2025-03-18T07:50:04Z' language: TypeScript archived: false stars: 7 watchers: 1 forks: 2 owner: criteria-labs logo: https://avatars.githubusercontent.com/u/88253836?v=4 license: MIT repoEtag: '"822ff3636be84311305c7a661f2f0dbff1789638e58fbdcabd7291a309ff1862"' repoLastModified: Tue, 18 Mar 2025 07:50:04 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/mikeparcewski/aws-openapi v3: true repositoryMetadata: base64Readme: >- # AWS + OpenAPI

A fast, massively scalable sample CRUD application with simple eventing that harnesses the power of AWS and OpenAPI.

### Table of Contents

* [What This Is](#what-this-is)
* [What You'll Need](#what-youll-need)
* [Setting It Up](#setting-it-up)
* [Making Sure It Works](#making-sure-it-works)
* [How It Works](#how-it-works)
  * [How to Customize](#srcmainresourcesopenapiyaml)
* [The Test Harness](#the-test-harness)
* [Taking It Further](#taking-it-further)
* [Cleaning Up](#cleaning-up)

### What This Is

Leveraging AWS, OpenAPI and BDD testing methodologies this a fast, massively scalable API created entirely through configuration, leveraging each of the technologies to simplify development and deployment.

* Amazon Web Services
  * [API Gateway](https://aws.amazon.com/api-gateway/) - To manage resource requests to AWS Lambda or DynamoDB
  * [AWS Lambda](https://aws.amazon.com/lambda) - To handle transformation of data
  * [DynamoDB](https://aws.amazon.com/dynamodb/) - A multi-master (e.g. read/write), regionally scalable
  * [AWS SNS](https://aws.amazon.com/sns/) - Alerting you in real time to changes in data
* [OpenAPI 3.x](https://www.openapis.org/) with AWS extensions (for deployment)
* [BDD For All](https://github.com/accenture/bdd-for-all) - A behavorial driven development grammar/test harness from Accenture

![AWS Diagram of Solution](content/diagram.png)

> Add Route 53 to route traffic based on best consumer region - https://medium.com/fintech-studios-engineering/automatic-multi-region-api-failover-and-geo-proximity-routing-on-aws-271f57752c1b

## What You'll Need

To run this application and create other API's you'll need the following installed...

* Modern IDE & Environment - Supports Maven, Java 8, NodeJS & Python
* AWS CLI - https://aws.amazon.com/cli/
* AWS CloudFormation Template Flip - https://github.com/awslabs/aws-cfn-template-flip

## What You'll Be Doing

Creating and testing a fast, and multi-region CRUD with no code using documentation and configuration.

**Features**
* CRUD API based on OpenAPI 3.x schema (either example one, or one you write)
* Launches in two regions, but can scale to all easily
* For reduced latency, all resources for an API exist within same region
  * All in-region resources resources configured to scale as well
* Backed by a global DynamoDB instance which makes each region instance multi-master read/write
* Change notifications sent via real time VIA SMS
* No code, other than reusable lambdas that could be used for any API.

## Setting It Up

First we need to generate the configuration.  In src/main/resources/ we have 3 YAML files...

* **aws-resources.yaml** - this outlines most of the components of the cloud formation stack
* **openapi.yaml** - this is the openapi spec for the service
* **aws-extensions.yaml** - these are openapi extensions from AWS that describe the API Gateway deployment

You can learn more about these in the [How This all Works](#how-this-all-works) section.  For now, let's get started.

* Go to the root of this projects directory
* First we need to convert our YAML configurations to AWS JSON.  We'll us the cfn-flip library to do this.
  ```sbtshell
  cfn-flip -j src/main/resources/aws-resources.yaml src/main/resources/aws-resources.json
  ```
  And...
  ```sbtshell
  cfn-flip -j src/main/resources/api-extensions.yaml src/main/resources/api-extensions.json
  ```
  Finally...
  ```sbtshell
  cfn-flip -j src/main/resources/openapi.yaml src/main/resources/openapi.json
  ```
* Next, we'll generate the final cloudformation template from these...
  ```sbtshell
  mvn clean generate-test-resources
  ```

And now we can start launching the stack...

```sbtshell
aws cloudformation create-stack \
  --stack-name ItemsExample \
  --template-body file:///$PWD/target/aws-config.json \
  --capabilities CAPABILITY_IAM \
  --region us-east-1
```

This creates the database, lambdas, and api gateway configuration and publishes demo endpoints in the us-east-1 region.

> You should jump to the console at this point - https://console.aws.amazon.com/cloudformation/home?region=us-east-1#/stacks?filter=active - and make sure it completes without error.  You can also check via the command line using `aws cloudformation describe-stacks --stack-name ItemsExample --region us-east-1` where you should see `"StackStatus": "CREATE_COMPLETE"`

Now, let's do the same for us-west-1...

```sbtshell
aws cloudformation create-stack \
  --stack-name ItemsExample \
  --template-body file:///$PWD/target/aws-config.json \
  --capabilities CAPABILITY_IAM \
  --region us-west-1
```
> Again jump to the console at this point - https://console.aws.amazon.com/cloudformation/home?region=us-west-1#/stacks?filter=active - and make sure it completes without error or check via the command line using the `aws cloudformation describe-stacks --stack-name ItemsExample --region us-west-1` command.

Once both stacks have been created, we want to make sure any database changes are synched between regions.  Since we can't specify global tables in the cloudformation template (as of Feb '19), we just need to issue the following command to bring the east/west databases together...

```sbtshell
aws dynamodb create-global-table \
  --region us-east-1 \
  --replication-group RegionName=us-east-1 RegionName=us-west-1 \
  --global-table-name ItemsExampleTable
```

To validate it worked, check out https://console.aws.amazon.com/dynamodb/home?region=us-east-1#tables:selected=ItemsExampleTable;tab=globaltables

There you should see lines for US East & West in the Global Table regions list.

### Making Sure It Works

In addition to writing an OpenAPI spec, we also write acceptance tests (BDD).  Let's see if the application works by executing those tests...

* First let's get the url's.  The easiest way... 
  ```sbtshell
  { \
  aws cloudformation describe-stack-resources \
    --stack-name ItemsExample \
    --region us-east-1 \
    | grep -C 5 "ApiGateway::RestApi" \
    | grep PhysicalResourceId \
    | sed 's/^.*: //; s/\"//g; s/^/host - https:\/\//; s/,/\.execute-api\.us-east-1\.amazonaws\.com\/demo/g' \
  ; \
  aws cloudformation describe-stack-resources \
    --stack-name ItemsExample \
    --region us-west-1 \
    | grep -C 5 "ApiGateway::RestApi" \
    | grep PhysicalResourceId \
    | sed 's/^.*: //; s/\"//g; s/^/alternative - https:\/\//; s/,/\.execute-api\.us-west-1\.amazonaws\.com\/demo/g' \
  } \
  | cat
  ```
  > Your output should look something like...
  ```sbtshell
  host - https://uoa9j84ufh.execute-api.us-east-1.amazonaws.com/demo/items
  alternative - https://39gzfyhlv9.execute-api.us-west-1.amazonaws.com/demo/items
  ```
* Now open src/test/resources/application.conf and take "host" from the command above (just the URL part) and put it where it says USEAST1 below.  Then take the "alternative" output from above and replace the USWEST1 location as shown below.
  ```scala
  bddcore {
  
    request {
  
      server {
        host = "USEAST1"
      }
  
      userAgent = "AWS-OPENAPI-SHOWCASE"
  
    }
  
    vars {
  
      Alternative {
  
        Region = "USWEST1"
  
      }
  
    }
  
  }
  ```
* Now for the SMS notifications, we'll add our phone.  First, we need to know the ARN...
  ```sbtshell
  aws cloudformation describe-stack-resources \
    --stack-name ItemsExample \
    --region us-east-1 \
    | grep -C 5 "AWS::SNS::Topic" \
    | grep PhysicalResourceId  \
    | sed 's/^.*: //; s/\"//g; s/,//g;'  
  ```
  > Your output should look something like...
  ```sbtshell
  arn:aws:sns:us-east-1:XXXXXXXXXXXX:ItemsExample-ChangeEventTopic-Q1W7ED1AEHQK
  ```
* Now we'll register using the arn (the output above) and your cell number
  ```sbtshell
  aws sns subscribe \
  --topic-arn <ARN_FROM_ABOVE> \
  --region us-east-1 \
  --protocol sms \
  --notification-endpoint <YOUR_PHONE>
  ```
  NOTE: Phone needs to include area code and us code (e.g. 15554444)
* Finally, let's execute the tests
  ```sbtshell
  mvn test
  ```
  
If everything looks good, all the tests will past and you should of got a bunch of text messages.  Now you can spin up postman or your favorite tool and start playing with the available endpoints.

> See [The Test Harness](#the-test-harness) section for more on what's happening here and check out the curl.log in the logs/ directory to grab some test curls to play with.

You can get a human readable version of the spec, to better understand what you have to work with, by...

* Going to https://editor.swagger.io/
* Click file -> "Import File"
* Select "/path/to/this/src/main/resources/openapi.yaml"

This will give you and idea of the model and endpoints.  In addition, checkout the logs/ directory.  If you ran the tests, there should be a curl.log.  You can copy those into your favorite tool as well to begin playing.

> To remove all the resources, see [Cleaning Up](#cleaning-up).

More fun, checkout the various consoles for each of the services we've launched...

* https://console.aws.amazon.com/cloudformation/home?region=us-east-1#/stacks (you could do us-west-1 too) - gives you a view of all the resources we just created and some management over them.
* https://console.aws.amazon.com/dynamodb/home?region=us-east-1 - checkout the global table you created and your additional configuration options
* https://console.aws.amazon.com/iam/home?region=us-east-1#/home - the roles and policies you've created
* https://console.aws.amazon.com/lambda/home?region=us-east-1#/functions - the nodejs endpoints you've launched
* https://console.aws.amazon.com/sns/v2/home?region=us-east-1#/home - the SNS topic you created
* https://console.aws.amazon.com/cloudwatch/home?region=us-east-1# - The logs all these new services are producing.

### How It Works

The magic starts with the configuration files mentioned way back in the beginning.  Let's begin with these...

#### src/main/resources/aws-resources.yaml

This file describes and configures the resources needed to run your new service in AWS and includes...

```yaml
  ## the actual table
  ItemsTable:
    Type: AWS::DynamoDB::Table
```

The above starts off the section for the applications database configuration.  It...

* Creates a table - This is the "ItemsTable" section and to make it easier for us to create a global table, we provide a name (e.g. "ItemsExampleTable").  If you wanted to repeat this setup for another application, the "TableName" attribute is actually the only thing you would need to change.
* Creates a scaling role and policy - The "ScalingRole" section allows this table to be managed by AWS Auto-Scaling services for both the DB itself and it's logs - https://docs.aws.amazon.com/AmazonECS/latest/developerguide/autoscale_IAM_role.html
* Creates read and write scalable targets - The "WriteCapacityScalableTarget", "WriteScalingPolicy", "ReadScalingTarget", and "ReadScalingPolicy" sections all tell AWS the scaling rules (e.g. how and when to scale) - https://docs.aws.amazon.com/amazondynamodb/latest/developerguide/AutoScaling.html
* Creates an event stream - "DataTableStream" creates a stream for table change events and registers our Lambda "" as the receiver

```yaml
  LambdaBasicExecutionRole:
    Type: AWS::IAM::Role
```

The above starts off the Lambda configuration.  Here we configure and deploy two lambdas using inline NodeJS code.  This section breaks down to...

* The execution role - "LambdaBasicExecutionRole" gives DynamoDB access to the lambdas, you can use this to configure access to any AWS service - https://docs.aws.amazon.com/lambda/latest/dg/intro-permission-model.html#lambda-intro-execution-role
* Ability to save JSON as a structured DynamoDB document - "SaveToDynamo" is a generic lambda that takes any JSON converts it to a DynamoDB friendly format, adds an ID, and then saves off to the table configured in the "ItemsTable" section.
* Ability to convert DynamoDB JSON to normalized JSON - "DynamoToJSON" is another generic lambda, that simply converts the DynamoDB format back to a JSON format in keeping with the API data model.
* Ability to update DynamoDB - "DynamoJSONUpdate", the last of the generic lambdas, converts a JSON request into an update statement.
* Ability to receive change events from dynamo - "HandleDynamoEvent" takes DynamoDB change events and publishes messages to SNS about what's happened.
* Finally, the "ChangeEventTopic" is the topic that the "HandleDynamoEvent" will use to publish it's messages.

```yaml
  APIGatewayRole:
    Type: AWS::IAM::Role
```

This section simply provides the API Gateway rights to use both DynamoDB and lambda functions so it can handle responses to API requests.

```yaml
  RestAPIv2:
    Type: AWS::ApiGateway::RestApi
```

Finally, the above, is the container for what will be the AWS Gateway deployment. Our OpenAPI document is appended here as part of the `mvn clean compile generate-resources` command.

#### src/main/resources/openapi.yaml

Nothing special here.  If you don't know about OpenAPI, we recommend you learn more - https://www.openapis.org/ - and come back.

You could replace this specification with any, as long as it's OpenAPI 3.x and you could generate a CRUD just for it. By default, this will handle any validations you describe with the spec.  It's pretty powerful for not writing any code!

You can even take this further by enabling transactions on DynamoDB - https://aws.amazon.com/blogs/aws/new-amazon-dynamodb-transactions/ - where you can configure transactional validations making it even more powerful without a line of code.

> Need to call AWS to enable for global tables, if you don't it won't work :)

#### src/main/resources/aws-extensions.yaml

More of AWS' secret sauce, this let's us describe how the endpoints interact with AWS or other services.  Typically this would be in the OpenAPI spec itself, but for readability we broke it out to a separate document and merge it with the same `mvn clean compile generate-resources` command that merges the OpenAPI spec into the AWS Resources document.

In this document there is a matching section for each endpoint, using same canonical structure, just with AWS specific extensions.  For each operation, we have some configuration...

**GET /item/{id}**

```yaml
      x-amazon-apigateway-integration:
        credentials: !GetAtt APIGatewayRole.Arn
        uri: !Sub arn:aws:apigateway:${AWS::Region}:lambda:path/2015-03-31/functions/${DynamoToJSON.Arn}/invocations
        responses:
          default:
            statusCode: "200"
        passthroughBehavior: "when_no_templates"
        httpMethod: "POST"
        type: "aws_proxy"
```

This section tells that any get request should be routed our "DynamoToJSON" described earlier.  API Gateway just acts as a proxy here, passing the request to the function and the response back to the consumer.

**DELETE /item/{id}**

```yaml
      x-amazon-apigateway-integration:
        credentials: !GetAtt APIGatewayRole.Arn
        uri: !Sub "arn:aws:apigateway:${AWS::Region}:dynamodb:action/DeleteItem"
        responses:
          default:
            statusCode: "200"
            responseTemplates:
              application/json: |-
                { "id": "$method.request.path.id" }
        requestTemplates:
          application/json: !Sub |
            {
              "TableName": "${ItemsTable}",
              "Key": {
                "id": {
                  "S": "$input.params('id')"
                }
              }
            }
        passthroughBehavior: when_no_templates
        httpMethod: "POST"
        type: aws
```

This endpoint puts a little more onus on the API Gateway.  It uses it's DynamoBD service integration to send a DELETE request directly to DynamoDB then respond with ID of the deleted item as confirmation.

**PUT /item/{id}**

```yaml
      x-amazon-apigateway-integration:
        credentials: !GetAtt APIGatewayRole.Arn
        uri: !Sub arn:aws:apigateway:${AWS::Region}:lambda:path/2015-03-31/functions/${DynamoJSONUpdate.Arn}/invocations
        responses:
          default:
            statusCode: "200"
        passthroughBehavior: "when_no_templates"
        httpMethod: "POST"
        type: "aws_proxy"
```

The above works like the GET, by proxying to "DynamoJSONUpdate" lambda and taking the path and the body to perform the update.

**GET /items/**

```yaml
      x-amazon-apigateway-integration:
        credentials: !GetAtt APIGatewayRole.Arn
        uri: !Sub arn:aws:apigateway:${AWS::Region}:lambda:path/2015-03-31/functions/${DynamoToJSON.Arn}/invocations
        responses:
          default:
            statusCode: "200"
        passthroughBehavior: "when_no_templates"
        httpMethod: "POST"
        type: "aws_proxy"
```

Like GET /items/{id}, this method works as a proxy to the "DynamoToJSON" lambda so it can return a list of requests.

**POST /items/**

```yaml
      x-amazon-apigateway-request-validator: "Validate body"
      x-amazon-apigateway-integration:
        credentials: !GetAtt APIGatewayRole.Arn
        uri: !Sub arn:aws:apigateway:${AWS::Region}:lambda:path/2015-03-31/functions/${SaveToDynamo.Arn}/invocations
        responses:
          default:
            statusCode: "200"
        passthroughBehavior: "when_no_templates"
        httpMethod: "POST"
        type: "aws_proxy"
```

Like the other endpoints that work as proxies, the POST endpoint does as well. You will notice one difference here...

```yaml
      x-amazon-apigateway-request-validator: "Validate body"
```

The above actually tells the API Gateway to validate the request payload before proxying it.  API Gateway uses the OpenAPI spec descriptors (required,min/max,allowed,etc...) to validate the payload automatically.  This is great if you're actually defining your field level requirements in the spec and is standard (e.g. NOT AWS specific) so the documentation, client libraries and other artifacts generated will also include this information.

> Amazon API Gateway Specs say things like patterns, exclusiveMinimum and other OAS properties don't work, but if you look at the tests, you'll see they actually do!

**Bringing It Together**

We use YAML because it's easier to read and has less markup then the JSON version of the documents.  To make merging easier, however, we convert to JSON.  This is why we have those `cfn-flip` statements.  There is some secret sauce there that we don't want to try to replicate ourselves when it comes to JSON <--> YAML conversion for CloudFormation templates.

When we combine this, with our simple JAVA program that merges all the docs as one we have our final CloudFormation template which describes all resources.  All those `aws cloudformation create-stack` commands really do is send that template off to AWS and telll it what region to push it too.

#### The Test Harness

Using the [BDD For All](https://github.com/Accenture/bdd-for-all) test harness from Accenture,  we created a few test cases to make sure we were truly cross-region and that some of the rules and validations we set up through the OAS Spec were actually working.  

Found in [src/test/resources/features/](src/test/resources/features/), we have test cases that cover invalid requests, checking for create/updates/deletes availablity cross region and more.

> Try writing your own tests, learn the grammar and other functionality available with the BDD For All library at - 

If you look in the logs directory after the tests are executed, you'll find a "curl.log", which has all the tests logged as cURLs that you can import into your favorite application to do some further testing.

### Taking it Further

What if we didn't want a simple NodeJS lambda backing our API, but instead wanted a Reactive JAVA application.  

That's easy!  Just type...

```
mvn clean spring-boot:run -Poas
```

and when you see...

```sbtshell
  .   ____          _            __ _ _
 /\\ / ___'_ __ _ _(_)_ __  __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
 \\/  ___)| |_)| | | | | || (_| |  ) ) ) )
  '  |____| .__|_| |_|_| |_\__, | / / / /
 =========|_|==============|___/=/_/_/_/
 :: Spring Boot ::        (v2.1.0.RELEASE)
```

Go to http://localhost:8080 and checkout your docs and somewhat functioning mocks.

#### What Did We Just Do?

The OpenAPITools project - https://github.com/OpenAPITools - has built on what the SmartBear - https://swagger.io/ - done to make some cool tools.  

The `-Poas` at the end of the previous mvn command, executes a maven plugin that exposes the OpenAPI Generator project - https://github.com/OpenAPITools/openapi-generator - which allows you to generate client libraries, server stubs, documentation and more for a variety of languages and platforms.

We generated a Spring Reactor shell, the code for which you can find at [src/main/java/io/github/mikeparcewski/demos/awsoas/generated](src/main/java/io/github/mikeparcewski/demos/awsoas/generated).  The shell pretty much contains all the plumbing.  All you really need to do implement, is...

* Add src/main/java/io/github/mikeparcewski/demos/awsoas/generated/api/ItemsApiDelegate.java to the .openapi-generator-ignore file (this keeps it from getting overwritten)
* Then edit ItemsApiDelegate to implement your business logic.

**Aside**

There are a lot of languages to choose from - https://github.com/OpenAPITools/openapi-generator#Overview - and by changing the following line to be the language/framework of your choice...

```xml
<generatorName>spring</generatorName>
```

and this...

```xml
<configOptions>
...
</configOptions>
```

To match the config for your language/framework option, you could build the app in almost any language your want.
  
## Cleaning Up

Remove the global table mapping...

```sbtshell
aws dynamodb update-global-table \
    --global-table-name ItemsExampleTable \
    --replica-updates 'Delete={RegionName=us-east-1,RegionName=us-west-1}' \
    --region us-east-1
```

Remove the US East stack...

```sbtshell
aws cloudformation delete-stack \
  --stack-name ItemsExample \
  --region us-east-1
```

Remove the US West stack...

```sbtshell
aws cloudformation delete-stack \
  --stack-name ItemsExample \
  --region us-west-1
``` readmeEtag: '"58f8c1b90f116c75f301810f526d21f0ce1dd0f1"' readmeLastModified: Tue, 24 Mar 2020 15:49:49 GMT repositoryId: 173618965 description: >- A massively scalable CRUD application built using AWS (CloudFormation) & OpenAPI 3.0 that focuses on configuration over code. created: '2019-03-03T19:17:31Z' updated: '2025-01-29T16:18:50Z' language: Gherkin archived: false stars: 6 watchers: 1 forks: 1 owner: mikeparcewski logo: https://avatars.githubusercontent.com/u/36518222?v=4 repoEtag: '"843c401a838cb16d6042ae6c6a399a05c71b72d36c70a7e6057d6e1122abbbf3"' repoLastModified: Wed, 29 Jan 2025 16:18:50 GMT foundInMaster: true category: Parsers id: e06d3b27ba90f952436feebcf0da6b60 - source: openapi3 tags repository: https://github.com/vanshg/weatherkit-openapi v3: true id: f84b7766e6993b11295307959969204a repositoryMetadata: base64Readme: >- IyBXZWF0aGVyS2l0LU9wZW5BUEkKT3BlbkFQSSBzcGVjIGZvciBBcHBsZSdzIFdlYXRoZXJLaXQKCiMjIFJlc291cmNlcwotIFRoZSAoY3VycmVudGx5LWluLUJldGEpIHNwZWM6IGh0dHBzOi8vZGV2ZWxvcGVyLmFwcGxlLmNvbS9kb2N1bWVudGF0aW9uL3dlYXRoZXJraXRyZXN0YXBpCi0gUHJpY2luZyBkZXRhaWxzOiBodHRwczovL2RldmVsb3Blci5hcHBsZS5jb20vd2VhdGhlcmtpdC9nZXQtc3RhcnRlZAotIFdXREMgd2Fsa3Rocm91Z2ggdmlkZW86IGh0dHBzOi8vZGV2ZWxvcGVyLmFwcGxlLmNvbS92aWRlb3MvcGxheS93d2RjMjAyMi8xMDAwMy8K readmeEtag: '"a34e288746a3a0fec5d92058e556662317dd2eb4"' readmeLastModified: Tue, 07 Jun 2022 23:53:05 GMT repositoryId: 500687327 description: OpenAPI spec for Apple's WeatherKit created: '2022-06-07T04:22:39Z' updated: '2024-07-16T09:38:31Z' language: null archived: false stars: 6 watchers: 1 forks: 2 owner: vanshg logo: https://avatars.githubusercontent.com/u/6133201?v=4 repoEtag: '"7d42a72887120575602ea708b812ff1e044a4723a4f833963e3a622e471898b0"' repoLastModified: Tue, 16 Jul 2024 09:38:31 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/mermade/swaggerplusplus v3: true repositoryMetadata: base64Readme: >- # swaggerplusplus

### A proposal for transitioning between Swagger 2.0 and OpenAPI 3.0.x

#### Version 1.0.0-rc4

The key words "MUST", "MUST NOT", "REQUIRED", "SHALL", "SHALL NOT", "SHOULD", "SHOULD NOT", "RECOMMENDED", "NOT RECOMMENDED", "MAY", and "OPTIONAL" in this document are to be interpreted as described in [BCP 14](https://tools.ietf.org/html/bcp14) [RFC2119](https://tools.ietf.org/html/rfc2119) [RFC8174](https://tools.ietf.org/html/rfc8174) when, and only when, they appear in all capitals, as shown here.

To aid the transition between Swagger 2.0 and OpenAPI 3.0.x, it is proposed to back-port the following features from the OpenAPI 3.0.0 specification (currently version 3.0.0) as [specification extensions](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#vendorExtensions) (formerly known as vendor extensions).

Tooling MAY make use of these features now, with minimal work required to support them in OpenAPI 3.0.x, definition authors MAY also use these features, knowing their API definition can be losslessly converted from **swaggerplusplus** to OpenAPI 3.0.x by any **swaggerplusplus**-compatible converter.

## Features

Path|Version 3.0.x Object|New Extension|Type|Description
---|---|---|---|---|
#/|servers|x-servers|[[Server Objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#server-object)]|When converting to OpenAPI 3.0.x, this array is used in place of any `servers` array converted from Swagger 2.0 metadata
#/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}|servers|x-servers|[[Server Objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#server-object)]|
#/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}/{[operation](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#operation-object)}|servers|x-servers|[[Server Objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#server-object)]|
#/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}/{[Operation](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#operation-object)}|trace|x-trace|[V2 Operation Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#operation-object)|This MUST be a Swagger 2.0 Operation Object, it MUST be treated as per any other Operation Object, for the `TRACE` HTTP method
#/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}|summary|x-summary|String|Also as per [Apiary extension](https://help.apiary.io/api_101/swagger-extensions/#x-summary-and-x-description)
#/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}|description|x-description|String|Also as per [Apiary extension](https://help.apiary.io/api_101/swagger-extensions/#x-summary-and-x-description)
#/[parameters](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#parameters-definitions-object) **OR** #/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}/parameters **OR** #/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}/{[Operation](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#operation-object)}/parameters|deprecated|x-deprecated|Boolean|Indicates the parameter is deprecated and SHOULD be transitioned out of use
#/paths/{[Path Item](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#path-item-object)}/[{Operation}](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#operation-object)|callbacks|x-callbacks|[[Callback Objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#server-object)]|
\* within [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object)|anyOf|x-anyOf|[[Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#schemaObject)]|Schema MUST be extracted and post-processed before being used for validation
\* within [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object)|oneOf|x-oneOf|[[Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#schemaObject)]|Schema MUST be extracted and post-processed before being used for validation
\* within [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object)|not|x-not|[Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#schemaObject)|Schema MUST be extracted and post-processed before being used for validation
\* within [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object)|required|x-required|Array|Where a property has been removed from `required` due to use of `x-anyOf`, `x-oneOf` or `x-not`, converters MUST merge these arrays when converting from **swaggerplusplus** to OpenAPI 3.0.x. When a converter converts from 3.0.x to **swaggerplusplus** it MUST remove any `required` properties hidden by `x-anyOf`, `x-oneOf` or `x-not` and move them into this array.
\* within [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object)|nullable|x-nullable|Boolean|Indicates whether the property is nullable or not.
\* within [Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#schema-object)|discriminator|x-discriminator|[Discriminator Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#discriminator-object)|A version 3.0.x `discriminator object` to replace the version 2.0 `discriminator` property.
\* within [Response Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#response-object)|links|x-links|Map {[Link Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#link-object)}|Links or references to reusable Link Objects
#/|components/links|x-links|Map {[Link Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#link-object)}|Contains reusable Link Objects
#/|components/callbacks|x-callbacks|Map {[Callback Object](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#callback-object)}|Contains reusable Callback Objects

## Conversion

Converters MUST validate the specification extensions are of the expected type, and that mandatory properties (if any) are present. Failing that, they MUST leave the specification extension unchanged.

## swaggerplusplus Tools

* [Converter/validator](https://github.com/mermade/swagger2openapi)
* [Rebilly Redoc](https://github.com/Rebilly/ReDoc) - supports a [subset](https://github.com/Rebilly/ReDoc/blob/master/docs/redoc-vendor-extensions.md) of **swaggerplusplus**

## swaggerplusplus in the wild

[Zappiti API](http://zappiti.com/api/zappiti-player-4k/swagger/)
 readmeEtag: '"a48c89f8924eb0e30ee63330a6f8d4185889f09c"' readmeLastModified: Thu, 28 Jun 2018 12:29:25 GMT repositoryId: 86082996 description: A proposal for transitioning between Swagger 2.0 and OpenAPI 3.0.x created: '2017-03-24T15:28:57Z' updated: '2025-07-06T22:14:13Z' language: null archived: false stars: 7 watchers: 3 forks: 0 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: Unlicense repoEtag: '"a418c11ce8ccc39f1409a32c4bf3358e1ff9168b7d951deeae3d00397219a584"' repoLastModified: Sun, 06 Jul 2025 22:14:13 GMT foundInMaster: true category: Parsers id: 0cf881514386d8ff2c26cf09e3967184 - source: - openapi3 tags - openapi31 tags repository: https://github.com/apiaddicts/openapi2soapui v3: true v3_1: true id: 9d4c1981e20f0c421731b1a708ab85cd repositoryMetadata: base64Readme: >- 
# 🛠️ OpenAPI2SoapUI ![Release](https://img.shields.io/badge/release-0.1.0-purple) ![Swagger](https://img.shields.io/badge/-soap-%23Clojure?style=flat&logo=swagger&logoColor=white) ![Java](https://img.shields.io/badge/java-%23ED8B00.svg?style=flat&logo=openjdk&logoColor=white)  [![License: LGPL v3](https://img.shields.io/badge/license-LGPL_v3-blue.svg)](https://www.gnu.org/licenses/lgpl-3.0) 

[API](./src/main/resources/static/api.yaml) to generate a SoapUI project from an OpenAPI Specification (fka Swagger Specification)

Given an OpenAPI Specification, either v2 or v3, a SoapUI project is generated with the _requests_ for each resource operation and a _test suite_. The response is the content of the SoapUI project in XML format to save as file and import into the SoapUI application.

### This repository is intended for :octocat: **community** use, it can be modified and adapted without commercial use. If you need a version, support or help for your **enterprise** or project, please contact us 📧 devrel@apiaddicts.org

[![Twitter](https://img.shields.io/badge/Twitter-%23000000.svg?style=for-the-badge&logo=x&logoColor=white)](https://twitter.com/APIAddicts) 
[![Discord](https://img.shields.io/badge/Discord-%235865F2.svg?style=for-the-badge&logo=discord&logoColor=white)](https://discord.gg/ZdbGqMBYy8)
[![LinkedIn](https://img.shields.io/badge/linkedin-%230077B5.svg?style=for-the-badge&logo=linkedin&logoColor=white)](https://www.linkedin.com/company/apiaddicts/)
[![Facebook](https://img.shields.io/badge/Facebook-%231877F2.svg?style=for-the-badge&logo=Facebook&logoColor=white)](https://www.facebook.com/apiaddicts)
[![YouTube](https://img.shields.io/badge/YouTube-%23FF0000.svg?style=for-the-badge&logo=YouTube&logoColor=white)](https://www.youtube.com/@APIAddictslmaoo)

# 🙌 Join the **OpenAPI2SoapUI** Adopters list 
📢 If OpenAPI2SoapUI is part of your organization's toolkit, we kindly encourage you to include your company's name in our Adopters list. 🙏 This not only significantly boosts the project's visibility and reputation but also represents a small yet impactful way to give back to the project.

| Organization  | Description of Use / Referenc |
|---|---|
|  [CloudAppi](https://cloudappi.net/)  | Apification and generation of microservices |
| [RSI](https://www.ruralserviciosinformaticos.com/)  | Generation of microservices  |

# 👩🏽‍💻  Contribute to ApiAddicts 

We're an inclusive and open community, welcoming you to join our effort to enhance ApiAddicts, and we're excited to prioritize tasks based on community input, inviting you to review and collaborate through our GitHub issue tracker.

Feel free to drop by and greet us on our GitHub discussion or Discord chat. You can also show your support by giving us some GitHub stars ⭐️, or by following us on Twitter, LinkedIn, and subscribing to our YouTube channel! 🚀

[!["Buy Me A Coffee"](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/apiaddicts)

# ⚙️ Functionalities

[Here](./src/main/resources/static/api.yaml) you can check the definition of the API Swagger to SoapUI

- Base64 Decoding of Open API Specification Content
- Parse Open API Specification Content into swagger-core representation as Java POJO
- Create SoapUI Project
  - Add OAuth 2.0 Profiles to SoapUI Project
  - Add new REST Service to SoapUI Project
    - Add Endpoints to REST Service
    - Set basePath to REST Service
    - Add Resources (Paths) to REST Service
      - Add Methods (Verbs / Operations) to each Resource
        - Set REST Request to each Method
          - Set Credentials (OAuth 2.0 Profile)
          - Set Parameters examples to REST Request
          - Set Request Body example to REST Request
          - Set Custom Headers to REST Request
  - Add new TestSuites to SoapUI Project for each Method
    - Add TestCases to TestSuite
      - Add Execution Test Step (REST Request)

Nomenclature used:
- SoapUI Project: {apiName}\_{apiVersion}
- REST Service: {apiName}
- Resource: {path}
- Method: {httpMethodInUppercase}
- Request: {defaultRequestName}
- Test Suite: {path}\_{httpMethodInUppercase}\_TestSuite
- Test Case (Default): Success\_TestCase
- Test Case: {testCaseName}\_TestCase
- Test Step: Execution\_{httpMethodInUppercase}\_TestStep

The variables are obtained from:
- apiName: property apiName of request body
- apiVersion: version defined in the 'info' section of the OpenAPI Spec
- path: each path defined in the OpenAPI Spec
- httpMethodInUppercase: each HTTP methods of paths defined in OpenAPI Spec
- testCaseName: each test case name defined in the property testCaseNames of request body

## Technology stack
### Overview

|Technology              |Description                 |
|------------------------|----------------------------|
|Core Framework          |Spring Boot 2               |

### Server - Backend

|Technology                                               |Description                                                                   |
|---------------------------------------------------------|------------------------------------------------------------------------------|
|[JDK 11](https://docs.oracle.com/en/java/javase/11/)                       |Java Development Kit                                                          |
|[Spring Boot 2](https://spring.io/projects/spring-boot)  |Framework to ease the bootstrapping and development of new Spring Applications|
|[Maven 3](https://maven.apache.org)                      |Dependency Management                                                         |
|[Tomcat 9](https://tomcat.apache.org)                    |Server deploy WAR                                                             |

###  Libraries and Plugins
|Technology              |Description                 |
|------------------------|----------------------------|
|[Lombok](https://projectlombok.org/) |Never write another getter or equals method again, with one annotation your class has a fully featured builder, Automate your logging variables, and much more.              |
|[Hibernate Validator](https://hibernate.org/validator/)|Express validation rules in a standardized way using annotation-based constraints and benefit from transparent integration with a wide variety of frameworks.|
|[Springdoc OpenAPI UI](https://springdoc.org/)|OpenAPI 3 Library for spring boot projects. Is based on swagger-ui, to display the OpenAPI description.|
|[SoapUI core module](https://www.soapui.org/open-source/)|SoapUI is the world's leading Functional Testing tool for SOAP and REST testing.|
|[Swagger Parser](https://github.com/swagger-api/swagger-parser)|Parses OpenAPI definitions in JSON or YAML format into swagger-core representation as Java POJO, returning any validation warnings/errors.|

# 📑 Getting started 

These instructions will get you a copy of the project up and running on your local machine for development and testing purposes.

### Prerequisites

* [JDK Installation](https://docs.oracle.com/en/java/javase/11/install/overview-jdk-installation.html#GUID-8677A77F-231A-40F7-98B9-1FD0B48C346A)
* [Apache Maven Installation](https://maven.apache.org/install.html)
* [Setting up Lombok](https://projectlombok.org/setup/overview)
  * [Eclipse and its offshoots](https://projectlombok.org/setup/eclipse)
  * [Intellij IDEA](https://projectlombok.org/setup/intellij)
  * [Netbeans](https://projectlombok.org/setup/netbeans)
  * [Microsoft Visual Studio Code](https://projectlombok.org/setup/vscode)

### External dependencies

The project use __"SoapUI Core Module"__ dependency, which is not maven dependency, so you must have internet access to download the dependency from the external repository, below is the repository configuration inside the pom.xml file.

```xml
  ...

  <repositories>
    <repository>
      <id>SmartBearPluginRepository</id>
      <url>https://rapi.tools.ops.smartbear.io/nexus/content/groups/public/</url>			
    </repository>
  </repositories>
  
  ...
```
### Running the application with IDE

There are several ways to run a Spring Boot application on your local machine. One way is to execute the `main` method in the `Openapi2SoapUIApplication` class from your IDE.

#### Example (Eclipse and its offshoots) :
* Download the zip or clone the Git repository.
* Unzip the zip file (if you downloaded one)
* Open IDE
	* File -> Import -> Existing Maven Project -> Navigate to the folder project 
	* Select the project
* Choose the Spring Boot Application file (search for @SpringBootApplication)
* Right Click on the file and Run as Java Application
* URL to access: **http://localhost:8080/api-openapi-to-soapui/v1/soap-ui-projects**

### Running the application locally with Maven

Alternatively you can use the [Spring Boot Maven plugin](https://docs.spring.io/spring-boot/docs/current/reference/html/build-tool-plugins-maven-plugin.html) like so:

* Download the zip or clone the Git repository.
* Unzip the zip file (if you downloaded one)
* Open Command Prompt and Change directory (cd) to folder containing pom.xml
* To build and start the server type

```shell
$ mvn spring-boot:run
```

* URL to access: **http://localhost:8080/api-openapi-to-soapui/v1/soap-ui-projects**

### Running the application in Docker

* Download the zip or clone the Git repository.
* Unzip the zip file (if you downloaded one)
* Open Command Prompt and Change directory (cd) to folder containing pom.xml
* To build and start the docker container type

```shell
$ mvn clean package -Pjar
$ docker-compose up -d
```

* URL to access: **http://localhost:8080/api-openapi-to-soapui/v1/soap-ui-projects**

### Running the application deploying WAR on Tomcat

The code can also be built into a war and then deployed on a Tomcat server.

* 	Download the zip or clone the Git repository.
* 	Unzip the zip file (if you downloaded one)
* 	Open Command Prompt and Change directory (cd) to folder containing pom.xml
* 	To build the war type

```sh
$ mvn clean package
```

* 	Once the war is built, copy the output WAR to Tomcat's webapps directory.

```sh
$CATALINA_HOME/webapps/openapi2soapui-<version>.war
```

* Restart Tomcat Server
* URL to access: **http://localhost:8080/openapi2soapui/api-openapi-to-soapui/v1/soap-ui-projects**
## Files and Directories Structure

The project directory has a particular directory structure. A representative project is shown below:

### Project Structure

```text
.
├── src
│   └── main
│       └── java
│           ├── org.apiaddicts.apitools.openapi2soapui
│           │ 
│           ├── org.apiaddicts.apitools.openapi2soapui.config
│           │  
│           ├── org.apiaddicts.apitools.openapi2soapui.constants
│           │ 
│           ├── org.apiaddicts.apitools.openapi2soapui.controller
│           │ 
│           ├── org.apiaddicts.apitools.openapi2soapui.error
│           ├── org.apiaddicts.apitools.openapi2soapui.error.exceptions
│           ├── org.apiaddicts.apitools.openapi2soapui.error.validators
│           │     
│           ├── org.apiaddicts.apitools.openapi2soapui.model
│           │
│           ├── org.apiaddicts.apitools.openapi2soapui.request
│           │
│           ├── org.apiaddicts.apitools.openapi2soapui.service
│           │
│           └── org.apiaddicts.apitools.openapi2soapui.util
├── src
│   └── main
│       └── resources
│           ├── static
│           │   └── api.yaml
│           │   
│           ├── application.properties
│           ├── log4j.properties
│           └── messages.properties
├── JRE System Library
├── Maven Dependencies
├── src
├── target
│   └──openapi2soapui-1.0.2
├── .gitlab-ci.yaml
├── lombok.config
├── mvnw
├── mvnw.cmd
├── pom.xml
└── README.md
```

### Packages

* 	`config` - app configurations;
* 	`constants` - app contants;
* 	`controller` - listen to the client;
* 	`error` - manage errors;
* 	`exceptions` - custom exception handling;
* 	`validators` - custom validations;
* 	`model` - entities;
* 	`request` - body request model/entities;
* 	`service` - business logic;
* 	`util` - utility classes;


### Resources
* 	`resources/` - contains all the static resources, templates and property files.
* 	`resources/static` - contains static resources.
* 	`resources/static/api.yaml` - contains Open API Specification.
* 	`resources/application.properties` - contains application-wide properties. Spring reads the properties defined in this file to configure your application. You can define server’s default port, server’s context path, database URLs etc, in this file.
* 	`resources/log4j.properties` - contains contains the entire runtime configuration used by log4j. This file will contain log4j appenders information, log level information and output file names for file appenders.
* 	`resources/messages.properties` - contains the error messages used in the application.
* mvnw / mvnw.cmd - This allows you to run the Maven project without having Maven installed and present in the path. Download the correct version of Maven if it can't be found (as far as I know by default in your user home directory). The mvnw file is for Linux (bash) and mvnw.cmd is for the Windows environment.
* 	`pom.xml` - contains all the project dependencies

## Deploy

* 	Build the war type

```sh
$ mvn clean package
```

* 	Once the war is built, copy the output WAR to Tomcat's webapps directory.

```sh
$CATALINA_HOME/webapps/openapi2soapui.war
```

* Restart Tomcat Server
* URL to access: **\<protocol\>://\<host\>:\<port\>/openapi2soapui/api-openapi-to-soapui/v1/soap-ui-projects**

## Documentation

- [cURL Example](example.sh)
- [Open API Specification](./src/main/resources/static/api.yaml)
- [Swagger UI](http://localhost:8080/swagger-ui.html) - `http://localhost:8080/swagger-ui.html`
- Find Java Doc in **javadoc** folder
- Java Doc is generated in ./target/site/apidocs` folder using the Maven command 

```sh
mvn javadoc:javadoc
```

## 💛 Sponsors
<p align="center">
	<a href="https://apiaddicts.org/">
    	<img src="https://apiaddicts.cloudappi.net/web/image/4248/LOGOCloudappi2020Versiones-01.png" alt="cloudappi" width="150"/>
        <img src="https://apiaddicts-web.s3.eu-west-1.amazonaws.com/wp-content/uploads/2022/03/17155736/cropped-APIAddicts-logotipo_rojo.png" height = "75">
	</a>
</p>

 readmeEtag: '"b7b5d345a985edce7396b3549da001259abf4c15"' readmeLastModified: Tue, 12 Sep 2023 10:57:37 GMT repositoryId: 352716389 description: >- API to generate a SoapUI project from an OpenAPI Specification (fka Swagger Specification) Given an OpenAPI Specification, either v2 or v3, a SoapUI project is generated with the requests for each resource operation and a test suite. The response is the content of the SoapUI project in XML format to save as file and import into the SoapUI application. created: '2021-03-29T16:47:42Z' updated: '2025-11-25T18:41:08Z' language: Java archived: false stars: 9 watchers: 2 forks: 4 owner: apiaddicts logo: https://avatars.githubusercontent.com/u/31730093?v=4 license: LGPL-3.0 repoEtag: '"6dcdfbbcdc91aa4783acaf01f388e78d8d3092e64ccb561cd7cd7a2335feb830"' repoLastModified: Tue, 25 Nov 2025 18:41:08 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/jshmrtn/openapi-compiler v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJQ29tcGlsZXIKClshW0hleCBkb2NzXShodHRwOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvaGV4LnBtLWRvY3MtZ3JlZW4uc3ZnP3N0eWxlPWZsYXQpXShodHRwczovL2hleGRvY3MucG0vb3BlbmFwaV9jb21waWxlcikKIVsuZ2l0aHViL3dvcmtmbG93cy9lbGl4aXIueW1sXShodHRwczovL2dpdGh1Yi5jb20vanNobXJ0bi9vcGVuYXBpLWNvbXBpbGVyL3dvcmtmbG93cy8uZ2l0aHViL3dvcmtmbG93cy9lbGl4aXIueW1sL2JhZGdlLnN2ZykKWyFbQ292ZXJhZ2UgU3RhdHVzXShodHRwczovL2NvdmVyYWxscy5pby9yZXBvcy9naXRodWIvanNobXJ0bi9vcGVuYXBpLWNvbXBpbGVyL2JhZGdlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly9jb3ZlcmFsbHMuaW8vZ2l0aHViL2pzaG1ydG4vb3BlbmFwaS1jb21waWxlcj9icmFuY2g9bWFzdGVyKQpbIVtIZXgucG1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vaGV4cG0vZHQvb3BlbmFwaV9jb21waWxlci5zdmcpXShodHRwczovL2hleC5wbS9wYWNrYWdlcy9vcGVuYXBpX2NvbXBpbGVyKQoKOndhcm5pbmc6IDp3YXJuaW5nOiBUaGlzIGxpYnJhcnkgaXMgaW5jb21wbGV0ZSBhbmQgZXhwZXJpbWVudGFsISA6d2FybmluZzogOndhcm5pbmc6CgpDb21waWxlIE9wZW5BcGkgMy4wIEVsaXhpciBDbGllbnQgZnJvbSBKU09OIC8gWWFtbC4KCkhlbHAgdG8gbWFrZSB0aGlzIGxpYnJhcnkgbW9yZSBzcGVjIGNvbXBsaWFudCBpcyB3ZWxjb21lIGluIGZvcm0gb2YgaXNzdWVzIC8gUFIncy4KCiMjIEluc3RhbGxhdGlvbgoKVGhlIHBhY2thZ2UgY2FuIGJlIGluc3RhbGxlZCBieSBhZGRpbmcgYG9wZW5hcGlfY29tcGlsZXJgIHRvIHlvdXIgbGlzdCBvZiBkZXBlbmRlbmNpZXMKaW4gYG1peC5leHNgOgoKYGBgZWxpeGlyCmRlZiBkZXBzIGRvCiAgWwogICAgezpvcGVuYXBpX2NvbXBpbGVyLCAifj4gMS4wLWJldGEifQogIF0KZW5kCmBgYAoKRG9jdW1lbnRhdGlvbiBjYW4gYmUgZ2VuZXJhdGVkIHdpdGggW0V4RG9jXShodHRwczovL2dpdGh1Yi5jb20vZWxpeGlyLWxhbmcvZXhfZG9jKQphbmQgcHVibGlzaGVkIG9uIFtIZXhEb2NzXShodHRwczovL2hleGRvY3MucG0pLiBUaGUgZG9jcyBjYW4gYmUgZm91bmQgYXQKW2h0dHBzOi8vaGV4ZG9jcy5wbS9vcGVuYXBpX2NvbXBpbGVyXShodHRwczovL2hleGRvY3MucG0vb3BlbmFwaV9jb21waWxlcikuCg== readmeEtag: '"bf382aafe1fb1d55ab387b51cb4735e682323308"' readmeLastModified: Tue, 05 Jul 2022 16:13:18 GMT repositoryId: 251401867 description: ':gear: Eixir OpenAPI Compiler' created: '2020-03-30T19:03:24Z' updated: '2022-07-07T18:58:33Z' language: Elixir archived: false stars: 6 watchers: 2 forks: 1 owner: jshmrtn logo: https://avatars.githubusercontent.com/u/10851004?v=4 license: MIT repoEtag: '"95d78d050b0a4c0c219b58b7779e1e58e7b2df8d7c7ae4f34b2e79d0edb59a67"' repoLastModified: Thu, 07 Jul 2022 18:58:33 GMT foundInMaster: true category: - Description Validators - Parsers id: 9b75226dafbc6d1d3410ef685b780166 - source: openapi3 tags repository: https://github.com/googee/openapi-builder v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQXBpIEdVSSBCdWlsZGVyCgpbZGVtb10oaHR0cHM6Ly9nb29nZWUuZ2l0aHViLmlvL09wZW5BUEktUGFnZS9idWlsZDEwMC8pCgojIyBQcm9qZWN0IHNldHVwCmBgYApucG0gaW5zdGFsbApgYGAKCiMjIyBDb21waWxlcyBhbmQgaG90LXJlbG9hZHMgZm9yIGRldmVsb3BtZW50CmBgYApucG0gcnVuIHNlcnZlCmBgYAoKIyMjIENvbXBpbGVzIGFuZCBtaW5pZmllcyBmb3IgcHJvZHVjdGlvbgpgYGAKbnBtIHJ1biBidWlsZApgYGAKCiMjIyBMaW50cyBhbmQgZml4ZXMgZmlsZXMKYGBgCm5wbSBydW4gbGludApgYGAKCiMjIyBDdXN0b21pemUgY29uZmlndXJhdGlvbgpTZWUgW0NvbmZpZ3VyYXRpb24gUmVmZXJlbmNlXShodHRwczovL2NsaS52dWVqcy5vcmcvY29uZmlnLykuCg== readmeEtag: '"2ae7b28459e5053f49ff08805c8788ee5029dcfc"' readmeLastModified: Sun, 07 Aug 2022 09:45:28 GMT repositoryId: 264067554 description: ':herb: OpenAPI GUI Builder' created: '2020-05-15T01:38:24Z' updated: '2024-05-03T14:54:03Z' language: TypeScript archived: false stars: 6 watchers: 1 forks: 2 owner: GooGee logo: https://avatars.githubusercontent.com/u/13238516?v=4 license: MIT repoEtag: '"e4e9d361ed4aa69d1ef0d922779849e1fcce704f9ff0658a2394f7b31666eb2d"' repoLastModified: Fri, 03 May 2024 14:54:03 GMT foundInMaster: true category: - Description Validators - Testing - Server Implementations id: 760db624d3fd67681974029110114e07 - source: - openapi3 tags - openapi31 tags repository: https://github.com/aptori-dev/sift-action v3: true v3_1: true id: 81a95e8d4e1634da0b6c0ea11d40c674 repositoryMetadata: base64Readme: >- WyFbQXB0b3JpXShodHRwczovL2FwcC5hcHRvcmkuZGV2L3Byb2plY3RNZXJjdXJ5UHVycGxlLnN2ZyldKGh0dHBzOi8vYXB0b3JpLmRldi8pCgojIEFwdG9yaSBTaWZ0IEFjdGlvbgoKCkFwdG9yaSBpcyBhbiBBSS1FbmFibGVkIFtkZXZlbG9wZXIgdG9vbCBmb3IgQVBJIFRlc3RpbmddKGh0dHBzOi8vYXB0b3JpLmRldi8pCmFuZCBBcHBsaWNhdGlvbiBTZWN1cml0eSBUZXN0aW5nLgoKQXB0b3JpIGxldmVyYWdlcyBBSSB0byBhc3Npc3QgaW4gdGhlIGRldmVsb3BtZW50IG9mIHNlY3VyZSwgaGlnaC1xdWFsaXR5CnNvZnR3YXJlLiBPdXIgU2VtYW50aWMgUmVhc29uaW5nIFRlY2hub2xvZ3kgY29tcHJlaGVuZHMgdGhlIHN0cnVjdHVyZSBvZiB5b3VyCmFwcGxpY2F0aW9uLCBhbGxvd2luZyBmb3IgYXV0b25vbW91cyBhcHBsaWNhdGlvbiB0ZXN0aW5nLiBUaGlzIGVmZmljaWVuY3kgZnJlZXMKZGV2ZWxvcGVycyBmcm9tIHRpbWUtY29uc3VtaW5nIHRlc3RpbmcgdGFza3Mgd2hpbGUgZW5zdXJpbmcgdGhlIHNvZnR3YXJlIGlzCnNlY3VyZSBhbmQgZnVuY3Rpb25zIGFzIGludGVuZGVkLgoKU2lmdCB1c2VzIEFJIHRvIHVuZGVyc3RhbmQgdGhlIGFwcGxpY2F0aW9uJ3MgQVBJIHNlbWFudGljYWxseSwgaWRlbnRpZnlpbmcgaG93CmRpZmZlcmVudCBvcGVyYXRpb25zIGNvbm5lY3QgYW5kIGNyZWF0aW5nIGEgbG9naWNhbCBmbG93Y2hhcnQgb2YgdGhlc2UKd29ya2Zsb3dzLiBJdCB0aGVuIHRlc3RzIHRoZXNlIHdvcmtmbG93cyBieSBwcm92aWRpbmcgYXBwcm9wcmlhdGUgaW5wdXQgdmFsdWVzLApjaGVja2luZyB0aGF0IHRoZSBhcHBsaWNhdGlvbidzIGJ1c2luZXNzIGxvZ2ljIGlzIGFjY3VyYXRlIGF0IGVhY2ggc3RhZ2UuCgpJbnRlZ3JhdGUgU2lmdCBpbiB5b3VyIENJIHRvIGlkZW50aWZ5IGJ1c2luZXNzIGxvZ2ljIGJ1Z3MgYW5kIHNlY3VyaXR5CnZ1bG5lcmFiaWxpdGllcyBpbiB5b3VyIGFwcGxpY2F0aW9ucyB3aXRoIEFwdG9yaSBBdXRvbm9tb3VzIFtBcHBsaWNhdGlvbgpTZWN1cml0eSBUZXN0aW5nXShodHRwczovL2FwdG9yaS5kZXYvYXBwc2VjL3NlY3VyaXR5LXRlc3RpbmcpLiBTaWZ0IEFjdGlvbiBpcyBhCltHaXRIdWIgQWN0aW9uXShodHRwczovL2RvY3MuZ2l0aHViLmNvbS9hY3Rpb25zKSB0aGF0IHJ1bnMgQXB0b3JpJ3MgU2lmdCBDTEkKdG9vbCBpbiBhIGNvbnRpbm91cyBpbnRlZ3JhdGlvbiAoQ0kpIHdvcmtmbG93LgoKCiMjIEFkZGl0aW9uYWwgSW5mb3JtYXRpb24KCiogV2F0Y2ggW0hvdyBBcHRvcmkgV29ya3NdKGh0dHBzOi8vYXB0b3JpLmRldi9wcm9kdWN0L3doeS1hcHRvcmkpIGZvciBhIGxvb2sgYXQgdGhlCiAgdGVjaG5vbG9neSB0aGF0IG1ha2VzIFNpZnQgdW5pcXVlLgoqIFtBcHRvcmkgRG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9kb2NzLmFwdG9yaS5kZXYvKSBoYXMgZnVydGhlciBkZXRhaWxzIG9uIHRoZQogIFNpZnQgQ0xJIHRvb2wgYW5kIHRoZSBBcHRvcmkgUGxhdGZvcm0uCiogW0dpdEh1YgogIFdvcmtmbG93c10oaHR0cHM6Ly9kb2NzLmdpdGh1Yi5jb20vYWN0aW9ucy91c2luZy13b3JrZmxvd3MvYWJvdXQtd29ya2Zsb3dzKQogIGhhcyBmdXJ0aGVyIGRldGFpbHMgYWJvdXQgY29uZmlndXJpbmcgam9icyBpbiBHaXRIdWIgQWN0aW9ucy4KCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCiMjIyAxLiBHZXQgeW91ciBBcHRvcmkgYWNjb3VudApUbyBydW4gU2lmdCwgeW91IG5lZWQgdG8gaGF2ZSBhbiBhY2NvdW50IG9uIEFwdG9yaS4gR2V0IGluIHRvdWNoIHdpdGggb3VyCmV4cGVydCB0ZWFtIHRvIHNldCB1cCBhIFt0cmlhbAphY2NvdW50XShodHRwczovL2FwdG9yaS5kZXYvZ2V0LWFwdG9yaS1ib29rLWEtZGVtbykuIEFmdGVyIGFjdGl2YXRpbmcgeW91cgphY2NvdW50LCB5b3UgY2FuIGVtcGxveSBgc2lmdC1hY3Rpb25gIHRvIGV2YWx1YXRlIHlvdXIgYXBwbGljYXRpb27igJlzIEFQSSwgYW5kCmF1dG9tYXRpY2FsbHkgdXBsb2FkIHRoZSBmaW5kaW5ncyB0byB0aGUgQXB0b3JpIFBsYXRmb3JtLgoKIyMjIDIuIEFkZCBTaWZ0IHRvIHlvdXIgd29ya2Zsb3cKR28gdG8gdGhlIEdpdEh1YiBSZXBvc2l0b3J5IG9uIHdoaWNoIHlvdSB3YW50IHRvIHJ1biBTaWZ0LiBBZGQgdGhlCmBzaWZ0LWFjdGlvbmAgdG8gYW4gZXhpc3RpbmcgR2l0SHViIHdvcmtmbG93IG9yIGNyZWF0ZSBhIG5ldyB3b3JrZmxvdy4KQSBzYW1wbGUgd29ya2Zsb3cgZmlsZSBpcyBwcm92aWRlZCBiZWxvdyB1bmRlciBbRXhhbXBsZSBXb3JrZmxvd10oI2V4YW1wbGUtd29ya2Zsb3cpLgoKSW4geW91ciBHaXRIdWIgcmVwb3NpdG9yeSBzZXR0aW5ncywgYWRkIHZhbHVlcyBpbiBzZWNyZXRzIGFuZCB2YXJpYWJsZXMgdGhhdAphcmUgdXNlZCBmb3IgaW5wdXQgdmFsdWVzIG9mIHRoZSBBY3Rpb24uCgoqIERlZmluZSBhCiAgW3NlY3JldF0oaHR0cHM6Ly9kb2NzLmdpdGh1Yi5jb20vYWN0aW9ucy9zZWN1cml0eS1ndWlkZXMvZW5jcnlwdGVkLXNlY3JldHMpCiAgbmFtZWQgYFNJRlRfUExBVEZPUk1fS0VZYCB0aGF0IGNvbnRhaW5zIHRoZSB2YWx1ZSBvZiBhbiBBcHRvcmkgUGxhdGZvcm0gS2V5LgogIFRoaXMgc2VjcmV0IGlzIHVzZWQgZm9yIHRoZSByZXF1aXJlZCBpbnB1dCBgcGxhdGZvcm1LZXlgIGluIHRoZSB3b3JrZmxvdwogIGV4YW1wbGUgYmVsb3cuCiogRGVmaW5lIGEKICBbdmFyaWFibGVdKGh0dHBzOi8vZG9jcy5naXRodWIuY29tL2VuL2FjdGlvbnMvbGVhcm4tZ2l0aHViLWFjdGlvbnMvdmFyaWFibGVzI2NyZWF0aW5nLWNvbmZpZ3VyYXRpb24tdmFyaWFibGVzLWZvci1hLXJlcG9zaXRvcnkpCiAgbmFtZWQgYFNJRlRfQ09ORklHVVJBVElPTl9JRGAgdGhhdCBjb250YWlucyB0aGUgSUQgb2YgYSBjb25maWd1cmF0aW9uIGNyZWF0ZWQKICBpbiB0aGUgQXB0b3JpIFBsYXRmb3JtLiAgVGhpcyB2YXJpYWJsZSBpcyB1c2VkIGZvciB0aGUgcmVxdWlyZWQgaW5wdXQKICBgY29uZmlndXJhdGlvbklkYCBpbiB0aGUgd29ya2Zsb3cKICBleGFtcGxlIGJlbG93LgoKCiMjIEFjdGlvbiBJbnB1dHMKCiMjIyBgcGxhdGZvcm1LZXlgCioqUmVxdWlyZWQqKiBcCkFwdG9yaSBQbGF0Zm9ybSBrZXkuCgojIyMgYGNvbmZpZ3VyYXRpb25JZGAKKipSZXF1aXJlZCoqIFwKSUQgb2YgY29uZmlndXJhdGlvbiB0byBmZXRjaCBmcm9tIEFwdG9yaSBQbGF0Zm9ybS4KCiMjIyBgdGFyZ2V0VXJsYAoqKk9wdGlvbmFsKiogXApVUkwgd2hlcmUgYXBwbGljYXRpb24gQVBJIGNhbiBiZSBhY2Nlc3NlZC4gVGhpcyBvdmVycmlkZXMgdGhlIFVSTCBzcGVjaWZpZWQgaW4gdGhlIGNvbmZpZ3VyYXRpb24uCgojIyMgYGxhYmVsc2AKKipPcHRpb25hbCoqICBcClNwYWNlLXNlcGFyYXRlZCBsaXN0IGxhYmVscyBpbiBrZXk9dmFsdWUgZm9ybSB0byBhcHBlbmQgdG8gdGhlIG1ldGRhdGEgZm9yIGEgdGVzdCBydW4uCgojIyMgYG5ldHdvcmtgCioqT3B0aW9uYWwqKiBcCkRvY2tlciBuZXR3b3JrIGJyaWRnZSBuYW1lLiBNYWtlIHN1cmUgeW91ciBhcHBsaWNhdGlvbiB1bmRlciB0ZXN0IHJ1bnMgaW4gdGhlIHNhbWUgbmV0d29yay4KCgojIyBBY3Rpb24gT3V0cHV0cwoKU2lmdCBvdXRwdXRzIHRlc3QgcmVzdWx0cyBpbiBTQVJJRiBmb3JtYXQgaW4gZmlsZSBgc2lmdC5zYXJpZmAgaW4gdGhlIHdvcmtzcGFjZQp3aGVyZSB0aGUgQWN0aW9uIGlzIHJ1bi4KCgojIyBFeGFtcGxlIFdvcmtmbG93CgpCZWxvdyBpcyBhIHNhbXBsZSB3b3JrZmxvdyB0aGF0IHVzZXMgU2lmdCBBY3Rpb24uICBUaGlzIHdvcmtmbG93IHVzZXMgYSBzZWNyZXQKKGBTSUZUX1BMQVRGT1JNX0tFWWApIGFuZCB2YXJpYWJsZSAoYFNJRlRfQ09ORklHVVJBVElPTl9JRGApIHRoYXQgYXJlIGRlZmluZWQgaW4KdGhlIHJlcG9zaXRvcnkgc2V0dGluZ3MuCgpgYGB5YW1sCiMgQW4gZXhhbXBsZSB3b3JrZmxvdyBmaWxlIHRvIGhlbHAgeW91IGdldCBzdGFydGVkCm5hbWU6IFJ1biBTaWZ0CgojIENvbnRyb2xzIHdoZW4gdGhlIHdvcmtmbG93IHdpbGwgcnVuCm9uOgogIHB1c2g6CiAgICBicmFuY2hlczogWyBtYWluIF0KICBwdWxsX3JlcXVlc3Q6CiAgICBicmFuY2hlczogWyBtYWluIF0KCmpvYnM6CiAgYXB0b3JpLXNpZnQ6CiAgICBuYW1lOiBBcHRvcmkgU2lmdAogICAgcnVucy1vbjogdWJ1bnR1LWxhdGVzdAoKICAgICMKICAgICMgUGVybWlzc2lvbnMgdG8gdXBsb2FkIHRoZSBDb2RlIFNjYW5uaW5nIHJlc3VsdAogICAgIyAob25seSBuZWNlc3NhcnkgaWYgdXNpbmcgR2l0SHViIEFkdmFuY2VkIFNlY3VyaXR5IGFuZCBTQVJJRiB1cGxvYWQgYmVsb3cpCiAgICAjCiAgICBwZXJtaXNzaW9uczoKICAgICAgYWN0aW9uczogcmVhZAogICAgICBjb250ZW50czogcmVhZAogICAgICBzZWN1cml0eS1ldmVudHM6IHdyaXRlCgogICAgc3RlcHM6CiAgICAtIG5hbWU6IENoZWNrIG91dCByZXBvCiAgICAgIHVzZXM6IGFjdGlvbnMvY2hlY2tvdXRAdjQKCiAgICAjCiAgICAjIEV4YW1wbGUgb2YgYnVpbGRpbmcgYW5kIHJ1bm5pbmcgYW4gYXBwbGljYXRpb24gd2l0aCBEb2NrZXIuCiAgICAjCiAgICAjIE5vdGUsIHRoaXMgaXMgYW4gZXhhbXBsZS4gIEFkZCB3b3JrZmxvdyBzdGVwKHMpIHRvIGRlcGxveSBvciBydW4gYW4KICAgICMgaW5zdGFuY2Ugb2YgeW91ciBhcHBsaWNhdGlvbi4KICAgICMKICAgIC0gbmFtZTogQnVpbGQgYW5kIHJ1biB0aGUgYXBwbGljYXRpb24KICAgICAgcnVuOiB8CiAgICAgICAgZG9ja2VyIGJ1aWxkIC10IG15YXBwOmxhdGVzdCAuCiAgICAgICAgZG9ja2VyIG5ldHdvcmsgY3JlYXRlIG5ldHdvcmsxCiAgICAgICAgZG9ja2VyIHJ1biAtZCAtcCA1MDAwOjUwMDAgLS1uZXR3b3JrIG5ldHdvcmsxIC0tbmFtZSBteWFwcCBteWFwcDpsYXRlc3QKCiAgICAjCiAgICAjIFJ1biBTaWZ0IHVzaW5nIHBsYXRmb3JtIGtleSBhbmQgY29uZmlndXJhdGlvbiBJRCBzdG9yZWQgaW4geW91ciByZXBvc2l0b3J5IHNlY3JldHMKICAgICMKICAgIC0gbmFtZTogUnVuIFNpZnQKICAgICAgdXNlczogQXB0b3JpLWRldi9zaWZ0LWFjdGlvbkB2MQogICAgICB3aXRoOgogICAgICAgIHBsYXRmb3JtS2V5OiAke3sgc2VjcmV0cy5TSUZUX1BMQVRGT1JNX0tFWSB9fQogICAgICAgIGNvbmZpZ3VyYXRpb25JZDogJHt7IHZhcnMuU0lGVF9DT05GSUdVUkFUSU9OX0lEIH19CiAgICAgICAgdGFyZ2V0VXJsOiBodHRwOi8vbXlhcHA6NTAwMAogICAgICAgIG5ldHdvcms6IG5ldHdvcmsxCiAgICAgICAgIyBVc2VyLWRlZmluZWQgbGFiZWxzIGNhbiBiZSB1c2VkIHRvIGhlbHAgZGlzdGluZ3Vpc2ggYSBwYXJ0aWN1bGFyIHJ1bgogICAgICAgIGxhYmVsczogfAogICAgICAgICAgR0lUSFVCX1JVTl9JRD0ke3tnaXRodWIucnVuX2lkfX0KICAgICAgICAgIEdJVEhVQl9SVU5fTlVNQkVSPSR7e2dpdGh1Yi5ydW5fbnVtYmVyfX0KICAgICAgICAgIGNvbW1pdD0ke3tnaXRodWIuc2hhfX0KCiAgICAjCiAgICAjIE9wdGlvbmFsOiBVcGxvYWQgU0FSSUYgZmlsZSB0byBHaXRIdWIgQ29kZSBTY2FubmluZwogICAgIyAocmVxdWlyZXMgR2l0SHViIEFkdmFuY2VkIFNlY3VyaXR5KQogICAgIwogICAgLSBuYW1lOiBVcGxvYWQgU0FSSUYgZmlsZQogICAgICB1c2VzOiBnaXRodWIvY29kZXFsLWFjdGlvbi91cGxvYWQtc2FyaWZAdjMKICAgICAgd2l0aDoKICAgICAgICAjIFBhdGggdG8gU0FSSUYgZmlsZSByZWxhdGl2ZSB0byB0aGUgcm9vdCBvZiB0aGUgcmVwb3NpdG9yeQogICAgICAgIHNhcmlmX2ZpbGU6IHNpZnQuc2FyaWYKICAgICAgICAjIENhdGVnb3J5IGZvciB0aGUgcmVzdWx0cyAtIHRvIGRpZmZlcmVudGlhdGUgbXVsdGlwbGUgcmVzdWx0cyBmb3Igb25lIGNvbW1pdAogICAgICAgIGNhdGVnb3J5OiBhcHRvcmktc2lmdApgYGAK readmeEtag: '"7c28d9f8e16595cf1eb1d9136de52abae8b81574"' readmeLastModified: Thu, 18 Sep 2025 15:56:24 GMT repositoryId: 681837537 description: Run Aptori Sift in GitHub Actions created: '2023-08-22T21:43:03Z' updated: '2025-09-18T15:56:28Z' language: JavaScript archived: false stars: 6 watchers: 2 forks: 0 owner: Aptori-dev logo: https://avatars.githubusercontent.com/u/140379261?v=4 license: MIT repoEtag: '"d0274020358802804988ce8a728a0593dcfa938320187dff77f66fcbe65cc8ed"' repoLastModified: Thu, 18 Sep 2025 15:56:28 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/odoo-plus/odootools-openapi v3: true id: 5908c15dabba77657f3b02534b6eb759 repositoryMetadata: base64Readme: >- T2Rvb1Rvb2xzIE9wZW5BUEkKPT09PT09PT09PT09PT09PT0KClRoaXMgcGFja2FnZSBpcyBhIGxpYnJhcnkgYW5kIGFuIGEgcGx1Z2luIGZvciBbb2Rvby10b29sc10oaHR0cHM6Ly9naXRodWIuY29tL2xsYWNyb2l4L29kb28tdG9vbHMpLgoKV2hlbiB1c2VkIGFzIGEgbGlicmFyeSwgaXQgZGVmaW5lIGFuIG9ybSB0aGF0IGhlbHBzIHVud3JhcCBqc29uIGJvZHkgaW50byBweXRob24gb2JqZWN0cyBhbmQgT2RvbwpvYmplY3RzIGJhY2sgaW50byBKU09OLgoKV2hlbiB1c2VkIGFzIGEgY29tbWFuZCwgaXQgd2lsbCBnZW5lcmF0ZSBhbiBPZG9vIE1vZHVsZSB3aXRoIGFwaSBtb2RlbHMgYW5kIGNvbnRyb2xsZXJzIGFzIGRlZmluZWQKaW4gYW4gb3BlbmFwaTMgc3BlYyB3aXRoIGN1c3RvbSBleHRlbnNpb25zIGZvciBPZG9vLgoKVGhlIGNvbW1hbmQgY2FuIGJlIGludm9rZWQgYXMgc3VjaDoKCiAgICBvZG9vdG9vbHMgZ2VuIG9wZW5hcGkgLS1kZXN0aW5hdGlvbiBteV9tb2R1bGVfYXBpIC0tcGF0aCBwYXRoX3RvX3NwZWMKICAgIG9kb290b29scyBnZW4gb3BlbmFwaSAtLWRlc3RpbmF0aW9uIG15X21vZHVsZV9hcGkgLS11cmwgdXJsX3RvX3NwZWMKClRoZW4gdGhlIG1vZHVsZSB3aWxsIGJlIGNyZWF0ZWQgaW5zaWRlIHRoZSBkZXN0aW5hdGlvbiBwYXRoLiBJdCB3aWxsIHVzZQp0aGUgdGl0bGUvZGVzY3JpcHRpb24gb2YgdGhlIGFwaSB0byBkZWZpbmUgdGhlIG5hbWUvc3VtbWFyeSBvZiB0aGUgbW9kdWxlLgoKCkV4YW1wbGUgQVBJOgoKICAgIG9wZW5hcGk6IDMuMC4wCiAgICBpbmZvOgogICAgICBjb250YWN0OgogICAgICAgIGVtYWlsOiBpbmZvQG9kb28ucGx1cwogICAgICAgIG5hbWU6IExvw69jIEZhdXJlLUxhY3JvaXgKICAgICAgICB1cmw6IGh0dHBzOi8vd3d3Lm9kb28ucGx1cwogICAgICBkZXNjcmlwdGlvbjogJycKICAgICAgbGljZW5zZToKICAgICAgICBuYW1lOiBBcGFjaGUgMi4wCiAgICAgICAgdXJsOiBodHRwczovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wLmh0bWwKICAgICAgdGl0bGU6ICcnCiAgICAgIHZlcnNpb246ICcxJwoKICAgIHRhZ3M6CiAgICAgIC0gbmFtZTogcHJvamVjdHMKICAgICAgICBkZXNjcmlwdGlvbjogU3VwZXIgcHJvamVjdHMKCiAgICBjb21wb25lbnRzOgogICAgICBzY2hlbWFzOgogICAgICAgIEFwaVJlc3VsdDoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBBcGkgUmVzdWx0CiAgICAgICAgICBwcm9wZXJ0aWVzOgogICAgICAgICAgICBpZDoKICAgICAgICAgICAgICB0aXRsZTogUmVzb3VyY2UgSWQKICAgICAgICAgICAgICB0eXBlOiBpbnRlZ2VyCiAgICAgICAgICB0eXBlOiBvYmplY3QKICAgICAgICBQcm9qZWN0OgogICAgICAgICAgZGVzY3JpcHRpb246IFByb2plY3QKICAgICAgICAgIHgtbW9kZWw6IHByb2plY3QucHJvamVjdAogICAgICAgICAgcHJvcGVydGllczoKICAgICAgICAgICAgaWQ6CiAgICAgICAgICAgICAgdGl0bGU6IElECiAgICAgICAgICAgICAgdHlwZTogaW50ZWdlcgogICAgICAgICAgICBuYW1lOgogICAgICAgICAgICAgIHRpdGxlOiBOYW1lCiAgICAgICAgICAgICAgdHlwZTogc3RyaW5nCgogICAgcGF0aHM6CiAgICAgIC9wcm9qZWN0L3twcm9qZWN0fToKICAgICAgICBnZXQ6CiAgICAgICAgICB0YWdzOgogICAgICAgICAgICAtIHByb2plY3RzCiAgICAgICAgICBwYXJhbWV0ZXJzOgogICAgICAgICAgICAtIG5hbWU6IHByb2plY3QKICAgICAgICAgICAgICB4LW1vZGVsOgogICAgICAgICAgICAgICAgJHJlZjogIiMvY29tcG9uZW50cy9zY2hlbWFzL1Byb2plY3QiCiAgICAgICAgICAgICAgaW46IHBhdGgKICAgICAgICAgICAgICByZXF1aXJlZDogdHJ1ZQogICAgICAgICAgICAgIHNjaGVtYToKICAgICAgICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgICAgICAgIGRlc2NyaXB0aW9uOiBHZXQgYSBwcm9qZWN0CiAgICAgICAgICBvcGVyYXRpb25JZDogZ2V0X3Byb2plY3QKICAgICAgICAgIHJlc3BvbnNlczoKICAgICAgICAgICAgMjAwOgogICAgICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgICAgICBhcHBsaWNhdGlvbi9qc29uOgogICAgICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICAgICAgJHJlZjogJyMvY29tcG9uZW50cy9zY2hlbWFzL0FwaVJlc3VsdCcKICAgICAgICAgICAgICBkZXNjcmlwdGlvbjogUHJvamVjdAoKClJlc3VsdCBjb250cm9sbGVyOgoKICAgIGZyb20gb2Rvb3Rvb2xzX29wZW5hcGkub2Rvby5odHRwIGltcG9ydCByZXF1ZXN0LCByb3V0ZSwgQ29udHJvbGxlcgoKICAgIGZyb20gLi5hcGlfbW9kZWxzLmFwaXJlc3VsdCBpbXBvcnQgQXBpUmVzdWx0CgoKICAgIGNsYXNzIFByb2plY3RzKENvbnRyb2xsZXIpOgogICAgICAgICIiIgogICAgICAgIFN1cGVyIHByb2plY3RzCiAgICAgICAgIiIiCgogICAgICAgIEByb3V0ZSgKICAgICAgICAgICAgcm91dGU9WyIvcHJvamVjdC88bW9kZWwoJ3Byb2plY3QucHJvamVjdCcpOnByb2plY3Q+Il0sCiAgICAgICAgICAgIG1ldGhvZHM9WydHRVQnXSwKICAgICAgICAgICAgY3NyZj1GYWxzZSwKICAgICAgICAgICAgdHlwZT0ncGxhaW5qc29uJywKICAgICAgICAgICAgYXV0aD0nbm9uZScsCiAgICAgICAgKQogICAgICAgIGRlZiBnZXRfcHJvamVjdChzZWxmLCBwcm9qZWN0KToKICAgICAgICAgICAgIiIiCiAgICAgICAgICAgIEdldCBhIHByb2plY3QKICAgICAgICAgICAgIiIiCiAgICAgICAgICAgIHBhc3MKICAgICAgICAK readmeEtag: '"1a748b58abea4e5ce09fb2961286198d80c7ec4c"' readmeLastModified: Tue, 07 Mar 2023 17:23:39 GMT repositoryId: 563113258 description: Odoo Tools OpenAPI created: '2022-11-07T23:43:36Z' updated: '2026-01-05T04:46:46Z' language: Python archived: false stars: 8 watchers: 2 forks: 4 owner: odoo-plus logo: https://avatars.githubusercontent.com/u/112507593?v=4 license: GPL-3.0 repoEtag: '"a8d7842356d97cb0b551651823db0a8595eab11c3c1de4edda6589b309fc8229"' repoLastModified: Mon, 05 Jan 2026 04:46:46 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/llacroix/odoo-tools-openapi - source: openapi3 tags repository: https://github.com/mdpearce/lemmy-ts-openapi v3: true id: 110ea9d687cc4a4cebce44becb89c336 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNjaGVtYSBHZW5lcmF0b3IgZm9yIHRoZSBMZW1teSBKUyBDbGllbnQgTGlicmFyeQojIyBPdmVydmlldwpUaGlzIGlzIGEgc2ltcGxlIHNjcmlwdCB3aGljaCBjb252ZXJ0cyB0aGUgW0xlbW15IEpTIGNsaWVudF0oaHR0cHM6Ly9naXRodWIuY29tL0xlbW15TmV0L2xlbW15LWpzLWNsaWVudCkgc291cmNlIGZpbGVzIGludG8gYW4gT3BlbkFQSSAzLjAgSnNvbiBzcGVjaWZpY2F0aW9uLgoKVGhlIGdlbmVyYXRlZCBPcGVuQVBJIHNjaGVtYSBjYW4gYmUgdXNlZCBieSB0aGUgW09wZW5BUEkgR2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKSB0byBnZW5lcmF0ZSBhIGNsaWVudCBTREsgaW4gYW55IHN1cHBvcnRlZCBsYW5ndWFnZS9mcmFtZXdvcmsuCgojIyBVc2FnZQoxLiBDbG9uZSB0aGUgW0xlbW15IEpTIGNsaWVudF0oaHR0cHM6Ly9naXRodWIuY29tL0xlbW15TmV0L2xlbW15LWpzLWNsaWVudCkgcmVwb3NpdG9yeSBzb21ld2hlcmUgb24gZGlzawoyLiBGcm9tIHRoZSBgbGVtbXktdHMtb3BlbmFwaWAgcm9vdCwgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB0byBwYXJzZSB0aGUgSlMgbGlicmFyeSBhbmQgZ2VuZXJhdGUgYW4gT3BlbkFQSSBzY2hlbWEgZmlsZToKYGBgc2gKbnBtIHN0YXJ0IC0tIC1jIC9wYXRoL3RvL2xlbW15LWpzLWNsaWVudCAtbyAvcGF0aC90by9vdXRwdXQvZmlsZW5hbWUuanNvbgpgYGAKCiMjIENvbnRyaWJ1dGluZwpDb250cmlidXRpb25zIGFyZSB3ZWxjb21lISBJIGFtIG5vdCBhIFR5cGVzY3JpcHQgZGV2ZWxvcGVyIGJ5IHRyYWRlLCBhbmQgdGhlcmUgaXMgbGlrZWx5IGEgbG90IG9mIHJvb20gZm9yIGltcHJvdmVtZW50LgoK readmeEtag: '"288c66caee3e282e72df8b6539b1dd089cc2ba52"' readmeLastModified: Fri, 04 Aug 2023 11:36:51 GMT repositoryId: 658645576 description: OpenAPI schema generator for the Lemmy JS client created: '2023-06-26T08:05:16Z' updated: '2024-11-03T12:42:44Z' language: TypeScript archived: false stars: 6 watchers: 1 forks: 1 owner: mdpearce logo: https://avatars.githubusercontent.com/u/1551362?v=4 license: LGPL-2.1 repoEtag: '"e7cf7be7795b00a67bc9c5412e98a10b7b20e250fad58cfa3704128b56d8e736"' repoLastModified: Sun, 03 Nov 2024 12:42:44 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/ralphbaer/spec2go v3: true id: bc8dff79cc50f188ccbfcc1bf788789d repositoryMetadata: base64Readme: >- IyBTcGVjMkdvIC0g8J+apyBXb3JrIGluIFByb2dyZXNzIChXSVApIPCfmqcKCioqc3BlYzJnbyBpcyBjdXJyZW50bHkgdW5kZXIgZGV2ZWxvcG1lbnQuIEZlYXR1cmVzIG1heSBiZSBpbmNvbXBsZXRlIGFuZCBkb2N1bWVudGF0aW9uIGlzIHN1YmplY3QgdG8gY2hhbmdlLiBVc2UgaW4gcHJvZHVjdGlvbiBlbnZpcm9ubWVudHMgaXMgbm90IHJlY29tbWVuZGVkIHVudGlsIHRoZSBvZmZpY2lhbCByZWxlYXNlLioqCgojIyBXaGF0IGlzIFNwZWMyR28/CgoqKlNwZWMyR28qKiBpcyBwb3dlcmZ1bCBjb21tYW5kLWxpbmUgdG9vbCBhdXRvbWF0ZXMgdGhlIGNyZWF0aW9uIG9mIEdvbGFuZyBhcHBsaWNhdGlvbnMgZGlyZWN0bHkgZnJvbSBPcGVuQVBJIDMuMCBhbmQgR28gcGtnLmdvLmRldi90ZXh0L3RlbXBsYXRlIHNwZWNpZmljYXRpb25zLiBXaXRoIGl0cyBkZWZhdWx0IHNldHRpbmcgdG8gZ2VuZXJhdGUgY29kZSBmb2xsb3dpbmcgdGhlIHByaW5jaXBsZXMgb2YgSGV4YWdvbmFsIEFyY2hpdGVjdHVyZSwgU3BlYzJHbyBlbnN1cmVzIGhpZ2ggbW9kdWxhcml0eSBhbmQgZWFzZSBvZiB0ZXN0aW5nLCBvZmZlcmluZyBhIHNvbGlkIGZvdW5kYXRpb24gZm9yIGJ1aWxkaW5nIHJvYnVzdCBhcHBsaWNhdGlvbnMuCgojIyBLZXkgRmVhdHVyZXMKCi0gKipEaXJlY3QgSW50ZWdyYXRpb24gd2l0aCBPcGVuQVBJIDMuMCoqOiBDb252ZXJ0IHlvdXIgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbnMgaW50byB3ZWxsLXN0cnVjdHVyZWQgR29sYW5nIGNvZGUgc2VhbWxlc3NseS4KLSAqKkhleGFnb25hbCBBcmNoaXRlY3R1cmUqKjogQmVnaW4geW91ciBwcm9qZWN0cyB3aXRoIGFuIGFyY2hpdGVjdHVyZSB0aGF0IHN1cHBvcnRzIGFuZCBlbmhhbmNlcyBtYWludGFpbmFiaWxpdHkgYW5kIHNjYWxhYmlsaXR5LgotICoqQ3VzdG9taXphYmxlIENvZGUgR2VuZXJhdGlvbioqOiBVdGlsaXplIEdvIHBrZy5nby5kZXYvdGV4dC90ZW1wbGF0ZSB0ZW1wbGF0ZXMgdG8gc2hhcGUgdGhlIG91dHB1dCBjb2RlIHRvIGJldHRlciBmaXQgeW91ciBwcm9qZWN0IHJlcXVpcmVtZW50cy4KLSAqKkNMSS1CYXNlZCBVdGlsaXR5Kio6IFJ1biBhcyBhIHN0YW5kYWxvbmUgY29tbWFuZC1saW5lIHRvb2wsIGFsbG93aW5nIGludGVncmF0aW9uIGludG8gYW55IHBhcnQgb2YgeW91ciBkZXZlbG9wbWVudCBwaXBlbGluZS4KCiMjIEluc3RhbGxhdGlvbgoKU3BlYzJHbyBpcyBlYXN5IHRvIGluc3RhbGwgd2l0aCBHby4gRW5zdXJlIHlvdSBoYXZlIEdvIGluc3RhbGxlZCBvbiB5b3VyIHN5c3RlbSAodmVyc2lvbiAxLjE2IG9yIGxhdGVyIGlzIHJlY29tbWVuZGVkKS4gWW91IGNhbiBpbnN0YWxsIFNwZWMyR28gZ2xvYmFsbHkgdXNpbmcgdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgc2gKZ28gaW5zdGFsbCBnaXRodWIuY29tL1JhbHBoYmFlci9zcGVjMmdvQGxhdGVzdApgYGAKClRoaXMgY29tbWFuZCB3aWxsIGluc3RhbGwgc3BlYzJnbyBnbG9iYWxseSBvbiB5b3VyIG1hY2hpbmUsIGFsbG93aW5nIHlvdSB0byBydW4gaXQgZnJvbSBhbnl3aGVyZSB3aXRoaW4geW91ciB0ZXJtaW5hbC4KCiMjIFF1aWNrIFN0YXJ0IEd1aWRlCgojIyMgU2V0dGluZyBVcCBZb3VyIFByb2plY3QKRW5zdXJlIHlvdXIgT3BlbkFQSSAzLjAgWUFNTCBmaWxlcyBhcmUgcmVhZHkuIEZvciBpbnN0YW5jZSwgeW91IG1pZ2h0IGhhdmUgYW4gQVBJIHNwZWNpZmljYXRpb24gZm9yIGEgbGVkZ2VyIGNvbXBvbmVudCB3aXRoaW4gYSBmaW5hbmNpYWwgYXBwbGljYXRpb24gbG9jYXRlZCBhdCAuL3BhdGgvdG8veW91ci9vcGVuYWkvZmlsZS55bWwuCgojIyMgR2VuZXJhdGluZyBDb2RlClRvIGdlbmVyYXRlIEdvbGFuZyBjb2RlIHVzaW5nIFNwZWMyR28sIHVzZSB0aGUgZm9sbG93aW5nIHN5bnRheDoKCmBgYHNoCnNwZWMyZ28gZ2VuZXJhdGUgLWkgLi9wYXRoL3RvL3lvdXIvb3BlbmFpL2ZpbGUueW1sIC1vIC4KYGBgCgpUaGlzIGNvbW1hbmQgd2lsbCBpbnN0cnVjdCBTcGVjMkdvIHRvOgoKLSAtaSAoLS1pbnB1dCk6IFBhdGggdG8geW91ciBPcGVuQVBJIHNwZWNpZmljYXRpb24gZmlsZS4KLSAtbyAoLS1vdXRwdXQpOiBPdXRwdXQgZGlyZWN0b3J5IHdoZXJlIHRoZSBnZW5lcmF0ZWQgR29sYW5nIGNvZGUgd2lsbCBiZSBwbGFjZWQuCgojIyMgQ3VzdG9taXppbmcgT3V0cHV0IHdpdGggVGVtcGxhdGVzClRvIGN1c3RvbWl6ZSB0aGUgb3V0cHV0LCB5b3UgY2FuIG1vZGlmeSBvciBjcmVhdGUgbmV3IEdvIHRtcGwgdGVtcGxhdGVzLgoKQSBzaW1wbGUgR28gdG1wbCB0ZW1wbGF0ZSBtaWdodCBsb29rIGxpa2UgdGhpczoKCmBgYAp7e2RlZmluZSAibWFpbiJ9fQpwYWNrYWdlIG1haW4KCmltcG9ydCAoCgkibG9nIgoJIm5ldC9odHRwIgoJLy8gV0FSTklORyEKCS8vIENoYW5nZSB0aGlzIHRvIGEgZnVsbHktcXVhbGlmaWVkIGltcG9ydCBwYXRoCgkvLyBvbmNlIHlvdSBwbGFjZSB0aGlzIGZpbGUgaW50byB5b3VyIHByb2plY3QuCgkvLyBGb3IgZXhhbXBsZSwKCS8vCgkvLyAgICBzdyAiZ2l0aHViLmNvbS9teW5hbWUvbXlyZXBvL3t7LlNvdXJjZUZvbGRlcn19IgoJLy8KCXt7LlBhY2thZ2VOYW1lfX0gIi4ve3suU291cmNlRm9sZGVyfX0iCikKCmZ1bmMgbWFpbigpIHsKCWxvZy5QcmludGxuKCJTZXJ2ZXIgc3RhcnRlZCBvbiA6MzAwMCIpCglodHRwLkhhbmRsZUZ1bmMoIi8iLCBmdW5jKHcgaHR0cC5SZXNwb25zZVdyaXRlciwgciAqaHR0cC5SZXF1ZXN0KSB7CgkJdy5Xcml0ZShbXWJ5dGUoIkhlbGxvLCBXb3JsZCEiKSkKCX0pCglsb2cuRmF0YWwoaHR0cC5MaXN0ZW5BbmRTZXJ2ZSgiOjMwMDAiLCBuaWwpKQp9Cnt7ZW5kfX0KYGBgCgpBZGp1c3QgdGhlIHRlbXBsYXRlIHRvIGluY2x1ZGUgYW55IHNwZWNpZmljIGNvbmZpZ3VyYXRpb25zIG9yIGNvZGluZyBzdGFuZGFyZHMgeW91ciBwcm9qZWN0IHJlcXVpcmVzLgoKIyMgQWR2YW5jZWQgQ29uZmlndXJhdGlvbgpZb3UgY2FuIGZ1cnRoZXIgY29uZmlndXJlIFNwZWMyR28gdGhyb3VnaCB2YXJpb3VzIGNvbW1hbmQtbGluZSBmbGFncyB0byBhZGFwdCB0aGUgdG9vbCB0byB5b3VyIHNwZWNpZmljIHdvcmtmbG93cyBhbmQgcHJlZmVyZW5jZXMuIEZvciBleGFtcGxlLCBhZGRpbmcgdmVyYm9zZSBvdXRwdXQsIGNvbmZpZ3VyaW5nIEFQSSB2ZXJzaW9uaW5nLCBvciBzcGVjaWZ5aW5nIGRpZmZlcmVudCB0ZW1wbGF0ZSBwYXRocy4KCiMjIENvbnRyaWJ1dGluZwpXZSB3ZWxjb21lIGNvbnRyaWJ1dGlvbnMgZnJvbSB0aGUgY29tbXVuaXR5ISBJZiB5b3UgaGF2ZSBpbXByb3ZlbWVudHMgb3IgYnVnIGZpeGVzLCBwbGVhc2UgZm9yayB0aGUgcmVwb3NpdG9yeSBhbmQgc3VibWl0IGEgcHVsbCByZXF1ZXN0LiBGb3IgbWFqb3IgY2hhbmdlcywgcGxlYXNlIG9wZW4gYW4gaXNzdWUgZmlyc3QgdG8gZGlzY3VzcyB3aGF0IHlvdSB3b3VsZCBsaWtlIHRvIGNoYW5nZS4KCiMjIExpY2Vuc2UKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZSAtIHNlZSB0aGUgTElDRU5TRSBmaWxlIGZvciBkZXRhaWxzLgoKIyMgQWNrbm93bGVkZ21lbnRzClRoYW5rcyB0byB0aGUgT3BlbkFQSSBJbml0aWF0aXZlIGZvciBtYWludGFpbmluZyB0aGUgc3BlY2lmaWNhdGlvbi4KQ29udHJpYnV0b3JzIHdobyBoYXZlIGhlbHBlZCBzaGFwZSBTcGVjMkdvIGludG8gYSByZWxpYWJsZSB0b29sIGZvciBkZXZlbG9wZXJzLgpXZSBob3BlIFNwZWMyR28gd2lsbCBtYWtlIHlvdXIgZGV2ZWxvcG1lbnQgcHJvY2VzcyBtb3JlIGVmZmljaWVudCBhbmQgZW5qb3lhYmxlIQoK readmeEtag: '"e10f80190da2dd2596ad46fe16db3f93db8015b3"' readmeLastModified: Fri, 26 Apr 2024 01:21:04 GMT repositoryId: 792058462 description: >- Streamline Go code generation from OpenAPI 3.0 specs and Go templates using this CLI tool. It implements the Hexagonal Architecture pattern and supports dependency injection through google/wire. created: '2024-04-25T22:33:14Z' updated: '2024-12-28T21:53:47Z' language: Shell archived: false stars: 6 watchers: 1 forks: 0 owner: Ralphbaer logo: https://avatars.githubusercontent.com/u/10503123?v=4 license: MIT repoEtag: '"fca73b89dd97bd893e30b20c656d1812c3e7fe40511309e31376e26944d5975f"' repoLastModified: Sat, 28 Dec 2024 21:53:47 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/jean85/symfony-openapi-example v3: true repositoryMetadata: base64Readme: >- IyBTeW1mb255IE9wZW5BUEkgZXhhbXBsZQpBbiBleGFtcGxlIG9mIHVzaW5nIGFuIE9wZW5BUEkgc3BlYyBpbiBhIFN5bWZvbnkgYXBwbGljYXRpb24uCgpUaGlzIHJlcG9zaXRvcnkgc2VydmVzIGFzIGNvZGUgZXhhbXBsZSBmb3IgW2EgdGFsa10oaHR0cHM6Ly9hbGVzc2FuZHJvbGFpLmRldi90YWxrcy8yMDIxLXNmZGF5LWFwaS1jb250cmFjdHMtb3BlbmFwaS1kdXJpbmctZGV2ZWxvcG1lbnQvKSB0aGF0IEkgcHJlc2VudGVkIGF0IFtzZmRheSBAIGxvY2FsaG9zdCAyMDIxXShodHRwczovLzIwMjEuc2ZkYXkuaXQpCg== readmeEtag: '"82269e6d705dc1b1fd45e154549ce1e2edd4c0c1"' readmeLastModified: Thu, 13 May 2021 07:37:34 GMT repositoryId: 349101796 description: An example of using an OpenAPI spec in a Symfony application created: '2021-03-18T14:21:00Z' updated: '2022-05-05T07:55:22Z' language: PHP archived: false stars: 6 watchers: 1 forks: 1 owner: Jean85 logo: https://avatars.githubusercontent.com/u/6729988?v=4 license: MIT repoEtag: '"11060c3bcbb88ce27a2972125a551485542918b5bc179a7ff5797b1b417e7e38"' repoLastModified: Thu, 05 May 2022 07:55:22 GMT foundInMaster: true category: - SDK - Server Implementations id: a4f86eaf0c9614329dcf62e5b3118e0c - source: openapi3 tags repository: https://github.com/misl/openapi-validator-maven-plugin v3: true repositoryMetadata: base64Readme: >- b3BlbmFwaS12YWxpZGF0b3ItbWF2ZW4tcGx1Z2luCj09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCkEgTWF2ZW4gcGx1Z2luIHRvIHN1cHBvcnQgdmFsaWRhdGlvbiBvZiBhbmQgY29udmVyc2lvbiAoWUFNTCB2cyBKU09OKSB0byBPcGVuQVBJIChzd2FnZ2VyKSBmaWxlcyAgdXNpbmcgdGhlIFtTd2FnZ2VyIFBhcnNlciBsaWJyYXJ5XShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1wYXJzZXIpLgoKSXQgbm90IG9ubHkgdmVyaWZpZXMgd2hldGhlciB0aGUgQVBJIHNwZWNpZmljYXRpb24gZmlsZXMgYXJlIHZhbGlkLCBidXQgYWxzbyBtZXJnZXMgc3BsaXR0ZWQgZmlsZXMgaW50byBhIHNpbmdsZSBZQU1MIG9yIEpTT04gZmlsZS4gVGhpcyBpcyBlc3BlY2lhbGx5IHVzZWZ1bGwgd2hlcmUgdG9vbHMgY2FuIG5vdCBjb3BlIHdpdGggSlNPTiBgJHJlZmAgcmVmZXJlbmNlcyB0byBleHRlcm5hbCBmaWxlcy4KCk1vcmUgaW5mbyBvbiBzcGxpdHRlZCBPcGVuQVBJIHNwZWNpZmljYXRpb24gZmlsZXMgcmVhZCBbRGVzaWduIFdlYiBBUElzXShodHRwczovL2FwaWhhbmR5bWFuLmlvL3dyaXRpbmctb3BlbmFwaS1zd2FnZ2VyLXNwZWNpZmljYXRpb24tdHV0b3JpYWwtcGFydC04LXNwbGl0dGluZy1zcGVjaWZpY2F0aW9uLWZpbGUvKQoKVXNhZ2UKPT09PT09PT09PT09PT09PT09PT09PT09PT09PQoKQWRkIHRvIHlvdXIgYGJ1aWxkLT5wbHVnaW5zYCBzZWN0aW9uIChkZWZhdWx0IHBoYXNlIGlzIGBnZW5lcmF0ZS1zb3VyY2VzYCBwaGFzZSkKYGBgeG1sCjxwbHVnaW4+CiAgPGdyb3VwSWQ+aXQudHJhZWNrLnRvb2xzLm9wZW5hcGk8L2dyb3VwSWQ+CiAgPGFydGlmYWN0SWQ+b3BlbmFwaS12YWxpZGF0b3ItbWF2ZW4tcGx1Z2luPC9hcnRpZmFjdElkPgogIDx2ZXJzaW9uPjEuMC4yPC92ZXJzaW9uPgogIDxleGVjdXRpb25zPgogICAgPGV4ZWN1dGlvbj4KICAgICAgPGdvYWxzPgogICAgICAgIDxnb2FsPmdlbmVyYXRlPC9nb2FsPgogICAgICA8L2dvYWxzPgogICAgICA8Y29uZmlndXJhdGlvbj4KICAgICAgICA8aW5wdXRTcGVjPiR7cHJvamVjdC5iYXNlZGlyfS9zcmMvbWFpbi9yZXNvdXJjZXMvYXBpLnlhbWw8L2lucHV0U3BlYz4KICAgICAgICA8b3V0cHV0UGF0aD4ke3Byb2plY3QuYnVpbGQuZGlyZWN0b3J5fS9nZW5lcmF0ZWQtc3BlY2lmaWNhdGlvbjwvb3V0cHV0UGF0aD4KICAgICAgICA8b3V0cHV0RmlsZW5hbWU+b3BlbmFwaTwvb3V0cHV0RmlsZW5hbWU+CiAgICAgICAgPG91dHB1dEZvcm1hdD5KU09OQU5EWUFNTDwvb3V0cHV0Rm9ybWF0PgogICAgICA8L2NvbmZpZ3VyYXRpb24+CiAgICA8L2V4ZWN1dGlvbj4KICA8L2V4ZWN1dGlvbnM+CjwvcGx1Z2luPgpgYGAKCkZvbGxvd2VkIGJ5OgoKYGBgCm12biBjbGVhbiBjb21waWxlCmBgYAoKIyMjIEdlbmVyYWwgQ29uZmlndXJhdGlvbiBwYXJhbWV0ZXJzCgpQYXJhbWV0ZXIgfCBEZXNjcmlwdGlvbiB8IFJlcXVpcmVkIHwgRGVmYXVsdAotLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLXwtLS0tLS0tLS0KYGlucHV0U3BlY2AgfCBPcGVuQVBJIFNwZWMgZmlsZSBwYXRoIHwgdHJ1ZSB8CmBvdXRwdXRQYXRoYCB8IHRhcmdldCBvdXRwdXQgcGF0aCB8IGZhbHNlIHwgYCR7cHJvamVjdC5idWlsZC5kaXJlY3Rvcnl9L2dlbmVyYXRlZC1zcGVjaWZpY2F0aW9uYApgb3V0cHV0RmlsZW5hbWVgIHwgT3V0cHV0IGZpbGVuYW1lICh3aXRob3V0IGV4dGVuc2lvbikgfCBmYWxzZSB8IG9wZW5hcGkKYG91dHB1dEZvcm1hdGAgfCBPdXRwdXQgZmlsZSBmb3JtYXQgKGBKU09OYCwgYFlBTUxgLCBgSlNPTkFORFlBTUxgKSB8IGZhbHNlIHwgSlNPTgpgZW5jb2RpbmdgIHwgZW5jb2RpbmcgdG8gdXNlIGZvciBvdXRwdXQgZmlsZXMgfCBmYWxzZSB8ICR7cHJvamVjdC5idWlsZC5zb3VyY2VFbmNvZGluZ30KYHByZXR0eVByaW50YCB8IHdoZXRoZXIgdG8gcHJldHR5IHByaW50ICh0cnVlKSBvdXRwdXQgb3Igbm90IHwgZmFsc2UgfCB0cnVlCgojIyMgU2FtcGxlIGNvbmZpZ3VyYXRpb25zCgpQbGVhc2Ugc2VlIFtleGFtcGxlIGNvbmZpZ3VyYXRpb25zXShleGFtcGxlcykgZm9yIGhvdyB0byB1c2UgdGhpcyBtYXZlbiBwbHVnaW4uCg== readmeEtag: '"72475f2d5220fa2950adcd11ddee93c7942fe953"' readmeLastModified: Sat, 19 Feb 2022 11:28:15 GMT repositoryId: 208227598 description: Plugin to validate and merge splitted OpenAPI specification files. created: '2019-09-13T08:46:19Z' updated: '2024-10-15T02:48:57Z' language: Java archived: false stars: 6 watchers: 1 forks: 3 owner: misl logo: https://avatars.githubusercontent.com/u/2040396?v=4 repoEtag: '"3490bd4d5c44ef2c38650b8832686329ababbb373d86f17851b6be0dc5f34fb6"' repoLastModified: Tue, 15 Oct 2024 02:48:57 GMT foundInMaster: true category: - Description Validators - Parsers id: 36cc2908099c7f4919d0f26f95932d7f - source: - openapi3 tags - openapi31 tags repository: https://github.com/hasanqq/openapi-starter v3_1: true id: 831fdb2a0ad33b8049fee532d28cee56 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFN0YXJ0ZXIKClVwZ3JhZGVkIHZlcnNpb24gb2YgUmVkb2NseSdzIFtvcGVuYXBpLXN0YXJ0ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L29wZW5hcGktc3RhcnRlcikgcmVwb3NpdG9yeSB3aXRoIHNvbWUgZmVhdHVyZXMgbGlrZToKCi0gICBHZW5lcmF0ZSBwb3N0bWFuIGNvbGxlY3Rpb24gb24gYnVpbGQKLSAgIFZpc3VhbCBpbXByb3ZlbWVudHMgKGZvbnQsIGJvcmRlci1yYWRpdXMsIHBhZGRpbmcgZXRjLikKLSAgIEdpdEh1YiB3b3JrZmxvdyBmb3IgbGludAoKIyMgU2NyZWVuc2hvdAoKIVtTY3JlZW5zaG90XSguLy5naXRodWIvc2NyZWVuc2hvdC5wbmcpCgojIyBDb21tYW5kcwoKTGlzdCBvZiBhdmFpbGFibGUgY29tbWFuZHMuCgp8IENvbW1hbmQgICAgICAgICAgIHwgRGVzY3JpcHRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgLS0tLS0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwKfCBgbnBtIHJ1biBkZXZgICAgICB8IFN0YXJ0IGEgZGV2IHNlcnZlciB3aXRoIGF1dG8gcmVmcmVzaC4gfAp8IGBucG0gcnVuIGJ1aWxkYCAgIHwgQnVpbGQgc3RhdGljIGZpbGVzLiAgICAgICAgICAgICAgICAgICB8CnwgYG5wbSBydW4gcHJldmlld2AgfCBQcmV2aWV3IHRoZSBidWlsZGVkIHZlcnNpb24uICAgICAgICAgIHwKfCBgbnBtIHJ1biBsaW50YCAgICB8IENoZWNrIHRoZSBPcGVuQVBJIHNwZWMuICAgICAgICAgICAgICAgfAoKIyMgQ29uZmlndXJhdGlvbgoKTGlzdCBvZiBhdmFpbGFibGUgY29uZmlndXJhdGlvbiBvcHRpb25zLgoKfCBGaWxlICAgICAgICAgICAgICAgICAgICAgfCBEb2N1bWVudGF0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8CnwgYHJlZG9jbHkueWFtbGAgICAgICAgICAgIHwgW1ZpZXddKGh0dHBzOi8vcmVkb2NseS5jb20vZG9jcy9yZWRvYy9jb25maWcpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgcG9zdG1hbi1jb252ZXJ0ZXIuanNvbmAgfCBbVmlld10oaHR0cHM6Ly9naXRodWIuY29tL3Bvc3RtYW5sYWJzL29wZW5hcGktdG8tcG9zdG1hbi9ibG9iL0hFQUQvT1BUSU9OUy5tZCkgfAoKCiMjIEV4YW1wbGVzClRoaXMgaXMgYSBsaXN0IG9mIEFQSSBkb2N1bWVudGF0aW9ucyB0aGF0IGFyZSB1c2VzIHRoaXMgdGVtcGxhdGUuCgoqIGh0dHBzOi8vYXBpLnNoaXBzZ28uY29tL2RvY3MvdjIv readmeEtag: '"ff3d90a7ad3b346a9d99d92d055ba019e2a6ead0"' readmeLastModified: Sat, 04 Jan 2025 14:18:56 GMT repositoryId: 846624183 description: >- API documentation boilerplate for redoc (OpenAPI) with some extra features like automatically generated postman collection. created: '2024-08-23T15:47:58Z' updated: '2025-01-14T13:41:46Z' language: TypeScript archived: false stars: 6 watchers: 1 forks: 0 owner: HasanQQ logo: https://avatars.githubusercontent.com/u/44672191?v=4 repoEtag: '"77bdfad6b3d82360afc5752fb51bc348d832b1415419731ac256ea116dbfab92"' repoLastModified: Tue, 14 Jan 2025 13:41:46 GMT category: Server Implementations foundInMaster: true v3: true - source: openapi3 tags repository: https://github.com/zchee/protoc-gen-openapi v3: true repositoryMetadata: base64Readme: >- IyBwcm90b2MtZ2VuLW9wZW5hcGkKClRoaXMgcGFja2FnZSBjb3B5IGZyb20gW2lzdGlvLmlvL3Rvb2xzL29wZW5hcGkvcHJvdG9jLWdlbi1vcGVuYXBpQDU5M2E0MWM3NmM1Y10oaHR0cHM6Ly9naXRodWIuY29tL2lzdGlvL3Rvb2xzL3RyZWUvNTkzYTQxYzc2YzVjODRhNGNkNTFhNGFiMGMzNDU2MzBjNWVkMzBiYS9vcGVuYXBpL3Byb3RvYy1nZW4tb3BlbmFwaSkuCgojIyBXaGF0J3MgdGhpcyBmb3I/CgpgcHJvdG9jLWdlbi1vcGVuYXBpYCBpcyBhIHBsdWdpbiBmb3IgdGhlIEdvb2dsZSBwcm90b2NvbCBidWZmZXIgY29tcGlsZXIgdG8gZ2VuZXJhdGUKb3BlbkFQSSBWMyBzcGVjIGZvciBhbnkgZ2l2ZW4gaW5wdXQgcHJvdG9idWYuIEl0IHJ1bnMgYXMgYSBgcHJvdG9jLWdlbi1gIGJpbmFyeSB0aGF0IHRoZQpwcm90b2J1ZiBjb21waWxlciBpbmZlcnMgZnJvbSB0aGUgYG9wZW5hcGlfb3V0YCBmbGFnLgoKIyMgQnVpbGQgYHByb3RvYy1nZW4tb3BlbmFwaWAKCmBwcm90b2MtZ2VuLW9wZW5hcGlgIGlzIHdyaXR0ZW4gaW4gR28sIHNvIGVuc3VyZSB0aGF0IGlzIGluc3RhbGxlZCBvbiB5b3VyIHN5c3RlbS4gWW91CmNhbiBmb2xsb3cgdGhlIGluc3RydWN0aW9ucyBvbiB0aGUgW2dvbGFuZyB3ZWJzaXRlXShodHRwczovL2dvbGFuZy5vcmcvZG9jL2luc3RhbGwpIG9yCm9uIERlYmlhbiBvciBVYnVudHUsIHlvdSBjYW4gaW5zdGFsbCBpdCBmcm9tIHRoZSBwYWNrYWdlIG1hbmFnZXI6CgpgYGBiYXNoCnN1ZG8gYXB0LWdldCBpbnN0YWxsIC15IGdvbGFuZwpgYGAKClRvIGJ1aWxkLCBmaXJzdCBlbnN1cmUgeW91IGhhdmUgdGhlIHByb3RvY29sIGNvbXBpbGVyIChwcm90b2MpOgoKYGBgYmFzaApnbyBnZXQgZ2l0aHViLmNvbS9nb2xhbmcvcHJvdG9idWYvcHJvdG8KYGBgClRvIGJ1aWxkLCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kIGZyb20gdGhpcyBwcm9qZWN0IGRpcmVjdG9yeToKCmBgYGJhc2gKZ28gYnVpbGQKYGBgCgpUaGVuIGVuc3VyZSB0aGUgcmVzdWx0aW5nIGBwcm90b2MtZ2VuLW9wZW5hcGlgIGJpbmFyeSBpcyBpbiB5b3VyIGBQQVRIYC4gQSByZWNvbW1lbmRlZCBsb2NhdGlvbgppcyBgJEhPTUUvYmluYDoKCmBgYGJhc2gKY3AgcHJvdG9jLWdlbi1vcGVuYXBpICRIT01FL2JpbgpgYGAKClNpbmNlIHRoZSBmb2xsb3dpbmcgaXMgb2Z0ZW4gaW4geW91ciBgJEhPTUUvLmJhc2hyY2AgZmlsZToKCmBgYGJhc2gKZXhwb3J0IFBBVEg9JEhPTUUvYmluOiRQQVRICmBgYAoKIyMgVXNpbmcgcHJvdG9jLWdlbi1vcGVuYXBpCgotLS0KKipUSVAqKgoKVGhlIC1JIG9wdGlvbiBpbiBwcm90b2MgaXMgdXNlZnVsIHdoZW4geW91IG5lZWQgdG8gc3BlY2lmeSBwcm90byBwYXRocyBmb3IgaW1wb3J0cy4KCi0tLQoKVGhlbiB0byBnZW5lcmF0ZSB0aGUgT3BlbkFQSSBzcGVjIG9mIHRoZSBwcm90b2J1ZiBkZWZpbmVkIGJ5IGZpbGUucHJvdG8sIHJ1bgoKYGBgYmFzaApwcm90b2MgLS1vcGVuYXBpX291dD1vdXRwdXRfZGlyZWN0b3J5IGlucHV0X2RpcmVjdG9yeS9maWxlLnByb3RvCmBgYAoKV2l0aCB0aGF0IGlucHV0LCB0aGUgb3V0cHV0IHdpbGwgYmUgd3JpdHRlbiB0bwoKCW91dHB1dF9kaXJlY3RvcnkvZmlsZS5qc29uCgpPdGhlciBzdXBwb3J0ZWQgb3B0aW9ucyBhcmU6CiogICBgcGVyX2ZpbGVgCiAgICAqICAgd2hlbiBzZXQgdG8gYHRydWVgLCB0aGUgb3V0cHV0IGlzIHBlciBwcm90byBmaWxlIGluc3RlYWQgb2YgcGVyIHBhY2thZ2UuCiogICBgc2luZ2xlX2ZpbGVgCiAgICAqICAgd2hlbiBzZXQgdG8gYHRydWVgLCB0aGUgb3V0cHV0IGlzIGEgc2luZ2xlIGZpbGUgb2YgYWxsIHRoZSBpbnB1dCBwcm90b3Mgc3BlY2lmaWVkLgoqICAgYHVzZV9yZWZgCiAgICAqICAgd2hlbiBzZXQgdG8gYHRydWVgLCB0aGUgb3V0cHV0IHVzZXMgdGhlIGAkcmVmYCBmaWVsZCBpbiBPcGVuQVBJIHNwZWMgdG8gcmVmZXJlbmNlIG90aGVyIHNjaGVtYXMuCiogICBgeWFtbGAKICAgICogICB3aGVuIHNldCB0byBgdHJ1ZWAsIHRoZSBvdXRwdXQgaXMgaW4geWFtbCBmaWxlLgo= readmeEtag: '"f31ced1b88dfdc36857ffb7531a5422beb32eb21"' readmeLastModified: Sat, 07 Sep 2019 17:43:11 GMT repositoryId: 207004447 description: >- protoc-gen-openapi is a plugin for the Google protocol buffer compiler to generate openAPI V3 spec for any given input protobuf. created: '2019-09-07T17:43:22Z' updated: '2023-04-17T23:58:47Z' language: Go archived: false stars: 6 watchers: 1 forks: 3 owner: zchee logo: https://avatars.githubusercontent.com/u/6366270?v=4 license: Apache-2.0 repoEtag: '"0cc9fbeb0f63c8c67386779e64ad98fe960dd022fe3e06c49d2c8b58be544489"' repoLastModified: Mon, 17 Apr 2023 23:58:47 GMT foundInMaster: true category: - Converters - Parsers id: 3c9503e39e26f0a16d4a9d0dd134824a - source: openapi3 tags repository: https://github.com/codecentric/reedelk-module-rest v3: true repositoryMetadata: base64Readme: >- IyBSZWVkZWxrIE1vZHVsZSBSRVNUCgpSRVNUIGNvbXBvbmVudHMgYW5kIGNvbm5lY3RvcnMgZm9yIFJlZWRlbGsgSW50ZWdyYXRpb24gUGxhdGZvcm06IGh0dHBzOi8vd3d3LnJlZWRlbGsuY29tL2RvY3VtZW50YXRpb24KCiMjIyBDcmVhdGluZyBhIHRlc3Qgb25seSBTU0wgQ2VydGlmaWNhdGUKCiogQ3JlYXRlIGEgY2VydGlmaWNhdGUgd2l0aCBTZWxmU2lnbmVkQ2VydGlmaWNhdGVCdWlsZGVyCiogY2FjZXJ0cyBmaWxlIChzdG9yZXMgdGhlIGNlcnRpZmljYXRlcykgaXMgaW4gdGhlIEphdmEgSG9tZSAoZS5nIC9MaWJyYXJ5L0phdmEvSmF2YVZpcnR1YWxNYWNoaW5lcy9qZGsxLjguMF82Ni5qZGsvQ29udGVudHMvSG9tZS9qcmUvbGliL3NlY3VyaXR5KQoqIExpc3QgY2VydGlmaWNhdGVzOiBrZXl0b29sIC1saXN0IC1rZXlzdG9yZSBjYWNlcnRzCiogRGVsZXRlIGNlcnRpZmljYXRlIChieSBnaXZlbiBhbGlhcyk6IHN1ZG8ga2V5dG9vbCAtZGVsZXRlIC1hbGlhcyByZWVkZWxrdGVzdCAta2V5c3RvcmUgY2FjZXJ0cwoqIEFkZCBjZXJ0aWZpY2F0ZSAod2l0aCBhbGlhcyk6IHN1ZG8ga2V5dG9vbCAtaW1wb3J0IC10cnVzdGNhY2VydHMgLWZpbGUgL1VzZXJzL3VzZXJuYW1lL0Rlc2t0b3AvbXljZXJ0LmNydCAtYWxpYXMgbG9jYWxob3N0IC1rZXlzdG9yZSBjYWNlcnRzIAo= readmeEtag: '"961767867872af867b86f065cd6e0e2522cf344f"' readmeLastModified: Fri, 23 Apr 2021 22:59:16 GMT repositoryId: 256693326 description: Reedelk REST module created: '2020-04-18T07:21:44Z' updated: '2025-03-17T09:09:35Z' language: Java archived: false stars: 5 watchers: 0 forks: 2 owner: codecentric logo: https://avatars.githubusercontent.com/u/1009716?v=4 license: Apache-2.0 repoEtag: '"4fc0ec88d267d7b03067f25b21e7a7c65349be949ea95cb216a319644a35e478"' repoLastModified: Mon, 17 Mar 2025 09:09:35 GMT foundInMaster: true category: Parsers id: 4e253326e6db56c587fbb0c78c612baf - source: openapi3 tags repository: https://github.com/numberfour/openapi-scala v3: true repositoryMetadata: base64Readme: >- # OpenAPI Scala

This project contains an opinionated library and SBT Plugin for Scala code generation from [OpenAPI 3.0](https://swagger.io/specification/) compliant YAML. 

This will generate data structures(i.e., schemas or types), Circe JSON serializers, and Paths (i.e., route definitions) for REST APIs.

We do not support all OpenAPI 3.0 features. For more details on what is supported, look at the [Limitations](#limitations) section.

## Modules

### sbt-openapi
The SBT module `sbt-openapi` is the interface for the generator toolchain and contains:
a) an interface to easily add the openAPI generator to a project
b) Put external openAPI resources/definitions/yaml-files next to internal definitions to make them referable. This enables
sharing artifact across multiple openAPI projects (like shared error messages/ data structures).
c) logic to load, resolve and aggregate openAPI definitions using the io.swagger.parser and generated a single combined
 openapi.json definition as output
d) trigger code generation. (`openapi-scala`) 

### openapi-scala
`openapi-scala`s main responsibility is to parse a single openAPI.json file into memory (`ast`), transform it into a
generator friendly intermediate representation (`repr`) and generate scala code (`generator`).

### openapi-lib & openapi-htt4s-lib
Libraries used by the generated code.

#### Routes

We support different types of Route generation, depending on the backend you need it for.  //TODO which backends do we have and how do we configure it?

- The generic Routes are each translated to a Scala `trait` declaring interfaces for that particular HTTP route.
- The http4s Routes translate into two files, one file with a trait for the implementation you'll need to provide and 
one file with an object apply function of which accepts mentioned implementation trait. 

The names of the functions implementing the route are either
 a) the concatenation of http method and path or
 b) the optionally more descriptive content of the `operationId` field (as specified in OAS 3.0) 
 
## Components

Components in OpenAPI are the types that can be referred to as inputs/outputs for routes. A common use of components is 
to define product types. We take these components and translate them to Scala case classes. Consider the following example:

```yaml
components:
    schemas:
        Person:
            description: My test description
            properties:
                name:
                    type: string
                weight:
                    type: number
```

The code above will be translated to following Scala code:

```scala
/**
 * My test description
**/
final case class Person(name: String, weight: Double)

object Person {
    implicit val customDecoders = deriveDecoder[Person](renaming.snakeCase, None)
    implicit val customEncoders = deriveEncoder[Person](renaming.snakeCase, true, None)

}
```

### sbt-openapi

SBT sub-project `sbt-openapi` contains an SBT plugin that allows the use of `openapi-scala` library to a given YAML file that will be loaded
and used to generate managed Scala sources.

## Quickstart

A release log of the plugin and the library can be found [here](https://github.numberfour.eu/Server/openapi-scala/releases).

To use the plugin, you first need to make it available in your SBT.

```scala
// project/plugins.sbt
addSbtPlugin("com.enfore" % "sbt-openapi" % "<openapi-scala-version>")
```

You will also need to make the library available to be able to use the generated code.

```scala
libraryDependencies += "com.enfore" %% "openapi-lib" % "<openapi-scala-version>"
```

Additionally, you will need to satisfy these library dependencies for `openapi-lib`. The provided versions are tested and made sure to work. You should be able to use any version compatible with your project though.

// TODO shouldn't we automatically add these dependencies via the SBT auto plugin via flag?
```scala
"com.beachape" %% "enumeratum"       % "1.5.13",
"com.beachape" %% "enumeratum-circe" % "1.5.20",
"io.circe"     %% "circe-derivation" % "0.11.0-M1",
"com.chuusai"  %% "shapeless"        % "2.3.3"
```

For http4s Routes you will also need: 

```scala
"org.http4s" %% s"http4s-dsl" % "0.20.0-M7",
"org.http4s" %% s"http4s-circe" % "0.20.0-M7"
```

Once the plugin is available in your project you can enable it on a given an SBT sub-project and use the setting `openAPIOutputPackage` to specify
the package name for your components. 

```scala
// build.sbt
lazy val root = (project in ("."))
    .settings(
        openAPIOutputPackage := "com.enfore.model",
        libraryDependencies += "com.enfore" %% "openapi-lib" % "<openapi-scala-version>"
    )
    .enablePlugins(OpenapiPlugin)
```

### SBT Settings

Following are the settings available for your SBT project.

**openAPISource**: Source directory for OpenAPI. Defaults to `src/main/openapi` of the project.

**openAPIOutput**: Output directory for the OpenAPI. Defaults to managed sources — `openapi`.

**openAPIOutputPacakge**: Name of the package to be used for OpenAPI components.

**routeImplementations**: A List of `com.enfore.apis.generator.RouteImplementation`, which controls which kind of routes should be generated. Find out more about [routes](#Routes)

## Refinements

OAS 3.0 allows one to set constraints on the primitive input types. We use the [Refined](https://github.com/fthomas/refined) library for Scala to reproduce these in the generated code. This means that you do not need to manually setup validations for any refinements and constraints in the inputs for your API. This, however, also means that you will have to return values with refinements if there are any in the return types.

In the example code below, we also generate a constructor for a value over which refinements exist. These refinements exist inside an object called `RefinementConstructors` that is further nested in the companion object for the generated case class. The names of the constructors will be the same as the values for which the refinements exist.

```scala
import shapeless._

final case class MyTestOutput(name: String Refined AllOf[MinSize[W.`1`.T] :: MaxSize[W.`256`.T] :: HNil])

object MyTestOutput {
    object RefinementConstructors {
        val name = String Refined AllOf[MinSize[W.`1`.T] :: MaxSize[W.`256`.T] :: HNil]

    }
}
```

## Read Only Properties

OpenAPI 3.0 supports marking properties in objects as read-only. This checks that the property is available if the type is used as the output of a route, but not for inputs.

We model this in Scala by generating two types in case a field is marked read-only. The generator will then automatically use the right type depending on weather the object is used as input or output.

```yaml
MyObject:
    type: object
    properties:
        id:
            type: string
            readOnly: true
        body:
            type: string
```

This will generate two Scala classes that are as follows:

```scala
final case class MyObjectRequest(body: String)

final case class MyObject(id: String, body: String)
```

## Unions & Enums

OpenAPI 3.0 uses `oneOf` types to represent unions or enumerations. In Scala, there is no way to have proper Algebraic Data Types (ADTs). While products can be represented using case classes, the only way to represent unions is to create classes that extend a given trait.

This, however, makes defining arbitrary unions very difficult. Therefore, we use [Shapeless' discriminated unions](https://github.com/milessabin/shapeless/wiki/Feature-overview:-shapeless-2.0.0#coproducts-and-discriminated-unions) to represent `anyOf` types in Scala.

```scala
final case class MyUnionWrapper(value: Union)

object MyUnionWrapper {
    type Union = Item1 :+: Item2 :+: CNil

    object jsonConversions extends Poly1 {
        implicit def caseItem1 = at[SingleTargetConversion](_.asJson.deepMerge(Json.obj("@type" -> Json.fromString("Item1"))))
        implicit def caseItem2 = at[MultiTargetConversion](_.asJson.deepMerge(Json.obj("@type" -> Json.fromString("Item2"))))

    }

    implicit val customerEncoders = new Encoder[MyUnionWrapper] {
      def apply(a: Conversion): Json = {
        (a.value map jsonConversions).unify
      }
    }

  implicit val customDecoder: Decoder[MyUnionWrapper] = new Decoder[Conversion] {
    def apply(c: HCursor): Decoder.Result[MyUnionWrapper] = {
      val output = c.downField("@type").as[String] match {
        case Right("Item1") => c.value.as[Item1].map(Coproduct[Union](_))
        case Right("Item2") => c.value.as[Item2].map(Coproduct[Union](_))
        case _ => Left(DecodingFailure.apply("Type information not available", List(CursorOp.DownField("@type"))))
      }
      output.map(MyUnionWrapper.apply)
    }
  }

}
```

Using these arbitrary unions is easier than it first appears. To create a value of type `MyUnionWrapper`, which is defined using `oneOf` in OpenAPI, we can use the following code:

```scala
import shapeless._
val item1Union = MyUnionWrapper(Coproduct[MyUnionWrapper.Union](item1))
val item2Union = MyUnionWrapper(Coproduct[MyUnionWrapper.Union](item2))
```

Handling the values of that are already wrapped and mapping them to something is slightly more work. We use the `Poly1`  from Shapeless to create functions that can map over a co-product.

```scala
object convertToT extends Poly1 {
    implicit val oneToT = at[Item1](x => oneToT(x))
    implicit val twoToT = at[Item2](x => twoToT(x))
}

val convertedItem: T = item1Union.fold(convertToT)
```

Our `convertToT` function converts every possible value to a type `T` and then we unify them. Since we know that our co-product will contain exactly one value of the possible given types, we can be certain that we will end up with at least one value of type `T`.

## Limitations

This generator is highly opinionated and skips over certain features defined OpenAPI 3.0 specification for the sake of API consistency and compatibility with modern typed languages.

### Limitations

- We do not support any anonymous types for representing HTTP requests and responses.
- We do not support nested type definitions. Therefore, any objects that are used must be defined in the components and must not contain
  any other nested objects. Nested arrays, however, are supported.
- We do not support more than one type for Content-Type header for incoming requests. Multiple types corresponding to their encodings are, however, available with an Accept-Encoding header on a request.
- We do not support `anyOf` types from OpenAPI 3.0. Suggestions as to how the behaviour for the same should be represented in generated Scala code are welcome.
- We do not support inlined schema definition for input and output object types in route definitions in OpenAPI input files. Therefore, these must be defined as components and referred.
- We do not support type aliasing or rich typing. For instance, if you have a primitive type with a particular set of refinements, you would have to point out those refinements everywhere with those refinements instead of referring them from a definition.
 readmeEtag: '"c70fd0cbc59430cfe21cb54d59302cbb69f78e9d"' readmeLastModified: Tue, 18 Aug 2020 13:29:31 GMT repositoryId: 259959923 description: >- An opinionated library and SBT plugin for generating Scala code from OpenAPI 3. created: '2020-04-29T15:08:59Z' updated: '2023-08-25T08:39:34Z' language: Scala archived: false stars: 6 watchers: 4 forks: 4 owner: NumberFour logo: https://avatars.githubusercontent.com/u/319804?v=4 license: MIT repoEtag: '"3f4189de565c20b1025a901ec58be43d6e52d0ff77aa4c654e1f0f90afc94551"' repoLastModified: Fri, 25 Aug 2023 08:39:34 GMT foundInMaster: true id: 59ff09178d2859e3ebe7eea756af30b7 - source: openapi3 tags repository: https://github.com/dsietz/actix-web-openapi v3: true repositoryMetadata: base64Readme: >- IyBBY3RpeCBXZWIgT3BlbkFQSQohW1ZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvdmVyc2lvbi0wLjAuMS1ibHVlLnN2ZykKIVtNSVQvQXBhY2hlLTIuMF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVCUyRkFwYWNoZS0tMi4wLWJsdWUuc3ZnKQoKLS0tCgojIyBVc2FnZQpgYGAKZXh0ZXJuIGNyYXRlIGFjdGl4X3dlYl9vcGVuYXBpOwoKZm4gbWFpbigpIHsKICAgIG1hdGNoIGFjdGl4X3dlYl9vcGVuYXBpOjpmcm9tX3BhdGgoIi4vZGF0YS92My4wL3BldHN0b3JlLnlhbWwiKSB7CiAgICAgICAgT2soc3BlYykgPT4gewogICAgICAgICAgICBsZXQgc2VydmVycyA9IHNwZWMuc2VydmVycy51bndyYXAoKTsKCiAgICAgICAgICAgIGZvciBzZXJ2ZXIgaW4gc2VydmVycy5pdGVyKCkgewogICAgICAgICAgICAgICAgbWF0Y2ggc2VydmVyLnRvX2NsaWVudF9yZXF1ZXN0KCkuZmluaXNoKCkgewogICAgICAgICAgICAgICAgICAgIE9rKGNsaWVudCkgPT4gewogICAgICAgICAgICAgICAgICAgICAgICBhc3NlcnRfZXEhKGNsaWVudC51cmkoKS5zY2hlbWVfc3RyKCksIFNvbWUoImh0dHAiKSk7CiAgICAgICAgICAgICAgICAgICAgICAgIGFzc2VydF9lcSEoY2xpZW50LnVyaSgpLmhvc3QoKSwgU29tZSgicGV0c3RvcmUuc3dhZ2dlci5pbyIpKTsKICAgICAgICAgICAgICAgICAgICAgICAgYXNzZXJ0X2VxIShjbGllbnQudXJpKCkucG9ydF91MTYoKSwgTm9uZSk7CiAgICAgICAgICAgICAgICAgICAgICAgIGFzc2VydF9lcSEoY2xpZW50LnVyaSgpLnBhdGgoKSwgIi92MSIpOwogICAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICAgICAgRXJyKF9lcnIpID0+IGFzc2VydCEoZmFsc2UpLAogICAgICAgICAgICAgICAgfSAKICAgICAgICAgICAgfQogICAgICAgIH0sCiAgICAgICAgRXJyKF9lcnIpID0+IGFzc2VydCEoZmFsc2UpLAogICAgfQp9CmBgYAoKIyMgTGljZW5zZQpUaGlzIHNvbHR1aW9uIHVzZXMgb3BlbiBzb3VyY2UgcHJvZHVjdHMgYW5kIGlzIGNvbnN0cmFpbmVkIHRvIHRoZSBpbmhlcml0ZWQgbGljZW5zZSBhZ3JlZW1lbnRzLgoKLSBBcGFjaGUgTGljZW5zZSwgVmVyc2lvbiAyLjAsIChMSUNFTlNFLUFQQUNIRSBvciBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjApCi0gTUlUIExpY2Vuc2UgKExJQ0VOU0UtTUlUIG9yIGh0dHA6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQpCg== readmeEtag: '"5bf6053932aeaec58b5fefa6105ae156711ffe40"' readmeLastModified: Mon, 22 Apr 2019 16:27:58 GMT repositoryId: 182823521 description: >- Contract testing of an OpenAPI document for REST service running on Actix Web created: '2019-04-22T16:13:04Z' updated: '2025-06-03T09:55:56Z' language: Rust archived: false stars: 6 watchers: 0 forks: 1 owner: dsietz logo: https://avatars.githubusercontent.com/u/21245551?v=4 license: Apache-2.0 repoEtag: '"0b76d65c6560ffffc710ca9c43c6e542992be1c29318804d56567ad49a19c2f3"' repoLastModified: Tue, 03 Jun 2025 09:55:56 GMT foundInMaster: true category: - Documentation - Parsers id: d1ef4b9f79b9d5f5499d73404fd347cd - source: openapi3 tags repository: https://github.com/seipan/bulma v3: true id: 5a221309bc76af6ae48f51f2e5f6c5a3 repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KCiFbTGFzdCBjb21taXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhc3QtY29tbWl0L3NlaXBhbi9ibHVtYT9zdHlsZT1mbGF0LXNxdWFyZSkKIVtSZXBvc2l0b3J5IFN0YXJzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9zdGFycy9zZWlwYW4vYmx1bWE/c3R5bGU9ZmxhdC1zcXVhcmUpCiFbSXNzdWVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMvc2VpcGFuL2JsdW1hP3N0eWxlPWZsYXQtc3F1YXJlKQohW09wZW4gSXNzdWVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtcmF3L3NlaXBhbi9ibHVtYT9zdHlsZT1mbGF0LXNxdWFyZSkKWyFbZ29dKGh0dHBzOi8vZ2l0aHViLmNvbS9zZWlwYW4vbG9naG9vay9hY3Rpb25zL3dvcmtmbG93cy9nby55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3NlaXBhbi9sb2dob29rL2FjdGlvbnMvd29ya2Zsb3dzL2dvLnltbCkKCjxpbWcgc3JjPSJodHRwczovL2kucGluaW1nLmNvbS83MzZ4LzdkLzVlLzBkLzdkNWUwZDhjZWE0NTJmZTkxOGUyNmYxZWIxNGVhODdiLmpwZyIgYWx0PSJleWVjYXRjaCIgaGVpZ2h0PSIyMDAiPgoKIyBidWxtYQoKIDpwdW5jaDogQ0xJIHRvb2wgdG8gcGFyc2UgT3BlbkFQSSBhbmQgc3RyZXNzIHRlc3QgZWFjaCBlbmRwb2ludC4gOnB1bmNoOgoKPGJyPgo8YnI+CgoKPC9kaXY+CgojIyBJbnN0YWxsCmBgYApnbyBpbnN0YWxsIGdpdGh1Yi5jb20vc2VpcGFuL2J1bG1hQGxhdGVzdApgYGAKCiMjIFVzYWdlCmBgYApVc2FnZToKICBibHVtYSBbZmxhZ3NdCgpGbGFnczoKICAtYiwgLS1iYXNlcGF0aCBzdHJpbmcgICAgIEJhc2VVUkwgZm9yIHN0cmVzcyB0ZXN0CiAgLWQsIC0tZHVyYXRpb24gZHVyYXRpb24gICBzdHJlc3MgdGVzdCBkdXJhdGlvbiAoZGVmYXVsdCAxbnMpCiAgLXAsIC0tZmlsZXBhdGggc3RyaW5nICAgICBGaWxlUGF0aCBmb3IgUGFyc2luZyBPcGVuQVBJCiAgLWYsIC0tZnJlcXVlbmN5IGludCAgICAgICBzdHJlc3MgdGVzdCBmcmVxdWVuY3kgKGRlZmF1bHQgMSkKICAtaCwgLS1oZWxwICAgICAgICAgICAgICAgIGhlbHAgZm9yIGJsdW1hCmBgYAoKIyMjIyBgLWJhc2VwYXRoYAoKVGhpcyBpcyB0aGUgYmFzZSBVUkwgd2hlcmUgeW91IHdhbnQgdG8gYXBwbHkgdGhlIGxvYWQuCgojIyMjIGAtZHVyYXRpb25gCgpUaGlzIGlzIHN0cmVzcyB0ZXN0IGR1cmF0aW9uLiAoZGVmYXVsdCAxbnMpCgojIyMjIGAtZmlsZXBhdGhgCgpUaGlzIGlzIEZpbGVQYXRoIGZvciBQYXJzaW5nIE9wZW5BUEkuCgojIyMjIGAtZnJlcXVlbmN5YAoKVGhpcyBpcyAgc3RyZXNzIHRlc3QgZnJlcXVlbmN5LiAoZGVmYXVsdCAxKQoKIyMgRXhhbXBsZQpgYGAKYnVsbWEgLS1wYXRoPXRlc3RkYXRhL2hlYWx0aC55YW1sIC0tYmFzZT1odHRwOi8vbG9jYWxob3N0OjgwODAKYGBgCgpgYGAKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1idWxtYSBhdHRhY2sgc3RhcnQtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tdmVnZXRhIGF0dGFjayB0byBodHRwOi8vbG9jYWxob3N0OjgwODAvaGVhbHRoLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KdmVnZXRhIGF0dGFjayB0byBtZXRob2Q6IEdFVApwYXRoIFN0YXR1c0NvZGU6IG1hcFsyMDA6MV0KCm1heCBwZXJjZW50aWxlOiAzMzMuNjIwNG1zCm1lYW4gcGVyY2VudGlsZTogMzMzLjYyMDRtcwp0b3RhbCBwZXJjZW50aWxlOiAzMzMuNjIwNG1zCjk5dGggcGVyY2VudGlsZTogMzMzLjYyMDRtcwoKIGVhcmxpZXN0OiAyMDIzLTA5LTIzIDA1OjIzOjA1LjE4Mzk3ODQgKzA5MDAgSlNUIG09KzEuMDI5MzAxNTAxCiBsYXRlc3Q6IDIwMjMtMDktMjMgMDU6MjM6MDUuMTgzOTc4NCArMDkwMCBKU1QgbT0rMS4wMjkzMDE1MDEKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS1idWxtYSBhdHRhY2sgZmluaXNoLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpgYGAKCiMjIExpY2Vuc2UKQ29kZSBsaWNlbnNlZCB1bmRlciAKW3RoZSBNSVQgTGljZW5zZV0oaHR0cHM6Ly9naXRodWIuY29tL3NlaXBhbi9idWxtYS9ibG9iL21haW4vTElDRU5TRSkuCg== readmeEtag: '"9c5a23f1a6d55933976941fe9c6e0cf8889a5e63"' readmeLastModified: Tue, 06 Aug 2024 15:30:11 GMT repositoryId: 671336325 description: ':punch: CLI tool to parse OpenAPI and stress test each endpoint. :punch:' created: '2023-07-27T05:04:35Z' updated: '2025-11-04T18:22:44Z' language: Go archived: false stars: 6 watchers: 1 forks: 1 owner: seipan logo: https://avatars.githubusercontent.com/u/88176012?v=4 license: MIT repoEtag: '"ea82aa6a1dee5151a10891879bd03e76362422859444dbca9e7ac4845c0cfefa"' repoLastModified: Tue, 04 Nov 2025 18:22:44 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/simplesolutiondev/spring-boot-api-doc v3: true repositoryMetadata: base64Readme: IyBzcHJpbmctYm9vdC1hcGktZG9jCmh0dHBzOi8vc2ltcGxlc29sdXRpb24uZGV2Lwo= readmeEtag: '"e479106ea25158730654b2bbeff2599cbe923fd0"' readmeLastModified: Wed, 03 Feb 2021 17:10:49 GMT repositoryId: 335696724 description: >- Spring Boot RESTful API Documentation with OpenAPI 3.0 and Swagger-UI using springdoc-openapi created: '2021-02-03T17:06:17Z' updated: '2025-10-26T17:58:48Z' language: Java archived: false stars: 6 watchers: 2 forks: 1 owner: simplesolutiondev logo: https://avatars.githubusercontent.com/u/48798703?v=4 repoEtag: '"2b4b0921b74c89ac03c938bb319af30335223537826dd45ae3bb3aaa4598434a"' repoLastModified: Sun, 26 Oct 2025 17:58:48 GMT foundInMaster: true category: - Low-level Tooling - Server - Server Implementations id: 58570b657a0d147e81d5a480138f034d - source: openapi3 tags repository: https://github.com/crazyoptimist/nest-starter v3: true id: e635e2fce4b0fcdb8d54a80a818fa0d1 repositoryMetadata: base64Readme: >- IyBOZXN0SlMgU3RhcnRlcgoKWyFbYnVpbGQgJiB0ZXN0XShodHRwczovL2dpdGh1Yi5jb20vY3JhenlvcHRpbWlzdC9uZXN0LXN0YXJ0ZXIvYWN0aW9ucy93b3JrZmxvd3MvYnVpbGQueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9jcmF6eW9wdGltaXN0L25lc3Qtc3RhcnRlci9hY3Rpb25zL3dvcmtmbG93cy9idWlsZC55bWwpCgpCdWlsZCBhIG1vZHVsYXIgUkVTVCBBUEkgd2l0aCBOZXN0SlMgZnJhbWV3b3JrLgoKIyMgVGFibGUgb2YgQ29udGVudHMKCi0gW1doYXQgSXMgVGhpcz9dKCN3aGF0LWlzLXRoaXMpCi0gW0dldHRpbmcgU3RhcnRlZF0oI2dldHRpbmctc3RhcnRlZCkKLSBbRGF0YWJhc2UgTWlncmF0aW9uc10oI2RhdGFiYXNlLW1pZ3JhdGlvbnMpCiAgLSBbR2VuZXJhdGUgYSBOZXcgTWlncmF0aW9uXSgjZ2VuZXJhdGUtYS1uZXctbWlncmF0aW9uKQogIC0gW1J1biBQZW5kaW5nIE1pZ3JhdGlvbnNdKCNydW4tcGVuZGluZy1taWdyYXRpb25zKQogIC0gW1JldmVydCBNaWdyYXRpb25zXSgjcmV2ZXJ0LW1pZ3JhdGlvbnMpCi0gW1Rlc3RzXSgjdGVzdHMpCi0gW0Vudmlyb25tZW50IENvbmZpZ3VyYXRpb25dKCNlbnZpcm9ubWVudC1jb25maWd1cmF0aW9uKQotIFtBUEkgRG9jdW1lbnRhdGlvbl0oI2FwaS1kb2N1bWVudGF0aW9uKQotIFtBdXRoZW50aWNhdGlvbl0oI2F1dGhlbnRpY2F0aW9uKQotIFtMaWNlbnNlXSgjbGljZW5zZSkKLSBbTWFpbnRhaW5lcnNdKCNtYWludGFpbmVycykKCiMjIFdoYXQgSXMgVGhpcz8KClRoaXMgaXMgYSBOZXN0SlMgUkVTVGZ1bCBBUEkgc3RhcnRlciB0aGF0IGluY2x1ZGVzIEpXVCBhdXRoZW50aWNhdGlvbiwgT3BlbkFQSSAzIGRvY3VtZW50YXRpb24sIGFuZCBUeXBlT1JNIGludGVncmF0aW9uLgoKVGhlIHJlYXNvbiBJIG1haW50YWluIGl0IGlzIGJlY2F1c2UgdXNpbmcgdGhpcyB0ZW1wbGF0ZSBhbGxvd3MgeW91IHRvIHF1aWNrbHkgc2V0IHVwIGEgcHJvZHVjdGlvbi1yZWFkeSBSRVNUZnVsIEFQSSB3aXRoIG1pbmltYWwgYm9pbGVycGxhdGUgY29kZS4KCiMjIEdldHRpbmcgU3RhcnRlZAoKQ3JlYXRlIGEgZG90ZW52IGZpbGUgYW5kIGZpbGwgaXQgb3V0IHdpdGggdGhlIGFwcHJvcHJpYXRlIHZhbHVlcy4KCmBgYGJhc2gKY3AgLmVudi5leGFtcGxlIC5lbnYKYGBgCgpJbnN0YWxsIGRlcGVuZGVuY2llcwoKYGBgYmFzaApucG0gaW5zdGFsbApgYGAKClJ1biB0aGUgYXBwbGljYXRpb24KCmBgYGJhc2gKbnBtIHJ1biBkZXYKYGBgCgpITVIoSG90IE1vZHVsZSBSZWxvYWQpIGlzIGNvbmZpZ3VyZWQgd2l0aCB3ZWJwYWNrLgoKIyMgRGF0YWJhc2UgTWlncmF0aW9ucwoKSWYgeW91IGRvbid0IHJlcXVpcmUgaW5jcmVtZW50YWwgZGF0YWJhc2UgbWlncmF0aW9ucywgc2V0IHRoZSBEQl9TWU5DIGVudmlyb25tZW50IHZhcmlhYmxlIHRvIHRydWUuIFRoaXMgd2lsbCBjb250aW51b3VzbHkgc3luY2hyb25pemUgdGhlIGRhdGFiYXNlIHN0cnVjdHVyZSB3aXRoIHRoZSBzY2hlbWEgZGVmaW5lZCBpbiB0aGUgY29kZWJhc2UuCgpfX0hvd2V2ZXIsIERPIE5PVCBzZXQgREJfU1lOQyB0byB0cnVlIGluIGEgcHJvZHVjdGlvbiBlbnZpcm9ubWVudCwgYXMgZG9pbmcgc28gbWF5IHJlc3VsdCBpbiBkYXRhIGxvc3MhX18KCiMjIyMgR2VuZXJhdGUgYSBOZXcgTWlncmF0aW9uCgpNYWtlIHN1cmUgdG8gdXNlIG5wbSBmb3IgdGhlIGZvbGxvd2luZyBjb21tYW5kcywgYXMgeWFybiBkb2VzIG5vdCBzdXBwb3J0IGAkbnBtX2NvbmZpZ19uYW1lYC4KCmBgYGJhc2gKbnBtIHJ1biBtaWdyYXRpb246Z2VuZXJhdGUgLS1uYW1lPUFkZEFnZUNvbHVtblRvVXNlcgpgYGAKCk1pZ3JhdGlvbiBmaWxlcyBhcmUgbG9jYXRlZCBpbiB0aGUgYHNyYy9taWdyYXRpb25zYCBkaXJlY3RvcnkuCgojIyMjIFJ1biBQZW5kaW5nIE1pZ3JhdGlvbnMKCmBgYGJhc2gKbnBtIHJ1biBtaWdyYXRpb246cnVuCmBgYAoKVXNpbmcgRG9ja2VyOgoKYGBgYmFzaApkb2NrZXIgZXhlYyBuZXN0IG5wbSBydW4gbWlncmF0aW9uOnJ1bgpgYGAKCiMjIyMgUmV2ZXJ0IE1pZ3JhdGlvbnMKClJldmVydCB0aGUgbGFzdCBtaWdyYXRpb24KCmBgYGJhc2gKbnBtIHJ1biBtaWdyYXRpb246cmV2ZXJ0CmBgYAoKIyMgVGVzdHMKCmBgYGJhc2gKIyB1bml0IHRlc3RzCm5wbSBydW4gdGVzdAoKIyBlMmUgdGVzdHMKbnBtIHJ1biB0ZXN0OmUyZQoKIyB0ZXN0IGNvdmVyYWdlCm5wbSBydW4gdGVzdDpjb3YKYGBgCgojIyBFbnZpcm9ubWVudCBDb25maWd1cmF0aW9uCgpgQG5lc3Rqcy9jb25maWdgIGhhcyBiZWVuIHVzZWQsIHNvIHlvdSBjYW4ganVzdCBpbmplY3QgYENvbmZpZ1NlcnZpY2VgIHRvIHJlYWQgZW52aXJvbm1lbnQgdmFyaWFibGVzIGZyb20gdGhlIGRvdGVudiBmaWxlLgoKIyMgQVBJIERvY3VtZW50YXRpb24KCk9wZW5BUEkgMy4wIGhhcyBiZWVuIGNvbmZpZ3VyZWQsIGFuZCB0aGUgQVBJIGRvY3VtZW50YXRpb24gaXMgaG9zdGVkIGF0IGBCQVNFX1VSTC9hcGkvZG9jc2AuCgojIyBBdXRoZW50aWNhdGlvbgoKSldUIGF1dGhlbnRpY2F0aW9uIGhhcyBiZWVuIGNvbmZpZ3VyZWQuCgojIyBMaWNlbnNlCgpNSVQKCiMjIE1haW50YWluZXJzCgpbY3JhenlvcHRpbWlzdF0oaHR0cHM6Ly9jcmF6eW9wdGltaXN0Lm5ldCkK readmeEtag: '"aa067beb2e72ff105e5ee7485f97fa18ad10f79d"' readmeLastModified: Thu, 27 Jun 2024 21:11:20 GMT repositoryId: 360806446 description: NestJS RESTful API starter for minimalists created: '2021-04-23T07:53:38Z' updated: '2024-12-26T17:54:09Z' language: TypeScript archived: false stars: 6 watchers: 1 forks: 5 owner: crazyoptimist logo: https://avatars.githubusercontent.com/u/55074937?v=4 license: MIT repoEtag: '"aad50893820aafea217eb394f20852516387bdc15549a85e4fba60b3ae9f93fd"' repoLastModified: Thu, 26 Dec 2024 17:54:09 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cjp2600/cutos v3: true repositoryMetadata: base64Readme: >- IyBDdXRvcyAtIChDdXJsIHRvIFN3YWdnZXIpIAoKQ3V0b3MgcHJvdmlkZXMgYSB0ZXJtaW5hbCBVSSBmb3Igd3JpdGluZyBPcGVuQVBJMyBhbmQgZG9jdW1lbnRhdGlvbiB3aXRoIHRoZSBlYXN5IHdheSBieSBjb3B5aW5nINGBdXJsIHF1ZXJpZXMgYW5kIGpzb24gcmVzcG9uc2VzLgpUaGUgcHJvamVjdCBpcyBpbiB0aGUgZWFybHkgYWxwaGEgdmVyc2lvbiBhbmQgaXMgbm90IGEgY2FuZGlkYXRlIGZvciB0aGUgbWFpbiBkZXZlbG9wZXIgaW5zdGFsbGF0aW9uLgpUaGUgcHJvamVjdCBpcyBhIHdlZWtlbmQgcHJvamVjdCBhbmQgd2FzIGNyZWF0ZWQgZXhjbHVzaXZlbHkgdG8gbWVldCB0aGUgbmVlZHMgb2YgdGhlIGF1dGhvci4KCi0tLQoKCiMjIEluc3RhbGxhdGlvbgoKY3V0b3MgaXMgYXZhaWxhYmxlIG9uIExpbnV4LCBtYWNPUyBhbmQgV2luZG93cyBwbGF0Zm9ybXMuCgoqIEJpbmFyaWVzIGZvciBMaW51eCwgV2luZG93cyBhbmQgTWFjIGFyZSBhdmFpbGFibGUgYXMgdGFyYmFsbHMgaW4gdGhlIFtyZWxlYXNlXShodHRwczovL2dpdGh1Yi5jb20vY2pwMjYwMC9jdXRvcy9yZWxlYXNlcykgcGFnZS4KKiBVc2UgYGdvIGdldGAgdG8gaW5zdGFsbCB0aGUgbGF0ZXN0IHZlcnNpb24gb2YgdGhlIHRvb2xzLiBUaGlzIGNvbW1hbmQgd2lsbCBpbnN0YWxsIHRoZSBjdXRvcyB3aXRoIGRlcGVuZGVuY2llczoKYGBgc2hlbGwKZ28gZ2V0IC11IGdpdGh1Yi5jb20vY2pwMjYwMC9jdXRvcwpgYGAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIAotLS0KCiMjIENyZWF0ZSBuZXcgc3dhZ2dlciBkb2N1bWVudApJZiBhIGZpbGUgd2l0aCBkb2N1bWVudGF0aW9uIGlzIG5vdCBjcmVhdGVkLCBpdCBzaG91bGQgYmUgY3JlYXRlZCB1c2luZyB0aGUgYGluaXRgIGNvbW1hbmQuIApgYGBzaGVsbApjYXRvcyBpbml0IHN3YWdnZXIuanNvbgpgYGAKPiBUaGUgdXRpbGl0eSB3aWxsIGFzayBmb3IgYmFzaWMgZGF0YSB0byBjcmVhdGUgYSBkb2N1bWVudGF0aW9uIGZpbGUKCiMjIEFwcGVuZCBuZXcgcGF0aCBmcm9tIGN1cmwKSWYgdGhlIGZpbGUgaGFzIGFscmVhZHkgYmVlbiBjcmVhdGVkIG9yIHdhcyBhbHJlYWR5IGNyZWF0ZWQsIHlvdSBjYW4gYWRkIGEgbmV3IHBhdGggdXNpbmcgdGhlIGZvbGxvd2luZyBjb21tYW5kOgpgYGBzaGVsbApjdXRvcyBhZGQgY3VybCBzd2FnZ2VyLmpzb24KYGBgCj4gVGhlIHV0aWxpdHkgd2lsbCBvcGVuIGEgbmV3IGxpc3RlbmVyIG9mIHlvdXIgY2xpcGJvYXJkIHRvIGNyZWF0ZSBhIHBhdGNoLgo+IFlvdSBjYW4gZWFzaWx5IHVzZSBwcm9ncmFtcyB0aGF0IGFsbG93IHlvdSB0byBleHBvcnQgdG8gdGhlIGN1cmwgZm9ybWF0LiBTdWNoIGFzIFtJbnNvbW5pYV0oaHR0cHM6Ly9pbnNvbW5pYS5yZXN0L2Rvd25sb2FkLyksIFtQb3N0bWFuXShodHRwczovL3d3dy5wb3N0bWFuLmNvbS8pIG9yIEJyb3dzZXIKCiMjIENvbmZpZ3VyYXRpb24KQnkgZGVmYXVsdCB0aGUgdXRpbGl0eSBsb29rcyBmb3IgdGhlIGNvbmZpZ3VyYXRpb24gZmlsZSBpbiB0aGlzIHBhdGggZGVmYXVsdCBpcyAqKiRIT01FLy5jdXRvcy55YW1sKioKQnV0IHlvdSBjYW4gYWx3YXlzIHJld3JpdGUgdGhlIHBhdGggdG8gdGhlIGNvbmZpZ3VyYXRpb24gZmlsZSBleHBsaWNpdGx5IGluIHRoZSBmb2xsb3dpbmcgcGFyYW1ldGVyICoqLS1jb25maWc9KioKYGBgc2hlbGwKY3V0b3MgYWRkIGN1cmwgc3dhZ2dlci5qc29uIC0tY29uZmlnPWNvbmZpZy55bWwKYGBgCiMjIEV4YW1wbGUgY29uZmlndXJhdGlvbjoKYGBgWUFNTAojCiMgQmFzaWMgYXBwbGljYXRpb24gc2V0dGluZ3MKIwphcHBsaWNhdGlvbl9zZXR0aW5nczoKCiAgIyBMaXN0IG9mIGhlYWRlcnMgcGFyYW1zIHRvIGJlIHNraXBwZWQKICBza2lwcGVkX2hlYWRlcnM6CiAgICAtIGFjY2VwdAogICAgLSBhdXRob3JpdHkKICAgIC0gdXNlci1hZ2VudAogICAgLSBjb250ZW50LXR5cGUKICAgIC0gb3JpZ2luCiAgICAtIHNlYy1mZXRjaC1zaXRlCiAgICAtIHNlYy1mZXRjaC1tb2RlCiAgICAtIHNlYy1mZXRjaC1kZXN0CiAgICAtIHJlZmVyZXIKYGBgCiAK readmeEtag: '"66f8e98928e5f99efbd5bc17255e903bb185b039"' readmeLastModified: Wed, 03 Aug 2022 11:04:47 GMT repositoryId: 309656196 description: The easiest way to write OpenAPI 3 documentation using curl copy created: '2020-11-03T10:52:31Z' updated: '2024-02-13T19:55:53Z' language: Go archived: false stars: 6 watchers: 1 forks: 0 owner: cjp2600 logo: https://avatars.githubusercontent.com/u/1980974?v=4 license: Apache-2.0 repoEtag: '"be28e425825603c4d3671f947d3087409bfc98cf2a400e4d79811645349a734e"' repoLastModified: Tue, 13 Feb 2024 19:55:53 GMT foundInMaster: true category: - Testing - Server Implementations id: b7ea64a2b5378c448b9a74deb95fbde8 - source: openapi3 tags repository: https://github.com/0xtheprodev/nestjs-clean-example v3: true id: cd38c02141d5f9c5903ff75154a4f9aa repositoryMetadata: base64Readme: >- IyBuZXN0anMtY2xlYW4tZXhhbXBsZQoKWyFbVHlwZVNjcmlwdF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS90eXBlc2NyaXB0LSUyMzAwN0FDQy5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPXR5cGVzY3JpcHQmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly90eXBlc2NyaXB0bGFuZy5vcmcvKQpbIVtOb2RlSlNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvbm9kZS5qcy02REE1NUY/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPW5vZGUuanMmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly9ub2RlanMub3JnLykKWyFbTmVzdEpTXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL25lc3Rqcy0lMjNFMDIzNEUuc3ZnP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1uZXN0anMmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly9uZXN0anMuY29tLykKWyFbRmFzdGlmeV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9mYXN0aWZ5LSUyMzAwMDAwMC5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWZhc3RpZnkmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly93d3cuZmFzdGlmeS5pby8pClshW09wZW5BUEldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvb3BlbmFwaS02QkE1Mzk/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPW9wZW5hcGktaW5pdGlhdGl2ZSZsb2dvQ29sb3I9ZmZmKV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykKWyFbR3JhcGhRTF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS8tR3JhcGhRTC1FMTAwOTg/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWdyYXBocWwmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly9ncmFwaHFsLm9yZy8pClshW0VTTGludF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9FU0xpbnQtNEIzMjYzP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1lc2xpbnQmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly9lc2xpbnQub3JnKQpbIVtQcmV0dGllcl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9QcmV0dGllci1ibGFjaz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289cHJldHRpZXImbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly9wcmV0dGllci5pbykKWyFbT3BlbiBJc3N1ZXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy1yYXcvMHhUaGVQcm9EZXYvbmVzdGpzLWNsZWFuLWV4YW1wbGU/c3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9uZXN0anMtY2xlYW4tZXhhbXBsZS9pc3N1ZXMpClshW0Nsb3NlZCBJc3N1ZXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy1jbG9zZWQtcmF3LzB4VGhlUHJvRGV2L25lc3Rqcy1jbGVhbi1leGFtcGxlP3N0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL2dpdGh1Yi5jb20vMHhUaGVQcm9EZXYvbmVzdGpzLWNsZWFuLWV4YW1wbGUvaXNzdWVzP3E9aXMlM0Fpc3N1ZStpcyUzQWNsb3NlZCkKWyFbT3BlbiBQdWxsc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvaXNzdWVzLXByLXJhdy8weFRoZVByb0Rldi9uZXN0anMtY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L25lc3Rqcy1jbGVhbi1leGFtcGxlL3B1bGxzKQpbIVtDbG9zZWQgUHVsbHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy1wci1jbG9zZWQtcmF3LzB4VGhlUHJvRGV2L25lc3Rqcy1jbGVhbi1leGFtcGxlP3N0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL2dpdGh1Yi5jb20vMHhUaGVQcm9EZXYvbmVzdGpzLWNsZWFuLWV4YW1wbGUvcHVsbHM/cT1pcyUzQXByK2lzJTNBY2xvc2VkKQpbIVtDb250cmlidXRvcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2NvbnRyaWJ1dG9ycy8weFRoZVByb0Rldi9uZXN0anMtY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L25lc3Rqcy1jbGVhbi1leGFtcGxlL2dyYXBocy9jb250cmlidXRvcnMpClshW0FjdGl2aXR5XShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9sYXN0LWNvbW1pdC8weFRoZVByb0Rldi9uZXN0anMtY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxhYmVsPW1vc3QlMjByZWNlbnQlMjBhY3Rpdml0eSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9uZXN0anMtY2xlYW4tZXhhbXBsZS9wdWxzZSkKCiMjIERlc2NyaXB0aW9uCgpfRXhhbXBsZSBBcHBsaWNhdGlvbiBJbnRlcmZhY2UgdXNpbmcgTmVzdEpTIGZyYW1ld29yayBpbiBUeXBlU2NyaXB0XwoKVGhpcyBleGFtcGxlIHNob3djYXNlcyBSZXBvc2l0b3J5IFBhdHRlcm4gaW4gSGV4YWdvbmFsIEFyY2hpdGVjdHVyZSBfKGFsc28ga25vd24gYXMgQ2xlYW4gQXJjaGl0ZWN0dXJlKV8uIEhlcmUgd2UgaGF2ZSB0d28gRW50aXRpZXMgLSBCb29rcyBhbmQgQXV0aG9ycywgd2hvc2UgcmVsYXRpb25zaGlwcyBoYXZlIGJlZW4gZXhwbG9pdGVkIHRvIGNyZWF0ZSBDUlVEIGVuZHBvaW50IGluIFJFU1QgdW5kZXIgT3BlbkFQSSBzdGFuZGFyZC4KCiMjIEluc3RhbGxhdGlvbgoKLSBJbnN0YWxsIGFsbCB0aGUgcHJvamVjdCBkZXBlbmRlbmN5IHVzaW5nIFtZYXJuXShodHRwczovL3lhcm5wa2cuY29tKToKCiAgYGBgc2gKICAkIHlhcm4KICBgYGAKCi0gUnVuIHRoZSBhcHBsaWNhdGlvbiBmcm9tIGNvbW1hbmQgcHJvbXB0OgoKICBgYGBzaAogICQgeWFybiBzdGFydAogIGBgYAoKIyMgVGVzdGluZwoKLSBSdW4gdGhlIGZvbGxvd2luZyBjb21tYW5kIHRvIGluaXRpYXRlIFVuaXQgdGVzdDoKICBgYGBzaAogICQgeWFybiB0ZXN0CiAgYGBgCi0gUnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB0byBpbml0aWF0ZSBFMkUgdGVzdDoKICBgYGBzaAogICQgeWFybiB0ZXN0OmUyZQogIGBgYAoKIyMgTGljZW5zZQoKJmNvcHk7IE1JVCBMaWNlbnNlCg== readmeEtag: '"d0240b56c75e047dee78bd5095a42dacf38f37ff"' readmeLastModified: Fri, 12 May 2023 09:03:17 GMT repositoryId: 507814008 description: Clean Architecture Example using NestJS on Fastify created: '2022-06-27T07:53:29Z' updated: '2025-10-09T13:15:39Z' language: TypeScript archived: false stars: 7 watchers: 0 forks: 1 owner: 0xTheProDev logo: https://avatars.githubusercontent.com/u/14367736?v=4 license: MIT repoEtag: '"04db310565f7f928d8d1e4c6f3daafd7a25440fce4c5b949f1d940b74e4084b6"' repoLastModified: Thu, 09 Oct 2025 13:15:39 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/progyan1997/nestjs-clean-example - source: openapi3 tags repository: https://github.com/stackql/stackql-provider-azure v3: true id: 612cb89afa40fd2312a6a1ba76b8d700 repositoryMetadata: base64Readme: >- # Azure OpenAPI Doc Generator

Generates a single OpenAPI document for each Azure Resource Manager service from a set of OpenAPI specifications sourced from [__azure-rest-api-specs__](https://github.com/Azure/azure-rest-api-specs).  

This project uses [__autorest__](https://github.com/Azure/autorest) via JavaScript to emit the latest set of OpenAPI specifications for each service,  then dereferences and combines the documents into a single YAML OpenAPI specification using [__APIDevTools/json-schema-ref-parser__](https://github.com/APIDevTools/json-schema-ref-parser). 

> a complete set of specs for each service is available [here](https://github.com/stackql/stackql-azure-openapi/tree/main/openapi/3-combined)

> a set of specs for each service used to generate the `azure` and `azure_extras` `stackql` provider is available [here](https://github.com/stackql/stackql-azure-openapi/tree/main/openapi/4-tagged)  

## Usage

> use the `prereq.sh` script or the equivalent to download the latest api documentation from [__azure-rest-api-specs__](https://github.com/Azure/azure-rest-api-specs)

> use `run.sh` script or the equivalent to run all the `stackql-azure-openapi` commands in a batch

The main entrypoint to the program is:  

```bash
bin/stackql-azure-openapi
```

This can be run without any arguments to get command line usage and help.  The `stackql-azure-openapi` commands include:

- `generate` : uses `autorest` to generate initial OpenAPI3 specs for one or many services
- `dereference` : dereferences all external JSON pointers in the generated OpenAPI specs
- `combine` : combines all the generated OpenAPI specs into a single OpenAPI spec
- `tag` : replaces existing tags for operations with the stackql resource name (optional)

These commands are intended to be run in sequence with the output of each command being the input to the next.  The flow is summarized below:  

### IMAGE

[![stackql_azure_openapi](images/stackql_azure_openapi.png)](images/stackql_azure_openapi.png)

Specific details about each command are provided in the sections below.  

## `generate` command

Uses `autorest` to generate initial OpenAPI3 specs for one or many services using the `autorest` configuration data in the `readme.md` in each service specification directory from the `main` branch of the [azure-rest-api-specs](https://github.com/Azure/azure-rest-api-specs) repo.  Conceptually this would be similar to running the following command in each subdirectory of the `specification` folder:      

```bash
autorest ./resource-manager/readme.md \
--openapi-type=arm \
--output-converted-oai3
```

The output of this command is a directory structure (n-levels deep) containing OpenAPI JSON documents with references to each other (using JSON `$refs`), the union of which forms a complete specification for the service.  

To generate specs for a single service (the `compute` service in this example), run the following command:

```bash
bin/stackql-azure-openapi generate compute
```

To generate specs for all services, run the following command:

```bash
bin/stackql-azure-openapi generate
```

## `dereference` command

Takes the output from the `generated` command, enumerates all JSON documents nested at any level in the directory structure, and dereferences all JSON pointers in each document, outputting a flat structure containing a document for each generated document found for each service with no external references.  

To dereference a set of `autorest` generated specs for a single service (the `compute` service in this example), run the following command:

```bash
bin/stackql-azure-openapi dereference compute
```

To dereference specs for all services, run the following command:

```bash
bin/stackql-azure-openapi dereference
```

## `combine` command

Takes the output from the `dereference` command, and combines all the generated documents into a single OpenAPI spec, outputting a single, self contained YAML document (removing the `examples` for brevity).  

To create a single OpenAPI document for a single service (the `compute` service in this example), run the following command:

```bash
bin/stackql-azure-openapi combine compute
```

To create a single OpenAPI document for all services, run the following command:

```bash
bin/stackql-azure-openapi combine
```

## `tag` command (Optional)

Takes the output from the `combine` command, generates the stackql resource name from stemming the `operationId` for each operation, pushes this value to the beginning of the `tags` array for each operation, and outputs the resulting document.    

To update tags for a single OpenAPI document for a single service (the `compute` service in this example), run the following command:  

```bash
bin/stackql-azure-openapi tag compute
```

To tags fo all OpenAPI documents for all services, run the following command:

```bash
bin/stackql-azure-openapi tag
```

## Testing locally with `stackql`
1. download the latest `stackql` binary, for example `curl -L https://bit.ly/stackql-zip -O && unzip stackql-zip` for Linux systems
2. run the following:
```
PROVIDER_REGISTRY_ROOT_DIR="$(pwd)/openapi"
REG_STR='{"url": "file://'${PROVIDER_REGISTRY_ROOT_DIR}'", "localDocRoot": "'${PROVIDER_REGISTRY_ROOT_DIR}'", "verifyConfig": {"nopVerify": true}}'
./stackql shell --registry="${REG_STR}"
```

### Run Test Suite

from the `stackql-provider-tests` directory:

```bash
cd ../../stackql-provider-tests

# azure
sh test-provider.sh \
azure \
false \
/mnt/c/LocalGitRepos/stackql/openapi-conversion/stackql-azure-openapi/openapi \
true

# azure_extras
sh test-provider.sh \
azure_extras \
false \
/mnt/c/LocalGitRepos/stackql/openapi-conversion/stackql-azure-openapi/openapi \
true

# azure_isv
sh test-provider.sh \
azure_isv \
false \
/mnt/c/LocalGitRepos/stackql/openapi-conversion/stackql-azure-openapi/openapi \
true

# azure_stack
sh test-provider.sh \
azure_stack \
false \
/mnt/c/LocalGitRepos/stackql/openapi-conversion/stackql-azure-openapi/openapi \
true
```

# Document

```bash
sh provider-dev/scripts/fix-broken-links.sh

# azure_stack
npm run generate-docs -- \
  --provider-name azure_stack \
  --provider-dir ./openapi/src/azure_stack/v00.00.00000 \
  --output-dir ./website/azure_stack \
  --provider-data-dir ./provider-dev/docgen/azure_stack

cd website/azure_stack
yarn build
yarn start
cd ../..

# azure_isv
npm run generate-docs -- \
  --provider-name azure_isv \
  --provider-dir ./openapi/src/azure_isv/v00.00.00000 \
  --output-dir ./website/azure_isv \
  --provider-data-dir ./provider-dev/docgen/azure_isv

cd website/azure_isv
yarn build
yarn start
cd ../..

# azure_extras
npm run generate-docs -- \
  --provider-name azure_extras \
  --provider-dir ./openapi/src/azure_extras/v00.00.00000 \
  --output-dir ./website/azure_extras \
  --provider-data-dir ./provider-dev/docgen/azure_extras

cd website/azure_extras
yarn build
yarn start
cd ../..

# azure
npm run generate-docs -- \
  --provider-name azure \
  --provider-dir ./openapi/src/azure/v00.00.00000 \
  --output-dir ./website/azure \
  --provider-data-dir ./provider-dev/docgen/azure

# note the azure site has to be built locally
cd website/azure
./local_build.sh
./deploy_to_netlify.sh
cd ../..
``` readmeEtag: '"0ca685165f84ee7ed53bf517cca1adf1584da864"' readmeLastModified: Wed, 17 Sep 2025 01:48:09 GMT repositoryId: 522469749 description: >- Generates a single OpenAPI document for each Azure Resource Manager service from a set of OpenAPI specifications sourced from Azure REST API specs created: '2022-08-08T08:41:58Z' updated: '2025-10-04T07:53:46Z' language: JavaScript archived: false stars: 7 watchers: 2 forks: 0 owner: stackql logo: https://avatars.githubusercontent.com/u/95105302?v=4 repoEtag: '"4f35624846ad7734c37507d399a0ae25fb3e5651919da47d732e0c64908eae80"' repoLastModified: Sat, 04 Oct 2025 07:53:46 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/stackql/stackql-azure-openapi - source: openapi3 tags repository: https://github.com/jordan-chalupka/ouraring v3: true repositoryMetadata: base64Readme: >- IyBPdXJhIFJpbmcgT3BlbkFQSXYzIFNjaGVtYSBEZWZpbml0aW9uICsgR28gQ2xpZW50ClRoaXMgcmVwbyBjb250YWlucyBhbiBPcGVuQVBJdjMgc2NoZW1hIGRlZmluaXRpb24gZm9yIFtPdXJhUmluZ10oaHR0cHM6Ly9vdXJhcmluZy5jb20vKQoKT3VyYVJpbmcgQVBJIGRvY3VtZW50YXRpb24gaHR0cHM6Ly9jbG91ZC5vdXJhcmluZy5jb20vZG9jcy8KCiMjIEdlbmVyYXRpbmcgdGhlIGNsaWVudApDbGllbnQgZ2VuZXJhdGlvbiBpcyBkb25lIHVzaW5nIGh0dHBzOi8vZ2l0aHViLmNvbS9kZWVwbWFwL29hcGktY29kZWdlbgoKVG8gcmVnZW5lcmF0ZSB0aGUgY2xpZW50LCBydW4gdGhlIGNvbW1hbmQKYGBgCm9hcGktY29kZWdlbiAtLXBhY2thZ2Ugb3VyYXJpbmcgb3VyYXJpbmdfb3BlbmFwaS55YW1sID4gb3VyYXJpbmcuZ2VuLmdvCmBgYAoKIyMgRXh0ZW5kaW5nIHRoZSBzY2hlbWEgZGVmaW5pdGlvbgpBdCB0aGlzIHRpbWUsIHRoZSBzY2hlbWEgZGVmaW5pdGlvbiBpcyBub3QgY29tcGxldGUuIAoKSWYgdGhlcmUgaXMgZGF0YSB3aGljaCB5b3Ugd291bGQgbGlrZSB0byBmZXRjaCBidXQgaXQgaXMgbm90IGF2YWlsYWJsZSwgY29uc2lkZXIgYWRkaW5nIGl0IHRvIHRoZSBzY2hlbWEgZGVmaW5pdGlvbgphbmQgcmVnZW5lcmF0aW5nIHRoZSBjbGllbnQuCgojIyBFeGFtcGxlIHVzYWdlCkdldCB5b3VyIHBlcnNvbmFsIGFjY2VzcyB0b2tlbiBmcm9tIEF1cmFSaW5nCmh0dHBzOi8vY2xvdWQub3VyYXJpbmcuY29tL3BlcnNvbmFsLWFjY2Vzcy10b2tlbnMKCmBgYGJhc2gKZXhwb3J0IE9VUkFfQUNDRVNTX1RPS0VOPSJZT1VSX1BFUlNPTkFMX0FDQ0VTU19UT0tFTiIKZ28gcnVuIGV4YW1wbGUvbWFpbi5nbwpgYGAKClNob3VsZCBmZXRjaCB5b3VyIGFjdGl2aXR5IHNjb3JlIGZvciB0b2RheToKYGBgCkFjdGl2aXR5IFNjb3JlOiA4MQpgYGAK readmeEtag: '"d336657fbb93d93dc9140b4b8628bd639aabb926"' readmeLastModified: Sat, 18 Dec 2021 22:55:24 GMT repositoryId: 439717045 description: Go client + OpenAPI v3 spec for Oura-Ring created: '2021-12-18T21:20:08Z' updated: '2024-05-23T16:59:30Z' language: Go archived: false stars: 6 watchers: 1 forks: 0 owner: jordan-chalupka logo: https://avatars.githubusercontent.com/u/9794216?v=4 repoEtag: '"2884c1da90cc2a6e7f114d0da7956c37afbe18303bd313d76c9714bdbc969b31"' repoLastModified: Thu, 23 May 2024 16:59:30 GMT foundInMaster: true category: - Converters - Server Implementations id: c0a9a71da09e3f7ae91e9f6c7ee05c41 - source: openapi3 tags repository: https://github.com/arno-di-loreto/api-specifications-documentation-as-data v3: true id: 944858e36a88377fd62fd15e2e48e808 repositoryMetadata: base64Readme: >- IyBBUEkgU3BlY2lmaWNhdGlvbiBEb2N1bWVudGF0aW9uIGFzIERhdGEgUHJvamVjdAoKUmVzZWFyY2gsIHRlc3RzIGFuZCBQT0NzIHRvIFtleHRyYWN0XShodHRwczovL2dpdGh1Yi5jb20vYXJuby1kaS1sb3JldG8vb3BlbmFwaS1zcGVjaWZpY2F0aW9uLWRvY3VtZW50YXRpb24tYXMtZGF0YS90cmVlL21haW4vb2FzLW1kLXBhcnNlci1weXRob24pIHNvbWUgW2RhdGFdKHNwZWNpZmljYXRpb25zLWRhdGEpIGZyb20gdGhlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBhbmQgQXN5bmNBUEkgU3BlY2lmaWNhdGlvbiBbZG9jdW1lbnRhdGlvbl0oc3BlY2lmaWNhdGlvbnMpIGFuZCBbdmlzdWFsaXplXShvYXMtZGF0YS12aWV3ZXItd2ViLWNvbXBvbmVudC8pIGl0LgpfV2FybmluZzogQWxsIHRoaXMgaXMgZXhwZXJpbWVudGFsLl8KCkNoZWNrIHRoZSBbc2NyaXB0c10oc2NyaXB0cykgZm9sZGVyIHRvIGluc3RhbGwgYW5kIHJ1biBlbGVtZW50cy4= readmeEtag: '"84bcee5a8d76a2e524f8985fd4d92b51ccba82a6"' readmeLastModified: Thu, 20 Apr 2023 15:44:52 GMT repositoryId: 544376926 description: >- Extracting and visualizing data from OpenAPI and AsyncAPI Specifications documentation (warning: experimental) created: '2022-10-02T10:21:52Z' updated: '2025-08-27T14:26:34Z' language: Mermaid archived: false stars: 6 watchers: 1 forks: 2 owner: arno-di-loreto logo: https://avatars.githubusercontent.com/u/10104551?v=4 repoEtag: '"be1e48aa22f85222bd86f798b51bdd9a353f6109ed04b883e4c7311192e36582"' repoLastModified: Wed, 27 Aug 2025 14:26:34 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/tomboyo/lily v3: true repositoryMetadata: base64Readme: >- = Lily OAS-To-Java Compiler

image:https://img.shields.io/maven-metadata/v/https/repo1.maven.org/maven2/io/github/tomboyo/lily/lily/maven-metadata.xml.svg[title="latest release version badge" link="https://mvnrepository.com/artifact/io.github.tomboyo.lily"]

Lily is a compiler that consumes OpenAPI (Swagger) specifications and produces Java source code.
It is intended to be an alternative to https://github.com/OpenAPITools/openapi-generator[OpenAPI Generator], which at time of writing is the only option, and substantially more complete than Lily.

== Usage

Please be aware that Lily is in the early stages of development and may not be suitable for production use-cases. Use Lily at your own risk.

That said, beta testing is the best way to link:CONTRIBUTING.adoc[contribute] to this project.

=== Java API

Generated source code looks like this:

[source,java]
----
List<Pet> exampleHappyPath() {
  var api = Api.newBuilder()
      .uri("https://example.com/")
      .build();

  try {
    var response = api.petsOperations()
        .listPets()
        .query(query -> query.limit(50))
        .sendSync();

    return switch (response) {
      case ListPets200 ok -> ok.body().value();
      case ListPetsDefault other -> throw new RuntimeException(other.body().message());
    };
  } catch (IOException | InterruptedException e) {
    // The java.net.http layer encountered an exception.
    throw new RuntimeException("Unable to complete Pets API request", e);
  }
}
----

Here are some of the Lily features we just saw:

* If an operation has the 'pets' tag, then we can access it via the `petsOperations()` _operation group_. Every operation is also part of the `everyOperation()` group, and operations without tags are also members of the `everyUntaggedOperation()` group. These groups are intended to help us explore the API using IDE type-ahead/auto-complete hints.

* Responses form a sealed interface. If we have the pattern-matching for switch expressions feature enabled, we can create an exhaustive switch expression to handle all possible responses, including undocumented and unexpected ones. Otherwise, we can use pattern-matching in an if-else ladder, or even access the status code via `response.httpResponse().statusCode()` (the native java.net.http API).

In the real world, OpenAPI specifications have errors in them that could prevent a generated API from successfully making requests. Rather than wait for service owners to update their specifications or try to fix them in a local copy ourselves, we can use Lily's API to do as much as possible, then dip down into the underlying java.net.http API for full customization and control:

[source,java]
----
var operation = api.petsOperations()
        .listPets()
        .query(query -> query.limit(50));
var request = HttpRequest.newBuilder(operation.httpRequest(), (k, v) -> true)
        .header("x-some-undocumented-header", "foo;bar;baz")
        .build();

// If the API has correctly documented responses, lily will help us deserialize
// the response and we can handle it like before.
var response = operation.sendSync(request);

// Otherwise, we can use the httpClient to get an HttpRequest of an InputStream
// and deserialize it however we see fit, including not at all.
var response = api.httpClient().send(request, BodyHandlers.ofInputStream());
----

Here's what we just saw:

* We can use the operation to customize an HttpRequest, then use the java.net.http API to copy-and-modify that request. We can use Lily for everything that is documented by the OpenAPI specification correctly, but then arbitrarily modify the request with the native API. This lets us accommodate nearly any specification error, and even flaws in Lily.

* We can then ask the operation to send the customized request, which will return a response that lazily deserializes the response body to the documented type. If we know the documented type is wrong, we can instead send the request with the native API and deserialize the InputStream however necessary, or not at all.

In other words, Lily is designed to facilitate HTTP interactions whenever possible, but fall back gracefully to the native java.net.http API in the presence of specification errors. Notably, all of these workarounds are _forwards-compatible_: Once the service owners update their OpenAPI specification to correct whatever errors were present, all of our code continues working. We can go back and update the code to use the generated API at our own pace.

=== Maven Dependency

image:https://img.shields.io/maven-metadata/v/https/repo1.maven.org/maven2/io/github/tomboyo/lily/lily/maven-metadata.xml.svg[title="latest release version badge" link="https://mvnrepository.com/artifact/io.github.tomboyo.lily"]

To generate sources from an OAS document in your maven project, and the following maven build plugin and dependencies:

```xml
<build>
    <plugins>
        <plugin>
            <groupId>io.github.tomboyo.lily</groupId>
            <artifactId>lily-compiler-maven-plugin</artifactId>
            <version>${lilyVersion}</version>
            <configuration>
                <!-- Any URI to an OAS document, be it https:// or file://. -->
                <uri>https://raw.githubusercontent.com/OAI/OpenAPI-Specification/main/examples/v3.0/petstore.yaml</uri>

                <!-- Uncomment to customize the default generated sources directory. -->
                <!-- <outputDir>target/generated-sources</outputDir> -->

                <basePackage>com.exmaple.my.api</basePackage>
            </configuration>
            <executions>
                <execution>
                    <goals>
                        <goal>compile-client</goal>
                    </goals>
                    <phase>generate-sources</phase>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>

 <dependencyManagement>
    <dependencies>
        <!-- BEGIN generated code dependency management -->
        <dependency>
            <groupId>com.fasterxml.jackson</groupId>
            <artifactId>jackson-bom</artifactId>
            <version>2.13.0</version>
            <scope>import</scope>
            <type>pom</type>
        </dependency>
        <!-- END generated code dependency management -->
    </dependencies>
</dependencyManagement>

<dependencies>
     <!-- BEGIN Generated code dependencies -->
    <dependency>
        <groupId>io.github.tomboyo.lily</groupId>
        <artifactId>lily-http</artifactId>
        <version>${lilyVersion}</version>
    </dependency>
    <dependency>
        <groupId>com.fasterxml.jackson.core</groupId>
        <artifactId>jackson-databind</artifactId>
    </dependency>
    <dependency>
        <!-- ZonedDatetime support -->
        <groupId>com.fasterxml.jackson.datatype</groupId>
        <artifactId>jackson-datatype-jsr310</artifactId>
    </dependency>
    <!-- END Generated code dependencies -->
</dependencies>
```

The generated source code relies on jackson and the lily-http library at runtime, which is why these dependencies are necessary.

These configurations can be stand-alone or embedded in a larger project.

== Goals

. Generate java source code directly from an OAS document within a java build pipeline (e.g. integrated with Maven or Gradle).
. Support OAS v3.
. Target the ~latest version of java, with special attention paid to upcoming language features.
. Help end-users work around incorrect or incomplete schema specifications so that they can make progress while awaiting upstream fixes.
. Expose a high-level API to guide the user through API interactions.
. Ensure that whenever possible, generated source code is compatible with user code between API specification revisions. In other words: "If I update to the latest API specification, and there are not breaking changes to the API, then Lily's generated source code doesn't break my application."
. Support all OpenAPI features, including unusual things like matrix-style requests.

== Non-Goals

. Do not (yet) support other languages than Java. It's not clear that a Java-oriented AST will cleanly translate to another language target.
. Do not support too many options. Options become confusing to maintain -- prefer opinionated code that works for most people who are doing sensible things.

== Design

Lily is a layered API with "high-level" layers that orchestrate full requests using generated code and "low-level" layers that help the developer implement requests from scratch if necessary.

High-level layers always allow the developer to move into lower levels. This allows the developer to use the convenient high-level API _as much as possible_, then resort to the lower-level API (which could be the java.net.http API itself) only as necessary to work around missing features or undocumented API parameters.

Lily should make simple things easy, and complex things possible.

== Quick Tour

Lily is composed of four modules in the `modules` directory:

- `example` compiles the v3.0 petstore YAML as an example. Check out the generated-sources directory after a build to see what Lily generates, and the test directory to see example usage of the generated code.

- `lily-compiler-maven-plugin` is a teensy-weensy Maven plugin that reads configuration from the pom and hands it off to the compiler project. This is what the user adds to their projects to compile code.

- `lily-compiler` is responsible for reading an OAS document, translating it to an intermediary AST (abstract syntax tree), rendering the AST as source code, and finally saving source code to disc.

- `lily-http` defines classes to help create and receive HTTP requests, including RFC6570 encoders, deser implementations, and the UriTemplate. This is a dependency of generated source code and may also be used directly by users to work around Lily or OAS shortcomings.
 readmeEtag: '"2df602ecf079064d3dd07fa6848724336fa31487"' readmeLastModified: Mon, 10 Mar 2025 00:36:53 GMT repositoryId: 423275796 description: Java 17 native HTTP client code generator for OpenAPI v3 created: '2021-10-31T22:37:39Z' updated: '2026-01-25T18:11:53Z' language: Java archived: false stars: 6 watchers: 3 forks: 1 owner: Tomboyo logo: https://avatars.githubusercontent.com/u/3467385?v=4 license: GPL-3.0 repoEtag: '"f1f1c0f105608bab9ad276a352699b258856e42aa4816646c8c5db3391b84a61"' repoLastModified: Sun, 25 Jan 2026 18:11:53 GMT foundInMaster: true category: Testing id: e8988f3eb5b572da48b6533f5ee9b3bf - source: openapi3 tags repository: https://github.com/huogerac/cookiecutter-flask-openapi v3: true repositoryMetadata: base64Readme: >- # Cookiecutter Flask OpenAPI boilerplate

[![GitHub issues](https://img.shields.io/github/issues/huogerac/cookiecutter-flask-openapi?style=for-the-badge)](https://github.com/huogerac/cookiecutter-flask-openapi/issues) [![GitHub stars](https://img.shields.io/github/stars/huogerac/cookiecutter-flask-openapi?style=for-the-badge)](https://github.com/huogerac/cookiecutter-flask-openapi/stargazers) [![GitHub license](https://img.shields.io/github/license/huogerac/cookiecutter-flask-openapi?style=for-the-badge)](https://github.com/huogerac/cookiecutter-flask-openapi/blob/master/LICENSE) [![code style: black](https://img.shields.io/badge/code%20style-black-000000.svg?style=for-the-badge)](https://github.com/psf/black)

> 💻 A full-featured Flask + API + OAS3 + JWT + SwaggerUI + ORM + Migrations + Great and Scalable structure project template

> 👉 It's using latest Flask 1.2

> Flask 2 is coming soon 😎

> It uses PostgreSQL everywhere, so Docker is necessary (SQLite with no Docker dependency is on the roadmap) 😎


## Why using this boilerplate ❓

- Focus on the business and creating value
- Faster project setup
- Standard project structure organization (easy to scale)
- Better QA


## What's Included (Features) 🎉


- API Design first using OpenAPI & Connexion
- API documentation using swagger UI
- Login using JWT
- Every layer is separated in context/domain
- Service layer for better tests and reuse
- Using Flask Factory to integrate with extensions
- Migrations using Alembic
- ORM using SQLAlchemy
- Optimized development and production settings
- Comes with user model ready to go, signup & signin
- Procfile for deploying to Heroku
- Customizable PostgreSQL version
- Tests using pytest
- Unit tests for the API layer
- Unit tests for the service layer

**Development**
- Code linter
- Code formatter (Black+iSort)
- Using .env file
- Docker support using docker-compose for development
- Docker using multistage (Production Ready Dockerfile)
- Postgres in development (using docker-compose)
- CI using Github Actions

## Structure

This project is organized in:
- Layers 🧅, which might not change in the project life cycle
- Modules 📦, for domain contexts, which might scale in terms of new features
- Configuration ⚙️ separated based on the extensions
- The 🎂 Business modules

```
Hackernews-Clone
.
├── hackernews
│   ├── app.py                    👉 Entrypoint (create_app)
│   ├── exceptions.py
│   ├── 🧅 ext                    👉 Settings
│   │   ├── ⚙️ configuration.py
│   │   ├── ⚙️ api.py
│   │   └── ⚙️ database.py
│   │   ...
│   ├── 🧅 api                    👉 API Routes
│   │   ├── 📦 auth.py
│   │   ├── 📦 news.py
│   │   └── 📦 openapi.yaml       👉 API Contract
│   │   ...
│   ├── 🧅 services               👉 Business rules
│   │   ├── 📦 auth.py 🎂
│   │   ├── 📦 news.py 🎂
│   │   └── 📦 token.py 🎂
│   │   ...
│   └── 🧅 models                 👉 ORM
│       ├── 📦 news.py
│       └── 📦 users.py
│       ...
├── ⚙️ migrations                 👉 Database versions
│   ├── alembic.ini
│   ├── env.py
│   ├── script.py.mako
│   └── versions
├── tests
│   ├── conftest.py
│   ├── api                      👉 Endpoint tests, input, output and validation 
│   ├── database                 👉 Database connection tests
│   └── services                 👉 Business rules tests
├── requirements.txt
├── pytest.ini
├── uwsgi.ini                    👉 Application server settings  
└── wsgi.py                      👉 WSGI Deploy file (Gunicorn/uWSGI)
```

## Requirements

- Python 3.8 or 3.9 (Help us test in other versions)

- Docker to run Postgres locally


## Contribute 🚀

Any help is more than welcome...

- 👉 It could be an [Issue](https://github.com/huogerac/cookiecutter-flask-openapi/issues)
- 💻 It could be using it and give a feedback
- 🌟 It could be a github star
- 🤔 It could be a [Question](https://github.com/huogerac/cookiecutter-flask-openapi/discussions/)
- 🤔 If you dislike this project, feel free to tell us what is wrong with it


## Get Started (Usage) 

Let's pretend you want to create a Flask project called "hackernewsclone". Rather than start from scratch by a ``app.py`` and add each library, Flask extesion and various other configurations that always get forgotten until the worst possible moment, get this cookiecutter template do all the work.

First, get Cookiecutter. Trust me, it's awesome::

```
    $ pip install "cookiecutter>=1.7.0"
```

Now run it against this repo::

```
    $ cookiecutter https://github.com/huogerac/cookiecutter-flask-openapi/
```

You'll be prompted for some values. Provide them...

```
project_name [Hackernews Clone]: 
project_slug [hackernews_clone]: hackernews
description [The Ultimate Flask Template]: 
main_model [News]: News
main_model_lower [news]: 
Select python_version:
1 - 3.8.10
2 - 3.9.5
Choose from 1, 2 [1]: 2
Select package_manager:
1 - requirements.txt
2 - Pipenv
Choose from 1, 2 [1]:  
Select postgresql_version:
1 - 13.3-alpine
2 - 13.5
3 - 14.1
Choose from 1, 2, 3 [1]: 
use_dockerfile [yes]: 
use_github_actions_CI [yes]: 
deploy_to_heroku [yes]: 
keep_vscode_settings [yes]: 
author_name [Roger Camargo]: 
email [roger-camargo@example.com]: huogerac@gmail.com
version [0.1.0]: 
 [INFO]:   - Using requirements.txt and virtualenv
 [SUCCESS]: 🐍 Your project is created! ✨ 🍰 ✨


What's next?
     cd hackernews
     Check the README_DOCKER 🐳
     Check the README_VIRTUALENV 🐍
 [INFO]: ⚠️ For more details, check the Makefile or run: make help

```

Then access 🚀 http://localhost:5000/api


## Articles

- 🇧🇷 [Estrutura e organização de pastas em projetos Flask](https://huogerac.hashnode.dev/estrutura-e-organizacao-de-pastas-em-projetos-flask)

- 🇧🇷 [API Design First](https://speakerdeck.com/huogerac/api-design-first-pt-br)


## API Design First Approach

The big difference in this template is the way APIs are created, it uses the [connexion](https://github.com/zalando/connexion) to build up the API Contract (YAML) first. So the API documentation is created from the contract and not from 
code.

### Understanding the structure from tests

We can start new implementation from building tests (TDD)

```python

# API tests

def test_should_return_title_is_a_required_field(token_valid_mock, client):
    # Given a request with a missing required field
    response = client.post(
        "/api/news",
        headers={"authorization": f"Bearer {token_valid_mock}"},
        json={},
    )

    # Then
    assert response.status_code == 400
    assert response.json["detail"] == "'title' is a required property"


def test_should_reject_title_less_than_min_title_length(token_valid_mock, client):
    # Given a request with an invalid title
    response = client.post(
        "/api/news",
        headers={"authorization": f"Bearer {token_valid_mock}"},
        json={"title": "tiny-title"},
    )

    # Then
    assert response.status_code == 400
    assert response.json["detail"] == "'tiny-title' is too short - 'title'"


@patch("hackernews.services.news.create_news")
def test_should_accept_null_description(news_mock, token_valid_mock, client):

    news_mock.return_value = {}

    # Given a request with an empty description (non string)
    response = client.post(
        "/api/news",
        headers={"authorization": f"Bearer {token_valid_mock}"},
        json={
            "title": "A valid and simple title",
            "description": None,
        },
    )

    # Then
    assert response.status_code == 201
    news_mock.assert_called_once_with("A valid and simple title", None)

```

Then, we jump up directly to the API Contract. Note that the contract has information
to do input validations

```YAML

  # hackernews/api/openapi.yaml
  /api/news:
    post:
      operationId: hackernews.api.news.create_news
      summary: Creates a news
      tags:
        - News
      security:
        - jwtAuth: [news:create]
      requestBody:
        required: true
        content:
          application/json:
            schema:
              type: object
              additionalProperties: false
              required:
                - title
              properties:
                title:
                  type: string
                  example: This is an awesome news
                  minLength: 12
                description:
                  type: string
                  example: Some extra information about the news
                  nullable: true

      responses:
        201:
          description: News created succesfully
          content:
            application/json:
              schema:
                $ref: "#/components/schemas/News"
        400:
          description: Invalid data
        401:
          description: No Authorization

components:

  schemas:
    News:
      type: object
      properties:
        id:
          type: integer
          example: 42
        title:
          type: string
          example: Python is a great option for backend and APIs
        description:
          type: string
          example: This is along text within more detailed information about the news
          nullable: true
        created_at:
          type: string
          format: date-time
          example: 2021-01-11T11:32:28Z
```




## Thanks and Inspirations

- This template is based on [cookiecutter-django](https://github.com/cookiecutter/cookiecutter-django). Thanks [PyDanny](https://github.com/pydanny) and the Cookiecutter maintainers

- This Flask structure is based on ["Arquitetura Definitiva para o Projeto Web Com Python e Flask"](https://www.youtube.com/watch?v=-qWySnuoaTM). Thanks [Bruno Rocha](https://github.com/rochacbruno) and all yours awesome Flask contents
 readmeEtag: '"69745b628de1c3737d6543a0abdb666da516811e"' readmeLastModified: Sun, 26 Dec 2021 15:36:50 GMT repositoryId: 439192571 description: >- Cookiecutter Flask OpenAPI is a template for jumpstarting production-ready Flask projects quickly. It has a well organized and scalable structure. It uses API design first created: '2021-12-17T02:56:07Z' updated: '2025-10-08T00:55:41Z' language: Python archived: false stars: 5 watchers: 1 forks: 1 owner: huogerac logo: https://avatars.githubusercontent.com/u/962233?v=4 license: BSD-3-Clause repoEtag: '"715dbfc11f7d94c4de523815c6fc90035d8924b06b020ce2440f82d44164847e"' repoLastModified: Wed, 08 Oct 2025 00:55:41 GMT foundInMaster: true category: Server Implementations id: 611541e75d358f861d14f5bb4f06070e - source: openapi3 tags repository: https://github.com/storm-platform/storm-ws-spec v3: true id: 9210ed330a23ca0f14c79b9ac96af28b repositoryMetadata: base64Readme: >- Li4KICAgIFRoaXMgZmlsZSBpcyBwYXJ0IG9mIFNwYXRpb1RlbXBvcmFsIE9wZW4gUmVzZWFyY2ggTWFuYWdlciBXZWIgU2VydmljZSBTcGVjaWZpY2F0aW9uLgogICAgQ29weXJpZ2h0IChDKSAyMDIxIElOUEUuCgogICAgU3BhdGlvVGVtcG9yYWwgT3BlbiBSZXNlYXJjaCBNYW5hZ2VyIFdlYiBTZXJ2aWNlIFNwZWNpZmljYXRpb24gaXMgZnJlZSBzb2Z0d2FyZTsgeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yIG1vZGlmeSBpdAogICAgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBNSVQgTGljZW5zZTsgc2VlIExJQ0VOU0UgZmlsZSBmb3IgbW9yZSBkZXRhaWxzLgoKCj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KU3BhdGlvVGVtcG9yYWwgT3BlbiBSZXNlYXJjaCBNYW5hZ2VyIFdlYiBTZXJ2aWNlIC0gU3BlY2lmaWNhdGlvbgo9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcnVuLUluc29tbmlhLWJsdWV2aW9sZXQKICAgICAgICA6dGFyZ2V0OiBodHRwczovL2luc29tbmlhLnJlc3QvcnVuLz9sYWJlbD1TdG9ybSUyMFdTJTIwaW1wbGVtZW50YXRpb24lMjBleGFtcGxlJnVyaT1odHRwcyUzQSUyRiUyRnJhdy5naXRodWJ1c2VyY29udGVudC5jb20lMkZzdG9ybS1wbGF0Zm9ybSUyRnN0b3JtLXdzLXNwZWMlMkZtYXN0ZXIlMkZleGFtcGxlJTJGc3Rvcm0td3MtaW5zb21uaWEuanNvbgogICAgICAgIDphbHQ6IFJ1biBpbiBJbnNvbW5pYQoKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2xpY2Vuc2UtTUlULWdyZWVuCiAgICAgICAgOnRhcmdldDogaHR0cHM6Ly9naXRodWIuY29tL3N0b3JtLXBsYXRmb3JtL3N0b3JtLXdzLXNwZWMvYmxvYi9tYXN0ZXIvTElDRU5TRQogICAgICAgIDphbHQ6IFNvZnR3YXJlIExpY2Vuc2UKCi4uIGltYWdlOjogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWZlY3ljbGUtbWF0dXJpbmctYmx1ZS5zdmcKICAgICAgICA6dGFyZ2V0OiBodHRwczovL3d3dy50aWR5dmVyc2Uub3JnL2xpZmVjeWNsZS8jbWF0dXJpbmcKICAgICAgICA6YWx0OiBTb2Z0d2FyZSBMaWZlIEN5Y2xlCgouLiBpbWFnZTo6IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3RhZy9zdG9ybS1wbGF0Zm9ybS9zdG9ybS13cy1zcGVjLnN2ZwogICAgICAgIDp0YXJnZXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9ybS1wbGF0Zm9ybS9zdG9ybS13cy1zcGVjL3JlbGVhc2VzCiAgICAgICAgOmFsdDogUmVsZWFzZQoKLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2Rpc2NvcmQvNjg5NTQxOTA3NjIxMDg1MTk4P2xvZ289ZGlzY29yZCZsb2dvQ29sb3I9ZmZmZmZmJmNvbG9yPTczODlEOAogICAgICAgIDp0YXJnZXQ6IGh0dHBzOi8vZGlzY29yZC5jb20vY2hhbm5lbHMvNjg5NTQxOTA3NjIxMDg1MTk4IwogICAgICAgIDphbHQ6IEpvaW4gdXMgYXQgRGlzY29yZAoKLi4gaW1hZ2U6OiBodHRwczovL2Nkbi5yYXdnaXQuY29tL3N5bDIwYm5yL3NwYWNlbWFjcy80NDJkMDI1Nzc5ZGEyZjYyZmM4NmMyMDgyNzAzNjk3NzE0ZGI2NTE0L2Fzc2V0cy9zcGFjZW1hY3MtYmFkZ2Uuc3ZnCiAgICAgICAgOnRhcmdldDogaHR0cHM6Ly9naXRodWIuY29tL3N5bDIwYm5yL3NwYWNlbWFjcwogICAgICAgIDphbHQ6IE1hZGUgd2l0aCBTcGFjZW1hY3MKCkFib3V0Cj09PT09CgpTcGF0aW9UZW1wb3JhbCBPcGVuIFJlc2VhcmNoIE1hbmFnZXIgV2ViIFNlcnZpY2UgYE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gPGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uPmBfLgoKUmVwb3NpdG9yeSBPcmdhbml6YXRpb24KPT09PT09PT09PT09PT09PT09PT09PT0KCi0gYGFwaSA8Li9hcGk+YF86IFN0b3JtIFdTIFNwZWNpZmljYXRpb24gdXNpbmcgYE9wZW5BUEkgMy4wIDxodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbj5gXzsKCi0gYGV4YW1wbGUgPC4vZXhhbXBsZT5gXzogYEluc29tbmlhIDxodHRwczovL2luc29tbmlhLnJlc3QvPmBfLWJhc2VkIGV4YW1wbGUgb2YgdXNpbmcgdGhlIGltcGxlbWVudGF0aW9uIG9mIHRoZSBzcGVjaWZpY2F0aW9uIHByb3ZpZGVkIGJ5IHRoZSBgU3Rvcm0gV1MgPGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9ybS1wbGF0Zm9ybS9zdG9ybS13cz5gXyBwYWNrYWdlLgoKQnVpbGRpbmcgdGhlIERvY3VtZW50YXRpb24KPT09PT09PT09PT09PT09PT09PT09PT09PT0KClJlcXVpcmVtZW50cwotLS0tLS0tLS0tLS0KClRoZSBidWlsZCBzeXN0ZW0gZm9yIHRoZSBSRVNUIEFQSSBkb2N1bWVudGF0aW9uIHJlbGllcyBvbiB0aGUgTm9kZS5qcyBydW4tdGltZSBlbnZpcm9ubWVudDoKCiAgLSBgTm9kZS5qcyA8aHR0cHM6Ly9ub2RlanMub3JnL2VuLz5gXyAoVmVyc2lvbiA4KykuCgogIC0gYFJlRG9jIDxodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9yZWRvYz5gXzogZ2VuZXJhdGVzIEhUTUwgcmVmZXJlbmNlIGRvY3VtZW50YXRpb24gZnJvbSBhbiBPcGVuQVBJIHNwZWNpZmljYXRpb24gZmlsZS4KCgpCdWlsZAotLS0tLQoKSWYgeW91IGhhdmUgTm9kZS5qcyBpbnN0YWxsZWQsIHBsZWFzZSwgZXhlY3V0ZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQgdG8gaW5zdGFsbCB0aGUgUmVEb2MgZGVwZW5kZW5jeToKCi4uIGNvZGUtYmxvY2s6OiBzaGVsbAoKICAgICQgbnBtIGluc3RhbGwKCgpBZnRlciB0aGF0LCBnZW5lcmF0ZSB0aGUgZG9jdW1lbnRhdGlvbjoKCi4uIGNvZGUtYmxvY2s6OiBzaGVsbAoKICAgICQgbnBtIHJ1biBidWlsZAoKClRoZSBhYm92ZSBjb21tYW5kIHdpbGwgY3JlYXRlIGEgZm9sZGVyIG5hbWVkIGBgZGlzdGBgIHdpdGggdGhlIGJ1bmRsZWQgZmlsZSBpbmRleC5odG1sLiBZb3UgbWF5IG9wZW4gaXQgaW4geW91ciB3ZWIgYnJvd3NlciBvciBtYXkgc2VydmUgaXQgd2l0aCBhbiBIVFRQIFNlcnZlci4KCkZvciBQeXRob24gZGV2ZWxvcGVycywgeW91IGNhbiBzZXJ2ZSB0aGUgSFRNTCB3aXRoOgoKLi4gY29kZS1ibG9jazo6IHNoZWxsCgogICAgICAgIHB5dGhvbjMuOCAtbSBodHRwLnNlcnZlciA4MDgwIC0tZGlyZWN0b3J5IGRpc3QKCgpMaWNlbnNlCj09PT09PT0KCi4uIGFkbW9uaXRpb246OgogICAgQ29weXJpZ2h0IChDKSAyMDIxIElOUEUuCgogICAgU3BhdGlvVGVtcG9yYWwgT3BlbiBSZXNlYXJjaCBNYW5hZ2VyIFdlYiBTZXJ2aWNlIGlzIGZyZWUgc29mdHdhcmU7IHlvdSBjYW4gcmVkaXN0cmlidXRlIGl0IGFuZC9vciBtb2RpZnkgaXQKICAgIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgTUlUIExpY2Vuc2U7IHNlZSBMSUNFTlNFIGZpbGUgZm9yIG1vcmUgZGV0YWlscy4K readmeEtag: '"751dddacd0c0bdc199da722b7ddec11b390edf08"' readmeLastModified: Sat, 12 Nov 2022 16:23:23 GMT repositoryId: 431249435 description: SpatioTemporal Open Research Manager Web Service Specification created: '2021-11-23T20:49:59Z' updated: '2022-12-26T15:55:04Z' language: null archived: false stars: 5 watchers: 3 forks: 0 owner: storm-platform logo: https://avatars.githubusercontent.com/u/93740635?v=4 license: MIT repoEtag: '"604a81eb98495526657f2abb2a7c5b8c30a643fd6f8d236c231fcc62fdf737b2"' repoLastModified: Mon, 26 Dec 2022 15:55:04 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/yndlingsfar/serverless-openapi-integration-helper v3: true repositoryMetadata: base64Readme: >- <p align="center">
  <a href="https://npmjs.com/package/serverless-openapi-integration-helper">
    <img src="https://flat.badgen.net/npm/v/serverless-openapi-integration-helper?icon=npm&label=npm@latest"></a>
<a href="https://www.npmjs.com/package/serverless-openapi-integration-helper">
    <img src="https://flat.badgen.net/npm/dt/serverless-openapi-integration-helper?icon=npm"></a>
  <a href="https://packagephobia.now.sh/result?p=serverless-openapi-integration-helper">
    <img src="https://flat.badgen.net/packagephobia/install/serverless-openapi-integration-helper"></a>
  <a href="https://www.npmjs.com/package/serverless-openapi-integration-helper">
    <img src="https://flat.badgen.net/npm/license/serverless-openapi-integration-helper"></a>
  <br/>
</p>

_Feedback is appreciated! If you have an idea for how this plugin/library can be improved (or even just a complaint/criticism) then please open an issue._

# Serverless Plugin: AWS Api Gateway integration helper

1. [Overview](#overview)
1. [Installation & Setup](#installation--setup)
1. [Plugin configuration](#plugin-configuration)  
1. [Usage](#usage)
1. [Command](#commands)
1. [CORS Generator](#cors-generator)
1. [AUTO-MOCK Generator](#auto-mock-generator)
1. [VALIDATION Generator](#validation-generator)
1. [PROXY Manager](#proxy-manager)
1. [Configuration Reference](#configuration-reference)
1. [Known Issues](#known-issues)
   1. [Stage Deployment](#stage-deployment)
   1. [Variable Resolving](#variable-resolving)
1. [Example](#example)
1. [Approach to a functional test of schema validation](#approach-to-a-functional-test-of-schema-validation)

# Overview 
The plugin provides the functionality to merge [OpenApiSpecification files](https://swagger.io/specification/) (formerly known as swagger) with one or multiple YML files containing the the x-amazon-apigateway extensions.
There are several use-cases to keep both information separated, e.g. it is needed to deploy different api gateway integrations depending on a stage environment.

When dealing with functional tests you do not want to test the production environment, but only a mocking response.

**The plugin supports YML based OpenApi3 specification files only**

## Features
- Serverless 3.x Support
- deploy stage dependent x-amazon-apigateway integrations
- separate infrastructure (aws) from openapi specification
- use mock integrations for functional testing
- auto-generating CORS methods, headers and api gateway mocking response
- hook into package & deploy lifeCycle and generate combined openApi files on the fly during deployment
- auto-inject generated openApi file into the Body property of specified API Gateway
- generate mocking responses without specifying x-amazon-apigateway-integration objects
- generate request-validation blocks
- generate all required x-amazon-apigateway-integration objects automatically
- full proxy generation support with **[NEW]:** feature: [PROXY Manager](#proxy-manager)

See the **examples** folder for a full working [example](https://github.com/yndlingsfar/serverless-openapi-integration-helper/tree/main/examples)

# Installation & Setup

Run `npm install` in your Serverless project.

`$ npm install --save-dev serverless-openapi-integration-helper`

Add the plugin to your serverless.yml file

```yml
plugins:
  - serverless-openapi-integration-helper
```
# Plugin configuration
You can configure the plugin under the key **openApiIntegration**. See
See [Configuration Reference](#configuration-reference) for a list of available options

The value for `openApiIntegration` must be either a single object or an array of objects.
For an array of objects, the `apiResourceName` property must be specified in each object to determine
which `Aws::ApiGateway::RestApi` resource to insert the merged API specification into,
while for a single object it's optional and will default to the first found
`Aws::ApiGateway::RestApi` resource.

The mapping array must be used to configure where the files containing the **x-amazon-apigateway-integration** blocks are located.

```yml
openApiIntegration:
    package: true #New feature! Hook into the package & deploy process
    inputFile: schema.yml
    mapping:
       - stage: [dev, prod] #multiple stages
         path: integrations
       - stage: test #single stage
         path: mocks
```

In the above example all YML files inside the _integrations_ directory will be processed and merged with the schema.yml file when deploying the dev stage
```shell
serverless deploy --stage=dev
```

To use a different x-amazon-apigateway to perform functional tests (with mocking responses e.g) the directory mock is processed and merged with the schema.yml file when deploying the test stage
```shell
serverless deploy --stage=test
```

## Multiple APIs

To handle multiple APIs, supply an array as the value of `openApiIntegrations`
instead of a single object, and specify the target `Aws::ApiGateway::RestApi`
resource for each item using the `apiResourceName` property, and make sure to specify
unique output files by setting `outputDirectory` and/or `outputFile` uniquely:

```yml
openApiIntegration:
  - package: true
    inputFile: schema-1.yml
    apiResourceName: Api1
    outputFile: api-1.yml
    mapping:
      - stage: [dev, prod]
        path: integrations/api-1
  - package: true
    inputFile: schema-2.yml
    apiResourceName: Api2
    outputFile: api-2.yml
    mapping:
       - stage: [dev, prod]
         path: integrations/api-2

...

resources:
  Resources:
    Api1:
      Type: AWS::ApiGateway::RestApi
      Properties:
        Body: ~ # auto-generated by plugin from schema-1

        ...

    Api2:
       Type: AWS::ApiGateway::RestApi
       Properties:
          Body: ~ # auto-generated by plugin from schema-2

          ...
```

# Usage
You can setup a fully working API GATEWAY with any openApi 3.0 specification file
First create the input file containing the [OpenApiSpecification](https://swagger.io/specification/)
```yml
# ./schema.yml
openapi: 3.0.0
info:
  description: User Registration
  version: 1.0.0
  title: UserRegistration
paths:
  /api/v1/user:
    post:
      summary: adds a user
      requestBody:
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/Customer'
      responses:
        '201':
          description: user created
components:
  schemas:
    Customer:
      type: object
      required:
        - email_address
        - password
      properties:
        email_address:
          type: string
          example: test@example.com
        password:
          type: string
          format: password
          example: someStrongPassword#
```

The plugin will generate the **x-amazon-apigateway integrations** objects for all methods that do not have an integration.

```yml
#generate a file containing a gateway mock integration in the directory /mocks
serverless integration create --output mocks --type mock --stage=test

#generate a file containing the production integration in the directory integrations/
serverless integration create --output integrations --type http --stage=prod
```

Supported types are
- http_proxy
- http
- aws
- aws_proxy
- mock


The plugin now generates a merged file during deployment that is automatically injected in your serverless resources

```shell
#Create OpenApi File containing mocking responses (usable in functional tests) and deploy to ApiGateway
serverless deploy --stage==test
```

```shell
#Create OpenApi File containing the production integration and deploy to ApiGateway
serverless deploy --stage=prod
```

The generated output is automatically injected in the **resources.Resources.YOUR_API_GATEWAY.Properties.Body** property
```yml
resources:
  Resources:
    ApiGatewayRestApi:
      Type: AWS::ApiGateway::RestApi
      Properties:
        ApiKeySourceType: HEADER
        Body: ~ #autogenerated by plugin
        Description: "Some Description"
        FailOnWarnings: false
        Name: ${opt:stage, self:provider.stage}-some-name
        EndpointConfiguration:
          Types:
            - REGIONAL
    ApiGatewayDeployment:
      Type: AWS::ApiGateway::Deployment
      Properties:
        RestApiId:
          Ref: ApiGatewayRestApi
        StageName: ${opt:stage, self:provider.stage}
```

# Commands

## Manual merge
The generate command can be used independently with
```yml
serverless integration merge --stage=dev
```
Of course then the API Gateway Body property has to be specified manually

```yml
resources:
  Resources:
    ApiGatewayRestApi:
      Type: AWS::ApiGateway::RestApi
      Properties:
        ApiKeySourceType: HEADER
        Body: ${file(openapi-integration/api.yml)}
```

# CORS generator

The plugin can generate full CORS support out of the box. 
```yml
openApiIntegration:
  cors: true
  ...
```

If enabled, the plugin generates all required OPTIONS methods as well as the required header informations and adds a mocking response to API Gateway. 
You can customize the CORS templates by placing your own files inside a directory **openapi-integration** (in your project root). The following files can be overwritten:

| Filename        | Description |
| ------------- |:-------------:| 
| headers.yml    | All headers required for CORS support |
| integration.yml      | Contains the x-amazon-apigateway-integration block  |
| path.yml| OpenApi specification for the OPTIONS method       |
| response-parameters.yml| The response Parameters of the x-amazon-apigateway-integration responses      |

See the [EXAMPLES](https://github.com/yndlingsfar/serverless-openapi-integration-helper/tree/main/examples) directory for detailed instructions.

# Auto Mock Generator
If enabled, the plugin generates mocking responses for all methods that do not have an x-amazon-apigateway-integration block defined.
It takes the first 2xx response defined in the openApi specification and generates a simple mocking response on the fly
```yml
openApiIntegration:
  autoMock: true
  ...
```

When using the autoMock feature, you do not need to specify inputPath mappings, since all endpoints are mocked automatically

```yml
openApiIntegration:
    package: true
    inputFile: schema.yml
    mapping: ~
```

# VALIDATION generator

The plugin supports full request validation out of the box
```yml
openApiIntegration:
  validation: true
  ...
```

If enabled, the plugin generates the x-amazon-apigateway-request-validators blocks and adds a basic request validation to all methods.
You can customize the VALIDATION template by placing your own files inside a directory **openapi-integration** (in your project root). The following files can be overwritten:

| Filename        | Description |
| ------------- |:-------------:| 
| request-validator.yml    | The x-amazon-apigateway-request-validators block |

See the [EXAMPLES](https://github.com/yndlingsfar/serverless-openapi-integration-helper/tree/main/examples) directory for detailed instructions.

# Proxy Manager

The proxymanager feature automates the complete generation of an HTTP proxy integration.
You only have to define the target URL and all necessary AWS integration blocks are generated on-the-fly during deployment.


```yml
openApiIntegration:
   cors: true
   validation: true
   mapping:
      - stage: [dev, prod]
        proxyManager:
           type: http_proxy
           baseUrl: https://www.example.com
           pattern: "(?<=api\/v1)\/.+"
  ...
```

**With this setting, no separate integration files need to be created**

A combination of your own and auto-generated files is still possible without any problems

## Proxy Manager configuration
### type
at the moment only http_proxy supported

### baseUrl
The base url is required to map the path variable from the openapi specification to the URI from the aws integration.

Example:

```yml
#original openapi specification
paths:
   /api/v1/user:
    post:
      ... 
```

will be translated to

```yml
#generated openapi specification output
paths:
   /api/v1/user:
      post:
         ...
         x-amazon-apigateway-integration:
            type: http_proxy
            passthroughBehavior: when_no_match
            httpMethod: POST
            uri: https://www.example.com/api/v1/user
```

### pattern

The pattern can be used to adapt the mapping of the base url using regexp, to remove a prefix, or a version string

Example:

```yml
baseUrl: https://www.example.com
pattern: "(?<=api\/v1)\/.+"
```

will translate the route **/api/v1/user** to https://www.example.com/user

# Configuration Reference

configure the plugin under the key **openApiIntegration**

```yml
openApiIntegration:
  inputFile: schema.yml #required
  package: true #optional, defaults to false 
  inputDirectory: ./ #optional, defaults to ./
  cors: true #optional, defaults to false
  autoMock: true #optional, defaults to false
  validation: true #optional, defaults to false
  mapping: #optional, can be completely blank if autoMock option is enabled
    - stage: [dev, prod] #multiple stages
      path: integrations
      proxyManager: #optional
         type: http_proxy
         baseUrl: https://example.com
         pattern: "(?<=v1)\/.+"
    - stage: test #single stage
      path: mocks/customer.yml
  outputFile: api.yml #optional, defaults to api.yml
  outputDirectory: openapi-integration #optional, defaults to ./openapi-integration
```

# Known Issues

## Stage deployment
When using serverless framework only to deploy your aws resources **without having any lambda functions or triggers**, the AWS Gateway deploymemt does not behave as expected.
Any deployment to an existing stage will be ignored, since CloudFormation does not redeploy a stage if the DeploymentIdentifier has not changed.

The plugin [serverless-random-gateway-deployment-id](https://www.npmjs.com/package/serverless-random-gateway-deployment-id) solves this problem by adding a random id to the deployment-name and all references to it on every deploy

See the **examples** folder for a full working [example](https://github.com/yndlingsfar/serverless-openapi-integration-helper/tree/main/examples)

## Variable Resolving
Serverless variables inside the openapi integration files are not resolved correctly when using the package & deploy hooks. This problem can be solved by using the api gateway STAGE VARIABLES.

See the **examples** folder for a full working [example](https://github.com/yndlingsfar/serverless-openapi-integration-helper/tree/main/examples)

# Example
```yml
service:
  name: user-registration

provider:
  name: aws
  stage: dev
  region: eu-central-1

plugins:
  - serverless-openapi-integration-helper
  
openApiIntegration:
  inputFile: schema.yml
  package: true
  mapping:
    - path: integrations
      stage: [dev, prod]
    - path: mocks/customer.yml
      stage: test

functions:

resources:
  Resources:
    ApiGatewayRestApi:
      Type: AWS::ApiGateway::RestApi
      Properties:
        ApiKeySourceType: HEADER
        Body: ~
        Description: "Some Description"
        FailOnWarnings: false
        Name: ${opt:stage, self:provider.stage}-some-name
        EndpointConfiguration:
          Types:
            - REGIONAL
    ApiGatewayDeployment:
      Type: AWS::ApiGateway::Deployment
      Properties:
        RestApiId:
          Ref: ApiGatewayRestApi
        StageName: ${opt:stage, self:provider.stage}
```

```
serverless deploy --stage=test
```

```
serverless deploy --stage=prod
```

# Approach to a functional test of schema validation
The plugin works well in combination with the [serverless-plugin-test-helper](https://www.npmjs.com/package/serverless-plugin-test-helper) to automate tests against the deployed api gateway

## Install The plugin test helper package

```shell
npm install --save-dev serverless-plugin-test-helper
```

add the plugin as a plugin dependency in your serverless configuration file and configure the plugin according to the [Readme](https://www.npmjs.com/package/serverless-plugin-test-helper)
```yml
#./serveless.yml
plugins:
  - serverless-plugin-test-helper
  - serverless-openapi-integration-helper

[...]

resources:
   Outputs:
      GatewayUrl: # This is the key that will be used in the generated outputs file
         Description: This is a helper for functional tests
         Value: !Join
            - ''
            - - 'https://'
              - !Ref ApiGatewayRestApi
              - '.execute-api.'
              - ${opt:region, self:provider.region}
              - '.amazonaws.com/'
              - ${opt:stage, self:provider.stage}

   Resources:
    ApiGatewayRestApi:
      Type: AWS::ApiGateway::RestApi
      Properties:
        ApiKeySourceType: HEADER
        Body: ~
        Description: User Registration (${opt:stage, self:provider.stage})
        FailOnWarnings: false
        Name: ${opt:stage, self:provider.stage}-gateway
        EndpointConfiguration:
          Types:
            - REGIONAL
    ApiGatewayDeployment:
      Type: AWS::ApiGateway::Deployment
      Properties:
        RestApiId:
          Ref: ApiGatewayRestApi
        StageName: ${opt:stage, self:provider.stage}
```

## Testing the schema validation
Add a functional test (e.g. with jest)

```javascript
//tests/registration.js
import {getOutput} from 'serverless-plugin-test-helper';
import axios from 'axios';

axios.defaults.adapter = require('axios/lib/adapters/http'); //Todo

const URL = getOutput('GatewayUrl');
test('request validation on registration', async () => {
    expect.assertions(1);
    const {status} = await axios.post(URL + '/api/v1/user',
        {
            "email_address": "test@example.com",
            "password": "someStrongPassword#"
        },
        {
            headers: {
                'Content-Type': 'application/json',
            }
        });
    expect(status).toEqual(201);
});

test('request validation on registration (invalid request)', async () => {
    expect.assertions(1);
    try {
        await axios.post(URL + '/api/v1/user',
            {
                "email": "test@example.com"
            },
            {
                headers: {
                    'Content-Type': 'application/json',
                }
            });
    } catch (e) {
        expect(e.response).toMatchObject({
            statusText: 'Bad Request',
            status: 400
        });
    }
});
```
Then perform the functional test
```shell
serverless deploy --stage=test
npm test
serverless remove --stage=test
```

The command will
- merge the openapi specification file with the MOCK integration configured before
- deploy to API Gateway in an isolated TEST infrastructure environment (your other environments will not be affected since we are deploying to a separated gateway)
- perform the test and verify that schema validation is correct
- remove all TEST resources if test succeeded

See the **examples** folder for a full working [example](https://github.com/yndlingsfar/serverless-openapi-integration-helper/tree/main/examples)
 readmeEtag: '"27c3a79bf8c3dde4733c71aebb44de2891ea8e7c"' readmeLastModified: Sat, 15 Apr 2023 06:50:31 GMT repositoryId: 354271536 description: >- The plugin separates x-amazon-apigateway extension syntax from your openapi3 files created: '2021-04-03T11:15:21Z' updated: '2024-11-30T21:04:41Z' language: JavaScript archived: false stars: 5 watchers: 1 forks: 5 owner: yndlingsfar logo: https://avatars.githubusercontent.com/u/8264387?v=4 license: MIT repoEtag: '"203cdac84bedf3917202bec3459d647e6de2db8b66e112745915af4f7ae4f5d5"' repoLastModified: Sat, 30 Nov 2024 21:04:41 GMT foundInMaster: true category: Testing id: 3fde089ee2ce97b7e54b375ff3c66db4 - source: openapi3 tags repository: https://github.com/0xtheprodev/spring-clean-example v3: true id: eb10ef37c7a934b2e312427da9c2dbe4 repositoryMetadata: base64Readme: >- IyBzcHJpbmctY2xlYW4tZXhhbXBsZQoKWyFbSmF2YV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9qYXZhLSUyM0VEOEIwMC5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWphdmEmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly93d3cuamF2YS5jb20vKQpbIVtTcHJpbmddKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvc3ByaW5nLSUyMzZEQjMzRi5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPXNwcmluZyZsb2dvQ29sb3I9d2hpdGUpXShodHRwczovL3NwcmluZy5pby8pClshW0FwYWNoZSBUb21jYXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvYXBhY2hlJTIwdG9tY2F0LSUyM0Y4REM3NS5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWFwYWNoZS10b21jYXQmbG9nb0NvbG9yPWJsYWNrKV0oaHR0cHM6Ly90b21jYXQuYXBhY2hlLm9yZy8pClshW09wZW5BUEldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvb3BlbmFwaS02QkE1Mzk/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPW9wZW5hcGktaW5pdGlhdGl2ZSZsb2dvQ29sb3I9ZmZmKV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykKWyFbR3JhcGhRTF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS8tR3JhcGhRTC1FMTAwOTg/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWdyYXBocWwmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly9ncmFwaHFsLm9yZy8pClshW09wZW4gSXNzdWVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtcmF3LzB4VGhlUHJvRGV2L3NwcmluZy1jbGVhbi1leGFtcGxlP3N0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL2dpdGh1Yi5jb20vMHhUaGVQcm9EZXYvc3ByaW5nLWNsZWFuLWV4YW1wbGUvaXNzdWVzKQpbIVtDbG9zZWQgSXNzdWVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtY2xvc2VkLXJhdy8weFRoZVByb0Rldi9zcHJpbmctY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L3NwcmluZy1jbGVhbi1leGFtcGxlL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FjbG9zZWQpClshW09wZW4gUHVsbHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy1wci1yYXcvMHhUaGVQcm9EZXYvc3ByaW5nLWNsZWFuLWV4YW1wbGU/c3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9zcHJpbmctY2xlYW4tZXhhbXBsZS9wdWxscykKWyFbQ2xvc2VkIFB1bGxzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMtcHItY2xvc2VkLXJhdy8weFRoZVByb0Rldi9zcHJpbmctY2xlYW4tZXhhbXBsZT9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tLzB4VGhlUHJvRGV2L3NwcmluZy1jbGVhbi1leGFtcGxlL3B1bGxzP3E9aXMlM0FwcitpcyUzQWNsb3NlZCkKWyFbQ29udHJpYnV0b3JzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9jb250cmlidXRvcnMvMHhUaGVQcm9EZXYvc3ByaW5nLWNsZWFuLWV4YW1wbGU/c3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vZ2l0aHViLmNvbS8weFRoZVByb0Rldi9zcHJpbmctY2xlYW4tZXhhbXBsZS9ncmFwaHMvY29udHJpYnV0b3JzKQpbIVtBY3Rpdml0eV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGFzdC1jb21taXQvMHhUaGVQcm9EZXYvc3ByaW5nLWNsZWFuLWV4YW1wbGU/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsYWJlbD1tb3N0JTIwcmVjZW50JTIwYWN0aXZpdHkpXShodHRwczovL2dpdGh1Yi5jb20vMHhUaGVQcm9EZXYvc3ByaW5nLWNsZWFuLWV4YW1wbGUvcHVsc2UpCgojIyBEZXNjcmlwdGlvbgoKX0V4YW1wbGUgQXBwbGljYXRpb24gSW50ZXJmYWNlIHVzaW5nIFNwcmluZyBmcmFtZXdvcmsgaW4gSmF2YV8KClRoaXMgZXhhbXBsZSBzaG93Y2FzZXMgUmVwb3NpdG9yeSBQYXR0ZXJuIGluIEhleGFnb25hbCBBcmNoaXRlY3R1cmUgXyhhbHNvIGtub3duIGFzIENsZWFuIEFyY2hpdGVjdHVyZSlfLiBIZXJlIHdlIGhhdmUgdHdvIEVudGl0aWVzIC0gQm9va3MgYW5kIEF1dGhvcnMsIHdob3NlIHJlbGF0aW9uc2hpcHMgaGF2ZSBiZWVuIGV4cGxvaXRlZCB0byBjcmVhdGUgQ1JVRCBlbmRwb2ludCBpbiBSRVNUIHVuZGVyIE9wZW5BUEkgc3RhbmRhcmQuCgojIyBJbnN0YWxsYXRpb24KCi0gUnVuIHRoZSBhcHBsaWNhdGlvbiB1c2luZyBbTWF2ZW5dKGh0dHBzOi8vbWF2ZW4uYXBhY2hlLm9yZy8pOgoKICBgYGBzaAogICQgLi9tdm53IHNwcmluZy1ib290OnJ1bgogIGBgYAoKLSBGb3IgV2luZG93cyB1c2VyczoKCiAgYGBgcG93ZXJzaGVsbAogICQgbXZudyBzcHJpbmctYm9vdDpydW4KICBgYGAKCiMjIFRlc3RpbmcKCi0gUnVuIHRlc3Qgc3VpdGUgYWxvbmcgd2l0aCBDb3ZlcmFnZSByZXBvcnRpbmc6CgogIGBgYHNoCiAgJCAuL212bncgamFjb2NvOnByZXBhcmUtYWdlbnQgdGVzdCBpbnN0YWxsIGphY29jbzpyZXBvcnQKICBgYGAKCiAgb3IgZm9yIFdpbmRvd3MKCiAgYGBgcG93ZXJzaGVsbAogICQgbXZudyBqYWNvY286cHJlcGFyZS1hZ2VudCB0ZXN0IGluc3RhbGwgamFjb2NvOnJlcG9ydAogIGBgYAoKIyMgU3dhZ2dlciBVSQoKLSBPcGVuIFN3YWdnZXIgVUkgYXQgYGxvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWlgIGFmdGVyIHJ1bm5pbmcgdGhlIGFwcGxpY2F0aW9uLgoKIyMgTGljZW5zZQoKJmNvcHk7IE1JVCBMaWNlbnNlCg== readmeEtag: '"be85193f84db044d8ace0e35e81f1cb84b558a6c"' readmeLastModified: Tue, 13 Dec 2022 10:58:53 GMT repositoryId: 510053852 description: Clean Architecture Example with Spring Framework (Java) created: '2022-07-03T14:57:09Z' updated: '2025-01-14T02:26:55Z' language: Java archived: false stars: 5 watchers: 0 forks: 0 owner: 0xTheProDev logo: https://avatars.githubusercontent.com/u/14367736?v=4 license: MIT repoEtag: '"18ce7a9d0e32407a1bf33d87db599297b5f68315831db91d70d8ac9fccedb57b"' repoLastModified: Tue, 14 Jan 2025 02:26:55 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/progyan1997/spring-clean-example - source: openapi3 tags repository: https://github.com/martingrambow/openisbt v3: true repositoryMetadata: base64Readme: >- IyBvcGVuSVNCVAoKT3BlbklTQlQgaXMgYW4gSW50ZWxsaWdlbnQgU2VydmljZSBCZW5jaG1hcmsgVG9vbCB0byBiZW5jaG1hcmsgbWljcm9zZXJ2aWNlLWJhc2VkIGFwcGxpY2F0aW9ucyBiYXNlZCBvbiB0aGVpciBPcGVuQVBJMy4wIGludGVyZmFjZSBkZXNjcmlwdGlvbiBmaWxlcy4gIApbQmVuY2htYXJraW5nIE1pY3Jvc2VydmljZSBQZXJmb3JtYW5jZTogQSBQYXR0ZXJuLWJhc2VkIEFwcHJvYWNoXShodHRwczovL2RiZXJtYmFjaC5naXRodWIuaW8vcHVibGljYXRpb25zLzIwMjAtc2FjLWRhZHMtbWljcm9zZXJ2aWNlcy5wZGYpIAoKSWYgeW91IHVzZSB0aGlzIHNvZnR3YXJlIGluIGEgcHVibGljYXRpb24sIHBsZWFzZSBjaXRlIGl0IGFzOgoKIyMjIFRleHQKTWFydGluIEdyYW1ib3csIEx1a2FzIE1ldXNlbCwgRXJpayBXaXR0ZXJuLCBEYXZpZCBCZXJtYmFjaC4gKipCZW5jaG1hcmtpbmcgTWljcm9zZXJ2aWNlIFBlcmZvcm1hbmNlOiBBIFBhdHRlcm4tYmFzZWQgQXBwcm9hY2gqKi4gSW46IFByb2NlZWRpbmdzIG9mIHRoZSAzNXRoIEFDTSBTeW1wb3NpdW0gb24gQXBwbGllZCBDb21wdXRpbmcgKFNBQyAyMDIwKS4gQUNNIDIwMjAuCgojIyMgQmliVGVYCmBgYFRlWApAaW5wcm9jZWVkaW5nc3twYXBlcl9ncmFtYm93X2JlbmNobWFya2luZ19taWNyb3NlcnZpY2VzLAoJdGl0bGUgPSB7e0JlbmNobWFya2luZyBNaWNyb3NlcnZpY2UgUGVyZm9ybWFuY2U6IEEgUGF0dGVybi1iYXNlZCBBcHByb2FjaH19LAoJYm9va3RpdGxlID0ge1Byb2NlZWRpbmdzIG9mIHRoZSAzNXRoIEFDTSBTeW1wb3NpdW0gb24gQXBwbGllZCBDb21wdXRpbmcgKFNBQyAyMDIwKX0sCglwdWJsaXNoZXIgPSB7QUNNfSwKCWF1dGhvciA9IHtNYXJ0aW4gR3JhbWJvdyBhbmQgTHVrYXMgTWV1c2VsIGFuZCBFcmlrIFdpdHRlcm4gYW5kIERhdmlkIEJlcm1iYWNofSwKCXllYXIgPSB7MjAyMH0KfQpgYGAKCkEgZnVsbCBsaXN0IG9mIG91ciBbcHVibGljYXRpb25zXShodHRwczovL3d3dy5tY2MudHUtYmVybGluLmRlL21lbnVlL2ZvcnNjaHVuZy9wdWJsaWthdGlvbmVuL3BhcmFtZXRlci9lbi8pIGFuZCBbcHJvdG90eXBlc10oaHR0cHM6Ly93d3cubWNjLnR1LWJlcmxpbi5kZS9tZW51ZS9mb3JzY2h1bmcvcHJvdG90eXBlcy9wYXJhbWV0ZXIvZW4vKSBpcyBhdmFpbGFibGUgb24gb3VyIGdyb3VwIHdlYnNpdGUuCgojIyBSZWxlYXNlcwoqIHZlcnNpb24gMC4xIChkMWI2YzRmKSB3YXMgdXNlZCBpbiBwdWJsaWNhdGlvbiAiQmVuY2htYXJraW5nIE1pY3Jvc2VydmljZSBQZXJmb3JtYW5jZTogQSBQYXR0ZXJuLWJhc2VkIEFwcHJvYWNoIgoqIHZlcnNpb24gMC4yIChmZDllNTA2KSBpbnRyb2R1Y2VzIGEgYmF0Y2ggYmVuY2htYXJrIHZpYSB0ZXJtaW5hbAoqIHZlcnNpb24gMC4zIChhMTA3MzdiKSBhcHBsaWNhdGlvbi13aWRlIGJlbmNobWFya2luZyB1c2luZyBtYW51YWwgc2VydmljZSBsaW5rcwoqIHZlcnNpb24gMC40ICh0YmQpIENsZWFuIHVwIGRlcHJlY2F0ZWQgY29tcG9uZW50cywgYmV0dGVyIGRvY3VtZW50YXRpb24KCiMjIExpY2Vuc2UKClRoZSBjb2RlIGluIHRoaXMgcmVwb3NpdG9yeSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIFtNSVRdKC4vTElDRU5TRSkgbGljZW5zZS4KCiMgVXNhZ2UKKiBbQmVuY2htYXJrXShvcGVuSVNCVEJhdGNoLm1kKQoKIyMgRnV0aGVyIEluZm9ybWF0aW9uIChmb3IgZGV2ZWxvcGVycykKKiBbQWJzdHJhY3Qgb3BlcmF0aW9ucyBhbmQgbWF0Y2hpbmcgdW5pdHNdKGFic3RyYWN0T3BlcmF0aW9ucy5tZCkKKiBbTGlua2luZyh1bml0cyldKGxpbmtpbmdVbml0cy5tZCkK readmeEtag: '"aa8703ff965e7666323692f1aba40f16937070ec"' readmeLastModified: Thu, 03 Sep 2020 21:50:12 GMT repositoryId: 196092022 description: >- OpenISBT is an Intelligent Service Benchmark Tool to benchmark microservice-based applications based on their OpenAPI3.0 interface description files. created: '2019-07-09T22:25:59Z' updated: '2024-09-26T18:26:19Z' language: Kotlin archived: false stars: 5 watchers: 1 forks: 1 owner: martingrambow logo: https://avatars.githubusercontent.com/u/18425060?v=4 license: MIT repoEtag: '"284ca0523591269dfe9ad6abd5928a6c2866d846bbbb2732b006c22042843bf4"' repoLastModified: Thu, 26 Sep 2024 18:26:19 GMT foundInMaster: true category: Server id: 75a45f8b179099c1bdb9fd6966eb5657 - source: openapi3 tags repository: https://github.com/webpro/parse-openapi v3: true repositoryMetadata: base64Readme: >- IyBwYXJzZS1vcGVuYXBpCgpUaGUgT3BlbkFQaSB2MyBwYXJzZXIgZXh0cmFjdGVkIGZyb20KW29wZW5hcGktdHlwZXNjcmlwdC1jb2RlZ2VuXShodHRwczovL2dpdGh1Yi5jb20vZmVyZGlrb29tZW4vb3BlbmFwaS10eXBlc2NyaXB0LWNvZGVnZW4pLgpTbyBmaXJzdCBvZiBhbGwsIGFsbCBjcmVkaXRzIGdvIHRvCltGZXJkaSBLb29tZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9mZXJkaWtvb21lbikgZm9yIGhpcyB3b3JrIG9uIHRoaXMgZ3JlYXQKbGlicmFyeS4gT25seSB0aGUgcGFyc2VyIGlzIG5lZWRlZCB0byByZW5kZXIgdGhlIG1vZGVscywgc2VydmljZXMgYW5kIGhvb2tzIHdpdGgKY3VzdG9tIHRlbXBsYXRlcywgaGVuY2UgdGhpcyBzdHJpcHBlZCBkb3duIHZlcnNpb24uCgojIyBJbnN0YWxsYXRpb24KCiAgICBucG0gaW5zdGFsbCAtRCBwYXJzZS1vcGVuYXBpIG9wZW5hcGktdHlwZXMKCiMjIEFQSQoKYGBgdHlwZXNjcmlwdAppbXBvcnQgeyByZWFkRmlsZSwgd3JpdGVGaWxlIH0gZnJvbSAnZnMvcHJvbWlzZXMnOwppbXBvcnQgeyBwYXJzZSB9IGZyb20gJ3BhcnNlLW9wZW5hcGknOwppbXBvcnQgeyByZW5kZXJNb2RlbCwgcmVuZGVyU2VydmljZSB9IGZyb20gJy4vdGVtcGxhdGVzJzsKaW1wb3J0IHR5cGUgeyBPcGVuQVBJVjMgfSBmcm9tICdvcGVuYXBpLXR5cGVzJzsKCmNvbnN0IHNwZWNzOiBPcGVuQVBJVjMgPSBKU09OLnBhcnNlKGF3YWl0IHJlYWRGaWxlKCdzd2FnZ2VyLmpzb24nLCAndXRmOCcpKTsKCmNvbnN0IHsgdmVyc2lvbiwgc2VydmVyLCBtb2RlbHMsIHNlcnZpY2VzIH0gPSBwYXJzZShzcGVjcyk7Cgptb2RlbHMuZm9yRWFjaChtb2RlbCA9PiB3cml0ZUZpbGUoYCR7bW9kZWwubmFtZX0udHNgLCByZW5kZXJNb2RlbChtb2RlbCkpKTsKc2VydmljZXMuZm9yRWFjaChzZXJ2aWNlID0+CiAgd3JpdGVGaWxlKGAke3NlcnZpY2UubmFtZX0udHNgLCByZW5kZXJTZXJ2aWNlKHNlcnZpY2UpKQopOwpgYGAKCiMjIEV4YW1wbGUgcmVuZGVyIGZ1bmN0aW9uCgpUaGlzIGNvdWxkIGJlIGFueXRoaW5nIGZyb20gSGFuZGxlYmFycyB0ZW1wbGF0ZXMgdG8gdGVtcGxhdGUgbGl0ZXJhbHMuIEhlcmUncyBhCm1pbmltYWwgZXhhbXBsZToKCmBgYHR5cGVzY3JpcHQKaW1wb3J0IHR5cGUgeyBPcGVyYXRpb24sIFNlcnZpY2UgfSBmcm9tICdwYXJzZS1vcGVuYXBpJzsKCmNvbnN0IHJlbmRlck9wZXJhdGlvbiA9IChvcGVyYXRpb246IE9wZXJhdGlvbik6IHN0cmluZyA9PiB7CiAgY29uc3QgW3Jlc3VsdF0gPSBvcGVyYXRpb24ucmVzdWx0czsKICByZXR1cm4gYAovKioKICogJHtvcGVyYXRpb24uc3VtbWFyeX0KICogJHtvcGVyYXRpb24uZGVzY3JpcHRpb259CiAqIEByZXR1cm5zICR7cmVzdWx0LnR5cGV9ICR7cmVzdWx0LmRlc2NyaXB0aW9ufQogJHtvcGVyYXRpb24uZGVwcmVjYXRlZCA/ICcqIEBkZXByZWNhdGVkJyA6ICcnfQogKi8KZXhwb3J0IGZ1bmN0aW9uICR7b3BlcmF0aW9uLm5hbWV9KFsuLi5dKTogUHJvbWlzZTwke3Jlc3VsdC50eXBlfT4gewogIHJldHVybiByZXF1ZXN0KHsKICAgIG1ldGhvZDogJyR7b3BlcmF0aW9uLm1ldGhvZH0nLAogICAgcGF0aDogXGAke29wZXJhdGlvbi5wYXRofVxgLAogICAgWy4uLl0KICB9KTsKfWA7Cn07CgpleHBvcnQgZGVmYXVsdCAoc2VydmljZTogU2VydmljZSkgPT4gewogIHJldHVybiBzZXJ2aWNlcy5vcGVyYXRpb25zLm1hcChyZW5kZXJPcGVyYXRpb24pLmpvaW4oJzsnKTsKfTsKYGBgCg== readmeEtag: '"02d1173e98535a6d16dc4936a6b82ded608a834e"' readmeLastModified: Thu, 31 Mar 2022 17:31:04 GMT repositoryId: 476396257 description: OpenAPI v3 parser created: '2022-03-31T16:48:53Z' updated: '2023-02-11T12:24:23Z' language: TypeScript archived: false stars: 5 watchers: 1 forks: 0 owner: webpro logo: https://avatars.githubusercontent.com/u/456426?v=4 license: MIT repoEtag: '"560a4f9b301cba8dda499ed7fc42877abf02d1017913ca36ae80f726565b4c64"' repoLastModified: Sat, 11 Feb 2023 12:24:23 GMT foundInMaster: true category: Parsers id: d21709b143857d908bd5a8d8fb73c5be - source: openapi3 tags repository: https://github.com/tuxivinc/mock-api v3: true repositoryMetadata: base64Readme: >- # Mock-api 
Api for Mock, performances testing, resilience and show functionalities in demo (docker for example, with healthcheck)

## Information
* sources : https://github.com/Tuxivinc/mock-api

## Functions
* Mock Json response (set by POST or mount volume)
* Mock response size (download file) 
* Mock response HTTP Status
* Get Hostname
* Get JVM args
* Get environment variables
* Get request information
  * body, headers and URL
* Simulation healthcheck
  * get http code
  * set http code return
* Sleep response time
  * Storage in memory or pass in request path
* Shutdown application with error code
* Latency on startup
* Demo session sharing (for load balancing example)
  * Store session local
  * Multicast session

## For testing
1/ Deploy container : docker run -d -p 8050:8050 --name mock-api tuxivinc/mock-api

2/ Call swagger for API infos : http://localhost:8050/mock-api/swagger-ui.html

## More information
### Functions list
#### Latency of startup
| | |
| --------------------- | ---------- |
| **What** | add time before service starting Up |
| **How** | add `-e STARTING_LATENCY_MS=[XXXX]` on command line docker run (XXXX is a time in ms) | 
| **Usage** | Gatling resilience testing, Docker restart policy |

#### JVM information
| | |
| --------------------- | ---------- |
| **What** | Get Jvm information |
| **How** | `curl -X GET "http://localhost:8050/mock-api/getinfos/jvm"` |
| **Usage** | Debug |

#### Environment variables
| | |
| --------------------- | ---------- |
| **What** | Get Environment variables |
| **How** | `curl -X GET "http://localhost:8050/mock-api/getinfos/varenv"` |
| **Usage** | Debug |

#### Hostname
| | |
| --------------------- | ---------- |
| **What** | Get Hostname of container |
| **How** | `curl -X GET "http://localhost:8050/mock-api/getinfos/hostname"` |
| **Usage** | Debug |

#### Shutdown
| | |
| --------------------- | ---------- |
| **What** | Kill JVM with specify exit Code |
| **How** |  |
| | Run container with --restart on-failure |
| | Call `curl -X POST "http://localhost:8050/mock-api/shutdown/12"` -> The container restart |
| | Call `curl -X POST "http://localhost:8050/mock-api/shutdown/0"` -> The container don't restart |
| **Usage** | Docker Restart policy, Gatling Resilience test |

#### HealthCheck
| | |
| --------------------- | ---------- |
| **What** | Set return code of Docker HealthCheck |
| **How** |  |
| | Call `curl -X GET "http://localhost:8050/mock-api/healthcheck"` -> Return healthcheck code (by default 200) |
| | Call `curl -X POST "http://localhost:8050/mock-api/healthcheck/setcode/500"` -> Set return code at 500 |
| **Usage** | Docker Restart policy, Docker HealthCheck, Gatling Resilience, Traefik expose service |

#### Sniffer
| | |
| --------------------- | ---------- |
| **What** | Get information of call (GET/PUT/POST/DELETE/OPTIONS/HEAD/PATCH) |
| | Hostname container |
| | HostIp |
| | Url received |
| | Uri received |
| | Quesry String |
| | List headers |
| | Body content |
| **How** | `curl -X GET "http://localhost:8050/mock-api/sniffer/headers"` |
| **Usage** | Demo, Debug |

#### Error code
| | |
| --------------------- | ---------- |
| **What** | Choose Http status of response |
| **How** |  |
| | return http status 404 : `curl -X GET "http://localhost:8050/mock-api/simulation/error/404"` |
| | return http status 500 : `curl -X GET "http://localhost:8050/mock-api/simulation/error/500"` |
| **Usage** | Gatling Resilience, Demo |

#### Response time
| | |
| --------------------- | ---------- |
| **What** | Choose response time (value in path) |
| **How** | Response in 5s : `curl -X GET "http://localhost:8050/mock-api/simulation/timeout/5000"` |
| **Usage** | Gatling Resilience |

#### Response time in memory
| | |
| --------------------- | ---------- |
| **What** | Choose response time (value store in memory of server, by default: 0ms) |
| **How** |  |
| | Set response time to 10s: `curl -X POST "http://localhost:8050/mock-api/simulation/timeout-memory/10000"`  |
| | Get response in 10s: `curl -X GET "http://localhost:8050/mock-api/simulation/timeout-memory"`  |
| **Usage** | Gatling Resilience |

#### Response size
| | |
| --------------------- | ---------- |
| **What** | Get file with specify size on request |
| **How** |  |
| | Get File 200Ko: `curl -X GET "http://localhost:8050/mock-api/simulation/response-size/200" --output /tmp/file200` |
| | Get File 2Mo: `curl -X GET "http://localhost:8050/mock-api/simulation/response-size/2000" --output /tmp/file2000` |
| **Usage** | Gatling Resilience, Demo, Test |

#### Session not shared
| | |
| --------------------- | ---------- |
| **What** | Store session (by token key) in server without share scales instances |
| **How** |  |
| | Run a secondary server (replace `-p 8050:8050` by `-p 8060:8050`) |
| | Call first server with token "test", return timestamp of created session: `curl -X GET "http://localhost:8050/mock-api/session/share/test"`  |
| | Call secondary server with token "test", return same timestamp of first server: `curl -X GET "http://localhost:8060/mock-api/session/share/test"`  |
| **Usage** | Demo serverless constraints Docker |

#### Shared Session
| | |
| --------------------- | ---------- |
| **What** | Share session (on multicast) between all servers |
| **How** |  |
| | Run a secondary server (replace `-p 8050:8050` by `-p 8060:8050`) |
| | Call first server with token "test", return timestamp of created session: `curl -X GET "http://localhost:8050/mock-api/session/notshare/test"`  |
| | Call fisrt server with token "test", return same timestamp |
| | Call secondary server with token "test", return timestamp of created session, but different of first server: `curl -X GET "http://localhost:8060/mock-api/session/notshare/test"`  |
| | Call secondary server with token "test", return same timestamp |
| **Usage** | Demo serverless constraints Docker  |

#### Mock Json Response
| | |
| --------------------- | ---------- |
| **What** | Return Json file store in `/mock-response` |
| **How** |  |
| | Create directory `/xxx/mock` |
| | In this directory, create json file (see examples after) |
| | Run container with -v /xxx/mock:/mock-response |
| | Call `curl -X GET "http://localhost:8050/mock-api/mock/test"` -> Return 200 with test file content |
| | Call `curl -X GET "http://localhost:8050/mock-api/mock/test2"` -> Return 200 with test2 file content |
| | Call `curl -X GET "http://localhost:8050/mock-api/mock/test3"` -> Return 404, no file found |
| **Usage** | Mock Json call, Gatling Resilience test |

Examples of File
* File test: 
    ```
    {
        "username":"John",
        "id":564
    }
    ``` 
* File test2: 
    ```
    {
        "username":"Robert",
        "id":789
    }
    ```

#### POST Mock Json Response
| | |
| --------------------- | ---------- |
| **What** | Add Json file store in `/mock-response` |
| **How** |  |
| | curl -X POST "http://localhost:8050/mock-api/mock/testrep" -H  "accept: application/json" -H  "Content-Type: application/json" -d "{\"key\":\"haha\"}" -> créer un fichier testrep |
| **Usage** | Mock Json call, Gatling Resilience test | readmeEtag: '"6268bf4ad56eccef518d070565379ffd54f7f72a"' readmeLastModified: Mon, 13 Sep 2021 11:48:35 GMT repositoryId: 216524928 description: >- SpringBoot app for mock API https://hub.docker.com/repository/docker/tuxivinc/mock-api created: '2019-10-21T09:09:03Z' updated: '2024-08-29T13:46:43Z' language: Java archived: false stars: 5 watchers: 1 forks: 2 owner: Tuxivinc logo: https://avatars.githubusercontent.com/u/29063459?v=4 repoEtag: '"a72b5dd5edb05866d527e40cbee8113c6424e17c5d5c2b9d99eb50e244e8a640"' repoLastModified: Thu, 29 Aug 2024 13:46:43 GMT foundInMaster: true category: Server Implementations id: f3c2edded6ea75e1012182cbee4cd398 - source: openapi3 tags repository: https://github.com/snehalkaranje/springboot-blog-rest-api v3: true repositoryMetadata: base64Readme: >- # springboot-blog-rest-api
<a href="https://maven.apache.org/"><img alt="maven" src="https://img.shields.io/badge/maven-%20-blue"/></a>
<a href="https://spring.io/projects/spring-boot/"><img alt="Spring Boot" src="https://img.shields.io/badge/SpringBoot-2.6.3-blue"/></a>
<a href="https://docs.oracle.com/en/java/javase/11/"><img alt="Java 11" src="https://img.shields.io/badge/Java-11-blue"/></a>
<a href="https://dev.mysql.com/"><img alt="MySQL" src="https://img.shields.io/badge/MySQL-8.0.28-blue"/></a>


## Dependencies
Dependencies used are as follows:

<a href="https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-web"><img alt="spring-boot-starter-web" src="https://img.shields.io/badge/dependency-SpringBootStarterWeb-blue"/></a>
<a href="https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-data-jpa"><img alt="spring-boot-starter-data-jpa" src="https://img.shields.io/badge/dependency-SpringBootStarterDataJPA-blue"/></a>
<a href="https://mvnrepository.com/artifact/mysql/mysql-connector-java"><img alt="mysql-connector-java" src="https://img.shields.io/badge/dependency-MySQLConnectorJava-blue"/></a>
<a href="https://mvnrepository.com/artifact/org.projectlombok/lombok"><img alt="lombok" src="https://img.shields.io/badge/dependency-Lombok-blue"/></a>
<a href="https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-devtools"><img alt="spring-boot-devtools" src="https://img.shields.io/badge/dependency-SpringBootDevTools-blue"/></a>
<a href="https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-test"><img alt="spring-boot-starter-test" src="https://img.shields.io/badge/dependency-SpringBootStarterTest-blue"/></a>
<a href="https://mvnrepository.com/artifact/org.modelmapper/modelmapper/3.0.0"><img alt="model-mapper" src="https://img.shields.io/badge/dependency-ModelMapper-blue"/></a>
<a href="https://mvnrepository.com/artifact/org.springframework.boot/spring-boot-starter-validation"><img alt="spring-boot-starter-validation" src="https://img.shields.io/badge/dependency-SpringBootStarterValidation-blue"/></a> 


## Project Architecture
```
3 layer Architecture : Client <--> Controller <--> Service <--> DAO <--> DB

- Controller
  - Keeps all spring REST controllers
  - Define end points
- Service
  - all service classes that hold business logic
- DAO
  - Repository layer
  - keep all spring JPA data repository
  - Communicates with database
```

## Project Structure Explained
```
- controller
  - Contains all Spring MVC controllers
- entity
  - Contains all JPA entities
- service
  - Contains service interfaces
- service.impl
  - Contains servie classes
- utils
  - Contains utilities and classes
- repository
  - Contains Spring Data JPA repositories. (JPA repositories are by default Transactional. Default implementation class of the JpaRepository interface is SimpleJpaRepository class)
- exception
  - Contains all custom exceptions
- payload
  - Contains DTOs (Data Transfer Objects) that act as a payload between client and server
```

## API document generation
OpenAPI 3 specification document for API is auto generated using <a href="https://github.com/springdoc/springdoc-openapi">springdoc-openapi-ui</a>

As per custom path provided in application.properties, auto generated API doc is generated at custom path: 
```
http://localhost:<port>/api-docs/ 
```

Doc can also be accessed in the Swagger UI at path:
```
http://localhost:<port>/swagger-ui.html
```


## Annotations
### Spring Boot Annotations
<a href="https://docs.spring.io/spring-boot/docs/2.0.x/reference/html/using-boot-using-springbootapplication-annotation.html"><img alt="SpringBootApplication" src="https://img.shields.io/badge/@SpringBootApplication-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/ResponseStatus.html"><img alt="ResponseStatus" src="https://img.shields.io/badge/@ResponseStatus-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/stereotype/Service.html"><img alt="Service" src="https://img.shields.io/badge/@Service-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/RestController.html"><img alt="RestController" src="https://img.shields.io/badge/@RestController-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/stereotype/Controller.html"><img alt="Controller" src="https://img.shields.io/badge/@Controller-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/ResponseBody.html"><img alt="ResponseBody" src="https://img.shields.io/badge/@ResponseBody-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/RequestMapping.html"><img alt="RequestMapping" src="https://img.shields.io/badge/@RequestMapping-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/PostMapping.html"><img alt="PostMapping" src="https://img.shields.io/badge/@PostMapping-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/GetMapping.html"><img alt="GetMapping" src="https://img.shields.io/badge/@GetMapping-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/PathVariable.html"><img alt="PathVariable" src="https://img.shields.io/badge/@PathVariable-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/PutMapping.html"><img alt="PutMapping" src="https://img.shields.io/badge/@PutMapping-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/DeleteMapping.html"><img alt="DeleteMapping" src="https://img.shields.io/badge/@DeleteMapping-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/RequestParam.html"><img alt="RequestParam" src="https://img.shields.io/badge/@RequestParam-blue"/></a>
<a href="https://docs.spring.io/spring-javaconfig/docs/1.0.0.M4/reference/html/ch02s"><img alt="Bean" src="https://img.shields.io/badge/@Bean-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/ExceptionHandler.html"><img alt="ExceptionHandler" src="https://img.shields.io/badge/@ExceptionHandler-blue"/></a>
<a href="https://docs.spring.io/spring-framework/docs/current/javadoc-api/org/springframework/web/bind/annotation/ControllerAdvice.html"><img alt="ControllerAdvice" src="https://img.shields.io/badge/@ControllerAdvice-blue"/></a>

### Lombok annotations
<a href="https://projectlombok.org/api/lombok/Data.html"><img alt="Data" src="https://img.shields.io/badge/@Data-blue"/></a>
<a href="https://projectlombok.org/api/lombok/AllArgsConstructor.html"><img alt="AllArgsConstructor" src="https://img.shields.io/badge/@AllArgsConstructor-blue"/></a>
<a href="https://projectlombok.org/api/lombok/NoArgsConstructor.html"><img alt="NoArgsConstructor" src="https://img.shields.io/badge/@NoArgsConstructor-blue"/></a>

### JPA annotations
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#Entity"><img alt="Entity" src="https://img.shields.io/badge/@Entity-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#Table"><img alt="Table" src="https://img.shields.io/badge/@Table-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#Id"><img alt="Id" src="https://img.shields.io/badge/@Id-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#GeneratedValue"><img alt="GeneratedValue" src="https://img.shields.io/badge/@GeneratedValue-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#Column"><img alt="Column" src="https://img.shields.io/badge/@Column-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#ManyToOne"><img alt="ManyToOne" src="https://img.shields.io/badge/@ManyToOne-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#JoinColumn"><img alt="JoinColumn" src="https://img.shields.io/badge/@JoinColumn-blue"/></a>
<a href="https://www.techferry.com/articles/hibernate-jpa-annotations.html#OneToMany"><img alt="OneToMany" src="https://img.shields.io/badge/@OneToMany-blue"/></a>

### Spring Boot starter validation (Hibernate Validator)
<a href="https://docs.oracle.com/javaee/7/api/javax/validation/constraints/NotNull.html"><img alt="NotNull" src="https://img.shields.io/badge/@NotNull-blue"/></a>
<a href="https://docs.oracle.com/javaee/7/api/javax/validation/constraints/Size.html"><img alt="Size" src="https://img.shields.io/badge/@Size-blue"/></a>
<a href="https://docs.oracle.com/javaee/7/api/javax/validation/constraints/Min.html"><img alt="Min" src="https://img.shields.io/badge/@Min-blue"/></a>
<a href="https://docs.oracle.com/javaee/7/api/javax/validation/constraints/Max.html"><img alt="Max" src="https://img.shields.io/badge/@Max-blue"/></a>
<a href="https://javaee.github.io/javaee-spec/javadocs/javax/validation/constraints/Email.html"><img alt="Email" src="https://img.shields.io/badge/@Email-blue"/></a>
<a href="https://javaee.github.io/javaee-spec/javadocs/javax/validation/constraints/NotEmpty.html"><img alt="NotEmpty" src="https://img.shields.io/badge/@NotEmpty-blue"/></a> 
<a href="https://javaee.github.io/javaee-spec/javadocs/javax/validation/constraints/NotBlank.html"><img alt="NotBlank" src="https://img.shields.io/badge/@NotBlank-blue"/></a> 
<a href="https://docs.oracle.com/javaee/7/api/javax/validation/Valid.html"><img alt="Valid" src="https://img.shields.io/badge/@Valid-blue"/></a>
 readmeEtag: '"ffec71145aca75f6f0feee527c8e309247fce13f"' readmeLastModified: Tue, 25 Jun 2024 09:30:27 GMT repositoryId: 453950856 description: 'Spring boot REST API application ' created: '2022-01-31T09:49:29Z' updated: '2024-09-15T20:07:21Z' language: Java archived: false stars: 5 watchers: 1 forks: 0 owner: SnehalKaranje logo: https://avatars.githubusercontent.com/u/58742622?v=4 repoEtag: '"e8842913ef947743d249cdace5124c89b013fcddd85421258c1e88dd1ce11fbb"' repoLastModified: Sun, 15 Sep 2024 20:07:21 GMT foundInMaster: true category: Server id: 92461140252d2f808a6e413eb747f965 - source: - openapi3 tags - openapi31 tags repository: https://github.com/apiaddicts/sonar-openapi v3: true v3_1: true id: 514c0837da7b67edc46795a00a5e8da2 repositoryMetadata: base64Readme: >- 
# 🛠️ Sonar OpenApi (plugin) ![Release](https://img.shields.io/badge/release-1.1.1-purple) ![Swagger](https://img.shields.io/badge/-openapi-%23Clojure?style=flat&logo=swagger&logoColor=white) ![Java](https://img.shields.io/badge/java-%23ED8B00.svg?style=flat&logo=openjdk&logoColor=white)  [![License: LGPL v3](https://img.shields.io/badge/license-LGPL_v3-blue.svg)](https://www.gnu.org/licenses/lgpl-3.0) 

Sonar OpenApi (plugin) is a code analyzer for OpenAPI specifications,  is the spiritual successor of [SonarOpenApi](https://github.com/societe-generale/sonar-openapi), carrying on from the point where it left off with support of Apiaddicts community.

### This repository is intended for :octocat: **community** use, it can be modified and adapted without commercial use. If you need a version, support or help for your **enterprise** or project, please contact us 📧 devrel@apiaddicts.org
### 💡 If you have an idea for a rule but you are not sure that everyone needs it you can implement a [custom rule](CustomRules.md) available only for you.

[![Twitter](https://img.shields.io/badge/Twitter-%23000000.svg?style=for-the-badge&logo=x&logoColor=white)](https://twitter.com/APIAddicts) 
[![Discord](https://img.shields.io/badge/Discord-%235865F2.svg?style=for-the-badge&logo=discord&logoColor=white)](https://discord.gg/ZdbGqMBYy8)
[![LinkedIn](https://img.shields.io/badge/linkedin-%230077B5.svg?style=for-the-badge&logo=linkedin&logoColor=white)](https://www.linkedin.com/company/apiaddicts/)
[![Facebook](https://img.shields.io/badge/Facebook-%231877F2.svg?style=for-the-badge&logo=Facebook&logoColor=white)](https://www.facebook.com/apiaddicts)
[![YouTube](https://img.shields.io/badge/YouTube-%23FF0000.svg?style=for-the-badge&logo=YouTube&logoColor=white)](https://www.youtube.com/@APIAddictslmaoo)

# 🙌 Join the **Sonar OpenApi (plugin)** Adopters list 
📢 If Sonar OpenApi is part of your organization's toolkit, we kindly encourage you to include your company's name in our Adopters list. 🙏 This not only significantly boosts the project's visibility and reputation but also represents a small yet impactful way to give back to the project.

| Organization  | Description of Use / Referenc |
|---|---|
|  [CloudAppi](https://cloudappi.net/)  | Apification and generation of microservices |
| [Madrid Digital](https://www.comunidad.madrid/servicios/sede-electronica/madrid-digital/)  | Generation of microservices  |
| [Apiquality](https://apiquality.io/)  | Generation of microservices  |

# 👩🏽‍💻  Contribute to ApiAddicts 

We're an inclusive and open community, welcoming you to join our effort to enhance ApiAddicts, and we're excited to prioritize tasks based on community input, inviting you to review and collaborate through our GitHub issue tracker.

Feel free to drop by and greet us on our GitHub discussion or Discord chat. You can also show your support by giving us some GitHub stars ⭐️, or by following us on Twitter, LinkedIn, and subscribing to our YouTube channel! 🚀

[!["Buy Me A Coffee"](https://www.buymeacoffee.com/assets/img/custom_images/orange_img.png)](https://www.buymeacoffee.com/apiaddicts)


## ⚙️ Features

* Full compatibility with OpenAPI v2.0, v3.0.0, v3.0.1, v3.0.2, v3.0.3 and v3.1.0

![SonarOpenApi in action](sonarqube.jpg)

<a name="install"></a>
## Installing

To install the plugin, you need to compile it, then install it in your SonarQube server.

1. Make sure you have at least JDK1.8 installed, as well as Maven 3.0.5 or later. They must be present in your PATH.
2. In the master directory of the project, type `mvn install`. This will compile the project and generate the artifacts.
3. Copy the file `sonar-openapi-plugin/target/sonar-openapi-plugin-<version>.jar` into directory `extensions/plugins/`of
   your SonarQube installation (you can install a local copy [from here](https://www.sonarqube.org/downloads/) for testing).
4. Restart your SonarQube server.

## Analyzing your projects

To analyze your projects, you must first [install](#install) the plugin.

### Configuring sonar-scanner

Once installed, configure the analysis properties by creating the `sonar-project.properties` at the root of your project.
[Sonar-scanner](https://docs.sonarqube.org/latest/analysis/scan/sonarscanner/) will look for this file when 
launching the analysis. Alternatively, you can define these properties as environment variables or using the Sonar Maven plugin.

An example configuration file is provided below for reference:

```properties
# must be unique in a given SonarQube instance
sonar.projectKey=test:openapi
# this is the name and version displayed in the SonarQube UI. Was mandatory prior to SonarQube 6.1.
sonar.projectName=OpenAPI plugin tests
sonar.projectVersion=1.0
 
# Path is relative to the sonar-project.properties file. Replace "\" by "/" on Windows.
# This property is optional if sonar.modules is set. 
sonar.sources=.
  
# Encoding of the source code. Default is default system encoding
sonar.sourceEncoding=UTF-8
# Select the language to use for analysis 
sonar.language=openapi
```

For details about how to configure SonarQube Scanner to analyze your projects, see [the documentation](https://docs.sonarqube.org/display/SCAN/Analyzing+with+SonarQube+Scanner).

### Configuring the plugin

The plugin automatically scans all `.yaml` and `.json` files that are compatible with the OpenAPI spec.

Is considered to be compatible with OpenAPI v2 spec if the file contains the root key `swagger` and compatible with the v3 if contains `openapi`.

### Running the analysis

* Make sure the SonarQube server is running
* [Generate a token](https://docs.sonarqube.org/latest/user-guide/user-token/) to authenticate to the server, or ask for one to your administrator
* With `sonar-scanner` in you path, just launch the tool from the directory where you have created `sonar-project.properties`.
* Make sure you specify the sonar server and token when launching the analysis

You should obtain an output similar to that:

```text
D:\git\testSonar>sonar-scanner -Dsonar.host.url=<your Sonar server> -Dsonar.login=<authorization token>
INFO: -------------  Scan OpenAPI plugin tests
INFO: Base dir: D:\git\testSonar
INFO: Working dir: d:\git\testSonar\.sonar
INFO: Source paths: .
INFO: Source encoding: UTF-8, default locale: en_US
INFO: Load server rules
INFO: Load server rules (done) | time=229ms
INFO: Index files
INFO: 4 files indexed
INFO: Quality profile for openapi: Sonar way
INFO: Sensor SonarJavaXmlFileSensor [java]
INFO: Sensor SonarJavaXmlFileSensor [java] (done) | time=1ms
INFO: Sensor OpenAPI Scanner Sensor [openapi]
INFO: Sensor OpenAPI Scanner Sensor [openapi] (done) | time=270ms
INFO: Sensor Zero Coverage Sensor
INFO: Sensor Zero Coverage Sensor (done) | time=8ms
INFO: No SCM system was detected. You can use the 'sonar.scm.provider' property to explicitly specify it.
INFO: Calculating CPD for 6 files
INFO: CPD calculation finished
INFO: Analysis report generated in 215ms, dir size=92 KB
INFO: Analysis reports compressed in 37ms, zip size=17 KB
INFO: Analysis report uploaded in 75ms
INFO: ANALYSIS SUCCESSFUL, you can browse <your Sonar server>/dashboard?id=test%3Aopenapi
INFO: Note that you will be able to access the updated dashboard once the server has processed the submitted analysis report
INFO: More about the report processing at <your Sonar server>/api/ce/task?id=AWZZE5MdehEa_CTMQA3m
INFO: Task total time: 3.356 s
INFO: ------------------------------------------------------------------------
INFO: EXECUTION SUCCESS
INFO: ------------------------------------------------------------------------
```

Then, log into your SonarQube server and go to your project to see the found violations (if any).

### Skipping rules

Sometimes, it makes sense to disable a rule altogether. The plugin comes with a way to control which rule is enabled on
a specific file. Use it with caution as it is generally a bad practice to disable a rule from code!

The `x-nosonar` OpenAPI extension completely disables a rule. Add it to the top-level OpenAPI document to disable a rule
or a set of rules:

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Swagger Petstore
  license:
    name: MIT
servers:
  - url: http://petstore.swagger.io/v1
x-nosonar: [ RuleId1, RuleId2 ]
```

You can pass either a string or an array of string to the extension.

To disable a rule only in a specific API element, use the `x-sonar-disable` extension. To enable an otherwise globally
disable rule, use the `x-sonar-enable` extension. They are recognized in any API element that supports extensions, except
on the top-level document.

```yaml
openapi: "3.0.0"
info:
  version: 1.0.0
  title: Swagger Petstore
  license:
    name: MIT
servers:
  - url: http://petstore.swagger.io/v1
x-nosonar: [ RuleId1, RuleId2 ]
paths:
  /pets:
    get:
      # This re-enables RuleId1 in this operation only (it is not inherited by child elements like tags or parameters)
      x-sonar-enable: RuleId1
      summary: List all pets
      operationId: listPets
      tags:
        - pets
      parameters:
        - name: filter
          in: query
          description: attribute on which to filter
          required: false
          schema:
            type: string
          # This disables RuleId3 locally in this parameter (it is not inherited by child elements like schema)
          x-sonar-disable: RuleId3
```

As for `x-nosonar`, the `x-sonar-disable` and `x-sonar-enable` extensions accept a single string or an array of strings.

<a name="testing"></a>
## Testing

To run tests locally follow these instructions.

### Build the Project and Run Unit Tests

To build the plugin and run its unit tests, execute this command from the project's root directory:

    mvn clean install

### Integration Tests

Integration tests are provided with the plugin. To include them, use the "its" profile:

    mvn -Pits clean install

If you are running behind an enterprise proxy, specify the java proxy options on the command line:

- http.proxyHost
- http.proxyPort
- http.proxyUser
- http.proxyPassword
- https.proxyHost
- https.proxyPort
- https.proxyUser
- https.proxyPassword

### Performing a new release

Validate that all is correct:

`mvn clean package -Prelease`

Deploy:

`mvn clean deploy -Prelease`

## 💛 Sponsors
<p align="center">
	<a href="https://apiaddicts.org/">
    	<img src="https://apiaddicts.cloudappi.net/web/image/4248/LOGOCloudappi2020Versiones-01.png" alt="cloudappi" width="150"/>
        <img src="https://www.comunidad.madrid/sites/default/files/styles/block_teaser_image/public/img/logos-simbolos/logo_centrado_md.png?itok=4rTUhmcj" alt="md" width="150"/>
        <img src="https://apiquality.io/wp-content/uploads/2022/09/cropped-logo-apiquality-principal-1-170x70.png" height = "75">
        <img src="https://apiaddicts-web.s3.eu-west-1.amazonaws.com/wp-content/uploads/2022/03/17155736/cropped-APIAddicts-logotipo_rojo.png" height = "75">
	</a>
</p> readmeEtag: '"1a316dfa03327419380fcce3f9f7165795eb4cf7"' readmeLastModified: Thu, 05 Sep 2024 15:43:59 GMT repositoryId: 474063683 description: Evaluation engine for OpenAPI/Swagger API definitions in SonarQube created: '2022-03-25T15:25:42Z' updated: '2025-12-08T23:23:04Z' language: Java archived: false stars: 13 watchers: 3 forks: 1 owner: apiaddicts logo: https://avatars.githubusercontent.com/u/31730093?v=4 license: LGPL-3.0 repoEtag: '"b9e444e4493828565b8a3287759a0a3be4ce8e01af9a10181abb2bef8400295f"' repoLastModified: Mon, 08 Dec 2025 23:23:04 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/llm-studios/swaggerboy v3: true id: e0a696d363aa2760afca627945d4ba39 repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyYm95CgpBbiBBSS1wb3dlcmVkIGNvcGlsb3QgZm9yIEFQSSBkZXNpZ24gYW5kIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBnZW5lcmF0aW9uLiBTd2FnZ2VyYm95IGhlbHBzIHRvIGRlc2lnbiBhbmQgZG9jdW1lbnQgQVBJcyB0aHJvdWdoIGFuIGludGVyYWN0aXZlLCBjb252ZXJzYXRpb24tYmFzZWQgYXBwcm9hY2guCgojIyBRdWljayBTdGFydAoKYGBgdHlwZXNjcmlwdApucHggc3dhZ2dlcmJveQpgYGAKCiMjIEZlYXR1cmVzCgotIPCfpJYgSW50ZXJhY3RpdmUgQVBJIGRlc2lnbiB0aHJvdWdoIG5hdHVyYWwgbGFuZ3VhZ2UgY29udmVyc2F0aW9ucwotIPCfk50gT3BlbkFQSSAzLjEuMCBzcGVjaWZpY2F0aW9uIGdlbmVyYXRpb24KLSDwn5KhIFByb2FjdGl2ZSBmZWF0dXJlIHJlY29tbWVuZGF0aW9ucwoKIyMgS2V5IENhcGFiaWxpdGllcyBvZiBTd2FnZ2VyYm95CgotIEd1aWRlcyB1c2VycyB0aHJvdWdoIGhpZ2gtbGV2ZWwgQVBJIGRlc2lnbgotIENyZWF0ZXMgZGV0YWlsZWQgc2NoZW1hIGNvbXBvbmVudHMgYW5kIGRhdGEgbW9kZWxzCi0gRXN0YWJsaXNoZXMgcmV1c2FibGUgY29tcG9uZW50cyAocmVzcG9uc2VzLCBwYXJhbWV0ZXJzLCBleGFtcGxlcykKLSBWYWxpZGF0ZXMgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucwotIEV4dGVuZHMgZW5kcG9pbnQgb3BlcmF0aW9ucyB3aXRoIGltcGxlbWVudGF0aW9uIGRldGFpbHMKCiMjIEJ1aWx0IFdpdGgKCi0gW09wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLjEuMF0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykKLSBbT3BlbkFJIGZvciBMTE0tYmFzZWQgY29udmVyc2F0aW9uYWwgQUldKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYWkpCi0gW1ZlcmNlbCBBSSBTREtdKGh0dHBzOi8vZ2l0aHViLmNvbS92ZXJjZWwvYWkpCi0gW1N0b3BsaWdodCBlbGVtZW50cyBmb3IgaW50ZXJhY3RpdmUgT3BlbkFQSV0oaHR0cHM6Ly9naXRodWIuY29tL3N0b3BsaWdodGlvL2VsZW1lbnRzKQotIFtTaGFkY24vdWldKGh0dHBzOi8vZ2l0aHViLmNvbS9zaGFkY24tdWkvdWkpCi0gW2Fzc2lzdGFudC11aV0oaHR0cHM6Ly9naXRodWIuY29tL1lvbm9tL2Fzc2lzdGFudC11aSkKLSBbTmV4dC5qc10oaHR0cHM6Ly9naXRodWIuY29tL3ZlcmNlbC9uZXh0LmpzKQotIFtvcGVuYXBpMy10c10oaHR0cHM6Ly9naXRodWIuY29tL21ldGFkZXZwcm8vb3BlbmFwaTMtdHMpCgpodHRwczovL2xsbXN0dWRpb3MuZGUvCgojIyBMaWNlbnNlCgpTd2FnZ2VyYm95IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4K readmeEtag: '"9f72d0adfa966c85234d4df88d70ad041241122f"' readmeLastModified: Mon, 09 Dec 2024 12:28:56 GMT repositoryId: 886914571 description: AI-Copilot for OpenAPI Description Documents created: '2024-11-11T20:49:34Z' updated: '2025-05-31T15:39:58Z' language: TypeScript archived: false stars: 7 watchers: 1 forks: 0 owner: LLM-Studios logo: https://avatars.githubusercontent.com/u/151792168?v=4 license: MIT repoEtag: '"9257a2b453019de9adcc75c775e459c4b2f4ed825a7f64a648fc30d6b80ed874"' repoLastModified: Sat, 31 May 2025 15:39:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/istreamlabs/rest-api-lint v3: true repositoryMetadata: base64Readme: >- IyBSRVNUIEFQSSBMaW50aW5nIEFjdGlvbgoKWyFbLmdpdGh1Yi93b3JrZmxvd3MvY2kueWFtbF0oaHR0cHM6Ly9naXRodWIuY29tL2lzdHJlYW1sYWJzL3Jlc3QtYXBpLWxpbnQvd29ya2Zsb3dzLy5naXRodWIvd29ya2Zsb3dzL2NpLnlhbWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2lzdHJlYW1sYWJzL3Jlc3QtYXBpLWxpbnQvYWN0aW9ucz9xdWVyeT13b3JrZmxvdyUzQS5naXRodWIlMkZ3b3JrZmxvd3MlMkZjaS55YW1sKSBbIVtEb2NrZXIgQ2xvdWQgQnVpbGQgU3RhdHVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2RvY2tlci9jbG91ZC9idWlsZC9pc3RyZWFtbGFicy9yZXN0LWFwaS1saW50KV0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL2lzdHJlYW1sYWJzL3Jlc3QtYXBpLWxpbnQpIFshW0RvY2tlciBQdWxsc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9kb2NrZXIvcHVsbHMvaXN0cmVhbWxhYnMvcmVzdC1hcGktbGludCldKGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9pc3RyZWFtbGFicy9yZXN0LWFwaS1saW50KQoKQSBEb2NrZXIgaW1hZ2Ugd2l0aCBvcGluaW9uYXRlZCBSRVNUIEFQSSBsaW50aW5nIGZvciBpU3RyZWFtUGxhbmV0LiBTdXBwb3J0cyBHaXRIdWIgQWN0aW9ucy4KClRoaXMgaXMgYnVpbHQgdXNpbmcgW1N0b3BsaWdodCdzIFNwZWN0cmFsXShodHRwczovL3N0b3BsaWdodC5pby9vcGVuLXNvdXJjZS9zcGVjdHJhbCkuIFRoZSBjdXN0b20gcnVsZXNldCBpcyBkZWZpbmVkIGluIFtpc3AtcnVsZXMueWFtbF0oaHR0cHM6Ly9naXRodWIuY29tL2lzdHJlYW1sYWJzL3Jlc3QtYXBpLWxpbnQvYmxvYi9tYXN0ZXIvaXNwLXJ1bGVzLnlhbWwpLCB3aGVyZSB5b3UgY2FuIGZpbmQgdGhlIGRlc2NyaXB0aW9ucyBhbmQgcnVsZSBuYW1lcyAoaWYgeW91IHdhbnQgdG8gZGlzYWJsZSBpbmRpdmlkdWFsIHJ1bGVzKS4KCiMjIFVzYWdlCgpUaGVyZSBhcmUgdHdvIHJlY29tbWVuZGVkIHdheXMgdG8gdXNlIHRoaXM6CgoxLiBHaXRIdWIgQWN0aW9ucwoyLiBBbnkgc3lzdGVtIHRoYXQgc3VwcG9ydHMgRG9ja2VyIGltYWdlcwoKIyMjIEdpdEh1YiBBY3Rpb25zIEV4YW1wbGUKClNldCB1cCBhIHdvcmtmbG93IHRoYXQgdXNlcyB0aGlzIGltYWdlIHRvIGRvIEFQSSBsaW50aW5nIGFuZCB5b3UgYXJlIGRvbmU6CgpgYGB5YW1sCm9uOiBbcHVzaF0KCmpvYnM6CiAgYnVpbGQ6CiAgICBydW5zLW9uOiB1YnVudHUtbGF0ZXN0CiAgICBuYW1lOiBCdWlsZCAmIFRlc3QKICAgIHN0ZXBzOgogICAgICAtIHVzZXM6IGFjdGlvbnMvY2hlY2tvdXRAdjEKICAgICAgLSBuYW1lOiBBUEkgTGludAogICAgICAgIHVzZXM6IGlzdHJlYW1sYWJzL3Jlc3QtYXBpLWxpbnRAbWFzdGVyCiAgICAgICAgd2l0aDoKICAgICAgICAgIGZpbGVuYW1lOiBwYXRoL3RvL29wZW5hcGkueWFtbApgYGAKClRoZSBwYXRoIHlvdSBwYXNzIHNob3VsZCBiZSByZWxhdGl2ZSB0byB0aGUgY2hlY2tvdXQgZGlyZWN0b3J5LCB3aGljaCBpcyBhdXRvbWF0aWNhbGx5IG1vdW50ZWQgaW4gdGhlIGNvbnRhaW5lciBhbmQgYmVjb21lcyB0aGUgd29ya2luZyBkaXJlY3RvcnkuCgojIyMgRG9ja2VyIEV4YW1wbGUKCldoaWxlIHNpbWlsYXIgdG8gdGhlIGFib3ZlLCB5b3Ugd2lsbCBuZWVkIHRvIGRvIGFuIGV4dHJhIHN0ZXAgd2hlbiB1c2luZyBEb2NrZXIgZGlyZWN0bHkuIFlvdSBtdXN0IGVuc3VyZSB0aGF0IHRoZSB3b3JraW5nIGRpcmVjdG9yeSBnZXRzIG1vdW50ZWQgcHJvcGVybHkgaW50byB0aGUgY29udGFpbmVyLgoKWW91IGNhbiBlaXRoZXIgdXNlIHRoZSBwdWJsaXNoZWQgW2Bpc3RyZWFtbGFicy9yZXN0LWFwaS1saW50YF0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL2lzdHJlYW1sYWJzL3Jlc3QtYXBpLWxpbnQpIGNvbnRhaW5lciBvciBidWlsZCBpdCB5b3Vyc2VsZiBsb2NhbGx5LgoKYGBgc2gKIyBDcmVhdGUgdGhlIGltYWdlLgokIGRvY2tlciBidWlsZCAtdCByZXN0LWFwaS1saW50IC4KCiMgTW91bnQgdGhlIGN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkgdG8gYC90bXBgIGluIHRoZSBjb250YWluZXIgYW5kIHJ1biBsaW50aW5nLgokIGRvY2tlciBydW4gLWl0IC12ICQocHdkKTovdG1wIHJlc3QtYXBpLWxpbnQ6bGF0ZXN0IC90bXAvbXktb3BlbmFwaS55YW1sCgojIElmIHlvdSBoYXZlIGEgY3VzdG9tIGAuc3BlY3RyYWwueWFtbGAgd2l0aCBvdmVycmlkZXM6CiQgZG9ja2VyIHJ1biAtaXQgLXcgL3dvcmtzcGFjZSAtdiAkKHB3ZCk6L3dvcmtzcGFjZSByZXN0LWFwaS1saW50OmxhdGVzdCBvcGVuYXBpLnlhbWwKYGBgCgpUaGUgY29tbWFuZCB3aWxsIHJldHVybiBhIGAwYCBvbiBzdWNjZXNzIGFuZCBhIGAxYCBpZiBhbnkgZXJyb3JzIGFyZSBmb3VuZC4gV2FybmluZ3MgZG8gbm90IHRyaWdnZXIgYSBmYWlsdXJlLgoKIyMjIEV4Y2VwdGlvbnMgJiBPdmVycmlkZXMKCldoZW4gW3N0b3BsaWdodGlvL3NwZWN0cmFsIzc0N10oaHR0cHM6Ly9naXRodWIuY29tL3N0b3BsaWdodGlvL3NwZWN0cmFsL2lzc3Vlcy83NDcpIGdldHMgZml4ZWQgeW91IHdpbGwgYmUgYWJsZSB0byBwcm92aWRlIGluZGl2aWR1YWwgZXhjZXB0aW9ucyBvbiBhIGNhc2UtYnktY2FzZSBiYXNpcy4KClRvIGRpc2FibGUgYW4gZW50aXJlIHJ1bGUsIGUuZy4gdG8gc3VwcG9ydCBhIGxlZ2FjeSBBUEksIHlvdSBjYW4gcHJvdmlkZSB5b3VyIG93biBgLnNwZWN0cmFsLnlhbWxgIGZpbGUgaW4gdGhlIHJvb3Qgb2YgeW91ciByZXBvIHdoaWNoIHNldHMgc29tZSBydWxlcyB0byBgZmFsc2VgLiBOb3RlIHRoYXQgdGhpcyBkaXNhYmxlcyB0aGUgcnVsZSBmb3IgdGhlIF9lbnRpcmUgZmlsZV86CgpgYGB5YW1sCnJ1bGVzOgogIGlzbzg2MDE6IGZhbHNlCiAgc2VydmVyLXZlcnNpb246IGZhbHNlCmBgYAoKSXQncyBhbHNvIHBvc3NpYmxlIHRvIHNldCBpdCB0byBgd2FybmAgb3IgYGluZm9gLgoKVGhlIGBleHRlbmRzYCBmaWVsZCBpcyBhdXRvbWF0aWNhbGx5IGFkZGVkIChvciBhcHBlbmRlZCB0bykgYnkgdGhlIGxpbnRlciBzY3JpcHQgdG8gaW5qZWN0IHRoZSBpU1AgcnVsZXNldCBpbnRvIHlvdXIgY29uZmlnLgoKIyMjIExvY2FsIERldmVsb3BtZW50CgoxLiBJbnN0YWxsIGBub2RlIDEzLjE0LjBgIGxvY2FsbHkuCjIuIFJ1biBgbnBtIHJ1biBidWlsZGAgdG8gY29tcGlsZSB0aGUgYGlzcC1mdW5jdGlvbnNgLgozLiBSdW4gYG5vZGUgLi9lbnRyeXBvaW50LmpzIDxQQVRIX1RPX09QRU5BUElfU1BFQz5gIHRvIHJ1biB0aGUgbGludGVyLgoKVGhlIGBidWlsZGAgY29tbWFuZCBtYXkgbmVlZCB0byBiZSByZS1ydW4gdG8gcGljayB1cCBjaGFuZ2VzIG1hZGUgdG8gc29tZSBvZiB0aGUgYGlzcC1mdW5jdGlvbnNgLgoKIyMjIyBUZXN0cwoKYGBgc2gKIyBDb21waWxlIHRoZSBpc3AtZnVuY3Rpb25zCm5wbSBydW4gYnVpbGQKCiMgUnVuIHRlc3QKbnBtIHJ1biB0ZXN0CmBgYAoKIyMgTGljZW5zZQoKQ29weXJpZ2h0IMKpIDIwMjAgaVN0cmVhbVBsYW5ldCBDby4sIExMQwoKTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7IHlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4K readmeEtag: '"2558795008088634d324e2464f3dc23c9adfd971"' readmeLastModified: Tue, 27 Jun 2023 17:23:32 GMT repositoryId: 237052565 description: REST API linting using OpenAPI and Spectral created: '2020-01-29T18:23:40Z' updated: '2025-11-07T09:47:26Z' language: JavaScript archived: false stars: 6 watchers: 6 forks: 2 owner: istreamlabs logo: https://avatars.githubusercontent.com/u/30184888?v=4 license: Apache-2.0 repoEtag: '"6d44ed1f0e896b2c29c256c5af929b90375463b64ac4932932d4152bde688215"' repoLastModified: Fri, 07 Nov 2025 09:47:26 GMT foundInMaster: true category: - Description Validators - Server Implementations id: 1794143fa313ceb332ca759e0105bde5 - source: openapi3 tags repository: https://github.com/danielgtaylor/apibin v3: true repositoryMetadata: base64Readme: >- IyBBUEkgQmluCgpbIVtIVU1BIFBvd2VyZWRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUG93ZXJlZCUyMEJ5LUh1bWEtZmY1Zjg3KV0oaHR0cHM6Ly9odW1hLnJvY2tzLykgWyFbV29ya3MgV2l0aCBSZXN0aXNoXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1dvcmtzJTIwV2l0aC1SZXN0aXNoLWZmNWY4NyldKGh0dHBzOi8vcmVzdC5zaC8pIFshW0dpdEh1Yl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9kYW5pZWxndGF5bG9yL2FwaWJpbildKGh0dHBzOi8vZ2l0aHViLmNvbS9kYW5pZWxndGF5bG9yL2FwaWJpbikKClByb3ZpZGVzIGEgc2ltcGxlLCBtb2Rlcm4sIGV4YW1wbGUgQVBJIGZvciBkZW1vaW5nIG9yIGRlYnVnZ2luZyB2YXJpb3VzIGZlYXR1cmVzLCBpbmNsdWRpbmc6CgotIEhUVFAsIEhUVFBTIChUTFMpLCBhbmQgW0hUVFAvMl0oaHR0cHM6Ly9odHRwMi5naXRodWIuaW8vKQotIFtPcGVuQVBJIDNdKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pICYgW0pTT04gU2NoZW1hXShodHRwczovL2pzb24tc2NoZW1hLm9yZy8pCi0gQ2xpZW50LWRyaXZlbiBjb250ZW50IG5lZ290aWF0aW9uCiAgLSBgZ3ppcGAgJiBgYnJgIGNvbnRlbnQgZW5jb2RpbmcgZm9yIGxhcmdlIHJlc3BvbnNlcwogIC0gYEpTT05gLCBgWUFNTGAsICYgYENCT1JgIGZvcm1hdHMKLSBDb25kaXRpb25hbCByZXF1ZXN0cyB2aWEgYEVUYWdgIG9yIGBMYXN0TW9kaWZpZWRgCi0gRWNobyBiYWNrIHJlcXVlc3QgaW5mbyB0byBoZWxwIGRlYnVnZ2luZwotIENhY2hlZCByZXNwb25zZXMgdG8gdGVzdCBwcm94eSAmIGNsaWVudC1zaWRlIGNhY2hpbmcKLSBFeGFtcGxlIHN0cnVjdHVyZWQgZGF0YQogIC0gU2hvd3Mgb2ZmIGBvYmplY3RgLCBgYXJyYXlgLCBgc3RyaW5nYCwgYGRhdGVgLCBgYmluYXJ5YCwgYGludGVnZXJgLCBgbnVtYmVyYCwgYGJvb2xlYW5gLCBldGMuCi0gQSBzYW1wbGUgQ1JVRCBBUEkgZm9yIGJvb2tzICYgcmV2aWV3cyB3aXRoIHNpbXVsYXRlZCBzZXJ2ZXItc2lkZSB1cGRhdGVzCi0gSW1hZ2UgcmVzcG9uc2VzIGBKUEVHYCwgYFdFQlBgLCBgR0lGYCwgYFBOR2AgJiBgSEVJQ2AKLSBbUkZDNzgwN10oaHR0cHM6Ly9kYXRhdHJhY2tlci5pZXRmLm9yZy9kb2MvaHRtbC9yZmM3ODA3KSBzdHJ1Y3R1cmVkIGVycm9ycwoKVGhpcyBwcm9qZWN0IGlzIG9wZW4gc291cmNlOiBodHRwczovL2dpdGh1Yi5jb20vZGFuaWVsZ3RheWxvci9hcGliaW4KCllvdSBjYW4gcnVuIGl0IGxvY2FseSB2aWEgRG9ja2VyOgoKYGBgc2gKIyBTdGFydCB0aGUgc2VydmVyCiQgZG9ja2VyIHJ1biAtcCA4ODg4Ojg4ODggZ2hjci5pby9kYW5pZWxndGF5bG9yL2FwaWJpbjpsYXRlc3QKCiMgTWFrZSBhIHJlcXVlc3QKJCByZXN0aXNoIDo4ODg4L3R5cGVzCmBgYAo= readmeEtag: '"e0501c1769bc643849d54d79c1d29b1cf8e4730b"' readmeLastModified: Sun, 04 Feb 2024 17:41:03 GMT repositoryId: 484840594 description: Example API with modern features created: '2022-04-23T19:39:15Z' updated: '2025-09-09T03:26:21Z' language: Go archived: false stars: 7 watchers: 1 forks: 1 owner: danielgtaylor logo: https://avatars.githubusercontent.com/u/106826?v=4 license: MIT repoEtag: '"3940f323889ba69563d6c31b468f134c9368e8ee2ba678bdfe4688123b3c3fc6"' repoLastModified: Tue, 09 Sep 2025 03:26:21 GMT foundInMaster: true category: - Documentation - Server Implementations id: c41c5a3a517f2769a0b3c04fad395fbd - source: openapi3 tags repository: https://github.com/cchacin/microgen v3: true repositoryMetadata: base64Readme: >- ![Eclipse MicroProfile logo](images/microprofile-logo.png)
![Eclipse JakartaEE logo](images/jakartaee-logo.png)

# MicroGen

 MicroGen is a tool to generate java code based on an [OpenAPI](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md) Spec file.

## Features

- Server API Contract (JAX-RS)
- Request and Response java objects
  - Inner static `Builder` class and utility methods
- Enum value classes
- `@BeanParams` for request parameters
- Maven Archetype to quick start a JakartaEE + MicroProfile application
- Dockerfile(s) with a JakartaEE + MicroProfile application server

## User Guide

Given an OpenAPI Spec file:

```yaml
openapi: 3.0.0
info:
  description: This is a sample server Petstore server.
  version: 1.0.0
  title: Swagger Petstore
tags:
  - name: pet
    description: Everything about your Pets
paths:
  /pet:
    post:
      tags:
        - pet
      summary: Add a new pet to the store
      description: ''
      operationId: addPet
      responses:
        '405':
          description: Invalid input
      security:
        - petstore_auth:
            - 'write:pets'
            - 'read:pets'
      requestBody:
        $ref: '#/components/requestBodies/Pet'
components:
  requestBodies:
    Pet:
      content:
        application/json:
          schema:
            $ref: '#/components/schemas/Pet'
        application/xml:
          schema:
            $ref: '#/components/schemas/Pet'
      description: Pet object that needs to be added to the store
      required: true
  schemas:
    Pet:
      type: object
      required:
        - name
        - photoUrls
      properties:
        id:
          type: integer
          format: int64
        category:
          $ref: '#/components/schemas/Category'
        name:
          type: string
          example: doggie
        photoUrls:
          type: array
          items:
            type: string
        tags:
          type: array
          items:
            $ref: '#/components/schemas/Tag'
        status:
          type: string
          description: pet status in the store
          enum:
            - available
            - pending
            - sold
```

MicroGen will generate the following:

#### Server API Contract (JAX-RS)

JAX-RS interface with all the annotations necessary annotations

```java
@javax.annotation.Generated(value = "org.openapitools.codegen.languages.MicroGen")
public interface PetApi {
  /**
   * Add a new pet to the store
   *
   * @param pet Pet object that needs to be added to the store (required)
   * @return {@code java.util.concurrent.CompletionStage<javax.ws.rs.core.Response>}
   */
  @javax.ws.rs.POST
  @javax.ws.rs.Path("pet")
  @javax.ws.rs.Consumes({"application/json", "application/xml"})
  java.util.concurrent.CompletionStage<javax.ws.rs.core.Response> addPet(
          @javax.validation.constraints.NotNull @javax.validation.Valid Pet pet
  ) throws javax.ws.rs.WebApplicationException;
  .
  .
  .
}
```

### `@BeanParams` for request parameters

`@BeanParams` wrapper class to minimize the breaking changes in the Java API contract, this includes:
  - `@PathParam`'s
  - `@QueryParam`'s
  - `@FormParam`'s
  - `@HeaderParam`'s
  - `@CookieParam`'s

But also the additional:

  - `@Context UriInfo`
  - `@Context HttpHeaders`

```java
public class AddPetParams {
    @javax.ws.rs.core.Context
    public javax.ws.rs.core.HttpHeaders coreHttpHeaders;

    public AddPetParams coreHttpHeaders(
            final javax.ws.rs.core.HttpHeaders coreHttpHeaders
    ) {
        this.coreHttpHeaders = coreHttpHeaders;
        return this;
    }

    @javax.ws.rs.core.Context
    public javax.ws.rs.core.UriInfo coreUriInfo;

    public AddPetParams coreUriInfo(
            final javax.ws.rs.core.UriInfo coreUriInfo
    ) {
        this.coreUriInfo = coreUriInfo;
        return this;
    }
}
```

### Request and Response objects:

Generate the Request and Response java objects with the `Json-b` annotations:
  - `@JsonbProperty`
  - `@JsonbCreator`
  - `@JsonbPropertyOrder`
  - `toString()`
  - `equals`
  - `hashCode`
  - Inner static `Builder` class and utility methods

  COMING SOON: Using immutables.org library

  ```java
  /**
   * Pet
   */
  @javax.json.bind.annotation.JsonbPropertyOrder({
    Pet.JSON_PROPERTY_ID,
    Pet.JSON_PROPERTY_CATEGORY,
    Pet.JSON_PROPERTY_NAME,
    Pet.JSON_PROPERTY_PHOTO_URLS,
    Pet.JSON_PROPERTY_TAGS,
    Pet.JSON_PROPERTY_STATUS
  })
  @javax.annotation.Generated(value = "org.openapitools.codegen.languages.MicroGen")
  @io.quarkus.runtime.annotations.RegisterForReflection
  public final class Pet {
      private final java.util.OptionalLong id;
      public static final String JSON_PROPERTY_ID = "id";

      @javax.validation.Valid
      private final java.util.Optional<Category> category;
      public static final String JSON_PROPERTY_CATEGORY = "category";

      @javax.validation.constraints.NotNull
      private final String name;
      public static final String JSON_PROPERTY_NAME = "name";

      @javax.validation.constraints.NotNull
      private final java.util.List<String> photoUrls;
      public static final String JSON_PROPERTY_PHOTO_URLS = "photoUrls";

      @javax.validation.Valid
      private final java.util.Optional<java.util.List<Tag>> tags;
      public static final String JSON_PROPERTY_TAGS = "tags";

      private final StatusEnum status;
      public static final String JSON_PROPERTY_STATUS = "status";

      @javax.json.bind.annotation.JsonbCreator
      public Pet(
          @javax.json.bind.annotation.JsonbProperty("id")
          final java.util.OptionalLong id,
          @javax.json.bind.annotation.JsonbProperty("category")
          final java.util.Optional<Category> category,
          @javax.json.bind.annotation.JsonbProperty("name")
          final String name,
          @javax.json.bind.annotation.JsonbProperty("photoUrls")
          final java.util.List<String> photoUrls,
          @javax.json.bind.annotation.JsonbProperty("tags")
          final java.util.Optional<java.util.List<Tag>> tags,
          @javax.json.bind.annotation.JsonbProperty("status")
          final StatusEnum status
    ) {
        this.id = id;
        this.category = category;
        this.name = name;
        this.photoUrls = photoUrls;
        this.tags = tags;
        this.status = status;
    }

    // getter omitted
    // more details explained below
  }
  ```

#### Inner static factory builder to make easier the instantiation:

  ```java
  public static Builder builder() {
      return Builder.create();
  }

  public static final class Builder {
      private java.util.OptionalLong                  id;
      private java.util.Optional<Category>            category;
      private String                                  name;
      private java.util.List<String>                  photoUrls;
      private java.util.Optional<java.util.List<Tag>> tags;
      private StatusEnum                              status;

      private Builder() {
      }

      public static Builder create() {
          return new Builder();
      }

      public Builder setId(final java.util.OptionalLong id) {
          this.id = id;
          return this;
      }
      // more info omitted
    }
  ```

### Enum values

Generating all the enum values with the `@JsonbSerializer`'s and `@JsonbDeserializer`'s

  ```java
  @javax.json.bind.annotation.JsonbTypeSerializer(Pet.StatusEnumSerializer.class)
  @javax.json.bind.annotation.JsonbTypeDeserializer(Pet.StatusEnumDeserializer.class)
  public static enum StatusEnum {

      AVAILABLE("available"),
      PENDING("pending"),
      SOLD("sold");

      private final String value;

      StatusEnum(final String value) {
          this.value = value;
      }

      public String getValue() {
          return this.value;
      }

      @Override
      public String toString() {
          return String.valueOf(this.value);
      }

      @javax.json.bind.annotation.JsonbCreator
      public static StatusEnum fromValue(final String text) {
          return java.util.Arrays.stream(StatusEnum.values())
                                 .filter(b -> java.util.Objects.equals(String.valueOf(b.value), text))
                                 .findFirst()
                                 .orElse(null);
      }
  }
  ```

#### Enum's `Jsonb` Serializers/Deserializer

  ```java
  public static class StatusEnumSerializer implements javax.json.bind.serializer.JsonbSerializer<StatusEnum> {

      public StatusEnumSerializer() { }

      @Override
      public void serialize(
              StatusEnum aEnum,
              javax.json.stream.JsonGenerator jsonGenerator,
              javax.json.bind.serializer.SerializationContext serializationContext) {
          jsonGenerator.write(aEnum.value);
      }
  }

  public static class StatusEnumDeserializer implements javax.json.bind.serializer.JsonbDeserializer<StatusEnum> {

      public StatusEnumDeserializer() {}

      @Override
      public StatusEnum deserialize(
              javax.json.stream.JsonParser jsonParser,
              javax.json.bind.serializer.DeserializationContext deserializationContext,
              java.lang.reflect.Type type) {
          return StatusEnum.fromValue(jsonParser.getString());
      }
  }
  ```

### Using MicroGen with the Maven Archetype

Execute this command

```bash
$ mvn archetype:generate \
    -DgroupId=com.example \
    -DartifactId=example \
    -DarchetypeGroupId=io.microgen \
    -DarchetypeArtifactId=microgen-archetype
```

Check what is created

```bash
$ cd example && tree . --dirsfirst
example
├── src
│   └── main
│       ├── java
│       │   └── com
│       │       └── example
│       │           └── JAXRSConfiguration.java
│       ├── resources
│       │   └── META-INF
│       │       ├── microprofile-config.properties
│       │       └── openapi.yml
│       └── webapp
│           └── WEB-INF
│               └── beans.xml
├── Dockerfile
├── README.md
├── mvnw
├── mvnw.cmd
└── pom.xml

9 directories, 9 files
```

Implement the JAX-RS API contract

```java
@Path("v1")
@ApplicationScoped
public class PetResource implements PetApi {

    @Override
    public CompletionStage<Response> addPet(Pet pet) throws WebApplicationException {
        return CompletableFuture.completedFuture(
                Response.noContent().build()
        );
    }

    @Override
    public CompletionStage<Response> getPetById(GetPetByIdParams params) throws WebApplicationException {
        return CompletableFuture.completedFuture(
                Response.ok(
                        Pet.builder()
                           .setId(OptionalLong.of(1))
                           .setName("pet")
                           .build()
                ).build()
        );
    }
```

Run and enjoy

```bash
$ ./mvnw clean install
```

```bash
$ docker rm -f example || true && docker run -d -p 9080:9080 --name example com.example/example:1.0-SNAPSHOT
```

```bash
$ curl -s -X GET -H "Accept: application/json" http://localhost:9080/api/v1/pet/1 | jq .
{
  "category": {
    "id": 1,
    "name": ""
  },
  "id": 1,
  "name": "name",
  "status": "available"
}
```

### Advance Features

### Dependencies

| Library              | Version | License |
|:---------------------|:--------|:--------|
| Eclipse JakartaEE    | 8.0.0   |         |
| Eclipse MicroProfile | 3.0     |         |
| OpenAPI Tools        | 4.1.1   |         |
| Immutables           | 2.7.5   |         |

[![forthebadge](https://forthebadge.com/images/badges/made-with-java.svg)](https://forthebadge.com) [![forthebadge](https://forthebadge.com/images/badges/built-with-love.svg)](https://forthebadge.com)

[![forthebadge](https://forthebadge.com/images/badges/approved-by-george-costanza.svg)](https://forthebadge.com) [![forthebadge](https://forthebadge.com/images/badges/contains-technical-debt.svg)](https://forthebadge.com)

![CodeQL](https://github.com/cchacin/MicroGen/workflows/Code%20scanning%20-%20action/badge.svg)
![Java CI](https://github.com/cchacin/MicroGen/workflows/Java%20CI/badge.svg)
![Greetings](https://github.com/cchacin/MicroGen/workflows/Greetings/badge.svg)
![Release Drafter](https://github.com/cchacin/MicroGen/workflows/Release%20Management/badge.svg)
[![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.microgen/microgen-parent/badge.svg)](https://maven-badges.herokuapp.com/maven-central/io.microgen/MicroGen)
[![PRs Welcome](https://img.shields.io/badge/PRs-welcome-brightgreen.svg?style=square)](http://makeapullrequest.com) [![Dependabot Status](https://api.dependabot.com/badges/status?host=github&repo=cchacin/MicroGen)](https://dependabot.com)
 readmeEtag: '"94f3ac84cace64ec43634908a17e33d1c951fae7"' readmeLastModified: Sun, 02 May 2021 03:12:19 GMT repositoryId: 144823441 description: An OpenAPI Spec Generator for Eclipse JakartaEE + Eclipse MicroProfile created: '2018-08-15T07:56:22Z' updated: '2025-05-31T01:42:06Z' language: Java archived: false stars: 5 watchers: 1 forks: 6 owner: cchacin logo: https://avatars.githubusercontent.com/u/292938?v=4 license: Apache-2.0 repoEtag: '"ebe4e943b3b3ebf3f4144892c4063bda2cf2a77d88cd95fc078049bf34b038d9"' repoLastModified: Sat, 31 May 2025 01:42:06 GMT foundInMaster: true category: Parsers id: ef50340f0e36fe66a7057fb768ecd29e - source: openapi3 tags repository: https://github.com/smallstep/smallstep-python v3: true id: d1bf3e75463bc83b0728b0f6a92967ca repositoryMetadata: base64Readme: >- IyBzbWFsbHN0ZXAtcHl0aG9uCgpzbWFsbHN0ZXAtcHl0aG9uIGlzIGEgUHl0aG9uIGxpYnJhcnkgYXQgYWxsb3dzIHlvdSB0byBpbnRlcmZhY2Ugd2l0aCB0aGUgU21hbGxzdGVwIEFQSS4KCldlIHVzZSBbb3BlbmFwaS1weXRob24tY2xpZW50XShodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1nZW5lcmF0b3JzL29wZW5hcGktcHl0aG9uLWNsaWVudCkgdG8gZ2VuZXJhdGUgYSBsb3cgbGV2ZWwgUHl0aG9uIGNsaWVudCB0aGF0IGlzIGxvY2F0ZWQgaW4gYHNtYWxsc3RlcC9hcGlfY2xpZW50L2AuIFNlZSB0aGUgYHNtYWxsc3RlcC9SRUFETUUubWRgIGZvciBtb3JlIGluZm9ybWF0aW9uIG9uIHVzaW5nIHRoaXMgZ2VuZXJhdGVkIGxpYnJhcnkuIFdlIGJ1aWx0IGEgd3JhcHBlciBsaWJyYXJ5IG9uIHRvcCBvZiBgc21hbGxzdGVwL2FwaV9jbGllbnRgIHRoYXQgaXMgbG9jYXRlZCBpbiBgYXBpLnB5YC4gVGhpcyBhZGRzIGEgaGFuZGZ1bCBvZiBoZWxwZXIgZmVhdHVyZXMuIEF0IHRoaXMgdGltZSBgc21hbGxzdGVwL2FwaS5weWAgb25seSBzdXBwb3J0cyBhIGZldyBBUEkgZW5kcG9pbnRzLiBTZWUgYHNtYWxsc3RlcC9hcGkucHlgIGZvciBkZXRhaWxzLgoKIyMgUmVxdWlyZW1lbnRzCgoqIFB5dGhvbiAzLjkKKiBbU21hbGxzdGVwIEFjY291bnRdKGh0dHBzOi8vc21hbGxzdGVwLmNvbS9zaWdudXApCiogQSB2ZXJ5IHN0cm9uZyBkZXNpcmUgdG8gYWxsb3cgYSBjdXRlIHNuYWtlIHRvIG1hbmFnZSB5b3VyIFNtYWxsc3RlcCBhY2NvdW50Cgo8aW1nIHNyYz0vaW1hZ2VzL3NtYWxsc3RlcC1zbmVrLnBuZyBhbHQ9IkEgY3V0ZSBzbmFrZSB0byBtYW5hZ2UgU21hbGxzdGVwIiB3aWR0aD0yNTA+CgojIyBJbnN0YWxsYXRpb24KCldlIHB1Ymxpc2ggdGhpcyBwYWNrYWdlIHRvIDxodHRwczovL3B5cGkub3JnL3Byb2plY3Qvc21hbGxzdGVwLXB5dGhvbi8+IGFuZCBpdCBjYW4gYmUgaW5zdGFsbGVkIHZpYSBwaXAuCgpgYGBiYXNoCnBpcCBpbnN0YWxsIHNtYWxsc3RlcC1weXRob24KYGBgCgojIyBEZXZlbG9wbWVudAoKIyMjIENvbmZpZ3VyYXRpb24KCkNyZWF0ZSBhIGAuZW52YCBmaWxlIGluIHRoZSBwcm9qZWN0IHJvb3QgYW5kIGFkZCB0aGUgZm9sbG93aW5nIGxpbmVzOgoKYGBgYmFzaAojIE5vdCBuZWVkZWQgdW5sZXNzIHlvdSBhcmUgdXNpbmcgb3VyIHJ1biBhbnl3aGVyZSBvZmZlcmluZwojIFNNQUxMU1RFUF9BUElfSE9TVD0iaHR0cHM6Ly9nYXRld2F5LnNtYWxsc3RlcC5jb20vYXBpIgpTTUFMTFNURVBfQVBJX1RPS0VOPSJ5b3VyX3NtYWxsc3RlcF9hcGlfdG9rZW4iCmBgYAoKQWRqdXN0IHRoZW0gdG8geW91ciBuZWVkcy4gU2VlIGBjb25maWcucHlgIGZvciBjb25maWd1cmF0aW9uIGRldGFpbHMuIFdlIHVzZSBbUHlkYW50aWMgU2V0dGluZ3NdKGh0dHBzOi8vZG9jcy5weWRhbnRpYy5kZXYvbGF0ZXN0L2NvbmNlcHRzL3B5ZGFudGljX3NldHRpbmdzLykgZm9yIGVhc3kgc2V0dGluZ3MgbWFuYWdlbWVudC4gQmVsb3cgaXMgdGhlIG9yZGVyIG9mIHByZWNlZGVuY2UgZm9yIGNvbmZpZ3VyYXRpb24gc2V0dGluZ3M6CgoxLiBBcmd1bWVudHMgcGFzc2VkIHRvIHRoZSBTZXR0aW5ncyBjbGFzcyBpbml0aWFsaXNlci4KMS4gRW52aXJvbm1lbnQgdmFyaWFibGVzLCBlLmcuIFNNQUxMU1RFUF9NWV9DT05GSUdfU0VUVElORyBhcyBkZXNjcmliZWQgYWJvdmUuCjEuIFZhcmlhYmxlcyBsb2FkZWQgZnJvbSBhIGRvdGVudiAoLmVudikgZmlsZS4KMS4gVmFyaWFibGVzIGxvYWRlZCBmcm9tIHRoZSBzZWNyZXRzIGRpcmVjdG9yeS4KMS4gVGhlIGRlZmF1bHQgZmllbGQgdmFsdWVzIGZvciB0aGUgU2V0dGluZ3MgbW9kZWwuCgpTZWUgdGhlIFB5ZGFudGljIFNldHRpbmdzIFtmaWVsZCB2YWx1ZSBwcmlvcml0eV0oaHR0cHM6Ly9kb2NzLnB5ZGFudGljLmRldi9sYXRlc3QvY29uY2VwdHMvcHlkYW50aWNfc2V0dGluZ3MvI2ZpZWxkLXZhbHVlLXByaW9yaXR5KSBzZWN0aW9uIGZvciBtb3JlIGluZm9ybWF0aW9uLgoKIyMgV2l0aCBQb2V0cnkKCkluc3RhbGwgUG9ldHJ5IG9uIHlvdXIgc3lzdGVtIHdpdGggW3RoaXNdKGh0dHBzOi8vcHl0aG9uLXBvZXRyeS5vcmcvZG9jcy8jaW5zdGFsbGF0aW9uKS4KClJ1biB0aGlzIGZyb20gdGhlIHJlcG8gZGlyZWN0b3J5OgoKYGBgYmFzaApwb2V0cnkgaW5zdGFsbApgYGAKCkVudGVyIHRoZSBQb2V0cnkgc2hlbGwgd2l0aCB0aGlzOgoKYGBgYmFzaApwb2V0cnkgc2hlbGwKcHJlLWNvbW1pdCBpbnN0YWxsCmBgYAoKIyMjIFdpdGggdmVudgoKYGBgYmFzaApleHBvcnQgVklSVFVBTF9FTlY9JHtQV0R9Ly52ZW52CnB5dGhvbjMgLW0gdmVudiAkVklSVFVBTF9FTlYKZXhwb3J0IFBBVEg9IiRWSVJUVUFMX0VOVi9iaW46JFBBVEgiCnBpcCBpbnN0YWxsIHdoZWVsCnBpcCBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQKLiAudmVudi9iaW4vYWN0aXZhdGUKcHJlLWNvbW1pdCBpbnN0YWxsCmBgYAoKIyMjIEFkZGluZyBwYWNrYWdlcwoKQWRkIHBhY2thZ2VzIHRvIFBvZXRyeSBhbmQgdGhlbiBydW4gcHJlLWNvbW1pdCB3aGljaCB3aWxsIGdlbmVyYXRlIGEgcmVxdWlyZW1lbnRzLnR4dCBmaWxlIGZvciB5b3UuCgpEZXZlbG9wbWVudDoKCmBgYGJhc2gKcG9ldHJ5IGFkZCAtLWdyb3VwPWRldiByaWNoCnByZS1jb21taXQKYGBgCgpQcm9kdWN0aW9uOgoKYGBgYmFzaApwb2V0cnkgYWRkIHJpY2gKcHJlLWNvbW1pdApgYGAKCiMjIExpY2Vuc2UKCltBcGFjaGUgTGljZW5zZSBWZXJzaW9uIDIuMF0oaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wKQoKQ29weXJpZ2h0IDIwMjMgU21hbGxzdGVwIExhYnMgSW5jLgo= readmeEtag: '"7b500e4c94187ccba1cbc2ea3bd3571ca40a3111"' readmeLastModified: Wed, 14 Aug 2024 18:10:28 GMT repositoryId: 703177207 description: 'A Python client library for the Smallstep API ' created: '2023-10-10T18:33:22Z' updated: '2026-01-23T22:30:31Z' language: Python archived: false stars: 7 watchers: 6 forks: 2 owner: smallstep logo: https://avatars.githubusercontent.com/u/23183426?v=4 license: Apache-2.0 repoEtag: '"90a930379f16db95e2580adce318a55ac96609b886f12f9cefcdf270337b4ed9"' repoLastModified: Fri, 23 Jan 2026 22:30:31 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/danielgtaylor/sdt v3: true repositoryMetadata: base64Readme: >- # Structured Data Templates

[![Go Reference](https://pkg.go.dev/badge/github.com/danielgtaylor/sdt.svg)](https://pkg.go.dev/github.com/danielgtaylor/sdt) [![Go Report Card](https://goreportcard.com/badge/github.com/danielgtaylor/sdt)](https://goreportcard.com/report/github.com/danielgtaylor/sdt) ![Build Status](https://github.com/danielgtaylor/sdt/actions/workflows/test.yaml/badge.svg?branch=main) [![codecov](https://codecov.io/gh/danielgtaylor/sdt/branch/main/graph/badge.svg?token=KB0QD0H6HP)](https://codecov.io/gh/danielgtaylor/sdt) [![VS Code Extension](https://img.shields.io/badge/vscode-extension-blue)](https://marketplace.visualstudio.com/items?itemName=danielgtaylor.structured-data-templates)

Structured data templates are a templating engine that takes a simplified set of input parameters and transforms them into a complex structured data output. Both the inputs and outputs can be validated against a schema.

Be sure to check out the [Visual Studio Code extension](https://marketplace.visualstudio.com/items?itemName=danielgtaylor.structured-data-templates) for syntax highlighting and schema validation as you type.

The goals of this project are to:

1. Provide a simple format: it's just JSON/YAML!
2. Give enough tools to be useful:
   - Interpolation `${my_value}` & `${num / 2 >= 5}`
   - Branching (if/then/else)
   - Looping (for/each)
3. Guarantee structural correctness
   - The structured data template is valid JSON / YAML
   - The input parameters are valid JSON / YAML
   - The output of the template is guaranteed to produce valid JSON
4. Provide tools for semantic correctness via schemas
   - The input types and values pass the schema
   - The template will produce output that should pass the schema
   - The output of the template after rendering passes the schema

## Structure

A structured data template document is made of two parts: schemas and a template. The schemas define the allowable input/output structure while the template defines the actual rendered output. An example document might look like:

```yaml
schemas:
  # Dialect selects the default JSON Schema version
  dialect: openapi-3.1
  input:
    # Input schema goes here
    type: object
    properties:
      name:
        type: string
        default: world
  output:
    # Output schema goes here, also supports refs:
    $ref: https://api.example.com/openapi.json#components/schemas/Greeting
template:
  # Templated output structure goes here
  greeting: Hello, ${name}!
```

## Installation

You can install via:

```sh
# Get the `sdt` command
$ go get -u github.com/danielgtaylor/sdt/cmd/...

# Install as a library
$ go get -u github.com/danielgtaylor/sdt
```

## Example

You can run the example like so:

```sh
# Validate the template
$ sdt validate ./samples/greeting.yaml

# Generate an example params file
$ sdt example -f yaml ./samples/hello/hello.yaml >params.yaml

# Render by passing in a file
$ sdt render ./samples/hello/hello.yaml <params.yaml
{
  "greeting": "Hello, SDT!"
}

# Render by using CLI shorthand syntax
$ sdt render ./samples/greeting.yaml name: Alice
{
  "greeting": "Hello, Alice!"
}
```

Input params for rendering can be passed via stdin as JSON/YAML and/or via command line arguments as [CLI shorthand syntax](https://github.com/danielgtaylor/shorthand#readme).

## Schemas

JSON Schema is used for all schemas. It defaults to JSON Schema 2020-12 but can be overridden via the `$schema` key or using `dialect` in the structured data template document like above. Available dialects:

- `openapi-3.0`
- `openapi-3.1`
- `https://json-schema.org/draft/2020-12/schema`
- `https://json-schema.org/draft/2019-09/schema`
- `https://json-schema.org/draft-07/schema`
- `https://json-schema.org/draft-06/schema`
- `https://json-schema.org/draft-04/schema`

The input schema describes the input parameters and the template will not render unless the passed parameters validate using the input schema. It also lets you set defaults for the input parameters, which default to `nil` if not passed.

The output schema describes the template's output structure. The validator is capable of understanding branches & loops to ensure that the output is semantically valid regardless of which path is taken during rendering.

## Template Language Specification

A template is just JSON/YAML. For example:

```yaml
hello: world
```

That is a valid static template. Nothing will change when rendered, which is not very useful. Normally, when a template is rendered, it is passed parameters, and these are used for interpolation, branching, and looping, which are specified using special syntax in strings or keywords as object property names:

- Interpolation: `${...}`
- Branching: `$if`, `$then`, `$else`
- Looping: `$for`, `$as`, `$each`
- Special operations: `$flatten`

These features make use of a basic expression language.

### Expressions

String interpolation, branching conditions, and loop variable selection all use an expression language. This allows you to make simple comparisons of the parameter context data. Examples:

- `foo > 50`
- `item.bars.length <= 5 or my_override`
- `"sdt" in name`
- `name startsWith "sdt"`
- `"foo" in my_array`
- `loop.index + 1`

See [danielgtaylor/mexpr syntax](https://github.com/danielgtaylor/mexpr#syntax) for details.

### String Interpolation

String interpolation is the act of replacing the contents of `${...}` within strings, where `...` corresponds to an expression that makes use of input parameters. For example:

```yaml
hello: ${name}
```

If passed `{"name": "Alice"}` as parameters this would render:

```json
{
  "hello": "Alice"
}
```

Whenever the string is just one `${...}` statement it will use whatever type it evaluates to in the result, so you are not limited to just strings. If the expression result is `nil`, then the property/item is not included in the rendered output.

It's also possible to add static text or multiple interpolation expressions in a single value:

```yaml
hello: Greetings, ${name}!
```

Given the same input that would result in:

```json
{
  "hello": "Greetings, Alice!"
}
```

#### Tricks

- Force a string output by using more than one expression: `${my_number}${""}`

### Branching

Branching allows one of multiple paths to be followed in the template at rendering time based on the result of an expression. The special properties `$if`, `$then`, and `$else` are used for this. For example:

```yaml
foo:
  $if: ${value > 5}
  $then: I am big
  $else: I am small
```

If rendered with `{"value": 1}` the result will be:

```json
{
  "foo": "I am small"
}
```

Notice that the special properties are completely removed and replaced with the contents of either the `$then` or `$else` clauses. So while in the _template_ `foo` is an object, the end result is that `foo` is a string and would pass the output schema.

If the expression is false and no `$then` is given, then the property is removed from the result.

### Looping

Looping allows an array of inputs to be expanded into the rendered output using a per-item template. The `$for`, `$as`, and `$each` special properties are used for this. For example:

```yaml
squares:
  $for: ${numbers}
  $each: ${item * item}
```

If rendered with `{"numbers": [1, 2, 3]}` the result will be:

```json
{
  "squares": [2, 4, 9]
}
```

The `$as` property controls the name of the variable holding the current item, which defaults to `item`. A local variable `loop` is also set, which includes an `index`, and whether the item is the `first` or `last` in the array. If using `$as` then the `loop` variable is named `loop_` + the `$as` value. This allows nested loops to access both their own and outer scope's loop variables. For example:

```yaml
things:
  $for: ${things}
  $as: thing
  $each:
    id: ${loop_thing.index}-${thing.name}
    tags:
      $for: ${tags}
      $as: tag
      $each: ${loop_thing.index}-${loop_tag.index}-${tag}
```

Given:

```json
{
  "things": [{ "name": "Alice" }, { "name": "Bob" }],
  "tags": ["big", "small"]
}
```

You would get as output:

```json
{
  "things": [
    {
      "id": "0-Alice",
      "tags": ["0-0-big", "0-1-small"]
    },
    {
      "id": "1-Bob",
      "tags": ["1-0-big", "1-1-small"]
    }
  ]
}
```

### Flatten

The `$flatten` special operator takes an array of arrays and flattens them one level into a single array. This can be useful for a number of scenarios like:

- Adding default items to a `$for` loop output
- Having one item of a `$for` clause generate multiple outputs

For a simple example:

```yaml
my_array:
  $flatten:
    - [0, 1, 2]
    - [3, 4, 5]
    - [6, 7, 8]
```

This would result in:

```json
{
  "my_array": [0, 1, 2, 3, 4, 5, 6, 7, 8]
}
```

More complex scenarios are possible when combined with `$for` clauses:

```yaml
# Loop through the items twice, generating one item at a time.
appended_array:
  $flatten:
    - $for: ${items}
      $each: ${item}
    - $for: ${items}
      $each: ${item * item}
# Loop through the items once, generating a list for each item.
merged_array:
  $flatten:
    $for: ${items}
    $each:
      - ${item}
      - ${item * item}
```

If given:

```json
{
  "items": [2, 3, 4]
}
```

You would get:

```json
{
  "appended_array": [2, 3, 4, 4, 9, 16],
  "merged_array": [2, 4, 3, 9, 4, 16]
}
```

## Open Questions

1. Should we support macros? Could be done with `$ref` in the template, and we could add a top-level `macros` or `definitions` for document-local refs. They would be drop-in only, no calling with arguments, but would render based on the current params context.

2. Should `nil` results from interpolation be rendered in the final output? Example: `name: ${name}` and what if `name` is `nil`?

3. Support for constants? Values that should always be present in the params that can contain complex and reusable data for the template?

4. Ability to sort `$for` loop output based on some expr?
 readmeEtag: '"8ffd0423b33657c111a60b76c8896c991707fbbd"' readmeLastModified: Thu, 16 Dec 2021 17:31:33 GMT repositoryId: 418250995 description: Structured Data Templates created: '2021-10-17T20:42:58Z' updated: '2025-02-13T08:07:24Z' language: Go archived: false stars: 6 watchers: 1 forks: 2 owner: danielgtaylor logo: https://avatars.githubusercontent.com/u/106826?v=4 license: MIT repoEtag: '"6b74ac3fcdbb0aadc76a8af83c3471b007bcb2b2ea51fc5a5bc9b5eaa4db0947"' repoLastModified: Thu, 13 Feb 2025 08:07:24 GMT foundInMaster: true category: Parsers id: 47cf5346c9ca65f18c438755c3812912 - source: openapi3 tags repository: https://github.com/skaffolder/skaffolder-vscode-extension v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+PGEgaHJlZj0iaHR0cHM6Ly9za2FmZm9sZGVyLmNvbSI+PGltZyBzcmM9Imh0dHBzOi8vc2thZmZvbGRlci5jb20vaW1nL2xvZ28vc2thZmZvbGRlcl9sb2dvLXdoaXRlX2JnLnBuZyIgd2lkdGg9IjcwJSI+PC9hPjwvcD4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hY3Rpb25zIj4KCSAgPGltZyBhbHQ9IkdpdEh1YiBXb3JrZmxvdyBTdGF0dXMiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvd29ya2Zsb3cvc3RhdHVzL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL21haW4iPgogIDwvYT4KICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vc2thZmZvbGRlci9za2FmZm9sZGVyLXZzY29kZS1leHRlbnNpb24vYmxvYi9tYXN0ZXIvTElDRU5TRSI+CiAgICA8aW1nIGFsdD0iR2l0SHViIExpY2Vuc2UiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbiI+CiAgPC9hPgo8L3A+CgohW0V4dGVuc2lvbiBHaWZdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hc3NldHMvZ2lmL3ZzY29kZV9zYW1wbGUuZ2lmKQoKIyBTa2FmZm9sZGVyIEdlbmVyYXRvcgoKVGhpcyBleHRlbnNpb24gYWxsb3dzIHRvIGNyZWF0ZSwgZWRpdCBhbmQgbWFuYWdlIGEgW1NrYWZmb2xkZXJdKGh0dHBzOi8vd3d3LnNrYWZmb2xkZXIuY29tKSBwcm9qZWN0LgoKSXQgYWxsb3dzIHRvIGNyZWF0ZSB3ZWIgYW5kIG1vYmlsZSBhcHBsaWNhdGlvbnMgc3RhcnRpbmcgZnJvbSB0aGUgdGVjaG5pY2FsIGRvY3VtZW50YXRpb24gb2YgZGF0YWJhc2UgbW9kZWxzLCBBUElzIGFuZCBwYWdlcy4KClRoZSBkb2N1bWVudGF0aW9uIGNhbiBiZSBkZWZpbmVkIGZyb20gdGhlIFtTa2FmZm9sZGVyIHdlYiBpbnRlcmZhY2VdKGh0dHBzOi8vYXBwLnNrYWZmb2xkZXIuY29tKSBoYXZpbmcgYSBmcmVlIFNrYWZmb2xkZXIgYWNjb3VudCBvciBmcm9tIHRoZSBvcGVuYXBpLnlhbWwgZmlsZSB0aGF0IGV4dGVuZHMgdGhlIE9wZW5BUEkgMy4wIHN0YW5kYXJkcy4gVGhpcyBleGVuc2lvbiBhbGxvd3MgdG8gZWRpdCB0aGUgb3BlbmFwaSBmaWxlIGZyb20gYSB2aXN1YWwgaW50ZXJmYWNlIGFjY29yZGluZyB0byB0aGUgU2thZmZvbGRlcidzIHN0YW5kYXJkcy4KCiMjIEZlYXR1cmVzCgotIENyZWF0ZSBGZWF0dXJlcwogIC0gW0NyZWF0ZSBwcm9qZWN0XSgjY3JlYXRlLXByb2plY3QpCiAgLSBbQ3JlYXRlIG1vZGVsXSgjY3JlYXRlLW1vZGVsKQogIC0gW0NyZWF0ZSBhcGldKCNjcmVhdGUtYXBpKQogIC0gW0NyZWF0ZSBwYWdlXSgjY3JlYXRlLXBhZ2UpCiAgLSBbQ3JlYXRlIGNydWRdKCNjcmVhdGUtY3J1ZCkKLSBbT3BlbiByZWFsZWF0aXZlIGZpbGVzXSgjb3Blbi1maWxlcykKLSBbR2VuZXJhdGUgcHJvamVjdF0oI2dlbmVyYXRlLWNvZGUpCi0gW0VkaXRpbmcgY29tcG9uZXRuc10oI2VkaXRpbmctY29tcG9uZW50cykKLSBTa2FmZm9sZGVyIEFjY291bnQgZmVhdHVyZXMKICAtIFNrYWZmb2xkZXIgTG9naW4vTG9nb3V0CiAgLSBbRXhwb3J0IHByb2plY3QgdG8gU2thZmZvbGRlciBQbGF0Zm9ybV0oI2V4cG9ydC1wcm9qZWN0KQotIFtTZXR0aW5nIGZvciBvbi1wcmVtaXNlXSgjZXh0ZW5zaW9uLXNldHRpbmdzLWZvci1vbi1wcmVtaXNlKQoKLS0tCgojIyMgQ3JlYXRlIFByb2plY3QKCkNyZWF0ZSBhIG5ldyBsb2NhbCBTa2FmZm9sZGVyIHByb2plY3QgYnkgY2xpY2tpbmcgb24gX0NyZWF0ZSBwcm9qZWN0c18uIFRoZSBidXR0b24gd2lsbCBvbmx5IGFwcGVhciBpZiB0aGUgZXh0ZW5zaW9uIGNhbid0IGZpbmQgYW4gb3BlbmFwaS55YW1sIGZpbGUgaW4gdGhlIHJvb3Qgb2YgdGhlIGN1cnJlbnQgd29ya3NwYWNlLgoKIVtDcmVhdGUgcHJvamVjdCBHaWZdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hc3NldHMvZ2lmL2NyZWF0ZV9wcm9qZWN0LmdpZikKCiMjIyBHZW5lcmF0ZSBjb2RlCgpHZW5lcmF0ZSB0aGUgc291cmNlIGNvZGUgb2YgeW91ciBwcm9qZWN0IGZyb20gdGhlIG9wZW5hcGkueWFtbCBmaWxlLgoKIVtHZW5lcmF0ZSBjb2RlIEdpZl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL2Fzc2V0cy9naWYvZ2VuZXJhdGVfY29kZS5naWYpCgojIyMgRXhwb3J0IHByb2plY3QKCkV4cG9ydCB5b3VyIGxvY2FsIHByb2plY3QgdG8gU2thZmZvbGRlciBQbGF0Zm9ybS4KQmVmb3JlIGV4cG9ydGluZyB5b3VyIHByb2plY3QgeW91IG5lZWQgdG8gbG9naW4gd2l0aCB5b3VyIFNrYWZmb2xkZXIgW2FjY291bnRdKCNyZXF1aXJlbWVudHMpLgoKIVtFeHBvcnQgcHJvamVjdCBHaWZdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hc3NldHMvZ2lmL2V4cG9ydF9wcm9qZWN0LmdpZikKCiMjIyBDcmVhdGUgbW9kZWwKCkNyZWF0ZSBhIG5ldyBtb2RlbCBpbiB0aGUgU2thZmZvbGRlciBwcm9qZWN0IGJ5IGNsaWNraW5nIG9uIHRoZSBzeW1ib2wgYCtgIG9mIGEgZGF0YWJhc2UgZnJvbSB0aGUgVHJlZVZpZXcgb2YgeW91ciBwcm9qZWN0OgoKIVtDcmVhdGUgbW9kZWwgZ2lmXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vc2thZmZvbGRlci9za2FmZm9sZGVyLXZzY29kZS1leHRlbnNpb24vYXNzZXRzL2dpZi9jcmVhdGVfbW9kZWwuZ2lmKQoKT3IgYnkgdHlwaW5nIGBzayBtb2RlbGAgaW4gdGhlIENvbW1hbmQgUGFsZXR0ZToKCiFbQ3JlYXRlIG1vZGVsIGdpZl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL2Fzc2V0cy9naWYvcGFsZXR0ZV9jcmVhdGVfbW9kZWwuZ2lmKQoKIyMjIENyZWF0ZSBhcGkKCkNyZWF0ZSBhIG5ldyBhcGkgaW50IHRoZSBTa2FmZm9sZGVyIHByb2plY3QgYnkgY2xpY2tpbmcgb24gdGhlIHN5bWJvbCBgK2Agb2YgYSBtb2RlbCBmcm9tIHRoZSBUcmVlVmlldyBvZiB5b3VyIHByb2plY3Q6CgohW0NyZWF0ZSBhcGkgZ2lmXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vc2thZmZvbGRlci9za2FmZm9sZGVyLXZzY29kZS1leHRlbnNpb24vYXNzZXRzL2dpZi9jcmVhdGVfYXBpLmdpZikKCk9yIGJ5IHR5cGluZyBgc2sgYXBpYCBpbiB0aGUgQ29tbWFuZCBQYWxldHRlOgoKIVtDcmVhdGUgYXBpIGdpZl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL2Fzc2V0cy9naWYvcGFsZXR0ZV9jcmVhdGVfYXBpLmdpZikKCiMjIyBDcmVhdGUgcGFnZQoKQ3JlYXRlIGEgbmV3IHBhZ2UgaW4gdGhlIFNrYWZmb2xkZXIgcHJvamVjdCBieSBjbGlja2luZyBvbiB0aGUgc3ltYm9sIGArYCBvbiB0aGUgX1BBR0VTXyBwYW5lbCBvZiB5b3VyIHByb2plY3Q6CgohW0NyZWF0ZSBwYWdlIGdpZl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL2Fzc2V0cy9naWYvY3JlYXRlX3BhZ2UuZ2lmKQoKT3IgYnkgdHlwaW5nIGBzayBwYWdlYCBpbiB0aGUgQ29tbWFuZCBQYWxldHRlOgoKIVtDcmVhdGUgcGFnZSBnaWZdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hc3NldHMvZ2lmL3BhbGV0dGVfY3JlYXRlX3BhZ2UuZ2lmKQoKIyMjIENyZWF0ZSBDUlVECgpDcmVhdGUgYSBDUlVEIGludGVyZmFjZSBmb3IgYSBtb2RlbCBieSBjbGlja2luZyBieSBfQ3JlYXRlIENSVURfIG9mIGEgdGhlIG1vZGVsIGVkaXRpbmcgcGFuZWw6CgohW0NyZWF0ZSBjcnVkIGdpZl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL2Fzc2V0cy9naWYvY3JlYXRlX2NydWQuZ2lmKQoKIyMjIE9wZW4gRmlsZXMKCk9wZW4gZmlsZXMgcmVsYXRlZCB0byBhIG1vZGVsLCBhcGkgb3IgYSBwYWdlIGJ5IGNsaWNraW5nIG9uIHRoZSBpY29uIG5leHQgdGhlIGVkaXQgaWNvbiBvciBieSBfT3BlbiByZWxhdGVkIGZpbGVzXyBpbiB0aGUgY29tcG9uZW50J3MgZWRpdGluZyBwYW5lbDoKCiFbT3BlbiBmaWxlcyBHaWZdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hc3NldHMvZ2lmL29wZW5fZmlsZXMuZ2lmKQoKIyMjIEVkaXRpbmcgY29tcG9uZW50cwoKRWRpdCBhIGNvbXBvbmVudCBieSBjbGlja2luZyB0aGUgZWRpdCBpY29uIGFuZCBhIG5ldyBlZGl0aW5nIHBhbmVsIHdpbGwgb3Blbi4gV2hlbiBzYXZpbmcsIGFueSBtb2RpZmljYXRpb24gd2lsbCBiZSB3cml0dGVuIGluIHRoZSBvcGVuYXBpLnlhbWwgZmlsZS4KCiFbQWRkIGF0dHJpYnV0ZSBmcm9tIHBhbmVsIGdpZl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3NrYWZmb2xkZXIvc2thZmZvbGRlci12c2NvZGUtZXh0ZW5zaW9uL2Fzc2V0cy9naWYvYWRkX2F0dHJpYnV0ZS5naWYpCgpPciB5b3UgY2FuIGVkaXQgZGlyZWN0bHkgZnJvbSB0aGUgb3BlbmFwaS55YW1sIGZpbGU6CgohW0VkaXQgYXR0cmlidXRlIGZyb20gZmlsZSBnaWZdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9za2FmZm9sZGVyL3NrYWZmb2xkZXItdnNjb2RlLWV4dGVuc2lvbi9hc3NldHMvZ2lmL2VkaXRfYXR0cmlidXRlX2ZpbGUuZ2lmKQoKIyMgUmVxdWlyZW1lbnRzCgpPcHRpb25hbCBmcmVlIFNrYWZmb2xkZXIgYWNjb3VudDoKaHR0cHM6Ly9za2FmZm9sZGVyLmNvbS9yZWdpc3RlcgoKLS0tCgojIyBFeHRlbnNpb24gU2V0dGluZ3MgZm9yIG9uLXByZW1pc2UKCi0gYHNrYWZmb2xkZXIuZW5kcG9pbnRgOiBjb25maWd1cmUgbGluayB3aXRoIG9uLXByZW1pc2UgU2thZmZvbGRlciBwbGF0Zm9ybQotIGBza2FmZm9sZGVyLmVuZHBvaW50RG9jc2A6IGNvbmZpZ3VyZSBsaW5rIHdpdGggb24tcHJlbWlzZSBTa2FmZm9sZGVyIHBsYXRmb3JtIGZvciBkaXNwbGF5aW5nIGRvY3VtZW50YXRpb24KCk1vcmUgaW5mbyBvbiBTa2FmZm9sZGVyIG9uLXByZW1pc2UgZm9yIGVudGVycHJpc2VzIFtoZXJlXShodHRwczovL3NrYWZmb2xkZXIuY29tL2VudGVycHJpc2Uvb3ZlcnZpZXcpLgoKR2V0IFNrYWZmb2xkZXIgb24tcHJlbWlzZSBmcm9tOgoKLSBbRG9ja2VyIEh1Yl0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9fL3NrYWZmb2xkZXItZW50ZXJwcmlzZSkKLSBbR29vZ2xlIE1hcmtldHBsYWNlXShodHRwczovL2NvbnNvbGUuY2xvdWQuZ29vZ2xlLmNvbS9tYXJrZXRwbGFjZS9kZXRhaWxzL3NrYWZmb2xkZXItcHVibGljL3NrYWZmb2xkZXItZW50ZXJwcmlzZSkKLSBbQVdTIE1hcmtldHBsYWNlXShodHRwczovL2F3cy5hbWF6b24uY29tL21hcmtldHBsYWNlL3BwL0IwN1NXNEdQRlkpCi0gW0RpZ2l0YWxPY2VhbiBNYXJrZXRwbGFjZV0oaHR0cHM6Ly9tYXJrZXRwbGFjZS5kaWdpdGFsb2NlYW4uY29tL2FwcHMvc2thZmZvbGRlci1lbnRlcnByaXNlKQoKLS0tCgojIyBDb250cmlidXRpbmcKClNrYWZmb2xkZXItZ2VuZXJhdG9yIGlzIGFuIG9wZW4tc291cmNlIHByb2plY3QuIEZlZWwgZnJlZSB0byBwcm9wb3NlIGVuaGFuY2VtZW50cyBzdWdnZXN0aW9ucywgcmVwb3J0IGJ1Z3MgYW5kIHRvIHN1Ym1pdCBwdWxsIHJlcXVlc3RzLgo= readmeEtag: '"d51ce444776820d34b678f0886b8179d40404f57"' readmeLastModified: Sat, 18 Jul 2020 18:17:07 GMT repositoryId: 237040227 description: >- This VSCode Extension allows to generate web app from documentation in your IDE. You can import APIs from a OpenAPI 3.0 file or define it by yourself. created: '2020-01-29T17:18:35Z' updated: '2024-03-26T08:58:48Z' language: TypeScript archived: false stars: 5 watchers: 2 forks: 6 owner: skaffolder logo: https://avatars.githubusercontent.com/u/35335745?v=4 license: GPL-3.0 repoEtag: '"b783319cfe55880d4d1deea00f3834b6240800759f268d190f7372b5c8ef299f"' repoLastModified: Tue, 26 Mar 2024 08:58:48 GMT foundInMaster: true category: SDK id: 4e921d6ee7f9e5bec7ded491eb1b25ba - source: openapi3 tags repository: https://github.com/correl/tornado-openapi3 v3: true repositoryMetadata: base64Readme: >- PT09PT09PT09PT09PT09PT09PQogVG9ybmFkbyBPcGVuQVBJIDMKPT09PT09PT09PT09PT09PT09PQoKLi4gaW1hZ2U6OiBodHRwczovL2dpdGh1Yi5jb20vY29ycmVsL3Rvcm5hZG8tb3BlbmFwaTMvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC55bWwvYmFkZ2Uuc3ZnP2JyYW5jaD1tYXN0ZXIKICAgIDp0YXJnZXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS9jb3JyZWwvdG9ybmFkby1vcGVuYXBpMy9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LnltbD9icmFuY2g9bWFzdGVyCi4uIGltYWdlOjogaHR0cHM6Ly9jb2RlY292LmlvL2doL2NvcnJlbC90b3JuYWRvLW9wZW5hcGkzL2JyYW5jaC9tYXN0ZXIvZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPUNUWVdXRFhUTDkKICAgIDp0YXJnZXQ6IGh0dHBzOi8vY29kZWNvdi5pby9naC9jb3JyZWwvdG9ybmFkby1vcGVuYXBpMwouLiBpbWFnZTo6IGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL3Rvcm5hZG8tb3BlbmFwaTMvYmFkZ2UvCiAgICA6dGFyZ2V0OiBodHRwczovL3Rvcm5hZG8tb3BlbmFwaTMucmVhZHRoZWRvY3MuaW8KLi4gaW1hZ2U6OiBodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NvZGUlMjBzdHlsZS1ibGFjay0wMDAwMDAuc3ZnCiAgICA6dGFyZ2V0OiBodHRwczovL2dpdGh1Yi5jb20vcHNmL2JsYWNrCgoKVG9ybmFkbyBPcGVuQVBJIDMgcmVxdWVzdCBhbmQgcmVzcG9uc2UgdmFsaWRhdGlvbiBsaWJyYXJ5LgoKUHJvdmlkZXMgaW50ZWdyYXRpb24gYmV0d2VlbiB0aGUgYFRvcm5hZG9gXyB3ZWIgZnJhbWV3b3JrIGFuZCBgT3BlbmFwaS1jb3JlYF8KbGlicmFyeSBmb3IgdmFsaWRhdGluZyByZXF1ZXN0IGFuZCByZXNwb25zZSBvYmplY3RzIGFnYWluc3QgYW4gYE9wZW5BUEkgM2BfCnNwZWNpZmljYXRpb24uCgpGdWxsIGRvY3VtZW50YXRpb24gaXMgYXZhaWxhYmxlIGF0IGh0dHBzOi8vdG9ybmFkby1vcGVuYXBpMy5yZWFkdGhlZG9jcy5pbwoKVXNhZ2UKPT09PT0KCkFkZGluZyB2YWxpZGF0aW9uIHRvIHJlcXVlc3QgaGFuZGxlcnMKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKLi4gY29kZTo6IHB5dGhvbgoKICAgaW1wb3J0IHRvcm5hZG8uaW9sb29wCiAgIGltcG9ydCB0b3JuYWRvLndlYgogICBmcm9tIHRvcm5hZG9fb3BlbmFwaTMuaGFuZGxlciBpbXBvcnQgT3BlbkFQSVJlcXVlc3RIYW5kbGVyCgoKICAgY2xhc3MgTXlSZXF1ZXN0SGFuZGxlcihPcGVuQVBJUmVxdWVzdEhhbmRsZXIpOgogICAgICAgc3BlY19kaWN0ID0gewogICAgICAgICAgICJvcGVuYXBpIjogIjMuMC4wIiwKICAgICAgICAgICAiaW5mbyI6IHsKICAgICAgICAgICAgICAgInRpdGxlIjogIlNpbXBsZSBFeGFtcGxlIiwKICAgICAgICAgICAgICAgInZlcnNpb24iOiAiMS4wLjAiLAogICAgICAgICAgIH0sCiAgICAgICAgICAgInBhdGhzIjogewogICAgICAgICAgICAgICAiLyI6IHsKICAgICAgICAgICAgICAgICAgICJnZXQiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgInJlc3BvbnNlcyI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIjIwMCI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJkZXNjcmlwdGlvbiI6ICJJbmRleCIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAiY29udGVudCI6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAidGV4dC9odG1sIjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAic2NoZW1hIjogeyJ0eXBlIjogInN0cmluZyJ9LAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgIH0KICAgICAgICAgICB9LAogICAgICAgfQoKCiAgIGNsYXNzIFJvb3RIYW5kbGVyKE15UmVxdWVzdEhhbmRsZXIpOgogICAgICAgYXN5bmMgZGVmIGdldChzZWxmKToKICAgICAgICAgICBzZWxmLmZpbmlzaCgiSGVsbG8sIFdvcmxkISIpCgoKICAgaWYgX19uYW1lX18gPT0gIl9fbWFpbl9fIjoKICAgICAgIGFwcCA9IHRvcm5hZG8ud2ViLkFwcGxpY2F0aW9uKFsociIvIiwgUm9vdEhhbmRsZXIpXSkKICAgICAgIGFwcC5saXN0ZW4oODg4OCkKICAgICAgIHRvcm5hZG8uaW9sb29wLklPTG9vcC5jdXJyZW50KCkuc3RhcnQoKQoKVmFsaWRhdGluZyByZXNwb25zZXMgaW4gdGVzdHMKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCi4uIGNvZGU6OiBweXRob24KCiAgIGltcG9ydCB1bml0dGVzdAoKICAgaW1wb3J0IHRvcm5hZG8ud2ViCiAgIGZyb20gdG9ybmFkb19vcGVuYXBpMy50ZXN0aW5nIGltcG9ydCBBc3luY09wZW5BUElUZXN0Q2FzZQoKCiAgIGNsYXNzIFJvb3RIYW5kbGVyKHRvcm5hZG8ud2ViLlJlcXVlc3RIYW5kbGVyKToKICAgICAgIGFzeW5jIGRlZiBnZXQoc2VsZik6CiAgICAgICAgICAgc2VsZi5maW5pc2goIkhlbGxvLCBXb3JsZCEiKQoKCiAgIGNsYXNzIEJhc2VUZXN0Q2FzZShBc3luY09wZW5BUElUZXN0Q2FzZSk6CiAgICAgICBzcGVjX2RpY3QgPSB7CiAgICAgICAgICAgIm9wZW5hcGkiOiAiMy4wLjAiLAogICAgICAgICAgICJpbmZvIjogewogICAgICAgICAgICAgICAidGl0bGUiOiAiU2ltcGxlIEV4YW1wbGUiLAogICAgICAgICAgICAgICAidmVyc2lvbiI6ICIxLjAuMCIsCiAgICAgICAgICAgfSwKICAgICAgICAgICAicGF0aHMiOiB7CiAgICAgICAgICAgICAgICIvIjogewogICAgICAgICAgICAgICAgICAgImdldCI6IHsKICAgICAgICAgICAgICAgICAgICAgICAicmVzcG9uc2VzIjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAiMjAwIjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIkluZGV4IiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJjb250ZW50IjogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJ0ZXh0L2h0bWwiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICJzY2hlbWEiOiB7InR5cGUiOiAic3RyaW5nIn0sCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgfQogICAgICAgICAgIH0sCiAgICAgICB9CgogICAgICAgZGVmIGdldF9hcHAoc2VsZik6CiAgICAgICAgICAgcmV0dXJuIHRvcm5hZG8ud2ViLkFwcGxpY2F0aW9uKFsociIvIiwgUm9vdEhhbmRsZXIpXSkKCiAgICAgICBkZWYgdGVzdF9yb290X2VuZHBvaW50KHNlbGYpOgogICAgICAgICAgIHJlc3BvbnNlID0gc2VsZi5mZXRjaCgiLyIpCiAgICAgICAgICAgc2VsZi5hc3NlcnRFcXVhbCgyMDAsIHJlc3BvbnNlLmNvZGUpCiAgICAgICAgICAgc2VsZi5hc3NlcnRFcXVhbChiIkhlbGxvLCBXb3JsZCEiLCByZXNwb25zZS5ib2R5KQoKCiAgIGlmIF9fbmFtZV9fID09ICJfX21haW5fXyI6CiAgICAgICB1bml0dGVzdC5tYWluKCkKCkNvbnRyaWJ1dGluZwo9PT09PT09PT09PT0KCkdldHRpbmcgU3RhcnRlZAotLS0tLS0tLS0tLS0tLS0KClRoaXMgcHJvamVjdCB1c2VzIGBQb2V0cnlgXyB0byBtYW5hZ2UgaXRzIGRlcGVuZGVuY2llcy4gVG8gc2V0IHVwIGEgbG9jYWwKZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnQsIGp1c3QgcnVuOgoKLi4gY29kZTo6IHNoCgogICAgcG9ldHJ5IGluc3RhbGwKCkZvcm1hdHRpbmcgQ29kZQotLS0tLS0tLS0tLS0tLS0KClRoZSBgQmxhY2tgXyB0b29sIGlzIHVzZWQgYnkgdGhpcyBwcm9qZWN0IHRvIGZvcm1hdCBQeXRob24gY29kZS4gSXQgaXMgaW5jbHVkZWQKYXMgYSBkZXZlbG9wbWVudCBkZXBlbmRlbmN5LCBhbmQgc2hvdWxkIGJlIHJ1biBvbiBhbGwgY29tbWl0dGVkIGNvZGUuIFRvIGZvcm1hdApjb2RlIHByaW9yIHRvIGNvbW1pdHRpbmcgaXQgYW5kIHN1Ym1pdHRpbmcgYSBQUiwgcnVuOgoKLi4gY29kZTo6IHNoCgogICAgcG9ldHJ5IHJ1biBibGFjayAuCgpSdW5uaW5nIFRlc3RzCi0tLS0tLS0tLS0tLS0KCmBweXRlc3RgXyBpcyB0aGUgcHJlZmVycmVkIHRlc3QgcnVubmVyIGZvciB0aGlzIHByb2plY3QuIEl0IGlzIGluY2x1ZGVkIGFzIGEKZGV2ZWxvcG1lbnQgZGVwZW5kZW5jeSwgYW5kIGlzIGNvbmZpZ3VyZWQgdG8gdHJhY2sgY29kZSBjb3ZlcmFnZSwgYEZsYWtlOGBfCnN0eWxlIGNvbXBsaWFuY2UsIGFuZCBgQmxhY2tgXyBjb2RlIGZvcm1hdHRpbmcuIFRlc3RzIGNhbiBiZSBydW4gaW4geW91cgpkZXZlbG9wbWVudCBlbnZpcm9ubWVudCBieSBydW5uaW5nOgoKLi4gY29kZTo6IHNoCgogICAgcG9ldHJ5IHJ1biBweXRlc3QKCkFkZGl0aW9uYWxseSwgdGVzdHMgY2FuIGJlIHJ1biB1c2luZyBgdG94YF8sIHdoaWNoIHdpbGwgcnVuIHRoZSB0ZXN0cyB1c2luZwptdWx0aXBsZSB2ZXJzaW9ucyBvZiBib3RoIFB5dGhvbiBhbmQgVG9ybmFkbyB0byBlbnN1cmUgYnJvYWQgY29tcGF0aWJpbGl0eS4KCkNvbmZpZ3VyaW5nIEh5cG90aGVzaXMKXl5eXl5eXl5eXl5eXl5eXl5eXl5eXgoKTWFueSBvZiB0aGUgdGVzdHMgbWFrZSB1c2Ugb2YgYEh5cG90aGVzaXNgXyB0byBzcGVjaWZ5IHRoZWlyIGV4cGVjdGF0aW9ucyBhbmQKZ2VuZXJhdGUgYSBsYXJnZSB2b2x1bWUgb2YgcmFuZG9taXplZCB0ZXN0IGlucHV0LiBCZWNhdXNlIG9mIHRoaXMsIHRoZSB0ZXN0cyBtYXkKdGFrZSBhIGxvbmcgdGltZSB0byBydW4gb24gc2xvd2VyIGNvbXB1dGVycy4gVHdvIHByb2ZpbGVzIGFyZSBkZWZpbmVkIGZvcgpIeXBvdGhlc2lzIHRvIHVzZSB3aGljaCBjYW4gYmUgc2VsZWN0ZWQgYnkgc2V0dGluZyB0aGUgYGBIWVBPVEhFU0lTX1BST0ZJTEVgYAplbnZpcm9ubWVudCB2YXJpYWJsZSB0byBvbmUgb2YgdGhlIGZvbGxvd2luZyB2YWx1ZXM6CgpgYGNpYGAKICBSdW5zIHRlc3RzIHVzaW5nIHRoZSBkZWZhdWx0IEh5cG90aGVzaXMgc2V0dGluZ3MgKDEwMCBleGFtcGxlcyBwZXIgdGVzdCkgYW5kCiAgbm8gY29tcGxldGlvbiBkZWFkbGluZS4KCmBgZGV2YGAKICBUaGUgZmFzdGVzdCBwcm9maWxlLCBtZWFudCBmb3IgbG9jYWwgZGV2ZWxvcG1lbnQgb25seS4gVXNlcyBvbmx5IDEwIGV4YW1wbGVzCiAgcGVyIHRlc3Qgd2l0aCBubyBjb21wbGV0aW9uIGRlYWRsaW5lLgoKCi4uIF9CbGFjazogaHR0cHM6Ly9naXRodWIuY29tL3BzZi9ibGFjawouLiBfRmxha2U4OiBodHRwczovL2ZsYWtlOC5weWNxYS5vcmcvCi4uIF9IeXBvdGhlc2lzOiBodHRwczovL2h5cG90aGVzaXMucmVhZHRoZWRvY3MuaW8vCi4uIF9PcGVuQVBJIDM6IGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLwouLiBfT3BlbmFwaS1jb3JlOiBodHRwczovL2dpdGh1Yi5jb20vcDFjMnUvb3BlbmFwaS1jb3JlCi4uIF9Qb2V0cnk6IGh0dHBzOi8vcHl0aG9uLXBvZXRyeS5vcmcvCi4uIF9Ub3JuYWRvOiBodHRwczovL3d3dy50b3JuYWRvd2ViLm9yZy8KLi4gX3B5dGVzdDogaHR0cHM6Ly9weXRlc3Qub3JnLwouLiBfdG94OiBodHRwczovL3RveC5yZWFkdGhlZG9jcy5pby8K readmeEtag: '"7394fcb2479f537d061a59604e8e8b91cb442d7c"' readmeLastModified: Mon, 21 Oct 2024 03:27:13 GMT repositoryId: 296513121 description: Tornado OpenAPI 3 request and response validation library. created: '2020-09-18T04:27:02Z' updated: '2024-10-21T03:53:28Z' language: Python archived: false stars: 5 watchers: 2 forks: 4 owner: correl logo: https://avatars.githubusercontent.com/u/289305?v=4 license: MIT repoEtag: '"13a4505b2b198abb6e42dfec9d300a70ff5a5a77305cfa58c787367f2537a758"' repoLastModified: Mon, 21 Oct 2024 03:53:28 GMT foundInMaster: true category: - Data Validators - Server Implementations id: 12f81e818f4c0355eff255ec1102d959 - source: openapi3 tags repository: https://github.com/elevup/mogen v3: true repositoryMetadata: base64Readme: >- ![Mogen logo](logo.svg "Mogen logo")

Mogen is a small library that converts Kotlin's models to other programming languages:

- Dart 2.12+,
- OpenApi 3.0 YML (generates `schemas` node),
- Swift,
- TypeScript.

### Motivation

In elevup we usually write backends in Kotlin. Unfortunately, frontends are whole other story. So the main goal was to
find a way to transfer the knowledge efficiently, and we went for OpenApi. But rewriting each diff in models to OpenApi
was more-less pain and literally no one wanted to do it.

Thanks to Mogen we were able to take all Kotlin models and convert them to OpenApi format. We felt that it was not 
enough cause all frontend devs had to rewrite them once again. So eventually we added convertors to more languages.

At this moment we generate frontend API models from backend models.

### Features
Supported features:
 - primitives - conversion from Kotlin to target language,
 - homogenous iterables - select collections and arrays are converted,
 - null-safety,
 - enums - including custom serialisation name,
 - classes - all public properties are transformed,
 - nested classes,
 - type aliases - only trivial cases, for example `typealias UserId = Long`.

Unsupported features:
 - maps,
 - generics,
 - inheritance,
 - abstract classes,
 - packages (= output is supposed to be printed into one file).

Have a look at `sample/src/main` where in `kotlin` folder are source files and in `resources` folder 
you can find generated output.

## Latest version

Library is available on Maven Central.

![Latest version](https://img.shields.io/maven-central/v/com.elevup/mogen "Latest lib version")
```groovy
implementation 'com.elevup:mogen:$version'
```

## How to use
First you need to gather all classes that need to be transformed. Simple create list of them.

```kotlin
val classes: List<KClass<*>> = listOf(Foo::class, Bar::class)
```

### Generate definitions
Depending on desired output pick a generator:

- `DartGenerator` ... Dart 2.12+
- `OpenApiGenerator` ... OpenApi 3.0
- `SwiftGenerator` ... Swift
- `TypeScriptGenerator` ... TypeScript

```kotlin
val generator: CachedGenerator = TypeScriptGenerator()
    .appendClass(SomeClass::class)
    .appendClasses(listOf(Foo::class, Bar::class))

// Access generated definitions
val classes = generator.generatedClasses
val enums = generator.generatedEnums
val types = generator.generatedTypeAliases
```

### Type aliases
When adding classes Mogen automatically detects presence of type aliases. However, in some cases
additional mapping is required. Imagine you want to serialise `LocalDateTime`. In some implementations it will become
`String` (ISO representation), in some `Long` (UNIX timestamp). This conversion is out of the scope of this
library but your frontend models must be compatible. In order to do so, you must define target language type.

For example in `TypeScript` date is represented by `Date`. To map it correctly, use `appendTypealias` method:

```kotlin
TypeScriptGenerator()
    .appendClass(User::class)
    .appendTypealias(Typealias(localClass = LocalDateTime::class, name = "Date"))
```

Input (Kotlin):
```kotlin
typealias UserId = Long

data class User(
    val id: UserId,
    val birthday: LocalDateTime,
)
```
Output (TypeScript):
```typescript
export type LocalDateTime = Date
export type UserId = number // copied automatically

export interface User {
    birthday: LocalDateTime; // Kotlin's class name is kept, type is created
    id: UserId;
}
```

### Indents
Everyone uses different indentation rules. To make it easier for you Mogen lets you configure basic indentation unit.
To adjust default indentation pass your own `Indentation` in generator's constructor. You can create your own instance or
use one of pre-defined indents:
- `GenericIndents` - designed for Dart, Swift and TypeScript,
- `OpenApiIndents` - designed for OpenApi.

```kotlin
TypeScriptGenerator(indents = GenericIndents(indent = "\t")) // For 'tabs' crew
OpenApiGenerator(indents = OpenApiIndents(indent = "  ")) // Fore 'spaces' crew
```

### Pretty print
In case you do not want to deal with formatting use `Printer` to convert generated definitions to single `String`
Library comes with bundled:
 - `GenericPrinter` that deals well with Dart, Swift and TypeScript,
 - `OpenApiPrinter` that is designed for OpenApi.

```kotlin
val allDefinitions: String = generator.print(GenericPrinter)
```

### Annotation processors
Sometimes it's useful to keep some metadata. Imagine you have just deprecated some property in backed, and you 
want to let others know that they should migrate their code. In that case you want to
use `DeprecatedAnnotationProcessor`! Annotation processors are passed in generator's constructor.

#### Deprecation
```kotlin
TypeScriptGenerator(annotationProcessors = listOf(DeprecatedAnnotationProcessor())) // <- pass them here
       .appendClass(User::class)
       .appendTypeAlias(Typealias(localClass = LocalDateTime::class, name = "Date"))
       .print(GenericPrinter)
       .let { println(it) }
```

Input (Kotlin):
```kotlin
data class User(
    val id: Long,
    @Deprecated("Not time-zone bulletproof, calculate it on your own")
    val age: Int,
    val birthday: LocalDateTime,
)
```

Output (TypeScript):
```typescript
export type LocalDateTime = Date

export interface User {
  /**
   * @deprecated Not time-zone bulletproof, calculate it on your own
   */
  age: number;
  birthday: LocalDateTime;
  id: number;
}

```

#### JavaX annotations
For validations like min, max, etc. on server; during conversion it will append comment where validation constraints 
are listed.

Input (Kotlin):
```kotlin
data class User(
    val id: Long,
    @field:Size(min = 1, max = 100)
    val firstName: String,
)
```

Output (TypeScript):
```typescript
export interface User {
  /**
   * min: 1
   * max: 100
   */
  firstName: string;
  id: number;
}
```

#### Jackson annotations
Handles `@JsonProperty` and `@JsonValue` annotations that can modify name of serialised property.

Input (Kotlin):
```kotlin
data class User(
    @JsonProperty("user_id")
    val id: Long,
    @JsonProperty("tea")
    val favouriteTea: Tea? = null,
) {
    enum class Tea {
        @JsonProperty("g") GREEN, 
        @JsonProperty("b") BLACK,
    }
}
```

Output (TypeScript):
```typescript
export enum UserTea {
  GREEN = 'g',
  BLACK = 'b',
}

export interface User {
    tea: UserTea | null;
    user_id: number;
}
```

### How to collect all models
If all models are in one package then [Reflections](https://github.com/ronmamo/reflections) library can be helpful! Then you can use this code to
get all classes:

```kotlin
val sourcePackage = "com.foo.bar" // Enter correct value
val reflections = Reflections(
    ConfigurationBuilder()
        .filterInputsBy(FilterBuilder().includePackage(sourcePackage))
        .setUrls(ClasspathHelper.forPackage(sourcePackage))
        .setScanners(SubTypesScanner(false))
)

val typeList = reflections.getSubTypesOf(Object::class.java) + reflections.getSubTypesOf(Enum::class.java)
val classes = typeList.map { c -> c.kotlin }.distinct()
```

## License

    Copyright 2021 elevup

    Licensed under the Apache License, Version 2.0 (the "License");
    you may not use this file except in compliance with the License.
    You may obtain a copy of the License at

       https://www.apache.org/licenses/LICENSE-2.0

    Unless required by applicable law or agreed to in writing, software
    distributed under the License is distributed on an "AS IS" BASIS,
    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
    See the License for the specific language governing permissions and
    limitations under the License.
 readmeEtag: '"8df68b06f4727560ecf498a05838d7c6ce321d65"' readmeLastModified: Thu, 31 Mar 2022 08:47:39 GMT repositoryId: 455560085 description: Converts Kotlin models to other languages created: '2022-02-04T13:36:46Z' updated: '2025-08-13T11:23:34Z' language: Kotlin archived: false stars: 6 watchers: 1 forks: 0 owner: elevup logo: https://avatars.githubusercontent.com/u/79261866?v=4 license: Apache-2.0 repoEtag: '"4bcdfdf7273b6ff3e16350f6926a617c3be2cc129c65167aaa8daeb18991efdf"' repoLastModified: Wed, 13 Aug 2025 11:23:34 GMT foundInMaster: true id: fe9ef951386e259cd49cdddcbd37f572 - source: openapi3 tags repository: https://github.com/ja88a/openapi-nestjs-auth-mongo v3: true id: 454350ec4d744135cda5c2bccc834fb3 repositoryMetadata: base64Readme: >- <!-- 
[![Contributors][sopenapi-contributors-shield]][sopenapi-contributors]
[![Forks][sopenapi-forks-shield]][sopenapi-forks]
[![Stargazers][sopenapi-stars-shield]][sopenapi-stars]
[![Issues][sopenapi-issues-shield]][sopenapi-issues]
[![MIT License][sopenapi-license-shield]][license]
-->

[![NestJs][nestjs-shield]][ref-nestjs]
[![NodeJs][nodejs-shield]][ref-nodejs]
[![Typescript][typescript-shield]][ref-typescript]
[![MongoDB][mongodb-shield]][ref-mongodb]
[![JWT][jwt-shield]][ref-jwt]
[![Jest][jest-shield]][ref-jest]
[![Yarn][yarn-shield]][ref-yarn]
[![Docker][docker-shield]][ref-docker]

# S*OpenAPI

## Purpose

> A mono repo to develop a set of NestJs-based Open API Services, Restful API Microservices with full auth support

Made with following best practices:
- [nodejs-best-practice](https://github.com/goldbergyoni/nodebestpractices) 
- [The Twelve-Factor App](https://12factor.net)
- [Microservice Architecture](https://microservices.io)

## Modules
- [account-api](packages/account-api/): User accounts, permissions, service settings & API keys management
- [common-utils](packages/common-utils): Common set of utilities for the MS APIs: authentication, security guards, pagination, filters, etc.
- [sales-api](packages/sales-api): set of APIs to query sales info

## Important

> Work in progress. Updates, customizations and new features are still required.

## Built With

Describes which version of the main packages and main tools.

| Name                          | Version  |
| ----------                    | -------- |
| [NestJs](ref-nestjs)          | v8.x     |
| [NodeJs](ref-nodejs)          | v17.x    |
| [Typescript](ref-typescript)  | v4.x     |
| [Mongoose](ref-mongoose)      | v6.x     |
| [MongoDB](ref-mongodb)        | v5.x     |
| [Yarn](ref-yarn)              | v1.x     |
| [NPM](ref-npm)                | v8.x     |
| [Docker](ref-docker)          | v20.x    |
| [Docker Compose](ref-docker-compose) | v2.x |


## Dev Instructions

### Init your dev environment

```
lerna bootstrap 
lerna link
lerna run build
```
OR
```
make bootstrap
```

### Test

```
lerna run test
```

## Prerequisites

1. Understand [NestJs Fundamental](http://nestjs.com), Main Framework. NodeJs Framework with support fully TypeScript.
2. Understand[Typescript Fundamental](https://www.typescriptlang.org), Programming Language. It will help us to write and read the code.
3. Understand [ExpressJs Fundamental](https://nodejs.org), a NodeJs based Framework. It helps in understanding how the NestJs Framework works.
4. Understand what NoSql is and how it works as a database, especially [MongoDB.](https://docs.mongodb.com)

## TODOs

Next developments:
- [ ] Production: Nestjs composition and Docker images generation
- [ ] Complete CI/CD of modules using Github actions
- [ ] Integrate Terraform for deploying the containers on k8s
- [ ] Deploy on Azure Kubernetes Services

## License

Distributed under [MIT licensed][license].


## Contact

[Jabba 01][author-email]

[![Github][github-shield]][author-github]
[![LinkedIn][linkedin-shield]][author-linkedin]

<!-- BADGE LINKS -->
[sopenapi-contributors-shield]: https://img.shields.io/github/contributors/ja88a/openapi-nestjs-auth-mongo?style=for-the-badge
[sopenapi-forks-shield]: https://img.shields.io/github/forks/ja88a/openapi-nestjs-auth-mongo?style=for-the-badge
[sopenapi-stars-shield]: https://img.shields.io/github/stars/ja88a/openapi-nestjs-auth-mongo?style=for-the-badge
[sopenapi-issues-shield]: https://img.shields.io/github/issues/ja88a/openapi-nestjs-auth-mongo?style=for-the-badge
[sopenapi-license-shield]: https://img.shields.io/github/license/ja88a/openapi-nestjs-auth-mongo?style=for-the-badge

[nestjs-shield]: https://img.shields.io/badge/nestjs-%23E0234E.svg?style=for-the-badge&logo=nestjs&logoColor=white
[nodejs-shield]: https://img.shields.io/badge/Node.js-339933?style=for-the-badge&logo=nodedotjs&logoColor=white
[typescript-shield]: https://img.shields.io/badge/TypeScript-007ACC?style=for-the-badge&logo=typescript&logoColor=white
[mongodb-shield]: https://img.shields.io/badge/MongoDB-white?style=for-the-badge&logo=mongodb&logoColor=4EA94B
[jwt-shield]: https://img.shields.io/badge/JWT-000000?style=for-the-badge&logo=JSON%20web%20tokens&logoColor=white
[jest-shield]: https://img.shields.io/badge/-jest-%23C21325?style=for-the-badge&logo=jest&logoColor=white
[yarn-shield]: https://img.shields.io/badge/yarn-%232C8EBB.svg?style=for-the-badge&logo=yarn&logoColor=white
[docker-shield]: https://img.shields.io/badge/docker-%230db7ed.svg?style=for-the-badge&logo=docker&logoColor=white

[github-shield]: https://img.shields.io/badge/GitHub-100000?style=for-the-badge&logo=github&logoColor=white
[linkedin-shield]: https://img.shields.io/badge/LinkedIn-0077B5?style=for-the-badge&logo=linkedin&logoColor=white

<!-- CONTACTS -->
[author-linkedin]: https://linkedin.com/in/srenault
[author-email]: mailto:r0g3r@tuta.io
[author-github]: https://github.com/ja88a

<!-- Repo LINKS -->
[sopenapi-endpoint]: /endpoints/endpoints.json

<!-- license -->
[license]: LICENSE.md
[endpoints]: endpoints.json

<!-- Documents -->
[sopenapi-docs]: https://andrechristikan.github.io/ack-nestjs-boilerplate-docs/
[sopenapi-docs-features]: https://andrechristikan.github.io/ack-nestjs-boilerplate-docs/#/features/readme
[sopenapi-docs-example]: https://andrechristikan.github.io/ack-nestjs-boilerplate-docs/#/example
[sopenapi-docs-tips]: https://andrechristikan.github.io/ack-nestjs-boilerplate-docs/#/tips/readme
[sopenapi-doc-env]: https://andrechristikan.github.io/ack-nestjs-boilerplate-docs/#/features/readme

<!-- Reference -->
[ref-nestjs]: http://nestjs.com
[ref-mongoose]: https://mongoosejs.com/
[ref-mongodb]: https://docs.mongodb.com/
[ref-nodejs-best-practice]: https://github.com/goldbergyoni/nodebestpractices
[ref-nodejs]: https://nodejs.org/
[ref-typescript]: https://www.typescriptlang.org/
[ref-jwt]: https://jwt.io
[ref-jest]: https://jestjs.io/docs/getting-started
[ref-docker]: https://docs.docker.com
[ref-docker-compose]: https://docs.docker.com
[ref-yarn]: https://yarnpkg.com
[ref-postman-import-export]: https://learning.postman.com/docs/getting-started/importing-and-exporting-data/
 readmeEtag: '"c6d8d3335be96045209443a85bbcb228b582f386"' readmeLastModified: Wed, 31 May 2023 09:24:27 GMT repositoryId: 504584334 description: >- Generic micro-services dev framework w/ user accounts, permissions and authentication management created: '2022-06-17T15:26:18Z' updated: '2025-05-20T09:04:33Z' language: TypeScript archived: false stars: 5 watchers: 1 forks: 0 owner: ja88a logo: https://avatars.githubusercontent.com/u/80005689?v=4 license: MIT repoEtag: '"bfd8a0feb24adabff456a831da46c9272abec350e2b9c0f146afffa9c459ffd9"' repoLastModified: Tue, 20 May 2025 09:04:33 GMT category: Server foundInMaster: true - source: openapi3 tags repository: https://github.com/phstudy/einvoice-api v3: true repositoryMetadata: repositoryId: 122307410 description: 財政部電子發票應用 API 規格 - OpenAPI 3.0 格式 created: '2018-02-21T08:00:03Z' updated: '2025-03-25T04:56:12Z' language: null archived: false stars: 6 watchers: 1 forks: 4 owner: phstudy logo: https://avatars.githubusercontent.com/u/2286805?v=4 license: MIT repoEtag: '"3a63104b9a28372dfd6934ec32544540386f107c0391c9a75d88aabc948653e1"' repoLastModified: Tue, 25 Mar 2025 04:56:12 GMT foundInMaster: true id: c508d10b2d4a66d9a84dd2a08e25e125 - source: openapi3 tags repository: https://github.com/ioggstream/openapi-resolver v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFJlc29sdmVyCgpbIVtDaXJjbGVDSV0oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvaW9nZ3N0cmVhbS9vcGVuYXBpLXJlc29sdmVyLnN2Zz9zdHlsZT1zdmcpXShodHRwczovL2NpcmNsZWNpLmNvbS9naC9pb2dnc3RyZWFtL29wZW5hcGktcmVzb2x2ZXIpClshW0NvZGVDb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9pb2dnc3RyZWFtL29wZW5hcGktcmVzb2x2ZXIvYnJhbmNoL21hc3Rlci9ncmFwaC9iYWRnZS5zdmcpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvaW9nZ3N0cmVhbS9vcGVuYXBpLXJlc29sdmVyKQoKVGhpcyBtb2R1bGUgcmVjdXJzaXZlbHkgcGFyc2VzIG9wZW5hcGkgc3BlY3MgcmVzb2x2aW5nIHJlZmVyZW5jZXMuCgojIyBUZXN0CgpUZXN0cyBydW4gbG9jYWxseSB2aWEgCgogICAgICAgIHRveAoKT3IgdmlhIFtjaXJjbGVjaS1sb2NhbF0oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZG9jcy8yLjAvbG9jYWwtY2xpLykKCiAgICAgICAgY2lyY2xlY2kgYnVpbGQgCgoKIyMgVXNhZ2UKClRoZSBtb2R1bGUgaGFzIGFuIGVtYmVkZGVkIHNjcmlwdCB0aGF0IGNhbiBiZSBydW4gdmlhCgogICAgICAgICQgcHl0aG9uIC1tIG9wZW5hcGlfcmVzb2x2ZXIgLS1oZWxwCgogICAgICAgIHVzYWdlOiBfX21haW5fXy5weSBbLWhdIHNyY19maWxlIFtkc3RfZmlsZV0KCiAgICAgICAgUmVjdXJzaXZlbHkgcmVzb2x2ZXMgYW5kIGJ1bmRsZXMgT3BlbkFQSSB2MyBmaWxlcy4KCiAgICAgICAgcG9zaXRpb25hbCBhcmd1bWVudHM6CiAgICAgICAgICBzcmNfZmlsZSAgICBBbiBPcGVuQVBJIHYzIHlhbWwgZmlsZS4KICAgICAgICAgIGRzdF9maWxlICAgIERlc3RpbmF0aW9uIGZpbGUsIGRlZmF1bHQgaXMgc3Rkb3V0LgoKICAgICAgICBvcHRpb25hbCBhcmd1bWVudHM6CiAgICAgICAgICAtaCwgLS1oZWxwICBzaG93IHRoaXMgaGVscCBtZXNzYWdlIGFuZCBleGl0CgpUbyBjcmVhdGUgYW4gb3BlbmFwaSBidW5kbGUgZnJvbSBhIHNwZWMgZmlsZSBqdXN0IHJ1bgoKICAgICAgICAkIHB5dGhvbiAtbSBvcGVuYXBpX3Jlc29sdmVyIHNhbXBsZS55YW1sCgpZb3UgY2FuIHVzZSB0aGlzIG1vZHVsZSB0byBub3JtYWxpemUgdHdvIHNwZWNzIGJlZm9yZSBkaWZmaW5nLCBlZzoKCiAgICAgICAgJCBweXRob24gLW0gb3BlbmFwaV9yZXNvbHZlciBvbmUueWFtbCBub3JtYWwtb25lLnlhbWwKICAgICAgICAkIHB5dGhvbiAtbSBvcGVuYXBpX3Jlc29sdmVyIHR3by55YW1sIG5vcm1hbC10d28ueWFtbAogICAgICAgICQgZGlmZiBub3JtYWwtb25lLnlhbWwgbm9ybWFsLXR3by55YW1sCgojIyBVc2Ugd2l0aCBkb2NrZXIKCkJ1aWxkIHRoZSBpbWFnZSB3aXRoOgoKYGBgCiQgZG9ja2VyIGJ1aWxkIC0tdGFnIG9wZW5hcGktcmVzb2x2ZXIgLiAKYGBgCgp0aGVuIHJ1biBkb2NrZXIgbWFwcGluZyB0aGUgb3BlbmFwaS55YW1sIGRpcmVjdG9yeQp0byB0aGUgYC9jb2RlYCB2b2x1bWUuCgpgYGAKZG9ja2VyIHJ1biAtaXQgLS1ybSAtdiAkKGRpcm5hbWUgcGF0aC10by1vcGVuYXBpLnlhbWwpOi9jb2RlIFwKICAgIG9wZW5hcGktcmVzb2x2ZXIgL2NvZGUvb3BlbmFwaS55YW1sIC9jb2RlL2J1bmRsZS55YW1sCmBgYAo= readmeEtag: '"15ddc680312dda7420a034eae0633cac8314fe4a"' readmeLastModified: Wed, 09 Oct 2019 14:05:38 GMT repositoryId: 145992506 description: A python 3 module for resolving $refs in OpenAPI specs and create bundles. created: '2018-08-24T13:05:35Z' updated: '2024-09-15T22:00:58Z' language: Python archived: false stars: 5 watchers: 1 forks: 5 owner: ioggstream logo: https://avatars.githubusercontent.com/u/1140844?v=4 license: MIT repoEtag: '"853a9c7182ea9dc8e79ac3775d850456f465e95fd3cf9cfd87d98577052c4a26"' repoLastModified: Sun, 15 Sep 2024 22:00:58 GMT foundInMaster: true category: - Description Validators - Parsers id: 40ab6688a0f4a02b4e8ac406853b4013 - source: openapi3 tags repository: https://github.com/clowre/docserver v3: true repositoryMetadata: base64Readme: >- IyBDbG93cmUgT3BlbkFQSSBEb2N1bWVudCBTZXJ2ZXIKCkEgc2ltcGxlIHByb2dyYW0gdGhhdCBtZXJnZXMgbXVsdGlwbGUgT3BlbkFQSS9Td2FnZ2VyIHNjaGVtYSBmaWxlcywgYW5kIHNlcnZlcyB0aGUgbWVyZ2VkIHJlc3VsdC4gVGhpcyBwcm9qZWN0IHdhcyBib3JuCnRvIHNvbHZlIGEgdmVyeSBzcGVjaWZpYyB1c2UtY2FzZSB3aGVyZSB3ZSBhdCBDbG93cmUgbmVlZGVkIHRvIHNlZSB0aGUgQVBJIGRvY3Mgb2YgYWxsIG91ciBtaWNyb3NlcnZpY2VzIGF0IG9uZSBwbGFjZS4KCkhvd2V2ZXIsIHRoaXMgcHJvamVjdCBhbHNvIGFpbXMgdG8gYmUgYSBnZW5lcmljIHNvbHV0aW9uIGZvciBPcGVuQVBJIFNjaGVtYSBtZXJnaW5nLgoKIyMgUnVubmluZwoKQnkgZGVmYXVsdCwgdGhlIHNlcnZlciB0cmllcyB0byByZXNvbHZlIGNvbmZpZ3VyYXRpb24gZnJvbSBhIFNwcmluZyBDbG91ZCBDb25maWcgU2VydmVyLCBzbyBldmVyeSBleHRlcm5hbCBjb25maWd1cmF0aW9uCm11c3Qgc3BlY2lmeSB0aGlzIHByb3BlcnR5IGV4cGxpY2l0bHk6CgpgYGB5YW1sCm1pY3JvbmF1dDoKICBjb25maWctY2xpZW50OgogICAgZW5hYmxlZDogZmFsc2UKYGBgCgpUaGUgcmVjb21tZW5kZWQgd2F5IHRvIHJ1biB0aGlzIHNlcnZlciBpcyB0aHJvdWdoIERvY2tlci4gVGhpcyBzZXJ2ZXIgc2VydmVzIG1lcmdlZCBPcGVuQVBJIFNjaGVtYXMgYXQKYGh0dHA6Ly9ob3N0OnBvcnQvZG9jcy5qc29uYC4KCiMjIyBXaXRoIGV4dGVybmFsIGNvbmZpZ3VyYXRpb24KCmBgYHNoZWxsCmRvY2tlciBydW4gLS1ybSAtaXQgLS1uYW1lPWRvY3NlcnZlciAtcCA4MDgwOjgwODAgXAogIC12ICIkKHB3ZCkvX2V4YW1wbGVzL2V4dGVybmFsLWNvbmZpZ3VyYXRpb24ueWFtbDovY29uZmlnLnlhbWwiIFwKICAtZSBNSUNST05BVVRfQ09ORklHX0ZJTEVTPS9jb25maWcueWFtbCBcCiAgY2xvd3JlL2RvY3NlcnZlcjpsYXRlc3QKYGBgCgojIyMgVXNpbmcgU3ByaW5nIENsb3VkIENvbmZpZyBTZXJ2ZXIKCklmIHlvdSB3aXNoIHRvIGNvbmZpZ3VyZSB0aGUgZG9jc2VydmVyIHVzaW5nIGEgQ2xvdWQgQ29uZmlnIGluc3RhbmNlLCBpdCBpcyByZXF1aXJlZCB0aGF0IGBDT05GSUdfU0VSVkVSX1VSSWAgcG9pbnRzIHRvCnlvdXIgY29uZmlndXJhdGlvbiBzZXJ2ZXIuIFRoZSBkb2NzZXJ2ZXIncyBhcHBsaWNhdGlvbiBJRCBpcyBgY2xvd3JlLWRvY3NlcnZlcmAsIHNvIG1ha2Ugc3VyZSB0aGF0IHRoZSBjb25maWd1cmF0aW9uIGNhbgpiZSByZXNvbHZlZCBieSB0aGlzIG5hbWUuCgpgYGBzaGVsbApkb2NrZXIgcnVuIC0tcm0gLWl0IC0tbmFtZT1kb2NzZXJ2ZXIgLXAgODA4MDo4MDgwIFwKICAtZSBDT05GSUdfU0VSVkVSX1VSST1odHRwOi8veW91ci1jb25maWctc2VydmVyOjg4ODggXAogIC1lIE1JQ1JPTkFVVF9FTlZJUk9OTUVOVFM9Y2xvdWQgXAogIGNsb3dyZS9kb2NzZXJ2ZXI6bGF0ZXN0CmBgYAoKIyMgQ29uZmlndXJhdGlvbgoKIyMjIFN3YWdnZXIgU291cmNlcwoKVGhpcyBpcyB0aGUgb25seSBjb25maWd1cmF0aW9uIHRoYXQgaXMgcmVxdWlyZWQgdG8gYmUgcHJlc2VudCB0byBydW4gYSBiYXJlLW1pbmltdW0gdmVyc2lvbiBvZiB0aGlzIHNlcnZlci4gVGhpcwpjb25maWd1cmF0aW9uIGlzIHNwZWNpZmllZCBieSB0aGUgYHN3YWdnZXI6YCBrZXkgaW4gdGhlIGNvbmZpZ3VyYXRpb24gWUFNTC4gSGVyZSdzIGEgc2FtcGxlIGNvbmZpZ3VyYXRpb24gdGhhdCBtZXJnZXMgYQpjb3VwbGUgb2Ygc2NoZW1hcy4KCmBgYHlhbWwKc3dhZ2dlcjoKICAjIyBXaWxsIGJlIHVzZWQgdG8gZ2VuZXJhdGUgdGhlIG1lcmdlZCBPcGVuQVBJIGZpbGUncyAnc2VydmVycycgc2VjdGlvbiAtIG9wdGlvbmFsLgogIHNlcnZlcnM6CiAgICAtIHVybDogaHR0cDovL215c2VydmVyLmxvY2FsCiAgICAgIGRlc2NyaXB0aW9uOiBBIHJlYWxseSBjb29sIHNlcnZlciB0aGF0IGRvZXMgc3R1ZmYuCiAgIyMgV2lsbCBiZSB1c2VkIHRvIGdlbmVyYXRlIHRoZSBtZXJnZWQgT3BlbkFQSSBmaWxlJ3MgJ2luZm8nIHNlY3Rpb24uCiAgaW5mbzoKICAgICMjIFRpdGxlIG9mIHRoZSBtZXJnZWQgT3BlbkFQSSBzY2hlbWEuIFJlcXVpcmVkLgogICAgdGl0bGU6IEEgYmFzaWMgY29tYmluZWQgQVBJIFlBTUwKICAgIGRlc2NyaXB0aW9uOiB8LQogICAgICAjIyBXaGF0IGl0IGlzCiAgICAgIEFuIGV4YW1wbGUgb2YgY2xvd3JlL2RvY3NlcnZlcidzIHNjaGVtYSBtZXJnZSBmdW5jdGlvbmFsaXR5LgoKICAgICAgIyMgV2hhdCBpdCBtZXJnZXMKICAgICAgVGhlIHN3YWdnZXIgUGV0IFN0b3JlIGFuZCBCb3hLbmlnaHQgQVBJIHNjaGVtYXMuCgogICMjIExpc3Qgb2YgT3BlbkFQSSBzY2hlbWFzIHRvIGJlIG1lcmdlZC4KICBzb3VyY2VzOgogICAgIyMgQSBmcmllbmRseSBuYW1lIG9mIHRoaXMgcmVzb3VyY2UgdG8gYmUgdXNlZCBpbiBsb2dnaW5nIC0gcmVxdWlyZWQuCiAgICAtIG5hbWU6IFBldCBTdG9yZQoKICAgICAgIyMgU3RyaW5nIHRvIGJlIHVzZWQgYXMgYSBwcmVmaXggb2YgQVBJIHBhdGhzIC0gcmVxdWlyZWQKICAgICAgcGF0aC1wcmVmaXg6IHBldC1zdG9yZQoKICAgICAgIyMgTG9jYXRpb24gb2YgT3BlbkFQSSBzY2hlbWEgSlNPTi9ZQU1MIC0gcmVxdWlyZWQKICAgICAgYWRkcmVzczogaHR0cHM6Ly9yZWRvY2x5LmdpdGh1Yi5pby9yZWRvYy9vcGVuYXBpLnlhbWwKCiAgICAgICMjIFZlcnNpb24gb2Ygc2NoZW1hIC0gVjIgb3IgVjMgLSByZXF1aXJlZAogICAgICB2ZXJzaW9uOiBWMwpgYGAKCk1vcmUgY29uZmlndXJhdGlvbiBleGFtcGxlcyBjYW4gYmUgZm91bmQgaW4gYF9leGFtcGxlc2AgZGlyZWN0b3J5IG9mIHRoaXMgcmVwb3NpdG9yeS4KCiMjIyBWaWV3cwoKVGhlIGRvY3NlcnZlciBjYW4gYWxzbyBzZXJ2ZSBvZmYgU3dhZ2dlciBhbmQgUmVEb2Mgdmlld3Mgb3V0IG9mIHRoZSBib3guIFRoaXMgZnVuY3Rpb25hbGl0eSBjYW4gYmUgZW5hYmxlZCBieSBwdXR0aW5nCnRoZSBmb2xsb3dpbmcgaW4geW91ciBjb25maWd1cmF0aW9uOgoKYGBgeWFtbAp2aWV3czoKICBlbmFibGUtcmVkb2M6IHRydWUKICBlbmFibGUtc3dhZ2dlcjogdHJ1ZQpgYGAKCk9uY2UgZW5hYmxlZCwgdGhlIHZpZXdzIGNhbiBiZSBhY2Nlc3NlZCBhdCBgaHR0cDovL2hvc3Q6cG9ydC92aWV3cy9yZWRvY2AgYW5kIGBodHRwOi8vaG9zdDpwb3J0L3ZpZXdzL3N3YWdnZXJgLgoKLS0tCgojIyBQcm9qZWN0IHVzYWJpbGl0eSBzdGF0dXMKClRoaXMgcHJvamVjdCBpcyBpbiAqKmFscGhhKiogc3RhdGUgcmlnaHQgbm93LCBhbmQgaXMgbm90IG1lYW50IHRvIGJlIHVzZWQgaW4gcHJvZHVjdGlvbi4gSG93ZXZlciwgdGhpcyB3aWxsIGtlZXAgb24KZXZvbHZpbmcgYW5kIHdpbGwgc2VlIGEgcHJvZHVjdGlvbiByZWxlYXNlIGFzIHdlIGF0IENsb3dyZSBlbmhhbmNlIGl0LgoKIyMgQ29udHJpYnV0aW9ucyB3ZWxjb21lCgpXZSdyZSBoYXBweSB0byB0YWtlIGNvbnRyaWJ1dGlvbnMgaW4gZm9ybSBvZiBpc3N1ZXMsIGZlYXR1cmUgcmVxdWVzdHMsIGRvY3VtZW50YXRpb24sIGFuZCBjb2RlIQ== readmeEtag: '"90210a8acbb2facb5140253546fc47a7d85af4d5"' readmeLastModified: Fri, 08 Apr 2022 17:54:18 GMT repositoryId: 469345400 description: Merge and Serve OpenAPI/Swagger Documents created: '2022-03-13T11:03:23Z' updated: '2024-11-14T20:45:27Z' language: Java archived: false stars: 5 watchers: 1 forks: 0 owner: clowre logo: https://avatars.githubusercontent.com/u/91081481?v=4 license: MIT repoEtag: '"ed2b60a51cbb0d771dfe53f38512d65acabc0d2c0d838ea35615d9fdf722d417"' repoLastModified: Thu, 14 Nov 2024 20:45:27 GMT foundInMaster: true category: Parsers id: a0183f83fb00660e796a6fd36906afc6 - source: openapi3 tags repository: https://github.com/umarov-safar/laravel-openapi-codegen v3: true id: 3bb9c1a130e5239fe359c38168f61892 repositoryMetadata: base64Readme: >- # Laravel OpenAPI Code Generation

## Introduction
This package provides a convenient way to generate Laravel routes, requests, controllers and etc. based on OpenAPI specifications in YAML format.


### Motivation
For example, we have this openapi.yaml
```yaml
openapi: 3.0.0
info:
  title: User API
  version: 1.0.0
paths:
  /users:
    get:
      #...
      x-og-route-name: listUsers
      x-og-controller: App\Http\Controllers\UsersController@index
      x-og-skip-request: true
      x-og-middlewares: auth
      x-og-skip-resource: false
      responses:
        '200':
          description: A list of users.
          content:
            application/json:
              schema:
                type: object
                properties:
                  data:
                    type: array
                    items:
                      $ref: '#/components/schemas/User'
    post:
      ...
      x-og-route-name: createUser
      x-og-controller: App\Http\Controllers\UsersController@store
      x-og-skip-request: false
      x-og-skip-resource: false
      x-og-middlewares: auth,admin
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/User'
      responses:
        '201':
          description: User created successfully.
          content:
            application/json:
              schema:
                type: object
                properties:
                  data:
                    $ref: '#/components/schemas/User'
  /users/{userId}:
    put:
      ...
      x-og-generation: true
      x-og-route-name: updateUser
      x-og-controller: App\Http\Controllers\UsersController@update
      x-og-skip-request: false
      x-og-middlewares: auth,admin
      x-og-skip-resource: false
      parameters:
        - name: userId
          in: path
          required: true
          description: ID of the user to update.
          schema:
            type: integer
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/User'
      responses:
        '200':
          description: User updated successfully.
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/User'
components:
  schemas:
    User:
      type: object
      properties:
        id:
          type: integer
          format: int64
        username:
          type: string
          pattern: '^[a-zA-Z0-9_-]{3,16}$'
        email:
          type: string
          format: email
          maxLength: 30
      required:
        - username
        - email
```
by using package after running `php artisan openapi:generate-code` package generates for you below code.

`routes/openapi-codegen.php`
```php
<?php

use Illuminate\Support\Facades\Route;
use App\Http\Controllers\UsersController;

Route::get('users', [UsersController::class, 'index'])->name('listUsers')->middleware(['auth']);
Route::post('users', [UsersController::class, 'store'])->name('createUser')->middleware(['auth', 'admin']);
Route::put('users/{userId}', [UsersController::class, 'update'])->name('updateUser')->middleware(['auth', 'admin']);
```
`app/Http/Controllers/UsersController.php`
```php
<?php

namespace App\Http\Controllers;

use Illuminate\Http\JsonResponse;
use Illuminate\Http\Request;
use App\Http\Resources\UsersResource;
use App\Http\Requests\StoreUsersRequest;
use App\Http\Requests\UpdateUsersRequest;

class UsersController
{
    public function index(): JsonResponse
    {
        // return UsersResource();
    }

    public function store(StoreUsersRequest $request): JsonResponse
    {
        // return UsersResource();
    }

    public function update(int $userId, UpdateUsersRequest $request): JsonResponse
    {
        // return UsersResource();
    }

}
```
`app/Http/Requests/StoreUsersRequest.php`
```php
<?php

namespace App\Http\Requests;

use Illuminate\Http\Request;

class StoreUsersRequest extends Request
{
    public function rules(): array
    {
        return [
            'username' => ['required', 'string', 'regex:/^[a-zA-Z0-9_-]{3,16}$/'],
            'email' => ['required', 'string', 'email', 'max:30'],
        ];
    }
}
```
`app/Http/Requests/UpdateUsersRequest.php`
```php
<?php

namespace App\Http\Requests;

use Illuminate\Http\Request;

class UpdateUsersRequest extends Request
{
    public function rules(): array
    {
        return [
            'username' => ['required', 'string', 'regex:/^[a-zA-Z0-9_-]{3,16}$/'],
            'email' => ['required', 'string', 'email', 'max:30'],
        ];
    }
}
```
`app/Http/Resources/UsersResource.php`
```php
<?php

namespace App\Http\Resources;

use Illuminate\Http\Request;
use Illuminate\Http\Resources\Json\JsonResource;

class UsersResource extends JsonResource
{
    public function toArray(Request $request): array
    {
        return [
            'id' => $this->id,
            'username' => $this->username,
            'email' => $this->email,
        ];
    }
}
```
## Installation
To install the package, you can use Composer:

```
composer require --dev laravel-openapi/codegen
```

### Configuration
After installing the package, you'll need to configure it to suit your project's needs. This package provides various configuration options to customize the code generation process.

#### Publishing Configuration
```
php artisan vendor:publish --provider="LaravelOpenapi\Codegen\LaravelOpenapiCodegenProvider"
```
This command will create an `openapi-codegen.php` file in your Laravel application's `config` directory.

### Configuration Options

In the openapi-codegen.php configuration file, you'll find the following options:
* `api_docs_url`: Specifies the URL where the OpenAPI documentation will be accessible. This URL should point to the location of your OpenAPI specification file.
* `entities`: Defines the entities for which code generation will be performed. You can specify which entities should be generated by adding them to this array. Supported entities include:
  * `route`: Generate Laravel routes based on OpenAPI paths.
  * `request`: Generate request classes based on OpenAPI request bodies.
  * `resource`: Generate resource classes for API responses.
  * `controller`: Generate controllers for handling API requests.
  
    By default, all supported entities will be generated. 
    To exclude specific entities, simply remove them from this array.
* `paths`: Specifies additional paths used in the application. Currently, only the routes_file option is available, which defines the file path where the OpenAPI routes will be generated.

### Usage
#### Generate code
To generate Laravel code from an OpenAPI YAML file, you can use the following Artisan command:

```
php artisan openapi:generate-code
```

This command will generate Laravel routes, requests, resource and controllers based on the provided OpenAPI YAML file.


### Contributing
I welcome contributions from the community! If you have any ideas for improvements or find any issues, please feel free to open an issue or submit a pull request on GitHub.

### License
This package is open-source software licensed under the MIT License.

### Support
If you encounter any problems or have questions about using the package, please don't hesitate to reach out to us via email (safarumarov@gmail.com) or GitHub issues.
 readmeEtag: '"8c5c22689e4f6391ac27a87d382a50d1ba011862"' readmeLastModified: Mon, 15 Dec 2025 20:52:22 GMT repositoryId: 766410829 description: Laravel OpenAPI Code Generation created: '2024-03-03T07:23:26Z' updated: '2025-12-15T20:52:26Z' language: PHP archived: false stars: 6 watchers: 1 forks: 0 owner: umarov-safar logo: https://avatars.githubusercontent.com/u/68920816?v=4 repoEtag: '"ae596ba65e18cff70b682162ee0a930dfc79e499b3c7f45411cc3716ddc1038e"' repoLastModified: Mon, 15 Dec 2025 20:52:26 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/vect0rpro/dropwizard-swagger v3: true repositoryMetadata: base64Readme: >- RHJvcHdpemFyZCBTd2FnZ2VyIEludGVncmF0aW9uCj09PT09PT09PT09PT09PT09PQoKWyFbQXBhY2hlIExpY2Vuc2UgVjIuMF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWNlbnNlLUFwYWNoZSUyMFYyLTUwY2EyMi5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vVmVjdDByUHJvL2Ryb3B3aXphcmQtc3dhZ2dlci9ibG9iL21hc3Rlci9MSUNFTlNFKQpbIVtNYXZlbiBDZW50cmFsXShodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9pbi52ZWN0b3Jwcm8uZHJvcHdpemFyZC9kcm9wd2l6YXJkLXN3YWdnZXI/bGFiZWw9TWF2ZW4lMjBDZW50cmFsJmNvbG9yPTUwY2EyMildKGh0dHBzOi8vY2VudHJhbC5zb25hdHlwZS5jb20vYXJ0aWZhY3QvaW4udmVjdG9ycHJvLmRyb3B3aXphcmQvZHJvcHdpemFyZC1zd2FnZ2VyKQoKQSBEcm9wd2l6YXJkIGJ1bmRsZSB0aGF0IHNlcnZlcyBTd2FnZ2VyIFVJIHN0YXRpYyBjb250ZW50IGFuZCBsb2FkcyBTd2FnZ2VyIGVuZHBvaW50cy4KCkN1cnJlbnQgdmVyc2lvbiBoYXMgYmVlbiB0ZXN0ZWQgd2l0aCBEcm9wd2l6YXJkIDIuMS4xMiBhbmQgU3dhZ2dlciAyLjIuMiB3aGljaCBzdXBwb3J0cyBPcGVuQXBpIDMuMCBzcGVjaWZpY2F0aW9ucwoKUmVxdWlyZW1lbnRzCi0tLS0tLS0tLS0tLS0tCiogRHJvcHdpemFyZCAyLjEuMTIKKiBTd2FnZ2VyIEFQSSAyLjIuMgoqIFN3YWdnZXIgVUkgNS4yNy4wCgpVc2FnZQotLS0tLS0tLS0tLS0tCgoqIEFkZCB0aGUgTWF2ZW4gZGVwZW5kZW5jeSAoYXZhaWxhYmxlIGluIE1hdmVuIENlbnRyYWwpCgpgYGB4bWwKPGRlcGVuZGVuY3k+CiAgICA8Z3JvdXBJZD5pbi52ZWN0b3Jwcm8uZHJvcHdpemFyZDwvZ3JvdXBJZD4KICAgIDxhcnRpZmFjdElkPmRyb3B3aXphcmQtc3dhZ2dlcjwvYXJ0aWZhY3RJZD4KICAgIDx2ZXJzaW9uPjIuMS4xMi0xPC92ZXJzaW9uPgo8L2RlcGVuZGVuY3k+CmBgYAoKCiogQWRkIHRoZSBmb2xsb3dpbmcgdG8geW91ciBDb25maWd1cmF0aW9uIGNsYXNzOgoKYGBgamF2YQpwdWJsaWMgY2xhc3MgWW91ckNvbmZpZ3VyYXRpb24gZXh0ZW5kcyBDb25maWd1cmF0aW9uIHsKCiAgICBASnNvblByb3BlcnR5KCJzd2FnZ2VyIikKICAgIHB1YmxpYyBTd2FnZ2VyQnVuZGxlQ29uZmlndXJhdGlvbiBzd2FnZ2VyQnVuZGxlQ29uZmlndXJhdGlvbjsKYGBgCgoqIEFkZCB0aGUgZm9sbG93aW5nIHlvdXIgY29uZmlndXJhdGlvbiB5YW1sICh0aGlzIGlzIHRoZSBtaW5pbWFsIGNvbmZpZ3VyYXRpb24geW91IG5lZWQpOgoKYGBgeWFtbApwcm9wMTogdmFsdWUxCnByb3AyOiB2YWx1ZTIKCiMgdGhlIG9ubHkgcmVxdWlyZWQgcHJvcGVydHkgaXMgcmVzb3VyY2VQYWNrYWdlLCBmb3IgbW9yZSBjb25maWcgb3B0aW9ucyBzZWUgYmVsb3cKc3dhZ2dlcjoKICByZXNvdXJjZVBhY2thZ2U6IDxhIGNvbW1hIHNlcGFyYXRlZCBzdHJpbmcgb2YgdGhlIHBhY2thZ2VzIHRoYXQgY29udGFpbiB5b3VyIEBPcGVuQVBJRGVmaW5pdGlvbiBhbm5vdGF0ZWQgcmVzb3VyY2VzPgoKICAjIENvbmZpZ3VyZSBwYWdlIHRpdGxlIG9yIGNvZGUgc25pcHBldCB0YXJnZXRzIGFzIG5lZWRlZAogIHN3YWdnZXJWaWV3Q29uZmlndXJhdGlvbjoKICAgIHBhZ2VUaXRsZTogIkFQSSBEb2N1bWVudGF0aW9uIgogICAgIyBEZWZhdWx0IHZhbHVlIG9mIGNvZGUgc25pcHBldCB0YXJnZXRzIGlzIGFsd2F5cyBwcm92aWRlZCBieSB0aGUgYnVuZGxlCiAgICAjIEluY2x1ZGUgc3VwcG9ydGVkIHNuaXBwZXQgdGFyZ2V0cyBhcyBwZXIgbGluayBwcm92aWRlZCBiZWxvdyBpZiBtb2RpZmljYXRpb24gdG8gZGVmYXVsdCB2YWx1ZSBpcyByZXF1aXJlZAogICAgY29kZVNuaXBwZXRUYXJnZXRzOgogICAgICAtIGphdmFfb2todHRwCiAgICAgIC0gcHl0aG9uX3JlcXVlc3RzCmBgYAoKU3VwcG9ydGVkIGNvZGUgc25pcHBldCB0YXJnZXRzOiBbbGlua10oaHR0cHM6Ly9naXRodWIuY29tL3Ryb250bzIwL3N3YWdnZXItc25pcHBldC1nZW5lcmF0b3I/dGFiPXJlYWRtZS1vdi1maWxlI3RhcmdldHMpCgoqIEluIHlvdXIgQXBwbGljYXRpb24gY2xhc3M6CgpgYGBqYXZhCkBPdmVycmlkZQpwdWJsaWMgdm9pZCBpbml0aWFsaXplKEJvb3RzdHJhcDxZb3VyQ29uZmlndXJhdGlvbj4gYm9vdHN0cmFwKSB7CiAgICBib290c3RyYXAuYWRkQnVuZGxlKG5ldyBTd2FnZ2VyQnVuZGxlPFlvdXJDb25maWd1cmF0aW9uPigpIHsKICAgICAgICBAT3ZlcnJpZGUKICAgICAgICBwcm90ZWN0ZWQgU3dhZ2dlckJ1bmRsZUNvbmZpZ3VyYXRpb24gZ2V0U3dhZ2dlckJ1bmRsZUNvbmZpZ3VyYXRpb24oWW91ckNvbmZpZ3VyYXRpb24gY29uZmlndXJhdGlvbikgewogICAgICAgICAgICByZXR1cm4gY29uZmlndXJhdGlvbi5zd2FnZ2VyQnVuZGxlQ29uZmlndXJhdGlvbjsKICAgICAgICB9CiAgICB9KTsKfQpgYGAKCiogQXMgdXN1YWwsIGFkZCBTd2FnZ2VyIGFubm90YXRpb25zIHRvIHlvdXIgcmVzb3VyY2UgY2xhc3NlcyBhbmQgbWV0aG9kcwoKKiBPcGVuIGEgYnJvd3NlciBhbmQgaGl0IGBodHRwOi8vbG9jYWxob3N0Ojx5b3VyX3BvcnQ+L3N3YWdnZXJgCgoqIFRvIHNlZSBhbGwgdGhlIHByb3BlcnRpZXMgdGhhdCBjYW4gYmUgdXNlZCB0byBjdXN0b21pemUgU3dhZ2dlciBVSSBzZWUgW1N3YWdnZXJCdW5kbGVDb25maWd1cmF0aW9uLmphdmFdKGRyb3B3aXphcmQtc3dhZ2dlci9zcmMvbWFpbi9qYXZhL2luL3ZlY3RvcnByby9kcm9wd2l6YXJkL3N3YWdnZXIvU3dhZ2dlckJ1bmRsZUNvbmZpZ3VyYXRpb24uamF2YSkK readmeEtag: '"b5cc29d28713a32dc720b7fe3045366d21aaa368"' readmeLastModified: Thu, 27 Nov 2025 05:22:34 GMT repositoryId: 384696629 description: Dropwizard Swagger 2.x and OpenAPI 3.x Integration created: '2021-07-10T12:49:31Z' updated: '2026-01-25T10:54:40Z' language: Java archived: false stars: 7 watchers: 3 forks: 6 owner: Vect0rPro logo: https://avatars.githubusercontent.com/u/87179257?v=4 license: Apache-2.0 repoEtag: '"1162ed1f82b95a0a65a991381a7f0fa5a87ac77eabe8f3319b58788439591b06"' repoLastModified: Sun, 25 Jan 2026 10:54:40 GMT foundInMaster: true category: SDK id: 9d2d990aafc59b92f9c7c2b92c20f519 - source: openapi3 tags repository: https://github.com/gichure/e-procure v3: true id: 8f0cf8f8364853a7341aeec8fc9712e6 repositoryMetadata: base64Readme: >- WyFbQ0ldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL0dpY2h1cmUvZS1wcm9jdXJlL3RyZWUvbWFpbi5zdmc/c3R5bGU9c3ZnKV0oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvR2ljaHVyZS9lLXByb2N1cmUvdHJlZS9tYWluKQpbIVtDb2RlIEFuYWx5c2lzXShodHRwczovL2dpdGh1Yi5jb20vR2ljaHVyZS9lLXByb2N1cmUvYWN0aW9ucy93b3JrZmxvd3MvY29kZXFsLWFuYWx5c2lzLnltbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vR2ljaHVyZS9lLXByb2N1cmUvYWN0aW9ucy93b3JrZmxvd3MvY29kZXFsLWFuYWx5c2lzLnltbCkKWyFbUHVibGlzaGluZyBQYWNrYWdlXShodHRwczovL2dpdGh1Yi5jb20vR2ljaHVyZS9lLXByb2N1cmUvYWN0aW9ucy93b3JrZmxvd3MvbWF2ZW4tcHVibGlzaC55bWwvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKV0oaHR0cHM6Ly9naXRodWIuY29tL0dpY2h1cmUvZS1wcm9jdXJlL2FjdGlvbnMvd29ya2Zsb3dzL21hdmVuLXB1Ymxpc2gueW1sKQoKIyBlLXByb2N1cmUKRS1Qcm9jdXJlbWVudCBTeXN0ZW0gdG8gbWVldCB0aGUgbmVlZHMgb2YgYXV0b21hdGlvbiBpbiB0aGUgaW50ZXJuYWwgcHJvY3VyZW1lbnQgcHJvY2VzcyBvZiBhIGNvbXBhbnkuICAKIyMgT3ZlcnZpZXcKMS4gIFRoZSBhcHBsaWNhdGlvbiBoYXMgYmVlbiBkZXZlbG9wZWQgdXNpbmcgbWljcm9zZXJ2aWNlIGFwcHJvYWNoLiAgCjIuICBXZSB1c2UgZ2l0IG1vbm9yZXBvIHdoZXJlIGFsbCBtaWNyb3NlcnZpY2VzIGFyZSB0aGUgbW9kdWxlcyBpbiBhIHNpbmdsZSByZXNwb3NpdG9yeS4KMy4gIEVhY2ggbWljcm9zZXJ2aWNlIHByb2R1Y3Qtc2VydmljZSBhbmQgcmV2aWV3LXNlcnZpY2UgY2FuIGJlIGJ1aWx0IGFuZCBydW4gaW5kZXBlbmRlbnRseSB1c2luZyBNYXZlbi4KNC4gIEVhY2ggbWljcm9zZXJ2aWNlIGNhbiBidWlsZCwgdGVzdCwgcnVuLCBkb2NrZXJpemUsIGRlcGxveSAmIHJ1biBpbmRlcGVuZGVudGx5IHVzaW5nIERvY2tlciwgQ2lyY2xlQ0kgYW5kIEt1YmVybmV0ZXMuCgojIyBUb29scyAmIFRlY2hvbG9naWVzCi0gIFNwcmluZyBCb290IGZvciBCdXNpbmVzcyBTZXJ2aWNlcwotICBNaWNyb3NlcnZpY2VzIEFyY2hpdGVjdHVyZQotICBTcHJpbmcgQ2xvdWQgZm9yIE1TQQotICBNeVNRTCBmb3IgUkRCTVMKLSAgQ2FtdW5kYSBmb3IgQlBNTgotICBNb2NraXRvIGZvciBUZXN0aW5nCi0gIERvY2tlciBmb3IgUGFja2FnaW5nIGFuZCBEZXBsb3ltZW50Ci0gIEZseXdheSBmb3IgRGF0YWJhc2UgVmVyc2lvbmluZwotICBBcGFjaGUgS2Fma2EgZm9yIE1lc3NhZ2luZwotICBLZXljbG9hY2sgZm9yIEF1dGhlbnRpY2F0aW9uIGFuZCBBdXRob3JpemF0aW9uCi0gIENpcmNsZUNJIGZvciBDSS9DRAotICBTd2FnZ2VyIGZvciBBUEkgRG9jdW1lbnRhdGlvbgoKCiMjIE1vZHVsZXMKMS4gIFNldHRpbmdzLCBQYXJhbWV0ZXJzIGFuZCBHZW5lcmFsIFNldHVwcwoyLiAgRW1wbG95ZWVzL1N0YWZmIE1hbmFnZW1lbnQKMy4gIEl0ZW1zL0ludmVudG9yeSBNYW5hZ2VtZW50CjQuICBQdXJjaGFzZSBSZXF1aXN0aW9uCjUuICBQdXJjaGFzZSBPcmRlcgo2LiAgVmVuZG9yIE1hbmFnZW1lbnQKNy4gIEJpZCBNYW5hZ2VtZW50CjguICBJbnZvaWNlIE1hbmFnZW1lbnQKOS4gIFBheW1lbnQgUHJvY2Vzc2luZwoxMC4gIEJ1c2luZXNzIFByb2Nlc3MgTWFuYWdlbWVudChXb3JrZmxvdykKCiMjIFNlcnZpY2VzCjEuICBbY29uZmlndXJhdGlvbnMtc2VydmljZV0oY29uZmlndXJhdGlvbnMtc2VydmljZS9SRUFETUUubWQpCjIuICBbZGlzY292ZXJ5LXNlcnZpY2VdKGRpc2NvdmVyeS1zZXJ2aWNlL1JFQURNRS5tZCkKMy4gIFtnYXRld2F5LXNlcnZpY2VdKGdhdGV3YXktc2VydmljZS9SRUFETUUubWQpCjQuICBbc2V0dXBzLXNlcnZpY2VdKHNldHVwcy1zZXJ2aWNlL1JFQURNRS5tZCkKNS4gIFtlbXBsb3llZXMtc2VydmljZV0oZW1wbG95ZWVzLXNlcnZpY2UvUkVBRE1FLm1kKQo2LiAgW3dvcmtmbG93LXNlcnZpY2VdKHdvcmtmbG93LXNlcnZpY2UvUkVBRE1FLm1kKQo3LiAgW25vdGlmaWNhdGlvbnMtc2VydmljZV0obm90aWZpY2F0aW9ucy1zZXJ2aWNlL1JFQURNRS5tZCkKOC4gIFtpbnZlbnRvcnktc2VydmljZV0oaW52ZW50b3J5LXNlcnZpY2UvUkVBRE1FLm1kKQo5LiAgW3N1cHBsaWVycy1zZXJ2aWNlXShzdXBwbGllcnMtc2VydmljZS9SRUFETUUubWQpCjEwLiBbb3JkZXJzLXNlcnZpY2VdKG9yZGVycy1zZXJ2aWNlL1JFQURNRS5tZCkKMTEuIFtpbnZvaWNlcy1zZXJ2aWNlXShpbnZvaWNlcy1zZXJ2aWNlL1JFQURNRS5tZCkKMTIuIFtwYXltZW50cy1zZXJ2aWNlXShwYXltZW50cy1zZXJ2aWNlL1JFQURNRS5tZCkKMTMuIFthdXRoZW50aWNhdGlvbi1zZXJ2aWNlXShhdXRoZW50aWNhdGlvbi1zZXJ2aWNlL1JFQURNRS5tZCkKMTQuIFtmcm9udGVuZC1zZXJ2aWNlXSh1aS1zZXJ2aWNlL1JFQURNRS5tZCkKCiMjIFN0YXJ0aW5nIFVwCkNsb25lIHRoaXMgcmVwb3NpdG9yeS4gIApGcm9tIHRoZSByb290IGZvbGRlciwgcnVuIG12biBjbGVhbiBwYWNrYWdlICAKUnVuIGRvY2tlciBjb21wb3NlIHVwCgojIyBQb3N0bWFuIENvbGxlY3Rpb24KWW91IGNhbiBhY2Nlc3MgdGhlIHBvc3RtYW4gY29sbGVjdGlvbiBbaGVyZV0oKSAgCiMjIyBQcm9jZXNzIEZsb3cKIVtFLVByb2N1cmUgUHJvY2VzcyBGbG93XShyZXNvdXJjZXMvaW1hZ2VzL3Byb2Nlc3NfZmxvdy5qcGcpCgojIyMjIENvbnRyaWJ1dGlvbnMKLSAgW1BhdWwgR2ljaHVyZV0oaHR0cHM6Ly9saW5rZWRpbi5jb20vaW4vZ2ljaHVyZSkuIFlvdSBjYW4gcmVhY2ggb3V0IGF0IGUtcHJvY3VyZUBnaWNodXJlLm1lLmtlCgojIyMjIExpY2Vuc2UKW01JVF0oTElDRU5TRS5tZCkKCiMjIyMgSXNzdWVzIFJlcG9ydGluZwpJc3N1ZXMgYXJlIHRvIGJlIHJlcG9ydGVkIGluIFt0aGlzIGZvcm1hdF0oLmdpdGh1Yi9JU1NVRV9URU1QTEFURS9idWdfcmVwb3J0Lm1kKQoKIyMjIyBEaXNjbGFpbWVyClRoaXMgcHJvamVjdCBpcyBwdXJlbHkgZm9yIGxlYXJuaW5nIHB1cnBvc2VzIGFuZCB0aHVzIGl0IGlzIHN1YmplY3QgdG8gY2hhbmdlIHdpdGhvdXQgbm90aWNlLiAgClNob3VsZCB5b3UgbmVlZCBhIHN0YWJsZSB2ZXJzaW9uLHJlYWNoIG91dCBhdCBlLXByb2N1cmVAZ2ljaHVyZS5tZS5rZSBvciBmb3JrIHRoaXMgcmVwb3NpdG9yeS4gCiBJIGFsc28gZG8gbm90IG9mZmVyIHN1cHBvcnQgc2VydmljZXMgYXMgd2VsbC4gIAo= readmeEtag: '"b90823636a3bcc61d2da20ee332c6c83f66a60fe"' readmeLastModified: Sun, 15 Dec 2024 14:09:34 GMT repositoryId: 489591997 description: >- E-Procurement System to meet the needs of automation in the internal procurement process of a company. Built on Springboot using micro-services Archicture created: '2022-05-07T06:39:29Z' updated: '2026-01-25T08:10:06Z' language: Java archived: false stars: 12 watchers: 2 forks: 6 owner: Gichure logo: https://avatars.githubusercontent.com/u/6824034?v=4 license: MIT repoEtag: '"97c5ce5c3bc6347db959e2d1b43a4a12f346fc2f24db96f39db049977d410fac"' repoLastModified: Sun, 25 Jan 2026 08:10:06 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/elazar/phanua v3: true repositoryMetadata: base64Readme: >- # Phanua

[![PHP Version Support](https://img.shields.io/static/v1?label=php&message=%3E=%207.4.0&color=blue)](https://packagist.org/packages/elazar/phanua)
[![Packagist Version](https://img.shields.io/static/v1?label=packagist&message=0.1.0&color=blue)](https://packagist.org/packages/elazar/phanua)
[![Software License](https://img.shields.io/badge/license-MIT-blue.svg?style=flat-square)](LICENSE.md)
[![Buy Me a Cofee](https://img.shields.io/badge/buy%20me%20a%20coffee-donate-blue.svg)](https://www.buymeacoffee.com/DIkm1qe)
[![Patreon](https://img.shields.io/badge/patreon-donate-blue.svg)](https://patreon.com/matthewturland)

[OpenAPI 3](https://swagger.io/specification/) + [Jane](https://jane.readthedocs.io/en/latest/) + [Cycle ORM](https://cycle-orm.dev) = 🔥

Phanua builds Cycle ORM schemas from OpenAPI 3 component schemas.

Released under the [MIT License](https://en.wikipedia.org/wiki/MIT_License).

**WARNING: This project is in an alpha state of development and may be subject to changes that break backward compatibility. It may contain bugs, lack features or extension points, or be otherwise unsuitable for production use. User discretion is advised.**

## Requirements

* PHP 7.4+
* [PDO](https://www.php.net/pdo) and [driver](https://www.php.net/manual/en/pdo.drivers.php) for desired database

## Installation

Use [Composer](https://getcomposer.org/).

```sh
composer require elazar/phanua
```

## Usage

```php
<?php

/**
 * 1. Configure the Phanua service provider.
 *
 * The next section covers this step in more detail.
 */

$provider = new \Elazar\Phanua\Service\Provider;

// Configure $provider as needed here.

/**
 * 2. Use the Phanua service provider to create a Phanua schema builder.
 */

$schemaBuilder = $provider->getSchemaBuilder();

/**
 * 3. Use the Phanua schema builder to generate a Cycle schema or ORM instance.
 */

// To configure an existing ORM instance to use the generated schema:
$orm = new \Cycle\ORM\ORM(/* ... */);
// ...
$schema = $schemaBuilder->buildSchema();
$orm = $orm->withSchema($schema);

// To have Phanua create a new ORM instance with the generated schema:
$orm = $schemaBuilder->buildOrm();
```

## Configuration

For its most basic use, Phanua needs three parameters:

1. the path to an OpenAPI 3 specification file in JSON or YAML format;
2. a PHP namespace used by [generated](https://jane.readthedocs.io/en/latest/documentation/OpenAPI.html#generating) [Jane model class files](https://jane.readthedocs.io/en/latest/documentation/OpenAPI.html#using) (without the `Model` subnamespace added by Jane); and
3. a Cycle ORM database configuration.

You can pass these parameters to Phanua by using an instance of the Phanua service provider class `Elazar\Phanua\Service\Provider`.

You can provide the path and namespace provided using your [Jane configuration file](https://jane.readthedocs.io/en/latest/documentation/OpenAPI.html#configuration-file) or by passing them directly to the service provider instance.

```php
<?php

use Elazar\Phanua\Service\Provider;

// To load the Jane configuration file, provide the path
$provider = (new Provider)
    ->withJaneConfiguration('/path/to/.jane-openapi.php');

// If file is already loaded, provide the contained array
$janeConfig = require '.jane-openapi.php';
$provider = (new Provider)
    ->withJaneConfiguration($janeConfig);

// To pass the same values directly:
$provider = (new Provider)
    ->withOpenApiSpecPath('path/to/openapi.json')
    ->withNamespace('\\Foo\\Generated');
```

You can provide the database configuration by specifying the same array passed to `Spiral\Database\Config\DatabaseConfig`. See [related Cycle documentation](https://cycle-orm.dev/docs/basic-connect#instantiate-dbal) for an example of this array.

```php
<?php

$databaseConfig = [ /* ... */ ];
$provider = (new \Elazar\Phanua\Service\Provider)
    ->withDatabaseConfig($databaseConfig);
```

Providing the database configuration in other ways requires an explanation of how Phanua handles its dependencies.

### Overriding Dependencies

To overriding a dependency of Phanua, you must provide a dependency injection container that includes that dependency.

Phanua can use any container that implements the [PSR-11](https://www.php-fig.org/psr/psr-11/) standard. An example of such a container is [Pimple](https://packagist.org/packages/pimple/pimple), which Phanua uses internally. Pimple is the recommended container to use if you aren't already using a different one.

```php
<?php

// Create a Pimple or PSR-11 container instance
$container = new \Pimple\Container;

// Then configure Phanua to use it
$provider = (new \Elazar\Phanua\Service\Provider)
    ->withDelegateContainer($container);
```

Phanua expects this container to use [fully-qualified class names](https://www.php.net/manual/en/language.namespaces.rules.php) as entry identifiers. Below are examples of alternate methods of configuring Pimple to supply the database configuration.

```php
<?php

$container = new \Pimple\Container;

// Compared to the earlier example of passing the database configuration to
// Phanua as an array, this is the next easiest / most low-level method of doing
// so if you're not already using a container in your application.
use Spiral\Database\Config\DatabaseConfig;
$container[DatabaseConfig::class] = fn() => new DatabaseConfig(
    // Pass the same array passed to Provider->withDatabaseConfig() in the earlier
    // example here.
);

// If you're already using a container and it includes a configured instance of
// the DatabaseManager class used by Cycle ORM, you can specify that instead.
use Spiral\Database\Config\DatabaseManager;
$container[DatabaseManager::class] = fn() => new DatabaseManager(
    new DatabaseConfig(/* ... */)
);

// Or you can specify an implementation of Cycle\ORM\FactoryInterface, such as
// the Cycle\ORM\Factory class.
use Spiral\Database\Config\{Factory, FactoryInterface};
$container[FactoryInterface::class] = fn() => new Factory(
    new DatabaseManager(/* ... */)
);

// Or you can specify an instance of Cycle\ORM\ORM.
use Cycle\ORM\ORM;
$container[ORM::class] = fn() => new ORM(
    new Factory(/* ... */)
);
```

If you're already using Pimple and want Phanua to use dependencies you've registered in your container, you can register your configured Phanua service provider instance with your container as a [provider](https://github.com/silexphp/Pimple#extending-a-container).

```php
<?php

$provider = new \Elazar\Phanua\Service\Provider;

// Configure $provider as needed here.

$container = new \Pimple\Container;
$container->register($provider);
```

Once you've configured the Phanua service provider with the necessary parameters, it's possible to use the container it builds independently.

```php
<?php

$provider = new \Elazar\Phanua\Service\Provider;

// Configure $provider as needed here.

// Pimple
$container = $provider->getContainer();

// PSR-11
$psrContainer = $provider->getPsrContainer();

// To get the schema builder:

use Elazar\Phanua\Schema\Builder;

// Pimple
$schemaBuilder = $container[Builder::class];

// PSR-11
$schemaBuilder = $container->get(Builder::class);
```

### Role Resolver

Cycle ORM uses the term ["role"](https://cycle-orm.dev/docs/advanced-schema#schema-properties) to refer to a unique name for an entity.

Phanua defines the interface `Elazar\Phanua\Entity\RoleResolverInterface` for classes used to determine the role for a given entity.

This interface contains a single method `getRole()` which receives a single parameter: a `string` containing the name of the OpenAPI component corresponding to the entity.

`getRole()` must return a `string` containing the determined role for the entity.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Entity\RoleResolver`. Its `getRole()` method returns the OpenAPI component name it receives so that the component name and role are the same.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Entity\RoleResolverInterface` and have it resolve to an instance of a class that implements the interface.

### Class Resolver

Entities in Cycle ORM schemas have corresponding [classes](https://cycle-orm.dev/docs/basic-crud#create-entity).

Phanua defines the interface `Elazar\Phanua\Entity\ClassResolverInterface` for classes used to determine the class for a given entity.

This interface contains a single method, `getClass()`, which receives a single parameter: a `string` containing the name of the OpenAPI component corresponding to the entity.

`getClass()` must return a `string` containing the [fully-qualified name](https://www.php.net/manual/en/language.namespaces.rules.php) of the class for the entity.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Entity\ClassResolver`. Its `getClass()` method returns a class name constructed with the namespace used by Jane from the Phanua service provider configuration and the OpenAPI component name it receives.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Entity\ClassResolverInterface` and have it resolve to an instance of a class that implements the interface.

### Table Resolver

Each Cycle ORM entity has a corresponding [table](https://cycle-orm.dev/docs/advanced-declaration#to-start).

Phanua defines the interface `Elazar\Phanua\Entity\TableResolverInterface` for classes used to determine the table associated with a given entity.

This interface contains a single method, `getTable()`, which receives three parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity;
2. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the component schema; and
3. an instance of `Cycle\Schema\Definition\Entity` representing the entity.

`getTable()` must return a `string` containing the name of the table for the entity.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Entity\TableResolver`. Its `getTable()` method returns the entity role, as determined by the instance of `Elazar\Phanua\Entity\RoleResolverInterface` in use, so that the role and table name are the same. See the ["Role Resolver"](#role-resolver) section above for further details.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Entity\TableResolverInterface` and have it resolve to an instance of a class that implements the interface.

### Name Resolver

Entities in Cycle ORM contain [fields](https://cycle-orm.dev/docs/advanced-dynamic-schema#using-schema-builder) that have names.

Phanua defines the interface `Elazar\Phanua\Field\NameResolverInterface` for classes used to determine the name of a field.

This interface contains a single method, `getName()`, which receives three parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity containing the field;
2. a `string` containing the name of the component property corresponding to the field; and
3. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the property schema.

`getName()` must return a `string` containing the name to assign to the field.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Field\NameResolver`. Its `getName()` method returns the OpenAPI property name it receives so that the property name and field name are the same.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Field\NameResolverInterface` and have it resolve to an instance of a class that implements the interface.

### Column Resolver

Cycle ORM entity fields have corresponding table [columns](https://cycle-orm.dev/docs/advanced-declaration#columns-and-abstract-types).

Phanua defines the interface `Elazar\Phanua\Field\ColumnResolverInterface` for classes used to determine the name of the column for a field.

This interface contains a single method, `getColumn()`, which receives three parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity containing the field;
2. a `string` containing the name of the component property corresponding to the field; and
3. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the property schema.

`getColumn()` must return a `string` containing the name to assign to the column.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Field\ColumnResolver`. Its `getColumn()` method returns the OpenAPI property name it receives so that the property name and column name are the same.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Field\ColumnResolverInterface` and have it resolve to an instance of a class that implements the interface.

### Type Resolver

Each Cycle ORM entity field has a corresponding [abstract type](https://cycle-orm.dev/docs/advanced-declaration#columns-and-abstract-types-abstract-types).

Phanua defines the interface `Elazar\Phanua\Field\TypeResolverInterface` for classes used to determine the type for a field.

This interface contains a single method, `getType()`, which receives three parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity containing the field;
2. a `string` containing the name of the component property corresponding to the field; and
3. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the property schema.

`getType()` may return a `string` containing the type to assign to the column or `null` if it's unable to resolve the column to a type.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Field\TypeResolver`. Its `getType()` method returns a type it derives from the [property type and format](https://swagger.io/specification/) as well as [validation constraints](https://swagger.io/specification/#schemaObject) on the property value (e.g. `minimum`, `exclusiveMinimum`, `maximum`, or `exclusiveMaximum` for numeric values, `minLength` and `maxLength` for string / text values). Lacking these constraints, `getType()` returns the largest available type to allow for maximal possible values.

**NOTE: This implementation does not handle nested objects, arrays, or references to other component schemas. This may change in the future.**

In the event of being unable to resolve to a more specific type, `getType()` supports falling back to a specified type. To reduce configuration needed for common use cases, the default value for this fallback type is `'string'`.

You may want change this fallback type if, for example, you want to compose `TypeResolver` with your own resolver implementation to resolve types that `TypeResolver` doesn't cover. In that case, you would set the fallback type to `null` so that your resolver knows when `TypeResolver` can't resolve a given type. Below is an example of overriding the fallback type using a Pimple container.

```php
<?php

use Elazar\Phanua\Field\TypeResolver;
$container[TypeResolver::class] = new TypeResolver(
    null // fallback value goes here
);
```

To override the default type resolver implementation with your own, add an entry to your container using the key `Elazar\Phanua\Field\TypeResolverInterface` and have it resolve to an instance of a class that implements the interface.

If you want to use the default type resolver, your own resolver, or another resolver together, you can compose them using `Elazar\Phanua\Field\CompositeFieldResolver`, which will use resolvers in turn until one returns a type.

```php
<?php

use Elazar\Phanua\Field\CompositeTypeResolver;
use Elazar\Phanua\Field\TypeResolver;
use Elazar\Phanua\Field\TypeResolverInterface;
use My\CustomTypeResolver;

// If you want your resolver to be tried first:
$container[TypeResolverInterface::class] = new CompositeTypeResolver(
    new CustomTypeResolver(),
    new TypeResolver(),
);

// If you want the default resolver to be tried first:
$container[TypeResolverInterface::class] = new CompositeTypeResolver(
    new TypeResolver(
        null // Required for CompositeTypeResolver to fall back to your resolver
    ),
    new CustomTypeResolver(),
);
```

### Primary Resolver

Each table must have a [primary index](https://cycle-orm.dev/docs/advanced-declaration#primary-index) on one or more columns for a compiled Cycle ORM schema to include it.

This circumstance is a limitation of the compiler that handles [compiling entities from the registry](https://cycle-orm.dev/docs/advanced-schema-builder#manually-define-entity) into a schema: it [skips entities without a primary index](https://github.com/cycle/schema-builder/blob/v1.2.0/src/Compiler.php#L68).

To draw attention to this behavior, Phanua throws an exception if primary index resolution fails for a given entity. To avoid this, you must explicitly exclude any entities in your OpenAPI specification that you do not want to include in the generated Cycle ORM schema. The [next section](#entity-resolver) discusses this further.

Phanua defines the interface `Elazar\Phanua\Field\PrimaryResolverInterface` for classes used to determine whether a given field should be part of the primary index of the table containing its respective column.

This interface contains a single method, `isPrimary()`, which receives three parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity containing the field;
2. a `string` containing the name of the component property corresponding to the field; and
3. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the property schema.

`isPrimary()` must return a `boolean` value where a value of `true` indicates that the given field is part of the primary index for the associated table and a value of `false` indicates that it's not.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Field\PrimaryResolver`. Its `isPrimary()` method returns `true` if the given property name is `id` or `false` otherwise. This class assumes that the property and column names are the same, which is the case when using the default implementation of `Elazar\Phanua\Field\ColumnResolverInterface` detailed in the ["Column Resolver"](#column-resolver) section above.

If your property and column names are not consistent with each other, or if any of your tables have a primary index that includes more than one column, create your own implementation of this interface to suit the needs of your database.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Field\PrimaryResolverInterface` and have it resolve to an instance of a class that implements the interface.

### Entity Resolver

Phanua represents the process of converting an OpenAPI component to a Cycle ORM entity with the interface `Elazar\Phanua\Entity\EntityResolverInterface`.

This interface contains a single method, `getEntity()`, which receives two parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity; and
2. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the component schema.

`getEntity()` must return a populated instance of `Cycle\Schema\Definition\Entity` representing the entity corresponding to the given component or `null` if it fails to resolve the component to an entity.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Entity\EntityResolver`. It composes an implementation of each of `Elazar\Phanua\Entity\RoleResolverInterface` and `Elazar\Phanua\Entity\ClassResolverInterface`, which it uses to determine the entity [role](#role-resolver) and [class](#class-resolver) respectively.

It also provides for the exclusion of specific components from having an entity in the generated schema, such as those components where the corresponding entity would not have a [primary index](#primary-resolver). The easiest way to exclude one or more components is by using the service provider: by default, it handles injecting a callback to filter components into `EntityResolver`.

```php
<?php

use Jane\Component\OpenApi3\JsonSchema\Model\Schema;

$componentFilter = fn (
    string $componentName,
    Schema $componentSchema
): bool =>
    !in_array($componentName, [
        // These components are excluded from the schema
        'component1',
        'component2',
        // ...
    ]);

$provider = (new \Elazar\Phanua\Service\Provider)
    ->withComponentFilter($componentFilter);
```

You can also exclude components by [overriding](#overriding-dependencies) the `EntityResolver` instance used by default with one that includes a callback to filter entities in the `$filterCallback` constructor parameter of `EntityResolver` or does something similar for your own entity resolver implementation. The example below does this using a Pimple container.

```php
<?php

use Elazar\Phanua\{
    Entity\ClassResolverInterface,
    Entity\EntityResolver,
    Entity\EntityResolverInterface,
    Entity\RoleResolverInterface,
    Service\Provider
};
use Jane\Component\OpenApi3\JsonSchema\Model\Schema;

$provider = new Provider;

/**
 * 1. Get a container from the Phanua service provider to override some of the
 *    default dependencies it defines.
 */

// Pimple
$phanuaContainer = $provider->getContainer();

// PSR-11
$phanuaContainer = $provider->getPsrContainer();

/**
 * 2. In your own container, use the key EntityResolverInterface::class and
 *    assign it an instance of EntityResolver or your own entity resolver
 *    implementation, injecting any default Phanua dependencies that you need.
 *    Below is an example of doing this using Pimple containers.
 */

$yourContainer = new \Pimple\Container;
$yourContainer[EntityResolverInterface::class] = fn() => new EntityResolver(
    $phanuaContainer[RoleResolverInterface::class],
    $phanuaContainer[ClassResolverInterface::class],
    function (
        string $componentName,
        string $propertyName,
        Schema $propertySchema
    ): bool {
        /* return TRUE to include or FALSE to exclude component */
    }
);

/**
 * 3. Set your container as the delegate container in the Phanua service
 *    provider.
 */

$provider = $provider->withDelegateContainer($yourContainer);

/**
 * 4. Get the schema builder as normal. It will use the custom dependencies
 *    you've defined.
 */

$schemaBuilder = $provider->getSchemaBuilder();
```

It's entirely optional to have your entity resolver use other Phanua dependencies, as the default Phanua implementation does; your entity resolver can function in whatever way you like.

Another approach to consider is having your entity resolver implementation compose the default implementation that Phanua uses. By doing so, you can use the default implementation for entities that are compatible with it and your own implementation for those that are not. See the example of this below.

```php
<?php

/**
 * 1. Define your entity resolver implementation.
 */

use Cycle\Schema\Definition\Entity;
use Elazar\Phanua\Entity\EntityResolverInterface;
use Jane\Component\OpenApi3\JsonSchema\Model\Schema;

class YourEntityResolver implements EntityResolverInterface
{
    private EntityResolverInterface $entityResolver;

    // Have the constructor accept another resolver implementation as a
    // parameter.

    public function __construct(
        EntityResolverInterface $entityResolver,
        /* ... */
    ) {
        $this->entityResolver = $entityResolver;
        /* ... */
    }

    // Then, in the implementation of the interface method...

    public function getEntity(
        string $componentName,
        Schema $componentSchema
    ): ?Entity
    {
        // $entity = ...

        // ... if your implementation can't resolve an entity...
        if ($entity === null) {

            // ... then have it defer to the injected resolver.
            $entity = $this->entityResolver->getEntity(
                $componentName,
                $componentSchema
            );

        }

        return $entity;
    }
}

/**
 * 2. Inject your implementation to have Phanua use it.
 */

// Get the Phanua container and configure your container as normal...
$phanuaContainer = $provider->getContainer();
$yourContainer = new \Pimple\Container;
$yourContainer[EntityResolverInterface::class] = fn() => new YourEntityResolver(

    // ... then inject the entity resolver implementation from the Phanua
    // container into your own implementation.
    $phanuaContainer[EntityResolverInterface::class]

);
```

### Field Resolver

Phanua includes a resolver for fields as it does for entities, represented by the `Elazar\Phanua\Field\FieldResolverInterface`.

This interface contains a single method, `getField()`, which receives three parameters:

1. a `string` containing the name of the OpenAPI component corresponding to the entity containing the field;
2. a `string` containing the name of the component property corresponding to the field; and
3. an instance of `Jane\Component\OpenApi3\JsonSchema\Model\Schema` representing the property schema.

`getField()` must return a populated instance of `Cycle\Schema\Definition\Field` representing the field corresponding to the given property or `null` if it fails to resolve the component to a field.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Field\FieldResolver`. It composes an implementation of each of `Elazar\Phanua\Field\ColumnResolverInterface`, `Elazar\Phanua\Field\TypeResolverInterface`, and `Elazar\Phanua\Field\PrimaryResolverInterface`, which it uses to determine the field [column](#column-resolver), [type](#type-resolver), and presence in the [primary index](#primary-resolver) respectively.

It also handles setting other options for the field, such as its default value and whether it's nullable based on corresponding [`default`](https://swagger.io/specification/#properties) and [`nullable`](https://swagger.io/specification/#fixed-fields-20) values from the given property schema.

`FieldResolver` also provides for the exclusion of specific properties from having a field in the generated schema. The easiest way to exclude one or more properties is by using the service provider: by default, it handles injecting a callback to filte properties into `FieldResolver`.

```php
<?php

use Jane\Component\OpenApi3\JsonSchema\Model\Schema;

$propertyFilter = fn (
    string $componentName,
    string $propertyName,
    Schema $propertySchema
): bool =>
    // Exclude all properties with this name
    $property !== 'propertyName'

    // Exclude the property only in the specified component
    && "$component.$property" !== 'componentName.propertyName';

$provider = (new \Elazar\Phanua\Service\Provider)
    ->withPropertyFilter($propertyFilter);
```

You can also exclude properties by [overriding](#overriding-dependencies) the `FieldResolver` instance used by default with one that includes a callback to filter properties in the `$filterCallback` constructor parameter of `FieldResolver`. The example below does this using a Pimple container.

```php
<?php

use Elazar\Phanua\{
    Field\ColumnResolverInterface,
    Field\FieldResolver,
    Field\FieldResolverInterface,
    Field\PrimaryResolverInterface,
    Field\TypeResolverInterface,
    Service\Provider
};
use Jane\Component\OpenApi3\JsonSchema\Model\Schema;

$provider = new Provider;

/**
 * 1. Get a container from the Phanua service provider to override some of the
 *    default dependencies it defines.
 */

// Pimple
$phanuaContainer = $provider->getContainer();

// PSR-11
$phanuaContainer = $provider->getPsrContainer();

/**
 * 2. In your own container, use the key FieldResolverInterface::class and
 *    assign it an instance of your entity resolver implementation, injecting
 *    any default Phanua dependencies that you need. Below is an example of
 *    doing this using Pimple containers.
 */

$yourContainer = new \Pimple\Container;
$yourContainer[FieldResolverInterface::class] = fn() => new FieldResolver(
    $phanuaContainer[ColumnResolverInterface::class],
    $phanuaContainer[PrimaryResolverInterface::class],
    $phanuaContainer[TypeResolverInterface::class],
    function (
        string $componentName,
        string $propertyName,
        Schema $propertySchema
    ): bool {
        /* return TRUE to include or FALSE to exclude property */
    }
);

/**
 * 3. Set your container as the delegate container in the Phanua service
 *    provider.
 */

$provider = $provider->withDelegateContainer($yourContainer);

/**
 * 4. Get the schema builder as normal. It will use the custom dependencies
 *    you've defined.
 */

$schemaBuilder = $provider->getSchemaBuilder();
```

It's entirely optional to have your field resolver use other Phanua dependencies, as the default Phanua implementation does; your field resolver can function in whatever way you like.

Another approach to consider is having your field resolver implementation compose the default implementation that Phanua uses. By doing so, you can use the default implementation for fields that are compatible with it and your own implementation for those that are not. See the example of this below.

```php
<?php

/**
 * 1. Define your field resolver implementation.
 */

use Cycle\Schema\Definition\Field;
use Elazar\Phanua\Field\FieldResolverInterface;
use Jane\Component\OpenApi3\JsonSchema\Model\Schema;

class YourFieldResolver implements FieldResolverInterface
{
    private FieldResolverInterface $fieldResolver;

    // Have the constructor accept another resolver implementation as a
    // parameter.

    public function __construct(
        FieldResolverInterface $fieldResolver,
        /* ... */
    ) {
        $this->fieldResolver = $fieldResolver;
        /* ... */
    }

    // Then, in the implementation of the interface method...

    public function getField(
        string $componentName,
        string $propertyName,
        Schema $propertySchema
    ): ?Field
    {
        // $field = ...

        // ... if your implementation can't resolve an entity...
        if ($field === null) {

            // ... then have it defer to the injected resolver.
            $field = $this->fieldResolver->getField(
                $componentName,
                $propertyName,
                $propertySchema
            );

        }

        return $field;
    }
}

/**
 * 2. Inject your implementation to have Phanua use it.
 */

// Get the Phanua container and configure your container as normal...
$phanuaContainer = $provider->getContainer();
$yourContainer = new \Pimple\Container;
$yourContainer[FieldResolverInterface::class] = fn() => new YourFieldResolver(

    // ... then inject the field resolver implementation from the Phanua
    // container into your own implementation.
    $phanuaContainer[FieldResolverInterface::class]

);
```

### Logger

Phanua supports any [PSR-3 logger](https://www.php-fig.org/psr/psr-3/). By default, it uses `Psr\Logger\NullLogger`, which discards any logged entries. To store these entries, you must override the default logger with one that does something with them. Below is an example of using the Pimple container to override the default logger with one from the [Monolog](https://github.com/Seldaek/monolog/) library.

```php
<?php

use Elazar\Phanua\Service\Provider;
use Monolog\Logger;
use Pimple\Container;
use Psr\Log\LoggerInterface;

$yourContainer = new Container;
$yourContainer[LoggerInterface::class] = function () {
    $logger = new Logger;
    // Configure $logger as needed here
    return $logger;
};

$provider = (new Provider)
    ->withDelegateContainer($yourContainer);
```

### Specification Loader

To convert an OpenAPI specification to a Cycle ORM schema, Phanua must first load and parse that specification.

Phanua defines the interface `Elazar\Phanua\Schema\SpecLoaderInterface` for classes used to load and parse OpenAPI specification files.

This interface contains a single method `load()` which receives a single parameter: a `string` containing the path to an OpenAPI specification file.

`load()` must return an instance of `Jane\Component\OpenApi3\JsonSchema\Model\OpenApi` containing the parsed specification or throw an instance of `Elazar\Phanua\Schema\Exception` if loading or parsing the specification fails.

The implementation of this interface that Phanua uses by default is `Elazar\Phanua\Schema\SpecLoader`. It handles logging events related to loading and parsing the specification.

To override the default implementation, add an entry to your container using the key `Elazar\Phanua\Schema\SpecLoaderInterface` and have it resolve to an instance of a class that implements the interface.

### Registry

As Phanua converts OpenAPI components to Cycle ORM entities, it uses an instance of `Cycle\Schema\Registry` to register the entities and link them to tables. This registry is ultimately used to compile the entities into an instance of `Cycle\ORM\Schema`.

By default, Phanua uses an empty `Registry` instance configured for your database. You want to use a different instance if, for example, you're compiling entities generated from other sources into the same schema with entities generated by Phanua.

In such situations, add an entry to your container using the key `Cycle\Schema\Registry` and have it resolve to your desired `Registry` instance to have Phanua use it. Note that you can use the Phanua container to either use a modified version of its default registry instance or to supply your registry's required implementation of `Spiral\Database\DatabaseProviderInterface` using the same instance of `Spiral\Database\DatabaseManager` that Phanua does. Below is an example of doing each with a Pimple container.

```php
<?php
use Cycle\Schema\Registry;
use Phanua\Service\Provider;
use Pimple\Container;
use Spiral\Database\DatabaseProviderInterface;

$provider = new Provider;

// If you're using Phanua\Service\Provider as a Pimple provider:
$yourContainer = new Container;
$yourContainer->register($provider);
$yourContainer->extend(Registry::class, function (Registry $registry) {
    // Change $registry as needed here
});

// If you're not using Phanua\Service\Provider as a Pimple provider:
$phanuaContainer = $provider->getContainer();
$yourContainer = new Container;
$yourContainer[Registry::class] = function () use ($phanuaContainer) {
    $databaseProvider = $phanuaContainer[DatabaseProviderInterface::class];
    $registry = new Registry($databaseProvider);
    // Configure $registry here as needed
    return $registry;
};
$phanuaContainer[Registry::class] = fn() => $yourContainer[Registry::class];
```

### Compiler Configuration

Phanua uses an instance of `Cycle\Schema\Compiler` to compile a registry of entities into a schema. This compiler can use [custom generators](https://cycle-orm.dev/docs/advanced-schema-builder#custom-generators) and [defaults](https://cycle-orm.dev/docs/advanced-default-classes). By default, Phanua uses default values for each of these (i.e. empty arrays).

If you want to use custom values for either of these, Phanua provides `Elazar\Phanua\Schema\CompilerConfiguration` to override them. Add an entry for this class to your container with a configured instance. Below is an example of doing this with a Pimple container.

```php
<?php

use Elazar\Phanua\Schema\CompilerConfiguration;
use Elazar\Phanua\Service\Provider;
use Pimple\Container;

$yourContainer = new Container;
$yourContainer[CompilerConfiguration::class] = fn() => (new CompilerConfiguration)
    ->withGenerators(/* ... */)
    ->withDefaults(/* ... */);

$provider = (new Provider)
    ->withDelegateContainer($yourContainer);
```

## FAQ

### Why did you build this?

I was already using OpenAPI and Jane in some projects and wanted to use Cycle ORM as well. I was already using model classes generated by Jane and didn't want to have to manually annotate or otherwise change them to be usable with Cycle. I built this so I wouldn't have to.

### Why the name "Phanua?"

It's taken from the latter half of the Swahili word *kufafanua*, which means "to define," and adapted to replace the "f" with "ph" as is common with PHP project names.

### How do you pronounce "Phanua?"

Fah-NOO-ah.
 readmeEtag: '"ac01a09cb96b5e6a046667396cf5bfe5ddf28755"' readmeLastModified: Sun, 01 Aug 2021 00:10:52 GMT repositoryId: 384845118 description: Builds Cycle ORM schemas from OpenAPI 3 component schemas created: '2021-07-11T02:35:08Z' updated: '2022-12-26T17:17:05Z' language: PHP archived: true stars: 5 watchers: 1 forks: 0 owner: elazar logo: https://avatars.githubusercontent.com/u/15487?v=4 license: MIT repoEtag: '"e419f7ed3db7bcb99dc14d4141064b30957d300301fa15d410a7f0292303e5c8"' repoLastModified: Mon, 26 Dec 2022 17:17:05 GMT foundInMaster: true id: d9fd84a229f29b78ac55c53e32cc1f38 - source: openapi3 tags repository: https://github.com/cossteam/cossim-server v3: true id: 33aedab32d669fdd29a2a08114f19cba repositoryMetadata: base64Readme: >- Y29zcy1zZXJ2ZXIKPT09PT09PT09PT09PT0KYGNvc3NpbS9jb3NzLXNlcnZlcmAg5piv55So5LqO5pSv5oyBY29zcy1jbGllbnTnmoRBUEnmnI3liqHjgIIKCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoqIFvnibnmgKddKCPnibnmgKcpCiogW+acjeWKoV0oI+acjeWKoSkKKiBb57uT5p6EXSgj57uT5p6EKQoqIFvlv6vpgJ/lkK/liqhdKCPlv6vpgJ/lkK/liqgpCiogW+mFjee9rl0oI+mFjee9rikKKiBb5paH5qGjXSgj5paH5qGjKQoqIFvmm7TlpJpdKCPmm7TlpJopCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCiMjIOeJueaApwoqIOmrmOaAp+iDvQoqIOe6r0dvbGFuZ+WunueOsAoqIOWIhuW4g+W8j+acjeWKoeaetuaehO+8iOacjeWKoemXtOmAmuS/oeS9v+eUqGdycGPvvIkKKiBERETpobnnm67mnrbmnoQKKiDmlK/mjIHliqjmgIHmiannvKnmnI3liqHlrp7kvovkuI7otJ/ovb3lnYfooaEKKiDmnI3liqHlrp7kvovmlK/mjIFwcHJvZuiwg+ivleWSjHByb21ldGhldXMgbWV0cmljcyjlj6/mjqXlhaVwcm9tZXRoZXVz5ZKMZ3JhZmFuYeWunueOsOWPr+inhuWMlueahOacjeWKoeebkeaOpykKKiDmlK/mjIHmnI3liqHliqjmgIHms6jlhozlj5HnjrDlkozphY3nva7kuK3lv4PvvIjln7rkuo5jb25zdWzvvIkKKiDlrp7ml7Yr56a757q/5o6o6YCB5pSv5oyB77yIU29ja2V0SU8rUmFiYml0TVHvvIkKKiDph4fnlKjpq5jmgKfog71BUEnnvZHlhbPvvIhhcGlzaXjvvIkKKiDmnI3liqHnlJ/lkb3lkajmnJ/lrp7njrDvvIhtYW5hZ2XvvIkKKiBPU1Plr7nosaHlrZjlgqjvvIhtaW5pb++8iQoqIOS8oOi+k+WKoOWvhu+8iG9wZW5wZ3DvvIkKCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQojIyDmnI3liqEKPiBjb3NzLXNlcnZlciDljIXlkKvku6XkuIvmnI3liqHvvJoKCiogdXNlcjog8J+RpOeUqOaIt+acjeWKoe+8jOWkhOeQhueUqOaIt+azqOWGjOOAgeeZu+W9leetieWKn+iDveOAggoqIGdyb3VwOiDwn5Gs576k57uE5pyN5Yqh77yM566h55CG55So5oi35LmL6Ze055qE576k57uE5YWz57O744CCCiogcHVzaDog4pyI77iP5raI5oGv5o6o6YCB5pyN5Yqh77yM6LSf6LSj5a6e5pe25raI5oGv5o6o6YCB5Yqf6IO944CCCiogbXNnOiDwn5Op5raI5oGv5pyN5Yqh77yM5aSE55CG55So5oi36Ze05raI5oGv55qE5pS25Y+R5Yqf6IO944CCCiogbGl2ZTog4piO77iP6YCa6K+d5pyN5Yqh77yM5pSv5oyB6K+t6Z+z6YCa6K+d5ZKM6KeG6aKR6YCa6K+d5Yqf6IO944CCCiogc3RvcmFnZTog8J+Xg+WtmOWCqOacjeWKoe+8jOi0n+i0o+aWh+S7tuWtmOWCqOWSjOeuoeeQhuOAggoqIHJlbGF0aW9uOiDwn6ea4oCN5YWz57O75pyN5Yqh77yM566h55CG55So5oi35LmL6Ze055qE56S+5Lqk5YWz57O744CCCiogYWRtaW46IPCfkbfigI3nrqHnkIblkZjmnI3liqHvvIznlKjkuo7nrqHnkIbns7vnu5/nlKjmiLflkozmnYPpmZDjgIIKCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQojIyDnu5PmnoQKPumhueebruebuOWFs+ebruW9leS7i+e7jeWmguS4i++8mgpgYGAK4pSc4pSA4pSAIGJ1aWxkICPnlJ/miJBwZ3Dlhaznp4HpkqUK4pSc4pSA4pSAIGNtZCAj5ZCv5Yqo5paH5Lu2CuKUnOKUgOKUgCBkZXBsb3kgI+mFjee9rgrilJzilIDilIAgZG9jcyAj5o6l5Y+j5paH5qGjCuKUnOKUgOKUgCBpbnRlcm5hbCAj5pyN5Yqh5Luj56CBCuKUlOKUgOKUgCBwa2cgI+acjeWKoemcgOimgeeUqOWIsOeahOWFrOWFseW3peWFt+WMhQpgYGAKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCgojIyDlv6vpgJ/lkK/liqgKPiDku6XkuIvkuKTnp43mlrnlvI/pg73pnIDopoHlronoo4Vkb2NrZXItY29tcG9zZeS4jmRvY2tlcgo+IAo+IOKaoO+4j+ivt+WFiOWuieijhVtjb3NzLWNsaeW3peWFt10oaHR0cHM6Ly9naXRodWIuY29tL2Nvc3NpbS9jb3NzLWNsaS9yZWxlYXNlcykKPiAKPiDmnInlhbNjbGnlt6Xlhbfmm7TlpJrnmoTkv6Hmga/or7db5Y+C6ZiFXShodHRwczovL2dpdGh1Yi5jb20vY29zc2ltL2Nvc3MtY2xpKQojIyMg5rqQ56CB5ZCv5YqoCmBgYAoxLuaLieWPluacgOaWsOS7o+eggQpnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2Nvc3NpbS9jb3NzLXNlcnZlcgptdiAuL2Nvc3MtY2xpLXh4eCBjb3NzLXNlcnZlci9jb3NzLWNsaQpjZCBjb3NzLXNlcnZlcgoKMi7nlJ/miJDphY3nva7mlofku7YKY2htb2QgYSt4IGNvc3MtY2xpCmNvc3MtY2xpIGdlbiAtLXBhdGggLi9kZXBsb3kvZG9ja2VyLwoKMy7lkK/liqjlv4XpnIDkuK3pl7Tku7YKZG9ja2VyLWNvbXBvc2UgLWYgZGVwbG95L2RvY2tlci9kb2NrZXItY29tcG9zZS5iYXNlLnlhbWwgdXAgLWQKCjQu5ZCv5Yqo5pyN5YqhCui/memHjOWPquaLv3VzZXLkuL7kvosKZ28gcnVuIC4vY21kL3VzZXIvbWFpbi5nbyAtY29uZmlnIGRlcGxveS9kb2NrZXIvY29uZmlnL3NlcnZpY2UvdXNlci55YW1sCmBgYAojIyMgY2xp5bel5YW35ZCv5YqoCj4g4pqg77iP6K+35YWI5a6J6KOFW2Nvc3MtY2xp5bel5YW3XShodHRwczovL2dpdGh1Yi5jb20vY29zc2ltL2Nvc3MtY2xpL3JlbGVhc2VzKQo+IAo+IOKaoO+4j+ivt+azqOaEj++8mmNsaeW3peWFt+WQr+WKqOaXtu+8jOS8muiHquWKqOeUn+aIkOmFjee9ruaWh+S7tuWcqOW9k+WJjeebruW9leS4i++8jOWmguaciemcgOimgeivt+WIm+W7uuaWh+S7tuWkuQpgYGAKbWtkaXIgY29zcy1zZXJ2ZXIKY2QgY29zcy1zZXJ2ZXIKY29zcy1jbGkgc3RhcnQKYGBgCi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKIyMg6YWN572uCioqY29uZmlnL2NvbW1vbioqCj4g5a2Y5pS+5YWs5YWx5Lit6Ze05Lu26YWN572u5paH5Lu2CgoqKmNvbmZpZy9wZ3AqKgo+IOWtmOaUvnBncOWFrOengemSpQoKKipjb25maWcvc2VydmljZSoqCj4g5a2Y5pS+5omA5pyJ5pyN5Yqh6YWN572u5paH5Lu2CgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KIyMg5paH5qGjClRPRE8KCiMjIOabtOWkmgpUT0RPCg== readmeEtag: '"2f523e7e0c8bb0d85b63a230e9014752bcfabe59"' readmeLastModified: Tue, 04 Jun 2024 09:01:56 GMT repositoryId: 736890548 description: coss-im-server created: '2023-12-29T07:07:13Z' updated: '2024-09-30T19:23:59Z' language: Go archived: false stars: 5 watchers: 0 forks: 4 owner: cossteam logo: https://avatars.githubusercontent.com/u/171768939?v=4 repoEtag: '"9c93315e1db56f5ff15768040bdf1f26ca0cae3c9350f22665fa20fd5d498e7f"' repoLastModified: Mon, 30 Sep 2024 19:23:59 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/input-output-hk/cardano-governance v3: true repositoryMetadata: base64Readme: >- IyBjYXJkYW5vLWdvdmVybmFuY2UKIVtDYXJkYW5vLUdvdmVybmFuY2UtcmVwb3NpdG9yeS01MDBdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzE3MDIxMjQ5LzE1MDA0NzI4OC1kMjE5NWI4MS02NGE2LTRkMGYtYTdlZi01OTBjZDZkMTdiMGEucG5nKQoK readmeEtag: '"a189815df425364a844146b0cf398a5f65d2aed3"' readmeLastModified: Wed, 19 Jan 2022 01:38:27 GMT repositoryId: 449402368 description: >- A public open governance github repository containing governance documentation, examples and Open APIs. created: '2022-01-18T18:32:22Z' updated: '2024-11-21T22:40:48Z' language: null archived: false stars: 5 watchers: 8 forks: 0 owner: input-output-hk logo: https://avatars.githubusercontent.com/u/12909177?v=4 repoEtag: '"02e5e6dd157d7f11a47e2827dc045f385ab2fccaf7ba46fccb6ff743ffec7224"' repoLastModified: Thu, 21 Nov 2024 22:40:48 GMT foundInMaster: true category: Low-level Tooling id: def1d8b02e3343ae5fb306289fd9c5ce - source: openapi3 tags repository: https://github.com/sergeyklay/provider-pact-example v3: true id: 3c013cdf6ffea735ac445dae12e3466a repositoryMetadata: base64Readme: >- Li4gcmF3OjogaHRtbAoKICAgIDxoMSBhbGlnbj0iY2VudGVyIj5Qcm92aWRlciBBUEkgRXhhbXBsZTwvaDE+CiAgICA8cCBhbGlnbj0iY2VudGVyIj4KICAgICAgICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vc2VyZ2V5a2xheS9wcm92aWRlci1wYWN0LWV4YW1wbGUvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC1jb250cmFjdHMueWFtbCI+CiAgICAgICAgICAgIDxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vc2VyZ2V5a2xheS9wcm92aWRlci1wYWN0LWV4YW1wbGUvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC1jb250cmFjdHMueWFtbC9iYWRnZS5zdmciIGFsdD0iVGVzdCBDb250cmFjdHMiIC8+CiAgICAgICAgPC9hPgogICAgICAgIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9zZXJnZXlrbGF5L3Byb3ZpZGVyLXBhY3QtZXhhbXBsZS9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LWNvZGUueWFtbCI+CiAgICAgICAgICAgIDxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vc2VyZ2V5a2xheS9wcm92aWRlci1wYWN0LWV4YW1wbGUvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC1jb2RlLnlhbWwvYmFkZ2Uuc3ZnIiBhbHQ9IlRlc3QgQ29kZSIgLz4KICAgICAgICA8L2E+CiAgICAgICAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3NlcmdleWtsYXkvcHJvdmlkZXItcGFjdC1leGFtcGxlL2FjdGlvbnMvd29ya2Zsb3dzL2xpbnQtb2FzLnlhbWwiPgogICAgICAgICAgICA8aW1nIHNyYz0iaHR0cHM6Ly9naXRodWIuY29tL3NlcmdleWtsYXkvcHJvdmlkZXItcGFjdC1leGFtcGxlL2FjdGlvbnMvd29ya2Zsb3dzL2xpbnQtb2FzLnlhbWwvYmFkZ2Uuc3ZnIiBhbHQ9IkxpbnQgT3BlbkFQSSIgLz4KICAgICAgICA8L2E+CiAgICAgICAgPGEgaHJlZj0iaHR0cHM6Ly9jb2RlY292LmlvL2doL3NlcmdleWtsYXkvcHJvdmlkZXItcGFjdC1leGFtcGxlIiA+CiAgICAgICAgICAgIDxpbWcgc3JjPSJodHRwczovL2NvZGVjb3YuaW8vZ2gvc2VyZ2V5a2xheS9wcm92aWRlci1wYWN0LWV4YW1wbGUvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPTJDOFcwVlpRR04iIGFsdD0iQ292ZXJhZ2UgU3RhdHVzIi8+CiAgICAgICAgPC9hPgogICAgPC9wPgoKLi4gdGVhc2VyLWJlZ2luCgpUaGlzIGlzIGEgUHl0aG9uIGFwcGxpY2F0aW9uIGZvciBleHBsYW5hdGlvbiBvZiBDb250cmFjdCBUZXN0aW5nIGJhc2VkIG9uCmBQYWN0IDxodHRwczovL2RvY3MucGFjdC5pbz5gXy4KCkhlcmUgeW91IGNhbiBmaW5kIG91dCBob3cgdG8gdXNlIFBhY3QgdXNpbmcgdGhlIFB5dGhvbiBsYW5ndWFnZS4gWW91IGNhbiBmaW5kCm1vcmUgb2YgYW4gb3ZlcnZpZXcgb24gUGFjdCBpbiB0aGUgYFBhY3QgSW50cm9kdWN0aW9uIDxodHRwczovL2RvY3MucGFjdC5pby8+YF8uCgpUaGlzIHByb2plY3QgdXNlczoKCiogYFBhY3QgPGh0dHBzOi8vcGFjdC5pbz5gXywgYSBjb2RlLWZpcnN0IHRvb2wgZm9yIHRlc3RpbmcgSFRUUCBhbmQgbWVzc2FnZQogIGludGVncmF0aW9ucyB1c2luZyBjb250cmFjdCB0ZXN0cwoqIGBwYWN0LXB5dGhvbiA8aHR0cHM6Ly9naXRodWIuY29tL3BhY3QtZm91bmRhdGlvbi9wYWN0LXB5dGhvbj5gXywgdG8gY3JlYXRlCiAgYW5kIHZlcmlmeSBjb25zdW1lciBkcml2ZW4gY29udHJhY3RzCiogYE9wZW5BUEkgPGh0dHBzOi8vc3dhZ2dlci5pbz5gXywgdG8gZGVzY3JpYmUgdGhlIFByb2R1Y3RzIEFQSQoqIGBGbGFzayA8aHR0cHM6Ly9mbGFzay5wYWxsZXRzcHJvamVjdHMuY29tPmBfLCBhIG1pY3JvIHdlYiBmcmFtZXdvcmsgZm9yCiAgYnVpbGRpbmcgQVBJCgouLiB0ZWFzZXItZW5kCgouLiBpbWFnZTo6IGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9zZXJnZXlrbGF5L3Byb3ZpZGVyLXBhY3QtZXhhbXBsZS9tYWluL2NkYy1leGFtcGxlLnBuZwogIDphbHQ6IEludGVyYWN0aW9uIGRpYWdyYW0KClByb3ZpZGVyCj09PT09PT09CgpQcm92aWRlciBBUEkgRXhhbXBsZSBpcyBhIHNhbXBsZSBGbGFzayBhcHBsaWNhdGlvbiB0aGF0IGV4cG9zZSBlbmRwb2ludHMgd2l0aApSRVNUIHN0YW5kYXJkLiBBcyBhbiBleGFtcGxlLCB0aGlzIHByb2plY3QgdXNlcyB0aGUgc2ltcGxlIFByb2R1Y3RzIEFQSS4gSGVyZQppcyB0aGUKYE9wZW5BUEkgc3BlYyA8aHR0cHM6Ly9naXRodWIuY29tL3NlcmdleWtsYXkvcHJvdmlkZXItcGFjdC1leGFtcGxlL2Jsb2IvbWFpbi9vcGVuYXBpL3N3YWdnZXIueWFtbD5gXwpkZXNjcmliZXMgdGhlIGludGVyYWN0aW9uIG9mIGNsaWVudHMgd2l0aCB0aGUgcHJvdmlkZXIuCgpDb25zdW1lcgo9PT09PT09PQoKRm9yIHRoZSBwdXJpdHkgb2YgdGhlIGV4cGVyaW1lbnQsIHRoZSBjb25zdW1lciBpcyBpbXBsZW1lbnRlZCBhcyBhIHNlcGFyYXRlCnByb2plY3QgYW5kIGNhbiBiZSBmb3VuZCBhdApgdGhlIGZvbGxvd2luZyByZXBvIDxodHRwczovL2dpdGh1Yi5jb20vc2VyZ2V5a2xheS9jb25zdW1lci1wYWN0LWV4YW1wbGU+YF8uCgpQYWN0Cj09PT0KClNhbXBsZSBjb250cmFjdHMgKHBhY3RzKSBhcmUgbG9jYXRlZCBoZXJlOgpodHRwczovL2dpdGh1Yi5jb20vc2VyZ2V5a2xheS9jb25zdW1lci1wYWN0LWV4YW1wbGUvdHJlZS9tYWluL3Rlc3RzL3BhY3RzCgpHZXR0aW5nIFN0YXJ0ZWQKPT09PT09PT09PT09PT09CgpQcmVyZXF1aXNpdGVzCi0tLS0tLS0tLS0tLS0KCldoYXQga2luZCBvZiB0aGluZ3MgeW91IG5lZWQgdG8gaW5zdGFsbCBvbiB5b3VyIHdvcmtzdGF0aW9uIHRvIHN0YXJ0OgoKKiBQeXRob24gPj0gMy4xMQoqIFNRTGl0ZTMKKiBOb2RlLmpzID49IDE2CgpJbnN0YWxsaW5nCi0tLS0tLS0tLS0KCkZpcnN0LCBpbnN0YWxsIFB5dGhvbiBkZXBlbmRlbmNpZXMgZm9yIHByb3ZpZGVyIChQcm9kdWN0cyBBUEkpOgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICQgbWFrZSBpbml0CiAgICQgbWFrZSBpbnN0YWxsCgoKQ3JlYXRlIHByb3ZpZGVyIGNvbmZpZ3VyYXRpb246CgouLiBjb2RlLWJsb2NrOjogY29uc29sZQoKICAgJCBjcCAuZW52LmV4YW1wbGUgLmVudgoKUnVuIGRhdGFiYXNlIG1pZ3JhdGlvbnMgZm9yIHByb3ZpZGVyOgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICQgbWFrZSBtaWdyYXRlCgpBZGQgcHJvdmlkZXIgc2VlZCBkYXRhIHRvIHRoZSBkYXRhYmFzZToKCi4uIGNvZGUtYmxvY2s6OiBjb25zb2xlCgogICAkIG1ha2Ugc2VlZAoKTmV4dCwgaW5zdGFsbCBOb2RlLmpzIGxpbnRlcnMgYW5kIHRvb2xzOgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICQgbnBtIGluc3RhbGwKClJ1biBBUEkgc2VydmVyCi0tLS0tLS0tLS0tLS0tCgpUbyBydW4gQVBJIHNlcnZlciB1c2UgdGhlIGNvbW1hbmQgYXMgZm9sbG93czoKCi4uIGNvZGUtYmxvY2s6OiBjb25zb2xlCgogICAkIG1ha2Ugc2VydmUKClJ1biB0ZXN0cwotLS0tLS0tLS0KClRvIHJ1biB1bml0IHRlc3RzIHVzZSB0aGUgY29tbWFuZCBhcyBmb2xsb3dzOgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICQgbWFrZSB0ZXN0CgpUbyB2ZXJpZnkgY29udHJhY3RzIChwYWN0cykgdXNlIHRoZSBjb21tYW5kIGFzIGZvbGxvd3M6CgouLiBjb2RlLWJsb2NrOjogY29uc29sZQoKICAgJCAuL3Rlc3RzL3J1bi1weXRlc3Quc2gKCk5vdGUgdGhhdCBiZWZvcmUgdGhlIGNvbnRyYWN0cyB2ZXJpZmljYXRpb24sIHlvdSBtdXN0IGhhdmUgZGVwbG95ZWQgdGhlIGJyb2tlciwKYXMgd2VsbCBhcyB0aGUgY29udHJhY3RzIG11c3QgYmUgcHVibGlzaGVkLgoKUnVuIGxpbnQgY2hlY2sKLS0tLS0tLS0tLS0tLS0KClRvIHJ1biBPcGVuQVBJIHNwZWMgY2hlY2tpbmcgdXNlIHRoZSBjb21tYW5kIGFzIGZvbGxvd3M6CgouLiBjb2RlLWJsb2NrOjogY29uc29sZQoKICAgJCBucG0gcnVuIGxpbnQKCgouLiAtcHJvamVjdC1pbmZvcm1hdGlvbi0KClByb2plY3QgSW5mb3JtYXRpb24KPT09PT09PT09PT09PT09PT09PQoKUHJvdmlkZXIgQVBJIEV4YW1wbGUgaXMgcmVsZWFzZWQgdW5kZXIgdGhlIGBNSVQgTGljZW5zZSA8aHR0cHM6Ly9jaG9vc2VhbGljZW5zZS5jb20vbGljZW5zZXMvbWl0Lz5gXywKYW5kIGl0cyBjb2RlIGxpdmVzIGF0IGBHaXRIdWIgPGh0dHBzOi8vZ2l0aHViLmNvbS9zZXJnZXlrbGF5L3Byb3ZpZGVyLXBhY3QtZXhhbXBsZT5gXy4KSXTigJlzIHJpZ29yb3VzbHkgdGVzdGVkIG9uIFB5dGhvbiAzLjExKy4KCklmIHlvdSdkIGxpa2UgdG8gY29udHJpYnV0ZSB0byBQcm92aWRlciBBUEkgRXhhbXBsZSB5b3UncmUgbW9zdCB3ZWxjb21lIQoKLi4gLXN1cHBvcnQtCgpTdXBwb3J0Cj09PT09PT0KClNob3VsZCB5b3UgaGF2ZSBhbnkgcXVlc3Rpb24sIGFueSByZW1hcmssIG9yIGlmIHlvdSBmaW5kIGEgYnVnLCBvciBpZiB0aGVyZSBpcwpzb21ldGhpbmcgeW91IGNhbid0IGRvIHdpdGggdGhlIFByb3ZpZGVyIEFQSSBFeGFtcGxlLCBwbGVhc2UKYG9wZW4gYW4gaXNzdWUgPGh0dHBzOi8vZ2l0aHViLmNvbS9zZXJnZXlrbGF5L3Byb3ZpZGVyLXBhY3QtZXhhbXBsZS9pc3N1ZXM+YF8uCg== readmeEtag: '"eca875a7a45a62339691a683ba4be1654a46e539"' readmeLastModified: Mon, 05 Aug 2024 12:03:43 GMT repositoryId: 592216369 description: Provider-side demo using consumer-driven contract testing created: '2023-01-23T08:22:09Z' updated: '2026-01-08T17:03:33Z' language: Python archived: true stars: 6 watchers: 1 forks: 2 owner: sergeyklay logo: https://avatars.githubusercontent.com/u/1256298?v=4 license: MIT repoEtag: '"472e8579e1d1e937f8ad269cbf09c792aa62c2e17e9a99d644237c9b2e6e0260"' repoLastModified: Thu, 08 Jan 2026 17:03:33 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/sergeyklay/contract-testing-example - https://github.com/sergeyklay/specmatic-testing-example - source: openapi3 tags repository: https://github.com/deepakbhalla/springboot-postgres-jsonb-jpa v3: true id: ee3ef4783b54d3953064d870157c1920 repositoryMetadata: base64Readme: >- IyBzcHJpbmdib290LXBvc3RncmVzLWpzb25iLWpwYQpBcHBsaWNhdGlvbiB0byBkZW1vbnN0cmF0ZSB0aGUgSlNPTkIgZGF0YSB0eXBlIHVzZWQgaW4gUG9zdGdyZVNRTCBkYXRhYmFzZSB1c2luZyBTcHJpbmdib290IEpQQSBpbXBsZW1lbnRhdGlvbi4KCiMjIyBKU09OIERhdGEgVHlwZXMKCkpTT04gZGF0YSB0eXBlcyBhcmUgZm9yIHN0b3JpbmcgSlNPTiAoSmF2YVNjcmlwdCBPYmplY3QgTm90YXRpb24pIGRhdGEuIFN1Y2ggZGF0YSBjYW4gYWxzbyBiZSBzdG9yZWQgYXMgdGV4dCwgYnV0IHRoZSAKSlNPTiBkYXRhIHR5cGVzIGhhdmUgdGhlIGFkdmFudGFnZSBvZiBlbmZvcmNpbmcgdGhhdCBlYWNoIHN0b3JlZCB2YWx1ZSBpcyB2YWxpZCBhY2NvcmRpbmcgdG8gdGhlIEpTT04gcnVsZXMuIFRoZXJlIGFyZSAKYWxzbyBhc3NvcnRlZCBKU09OLXNwZWNpZmljIGZ1bmN0aW9ucyBhbmQgb3BlcmF0b3JzIGF2YWlsYWJsZSBmb3IgZGF0YSBzdG9yZWQgaW4gdGhlc2UgZGF0YSB0eXBlcy4KClBvc3RncmVTUUwgb2ZmZXJzIHR3byB0eXBlcyBmb3Igc3RvcmluZyBKU09OIGRhdGE6IAoKLSBKU09OIAotIEpTT05CLiAKClRvIGltcGxlbWVudCBlZmZpY2llbnQgcXVlcnkgbWVjaGFuaXNtcyBmb3IgdGhlc2UgCmRhdGEgdHlwZXMsIFBvc3RncmVTUUwgYWxzbyBwcm92aWRlcyB0aGUganNvbnBhdGggZGF0YSB0eXBlLgoKVGhlIGpzb24gYW5kIGpzb25iIGRhdGEgdHlwZXMgYWNjZXB0IGFsbW9zdCBpZGVudGljYWwgc2V0cyBvZiB2YWx1ZXMgYXMgaW5wdXQuIFRoZSBtYWpvciBwcmFjdGljYWwgZGlmZmVyZW5jZSBpcyBvbmUgCm9mIGVmZmljaWVuY3kuIFRoZSBqc29uIGRhdGEgdHlwZSBzdG9yZXMgYW4gZXhhY3QgY29weSBvZiB0aGUgaW5wdXQgdGV4dCwgd2hpY2ggcHJvY2Vzc2luZyBmdW5jdGlvbnMgbXVzdCByZXBhcnNlIG9uIAplYWNoIGV4ZWN1dGlvbjsgd2hpbGUganNvbmIgZGF0YSBpcyBzdG9yZWQgaW4gYSBkZWNvbXBvc2VkIGJpbmFyeSBmb3JtYXQgdGhhdCBtYWtlcyBpdCBzbGlnaHRseSBzbG93ZXIgdG8gaW5wdXQgZHVlIHRvIAphZGRlZCBjb252ZXJzaW9uIG92ZXJoZWFkLCBidXQgc2lnbmlmaWNhbnRseSBmYXN0ZXIgdG8gcHJvY2Vzcywgc2luY2Ugbm8gcmVwYXJzaW5nIGlzIG5lZWRlZC4ganNvbmIgYWxzbyBzdXBwb3J0cyAKaW5kZXhpbmcsIHdoaWNoIGNhbiBiZSBhIHNpZ25pZmljYW50IGFkdmFudGFnZS4KCkJlY2F1c2UgdGhlIGpzb24gdHlwZSBzdG9yZXMgYW4gZXhhY3QgY29weSBvZiB0aGUgaW5wdXQgdGV4dCwgaXQgd2lsbCBwcmVzZXJ2ZSBzZW1hbnRpY2FsbHktaW5zaWduaWZpY2FudCB3aGl0ZSBzcGFjZSAKYmV0d2VlbiB0b2tlbnMsIGFzIHdlbGwgYXMgdGhlIG9yZGVyIG9mIGtleXMgd2l0aGluIEpTT04gb2JqZWN0cy4gQWxzbywgaWYgYSBKU09OIG9iamVjdCB3aXRoaW4gdGhlIHZhbHVlIGNvbnRhaW5zIHRoZSAKc2FtZSBrZXkgbW9yZSB0aGFuIG9uY2UsIGFsbCB0aGUga2V5L3ZhbHVlIHBhaXJzIGFyZSBrZXB0LiAoVGhlIHByb2Nlc3NpbmcgZnVuY3Rpb25zIGNvbnNpZGVyIHRoZSBsYXN0IHZhbHVlIGFzIHRoZSAKb3BlcmF0aXZlIG9uZS4pIEJ5IGNvbnRyYXN0LCBqc29uYiBkb2VzIG5vdCBwcmVzZXJ2ZSB3aGl0ZSBzcGFjZSwgZG9lcyBub3QgcHJlc2VydmUgdGhlIG9yZGVyIG9mIG9iamVjdCBrZXlzLCBhbmQgZG9lcyAKbm90IGtlZXAgZHVwbGljYXRlIG9iamVjdCBrZXlzLiBJZiBkdXBsaWNhdGUga2V5cyBhcmUgc3BlY2lmaWVkIGluIHRoZSBpbnB1dCwgb25seSB0aGUgbGFzdCB2YWx1ZSBpcyBrZXB0LgoKSW4gZ2VuZXJhbCwgbW9zdCBhcHBsaWNhdGlvbnMgc2hvdWxkIHByZWZlciB0byBzdG9yZSBKU09OIGRhdGEgYXMganNvbmIsIHVubGVzcyB0aGVyZSBhcmUgcXVpdGUgc3BlY2lhbGl6ZWQgbmVlZHMsIHN1Y2ggCmFzIGxlZ2FjeSBhc3N1bXB0aW9ucyBhYm91dCBvcmRlcmluZyBvZiBvYmplY3Qga2V5cy4KCkpTT05CIGlzIGEgImJldHRlciIgdmVyc2lvbiBvZiBKU09OLgoKTGV0J3MgbG9vayBhdCBhbiBleGFtcGxlOgoKLSBKU09OCgohW2ltZy5wbmddKHNjcmVlbnNob3RzLzE5X2pzb25fZGF0YV90eXBlX2V4YW1wbGUucG5nKQoKLSBKU09OQgoKIVtpbWcucG5nXShzY3JlZW5zaG90cy8yMF9qc29uYl9kYXRhX3R5cGVfZXhhbXBsZS5wbmcpCgpJbiBzdW1tYXJ5LAoKMS4gSlNPTiBzdG9yZXMgd2hpdGUgc3BhY2UsIGFuZCB0aGF0IGlzIHdoeSB3ZSBjYW4gc2VlIHNwYWNlcyB3aGVuIGtleSAiYSIgaXMgc3RvcmVkLCB3aGlsZSBKU09OQiBkb2VzIG5vdC4KMi4gSlNPTiBzdG9yZXMgYWxsIHRoZSB2YWx1ZXMgb2YgYSBrZXkuIFRoaXMgaXMgdGhlIHJlYXNvbiB5b3UgY2FuIHNlZSBtdWx0aXBsZSB2YWx1ZXMgKDIgYW5kIDEpIGFnYWluc3QgdGhlIGtleSAiYSIsIHdoaWxlIEpTT05CIG9ubHkgInN0b3JlcyIgdGhlIGxhc3QgdmFsdWUuCjMuIEpTT04gbWFpbnRhaW5zIHRoZSBvcmRlciBpbiB3aGljaCBlbGVtZW50cyBhcmUgaW5zZXJ0ZWQsIHdoaWxlIEpTT05CIG1haW50YWlucyB0aGUgInNvcnRlZCIgb3JkZXIuCjQuIEpTT05CIG9iamVjdHMgYXJlIHN0b3JlZCBhcyBhIGRlY29tcHJlc3NlZCBiaW5hcnkgYXMgb3Bwb3NlZCB0byAicmF3IGRhdGEiIGluIEpTT04sIHdoZXJlIG5vIHJlcGFyc2luZyBvZiBkYXRhIGlzIHJlcXVpcmVkIGR1cmluZyByZXRyaWV2YWwuCjUuIEpTT05CIGFsc28gc3VwcG9ydHMgaW5kZXhpbmcsIHdoaWNoIGNhbiBiZSBhIHNpZ25pZmljYW50IGFkdmFudGFnZS4KCiMjIyBKU09OIEZ1bmN0aW9ucyBhbmQgT3BlcmF0b3JzCgpUaGlzIHNlY3Rpb24gZGVzY3JpYmVzOgoKLSBmdW5jdGlvbnMgYW5kIG9wZXJhdG9ycyBmb3IgcHJvY2Vzc2luZyBhbmQgY3JlYXRpbmcgSlNPTiBkYXRhCi0gdGhlIFNRTC9KU09OIHBhdGggbGFuZ3VhZ2UKClRvIHByb3ZpZGUgbmF0aXZlIHN1cHBvcnQgZm9yIEpTT04gZGF0YSB0eXBlcyB3aXRoaW4gdGhlIFNRTCBlbnZpcm9ubWVudCwgUG9zdGdyZVNRTCBpbXBsZW1lbnRzIHRoZSBTUUwvSlNPTiBkYXRhIG1vZGVsLiAKVGhpcyBtb2RlbCBjb21wcmlzZXMgc2VxdWVuY2VzIG9mIGl0ZW1zLiBFYWNoIGl0ZW0gY2FuIGhvbGQgU1FMIHNjYWxhciB2YWx1ZXMsIHdpdGggYW4gYWRkaXRpb25hbCBTUUwvSlNPTiBudWxsIHZhbHVlLCAKYW5kIGNvbXBvc2l0ZSBkYXRhIHN0cnVjdHVyZXMgdGhhdCB1c2UgSlNPTiBhcnJheXMgYW5kIG9iamVjdHMuIFRoZSBtb2RlbCBpcyBhIGZvcm1hbGl6YXRpb24gb2YgdGhlIGltcGxpZWQgZGF0YSBtb2RlbCAKaW4gdGhlIEpTT04gc3BlY2lmaWNhdGlvbi4KClNRTC9KU09OIGFsbG93cyB5b3UgdG8gaGFuZGxlIEpTT04gZGF0YSBhbG9uZ3NpZGUgcmVndWxhciBTUUwgZGF0YSwgd2l0aCB0cmFuc2FjdGlvbiBzdXBwb3J0LCBpbmNsdWRpbmc6CgotIFVwbG9hZGluZyBKU09OIGRhdGEgaW50byB0aGUgZGF0YWJhc2UgYW5kIHN0b3JpbmcgaXQgaW4gcmVndWxhciBTUUwgY29sdW1ucyBhcyBjaGFyYWN0ZXIgb3IgYmluYXJ5IHN0cmluZ3MuCi0gR2VuZXJhdGluZyBKU09OIG9iamVjdHMgYW5kIGFycmF5cyBmcm9tIHJlbGF0aW9uYWwgZGF0YS4KLSBRdWVyeWluZyBKU09OIGRhdGEgdXNpbmcgU1FML0pTT04gcXVlcnkgZnVuY3Rpb25zIGFuZCBTUUwvSlNPTiBwYXRoIGxhbmd1YWdlIGV4cHJlc3Npb25zLgoKIyMjIE9wZW5BUEkgU3BlY2lmaWNhdGlvbgoKaHR0cDovL2xvY2FsaG9zdDo5MDkxL215LWFwcGxpY2F0aW9uL3N3YWdnZXItdWkvaW5kZXguaHRtbAoKIVtpbWcucG5nXShzY3JlZW5zaG90cy8yMV9vcGVuYXBpX3N3YWdnZXJfc3BlY2lmaWNhdGlvbnMucG5nKQoKIyMjIERpc2FibGUgT3BlbkFQSSBTd2FnZ2VyIGZvciBQcm9kdWN0aW9uIEVudmlyb25tZW50CgpXZSBjYW4gZGlzYWJsZSBPcGVuQVBJIHN3YWdnZXIgZm9yIGFueSBlbnZpcm9ubWVudCBiYXNlZCB1cG9uIHByb2ZpbGVzLiBXZSBjYW4gc3VwcGx5IGEgVk0gYXJndW1lbnQgCictRHNwcmluZy5wcm9maWxlcy5hY3RpdmU9PGVudmlyb25tZW50IG5hbWU+JyB0byB0aGUgYXBwbGljYXRpb24gY29uZmlndXJhdGlvbnMuClVzaW5nIHNwcmluZyBwcm9maWxlIGFubm90YXRpb24gQFByb2ZpbGUoInByb2QiKSwgd2UgY2FuIGNvbnRyb2wgdGhlIGRpc3BsYXkgb2Ygc3dhZ2dlci4KCi0gVk0gQXJndW1lbnQKCiFbaW1nLnBuZ10oc2NyZWVuc2hvdHMvMjJfdm1fYXJndW1lbnRfc3ByaW5nX2FjdGl2ZV9wcm9maWxlLnBuZykKCklmIHRoZSB2YWx1ZSBvZiBzcHJpbmcgcHJvZmlsZSBpcyAncHJvZCcsIHRoZW4gc3dhZ2dlciB3b24ndCBiZSBhdmFpbGFibGUuIFBsZWFzZSByZWZlciB0aGUgYmVsb3cgc2NyZWVuc2hvdDoKCiFbaW1nLnBuZ10oc2NyZWVuc2hvdHMvMjNfc3dhZ2dlcl9ub3RfYXZhaWxhYmxlLnBuZykKClJlZmVyZW5jZXM6CgotIGh0dHBzOi8vd3d3LnBvc3RncmVzcWwub3JnL2RvY3MvY3VycmVudC9kYXRhdHlwZS1qc29uLmh0bWwKLSBodHRwczovL3d3dy5wb3N0Z3Jlc3FsLm9yZy9kb2NzL2N1cnJlbnQvZnVuY3Rpb25zLWpzb24uaHRtbAo= readmeEtag: '"dca96c49d87c408f4d74578cada0f7a61dcfe458"' readmeLastModified: Thu, 26 Oct 2023 18:27:53 GMT repositoryId: 704453985 description: >- Spring boot application to demonstrate the JSONB data type to be used in Postgre SQL database using Springboot JPA. created: '2023-10-13T09:40:28Z' updated: '2026-01-15T05:24:49Z' language: Java archived: false stars: 4 watchers: 1 forks: 1 owner: deepakbhalla logo: https://avatars.githubusercontent.com/u/6603640?v=4 repoEtag: '"a6cd6654a116a2f849d56ddba14c229bc62711703ad55081d93bbbf16a65d181"' repoLastModified: Thu, 15 Jan 2026 05:24:49 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/progremir/django-swagger-render v3: true repositoryMetadata: base64Readme: >- IyBEamFuZ28gU3dhZ2dlciBSZW5kZXIKCiMjIEdldHRpbmcgU3RhcnRlZAoKIyMjIFByZXJlcXVpc2l0ZXMKCi0gcHl0aG9uID49IDMuNQotIERqYW5nbyA+PSAyLjAKCiMjIyBJbnN0YWxsYXRpb24KCkluc3RhbGwgdXNpbmcgcGlwCgpgYGAKcGlwIGluc3RhbGwgZGphbmdvLXN3YWdnZXItcmVuZGVyCmBgYAoKQWRkICdzd2FnZ2VyX3JlbmRlcicgdG8geW91ciBJTlNUQUxMRURfQVBQUyBzZXR0aW5nLgoKYGBgCklOU1RBTExFRF9BUFBTID0gWwogICAgLi4uCiAgICAnc3dhZ2dlcl9yZW5kZXInLApdCmBgYAoKQ3JlYXRlIHRoZSBmb2xkZXIgd2hlcmUgeW91IHdpbGwgc3RvcmUgeW91ciBkb2N1bWVudGF0aW9uCgpgYGAKbWtkaXIgZG9jcwpgYGAKCkNyZWF0ZSB0aGUgYGluZGV4LnltbGAgZmlsZSB3aXRoIHNvbWUgYE9QRU5BUElgIG9yIGBTd2FnZ2VyYCBzcGVjaWZpY2F0aW9ucwoKYGBgCnRvdWNoIGRvY3MvaW5kZXgueW1sCmBgYAoKU2VydmUgeW91ciBkb2N1bWVudGF0aW9uIGZpbGVzCmBgYAp1cmxwYXR0ZXJucyArPSBzdGF0aWMoJy9kb2NzLycsIGRvY3VtZW50X3Jvb3Q9J2RvY3MnKQpgYGAKCkFkZCBgU1dBR0dFUl9ZQU1MX0ZJTEVOQU1FYCBzZXR0aW5nIHRvIHlvdXIgYHNldHRpbmdzLnB5YAoKYGBgClNXQUdHRVJfWUFNTF9GSUxFTkFNRSA9ICcvZG9jcy9pbmRleC55bWwnCmBgYAoKQWRkIHRoZSBgU3dhZ2dlclVJVmlld2AgdG8geW91ciB1cmxzCgpgYGAKZnJvbSBzd2FnZ2VyX3JlbmRlci52aWV3cyBpbXBvcnQgU3dhZ2dlclVJVmlldwoKCnVybHBhdHRlcm5zID0gWwogICAgLi4uCiAgICBwYXRoKCdzd2FnZ2VyLycsIFN3YWdnZXJVSVZpZXcuYXNfdmlldygpKSwKXQpgYGAKClZvaWxhIQo= readmeEtag: '"880cca32dd57b1eb6398cdcd4c9d9d75e3e3982d"' readmeLastModified: Sat, 27 Jun 2020 11:25:49 GMT repositoryId: 200704843 description: Swagger documentation in Django created: '2019-08-05T18:04:35Z' updated: '2023-07-21T06:43:55Z' language: HTML archived: false stars: 5 watchers: 2 forks: 2 owner: progremir logo: https://avatars.githubusercontent.com/u/17231674?v=4 license: MIT repoEtag: '"0ed75eb58e4fc40f5ce3c1ccfe68e6349430b92d6db5726a60c1810603b92f94"' repoLastModified: Fri, 21 Jul 2023 06:43:55 GMT foundInMaster: true category: - Description Validators - Server Implementations id: e98db438e3bce57985e796d2d2a012e2 - source: openapi3 tags repository: https://github.com/shakibmostahid/writing-swagger v3: true id: bff913362ef7940f3409c2d9e378af3a repositoryMetadata: base64Readme: >- # What is Swagger Documentation? 

Swagger is the standard way of documenting REST APIs. *Swagger is a set of open-source tools built around the **OpenAPI Specification** that can help to design, build, document and consume REST APIs.* This is a popular framework which used for documenting the APIs in a common format.

To learn more about Swagger, please follow the [documentation](https://swagger.io/docs/specification/about/ "documentation").

# Why is Swagger is helpful? 

We can think swagger documentation as a blueprint of the APIs. By checking the blueprint developer can get an idea of the task of an API. Swagger gives a visual idea of the APIs, where user doesn’t need to know what technologies is being used to develop the APIs. API is documented in a common format in swagger where HTTP Verb, resopnse samples, request samples, description, etc is present. so anyone can get a clear picture of the API just by checking it. 

# How to Write Swagger Documentation?

Swagger can be written in yml and json. We will use yml here. And, we will use [Swagger Editor tool](https://editor.swagger.io/ "Swagger Editor tool") to write the documentation. Here, we will use an example of User Management APIs, where user details can be Retrieved, Updated and Deleted.

#### Let’s Start:
- First we need to define the OpenAPI version, here we will use version 3.

```yaml
openapi: 3.0.0
```
- Then we need to add information about the projects.

```yaml
info:
  title: User Management APIs
  description: This is a sample project for managing users.
  contact:
    name: John Doe
    email: john@example.com
  version: "1.0"
```
Here, `info` attribute holds the projects information. `title` represents project title. `version` represnts the API version.

------------

- Now, we will add the protocol and domain of our API, also we will add other fixed reourceses which is fixed for all the APIs like API version. All the information will be under `servers` attribute.

 ```yaml
 servers:
 - url: https://example.com/v1
 ```
Here, we have set our API domain URL for HTTPS protocol. So, what will happen if our API supports both HTTP and HTTPS. How will we add HTTP protocol here?
We can simply add another value here in the array.

 ```yaml
 servers:
 - url: https://example.com/v1
 - url: http://example.com/v1
 ```
now, swagger will show us both of the servers information in a dropdown.

- what happens if we have multiple domain names? We can add more values into the array. But there’s another way. We can use variables attribute to make it more dynamic.

```yaml
servers:
- url: "{protocol}://{base_url}/v1"
  variables:
    protocol:
      default: https
      enum:
        - https
        - http
    base_url:
      default: example.com
      enum:
        - example.com
        - bd.example.com
```

here, we defined `protocol` and `base_url` as variable into curly braces, these variable name can be anything. Then we set the value of these under `variables` attribute.
After setting the values we can see multiple dropdowns for each variable.

------------

#### Now after adding these lines into swagger editor, the output will be as below:

![swagger_servers_output](https://github.com/shakibmostahid/writing-swagger/blob/main/images/swagger_1.png?raw=true)

------------

#### We need to add the paths information.

- Now, we will add the API details under `paths` attribute. We will add the API summary and reponse sample in the documentation. For example we have an API for getting customer information for given ID, we will add information for it:

```yaml
paths:
  /users/1:
    get:
      summary: Get User details
      operationId: getUser
      responses:
        200:
          description: API to fetch the user information by user id.
          headers:
            Cache-Control:
              schema:
                type: string
                example: no-cache, private
          content:
            application/json:
              schema:
                type: object
                properties:
                  id:
                    type: integer
                    example: 1
                  first_name:
                    type: string
                    example: Jalil
                  last_name:
                    type: string
                    example: Uddin
                  address:
                    type: object
                    properties:
                      area:
                        type: string
                        example: Mirpur
                      city:
                        type: string
                        example: Dhaka
                      country:
                        type: string
                        example: Bangladesh
```

here, we have set our API path `customers/1` under `paths` attribute. Then we have define the request type. This is a GET request so, we have define it as `get`.
Then we have set a short summary for this API in `summary` attribute. `operationId` represents the identity of the operation, this should be unique.
Then we have added response details under `responses` attribute.
First we need to define the status codes, there can be multiple responses with different status code. Here we have set only one for `200` status, we will add one error response later.
The response body needs to define under `content` attribute.
We need to add the content type here; we have added `application/json` as content type.
Then, response schema needs to define under `schema` attribute.
our response is a JSON object, that’s why we have set `type: object`. As we have set type = object so, we need to define the object details under `properties` attribute.

------------

- Now, we will add the headers info for the success response. We have set the response example under `content` attribute of `200` status, we will add the headers informations under `headers` attribute of `200` status:

```yaml
headers:
  Cache-Control:
    schema:
      type: string
      example: no-cache, private
  Date:
    schema:
      type: string
      example: Thu, 17 Nov 2022 09:52:10 GMT
```
We have added only two headers, we can add as many headers as we want.

#### Now after adding the response example lines into swagger editor, the output will be as below:

![swagger_response_output](https://github.com/shakibmostahid/writing-swagger/blob/main/images/swagger_2.png?raw=true)

------------

- We might need the example of `user` object on others API. For example, we will return the same user object after creating an user. Then what we can do is copy pasting the response content. That would definitely work, but the contents of the file keeps increasing. It would be better if we can avoid the duplication.
The solution for this problem is to add common definitions in the global `components` attribute and add the references using `$ref` of the definitions wherever we need. 
Let's write the user schema under `components` attribute:

```yaml
components:
  schemas:
    User:
      type: object
      properties:
        id:
          type: integer
          example: 1
        first_name:
          type: string
          example: Jalil
        last_name:
          type: string
          example: Uddin
        address:
          type: object
          properties:
            area:
              type: string
              example: Mirpur
            city:
              type: string
              example: Dhaka
            country:
              type: string
              example: Bangladesh
```

Here we have added the user details under `schemas` atrribute which parent element is `components`. We have named our schema with `User`. Then we have added the schema details. 

Now, our next task is to add the references in the success response content. We need to remove the user schema from there, and need to add only one line to refer the object. So the changes would be like: 

```yaml
content:
  application/json:
    schema:
      "$ref": "#/components/schemas/User"
```

We have just set the reference under `schema` by setting the value of `$ref` attribute. The format is `#/components/<type>/<OjbectName>`. Our type is `schemas` and our object name is `User`. That's why we have set the value `#/components/schemas/User`.

After adding these changes, the response object would be as previous. But there will be shown a new component named `Schemas`.
That would look like:

![swagger_schema_output](https://github.com/shakibmostahid/writing-swagger/blob/main/images/swagger_3.png?raw=true)

------------

- In the example, we have used user's `id` in path with the value 1. So, the path become, `/users/1`, which looks like a static route. But our route is dynamic and it is based on the user's unique id. Swagger give us the ability to make the path dynamic. We need to add placeholder in the `id` portion. To add place holder we use curly braces `{}`, and inside the curly braces we add the variable name. Now, we will make changes in the `path` section like this:

```yaml
path:
  users/{id}:
    get:
```

Now after adding this, we will see an semantic error saying `Declared path parameter "id" needs to be defined as a path parameter at either the path or operation level`.

To fix the error, we need to add set the information for the `{id}` in the `parameters` attribute. We can have multiple variables for a single route, all the information should be added under `parameters`. We can use variable in query params, body, header as well.

Here we have only one variable `id`, so we will add the information for it only.

```yaml
paths:
  /users/{id}:
    get:
      summary: Get User details
      operationId: getUser
      parameters:
        - in: path
          name: id
          schema:
            type: integer
            example: 1
          required: true
```

Here, as our variable location is in path, that's why we have set `path` value for `in` attribute. Then, we have entered the schema details for `id` variable. We have set example `1` as user's id. 

After making these changes, there will be a new row with parameters information in the output section. It will look like:


![swagger_parameters](https://github.com/shakibmostahid/writing-swagger/blob/main/images/swagger_4.png?raw=true)

------------

- Let's add an `POST` API details in the swagger doc. Now, we will add an API specification for creating an user to our system. First we will add the route and other details like we did in the `GET` API. We need to create an key with the route name under `paths` attribute. And the key must be siblings of other routes. For example, we are setting this route as `/users`. So, it will be siblings of `/users/{id}`.

```yaml
paths:
  /users/{id}:
  ..............
  /users:
    post:
      summary: Create New User
      operationId: createUser
```

Then, we will add the request payload details which contains the users details. To add the request payload, we need to define it under `requestBody` attribute. Let's add the information:

```yaml
/users:
  post:
    summary: Create New User
    operationId: createUser
    requestBody:
      description: Request payload for creating new user 
      required: true
      content:
        application/json:
          schema:
            $ref: '#/components/schemas/UserPayload'
components:
  schemas:
    UserPayload:
      type: object
      properties:
        first_name:
          type: string
          example: Hamid
        last_name:
          type: string
          example: Ahmed
        address:
          type: object
          properties:
            area:
              type: string
              example: Mirpur
            city:
              type: string
              example: Dhaka
            country:
              type: string
              example: Bangladesh
```
Here, `description` under `requestBody` represents the summary of the request body. Then we have set the payload content type under `content` attribute. As our request will contain JSON data, we have set our schema details under `application/json`. We have create a new schema for the request payload, and set the references on the `schema` attribute.

We can add the full request body as reference, then we can use the reference directly if we need the request body anywhere. To achieve this, we need to add `requestBodies` key under `components` attribute, then we need to add the details there with a custom key name. Let's add it:

```yaml
components:
  requestBodies:
    UserRequestPayload:
      description: Request payload for creating new user 
      required: true
      content:
        application/json:
          schema:
            $ref: '#/components/schemas/UserPayload'
```

Then, we can use `UserRequestPayload` reference in the API specifications. So, we will make the changes now: 

```yaml
/users:
  post:
    summary: Create New User
    operationId: createUser
    requestBody:
        "$ref": '#/components/requestBodies/UserRequestPayload'
```

Now, the only part left here is adding the response information. The response will be same as our `GET` request. So, our complete code will be looking like below:

```yaml
/users:
  post:
    summary: Create New User
    operationId: createUser
    requestBody:
      "$ref": '#/components/requestBodies/UserRequestPayload'
    responses:
      201:
        description: API to Create the user information and get created user's details.
        headers:
          Cache-Control:
            schema:
              type: string
              example: no-cache, private
          Date:
            schema:
              type: string
              example: Thu, 17 Nov 2022 09:52:10 GMT
        content:
          application/json:
            schema:
              "$ref": "#/components/schemas/User"
```

After adding the `POST` API specification, there will be a new block for the user create API. After expanding the block, It will look like:

![swagger_parameters](https://github.com/shakibmostahid/writing-swagger/blob/main/images/swagger_5.png?raw=true)

------------

- All the API details are shown under *default* group, It is called `tags`. If we don't set any tags for the route or API it will be set to *default*. Let's create a tag here and set to the routes.

To create a tag, we need to define the details under `tags` attribute. This attribute will be siblings of `paths` attribute. It has no parent element.

```yml
paths:
...................

tags:
  - name: Users
    description: Users API specifications
```

Then, we need to add the tag name in the routes details to group it together under `tags` attribute as array value. Now the changes will be:

```yml
paths:
  /users/{id}:
    get:
      tags:
        - Users
      .............................
  /users:
    post:
      tags:
        - Users
      .............................
```

Now both of the routes group name is changed to `Users`. 

![swagger_parameters](https://github.com/shakibmostahid/writing-swagger/blob/main/images/swagger_6.png?raw=true)

------------ readmeEtag: '"f7b25084169d09619ab2693aca19851df8355026"' readmeLastModified: Mon, 30 Oct 2023 05:02:48 GMT repositoryId: 568020156 description: This is a guideline for beginner who wants to learn swagger with examples. created: '2022-11-19T07:32:30Z' updated: '2025-04-25T17:01:41Z' language: null archived: false stars: 5 watchers: 1 forks: 0 owner: shakibmostahid logo: https://avatars.githubusercontent.com/u/36795674?v=4 license: Unlicense repoEtag: '"84fcf9e5938048a2aa94f7a02293623339821cb0bfe4238c6b501b1f717131f3"' repoLastModified: Fri, 25 Apr 2025 17:01:41 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/ipranjal/openapi-spec v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+PGltZyBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9PU1NOQVBJL1N1Ym1pc3Npb24tSW1hZ2VzL2Jsb2IvbWFpbi9sb2dvLnBuZz9yYXc9dHJ1ZSIgd2lkdGg9IjIwMCIgaGVpZ2h0PSIyMDAiPjwvcD4KCiMgT3BlbkFQSS1zcGVjCk9wZW4gQVBJIDMuMCBzcGVjIGZvciBjcmVhdGluZyBzb2NpYWwgbmV0d29ya2luZyBBUEkKClRoaXMgc3BlY2lmaWNhdGlvbiB3YXMgY3JlYXRlZCBhcyBwYXJ0IG9mICoqVGhlIFBvc3RtYW4gQVBJIEhhY2sqKiBhbmQgcHJvdmlkZXMgYSAqKk9wZW5BUEkgMy4wKiogc3BlY2lmaWNhdGlvbiBmb3IgY3JlYXRpbmcgU29jaWFsIE5ldHdvcmtpbmcgQVBJLCB1c2luZyB0aGVzZSBBUEkncyB5b3UgY2FuIGNyZWF0ZSBhIHNvY2lhbCBuZXR3b3JraW5nIHdlYnNpdGUgbGlrZSBGYWNlYm9vaywgVHdpdHRlcgoKWW91IGNhbiBnb3RvICB0byBmaW5kIHRoZSB3b3Jrc3BhY2Ugd2l0aCBBUEkgY29sbGVjdGlvbiAsIFRlc3RzIGFuZCBNb2NrIFNlcnZlciBmb3IgZGV2ZWxvcGluZyB1c2luZyB0aGlzIEFQSSBzcGVjaWZpY2F0aW9uLiAKCgojIyBVc2VmdWwgTGlua3MKLSBCbG9nIFBvc3RzIGFib3V0IHRoZSBwcm9qZWN0IGFuZCBoYWNrYXRob24gLSBodHRwczovL3BoeXNjb2NvZGUuY29tL2NhdGVnb3J5L3Bvc3RtYW4vCi0gRG9jdW1lbnQgZXhwbGFpbmluZyB0aGUgQVBJJ3MgY2FuIGJlIGZvdW5kIGhlcmUgLSBodHRwczovL2RvY3VtZW50ZXIuZ2V0cG9zdG1hbi5jb20vdmlldy8xNzIyMTExL1RXNnRMcTUzCi0gWW91IGNhbiBmaW5kIHRoZSBwdWJsaWMgd29ya3NwYWNlIGhlcmUgLSBodHRwczovL3d3dy5wb3N0bWFuLmNvbS9wcmFuamFsc2hvcC93b3Jrc3BhY2UvdGhlLXNvY2lhbC1uZXR3b3JrLWFwaS1zL292ZXJ2aWV3Ci0gQVBJIGRlc2lnbiBjb25jZXB0IC0gaHR0cHM6Ly9waHlzY29jb2RlLmNvbS9vc3NuLWFwaS1vcGVuLXNvdXJjZS1zb2NpYWwtbmV0d29ya2luZy1hcGktZGVzaWduLWNvbmNlcHQvCi0gU2VydmVyIHNpZGUgaW1wbGVtZW50YXRpb24gb2Ygc3BlY2lmaWNhdGlvbiAtIGh0dHBzOi8vZ2l0aHViLmNvbS9PU1NOQVBJL09TU04tc2VydmVyLXBocAoKIyMgVGhpcyBzcGVjaWZpY2F0aW9uIGRlZmluZXMgaW1wbGVtZW50YWlvbiBvZiBmb2xsb3dpbmcgQVBJOgoxKSAqKlVzZXIgQVBJIGNvbGxlY3Rpb24qKiAtIENvbGxlY3Rpb24gb2YgQVBJJ3MgdG8gbWFuYWdlIHVzZXJzCjIpICoqUG9zdHMgQVBJIGNvbGxlY3Rpb24qKiAtIENvbGxlY3Rpb24gb2YgQVBJIHRvIG1hbmFnZSBwb3N0cyB0aGF0IGNhbiByYW5nZSBmcm9tIHNtYWxsIHR3ZWV0cyB0byBhIGZ1bGwgaW1hZ2UgcG9zdCBvciBldmVuIGEgc2xpZGUgb2YgaW1hZ2VzCjMpICoqTGlrZSBhbmQgQ29tbWVudCBBUEkgY29sbGVjdGlvbioqIC0gQVBJIHRvIG1hbmFnZSBsaWtlcyBhbmQgY29tbWVudHMgYXR0YWNoZWQgdG8gcG9zdHMKNCkgKipVc2VyIFJlbGF0aW9uIEFQSSBjb2xsZWN0aW9uKiogLSBDb2xsZWN0aW9uIG9mIGFwaSB0byBtYW5nZSByZWxhdGlvbiBiZXR3ZWVuIHVzZXJzIChmcmllbmRzLGZvbGxvdyxjaXJlY2xlcyBldGMpCjUpICoqTWVzc2FnaW5nIEFQSSBjb2xsZWN0aW9uKiogLSBDb2xsZWN0aW9uIG9mIGFwaSBmb3IgYnVpbGRpbmcgMS0xIG9yIGdyb3VwCjYpICoqRmVlZCBBUEkgY29sbGVjdGlvbioqIC0gQ29sbGVjdGlvbiBvZCBhcGkgdG8gbWFuYWdlIGFjdGl2aXR5IGZlZWQKCiMjIFRoaXMgc3BlY2lmaWNhdGlvbiBkZWZpbmVzIGZvbGxvd2luZyBtb2RlbDoKMSkgKipVc2VyKiogLSB0byBzdG9yZSBhbmQgcmV0cmlldmUgdXNlciBkYXRhCjIpICoqUG9zdCoqIC0gdGhpcyBpcyB0aGUgYmFzaWMgYmFja2JvbmUgb2Ygc29jaWFsIGZlYXR1cmUsIGEgcG9zdCBjYW4gYmUgYW55dGhpbmcgZnJvbSBzaG9ydCB0d2VldHMsIHRvIGltYWdlIGJhc2VkIHBvc3QgCjMpICoqRmVlZCoqIC0gdGhpcyBpcyB1c2VkIHRvIHN0b3JlIGludGVyYWN0aW9uIGJldHdlZW4gdXNlcnMgLCB1c2VyIGFuZCBwb3N0IGFuZCBvdGhlciBhY3Rpdml0eSBvZiB1c2Vycy4gRWcgOiBYIGxpa2VzIGEgcG9zdCAsIFggaXMgbm93IGZyaWVuZHMgd2l0aCBZIGV0Yy4gVGhlc2UgYWN0aXZpdGllcyBjYW4gdGhlbiBiZSBkaXNwbGF5ZWQgb24gZmVlZAo0KSAqKkxpa2UqKiAtIHN0b3JlIGxpc3Qgb2YgYWxsIHVzZXIgd2hvIGxpa2VzIGEgcGFydGljdWxhciBwb3N0CjUpICoqQ29tbWVudHMqKiAtIHN0b3JlIGFsbCB0aGUgY29tbWVudHMgZm9yIGEgcG9zdAo2KSAqKlVzZXJSZWxhdGlvbioqIC0gdGhpcyBzdG9yZXMgcmVsYXRpb24gYmV0d2VlbiB0d28gdXNlcnMgLCB0aGlzIGNhbiBiZSB1c2VkIHRvIGltcGxlbWVudCByZWxhdGlvbnMgbGlrZSAnZnJpZW5kcycsICdmb2xsb3cnIG9yIHNvbWV0aGluZyBhcyBjb21wbGV4IGFzICdjaXJjbGVzJyB0aGF0IHdhcyBpbnRyb2R1Y2VkIGJ5IEdvb2dsZSArCjcpICoqTWVzc2FnZSoqIC0gdGhpcyBzdG9yZXMgbWVzc2VnZXMgc2VudCBpbiBncm91cHMgb3IgMS0xIGludGVyYWN0aW9uCjgpICoqTWVzc2FnZVJvb20qKiAtIHRoaXMgc3RvcmVzIHRoZSBtZXNzYWdpbmcgcm9vbXMgLCByb29tcyBjYW4gYmUgMS0xIG1lc3NhZ2luZyByb29tcyBvciBncm91cCBtZXNzYWdlIHJvb21zLiBNZXNzYWdlIHJvb21zIGFyZSB3aGVyZSBtZXNzYWdlcyBhcmUgZXhjaGFuZ2VkCgojIyBMaWNlbnNlClRoaXMgcHJvamVjdCBpcyBjcmVhdGVkIGJ5IFtQcmFuamFsIFBhbmRleV0oaHR0cHM6Ly9naXRodWIuY29tL3BoeXNjb2NvZGUpIGFuZCBpcyBhdmFpbGFibGUgdW5kZXIgTUlUIGxpY2Vuc2UK readmeEtag: '"b133ee07521830b044ea3b5505ae41045863b6cc"' readmeLastModified: Sun, 31 Jan 2021 19:59:57 GMT repositoryId: 331009814 description: Open API 3.0 spec for creating social networking API created: '2021-01-19T14:37:48Z' updated: '2023-12-11T22:37:01Z' language: null archived: false stars: 4 watchers: 1 forks: 0 owner: ipranjal logo: https://avatars.githubusercontent.com/u/7591484?v=4 license: MIT repoEtag: '"73d399453dc334113b60a2ea638f1446ca941b92504c6b8d63c206c921b6924d"' repoLastModified: Mon, 11 Dec 2023 22:37:01 GMT foundInMaster: true category: Server Implementations id: ad45934ff23eea424cd904c63c539912 oldLocations: - https://github.com/scrawler-archive/openapi-spec - source: openapi3 tags repository: https://github.com/salesforce/anypoint-automation-client-generator v3: true repositoryMetadata: base64Readme: >- IyBDbG91ZGh1YiBBUEkgQ2xpZW50IEdlbmVyYXRvcgoKIyMgSW5zdGFsbAoKTWFrZSBzdXJlIHRvIGluc3RhbGwgbm9kZWpzIGFuZCBucG0gYW5kIGV4ZWN1dGUgdGhlIGZvbGxvd2luZyBpbiB0aGUgcm9vdCBwcm9qZWN0CgpgYGBiYXNoCm5wbSBpbnN0YWxsCmBgYAoKIyMgRXhlY3V0ZSBnZW5lcmF0b3IKCk1ha2Ugc3VyZSB0byBhZGQgYEFOWVBPSU5UX0dFTkVSQVRPUl9HT19ERVNUYCBlbnZpcm9ubWVudCB2YXJpYWJsZSB0aGF0IHNob3VsZCBwb2ludCB0byB0aGUgZGVzdGluYXRpb24gZ28gZm9sZGVyLiBZb3UgY2FuIHVzZSB0aGUgYGRpc3RgIGZvbGRlciBvZiB0aGUgcHJvamVjdCB3aGljaCBpcyBpZ25vcmVkIGluIGdpdCBieSBleGVjdXRpbmcgdGhlIGZvbGxvd2luZyBpbnNpZGUgdGhlIHByb2plY3QncyBmb2xkZXI6CgpgYGBiYXNoCmV4cG9ydCBBTllQT0lOVF9HRU5FUkFUT1JfR09fREVTVD1gcHdkYC9kaXN0CmBgYAoKVXNlIHRoZSBmb2xsb3dpbmcgdG8gZ2V0IHRoZSBtYW51YWwKCmBgYGJhc2gKbnB4IG9wZW5hcGktZ2VuZXJhdG9yLWNsaSBoZWxwCmBgYAoKVXNlIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB0byBnZW5lcmF0ZSB1c2luZyB0aGUgY29uZmlndXJhdGlvbiBzdG9yZWQgaW4gYG9wZW5hcGl0b29scy5qc29uYAoKYGBgYmFzaApucHggb3BlbmFwaS1nZW5lcmF0b3ItY2xpIGdlbmVyYXRlCmBgYAoKIyMgSG93IHRvIHdyaXRlIHlvdXIgc3BlYyA/CgpIZXJlJ3Mgc29tZSBydWxlcyB0byBrZWVwIGluIG1pbmQgd2hlbiB5b3Ugd3JpdGUgeW91ciBzcGVjOgoKKiBVc2UgT0FTIDMuMC4wCiogRGVmaW5lIHlvdXIgb2JqZWN0cyBzY2hlbWFzIGluIGAjL2NvbXBvbmVudHMvc2NoZW1hYCBhbmQgcmVmZXJlbmNlIHRoZW0gaW4geW91ciBwYXRoIGRlZmluaXRpb25zLgoqIFVzZSBlbmhlcml0YW5jZSBhbmQgcG9seW1vcnBoaXNtIHdoZW4geW91IGRlZmluZSB5b3VyIHNjaGVtYXMgdG8gYXZvaWQgcmVkZWZpbmluZyB0aGUgc2FtZSBhdHRyaWJ1dGVzIG11bHRpcGxlIHRpbWVzLiBbcmVmXShodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2RhdGEtbW9kZWxzL2luaGVyaXRhbmNlLWFuZC1wb2x5bW9ycGhpc20vKS4KKiBGb3IgZWFjaCBhdHRyaWJ1dGUgb2YgeW91ciBzY2hlbWFzIG1ha2Ugc3VyZSB0byBwcm92aWRlIGEgYHRpdGxlYC4gVGhlIGB0aXRsZWAgd2lsbCBiZSB1c2VkIHRvIGdlbmVyYXRlIGEgc3RydWN0dXJlIG5hbWUgZm9yIHRoYXQgYXR0cmlidXRlLCB0aGVyZWZvcmUgdXNlIGEgc2hvcnQgYW5kIGNvbnNpdGVudCBuYW1lLiBsb29rIGluIHRoZSBleGlzdGVudCBzcGVjcyBmb3IgZXhhbXBsZXMuCiogVXNlICoqY2FtbCBjYXNlKiogaW4gYHRpdGxlYCB0byBuYW1lIHlvdXIgc2NoZW1hcyBhdHRyaWJ1dGVzLgoqIFdoZW4gdXNpbmcgYHRpdGxlYCB0byBkZXNjcmliZSBhbiBvYmplY3Qgb3IgYXJyYXksIGl0cyBuYW1lIHNob3VsZCBiZSB1bmlxdWUgYWNjcm9zcyB0aGUgc2FtZSBzcGVjaWZpY2F0aW9uLgoqIEZvciBlYWNoIG9mIHlvdXIgYXBpIHJlc291cmNlLCByZXR1cm4gYSBkZXRlcm1pbmlzdGljIHJlc3VsdCBhbmQgbmV2ZXIgdXNlciBPbmVPZiBmb3IgZXhhbXBsZS4gVGhlIGdlbmVyYXRvciB3aWxsIG5vdCBiZSBhYmxlIHRvIGNyZWF0ZSB0aGUgYXBwcm9wcmlhdGUgVHlwZS4KKiBGb3IgYWxsIHNpbXBsZSB0eXBlcyAoZS5nIGludGVnZXIsIHN0cmluZyAuLi4pIERvbid0IHVzZSBgJHJlZmAgdG8gcmVmZXIgdG8gdGhlIGRlZmluaXRpb24uCiogTWFrZSBzdXJlIHRvIHJlbW92ZSBhbnkgdW5uZWNlc3NhcnkgYHJlcXVpcmVkYCBhdHRyaWJ1dGVzLgoqIFRob3VnaCB0aGUgdXNlIG9mIGJvZHkgcGFyYW1ldGVyIGlzIGRpc2NvdXJhZ2VkIGZvciBgREVMRVRFYCBvcGVyYXRpb25zLCBzb21lIEFQSXMgbWF5IHJlcXVpcmUgaXQgKHRvIGRlbGV0ZSBtdWx0aXBsZSBzcGVjaWZpYyBpdGVtcyBmb3IgZXhhbXBsZS4uLikuIEluIHRoYXQgY2FzZSBzb21lIGVkaXRvcnMgbGlrZSBTd2FnZ2VyIG1heSBzaG93IGFuIGVycm9yIHdoZW4geW91IGFkZCBhIGByZXF1ZXN0Qm9keWAgb24gYSBgREVMRVRFYCByZXNvdXJjZSBvcGVyYXRpb24uIERvbid0IHBheSBhdHRlbnRpb24gdG8gdGhhdCBlcnJvciBzaW5jZSBvdXIgY29kZSBnZW5lcmF0b3Igc3VwcG9ydHMgaXQuIElmIHlvdSBhcmUgYW5ub3llZCBieSB0aGF0LCB0cnkgdXNpbmcgeW91ciBmYXZvcml0ZSBlZGl0b3IncyBwbHVnaW4gZm9yIG9wZW5hcGkgKGV4YW1wbGUgb24gdnNjb2RlIG9wZW5hcGkgW3BsdWdpbl0oaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPTQyQ3J1bmNoLnZzY29kZS1vcGVuYXBpKSkuCiogSWYgYSBnaXZlbiBmaWVsZCBzdXBwb3J0cyBtdWx0aXBsZSB0eXBlcywgZG9uJ3QgdXNlIGBvbmVPZmAgb3IgYGFueU9mYC4gSW5zdGVhZCwgbGVhdmUgdGhlIHR5cGUgZW1wdHkgKGFkZCB0aXRsZSBmaWVsZCBvciBkZXNjcmlwdGlvbiBvbmx5IGZvciBleGFtcGxlKS4KKiBVc2UgYG9wZXJhdGlvbklkYCB0byBuYW1lIHJlc291cmNlcy4uCgojIyBSZWZlcmVuY2UKCiogW29wZW4gYXBpIGdlbmVyYXRvcl0oaHR0cHM6Ly9vcGVuYXBpLWdlbmVyYXRvci50ZWNoLykKKiBbb3BlbmFwaSBwbGFjZWhvbGRlcnMgbWF2ZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IvYmxvYi9tYXN0ZXIvbW9kdWxlcy9vcGVuYXBpLWdlbmVyYXRvci1tYXZlbi1wbHVnaW4vUkVBRE1FLm1kKQoqIFtvcGVuYXBpIGNsaSBwbGFjZWhvbGRlcnNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3ItY2xpL3RyZWUvbWFzdGVyL2FwcHMvZ2VuZXJhdG9yLWNsaS9zcmMjYXZhaWxhYmxlLXBsYWNlaG9sZGVycykKKiBbSlNPTiB0byBKU09OU2NoZW1hIENvbnZlcnRlciBUb29sXShodHRwczovL3d3dy5qc29uc2NoZW1hLm5ldC9ob21lKQoKIyMgRGlzY2xhaW1lcgoKKipUaGlzIGlzIGFuIFtvcGVuIHNvdXJjZSBzb2Z0d2FyZSwgcGxlYXNlIHJldmlldyB0aGUgY29uc2lkZXJhdGlvbnNdKExJQ0VOU0UubWQpLioqClRoaXMgaXMgYW4gb3BlbiBzb3VyY2UgcHJvamVjdCwgaXQgZG9lcyBub3QgZm9ybSBwYXJ0IG9mIHRoZSBvZmZpY2lhbCBNdWxlU29mdCBwcm9kdWN0IHN0YWNrLCBhbmQgaXMgdGhlcmVmb3JlIG5vdCBpbmNsdWRlZCBpbiBNdWxlU29mdCBzdXBwb3J0IFNMQXMuIElzc3VlcyBzaG91bGQgYmUgZGlyZWN0ZWQgdG8gdGhlIGNvbW11bml0eSwgd2hvIHdpbGwgdHJ5IHRvIGFzc2lzdCBvbiBhIGJlc3QgZW5kZWF2b3VycyBiYXNpcy4gVGhpcyBhcHBsaWNhdGlvbiBpcyBkaXN0cmlidXRlZCAqKmFzIGlzKiouCg== readmeEtag: '"f49fb3502a8d0b47b5875c31f2e595f790aa972c"' readmeLastModified: Tue, 06 Feb 2024 14:37:39 GMT repositoryId: 350462203 description: OAS specifications for Anypoint Platform Resources created: '2021-03-22T19:19:01Z' updated: '2025-11-10T14:41:44Z' language: null archived: false stars: 3 watchers: 3 forks: 15 owner: salesforce logo: https://avatars.githubusercontent.com/u/453694?v=4 license: MIT repoEtag: '"2103c40027b22640b67818c901f3ff547a73ca2cc8450516a9a189eb64772646"' repoLastModified: Mon, 10 Nov 2025 14:41:44 GMT foundInMaster: true category: Parsers id: 5169bc3ad6bc176c4877e87d5d4d22c3 oldLocations: - >- https://github.com/mulesoft-consulting/anypoint-automation-client-generator - source: openapi3 tags repository: https://github.com/llcfreedom-space/fs-itunes-search-client v3: true id: 8e18353fd2f762ba3c4be5a9e22f89d9 repositoryMetadata: base64Readme: >- IyBGU0lUdW5lc1NlYXJjaENsaWVudAoKWyFbU3dpZnQgVmVyc2lvbl1bc3dpZnQtaW1hZ2VdXVtzd2lmdC11cmxdClshW0xpY2Vuc2VdW2xpY2Vuc2UtaW1hZ2VdXVtsaWNlbnNlLXVybF0KIVtHaXRIdWIgcmVsZWFzZSAod2l0aCBmaWx0ZXIpXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3JlbGVhc2UvTExDRnJlZWRvbS1TcGFjZS9mcy1pdHVuZXMtc2VhcmNoLWNsaWVudCkKIFshW1JlYWQgdGhlIERvY3NdKGh0dHBzOi8vcmVhZHRoZWRvY3Mub3JnL3Byb2plY3RzL2RvY3MvYmFkZ2UvP3ZlcnNpb249bGF0ZXN0KV0oaHR0cHM6Ly9sbGNmcmVlZG9tLXNwYWNlLmdpdGh1Yi5pby9mcy1pdHVuZXMtc2VhcmNoLWNsaWVudC8pCiFbZXhhbXBsZSB3b3JrZmxvd10oaHR0cHM6Ly9naXRodWIuY29tL0xMQ0ZyZWVkb20tU3BhY2UvZnMtaXR1bmVzLXNlYXJjaC1jbGllbnQvYWN0aW9ucy93b3JrZmxvd3MvZG9jYy55bWwvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKQohW2V4YW1wbGUgd29ya2Zsb3ddKGh0dHBzOi8vZ2l0aHViLmNvbS9MTENGcmVlZG9tLVNwYWNlL2ZzLWl0dW5lcy1zZWFyY2gtY2xpZW50L2FjdGlvbnMvd29ya2Zsb3dzL2xpbnQueW1sL2JhZGdlLnN2Zz9icmFuY2g9bWFpbikKIVtleGFtcGxlIHdvcmtmbG93XShodHRwczovL2dpdGh1Yi5jb20vTExDRnJlZWRvbS1TcGFjZS9mcy1pdHVuZXMtc2VhcmNoLWNsaWVudC9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LnltbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pCiBbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2l0aHViL0xMQ0ZyZWVkb20tU3BhY2UvZnMtaXR1bmVzLXNlYXJjaC1jbGllbnQvZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPTJFVUlBNE9HUzkpXShodHRwczovL2NvZGVjb3YuaW8vZ2l0aHViL0xMQ0ZyZWVkb20tU3BhY2UvZnMtaXR1bmVzLXNlYXJjaC1jbGllbnQpCgpUaGlzIFN3aWZ0IHBhY2thZ2UgcHJvdmlkZXMgYSBjbGllbnQgbGlicmFyeSBmb3IgaW50ZXJhY3Rpbmcgd2l0aCB0aGUgaVR1bmVzIFNlYXJjaCBBUEkuIEl0IGxldmVyYWdlcyB0aGUgcG93ZXIgb2YgW1N3aWZ0IE9wZW5BUEkgR2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vYXBwbGUvc3dpZnQtb3BlbmFwaS1nZW5lcmF0b3IvdHJlZS9tYWluKSB0byBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlIGNvZGUgZnJvbSB0aGUgcHJvdmlkZWQgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZpbGUgKG9wZW5hcGkueWFtbCkuCgojIyBGZWF0dXJlcwoKKiBGZXRjaCBhcHAgaW5mb3JtYXRpb24gYnkgYnVuZGxlIElEIGFuZCBvcHRpb25hbCBjb3VudHJ5IGNvZGUuCiogSGFuZGxlcyBkaWZmZXJlbnQgcmVzcG9uc2UgZm9ybWF0cyAoSlNPTiBhbmQgSlNPTlApLgoqIFRocm93cyBpbmZvcm1hdGl2ZSBlcnJvcnMgZm9yIHNlcnZlciBlcnJvcnMgYW5kIG5vdCBmb3VuZCBjYXNlcy4KCiMjIEluc3RhbGxhdGlvbgoKMS4gQWRkIHRoZSBwYWNrYWdlIGRlcGVuZGVuY3kgdG8geW91ciBQYWNrYWdlLnN3aWZ0IGZpbGU6CgpgYGBzd2lmdApkZXBlbmRlbmNpZXM6IFsKLnBhY2thZ2UodXJsOiAiaHR0cHM6Ly9naXRodWIuY29tL0xMQ0ZyZWVkb20tU3BhY2UvZnMtaXR1bmVzLXNlYXJjaC1jbGllbnQiLCBmcm9tOiAiMS4wLjAiKQpdCmBgYAoKMi4gSW1wb3J0IHRoZSBsaWJyYXJ5IGluIHlvdXIgU3dpZnQgY29kZToKCmBgYHN3aWZ0CmltcG9ydCBJVHVuZXNTZWFyY2hDbGllbnQKYGBgCgojIyBVc2FnZQoKSGVyZSdzIGFuIGV4YW1wbGUgb2YgaG93IHRvIHVzZSB0aGUgSVR1bmVzU2VhcmNoQ2xpZW50IHRvIGZldGNoIGluZm9ybWF0aW9uIGFib3V0IGFuIGFwcDoKCmBgYHN3aWZ0CmxldCBjbGllbnQgPSB0cnkgSVR1bmVzU2VhcmNoQ2xpZW50KCkKCmRvIHsKbGV0IGFwcEluZm8gPSB0cnkgYXdhaXQgY2xpZW50LmZldGNoQXBwSW5mbyhieTogImNvbS5leGFtcGxlLk15QXBwIikKcHJpbnQoIkFwcCBWZXJzaW9uOiBcKGFwcEluZm8udmVyc2lvbiA/PyAiTm90IGF2YWlsYWJsZSIpIikKcHJpbnQoIkJ1bmRsZSBJRDogXChhcHBJbmZvLmJ1bmRsZUlkISkiKQovLyAuLi4gYWNjZXNzIG90aGVyIHByb3BlcnRpZXMgb2YgQXBwSW5mbwp9IGNhdGNoIHsKcHJpbnQoIkVycm9yIGZldGNoaW5nIGFwcCBpbmZvOiBcKGVycm9yKSIpCn0KYGBgCgojIyBDb250cmlidXRpb25zCgpXZSB3ZWxjb21lIGNvbnRyaWJ1dGlvbnMgdG8gdGhpcyBwcm9qZWN0ISBQbGVhc2UgZmVlbCBmcmVlIHRvIG9wZW4gaXNzdWVzIG9yIHB1bGwgcmVxdWVzdHMgdG8gaGVscCBpbXByb3ZlIHRoZSBwYWNrYWdlLgoKIyMgTGlua3MKCkxMQyBGcmVlZG9tIFNwYWNlIOKAkyBbQExMQ0ZyZWVkb21TcGFjZV0oaHR0cHM6Ly90d2l0dGVyLmNvbS9sbGNmcmVlZG9tc3BhY2UpIOKAkyBbc3VwcG9ydEBmcmVlZG9tc3BhY2UuY29tcGFueV0obWFpbHRvOnN1cHBvcnRAZnJlZWRvbXNwYWNlLmNvbXBhbnkpCgpEaXN0cmlidXRlZCB1bmRlciB0aGUgR05VIEFGRkVSTyBHRU5FUkFMIFBVQkxJQyBMSUNFTlNFIFZlcnNpb24gMy4gU2VlIFtMSUNFTlNFLm1kXVtsaWNlbnNlLXVybF0gZm9yIG1vcmUgaW5mb3JtYXRpb24uCgogW0dpdEh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL0xMQ0ZyZWVkb20tU3BhY2UpCgpbc3dpZnQtaW1hZ2VdOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvc3dpZnQtNS44LW9yYW5nZS5zdmcKW3N3aWZ0LXVybF06IGh0dHBzOi8vc3dpZnQub3JnLwpbbGljZW5zZS1pbWFnZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTGljZW5zZS1HUEx2My1ibHVlLnN2ZwpbbGljZW5zZS11cmxdOiBMSUNFTlNFCg== readmeEtag: '"3238de09b1e741b1364b00851754bc8ca20bd693"' readmeLastModified: Wed, 17 Apr 2024 08:55:29 GMT repositoryId: 782396471 description: >- This Swift package provides a client library for interacting with the iTunes Search API. It leverages the power of Swift OpenAPI Generator to automatically generate code from the provided OpenAPI specification file (openapi.yaml). created: '2024-04-05T08:09:57Z' updated: '2024-07-30T14:45:17Z' language: Swift archived: false stars: 4 watchers: 1 forks: 0 owner: LLCFreedom-Space logo: https://avatars.githubusercontent.com/u/144135921?v=4 license: AGPL-3.0 repoEtag: '"fbd39c86f29abc69acf06d936b5bbe16473c1d964588d61a4ccb08f999193c2f"' repoLastModified: Tue, 30 Jul 2024 14:45:17 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/onrik/gaws v3: true id: 26ff5e96931349992339526fe9eab9cb repositoryMetadata: base64Readme: >- IyBHYXdzCgpPcGVuQVBJIChzd2FnZ2VyKSBkb2NzIGdlbmVyYXRvciBmb3IgR29sYW5nLgoKIyMgRXhhbXBsZXMKCmBgYGdvbGFuZwpwYWNrYWdlIHVzZXJzCgppbXBvcnQgKAoJIm5ldC9odHRwIgoJInRpbWUiCikKCnR5cGUgR3JvdXAgc3RydWN0IHsKCU5hbWUgc3RyaW5nIGBqc29uOiJuYW1lImAKfQoKdHlwZSBVc2VyIHN0cnVjdCB7CglfICAgICAgICAgc3RydWN0e30gIGBqc29uOiItIiBvcGVuYXBpRGVzYzoiVXNlciJgIC8vIERlc2NyaXB0aW9uIGZvciBlbml0cmUgc3RydWN0CglJRCAgICAgICAgdWludCAgICAgIGBqc29uOiJpZCJgCglOYW1lICAgICAgc3RyaW5nICAgIGBqc29uOiJuYW1lIiBvcGVuYXBpRXh0OiJ4LWV4dC1rZXk6ZXh0LXZhbHVlImAgLy8gIFBhc3Npbmcgb3BlbmFwaSBleHRlbnNpb25zIHRvIHNwZWMgYXMgaXMKCUVtYWlsICAgICBzdHJpbmcgICAgYGpzb246ImVtYWlsIiBvcGVuYXBpRGVzYzoiVXNlcidzIGVtYWlsImAgLy8gRGVzY3JpcHRpb24gZm9yIGN1cnJlbnQgZmllbGQKCUlzQWRtaW4gICBib29sICAgICAgYGpzb246ImlzX2FkbWluImAKCUdyb3VwcyAgICBbXUdyb3VwICAgYGpzb246Imdyb3VwcyJgCglDcmVhdGVkQXQgdGltZS5UaW1lIGBqc29uOiJjcmVhdGVkX2F0ImAKCVN0YXR1cyAgICBzdHJpbmcgICAgYGpzb246InN0YXR1cyIgb3BlbmFwaUVudW06Im5ldyxjb25maXJtZWQsZGVsZXRlZCJgIC8vIEVudW0gZm9yIGN1cnJlbnQgZmllbGQgdmFsdWVzCQp9Cgp0eXBlIGNyZWF0ZVVzZXJSZXF1ZXN0IHN0cnVjdCB7CglFbWFpbCAgICAgc3RyaW5nICAgYGpzb246ImVtYWlsImAKCU5hbWUgICAgICBzdHJpbmcgICBganNvbjoibmFtZSJgCglQYXNzd29yZDEgc3RyaW5nICAgYGpzb246InBhc3N3b3JkMSJgCglQYXNzd29yZDIgc3RyaW5nICAgYGpzb246InBhc3N3b3JkMiJgCglJc0FkbWluICAgYm9vbCAgICAgYGpzb246ImlzX2FkbWluImAKCUdyb3VwcyAgICBbXXN0cmluZyBganNvbjoiZ3JvdXBzImAKfQoKdHlwZSB1cGRhdGVVc2VyUmVxdWVzdCBzdHJ1Y3QgewoJRW1haWwgICAgKnN0cmluZyAgIGBqc29uOiJlbWFpbCxvbWl0ZW1wdHkiYAoJTmFtZSAgICAgKnN0cmluZyAgIGBqc29uOiJuYW1lLG9taXRlbXB0eSJgCglQYXNzd29yZCAqc3RyaW5nICAgYGpzb246InBhc3N3b3JkLG9taXRlbXB0eSJgCglJc0FkbWluICAqYm9vbCAgICAgYGpzb246ImlzX2FkbWluLG9taXRlbXB0eSJgCglHcm91cHMgICAqW11zdHJpbmcgYGpzb246Imdyb3VwcyxvbWl0ZW1wdHkiYAp9CgovKgpVc2VycyByZXR1cm5zIHVzZXJzIGxpc3QKQG9wZW5hcGkgR0VUIC9hcGkvdjEvdXNlcnMKQG9wZW5hcGlQYXJhbSBxIGluPXF1ZXJ5LCB0eXBlPXN0cmluZywgZXhhbXBsZT1Kb2huCkBvcGVuYXBpUmVzcG9uc2UgMjAwIGFwcGxpY2F0aW9uL2pzb24geyJ1c2VycyI6IFtdVXNlcn0KKi8KZnVuYyBVc2Vycyh3IGh0dHAuUmVzcG9uc2VXcml0ZXIsIHIgKmh0dHAuUmVxdWVzdCkgewp9CgovKgpDcmVhdGVVc2VyIGNyZWF0ZXMgdXNlcgpAb3BlbmFwaSBQT1NUIC9hcGkvdjEvdXNlcnMKQG9wZW5hcGlSZXF1ZXN0IGFwcGxpY2F0aW9uL2pzb24gY3JlYXRlVXNlclJlcXVlc3QKQG9wZW5hcGlSZXNwb25zZSA0MDAgYXBwbGljYXRpb24vanNvbiB7Im1lc3NhZ2UiOiAiZW1haWw9ZW1haWw7bmFtZT1yZXF1aXJlZCJ9CkBvcGVuYXBpUmVzcG9uc2UgMjAwIGFwcGxpY2F0aW9uL2pzb24geyJ1c2VyIjogVXNlcn0KKi8KZnVuYyBDcmVhdGVVc2VyKHcgaHR0cC5SZXNwb25zZVdyaXRlciwgciAqaHR0cC5SZXF1ZXN0KSB7Cn0KCi8qClVwZGF0ZVVzZXIgdXBkYXRlcyB1c2VyCkBvcGVuYXBpIFBPU1QgL2FwaS92MS91c2Vycy97aWR9CkBvcGVuYXBpUGFyYW0gaWQgaW49cGF0aCwgdHlwZT1pbnQsIGV4YW1wbGU9NTYKQG9wZW5hcGlSZXF1ZXN0IGFwcGxpY2F0aW9uL2pzb24gdXBkYXRlVXNlclJlcXVlc3QKQG9wZW5hcGlSZXNwb25zZSA0MDQgYXBwbGljYXRpb24vanNvbiB7Im1lc3NhZ2UiOiAiTm90IEZvdW5kIn0KQG9wZW5hcGlSZXNwb25zZSAyMDAgYXBwbGljYXRpb24vanNvbiB7InVzZXIiOiBVc2VyfQoqLwpmdW5jIFVwZGF0ZVVzZXIodyBodHRwLlJlc3BvbnNlV3JpdGVyLCByICpodHRwLlJlcXVlc3QpIHsKfQoKLyoKRGVsZXRlVXNlciBkZWxldGUgdXNlcgpAb3BlbmFwaSBERUxFVEUgL2FwaS92MS91c2Vycy97aWR9CkBvcGVuYXBpUGFyYW0gaWQgaW49cGF0aCwgdHlwZT1pbnQsIGV4YW1wbGU9NTYKQG9wZW5hcGlSZXNwb25zZSA0MDQgYXBwbGljYXRpb24vanNvbiB7Im1lc3NhZ2UiOiAiTm90IEZvdW5kIn0KQG9wZW5hcGlSZXNwb25zZSAyMDAgYXBwbGljYXRpb24vanNvbiB7fQoqLwpmdW5jIERlbGV0ZVVzZXIodyBodHRwLlJlc3BvbnNlV3JpdGVyLCByICpodHRwLlJlcXVlc3QpIHsKfQoKYGBgCg== readmeEtag: '"150b2a4f9a3e3eec503db50501996759055b440e"' readmeLastModified: Mon, 08 Apr 2024 05:16:18 GMT repositoryId: 515461559 description: OpenAPI doc generator for Golang created: '2022-07-19T06:25:23Z' updated: '2025-10-14T07:27:40Z' language: Go archived: false stars: 4 watchers: 1 forks: 3 owner: onrik logo: https://avatars.githubusercontent.com/u/4413900?v=4 license: MIT repoEtag: '"52d2df0e7deffae2e33df2c4c438ed81d3a9dbaddd1ab3ec358ae968eeb2956d"' repoLastModified: Tue, 14 Oct 2025 07:27:40 GMT foundInMaster: true category: Parsers - source: openapi3 tags repository: https://github.com/ondata/taskforse.it v3: true repositoryMetadata: base64Readme: >- IyBUYXNrIEZvcnNlCgo+IE5lbCBkdWJiaW8uLi4gdGFzayBmb3JzZSEKCiJUYXNrIEZvcnNlIiBpcyBhbiBpbmRlcGVuZGVudCBwcm9qZWN0IGZvciBtb25pdG9yaW5nIG9mIHB1YmxpYyB0YXNrIGZvcmNlcycgYWN0aXZpdHkgZHVyaW5nIENPVklELTE5IGVtZXJnZW5jeSBpbiAyMDIwIGluIEl0YWx5OiBodHRwczovL3d3dy50YXNrZm9yc2UuaXQvLgoKSXQncyBhIG5vLXByb2ZpdCBhbmQgY29tbXVuaXR5LWRyaXZlbiBpbml0aWF0aXZlIHN1cHBvcnRlZCBieSBbT25kYXRhXShodHRwczovL29uZGF0YS5pdCkgYW5kIGJhc2VkIG9uIGEgc2hhcmVkIGRhdGFiYXNlIGhvc3RlZCBieSBbR29vZ2xlIFNoZWV0c10oaHR0cHM6Ly93d3cuZ29vZ2xlLmNvbS9pbnRsL2l0L3NoZWV0cy9hYm91dC8pIGFuZCBbR29vZ2xlIEZvcm1dKGh0dHBzOi8vd3d3Lmdvb2dsZS5pdC9pbnRsL2l0L2Zvcm1zL2Fib3V0LykgYW5kIGEgd2ViIGFwcGxpY2F0aW9uIGJ1aWx0IG9uIHRvcCBvZiBbTmV4dEpTXShodHRwczovL25leHRqcy5vcmcvKS4KClRoZSBuYW1lICJUYXNrIEZvcnNlIiBpcyBhIGpva2UgaW4gSXRhbGlhbiBsYW5ndWFnZTogX2ZvcmNlXyBpbiBFbmdsaXNoIGFuZCBfZm9yc2VfIGluIEl0YWxpYW4gKF9tYXliZV8gaW4gRW5nbGlzaCkgc291bmQgc2ltaWxhciwgc28gaW4gSXRhbHkgdGhlIHdvcmQgX3Rhc2sgZm9yc2VfIHN1Z2dlc3RzIHNvbWUgYW1vdW50IG9mIHVuY2VydGFpbnR5IGluIHRoZSBtYW5hZ2VtZW50IG9mIGVtZXJnZW5jeSB1c2luZyB0YXNrIGZvcmNlcyBhcHBvaW50ZWQgYnkgdGhlIGdvdmVybm1lbnQgYW5kIGxvY2FsIHB1YmxpYyBpbnN0aXR1dGlvbnMuCgojIyBEYXRhYmFzZQoKU2hhcmVkIGRhdGFiYXNlOiBbMTVMbUNpWUtnMmNXem92QWlxcXVocF9sWXNhQlN1R05hdTdzdVVrUWRkbDhdKGh0dHBzOi8vZG9jcy5nb29nbGUuY29tL3NwcmVhZHNoZWV0cy9kLzE1TG1DaVlLZzJjV3pvdkFpcXF1aHBfbFlzYUJTdUdOYXU3c3VVa1FkZGw4L2VkaXQ/dXNwPXNoYXJpbmcpLgoKU2hlZXRzOgotIFtNZXRhXShodHRwczovL2RvY3MuZ29vZ2xlLmNvbS9zcHJlYWRzaGVldHMvZC8xNUxtQ2lZS2cyY1d6b3ZBaXFxdWhwX2xZc2FCU3VHTmF1N3N1VWtRZGRsOC9lZGl0I2dpZD0wKTogZGF0YWJhc2UgbWV0YWRhdGEgd2l0aCB0YWJsZXMgYW5kIGZpZWxkcyBkZXNjcmlwdGlvbnMuCi0gW1Rhc2sgZm9yc2VzXShodHRwczovL2RvY3MuZ29vZ2xlLmNvbS9zcHJlYWRzaGVldHMvZC8xNUxtQ2lZS2cyY1d6b3ZBaXFxdWhwX2xZc2FCU3VHTmF1N3N1VWtRZGRsOC9lZGl0I2dpZD0xNDI4MTQ0NDQwKSAodGFzayBmb3JjZXMpOiB0YWJsZSB3aXRoIGFsbCBtb25pdG9yZWQgdGFzayBmb3JjZXMKLSBbTWVtYnJpXShodHRwczovL2RvY3MuZ29vZ2xlLmNvbS9zcHJlYWRzaGVldHMvZC8xNUxtQ2lZS2cyY1d6b3ZBaXFxdWhwX2xZc2FCU3VHTmF1N3N1VWtRZGRsOC9lZGl0I2dpZD04ODY1MTk2MTYpIChtZW1iZXJzKTogdGFibGUgd2l0aCBhbGwga25vd24gbWVtYmVycyB3aGljaCBhcmUgcGFydCBvZiBhdCBsZWFzdCBvbmUgdGFzayBmb3JjZQotIFtWZXJiYWxpXShodHRwczovL2RvY3MuZ29vZ2xlLmNvbS9zcHJlYWRzaGVldHMvZC8xNUxtQ2lZS2cyY1d6b3ZBaXFxdWhwX2xZc2FCU3VHTmF1N3N1VWtRZGRsOC9lZGl0I2dpZD05Mzk0Nzg3ODQpIChtaW51dGVzKTogdGFibGUgd2l0aCBhbGwgb2ZmaWNpYWwgbWludXRlcyBwcm9kdWNlZCBhbmQgcHlibGlzaGVkIGJ5IHRhc2sgZm9yY2VzCi0gW1Jpc29yc2VdKGh0dHBzOi8vZG9jcy5nb29nbGUuY29tL3NwcmVhZHNoZWV0cy9kLzE1TG1DaVlLZzJjV3pvdkFpcXF1aHBfbFlzYUJTdUdOYXU3c3VVa1FkZGw4L2VkaXQjZ2lkPTM2OTc2ODY4NykgKHJlc291cmNlcyk6IHRhYmxlIHdpdGggYWxsIHNvcnQgb2YgZXh0ZXJuYWwgcmVzb3VyY2VzIHJlbGF0ZWQgdG8gdGFzayBmb3JjZXMnIGFjdGl2aXR5CgpZb3UgY2FuIGZyZWVseSBhY2Nlc3MgdGhlIHNoYXJlZCBkYXRhYmFzZSwgZG93bmxvYWQgaXQgaW4gYWxsIGZvcm1hdHMgc3VwcG9ydGVkIGJ5IEdvb2dsZSBTaGVldCBhbmQgcmV1c2UgaXQgaW4gYWdyZWVtZW50IHdpdGggYSBbQ3JlYXRpdmUgQ29tbW9ucyBBdHRyaWJ1dGlvbiAoQ0MtQlkpXShodHRwczovL2NyZWF0aXZlY29tbW9ucy5vcmcvbGljZW5zZXMvYnkvNC4wL2RlZWQuaXQpIGxpY2Vuc2UuCgojIyBXZWIgYXBwbGljYXRpb24KClJ1bm5pbmcgaW5zdGFuY2U6IGh0dHBzOi8vd3d3LnRhc2tmb3JzZS5pdC8uCgojIyMgU291cmNlIGNvZGUKClB1YmxpYyByZXBvc2l0b3J5OiBodHRwczovL2dpdGh1Yi5jb20vb25kYXRhL3Rhc2tmb3JzZS5pdC4KClNvdXJjZSBjb2RlIGlzIHJlbGVhc2VkIHVuZGVyIGEgW01JVF0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQpIGxpY2Vuc2Ugb24gR2l0aHViLiBDb250cmlidXRpb25zIGFyZSB3ZWxjb21lLiBGZWVsIGZyZWUgdG8gb3BlbiBpc3N1ZXMgYW5kIHN1Ym1pdCBhIHB1bGwgcmVxdWVzdCBhdCBhbnkgdGltZSEKCiMjIyBTb2Z0d2FyZSBhcmNoaXRlY3R1cmUKClRoaXMgYXBwbGljYXRpb24gaXMgY29tcG9zZWQgYnkgdGhyZWUgc2VydmljZXM6Ci0gYSByZXZlcnNlIHByb3h5IHdpdGggY2FjaGluZyBhcyBhIG1pZGRsZXdhcmUgYmV0d2VlbiBjbGllbnQgYW5kIG5leHRqcyBhcHBsaWNhdGlvbiBhbmQgYmV0d2VlbiBuZXh0anMgYXBwbGljYXRpb24gYW5kIHJlbW90ZSBnb29nbGUgc2hlZXQKLSBhIG5leHRqcyBhcHBsaWNhdGlvbiB3aXRoIFNTUiBlbmFibGVkCi0gYW4gaW50ZXJhY3RpdmUgQVBJIGRvY3VtZW50YXRpb24KClRoZSByZW1vdGUgZGF0YWJhc2UgaXMgYSBzaW1wbGUgR29vZ2xlIFNoZWV0IHdpdGggcHVibGljIHZpc2liaWxpdHkgYWNjZXNzZWQgYnkgbmV4dGpzIGFwcGxpY2F0aW9uIG9ubHkgb24gc2VydmVyLXNpZGUuCgojIyMgUHJvZHVjdGlvbgoKWW91IGNhbiBkZXBsb3kgYW5kIHJ1biB0aGUgYXBwbGljYXRpb24gaW4gYSBwcm9kdWN0aW9uIGVudmlyb25tZW50IHVzaW5nIFtEb2NrZXJdKGh0dHBzOi8vd3d3LmRvY2tlci5jb20vKS4gQSBgZG9ja2VyLWNvbXBvc2UueW1sYCBpcyBwcm92aWRlZCB0byBzaW1wbGlmeSBkZXBsb3kgdXNpbmcgW0RvY2tlciBDb21wb3NlXShodHRwczovL2RvY3MuZG9ja2VyLmNvbS9jb21wb3NlLykgdXRpbGl0eS4KClRoZSBhcHBsaWNhdGlvbiBpcyBjb21wb3NlZCBieSAzIGNvbnRhaW5lcnMgYmFzZWQgb24gaW1hZ2VzIGhvc3RlZCBieSBbR2l0aHViIFBhY2thZ2VzXShodHRwczovL2dpdGh1Yi5jb20vZmVhdHVyZXMvcGFja2FnZXMpOiBodHRwczovL2dpdGh1Yi5jb20vb25kYXRhL3Rhc2tmb3JzZS5pdC9wYWNrYWdlcy4KLSBbYXBwOmxhdGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL29uZGF0YS90YXNrZm9yc2UuaXQvcGFja2FnZXMvMjMzMTcxKTogbmV4dGpzIGFwcGxpY2F0aW9uCi0gW3Byb3h5OmxhdGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL29uZGF0YS90YXNrZm9yc2UuaXQvcGFja2FnZXMvMjMzMTY5KTogbmdpbnggcmV2ZXJzZSBwcm94eSB3aXRoIGNhY2hpbmcKLSBbb3NhOmxhdGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL29uZGF0YS90YXNrZm9yc2UuaXQvcGFja2FnZXMvMjMzOTE4KTogc3dhZ2dlciB1aSBmb3IgQVBJIGRvY3VtZW50YXRpb24KCllvdSBjYW4gc2ltcGx5IGNvcHkgYW5kIHBhc3RlIHRoZSBbZG9ja2VyLWNvbXBvc2UueW1sIGZpbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9vbmRhdGEvdGFza2ZvcnNlLml0L2Jsb2IvbWFzdGVyL2RvY2tlci1jb21wb3NlLnltbCkgcHJvdmlkZWQgYW5kIHJ1biBgZG9ja2VyLWNvbXBvc2UgdXBgLiBUaGVuIHlvdSBjYW4gb3BlbiBgaHR0cDovL2xvY2FsaG9zdDo4MDgwYCB1c2luZyB5b3VyIGZhdm91cml0ZSBicm93c2VyLgoKWW91IGNhbiBzZXQgZW52aXJvbm1lbnQgdmFyaWFibGVzIHVzaW5nIGEgYC5lbnZgIGZpbGU6Ci0gUFJPWFlfVVJMIC0gVVJMIG9mIG5naW54IHNlcnZlciB3aXRoIHBhdGggZm9yIHByb3hpZWQgR29vZ2xlIFNoZWV0LCBkZWZhdWx0OiBodHRwOi8vcHJveHkvZGIKLSBBUFBfVVJMIC0gVVJMIG9mIG5leHRqcyBhcHBsaWNhdGlvbiwgZGVmYXVsdDogaHR0cDovL2FwcDozMDAwCi0gT0FTX1VSTCAtIFVSTCBvZiBzd2FnZ2VyIHVpIHNlcnZpY2UsIGRlZmF1bHQ6IGh0dHA6Ly9vYXMtdWk6ODA4MAotIFBST0RfUE9SVCAtIFBvcnQgZXhwb3NlZCBvbiBob3N0LCBkZWZhdWx0OiA4MDgwCgojIyMgRGV2ZWxvcG1lbnQKCklmIHlvdSB3YW50IHRvIGNvbnRyaWJ1dGUgdG8gdGhlIGRldmVsb3BtZW50LCB5b3UgY2FuIGZvcmsgYW5kIGNsb25lIHRoaXMgcmVwb3NpdG9yeSwgbWFrZSB5b3VyIGNoYW5nZXMgYW5kIGZpbmFsbHkgc2VuZCBhIHB1bGwgcmVxdWVzdC4KCk5leHRKUyBpcyBhIG5vZGUgYXBwbGljYXRpb24sIHNvIGFmdGVyIGNsb25pbmcgeW91IG11c3QgaW5zdGFsbCBkZXBlbmRlbmNpZXMgcnVubmluZyBgbnBtIGluc3RhbGxgLgoKTWFpbiBkZXBlbmRlbmNpZXM6Ci0gW0ZvbnQgQWF3ZXNvbWUgaWNvbnMgZm9yIHJlYWN0XShodHRwczovL2ZvcnRhd2Vzb21lLmNvbS8pCi0gW01hdGVyaWFsLVVJXShodHRwczovL21hdGVyaWFsLXVpLmNvbS8pCi0gW0F4aW9zXShodHRwczovL2dpdGh1Yi5jb20vYXhpb3MvYXhpb3MpCi0gW0xvZGFzaF0oaHR0cHM6Ly9sb2Rhc2guY29tLykKLSBbTmV4dEpTXShodHRwczovL25leHRqcy5vcmcvKQotIFtSZWFjdF0oaHR0cHM6Ly9yZWFjdGpzLm9yZy8pCgpZb3UgY2FuIHN0YXJ0IGRldmVsb3BtZW50IHNlcnZlciB3aXRoIGhvdCByZWxvYWRpbmcgcnVubmluZyBgbnBtIHJ1biBkZXZgLiBZb3UgY2FuIGFsc28gdGVzdCB0aGUgcHJvZHVjdGlvbiBpbnN0YW5jZSBydW5uaW5nIGBucG0gcnVuIHN0YXJ0YC4gVGhlbiB5b3UgY2FuIG9wZW4gYGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMGAgdXNpbmcgeW91ciBmYXZvdXJpdGUgYnJvd3Nlci4gTm90ZSB0aGF0IHJldmVyc2UgcHJveHkgYW5kIGNhY2hpbmcgYXJlIHVuYXZhaWxhYmxlIG91dHNpZGUgZG9ja2VyIGNvbnRhaW5lcnMuCgojIyMgQVBJCgpBbGwgZGF0YSBhcmUgYXZhaWxhYmxlIG9uIGEgUkVTVCBBUEkgKG9ubHkgR0VUIHZlcmIgaXMgc3VwcG9ydGVkKTogaHR0cHM6Ly93d3cudGFza2ZvcnNlLml0L2FwaS92MS4gVGhlcmUgaXMgYWxzbyBhbiBpbnRlcmFjdGl2ZSBkb2N1bWVudGF0aW9uIHBvd2VyZWQgYnkgW1N3YWdnZXIgVUldKGh0dHBzOi8vc3dhZ2dlci5pby90b29scy9zd2FnZ2VyLXVpLykgYW5kIGNvbXBsaWFudCB0byBbT3BlbkFQSSBTcGVjaWZpY2F0aW9uIHYzXShodHRwOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMyk6IGh0dHBzOi8vd3d3LnRhc2tmb3JzZS5pdC9vYXMvLgoKSWYgeW91IHdhbnQgdG8gY29udHJpYnV0ZSB0byB0aGUgQVBJIGRvY3VtZW50YXRpb24sIHlvdSBjYW4gcnVuIFtTd2FnZ2VyIEVkaXRvcl0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3Rvb2xzL3N3YWdnZXItZWRpdG9yLykgdXNpbmcgdGhlIGRvY2tlci1jb21wb3NlLnltbCBmaWxlIGluIGBvYXMvYCBmb2xkZXIuCgo+IFdhcm5pbmc6IFtDT1JTXShodHRwczovL2RldmVsb3Blci5tb3ppbGxhLm9yZy9pdC9kb2NzL1dlYi9IVFRQL0NPUlMpIGlzIGRpc2FibGVkIGJ5IGRlZmF1bHQgZm9yIGFsbCBvcmlnaW5zLCBpZiB5b3UgbmVlZCB0byBhY2Nlc3MgQVBJIGZyb20gYSBjbGllbnQtc2lkZSBhcHBsaWNhdGlvbiBmZWVsIGZyZWUgdG8gb3BlbiBhbiBpc3N1ZS4KCiMjIExpY2Vuc2UKClRoZSBzb3VyY2UgY29kZSBpcyByZWxlYXNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkuCgpUaGUgZGF0YWJhc2UgaXMgcmVsZWFzZWQgdW5kZXIgdGhlIFtDcmVhdGl2ZSBDb21tb25zIEF0dHJpYnV0aW9uIExpY2Vuc2VdKGh0dHBzOi8vY3JlYXRpdmVjb21tb25zLm9yZy9saWNlbnNlcy9ieS80LjAvZGVlZC5pdCkuCg== readmeEtag: '"6eb1c6af6a1ec993f84a535e8a9c172bd7d380a0"' readmeLastModified: Fri, 07 Aug 2020 13:21:20 GMT repositoryId: 257328202 description: Nel dubbio... task forse! created: '2020-04-20T15:43:25Z' updated: '2021-11-01T15:42:59Z' language: JavaScript archived: false stars: 4 watchers: 6 forks: 0 owner: ondata logo: https://avatars.githubusercontent.com/u/10263296?v=4 license: MIT repoEtag: '"71d1654db6ce6bfd132c8bb1a7f76efe2f64a7d9ea1b7ddf59f68145cb4cbd1f"' repoLastModified: Mon, 01 Nov 2021 15:42:59 GMT foundInMaster: true category: Server Implementations id: d262bd466cd5b168d2d9100464ef3822 - source: openapi3 tags repository: https://github.com/opensourcewebsite-org/apidocs-opensourcewebsite-org v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPk9wZW5Tb3VyY2VXZWJzaXRlIEFQSSBEb2N1bWVudGF0aW9uPC9oMT4KCkxpdmU6IFthcGlkb2NzLm9wZW5zb3VyY2V3ZWJzaXRlLm9yZ10oaHR0cHM6Ly9hcGlkb2NzLm9wZW5zb3VyY2V3ZWJzaXRlLm9yZykKClNwZWNpZmljYXRpb246IFtvcGVuYXBpLnlhbWxdKGh0dHBzOi8vYXBpZG9jcy5vcGVuc291cmNld2Vic2l0ZS5vcmcvb3BlbmFwaS55YW1sKQoKVGhlIHJlcG9zaXRvcnkgaXMgYSBwYXJ0IG9mIHRoZSBbT3BlblNvdXJjZVdlYnNpdGUgT3JnYW5pemF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vb3BlbnNvdXJjZXdlYnNpdGUtb3JnKS4gVGhpcyBwcm9qZWN0IGFuZCBldmVyeW9uZSBwYXJ0aWNpcGF0aW5nIGluIGl0IGlzIGdvdmVybmVkIGJ5IHRoZSBbQ29kZSBvZiBDb25kdWN0XShDT0RFX09GX0NPTkRVQ1QubWQpLgoKIyMgQ29udHJpYnV0aW5nCgpQbGVhc2UgcmVhZCB0aHJvdWdoIG91ciBbQ29udHJpYnV0aW5nIEd1aWRlbGluZXNdKENPTlRSSUJVVElORy5tZCkuCgojIyBPcGVuQVBJIFNwZWNpZmljYXRpb24KCmh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZwoKaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24KCmh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vYWJvdXQvCgojIyMgUmVzb2x2aW5nIGxvY2FsIGVycm9ycwoKaHR0cHM6Ly9naXRodWIuY29tL21yZG9vYi90aHJlZS5qcy93aWtpL0hvdy10by1ydW4tdGhpbmdzLWxvY2FsbHkKCiMjIEZlZWRiYWNrCgpUbyByZXF1ZXN0IGEgbmV3IGZlYXR1cmUsIHN1Ym1pdCBhIGJ1ZyByZXBvcnQsIGdpdmUgdXMgZmVlZGJhY2ssIHN0YXJ0IGEgZGVzaWduIGRpc2N1c3Npb24gb3IgaGF2ZSBhbiBpZGVhIHRvIG1ha2UgdGhpcyBjb2RlIGJldHRlciBmZWVsIGZyZWUgdG8gW29wZW4gYW4gaXNzdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuc291cmNld2Vic2l0ZS1vcmcvYXBpZG9jcy1vcGVuc291cmNld2Vic2l0ZS1vcmcvaXNzdWVzKSwgb3IgW2NyZWF0ZSBhIHB1bGwgcmVxdWVzdF0oaHR0cHM6Ly9naXRodWIuY29tL29wZW5zb3VyY2V3ZWJzaXRlLW9yZy9hcGlkb2NzLW9wZW5zb3VyY2V3ZWJzaXRlLW9yZy9wdWxscykuCgpQbGVhc2Ugc2VuZCBhbGwgc2VjdXJpdHkgaXNzdWVzIHRvIFtzZWN1cml0eUBvcGVuc291cmNld2Vic2l0ZS5vcmddKG1haWx0bzpzZWN1cml0eUBvcGVuc291cmNld2Vic2l0ZS5vcmcpLgoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIG9wZW4gc291cmNlIGFuZCBhdmFpbGFibGUgZnJlZWx5IHVuZGVyIHRoZSBbTUlUIGxpY2Vuc2VdKExJQ0VOU0UubWQpLgo= readmeEtag: '"ad7d1612469f9e76006bb9e187fa5b1ec68675ea"' readmeLastModified: Tue, 18 Jun 2024 05:46:49 GMT repositoryId: 210095533 description: OpenSourceWebsite API Documentation created: '2019-09-22T05:29:15Z' updated: '2025-12-18T10:49:50Z' language: HTML archived: false stars: 5 watchers: 1 forks: 1 owner: opensourcewebsite-org logo: https://avatars.githubusercontent.com/u/42433667?v=4 license: MIT repoEtag: '"9abbe514a1799e814044fc8d7b9c080635ac8320335059ed560e36474cb8ebf1"' repoLastModified: Thu, 18 Dec 2025 10:49:50 GMT foundInMaster: true category: Server id: 724ad1bbafd532e3bc9d7f8b208672cd - source: openapi3 tags repository: https://github.com/forter/oktabeat v3: true repositoryMetadata: base64Readme: >- IyBPa3RhYmVhdAoKV2VsY29tZSB0byBPa3RhYmVhdC4KCkVuc3VyZSB0aGF0IHRoaXMgZm9sZGVyIGlzIGF0IHRoZSBmb2xsb3dpbmcgbG9jYXRpb246CmAke0dPUEFUSH0vc3JjL2dpdGh1Yi5jb20vZm9ydGVyL29rdGFiZWF0YAoKIyMgR2V0dGluZyBTdGFydGVkIHdpdGggT2t0YWJlYXQKCiMjIyBSZXF1aXJlbWVudHMKCiogW0dvbGFuZ10oaHR0cHM6Ly9nb2xhbmcub3JnL2RsLykgMS43CgojIyMgSW5pdCBQcm9qZWN0ClRvIGdldCBydW5uaW5nIHdpdGggT2t0YWJlYXQgYW5kIGFsc28gaW5zdGFsbCB0aGUKZGVwZW5kZW5jaWVzLCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgCm1ha2Ugc2V0dXAKYGBgCgpJdCB3aWxsIGNyZWF0ZSBhIGNsZWFuIGdpdCBoaXN0b3J5IGZvciBlYWNoIG1ham9yIHN0ZXAuIE5vdGUgdGhhdCB5b3UgY2FuIGFsd2F5cyByZXdyaXRlIHRoZSBoaXN0b3J5IGlmIHlvdSB3aXNoIGJlZm9yZSBwdXNoaW5nIHlvdXIgY2hhbmdlcy4KClRvIHB1c2ggT2t0YWJlYXQgaW4gdGhlIGdpdCByZXBvc2l0b3J5LCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kczoKCmBgYApnaXQgcmVtb3RlIHNldC11cmwgb3JpZ2luIGh0dHBzOi8vZ2l0aHViLmNvbS9mb3J0ZXIvb2t0YWJlYXQKZ2l0IHB1c2ggb3JpZ2luIG1hc3RlcgpgYGAKCkZvciBmdXJ0aGVyIGRldmVsb3BtZW50LCBjaGVjayBvdXQgdGhlIFtiZWF0IGRldmVsb3BlciBndWlkZV0oaHR0cHM6Ly93d3cuZWxhc3RpYy5jby9ndWlkZS9lbi9iZWF0cy9saWJiZWF0L2N1cnJlbnQvbmV3LWJlYXQuaHRtbCkuCgojIyMgQnVpbGQKClRvIGJ1aWxkIHRoZSBiaW5hcnkgZm9yIE9rdGFiZWF0IHJ1biB0aGUgY29tbWFuZCBiZWxvdy4gVGhpcyB3aWxsIGdlbmVyYXRlIGEgYmluYXJ5CmluIHRoZSBzYW1lIGRpcmVjdG9yeSB3aXRoIHRoZSBuYW1lIG9rdGFiZWF0LgoKYGBgCm1ha2UKYGBgCgoKIyMjIFJ1bgoKVG8gcnVuIE9rdGFiZWF0IHdpdGggZGVidWdnaW5nIG91dHB1dCBlbmFibGVkLCBydW46CgpgYGAKLi9va3RhYmVhdCAtYyBva3RhYmVhdC55bWwgLWUgLWQgIioiCmBgYAoKCiMjIyBUZXN0CgpUbyB0ZXN0IE9rdGFiZWF0LCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgCm1ha2UgdGVzdHN1aXRlCmBgYAoKYWx0ZXJuYXRpdmVseToKYGBgCm1ha2UgdW5pdC10ZXN0cwptYWtlIHN5c3RlbS10ZXN0cwptYWtlIGludGVncmF0aW9uLXRlc3RzCm1ha2UgY292ZXJhZ2UtcmVwb3J0CmBgYAoKVGhlIHRlc3QgY292ZXJhZ2UgaXMgcmVwb3J0ZWQgaW4gdGhlIGZvbGRlciBgLi9idWlsZC9jb3ZlcmFnZS9gCgojIyMgVXBkYXRlCgpFYWNoIGJlYXQgaGFzIGEgdGVtcGxhdGUgZm9yIHRoZSBtYXBwaW5nIGluIGVsYXN0aWNzZWFyY2ggYW5kIGEgZG9jdW1lbnRhdGlvbiBmb3IgdGhlIGZpZWxkcwp3aGljaCBpcyBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlZCBiYXNlZCBvbiBgZmllbGRzLnltbGAgYnkgcnVubmluZyB0aGUgZm9sbG93aW5nIGNvbW1hbmQuCgpgYGAKbWFrZSB1cGRhdGUKYGBgCgoKIyMjIENsZWFudXAKClRvIGNsZWFuICBPa3RhYmVhdCBzb3VyY2UgY29kZSwgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYAptYWtlIGZtdApgYGAKClRvIGNsZWFuIHVwIHRoZSBidWlsZCBkaXJlY3RvcnkgYW5kIGdlbmVyYXRlZCBhcnRpZmFjdHMsIHJ1bjoKCmBgYAptYWtlIGNsZWFuCmBgYAoKCiMjIyBDbG9uZQoKVG8gY2xvbmUgT2t0YWJlYXQgZnJvbSB0aGUgZ2l0IHJlcG9zaXRvcnksIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmRzOgoKYGBgCm1rZGlyIC1wICR7R09QQVRIfS9zcmMvZ2l0aHViLmNvbS9mb3J0ZXIvb2t0YWJlYXQKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9mb3J0ZXIvb2t0YWJlYXQgJHtHT1BBVEh9L3NyYy9naXRodWIuY29tL2ZvcnRlci9va3RhYmVhdApgYGAKCgpGb3IgZnVydGhlciBkZXZlbG9wbWVudCwgY2hlY2sgb3V0IHRoZSBbYmVhdCBkZXZlbG9wZXIgZ3VpZGVdKGh0dHBzOi8vd3d3LmVsYXN0aWMuY28vZ3VpZGUvZW4vYmVhdHMvbGliYmVhdC9jdXJyZW50L25ldy1iZWF0Lmh0bWwpLgoKCiMjIFBhY2thZ2luZwoKVGhlIGJlYXQgZnJhbWV3b3JrcyBwcm92aWRlcyB0b29scyB0byBjcm9zc2NvbXBpbGUgYW5kIHBhY2thZ2UgeW91ciBiZWF0IGZvciBkaWZmZXJlbnQgcGxhdGZvcm1zLiBUaGlzIHJlcXVpcmVzIFtkb2NrZXJdKGh0dHBzOi8vd3d3LmRvY2tlci5jb20vKSBhbmQgdmVuZG9yaW5nIGFzIGRlc2NyaWJlZCBhYm92ZS4gVG8gYnVpbGQgcGFja2FnZXMgb2YgeW91ciBiZWF0LCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgCm1ha2UgcmVsZWFzZQpgYGAKClRoaXMgd2lsbCBmZXRjaCBhbmQgY3JlYXRlIGFsbCBpbWFnZXMgcmVxdWlyZWQgZm9yIHRoZSBidWlsZCBwcm9jZXNzLiBUaGUgd2hvbGUgcHJvY2VzcyB0byBmaW5pc2ggY2FuIHRha2Ugc2V2ZXJhbCBtaW51dGVzLgo= readmeEtag: '"5b82c112e8ab01564167814309b7dda4fd8994f3"' readmeLastModified: Mon, 13 May 2019 16:32:11 GMT repositoryId: 186449834 description: >- An elasticbeat that polls the Okta System Log https://developer.okta.com/docs/api/resources/system_log/#getting-started created: '2019-05-13T15:43:27Z' updated: '2024-11-18T16:24:41Z' language: Go archived: true stars: 4 watchers: 3 forks: 3 owner: forter logo: https://avatars.githubusercontent.com/u/6420463?v=4 license: NOASSERTION repoEtag: '"fc16c50176ee5bacd4eabc6ad1cdace77d43a80964c3ca4641f96d1872816e9a"' repoLastModified: Mon, 18 Nov 2024 16:24:41 GMT foundInMaster: true category: - Testing - Server Implementations id: 3e376160f047267d96a9ead763511c1b - source: openapi3 tags repository: https://github.com/zhangmazi/json2swagger3-php v3: true repositoryMetadata: base64Readme: >- IyBKc29uMnN3YWdnZXIzLXBocAojIyBKc29uMnN3YWdnZXIzLXBocOaYr+S4gOS4quWFjei0ueW8gOa6kOW3peWFt++8jOS4u+imgeaKilBvc3RtYW7nmoRSZXF1ZXN05ZKMUmVzcG9uc2Xnu5PmnoTovaxTd2FnZ2VyMy1QSFAoT3BlbkFQSTMpCgojIyDlnKjnur/kvb/nlKgKW+eCueaIkeWcqOe6v+ebtOaOpeeUqF0oaHR0cHM6Ly96aGFuZ21hemkuZ2l0aHViLmlvL2pzb24yc3dhZ2dlcjMtcGhwL2luZGV4Lmh0bWwpCgojIyDliJ3oobcK5pa55L6/5YWs5Y+45ZCM5LqL5oqKUG9zdG1hbumHjOeahOivt+axguWSjOi/lOWbnkpzb27ku6PnoIHlv6vmjbfnmoTlhpnlhaVQSFDku6PnoIHph4zvvIzpgJrov4dzd2FnZ2VyLXVp5bGV546w5omA5pyJ5o6l5Y+j77yM5omA5Lul5q2k5bel5YW36K+e55Sf5LqGLgoKIyMjIOacrOW3peWFt+WfuuS6jltTd2FnRGVmR2VuXShodHRwczovL2dpdGh1Yi5jb20vUm9nZXIxMy9Td2FnRGVmR2VuKSwg5bqU6K+l5pWw5Lit5Zu95Y2H57qn54mI5ZCn77yM5pSv5oyBU3dhZ2dlclBIUC1VSSxPcGVuQXBpMy54CgojIyMg5aaC5p6c5Zac5qyi77yM5Y+v5Lul5pSv5LuY5a6d5omT6LWP5LiA5LiL77yM6LCi6LCi77yBCiFb5oSf6LCi5aSn5L2sXShodHRwOi8vd3d3Lm5pbmphOTExLmNvbS9pbWFnZXMvcGF5Mm1lL2FsaXBheS5wbmcpCgo= readmeEtag: '"4c43fabacef97a6b8ed4de08f3528ed5897035a6"' readmeLastModified: Sat, 30 Mar 2019 05:06:02 GMT repositoryId: 177393777 description: Postman的Request和Response结构转Swagger3-PHP(OPENAPI3)工具 created: '2019-03-24T09:22:37Z' updated: '2021-06-28T10:08:26Z' language: JavaScript archived: false stars: 4 watchers: 1 forks: 0 owner: zhangmazi logo: https://avatars.githubusercontent.com/u/1011088?v=4 license: MIT repoEtag: '"8934afb7a79138f297d9d84522a32eec4f3ee6305e2184f68fc45813913ca6cf"' repoLastModified: Mon, 28 Jun 2021 10:08:26 GMT foundInMaster: true category: SDK id: c5fdb5a7e261ee9e90b0ab53e2537c0f - source: openapi3 tags repository: https://github.com/noahehall/nodeproto v3: true repositoryMetadata: base64Readme: >- IyBAbm9kZXByb3RvCgotIHByb2R1Y3QgZGV2ZWxvcG1lbnQgc3RhcnRlcmtpdAogIC0gZm9jdXNlcyBvbiBmdWxsIHN0YWNrIG5vZGUrcmVhY3QgYXBwcyBydW5uaW5nIG9uIEFXUwogIC0gdmFncmFudCArIGRvY2tlciBpbiBkZXYgKHRvZG8pCiAgLSBkb2NrZXIgaW4gcHJvZCAodG9kbykKICAtIHRoaXMgcHJvamVjdCB3aWxsIGJlY29tZSB0aGUgYmFzZSBmb3IgdGhlIGNvcmUgbmlydmFpIHdlYiBhcHAKCi0gaGFyZCByZXF1aXJlbWVudHMgZm9yIGRldmVsb3AgYnJhbmNoCiAgLSBkZXBlbmRlbmN5IHJlcXVpcmVtZW50cwogICAgLSBhbGwgaW1hZ2VzIHB1c2hlZCB0byBbQG5vZGVwcm90byByZWdpc3RyeSBvbiBhd3NdKGh0dHBzOi8vZ2FsbGVyeS5lY3IuYXdzL3owYzNuNGg1L25vZGVwcm90by9hcHBzKQogICAgLSBBcmFuZ29kYiAodG9kbykKICAgIC0gQ2xpY2tIb3VzZSAodG9kbykKICAgIC0gQ29uc3VsICh0b2RvKQogICAgLSBmbG93dHlwZSA+PSAwLjE3MS4wCiAgICAtIGhhcHJveHkgKG5lZWRzIHZpcnR1YWxpemF0aW9uKQogICAgLSBrb2EgPT4gMgogICAgLSBub2RlICsgY29yZXBhY2sgPj0gMTcuNSAoMTcuNiBpcyBvdXQhKQogICAgLSBPcGVuU1NMID49IDEuMS4xIGZvciBUTFN2MS4zIHN1cHBvcnQKICAgIC0gcG5wbSA+PSA2LjMwLjEKICAgIC0gcmVhY3QgKyByZWFjdC1kb20gQHJjIChvciBAbmV4dCkKICAgIC0gU2lnbm96IEFQTSAodG9kbykKICAtIHByb2Nlc3MgcmVxdWlyZW1lbnRzCiAgICAtIHBhY2thZ2UuanNvbi5jb25maWcgaXMgY29udHJvbGxlZCAgYnkgcm9vdC9wYWNrYWdlLmpzb24gYW5kIHN5bmNlZCB0byBjaGlsZCBwYWNrYWdlcyB2aWEgdG9vbHMvanN5bmMKICAgIC0gYWxsIHBhY2thZ2VzIGV4cG9ydCByZXVzYWJsZSB0eXBlIGRlZmluaXRpb25zIGFzIGBAbm9kZXByb3RvL3BhY2thZ2VuYW1lL3NyYy9saWJkZWZzYAogIC0gY3VsdHVyYWwgYm91bmRhcmllcwogICAgLSBiZXN0IGluIGNsYXNzID4gbW9zdCBwb3B1bGFyCiAgICAtIGJsZWVkaW5nIGVkZ2UgYWx3YXlzCiAgICAtIG1vbm9yZXBvIG1pY3Jvc2VydmljZXMgPiBtdWx0aXJlcG8gbWljcm9zZXJ2aWNlcwogICAgLSByZWFkIHRoZSBjb2RlID4gcmVhZCB0aGUgY29tbWVudHMKICAgIC0gc3BlY2lmaWNhdGlvbiA+IGtpdGNoZW4gc2lua3MKICAgIC0gdGVyc2UgY2xhcml0eSA+IHZlcmJvc2UgZXhwcmVzc2l2ZW5lc3MKICAgIC0gdGVzdHMgYXJlIGZpcnN0IGNsYXNzIGNpdGl6ZW5zCiAgICAtIHRydW5rID4gYnJhbmNoCgojIyBUTERSCgoKLSBzZXJ2aWNlcwogIC0gW0Bub2RlcHJvdG8vY2xpZW50IC0gUmVhY3RAbmV4dCBmcm9udGVuZF0ocGFja2FnZXMvYXBwcy9jbGllbnQvUkVBRE1FLm1kKQogIC0gW0Bub2RlcHJvdG8vYXV0aG56IC0ga29hL29wZW5hcGkgYmFja2VuZF0ocGFja2FnZXMvYXBwcy9hdXRobnovUkVBRE1FLm1kKQotIGxpYnJhcmllcwogIC0gW0Bub2RlcHJvdG8vY29uZmlncHJvdG8gLSBTdGF0aWMgY29uZmlndXJhdGlvbnNdKHBhY2thZ2VzL2xpYnJhcmllcy9jb25maWdwcm90by9SRUFETUUubWQpCiAgLSBbQG5vZGVwcm90by9zaGFyZWQgLSB1dGlsaXR5IGZuc10ocGFja2FnZXMvbGlicmFyaWVzL3NoYXJlZC9SRUFETUUubWQpCiAgLSBbQG5vZGVwcm90by90ZXN0cHJvdG8gLSBIaWdoIFZlbG9jaXR5IHRlc3Qgc3VpdGVdKHBhY2thZ2VzL2xpYnJhcmllcy90ZXN0cHJvdG8vUkVBRE1FLm1kKQogIC0gW0Bub2RlcHJvdG8vZW52cHJvdG8gLSBlbnYvU1NMIG1hbmFnZW1lbnRdKHBhY2thZ2VzL2xpYnJhcmllcy9lbnZwcm90by9SRUFETUUubWQpCiAgLSBbQG5vZGVwcm90by9idWlsZHByb3RvIC0gZXNidWlsZC93ZWJwYWNrOiBidWlsZCBhbnl0aGluZyBydW4gZXZlcnl3aGVyZV0ocGFja2FnZXMvbGlicmFyaWVzL2J1aWxkcHJvdG8vUkVBRE1FLm1kKQogIC0gW0Bub2RlcHJvdG8vd3RmIC0gV2hlcmUgdGhlIGZpbGU/IHN5c3RlbSBsb2NhdGlvbnMgJiBmaWxlIG1hbmFnZW1lbnRdKHBhY2thZ2VzL2xpYnJhcmllcy93dGYvUkVBRE1FLm1kKQotIHRvb2xzCiAgLSBbQG5vZGVwcm90by9qc3luYyAtIHN5bmNocm9uaXplIHBhY2thZ2UuanNvbltjXSBmaWxlc10ocGFja2FnZXMvdG9vbHMvanN5bmMvUkVBRE1FLm1kKQoKIyMjIG5wbSBzY3JpcHRzCgotIHlvdSBjYW4gcHJlcGVuZCBiZWZvcmUgYW55IG5wbSBzY3JpcHQKICAtIGBwcm90b2Agb3IgYHByb3RvOnNjcmlwdGAgY21kIHRvIHJ1biBpbiBhbGwgcGFja2FnZXMKICAgIC0gcHJvdG8gcnVucyBjbWRzIGluIGVhY2ggcGtnIHZpYSB1bHRyYQogICAgLSBwcm90bzpzY3JpcHQgcnVucyBjbWRzIGluIGVhY2ggcGtnIHZpYSBwbnBtCiAgLSBgdWx0cmFgIHRvIHJ1biB2aWEgdWx0cmEtcnVubmVyCgojIyMgdW5pbnN0YWxsCgotIHRvIHJlbW92ZSB0aGlzIHJlcG9zaXRvcnkgY29tcGxldGVseSBmcm9tIHlvdXIgc3lzdGVtCi0gZGVsZXRlIHRoZSBgL3Zhci8ubm9kZXByb3RvYCBkaXJlY3RvcnksIGFuZCB0aGlzIHJlcG9zaXRvcnkKCiMjIyBpbnN0YWxsYXRpb24gKGNvbnRyaWJ1dGluZykKCi0gdmFncmFudCAocmVjb21tZW5kZWQpLCByZXF1aXJlcyA+PSB2Mi4yCgpgYGBzaAogIHZhZ3JhbnQgdXAKICB2YWdyYW50IHNzaAogIGNkIC9vcHQvbm9kZXByb3RvCmBgYAoKLSBiYXJlIG1ldGFsCmBgYHNoCiAgIyBkZXBlbmRlbmN5ICsgcG5wbSBtZXRhZGF0YQogIHN1ZG8gbWtkaXIgL3Zhci8ubm9kZXByb3RvCiAgc3VkbyBjaG93biAtUiAkKHdob2FtaSk6JCh3aG9hbWkpIC92YXIvLm5vZGVwcm90bwogIHBucG0gaW5zdGFsbC1jb21wbGV0aW9uCiAgcG5wbSBpbnN0YWxsCiAgcG5wbSBwcm90bzpzY3JpcHQgYnVpbGQKICBwbnBtIHByb3RvOnNjcmlwdCByZXBvOnRlc3QKYGBgCgotIG90aGVyIGhlbGxwZnVsIGNtZHMKCmBgYHNoCiAgIyBkZXBlbmRlbmNpZXMKICBwbnBtIHByb3RvIHJlcG86Zmxvd3R5cGVkOmluc3RhbGwgIyBpbnN0YWxsIGZsb3d0eXBlIGRlZnMKICBwbnBtIHByb3RvIHJlcG86anN5bmMgIyBzeW5jaHJvbml6ZSByb290L3BhY2thZ2UuanNvbiBpbnRvIGVhY2ggcGFja2FnZS9wYWNrYWdlLmpzb24KCiAgIyB2YWxpZGF0aW9uCiAgcG5wbSBwcm90byByZXBvOmxpbnQKCiAgIyBpbnRyb3NwZWN0aW9uCiAgcG5wbSByZXBvOmRlcHM6Z3JhcGgKICBwbnBtIHJlcG86c2NyaXB0czp2CmBgYAoKCiMjIyB1cGRhdGluZwoKYGBgc2gKICBwbnBtIHByb3RvIHJlcG86dXBkYXRlICMgdXBkYXRlcyBhbGwgZGVwcyB0byBsYXRlc3QKYGBgCgojIyMgcnVubmluZyB0ZXN0cyAmIGxpbnRzCgpgYGBzaAogIHBucG0gcmVwbzpsaW50ICMgcnVucyBlc2xpbnQgKGZpeCBtb2RlKSwgcHJldHRpZXIsIGZsb3cgKyB0eXBlIGNvdmVyYWdlCiAgcG5wbSByZXBvOmVzbGludCAjIHJ1bnMgZXNsaW50IGZsb3cgJiBwcmV0dGllciwgZmFpbHMgb24gZmlyc3QgcGFja2FnZSB3aXRoIGxpbnQgZXJycwogIHBucG0gcmVwbzplc2xpbnQ6Zml4ICMgc2FtZSBhcyBhYm92ZSwgYnV0IGRvZXNudCBmYWlsCiAgcG5wbSByZXBvOmZsb3c6Y292ZXJhZ2UgIyBydW5zIGZsb3cgd2l0aCB0eXBlIGRlZiBjb3ZlcmFnZQogIHBucG0gZmxvdyAjIGRldGFpbGVkIGZsb3d0eXBlIGFuYWx5c2lzCiAgcG5wbSBmbG93IHN0b3B8c3RhcnQgIyByZXN0YXJ0IGZsb3cgc2VydmVyLCBzb21ldGltZXMgaXQgZmFpbHMgdG8gcGljayB1cCBjaGFuZ2VzIHRvIGRlZXAgZXh0ZXJuYWwgbGliZGVmcwogIHBucG0gcmVwbzp0ZXN0ICMgcnVuIHRlc3RzCiAgcG5wbSByZXBvOnRlc3RpbmcgIyB3YXRjaCAmIHJlcnVuIHRlc3RzIGluIGEgc2luZ2xlIHBhY2thZ2UKICBwbnBtIHJlcG86dGVzdDpmaWxlIHNvbWVmaWxlICMgcnVuIGEgc3BlY2lmaWMgdGVzdCBmaWxlCgpgYGAK readmeEtag: '"442171f34684c5a02b29465f4b5dd66ec5748f54"' readmeLastModified: Fri, 15 Apr 2022 00:02:03 GMT repositoryId: 369644109 description: build prototypes, fast! created: '2021-05-21T20:12:07Z' updated: '2023-10-17T15:24:24Z' language: HTML archived: true stars: 4 watchers: 1 forks: 0 owner: noahehall logo: https://avatars.githubusercontent.com/u/10324554?v=4 license: NOASSERTION repoEtag: '"309a96239f4688c0ad569d6d9e4b05a89ec4ea0623bdaf941dc499a40757f780"' repoLastModified: Tue, 17 Oct 2023 15:24:24 GMT foundInMaster: true category: Testing id: 37b213c383b786f80ef2ad7961c30558 - source: openapi3 tags repository: https://github.com/apiportal/abyss-spec-transformer v3: true repositoryMetadata: base64Readme: >- [![Build Status](https://travis-ci.org/apiportal/abyss-spec-transformer.svg?branch=master)](https://travis-ci.org/apiportal/abyss-spec-transformer)

# ABYSS SPEC TRANSFORMER

This application transforms provided `WSDL` into an `OpenAPI v3` spec yaml file. WSDL should be provided via path, URI or String  

An example is below

**Input**

```xml
<?xml version="1.0" standalone="no"?>
<wsdl:definitions xmlns:wsdl="http://schemas.xmlsoap.org/wsdl/"
                  xmlns="http://schemas.xmlsoap.org/wsdl/"
                  xmlns:wsaw="http://www.w3.org/2006/05/addressing/wsdl"
                  xmlns:http="http://schemas.xmlsoap.org/wsdl/http/"
                  xmlns:tns="http://thomas-bayer.com/blz/"
                  xmlns:xsd="http://www.w3.org/2001/XMLSchema"
                  xmlns:mime="http://schemas.xmlsoap.org/wsdl/mime/"
                  xmlns:soap="http://schemas.xmlsoap.org/wsdl/soap/"
                  xmlns:soap12="http://schemas.xmlsoap.org/wsdl/soap12/"
                  targetNamespace="http://thomas-bayer.com/blz/">
    <wsdl:documentation>BLZService</wsdl:documentation>
    <wsdl:types>
        <xsd:schema attributeFormDefault="unqualified" elementFormDefault="qualified" targetNamespace="http://thomas-bayer.com/blz/">
            <xsd:element name="getBank" type="tns:getBankType"></xsd:element>
            <xsd:element name="getBankResponse" type="tns:getBankResponseType"></xsd:element>
            <xsd:complexType name="getBankType">
                <xsd:sequence>
                    <xsd:element name="blz" type="xsd:string"></xsd:element>
                </xsd:sequence>
            </xsd:complexType>
            <xsd:complexType name="getBankResponseType">
                <xsd:sequence>
                    <xsd:element name="details" type="tns:detailsType"></xsd:element>
                </xsd:sequence>
            </xsd:complexType>
            <xsd:complexType name="detailsType">
                <xsd:sequence>
                    <xsd:element minOccurs="0" name="bezeichnung" type="xsd:string"></xsd:element>
                    <xsd:element minOccurs="0" name="bic" type="xsd:string"></xsd:element>
                    <xsd:element minOccurs="0" name="ort" type="xsd:string"></xsd:element>
                    <xsd:element minOccurs="0" name="plz" type="xsd:string"></xsd:element>
                </xsd:sequence>
            </xsd:complexType>
        </xsd:schema>
    </wsdl:types>
    <wsdl:message name="getBank">
        <wsdl:part name="parameters" element="tns:getBank"></wsdl:part>
    </wsdl:message>
    <wsdl:message name="getBankResponse">
        <wsdl:part name="parameters" element="tns:getBankResponse"></wsdl:part>
    </wsdl:message>
    <wsdl:portType name="BLZServicePortType">
        <wsdl:operation name="getBank">
            <wsdl:input message="tns:getBank"></wsdl:input>
            <wsdl:output message="tns:getBankResponse" wsaw:Action="http://thomas-bayer.com/blz/BLZService/getBankResponse"></wsdl:output>
        </wsdl:operation>
    </wsdl:portType>
    <wsdl:binding name="BLZServiceSOAP11Binding" type="tns:BLZServicePortType">
        <soap:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"></soap:binding>
        <wsdl:operation name="getBank">
            <soap:operation style="document" soapAction=""></soap:operation>
            <wsdl:input>
                <soap:body use="literal"></soap:body>
            </wsdl:input>
            <wsdl:output>
                <soap:body use="literal"></soap:body>
            </wsdl:output>
        </wsdl:operation>
    </wsdl:binding>
    <wsdl:binding name="BLZServiceSOAP12Binding" type="tns:BLZServicePortType">
        <soap12:binding style="document" transport="http://schemas.xmlsoap.org/soap/http"></soap12:binding>
        <wsdl:operation name="getBank">
            <soap12:operation style="document" soapAction=""></soap12:operation>
            <wsdl:input>
                <soap12:body use="literal"></soap12:body>
            </wsdl:input>
            <wsdl:output>
                <soap12:body use="literal"></soap12:body>
            </wsdl:output>
        </wsdl:operation>
    </wsdl:binding>
    <wsdl:binding name="BLZServiceHttpBinding" type="tns:BLZServicePortType">
        <http:binding verb="POST"></http:binding>
        <wsdl:operation name="getBank">
            <http:operation location="BLZService/getBank"></http:operation>
            <wsdl:input>
                <mime:content part="getBank" type="text/xml"></mime:content>
            </wsdl:input>
            <wsdl:output>
                <mime:content part="getBank" type="text/xml"></mime:content>
            </wsdl:output>
        </wsdl:operation>
    </wsdl:binding>
    <wsdl:service name="BLZService">
        <wsdl:port name="BLZServiceSOAP11port_http" binding="tns:BLZServiceSOAP11Binding">
            <soap:address location="http://www.thomas-bayer.com/axis2/services/BLZService"></soap:address>
        </wsdl:port>
        <wsdl:port name="BLZServiceSOAP12port_http" binding="tns:BLZServiceSOAP12Binding">
            <soap12:address location="http://www.thomas-bayer.com/axis2/services/BLZService"></soap12:address>
        </wsdl:port>
        <wsdl:port name="BLZServiceHttpport" binding="tns:BLZServiceHttpBinding">
            <http:address location="http://www.thomas-bayer.com/axis2/services/BLZService"></http:address>
        </wsdl:port>
    </wsdl:service>
</wsdl:definitions>
```

>credit to http://www.thomas-bayer.com/


**Usage**

```java
String sourcePath = "http://thomas-bayer.com/axis2/services/BLZService?wsdl";
return new OpenAPITransformer().transform(sourcePath);        
```

**Output**

```yaml
openapi: 3.0.1
info:
  title: BLZService
  version: "1.0"
servers:
- url: http://www.thomas-bayer.com/axis2/services/BLZService
tags:
- name: BLZServiceSOAP11Binding
  externalDocs:
    description: Soap 1.1 Service
    url: http://www.thomas-bayer.com/axis2/services/BLZService
- name: BLZServiceHttpBinding
  externalDocs:
    description: HTTP Service
    url: http://www.thomas-bayer.com/axis2/services/BLZService
- name: BLZServiceSOAP12Binding
  externalDocs:
    description: Soap 1.2 Service
    url: http://www.thomas-bayer.com/axis2/services/BLZService
paths:
  /BLZService/getBank:
    post:
      tags:
      - BLZServiceHttpBinding
      operationId: getBank_HttpService
      requestBody:
        $ref: '#/components/requestBodies/getBank'
      responses:
        200:
          $ref: '#/components/responses/200_getBankResponse'
  /getBank:
    post:
      tags:
      - BLZServiceSOAP12Binding
      - BLZServiceSOAP11Binding
      operationId: getBank_SoapService
      requestBody:
        $ref: '#/components/requestBodies/getBank'
      responses:
        200:
          $ref: '#/components/responses/200_getBankResponse'
components:
  schemas:
    getBankResponseType:
      title: getBankResponseType
      required:
      - details
      type: object
      properties:
        details:
          $ref: '#/components/schemas/detailsType'
    getBankResponse:
      title: getBankResponse
      required:
      - getBankResponseType
      type: object
      properties:
        getBankResponseType:
          $ref: '#/components/schemas/getBankResponseType'
    getBankType:
      title: getBankType
      required:
      - blz
      type: object
      properties:
        blz:
          type: string
    detailsType:
      title: detailsType
      required:
      - bezeichnung
      - bic
      - ort
      - plz
      type: object
      properties:
        bezeichnung:
          type: string
        bic:
          type: string
        ort:
          type: string
        plz:
          type: string
    getBank:
      title: getBank
      required:
      - getBankType
      type: object
      properties:
        getBankType:
          $ref: '#/components/schemas/getBankType'
  responses:
    200_getBankResponse:
      description: Successful operation
      content:
        application/xml:
          schema:
            $ref: '#/components/x-messages/getBankResponse'
  requestBodies:
    getBank:
      description: getBank message object as request body
      content:
        application/xml:
          schema:
            $ref: '#/components/x-messages/getBank'
      required: true
  x-messages:
    getBankResponse:
      $ref: '#/components/schemas/getBankResponse'
    getBank:
      $ref: '#/components/schemas/getBank'
``` readmeEtag: '"55c05ba0aa538774c6ccee25f0a5436d9cecd18e"' readmeLastModified: Wed, 08 May 2019 16:25:42 GMT repositoryId: 182996886 description: Abyss Spec Transformer Project https://apiportal.com created: '2019-04-23T11:04:16Z' updated: '2019-10-08T20:22:59Z' language: Java archived: false stars: 4 watchers: 3 forks: 2 owner: apiportal logo: https://avatars.githubusercontent.com/u/49755495?v=4 license: Apache-2.0 repoEtag: '"51adc3ece8f467a3a73ac1af22b1634ad5a45f024f321c53bc61be1b5b31cbe8"' repoLastModified: Tue, 08 Oct 2019 20:22:59 GMT foundInMaster: true category: Parsers id: 8f099ab923615e467822d1db90695ec1 - source: openapi3 tags repository: https://github.com/mohitkumarsahni/todo-api v3: true id: bc0238674e6a94f258aeca89c30ea2af repositoryMetadata: base64Readme: >- IyMgVG9Eby1BUEkKQW4gQVBJIGZvciBjcmVhdGluZyBhbmQgbWFuYWdpbmcgdGFza3MuCgojIyMgVGhlIEFQSSBjYW4gcGVyZm9ybSBmb2xsb3dpbmcgb3BlcmF0aW9uczoKCi0gW3hdIENyZWF0ZSBhIG5ldyB0YXNrLWxpc3QuCi0gW3hdIFVwZGF0ZSBhIHRhc2stbGlzdC4KLSBbeF0gRmV0Y2ggYSB0YXNrLWxpc3QuCi0gW3hdIERlbGV0ZSBhIHRhc2stbGlzdC4KLSBbeF0gQ3JlYXRlIGEgbmV3IHRhc2sgaW4gYSB0YXNrLWxpc3QuCi0gW3hdIFVwZGF0ZSBhIHRhc2sgaW4gYSB0YXNrLWxpc3QuCi0gW3hdIEZldGNoIGEgdGFzay4KLSBbeF0gRmV0Y2ggYWxsIHRhc2tzIGluIGEgdGFzay1saXN0LgotIFt4XSBEZWxldGUgYSB0YXNrIGluIGEgdGFzay1saXN0LgoKUGxlYXNlIHNlZSB0aGUgW29wZW5hcGkgY29udHJhY3RdKC4vZG9jcy9hcGktY29udHJhY3QueWFtbCkgZm9yIEFQSSBlbmRwb2ludHMgYW5kIGRldGFpbHMuCgojIyMgUHJlcmVxdWlzaXRlcwoKIyMjIyBJbnN0YWxsIEphdmEgMTEKLSBUaGlzIGFwcGxpY2F0aW9uIHJlcXVpcmVzIEphdmEgMTEuCi0gUGxlYXNlIHJlZmVyIHRvIHRoZSBmb2xsb3dpbmcgbGluayB0byBkb3dubG9hZCBhbmQgaW5zdGFsbCBKYXZhLTExOiBbSmF2YS0xMV0oaHR0cHM6Ly93d3cub3JhY2xlLmNvbS9pbi9qYXZhL3RlY2hub2xvZ2llcy9qYXZhc2UvamRrMTEtYXJjaGl2ZS1kb3dubG9hZHMuaHRtbCkKCiMjIyMgSW5zdGFsbCAmIFNldHVwIERhdGFiYXNlCi0gVGhpcyBhcHBsaWNhdGlvbiByZXF1aXJlcyBNeVNRTCA4LgotIFBsZWFzZSByZWZlciB0byB0aGUgZm9sbG93aW5nIGxpbmsgdG8gZG93bmxvYWQgYW5kIGluc3RhbGwgTXlTUUwtODogW015U1FMLThdKGh0dHBzOi8vZGV2Lm15c3FsLmNvbS9kb2MvcmVmbWFuLzguMC9lbi9pbnN0YWxsaW5nLmh0bWwpCi0gVG8gY3JlYXRlIGRhdGFiYXNlIGFuZCB1c2VyIHlvdSBjYW4gcnVuIGZvbGxvd2luZyBzY3JpcHQ6IFtkYl9zZXR1cC5zcWxdKC4vZG9jcy9kYXRhYmFzZS9kYl9zZXR1cC5zcWwpCgojIyMgUnVubmluZyB0aGUgYXBwbGljYXRpb24KLSBDbG9uZSB0aGUgcmVwb3NpdG9yeS4KLSBPcGVuIHRoZSBwcm9qZWN0IGluIEludGVsbGlKIElERUEgb3IgYW55IHN1aXRhYmxlIElERSBvZiB5b3VyIGNob2ljZS4KLSBBZGQgZm9sbG93aW5nIGVudmlyb25tZW50IHZhcmlhYmxlcyBpbiBjb25maWd1cmF0aW9uczoKICAtIERCX0hPU1QKICAtIERCX05BTUUKICAtIERCX1BBU1NXT1JECiAgLSBEQl9QT1JUCiAgLSBEQl9VU0VSTkFNRQotIEluIEludGVsbGlKIElERUEsIGVudmlyb25tZW50IHZhcmlhYmxlcyBjYW4gYmUgYWRkZWQgZnJvbSBSdW4gLT4gRWRpdCBDb25maWd1cmF0aW9ucyAtPiBFbnZpcm9ubWVudCBWYXJpYWJsZXMKLSBSdW4gdGhlIHByb2plY3QuCgojIyMgQnVpbGQgJiBydW4gd2l0aCBEb2NrZXIgb24gVWJ1bnR1Ci0gSW5zdGFsbCBEb2NrZXIgZnJvbTogW0RvY2tlcl0oaHR0cHM6Ly9kb2NzLmRvY2tlci5jb20vZW5naW5lL2luc3RhbGwvdWJ1bnR1LykKLSBJbiB0ZXJtaW5hbCBzd2l0Y2ggdG8gcHJvamVjdCBkaXJlY3RvcnkuCi0gVXNlIGZvbGxvd2luZyBjb21tYW5kOiBgZG9ja2VyIGJ1aWxkIC10IHRvZG8tYXBpOjEuMC4wIC5gCi0gQWJvdmUgY29tbWFuZCB3aWxsIGJ1aWxkIHRoZSBpbWFnZS4KLSBOb3cgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCiAgYHN1ZG8gZG9ja2VyIHJ1biB0b2RvLWFwaToxLjAuMCAtZSAiREJfSE9TVD08REJfSE9TVD4iIC1lICJEQl9QT1JUPTxEQl9QT1JUPiIgLWUgIkRCX05BTUU9PERCX05BTUU+IiAtZSAiREJfVVNFUk5BTUU9PERCX1VTRVJOQU1FPiIgLWUgIkRCX1BBU1NXT1JEPTxEQl9QQVNTV09SRD4iIC1wIDgwODA6ODA4MCAtZCBgCi0gQ2hhbmdlIHRoZSBhY3R1YWwgdmFyaWFibGUgdmFsdWUgaW4gYW5nbGUgYnJhY2tldHMuCi0gSXQgd2lsbCBzcGluIHVwIGEgY29udGFpbmVyIHdoaWNoIGNhbiBiZSBhY2Nlc3NlZCBvbiBgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2AKCiMjIyBQZW5kaW5nIFdvcmsKLSBVbml0IFRlc3RpbmcKLSBIb3cgdG8gZGVwbG95IHdpdGggZG9ja2VyIG9uIHVidW50dQotIEhvdyB0byBkZXBsb3kgd2l0aCBBenVyZSBXZWIgQXBwIFNlcnZpY2UKLSBBdXRob3JpemF0aW9uICYgQXV0aGVudGljYXRpb24= readmeEtag: '"69a5e4e83f78d2b01ffab98173faa2b10bfc2845"' readmeLastModified: Sat, 07 Jan 2023 17:43:59 GMT repositoryId: 581242250 description: An API for creating and managing tasks. created: '2022-12-22T16:43:37Z' updated: '2025-11-01T23:42:46Z' language: Java archived: false stars: 5 watchers: 1 forks: 15 owner: mohitkumarsahni logo: https://avatars.githubusercontent.com/u/25563097?v=4 license: MIT repoEtag: '"79df82f98c8d4b2fb8e10b0a52c3e7b0cac7c5955ac423dbddb0e1c5f5490386"' repoLastModified: Sat, 01 Nov 2025 23:42:46 GMT category: - Parsers - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/aereal/go-openapi3-validation-middleware v3: true repositoryMetadata: base64Readme: >- IVtDSV1bY2ktc3RhdHVzXQpbIVtQa2dHb0Rldl1bcGtnLWdvLWRldi1iYWRnZV1dW3BrZy1nby1kZXZdCgojIGdvLW9wZW5hcGkzLXZhbGlkYXRpb24tbWlkZGxld2FyZQoKbmV0L2h0dHAgbWlkZGxld2FyZSB0byB2YWxpZGF0ZSBIVFRQIHJlcXVlc3RzL3Jlc3BvbnNlcyBhZ2FpbnN0IE9wZW5BUEkgMyBzY2hlbWEgdXNpbmcgW2tpbi1vcGVuYXBpXVtdLgoKIyMgSW5zdGFsbGF0aW9uCgpgYGBzaApnbyBnZXQgZ2l0aHViLmNvbS9hZXJlYWwvZ28tb3BlbmFwaTMtdmFsaWRhdGlvbi1taWRkbGV3YXJlCmBgYAoKIyMgU3lub3BzaXMKCmBgYGdvCmltcG9ydCAoCgkibmV0L2h0dHAiCgoJImdpdGh1Yi5jb20vYWVyZWFsL2dvLW9wZW5hcGkzLXZhbGlkYXRpb24tbWlkZGxld2FyZSIKCSJnaXRodWIuY29tL2dldGtpbi9raW4tb3BlbmFwaS9yb3V0ZXJzIgopCgpmdW5jIG1haW4oKSB7Cgl2YXIgcm91dGVyIHJvdXRlcnMuUm91dGVyIC8vIG11c3QgYmUgYnVpbHQgd2l0aCBjZXJ0YWluIHdheQoJbXcgOj0gb3BlbmFwaTNtaWRkbGV3YXJlLldpdGhWYWxpZGF0aW9uKG9wZW5hcGkzbWlkZGxld2FyZS5NaWRkbGV3YXJlT3B0aW9uc3tSb3V0ZXI6IHJvdXRlcn0pCglodHRwLkhhbmRsZSgiLyIsIG13KGh0dHAuSGFuZGxlckZ1bmMoZnVuYyAodyBodHRwLlJlc3BvbnNlV3JpdGVyLCByICpodHRwLlJlcXVlc3QpIHsKCQkvLyB0aGlzIGhhbmRsZXIgaXMgY2FsbGVkIGlmIHZhbGlkYXRpb24gc3VjY2VlZHMKCX0pKSkKfQpgYGAKCiMjIExpY2Vuc2UKClNlZSBMSUNFTlNFIGZpbGUuCgpbcGtnLWdvLWRldl06IGh0dHBzOi8vcGtnLmdvLmRldi9naXRodWIuY29tL2FlcmVhbC9nby1vcGVuYXBpMy12YWxpZGF0aW9uLW1pZGRsZXdhcmUKW3BrZy1nby1kZXYtYmFkZ2VdOiBodHRwczovL3BrZy5nby5kZXYvYmFkZ2UvYWVyZWFsL2dvLW9wZW5hcGkzLXZhbGlkYXRpb24tbWlkZGxld2FyZQpbY2ktc3RhdHVzXTogaHR0cHM6Ly9naXRodWIuY29tL2FlcmVhbC9nby1vcGVuYXBpMy12YWxpZGF0aW9uLW1pZGRsZXdhcmUvd29ya2Zsb3dzL0NJL2JhZGdlLnN2Zz9icmFuY2g9bWFpbgpba2luLW9wZW5hcGldOiBodHRwczovL2dpdGh1Yi5jb20vZ2V0a2luL2tpbi1vcGVuYXBpCg== readmeEtag: '"8579a1a295d819d4ee0ec05621b7e4e6a136729f"' readmeLastModified: Wed, 13 Dec 2023 12:23:59 GMT repositoryId: 375605593 description: >- net/http middleware to validate HTTP requests/responses against OpenAPI 3 schema using kin-openapi. created: '2021-06-10T07:13:33Z' updated: '2025-11-07T00:02:07Z' language: Go archived: false stars: 5 watchers: 1 forks: 1 owner: aereal logo: https://avatars.githubusercontent.com/u/87649?v=4 license: MIT repoEtag: '"d862127b12f8832d693501bc66972c0f2d11ff258099581e145bb896de7317fe"' repoLastModified: Fri, 07 Nov 2025 00:02:07 GMT foundInMaster: true category: Data Validators id: 8f5fa1fe1b8d8d7bed2b2dc8007802e9 - source: openapi3 tags repository: https://github.com/dna-evolutions/angular-demo-application-source v3: true repositoryMetadata: base64Readme: >- # TourOptimizer Angular Demo Application - Source

<a href="https://dna-evolutions.com/" target="_blank"><img src="https://docs.dna-evolutions.com/indexres/dna-temp-logo.png" width="110"
title="DNA-Evolutions" alt="DNA-Evolutions"></a>

To utilize <a href="https://github.com/DNA-Evolutions/Docker-REST-TourOptimizer/blob/main/README.md#how-to-start-jopttouroptimizer-docker" target="_blank">JOptTourOptimizer-Docker</a>, we created an Angular-Demo application. This demo application is hosted on <a href="https://azure.microsoft.com/" target="_blank">Microsoft Azure</a> and is made available via <a href="https://demo.dna-evolutions.com/" target="_blank">https://demo.dna-evolutions.com</a>. Here we would like to share the latest source-code of this project. This is not a classical "Hello World" project, as it is using multiple dependencies and advanced concepts. However, feel free to explore the project and extract whatever you need to get started with JOpt.

**Update January 11th 2025:**

From now on, the Angular Demo uses Angular 19.1. We also updated/upgraded multiple other dependencies. The generated models are using JOpt.TourOptimizer with **version 1.3.0 or higher**.

**Update July 12th 2024:**

From now on, the Angular Demo uses Angular 18. We also updated/upgraded multiple other dependencies. The generated models are using JOpt.TourOptimizer with **version 1.2.7 or higher**.

**Update January 7th 2024:**

From now on, the Angular Demo uses Angular 17. We also updated/upgraded multiple other dependencies. The generated models are using JOpt.TourOptimizer with **version 1.2.6 or higher**. In addition, a "free moving example" was added. Nodes and Resources can be freely positioned via drag and drop.

**Update June 8th 2021:**

From now on, the Angular Demo uses our new swagger definition (see 
<a href="https://swagger.dna-evolutions.com/v3/api-docs/Optimize" target="_blank">swagger.dna-evolutions.com/v3/api-docs/Optimize</a>). This requires using JOpt.TourOptimizer with **version 1.0.3 or higher**. This swagger definition has the advantage that it enables almost all features of JOpt at the REST-endpoint.

**Update 21st September 2021:**
The replacement described in *Update June 2021* is no longer necessary. However, models already created need to be recreated. Further, the legacy endpoint is dropped, and the main-endpoint changed to <a href="https://swagger.dna-evolutions.com/v3/api-docs/Optimize" target="_blank">swagger.dna-evolutions.com/v3/api-docs/Optimize</a>.

---

# Contact

If you need any help, please contact us via our company website <a href="https://www.dna-evolutions.com" target="_blank">www.dna-evolutions.com</a> or write an email to <a href="mailto:info@dna-evolutions.com">info@dna-evolutions.com</a>.

---

## Further Documentation and helpful Links

Our content:
- Further documentation - <a href="https://docs.dna-evolutions.com" target="_blank">docs.dna-evolutions.com</a>
- Special features	- <a href="https://docs.dna-evolutions.com/overview_docs/special_features/Special_Features.html" target="_blank">Overview of special features</a>
- Our company website - <a href="https://www.dna-evolutions.com" target="_blank">www.dna-evolutions.com</a>
- Our official repository - <a href="https://public.repo.dna-evolutions.com" target="_blank">public.repo.dna-evolutions.com</a>
- Our official JavaDocs - <a href="https://public.javadoc.dna-evolutions.com" target="_blank">public.javadoc.dna-evolutions.com</a>
- Our YouTube channel - <a href="https://www.youtube.com/channel/UCzfZjJLp5Rrk7U2UKsOf8Fw" target="_blank">DNA Tutorials</a>
- Our DockerHub channel - <a href="https://hub.docker.com/u/dnaevolutions" target="_blank">DNA DockerHub</a>
- Our LinkedIn channel - <a href="https://www.linkedin.com/company/dna-evolutions/" target="_blank">DNA LinkedIn</a>

Third-party:
- <a href="https://angular.io/cli" target="_blank">Angular CLI</a>
- <a href="https://www.npmjs.com/package/@openapitools/openapi-generator-cli" target="_blank">Open Api Generator</a>
- <a href="https://www.mokkapps.de/blog/how-to-generate-angular-and-spring-code-from-open-api-specification/" target="_blank">Generate Angular and Spring code from OpenApi</a>
- <a href="https://material.angular.io/" target="_blank">Angular Material</a>

---

## Overview

* [General - DNA Demo Application](#general-dna-demo-application)
* [Serve with Docker (recommended)](#serve-with-docker-recommended)
* [Serve without Docker](#serve-without-docker)
* [Update/Generate the typescript models using OpenApi Generator](#update-generate-the-typescript-models-using-openapi-generator)
* [Video - How to use the Demo-Application](#video-how-to-use-the-demo-application)
* [What's next?](#what-s-next)

---

## General - DNA Demo Application

The angular-demo application was generated with <a href="https://github.com/angular/angular-cli" target="_blank">Angular CLI</a>. Further, we utilized <a href="https://github.com/OpenAPITools/openapi-generator" target="_blank">OpenApi Generator</a> to automatically create the required TypeScript models from the OpenApi definition provided from our Swagger endpoint of JOptTourOptimizer.

<img src="https://dna-evolutions.com/wp-content/uploads/2020/11/how-to-video-prev.gif" width="450"
title="DNA-Evolutions Demo Application Preview" alt="DNA-Evolutions Demo Application Preview">

[![Open in StackBlitz](https://developer.stackblitz.com/img/open_in_stackblitz.svg)](https://stackblitz.com/github/DNA-Evolutions/Angular-Demo-Application-Source) (this takes quite a while).
Please make sure you set a valid touroptizer-application endpoint. By default it is <a href="http://localhost:8081" target="_blank">[Angular CLI](http://localhost:8081)</a>. 
You can modify the endpoint in the ``src/environments/environment.ts`` or ``src/environments/environment.prod.ts`` under the key ``host`` and ``port``.

---

## Serve with Docker (recommended)

We recommend serving the Demo Application with docker, especially when you are new to Angular. The docker build will install all the required dependencies for you. You can modify the code in your desired IDE and build your custom image.

### Prerequisites

1) Install a Docker environment and make sure ``docker`` is available in your command line.

2) Even though the Demo-Application needs no TourOptimizer to start, you need to provide a running TourOptimizer instance to test the full functionality of the Demo-Application. Please read the documentation on how to <a href="https://github.com/DNA-Evolutions/Docker-REST-TourOptimizer/blob/main/README.md#how-to-start-jopttouroptimizer-docker" target="_blank">start the TourOptimizer</a>.


### Start the image

Please visit <a href="https://hub.docker.com/repository/docker/dnaevolutions/jopt_demoapplication/general" target="_blank">https://hub.docker.com/repository/docker/dnaevolutions/jopt_demoapplication/general</a> and follow the instructions if you want to start the pre-built image without using the sources. 

### Build the image from scratch

1) Clone this repository and open it in your desired IDE (e.g. <a href="https://code.visualstudio.com/" target="_blank">Visual Studio Code</a>) and modify the code if you like.
2) <a href="https://docs.docker.com/engine/reference/commandline/build/" target="_blank">Build</a> the docker image using the ``Dockerfile`` in the main project path, by running:

ATTENTION: Node.js during the build process can be quite memory consuming. So make sure your docker environment has at least 4 GB of Ram. 

```xml
docker build -t dna_custom_demo_image .
```

### Run a container

1) Run the image based on your freshly created image:

```xml
docker run -d --rm \
	--name myJOptTourOptimizerDemo \
	-p 3000:80 \
	-v ${PWD}:/usr/src/app \
	-e JOPT_SWAGGER_HOST="http://localhost" \
	-e JOPT_SWAGGER_PORT="8081" \
	dna_custom_demo_image
```

Same command as a single line:

```xml
docker run -d --rm --name myJOptTourOptimizerDemo -p 3000:80 -v ${PWD}:/usr/src/app -e JOPT_SWAGGER_HOST="http://localhost" -e JOPT_SWAGGER_PORT="8081" dna_custom_demo_image
```

Open <a href="http://localhost:3000" target="_blank">http://localhost:3000</a>, and you should see the Demo Application. If <a href="https://github.com/DNA-Evolutions/Docker-REST-TourOptimizer/blob/main/README.md#how-to-start-jopttouroptimizer-docker" target="_blank">JOptTourOptimizer</a> is not running in the background, you can **NOT** start any Optimization. However, you should see the starting page of the Demo Application. By default, the TourOptimizer is assumed to be available at <a href="http://localhost:8081" target="_blank">http://localhost:8081</a> and is started with the "cors" profile. You can adjust the default settings for host and port in the ``environment.ts`` file.

You can provide three environment variables:
* ``JOPT_SWAGGER_HOST``: The address of your running TourOptimizer. If you run the docker image of the TourOptimizer locally, this is ``http://localhost``.
* ``JOPT_SWAGGER_PORT``: The port of your running TourOptimizer. If you run the docker image of the TourOptimizer locally, this is ``8081``.
* ``INPRODUCTION``: This value is ``false`` by default. If you are changing it to ``true``, our official TourOptimizer endpoint is used by default. However, **try to avoid this**, as we are going to block IPs with too many requests. Further, our official endpoint has a limitation of 20 elements to be optimized, regardless of your license. You can change the production environment settings by editing the file: ``environment.prod.ts``.


---

## Serve without Docker

Of course, you can host the Demo-Application Angular Project locally without using docker. However, you have to make sure your environment is set up correctly.

### Prerequisites (for using ``npm`` and ``ng``)

1) First, you need to make sure <a href="https://www.npmjs.com/get-npm" target="_blank">npm (Node Package Manager)</a> is installed.   

2) Further, you need to install <a href="https://angular.io/cli" target="_blank">ng (Angular CLI)</a>

3) Even though the Demo-Application needs no TourOptimizer to start, you need to provide a running TourOptimizer instance to test the full functionality of the Demo-Application. Please read the documentation on how to <a href="https://github.com/DNA-Evolutions/Docker-REST-TourOptimizer/blob/main/README.md#how-to-start-jopttouroptimizer-docker" target="_blank">start the TourOptimizer</a>.


### Serve the Demo Application

1. Clone this repository and open it in your desired IDE (e.g. <a href="https://code.visualstudio.com/" target="_blank">Visual Studio Code</a>) if you like to modify the code.

2. You can serve the project by using ``ng serve`` from within the project path.

3. Open <a href="http://localhost:5000" target="_blank">http://localhost:5000</a>, and you should see the Demo Application. If <a href="https://github.com/DNA-Evolutions/Docker-REST-TourOptimizer/blob/main/README.md#how-to-start-jopttouroptimizer-docker" target="_blank">JOptTourOptimizer</a> is not running in the background, you can **NOT** start any Optimization. However, you should see the starting page of the Demo Application. By default, the TourOptimizer is assumed to be available at <a href="http://localhost:8081" target="_blank">http://localhost:8081</a> and is started with the "cors" profile. You can adjust the default settings for host and port in the ``environment.ts`` file. The default port, the Demo Application is hosted, is port 5000, you can change this behavior in the ``angular.json``.

(Please also read the <a href="https://angular.io/cli/build" target="_blank">Angular Documentation</a> to get further help on how to build projects.)


---
## Update/Generate the TypeScript models using OpenApi Generator

If you would like to automatically generate the models based on the <a href="https://swagger.dna-evolutions.com/v3/api-docs" target="_blank">latest Swagger API-Docs</a> of the
<a href="https://github.com/DNA-Evolutions/Docker-REST-TourOptimizer" target="_blank">TourOptimizer</a>, you will have to install <a href="https://www.npmjs.com/package/@openapitools/openapi-generator-cli" target="_blank">Openapi-Generator-CLI</a>.

Note: OpenApiGenerator can also be run as Docker container or Maven depedency.

### Create a touroptimizer_spec.json file
The ``touroptimizer_spec.json`` containing the API-Docs is not part of this repository. Copy and paste the Swagger definition under <a href="https://swagger.dna-evolutions.com/v3/api-docs" target="_blank">swagger.dna-evolutions.com/v3/api-docs</a> (or from <a href="http://localhost:8081/v3/api-docs" target="_blank">http://localhost:8081/v3/api-docs</a>). **Please, clean all 'oneOf' keys before using the schema.**

### Download touroptimizer_spec_cleaned.json file
 Visit <a href="https://github.com/DNA-Evolutions/Java-REST-Client-Examples/tree/master/src/main/resources/swagger/touroptimizer/spec/touroptimizer_spec_cleaned.json" target="_blank">cleaned schema on GitHub</a>) and save as a new file called ``touroptimizer_spec.json``. 

**Update June 2021:**

Native JSON is also supported from now on.

### Make the touroptimizer_spec.json available

Within the file ``package.json`` of the Demo-Application project, the script ``npm run generate:api`` is defined. It expects the ``touroptimizer_spec.json`` to be saved at ``../openapi/touroptimizer_spec.json``. Meaning, you will have to create a new folder called ``openapi`` next to your project folder (NOT inside the project folder itself) and save your ``touroptimizer_spec.json`` inside the ``openapi`` folder.

If this is done, you can call:

```xml
npm run generate:api
```

to update/generate the TypeScript models.

---

## Video - How to use the Demo-Application

Click to open video:

<a href="https://www.youtube.com/watch?v=2q7cYYArKm8 " target="_blank"><img src="https://img.youtube.com/vi/2q7cYYArKm8/maxresdefault.jpg" width="500"
title="Tutorial - How to use the Demo-Application" alt="Tutorial - How to use the Demo-Application"></a>
---

## What's next?

We are going to publish some tutorials on "how to create your own examples". Further, we are going to give some more insides on the structure of the Demo-Application.

---

## TODO's

* Improve in-code documentation.

---

## Agreement
For reading our license agreement and for further information about license plans, please visit <a href="https://www.dna-evolutions.com" target="_blank">www.dna-evolutions.com</a>.

--- 

## Authors
A product by [dna-evolutions ](https://www.dna-evolutions.com)&copy;

 readmeEtag: '"5938824631b590597504a3cb2a533a95d9ee8c31"' readmeLastModified: Tue, 11 Feb 2025 22:54:22 GMT repositoryId: 340891355 description: The latest source code of our Angular-Demo Application. created: '2021-02-21T12:03:16Z' updated: '2025-07-24T09:22:08Z' language: TypeScript archived: false stars: 4 watchers: 0 forks: 1 owner: DNA-Evolutions logo: https://avatars.githubusercontent.com/u/62150493?v=4 repoEtag: '"ebe9e73d38a68bad749d78c4c89638cc70289eefb4fdd7ebd4831c4fbc723014"' repoLastModified: Thu, 24 Jul 2025 09:22:08 GMT foundInMaster: true category: Server Implementations id: 32f0648c116dcc86eaec5b5beec2a428 - source: openapi3 tags repository: https://github.com/edavis25/plex-api-spec v3: true repositoryMetadata: base64Readme: >- IyBQbGV4IEFQSSBEb2N1bWVudGF0aW9uDQpPcGVuQVBJIHNwZWNpZmljYXRpb24gZm9yIHRoZSBQbGV4IHdlYiBBUEkuDQoNCiMjIyA6bGluazogW1ZpZXcgdGhlIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vZWRhdmlzMjUuZ2l0aHViLmlvL3BsZXgtYXBpLXNwZWMvKQ0KDQoNCiMjIyMgRHNlY3JpcHRpb24NClRoaXMgcHJvamVjdCBpcyBhbiBhdHRlbXB0IGF0IGRvY3VtZW50aW5nIHNvbWUgb2YgdGhlIG1hbnkgQVBJIGVuZHBvaW50cyB1c2VkIGJ5IHRoZSBQbGV4IHdlYiBhcHBsaWNhdGlvbi4gU2luY2UgUGxleCBpcyBjbG9zZWQtc291cmNlIGFuZCBkb2VzIG5vdCBwdWJsaWNseSBleHBvc2UgZG9jdW1lbnRhdGlvbiBmb3IgdGhlaXIgQVBJLCB0aGlzIHByb2plY3QgYXR0ZW1wdHMgdG8gZG9jdW1lbnQgdGhlIEFQSSdzIGZ1bmN0aW9uYWxpdHkgdXNpbmcgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbi4NCg0KVGhpcyBwcm9qZWN0IHdhcyBpbnNwaXJlZCBieSBNYXJjIFN0ZXBoZW5zb24ncyBbVW5vZmZpY2lhbCBQbGV4IEFQSSBEb2N1bWVudGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vQXJjYW5lbWFndXMvcGxleC1hcGkpLiBIaXMgZG9jdW1lbnRhdGlvbiBpcyBhbm90aGVyIGV4Y2VsbGVudA0KcmVzb3VyY2UgdGhhdCBJIGhpZ2hseSByZWNvbW1lbmQgd2hlbiBsZWFybmluZyBob3cgdG8gY29uc3VtZSB0aGUgUGxleCBBUEkuDQo= readmeEtag: '"3d6c96c9541543a939381186a8aff62e29652b19"' readmeLastModified: Thu, 10 Mar 2022 22:50:22 GMT repositoryId: 443420796 description: Documentation for the Plex Media Server web API. created: '2021-12-31T20:28:10Z' updated: '2023-09-08T18:30:12Z' language: null archived: false stars: 4 watchers: 2 forks: 2 owner: edavis25 logo: https://avatars.githubusercontent.com/u/14096584?v=4 repoEtag: '"6faa31cc3055fe2a43b3a2f786c54191c062ebe75cebcad60246b15849721527"' repoLastModified: Fri, 08 Sep 2023 18:30:12 GMT foundInMaster: true category: - Server - Server Implementations id: 09854d87bfafe437e01abb6fa83211b5 - source: openapi3 tags repository: https://github.com/data-studio/data-studio v3: true repositoryMetadata: base64Readme: >- IyBEYXRhIFN0dWRpbyBieSBFdmlyYXRlYwoKV1AgZXhwb3J0IGluOiBgLi9kb2MvZGIvZGF0YXN0dWRpb2J5ZXZpcmF0ZWMuLi54bWxgCgojIyBTcGVjaWFsIG5vdGVzLCBmdXJ0aGVyIHJlYWRpbmcsIGFuZCB0aGFua3MgdG8gLi4uCgpQbGVhc2Ugbm90ZSAqKnRoaXMgcHJvamVjdCBpcyBpbiBOTyBXQVkgYWZmaWxpYXRlZCB3aXRoOyBvciBzdXBwb3J0ZWQsIG9yIGVuZG9yc2VkIGJ5OyBhbnkgb2YgdGhlIGZvbGxvd2luZyBwZW9wbGUgb3Igb3JnYW5pc2F0aW9ucy4qKgoKVGhleSBoYXZlIHNwZWNpYWwgbWVudGlvbiBoZXJlIGFzIEkgZmVlbCB0aGV5IGRlc2VydmUgaXQgOikKCiMjIyBDUVJTCgpDb21tYW5kIFF1ZXJ5IFJlc3BvbnNpYmlsaXR5IFNlZ3JlZ2F0aW9uCgotIFtNYXJ0aW4gRm93bGVyXShodHRwczovL21hcnRpbmZvd2xlci5jb20pIGZvciB0aGlzIGd1aWRlIG9uIFtDUVJTXShodHRwczovL21hcnRpbmZvd2xlci5jb20vYmxpa2kvQ1FSUy5odG1sKQotIFtHcmVnIFlvdW5nXShodHRwczovL3R3aXR0ZXIuY29tL2dyZWd5b3VuZykgYXMgaGUgaXMgbWVudGlvbmVkIG9uIGh0dHBzOi8vbWFydGluZm93bGVyLmNvbS9ibGlraS9DUVJTLmh0bWwKLSBbRG91Z2xhcyBSZWl0aF0oaHR0cHM6Ly93d3cubGlua2VkaW4uY29tL2luL2RvdWdsYXMtcmVpdGgvKSBmb3IgaW50cm9kdWNpbmcgbWUgdG8gQ1FSUyBhbmQgZXhwbGFpbmluZyBpdCBpbiBhIHdheSBJIGNvdWxkIHVuZGVyc3RhbmQKCiMjIyBPcGVuQVBJCgotIFtPcGVuQVBJIEluaXRpYXRpdmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkpOiBodHRwczovL29wZW5hcGlzLm9yZyBhbmQgaXQncyBjb250cmlidXRvcnMuCgojIyMgTWlzc2luZyBzb21lb25lL3NvbWV0aGluZz8KCkxldCBtZSBrbm93IDopCgojIyBNb2RlbHMKCmBgYAphcHAgICAgICAgICAgIEFwcCAgICAgICAgICAgQXBwcyAgICAgICAgICAgYXBwICAgICAgICAgICBhcHBzCmxvZ2ljX2dyb3VwICAgTG9naWNHcm91cCAgICBMb2dpYyBHcm91cHMgICBsb2dpYyBncm91cCAgIGxvZ2ljIGdyb3Vwcwptb2RlbCAgICAgICAgIE1vZGVsICAgICAgICAgTW9kZWxzICAgICAgICAgbW9kZWwgICAgICAgICBtb2RlbHMKYXR0cmlidXRlICAgICBBdHRyaWJ1dGUgICAgIEF0dHJpYnV0ZXMgICAgIGF0dHJpYnV0ZSAgICAgYXR0cmlidXRlcwpjb21tYW5kICAgICAgIENvbW1hbmQgICAgICAgQ29tbWFuZHMgICAgICAgY29tbWFuZCAgICAgICBjb21tYW5kcwpxdWVyeSAgICAgICAgIFF1ZXJ5ICAgICAgICAgUXVlcmllcyAgICAgICAgcXVlcnkgICAgICAgICBxdWVyaWVzCmBgYAoKIyMgUXVlcnkKCmBgYApnZXRBcHAKZ2V0TG9naWNHcm91cApnZXRNb2RlbApnZXRBdHRyaWJ1dGUKZ2V0Q29tbWFuZApnZXRRdWVyeQoKZ2V0QXBwcwpnZXRMb2dpY0dyb3Vwc0J5QXBwCmdldE1vZGVsc0J5TG9naWNHcm91cApnZXRBdHRyaWJ1dGVzQnlNb2RlbApnZXRDb21tYW5kc0J5TG9naWNHcm91cApnZXRRdWVyaWVzQnlMb2dpY0dyb3VwCmBgYAoKIyMgQ01ECgpgYGAKY3JlYXRlQXBwCmNyZWF0ZUxvZ2ljR3JvdXAKY3JlYXRlTW9kZWwKY3JlYXRlQXR0cmlidXRlCmNyZWF0ZUNvbW1hbmQKY3JlYXRlUXVlcnkKCmRlbGV0ZUFwcApkZWxldGVMb2dpY0dyb3VwCmRlbGV0ZU1vZGVsCmRlbGV0ZUF0dHJpYnV0ZQpkZWxldGVDb21tYW5kCmRlbGV0ZVF1ZXJ5Cgp1cGRhdGVBcHBOYW1lCnVwZGF0ZUxvZ2ljR3JvdXBOYW1lCnVwZGF0ZU1vZGVsTmFtZQp1cGRhdGVBdHRyaWJ1dGVOYW1lCnVwZGF0ZUNvbW1hbmROYW1lCnVwZGF0ZVF1ZXJ5TmFtZQpgYGAKCiMjIyBSZXF1aXJlcyBXb3JkUHJlc3MgUGx1Z2lucwotIEFkdmFuY2VkIEN1c3RvbSBGaWVsZHMKLSBQcm9maWxlIEJ1aWxkZXIKCiMjIEJhc2VkIG9uIFtIVE1MNSBCbGFua10oaHR0cDovL2h0bWw1YmxhbmsuY29tKQoKSFRNTDUgQmxhbmsgaXMgYSBwb3dlcmZ1bCBzaGVsbCBmb3IgcmFwaWRseSBkZXBsb3lpbmcgeW91ciBXb3JkUHJlc3MgcHJvamVjdHMuCgoqIFByb2plY3Q6IFtnaXRodWIuY29tL3RvZGRtb3R0by9odG1sNWJsYW5rXShodHRwczovL2dpdGh1Yi5jb20vdG9kZG1vdHRvL2h0bWw1YmxhbmspCiogV2Vic2l0ZTogW2h0bWw1YmxhbmsuY29tXShodHRwOi8vaHRtbDVibGFuay5jb20pCiogVHdpdHRlcjogW0BodG1sNWJsYW5rXShodHRwOi8vdHdpdHRlci5jb20vaHRtbDVibGFuaykKKiBBdXRob3IgOiBbVG9kZCBNb3R0b10oaHR0cDovL3RvZGRtb3R0by5jb20pIC8vIFtAdG9kZG1vdHRvXShodHRwOi8vdHdpdHRlci5jb20vdG9kZG1vdHRvKQoKIyMjIENvbnRyaWJ1dG9ycyAoaW4gb3JkZXIgb2YgcHVsbCByZXF1ZXN0KQpbRGF2aWQgTXVubl0oaHR0cHM6Ly9naXRodWIuY29tL011bm5kYXkpLCBbUGF0cmljayBaZWluZXJ0XShodHRwczovL2dpdGh1Yi5jb20vQ29ldXNDQyksIFtKLVJhYmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9KLVJhYmUpLCBbU3RldmUgU3RlaW5lcl0oaHR0cHM6Ly9naXRodWIuY29tL3NzdGVpbmVyeCksIFtLeWxlIEh1ZHNvbl0oaHR0cHM6Ly9naXRodWIuY29tL2Rpc2todWIpLCBbY2hyaXNkbF0oaHR0cHM6Ly9naXRodWIuY29tL2NocmlzZGwpLCBbTWFyY2VsIE1pcmFuZGFdKGh0dHBzOi8vZ2l0aHViLmNvbS9yZWFrdGl2byksIFtGeCBCw6luYXJkXShodHRwczovL2dpdGh1Yi5jb20vZnhiZW5hcmQpLCBJb2FuIFZpcmFnLCBbTW9oYW1lZCBFbGtlYmlyXShodHRwczovL2dpdGh1Yi5jb20vZWxrZWJpcm1lZCksIFtscmVnbGFdKGh0dHBzOi8vZ2l0aHViLmNvbS9scmVnbGEpLCBDYXJsb3MgUGluYXIsIFtKb3NodWEgTHltYW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9qbHltYW4pLCBbS2V2aW4gUGxhdHRyZXRdKGh0dHBzOi8vZ2l0aHViLmNvbS9rZXZpbnBsYXR0cmV0KSwgW1dlc2xsZWkgSGVucmlxdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS93ZXNsbGVpKSwgW1N0YWNleSBDb3Jkb25pXShodHRwczovL2dpdGh1Yi5jb20vc3RhY2V5Y29yZG9uaSkuCgojIyBNb3JlIGluZm8gYWJvdXQgSFRNTDUgQmxhbmsKCkRvd25sb2FkIHRoZSBsYXRlc3QgdmVyc2lvbiBmcm9tIFtodG1sNWJsYW5rLmNvbV0oaHR0cDovL2h0bWw1YmxhbmsuY29tKSwgb3IgW2dpdGh1Yi5jb20vdG9kZG1vdHRvL2h0bWw1YmxhbmtdKGh0dHBzOi8vZ2l0aHViLmNvbS90b2RkbW90dG8vaHRtbDVibGFuaykuCgojIyBNSVQgTGljZW5zZQoKQ29weXJpZ2h0ICZjb3B5OyAyMDE4IENhbGxhbiBQZXRlciBNaWxuZQoKQnVpbHQgYnkgQ2FsbGFuIE1pbG5lIC8vIEBldmlyYXRlYwpCYXNlZCBvbiB3b3JrIChbSFRNTDUgQmxhbmtdKGh0dHA6Ly9odG1sNWJsYW5rLmNvbSkpIGJ5IFRvZGQgTW90dG8gLy8gQHRvZGRtb3R0bwoKUGVybWlzc2lvbiBpcyBoZXJlYnkgZ3JhbnRlZCwgZnJlZSBvZiBjaGFyZ2UsIHRvIGFueSBwZXJzb24gb2J0YWluaW5nIGEgY29weSBvZiB0aGlzIHNvZnR3YXJlIGFuZCBhc3NvY2lhdGVkIGRvY3VtZW50YXRpb24gZmlsZXMgKHRoZSAiU29mdHdhcmUiKSwgdG8gZGVhbCBpbiB0aGUgU29mdHdhcmUgd2l0aG91dCByZXN0cmljdGlvbiwgaW5jbHVkaW5nIHdpdGhvdXQgbGltaXRhdGlvbiB0aGUgcmlnaHRzIHRvIHVzZSwgY29weSwgbW9kaWZ5LCBtZXJnZSwgcHVibGlzaCwgZGlzdHJpYnV0ZSwgc3VibGljZW5zZSwgYW5kL29yIHNlbGwgY29waWVzIG9mIHRoZSBTb2Z0d2FyZSwgYW5kIHRvIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIFNvZnR3YXJlIGlzIGZ1cm5pc2hlZCB0byBkbyBzbywgc3ViamVjdCB0byB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnM6CgpUaGUgYWJvdmUgY29weXJpZ2h0IG5vdGljZSBhbmQgdGhpcyBwZXJtaXNzaW9uIG5vdGljZSBzaGFsbCBiZSBpbmNsdWRlZCBpbiBhbGwgY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS4KClRIRSBTT0ZUV0FSRSBJUyBQUk9WSURFRCAiQVMgSVMiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELCBFWFBSRVNTIE9SIElNUExJRUQsIElOQ0xVRElORyBCVVQgTk9UIExJTUlURUQgVE8gVEhFIFdBUlJBTlRJRVMgT0YgTUVSQ0hBTlRBQklMSVRZLCBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULiBJTiBOTyBFVkVOVCBTSEFMTCBUSEUgQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWSBDTEFJTSwgREFNQUdFUyBPUiBPVEhFUiBMSUFCSUxJVFksIFdIRVRIRVIgSU4gQU4gQUNUSU9OIE9GIENPTlRSQUNULCBUT1JUIE9SIE9USEVSV0lTRSwgQVJJU0lORyBGUk9NLCBPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRSBTT0ZUV0FSRSBPUiBUSEUgVVNFIE9SIE9USEVSIERFQUxJTkdTIElOIFRIRSBTT0ZUV0FSRS4K readmeEtag: '"67ddde20d93ef634c55bec672ecb3e66f354da75"' readmeLastModified: Wed, 24 Oct 2018 00:11:46 GMT repositoryId: 133067860 description: DataStudio by Eviratec (datastudio.xyz) as a WordPress Theme created: '2018-05-11T17:13:28Z' updated: '2024-05-03T14:59:12Z' language: PHP archived: false stars: 4 watchers: 1 forks: 1 owner: data-studio logo: https://avatars.githubusercontent.com/u/28729308?v=4 license: NOASSERTION repoEtag: '"f1426647c457824b4a7625d331d75059e00a961b4b033fb63e525091a7c8329b"' repoLastModified: Fri, 03 May 2024 14:59:12 GMT foundInMaster: true category: Server id: 565b3259a85c277be27ca7225a6355cf - source: - openapi3 tags - openapi31 tags repository: https://github.com/apiaddicts/sonaropenapi-rules v3: true id: 2be35a610f6c92cf6cbb96985485bb4b repositoryMetadata: base64Readme: >- CgojIPCfm6DvuI8gc29uYXJvcGVuYXBpLXJ1bGVzICAhW1JlbGVhc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcmVsZWFzZS0xLjIuMS1wdXJwbGUpICFbSmF2YV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9qYXZhLSUyM0VEOEIwMC5zdmc/c3R5bGU9ZmxhdCZsb2dvPW9wZW5qZGsmbG9nb0NvbG9yPXdoaXRlKSAgWyFbTGljZW5zZTogTEdQTCB2M10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLUxHUExfdjMtYmx1ZS5zdmcpXShodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2xncGwtMy4wKSAKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBhIHNldCBvZiBjdXN0b20gU29uYXJRdWJlIHJ1bGVzIHNwZWNpZmljYWxseSBkZXNpZ25lZCB0byBhbmFseXplIGFuZCBpbXByb3ZlIHRoZSBxdWFsaXR5IG9mIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMuIEJ5IGludGVncmF0aW5nIHRoZXNlIHJ1bGVzLCB0ZWFtcyBjYW4gZW5zdXJlIGJlc3QgcHJhY3RpY2VzLCBtYWludGFpbmFiaWxpdHksIGFuZCBjb25zaXN0ZW5jeSBpbiB0aGVpciBBUEkgZGVmaW5pdGlvbnMuCgojIyMgVGhpcyByZXBvc2l0b3J5IGlzIGludGVuZGVkIGZvciA6b2N0b2NhdDogKipjb21tdW5pdHkqKiB1c2UsIGl0IGNhbiBiZSBtb2RpZmllZCBhbmQgYWRhcHRlZCB3aXRob3V0IGNvbW1lcmNpYWwgdXNlLiBJZiB5b3UgbmVlZCBhIHZlcnNpb24sIHN1cHBvcnQgb3IgaGVscCBmb3IgeW91ciAqKmVudGVycHJpc2UqKiBvciBwcm9qZWN0LCBwbGVhc2UgY29udGFjdCB1cyDwn5OnIGRldnJlbEBhcGlhZGRpY3RzLm9yZwojIyMg8J+SoSBJZiB5b3UgaGF2ZSBhbiBpZGVhIGZvciBhIHJ1bGUgYnV0IHlvdSBhcmUgbm90IHN1cmUgdGhhdCBldmVyeW9uZSBuZWVkcyBpdCB5b3UgY2FuIGltcGxlbWVudCBhIFtjdXN0b20gcnVsZV0oQ3VzdG9tUnVsZXMubWQpIGF2YWlsYWJsZSBvbmx5IGZvciB5b3UuCgpbIVtUd2l0dGVyXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1R3aXR0ZXItJTIzMDAwMDAwLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289eCZsb2dvQ29sb3I9d2hpdGUpXShodHRwczovL3R3aXR0ZXIuY29tL0FQSUFkZGljdHMpIApbIVtEaXNjb3JkXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0Rpc2NvcmQtJTIzNTg2NUYyLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289ZGlzY29yZCZsb2dvQ29sb3I9d2hpdGUpXShodHRwczovL2Rpc2NvcmQuZ2cvWmRiR3FNQll5OCkKWyFbTGlua2VkSW5dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGlua2VkaW4tJTIzMDA3N0I1LnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289bGlua2VkaW4mbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly93d3cubGlua2VkaW4uY29tL2NvbXBhbnkvYXBpYWRkaWN0cy8pClshW0ZhY2Vib29rXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0ZhY2Vib29rLSUyMzE4NzdGMi5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPUZhY2Vib29rJmxvZ29Db2xvcj13aGl0ZSldKGh0dHBzOi8vd3d3LmZhY2Vib29rLmNvbS9hcGlhZGRpY3RzKQpbIVtZb3VUdWJlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1lvdVR1YmUtJTIzRkYwMDAwLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289WW91VHViZSZsb2dvQ29sb3I9d2hpdGUpXShodHRwczovL3d3dy55b3V0dWJlLmNvbS9AQVBJQWRkaWN0c2xtYW9vKQoKIyDwn5mMIEpvaW4gdGhlICoqZG9Tb25hckFwaSoqIEFkb3B0ZXJzIGxpc3QgCvCfk6IgSWYgZG9Tb25hckFwaSBpcyBwYXJ0IG9mIHlvdXIgb3JnYW5pemF0aW9uJ3MgdG9vbGtpdCwgd2Uga2luZGx5IGVuY291cmFnZSB5b3UgdG8gaW5jbHVkZSB5b3VyIGNvbXBhbnkncyBuYW1lIGluIG91ciBBZG9wdGVycyBsaXN0LiDwn5mPIFRoaXMgbm90IG9ubHkgc2lnbmlmaWNhbnRseSBib29zdHMgdGhlIHByb2plY3QncyB2aXNpYmlsaXR5IGFuZCByZXB1dGF0aW9uIGJ1dCBhbHNvIHJlcHJlc2VudHMgYSBzbWFsbCB5ZXQgaW1wYWN0ZnVsIHdheSB0byBnaXZlIGJhY2sgdG8gdGhlIHByb2plY3QuCgp8IE9yZ2FuaXphdGlvbiAgfCBEZXNjcmlwdGlvbiBvZiBVc2UgLyBSZWZlcmVuYyB8CnwtLS18LS0tfAp8ICBbQ2xvdWRBcHBpXShodHRwczovL2Nsb3VkYXBwaS5uZXQvKSAgfCBBcGlmaWNhdGlvbiBhbmQgZ2VuZXJhdGlvbiBvZiBtaWNyb3NlcnZpY2VzIHwKfCBbTWFkcmlkIERpZ2l0YWxdKGh0dHBzOi8vd3d3LmNvbXVuaWRhZC5tYWRyaWQvc2VydmljaW9zL3NlZGUtZWxlY3Ryb25pY2EvbWFkcmlkLWRpZ2l0YWwvKSAgfCBHZW5lcmF0aW9uIG9mIG1pY3Jvc2VydmljZXMgIHwKfCBbQXBpcXVhbGl0eV0oaHR0cHM6Ly9hcGlxdWFsaXR5LmlvLykgIHwgR2VuZXJhdGlvbiBvZiBtaWNyb3NlcnZpY2VzICB8CgojIPCfkanwn4+94oCN8J+SuyAgQ29udHJpYnV0ZSB0byBBcGlBZGRpY3RzIAoKV2UncmUgYW4gaW5jbHVzaXZlIGFuZCBvcGVuIGNvbW11bml0eSwgd2VsY29taW5nIHlvdSB0byBqb2luIG91ciBlZmZvcnQgdG8gZW5oYW5jZSBBcGlBZGRpY3RzLCBhbmQgd2UncmUgZXhjaXRlZCB0byBwcmlvcml0aXplIHRhc2tzIGJhc2VkIG9uIGNvbW11bml0eSBpbnB1dCwgaW52aXRpbmcgeW91IHRvIHJldmlldyBhbmQgY29sbGFib3JhdGUgdGhyb3VnaCBvdXIgR2l0SHViIGlzc3VlIHRyYWNrZXIuCgpGZWVsIGZyZWUgdG8gZHJvcCBieSBhbmQgZ3JlZXQgdXMgb24gb3VyIEdpdEh1YiBkaXNjdXNzaW9uIG9yIERpc2NvcmQgY2hhdC4gWW91IGNhbiBhbHNvIHNob3cgeW91ciBzdXBwb3J0IGJ5IGdpdmluZyB1cyBzb21lIEdpdEh1YiBzdGFycyDirZDvuI8sIG9yIGJ5IGZvbGxvd2luZyB1cyBvbiBUd2l0dGVyLCBMaW5rZWRJbiwgYW5kIHN1YnNjcmliaW5nIHRvIG91ciBZb3VUdWJlIGNoYW5uZWwhIPCfmoAKClshWyJCdXkgTWUgQSBDb2ZmZWUiXShodHRwczovL3d3dy5idXltZWFjb2ZmZWUuY29tL2Fzc2V0cy9pbWcvY3VzdG9tX2ltYWdlcy9vcmFuZ2VfaW1nLnBuZyldKGh0dHBzOi8vd3d3LmJ1eW1lYWNvZmZlZS5jb20vYXBpYWRkaWN0cykKCgojIPCfk5EgR2V0dGluZyBzdGFydGVkIAoKIyMg8J+UjSBDb25maWd1cmUgc2Nhbm5lcgoKIyMjIE1hdmVuIHBsdWdpbgoKIyMjIyBDb25maWd1cmUgcHJvcGVydGllcwoKSW4gYHBvbS54bWxgIGNvbmZpZ3VyZToKCmBgYGB4bWwKICAgIDxwcm9wZXJ0aWVzPgogICAgICAgIDwhLS0gT3B0aW9uYWwsIFdoZW4gaXMgc2V0IG9ubHkgdGhlIGxhbmd1YWdlIHNwZWNpZmllZCBpcyBhbmFseXplZCAtLT4KICAgICAgICA8c29uYXIubGFuZ3VhZ2U+b3BlbmFwaTwvc29uYXIubGFuZ3VhZ2U+CiAgICAgICAgPCEtLSBPcHRpb25hbCwgRGVmYXVsdCB2YWx1ZSBpcyBzcmMvbWFpbixwb20ueG1sIC0tPgogICAgICAgIDxzb25hci5zb3VyY2VzPi48L3NvbmFyLnNvdXJjZXM+CiAgICA8L3Byb3BlcnRpZXM+CmBgYGAKCiMjIyMgUnVuIHNjYW5uZXIKCmBtdm4gc29uYXI6c29uYXIgLURzb25hci5ob3N0LnVybD08SE9TVD4gLURzb25hci5sb2dpbj08S0VZPmAKCiMjIyBFeHRlcm5hbCBgc29uYXItc2Nhbm5lcmAKCiMjIyMgSW5zdGFsbCBgc29uYXItc2Nhbm5lcmAKCkRvd25sb2FkIHRoZSBgc29uYXItc2Nhbm5lcmAgZnJvbSBodHRwczovL2RvY3Muc29uYXJxdWJlLm9yZy9sYXRlc3QvYW5hbHlzaXMvc2Nhbi9zb25hcnNjYW5uZXIvIGFuZCBtYWtlIGl0IGFjY2Vzc2libGUuCgojIyMjIENvbmZpZ3VyZSBwcm9wZXJ0aWVzCgpJbiBgc29uYXItcHJvamVjdC5wcm9wZXJ0aWVzYCAoZmlsZSBpbiByb290IHByb2plY3QgZm9sZGVyKSBjb25maWd1cmU6CgpgYGBgcHJvcGVydGllcwojIG11c3QgYmUgdW5pcXVlIGluIGEgZ2l2ZW4gU29uYXJRdWJlIGluc3RhbmNlCnNvbmFyLnByb2plY3RLZXk9dGVzdDp0ZXN0CiMgdGhpcyBpcyB0aGUgbmFtZSBhbmQgdmVyc2lvbiBkaXNwbGF5ZWQgaW4gdGhlIFNvbmFyUXViZSBVSS4gV2FzIG1hbmRhdG9yeSBwcmlvciB0byBTb25hclF1YmUgNi4xLgpzb25hci5wcm9qZWN0TmFtZT1PcGVuQVBJIHBsdWdpbiB0ZXN0cwpzb25hci5wcm9qZWN0VmVyc2lvbj0xLjAtU05BUFNIT1QKCiMgUGF0aCBpcyByZWxhdGl2ZSB0byB0aGUgc29uYXItcHJvamVjdC5wcm9wZXJ0aWVzIGZpbGUuIFJlcGxhY2UgIlwiIGJ5ICIvIiBvbiBXaW5kb3dzLgojIFRoaXMgcHJvcGVydHkgaXMgb3B0aW9uYWwgaWYgc29uYXIubW9kdWxlcyBpcyBzZXQuCnNvbmFyLnNvdXJjZXM9LgoKIyBFbmNvZGluZyBvZiB0aGUgc291cmNlIGNvZGUuIERlZmF1bHQgaXMgZGVmYXVsdCBzeXN0ZW0gZW5jb2RpbmcKc29uYXIuc291cmNlRW5jb2Rpbmc9VVRGLTgKIyBTZWxlY3QgdGhlIGxhbmd1YWdlIHRvIHVzZSBmb3IgYW5hbHlzaXMKc29uYXIubGFuZ3VhZ2U9b3BlbmFwaQpgYGBgCgojIyMjIOKWtu+4jyBSdW4gc2Nhbm5lcgoKYHNvbmFyLXNjYW5uZXIgLURzb25hci5ob3N0LnVybD08SE9TVD4gLURzb25hci5sb2dpbj08S0VZPmAKCiMjIOKchSBDb21wYXRpYmlsaXR5CgpUaGlzIHBsdWdpbiBpcyBzdXBwb3J0ZWQgYnkgU29uYXJRdWJlIHZlcnNpb25zIGdyZWF0ZXIgb3IgZXF1YWwgdG8gYDYuNy40YAoKIyMjIEV4cGxpY2l0IGNvbXBhdGliaWxpdHkgdmVyc2lvbnMgdGVzdGVkCgp8IFZlcnNpb24gfAp8LS0tLS0tLS0tfAp8IGA2LjcuNGAgfAp8IGA3LjktY29tbXVuaXR5YCB8CnwgYDguMy1jb21tdW5pdHlgIHwKCiMjIPCfkpsgU3BvbnNvcnMKPGltZyBzcmM9Imh0dHBzOi8vYXBpYWRkaWN0cy5jbG91ZGFwcGkubmV0L3dlYi9pbWFnZS80MjQ4L0xPR09DbG91ZGFwcGkyMDIwVmVyc2lvbmVzLTAxLnBuZyIgYWx0PSJjbG91ZGFwcGkiIHdpZHRoPSIxNTAiLz4KPGltZyBzcmM9Imh0dHBzOi8vd3d3LmNvbXVuaWRhZC5tYWRyaWQvc2l0ZXMvZGVmYXVsdC9maWxlcy9zdHlsZXMvYmxvY2tfdGVhc2VyX2ltYWdlL3B1YmxpYy9pbWcvbG9nb3Mtc2ltYm9sb3MvbG9nb19jZW50cmFkb19tZC5wbmc/aXRvaz00clRVaG1jaiIgYWx0PSJtZCIgd2lkdGg9IjE1MCIvPgo8aW1nIHNyYz0iaHR0cHM6Ly9hcGlxdWFsaXR5LmlvL3dwLWNvbnRlbnQvdXBsb2Fkcy8yMDIyLzA5L2Nyb3BwZWQtbG9nby1hcGlxdWFsaXR5LXByaW5jaXBhbC0xLTE3MHg3MC5wbmciIGhlaWdodCA9ICI3NSI+CjxpbWcgc3JjPSJodHRwczovL2FwaWFkZGljdHMtd2ViLnMzLmV1LXdlc3QtMS5hbWF6b25hd3MuY29tL3dwLWNvbnRlbnQvdXBsb2Fkcy8yMDIyLzAzLzE3MTU1NzM2L2Nyb3BwZWQtQVBJQWRkaWN0cy1sb2dvdGlwb19yb2pvLnBuZyIgaGVpZ2h0ID0gIjc1Ij4K readmeEtag: '"bdd2aba7b120f647ba7bf8b8ffe41420c78e9880"' readmeLastModified: Mon, 19 May 2025 14:55:19 GMT repositoryId: 317901907 description: >- sonaropenapi-rules is a set of rules to validate OpenAPI / Swagger api definitions in SonarQube, you also need to install the evaluation engine. created: '2020-12-02T15:15:59Z' updated: '2026-01-08T16:42:17Z' language: HTML archived: false stars: 19 watchers: 2 forks: 0 owner: apiaddicts logo: https://avatars.githubusercontent.com/u/31730093?v=4 license: LGPL-3.0 repoEtag: '"54dc86e533d8fbf9fa7f23a9652d7022068660abb1d872c82c3452fca46bd53f"' repoLastModified: Thu, 08 Jan 2026 16:42:17 GMT category: - Code Generators - Server Implementations foundInMaster: true v3_1: true oldLocations: - https://github.com/apiaddicts/dosonarapi - source: openapi3 tags repository: https://github.com/guillaumedeconinck/todos-fastify v3: true repositoryMetadata: base64Readme: >- IyBUb2RvcyBBUEkKCiMjIFdoYXQgaXMgdGhpcyByZXBvc2l0b3J5ID8KClRoaXMgcmVwb3NpdG9yeSwgYXMgdGhlIG5hbWUgc3VnZ2VzdHMsIGlzIGFuIGltcGxlbWVudGF0aW9uIG9mIGEgVG9kb3MgQVBJLiBUaGUgbWFpbiBnb2FsIG9mIHRoaXMgQVBJIGlzIHRvIHNlcnZlIGFzIGEgInByb29mIiAob3IgInBvcnRmb2xpbyIpIG9mIG15IG93biBza2lsbHMuIFRoaXMgQVBJIGlzIGltcGxlbWVudGVkIGluICoqVHlwZXNjcmlwdCoqIGFuZCB1c2VzICoqTm9kZUpTIDE2KiouCgpXaGlsZSBidWlsZGluZyB0aGlzIEFQSSwgSSdtIGFsc28gc29tZXRpbWVzIGxlYXJuaW5nIG5ldyBzdHVmZiBhbmQgYXBwbHlpbmcgbmV3IGNvbmNlcHRzIHRoYXQgSSBoYXZlbid0IGhhZCB0aGUgb2NjYXNpb24gdG8gdXNlIGJlZm9yZS4KCklkZWFsbHksIEkgd291bGQgbGlrZSB0aGlzIEFQSSB0byBmb2xsb3cgYSBmZXcgcHJpbmNpcGxlczoKCi0gWzEyLWZhY3RvciBhcHBdKGh0dHBzOi8vMTJmYWN0b3IubmV0LykKLSBESSAoRGVwZW5kZW5jeSBJbmplY3Rpb24pIHdpdGggaW50ZXJmYWNlcywgdGhhbmtzIHRvIFt0c3lyaW5nZV0oaHR0cHM6Ly9naXRodWIuY29tL21pY3Jvc29mdC90c3lyaW5nZSkKLSBNaWNyb3NlcnZpY2VzIHBhdHRlcm5zCiAgLSBNb3JlIHNwZWNpZmljYWxseSBlbnN1cmUgdGhhdCBldmVudHMgYXJlIGVtaXR0ZWQgYXQgbGVhc3Qgb25jZSAobm90IHlldCBpbXBsZW1lbnRlZCwgc2VlIFsjMTVdKGh0dHBzOi8vZ2l0aHViLmNvbS9HdWlsbGF1bWVEZWNvbmluY2svdG9kb3MtZmFzdGlmeS9pc3N1ZXMvMTUpKQogIC0gTW9yZSBEREQgbWF5YmUgKEkgaGF2ZSB0byAocmUpcmVhZCBtb3JlIGFydGljbGVzL2Jvb2tzIG9uIHRoaXMpCi0gUHJvcGVybHkgZG9jdW1lbnRlZAotIEdvb2QgZXJyb3IgbWVzc2FnZXMsIGFzIHRoZSBoYXBweSBmbG93IGlzIG5vdCB0aGUgb25seSBmbG93CgojIyBTdHJ1Y3R1cmUgb2YgdGhlIHByb2plY3QKCnwgRm9sZGVyIG9yIGZpbGUgICAgICAgICB8IEV4cGxhbmF0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8CnwgYF9fdGVzdHNfX2AgICAgICAgICAgICB8IFVuaXQgYW5kIGUyZSB0ZXN0cyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgYC5naXRodWJgICAgICAgICAgICAgICB8IEdpdGh1YiB3b3JrZmxvdyBmb2xkZXIgKENJL0NEKSAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgYHNyY2AgICAgICAgICAgICAgICAgICB8IFNvdXJjZSBjb2RlLCBhcyBleHBlY3RlZCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgYGRvY2tlci1jb21wb3NlKi55YW1sYCB8IEFsbCB0aGUgZG9ja2VyLWNvbXBvc2UgZmlsZXMsIHVzZWQgZm9yIGRldiAmIGUyZSB0ZXN0cyB8CnwgYG9wZW5hcGkueWFtbGAgICAgICAgICB8ICoqT3BlbkFQSSBkZWZpbml0aW9uKiogb2YgdGhlIEFQSSAgICAgICAgICAgICAgICAgICAgICB8CgpNb3Jlb3ZlciwgdGhlIGBzcmNgIGZvbGRlciBpcyBkaXZpZGVkIGFzIGZvbGxvd3MgKG5vdCBpbiBhbHBoYWJldGljYWwgb3JkZXIpOgoKfCBGb2xkZXIgb3IgZmlsZSAgIHwgRXhwbGFuYXRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8CnwgYHJlc3RgICAgICAgICAgICB8IFJFU1QsIHNvIHJvdXRlIGhhbmRsZXJzLCBtaWRkbGV3YXJlcywgZXRjLiBObyBsb2dpYyBoZXJlICEgICAgICAgICAgICAgICAgICAgIHwKfCBgYXBwbGljYXRpb25gICAgIHwgU2VydmljZXMgZG9pbmcgdGhlIGxpbmsgYmV0d2VlbiBSRVNUIGFuZCBkb21haW4gKG5vdCByZWFsbHkgdXNlZnVsIHJpZ2h0IG5vdykgfAp8IGBkb21haW5gICAgICAgICAgfCBUaGUgbW9kZWxzIGFuZCByZXBvc2l0b3JpZXMnIGludGVyZmFjZXMsIHRoYXQncyB3aGVyZSB0aGUgbG9naWMgc2hvdWxkIGJlICAgICB8CnwgYGluZnJhc3RydWN0dXJlYCB8IEFsbCB0aGUgcmVwb3NpdG9yaWVzIGltcGxlbWVudGF0aW9ucywgZS5nLiBQb3N0Z3JlcyAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgdG9vbHNgICAgICAgICAgIHwgVXRpbGl0aWVzIGZpbGVzIHN1Y2ggYXMgZW52IHZhcnMsIGxvZ2dlciAmIG1haW50ZW5hbmNlIHN0dWZmICAgICAgICAgICAgICAgICAgfAoKPiBUZWNobmljYWxseSBgcmVzdGAgY291bGQgYmUgaW4gYGluZnJhc3RydWN0dXJlYAoKPiAqKlRoaXMgc3RydWN0dXJlIGNhbiBiZSBzZWVuIGFzIG92ZXJraWxsIGZvciB0aGUgc2l6ZSBvZiB0aGlzIEFQSSAoYXMgdGhlcmUgaXMgb25seSBvbmUgcmVzb3VyY2UgZm9yIG5vdykqKgoKIyMgSG93IHRvIHVzZQoKIyMjIFByZXJlcXVpc2l0ZXMKCkZvciBydW5uaW5nIHRoaXMgcHJvamVjdCwgaXQncyBleHBlY3RlZCB0aGF0IHlvdSBoYXZlIGFscmVhZHkgb24geW91ciBtYWNoaW5lCgotIE5vZGUgMTYgJiBucG0gOAotIERvY2tlcgoKIyMjIFJ1bm5pbmcgdGhlIEFQSQoKVGhlIHByb2plY3QgaXMgcXVpdGUgc2ltcGxlIHRvIHVzZSwgZmlyc3QgY2xvbmUgdGhlIHJlcG9zaXRvcnkgYW55d2hlcmUgb24geW91ciBtYWNoaW5lOgoKYGBgc2gKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9HdWlsbGF1bWVEZWNvbmluY2svdG9kb3MtZmFzdGlmeS5naXQKYGBgCgpUaGVuLCBpbnN0YWxsIHRoZSBkZXBlbmRlbmNpZXM6CgpgYGBzaApucG0gY2kKYGBgCgo+IGBucG1gIGlzIHVzZWQgaGVyZSwgYnV0IHlvdSBjYW4gdXNlIGB5YXJuYCBvciBgcG5wbWAuCgpBcyB0aGUgQVBJIGlzIGV4cGVjdGVkIHRvIHJ1biBpbiBhIGNvbnRhaW5lciAoZS5nLiBpbiBhIEt1YmVybmV0ZXMgY2x1c3RlciksIEkgcmVseSBoZWF2aWx5IG9uIERvY2tlciwgZXZlbiBmb3IgZGV2ZWxvcG1lbnQgYW5kIHRlc3RzLiBGb3IgcnVubmluZyB0aGUgYXBpLCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgc2gKbnBtIHJ1biBkZXYKYGBgCgpBZnRlciBhIH5+c2hvcnR+fiB0aW1lLCB5b3Ugc2hvdWxkIGhhdmUgYSBydW5uaW5nIEFQSSBvbiBwb3J0IGA5MDAyYCAoZGVmYXVsdCkKCiMjIyBNb25pdG9yaW5nCgpUaGUgQVBJIGNhbiBwcm92aWRlIHRyYWNlcyBvZiB3aGF0IGlzIGhhcHBlbmluZyB0byB0aGUgZW52IHZhciBgT1RMUF9HUlBDX0VORFBPSU5UYC4KClRoZSBzcGFucyBhcmUgZ2VuZXJhdGVkIHdpdGggW09wZW5UZWxlbWV0cnldKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuLXRlbGVtZXRyeS9vcGVudGVsZW1ldHJ5LWpzKSBhbmQgaGF2ZSBiZWVuIHRlc3RlZCB3aXRoIFtTaWdOb3pdKGh0dHBzOi8vZ2l0aHViLmNvbS9TaWdOb3ovc2lnbm96KS4KCjxkZXRhaWxzPgogIDxzdW1tYXJ5PlNjcmVlbnNob3RzIGluIFNpZ05vejwvc3VtbWFyeT4KICAKICAhW0xpc3Qgb2YgZW5kcG9pbnRzIHdpdGggdGhlaXIgbWV0cmljc10oZG9jcy9pbWFnZXMvVG9kb3NUcmFjZXNFbmRwb2ludHMucG5nKQogICFbU3BhbnMgb2YgYSBQVVQgcmVxdWVzdF0oZG9jcy9pbWFnZXMvVG9kb3NUcmFjZXNQVVQucG5nKQo8L2RldGFpbHM+CgojIyMgTWlncmF0aW9ucwoKPiBPdXRkYXRlZCwgdG8gYmUgcmV3cml0dGVuIHNvb24KCkFzIHlvdSBtYXkgaGF2ZSBub3RpY2VkLCBtaWdyYXRpb25zIHdlcmVuJ3QgbWVudGlvbmVkIGF0IGFsbCBpbiB0aGUgcHJldmlvdXMgc2VjdGlvbi4gKipUaGUgbWlncmF0aW9ucyBhcmUgaW4gZmFjdCBydW4gZHVyaW5nIHRoZSBzdGFydHVwIG9mIHRoZSBBUEkuKiogVGhpcyBpcyBub3QgYWx3YXlzIHRoZSBiZXN0IHRoaW5nIHRvIGRvIChlLmcuIGEgaGlnaGx5IHNlbnNpdGl2ZSBtaWdyYXRpb24gdGhhdCBjYW5ub3QgYmUgYXZvaWRlZCksIGJ1dCBmb3IgdGhpcyBzaW1wbGUgQVBJLCBpdCdzIG9rYXkuCgpUbyBjcmVhdGUgYSBuZXcgbWlncmF0aW9uLCBzaW1wbHkgcnVuCgpgYGBzaApucHggZGItbWlncmF0ZSBjcmVhdGUgdGhlLW5hbWUtb2YtdGhlLW1pZ3JhdGlvbgpgYGAKCkl0IHdpbGwgY3JlYXRlIGEgbmV3IGZpbGUgaW4gdGhlIGBtaWdyYXRpb25zYCBmb2xkZXIsIHdpdGggYSB0aW1lc3RhbXAgYW5kIHRoZSBuYW1lIHByb3ZpZGVkLgoKIyMgVGVzdGluZyB0aGUgQVBJCgojIyMgVW5pdCB0ZXN0cwoKU2ltcGx5IHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQKCmBgYHNoCm5wbSBydW4gdGVzdApgYGAKCiMjIyBFMkUgdGVzdHMKClJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQKCmBgYHNoCm5wbSBydW4gZG9ja2VyOmUyZQpgYGAKClRoaXMgd2lsbCBsYXVuY2ggYSBmZXcgY29udGFpbmVyczoKCi0gVGhlIGN1cnJlbnQgQVBJCi0gQSBQb3N0Z3JlcyBkYXRhYmFzZSwgdXNlZCBieSB0aGUgQVBJCi0gQSB0ZXN0IHJ1bm5lciAtPiBleGVjdXRlcyBKZXN0CgpUaGUgQVBJIGhhcyBhIHZvbHVtZSBtYXBwZWQgdG8gYHNyY2Agd2l0aCBgbm9kZW1vbmAgcnVubmluZy4gU28gYW55IGNoYW5nZSB0byB0aGUgc291cmNlIGNvZGUgd2lsbCB0cmlnZ2VyIGEgcmVzdGFydCBvZiB0aGUgQVBJLgoKVGhlIHRlc3QgcnVubmVyIGhhcyBhIHZvbHVtZSBtYXBwZWQgdG8gYF9fdGVzdHNfX2Agd2l0aCBgbm9kZW1vbmAgcnVubmluZyBhcyB3ZWxsLiBBbnkgY2hhbmdlIHRvIGEgdGVzdCB3aWxsIHJlLWxhdW5jaCB0aGUgd2hvbGUgRTJFIHN1aXRlLgoKPiBZb3UgY2FuIHRlc3QvZm9jdXMgb25seSBvbiBvbmUgdGVzdCBmaWxlIGJ5IGNoYW5naW5nIHRoZSBgY29tbWFuZDpgIG9wdGlvbiBpbiB0aGUgZmlsZSBgZG9ja2VyLWNvbXBvc2UuZTJlLmxvY2FsLnlhbWxgLgoKIyMgQ29udmVudGlvbnMKCkEgYC5lZGl0b3Jjb25maWdgIGZpbGUgd2lsbCBzb29uIGJlIGFkZGVkLiBJbiB0aGUgbWVhbiB0aW1lLCB0aGlzIHByb2plY3QgdXNlcyBgcHJldHRpZXJgIHdpdGggYGVzbGludGAgdG8gZW5zdXJlIHRoYXQgdGhlIGNvZGViYXNlIGZvbGxvd3MgdGhlIGNvbnZlbnRpb25zIHByb3Blcmx5LgoKIyMgUm9hZG1hcAoKVGhlIHJvYWRtYXAgaXMga2luZGEgZGVzY3JpYmVkIGluIHRoZSBpc3N1ZXMgb2YgdGhpcyByZXBvc2l0b3J5LCBidXQgaW4gInNob3J0IiBhbmQgbm90IGluIG9yZGVyOgoKLSBGaW5pc2ggdGhlIENSVUQgb2YgVG9kb3MKLSBBZGQgQ1JVRCBvZiBUYWdzLCBhbmQgbGluayB0aGVzZSB0byBUb2RvcwotIE1pZ3JhdGUgY3VycmVudCBtYW51YWwgdXNhZ2Ugb2YgYHBnYCB0byBhbiBPUk0gbWF5YmUgPwotIEV2ZW50IGVtaXNzaW9uIHRvIFJhYmJpdE1RIHdoZW4gc29tZXRoaW5nIGlzIGNyZWF0ZWQvY2hhbmdlZC9kZWxldGVkCi0gSGFuZGxlIGVycm9ycyBtb3JlIGdyYWNlZnVsbHkKLSBIYW5kbGUgc2h1dGRvd24gc2lnbmFsIGdyYWNlZnVsbHkgKGUuZy4gZG8gbm90IGtpbGwgYWxsIG9uZ29pbmcgcmVxdWVzdHMpCi0gUHJvbWV0aGV1cyBtZXRyaWNzCi0gUHJvcGVyIGxvZ2dpbmcgaW4gSlNPTiB3aXRoIGdvb2QgbWV0YWRhdGEvZXJyb3IgaGFuZGxpbmcKLSBIZWFsdGhjaGVjay9yZWFkaW5lc3Mgcm91dGVzCi0gQXV0aCB3aXRoIEpXVAotIEF1dG9tYXRlZCBjbGVhbnVwICh0byBhdm9pZCB0aGUgZnJlZSBIZXJva3UgREIgdG8gZ28gZnVsbCkKLSBSYXRlIGxpbWl0aW5nIChnbG9iYWwgYW5kIHBlciB1c2VyID8pCi0gSW1wbGVtZW50IGUyZS9pbnRlZ3JhdGlvbiB0ZXN0cyB3aXRoIHN1cGVydGVzdCBvciBzb21ldGhpbmcgc2ltaWxhciA/Ci0gQWRkIHVuaXQgdGVzdHMuLi4gQnV0IHJpZ2h0IG5vdyB0aGVyZSdzIGxpdGVyYWxseSBubyBsb2dpYyB0byB0ZXN0IDopCg== readmeEtag: '"b0e4e597ac4e8587bd04b2a7531f2e2e89b824c4"' readmeLastModified: Sat, 09 Jul 2022 13:04:51 GMT repositoryId: 389616942 description: Todos API written in Typescript, trying to implement the best practices. created: '2021-07-26T11:59:04Z' updated: '2022-12-03T08:30:52Z' language: TypeScript archived: false stars: 4 watchers: 1 forks: 0 owner: GuillaumeDeconinck logo: https://avatars.githubusercontent.com/u/8962903?v=4 repoEtag: '"363e791d4c8e58b888f720e1236417be2db05884bc475400b225f1f40d0bfb52"' repoLastModified: Sat, 03 Dec 2022 08:30:52 GMT foundInMaster: true category: Server Implementations id: dcbaa2bec964bbacbd35706dd3d03fe9 - source: openapi3 tags repository: https://github.com/swaggest/php-swagger2-schema v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIDMuMCBhbmQgU3dhZ2dlciAyLjAgc2NoZW1hIFBIUCBtYXBwaW5ncwoKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvc3dhZ2dlc3QvcGhwLXN3YWdnZXIyLXNjaGVtYS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9zd2FnZ2VzdC9waHAtc3dhZ2dlcjItc2NoZW1hKQpbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvc3dhZ2dlc3QvcGhwLXN3YWdnZXIyLXNjaGVtYS9icmFuY2gvbWFzdGVyL2dyYXBoL2JhZGdlLnN2ZyldKGh0dHBzOi8vY29kZWNvdi5pby9naC9zd2FnZ2VzdC9waHAtc3dhZ2dlcjItc2NoZW1hKQoKQWNjZXNzIGFuZCB2YWxpZGF0ZSBgT3BlbkFQSSAzLjBgIGFuZCBgU3dhZ2dlciAyLjBgIHNjaGVtYXMgZnJvbSBQSFAuCgojIyBJbnN0YWxsYXRpb24KCmBgYApjb21wb3NlciByZXF1aXJlIHN3YWdnZXN0L3N3YWdnZXIyLXNjaGVtYQpgYGAKCiMjIFVzYWdlCgojIyMgT3BlbkFQSSAzLjAKCmBgYHBocAovLyBMb2FkIHNjaGVtYQokanNvbiA9IGpzb25fZGVjb2RlKGZpbGVfZ2V0X2NvbnRlbnRzKF9fRElSX18gLiAnLy4uLy4uLy4uL3NwZWMvcGV0c3RvcmUtb3BlbmFwaTMuanNvbicpKTsKCi8vIEltcG9ydCBhbmQgdmFsaWRhdGUKJHNjaGVtYSA9IE9wZW5BUEkzU2NoZW1hOjppbXBvcnQoJGpzb24pOwoKLy8gQWNjZXNzIGRhdGEgdGhyb3VnaCBQSFAgY2xhc3NlcwokdGhpcy0+YXNzZXJ0U2FtZSgnU3dhZ2dlciBQZXRzdG9yZScsICRzY2hlbWEtPmluZm8tPnRpdGxlKTsKJG9wcyA9ICRzY2hlbWEtPnBhdGhzWycvcGV0cyddLT5nZXRHZXRQdXRQb3N0RGVsZXRlT3B0aW9uc0hlYWRQYXRjaFRyYWNlVmFsdWVzKCk7CiR0aGlzLT5hc3NlcnRTYW1lKCdMaXN0IGFsbCBwZXRzJywgJG9wc1snZ2V0J10tPnN1bW1hcnkpOwoKJHJlc3BvbnNlU2NoZW1hID0gJG9wc1snZ2V0J10tPnJlc3BvbnNlc1syMDBdLT5jb250ZW50WydhcHBsaWNhdGlvbi9qc29uJ10tPnNjaGVtYTsKJHRoaXMtPmFzc2VydFNhbWUoJ2FycmF5JywgJHJlc3BvbnNlU2NoZW1hLT50eXBlKTsKYGBgCgojIyMgU3dhZ2dlciAyLjAKCmBgYHBocAovLyBMb2FkIHNjaGVtYQokanNvbiA9IGpzb25fZGVjb2RlKGZpbGVfZ2V0X2NvbnRlbnRzKF9fRElSX18gLiAnLy4uLy4uL3NwZWMvcGV0c3RvcmUtc3dhZ2dlci5qc29uJykpOwoKLy8gSW1wb3J0IGFuZCB2YWxpZGF0ZQokc2NoZW1hID0gU3dhZ2dlclNjaGVtYTo6aW1wb3J0KCRqc29uKTsKCi8vIEFjY2VzcyBkYXRhIHRocm91Z2ggUEhQIGNsYXNzZXMKJHRoaXMtPmFzc2VydFNhbWUoJ1N3YWdnZXIgUGV0c3RvcmUnLCAkc2NoZW1hLT5pbmZvLT50aXRsZSk7CmBgYAo= readmeEtag: '"8e722603a21649933772da39d0b264692af2891e"' readmeLastModified: Thu, 17 Jun 2021 22:37:53 GMT repositoryId: 116000397 description: OpenAPI 3.0 / Swagger 2.0 schema PHP mappings created: '2018-01-02T09:52:55Z' updated: '2026-01-09T20:28:35Z' language: PHP archived: false stars: 6 watchers: 1 forks: 0 owner: swaggest logo: https://avatars.githubusercontent.com/u/19609628?v=4 repoEtag: '"6576b57713df03ce038f3e6676d7504da16758271122adba7fb743b29071233f"' repoLastModified: Fri, 09 Jan 2026 20:28:35 GMT foundInMaster: true category: Parsers id: 891b1d19c1d5b9398f4b4df84df90b50 - source: openapi3 tags repository: https://github.com/goodforgod/micronaut-management-openapi v3: true repositoryMetadata: base64Readme: >- IyBNaWNyb25hdXQgTWFuYWdlbWVudCBPcGVuQVBJCgpbIVtNaW5pbXVtIHJlcXVpcmVkIEphdmEgdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9KYXZhLTE3JTJCLWJsdWU/bG9nbz1vcGVuamRrKV0oaHR0cHM6Ly9vcGVuamRrLm9yZy9wcm9qZWN0cy9qZGsvMTcvKQpbIVtNYXZlbiBDZW50cmFsXShodHRwczovL21hdmVuLWJhZGdlcy5oZXJva3VhcHAuY29tL21hdmVuLWNlbnRyYWwvaW8uZ29vZGZvcmdvZC9taWNyb25hdXQtbWFuYWdlbWVudC1vcGVuYXBpL2JhZGdlLnN2ZyldKGh0dHBzOi8vbWF2ZW4tYmFkZ2VzLmhlcm9rdWFwcC5jb20vbWF2ZW4tY2VudHJhbC9pby5nb29kZm9yZ29kL21pY3JvbmF1dC1tYW5hZ2VtZW50LW9wZW5hcGkpCiFbSmF2YSBDSV0oaHR0cHM6Ly9naXRodWIuY29tL0dvb2Rmb3JHb2QvbWljcm9uYXV0LW1hbmFnZW1lbnQtb3BlbmFwaS93b3JrZmxvd3MvQ0klMjBNYXN0ZXIvYmFkZ2Uuc3ZnKQpbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1Hb29kZm9yR29kX21pY3JvbmF1dC1tYW5hZ2VtZW50LW9wZW5hcGkmbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9R29vZGZvckdvZF9taWNyb25hdXQtbWFuYWdlbWVudC1vcGVuYXBpKQpbIVtDb3ZlcmFnZV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9R29vZGZvckdvZF9taWNyb25hdXQtbWFuYWdlbWVudC1vcGVuYXBpJm1ldHJpYz1jb3ZlcmFnZSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9R29vZGZvckdvZF9taWNyb25hdXQtbWFuYWdlbWVudC1vcGVuYXBpKQpbIVtNYWludGFpbmFiaWxpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1Hb29kZm9yR29kX21pY3JvbmF1dC1tYW5hZ2VtZW50LW9wZW5hcGkmbWV0cmljPXNxYWxlX3JhdGluZyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9R29vZGZvckdvZF9taWNyb25hdXQtbWFuYWdlbWVudC1vcGVuYXBpKQoKTGlicmFyeSBwcm92aWRlcyBNaWNyb25hdXQgKmNsb3VkLWZyaWVuZGx5KiBPcGVuQVBJL1N3YWdnZXItVUkvUmFwaWRvYyBtYW5hZ2VtZW50IGVuZHBvaW50cy4KCkZlYXR1cmVzOgotIENsb3VkLWZyaWVuZGx5LCAqb3B0aW1pemVkIGZvciBmaWxlIHN0cmVhbWluZyogT3BlbkFQSS9Td2FnZ2VyLVVJL1JhcGlkb2MgZW5kcG9pbnRzCi0gTWVyZ2luZyBtdWx0aXBsZSBPcGVuQVBJIGZpbGVzIGludG8gb25lCi0gT3BlbkFQSSBleHBvc3VyZSBlbmRwb2ludAotIFN3YWdnZXIgVUkgZXhwb3N1cmUgZW5kcG9pbnQKLSAqU3dhZ2dlciBVSSBEYXJrXExpZ2h0IHRoZW1lKgotIFJhcGlkb2MgZXhwb3N1cmUgZW5kcG9pbnQKCiMjIERlcGVuZGVuY3kgOnJvY2tldDoKCioqR3JhZGxlKioKYGBgZ3Jvb3Z5CmltcGxlbWVudGF0aW9uICJpby5nb29kZm9yZ29kOm1pY3JvbmF1dC1tYW5hZ2VtZW50LW9wZW5hcGk6Mi4wLjAiCmBgYAoKKipNYXZlbioqCmBgYHhtbAo8ZGVwZW5kZW5jeT4KICAgIDxncm91cElkPmlvLmdvb2Rmb3Jnb2Q8L2dyb3VwSWQ+CiAgICA8YXJ0aWZhY3RJZD5taWNyb25hdXQtbWFuYWdlbWVudC1vcGVuYXBpPC9hcnRpZmFjdElkPgogICAgPHZlcnNpb24+Mi4wLjA8L3ZlcnNpb24+CjwvZGVwZW5kZW5jeT4KYGBgCgojIyBFeGFtcGxlCgpIZXJlIGlzIFtzaW1wbGUgTWljcm9uYXV0IEhUVFAgYXBwbGljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9Hb29kZm9yR29kL21pY3JvbmF1dC1qYXZhLWh0dHAtdGVtcGxhdGUpCndpdGggY29uZmlndXJlZCBsaWJyYXJ5IGFuZCBPcGVuQVBJLCB5b3UgcGxheSB3aXRoIGl0LgoKIyMgT3BlbkFQSSBHZW5lcmF0aW9uCgpMaWJyYXJ5IG9ubHkgZXhwb3NlcyAqT3BlbkFQSSosIGxpYnJhcnkgKipET0VTTidUKiogZ2VuZXJhdGUgaXQsIHRoaXMgaXMgeW91ciByZXNwb25zaWJpbGl0eSB0byBnZW5lcmF0ZSBPcGVuQVBJIGZpbGVzLgoKVGhlcmUgaXMgTWljcm9uYXV0IE9wZW5BUEkgZ2VuZXJhdG9yLCBbR3JhZGxlIGNvbmZpZ10oaHR0cHM6Ly9naXRodWIuY29tL0dvb2Rmb3JHb2QvbWljcm9uYXV0LWh0dHAtdGVtcGxhdGUvYmxvYi9tYXN0ZXIvYnVpbGQuZ3JhZGxlI0w0MCk6CgpgYGB5YW1sCmRlcGVuZGVuY2llcyB7CiAgICBhbm5vdGF0aW9uUHJvY2Vzc29yKCJpby5taWNyb25hdXQub3BlbmFwaTptaWNyb25hdXQtb3BlbmFwaSIpCiAgICBjb21waWxlT25seSgiaW8uc3dhZ2dlci5jb3JlLnYzOnN3YWdnZXItYW5ub3RhdGlvbnM6Mi4yLjE1IikKfQpgYGAKCk1vcmUgaW5mbyBhYm91dCBNaWNyb25hdXQgT3BlbkFQSSBnZW5lcmF0b3IgW2luIG9mZmljaWFsIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vbWljcm9uYXV0LXByb2plY3RzLmdpdGh1Yi5pby9taWNyb25hdXQtb3BlbmFwaS9sYXRlc3QvZ3VpZGUvaW5kZXguaHRtbCkuCgojIyBFbmRwb2ludHMKCkxpYnJhcnkgYXV0b21hdGljYWxseSAqc2NhbiogZm9yIE9wZW5BUEkgZmlsZXMgaW5zaWRlIEpBUiBpbiAqZGVmYXVsdC1kaXJlY3RvcnkqIGFuZCBleHBvc2UgdGhlbSB2aWEgT3BlbkFQSSBlbmRwb2ludC4KCkVuZHBvaW50czoKLSAqKi9vcGVuYXBpKiogLSBbT3BlbkFQSV0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMS4wKSBlbmRwb2ludC4KLSAqKi9zd2FnZ2VyLXVpKiogLSBbU3dhZ2dlciBVSV0oaHR0cHM6Ly9wZXRzdG9yZS5zd2FnZ2VyLmlvLykgZW5kcG9pbnQuCi0gKiovcmFwaWRvYyoqIC0gW1JhcGlkb2NdKGh0dHBzOi8vbXJpbjkuZ2l0aHViLmlvL1JhcGlEb2MvZXhhbXBsZXMvZXhhbXBsZTIuaHRtbCkgZW5kcG9pbnQuCgpTd2FnZ2VyIFVJIGhhdmUgKipMaWdodFxEYXJrKiogdGhlbWUgc3dpdGNoISEhCgojIyBDb25maWd1cmF0aW9uCgpNb3N0IG9mIHRoZSBzZXR0aW5ncyBhcmUgKmNsb3VkLWZyaWVuZGx5KiBieSBkZWZhdWx0LgoKYGBgeWFtbApvcGVuYXBpOgogIHBhdGg6IC9vcGVuYXBpICAgICAgICAgICAgICAgICAgICAgICAgLy8gUGF0aCBmb3IgT3BlbkFQSSBlbmRwb2ludCAgICAgICAgICAgICAgICAgICAgICAgICAgKGRlZmF1bHQgLSAvb3BlbmFwaSkKICBlbmFibGVkOiB0cnVlICAgICAgICAgICAgICAgICAgICAgICAgIC8vIEVuYWJsZSBPcGVuQVBJIGV4cG9zdXJlICAgICAgICAgICAgICAgICAgICAgICAgICAgIChkZWZhdWx0IC0gdHJ1ZSkKICBtZXJnZTogZmFsc2UgICAgICAgICAgICAgICAgICAgICAgICAgIC8vIEVuYWJsZSBtZXJnaW5nIE9wZW5BUEkgZm91bmQgaW4gZGVmYXVsdC1kaXJlY3RvcnkgIChkZWZhdWx0IC0gZmFsc2UpCiAgZGVmYXVsdC1kaXJlY3Rvcnk6IE1FVEEtSU5GL3N3YWdnZXIgICAvLyBQYXRoIGluc2lkZSBKQVIgd2hlcmUgdG8gc2VhcmNoIE9wZW5BUEkgICAgICAgICAgICAoZGVmYXVsdCAtIE1FVEEtSU5GL3N3YWdnZXIpCiAgZXhjbHVkZTogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBPcGVuQVBJIGZpbGVzIHRvIGV4Y2x1ZGUgZnJvbSBleHBvc3VyZSAgICAgICAgICAgICAocGF0aCBvciBmaWxlbmFtZSkKICAgIC0gb3BlbmFwaS0xLnltbCAgICAgICAgICAgICAgICAgICAgIC8vIFBhdGggb3IgZmlsZW5hbWUgKGRlZmF1bHQtZGlyZWN0b3J5KQogICAgLSBNRVRBLUlORi9zd2FnZ2VyL29wZW5hcGktMi55bWwgICAgLy8gUGF0aCBvciBmaWxlbmFtZSAoZGVmYXVsdC1kaXJlY3RvcnkpCiAgaW5jbHVkZTogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAvLyBJbmNsdWRlIE9OTFkgc3BlY2lmaWVkIE9wZW5BUEkgZmlsZXMgZm9yIGV4cG9zdXJlICAocGF0aCBvbmx5KQogICAgLSBNRVRBLUlORi9zd2FnZ2VyL29wZW5hcGktMy55bWwgICAgLy8gUGF0aCB0byBmaWxlIGluc2lkZSBKQVIKICAgIC0gZXh0ZXJuYWwtc3dhZ2dlci9vcGVuYXBpLTQueW1sICAgIC8vIFBhdGggdG8gZmlsZSBpbnNpZGUgSkFSCiAgCiAgc3dhZ2dlci11aToKICAgIHBhdGg6IC9zd2FnZ2VyLXVpICAgICAgICAgICAgICAgICAgIC8vIFBhdGggZm9yIFN3YWdnZXItVUkgZW5kcG9pbnQgICAgICAgICAgICAgICAgICAgICAgIChkZWZhdWx0IC0gL3N3YWdnZXItdWkpCiAgICBlbmFsYmVkOiBmYWxzZSAgICAgICAgICAgICAgICAgICAgICAvLyBFbmFibGUgU3dhZ2dlci1VSSBleHBvc3VyZSAgICAgICAgICAgICAgICAgICAgICAgICAoZGVmYXVsdCAtIGZhbHNlKQogIAogIHJhcGlkb2M6CiAgICBwYXRoOiAvcmFwaWRvYyAgICAgICAgICAgICAgICAgICAgICAvLyBQYXRoIGZvciBSYXBpZG9jIGVuZHBvaW50ICAgICAgICAgICAgICAgICAgICAgICAgICAoZGVmYXVsdCAtIC9yYXBpZG9jKQogICAgZW5hbGJlZDogZmFsc2UgICAgICAgICAgICAgICAgICAgICAgLy8gRW5hYmxlIFJhcGlkb2MgZXhwb3N1cmUgICAgICAgICAgICAgICAgICAgICAgICAgICAgKGRlZmF1bHQgLSBmYWxzZSkgCmBgYAoKIyMjIE1lcmdlCgotICptZXJnZSogaXMgZGlzYWJsZWQgdGhlbiBhbnkgZmlyc3QgT3BlbkFQSSBmaWxlIHdpbGwgYmUgZXhwb3NlZCAoYWNjb3JkaW5nIHRvICpleGNsdWRlKiBhbmQgKmluY2x1ZGUqIGNvbmZpZ3VyYXRpb24pLgotICptZXJnZSogaXMgZW5hYmxlZCB0aGVuIGFsbCBzdWl0YWJsZSAoYWNjb3JkaW5nIHRvICpleGNsdWRlKiBhbmQgKmluY2x1ZGUqIGNvbmZpZ3VyYXRpb24pIE9wZW5BUEkgZmlsZXMgd2lsbCBiZSBtZXJnZWQgaW50byBvbmUgYW5kIGV4cG9zZWQuCgojIyMgU2VjdXJpdHkKCldoZW4geW91IGhhdmUgc2VjdXJpdHkgZW5hYmxlZCBhbmQgd2FudCB0byBwcm92aWRlICpub24tYXV0aCogYWNjZXNzIGZvciB5b3VyIE9wZW5BUEkvU3dhZ2dlci1VSS9SYXBvZGljIGVuZHBvaW50cyBoZXJlIGlzIGNvbmZpZ3VyYXRpb24gZm9yIHN1Y2ggY2FzZToKCmBgYHlhbWwKbWljcm9uYXV0OgogIHNlY3VyaXR5OgogICAgaW50ZXJjZXB0LXVybC1tYXA6CiAgICAgIC0KICAgICAgICBwYXR0ZXJuOiAvb3BlbmFwaXwvc3dhZ2dlci11aQogICAgICAgIGh0dHAtbWV0aG9kOiBHRVQKICAgICAgICBhY2Nlc3M6CiAgICAgICAgICAtIGlzQW5vbnltb3VzKCkKYGBgCgoKIyMgTWljcm9uYXV0IENvbXBhdGFiaWxpdHkKClN0YXJ0aW5nIGZyb20gdmVyc2lvbiAqMi4wLjAqIGxpYnJhcnkgc2hpcHMgZm9yICpNaWNyb25hdXQgNCogJiBKYXZhIDE3IGlzIHJlcXVpcmVkLgoKU3RhcnRpbmcgZnJvbSB2ZXJzaW9uICoxLjAuMCogbGlicmFyeSBzaGlwcyBmb3IgKk1pY3JvbmF1dCAzKiAmIEphdmEgMS44KyBpcyByZXF1aXJlZC4KCiMjIExpY2Vuc2UKClRoaXMgcHJvamVjdCBsaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UgMi4wIC0gc2VlIHRoZSBbTElDRU5TRV0oTElDRU5TRSkgZmlsZSBmb3IgZGV0YWlscy4K readmeEtag: '"ef10a48b4778e7a3753774d234cfbe9e2e637607"' readmeLastModified: Thu, 10 Aug 2023 05:55:15 GMT repositoryId: 297153893 description: ⚙️ Micronaut cloud-friendly OpenAPI/Swagger-UI/Rapidoc management. created: '2020-09-20T20:02:27Z' updated: '2024-02-20T06:46:27Z' language: Java archived: false stars: 4 watchers: 1 forks: 1 owner: GoodforGod logo: https://avatars.githubusercontent.com/u/9437175?v=4 license: Apache-2.0 repoEtag: '"38b96658b56497790f7b3744c165a19d2b9ab2c5c864849e60c1cb46c30d140a"' repoLastModified: Tue, 20 Feb 2024 06:46:27 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: f9d87f105e5a35d535b3f7c801437b29 - source: openapi3 tags repository: https://github.com/rabestro/pig-latin-rest v3: true id: a28256e662d8b3f9b919787f4b645788 repositoryMetadata: base64Readme: >- WyFbUXVhbGl0eSBHYXRlIFN0YXR1c10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9bHYuaWQuamMlM0FwaWctbGF0aW4tcmVzdCZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9bHYuaWQuamMlM0FwaWctbGF0aW4tcmVzdCkKWyFbQVBJIGRvY3NdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvQVBJLWRvY3MtaW5mb3JtYXRpb25hbCldKGh0dHBzOi8vcGlnbGF0aW4uYXp1cmV3ZWJzaXRlcy5uZXQvc3dhZ2dlci11aS9pbmRleC5odG1sKQpbIVtEZXBsb3llZCBvbiBBenVyZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9EZXBsb3llZCUyMG9uLUF6dXJlLWJsdWUpXShodHRwczovL3BpZ2xhdGluLmF6dXJld2Vic2l0ZXMubmV0KQoKIyBQaWcgTGF0aW4gVHJhbnNsYXRvcgoKVGhpcyBpcyBhIFJFU1QgYXBwbGljYXRpb24gdGhhdCB0cmFuc2xhdGVzIEVuZ2xpc2ggc2VudGVuY2VzIGludG8gUGlnIExhdGluLiBUaGUgcnVsZXMgYW5kIGluc3RydWN0aW9ucyBmb3IgdGhlIFBpZyBMYXRpbiB0cmFuc2xhdGlvbiB3ZXJlIG9idGFpbmVkIGZyb20gdGhlIFtQaWcgTGF0aW4gZXhlcmNpc2Ugb24gdGhlIEV4ZXJjaXNtIEphdmEgVHJhY2tdKGh0dHBzOi8vZXhlcmNpc20ub3JnL3RyYWNrcy9qYXZhL2V4ZXJjaXNlcy9waWctbGF0aW4pLgoKVGhlIGFwcGxpY2F0aW9uIHdhcyBjcmVhdGVkIHRvIGV4cGVyaW1lbnQgd2l0aCB0aGUgQVBJLWZpcnN0IGFwcHJvYWNoOiB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIHdhcyBjcmVhdGVkIGZpcnN0LCB0aGVuIHRoZSBjb2RlIHdhcyBnZW5lcmF0ZWQgZm9yIFNwcmluZywgYW5kIGZpbmFsbHksIGN1c3RvbSBjb2RlIHdhcyBhZGRlZCB0aGF0IGRlcGVuZHMgb24gdGhlIGdlbmVyYXRlZCBjb2RlLgoKIyMgR2V0dGluZyBTdGFydGVkCgpCZWZvcmUgb3BlbmluZyB0aGUgY29kZSBmb3IgZWRpdGluZywgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB0byBnZW5lcmF0ZSB0aGUgbmVjZXNzYXJ5IGNvZGU6CgpgYGBiYXNoCm12biBjbGVhbiBjb21waWxlCmBgYAoKVGhpcyBjb21tYW5kIGNsZWFucyB0aGUgcHJvamVjdCwgY29tcGlsZXMgdGhlIHNvdXJjZSBjb2RlLCBhbmQgZ2VuZXJhdGVzIHRoZSBjb2RlIHRoYXQgdGhlIGN1c3RvbSBjb2RlIGRlcGVuZHMgb24uIEFmdGVyIHJ1bm5pbmcgdGhpcyBjb21tYW5kLCB5b3Ugc2hvdWxkIGJlIGFibGUgdG8gb3BlbiB0aGUgY29kZSB3aXRob3V0IGdldHRpbmcgYW55IGVycm9yIG1lc3NhZ2VzLgoKIyMgUnVubmluZyB0aGUgQXBwbGljYXRpb24gTG9jYWxseQoKVG8gcnVuIHRoZSBhcHBsaWNhdGlvbiBsb2NhbGx5LCB1c2UgdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgYmFzaAptdm4gY2xlYW4gY29tcGlsZSBzcHJpbmctYm9vdDpydW4KYGBgClRoaXMgd2lsbCBzdGFydCB0aGUgYXBwbGljYXRpb24gb24gbG9jYWxob3N0OjgwODAuIFlvdSBjYW4gYWNjZXNzIHRoZSBTd2FnZ2VyIFVJIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpLyBhbmQgdGhlIGFwcGxpY2F0aW9uIGVuZHBvaW50IGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9waWctbGF0aW4uCgojIyBEZXBsb3llZCBBcHBsaWNhdGlvbgoKVGhlIGFwcGxpY2F0aW9uIGlzIGRlcGxveWVkIG9uIEF6dXJlIGF0IGh0dHBzOi8vcGlnbGF0aW4uYXp1cmV3ZWJzaXRlcy5uZXQuCgpZb3UgY2FuIHVzZSB0aGUgc2ltcGxlIFt3ZWIgaW50ZXJmYWNlXShodHRwczovL3BpZ2xhdGluLmF6dXJld2Vic2l0ZXMubmV0KSBhbmQgdGhlIFtTd2FnZ2VyIFVJXShodHRwczovL3BpZ2xhdGluLmF6dXJld2Vic2l0ZXMubmV0L3N3YWdnZXItdWkvaW5kZXguaHRtbCkuCgpQbGVhc2Ugbm90ZSB0aGF0IHRoZSBhcHBsaWNhdGlvbiBpcyBkZXBsb3llZCBvbiB0aGUgZnJlZS10aWVyIEYxIHBsYW4uIFRoaXMgbWVhbnMgdGhhdCBpZiB0aGUgYXBwbGljYXRpb24gaXMgaW4gc2xlZXAgbW9kZSwgaXQgbWF5IHRha2UgYWJvdXQgYSBtaW51dGUgZm9yIHRoZSBmaXJzdCBBUEkgY2FsbCB0byB3YWtlIGl0IHVwLiBTdWJzZXF1ZW50IEFQSSBjYWxscyBzaG91bGQgYmUgZmFzdGVyLgoKIyMgQVBJIFRlc3RpbmcKClRoZSBhcHBsaWNhdGlvbiBpbmNsdWRlcyBBUEkgdGVzdHMgd3JpdHRlbiB3aXRoIGZyYW1ld29ya3M6Ci0gW0thcmF0ZV0oaHR0cHM6Ly93d3cua2FyYXRlbGFicy5pby8pCi0gW0JydW5vXShodHRwczovL3d3dy51c2VicnVuby5jb20vKQotIFtIdXJsXShodHRwczovL2h1cmwuZGV2LykKLSBbaHR0cFlhY10oaHR0cHM6Ly9odHRweWFjLmdpdGh1Yi5pby8pCi0gW0pldEJyYWlucyBIVFRQIENsaWVudF0oaHR0cHM6Ly93d3cuamV0YnJhaW5zLmNvbS9oZWxwL2lkZWEvaHR0cC1jbGllbnQtaW4tcHJvZHVjdC1jb2RlLWVkaXRvci5odG1sKQotIFtBcGFjaGUgak1ldGVyXShodHRwczovL2ptZXRlci5hcGFjaGUub3JnLykKCiMjIyBCcnVubwoKVG8gcnVuIHRoZSBCcnVubyBBUEkgdGVzdHMsIHVzZSB0aGUgZm9sbG93aW5nIGNvbW1hbmRzOgoKYGBgYmFzaApjZCBicnVuby10ZXN0Cm5wbSBydW4gdGVzdApgYGAKClN1Y2Nlc3NmdWwgdGVzdCByZXBvcnQ6Cmh0dHBzOi8vZ2l0aHViLmNvbS9yYWJlc3Ryby9waWctbGF0aW4tcmVzdC9ydW5zLzIwMzU3NjY5ODEyCgpGYWlsZWQgdGVzdCByZXBvcnQ6Cmh0dHBzOi8vZ2l0aHViLmNvbS9yYWJlc3Ryby9waWctbGF0aW4tcmVzdC9ydW5zLzIwNzA2MTY3MTQ0I3IwczEKCj4gWyFOT1RFXQo+IE5vdGUgLSBCcnVubydzIHNjcmlwdCBzdXBwb3J0IG1pZ2h0IGFsc28gYmUgdXNlZCBmb3IgbGlnaHQgdGVzdCBhdXRvbWF0aW9uIGJhc2VkIG9uIHByb2plY3Qgc3BlY2lmaWNzLgoKIyMjIEpldEJyYWlucyBIVFRQIENsaWVudAoKWyFbSFRUUCBDbGllbnQgLSBTZWNyZXQgV2VhcG9uIGZvciBXZWIgU2VydmljZSBUZXN0aW5nXShodHRwczovL2ltZy55b3V0dWJlLmNvbS92aS9WTVVhT1o2a3ZKMC9kZWZhdWx0LmpwZyldKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9Vk1VYU9aNmt2SjApCgpbIVtIVFRQIENsaWVudCBDTEldKGh0dHBzOi8vaW1nLnlvdXR1YmUuY29tL3ZpL213aUhBdWtiV2pNL2RlZmF1bHQuanBnKV0oaHR0cHM6Ly93d3cueW91dHViZS5jb20vd2F0Y2g/dj1td2lIQXVrYldqTSkKCgpTdWNjZXNzZnVsIHRlc3QgcmVwb3J0OgpodHRwczovL2dpdGh1Yi5jb20vcmFiZXN0cm8vcGlnLWxhdGluLXJlc3QvcnVucy8yMDI0MDcxODM3MwoKRmFpbGVkIHRlc3QgcmVwb3J0OgpodHRwczovL2dpdGh1Yi5jb20vcmFiZXN0cm8vcGlnLWxhdGluLXJlc3QvcnVucy8yMDI0MTcyNDk4NwoKIyMjIGh0dHBZYWMgLSB0aGUgRnJlZSBhbHRlcm5hdGl2ZSBvZiBKZXRCcmFpbnMgSFRUUCBDbGllbnQKCmh0dHBzOi8vaHR0cHlhYy5naXRodWIuaW8vCgojIyMgS2FyYXRlCgpUbyBydW4gdGhlIEthcmF0ZSBBUEkgdGVzdHMsIHVzZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBiYXNoCm12biBjbGVhbiB0ZXN0IC1EdGVzdD1UZXN0UnVubmVyCmBgYAoKIyMjIFBlcmZvcm1hbmNlIFRlc3RpbmcgYnkgQXBhY2hlIGpNZXRlcgoKTG9hZCBUZXN0IFtEZXNjcmlwdGl2ZSBTdW1tYXJ5L0NvbmNsdXNpb25zXShodHRwczovL2EuYmxhemVtZXRlci5jb20vYXBwL2V4ZWN1dGl2ZS1zdW1tYXJ5L2luZGV4Lmh0bWw/bWFzdGVyX2lkPTcxNzU4MTk4JnNlbGVjdGVkVGltZU1lYXN1cmU9bWlsbGlzZWNvbmRzIy8pIGFuZCBbRnVsbCBSZXBvcnRdKGh0dHBzOi8vYS5ibGF6ZW1ldGVyLmNvbS9hcHAvP3B1YmxpYy10b2tlbj1OOEVBeWdSYWN6Z1ZyRzVkVmZ6YWlXMDhLRVR0RUNYdnBoM1g2QkRrWFY2Q29JaTJwTSMvYWNjb3VudHMvMTg4Njg0MC93b3Jrc3BhY2VzLzE5NTgxODYvcHJvamVjdHMvMjMxMTQzMC9tYXN0ZXJzLzcxNzU4MTk4L3N1bW1hcnkpCg== readmeEtag: '"4edb803c82dedf62bdbf89bf33c2706405e5b699"' readmeLastModified: Tue, 30 Jul 2024 05:49:45 GMT repositoryId: 735633231 description: A project for experimenting with OpenAPI specification and API testing. created: '2023-12-25T16:10:39Z' updated: '2024-07-30T05:49:49Z' language: Groovy archived: false stars: 4 watchers: 2 forks: 1 owner: rabestro logo: https://avatars.githubusercontent.com/u/52966251?v=4 license: MIT repoEtag: '"4dc23cc8277243edc292384dce82e24d65aa1aaa435b1bde2d1e8c31404165c1"' repoLastModified: Tue, 30 Jul 2024 05:49:49 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/go-lite/lite v3: true id: 573819accf54a9402aa0d27427ba0907 repositoryMetadata: base64Readme: >- PHAgc3R5bGU9InRleHQtYWxpZ246IGNlbnRlcjsiPgogIDxpbWcgc3JjPSIuL2xvZ28vbGl0ZS5wbmciIGhlaWdodD0iMjAwIiBhbHQ9IkxpdGUgTG9nbyIgLz4KPC9wPgoKWyFbR29dKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1saXRlL2xpdGUvYWN0aW9ucy93b3JrZmxvd3MvZ28ueW1sL2JhZGdlLnN2Zz9icmFuY2g9bWFpbildKGh0dHBzOi8vZ2l0aHViLmNvbS9nby1saXRlL2xpdGUvYWN0aW9ucy93b3JrZmxvd3MvZ28ueW1sKQpbIVtHbyBSZWZlcmVuY2VdKGh0dHBzOi8vcGtnLmdvLmRldi9iYWRnZS9naXRodWIuY29tL2dvLWxpdGUvbGl0ZS5zdmcpXShodHRwczovL3BrZy5nby5kZXYvZ2l0aHViLmNvbS9nby1saXRlL2xpdGUpClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS9nby1saXRlL2xpdGUpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20vZ28tbGl0ZS9saXRlKQpbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvZ28tbGl0ZS9saXRlL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj01T0ZYVFFLSEVFKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2dvLWxpdGUvbGl0ZSkKCiMgTGl0ZTogQSBUeXBlZCBXcmFwcGVyIGZvciBHb0ZpYmVyCiMjIE92ZXJ2aWV3CgpUaGUgYGxpdGVgIHBhY2thZ2UgcHJvdmlkZXMgZnVuY3Rpb25hbGl0aWVzIGZvciBhdXRvbWF0aWNhbGx5IGdlbmVyYXRpbmcgT3BlbkFQSSBkb2N1bWVudGF0aW9uIGZvciBIVFRQIG9wZXJhdGlvbnMgYmFzZWQgCm9uIHRoZSBzdHJ1Y3QgZGVmaW5pdGlvbnMgYW5kIHRoZWlyIHRhZ3Mgd2l0aGluIGEgR28gYXBwbGljYXRpb24uIFRoaXMgZG9jdW1lbnQgZXhwbGFpbnMgaG93IHRvIHVzZSB0aGUgYGxpdGVgIHBhY2thZ2UsIApzcGVjaWZpY2FsbHkgZm9jdXNpbmcgb24gdGhlIHVzYWdlIG9mIHRoZSBgbGl0ZWAgdGFnLgoKIyMgSW5zdGFsbGF0aW9uCgpUbyB1c2UgdGhlIGBsaXRlYCBwYWNrYWdlLCB5b3UgbmVlZCB0byBpbnN0YWxsIGl0IGZpcnN0LiBBc3N1bWluZyB5b3UgaGF2ZSBHbyBpbnN0YWxsZWQsIHlvdSBjYW4gYWRkIGl0IHRvIHlvdXIgcHJvamVjdCB3aXRoOgoKYGBgYmFzaApnbyBnZXQgZ2l0aHViLmNvbS9nby1saXRlL2xpdGUKYGBgCgojIyBVc2FnZQojIyMgU2ltcGxlIEV4YW1wbGUKSGVyZSBpcyBhIHNpbXBsZSBleGFtcGxlIG9mIGhvdyB0byB1c2UgTGl0ZToKCmBgYGdvCnBhY2thZ2UgbWFpbgoKaW1wb3J0ICgKCSJnaXRodWIuY29tL2dvLWxpdGUvbGl0ZSIKCSJsb2ciCikKCnR5cGUgUmVzcG9uc2Ugc3RydWN0IHsKCU1lc3NhZ2Ugc3RyaW5nIGBqc29uOiJtZXNzYWdlImAKfQoKZnVuYyBtYWluKCkgewoJYXBwIDo9IGxpdGUuTmV3KCkKCglsaXRlLkdldChhcHAsICIvIiwgZnVuYyhjICpsaXRlLkNvbnRleHROb1JlcXVlc3QpIChSZXNwb25zZSwgZXJyb3IpIHsKCQlyZXR1cm4gUmVzcG9uc2V7TWVzc2FnZTogIkhlbGxvLCB3b3JsZCEifSwgbmlsCgl9KQoKCWxvZy5GYXRhbChhcHAuTGlzdGVuKCI6MzAwMCIpKQp9CmBgYApUaGUgc3dhZ2dlciBzcGVjcyBpcyBhdmFpbGFibGUgYXQgYGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMC9zd2FnZ2VyL2luZGV4Lmh0bWxgIGlmIHBvcnQgYDMwMDBgIGlzIHVzZWQuCgojIyMgT3RoZXIgRXhhbXBsZQpgYGBnbwpwYWNrYWdlIG1haW4KCmltcG9ydCAoCgkiaW8iCgkibG9nIgoJIm1pbWUvbXVsdGlwYXJ0IgoJIm9zIgoKCSJnaXRodWIuY29tL2dvLWxpdGUvbGl0ZSIKCSJnaXRodWIuY29tL2dvLWxpdGUvbGl0ZS9lcnJvcnMiCgkiZ2l0aHViLmNvbS9nby1saXRlL2xpdGUvbWltZSIKCSJnaXRodWIuY29tL2dvZmliZXIvZmliZXIvdjIvbWlkZGxld2FyZS9sb2dnZXIiCgkiZ2l0aHViLmNvbS9nb2ZpYmVyL2ZpYmVyL3YyL21pZGRsZXdhcmUvcmVjb3ZlciIKKQoKdHlwZSBJbWFnZVJlc3BvbnNlID0gW11ieXRlCgp0eXBlIEltYWdlUGF5bG9hZCBzdHJ1Y3QgewoJQm9keSBJbWFnZSBgbGl0ZToicmVxPWJvZHksbXVsdGlwYXJ0L2Zvcm0tZGF0YSJgCn0KCnR5cGUgSW1hZ2Ugc3RydWN0IHsKCUluZm8gIGluZm8gICAgICAgICAgICAgICAgICBgZm9ybToiaW5mbyJgCglJbWFnZSAqbXVsdGlwYXJ0LkZpbGVIZWFkZXIgYGZvcm06ImltYWdlImAKfQoKdHlwZSBpbmZvIHN0cnVjdCB7CglGaWxlTmFtZSBzdHJpbmcgYGZvcm06ImZpbGVuYW1lImAKfQoKZnVuYyBtYWluKCkgewoJYXBwIDo9IGxpdGUuTmV3KCkKCglsaXRlLlVzZShhcHAsIGxvZ2dlci5OZXcoKSkKCWxpdGUuVXNlKGFwcCwgcmVjb3Zlci5OZXcoKSkKCglsaXRlLlBvc3QoYXBwLCAiL3YxL2ltYWdlL2FuYWx5c2UiLCBmdW5jKGMgKmxpdGUuQ29udGV4dFdpdGhSZXF1ZXN0W0ltYWdlUGF5bG9hZF0pIChJbWFnZVJlc3BvbnNlLCBlcnJvcikgewoJCXJlcSwgZXJyIDo9IGMuUmVxdWVzdHMoKQoJCWlmIGVyciAhPSBuaWwgewoJCQlyZXR1cm4gSW1hZ2VSZXNwb25zZXt9LCBlcnJvcnMuTmV3QmFkUmVxdWVzdEVycm9yKGVyci5FcnJvcigpKQoJCX0KCgkJaW1hZ2UgOj0gcmVxLkJvZHkuSW1hZ2UKCgkJaWYgZXJyID0gYy5TYXZlRmlsZShpbWFnZSwgIi4vZXhhbXBsZXMvZmlsZS91cGxvYWRzLyIraW1hZ2UuRmlsZW5hbWUpOyBlcnIgIT0gbmlsIHsKCQkJcmV0dXJuIEltYWdlUmVzcG9uc2V7fSwgZXJyCgkJfQoKCQkvLyBnZXQgdGhlIGZpbGUKCQlmLCBlcnIgOj0gb3MuT3BlbigiLi9leGFtcGxlcy9maWxlL3VwbG9hZHMvIiArIGltYWdlLkZpbGVuYW1lKQoJCWlmIGVyciAhPSBuaWwgewoJCQlyZXR1cm4gSW1hZ2VSZXNwb25zZXt9LCBlcnIKCQl9CgoJCS8vIER1bW15IGRhdGEgZm9yIHRoZSByZXNwb25zZQoJCXJlc3BvbnNlLCBlcnIgOj0gaW8uUmVhZEFsbChmKQoJCWlmIGVyciAhPSBuaWwgewoJCQlsb2cuRmF0YWxmKCJmYWlsZWQgcmVhZGluZyBmaWxlOiAlcyIsIGVycikKCQl9CgoJCWMuU2V0Q29udGVudFR5cGUobWltZS5JbWFnZVBuZykKCgkJcmV0dXJuIHJlc3BvbnNlLCBuaWwKCX0pLlNldFJlc3BvbnNlQ29udGVudFR5cGUoImltYWdlL3BuZyIpCglsaXRlLlBvc3QoYXBwLCAiL3YxL3BkZiIsIGZ1bmMoYyAqbGl0ZS5Db250ZXh0V2l0aFJlcXVlc3RbW11ieXRlXSkgKGFueSwgZXJyb3IpIHsKCQlyZXEsIGVyciA6PSBjLlJlcXVlc3RzKCkKCQlpZiBlcnIgIT0gbmlsIHsKCQkJcmV0dXJuIG5pbCwgZXJyb3JzLk5ld0JhZFJlcXVlc3RFcnJvcihlcnIuRXJyb3IoKSkKCQl9CgoJCWxvZy5QcmludGxuKHN0cmluZyhyZXEpKQoKCQlyZXR1cm4gbmlsLCBuaWwKCX0pCgoJYXBwLkFkZFNlcnZlcigiaHR0cDovL2xvY2FsaG9zdDo5MDAwIiwgImV4YW1wbGUgc2VydmVyIikKCglpZiBlcnIgOj0gYXBwLlJ1bigpOyBlcnIgIT0gbmlsIHsKCQlyZXR1cm4KCX0KfQpgYGAKCiMjIyBTdXBwb3J0ZWQgVGFncwoKVGhlIGBsaXRlYCBwYWNrYWdlIHN1cHBvcnRzIHRoZSBmb2xsb3dpbmcgdGFncyB3aXRoaW4gc3RydWN0IGRlZmluaXRpb25zIHRvIG1hcCBmaWVsZHMgdG8gZGlmZmVyZW50IHBhcnRzIG9mIGFuIEhUVFAgcmVxdWVzdCBvciByZXNwb25zZToKCnwgVGFnICAgICB8IFNldERlc2NyaXB0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IEV4YW1wbGUgICAgICAgICAgICAgICAgICAgIHwKfC0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8IGBwYXJhbXNgfCBNYXBzIHRvIGEgVVJMIHBhdGggcGFyYW1ldGVyICAgICAgICAgICAgICAgfCBgbGl0ZToicGFyYW1zPWlkImAgICAgICAgICAgIHwKfCBgcXVlcnlgIHwgTWFwcyB0byBhIFVSTCBxdWVyeSBwYXJhbWV0ZXIgICAgICAgICAgICAgIHwgYGxpdGU6InF1ZXJ5PW5hbWUiYCAgICAgICAgfAp8IGBoZWFkZXJgfCBNYXBzIHRvIGFuIEhUVFAgaGVhZGVyICAgICAgICAgICAgICAgICAgICAgfCBgbGl0ZToiaGVhZGVyPUF1dGgiYCAgICAgICB8CnwgYGNvb2tpZWB8IE1hcHMgdG8gYW4gSFRUUCBjb29raWUgICAgICAgICAgICAgICAgICAgICB8IGBsaXRlOiJjb29raWU9c2Vzc2lvbl9pZCJgIHwKfCBgcmVxYCAgIHwgTWFwcyB0byB0aGUgcmVxdWVzdCBib2R5ICAgICAgICAgICAgICAgICAgIHwgYGxpdGU6InJlcT1ib2R5ImAgICAgICAgICAgfAp8IGBlbnVtc2AgfCBNYXBzIHRvIGEgc3RyaW5nIGVudW1zICAgICAgICAgICAgICAgICAgICAgfCBgZW51bXM6Im1hbGUsZmVtYWxlImAgfCAgICAKCgojIyBDb250cmlidXRpbmcKQ29udHJpYnV0aW9ucyBhcmUgd2VsY29tZSEgUGxlYXNlIGZlZWwgZnJlZSB0byBzdWJtaXQgYSBwdWxsIHJlcXVlc3Qgb3Igb3BlbiBhbiBpc3N1ZS4KCiMjIFNldExpY2Vuc2UKTGl0ZSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIFNldExpY2Vuc2UuIFNlZSBbTElDRU5TRV0oTElDRU5TRSkgZm9yIG1vcmUgaW5mb3JtYXRpb24uCltdOiAjIChFTkQpCg== readmeEtag: '"dea69b1225e1d35572537bb0bd1606d5e2f775db"' readmeLastModified: Mon, 05 Aug 2024 18:05:29 GMT repositoryId: 808226704 description: Lite - fiber wrapper OpenAPI 3 spec auto generate created: '2024-05-30T16:25:58Z' updated: '2025-10-07T08:09:48Z' language: Go archived: false stars: 4 watchers: 1 forks: 0 owner: go-lite logo: https://avatars.githubusercontent.com/u/173594949?v=4 license: MIT repoEtag: '"0f39bffc5e8e8b62806a387554d16dcab5b9bfb55fca31403a518d35f7705590"' repoLastModified: Tue, 07 Oct 2025 08:09:48 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/keisuke6065/openapi-combine v3: true repositoryMetadata: base64Readme: >- b3BlbmFwaS1jb21iaW5lCj09PT09PT09PT09PT09PQoKCiFbdGVzdCBDSV0oaHR0cHM6Ly9naXRodWIuY29tL2tlaXN1a2U2MDY1L29wZW5hcGktY29tYmluZS93b3JrZmxvd3MvdGVzdCUyMENJL2JhZGdlLnN2Zz9icmFuY2g9bWFzdGVyKQpbIVtvY2xpZl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jbGktb2NsaWYtYnJpZ2h0Z3JlZW4uc3ZnKV0oaHR0cHM6Ly9vY2xpZi5pbykKWyFbVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vcGVuYXBpLWNvbWJpbmUuc3ZnKV0oaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZS9vcGVuYXBpLWNvbWJpbmUpClshW0Rvd25sb2Fkcy93ZWVrXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9kdy9vcGVuYXBpLWNvbWJpbmUuc3ZnKV0oaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZS9vcGVuYXBpLWNvbWJpbmUpClshW0xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2wvb3BlbmFwaS1jb21iaW5lLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9rZWlzdWtlNjA2NS9vcGVuYXBpLWNvbWJpbmUvYmxvYi9tYXN0ZXIvcGFja2FnZS5qc29uKQpbIVtEb2NrZXIgSHViIHBhY2thZ2VdKGh0dHBzOi8vZG9ja2VyaS5jby9pbWFnZS9rZWlzdWtlNjA2NS9vcGVuYXBpLWNvbWJpbmUpXShodHRwczovL2h1Yi5kb2NrZXIuY29tL3Iva2Vpc3VrZTYwNjUvb3BlbmFwaS1jb21iaW5lKQoKPCEtLSB0b2MgLS0+CiogW0ZlYXR1cmVzXSgjZmVhdHVyZXMpCiogW1VzYWdlXSgjdXNhZ2UpCiogW0NvbW1hbmRzXSgjY29tbWFuZHMpCjwhLS0gdG9jc3RvcCAtLT4KIyBGZWF0dXJlcwpTdXBwb3J0cyBgJHJlZmAgU3ludGF4IGFuZCBgJHJlZnNgIEV4dGVuZGVkIHN5bnRheCAgCmAkcmVmYCBzdXBwb3J0cyBiYXNlZCBvbiBbcmZjMzk4Nl0oaHR0cHM6Ly90b29scy5pZXRmLm9yZy9odG1sL3JmYzM5ODYpICAgIApUaGUgZXh0ZW5kZWQgc3ludGF4IG9mIGAkcmVmc2AgaXMgYSBzeW50YXggdGhhdCBjYW4gaGF2ZSBgJHJlZmAgYXMgYW4gYXJyYXkgIAoKIyMgRXh0ZW5kZWQgc3ludGF4IHVzYWdlIGV4YW1wbGVzCkRlZmluZSBtdWx0aXBsZSBgJHJlZmAgcmVmZXJlbmNlcyB1c2luZyBgJHJlZnNgIGFzIHNob3duIGJlbG93CihDdXJyZW50bHkgb25seSBMb2NhbCBSZWZlcmVuY2Ugc3VwcG9ydCkKCmBgYHlhbWwKb3BlbmFwaTogIjMuMC4wIgppbmZvOgogIHZlcnNpb246IDEuMC4wCiAgdGl0bGU6IGV4YW1wbGVzCnNlcnZlcnM6CiAgLSB1cmw6IGh0dHA6Ly9wZXRzdG9yZS5zd2FnZ2VyLmlvL3YxCnBhdGhzOgogIC9wZXRzL3twZXRJZH06CiAgICAkcmVmOiAuL3BhdGhzL3BldHMvaWQvaW5kZXgueWFtbApjb21wb25lbnRzOgogIHNjaGVtYXM6CiAgICAkcmVmczoKICAgICAgLSAuL2NvbXBvbmVudHMvc2NoZW1hcy9wZXQueWFtbAogICAgICAtIC4vY29tcG9uZW50cy9zY2hlbWFzL3BldHMueWFtbAogICAgRXJyb3I6CiAgICAgICRyZWY6IC4vY29tcG9uZW50cy9zY2hlbWFzL2Vycm9yLnlhbWwKYGBgCnNlZSBjb252ZXJ0IGV4YW1wbGUgW2V4YW1wbGUvb3BlbmFwaS55YW1sXShleGFtcGxlL29wZW5hcGkueWFtbCkgLT4gW2V4YW1wbGUvb3BlbmFwaS55YW1sXShleGFtcGxlL3Jlc3VsdC55YW1sKQoKCiMgVXNhZ2UKPCEtLSB1c2FnZSAtLT4KYGBgc2gtc2Vzc2lvbgokIG5wbSBpbnN0YWxsIC1nIG9wZW5hcGktY29tYmluZQokIG9wZW5hcGktY29tYmluZSBDT01NQU5ECnJ1bm5pbmcgY29tbWFuZC4uLgokIG9wZW5hcGktY29tYmluZSAoLXZ8LS12ZXJzaW9ufHZlcnNpb24pCm9wZW5hcGktY29tYmluZS8wLjMuMSBsaW51eC14NjQgbm9kZS12MTYuMTUuMAokIG9wZW5hcGktY29tYmluZSAtLWhlbHAgW0NPTU1BTkRdClVTQUdFCiAgJCBvcGVuYXBpLWNvbWJpbmUgQ09NTUFORAouLi4KYGBgCjwhLS0gdXNhZ2VzdG9wIC0tPgojIENvbW1hbmRzCjwhLS0gY29tbWFuZHMgLS0+CiogW2BvcGVuYXBpLWNvbWJpbmUgaGVscCBbQ09NTUFORF1gXSgjb3BlbmFwaS1jb21iaW5lLWhlbHAtY29tbWFuZCkKKiBbYG9wZW5hcGktY29tYmluZSBtZXJnZWBdKCNvcGVuYXBpLWNvbWJpbmUtbWVyZ2UpCgojIyBgb3BlbmFwaS1jb21iaW5lIGhlbHAgW0NPTU1BTkRdYAoKRGlzcGxheSBoZWxwIGZvciBvcGVuYXBpLWNvbWJpbmUuCgpgYGAKVVNBR0UKICAkIG9wZW5hcGktY29tYmluZSBoZWxwIFtDT01NQU5EXQoKQVJHVU1FTlRTCiAgQ09NTUFORCAgQ29tbWFuZCB0byBzaG93IGhlbHAgZm9yLgoKT1BUSU9OUwogIC1uLCAtLW5lc3RlZC1jb21tYW5kcyAgSW5jbHVkZSBhbGwgbmVzdGVkIGNvbW1hbmRzIGluIHRoZSBvdXRwdXQuCmBgYAoKX1NlZSBjb2RlOiBbQG9jbGlmL3BsdWdpbi1oZWxwXShodHRwczovL2dpdGh1Yi5jb20vb2NsaWYvcGx1Z2luLWhlbHAvYmxvYi92NS4xLjEyL3NyYy9jb21tYW5kcy9oZWxwLnRzKV8KCiMjIGBvcGVuYXBpLWNvbWJpbmUgbWVyZ2VgCgpkZXNjcmliZSB0aGUgY29tbWFuZCBoZXJlCgpgYGAKVVNBR0UKICAkIG9wZW5hcGktY29tYmluZSBtZXJnZQoKT1BUSU9OUwogIC1pLCAtLWlucHV0PWlucHV0ICAgICAocmVxdWlyZWQpIGlucHV0IHRhcmdldCB5YW1sIGZpbGUKICAtbywgLS1vdXRwdXQ9b3V0cHV0ICAgW2RlZmF1bHQ6IC4vb3V0cHV0L29wZW5hcGkueWFtbF0gb3V0cHV0IHRhcmdldCB5YW1sIGZpbGUKICAtdCwgLS10eXBlPXlhbWx8anNvbiAgW2RlZmF1bHQ6IHlhbWxdIG91dHB1dCBmb3JtYXQgeWFtbCBvciBqc29uCgpFWEFNUExFCiAgJCBvcGVuYXBpLWNvbWJpbmUgbWVyZ2UgLWkgLi9leGFtcGxlL29wZW5hcGkueWFtbCAtbyAuL2J1aWxkL29wZW5hcGkueWFtbApgYGAKPCEtLSBjb21tYW5kc3N0b3AgLS0+Cg== readmeEtag: '"25fa7a46e17d3466f564b6c137a1207301c0c096"' readmeLastModified: Thu, 09 Jun 2022 03:09:18 GMT repositoryId: 290097194 description: openapi combine tools Extended syntax `$refs` created: '2020-08-25T02:57:22Z' updated: '2022-12-15T07:21:32Z' language: TypeScript archived: false stars: 4 watchers: 2 forks: 0 owner: keisuke6065 logo: https://avatars.githubusercontent.com/u/6197330?v=4 license: MIT repoEtag: '"ff227a7bbf862a2db31167906635e1572604f480ba2d5168aeee579143378f81"' repoLastModified: Thu, 15 Dec 2022 07:21:32 GMT foundInMaster: true category: Parsers id: 54556e5124d461a37adbe50e3129138b - source: openapi3 tags repository: https://github.com/eliezio/simple-pki v3: true repositoryMetadata: base64Readme: >- aWZuZGVmOjplbnYtZ2l0aHViW10KOmRvY3R5cGU6IGJvb2sKOmljb25zOiBmb250Cjpzb3VyY2UtaGlnaGxpZ2h0ZXI6IGhpZ2hsaWdodGpzCjp0b2M6IGxlZnQKOnRvY2xldmVsczogMwo6c2VjdGxpbmtzOgoKKysrKwo8ZGl2IHN0eWxlPSJwb3NpdGlvbjpyZWxhdGl2ZTttaW4td2lkdGg6OTYwcHgiPgo8YSBocmVmPSIgaHR0cHM6Ly9naXRodWIuY29tL2VsaWV6aW8vc2ltcGxlLXBraSI+PGltZyB3aWR0aD0iMTQ5IiBoZWlnaHQ9IjE0OSIgc3JjPSJodHRwczovL2dpdGh1Yi5ibG9nL3dwLWNvbnRlbnQvdXBsb2Fkcy8yMDA4LzEyL2ZvcmttZV9yaWdodF9kYXJrYmx1ZV8xMjE2MjEucG5nP3Jlc2l6ZT0xNDklMkMxNDkiIGNsYXNzPSJhdHRhY2htZW50LWZ1bGwgc2l6ZS1mdWxsIiBhbHQ9IkZvcmsgbWUgb24gR2l0SHViIiBkYXRhLXJlY2FsYy1kaW1zPSIxIiBzdHlsZT0iIHBvc2l0aW9uOiBhYnNvbHV0ZTtyaWdodDowO3RvcDowIj48L2E+CjwvZGl2PgorKysrCmVuZGlmOjpbXQoKPSBTaW1wbGUgUEtJCkVsacOpemlvIE9saXZlaXJhIDxlbGllemlvQHBtLm1lPgp2e3Byb2plY3QtdmVyc2lvbn0KCmltYWdlOmh0dHBzOi8vdHJhdmlzLWNpLm9yZy9lbGllemlvL3NpbXBsZS1wa2kuc3ZnP2JyYW5jaD1tYXN0ZXJbQnVpbGQsIGxpbms9Imh0dHBzOi8vdHJhdmlzLWNpLm9yZy9lbGllemlvL3NpbXBsZS1wa2kiXQppbWFnZTpodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1lbGllemlvX3NpbXBsZS1wa2kmbWV0cmljPWFsZXJ0X3N0YXR1c1tTb25hclF1YmUgUXVhbGl0eSBHYXRlLCBsaW5rPSJodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPWVsaWV6aW9fc2ltcGxlLXBraSJdCmltYWdlOmh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PWVsaWV6aW9fc2ltcGxlLXBraSZtZXRyaWM9Y292ZXJhZ2VbQ292ZXJhZ2UsIGxpbms9Imh0dHBzOi8vc29uYXJjbG91ZC5pby9wcm9qZWN0L2FjdGl2aXR5P2N1c3RvbV9tZXRyaWNzPWNvdmVyYWdlJmdyYXBoPWN1c3RvbSZpZD1lbGllemlvX3NpbXBsZS1wa2kiXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL2VuZHBvaW50LnN2Zz91cmw9aHR0cHM6Ly9lbGllemlvLmdpdGh1Yi5pby9zaW1wbGUtcGtpL3BpdGVzdC9zaGllbGRzLmpzb25bUGl0ZXN0LCBsaW5rPSJodHRwczovL2VsaWV6aW8uZ2l0aHViLmlvL3NpbXBsZS1wa2kvcGl0ZXN0LyJdCgo9PSBPdmVydmlldwoKVGhpcyBhcHBsaWNhdGlvbiBwcm92aWRlcyB0aGUgYmFzaWMgc2VydmljZXMgdHlwaWNhbGx5IGZvdW5kIG9uIGFueSBQS0k6CgotIEdlbmVyYXRlcyBYLjUwOSBjZXJ0aWZpY2F0ZXM7Ci0gQWxsb3cgcHJlZW1wdGl2ZSByZXZvY2F0aW9uIG9mIGlzc3VlZCBjZXJ0aWZpY2F0ZXM7Ci0gR2VuZXJhdGVzIGFuIHVwZGF0ZWQgQ1JMIChDZXJ0aWZpY2F0ZSBSZXZvY2F0aW9uIExpc3QpOwotIFN1cHBseSB0aGUgQ0EgY2VydGlmaWNhdGUuCgpUaGUgUEtJIGltcGxlbWVudGVkIGlzIGNvbXByaXNlZCBvZiBhIHNpbmdsZSBDQSB3aXRoIF9ub18gaW50ZXJtZWRpYXRlIENBcy4KCj09IENvZGUgQ292ZXJhZ2UKCmltYWdlOmh0dHBzOi8vY29kZWNvdi5pby9naC9lbGllemlvL3NpbXBsZS1wa2kvYnJhbmNoL21hc3Rlci9ncmFwaHMvdHJlZS5zdmdbbGluaz0iaHR0cHM6Ly9jb2RlY292LmlvL2doL2VsaWV6aW8vc2ltcGxlLXBraS9icmFuY2gvbWFzdGVyL2dyYXBocy90cmVlLnN2ZyJdCgo9PSBCdWlsZGluZyB0aGUgYXBwbGljYXRpb24KClRvIGJ1aWxkIHRoZSBhcHBsaWNhdGlvbiBmcm9tIHRoZSBzb3VyY2VzIGp1c3QgcnVuOgoKYGBgCi4vZ3JhZGxldyBidWlsZApgYGAKCj09IERvY2tlciBJbWFnZQoKWW91IGNhbiBhbHNvIGJ1aWxkIGEgRG9ja2VyIGltYWdlIGZyb20gdGhlIHNvdXJjZSBjb2RlIHVzaW5nIHRoZQpbSmliIHBsdWdpbl0oaHR0cHM6Ly9naXRodWIuY29tL0dvb2dsZUNvbnRhaW5lclRvb2xzL2ppYi90cmVlL21hc3Rlci9qaWItZ3JhZGxlLXBsdWdpbiNidWlsZC15b3VyLWltYWdlKSBsaWtlLCBmb3IgZXhhbXBsZToKCmBgYAouL2dyYWRsZXcgamliRG9ja2VyQnVpbGQKYGBgCgo9PSBJbnN0YWxsYXRpb24KClRoZSBmaXJzdCBzdGVwIGlzIGNyZWF0ZSB0aGUga2V5c3RvcmUgd2l0aCB0aGUgQ0EgZW50aXR5LgpBcyBhbiBleGFtcGxlLCB5b3UgY2FuIHJ1biB0aGUgYHNyYy9tYWluL3NjcmlwdHMvY3JlYXRlLWNhLnNoYCB0aGF0IHdpbGwgY3JlYXRlIGEgYC4vZGF0YS9jYS5wMTJgIGtleXN0b3JlIHdpdGggZGVmYXVsdCBwYXNzd29yZHMgc2V0IHRvIGBjaGFuZ2VpdGAuCgo9PSBSdW5uaW5nIHdpdGggJ2RlYnVnJyBwcm9maWxlIGFjdGl2YXRlZAoKSWYgdXNpbmcgR3JhZGxlLCBydW46CgpgYGAKLi9ncmFkbGV3IGJvb3RSdW4gLS1hcmdzPSctLXNwcmluZy5wcm9maWxlcy5hY3RpdmU9ZGVidWcnCmBgYAoKPT0gTGljZW5zZQoKQ29weXJpZ2h0IChDKSAyMDE5LTIwMjIgTm9yZGl4IEZvdW5kYXRpb24uCg== readmeEtag: '"640141e2f03f828adbd519c342f339e3e55076fb"' readmeLastModified: Tue, 02 Jan 2024 17:59:53 GMT repositoryId: 193696687 description: A Simple PKI created: '2019-06-25T11:42:04Z' updated: '2024-11-28T18:10:53Z' language: Kotlin archived: false stars: 4 watchers: 0 forks: 1 owner: eliezio logo: https://avatars.githubusercontent.com/u/1125375?v=4 license: NOASSERTION repoEtag: '"23cb926311e4b7d0dcc4bb437d90c5bdb9f6fca5c43f033cf8b67fa1349a200a"' repoLastModified: Thu, 28 Nov 2024 18:10:53 GMT foundInMaster: true category: Server id: 001f83813b19f87d61e5fffe2bc21055 - source: - openapi3 tags - openapi31 tags repository: https://github.com/swaggerexpert/apidom-validate v3: true v3_1: true id: 6d163bb2b4f6635f7cd80f2e92e89b13 repositoryMetadata: base64Readme: >- IyBBcGlET00gVmFsaWRhdG9yIEdpdEh1YiBBY3Rpb24KClRoaXMgR2l0SHViIEFjdGlvbiB2YWxpZGF0ZXMgW09wZW5BUEkgMi4wXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21haW4vdmVyc2lvbnMvMi4wLm1kKSwgW09wZW5BUEkgMy54LnldKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvI29wZW5hcGktc3BlY2lmaWNhdGlvbikgCmFuZCBbQXN5bmNBUEkgMi54XShodHRwczovL3d3dy5hc3luY2FwaS5jb20vZG9jcy9yZWZlcmVuY2Uvc3BlY2lmaWNhdGlvbi92Mi54KSBkZWZpbml0aW9uIGZpbGUgdXNpbmcgW0FwaURPTSBMYW5ndWFnZSBTZXJ2aWNlXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvYXBpZG9tKS4KVmFsaWRhdGlvbiBydWxlcyBhcmUgZXhhY3RseSB0aGUgc2FtZSBhcyB0aGUgb25lIHRoYXQgaHR0cHM6Ly9lZGl0b3ItbmV4dC5zd2FnZ2VyLmlvLyB1c2VzLgoKKipTdXBwb3J0ZWQgc3BlY2lmaWNhdGlvbnM6KioKCi0gW09wZW5BUEkgMi4wXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21haW4vdmVyc2lvbnMvMi4wLm1kKQotIFtPcGVuQVBJIDMuMC4wXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4wLjAuaHRtbCkKLSBbT3BlbkFQSSAzLjAuMV0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMC4xLmh0bWwpCi0gW09wZW5BUEkgMy4wLjJdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMi5odG1sKQotIFtPcGVuQVBJIDMuMC4zXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4wLjMuaHRtbCkKLSBbT3BlbkFQSSAzLjAuNF0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMC40Lmh0bWwpCi0gW09wZW5BUEkgMy4xLjBdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMC5odG1sKQotIFtBc3luY0FQSSAyLjAuMF0oaHR0cHM6Ly92Mi5hc3luY2FwaS5jb20vZG9jcy9yZWZlcmVuY2Uvc3BlY2lmaWNhdGlvbi92Mi4wLjApCi0gW0FzeW5jQVBJIDIuMS4wXShodHRwczovL3YyLmFzeW5jYXBpLmNvbS9kb2NzL3JlZmVyZW5jZS9zcGVjaWZpY2F0aW9uL3YyLjEuMCkKLSBbQXN5bmNBUEkgMi4yLjBdKGh0dHBzOi8vdjIuYXN5bmNhcGkuY29tL2RvY3MvcmVmZXJlbmNlL3NwZWNpZmljYXRpb24vdjIuMi4wKQotIFtBc3luY0FQSSAyLjMuMF0oaHR0cHM6Ly92Mi5hc3luY2FwaS5jb20vZG9jcy9yZWZlcmVuY2Uvc3BlY2lmaWNhdGlvbi92Mi4zLjApCi0gW0FzeW5jQVBJIDIuNC4wXShodHRwczovL3YyLmFzeW5jYXBpLmNvbS9kb2NzL3JlZmVyZW5jZS9zcGVjaWZpY2F0aW9uL3YyLjQuMCkKLSBbQXN5bmNBUEkgMi41LjBdKGh0dHBzOi8vdjIuYXN5bmNhcGkuY29tL2RvY3MvcmVmZXJlbmNlL3NwZWNpZmljYXRpb24vdjIuNS4wKQotIFtBc3luY0FQSSAyLjYuMF0oaHR0cHM6Ly92Mi5hc3luY2FwaS5jb20vZG9jcy9yZWZlcmVuY2Uvc3BlY2lmaWNhdGlvbi92Mi42LjApCgojIyBJbnB1dHMKCiMjIGBkZWZpbml0aW9uLWZpbGVgCgoqKlJlcXVpcmVkKiogUGF0aCB0byBkZWZpbml0aW9uIGZpbGUuCgojIyBgZmFpbHMtb25gCgpTZXZlcml0eSBsZXZlbCBhdCB3aGljaCB0byBmYWlsIGFjdGlvbi4gRGVmYXVsdCBgMWAsIGlmIG5vdCBzcGVjaWZpZWQuCi0gYDFgOiBGYWlscyBpZiAqKmVycm9yKiogbWVzc2FnZXMgZXhpc3QgaW4gdmFsaWRhdGlvbiBvdXRwdXQKLSBgMmA6IEZhaWxzIGlmICoqZXJyb3IqKiBvciAqKndhcm5pbmcqKiBtZXNzYWdlcyBleGlzdCBpbiB2YWxpZGF0aW9uIG91dHB1dAotIGAzYDogRmFpbHMgaWYgKiplcnJvcioqLCAqKndhcm5pbmcqKiBvciAqKmluZm9ybWF0aW9uKiogbWVzc2FnZXMgZXhpc3QgaW4gdmFsaWRhdGlvbiBvdXRwdXQKLSBgNGA6IEZhaWxzIGlmICoqZXJyb3IqKiwgKip3YXJuaW5nKiosICoqaW5mb3JtYXRpb24qKiBvciAqKmhpbnQqKiBtZXNzYWdlcyBleGlzdCBpbiB2YWxpZGF0aW9uIG91dHB1dAoKIyMgRXhhbXBsZSB1c2FnZQoKYGBgeWFtbAp1c2VzOiBzd2FnZ2VyZXhwZXJ0L2FwaWRvbS12YWxpZGF0ZUB2MQp3aXRoOgogIGRlZmluaXRpb24tZmlsZTogJ3BhdGgvdG8vbXkvb3BlbmFwaS55YW1sJwogIGZhaWxzLW9uOiAyCmBgYAo= readmeEtag: '"9fe345231f7e9fc9097fdffc296179137b8a0575"' readmeLastModified: Tue, 31 Dec 2024 07:40:07 GMT repositoryId: 616639890 description: >- This GitHub Action validates OpenAPI 2.0, OpenAPI 3.x.y and AsyncAPI 2.x definition file using ApiDOM Language Service. created: '2023-03-20T19:34:02Z' updated: '2026-02-03T09:41:49Z' language: JavaScript archived: false stars: 10 watchers: 2 forks: 4 owner: swaggerexpert logo: https://avatars.githubusercontent.com/u/172408630?v=4 license: Apache-2.0 repoEtag: '"e0021031f418c885c58e33f91a8eec5ec51bc6e60134f841c1b3f0ce334bc85e"' repoLastModified: Tue, 03 Feb 2026 09:41:49 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/futuristicoder/nomad.oas v3: true repositoryMetadata: base64Readme: >- IyBOb21hZC5PQVMKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9kZXYuYXp1cmUuY29tL0Z1dHVyaXN0aUNvZGVyL05vbWFkLk9BUy9fYXBpcy9idWlsZC9zdGF0dXMvRnV0dXJpc3RpQ29kZXIuTm9tYWQuT0FTP2JyYW5jaE5hbWU9bWFzdGVyKV0oaHR0cHM6Ly9kZXYuYXp1cmUuY29tL0Z1dHVyaXN0aUNvZGVyL05vbWFkLk9BUy9fYnVpbGQvbGF0ZXN0P2RlZmluaXRpb25JZD0xMiZicmFuY2hOYW1lPW1hc3RlcikKCi0tLQoKSGFzaGlDb3JwIE5vbWFkIG9wZW4gYXBpIHNwZWNpZmljYXRpb24KCiMjIENsaWVudCBsaWJyYXJpZXMgKGdlbmVyYXRlZCkKCiMjIyAuTmV0CgotIE5vbWFkLkNsaWVudCBbIVtOdUdldCBTdGF0dXNdKGh0dHBzOi8vYmFkZ2VuLm5ldC9udWdldC92L25vbWFkLmNsaWVudCldKGh0dHBzOi8vd3d3Lm51Z2V0Lm9yZy9wYWNrYWdlcy9Ob21hZC5DbGllbnQpCgpjbGllbnQgbGlicmFyeSBpbiBvdGhlciBsYW5ndWFnZXMgY2FuIGJlIGdlbmVyYXRlZAoKIyMgZW5kcG9pbnRzCgp8IEFwaSAgICAgICAgICAgICAgIHwgSW1wbGVtZW50IHwgQ29tbWVudCAgICAgICAgfAp8LS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tfAp8IEFDTCBQb2xpY2llcyAgICAgIHwgICAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IEFDTCBUb2tlbnMgICAgICAgIHwgICAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IEFnZW50ICAgICAgICAgICAgIHwg4pyTICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgQWxsb2NhdGlvbnMgICAgICAgfCDinJMgICAgICAgICB8ICAgICAgICAgICAgICAgIHwKfCBDbGllbnQgICAgICAgICAgICB8IOKckyAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IERlcGxveW1lbnRzICAgICAgIHwg4pyTICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgRXZhbHVhdGlvbnMgICAgICAgfCDinJMgICAgICAgICB8ICAgICAgICAgICAgICAgIHwKfCBKb2JzICAgICAgICAgICAgICB8IOKckyAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IE5hbWVzcGFjZXMgICAgICAgIHwgICAgICAgICAgIHwgZW50ZXJwcmlzZSBhcGkgfAp8IE5vZGVzICAgICAgICAgICAgIHwg4pyTICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgTWV0cmljcyAgICAgICAgICAgfCAgICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgT3BlcmF0b3IgICAgICAgICAgfCAgICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgUGx1Z2lucyAgICAgICAgICAgfCAgICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgUXVvdGFzICAgICAgICAgICAgfCAgICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgUmVnaW9ucyAgICAgICAgICAgfCDinJMgICAgICAgICB8ICAgICAgICAgICAgICAgIHwKfCBTY2FsaW5nIFBvbGljaWVzICB8ICAgICAgICAgICB8ICAgICAgICAgICAgICAgIHwKfCBTZWFyY2ggICAgICAgICAgICB8IOKckyAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IFNlbnRpbmVsIFBvbGljaWVzIHwgICAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IFN0YXR1cyAgICAgICAgICAgIHwg4pyTICAgICAgICAgfCAgICAgICAgICAgICAgICB8CnwgU3lzdGVtICAgICAgICAgICAgfCDinJMgICAgICAgICB8ICAgICAgICAgICAgICAgIHwKfCBWYWxpZGF0ZSAgICAgICAgICB8IOKckyAgICAgICAgIHwgICAgICAgICAgICAgICAgfAp8IFZvbHVtZXMgICAgICAgICAgIHwg4pyTICAgICAgICAgfCAgICAgICAgICAgICAgICB8Cg== readmeEtag: '"d52ddab7c48aaeb4702014b5c3fb3f6bdc34693c"' readmeLastModified: Thu, 19 Nov 2020 16:55:36 GMT repositoryId: 249948663 description: Nomad OpenApi Specification created: '2020-03-25T10:27:46Z' updated: '2023-09-08T18:05:01Z' language: C# archived: false stars: 4 watchers: 0 forks: 3 owner: FuturistiCoder logo: https://avatars.githubusercontent.com/u/46922135?v=4 license: MIT repoEtag: '"ffdb4d18d7c00bef9c695c2c2f6e6ddab61ad49fda95098e4bd3c007b581ae64"' repoLastModified: Fri, 08 Sep 2023 18:05:01 GMT foundInMaster: true category: Description Validators id: a08f0fbe1cd600070fd4afe4ee90dbb0 - source: openapi3 tags repository: https://github.com/deiteris/api-contractor v3: true repositoryMetadata: base64Readme: >- IyBWaXN1YWwgU3R1ZGlvIENvZGUgZXh0ZW5zaW9uIGZvciBSRVNUIEFQSXMgZGVzaWduaW5nIGFuZCB2YWxpZGF0aW9uCgpTZWUgYGV4dGVuc2lvbi9SRUFETUUubWRgIGZvciBpbmZvcm1hdGlvbiByZWdhcmRpbmcgdGhlIGV4dGVuc2lvbi4KClNlZSBgYXBpLWNvbnNvbGUvUkVBRE1FLm1kYCBmb3IgaW5mb3JtYXRpb24gcmVnYXJkaW5nIHRoZSBBUEkgY29uc29sZSB0aGF0IGlzIHVzZWQgZm9yIHByZXZpZXdpbmcuCg== readmeEtag: '"6eed0bda5116e956daadfd9391c322d2667dced3"' readmeLastModified: Sat, 04 Nov 2023 14:11:20 GMT repositoryId: 359785542 description: >- A Visual Studio Code extension for working with the API contracts in OpenAPI and RAML formats. created: '2021-04-20T11:09:09Z' updated: '2024-05-23T08:53:02Z' language: TypeScript archived: false stars: 4 watchers: 1 forks: 0 owner: deiteris logo: https://avatars.githubusercontent.com/u/6103913?v=4 license: MIT repoEtag: '"a60cee86b9583ea8509b962868c3b24752077fd9fd6a51ede346bb817c9964df"' repoLastModified: Thu, 23 May 2024 08:53:02 GMT foundInMaster: true category: - Testing - Server Implementations id: 64a794fc3fcfdaebd6e5e5d1374ee8ac - source: openapi3 tags repository: https://github.com/xseman/superfaktura.openapi v3: true id: 13977644151f7e71ecab9beace0017dc repositoryMetadata: base64Readme: >- IyBTdXBlcmZha3R1cmEgT3BlbkFQSQoKIVtBbHQgdGV4dF0oZG9jcy9zd2FnZ2VyLXVpLnBuZykKClRoaXMgc3BlY2lmaWNhdGlvbiBoYXMgYmVlbiBtYW51YWxseSBjcmVhdGVkIHVzaW5nIGF2YWlsYWJsZSBbZG9jdW1lbnRhdGlvbl0sCmJ1dCBpdCdzIHBvc3NpYmxlIHRoYXQgdGhlcmUgYXJlIGVycm9ycyBvciBtaXNzaW5nIG5ldyBmZWF0dXJlcyBhcyB0aGUgQVBJCmV2b2x2ZXMgb3ZlciB0aW1lLiBJJ20gbWFraW5nIGFuIGVmZm9ydCB0byBrZWVwIGl0IHVwIHRvIGRhdGUuCgpbZG9jdW1lbnRhdGlvbl06IGh0dHBzOi8vZ2l0aHViLmNvbS9zdXBlcmZha3R1cmEvZG9jcwoKSSB3b3VsZCBncmVhdGx5IGFwcHJlY2lhdGUgYW55IGNvbnRyaWJ1dGlvbnMgb3IgdXBkYXRlcyB0byB0aGUgT3BlbkFQSSBmaWxlIHRvCmFkZHJlc3MgYW55IGlzc3Vlcy4KCiMgQXV0aAoKU2VlIFtkb2N1bWVudGF0aW9uXVthdXRoLWRvY3VtZW50YXRpb25dIHdpdGggZXhhbXBsZXMuCgpbYXV0aC1kb2N1bWVudGF0aW9uXTogaHR0cHM6Ly9naXRodWIuY29tL3N1cGVyZmFrdHVyYS9kb2NzL2Jsb2IvbWFzdGVyL2ludHJvLm1kI2F1dGhlbnRpY2F0aW9uCg== readmeEtag: '"5846138d0c1b8a7e15d146b5fe0bbe6f2cfe2c46"' readmeLastModified: Sat, 10 Aug 2024 09:07:44 GMT repositoryId: 736693265 description: SuperFaktura OpenAPI created: '2023-12-28T15:49:58Z' updated: '2025-09-04T09:00:53Z' language: null archived: false stars: 5 watchers: 1 forks: 0 owner: xseman logo: https://avatars.githubusercontent.com/u/9111485?v=4 license: Apache-2.0 repoEtag: '"bdc7d56dc7080696e7432608736cd11f46850eef3d5ec995319b6af25d9ba7ca"' repoLastModified: Thu, 04 Sep 2025 09:00:53 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/inkognitro/oas-tszod-gen v3: true id: 10f3430580c5b091115eb11b9ca1f843 repositoryMetadata: base64Readme: >- [![continuous integration pipeline](https://github.com/inkognitro/oas-tszod-gen/actions/workflows/ci.yml/badge.svg)](https://github.com/inkognitro/oas-tszod-gen/actions?query=workflow%3Aci)
![test coverage](https://github.com/inkognitro/oas-tszod-gen/blob/main/badges/jest/coverage-total.svg)

# oas-tszod-gen
A code generator to convert [OpenApi version 3 (OAS3)](https://swagger.io/specification/) specifications into endpoint caller functions for [TS](https://www.typescriptlang.org/) and [Zod](https://zod.dev).

[![Demo Video](https://github.com/inkognitro/oas-tszod-gen/blob/main/youtube-demo-thumb.png)](https://www.youtube.com/watch?v=4Ao3GpuFmFE)  
The code of the demo video is available [here](https://github.com/inkognitro/oas-tszod-gen-demo).

## Why yet another library for this task?
Before you consider using this library, I suggest having a look at the combination of [Zodios](https://www.zodios.org/)
and [openapi-zod-client](https://github.com/astahmer/openapi-zod-client) or [OpenApi Typescript](https://openapi-ts.dev/) (if you are OK with just Typescript). This code generator serves as a combined alternative for these three libraries.
However, I coded my own solution because I wanted to:

- have full ownership of my production code without any additional dependency
- have generated endpoint caller functions representing the reality which is given from some OAS3 specs
- be able to easily test these endpoint caller functions with exchangeable `RequestHandler` implementations
- have a uniform adapter API which passes an endpoint schema inside the request adapter object
- have separated functions and type definitions for each endpoint in a single file, located in its context folder
- be able to go with other implementations under the hood than [Axios](https://axios-http.com/docs/intro)
- have the opt-in possibility for generated [Zod](https://zod.dev) schemas with included runtime validation of requests
and responses
- have a uniform way as an alternative to [Axios' interceptors](https://axios-http.com/docs/interceptors)
through different `RequestHandler` implementations

> :bulb: [Axios](https://axios-http.com/docs/intro) is great. It offers a lot, and it works with
> [XMLHttpRequest](https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest_API) under the hood by default.
> But one might have PWAs and [service workers](https://developer.mozilla.org/en-US/docs/Web/API/Service_Worker_API)
> in mind where XMLHttpRequest is out of scope. There are lots of articles out there which discuss things like
> [Axios Vs Fetch](https://blog.logrocket.com/axios-vs-fetch-best-http-requests/).
> Even though, since version 1.7.0 Axios does natively support a
> [fetch adapter](https://github.com/axios/axios?tab=readme-ov-file#-fetch-adapter).

## Generator: Setup and usage
First, install the package as a dev dependency:
```
npm install oas-tszod-gen --save-dev
```

If you also want to use [Zod](https://zod.dev) (see `withZod: true` configuration below) and it is not yet installed
as a dependency of your project:
```
npm install zod --save
```

Now create a new file `api.specs.json` in the root folder of your project and paste your OAS3 specification into it.
The next step is to create a file `api.generate.js` - also in the root folder of your project - and paste the following
code into it:

```typescript
// api.generate.js

const {generateOas3ToTs} = require('oas-tszod-gen');
const oas3Specification = require('./api.specs.json');

generateOas3ToTs({
  // required:
  getSpecification: () => {
    return new Promise(resolve => {
      resolve(oas3Specification);
    });
  },
  
  // required:
  outputFolderPath: './generated-api',

  // optional:
  // The list of separators which should be considered for the outputPath creation from a
  // component name. The outputPath itself is an array of strings (see explanation in one
  // of the sections below). By default, the following configuration is taken:
  // ['.', '/', '\\']
  componentOutputPathSeparators: ['.', '/', '\\', '_'],
  
  // optional:
  // Same as "componentOutputPathSeparators" but in the context of defined "operationId" properties
  // of an endpoint
  operationIdOutputPathSeparators: ['.', '/', '\\', '_'],
  
  // optional:
  // The folder structure is generated from resulting outputPaths of the given operationIds.
  // If some additional sub folders are required the configuration can be done as follow:
  predefinedFolderOutputPaths: [
    // For outputs having an `OutputPath` starting with ['core']:
    // Variables and type definitions are put in the `{outputFolderPath}/core` folder
    ['core'],
    
    // For outputs having an `OutputPath` starting with ['util', 'foo']:
    // Variables and type definitions are put in the `{outputFolderPath}/util/foo` folder
    ['util', 'foo'],
  ],
  
  // required:
  logger: {
    log: (content) => {
      console.log(content);
    },
  },
  
  // optional:
  // By default or in case of `false`, only TypeScript types are generated without Zod schemas
  withZod: true,
  
  // optional:
  // The following will add only the defined templates.
  // An empty array results in no added templates.
  // By default, all available templates will be added to your codebase.
  templates: [
    'AxiosRequestHandler',
    'AuthRequestHandler',
    'ResponseExtractors',
    'ZodValidationRequestHandler'
  ],

  // optional: Modify output paths which result from an operation
  createOperationOutputPath: (_path, _method, _endpointSchema, defaultOutputPath) => {
    return defaultOutputPath[0] === 'v1' ? defaultOutputPath.slice(1) : defaultOutputPath;
  },
  
  // optional: Whether a specific operation (endpoint respectively) should be added
  shouldAddOperation: (_path, _method, _endpointSchema) => {
    return true; // adjust according to your needs
  },
  
  // optional: Whether a specific content type request body should be added
  shouldAddRequestBodyContent: (contentType, bodyByContentTypeMap) => {
    const lowerCaseContentType = contentType.toLowerCase();
    if (lowerCaseContentType.match(/application\/[^+]*[+]?(json);?.*/)) {
      return true;
    }
    const hasJsonAlternative = Object.keys(bodyByContentTypeMap).find(ct =>
      ct.toLowerCase().match(/application\/[^+]*[+]?(json);?.*/)
    );
    if (hasJsonAlternative) {
      return false;
    }
    return !!lowerCaseContentType.match(/multipart\/form-data;?.*/);
  },
  
  // optional: Whether a specific content type response body should be added
  shouldAddResponseBodyContent: (contentType, _bodyByContentTypeMap) => {
    return ['multipart/form-data', 'application/json'].includes(contentType);
  },
});
```

Finally run the script to generate the output files into the `./generated-api` folder:
```
node api.generate.js
```

**Important:**
It is recommended to regenerate the output in your CI pipeline. On every call of the `generateOas3ToTs` function,
the output folder is going to be deleted and the code will be generated from scratch.

> :bulb: Node version `>= 18.0.0` is required to support
> the [Blob](https://nodejs.org/docs/v20.17.0/api/globals.html#class-blob) class
> if you want to use the generated code on the backend side.

## What is an OutputPath?
Variables, functions and type definitions which are generated by this generator come in form of an `output`.
A `output` is an object which has a `path` property in it. This property is of type `OutputPath` (= `string[]`).
The first few entries of the `OutputPath` array do represent the filepath in which the variables and type definitions are going to be put in.
The remaining entries of this array are used for the type or the variable name.

Let's consider some example endpoints which might be defined in your OAS3 specification under the `paths` property:

1. The generated code for the endpoint with the operationId `auth.authenticate` will be put into `./auth/authenticate.ts`
2. The generated code for the endpoint with the operationId `userManagement.getUsers` will be put into `./user-management/getUsers.ts`
3. The generated code for the endpoint with the operationId `userManagement.admin.getUsers` will be put into `./user-management/admin/getUsers.ts`

The same principles that apply to dots `.` do also apply to slashes `/` and backslashes `\` by default.
This can be configured with the configuration prop `operationIdOutputPathSeparators` for operationIds and
with `componentOutputPathSeparators` for component names (see above).

After generating endpoint caller functions from an OAS3 specification into your codebase,
it's time to understand how the generated code can be used. The following sections should bring some clarity.

## Usage of the generated code by example
The following example code parts do illustrate a login process.
In order to process this, a request is triggered by calling the generated `authenticate` endpoint caller function.
After receiving the `RequestResult` followed by the check of the according response status 200, TypeScript recognizes
the according types of the received response.

The following code shows how to create a `RequestHandler` instance, which ideally is provided as a service in your app.
Different `RequestHandler` implementations are composed together.
These implementations came according due to the `templates` configuration of the code generator.

```typescript
// requestHandler.ts

import {
  AuthRequestHandler,
  AuthRequestHandlerExecutionConfig,
  AxiosRequestHandler,
  AxiosRequestHandlerExecutionConfig,
  BearerAuthProvider,
  ZodValidationRequestHandler,
  ZodValidationRequestHandlerExecutionConfig,
} from './generated-api/core';
import {authenticate} from './generated-api/auth';
import axios from 'axios';
import {parse} from 'qs';
import {BearerAuthProvider} from "./authRequestHandler";

declare global {
  interface RequestHandlerExecutionConfig
    extends AxiosRequestHandlerExecutionConfig,
      AuthRequestHandlerExecutionConfig,
      ZodValidationRequestHandlerExecutionConfig {
  }
}

class MyJwtAuthProvider implements BearerAuthProvider {
  public readonly type: 'bearer';

  // This is the name of one of your security definition in your OAS3 specification
  public readonly securitySchemeName: 'myJwtAuth';

  private readonly accessToken: null | string;

  setToken(accessToken: string | null) {
    this.accessToken = accessToken;
  }

  findToken(): null | string {
    return this.accessToken;
  }
}

export const myJwtAuthProvider = new MyJwtAuthProvider();

const myAxiosInstance = axios.create({
  baseURL: 'https://api.acme.com',

  // In case of browsers:
  // Allow cookies to be passed along with the request for a different api (sub-)domain (CORS)
  // Source: https://developer.mozilla.org/en-US/docs/Web/API/XMLHttpRequest/withCredentials
  withCredentials: true,
});

const axiosRequestHandler = new AxiosRequestHandler({
  axios: myAxiosInstance,
  urlDecodeQueryString: (queryString: string) => {
    return parse(queryString);
  },
});

const zodRequestHandler = new ZodValidationRequestHandler(axiosRequestHandler);

export const requestHandler = new AuthRequestHandler(
  // In case of multiple authentication providers, the order does not matter.
  // Multiple authentication providers are prioritized by the sorting of the names which
  // are listed in the "request.endpointSchema.supportedSecuritySchemas" array.
  // This array has the same order as the security schemes in the OAS3 specs of that operation,
  // or endpoint respectively. The first found token received from the "findToken" method of a
  // supported authentication provider is then added to the request headers.
  [myJwtAuthProvider],

  zodRequestHandler
);
```

Below are three different variants to show how an endpoint call can look like when
using a generated endpoint caller functions.

The first function does directly **reveal** the response body.
This is done by the provided extractor function `getRevealedResponseOrReject`.
With this extractor function an error is thrown in case the actual received response does not correspond
to the expected one. This happens by checking the status and the content type of the response.

```typescript
// login1.ts

import { requestHandler, myJwtAuthProvider } from './requestHandler';
import { getRevealedResponseOrReject } from './generated-api/core';
import { authenticate } from './generated-api/auth';

async function loginOrThrowError() {
  const res = await getRevealedResponseOrReject(200, 'application/json', authenticate(
    requestHandler,
    {
      contentType: 'application/json',
      body: {
        usernameOrEmail: 'inkognitro',
        password: '12345678'
      },
    },
    optionalConfig
  ));
  myJwtAuthProvider.setToken(res.body.accessToken);
}
```
This can be useful when using [@tanstack/query](https://tanstack.com/query/latest).

With the second variant one can achieve the same as in the first example without throwing an error in case of
another response than expected. This can be achieved by the `findRevealedResponse` like so:

```typescript
// login2.ts

import { requestHandler, myJwtAuthProvider } from './requestHandler';
import { findRevealedResponse } from './generated-api/core';
import { authenticate } from './generated-api/auth';

async function loginOrDoNothing() {
  const res = await findRevealedResponse(200, 'application/json', authenticate(
    requestHandler,
    {
      contentType: 'application/json',
      body: {
        usernameOrEmail: 'inkognitro',
        password: '12345678'
      },
    },
    optionalConfig
  ));
  if (!res) {
    console.log('response is ignored...');
    return;
  }
  myJwtAuthProvider.setToken(res.body.accessToken);
}
```

Last but not least, the third variant. This variant serves as an overview of what is going on under the hood in
those extractor functions from above.
```typescript
// login3.ts

import { requestHandler, myJwtAuthProvider } from './requestHandler';
import { authenticate } from './generated-api/auth';

async function loginWithExplicitResponseBodyRevealation() {
  const requestResult = await authenticate(
    requestHandler,
    {
      contentType: 'application/json',
      body: {
        usernameOrEmail: 'inkognitro',
        password: '12345678'
      },
    },
  );

  if (!requestResult.response || requestResult.response.status !== 200) {
    console.error('Something went wrong...');
    return;
  }

  // The following check is only required if the response body of the response
  // with the status code 200 can have ambiguous content types according to the
  // given OAS3 specs.
  if (requestResult.response.contentType !== 'application/json') {
    console.error('unsupported response body:', requestResult.response.body);
    return;
  }

  const body = await requestResult.response.revealBody();
  myJwtAuthProvider.setToken(body.accessToken);
}
```
There exist another two extractor functions `getResponseOrReject` and `findResponse`.
These two functions do return a response with a body which was not yet **revealed**.
Anyway, this is for convenience, so we don't have to write that much code.

> :bulb: The `response.contentType` property does not always match the exact value of the actual `response.headers['content-type']`
> which was received from the server. The `response.contentType` property's reason to exist is to make sure that we
> are able to distinguish between different response types which were defined by the OAS3 specs.
> The OAS3 content type that has the most characters matching the actual `content-type` header is adopted during runtime.

## RequestHandler implementations
The decision of having a `RequestHandler` interface with granular implementations was made with different environments
in mind: `server` vs `client` | `prod` vs `test` | custom "middleware" behaviour.
You can write your own implementations or just combine some of the existing ones below, according to your needs.
The `RequestHandler` interface for custom implementations can be found
[here](https://github.com/inkognitro/oas-tszod-gen/blob/main/src/templates/core/core.ts#L343).

### AxiosRequestHandler
This implementation is responsible for executing your requests through the http(s) protocol.
It is usually the most inner implementation of an onion bootstrapped request handler object.
It requires the installation of the [Axios](https://axios-http.com/docs/intro) library in your code base.
For the recommended Axios version have a look at the `peerDependencies` in the `package.json` of this project.
Creating an AxiosRequestHandler instance comes with the following steps:

```
npm install axios --save
```

To support `application/x-www-form-urlencoded` response bodies, this request handler implementation requires
a custom `decoding` function to decode url encoded data.
The [qs](https://www.npmjs.com/package/qs) library currently is the most commonly used library
for such tasks and therefore recommended:

```
npm install qs --save

npm install @types/qs --save-dev
```

And finally:

```typescript
const requestHandler = new AxiosRequestHandler({
  axios: axios.create({
    baseURL: `http://localhost:3000`,
  }),
  urlDecodeQueryString: (queryString: string) => {
    return parse(queryString);
  },
});
```

> :bulb: The AxiosRequestHandler forces Axios to always receive response bodies in form of an `ArrayBuffer`.
> This is achieved through setting `responseType: 'arraybuffer'` inside
> the [AxiosRequestConfig](https://axios-http.com/docs/req_config).
> A preconfigured `responseType` property is therefore always overwritten by the request handler.
> The same applies to `cancelToken`.

### FetchApiRequestHandler
This implementation is responsible for executing your requests through the http(s) protocol.
It is usually the most inner implementation of an onion bootstrapped request handler object
and uses the built-in [Fetch API](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API).

To support `application/x-www-form-urlencoded` request and response bodies, this request handler implementation requires
custom `decoding` and `encoding` functions to convert data from and to query string format.
The [qs](https://www.npmjs.com/package/qs) library currently is the most commonly used library
for such tasks and therefore recommended:
```
npm install qs --save

npm install @types/qs --save-dev
```

Create a FetchApiRequestHandler instance as written below:
```typescript
import { FetchApiRequestHandler } from './generated-api/core';
import { stringify } from 'qs';

const fetchApiRequestHandler = new FetchApiRequestHandler({
  // These two functions are required
  urlEncodeQueryString: plainObject => stringify(queryParams),
  urlDecodeQueryString: (queryString: string) => {
    return parse(queryString);
  },

  // Following is optional stuff
  baseUrl: 'http://localhost:8000',
  generalRequestInit: {
    credentials: 'include', // Send credential headers
    mode: 'cors', // Enable CORS
  },
});
```

### AuthRequestHandler
This implementation can be taken to add authentication information to requests automatically.
This is done as defined in the [OAS Authentication specs](https://swagger.io/docs/specification/v3_0/authentication/),
according to the given
[AuthProvider](https://github.com/inkognitro/oas-tszod-gen/blob/main/src/templates/core/authRequestHandler.ts#L29)(s).
As of the time of writing this, `basic`, `bearer` and `apiKey` authentication is supported.

### ScopedRequestHandler
With this implementation you are able to make sure that the methods `cancelRequestById` and `cancelAllRequests` do only
cancel the requests which were made through the `execute` method of exactly that `ScopedRequestHandler` instance.
This might be useful when you want to provide a separate `ScopedRequestHandler` instance for React components.
Anyway, React stuff is out-of-scope of this package.

### ZodValidationRequestHandler
This implementation is responsible for validating the request and response data through their Zod schemas.
This one is only available when Zod schemas are generated due to the `withZod: true` configuration.
This request handler does `reject` the promise with the Zod error when requests or responses do not comply with the
Zod schema definitions which are defined in the requests' `endpointSchema` property.
It uses the [safeParse](https://zod.dev/?id=safeparse) method for this task.

> :bulb: In case of "multipart/form-data" requests, you are allowed to either pass a FormData instance or
> a plain object in the `body` property of the endpoint caller function payload.
> The ZodValidationRequestHandler only validates plain object request bodies and skips validation for FormData instances.
> This also applies to response bodies which come as FormData after calling the `revealBody` method of a response.

## Promises: `resolve` vs `reject`
The provided `RequestHandler` implementations distinguish between "expected" and
"unexpected" events: An "expected" event is for example a response which could not be received due to cancellation or network issues.
A response with no 2xx status code is considered to be "expected" as well.
In such cases, the Promise which was delivered by the `requestHandler.execute` method, is going to be `resolved`.
On the other hand, things like programming errors or other runtime errors due to wrong source code are considered
to be "unexpected" and will result in a `rejected` Promise. This applies to all provided RequestHandler implementations.

This behaviour should lead to a better developer experience because thrown errors must only be caught at a root level
of an application then.

## Semantic versioning
**Worried about different code generation outputs after updating this library?**
As long as the API contract on the backend side was not violated through breaking changes,
you don't have to worry about breaking changes in generated code output when upgrading to the next **minor** version.

## Lacking OAS3 support
- Although `default` is supported for response status codes,
status code pattern matching (e.g. for [5XX](https://swagger.io/docs/specification/describing-responses/))
is ignored by the generator and will not get recognized by TS.
As of right now, it is added as a possible response having `any` status code.
- The generated code for the ["not" keyword](https://swagger.io/docs/specification/data-models/oneof-anyof-allof-not/#not)
is `any` for TS and `z.any()` for Zod.
- Only **Local References** are supported for [$ref syntax](https://swagger.io/docs/specification/using-ref/).
One might collect all external `$ref` definitions and modify the specs accordingly to have only local references
when passing the specs to the generator.
- The `discriminator.mapping` property - as described
[here](https://swagger.io/docs/specification/data-models/inheritance-and-polymorphism/) is ignored.
It is recommended to define the discriminating literal value directly in the schemas which are referenced in
`oneOf` or `anyOf` (as long as the discriminator is defined in there). This can be achieved through a string enum.
- The properties `readOnly` and `writeOnly` described [here](https://swagger.io/docs/specification/data-models/data-types/) are ignored.
It is recommended to have separate definitions for read and write objects, especially with changing requirements
(see also [CQRS pattern](https://martinfowler.com/bliki/CQRS.html)).
Anyway, this is planned to be implemented but currently has low priority to me.

## Extra
- Request `headers` and `cookies` are always optional. At the moment there is no configuration setting for this.
This is because one might avoid defining common headers for every call of an endpoint caller function and define
those in a separate underlying request handler. However, as soon as a request `headers` object is passed in to the
endpoint caller function, it must fulfill the definition of the given OAS3 specs, e.g. `required: true`
of a specific header.
- [AnyOf](https://swagger.io/docs/specification/v3_0/data-models/oneof-anyof-allof-not/#anyof) does officially support
only ObjectSchemas or ObjectSchemaComponentRefs.
However, in case of an OAS3 spec which defines another type in the `anyOf` property
(e.g. a [StringSchema](https://swagger.io/docs/specification/v3_0/data-models/data-types/#string)), the generator does
generate code which treats non-ObjectSchema types as another union option.

## Out of scope
Linting is out of scope of this package. The generator is responsible to generate valid TypeScript (and Zod) definitions.
That's what it does. Your code setup should be responsible for linting your code, so it might make sense for you to run
[ESLint](https://eslint.org/) right after the `generateOas3ToTs` function was executed, e.g. with a script in your
`package.json` like so:
```json
"scripts": {
  "api:generate": "node api.generate.js && eslint --fix --quiet ./generated-api"
}
```

## Bug report / feature request
If you think an unsupported feature needs to be implemented, feel free to open an issue or create a new
pull request. Please refer to the pull request criteria which is defined [here](.github/pull_request_template.md).
 readmeEtag: '"398ac0ed42a7331f49ed33490ac1e5fa62563fd7"' readmeLastModified: Sun, 20 Oct 2024 14:00:48 GMT repositoryId: 763628347 description: >- Client SDK code generator to convert OpenApi v3 specifications into TS endpoint caller functions with Zod types. created: '2024-02-26T16:38:59Z' updated: '2026-01-06T05:08:53Z' language: TypeScript archived: false stars: 8 watchers: 1 forks: 0 owner: inkognitro logo: https://avatars.githubusercontent.com/u/22317587?v=4 license: MIT repoEtag: '"0be6ac4f7411df838e5f7b5bf529a065869d0d8fce117c3a952471f6f39f928c"' repoLastModified: Tue, 06 Jan 2026 05:08:53 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/falcoframework/falco.openapi v3: true id: 200b9168cb016f82f97f3f6278a73df0 repositoryMetadata: base64Readme: >- # Falco.OpenAPI

[![NuGet Version](https://img.shields.io/nuget/v/Falco.OpenApi.svg)](https://www.nuget.org/packages/Falco.OpenApi)
[![build](https://github.com/FalcoFramework/Falco.OpenApi/actions/workflows/build.yml/badge.svg)](https://github.com/FalcoFramework/Falco.OpenApi/actions/workflows/build.yml)

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.name "HelloWorld"
            |> OpenApi.summary "This is a summary"
            |> OpenApi.description "This is a test description, which is the long form of the summary."
            |> OpenApi.returnType typeof<string>
    ]
```

[Falco.OpenAPI](https://github.com/FalcoFramework/Falco.OpenAPI) is a library for generating OpenAPI documentation for [Falco](https://github.com/FalcoFramework/Falco) applications. It provides a set of combinators for annotating Falco routes with OpenAPI metadata, which can be used to generate OpenAPI documentation.

## Key Features

- Generates OpenAPI 3.0 documentation.
- Provides a set of combinators for annotating Falco routes with OpenAPI metadata.
- Generates OpenAPI documentation from metadata associated to Falco routes.
- Allows for customization of the generated OpenAPI documentation.

## Design Goals

- Minimalistic and easy to use.
- Explicit, allowing for fine-grained control over the generated OpenAPI documentation.
- Type-safe and to leverage the type system to prevent common mistakes.
- Composable, allowing for the generation of complex OpenAPI documentation.

## Getting Started

This guide assumes you have a [Falco](https://github.com/FalcoFramework/Falco) project setup. If you don't, you can create a new Falco project using the following command:

```shell
> dotnet new web -lang F# -o HelloWorld
> cd HelloWorldApp
```

Install the nuget package:

```shell
> dotnet add package Falco
> dotnet add package Falco.OpenApi
```

Remove any `*.fs` files created automatically, create a new file named `Program.fs` and set the contents to the following:

```fsharp
open Falco
open Falco.Routing
open Microsoft.AspNetCore.Builder

let bldr = WebApplication.CreateBuilder()
let wapp = bldr.Build()

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
    ]

wapp.UseRouting()
    .UseFalco(endpoints)
    .Run()
```

Now, let's incorporate OpenAPI documentation into our Falco application. Update the `Program.fs` file to the following:

```fsharp
// ..

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.name "HelloWorld"
            |> OpenApi.summary "This is a summary"
            |> OpenApi.description "This is a test description, which is the long form of the summary."
            |> OpenApi.returnType typeof<string>
    ]

// ..
```

Which produces the following OpenAPI documentation:

```json
{
  "openapi": "x.x.x",
  "info": {
    "title": "HelloWorld",
    "version": "1.0"
  },
  "paths": {
    "/": {
      "get": {
        "tags": [
          "HelloWorld"
        ],
        "summary": "This is a summary",
        "description": "This is a test description, which is the long form of the summary",
        "operationId": "HelloWorld",
        "responses": {
          "200": {
            "description": "OK",
            "content": {
              "text/plain": {
                "schema": {
                  "type": "string"
                }
              }
            }
          }
        }
      }
    }
  },
  "components": { }
}
```

## Specs

### Name (Operation ID)

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.name "HelloWorld"
    ]
```

### Summary

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.summary "This is a summary"
    ]
```

### Description

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.description "This is a description"
    ]
```

### Return Type

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.returns typeof<string>
            // or,
            // |> OpenApi.returns { Return = typeof<string>; ContentType = ["text/plain"]; Status = 200 }
    ]
```

### Route Parameters

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        mapGet "/hello/{name?}"
            (fun route -> route?name.AsString("world"))
            Response.ofPlainText
            |> OpenApi.route [
                { Type = typeof<string>; Name = "name"; Required = false } ]
    ]
```

### Query Parameters

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        mapGet "/hello"
            (fun route ->
                let age = route?age.AsInt(0)
                $"Hello there, your are {age} years old.")
            Response.ofPlainText
            |> OpenApi.route [
                { Type = typeof<int>; Name = "age"; Required = false } ]
    ]
```

### Header Parameters

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/"
            (fun ctx ->
                let headers = Request.getHeaders ctx
                let versionId = headers.GetString "X-Version-ID"
                $"Hello, you are using version {versionId}")
            |> OpenApi.header [
                { Type = typeof<string>; Name = "X-Version-ID"; Required = true } ]
    ]
```

### Tags

```fsharp
open Falco
open Falco.OpenApi
open Falco.Routing

let endpoints =
    [
        get "/" (Response.ofPlainText "Hello World!")
            |> OpenApi.tags [
                "tag1"
                "tag2" ]
    ]
```

## Find a bug?

There's an [issue](https://github.com/FalcoFramework/Falco.OpenApi/issues) for that.

## License

Licensed under [Apache License 2.0](https://github.com/FalcoFramework/Falco.OpenApi/blob/master/LICENSE).
 readmeEtag: '"9f9887dfe1cbf6d935809b2462a996dd630dabcf"' readmeLastModified: Wed, 17 Sep 2025 12:17:38 GMT repositoryId: 874218513 description: OpenAPI integration for Falco. created: '2024-10-17T12:58:45Z' updated: '2025-12-25T09:29:18Z' language: F# archived: false stars: 11 watchers: 2 forks: 1 owner: falcoframework logo: https://avatars.githubusercontent.com/u/204376955?v=4 license: Apache-2.0 repoEtag: '"3d06691daad21bb1a3935e5f1de75c9e5d0e3ed21ebfdf81a281bb58b660ac73"' repoLastModified: Thu, 25 Dec 2025 09:29:18 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/pimbrouwers/falco.openapi - source: openapi3 tags repository: https://github.com/gamefabric/openapi v3: true id: 22de0ad83c5048756bcfefbc075ec4f6 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNpZmljYXRpb24gR2VuZXJhdG9yCgpgb3BlbmFwaWAgcHJvdmlkZXMgYSBmcmFtZXdvcmsgZm9yIGdlbmVyYXRpbmcgT3BlbkFQSSB2MyBzcGVjaWZpY2F0aW9ucywgdXNpbmcgYSBjaGkgbXV4LCBmcm9tIGNvZGUuIEl0IGF0dGFjaGVzIGFzIG9uZSBvcgptb3JlIG1pZGRsZXdhcmUgdG8gZGVzY3JpYmUgZWFjaCByb3V0ZSwgcmVtb3ZpbmcgaXRzZWxmIGF0IHJ1bnRpbWUgdG8gaGF2ZSBubyBwZXJmb3JtYW5jZSBpbXBhY3QuCgojIyBVc2FnZQoKIyMjIFN0cnVjdCBHZW5lcmF0b3IKCmBvYXBpLWdlbmAgaXMgYSBzdHJ1Y3QgZnVuY3Rpb24gZ2VuZXJhdG9yIHVzZWQgdG8gY3JlYXRlIGRvY3VtZW50YXRpb24gYW5kIGF0dHJpdWJ1dGVzIGZyb20gc3RydWN0IGZpZWxkIGNvbW1lbnRzLgoKQW55IHN0cnVjdCB3aXRoIHRoZSBkaXJlY3RpdmUgYG9wZW5hcGk6Z2VuYCB3aWxsIGhhdmUgYSBkb2N1bWVudGF0aW9uIGZ1bmN0aW9uIGdlbmVyYXRlZC4KCiMjIyMgSW5zdGFsbAoKYGBgc2hlbGwKJCBnbyBpbnN0YWxsIGdpdGh1Yi5jb20vZ2FtZWZhYnJpYy9vcGVuYXBpL2NtZC9vYXBpLWdlbkA8dmVyc2lvbj4KYGBgCgpBIHNpbXBsZSB3YXkgdG8ga2VlcCB0aGUgZ2VuZXJhdGlvbiB1cC10by1kYXRlIGlzIHRvIHVzZSBHbydzIGdlbmVyYXRpb24gZnJhbWV3b3JrIG9uIGVhY2ggcGFja2FnZSB0aGF0IG5lZWRzIGRvY3VtZW50YXRpb24gZnVuY3Rpb25zLgoKYGBgZ28KLy9nbzpnZW5lcmF0ZSBvYXBpLWdlbgpgYGAKClRoZW4gcnVuOgoKYGBgc2hlbGwKZ28gZ2VuZXJhdGUgLi8uLi4KYGBgCgojIyMjIEV4YW1wbGUKClJ1bm5pbmcgYG9hcGktZ2VuYCBvbiB0aGUgZm9sbG93aW5nIHBhY2thZ2U6CgpgYGBnbwpwYWNrYWdlIHNvbWVwYWNrYWdlCgovLyBUZXN0T2JqZWN0IGlzIGEgdGVzdCBvYmplY3QuCi8vCi8vb3BlbmFwaTpkb2NzCnR5cGUgVGVzdE9iamVjdCBzdHJ1Y3QgewoJLy8gQSBpcyBhbiBleGFtcGxlIGZpZWxkLgoJQSBzdHJpbmcgYGpzb246ImEiYAoKCS8vIEIgaXMgYW5vdGhlciBleGFtcGxlIGZpZWxkLgoJLy8KCS8vb3BlbmFwaTpyZXF1aXJlZAoJLy9vcGVuYXBpOmZvcm1hdD1pcHY0CglCIHN0cmluZwp9CmBgYAoKV2lsbCBwcm9kdWNlIGEgZG9jdW1lbnRhdGlvbiBmaWxlIGB6el9nZW5lcmF0ZWQuZG9jcy5nb2Agd2l0aCB0aGUgY29udGVudDoKCmBgYGdvCnBhY2thZ2Ugc29tZXBhY2thZ2UKCi8vIENvZGUgZ2VuZXJhdGVkIGJ5IG9hcGktZG9jZ2VuLiBETyBOT1QgRURJVC4KCi8vIERvY3MgcmV0dXJucyBhIHNldCBvZiBwcm9wZXJ0eSBkZXNjcmlwdGlvbnMgcGVyIHByb3BlcnR5LgpmdW5jIChUZXN0T2JqZWN0KSBEb2NzKCkgbWFwW3N0cmluZ11zdHJpbmcgewoJcmV0dXJuIG1hcFtzdHJpbmddc3RyaW5newoJCSJCIjogIkIgaXMgYW5vdGhlciBleGFtcGxlIGZpZWxkLiIsCgkJImEiOiAiQSBpcyBhbiBleGFtcGxlIGZpZWxkLiIsCgl9Cn0KCi8vIEF0dHJpYnV0ZXMgcmV0dXJucyBhIHNldCBvZiBwcm9wZXJ0eSBhdHRyaWJ1dGVzIHBlciBwcm9wZXJ0eS4KZnVuYyAoVGVzdE9iamVjdCkgQXR0cmlidXRlcygpIG1hcFtzdHJpbmddc3RyaW5nIHsKCXJldHVybiBtYXBbc3RyaW5nXXN0cmluZ3sKCQkiQiI6ICJyZXF1aXJlZCIsCgl9Cn0KCi8vIEZvcm1hdHMgcmV0dXJucyBhIHNldCBvZiBwcm9wZXJ0eSBmb3JtYXRzIHBlciBwcm9wZXJ0eS4KZnVuYyAoVGVzdE9iamVjdCkgRm9ybWF0cygpIG1hcFtzdHJpbmddc3RyaW5nIHsKCXJldHVybiBtYXBbc3RyaW5nXXN0cmluZ3sKCQkiQiI6ICJpcHY0IiwKCX0KfQpgYGAKClRoZSBmb2xsb3dpbmcgZGlyZWN0aXZlcyBjYW4gYmUgdXNlZCBvbiBzdHJ1Y3QgZmllbGRzOgoKKiBgb3BlbmFwaTpyZXF1aXJlZGA6IE1hcmtzIHRoZSBmaWVsZCBhcyByZXF1aXJlZC4KKiBgb3BlbmFwaTpyZWFkb25seWA6IE1hcmtzIHRoZSBmaWVsZCBhcyByZWFkIG9ubHkuCiogYG9wZW5hcGk6Zm9ybWF0PTxGT1JNQVQ+YDogU2V0cyB0aGUgZm9ybWF0IG9mIHRoZSBmaWVsZCwgZS5nLiAiZGF0ZSIgb3IgImlwdjQiLiBTZWUgW2xpc3Qgb2YgdmFsaWQgZm9ybWF0c10oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9yZWdpc3RyeS9mb3JtYXQvKS4KKiBgb3BlbmFwaTplbnVtPTEsMiwzYDogU2V0cyB0aGUgYWxsb3dlZCBlbnVtIHZhbHVlcyBmb3IgdGhpcyBmaWVsZC4KCiMjIyMgTW9yZSBPcHRpb25zCgpgb2FwaS1nZW5gIGNvbW1hbmQgc3VwcG9ydHMgdGhlIGZvbGxvd2luZyBhZGRpdGlvbmFsIGFyZ3VtZW50cy4KCmBgYHNoZWxsCk9wdGlvbnM6CiAgLWFsbAogICAgCVBhcnNlIGFsbCBzdHJ1Y3RzLgogIC1wYXRoIHN0cmluZwogICAgCVRoZSBwYXRoIHRvIHBhcnNlIGZvciBkb2N1bWVudGF0aW9uLiBEZWZhdWx0cyB0byB0aGUgY3VycmVudCB3b3JraW5nIGRpcmVjdG9yeS4KICAtcQlTdXBwcmVzcyBnZW5lcmF0aW9uIG91dHB1dC4KICAtdGFnIHN0cmluZwogICAgCVRoZSB0YWcgdG8gb3ZlcnJpZGUgdGhlIGRvY3VtZW50YXRpb24ga2V5LiAoZGVmYXVsdCAianNvbiIpCmBgYAo= readmeEtag: '"b7d9bb15073e3c5f94acb710eb960ad71383dced"' readmeLastModified: Tue, 01 Jul 2025 07:11:16 GMT repositoryId: 865821942 description: >- openapi provides a framework for generating OpenAPI v3 specifications, using a chi mux, from code. created: '2024-10-01T07:35:24Z' updated: '2026-01-23T10:24:27Z' language: Go archived: false stars: 4 watchers: 3 forks: 3 owner: GameFabric logo: https://avatars.githubusercontent.com/u/179205010?v=4 license: MIT repoEtag: '"e7b1af785ce1b532bab029e87df332a1183f7db6a5a6d43335edfad8afcc66e0"' repoLastModified: Fri, 23 Jan 2026 10:24:27 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/alma-cfao/pyapifirstdemo v3: true repositoryMetadata: base64Readme: >- # PyApiFirstDemo
This project contains demo code of 12/12/2019 python meetup in grenoble called:

Tools to write clean microservices in python without headaches. (demo project)


You can find slides [here](https://github.com/Alma-CFAO/PyApiFirstDemo/blob/master/Meetup%20Python%20microservices%20decembre%202019.pdf).

This project is a demo project you can use as microservice project template.
It conbines django ORM and connexion (flask) framework together.
You can find below all the things you can do with this project template.

## Project scructure
    .
    ├── apps
    │   ├── boilerplace
    │   │   ├── migrations
    │   ├── db
    │   │   ├── migrations
    ├── codegen_template
    ├── docs
    ├── locale
    ├── requirements
    ├── scripts
    ├── settings
    └── swagger_server
        ├── controllers
        ├── functions
        ├── models
        ├── openapi
        ├── repositories
        ├── serializers
        ├── stories
        └── tests

**apps** is a directory containing our django apps

**apps/boilerplace** a django app containing our custom user model

**apps/db** as we are using django ORM with connexion this app contains all project related db models

**codegen_template** the template we are using for code generation

**docs** the place where our api first specification is

**locale** the i18n po and mo files folder

**requirements** our project python requirements files

**scripts** this directory contains curstom scripts for convenience

**swagger_server** the root directory of connexion framework

**swagger_server/controllers** where connexion (flask) controller belongs

**swagger_server/functions** Where all utils functions belongs

**swagger_server/models** Where generated view models (api related models, not db models) belongs

**swagger_server/openapi** Where generated version of specification belong (with various things added like operationId... etc.)

**swagger_server/repositories** A layer between django models and our business code to help us switching from django ORM to anything else later.

**swagger_server/serializers** This is a convenient layer to generate a view model from a django model when fields are the same in both of them.

**swagger_server/stories** Where business transaction belongs (using dry-python/stories https://github.com/dry-python/stories)

**swagger_server/tests** Where tests and test framework fixtures configuration belongs

All of those layers are injected inside a "container" class using dependencies python library (https://github.com/dry-python/dependencies)

Those containers are defined inside the implemented.py file.
Then, those container classes can be used inside controllers.

## What is included ?
* Full features python test framework (with coverage, html report and testing on various python verions)
* Lint tools (flake8 and isort)
* Performance analysis tool (py-spy running on top of the test framework)
* i18n
* Code duplication analyser
* Dependency injection and business transaction the awesome dry-python's stories and dependencies python libraries
* Swagger ui
* Generate db er-diagram
* Other convinient scripts...

## Requirements
* yarn
* python3.6+
* sqlite
* pew
* pmd (https://pmd.github.io/)

### Before first run, execute those commands
* Create new python virtualenv for project deps
```bash
pew new --python=python3.7 meetup121219
```
* Install python dependencies in virtualenv
```
    pew in meetup121219 pip install -r requirements/requirements.txt
```
* Install node dependencies (for codegem)
```bash
yarn
```
* Migrate database
```bash
npm run db:migrate
```

## Use this project without jwt key for test purpose
If your code is running in debug we have implemented fake users in fake_user_if_fake_jwt_token function in swagger_server/functions/auth.py file.

Examples of token payload can be found inside this method.

In swagger ui or using an http client those values can be used as fake users :
- super_user
- staff_user
- standard_user
- unverified_standard_user
- blocked_standard_user

## How to... ?

As you will see this project use npm as a task runner to make various tasks available.

### Start server
```npm run dev```

### Lint code
```npm run flake8```
Find pep8 violations & other things.

### Reformat imports automatically
```npm run isort```

### Add python coding utf8 everywhere it is missing
```npm run fix:coding```

### Fix endl everywhere it is needed
```npm run fix:endl```

### Find duplicated code
```npm run cpd```
Needs $PMD_BIN to be set correctly (bin directory of pmd where run.sh script is).
Generates report called cpd_report.html

### Generate view models, dummy controllers & spec to be used to run the server (the same spec but with operationId & things like this...)
```npm run generate```
Then:
  - Copy generated/swagger_server/models/*.py to swagger_server/models/
  - Copy new controller functions to the right corresponding file
  - Forget everything else. (it is mostly useless)

Remark:
  We have customize the code-generation template to wrap controllers with a login_required wrapper.
  This wrapper extract user info from the token to create a user object
  to be used by django guardian or other django apps.

### Use swagger ui
Go to 127.0.0.1:5000/v1/ui

### Run tests
```npm run test```
Will run only last failed test by default.

```npm run test:all```
Will run all existing tests from tests directory (swagger_server/tests)

```npm run test:tox```
Will run all tests on python3.6 and python3.7

```npm run test:cdb```
'cdp' means create db.
Will re-create test db before running tests.

```npm run test:vv```
Run pytest in very verbose mode.

```npm run test:withprof```
Run pytest with py-spy to generate a flamegraph

```npm run test:nocov```
Run pytest and do not generat a coverage report.

### Find coverage info
After running `npm run test:all`, coverage report can be found in *htmlcov* directory (firefox index.html).

### Extract i18n strings from code
```npm run i18n:extract```
Extract strings to *locale* directory.

### Translate strings
Open django.po files from locale directory with poedit (https://poedit.net/)

### Make django or flask using translated strings
```npm run i18n:build```
Generates *.mo files to be used by django for translations.

### Grenerate db er-diagram
```npm run db:erdiagram```
Generates er_diagram.png file.

 readmeEtag: '"a05856d353953a5f9d21ff4fe26118cd59b0b6aa"' readmeLastModified: Wed, 21 Apr 2021 18:41:49 GMT repositoryId: 227149750 description: >- Tools to write clean microservices in python without headaches. (demo project) created: '2019-12-10T15:07:35Z' updated: '2025-10-08T00:55:25Z' language: Python archived: true stars: 4 watchers: 0 forks: 0 owner: Alma-CFAO logo: https://avatars.githubusercontent.com/u/54944051?v=4 license: MIT repoEtag: '"856e4260d2a04e1b4612150cca182e309112ff0e252a986788d9f19b0430dff0"' repoLastModified: Wed, 08 Oct 2025 00:55:25 GMT foundInMaster: true category: Server Implementations id: b136a8f4c9ec91fb8287c2b81073bdc1 - source: openapi3 tags repository: https://github.com/medly/apifi v3: true repositoryMetadata: base64Readme: >- IyBBcGlmaSAKClNwZWMgZHJpdmVuIFJFU1QgQVBJcwoKIVtCdWlsZCB0aGUgRGlzdHJpYnV0aW9uXShodHRwczovL2dpdGh1Yi5jb20vbWVkbHkvYXBpZmkvd29ya2Zsb3dzL0J1aWxkJTIwdGhlJTIwRGlzdHJpYnV0aW9uL2JhZGdlLnN2ZykKCiMjIEluY2x1ZGUgaW4gZ3JhZGxlIHByb2plY3QKMVwuIEFkZCBqaXRwYWNrIHJlcG9zaXRvcnkgaW4gc2V0dGluZ3MuZ3JhZGxlOgpgYGAKcGx1Z2luTWFuYWdlbWVudCB7CiAgICByZXBvc2l0b3JpZXMgewogICAgICAgIGdyYWRsZVBsdWdpblBvcnRhbCgpCiAgICAgICAgbWF2ZW4gewogICAgICAgICAgICB1cmwgJ2h0dHBzOi8vaml0cGFjay5pbycKICAgICAgICAgICAgY29udGVudCB7CiAgICAgICAgICAgICAgICBpbmNsdWRlR3JvdXAgJ2NvbS5tZWRseS5hcGlmaScKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KICAgIHJlc29sdXRpb25TdHJhdGVneSB7CiAgICAgICAgZWFjaFBsdWdpbiB7CiAgICAgICAgICAgIGlmIChyZXF1ZXN0ZWQuaWQubmFtZXNwYWNlID09ICdjb20ubWVkbHknKSB7CiAgICAgICAgICAgICAgICB1c2VNb2R1bGUoIiR7cmVxdWVzdGVkLmlkfToke3JlcXVlc3RlZC5pZH0uZ3JhZGxlLnBsdWdpbjphYjgxNmQxYTg0IikKICAgICAgICAgICAgfQogICAgICAgIH0KICAgIH0KfQpgYGAKMlwuIEFkZCBwbHVnaW4gaW4gYnVpbGQuZ3JhZGxlOgpgYGAKcGx1Z2lucyB7CiAgICAuLi4uCiAgICBpZCAiY29tLm1lZGx5LmFwaWZpIgogICAgLi4uCn0KYGBgCgozXC4gQWRkIGdyYWRsZSB0YXNrIHRvIGdlbmVyYXRlIGNvZGU6CmBgYAphcGlmaSB7CiAgICBvcGVuQXBpU3BlYyBmaWxlKCJwYXRoL3RvL3NwZWMtZmlsZSIpCiAgICBnZW5lcmF0ZWRTb3VyY2VEaXIgZmlsZSgicGF0aC90by9nZW4iKQogICAgYmFzZVBhY2thZ2VOYW1lICJjb20ueW91ci5wYWNrYWdlLm5hbWUiCn0KYGBgCgojIyBHZW5lcmF0ZSBBUElzCgpgYGAKLi9ncmFkbGV3IGFwaWZpCmBgYA== readmeEtag: '"46f042d44f4f7166b67848685ee99b7ad3f4735f"' readmeLastModified: Fri, 18 Sep 2020 12:25:39 GMT repositoryId: 259544047 description: Open API spec driven HTTP APIs created: '2020-04-28T05:50:27Z' updated: '2021-10-01T11:34:56Z' language: Kotlin archived: false stars: 4 watchers: 3 forks: 5 owner: medly logo: https://avatars.githubusercontent.com/u/54950577?v=4 repoEtag: '"2aa31c588ffd4204be01a6e0feeb896aedc7232863a9a7a6852c410e7348335d"' repoLastModified: Fri, 01 Oct 2021 11:34:56 GMT foundInMaster: true category: SDK id: 5f1612e5badbd0eab91bebe51e68ae1b - source: openapi3 tags repository: https://github.com/fazzani/synker v3: true repositoryMetadata: base64Readme: >- IyAhW1N5bmtlciBBUEldKGxvZ28ucG5nKQoKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2Rldi5henVyZS5jb20vaGVuaWZhenphbmkvU3lua2VyQVBJL19hcGlzL2J1aWxkL3N0YXR1cy9TeW5rZXJBUEktQ0k/YnJhbmNoTmFtZT1tYXN0ZXIpXShodHRwczovL2Rldi5henVyZS5jb20vaGVuaWZhenphbmkvU3lua2VyQVBJL19idWlsZC9sYXRlc3Q/ZGVmaW5pdGlvbklkPTEyJmJyYW5jaE5hbWU9bWFzdGVyKQpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vZGV2LmF6dXJlLmNvbS9oZW5pZmF6emFuaS9TeW5rZXJBUEkvX2FwaXMvYnVpbGQvc3RhdHVzL1N5bmtlckFQSS1Eb2NrZXIlMjBDST9icmFuY2hOYW1lPW1hc3RlcildKGh0dHBzOi8vZGV2LmF6dXJlLmNvbS9oZW5pZmF6emFuaS9TeW5rZXJBUEkvX2J1aWxkL2xhdGVzdD9kZWZpbml0aW9uSWQ9MTQmYnJhbmNoTmFtZT1tYXN0ZXIpCgpbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1GYXp6YW5pX1N5bmtlciZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1GYXp6YW5pX1N5bmtlcikKWyFbQ292ZXJhZ2VdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PUZhenphbmlfU3lua2VyJm1ldHJpYz1jb3ZlcmFnZSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9RmF6emFuaV9TeW5rZXIpClshW1JlbGlhYmlsaXR5IFJhdGluZ10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9RmF6emFuaV9TeW5rZXImbWV0cmljPXJlbGlhYmlsaXR5X3JhdGluZyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9RmF6emFuaV9TeW5rZXIpClshW1Z1bG5lcmFiaWxpdGllc10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9RmF6emFuaV9TeW5rZXImbWV0cmljPXZ1bG5lcmFiaWxpdGllcyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9RmF6emFuaV9TeW5rZXIpClshW0NvZGUgU21lbGxzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1GYXp6YW5pX1N5bmtlciZtZXRyaWM9Y29kZV9zbWVsbHMpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPUZhenphbmlfU3lua2VyKQoKIyMgQXJjaGl0ZWN0dXJlIChDbGVhbiBhcmNoaXRlY3R1cmUpCgpJbnNwaXJlZCBmcm9tIFtOb3J0aHdpbmRUcmFkZXJzXVtOb3J0aHdpbmRUcmFkZXJzXQpbU2xpZGVzXShodHRwczovL2dpdGh1Yi5jb20vRmF6emFuaS9TeW5rZXIvcmF3L21hc3Rlci9kb2NzL1NsaWRlcy5wZGYpCgojIyMgU2NoZW1hCgohW2FyY2hpX2NsZWFuX2RpYWdyYW1dKGh0dHBzOi8vZnVsbHN0YWNrbWFyay5jb20vaW1nL3Bvc3RzLzExL2NsZWFuLWFyY2hpdGVjdHVyZS1jaXJjbGUtZGlhZ3JhbS5qcGcpCgojIyMgTWljcm8tc2VydmljZXMgc2V0CgotIFBsYXlsaXN0cyBzZXJ2aWNlCiAgLSBDUlVECiAgLSBTeW5jaHJvIChieSBwbGF5bGlzdCAmJiBieSBncm91cCkKICAtIERlY29yYXRvcnMKICAgIC0gQ2xlYW5pbmcgbmFtZXMKICAgIC0gQXV0byBncm91cGluZwogICAgLSBTaGlmdCB0aW1lCiAgICAtIEN1c3RvbSAmJiBkeW5hbWljCi0gRGF0YVNvdXJjZXMKLSBFUEcgc2VydmljZQotIFBpY29ucyBzZXJ2aWNlCi0gQXV0aGVudGljYXRpb24gU2VydmljZSAoSURQKQotIE5vdGlmaWNhdGlvbiBTZXJ2aWNlCgojIyBIb3cgaXQgd29ya3MKClRoaXMgaXMgdXNpbmcgQVNQLk5FVCBDb3JlIHdpdGg6CgotIEVudGl0eSBGcmFtZXdvcmsgQ29yZSBvbiBQb3N0Z3JlU1FMIGFuZCBJbk1lbW9yeSBmb3IgdGVzdGluZwotIE9QRU4gQVBJIDMgd2l0aCBbTlN3YWddW25zd2FnX3JlcG9dCi0gRXZlbnQgc291cmNpbmcKLSBbTWVkaWF0Ul0oaHR0cHM6Ly9naXRodWIuY29tL2pib2dhcmQvTWVkaWF0Ui93aWtpKQotIFRERDoKICAtIFh1bml0LCAKICAtIFthbmdsZXNoYXJwXVthbmdsZXNoYXJwXSAKICAtIFtNb3FdW21vcV9yZXBvXQogIC0gZTJlIHdpdGggW0N5cHJlc3NdW2N5cHJlc3NdCi0gUHJvbWV0aGV1cyBtZXRyaWNzCi0gW0JlYXQgUHVsc2VdW2JlYXRfcHVsc2VfZ2l0aHViXSBmb3IgbGl2ZW5lc3MgYW5kIHJlYWRpbmVzcyBjaGVjawogIC0gQmVhdFB1bHNlLklkU3ZyCiAgLSBCZWF0UHVsc2UuRWxhc3RpY3NlYXJjaAogIC0gQmVhdFB1bHNlLk5wZ3NxbAotIFNlcmlsb2cgZm9yIGxvZ2dpbmcKLSBbQXV0b01hcHBlcl0oaHR0cHM6Ly9hdXRvbWFwcGVyLm9yZy8pCi0gW0ZsdWVudHZhbGlkYXRpb25dKGh0dHBzOi8vZmx1ZW50dmFsaWRhdGlvbi5uZXQvc3RhcnQpCi0gW0NvbW1hbmQgQ1FSUyB2YWxpZGF0aW9uXShodHRwczovL3d3dy5saW5rZWRpbi5jb20vcHVsc2UvdmFsaWRhdGlvbi1kZGQtY3Fycy1sdWNhLWJyaWd1Z2xpYS8pCi0gW0RERCBBZ2dyZWdhdGUgcGF0dGVybl0oaHR0cHM6Ly9tYXJ0aW5mb3dsZXIuY29tL2JsaWtpL0RERF9BZ2dyZWdhdGUuaHRtbCkKCiMjIENsZWFuIEFyY2hpdGVjdHVyZQoKIyMjIERlc2lnbmluZyBEb21haW4gTW9kZWwgTGF5ZXIKCjEuICoqUGVyc2lzdGVuY2UgSWdub3JhbmNlIChQSSkgcHJpbmNpcGxlKiogc2F5cyB0aGF0IHRoZSBEb21haW4gTW9kZWwgc2hvdWxkIGJlIGlnbm9yYW50IG9mIGhvdyBpdHMgZGF0YSBpcyBzYXZlZCBvciByZXRyaWV2ZWQ6CiAgIC0gTm8gZGF0YSBhY2Nlc3MgY29kZQogICAtIE5vIGRhdGEgYW5ub3RhdGlvbnMgZm9yIG91ciBlbnRpdGllcwogICAtIE5vIGluaGVyaXRhbmNlIGZyb20gYW55IGZyYW1ld29yayBjbGFzc2VzLCBlbnRpdGllcyBzaG91bGQgYmUgUGxhaW4gT2xkIENMUiBPYmplY3QKMi4gUHV0dGluZyBhbGwgZGF0YSBhY2Nlc3MgY29kZSBvdXRzaWRlIG91ciBkb21haW4gbW9kZWwgaW1wbGVtZW50YXRpb24KMy4gVXNpbmcgRW50aXR5IEZyYW1ld29yayBDb3JlIGZlYXR1cmVzOiAKICAgLSBTaGFkb3cgUHJvcGVydGllcwogICAtIE93bmVkIEVudGl0eSBUeXBlcwogICAtIFByaXZhdGUgZmllbGRzIG1hcHBpbmcKICAgLSBWYWx1ZSBDb252ZXJzaW9ucwoKIyMgR28gZnVydGhlciAmIHJlZmVyZW5jZXMKCi0gW0NsZWFuIEFyY2hpdGVjdHVyZV1bY2xlYW5fYXJjaGldCi0gW09uaW9uIEFyY2hpdGVjdHVyZV1bb25pb25fYXJjaGldCi0gW1BvcnRzIEFuZCBBZGFwdGVycyAvIEhleGFnb25hbCBBcmNoaXRlY3R1cmVdW3BvcnRfYWRhcHRlcl9oZXhhXQotIFtFeGFtcGxlIENsZWFuIEFyY2hpIDFdW2V4YW1wbGVfYXJjaGlfMV0KLSBbRXhhbXBsZSBDbGVhbiBBcmNoaSAyXVtleGFtcGxlX2FyY2hpXzJdCgpbYmVhdF9wdWxzZV9naXRodWJdOmh0dHBzOi8vZ2l0aHViLmNvbS9YYWJhcmlsL0JlYXRQdWxzZQpbTm9ydGh3aW5kVHJhZGVyc106aHR0cHM6Ly9naXRodWIuY29tL0phc29uR1QvTm9ydGh3aW5kVHJhZGVycwpbUGVyc2lzdGVuY2UgSWdub3JhbmNlXTpodHRwOi8vd3d3LmthbWlsZ3J6eWJlay5jb20vZGVzaWduL2RvbWFpbi1tb2RlbC1lbmNhcHN1bGF0aW9uLWFuZC1waS13aXRoLWVudGl0eS1mcmFtZXdvcmstMi0yLwpbbnN3YWdfcmVwb106aHR0cHM6Ly9naXRodWIuY29tL1JpY29TdXRlci9OU3dhZwpbbW9xX3JlcG9dOmh0dHBzOi8vZ2l0aHViLmNvbS9tb3EvbW9xNApbYW5nbGVzaGFycF06aHR0cHM6Ly9hbmdsZXNoYXJwLmdpdGh1Yi5pby8KW2NsZWFuX2FyY2hpXTpodHRwOi8vYmxvZy5jbGVhbmNvZGVyLmNvbS91bmNsZS1ib2IvMjAxMi8wOC8xMy90aGUtY2xlYW4tYXJjaGl0ZWN0dXJlLmh0bWwKW29uaW9uX2FyY2hpXTpodHRwczovL2plZmZyZXlwYWxlcm1vLmNvbS8yMDA4LzA3L3RoZS1vbmlvbi1hcmNoaXRlY3R1cmUtcGFydC0xCltwb3J0X2FkYXB0ZXJfaGV4YV06aHR0cHM6Ly9oZXJiZXJ0b2dyYWNhLmNvbS8yMDE3LzA5LzE0L3BvcnRzLWFkYXB0ZXJzLWFyY2hpdGVjdHVyZS8KW2V4YW1wbGVfYXJjaGlfMV06aHR0cHM6Ly9mdWxsc3RhY2ttYXJrLmNvbS9wb3N0LzExL2JldHRlci1zb2Z0d2FyZS1kZXNpZ24td2l0aC1jbGVhbi1hcmNoaXRlY3R1cmUKW2V4YW1wbGVfYXJjaGlfMl06aHR0cHM6Ly9mdWxsc3RhY2ttYXJrLmNvbS9wb3N0LzE4L2J1aWxkaW5nLWFzcG5ldC1jb3JlLXdlYi1hcGlzLXdpdGgtY2xlYW4tYXJjaGl0ZWN0dXJlCltjeXByZXNzXTpodHRwczovL3d3dy5jeXByZXNzLmlvLw== readmeEtag: '"6d5ad0ca66b9a3cc96d05b217ae03df4c2e7e4b9"' readmeLastModified: Wed, 28 Oct 2020 15:47:24 GMT repositoryId: 185035728 description: Synker Clean architecture created: '2019-05-05T13:34:51Z' updated: '2020-10-28T15:47:27Z' language: C# archived: false stars: 4 watchers: 1 forks: 0 owner: Fazzani logo: https://avatars.githubusercontent.com/u/2669070?v=4 repoEtag: '"0204670e71a36bc10e05d8019b0c1790389515641e8f307d862e85f4e0b4ec1f"' repoLastModified: Wed, 28 Oct 2020 15:47:27 GMT foundInMaster: true category: Server id: 24f3174a585318a2e0fc37f0502c0de9 - source: openapi3 tags repository: https://github.com/alexferl/echo-openapi v3: true id: ce1a77b9cc878c17a7dde009e1eecfdd repositoryMetadata: base64Readme: >- IyBlY2hvLW9wZW5hcGkgWyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL2FsZXhmZXJsL2VjaG8tb3BlbmFwaSldKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9hbGV4ZmVybC9lY2hvLW9wZW5hcGkpIFshW2NvZGVjb3ZdKGh0dHBzOi8vY29kZWNvdi5pby9naC9hbGV4ZmVybC9lY2hvLW9wZW5hcGkvYnJhbmNoL21hc3Rlci9ncmFwaC9iYWRnZS5zdmcpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvYWxleGZlcmwvZWNoby1vcGVuYXBpKQoKQW4gW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIG1pZGRsZXdhcmUgZm9yIHRoZSBbRWNob10oaHR0cHM6Ly9naXRodWIuY29tL2xhYnN0YWNrL2VjaG8pIGZyYW1ld29yayB1c2luZwpbZ2V0a2luL2tpbi1vcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vZ2V0a2luL2tpbi1vcGVuYXBpKSB0byB2YWxpZGF0ZSBIVFRQIHJlcXVlc3RzIGFuZCByZXNwb25zZXMuCgojIyBJbnN0YWxsaW5nCmBgYHNoZWxsCmdvIGdldCBnaXRodWIuY29tL2FsZXhmZXJsL2VjaG8tb3BlbmFwaQpgYGAKCiMjIFVzaW5nCgojIyMgQ29kZSBleGFtcGxlCmBgYGdvCnBhY2thZ2UgbWFpbgoKaW1wb3J0ICgKICAgICJuZXQvaHR0cCIKCiAgICBtdyAiZ2l0aHViLmNvbS9hbGV4ZmVybC9lY2hvLW9wZW5hcGkiCiAgICAiZ2l0aHViLmNvbS9sYWJzdGFjay9lY2hvL3Y0IgogICAgImdpdGh1Yi5jb20vbGFic3RhY2svZWNoby92NC9taWRkbGV3YXJlIgopCgovKgojIG9wZW5hcGkueWFtbApvcGVuYXBpOiAzLjAuNAppbmZvOgogIHZlcnNpb246IDEuMC4wCiAgdGl0bGU6IFRlc3QgQVBJCiAgZGVzY3JpcHRpb246IEEgdGVzdCBBUEkKcGF0aHM6CiAgL2hlbGxvOgogICAgcG9zdDoKICAgICAgZGVzY3JpcHRpb246IEhlbGxvCiAgICAgIHBhcmFtZXRlcnM6CiAgICAgICAgLSBuYW1lOiBtZXNzYWdlCiAgICAgICAgICBpbjogcXVlcnkKICAgICAgICAgIHJlcXVpcmVkOiB0cnVlCiAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgICBtaW5MZW5ndGg6IDEKICAgICAgICAgICAgbWF4TGVuZ3RoOiAxMDAKICAgICAgcmVzcG9uc2VzOgogICAgICAgICcyMDAnOgogICAgICAgICAgZGVzY3JpcHRpb246IFN1Y2Nlc3NmdWwgcmVzcG9uc2UKICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgICAgdHlwZTogb2JqZWN0CiAgICAgICAgICAgICAgICBhZGRpdGlvbmFsUHJvcGVydGllczogZmFsc2UKICAgICAgICAgICAgICAgIHJlcXVpcmVkOgogICAgICAgICAgICAgICAgICAtIG1lc3NhZ2UKICAgICAgICAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgICAgICAgIG1lc3NhZ2U6CiAgICAgICAgICAgICAgICAgICAgdHlwZTogc3RyaW5nCiAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb246IFdlbGNvbWUgbWVzc2FnZQogICAgICAgICAgICAgICAgICAgIG1pbkxlbmd0aDogNAoqLwoKdHlwZSBIYW5kbGVyIHN0cnVjdCB7CiAgICAqbXcuSGFuZGxlcgp9CgpmdW5jIChoICpIYW5kbGVyKSBIZWxsbyhjIGVjaG8uQ29udGV4dCkgZXJyb3IgewogICAgbXNnIDo9IGMuUXVlcnlQYXJhbSgibWVzc2FnZSIpCiAgICByZXR1cm4gaC5WYWxpZGF0ZShjLCBodHRwLlN0YXR1c09LLCBlY2hvLk1hcHsibWVzc2FnZSI6IG1zZ30pCn0KCmZ1bmMgbWFpbigpIHsKICAgIGUgOj0gZWNoby5OZXcoKQoKICAgIGggOj0gJkhhbmRsZXJ7bXcuTmV3SGFuZGxlcigpfQogICAgZS5BZGQoaHR0cC5NZXRob2RQb3N0LCAiL2hlbGxvIiwgaC5IZWxsbykKCiAgICBlLlVzZShtaWRkbGV3YXJlLkxvZ2dlcigpKQogICAgZS5Vc2UobXcuT3BlbkFQSSgiLi9vcGVuYXBpLnlhbWwiKSkKCiAgICBlLkxvZ2dlci5GYXRhbChlLlN0YXJ0KCJsb2NhbGhvc3Q6MTMyMyIpKQp9CmBgYApTZW5kIGFuIGludmFsaWQgcmVxdWVzdCB0byB0ZXN0IHJlcXVlc3QgdmFsaWRhdGlvbjoKYGBgc2hlbGwKY3VybCAtaSAtWCBQT1NUIGh0dHA6Ly9sb2NhbGhvc3Q6MTMyMy9oZWxsbwpIVFRQLzEuMSA0MjIgVW5wcm9jZXNzYWJsZSBFbnRpdHkKQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uOyBjaGFyc2V0PVVURi04CkRhdGU6IFNhdCwgMTIgTm92IDIwMjIgMTc6MzE6MjQgR01UCkNvbnRlbnQtTGVuZ3RoOiAxMTcKCnsibWVzc2FnZSI6IlZhbGlkYXRpb24gZXJyb3IiLCJlcnJvcnMiOlsicGFyYW1ldGVyICdtZXNzYWdlJyBpbiBxdWVyeSBoYXMgYW4gZXJyb3I6IHZhbHVlIGlzIHJlcXVpcmVkIGJ1dCBtaXNzaW5nIl19CmBgYAoKU2VuZCBhIHZhbGlkIHJlcXVlc3Q6CmBgYHNoZWxsCmN1cmwgLWkgLVggUE9TVCBodHRwOi8vbG9jYWxob3N0OjEzMjMvaGVsbG9cP21lc3NhZ2VcPWhlbGxvCkhUVFAvMS4xIDIwMCBPSwpDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24KRGF0ZTogU2F0LCAxMiBOb3YgMjAyMiAxNzozMTo0NyBHTVQKQ29udGVudC1MZW5ndGg6IDE5Cgp7Im1lc3NhZ2UiOiJoZWxsbyJ9CmBgYAoKU2VuZCBhIHZhbGlkIHJlcXVlc3Qgd2l0aCBhbiBpbnZhbGlkIHJlc3BvbnNlOgpgYGBzaGVsbApjdXJsIC1pIC1YIFBPU1QgaHR0cDovL2xvY2FsaG9zdDoxMzIzL2hlbGxvXD9tZXNzYWdlXD1hCkhUVFAvMS4xIDUwMCBJbnRlcm5hbCBTZXJ2ZXIgRXJyb3IKQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uCkRhdGU6IFNhdCwgMTIgTm92IDIwMjIgMTc6MzE6MDEgR01UCkNvbnRlbnQtTGVuZ3RoOiAzNgoKeyJtZXNzYWdlIjoiSW50ZXJuYWwgU2VydmVyIEVycm9yIn0KYGBgCllvdSBzaG91bGQgYWxzbyBoYXZlIHRoZSBmb2xsb3dpbmcgaW4gdGhlIHNlcnZlcidzIGxvZyB0byBoZWxwIHlvdSBkZWJ1ZyB5b3VyIHNjaGVtYToKYGBgc2hlbGwKeyJlcnJvciI6ImZhaWxlZCB2YWxpZGF0aW5nIHJlc3BvbnNlOiBtZXNzYWdlOiBtaW5pbXVtIHN0cmluZyBsZW5ndGggaXMgNCJ9CmBgYAoKIyMjIENvbmZpZ3VyYXRpb24KYGBgZ28KdHlwZSBDb25maWcgc3RydWN0IHsKICAgIC8vIFNraXBwZXIgZGVmaW5lcyBhIGZ1bmN0aW9uIHRvIHNraXAgbWlkZGxld2FyZS4KICAgIFNraXBwZXIgbWlkZGxld2FyZS5Ta2lwcGVyCgogICAgLy8gU2NoZW1hIGRlZmluZXMgdGhlIE9wZW5BUEkgdGhhdCB3aWxsIGJlIGxvYWRlZCBhbmQKICAgIC8vIHRoYXQgdGhlIHJlcXVlc3QgYW5kIHJlc3BvbnNlcyB3aWxsIGJlIHZhbGlkYXRlZCBhZ2FpbnN0LgogICAgLy8gUmVxdWlyZWQuCiAgICBTY2hlbWEgc3RyaW5nCgogICAgLy8gQ29udGV4dEtleSBkZWZpbmVzIHRoZSBrZXkgdGhhdCB3aWxsIGJlIHVzZWQgdG8gc3RvcmUgdGhlIHZhbGlkYXRvcgogICAgLy8gb24gdGhlIGVjaG8uQ29udGV4dCB3aGVuIHRoZSByZXF1ZXN0IGlzIHN1Y2Nlc3NmdWxseSB2YWxpZGF0ZWQuCiAgICAvLyBPcHRpb25hbC4gRGVmYXVsdHMgdG8gInZhbGlkYXRvciIuCiAgICBDb250ZXh0S2V5IHN0cmluZwoKICAgIC8vIEV4ZW1wdFJvdXRlcyBkZWZpbmVzIHJvdXRlcyBhbmQgbWV0aG9kcyB0aGF0IGRvbid0IHJlcXVpcmUgdG9rZW5zLgogICAgLy8gT3B0aW9uYWwuCiAgICBFeGVtcHRSb3V0ZXMgbWFwW3N0cmluZ11bXXN0cmluZwp9Cgp0eXBlIEhhbmRsZXJDb25maWcgc3RydWN0IHsKICAgIC8vIENvbnRlbnRUeXBlIHNldHMgdGhlIENvbnRlbnQtVHlwZSBoZWFkZXIgb2YgdGhlIHJlc3BvbnNlLgogICAgLy8gT3B0aW9uYWwuIERlZmF1bHRzIHRvICJhcHBsaWNhdGlvbi9qc29uIi4KICAgIENvbnRlbnRUeXBlIHN0cmluZwoKICAgIC8vIFZhbGlkYXRvcktleSBkZWZpbmVzIHRoZSBrZXkgdGhhdCB3aWxsIGJlIHVzZWQgdG8gcmVhZCB0aGUKICAgIC8vICpvcGVuYXBpM2ZpbHRlci5SZXF1ZXN0VmFsaWRhdGlvbklucHV0IGZyb20gdGhlIGVjaG8uQ29udGV4dAogICAgLy8gc2V0IGJ5IHRoZSBtaWRkbGV3YXJlLgogICAgLy8gT3B0aW9uYWwuIERlZmF1bHRzIHRvICJ2YWxpZGF0b3IiLgogICAgVmFsaWRhdG9yS2V5IHN0cmluZwoKICAgIC8vIEV4Y2x1ZGVSZXF1ZXN0Qm9keSBtYWtlcyBWYWxpZGF0ZSBza2lwcyByZXF1ZXN0IGJvZHkgdmFsaWRhdGlvbi4KICAgIC8vIE9wdGlvbmFsLiBEZWZhdWx0cyB0byBmYWxzZS4KICAgIEV4Y2x1ZGVSZXF1ZXN0Qm9keSBib29sCgogICAgLy8gRXhjbHVkZVJlc3BvbnNlQm9keSBtYWtlcyBWYWxpZGF0ZSBza2lwcyByZXNwb25zZSBib2R5IHZhbGlkYXRpb24uCiAgICAvLyBPcHRpb25hbC4gRGVmYXVsdHMgdG8gZmFsc2UuCiAgICBFeGNsdWRlUmVzcG9uc2VCb2R5IGJvb2wKCiAgICAvLyBJbmNsdWRlUmVzcG9uc2VTdGF0dXMgbWFrZXMgVmFsaWRhdGUgZmFpbCBvbiByZXNwb25zZQogICAgLy8gc3RhdHVzZXMgbm90IGRlZmluZWQgaW4gdGhlIE9wZW5BUEkgc3BlYy4KICAgIC8vIE9wdGlvbmFsLiBEZWZhdWx0cyB0byB0cnVlLgogICAgSW5jbHVkZVJlc3BvbnNlU3RhdHVzIGJvb2wKfQpgYGAK readmeEtag: '"31ac2ac48f8c5aa9e948876a85de3bfa94d8b608"' readmeLastModified: Tue, 06 Feb 2024 20:18:00 GMT repositoryId: 557002965 description: OpenAPI middleware for the Echo framework created: '2022-10-24T23:08:22Z' updated: '2024-10-02T14:58:39Z' language: Go archived: false stars: 4 watchers: 2 forks: 2 owner: alexferl logo: https://avatars.githubusercontent.com/u/3533424?v=4 license: MIT repoEtag: '"b9fa92f8547979b03b971931995b413e4bd96933482945068d4eaa65f27be1b1"' repoLastModified: Wed, 02 Oct 2024 14:58:39 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/kulgan/flaskdoc v3: true repositoryMetadata: base64Readme: >- flaskdoc
========

|PyPi version| |Python version| |ci| |docs| |license| |coverage| |code quality|

FlaskDoc allows developers to programmatically compose openapi specifications for flask endpoints as a part of code
without needing to write a separate yaml file, and it comes with SwaggerUI embedded. Its main focus is on documentation
which frees developers to focus on getting their services coded.

Why flaskdoc
------------

* Focus only on documentation and not introduce some fancy new way of using flask.
* Easily add to existing code without needing to refactor of change the way the code has been written
* Little or no learning curve, as long as a developer is comforatble using flask developers, they can use flaskdoc.
  to learn quickly and not distract So developers focus on writing code
* SwaggerUI integration for quickly testing and iterating through versions
* Automatic data model to JSON Schema transformation that allows for finer grain configuration


Getting Started
---------------
Visit `documentation <https://flaskdoc.readthedocs.io>`_ page for more details.

Install
"""""""
from pypi

.. code-block::

    $ pip install flaskdoc

from github

.. code-block::

    $ pip install https://github.com/kulgan/flaskdoc/tarball/master

To run examples you will need to install the dev extension

.. code-block::

    $ pip install flaskdoc[dev]

Register OpenAPI
""""""""""""""""
Add top level openapi objects like `Info <https://swagger.io/specification/#info-object>`_,
`Contact <https://swagger.io/specification/#contact-object>`_, `License <https://swagger.io/specification/#license-object>`_ etc

.. code-block:: python

    import flask
    from flaskdoc import register_openapi, swagger

    app = flask.Flask()
    # initialize app, add all the blueprints you care about

    # Create top level OpenAPI objects
    # the info object
    info = swagger.Info(
        title="Test",
        version="1.2.2",
        contact=swagger.Contact(
            name="Rowland", email="r.ogwara@gmail.com", url="https://github.com/kulgan"
        ),
        license=swagger.License(name="Apache 2.0", url="https://www.example.com/license"),
    )

    # servers names and variables if necessary
    servers = [swagger.Server(url="http://localhost:15172")]

    # top level tags
    tags = [
        swagger.Tag(name="admin", description="Secured Admin-Only calls"),
        swagger.Tag(name="developers", description="Operations available to regular developers"),
    ]

    security_schemes = {
        "api_key": swagger.ApiKeySecurityScheme(name="api_key"),
    }

    # register spec
    register_openapi(app, info=info, servers=servers, tags=tags, security=security_schemes)

This adds the following endpoints to your list

* /docs
* /docs/openapi.yaml
* /docs/openapi.json

Start Documenting
"""""""""""""""""
Now start documenting you flask routes

A simple post example

.. code-block:: python

    blp = flask.Blueprint("Dummy", __name__, url_prefix="/v1")
    @swagger.POST(
        tags=["administrator"],
        description="Posts an Echo",
        responses={"201": swagger.ResponseObject(description="OK")},
    )
    @blp.route("/echo", methods=["POST"])
    def post():
        req = flask.request.get_json(force=True)
        return flask.jsonify(req), 200

A GET example with path parameter

.. code-block:: python

    blp = flask.Blueprint("Dummy", __name__, url_prefix="/v1")

    @swagger.GET(
        tags=["getEcho"],
        operation_id="getEcho",
        parameters=[swagger.PathParameter(name="sample", schema=str)],
        description="Retrieve echos wit Get",
        responses={
            "200": swagger.ResponseObject(
                description="Success", content=jo.PlainText(schema=jo.Email()),
            )
        },
    )
    @blp.route("/echo/<string:sample>", methods=["GET"])
    def echo(sample: str):
        """
        Sample GET request
        Returns: Echos back whatever was sent

        """
        return sample

Run your app and visit `/docs` to see the generated openapi specs

Running Examples
================

Two example projects are currently provided

* `inventory <src/flaskdoc/examples/inventory.py>`_
* `petstore <src/flaskdoc/examples/petstore.py>`_ source `OpenAPI Petstore <https://petstore.swagger.io>`_
* `link-example <src/flaskdoc/examples/link_example/v0.py>`_ - source `OpenAPI link example <https://github.com/OAI/OpenAPI-Specification/blob/master/examples/v3.0/link-example.json>`_
* `api-with-example <src/flaskdoc/examples/api_with_example.py>`_ - source `OpenAPI api_with_examples <https://github.com/OAI/OpenAPI-Specification/blob/master/examples/v3.0/api-with-examples.json>`_

To run

.. code-block:: bash

    $ pip install flaskdoc[dev]
    $ flaskdoc start -n petstore

Contributing
------------

Don't hesitate to create a `Github issue <https://github.com/kulgan/flaskdoc/issues>`__ for any bugs or suggestions

.. |ci| image:: https://github.com/kulgan/flaskdoc/workflows/ci/badge.svg
    :target: https://github.com/kulgan/flaskdoc/
    :alt: build

.. |PyPi version| image:: https://img.shields.io/pypi/v/flaskdoc.svg
    :target: https://pypi.org/project/flaskdoc/
    :alt: PyPi downloads

.. |Python version| image:: https://img.shields.io/pypi/pyversions/flaskdoc.svg
    :target: https://pypi.org/project/flaskdoc/
    :alt: Python versions

.. |license| image:: https://img.shields.io/pypi/l/flaskdoc.svg
    :target: https://pypi.org/project/flaskdoc/
    :alt: license
.. |docs| image:: https://readthedocs.org/projects/flaskdoc/badge/?version=latest
    :target: https://flaskdoc.readthedocs.io/en/latest/?badge=latest
    :alt: Documentation Status

.. |code quality| image:: https://app.codacy.com/project/badge/Grade/2dafebf021354a42aa62b11d6ab42654
    :target: https://www.codacy.com/manual/kulgan/flaskdoc?utm_source=github.com&amp;utm_medium=referral&amp;utm_content=kulgan/flaskdoc&amp;utm_campaign=Badge_Grade
    :alt: Code Quality

.. |coverage| image:: https://app.codacy.com/project/badge/Coverage/2dafebf021354a42aa62b11d6ab42654
    :target: https://www.codacy.com/manual/kulgan/flaskdoc?utm_source=github.com&amp;utm_medium=referral&amp;utm_content=kulgan/flaskdoc&amp;utm_campaign=Badge_Coverage
    :alt: Coverage
 readmeEtag: '"b59128c6ef4fa4d4bada6b3a06403e52075d5ba8"' readmeLastModified: Thu, 10 Jun 2021 22:19:38 GMT repositoryId: 208150427 description: Flask OpenAPI annotations created: '2019-09-12T21:35:27Z' updated: '2022-06-21T18:40:09Z' language: Python archived: false stars: 4 watchers: 0 forks: 1 owner: kulgan logo: https://avatars.githubusercontent.com/u/892017?v=4 license: Apache-2.0 repoEtag: '"d110529b32d24aa98782c4beadc594c17e44db5dc2e44fe06013a14c8e17470a"' repoLastModified: Tue, 21 Jun 2022 18:40:09 GMT foundInMaster: true category: Server Implementations id: db4ac60d7f3ef260b6df2e06d75adb79 - source: openapi3 tags repository: https://github.com/budwk/budwk-openapi-viewer v3: true repositoryMetadata: base64Readme: >- CiMgU3dhZ2dlciBPcGVuQXBpIFYzIFZpZXdlcgpBbiBPcGVuQVBJIDMuMCBhbmQgMi4wIFNwZWMgdmlld2VyCgojIOivtOaYjgoKKiDmnKzpobnnm67lnKggT3BlbkFQSS1WaWV3ZXIg55qE5Z+656GA5LiKLCDkv67lpI3kuIDkupsgT3BlbkFQSSBWMyDlhbzlrrnmgKdidWcg5Li65LqG5pu05aW955qE5bqU55So5LqOIEJ1ZFdrIOmhueebruS4igoqIOWOn+Wni+S7o+eggTogW2h0dHBzOi8vZ2l0aHViLmNvbS9tcmluOS9PcGVuQVBJLVZpZXdlcl0oaHR0cHM6Ly9naXRodWIuY29tL21yaW45L09wZW5BUEktVmlld2VyKQoqIOa8lOekuuWcsOWdgDogW2h0dHBzOi8vZGVtby5idWR3ay5jb20vc3dhZ2dlcl0oaHR0cHM6Ly9kZW1vLmJ1ZHdrLmNvbS9zd2FnZ2VyKQoKIyMg5a6J6KOFCmBgYAp5YXJuIGluc3RhbGwKYGBgCiMjIOiwg+ivlQpgYGAKeWFybiBzZXJ2ZQpgYGAKIyMg55Sf5oiQCmBgYAp5YXJuIHJ1biBidWlsZCAKYGBg readmeEtag: '"fe6ceb67be156b006a274bf0daa37e4366018d91"' readmeLastModified: Wed, 09 Jun 2021 07:15:30 GMT repositoryId: 240551610 description: 基于Vue 的 Swagger OpenApi V3 API文档查看器 created: '2020-02-14T16:26:37Z' updated: '2023-08-18T15:44:16Z' language: Vue archived: false stars: 4 watchers: 1 forks: 0 owner: budwk logo: https://avatars.githubusercontent.com/u/58797370?v=4 license: Apache-2.0 repoEtag: '"657471db431b655510b9c5994cd0678839a7e20472bf617dd914894df38c1da0"' repoLastModified: Fri, 18 Aug 2023 15:44:16 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 8646e719f1e82252200279911158be75 - source: openapi3 tags repository: https://github.com/sohoffice/swagger-down v3: true repositoryMetadata: base64Readme: >- c3dhZ2dlci1kb3duIGlzIGEgT3BlbkFQSSAzIChTd2FnZ2VyIDMuMCkgc3BlYyB0byBtYXJrZG93biAvIGh0bWwgcmVuZGVyZXIuCgpJdCBpcyBhIGNvbW1hbmQgbGluZSBhcHBsaWNhdGlvbiB0aGF0IHRha2UgdGhlIHN3YWdnZXIgc3BlYyBmaWxlIGFzIGlucHV0IGFuZCByZW5kZXIgdG8gc3Rkb3V0LgpUeXBpY2FsIHVzYWdlIHdpbGwgcmVkaXJlY3QgdGhlIG91dHB1dCBkaXJlY3RseSB0byBhIGZpbGUgZm9yIGxhdGVyIHByb2Nlc3NpbmcuCgpFeGFtcGxlOgoKYGBgCmphdmEgLWphciBzd2FnZ2VyLWRvd24tYWxsLmphciBmb28ueW1sID4gZm9vLm1kCmBgYAoKc3dhZ2dlci1kb3duIGRvZXNuJ3Qgc3VwcG9ydCBzd2FnZ2VyIDIuMCwgdGhlcmUncyBhIG51bWJlciBvZiBvcHRpb25zIHN1cHBvcnRpbmcgaXQuClNvIHdlIGp1c3QgZGVjaWRlZCBub3QgdG8gcmUtaW52ZW50IHRoZSB3aGVlbC4KCkl0IGlzIGFsc28gbW9yZSB1c2VmdWwgd2hlbiBwcm9jZXNzaW5nIFlBTUwgc3BlYyBmaWxlLiBUaGUgYXBwbGljYXRpb24gb3V0cHV0cyBtYXJrZG93bgphbmQgaXQncyBhIGxvdCBuYXR1cmFsIHRvIHdyaXRlIG1hcmtkb3duIGluIFlBTUwgdGhhbiBpdCBpcyBpbiBKU09OLiBUaGUgMiBhcmUgcGVyZmVjdCBtYXRjaC4KSG93ZXZlciwgSlNPTiBzcGVjIGZpbGUgaXMgc3RpbGwgc3VwcG9ydGVkLCBqdXN0IHRoZSBvdXRwdXQgbWF5IGJlIGxlc3MgcHJldHR5LgoKT3V0cHV0IERvY3VtZW50Ci0tLS0tLS0tLS0tLS0tLQoKUGV0c3RvcmUgZXhhbXBsZTogaHR0cHM6Ly9naXRodWIuY29tL3NvaG9mZmljZS9zd2FnZ2VyLWRvd24vYmxvYi9tYXN0ZXIvc3JjL3Rlc3QvcmVzb3VyY2VzL3BldHN0b3JlLm1kCgpUaGUgYmFzaWMgaWRlYSBhcm91bmQgdGhlIGRlc2lnbiBvZiBnZW5lcmF0ZWQgZG9jdW1lbnQgaXMgYGZsYXR0ZW5gLgoKU3dhZ2dlciBzcGVjIHByb21vdGVzIHJlLXVzYWJpbGl0eSwgc28gdGhlIHNjaGVtYSBjYW4gYmUgZGVmaW5lZCBhcyBhIGNvbXBvbmVudCBhbmQgcmVmZXJlbmNlZCB3aGVuIG5lY2Vzc2FyeS4KU3dhZ2dlciBVSSBwcmVzZXJ2ZXMgdGhlIGNvbW1vbiBjb21wb25lbnQgc2VjdGlvbiwgYnV0IGFsc28gc2hvdyBzY2hlbWEgZGlyZWN0bHkgd2hlcmUgaXQgaXMgbmVlZGVkLgoKQSBtYXJrZG93biBkb2N1bWVudCBpcyBsZXNzIGludGVyYWN0aXZlLgoKSWYgdGhlIHJlYWRlciBoYXMgdG8ganVtcCBiYWNrIGFuZCBmb3J0aCB0byBsb29rdXAgc2NoZW1hIGRlZmluaXRpb24sIGl0IHdpbGwgc29vbiBtYWtlcyByZWFkaW5nIGV4cGVyaWVuY2UgbGVzcyBqb3lmdWwuClNvIGEgc3dhZ2dlci1kb3duIGdlbmVyYXRlZCBkb2N1bWVudCBiYXNpY2FsbHkgZmxhdHRlbnMgYWxsIHNjaGVtYSBhbmQgbWFrZSB0aGVtIGF2YWlsYWJsZSB3aGVyZSB0aGV5IGFyZSBuZWVkZWQuCkZvciBleGFtcGxlOiBpZiB0aGUgc2NoZW1hIGlzIHJlZmVyZW5jZWQgaW4gcmVxdWVzdCBib2R5LCB0aGUgZGV0YWlscyB3aWxsIGJlIGFkZGVkIHJpZ2h0IGluIHRoZSByZWxldmFudCBzZWN0aW9uLgpUaGUgc2FtZSBydWxlIGFwcGxpZXMgdG8gcmVzcG9uc2VzLgoKQWxzbywgdGhlIHN3YWdnZXIgc3BlYyBncm91cHMgZGlmZmVyZW50IG1ldGhvZHMgdW5kZXIgdGhlIHNhbWUgcGF0aC4gVGhpcyBtYWtlcyBwZXJmZWN0IHNlbnNlIHdpdGggU3dhZ2dlciBVSSwKYnV0IGFsc28gbWFrZSB0aGUgbWFya2Rvd24gZG9jdW1lbnQgbW9yZSB2ZXJib3NlIGFuZCBjcmVhdGVzIG1vcmUgaGllcmFyY2h5LgpBcyBhIHJlc3VsdCwgYWxsIEFQSXMgYXJlIGZsYXR0ZW5lZCBpbnRvIGEgbGlzdCwgYnV0IHRob3NlIHdpdGggdGhlIHNhbWUgcGF0aCBhcmUgcGxhY2VkIG5leHQgdG8gZWFjaCBvdGhlciB0byBtYWtlIApyZWFkaW5nIGV4cGVyaWVuY2UgbW9yZSBzbW9vdGhseS4gVGhpcyBhbHNvIGFwcGxpZXMgdG8gZGlmZmVyZW50IHJlcXVlc3QgY29udGVudCB0eXBlcyBvciBkaWZmZXJlbnQgcmVzcG9uc2Ugc3RhdHVzIApjb2RlICsgY29udGVudCB0eXBlcy4KCiMjIyMgSlNPTiBzY2hlbWEKCkpTT04gaXMgdGhlIGRlLWZhY3RvIGZvcm1hdCBvZiBBUEkgaW4gdGhlIG1vZGVybiBhZ2UsIGJ1dCBkb2N1bWVudGluZyBKU09OIGlzIG5vdCBhcyBzdHJhaWdodCBmb3J3YXJkIGFzIGl0IHNlZW1zLgoKc3dhZ2dlci1kb3duIHRha2VzIHRoZSBmbGF0dGVuaW5nIGFwcHJvYWNoIGFuZCBjb252ZXJ0cyB0aGUgZW50aXJlIHNjaGVtYSBpbnRvIGEgdGFibGUsIHdpdGgga2V5IHJlcHJlc2VudGVkIGluIEpTT04gcGF0aC4KClRha2UgdGhpcyBKU09OIGZvciBleGFtcGxlOgoKYGBganNvbgp7CiAgImNvZGUiOiAiRk9PIiwKICAibWVzc2FnZSI6ICJGb28gbWVzc2FnZSIsCiAgInBhcmFtZXRlcnMiOiBbCiAgICB7CiAgICAgICJuYW1lIjogInBhcmFtMSIsCiAgICAgICJ2YWx1ZSI6ICJ2YWx1ZSIKICAgIH0KICBdCn0KYGBgCgpUaGUgc2NoZW1hIHdpbGwgYmUgZG9jdW1lbnRlZCBpbiBiZWxvdyBmb3JtYXQuCgp8IEtleSB8IFR5cGUgfCBEZXNjcmlwdGlvbiB8CnwgLS0tIHwgLS0tLSB8IC0tLS0tLS0tLS0tIHwKfCAkLmNvZGUgfCBzdHJpbmcgfCBFcnJvciBjb2RlIHwKfCAkLm1lc3NhZ2UgfCBzdHJpbmcgfCBIdW1hbiByZWFkYWJsZSBlcnJvciBtZXNzYWdlIHwKfCAkLnBhcmFtZXRlcnNbXS5uYW1lIHwgc3RyaW5nIHwgUGFyYW1ldGVyIG5hbWUgfAp8ICQucGFyYW1ldGVyc1tdLnZhbHVlIHwgc3RyaW5nIHwgUGFyYW1ldGVyIHZhbHVlIHwKCldlIGhvcGUgdGhlIHRhYnVsYXIgZGlzcGxheSBoZWxwcyB0byBleHByZXNzIHRoZSBzY2hlbWEgZWxlbWVudCBpbiBhIG1vcmUgb3JnYW5pemVkIG1hbm5lci4KCiMjIyMgRG9jdW1lbnQgb3JnYW5pemF0aW9uCgpXaXRoIHRoZSBhYm92ZSBjb25jZXB0cyBpbiBtaW5kLCB0aGUgb3V0cHV0IGRvY3VtZW50IGlzIG9yZ2FuaXplZCBpbiB0aGUgZm9sbG93aW5nIG9yZGVyOgoKMS4gSW5mbwoyLiBTZXJ2ZXJzCjMuIEFQSSBmbGF0dGVuZWQgaW50byBgbWV0aG9kIHBhdGhgIChleDogR0VUIC9mb28vYmFyKQoKV2l0aGluIGVhY2ggQVBJIGVuZHBvaW50LCB0aGUgc2VjdGlvbnMgYXJlIHJlbmRlcmVkIGFzIHRoZSBiZWxvdwoKMS4gU3VtbWFyeQoyLiBEZXNjcmlwdGlvbgozLiBQYXJhbWV0ZXJzCjQuIFJlcXVlc3QgYm9keQo1LiBSZXNwb25zZXMgZmxhdHRlbmVkIGludG8gYHN0YXR1cyBjb250ZW50LXR5cGVgIChleDogMjAwIGFwcGxpY2F0aW9uL2pzb24pCgojIyMjIE91dHB1dCBmb3JtYXQKCi0gbWFya2Rvd24KLSBodG1sCgpUaGUgSFRNTCBpcyB1bnN0eWxlZCBhbmQgaXMgbW9yZSBzdWl0YWJsZSB3aGVuIGFueSBvZiB0aGUgZm9sbG93aW5nIGlzIHRydWUKCi0gWW91IGhhdmUgSFRNTCBtYXJrdXAgaW4geW91ciBzcGVjIGZpbGUuIFlvdSdsbCBuZWVkIHRvIHN0eWxlIHRoZSBvdXRwdXQgeW91cnNlbGYuCi0gWW91IGhhdmUgYSB3YXkgdG8gc3R5bGUgeW91ciBIVE1MLiBGb3IgZXhhbXBsZSwgaXQgY2FuIGJlIHBhc3RlZCBpbnRvIENvbmZsdWVuY2Ugd2lraS4KCktub3duIExpbWl0YXRpb25zCi0tLS0tLS0tLS0tLS0tLS0tCgotIFN3YWdnZXIgMyBvbmx5Ci0gQWR2YW5jZWQgc3dhZ2dlciBmZWF0dXJlIGxpa2Ugb25lT2YsIGFueU9mIGFyZSBub3Qgc3VwcG9ydGVkLiBXZSdyZSBsb29raW5nIHRvIGFkZCB0aGVtIGlmIHJlcXVlc3RlZC4KLSBPcGVuIElEIHNlY3VyaXR5IHN0cnVjdHVyZXMgYXJlIG5vdCBzdXBwb3J0ZWQKCk1haW50ZW5hbmNlCi0tLS0tLS0tLS0tCgpTZWUgW0RFVi5tZF0oREVWLm1kKSBmb3IgcmVmZXJlbmNlCgpQcmUtcmVxdWlzaXRlCi0tLS0tLS0tLS0tLS0KCllvdSBuZWVkIEphdmEgOCsgdG8gcnVuLgoKUnVubmluZwotLS0tLS0tCgpEb3dubG9hZCB0aGUgbGF0ZXN0IGJpbmFyeSBmcm9tIFtyZWxlYXNlXShodHRwczovL2dpdGh1Yi5jb20vc29ob2ZmaWNlL3N3YWdnZXItZG93bi9yZWxlYXNlcykgCmFuZCBleGVjdXRlIGluIGNvbW1hbmQgcHJvbXB0IHdpdGg6CgpgYGAKamF2YSAtamFyIHN3YWdnZXItZG93bi1hbGwuamFyIFtvcHRpb25zXSA8c3BlY19maWxlbmFtZT4KYGBgCgpUaGUgZG93bmxvYWRlZCBqYXIgaXMgYSBgZmF0IGphcmAsIGl0IGNhbiBiZSBleGVjdXRlZCB3aXRob3V0IGZ1cnRoZXIgZGVwZW5kZW5jaWVzLgoKIyMgVXNhZ2UKCmBgYApqYXZhIC1qYXIgYnVpbGQvbGlicy9zd2FnZ2VyLWRvd24tYWxsLmphciAtaGVscApVc2FnZTogPG1haW4gY2xhc3M+IFstaF0gWy1mPTxmb3JtYXQ+XSBbLXI9PGZsYXZvcj5dIEZJTEUKICAgICAgRklMRSAgICAgICAgICAgICAgICBTd2FnZ2VyIFlBTUwgZmlsZXMgdG8gY29udmVydAogIC1mLCAtLWZvcm1hdD08Zm9ybWF0PiAgIFRoZSBvdXRwdXQgZm9ybWF0LiBWYWx1ZXM6IE1ELCBIVE1MCiAgLWgsIC0taGVscCAgICAgICAgICAgICAgZGlzcGxheSBhIGhlbHAgbWVzc2FnZQogIC1yLCAtLWZsYXZvcj08Zmxhdm9yPiAgIFRoZSBnZW5lcmF0aW9uIGZsYXZvci4gVmFsdWVzOiBTVEFOREFSRApgYGAK readmeEtag: '"6b45a5fd8c3b91be9a2fde89dcac42e6d386edc1"' readmeLastModified: Sun, 28 Mar 2021 02:54:00 GMT repositoryId: 340091372 description: >- A CLI application to render OpenAPI 3 (swagger 3.0) spec to markdown / html created: '2021-02-18T15:23:24Z' updated: '2023-09-01T09:00:30Z' language: Kotlin archived: false stars: 4 watchers: 1 forks: 0 owner: sohoffice logo: https://avatars.githubusercontent.com/u/3193955?v=4 repoEtag: '"0f512639c0a1b6cab5acbee7da8abe11a096c009b5926c8c76593cacd3426361"' repoLastModified: Fri, 01 Sep 2023 09:00:30 GMT foundInMaster: true category: Parsers id: 970de5664bcc609366a6f02d1e11252f - source: openapi3 tags repository: https://github.com/rodnansol/openapi-extender v3: true id: 653701984139345acce908cfa00bd535 repositoryMetadata: base64Readme: >- PSBPcGVuQVBJIEV4dGVuZGVyCmlmbmRlZjo6ZW52LWdpdGh1YltdCjppY29uczogZm9udAplbmRpZjo6W10KaWZkZWY6OmVudi1naXRodWJbXQo6Y2F1dGlvbi1jYXB0aW9uOiA6ZmlyZToKOmltcG9ydGFudC1jYXB0aW9uOiA6ZXhjbGFtYXRpb246Cjpub3RlLWNhcHRpb246IDpwYXBlcmNsaXA6Cjp0aXAtY2FwdGlvbjogOmJ1bGI6Cjp3YXJuaW5nLWNhcHRpb246IDp3YXJuaW5nOgplbmRpZjo6W10KOnRvYzoKOnRvYy1wbGFjZW1lbnQhOgo6dG9jbGV2ZWxzOiA0CgppbWFnZTo6YmFubmVyLnBuZ1tiYW5uZXIucG5nXQpbLnRleHQtY2VudGVyXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9vcmcucm9kbmFuc29sL29wZW5hcGktZXh0ZW5kZXItcGFyZW50LnN2Z1tNYXZlbiBDZW50cmFsXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtQXBhY2hlXzIuMC1ibHVlLnN2Z1tBcGFjaGUgMi4wXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL3R3aXR0ZXIvdXJsL2h0dHBzL3R3aXR0ZXIuY29tL3JvZG5hbnNvbC5zdmc/c3R5bGU9c29jaWFsJmxhYmVsPUZvbGxvdyUyMCU0MFJvZG5hblNvbFtdCmltYWdlOmh0dHBzOi8vZGNiYWRnZS52ZXJjZWwuYXBwL2FwaS9zZXJ2ZXIvVVN5aDZYVWp2UFtEaXNjb3JkXQoKdG9jOjpbXQoKPT0gTW90aXZhdGlvbgoKV3JpdGluZyBwcm9wZXIgT3BlbkFQSSBkb2N1bWVudGF0aW9ucyBhcmUgdGltZS1jb25zdW1pbmcsIGlmIHlvdSByZWFsbHkKd2FudCB0byBtYWtlIHN1cmUgeW91ciBjbGllbnRzIGFyZSB1bmRlcnN0YW5kaW5nIHRoZSBBUEkgaXRzZWxmLCBtYXNzaXZlCmFtb3VudCBvZiBzdW1tYXJpZXMgYW5kIGV4YW1wbGVzIHNob3VsZCBiZSBwcm92aWRlZCwgbm90IG1hbmRhdG9yeSwgYnV0CmlmIHlvdSBhcmUgb24gdGhlIG90aGVyIHNpZGUgb2YgdGhlIEFQSSwgeW91IG1pZ2h0IGJlIGFsc28gaGFwcGllciBpZiBhbgpBUEkgZG9jdW1lbnRhdGlvbiBpcyB2ZXJib3NlIGFzIHBvc3NpYmxlLgoKPT0gR29hbHMKClRoZSBnb2FsIG9mIHRoaXMgcHJvamVjdCB0byBtYWtlIHN1cmUgdGhhdCB5b3VyIE9wZW5BUEkgZG9jdW1lbnRhdGlvbiBpcwpoYXZpbmcgZW5vdWdoIGV4dHJhIGRldGFpbHMsIHRoYXQgeW91IG1pZ2h0IG5vdCB3YW50IHRvIHdyaXRlIGJ5CnlvdXJzZWxmIGluIHRoZSBkb2N1bWVudGF0aW9uLCBidXQgeW91IHdvdWxkIGxpa2UgdG8gdXNlIG90aGVyLCBlYXJsaWVyCndyaXR0ZW4gcmVzb3VyY2VzIHRvIGZ1bGZpbCB0aGVzZSByZXF1aXJlbWVudHMvbmVlZHMvZ2FwcyBpbiB5b3VyCmRvY3VtZW50YXRpb24uCgpEZXZlbG9wZXJzIGFyZSB3cml0aW5nIHVuaXQgYW5kIGludGVncmF0aW9uIHRlc3RzIG9uIHRoZWlyIHByb2plY3RzLCBhbmQKYSBnb29kIHRlc3QgY291bGQgYmUgcHJvdmlkaW5nIHRoZSBkaWZmZXJlbnQgaW5wdXQgYW5kIG91dHB1dCB2YWx1ZXMgZm9yCmEgY29tcG9uZW50LiBBUEkgZGV2ZWxvcGVycyBhcmUgd3JpdGluZyB0b25zIG9mIGludGVncmF0aW9uIHRlc3RzIGZvcgp0aGUgZGlmZmVyZW50IHVzZSBjYXNlcyBhbmQgaXQgaGFwcGVucyBzb21ldGltZXMgdGhhdCB0aGV5IGFyZSBub3QKZ2l2aW5nIG91dCB0aGVzZSB0ZXN0cywgYmVjYXVzZSBvZiBjb3Vyc2Ugc2VjdXJpdHkgYW5kIHByaXZhY3kgcnVsZXMsCmJ1dCBzb21lIG91dHB1dCBvZiB0aGVzZSB0ZXN0cyBjb3VsZCBiZSB1c2VkIHRvIGV4dGVuZCB0aGUgYWxyZWFkeQpleGlzdGluZyBBUEkuCgo9PSBBbHRlcm5hdGl2ZXMKCiogaHR0cHM6Ly9naXRodWIuY29tL3NwcmluZy1wcm9qZWN0cy9zcHJpbmctcmVzdGRvY3NbU3ByaW5n4oCZcyBSRVNUIERvY3NdCmRvZXMgc29tZXRoaW5nIHNpbWlsYXIsIGJ1dCBpdCBpcyBub3QgaGVhdmlseSBpbnZvbHZlZCBpbiB0aGUgT3BlbkFQSQppbml0aWF0aXZlLgoqIGh0dHBzOi8vZ2l0aHViLmNvbS9lUGFnZXMtZGUvcmVzdGRvY3MtYXBpLXNwZWNbU3ByaW5nIFJFU1QgRG9jcyBBUEkKc3BlY2lmaWNhdGlvbiBJbnRlZ3JhdGlvbl0ga2luZGEgdGhlIHdhbnRlZCBmZWF0dXJlcywgYnV0IGl0IGlzIHRvdGFsbHkKdGVzdCBkcml2ZW4uCgo9PSBJbXBsZW1lbnRhdGlvbgoKVGhlIHByb2plY3QgaXMgc3BsaXQgdXAgaW50byBtdWx0aXBsZSBtb2R1bGVzIHRvIG1ha2Ugc3VyZSB0aGUgZW5kIHVzZXJzCmFyZSBub3QgZGVwZW5kaW5nIG9uIHNvbWUgdW5uZWNlc3NhcnkgZGVwZW5kZW5jaWVzOgoKKiBvcGVuYXBpLWV4dGVuZGVyIC0gQ29yZSBwYXJ0LCB0aGF0IGRvZXMgdGhlIGBgaGVhdnktbGlmdGluZycnIHRoYXQKd29ya3Mgd2l0aCB0aGUgT3BlbkFQSSBtb2RlbCBjbGFzc2VzCiogb3BlbmFwaS1leHRlbmRlci1yZXNvdXJjZS1nZW5lcmF0b3IgLSBSZXNvdXJjZSBnZW5lcmF0b3IgbW9kdWxlIHRoYXQKaXMgZ2VuZXJhdGluZyB0aGUgcmVzb3VyY2UgZmlsZXMgYmFzZWQgb24gZGlmZmVyZW50IGNvbnRlbnQKKiBvcGVuYXBpLWV4dGVuZGVyLXNwcmluZy10ZXN0IC0gU3ByaW5nIFRlc3QgZXh0ZW5zaW9uCiogb3BlbmFwaS1leHRlbmRlci1zcHJpbmdkb2MgLSBTcHJpbmdEb2MgZXh0ZW5zaW9uCgo9PSBVc2FnZQoKUGxlYXNlIGNoZWNrIHRoZSBmb2xsb3dpbmcgZG9jdW1lbnRhdGlvbiBzZWN0aW9ucyBmb3IgeW91ciByZXF1ZXN0ZWQgZmVhdHVyZXMKCj09PSA8PGRvY3MvZ2VuZXJhdGVkLWV4YW1wbGVzLmFkb2MjZ2VuZXJhdGVkLWV4YW1wbGVzLCBHZW5lcmF0ZWQgZXhhbXBsZXMgd2l0aCB0ZXN0cz4+CgppZm5kZWY6OmVudi1naXRodWJbXQppbmNsdWRlOjpkb2NzL2dlbmVyYXRlZC1leGFtcGxlcy5hZG9jW10KZW5kaWY6OltdCgo9PT0gPDxkb2NzL2V4dHJhLXN1bW1hcmllcy5hZG9jI2V4dHJhLXN1bW1hcmllcywgRXh0cmEgc3VtbWFyeSBhbmQgZGVzY3JpcHRpb24gZm9yIG9wZXJhdGlvbnMgPj4KCmlmbmRlZjo6ZW52LWdpdGh1YltdCmluY2x1ZGU6OmRvY3MvZXh0cmEtc3VtbWFyaWVzLmFkb2NbXQplbmRpZjo6W10K readmeEtag: '"84aaf4cfeb2ad6551317d8996642f9efcad24007"' readmeLastModified: Mon, 23 Jan 2023 08:16:03 GMT repositoryId: 534821652 description: >- OpenAPI Extender that will help you to move faster with your API documentation. created: '2022-09-09T22:14:58Z' updated: '2025-05-28T12:33:16Z' language: Java archived: false stars: 5 watchers: 1 forks: 0 owner: rodnansol logo: https://avatars.githubusercontent.com/u/107804610?v=4 license: Apache-2.0 repoEtag: '"3cad0ffc3d8d1835bb5c4cfe585cf7ee7efe1358834b67d666ac136727b1f062"' repoLastModified: Wed, 28 May 2025 12:33:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/louiscavalcante/ts-node-boilerplate-2024 v3: true id: 51a3ee3d0ec902f994c8cc1da04f2258 repositoryMetadata: base64Readme: >- IyMgVXNhZ2UKCkluc3RhbGwgdGhlIHBhY2thZ2VzLgoKYGBgYmFzaAokIG5wbSBpCmBgYAoKSW5zdGFsbCB0aGUgcmVxdWlyZW1lbnRzLgoKYGBgYmFzaAokIG5wbSBydW4gZGV2OnJlcXVpcmVtZW50cwpgYGAKCiMjIFRpcHM6CgotICAgVGhpcyBwcm9qZWN0IHNlcGFyYXRlcyBgcHJldHRpZXJgIGp1c3QgZm9yIGZvcm1hdHRpbmcgYW5kIGBlc2xpbnRgIGp1c3QgZm9yIGxpbnRpbmcsIGFzIGl0J3MgYWR2aXNlZCBieSB0aGUgZGV2ZWxvcGVycy4gVGhhdCdzIHdoeSBJIGRvbid0IHVzZSBgZXNsaW50LXBsdWdpbi1wcmV0dGllcmAKCi0gICBXaGVuIHlvdSBjb21taXQgc29tZXRoaW5nLCBgaHVza3lgLCBgbXJtYCBhbmQgYGxpbnQtc3RhZ2VgIGFyZSBzZXQgdG8gcnVuIGBwcmV0dGllcmAsIGBlc2xpbnRgIGFuZCBgamVzdGAuPGJyPgogICAgVGhpcyB3aWxsIGZpeCBldmVyeXRoaW5nIGFuZCBsZXQgeW91IGtub3cgaWYgeW91IGZvcmdvdCBhYm91dCBhbnkgZXJyb3IuPGJyPgogICAgSXQgd2lsbCBhbHNvIHJ1biB0ZXN0cyB0aGF0IHdlcmUgY3JlYXRlZCBvciBtb2RpZmllZC4KCi0gICBFdmVyeSB0aW1lIHlvdSB1cGRhdGUgdGhlIGBvcGVuYXBpLnlhbWxgIGZpbGUsIHlvdSBuZWVkIHRvIHJ1biB0aGUgYG5wbSBydW4gdXBkYXRlOm9wZW5hcGlgIHNjcmlwdCwgdG8ga2VlcCBgb3BlbmFwaS1zY2hlbWEuZC50c2AgdXBkYXRlZCwgc28geW91IGNhbiB1c2UgdGhlIG9wZW5hcGkncyBzY2hlbWFzIGFzIHR5cGVzIGZvciB0aGlzIGFwcGxpY2F0aW9uLgoKLSAgIFRoZSBgZXJyb3ItaGFuZGxlci5taWRkbGV3YXJlYCB3aWxsIGhhbmRsZSBhbGwgZXJyb3JzLCB0aGlzIGFsc28gaW5jbHVkZXMgYW55IHVua25vd24gcm91dGVzIGJ5IHRoZSBgb3BlbmFwaS55YW1sYCBhbmQgdGhlIGVycm9ycyB0aHJvd24gYnkgdGhlIGNsYXNzZXMgYERvbWFpbkVycm9yYCBhbmQgYEF4aW9zUmVxdWVzdEVycm9yYC48YnI+CiAgICBXaGVuIGBOT0RFX0VOVmAgaXMgc2V0IHRvIGBwcm9kdWN0aW9uYC4gSXQgd2lsbCBOT1QgaW50ZXJmZXJlIHdpdGggdGhlIGxvZ3MsIGJ1dCB3aWxsIGNoYW5nZSB0aGUgZXJyb3IgcmVzcG9uc2UgdG86PGJyPgogICAgYGh0dHAgc3RhdHVzIDUwMGAgYW5kIHRoZSBib2R5IHJlc3BvbnNlIHRvOiBgU29tZXRoaW5nIGJhZCBoYXBwZW5lZGA8YnI+CiAgICBVbmxlc3MgaXQncyBhIGBEb21haW5FcnJvcmAgdGhhdCB0aGUgY2xpZW50IG11c3Qga25vdyBhYm91dCEKCi0gICBVbmtub3duIGV4Y2VwdGlvbnMgYnkgdGhlIGFwcGxpY2F0aW9uLCB3aWxsIHVzZSB0aGUgRXJsYW5nJ3MgbGV0IGl0IGNyYXNoIHBoaWxvc29waHkuPGJyPgogICAgQmFzaWNhbGx5IGl0IGxldHMgdGhlIGFwcCBjcmFzaCBzbyB0aGUgb3JjaGVzdHJhdG9yIGNhbiByZXN0YXJ0IHRoZSBhcHBsaWNhdGlvbi48YnI+CiAgICBCdXQgaG93IHRoaXMgcmVhbGx5IHdvcmtzPzxicj4KICAgIEZpcnN0IHRoZSBhcHAgd2lsbCBzdG9wIHJlY2VpdmluZyByZXF1ZXN0LCB0aGVuIGZpbmlzaCB0aGUgcmVxdWVzdHMgdGhhdCB3ZXJlIGFscmVhZHkgaW4gcHJvZ3Jlc3MgYW5kIGdyYWNlZnVsbHkgc2h1dHMgZG93biBldmVyeXRoaW5nLjxicj4KICAgIFRoaXMgYXBwcm9hY2ggaW1wcm92ZXMgc3lzdGVtIHJlbGlhYmlsaXR5Ljxicj4KICAgIElmIHlvdSdyZSBvbiBLdWJlcm5ldGVzLCB5b3UgY2FuIHVzZSBJbmdyZXNzIG9yIG90aGVyIGxvYWQgYmFsYW5jZXIgc3RyYXRlZ2llcyBmb3IgeW91ciBhcHBsaWNhdGlvbi4KCi0gICBUaGVyZSdzIGEgYHJhdGVMaW1pdGVyYCBtZXRob2QgY29uZmlndXJlZCB0byBsaW1pdCByZXF1ZXN0IHBlciB1c2VyJ3MgaXAuPGJyPgogICAgWW91IGNhbiBkZWxldGUgaXQgaWYgeW91IGRlY2lkZSB0byB1c2UgdGhpcyBwcm9qZWN0IGFzIGEgcHJpdmF0ZSBBUEkuCgotICAgZG90ZW52IGlzIHNldCBqdXN0IGZvciBKZXN0Ljxicj4KICAgIEZvciBkZXZlbG9wbWVudCBpdCB1c2VzIGAtLWVudi1maWxlIC4vZW52aXJvbm1lbnRzLy5lbnZgCgotICAgSW5zaWRlIHRoZSBgdHNjb25maWcuanNvbmAgdW5kZXIgYHBhdGhzOiB7fWAgeW91IGNhbiBzZXQgY3VzdG9tIHBhdGhzLjxicj4KICAgIFlvdSBkb24ndCBuZWVkIHRvIHNldCB0aGVtIGFueXdoZXJlIGVsc2UsIHRoaXMgcHJvamVjdCBhbHJlYWR5IGRvZXMgdGhhdCBmb3IgeW91LgoKLSAgIFRoZXJlIGFyZSB0d28gZGVidWdnZXJzIGNvbmZpZ3VyZWQgd2l0aCBhdXRvIHJlbG9hZCB0byBmYWNpbGl0YXRlIHlvdXIgd29ya2Zsb3cuPGJyPgogICAgU28geW91IGRvbid0IG5lZWQgdG8ga2VlcCByZXN0YXJ0aW5nIHlvdXIgZGVidWdnZXIgbWFudWFsbHkuPGJyPgogICAgT25lIGZvciBOb2RlIGFuZCB0aGUgb3RoZXIgZm9yIEplc3QuCgotICAgWW91IGNhbiBydW4vZGVidWcganVzdCBvbmUgdGVzdCBhdCBhIHRpbWUgd2l0aCB0aGUgZXh0ZW5zaW9uIGBKZXN0IFJ1bm5lcmAgZnJvbSB0aGUgYXV0aG9yIGBmaXJzdHRyaXMgYDxicj4KICAgIFRoZSBwcm9qZWN0IGlzIGFscmVhZHkgc2V0IHRvIHJ1biBpdCB3aXRob3V0IGFueSBwcm9ibGVtcy48YnI+CiAgICBQUzogSSd2ZSBmb3VuZCBhIGJ1ZyBpbiB0aGlzIGV4dGVuc2lvbiBvbiBsaW51eCwgYnV0IHRoZSBidWcganVzdCBvY2N1cnMgaWYgeW91IHNldCB5b3VyIGBpdChgYCwgKCkgPT4ge30pYCB3aXRoIHRlbXBsYXRlIHN0cmluZ3MgdGhhdCBoYXMgbXVsdGlwbGUgbGluZXMuIFRoZSBidWcgaGFwcGVucyB3aGVuIHRoZSBmaXJzdCBsaW5lIGhhcyBhIHNpbWlsYXIgZGVzY3JpcHRpb24gd2l0aCBhbm90aGVyIHRlc3QgYmVmb3JlIHRoZSBsaW5lIGJyZWFrcy48YnI+CiAgICBCZXNpZGVzIHRoYXQgaXQgd29ya3MgZ3JlYXQhCgojIyBXWVAKCi0gICBVbml0IHRlc3RzCg== readmeEtag: '"f9e7803c7ba79f90e84ed371ec969fd46ef4188e"' readmeLastModified: Tue, 23 Jan 2024 13:47:23 GMT repositoryId: 739321785 description: null created: '2024-01-05T09:37:17Z' updated: '2025-03-13T01:54:32Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 0 owner: louiscavalcante logo: https://avatars.githubusercontent.com/u/32718388?v=4 license: MIT repoEtag: '"b8ecb67a9c6e09e04024dbed2dcdc12ae6cf5168c62b8ab1eb4a1fa29fa36289"' repoLastModified: Thu, 13 Mar 2025 01:54:32 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/methodus/homematic_openapi v3: true repositoryMetadata: base64Readme: >- IyBob21lbWF0aWNfb3BlbmFwaQpPcGVuQVBJIDMuMCBzcGVjIGJhc2VkIEhvbWVtYXRpYyBBUEkK readmeEtag: '"0981c8bbbd52f0de41671bde0a9a84c15dc02937"' readmeLastModified: Sat, 16 Mar 2019 20:41:03 GMT repositoryId: 170865647 description: OpenAPI 3.0 spec based Homematic API created: '2019-02-15T13:01:53Z' updated: '2022-09-15T06:37:47Z' language: Tcl archived: false stars: 4 watchers: 3 forks: 0 owner: methodus logo: https://avatars.githubusercontent.com/u/4979595?v=4 license: Apache-2.0 repoEtag: '"218d44d2a93d64e2d3e5d11f7aa6cd5136b9f1e9d69e3cd69693470527e10dd5"' repoLastModified: Thu, 15 Sep 2022 06:37:47 GMT foundInMaster: true category: Parsers id: 4f9c3ac7e771ac85e88ed15e1140c97e - source: openapi3 tags repository: https://github.com/felixthec/openapi-fastapi-client v3: true id: 4c04f44b7d6d0aba1b972175daf2fd27 repositoryMetadata: base64Readme: >- WyFbUHl0aG9uIDMuMTBdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcHl0aG9uLTMuMTAtYmx1ZS5zdmcpXShodHRwczovL3d3dy5weXRob24ub3JnL2Rvd25sb2Fkcy9yZWxlYXNlL3B5dGhvbi0zMTAwLykKIVtQeXRob24gYXBwbGljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9GZWxpeFRoZUMvb3BlbmFwaS1mYXN0YXBpLWNsaWVudC93b3JrZmxvd3MvUHl0aG9uJTIwYXBwbGljYXRpb24vYmFkZ2Uuc3ZnKQpbIVtDb2RlIHN0eWxlOiBibGFja10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlJTIwc3R5bGUtYmxhY2stMDAwMDAwLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9wc2YvYmxhY2spClshW0ltcG9ydHM6IGlzb3J0XShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlLyUyMGltcG9ydHMtaXNvcnQtJTIzMTY3NGIxP3N0eWxlPWZsYXQmbGFiZWxDb2xvcj1lZjgzMzYpXShodHRwczovL3B5Y3FhLmdpdGh1Yi5pby9pc29ydC8pCgoKIyBPcGVuYXBpIHlhbWwgZmlsZSB0byBGYXN0QXBpIENsaWVudApBIGNvbW1hbmRsaW5lIHRvb2wgdG8gZ2VuZXJhdGUgQXBpIGBmdW5jdGlvbnNgIGFuZCB0aGVpciByZXF1aXJlZCBgcHlkYW50aWMgTW9kZWxgIFNjaGVtYSBmcm9tIGFuIGBvcGVuYXBpLnlhbWxgIG9mIHZlcnNpb24gMwoKIyMgSW5zdGFsbGF0aW9uCmBgYHNoZWxsCnBpcCBpbnN0YWxsIG9wZW5hcGktZmFzdGFwaS1jbGllbnQKYGBgCgojIyBVc2FnZQpgYGBzaGVsbApvcGVuYXBpLWZhc3RhcGktY2xpZW50IC4vb3BlbmFwaS55YW1sIC4vbXktY2xpZW50CmBgYApgYGBzaGVsbApvcGVuYXBpLWZhc3RhcGktY2xpZW50IC4vb3BlbmFwaS55YW1sIC4vbXktY2xpZW50IC0tYXN5bmMKYGBgCi0gdGhpcyB3aWxsIGdlbmVyYXRlIHVuZGVyIHRoZSBmb2xkZXIgYG15LWNsaWVudGAgZm9sbG93aW5nIGZpbGVzCiAgLSBgX19pbml0X18ucHlgIGlmIG5vdCBleGlzdHMKICAtIGBhcGkucHlgIGhlcmUgYXJlIGFsbCBmdW5jdGlvbiBjYWxscyB0byB0aGUgZXh0ZXJuYWwgYXBpCiAgLSBgc2NoZW1hLnB5YCBoZXJlIGFyZSBhbGwgcHlkYW50aWMgTW9kZWxzCiAgCgojIyBBcmd1bWVudHMKLSBgT1BFTkFQSV9GSUxFICBbcmVxdWlyZWRdYAotIGBPVVRQVVRfUEFUSCAgIFtyZXF1aXJlZF1gCgojIyBPcHRpb25zCi0gYC0tc3luY2AgIEFsbCByZXF1ZXN0cyB0byB0aGUgY2xpZW50IGFyZSBzeW5jaHJvbm91cy4gIF9kZWZhdWx0XwotIGAtLWFzeW5jYCBBbGwgcmVxdWVzdHMgdG8gdGhlIGNsaWVudCBhcmUgYXN5bmNocm9ub3VzIHdpdGggX19haW9odHRwX18uCgojIyBIZWxwCmBgYHNoZWxsCm9wZW5hcGktZmFzdGFwaS1jbGllbnQgLS1oZWxwCmBgYAoKIVtdKG9wZW5hcGktZmFzdGFwaS1jbGllbnRfbG9uZy5naWYpCg== readmeEtag: '"1a0cd96f4b2b1ce9a055f572f6f62050b5275e64"' readmeLastModified: Sat, 19 Nov 2022 13:52:19 GMT repositoryId: 564465979 description: A tool to autogenerate FastApi Clients from given openapi.yaml. created: '2022-11-10T19:27:40Z' updated: '2026-01-11T18:44:30Z' language: Python archived: false stars: 9 watchers: 1 forks: 0 owner: FelixTheC logo: https://avatars.githubusercontent.com/u/15969581?v=4 license: MIT repoEtag: '"e2c3dd5b3d3e5389da3c41aad33f77e6d03b700f3022ddc43b3a46f5e78589cf"' repoLastModified: Sun, 11 Jan 2026 18:44:30 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/hardiksinghbehl/spring-boot-freemarker-java-mail-api v3: true repositoryMetadata: base64Readme: >- IyBIZWR3aWcKIyMjIyBQcm9vZi1vZi1jb25jZXB0IGFwcGxpY2F0aW9uIHRvIHNlbmQgZW1haWxzIHRvIHVzZXIgb24gZXZlbnQgb2NjdXJlbmNlCgpbQ2xpY2sgSGVyZSBUbyBVc2UgUnVubmluZyBBcHBsaWNhdGlvbl0oaHR0cHM6Ly9qYXZhLW1haWwtZnJlZW1hcmtlci1zcHJpbmcuaGVyb2t1YXBwLmNvbS9oZWR3aWcvc3dhZ2dlci11aS5odG1sKQoKIyMgVGVjaG5vbG9naWVzIFVzZWQKKiBKYXZhIFNwcmluZyBCb290CiogSmF2YU1haWwtQVBJCiogU3ByaW5nIEV2ZW50cwoqIEFwYWNoZSBGcmVlbWFya2VyCiogSlBBL0gyIGluLW1lbW9yeSBkYXRhYmFzZQoqIExvbWJvawoqIE9wZW4tQVBJIChzd2FnZ2VyLXVpKSAKKiBbQm9vdHN0cmFwLUVtYWlsIEVkaXRvcl0oaHR0cHM6Ly9hcHAuYm9vdHN0cmFwZW1haWwuY29tL2VkaXRvcikKCiMjIFNwcmluZyBFdmVudHMKKiBbUmVmZXJlbmNlIEFydGljbGVdKGh0dHBzOi8vd3d3LmJhZWxkdW5nLmNvbS9zcHJpbmctZXZlbnRzKQoqIEV2ZW50IGNsYXNzZXMgd2lsbCByZXByZXNlbnQgYW4gZXZlbnQgZm9yIHdoaWNoIGEgbWFpbCBoYXMgdG8gYmUgc2VudCAoYWNjb3VudC1jcmVhdGlvbi1ldmVudCBpbiB0aGlzIGNvbnRleHQgY2FuIGJlIGFueXRoaW5nIGRlcGVuZGluZyBvbiBidXNpbmVzcyBsb2dpYykKKiBBIGxpc3RlbmVyIGNsYXNzIHRoYXQgbGlzdGVucyB0byB0aGUgYWJvdmUgY3JlYXRlZCBldmVudCBhbmQgZXhlY3V0ZXMgY29kZSB3aGVuZXZlciB0aGUgYWJvdmUgZXZlbnQgb2NjdXJlcwoqIEEgcHVibGlzaGVyIHdpbGwgcHVibGlzaCB0aGUgZXZlbnQgKG5vdGlmeSB0aGF0IGV2ZW50IGhhcyBvY2N1cnJlZCkgYW5kIGNhbiBiZSBpbmplY3RlZCBpbiB0aGUgc2VydmljZSBsYXllcgoqIExpc3RlbmVyIG1ldGhvZCB0byBiZSBleGVjdXRlZCBhc3luY2hyb25vdXNseSAocmVjb21tZW5kZWQpCgojIyBNYWluIGNsYXNzZXMgKERyYWcgdG8gbmV3IHdpbmRvdykKKiBbRW1haWwtU2VydmljZV0oaHR0cHM6Ly9naXRodWIuY29tL2hhcmRpa1NpbmdoQmVobC9zcHJpbmctYm9vdC1mcmVlbWFya2VyLWphdmEtbWFpbC1hcGkvYmxvYi9tYWluL3NyYy9tYWluL2phdmEvY29tL2hhcmRpay9oZWR3aWcvbWFpbC9zZXJ2aWNlL0VtYWlsU2VydmljZS5qYXZhKSAodG8gc2VuZCBlbWFpbHMgdG8gcHJvdmlkZWQgZW1haWwtaWQgd2l0aCBwcm92aWRlZCB0ZW1wbGF0ZSBhbmQgc3ViamVjdCkKKiBbQWNjb3VudCBDcmVhdGlvbiBFdmVudF0oaHR0cHM6Ly9naXRodWIuY29tL2hhcmRpa1NpbmdoQmVobC9zcHJpbmctYm9vdC1mcmVlbWFya2VyLWphdmEtbWFpbC1hcGkvYmxvYi9tYWluL3NyYy9tYWluL2phdmEvY29tL2hhcmRpay9oZWR3aWcvbWFpbC9ldmVudC9Vc2VyQWNjb3VudENyZWF0aW9uRXZlbnQuamF2YSkKKiBbQWNjb3VudCBDcmVhdGlvbiBFdmVudCBMaXN0ZW5lcl0oaHR0cHM6Ly9naXRodWIuY29tL2hhcmRpa1NpbmdoQmVobC9zcHJpbmctYm9vdC1mcmVlbWFya2VyLWphdmEtbWFpbC1hcGkvYmxvYi9tYWluL3NyYy9tYWluL2phdmEvY29tL2hhcmRpay9oZWR3aWcvbWFpbC9saXN0ZW5lci9Vc2VyQWNjb3VudENyZWF0aW9uTGlzdGVuZXIuamF2YSkgKHJlY2VpdmVzIHVzZXJDcmVhdGlvblJlcXVlc3REdG8gYXMgaW5wdXQgYW5kIHNlbmRzIG1haWwgdG8gcHJvdmlkZWQgZW1haWwtaWQgYXN5bmNocm9ub3VzbHkpCiogW1VzZXIgU2VydmljZV0oaHR0cHM6Ly9naXRodWIuY29tL2hhcmRpa1NpbmdoQmVobC9zcHJpbmctYm9vdC1mcmVlbWFya2VyLWphdmEtbWFpbC1hcGkvYmxvYi9tYWluL3NyYy9tYWluL2phdmEvY29tL2hhcmRpay9oZWR3aWcvc2VydmljZS9Vc2VyU2VydmljZS5qYXZhKSAoY2xhc3MgdGhhdCBjcmVhdGVzIHVzZXIgYWNjb3VudCBpbiB0aGUgc3lzdGVtIGFuZCBwdWJsaXNoZXMgdGhhdCBBY2NvdW50Q3JlYXRpb25FdmVudCBoYXMgb2NjdXJyZWQpCgojIyBCb290c3RyYXAgRW1haWwKCiogSFRNTCBmb3IgZW1haWxzIGlzIHNsaWdodGx5IGRpZmZlcmVudCBmcm9tIEhUTUwgZm9yIHdlYgoqIENvcHkgdGhlIEhUTUwgZ2VuZXJhdGVkIGZyb20gYm9vdHN0cmFwIGVtYWlsIGVkaXRvciAocmlnaHQgcGFydCBvZiBzY3JlZW4pIHRvIGEgLmZ0bCBmaWxlIHVuZGVyICoqc3JjL21haW4vcmVzb3VyY2VzL3RlbXBsYXRlcyoqIChjb25maWd1cmVkIGluIEZyZWVtYXJrZXJDb25maWd1cmF0aW9uLmNsYXNzKQoqICoqYWNjb3VudC1jcmVhdGlvbi1zdWNjZXNzLmZ0bCoqIGluIHRlbXBsYXRlcyBmb2xkZXIgd2lsbCBiZSBnZW5lcmF0ZWQgdG8gdGhlIGJlbG93IGltYWdlCiogJHtlbWFpbElkfSBhbmQgJHtmdWxsTmFtZX0gd2lsbCBiZSBjb252ZXJ0ZWQgdG8gcHJvdmlkZWQgZW1haWxJZCBhbmQgZnVsbG5hbWUgd2l0aCB0aGUgaGVscCBvZiBhcGFjaGUgZnJlZW1hcmtlcgoKIVttYWlsXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS82OTY5MzYyMS8xMjAyNDA4OTUtNGZjNmRkMDAtYzI3Zi0xMWViLTlmYzUtZWY3ZGE5MTQ0ZDVkLnBuZykKCiMjIExvY2FsIFNldHVwCgoqIEluc3RhbGwgSmF2YSAxNQoqIEluc3RhbGwgTWF2ZW4KClJlY29tbWVuZGVkIHdheSBpcyB0byB1c2UgW3Nka21hbl0oaHR0cHM6Ly9zZGttYW4uaW8vKSBmb3IgaW5zdGFsbGluZyBib3RoIG1hdmVuIGFuZCBqYXZhCgpHbyB0byBhcHBsaWNhdGlvbi5wcm9wZXJ0aWVzIGFuZCBjb25maWd1cmUgeW91ciBlbWFpbCBhbmQgW2FwcCBwYXNzd29yZF0oaHR0cHM6Ly9kZXZhbnN3ZXJzLmNvL2NyZWF0ZS1hcHBsaWNhdGlvbi1zcGVjaWZpYy1wYXNzd29yZC1nbWFpbC8pIChkaWZmcmVudCBmcm9tIHJlZ3VsYXIgcGFzc3dvcmRzKQoKYGBgCnNwcmluZy5tYWlsLnVzZXJuYW1lID0gPEVNQUlMLUlELUhFUkU+CnNwcmluZy5tYWlsLnBhc3N3b3JkID0gPFBBU1NXT1JELUhFUkU+CmBgYAoKUnVuIHRoZSBiZWxvdyBjb21tYW5kcyBpbiB0aGUgY29yZQoKYGBgCm12biBjbGVhbiBpbnN0YWxsCmBgYAoKYGBgCm12biBzcHJpbmctYm9vdDpydW4KYGBgCgpzZXJ2ZXIgcG9ydCBpcyBjb25maWd1cmVkIHRvIDkwOTAgYW5kIGJhc2UtdXJsIHRvIC9oZWR3aWcgd2hpY2ggY2FuIGJlIGNoYW5nZWQgaW4gYXBwbGljYXRpb24ucHJvcGVydGllcyBmaWxlCgpHbyB0byB0aGUgYmVsb3cgdXJsIHRvIHZpZXcgc3dhZ2dlci11aSAoQVBJIGRvY3MpCgpgYGAKaHR0cDovL2xvY2FsaG9zdDo5MDkwL2hlZHdpZy9zd2FnZ2VyLXVpLmh0bWwKYGBgCgpSdW4gVGVzdHMKCmBgYAptdm4gdGVzdApgYGA= readmeEtag: '"e5c50f99c61fbb8cebe0b2911499b1a6f0e2e0d3"' readmeLastModified: Sun, 13 Jun 2021 09:50:13 GMT repositoryId: 372554187 description: >- POC that sends emails to user on account creation event occurrence: made using java spring boot, apache freemarker, java-mail-API, bootstrap email editor and spring events. created: '2021-05-31T15:35:13Z' updated: '2024-10-08T18:15:11Z' language: Java archived: false stars: 4 watchers: 1 forks: 3 owner: hardikSinghBehl logo: https://avatars.githubusercontent.com/u/69693621?v=4 repoEtag: '"3d44be34f0edd04ead67adf8843a60d4ed684b87b61aaaa431a7a76dad88e236"' repoLastModified: Tue, 08 Oct 2024 18:15:11 GMT foundInMaster: true category: Testing id: 9ca6ae8f94bc87c4e5521849e283ea42 - source: openapi3 tags repository: https://github.com/trellix-opensource/intelligent-sandbox-api v3: true id: cfc5cc1c72af92b4d8b81459311749fc repositoryMetadata: base64Readme: >- IyBUcmVsbGl4IEludGVsbGlnZW50IFNhbmRib3ggKEZvcm1lcjogQVREKQpUaGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZvciBpbnRlcmFjdGluZyB3aXRoIFJFU1QgQVBJcyB0aGF0IFtUcmVsbGl4IEludGVsbGlnZW50IFNhbmRib3hdKGh0dHBzOi8vZG9jcy50cmVsbGl4LmNvbS9lbi9idW5kbGU/bGFiZWxrZXk9cHJvZC1hZHZhbmNlZC10aHJlYXQtZGVmZW5zZSkgKGZvcm1lcjogQVREKSBvZmZlcnMuIDxici8+CiMjIEhvdyB0byBnZW5lcmF0ZSBTREtzPwoKIyMjIFNldHRpbmcgdXAgdGhlIE9wZW5BUEkgR2VuZXJhdG9yIGluIHlvdXIgZW52aXJvbm1lbnQgKG9uZSB0aW1lIHN0ZXApCjEuIE5hdmlnYXRlIHRvOiBbT3BlbkFQSSBHZW5lcmF0b3IgaW5zdGFsbGF0aW9uXShodHRwczovL29wZW5hcGktZ2VuZXJhdG9yLnRlY2gvZG9jcy9pbnN0YWxsYXRpb24vKQoyLiBDaG9vc2UgdGhlIENMSSBjbGllbnQgdGhhdCBzdWl0cyB5b3VyIG5lZWQvaW50ZWdyYXRlcyB3aXRoIHlvdXIgcGxhdGZvcm0gYmV0dGVyLiAoKipSZWNvbW1lbmRlZCB0byB1c2UgdjUuMC4xKykqKiAgRm9yIG1vcmUgaW5mbyBwbGVhc2UgdmlzaXQ6IGh0dHBzOi8vb3BlbmFwaS1nZW5lcmF0b3IudGVjaCA8YnIvPgozLiBUaGUgT3BlbkFQSSBHZW5lcmF0b3Igc3VwcG9ydHMgbWFueSBkaWZmZXJlbnQgaW50ZWdyYXRpb25zIGFuZCB1c2UgY2FzZXMsIGluY2x1ZGluZyAoYnV0IG5vdCBsaW1pdGVkIHRvKToKICAgIC0gTWF2ZW4gUGx1Z2luCiAgICAtIEdyYWRsZSBQbHVnaW4KICAgIC0gQmF6ZWwgUGx1Z2luCiAgICAtIFNCVCBQbHVnaW4KICAgIC0gQ2FrZSBQbHVnaW4KICAgIC0gQ0xJIHZpYSBIb21lYnJldwogICAgLSBDTEkgdmlhIERvY2tlcgogICAgLSBDTEkgdmlhIG5wbQogICAgLSBHZW5lcmF0b3IgU2FhUzxicj4KICBGb3IgZGV0YWlscywgc2VlIFtXb3JrZmxvdyBJbnRlZ3JhdGlvbnNdKGh0dHBzOi8vb3BlbmFwaS1nZW5lcmF0b3IudGVjaC9kb2NzL2ludGVncmF0aW9ucy8pCgoqKk5PVEU6KiogCjxici8+CkluIHNvbWUgY2FzZXMgdGhlIGNsaSBnZW5lcmF0b3IgbWlnaHQgYmUgbmFtZWQgYXMgPGNvZGU+b3BlbmFwaS1nZW5lcmF0b3ItY2xpPC9jb2RlPiBpbnN0ZWFkIG9mIDxjb2RlPm9wZW5hcGktZ2VuZXJhdG9yPC9jb2RlPiwgcGxlYXNlIGJlIGNhdXRpb3VzLgoKIyMjIEdlbmVyYXRpbmcgdGhlIFNESyAKMS4gUnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCBhbmQgbWFrZSBzdXJlIHlvdSBhcmUgKipydW5uaW5nIHZlcnNpb24gNS4wKy4qKiBbRmluZCB0aGUgbGF0ZXN0IHZlcnNpb24gaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL09wZW5BUElUb29scy9vcGVuYXBpLWdlbmVyYXRvci9yZWxlYXNlcykgPGJyPgogICAgPGNvZGU+b3BlbmFwaS1nZW5lcmF0b3IgdmVyc2lvbjwvY29kZT4KMi4gRm9yIGxpc3Rpbmcgb3V0IHRoZSBhdmFpbGFibGUgbGFuZ3VhZ2VzIHlvdSBjYW4gdXNlOjxici8+CiAgPGNvZGU+b3BlbmFwaS1nZW5lcmF0b3IgbGlzdDwvY29kZT48YnI+CiAgQWx0ZXJuYXRpdmVseSwgeW91IGNhbiB2aXNpdDogW0xpc3Qgb2YgY2xpZW50IGdlbmVyYXRvcnNdKGh0dHBzOi8vb3BlbmFwaS1nZW5lcmF0b3IudGVjaC9kb2NzL2dlbmVyYXRvcnMvI2NsaWVudC1nZW5lcmF0b3JzKSBmb3IgbGlzdCBvZiBzdXBwb3J0ZWQgY2xpZW50IFNES3MuCjMuIEZvciBnZW5lcmF0aW5nIHRoZSBjbGllbnQgaW4gdGhlIGxhbmd1YWdlIG9mIHlvdXIgY2hvaWNlLCBydW4gdGhlIGZvbGxvd2luZzogPGJyLz4KIDxjb2RlPm9wZW5hcGktZ2VuZXJhdG9yIGdlbmVyYXRlIC1pIGlucHV0X2ZpbGVfbmFtZS55YW1sIC1nIGFfbGFuZ3VhZ2VfbmFtZV9mcm9tX2xpc3QgLW8gL3BhdGgvdG8vb3V0cHV0X2ZvbGRlcjwvY29kZT4gPGJyLz4KIDxici8+CjxiPk5PVEU6PC9iPgo8YnIvPgogMS4gSWYgZHVyaW5nIGV4ZWN1dGlvbiB5b3UgZW5jb3VudGVyIGFuIGVycm9yIHNheWluZyAiYW55IG1vZHVsZS9tb2RlbCBuYW1lIiBub3QgZm91bmQsIGtpbmRseSBnZW5lcmF0ZSB0aGUgY2xpZW50IHVzaW5nIHRoZSBmb2xsb3dpbmcgZ2xvYmFsIHByb3BlcnR5Ojxici8+CiA8Y29kZT5vcGVuYXBpLWdlbmVyYXRvciBnZW5lcmF0ZSAtaSBmaWxlX25hbWUueWFtbCAtZyBhX2xhbmd1YWdlX25hbWVfZnJvbV9saXN0IC1vIC9wYXRoL3RvL291dHB1dF9mb2xkZXIgLS1nbG9iYWwtcHJvcGVydHkgc2tpcEZvcm1Nb2RlbD1mYWxzZTwvY29kZT48YnIvPgoKIyMgSW1wb3J0aW5nIG9wZW5hcGkgc3BlYyBpbiBQb3N0bWFuCkluIG9yZGVyIHRvIGltcG9ydCB0aGUgW09wZW5BUEldKG9wZW5hcGkueWFtbCkgZmlsZSBpbiBwb3N0bWFuLCBwbGVhc2UgcmVmZXIgdG8gOiBbd29ya2luZyB3aXRoIG9wZW5BUEldKGh0dHBzOi8vbGVhcm5pbmcucG9zdG1hbi5jb20vZG9jcy9pbnRlZ3JhdGlvbnMvYXZhaWxhYmxlLWludGVncmF0aW9ucy93b3JraW5nLXdpdGgtb3BlbkFQSS8pCgojIyBDb250cmlidXRpbmcgYW5kIENvZGUgb2YgY29uZHVjdAogRm9yIENvZGUgb2YgY29uZHVjdCBhbmQgY29udHJpYnV0aW9uIGd1aWRlbGluZXMgcGxlYXNlIHJlZmVyIHRvIFtDT0RFX09GX0NPTkRVQ1RdKENPREVfT0ZfQ09ORFVDVC5tZCkgYW5kIFtDT05UUklCVVRJTkddKENPTlRSSUJVVElORy5tZCkgcmVzcGVjdGl2ZWx5LgogCgo= readmeEtag: '"adf909021b8a1a8dfe64cd5973d9ef02a05fb6f8"' readmeLastModified: Tue, 16 Aug 2022 05:22:33 GMT repositoryId: 521001395 description: >- The repository contains OpenAPI specification for interacting with REST APIs that Trellix Intelligent Sandbox (former: ATD) offers. created: '2022-08-03T19:05:34Z' updated: '2024-04-30T12:31:11Z' language: HTML archived: false stars: 4 watchers: 2 forks: 0 owner: trellix-opensource logo: https://avatars.githubusercontent.com/u/106668524?v=4 license: Apache-2.0 repoEtag: '"f0ccc030b1affa92f74347d216bdaac50aa67ead84ae5170a790459cad70267d"' repoLastModified: Tue, 30 Apr 2024 12:31:11 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/stefh/ramltoopenapiconverter v3: true repositoryMetadata: base64Readme: >- IyBSYW1sVG9PcGVuQXBpQ29udmVydGVyDQpDb252ZXJ0cyBhIFJBTUwgdG8gT3BlbiBBUEkgU3BlY2lmaWNhdGlvbg0KDQpbIVtOdUdldDogUmFtbFRvT3BlbkFwaUNvbnZlcnRlcl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9udWdldC92L1JhbWxUb09wZW5BcGlDb252ZXJ0ZXIpXShodHRwczovL3d3dy5udWdldC5vcmcvcGFja2FnZXMvUmFtbFRvT3BlbkFwaUNvbnZlcnRlcikNCg0KIyMgVXNhZ2UNCg0KIyMjIENvbnZlcnQgYSBSQU1MIGZpbGUNCmBgYCBjIw0KbmV3IFJhbWxDb252ZXJ0ZXIoKQ0KICAuQ29udmVydFRvRmlsZSgiTWVkaWFXaWtpLnJhbWwiLCAiTWVkaWFXaWtpLmpzb24iLCBPcGVuQXBpU3BlY1ZlcnNpb24uT3BlbkFwaTNfMCwgT3BlbkFwaUZvcm1hdC5Kc29uKTsNCmBgYA0KDQoNCiMjIyBMaW1pdHMNCi0gLi4uDQoNCg0KIyMgRGV0YWlscw0KVGhpcyBwcm9qZWN0IHVzZXMgdGhlIGZvbGxvd2luZyB0b29sczoNCi0gWWFtbERvdE5ldCAtLT4gdG8gcmVhZCB0aGUgUkFNTCAoYXMgWUFNTCBvciBKU09OKQ0KLSBNaWNyb3NvZnQuT3BlbkFwaS5ZYW1sUmVhZGVyIC0tPiB0byBwcm9jZXNzIHRoZSBPcGVuIEFQSSBNb2RlbCBhbmQgY29udmVydCB0aGUgbW9kZWwgdG8gdGhlIG91dHB1dCBmaWxlDQoNCiANCiMjIFNwb25zb3JzDQoNCltFbnRpdHkgRnJhbWV3b3JrIEV4dGVuc2lvbnNdKGh0dHBzOi8vZW50aXR5ZnJhbWV3b3JrLWV4dGVuc2lvbnMubmV0Lz91dG1fc291cmNlPVN0ZWZIKSBhbmQgW0RhcHBlciBQbHVzXShodHRwczovL2RhcHBlci1wbHVzLm5ldC8/dXRtX3NvdXJjZT1TdGVmSCkgYXJlIG1ham9yIHNwb25zb3JzIGFuZCBwcm91ZCB0byBjb250cmlidXRlIHRvIHRoZSBkZXZlbG9wbWVudCBvZiAqKlJhbWxUb09wZW5BcGlDb252ZXJ0ZXIqKi4NCg0KWyFbRW50aXR5IEZyYW1ld29yayBFeHRlbnNpb25zXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vU3RlZkgvcmVzb3VyY2VzL21haW4vc3BvbnNvci9lbnRpdHktZnJhbWV3b3JrLWV4dGVuc2lvbnMtc3BvbnNvci5wbmcpXShodHRwczovL2VudGl0eWZyYW1ld29yay1leHRlbnNpb25zLm5ldC9idWxrLWluc2VydD91dG1fc291cmNlPVN0ZWZIKQ0KDQpbIVtEYXBwZXIgUGx1c10oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL1N0ZWZIL3Jlc291cmNlcy9tYWluL3Nwb25zb3IvZGFwcGVyLXBsdXMtc3BvbnNvci5wbmcpXShodHRwczovL2RhcHBlci1wbHVzLm5ldC9idWxrLWluc2VydD91dG1fc291cmNlPVN0ZWZIKQ== readmeEtag: '"e4f9a9a47906e7bf329fc4b87a7634c6f2c97eb1"' readmeLastModified: Mon, 16 Jun 2025 06:10:01 GMT repositoryId: 237777850 description: Converts a RAML to Open API Specification (native C# dotnet) created: '2020-02-02T13:47:59Z' updated: '2025-09-05T07:30:42Z' language: C# archived: false stars: 4 watchers: 2 forks: 4 owner: StefH logo: https://avatars.githubusercontent.com/u/249938?v=4 license: MIT repoEtag: '"e5242a5be5119a4f46f6ce352de402112650c7bbb4109009e3189bd5eb656c55"' repoLastModified: Fri, 05 Sep 2025 07:30:42 GMT foundInMaster: true category: Parsers id: fc6d82a523fbcf983a51caa78aa81344 - source: openapi3 tags repository: https://github.com/membrane-php/membrane-core v3: true id: f7a3f57a79b6b3b13e5ff19c0dd5fd7a repositoryMetadata: base64Readme: >- IyBNZW1icmFuZQoKTWVtYnJhbmUgaXMgYSBsaWdodHdlaWdodCBpbnB1dCB2YWxpZGF0aW9uIGxheWVyIHdoaWNoIGNhbiBiZSB1c2VkIHRvIGFuc3dlciB0aGUgcXVlc3Rpb24gIklzIHRoaXMgaW5wdXQgc29tZXRoaW5nIHdlCm1pZ2h0IGJlIGFibGUgdG8gcHJvY2Vzcz8iLiBJdHMKcHVycG9zZSBpcyB0byByZWNlaXZlICJyYXciIHVzZXIgaW5wdXQgYW5kIHByb2R1Y2UgYSBjbGVhbmVkIHVwLCBmdWxseSB2YWxpZGF0ZWQgb2JqZWN0IG9yIGFycmF5IHJlcHJlc2VudGluZyB0aGF0IGRhdGEsCndoaWNoIHRoZSByZXN0IG9mIHlvdXIgYXBwbGljYXRpb24KY2FuIHVzZSB3aXRoIGNvbmZpZGVuY2UuCgpJdCBpcyBub3QgaW50ZW5kZWQgdG8gYmUgYSBmdWxseSBmZWF0dXJlZCB2YWxpZGF0aW9uIGxpYnJhcnksIGJ1dCBpbnN0ZWFkIGZvY3VzIG9uIGVuc3VyaW5nIHRoYXQgdGhlIGRhdGEgeW91IGhhdmUKcmVjZWl2ZWQgbG9va3MgY29ycmVjdCBieSBmb2N1c2luZwpvbiBkYXRhIHR5cGVzLCByYW5nZXMgYW5kIGZvcm1hdHMuCgpUbyBwdXQgdGhpcyBpbiBjb250ZXh0LCBtZW1icmFuZSBjb3VsZCBiZSB1c2VkIHRvIHZhbGlkYXRlIHRoYXQgYSB1c2VyIHN1Ym1pdHRpbmcgYSByZXF1ZXN0IHRvIGVkaXQgYSBibG9nIHBvc3QgaGFzOgoKLSBhbiBpbnRlZ2VyIGlkLCBncmVhdGVyIHRoYW4gemVybzsKLSBhIHN0cmluZyB0aXRsZSBiZXR3ZWVuIDEwIGFuZCA4MCBjaGFyYWN0ZXJzOwotIGEgc3RyaW5nIHBvc3QgYm9keSBhdCBsZWFzdCAyNSBjaGFyYWN0ZXJzIGxvbmc7Ci0gYSBsaXN0IG9mIG5vIG1vcmUgdGhhbiA1IHN0cmluZyB0YWdzLCBlYWNoIHdpdGggYSBsZW5ndGggYmV0d2VlbiAzIGFuZCAxNSBjaGFyYWN0ZXJzLgoKSG93ZXZlciwgaXQgc2hvdWxkIG5vdCBiZSB1c2VkIHRvIHZhbGlkYXRlIHRoYXQ6CgotIHRoZSBibG9nIHBvc3QgYWxyZWFkeSBleGlzdHM7Ci0gdGhhdCB0aGUgdXNlciBoYXMgcGVybWlzc2lvbiB0byBlZGl0IHRoZSBibG9nIHBvc3Q7Ci0gdGhhdCB0aGUgYmxvZyBwb3N0IGlzIHVucHVibGlzaGVkIGFuZCB0aHVzIHN0aWxsIGFsbG93ZWQgdG8gYmUgZWRpdGVkLgoKVGhlc2UgYXJlIGFsbCBidXNpbmVzcyBydWxlcyB3aGljaCBzaG91bGQgYmUgdGFrZW4gY2FyZSBvZiBpbiBhIHNlcGFyYXRlIGxheWVyLgoKTWVtYnJhbmUgd2lsbCB1c3VhbGx5IGJlIHlvdXIgZmlyc3QgbGluZSBiZXR3ZWVuIGEgd2ViIHJlcXVlc3QgYW5kIHlvdXIgYXBwbGljYXRpb24sIHNvIEkgaGF2ZSBtYWRlIHRoZSBkZXNpZ24gZGVjaXNpb24KdGhhdCBtZW1icmFuZSB3aWxsICpORVZFUiogdGhyb3cgYW4KZXhjZXB0aW9uIGJlY2F1c2Ugb2YgYmFkIHVzZXIgZGF0YS4gRXhjZXB0aW9ucyB3aWxsIG9ubHkgYmUgdGhyb3duIGR1ZSB0byBtZW1icmFuZSBiZWluZyBzZXR1cCBpbmNvcnJlY3RseSBieSB0aGUKZGV2ZWxvcGVyLiBBcyBzdWNoLCB5b3Ugd2lsbCBhbHdheXMgZ2V0IGEKcmVzdWx0IG9iamVjdCBiYWNrIHdoZW4gdmFsaWRhdGluZyB1c2VyIGlucHV0IGV2ZW4gaWYgdGhlIGlucHV0IHdhcyBjb21wbGV0ZSBnYXJiYWdlLgo= readmeEtag: '"fb9d6c89006ebdfbf89771a413a72160129e683a"' readmeLastModified: Thu, 25 Apr 2024 16:02:54 GMT repositoryId: 485017464 description: >- Membrane is a general purpose input validation library, supports both PHP Attributes and OpenAPI specifications created: '2022-04-24T12:10:16Z' updated: '2025-11-18T14:56:51Z' language: PHP archived: false stars: 6 watchers: 2 forks: 4 owner: membrane-php logo: https://avatars.githubusercontent.com/u/104306108?v=4 license: NOASSERTION repoEtag: '"43eb6e56b9c6a88eb78db74714d792a544bb06fc197d266a27ea512722607ce0"' repoLastModified: Tue, 18 Nov 2025 14:56:51 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/matthewtrask/disapi v3: true repositoryMetadata: base64Readme: >- IyMgRGlzQVBJCgojIyMgVGhpcyBpcyBhIFdJUCwgdXNlIGF0IHlvdXIgb3duIHJpc2suCgpBIFJFU1RmdWwgT3BlbkFQSSBzcGVjIEFQSSBhcm91bmQgdGhlIFdhbHQgRGlzbmV5IFdvcmxkIFJlc29ydCAKCiMjIyMgVG8gc2VlIFJlRG9jOiAKYGBgCnlhcm4gcnVuIHNwZWNjeSBzZXJ2ZSBzcGVjcy9vcGVuYXBpLnltbApgYGAKCkFuZCB0aGVuIGdvIHRvIGxvY2FsaG9zdDo1MDAwIHRvIHNlZSB0aGUgZG9jdW1lbnRhdGlvbgoKIyMjIyBFbmRwb2ludHMKCldvcmsgd2l0aCB0aGUgUGFya3MgUmVzb3VyY2UgQ29sbGVjdGlvbgpgYGBwaHAKICAgIEdFVC9QT1NUIC9wYXJrcwpgYGAKYGBgcGhwCiAgICBHRVQvUFVUL0RFTEVURSAvcGFya3Mve2lkfQpgYGAKCldvcmsgd2l0aCB0aGUgUmlkZXMgUmVzb3VyY2UgQ29sbGVjdGlvbgpgYGBwaHAKICAgIEdFVC9QT1NUIC9yaWRlcwpgYGAKYGBgcGhwCiAgICBHRVQvUFVUL0RFTEVURSByaWRlcy97aWR9CmBgYAoKV29yayB3aXRoIHRoZSBSZXN0YXVyYW50cyBSZXNvdXJjZSBDb2xsZWN0aW9uCmBgYHBocAogICAgR0VUL1BPU1QgL3Jlc3RhdXJhbnRzCmBgYApgYGBwaHAKICAgIEdFVC9QVVQvREVMRVRFIC9yZXN0YXVyYW50cwpgYGAKCldvcmsgd2l0aCB0aGUgUmVzb3J0cyBSZXNvdXJjZSBDb2xsZWN0aW9uCmBgYHBocAogICAgR0VUL1BPU1QgL3Jlc29ydHMKYGBgCmBgYHBocAogICAgR0VUL1BVVC9ERUxFVEUgL3Jlc3RhdXJhbnRzL3tpZH0KYGBgCgojIyMjIFJlcXVpcmVtZW50cwoKKiBDb21wb3NlcgoqIFlhcm4KCiMjIyMgSW5zdGFsbGF0aW9uCgpJZiB5b3Ugd2FudCB0byBpbnN0YWxsIGEgbG9jYWwgY29weSBvZiB0aGlzIHByb2plY3QKCiogQ2xvbmUgdG8geW91ciBsb2NhbAoqIENkIHRvIHByb2plY3QsIGFuZCBjb21wb3NlciBpbnN0YWxsIHRoZSBkZXBlbmRlbmNpZXMKKiBUbyB1c2UgSG9tZXN0ZWFkLCBlZGl0IHRoZSBwYXRoIGluIEhvbWVzdGVhZC55YW1sIHRvIHJlZmxlY3QgeW91ciBsb2NhbCBhbmQ6IApgYGBwaHAKJCAuL3ZlbmRvci9iaW4vaG9tZXN0ZWFkIG1ha2UgJiYgdmFncmFudCB1cApgYGAKKiBFZGl0IHlvdXIgL2V0Yy9ob3N0cyAobm90IHN1cmUgYWJvdXQgd2luZG93cykgYW5kIGFkZCBgYGAxOTIuMTY4LjEwLjEwIGhvbWVzdGVhZC50ZXN0YGBgIG9yIHdoYXRldmVyIHlvdSB3YW50IGl0IHRvIGJlLgoqIGBgYHZhZ3JhbnQgc3NoYGBgIGFuZCBjZCB0byByb290IG9mIHByb2plY3QKKiBJbnN0YWxsIEpTIGRlcGVuZGVuY2llcyB3aXRoIFlhcm4gKG5lZWRlZCBmb3IgU3BlY2N5KQoqIE1pZ3JhdGUgYW5kIFNlZWQgdGhlIERhdGFiYXNlIHdpdGggYGBgcGhwIGFydGlzYW4gbWlncmF0ZSAmJiBwaHAgYXJ0aXNhbiBkYjpzZWVkYGBgCiogQWx0ZXJuYXRpdmVseSwgeW91IGNvdWxkIGltcG9ydCB0aGUgZGF0YS5zcWwgZmlsZSBpbnRvIHlvdXIgbG9jYWwgbXlzcWwgaW5zdGFuY2UKKiBPcGVuIFBvc3RtYW4gYW5kIGhpdCB0aGUgZW5kcG9pbnRzIGFib3ZlIHRvIGVuc3VyZSBpdCB3b3Jrcy4gCgpJZiB5b3Ugc2VlIGFuIGlzc3VlLCBoaXQgbWUgdXAgb24gdHdpdHRlciBAbWF0dGhld3RyYXNr readmeEtag: '"bfc1c5ddebf302bf594543e315688fb420a70997"' readmeLastModified: Thu, 28 Mar 2019 02:05:42 GMT repositoryId: 144517308 description: An API around Walt Disney World Resort. created: '2018-08-13T02:02:16Z' updated: '2019-03-28T02:05:45Z' language: PHP archived: false stars: 4 watchers: 1 forks: 2 owner: matthewtrask logo: https://avatars.githubusercontent.com/u/4731244?v=4 repoEtag: '"47db64ccc2b971b7410a07236912bd5285690365d29314ebb335e0da7080da44"' repoLastModified: Thu, 28 Mar 2019 02:05:45 GMT foundInMaster: true category: - Testing - Server Implementations id: 95241dd18a6bfddc4e65c4c7becfc3bf - source: openapi3 tags repository: https://github.com/tidepool-org/tidepoolapi v3: true id: a7c2fef8f920a4385bf8aae75ba61407 repositoryMetadata: base64Readme: >- IyBUaWRlcG9vbEFwaSAKCiFbVGlkZXBvb2wgTG9nb10oLi9hc3NldHMvaW1hZ2VzL1RpZGVwb29sX0xvZ29fRGFya19MYXJnZS5wbmcpCgpbIVtwdWJsaXNoXShodHRwczovL2dpdGh1Yi5jb20vdGlkZXBvb2wtb3JnL1RpZGVwb29sQXBpL2FjdGlvbnMvd29ya2Zsb3dzL2NoZWNrLWFuZC1wdWJsaXNoLnltbC9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vZ2l0aHViLmNvbS90aWRlcG9vbC1vcmcvVGlkZXBvb2xBcGkvYWN0aW9ucy93b3JrZmxvd3MvY2hlY2stYW5kLXB1Ymxpc2gueW1sKQoKVGhpcyByZXBvc2l0b3J5IGNvbnRhaW5zIFRpZGVwb29sIFBsYXRmb3JtIEFQSSBkb2N1bWVudGF0aW9uIGluIFtPcGVuQVBJIHYzXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKSBmb3JtYXQgd2l0aCBhZGRpdGlvbmFsIG5hcnJhdGl2ZSBjb250ZW50IGluIFtNYXJrZG9jXShodHRwczovL3JlZG9jbHkuY29tL2xlYXJuL21hcmtkb2Mvd3JpdGUtd2l0aC1tYXJrZG9jKSBmb3JtYXQgd2hpY2ggaW4gdHVybiBleHRlbmRzIG9uIFtNYXJrZG93bl0oaHR0cHM6Ly93d3cubWFya2Rvd25ndWlkZS5vcmcvKS4KVGhlc2UgQVBJIGRlZmluaXRpb25zIGNhbiBiZSB1c2VkIHRvIGdlbmVyYXRlIHN0dWIgY29kZSBmb3IgZWl0aGVyIHNlcnZlciBvciBjbGllbnQgc2lkZS4gQ3VycmVudGx5LCB3ZSBvbmx5IGRvIHRoaXMgd2l0aCB0aGUgYGNsaW5pY2Agc2VydmljZSAoc2VlIGJlbG93IGZvciBzcGVjaWFsIG5vdGUgb24gdGhhdCkuCgojIyBXb3JrZmxvdwoKV2UgaGF2ZSBhbiBhY2NvdW50IGluIFtSZWRvY2x5XShodHRwczovL3RpZGVwb29sLnJlZG9jbHkuYXBwKSBmb3IgcHVibGlzaGluZyB0aGUgZG9jdW1lbnRhdGlvbiBhbmQgQVBJIHNwZWNpZmljYXRpb25zLiBUaGUgZ29hbCBpcyBmb3IgdGhhdCBzaXRlIHRvIHNvb24gcmVwbGFjZSB0aGUgY3VycmVudCBbZGV2ZWxvcGVyIHBvcnRhbF0oaHR0cHM6Ly9kZXZlbG9wZXIudGlkZXBvb2wub3JnKS4gSXQgKmlzKiBwb3NzaWJsZSB0byBlZGl0IHRoZSBkb2N1bWVudGF0aW9uIGZpbGVzIGRpcmVjdGx5IG9ubGluZSBpbiBSZWRvY2x5IC0gaG93ZXZlciB3ZSBhcmUgKm5vdCogdXNpbmcgdGhhdCBjYXBhYmlsaXR5LCBmb3Igc2V2ZXJhbCByZWFzb25zOgoKMS4gRG9pbmcgc28gcmVxdWlyZXMgdXNpbmcgb25lIG9mIGEgbGltaXRlZCBudW1iZXIgb2Ygc2VhdHMgKDUpLgoyLiBJdCBkb2VzIG5vdCBmaXQgd2VsbCBpbnRvIG91ciBub3JtYWwgcmV2aWV3L2FwcHJvdmFsIHdvcmtmbG93LgozLiBJdCBkb2VzIG5vdCBlbmFibGUgdXMgdG8gcnVuIHByZS1tZXJnZSBjaGVja3MsIGluIHBhcnRpY3VsYXIgdG8gdmFsaWRhdGUgY2hhbmdlcyB0aGF0IGNvdWxkIGJyZWFrIGdlbmVyYXRlZCBzZXJ2ZXIgc3R1YiBjb2RlIHN1Y2ggYXMgdGhlIFtgY2xpbmljYF0oaHR0cHM6Ly9naXRodWIuY29tL3RpZGVwb29sLW9yZy9jbGluaWMpIHNlcnZpY2UuIFNlZSB0aGUgYGNoZWNrYCBhbmQgYGdlbmVyYXRlYCBzdGVwcyBpbiB0aGUgW01ha2VmaWxlXSguL01ha2VmaWxlKS4KClRoZSBwcmVmZXJyZWQgd29ya2Zsb3cgaXMgdG8gZWRpdCB0aGVzZSBmaWxlcyBvZmZsaW5lLCB0aGVuIGNvbW1pdCB0aGVtIHRvIEdpdEh1YiB3aGljaCBhdXRvbWF0aWNhbGx5IHB1c2hlcyB1cGRhdGVzIGludG8gUmVkb2NseSwgaW5jbHVkaW5nIHdvcmsgYnJhbmNoZXMgdGhvdWdoIHRoZXkgYXJlIG5vdCB2aXNpYmxlIHRvIHBlb3BsZSBvdXRzaWRlIG9mIFRpZGVwb29sLiBIZXJlIGFyZSB0aGUgd29ya2Zsb3cgZGV0YWlsczoKCjEuIENsb25lIHRoaXMgcmVwb3NpdG9yeSBhbmQgaW5zdGFsbCB0aGUgdmFsaWRhdGlvbiAmIHB1Ymxpc2hpbmcgdG9vbHMuIFlvdSBvbmx5IG5lZWQgdG8gdG8gdGhpcyBvbmNlLgoKICAgIGBgYHNoZWxsCiAgICBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL3RpZGVwb29sLW9yZy9UaWRlcG9vbEFwaS5naXQKICAgIGNkIFRpZGVwb29sQXBpCiAgICBtYWtlIGluc3RhbGxfdG9vbHMKICAgIGBgYAoKMi4gQ3JlYXRlIGEgKndvcmsgYnJhbmNoKiBmb3IgeW91ciBlZGl0czoKCiAgICBgYGBzaGVsbAogICAgZ2l0IGNoZWNrb3V0IG1hc3RlcgogICAgZ2l0IHB1bGwKICAgIGdpdCBjaGVja291dCBtYXN0ZXIgLWIge2JyYW5jaH0KICAgIGBgYAoKMy4gRWRpdCBhbmQgcHJldmlldyB0aGUgZmlsZXMgb2ZmbGluZSB1c2luZyBhbnkgb2YgdGhlIFt0b29sc10oI2VkaXRpbmctdG9vbHMpIGxpc3RlZCBiZWxvdy4KNC4gVmFsaWRhdGUgeW91ciBjaGFuZ2VzIGxvY2FsbHk6CgogICAgYGBgc2hlbGwKICAgIG1ha2UgY2hlY2sKICAgIG1ha2UgcHJlcGFyZQogICAgYGBgCgo1LiBDb21taXQgeW91ciBjaGFuZ2VzIHRvIHRoZSB3b3JrIGJyYW5jaCBhbmQgcHVzaCB0byBHaXRIdWIuIFRoaXMgd2lsbCBhdXRvbWF0aWNhbGx5IGtpY2sgb2ZmIGEgW0dpdEh1YiBBY3Rpb25dKC5naXRodWIvd29ya2Zsb3dzLykgdGhhdCBjaGVja3MgJiBwdWJsaXNoZXMgdGhlIG5ldyAoZHJhZnQpIGRvY3VtZW50YXRpb24gaW50byBhIGJyYW5jaCBpbiBSZWRvY2x5LgoKICAgIGBgYHNoZWxsCiAgICBnaXQgYWRkIHtmaWxlKHMpfQogICAgZ2l0IGNvbW1pdCAtbSAie2Rlc2NyaXB0aW9ufSIKICAgIGdpdCBjaGVja291dCBtYXN0ZXIKICAgIGdpdCBwdWxsCiAgICBnaXQgY2hlY2tvdXQge2JyYW5jaH0KICAgIGdpdCBtZXJnZSAtLW5vLWZmIG1hc3RlcgogICAgZ2l0IHB1c2ggLXUgb3JpZ2luIEhFQUQKICAgIGBgYAoKNi4gUG9zdCBhIHB1bGwgcmVxdWVzdCAoUFIpIHRvIHJldmlldyB0aGUgY2hhbmdlcwo3LiBPbmNlIHRoZSBQUiBpcyBhcHByb3ZlZCwgbWVyZ2UgdG8gbWFzdGVyIHdoaWNoIHdpbGwgYWdhaW4gYXV0b21hdGljYWxseSB1cGRhdGUgdGhlIG1hc3RlciBicmFuY2ggaW4gdGhlIFJlZG9jbHkgc2l0ZS4KCiMjIEVkaXRpbmcgVG9vbHMKCjEuIEZyZWUgW1N0b3BsaWdodCBTdHVkaW9dKGh0dHBzOi8vc3RvcGxpZ2h0LmlvL3N0dWRpby8pIGZvciB2YWxpZGF0aW5nIGFuZCByZW5kZXJpbmcgT3BlbkFQSSB2MyBzcGVjaWZpY2F0aW9ucy4KMi4gRnJlZSBNaWNyb3NvZnQgW1Zpc3VhbCBTdHVkaW8gQ29kZV0oaHR0cHM6Ly9jb2RlLnZpc3VhbHN0dWRpby5jb20vKSwgd2l0aCBwbHVnLWlucyBmb3IgdmFsaWRhdGluZyBhbmQgcmVuZGVyaW5nIE9wZW5BUEkgdjMgc3BlY2lmaWNhdGlvbnMgYW5kIE1hcmtkb3duIGRvY3VtZW50YXRpb24uCjMuIE1hbnkgb3RoZXIgSURFcyBvZmZlciBzaW1pbGFyIHBsdWctaW5zLgoKIyMgT3RoZXIgVG9vbHMKClRoZSBbTWFrZWZpbGVdKC4vTWFrZWZpbGUpIG1ha2VzIHVzZSBvZiBzZXZlcmFsIENMSSB0b29scyB0byBjaGVjaywgcHJlcGFyZSwgYW5kIHB1Ymxpc2ggdGhlIGRvY3VtZW50YXRpb24gYW5kIHNwZWNpZmljYXRpb25zLgpZb3UgY2FuIGluc3RhbGwgdGhlIHRvb2xzIGJ5IGV4ZWN1dGluZyB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBzaGVsbAptYWtlIGluc3RhbGxfdG9vbHMKYGBgCgpZb3UgY2FuIGNoZWNrIGlmIHlvdSBoYXZlIGFsbCB0aGUgdG9vbHMgaW5zdGFsbGVkIGJ5IGV4ZWN1dGluZyB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBzaGVsbAptYWtlIGNoZWNrX3Rvb2xzCmBgYAoKfCBUb29sIHwgRGVzY3JpcHRpb24gfAp8LS0tLS0tfC0tLS0tLS0tLS0tLS18CnwgW21hcmtkb3dubGludF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvbWFya2Rvd25saW50KSB8IFZhbGlkYXRlcyBNYXJrZG93biBmaWxlcy4gfAp8IFttYXJrZG93bi1saW5rLWNoZWNrXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9tYXJrZG93bi1saW5rLWNoZWNrKSB8IFZhbGlkYXRlcyBoeXBlcmxpbmtzIGluIE1hcmtkb3duIGZpbGVzLiB8CnwgW3NwZWN0cmFsXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9Ac3RvcGxpZ2h0L3NwZWN0cmFsKSB8IFZhbGlkYXRlcyBPcGVuQVBJIDMuMCBzcGVjaWZpY2F0aW9uIGZpbGVzLiB8CnwgW3N3YWdnZXItY2xpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9zd2FnZ2VyLWNsaSkgfCBWYWxpZGF0ZXMgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbiBmaWxlcy4gQWxzbyBidW5kbGVzIG11bHRpcGxlIE9BUzMgZmlsZXMgaW50byBhIHNpbmdsZSBmaWxlLCB0aGF0IGlzIHJlcXVpcmVkIGJ5IHNvbWUgZG93bnN0cmVhbSB1c2UtY2FzZXMuIHwKfCBbcmVkb2NseV0oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvcmVkb2NseS1jbGkpIHwgVmFsaWRhdGVzIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gZmlsZXMuIEFsc28gYnVuZGxlcyBtdWx0aXBsZSBPQVMzIGZpbGVzIGludG8gYSBzaW5nbGUgZmlsZSwgdGhhdCBpcyByZXF1aXJlZCBieSBzb21lIGRvd25zdHJlYW0gdXNlLWNhc2VzLiB8CnwgW29wZW5hcGktbWVyZ2UtY2xpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLW1lcmdlLWNsaSkgfCBNZXJnZXMgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbiBmaWxlcyBpbnRvIHNpbmdsZSBmaWxlLiB8CnwgW29hcGktY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL2RlZXBtYXAvb2FwaS1jb2RlZ2VuKSB8IEdlbmVyYXRlcyBzZXJ2ZXIgYW5kIGNsaWVudCBzdHViIGNvZGUgZnJvbSBPcGVuQVBJIDMuMCBzcGVjaWZpY2F0aW9ucy4gVXNlZCBjdXJyZW50bHkgdG8gZ2VuZXJhdGUgdGhlIFtgY2xpbmljYF0oaHR0cHM6Ly9naXRodWIuY29tL3RpZGVwb29sLW9yZy9jbGluaWMpIHNlcnZpY2UgY29kZS4gfAo= readmeEtag: '"942c1c926904cc6f0967d7a0a1f1fc15c2a06569"' readmeLastModified: Thu, 11 Sep 2025 15:09:49 GMT repositoryId: 239433889 description: >- Tidepool APIs documented with OpenAPI v3, for schema validation and client and server code generation. created: '2020-02-10T05:20:54Z' updated: '2026-02-05T02:14:58Z' language: Shell archived: false stars: 9 watchers: 12 forks: 3 owner: tidepool-org logo: https://avatars.githubusercontent.com/u/4390820?v=4 license: BSD-2-Clause repoEtag: '"50a7902a9a42c5676d4bd983f6bf136fde64af66083e7795a326f514d098d9aa"' repoLastModified: Thu, 05 Feb 2026 02:14:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/lschlesinger/the-game v3: true repositoryMetadata: base64Readme: >- 
<!-- PROJECT LOGO -->
<br />

<p align="center">
  <a href="https://github.com/lschlesinger/the-game">
    <img src="game-web-ui/src/assets/images/logo.png" alt="Logo" width="500">
  </a>
  <p align="center">
    Swift implementation of the card game "The Game", by Steffen Benndorf using the <a href="https://github.com/Apodini/Apodini">Apodini Framework</a>.
    The Endangered Nature Edition is based on an environmental education concept, which aims to raise awareness of the current threats to certain species.
      </p>
    <br/>
  </p> 
</p>

<!-- TABLE OF CONTENTS -->

## Table of Contents

* [About the Project](#about-the-project)
  * [Motivation](#motivation)
  * [Technical Background](#technical-background)
  * [Built With](#built-with)
* [Getting Started](#getting-started)
* [Critical Reflection of the Software](#critical-reflection)
* [License](#license)

<!-- ABOUT THE PROJECT -->

## About the Project

The current Corona situation forces us in many cases to limit ourselves to digital contact and digital communication alone. Unfortunately, in times of physical distance and isolation, inviting a large group of people over for a fun evening with card or board games is more a memory of an earlier time than part of our current weekend plan - and we miss that! That is why we came up with the idea to implement our favorite card game as an online game. We can now play together with a group of friends and our families, spread all over the country.

Find more information about the original game [here](http://middys.nsv.de/middys/the-game/).
This is how our edition looks like:
<p align="center">
  <img src="game-web-ui/src/assets/images/game-example.png" alt="Logo" width="700">
</p>

We hope you have fun playing it! Please find acknowledgement as well as the source of the pictures used [here](Documentation/Acknowledgement.md).

### Motivation

Biodiversity is the basis for functioning ecosystems, on which we humans ourselves ultimately depend. Fortunately, the contribution of nature to a good quality of life is nowadays increasingly valued. Nevertheless, anthropogenic activities are destroying species habitats and thus fueling species extinction unabated.

The idea behind the card design is to raise awareness for [Nature’s Red List of Threatened Species](https://www.iucnredlist.org/about/background-history). 
The mere sight of the charming representatives of the vulnerable and endangered species should ideally contribute to advocating a more nature-friendly way of life.

### Technical Background

By using the [Apodini Framework](https://github.com/Apodini/Apodini), this sample project tackles the following problem:
<p align="center">
  <img src="./Documentation/info-material/Apodini-Problem-Statement.png" alt="Logo" width="600">
</p>

You can find a step-by-step manual on how to accomplish building a `GameWebService` with Apodini using OAS [here](Documentation/README.md).

### Built With

* [Apodini](https://github.com/Apodini/Apodini)
* [Swagger Codegen](https://github.com/swagger-api/swagger-codegen) using [OpenAPI Specification 3.0.3](https://swagger.io/specification/)
* [Angular](https://angular.io/)
* [Ant Design](https://ant.design/docs/react/introduce)
* [Docker](https://www.docker.com/)
* [NGINX](https://www.nginx.com/)

<!-- GETTING STARTED -->

## Getting Started

### Docker Setup

#### Requirements

* Docker
* Docker Compose
* Swagger codegen

#### Run

To get a local copy up and running follow these simple steps:

1. Clone the repo

```sh
git clone git@github.com:lschlesinger/the-game.git
```
2. Start backend and nginx which serves frontend

```sh
docker-compose up
```

3. Visit `http://localhost:80`

### Development Setup

#### Requirements

* Node.js
* npm
* Swift

#### Run

To get a local copy up and running follow these simple steps:

1. Clone the repo

```sh
git clone git@github.com:lschlesinger/the-game.git
```
2. Start backend

```sh
swift run
```
3. Start frontend (in separate terminal)

```sh
cd game-web-ui
npm install # only the first time
npm run start
```

4. Visit `http://localhost:4200` for frontend (all requests to backend will be proxied)
	1. Visit `http://localhost:8080/doc/openapi` (if not configured differently) to see the generated OpenAPI specification
	2. Visit `http://localhost:8080/ui/swagger` (if not configured differently) to explore the web app with Swagger-UI


<!-- CRITICAL REFLECTION -->

## Critical Reflection of the Software

Browsing the web, people with disabilitites often rely on assistive technologies, such as screen readers, magnification software, text readers, head pointers, and motion or eye trackers to access content. As this project does not include features to support these assistive technologies so far, the software is not barrier-free for all users. 
In the future, the accessibility of the online game should be guaranteed for all people, no matter where these people come from, what language they speak, what technology they use or what social background they have. That is why we are working to achieve these [Web Content Accessibility Guidelines (WCAG) 2.0](https://www.w3.org/TR/WCAG20/) in the long term. 

Moreover, the due to the mission that should be transported by the use of pictures, the page weight has potential for optimization. A generated [Page Weight](https://pageweight.imgix.com/) report suggests, that there are 83% potential savings in reducing image sizes. The current page weight of the online games does not only bear the risk of, slow load times, impeded performance, and potentially wasted energy. According to the digital agency [Mightybytes](https://www.mightybytes.com/blog/page-weight-budget-to-speed-up-your-site/) "this can frustrate many users, especially those on older mobile devices or in rural areas with restricted bandwidth". In the future, the online game should bring it's page weight to a minimum and thereby reducing the digital footprint.

In their blog post series [Sustainable Web Design](https://www.mightybytes.com/blog/sustainable-web-design/) Mightybytes introduce sustainable web design principles that generally focus on reducing electricity use but also cover the use of ‘green’ ingredients, such as clean energy-powered web hosting, for example.
Having in mind that web designers can make a measurable difference to decrease the carbon footprint of the Internet, we should face the challenge of making the online game adhereing to the sustainble web design principles as far as possible in the future.

<!-- LICENSE -->

## License

Distributed under the GPL-3.0 License. See `LICENSE` for more information.

## Contributors

:two_women_holding_hands: L&L Schlesinger
 readmeEtag: '"279e9dc1d30dd22089cf807314b1dd9a6923e1c5"' readmeLastModified: Wed, 10 Mar 2021 19:38:47 GMT repositoryId: 332280786 description: >- Swift implementation of the card game "The Game", by Steffen Benndorf using the Apodini Framework, Angular, OAS3.0 and swagger-codegen created: '2021-01-23T18:30:16Z' updated: '2023-01-11T08:34:03Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 0 owner: lschlesinger logo: https://avatars.githubusercontent.com/u/41366522?v=4 license: GPL-3.0 repoEtag: '"f3999988bd15e5c109c9cf901a625cead3f54c7397dd37be4dc48fbf4d050ef0"' repoLastModified: Wed, 11 Jan 2023 08:34:03 GMT foundInMaster: true category: - Documentation - Server Implementations id: 429f4d721f8935dd2cdc97e743793b7f - source: openapi3 tags repository: https://github.com/berislavlopac/pyapi-server v3: true id: b5e0fb24b4b1c517abdf5cd50ec99acc repositoryMetadata: base64Readme: >- IyBQeUFQSSBTZXJ2ZXIKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9iMTFjLnNlbWFwaG9yZWNpLmNvbS9iYWRnZXMvcHlhcGktc2VydmVyL2JyYW5jaGVzL21haW4uc3ZnP3N0eWxlPXNoaWVsZHMma2V5PWU5ZWViOWQyLTY0ODctNGFiYS05MjA3LWU0NmM4NGY5YmM2ZildKGh0dHBzOi8vYjExYy5zZW1hcGhvcmVjaS5jb20vcHJvamVjdHMvcHlhcGktc2VydmVyKQpbIVtEb2N1bWVudGF0aW9uIFN0YXR1c10oaHR0cHM6Ly9yZWFkdGhlZG9jcy5vcmcvcHJvamVjdHMvcHlhcGktc2VydmVyL2JhZGdlLz92ZXJzaW9uPWxhdGVzdCldKGh0dHBzOi8vcHlhcGktc2VydmVyLnJlYWR0aGVkb2NzLmlvL2VuL2xhdGVzdC8/YmFkZ2U9bGF0ZXN0KQoKKipQeUFQSSBTZXJ2ZXIqKiBpcyBhIFB5dGhvbiBsaWJyYXJ5IGZvciBzZXJ2aW5nIFJFU1QgQVBJcyBiYXNlZCBvbgpbT3BlbkFQSV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3Jlc291cmNlcy9vcGVuLWFwaS8pIHNwZWNpZmljYXRpb25zLiBJdCBpcyBiYXNlZCBvbiBbU3RhcmxldHRlXShodHRwczovL3d3dy5zdGFybGV0dGUuaW8pIGFuZCBpcyBmdW5jdGlvbmFsbHkgdmVyeSBzaW1pbGFyIHRvIFtjb25uZXhpb25dKGh0dHBzOi8vY29ubmV4aW9uLnJlYWR0aGVkb2NzLmlvKSwgZXhjZXB0IHRoYXQgaXQgYWltcyB0byBiZSBmdWxseSBbQVNHSV0oaHR0cHM6Ly9hc2dpLnJlYWR0aGVkb2NzLmlvKS1jb21wbGlhbnQuCgoqKldBUk5JTkc6KiogVGhpcyBpcyBzdGlsbCBhIHdvcmsgaW4gcHJvZ3Jlc3MgYW5kIG5vdCBxdWl0ZSByZWFkeSBmb3IgcHJvZHVjdGlvbiB1c2FnZS4gVW50aWwgdmVyc2lvbiAxLjAgaXMgcmVsZWFzZWQsIGFueSBuZXcgcmVsZWFzZSBjYW4gYmUgZXhwZWN0ZWQgdG8gYnJlYWsgYmFja3dhcmQgY29tcGF0aWJpbGl0eS4KCgojIyBRdWljayBTdGFydAoKYGBgcHl0aG9uCmZyb20gcHlhcGkuc2VydmVyIGltcG9ydCBBcHBsaWNhdGlvbgpmcm9tIHNvbWUucGF0aCBpbXBvcnQgZW5kcG9pbnRzCgphcHAgPSBBcHBsaWNhdGlvbi5mcm9tX2ZpbGUoInBhdGgvdG8vb3BlbmFwaS55YW1sIiwgbW9kdWxlPWVuZHBvaW50cykKYGBgCg== readmeEtag: '"e330bf28b91a7cd64bdb084322671787282f5380"' readmeLastModified: Thu, 25 Jul 2024 14:14:16 GMT repositoryId: 524173113 description: >- Lightweight API framework using an OpenAPI spec for routing and validation. created: '2022-08-12T17:32:43Z' updated: '2026-01-22T12:12:58Z' language: Python archived: false stars: 5 watchers: 1 forks: 1 owner: berislavlopac logo: https://avatars.githubusercontent.com/u/754090?v=4 repoEtag: '"b2c09bc978dde2497539d386fd55c4cb4f06e761840d16ed40da6d75bc67fae4"' repoLastModified: Thu, 22 Jan 2026 12:12:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/burgalon/fezto v3: true id: 7baea5051107a6d799f91ea77fbfbe59 repositoryMetadata: base64Readme: >- IyBGcm9udGVuZCBaZXJvIHRvIE9uZQpBbiBhdXRvbWF0aWMgb24tdGhlLWZseSBhcHAgZ2VuZXJhdG9yIGZyb20gT3BlbkFQSSAzLjAgc3BlYyAob3IgU3dhZ2dlciAyLjApIAoKRkVaVE8gcGFyc2VzIHRoZSBPcGVuQVBJIHNwZWMgYW5kIHRyaWVzIHRvIGRldGVybWluZSB3aGljaCByZXNvdXJjZXMgYXJlIGF2YWlsYWJsZSB0byBwcmVzZW50LCBhbmQgY3JlYXRlcyBDUlVEIHNjcmVlbiBmb3IgY3JlYXRpbmcsIHJlYWRpbmcgYW5kIHVwZGF0aW5nIGVudGl0aWVzLiBJdCBhbHNvIHRyaWVzIHRvIHJlc29sdmUgZm9yZWlnbiBrZXlzIHRvIGNyZWF0ZSBkcm9wZG93bnMgYW5kIGFkZCBsaW5rYWJsZSByZXNvdXJjZXMuCgo= readmeEtag: '"24a49c25bac7949f72d53910052d0734574f512a"' readmeLastModified: Thu, 26 May 2022 09:00:36 GMT repositoryId: 496548279 description: Frontend Zero to One created: '2022-05-26T08:56:21Z' updated: '2025-01-06T11:03:54Z' language: null archived: false stars: 3 watchers: 1 forks: 0 owner: burgalon logo: https://avatars.githubusercontent.com/u/47217?v=4 repoEtag: '"f5cd9eb9334b7cbb6c87536803363ab568fcc4b653990f7b084e56a53b6077ce"' repoLastModified: Mon, 06 Jan 2025 11:03:54 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/cloudy-sky-software/pulumi-provider-template v3: true id: 5e5e71c5d4ff048e6be5da2103af3c92 repositoryMetadata: base64Readme: >- # Pulumi Provider Template For OpenAPI-based Providers

This repository is a boilerplate showing how to create a native Pulumi provider using:

1. [`pulschema`](https://github.com/cloudy-sky-software/pulschema) to convert an OpenAPI spec to Pulumi schema
2. [`pulumi-provider-framework`](https://github.com/cloudy-sky-software/pulumi-provider-framework) to make HTTP requests against the cloud provider API. It uses the metadata returned by `pulschema` as one of the outputs from
   converting an OpenAPI spec.

## Background

Follow this link to see what a [Pulumi native provider is comprised of](https://github.com/cloudy-sky-software/cloud-provider-api-conformance/blob/main/pulumi.md).

A Pulumi Resource Provider:

- is a gRPC server which allows for the Pulumi engine to create resources in a specific cloud
- holds the lifecycle logic for these cloud resources
- holds a pulumi JSON schema that describes the provider
- provides language-specific SDKs so resources can be created in whichever language you prefer

When we speak of a "native" provider, we mean that all implementation is native to Pulumi, as opposed
to [Terraform-based providers](https://github.com/pulumi/pulumi-tf-provider-boilerplate).

## Authoring a Pulumi Native Provider

This boilerplate creates a working Pulumi-owned provider named `xyz`.

### Prerequisites

Ensure the following tools are installed and present in your `$PATH`:

- [`pulumictl`](https://github.com/pulumi/pulumictl#installation)
- [Go 1.21](https://golang.org/dl/) or 1.latest
- [NodeJS](https://nodejs.org/en/) 18.x. We recommend using [nvm](https://github.com/nvm-sh/nvm) to manage NodeJS installations.
- [Yarn](https://yarnpkg.com/)
- [TypeScript](https://www.typescriptlang.org/)
- [Python](https://www.python.org/downloads/) (called as `python3`)
- [.NET](https://dotnet.microsoft.com/download)

### Creating and Initializing the Repository

Pulumi offers this repository as a [GitHub template repository](https://docs.github.com/en/repositories/creating-and-managing-repositories/creating-a-repository-from-a-template) for convenience. From this repository:

1. Click "Use this template".
1. Set the following options:
   - Owner: `<your GH organization>`
   - Repository name: pulumi-xyz (replace "xyz" with the name of your provider)
     - Providers built from Cloudy Sky Software's templates are _always_ native providers, by default.
     - However, if there is already a TF-bridged provider with that name, you should add the suffix `-native` so that the package name in some package registries do not conflict with the other providers.
   - Description: Pulumi provider for xyz
   - Repository type: Public
1. Clone the generated repository.

From the templated repository:

Search-replace the following occurrences with the corresponding names.

| Search                                    | Replace With                                                                   |
| ----------------------------------------- | ------------------------------------------------------------------------------ |
| github.com/cloudy-sky-software/pulumi-xyz | Remote location of your repository                                             |
| xyz                                       | Lower-cased name of the provider                                               |
| Xyz                                       | Pascal-case name of the provider                                               |
| XYZ                                       | Upper-cased name of the provider                                               |
| x_y_z                                     | Lower snake-cased name of the provider if the provider name has multiple words |

Update the `Homepage` and `Publisher` values in `provider/pkg/gen/schema.go - PulumiSchema` appropriately.

#### Embed the OpenAPI spec

The OpenAPI spec file for the provider you are building must be placed in the `provider/cmd/pulumi-gen-*` folder as `openapi.yml`.
Unlike some of Pulumi's own native providers which download the OpenAPI spec from an upstream repo, this template does not do that
as it does not know where to download the OpenAPI spec from.

- You can, of course, add a Make target similar to what Pulumi does with some of its native providers and have it download the latest
OpenAPI spec from an upstream repo.
- You can also rename the file to something other than `openapi.yml` if you wish. Be sure to change the name of the file that Go
should embed in `provider/cmd/pulumi-gen-*/main.go`.

#### Generate Pulumi schema

Now that you have an OpenAPI spec downloaded and all the placholders renamed to the appropriate provider/package name,
i.e. replaced `xyz` with the appropriate name (see the section above), you can generate a Pulumi schema by running
`make gen generate_schema`. You must have a Pulumi schema generated successfully in order to generate the language
SDKs.

The larger the spec the more likely there are errors in the spec itself. In most cases, it has nothing to do with `pulschema`.
If it's a genuine bug in `pulschema`, please open an [issue](https://github.com/cloudy-sky-software/pulschema/issues).
But it's more than likely you'll need to patch the OpenAPI spec itself. You can do that using Go instead of manually editing the spec file
which can be quite cumbersome, especially if you are dealing with a very large spec.
Anyway, here's where you can write Go code to modify the spec: https://github.com/cloudy-sky-software/pulumi-provider-template/blob/main/provider/pkg/gen/openapi_fixes.go.
Here's an example of an OpenAPI spec that needed to be modified: https://github.com/cloudy-sky-software/pulumi-digitalocean-native/blob/main/provider/pkg/gen/openapi_fixes.go

If there are endpoints in the spec that you don't care about and want to exclude them from Pulumi,
you can pass a [list](https://github.com/cloudy-sky-software/pulumi-provider-template/blob/main/provider/pkg/gen/schema.go#L94) of the endpoint paths exactly as they appear in the spec.

#### Build the provider and install the plugin

```bash
$ make build install
```

This will:

1. Create the SDK codegen binary and place it in a `./bin` folder (gitignored)
2. Create the provider binary and place it in the `./bin` folder (gitignored)
3. Generate the dotnet, Go, Node, and Python SDKs and place them in the `./sdk` folder
4. Install the provider on your machine.

Feel free to modify any of the Make targets (or add news ones) to fit your needs.
If you feel others might find them useful, please consider contributing it back to
this template repo. :)

#### Test against the example

```bash
$ cd examples/simple
$ yarn link @pulumi/xyz
$ yarn install
$ pulumi stack init test
$ pulumi up
```

#### A brief repository overview

You now have:

1. A `provider/` folder containing the building and implementation logic
   1. `cmd/`
      1. `pulumi-gen-xyz/` - generates language SDKs from the schema
      2. `pulumi-resource-xyz/` - holds the package schema, injects the package version, and starts the gRPC server
   2. `pkg`
      1. `provider` - holds the gRPC methods (and for now, the sample implementation logic) required by the Pulumi engine
      2. `version` - semver package to be consumed by build processes
3. `sdk` - holds the generated code libraries created by `pulumi-gen-xyz/main.go`
4. `examples` a folder of Pulumi programs to try locally and/or use in CI.
5. A `Makefile` and this `README`.

### Implementing the provider callback methods

You will find a mostly blank implementation of these in `pkg/provider/provider.go`.
Note that these methods do not link 1:1 to the Pulumi resource provider interface
because `pulumi-provider-framework` provides a convenient callback mechanism
and handles all other responsibilities. You should use the callback methods
to alter the HTTP request (in Pre* methods) or the response (in Post*) as
needed. If you don't need any customization, then you don't need to do
anything at all. Yep! The framework handles it all.

### Build Examples

Create an example program using the resources defined in your provider, and place it in the `examples/` folder.

You can now repeat the steps for [build, install, and test](#test-against-the-example).

## Documentation

Please [follow this guide to add documentation to your provider](https://www.pulumi.com/docs/guides/pulumi-packages/how-to-author/#write-documentation).

### Importing Existing Resources

Import IDs should satisfy all ID segments in the `GET` endpoint for the resource
you are importing. The IDs required in the path should be separated by `/`.
First, start by identifying the `GET` endpoint in the OpenAPI spec
for the provider.

For example, let's assume such a GET endpoint path for some resource is: `/services/{serviceId}/someResource/{someResourceId}`.

Thus, the `pulumi import` command to run is:

```bash
# The resource type token can be easily found by using your IDEs
# Go To Definition functionality for the resource and looking at the type
# property defined in the custom resource's class definition.
pulumi import {resourceTypeToken} {resourceName} /{serviceId}/{someResourceId}
```

Alternatively, you can also import using the `import` Pulumi resource option.
Run `pulumi up` to import the resource into your stack's state. Once imported,
you should remove the `import` resource option.

```typescript
const someResource = new SomeResource(
  "myResourceName",
  { //inputs for the reosurce },
  {
    protect: true,
    import: `/{serviceId}/{someResourceId}`,
  }
);
```

Refer to the Pulumi [docs](https://www.pulumi.com/docs/iac/adopting-pulumi/import/) for importing a
resource.

## Configuring CI and releases

1. Follow the instructions laid out in the [deployment templates](./deployment-templates/README-DEPLOYMENT.md).
 readmeEtag: '"e76c9f1eed3a01c32d4bd4fdea89702e1e6bb390"' readmeLastModified: Fri, 18 Jul 2025 17:43:19 GMT repositoryId: 563307038 description: >- Template repo for creating Pulumi providers based on pulschema and pulumi-provider-framework created: '2022-11-08T10:44:25Z' updated: '2026-02-06T01:25:57Z' language: Go archived: false stars: 4 watchers: 1 forks: 1 owner: cloudy-sky-software logo: https://avatars.githubusercontent.com/u/97355959?v=4 license: Apache-2.0 repoEtag: '"10340170d3743aeea2f5217b38337547fae3b355f27c451a0ee33f122df82cb6"' repoLastModified: Fri, 06 Feb 2026 01:25:57 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/golangid/goruda v3: true repositoryMetadata: base64Readme: >- IyBHb3J1ZGEKCkdvcnVkYSBpcyBHb2xhbmcgQ0xJIEFwcCB0byBjb252ZXJ0IE9wZW5BUEkgMyBTcGVjaWZpY2F0aW9ucwppbnRvIHNpbXBsZSBib2lsZXJwbGF0ZSBjb2RlLiAKClRoaXMgYXBwIGlzIGNyZWF0ZWQgYmVjYXVzZSB0aGUgcHJvY2VzcyBvZiBjcmVhdGluZyBib2lsZXJwbGF0ZSAKSFRUUCBzZXJ2ZXIgaW4gR28gaXMgcmVwZXRpdGl2ZSwgc28gaW4gb3JkZXIgdG8gcmVkdWNlIHRoYXQga2luZCBvZiB3b3JrLCAKR29ydWRhIHdpbGwgcmVhZCBhbGwgeW91ciBPcGVuQVBJIDMgc3BlY2lmaWNhdGlvbnMgYW5kIGNvbnZlcnQgaXQKaW50byBzaW1wbGUgcnVubmluZyBIVFRQIFNlcnZlci4KCiMjIFJlcXVpcmVtZW50cwoKLSBHbyBhdCBsZWFzdCB2ZXIuIDEuMTEKLSBXb3JraW5nIE9wZW5BUEkgMyBGaWxlCgojIyBIb3cgdG8gUnVuCgpgYGBiYXNoCn4gbWFrZSBidWlsZAp+IC4vZ29ydWRhIGdlbmVyYXRlIFtwYXRoX3RvX29wZW5hcGlfZmlsZV0KYGBgCgo= readmeEtag: '"d5af7162b1282fd9326909034a59d325a120ae7f"' readmeLastModified: Mon, 22 Jun 2020 11:43:46 GMT repositoryId: 143961171 description: Open API 3 Specs to Golang HTTP Server created: '2018-08-08T04:30:14Z' updated: '2025-09-25T14:26:02Z' language: Go archived: true stars: 3 watchers: 2 forks: 2 owner: golangid logo: https://avatars.githubusercontent.com/u/42196748?v=4 license: MIT repoEtag: '"439ccec818afead9262e686a604b7610c78dacba78fa23664dbd85348eb080c1"' repoLastModified: Thu, 25 Sep 2025 14:26:02 GMT foundInMaster: true category: Parsers id: 076704721a0716ce327f2ced3ff53046 - source: openapi3 tags repository: https://github.com/quetz-al/quetzal v3: true repositoryMetadata: base64Readme: >- Li4gY2xhc3M6OiBjZW50ZXIKCiAgICAuLiBpbWFnZTo6IGRvY3Mvc291cmNlL19zdGF0aWMvbG9nb19oLnBuZwogICAgICAgOmhlaWdodDogMTAwcHgKICAgICAgIDphbHQ6IFF1ZXR6YWwgbG9nbwoKICAgIFF1ZXR6YWwg4oCUIEEgUkVTVGZ1bCBBUEkgZm9yIGRhdGEgYW5kIG1ldGFkYXRhIG1hbmFnZW1lbnQuCgouLiBiYWRnZXMgd2lsbCBnbyBoZXJlCgoKUXVldHphbAo9PT09PT09CgouLiBhYnN0cmFjdF9zdGFydAoKUXVldHphbCAoc2hvcnQgZm9yIFF1ZXR6YWxjw7NhdGwsIHRoZSBmZWF0aGVyZWQgc25ha2UpLCBhIFJFU1RmdWwgQVBJIGRlc2lnbmVkCnRvIHN0b3JlIGRhdGEgZmlsZXMgYW5kIG1hbmFnZSB0aGVpciBhc3NvY2lhdGVkIG1ldGFkYXRhLgoKUXVldHphbCBpcyBhbiBhcHBsaWNhdGlvbiB0aGF0IHVzZXMgQ2xvdWQgc3RvcmFnZSBwcm92aWRlcnMgYW5kIG5vbi1zdHJ1Y3R1cmVkCmRhdGFiYXNlcyB0byBoZWxwIHJlc2VhcmNoZXJzIG9yZ2FuaXplIHRoZWlyIGRhdGEgYW5kIG1ldGFkYXRhIGZpbGVzLgpJdHMgbWFpbiBmZWF0dXJlIGlzIHRvIHByb3ZpZGUgYSByZW1vdGUsIHZpcnR1YWxseSBpbmZpbml0ZSwgc3RvcmFnZSBsb2NhdGlvbgpmb3IgcmVzZWFyY2hlcnMnIGRhdGEsIHdoaWxlIHByb3ZpZGluZyBhbiBBUEkgdG8gZW5jYXBzdWxhdGUgZGF0YS9tZXRhZGF0YQpvcGVyYXRpb25zLiBJbiBvdGhlciB3b3JkcywgcmVzZWFyY2hlcnMgYW5kIHRlYW1zIGNhbiB3b3JrIHdpdGggbGFyZ2UgYW1vdW50cwpvZiBkYXRhIHRoYXQgd291bGQgYmUgdG9vIGxhcmdlIGZvciBsb2NhbCBhbmFseXNlcywgdXNpbmcgUXVldHphbCB0byBzaW1wbGlmeQp0aGUgY29tcGxleGl0eSBvZiBDbG91ZCByZXNvdXJjZSBtYW5hZ2VtZW50LgoKUXVldHphbCdzIG1pZC10ZXJtIHJvYWRtYXAgaXMgdG8gaW50ZWdyYXRlIHdpdGggbGFyZ2UgcHVibGljIHBoeXNpb2xvZ2ljYWwKc2lnbmFsIGRhdGFiYXNlcyBsaWtlIFBoeXNpb05ldF8sIE1JUERCXywgVFVIXywgYW1vbmcgb3RoZXJzLiBUaGEgbWFpbiBvYmplY3RpdmUKaXMgdG8gcHJvdmlkZSByZXNlYXJjaGVycyBhbmQgZGF0YSBzY2llbnRpc3RzIGEgdW5pcXVlIGJhbmsgb2YgZmlsZSBkYXRhc2V0cwp3aXRoIGEgdW5pZmllZCBBUEkgdG8gYWNjZXNzIHRoZSBkYXRhIGFuZCB0byBlbmNhcHN1bGF0ZSB0aGUgaGV0ZXJvbmVnZWl0eSBvZgp0aGVzZSBkYXRhc2V0cy4KCi4uIF9QaHlzaW9OZXQ6IGh0dHBzOi8vcGh5c2lvbmV0Lm9yZy8KLi4gX01JUERCOiBodHRwOi8vZmNvbl8xMDAwLnByb2plY3RzLm5pdHJjLm9yZy9pbmRpL2NtaV9lZWcvCi4uIF9UVUg6IGh0dHBzOi8vd3d3LmlzaXAucGljb25lcHJlc3MuY29tL3Byb2plY3RzL3R1aF9lZWcvaHRtbC9vdmVydmlldy5zaHRtbAoKRmVhdHVyZXMKLS0tLS0tLS0KClRoZXJlIGFyZSB0d28gc2NlbmFyaW9zIHdoZXJlIFF1ZXR6YWwgd2FzIGRlc2lnbmVkIHRvIGhlbHA6CgoqIEltYWdpbmUgeW91IHdhbnQgdG8gYXBwbHkgYSBkYXRhIHByb2Nlc3NpbmcgcGlwZWxpbmUgdG8gYSBsYXJnZSBkYXRhc2V0LgogIFRoZXJlIGFyZSBzZXZlcmFsIHNvbHV0aW9ucyBvbiBob3cgdG8gZXhlY3V0ZSBhbmQgcGFyYWxsZWxpemUgeW91ciBjb2RlLCBidXQKICAqd2hlcmUgaXMgdGhlIGRhdGE/KiBNb3Jlb3ZlciwgaW1hZ2luZSB0aGF0IHlvdSB3YW50IHRvIGRvIGEgdHJhbnN2ZXJzZSBzdHVkeToKICBIb3cgZG8geW91IG1hbmFnZSB0aGUgZGlmZmVyZW50IHNvdXJjZXM/IEhvdyB0byBkb3dubG9hZCB0aGVtPwoKICBRdWV0emFsIHByb3ZpZGVzIGEgc2luZ2xlIGRhdGEgc291cmNlIHdpdGggYSBzaW1wbGUgQVBJIHRoYXQgd2lsbCBsZXQgeW91CiAgZGVmaW5lIGVhc2lseSB0aGUgc2NvcGUgb2YgeW91ciBzdHVkeSBhbmQsIHdpdGggYSBicmllZiBQeXRob24gY29kZSB0aGF0CiAgdXNlcyBgUXVldHphbCBjbGllbnQgPGh0dHBzOi8vZ2l0aHViLmNvbS9xdWV0ei1hbC9xdWV0emFsLWNsaWVudD5gXywgeW91IHdpbGwKICBiZSBhYmxlIHRvIGRvd25sb2FkIHlvdXIgZGF0YXNldC4KCiogTGV0J3Mgc2F5IHRoYXQgeW91IGFyZSBwcmVwYXJpbmcgYSBuZXcgc3R1ZHkgaW1wbHlpbmcgc29tZSBkYXRhIGNvbGxlY3Rpb24KICBwcm90b2NvbC4gWW91IGNvdWxkIGRlZmluZSBhIHByb2NlZHVyZSB3aGVyZSB0aGUgZGF0YSBvcGVyYXRvcnMgb3IgdGVjaG5pY2lhbnMKICB0YWtlIGNhcmUgdG8gY29weSB0aGUgZGF0YSBmaWxlcyBpbiBhIGRpc2ssIEdvb2dsZSBEcml2ZSBvciBEcm9wYm94LCBhbG9uZwogIHdpdGggdGhlIG5vdGVzIGFzc29jaWF0ZWQgd2l0aCBlYWNoIHNlc3Npb24sIGxpa2Ugc3ViamVjdCBzdHVkeSBpZGVudGlmaWVyLAogIGRhdGUsIGFnZSwgdGVtcGVyYXR1cmUsIGV0Yy4gRG9pbmcgdGhpcyBtYW51YWxseSB3b3VsZCBiZSBlcnJvci1wcm9uZS4KICBNb3Jlb3ZlciwgdGhlIHN0cnVjdHVyZSBvZiB0aGVzZSBub3RlcyAoaS5lLiB0aGUgbWV0YWRhdGEpIG1heSBldm9sdmUgcXVpY2tseSwKICBzbyB5b3UgZWl0aGVyIHNhdmUgdGhlbSBhcyBtYW51YWwgbm90ZXMsIHRleHQgZmlsZXMsIG9yIHNvbWUgZGF0YWJhc2UgdGhhdAogIGdpdmVzIHlvdSB0aGUgZmxleGliaWxpdHkgdG8gcXVpY2tseSBhZGFwdCBpdHMgc3RydWN0cmUuCgogIFVzaW5nIHRoZSBRdWV0emFsIEFQSSwgeW91IGF1dG9tYXRlIHRoZSB1cGxvYWQgYW5kIHNhZmUgc3RvcmFnZSBvZiB0aGUgc3R1ZHkKICBmaWxlcywgYXNzb2NpYXRlIHRoZSBtZXRhZGF0YSBvZiB0aGVzZSBmaWxlcyB3aGlsZSBoYXZpbmcgdGhlIGxpYmVydHkgdG8gc2V0CiAgYW5kIG1vZGlmeSB0aGUgbWV0YWRhdGEgc3RydWN0dXJlIGFzIHlvdSBzZWUgZml0LgoKCkluIGJyaWVmLCBRdWV0emFsIG9mZmVycyB0aGUgZm9sbG93aW5nIG1haW4gZmVhdHVyZXM6CgoqICoqU3RvcmFnZSoqIG9mIGRhdGEgZmlsZXMsIGJhc2VkIG9uIGNsb3VkIHN0b3JhZ2UgcHJvdmlkZXJzLCB3aGljaCBiZW5lZml0cwogIGZyb20gYWxsIG9mIHRoZSBmZWF0dXJlcyBmcm9tIHRoZSBwcm92aWRlciwgc3VjaCBhcyB2aXJ0dWFsbHkgaW5maW5pdGUKICBzdG9yYWdlIHNpemUuCiogKipVbnN0cnVjdHVyZWQgbWV0YWRhdGEqKiBhc3NvY2lhdGVkIHRvIGVhY2ggZmlsZSouIFF1ZXR6YWwgZG9lcyBub3QgZm9yY2UKICB0aGUgdXNlciB0byBvcmdhbml6ZSB5b3VyIG1ldGFkYXRhIGluIGEgcGFydGljdWxhciB3YXksIGl0IGxldHMgdGhlIHVzZXIga2VlcAogIHdoYXRldmVyIHN0cnVjdHVyZSB0aGV5IHByZWZlci4KKiAqKlN0cnVjdHVyZWQgbWV0YWRhdGEgdmlld3MqKiBmb3IgbWV0YWRhdGEgZXhwbG9yYXRpb24gb3IgZGF0YXNldCBkZWZpbml0aW9uLgogIEJ5IGxldmVyYWdpbmcgUG9zdGdyZXMgU1FMLCB1bnN0cnVjdHVyZWQgbWV0YWRhdGEgY2FuIGJlIHF1ZXJpZWQgYXMgSlNPTgogIG9iamVjdHMsIGxldHRpbmcgdGhlIHVzZXIgZXhwcmVzcyB3aGF0IHN1YnNldCBvZiB0aGUgZGF0YSB0aGV5IHdhbnQgdG8gdXNlLgoqICoqTWV0YWRhdGEgdmVyc2lvbmluZyoqLiBDaGFuZ2VzIG9uIG1ldGFkYXRhIGFyZSB2ZXJzaW9uZWQsIHdoaWNoIGlzCiAgcGFydGljdWxhcmx5IHVzZWZ1bCB0byBlbnN1cmUgdGhhdCBhIGRhdGFzZXQgYXJlIHJlcHJvZHVjaWJsZS4KKiBFbmRwb2ludHMgYW5kIG9wZXJhdGlvbnMgZGVmaW5lZCB1c2luZyB0aGUKICBgT3BlbkFQSSB2MyBzcGVjaWZpY2F0aW9uIDxodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbj5gXy4KCi4uIGFic3RyYWN0X2VuZAoKRG9jdW1lbnRhdGlvbgotLS0tLS0tLS0tLS0tCgpRdWV0emFsJ3MgZG9jdW1lbnRhdGlvbiBpcyBhdmFpbGFibGUgb24KYHJlYWR0aGVkb2NzIDxodHRwczovL3F1ZXR6YWwtYXBpLnJlYWR0aGVkb2NzLm9yZz5gXy4gVGhlIEFQSSBkb2N1bWVudGF0aW9uIGlzCmVtYmVkZGVkIGludG8gaXRzIHNwZWNpZmljYXRpb247IHRoZSBiZXN0IHdheSB0byB2aXN1YWxpemUgaXQgaXMgdGhyb3VnaCB0aGUKaXMgYWxzbyBhCmBSZURvYyBBUEkgcmVmZXJlbmNlIGRvY3VtZW50YXRpb24gc2l0ZSA8aHR0cHM6Ly9zdGFnZS5xdWV0ei5hbC9yZWRvYz5gXy4KCgoKQ29udHJpYnV0ZQotLS0tLS0tLS0tCgotIElzc3VlIFRyYWNrZXI6IGh0dHBzOi8vZ2l0aHViLmNvbS9xdWV0ei1hbC9xdWV0emFsL2lzc3VlcwotIFNvdXJjZSBDb2RlOiBodHRwczovL2dpdGh1Yi5jb20vcXVldHotYWwvcXVldHphbAoKU3VwcG9ydAotLS0tLS0tCgpJZiB5b3UgYXJlIGhhdmluZyBpc3N1ZXMsIHBsZWFzZSBsZXQgdXMga25vdyBieSBvcGVuaW5nIGFuIGlzc3VlIG9yIGJ5IHNlbmRpbmcKYW4gZW1haWwgdG8gc3VwcG9ydEBxdWV0ei5hbC4KCkxpY2Vuc2UKLS0tLS0tLQoKVGhlIHByb2plY3QgaXMgdW5kZXIgdGhlIEJTRCAzLWNsYXVzZSBsaWNlbnNlLgoKU2VlIHRoZSBgYXV0aG9ycyA8Li9BVVRIT1JTLnJzdD5gXyBwYWdlIGZvciBtb3JlIGluZm9ybWF0aW9uIG9uIHRoZSBhdXRob3JzIGFuZApjb3B5cmlnaHQgaG9sZGVycy4K readmeEtag: '"3c831224d83f4aa0c17bc8b49a787f06b1ad0002"' readmeLastModified: Tue, 24 Mar 2020 16:17:35 GMT repositoryId: 173719463 description: >- Quetzal API (short for Quetzalcoatl): a data and metadata management application created: '2019-03-04T10:00:31Z' updated: '2022-08-05T03:40:39Z' language: Python archived: false stars: 3 watchers: 0 forks: 2 owner: quetz-al logo: https://avatars.githubusercontent.com/u/47522179?v=4 license: BSD-3-Clause repoEtag: '"a9342d638aec7c8cdcbb0dd1b130d22e0e768f1d7409172fb1bb3fbfa653b136"' repoLastModified: Fri, 05 Aug 2022 03:40:39 GMT foundInMaster: true category: - Testing - Server Implementations id: 76a2a1bb0a23b588699359bf97edbc48 - source: openapi3 tags repository: https://github.com/ninjanas/forteapi v3: true id: b3809529738bd1da943a65032673b5ae repositoryMetadata: base64Readme: >- # ForteAPI
<div align="center">
<h3></h3>
<img width="100" height="100" src="https://img.icons8.com/ios-filled/100/curly-brackets.png" alt="curly-brackets"/>
<h3></h3>
</div>

<div align="center">
<img alt="GitHub last commit (branch)" src="https://img.shields.io/github/last-commit/NinjaNas/ForteAPI/main">
<a href="https://github.com/NinjaNas/ForteAPI/issues"><img src="https://img.shields.io/github/issues/NinjaNas/ForteAPI" alt="issues - ForteAPI"></a>
<img alt="GitHub Repo stars" src="https://img.shields.io/github/stars/NinjaNas/ForteAPI">
<a href="https://typescriptlang.org" title="Go to TypeScript homepage"><img src="https://img.shields.io/badge/TypeScript-5-blue?logo=typescript&logoColor=white" alt="Made with TypeScript"></a>
<a href="https://github.com/NinjaNas/ForteAPI/blob/main/LICENSE"><img src="https://img.shields.io/badge/License-MIT-blue" alt="License"></a>

Forte API is a music theory API that provides a way to query set classes in 12 tone equal temperament. It uses data scraped from this <a href="https://en.wikipedia.org/wiki/List_of_set_classes">wikipedia page</a> to provide Forte numbers, prime forms, interval vectors, zygotic or twinned sets, and complements of sets.

##

**Currently this API is live [here](https://hcda8f8dtk.execute-api.us-east-1.amazonaws.com/prod/api/data/) using AWS API Gateway + AWS Lambda!**
**Check out the OpenAPI docs on SwaggerHub [here](https://app.swaggerhub.com/apis-docs/NinjaNas/ForteAPI/1.3.1)!**

**The API is rate-limited at 500 requests per day. Make an issue if you need more requests.**

**Please make an issue for any bugs and typos you may find!**

</div>

## Table of Contents

- [App Example](#app-example)
- [What are Forte Numbers? and More!](#what-are-forte-numbers-and-more)
  - [Set Theory](#set-theory)
  - [Prime Form](#prime-form)
  - [Interval Vector](#interval-vector)
  - [Forte Numbers](#forte-numbers)
  - [Complements](#complements)
- [Should I use this API?](#should-i-use-this-api)
- [DataSet Type](#dataset-type)
- [Status Codes](#status-codes)
- [Endpoints](#endpoints)
  - [GET /api/data/](#get-apidata)
  - [GET /api/data/:queryProp/](#get-apidataqueryprop)
  - [GET /api/flatdata/:queryprop/](#get-apiflatdataqueryprop)
    - [number](#number)
    - [primeForm](#primeform)
    - [vec](#vec)
    - [z](#z)
    - [complement](#complement)
    - [inversion](#inversion)
  - [GET /api/hashdata/:queryPropKey/:queryPropValue/](#get-apihashdataquerypropkeyquerypropvalue)
  - [GET /api/data/number/:querySearch/](#get-apidatanumberquerysearch)
    - [Exact Search](#exact-search)
    - [Starts With Search](#starts-with-search)
    - [Ends With Search](#ends-with-search)
    - [Contains Search](#contains-search)
    - [Substring Search](#substring-search)
    - [Not Search](#not-search)
    - [Exclude Search](#exclude-search)
    - [Chaining Methods](#chaining-methods-no-duplicates)
  - [GET /api/data/primeForm/:querySearch/](#get-apidataprimeformquerysearch)
    - [Exact Search](#exact-search-1)
    - [Starts With Search](#starts-with-search-1)
    - [Ends With Search](#ends-with-search-1)
    - [Contains Search](#contains-search-1)
    - [Substring Search](#substring-search-1)
    - [Not Search](#not-search-1)
    - [Exclude Search](#exclude-search-1)
    - [Chaining Methods](#chaining-methods-no-duplicates-1)
  - [GET /api/data/vec/:querySearch/](#get-apidatavecquerysearch)
    - [Exact Search](#exact-search-2)
    - [Starts With Search](#starts-with-search-2)
    - [Ends With Search](#ends-with-search-2)
    - [Contains Search](#contains-search-2)
    - [Substring Search](#substring-search-2)
    - [Not Search](#not-search-2)
    - [Exclude Search](#exclude-search-2)
    - [Chaining Methods](#chaining-methods-no-duplicates-2)
  - [GET /api/data/vec/:querySearch/:queryInequality](#get-apidatavecquerysearchqueryinequality)
    - [Equal Search](#equal-search)
    - [Less Than Search](#less-than-search)
    - [Less Than Or Equal To Search](#less-than-or-equal-to-search)
    - [Greater Than Search](#greater-than-search)
    - [Greater Than Or Equal To Search](#greater-than-or-equal-to-search)
  - [GET /api/data/z/:querySearch/](#get-apidatazquerysearch)
    - [Exact Search](#exact-search-3)
    - [Starts With Search](#starts-with-search-3)
    - [Ends With Search](#ends-with-search-3)
    - [Contains Search](#contains-search-3)
    - [Substring Search](#substring-search-3)
    - [Not Search](#not-search-3)
    - [Exclude Search](#exclude-search-3)
    - [Chaining Methods](#chaining-methods-no-duplicates-3)
  - [GET /api/data/complement/:querySearch/](#get-apidatacomplementquerysearch)
    - [Exact Search](#exact-search-4)
    - [Starts With Search](#starts-with-search-4)
    - [Ends With Search](#ends-with-search-4)
    - [Contains Search](#contains-search-4)
    - [Substring Search](#substring-search-4)
    - [Not Search](#not-search-4)
    - [Exclude Search](#exclude-search-4)
    - [Chaining Methods](#chaining-methods-no-duplicates-4)
  - [GET /api/data/inversion/:querySearch/](#get-apidatainversionquerysearch)
    - [Exact Search](#exact-search-5)
    - [Starts With Search](#starts-with-search-5)
    - [Ends With Search](#ends-with-search-5)
    - [Contains Search](#contains-search-5)
    - [Substring Search](#substring-search-5)
    - [Not Search](#not-search-5)
    - [Exclude Search](#exclude-search-5)
    - [Chaining Methods](#chaining-methods-no-duplicates-5)
  - [GET /api/data/:queryProp/PROPERTY/querySearch](#get-apidataqueryproppropertyquerysearch)
    - [Endpoints](#endpoints-1) 
  - [GET /api/data/d3/:querySearch/](#get-apidatad3querysearch)
    - [Types](#types)
    - [Cardinality-Increasing vs Strict-Increasing](#cardinality-increasing-vs-strict-increasing)
    - [Vector-Similarity](#vector-similarity)
    - [Original vs Inversion](#original-vs-inversion)
    - [Manual Construction](#manual-construction)
        - [Fetch](#fetch)
    	- [Links](#links)
     	- [Dag](#dag)
    - [How to Use JSON in D3Dag](#how-to-use-json-in-d3dag)
    - [Endpoints](#endpoints)
- [API Development](#api-development)
  - [Add .env File](#add-env-file)
- [Star This Repo](#star-this-repo)

## App Example

Check out my app [set-class-visualizer](https://github.com/NinjaNas/set-class-visualizer) using ForteAPI to serve its data and dags!

## What are Forte Numbers? and More!

If you have the time watch this great playlist of videos by Jay Beard on [musical set theory](https://www.youtube.com/watch?v=3fe_NatE5w8&list=PLKWIRLQnfaw1KUMTG3b9mFrt0D3MFHvPF)!

Before getting into Forte numbers we should learn basic set theory, prime form, and interval vectors.

### Set Theory

There are 12 notes or 12 pitch classes in 12 tone equal temperament.

We can label them as so:

Set Theory Labels: **0, 1, 2, 3, 4, 5, 6, 7, 8, 9, T, E (0 to 11)**

Note Names: **C, C#/Db, D, D#/Eb, E, F, F#/Gb, G, G#/Ab, A, A#/Bb, B**

### Prime Form

Prime forms are a way to standardize sets so they can be compared. Prime form requires us to alway to have 0 first by transposition, unless it is the empty set, and have the smallest intervals as possible (compactness) going from left to right as there are different ways to order the same set

- Ex: {0,1,2,4} - Note that {1,2,3,5} is not in prime form an is a transposition of {0,1,2,4} as we can subtract 1 from all of the elements to make {0,1,2,4}

### Interval Vector

An interval vector is a six-number sequence that represents the distribution of intervals within a pitch-class set. Each number in the vector indicates the number of times a specific interval (from 1 to 6 semitones) appears between any two pitches in the set.

- Ex: <6,4,5,6,5,2>
- Ex: <C,C,C,C,C,6> (C stands for 12 here)

### Forte Numbers

Forte numbers are a system of labeling pitch-class sets in the musical set theory, named after the musicologist Allen Forte who devised this classification system. They offer a systematic way to catalog and study the various pitch-class sets in twelve-tone music.

The labels are structured by cardinality-ordinal number.

- Ex. 3-6 meaning the 6th set in order of sets containing 3 pitch classes/notes.
- In general the smaller the ordinal number the more compact the set is.

Allen Forte did not differentiate between inversions in his book so 3-11A and 3-11B would just be 3-11 using 3-11A as its most compact form. Later on, notation was made to differentiate between inversions. If there is A or B appended to the end it means that set has a distinct inversion, where A is given to the most compact version. Both A and B sets share the same interval vector.

- Ex. 3-11A and 3-11B
- 3-11A has a prime form of {0,3,7} and 3-11B has a prime form of {0,3,7}
- Let's invert {0,3,7} in this system we are using mod 12 because we have 12 pitch classes
- {12-0 mod 12, 12-3 mod 12, 12-7 mod 12} = {0, 9, 5}
- Order for compactness: {5, 9, 0} has the smallest intervals going from left to right
- Transpose so first element is 0: {5-5, 9-5, 12(0)-5) = {0,4,7}

If there is a Z in the Forte number this means it is a zygotic/twinned set that has a twin set with the same interval vector and is not an inversion of the set. In this API, the Z property is linked only A when applicable for consistency.

- Ex: 4-z15A and 4-z29A
- Ex: 6-z4 and 6-z37

### Complements

Complements are pairs of sets that add up to set with all pitch classes {0,1,2,3,4,5,6,7,8,9,T,E}. If a set does not have a complement that means they are a complement of themselves

- Ex: Prime form of {} is a complement of {0,1,2,3,4,5,6,7,8,9,T,E}
- Ex: {0,1,3,4} is a complement of {0,1,2,3,4,5,6,9} after some transposition

## Should I use this API?

You should only use this API either if you care about your intital load times as the full data size is ~42KB or if you want a prebuilt solution to query set classes. You can use this API to get and serve prebuilt d3dag graphs for visualization or the just the links for you to create your own custom graphs. 

If you do not need to use this API, you should just download the json [here](https://github.com/NinjaNas/ForteAPI/blob/main/data/set_classes.json). Report any typos you may find or suggest new properties.

There are also d3dag graphs and links avaliable in JSON [here](https://github.com/NinjaNas/ForteAPI/blob/main/data/d3).

## DataSet Type

**Note: Properties are case-sensitive!**

```ts
type DataSet = {
	number: string;
	primeForm: string; // can be easily formatted to a string[] by JSON.parse(primeFormString)
	vec: string;
	z: null | string;
	complement: null | string;
	inversion: null | string;
}[];
```

## Status Codes

500 - Internal Server Error
414 - URI or Subquery Too Long
400 - Bad Property / Request Returned Nothing
200 - Success

## Endpoints

### GET /api/data/

The endpoint returns all of the data from [/data/set-classes.json](https://github.com/NinjaNas/ForteAPI/blob/main/data/set_classes.json)

```ts
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "12-1",
    "inversion": null
  },
    ...
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

### GET /api/data/:queryProp/

The endpoint returns the full data given the properties provided in a comma separated list (number, primeForm, vec, z, complement, inversion)

- Max URI length: No more than 43 characters
- Subquery length: 1-10 characters

```ts
// GET /api/data/number
[
  {
    "number": "0-1"
  },
    ...
  {
    "number": "12-1"
  },
];

// GET /api/data/number,primeForm
[
  {
    "number": "0-1"
    "primeForm": "[\"\"]",
  },
    ...
  {
    "number": "12-1"
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]"
  },
];
```

### GET /api/flatdata/:queryProp/

The endpoint returns a flatmap of the valid properties (number, primeForm, vec, z, complement, inversion)

- Max URI length: No more than 10 characters

#### number

```ts
// GET /api/flatdata/number
["0-1", ..., "4-z15A", ..., "12-1"];
```

#### primeForm

```ts
// GET /api/flatdata/primeForm
["[\"\"]", ..., "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]"];
```

#### vec

```ts
// GET /api/flatdata/vec
["<0,0,0,0,0,0>", ..., "<C,C,C,C,C,6>"];
```

#### z

```ts
// GET /api/flatdata/z
[null, ..., "4-z15A", ..., null];
```

#### complement

```ts
// GET /api/flatdata/complement
["12-1", ..., "0-1"];
```
#### inversion

```ts
// GET /api/flatdata/inversion
["null", ..., "[\"0\",\"2\",\"5\",\"6\"]", ..., "null"];
```


### GET /api/hashdata/:queryPropKey/:queryPropValue/

The endpoint returns a hashmap with two the valid properties (number, primeForm, vec, z, complement, inversion)

- Max URI length For Both Queries: No more than 10 characters

```ts
// GET /api/hashdata/number/primeForm
{
  "0-1": "[\"\"]",
    ...
  "12-1": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]"
};
```


### GET /api/data/number/:querySearch/

The endpoint returns an array of objects based on the query on the number property

- Max URI length: No more than 100 characters
- Subquery length: 2-15 characters
	- Ex. ^1 or !`6-z25A~6-z25B

#### Exact Search

```ts
// GET /api/data/number/1-1
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "11-1",
    "inversion": null
  }
];
```

#### Starts With Search

```ts
// GET /api/data/number/^4-z15
[
  {
    "number": "4-z15A",
    "primeForm": "[\"0\",\"1\",\"4\",\"6\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z29A",
    "complement": "8-z15B",
    "inversion": "[\"0\",\"2\",\"5\",\"6\"]"
  },
  {
    "number": "4-z15B",
    "primeForm": "[\"0\",\"2\",\"5\",\"6\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z29A",
    "complement": "8-z15A",
    "inversion": "[\"0\",\"1\",\"4\",\"6\"]"
  }
];
```

#### Ends With Search

```ts
// GET /api/data/number/-z50$
[
  {
    "number": "6-z50",
    "primeForm": "[\"0\",\"1\",\"4\",\"6\",\"7\",\"9\"]",
    "vec": "<2,2,4,2,3,2>",
    "z": "6-z29",
    "complement": "6-z29",
    "inversion": null
  }
];
```

#### Contains Search

```ts
// GET /api/data/number/@12A or GET /api/data/number/@21A
[
  {
    "number": "4-12A",
    "primeForm": "[\"0\",\"2\",\"3\",\"6\"]",
    "vec": "<1,1,2,1,0,1>",
    "z": null,
    "complement": "8-12A",
    "inversion": "[\"0\",\"3\",\"4\",\"6\"]"
  },
  {
    "number": "5-21A",
    "primeForm": "[\"0\",\"1\",\"4\",\"5\",\"8\"]",
    "vec": "<2,0,2,4,2,0>",
    "z": null,
    "complement": "7-21B",
    "inversion": "[\"0\",\"3\",\"4\",\"7\",\"8\"]"
  },
  {
    "number": "6-z12A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"6\",\"7\"]",
    "vec": "<3,3,2,2,3,2>",
    "z": "6-z41A",
    "complement": "6-z41B",
    "inversion": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\"]"
  },
  {
    "number": "6-21A",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"8\"]",
    "vec": "<2,4,2,4,1,2>",
    "z": null,
    "complement": null,
    "inversion": "[\"0\",\"2\",\"4\",\"5\",\"6\",\"8\"]"
  },
  {
    "number": "7-21A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"8\",\"9\"]",
    "vec": "<4,2,4,6,4,1>",
    "z": null,
    "complement": "5-21B",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"8\",\"9\"]"
  },
  {
    "number": "8-12A",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\",\"7\",\"9\"]",
    "vec": "<5,5,6,5,4,3>",
    "z": null,
    "complement": "4-12A",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"6\",\"8\",\"9\"]"
  }
];
```

#### Substring Search

```ts
// GET /api/data/number/*12A
[
  {
    "number": "4-12A",
    "primeForm": "[\"0\",\"2\",\"3\",\"6\"]",
    "vec": "<1,1,2,1,0,1>",
    "z": null,
    "complement": "8-12A",
    "inversion": "[\"0\",\"3\",\"4\",\"6\"]"
  },
  {
    "number": "6-z12A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"6\",\"7\"]",
    "vec": "<3,3,2,2,3,2>",
    "z": "6-z41A",
    "complement": "6-z41B",
    "inversion": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\"]"
  },
  {
    "number": "8-12A",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\",\"7\",\"9\"]",
    "vec": "<5,5,6,5,4,3>",
    "z": null,
    "complement": "4-12A",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"6\",\"8\",\"9\"]"
  }
];
```

#### Range Search (inclusive)

The only filtering methods that you can use with range search are \` and ! (which is also !\`).
  - Ex: \`1-1\~2-1, !1-1\~2-1, !\`1-1\~2-1

```ts
// GET /api/data/number/1-1~2-1
// WARNING: 1-1~1-1 is invalid and will send a 400 status code, use 1-1 instead
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "11-1",
    "inversion": null
  },
  {
    "number": "2-1",
    "primeForm": "[\"0\",\"1\"]",
    "vec": "<1,0,0,0,0,0>",
    "z": null,
    "complement": "10-1",
    "inversion": null
  }
];
```

#### Not Search

- Works with !1-1, !^1-1, !1-1$, !@A, !*A, !1-1~2-2
  - ! is equivalent to !` 
  - `! is not a valid search method

```ts
// GET /api/data/number/!1-1
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "11-1",
    "inversion": null
  },
    ...,
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Exclude Search

- Works with \`1-1, \`^1-1, \`1-1$, \`@A, \`*A, \`1-1~2-2

```ts
// GET /api/data/number/1-1~2-4,`1-1~2-2
[
  {
    "number": "2-3",
    "primeForm": "[\"0\",\"3\"]",
    "vec": "<0,0,1,0,0,0>",
    "z": null,
    "complement": "10-3",
    "inversion": null
  },
  {
    "number": "2-4",
    "primeForm": "[\"0\",\"4\"]",
    "vec": "<0,0,0,1,0,0>",
    "z": null,
    "complement": "10-4",
    "inversion": null
  }
];
```

#### Chaining Methods (no duplicates)

```ts
// GET /api/data/number/1-1,^4-z15,-z50$,1-1~2-1,`4-z15B
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "11-1",
    "inversion": null
  },
  {
    "number": "4-z15A",
    "primeForm": "[\"0\",\"1\",\"4\",\"6\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z29A",
    "complement": "8-z15B",
    "inversion": "[\"0\",\"2\",\"5\",\"6\"]"
  },
  {
    "number": "6-z50",
    "primeForm": "[\"0\",\"1\",\"4\",\"6\",\"7\",\"9\"]",
    "vec": "<2,2,4,2,3,2>",
    "z": "6-z29",
    "complement": "6-z29",
    "inversion": null
  },
  {
    "number": "2-1",
    "primeForm": "[\"0\",\"1\"]",
    "vec": "<1,0,0,0,0,0>",
    "z": null,
    "complement": "10-1",
    "inversion": null
  }
];
```

### GET /api/data/primeForm/:querySearch/

The endpoint returns an array of objects based on the query on the primeForm property

Use T for 10, E for 11, and C for 12.

- Max URI length: No more than 100 characters
- Subquery length: 2-14 characters
	- Ex. ^0 or !`0123456789TE

#### Exact Search

```ts
// GET /api/data/primeForm/0123
[
  {
    "number": "4-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\"]",
    "vec": "<3,2,1,0,0,0>",
    "z": null,
    "complement": "8-1",
    "inversion": null
  }
];
```

#### Starts With Search

```ts
// GET /api/data/primeForm/^0356
[
  {
    "number": "4-13B",
    "primeForm": "[\"0\",\"3\",\"5\",\"6\"]",
    "vec": "<1,1,2,0,1,1>",
    "z": null,
    "complement": "8-13A",
    "inversion": "[\"0\",\"1\",\"3\",\"6\"]"
  },
  {
    "number": "5-25B",
    "primeForm": "[\"0\",\"3\",\"5\",\"6\",\"8\"]",
    "vec": "<1,2,3,1,2,1>",
    "z": null,
    "complement": "7-25A",
    "inversion": "[\"0\",\"2\",\"3\",\"5\",\"8\"]"
  },
  {
    "number": "5-z36B",
    "primeForm": "[\"0\",\"3\",\"5\",\"6\",\"7\"]",
    "vec": "<2,2,2,1,2,1>",
    "z": "5-z12",
    "complement": "7-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"7\"]"
  },
  {
    "number": "6-z40B",
    "primeForm": "[\"0\",\"3\",\"5\",\"6\",\"7\",\"8\"]",
    "vec": "<3,3,3,2,3,1>",
    "z": "6-z11A",
    "complement": "6-z11A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"8\"]"
  }
];
```

#### Ends With Search

```ts
// GET /api/data/primeForm/2$
[
  {
    "number": "2-2",
    "primeForm": "[\"0\",\"2\"]",
    "vec": "<0,1,0,0,0,0>",
    "z": null,
    "complement": "10-2",
    "inversion": null
  },
  {
    "number": "3-1",
    "primeForm": "[\"0\",\"1\",\"2\"]",
    "vec": "<2,1,0,0,0,0>",
    "z": null,
    "complement": "9-1",
    "inversion": null
  }
];
```

#### Contains Search

```ts
// GET /api/data/primeForm/@12346789
[
  {
    "number": "9-5A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<7,6,6,6,7,4>",
    "z": null,
    "complement": "3-5B",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "10-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<9,8,8,8,8,4>",
    "z": null,
    "complement": "2-1",
    "inversion": null
  },
  {
    "number": "10-6",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\",\"T\"]",
    "vec": "<8,8,8,8,8,5>",
    "z": null,
    "complement": "2-6",
    "inversion": null
  },
  {
    "number": "11-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\"]",
    "vec": "<T,T,T,T,T,5>",
    "z": null,
    "complement": "1-1",
    "inversion": null
  },
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Substring Search

```ts
// GET /api/data/primeForm/*12346789
[
  {
    "number": "9-5A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<7,6,6,6,7,4>",
    "z": null,
    "complement": "3-5B",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "10-6",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\",\"T\"]",
    "vec": "<8,8,8,8,8,5>",
    "z": null,
    "complement": "2-6",
    "inversion": null
  }
];
```

#### Not Search

- Works with !0, !^0, !0$, !@0, !*0,
  - ! is equivalent to !` 
  - `! is not a valid search method
- Use %20 for whitespace equivalent to the empty array for 0-1  

```ts
// GET /api/data/primeForm/!%20
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "11-1",
    "inversion": null
  },
    ...,
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Exclude Search

- Works with \`0, \`^0, \`0$, \`@0, \`*0

```ts
// GET /api/data/primeForm/01,012,`012
[
  {
    "number": "2-1",
    "primeForm": "[\"0\",\"1\"]",
    "vec": "<1,0,0,0,0,0>",
    "z": null,
    "complement": "10-1",
    "inversion": null
  }
];
```

#### Chaining Methods (no duplicates)

```ts
// GET /api/data/primeForm/%20,^012567,01245$,`@9
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "12-1",
    "inversion": null
  },
  {
    "number": "6-z6",
    "primeForm": "[\"0\",\"1\",\"2\",\"5\",\"6\",\"7\"]",
    "vec": "<4,2,1,2,4,2>",
    "z": "6-z38",
    "complement": "6-z38",
    "inversion": null
  },
  {
    "number": "7-7B",
    "primeForm": "[\"0\",\"1\",\"2\",\"5\",\"6\",\"7\",\"8\"]",
    "vec": "<5,3,2,3,5,3>",
    "z": null,
    "complement": "5-7A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"6\",\"7\",\"8\"]"
  },
  {
    "number": "5-3A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\"]",
    "vec": "<3,2,2,2,1,0>",
    "z": null,
    "complement": "7-3B",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\"]"
  }
];
```

### GET /api/data/vec/:querySearch/

The endpoint returns an array of objects based on the query on the vec property

Use T for 10, E for 11, and C for 12.

- Max URI length: No more than 100 characters
- Subquery length: 2-8 characters
	- Ex. ^1 or !`000000
 
#### Exact Search

```ts
// GET /api/data/vec/321000
[
  {
    "number": "4-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\"]",
    "vec": "<3,2,1,0,0,0>",
    "z": null,
    "complement": "8-1",
    "inversion": null
  }
];
```

#### Wildcard Search

```ts
// GET /api/data/vec/1121X1
// capital X must be used as the wildcard
[
  {
    "number": "4-12A",
    "primeForm": "[\"0\",\"2\",\"3\",\"6\"]",
    "vec": "<1,1,2,1,0,1>",
    "z": null,
    "complement": "8-12A",
    "inversion": "[\"0\",\"3\",\"4\",\"6\"]"
  },
  {
    "number": "4-12B",
    "primeForm": "[\"0\",\"3\",\"4\",\"6\"]",
    "vec": "<1,1,2,1,0,1>",
    "z": null,
    "complement": "8-12B",
    "inversion": "[\"0\",\"2\",\"3\",\"6\"]"
  }
];
```

#### Starts With Search

```ts
// GET /api/data/vec/^321
[
  {
    "number": "4-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\"]",
    "vec": "<3,2,1,0,0,0>",
    "z": null,
    "complement": "8-1",
    "inversion": null
  },
  {
    "number": "5-5A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"7\"]",
    "vec": "<3,2,1,1,2,1>",
    "z": null,
    "complement": "7-5B",
    "inversion": "[\"0\",\"4\",\"5\",\"6\",\"7\"]"
  },
  {
    "number": "5-5B",
    "primeForm": "[\"0\",\"4\",\"5\",\"6\",\"7\"]",
    "vec": "<3,2,1,1,2,1>",
    "z": null,
    "complement": "7-5A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"7\"]"
  }
];
```

#### Ends With Search

```ts
// GET /api/data/vec/0111$
[
  {
    "number": "4-5A",
    "primeForm": "[\"0\",\"1\",\"2\",\"6\"]",
    "vec": "<2,1,0,1,1,1>",
    "z": null,
    "complement": "8-5B",
    "inversion": "[\"0\",\"4\",\"5\",\"6\"]"
  },
  {
    "number": "4-5B",
    "primeForm": "[\"0\",\"4\",\"5\",\"6\"]",
    "vec": "<2,1,0,1,1,1>",
    "z": null,
    "complement": "8-5A",
    "inversion": "[\"0\",\"1\",\"2\",\"6\"]"
  }
];
```

#### Contains Search

```ts
// GET /api/data/vec/@3250
[
  {
    "number": "6-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\"]",
    "vec": "<5,4,3,2,1,0>",
    "z": null,
    "complement": null,
    "inversion": null
  },
  {
    "number": "6-32",
    "primeForm": "[\"0\",\"2\",\"4\",\"5\",\"7\",\"9\"]",
    "vec": "<1,4,3,2,5,0>",
    "z": null,
    "complement": null,
    "inversion": null
  }
];
```

#### Substring Search

```ts
// GET /api/data/vec/*3250 or // GET /api/data/vec/*325
[
  {
    "number": "6-32",
    "primeForm": "[\"0\",\"2\",\"4\",\"5\",\"7\",\"9\"]",
    "vec": "<1,4,3,2,5,0>",
    "z": null,
    "complement": null,
    "inversion": null
  }
];
```

#### Not Search

- Works with !0, !^0, !0$, !@0, !*0,
  - ! is equivalent to !` 
  - `! is not a valid search method

```ts
// GET /api/data/primeForm/!000000
[
  {
    "number": "2-1",
    "primeForm": "[\"0\",\"1\"]",
    "vec": "<1,0,0,0,0,0>",
    "z": null,
    "complement": "10-1",
    "inversion": null
  },
    ...,
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Exclude Search

- Works with \`0, \`^0, \`0$, \`@0, \`*0

```ts
// GET /api/data/vec/!000000,`@0,`@1,`@2,`^8,`^6,`@5,`@4,`@7
[
  {
    "number": "7-31A",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"6\",\"7\",\"9\"]",
    "vec": "<3,3,6,3,3,3>",
    "z": null,
    "complement": "5-31B",
    "inversion": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"8\",\"9\"]"
  },
  {
    "number": "7-31B",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"8\",\"9\"]",
    "vec": "<3,3,6,3,3,3>",
    "z": null,
    "complement": "5-31A",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"6\",\"7\",\"9\"]"
  },
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Chaining Methods (no duplicates)

```ts
// GET /api/data/vec/321000,^353,441$
[
  {
    "number": "4-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\"]",
    "vec": "<3,2,1,0,0,0>",
    "z": null,
    "complement": "8-1",
    "inversion": null
  },
  {
    "number": "7-24A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"7\",\"9\"]",
    "vec": "<3,5,3,4,4,2>",
    "z": null,
    "complement": "5-24B",
    "inversion": "[\"0\",\"2\",\"4\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "7-24B",
    "primeForm": "[\"0\",\"2\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<3,5,3,4,4,2>",
    "z": null,
    "complement": "5-24A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"7\",\"9\"]"
  },
  {
    "number": "7-11A",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\",\"8\"]",
    "vec": "<4,4,4,4,4,1>",
    "z": null,
    "complement": "5-11B",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\"]"
  },
  {
    "number": "7-11B",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\"]",
    "vec": "<4,4,4,4,4,1>",
    "z": null,
    "complement": "5-11A",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\",\"8\"]"
  }
];
```

### GET /api/data/vec/:querySearch/:queryInequality/

The endpoint returns an array of objects based on the query on the vec property using inequalities. It checks the truthiness of the chosen inequality for every position.

The endpoint only works with the exact search method as in the /api/data/vec/:querySearch endpoint, you can still use ! and \` to filter out what you do not need. Other inclusion methods (\^\$\@\*) work but do not get effected by :queryInequality.
	- /api/data/vec/:querySearch/e is equivalent to /api/data/vec/:querySearch/

- Max URI length: No more than 100 characters
- Subquery length: 2-8 characters
	- Ex. ^1 or !`000000
- Subquery length for Inequality: 1-2 characters
	- Ex. e or le

#### Equal Search

```ts
// GET /api/data/vec/555553/e
[
  {
    "number": "8-z15A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15B",
    "inversion": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-z15B",
    "primeForm": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]"
  },
  {
    "number": "8-z29A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z15A",
    "complement": "4-z29B",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-z29B",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z15A",
    "complement": "4-z29A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"9\"]"
  }
];
```


#### Less Than Search

```ts
// GET /api/data/vec/555553/l
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "12-1",
    "inversion": null
  },
    ...
  {
    "number": "7-z38B",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"6\",\"7\",\"8\"]",
    "vec": "<4,3,4,4,4,2>",
    "z": "7-z18A",
    "complement": "5-z38A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"7\",\"8\"]"
  }
];
```

#### Less Than Or Equal To Search

```ts
// GET /api/data/vec/555553/le
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "12-1",
    "inversion": null
  },
    ...
  {
    "number": "8-z29B",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z15A",
    "complement": "4-z29A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"9\"]"
  }
];
```

#### Greater Than Search

```ts
// GET /api/data/vec/555553/g
[
  {
    "number": "9-5A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<7,6,6,6,7,4>",
    "z": null,
    "complement": "3-5B",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
    ...
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Greater Than Or Equal To Search

```ts
// GET /api/data/vec/555553/ge
[
  {
    "number": "8-z15A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15B",
    "inversion": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
    ...
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

### GET /api/data/z/:querySearch/

The endpoint returns an array of objects based on the query on the z property

- Max URI length: No more than 100 characters
- Subquery length: 2-8 characters
	- Ex. ^0 or !`6-z25A

#### Exact Search

```ts
// GET /api/data/z/null
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null
    "complement": "12-1",
    "inversion": null
  },
   ...
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];

// GET /api/data/z/5-z37
[
  {
    "number": "5-z17",
    "primeForm": "[\"0\",\"3\",\"4\",\"5\",\"8\"]",
    "vec": "<2,1,2,3,2,0>",
    "z": "5-z37",
    "complement": "7-z17",
    "inversion": null
  }
];
```

#### Starts With Search

```ts
// GET /api/data/z/^5-z3
[
  {
    "number": "5-z12",
    "primeForm": "[\"0\",\"1\",\"3\",\"5\",\"6\"]",
    "vec": "<2,2,2,1,2,1>",
    "z": "5-z36A",
    "complement": "7-z12",
    "inversion": null
  },
  {
    "number": "5-z17",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"8\"]",
    "vec": "<2,1,2,3,2,0>",
    "z": "5-z37",
    "complement": "7-z17",
    "inversion": null
  },
  {
    "number": "5-z18A",
    "primeForm": "[\"0\",\"1\",\"4\",\"5\",\"7\"]",
    "vec": "<2,1,2,2,2,1>",
    "z": "5-z38A",
    "complement": "7-z18A",
    "inversion": "[\"0\",\"2\",\"3\",\"6\",\"7\"]"
  },
  {
    "number": "5-z18B",
    "primeForm": "[\"0\",\"2\",\"3\",\"6\",\"7\"]",
    "vec": "<2,1,2,2,2,1>",
    "z": "5-z38A",
    "complement": "7-z18B",
    "inversion": "[\"0\",\"1\",\"4\",\"5\",\"7\"]"
  }
];
```

#### Ends With Search

```ts
// GET /api/data/z/23$
[
  {
    "number": "6-z45",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"9\"]",
    "vec": "<2,3,4,2,2,2>",
    "z": "6-z23",
    "complement": "6-z23",
    "inversion": null
  }
];
```

#### Contains Search

```ts
// GET /api/data/z/@45
[
  {
    "number": "4-z29A",
    "primeForm": "[\"0\",\"1\",\"3\",\"7\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z15A",
    "complement": "8-z29B",
    "inversion": "[\"0\",\"4\",\"6\",\"7\"]"
  },
  {
    "number": "4-z29B",
    "primeForm": "[\"0\",\"4\",\"6\",\"7\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z15A",
    "complement": "8-z29A",
    "inversion": "[\"0\",\"1\",\"3\",\"7\"]"
  },
  {
    "number": "6-z23",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"8\"]",
    "vec": "<2,3,4,2,2,2>",
    "z": "6-z45",
    "complement": "6-z45",
    "inversion": null
  }
];
```

#### Substring Search

```ts
// GET /api/data/z/*45
[
  {
    "number": "6-z23",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"8\"]",
    "vec": "<2,3,4,2,2,2>",
    "z": "6-z45",
    "complement": "6-z45",
    "inversion": null
  }
];
```

#### Not Search

- Works with !1-1, !^1-1, !1-1$, !@A, !*A,
  - ! is equivalent to !` 
  - `! is not a valid search method
- Use %20 for whitespace equivalent to the empty array for 0-1  

```ts
// GET /api/data/z/!null
[
  {
    "number": "4-z15A",
    "primeForm": "[\"0\",\"1\",\"4\",\"6\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z29A",
    "complement": "8-z15B",
    "inversion": "[\"0\",\"2\",\"5\",\"6\"]"
  },
    ...,
  {
    "number": "8-z29B",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z15A",
    "complement": "4-z29A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"9\"]"
  }
];
```

#### Exclude Search

- Works with \`1-1, \`^1-1, \`1-1$, \`@A, \`*A

```ts
// GET /api/data/z/7-z12,8-z29A,`7-z12
[
  {
    "number": "8-z15A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15B",
    "inversion": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-z15B",
    "primeForm": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]"
  }
];
```

#### Chaining Methods (no duplicates)

```ts
// GET /api/data/z/8-z29A,^7-z1,-z29A,`7-z18A
[
  {
    "number": "8-z15A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15B",
    "inversion": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-z15B",
    "primeForm": "[\"0\",\"1\",\"3\",\"5\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<5,5,5,5,5,3>",
    "z": "8-z29A",
    "complement": "4-z15A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"6\",\"8\",\"9\"]"
  },
  {
    "number": "7-z36A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"8\"]",
    "vec": "<4,4,4,3,4,2>",
    "z": "7-z12",
    "complement": "5-z36B",
    "inversion": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\"]"
  },
  {
    "number": "7-z36B",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\"]",
    "vec": "<4,4,4,3,4,2>",
    "z": "7-z12",
    "complement": "5-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"8\"]"
  },
  {
    "number": "7-z37",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"7\",\"8\"]",
    "vec": "<4,3,4,5,4,1>",
    "z": "7-z17",
    "complement": "5-z37",
    "inversion": null
  }
];
```

### GET /api/data/complement/:querySearch/

The endpoint returns an array of objects based on the query on the complement property

- Max URI length: No more than 100 characters
- Subquery length: 2-8 characters
	- Ex. ^0 or !`6-z25A

#### Exact Search

```ts
// GET /api/data/complement/null
[
  {
    "number": "6-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\"]",
    "vec": "<5,4,3,2,1,0>",
    "z": null,
    "complement": null,
    "inversion": null
  },
   ...
  {
    "number": "6-35",
    "primeForm": "[\"0\",\"2\",\"4\",\"6\",\"8\",\"T\"]",
    "vec": "<0,6,0,6,0,3>",
    "z": null,
    "complement": null,
    "inversion": null
  }
];

// GET /api/data/complement/5-z37
[
  {
    "number": "7-z37",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"7\",\"8\"]",
    "vec": "<4,3,4,5,4,1>",
    "z": "7-z17",
    "complement": "5-z37",
    "inversion": null
  }
];
```

#### Starts With Search

```ts
// GET /api/data/complement/^5-z36
[
  {
    "number": "7-z36A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"8\"]",
    "vec": "<4,4,4,3,4,2>",
    "z": "7-z12",
    "complement": "5-z36B",
    "inversion": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\"]"
  },
  {
    "number": "7-z36B",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\"]",
    "vec": "<4,4,4,3,4,2>",
    "z": "7-z12",
    "complement": "5-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"8\"]"
  }
];
```

#### Ends With Search

```ts
// GET /api/data/complement/23$
[
  {
    "number": "4-23",
    "primeForm": "[\"0\",\"2\",\"5\",\"7\"]",
    "vec": "<0,2,1,0,3,0>",
    "z": null,
    "complement": "8-23",
    "inversion": null
  },
  {
    "number": "6-z45",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"9\"]",
    "vec": "<2,3,4,2,2,2>",
    "z": "6-z23",
    "complement": "6-z23",
    "inversion": null
  },
  {
    "number": "8-23",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"7\",\"8\",\"T\"]",
    "vec": "<4,6,5,4,7,2>",
    "z": null,
    "complement": "4-23",
    "inversion": null
  }
];
```

#### Contains Search

```ts
// GET /api/data/complement/@36A
[
  {
    "number": "5-z36B",
    "primeForm": "[\"0\",\"3\",\"5\",\"6\",\"7\"]",
    "vec": "<2,2,2,1,2,1>",
    "z": "5-z12",
    "complement": "7-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"7\"]"
  },
  {
    "number": "6-z3B",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\"]",
    "vec": "<4,3,3,2,2,1>",
    "z": "6-z36A",
    "complement": "6-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\"]"
  },
  {
    "number": "6-z10B",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\",\"6\",\"7\"]",
    "vec": "<3,3,3,3,2,1>",
    "z": "6-z39A",
    "complement": "6-z39A",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"7\"]"
  },
  {
    "number": "6-z17B",
    "primeForm": "[\"0\",\"1\",\"4\",\"6\",\"7\",\"8\"]",
    "vec": "<3,2,2,3,3,2>",
    "z": "6-z43A",
    "complement": "6-z43A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"7\",\"8\"]"
  },
  {
    "number": "6-z36B",
    "primeForm": "[\"0\",\"3\",\"4\",\"5\",\"6\",\"7\"]",
    "vec": "<4,3,3,2,2,1>",
    "z": "6-z3A",
    "complement": "6-z3A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"7\"]"
  },
  {
    "number": "7-z36B",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\"]",
    "vec": "<4,4,4,3,4,2>",
    "z": "7-z12",
    "complement": "5-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"8\"]"
  }
];
```

#### Substring Search

```ts
// GET /api/data/complement/*36A
[
  {
    "number": "5-z36B",
    "primeForm": "[\"0\",\"3\",\"5\",\"6\",\"7\"]",
    "vec": "<2,2,2,1,2,1>",
    "z": "5-z12",
    "complement": "7-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"7\"]"
  },
  {
    "number": "6-z3B",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\"]",
    "vec": "<4,3,3,2,2,1>",
    "z": "6-z36A",
    "complement": "6-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\"]"
  },
  {
    "number": "7-z36B",
    "primeForm": "[\"0\",\"2\",\"3\",\"5\",\"6\",\"7\",\"8\"]",
    "vec": "<4,4,4,3,4,2>",
    "z": "7-z12",
    "complement": "5-z36A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"8\"]"
  }
];
```

#### Not Search

- Works with !1-1, !^1-1, !1-1$, !@A, !*A,
  - ! is equivalent to !` 
  - `! is not a valid search method
- Use %20 for whitespace equivalent to the empty array for 0-1  

```ts
// GET /api/data/complement/!null
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "12-1",
    "inversion": null
  },
    ...,
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];
```

#### Exclude Search

- Works with !1-1, !^1-1, !1-1$, !@A, !*A,

```ts
// GET /api/data/complement/1-1,11-1,`1-1
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null,
    "complement": "11-1",
    "inversion": null
  }
];
```

#### Chaining Methods (no duplicates)

```ts
// GET /api/data/complement/8-z29A,^7-z1,-z29A,`7-z18A,`B$
[
  {
    "number": "4-z29B",
    "primeForm": "[\"0\",\"4\",\"6\",\"7\"]",
    "vec": "<1,1,1,1,1,1>",
    "z": "4-z15A",
    "complement": "8-z29A",
    "inversion": "[\"0\",\"1\",\"3\",\"7\"]"
  },
  {
    "number": "5-z12",
    "primeForm": "[\"0\",\"1\",\"3\",\"5\",\"6\"]",
    "vec": "<2,2,2,1,2,1>",
    "z": "5-z36A",
    "complement": "7-z12",
    "inversion": null
  },
  {
    "number": "5-z17",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"8\"]",
    "vec": "<2,1,2,3,2,0>",
    "z": "5-z37",
    "complement": "7-z17",
    "inversion": null
  }
];
```
### GET /api/data/inversion/:querySearch/

The endpoint returns an array of objects based on the query on the inversion property

Use T for 10, E for 11, and C for 12.

- Max URI length: No more than 100 characters
- Subquery length: 2-14 characters
	- Ex. ^0 or !`0123456789TE

#### Exact Search

```ts
// GET /api/data/inversion/null
[
  {
    "number": "0-1",
    "primeForm": "[\"\"]",
    "vec": "<0,0,0,0,0,0>",
    "z": null
    "complement": "12-1",
    "inversion": null
  },
   ...
  {
    "number": "12-1",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\",\"T\",\"E\"]",
    "vec": "<C,C,C,C,C,6>",
    "z": null,
    "complement": "0-1",
    "inversion": null
  }
];

// GET /api/data/inversion/026
[
  {
    "number": "3-8B",
    "primeForm": "[\"0\",\"4\",\"6\"]",
    "vec": "<0,1,0,1,0,1>",
    "z": null,
    "complement": "9-8A",
    "inversion": "[\"0\",\"2\",\"6\"]"
  }
];
```

#### Starts With Search

```ts
// GET /api/data/inversion/^02367
[
  {
    "number": "5-z18A",
    "primeForm": "[\"0\",\"1\",\"4\",\"5\",\"7\"]",
    "vec": "<2,1,2,2,2,1>",
    "z": "5-z38A",
    "complement": "7-z18A",
    "inversion": "[\"0\",\"2\",\"3\",\"6\",\"7\"]"
  },
  {
    "number": "6-z43A",
    "primeForm": "[\"0\",\"1\",\"2\",\"5\",\"6\",\"8\"]",
    "vec": "<3,2,2,3,3,2>",
    "z": "6-z17A",
    "complement": "6-z17B",
    "inversion": "[\"0\",\"2\",\"3\",\"6\",\"7\",\"8\"]"
  }
];
```

#### Ends With Search

```ts
// GET /api/data/inversion/23$
[
  {
    "number": "3-2A",
    "primeForm": "[\"0\",\"1\",\"3\"]",
    "vec": "<1,1,1,0,0,0>",
    "z": null,
    "complement": "9-2A",
    "inversion": "[\"0\",\"2\",\"3\"]"
  }
];
```

#### Contains Search

```ts
// GET /api/data/inversion/@45789
[
  {
    "number": "7-27A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"7\",\"9\"]",
    "vec": "<3,4,4,4,5,1>",
    "z": null,
    "complement": "5-27B",
    "inversion": "[\"0\",\"2\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-11A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"7\",\"9\"]",
    "vec": "<5,6,5,5,5,2>",
    "z": null,
    "complement": "4-11B",
    "inversion": "[\"0\",\"2\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-14A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"7\",\"9\"]",
    "vec": "<5,5,5,5,6,2>",
    "z": null,
    "complement": "4-14A",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-19A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"8\",\"9\"]",
    "vec": "<5,4,5,7,5,2>",
    "z": null,
    "complement": "4-19B",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "9-2A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"9\"]",
    "vec": "<7,7,7,6,6,3>",
    "z": null,
    "complement": "3-2A",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "9-3A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"6\",\"8\",\"9\"]",
    "vec": "<7,6,7,7,6,3>",
    "z": null,
    "complement": "3-3B",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "9-4A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]",
    "vec": "<7,6,6,7,7,3>",
    "z": null,
    "complement": "3-4B",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "9-4B",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<7,6,6,7,7,3>",
    "z": null,
    "complement": "3-4A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  }
];
```

#### Substring Search

```ts
// GET /api/data/inversion/*45789
[
  {
    "number": "7-27A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"7\",\"9\"]",
    "vec": "<3,4,4,4,5,1>",
    "z": null,
    "complement": "5-27B",
    "inversion": "[\"0\",\"2\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-14A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"7\",\"9\"]",
    "vec": "<5,5,5,5,6,2>",
    "z": null,
    "complement": "4-14A",
    "inversion": "[\"0\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "8-19A",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"8\",\"9\"]",
    "vec": "<5,4,5,7,5,2>",
    "z": null,
    "complement": "4-19B",
    "inversion": "[\"0\",\"1\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  },
  {
    "number": "9-4B",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"7\",\"8\",\"9\"]",
    "vec": "<7,6,6,7,7,3>",
    "z": null,
    "complement": "3-4A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"4\",\"5\",\"7\",\"8\",\"9\"]"
  }
];
```

#### Not Search

- Works with !0, !^0, !0$, !@0, !*0,
  - ! is equivalent to !` 
  - `! is not a valid search method
- Use %20 for whitespace equivalent to the empty array for 0-1  

```ts
// GET /api/data/inversion/!null
[
  {
    "number": "3-2A",
    "primeForm": "[\"0\",\"1\",\"3\"]",
    "vec": "<1,1,1,0,0,0>",
    "z": null,
    "complement": "9-2A",
    "inversion": "[\"0\",\"2\",\"3\"]"
  },
    ...,
  {
    "number": "9-11B",
    "primeForm": "[\"0\",\"1\",\"2\",\"4\",\"5\",\"6\",\"7\",\"9\",\"T\"]",
    "vec": "<6,6,7,7,7,3>",
    "z": null,
    "complement": "3-11A",
    "inversion": "[\"0\",\"1\",\"2\",\"3\",\"5\",\"6\",\"7\",\"9\",\"T\"]"
  }
];
```

#### Exclude Search

- Works with \`0, \`^0, \`0$, \`@0, \`*0

```ts
// GET /api/data/inversion/0124,01245,`0124
[
  {
    "number": "5-3B",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\"]",
    "vec": "<3,2,2,2,1,0>",
    "z": null,
    "complement": "7-3A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"5\"]"
  }
];
```

#### Chaining Methods (no duplicates)

```ts
// GET /api/data/inversion/0124,^012567,01245$,`@9
[
  {
    "number": "4-2B",
    "primeForm": "[\"0\",\"2\",\"3\",\"4\"]",
    "vec": "<2,2,1,1,0,0>",
    "z": null,
    "complement": "8-2A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\"]"
  },
  {
    "number": "7-7A",
    "primeForm": "[\"0\",\"1\",\"2\",\"3\",\"6\",\"7\",\"8\"]",
    "vec": "<5,3,2,3,5,3>",
    "z": null,
    "complement": "5-7B",
    "inversion": "[\"0\",\"1\",\"2\",\"5\",\"6\",\"7\",\"8\"]"
  },
  {
    "number": "5-3B",
    "primeForm": "[\"0\",\"1\",\"3\",\"4\",\"5\"]",
    "vec": "<3,2,2,2,1,0>",
    "z": null,
    "complement": "7-3A",
    "inversion": "[\"0\",\"1\",\"2\",\"4\",\"5\"]"
  }
];
```

### GET /api/data/:queryProp/PROPERTY/:querySearch/

Using the GET /api/data/:queryProp endpoint you can filter out the properties you want from the resulting querySearch

Lengths For :queryProp\:
- Max URI length: No more than 43 characters
- Subquery length: 1-10 characters

##### Endpoints:
- /api/data/:queryProp/number/:querySearch/
- /api/data/:queryProp/primeForm/:querySearch/
- /api/data/:queryProp/vec/:querySearch/
- /api/data/:queryProp/vec/:querySearch/:queryInequality/
- /api/data/:queryProp/z/:querySearch/
- /api/data/:queryProp/complement/:querySearch/
- /api/data/:queryProp/inversion/:querySearch/

```ts
// GET /api/data/primeForm/number/1-1
[
  {
    "primeForm": "[\"0\"]"
  }
];

// GET /api/data/number,primeForm/number/1-1
[
  {
    "number": "1-1",
    "primeForm": "[\"0\"]"
  }
];
```

### GET /api/data/d3/:querySearch/

The endpoint returns either json for a valid d3dag graph or json for valid dag links

A option for forte numbers, encoded like this "[\"0\"]|1-1|000000", is added to allow the ability to toggle between primeForm, number, and vec on a d3dag by splitting on "|"

- Max URI length: No more than 22 characters

#### Types
```ts
type Links = { source: string; target: string }[];
type DagJSONObject = {
	size: { width: number; height: number };
	nodes: { x: number; y: number; data: string }[];
	links: { source: string; target: string; points: number[][]; data: Links };
	v: number;
};
```

#### Cardinality-Increasing vs Strict-Increasing

The operations here are done on the primeForm property.

Cardinality-increasing here means the target set is a proper superset of the source set where the target set is greater in length by 1

```ts
  {
    "source": "[\"0\"]",
    "target": "[\"0\",\"1\"]"
  }
```

Strict-increasing here means the target set is a proper superset of the source set where the target set is greater in length by 1 AND the next number of the target set must be greater than the largest number in source set (lexicographically greater).

When comparing [\"0\",\"2\",\"3\",\"4\"] > [\"0\",\"2\",\"3\"], it returns true
While comparing [\"0\",\"1\",\"3\",\"4\"] > [\"0\",\"2\",\"3\"], it returns false
```ts
  {
    "source": "[\"0\",\"2\",\"3\"]",
    "target": "[\"0\",\"2\",\"3\",\"4\"]"
  }
```

#### Vector-Similarity

The operations here are done on the vec property.

Vectors are compared using cosine similarity and are linked to the set with the highest cosine similarity.

#### Original vs Inversion

Original means no inversions (no B sets), but sets are still labelled 3-11A instead of 3-11.

Inversion means with inversions so it includes both A and B sets.

#### Manual Construction

The json files here can be recreated using the set_classes.json file.

##### Fetch

```ts
type Link = { source: string; target: string }
const links = ref<null | Link[]>(null)

const fetchData = async () => {
  try {
    const res = await fetch(
      'https://hcda8f8dtk.execute-api.us-east-1.amazonaws.com/prod/api/data/number,primeForm,vec/',
      { signal: abortController.signal }
    )
    if (res.ok) {
      const data: { number: string; primeForm: string; vec: string }[] = await res.json()
      links.value = linkBuilder(data)
      localStorage.setItem('links', JSON.stringify(links.value))
    } else {
      console.log('Not 200')
    }
  } catch (error) {
    if ((error as Error).name == 'AbortError') {
      console.log('AbortError', error)
    }
    console.log(error)
  }
}
```

##### Links
```ts
// Cardinality-Increasing
const linkBuilder = (
  data: { number: string; primeForm: string; vec: string }[]
) => {
  const newData: { number: string; primeForm: string[]; vec: string }[] = data.map((s) => ({
    primeForm: s.primeForm.slice(1, -1).split(','),
    number: s.number,
    vec: s.vec
  }))
  let links: Link[] = [{ source: '[""]|0-1|000000', target: '["0"]|1-1|000000' }]

  for (const s of newData) {
    for (const t of newData) {
      if (
        s.primeForm.every((e) => t.primeForm.includes(e)) &&
        s.primeForm.length === t.primeForm.length - 1
      ) {
        links.push({
          source:
            '[' + s.primeForm.toString() + ']' + '|' + s.number + '|' + s.vec.replace(/\D/g, ''),
          target:
            '[' + t.primeForm.toString() + ']' + '|' + t.number + '|' + t.vec.replace(/\D/g, '')
        })
      }
    }
  }
  return links
}

// Strictly-Increasing
const linkBuilder = (
  data: { number: string; primeForm: string; vec: string }[]
) => {
  const newData: { number: string; primeForm: string[]; vec: string }[] = data.map((s) => ({
    primeForm: s.primeForm.slice(1, -1).split(','),
    number: s.number,
    vec: s.vec
  }))
  let links: Link[] = [{ source: '[""]|0-1|000000', target: '["0"]|1-1|000000' }]

  for (const s of newData) {
    for (const t of newData) {
      if (
        s.primeForm.every((e) => t.primeForm.includes(e)) &&
        s.primeForm.length === t.primeForm.length - 1 &&
        t.primeForm > s.primeForm
      ) {
        links.push({
          source:
            '[' + s.primeForm.toString() + ']' + '|' + s.number + '|' + s.vec.replace(/\D/g, ''),
          target:
            '[' + t.primeForm.toString() + ']' + '|' + t.number + '|' + t.vec.replace(/\D/g, '')
        })
      }
    }
  }
  return links
}

// Vector-Similarity
const linkBuilder = (
  data: { number: string; primeForm: string; vec: string }[]
) => {
  const newData: { number: string; primeForm: string[]; vec: string }[] = data.map((s) => ({
    primeForm: s.primeForm.slice(1, -1).split(','),
    number: s.number,
    vec: s.vec
  }))
  let links: Link[] = [{ source: '[""]|0-1|000000', target: '["0"]|1-1|000000' }]

  for (const [i, s] of newData.entries()) {
    let max = 0
    let newS = null
    let newT = null
    for (const [j, t] of newData.entries()) {
      const sVec = s.vec.replace(/[<>]/g, '').replace(/C/g, '12').replace(/T/g, '10').split(',')
      const tVec = t.vec.replace(/[<>]/g, '').replace(/C/g, '12').replace(/T/g, '10').split(',')

      const dotProd = tVec
        .map((a, i) => parseInt(a) * parseInt(sVec[i]))
        .reduce((acc, curr) => acc + curr, 0)
      const cosSim =
        dotProd /
        (Math.sqrt(sVec.reduce((sum, val) => sum + parseInt(val) * parseInt(val), 0)) *
          Math.sqrt(tVec.reduce((sum, val) => sum + parseInt(val) * parseInt(val), 0)))

      if (i < j && cosSim > max) {
        max = cosSim
        newS = s
        newT = t
      }
    }
    if (newS && newT) {
      links.push({
        source:
          '[' +
          newS.primeForm.toString() +
          ']' +
          '|' +
          newS.number +
          '|' +
          newS.vec.replace(/(?![TEC])\D/g, ''),
        target:
          '[' +
          newT.primeForm.toString() +
          ']' +
          '|' +
          newT.number +
          '|' +
          newT.vec.replace(/(?![TEC])\D/g, '')
      })
    }
  }
  return links
}

// For original dags add this to the condition
!s.number.endsWith('B') &&
!t.number.endsWith('B')
```

##### Dag

```ts
import * as d3 from 'd3'
import * from 'd3-dag'
const builder = graphConnect()
    .sourceId(({ source }: { source: string }) => source)
    .targetId(({ target }: { target: string }) => target)
const dagBuild = builder(links.value)
const layout = sugiyama()
    .layering(layeringSimplex())
    .decross(decrossTwoLayer().order(twolayerGreedy().base(twolayerAgg())))
    .coord(coordSimplex())
    .nodeSize([2 * NODE_RADIUS, 2 * NODE_RADIUS])
    .gap([NODE_RADIUS, NODE_RADIUS])
    .tweaks([tweakShape([2 * NODE_RADIUS, 2 * NODE_RADIUS], shapeEllipse)])

const { width, height } = layout(dagBuild as any)

JSON.stringify(dagBuild) // NOTE: I added size property with width and height during post-processing, it is not a default property returned in the JSON by d3dag
```

#### How to Use JSON in D3Dag

```ts
type Link = { source: string; target: string }
type DagJSONObject = {
	size: { width: number; height: number };
	nodes: { x: number; y: number; data: string }[];
	links: { source: string; target: string; points: number[][]; data: Link[] };
	v: number;
};

const res = await fetch('.../api/data/d3/strictdagprimeform/')
const data: DagJSONObject = await res.json() 
const size: {width: number, height: number} = data.size // use to adjust the starting position of the graph or set the graph size

const builder = graphJson()
      .nodeDatum((data) => data as string)
      .linkDatum((data) => data as Link)
const dag = builder(JSON.parse(data))
// perform d3 visualizations using dag.nodes() and dag.links()
```

#### Endpoints

Valid queries are:
  - cardinalinversiondag
  - cardinalinversionlinks
  - cardinaloriginaldag
  - cardinaloriginallinks
  - strictinversiondag
  - strictinversionlinks
  - strictoriginaldag
  - strictoriginallinks
  - vectorinversiondag
  - vectorinversionlinks
  - vectororiginaldag
  - vectororiginallinks

## API Development

Run ```npm i```

### Add .env File

```env
NODE_ENV=development
PORT=[choose any port]
```

## Star This Repo

If you like this API, please give it a star! _\~ Created by Khang Tran_
 readmeEtag: '"5977867a76a2944632a5773c48c1a067fc0336a1"' readmeLastModified: Mon, 20 Nov 2023 05:04:25 GMT repositoryId: 676712807 description: >- A music theory API that provides a way to query set classes using Express and TypeScript, Tested with Mocha-Chai created: '2023-08-09T20:46:55Z' updated: '2023-09-22T21:00:34Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 0 owner: NinjaNas logo: https://avatars.githubusercontent.com/u/54213302?v=4 license: MIT repoEtag: '"59b4e88ba3489801cba75d58087d47b474d65b203824992706beefc20bc73f6b"' repoLastModified: Fri, 22 Sep 2023 21:00:34 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/jorgebsa/spring-demo v3: true repositoryMetadata: base64Readme: >- ## Spring demo microservice
[![GitHub Workflow Status](https://img.shields.io/github/workflow/status/jorgebsa/spring-demo/Java%20CI%20with%20Gradle?style=for-the-badge)](https://github.com/jorgebsa/spring-demo/actions/workflows/gradle.yml) 
[![Codecov](https://img.shields.io/codecov/c/github/jorgebsa/spring-demo?style=for-the-badge&token=55F3FYDVUN)](https://codecov.io/gh/jorgebsa/spring-demo) 
[![GitHub last commit](https://img.shields.io/github/last-commit/jorgebsa/spring-demo?style=for-the-badge)](https://github.com/jorgebsa/spring-demo/commits/main) 

[![Java version](https://img.shields.io/badge/Java%20version-17-brightgreen?style=for-the-badge)](https://openjdk.java.net/projects/jdk/17/)
[![Spring Boot version](https://img.shields.io/badge/Spring%20Boot%20version-2.6.2-brightgreen?style=for-the-badge)](https://docs.spring.io/spring-boot/docs/2.6.2/reference/html/getting-started.html#getting-started)
[![Gradle wrapper version](https://img.shields.io/badge/Gradle%20version-7.3.3-brightgreen?style=for-the-badge)](https://docs.gradle.org/7.3.3/release-notes.html)
[![MongoDB container version](https://img.shields.io/badge/MongoDB%20version-5.0.4-brightgreen?style=for-the-badge)](https://docs.mongodb.com/manual/release-notes/5.0-changelog/#std-label-5.0.4-changelog)

## Table of contents

1. [Introduction](#introduction)
2. [About the API](#about-the-api)
3. [Language Version](#language-version)
   1. [JDK 17](#jdk-17)
   2. [Managing JDKs with SDKMAN!](#managing-jdks-with-sdkman)
   3. [Available Toolchains](#available-toolchains)
4. [How to Build](#how-to-build)
5. [Testing](#testing)
   1. [Why JUnit 5?](#why-junit-5)
   2. [Why AssertJ?](#why-assertj)
   3. [Why Testcontainers?](#why-testcontainers)
   4. [What is JaCoCo?](#what-is-jacoco)
      1. [JaCoCo logging](#jacoco-logging)
   5. [Mutation Testing](#mutation-testing)
   6. [Future-proofing](#future-proofing)
6. [Required services](#required-services)
   1. [Docker Compose](#docker-compose)
   2. [Keycloak](#keycloak)
7. [Running the microservice](#running-the-microservice)
   1. [Running the application with Gradle](#running-the-application-with-gradle)
   2. [Consuming the API](#consuming-the-api)
8. [Contributing](#contributing)

## Introduction

This project is about showcasing some of the best practices and technologies that can be 
used to ensure code quality while developing a microservice in Java with Spring Boot. 
Therefore, the focus isn't on creating a complex API but rather on how to achieve a high 
code quality standard, using a variety of mature and free libraries and technologies.

The project has been implemented with:
* Language: Java 17
* Framework: Spring Boot 
* Database: MongoDB
* Build tool: Gradle
* Security: Spring Security + Keycloak

Relevant testing information and libraries will be described in the appropriate section
of this document.

If you are not familiar with Gradle, the most relevant commands and tasks will be explained
in the following sections. If you need further help, check its [documentation](https://docs.gradle.org/7.3.3/userguide/userguide.html)
or execute the help task:

```shell
./gradlew help
```

## About the API

The API is rather simple: this microservice allows users to keep track of their "notes", 
and offers endpoints for the basic [CRUD](https://en.wikipedia.org/wiki/Create,_read,_update_and_delete) 
operations. 

Endpoints are secured by OAuth2, so each request must include a valid access token to be
processed. These tokens are issued by Keycloak, and the instructions on how to obtain them
will be detailed in a following section of this document.

No frontend application was developed, but a Swagger UI is available in order to explore
and consume the API.

## Language Version

Gradle's [toolchain](https://docs.gradle.org/7.3.3/userguide/toolchains.html) concept is 
used in order to define which java version will be used to build the project. This is 
declared in the [build.gradle.kts](build.gradle.kts) file:

```
java {
    toolchain {
        languageVersion.set(JavaLanguageVersion.of(17))
    }
}
```

### JDK 17

Since the project is implemented with Java 17, a JDK for Java 17 must be available on your 
machine in order to build this project.

Don't worry! If a matching JDK is not found by Gradle on your machine, it will automatically
download it before building the project. If you want Gradle to download a distribution JDK 
from a specific vendor, you can do so by setting the [vendor property](https://docs.gradle.org/7.3.3/userguide/toolchains.html#sec:vendors)
in the [build.gradle.kts](build.gradle.kts) file:

```
java {
    toolchain {
        languageVersion.set(JavaLanguageVersion.of(17))
        vendor.set(JvmVendorSpec.GRAAL_VM)
    }
}
```

### Managing JDKs with SDKMAN!

Most developers want to control which JDKs are installed on their machines. If you would
like to manage JDKs by yourself, you can use tools such as [SDKMAN!](https://sdkman.io/) 
to simplify that task. This is not meant to compare SDKMAN! with any other tool, but rather
explain how you can easily manage SDKs with a mature tool.

The [installation](https://sdkman.io/install) is very straightforward, and you can easily
install any JDK with a single command. For example, in order to install GraalVM 21.3.0 
with Java 17 you can execute the following command:

```shell
sdk install java 21.3.0.r17-grl 
```

After installing it, SDKMAN! will ask if you want to set it as the default JDK on your 
machine. If you chose to not set it as the default JDK, you can later set it as the 
active JDK in your terminal window by running the `use` command:

```shell
sdk use java 21.3.0.r17-grl
```

More info on SDKMAN usage can be found [here](https://sdkman.io/usage)

### Available Toolchains

In order to check which toolchains are available to Gradle, you can execute the 
`javaToolchains` task:

```shell
./gradlew javaToolchains
```

Here's an example of the output you should expect from the task when multiple 
toolchains are available:

```
> Task :javaToolchains

 + GraalVM Community JDK 11.0.13+7-jvmci-21.3-b05
     | Location:           /Users/username/.sdkman/candidates/java/21.3.0.r11-grl
     | Language Version:   11
     | Vendor:             GraalVM Community
     | Is JDK:             true
     | Detected by:        Current JVM

 + GraalVM Community JDK 17.0.1+12-jvmci-21.3-b05
     | Location:           /Users/username/.sdkman/candidates/java/21.3.0.r17-grl
     | Language Version:   17
     | Vendor:             GraalVM Community
     | Is JDK:             true
     | Detected by:        SDKMAN!

 + OpenJDK 18-ea+24-1608
     | Location:           /Users/username/.sdkman/candidates/java/18.ea.24-open
     | Language Version:   18
     | Vendor:             Oracle
     | Is JDK:             true
     | Detected by:        SDKMAN!

 + Oracle JDK 1.8.0_162-b12
     | Location:           /Library/Java/JavaVirtualMachines/jdk1.8.0_162.jdk/Contents/Home
     | Language Version:   8
     | Vendor:             Oracle
     | Is JDK:             true
     | Detected by:        macOS java_home
```

## How to Build

Gradle was chosen as the build tool for this project and its wrapper has been included 
in the repository. There are at least two very good reasons to advocate for wrapper
inclusion and usage:

* Project can be immediately built after cloning is finished, there's no need for 
the developer to install Gradle beforehand.
* The wrapper will always match the expected Gradle version that is meant to be used
when building the project. 

In order to build the project, execute the following command:

```shell
./gradlew build
```

Please note that the `build` task will also execute the tests. If you simply want to 
compile the source code and resources, you can do so by executing the `assemble` task:

```shell
./gradlew assemble
```

## Testing

Testing is crucial to ensure code is reliable, maintainable, reusable, extensible. All
of these factors influence the code's quality standard.

Many libraries can be used to improve the testing experience. In this project some great
libraries and tools are showcased, such as:

* [JUnit 5](https://junit.org/junit5/) to create and run our tests
* [AssertJ](https://joel-costigliola.github.io/assertj/) to write amazing assertions in 
our test cases
* [Testcontainers](https://www.testcontainers.org/) to reproduce a real environment for
our tests
* [JaCoCo](https://www.jacoco.org/jacoco/trunk/index.html) to generate the coverage 
report of our project
* [Pitest](https://pitest.org/) to execute mutation testing
* [Codecov](https://about.codecov.io/) to keep track of our repository's coverage

Integration and Unit tests can be found in the `src/test` directory. Usually, they
could've been split into different source sets, such as `src/test` for unit tests and 
`src/it` for integration tests, but that was not deemed necessary for this project's 
purpose, given its size, complexity and scope. However, this might change in the future.

In order to run the tests in this project, execute the `test` task:

```shell
./gradlew test
```

When execution ends, reports will be available at:

* `build/reports/tests/test` for JUnit reports
* `build/reports/jacoco/test` for JaCoCo reports

### Why JUnit 5?

JUnit is the most used testing platform for Java, with version 5 being its most 
recent major version. It has many features and advantages over JUnit 4, some
of which will be used in our tests, such as:

* `@ParameterizedTest`: this allows the execution of the same test case with any 
number of varying inputs (example: testing the same endpoint with different payloads)
* `Assertions.assertAll(...)`: executes all assertions in a block without failing
the test as soon as the first assertion fails, allowing for all failed assertions 
to be grouped and displayed together
* `Assertions.assertThrows(...)`: which helps the assertion of an exception and 
its properties 

### Why AssertJ?

The purpose of this project is not to compare assertion libraries, such as AssertJ,
Truth, Hamcrest and others. 

AssertJ is an amazing library, and is already included when importing the Spring 
Boot's testing dependency, so that's why it was used in this project. 

Writing meaningful and helpful assertions with ease is the goal, and AssertJ does a 
great job at it.

### Why Testcontainers?

You want your API to be reliable, extensible, maintainable, reusable, and that 
assumption has many consequences. 

One way to ensure that you don't break anything when developing new features or fixing
bugs is to have a very thorough test suite, and having a continuous integration pipeline
that automatically runs those tests and lets you know when things go wrong.

However, what if your tests do not reproduce the real environment? The execution of the
test suite can be successful and once you deploy the new version to your real world 
production environment, things that you were lead to assume were fine still run the risk
of breaking because the environment they were deployed to was nothing like what you 
tested against.

One of the ways to reduce this kind of problem is to use real instances of the services that
the application relies on, with the same version as the ones in your production environment.
For example, instead of testing an application against an embedded database, have it connect
to a real database server. 

This is where Testcontainers comes in handy, allowing you to spin up disposable Docker 
containers that will live only during the execution of your test suite. 

In this case, the existing `org.testcontainers:mongodb` dependency was used to import its 
`MongoDBContainer` class which was extended in the `ExtendedMongoDBContainer` class in 
order to ensure that the appropriate `URL` connection value was made available to Spring
Data once the container was ready to accept connections.

For the Keycloak authorization server, the `com.github.dasniko:testcontainers-keycloak` 
dependency was used. Similarly to the MongoDB case, the `KeycloakContainer` was extended 
by the `ExtendedKeycloakContainer` class, which is used to configure Keycloak's test realm 
as well as making its auth URL available to Spring Security.

#### IMPORTANT!

Using `testcontainers` means that most of the project's tests REQUIRE docker to be available 
in the host machine in order to run at all.

The Docker images being used in the tests are:

* `mongo:5.0.4`
* `quay.io/keycloak/keycloak:15.0.2`

If they are not found in the host machine, `testcontainers` will automatically download 
them. This can give the impression that the tasks are taking too long to start or finish,
so if you wish to manually download the images before executing the tests, you can by
executing the `pull` command:

```shell
docker pull mongo:5.0.4
docker pull quay.io/keycloak/keycloak:15.0.2
```

### What is JaCoCo?

JaCoCo is an amazing tool that generates code coverage reports for java. In order to 
generate these reports, the [JaCoCo Gradle plugin](https://docs.gradle.org/7.3.3/userguide/jacoco_plugin.html) 
is used, and is declared in the [build.gradle.kts](build.gradle.kts) plugins block. 

The plugin provides two useful tasks: 

* `jacocoTestReport`:  Generates the report and depends on the output of tests. In this
project, the gradlew `test` task is finalized by this task, so that the coverage report
is always generated after test execution
* `jacocoTestCoverageVerification`: verifies if the coverage rules are met. If a project
declares min 99% coverage, the task will fail if the coverage is below that threshold

In this project, the plugin is configured to generate both HTML and XML reports. 
 * HTML provides the best visual experience, as it allows a developer to navigate through
the packages and classes, checking all the coverage details.
 * The XML report is created so that is exported to [Codecov](https://codecov.io/) when 
the GitHub Actions workflow is executed

These reports can be found in `build/reports/jacoco`, and they include information about:

* Class coverage 
* Method coverage 
* Branch coverage 
* Line coverage
* Instruction coverage 
* Complexity coverage 

More information on what those counters mean can be found [here](https://www.jacoco.org/jacoco/trunk/doc/counters.html).

#### JaCoCo logging

Another neat plugin used is [jacocolog](https://gitlab.com/barfuin/gradle-jacoco-log),
which provides the`jacocoLogTestCoverage` task which prints all those metrics 
in the console in case you are only looking for the consolidated coverage numbers

Having near 100% code coverage does not mean that a software is bug-free or flawless.
But it usually means that a team is invested into making it reliable, and that type
of confidence in the code base provides a lot of comfort when planning and building
new features or simply fixing bugs. 

### Mutation Testing

This is not a very widespread concept, but is a rather powerful one. Below is the 
explanation of why you should care about it, from the folks at [Pitest](https://pitest.org):

> Traditional test coverage (i.e line, statement, branch, etc.) measures only which 
> code is executed by your tests. It does not check that your tests are actually able
> to detect faults in the executed code. It is therefore only able to identify code
> that is definitely not tested.
>
> The most extreme examples of the problem are tests with no assertions. Fortunately 
> these are uncommon in most code bases. Much more common is code that is only 
> partially tested by its suite. A suite that only partially tests code can still 
> execute all its branches (examples).
> 
> As it is actually able to detect whether each statement is meaningfully tested, 
> mutation testing is the gold standard against which all other types of coverage
> are measured.

In order to run mutation testing, the [Pitest](https://gradle-pitest-plugin.solidsoft.info/)
Gradle plugin is used. It provides the `pitest` gradle task, which runs the mutated 
tests, and the generated report can be found in`build/reports/pitest/index.html`

#### IMPORTANT!

Given many mutations are generated for each test case, the execution time of the 
`pitest` task is much longer than the time from the `test` task. Since this may take
several minutes, there's no GitHub action configured for it.

The current version of the Pitest Gradle plugin is not compatible with the Gradle 
Toolchain concept. Therefore, if the JVM used by Gradle to run does not match the 
language version declared by the toolchain, the execution of `pitest` task will
fail. In order to avoid this, make sure you set the JDK used by Gradle to be
compatible with the toolchain version. If you are using SDKMAN! this can be easily
done by executing the `sdk use java` command before you invoke the `pitest` task.

### Future-proofing

One way to future-proof your application is to test it against early access builds of
future JDKs. This can save you a lot of time when migrating to newer versions of the
JDK, as it will make it clear if your current code and dependencies are compatible
with the next release.

Since this project is written in Java 17, the only early access builds it can be tested
against are from JDK 18 (at the time of writing this). Therefore, a Gradle task called 
`testsOn18` has been created and declared in the [build.gradle.kts](build.gradle.kts) 
file. This task sets the Java Toolchain to use Java Language Version 18 and executes the
project's tests.

To run the tests on JDK 18, execute the following command:

````shell
./gradlew testsOn18
````

Once the tests finish, you can access the reports at `build/reports/tests/testsOn18`

Please note that since an early access build of the JDK is required to test against a
future version of the JDK, you must install it manually, as Gradle's toolchain isn't 
capable of downloading one. This might change in the future, as Gradle has an open 
[issue](https://github.com/gradle/gradle/issues/14814) for this feature. 

If you are using SDKMAN! as suggested in the previous sections, you can easily
download an early access build of the JDK. At the time of writing, the latest build
was `18.ea.25` which can be installed by executing:

```shell
sdk install java 18.ea.25-open
```

Once installed, Gradle's Toolchain is capable of automatically locating the early
access build of the JDK and use it when required.

#### IMPORTANT

The current version of JaCoCo is `0.8.7` and it isn't compatible with Java 18, 
therefore an exception will be thrown when running the tests on 18. However, this 
doesn't mean that the tests have failed or won't be executed, just that JaCoCo won't
be able to generate its reports from the test output. This should be fixed when 
JaCoCo version `0.8.8` is released.

## Required services

This microservice requires both MongoDB and Keycloak servers to be available in order
to function. If you have both servers installed or available on a remote URL, you can 
use such instances, just don't forget to update corresponding properties on 
`src/main/resources/application.yml` (or another profile's file) with the appropriate
connection URLs.

If you don't want to alter the properties file, you can also provide the URIs as 
JVM properties

```
-Dkeycloak.auth-server-url=http://some.host:8080/auth
-Dspring.data.mongodb.uri=mongodb://user:password@some.host:27017/spring-demo
```

or even as application arguments.

```
--keycloak.auth-server-url=http://some.host:8080/auth
--spring.data.mongodb.uri=mongodb://user:password@some.host:27017/spring-demo
```

### Docker Compose

However, you can also easily spin up both MongoDB and Keycloak as docker containers. 
A file called `docker-compose.yml` has been added to the project, and you can use 
[Docker Compose](https://docs.docker.com/compose/) in order to take advantage of it.

To start the containers with Docker Compose, execute:

```shell
docker-compose up -d
```

To stop them, execute:

```shell
docker-compose down
```

Please note that the services described in the `docker-compose.yml` file do not 
have persistent storage configured, which will cause modifications to the 
databases to be lost once the containers are stopped. This is intentional 
behaviour. 

If you'd like data to be persisted when the containers are stopped, you can use
the `docker-compose-persistent.yml` file instead. To do so, execute the commands
above with the `-f` flag:

```shell
docker-compose -f docker-compose-persistent.yml up -d
docker-compose -f docker-compose-persistent.yml down
```

The main difference in this approach is that PostgreSQL is used as the backing
database for Keycloak, running on its own container, and that both MongoDB and 
PostgreSQL containers are mounting data [volumes](https://docs.docker.com/storage/volumes/),
which are managed by the Docker Engine, in the appropriate container directories 

### Keycloak

Keycloak uses the concept of [realm](https://huongdanjava.com/overview-about-realm-in-keycloak.html) 
to manage Users, Clients, Roles and other security details. There are many ways
to configure a realm:

* You can do it [manually](https://www.keycloak.org/docs/latest/server_admin/#_create-realm)
* You can use Keycloak's [REST API](https://www.keycloak.org/docs-api/15.0/rest-api/index.html) 
(or one if its existing clients)
* You can import an existing `realm` configuration.

In the [realm.json](realm.json) file, you can inspect the basic Keycloak realm 
configuration needed in order for the microservice to work as expected. This file
is automatically imported by Keycloak when it starts if you use the project's 
provided docker compose files.

During tests, instead of importing the realm, it is created programmatically. 
This is done in the `com.github.jorgebsa.spring.demo.util.ExtendedKeycloakContainer` 
class. To achieve this, the `Keycloak` REST client from the 
`org.keycloak:keycloak-admin-client` dependency is used. 

## Running the microservice

Once your MongoDB and Keycloak instances are reachable, there are a few ways
to launch the application: 

* You can run it directly from your IDE, by invoking the main method of
`com.github.jorgebsa.spring.demo.Application`
* You can run it using the [Spring Boot Gradle plugin](https://docs.spring.io/spring-boot/docs/2.6.2/gradle-plugin/reference/htmlsingle/#running-your-application)

### Running the application with Gradle
 
Is as easy as invoking a simple [task](https://docs.spring.io/spring-boot/docs/2.6.2/gradle-plugin/reference/htmlsingle/#running-your-application)
from the Spring Boot Gradle Plugin:

```shell
./gradlew bootRun
```

### Consuming the API

After the application starts, you can send HTTP requests directly to it or 
access the Swagger UI available at `http://localhost:8080/swagger-ui.html`.

Remember that the endpoints are secured, therefore each request must include
an access token in its `Authorization` header. To obtain an access token, you
must request one from Keycloak with the appropriate payload, for example:

```http request
POST http://localhost:8081/auth/realms/tests/protocol/openid-connect/token
Accept: application/json
Cache-Control: no-cache
Content-Type: application/x-www-form-urlencoded

client_id=spring-demo&client_secret=the-secret&username=the-admin&password=123Admin&grant_type=password
```

You are looking for the `access_token` value from the response body, which will
be in the following format:

```json
{
  "access_token": "eyJhbGciOiJSUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJ2WENxeHpWT25aNW92a2Y5N0x0UEdZX0JXUkNuSWVRbUYwNU5nUUVzb2hRIn0.eyJleHAiOjE2Mzc5ODE3NjEsImlhdCI6MTYzNzk0NTc2MiwianRpIjoiMmU0ZDgxM2EtM2EwMS00MTg1LWI0NTUtY2EzNzY4ZDljYTNhIiwiaXNzIjoiaHR0cDovL2xvY2FsaG9zdDo4MDgxL2F1dGgvcmVhbG1zL3Rlc3RzIiwic3ViIjoiNDAxZTVjZGEtNzEzOS00YzljLWFjMjUtYWIxYTE5MjYzZDM5IiwidHlwIjoiQmVhcmVyIiwiYXpwIjoic3ByaW5nLWRlbW8iLCJzZXNzaW9uX3N0YXRlIjoiOWZlYzU4MzktMjNiZS00Y2RiLThiOWQtMTQ4OTg5NjMzNmJiIiwiYWNyIjoiMSIsImFsbG93ZWQtb3JpZ2lucyI6WyJodHRwOi8vbG9jYWxob3N0OjgwODAiXSwicmVzb3VyY2VfYWNjZXNzIjp7InNwcmluZy1kZW1vIjp7InJvbGVzIjpbIm5vdGVzLWFkbWluIiwibm90ZXMtdXNlciJdfX0sInNjb3BlIjoicHJvZmlsZSBlbWFpbCIsInNpZCI6IjlmZWM1ODM5LTIzYmUtNGNkYi04YjlkLTE0ODk4OTYzMzZiYiIsImVtYWlsX3ZlcmlmaWVkIjpmYWxzZSwicHJlZmVycmVkX3VzZXJuYW1lIjoidGhlLWFkbWluIn0.CII62FsOdCfe9qIyf9BfzBJUbfWjODgP6-BuhqFjQAeqt12yIXGcGXcyy4ZD55uYaBywMovOofMlmhIT_IGx6bRAHLwjYq6sH-TW_5wC_42rrZ4---UWkQn1zn8atdYJnDNHagEPRqMnzvE0R38Nk2otFiVpiYGodP_K-LuoORPKtqpb4LdbiR9kM-uhIuYnqxv_4cJuhH_3-wBFBuhI2uuAi12aHwrgfgWKDkAgc0I4uwNaQ8jIKG7MkeJhPOu81p1kYM99CK2mDYFhjggroR_wdwlvIG33Z_V2tVvHxuNdUCeg7JXTwGMT6t2xRGApW0I0IqeEQ_AdBitxBEeTaA",
  "expires_in": 35999,
  "refresh_expires_in": 1800,
  "refresh_token": "eyJhbGciOiJIUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJmMGVjMzU3OC1kZTA1LTRkZmItOWFjMC1kZWEwMTYzY2EzYjgifQ.eyJleHAiOjE2Mzc5NDc1NjIsImlhdCI6MTYzNzk0NTc2MiwianRpIjoiZjIzOWM1NzUtYzlhZC00OWNmLThmZGEtMDgwNzc4MDY0Y2Q1IiwiaXNzIjoiaHR0cDovL2xvY2FsaG9zdDo4MDgxL2F1dGgvcmVhbG1zL3Rlc3RzIiwiYXVkIjoiaHR0cDovL2xvY2FsaG9zdDo4MDgxL2F1dGgvcmVhbG1zL3Rlc3RzIiwic3ViIjoiNDAxZTVjZGEtNzEzOS00YzljLWFjMjUtYWIxYTE5MjYzZDM5IiwidHlwIjoiUmVmcmVzaCIsImF6cCI6InNwcmluZy1kZW1vIiwic2Vzc2lvbl9zdGF0ZSI6IjlmZWM1ODM5LTIzYmUtNGNkYi04YjlkLTE0ODk4OTYzMzZiYiIsInNjb3BlIjoicHJvZmlsZSBlbWFpbCIsInNpZCI6IjlmZWM1ODM5LTIzYmUtNGNkYi04YjlkLTE0ODk4OTYzMzZiYiJ9.RCiVcFlkK3hkHtFhVJHKze1oBf_upmB6vxVuZyCWYxk",
  "token_type": "Bearer",
  "not-before-policy": 0,
  "session_state": "9fec5839-23be-4cdb-8b9d-1489896336bb",
  "scope": "profile email"
}
```

Once you have the access token, you can use it to Authorize Swagger UI's requests by 
clicking on the `Authorize` button and pasting the value into the field. If you opt
to work with traditional HTTP requests, make sure to set the value of the `Authorization`
header to be of type `Bearer`, here's an example with the access token from above:

```http request
GET http://localhost:8080/notes
Accept: application/json
Authorization: Bearer eyJhbGciOiJIUzI1NiIsInR5cCIgOiAiSldUIiwia2lkIiA6ICJmMGVjMzU3OC1kZTA1LTRkZmItOWFjMC1kZWEwMTYzY2EzYjgifQ.eyJleHAiOjE2Mzc5NDc1NjIsImlhdCI6MTYzNzk0NTc2MiwianRpIjoiZjIzOWM1NzUtYzlhZC00OWNmLThmZGEtMDgwNzc4MDY0Y2Q1IiwiaXNzIjoiaHR0cDovL2xvY2FsaG9zdDo4MDgxL2F1dGgvcmVhbG1zL3Rlc3RzIiwiYXVkIjoiaHR0cDovL2xvY2FsaG9zdDo4MDgxL2F1dGgvcmVhbG1zL3Rlc3RzIiwic3ViIjoiNDAxZTVjZGEtNzEzOS00YzljLWFjMjUtYWIxYTE5MjYzZDM5IiwidHlwIjoiUmVmcmVzaCIsImF6cCI6InNwcmluZy1kZW1vIiwic2Vzc2lvbl9zdGF0ZSI6IjlmZWM1ODM5LTIzYmUtNGNkYi04YjlkLTE0ODk4OTYzMzZiYiIsInNjb3BlIjoicHJvZmlsZSBlbWFpbCIsInNpZCI6IjlmZWM1ODM5LTIzYmUtNGNkYi04YjlkLTE0ODk4OTYzMzZiYiJ9.RCiVcFlkK3hkHtFhVJHKze1oBf_upmB6vxVuZyCWYxk
```

## Contributing

This project is meant to be a way of showcasing some of the best practices and 
technologies I've come across in the past few years. I'm always interested in
learning more, discussing different points of views and approaches. If you can
contribute to this repository in any meaningful way, be it by showcasing something
new or improving what has been implemented already, you are more than welcome to
open a pull request and start a discussion! readmeEtag: '"1fac9e76a587c109e41ac069da91cece9f5f2569"' readmeLastModified: Wed, 22 Dec 2021 23:31:44 GMT repositoryId: 424862094 description: >- Demo microservice implemented with Spring Boot, showcasing tools such as Gradle, JaCoCo, Testcontainers and others created: '2021-11-05T07:27:48Z' updated: '2024-11-06T09:17:30Z' language: Java archived: true stars: 3 watchers: 1 forks: 0 owner: jorgebsa logo: https://avatars.githubusercontent.com/u/7410139?v=4 repoEtag: '"08225033905761dde0d7d184b8bee768153faaacd064330a7a27a0cb520ddcf3"' repoLastModified: Wed, 06 Nov 2024 09:17:30 GMT foundInMaster: true category: - Testing - Server Implementations id: 8caced4e5b7df4143e574e47313418d4 - source: - openapi3 tags - openapi31 tags repository: https://github.com/wojtekkarwacki/python-openapi-generator-rust v3: true v3_1: true id: d4a37ecbba0d8265cc4f653f92ffcf2d repositoryMetadata: base64Readme: >- <p>
  <b>Python Server OpenAPI Generator, Python Client OpenAPI Generator, Kotlin Server OpenAPI Generator</b>
</p>

---

<p align="center">
  <b>Documentation, source code: <a href="https://github.com/wkarwacki/python-openapi-generator-rust" target="_blank">https://github.com/wkarwacki/python-openapi-generator-rust</a></b>
</p>

---
# Trust Spec

Trust Spec is a web integration specification and a set of code generators aiming to be a substitute for OpenAPI. It provides a unified way to describe data transfer interfaces and generates modern, type-safe code.

## Table of Contents
- [I just want to glue my services together](#i-just-want-to-glue-my-services-together)
- [Requirements](#requirements)
- [Specification](#specification)
  - [Overview](#overview)
  - [Common Use Cases](#common-use-cases)
- [Code Generation](#code-generation)
  - [Generator Options](#generator-options)
  - [Supported Generators](#supported-generators)
    - [Experimental Generators](#experimental-generators)
  - [Generators Documentation](#generators-documentation)
- [Conversion from OpenAPI](#conversion-from-openapi)
- [Conversion to OpenAPI](#conversion-to-openapi)
- [Building Trust CLI](#building-trust-cli)
  - [With Docker](#with-docker)
  - [From Source](#from-source)

## I just want to glue my services together
Install Trust:
```shell
$ pip install trustspeccli
```
Then, run:
```shell
$ trust

Usage: trust <COMMAND>

Commands:
  from-open-api  Convert an OpenAPI specification to a Trust specification. Integrate this into your build process to utilize Trust's code generators
  to-open-api    Convert a Trust specification back to an OpenAPI specification, useful when a Trust code generator is not available for your target language
  generate       Generate code based on a Trust specification
  help           Print this message or the help of the given subcommand(s)

Options:
  -h, --help     Print help
  -V, --version  Print version
```

- If you already have an OpenAPI spec:
  1. first convert it to Trust spec with `from-open-api` command
  2. and then generate the glue code with `generate` command.

  > :exclamation: Note that you may easily automate the whole process combining the two steps above, however it is recommended to make a migration once and leverage Trust spec's expressiveness in long-term.

- **Without an OpenAPI spec:**
  - Start directly with Trust spec.

- **To continue using OpenAPI for code generation:**
  - Use Trust spec as an intermediate format with the `to-open-api` command.

## Requirements
- Python 3.14 - currently the only supported version

## Specification
Trust specification enhances current integration standards like [OpenAPI](https://github.com/OAI/OpenAPI-Specification). Key benefits include:
- **Clear Notation** - Single way to model and interpret an API.
- **Generic Types** - Customizable types with parameters for different contexts.
- **Minimalistic** - Simple, efficient language without redundant features.
- **Web System Integration Focused** - Ideal for type-safe system integration.
- **Highly Customizable** - Extensive [handlebars](https://github.com/sunng87/handlebars-rust) helpers for template modification.
- **Protocol-Agnostic** - Designed for HTTP but applicable to any API type.

It addresses inherent OpenAPI issues with:
- **Enclosed Algebraic Data Types** - All subtypes of an ADT are grouped in a single `adt` node.

### Overview
Examples and more usage details can be found in [tests](https://github.com/wkarwacki/python-openapi-generator-rust/tree/master/src/lib/test).

**Data Types:**
- **Simple Types:**
  - `type: bool` - equivalent to OpenAPI `type: boolean`
  - `type: int` - equivalent to OpenAPI `type: integer` with `format: int64`
  - `type: dec` - equivalent to OpenAPI `type: number`
  - `type: str` - equivalent to OpenAPI `type: string`
  - `type: enum` - equivalent to OpenAPI `type: string` with `enum`
- **Complex Types:**
  - `type: obj` - equivalent to OpenAPI `type: object`
  - `type: seq` - equivalent to OpenAPI `type: array`
  - `type: map` - equivalent to OpenAPI `type: object` with `additionalProperties`
- **Special Types:**
  - `type: alias` - equivalent to OpenAPI `$ref`
  - `type: struct` - equivalent to OpenAPI empty schema (`{}`)
  - `type: const` - equivalent to OpenAPI `const`

### Common Use Cases

**Including Variables from Other Types:**
  ```yaml
  Parent:
    type: obj
    vars:
      parentVar:
        type: dec
  AnotherParent:
    type: obj
    vars:
      anotherParentVar:
        type: bool
  WithParentsVars:
    type: obj
    mix:
      - path: "defs.Parent"
      - path: "defs.AnotherParent"
    vars:
      ownVar:
        type: int
  ```
Results in:
  ```json
  {
    "parentVar": 1.0,
    "anotherParentVar": true,
    "ownVar": 1
  }
  ```

**Algebraic Data Types (Union Types):**
  ```yaml
  AdtType:
    type: obj
    vars: 
      discriminatorVar:
        type: str
      someOtherVar:
        type: dec
    adt:
      var: discriminatorVar
      map:
        firstSubtype:
          vars:
            firstSubtypeVar:
              type: int
        secondSubtype:
          vars:
            secondSubtypeVar:
              type: bool
  ```
Interpreted as:
  ```json
  {
    "discriminatorVar": "firstSubtype",
    "someOtherVar": 1.0,
    "firstSubtypeVar": 1
  }
  ```
  ```json
  {
    "discriminatorVar": "secondSubtype",
    "someOtherVar": 1.0,
    "secondSubtypeVar": true
  }
  ```


**Generic Types:**
  ```yaml
  defs:
    SomeNamedString:
      type: str
    ParameterizedType:
      type: obj
      vars:
        varOfParamAbcType:
          param: ParamAbc # to declare a generic type, you need to simply use a 'param' keyword
        varOfParamXyzType:
          param: ParamXyz # similarly to the above, this time with a different name
        anoterVarOfParamXyzType:
          param: ParamXyz # similarly to the above, this time with a different name
    SubtypeOfParameterizedType:
      type: obj
      ext: # extending a generic type
        path: 'defs.ParameterizedType'
        args: # with below type-arguments
          ParamAbc:
            type: bool
          ParamXyz:
            path: 'defs.SomeType'
  ```
Equivalent to:
  ```java
  interface ParameterizedType<ParamAbc, ParamXyz> { 
    ParamAbc varOfParamAbcType; 
    ParamXyz varOfParamXyzType; 
    ParamXyz anoterVarOfParamXyzType; 
  }
  interface SubtypeOfParameterizedType extends ParameterizedType<Boolean, SomeType> { }
  ```

## Code Generation

```shell
$ trust generate -h
Generate code based on a Trust specification

Usage: trust generate [OPTIONS] <LANG> <ROLE> <INPUT> <OUTPUT>

Arguments:
  <LANG>    Select the target programming language for the generated code [possible values: kotlin, python, scala, type-script]
  <ROLE>    Specify whether to generate server or client code [possible values: client, server]
  <INPUT>   Path to the Trust specification file
  <OUTPUT>  Directory where the generated code will be saved

Options:
  -c <CONFIG>              Optional path to a generator configuration file. Refer to the Trust documentation for details
  -t <TEMPLATES_PATH>      Optional path to a custom templates directory. For instance, you can override any template found at https://github.com/wkarwacki/python-openapi-generator-rust/tree/master/src/lib/gen/python/server/templates, however this can be configured for all languages and roles
  -h, --help               Print help
```

### Generator Options
You can customize the generator behavior by passing a relevant `yml` configuration file. The following options are available:
* `typeMapping: dict[str, str]` - Map Trust Spec type to any provided type in a type-safe way. Generated code for both server and client is supposed to pick up on mapping provided by the user so that any errors in such will be caught at compile time.
* `module: str` - Specify the module name for the generated code
* `dtoName: str` - Provide the custom Handlebars template for naming DTO classes, by default it is `{{val}}Dto`
* `autoImplement: bool` This option is a fundamental part of the [Trust Spec integration tests suite](https://github.com/wkarwacki/python-openapi-generator-rust/blob/master/do_test.sh#L13).
  * For server generators - Provides default implementation for all the operations
  * For client generators - Generates tests with all required params that verify server's correctness 

### Supported Generators:

- <b>Python HTTP Server ([fastapi](https://github.com/tiangolo/fastapi))</b>
- <b>Python HTTP Client ([httpx](https://github.com/encode/httpx))</b>

#### Experimental Generators:
> :exclamation: Not fully implemented. Use at your own risk.

- Kotlin HTTP Server ([spring](https://github.com/spring-projects/spring-framework))
- Scala HTTP Server ([cask](https://github.com/com-lihaoyi/cask))

### Generators Documentation
For detailed documentation, features and limitations on the supported code generators, refer to:
- [Python Http Client Generator Documentation](https://github.com/wkarwacki/python-openapi-generator-rust/blob/master/src/lib/gen/python/client/Readme.md)
- [Python Http Server Generator Documentation](https://github.com/wkarwacki/python-openapi-generator-rust/blob/master/src/lib/gen/python/server/Readme.md)


## Conversion from OpenAPI
```shell
$ trust from-open-api -h
Convert an OpenAPI specification to a Trust specification. Integrate this into your build process to utilize Trust's code generators

Usage: trust from-open-api [OPTIONS] <INPUT> <OUTPUT>

Arguments:
  <INPUT>   Path to the OpenAPI specification file
  <OUTPUT>  Directory where the output Trust specification will be saved

Options:
  -l <LAYOUT>      Specify the structure of the converted Trust specification [default: default] [possible values: default, tag]
  -h, --help       Print help (see more with '--help')

```
* Control the layout of the generated Trust spec with the `-l` option. For instance, setting it to `tag` organizes the Trust spec by OpenAPI tags, as shown in this [example](https://github.com/wkarwacki/python-openapi-generator-rust/blob/master/test/integration/specs/openapi_fastapi/api.yml#L9).

## Conversion to OpenAPI

```shell
$ trust to-open-api -h
Convert a Trust specification back to an OpenAPI specification, useful when a Trust code generator is not available for your target language

Usage: trust to-open-api <INPUT>

Arguments:
  <INPUT>  Path to the Trust specification file

Options:
  -h, --help  Print help
```

## Building Trust Cli

### With Docker
Prerequisites:
- [Docker](https://docs.docker.com/engine/install/)
  ```shell
  $ ./docker/build.sh
  $ docker run trust
  ```

### From Source
Prerequisites:
- [Rust](https://www.rust-lang.org/tools/install)
  ```shell
  $ cargo run trust
  ```
 readmeEtag: '"79872888a2e1352524c01ec44eb39863121122d0"' readmeLastModified: Fri, 31 Oct 2025 08:36:48 GMT repositoryId: 834278095 description: >- Typesafe Python Client and Server OpenAPI generator written in Rust / Trust created: '2024-07-26T20:36:55Z' updated: '2026-01-14T16:58:58Z' language: Rust archived: false stars: 3 watchers: 1 forks: 0 owner: WojtekKarwacki logo: https://avatars.githubusercontent.com/u/24361894?v=4 license: Apache-2.0 repoEtag: '"a447b09ee0d6c8ce1ee41e8d4e8163411c4f4fb67f6ab55c215098bc7edc4a68"' repoLastModified: Wed, 14 Jan 2026 16:58:58 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/wkarwacki/python-openapi-generator-rust - source: openapi3 tags repository: https://github.com/chrimle/openapi-to-java-records-mustache-templates v3: true id: 3ea340810450fecfe6faffbbb7d5e31b repositoryMetadata: base64Readme: >- # openapi-to-java-records-mustache-templates
[![Java CI with Maven](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/actions/workflows/maven.yml/badge.svg)](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/actions/workflows/maven.yml)
[![Maven Package](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/actions/workflows/maven-publish.yml/badge.svg)](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/actions/workflows/maven-publish.yml)
[![pages-build-deployment](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/actions/workflows/pages/pages-build-deployment/badge.svg)](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/actions/workflows/pages/pages-build-deployment)

Project containing [Mustache-templates](https://mustache.github.io/) used by [openapi-generator-maven-plugin](https://github.com/OpenAPITools/openapi-generator/blob/master/modules/openapi-generator-maven-plugin/README.md) to generate [Java Records](https://docs.oracle.com/en/java/javase/17/language/records.html) from [OpenAPI Specifications](https://swagger.io/specification/).

This project contains the **mustache templates**.

## Apache License 2.0
Starting with project version **2.0.0**, the project is licensed under the **Apache License 2.0**.
> [!NOTE]
> Prior to version **2.0.0**, the project was licensed under the **MIT License**. See [License](#license).

## Support the Project
If you find this project useful, please ⭐ **Star** ⭐ it and share it with others!
This is the best way to show appreciation for this project - Thank you! ❤️

If you have feedback or suggestions, please share it in either [Discussions](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/discussions) or [Issues](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/issues)!

> [!NOTE]
> This project is, _and will continue to be_, solely maintained by [Chrimle](https://github.com/Chrimle). 
> While _direct_ code contributions are disallowed, your feedback is the most valuable contribution - please share it!

# Getting Started
The mustache templates are best acquired by importing the project as a dependency.

> [!TIP]
> If you want a more detailed guide with simple examples to get started, check out the Wiki-page 
> [Beginner Guide (Step-by-Step)](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/wiki/Beginner-Guide-%28Step‐by‐Step%29).

## Import Dependency
> [!NOTE]
> The Maven artifact **only** contains the `.mustache` template files and a `LICENSE.txt`. No other files are included in the imported artifact.

```xml
<dependency>
    <groupId>io.github.chrimle</groupId>
    <artifactId>openapi-to-java-records-mustache-templates</artifactId>
    <version>3.3.0</version>
</dependency>
```
It is **strongly recommended** to import the project as a dependency. It has officially been published to:
- [Maven Central Repository](https://central.sonatype.com/artifact/io.github.chrimle/openapi-to-java-records-mustache-templates)
- [GitHub Packages](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/packages/)

> [!WARNING]
> While it is _possible_ to download the Mustache templates directly
> from [GitHub](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/tree/main/mustache-templates/target/classes/templates),
> this approach is **not recommended**. Templates obtained this way are not guaranteed to be versioned correctly and
> is explicitly **exempt** from the [Semantic Versioning](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/wiki/Semantic-Versioning) considerations.

## Use the `.mustache` templates when generating
Place the file(s) in desired directory. Then, in the Maven build configuration, set the property `<templateDirectory>` to the directory path. Example:
```xml
<build>
  <plugins>
    <plugin>
      <groupId>org.openapitools</groupId>
      <artifactId>openapi-generator-maven-plugin</artifactId>
      <executions>
        <execution>
          <goals>
            <goal>generate</goal>
          </goals>
          <configuration>
            <inputSpec><!-- Relative directory path to the openapi.yaml file --></inputSpec>
            <templateDirectory><!-- Relative directory path to the mustache templates --></templateDirectory>
            <output><!-- Relative directory path to where generated classes should be placed --></output>
          </configuration>
        </execution>
      </executions>
    </plugin>
  </plugins>
</build>
```
## Additional Configurations
The generated classes are customizable by using `<configuration>`-properties.

In this example, each generated class field will be annotated with [Jakarta Bean Validation annotations](https://jakarta.ee/specifications/bean-validation/3.0/jakarta-bean-validation-spec-3.0.html#builtinconstraints).
```xml
  <configuration>
    <!-- ... more configurations ... -->
    <configOptions>
      <useBeanValidation>true</useBeanValidation>
      <!-- ... more configOptions ... -->
    </configOptions>
  </configuration>
```

> [!TIP]
> See [Supported 'openapi‐generator‐maven‐plugin' Configuration options](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/wiki/Supported-%27openapi‐generator‐maven‐plugin%27-Configuration-options)

## OpenAPI Specification
<details open>
<summary>Example OpenAPI Specification</summary>
    
```yaml
components:
  schemas:
    Name:
      description: Name Information
      type: object
      required:
        - firstName
        - lastName
      properties:
        firstName:
          description: First Name
          type: string
          minLength: 2
        lastName:
          description: Last Name
          type: string
          minLength: 2
        middleName:
          description: Middle Name
          type: string
          nullable: true
    Person:
      description: Personal information
      deprecated: true
      type: object
      required:
        - name
        - age
        - gender
        - height
        - ssn
        - aliases
        - email
        - trackingCode
        - uuid
      properties:
        name:
          description: Name
          type: object
          $ref: '#/components/schemas/Name'
        age:
          description: Age (years)
          type: integer
          minimum: 0
          maximum: 100
        gender:
          description: Gender
          type: string
          enum:
            - Male
            - Female
        height:
          description: Height (m)
          type: number
          pattern: float
          minimum: 0
        legalGuardian:
          description: Legal Guardian
          type: object
          $ref: '#/components/schemas/Person'
        ssn:
          description: Social Security Number
          type: string
          pattern: '^\d{3}-\d{2}-\d{4}$'
        aliases:
          description: Known Aliases
          type: array
          uniqueItems: true
          minItems: 1
          maxItems: 3
          items:
            type: string
        telephoneNumber:
          description: Telephone Number
          type: string
          nullable: true
        email:
          description: Email Address
          type: string
          format: email
        trackingCode:
          description: Tracking code for Web analytics
          type: string
          minLength: 5
          maxLength: 50
          default: "utm_source=default"
        uuid:
          description: An Universally Unique Identifier
          type: string
          format: uuid
```

</details>

> [!TIP]
> See [Supported OpenAPI Specification properties](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/wiki/Supported-OpenAPI-Specification-properties)

## Generate models
Compile the project, for example via:
```console
mvn compile
```

> [!TIP]
> Further information about how to generate models can be found on [openapi-generator-maven-plugin](https://github.com/OpenAPITools/openapi-generator/blob/master/modules/openapi-generator-maven-plugin/README.md).

## Generated Java Record
Unless the configuration property `<output>` has been set, the generated classes should be found under `./target/generated-sources/openapi`.

<details open>
    
<summary>Resulting Java record from example OpenAPI spec.</summary>

```java
package io.github.chrimle.o2jrm;

/**
 * Personal information
 *
 * @deprecated
 * @param name Name.
 * @param age Age (years)
 * @param gender Gender
 * @param height Height (m)
 * @param legalGuardian Person.
 * @param ssn Social Security Number
 * @param aliases Known Aliases
 * @param telephoneNumber Telephone Number
 * @param email Email Address
 * @param trackingCode Tracking code for Web analytics
 * @param uuid An Universally Unique Identifier
 */
@Deprecated
public record Person(
    @Valid @NotNull Name name,
    @NotNull @Min(0) @Max(100) Integer age,
    @NotNull GenderEnum gender,
    @NotNull @DecimalMin("0") BigDecimal height,
    @Valid Person legalGuardian,
    @NotNull @Pattern(regexp = "^\\d{3}-\\d{2}-\\d{4}$") String ssn,
    @NotNull @Size(min = 1, max = 3) Set<String> aliases,
    String telephoneNumber,
    @NotNull @Email String email,
    @NotNull @Size(min = 5, max = 50) String trackingCode,
    @NotNull UUID uuid) {

  public Person(
      final Name name,
      final Integer age,
      final GenderEnum gender,
      final BigDecimal height,
      final Person legalGuardian,
      final String ssn,
      final Set<String> aliases,
      final String telephoneNumber,
      final String email,
      final String trackingCode,
      final UUID uuid) {
    this.name = name;
    this.age = age;
    this.gender = gender;
    this.height = height;
    this.legalGuardian = legalGuardian;
    this.ssn = ssn;
    this.aliases = Objects.requireNonNullElseGet(aliases, () -> new LinkedHashSet<>());
    this.telephoneNumber = telephoneNumber;
    this.email = email;
    this.trackingCode = Objects.requireNonNullElse(trackingCode, "utm_source=default");
    this.uuid = uuid;
  }

  /**
   * Gender
   */
  public enum GenderEnum {
    MALE("Male"),
    FEMALE("Female");

    private final String value;

    GenderEnum(final String value) {
      this.value = value;
    }

    /**
     * Gets the {@code value} of this enum.
     *
     * @return the value of this enum.
     */
    public String getValue() {
      return value;
    }

    /**
     * Case-sensitively matches the given {@code value} to an enum constant using {@link
     * #getValue()}.
     *
     * <p><b>NOTE:</b> if multiple enum constants have a matching value, the first enum constant is
     * returned, by the order they are declared.
     *
     * @param value of the enum.
     * @return a {@link GenderEnum } with the matching value.
     * @throws IllegalArgumentException if no enum has a value matching the given value.
     */
    public static GenderEnum fromValue(final String value) {
      for (final GenderEnum constant : GenderEnum.values()) {
        if (constant.getValue().equals(value)) {
          return constant;
        }
      }
      throw new IllegalArgumentException("Unexpected value '" + value + "'");
    }
  }
}
```

</details>

## Further examples

Refer to the test-cases for generated classes, as these list supported plugin `<configuration>`-options and OpenAPI
Specification-properties. The unit-tests could clarify expected behaviors, and the OpenAPI Specification could also
provide concrete examples and use-cases. For reference:

- [OpenAPI Specification](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/blob/main/test-common/src/main/resources/api.yaml)<br/>
  Used as the input schema to generate **all** classes used in testing.
- [Maven plugin executions](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/blob/main/tests/pom.xml#L139)<br/>
  Generates classes from the OpenAPI spec, with different `openapi-generator-maven-plugin` configuration options. The
  resulting classes are placed in sub-packages, named after the plugin-execution.
- [Generated classes](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/tree/main/tests/target/generated-sources/openapi/src/src/gen/java/main/io/github/chrimle/o2jrm)<br/>
  The resulting classes are organized into sub-packages, which facilitates testing. These files are - _albeit in no way
  necessary_ - tracked and versioned in order to be accessible without needing to compile anything. Furthermore, it
  makes it easier to spot differences in generated classes after making a change to the mustache templates.
- [Test Suite](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/tree/main/tests/src/test/java/io/github/chrimle/o2jrm/tests)<br/>
  Contains all test-cases for generated `record` and `enum` classes. These tests are _parameterized_, to test **all**
  classes in the OpenAPI Specification in combination with **all** plugin-executions.

## Encountered an issue?
Firstly, make a minimal reproducible example - it will greatly facilitate troubleshooting!

Please, verify these steps _without_ custom mustache-template files:
1. Verify that the Maven Build Configuration is correct.
2. Verify that the OpenAPI Specification is valid.
3. Verify that classes are generated successfully.
4. Verify that needed dependencies are imported.

Once verified, use the custom mustache-template files and verify the following:
1. Verify that the `openapi-generator-maven-plugin` configuration options are supported. See [Supported 'openapi‐generator‐maven‐plugin' Configuration options](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/wiki/Supported-%27openapi‐generator‐maven‐plugin%27-Configuration-options).
   - If no configuration options are set, please proceed to the next step.
   - If the configuration option is not listed as supported - please request it via [open an issue](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/issues/new/choose).
2. Verify that the OpenAPI Specification properties are supported. See [Supported OpenAPI Specification properties](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/wiki/Supported-OpenAPI-Specification-properties).
   - If the property is not listed as supported - please request it via [open an issue](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/issues/new/choose).
3. Compare `openapi-generator-maven-plugin` versions
   - As a last resort, it could be due to using an older/newer version than what is used within this project for testing.
     Even if this would solve the issue - please report it via [open an issue](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/issues/new/choose).

If problems persist, check the [open issues](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/issues). 
If the problem you are facing has not already been reported, please [open an issue](https://github.com/Chrimle/openapi-to-java-records-mustache-templates/issues/new/choose) with details and instructions to reproduce.

### Useful Resources

- [Maven in 5 minutes](https://maven.apache.org/guides/getting-started/maven-in-five-minutes.html)
- [OpenAPI Basic Structure](https://swagger.io/docs/specification/basic-structure/)
- [openapi-generator-maven-plugin](https://github.com/OpenAPITools/openapi-generator/blob/master/modules/openapi-generator-maven-plugin/README.md)
- [Mustache](https://mustache.github.io/)

# License
                                 Apache License
                           Version 2.0, January 2004
                        http://www.apache.org/licenses/

   TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION

   1. Definitions.

      "License" shall mean the terms and conditions for use, reproduction,
      and distribution as defined by Sections 1 through 9 of this document.

      "Licensor" shall mean the copyright owner or entity authorized by
      the copyright owner that is granting the License.

      "Legal Entity" shall mean the union of the acting entity and all
      other entities that control, are controlled by, or are under common
      control with that entity. For the purposes of this definition,
      "control" means (i) the power, direct or indirect, to cause the
      direction or management of such entity, whether by contract or
      otherwise, or (ii) ownership of fifty percent (50%) or more of the
      outstanding shares, or (iii) beneficial ownership of such entity.

      "You" (or "Your") shall mean an individual or Legal Entity
      exercising permissions granted by this License.

      "Source" form shall mean the preferred form for making modifications,
      including but not limited to software source code, documentation
      source, and configuration files.

      "Object" form shall mean any form resulting from mechanical
      transformation or translation of a Source form, including but
      not limited to compiled object code, generated documentation,
      and conversions to other media types.

      "Work" shall mean the work of authorship, whether in Source or
      Object form, made available under the License, as indicated by a
      copyright notice that is included in or attached to the work
      (an example is provided in the Appendix below).

      "Derivative Works" shall mean any work, whether in Source or Object
      form, that is based on (or derived from) the Work and for which the
      editorial revisions, annotations, elaborations, or other modifications
      represent, as a whole, an original work of authorship. For the purposes
      of this License, Derivative Works shall not include works that remain
      separable from, or merely link (or bind by name) to the interfaces of,
      the Work and Derivative Works thereof.

      "Contribution" shall mean any work of authorship, including
      the original version of the Work and any modifications or additions
      to that Work or Derivative Works thereof, that is intentionally
      submitted to Licensor for inclusion in the Work by the copyright owner
      or by an individual or Legal Entity authorized to submit on behalf of
      the copyright owner. For the purposes of this definition, "submitted"
      means any form of electronic, verbal, or written communication sent
      to the Licensor or its representatives, including but not limited to
      communication on electronic mailing lists, source code control systems,
      and issue tracking systems that are managed by, or on behalf of, the
      Licensor for the purpose of discussing and improving the Work, but
      excluding communication that is conspicuously marked or otherwise
      designated in writing by the copyright owner as "Not a Contribution."

      "Contributor" shall mean Licensor and any individual or Legal Entity
      on behalf of whom a Contribution has been received by Licensor and
      subsequently incorporated within the Work.

   2. Grant of Copyright License. Subject to the terms and conditions of
      this License, each Contributor hereby grants to You a perpetual,
      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
      copyright license to reproduce, prepare Derivative Works of,
      publicly display, publicly perform, sublicense, and distribute the
      Work and such Derivative Works in Source or Object form.

   3. Grant of Patent License. Subject to the terms and conditions of
      this License, each Contributor hereby grants to You a perpetual,
      worldwide, non-exclusive, no-charge, royalty-free, irrevocable
      (except as stated in this section) patent license to make, have made,
      use, offer to sell, sell, import, and otherwise transfer the Work,
      where such license applies only to those patent claims licensable
      by such Contributor that are necessarily infringed by their
      Contribution(s) alone or by combination of their Contribution(s)
      with the Work to which such Contribution(s) was submitted. If You
      institute patent litigation against any entity (including a
      cross-claim or counterclaim in a lawsuit) alleging that the Work
      or a Contribution incorporated within the Work constitutes direct
      or contributory patent infringement, then any patent licenses
      granted to You under this License for that Work shall terminate
      as of the date such litigation is filed.

   4. Redistribution. You may reproduce and distribute copies of the
      Work or Derivative Works thereof in any medium, with or without
      modifications, and in Source or Object form, provided that You
      meet the following conditions:

      (a) You must give any other recipients of the Work or
          Derivative Works a copy of this License; and

      (b) You must cause any modified files to carry prominent notices
          stating that You changed the files; and

      (c) You must retain, in the Source form of any Derivative Works
          that You distribute, all copyright, patent, trademark, and
          attribution notices from the Source form of the Work,
          excluding those notices that do not pertain to any part of
          the Derivative Works; and

      (d) If the Work includes a "NOTICE" text file as part of its
          distribution, then any Derivative Works that You distribute must
          include a readable copy of the attribution notices contained
          within such NOTICE file, excluding those notices that do not
          pertain to any part of the Derivative Works, in at least one
          of the following places: within a NOTICE text file distributed
          as part of the Derivative Works; within the Source form or
          documentation, if provided along with the Derivative Works; or,
          within a display generated by the Derivative Works, if and
          wherever such third-party notices normally appear. The contents
          of the NOTICE file are for informational purposes only and
          do not modify the License. You may add Your own attribution
          notices within Derivative Works that You distribute, alongside
          or as an addendum to the NOTICE text from the Work, provided
          that such additional attribution notices cannot be construed
          as modifying the License.

      You may add Your own copyright statement to Your modifications and
      may provide additional or different license terms and conditions
      for use, reproduction, or distribution of Your modifications, or
      for any such Derivative Works as a whole, provided Your use,
      reproduction, and distribution of the Work otherwise complies with
      the conditions stated in this License.

   5. Submission of Contributions. Unless You explicitly state otherwise,
      any Contribution intentionally submitted for inclusion in the Work
      by You to the Licensor shall be under the terms and conditions of
      this License, without any additional terms or conditions.
      Notwithstanding the above, nothing herein shall supersede or modify
      the terms of any separate license agreement you may have executed
      with Licensor regarding such Contributions.

   6. Trademarks. This License does not grant permission to use the trade
      names, trademarks, service marks, or product names of the Licensor,
      except as required for reasonable and customary use in describing the
      origin of the Work and reproducing the content of the NOTICE file.

   7. Disclaimer of Warranty. Unless required by applicable law or
      agreed to in writing, Licensor provides the Work (and each
      Contributor provides its Contributions) on an "AS IS" BASIS,
      WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
      implied, including, without limitation, any warranties or conditions
      of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
      PARTICULAR PURPOSE. You are solely responsible for determining the
      appropriateness of using or redistributing the Work and assume any
      risks associated with Your exercise of permissions under this License.

   8. Limitation of Liability. In no event and under no legal theory,
      whether in tort (including negligence), contract, or otherwise,
      unless required by applicable law (such as deliberate and grossly
      negligent acts) or agreed to in writing, shall any Contributor be
      liable to You for damages, including any direct, indirect, special,
      incidental, or consequential damages of any character arising as a
      result of this License or out of the use or inability to use the
      Work (including but not limited to damages for loss of goodwill,
      work stoppage, computer failure or malfunction, or any and all
      other commercial damages or losses), even if such Contributor
      has been advised of the possibility of such damages.

   9. Accepting Warranty or Additional Liability. While redistributing
      the Work or Derivative Works thereof, You may choose to offer,
      and charge a fee for, acceptance of support, warranty, indemnity,
      or other liability obligations and/or rights consistent with this
      License. However, in accepting such obligations, You may act only
      on Your own behalf and on Your sole responsibility, not on behalf
      of any other Contributor, and only if You agree to indemnify,
      defend, and hold each Contributor harmless for any liability
      incurred by, or claims asserted against, such Contributor by reason
      of your accepting any such warranty or additional liability.

   END OF TERMS AND CONDITIONS

   APPENDIX: How to apply the Apache License to your work.

      To apply the Apache License to your work, attach the following
      boilerplate notice, with the fields enclosed by brackets "[]"
      replaced with your own identifying information. (Don't include
      the brackets!)  The text should be enclosed in the appropriate
      comment syntax for the file format. We also recommend that a
      file or class name and description of purpose be included on the
      same "printed page" as the copyright notice for easier
      identification within third-party archives.

   Copyright 2024-2026 Chrimle

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

       http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.

 readmeEtag: '"64910a6a9ba6ddd320e8355bf99a43618fa42f73"' readmeLastModified: Tue, 20 Jan 2026 20:10:10 GMT repositoryId: 853046067 description: Generate Java Records from OpenAPI Specifications created: '2024-09-05T22:21:02Z' updated: '2026-01-20T20:10:18Z' language: Java archived: false stars: 10 watchers: 1 forks: 0 owner: Chrimle logo: https://avatars.githubusercontent.com/u/28791817?v=4 license: Apache-2.0 repoEtag: '"67b6bb3fd8f197da5609aa4f4bb71a922e6423308451eed6f4ad204ba4b85a64"' repoLastModified: Tue, 20 Jan 2026 20:10:18 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/theogravity/immutable-x-openapi v3: true repositoryMetadata: base64Readme: >- IyBpbW11dGFibGUteC1vcGVuYXBpCgpUaGlzIGNvbnRhaW5zIGFuICp1bm9mZmljaWFsKiBPcGVuQVBJIDMuMCBkZWZpbml0aW9ucyBmb3IgCgotIHRoZSBbSW1tdXRhYmxlIFggQVBJc10oaHR0cHM6Ly9kb2NzLnguaW1tdXRhYmxlLmNvbS9yZWZlcmVuY2UpCi0gdGhlIFtHb2RzIFVuY2hhaW5lZCBBUElzXShodHRwczovL2dpdGh1Yi5jb20vaW1tdXRhYmxlL2dvZHMtdW5jaGFpbmVkLWFwaSkKClRoZSBkZWZpbml0aW9ucyB3ZXJlIGNyZWF0ZWQgdXNpbmcgW1N0b3BsaWdodCBTdHVkaW9dKGh0dHBzOi8vc3RvcGxpZ2h0LmlvL3N0dWRpby8pLgoKWW91IGNhbiB1c2UgdGhlIGRlZmluaXRpb24gdG8gW2dlbmVyYXRlIHlvdXIgb3duIFNES10oaHR0cHM6Ly9vcGVuYXBpLnRvb2xzLyksIG9yIGltcG9ydCBpbnRvIHlvdXIgZmF2b3JpdGUKQVBJIGNsaWVudCBzdWNoIGFzIFtJbnNvbW5pYV0oaHR0cHM6Ly9pbnNvbW5pYS5yZXN0LykgYW5kIFtQb3N0bWFuXShodHRwczovL3d3dy5wb3N0bWFuLmNvbS8pLgoKIVtdKGFzc2V0cy9pbnNvbW5pYS5wbmcpCgojIyBVc2luZyBhIHJlYWR5LW1hZGUgU0RLCgpZb3UgY2FuIGZpbmQgZG9jdW1lbnRhdGlvbiBoZXJlIHRoYXQgaGFzIGJlZW4gZ2VuZXJhdGVkIHVzaW5nIHRoZSBkZWZpbml0aW9ucy4gU2VsZWN0IHRoZSAKbGFuZ3VhZ2Ugb3B0aW9uIG9uIHRoZSByaWdodCBzaWRlIGZvciB5b3VyIGNsaWVudC4gVGhlcmUgaXMgYSB2YW5pbGxhIEphdmFzY3JpcHQgb3B0aW9uIGlmIHlvdQpjbGljayBvbiB0aGUgdmVydGljYWwgdGhyZWUgZG90cyBuZXh0IHRvIHRoZSBsYW5ndWFnZXMsIHdoaWNoIHdpbGwgc2hvdyB5b3UKaG93IHRvIGNhbGwgdGhlIEFQSSBpbiB2YW5pbGxhIEpTLgoKaHR0cHM6Ly9pbXgucmVhZG1lLmlvL3JlZmVyZW5jZS9saXN0Y2FyZHMKCiMjIEJ1aWxkaW5nIGFuIFNESwoKIyMjIFR5cGVzY3JpcHQgKHZpYSBub2RlLmpzKQoKVXNlIHRoZSBbb3BlbmFwaS10eXBlc2NyaXB0LWNvZGVnZW5dKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlZ2VuKSBwYWNrYWdlIHRvIApidWlsZCBhbiBTREsgY29tcGxldGUgd2l0aCBUeXBlc2NyaXB0IGRlZmluaXRpb25zLgoKSWYgeW91IGNvcHkgdGhlIGFwcHJvcHJpYXRlIGByZWZlcmVuY2UvKi55YW1sYCBmaWxlIHRvIHlvdXIgcHJvamVjdCBhbmQgaW5zdGFsbCBgb3BlbmFwaS10eXBlc2NyaXB0LWNvZGVnZW5gLAp5b3UgY2FuIGdlbmVyYXRlIHRoZSBTREsgdXNpbmcgYSBgc2NyaXB0YCBjb21tYW5kIGluIHlvdXIgYHBhY2thZ2UuanNvbmA6CgpgYGBqc29uCnsKICAic2NyaXB0cyI6IHsKICAgICJndS1jbGllbnQ6YnVpbGQiOiAib3BlbmFwaSAtLWlucHV0IEdvZHNVbmNoYWluZWQtQVBJLU9wZW5BUEkzLnlhbWwgLS11c2VPcHRpb25zIC0tb3V0cHV0IC4vc3JjL2d1LWFwaS1jbGllbnQiCiAgfQp9CmBgYAoKIyMgTm90ZXMKClRoZXNlIGRlZmluaXRpb25zIHdlcmUgY3JhZnRlZCBieSBoYW5kIHVzaW5nIHRoZSBhdmFpbGFibGUgZG9jdW1lbnRhdGlvbi4KCkFsdGhvdWdoIGEgYmVzdCBlZmZvcnQgd2FzIG1hZGUgdG8gZW5zdXJlIGFjY3VyYWN5IGluIHJlc3VsdHMsIHRoZXJlIG1heSBiZSBtaXN0YWtlcyBpbiB0aGUgZGVmaW5pdGlvbi4gSWYgeW91IGZpbmQKc3VjaCBhIG1pc3Rha2UsIHBsZWFzZSBvcGVuIGEgcHVsbCByZXF1ZXN0IHdpdGggdGhlIG5lY2Vzc2FyeSBjaGFuZ2VzLgoKRm9yIG1hbnkgZW5kcG9pbnRzLCBtb2RlbCBkZWZpbml0aW9ucyBoYXZlIGJlZW4gY3JlYXRlZCBmb3IgcmVzcG9uc2VzLCB3aGljaCBhIGNvZGUgZ2VuZXJhdG9yIGNhbiB1c2UgdG8gY3JlYXRlCmNvcnJlc3BvbmRpbmcgY2xhc3Nlcywgb3IgdHlwZWQgaW50ZXJmYWNlcyBpZiB1c2luZyBhIFR5cGVzY3JpcHQgZ2VuZXJhdG9yLgoKIyMgVW5kb2N1bWVudGVkIEFQSXMKCiMjIyBJbW11dGFibGUgWAoKVGhlIGZvbGxvd2luZyBlbmRwb2ludHMgd2VyZSB1bmFibGUgdG8gYmUgZG9jdW1lbnRlZCBhdCB0aGUgdGltZSBkdWUgdG8gdW5hdmFpbGFiaWxpdHkgb2YgdGhlIGRvY3VtZW50YXRpb246CgotIGh0dHBzOi8vZG9jcy54LmltbXV0YWJsZS5jb20vcmVmZXJlbmNlL3Bvc3RfdjEtc2lnbmFibGUtZGVwb3NpdC1kZXRhaWxzLTEKLSBodHRwczovL2RvY3MueC5pbW11dGFibGUuY29tL3JlZmVyZW5jZS9wb3N0X3YxLXNpZ25hYmxlLW9yZGVyLWRldGFpbHMtMQotIGh0dHBzOi8vZG9jcy54LmltbXV0YWJsZS5jb20vcmVmZXJlbmNlL3Bvc3RfdjEtc2lnbmFibGUtdHJhbnNmZXItZGV0YWlscy0xCi0gaHR0cHM6Ly9kb2NzLnguaW1tdXRhYmxlLmNvbS9yZWZlcmVuY2UvcG9zdF92MS1zaWduYWJsZS13aXRoZHJhd2FsLWRldGFpbHMtMQoKIyMgQ29udHJpYnV0aW5nCgotIEZvcmsgdGhpcyByZXBvCi0gVXNlIFN0b3BsaWdodCBTdHVkaW8gdG8gb3BlbiB0aGUgcmVwbyBkaXJlY3RvcnksIGFuZCBtYWtlIHlvdXIgZWRpdHMKLSBNYWtlIHN1cmUgdGhlIGxpbnRlciBpbiBTdG9wbGlnaHQgU3R1ZGlvIGhhcyBubyBlcnJvcnMvd2FybmluZ3MKLSBNYWtlIGEgcHVsbCByZXF1ZXN0IHdpdGggeW91ciBjaGFuZ2VzCgojIyBEaXNjbGFpbWVyCgpUaGUgQVBJIGRlZmluaXRpb25zIGRlcGVuZCBvbiB0aGUgY29tbXVuaXR5IHRvIGtlZXAgdXAtdG8tZGF0ZSwgb3IgdW50aWwgdGhlIEltbXV0YWJsZSB0ZWFtCnByb3ZpZGVzIGFuIG9mZmljaWFsIGRlZmluaXRpb24uIFVzZSBhdCB5b3VyIG93biByaXNrLCBub3QgcmVzcG9uc2libGUgZm9yIGxvc3NlcyBpbmN1cnJlZCBmcm9tIGFuIGluY29ycmVjdApBUEkgZGVmaW5pdGlvbi4K readmeEtag: '"dffaecc96e8d51b6bc3ce4d59b72e19ac3342a6f"' readmeLastModified: Fri, 25 Feb 2022 05:11:50 GMT repositoryId: 443897863 description: Immutable X and Gods Unchained OpenAPI 3.0 definitions created: '2022-01-03T00:07:45Z' updated: '2023-04-13T17:11:33Z' language: null archived: false stars: 3 watchers: 1 forks: 1 owner: theogravity logo: https://avatars.githubusercontent.com/u/855434?v=4 license: MIT repoEtag: '"645cad8dcbf6627f2646c5015795c2f41cbb05b9b7a042c0214285f9edbad5a1"' repoLastModified: Thu, 13 Apr 2023 17:11:33 GMT foundInMaster: true category: Parsers id: df13bb01ef12259dd9da2e6209165294 - source: openapi3 tags repository: https://github.com/0x29a/web_template v3: true id: 7b4bda04c7ead8d5a5dc42a240ad7dc9 repositoryMetadata: base64Readme: >- IyBXZWJUZW1wbGF0ZQoKWyFbR2l0TGFiIENJIHBpcGVsaW5lIHN0YXR1cy5dW3BpcGVsaW5lLWltYWdlXV1bcGlwZWxpbmUtdXJsXQoKVGVtcGxhdGUgZm9yIGEgcXVpY2sgd2ViIHNlcnZpY2VzIHByb3RvdHlwaW5nLiBJdCdzIGludGVuZGVkIHRvIHNwZWVkIHVwIGJvb3RzdHJhcHBpbmcKb2YgcHJvZHVjdGlvbi1yZWFkeSBhcHBsaWNhdGlvbnMsIHdpdGggRGphbmdvLWJhc2VkIGJhY2tlbmQsIGFuZCBOZXh0LmpzLWJhc2VkIGZyb250ZW5kLgpJbiBhdHRlbXB0IHRvIHJlZHVjZSBib2lsZXJwbGF0ZSB0byB0aGUgYWJzb2x1dGUgcG9zc2libGUgbWluaW11bSwgaXQgdXRpbGl6ZXMgdGVjaG5vbG9naWVzIHN1Y2ggYXM6CgotIFtkcmYtc3BlY3RhY3VsYXJdKGh0dHBzOi8vZ2l0aHViLmNvbS90ZnJhbnplbC9kcmYtc3BlY3RhY3VsYXIpIHRvIGdlbmVyYXRlIE9wZW5BUEkgMyBzY2hlbWEuCi0gW3JlZHV4LXRvb2xraXRdKGh0dHBzOi8vZ2l0aHViLmNvbS9yZWR1eGpzL3JlZHV4LXRvb2xraXQpIGFuZCBbQHJ0ay1xdWVyeS9jb2RlZ2VuLW9wZW5hcGldKGh0dHBzOi8vcmVkdXgtdG9vbGtpdC5qcy5vcmcvcnRrLXF1ZXJ5L3VzYWdlL2NvZGUtZ2VuZXJhdGlvbiNvcGVuYXBpKSB0byBnZW5lcmF0ZSBBUEkgY2xpZW50IGJhc2VkIG9uIFJUSyBRdWVyeS4KLSBbcGlwLXRvb2xzXShodHRwczovL2dpdGh1Yi5jb20vamF6emJhbmQvcGlwLXRvb2xzKSB0byBtYW5hZ2UgZGVwZW5kZW5jaWVzLgotIFtOZXh0LmpzXShodHRwczovL25leHRqcy5vcmcvKSBmb3IgaXRzIGZpbGUtYmFzZWQgcm91dGluZywgbmljZSBzdGF0aWMgcGFnZXMgaGFuZGxpbmcgYW5kIGRvemVucyBvZiBvdGhlciBvcHRpbWl6YXRpb25zLgotIFtUYWlsd2luZCBVSV0oaHR0cHM6Ly90YWlsd2luZHVpLmNvbSksIGFzIHV0aWxpdHkgY2xhc3NlcyBwbGF5IHJlYWxseSBuaWNlIHdpdGggcmV1c2FibGUgUmVhY3QgY29tcG9uZW50cy4KLSBbR2l0TGFiIENJXShodHRwczovL2RvY3MuZ2l0bGFiLmNvbS9lZS9jaS8pLCB3aGljaCwgaW4gYWRkaXRpb24gdG8gcXVhbGl0eSBjaGVja2luZywgaXMgdXNlZCB0byBidWlsZCBhbmQgaG9zdCBPQ0kgaW1hZ2VzLgotIC4uLmFuZCBtYW55IG90aGVyLgoKQnkgZGVmYXVsdCwgdGhlIHRlbXBsYXRlIGlzIGludGVuZGVkIGJlIGRlcGxveWVkIHRvIGEgRGlnaXRhbE9jZWFuIGRyb3BsZXQsIGJ1dCBzaW5jZSB0aGUgaW1hZ2VzIGFyZSBwcmUtYnVpbHQgY29udGludW91c2x5LCBpdCdzIHRyaXZpYWwgdG8gc3dpdGNoIHRvIGZseS5pbywgQVdTIEVDUyBvciBLdWJlcm5ldGVzLgoKSW4gYWRkaXRpb24gdG8gYWxsIHRoZSBjb29sIHRvb2xpbmcsIEkgdHJ5IHRvIGRvY3VtZW50IGFsbCB0ZWNobm9sb2dpY2FsIC8gYXJjaGl0ZWN0dXJhbCBkZWNpc2lvbnMgaW4gYGRvY3MvYCwgc28gaXQncyBlYXNpZXIKdG8gY29udGludWUgd29ya2luZyBvbiB0aGlzIHByb2plY3QgYWZ0ZXIgbGFyZ2UgYnJlYWtzLgoKIyMgR2V0dGluZyBzdGFydGVkCgoxLiBDcmVhdGUgYC5lbnZgOgoKICAgIGBgYGJhc2gKICAgIGNwIC5lbnYubG9jYWwgLmVudgogICAgYGBgCgoxLiBDcmVhdGUgYGRvY2tlci1jb21wb3NlLm92ZXJyaWRlLnltbGA6CgogICAgYGBgYmFzaAogICAgY3AgZG9ja2VyLWNvbXBvc2Uub3ZlcnJpZGUueW1sLmxvY2FsIGRvY2tlci1jb21wb3NlLm92ZXJyaWRlLnltbAogICAgYGBgCgoxLiBTdGFydCBjb250YWluZXJzOgoKICAgIGBgYGJhc2gKICAgIG1ha2UgdXAKICAgIGBgYAoKMS4gT3BlbiBgaHR0cDovL2xvY2FsaG9zdDo4MTI0L2AgaW4gYSBicm93c2VyLgoKPCEtLSBCYWRnZXMgLS0+CgpbcGlwZWxpbmUtaW1hZ2VdOiBodHRwczovL2dpdGxhYi5jb20vMHgyOWEvd2ViX3RlbXBsYXRlL2JhZGdlcy9tYXN0ZXIvcGlwZWxpbmUuc3ZnCltwaXBlbGluZS11cmxdOiBodHRwczovL2dpdGxhYi5jb20vMHgyOWEvd2ViX3RlbXBsYXRlLy0vcGlwZWxpbmVzCg== readmeEtag: '"2f7c1f91578222404a2a546b27d88f7e4b037769"' readmeLastModified: Tue, 13 Feb 2024 07:30:07 GMT repositoryId: 356602391 description: Next.js / Django template for quick prototyping. created: '2021-04-10T14:15:56Z' updated: '2024-03-07T06:27:17Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 0 owner: '0x29a' logo: https://avatars.githubusercontent.com/u/18251194?v=4 license: MIT repoEtag: '"c9fa81813ecfc9e5ce671b96738fe6bbc4a8d43e829dbb1609f09b8a574e535a"' repoLastModified: Thu, 07 Mar 2024 06:27:17 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cnsukidayo/wword v3: true id: 125190234ba25676da0af2c14d1e5e22 repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiID4KICAgIDxiPldXb3JkPC9iPgo8L2gxPgo8cCBhbGlnbj0iY2VudGVyIj4KICAgIDxhIGhyZWY9Imh0dHBzOi8vb3Blbmpkay5vcmcvcHJvamVjdHMvamRrLzE3LyI+PGltZyBhbHQ9IkdpdEh1YiBsYXN0IGNvbW1pdCIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0pESy0xNy1yZWQuc3ZnP3N0eWxlPWZsYXQmbG9nbz1PcmFjbGUmbGFiZWxDb2xvcj0yQjlDNEMmY29sb3I9REMyNTAwIiAvPjwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9DTlN1a2lkYXlvL1dXb3JkL2NvbW1pdHMiPjxpbWcgYWx0PSJHaXRIdWIgbGFzdCBjb21taXQiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGFzdC1jb21taXQvQ05TdWtpZGF5by9XV29yZCIgLz48L2E+CiAgICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vQ05TdWtpZGF5by9XV29yZC9ibG9iL21hc3Rlci9MSUNFTlNFIj48aW1nIGFsdD0iR2l0SHViIGxhc3QgY29tbWl0IiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGljZW5zZS1HUEwgMy4wLTRFQjFCQSIgLz48L2E+CjwvcD4KCiMjIyDku4vnu40KCuS4h+ivreivjSzmraPlpoLlhbblkI3lroPoh7Tlipvkuo7mj5DkvpvkuIDlpZfnu5/kuIDnmoTpgLvovpHlrozmiJDku7vmhI/or63oqIDor43msYfnmoTlrabkuaAuCuW8leWFpeeLrOWIm+eahOWNleivjeagh+iusOaKgOacr+aWueS+v+eUqOaIt+abtOWlveWcsOWvuemBl+W/mOefpeivhui/m+ihjOafpea8j+ihpee8uiwK55So5oi35Y+v5Lul5Zyo6K+l5bmz5Y+w5Y+R5biDTWFya2Rvd27moLzlvI/nmoTluJblrZDliIbkuqvor63oqIDlrabkuaDkuK3nmoTlkITnp43lv4PlvpcuCumhueebruWfuuS6jlNwcmluZ0Jvb3QrTXliYXRpc+WunueOsCzph4fnlKhEb2NrZXLlrrnlmajljJbpg6jnvbIuCuWMheWQq+euoeeQhuWRmOaooeWdl+OAgeadg+mZkOaooeWdl+OAgeaguOW/g+WKn+iDveaooeWdl+OAgeaQnOe0ouaooeWdl+etiS4KCiMjIyDova/ku7bmnrbmnoQKCiFb6L2v5Lu25p625p6EXShpbWFnZXMvZGVzaWduLnBuZykKCiMjIyDmqKHlnZfliJLliIYKCiogKirms6jlhozkuK3lv4MqKjpuYWNvcwoqICoq572R5YWzZ2F0ZXdheSoqOkdhdGV3YXkg6IGa5ZCI5omA5pyJ55qE5o6l5Y+jLOe7n+S4gOaOpeWPl+WkhOeQhuWJjeerr+eahOivt+axgjvlubbkuJTlnKjovazlj5HliY3lvILmraXosIPnlKjmnYPpmZDmqKHlnZfov5vooYzpibTmnYMuCiogKirlhazlhbHmqKHlnZcqKjrlsIbmiYDmnInmnI3liqHmqKHlnZfpnIDopoHnmoTlip/og73ljZXni6zmir3nprso5YyF5ous5YWo5bGA5byC5bi45ZON5bqU5ZKM5byC5bi45aSE55CGLOi/h+a7pOWZqCkuCiogKirlrp7kvZPnsbvmqKHlnZcqKjrpgJrov4fmi4bliIbkuLrliY3nq6/jgIHlhazlhbHjgIHlkI7nq6/kuInkuKrpg6jliIbnmoTlrp7kvZPnsbvmqKHlnZcs5L2/5b6X57O757uf5ZCE5bGC5LmL6Ze055u45LqS54us56uL6Kej6ICmLgoqICoq6Ym05p2D5qih5Z2XKio655So5oi35rOo5YaM44CB6I635Y+W55So5oi35L+h5oGv44CB5Yik5pat55So5oi35piv5ZCm5pyJ55uu5qCH5o6l5Y+j5p2D6ZmQ44CB6KeS6Imy566h55CG44CB6KeS6Imy5YiG6YWN5o6l5Y+j5p2D6ZmQ562JLgoqICoq5qC45b+D5qih5Z2XKio65YyF5ous55So5oi35pS26JeP5aS55Yqf6IO9566h55CG44CBbWFya2Rvd27luJblrZDnrqHnkIYKKiAqKueuoeeQhuWRmOaooeWdlyoqOuWNleivjeWIkuWIhueuoeeQhuOAgeWvvOWFpeWNleivjeeuoeeQhgoqICoq5pCc57Si5qih5Z2XKio6RWxhc3RpY1NlYXJjaOWNleivjeaQnOe0ouOAgeabtOaWsEVT5YiX6KGoCgojIyMg55WM6Z2i5bGV56S6Clvku5PlupPlnLDlnYA6aHR0cHM6Ly9naXRodWIuY29tL0NOU3VraWRheW8vQW55TGFuZ3VhZ2VXb3JkXShodHRwczovL2dpdGh1Yi5jb20vQ05TdWtpZGF5by9BbnlMYW5ndWFnZVdvcmQpCjx0YWJsZSBib3JkZXI9IjBweCI+CiAgICA8dHI+CiAgICAgICAgPGltZyBzcmM9ImltYWdlcy9ob21lX3BhZ2VfMS5wbmciIGFsdD0i55m76ZmG55WM6Z2iIiAvPgogICAgPC90cj4gICAgICAgCiAgICA8dHI+CiAgICAgICAgPGltZyBzcmM9ImltYWdlcy9ob21lX3BhZ2VfMi5wbmciIGFsdD0i6IOM6K+N55WM6Z2iIiAvPgogICAgPC90cj4gICAgCiAgICA8dHI+CiAgICAgICAgPGltZyBzcmM9ImltYWdlcy9ob21lX3BhZ2VfMy5wbmciIGFsdD0i5pS26JeP5aS55ZKMbWFya2Rvd27mmL7npLoiIC8+CiAgICA8L3RyPiAgICAKPC90YWJsZT4= readmeEtag: '"ebdab30a16a3adbb63cb7b86fb3cdf522b82ffa7"' readmeLastModified: Sun, 25 Feb 2024 05:33:42 GMT repositoryId: 669420168 description: >- 万语词,正如其名它致力于提供一套统一的逻辑完成任意语言词汇的学习.引入独创的单词标记技术方便用户更 好地对遗忘知识进行查漏补缺,用户可以在该平台发布Markdown格式的帖子分享语言学习中的各种心得.项目基于SpringBoot+Mybatis实现,采用Docker容器化部署.包含管理员模块、权限模块、核心功能模块、搜索模块等. created: '2023-07-22T08:08:10Z' updated: '2024-08-23T12:18:09Z' language: Java archived: false stars: 3 watchers: 1 forks: 0 owner: CNSukidayo logo: https://avatars.githubusercontent.com/u/53049252?v=4 license: GPL-3.0 repoEtag: '"0642d71b379762d5c3d76d94c6fc0541912924c9b5c5b8fd2ce1d73666511d7d"' repoLastModified: Fri, 23 Aug 2024 12:18:09 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/raghav2211/spring-web-flux-todo-app v3: true id: bc4cf2d2272ad2f80158393eb9d3da0f repositoryMetadata: base64Readme: >- IyBUT0RPIEFQUApbIVtHcmFkbGUgUGlwZWxpbmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9SYWdoYXYyMjExL3NwcmluZy13ZWItZmx1eC10b2RvLWFwcC9hY3Rpb25zL3dvcmtmbG93cy9ncmFkbGUtcGlwZWxpbmUueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9SYWdoYXYyMjExL3NwcmluZy13ZWItZmx1eC10b2RvLWFwcC9hY3Rpb25zL3dvcmtmbG93cy9ncmFkbGUtcGlwZWxpbmUueW1sKQpbIVtNYXZlbiBQaXBlbGluZV0oaHR0cHM6Ly9naXRodWIuY29tL1JhZ2hhdjIyMTEvc3ByaW5nLXdlYi1mbHV4LXRvZG8tYXBwL2FjdGlvbnMvd29ya2Zsb3dzL21hdmVuLXBpcGVsaW5lLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vUmFnaGF2MjIxMS9zcHJpbmctd2ViLWZsdXgtdG9kby1hcHAvYWN0aW9ucy93b3JrZmxvd3MvbWF2ZW4tcGlwZWxpbmUueW1sKQoKVGhpcyBhcHBsaWNhdGlvbiBpcyBhbiBzY2FmZm9sZCB3aGljaCBpcyBhIGJhc2ljIHNldHVwIG9mIHNwcmluZy1jbG91ZC1nYXRld2F5LCBjb25maWctc2VydmVyIGFuZCBhIHJlc291cmNlIHNlcnZlcihBIHRvZG8gYXBwbGljYXRpb24pLiBJdCBkZW1vbnN0cmF0ZSBob3cgZGF0YSBmbG93cyBiZXR3ZWVuIGdhdGV3YXkgJiByZXNvdXJjZS1zZXJ2ZXIgdXNpbmcgY29uZmlnLXNlcnZlciB3aGljaCBob2xkcyBzcGVjaWZpYyBlbnZpb3JubWVudCBjb25maWdzIGFuZCBzZWNyZXRzIAoKIyMgV29ya0Zsb3cKIVtXb3JrZmxvd10oZG9jdW1lbnRhdGlvbi93b3JrZmxvdy5wbmcpCgojIyBUZWNoClRPRE8gQXBwIHVzZXMgZm9sbG93aW5nIHRlY2hub2xvZ2llczoKCiogW0phdmEgMTFdIC0gSkRLCiogW01hdmVuLTMuNi4zXSAtIEJ1aWxkIHRvb2wKKiBbR3JhZGxlLTYuNy4xXSAtIEJ1aWxkIHRvb2wKKiBbU3ByaW5nYm9vdC0yLjcuMF0gLSBPcGVuIHNvdXJjZSBKYXZhLWJhc2VkIE1pY3JvLVNlcnZpY2UgZnJhbWV3b3JrCiogW09wZW5BcGktMS42LjldIC0gUmVzdCBBUEkgZG9jdW1lbnRhdGlvbgoqIFtKdW5pdC01LjcuMF0gLSBVbml0IHRlc3QKKiBbTW9uZ29EQi00LjIuMjFdIC0gQmFja2VuZCBkYXRhIHN0b3JlCiogW0RvY2tlci0xOS4wMy44XSAtIE9TIGxldmVsIHZpcnR1YWxpemF0aW9uCgojIyBCdWlsZCAjIwoxLiBUb2RvIEFwcAogICBbQnVpbGRdKC4vdG9kby1hcHAvUkVBRE1FLm1kKQoyLiBFZGdlIFNlcnZpY2UKICAgW0J1aWxkXSguL2VkZ2Utc2VydmljZS9SRUFETUUubWQpCjMuIENvbmZpZyBTZXJ2ZXIKICAgW0J1aWxkXSguL2NvbmZpZy1zZXJ2ZXIvUkVBRE1FLm1kKSAgIAogICAgCgoKIyMgRGVwbG95ICMjCiAKW1RvZG8gSW5mcmFdKGh0dHBzOi8vZ2l0aHViLmNvbS9SYWdoYXYyMjExL3RvZG8tYXBwLWluZnJhKQo= readmeEtag: '"712a9e04ce44758e96cbdffac5cd8f4cfd93cef1"' readmeLastModified: Sun, 21 Aug 2022 14:41:13 GMT repositoryId: 500385093 description: Todo Application with Cloud gateway & Config server created: '2022-06-06T10:19:14Z' updated: '2024-04-26T08:54:03Z' language: Java archived: false stars: 3 watchers: 1 forks: 2 owner: Raghav2211 logo: https://avatars.githubusercontent.com/u/7526431?v=4 repoEtag: '"888ce2e21649a2b5ffcbf4b0df51ab17def6e6378062912f0807fea8b594674f"' repoLastModified: Fri, 26 Apr 2024 08:54:03 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/manjeshpv/awesome-fullstack-sandbox v3: true id: 9210dd963ad7560db8ce67db9fd0b7b1 repositoryMetadata: base64Readme: >- IyBhd2Vzb21lLWZ1bGxzdGFjay1zYW5kYm94CgpbV2hhdHNBcHAgQ2hhbm5lbF0oaHR0cHM6Ly93aGF0c2FwcC5jb20vY2hhbm5lbC8wMDI5VmFCaUJUUDROVmliSzdlakFzMTQpCgpMb3cgQ29kZSBGdWxsIHN0YWNrIC0gQ2xvdWQgTmF0aXZlIERvY2tlcml6ZWQgU2FuZGJveCBmb3IgSW5kaWFuIFN0YXJ0dXBzIHdpdGggQ0kvQ0QKCgojIyMgQmFja2dyb3VuZAotIFdoYXQgaXMgW1NvZnR3YXJlIFNhbmRib3hdKGh0dHBzOi8vd3d3LnRlY2hvcGVkaWEuY29tL2RlZmluaXRpb24vMjc2ODEvc2FuZGJveC1zb2Z0d2FyZS10ZXN0aW5nKQotIFtJbnRyb2R1Y3Rpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9tYW5qZXNocHYvYXdlc29tZS1zYW5kYm94L2Jsb2IvbWFpbi9kb2NzL3JlYWRtZS5tZCkKLSBbR29hbHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9tYW5qZXNocHYvYXdlc29tZS1zYW5kYm94L2Jsb2IvbWFpbi9kb2NzL2dvYWxzLm1kKQotIFtUZWNobm9sb2d5IFN0YWNrXShodHRwczovL2dpdGh1Yi5jb20vbWFuamVzaHB2L2F3ZXNvbWUtc2FuZGJveC9ibG9iL21haW4vZG9jcy90ZWNoLXN0YWNrLm1kKQoKIyMjIENoaWxkIFByb2plY3RzCi0gaHR0cHM6Ly9naXRodWIuY29tL2NqZWxpdmFhcy9uZXN0anMtcGluby1kaXN0cmlidXRlZC10cmFjaW5nLW1vbm8KCiMjIyBVc2UgY2FzZXMKLSBGb3IgU3RhcnR1cCBGb3VuZGVycyAmIENYTydzCi0gRm9yIFN0YXJ0dXAgUHJvZHVjdCBNYW5hZ2VycwotIEZvciBTdGFydHVwIEFyY2hpdGVjdHMKLSBGb3IgU3RhcnR1cCBTZW5pb3IgRGV2ZWxvcGVycwotIEZvciBTdGFydHVwIEVuZ2luZWVycwotIEZvciBTdGFydHVwIFFBIEVuZ2luZWVycywgUGVyZm9ybWFuY2UgVGVzdGVycwotIEZvciBTdGFydHVwIFNlY3VyaXR5IEFkdmlzb3JzICYgUGVuIFRlc3RlcnMKCiMjIyBTcG9uc29ycwotIEVsaXZhYXM6IFtKb2IgT3BlbmluZ3NdKGh0dHBzOi8vd3d3Lmluc3RhaHlyZS5jb20vam9icy1hdC1lbGl2YWFzLykgfCBbRm9sbG93IG9uIEdpdGh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL2VsaXZhYXNkKQoKCiMjIyBDb250cmlidXRvcnMKLSBodHRwczovL2dpdGh1Yi5jb20vbWFuamVzaHB2Ci0gaHR0cHM6Ly9naXRodWIuY29tL0FzaHV0b3NoTW80OQotIGh0dHBzOi8vZ2l0aHViLmNvbS9BeXVzaENyaW8K readmeEtag: '"82948b98729d78c6c0104ba3b5af9b3047368cb4"' readmeLastModified: Sun, 07 Jul 2024 12:26:10 GMT repositoryId: 704798735 description: Cloud Native Docker Sandbox for Indian Startups created: '2023-10-14T05:59:20Z' updated: '2025-09-19T05:29:51Z' language: null archived: false stars: 3 watchers: 1 forks: 1 owner: manjeshpv logo: https://avatars.githubusercontent.com/u/1999957?v=4 repoEtag: '"a36577e88630b655b8dd61cfa1f54e5d4d12358b7e76dde59b269ad5dd30c681"' repoLastModified: Fri, 19 Sep 2025 05:29:51 GMT category: Server foundInMaster: true - source: openapi3 tags repository: https://github.com/orisai/openapi v3: true id: a755d23d53fa42bbbac1908bcd25d285 repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgoJPGltZyBzcmM9Imh0dHBzOi8vZ2l0aHViLmNvbS9vcmlzYWkvLmdpdGh1Yi9ibG9iL21haW4vaW1hZ2VzL3JlcG9fdGl0bGUucG5nP3Jhdz10cnVlIiBhbHQ9Ik9yaXNhaSIvPgoJPGJyLz4KCU9wZW5BUEkKPC9oMT4KCjxwIGFsaWduPSJjZW50ZXIiPgoJT3BlbkFQSSBidWlsZGVyLCBwYXJzZXIgYW5kIHNlcmlhbGl6ZXIKPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CglDb21wbGlhbnQgd2l0aCB0aGUgPGEgaHJlZj0iaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMS4wIj52My4xLjAgc3BlY3M8L2E+CjwvcD4KCjxwIGFsaWduPSJjZW50ZXIiPgoJ8J+ThCBDaGVjayBvdXQgb3VyIDxhIGhyZWY9ImRvY3MvUkVBRE1FLm1kIj5kb2N1bWVudGF0aW9uPC9hPi4KPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+Cgnwn5K4IElmIHlvdSBsaWtlIE9yaXNhaSwgcGxlYXNlIDxhIGhyZWY9Imh0dHBzOi8vb3Jpc2FpLmRldi9zcG9uc29yIj5tYWtlIGEgZG9uYXRpb248L2E+LiBUaGFuayB5b3UhCjwvcD4KCjxwIGFsaWduPSJjZW50ZXIiPgoJPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL29yaXNhaS9vcGVuYXBpL2FjdGlvbnM/cXVlcnk9d29ya2Zsb3c6Q0krYnJhbmNoOnYxLngiPjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vb3Jpc2FpL29wZW5hcGkvYWN0aW9ucy93b3JrZmxvd3MvY2kueWFtbC9iYWRnZS5zdmc/YnJhbmNoPXYxLngiPjwvYT4KCTxhIGhyZWY9Imh0dHBzOi8vY292ZXJhbGxzLmlvL2dpdGh1Yi9vcmlzYWkvb3BlbmFwaT9icmFuY2g9djEueCI+PGltZyBzcmM9Imh0dHBzOi8vYmFkZ2VuLm5ldC9jb3ZlcmFsbHMvYy9naXRodWIvb3Jpc2FpL29wZW5hcGkvdjEueD9jYWNoZT0zMDAiPjwvYT4KCTxhIGhyZWY9Imh0dHBzOi8vZGFzaGJvYXJkLnN0cnlrZXItbXV0YXRvci5pby9yZXBvcnRzL2dpdGh1Yi5jb20vb3Jpc2FpL29wZW5hcGkvdjEueCI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZW5kcG9pbnQ/c3R5bGU9ZmxhdCZ1cmw9aHR0cHM6Ly9iYWRnZS1hcGkuc3RyeWtlci1tdXRhdG9yLmlvL2dpdGh1Yi5jb20vb3Jpc2FpL29wZW5hcGkvdjEueCI+PC9hPgoJPGEgaHJlZj0iaHR0cHM6Ly9wYWNrYWdpc3Qub3JnL3BhY2thZ2VzL29yaXNhaS9vcGVuYXBpIj48aW1nIHNyYz0iaHR0cHM6Ly9iYWRnZW4ubmV0L3BhY2thZ2lzdC9kdC9vcmlzYWkvb3BlbmFwaT9jYWNoZT0zNjAwIj48L2E+Cgk8YSBocmVmPSJodHRwczovL3BhY2thZ2lzdC5vcmcvcGFja2FnZXMvb3Jpc2FpL29wZW5hcGkiPjxpbWcgc3JjPSJodHRwczovL2JhZGdlbi5uZXQvcGFja2FnaXN0L3Yvb3Jpc2FpL29wZW5hcGk/Y2FjaGU9MzYwMCI+PC9hPgoJPGEgaHJlZj0iaHR0cHM6Ly9jaG9vc2VhbGljZW5zZS5jb20vbGljZW5zZXMvbXBsLTIuMC8iPjxpbWcgc3JjPSJodHRwczovL2JhZGdlbi5uZXQvYmFkZ2UvbGljZW5zZS9NUEwtMi4wL2JsdWU/Y2FjaGU9MzYwMCI+PC9hPgo8cD4KCiMjCg== readmeEtag: '"9f1699f73d78f9ee72d7af92b3271eb6adbd6844"' readmeLastModified: Fri, 21 Jun 2024 22:26:58 GMT repositoryId: 547861622 description: '[WIP] OpenAPI builder, parser and serializer' created: '2022-10-08T12:47:28Z' updated: '2024-12-29T16:06:12Z' language: PHP archived: false stars: 3 watchers: 1 forks: 0 owner: orisai logo: https://avatars.githubusercontent.com/u/68913382?v=4 license: MPL-2.0 repoEtag: '"25dd62329ede859b543f558f93850431bcf5c4f255719630bb0d44d7b9d5e741"' repoLastModified: Sun, 29 Dec 2024 16:06:12 GMT category: Testing foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/api-flows/openapi-workflow-parser v3: true v3_1: true id: de663aa29fdc4381a95f3bcf538e307f repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFdvcmtmbG93IChBcmF6em8pIFBhcnNlcgoKWyFbXShodHRwczovL2JhZGdlbi5uZXQvZ2l0aHViL2xpY2Vuc2UvQVBJLUZsb3dzL29wZW5hcGktd29ya2Zsb3ctcGFyc2VyKV0oTElDRU5TRSkKWyFbXShodHRwczovL2JhZGdlbi5uZXQvbWF2ZW4vdi9tYXZlbi1jZW50cmFsL2NvbS5hcGktZmxvd3Mvb3BlbmFwaS13b3JrZmxvdy1wYXJzZXIpXShodHRwczovL3JlcG8xLm1hdmVuLm9yZy9tYXZlbjIvY29tL2FwaS1mbG93cy9vcGVuYXBpLXdvcmtmbG93LXBhcnNlci8pClshW1F1YWxpdHkgR2F0ZSBTdGF0dXNdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PUFQSS1GbG93c19vcGVuYXBpLXdvcmtmbG93LXBhcnNlciZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9QVBJLUZsb3dzX29wZW5hcGktd29ya2Zsb3ctcGFyc2VyKQoKUGFyc2luZyBBUEkgd29ya2Zsb3dzLgoKIyMgT3ZlcnZpZXcKClRoZSBPcGVuQVBJIFdvcmtmbG93IHBhcnNlciBpcyBhbiBvcGVuLXNvdXJjZSBKYXZhIGxpYnJhcnkgZGVzaWduZWQgdG8gcGFyc2UgdGhlIFtPQUkgQXJhenpvIHNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvQXJhenpvLVNwZWNpZmljYXRpb24vKSBmaWxlcy4gSXQgcmVhZHMgYW4gQXJhenpvIGZpbGUgKEpTT04gb3IgWUFNTCBmb3JtYXRzIGFyZSBzdXBwb3J0ZWQpIGFuZCBjcmVhdGVzIHRoZSBjb3JyZXNwb25kaW5nIEphdmEgb2JqZWN0cy4gIAoKVGhlIHBhcnNlcidzIGdvYWwgaXMgdG8gc2ltcGxpZnkgdGhlIGV4dHJhY3Rpb24gYW5kIG1hbmlwdWxhdGlvbiBvZiB3b3JrZmxvd3MsIGhlbHBpbmcgZGV2ZWxvcGVycyBjcmVhdGUgYXBwbGljYXRpb25zIGFuZCB0b29scyB0aGF0IGhhcm5lc3MgdGhlIHNlbWFudGljIHN0cnVjdHVyZSBvZiBBUEkgZmxvd3MuCgojIyBGZWF0dXJlcwoKLSAqKldvcmtmbG93IFBhcnNpbmc6KiogUmVhZHMgT0FJIEFyYXp6byBzcGVjaWZpY2F0aW9uIGZpbGVzIGxvYWRpbmcgdGhlIGNvcnJlc3BvbmRpbmcgSmF2YSBvYmplY3RzLgotICoqRWFzZSBvZiBVc2U6KiogUHJvdmlkZXMgYSBzaW1wbGUgd2F5IGZvciBkZXZlbG9wZXJzIHRvIHBhcnNlIHdvcmtmbG93cy4KLSAqKkNvbXBhdGliaWxpdHk6KiogU3VwcG9ydHMgYm90aCBKU09OIGFuZCBZQU1MIGZvcm1hdHMuCi0gKipWYWxpZGF0aW9uOioqIFZhbGlkYXRlcyB0aGUgc3BlY2lmaWNhdGlvbiBhY2NvcmRpbmcgdG8gdGhlIFtBcmF6em8gU3BlY2lmaWNhdGlvbiB2MS4wLjBdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvQXJhenpvLVNwZWNpZmljYXRpb24vKS4KICAKIyMgVXNhZ2UKCiMjIyBBZGQgdG8gdGhlIHByb2plY3QKCllvdSBjYW4gaW5jbHVkZSB0aGlzIGxpYnJhcnkgZnJvbSBNYXZlbiBjZW50cmFsOgpgYGAKICA8ZGVwZW5kZW5jeT4KICAgIDxncm91cElkPmNvbS1hcGktZmxvd3M8L2dyb3VwSWQ+CiAgICA8YXJ0aWZhY3RJZD5vcGVuYXBpLXdvcmtmbG93LXBhcnNlcjwvYXJ0aWZhY3RJZD4KICAgIDx2ZXJzaW9uPjAuMC4zPC92ZXJzaW9uPgogIDwvZGVwZW5kZW5jeT4KYGBgCgpQYXJzZSBmcm9tIGZpbGU6CmBgYGphdmEKICBmaW5hbCBTdHJpbmcgV09SS0ZMT1dTX1NQRUMgPSAicGF0aC9wZXQtY291cG9ucy53b3JrZmxvdy55YW1sIjsKCiAgT3BlbkFQSVdvcmtmbG93UGFyc2VyUmVzdWx0IHJlc3VsdCA9IHBhcnNlci5wYXJzZShXT1JLRkxPV1NfU1BFQyk7CgogIGJvb2xlYW4gdmFsaWQgPSByZXN1bHQuaXNWYWxpZCgpOwogIFN0cmluZyB0aXRsZSA9IHJlc3VsdC5nZXRPcGVuQVBJV29ya2Zsb3coKS5nZXRJbmZvKCkuZ2V0VGl0bGUoKTsKYGBgCgpQYXJzZSBmcm9tIFVSTDoKYGBgamF2YQogIGZpbmFsIFN0cmluZyBXT1JLRkxPV1NfU1BFQyA9ICJodHRwczovL2hvc3QvcGF0aC9wZXQtY291cG9ucy53b3JrZmxvdy55YW1sIjsKCiAgT3BlbkFQSVdvcmtmbG93UGFyc2VyUmVzdWx0IHJlc3VsdCA9IHBhcnNlci5wYXJzZShXT1JLRkxPV1NfU1BFQyk7CgogIGJvb2xlYW4gdmFsaWQgPSByZXN1bHQuaXNWYWxpZCgpOwogIFN0cmluZyB0aXRsZSA9IHJlc3VsdC5nZXRPcGVuQVBJV29ya2Zsb3coKS5nZXRJbmZvKCkuZ2V0VGl0bGUoKTsKYGBgCgojIyBCdWlsZCBmcm9tIHNvdXJjZQoKQ2xvbmUgZnJvbSB0aGUgR2l0SHViIHJlcG9zaXRvcnkKCmBgYGJhc2gKICBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL0FQSS1GbG93cy9vcGVuYXBpLXdvcmtmbG93LXBhcnNlci5naXQKICBjZCBvcGVuYXBpLXdvcmtmbG93LXBhcnNlcgogIG12biBwYWNrYWdlCmBgYAoKIyMgVXNlIHNuYXBzaG90cwoKQWRkIHRoZSBNYXZlbiByZXBvc2l0b3J5IGZvciB0aGUgT3BlbkFQSSB3b3JrZmxvdyBwYXJzZXIgc25hcHNob3RzOgpgYGB4bWwKIDxyZXBvc2l0b3JpZXM+CiAgICA8cmVwb3NpdG9yeT4KICAgICAgPGlkPmdpdGh1YjwvaWQ+CiAgICAgIDx1cmw+aHR0cHM6Ly9tYXZlbi5wa2cuZ2l0aHViLmNvbS9BUEktRmxvd3Mvb3BlbmFwaS13b3JrZmxvdy1wYXJzZXI8L3VybD4KICAgICAgPHNuYXBzaG90cz4KICAgICAgICA8ZW5hYmxlZD50cnVlPC9lbmFibGVkPgogICAgICA8L3NuYXBzaG90cz4KICAgIDwvcmVwb3NpdG9yeT4KICA8L3JlcG9zaXRvcmllcz4KYGBgCkFkZCB0aGUgU05BUFNIT1QgZGVwZW5kZW5jeSAoW2NoZWNrIGxhdGVzdF0ocG9tLnhtbCkgYXZhaWxhYmxlKSBpbiB5b3VyIFBPTSBmaWxlOgpgYGB4bWwKICAgIDxkZXBlbmRlbmN5PgogICAgICA8Z3JvdXBJZD5jb20uYXBpLWZsb3dzPC9ncm91cElkPgogICAgICA8YXJ0aWZhY3RJZD5vcGVuYXBpLXdvcmtmbG93LXBhcnNlcjwvYXJ0aWZhY3RJZD4KICAgICAgPHZlcnNpb24+MC4wLjQtU05BUFNIT1Q8L3ZlcnNpb24+CiAgICA8L2RlcGVuZGVuY3k+CmBgYAo= readmeEtag: '"6a706544a68f764fb1ce2700116a45db7baea395"' readmeLastModified: Tue, 13 Aug 2024 10:48:40 GMT repositoryId: 729817092 description: Java parser of the OpenAPI workflow (Arazzo) specification created: '2023-12-10T13:02:41Z' updated: '2025-08-18T07:56:48Z' language: Java archived: false stars: 4 watchers: 1 forks: 1 owner: API-Flows logo: https://avatars.githubusercontent.com/u/153562650?v=4 license: Apache-2.0 repoEtag: '"6eab4b40a4db59f0ff1a7479076154faef3d3433fcc441076be9c88260a836b4"' repoLastModified: Mon, 18 Aug 2025 07:56:48 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mhapach/swaggermodelgenerator v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyTW9kZWxHZW5lcmF0b3IKClshW0xhdGVzdCBWZXJzaW9uIG9uIFBhY2thZ2lzdF1baWNvLXZlcnNpb25dXVtsaW5rLXBhY2thZ2lzdF0KWyFbVG90YWwgRG93bmxvYWRzXVtpY28tZG93bmxvYWRzXV1bbGluay1kb3dubG9hZHNdClshW0J1aWxkIFN0YXR1c11baWNvLXRyYXZpc11dW2xpbmstdHJhdmlzXQpbIVtTdHlsZUNJXVtpY28tc3R5bGVjaV1dW2xpbmstc3R5bGVjaV0KClRoaXMgbGlicmFyeSBnZW5lcmF0ZXMgbW9kZWxzIGFuZCBzZXJ2aWNlIHdpdGggbWV0aG9kcyBiYXNlZCBvbiBTd2FnZ2VyIHNjaGVtZS4gCkN1cnJlbnQgcmVsZWFzZSBzdXBwb3J0cyBPcGVuQVBJIDIuMCAoQWthIE9BUykgYW5kIE9wZW5BcGkgMy4wCgojIyBJbnN0YWxsYXRpb24KClN0ZXAgMS4gQWRkIEVsb3F1ZW50IE1vZGVsIEdlbmVyYXRvciB0byB5b3VyIHByb2plY3QgdmlhIENvbXBvc2VyCgpgYGAgYmFzaAokIGNvbXBvc2VyIHJlcXVpcmUgbWhhcGFjaC9zd2FnZ2VybW9kZWxnZW5lcmF0b3IKYGBgClN0ZXAgMi4gUmVnaXN0ZXIgU3dhZ2dlck1vZGVsR2VuZXJhdG9yU2VydmljZVByb3ZpZGVyIGluIGNvbmZpZy9hcHAucGhwCmBgYCAgCidwcm92aWRlcnMnID0+IFsKICAgIC8vLi4uCiAgICBcbWhhcGFjaFxTd2FnZ2VyTW9kZWxHZW5lcmF0b3JcU3dhZ2dlck1vZGVsR2VuZXJhdG9yU2VydmljZVByb3ZpZGVyOjpjbGFzcywKXQpgYGAKCiMjIFVzYWdlCiMjIyBNb2RlbHMgYW5kIHNlcnZpY2UgZ2VuZXJhdGlvbgogICAgICAKICAgICRzZXJ2aWNlU2NoZW1lID0gImh0dHA6Ly95b3VyLXNlcnZpY2UuY29tL3NjaGVtZSI7CiAgICAkbW9kZWxzTmFtZXNwYWNlID0gJ0FwcFxTZXJ2aWNlc1xNb2RlbHMnOwogICAgJHNlcnZpY2VOYW1lc3BhY2UgPSAnQXBwXFNlcnZpY2VzJzsKICAgICRtb2RlbHNQYXRoID0gIi95b3VyLXByb2plY3QvYXBwL1NlcnZpY2VzL01vZGVscyI7CiAgICAkc2VydmljZVBhdGggPSAiL3lvdXItcHJvamVjdC9hcHAvU2VydmljZXMiOwoKICAgIC8qKiBAdmFyIFN3YWdnZXIgJGNvbnZlcnRlckluc3RhbmNlICovCiAgICAkY29udmVydGVySW5zdGFuY2UgPSAobmV3IFN3YWdnZXJNb2RlbEdlbmVyYXRvcigkc2VydmljZVNjaGVtZSwgdHJ1ZSkpLT5nZXRDb252ZXJ0ZXJJbnN0YW5jZSgkbW9kZWxzTmFtZXNwYWNlLCAkc2VydmljZU5hbWVzcGFjZSk7CiAgICAkY29udmVydGVySW5zdGFuY2UtPmdlbk1vZGVscygkbW9kZWxzUGF0aCk7ICAgICAgICAKICAgICRjb252ZXJ0ZXJJbnN0YW5jZS0+Z2VuU2VydmljZSgkc2VydmljZVBhdGgpOwogICAgCiMjIyBHZW5lcmF0ZWQgc2VydmljZSBhbmQgbW9kZWxzIHVzYWdlLiBFeGFtcGxlIDEKICAgICRzZXJ2aWNlQWRkcmVzcyA9ICJodHRwOi8veW91ci1zZXJ2aWNlLmNvbS9zb21lLW5hbWUiOwogICAgJHNlcnZpY2UgPSBuZXcgU2VydmljZSgkc2VydmljZUFkZHJlc3MpOwogICAgJGJjID0gJHNlcnZpY2UtPmJlbmVmaXRDYXRlZ29yaWVzVXNpbmdHRVQoWwogICAgICAgICdwYXRoJyA9PiBbJ2lkJyA9PiAxMTIyXSwKICAgICAgICAncXVlcnknID0+IFsnc29tZS1wYXJhbScgPT4gMV0sCiAgICBdKTsgICAKIyMjIEdlbmVyYXRlZCBzZXJ2aWNlIGFuZCBtb2RlbHMgdXNhZ2UuIEV4YW1wbGUgMgogICAgJHNlcnZpY2VBZGRyZXNzID0gImh0dHA6Ly95b3VyLXNlcnZpY2UuY29tL3NvbWUtbmFtZSI7CiAgICAkdG9rZW4gPSAnZGFrc2Rsa2Egc2hkbGtqYWhzbGtkaiBoPT0nOwogICAgJGhlYWRlcnMgPSBbCiAgICAgICAgJ0FjY2VwdCcgPT4gJyovKicsCiAgICAgICAgJ0NvbnRlbnQtVHlwZScgPT4gJ2FwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZCcsCiAgICAgICAgJ0NhY2hlLUNvbnRyb2wnID0+ICduby1jYWNoZScsCiAgICAgICAgIkF1dGhvcml6YXRpb24iID0+ICJCZWFyZXIgeyR0b2tlbn0iCiAgICBdOwogICAgJHNlcnZpY2UgPSBuZXcgU2VydmljZSgkc2VydmljZUFkZHJlc3MsICRoZWFkZXJzKTsKICAgICRiYyA9ICRzZXJ2aWNlLT5iZW5lZml0Q2F0ZWdvcmllc1VzaW5nR0VUKFsKICAgICAgICAncXVlcnknID0+IFsnc29tZS1wYXJhbScgPT4gMV0sCiAgICAgICAgJ2JvZHknID0+ICd7InBhcmFtMSI6MTAwLCAicGFyYW0yIjoyMDB9JwogICAgXSk7CiAgICAgICAKIyMjIERlYnVnIGxvZyBlbmFibGluZy4gRXhhbXBsZSAzCiAgICAkc2VydmljZUFkZHJlc3MgPSAiaHR0cDovL3lvdXItc2VydmljZS5jb20vc29tZS1uYW1lIjsKICAgICRjb25maWcgPSBjb25maWcoJ2xvZ2dpbmcuY2hhbm5lbHMuJy5jb25maWcoJ2xvZ2dpbmcuZGVmYXVsdCcpKTsKICAgICRjb25maWdbJ3BhdGgnXSA9IHN0b3JhZ2VfcGF0aCgibG9ncy9yZXN0LmxvZyIpOyAvLyBvcHRpb25hbAogICAgJGxvZ2dlciA9IFxMb2c6OmJ1aWxkKCRjb25maWcpOwogICAgJHNlcnZpY2UgPSBuZXcgU2VydmljZSgkc2VydmljZUFkZHJlc3MsIFsnQ29udGVudC1UeXBlJyA9PiAnYXBwbGljYXRpb24vanNvbiddLCAkbG9nZ2VyKTsKICAgIC8vRW5hYmxpbmcgbG9nICAgIAogICAgJHNlcnZpY2UtPmVuYWJsZUxvZygpOwogICAgLy8gY2FsbCBzZXJ2aWNlIG1ldGhvZCAgICAgICAgCiAgICAkYmMgPSAkc2VydmljZS0+YmVuZWZpdENhdGVnb3JpZXNVc2luZ0dFVChbCiAgICAgICAgJ3F1ZXJ5JyA9PiBbJ3NvbWUtcGFyYW0nID0+IDFdLAogICAgICAgICdib2R5JyA9PiAneyJwYXJhbTEiOjEwMCwgInBhcmFtMiI6MjAwfScKICAgIF0pOyAgIAogICAgCiMjIENoYW5nZSBsb2cKClBsZWFzZSBzZWUgdGhlIFtjaGFuZ2Vsb2ddKGNoYW5nZWxvZy5tZCkgZm9yIG1vcmUgaW5mb3JtYXRpb24gb24gd2hhdCBoYXMgY2hhbmdlZCByZWNlbnRseS4KCiMjIFRlc3RpbmcKCmBgYCBiYXNoCiQgY29tcG9zZXIgdGVzdApgYGAKCiMjIENvbnRyaWJ1dGluZwoKUGxlYXNlIHNlZSBbY29udHJpYnV0aW5nLm1kXShjb250cmlidXRpbmcubWQpIGZvciBkZXRhaWxzIGFuZCBhIHRvZG9saXN0LgoKIyMgU2VjdXJpdHkKCklmIHlvdSBkaXNjb3ZlciBhbnkgc2VjdXJpdHkgcmVsYXRlZCBpc3N1ZXMsIHBsZWFzZSBlbWFpbCBhdXRob3IgZW1haWwgaW5zdGVhZCBvZiB1c2luZyB0aGUgaXNzdWUgdHJhY2tlci4KCiMjIENyZWRpdHMKCi0gW2F1dGhvciBuYW1lXVtsaW5rLWF1dGhvcl0KLSBbQWxsIENvbnRyaWJ1dG9yc11bbGluay1jb250cmlidXRvcnNdCgojIyBMaWNlbnNlCgpsaWNlbnNlLiBQbGVhc2Ugc2VlIHRoZSBbbGljZW5zZSBmaWxlXShsaWNlbnNlLm1kKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCltpY28tdmVyc2lvbl06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L3YvbWhhcGFjaC9zd2FnZ2VybW9kZWxnZW5lcmF0b3Iuc3ZnP3N0eWxlPWZsYXQtc3F1YXJlCltpY28tZG93bmxvYWRzXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9wYWNrYWdpc3QvZHQvbWhhcGFjaC9zd2FnZ2VybW9kZWxnZW5lcmF0b3Iuc3ZnP3N0eWxlPWZsYXQtc3F1YXJlCltpY28tdHJhdmlzXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90cmF2aXMvbWhhcGFjaC9zd2FnZ2VybW9kZWxnZW5lcmF0b3IvbWFzdGVyLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZQpbaWNvLXN0eWxlY2ldOiBodHRwczovL3N0eWxlY2kuaW8vcmVwb3MvMTIzNDU2Nzgvc2hpZWxkCgpbbGluay1wYWNrYWdpc3RdOiBodHRwczovL3BhY2thZ2lzdC5vcmcvcGFja2FnZXMvbWhhcGFjaC9zd2FnZ2VybW9kZWxnZW5lcmF0b3IKW2xpbmstZG93bmxvYWRzXTogaHR0cHM6Ly9wYWNrYWdpc3Qub3JnL3BhY2thZ2VzL21oYXBhY2gvc3dhZ2dlcm1vZGVsZ2VuZXJhdG9yCltsaW5rLXRyYXZpc106IGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9taGFwYWNoL3N3YWdnZXJtb2RlbGdlbmVyYXRvcgpbbGluay1zdHlsZWNpXTogaHR0cHM6Ly9zdHlsZWNpLmlvL3JlcG9zLzEyMzQ1Njc4CltsaW5rLWF1dGhvcl06IGh0dHBzOi8vZ2l0aHViLmNvbS9taGFwYWNoCltsaW5rLWNvbnRyaWJ1dG9yc106IC4uLy4uL2NvbnRyaWJ1dG9ycwo= readmeEtag: '"6d5573fd487cae3f0cfff219e63786cae1dd1c0d"' readmeLastModified: Tue, 14 May 2024 11:07:14 GMT repositoryId: 257096773 description: >- Laravel library that generates models and service with methods based on Swagger scheme. Current release supports OpenAPI 2.0 (Aka OAS) and OpenApi 3.0 created: '2020-04-19T20:34:51Z' updated: '2024-09-16T15:27:48Z' language: PHP archived: false stars: 3 watchers: 1 forks: 0 owner: mhapach logo: https://avatars.githubusercontent.com/u/6570258?v=4 license: NOASSERTION repoEtag: '"5e6f37340d57e51bbd18953e22596b0e3399ceaad1ba2b5db27111c7ef51da46"' repoLastModified: Mon, 16 Sep 2024 15:27:48 GMT foundInMaster: true category: Data Validators id: c60edd99eed7b5341001026b01957e0b - source: - openapi3 tags - openapi31 tags repository: https://github.com/bump-sh-examples/rails-design-first v3: true v3_1: true id: 04a4ac10f8421c07973b6c7150ad5567 repositoryMetadata: base64Readme: >- IyBSYWlscyBIZWxsbyBPcGVuQVBJCgpUaGlzIHJlcG9zaXRvcnkgd2FzIGJ1aWx0IGFzIHNhbXBsZSBjb2RlIGZvciB0aGUgQnVtcC5zaCBndWlkZSBvbiBfW1VzaW5nIE9wZW5BUEkgdG8gc2ltcGxpZnkgYnVpbGRpbmcgYW5kIHRlc3RpbmcgUnVieSBvbiBSYWlscyBBUElzXShodHRwczovL2RvY3MuYnVtcC5zaC9ndWlkZXMvb3BlbmFwaS9kZXNpZ24tZmlyc3QtcmFpbHMvKV8uIExlYXJuIHRvIHVzZSB0aGUgQVBJIERlc2lnbiBGaXJzdCB3b3JrZmxvdywgYW5kIHNpbXBsaWZ5IHlvdXIgUmFpbHMgY29kZSBieSBub3QgaGF2aW5nIHRvIHJlcGVhdCB5b3VyIEFQSSBjb250cmFjdCBpbiB2YWxpZGF0aW9uIGFuZCBjb250cmFjdCB0ZXN0aW5nIGFzIHdlbGwgYXMgZG9jdW1lbnRhdGlvbi4gSnVzdCBkbyBpdCBvbmNlLCBtYWtlIGVhc3kgY29kZSwgdGhlbiBwb3AgaXQgYWxsIG9udG8gQnVtcC5zaCB0byBoYXZlIFtncmVhdCBBUEkgZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9idW1wLnNoL2J1bXAtZXhhbXBsZXMvaHViL2NvZGUtc2FtcGxlcy9kb2MvcmFpbHMtZGVzaWduLWZpcnN0KS4KCiMjIFVzYWdlCgpDbG9uZSB0aGUgcmVwb3NpdG9yeSBkb3duIHRvIGdpdmUgaXQgYSB0cnkuCgpgYGAKIyBTZXQgZXZlcnl0aGluZyB1cAokIGJ1bmRsZSBpbnN0YWxsCgojIFN0YXJ0IHRoZSBzZXJ2ZXIKJCByYWlscyBzCgojIFBva2UgdGhlIEFQSSBhbmQgZ2V0IGF1dG9tYXRlZCAgZXJyb3JzCiQgY3VybCAtWCBQT1NUIGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMC93aWRnZXRzIC1IICJDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24iIC1kICd7fScgIHwganEgLgoKewogICJ0aXRsZSI6ICJCYWQgUmVxdWVzdCBCb2R5IiwKICAic3RhdHVzIjogNDAwLAogICJlcnJvcnMiOiBbCiAgICB7CiAgICAgICJtZXNzYWdlIjogIm9iamVjdCBhdCByb290IGlzIG1pc3NpbmcgcmVxdWlyZWQgcHJvcGVydGllczogbmFtZSIsCiAgICAgICJwb2ludGVyIjogIiIsCiAgICAgICJjb2RlIjogInJlcXVpcmVkIgogICAgfQogIF0KfQpgYGAKCkdpdmUgaXQgYSB0cnksIHBsYXkgYXJvdW5kIHdpdGggdGhlIE9wZW5BUEksIGFuZCBzZWUgaG93IGl0IHJlc3BvbmRzIHRvIGRpZmZlcmVudCBzY2VuYXJpb3MuIAoKVGhlbiB5b3UgY2FuIHJ1biBgcnNwZWNgIHRvIHNlZSBpZiB0aGUgQVBJIHJlc3BvbnNlcyBtYXRjaCB3aGF0IE9wZW5BUEkgZXhwZWN0LCB3aGljaCB3aGVuIGltcGxlbWVudGVkIGluIHlvdXIgYXBwbGljYXRpb24gd2lsbCBoZWxwIG1ha2Ugc3VyZSB5b3VyIEFQSSBpcyBhY3R1YWxseSBkb2luZyB3aGF0IHlvdXIgZG9jcyBhcmUgc2F5aW5nLCBvciBtYWtlIHN1cmUgeW91ciBkb2NzIGFyZSBzYXlpbmcgd2hhdCB5b3VyIEFQSSBpcyBkb2luZywgd2hpY2hldmVyIHdheSByb3VuZCB5b3UgcHJlZmVyIHRvIHRoaW5rIG9mIGl0LgoKUHJldmlldyBob3cgdGhlIEFQSSByZWZlcmVuY2UgZG9jcyBsb29rIFtvbiBCdW1wLnNoXShodHRwczovL2J1bXAuc2gvYnVtcC1leGFtcGxlcy9odWIvY29kZS1zYW1wbGVzL2RvYy9yYWlscy1kZXNpZ24tZmlyc3QpLgoKIyMgTGljZW5zZQoKVGhlIGNvbnRlbnRzIG9mIHRoaXMgcmVwb3NpdG9yeSBhcmUgbGljZW5zZWQgdW5kZXIgW0NDIEJZLU5DLVNBCiAgNC4wXSguL0xJQ0VOU0VfQ0MtQlktTkMtU0EtNC4wKS4K readmeEtag: '"c03cf67788386b81839994ea8f387da606473b63"' readmeLastModified: Mon, 11 Mar 2024 16:05:27 GMT repositoryId: 740102881 description: >- Speed up your Rails API development with OpenAPI request validation and contract testing. created: '2024-01-07T14:36:56Z' updated: '2025-06-26T12:27:43Z' language: Ruby archived: false stars: 3 watchers: 2 forks: 0 owner: bump-sh-examples logo: https://avatars.githubusercontent.com/u/157144805?v=4 license: NOASSERTION repoEtag: '"2b6c7347c55622bdcd1228ce1e6b1bf808ccd5ada11ceb3ecff88923a5fc05c3"' repoLastModified: Thu, 26 Jun 2025 12:27:43 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/tsingsun/woocoo v3: true id: e194113fddfe7be854f117e3eadbe339 repositoryMetadata: base64Readme: >- IyBXb29Db28KClshW0xhbmd1YWdlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xhbmd1YWdlLUdvLWJsdWUuc3ZnKV0oaHR0cHM6Ly9nb2xhbmcub3JnLykKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL3RzaW5nc3VuL3dvb2Nvby9icmFuY2gvbWFpbi9ncmFwaC9iYWRnZS5zdmcpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvdHNpbmdzdW4vd29vY29vKQpbIVtHbyBSZXBvcnQgQ2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vdHNpbmdzdW4vd29vY29vKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL3RzaW5nc3VuL3dvb2NvbykKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vdHNpbmdzdW4vd29vY29vL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vdHNpbmdzdW4vd29vY29vL2FjdGlvbnM/cXVlcnk9YnJhbmNoJTNBbWFpbikKWyFbUmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvcmVsZWFzZS90c2luZ3N1bi93b29jb28uc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9naXRodWIuY29tL3RzaW5nc3VuL3dvb2Nvby9yZWxlYXNlcykKWyFbR29Eb2NdKGh0dHBzOi8vcGtnLmdvLmRldi9iYWRnZS9naXRodWIuY29tL3RzaW5nc3VuL3dvb2Nvbz9zdGF0dXMuc3ZnKV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2dpdGh1Yi5jb20vdHNpbmdzdW4vd29vY29vP3RhYj1kb2MpCgojIyBJbnRyb2R1Y3Rpb24KCmBXb29Db29gIGlzIGFuIGFwcGxpY2F0aW9uIGRldmVsb3BtZW50IGZyYW1ld29yayBhbmQgdG9vbGtpdCB3cml0dGVuIGluIEdPKEdvbGFuZykuIEl0IGlzIGVhc3kgdG8gZGV2ZWxvcCBXZWJBcGkgYXBwbGljYXRpb25zIG9yIFJQQyBzZXJ2aWNlcy4KCmBXb29Db29gIG1haW5seSBwbGF5cyBhIHJvbGUgb2YgYWRoZXNpdmUsIGFuZCBpdHMgY29yZSBjb21wb25lbnRzIGFyZSBmcm9tIG90aGVyIG9wZW4gc291cmNlIHByb2plY3RzLiAKVGhlIGN1cnJlbnQgZmVhdHVyZXMgYXJlIGFzIGZvbGxvd3M6CgojIEZlYXR1cmVzCi0gW3hdIGNvbXBvbmVudCBjb25maWd1cmFibGUsZWFzeSB0byBzcGxpdCBtdWx0aSBlbnZpcm9ubWVudHMKLSBbeF0gbG9nZ2VyIGFuZCByb3RhdGUgc3VwcG9ydC4gW0RldGFpbF0oZG9jcy9tZC9sb2dnZXIubWQpLAotIFt4XSBPcGVuVGVsZW1ldHJ5IHN1cHBvcnQuIFtEZXRhaWxdKGRvY3MvbWQvb3RlbC5tZCkKLSBbeF0gYnVpbHQtaW4gd2ViIHJvdXRlcixzdXBwb3J0cyBHcmFwaFFMLgotIFt4XSBidWlsdC1pbiBncnBjIHNlcnZlciBhbmQgZWFzeSB0byB1c2UgZ3JwYyBjbGllbnQuCi0gW3hdIEpXVC1iYXNlZCB2YWxpZGF0aW9uCi0gW3hdIG1pY3Jvc2VydmljZSByZWdpc3RyeSBhbmQgZGlzY292ZXJ5OiAKICAtIGV0Y2R2MzogcmVnaXN0ZXIgYW5kIGRpc2NvdmVyeSBzZXJ2aWNlcyAKICAtIFtQb2xhcmlzXShodHRwczovL2dpdGh1Yi5jb20vcG9sYXJpc21lc2gvcG9sYXJpcyk6IHNlcnZpY2UgZGlzY292ZXJ5IGFuZCBnb3Zlcm5hbmNlCgojIyBUb29scwoKLSB3b2NvLWNsaTogY29tbWFuZCBsaW5lIHRvb2wsIGluY2x1ZGUgZmVhdHVyZToKCiAgLSBbeF0gZ2VuZXJhdGUgY29kZSBzdXBwb3J0OiBgRW50YAogIC0gW3hdIHNlcnZlciBjb2RlIGdlbmVyYXRvciBmb3Igb3BlbmFwaSAzLjAgc3BlYy4gc2VlW09wZW5BcGkzIEdlbmVyYXRvcl0oZG9jcy9tZC9vYXNnZW4ubWQpCgojIyBXb3JrIFdpdGgKCi0gW2ZhY2Vib29rIGVudF0oaHR0cHM6Ly9naXRodWIuY29tL2VudC9lbnQpCi0gR3JhcGhxbDogYnkgZW50CgojIyBleGFtcGxlcwoKW3dvb2NvbyBleGFtcGxlc10oaHR0cHM6Ly9naXRodWIuY29tL3RzaW5nc3VuL3dvb2Nvby1leGFtcGxlKQoKIyMgb3RoZXJzOgoKY29udGFjdDoKLSBRUTogMjE5OTcyNzIKCiMjIFRoYW5rcwoKIVtpbWFnZV0oaHR0cHM6Ly9yZXNvdXJjZXMuamV0YnJhaW5zLmNvbS9zdG9yYWdlL3Byb2R1Y3RzL2NvbXBhbnkvYnJhbmQvbG9nb3MvamJfYmVhbS5zdmcp readmeEtag: '"a5a57ac9a75461312cf043b31c1a5171a33c6e8c"' readmeLastModified: Tue, 13 Aug 2024 08:15:25 GMT repositoryId: 129063014 description: an application development framework and toolkit created: '2018-04-11T08:44:52Z' updated: '2025-12-24T09:20:23Z' language: Go archived: false stars: 7 watchers: 2 forks: 0 owner: tsingsun logo: https://avatars.githubusercontent.com/u/5848549?v=4 license: Apache-2.0 repoEtag: '"a522bb5944a526604fd60fc99ddf9211cb2ca3346a223796c8dbb0b66da73bdb"' repoLastModified: Wed, 24 Dec 2025 09:20:23 GMT category: Server Implementations foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/swaggerexpert/openapi-server-url-templating v3: true v3_1: true id: 28ee2f640ed6f6fc55bfdae1fdd8fa96 repositoryMetadata: base64Readme: >- # openapi-server-url-templating

[![npmversion](https://badge.fury.io/js/openapi-server-url-templating.svg)](https://www.npmjs.com/package/openapi-server-url-templating)
[![npm](https://img.shields.io/npm/dm/openapi-server-url-templating.svg)](https://www.npmjs.com/package/openapi-server-url-templating)
[![Test workflow](https://github.com/swaggerexpert/openapi-server-url-templating/actions/workflows/test.yml/badge.svg)](https://github.com/swaggerexpert/openapi-server-url-templating/actions)
[![Dependabot enabled](https://img.shields.io/badge/Dependabot-enabled-blue.svg)](https://dependabot.com/)
[![try on RunKit](https://img.shields.io/badge/try%20on-RunKit-brightgreen.svg?style=flat)](https://npm.runkit.com/openapi-server-url-templating)
[![Tidelift](https://tidelift.com/badges/package/npm/openapi-server-url-templating)](https://tidelift.com/subscription/pkg/npm-openapi-server-url-templating?utm_source=npm-openapi-server-url-templating&utm_medium=referral&utm_campaign=readme)

[Server URL Templating](https://spec.openapis.org/oas/v3.1.1.html#server-object) supports [Server Variables](https://spec.openapis.org/oas/v3.1.1.html#server-variable-object). Variable substitutions will be made when a variable is named in `{`brackets`}`.
This mechanism is used by [Server Object](https://spec.openapis.org/oas/v3.1.1.html#server-object)
of [OpenAPI specification](https://spec.openapis.org/).

`openapi-server-url-templating` is a **parser**, **validator** and **substitution mechanism** for OpenAPI Server URL Templating,
which played a [foundational role](https://github.com/OAI/OpenAPI-Specification/pull/4264) in defining the official ANBF grammar for Server URL Templating.

It supports Server Object URL Templating defined in following OpenAPI specification versions:

- [OpenAPI 3.0.0](https://spec.openapis.org/oas/v3.0.0.html)
- [OpenAPI 3.0.1](https://spec.openapis.org/oas/v3.0.1.html)
- [OpenAPI 3.0.2](https://spec.openapis.org/oas/v3.0.2.html)
- [OpenAPI 3.0.3](https://spec.openapis.org/oas/v3.0.3.html)
- [OpenAPI 3.0.4](https://spec.openapis.org/oas/v3.0.4.html)
- [OpenAPI 3.1.0](https://spec.openapis.org/oas/v3.1.0.html)
- [OpenAPI 3.1.1](https://spec.openapis.org/oas/v3.1.1.html)

<table>
  <tr>
    <td align="right" valign="middle">
        <img src="https://cdn2.hubspot.net/hubfs/4008838/website/logos/logos_for_download/Tidelift_primary-shorthand-logo.png" alt="Tidelift" width="60" />
      </td>
      <td valign="middle">
        <a href="https://tidelift.com/subscription/pkg/npm-openapi-server-url-templating?utm_source=npm-openapi-server-url-templating&utm_medium=referral&utm_campaign=readme">
            Get professionally supported openapi-server-url-templating with Tidelift Subscription.
        </a>
      </td>
  </tr>
</table>

## Table of Contents

- [Getting started](#getting-started)
  - [Installation](#installation)
  - [Usage](#usage)
    - [Parsing](#parsing)
    - [Validation](#validation)
    - [Substitution](#substitution)
    - [Grammar](#grammar)
- [More about OpenAPI Server URL Templating](#more-about-openapi-server-url-templating)
- [License](#license)

## Getting started

### Installation

You can install `openapi-server-url-templating` using `npm`:

```sh
 $ npm install openapi-server-url-templating
```

### Usage

`openapi-server-url-templating` currently supports **parsing**, **validation** and **substitution**.
Both parser and validator are based on a superset of [ABNF](https://www.rfc-editor.org/rfc/rfc5234) ([SABNF](https://github.com/ldthomas/apg-js2/blob/master/SABNF.md))
and use [apg-lite](https://github.com/ldthomas/apg-lite) parser generator.

#### Parsing

Parsing a Server URL Templating is as simple as importing the **parse** function
and calling it.

```js
import { parse } from 'openapi-server-url-templating';

const parseResult = parse('https://{username}.gigantic-server.com:{port}/{basePath}');
parseResult.result.success; // => true
```

**parseResult** variable has the following shape:

```
{
  result: {
    success: true,
    state: 101,
    stateName: 'MATCH',
    length: 56,
    matched: 56,
    maxMatched: 56,
    maxTreeDepth: 12,
    nodeHits: 758
  },
  ast: fnast {
    callbacks: [
      'server-url-template': [Function: serverUrlTemplate],
      'server-variable': [Function: serverVariable],
      'server-variable-name': [Function: serverVariableName],
      literals: [Function: literals]
    ],
    init: [Function (anonymous)],
    ruleDefined: [Function (anonymous)],
    udtDefined: [Function (anonymous)],
    down: [Function (anonymous)],
    up: [Function (anonymous)],
    translate: [Function (anonymous)],
    setLength: [Function (anonymous)],
    getLength: [Function (anonymous)],
    toXml: [Function (anonymous)]
  }
}
```

###### Interpreting AST as list of entries

```js
import { parse } from 'openapi-server-url-templating';

const parseResult = parse('https://{username}.gigantic-server.com:{port}/{basePath}');
const parts = [];

parseResult.ast.translate(parts);
```

After running the above code, **parts** variable has the following shape:

```js
[
  [
    'server-url-template',
    'https://{username}.gigantic-server.com:{port}/{basePath}'
  ],
  [ 'literals', 'https://' ],
  [ 'server-variable', '{username}' ],
  [ 'server-variable-name', 'username' ],
  [ 'literals', '.gigantic-server.com:' ],
  [ 'server-variable', '{port}' ],
  [ 'server-variable-name', 'port' ],
  [ 'literals', '/' ],
  [ 'server-variable', '{basePath}' ],
  [ 'server-variable-name', 'basePath' ]
]
```

###### Interpreting AST as XML

```js
import { parse } from 'openapi-server-url-templating';

const parseResult = parse('https://{username}.gigantic-server.com:{port}/{basePath}');
const xml = parseResult.ast.toXml();
```

After running the above code, **xml** variable has the following content:

```xml
<?xml version="1.0" encoding="utf-8"?>
<root nodes="10" characters="56">
  <!-- input string -->
  https://{username}.gigantic-server.com:{port}/{basePath}
  <node name="server-url-template" index="0" length="56">
    https://{username}.gigantic-server.com:{port}/{basePath}
    <node name="literals" index="0" length="8">
      https://
    </node><!-- name="literals" -->
    <node name="server-variable" index="8" length="10">
      {username}
      <node name="server-variable-name" index="9" length="8">
        username
      </node><!-- name="server-variable-name" -->
    </node><!-- name="server-variable" -->
    <node name="literals" index="18" length="21">
      .gigantic-server.com:
    </node><!-- name="literals" -->
    <node name="server-variable" index="39" length="6">
      {port}
      <node name="server-variable-name" index="40" length="4">
        port
      </node><!-- name="server-variable-name" -->
    </node><!-- name="server-variable" -->
    <node name="literals" index="45" length="1">
      /
    </node><!-- name="literals" -->
    <node name="server-variable" index="46" length="10">
      {basePath}
      <node name="server-variable-name" index="47" length="8">
        basePath
      </node><!-- name="server-variable-name" -->
    </node><!-- name="server-variable" -->
  </node><!-- name="server-url-template" -->
</root>
```

> NOTE: AST can also be traversed in classical way using [depth first traversal](https://www.tutorialspoint.com/data_structures_algorithms/depth_first_traversal.htm). For more information about this option please refer to [apg-js](https://github.com/ldthomas/apg-js) and [apg-js-examples](https://github.com/ldthomas/apg-js-examples).

#### Validation

Validating a Server URL Templating is as simple as importing the **test** function and calling it.


```js
import { test } from 'openapi-server-url-templating';

test('https://{username}.gigantic-server.com:{port}/{basePath}'); // => true
test('https://gigantic-server.com/base-path'); // => true
test('https://gigantic-server.com/base-path', { strict: true }); // => false (doesn't contain any server-variable)
```

#### Substitution

Performing Server URL template substitution is as simple as importing the **substitute** function and calling it.

```js
import { substitute } from 'openapi-server-url-templating';

subtitute('https://{username}.gigantic-server.com', { username: 'char0n' }); // => "https://char0n.gigantic-server.com"
```

Substituted Server URL Templating is automatically encoded using [encodeURIComponent](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURIComponent) function.
It is possible to provide a custom encoder.

```js
import { substitute } from 'openapi-server-url-templating';

substitute('https://{username}.gigantic-server.com', { username: '/?#' }, {
  encoder: (serverVariable) => serverVariable, // no encoding
}); // => "https:///?#.gigantic-server.com"
```

#### Grammar

New grammar instance can be created in following way:

```js
import { Grammar } from 'openapi-server-url-templating';

const grammar = new Grammar();
```

To obtain original ABNF (SABNF) grammar as a string:

```js
import { Grammar } from 'openapi-server-url-templating';

const grammar = new Grammar();

grammar.toString();
// or
String(grammar);
```

## More about OpenAPI Server URL Templating

The Server URL Templating is defined by the following [ABNF](https://tools.ietf.org/html/rfc5234) syntax

```abnf
; OpenAPI Server URL templating ABNF syntax
server-url-template    = 1*( literals / server-variable ) ; variant of https://www.rfc-editor.org/rfc/rfc6570#section-2
server-variable        = "{" server-variable-name "}"
server-variable-name   = 1*( %x00-7A / %x7C / %x7E-10FFFF ) ; every UTF8 character except { and } (from OpenAPI)

; https://www.rfc-editor.org/rfc/rfc6570#section-2.1
; https://www.rfc-editor.org/errata/eid6937
literals               = 1*( %x21 / %x23-24 / %x26-3B / %x3D / %x3F-5B
                       / %x5D / %x5F / %x61-7A / %x7E / ucschar / iprivate
                       / pct-encoded)
                            ; any Unicode character except: CTL, SP,
                            ;  DQUOTE, "%" (aside from pct-encoded),
                            ;  "<", ">", "\", "^", "`", "{", "|", "}"

; https://www.rfc-editor.org/rfc/rfc6570#section-1.5
DIGIT          =  %x30-39             ; 0-9
HEXDIG         =  DIGIT / "A" / "B" / "C" / "D" / "E" / "F" ; case-insensitive

pct-encoded    =  "%" HEXDIG HEXDIG

ucschar        =  %xA0-D7FF / %xF900-FDCF / %xFDF0-FFEF
               /  %x10000-1FFFD / %x20000-2FFFD / %x30000-3FFFD
               /  %x40000-4FFFD / %x50000-5FFFD / %x60000-6FFFD
               /  %x70000-7FFFD / %x80000-8FFFD / %x90000-9FFFD
               /  %xA0000-AFFFD / %xB0000-BFFFD / %xC0000-CFFFD
               /  %xD0000-DFFFD / %xE1000-EFFFD

iprivate       =  %xE000-F8FF / %xF0000-FFFFD / %x100000-10FFFD
```

## License

`openapi-server-url-templating` is licensed under [Apache 2.0 license](https://github.com/swaggerexpert/openapi-server-url-templating/blob/main/LICENSE).
`openapi-server-url-templating` comes with an explicit [NOTICE](https://github.com/swaggerexpert/openapi-server-url-templating/blob/main/NOTICE) file
containing additional legal notices and information.
 readmeEtag: '"10de8be9d70e6837a9da15ee2efee0d1e6a2196e"' readmeLastModified: Sat, 28 Dec 2024 13:39:26 GMT repositoryId: 809665855 description: >- OpenAPI Server URL templating parser, validator and substitution mechanism. created: '2024-06-03T08:05:16Z' updated: '2026-01-21T22:55:39Z' language: JavaScript archived: false stars: 4 watchers: 1 forks: 1 owner: swaggerexpert logo: https://avatars.githubusercontent.com/u/172408630?v=4 license: Apache-2.0 repoEtag: '"6e42a30e31230f22eff3526ca1935c4af74dd01f802284634ed9f8403acc0b9e"' repoLastModified: Wed, 21 Jan 2026 22:55:39 GMT category: Parsers foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/gopher-fleece/gleece v3: true v3_1: true id: a64dbea139404dd2c8731cb92e92bcaa repositoryMetadata: base64Readme: >- **Gleece** - Bringing joy and ease to API development in Go! 🚀   

<!-- Source code health & info -->
[![gleece](https://github.com/gopher-fleece/gleece/actions/workflows/build.yml/badge.svg?branch=main)](https://github.com/gopher-fleece/gleece/actions/workflows/build.yml)
[![Go Report Card](https://goreportcard.com/badge/github.com/gopher-fleece/gleece)](https://goreportcard.com/report/gopher-fleece/gleece)
[![Coverage Status](https://coveralls.io/repos/github/gopher-fleece/gleece/badge.svg?branch=main)](https://coveralls.io/github/gopher-fleece/gleece?branch=main)
[![Go Version](https://img.shields.io/github/go-mod/go-version/gopher-fleece/gleece)](https://github.com/gopher-fleece/gleece/blob/main/go.mod)

<!-- Packages, Releases etc -->
[![VSCode Extension](https://img.shields.io/visual-studio-marketplace/v/haim-kastner.gleece-extension?label=VSCode%20Extension)](https://marketplace.visualstudio.com/items?itemName=haim-kastner.gleece-extension)
<a href="https://docs.gleece.dev">
    <img src="https://img.shields.io/badge/docs-gleece.dev-blue" alt="Documentation">
</a>
[![LibHunt - Gleece](https://img.shields.io/badge/LibHunt-Gleece-blue)](https://www.libhunt.com/r/gleece)
[![Latest Release](https://img.shields.io/github/v/release/gopher-fleece/gleece)](https://github.com/gopher-fleece/gleece/releases)
[![Go Reference](https://pkg.go.dev/badge/github.com/gopher-fleece/gleece.svg)](https://pkg.go.dev/github.com/gopher-fleece/gleece)

<!-- Supported standards -->
[![OpenAPI 3.0](https://img.shields.io/badge/OpenAPI-3.0.0-green.svg)](https://spec.openapis.org/oas/v3.0.0)
[![OpenAPI 3.1](https://img.shields.io/badge/OpenAPI-3.1.0-green.svg)](https://spec.openapis.org/oas/v3.1.0)

<!-- Supported frameworks -->
[![Gin Support](https://img.shields.io/badge/Gin-Supported-blue)](https://gin-gonic.com/)
[![Echo Support](https://img.shields.io/badge/Echo-Supported-blue)](https://echo.labstack.com/)
[![Gorilla Mux Support](https://img.shields.io/badge/Gorilla_Mux-Supported-blue)](https://github.com/gorilla/mux)
[![Chi Support](https://img.shields.io/badge/Chi-Supported-blue)](https://github.com/go-chi/chi)
[![Fiber Support](https://img.shields.io/badge/Fiber-Supported-blue)](https://github.com/gofiber/fiber)

<!-- Social -->
[![GitHub stars](https://img.shields.io/github/stars/gopher-fleece/gleece.svg?style=social&label=Stars)](https://github.com/gopher-fleece/gleece/stargazers) 
[![License](https://img.shields.io/github/license/gopher-fleece/gleece.svg?style=social)](https://github.com/gopher-fleece/gleece/blob/master/LICENSE)


---

## About  
Developing APIs doesn’t have to be a chore - it should be simple, efficient, and enjoyable.  

Gone are the days of manually writing repetitive boilerplate code or struggling to keep your API documentation in sync with your implementation.


With Gleece, you can:  
- **Simplify and standardize**  your API development process.  
- Automatically **generate OpenAPI v3.0.0 / v3.1.0** specifications directly from your code.  
- Ensure your APIs are always **up-to-date, well-documented and consistent**.  
- Effortlessly **validate input data** to ensure your APIs are reliable and secure.
- Adopt a **security first** approach with simple yet powerful authorization flows.
- **Customize behavior** to your exact needs by extending or overriding the routes templates.
- Bring your own router - works seamlessly with **Gin, Echo, Gorilla Mux, Chi, & Fiber**.

Gleece aims to make Go developers’ lives easier by seamlessly integrating API routing, validation, and documentation into a single cohesive workflow **without sacrificing performance**.

For more information, check out the [concept](https://docs.gleece.dev/docs/about/concept) section of our documentation


## Documentation

Explore the complete [documentation site](https://docs.gleece.dev/docs/intro) to get started quickly.  

You can also check out our [example project](https://github.com/gopher-fleece/gleecexample#readme), which demonstrates how to set up and use Gleece in a real-world scenario. This hands-on reference will help you integrate Gleece seamlessly.

If you've encountered a gap in the documentation or feel some topic is inadequately covered, please let us know via [GitHub Issues](https://github.com/gopher-fleece/gleece/issues)


## Under the hood

*Gleece* is almost exclusively a **design/build time** framework/transpiler- there are no magic tricks or runtime shenanigans.

Once your API implementation is ready, use *Gleece*'s CLI to generate routes according to your chosen engine.

These generated routes work just like any manually written code and need to be registered to your engine's router instance through your application code.

You can continue to use your engine's native calls, middlewares, and other features as before - *Gleece*'s generated code acts only as a complementary plugin to your router.

For a deeper look into *Gleece*'s internals, please see the [architecture](https://docs.gleece.dev/docs/about/architecture) section of our documentation.


## Look & Feel  

Here’s a practical snippet of how Gleece simplifies your API development:  

![Screenshot](https://raw.githubusercontent.com/gophar-fleece/.github/main/docs/screenshots/usage-example.png)

All other aspects, including HTTP routing generation, authorization enforcement, payload validation, error handling, and OpenAPI v3 specification generation, are handled by Gleece.

## Visual Studio Code Extension

To enhance your development experience with Gleece, we provide an official Visual Studio Code extension that provides intelligent annotation highlighting and improved code visibility.

For more information and capabilities see the [Gleece VS Code Extension](https://github.com/gopher-fleece/gleece-vscode-extension#readme).

To install it search `Gleece` in the "Extension" tab or go to the [Visual Studio Marketplace](https://marketplace.visualstudio.com/items?itemName=haim-kastner.gleece-extension).

## Our Initiative

We believe that API development should be code-first, focusing on the developer experience while maintaining industry standards. Coming from the TypeScript ecosystem, we were inspired by frameworks like [TSOA](https://github.com/lukeautry/tsoa) that handle everything from routing and validation to OpenAPI generation - all from a single source of truth: your code.

Read more about our initiative and development philosophy:
* [Haim Kastner's Gleece Project blog post](https://blog.castnet.club/en/blog/gleece-project)

* [A Holistic View on APIs as an Ecosystem](https://zuplo.com/learning-center/holistic-view-of-apis) in *Zuplo*'s learning center


## Security

To report security concerns or vulnerabilities, please follow the instructions in our [Security Policy](./SECURITY.md).

## Contact

For general inquiries or feedback, please reach out to us at [contact@gleece.dev](mailto:contact@gleece.dev).

Anyway, we'd love to hear from you, especially if you or your company use Gleece.

## Help us grow!

*Gleece* is a two-man passion project and requires enormous efforts;

If you find it interesting or useful, please help us grow by dropping a star or telling your colleagues about us!

Your support keeps us going!

---

## License  
Gleece is licensed under the [MIT LICENSE](./LICENSE).  readmeEtag: '"79015fe4fee129f564b50a7e7fa154d045af4126"' readmeLastModified: Mon, 22 Sep 2025 16:17:48 GMT repositoryId: 913709084 description: >- Building, documenting, validating, securing and customizing REST APIs through code-first development created: '2025-01-08T07:53:15Z' updated: '2026-01-12T02:49:11Z' language: Go archived: false stars: 82 watchers: 2 forks: 1 owner: gopher-fleece logo: https://avatars.githubusercontent.com/u/195629235?v=4 license: MIT repoEtag: '"5ca107a84980dec11081010c6831fdf60d4b41e6b4eaeb341184fee66c26b2af"' repoLastModified: Mon, 12 Jan 2026 02:49:11 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/codecentric/spring-boot-api-first v3: true repositoryMetadata: base64Readme: >- ICMgQVBJIGZpcnN0IFNwcmluZyBCb290IFNlcnZpY2UgKFdJUCkKCiBUaGlzIFNwcmluZyBCb290IFNlcnZpY2Ugd2lsbCBiZSBjcmVhdGVkIGJ5IHRoZSBBUEkgZmlyc3QgYXBwcm9hY2ggYW5kIHRlc3RlZCB3aXRoIFtLYXJhdGVdKGh0dHBzOi8vaW50dWl0LmdpdGh1Yi5pby9rYXJhdGUvKS4gRHVlIHRvIG1pc3NpbmcgY2FwYWJpbGl0eSBvZiBPcGVuQVBJIEdlbmVyYXRvciBpbiBWZXJzaW9uIDUuMC4wIHRoZSBkaXJlY3QgZG93bmxvYWQgb2YgdGhlIEFQSSBzcGVjIGlzIG5vdCBwb3NzaWJsZSBhdCB0aGUgbW9tZW50LiBTbyB3ZSB3aWxsIHJlbHkgb24gdXNpbmcgW0dyYWRsZSBEb3dubG9hZCBUYXNrXShodHRwczovL2dpdGh1Yi5jb20vbWljaGVsLWtyYWVtZXIvZ3JhZGxlLWRvd25sb2FkLXRhc2spLiBUaGVyZSBpcyBhbHNvIGFscmVhZHkgYSBbYnVnIHJlcG9ydF0oaHR0cHM6Ly9naXRodWIuY29tL09wZW5BUElUb29scy9vcGVuYXBpLWdlbmVyYXRvci9pc3N1ZXMvODI1NSkgd3JpdHRlbiBieSBbam9zY2hpXShodHRwczovL2dpdGh1Yi5jb20vam9zY2hpKS4gQW5kIHRoZXJlIGlzIGFuIG9wZW4gW2ZlYXR1cmUgcmVxdWVzdF0oaHR0cHM6Ly9naXRodWIuY29tL09wZW5BUElUb29scy9vcGVuYXBpLWdlbmVyYXRvci9pc3N1ZXMvOTA4MykgZm9yIE9wZW5BUEkgR2VuZXJhdG9yIHRvIHN1cHBvcnQgT3BlbkFQSSBTcGVjIDMuMS4gCiBEdXJpbmcgdGhlIHdyaXRpbmcgcHJvY2VzcyBPcGVuQVBJIEdlbmVyYXRvciB3YXMgcmVtb3ZlZCBhbmQgcmVwbGFjZWQgW09wZW5BUEkgUHJvY2Vzc29yXShodHRwczovL2RvY3Mub3BlbmFwaXByb2Nlc3Nvci5pby9vYXAvaG9tZS9ob21lLmh0bWwpLg== readmeEtag: '"d0c78f8a85e68b7aff4392e02cf88c1f6fa22c47"' readmeLastModified: Tue, 06 Jul 2021 09:48:42 GMT repositoryId: 324130901 description: Example project using Spring Boot and OpenAPI Processor created: '2020-12-24T10:34:38Z' updated: '2025-10-08T00:55:32Z' language: Java archived: false stars: 3 watchers: 1 forks: 4 owner: codecentric logo: https://avatars.githubusercontent.com/u/1009716?v=4 repoEtag: '"2148e7b2826b8abcd86f78cca4c50fd7fe1b1f60e54caefa6e8020506ca0196c"' repoLastModified: Wed, 08 Oct 2025 00:55:32 GMT foundInMaster: true category: SDK id: 36f5dbfc3daf547fbd1460be064da28d - source: openapi3 tags repository: https://github.com/zchee/openapi2protobuf v3: true id: f1ab0225162b832a64bce243ce55a302 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpMnByb3RvYnVmCgpbIVtMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL2dvLWxhbmd1YWdlLXNlcnZlci9vcGVuYXBpMnByb3RvYnVmP2NvbG9yPWJsdWUmbG9nbz1zcGR4JmxvZ29Db2xvcj0lMjM1QTk2Qzgmc3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vc3BkeC5vcmcvbGljZW5zZXMvQlNELTMtQ2xhdXNlLmh0bWwpCgpvcGVuYXBpMnByb3RvYnVmIGdlbmVyYXRlcyBQcm90b2NvbCBCdWZmZXJzIHYzIGFuZCBnUlBDIHNlcnZpY2VzIGRlZmluaXRpb25zIGZyb20gdGhlIE9wZW5BUEkvU3dhZ2dlciBzY2hlbWEuCg== readmeEtag: '"81277246368f666f71fadfbc3ca097b5df95e841"' readmeLastModified: Fri, 06 Jan 2023 11:39:32 GMT repositoryId: 469262087 description: >- openapi2protobuf generates Protocol Buffers v3 schema and gRPC service definitions from OpenAPI/Swagger schema definitions created: '2022-03-13T03:43:19Z' updated: '2024-06-28T05:49:16Z' language: Go archived: false stars: 3 watchers: 1 forks: 0 owner: zchee logo: https://avatars.githubusercontent.com/u/6366270?v=4 license: BSD-3-Clause repoEtag: '"7095b667f1c229813972dbf7fb2360974d020ecb45ef4eda348c5b1ba80cff85"' repoLastModified: Fri, 28 Jun 2024 05:49:16 GMT category: Parsers oldLocations: - https://github.com/go-language-server/openapi2protobuf foundInMaster: true - source: openapi3 tags repository: https://github.com/moochub/schema v3: true repositoryMetadata: base64Readme: >- IyBNT09DaHViIFNjaGVtYQoKVGhpcyByZXBvc2l0b3J5IGNvbnRhaW5zIHNwZWNpZmljYXRpb24gZmlsZXMgZm9yIGRhdGEgZXhjaGFuZ2UgYmV0d2VlbiBNT09DIHByb3ZpZGVycyBwYXJ0aWNpcGF0aW5nIGluIHRoZSBbTU9PQ2h1Yl0oaHR0cHM6Ly9tb29jaHViLm9yZykuIEJhc2VkIG9uIHByZXZpb3VzIHdvcmsgcmVnYXJkaW5nIFtNT09DIHN0YW5kYXJkc10oaHR0cHM6Ly9naXRodWIuY29tL29wZW5IUEkvbW9vYy1zdGFuZGFyZHMpLCB0aGUgSlNPTiBmb3JtYXQgc3BlY2lmaWVkIGVuYWJsZXMgYSBzZWFtbGVzcyBleGNoYW5nZSBvZiBjb3Vyc2UgaW5mb3JtYXRpb24sIGluc3RydWN0b3JzLCBhbmQgdGhlIGluc3RpdHV0ZXMgb2ZmZXJpbmcgYSBjb3Vyc2UuIE1pbmltYWwgaW5mb3JtYXRpb24gaXMgcmVxdWlyZWQgcGVyIGNvdXJzZSB0byBlYXNlIG9uYm9hcmRpbmcgZm9yIG5ldyBwYXJ0bmVycyB3aGlsZSBhbGxvd2luZyBjdXN0b21pemFiaWxpdHkgdG8gZml0IGluZGl2aWR1YWwgbmVlZHMuCgojIyBTcGVjaWZpY2F0aW9uCgpUaGUgTU9PQ2h1YiBBUEkgc3BlY2lmaWNhdGlvbiBjb25zaXN0cyBvZiB0d28gbWFqb3IgcGFydHM6ICgxKSBBIEpTT04gZm9ybWF0IGFuZCAoMikgYW4gQVBJIHZlcnNpb25pbmcgY29uY2VwdC4KCjEuIEpTT04gZm9ybWF0CgogICBUaGUgSlNPTiBmb3JtYXQgc3BlY2lmaWVkIGluIHRoaXMgcmVwb3NpdG9yeSBpcyBiYXNlZCBvbiB0aGUgW0pTT046QVBJXShodHRwczovL2pzb25hcGkub3JnKSBzY2hlbWEgYW5kIHVzZXMgdGhlaXIgcHJvcG9zZWQgcGFnaW5hdGlvbiBhcHByb2FjaC4gSW4gYSB2YWxpZCByZXNwb25zZSwgb25seSBhIGZldyBhdHRyaWJ1dGVzIG11c3QgYmUgcHJlc2VudCwgbW9zdCBjYW4gYmUgb21pdHRlZCBvciBjYW4gYmUgYG51bGxgLiBBcyB3ZSBzZWUgYSBzZW1hbnRpY2FsIGRpZmZlcmVuY2UgYmV0d2VlbiBhbiBvbWl0dGVkIGFuZCBhIGBudWxsYCB2YWx1ZSwgd2Uga2luZGx5IGFzayB5b3UgdG8gcHJvdmlkZSBhcyBtdWNoIGluZm9ybWF0aW9uIGFzIHBvc3NpYmxlLgoKMi4gQVBJIHZlcnNpb25pbmcgY29uY2VwdAoKICAgVXBvbiByZXF1ZXN0IG9mIHRoZSBBUEksIHRoZSBzZXJ2ZXIgYW5kIGNsaWVudCBtaWdodCBuZWdvdGlhdGUgdGhlIG1vc3Qgc3VpdGFibGUgdmVyc2lvbiBvZiB0aGUgQVBJIHVzaW5nIEhUVFAgaGVhZGVycy4gUGxlYXNlIHJlZmVyIHRvIHRoZSBbZGVkaWNhdGVkIHZlcnNpb25pbmcgc3BlY2lmaWNhdGlvbl0obW9vY2h1Yi12ZXJzaW9uaW5nLm1kKSBmb3IgbW9yZSBkZXRhaWxzLgoKIyMgUmVsZWFzZXMKClRoZSBbbGF0ZXN0IGZpbmFsaXplZCB2ZXJzaW9uXShodHRwczovL2dpdGh1Yi5jb20vbW9vY2h1Yi9zY2hlbWEvcmVsZWFzZXMvbGF0ZXN0KSBvZiB0aGUgc2NoZW1hIGlzIHJlbGVhc2VkIGluIHRoaXMgcmVwb3NpdG9yeSBhbmQgdGFnZ2VkIGFwcHJvcHJpYXRlbHkuIEFsbCBNT09DaHViIGNvbXBsaWFudCBzb2Z0d2FyZSBpcyByZXF1ZXN0ZWQgdG8gaW1wbGVtZW50IHRoZSB2ZXJzaW9uaW5nIGFzIG91dGxpbmVkIGluIHRoZSBbZGVkaWNhdGVkIEFQSSB2ZXJzaW9uaW5nIHNwZWNpZmljYXRpb25dKG1vb2NodWItdmVyc2lvbmluZy5tZCkuIFdlIGZvbGxvdyBhIFtzZW1hbnRpYyB2ZXJzaW9uaW5nXShodHRwczovL3NlbXZlci5vcmcpIGFwcHJvYWNoIGZvciBhbGwgcmVsZWFzZXMgdG8gZWFzZSBjb25zdW1pbmcgc2NoZW1hLWNvbXBsaWFudCBBUElzLgoKIyMgQ29udHJpYnV0aW9ucwoKVGhlIHNjaGVtYSBpcyBtYWlubHkgZHJpdmVuIGJ5IHRoZSBbTU9PQ2h1YiBvcmdhbml6YXRpb24gYW5kIHRoZWlyIHBhcnRuZXJzXShodHRwczovL21vb2NodWIub3JnL3BhcnRuZXJzLykuIFRoZXJlZm9yZSwgaXQgaXMgZGVzaWduZWQgdG8gcHJvdmlkZSBhIGNvbnNpc3RlbnQgYW5kIHVuaWZvcm0gbWVjaGFuaXNtIHRvIHRyYW5zcG9ydCByZXF1aXJlZCBpbmZvcm1hdGlvbiBiZXR3ZWVuIHRoZSBwYXJ0bmVycy4gV2Ugd2lsbCBldm9sdmUgdGhlIHNjaGVtYSBiYXNlZCBvbiByZXF1aXJlbWVudHMgaW50cm9kdWNlZCBieSBwYXJ0bmVycyBhbmQgY29uc3VtZXJzLiBBZGRpdGlvbmFsIGNvbnRyaWJ1dGlvbnMgdG8gdGhlIHNjaGVtYSBhcmUgd2VsY29tZSBhbmQgd2lsbCBiZSBjb25zaWRlcmVkIGZvciBhbiB1cGNvbWluZyByZWxlYXNlLiBQbGVhc2UgY3JlYXRlIGFuIGlzc3VlIG9yIGEgcHVsbCByZXF1ZXN0IHRvIHN0YXJ0IGNvbnRyaWJ1dGluZyEKCiMjIENvbnRhY3QKCklmIHlvdSBoYXZlIGZ1cnRoZXIgcXVlc3Rpb25zIG9yIHdvdWxkIGxpa2UgdG8gZ2V0IGxpc3RlZCBpbiB0aGUgW01PT0NodWJdKGh0dHBzOi8vbW9vY2h1Yi5vcmcpIHVzaW5nIGFuIGVuZHBvaW50IGZvbGxvd2luZyB0aGlzIHNjaGVtYSwgcGxlYXNlIGRyb3AgdXMgYSBtYWlsIGF0IG9mZmljZUBtb29jaHViLm9yZy4K readmeEtag: '"0b446e49b84e772b5555d586de201d8b311528b8"' readmeLastModified: Tue, 18 Jun 2024 14:40:56 GMT repositoryId: 343502074 description: >- A cross-platform schema definition used for data exchange between cooperating MOOC providers created: '2021-03-01T17:34:18Z' updated: '2025-12-11T14:49:09Z' language: null archived: false stars: 3 watchers: 4 forks: 3 owner: MOOChub logo: https://avatars.githubusercontent.com/u/79694863?v=4 license: MIT repoEtag: '"b8ee8ef1a52b077b3a21e388ebb2189f7a7e7606b53a3cf36640fba4a76fbd3a"' repoLastModified: Thu, 11 Dec 2025 14:49:09 GMT foundInMaster: true category: Parsers id: f82539087f07461ed2a41b256e61e762 - source: openapi3 tags repository: https://github.com/postman-solutions-eng/aippealing-companies-template v3: true id: dc22dd446e2db39aefb9053184a362f0 repositoryMetadata: base64Readme: >- # A(i)ppealing Companies :office: :arrow_right: :robot: :arrow_right: :chocolate_bar: :car: :framed_picture: 

[API](https://auto-demo.postmansolutions.com) to create appealing images of company products made of chocolate, gold or lego.
Reference implementation is based on OpenAI (DALL-E, GPT-3).

[![Run in Postman](https://run.pstmn.io/button.svg)](https://app.getpostman.com/run-collection/24483733-cc806d3a-75bf-4025-bfb1-b399b10ba03d?action=collection%2Ffork&collection-url=entityId%3D24483733-cc806d3a-75bf-4025-bfb1-b399b10ba03d%26entityType%3Dcollection%26workspaceId%3D210e7615-f69f-4322-a87f-1d8135fe05bd#?env%5BPublicMock%5D=W3sia2V5IjoiYmFzZVVybCIsInZhbHVlIjoiaHR0cHM6Ly9lYjIwMmRkYi1kMTNmLTRiZjctYmFjYS1hYWVhMmE3NWEwM2EubW9jay5wc3Rtbi5pbyIsImVuYWJsZWQiOnRydWUsInR5cGUiOiJkZWZhdWx0Iiwic2Vzc2lvblZhbHVlIjoiaHR0cHM6Ly9lYjIwMmRkYi1kMTNmLTRiZjctYmFjYS1hYWVhMmE3NWEwM2EubW9jay5wc3Rtbi5pbyIsInNlc3Npb25JbmRleCI6MH0seyJrZXkiOiJPcGVuQUlUb2tlbiIsInZhbHVlIjoiPHlvdSBkb24ndCBuZWVkIGFueSByZWFsIHRva2VuIGZvciB0aGUgbW9jayBzZXJ2ZXI+IiwiZW5hYmxlZCI6dHJ1ZSwidHlwZSI6InNlY3JldCIsInNlc3Npb25WYWx1ZSI6Im1vY2stdG9rZW4iLCJzZXNzaW9uSW5kZXgiOjF9XQ==)

[![Postman Governance checks on branch](https://github.com/jonico/aippealing-companies-impl-openai/actions/workflows/governance-checks-branch.yml/badge.svg)](https://github.com/jonico/aippealing-companies-impl-openai/actions/workflows/governance-checks-branch.yml)

[![Integration Tests Production](https://github.com/jonico/aippealing-companies-impl-openai/actions/workflows/integration-tests-production.yml/badge.svg)](https://github.com/jonico/aippealing-companies-impl-openai/actions/workflows/integration-tests-production.yml)



## Motivation

There are some amazing, visually appealing company products made of chocolate or lego

<img width="197" alt="image" src="https://user-images.githubusercontent.com/1872314/208320977-06e18d27-4938-4be3-ab9d-8b2ffc2c85e1.png"> <img width="239" alt="image" src="https://user-images.githubusercontent.com/1872314/208321054-130afbaa-f7db-453e-8db1-114ebce97f4e.png"><img src="https://user-images.githubusercontent.com/1872314/213943475-a74c1b47-20bc-4761-8d73-5e4d052fefc3.png" alt="Soccer%20Stadium%20made%20of%20Lego" width="350">

Inspired by the image and text generation capabilities of [OpenAI](https://www.postman.com/devrel/workspace/openai/documentation/13183464-90abb798-cb85-43cb-ba3a-ae7941e968da), this API generates images of the products, companies are most known for, and turns them into gold, chocolate, lego or any other material.

#### Volkswagen / Porsche

<img width="200" alt="image" src="https://user-images.githubusercontent.com/1872314/208321137-0ab3cdb9-a2d2-47ad-9f9e-172188a848d5.png"> <img width="200" alt="image" src="https://user-images.githubusercontent.com/1872314/208321209-9a82e2b5-47aa-4d9d-a267-1659e2964b0b.png">


#### Allianz

<img width="200" alt="image" src="https://user-images.githubusercontent.com/1872314/208321233-117ba82b-1c3b-4bab-860a-3fc94b9df407.png"> <img width="200" alt="image" src="https://user-images.githubusercontent.com/1872314/208321469-9a3d2c0a-ced0-4568-b61e-45f5e46f4524.png">


### Mercedes-Benz

<img width="200" alt="image" src="https://user-images.githubusercontent.com/1872314/208321375-19d9198a-858d-4fc9-b95c-6a31fe58096a.png"> <img width="200" alt="image" src="https://user-images.githubusercontent.com/1872314/208321413-806d2afc-d3fb-4660-9299-7ca86be90e2f.png">

## Trying it out :office: :arrow_right: :robot: :arrow_right: :chocolate_bar: :car: :framed_picture:  :tada: 

Follow the [generated API documentation](https://auto-demo.postmansolutions.com) and click the "Run in Postman" button

<img width="1290" alt="image" src="https://user-images.githubusercontent.com/1872314/208407634-c0a48387-56cc-44f1-9c16-59041a32e903.png">

# OpenAPI Generated JavaScript/Express Server

## Overview
This server was generated using the [OpenAPI Generator](https://openapi-generator.tech) project.  The code generator, and it's generated code allows you to develop your system with an API-First attitude, where the API contract is the anchor and definer of your project, and your code and business-logic aims to complete and comply to the terms in the API contract.

### prerequisites
- NodeJS >= 10.6
- NPM >= 6.10.0

The code was written on a mac, so assuming all should work smoothly on Linux-based computers. However, there is no reason not to run this library on Windows-based machines. If you find an OS-related problem, please open an issue and it will be resolved.

### Running the server
#### This is a long read, but there's a lot to understand. Please take the time to go through this.
1. Use the OpenAPI Generator to generate your application:
Assuming you have Java (1.8+), and [have the jar](https://github.com/openapitools/openapi-generator#13---download-jar) to generate the application, run:
```java -jar {path_to_jar_file} generate -g nodejs-express-server -i {openapi yaml/json file} -o {target_directory_where_the_app_will_be_installed} ```
If you do not have the jar, or do not want to run Java from your local machine, follow instructions on the [OpenAPITools page](https://github.com/openapitools/openapi-generator). You can run the script online, on docker, and various other ways.
2. Go to the generated directory you defined. There's a fully working NodeJS-ExpressJs server waiting for you. This is important - the code is yours to change and update! Look at config.js and see that the settings there are ok with you - the server will run on port 3000, and files will be uploaded to a new directory 'uploaded_files'.
3. The server will base itself on an openapi.yaml file which is located under /api/openapi.yaml. This is not exactly the same file that you used to generate the app:
I.  If you have `application/json` contentBody that was defined inside the path object - the generate will have moved it to the components/schemas section of the openapi document.
II. Every process has a new element added to it - `x-eov-operation-handler: controllers/PetController` which directs the call to that file.
III. We have a Java application that translates the operationId to a method, and a nodeJS script that does the same process to call that method. Both are converting the method to `camelCase`, but might have discrepancy. Please pay attention to the operationID names, and see that they are represented in the `controllers` and `services` directories.
4. Take the time to understand the structure of the application. There might be bugs, and there might be settings and business-logic that does not meet your expectation. Instead of dumping this solution and looking for something else - see if you can make the generated code work for you.
To keep the explanation short (a more detailed explanation will follow): Application starts with a call to index.js (this is where you will plug in the db later). It calls expressServer.js which is where the express.js and openapi-validator kick in. This is an important file. Learn it. All calls to endpoints that were configured in the openapi.yaml document go to `controllers/{name_of_tag_which_the_operation_was_associated_with}.js`, which is a very small method. All the business-logic lies in `controllers/Controller.js`, and from there - to `services/{name_of_tag_which_the_operation_was_associated_with}.js`.

5. Once you've understood what is *going* to happen, launch the app and ensure everything is working as expected:
```
npm start
```

### View and test the API
(Assuming no changes were made to config.js)

1. API documentation, and to check the available endpoints:
http://localhost:3000/api-docs/. To
2. Download the openapi.yaml document: http://localhost:3000/openapi.
3.  Every call to an endpoint that was defined in the openapi document will return a 200 and a list of all the parameters and objects that were sent in the request.
4. Endpoints that require security need to have security handlers configured before they can return a successful response. At this point they will return [ a response code of 401](https://developer.mozilla.org/en-US/docs/Web/HTTP/Status/401).
5. ##### At this stage the server does not support document body sent in xml format.

### Project Files
#### Root Directory:
In the root directory we have (besides package.json, config.js, and log files):
- **logger.js** - where we define the logger for the project. The project uses winston, but the purpose of this file is to enable users to change and modify their own logger behavior.
- **index.js** - This is the project's 'main' file, and from here we launch the application. This is a very short and concise file, and the idea behind launching from this short file is to allow use-cases of launching the server with different parameters (changing config and/or logger) without affecting the rest of the code.
- **expressServer.js** - The core of the Express.js server. This is where the express server is initialized, together with the OpenAPI validator, OpenAPI UI, and other libraries needed to start our server. If we want to add external links, that's where they would go. Our project uses the [express-openapi-validator](https://www.npmjs.com/package/express-openapi-validator) library that acts as a first step in the routing process - requests that are directed to paths defined in the `openapi.yaml` file are caught by this process, and it's parameters and bodyContent are validated against the schema. A successful result of this validation will be a new 'openapi' object added to the request. If the path requested is not part of the openapi.yaml file, the validator ignores the request and passes it on, as is, down the flow of the Express server.

#### api/
- **openapi.yaml** - This is the OpenAPI contract to which this server will comply. The file was generated using the codegen, and should contain everything needed to run the API Gateway - no references to external models/schemas.

#### utils/
Currently a single file:

- **openapiRouter.js** - This is where the routing to our back-end code happens. If the request object includes an ```openapi``` object, it picks up the following values (that are part of the ```openapi.yaml``` file): 'x-openapi-router-controller', and 'x-openapi-router-service'. These variables are names of files/classes in the controllers and services directories respectively. The operationId of the request is also extracted. The operationId is a method in the controller and the service that was generated as part of the codegen process. The routing process sends the request and response objects to the controller, which will extract the expected variables from the request, and send it to be processed by the service, returning the response from the service to the caller.

#### controllers/
After validating the request, and ensuring this belongs to our API gateway, we send the request to a `controller`, where the variables and parameters are extracted from the request and sent to the relevant `service` for processing. The `controller` handles the response from the `service` and builds the appropriate HTTP response to be sent back to the user.

- **index.js** - load all the controllers that were generated for this project, and export them to be used dynamically by the `openapiRouter.js`. If you would like to customize your controller, it is advised that you link to your controller here, and ensure that the codegen does not rewrite this file.

- **Controller.js** - The core processor of the generated controllers. The generated controllers are designed to be as slim and generic as possible, referencing to the `Controller.js` for the business logic of parsing the needed variables and arguments from the request, and for building the HTTP response which will be sent back. The `Controller.js` is a class with static methods.

- **.js** - auto-generated code, processing all the operations. The Controller is a class that is constructed with the service class it will be sending the request to. Every request defined by the `openapi.yaml`  has an operationId. The operationId is the name of the method that will be called. Every method receives the request and response, and calls the `Controller.js` to process the request and response, adding the service method that should be called for the actual business-logic processing.

#### services/
This is where the API Gateway ends, and the unique business-logic of your application kicks in. Every endpoint in the `openapi.yaml` has a variable 'x-openapi-router-service', which is the name of the service class that is generated. The operationID of the endpoint is the name of the method that will be called. The generated code provides a simple promise with a try/catch clause. A successful operation ends with a call to the generic `Service.js` to build a successful response (payload and response code), and a failure will call the generic `Service.js` to build a response with an error object and the relevant response code. It is recommended to have the services be generated automatically once, and after the initial build add methods manually.

- **index.js** - load all the services that were generated for this project, and export them to be used dynamically by the `openapiRouter.js`. If you would like to customize your service, it is advised that you link to your controller here, and ensure that the codegen does not rewrite this file.

- **Service.js** - A utility class, very simple and thin at this point, with two static methods for building a response object for successful and failed results in the service operation. The default response code is 200 for success and 500 for failure. It is recommended to send more accurate response codes and override these defaults when relevant.

- **.js** - auto-generated code, providing a stub Promise for each operationId defined in the `openapi.yaml`. Each method receives the variables that were defined in the `openapi.yaml` file, and wraps a Promise in a try/catch clause. The Promise resolves both success and failure in a call to the `Service.js` utility class for building the appropriate response that will be sent back to the Controller and then to the caller of this endpoint.

#### tests/
- **serverTests.js** - basic server validation tests, checking that the server is up, that a call to an endpoint within the scope of the `openapi.yaml` file returns 200, that a call to a path outside that scope returns 200 if it exists and a 404 if not.
- **routingTests.js** - Runs through all the endpoints defined in the `openapi.yaml`, and constructs a dummy request to send to the server. Confirms that the response code is 200. At this point requests containing xml or formData fail - currently they are not supported in the router.
- **additionalEndpointsTests.js** - A test file for all the endpoints that are defined outside the openapi.yaml scope. Confirms that these endpoints return a successful 200 response.


Future tests should be written to ensure that the response of every request sent should conform to the structure defined in the `openapi.yaml`. This test will fail 100% initially, and the job of the development team will be to clear these tests.


#### models/
Currently a concept awaiting feedback. The idea is to have the objects defined in the openapi.yaml act as models which are passed between the different modules. This will conform the programmers to interact using defined objects, rather than loosely-defined JSON objects. Given the nature of JavaScript programmers, who want to work with their own bootstrapped parameters, this concept might not work. Keeping this here for future discussion and feedback.















 readmeEtag: '"be947f884b56f52f14003bf276763525b372d045"' readmeLastModified: Tue, 06 Feb 2024 16:43:15 GMT repositoryId: 579412089 description: >- API to create appealing images of company products made of chocolate, gold or lego created: '2022-12-17T16:00:42Z' updated: '2025-10-08T00:55:50Z' language: HTML archived: false stars: 3 watchers: 2 forks: 1 owner: postman-solutions-eng logo: https://avatars.githubusercontent.com/u/91276282?v=4 license: Apache-2.0 repoEtag: '"0865060d350054066af4a1aa762495d83f8870a1483d11ce6501fd020d2e9b4c"' repoLastModified: Wed, 08 Oct 2025 00:55:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sdclarkelab/covid19-jamaica-api v3: true repositoryMetadata: base64Readme: >- IyBDb3ZpZC0xOSBKYW1haWNhIEFQSQpUaGlzIHByb2plY3RzIHB1bGxzIENvdmlkLTE5IHJlcG9ydHMgZnJvbSB0aGUgKltNaW5pc3RyeSBvZiBIZWFsdGgsIEphbWFpY2FdKGh0dHBzOi8vd3d3Lm1vaC5nb3Yuam0vdXBkYXRlcy9jb3JvbmF2aXJ1cy9jb3ZpZC0xOS1jbGluaWNhbC1tYW5hZ2VtZW50LXN1bW1hcnkvKSogYW5kIHNldmVycyB0aGF0IGRhdGEgaW4gSlNPTiBmb3JtYXQgdmlhIFJFU1QgQVBJLgoKIyMgSG93IHRvIFJ1bgpgYGBzaAokIHBpcCBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQKJCBweXRob24gYXBpLnB5CmBgYAoKIyMgQVBJIGRvY3VtZW50YXRpb24KTmF2aWdhdGUgdG8gdGhlIGZvbGxvd2luZyBVUkwgKGh0dHA6Ly9sb2NhbGhvc3Q6NTAwMC9jb3ZpZDE5LWphbS9hcGkvdjEvdWkpIGxvY2FsbHkgdG8gdmlldyB0aGUgT3BlbkFQSSB1c2VyIGludGVyZmFjZSBvciB2aXNpdCAoaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcy9zZGNsYXJrZWxhYi9jb3ZpZDE5X2phbV9hcGkvdjEpCg== readmeEtag: '"deb5df79301773cb8d7c3c7c63f16e3c85bbe8fc"' readmeLastModified: Sat, 23 Oct 2021 22:21:02 GMT repositoryId: 416054590 description: Jamaica COVID-19 API data scraped from the MOH's website created: '2021-10-11T19:09:28Z' updated: '2024-12-04T17:11:49Z' language: Python archived: false stars: 3 watchers: 1 forks: 0 owner: sdclarkelab logo: https://avatars.githubusercontent.com/u/44792415?v=4 repoEtag: '"f2f1f7a260072d36014495d50f3947b5c11b6255f441d13efe928cc94c9fadfb"' repoLastModified: Wed, 04 Dec 2024 17:11:49 GMT foundInMaster: true category: Server Implementations id: eb4c3835c1410cc0afc0e3b22c7bb093 - source: openapi3 tags repository: https://github.com/ideal-postcodes/openapi v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgogIDxpbWcgc3JjPSJodHRwczovL2ltZy5pZGVhbC1wb3N0Y29kZXMuY28udWsvT3BlbkFQSSUyMExvZ29AM3gucG5nIiBhbHQ9Ik9wZW5BUEkiPgo8L2gxPgoKPiBJZGVhbCBQb3N0Y29kZXMgT3BlbkFQSSAodjMpIFNwZWNpZmljYXRpb24gKGFwaS5pZGVhbC1wb3N0Y29kZXMuY28udWspCgpbIVtDSV0oaHR0cHM6Ly9naXRodWIuY29tL2lkZWFsLXBvc3Rjb2Rlcy9vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vaWRlYWwtcG9zdGNvZGVzL29wZW5hcGkvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sKQoKIyMgRmVhdHVyZXMKCi0gT3BlbkFQSSB2MyBzcGVjaWZpY2F0aW9uIGZvciBhcGkuaWRlYWwtcG9zdGNvZGVzLmNvLnVrCi0gRXhwb3J0cyBBUEkgVHlwaW5ncwoKIyMgTGlua3MKCi0gW0dpdGh1YiBSZXBvc2l0b3J5XShodHRwczovL2dpdGh1Yi5jb20vaWRlYWwtcG9zdGNvZGVzL29wZW5hcGkpCi0gW0FQSSBSZWZlcmVuY2VdKGh0dHBzOi8vb3BlbmFwaS5pZGVhbC1wb3N0Y29kZXMuY28udWspCi0gT3BlbkFQSSB2MyBSYXcgRmlsZXM6IFtKU09OXShodHRwczovL29wZW5hcGkuaWRlYWwtcG9zdGNvZGVzLmNvLnVrL29wZW5hcGkuanNvbikKLSBPcGVuQVBJIHYzIFJhdyBGaWxlczogW1lBTUxdKGh0dHBzOi8vb3BlbmFwaS5pZGVhbC1wb3N0Y29kZXMuY28udWsvb3BlbmFwaS55YW1sKQotIFtOUE0gUGFja2FnZV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQGlkZWFsLXBvc3Rjb2Rlcy9vcGVuYXBpKQoKIyMgR2V0dGluZyBTdGFydGVkCgojIyMgQnVpbGQgVHlwaW5ncwoKYGBgCm5wbSBydW4gYnVpbGQKYGBgCgojIyMgVmFsaWRhdGUgU3BlYwoKYGBgCm5wbSB0ZXN0CmBgYAoKIyMjIERvd25sb2FkIGFuZCBBY2Nlc3MgdmlhIG5wbQoKIyMjIyBJbnN0YWxsCgpgYGBiYXNoCm5wbSBpbnN0YWxsIEBpZGVhbC1wb3N0Y29kZXMvb3BlbmFwaQpgYGAKCiMjIyMgUmVxdWlyZSBzcGVjcwoKSlNPTiBhbmQgWUFNTCBzcGVjcyBjYW4gYmUgYWNjZXNzZWQgYXQgYG5vZGVfbW9kdWxlcy9AaWRlYWwtcG9zdGNvZGVzL29wZW5hcGkvZGlzdC9vcGVuYXBpLltqc29ufHlhbWxdYAoKIyMgTGljZW5jZQoKQ29weXJpZ2h0IElERFFEIExpbWl0ZWQK readmeEtag: '"32a53763157090edbf5fa882f3e3f2dc0128e2ed"' readmeLastModified: Wed, 27 Aug 2025 12:00:56 GMT repositoryId: 181899806 description: OpenAPI V3 specifications for api.ideal-postcodes.co.uk created: '2019-04-17T13:41:02Z' updated: '2026-02-04T15:40:14Z' language: TypeScript archived: false stars: 3 watchers: 3 forks: 0 owner: ideal-postcodes logo: https://avatars.githubusercontent.com/u/4996310?v=4 license: MIT repoEtag: '"4811f2a04115adad602ada54a641769aea1c77cec19f163ee7c1decb3cbdce83"' repoLastModified: Wed, 04 Feb 2026 15:40:14 GMT foundInMaster: true category: SDK id: e5fcae1b1547e9581f18e43328e332ef - source: openapi3 tags repository: https://github.com/muskratjs/muskrat v3: true repositoryMetadata: base64Readme: IyBtdXNrcmF0CgpSdW50aW1lIEFQSSBzY2hlbWEgZ2VuZXJhdGlvbgo= readmeEtag: '"f079a4c9e4bd2bd72dc78013f795c4502fc7da02"' readmeLastModified: Tue, 29 Jan 2019 12:39:17 GMT repositoryId: 135346267 description: ':rat: Generate project metadata' created: '2018-05-29T19:55:40Z' updated: '2019-01-29T12:39:19Z' language: TypeScript archived: false stars: 3 watchers: 2 forks: 0 owner: muskratjs logo: https://avatars.githubusercontent.com/u/41447837?v=4 license: MIT repoEtag: '"5810c4fed29be686d6260271e00fb7efbbc39fd8310b7f735b5620bba6604efb"' repoLastModified: Tue, 29 Jan 2019 12:39:19 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: e426cf26d8718a6a03b77573b5f15664 - source: openapi3 tags repository: https://github.com/axiomsamarth/gita.io v3: true repositoryMetadata: base64Readme: >- IyBnaXRhLmlvCkdpdGEuaW8gaXMgYW4gQVBJIGFkaGVyaW5nIHRvIHN0YW5kYXJkcyBvZiBPcGVuQVBJIHYzLjAsIHRvIGxldmVyYWdlIHRoZSBHaXRhIHZlcnNlcyBwcm9ncmFtbWF0aWNhbGx5LiBUaGUgR2l0YS5pbyBBUEkgc2VydmVzIHRoZSB2ZXJzZXMgaW4gdGhlIG5hdGl2ZSBEZXZhbmFnYXJpIHNjcmlwdCBhbG9uZyB3aXRoIHRoZWlyIHRyYW5zbGl0ZXJhdGlvbiBhbmQgbWVhbmluZy4KClRoaXMgcmVzcG9zaXRvcnkgcHJvdmlkZXMgdGhlIHNvdXJjZSBjb2RlIG9mIHRoZSBBUEkgaW4gdGhlIGZvbGRlciBgL2dpdGEtYXBpYCBhbG9uZyB3aXRoIGFuIGV4YW1wbGUgUHl0aG9uLUZsYXNrIHdlYnNpdGUgYnVpbHQgZW1wbG95aW5nIHRoZSBnaXRhLmlvIEFQSSBpbiB0aGUgZm9sZGVyIGAvZ2l0YS1hcGktaW9gLgoKIyMgQVBJIERvY3VtZW50YXRpb24KClRoZSBvZmZpY2lhbCBBUEkgZG9jdW1lbnRhdGlvbiBmb3IgZGV2ZWxvcGVyIHVzZSBjYW4gYmUgZm91bmQgYXQgW0dpdGEuaW8gU3dhZ2dlciBEb2N1bWVudGF0aW9uXShodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzLWRvY3MvQXhpb21TYW1hcnRoL2dpdGEuaW8vMS4wLjApCgojIyBQcmVyZXF1aXNpdGVzClRoZSBmb2xsb3dpbmcgYXJlIHRoZSBwcmVyZXF1aXNpdGVzIGZvciBhIHNlYW1sZXNzIGV4ZWN1dGlvbiBvZiB0aGUgQVBJLiAKCi0gUHJlZmVycmFibHkgbGF0ZXN0IE1TIFdpbmRvd3MvTGludXgvTWFjT1MgUGxhdGZvcm0KLSBQeXRob24gPj0gMy4wCi0gUHl0aG9uIEZsYXNrID49IDEuMS4yCi0gUHltb25nbyA+PSAzLjEwLjEKLSBNb25nb0RCIGFzIHRoZSBkYXRhYmFzZQoKIyMgU2V0dXAgaW5zdHJ1Y3Rpb25zCgotIFBsZWFzZSByZWFkIHRoZSBbUkVBRE1FLm1kXSgvZ2l0YS1hcGkvUkVBRE1FLm1kKSBmb3Igc2V0dGluZyB1cCB0aGUgZ2l0YS5pbyBBUEkKCi0gUGxlYXNlIHJlYWQgdGhlIFtSRUFETUUubWRdKC9naXRhLWFwaS1pby9SRUFETUUubWQpIGZvciBzZXR0aW5nIHVwIHRoZSBleGFtcGxlIGZsYXNrIHdlYiBhcHBsaWNhdGlvbiB0aGF0IGxldmVyYWdlcyB0aGUgZ2l0YS5pbyBBUEkuIAoKIyMgRGVtbyB2aWRlb3MKVGhlIGRlbW8gdmlkZW9zIG9mIHRoZSBnaXRhLmlvIGFuZCB0aGUgd2Vic2l0ZSBidWlsdCBlbXBsb3lpbmcgZ2l0YS5pbyBjYW4gYmUgc2VlbiBvbiBteSBZb3VUdWJlIGNoYW5uZWwgYXQ6CgotIFtnaXRhLmlvIEFQSSBpbiBhY3Rpb25dKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9eFF6UGJVMjdZdVEmbGlzdD1QTEdnemVfWFdZSW95cy1ldFhUQkw0NmN0ZEhkREF0RldrJmluZGV4PTEyJnQ9MHMpCgotIFtnaXRhLmlvIEFQSSBmb3IgZGV2ZWxvcGVycy4gRGVtbyBpbiBQb3N0bWFuXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PUstZmJVNTU2LW5jJmxpc3Q9UExHZ3plX1hXWUlveXMtZXRYVEJMNDZjdGRIZERBdEZXayZpbmRleD0xMyZ0PTBzKQoKIyMgQ29udHJpYnV0aW9uIEd1aWRlbGluZXMKClRoYW5rIHlvdSBzbyBtdWNoIGZvciBsb29raW5nIGZvcndhcmQgdG8gY29udHJpYnV0ZS4gUGxlYXNlIHJlYWQgdGhlIFtjb250cmlidXRpb24gZ3VpZGVsaW5lc10oL0NvbnRyaWJ1dGlvbi5tZCkgYmVmb3JlIHByb2NlZWRpbmcuIFdlIHN0cmljdGx5IGFkaGVyZSB0byB0aGVtLg== readmeEtag: '"d6a1ade6de6f2968ffb917afac29fd3dde6c6c30"' readmeLastModified: Fri, 26 Mar 2021 13:13:27 GMT repositoryId: 275591455 description: >- Gita.io is an API adhering to standards of OpenAPI v3.0, to leverage the Gita verses programmatically. The Gita.io API serves the verses in the native Devanagari script along with their transliteration and meaning. created: '2020-06-28T13:32:32Z' updated: '2023-01-13T17:29:31Z' language: HTML archived: false stars: 3 watchers: 1 forks: 1 owner: AxiomSamarth logo: https://avatars.githubusercontent.com/u/32246441?v=4 repoEtag: '"02ae5e9d7027de8bc370709c82687110330caf8d5c5d8fb7abdfe3631ed5bbe0"' repoLastModified: Fri, 13 Jan 2023 17:29:31 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 376287ab5e6f1dcd1912759578fee798 - source: openapi3 tags repository: https://github.com/cope-systems/bottle-openapi-3 v3: true repositoryMetadata: base64Readme: >- PT09PT09PT09PT09PT09PT09PT09PT09PT0KQm90dGxlIE9wZW5BUEkgMyBQbHVnaW4KPT09PT09PT09PT09PT09PT09PT09PT09PT0KLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KQWJvdXQKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KClRoZSBCb3R0bGUgT3BlbkFQSSAzIFBsdWdpbiBpcyBhIHRvb2xraXQgZm9yIHBlcmZvcm1pbmcgdmFsaWRhdGlvbiBvZiByZXF1ZXN0cwphZ2FpbnN0IGFuIE9wZW5BUEkgZG9jdW1lbnQgZm9yIGBCb3R0bGUgPGh0dHBzOi8vYm90dGxlcHkub3JnL2RvY3MvMC4xMi8+YF8gYXBwbGljYXRpb25zLiBJdCBpcyBidWlsdCBvbiB0aGUgYG9wZW5hcGktY29yZSA8aHR0cHM6Ly9naXRodWIuY29tL3AxYzJ1L29wZW5hcGktY29yZT5gXwphbmQgYG9wZW5hcGktc3BlYy12YWxpZGF0b3IgPGh0dHBzOi8vZ2l0aHViLmNvbS9wMWMydS9vcGVuYXBpLXNwZWMtdmFsaWRhdG9yPmBfIGxpYnJhcmllcywgYW5kIHN1cHBvcnRzCnRoZSBgT3BlbkFQSSAzIHNwZWNpZmljYXRpb24gPGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kPmBfLgoKLS0tLS0tLS0KTGljZW5zZQotLS0tLS0tLQoKVGhpcyBjb2RlYmFzZSBpcyBNSVQgbGljZW5zZWQuCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpSZXF1aXJlbWVudHMKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCkEgcmVsYXRpdmVseSByZWNlbnQgdmVyc2lvbiBvZiBQeXRob24gKDMuNSspIGlzIHJlcXVpcmVkLiBUaGlzIHBsdWdpbiBkZXBlbmRzIG9uIHRoZQphZm9yZW1lbnRpb25lZCBgYG9wZW5hcGktY29yZWBgIGFuZCBgYG9wZW5hcGktc3BlYy12YWxpZGF0b3JgYCBsaWJyYXJpZXMsIGFuZCBhbHNvIHJlcXVpcmVzCmEgcmVsYXRpdmVseSByZWNlbnQgdmVyc2lvbiBvZiBgYGJvdHRsZWBgICgwLjEyKykuCgoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KUXVpY2tzdGFydAotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKVGhlIEJvdHRsZSBPcGVuQVBJIDMgcGx1Z2luIG1heSBlaXRoZXIgYmUgaW5zdGFsbGVkIGZyb20gYHB5cGkgPGh0dHBzOi8vcHlwaS5vcmcvcHJvamVjdC9ib3R0bGUtb3BlbmFwaS0zLz5gXyAgYXMgdGhlIGBgYm90dGxlLW9wZW5hcGktM2BgIHBhY2thZ2U6CgogICAgcGlwIGluc3RhbGwgYm90dGxlLW9wZW5hcGktMwoKb3IgbWF5IGJlIGluc3RhbGxlZCBmcm9tIHNvdXJjZSBmcm9tIHRoZSBgZ2l0IHJlcG9zaXRvcnkgPGh0dHBzOi8vZ2l0aHViLmNvbS9jb3BlLXN5c3RlbXMvYm90dGxlLW9wZW5hcGktMz5gXzoKCiAgICBweXRob24gc2V0dXAucHkgaW5zdGFsbAoKT25jZSB0aGUgcGx1Z2luIGlzIGluc3RhbGxlZCwgaXQgbWF5IGJlIHVzZWQgaW4gYSBCb3R0bGUgYXBwbGljYXRpb24gYnkgbG9hZGluZyB0aGUgT3BlbkFQSSBzY2hlbWEgYW5kIGluc3RhbGxpbmcgdGhlCnBsdWdpbi4gQW4gZXhhbXBsZToKCi4uIGNvZGUtYmxvY2s6OiBweXRob24KCiAgICBpbXBvcnQgYm90dGxlCiAgICBpbXBvcnQgeWFtbAogICAgZnJvbSBib3R0bGVfb3BlbmFwaV8zIGltcG9ydCBPcGVuQVBJUGx1Z2luCgogICAgYXBwID0gYm90dGxlLkJvdHRsZSgpCgogICAgd2l0aCBvcGVuKCJzd2FnZ2VyLnlhbWwiKSBhcyBmOgogICAgICAgIHNwZWMgPSB5YW1sLmxvYWQoZikKCiAgICBAYXBwLnJvdXRlKCIvYXBpL2ZvbyIpCiAgICBkZWYgZm9vX2hhbmRsZXIoKToKICAgICAgICByZXR1cm4geyJmb28iOiAiYmFyIn0KCiAgICBhcHAuaW5zdGFsbChPcGVuQVBJUGx1Z2luKHNwZWMpKQoKICAgIGFwcC5ydW4oKQoKVGhlIGV4YW1wbGUncyBzcGVjaWZpY2F0aW9uOgoKLi4gY29kZS1ibG9jazo6IHlhbWwKCiAgICBvcGVuYXBpOiAzLjAuMAogICAgaW5mbzoKICAgICAgdGl0bGU6IE15IEFQSQogICAgc2VydmVyczoKICAgICAgLSB1cmw6IC9hcGkKICAgIHBhdGhzOgogICAgICAvZm9vOgogICAgICAgIGdldDoKICAgICAgICAgICBzdW1tYXJ5OiBGZXRjaCBhbiBvYmplY3QKICAgICAgICAgICByZXNwb25zZXM6CiAgICAgICAgICAgICAiMjAwIjoKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uOiAiQW4gb2JqZWN0IHdhcyBzdWNjZXNzZnVsbHkgZ2VuZXJhdGVkLiIKICAgICAgICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgICAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgICAgICAgICAgICAgc2NoZW1hOiB7InR5cGUiOiAib2JqZWN0In0KCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpBZHZhbmNlZCBVc2FnZQotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKVE9ETwoKCgotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQpDaGFuZ2Vsb2cKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KCjAuMS4yIChNYXkgMjAyMSkKKioqKioqKioqKioqKioqKioKCkZpeGVkIGFuIGlzc3VlIGRlY29kaW5nIHRoZSByZXF1ZXN0IGJvZHkgZm9yIEhUVFAgbWV0aG9kcyBsaWtlClBPU1QsIFBVVCwgZXRjLgoKCjAuMS4wIChKYW4gMjAyMSkKKioqKioqKioqKioqKioqKioKCkluaXRpYWwgYWxwaGEgcmVsZWFzZSBvZiB0aGUgT3BlbkFQSSAzIHBsdWdpbiBmb3IKQm90dGxlLiBNb3N0IGZ1bmN0aW9uYWxpdHkgc2hvdWxkIGJlIGltcGxlbWVudGVkLg== readmeEtag: '"0c1057831bb5620d6168385ada5a00d0b2570a31"' readmeLastModified: Sun, 03 Apr 2022 14:51:31 GMT repositoryId: 313789033 description: OpenAPI 3.0 Support for the Bottle Web Framework created: '2020-11-18T01:27:57Z' updated: '2023-10-25T10:49:15Z' language: Python archived: false stars: 3 watchers: 2 forks: 4 owner: cope-systems logo: https://avatars.githubusercontent.com/u/50466330?v=4 license: MIT repoEtag: '"29c329428fb78326b476dab3f11cb4a1c7f961700be8d0997947617b56b09e89"' repoLastModified: Wed, 25 Oct 2023 10:49:15 GMT category: Parsers foundInMaster: true id: db2d64670d722bc3675c6b471ee74aa1 - source: openapi3 tags repository: https://github.com/funa1g/multi-file-oas-example v3: true repositoryMetadata: base64Readme: >- IyBtdWx0aS1maWxlLW9hcy1leGFtcGxlCkFuIGV4YW1wbGUgb2YgbXVsdGkgZmlsZSBPcGVuQVBJIFNwZWNpZmljYXRpb24gMy4wCgojIyBVc2FnZQpJZiB5b3UgY2hlY2sgdGhpcyBleGFtcGxlLCB1c2UgZG9ja2VyIGVudmlyb25tZW50LgoKYGBgCmRvY2tlci1jb21wb3NlIHVwCmBgYAoKQW5kIHNob3cgYGxvY2FsaG9zdDo4MDgwYAoKIyMgRGV0YWlscwpUaGlzIHNhbXBsZSB1c2VzIGBtdWx0aS1maWxlLXN3YWdnZXJgLgoKIyBDb3B5cmlnaHQKVGhpcyBzb2Z0d2FyZSBpbmNsdWRlcyB0aGUgd29yayB0aGF0IGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBBcGFjaGUgTGljZW5zZSAyLjAK readmeEtag: '"262fff406391a7549d32cfae001eccf629f108f4"' readmeLastModified: Sun, 20 May 2018 09:18:01 GMT repositoryId: 121081045 description: An example of multi file OpenAPI Specification 3.0 created: '2018-02-11T03:39:43Z' updated: '2023-09-08T08:19:44Z' language: HTML archived: false stars: 3 watchers: 1 forks: 7 owner: funa1g logo: https://avatars.githubusercontent.com/u/9946679?v=4 license: MIT repoEtag: '"e68d3854597319e28126cb79c4fd8b66abc157921fc44dfbbdf15a93678c1731"' repoLastModified: Fri, 08 Sep 2023 08:19:44 GMT foundInMaster: true category: - Documentation - Parsers - Server Implementations id: 7fc78b8250209771d87b7f5c12781671 - source: openapi3 tags repository: https://github.com/vkazakevich/slim-vue-app v3: true id: eba520242ad6418352b035717a1f87a2 repositoryMetadata: base64Readme: >- IyBTbGltIDQgKyBWdWUgMyBhcHBsaWNhdGlvbgoKUHJvamVjdCBkZW1vbnN0cmF0aW5nIHVzaW5nIHRoZSBTbGltIDQgZnJhbWV3b3JrIHRvIGNyZWF0ZSBhIFJlc3RmdWwgQVBJLiBUaGUgZGF0YWJhc2UgaXMgcG93ZXJlZCBieSBFbG9xdWVudCwgYW5kIHRoZSBjbGllbnQgc2lkZSBpcyBkZXZlbG9wZWQgdXNpbmcgVnVlIDMuCgoKIyMgR2V0dGluZyBTdGFydGVkCgpIZXJlIHlvdSBjYW4gc2VlIGhvdyB0byBzZXR1cCB0aGUgcHJvamVjdC4gU2luY2UgdGhlIHByb2plY3QgaXMgdXNpbmcgZG9ja2VyIHlvdSBoYXZlIHRvIGhhdmUgaXQgaW5zdGFsbGVkIGFuZCBydW5uaW5nLgoKVXNlIHRoZSBmb2xsb3dpbmcgY29tbWFuZHMgdG8gaW5pdGlhbGl6ZSB0aGUgcHJvamVjdDoKCmBgYGJhc2gKIyBZb3UgY2FuIG5vdyBlZGl0IHRoZSBzZXR0aW5ncyBpbiB0aGUgLmVudiBmaWxlCmNwIC5lbnYuZXhhbXBsZSAuZW52CgojIFJ1biBkb2NrZXIgYnVpbGQgY29tbWFuZAptYWtlIGJ1aWxkCgojIEluc3RhbGwgdGhlIENvbXBvc2VyIGRlcGVuZGVuY2llcwptYWtlIGNvbXBvc2VyLWluc3RhbGwgCgojIENyZWF0ZSBhIGRhdGFiYXNlCm1ha2UgY3JlYXRlLWRhdGFiYXNlCmBgYAoKVG8gcG9wdWxhdGUgdGhlIGRhdGFiYXNlIHdpdGggZmFrZSB2YWx1ZXMsIHVzZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBiYXNoCm1ha2UgcG9wdWxhdGUtZGF0YWJhc2UKYGBgCgpUbyBzdG9wIG9yIHN0YXJ0IHRoZSBjb250YWluZXIsIHVzZSBgbWFrZSBzdGFydGAgYW5kIGBtYWtlIHN0b3BgLgoKVGhlIHByb2plY3QgaXMgYWNjZXNzaWJsZSBieSBgbG9jYWxob3N0OjgwODBgIGhvc3QuCgojIyBQb3N0bWFuCgpGb3IgeW91ciBjb252ZW5pZW5jZSwgdGhpcyByZXBvc2l0b3J5IGhhcyBhIFtPcGVuQVBJIFNwZWNpZmljYXRpb25dKC4vb3BlbmFwaS5qc29uKSB0aGF0IGNhbiBiZSBpbXBvcnRlZCBpbnRvIFBvc3RtYW4gKFttb3JlIGRldGFpbHNdKGh0dHBzOi8vbGVhcm5pbmcucG9zdG1hbi5jb20vZG9jcy9pbnRlZ3JhdGlvbnMvYXZhaWxhYmxlLWludGVncmF0aW9ucy93b3JraW5nLXdpdGgtb3BlbkFQSS8pKS4K readmeEtag: '"08a60f167948685d361ddbb722e7395cdfc36c3a"' readmeLastModified: Tue, 11 Jul 2023 06:46:45 GMT repositoryId: 664813397 description: >- Demo application on Slim 4 bundled with Eloquent and OpenAPI, and Vue 3 on the client side. created: '2023-07-10T20:05:20Z' updated: '2024-03-31T21:58:45Z' language: PHP archived: false stars: 3 watchers: 1 forks: 0 owner: vkazakevich logo: https://avatars.githubusercontent.com/u/9676524?v=4 license: MIT repoEtag: '"a1c9056ee4c27cfd9f0e82a1aa04e373fbe9a42df125ee5b29a37e0146b07eb8"' repoLastModified: Sun, 31 Mar 2024 21:58:45 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/vkthedev/slim-vue-app - source: openapi3 tags repository: https://github.com/yusufsefasezer/jax-rs-example v3: true repositoryMetadata: base64Readme: >- IyBKYXZhIEpBWC1SUyBSRVNUCgpBIHNpbXBsZSBSRVNUIGJhc2VkIGFwcGxpY2F0aW9uIGRldmVsb3BlZCB3aXRoIEpBVkEsIEpBWC1SUywgU3dhZ2dlcihPcGVuQVBJKSBhbmQgQW5ndWxhci4KCiMjIFtEb3dubG9hZF0oaHR0cHM6Ly9naXRodWIuY29tL3l1c3Vmc2VmYXNlemVyL2pheC1ycy1leGFtcGxlL2FyY2hpdmUvbWFzdGVyLnppcCkKCiMjIEhvdyB0byBydW4KCk1hdmVuIG11c3QgYmUgaW5zdGFsbGVkIHRvIHJ1biB0aGlzIGFwcGxpY2F0aW9uLgoKSWYgeW91IGhhdmUgbWF2ZW4gZXhlY3V0ZSB0aGUgYmVsb3cgY29tbWFuZCB0byBydW4uCgpgYGAKbXZuIGNsZWFuIHBhY2thZ2UKYGBgCgpUaGVuIGRlcGxveSBnZW5lcmF0ZWQgLndhciBmaWxlKHNlcnZlci90YXJnZXQpIHRvIHNlcnZsZXQgY29udGFpbmVyKHRvbWNhdCwgamV0dHkpIG9yIGFwcGxpY2F0aW9uIHNlcnZlcihwYXlhcmEsIGdsYXNzZmlzaCwgdG9tZWUpCgoqKk5PVEU6KiogU2VydmxldCBDb250YWluZXIgbXVzdCBzdXBwb3J0IFNlcnZsZXQgNi4wIHNwZWNzCgojIyBEb2NrZXIKCioqRG9ja2VyIG11c3QgYmUgaW5zdGFsbGVkLioqCgpCdWlsZCB0aGUgRG9ja2VyIGltYWdlIHdpdGggdGhlIHRhZyAiamF2YS1zZXJ2bGV0LWNvbnRhY3QiCgpgYGAKZG9ja2VyIGJ1aWxkIC10IGphdmEtc2VydmxldC1jb250YWN0IC4KYGBgCgpgYGAKZG9ja2VyIHJ1biAtcCA4MDo4MDgwIGphdmEtc2VydmxldC1jb250YWN0CmBgYAoKWW91IGNhbiBhY2Nlc3MgdGhlIGFwcGxpY2F0aW9uIHVzaW5nIGBsb2NhbGhvc3QvamF4LXJzLWV4YW1wbGVgIGluIHlvdXIgd2ViIGJyb3dzZXIuCgojIyBTY3JlZW5zaG90CgotIFtEZWZhdWx0XShzY3JlZW5zaG90L2RlZmF1bHQucG5nKQotIFtDcmVhdGVdKHNjcmVlbnNob3QvY3JlYXRlLnBuZykKLSBbU2hvd10oc2NyZWVuc2hvdC9zaG93LnBuZykKLSBbRWRpdF0oc2NyZWVuc2hvdC9lZGl0LnBuZykKLSBbRG9jdW1lbnRhdGlvbiAvIFN3YWdnZXJdKHNjcmVlbnNob3QvZG9jdW1lbnRhdGlvbi5wbmcpCi0gW0RvY3VtZW50YXRpb24gLyBTd2FnZ2VyIEdFVF0oc2NyZWVuc2hvdC9kb2N1bWVudGF0aW9uLWdldC5wbmcpCi0gW0RvY3VtZW50YXRpb24gLyBTd2FnZ2VyIFBPU1RdKHNjcmVlbnNob3QvZG9jdW1lbnRhdGlvbi1wb3N0LnBuZykKCiMgTGljZW5zZQpUaGlzIHByb2plY3QgaXMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIFtMSUNFTlNFXShMSUNFTlNFKSBmaWxlIGZvciBkZXRhaWxzCgpDcmVhdGVkIGJ5IFtZdXN1ZiBTZXplcl0oaHR0cHM6Ly93d3cueXVzdWZzZXplci5jb20pCg== readmeEtag: '"c468686b2cdc4c43029b9e0e8b13736d0c4fba25"' readmeLastModified: Wed, 13 Mar 2024 19:28:34 GMT repositoryId: 284446325 description: >- A simple REST based application developed with JAVA, JAX-RS, Swagger and Angular. created: '2020-08-02T11:16:21Z' updated: '2024-03-13T19:30:14Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 1 owner: yusufsefasezer logo: https://avatars.githubusercontent.com/u/9640186?v=4 license: MIT repoEtag: '"49f8a8072ed60af105bfefadab7f5cd2d0c170ba3f68041d910f5667c2b81359"' repoLastModified: Wed, 13 Mar 2024 19:30:14 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 0dd7d55cb3c92f112418b9838b65625f - source: openapi3 tags repository: https://github.com/oai/stories v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIEluaXRpYXRpdmUgKE9BSSkgU3RvcmllcwpUaGlzIGlzIGEgcHVibGljIHJlcG9zaXRvcnkgZm9yIHN1Z2dlc3RpbmcsIGNyZWF0aW5nLCBhbmQgZXZvbHZpbmcgc3RvcmllcyBhYm91dCB0aGUgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIChPQVMpLiBJbiBhbiBlZmZvcnQgdG8gZW5nYWdlIG1vcmUgd2l0aCB0aGUgY29tbXVuaXR5IHdlIGFyZSBtb3ZpbmcgdGhpcyBwcm9jZXNzIG9udG8gR2l0aHViLCBhbmQgdXNpbmcgaXNzdWVzLCBkaXNjdXNzaW9ucywgYW5kIHByb2plY3QgdG8gZGV2ZWxvcCBhIHJlZ3VsYXIgd2F2ZSBvZiBzdG9yeXRlbGxpbmcgYXJvdW5kIHRoZSBzcGVjaWZpY2F0aW9uLgoKVGhpcyByZXBvIGlzIHVzZWQgdG8gZ2F0aGVyIGlkZWFzIGFuZCBtb3ZlIGZvcndhcmQgc3RvcmllcyB0aGF0IHF1aXRlIG9mdGVuIHdpbGwgZW5kIHVwIG9uIHRoZSBPQUkgYmxvZywgYnV0IGNhbiBhbHNvIGJlIHB1Ymxpc2hlZCBvciBzeW5kaWNhdGVkIHRvIG90aGVyIGxvY2F0aW9ucywgaGVscGluZyBleHRlbmQgdGhlIHJlYWNoIG9mIHRoZSBPQUksIGFuZCBlZHVjYXRlIG1vcmUgb2YgdGhlIGNvbW11bml0eSBhYm91dCB3aGF0IHRoZSBzcGVjaWZpY2F0aW9uIGNhbiBkby4KCi0gW09wZW5BUEkgQmxvZyBHdWlkZWxpbmVzXShodHRwczovL2RvY3MuZ29vZ2xlLmNvbS9kb2N1bWVudC9kLzFDanBtUnpBRzZJdWpTdl90c053VXc2dnktMjlramJTUlY2NXUzMXNBSVJZL2VkaXQjKQoKIyMgQ3VycmVudCBCbG9nIFBvc3RzIGluIHRoZSBRdWV1ZQpUaGVzZSBhcmUgc29tZSBvZiB0aGUgYmxvZyBwb3N0cyB0aGF0IGhhdmUgYmVlbiBzdWdnZXN0ZWQgYW5kIGFyZSBpbiBuZWVkIG9mIGNoYW1waW9ucyB0byBoZWxwIG1vdmUgZWFjaCBzdG9yeSBmb3J3YXJkLgoKKiAqKk9BSSBTSUdzKiogKFtJc3N1ZV0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcy8xMSkpIChXb3JrZmxvdyBTdGF0dXM6IFtJZGVhc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBaWRlYXMpIC8gVG9waWNzKQoqICoqSlNPTiBTY2hlbWEgQnVuZGxpbmcqKiAoW0lzc3VlXShodHRwczovL2dpdGh1Yi5jb20vT0FJL1N0b3JpZXMvaXNzdWVzLzExKSkgKFdvcmtmbG93IFN0YXR1czogW1N0b3JpZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvU3Rvcmllcy9pc3N1ZXM/cT1pcyUzQWlzc3VlK2lzJTNBb3BlbitsYWJlbCUzQXN0b3JpZXMpIC8gRHJhZnQpCiogKipBUEkgQ29weXJpZ2h0IHdpdGggT0FTKiogKFtJc3N1ZV0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcy8yKSkgKFtQcm9qZWN0XShodHRwczovL2dpdGh1Yi5jb20vT0FJL1N0b3JpZXMvcHJvamVjdHMvMSkpIChXb3JrZmxvdyBTdGF0dXM6IFtTdHVjdHVyYWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvU3Rvcmllcy9pc3N1ZXM/cT1pcyUzQWlzc3VlK2lzJTNBb3BlbitsYWJlbCUzQXN0cnVjdHVyYWwpIC8gTm8pCiogKipYZXJvIE9wZW5BUEkgSm91cm5leSoqIChbSXNzdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvU3Rvcmllcy9pc3N1ZXMvMykpIChbUHJvamVjdF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL3Byb2plY3RzLzIpKSAoV29ya2Zsb3cgU3RhdHVzOiBbU3Rvcmllc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBc3RvcmllcykgLyBEcmFmdCkKKiAqKkdpdGh1YiBPcGVuQVBJIEpvdXJuZXkqKiAoSXNzdWUpIChbUHJvamVjdF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL3Byb2plY3RzLzMpKSAoV29ya2Zsb3cgU3RhdHVzOiBbU3Rvcmllc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBc3RvcmllcykgLyBEcmFmdCkKKiAqKk9wZW5BUEkgRm9ybWF0IFRvb2wqKiAoW0lzc3VlXShodHRwczovL2dpdGh1Yi5jb20vT0FJL1N0b3JpZXMvaXNzdWVzLzUpKSAoV29ya2Zsb3cgU3RhdHVzOiBbU3Rvcmllc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBc3RvcmllcykgLyBOb3RlcykKKiAqKk9wZW5BUEkgRGlmZiBUb29sKiogKFtJc3N1ZV0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcy82KSkgKFdvcmtmbG93IFN0YXR1czogW1N0b3JpZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvU3Rvcmllcy9pc3N1ZXM/cT1pcyUzQWlzc3VlK2lzJTNBb3BlbitsYWJlbCUzQXN0b3JpZXMpIC8gTm90ZXMpCiogKipPdmVybGF5cyoqIChbSXNzdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvU3Rvcmllcy9pc3N1ZXMvNykpIChXb3JrZmxvdyBTdGF0dXM6IFtJZGVhc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBaWRlYXMpIC8gVG9waWNzKQoqICoqRXh0ZW5zaW9ucyoqIChbSXNzdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvU3Rvcmllcy9pc3N1ZXMvOCkpIChXb3JrZmxvdyBTdGF0dXM6IFtJZGVhc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9TdG9yaWVzL2lzc3Vlcz9xPWlzJTNBaXNzdWUraXMlM0FvcGVuK2xhYmVsJTNBaWRlYXMpICAvIFRvcGljcykKCkZlZWwgZnJlZSB0byBzdWJtaXQgeW91ciBpZGVhIGZvciBhbiBPcGVuQVBJIHN0b3J5IGFzIGEgR2l0aHViIGlzc3VlLCBvciBqdW1wIGluIG9uIHRoZSBwcm9qZWN0IHBhZ2UgZm9yIG9uZSBhbHJlYWR5IGluIG1vdGlvbi4KCiMjIFN0b3J5IFdvcmtmbG93CldlIGFyZSBkZXZlbG9waW5nIGEgd29ya2Zsb3cgZm9yIHB1c2hpbmcgc3RvcmllcyB0byBhIGZpbmlzaCBzdGF0ZS4gTm90IGFsbCBzdG9yaWVzIHdpbGwgdWx0aW1hdGVseSBtYWtlIGl0IHRvIGEgZmluaXNoZWQgc3RhdGUsIGxldCBhbG9uZSB0aGUgT0FJIGJsb2csIGJ1dCB3ZSB1c2UgdGhpcyBwcm9jZXNzIHRvIGRldmVsb3Agc3RvcmllcyBhbmQgZXZvbHZlIHRoZW0gZm9yIGV2ZW50cyBsaWtlIEFTQywgY3JlYXRpb24gb2YgdmlkZW9zLCBhbmQgb3RoZXIgY2hhbm5lbHMuCgojIyMgSWRlYXMgLSBUaGUgc3VibWlzc2lvbiBvZiBhIG5ldyBpZGVhIGZvciBhIHN0b3J5LgoKKiAqKlRvcGljcyoqIC0gQSB0b3BpYyB0byB0ZWxsIGEgc3RvcnkgYWJvdXQuCiogKipQZW9wbGUqKiAtIFNvbWVvbmUgdG8gdGVsbCB0aGUgc3RvcnkuCiogKipDb21wYW5pZXMqKiAtIFdobyBhIHN0b3J5IGludm9sdmVzLgoKIyMjIFN0b3JpZXMgLSBNb3Zpbmcgc29tZXRoaW5nIGZyb20gaWRlYSB0byBzdG9yeS4KCiogKipJbnRlcnZpZXdzKiogLSBDb25kdWN0IGludGVydmlld3MgZm9yIHN0b3J5LgoqICoqTm90ZXMqKiAtIENvbXBpbGUgYW5kIHNoYXJlIG5vdGVzIGZvciBzdG9yeS4KKiAqKkRyYWZ0KiogLSBDcmVhdGUgYSBkcmFmdCBmb3IgYSBzaW5nbGUgc3RvcnkuCgojIyMgU3RydWN0dXJhbCAtIFRoZSB0aW1lIGZvciBtYWtpbmcgYmlnIGNoYW5nZXMuCgoqICoqWWVzIC8gTm8qKiAtIERvIHdlIGV2ZW4gZG8gdGhpcyBzdG9yeT8KKiAqKkJpZyBDaGFuZ2VzKiogLSBTaGlmdGluZyB0aGUgZW50aXJlIHN0cnVjdHVyZS4KKiAqKkN1bHRpdmF0ZSoqIC0gRG8gbW9yZSByZXNlYXJjaCBhbmQgZmxlc2ggb3V0LgoqICoqQ3JlYXRpdmUqKiAtIERvIHdlIG5lZWQgYW55IGltYWdlcz8KCiMjIyBFZGl0aW5nIC0gVGhlIHByb2Nlc3Mgb2YgcmVmaW5pbmcgYSBzdG9yeSBmb3IgcHVibGlzaGluZwoKKiAqKkNoYW5nZXMqKiAtIE5vIG1vcmUgYmlnIGNoYW5nZXMgdG8gYSBzdG9yeS4KKiAqKkNvcHkgRWRpdHMqKiAtIEludHJvZHVjaW5nIGVkaXRzIHRvIHRoZSBzdG9yeSBmcm9tIHRoZSBlZGl0b3IuCiogKipDcmVhdGl2ZSoqIC0gQWRkIGltYWdlcywgcXVvdGVzLCBsaW5rcywgZXRjLgoKIyMjIEdvIExpdmUgLSBCZWdpbiB0byBtb3ZlIHRvd2FyZHMgbWFraW5nIGxpdmUuCgoqICoqSW50ZXJuYWwgQXBwcm92YWwqKiAtIEdldCBpbnRlcm5hbCBhcHByb3ZhbHMuCiogKipFeHRlcm5hbCBBcHByb3ZhbCoqIC0gR2V0IG90aGVyIGFwcHJvdmFscy4KKiAqKlB1Ymxpc2gqKiAtIEdvIGxpdmUgd2l0aCBhIHN0b3J5IG9uIGJsb2cgJiBzb2NpYWwuCiogKipTb2NpYWwqKiAtIEFtcGxpZnkgdXNpbmcgc29jaWFsIG1lZGlhLgoKIyMgR2V0IEludm9sdmVkClRoaXMgaXMgYSBjb21tdW5pdHkgcHJvamVjdCB3aXRoaW4gdGhlIE9BSSBhbmQgd2UgYXJlIGxvb2tpbmcgZm9yIHN0b3J5IGlkZWEgc3VibWlzc2lvbnMgYXMgd2VsbCBhcyBmb2xrcyBpbiB0aGUgZm9sbG93aW5nIHJvbGVzIHRvIGhlbHAgbW92ZSBzdG9yaWVzIGZvcndhcmQ6CgoqICoqSW50ZXJ2aWV3ZXJzKiogLSBQZW9wbGUgd2hvIGNhbiBoZWxwIGludGVydmlldyBwcm92aWRlcnMsIHRvb2xpbmcgbWFrZXJzLCBhbmQgb3RoZXIgcGVvcGxlIHB1dHRpbmcgT3BlbkFQSSB0byB3b3JrLgoqICoqV3JpdGVycyoqIC0gUGVvcGxlIHdobyBjYW4gaGVscCBhY3R1YWxseSB0YWtlIGludGVydmlld3Mgb3IgdG9waWNzLCBkbyB0aGUgcmVzZWFyY2gsIGFuZCBwcm9kdWNlIHN0b3JpZXMgZm9yIHB1Ymxpc2hpbmcuCiogKipFZGl0b3JzKiogKiotIFBlb3BsZSB3aG8gY2FuIGhlbHAgY29weSBlZGl0IHN0b3JpZXMgdGhhdCBhcmUgc3VibWl0dGVkIG55IGF1dGhvcnMgYW5kIGhlbHAgdXMgcHJvZHVjZSBoaWdoIHF1YWxpdHkgc3Rvcmllcy4KCkZlZWwgZnJlZSB0byBqdW1wIGluIG9uIG9uZSBvZiB0aGUgcHJvamVjdHMgYWJvdmUsIG9yIHN1Ym1pdCBhIEdpdGh1YiBpc3N1ZSBpZiB5b3UgaGF2ZSBhbnkgY29tbWVudHMsIHF1ZXN0aW9ucywgb3IgZmVlZGJhY2sgYWJvdXQgc3Rvcmllcy4gSWYgeW91J2QgbGlrZSB0byBiZSBhZGRlZCBhcyBhIGNvbnRyaWJ1dG9yIHBsZWFzZSBzdWJtaXQgYW4gaXNzdWUgc3RhdGluZyBob3cgeW91J2QgbGlrZSB0byBoZWxwIG91dCBhbmQgd2UgY2FuIGFkZCB5b3UgYXMgYSBjb250cmlidXRvciB0byB0aGUgcmVwb3NpdG9yeS4K readmeEtag: '"8630431c01481edf446dd57bc75e98faabc6dcfe"' readmeLastModified: Fri, 18 Jun 2021 01:33:17 GMT repositoryId: 366088081 description: >- Submit ideas, writing, and publishing stories about the OpenAPI specification. created: '2021-05-10T15:21:31Z' updated: '2024-08-02T02:22:50Z' language: null archived: false stars: 3 watchers: 3 forks: 4 owner: OAI logo: https://avatars.githubusercontent.com/u/16343502?v=4 repoEtag: '"2ef91866bdffc00ad5445555ed5ceb69a3920aee78117751636ff1ee865bf0ca"' repoLastModified: Fri, 02 Aug 2024 02:22:50 GMT foundInMaster: true category: Testing id: af4943447a676919750a937a8b3f5fd6 - source: openapi3 tags repository: https://github.com/edenreich/kopgen v3: true id: 72f4700c4f5d0e434b9451e77261d6e1 repositoryMetadata: base64Readme: >- <p align="center">
  <img src="logo/kopgen.webp" alt="Kopgen Logo" width="400"/>
</p>
<h1 align="center">🦀 Kopgen 🦀</h1>

<p align="center">
<a href="https://github.com/edenreich/kopgen/actions"><img src="https://github.com/edenreich/kopgen/actions/workflows/ci.yml/badge.svg" alt="CI Status"/></a>
<a href="https://github.com/edenreich/kopgen/releases"><img src="https://img.shields.io/github/v/release/edenreich/kopgen?color=blue&style=flat-square" alt="Version"/></a>
<a href="https://github.com/edenreich/kopgen/actions"><img src="https://github.com/edenreich/kopgen/actions/workflows/docs.yml/badge.svg" alt="Documentation Status"/></a>
<a href="https://github.com/edenreich/kopgen/blob/main/LICENSE"><img src="https://img.shields.io/github/license/edenreich/kopgen?color=blue&style=flat-square" alt="License"/></a>
</p>

Welcome to the Kopgen tool! This project provides a generator for creating a Create, Read, Update and Delete (CRUD) Kubernetes operator from an OpenAPI Specification (OAS), all written in the powerful and efficient Rust language.

If you have a running API with a valid OpenAPI Specification but lack an operator to manage its complexity, this generator is the perfect solution. Kubernetes operators also align seamlessly with the GitOps methodology, enhancing your operational efficiency.

It's a well-established fact that using YAML to define API resources not only simplifies complexity but also enhances collaboration between developers and operations teams. This approach allows teams to concentrate on what truly matters - delivering value.

- [Documentation](#documentation)
- [Quick Start Guide](#quick-start-guide)
- [Contributing](#contributing)
- [Need Help?](#need-help)
- [Motivation](#motivation)
- [Why OpenAPI Specification?](#why-openapi-specification)

## Documentation

The documentation is available here: https://edenreich.github.io/kopgen/introduction.html

For documentation Github-Pages is being used.
It's built using rust mdbook which already comes pre-installed on the DevContainer.

If you need to make adjustment to the documentation you can serve it locally, run:

```bash
task docs
```

## Quick Start Guide

Before you proceed, ensure that [Docker](https://docs.docker.com/engine/install/) is installed on your system.

Download and install the latest kopgen CLI:

```sh
curl -sSL https://raw.githubusercontent.com/edenreich/kopgen/refs/heads/main/scripts/install.sh | sh
```

Start a new project in an empty folder:

```bash
kopgen init .
```

This will generate the project for the operator including a dev container environment, therefore it is recommended you use VScode.

Open the project in VScode, a prompt to open the project inside of a Dev Container will be shown, click `open`.

A Dev Container is essentially an encapsulate environment with all the necessary tools you would need for the development of this project.

To get started, follow these steps:

1. Make sure you have Docker and VSCode installed.
2. Run: `kopgen init <directory>`.
3. Open the directory in vscode: `code <directory>`.
4. You supposed to be prompted to open DevContainer, click on "Reopen in Container".
5. Configure: `cp .env.example .env`.
6. Generate the operator including all of its dependencies, run: `task generate`.
7. Run the operator: `task run`.

## Contributing

If you'd like to contribute to this project, we highly recommend using **Visual Studio Code**. VS Code comes with a pre-configured development environment through our dev container, ensuring you have all the necessary project tooling set up seamlessly.

Additionally, please install the Git pre-commit hooks. These hooks help maintain our project's coding standards by automatically checking your commits for compliance before they're finalized.

## Need Help?

If you're not sure what to do, just run `task --list-all` to see a list of all available tasks.

## Motivation

Why create a Kubernetes operator in Rust? Great question! Here are a few reasons:

1. **Performance**: Rust is known for its blazing fast performance. It's a system programming language that runs without any extra runtime or garbage collector. This makes Rust a great choice for writing Kubernetes operators where performance is key.

2. **Memory Safety**: Rust's unique ownership model guarantees memory safety without needing a garbage collector. This means you can write high-performance operators without worrying about manual memory management.

3. **Concurrency**: Rust has first-class support for concurrent programming. This is crucial for writing operators, which often need to manage multiple resources concurrently.

4. **Interoperability**: Rust can easily interoperate with C and other languages. This makes it easy to leverage existing libraries when writing your operator.

5. **Tooling**: Rust's tooling is top-notch. With Cargo, Rust's package manager and build system, managing dependencies and building your project is a breeze.

6. **Strong Community**: The Rust community is known for being friendly and helpful, which is always a plus when learning a new language or starting a new project.

## Why OpenAPI Specification?

The OpenAPI Specification (OAS), formerly known as Swagger, is a standard for defining APIs. It provides a way to describe the capabilities of your API in a machine-readable format. This has several benefits:

1. **Documentation**: With OAS, your API documentation is always up-to-date because it's generated directly from your API definition. This makes it easier for developers to understand what your API does and how to use it.

2. **Client SDK Generation**: Tools like the OpenAPI Generator can generate client SDKs in various languages from an OAS document. This means developers can start using your API in their language of choice without having to write a lot of boilerplate code.

3. **Server Stub Generation**: Similarly, you can generate server stubs from an OAS document. This can speed up the initial development of your API.

4. **Validation**: You can use your OAS document to validate API requests and responses. This helps catch errors before they become problems.

5. **Integration**: Because OAS is a standard, there are many tools that can import OAS documents and provide additional functionality, such as testing tools, API gateways, and more.

In this project, we use the OpenAPI Specification to generate the necessary types and controllers for our Kubernetes operator. This allows us to easily keep our operator in sync with the latest version of our API.
 readmeEtag: '"09121f087c8ea54ed77c010168f0ca2489ba3819"' readmeLastModified: Tue, 19 Nov 2024 23:56:13 GMT repositoryId: 631552666 description: >- A generator for creating a CRUD Kubernetes operator from OpenAPI Specification written in Rust. created: '2023-04-23T11:43:16Z' updated: '2025-07-02T11:28:07Z' language: Rust archived: false stars: 7 watchers: 1 forks: 2 owner: edenreich logo: https://avatars.githubusercontent.com/u/16985712?v=4 license: Apache-2.0 repoEtag: '"d4752f2fe10d0a1427b56873e6b8ca5f07bca1e1310065d2dfb39a252a1cabd7"' repoLastModified: Wed, 02 Jul 2025 11:28:07 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/edenreich/k8s-operator-generator - source: openapi3 tags repository: https://github.com/koriit/ktor-controllers-openapi v3: true repositoryMetadata: base64Readme: >- PSBLdG9yIENvbnRyb2xsZXJzIE9wZW5BUEkKCmltYWdlOmh0dHBzOi8vZ2l0aHViLmNvbS9Lb3JpaXQva3Rvci1jb250cm9sbGVycy1vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnlhbWwvYmFkZ2Uuc3ZnW0J1aWxkLCBsaW5rPSJodHRwczovL2dpdGh1Yi5jb20vS29yaWl0L2t0b3ItY29udHJvbGxlcnMtb3BlbmFwaS9hY3Rpb25zL3dvcmtmbG93cy9idWlsZC55YW1sIl0KaW1hZ2U6aHR0cHM6Ly93d3cuY29kZWZhY3Rvci5pby9yZXBvc2l0b3J5L2dpdGh1Yi9rb3JpaXQva3Rvci1jb250cm9sbGVycy1vcGVuYXBpL2JhZGdlW0NvZGVGYWN0b3IsbGluaz1odHRwczovL3d3dy5jb2RlZmFjdG9yLmlvL3JlcG9zaXRvcnkvZ2l0aHViL2tvcmlpdC9rdG9yLWNvbnRyb2xsZXJzLW9wZW5hcGldCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvY29kZSUyMHN0eWxlLSVFMiU5RCVBNC1GRjQwODEuc3ZnW2t0bGludCxsaW5rPWh0dHBzOi8va3RsaW50LmdpdGh1Yi5pby9dCgppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9jb20ua29yaWl0LmtvdGxpbi9rdG9yLWNvbnRyb2xsZXJzLW9wZW5hcGkuc3ZnP2xhYmVsPU1hdmVuJTIwQ2VudHJhbFtNYXZlbiBDZW50cmFsLCBsaW5rPSJodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9ZzolMjJjb20ua29yaWl0LmtvdGxpbiUyMiUyMEFORCUyMGE6JTIya3Rvci1jb250cm9sbGVycy1vcGVuYXBpJTIyIl0KaW1hZ2U6aHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9rb3JpaXQva3Rvci1jb250cm9sbGVycy1vcGVuYXBpW0dpdEh1Yl0KCldBUk5JTkc6IEZyb20gdmVyc2lvbiAwLjUuMCBhbGwgcGFja2FnZSBuYW1lcyBoYXZlIGJlZW4gcmVuYW1lZCB0byBtYXRjaCBuZXcgYXJ0aWZhY3QgZ3JvdXAgaWQuCgpTdXBwb3J0IGZvciBPcGVuQVBJMyBpbiBodHRwczovL2dpdGh1Yi5jb20vS29yaWl0L2t0b3ItY29udHJvbGxlcnNba3Rvci1jb250cm9sbGVyc10uCgpUaGUgaWRlYSBpcyB0byB2YWxpZGF0ZSB5b3VyIE9wZW5BUEkgc3BlYyBhZ2FpbnN0IHlvdXIgY29kZSBpbnN0ZWFkIG9mIGdlbmVyYXRpbmcgaXQgZnJvbSB5b3VyIGNvZGUuCgpbV0FSTklOR10KVGhpcyBsaWJyYXJ5IGlzICpXb3JrIEluIFByb2dyZXNzKi4gSXQgZG9lc24ndCBzdXBwb3J0IGFsbCAqcmVsZXZhbnQqIGVsZW1lbnRzIG9mIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBsaWtlCm9hdXRoLCBjYWxsYmFjayBvciBlbmNvZGluZyB5ZXQuIEhvd2V2ZXIsIGFsbCBzdHJ1Y3R1cmFsIGVsZW1lbnRzIHNob3VsZCBiZSBzdXBwb3J0ZWQuCgoKPT0gRXhhbXBsZQpbc291cmNlLGtvdGxpbl0KLS0tLQppbnN0YWxsKFJvdXRpbmcpIHsKICAgIHZhbCBvcGVuYXBpID0ge30uamF2YUNsYXNzLmdldFJlc291cmNlQXNTdHJlYW0oIi9vcGVuYXBpLnlhbWwiKS5yZWFkZXIoKS5yZWFkVGV4dCgpCgogICAgb3BlbkFQSUNvbnRyb2xsZXIob3BlbmFwaSkKICAgIHN3YWdnZXJVSUNvbnRyb2xsZXIoKQp9Ci0tLS0KCj09IFRlc3RpbmcKW3NvdXJjZSxrb3RsaW5dCi0tLS0KdmFsIHNlcnZlciA9IFRlc3RBcHBsaWNhdGlvbkVuZ2luZShzZXJ2ZXJDb25maWcoKSkKc2VydmVyLnN0YXJ0KCkKCkxPRy5pbmZvKCJBbmFseXppbmcgS3RvciBzZXJ2ZXIuLi4iKQp2YWwgYW5hbHl6ZXIgPSBLdG9yT3BlbkFQSUFuYWx5emVyKAogICAgICAgIGt0b3IgPSBzZXJ2ZXIuYXBwbGljYXRpb24sCiAgICAgICAgYmFzZVBhdGhzID0gbGlzdE9mKCIvYXBpIiksCiAgICAgICAgZGVmYXVsdEhlYWRlcnMgPSBsaXN0T2YoSHR0cEhlYWRlcihYUmVxdWVzdElkKSksCiAgICAgICAgZGVmYXVsdEVycm9yVHlwZSA9IEFwaUVycm9yOjpjbGFzcwopCgp2YWwgc291cmNlID0gYW5hbHl6ZXIuYW5hbHl6ZSgpCnNlcnZlci5zdG9wKDBMLCAwTCwgVGltZVVuaXQuTUlMTElTRUNPTkRTKQoKTE9HLmluZm8oIlJlYWRpbmcgT3BlbkFQSSBzcGVjLi4uIikKdmFsIGRvYyA9IE9wZW5BUElSZWFkZXIoKS5sb2FkKHt9LmphdmFDbGFzcy5nZXRSZXNvdXJjZUFzU3RyZWFtKCIvb3BlbmFwaS55YW1sIikpCgpMT0cuaW5mbygiVmFsaWRhdGluZyBzcGVjLi4uIikKdmFsIGVycm9ycyA9IE9wZW5BUElNYXRjaGVyKCkubWF0Y2goZG9jLCBzb3VyY2UpCgppZiAoZXJyb3JzLmlzTm90RW1wdHkoKSkgewogICAgTE9HLmluZm8oIlJlc3VsdCBvZiBzZXJ2ZXIgYW5hbHlzaXM6XG57fSIsIHNvdXJjZSkKCiAgICBlcnJvcnMuZm9yRWFjaCB7CiAgICAgICAgTE9HLmVycm9yKGl0KQogICAgfQoKICAgIGZhaWwoIlRoZXJlIGFyZSAke2Vycm9ycy5zaXplfSB2YWxpZGF0aW9uIGVycm9ycyEiKQoKfSBlbHNlIHsKICAgIExPRy5pbmZvKCJPSyEiKQp9Ci0tLS0K readmeEtag: '"8170c81300eca3c42ed213ee826e02d851f63a60"' readmeLastModified: Tue, 01 Feb 2022 00:25:38 GMT repositoryId: 217298102 description: Support for OpenAPI3 in ktor-controllers created: '2019-10-24T12:50:34Z' updated: '2023-09-15T12:46:53Z' language: Kotlin archived: false stars: 3 watchers: 1 forks: 0 owner: Koriit logo: https://avatars.githubusercontent.com/u/8916393?v=4 license: MIT repoEtag: '"85833dee171f236180302ab03b75328f1be10761b39f245ae4891326d6d568d3"' repoLastModified: Fri, 15 Sep 2023 12:46:53 GMT foundInMaster: true category: Data Validators id: d8110a35dbaeb0bf31f7edaeeb7db995 - source: openapi3 tags repository: https://github.com/greyteam2020/gooutsafe_microservice v3: true repositoryMetadata: base64Readme: >- IyBHb091dFNhZmUgd2l0aCBtaWNyb3NlcnZpY2VzCgohW0dpdEh1YiBXb3JrZmxvdyBTdGF0dXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3dvcmtmbG93L3N0YXR1cy9HcmV5VGVhbTIwMjAvR29PdXRTYWZlX21pY3Jvc2VydmljZS9kb2NrZXItY29tcG9zZS1hY3Rpb25zLXdvcmtmbG93P3N0eWxlPWZvci10aGUtYmFkZ2UpCiFbQ29kZWNvdl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9jb2RlY292L2MvZ2l0aHViL0dyZXlUZWFtMjAyMC9Hb091dFNhZmVfbWljcm9zZXJ2aWNlP3N0eWxlPWZvci10aGUtYmFkZ2UpCgojIyBUYWJsZSBvZiBDb250ZW50CgotIFtBcHAgQXJjaGl0ZWN0dXJlXSgjKQotIFtIb3cgQnVpbGQgYW5kIHJ1biB3aXRoIERvY2tlcl0oaHR0cHM6Ly9ncmV5dGVhbTIwMjAuZ2l0aHViLmlvL0dvT3V0U2FmZV9taWNyb3NlcnZpY2UvcnVuLXRlc3Qtb24tdGhlLWhvc3QtbWFjaGluZSkKLSBbRGV2ZWxvcGluZ10oaHR0cHM6Ly9ncmV5dGVhbTIwMjAuZ2l0aHViLmlvL0dvT3V0U2FmZV9taWNyb3NlcnZpY2UvR29PdXRTYWZlX21pY3Jvc2VydmljZSNkZXZlbG9waW5nKQotIFtBZGRpdGlvbmFsIGluZm9ybWF0aW9uXShodHRwczovL2dyZXl0ZWFtMjAyMC5naXRodWIuaW8vR29PdXRTYWZlX21pY3Jvc2VydmljZS9hZGRpdGlvbmFsLWluZm9ybWF0aW9uKQoKIyMgQXBwIEFyY2hpdGVjdHVyZQoKIVtdKGh0dHBzOi8vaS5pYmIuY28vaGMzVjFtay9waG90by0yMDIwLTExLTI2LTEwLTA5LTU4LmpwZykKCiFbXShodHRwczovL2kuaWJiLmNvL0NIUlIxMVcvU2VsZWN0aW9uLTA0Ni5wbmcpCgojIyBIb3cgY2xvbmUgaXQKCkVhY2ggbWljcm9zZXJ2aWNlcyBpcyBhIHNlcGFyYXRlIHJlcG9zaXRvcnkgdGhhdCB5b3UgY2FuIGNsb25lIHdpdGggdGhlIGZvbGxvd2luZyBjb21tYW5kcwoKYGBgYmFzaApnaXQgY2xvbmUgLS1yZWN1cnNlLXN1Ym1vZHVsZXMgaHR0cHM6Ly9naXRodWIuY29tL0dyZXlUZWFtMjAyMC9Hb091dFNhZmVfbWljcm9zZXJ2aWNlLmdpdApgYGAKCltTb3VyY2VdKGh0dHBzOi8vc3RhY2tvdmVyZmxvdy5jb20vYS8zNzk3MDYxLzcyOTA1NjIpCgpUaGUgc3VibW9kdWxlcyBhcmUgdGhlIG1pY3Jvc2VydmljZXMgd2l0aCB0aGUgZm9sbG93aW5nIHByb3ByaWV0aWVzOgoKLSBbRW1haWwgTWljcm9zZXJ2aWNlc10oaHR0cHM6Ly9naXRodWIuY29tL0dyZXlUZWFtMjAyMC9lbWFpbF9taWNyb3NlcnZpY2UpOiBSdW5uaW5nIG9uIHBvcnQgNTAwMSBhbmQgaW5jbHVkZSBzb21lIFtjZWxlcnldKGh0dHBzOi8vZG9jcy5jZWxlcnlwcm9qZWN0Lm9yZy9lbi9zdGFibGUvZ2V0dGluZy1zdGFydGVkL2ludHJvZHVjdGlvbi5odG1sKSB0YXNrcy4KLSBbVXNlciBNaWNyb3NlcnZpY2VzXShodHRwczovL2dpdGh1Yi5jb20vR3JleVRlYW0yMDIwL3VzZXJzX21pY3Jvc2VydmljZSk6IFJ1bm5pbmcgb24gcG9ydCA1MDAyOwotIFtSZXN0YXVyYW50cyBNaWNyb3NlcnZpY2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9HcmV5VGVhbTIwMjAvcmVzdGF1cmFudHNfbWljcm9zZXJ2aWNlKTogUnVubmluZyBvbiBwb3J0IDUwMDMgYW5kIGluY2x1ZGUgc29tZSBbY2VsZXJ5XShodHRwczovL2RvY3MuY2VsZXJ5cHJvamVjdC5vcmcvZW4vc3RhYmxlL2dldHRpbmctc3RhcnRlZC9pbnRyb2R1Y3Rpb24uaHRtbCkgYmFja2dyb3VuZCB0YXNrcy47Ci0gW0Jvb2tpbmcgTWljcm9zZXJ2aWNlXShodHRwczovL2dpdGh1Yi5jb20vR3JleVRlYW0yMDIwL2Jvb2tpbmdfbWljcm9zZXJ2aWNlKTogUnVubmluZyBvbiBwb3J0IDUwMDQKCiMjIEhvdyBCdWlsZCBhbmQgcnVuIHdpdGggRG9ja2VyCgpDbG9uZSB0aGUgcmVwb3NpdG9yeSB3aXRoIHRoZSBjb21tYW5kIGFib3ZlIGFuZCBydW4gaW4gdGhlIHJvb3QgZm9sZGVyIHRoZSBjb21tYW5kCmBkb2NrZXItY29tcG9zZSB1cGAKVGhpcyB3aWxsIG1ha2UgZG9ja2VyIGRvd25sb2FkcyBhbGwgdGhlIGZpbGUgbmVlZGVkIGFuZCBzdGFydCBidWlsZGluZyB0aGUgY29udGFpbmVycy4gCkFmdGVyIHRoYXQsIHlvdSBjYW4gYnJvd3NlIHRvIGh0dHA6Ly9sb2NhbGhvc3QvIHRvIHVzZSB0aGUgYXBwLgoKIyMgUnVuIHRlc3Qgb24gdGhlIGhvc3QgbWFjaGluZQoKWW91IGNhbid0IHdpdGhvdXQgZG9ja2VyIHRoaXMgdGltZSwgdG8gcnVuIGl0IG9uIHRoZSBob3N0IG1hY2hpbmUgd2l0aG91dCBkb2NrZXIgeW91IGNhbiBydW4gdGhlIG1vbm9saXRoIHZlcnNpb24KYXZhaWxhYmxlIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vR3JleVRlYW0yMDIwL0dvT3V0U2FmZV9QcmltZXIyMDIwKQoKIyMgRGV2ZWxvcGluZwoKRWFjaCBwcm9ncmFtbWVyIGhhcyBhIHBlcnNvbmFsIHN0eWxlIG9uIHdyaXRlIGNvZGUgYW5kIHdlIGFjY2VwdCB0aGlzLCBidXQgdG8gbWFrZSByZWFkYWJpbGl0eSB0aGUKY29kZSBmcm9tIGFsbCBjb21wb25lbnQgb2YgdGhlIHRlYW0sIHdlIHVzZWQgYSBnb29kIHRvb2wgdG8gZm9ybWF0IHRoZSBjb2RlIGluIGF1dG9tYXRpY2FsbHkuCgpJdCBpcyBbYmxhY2tdKGh0dHBzOi8vZ2l0aHViLmNvbS9wc2YvYmxhY2spLCBhbmQgaXQgaXMgaW5zdGFsbGVkIHdpdGggdGhlIHJlcXVpcmVtZW50cy50eHQKClRvIGZvcm1hdCB0aGUgY29kZSB5b3UgY2FuIHJ1biB0aGUgY29tbWFuZCBiZWxvdyBhZnRlciBgcGlwMyBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQgLS11c2VyYAoKYGJsYWNrIG1vbm9saXRoYAoKV2hlbiB5b3Ugc2VlIHRoZSBmb2xsb3dpbmcgbGluZSwgeW91IGFyZSBkb25lIHRvIHB1c2ggeW91ciBQUgoKQWxsIGRvbmUhIOKcqCDwn42wIOKcqAoKCiMjIEFkZGl0aW9uYWwgaW5mb3JtYXRpb24KCi0gRGVhZGxpbmUgLSBXZWRuZXNkYXksIDI1IE5vdmVtYmVyIDIwMjAsIDIzOjU5Cg== readmeEtag: '"6a38e50f5f93b7ba25d603dd9e86ab68f3b44c25"' readmeLastModified: Tue, 01 Mar 2022 10:29:30 GMT repositoryId: 312033852 description: >- :octocat: GoOutSafe a web app (with microservices architecture) that implements some cool features to survive during this covid19 Pandemic. This web app is developed during the Advanced Software Engineering course by the Univerisity of Pisa :computer: created: '2020-11-11T16:56:03Z' updated: '2023-04-04T04:13:22Z' language: Python archived: false stars: 3 watchers: 2 forks: 3 owner: GreyTeam2020 logo: https://avatars.githubusercontent.com/u/73334241?v=4 repoEtag: '"770c31ec6818475901c161eebfa4fc6fe77956418f6a0e6392fa30f29edef4c3"' repoLastModified: Tue, 04 Apr 2023 04:13:22 GMT foundInMaster: true category: - Description Validators - Server Implementations id: eed683d5286fd183b4fd5b4e297b83f1 - source: openapi3 tags repository: https://github.com/okaluzny/app-assistant v3: true id: 54199e746a1f9db1639f71e53fd9b498 repositoryMetadata: base64Readme: >- IyMgYXBwLWFzc2lzdGFudAoKVGhpcyBpcyBhIHNpbXBsZSBsb2dpc3RpYyBhcHBsaWNhdGlvbiBkZW1vbnN0cmF0aW5nIHNvbWUgY2FwYWJpbGl0aWVzIG9mIFNwcmluZyBCb290IGFzIGJhY2tlbmQgc2VydmljZSBjYXBhYmlsaXRpZXMgaW5jbHVkZToKKiBBcHAgYXJjaGl0ZWN0dXJlOwoqIERhdGFiYXNlIG1pZ3JhdGlvbiBhbmQgc2NoZW1hIGNoYW5nZSBtYW5hZ2VtZW50IHVzaW5nIExpcXVpYmFzZTsKKiBldGMuCgojIyBIb3cgaXQgd29ya3M6CioqMS4gRG9ja2VyLiBGaXJzdCB5b3UgbmVlZCB0byBpbnN0YWxsIGRvY2tlcioqCiogRG93bmxvYWQgRG9ja2VyIFtIZXJlXShodHRwczovL2RvY3MuZG9ja2VyLmNvbS9kb2NrZXItZm9yLXdpbmRvd3MvaW5zdGFsbC8pLiBIaW50OiBFbmFibGUgSHlwZXItViBmZWF0dXJlIG9uIHdpbmRvd3MgYW5kIHJlc3RhcnQ7CiogVGhlbiBvcGVuIHBvd2Vyc2hlbGwgYW5kIGNoZWNrOgpgYGBiYXNoCmRvY2tlciBpbmZvCmBgYApvciwgYW5kIHlvdSBzZWUgdmVyc2lvbnMgZG9ja2VyICYgZG9ja2VyIGNvbXBvc2UKYGBgYmFzaApkb2NrZXIgLXYKYGBgCmBgYGJhc2gKZG9ja2VyLWNvbXBvc2UgLXYKYGBgCgoqKjIuIFNwcmluZyBib290IGFwcCoqCiogQ2xvbmUgdGhlIHJlcG9zaXRvcnk6CmBgYGJhc2gKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9PS2FsdXpueS9taWNyb3NlcnZpY2UtYXNzaXN0YW50LmdpdApgYGAKKiBCdWlsZCB0aGUgbWF2ZW4gcHJvamVjdDoKYGBgYmFzaAptdm4gY2xlYW4gaW5zdGFsbApgYGAKKiBOb3cgcnVuOgpgYGBiYXNoCmRvY2tlci1jb21wb3NlIHVwCmBgYAoKQXBwZW5kaXggQS4KCkFsbCBjb21tYW5kcyBzaG91bGQgYmUgcnVuIGZyb20gcHJvamVjdCByb290ICh3aGVyZSBkb2NrZXItY29tcG9zZS55bWwgbG9jYXRlcykK readmeEtag: '"8afe9efa60365148dd1320f36456e6620ea495c7"' readmeLastModified: Tue, 28 Nov 2023 08:26:56 GMT repositoryId: 229004551 description: null created: '2019-12-19T07:59:29Z' updated: '2023-08-28T21:04:33Z' language: Java archived: false stars: 3 watchers: 1 forks: 1 owner: OKaluzny logo: https://avatars.githubusercontent.com/u/13517065?v=4 repoEtag: '"4332b203898ea744c1020b96dd1d053b523ffe45f8fd95c98aff99927b576101"' repoLastModified: Mon, 28 Aug 2023 21:04:33 GMT category: Server Implementations oldLocations: - https://github.com/okaluzny/microservice-assistant foundInMaster: true - source: openapi3 tags repository: https://github.com/koumoul-dev/soas v3: true repositoryMetadata: base64Readme: >- IyBTZW1hbnRpYyBPcGVuIEFQSSBTcGVjaWZpY2F0aW9uIGxpYnJhcnkKSG9sZHMgdXRpbGl0eSBmdW5jdGlvbnMgdG8gbWFuaXB1bGF0ZSBhbiBBUEkgZGVzY3JpYmVkIHdpdGggW09wZW5BUEkgMy4wIHNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kKSBhbmQgW3NlbWFudGljYWxseSBhbm5vdGF0ZWRdKGh0dHA6Ly93d3cuaW50ZWxsaWdlbmNlLnR1Yy5nci9+cGV0cmFraXMvcHVibGljYXRpb25zL1NPQVM0LnBkZikuCgojIyBRdWljayBzdGFydApgYGAKbnBtIGluc3RhbGwgLS1zYXZlCmBgYAoKYGBgamF2YXNjcmlwdAppbXBvcnQgc29hc0xvYWRlciBmcm9tICdzb2FzJwppbXBvcnQgYXBpRG9jIGZyb20gJ3BhdGgtdG8vbXktYXBpLmpzb24nCgpjb25zdCBzb2FzID0gc29hc0xvYWRlcihhcGlEb2MpCgpjb25zdCBlbmRQb2ludHMgPSBzb2FzLmVuZFBvaW50cygpCmNvbnN0IGFjdGlvbnMgPSBzb2FzLmFjdGlvbnMoKQoKY29uc3QgaW5wdXQxID0gewogICJodHRwOi8vc2NoZW1hLm9yZy9zdHJlZXRBZGRyZXNzIjogIjcgUnVlIFZpY3RvciBTY2hvZWxjaGVyIiwKICAiaHR0cDovL3JkZi5pbnNlZS5mci9kZWYvZ2VvI2NvZGVDb21tdW5lIjoiMjIwNTAiCn0KY29uc3Qgb3V0cHV0MSA9IGF3YWl0IHNvYXMuZXhlY3V0ZSgnZ2V0Q29vcmQnLCBpbnB1dDEpCi8vIG91dHB1dDEgaXMgOgovLyB7Imh0dHA6Ly9zY2hlbWEub3JnL2lkZW50aWZpZXIiOiIxIiwKLy8gImh0dHA6Ly9zY2hlbWEub3JnL2xhdGl0dWRlIjoiNDguMyIsCi8vICJodHRwOi8vc2NoZW1hLm9yZy9sb25naXR1ZGUiOiItMy40In0KCi8vIHdvcmtzIHdpdGggc3RyZWFtcyB0b28KY29uc3QgaW5wdXQyID0gZnMuY3JlYXRlUmVhZFN0cmVhbShwYXRoLmpvaW4oX19kaXJuYW1lLCAnYWRkcmVzc2VzLm5kanNvbicpKQogICAgICAgICAgICAgICAgICAucGlwZShtaW1lVHlwZVN0cmVhbSgnYXBwbGljYXRpb24veC1uZGpzb24nKS5wYXJzZXIoKSkKY29uc3Qgb3V0cHV0MiA9IGF3YWl0IHNvYXMuZXhlY3V0ZSgncG9zdENvb3JkcycsIGlucHV0KQpvdXRwdXQyLnBpcGUobWltZVR5cGVTdHJlYW0oJ2FwcGxpY2F0aW9uL3gtbmRqc29uJykuc2VyaWFsaXplcigpKS5waXBlKHByb2Nlc3Muc3Rkb3V0KQouLi4KYGBgCiMjIERvY3VtZW50YXRpb24KCiMjIyBDb25zdHJ1Y3RvcgpTZWUgdGhlIFtPcGVuQVBJIDMuMCBzcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMC5tZCkgZm9yIHRoZSBjb3JyZWN0IGZvcm1hdCBvZiB0aGUgQVBJIGRlc2NyaXB0aW9uLgoKYGBgamF2YXNjcmlwdAppbXBvcnQgc29hc0xvYWRlciBmcm9tICdzb2FzJwppbXBvcnQgYXBpRG9jIGZyb20gJ3BhdGgtdG8vbXktYXBpLmpzb24nCgovLyBhcGlEb2MgaXMgYSBKU09OIGRlc2NyaWJpbmcgYW4gQVBJIHdpdGggdGhlIE9wZW5BcGkgMy4wIHNwZWNpZmljYXRpb24KLy8gVG8ga2VlcCB0aGlzIGxpYnJhcnkgbGlnaHQsIHNjaGVtYSB2YWxpZGF0aW9uIGlzIG5vdCBoYW5kbGVkCmNvbnN0IHNvYXMgPSBzb2FzTG9hZGVyKGFwaURvYykKYGBgCgojIyMgZW5kUG9pbnRzKCkKTGlzdCBBUEkgZW5kcG9pbnRzLiBSZXR1cm4gYW4gYXJyYXkgb2Ygb2JqZWN0cyB3aXRoIHRoZSBmb2xsb3dpbmcgcHJvcGVydGllcyA6CiAqICoqbWV0aG9kKiogOiB0aGUgSFRUUCBtZXRob2QKICogKipwYXRoKiogOiB0aGUgcGF0aCB0aGlzIGVuZHBvaW50IHJlZmVycyB0bwogKiAqKm9wZXJhdGlvbioqIDogdGhlIG9wZXJhdGlvbiBkZXNjcmliZWQgYnkgdGhpcyBlbmRwb2ludC4gU2VlIHRoZSBbT3BlcmF0aW9uIE9iamVjdF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjAubWQjb3BlcmF0aW9uT2JqZWN0KQoKIyMjIGFjdGlvbnMoKQpMaXN0IGFsbCBhY3Rpb25zLiBBbiBhY3Rpb24gaXMgYW4gQVBJIGVuZHBvaW50IHNlbWFudGl6ZWQgd2l0aCBbYW5ub3RhdGlvbnMgZGVzY3JpYmVkIGhlcmVdKGh0dHA6Ly93d3cuaW50ZWxsaWdlbmNlLnR1Yy5nci9+cGV0cmFraXMvcHVibGljYXRpb25zL1NPQVM0LnBkZikuIEl0IGhhcyB0aGUgZm9sbG93aW5ncyBwcm9wZXJ0aWVzIDoKICogKippZCoqIDogdGhlIGlkZW50aWZpZXIgb2YgdGhlIGFjdGlvbgogKiAqKnR5cGUqKiA6IHRoZSB0eXBlIG9mIGFjdGlvbi4gVGhpcyBzaG91bGQgYmUgYSBzdWJ0eXBlIG9mIFtzY2hlbWEub3JnIEFjdGlvbl0oaHR0cDovL3NjaGVtYS5vcmcvQWN0aW9uKSwgb3Igc29tZXRoaW5nIHNpbWlsYXIuCiAqICoqb3BlcmF0aW9uKiogOiBwYXRoIGFuZCBtZXRob2QKICogKippbnB1dCoqIDogYW4gbWFwIG9mIGNvbmNlcHRzID0+IHBhcmFtZXRlcnMgbG9jYXRpb24KICogKippbnB1dENvbGxlY3Rpb24qKiA6IGJvb2xlYW4gaW5kaWNhdGluZyBpZiB0aGUgYWN0aW9uIHRha2VzIGEgY29sbGVjdGlvbiBhcyBpbnB1dAogKiAqKmlucHV0Qm9keVR5cGVzKiogOiB0aGUgcG9zc2libGUgbWltZS10eXBlcyBmb3IgdGhlIHJlcXVlc3QgYm9keSwgaWYgdGhlcmUgaXMgb25lCiAqICoqb3V0cHV0KiogOiBhbiBtYXAgb2YgY29uY2VwdHMgPT4gb2JqZWN0IGZpZWxkcwogKiAqKm91dHB1dFNjaGVtYSoqOiBUaGUgSlNPTiBzY2hlbWEgb2Ygb3V0cHV0CiAqICoqb3V0cHV0Q29sbGVjdGlvbioqIDogYm9vbGVhbiBpbmRpY2F0aW5nIGlmIHRoZSBhY3Rpb24gb3V0cHV0cyBhIGNvbGxlY3Rpb24KICogKipvdXRwdXRCb2R5VHlwZXMqKiA6IHRoZSBwb3NzaWJsZSBtaW1lLXR5cGVzIGZvciB0aGUgcmVzcG9uc2UgYm9keSwgaWYgdGhlcmUgaXMgb25lCiAqICoqc3VtbWFyeSoqIDogdGhlIHN1bW1hcnkgb2YgdGhlIG9wZXJhdGlvbiB0aGlzIGFjdGlvbiByZWZlcnMgdG8KICogKipjYW5Vc2UqKiA6IGEgYm9vbGVhbiBpbmRpY2F0aW5nIGlmIHRoaXMgbGlicmFyeSBpcyBhYmxlIHRvIHVzZSB0aGUgYWN0aW9uIG9yIG5vdAoKIyMjIGV4ZWN1dGUoYWN0aW9uSWQgWywgaW5wdXQsIHNlcnZlcl0pClJldHVybiBhIHByb21pc2UuIE9uY2UgcmVzb2x2ZWQsIGNhbiBiZSBhbiBvYmplY3Qgb3IgYSBzdHJlYW0gb2Ygb2JqZWN0cyB3aGljaCBhcmUgbWFwcyBvZiBjb25jZXB0cyBhbmQgdGhlaXIgdmFsdWUuCgoqKmFjdGlvbklkKiogaXMgdGhlIGlkIG9mIGFuIGFjdGlvbiBsaXN0ZWQgd2l0aCBgYWN0aW9ucygpYC4gKippbnB1dCoqIGlzIGlzIGEgbWFwIG9mIGNvbmNlcHRzIGFuZCB0aGVpciB2YWx1ZS4gSXQgY2FuIGJlIGEgc3RyZWFtIG9mIG9iamVjdHMgdG9vLiAqKnNlcnZlcioqIGlzIGFuIHVybCB0byBxdWVyeSwgaWYgbm90IHByb3ZpZGVkIGl0IHdpbGwgYmUgcmVzb2x2ZWQgdG8gdGhlIHVybCBvZiB0aGUgZmlyc3QgW1NlcnZlciBPYmplY3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kI3NlcnZlck9iamVjdCkgb2YgdGhlIGBzZXJ2ZXJzYCBwcm9wZXJ0eSBvZiB0aGUgQVBJIGRlc2NyaXB0aW9uLgo= readmeEtag: '"8646eda7d1137d4c3aad05f2796069c510c1636e"' readmeLastModified: Wed, 12 May 2021 11:30:06 GMT repositoryId: 112844830 description: Semantic Open API Specification library created: '2017-12-02T13:37:23Z' updated: '2021-05-12T11:30:18Z' language: JavaScript archived: false stars: 3 watchers: 2 forks: 0 owner: koumoul-dev logo: https://avatars.githubusercontent.com/u/16051219?v=4 license: MIT repoEtag: '"a82d0d1bece3a3dc9718066202396fdfa5d218db04321202543defcb334dcd87"' repoLastModified: Wed, 12 May 2021 11:30:18 GMT foundInMaster: true category: Parsers id: 68d9f6193089b720b24dc37563d11597 - source: openapi3 tags repository: https://github.com/himenon/openapi-typescript-practice v3: true repositoryMetadata: base64Readme: >- IyBAaGltZW5vbi9vcGVuYXBpLXR5cGVzY3JpcHQtcHJhY3RpY2UKCi0gW+ODieOCreODpeODoeODs+ODiCAtIGh0dHBzOi8vaGltZW5vbi5naXRodWIuaW8vb3BlbmFwaS10eXBlc2NyaXB0LXByYWN0aWNlXShodHRwczovL2hpbWVub24uZ2l0aHViLmlvL29wZW5hcGktdHlwZXNjcmlwdC1wcmFjdGljZSkKCiMjIERldmVsb3BtZW50Cgp8IHNjcmlwdHMgICAgICAgICB8IGRlc2NyaXB0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCA6LS0tLS0tLS0tLS0tLS0gfCA6LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLSB8CnwgYnVpbGQgICAgICAgICAgIHwgYnVpbGQ6Y29kZSwgYnVpbGQ6ZG9jcyDjgpLpoIbmrKHlrp/ooYwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgYnVpbGQ6Y29kZSAgICAgIHwgVHlwZVNjcmlwdCDjga7jgrPjg7zjg4njgpLnlJ/miJAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBidWlsZDpkb2NzICAgICAgfCDjg4njgq3jg6Xjg6Hjg7Pjg4jjgpLnlJ/miJAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgY2xlYW4gICAgICAgICAgIHwgYnVpbGQg6Zai6YCj44Gu44OV44Kh44Kk44Or44O744OH44Kj44Os44Kv44OI44Oq44KS5YmK6ZmkICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwganNvbjJzY2hlbWEgICAgIHwganNvbiDjgYvjgokgT3BlbkFwaSAzLjAg44GuIHNjaGVtYSDjgpLlkJDjgY3lh7rjgZnjgILlt6jlpKfjgaoganNvbiDjgarjganjgavliKnnlKjjgZnjgovjgajoia/jgYQgICAgIHwKfCBtb2NrOnNlcnZlciAgICAgfCBgeWFybiBydW4gbW9jazpzZXJ2ZXIgbG9jYWxob3N0YOOBqOWun+ihjOOBmeOCi+OBk+OBqOOBpyBBUEkgTW9jayBTZXJ2ZXIg44KS6LW35YuVIHwKfCBwdWJsaXNoOmdocGFnZXMgfCBkb2NzIOOCkiBnaHBhZ2VzIOOBqyBwdWJsaXNoICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCB0cyAgICAgICAgICAgICAgfCB0cy1ub2RlIOOCkuWun+ihjOOBmeOCiyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKCiMjIEJ1aWxkIFBpcGVsaW5lCgojIyMgeWFybiBidWlsZDpjb2RlCgohW2J1aWxkOmNvZGVdKC4vYnVpbGQtcGlwZWxpbmUucG5nKQoKIyMjIFJlbGVhc2UKCnJlbGVhc2UgdmVyc2lvbgoKYGBgYmFzaAp5YXJuIHJ1biBsZXJuYSB2ZXJzaW9uIC0teWVzCmBgYAoKIyMgVG9vbHMKCi0gaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPXBoaWxvc293YWZmbGUub3BlbmFwaS1kZXNpZ25lcgotIGh0dHBzOi8vaHViLmRvY2tlci5jb20vci9zd2FnZ2VyYXBpL3N3YWdnZXItdWkvCi0gaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQHN0b3BsaWdodC9wcmlzbS1jbGkKLSBodHRwczovL2dpdGh1Yi5jb20vc3RvcGxpZ2h0aW8vcHJpc20KCiMjIFJlZmVyZW5jZQoKLSBbVlNDb2RlIOOBp+ODquODleOCoeOCr+OCv+ODquODs+OCsOODu+S/neWuiOOBmeOCiyBPcGVuQVBJIC0gUWlpdGFdKGh0dHBzOi8vcWlpdGEuY29tL3RNaW5hbWkvaXRlbXMvNWIxYTkyMWU4MmI0Yzc5NzljZDEpCi0gW+S/uueahOOAkE9BU+OAkeOBqOOBruWQkeOBjeWQiOOBhOaWuSAo54iG6YCf44GnIE9wZW5BUEkg44Go5Y+L6YGU44Gr44Gq44KN44GGKV0oaHR0cHM6Ly90ZWNoLWJsb2cub3B0aW0uY28uanAvZW50cnkvMjAyMC8wNC8xMy8xMDAwMDApCgojIyBMSUNFTlNFCgpNSVQK readmeEtag: '"856dfa0c3657e95f8346bd85ef85713f531203e2"' readmeLastModified: Sun, 31 Jan 2021 08:30:24 GMT repositoryId: 305648117 description: OpenAPIを利用したAPI作成と、開発、リリースフローまでの仕組みづくりの練習 created: '2020-10-20T08:56:12Z' updated: '2022-09-14T18:32:19Z' language: HTML archived: false stars: 3 watchers: 0 forks: 0 owner: Himenon logo: https://avatars.githubusercontent.com/u/6715229?v=4 license: MIT repoEtag: '"12fdff1941d3dcd3268e80ff0b6168325a54880cdd69114371e2169abc255c74"' repoLastModified: Wed, 14 Sep 2022 18:32:19 GMT foundInMaster: true category: SDK id: d86d9ef52052c9adffcee267a44e13f2 - source: openapi3 tags repository: https://github.com/nitianabhigyan/smitch-api v3: true id: 1b3e47b8baebff63549eb3f48deb1b78 repositoryMetadata: base64Readme: >- IyBTbWl0Y2gtQVBJCkEgKipwb3N0bWFuIGNvbGxlY3Rpb24qKiBhbmQgKipvcGVuYXBpIHNwZWMuKiogdG8gY29udHJvbCBzbWl0Y2ggc21hcnQgZGV2aWNlcyBpbmNsdWRpbmcgbGlnaHRzIGFuZCBzd2l0Y2hlcy4KCk9mZmljYWwgZG9jdW1lbnRhdGlvbiBmb3IgdGhlIFNtaXRjaCBTbWFydCBhcHAgY29udHJvbDogW2h0dHBzOi8vZG9jcy5kZXZlbG9wZXIubXlzbWl0Y2guY29tXShodHRwczovL2RvY3MuZGV2ZWxvcGVyLm15c21pdGNoLmNvbSkuIDxicj4KT2ZmaWNhbCAiR2V0IHN0YXJ0ZWQiIGd1aWRlOiBbR2V0LVN0YXJ0ZWRdKGh0dHBzOi8vZG9jcy5kZXZlbG9wZXIubXlzbWl0Y2guY29tL2RldmVsb3Blci1wb3J0YWwvZ2V0LXN0YXJ0ZWQpLgoKIyMgSG93IHRvIHVzZSB0aGUgcG9zdG1hbiBjb2xsZWN0aW9uCiMjIyBQcmVyZXF1aXNpdGVzClBsZWFzZSBmYW1pbGlhcmlzZSB5b3Vyc2VsZiB3aXRoIHRoZSBiYXNpYyBwcmVyZXF1aXNpdGUgYXQ6IFtQcmVyZXF1aXNpdGVzXShodHRwczovL2RvY3MuZGV2ZWxvcGVyLm15c21pdGNoLmNvbS8jMC4tcHJlcmVxdWlzaXRlcykKLSBSZWdpc3RlciBhIGRldmVsb3BlciBhY2NvdW50IGF0OiBbRGV2ZWxvcGVyLVBvcnRhbF0oaHR0cHM6Ly9kZXZlbG9wZXIubXlzbWl0Y2guY29tKSBieSBzZWxlY3RpbmcgU2lnbiBVcCAoU2lnbiBJbiBpZiB5b3UgYWxyZWFkeSBjcmVhdGVkIGFuIGFjY291bnQpCi0gQWRkIGFuIEFwcCBpbiB5b3VyIGRldmVsb3BlciBhY2NvdW50IGFuZCBjcmVhdGUgYSAiVEVTVCIgQVBJIEtFWS4gTW9yZSBkZXRhaWxzOiBbYXBpLWtleXNdKGh0dHBzOi8vZG9jcy5kZXZlbG9wZXIubXlzbWl0Y2guY29tL2FwcC9hcGkta2V5cykuIFNhdmUgdGhpcyBhcGkta2V5IHNvbWV3aGVyZSBzYWZlLgotIExpbmsgYSAidGVzdGVyIiB0byB5b3VyIGFwcGxpY2F0aW9uLiBNb3JlIGRldGFpbHM6IFt1c2VyLXRlc3Rlcl0oaHR0cHM6Ly9kb2NzLmRldmVsb3Blci5teXNtaXRjaC5jb20vdXNlcnMvdXNlci10ZXN0ZXIpLCBtYWtlIHN1cmUgdGhlIHRlc3RlcidzIGFjY291bnQgdXNlZCBoZXJlIGlzIHRoZSBzYW1lIGFzIHRoZSBvbmUgeW91IHVzZSB0byBzaWduIGluIHRoZSBTbWl0Y2ggbW9iaWxlIGFwcC4KCiMjIyBVc2luZyB0aGUgY29sbGVjdGlvbi4KCjEuIExvYWQgdGhlIGNvbGxlY3Rpb24gbG9jYXRlZCBhdCBbY29sbGVjdGlvbl0oLi9TbWl0Y2gmIzMyO1Bvc3RtYW4mIzMyO0FQSSYjMzI7Y29sbGVjdGlvbi5wb3N0bWFuX2NvbGxlY3Rpb24uanNvbikgaW4geW91ciBsb2NhbC93ZWIgaW5zdGFuY2Ugb2YgUG9zdG1hbi4KMi4gT3BlbiB0aGUgIlNtaXRjaCBQb3N0bWFuIEFQSSBjb2xsZWN0aW9uIiwgc2VsZWN0IHRoZSB2YXJpYWJsZXMgdGFiIGFuZCBzZXQgdGhlICJhcGlfa2V5IiB2YXJpYWJsZSdzIHZhbHVlIHRvIHlvdXIgYXBpX2tleSBvYnRhaW5lZCBmcm9tLgozLiBTZWxlY3QgdGhlICJFdmVyeXRoaW5nX2lzX3ZhbGlkIiByZXF1ZXN0IGFuZCBzZW5kIGl0LiBJZiBldmVyeXRoaW5nIHdhcyBkb25lIGNvcnJlY3RseSwgdGhpcyBzaG91bGQgcmV0dXJuIGEgICJzdGF0dXMiOiAic3VjY2VzcyIuIElmIG5vdCwgcGxlYXNlIHZhbGlkYXRlIHlvdXIgc3RlcHMgYWdhaW4uCjQuIFNlbGVjdCB0aGUgImxpc3Rfb2ZfdXNlcnMiIHJlcXVlc3QgYW5kIHNlbmQgaXQuIE9idGFpbiB0aGUgdXNlcl9pZCB2YXJpYWJsZSBvZiB0aGUgZGVzaXJlZCB1c2VyIChtYWtlIHN1cmUgdGhlIHNhaWQgYWNjb3VudCBpcyBhZGRlZCBhcyB0ZXN0ZXIgYXMgcGVyIGFib3ZlKSwgYWRkIHRoaXMga2V5J3MgdmFsdWUgdG8geW91ciBjb2xsZWN0aW9uJ3MgdmFyaWFibGUgInVzZXJfaWQiIHZhcmlhYmxlJ3MgdmFsdWUuCjUuIFRvIGNvbnRyb2wgYSBkZXZpY2UsIHlvdSBmaXJzdCBuZWVkIHRvIGlkZW50aWZ5IGl0J3MgImRldmljZV9pZCIgYW5kIGFkZCBpdCBpbiBjb2xsZWN0aW9uJ3MgdmFyaWFibGVzLiBUbyBvYnRhaW4gYSBsaXN0IG9mIGFsbCBkZXZpY2VzIGFkZGVkIGluIHVzZXIncyBhY2NvdW50IHBsZWFzZSB1c2UgdGhlICJnZXRfdXNlcl9kZXRhaWxzIiByZXF1ZXN0LgoKIyMgVG8gQmUgRG9uZQogLSB+fkNoZWNrIGZlYXNpYmlsaXR5IG9mIGNvbnZlcnRpbmcgdGhpcyB0byBvcGVuYXBpLn5+ICoqUmVzdWx0Kio6IERvYWJsZS4KIC0gfn5Db252ZXJ0IHRoaXMgdG8gT3BlbkFQSSB2ZXJzaW9uIDMgc3BlYy5+fgogLSBVc2Ugb3BlbmFwaSBhbmQgc3dhZ2dlci11aSB0byBob3N0IHRoaXMgaW4gdGhlIGZvcm0gb2YgYSBnaXRodWIgcGFnZS4KIC0gQ3VzdG9taXNlIHRoZSBvcGVuYXBpLnlhbWwgc3BlYyB0byBiZXR0ZXIgc3VpdGUgdGhlIG5lZWRzIG9mIHRoZSB2ZXJzaW9uLgogLSBEbyB0ZXN0IHJ1bnMKIC0gQWRkIGRvY3VtZW50YXRpb24gZm9yIHVzaW5nIG9wZW5hcGkueWFtbCB0byBjcmVhdGUgU0RLcywgYW5kIGZpbmFsbHkgYmUgYXQgcGVhY2Ug4p2k77iPCg== readmeEtag: '"096c1863cbf45dae8a1e3dc591aa5ce56d5c2bba"' readmeLastModified: Wed, 17 Aug 2022 07:40:12 GMT repositoryId: 507033057 description: >- A postman collection to control switch smart devices. Offical documentation for the Smitch Smart app control: https://docs.developer.mysmitch.com created: '2022-06-24T14:06:08Z' updated: '2025-08-21T12:31:35Z' language: null archived: false stars: 4 watchers: 1 forks: 1 owner: nitianabhigyan logo: https://avatars.githubusercontent.com/u/51279790?v=4 license: MIT repoEtag: '"b1337c7912c862115258eec3e1469624c8a5244d63b1000913531f137ed70b6e"' repoLastModified: Thu, 21 Aug 2025 12:31:35 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/egodigital/swagger-ui-cli v3: true repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyLXVpLWNsaQoKIyMgUmVwb3NpdG9yeSBoYXMgbW92ZWQgdG8gW2Vnb21vYmlsZS9zd2FnZ2VyLXVpLWNsaV0oaHR0cHM6Ly9naXRodWIuY29tL2Vnb21vYmlsZS9zd2FnZ2VyLXVpLWNsaSkK readmeEtag: '"991fecae6fe1e2df3c8dc26fe87c0243695121a8"' readmeLastModified: Mon, 31 May 2021 11:54:09 GMT repositoryId: 291583353 description: A standalone CLI application, serving Swagger UIs via a HTTP server. created: '2020-08-31T01:17:43Z' updated: '2023-01-28T08:35:06Z' language: TypeScript archived: true stars: 3 watchers: 1 forks: 2 owner: egodigital logo: https://avatars.githubusercontent.com/u/41436054?v=4 license: GPL-3.0 repoEtag: '"8bb23d737208cb4e35d45ac7069f6864e354b13b5fd43f00005dd82f5e97c2b0"' repoLastModified: Sat, 28 Jan 2023 08:35:06 GMT foundInMaster: true category: - Low-level Tooling - Code Generators id: 1db9e8eaf18d0783bce830fbe3ccb1db - source: openapi3 tags repository: https://github.com/wtask-go/mixpanel v3: true repositoryMetadata: base64Readme: >- IyBNaXhwYW5lbAoKTWl4cGFuZWwgQVBJIGNsaWVudCBmb3IgR28KClshW0dvIFJlZmVyZW5jZV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2JhZGdlL2dpdGh1Yi5jb20vd3Rhc2stZ28vbWl4cGFuZWwuc3ZnKV0oaHR0cHM6Ly9wa2cuZ28uZGV2L2dpdGh1Yi5jb20vd3Rhc2stZ28vbWl4cGFuZWwpCgojIyBSZXF1aXJlbWVudHMKCiogR28gMS4xNiBhbmQgYWJvdmUuIFRoaXMgbW9kdWxlIHVzZXMgYGVtYmVkYCBwYWNrYWdlIGludHJvZHVjZWQgZm9yIEdvIDEuMTYKCiMjIERlcGVuZGVuY2llcwoKKiBbZ2l0aHViLmNvbS9zYW50aG9zaC10ZWt1cmkvanNvbnNjaGVtYV0oaHR0cHM6Ly9naXRodWIuY29tL3NhbnRob3NoLXRla3VyaS9qc29uc2NoZW1hKSAtLSBmb3IgSW5nZXN0aW9uIEFQSSBFdmVudCB2YWxpZGF0aW9uCiogW2dpdGh1Yi5jb20vZ2V0a2luL2tpbi1vcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vZ2V0a2luL2tpbi1vcGVuYXBpKSAtLSBmb3IgcmVxdWVzdHMgdmFsaWRhdGlvbiBhZ2FpbnN0IE9wZW5BUEkgdjMgc3BlY2lmaWNhdGlvbgoKIyMgSW5nZXN0aW9uIEFQSQoKQ2hlY2sgb2ZmaWNpYWwgZG9jcyBmb3IgW0luZ2VzdGlvbiBBUEldKGh0dHBzOi8vZGV2ZWxvcGVyLm1peHBhbmVsLmNvbS9yZWZlcmVuY2UvaW5nZXN0aW9uLWFwaSkgZGV0YWlscy4KCk91ciBjbGllbnQgb2ZmZXJzIHRvcC1sZXZlbCBpbnRlcmZhY2UgdG8gaW50ZXJhY3Qgd2l0aCBNaXhwYW5lbCBlbmRwb2ludHMuCldlIHVzZSBzZW1pLW9mZmljaWFsIGpzb24gc2NoZW1hIG9mIEV2ZW50IG9iamVjdCBpbiB0ZXN0cyB0byB2YWxpZGF0ZSBwcmVwYXJlZCBldmVudCBkYXRhLiBDaGVjayBbdGhpcyBwYWdlIGluIGRvY3NdKGh0dHBzOi8vZGV2ZWxvcGVyLm1peHBhbmVsLmNvbS9kb2NzL2RhdGEtbW9kZWwjYW5hdG9teS1vZi1hbi1ldmVudCkgZm9yIHRoZSBbc2NoZW1hIGxpbmtdKGh0dHBzOi8vZ2lzdC5naXRodWIuY29tL2pid3ltZS9mMDFmMGE2ZjZmOGI4ZGIyNDcyY2I4NzcxZjdhNTA1YykuCgpBbHNvIHdlIG1hZGUgb3duIFtPcGVuQVBJIHNjaGVtYV0oLi9pbnRlcm5hbC9hc3NldHMvb3BlbmFwaS9pbmdlc3Rpb24ub3BlbmFwaS55bWwpIHRvIGRlc2NyaWJlIGV4dGVybmFsIE1peHBhbmVsIEluZ2VzdGlvbiBBUEkuIFRoZSBtb2R1bGUgdXNlcyBtZW50aW9uZWQgc2NoZW1hIHRvIHZhbGlkYXRlIHByZXBhcmVkIEhUVFAgcmVxdWVzdHMgaW4gdGVzdHMgb25seS4KCiMjIyBFdmVudHMKCiogVHJhY2sgRXZlbnQ6IGBpbmdlc3Rpb24uQ2xpZW50LlRyYWNrKClgCiogVHJhY2sgRXZlbnQgd2l0aCBEZWR1cGxpY2F0aW9uOiBgaW5nZXN0aW9uLkNsaWVudC5UcmFja0RlZHVwbGljYXRlKClgCiogVHJhY2sgTXVsdGlwbGUgRXZlbnRzOiBgaW5nZXN0aW9uLkNsaWVudC5UcmFja0JhdGNoKClgCgojIyMgVXNlciBQcm9maWxlcwoKKiBTZXQgUHJvcGVydHksIFNldCBQcm9wZXJ0eSBPbmNlLCBJbmNyZW1lbnQgTnVtZXJpY2FsIFByb3BlcnR5LCBBcHBlbmQgdG8gTGlzdCBQcm9wZXJ0eSwgUmVtb3ZlIGZyb20gTGlzdCBQcm9wZXJ0eSwgRGVsZXRlIFByb3BlcnR5OiBgaW5nZXN0aW9uLkNsaWVudC5FbmdhZ2UoKWAKKiBVcGRhdGUgTXVsdGlwbGUgUHJvZmlsZXM6IGBpbmdlc3Rpb24uQ2xpZW50LkVuZ2FnZUJhdGNoKClgCg== readmeEtag: '"5263abf375ecbbaee557919fecd74e2423526ee7"' readmeLastModified: Wed, 22 Jun 2022 21:05:41 GMT repositoryId: 354055700 description: Mixpanel API client for Go created: '2021-04-02T15:15:49Z' updated: '2023-05-06T23:18:03Z' language: Go archived: false stars: 3 watchers: 0 forks: 1 owner: wtask-go logo: https://avatars.githubusercontent.com/u/46242331?v=4 license: MIT repoEtag: '"6aea68d9fa9d906381d410a67617b2412b05b7a2703c9ce1d61e948dbbfc6cc4"' repoLastModified: Sat, 06 May 2023 23:18:03 GMT foundInMaster: true category: Parsers id: 61cf00fb32df2c743392e5a46f5ad8cf - source: openapi3 tags repository: https://github.com/estin/oaph v3: true id: e60b95b16fb191f591862c4061fc8a26 repositoryMetadata: base64Readme: >- IyBPQVBICgpIZWxwcyB0byBzdWJ0aXR1YXRlIHF1ZXJ5IHBhcmFtcyBhbmQgc2NoZW1hIGRlZmluaXRpb25zIHRvIG9wZW5hcGkzL2FzeW5jYXBpIHlhbWwuCgpUaGlzIGlzIGZlYXR1cmUtbGVzcyBzaW1wbGUgY3JhdGUgd2l0aCBubyBhbWJpdGlvbnMgdG8gc3VwcG9ydCB3aG9sZSBvcGVuYXBpMy9hc3luY2FwaSBzcGVjcyBhbmQgY292ZXIgYWxsIGNhc2VzIChhdCBsZWFzdCBjb3ZlciBteSBwZXJzb25hbCB1c2UtY2FzZXMpLgoKYGBgcnVzdAp1c2Ugc2VyZGU6OntTZXJpYWxpemUsIERlc2VyaWFsaXplfTsKdXNlIG9hcGg6OntPcGVuQXBpUGxhY2VIb2xkZXIsIHNjaGVtYXJzOjp7c2VsZiwgSnNvblNjaGVtYX19OwoKCiNbYWxsb3coZGVhZF9jb2RlKV0KI1tkZXJpdmUoRGVzZXJpYWxpemUsIEpzb25TY2hlbWEpXQpzdHJ1Y3QgU2VhcmNoUXVlcnkgewogICAgLy8vIHNvbWUgZGVzY3JpcHRpb24gZm9yIHRoaXMgZmxhZyAoc2VlIGh0dHBzOi8vZ3JhaGFtLmNvb2wvc2NoZW1hcnMvZXhhbXBsZXMvNi1kb2NfY29tbWVudHMvKQogICAgZmxhZzogYm9vbCwKfQoKI1thbGxvdyhkZWFkX2NvZGUpXQojW2Rlcml2ZShEZXNlcmlhbGl6ZSwgSnNvblNjaGVtYSldCnN0cnVjdCBTZWFyY2hSZXNwb25zZSB7CiAgICBzdWNjZXNzOiBib29sLAogICAgY291bnQ6IHVzaXplLAogICAgaXRlbXM6IFZlYzxJdGVtPiwKfQoKI1thbGxvdyhkZWFkX2NvZGUpXQojW2Rlcml2ZShEZXNlcmlhbGl6ZSwgSnNvblNjaGVtYSldCnN0cnVjdCBJdGVtIHsKICAgIGlkOiB1c2l6ZSwKICAgIHZhbHVlOiBTdHJpbmcsCn0KCmZuIG1haW4oKSAtPiBSZXN1bHQ8KCksIEJveDxkeW4gc3RkOjplcnJvcjo6RXJyb3I+PiB7CiAgICBsZXQgb3BlbmFwaTNfeWFtbCA9IE9wZW5BcGlQbGFjZUhvbGRlcjo6bmV3KCkKICAgICAgICAucXVlcnlfcGFyYW1zOjo8U2VhcmNoUXVlcnk+KCJTZWFyY2hRdWVyeSIpPwogICAgICAgIC5zY2hlbWE6OjxTZWFyY2hSZXNwb25zZT4oIlNlYXJjaFJlc3BvbnNlIik/CiAgICAgICAgLnJlbmRlcl90byhyIyIKb3BlbmFwaTogMy4wLjAKaW5mbzoKICB0aXRsZTogb2FwaCBleGFtcGxlCiAgdmVyc2lvbjogMS4wLjAKcGF0aHM6CiAgL3NlYXJjaDoKICAgIGdldDoKICAgICAgdGFnczoKICAgICAgLSBkZW1vCiAgICAgIGRlc2NyaXB0aW9uOiBkZW1vIGFwaQogICAgICBwYXJhbWV0ZXJzOgogICAgICAgIHt7U2VhcmNoUXVlcnl9fQogICAgICByZXNwb25zZXM6CiAgICAgICAgJzIwMSc6CiAgICAgICAgICBjb250ZW50OgogICAgICAgICAgICBhcHBsaWNhdGlvbi9qc29uOgogICAgICAgICAgICAgIHNjaGVtYToKICAgICAgICAgICAgICAgIHt7U2VhcmNoUmVzcG9uc2V9fQpkZWZpbml0aW9uczoKICB7e29hcGg6OmRlZmluaXRpb25zfX0KIiMpPzsKCiAgICBwcmludGxuISgie30iLCBvcGVuYXBpM195YW1sKTsKICAgIE9rKCgpKQp9CmBgYAoKQW5kIG91dHB1dCB3b3VsZCBiZQoKYGBgeWFtbApvcGVuYXBpOiAzLjAuMAppbmZvOgogIHRpdGxlOiBvYXBoIGV4YW1wbGUKICB2ZXJzaW9uOiAxLjAuMApwYXRoczoKICAvc2VhcmNoOgogICAgZ2V0OgogICAgICB0YWdzOgogICAgICAtIGRlbW8KICAgICAgZGVzY3JpcHRpb246IGRlbW8gYXBpCiAgICAgIHBhcmFtZXRlcnM6CiAgICAgICAgLSBpbjogcXVlcnkKICAgICAgICAgIG5hbWU6IGZsYWcKICAgICAgICAgIGRlc2NyaXB0aW9uOiBzb21lIGRlc2NyaXB0aW9uIGZvciB0aGlzIGZsYWcgKHNlZSBodHRwczovL2dyYWhhbS5jb29sL3NjaGVtYXJzL2V4YW1wbGVzLzYtZG9jX2NvbW1lbnRzLykKICAgICAgICAgIHJlcXVpcmVkOiB0cnVlCiAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgIHR5cGU6IGJvb2xlYW4KICAgICAgcmVzcG9uc2VzOgogICAgICAgICcyMDEnOgogICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICB0aXRsZTogU2VhcmNoUmVzcG9uc2UKICAgICAgICAgICAgICAgIHR5cGU6IG9iamVjdAogICAgICAgICAgICAgICAgcmVxdWlyZWQ6CiAgICAgICAgICAgICAgICAgIC0gY291bnQKICAgICAgICAgICAgICAgICAgLSBpdGVtcwogICAgICAgICAgICAgICAgICAtIHN1Y2Nlc3MKICAgICAgICAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgICAgICAgIHN1Y2Nlc3M6CiAgICAgICAgICAgICAgICAgICAgdHlwZTogYm9vbGVhbgogICAgICAgICAgICAgICAgICBjb3VudDoKICAgICAgICAgICAgICAgICAgICB0eXBlOiBpbnRlZ2VyCiAgICAgICAgICAgICAgICAgICAgZm9ybWF0OiB1aW50CiAgICAgICAgICAgICAgICAgICAgbWluaW11bTogMC4wCiAgICAgICAgICAgICAgICAgIGl0ZW1zOgogICAgICAgICAgICAgICAgICAgIHR5cGU6IGFycmF5CiAgICAgICAgICAgICAgICAgICAgaXRlbXM6CiAgICAgICAgICAgICAgICAgICAgICAkcmVmOiAiIy9kZWZpbml0aW9ucy9JdGVtIgpkZWZpbml0aW9uczoKICBJdGVtOgogICAgdHlwZTogb2JqZWN0CiAgICByZXF1aXJlZDoKICAgICAgLSBpZAogICAgICAtIHZhbHVlCiAgICBwcm9wZXJ0aWVzOgogICAgICBpZDoKICAgICAgICB0eXBlOiBpbnRlZ2VyCiAgICAgICAgZm9ybWF0OiB1aW50CiAgICAgICAgbWluaW11bTogMC4wCiAgICAgIHZhbHVlOgogICAgICAgIHR5cGU6IHN0cmluZwpgYGAKCkNoZWNrIFtleGFtcGxlXShleGFtcGxlcy9zaW1wbGUvc3JjL21haW4ucnMpIHRvIHNlcnZlIGRvY3Mgb24gW250ZXhdKGh0dHBzOi8vZ2l0aHViLmNvbS9udGV4LXJzL250ZXgpIHN0YWNrLgoKCiMjIFRoYW5rcyB0bwoKIC0gW3NlcmRlXShodHRwczovL2dpdGh1Yi5jb20vc2VyZGUtcnMvc2VyZGUpIGFuZCBidWRkaWVzIGNyYXRlcwogLSBbc2NoZW1hcnNdKGh0dHBzOi8vZ2l0aHViLmNvbS9HUkVzYXUvc2NoZW1hcnMpCgojIyBMaWNlbnNlCgpUaGlzIHByb2plY3QgaXMgbGljZW5zZWQgdW5kZXIKCiogTUlUIGxpY2Vuc2UgKFtMSUNFTlNFXShMSUNFTlNFKSBvciBbaHR0cDovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVF0oaHR0cDovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkpCg== readmeEtag: '"d624e8ba17712a9b6c265bafb9636c2b84cfaee7"' readmeLastModified: Fri, 19 Apr 2024 17:13:45 GMT repositoryId: 372291486 description: >- Helps to subtituate query params and schema definitions to openapi3/asyncapi yaml. created: '2021-05-30T18:47:04Z' updated: '2024-04-19T17:15:31Z' language: Rust archived: false stars: 3 watchers: 1 forks: 0 owner: estin logo: https://avatars.githubusercontent.com/u/520814?v=4 license: MIT repoEtag: '"90fceb169cb7a403f1b4f1bc484496b03ff744826354fa61cff056b5e1e601d4"' repoLastModified: Fri, 19 Apr 2024 17:15:31 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/goodforgod/openapi-resources v3: true id: fe672ce8b2c7837fb898ffc1fc3de8f6 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFJlc291cmNlcwoKQ29sbGVjdGlvbiBvZiBzb21lIE9wZW5BUEkgcmVsYXRlZCByZXNvdXJjZXMuCgpGb3IgU3dhZ2dlciBVSSBEYXJrIFRoZW1lIHRoYW5rcyB0byBbUm9tYW5zIFBva3JvdnNraXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9BbW9lbnVzL1N3YWdnZXJEYXJrKQ== readmeEtag: '"6ec2308403ce20e5643a970a542763cc050565ed"' readmeLastModified: Sun, 26 Feb 2023 09:46:28 GMT repositoryId: 439724448 description: 📦 OpenAPI Resources created: '2021-12-18T22:07:59Z' updated: '2024-06-05T00:27:56Z' language: CSS archived: false stars: 3 watchers: 1 forks: 1 owner: GoodforGod logo: https://avatars.githubusercontent.com/u/9437175?v=4 repoEtag: '"8d779ee9fd47e2489060d92066ec1c417edc5bde60c64cd97d7c5dce66d81396"' repoLastModified: Wed, 05 Jun 2024 00:27:56 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/namsor/namsor-java-sdk2 v3: true repositoryMetadata: base64Readme: >- # namsor-sdk2

NamSor API v2
- API version: 2.0.30
  - Build date: 2024-04-11T10:56:47.519625+02:00[Europe/Berlin]

NamSor API v2 : enpoints to process personal names (gender, cultural origin or ethnicity) in all alphabets or languages. By default, enpoints use 1 unit per name (ex. Gender), but Ethnicity classification uses 10 to 20 units per name depending on taxonomy. Use GET methods for small tests, but prefer POST methods for higher throughput (batch processing of up to 100 names at a time). Need something you can't find here? We have many more features coming soon. Let us know, we'll do our best to add it! 

  For more information, please visit [http://www.namsor.com/](http://www.namsor.com/)

*Automatically generated by the [OpenAPI Generator](https://openapi-generator.tech)*


## Requirements

Building the API client library requires:
1. Java 1.7+
2. Maven/Gradle

## Installation

To install the API client library to your local Maven repository, simply execute:

```shell
mvn clean install
```

To deploy it to a remote Maven repository instead, configure the settings of the repository and execute:

```shell
mvn clean deploy
```

Refer to the [OSSRH Guide](http://central.sonatype.org/pages/ossrh-guide.html) for more information.

### Maven users

Add this dependency to your project's POM:

```xml
<dependency>
  <groupId>com.namsor</groupId>
  <artifactId>namsor-sdk2</artifactId>
  <version>2.0.30</version>
  <scope>compile</scope>
</dependency>
```

### Gradle users

Add this dependency to your project's build file:

```groovy
compile "com.namsor:namsor-sdk2:2.0.30"
```

### Others

At first generate the JAR by executing:

```shell
mvn clean package
```

Then manually install the following JARs:

* `target/namsor-sdk2-2.0.30.jar`
* `target/lib/*.jar`

## Getting Started

Please follow the [installation](#installation) instruction and execute the following Java code:

```java

import com.namsor.sdk2.invoke.*;
import com.namsor.sdk2.invoke.auth.*;
import com.namsor.sdk2.model.*;
import com.namsor.sdk2.api.AdminApi;

import java.io.File;
import java.util.*;

public class AdminApiExample {

    public static void main(String[] args) {
        ApiClient defaultClient = Configuration.getDefaultApiClient();
        
        // Configure API key authorization: api_key
        ApiKeyAuth api_key = (ApiKeyAuth) defaultClient.getAuthentication("api_key");
        api_key.setApiKey("YOUR API KEY");
        // Uncomment the following line to set a prefix for the API key, e.g. "Token" (defaults to null)
        //api_key.setApiKeyPrefix("Token");

        AdminApi apiInstance = new AdminApi();
        String source = "source_example"; // String | 
        Boolean anonymized = true; // Boolean | 
        try {
            apiInstance.anonymize(source, anonymized);
        } catch (ApiException e) {
            System.err.println("Exception when calling AdminApi#anonymize");
            e.printStackTrace();
        }
    }
}

```

## Documentation for API Endpoints

All URIs are relative to *https://v2.namsor.com/NamSorAPIv2*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AdminApi* | [**anonymize**](docs/AdminApi.md#anonymize) | **GET** /api2/json/anonymize/{source}/{anonymized} | Activate/deactivate anonymization for a source.
*AdminApi* | [**anonymize1**](docs/AdminApi.md#anonymize1) | **GET** /api2/json/anonymize/{source}/{anonymized}/{token} | Activate/deactivate anonymization for a source.
*AdminApi* | [**apiKeyInfo**](docs/AdminApi.md#apiKeyInfo) | **GET** /api2/json/apiKeyInfo | Read API Key info.
*AdminApi* | [**apiStatus**](docs/AdminApi.md#apiStatus) | **GET** /api2/json/apiStatus | Prints the current status of the classifiers. A classifier name in apiStatus corresponds to a service name in apiServices.
*AdminApi* | [**apiUsage**](docs/AdminApi.md#apiUsage) | **GET** /api2/json/apiUsage | Print current API usage.
*AdminApi* | [**apiUsageHistory**](docs/AdminApi.md#apiUsageHistory) | **GET** /api2/json/apiUsageHistory | Print historical API usage.
*AdminApi* | [**apiUsageHistoryAggregate**](docs/AdminApi.md#apiUsageHistoryAggregate) | **GET** /api2/json/apiUsageHistoryAggregate | Print historical API usage (in an aggregated view, by service, by day/hour/min).
*AdminApi* | [**availableServices**](docs/AdminApi.md#availableServices) | **GET** /api2/json/apiServices | List of classification services and usage cost in Units per classification (default is 1&#x3D;ONE Unit). Some API endpoints (ex. Corridor) combine multiple classifiers.
*AdminApi* | [**disable**](docs/AdminApi.md#disable) | **GET** /api2/json/disable/{source}/{disabled} | Activate/deactivate an API Key.
*AdminApi* | [**learnable**](docs/AdminApi.md#learnable) | **GET** /api2/json/learnable/{source}/{learnable}/{token} | Activate/deactivate learning from a source.
*AdminApi* | [**learnable1**](docs/AdminApi.md#learnable1) | **GET** /api2/json/learnable/{source}/{learnable} | Activate/deactivate learning from a source.
*AdminApi* | [**regions**](docs/AdminApi.md#regions) | **GET** /api2/json/regions | Print basic source statistics.
*AdminApi* | [**softwareVersion**](docs/AdminApi.md#softwareVersion) | **GET** /api2/json/softwareVersion | Get the current software version
*AdminApi* | [**taxonomyClasses**](docs/AdminApi.md#taxonomyClasses) | **GET** /api2/json/taxonomyClasses/{classifierName} | Print the taxonomy classes valid for the given classifier.
*ChineseApi* | [**chineseNameCandidates**](docs/ChineseApi.md#chineseNameCandidates) | **GET** /api2/json/chineseNameCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming
*ChineseApi* | [**chineseNameCandidatesBatch**](docs/ChineseApi.md#chineseNameCandidatesBatch) | **POST** /api2/json/chineseNameCandidatesBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**chineseNameCandidatesGenderBatch**](docs/ChineseApi.md#chineseNameCandidatesGenderBatch) | **POST** /api2/json/chineseNameCandidatesGenderBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname) ex. Wang Xiaoming.
*ChineseApi* | [**chineseNameGenderCandidates**](docs/ChineseApi.md#chineseNameGenderCandidates) | **GET** /api2/json/chineseNameGenderCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin}/{knownGender} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming - having a known gender (&#39;male&#39; or &#39;female&#39;)
*ChineseApi* | [**chineseNameMatch**](docs/ChineseApi.md#chineseNameMatch) | **GET** /api2/json/chineseNameMatch/{chineseSurnameLatin}/{chineseGivenNameLatin}/{chineseName} | Return a score for matching Chinese name ex. 王晓明 with a romanized name ex. Wang Xiaoming
*ChineseApi* | [**chineseNameMatchBatch**](docs/ChineseApi.md#chineseNameMatchBatch) | **POST** /api2/json/chineseNameMatchBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**genderChineseName**](docs/ChineseApi.md#genderChineseName) | **GET** /api2/json/genderChineseName/{chineseName} | Infer the likely gender of a Chinese full name ex. 王晓明
*ChineseApi* | [**genderChineseNameBatch**](docs/ChineseApi.md#genderChineseNameBatch) | **POST** /api2/json/genderChineseNameBatch | Infer the likely gender of up to 100 full names ex. 王晓明
*ChineseApi* | [**genderChineseNamePinyin**](docs/ChineseApi.md#genderChineseNamePinyin) | **GET** /api2/json/genderChineseNamePinyin/{chineseSurnameLatin}/{chineseGivenNameLatin} | Infer the likely gender of a Chinese name in LATIN (Pinyin).
*ChineseApi* | [**genderChineseNamePinyinBatch**](docs/ChineseApi.md#genderChineseNamePinyinBatch) | **POST** /api2/json/genderChineseNamePinyinBatch | Infer the likely gender of up to 100 Chinese names in LATIN (Pinyin).
*ChineseApi* | [**parseChineseName**](docs/ChineseApi.md#parseChineseName) | **GET** /api2/json/parseChineseName/{chineseName} | Infer the likely first/last name structure of a name, ex. 王晓明 -&gt; 王(surname) 晓明(given name)
*ChineseApi* | [**parseChineseNameBatch**](docs/ChineseApi.md#parseChineseNameBatch) | **POST** /api2/json/parseChineseNameBatch | Infer the likely first/last name structure of a name, ex. 王晓明 -&gt; 王(surname) 晓明(given name).
*ChineseApi* | [**pinyinChineseName**](docs/ChineseApi.md#pinyinChineseName) | **GET** /api2/json/pinyinChineseName/{chineseName} | Romanize the Chinese name to Pinyin, ex. 王晓明 -&gt; Wang (surname) Xiaoming (given name)
*ChineseApi* | [**pinyinChineseNameBatch**](docs/ChineseApi.md#pinyinChineseNameBatch) | **POST** /api2/json/pinyinChineseNameBatch | Romanize a list of Chinese name to Pinyin, ex. 王晓明 -&gt; Wang (surname) Xiaoming (given name).
*GeneralApi* | [**nameType**](docs/GeneralApi.md#nameType) | **GET** /api2/json/nameType/{properNoun} | Infer the likely type of a proper noun (personal name, brand name, place name etc.)
*GeneralApi* | [**nameTypeBatch**](docs/GeneralApi.md#nameTypeBatch) | **POST** /api2/json/nameTypeBatch | Infer the likely common type of up to 100 proper nouns (personal name, brand name, place name etc.)
*GeneralApi* | [**nameTypeGeo**](docs/GeneralApi.md#nameTypeGeo) | **GET** /api2/json/nameTypeGeo/{properNoun}/{countryIso2} | Infer the likely type of a proper noun (personal name, brand name, place name etc.)
*GeneralApi* | [**nameTypeGeoBatch**](docs/GeneralApi.md#nameTypeGeoBatch) | **POST** /api2/json/nameTypeGeoBatch | Infer the likely common type of up to 100 proper nouns (personal name, brand name, place name etc.)
*IndianApi* | [**casteIndianBatch**](docs/IndianApi.md#casteIndianBatch) | **POST** /api2/json/casteIndianBatch | [USES 10 UNITS PER NAME] Infer the likely Indian name caste of up to 100 personal Indian Hindu names. 
*IndianApi* | [**castegroupIndian**](docs/IndianApi.md#castegroupIndian) | **GET** /api2/json/castegroupIndian/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of a first / last name.
*IndianApi* | [**castegroupIndianBatch**](docs/IndianApi.md#castegroupIndianBatch) | **POST** /api2/json/castegroupIndianBatch | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of up to 100 personal first / last names. 
*IndianApi* | [**castegroupIndianFull**](docs/IndianApi.md#castegroupIndianFull) | **GET** /api2/json/castegroupIndianFull/{subDivisionIso31662}/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of a personal full name.
*IndianApi* | [**castegroupIndianFullBatch**](docs/IndianApi.md#castegroupIndianFullBatch) | **POST** /api2/json/castegroupIndianFullBatch | [USES 10 UNITS PER NAME] Infer the likely Indian name castegroup of up to 100 personal full names. 
*IndianApi* | [**castegroupIndianHindu**](docs/IndianApi.md#castegroupIndianHindu) | **GET** /api2/json/casteIndian/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely Indian name caste of a personal Hindu name.
*IndianApi* | [**religion**](docs/IndianApi.md#religion) | **GET** /api2/json/religionIndianFull/{subDivisionIso31662}/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal Indian full name, provided the Indian state or Union territory (NB/ this can be inferred using the subclassification endpoint).
*IndianApi* | [**religion1**](docs/IndianApi.md#religion1) | **GET** /api2/json/religionIndian/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal Indian first/last name, provided the Indian state or Union territory (NB/ this can be inferred using the subclassification endpoint).
*IndianApi* | [**religionIndianBatch**](docs/IndianApi.md#religionIndianBatch) | **POST** /api2/json/religionIndianBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal first/last Indian names, provided the subclassification at State or Union territory level (NB/ can be inferred using the subclassification endpoint).
*IndianApi* | [**religionIndianFullBatch**](docs/IndianApi.md#religionIndianFullBatch) | **POST** /api2/json/religionIndianFullBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal full Indian names, provided the subclassification at State or Union territory level (NB/ can be inferred using the subclassification endpoint).
*IndianApi* | [**subclassificationIndian**](docs/IndianApi.md#subclassificationIndian) | **GET** /api2/json/subclassificationIndian/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on the name.
*IndianApi* | [**subclassificationIndianBatch**](docs/IndianApi.md#subclassificationIndianBatch) | **POST** /api2/json/subclassificationIndianBatch | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on a list of up to 100 names.
*IndianApi* | [**subclassificationIndianFull**](docs/IndianApi.md#subclassificationIndianFull) | **GET** /api2/json/subclassificationIndianFull/{fullName} | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on the name.
*IndianApi* | [**subclassificationIndianFullBatch**](docs/IndianApi.md#subclassificationIndianFullBatch) | **POST** /api2/json/subclassificationIndianFullBatch | [USES 10 UNITS PER NAME] Infer the likely Indian state of Union territory according to ISO 3166-2:IN based on a list of up to 100 names.
*JapaneseApi* | [**genderJapaneseNameFull**](docs/JapaneseApi.md#genderJapaneseNameFull) | **GET** /api2/json/genderJapaneseNameFull/{japaneseName} | Infer the likely gender of a Japanese full name ex. 王晓明
*JapaneseApi* | [**genderJapaneseNameFullBatch**](docs/JapaneseApi.md#genderJapaneseNameFullBatch) | **POST** /api2/json/genderJapaneseNameFullBatch | Infer the likely gender of up to 100 full names
*JapaneseApi* | [**genderJapaneseNamePinyin**](docs/JapaneseApi.md#genderJapaneseNamePinyin) | **GET** /api2/json/genderJapaneseName/{japaneseSurname}/{japaneseGivenName} | Infer the likely gender of a Japanese name in LATIN (Pinyin).
*JapaneseApi* | [**genderJapaneseNamePinyinBatch**](docs/JapaneseApi.md#genderJapaneseNamePinyinBatch) | **POST** /api2/json/genderJapaneseNameBatch | Infer the likely gender of up to 100 Japanese names in LATIN (Pinyin).
*JapaneseApi* | [**japaneseNameGenderKanjiCandidatesBatch**](docs/JapaneseApi.md#japaneseNameGenderKanjiCandidatesBatch) | **POST** /api2/json/japaneseNameGenderKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName &#x3D; japaneseGivenName; lastName&#x3D;japaneseSurname) with KNOWN gender, ex. Yamamoto Sanae
*JapaneseApi* | [**japaneseNameKanjiCandidates**](docs/JapaneseApi.md#japaneseNameKanjiCandidates) | **GET** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**japaneseNameKanjiCandidates1**](docs/JapaneseApi.md#japaneseNameKanjiCandidates1) | **GET** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{knownGender} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae - and a known gender.
*JapaneseApi* | [**japaneseNameKanjiCandidatesBatch**](docs/JapaneseApi.md#japaneseNameKanjiCandidatesBatch) | **POST** /api2/json/japaneseNameKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName &#x3D; japaneseGivenName; lastName&#x3D;japaneseSurname), ex. Yamamoto Sanae
*JapaneseApi* | [**japaneseNameLatinCandidates**](docs/JapaneseApi.md#japaneseNameLatinCandidates) | **GET** /api2/json/japaneseNameLatinCandidates/{japaneseSurnameKanji}/{japaneseGivenNameKanji} | Romanize japanese name, based on the name in Kanji.
*JapaneseApi* | [**japaneseNameLatinCandidatesBatch**](docs/JapaneseApi.md#japaneseNameLatinCandidatesBatch) | **POST** /api2/json/japaneseNameLatinCandidatesBatch | Romanize japanese names, based on the name in KANJI
*JapaneseApi* | [**japaneseNameMatch**](docs/JapaneseApi.md#japaneseNameMatch) | **GET** /api2/json/japaneseNameMatch/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | Return a score for matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**japaneseNameMatchBatch**](docs/JapaneseApi.md#japaneseNameMatchBatch) | **POST** /api2/json/japaneseNameMatchBatch | Return a score for matching a list of Japanese names in KANJI ex. 山本 早苗 with romanized names ex. Yamamoto Sanae
*JapaneseApi* | [**japaneseNameMatchFeedbackLoop**](docs/JapaneseApi.md#japaneseNameMatchFeedbackLoop) | **GET** /api2/json/japaneseNameMatchFeedbackLoop/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | [CREDITS 1 UNIT] Feedback loop to better perform matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**parseJapaneseName**](docs/JapaneseApi.md#parseJapaneseName) | **GET** /api2/json/parseJapaneseName/{japaneseName} | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae
*JapaneseApi* | [**parseJapaneseNameBatch**](docs/JapaneseApi.md#parseJapaneseNameBatch) | **POST** /api2/json/parseJapaneseNameBatch | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae 
*PersonalApi* | [**communityEngage**](docs/PersonalApi.md#communityEngage) | **GET** /api2/json/communityEngage/{countryIso2}/{firstName}/{lastName} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora, country, gender of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.) for community engagement (require special module/pricing)
*PersonalApi* | [**communityEngageBatch**](docs/PersonalApi.md#communityEngageBatch) | **POST** /api2/json/communityEngageBatch | Infer the likely ethnicity/diaspora, country, gender of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.) for community engagement (require special module/pricing)
*PersonalApi* | [**communityEngageFull**](docs/PersonalApi.md#communityEngageFull) | **GET** /api2/json/communityEngageFull/{countryIso2}/{personalNameFull} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora, country, gender of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.) for community engagement (require special module/pricing)
*PersonalApi* | [**communityEngageFullBatch**](docs/PersonalApi.md#communityEngageFullBatch) | **POST** /api2/json/communityEngageFullBatch | Infer the likely ethnicity/diaspora, country, gender of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.) for community engagement (require special module/pricing)
*PersonalApi* | [**corridor**](docs/PersonalApi.md#corridor) | **GET** /api2/json/corridor/{countryIso2From}/{firstNameFrom}/{lastNameFrom}/{countryIso2To}/{firstNameTo}/{lastNameTo} | [USES 50 UNITS PER NAME COUPLE] Infer several classifications for a cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**corridorBatch**](docs/PersonalApi.md#corridorBatch) | **POST** /api2/json/corridorBatch | [USES 50 UNITS PER NAME PAIR] Infer several classifications for up to 100 cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**corridorFull**](docs/PersonalApi.md#corridorFull) | **GET** /api2/json/corridorFull/{countryIso2From}/{personalNameFrom}/{countryIso2To}/{personalNameTo} | [USES 50 UNITS PER NAME COUPLE] Infer several classifications for a cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**corridorFullBatch**](docs/PersonalApi.md#corridorFullBatch) | **POST** /api2/json/corridorFullBatch | [USES 50 UNITS PER NAME PAIR] Infer several classifications for up to 100 cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**country**](docs/PersonalApi.md#country) | **GET** /api2/json/country/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely country of residence of a personal full name, or one surname. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**countryBatch**](docs/PersonalApi.md#countryBatch) | **POST** /api2/json/countryBatch | [USES 10 UNITS PER NAME] Infer the likely country of residence of up to 100 personal full names, or surnames. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**countryFnLn**](docs/PersonalApi.md#countryFnLn) | **GET** /api2/json/countryFnLn/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely country of residence of a personal first / last name, or one surname. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**countryFnLnBatch**](docs/PersonalApi.md#countryFnLnBatch) | **POST** /api2/json/countryFnLnBatch | [USES 10 UNITS PER NAME] Infer the likely country of residence of up to 100 personal first / last names, or surnames. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**diaspora**](docs/PersonalApi.md#diaspora) | **GET** /api2/json/diaspora/{countryIso2}/{firstName}/{lastName} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**diasporaBatch**](docs/PersonalApi.md#diasporaBatch) | **POST** /api2/json/diasporaBatch | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**diasporaFull**](docs/PersonalApi.md#diasporaFull) | **GET** /api2/json/diasporaFull/{countryIso2}/{personalNameFull} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**diasporaFullBatch**](docs/PersonalApi.md#diasporaFullBatch) | **POST** /api2/json/diasporaFullBatch | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**gender**](docs/PersonalApi.md#gender) | **GET** /api2/json/gender/{firstName} | Infer the likely gender of a just a fiven name, assuming default &#39;US&#39; local context. Please use preferably full names and local geographic context for better accuracy.
*PersonalApi* | [**gender1**](docs/PersonalApi.md#gender1) | **GET** /api2/json/gender/{firstName}/{lastName} | Infer the likely gender of a name.
*PersonalApi* | [**genderBatch**](docs/PersonalApi.md#genderBatch) | **POST** /api2/json/genderBatch | Infer the likely gender of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**genderFull**](docs/PersonalApi.md#genderFull) | **GET** /api2/json/genderFull/{fullName} | Infer the likely gender of a full name, ex. John H. Smith
*PersonalApi* | [**genderFullBatch**](docs/PersonalApi.md#genderFullBatch) | **POST** /api2/json/genderFullBatch | Infer the likely gender of up to 100 full names, detecting automatically the cultural context.
*PersonalApi* | [**genderFullGeo**](docs/PersonalApi.md#genderFullGeo) | **GET** /api2/json/genderFullGeo/{fullName}/{countryIso2} | Infer the likely gender of a full name, given a local context (ISO2 country code).
*PersonalApi* | [**genderFullGeoBatch**](docs/PersonalApi.md#genderFullGeoBatch) | **POST** /api2/json/genderFullGeoBatch | Infer the likely gender of up to 100 full names, with a given cultural context (country ISO2 code).
*PersonalApi* | [**genderGeo**](docs/PersonalApi.md#genderGeo) | **GET** /api2/json/genderGeo/{firstName}/{lastName}/{countryIso2} | Infer the likely gender of a name, given a local context (ISO2 country code).
*PersonalApi* | [**genderGeoBatch**](docs/PersonalApi.md#genderGeoBatch) | **POST** /api2/json/genderGeoBatch | Infer the likely gender of up to 100 names, each given a local context (ISO2 country code).
*PersonalApi* | [**origin**](docs/PersonalApi.md#origin) | **GET** /api2/json/origin/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely country of origin of a personal name. Assumes names as they are in the country of origin. For US, CA, AU, NZ and other melting-pots : use &#39;diaspora&#39; instead.
*PersonalApi* | [**originBatch**](docs/PersonalApi.md#originBatch) | **POST** /api2/json/originBatch | [USES 10 UNITS PER NAME] Infer the likely country of origin of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**originFull**](docs/PersonalApi.md#originFull) | **GET** /api2/json/originFull/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely country of origin of a personal name. Assumes names as they are in the country of origin. For US, CA, AU, NZ and other melting-pots : use &#39;diaspora&#39; instead.
*PersonalApi* | [**originFullBatch**](docs/PersonalApi.md#originFullBatch) | **POST** /api2/json/originFullBatch | [USES 10 UNITS PER NAME] Infer the likely country of origin of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**parseName**](docs/PersonalApi.md#parseName) | **GET** /api2/json/parseName/{nameFull} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. 
*PersonalApi* | [**parseNameBatch**](docs/PersonalApi.md#parseNameBatch) | **POST** /api2/json/parseNameBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John.
*PersonalApi* | [**parseNameGeo**](docs/PersonalApi.md#parseNameGeo) | **GET** /api2/json/parseName/{nameFull}/{countryIso2} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. For better accuracy, provide a geographic context.
*PersonalApi* | [**parseNameGeoBatch**](docs/PersonalApi.md#parseNameGeoBatch) | **POST** /api2/json/parseNameGeoBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. Giving a local context improves precision. 
*PersonalApi* | [**religion2**](docs/PersonalApi.md#religion2) | **GET** /api2/json/religion/{countryIso2}/{subDivisionIso31662}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal first/last name. NB: only for INDIA (as of current version).
*PersonalApi* | [**religionBatch**](docs/PersonalApi.md#religionBatch) | **POST** /api2/json/religionBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal first/last names. NB: only for India as of currently.
*PersonalApi* | [**religionFull**](docs/PersonalApi.md#religionFull) | **GET** /api2/json/religionFull/{countryIso2}/{subDivisionIso31662}/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely religion of a personal full name. NB: only for INDIA (as of current version).
*PersonalApi* | [**religionFullBatch**](docs/PersonalApi.md#religionFullBatch) | **POST** /api2/json/religionFullBatch | [USES 10 UNITS PER NAME] Infer the likely religion of up to 100 personal full names. NB: only for India as of currently.
*PersonalApi* | [**subclassification**](docs/PersonalApi.md#subclassification) | **GET** /api2/json/subclassification/{countryIso2}/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely origin of a name at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#39;IN&#39;).
*PersonalApi* | [**subclassificationBatch**](docs/PersonalApi.md#subclassificationBatch) | **POST** /api2/json/subclassificationBatch | [USES 10 UNITS PER NAME] Infer the likely origin of a list of up to 100 names at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#39;IN&#39;).
*PersonalApi* | [**subclassificationFull**](docs/PersonalApi.md#subclassificationFull) | **GET** /api2/json/subclassificationFull/{countryIso2}/{fullName} | [USES 10 UNITS PER NAME] Infer the likely origin of a name at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#39;IN&#39;).
*PersonalApi* | [**subclassificationFullBatch**](docs/PersonalApi.md#subclassificationFullBatch) | **POST** /api2/json/subclassificationFullBatch | [USES 10 UNITS PER NAME] Infer the likely origin of a list of up to 100 names at a country subclassification level (state or regeion). Initially, this is only supported for India (ISO2 code &#39;IN&#39;).
*PersonalApi* | [**usRaceEthnicity**](docs/PersonalApi.md#usRaceEthnicity) | **GET** /api2/json/usRaceEthnicity/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer a US resident&#39;s likely race/ethnicity according to US Census taxonomy W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**usRaceEthnicityBatch**](docs/PersonalApi.md#usRaceEthnicityBatch) | **POST** /api2/json/usRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#39;s likely race/ethnicity according to US Census taxonomy. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**usRaceEthnicityFull**](docs/PersonalApi.md#usRaceEthnicityFull) | **GET** /api2/json/usRaceEthnicityFull/{personalNameFull} | [USES 10 UNITS PER NAME] Infer a US resident&#39;s likely race/ethnicity according to US Census taxonomy W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**usRaceEthnicityFullBatch**](docs/PersonalApi.md#usRaceEthnicityFullBatch) | **POST** /api2/json/usRaceEthnicityFullBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#39;s likely race/ethnicity according to US Census taxonomy. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**usRaceEthnicityZIP5**](docs/PersonalApi.md#usRaceEthnicityZIP5) | **GET** /api2/json/usRaceEthnicityZIP5/{firstName}/{lastName}/{zip5Code} | [USES 10 UNITS PER NAME] Infer a US resident&#39;s likely race/ethnicity according to US Census taxonomy, using (optional) ZIP5 code info. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**usZipRaceEthnicityBatch**](docs/PersonalApi.md#usZipRaceEthnicityBatch) | **POST** /api2/json/usZipRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#39;s likely race/ethnicity according to US Census taxonomy, with (optional) ZIP code. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*SocialApi* | [**phoneCode**](docs/SocialApi.md#phoneCode) | **GET** /api2/json/phoneCode/{firstName}/{lastName}/{phoneNumber} | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, given a personal name and formatted / unformatted phone number.
*SocialApi* | [**phoneCodeBatch**](docs/SocialApi.md#phoneCodeBatch) | **POST** /api2/json/phoneCodeBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, detecting automatically the local context given a name and formatted / unformatted phone number.
*SocialApi* | [**phoneCodeGeo**](docs/SocialApi.md#phoneCodeGeo) | **GET** /api2/json/phoneCodeGeo/{firstName}/{lastName}/{phoneNumber}/{countryIso2} | [USES 11 UNITS PER NAME] Infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).
*SocialApi* | [**phoneCodeGeoBatch**](docs/SocialApi.md#phoneCodeGeoBatch) | **POST** /api2/json/phoneCodeGeoBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, with a local context (ISO2 country of residence).
*SocialApi* | [**phoneCodeGeoFeedbackLoop**](docs/SocialApi.md#phoneCodeGeoFeedbackLoop) | **GET** /api2/json/phoneCodeGeoFeedbackLoop/{firstName}/{lastName}/{phoneNumber}/{phoneNumberE164}/{countryIso2} | [CREDITS 1 UNIT] Feedback loop to better infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).


## Documentation for Models

 - [APIBillingPeriodUsageOut](docs/APIBillingPeriodUsageOut.md)
 - [APIClassifierOut](docs/APIClassifierOut.md)
 - [APIClassifierTaxonomyOut](docs/APIClassifierTaxonomyOut.md)
 - [APIClassifiersStatusOut](docs/APIClassifiersStatusOut.md)
 - [APICounterV2Out](docs/APICounterV2Out.md)
 - [APIKeyOut](docs/APIKeyOut.md)
 - [APIPeriodUsageOut](docs/APIPeriodUsageOut.md)
 - [APIPlanSubscriptionOut](docs/APIPlanSubscriptionOut.md)
 - [APIServiceOut](docs/APIServiceOut.md)
 - [APIServicesOut](docs/APIServicesOut.md)
 - [APIUsageAggregatedOut](docs/APIUsageAggregatedOut.md)
 - [APIUsageHistoryOut](docs/APIUsageHistoryOut.md)
 - [BatchCommunityEngageFullOut](docs/BatchCommunityEngageFullOut.md)
 - [BatchCommunityEngageOut](docs/BatchCommunityEngageOut.md)
 - [BatchCorridorFullIn](docs/BatchCorridorFullIn.md)
 - [BatchCorridorFullOut](docs/BatchCorridorFullOut.md)
 - [BatchCorridorIn](docs/BatchCorridorIn.md)
 - [BatchCorridorOut](docs/BatchCorridorOut.md)
 - [BatchFirstLastNameCasteOut](docs/BatchFirstLastNameCasteOut.md)
 - [BatchFirstLastNameCastegroupOut](docs/BatchFirstLastNameCastegroupOut.md)
 - [BatchFirstLastNameDiasporaedOut](docs/BatchFirstLastNameDiasporaedOut.md)
 - [BatchFirstLastNameGenderIn](docs/BatchFirstLastNameGenderIn.md)
 - [BatchFirstLastNameGenderedOut](docs/BatchFirstLastNameGenderedOut.md)
 - [BatchFirstLastNameGeoIn](docs/BatchFirstLastNameGeoIn.md)
 - [BatchFirstLastNameGeoOut](docs/BatchFirstLastNameGeoOut.md)
 - [BatchFirstLastNameGeoSubclassificationOut](docs/BatchFirstLastNameGeoSubclassificationOut.md)
 - [BatchFirstLastNameGeoSubdivisionIn](docs/BatchFirstLastNameGeoSubdivisionIn.md)
 - [BatchFirstLastNameGeoZippedIn](docs/BatchFirstLastNameGeoZippedIn.md)
 - [BatchFirstLastNameIn](docs/BatchFirstLastNameIn.md)
 - [BatchFirstLastNameOriginedOut](docs/BatchFirstLastNameOriginedOut.md)
 - [BatchFirstLastNamePhoneCodedOut](docs/BatchFirstLastNamePhoneCodedOut.md)
 - [BatchFirstLastNamePhoneNumberGeoIn](docs/BatchFirstLastNamePhoneNumberGeoIn.md)
 - [BatchFirstLastNamePhoneNumberIn](docs/BatchFirstLastNamePhoneNumberIn.md)
 - [BatchFirstLastNameReligionedOut](docs/BatchFirstLastNameReligionedOut.md)
 - [BatchFirstLastNameSubdivisionIn](docs/BatchFirstLastNameSubdivisionIn.md)
 - [BatchFirstLastNameUSRaceEthnicityOut](docs/BatchFirstLastNameUSRaceEthnicityOut.md)
 - [BatchMatchPersonalFirstLastNameIn](docs/BatchMatchPersonalFirstLastNameIn.md)
 - [BatchNameGeoIn](docs/BatchNameGeoIn.md)
 - [BatchNameIn](docs/BatchNameIn.md)
 - [BatchNameMatchCandidatesOut](docs/BatchNameMatchCandidatesOut.md)
 - [BatchNameMatchedOut](docs/BatchNameMatchedOut.md)
 - [BatchPersonalNameCastegroupOut](docs/BatchPersonalNameCastegroupOut.md)
 - [BatchPersonalNameDiasporaedOut](docs/BatchPersonalNameDiasporaedOut.md)
 - [BatchPersonalNameGenderedOut](docs/BatchPersonalNameGenderedOut.md)
 - [BatchPersonalNameGeoIn](docs/BatchPersonalNameGeoIn.md)
 - [BatchPersonalNameGeoOut](docs/BatchPersonalNameGeoOut.md)
 - [BatchPersonalNameGeoSubclassificationOut](docs/BatchPersonalNameGeoSubclassificationOut.md)
 - [BatchPersonalNameGeoSubdivisionIn](docs/BatchPersonalNameGeoSubdivisionIn.md)
 - [BatchPersonalNameIn](docs/BatchPersonalNameIn.md)
 - [BatchPersonalNameOriginedOut](docs/BatchPersonalNameOriginedOut.md)
 - [BatchPersonalNameParsedOut](docs/BatchPersonalNameParsedOut.md)
 - [BatchPersonalNameReligionedOut](docs/BatchPersonalNameReligionedOut.md)
 - [BatchPersonalNameSubdivisionIn](docs/BatchPersonalNameSubdivisionIn.md)
 - [BatchPersonalNameUSRaceEthnicityOut](docs/BatchPersonalNameUSRaceEthnicityOut.md)
 - [BatchProperNounCategorizedOut](docs/BatchProperNounCategorizedOut.md)
 - [CommunityEngageOptionOut](docs/CommunityEngageOptionOut.md)
 - [CommunityEngageOut](docs/CommunityEngageOut.md)
 - [CorridorFullOut](docs/CorridorFullOut.md)
 - [CorridorIn](docs/CorridorIn.md)
 - [CorridorOut](docs/CorridorOut.md)
 - [FeedbackLoopOut](docs/FeedbackLoopOut.md)
 - [FirstLastNameCasteOut](docs/FirstLastNameCasteOut.md)
 - [FirstLastNameCastegroupOut](docs/FirstLastNameCastegroupOut.md)
 - [FirstLastNameDiasporaedOut](docs/FirstLastNameDiasporaedOut.md)
 - [FirstLastNameGenderIn](docs/FirstLastNameGenderIn.md)
 - [FirstLastNameGenderedOut](docs/FirstLastNameGenderedOut.md)
 - [FirstLastNameGeoIn](docs/FirstLastNameGeoIn.md)
 - [FirstLastNameGeoOut](docs/FirstLastNameGeoOut.md)
 - [FirstLastNameGeoSubclassificationOut](docs/FirstLastNameGeoSubclassificationOut.md)
 - [FirstLastNameGeoSubdivisionIn](docs/FirstLastNameGeoSubdivisionIn.md)
 - [FirstLastNameGeoZippedIn](docs/FirstLastNameGeoZippedIn.md)
 - [FirstLastNameIn](docs/FirstLastNameIn.md)
 - [FirstLastNameOriginedOut](docs/FirstLastNameOriginedOut.md)
 - [FirstLastNameOut](docs/FirstLastNameOut.md)
 - [FirstLastNamePhoneCodedOut](docs/FirstLastNamePhoneCodedOut.md)
 - [FirstLastNamePhoneNumberGeoIn](docs/FirstLastNamePhoneNumberGeoIn.md)
 - [FirstLastNamePhoneNumberIn](docs/FirstLastNamePhoneNumberIn.md)
 - [FirstLastNameReligionedOut](docs/FirstLastNameReligionedOut.md)
 - [FirstLastNameSubdivisionIn](docs/FirstLastNameSubdivisionIn.md)
 - [FirstLastNameUSRaceEthnicityOut](docs/FirstLastNameUSRaceEthnicityOut.md)
 - [MatchPersonalFirstLastNameIn](docs/MatchPersonalFirstLastNameIn.md)
 - [NameGeoIn](docs/NameGeoIn.md)
 - [NameIn](docs/NameIn.md)
 - [NameMatchCandidateOut](docs/NameMatchCandidateOut.md)
 - [NameMatchCandidatesOut](docs/NameMatchCandidatesOut.md)
 - [NameMatchedOut](docs/NameMatchedOut.md)
 - [PersonalNameCastegroupOut](docs/PersonalNameCastegroupOut.md)
 - [PersonalNameDiasporaedOut](docs/PersonalNameDiasporaedOut.md)
 - [PersonalNameGenderedOut](docs/PersonalNameGenderedOut.md)
 - [PersonalNameGeoIn](docs/PersonalNameGeoIn.md)
 - [PersonalNameGeoOut](docs/PersonalNameGeoOut.md)
 - [PersonalNameGeoSubclassificationOut](docs/PersonalNameGeoSubclassificationOut.md)
 - [PersonalNameGeoSubdivisionIn](docs/PersonalNameGeoSubdivisionIn.md)
 - [PersonalNameIn](docs/PersonalNameIn.md)
 - [PersonalNameOriginedOut](docs/PersonalNameOriginedOut.md)
 - [PersonalNameParsedOut](docs/PersonalNameParsedOut.md)
 - [PersonalNameReligionedOut](docs/PersonalNameReligionedOut.md)
 - [PersonalNameSubdivisionIn](docs/PersonalNameSubdivisionIn.md)
 - [PersonalNameUSRaceEthnicityOut](docs/PersonalNameUSRaceEthnicityOut.md)
 - [ProperNounCategorizedOut](docs/ProperNounCategorizedOut.md)
 - [RegionISO](docs/RegionISO.md)
 - [RegionOut](docs/RegionOut.md)
 - [ReligionStatOut](docs/ReligionStatOut.md)
 - [SoftwareVersionOut](docs/SoftwareVersionOut.md)


## Documentation for Authorization

Authentication schemes defined for the API:
### api_key

- **Type**: API key
- **API key parameter name**: X-API-KEY
- **Location**: HTTP header


## Recommendation

It's recommended to create an instance of `ApiClient` per thread in a multithreaded environment to avoid any potential issues.

## Author

contact@namsor.com

 readmeEtag: '"7b2d272c7a87dd96811652eab9221dbc3e47c903"' readmeLastModified: Thu, 11 Apr 2024 09:05:05 GMT repositoryId: 165628241 description: >- NamSor API v2 Java SDK - classify personal names accurately by gender, country of origin, or ethnicity. created: '2019-01-14T08:57:20Z' updated: '2025-12-18T09:27:29Z' language: Java archived: false stars: 3 watchers: 2 forks: 0 owner: namsor logo: https://avatars.githubusercontent.com/u/6951565?v=4 license: AGPL-3.0 repoEtag: '"78e28fe6af90cbe999a17b374d2a0cfcbce113c8082666a9331bc3513066dc8d"' repoLastModified: Thu, 18 Dec 2025 09:27:29 GMT foundInMaster: true category: Description Validators id: 13d0d90ebabd4eb9f1a420d9c3406807 - source: openapi3 tags repository: https://github.com/ironcore-dev/openapi-extractor v3: true id: 15586772b825fc246137cce1d6a2d7c7 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWV4dHJhY3RvcgoKWyFbUkVVU0Ugc3RhdHVzXShodHRwczovL2FwaS5yZXVzZS5zb2Z0d2FyZS9iYWRnZS9naXRodWIuY29tL2lyb25jb3JlLWRldi9vcGVuYXBpLWV4dHJhY3RvcildKGh0dHBzOi8vYXBpLnJldXNlLnNvZnR3YXJlL2luZm8vZ2l0aHViLmNvbS9pcm9uY29yZS1kZXYvb3BlbmFwaS1leHRyYWN0b3IpClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS9pcm9uY29yZS1kZXYvb3BlbmFwaS1leHRyYWN0b3IpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20vaXJvbmNvcmUtZGV2L29wZW5hcGktZXh0cmFjdG9yKQpbIVtHaXRIdWIgTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9zdGF0aWMvdjE/bGFiZWw9TGljZW5zZSZtZXNzYWdlPUFwYWNoZS0yLjAmY29sb3I9Ymx1ZSldKExJQ0VOU0UpClshW1BScyBXZWxjb21lXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1BScy13ZWxjb21lLWJyaWdodGdyZWVuLnN2ZyldKGh0dHBzOi8vbWFrZWFwdWxscmVxdWVzdC5jb20pCgpUaGUgYG9wZW5hcGktZXh0cmFjdG9yYCBleHRyYWN0cyB0aGUgT3BlbkFQSSB2MiBhbmQgdjMgc3BlY2lmaWNhdGlvbnMgb2YgYSBnaXZlbiBLdWJlcm5ldGVzIEFQSSBzZXJ2ZXIuCgojIyBJbnN0YWxsYXRpb24KCiMjIyBGcm9tIHNvdXJjZQoKVG8gaW5zdGFsbCB0aGUgYG9wZW5hcGktZXh0cmFjdG9yYCBiaW5hcnkgaW50byB5b3VyIEdvIGJpbiBwYXRoIHJ1bgoKYGBgYmFzaApnbyBpbnN0YWxsIGdpdGh1Yi5jb20vaXJvbmNvcmUtZGV2L29wZW5hcGktZXh0cmFjdG9yL2NtZC9vcGVuYXBpLWV4dHJhY3RvckBtYWluCmBgYAoKIyMgVXNhZ2UKCiMjIyBDb21tYW5kIGJhc2VkIGV4dHJhY3Rpb24KCkluIGNhc2UgeW91IGhhdmUgdGhlIGFwaSBzZXJ2ZXIgYmluYXJ5IHByZXNlbnQsIHlvdSBjYW4gZXh0cmFjdCB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBieSBydW5uaW5nCgpgYGBzaGVsbApvcGVuYXBpLWV4dHJhY3RvciAtLWFwaXNlcnZlci1jb21tYW5kPTxQQVRILVRPLUFQSVNFUlZFUi1CSU4+IFwKICAtLWFwaXNlcnZpY2VzPTxQQVRILVRPLUFQSVNFUlZJQ0VTLURJUj4KYGBgCgojIyMgR28gbW9kdWxlIGJhc2VkIGV4dHJhY3Rpb24KClRoZSBbYHNhbXBsZWBdKC9zYW1wbGUpIGZvbGRlciBjb250YWlucyBhbiBleGFtcGxlIG9uIGhvdyB0byBleHRyYWN0IHRoZSBPcGVuIEFQSSBzcGVjIGZyb20gYW4gYXBpIHNlcnZlciBwYWNrYWdlLiBJbiAKb3VyIGV4YW1wbGUgd2UgYXJlIHVzaW5nIHRoZSBbYGlyb25jb3JlYF0oaHR0cHM6Ly9naXRodWIuY29tL2lyb25jb3JlLWRldi9pcm9uY29yZSkgYWdncmVnYXRlZCBhcGkgc2VydmVyLgoKYGBgc2hlbGwKb3BlbmFwaS1leHRyYWN0b3IgLS1hcGlzZXJ2ZXItcGFja2FnZT1naXRodWIuY29tL2lyb25jb3JlLWRldi9pcm9uY29yZS9jbWQvaXJvbmNvcmUtYXBpc2VydmVyIFwKICAtLWFwaXNlcnZlci1idWlsZC1vcHRzPW1vZCBcCiAgLS1hcGlzZXJ2aWNlcz08UEFUSC1UTy1BUElTRVJWSUNFUy1ESVI+CmBgYAoKSW4gY2FzZSB5b3Ugd2FudCB0byB1c2UgeW91ciBvd24gcGFja2FnZSwgZmlyc3QgYGdvIGdldGAgaXQgc28geW91IGhhdmUgdG8gY29ycmVjdCBkZXBlbmRlbmNpZXMgaW4geW91ciBgZ28ubW9kYCBmaWxlIGFuZAphZGp1c3QgdGhlIGAtLWFwaXNlcnZlci1wYWNrYWdlYCBmbGFnIGFjY29yZGluZ2x5LgoKIyMjIE91dHB1dAoKVGhlIGV4dHJhY3RlZCBPcGVuQVBJIHYyIGFuZCB2MyBmaWxlcyBjYW4gYmUgZm91bmQgaW4gY3VycmVudCBmb2xkZXIgd2hlcmUgdGhlIHYyIHZlcnNpb24gd2lsbCBiZSBzdG9yZWQgaW4gdGhlIGBzd2FnZ2VyLmpzb25gCmZpbGUgYW5kIHRoZSB2MyB2ZXJzaW9ucyB3aWxsIGJlIHN0b3JlZCBpbiBpbmRpdmlkdWFsIGZpbGVzIHBlciBncm91cCBpbiB0aGUgYC4vdjNgIGZvbGRlci4KClRvIG92ZXJyaWRlIHRoZSBsb2NhdGlvbiBvZiB0aGUgb3V0cHV0IHBhc3Mgb24gdGhlIGAtLW91dHB1dGAgZmxhZyBlLmcuIHZpYSBgLS1vdXRwdXQ9ZGV2YCBzdG9yZSBleHRyYWN0IHRoZSBmaWxlcyBpbnRvCnRoZSBgLi9kZXZgIGZvbGRlci4KCiMjIENvbnRyaWJ1dGluZwoKV2UnZCBsb3ZlIHRvIGdldCBmZWVkYmFjayBmcm9tIHlvdS4gUGxlYXNlIHJlcG9ydCBidWdzLCBzdWdnZXN0aW9ucyBvciBwb3N0IHF1ZXN0aW9ucyBieSBvcGVuaW5nIGEgR2l0SHViIGlzc3VlLgoKIyMgTGljZW5zZQoKQ29weXJpZ2h0IDIwMjIuCgpMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgpZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKCiAgICBodHRwOi8vd3d3LmFwYWNoZS5vcmcvbGljZW5zZXMvTElDRU5TRS0yLjAKClVubGVzcyByZXF1aXJlZCBieSBhcHBsaWNhYmxlIGxhdyBvciBhZ3JlZWQgdG8gaW4gd3JpdGluZywgc29mdHdhcmUKZGlzdHJpYnV0ZWQgdW5kZXIgdGhlIExpY2Vuc2UgaXMgZGlzdHJpYnV0ZWQgb24gYW4gIkFTIElTIiBCQVNJUywKV0lUSE9VVCBXQVJSQU5USUVTIE9SIENPTkRJVElPTlMgT0YgQU5ZIEtJTkQsIGVpdGhlciBleHByZXNzIG9yIGltcGxpZWQuClNlZSB0aGUgTGljZW5zZSBmb3IgdGhlIHNwZWNpZmljIGxhbmd1YWdlIGdvdmVybmluZyBwZXJtaXNzaW9ucyBhbmQKbGltaXRhdGlvbnMgdW5kZXIgdGhlIExpY2Vuc2UuCg== readmeEtag: '"292c94eae99bf01ce379b4552a33b4fc65ba3a17"' readmeLastModified: Mon, 12 Aug 2024 14:27:27 GMT repositoryId: 546069232 description: >- This project extracts the OpenAPI v2 and v3 specifications of a given Kubernetes API server created: '2022-10-05T13:22:32Z' updated: '2026-02-03T21:16:52Z' language: Go archived: false stars: 3 watchers: 3 forks: 1 owner: ironcore-dev logo: https://avatars.githubusercontent.com/u/147836484?v=4 license: Apache-2.0 repoEtag: '"ba9c96bc5290eb9cf3567d7380e472b357c1e98aa6ef1a548bdb7449ea392866"' repoLastModified: Tue, 03 Feb 2026 21:16:52 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/kinecosystem/ecosystem-api v3: true repositoryMetadata: base64Readme: >- # Kin Ecosystem SDK API

## marketplace API
The `openapi.yaml` file contains definitions of the services and models that will be available for our client to access on the server.

To edit the file you can use the [online editor](http://editor.swagger.io/?url=https://raw.githubusercontent.com/kinfoundation/ecosystem-api/master/openapi.yaml)

### JWT specs and examples:
#### JWT encoded messages
##### JWT header format

For all 3 JWTs we will use the following template:
```
{
	alg: string; // ES256 but can be discussed
	typ: string; // JWT
	kid: string; // identifier of the keypair that was used to sign the JWT. identifiers and public keys will be provided by signer authority. This enables using multiple private/public key pairs (a list of public keys and their ids need to be provided by signer authority to verifier in advanced)
}
```

#### Register payload
```
{
	// common/ standard fields
	iat: number;  // issued at - seconds from epoc
	iss: string; // issuer
	exp: number; // expiration
	sub: string; // subject - "register"

	// application fields
	user_id: string; // id of the user - or a deterministic unique id for the user (hash)
}
```
##### [Example (viewable on jwt.io)](https://jwt.io/?token=eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6InJlZ2lzdGVyIiwidXNlcl9pZCI6ImFfaGFzaF9vbl9raWtfdXNlcl9pZCJ9.6DwojspQ46inlwYwn3NNH0bmQIzKd7mL0VX8V2ZmlH8aZqWF2UbK_Md5kcxgnXgq0P6tExVTr1vxpwhfj7_3dg)
```
eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6InJlZ2lzdGVyIiwidXNlcl9pZCI6ImFfaGFzaF9vbl9raWtfdXNlcl9pZCJ9.6DwojspQ46inlwYwn3NNH0bmQIzKd7mL0VX8V2ZmlH8aZqWF2UbK_Md5kcxgnXgq0P6tExVTr1vxpwhfj7_3dg
```

#### Spend payload
```
{
	iat: number;  // issued at - seconds from epoc
	iss: string; // issuer
	exp: number; // expiration
	sub: string; // subject - "spend"

	nonce: string; // optional, to create a unique pair for offer id per user
	offer: {
		id: string; // offer id - id is decided by digital service
		amount: number; // amount of kin for this offer - price
	};
	sender: {
		user_id: string; // optional: user_id who will perform the
		title: string; // order title - appears in order history
		description: string; // order description - appears in order history
	};
}
```
##### [Example (viewable on jwt.io)](https://jwt.io/?token=eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6InNwZW5kIiwib2ZmZXIiOnsiaWQiOiJPMTIzMTIzMTIzIiwiYW1vdW50Ijo1MDAwfSwic2VuZGVyIjp7InRpdGxlIjoiQmx1ZSBUaGVtZSIsImRlc2NyaXB0aW9uIjoiT2NlYW4gQmx1ZSIsInVzZXJfaWQiOiJzb21lX3VzZXIifX0.BRacJ37zbbaKivWD_uhC2JXozzrHDN5B9VeG7d1BXjkpPYEfpaEy_kKF6xqp7oUMjEG2ltrOQPJ-UkFcFl6H-g)
```
eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6InNwZW5kIiwib2ZmZXIiOnsiaWQiOiJPMTIzMTIzMTIzIiwiYW1vdW50Ijo1MDAwfSwic2VuZGVyIjp7InRpdGxlIjoiQmx1ZSBUaGVtZSIsImRlc2NyaXB0aW9uIjoiT2NlYW4gQmx1ZSIsInVzZXJfaWQiOiJzb21lX3VzZXIifX0.BRacJ37zbbaKivWD_uhC2JXozzrHDN5B9VeG7d1BXjkpPYEfpaEy_kKF6xqp7oUMjEG2ltrOQPJ-UkFcFl6H-g
```
#### Earn payload
```
{
	iat: number;  // issued at - seconds from epoc
	iss: string; // issuer
	exp: number; // expiration
	sub: string; // subject - "earn"

	nonce: string; // optional, to create a unique pair for offer id per user
	offer: {
		id: string; // offer id - id is decided by digital service
		amount: number; // amount of kin for this offer - price
	};
	recipient: {
		user_id: string; // user_id who will perform the order
		title: string; // order title - appears in order history
		description: string; // order description - appears in order history
	};
}
```
##### [Example (viewable on jwt.io)](https://jwt.io/?token=eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6ImVhcm4iLCJvZmZlciI6eyJpZCI6Ik8xMjMxMjMxMjMiLCJhbW91bnQiOjUwMDB9LCJyZWNpcGllbnQiOnsidGl0bGUiOiJCbHVlIFRoZW1lIiwiZGVzY3JpcHRpb24iOiJPY2VhbiBCbHVlIiwidXNlcl9pZCI6InNvbWVfdXNlciJ9fQ.R7OpvaZQIAzjQ0MSi5nC1c39oC9oN08NVKwricMyWnuMbK5FD9Qn6ecmol4JnMGE5IZA7j_LR-EEbVhhEYi57g)
```
eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6ImVhcm4iLCJvZmZlciI6eyJpZCI6Ik8xMjMxMjMxMjMiLCJhbW91bnQiOjUwMDB9LCJyZWNpcGllbnQiOnsidGl0bGUiOiJCbHVlIFRoZW1lIiwiZGVzY3JpcHRpb24iOiJPY2VhbiBCbHVlIiwidXNlcl9pZCI6InNvbWVfdXNlciJ9fQ.R7OpvaZQIAzjQ0MSi5nC1c39oC9oN08NVKwricMyWnuMbK5FD9Qn6ecmol4JnMGE5IZA7j_LR-EEbVhhEYi57g
```
#### PayToUser payload
```
{
	iat: number;  // issued at - seconds from epoc
	iss: string; // issuer
	exp: number; // expiration
	sub: string; // subject - "pay_to_user"

	nonce: string; // optional, to create a unique pair for offer id per user
	offer: {
		id: string; // offer id - id is decided by digital service
		amount: number; // amount of kin for this offer - price
	};
	sender: {
		user_id: string; // optional: user_id who will perform the order
		title: string; // offer title - appears in order history
		description: string; // offer description - appears in order history
	};
	recipient: {
		user_id: string; // user_id who will receive the order
		title: string; // offer title - appears in order history
		description: string; // offer description - appears in order history
	};
}
```
##### [Example (viewable on jwt.io)](https://jwt.io/?tokeneyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6InBheV90b191c2VyIiwib2ZmZXIiOnsiaWQiOiJPMTIzMTIzMTIzIiwiYW1vdW50Ijo1MDAwfSwic2VuZGVyIjp7InRpdGxlIjoiVGlwIEZyb20gRG9vZHkiLCJkZXNjcmlwdGlvbiI6IkRvb2R5IHRpcHBlZCB5b3UiLCJ1c2VyX2lkIjoidXNlcjpuaXR6YW4ifSwicmVjaXBpZW50Ijp7InRpdGxlIjoiVGlwIFRvIE5pdHphbiIsImRlc2NyaXB0aW9uIjoiWW91IHRpcHBlZCBOaXR6YW4iLCJ1c2VyX2lkIjoidXNlcjpkb29keSJ9fQ.GVufyAI2UkpzzLlSL7a_kb5JcdSkgBb1PnzdL7wUkIGx-IUUu_pgTecElwTbZvWrCSr_GkJ4leD1MnXHSUb_QQ)
```
eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayBzZXJ2ZXIiLCJleHAiOjE1MjYyMzkwMjIsInN1YiI6InBheV90b191c2VyIiwib2ZmZXIiOnsiaWQiOiJPMTIzMTIzMTIzIiwiYW1vdW50Ijo1MDAwfSwic2VuZGVyIjp7InRpdGxlIjoiVGlwIEZyb20gRG9vZHkiLCJkZXNjcmlwdGlvbiI6IkRvb2R5IHRpcHBlZCB5b3UiLCJ1c2VyX2lkIjoidXNlcjpuaXR6YW4ifSwicmVjaXBpZW50Ijp7InRpdGxlIjoiVGlwIFRvIE5pdHphbiIsImRlc2NyaXB0aW9uIjoiWW91IHRpcHBlZCBOaXR6YW4iLCJ1c2VyX2lkIjoidXNlcjpkb29keSJ9fQ.GVufyAI2UkpzzLlSL7a_kb5JcdSkgBb1PnzdL7wUkIGx-IUUu_pgTecElwTbZvWrCSr_GkJ4leD1MnXHSUb_QQ
```
#### PaymentConfirmation payload
```
{
	iat: number;  // issued at - seconds from epoc
	iss: string; // issuer
	exp: number; // expiration
	sub: string; // subject - "payment_confirmation"

	sender_user_id: string; // user identifier - same value as given by register
	recipient_user_id: string; // user identifier - same value as given by register
	offer_id: string; // offer id - id is decided by digital service
	nonce: string; // the same as was send in the order JWT, or a default value in case none was used
	payment: {
		blockchain: string; // identifier of the blockchain network the transaction was made on
		transaction_id: string; // stellar identifier of the blockchain transaction
	};
}
```
##### [Example (viewable on jwt.io)](https://jwt.io/?token=eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayIsImV4cCI6MTUyNjIzOTAyMiwic3ViIjoicGF5bWVudF9jb25maXJtYXRpb24iLCJvZmZlcl9pZCI6Ik8xMjMxMjMxMjMiLCJzZW5kZXJfdXNlcl9pZCI6InVzZXI6ZG9vZHkiLCJyZWNpcGllbnRfdXNlcl9pZCI6InVzZXI6bml0emFuIiwicGF5bWVudCI6eyJibG9ja2NoYWluIjoic3RlbGxhci1tYWlubmV0IiwidHJhbnNhY3Rpb25faWQiOiJ0cmFuc2FjdGlvbjoxMjM0NSJ9fQ.-AbZOfC69eY1It43RccOXluY-sjWSi4JFvQkVKO9D2UgYU3jNPbEcBERLrqBHPSpS6f26LVpIsg5A81UQNoukw)
```
eyJhbGciOiJFUzI1NiIsInR5cCI6IkpXVCIsImtpZCI6InNvbWVfaWQifQ.eyJpYXQiOjE1MTYyMzkwMjIsImlzcyI6ImtpayIsImV4cCI6MTUyNjIzOTAyMiwic3ViIjoicGF5bWVudF9jb25maXJtYXRpb24iLCJvZmZlcl9pZCI6Ik8xMjMxMjMxMjMiLCJzZW5kZXJfdXNlcl9pZCI6InVzZXI6ZG9vZHkiLCJyZWNpcGllbnRfdXNlcl9pZCI6InVzZXI6bml0emFuIiwicGF5bWVudCI6eyJibG9ja2NoYWluIjoic3RlbGxhci1tYWlubmV0IiwidHJhbnNhY3Rpb25faWQiOiJ0cmFuc2FjdGlvbjoxMjM0NSJ9fQ.-AbZOfC69eY1It43RccOXluY-sjWSi4JFvQkVKO9D2UgYU3jNPbEcBERLrqBHPSpS6f26LVpIsg5A81UQNoukw
```

#### Key Pair used in Examples
Public:
```
-----BEGIN PUBLIC KEY-----
MFkwEwYHKoZIzj0CAQYIKoZIzj0DAQcDQgAEEVs/o5+uQbTjL3chynL4wXgUg2R9
q9UU8I5mEovUf86QZ7kOBIjJwqnzD1omageEHWwHdBO6B+dFabmdT9POxg==
-----END PUBLIC KEY-----
```
Private:
```
-----BEGIN PRIVATE KEY-----
MIGHAgEAMBMGByqGSM49AgEGCCqGSM49AwEHBG0wawIBAQQgevZzL1gdAFr88hb2
OF/2NxApJCzGCEDdfSp6VQO30hyhRANCAAQRWz+jn65BtOMvdyHKcvjBeBSDZH2r
1RTwjmYSi9R/zpBnuQ4EiMnCqfMPWiZqB4QdbAd0E7oH50VpuZ1P087G
-----END PRIVATE KEY-----
```

## internal payment

To edit the file you can use the [online editor](http://editor.swagger.io/?url=https://raw.githubusercontent.com/kinfoundation/ecosystem-api/master/payment.yaml)

## creating client stubs

openapi3 code generation is supported by a few projects

### java

using `github.com:swagger-api/swagger-codegen` on branch `v3.0.0-rc0` compile the maven project and run the following:
```
java -jar modules/swagger-codegen-cli/target/swagger-codegen-cli.jar generate -i ~/Workspace/ecosystem-api/openapi.yaml -l java  -o java_client
```
 readmeEtag: '"3d07024df169f71b5b319c3262feda2c79c513fb"' readmeLastModified: Wed, 12 Jun 2019 09:07:10 GMT repositoryId: 118761691 description: Apis for client server communication created: '2018-01-24T12:21:15Z' updated: '2023-01-28T08:54:54Z' language: Java archived: true stars: 3 watchers: 15 forks: 4 owner: kinecosystem logo: https://avatars.githubusercontent.com/u/28860603?v=4 repoEtag: '"55701ae4af9cece1b2ec4c39d6fd2262c6b62bf1fa2b13e6629e35b126426777"' repoLastModified: Sat, 28 Jan 2023 08:54:54 GMT foundInMaster: true category: Parsers id: e55fb0f4040961fcbf22287ee716269c - source: openapi3 tags repository: https://github.com/mantisnet/yang-to-openapi v3: true repositoryMetadata: base64Readme: >- IyB5YW5nLXRvLW9wZW5hcGkKWUFORyB0byBPcGVuQVBJIDMuMC54IGNvbnZlcnRlciB0byBzZWxmLWRvY3VtZW50IFlBTkcgbW9kZWxzCg== readmeEtag: '"3a038dba68cbabdef7e20976d3d97ada3ac13110"' readmeLastModified: Wed, 03 Jun 2020 20:39:25 GMT repositoryId: 269189795 description: YANG to OpenAPI 3.0.x converter to self-document YANG models created: '2020-06-03T20:39:24Z' updated: '2020-06-08T10:41:39Z' language: null archived: false stars: 3 watchers: 3 forks: 0 owner: MantisNet logo: https://avatars.githubusercontent.com/u/66088252?v=4 license: Apache-2.0 repoEtag: '"b83854af0ca531b4ba475bb3addb47c0c0fad22c8e178fec4a758003bb38ab05"' repoLastModified: Mon, 08 Jun 2020 10:41:39 GMT foundInMaster: true category: Parsers id: 6cf84626e965ab5c8d7b428d18aafde4 - source: openapi3 tags repository: https://github.com/husniadil/next-swagger-ui v3: true repositoryMetadata: base64Readme: >- IyBOZXh0LVN3YWdnZXItVUkKCioqTmV4dC1Td2FnZ2VyLVVJKiogd3JhcHMgW1N3YWdnZXIgVUldKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLXVpKSB1c2luZyBbTmV4dC5qc10oaHR0cHM6Ly9naXRodWIuY29tL3ZlcmNlbC9uZXh0LmpzKSBhbmQgW3N3YWdnZXItdWktcmVhY3RdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3N3YWdnZXItdWktcmVhY3QpIGNvbXBvbmVudC4KClVubGlrZSB0aGUgZGVmYXVsdCBpbXBsZW1lbnRhdGlvbiBvZiBbc3dhZ2dlci11aS1yZWFjdF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvc3dhZ2dlci11aS1yZWFjdCksIHRoZSAqKk5leHQtU3dhZ2dlci1VSSoqIGluY2x1ZGVzIFRvcGJhciBvdXQtb2YtdGhlLWJveCBmb3IgZHluYW1pYyBzY2hlbWEgbG9hZGluZy4KCiMjIERlcGxveSB5b3VyIG93bgoKWyFbRGVwbG95IHdpdGggVmVyY2VsXShodHRwczovL3ZlcmNlbC5jb20vYnV0dG9uKV0oaHR0cHM6Ly92ZXJjZWwuY29tL2ltcG9ydC9wcm9qZWN0P3RlbXBsYXRlPWh0dHBzOi8vZ2l0aHViLmNvbS9odXNuaWFkaWwvbmV4dC1zd2FnZ2VyLXVpKQo= readmeEtag: '"a32c8ad9a2865fd68a0862d2fa77796e7e16b4c9"' readmeLastModified: Tue, 01 Dec 2020 17:09:58 GMT repositoryId: 315043097 description: Wraps Swagger UI using Next.js created: '2020-11-22T13:23:45Z' updated: '2022-04-26T02:48:09Z' language: JavaScript archived: false stars: 3 watchers: 1 forks: 0 owner: husniadil logo: https://avatars.githubusercontent.com/u/10581130?v=4 license: Apache-2.0 repoEtag: '"f6e68f7d3383a7a6aea050d262bcc6ab00cfec103674434665b32d80b8f9963f"' repoLastModified: Tue, 26 Apr 2022 02:48:09 GMT foundInMaster: true category: - SDK - Code Generators id: a824a1b919bb33cccd842418c40d075b - source: - openapi3 tags - openapi31 tags repository: https://github.com/api-flows/api-flows-studio v3: true v3_1: true id: 49c00ed55edcee7a3f8e051e5e1b6334 repositoryMetadata: base64Readme: >- IyBBUEkgRmxvd3MgU3R1ZGlvCgpBUEkgd29ya2Zsb3cgdmlld2VyLgoKIyMgT3ZlcnZpZXcKClRoZSBBUEkgRmxvd3MgU3R1ZGlvIGlzIGEgd2ViIGFwcGxpY2F0aW9uIHRoYXQgbG9hZHMgYW5kIGRpc3BsYXlzIGFuIE9wZW5BUEkgd29ya2Zsb3cgZmlsZSwgYmFzZWQgb24gdGhlIFtPQUkgQXJhenpvIHNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvQXJhenpvLVNwZWNpZmljYXRpb24vKS4KCjxwIGFsaWduPSJjZW50ZXIiPgogICAgPGltZyBzcmM9ImhvbWUuZ2lmIj4KPC9wPgoKIyMgQWNjZXNzIHRoZSB3ZWIgc2l0ZQoKVHJ5IGl0IG91dCBvbiBbYXBpLWZsb3dzLmNvbV0oaHR0cHM6Ly9hcGktZmxvd3MuY29tLykKCiMjIFJ1biBvbiBHaXRwb2QKClshW09wZW4gaW4gR2l0cG9kXShodHRwczovL2dpdHBvZC5pby9idXR0b24vb3Blbi1pbi1naXRwb2Quc3ZnKV0oaHR0cHM6Ly9naXRwb2QuaW8vI2h0dHBzOi8vZ2l0aHViLmNvbS9BUEktRmxvd3MvYXBpLWZsb3dzLXN0dWRpby90cmVlL21haW4pCgoKIyMgUnVuIG9uIGxvY2FsCgpDbG9uZSBmcm9tIHRoZSBHaXRIdWIgcmVwb3NpdG9yeQoKYGBgYmFzaAogIGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vQVBJLUZsb3dzL2FwaS1mbG93cy1zdHVkaW8uZ2l0CiAgY2QgYXBpLWZsb3dzLXN0dWRpbwogIG12biBzcHJpbmctYm9vdDpydW4gLURza2lwLmZyb250ZW5kLmJ1aWxkPWZhbHNlCmBgYAo= readmeEtag: '"55f24c71d7032a7513376c3a4fe65828fdaf7215"' readmeLastModified: Tue, 18 Jun 2024 18:37:39 GMT repositoryId: 736327040 description: 'Visualize OpenAPI (Arazzo) workflows ' created: '2023-12-27T15:51:24Z' updated: '2025-09-17T13:49:19Z' language: Java archived: false stars: 6 watchers: 1 forks: 0 owner: API-Flows logo: https://avatars.githubusercontent.com/u/153562650?v=4 license: Apache-2.0 repoEtag: '"b6a5629e18b4eb45f8df091a97b5b08f80a2c932c6b25965e15e4f9c74e431cc"' repoLastModified: Wed, 17 Sep 2025 13:49:19 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/bhaveshdev09/vendor-hub-api v3: true id: db572b2a5135603185743334e3579f99 repositoryMetadata: base64Readme: >- # Vendor Hub API

A Vendor Management System(VMS) apis  for efficient vendor profile management, purchase order tracking, and vendor performance metrics calculation.

![Python version](https://img.shields.io/badge/Python-3.10.8-4c566a?logo=python&&longCache=true&logoColor=white&colorB=pink&style=flat-square&colorA=4c566a) ![Django version](https://img.shields.io/badge/Django-4.2.8-4c566a?logo=django&&longCache=truelogoColor=white&colorB=pink&style=flat-square&colorA=4c566a) ![Django-RestFramework version](https://img.shields.io/badge/Django_Rest_Framework-3.14.0-red.svg?longCache=true&style=flat-square&logo=django&logoColor=white&colorA=4c566a&colorB=pink)  ![Last Commit](https://img.shields.io/github/last-commit/bhaveshdev09/vendor-hub-api/master?&&longCache=true&logoColor=white&colorB=green&style=flat-square&colorA=4c566a) [![License](https://img.shields.io/badge/license-MIT-blue.svg)](https://opensource.org/licenses/MIT)

## Table of Contents

- [Core Features](#features)
- [Project Plan](#planning)
- [Installation](#installation)
- [API Endpoints](#api-endpoints)
- [Testing](#testing)
- [Contributing](#contributing)
- [License](#license)

## Core Features

1. **Vendor Profile Management:**

   - Create, retrieve, update, and delete vendor profiles.
   - Track vendor information including name, contact details, address, and a unique vendor code.
2. **Purchase Order Tracking:**

   - Create, retrieve, update, and delete purchase orders.
   - Track purchase order details such as PO number, vendor reference, order date, items, quantity, and status.
3. **Vendor Performance Evaluation:**

   - Calculate vendor performance metrics, including on-time delivery rate, quality rating average, average response time, and fulfillment rate.
   - Retrieve performance metrics for a specific vendor.

## Project Plan 

To access the project plan, please visit the following URL: [Vendor Hub API](https://github.com/users/bhaveshdev09/projects/1)

## Installation

1. Clone the repository:

   ```bash
   git clone https://github.com/bhaveshdev09/vendor-hub-api.git backend
   cd backend
   ```
2. Create a virtual environment and install dependencies:

   ```bash
   python -m venv venv
   source venv/bin/activate  # On Windows, use `venv\Scripts\activate`
   pip install -r requirements.txt
   ```
3. Apply database migrations:

   ```bash
   python manage.py migrate
   ```
4. Run the development server:

   ```bash
   python manage.py runserver
   ```
5. Set the environment variables
   
   > Please refer **.env.example** for must have variables

6. Access the application at http://localhost:8000.

## API Endpoints

Below is a summary of the available API endpoints:

**Vendor Profile Management**

| Endpoint               | Method      | Description                           |
| ---------------------- | ----------- | ------------------------------------- |
| `/api/vendors/`      | POST        | Create a new vendor.                  |
| `/api/vendors/`      | GET         | List all vendors.                     |
| `/api/vendors/{id}/` | GET         | Retrieve a specific vendor's details. |
| `/api/vendors/{id}/` | PUT / PATCH | Update a vendor's details.            |
| `/api/vendors/{id}/` | DELETE      | Delete a vendor.                      |

**Purchase Order Tracking**

| Endpoint                                  | Method       | Description                                    |
| ----------------------------------------- | ------------ | ---------------------------------------------- |
| `/api/purchase_orders/`                 | POST         | Create a purchase order.                       |
| `/api/purchase_orders/`                 | GET          | List all purchase orders.                      |
| `/api/purchase_orders/{id}/`            | GET          | Retrieve details of a specific purchase order. |
| `/api/purchase_orders/{id}/`            | PUT / PATCH  | Update a purchase order.                       |
| `/api/purchase_orders/{id}/`            | DELETE       | Delete a purchase order.                       |
| `/api/purchase_orders/{id}/acknowledge` | PUT / PATCH | Acknowledge a purchase order.                  |

**Vendor Performance Evaluation**

| Endpoint                                  | Method | Description                              |
| ----------------------------------------- | ------ | ---------------------------------------- |
| `/api/vendors/{id}/performance`         | GET    | Retrieve a vendor's performance metrics. |
| `/api/vendors/{id}/performance/history` | GET    | Retrieve a vendor's performance history. |

**Additional Docs**

| Endpoint                                      | Method | Description                                                   |
| --------------------------------------------- | ------ | ------------------------------------------------------------- |
| `/api/vendors/{id}/performance/schema/doc/` | GET    | List all the apis in Open API3 documentation format.          |
| `/api/vendors/{id}/performance/schema/doc/` | GET    | List all the apis in Open API3 detailed documentation format. |

## Testing

Run the tests using the following command:

```bash
python manage.py test
```

To check coverage report of the entire codebases try this command:

```bash
coverage run manage.py test

# to get the coverage report on command line
coverage report -m

# To get the html report of code coverage try this
coverage html
```

## Contributing

Inputs and contributions to this project are appreciated. To make them as transparent and easy as possible, please follow this steps:

- ### How to contribute:


  1. Fork the repository and create your branch from master with different name.
  2. Clone the project to your own machine
  3. Commit changes to your own branch
  4. Push your work back up to your fork
  5. Submit a Pull request

  ### Don't:

  - Don't include any license information when submitting your code as this repository is MIT licensed, and so your submissions are understood to be under the same MIT License as well.
- ### How to report a bug:


  1. Open a new Issue.
  2. Write a bug report with details, background, and when possible sample code. That's it!

## License

This project is licensed under the [MIT License](https://opensource.org/licenses/MIT).
 readmeEtag: '"ba1579aea290e702ce8186039366fbace90cc512"' readmeLastModified: Sat, 09 Dec 2023 20:19:38 GMT repositoryId: 726566533 description: >- A comprehensive Vendor Management System(VMS) apis for efficient vendor profile management, purchase order tracking, and vendor performance metrics calculation created: '2023-12-02T18:34:06Z' updated: '2025-11-23T19:00:58Z' language: Python archived: false stars: 4 watchers: 1 forks: 0 owner: bhaveshdev09 logo: https://avatars.githubusercontent.com/u/45851391?v=4 license: MIT repoEtag: '"9faa2bbe2abdb5d0dec648158dfea229a2191e754016c28e4301be41adf606d7"' repoLastModified: Sun, 23 Nov 2025 19:00:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/swaggest/rest-fasthttp v3: true id: 6d32533fe2615e6613b2e30dc8a93bcd repositoryMetadata: base64Readme: >- # REST with Clean Architecture for Go

[![Build Status](https://github.com/swaggest/rest/workflows/test/badge.svg)](https://github.com/swaggest/rest/actions?query=branch%3Amaster+workflow%3Atest)
[![Coverage Status](https://codecov.io/gh/swaggest/rest/branch/master/graph/badge.svg)](https://codecov.io/gh/swaggest/rest)
[![GoDevDoc](https://img.shields.io/badge/dev-doc-00ADD8?logo=go)](https://pkg.go.dev/github.com/swaggest/rest)
[![Time Tracker](https://wakatime.com/badge/github/swaggest/rest.svg)](https://wakatime.com/badge/github/swaggest/rest)
![Code lines](https://sloc.xyz/github/swaggest/rest/?category=code)
![Comments](https://sloc.xyz/github/swaggest/rest/?category=comments)

This module implements HTTP transport level for [`github.com/swaggest/usecase`](https://github.com/swaggest/usecase) 
to build REST services.

## This Fork

This is a reduced fork of [`github.com/swaggest/rest`](https://github.com/swaggest/rest) 
that uses [`fasthttp`](https://github.com/valyala/fasthttp) instead of standard `net/http`.

`fasthttp` brings better performance and memory efficiency to highly loaded services, 
but it is not compatible with many existing middlewares and may require additional 
dev effort for such cases.

## Goals

* Maintain single source of truth for documentation, validation and input/output of HTTP API.
* Avoid dependency on compile time code generation.
* Improve productivity and reliability by abstracting HTTP details with simple API for common case.
* Allow low-level customizations for advanced cases.
* Maintain reasonable performance with low GC impact.

## Non-Goals

* Support for legacy documentation schemas like Swagger 2.0 or RAML.
* Zero allocations.
* Explicit support for XML in request or response bodies.

## Features

* Compatible with `net/http`.
* Built with [`github.com/go-chi/chi`](https://github.com/go-chi/chi) router.
* Modular flexible structure.
* HTTP [request mapping](#request-decoder) into Go value based on field tags.
* Decoupled business logic with Clean Architecture use cases.
* Automatic type-safe OpenAPI 3 documentation with [`github.com/swaggest/openapi-go`](https://github.com/swaggest/openapi-go).
* Single source of truth for the documentation and endpoint interface.
* Automatic request/response JSON schema validation with [`github.com/santhosh-tekuri/jsonschema`](https://github.com/santhosh-tekuri/jsonschema).
* Dynamic gzip compression and fast pass through mode.
* Optimized performance.
* Embedded [Swagger UI](https://swagger.io/tools/swagger-ui/).
* Integration test helpers.
* Generic interface for [use case interactors](https://pkg.go.dev/github.com/swaggest/usecase#NewInteractor). 

## Usage

Please check this [tutorial](https://dev.to/vearutop/tutorial-developing-a-restful-api-with-go-json-schema-validation-and-openapi-docs-2490) for end-to-end usage example.

### Request Decoder

Go struct with optional field tags defines input port. 
Request decoder populates field values from `http.Request` data before use case interactor is invoked. 

```go
// Declare input port type.
type helloInput struct {
    Locale string `query:"locale" default:"en-US" pattern:"^[a-z]{2}-[A-Z]{2}$" enum:"ru-RU,en-US"`
    Name   string `path:"name" minLength:"3"` // Field tags define parameter location and JSON schema constraints.

    // Field tags of unnamed fields are applied to parent schema, 
	// they are optional and can be used to disallow unknown parameters.
    // For non-body params, name tag must be provided explicitly.
    // E.g. here no unknown `query` and `cookie` parameters allowed,
    // unknown `header` params are ok.
    _ struct{} `query:"_" cookie:"_" additionalProperties:"false"`
}
```

Input data can be located in:
* `path` parameter in request URI, e.g. `/users/{name}`,
* `query` parameter in request URI, e.g. `/users?locale=en-US`,
* `formData` parameter in request body with `application/x-www-form-urlencoded` or `multipart/form-data` content,
* `json` parameter in request body with `application/json` content,
* `cookie` parameter in request cookie,
* `header` parameter in request header.

For more explicit separation of concerns between use case and transport it is possible to provide request mapping 
separately when initializing handler (please note, such mapping is [not applied](https://github.com/swaggest/rest/issues/61#issuecomment-1059851553) to `json` body).

```go
// Declare input port type.
type helloInput struct {
    Locale string `default:"en-US" pattern:"^[a-z]{2}-[A-Z]{2}$"`
    Name   string `minLength:"3"` // Field tags define parameter location and JSON schema constraints.
}
```

```go
// Add use case handler with custom input mapping to router.
r.Method(http.MethodGet, "/hello/{name}", fhttp.NewHandler(u,
    fhttp.RequestMapping(new(struct {
       Locale string `query:"locale"`
       Name   string `path:"name"` // Field tags define parameter location and JSON schema constraints.
    })),
))
```

Additional field tags describe JSON schema constraints, please check 
[documentation](https://pkg.go.dev/github.com/swaggest/jsonschema-go#Reflector.Reflect).

More schema customizations are possible with [`github.com/swaggest/jsonschema-go interfaces`](https://github.com/swaggest/jsonschema-go#implementing-interfaces-on-a-type).

By default `default` tags are only contributing to documentation, 
if [`request.DecoderFactory.ApplyDefaults`](https://pkg.go.dev/github.com/swaggest/rest/request#DecoderFactory) is 
set to `true`, fields of request structure that don't have explicit value but have `default` will be populated with 
default value.

If input structure implements [`request.Loader`](https://pkg.go.dev/github.com/swaggest/rest/request#Loader),  
then `LoadFromHTTPRequest(r *http.Request) error` method will be invoked to populate input structure instead 
of automatic decoding. This allows low level control for cases that need it.

### Response Encoder

Go struct with field tags defines output port.
Response encoder writes data from output to `http.ResponseWriter` after use case interactor invocation finishes.

```go
// Declare output port type.
type helloOutput struct {
    Now     time.Time `header:"X-Now" json:"-"`
    Message string    `json:"message"`
}
```

Output data can be located in:
* `json` for response body with `application/json` content,
* `header` for values in response header.

For more explicit separation of concerns between use case and transport it is possible to provide response header mapping 
separately when initializing handler.

```go
// Declare output port type.
type helloOutput struct {
    Now     time.Time `json:"-"`
    Message string    `json:"message"`
}
```

```go
// Add use case handler with custom output headers mapping to router.
r.Method(http.MethodGet, "/hello/{name}", fhttp.NewHandler(u,
    fhttp.ResponseHeaderMapping(new(struct {
        Now     time.Time `header:"X-Now"`
    })),
))
```

Additional field tags describe JSON schema constraints, please check 
[documentation](https://pkg.go.dev/github.com/swaggest/jsonschema-go#Reflector.Reflect).

### Creating Use Case Interactor

HTTP transport is decoupled from business logic by adapting
[use case interactors](https://pkg.go.dev/github.com/swaggest/usecase#Interactor).

Use case interactor can define input and output ports that are used to map data between Go values and transport.
It can provide information about itself that will be exposed in generated documentation.

```go
// Create use case interactor with references to input/output types and interaction function.
u := usecase.NewIOI(new(helloInput), new(helloOutput), func(ctx context.Context, input, output interface{}) error {
    var (
        in  = input.(*helloInput)
        out = output.(*helloOutput)
    )

    msg, available := messages[in.Locale]
    if !available {
        return status.Wrap(errors.New("unknown locale"), status.InvalidArgument)
    }

    out.Message = fmt.Sprintf(msg, in.Name)
    out.Now = time.Now()

    return nil
})
```

For modularity particular use case interactor instance can be assembled by embedding relevant traits in a struct,
for example you can skip adding `usecase.WithInput` if your use case does not imply any input.

```go
// Create use case interactor.
u := struct {
    usecase.Info
    usecase.Interactor
    usecase.WithInput
    usecase.WithOutput
}{}

// Describe use case interactor.
u.SetTitle("Greeter")
u.SetDescription("Greeter greets you.")
u.Input = new(helloInput)
u.Output = new(helloOutput)
u.Interactor = usecase.Interact(func(ctx context.Context, input, output interface{}) error {
    // Do something about input to prepare output.
    return nil
})
```

### Initializing Web Service

[Web Service](https://pkg.go.dev/github.com/swaggest/rest/web#DefaultService) is an instrumented facade in front of 
router, it simplifies configuration and provides more compact API to add use cases.

```go
// Service initializes router with required middlewares.
service := web.DefaultService()

// It allows OpenAPI configuration.
service.OpenAPI.Info.Title = "Albums API"
service.OpenAPI.Info.WithDescription("This service provides API to manage albums.")
service.OpenAPI.Info.Version = "v1.0.0"

// Additional middlewares can be added.
service.Use(
    middleware.StripSlashes,
)

// Use cases can be mounted using short syntax .<Method>(...).
service.Post("/albums", postAlbums(), fhttp.SuccessStatus(http.StatusCreated))

log.Println("Starting service at http://localhost:8080")

if err := http.ListenAndServe("localhost:8080", service); err != nil {
    log.Fatal(err)
}

```

Usually, `web.Service` API is sufficient, but if it is not, router can be configured manually, please check 
the documentation below.

### Adding use case to router

```go
// Add use case handler to router.
r.Method(http.MethodGet, "/hello/{name}", fhttp.NewHandler(u))
```

## API Schema Collector

OpenAPI schema should be initialized with general information about REST API.

It uses [type-safe mapping](https://github.com/swaggest/openapi-go) for the configuration, 
so any IDE will help with available fields. 

```go
// Init API documentation schema.
apiSchema := &openapi.Collector{}
apiSchema.Reflector().SpecEns().Info.Title = "Basic Example"
apiSchema.Reflector().SpecEns().Info.WithDescription("This app showcases a trivial REST API.")
apiSchema.Reflector().SpecEns().Info.Version = "v1.2.3"
```

## Router Setup

REST router is based on [`github.com/go-chi/chi`](https://github.com/go-chi/chi), wrapper allows unwrapping instrumented
handler in middleware.

These middlewares are required:
* `nethttp.OpenAPIMiddleware(apiSchema)`, 
* `request.DecoderMiddleware(decoderFactory)`,
* `response.EncoderMiddleware`.

Optionally you can add more middlewares with some performance impact:
* `request.ValidatorMiddleware(validatorFactory)` (request validation, recommended)
* `response.ValidatorMiddleware(validatorFactory)`
* `gzip.Middleware`

You can also add any other 3rd party middlewares compatible with `net/http` at your discretion.

```go
// Setup request decoder and validator.
validatorFactory := jsonschema.NewFactory(apiSchema, apiSchema)
decoderFactory := request.NewDecoderFactory()
decoderFactory.SetDecoderFunc(rest.ParamInPath, chirouter.PathToURLValues)

// Create router.
r := chirouter.NewWrapper(fchi.NewRouter())

// Setup middlewares.
r.Use(
    middleware.Recoverer,                          // Panic recovery.
    nethttp.OpenAPIMiddleware(apiSchema),          // Documentation collector.
    request.DecoderMiddleware(decoderFactory),     // Request decoder setup.
    request.ValidatorMiddleware(validatorFactory), // Request validator setup.
    response.EncoderMiddleware,                    // Response encoder setup.
    gzip.Middleware,                               // Response compression with support for direct gzip pass through.
)
```

Register Swagger UI to serve documentation at `/docs`.

```go
// Swagger UI endpoint at /docs.
r.Method(http.MethodGet, "/docs/openapi.json", apiSchema)
r.Mount("/docs", v3cdn.NewHandler(apiSchema.Reflector().Spec.Info.Title,
    "/docs/openapi.json", "/docs"))
```

## Security Setup

```go
// Prepare middleware with suitable security schema.
// It will perform actual security check for every relevant request.
adminAuth := middleware.BasicAuth("Admin Access", map[string]string{"admin": "admin"})

// Prepare API schema updater middleware.
// It will annotate handler documentation with security schema.
adminSecuritySchema := nethttp.HTTPBasicSecurityMiddleware(apiSchema, "Admin", "Admin access")

// Endpoints with admin access.
r.Route("/admin", func(r fchi.Router) {
    r.Group(func(r fchi.Router) {
        r.Wrap(adminAuth, adminSecuritySchema) // Add both middlewares to routing group to enforce and document security.
        r.Method(http.MethodPut, "/hello/{name}", fhttp.NewHandler(u))
    })
})
```

See [example](./_examples/task-api/internal/infra/nethttp/router.go).

## Handler Setup

Handler is a generalized adapter for use case interactor, so usually setup is trivial.

```go
// Add use case handler to router.
r.Method(http.MethodGet, "/hello/{name}", fhttp.NewHandler(u))
```

## Example

For non-generic use case, see another [example](./_examples/basic/main.go).

```go
package main

import (
	"context"
	"errors"
	"fmt"
	"log"
	"time"

	"github.com/swaggest/fchi"
	"github.com/swaggest/rest-fasthttp/response/gzip"
	"github.com/swaggest/rest-fasthttp/web"
	swgui "github.com/swaggest/swgui/v4emb"
	"github.com/swaggest/usecase"
	"github.com/swaggest/usecase/status"
	"github.com/valyala/fasthttp"
)

func main() {
	s := web.DefaultService()

	// Init API documentation schema.
	s.OpenAPI.Info.Title = "Basic Example"
	s.OpenAPI.Info.WithDescription("This app showcases a trivial REST API.")
	s.OpenAPI.Info.Version = "v1.2.3"

	// Setup middlewares.
	s.Wrap(
		gzip.Middleware, // Response compression with support for direct gzip pass through.
	)

	// Declare input port type.
	type helloInput struct {
		Locale string `query:"locale" default:"en-US" pattern:"^[a-z]{2}-[A-Z]{2}$" enum:"ru-RU,en-US"`
		Name   string `path:"name" minLength:"3"` // Field tags define parameter location and JSON schema constraints.

		// Field tags of unnamed fields are applied to parent schema.
		// they are optional and can be used to disallow unknown parameters.
		// For non-body params, name tag must be provided explicitly.
		// E.g. here no unknown `query` and `cookie` parameters allowed,
		// unknown `header` params are ok.
		_ struct{} `query:"_" cookie:"_" additionalProperties:"false"`
	}

	// Declare output port type.
	type helloOutput struct {
		Now     time.Time `header:"X-Now" json:"-"`
		Message string    `json:"message"`
	}

	messages := map[string]string{
		"en-US": "Hello, %s!",
		"ru-RU": "Привет, %s!",
	}

	// Create use case interactor with references to input/output types and interaction function.
	u := usecase.NewInteractor(func(ctx context.Context, input helloInput, output *helloOutput) error {
		msg, available := messages[input.Locale]
		if !available {
			return status.Wrap(errors.New("unknown locale"), status.InvalidArgument)
		}

		output.Message = fmt.Sprintf(msg, input.Name)
		output.Now = time.Now()

		return nil
	})

	// Describe use case interactor.
	u.SetTitle("Greeter")
	u.SetDescription("Greeter greets you.")

	u.SetExpectedErrors(status.InvalidArgument)

	// Add use case handler to router.
	s.Get("/hello/{name}", u)

	// Swagger UI endpoint at /docs.
	s.Docs("/docs", swgui.New)

	// Start server.
	log.Println("http://localhost:8011/docs")
	if err := fasthttp.ListenAndServe(":8011", fchi.RequestHandler(s)); err != nil {
		log.Fatal(err)
	}
}
```

![Documentation Page](./_examples/basic/screen.png)

## Versioning

This project adheres to [Semantic Versioning](https://semver.org/#semantic-versioning-200).

Before version `1.0.0`, breaking changes are tagged with `MINOR` bump, features and fixes are tagged with `PATCH` bump.
After version `1.0.0`, breaking changes are tagged with `MAJOR` bump.

Breaking changes are described in [UPGRADE.md](./UPGRADE.md).
 readmeEtag: '"58c198128c1f22c1ede67899d800e7273202e40e"' readmeLastModified: Mon, 27 Jun 2022 07:33:50 GMT repositoryId: 507534984 description: Web services with OpenAPI and JSON Schema done quick in Go created: '2022-06-26T09:49:48Z' updated: '2025-05-12T06:10:23Z' language: Go archived: false stars: 4 watchers: 0 forks: 0 owner: swaggest logo: https://avatars.githubusercontent.com/u/19609628?v=4 license: MIT repoEtag: '"ba4cb14311eca673ca25b26323c0e6c3689bd1e71f63bf337e06c8ccd43cdcea"' repoLastModified: Mon, 12 May 2025 06:10:23 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/srfrnk/crd-api-doc-gen v3: true repositoryMetadata: base64Readme: >- IyBjcmQtYXBpLWRvYy1nZW4KCkt1YmVybmV0ZXMgQ1JEIEFQSSBHZW5lcmF0b3IKCiMjIEJhc2ljIFVzYWdlCgpUbyBnZW5lcmF0ZSBhbiBgSFRNTGAgQVBJIGRvY3VtZW50YXRpb24gcGFnZToKCjEuIE9idGFpbiBzb21lIGBqc29uYCBvciBgeWFtbGAgbWFuaWZlc3QgZmlsZXMgdGhhdCBjb250YWluIG9uZSBvciBtb3JlIGBDdXN0b21SZXNvdXJjZURlZmluaXRpb25gIG9iamVjdHMKMS4gKipPcHRpb25hbGx5Kiogb2J0YWluIGEgZmlsZSB3aXRoIHNvbWUgYXBpIGluZm8uCjEuIFJ1bjogYGRvY2tlciBydW4gLS1ybSAtdiAkUFdELzxQQVRIX1RPX1lPVVJfRklMRVM+Oi88QU5ZX1BBVEg+IGdoY3IuaW8vc3Jmcm5rL2NyZC1hcGktZG9jLWdlbjo8VkVSU0lPTl9UQUc+IDxJTlBVVF9GT0xERVJfUEFUSD4gPE9VVFBVVF9GT0xERVJfUEFUSD4gWzxBUElfSU5GT19QQVRIPl1gCjEuICoqTm90ZSoqIHRoYXQgdGhlc2UgcGF0aHMgYXJlIGludGVybmFsIHRvIHRoZSBjb250YWluZXIuIEUuZy4gYGRvY2tlciBydW4gLS1ybSAtdiAvbXlib3gvbXlmaWxlczovaW50ZXJuYWwgZ2hjci5pby9zcmZybmsvY3JkLWFwaS1kb2MtZ2VuOmxhdGVzdCAvaW50ZXJuYWwvc29tZS9wYXRoIC9pbnRlcm5hbC9hbm90aGVyL3BhdGhgCjEuIGA8SU5QVVRfRk9MREVSX1BBVEg+YCBuZWVkcyB0byBjb250YWluIGFueSBgeWFtbGAgb3IgYGpzb25gIEs4cyBtYW5pZmVzdCBmaWxlcyB3aXRoIENSRCBkZWZpbml0aW9ucy4KMS4gRmluZCBnZW5lcmF0ZWQgSFRNTCBwYWdlIGBpbmRleC5odG1sYCBpbnNpZGUgdGhlIHNwZWNpZmllZCBvdXRwdXQgZm9sZGVyLgoKIyMgQVBJIEluZm8gZmlsZQoKVG8gY3VzdG9taXplIGFwaSBpbmZvIHlvdSBjYW4gc3BlY2lmeSBhbiBhcGktaW5mbyBmaWxlLgpUaGlzIGZpbGUgbXVzdCBiZSBpbiBgeWFtbGAvYGpzb25gIGZvcm1hdCBhbmQgaGF2ZSB0aGUgZm9sbG93aW5nIHN0cnVjdHVyZToKCmBgYHlhbWwKYXBpLWluZm86CiAgdmVyc2lvbjogPEFOWSBTVFJJTkc+CiAgdGl0bGU6IDxBTlkgU1RSSU5HPgogIGRlc2NyaXB0aW9uOiA8QU5ZIE1BUktET1dOIFRFWFQ+CmBgYAoKIyMgQ3JlZGl0cwoKMS4gQmFzZWQgb24gW29wZW5hcGktZ2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKQoxLiBVc2luZyBbeXFdKGh0dHBzOi8vZ2l0aHViLmNvbS9taWtlZmFyYWgveXEpCg== readmeEtag: '"81400118c011ebfbb4996d87a2f1fe67bad4fba8"' readmeLastModified: Sun, 19 Dec 2021 16:57:32 GMT repositoryId: 426717001 description: Kubernetes CRD API Generator created: '2021-11-10T17:36:05Z' updated: '2025-09-03T07:08:17Z' language: Mustache archived: false stars: 5 watchers: 0 forks: 3 owner: srfrnk logo: https://avatars.githubusercontent.com/u/1501654?v=4 license: MIT repoEtag: '"6310fb0cf4e2dc98082ce5a9a4015e9eb3178ae2cba34fc034afadf729f553fa"' repoLastModified: Wed, 03 Sep 2025 07:08:17 GMT foundInMaster: true category: Code Generators id: 11c2eb0e66e56fa8344ebdd16b47d794 - source: openapi3 tags repository: https://github.com/dernasherbrezon/vue-openapi-bootstrap v3: true repositoryMetadata: base64Readme: >- IyMgU2NyZWVuc2hvdHMKCiFbc2NyZWVuMV0oL2RvYy8xLnBuZz9yYXc9dHJ1ZSkKIVtzY3JlZW4yXSgvZG9jLzIucG5nP3Jhdz10cnVlKQoKIyMgVXNhZ2UKCmBgYGphdmFzY3JpcHQKPHRlbXBsYXRlPgogIDx2dWUtb3BlbmFwaS1ib290c3RyYXAgOm9wZW5hcGk9Im9wZW5hcGkiPjwvdnVlLW9wZW5hcGktYm9vdHN0cmFwPgo8L3RlbXBsYXRlPgoKPHNjcmlwdD4KaW1wb3J0IHZ1ZU9wZW5hcGlCb290c3RyYXAgZnJvbSAndnVlLW9wZW5hcGktYm9vdHN0cmFwL3NyYy92dWUtb3BlbmFwaS1ib290c3RyYXAnCmV4cG9ydCBkZWZhdWx0IHsKICBjb21wb25lbnRzOiB7dnVlT3BlbmFwaUJvb3RzdHJhcH0sCiAgZGF0YSAoKSB7CiAgICByZXR1cm4gewogICAgICBvcGVuYXBpOiB7fQogICAgfQogIH0KfQo8L3NjcmlwdD4KYGBgCgojIyBCdWlsZCBTZXR1cAoKYGBgIGJhc2gKIyBpbnN0YWxsIGRlcGVuZGVuY2llcwpucG0gaW5zdGFsbAoKIyBidWlsZCBmb3IgcHJvZHVjdGlvbiB3aXRoIG1pbmlmaWNhdGlvbgpucG0gcnVuIGJ1aWxkCgpgYGAKCkZvciBhIGRldGFpbGVkIGV4cGxhbmF0aW9uIG9uIGhvdyB0aGluZ3Mgd29yaywgY2hlY2sgb3V0IHRoZSBbZ3VpZGVdKGh0dHA6Ly92dWVqcy10ZW1wbGF0ZXMuZ2l0aHViLmlvL3dlYnBhY2svKSBhbmQgW2RvY3MgZm9yIHZ1ZS1sb2FkZXJdKGh0dHA6Ly92dWVqcy5naXRodWIuaW8vdnVlLWxvYWRlcikuIAo= readmeEtag: '"2f3f3bc03fe99ba3ffbae4cc81843eb6a1805fff"' readmeLastModified: Fri, 11 Jun 2021 07:21:08 GMT repositoryId: 222297022 description: OpenAPI viewer component for VueJS created: '2019-11-17T19:00:16Z' updated: '2024-11-27T17:42:28Z' language: Vue archived: false stars: 3 watchers: 2 forks: 0 owner: dernasherbrezon logo: https://avatars.githubusercontent.com/u/1614424?v=4 license: Apache-2.0 repoEtag: '"16c12d110a1e48e623c951aab32801d95bc327175984847abbb91e5c0ec14512"' repoLastModified: Wed, 27 Nov 2024 17:42:28 GMT foundInMaster: true category: - Documentation - Parsers id: 5907b7f8d6ec3847e6b8965336450484 - source: openapi3 tags repository: https://github.com/adamko-dev/zally-gradle-plugin v3: true id: 0153bb0c58b4017bbb9027350ca69752 repositoryMetadata: base64Readme: >- IyBaYWxseSBHcmFkbGUgUGx1Z2luCgpbWmFsbHkgR3JhZGxlIFBsdWdpbl0oaHR0cHM6Ly9naXRodWIuY29tL2FkYW1rby1kZXYvemFsbHktZ3JhZGxlLXBsdWdpbikgaXMgYW4KdW5vZmZpY2lhbCBHcmFkbGUgUGx1Z2luIGZvciBbWmFsbHldKGh0dHBzOi8vZ2l0aHViLmNvbS96YWxhbmRvL3phbGx5KSwgdGhlCm1pbmltYWxpc3RpYywgc2ltcGxlLXRvLXVzZSBPcGVuQVBJIDIgYW5kIDMgbGludGVyLgoKVGhlIFphbGx5IEdyYWRsZSBwbHVnaW4gaXMgYmFzZWQgb24gdGhlCltaYWxseSBNYXZlbiBwbHVnaW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9ldGhsby96YWxseS1tYXZlbi1wbHVnaW4pLgoKKipUaGlzIHByb2plY3QgaXMgdW5maW5pc2hlZCoqLiBJJ20gc2hhcmluZyB0aGUgd29yayBzbyBmYXIgdG8gZ2V0IHRoZSBiYWxsIHJvbGxpbmcuCgojIyMgU2V0dXAKCmBgYGtvdGxpbgovLyBidWlsZC5ncmFkbGUua3RzCgpidWlsZHNjcmlwdCB7CiAgcmVwb3NpdG9yaWVzIHsKICAgIG1hdmVuQ2VudHJhbCgpCiAgICBncmFkbGVQbHVnaW5Qb3J0YWwoKQogICAgbWF2ZW4oImh0dHBzOi8vaml0cGFjay5pbyIpCiAgfQp9CgpwbHVnaW5zIHsKLy8gIGlkKCJkZXYuYWRhbWtvLnphbGx5IikKICBpZCgiY29tLmdpdGh1Yi5hZGFta28tZGV2LnphbGx5LWdyYWRsZS1wbHVnaW4iKSB2ZXJzaW9uICJtYWluLVNOQVBTSE9UIgp9Cgp6YWxseSB7CiAgb3BlbkFwaVNwZWMuc2V0KAogICAgbGF5b3V0LnByb2plY3REaXJlY3RvcnkuZmlsZSgic3JjL21haW4vcmVzb3VyY2VzL29wZW5hcGkueW1sIikKICApCn0KYGBgCgpSdW46IGAuL2dyYWRsZXcgemFsbHlgCg== readmeEtag: '"3952c3f48a22e45f68833bf2dab336399ec00ac1"' readmeLastModified: Wed, 17 Aug 2022 11:25:51 GMT repositoryId: 520641991 description: Use Zally to validate OpenAPI specs in a Gradle project created: '2022-08-02T20:33:41Z' updated: '2023-12-11T16:25:27Z' language: Kotlin archived: false stars: 3 watchers: 1 forks: 0 owner: adamko-dev logo: https://avatars.githubusercontent.com/u/93070146?v=4 license: Apache-2.0 repoEtag: '"fa0018cbfd7b387c41e2a5ec06d0d7cf7f6fe591fa72683d1255086209e330c1"' repoLastModified: Mon, 11 Dec 2023 16:25:27 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/derberg/convert-swagger-to-openapi-playground v3: true repositoryMetadata: base64Readme: >- IyBDb252ZXJ0IFN3YWdnZXIgdG8gT3BlbkFQSSBQbGF5Z3JvdW5kCgpTdGF5aW5nIHdpdGggU3dhZ2dlciAyLjAgaXMgbGlrZSBzdGF5aW5nIHdpdGggLi4uIHN0aWxsIHdvcmtpbmcgb24gYSBnb29kIGNvbXBhcmlzb24uIEFueXdheSwgeW91IHNob3VsZCBtaWdyYXRlIHRvIE9wZW5BUEksIGFuZCBteSBnb2FsIGhlcmUgaXMgbm90IHRvIGNvbnZpbmNlIHlvdSB3aHkuIEkgYXNzdW1lIHlvdSBhbHJlYWR5IG1hZGUgdGhpcyBzbWFydCBkZWNpc2lvbiwgYW5kIEkganVzdCB3YW50IHRvIG1ha2UgaXQgZWFzaWVyIGZvciB5b3UuCgpJIGRpZCBub3QgcGVyZm9ybSBhbnkgbW9yZSBpbi1kZXB0aCBpbnZlc3RpZ2F0aW9uIG9mIGF2YWlsYWJsZSB0b29scy4gVGhlcmUgYXJlIFt0b29sc10oaHR0cHM6Ly9naXRodWIuY29tL01lcm1hZGUvb2FzLWtpdCkgcHJvdmlkZWQgYnkgTWVybWFkZSBTb2Z0d2FyZS4gVG8gYmUgbW9yZSBzcGVjaWZpYywgYnkgW01pa2UgUmFscGhzb25dKGh0dHBzOi8vdHdpdHRlci5jb20vUGVybWl0dGVkU29jKSB0aGF0IGlzIGEgbWVtYmVyIG9mIFtUZWNobmljYWwgU3RlZXJpbmcgQ29tbWl0dGVlXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci9NQUlOVEFJTkVSUy5tZCkuIEZvciBtZSwgdGhpcyBpcyBhIGdvb2QgcmVhc29uIG5vdCB0byByZXNlYXJjaCBkaWZmZXJlbnQgdG9vbHMuCgotIFtDb252ZXJ0IGluIGEgQnJvd3Nlcl0oI2NvbnZlcnQtaW4tYS1icm93c2VyKQotIFtDb252ZXJ0IGluIGEgVGVybWluYWxdKCNjb252ZXJ0LWluLWEtdGVybWluYWwpCiAgICAtIFtOUE1dKCNucG0pCiAgICAtIFtOUFhdKCNucHgpCiAgICAtIFtEb2NrZXJdKCNkb2NrZXIpCi0gW0NvbnZlcnQgTXVsdGlwbGUgRmlsZXNdKCNjb252ZXJ0LW11bHRpcGxlLWZpbGVzKQoKIyMgQ29udmVydCBpbiBhIEJyb3dzZXIKCkNvbnZlcnNpb24gaW4gYSBicm93c2VyIGlzIGFkZHJlc3NlZCBieSBbS2luIExhbmVdKGh0dHBzOi8vdHdpdHRlci5jb20va2lubGFuZSkgYWthIEFQSSBFdmFuZ2VsaXN0IGluIFt0aGlzIHRocmVhZCBhbmQgdGhlIHZpZGVvXShodHRwczovL3R3aXR0ZXIuY29tL2FwaWV2YW5nZWxpc3Qvc3RhdHVzLzEyOTU1MDIwOTM2MjU2NDMwMDgpIHNvIHdhdGNoIGl0IGFuZCBnbyB0byBodHRwczovL21lcm1hZGUub3JnLnVrL29wZW5hcGktY29udmVydGVyCgojIyBDb252ZXJ0IGluIGEgVGVybWluYWwKClVzZSBbc3dhZ2dlcjJvcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vTWVybWFkZS9vYXMta2l0L2Jsb2IvbWFzdGVyL3BhY2thZ2VzL3N3YWdnZXIyb3BlbmFwaS9SRUFETUUubWQpIGFuZCBoYXZlIGZ1bjoKCiMjIyBOUE0KCjEuIEluc3RhbGwgdGhlIHRvb2wgYG5wbSBpbnN0YWxsIC1nIHN3YWdnZXIyb3BlbmFwaWAKMS4gUnVuIGNvbnZlcnNpb24gYHN3YWdnZXIyb3BlbmFwaSAtLXlhbWwgLS1vdXRmaWxlIG9wZW5hcGkueWFtbCBodHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjIvc3dhZ2dlci5qc29uYAoKQW5kIHRoYXQgaXMgaXQuIENoZWNrIHlvdXIgbmV3IGBvcGVuYXBpLnlhbWxgIGZpbGUuCgojIyMgTlBYCgpbTlBYXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9ucHgpIGlzIHVzZWZ1bCBpbiBDSS9DRCB3aGVyZSB5b3UgZG8gbm90IHdhbnQgdG8gaW5zdGFsbCBgc3dhZ2dlcjJvcGVuYXBpYCBnbG9iYWxseS4KCmBucHggLXAgc3dhZ2dlcjJvcGVuYXBpIHN3YWdnZXIyb3BlbmFwaSAtLXlhbWwgLS1vdXRmaWxlIG9wZW5hcGkueWFtbCBodHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjIvc3dhZ2dlci5qc29uYAoKIyMjIERvY2tlcgoKTlBNIGFuZCBOUFggaXMgbm90IHlvdXIgdGhpbmd5PyB1c2UgdGhlIERvY2tlciBpbWFnZSBwcm92aWRlZCBieSBNaWtlLgoKYGBgYmFzaAojdGhpcyBwYXJ0ICItdiAke1BXRH06L3Vzci9zcmMvYXBwIiBtb3VudHMgdGhlIGRpcmVjdG9yeSB3aGVyZSB5b3Ugc3RhcnRlZCAiZG9ja2VyIHJ1biIgaW5zaWRlIHRoZSBjb250YWluZXIgd2hlcmUgQ0xJIGlzIHRyaWdnZXJlZCwgdGhpcyB3YXkgZ2VuZXJhdGVkICJvcGVuYXBpLnlhbWwiIGdldHMgaW50byB5b3VyIGxvY2FsIGRyaXZlCmRvY2tlciBydW4gLS1ybSAtdiAke1BXRH06L3Vzci9zcmMvYXBwIG1lcm1hZGUvc3dhZ2dlcjJvcGVuYXBpIHN3YWdnZXIyb3BlbmFwaSAtLXlhbWwgLS1vdXRmaWxlIG9wZW5hcGkueWFtbCBodHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjIvc3dhZ2dlci5qc29uCmBgYAoKIyMgQ29udmVydCBNdWx0aXBsZSBGaWxlcwoKWW91IG1vc3QgcHJvYmFibHkgaGF2ZSBtYW55IHNlcnZpY2VzLCBhbmQgeW91IG5lZWQgdG8gY29udmVydCBtYW55IFN3YWdnZXIgZmlsZXMsIGFuZCB5b3UgZG8gbm90IHdhbnQgdG8gZG8gaXQgb25lIGJ5IG9uZSBidXQgYWxsIGF0IG9uY2Ugd2l0aCBhIHNjcmlwdC4gWW91IGNhbiB1c2UgQmFzaCBhbmQgd3JpdGUgc29tZSBzY3JpcHQgdGhhdCBydW5zIHRoZSBDTEksIGJ1dCB3cml0aW5nIEJhc2ggc2NyaXB0cyBpcyBsaWtlIC4uLiB5ZWFoLCBvbmUgZGF5IEknbGwgZmluZCBhIGdvb2QgY29tcGFyaXNvbi4KCkp1c3QgdXNlIGBzd2FnZ2VyMm9wZW5hcGlgIGFzIGEgbGlicmFyeS4gSW4gYGNvbnZlcnRtdWx0aXBsZWAgZGlyZWN0b3J5LCB5b3UgY2FuIGZpbmQgYSBbc2FtcGxlIGNvZGVdKGNvbnZlcnRtdWx0aXBsZS9pbmRleC5qcykgdGhhdCB5b3UgY2FuIGVhc2lseSBlZGl0IGZvciB5b3VyIG5lZWRzLiBJdHMgY29kZSBoYXMgcHJvcGVyIGNvbW1lbnRzIHRoYXQgd2lsbCBoZWxwIHlvdSB1bmRlcnN0YW5kIHdoYXQgcGFydHMgeW91IG1pZ2h0IGNoYW5nZSBpbiBjYXNlIHlvdSBoYXZlIGRpZmZlcmVudCBkaXJlY3Rvcnkgc3RydWN0dXJlIG9yIGZpbGVuYW1lcyB0aGFuIHRoZSBvbmVzIHVzZWQgaW4gdGhlIHNhbXBsZQoKVG8gcGxheSB3aXRoIHRoZSBleGFtcGxlLCBwZXJmb3JtIHRoZSBmb2xsb3dpbmcgc3RlcHM6CjEuIGBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2RlcmJlcmcvY29udmVydC1zd2FnZ2VyLXRvLW9wZW5hcGktcGxheWdyb3VuZC5naXRgCjEuIGBjZCBjb252ZXJ0LXN3YWdnZXItdG8tb3BlbmFwaS1wbGF5Z3JvdW5kL2NvbnZlcnRtdWx0aXBsZWAKMS4gYG5wbSBpbnN0YWxsYAoxLiBgbnBtIHN0YXJ0YAo= readmeEtag: '"c09bc73a43355ecfcbe670a69e08691339aa6619"' readmeLastModified: Tue, 18 Aug 2020 13:01:09 GMT repositoryId: 288430830 description: >- Some instructions and sample how to convert Swagger 2.0 files to OpenAPI 3.0 created: '2020-08-18T10:58:26Z' updated: '2025-03-10T13:27:54Z' language: JavaScript archived: false stars: 4 watchers: 1 forks: 2 owner: derberg logo: https://avatars.githubusercontent.com/u/6995927?v=4 repoEtag: '"995591b4ebc0a54ffefc4f71c97b82bc33d9a08fb547b563e03b242775f32b58"' repoLastModified: Mon, 10 Mar 2025 13:27:54 GMT foundInMaster: true category: - Description Validators - Parsers id: c4af853e2cb7950af6ef30a892c513b0 - source: openapi3 tags repository: https://github.com/joar/py-openapilib v3: true repositoryMetadata: base64Readme: >- Li4gdmltOnRhYnN0b3A9NDpzaGlmdHdpZHRoPTQ6c29mdHRhYnN0b3A9NDoKCi4uIHxuYW1lfCByZXBsYWNlOjogYGBvcGVuYXBpbGliYGAKLi4gfGRlc2NyaXB0aW9ufCByZXBsYWNlOjogT3BlbkFQSSAzIE9iamVjdCBNb2RlbAoKIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMKICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8bmFtZXwgLSB8ZGVzY3JpcHRpb258CiMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjIyMjCgo9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQpTdGF0dXMKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCi0gICB8bmFtZXwgaXMgaW4gZGV2ZWxvcG1lbnQuCi0gICB8bmFtZXwgaW1wbGVtZW50cyB0aGUgbWFqb3JpdHkgb2YgdGhlIGZpZWxkcy4KCkNvbXBhdGliaWxpdHkKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCi0gICB8bmFtZXwgYWltcyB0byBpbXBsZW1lbnQgdGhlIGBPcGVuQVBJIDMuMC4wIFNwZWNpZmljYXRpb25gXy4KLSAgIHxuYW1lfCBpcyBjb21wYXRpYmxlIHdpdGggUHl0aG9uID49IDMuNiwgZHVlIHRvIGl0J3MgdXNhZ2Ugb2YgZi1zdHJpbmdzLgoKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KUmVmZXJlbmNlcwo9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQoKLSAgIGBPcGVuQVBJIDMuMC4wIFNwZWNpZmljYXRpb24gPGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kPmBfCg== readmeEtag: '"241e21c7f9feaca803089c83f1a504f389cb96e4"' readmeLastModified: Mon, 26 Mar 2018 15:02:44 GMT repositoryId: 107727934 description: Python 3.6 OpenAPI 3 Object Model - built with attrs created: '2017-10-20T21:24:50Z' updated: '2024-05-21T10:16:12Z' language: Python archived: false stars: 3 watchers: 1 forks: 2 owner: joar logo: https://avatars.githubusercontent.com/u/20814?v=4 license: BSD-2-Clause repoEtag: '"fa4be6fae6986251a2888b92201fd56dd4b1e33cba0cad46db99dcb2a0c3822e"' repoLastModified: Tue, 21 May 2024 10:16:12 GMT foundInMaster: true category: Parsers id: 62c05e9899c36eb5cb5f25061e0fce48 - source: openapi3 tags repository: https://github.com/itzlyg/iot v3: true id: 246f8e81b414ff1c8017ebcd15da30ac repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9TcHJpbmclMjBCb290LTIuNy4xOC1ibHVlLnN2ZyIgYWx0PSJEb3dubG9hZHMiPgogPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvVnVlLTIuNi4xNC1ibHVlLnN2ZyIgYWx0PSJEb3dubG9hZHMiPgo8L3A+CgojIyAg566A5LuLCuacrOmhueebruaYr+S4gOS4quWfuuS6jmBKYXZhYCBgVnVlYOW8gOWPkeeahOeJqeiBlOe9keWfuuehgOW5s+WPsO+8jOWPr+S7peW/q+mAn+WunueOsOehrOS7tuiuvuWkh+aOpeWFpeOAggoKKiBKYXZhIOWQjuerr++8mmBtYXN0ZXJgIOWIhuaUr+S4uiBKREsgOCArIFNwcmluZyBCb290IDIuNy4xOAoqIOeuoeeQhuWQjuWPsOeahOeUteiEkeerr++8mlZ1ZTIg5o+Q5L6bIGBlbGVtZW50LXVpYCDniYjmnKwKKiDlkI7nq6/ph4fnlKggU3ByaW5nIEJvb3Qg5aSa5qih5Z2X5p625p6E44CBTXlTUUwgKyBNeUJhdGlzIFBsdXPjgIFSZWRpcyArIFJlZGlzc29u77yMVmVydC54Ciog5pSv5oyB5aSa56ef5oi35qih5byPCiog57uf5LiA5o6l5Y+j5o6l5pS25ZKM6L+U5Zue5qC85byP77yM5L2/55SoT3BlbkFwaSAzK2tuaWZlNGrkvZzkuLrmjqXlj6PmlofmoaPlt6XlhbcKKiDml7bluo/mlbDmja7lupPkvb/nlKhURGVuZ2luZe+8jOWPr+WPpuihjOaJqeWxlQoqIOa2iOaBr+mYn+WIl+W3sue7j+WunueOsFJvY2tldE1R5ZKM5Z+65LqOdmVydHjnmoRldmVudGJ1c++8jOWPr+S7peiHquihjOagueaNrumcgOimgeaJqeWxleS9v+eUqOWFtuS7luS4remXtOS7tu+8mkthZmth44CBUmFiYml0TVHnrYkKKiDlrp7ml7bpgJrkv6Hkvb/nlKh3ZWJzb2NrZXTvvIzln7rkuo52ZXJ0LnjmoYbmnrblrp7njrAKKiDpm4bmiJDkuIPniZvkupHkupHlrZjlgqjmnI3liqHvvIzpooTnlZnlhbbku5bmnI3liqHllYbmjqXlj6PvvIzoh6rooYzlrp7njrDmianlsZUKCiMjIyMg5Lqn5ZOB54m55oCnCgotIOiuvuWkh+aOpeWFpe+8muaUr+aMgeaOpeWFpeihjOS4muagh+WHhueJqeiBlOe9keWNj+iuru+8iG1xdHTjgIFodHRw44CBbW9kYnVz562J77yJ77yM5a6e546w5rW36YeP6K6+5aSH5o6l5YWl77yM5Lmf5Y+v5Lul6YCa6L+H6Ieq5a6a5LmJ55qE5pa55byP5o6l5YWl6Ieq5a6a5LmJ5Y2P6K6u44CCCi0g54mp5qih5Z6L77ya5pSv5oyB54mp5qih5Z6L77yM5Y+v5Lul5pyJ5pWI55qE566h55CG6K6+5aSH55qE5bGe5oCn44CB5Yqf6IO944CB5LqL5Lu244CCCi0g5Lqn5ZOB44CB6K6+5aSH5YiG57uE5qaC5b+177ya5pSv5oyB6K6+5aSH5qC55o2u5bqU55So5Zy65pmv5ZKM55So5oi35p2D6ZmQ5a6e546w6ZqU56a777yM5pa55L6/55So5oi357uf5LiA566h55CG44CCCi0g5p2D6ZmQ566h55CG77ya57O757uf5pSv5oyB5Z+65pys55qE6KeS6Imy566h55CG77yM5pSv5oyB55So5oi344CB6KeS6Imy44CB6I+c5Y2V44CB5oyJ6ZKu5Z+65pys5p2D6ZmQ5Yqf6IO944CCCi0g6Ieq5Li75Y+v5o6n77ya5Y+v5pys5Zyw44CB56eB5pyJ5LqR44CB5YWs5pyJ5LqR6YOo572y44CCCi0g5pe25bqP5pWw5o2u5bqT77ya5pWw5o2u5LiL5Y+R44CB5LiK5oql5pWw5o2u6YeH55So5pWw5bqP5pWw5o2u5bqT77yM5pSv5oyB5rW36YeP5bm25Y+R44CCCi0g5bmz5Y+w5Z+65bqn77ya5Y+v5L2c5Li654mp6IGU572R6KGM5Lia6Kej5Yaz5pa55qGI5bmz5Y+w5Z+65bqn77yM5pSv5oyB5aSa5Liq6KGM5Lia6Kej5Yaz5pa55qGI5YWx55So5LiA5Liq54mp6IGU572R5bmz5Y+w5Z+65bqn77yM5Li65LyB5Lia5YeP5bCR5oiQ5pys77yM6I635Y+W5pu05aSa55qE6KGM5Lia57uP6aqM5ZKM5pa55qGI44CCCgojIyMjIOWFtuS7lgrikaAg5Luj56CB5pW05rSB44CB5p625p6E5pW05rSB77yM6YG15b6q44CK6Zi/6YeM5be05be0IEphdmEg5byA5Y+R5omL5YaM44CL6KeE6IyD44CCCgrikaEg5bel5YW357G75Y+q5L6d6LWWQXBhY2hl55u45YWz55qE77yM5oiW566A5Y2V6Ieq6KGM57yW5YaZ77yM5YeP5bCR56ys5LiJ5pa55L6d6LWW77yM6YG/5YWN6auY5Y2x5ryP5rSe44CCCgrikaIg57uf5LiA5o6l5Y+j5o6l5pS25ZKM6L+U5Zue5qC85byPCgoKIyMjIyDnianogZTnvZHpobnnm64g5qih5Z2X57uE5oiQCgojIyMjIyBwcm90b2NvbCDljY/orq7lsYLku6PnoIEKLSAx44CBYmFzZSDljY/orq7ln7rnoYDmir3lj5YKLSAy44CBc2VydmljZSDlrp7njrDljY/orq7nu4Tku7bvvIzlrp7njrDmlbTkuKrpgJrorq/pgJrpgZPjgIHnvJbop6PnoIHvvIzlrozmiJDmlbDmja7nmoTmtYHovazjgILmj5Dkvpvmlrnms5Xnu5nkuJrliqHlsYLosIPnlKgKLSAz44CBdGNw44CBaHR0cOOAgXdz44CBbXF0dOaYr+WFt+S9k+WNj+iurueahOWunueOsOWMhQoKIyMjIyMgbW9kdWxlIOaKveixoeWHuuWPr+S7peWkmuWunueOsOaWueW8j+eahOe7hOS7tgotIDHjgIFvc3Mg5paH5Lu25a2Y5YKoCi0gMuOAgW5vdGlmeSDlkYrorabpgJrnn6UKLSAz44CBc2luayDnianmqKHlnovmlbDmja7okL3lnLDliLDmlbDmja7lupMKCiMjIyMjIGFjaGlldmUg55So5p2l5a6e546wIG1vZHVsZee7hOS7tueahAotIDHjgIFtcS12ZXJ0eCDnlKh2ZXJ0Lngg5p2l5a6e546w5raI5oGv6Zif5YiXCi0gMuOAgXJvY2tldC1tcSByb2NrZXRNceeahOWunueOsAotIDPjgIFvc3MtcW4g5LiD54mb5a6e546w5a2Y5YKo5pa55qGICi0gNOOAgXNpbmstdGQgVERlbmdpbmUg5a+56K6+5aSH5LiK5oql5LiL5Y+R5pWw5o2u6K6w5b2V44CB6K6+5aSH55qE5bGe5oCn562J5pWw5o2u6YeP5b6I5aSn55qE5Zy65pmvCi0gNeOAgXNpbmstbXlzcWwgbXlzcWwg55qE5pa55byP5a6e546w5pWw5o2u6JC95ZywCgojIyMjIyBiaXog5Lia5Yqh5bGCCi0gMeOAgWJhc2Ug5bel5YW357G777yb5Z+656GA5p6a5Li+5YC844CBdm/nmoTlrprkuYnvvJvns7vnu5/phY3nva4KLSAy44CBc2VydmljZe+8jOWFt+S9k+eahOS4muWKoe+8jOaKveemu+WkhOeQhu+8jOWPr+S7peS+m+Wkmuerr+iwg+eUqOWunueOsO+8mndlYuOAgW9wZW5BcGnnrYkKLSAz44CBb3Blbi1hcGkg5a+556ys5LiJ5pa55a+55o6l5o+Q5L6b5o6l5Y+jCi0gM+OAgXRlc3Qg5rWL6K+V5Luj56CBCi0gNOOAgXdlYiB3ZWLnq6/nmoTmjqXlj6MKCiMjIyMg5Yqf6IO954K577yaCi0gMeOAgeaUr+aMgW1xdHTjgIFodHRw44CBdGNw44CBd2Vic29ja2V0562J5Y2P6K6u6YCa6YGTCi0gMuOAgeWunueOsOeJqeiBlOe9keiuvuWkh+WfuuacrOWNj+iuru+8jOWunueOsOaWsOiuvuWkh+aOpeWFpeaXtuWAme+8jOWPr+S7peWIqeeUqOaPkuaLlOW8j+eahOaWueW8j+W/q+mAn+aOpeWFpe+8jOaWsOiuvuWkh+ino+aekOaWueW8j+WPr+S7peeUqGBqc2Ag5oiW6ICFYHB5dGhvbmDnmoTmlrnlvI/lv6vpgJ/lrp7njrDvvIzlh4/lsJHlvIDlj5HmiJDmnKwKLSAz44CB5a6e546w6K6+5aSH5bGe5oCn5ZKM5Yqf6IO955qE6Ieq5a6a5LmJ77yM5b+r6YCf5a6e546w6K6+5aSH55qE5Yqf6IO9Ci0gNOOAgXdlYnNvY2tldOaOqOmAgeiuvuWkh+WunuaXtua2iOaBr+WIsOmhtemdogotIDXjgIHorr7lpIflkYrorabns7vnu58KLSA244CB6K6w5b2V5pON5L2c5pel5b+X5ZKM5pWw5o2u5byC5Yqo5pel5b+XCi0gN+OAgeeUqOaIt+adg+mZkOetieWfuuehgOWKn+iDvQotIDjjgIFvcGVuIGFwaeW8gOaUvgoKIyMjIyDkuqflk4Hlip/og73vvJoKfCDlip/og70gICAgIHwg5o+P6L+wICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwKfCDpppbpobUgICAgIHwg5aSn5bGP55yL5p2/5bGV56S677yM6K6+5aSH6L+Q6KGM54q25oCB44CB5pWw6YeP44CB5ZGK6K2m562J5Z+65pys5L+h5oGvIHwKfCDnu4Tnu4fnrqHnkIYgfCDnlKjmiLfnrqHnkIbjgIHpg6jpl6jnrqHnkIbjgIHop5LoibLnrqHnkIbjgIHoj5zljZXnrqHnkIYgICAgICAgICAgIHwKfCDpgJrkv6HljY/orq4gfCDmtojmga/ovazljJblmajjgIHpgJrkv6HljY/orq7nrqHnkIYgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IOS6p+WTgeeuoeeQhiB8IOS6p+WTgeeuoeeQhuOAgeS6p+WTgeWIhuexuyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8Cnwg6K6+5aSH566h55CGIHwg6K6+5aSH5YiG57uE44CB6K6+5aSH566h55CG44CB6K6+5aSH6K+m5oOFICAgICAgICAgICAgICAgICAgICAgfAp8IOWcuuaZr+iBlOWKqCB8IOWcuuaZr+iBlOWKqOeuoeeQhiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8Cnwg5ZGK6K2m6YCa55+lIHwg5ZGK6K2m5qih5p2/44CB5ZGK6K2m5rig6YGT44CB5raI5oGv566h55CGICAgICAgICAgICAgICAgICAgICAgfAoKCgoKIyMjIyDlhbbku5YKW+a8lOekuuWcsOWdgF0oaHR0cDovL3d3dy5qaXViYW5rZWppLmNuL3dlYl9pb3QpCgrluJDlj7cv5a+G56CB77yaCmAxODgxMjM0NTY3OGAgYElvQHQwMzE1SmIhYAoKCjxpbWcgYWx0PSJEb3dubG9hZHMiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9waG9uZS0xMzQyODY5OTYyOS1ncmVlbi5zdmciPgoKKirlpoLmnInnlpHpl67miJbogIXpnIDopoHmupDnoIHnmoTlj6/ku6Xpmo/ml7bogZTns7vvvJoqKgoKPGltZyBzcmM9Ii4vaW90LWFwL2RvYy9qYl9xcmNvZGUuanBnIiB3aWR0aD0iMTUwIiBoZWlnaHQ9IjE1MCIvPgoKKOaJq+eggeaIluiAheebtOaOpeaQnOe0omBqaXViYW55dW5faW90YOa3u+WKoOW+ruS/oSkK readmeEtag: '"88ca607c4250c328a50c78db77b1916ce14e9e40"' readmeLastModified: Sun, 01 Dec 2024 14:28:11 GMT repositoryId: 771891137 description: 物联网平台 物联网应用 物联网 智慧平台 created: '2024-03-14T06:23:53Z' updated: '2026-01-13T12:50:37Z' language: Java archived: false stars: 6 watchers: 1 forks: 1 owner: itzlyg logo: https://avatars.githubusercontent.com/u/17636877?v=4 license: GPL-3.0 repoEtag: '"02f733d2bb140ee5b7d262feecb9e894260849e0d80255bf36cb78fe13d50788"' repoLastModified: Tue, 13 Jan 2026 12:50:37 GMT category: Code Generators foundInMaster: true - source: openapi3 tags repository: https://github.com/denismakogon/fn-openapi-v3 v3: true repositoryMetadata: base64Readme: >- T3BlbkFQSSB2MyBzcGVjIGdlbmVyYXRvciB0b29sIGZvciBGbiBhcHBsaWNhdGlvbnMKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KCklkZWEKLS0tLQpUaGUgRm4gYXBwbGljYXRpb24gaXMgbm90aGluZyBidXQgc2V0IG9mIGZ1bmN0aW9uIHdoZXJlIGVhY2ggaGFzIGl0cyBvd24gSFRUUCByb3V0ZSBmb3IgZXhlY3V0aW9uLgoKV2hhdCBpZiB0aGVyZSdzIGEgd2F5IHRvIGJ1aWxkIGNsaWVudCBiaW5kaW5nIGZvciB0aGUgcGFydGljdWxhciBzZXJ2ZXJsZXNzIGFwcGxpY2F0aW9uPwoKVGhpcyBsaWJyYXJ5IGFuZCB0b29sIGFyZSBkZXNpZ25lZCB0byBwcm92aWRlIG5lY2Vzc2FyeSBBUEkgdG8gZ2VuZXJhdGUgT3BlbkFQSSB2My4wLjAgc3BlY2lmaWNhdGlvbiB1c2luZyBGdW5jdGlvbiBzcGVjIGxhbmd1YWdlLgoKCkZ1bmN0aW9uIHNwZWMgbGFuZ3VhZ2UKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKVGhpcyB0b29sIHJlbGllcyBvbiBpbXByb3ZlZCBTd2FnZ2VyIEFQSSAyLjAgcGx1cyBhZGRpdGlvbmFsIGlubGluZSByZWZlcmVuY2luZyBmZWF0dXJlcyAKdGhhdCBhcmUgbWlzc2luZyBpbiBib3RoIFN3YWdnZXIgMi4wIGFuZCBPcGVuQVBJIDMuMCB0aGF0IGFyZSBhbGxvd2luZyBkZXZlbG9wZXJzIHRvIHN0cnVjdHVyZSB0aGVpciBhcHBsaWNhdGlvbiBpbiBtb3JlIG1vZHVsYXIgd2F5LgoKCkZ1bmN0aW9uIHNwZWMgZXhhbXBsZQotLS0tLS0tLS0tLS0tLS0tLS0tLS0KCmBgYHlhbWwKdmVyc2lvbjogMC4wLjEKZGVzY3JpcHRpb246IEZ1bmN0aW9ucyBzcGVjIHRoYXQgZGVzY3JpYmVzIEZuLXBvd2VyZWQgc2VydmVybGVzcyBhcHBsaWNhdGlvbgpmdW5jdGlvbnM6CiAgY3JlYXRlVXNlcjoKICAgIGhhbmRsZXI6IGhhbmRsZXIuY3JlYXRlCiAgICBldmVudHM6CiAgICAgIC0gaHR0cDoKICAgICAgICAgIG1ldGhvZDogcG9zdAogICAgICAgICAgZm46ICR7ZmlsZShtb2RlbHMvZnVuYy55bWwpOmZpcnN0fQogICAgICAgICAgZG9jdW1lbnRhdGlvbjoKICAgICAgICAgICAgc3VtbWFyeTogQ3JlYXRlIFVzZXIKICAgICAgICAgICAgZGVzY3JpcHRpb246IENyZWF0ZXMgYSB1c2VyIGFuZCB0aGVuIHNlbmRzIGEgZ2VuZXJhdGVkIHBhc3N3b3JkIGVtYWlsCiAgICAgICAgICAgIHJlcXVlc3RCb2R5OgogICAgICAgICAgICAgIHNjaGVtYTogJHtmaWxlKG1vZGVscy9yZXF1ZXN0Lmpzb24pfQogICAgICAgICAgICBwYXJhbWV0ZXJzOgogICAgICAgICAgICAgIC0gbmFtZTogdXNlcm5hbWUKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uOiBUaGUgdXNlcm5hbWUgZm9yIGEgdXNlciB0byBjcmVhdGUKICAgICAgICAgICAgICAgIHJlcXVpcmVkOiB0cnVlCiAgICAgICAgICAgICAgICBpbjogcGF0aAogICAgICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgICAgICAgICAgcGF0dGVybjogIl5bLWEtejAtOV9dKyQiCiAgICAgICAgICAgICAgLSBuYW1lOiBtZW1iZXJzaGlwVHlwZQogICAgICAgICAgICAgICAgZGVzY3JpcHRpb246IFRoZSB1c2VyJ3MgTWVtYmVyc2hpcCBUeXBlCiAgICAgICAgICAgICAgICByZXF1aXJlZDogdHJ1ZQogICAgICAgICAgICAgICAgaW46IHF1ZXJ5CiAgICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgICAgICAgICBlbnVtOgogICAgICAgICAgICAgICAgICAgIC0gcHJlbWl1bQogICAgICAgICAgICAgICAgICAgIC0gc3RhbmRhcmQKICAgICAgICAgICAgcmVzcG9uc2VzOgogICAgICAgICAgICAgIDIwMDoKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uOiBjcmVhdGUgYSB1c2VyCiAgICAgICAgICAgICAgICBjb250ZW50OgogICAgICAgICAgICAgICAgICBhcHBsaWNhdGlvbi9qc29uOgogICAgICAgICAgICAgICAgICAgIHNjaGVtYTogJHtmaWxlKG1vZGVscy9yZXF1ZXN0Lmpzb24pfQogICAgICAgICAgICAgIDUwMDoKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uOiBlcnJvcgogICAgICAgICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgICAgICAgICBzY2hlbWE6ICR7ZmlsZShtb2RlbHMvZXJyb3IuanNvbil9CmBgYApUaGlzIHNhbXBsZSB5b3UgY2FuIGZpbmQgW2hlcmVdKGV4YW1wbGVzL2ZuLnltbCkKCnVzaW5nIHRoZSBmb2xsb3dpbmcgY29kZToKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgKAoJImZtdCIKCSJnaXRodWIuY29tL2RlbmlzbWFrb2dvbi9mbi1vcGVuYXBpL21vZGVscyIKCSJpby9pb3V0aWwiCgkib3MiCikKCmZ1bmMgbWFpbigpIHsKCgl5YW1sRmlsZSwgZXJyIDo9IGlvdXRpbC5SZWFkRmlsZShvcy5BcmdzWzFdKQoJaWYgZXJyICE9IG5pbCB7CgkJZm10LlByaW50bG4oZXJyLkVycm9yKCkpCgkJb3MuRXhpdCgxKQoJfQoKCXZhciBmbiBtb2RlbHMuRm4KCgllcnIgPSBmbi5Vbm1hcnNoYWwoeWFtbEZpbGUsIG9zLlN0ZG91dCkKCWlmIGVyciAhPSBuaWwgewoJCWZtdC5QcmludGxuKGVyci5FcnJvcigpKQoJCW9zLkV4aXQoMSkKCX0KCgl2YXIgb2FpIG1vZGVscy5PcGVuQVBJU3BlYwoJZXJyID0gb2FpLkZyb21GblNwZWMoImh0dHA6Ly9sb2NhbGhvc3Q6ODA4MCIsICZmbikKCWVyciA9IG9haS5NYXJzaGFsKG9zLlN0ZG91dCkKCWlmIGVyciAhPSBuaWwgewoJCWZtdC5QcmludGxuKGVyci5FcnJvcigpKQoJCW9zLkV4aXQoMSkKCX0KfQpgYGAKCml0IGlzIHBvc3NpYmxlIHRvIHR1cm4gRnVuY3Rpb24gc3BlYyBpbnRvIHZhbGlkIE9wZW5BUEkgdjMgc3BlY2lmaWNhdGlvbi4KVG8gY29uZmlybSB0aGF0IHNwZWMgaXMgdmFsaWQgdXNlIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKYGBgYmFzaApkb2NrZXIgcnVuIC0tcm0gLWkgLXYgYHB3ZGA6L2dvIGZucHJvamVjdC9vcGVuYXBpdjMtdmFsaWRhdG9yOjAuMC4xIC9nby9leGFtcGxlcy9vcGVuYXBpLnltbApgYGAK readmeEtag: '"69529cf07d34bc2737095a9d5caf358a480f1cd3"' readmeLastModified: Wed, 17 Jan 2018 18:35:01 GMT repositoryId: 117738868 description: Fn-powered serverless application OpenAPI v3.0.0 generator tool created: '2018-01-16T20:38:36Z' updated: '2019-06-11T20:40:03Z' language: Go archived: false stars: 3 watchers: 2 forks: 0 owner: denismakogon logo: https://avatars.githubusercontent.com/u/3034091?v=4 license: Apache-2.0 repoEtag: '"c56c5e17e43230736e2ae7a5c7b42a85857ef516780b51e4fa432668ac9035b8"' repoLastModified: Tue, 11 Jun 2019 20:40:03 GMT foundInMaster: true category: Parsers id: 61087310dc28f1cc6a4eb90cfa714b60 - source: openapi3 tags repository: https://github.com/confuser/exegesis-koa v3: true repositoryMetadata: base64Readme: >- IyBleGVnZXNpcy1rb2EKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9hcGkudHJhdmlzLWNpLm9yZy9jb25mdXNlci9leGVnZXNpcy1rb2Euc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvY29uZnVzZXIvZXhlZ2VzaXMta29hKQpbIVtDb3ZlcmFnZSBTdGF0dXNdKGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9jb25mdXNlci9leGVnZXNpcy1rb2EvYmFkZ2Uuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL2NvdmVyYWxscy5pby9naXRodWIvY29uZnVzZXIvZXhlZ2VzaXMta29hP2JyYW5jaD1tYXN0ZXIpClshW0tub3duIFZ1bG5lcmFiaWxpdGllc10oaHR0cHM6Ly9zbnlrLmlvL3Rlc3QvZ2l0aHViL2NvbmZ1c2VyL2V4ZWdlc2lzLWtvYS9iYWRnZS5zdmc/dGFyZ2V0RmlsZT1wYWNrYWdlLmpzb24pXShodHRwczovL3NueWsuaW8vdGVzdC9naXRodWIvY29uZnVzZXIvZXhlZ2VzaXMta29hP3RhcmdldEZpbGU9cGFja2FnZS5qc29uKQoKPiAjIyAqZXhlZ2VzaXMqCj4KPiAqbi4qIEFuIGV4cGxhbmF0aW9uIG9yIGNyaXRpY2FsIGludGVycHJldGF0aW9uIG9mIGEgdGV4dCwgZXNwZWNpYWxseSBhbgo+IEFQSSBkZWZpbml0aW9uIGRvY3VtZW50Lgo+Cj4gLS0gTm8gZGljdGlvbmFyeSBldmVyCgpUaGlzIGxpYnJhcnkgaW1wbGVtZW50cyBhIEtvYSBtaWRkbGV3YXJlIGZvcgpbT3BlbkFQSSAzLnhdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4xLm1kI3JlcXVlc3RCb2R5T2JqZWN0KS4KCiMjCmBgYApucG0gaW5zdGFsbCBleGVnZXNpcy1rb2EKYGBgCgojIyBUdXRvcmlhbAoKQ2hlY2sgb3V0IHRoZSB0dXRvcmlhbCBbaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL2V4ZWdlc2lzLWpzL2V4ZWdlc2lzL2Jsb2IvbWFzdGVyL2RvY3MvVHV0b3JpYWwubWQpLgoKIyMgVXNhZ2UKYGBganMKY29uc3QgS29hID0gcmVxdWlyZSgna29hJykKY29uc3QgcGF0aCA9IHJlcXVpcmUoJ3BhdGgnKQpjb25zdCBleGVnZXNpc0tvYSA9IHJlcXVpcmUoJ2V4ZWdlc2lzLWtvYScpCgphc3luYyBmdW5jdGlvbiBjcmVhdGVTZXJ2ZXIoKSB7CiAgICAvLyBTZWUgaHR0cHM6Ly9naXRodWIuY29tL2V4ZWdlc2lzLWpzL2V4ZWdlc2lzL2Jsb2IvbWFzdGVyL2RvY3MvT3B0aW9ucy5tZAogICAgY29uc3Qgb3B0aW9ucyA9IHsKICAgICAgICBjb250cm9sbGVyczogcGF0aC5yZXNvbHZlKF9fZGlybmFtZSwgJy4vY29udHJvbGxlcnMnKQogICAgfQogICAgY29uc3QgZXhlZ2VzaXNNaWRkbGV3YXJlID0gZXhlZ2VzaXNLb2EocGF0aC5yZXNvbHZlKF9fZGlybmFtZSwgJy4vb3BlbmFwaS55YW1sJyksIG9wdGlvbnMpCgogICAgY29uc3QgYXBwID0gbmV3IEtvYSgpCgogICAgLy8gSWYgeW91IGhhdmUgYW55IGJvZHkgcGFyc2VycywgdGhpcyBzaG91bGQgZ28gYmVmb3JlIHRoZW0uCiAgICBhcHAudXNlKGFzeW5jIChjdHgsIG5leHQpID0+IHsKICAgICAgdHJ5IHsKICAgICAgICBhd2FpdCBuZXh0KCkKICAgICAgfSBjYXRjaCAoZXJyKSB7CiAgICAgICAgY3R4LnN0YXR1cyA9IDUwMAogICAgICAgIGN0eC5ib2R5ID0gYEludGVybmFsIGVycm9yOiAke2Vyci5tZXNzYWdlfWAKICAgICAgfQogICAgfSkKICAgIGFwcC51c2UoZXhlZ2VzaXNNaWRkbGV3YXJlKQogICAgYXBwLnVzZShhc3luYyAoY3R4KSA9PiB7CiAgICAgIGlmIChjdHguc3RhdHVzID09PSA0MDQpIHsKICAgICAgICBjdHguc3RhdHVzID0gNDA0CiAgICAgIH0KICAgIH0pCgogICAgYXBwLmxpc3RlbigpCn0KYGBgCgpDYWxsaW5nIGBleGVnZXNpc2tvYShvcGVuQXBpRmlsZSwgb3B0aW9ucylgIHdpbGwgcmV0dXJuIGEgUHJvbWlzZQp3aGljaCByZXNvbHZlcyB0byBhIGtvYSBtaWRkbGV3YXJlLgoKYG9wZW5BcGlGaWxlYCBpcyBlaXRoZXIgYSBwYXRoIHRvIHlvdXIgb3BlbmFwaS55YW1sIG9yIG9wZW5hcGkuanNvbiBmaWxlLApvciBpdCBjYW4gYmUgYSBKU09OIG9iamVjdCB3aXRoIHRoZSBjb250ZW50cyBvZiB5b3VyIE9wZW5BUEkgZG9jdW1lbnQuICBUaGlzCnNob3VsZCBoYXZlIHRoZSBbYHgtZXhlZ2VzaXMtY29udHJvbGxlcmBdKGh0dHBzOi8vZ2l0aHViLmNvbS9leGVnZXNpcy1qcy9leGVnZXNpcy9ibG9iL21hc3Rlci9kb2NzL09BUzMlMjBTcGVjaWZpY2F0aW9uJTIwRXh0ZW5zaW9ucy5tZCkKZXh0ZW5zaW9uIGRlZmluZWQgb24gYW55IHBhdGhzIHlvdSB3YW50IHRvIGJlIGFibGUgdG8gYWNjZXNzLgoKYG9wdGlvbnNgIGNhbiBiZSBbYW55dGhpbmcgeW91IGNhbiBwYXNzIHRvIGV4ZWdlc2lzXShodHRwczovL2dpdGh1Yi5jb20vZXhlZ2VzaXMtanMvZXhlZ2VzaXMvYmxvYi9tYXN0ZXIvZG9jcy9PcHRpb25zLm1kKS4gIEF0IGEKbWluaW11bSwgeW91J2xsIHByb2JhYmx5IHdhbnQgdG8gcHJvdmlkZSBgb3B0aW9ucy5jb250cm9sbGVyc2AsIGEgcGF0aCB0byB3aGVyZQp5b3VyIFtjb250cm9sbGVyIG1vZHVsZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9leGVnZXNpcy1qcy9leGVnZXNpcy9ibG9iL21hc3Rlci9kb2NzL0V4ZWdlc2lzJTIwQ29udHJvbGxlcnMubWQpCmNhbiBiZSBmb3VuZC4gIElmIHlvdSBoYXZlIGFueSBzZWN1cml0eSByZXF1aXJlbWVudHMgZGVmaW5lZCwgeW91J2xsIGFsc28Kd2FudCB0byBwYXNzIGluIHNvbWUgW2F1dGhlbnRpY2F0b3JzXShodHRwczovL2dpdGh1Yi5jb20vZXhlZ2VzaXMtanMvZXhlZ2VzaXMvYmxvYi9tYXN0ZXIvZG9jcy9PQVMzJTIwU2VjdXJpdHkubWQpLgpUbyBlbmFibGUgcmVzcG9uc2UgdmFsaWRhdGlvbiwgeW91J2xsIHdhbnQgdG8gcHJvdmlkZSBhIHZhbGlkYXRpb24gY2FsbGJhY2sKZnVuY3Rpb24gdmlhIFtgb25SZXNwb25zZVZhbGlkYXRpb25FcnJvcigpYF0oaHR0cHM6Ly9naXRodWIuY29tL2V4ZWdlc2lzLWpzL2V4ZWdlc2lzL2Jsb2IvbWFzdGVyL2RvY3MvT3B0aW9ucy5tZCNvbnJlc3BvbnNldmFsaWRhdGlvbmVycm9yKS4KRXhlZ2VzaXMncyBmdW5jdGlvbmFsaXR5IGNhbiBhbHNvIGJlIGV4dGVuZGVkIHVzaW5nIFtwbHVnaW5zXShodHRwczovL2dpdGh1Yi5jb20vZXhlZ2VzaXMtanMvZXhlZ2VzaXMvdHJlZS9tYXN0ZXIvZG9jcyksCndoaWNoIHJ1biBvbiBldmVyeSByZXF1ZXN0LiAgUGx1Z2lucyBsZXQgeW91IGFkZCBmdW5jdGlvbmFsaXR5IGxpa2UKW3JvbGUgYmFzZSBhdXRob3JpemF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vZXhlZ2VzaXMtanMvZXhlZ2VzaXMtcGx1Z2luLXJvbGVzKSwKb3IgQ09SUy4K readmeEtag: '"b3b57b5f9c525f047e01a4d61359b7820a7804ae"' readmeLastModified: Sun, 03 Jun 2018 12:00:14 GMT repositoryId: 135899462 description: 'Koa middleware to handle OpenAPI 3.x. ' created: '2018-06-03T11:48:14Z' updated: '2022-09-23T15:25:12Z' language: JavaScript archived: false stars: 3 watchers: 1 forks: 1 owner: confuser logo: https://avatars.githubusercontent.com/u/1658997?v=4 license: MIT repoEtag: '"cb11ea2853c22d46f87b3b718e5c85459f9cc6fcf59adc5073acd16d13e1019f"' repoLastModified: Fri, 23 Sep 2022 15:25:12 GMT foundInMaster: true category: Server Implementations id: 75c502e1a557c910665da48dc9eec0eb - source: openapi3 tags repository: https://github.com/afshinparhizkari/cloud v3: true repositoryMetadata: base64Readme: IyBDbG91ZApEREQgUHJvamVjdCB3aXRoIERldm9wcyB0b29scwo= readmeEtag: '"7aa4a270fdeb7f939a62d5a08b03b4c2907eb9cc"' readmeLastModified: Sat, 30 Sep 2023 12:01:24 GMT repositoryId: 427974337 description: the simple DDD Project with Spring Cloud library created: '2021-11-14T16:02:42Z' updated: '2022-11-03T07:13:50Z' language: Java archived: false stars: 3 watchers: 1 forks: 1 owner: AfshinParhizkari logo: https://avatars.githubusercontent.com/u/32636285?v=4 license: LGPL-2.1 repoEtag: '"77a8fd98b0f57bd154859b0b0b350f6da7f7ea2bb5c6d72b173fb6d1464963e8"' repoLastModified: Thu, 03 Nov 2022 07:13:50 GMT foundInMaster: true category: - Server - Server Implementations id: 5046b14ee2dc167c4183150526d560a7 - source: openapi3 tags repository: https://github.com/ccwukong/snitch v3: true id: 09c3dafe2da696b61a144e189ea8cb5d repositoryMetadata: base64Readme: >- # snitch <img src="https://img.shields.io/badge/pypi-v0.1.34-green" /> <img src="https://img.shields.io/badge/Tested%20on-macOS%20Ventura-brightgreen" /> <img src="https://img.shields.io/badge/Tested%20on-Ubuntu%2022.10-brightgreen" /> 

![logo](docs/logo.png)

# Table of contents

<!--ts-->
  * [Introduction](#introduction)
  * [Main features](#main-features)
    * [Demo](#demo)
  * [Installation](#installation)
  * [Use cases](#use-cases)
  * [How this works](#how-this-works)
  * [Flags](#flags)
  * [Idempotency](#idempotency)
  * [How to contribute](#how-to-contribute)
<!--te-->

## Introduction

snitch is a CLI tool that helps you do health check, API idempotency check and more for your APIs.

This is not a replacement for your existing testing tools, but rather it provides a convenient way to check your APIs swiftly.

## Main features

- Running health check concurrently (using coroutine)
- Running Idempotency check by mixing multithreading(for synchronous requests) and coroutines in the same event loop. This will improve I/O performance.
- More to come

[CHANGELOG](CHANGELOG.md)

### Demo

![Demo](docs/demo.gif)

## Installation

Install snitch via pip. Make sure you have Python 3 installed on your machine.

**❗CAUTION** The package name is **api-snitch** instead of **snitch**.

```
pip3 install api-snitch
```

Once you have the config JSON file ready, you can run this in your commandline prompt:

```
snitch -p your_config_json_file_path [-o your_output_directory]
```

### Use cases

Senario 1: I want to run all tasks(API health check and API idempotency check)

```concole
snitch -p your_config_json_file_path
```

Senario 2: I want to run the API Idempotency check task only

```concole
snitch -p your_config_json_file_path -t id
```

Senario 3: I don't have a configuration json file, and I want to create one named **config.json** using the default template and stored it in my **current directory**

```concole
snitch -i ./config.json
```

Senario 4: I want to run all tasks(API health check and API idempotency check) with all API responses printed.

```concole
snitch -p your_config_json_file_path -v
```

## How this works

snitch accepts 2 types of API contracts: Postman collection file version **>= 2.0** or OpenAPI(Swagger) file version **>= 3.0.0**.

First, you need to have a global .json file which provides all essential configurations in order to run the test. 

You can create it by running

```console
snitch -i path_to_store_the_config_json_file/your_json_file_name.json
```

Or, you can use this template:
```json
{
  "postmanCollection": {
    "version": "2.1", 
    "filePath": "absolute_path_to_the_postman_collection_json_file",
    "metadata":{ // you can put all your placeholder strings here, for instance, the placeholder string for the host of the REST endpoints
      "{{restUrl}}": "https://your_api_domain",
      "{{accessToken}}": "the access token string",
      "{{apiKey}}": "api key string",
      ...
    }
  },
  "openApi": {
    "version": "3.0.0", 
    "filePath": "absolute_path_to_the_open_api_yaml_file",
    "metadata":{
      "{{restUrl}}": "https://your_api_domain",
      "{{accessToken}}": "the access token string",
      "{{apiKey}}":"api key string",
      ...
    }
  }
}
```

## Flags

Available flags for snitch CLI commands.

| flag      | Description |
| ----------- | ----------- |
| -i      | ***OPTIONAL*** Create a new config JSON file with default template |
| --init   | same as -i        |
| -o      | ***OPTIONAL*** Your output directory for storing the test results. It has to be a directory |
| --output   | same as -o        |
| -p      | Your config JSON file path |
| --path   | same as -p        |
| -t      | ***OPTIONAL*** There are 2 tasks available:<br /> - hc (API health check)<br /> - id (API idempotency check) |
| --task   | same as -t        |
| -v      | ***OPTIONAL*** Print API responses if -v or --verbose flag is specified |
| --verbose   | same as -v        |
| --version   | Display the current version of snitch   |

## Idempotency

Explanation from [https://www.restapitutorial.com/](https://www.restapitutorial.com/lessons/idempotency.html#:~:text=From%20a%20RESTful%20service%20standpoint,as%20making%20a%20single%20request.)

> From a RESTful service standpoint, for an operation (or service call) to be idempotent, clients can make that same call repeatedly while producing the same result. In other words, making multiple identical requests has the same effect as making a single request. Note that while idempotent operations produce the same result on the server (no side effects), the response itself may not be the same (e.g. a resource's state may change between requests).

> The PUT and DELETE methods are defined to be idempotent. However, there is a caveat on DELETE. The problem with DELETE, which if successful would normally return a 200 (OK) or 204 (No Content), will often return a 404 (Not Found) on subsequent calls, unless the service is configured to "mark" resources for deletion without actually deleting them. However, when the service actually deletes the resource, the next call will not find the resource to delete it and return a 404. However, the state on the server is the same after each DELETE call, but the response is different.

> GET, HEAD, OPTIONS and TRACE methods are defined as safe, meaning they are only intended for retrieving data. This makes them idempotent as well since multiple, identical requests will behave the same.

POST method usually is not idempotent, however, for applications such as online banking, digital payment etc. it is also important to keep POST method idempotent to avoid duplicated payments and transactions. It's subjected to application owners to decide whether or not idempotency is non-trival to certain POST endpoints of their services, and snitch only simply checks if same API with same parameteres returns same response or not.

## How to contribute

**Step 1**: Clone this repo

**Step 2**: Create a Python 3 virtual environment

Run the following commands under **root** directory(NOT snitch/)

```console
python3 -m venv venv

source ./venv/bin/activate
```

**Step 3**: Create a new branch, add your code and test cases, make sure nothing breaks

Run unit testing under **root** directory(NOT snitch/)
```console
tox -e py310 -- ./tests
```

**Step 4**: Test the program manually

Run the following command under **root** directory(NOT snitch/) to avoid the relative import path issue.

```console 
python3 -m snitch.main -p your_config_json_file_path [-o your_output_directory]

```

**Step 5**: Push your branch and create a PR for review

 readmeEtag: '"0b7a8d76ac2aff61285f215e915e721fffc9a84f"' readmeLastModified: Thu, 12 Jan 2023 03:04:54 GMT repositoryId: 577734803 description: A CLI tool that does health and idempotency checks for your REST APIs created: '2022-12-13T12:06:29Z' updated: '2025-12-27T13:16:40Z' language: Python archived: false stars: 3 watchers: 1 forks: 0 owner: ccwukong logo: https://avatars.githubusercontent.com/u/26168452?v=4 license: MIT repoEtag: '"4e9a5acdee48e9ac1748b7ce30eba611b086b812d281c1fe0b50256351313de9"' repoLastModified: Sat, 27 Dec 2025 13:16:40 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/easymetahub/emh-marklogic-glossary v3: true repositoryMetadata: base64Readme: >- # EasyMetaHub Glossary Manager for MarkLogic

## Introduction

This application is a SKOS glossary manager and faceted search application that can 
manage multiple glossaries.  It is intended for organizations that need to manage
one or more glossaries.

It was created as an easily customizable search application.  It abstracts out the
common code for faceted search and gives an easy development interface to customize
for uses other than a glossary manager.

#### About the author

[Loren Cahlander](https://www.linkedin.com/in/lorencahlander/) is the creator of
this tool and the [sister glossary application](https://github.com/easymetahub/emh-exist-glossary)
for [eXist-db](http://exist-db.org).  

He has also taken over the maintenance
of [xqDoc](http://xqdoc.org) that provides a simple vendor neutral solution 
for documenting XQuery library and main modules, as well as tools to generate 
a user friendly presentation of this documentation and cross reference information.

#### Extras

This application also has an 

* xqDoc viewer to view the documentation of the codebase for this application
* RESTful OpenAPI viewer for the REST services defined using the XQRS 
implementation of [RestXQ](http://exquery.github.io/exquery/exquery-restxq-specification/restxq-1.0-specification.html).

#### Consulting 
[EasyMetaHub](http://easymetahub.com) is available for consulting in developing your
own customization of this tool.

## Basic installation and getting started is here:

In the root directory of the application run:

```gradle mlDeploy```

The application is deployed to port *8006* as the application *emh-glossary-xqrs*

Open 
[http://localhost:8006](http://localhost:8006) and you get the following.

![images/emh-glossary-01.png](images/emh-glossary-01.png)

Searching requires a user with the role *emh-glossary-reader*. 
To load a glossary, login as *emh-glossary-writer* with the 
password *emh-glossary-writer*.  Click on the *lock* to open
the login dialog.

![images/emh-glossary-02.png](images/emh-glossary-02.png)

The toolbar changes to show the icons for the administrative screen and to logout.

![images/emh-glossary-03.png](images/emh-glossary-03.png)

Click on the gear icon for the administrative screen and the page changes to the admin screen.

![images/emh-glossary-admin-01.png](images/emh-glossary-admin-01.png)

Click on the *Upload Files...* button.
Select the *IVOAT.rdf*  in the *sample-data* folder.

![images/emh-glossary-05.png](images/emh-glossary-05.png)

The loaded glossary is listed in the admin page after this.

![images/emh-glossary-05a.png](images/emh-glossary-05a.png)


Click the left pointing chevron icon to return to the search page.

![images/emh-glossary-06.png](images/emh-glossary-06.png)

Type *galaxy* in the search bar and you get the following.

![images/emh-glossary-07.png](images/emh-glossary-07.png)

You can then select a facet to narrow the search results.  You can also expand a result item by selecting *Show Details*

![images/emh-glossary-08.png](images/emh-glossary-08.png)

If you select one of the buttons for *Related*, *Broader*, or *Narrower*, then you will be hyperlinked to that *Concept*

![images/emh-glossary-09.png](images/emh-glossary-09.png)

## Customizing

The following three files are the ones to customize to change the application from a 
glossary application to one of your own content.

* src/main/ml-modules/root/modules/custom/custom.xqm
* src/main/ml-config/databases/content-database.json
* src/main/polymer/root/src/emh-accelerator-app/emh-accelerator-app.js
* src/main/polymer/root/src/emh-accelerator-app/result-item.js

### custom.xqm

This file has the abstracted out customization layer for handling *search* and *content upload*

#### Search

The search cutomization is handled in two different functions in this 
XQuery Library Module.

```
(:~
 : Returns the search option for the query.
 :
 : @return The search options for the search:search() call in the search module.
 :)
declare function custom:search-options()
```
That returns the XML search options.  These options define the constraints 
for the search as well as the facets for the search.

```
(:~
 : Generates the JSON object for a result item.
 :
 : @param $result A 'search:result' object from the search results
 : @param $show-snippets A flag for whether to show the snippets.
 : @return The JSON object that represents a result item in the client page
 :)
declare function custom:result-object($result as node(), $show-snippets as xs:boolean)
```

This is called for each result that is returned from the search.



#### Upload

The upload customization is used to take the uploaded file and perform the 
proper processing of the file before it is stored in the content database.

```
(:~
 : This function processes the file that was uploaded through the upload dialog.
 :
 :  @param $filename The name of the file that has ben uploaded
 :  @param $file     The file that has been uploaded.
 :  @return An array of JSON objects as { "type": error-type, "message": error-message }
 :)
declare function custom:process-upload($filename as xs:string, $file)
as object-node()*
```

### content-database.json

### result-item.js

### emh-accelerator-app.js

## Others

* README.md
* build.gradle
* gradle.properties
* src/main/ml-config/databases/content-database.json
* src/main/ml-config/rest-api.json
* src/main/ml-config/security/roles/1-emh-glossary-nobody-role.json
* src/main/ml-config/security/roles/2-emh-glossary-reader-role.json
* src/main/ml-config/security/roles/3-emh-glossary-writer-role.json
* src/main/ml-config/security/roles/4-emh-glossary-internal-role.json
* src/main/ml-config/security/roles/5-emh-glossary-admin-role.json
* src/main/ml-config/security/users/emh-glossary-admin-user.json
* src/main/ml-config/security/users/emh-glossary-reader-user.json
* src/main/ml-config/security/users/emh-glossary-writer-user.json
* src/main/ml-config/servers/xqrs-server.json
* src/main/ml-modules/options/emh-glossary-options.xml
* src/main/ml-modules/rest-properties.json
* src/main/ml-modules/root/example-guestbook.xqy
* src/main/ml-modules/root/example.xqy
* src/main/ml-modules/root/favicon.ico
* src/main/ml-modules/root/guestbook.css
* src/main/ml-modules/root/images/app-icon-144.png
* src/main/ml-modules/root/images/app-icon-32.png
* src/main/ml-modules/root/images/chrome-splashscreen-icon-384x384.png
* src/main/ml-modules/root/images/chrome-touch-icon-192x192.png
* src/main/ml-modules/root/images/icon-128x128.png
* src/main/ml-modules/root/images/icon-512x512.png
* src/main/ml-modules/root/images/icon-72x72.png
* src/main/ml-modules/root/images/icon-96x96.png
* src/main/ml-modules/root/images/ms-icon-144x144.png
* src/main/ml-modules/root/images/powered-by-marklogic.png
* src/main/ml-modules/root/modules/config.xqm
* src/main/ml-modules/root/modules/custom/custom.xqm
* src/main/ml-modules/root/modules/emh-json.xqm
* src/main/ml-modules/root/modules/search-xqrs.xqy
* src/main/ml-modules/root/modules/upload-xqrs.xqy
* src/main/ml-modules/root/openapi/index.html
* src/main/ml-modules/root/openapi/openapi.xqy
* src/main/ml-modules/root/openapi/rxq2xqdoc2openapi-lib.xqy
* src/main/ml-modules/root/openapi/services2xqdoc2openapi-lib.xqy
* src/main/ml-modules/root/openapi/xqrs2xqdoc2openapi-lib.xqy
* src/main/ml-modules/root/xqdoc-lib.xqy
* src/main/polymer/root/favicon.ico
* src/main/polymer/root/images/app-icon-144.png
* src/main/polymer/root/images/app-icon-32.png
* src/main/polymer/root/images/chrome-splashscreen-icon-384x384.png
* src/main/polymer/root/images/chrome-touch-icon-192x192.png
* src/main/polymer/root/images/icon-128x128.png
* src/main/polymer/root/images/icon-512x512.png
* src/main/polymer/root/images/icon-72x72.png
* src/main/polymer/root/images/icon-96x96.png
* src/main/polymer/root/images/ms-icon-144x144.png
* src/main/polymer/root/images/powered-by-marklogic.png
* src/main/polymer/root/index.html
* src/main/polymer/root/manifest.json
* src/main/polymer/root/package-lock.json
* src/main/polymer/root/package.json
* src/main/polymer/root/polymer.json
* src/main/polymer/root/src/emh-accelerator-app/emh-accelerator-app.js
* src/main/polymer/root/src/emh-accelerator-app/facet-card.js
* src/main/polymer/root/src/emh-accelerator-app/facet-selector.js
* src/main/polymer/root/src/emh-accelerator-app/result-item-button.js
* src/main/polymer/root/src/emh-accelerator-app/result-item.js
* src/main/polymer/root/src/emh-accelerator-app/search-snippet-highlight.js
* src/main/polymer/root/src/emh-accelerator-app/upload-item.js
* src/main/polymer/xqDoc/README.md
* src/main/polymer/xqDoc/index.html
* src/main/polymer/xqDoc/manifest.json
* src/main/polymer/xqDoc/package-lock.json
* src/main/polymer/xqDoc/package.json
* src/main/polymer/xqDoc/polymer.json
* src/main/polymer/xqDoc/src/xqdoc-app/function-detail.js
* src/main/polymer/xqDoc/src/xqdoc-app/hash-button.js
* src/main/polymer/xqDoc/src/xqdoc-app/import-detail.js
* src/main/polymer/xqDoc/src/xqdoc-app/module-selector.js
* src/main/polymer/xqDoc/src/xqdoc-app/variable-detail.js
* src/main/polymer/xqDoc/src/xqdoc-app/xqdoc-app.js
* src/main/polymer/xqDoc/src/xqdoc-app/xqdoc-comment.js
* src/main/polymer/xqDoc/src/xqdoc-app/xqdoc-module.js
* xqDoc/openapi.json

## xqDoc

Click on the xqDoc icon and you get the following in a new tab.

![images/xqdoc-01.png](images/xqdoc-01.png)

## OpenAPI

Click on the OpenAPI icon and you get the Swagger UI of the RESTful services
of the application in a new tab.

![images/openapi-01.png](images/openapi-01.png)

## Training Available

[EasyMetaHub](http://easymetahub.com) is available for training in proper
XQuery documentation and OpenAPI generation. readmeEtag: '"de3b387ffa71a2e41370e6b27287397db4cd47b6"' readmeLastModified: Tue, 05 Dec 2023 00:00:15 GMT repositoryId: 224879779 description: >- This is a template application for MarkLogic that is for a glossary viewer that also shows xqDoc and OpenAPI (Swagger-UI) created: '2019-11-29T15:34:21Z' updated: '2025-06-11T06:14:38Z' language: XQuery archived: false stars: 4 watchers: 2 forks: 0 owner: easymetahub logo: https://avatars.githubusercontent.com/u/17910373?v=4 license: Apache-2.0 repoEtag: '"eafc47fac8e66c3d210db395a9d55a9238753a987807c8782589a94c661f36c0"' repoLastModified: Wed, 11 Jun 2025 06:14:38 GMT foundInMaster: true category: Testing id: edb1cfa247ab744d31e76832e8660d92 - source: openapi3 tags repository: https://github.com/which0113/api-frontend v3: true id: 8c7c34b973e448a59fbc3042c39cbc07 repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgICA8aW1nIHNyYz1kb2MvbG9nby5wbmcgd2lkdGg9MTg4Lz4KPC9wPgo8aDEgYWxpZ249ImNlbnRlciI+5ZK46bG8LUFQSSDmjqXlj6PlvIDmlL7lubPlj7A8L2gxPgo8cCBhbGlnbj0iY2VudGVyIj48c3Ryb25nPuWSuOmxvC1BUEkg5o6l5Y+j5byA5pS+5bmz5Y+w5piv5LiA5Liq5Li655So5oi35ZKM5byA5Y+R6ICF5o+Q5L6b5YWo6Z2iQVBJ5o6l5Y+j6LCD55So5pyN5Yqh55qE5bmz5Y+wIPCfm6A8L3N0cm9uZz48L3A+CjxkaXYgYWxpZ249ImNlbnRlciI+CjxhIHRhcmdldD0iX2JsYW5rIiBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vd2hpY2gwMTEzL2FwaS1iYWNrZW5kIj4KICAgIDxpbWcgYWx0PSIiIHNyYz0iaHR0cHM6Ly9naXRodWIuY29tL3doaWNoMDExMy9hcGktYmFja2VuZC9iYWRnZS9zdGFyLnN2Zz90aGVtZT1ndnAiLz4KPC9hPgogICAgPGltZyBhbHQ9Ik1hdmVuIiBzcmM9Imh0dHBzOi8vcmFzdGVyLnNoaWVsZHMuaW8vYmFkZ2UvTWF2ZW4tMy44LjEtcmVkLnN2ZyIvPgo8YSB0YXJnZXQ9Il9ibGFuayIgaHJlZj0iaHR0cHM6Ly93d3cub3JhY2xlLmNvbS90ZWNobmV0d29yay9qYXZhL2phdmFzZS9kb3dubG9hZHMvaW5kZXguaHRtbCI+CiAgICAgICAgPGltZyBhbHQ9IiIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0pESy0xLjgrLWdyZWVuLnN2ZyIvPgo8L2E+CiAgICA8aW1nIGFsdD0iU3ByaW5nQm9vdCIgc3JjPSJodHRwczovL3Jhc3Rlci5zaGllbGRzLmlvL2JhZGdlL1NwcmluZ0Jvb3QtMi43Ky1ncmVlbi5zdmciLz4KPC9kaXY+CgojIyDpobnnm67ku4vnu40KCvCfmIAg5L2c5Li655So5oi35oKo5Y+v5Lul6YCa6L+H5rOo5YaM55m75b2V6LSm5oi377yM6I635Y+W5o6l5Y+j6LCD55So5p2D6ZmQ77yM5bm25qC55o2u6Ieq5bex55qE6ZyA5rGC5rWP6KeI5ZKM6YCJ5oup6YCC5ZCI55qE5o6l5Y+j44CC5oKo5Y+v5Lul5Zyo57q/6L+b6KGM5o6l5Y+j6LCD6K+V77yM5b+r6YCf6aqM6K+B5o6l5Y+j55qE5Yqf6IO95ZKM5pWI5p6c44CCCgrwn5K7IOS9nOS4uuW8gOWPkeiAhe+8jOaIkeS7rOaPkOS+m+S6hlvlkrjpsbwtQVBJIFNES10oaHR0cHM6Ly9naXRodWIuY29tL3doaWNoMDExMy9hcGktc2RrKe+8jArpgJrov4db5byA5Y+R6ICF5Yet6K+BXShodHRwczovL3d3dy5mcmVlZmlzaC5sb3ZlL2FjY291bnQvY2VudGVyKeWNs+WPr+Wwhui9u+advumbhuaIkOaOpeWPo+WIsOaCqOeahOmhueebruS4re+8jOWunueOsOabtOmrmOaViOeahOW8gOWPkeWSjOiwg+eUqOOAggoK8J+knSDmgqjlj6/ku6XlsIboh6rlt7HnmoTmjqXlj6PmjqXlhaXliLAgKipf5ZK46bG8LUFQSV8qKiDmjqXlj6PlvIDmlL7lubPlj7DlubPlj7DkuIrvvIzlubblj5HluIPnu5nlhbbku5bnlKjmiLfkvb/nlKjjgIIK5oKo5Y+v5Lul566h55CG5ZKM5ZCE5Liq5o6l5Y+j77yM5Lul5L6/5pu05aW95Zyw5YiG5p6Q5ZKM5LyY5YyW5o6l5Y+j5oCn6IO944CCCgrwn5SOIOaCqOWPqumcgOimgeWvvOWFpeacgOWOn+Wni+eahOaVsOaNrumbhu+8jOi+k+WFpemcgOimgeWIhuaekOeahOebruagh++8jOWwseiDveWIqeeUqFtBSeaZuuiDveeUn+aIkF0oaHR0cHM6Ly93d3cuZnJlZWZpc2gubG92ZS9hbmFseXNlKQrkuIDkuKrlj6/kuIvovb3nmoTmlbDmja7liIbmnpDlm77ooajlkozliIbmnpDnu5PorrrjgIIKCvCfj4Eg5peg6K665oKo5piv55So5oi36L+Y5piv5byA5Y+R6ICF77yMKipf5ZK46bG8LUFQSV8qKgrlkrjpsbwtQVBJIOaOpeWPo+W8gOaUvuW5s+WPsOmDveiHtOWKm+S6juaPkOS+m+eos+WumuOAgeWuieWFqOOAgemrmOaViOeahOaOpeWPo+iwg+eUqOWSjOaVsOaNruWIhuaekOacjeWKoe+8jOW4ruWKqeaCqOWunueOsOabtOmrmOaViOOAgeS+v+aNt+WMlueahOW8gOWPkeWSjOiwg+eUqOS9k+mqjOOAggoKIyMg572R56uZ5a+86IiqCgotIFvpobnnm67lnKjnur/mvJTnpLrlnLDlnYAg8J+Ul10oaHR0cHM6Ly93d3cuZnJlZWZpc2gubG92ZSkKLSDmvJTnpLrotKblj7fvvJpkZW1vCi0g5a+G56CB77yaMTIzNDU2NzgKLSBb6aG555uu5YmN56uv5Zyw5Z2AIPCflJddKGh0dHBzOi8vZ2l0aHViLmNvbS93aGljaDAxMTMvYXBpLWZyb250ZW5kKQotIFvpobnnm67lkI7nq6/lnLDlnYAg8J+Ul10oaHR0cHM6Ly9naXRodWIuY29tL3doaWNoMDExMy9mcmVlZmlzaC1hcGkpCgojIyDkvb/nlKjmjIflr7wKCiMjIyDlhYvpmobpobnnm67liLDmnKzlnLAKCmBgYGJhc2gKZ2l0IGNsb25lIGdpdEBnaXRodWIuY29tOndoaWNoMDExMy9hcGktZnJvbnRlbmQuZ2l0CmBgYAoKIyMjIOi/kOihjOWJjeerrwoK56Gu5L+dIG5vZGUgPj0gMTjvvIzmn6XnnIsgbm9kZSDniYjmnKwKCmBgYGJhc2gKbm9kZSAtdgpgYGAKCiMjIyMg5a6J6KOF5L6d6LWWCgpgYGBiYXNoCm5wbSBpbnN0YWxsCmBgYAoKIyMjIyDlkK/liqgKCmBgYGJhc2gKbnBtIHJ1biBkZXYKYGBgCgojIyMjIOWFtuS7lgoKLSBbcmVxdWVzdENvbmZpZy50c10oc3JjJTJGcmVxdWVzdENvbmZpZy50cykg5paH5Lu25Y+v5L+u5pS55a+55o6l5ZCO56uv5pyN5Yqh5Zmo5o6l5Y+j55qE5Zyw5Z2A77yM6buY6K6k77yaaHR0cDovL2xvY2FsaG9zdDo5MDAxCi0gW2luZGV4LnRzeF0oc3JjJTJGcGFnZXMlMkZNeUNoYXJ0JTJGaW5kZXgudHN4KSDlj6/kv67mlLkgV2ViU29ja2V0IOiHquWumuS5iei/nuaOpeWcsOWdgO+8jOm7mOiupO+8mndzOi8vbG9jYWxob3N0OjkwMDEvYXBpL3dzLwoKIyMjIyDpg6jnvbLliY3nq68KCuaWueazleS4gO+8mueUn+aIkCBkaXN0IOaWh+S7tumDqOe9sgoKYGBgYmFzaApucG0gcnVuIGJ1aWxkCmBgYAoK5pa55rOV5LqM77ya5L2/55SoIGRvY2tlciDlrrnlmajpg6jnvbIKCuaehOW7uu+8mgoKYGBgYmFzaApkb2NrZXIgYnVpbGQgLXQgZnJlZWZpc2gtYXBpLWZyb250ZW5kOnYwLjAuMSAuCmBgYAoK6L+Q6KGM77yaCgpgYGBiYXNoCmRvY2tlciBydW4gLXAgODA6ODAgLWQgZnJlZWZpc2gtYXBpLWZyb250ZW5kOnYwLjAuMQpgYGAKCiMjIOmhueebruWxleekugoKIyMjIOmmlumhtQoKIVtob21lLnBuZ10oZG9jJTJGaG9tZS5wbmcpCgojIyMg5o6l5Y+j5bm/5Zy6CgojIyMjIOaOpeWPo+WxleekugoKIVthcGktZGlzcGxheS5wbmddKGRvYyUyRmFwaS1kaXNwbGF5LnBuZykKCiMjIyMg5o6l5Y+j6K+m5oOFCgohW2FwaS1kZXRhaWxzLnBuZ10oZG9jJTJGYXBpLWRldGFpbHMucG5nKQoKIyMjIyMgQVBJ5paH5qGjCgohW2FwaS1kb2MucG5nXShkb2MlMkZhcGktZG9jLnBuZykKCiMjIyMjIOmUmeivr+eggQoKIVtlcnJvci1jb2Rlcy5wbmddKGRvYyUyRmVycm9yLWNvZGVzLnBuZykKCiMjIyMjIOekuuS+i+S7o+eggQoKIVtzYW1wbGUtY29kZS5wbmddKGRvYyUyRnNhbXBsZS1jb2RlLnBuZykKCiMjIyMjIOWcqOe6v+iwg+ivlQoKIVtvbmxpbmUtY29tbWlzc2lvbmluZy5wbmddKGRvYyUyRm9ubGluZS1jb21taXNzaW9uaW5nLnBuZykKCiMjIyDmmbrog73liIbmnpAKCiFbaW50ZWxsaWdlbnQtYW5hbHl0aWNzLnBuZ10oZG9jJTJGaW50ZWxsaWdlbnQtYW5hbHl0aWNzLnBuZykKCiMjIyDmiJHnmoTlm77ooagKCiMjIyMg5Zu+6KGo5bGV56S6CgohW2NoYXJ0LWRpc3BsYXkucG5nXShkb2MlMkZjaGFydC1kaXNwbGF5LnBuZykKCiMjIyMg5Zu+6KGo5LiL6L29CgohW2NoYXJ0LWRvd25sb2FkLnBuZ10oZG9jJTJGY2hhcnQtZG93bmxvYWQucG5nKQoKIyMjIOeUqOaIt+WKn+iDvQoKIyMjIyDnmbvlvZUKCiFbbG9naW4ucG5nXShkb2MlMkZsb2dpbi5wbmcpCgojIyMjIOazqOWGjAoKIVtyZWdpc3Rlci5wbmddKGRvYyUyRnJlZ2lzdGVyLnBuZykKCiMjIyMg5Liq5Lq65Lit5b+DCgohW3BlcnNvbmFsLWNlbnRlci5wbmddKGRvYyUyRnBlcnNvbmFsLWNlbnRlci5wbmcpCgojIyMg566h55CG5ZGY5Yqf6IO9CgojIyMjIOaOpeWPo+euoeeQhgoKIVtpbnRlcmZhY2UtbWFuYWdlbWVudC5wbmddKGRvYyUyRmludGVyZmFjZS1tYW5hZ2VtZW50LnBuZykKCiMjIyMg5Zu+6KGo566h55CGCgohW2NoYXJ0LW1hbmFnZW1lbnQucG5nXShkb2MlMkZjaGFydC1tYW5hZ2VtZW50LnBuZykKCiMjIyMg55So5oi3566h55CGCgohW3VzZXItbWFuYWdlbWVudC5wbmddKGRvYyUyRnVzZXItbWFuYWdlbWVudC5wbmcpCgojIyDmioDmnK/moIgKCi0gUmVhY3QgMTgKLSBBbnQgRGVzaWduIFBybyA1Lngg6ISa5omL5p62Ci0gQW50IERlc2lnbiAmIFByb2NvbXBvbmVudHMg57uE5Lu25bqTCi0gVW1pIDQg5YmN56uv5qGG5p62Ci0gT3BlbkFQSSDliY3nq6/ku6PnoIHnlJ/miJAK readmeEtag: '"a84070c88e22d9a6dee11cea21f247ef9622823c"' readmeLastModified: Sun, 06 Oct 2024 14:18:51 GMT repositoryId: 736579965 description: >- 本项目基于 React + Spring Boot + Dubbo + AIGC,致力于为用户和开发者提供全面 API 接口调用和 AI 数据分析服务。 created: '2023-12-28T09:39:19Z' updated: '2025-06-23T23:57:26Z' language: TypeScript archived: false stars: 5 watchers: 2 forks: 0 owner: which0113 logo: https://avatars.githubusercontent.com/u/119718783?v=4 repoEtag: '"ce7d5fae9c4948bdfc0838707aa1a05cf8bb2f19cf6330e57cd79a128547e902"' repoLastModified: Mon, 23 Jun 2025 23:57:26 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/yaniv-golan/airtable-openapi-schema-generator v3: true id: b0272f455ab27bcc26743cbba3ce0c1b repositoryMetadata: base64Readme: >- # Airtable OpenAPI 3.1.0 Schema Generator for GPT Actions (Read-Only)

This script generates an OpenAPI 3.1.0 schema based on your Airtable base structure. The schema is designed to be used with Custom GPT (ChatGPT) Actions, allowing the GPT assistant to interact with your Airtable data via the Airtable API.

## Table of Contents

- [Purpose](#purpose)
- [Features](#features)
- [Prerequisites](#prerequisites)
- [Setup and Usage](#setup-and-usage)
  - [1. Open the Airtable Scripting Extension](#1-open-the-airtable-scripting-extension)
  - [2. Paste the Script](#2-paste-the-script)
  - [3. Run the Script](#3-run-the-script)
  - [4. Retrieve the Generated Schema](#4-retrieve-the-generated-schema)
  - [5. Configure Your Custom GPT Assistant](#5-configure-your-custom-gpt-assistant)
  - [6. Specify the API Key](#6-specify-the-api-key)
  - [7. Update the Assistant Instructions](#7-update-the-assistant-instructions)
  - [8. Test the Assistant](#8-test-the-assistant)
- [Script Explanation](#script-explanation)
- [Generated Schema Usage](#generated-schema-usage)
- [Handling Linked Records](#handling-linked-records)
- [Unmapped Field Types](#unmapped-field-types)
- [Troubleshooting](#troubleshooting)
- [Contributing](#contributing)
- [License](#license)

## Purpose

The script automates the creation of an OpenAPI schema that reflects your Airtable base's tables, fields, and relationships. This schema can then be used to configure a Custom GPT assistant that can answer queries by interacting directly with your Airtable data.

## Features

- **Automatic Schema Generation**: Generates an OpenAPI 3.1.0 schema based on your Airtable base.
- **Field Type Mapping**: Maps Airtable field types to appropriate OpenAPI data types.
- **Linked Records Handling**: Expresses relationships between tables using custom `x-linkedTable` properties.
- **Preservation of Field Descriptions**: Includes existing field descriptions from your Airtable base.
- **Error Reporting**: Identifies and reports unmapped field types for manual attention.

## Prerequisites

- **Airtable Account**: Access to the base you wish to generate a schema for.
- **Airtable Scripting Extension**: Enabled for your base.
- **OpenAI Account**: With access to the Custom GPT feature and the ability to configure actions and upload schemas.
- **API Key**: Your Airtable API key.
  - You can find your Airtable API key by logging into Airtable and visiting your account page.
  - Keep your API key secure and do not share it publicly.

## Setup and Usage

### 1. Open the Airtable Scripting Extension

- Navigate to your Airtable base.
- Click on the "Extensions" button in the top-right corner.
- Add a new extension and select "Scripting".

### 2. Paste the Script

- Copy the entire script from the `script.js` file.
- Paste it into the scripting editor within the Airtable Scripting Extension.

### 3. Run the Script

- Click the "Run" button to execute the script.
- The script will generate the OpenAPI schema based on your base's structure.

### 4. Retrieve the Generated Schema

- After running, the script outputs markdown containing the OpenAPI schema JSON.
- Copy the JSON text displayed in the output section.

### 5. Configure Your Custom GPT Assistant

- Log into your OpenAI account and navigate to the Custom GPT settings.
- Create a new GPT or select an existing one to configure.
- In the "Actions" section, click on "Create new action" to create a new action.
- Paste the OpenAPI schema you copied into the **Schema** input field.
- **Set the Authentication Method**:
  - Select **"API Key"** as the authentication method.
- Proceed to the next step to set up the authentication details.

### 6. Specify the API Key

- In the **API Key** authentication settings:
  - **API Key Header Name**: Enter `Authorization`.
  - **API Key Value Prefix**: Enter `Bearer ` (including the space).
  - **API Key**: Enter your Airtable API key.
    - Ensure that your API key is stored securely.
    - The API key will be encrypted and stored securely by OpenAI.

### 7. Update the Assistant Instructions

- Copy the sample assistant instructions from the [sample_gpt_instructions.md](sample_gpt_instructions.md) file.
- Paste them into the **Instructions** field of your Custom GPT assistant.
- These instructions guide the assistant on how to interact with the API and handle data appropriately.

### 8. Test the Assistant

- Interact with your GPT assistant by asking queries related to your Airtable data.
- Verify that it can retrieve and present information accurately.
- Troubleshoot any issues using the guidance provided in this README.

## Script Explanation

The script performs the following actions:

- **Field Type Mapping**: Maps Airtable field types to OpenAPI data types via the `AIRTABLE_TO_OPENAPI_DEFS` object.
- **Schema Generation**: Iterates through each table in the base to generate corresponding schemas and endpoint definitions.
- **Linked Records Handling**:
  - Identifies fields of type `multipleRecordLinks`.
  - Uses a helper function `getTableById` to resolve linked table IDs to names.
  - Adds a custom `x-linkedTable` property to express relationships without nesting.
- **Preserving Descriptions**: Includes any existing field descriptions in the schema.
- **Output Generation**: Constructs the OpenAPI schema and outputs it as JSON.

## Generated Schema Usage

- The generated OpenAPI schema defines the structure of your Airtable base for API interactions.
- **Endpoints**: Each table has an endpoint for listing records.
- **Security**: The schema includes a `BearerAuth` security scheme for API key authentication.
- **Schemas**: Defines the data structures for each table, including fields and their types.
- **Linked Records**:
  - Represented as arrays of record IDs (strings).
  - Include an `x-linkedTable` property indicating the related table.

## Handling Linked Records

- **No Nested Data**: The Airtable API returns linked records as arrays of record IDs, not nested objects.
- **Expressing Relationships**: The `x-linkedTable` property in the schema indicates which table a linked record field references.
- **Assistant Behavior**:
  - When encountering a linked record field, use the `x-linkedTable` property to determine the related table.
  - Make additional API calls to the appropriate table's endpoint to retrieve related records using the record IDs.
  - Do not expose record IDs to the user; present human-readable names or relevant information.

## Unmapped Field Types

- If the script encounters field types that are not mapped in `AIRTABLE_TO_OPENAPI_DEFS`, it will:
  - Add them to the `unmappedFieldTypes` set.
  - Output a warning at the end of script execution.
- **Action Required**:
  - Review any unmapped field types listed.
  - Update the `AIRTABLE_TO_OPENAPI_DEFS` object to include appropriate mappings.
  - Rerun the script to regenerate the schema with the new mappings.

## Troubleshooting

### Error: `UnrecognizedKwargsError: fields`

- **Cause**: The parameter name `fields[]` may cause issues with serialization.
- **Solution**:
  - The script has been updated to use `fields` instead of `fields[]`.
  - Ensure you are using the latest version of the script.
  - Regenerate and update the schema in your Custom GPT configuration.

### Assistant Not Resolving Linked Records

- **Cause**: The assistant may not be programmed to utilize the `x-linkedTable` property.
- **Solution**:
  - Update the assistant's instructions to handle linked records using the `x-linkedTable` metadata (see `instructions.txt`).
  - Ensure it makes additional API calls to fetch related records as needed.

### Unmapped Field Types Warning

- **Cause**: New or custom field types not defined in the script.
- **Solution**:
  - Add mappings for these field types in the `AIRTABLE_TO_OPENAPI_DEFS` object.
  - If unsure, map them to `'type': 'string'` as a default.
  - Rerun the script after making changes.

## Contributing

- If you make improvements to the script or encounter issues, consider sharing your updates.
- Contributions can help others who are using similar setups.
- Feel free to fork the script or submit pull requests with enhancements.

## License

This script is provided as-is without any warranty. Use it at your own risk. Make sure to comply with Airtable's API usage policies and OpenAI's terms of service when using the script and integrating it with GPT assistants.

---

**Note**: Always ensure that you handle sensitive data securely and comply with any relevant data protection regulations when exposing your Airtable data to external systems.

---

**Disclaimer**: The instructions provided in this README are meant for informational purposes and to aid in setting up the script and GPT assistant. OpenAI's policies and Airtable's terms may change over time; always refer to the official documentation for the most current information.
 readmeEtag: '"0570379ca51b9bd28e356d5f0c058cd45a8376c0"' readmeLastModified: Thu, 26 Sep 2024 15:50:13 GMT repositoryId: 863414293 description: >- A script that generates an OpenAPI 3.1.0 schema based on your Airtable base structure. This schema is designed for use with Custom GPT (ChatGPT) Actions, allowing a GPT assistant to interact with your Airtable data via the Airtable API. created: '2024-09-26T08:55:04Z' updated: '2025-11-03T00:55:17Z' language: JavaScript archived: false stars: 12 watchers: 2 forks: 1 owner: yaniv-golan logo: https://avatars.githubusercontent.com/u/5079117?v=4 license: MIT repoEtag: '"ac7780326bba45e1acd3f0b75fa93bdff1c3de0a1393c1bf9e95bb27380a8a74"' repoLastModified: Mon, 03 Nov 2025 00:55:17 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/hiroyoshii/go-aas-proxy v3: true id: dc073f77dba2b9ffe063aa18b4df40a5 repositoryMetadata: base64Readme: >-   <div align="center">
  <h1>go-aas-proxy</h1>
  <p>
  Asset Administration Shell(AAS) proxy for RDBMS backend
  </p>

  [![Build](https://github.com/hiroyoshii/go-aas-proxy/actions/workflows/go_application.yml/badge.svg)](https://github.com/hiroyoshii/go-aas-proxy/actions/workflows/go_application.yml)
  [![CodeQL](https://github.com/hiroyoshii/go-aas-proxy/actions/workflows/codeql.yml/badge.svg)](https://github.com/hiroyoshii/go-aas-proxy/actions/workflows/codeql.yml)
  [![OpenSSF Scorecard](https://api.securityscorecards.dev/projects/github.com/hiroyoshii/go-aas-proxy/badge)](https://securityscorecards.dev/viewer/?uri=github.com/hiroyoshii/go-aas-proxy)
  [![Go Report Card](https://goreportcard.com/badge/github.com/hiroyoshii/go-aas-proxy)](https://goreportcard.com/report/github.com/hiroyoshii/go-aas-proxy)
  [![OpenSSF Best Practices](https://www.bestpractices.dev/projects/7795/badge)](https://www.bestpractices.dev/projects/7795)
  ![GitHub go.mod Go version](https://img.shields.io/github/go-mod/go-version/hiroyoshii/go-aas-proxy)
  [![License](https://img.shields.io/github/license/hiroyoshii/go-aas-proxy)](LICENSE)

  </div>


  ## About

  The go-aas-proxy is asset administration shell(aas) server which is implemented by go and proxy to RDBMS backend. **not ready for production**.

  This implementation is inspired from "aas-proxy" in [Representing the Virtual: Using AAS to Expose DigitalAssets](https://ceur-ws.org/Vol-3291/paper5.pdf) paper.
  ![aas-proxy](./assets/aas-proxy.png)

  "aas-proxy" means that submodels are configured by referencing the RDB of other applications.
  Therefore, only the AAS and the AAS-Submodel relationship can be created, updated, or deleted, while submodels are read-only.

  ## Features
  - Open API Endpoints compatible with [Basyx API](https://app.swaggerhub.com/apis/BaSyx/basyx_asset_administration_shell_http_rest_api/v1). But only supports following endpoints:
    - :white_check_mark: /shells: GET
    - :white_check_mark: /shells/{aasId}: GET/PUT/DELETE
    - :white_check_mark: /shells/{aasId}/aas/submodels: GET
    - :white_check_mark: /shells/{aasId}/aas/submodels/{submodelIdShort}: GET
    - :white_check_mark: /shells/{aasId}/aas/submodels/{submodelIdShort}/submodel/submodelElements: GET
    - :white_check_mark: /shells/{aasId}/aas/submodels/{submodelIdShort}/submodel/values: GET
    - :warning:	 /shells/{aasId}/aas/submodels/{submodelIdShort}: PUT(only relation to aas)
  - Support multiple RDBMS locations and types (Postgres, MySQL, Oracle)

  Unsuported:
  - :x: create/update/delete submodels and submodel elements
  - :x: Endpoints containing \{seIdShortPath\}
  - :x: Invocation Endpoints

  ## Implementation Overview
  ![architecture](./assets/architecture.drawio.png)
  - configurablity for submodel
    - reference databases per semanticID of submodel
    - query for databases
    - response json content for submodels

  ## How to run
  1. screate following submodel configuration file.
    - there are multiple reference database and query templates.
    - SQL query template has requested path parameter variables: .AasID, .SubmodelIDShort.
    - response template has query results variables: `.<template file name>.Results.<column name>` and `.<template file name>.Columns.<column name>`.
      - for example, template file path is sample/query.sql and column name is sample, variables are `.Query.Results.Sample` and `.Query.Columns.Sample`
  ```yaml
  submodels:
    - semanticID: "<submodel semantic ID>"
      queryTemplates:
        - path: "<SQL query file path (go template)>"
          dbName: "<database name for query execution>"
      responseTemplatePath: "<submodel json template file path>"
  databases:
    - name: "<database name referenced by submodels.queryTemplates.dbName>"
      dbType: <type of RDBMS: postgres, mysql or oracle>
      dbConfig:
        host: <database host>
        port: <database port number>
        user: "<database user>"
        password: "<database password>"
        database: "<database name>"
        sslmode: "<sslmode if dbType is postgres>"
        options: "<database connection options>"
  ```
  2. set up postgresql database for aas and asset
  3. docker run
  ```
  docker run \
  -e SUBMODEL_CONFIG_PATH=<configuration file path in procedure 1.> \
  -e AAS_TABLES_CREATED=true \
  -e AAS_DB_HOST=<database host name in procedure 2.> \
  hiroyoshii/go-aas-proxy:latest
  ```

  ### Example
  * run with sample data (run docker compose and create sample deta)
  ```
  ./e2e/scenario_setup.sh
  ```
  * Submodel configurations can be found in the ./e2e/application/configs directory.

  ### Environment Variables
  | variables            | overview                                                              | sample               | required                | 
  | -------------------- | --------------------------------------------------------------------- | -------------------- | ----------------------- | 
  | SUBMODEL_CONFIG_PATH | submodel configuration file path.<br>                                 | submodel_config.yaml | yes                     | 
  | AAS_TABLES_CREATED   | DDL stored AAS data is created or not when go-aas-proxy is started.   | true                 | no (default: true)      | 
  | AAS_DB_HOST          | Postgresql database host name for AAS stored                          | 127.0.0.1            | no (default: 127.0.0.1) | 
  | AAS_DB_PORT          | Postgresql database port number for AAS stored                        | 5432                 | no (default: 5432)      | 
  | AAS_DB_USER          | Postgresql database user for AAS stored                               | postgres             | no (default: postgres)  | 
  | AAS_DB_PASSWORD      | Postgresql database password for AAS stored                           | password             | no (default: password)  | 
  | AAS_DB_DATABASE      | Postgresql database name for AAS stored                               | sample               | no (default: sample)    | 
  | AAS_DB_SSL_MODE      | Postgresql database sslmode for AAS stored                            | disable              | no (default: disable)   | 

  ## License
  Apache License 2.0, see [LICENSE](./LICENSE).
 readmeEtag: '"6ab2233e54b28e6c1982e82ae1173ec28ab9c5eb"' readmeLastModified: Tue, 22 Oct 2024 18:11:30 GMT repositoryId: 672491411 description: Asset Administration Shell(AAS) proxy for RDBMS backend created: '2023-07-30T09:12:05Z' updated: '2025-10-04T01:51:39Z' language: Go archived: false stars: 3 watchers: 1 forks: 1 owner: hiroyoshii logo: https://avatars.githubusercontent.com/u/40140638?v=4 license: Apache-2.0 repoEtag: '"db28435294fba1ae389a1b6052a35009939d82529d59416b3397f6f29cba5ef3"' repoLastModified: Sat, 04 Oct 2025 01:51:39 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/membrane-php/membrane-laravel v3: true id: 97d0ee9099351cdd23940bde3ad0428f repositoryMetadata: base64Readme: >- IyBNZW1icmFuZS1MYXJhdmVsCgpJbnRlZ3JhdGVzIFtNZW1icmFuZS1jb3JlXShodHRwczovL2dpdGh1Yi5jb20vbWVtYnJhbmUtcGhwL21lbWJyYW5lLWNvcmUpIHdpdGggW0xhcmF2ZWxdKGh0dHBzOi8vbGFyYXZlbC5jb20vKS4KCiMjIEFib3V0CgpNaWRkbGV3YXJlIHRoYXQgdmFsaWRhdGVzIHRoZSByYXcgdXNlciBpbnB1dCBmcm9tIGluY29taW5nIEhUVFAgcmVxdWVzdHMgYWdhaW5zdCB5b3VyIE9wZW5BUEkgc3BlYy4gIApBZGRzIGEgYE1lbWJyYW5lXFJlc3VsdFxSZXN1bHRgIG9udG8geW91ciBgSWxsdW1pbmF0ZVxDb250cmFjdHNcQ29udGFpbmVyXENvbnRhaW5lcmAuICAKVGhlIFJlc3VsdCBvYmplY3QgY29udGFpbnMgdGhlIGNsZWFuZWQgdXAgZGF0YSBhbmQgYWRkaXRpb25hbCBkZXRhaWxzIGluIHRoZSBjYXNlIG9mIGludmFsaWQgcmVxdWVzdHMuCgojIyBTZXR1cAoKIyMjIEluc3RhbGxhdGlvbgoKUmVxdWlyZSB0aGUgYG1lbWJyYW5lL2xhcmF2ZWxgIHBhY2thZ2UgaW4geW91ciBjb21wb3Nlci5qc29uIGFuZCB1cGRhdGUgeW91ciBkZXBlbmRlbmNpZXM6CgpgYGBzaGVsbApjb21wb3NlciByZXF1aXJlIG1lbWJyYW5lL2xhcmF2ZWwKYGBgCgojIyMgQ29uZmlndXJhdGlvbgoKVGhlIGRlZmF1bHRzIGFyZSBzZXQgaW4gYGNvbmZpZy9tZW1icmFuZS5waHBgLiAgClRvIHB1Ymxpc2ggYSBjb3B5IHRvIHlvdXIgb3duIGNvbmZpZywgdXNlIHRoZSBmb2xsb3dpbmc6CgpgYGBzaGVsbApwaHAgYXJ0aXNhbiB2ZW5kb3I6cHVibGlzaCAtLXRhZz0ibWVtYnJhbmUiCmBgYAoKIyMjIyBBUEkgU3BlYyBGaWxlCgpUaGlzIGlzIHRoZSAqKmFic29sdXRlKiogZmlsZXBhdGggb2YgeW91ciBPcGVuQVBJLgoKQnkgZGVmYXVsdCwgaXQgbG9va3MgZm9yIGA8eW91ci1wcm9qZWN0LWRpcmVjdG9yeT4vYXBpL29wZW5hcGkueWFtbGAuCgojIyMjIFZhbGlkYXRpb24gRXJyb3IgUmVzcG9uc2UgQ29kZQoKU2V0IGAndmFsaWRhdGlvbl9lcnJvcl9yZXNwb25zZV9jb2RlJ2AgdG8gdGhlICoqaW50ZWdlcioqIHZhbHVlIG9mIHRoZSBkZWZhdWx0IGh0dHAgc3RhdHVzIGNvZGUgZm9yIGludmFsaWQgcmVzdWx0cy4KCiMjIyMgVmFsaWRhdGlvbiBFcnJvciBSZXNwb25zZSBUeXBlCgpTZXQgYCd2YWxpZGF0aW9uX2Vycm9yX3Jlc3BvbnNlX3R5cGUnYCB0byB0aGUgKipzdHJpbmcqKiB2YWx1ZSBvZiB0aGUgZGVmYXVsdCByZXNwb25zZSB0eXBlIGZvciBBUEkgcHJvYmxlbXMuCgojIyMjIEFQSSBQcm9ibGVtIFJlc3BvbnNlIFR5cGVzCgpXaXRoaW4gdGhlIGAnYXBpX3Byb2JsZW1fcmVzcG9uc2VfdHlwZXMnYCBhcnJheToKU2V0ICoqaW50ZWdlcioqIGh0dHAgc3RhdHVzIGNvZGUgPT4gKipzdHJpbmcqKiByZXNwb25zZSB0eXBlIHBhaXJzLiAgClRoZXNlIGFyZSBtb3JlIHNwZWNpZmljIGFuZCB3aWxsIG92ZXJyaWRlIHRoZSBkZWZhdWx0IHZhbHVlIHNldCBieSBgJ3ZhbGlkYXRpb25fZXJyb3JfcmVzcG9uc2VfdHlwZSdgCgojIyBVc2FnZQoKIyMjIFJlcXVlc3RzCgpUaGUgYFxNZW1icmFuZVxMYXJhdmVsXE1pZGRsZXdhcmVcUmVxdWVzdFZhbGlkYXRpb25gIG1pZGRsZXdhcmUgd2lsbCB2YWxpZGF0ZSBvciBpbnZhbGlkYXRlIGluY29taW5nIHJlcXVlc3RzIGFuZCBsZXQKeW91IGRlY2lkZQpob3cgdG8gcmVhY3QuCllvdSBjYW4gZm9sbG93IGl0IHdpdGggeW91ciBvd24gY3VzdG9tIG1pZGRsZXdhcmUgb3Igd2l0aCBvbmUgb2YgdGhlIGZvbGxvd2luZyBidWlsdC1pbiBvcHRpb25zIHRvIHByb2R1Y2UgYW4gZXJyb3IKcmVzcG9uc2U6CgojIyMgUmVzcG9uc2VzCgpBbnkgcmVzcG9uc2UgbWlkZGxld2FyZSBNVVNUIGZvbGxvdyB0aGUgYFJlcXVlc3RWYWxpZGF0aW9uYCBtaWRkbGV3YXJlIGFzIGl0IHJlcXVpcmVzIHRoZSBgcmVzdWx0YCBvYmplY3QgYmVpbmcgYWRkZWQgdG8KeW91ciBjb250YWluZXIuICAKVGhlc2UgbWlkZGxld2FyZXMgd2lsbCBjaGVjayB3aGV0aGVyIHRoZSByZXF1ZXN0IGhhcyBwYXNzZWQgb3IgZmFpbGVkIHZhbGlkYXRpb24uICAKSW52YWxpZCByZXF1ZXN0cyB3aWxsIHJldHVybiBhbiBhcHByb3ByaWF0ZSByZXNwb25zZSBkZXRhaWxpbmcgdGhlIHJlYXNvbnMgdGhlIHJlcXVlc3Qgd2FzIGludmFsaWQuCgpZb3VyIHJlc3BvbnNlIGNhbiBiZSBpbiBvbmUgb2YgdGhlIGZvbGxvd2luZyBmb3JtYXRzLgoKIyMjIyBGbGF0IEpzb24KCmBcTWVtYnJhbmVcTGFyYXZlbFxNaWRkbGV3YXJlXFJlc3BvbnNlSnNvbkZsYXRgCgpgYGBqc29uCnsKICAgICJlcnJvcnMiOnsKICAgICAgICAicGV0LT5pZCI6WyJtdXN0IGJlIGFuIGludGVnZXIiXSwKICAgICAgICAicGV0IjpbIm5hbWUgaXMgYSByZXF1aXJlZCBmaWVsZCJdCiAgICB9LAogICAgInRpdGxlIjoiUmVxdWVzdCBwYXlsb2FkIGZhaWxlZCB2YWxpZGF0aW9uIiwKICAgICJ0eXBlIjoiYWJvdXQ6YmxhbmsiLAogICAgInN0YXR1cyI6NDAwCn0KYGBgCgojIyMjIE5lc3RlZCBKc29uCgpgXE1lbWJyYW5lXExhcmF2ZWxcTWlkZGxld2FyZVxSZXNwb25zZUpzb25OZXN0ZWRgCgpgYGBqc29uCnsKICAgICJlcnJvcnMiOnsKICAgICAgICAiZXJyb3JzIjpbXSwKICAgICAgICAiZmllbGRzIjp7CiAgICAgICAgICAgICJwZXQiOnsKICAgICAgICAgICAgICAgICJlcnJvcnMiOlsKICAgICAgICAgICAgICAgICAgICAibmFtZSBpcyBhIHJlcXVpcmVkIGZpZWxkIgogICAgICAgICAgICAgICAgXSwKICAgICAgICAgICAgICAgICJmaWVsZHMiOnsKICAgICAgICAgICAgICAgICAgICAiaWQiOnsKICAgICAgICAgICAgICAgICAgICAgICAgImVycm9ycyI6WwogICAgICAgICAgICAgICAgICAgICAgICAgICAgIm11c3QgYmUgYW4gaW50ZWdlciIKICAgICAgICAgICAgICAgICAgICAgICAgXSwKICAgICAgICAgICAgICAgICAgICAgICAgImZpZWxkcyI6W10KICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9LAogICAgInRpdGxlIjoiUmVxdWVzdCBwYXlsb2FkIGZhaWxlZCB2YWxpZGF0aW9uIiwKICAgICJ0eXBlIjoiYWJvdXQ6YmxhbmsiLAogICAgInN0YXR1cyI6NDAwCn0KYGBgCgojIyMgR2xvYmFsIFVzYWdlCgpUbyB1c2UgYW55IG9mIHRoZSBhYm92ZSBtaWRkbGV3YXJlcyBvbiBhbGwgcm91dGVzLCBnbyBpbnRvIHlvdXIgYGFwcC9IdHRwL0tlcm5lbC5waHBgIGFuZCBhZGQgdGhlbSB0byB5b3VyIGBtaWRkbGV3YXJlYAphcnJheS4KCkZvciBleGFtcGxlOgoKYGBgcGhwCnByb3RlY3RlZCAkbWlkZGxld2FyZSA9IFsKICBcTWVtYnJhbmVcTGFyYXZlbFxNaWRkbGV3YXJlXFJlcXVlc3RWYWxpZGF0aW9uOjpjbGFzcywKICBcTWVtYnJhbmVcTGFyYXZlbFxNaWRkbGV3YXJlXFJlc3BvbnNlSnNvbkZsYXQ6OmNsYXNzCiAgLy8gLi4uCl07CmBgYAoKSW4gTGFyYXZlbCAxMS54LzEyLnggYGFwcC9IdHRwS2VybmVsLnBocGAgaGFzIGJlZW4gcmVtb3ZlZCwgZ2xvYmFsIGNvbmZpZ3VyYXRpb24gZ29lcyBpbnRvIGBib290c3RyYXAvYXBwLnBocGA6CgpGb3IgZXhhbXBsZToKCmBgYHBocApyZXR1cm4gQXBwbGljYXRpb246OmNvbmZpZ3VyZShiYXNlUGF0aDogZGlybmFtZShfX0RJUl9fKSkKICAgIC8vIC4uLgogICAgLT53aXRoTWlkZGxld2FyZShmdW5jdGlvbiAoTWlkZGxld2FyZSAkbWlkZGxld2FyZSkgewogICAgICAgIC8vIC4uLgogICAgICAgICRtaWRkbGV3YXJlLT5hcHBlbmRUb0dyb3VwKCdhcGknLCBbCiAgICAgICAgICAgIFxNZW1icmFuZVxMYXJhdmVsXE1pZGRsZXdhcmVcUmVxdWVzdFZhbGlkYXRpb246OmNsYXNzLAogICAgICAgICAgICBcTWVtYnJhbmVcTGFyYXZlbFxNaWRkbGV3YXJlXFJlc3BvbnNlSnNvbkZsYXQ6OmNsYXNzLAogICAgICAgIF0pOwogICAgfSkKICAgIC8vIC4uLgogICAgLT5jcmVhdGUoKTsKYGBgCg== readmeEtag: '"afc2f469137e6489d7c17a821221a64f0fda4b99"' readmeLastModified: Mon, 30 Jun 2025 13:28:15 GMT repositoryId: 574444174 description: Laravel integration for Membrane validation library. created: '2022-12-05T10:25:52Z' updated: '2025-07-14T07:43:12Z' language: PHP archived: false stars: 4 watchers: 2 forks: 3 owner: membrane-php logo: https://avatars.githubusercontent.com/u/104306108?v=4 license: NOASSERTION repoEtag: '"b5572804e61b396250a01ad7b33762d81b5437b102981d9db65c27e2ff708795"' repoLastModified: Mon, 14 Jul 2025 07:43:12 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/leo-lem/studentscrollapi v3: true id: 9d85e4b5c6f3f2586fc775819a7ac60f repositoryMetadata: base64Readme: >- IyBTdHVkZW50U2Nyb2xsIEFQSQoKWyFbQ0kvQ0RdKGh0dHBzOi8vZ2l0aHViLmNvbS9MZW8tTGVtL1N0dWRlbnRTY3JvbGxBUEkvYWN0aW9ucy93b3JrZmxvd3MvY2ljZC55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL0xlby1MZW0vU3R1ZGVudFNjcm9sbEFQSS9hY3Rpb25zL3dvcmtmbG93cy9jaWNkLnltbCkKClRoaXMgaXMgdGhlIFJFU1QgQVBJIGJlaGluZCB0aGUgU3R1ZGVudFNjcm9sbCBhcHAgKFtPcGVuQVBJIHNwZWNzXShodHRwczovL3N0dWRlbnRzY3JvbGwubmV0L2FwaS92MS9kb2NzL3N3YWdnZXIpKS4KCkhlcmUgd2UgcHJvdmlkZSBvdXIgW1N0dWRlbnRTY3JvbGwgd2ViIGFwcF0oaHR0cHM6Ly9naXRodWIuY29tL2xlby1sZW0vc3R1ZGVudHNjcm9sbGFwcCkgd2l0aCBpdHMgZnVuY3Rpb25hbGl0eS4KClsvYXBpL3YyXShodHRwczovL3N0dWRlbnRzY3JvbGwubmV0L2FwaS92Mi9kb2NzL3N3YWdnZXItdWkvaW5kZXguaHRtbCkgVGhlIFJFU1QgQVBJICh2MikgaXMgZGVzaWduZWQgdG8gYmUgYXMgc2ltcGxlIGFuZCBpbnR1aXRpdmUgYXMgcG9zc2libGUsIHdoaWxzdCBwcm92aWRpbmcgYWxsIHJlcXVpcmVkIGZ1bmN0aW9uYWxpdHk6CgotIFsvYWNjb3VudF0oaHR0cHM6Ly9zdHVkZW50c2Nyb2xsLm5ldC9hcGkvdjIvZG9jcy9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwjL2FjY291bnQpIFRoaXMgZW5kcG9pbnQgZGVzY3JpYmVzIG91ciBhY2NvdW50IGZlYXR1cmUuIEhlcmUgd2UgaGFuZGxlIGV2ZXJ5dGhpbmcgZnJvbSBzaWduIHVwLCB0aHJvdWdoIHNpZ24gaW4sIHRvIHVwZGF0aW5nIGNyZWRlbnRpYWxzIChmb3Jnb3R0ZW4gcGFzc3dvcmQsIGV0Yy4pIGFuZCBkZWxldGluZyB5b3VyIGFjY291bnQuCgogIC0gWy9zZXR0aW5nc10oaHR0cHM6Ly9zdHVkZW50c2Nyb2xsLm5ldC9hcGkvdjIvZG9jcy9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwjL3NldHRpbmdzKSBGdXJ0aGVybW9yZSwgeW91IGNhbiBpbmRpdmlkdWFsaXNlIHlvdXIgYWNjb3VudCBieSB1cGRhdGluZyB5b3VyIHNldHRpbmdzLgoKLSBbL3N0dWRlbnRzXShodHRwczovL3N0dWRlbnRzY3JvbGwubmV0L2FwaS92Mi9kb2NzL3N3YWdnZXItdWkvaW5kZXguaHRtbCMvc3R1ZGVudHMpIFRoaXMgbWFqb3IgZW5kcG9pbnQgZGVzY3JpYmVzIG91ciBwcm9maWxlIGZlYXR1cmUuIEhlcmUgeW91IGNhbiB1cGRhdGUgeW91ciBpbmRpdmlkdWFsIHByb2ZpbGUsIHNlYXJjaCBmb3Igb3RoZXIgc3R1ZGVudCdzIHByb2ZpbGVzLgoKICAtIFsvZm9sbG93ZXJzXShodHRwczovL3N0dWRlbnRzY3JvbGwubmV0L2FwaS92Mi9kb2NzL3N3YWdnZXItdWkvaW5kZXguaHRtbCMvc3R1ZGVudHMpIFdlIGhhdmUgYWxzbyBlbWJlZGRlZCB0aGUgZm9sbG93aW5nIG1lY2hhbmljIGJlbmVhdGggdGhpcyBlbmRwb2ludC4gSGVyZSB5b3UgY2FuIGZvbGxvdyBhbmQgdW5mb2xsb3cgb3RoZXIgc3R1ZGVudHMuCgotIFsvcG9zdHNdKGh0dHBzOi8vc3R1ZGVudHNjcm9sbC5uZXQvYXBpL3YyL2RvY3Mvc3dhZ2dlci11aS9pbmRleC5odG1sIy9wb3N0cykgT24gdGhlIHBvc3RzIGVuZHBvaW50LCB5b3UgY2FuLCB3ZWxsLCBwb3N0IHdoYXQncyBvbiB5b3VyIG1pbmQuIEFMc28sIHlvdSBjYW4gc2VhcmNoIGZvciBwb3N0cyBhbmQgdXBkYXRlLCBvciBkZWxldGUsIHlvdXIgcG9zdHMuCgotIFsvY2hhdHNdKGh0dHBzOi8vc3R1ZGVudHNjcm9sbC5uZXQvYXBpL3YyL2RvY3Mvc3dhZ2dlci11aS9pbmRleC5odG1sIy9jaGF0cykgT3VyIGNoYXRzIGZlYXR1cmUgZW5hYmxlcyB5b3UgdG8gY3JlYXRlLCBmaW5kLCBhbmQgZGVsZXRlIGNoYXRzLgoKICAtIFsvbWVzc2FnZXNdKGh0dHBzOi8vc3R1ZGVudHNjcm9sbC5uZXQvYXBpL3YyL2RvY3Mvc3dhZ2dlci11aS9pbmRleC5odG1sIy9tZXNzYWdlcykgVGhlIGVtYmVkZGVkIG1lc3NhZ2VzIGVuZHBvaW50IGNvbnRhaW5zIHRoZSBtZXNzYWdlcyBpbiBhIGdpdmVuIGNoYXQuIEhlcmUgeW91IGNhbiBzZW5kIG1lc3NhZ2VzLCB1cGRhdGUgb3IgZGVsZXRlIHRoZW0sIGFzIHdlbGwgYXMgZmluZCBhIGdpdmVuIG1lc3NhZ2UuCgotIFsvbWFwc10oaHR0cHM6Ly9zdHVkZW50c2Nyb2xsLm5ldC9hcGkvdjIvZG9jcy9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwjL21hcHMpIFRoaXMgbWlub3IgZW5kcG9pbnQgcmV0cmlldmVzIHRoZSBHb29nbGUgTWFwcyBBUEkga2V5IGZyb20gb3VyIGJhY2tlbmQuCgojIEJ1aWxkaW5nLCB0ZXN0aW5nLCBhbmQgcnVubmluZyB0aGUgU3R1ZGVudFNjcm9sbEFQSQoKVGhlIFN0dWRlbnRTY3JvbGxBUEkgdXNlcyBNYXZlbiBhcyBkZXBlbmRlbmN5IG1hbmFnZXIgYW5kIGJ1aWxkIHRvb2wuCgpTaW5jZSB0aGUgYmVnaW5uaW5nLCB3ZSBoYXZlIHVwZGF0ZWQgdGhlIHdheSB3ZSBydW4gdGhpbmdzLiBXZSBub3cgaW5jbHVkZSBhIE1hdmVuIHdyYXBwZXIuIFdpdGggdGhpcyB5b3UgY2FuIHRyaWdnZXIgYW55IG9mIHRoZSBbTWF2ZW4gbGlmZWN5Y2xlIHBoYXNlc10oaHR0cHM6Ly9tYXZlbi5hcGFjaGUub3JnL2d1aWRlcy9pbnRyb2R1Y3Rpb24vaW50cm9kdWN0aW9uLXRvLXRoZS1saWZlY3ljbGUuaHRtbCkgYnkgcnVubmluZyB0aGUgZm9sbG93aW5nIGZyb20gdGhlIHJlcG9zaXRvcnkgcm9vdDoKCmBgYGJhc2gKPiAlIC4vbXZudyBbY29tcGlsZSB8IHRlc3QgfCBwYWNrYWdlIHwgaW5zdGFsbCB8IC4uLl0gIyBVbml4CiMgb3IKPiAlIC4vbXZudy5jbWQgW2NvbXBpbGUgfCB0ZXN0IHwgcGFja2FnZSB8IGluc3RhbGwgfCAuLi5dICMgV2luZG93cwpgYGAKCkFkZGl0aW9uYWxseSwgU3ByaW5nIEJvb3QgcHJvdmlkZXMgYSBjdXN0b20gcnVuIGNvbW1hbmQgZm9yIHRyeWluZyB0aGUgcHJvamVjdCBvdXQgb24geW91ciBsb2NhbCBtYWNoaW5lOgoKYGBgYmFzaAo+ICUgLi9tdm53IHNwcmluZy1ib290OnJ1biAjIFVuaXgKIyBvcgo+ICUgLi9tdm53LmNtZCBzcHJpbmctYm9vdDpydW4gIyBXaW5kb3dzCmBgYAo= readmeEtag: '"883d57535f65fcde1a97f6fce6bdaac3b5b9d6ae"' readmeLastModified: Sat, 03 Jun 2023 23:20:48 GMT repositoryId: 613641216 description: This is the REST API behind StudentScroll. created: '2023-03-14T01:14:36Z' updated: '2025-12-15T18:20:54Z' language: Java archived: true stars: 7 watchers: 2 forks: 0 owner: leo-lem logo: https://avatars.githubusercontent.com/u/81310398?v=4 license: MIT repoEtag: '"c8c46a087a96af7eaed437ad127e39202a06392458cd7fb900b166beb2de23e1"' repoLastModified: Mon, 15 Dec 2025 18:20:54 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/suud/cdk-openapigateway v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpZ2F0ZXdheQo+IENyZWF0ZSBhbiBBbWF6b24gQVBJIEdhdGV3YXkgZnJvbSBhbiBPcGVuQVBJIDMgRG9jdW1lbnQuCgohW1B5UEldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcHlwaS92L29wZW5hcGlnYXRld2F5KQohW1B5UEkgLSBQeXRob24gVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3B5dmVyc2lvbnMvb3BlbmFwaWdhdGV3YXkpClshW0NvZGUgc3R5bGU6IGJsYWNrXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NvZGUlMjBzdHlsZS1ibGFjay0wMDAwMDAuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3BzZi9ibGFjaykKIVtQeVBJIC0gTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL2wvb3BlbmFwaWdhdGV3YXkpCgpBV1MgQ0RLIENvbnN0cnVjdCB0aGF0IGNyZWF0ZXMgYW4gQW1hem9uIEFQSSBHYXRld2F5IEh0dHBBcGkgYmFzZWQgb24gYQpwYXJhbWV0ZXJpemVkIE9wZW5BUEkgMyBEb2N1bWVudC4KCiMjIEluc3RhbGxhdGlvbgoKYGBgc2gKcGlwIGluc3RhbGwgb3BlbmFwaWdhdGV3YXkKYGBgCgojIyBVc2FnZQoKIyMjIEV4YW1wbGUgMTogQVBJIGJhY2tlZCBieSBMYW1iZGEgRnVuY3Rpb24KCipvcGVuYXBpLnlhbWw6KgpgYGB5YW1sClsuLi5dCnBhdGhzOgogIC9wZXRzOgogICAgZ2V0OgogICAgICBzdW1tYXJ5OiBMaXN0IGFsbCBwZXRzCiAgICAgIHJlc3BvbnNlczoKICAgICAgICBbLi4uXQogICAgICB4LWFtYXpvbi1hcGlnYXRld2F5LWludGVncmF0aW9uOgogICAgICAgIHVyaTogIiR7QVBJX0xBTUJEQV9BUk59IgogICAgICAgIHR5cGU6ICJBV1NfUFJPWFkiCiAgICAgICAgaHR0cE1ldGhvZDogIlBPU1QiCiAgICAgICAgY29ubmVjdGlvblR5cGU6ICJJTlRFUk5FVCIKICAgICAgICBwYXlsb2FkRm9ybWF0VmVyc2lvbjogIjIuMCIKICAgICAgeC1hbWF6b24tYXBpZ2F0ZXdheS1yZXF1ZXN0LXZhbGlkYXRvcjoKICAgICAgICB2YWxpZGF0ZVJlcXVlc3RCb2R5OiB0cnVlCiAgICAgICAgdmFsaWRhdGVSZXF1ZXN0UGFyYW1ldGVyczogdHJ1ZQpbLi4uXQpgYGAKCipvcGVuX2FwaV9zdGFjay5weToqCmBgYHB5dGhvbgpmcm9tIGF3c19jZGsgaW1wb3J0IGNvcmUsIGF3c19pYW0gYXMgaWFtLCBhd3NfbGFtYmRhIGFzIF9sYW1iZGEKZnJvbSBvcGVuYXBpZ2F0ZXdheSBpbXBvcnQgT3BlbkFwaUdhdGV3YXkKCgpjbGFzcyBPcGVuQXBpU3RhY2soY29yZS5TdGFjayk6CiAgICBkZWYgX19pbml0X18oCiAgICAgICAgc2VsZiwgc2NvcGU6IGNvcmUuQ29uc3RydWN0LCBjb25zdHJ1Y3RfaWQ6IHN0ciwgKiprd2FyZ3MKICAgICkgLT4gTm9uZToKICAgICAgICBzdXBlcigpLl9faW5pdF9fKHNjb3BlLCBjb25zdHJ1Y3RfaWQsICoqa3dhcmdzKQoKICAgICAgICAjIGZ1bmN0aW9uIHRoYXQgaGFuZGxlcyBhcGkgcmVxdWVzdChzKQogICAgICAgIGFwaV9sYW1iZGEgPSBfbGFtYmRhLkZ1bmN0aW9uKFsuLi5dKQoKICAgICAgICAjIGNyZWF0ZSBhcGkgZnJvbSBvcGVuYXBpIGRvY3VtZW50IGFuZCByZXBsYWNlIHBhcmFtcwogICAgICAgIG9wZW5hcGkgPSBPcGVuQXBpR2F0ZXdheSgKICAgICAgICAgICAgc2VsZiwKICAgICAgICAgICAgIk9wZW5BUEkgR2F0ZXdheSIsCiAgICAgICAgICAgIG9wZW5hcGlfcGF0aD0ib3BlbmFwaS55YW1sIiwKICAgICAgICAgICAgcGFyYW1fdmFsdWVfZGljdD17IkFQSV9MQU1CREFfQVJOIjogYXBpX2xhbWJkYS5mdW5jdGlvbl9hcm59LAogICAgICAgICAgICBmYWlsX29uX3dhcm5pbmdzPVRydWUsCiAgICAgICAgKQoKICAgICAgICAjIGdyYW50IEh0dHBBcGkgcGVybWlzc2lvbiB0byBpbnZva2UgYXBpIGxhbWJkYSBmdW5jdGlvbgogICAgICAgIGFwaV9sYW1iZGEuYWRkX3Blcm1pc3Npb24oCiAgICAgICAgICAgIGYiSW52b2tlIEJ5IHtvcGVuYXBpLmh0dHBfYXBpLm5vZGUuaWR9IFBlcm1pc3Npb24iLAogICAgICAgICAgICBwcmluY2lwYWw9aWFtLlNlcnZpY2VQcmluY2lwYWwoImFwaWdhdGV3YXkuYW1hem9uYXdzLmNvbSIpLAogICAgICAgICAgICBhY3Rpb249ImxhbWJkYTpJbnZva2VGdW5jdGlvbiIsCiAgICAgICAgICAgIHNvdXJjZV9hcm49b3BlbmFwaS5odHRwX2FwaV9hcm4sCiAgICAgICAgKQpgYGAKCiMjIFtEb2N1bWVudGF0aW9uXShodHRwczovL3N1dWQuZ2l0aHViLmlvL2Nkay1vcGVuYXBpZ2F0ZXdheS9vcGVuYXBpZ2F0ZXdheS5odG1sKQoKIyMgRGV2ZWxvcG1lbnQgc2V0dXAKCiMjIyBvcHRpb25hbDogdXNlIHZpcnR1YWxlbnYKCmBgYHNoCiMgY3JlYXRlIHZpcnR1YWxlbnYgb24gTWFjT1MgYW5kIExpbnV4CnB5dGhvbjMgLW0gdmVudiAudmVudgojIGFjdGl2YXRlIHZpcnR1YWxlbnYKc291cmNlIC52ZW52L2Jpbi9hY3RpdmF0ZQpgYGAKCiMjIyBpbnN0YWxsIGRlcGVuZGVuY2llcwoKVG8gaW5zdGFsbCB0aGlzIHBhY2thZ2UsIGFsb25nIHdpdGggdGhlIHRvb2xzIHlvdSBuZWVkIHRvIGRldmVsb3AgYW5kIHB1Ymxpc2gKaXQsIHJ1biB0aGUgZm9sbG93aW5nOgoKYGBgc2gKcGlwIGluc3RhbGwgLWUgIi5bZGV2XSIKYGBgCgojIyBDb250cmlidXRpbmcKCjEuIFtGb3JrIHRoaXMgcmVwb3NpdG9yeV0oaHR0cHM6Ly9naXRodWIuY29tL3N1dWQvY2RrLW9wZW5hcGlnYXRld2F5L2ZvcmspCjIuIENyZWF0ZSB5b3VyIGZlYXR1cmUgYnJhbmNoIChgZ2l0IGNoZWNrb3V0IC1iIGZlYXR1cmUvZm9vQmFyYCkKMy4gQ29tbWl0IHlvdXIgY2hhbmdlcyAoYGdpdCBjb21taXQgLWFtICdBZGQgc29tZSBmb29CYXInYCkKNC4gUHVzaCB0byB0aGUgYnJhbmNoIChgZ2l0IHB1c2ggb3JpZ2luIGZlYXR1cmUvZm9vQmFyYCkKNS4gQ3JlYXRlIGEgbmV3IFB1bGwgUmVxdWVzdAoKIyMgTGljZW5zZQoKTUlUCg== readmeEtag: '"59392f141d23ea5cd395744d56522cd550df646d"' readmeLastModified: Tue, 29 Dec 2020 19:49:29 GMT repositoryId: 324788192 description: AWS CDK - API Gateway from OpenAPI 3 Specs created: '2020-12-27T15:15:15Z' updated: '2022-09-08T21:22:14Z' language: Python archived: false stars: 3 watchers: 1 forks: 2 owner: suud logo: https://avatars.githubusercontent.com/u/10183975?v=4 license: MIT repoEtag: '"37e0aced9464d135af44bc3ec751153915f96462f75ac40b861eac864e69f743"' repoLastModified: Thu, 08 Sep 2022 21:22:14 GMT foundInMaster: true category: Data Validators id: 1aba64373963ca20f37c64c9a809c53d - source: openapi3 tags repository: https://github.com/adarro/parsehub-openapi v3: true repositoryMetadata: base64Readme: >- IyBwYXJzZWh1Yi1vcGVuYXBpClN3YWdnZXIgKE9wZW5BUEkpIGRvY3VtZW50YXRpb24gZm9yIFBhcnNlSHViIFJFU1QgQVBJCgpUaGlzIGlzIGFuIGltcGxlbWVudGF0aW9uIG9mIGEgU3dhZ2dlciAoT3BlbkFQSSAzKSBkb2N1bWVudCBvZiB0aGUgW1BhcnNlSHViXShodHRwczovL3BhcnNlaHViLmNvbSkgUkVTVCBhcGkuCgpBdCBjdXJyZW50LCB0aGUgbWFzdGVyIGRvY3VtZW50IGxpdmVzIG9uIFN3YWdnZXJIdWIgW0hFUkVdKGh0dHBzOi8vYXBwLnN3YWdnZXJodWIuY29tL2FwaXMvYWRhcnJvL3BhcnNlLWh1Yi8wLjAuMSkKCkdpdGh1YiAvIEdpdGxhYiBpbnRlZ3JhdGlvbiB3aXRoIFN3YWdnZXJIdWIgZG9lcyBub3Qgd29yayBmb3IgT3BlbkFQSSAzLjAgYXQgdGhpcyB0aW1lLCBidXQgKnNob3VsZCogaW4gdGhlIG5lYXIgZnV0dXJlLgoKIyMgVmVyc2lvbmluZwpJIHdvdWxkIGxpa2UgdG8gZm9sbG93IFNlbWFudGljIFZlcnNpb25pbmcgW1NlbVZlcl0oaHR0cHM6Ly9zZW12ZXIub3JnLykgYnV0IHdvdWxkIGxpa2UgdG8ga2VlcCB0aGlzIGluIHN5bmMgd2l0aCBhbnkgUGFyc2VodWIgY2hhbmdlcy4KClRoaXMgcHJvamVjdCBpcyBBcGFjaGUgMi4wIExpY2Vuc2VkLCBidXQgdXNlIG9mIFBhcnNlSHViJ3Mgc2VydmljZXMgcmVxdWlyZXMgYWdyZWVtZW50IHRvIHRoZWlyIFtUZXJtcyBvZiBTZXJ2aWNlXShodHRwczovL3d3dy5wYXJzZWh1Yi5jb20vdGVybXMp readmeEtag: '"59d57bff96d8840d5388162325f16c9fcd2767fe"' readmeLastModified: Thu, 06 Feb 2020 14:08:17 GMT repositoryId: 139917432 description: Swagger (OpenAPI) documentation for ParseHub REST API created: '2018-07-06T01:19:24Z' updated: '2023-07-26T08:05:24Z' language: null archived: false stars: 3 watchers: 1 forks: 0 owner: adarro logo: https://avatars.githubusercontent.com/u/101220?v=4 license: Apache-2.0 repoEtag: '"c3d62f680202c56930c2633e291542448f75d1256b85b2fd9ccaf0ebebd66269"' repoLastModified: Wed, 26 Jul 2023 08:05:24 GMT foundInMaster: true category: Parsers id: 4febdf85e579394614b5e30de52823ba - source: openapi3 tags repository: https://github.com/bump-sh-examples/laravel-design-first v3: true id: 22a90a57d3f8d2cbe22c7514cc808a0c repositoryMetadata: base64Readme: >- IyBMYXJhdmVsIENvZGUgRmlyc3QKClRoaXMgcmVwb3NpdG9yeSB3YXMgYnVpbHQgYXMgc2FtcGxlIGNvZGUgZm9yIHRoZSBCdW1wLnNoIGd1aWRlIG9uIF9bVXNpbmcgT3BlbkFQSSB0byBzaW1wbGlmeSBidWlsZGluZyBhbmQgdGVzdGluZyBMYXJhdmVsIEFQSXNdKGh0dHBzOi8vZG9jcy5idW1wLnNoL2d1aWRlcy9vcGVuYXBpL2Rlc2lnbi1maXJzdC1sYXJhdmVsLylfLiBMZWFybiB0byB1c2UgdGhlIEFQSSBEZXNpZ24gRmlyc3Qgd29ya2Zsb3csIGFuZCBzaW1wbGlmeSB5b3VyIExhcmF2ZWwgY29kZSBieSBub3QgaGF2aW5nIHRvIHJlcGVhdCB5b3VyIEFQSSBjb250cmFjdCBpbiB2YWxpZGF0aW9uIGFuZCBjb250cmFjdCB0ZXN0aW5nIGFzIHdlbGwgYXMgZG9jdW1lbnRhdGlvbi4gSnVzdCBkbyBpdCBvbmNlLCBtYWtlIGVhc3kgY29kZSwgdGhlbiBwb3AgaXQgYWxsIG9udG8gQnVtcC5zaCB0byBoYXZlIFtncmVhdCBBUEkgZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9idW1wLnNoL2J1bXAtZXhhbXBsZXMvaHViL2NvZGUtc2FtcGxlcy9kb2MvbGFyYXZlbC1kZXNpZ24tZmlyc3QpLgoKIyMgVXNhZ2UKCkNsb25lIHRoZSByZXBvc2l0b3J5IGRvd24gdG8gZ2l2ZSBpdCBhIHRyeS4KCmBgYAojIFNldCBldmVyeXRoaW5nIHVwCiQgY29tcG9zZXIgaW5zdGFsbAoKIyBTdGFydCB0aGUgc2VydmVyCiQgcGhwIGFydGlzYW4gc2VydmUKCiMgUG9rZSB0aGUgQVBJIGFuZCBnZXQgYXV0b21hdGVkICBlcnJvcnMKJCBjdXJsIC1YIFBPU1QgaHR0cDovL2xvY2FsaG9zdDo4MDAwL2FwaS93aWRnZXRzIFwKICAgIC1IICJDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24iIFwKICAgIC1kICd7Im5hbWUiIDogIlJlcGxpY2F0b3IifScKCnsKICAiZXJyb3JzIjogewogICAgInJlcXVlc3RCb2R5IjogWwogICAgICAiZGVzY3JpcHRpb24gaXMgYSByZXF1aXJlZCBmaWVsZCIKICAgIF0KICB9LAogICJ0aXRsZSI6ICJSZXF1ZXN0IHBheWxvYWQgZmFpbGVkIHZhbGlkYXRpb24iLAogICJ0eXBlIjogImFib3V0OmJsYW5rIiwKICAic3RhdHVzIjogNDAwCn0KYGBgCgpHaXZlIGl0IGEgdHJ5LCBwbGF5IGFyb3VuZCB3aXRoIHRoZSBPcGVuQVBJLCBhbmQgc2VlIGhvdyBpdCByZXNwb25kcyB0byBkaWZmZXJlbnQgc2NlbmFyaW9zLiAKClRoZW4geW91IGNhbiBydW4gYGNvbXBvc2VyIHRlc3RgIHRvIHNlZSBpZiB0aGUgQVBJIHJlc3BvbnNlcyBtYXRjaCB3aGF0IE9wZW5BUEkgZXhwZWN0LCB3aGljaCB3aGVuIGltcGxlbWVudGVkIGluIHlvdXIgYXBwbGljYXRpb24gd2lsbCBoZWxwIG1ha2Ugc3VyZSB5b3VyIEFQSSBpcyBhY3R1YWxseSBkb2luZyB3aGF0IHlvdXIgZG9jcyBhcmUgc2F5aW5nLCBvciBtYWtlIHN1cmUgeW91ciBkb2NzIGFyZSBzYXlpbmcgd2hhdCB5b3VyIEFQSSBpcyBkb2luZywgd2hpY2hldmVyIHdheSByb3VuZCB5b3UgcHJlZmVyIHRvIHRoaW5rIG9mIGl0LgoKUHJldmlldyBob3cgdGhlIEFQSSByZWZlcmVuY2UgZG9jcyBsb29rIFtvbiBCdW1wLnNoXShodHRwczovL2J1bXAuc2gvYnVtcC1leGFtcGxlcy9odWIvY29kZS1zYW1wbGVzL2RvYy9sYXJhdmVsLWRlc2lnbi1maXJzdCkuCgojIyBMaWNlbnNlCgpUaGUgY29udGVudHMgb2YgdGhpcyByZXBvc2l0b3J5IGFyZSBsaWNlbnNlZCB1bmRlciBbQ0MgQlktTkMtU0EKICA0LjBdKC4vTElDRU5TRV9DQy1CWS1OQy1TQS00LjApLgo= readmeEtag: '"7ee963ec709358519fd61bec17566d66a16b2177"' readmeLastModified: Tue, 27 Feb 2024 16:05:25 GMT repositoryId: 743755264 description: >- Create your OpenAPI then use it to power your application, covering both request validation and contract testing. created: '2024-01-15T23:12:27Z' updated: '2025-03-12T20:31:31Z' language: PHP archived: false stars: 4 watchers: 1 forks: 0 owner: bump-sh-examples logo: https://avatars.githubusercontent.com/u/157144805?v=4 license: NOASSERTION repoEtag: '"f8fce7e0f7eb94ebd295376350dbdd5c4ea192e588fff7cfc63658084bfb31c6"' repoLastModified: Wed, 12 Mar 2025 20:31:31 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/sudokuru v3: true id: 88726e2bfb53ac81a29d3f16ff3a6a14 repositoryMetadata: base64Readme: >- ![Sudokuru Logo](https://sudokuru.s3.amazonaws.com/goldLogoText.png)
# Sudoku Puzzle Generator
[![NPM version](https://img.shields.io/npm/v/sudokuru.svg?style=flat)](https://npmjs.org/package/sudokuru)
[![NPM downloads](https://img.shields.io/npm/dm/sudokuru.svg?style=flat)](https://npmjs.org/package/sudokuru)
![NPM License](https://img.shields.io/npm/l/sudokuru)
[![Pipeline](https://github.com/SudoKuru/Sudokuru/actions/workflows/pipeline.yml/badge.svg)](https://github.com/SudoKuru/Sudokuru/actions/workflows/pipeline.yml)
[![Coveralls Coverage](https://coveralls.io/repos/github/SudoKuru/Sudokuru/badge.svg?branch=main)](https://coveralls.io/github/SudoKuru/Sudokuru?branch=main)
[![Codecov Coverage](https://codecov.io/gh/SudoKuru/Sudokuru/graph/badge.svg?token=SISS85TX20)](https://codecov.io/gh/SudoKuru/Sudokuru)

[![Common Changelog](https://common-changelog.org/badge.svg)](https://common-changelog.org)


> Generate data about Sudoku puzzles

> [!NOTE]  
> This project uses [BreakVer](https://www.taoensso.com/break-versioning) for project versioning. 

# Table of Contents

*   [Installation](#installation)
*   [Usage](#usage)
    *   [JavaScript](#javascript)
        *   [Hint File](#hint-file)
            *   [getHint()](#gethint)
        *   [PuzzleData File](#puzzledata-file)
            *   [getPuzzleData()](#getpuzzledata)
        *   [Drill File](#drill-file)
            *   [getDrillPuzzleString()](#getdrillpuzzlestring)
*   [hint Object Properties](#hint-object-properties)
    *   [Example 1: Amend Notes](#example-1-amend-notes)
        *   [strategy](#strategy)
        *   [cause](#cause)
        *   [groups](#groups)
        *   [placements](#placements)
        *   [removals](#removals)
        *   [info](#info)
        *   [action](#action)
    *   [Example 2: Simplify Notes](#example-2-simplify-notes)
        *   [strategy](#strategy-1)
        *   [cause](#cause-1)
        *   [groups](#groups-1)
        *   [placements](#placements-1)
        *   [removals](#removals-1)
        *   [info](#info-1)
        *   [action](#action-1)
    *   [Example 3: Obvious Single](#example-3-obvious-single)
        *   [strategy](#strategy-2)
        *   [cause](#cause-2)
        *   [groups](#groups-2)
        *   [placements](#placements-2)
        *   [removals](#removals-2)
        *   [info](#info-2)
        *   [action](#action-2)
    *   [Example 4: Obvious Pair](#example-4-obvious-pair)
        *   [strategy](#strategy-3)
        *   [cause](#cause-3)
        *   [groups](#groups-3)
        *   [placements](#placements-3)
        *   [removals](#removals-3)
        *   [info](#info-3)
        *   [action](#action-3)
    *   [Example 5: Other Obvious Sets](#example-5-other-obvious-sets)
        *   [strategy](#strategy-4)
        *   [strategy](#strategy-5)
    *   [Example 6: Hidden Single](#example-6-hidden-single)
        *   [strategy](#strategy-6)
        *   [cause](#cause-4)
        *   [groups](#groups-4)
        *   [placements](#placements-4)
        *   [removals](#removals-4)
        *   [info](#info-4)
        *   [action](#action-4)
    *   [Example 7: Hidden Pair](#example-7-hidden-pair)
        *   [strategy](#strategy-7)
        *   [cause](#cause-5)
        *   [groups](#groups-5)
        *   [placements](#placements-5)
        *   [removals](#removals-5)
        *   [info](#info-5)
        *   [action](#action-5)
    *   [Example 8: Other Hidden Sets](#example-8-other-hidden-sets)
        *   [strategy](#strategy-8)
        *   [strategy](#strategy-9)
    *   [Example 9: Pointing Pair](#example-9-pointing-pair)
        *   [strategy](#strategy-10)
        *   [cause](#cause-6)
        *   [groups](#groups-6)
        *   [placements](#placements-6)
        *   [removals](#removals-6)
        *   [info](#info-6)
        *   [action](#action-6)
*   [PuzzleData Object Properties](#puzzledata-object-properties)
    *   [solution](#solution)
    *   [difficulty](#difficulty)
*   [Developer Tools](#developer-tools)

# Installation

```shell
npm install sudokuru
```

# Usage

## JavaScript

### Hint File

#### getHint()
1. Setup: 
    ```typescript
    import {getHint} from 'sudokuru';
    ```
2. Description: Returns a hint based on the puzzle and notes provided
3. Syntax
    ```typescript
    getHint(board, notes, strategies, solution);
    ```
4. Parameters:
    - board: 2d board array (9 arrays, one for each row, each with 9 strings representing values or "0" if empty)
    - notes: 2d notes array (81 arrays, one for each cell containing each note that is left in it)
    - strategies: optional parameter specifying which strategies are allowed to be used in the hint given in order of precedence (allowed values are the strings in the StrategyEnum in Sudoku.ts)
    - solution: optional parameter specifying boards solution so that amend notes hints can correct users mistakes
5. Return Value: [hint](#hint-object-properties)

### PuzzleData File

#### getPuzzleData()
1. Setup: 
    ```typescript
    import {getPuzzleData} from 'sudokuru';
    ```
2. Description: Returns data about a given puzzle
3. Syntax
    ```typescript
    getPuzzleData(board);
    ```
4. Parameters:
    - board: puzzle board string (81 characters, one for each cell containing each value or "0" if empty, left to right top to bottom)
5. Return Value: [puzzleData](#puzzledata-object-properties)

### Drill File

#### getDrillPuzzleString()
1. Setup:
    ```typescript
    import {getDrillPuzzleString} from 'sudokuru';
    ```
2. Description: Returns puzzle string updated to right before drill strategy move can be played
3. Syntax
    ```typescript
    getDrillPuzzleString(puzzleString, moveNumber);
    ```
4. Parameters:
    - puzzleString: puzzle board string (81 characters, one for each cell containing each value or "0" if empty, left to right top to bottom)
    - moveNumber: number of cells filled in (so index 30 has 30 cells filled in including givens), returned by getPuzzleData
5. Return Value: puzzle board string (81 characters, one for each cell containing each value or "0" if empty, left to right top to bottom)

# hint Object Properties
## Example 1: Amend Notes
![Example 1](https://sudokuru.s3.amazonaws.com/hintExample1-V2.png)
### strategy
```json
"AMEND_NOTES"
```
Name of strategy used by the hint. Amend notes works removing all notes from a cell then adding back in just the ones that don't conflict with placed values in the same row, column, and box as the cell. In the picture the cell with the red border was previously empty and amend notes was able to add in the 2 and 8 only as the others conflicted with the highlighted cells.
### cause
```json
[[7,4],[7,5],[7,6],[7,8],[4,2],[8,0],[8,1]]
```
Coordinates of cells that "cause" strategy to be applicable. This is a 2d array so cause[0] is [7, 4] referring to the cell with the 9 highlighted in blue in the next to last row (rows and columns are zero-indexed). cause[0][0] refers to it being in the 8th row and cause[0][1] refers to it being in the 5th column. Similarly the last cell cause[6] is the highlighted 3 that it shares a box with.
### groups
```json
[[0,7],[1,2],[2,6]]
```
Group type and index of groups that "cause strategy". This is a 2d array so group[0] is [0, 7] referring to the next to next to last row with green borders. group[0][0] refers to the group type (0 = row, 1 = column, 2 = box) and group[0][1] refers to it being the 8th row. Similarly the last group group[2] is the bottom left box whose cells also have green borders.
### placements
```json
[]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This does not apply to this strategy.
### removals
```json
[[7,2,1,3,4,5,6,7,9]]
```
Notes that can be removed from cells along with their row and columns. The first two values represent the row and column respectively so removals[0][0] and removals[0][1] refer to the 3rd cell in the 8th row. The rest of the numbers in the subarray are the notes to be removed which is every number but 2 and 8. Amend notes is unique in that every note that shouldn't be removed should be added back in (2 and 8 in this example). In this example the cell had no notes in it before amend notes.
### info
```json
"Amend notes are when you reset a cell's notes to contain every nonconflicting number"
```
Info about the strategy being used by the hint.
### action
```json
"When you see an amend notes you can remove all notes then add all nonconflicting numbers to its notes"
```
Describes the action that the hint is suggesting.
## Example 2: Simplify Notes
![Example 2](https://sudokuru.s3.amazonaws.com/hintExample2-V2.png)
### strategy
```json
"SIMPLIFY_NOTES"
```
Name of strategy used by the hint. Simplify notes works by using a cell with a placed value (the blue highlighted 8) to remove a note of the same value from a cell sharing a group (the red 8 note in a cell in the same column).
### cause
```json
[[7,0]]
```
Coordinates of cells that "cause" strategy to be applicable. This is a 2d array so cause[0] is [7, 0] referring to the cell with the 8 highlighted in blue in the next to last row (rows and columns are zero-indexed). cause[0][0] refers to it being in the 8th row and cause[0][1] refers to it being in the 1st column.
### groups
```json
[[1,0]]
```
Group type and index of groups that "cause strategy". This is a 2d array so group[0] is [1, 0] referring to the first column with green borders. group[0][0] refers to the group type (0 = row, 1 = column, 2 = box) and group[0][1] refers to it being the 1st column.
### placements
```json
[]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This does not apply to this strategy.
### removals
```json
[[1,0,8]]
```
Notes that can be removed from cells along with their row and columns. The first two values represent the row and column respectively so removals[0][0] and removals[0][1] refer to the 1st cell in the 2nd row. The rest of the numbers in the subarray are the notes to be removed which is just 8.
### info
```json
"You can simplify notes using values already placed in cells at the start of the game"
```
Info about the strategy being used by the hint.
### action
```json
"When there is a value already placed in a cell than it can be removed from all other cells notes in its row, column, and box"
```
Describes the action that the hint is suggesting.
## Example 3: Obvious Single
![Example 3](https://sudokuru.s3.amazonaws.com/hintExample3-V1.png)
### strategy
```json
"OBVIOUS_SINGLE"
```
Name of strategy used by the hint. Obvious single works by placing a value in a cell where it is the only remaining possibility. In the example the green highlighted 2 is the last remaining possibility in the blue highlighted cell.
### cause
```json
[[7,2]]
```
Coordinates of cells that "cause" strategy to be applicable. This is a 2d array so cause[0] is [7, 2] referring to the 3rd cell in the next to last row (rows and columns are zero-indexed). cause[0][0] refers to it being in the 8th row and cause[0][1] refers to it being in the 3rd column.
### groups
```json
[]
```
Group type and index of groups that "cause strategy". This does not apply to this strategy.
### placements
```json
[[7,2,2]]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This is a 2d array so placements[0] refers to the first value that needs to be placed. The first value in the subarray refers to the cells row, the next value is the column, and the final value is the value to be placed ("2" in this case).
### removals
```json
[]
```
Notes that can be removed from cells along with their row and columns. This does not apply to this strategy.
### info
```json
"Obvious singles are when you only have one number left as a possibility in a cell"
```
Info about the strategy being used by the hint.
### action
```json
"When you see a obvious single you can fill it in with its last remaining possibility"
```
Describes the action that the hint is suggesting.
## Example 4: Obvious Pair
![Example 4](https://sudokuru.s3.amazonaws.com/hintExample4-V1.png)
### strategy
```json
"OBVIOUS_PAIR"
```
Name of strategy used by the hint. Obvious pair works using two cells that share a group and only have the same two notes left as possibilities to eliminate those notes from other cells in the group. In the example 7 and 8 are the last remaining possibilities in the blue highlighted cells which are used to remove all the other 7 and 8s in the column shown highlighted in red.
### cause
```json
[[1,8],[6,8]]
```
Coordinates of cells that "cause" strategy to be applicable. This is a 2d array so cause[0] is [1, 8] referring to the 9th cell in the second row (rows and columns are zero-indexed). cause[0][0] refers to it being in the 2nd row and cause[0][1] refers to it being in the 9th column.
### groups
```json
[[1,8]]
```
Group type and index of groups that "cause strategy". This is a 2d array so group[0] is [1, 8] referring to the last column with green borders. group[0][0] refers to the group type (0 = row, 1 = column, 2 = box) and group[0][1] refers to it being the 9th column.
### placements
```json
[]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This does not apply to this strategy.
### removals
```json
[[3,8,7,8],[4,8,7,8]]
```
Notes that can be removed from cells along with their row and columns. The first two values in each subarray represent the row and column respectively so removals[0][0] and removals[0][1] refer to the 9th cell in the 4th row. The rest of the numbers in the subarray are the notes to be removed which is 7 and 8 (although only 8 is there to begin with and is highlighted in red).
### info
```json
"Obvious pairs are when you only have the same two numbers left as a possibility in two cells in the same row, column, or box"
```
Info about the strategy being used by the hint.
### action
```json
"When you see a obvious pair you can remove them from the notes of every other cell in the row, column, or box that they share"
```
Describes the action that the hint is suggesting.
## Example 5: Other Obvious Sets
### strategy
```json
"OBVIOUS_TRIPLET"
```
### strategy
```json
"OBVIOUS_QUADRUPLET"
```
Name of strategy used by the hint. Obvious triplets through octuplets are scaled up versions of obvious pairs except instead of using two cells and notes they share 3-8. Note: while cells in say a obvious quadruplet must share the same 4 notes the 4 cells don't have to individually have all 4 notes they just can't have any other notes.
## Example 6: Hidden Single
![Example 6](https://sudokuru.s3.amazonaws.com/hintExample5-V1.png)
### strategy
```json
"HIDDEN_SINGLE"
```
Name of strategy used by the hint. Hidden single works by having only one cell in a group that still has a number as a possibility in which case all other notes can be removed from it. This is shown by the 8 obvious single which in the only 8 note in the box with the green border. Therefore, the other notes highlighted in red can be removed.
### cause
```json
[[3,8],[4,8],[5,6],[5,8]]
```
Coordinates of cells that "cause" strategy to be applicable. This is a 2d array so cause[0] is [3, 8] referring to the 9th cell in the 4th row (rows and columns are zero-indexed). cause[0][0] refers to it being in the 4th row and cause[0][1] refers to it being in the 9th column.
### groups
```json
[[2,5]]
```
Group type and index of groups that "cause strategy". This is a 2d array so group[0] is [2, 5] referring to the 6th box with green borders (boxes are zero-indexed starting with the top left and going left to right top to bottom). group[0][0] refers to the group type (0 = row, 1 = column, 2 = box) and group[0][1] refers to it being the 6th box.
### placements
```json
[]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This does not apply to this strategy.
### removals
```json
[[3,6,1,2,3,9]]
```
Notes that can be removed from cells along with their row and columns. The first two values in each subarray represent the row and column respectively so removals[0][0] and removals[0][1] refer to the 7th cell in the 4th row. The rest of the numbers in the subarray are the notes to be removed which is 1, 2, 3, and 9 (although only 1 and 3 are present and are highlighted in red).
### info
```json
"Hidden singles are when you only have one cell left still containing a specific value in a row, column, or box"
```
Info about the strategy being used by the hint.
### action
```json
"When you see a hidden single you can remove all notes other than the single from the cell"
```
Describes the action that the hint is suggesting.
## Example 7: Hidden Pair
![Example 7](https://sudokuru.s3.amazonaws.com/hintExample6-V1.png)
### strategy
```json
"HIDDEN_PAIR"
```
Name of strategy used by the hint. Hidden pair works by having only two cells in a group that still have two specific numbers as possibilities in which case all other notes can be removed from the pair of cells. This is shown by the obvious pair in the top right in which they have the only 2 and 3's in the box with the green border. Therefore, the other notes highlighted in red can be removed.
### cause
```json
[[0,7],[1,7],[1,8]]
```
Coordinates of cells that "cause" strategy to be applicable. This is a 2d array so cause[0] is [0, 7] referring to the 8th cell in the 1st row (rows and columns are zero-indexed). cause[0][0] refers to it being in the 1st row and cause[0][1] refers to it being in the 8th column.
### groups
```json
[[2,2]]
```
Group type and index of groups that "cause strategy". This is a 2d array so group[0] is [2, 2] referring to the 3rd box with green borders (boxes are zero-indexed starting with the top left and going left to right top to bottom). group[0][0] refers to the group type (0 = row, 1 = column, 2 = box) and group[0][1] refers to it being the 3rd box.
### placements
```json
[]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This does not apply to this strategy.
### removals
```json
[[0,6,1,7,8],[2,6,1,7,8]]
```
Notes that can be removed from cells along with their row and columns. The first two values in each subarray represent the row and column respectively so removals[0][0] and removals[0][1] refer to the 7th cell in the 1st row. The rest of the numbers in the subarray are the notes to be removed which is 1, 7, and 8 (although only 1 and 8 are present and are highlighted in red in the cell in the first row).
### info
```json
"Hidden pairs are when you only have two cells left still containing two specific values in a shared row, column, or box"
```
Info about the strategy being used by the hint.
### action
```json
"When you see a hidden pair you can remove all notes other than the pair from the cells"
```
Describes the action that the hint is suggesting.
## Example 8: Other Hidden Sets
### strategy
```json
"HIDDEN_TRIPLET"
```
### strategy
```json
"HIDDEN_QUADRUPLET"
```
Name of strategy used by the hint. Hidden triplets through octuplets are scaled up versions of hidden pairs except instead of using two cells and notes they share 3-8. Note: while cells in say a hidden quadruplet must share the same 4 notes the 4 cells don't have to individually have all 4 notes they just have to be the only cells in the group that have those notes.
## Example 9: Pointing Pair
![Example 8](https://sudokuru.s3.amazonaws.com/hintExample7-V1.png)
### strategy
```json
"POINTING_PAIR"
```
Name of strategy used by the hint. Pointing pair works by having only two cells in a box that still have a specific numbers as a possibility and also share a row or column in which case all other notes can be removed from any other cells in the shared row or column except for the pointing pair themself. This is shown by the pointing pair in the top right in which they have the only 6's in the box are in the two cells highlighted in blue on the same row. Therefore, the other notes highlighted in red can be removed.
### cause
```json
[[0,6],[0,8]]
```
Coordinates of cells that "cause" strategy to be applicable.
### groups
```json
[[0,0],[2,2]]
```
Group type and index of groups that "cause strategy".
### placements
```json
[]
```
Row, column, and values for cells that have had values placed in them as result of strategy. This does not apply to this strategy.
### removals
```json
[[0,1,6],[0,2,6]]
```
Notes that can be removed from cells along with their row and columns. The first two values in each subarray represent the row and column respectively.
### info
```json
"Pointing pairs are when you only have two cells in a row or column left still containing a specific value and the cells also share a box"
```
Info about the strategy being used by the hint.
### action
```json
"When you see a pointing pair you can remove all other instances of the shared note from the box (except for those in the pair themself)"
```
Describes the action that the hint is suggesting.
# PuzzleData Object Properties
## solution
81 character string representing the solution to the puzzle (left to right top to bottom)
## difficulty
Number representing the difficulty of the puzzle (higher is harder)
## givensCount
Number representing the number of givens i.e. pre-filled cells in the puzzle
## puzzleStrategies
Boolean array representing which strategies were used to solve the puzzle in following order (subset of StrategyEnum):
- 0: OBVIOUS_SINGLE,
- 1: HIDDEN_SINGLE,
- 2: OBVIOUS_PAIR,
- 3: HIDDEN_PAIR,
- 4: POINTING_PAIR,
- 5: OBVIOUS_TRIPLET,
- 6: HIDDEN_TRIPLET,
- 7: POINTING_TRIPLET,
- 8: OBVIOUS_QUADRUPLET,
- 9: HIDDEN_QUADRUPLET,
## drills
- 1D number array representing last time strategy can be reasonably used as drill or -1 if it can't be used as a drill
- Each index is defined as the puzzle when it has a certain number of cells filled in (so index 30 has 30 cells filled in including givens).
- Reasonably is defined as to mean removing strategies if they overlap with their prereqs (i.e. if a obvious pair is made up of two obvious singles the pair will be excluded) and not counting a strategy if multiple instances of it are present so their is only one correct answer.
- The same number index to strategy mapping as puzzleStrategies is used.

# Developer Tools

Clone Repository
```shell
git clone https://github.com/SudoKuru/Sudokuru.git
```

```shell
bun install # or npm install
```

[Install pre-commit hooks](https://pre-commit.com/#install)

Run the below command to setup pre-commit hooks:

```shell
pre-commit install
```

Pre-commit hooks can be run manually with below command, but
will always run before git commit and git push if setup correctly.

```shell
npm run pre-commit
```

```bun......................................................................Passed```

Should appear in the git output logs after a commit.


```shell
# Install Developer Dependencies
bun install
```

Run Tests

Coverage reports are automatically generated can can be viewed in `jest-coverage` folder
```shell
npm run test
```
Run Tests without coverage (faster)
```shell
bun test
```

Build dist Folder
```shell
bun build
```

Create Local TypeDoc Documentation
```shell
bun update-docs
```

The following can also be done after installing the npm package by navigating to node_modules/sudokuru

Run Demo Server on http://localhost:3100/
```shell
bun start
```
```shell
# Can open interactive Solver Demo in browser while Demo Server is running (Generator/Demo/demo.html)

# Demo Server provides the following fakes for Puzzle Class (use http://localhost:3100/ as url)
# Puzzles.startGame(): Will overwrite text file with activeGame constant and return it to user
# Puzzles.getGame(): Will return the activeGame from text file or return 404 error if the text file doesn't exist
# Puzzles.saveGame(): Attempts to save changes to activeGame stored locally in a text file and returns true if successful
# Puzzles.finishGame(): Attempts to delete activeGame stored locally in a text file and returns true if successful
# Drills.getGame(): Will return a puzzle string constant

# Can modify a lessons by doing the following:
1. Log into AWS Console using team IAM user
2. Navigate to S3
3. Navigate to sudokuru bucket
4. Navigate to Lessons folder
5. Download the lessons json file e.g. AMEND_NOTES.json
6. Make changes locally to the json file
7. Upload the json file (it will overwrite the old version)

# Can add a lesson by doing the following:
1. Navigate to S3 sudokuru bucket Lessons folder same as in modifying lesson instructions above
2. Upload the new lesson json file using the same format as the existing ones
3. Download strategies.json
4. Add the new strategies name (same as uploaded file name) to strategies.json locally
5. Upload the json file (it will overwrite the old version)
```
Official TypeDoc Documentation is Hosted Here: https://sudokuru.github.io/SudokuPuzzleGenerator/

 readmeEtag: '"611074b7b41212110e593aa5ef7db41fa87663f3"' readmeLastModified: Thu, 24 Jul 2025 23:51:43 GMT repositoryId: 559359396 description: >- Code for generating data to be stored in the database related to Sudoku Puzzles. created: '2022-10-29T21:38:29Z' updated: '2025-07-24T23:51:48Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 0 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"b170fcd6641e2d10ef0b2fbd73174ff29e5e1504357c319bfd4fe573cb3021bf"' repoLastModified: Thu, 24 Jul 2025 23:51:48 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/sudokuru/sudokupuzzlegenerator - source: openapi3 tags repository: https://github.com/yugokato/openapi-test-client v3: true id: 7e22733a69cba241a5a0687dfb132f2c repositoryMetadata: base64Readme: >- OpenAPI Test Client
======================
[![test](https://github.com/yugokato/openapi-test-client/actions/workflows/test.yml/badge.svg)](https://github.com/yugokato/openapi-test-client/actions/workflows/test.yml)
[![Code style: black](https://img.shields.io/badge/code%20style-black-000000.svg)](https://github.com/psf/black)

# Overview

OpenAPI Test Client (with an emphasis on "test" client) can generate/update API clients in Python from any 
OpenAPI 3.x specifications.  

The goal of the project is to provide easy-to-use/maintain auto-generated API clients for QA engineers who need to 
thoroughly perform automated API testing against API-first applications in a fast-paced development environment. 

During each software release cycle, QA will need to perform various testing to ensure that new APIs to be released work 
as expected. The testing tool for QA requires to have a lot of flexibility and intelligence to achieve this efficiently, 
and to fully automate everything which of course includes negative cases and some unusual scenarios.
This API test client will make the entire testing process very easy and efficient by automatically handling a lot of 
boilerplate under the hood, while also providing you with full control over how and what you want to test, as well as 
how you want to extend or customize it for your own use case.


# SDK client v.s. Test client

One important thing to clarify is that this is NOT an SDK client but a "test" client although both may share some 
common capabilities. An SDK client utilizes stable APIs under the hood for development purposes, while an API test 
client is designed for testing raw APIs that are currently in development and may still contain bugs. What this means 
is the test client will NOT automatically validate requests and responses for you. It is the QA's main responsibility to 
work on, and the test client will provide everything needed for testing APIs. 

> [!TIP]
> If you would like to expand the client's use case beyond testing, you can optionally enable "validation mode" to make 
> it partially behave like an SDK. When validation mode is enabled, your request payload will be validated in 
> Pydantic strict mode on the client side.  
> Please note that validation against response is not currently supported.


# Async Support
The same client code can support both `sync` (the default) and `async` modes. You can switch to the async mode by 
giving `async_mode=True` when instantiating an API client, then specify `await` to each API function call. 


# Try it out

## 1. Setup

> [!Important]
> Before moving forward, please make sure to create a new `venv` (requires Python 3.11+) and start with a fresh 
> environment. We will use `pip install` to simplify the setup process, but this project is not intended to be used as 
> a publicly available Python package. Since all dependencies are intentionally pinned strictly to exact versions, 
> dependency conflicts may occur if the installation is done within your existing project. 

```
pip install git+https://github.com/yugokato/openapi-test-client.git
```

Or, if you want to try the demo client with the Quart-based [demo backend app](src/demo_app):  
```
pip install "openapi-test-client[app] @ git+https://github.com/yugokato/openapi-test-client.git"
```
Alternatively, you can clone the repo and run `pip install -e .` (or `pip install -e .[app]`) if you prefer to reuse 
this project as your base project. 


## 2. Generate/Update API client code

Once the setup is complete, the `openapi-client` CLI command should be available for generating and updating an API 
client from an OpenAPI spec file. It will take either `generate` or `update` as the first argument. Available command 
options will change based on which action you choose. (see help of each action with `-h` for more details)

```shell
$ openapi-client -h
usage: openapi-client [-h] {generate,update} ...

positional arguments:
  {generate,update}  Action to take
    generate         Generate a new API client from an OpenAPI spec URL
    update           Update an existing API client

options:
  -h, --help         show this help message and exit

```

### Generate a new API client

You can generate a new API client from OpenAPI specs by running the `generate` command:
```
openapi-client generate -u <url> -a <app_name> [--dir <directory>]
```
`url`: The URL of the OpenAPI spec file (JSON/YAML, must be version 3.x) from which you want to generate the API client  
`app_name`: The name of the target application. The value will be used for generating the client name  
`directory`: A directory path to save the generated client modules. The specified directory will become the top-level 
module for accessing all your clients. (This is optional if your project was cloned from the original repository)

<img src="images/generate.gif" width="800"/>

To generate another client for a different app, simply run the same command with the same `--dir` value, but with new 
`-u` and `-a` values. 

> [!NOTE]
> The following manual step(s) will be needed after generating a client
> - For each auto-generated API function, rename the function name to whatever value you like. All functions will be generated as "*\_unnamed_endpoint\_{idx}*" by default
> - (If needed) Adjust the value of the API base URL in the `cfg/urls.json`, and the `doc` value in the generated API client's `__init__()` code. These values will be automatically added during the client generation based on the `url` value you specify, but your application may expect to split the URL differently


### Update the existing API client

You can update the existing client code by running the `update` command to reflect the latest OpenAPI specs: 

```
openapi-client update -c <client_name>
```
`client_name`: The name of the client assigned at the client generation step

<img src="images/update.gif" width="800"/>

This update functionality primarily focuses on reflecting the latest OpenAPI specs. It will only touch the following areas: 
- Existing API class tag
- Existing API class functions (signatures and docstrings)
- Existing API parameter models
- Currently missing API functions and parameter models (a new function will be added as "*\_unnamed_endpoint_{idx}*")
- Currently missing API classes (this requires `-A` option)

Anything else, (eg. API class/function names, function body, decorators applied on API functions, etc.) will remain 
intact so that updated client code won't break your existing test code.

> [!TIP]
> - You might want to run the command with `-d`/`--dry-run` option first to see changes before actually updating code
> - You can limit the scope of the update (eg. specific tag, specific endpoint, etc.) by using various command options. 
> See more details with `-h`

    
## 3. Use the API client

Your client should be ready to use by now. For demonstration purposes, the rest of the sections will be explained using 
the [DemoAppAPIClient](src/openapi_test_client/clients/demo_app) that comes with this repository as an example client. This demo client was actually 
generated by the following `openapi-client generate` command against the [demo app](src/demo_app) 
demo application:
```
openapi-client generate -u http://127.0.0.1:5000/openapi.json -a demo_app
``` 
It was also manually extended with a custom [post-request hook](src/openapi_test_client/clients/demo_app/api/request_hooks/post_request.py) for automatically managing auth tokens 
before login and after logout.

> [!NOTE]
> If you would like to try this `demo_app` API client by yourself, follow the following steps to start the demo application before moving forward:
> 1. Clone the repo
> 2. Set up with `app` server-side extra dependency  (see the "Setup" section)
> 3. Start the application with `quart -A demo_app run` command
> ```shell
> $ quart -A demo_app run
> * Serving Quart app 'demo_app'
> * Debug mode: False
> * Please use an ASGI server (e.g. Hypercorn) directly in production
> * Running on http://127.0.0.1:5000 (CTRL + C to quit)
>[2024-01-01 00:00:00 -0000] [39691] [INFO] Running on http://127.0.0.1:5000 (CTRL + C to quit)
> ```


### Instantiate the client

Once you have generated an API client, the client class will be importable as 
`from <your_module>.clients.<client_name> import <ClientName>APIClient`. 

<details open>
<summary>Sync Client</summary>

```pycon
>>> from openapi_test_client.clients.demo_app import DemoAppAPIClient
>>> client = DemoAppAPIClient()
```
</details>

<details>
<summary>Async Client</summary>

```pycon
>>> from openapi_test_client.clients.demo_app import DemoAppAPIClient
>>> client = DemoAppAPIClient(async_mode=True)
```
</details>

Make sure to replace the "openapi_test_clients" part with your own module name when importing your own clients.

Alternatively, you can instantiate your client directly from the parent `OpenAPIClient` class.

<details open>
<summary>Sync Client</summary>

```pycon
>>> from openapi_test_client.clients import OpenAPIClient
>>> client = OpenAPIClient.get_client("<client_name>")
```
</details>
<details>
<summary>Async Client</summary>

```pycon
>>> from openapi_test_client.clients import OpenAPIClient
>>> client = OpenAPIClient.get_client("<client_name>", async_mode=True)
```
</details>

### Make an API request

To make an API request with your API client, call an API function as  `client.<ApiTag>.<api_function_name>()`. 
The function will take all parameters documented in the OpenAPI specs as keyword arguments. 

eg. To call the login API defined under the Auth tag:

<details open>
<summary>Sync Client</summary>

```pycon
>>> r = client.Auth.login(username='foo', password='bar')
2024-01-01T00:00:00.863-0800 - request: POST http://127.0.0.1:5000/v1/auth/login
2024-01-01T00:00:00.877-0800 - response: 201 (Created)
- request_id: a2b20acf-22d5-4131-ac0d-6796bf19d2af
- request: POST http://127.0.0.1:5000/v1/auth/login
- payload: {"username": "foo", "password": "***"}
- status_code: 201 (Created)
- response: {
    "token": "IjFlNTAxNmI2LTVlZjctNGQxYi1iMGJhLTYxY2M3ZWIzY2VmYSI.ZadDlA.tKS_La5uDuteXH7OMT_WZVK1o_hAnWZVn_J5rSsJQILHu1juXYg0EYLkgpH1LChzeOhN_YUUXaO37rlt_UV1Ag"
}
- response_time: 0.01344s
```
</details>

<details>
<summary>Async Client</summary>

```pycon
# NOTE: This example uses asyncio REPL (python -m asyncio) 
>>> r = await client.Auth.login(username='foo', password='bar')
2024-01-01T00:00:00.863-0800 - request: POST http://127.0.0.1:5000/v1/auth/login
2024-01-01T00:00:00.877-0800 - response: 201 (Created)
- request_id: a2b20acf-22d5-4131-ac0d-6796bf19d2af
- request: POST http://127.0.0.1:5000/v1/auth/login
- payload: {"username": "foo", "password": "***"}
- status_code: 201 (Created)
- response: {
    "token": "IjFlNTAxNmI2LTVlZjctNGQxYi1iMGJhLTYxY2M3ZWIzY2VmYSI.ZadDlA.tKS_La5uDuteXH7OMT_WZVK1o_hAnWZVn_J5rSsJQILHu1juXYg0EYLkgpH1LChzeOhN_YUUXaO37rlt_UV1Ag"
}
- response_time: 0.01344s
```
</details>

> [!NOTE] 
> - A new UUID will be generated at each request and will be set to the `X-Request-ID` header
> - `DemoAppAPIClient` will automatically set the `token` value to the API session in the client's post-request logic mentioned earlier

The function call will return a `RestResponse` object where the decoded response is accessible as `response`. If you 
need to access the raw request and response objects from the underlying `httpx` library, you can do so through the 
`request` and `_response` attributes.

```pycon
>>> r.status_code
201
>>> r.response
{'token': 'IjFlNTAxNmI2LTVlZjctNGQxYi1iMGJhLTYxY2M3ZWIzY2VmYSI.ZadDlA.tKS_La5uDuteXH7OMT_WZVK1o_hAnWZVn_J5rSsJQILHu1juXYg0EYLkgpH1LChzeOhN_YUUXaO37rlt_UV1Ag'}
>>> pprint(r)
RestResponse(_response=<Response [201 ],
             request_id='a2b20acf-22d5-4131-ac0d-6796bf19d2af',
             status_code=201,
             response={'token': 'IjFlNTAxNmI2LTVlZjctNGQxYi1iMGJhLTYxY2M3ZWIzY2VmYSI.ZadDlA.tKS_La5uDuteXH7OMT_WZVK1o_hAnWZVn_J5rSsJQILHu1juXYg0EYLkgpH1LChzeOhN_YUUXaO37rlt_UV1Ag'},
             response_time=0.01344,
             request=<Request('POST', 'http://127.0.0.1:5000/v1/auth/login')>,
             ok=True,
             is_stream=False)
```
Note that we extend the `httpx` library's `Request` and `Client` classes to add a few small capabilities. 
As an example, you can get the request start/end time (UTC) through `request` as a `datatime` object.
 
```pycon
>>> r.request.start_time
datetime.datetime(2024, 1, 1, 0, 0, 0, 86309, tzinfo=datetime.timezone.utc)
>>> r.request.end_time
datetime.datetime(2024, 1, 1, 0, 0, 0, 87714, tzinfo=datetime.timezone.utc)
```

## 4. Customize API functions (optional)

The test client provides a few ways to customize how an API call via an API function will be processed.

### Implement your custom function logic
By default, each auto-generated API function will look like a stub function with a placeholder (`...`). For example:
```python
@endpoint.is_public
@endpoint.post("/v1/auth/login")
def login(self, *, username: str = Unset, password: str = Unset, **kwargs: Unpack[Kwargs]) -> APIResponse:
    """Login"""
    ...
```

Even without having actual function logic, API requests made through these functions will just work - The function call will automatically make an HTTP request to the associated endpoint with the provided parameters, parse the response, and return it as a `RestResponse` object. This behavior is managed by the `@endpoint.<method>(path>)` decorator.  
In most cases, you won’t need to modify the generated function body at all.

However, if you need to implement custom logic for making an API request for specific API functions, you can replace the placeholder with your own code. Just ensure that the function returns a `RestResponse` object, which is what the underlying Rest API client we use will return.

> [!NOTE]
> The client will raise a `RuntimeError` if anything other than a `RestResponse` object or `None` is returned.

> [!TIP]
> If you only need to add extra behavior before or after the request, consider using decorators or request hooks instead

### Decorators
To extend the default API call logic, you can apply decorators on an API function. Note that each decorator MUST be decorated with `@endpoint.decorator` for it to work properly.  

For example:
- Define a decorator
```python
from functools import wraps

from openapi_test_client.libraries.api import endpoint

@endpoint.decorator
def my_decorator(f):
    @wraps(f)
    def wrapper(*args, **kwargs):
        # Do something before request
        response = f(**args, **kwargs)
        # Do something after request
        return response
    return wrapper
```

- Apply it

```python
@my_decorator
@endpoint.is_public
@endpoint.post("/v1/auth/login")
def login(self, *, username: str = Unset, password: str = Unset, **kwargs: Unpack[Kwargs]) -> APIResponse:
    """Login"""
    ...
```

### Request Hooks

The test client supports three types of request hooks in API classes:
- Pre-request Hook: Called before an API request is made
- Post-request Hook: Called after an API request is completed
- Request wrapper: Wraps the original API request logic before any decorators are applied, allowing you to apply additional behavior before and after a request

Hooks work similarly to decorators but are more useful for applying general control that affects multiple or all endpoints at the central place.

For examples and usage, see [demo_app client hooks](src/openapi_test_client/clients/demo_app/api/request_hooks)


# Deep Dive 

The following are some of the important building blocks of the API test client:
- API client
- API class and functions
- API endpoint function
- API endpoint model
- API parameter model
- Pydantic model (validation mode)


## API client

All API clients will inherit the base client `OpenAPIClient`, which will handle parsing OpenAPI specs and resolving 
`$ref`. The generated API client will have a `@cached_property` definition for each API tag (or set of tags when 
managing APIs in multiple tags), where an API class instance will be returned. 

```python
# openapi_test_client/clients/demo_app/demo_app_client.py

from functools import cached_property

from openapi_test_client.clients.base import OpenAPIClient

from .api.auth import AuthAPI
from .api.users import UsersAPI


class DemoAppAPIClient(OpenAPIClient):
    """API client for demo_app"""

    def __init__(self, env: str = "dev", async_mode: bool=False) -> None:
        super().__init__("demo_app", env=env, doc="openapi.json", async_mode=async_mode)

    @cached_property
    def Auth(self) -> AuthAPI:
        return AuthAPI(self)

    @cached_property
    def Users(self) -> UsersAPI:
        return UsersAPI(self)
```


## API class and functions

All API endpoints for a tag (or set of tags) will be managed inside an "API class", where each function will map to a 
specific endpoint documented in the OpenAPI spec. 

For example, if the OpenAPI spec defines Auth APIs like this: 
```json
"paths": {
    "/v1/auth/login": {
      "post": {
        "parameters": [],
        "responses": {},
        "description": "",
        "summary": "Login",
        "tags": [
          "Auth"
        ],
        "security":[],
        "requestBody": {
          "content": {
            "application/json": {
              "schema": {
                "properties": {
                  "username": {
                    "title": "Username",
                    "type": "string"
                  },
                  "password": {
                    "title": "Password",
                    "type": "string"
                  }
                },
                "required": [
                  "username",
                  "password"
                ],
                "title": "LoginData",
                "type": "object"
              }
            }
          }
        },
        "operationId": "post_login"
      }
    },
    "/v1/auth/logout": {
      "get": {
        "parameters": [],
        "responses": {},
        "description": "",
        "summary": "Logout",
        "tags": [
          "Auth"
        ],
        "security":[],
        "operationId": "get_logout"
      }
    },
```

Then the API class and functions will be generated like this:

```python
# openapi_test_client/clients/demo_app/api/auth.py

from typing import Unpack

from common_libs.clients.rest_client import APIResponse

from openapi_test_client.clients.demo_app.api.base import DemoAppBaseAPI
from openapi_test_client.libraries.api.api_functions import endpoint
from openapi_test_client.libraries.api.types import Kwargs, Unset


class AuthAPI(DemoAppBaseAPI):
    TAGs = ("Auth",)

    @endpoint.is_public
    @endpoint.post("/v1/auth/login")
    def login(self, *, username: str = Unset, password: str = Unset, **kwargs: Unpack[Kwargs]) -> APIResponse:
        """Login"""
        ...

    @endpoint.is_public
    @endpoint.get("/v1/auth/logout")
    def logout(self, **kwargs: Unpack[Kwargs]) -> APIResponse:
        """Logout"""
        ...

```
A function can take API path parameters and query/body parameters as arguments, where path parameters are always 
defined as positional-only arguments, and other parameters are always defined as keyword-only arguments with a default 
value of `Unset`. Any parameters with this sentinel value will be excluded from the actual API call parameters.   
Additionally, it will always have `**kwargs` for supporting making a call with any undocumented 
endpoint parameters, as well as with some internal endpoint function call options such as `raw_options`, 
which takes raw `httpx` library options (eg. `timeout`, `headers`, etc.) passed to `httpx.Client.request()` if you need to. 

> [!NOTE]
> Parameters defined as optional in the OpenAPI spec will be annotated with `Optional[]` in the function signature, but 
> this does not mean the parameter is actually optional for the endpoint. It just means you can hit the endpoint 
> without specifying that parameter, whether it is actually required or not (for negative case testing)

Some attributes available from the API class: 
```pycon
>>> # Get tag data
>>> client.Auth.TAGs
('Auth',)
>>> # Get available endpoints under this API class 
>>> pprint(client.Auth.endpoints)
[Endpoint(tags=('Auth',),
          api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>,
          method='post',
          path='/v1/auth/login',
          func_name='login',
          model=<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.AuthAPILoginEndpointModel'>,
          url=None,
          content_type=None,
          is_public=True,
          is_documented=True,
          is_deprecated=False),
 Endpoint(tags=('Auth',),
          api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>,
          method='get',
          path='/v1/auth/logout',
          func_name='logout',
          model=<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.AuthAPILoginEndpointModel'>,
          url=None,
          content_type=None,
          is_public=True,
          is_documented=True,
          is_deprecated=False)]
```

A list of defined API classes are available as `API_CLASSES`.
```pycon
>>> from itertools import chain
>>> from openapi_test_client.clients.demo_app.api import API_CLASSES
# list all endpoints
>>> all_endpoints = chain(*[x.endpoints for x in API_CLASSES])
>>> for endpoint in all_endpoints:
...     print(endpint)
... 
POST /v1/auth/login
GET /v1/auth/logout
POST /v1/users
GET /v1/users/{user_id}
GET /v1/users
POST /v1/users/images
DELETE /v1/users/{user_id}
```


## API endpoint function

Each API class function is decorated with a `@endpoint.<method>(<path>)` endpoint decorator. This decorator converts the original function into an instance of the `EndpointHandler` class at runtime. The `EndpointHandler` object acts as a proxy to a per-endpoint `EndpointFunc` object, which is also created at runtime and is responsible for handling the actual API calls via its base class's `__call__()` method. Additionally, the `EndpointFunc` object provides various capabilities and attributes related to the endpoint.

eg. The Login API is accessible via `client.Auth.login()` API function, which is actually an instance of 
`AuthAPILoginEndpointFunc` class returned by its associated `EndpointHandler` obj.

```pycon
>>> client.Auth.login
<openapi_test_client.libraries.api.api_functions.endpoints.AuthAPILoginEndpointFunc object at 0x1074abf10>
(mapped to: <function AuthAPI.login at 0x10751c360>)
```

The endpoint function is also accessible directly from the API class:
```pycon
>>> from openapi_test_client.clients.demo_app.api.auth import AuthAPI
>>> AuthAPI.login
<openapi_test_client.libraries.api.api_functions.endpoints.AuthAPILoginEndpointFunc object at 0x107650b50>
(mapped to: <function AuthAPI.login at 0x10751c360>)
```

Various endpoint data is available from the endpoint function via `endpoint` property:
```pycon
>>> print(client.Auth.login.endpoint)
POST /v1/auth/login
>>> pprint(client.Auth.login.endpoint)
Endpoint(tags=('Auth',),
         api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>,
         method='post',
         path='/v1/auth/login',
         func_name='login',
         model=<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.AuthAPILoginEndpointModel'>,
         url='http://127.0.0.1:5000/v1/auth/login',
         content_type=None,
         is_public=True,
         is_documented=True,
         is_deprecated=False)
>>> client.Auth.login.endpoint.method
'post'
>>> client.Auth.login.endpoint.path
'/v1/auth/login'
>>> client.Auth.login.endpoint.url
'http://127.0.0.1:5000/v1/auth/login'
```

Note that the same endpoint data is also available directly from the API class, except for `url` will always be `None`.
```pycon
>>> from openapi_test_client.clients.demo_app.api.auth import AuthAPI
>>> client.Auth.login.endpoint == AuthAPI.login.endpoint
True
>>> print(AuthAPI.login.endpoint)
POST /v1/auth/login
>>> pprint(AuthAPI.login.endpoint)
Endpoint(tags=('Auth',),
         api_class=<class 'openapi_test_client.clients.demo_app.api.auth.AuthAPI'>,
         method='post',
         path='/v1/auth/login',
         func_name='login',
         model=<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.AuthAPILoginEndpointModel'>,
         url=None,
         content_type=None,
         is_public=True,
         is_documented=True,
         is_deprecated=False)
```

An example of the additional capability the `EndpointFunc` obj provides - Automatic retry:
```pycon
# Call the endpoint with the automatic retry (you can specify a retry condition if needed)
>>> r = client.Auth.login.with_retry(username='foo', password='bar')
2024-01-01T00:00:00.153-0000 - request: POST http://127.0.0.1:5000/v1/auth/login
2024-01-01T00:00:00.158-0000 - response: 429 (Too Many Requests)
- request_id: 1b028ff7-0880-430c-b5a3-12aa057892cf
- request: POST http://127.0.0.1:5000/v1/auth/login
- payload: {"username": "foo", "password": "***"}
- status_code: 429 (Too Many Requests)
- response: {
    "error": {
        "code": 429,
        "message": "Too many requests",
        "request_id": "1b028ff7-0880-430c-b5a3-12aa057892cf"
    }
}
- response_time: 0.004804s
2024-01-01T00:00:00.159-0000 - Retry condition matched. Retrying in 5 seconds...
2024-01-01T00:00:05.166-0000 - request: POST http://127.0.0.1:5000/v1/auth/login
2024-01-01T00:00:05.171-0000 - response: 201 (Created)
- request_id: 1a34195e-5cec-4d4e-abe0-61acbfcdae1a
- request: POST http://127.0.0.1:5000/v1/auth/login
- payload: {"username": "foo", "password": "***"}
- status_code: 201 (Created)
- response: {
    "token": "ImJjMjA5YjkxLTEzYmItNDE3Yi1iN2ExLTdhNmFlNWFjYjFiNSI.ZuSs3Q.KD-zA6KxCUEC14YUAuCcuQNGNoulp2POuKq0yMIfkGIS0E--V473kssohnvGoIHo97FwwSeOV94NnwaZdGtSxA"
}
- response_time: 0.005143s
>>> 
>>> r.request.retried.request_id
'1b028ff7-0880-430c-b5a3-12aa057892cf'
```


## API endpoint model

Each endpoint is represented as an `EndpointModel` dataclass model, which holds various context around each 
parameter (eg. type annotation).
```pycon
>>> model = client.Auth.login.endpoint.model
>>> print(model)
<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.AuthAPILoginEndpointModel'>
>>> pprint(model.__dataclass_fields__, sort_dicts=False)
{'username': Field(name='username',type=<class 'str'>,default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD),
 'password': Field(name='password',type=<class 'str'>,default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD)}
```

## API parameter model

If a parameter type is documented as `object` (or `array` of objects) with `properties` in the OpenAPI specs, it will 
be represented as a `ParamModel` dataclass model. These models will be automatically generated as a model file (`<your_module_dir>/clients/<client_name>/models/<api_tag>.py`) for each API class.

eg. 

- API function definition (note that the `metadata` parameter is of type `Metadata`)
```python
# openapi_test_client/clients/demo_app/api/users.py

from typing import Annotated, Literal, Unpack

from common_libs.clients.rest_client import APIResponse

from openapi_test_client.clients.demo_app.api.base import DemoAppBaseAPI
from openapi_test_client.libraries.api.api_functions import endpoint
from openapi_test_client.libraries.api.types import Constraint, Format, Kwargs, Optional, Unset

from ..models.users import Metadata


class UsersAPI(DemoAppBaseAPI):
    TAGs = ("Users",)

    @endpoint.post("/v1/users")
    def create_user(
        self,
        *,
        first_name: Annotated[str, Constraint(min_len=1, max_len=255)] = Unset,
        last_name: Annotated[str, Constraint(min_len=1, max_len=255)] = Unset,
        email: Annotated[str, Format("email")] = Unset,
        role: Literal["admin", "viewer", "support"] = Unset,
        metadata: Optional[Metadata] = Unset,
        **kwargs: Unpack[Kwargs],
    ) -> APIResponse:
        """Create a new user"""
        ...
```

- Parameter model definition
```python
# openapi_test_client/clients/demo_app/models/users.py

from dataclasses import dataclass
from typing import Annotated, Literal

from openapi_test_client.libraries.api.types import Constraint, Format, Optional, ParamModel, Unset


@dataclass
class Preferences(ParamModel):
    theme: Optional[Literal["light", "dark", "system"]] = Unset
    language: Optional[str] = Unset
    font_size: Optional[Annotated[int, Constraint(min=8, max=40, multiple_of=2)]] = Unset


@dataclass
class SocialLinks(ParamModel):
    facebook: Optional[Annotated[str, Format("uri"), Constraint(min_len=1)]] = Unset
    instagram: Optional[Annotated[str, Format("uri"), Constraint(min_len=1)]] = Unset
    linkedin: Optional[Annotated[str, Format("uri"), Constraint(min_len=1)]] = Unset
    github: Optional[Annotated[str, Format("uri"), Constraint(min_len=1)]] = Unset


@dataclass
class Metadata(ParamModel):
    preferences: Optional[Preferences] = Unset
    social_links: Optional[SocialLinks] = Unset

```

> [!NOTE]
> Same like API function parameters:
> - All model attributes will be defined with a default value of `Unset`. Any parameters with this sentinel value will be excluded from an API call payload
> - Parameters defined as optional in the OpenAPI spec will be annotated with `Optional[]` in the model definition

A parameter model is a customized `dataclass` that is intended to function exactly the same as a dictionary under the hood. 
Unlike the native dataclass, a parameter model instance will only hold fields that were explicitly given at instantiation. 
This is very important when you want to differentiate test scenarios for "not including a parameter in the payload" v.s. 
"explicitly giving a `null` value for a parameter".

```pycon
>>> from dataclasses import asdict, is_dataclass
>>> from openapi_test_client.clients.demo_app.models.users import *
>>> metadata = Metadata(preferences=Preferences(theme='dark'))
>>> print(metadata)
Metadata(preferences=Preferences(theme='dark'))
>>> print(metadata.preferences)
Preferences(theme='dark')
>>> print(metadata["preferences"])
Preferences(theme='dark')
>>> is_dataclass(metadata) and isinstance(metadata, dict)
True
>>> metadata.items()
dict_items([('preferences', Preferences(theme='dark'))])
>>> asdict(metadata)
{'preferences': {'theme': 'dark'}}
```

Parameter models function both as a dataclass and a dictionary, and a change made to one will be reflected in the other.
```pycon
>>> # Update a model field value
>>> metadata.preferences.theme = "light"
>>> print(metadata)
Metadata(preferences=Preferences(theme='light'))
>>> asdict(metadata)
{'preferences': {'theme': 'light'}}
>>>
>>> # Update as a dictionary
>>> metadata.preferences["theme"] = "dark"
>>> print(metadata)
Metadata(preferences=Preferences(theme='dark'))
>>> asdict(metadata)
{'preferences': {'theme': 'dark'}}
```

You can also dynamically add or delete model fields similar to how you can add or remove a key from a dictionary, if you need to.
```pycon
>>> # Add a new field to the model
>>> metadata.new_attr1 = 123
>>> print(metadata)
Metadata(preferences=Preferences(theme='dark'), new_attr1=123)
>>> asdict(metadata)
{'preferences': {'theme': 'dark'}, 'new_attr1': 123}
>>> 
>>> # Add a new field to the dictionary
>>> metadata["new_attr2"] = 456
>>> print(metadata)
Metadata(preferences=Preferences(theme='dark'), new_attr1=123, new_attr2=456)
>>> asdict(metadata)
{'preferences': {'theme': 'dark'}, 'new_attr1': 123, 'new_attr2': 456}
>>>
>>> # Delete a field from the model
>>> delattr(metadata, "new_attr2")
>>> print(metadata)
Metadata(preferences=Preferences(theme='dark'), new_attr1=123)
>>> asdict(metadata)
{'preferences': {'theme': 'dark'}, 'new_attr1': 123}
>>>
>>> # Delete a field from the dictionary
>>> del metadata["new_attr1"]
>>> print(metadata)
Metadata(preferences=Preferences(theme='dark'))
>>> asdict(metadata)
{'preferences': {'theme': 'dark'}}
```

## Pydantic models (Validation mode)

By default, our client models will not perform any validation for you, since they are based on `dataclass`. This allows 
the test client to send whatever invalid values for a parameter to verify the server-side validation logic.    
To expand the use case of the client beyond testing, you can optionally enable "validation mode". When enabled, all of 
our dataclass models (Endpoint/Param models) will be automatically converted into native Pydantic models that inherit 
this custom base model class:

```python
from typing import ClassVar

from pydantic import BaseModel, ConfigDict


class PydanticModel(BaseModel):
    """Base class for Pydantic endpoint/param models"""

    model_config: ClassVar = ConfigDict(extra="forbid", validate_assignment=True, strict=True)
```

With this dynamic conversion, request parameters will be automatically validated in Pydantic strict mode before an API 
request call.  

Here are some comparisons between regular models and pydantic models:

- Regular dataclass model (Validation will be done on the server-side)
```pycon 
>>> # Model definition
>>> model = client.Users.create_user.endpoint.model
>>> print(model)
<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.UsersAPICreateUserEndpointModel'>
>>> pprint(model.__dataclass_fields__, sort_dicts=False)
{'first_name': Field(name='first_name',type=typing.Annotated[str, Constraint(min_len=1, max_len=255)],default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD),
 'last_name': Field(name='last_name',type=typing.Annotated[str, Constraint(min_len=1, max_len=255)],default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD),
 'email': Field(name='email',type=typing.Annotated[str, Format(value='email')],default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD),
 'role': Field(name='role',type=typing.Literal['admin', 'viewer', 'support'],default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD),
 'metadata': Field(name='metadata',type=typing.Optional[openapi_test_client.clients.demo_app.models.users.Metadata],default=<object object at 0x107b410b0>,default_factory=<dataclasses._MISSING_TYPE object at 0x107ea61d0>,init=True,repr=True,hash=None,compare=True,metadata=mappingproxy({}),kw_only=True,_field_type=_FIELD)}
>>>
>>> # Make an API request with the invalid parameter values
>>> r = client.Users.create_user(first_name=123, email="foo", role="something", metadata=Metadata(social_links=SocialLinks(facebook="test")), extra=123)
2024-01-01T00:00:00.741-0800 - The request contains one or more parameters UsersAPI.create_user() does not expect:
- extra
2024-01-01T00:00:00.742-0800 - request: POST http://127.0.0.1:5000/v1/users
2024-01-01T00:00:00.752-0800 - response: 400 (Bad Request)
- request_id: 91b4ac05-1280-473b-ac41-0be30903c448
- request: POST http://127.0.0.1:5000/v1/users
- payload: {"first_name": 123, "email": "foo", "role": "something", "metadata": {"social_links": {"facebook": "test"}}, "extra": 123}
- status_code: 400 (Bad Request)
- response: {
    "error": {
        "code": 400,
        "message": [
            {
                "type": "string_type",
                "loc": [
                    "first_name"
                ],
                "msg": "Input should be a valid string",
                "input": 123,
                "url": "https://errors.pydantic.dev/2.9/v/string_type"
            },
            {
                "type": "missing",
                "loc": [
                    "last_name"
                ],
                "msg": "Field required",
                "input": {
                    "first_name": 123,
                    "email": "foo",
                    "role": "something",
                    "metadata": {
                        "social_links": {
                            "facebook": "test"
                        }
                    },
                    "extra": 123
                },
                "url": "https://errors.pydantic.dev/2.9/v/missing"
            },
            {
                "type": "value_error",
                "loc": [
                    "email"
                ],
                "msg": "value is not a valid email address: An email address must have an @-sign.",
                "input": "foo",
                "ctx": {
                    "reason": "An email address must have an @-sign."
                },
                "url": "https://errors.pydantic.dev/2.9/v/value_error"
            },
            {
                "type": "enum",
                "loc": [
                    "role"
                ],
                "msg": "Input should be 'admin', 'viewer' or 'support'",
                "input": "something",
                "ctx": {
                    "expected": "'admin', 'viewer' or 'support'"
                },
                "url": "https://errors.pydantic.dev/2.9/v/enum"
            },
            {
                "type": "url_parsing",
                "loc": [
                    "metadata",
                    "social_links",
                    "facebook"
                ],
                "msg": "Input should be a valid URL, relative URL without a base",
                "input": "test",
                "ctx": {
                    "error": "relative URL without a base"
                },
                "url": "https://errors.pydantic.dev/2.9/v/url_parsing"
            }
        ],
        "request_id": "91b4ac05-1280-473b-ac41-0be30903c448"
    }
}
- response_time: 0.010107s
```

<br>

- Pydantic model (Validation will be done on the client-side)
```pycon
>>> # Model definition
>>> pydantic_model = client.Users.create_user.endpoint.model.to_pydantic()
>>> print(pydantic_model)
<class 'openapi_test_client.libraries.api.api_functions.utils.endpoint_model.UsersAPICreateUserEndpointModelPydantic'>
>>> pprint(pydantic_model.model_fields, sort_dicts=False)
{'first_name': FieldInfo(annotation=str, required=True, metadata=[MinLen(min_length=1), MaxLen(max_length=255)]),
 'last_name': FieldInfo(annotation=str, required=True, metadata=[MinLen(min_length=1), MaxLen(max_length=255)]),
 'email': FieldInfo(annotation=EmailStr, required=True, metadata=[Format(value='email')]),
 'role': FieldInfo(annotation=Literal['admin', 'viewer', 'support'], required=True),
 'metadata': FieldInfo(annotation=Union[MetadataPydantic, NoneType], required=False, default=None)}
>>>
>>> # Make an API request with the same invalid parmeter values, but with validate=True option
>>> r = client.Users.create_user(first_name=123, email="foo", role="something", metadata=Metadata(social_links=SocialLinks(facebook="test")), extra=123, validate=True)
2024-01-01T00:00:00.830-0800 - The request contains one or more parameters UsersAPI.create_user() does not expect:
- extra
Traceback (most recent call last):
  File "<stdin>", line 1, in <module>
    <snip>
    raise ValueError(
ValueError: Request parameter validation failed.
6 validation errors for UsersAPICreateUserEndpointModelPydantic
first_name
  Input should be a valid string [type=string_type, input_value=123, input_type=int]
    For further information visit https://errors.pydantic.dev/2.9/v/string_type
last_name
  Field required [type=missing, input_value={'first_name': 123, 'emai... 'test'}}, 'extra': 123}, input_type=dict]
    For further information visit https://errors.pydantic.dev/2.9/v/missing
email
  value is not a valid email address: An email address must have an @-sign. [type=value_error, input_value='foo', input_type=str]
role
  Input should be 'admin', 'viewer' or 'support' [type=literal_error, input_value='something', input_type=str]
    For further information visit https://errors.pydantic.dev/2.9/v/literal_error
metadata.social_links.facebook
  Input should be a valid URL, relative URL without a base [type=url_parsing, input_value='test', input_type=str]
    For further information visit https://errors.pydantic.dev/2.9/v/url_parsing
extra
  Extra inputs are not permitted [type=extra_forbidden, input_value=123, input_type=int]
    For further information visit https://errors.pydantic.dev/2.9/v/extra_forbidden
```

> [!TIP]
> The above example enables validation mode "per request" by passing `validate=True` to the API function call. 
> Alternatively, you can also set `VALIDATION_MODE='true'` environment variable to globally enable the validation mode.
 readmeEtag: '"a19e58a9177f1919ad346228cc575ce8be07a6c4"' readmeLastModified: Mon, 15 Dec 2025 19:29:14 GMT repositoryId: 763312966 description: >- Dynamically generate/update API test clients from any OpenAPI 3.x specifications. Supports both sync and async. created: '2024-02-26T03:47:38Z' updated: '2025-12-18T19:38:56Z' language: Python archived: false stars: 4 watchers: 2 forks: 0 owner: yugokato logo: https://avatars.githubusercontent.com/u/15665210?v=4 license: MIT repoEtag: '"42ecb3ec11d4020fc6137a98f38a8cee22a4caec1289abf843647891e63b20f9"' repoLastModified: Thu, 18 Dec 2025 19:38:56 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/wtx-labs/woocommerce-api-openapi-specification v3: true id: 5ef6b6f7a0643145ce282df554713a32 repositoryMetadata: base64Readme: >- IyBXb29Db21tZXJjZSBSRVNUIEFQSSBzcGVjaWZpY2F0aW9uIGluIE9wZW5BUEkgMy4wIGZvcm1hdAoKVGhlIFdvb0NvbW1lcmNlIFJFU1QgQVBJIHNwZWNpZmljYXRpb24gaW4gT3BlbkFQSSAzLjAgZm9ybWF0LgoKIyMg8J+aqCBQcm9qZWN0IFN0YXR1cwoKPiDimqDvuI8gKipOb3RlOiBUaGlzIGlzIGEgZGV2ZWxvcG1lbnQgdmVyc2lvbiEqKgo+IAo+IFRoZSBzcGVjaWZpY2F0aW9uIGlzIHVuZGVyIGRldmVsb3BtZW50IGFuZCB3aWxsIGJlIGdyYWR1YWxseSBleHBhbmRlZCB0byBjb3ZlciB0aGUga2V5IGZ1bmN0aW9uYWxpdGllcyBvZiB0aGUgY29tbWVyY2UgZW5naW5lLgoKIyMg8J+TpiBWZXJzaW9uIEluZm9ybWF0aW9uCgotICoqQ3VycmVudCBWZXJzaW9uKio6IGAwLjkuN2AKLSAqKk9wZW5BUEkgU3BlY2lmaWNhdGlvbiBWZXJzaW9uKio6IGAzLjBgCi0gKipTdXBwb3J0ZWQgV29vQ29tbWVyY2UgQVBJIFZlcnNpb24qKjogYHYzYAoKIyMg8J+UkyBMaWNlbnNlCgoqKk1JVCBMaWNlbnNlKioKClBlcm1pc3Npb24gaXMgaGVyZWJ5IGdyYW50ZWQsIGZyZWUgb2YgY2hhcmdlLCB0byBhbnkgcGVyc29uIG9idGFpbmluZyBhIGNvcHkgb2YgdGhpcyBzb2Z0d2FyZSBhbmQgYXNzb2NpYXRlZCBkb2N1bWVudGF0aW9uIGZpbGVzLCB0byB1c2UsIGNvcHksIG1vZGlmeSwgbWVyZ2UsIHB1Ymxpc2gsIGRpc3RyaWJ1dGUsIHN1YmxpY2Vuc2UsIGFuZC9vciBzZWxsIGNvcGllcyBvZiB0aGUgc29mdHdhcmUuCgpUaGUgb25seSByZXF1aXJlbWVudCBpcyB0byBwcmVzZXJ2ZSB0aGUgb3JpZ2luYWwgYXV0aG9yIGF0dHJpYnV0aW9uIGluIHRoZSBzb3VyY2UgY29kZSBhbmQgZG9jdW1lbnRhdGlvbi4KCiMjIPCflI0gUXVpY2sgR2xhbmNlIGF0IHRoZSBBUEkgaW4gU3dhZ2dlcgoKWW91IGNhbiB0YWtlIGEgcXVpY2sgbG9vayBhdCB0aGUgQVBJIGluIFN3YWdnZXIgW2hlcmVdKGh0dHBzOi8vZWRpdG9yLnN3YWdnZXIuaW8vP3VybD1odHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vd3R4LWxhYnMvd29vY29tbWVyY2UtYXBpLW9wZW5hcGktc3BlY2lmaWNhdGlvbi9tYWluL3dvb2NvbW1lcmNlLXJlc3QtYXBpLW9wZW5hcGktc3BlY2lmaWNhdGlvbi55bWwpLgoKIVtXb29Db21tZXJjZSBSRVNUIEFQSSBpbiBTd2FnZ2VyIEVkaXRvcl0od3R4LWxhYnMtd29vY29tbWVyY2UtcmVzdC1hcGktZWRpdG9yLXN3YWdnZXItaW8tZXhhbXBsZS5wbmcpCgojIyDwn5qAIFdvb0NvbW1lcmNlIFJFU1QgQVBJIENsaWVudCBmb3IgSmF2YQoKQXJlIHlvdSBsb29raW5nIHRvIHNlYW1sZXNzbHkgaW50ZWdyYXRlIHlvdXIgSmF2YSBhcHBsaWNhdGlvbnMgd2l0aCBXb29Db21tZXJjZT8gTG9vayBubyBmdXJ0aGVyISBPdXIgW1dvb0NvbW1lcmNlIFJFU1QgQVBJIENsaWVudCBmb3IgSmF2YV0oaHR0cHM6Ly9naXRodWIuY29tL3d0eC1sYWJzL3dvb2NvbW1lcmNlLWFwaS1jbGllbnQtamF2YSkgaXMgZGVzaWduZWQgdG8gbWFrZSB5b3VyIGludGVncmF0aW9uIHByb2Nlc3Mgc21vb3RoIGFuZCBlZmZpY2llbnQuIFRyeSBpdCBhbmQgZ2l2ZSB1cyB5b3VyIGZlZWRiYWNrIQoKIyMg8J+UlyBTdGF5IENvbm5lY3RlZAoK4pyoIFdlJ3JlIGNvbnN0YW50bHkgaW1wcm92aW5nIHRoaXMgc3BlY2lmaWNhdGlvbiEK8J+SoSBIYXZlIHN1Z2dlc3Rpb25zIG9yIG5lZWQgaGVscD8KLSBbUmVwb3J0IGEgQnVnXShodHRwczovL2dpdGh1Yi5jb20vd3R4LWxhYnMvd29vY29tbWVyY2UtYXBpLW9wZW5hcGktc3BlY2lmaWNhdGlvbi9pc3N1ZXMvbmV3P3RlbXBsYXRlPWJ1Z19yZXBvcnQueW1sKQotIFtSZXF1ZXN0IGEgRmVhdHVyZV0oaHR0cHM6Ly9naXRodWIuY29tL3d0eC1sYWJzL3dvb2NvbW1lcmNlLWFwaS1vcGVuYXBpLXNwZWNpZmljYXRpb24vaXNzdWVzL25ldz90ZW1wbGF0ZT1mZWF0dXJlX3JlcXVlc3QueW1sKQotIFtJbXByb3ZlIERvY3VtZW50YXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS93dHgtbGFicy93b29jb21tZXJjZS1hcGktb3BlbmFwaS1zcGVjaWZpY2F0aW9uL2lzc3Vlcy9uZXc/dGVtcGxhdGU9ZG9jdW1lbnRhdGlvbl9pc3N1ZS55bWwpCgrwn5qAIEhhcHB5IGNvZGluZyEg8J+YigoKKipZb3VyIFdUWCBMYWJzIFRlYW0qKiDwn5qACg== readmeEtag: '"8ad2a4cb87075c44062233cd837f6c507f940a0a"' readmeLastModified: Fri, 20 Jun 2025 12:22:53 GMT repositoryId: 906266270 description: The WooCommerce REST API specification in OpenAPI 3.0 format created: '2024-12-20T14:13:24Z' updated: '2026-02-05T08:10:46Z' language: null archived: false stars: 7 watchers: 2 forks: 0 owner: wtx-labs logo: https://avatars.githubusercontent.com/u/192332221?v=4 repoEtag: '"e15256d314a8e39944830a2e9c2637cb9c69069542c3225abaf7c6b5a05a8a15"' repoLastModified: Thu, 05 Feb 2026 08:10:46 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/wtx-labs/woocommerce-api-openapi-specification- - source: openapi3 tags repository: https://github.com/x1n13y84issmd42/oasis v3: true repositoryMetadata: base64Readme: >- IyBPQVNJUwpPcGVuIEFQSSBTcGVjaWZpY2F0aW9uIEludGVsbGlnZW5jZSBTZXJ2aWNlcy4gT3IgaW4gYSBsZXNzIGZhbmN5IHdheSwgYSB0b29sIHRvIHRlc3QgQVBJcyB3aGljaCB1c2VzIE9BUy9Td2FnZ2VyIHNwZWMgZmlsZXMgYXMgYSB0ZXN0IHN1aXRlLgoKWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5jb20veDFuMTN5ODRpc3NtZDQyL29hc2lzLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kuY29tL3gxbjEzeTg0aXNzbWQ0Mi9vYXNpcykKPGEgaHJlZj0iaHR0cHM6Ly9jb2RlY2xpbWF0ZS5jb20vZ2l0aHViL3gxbjEzeTg0aXNzbWQ0Mi9vYXNpcy9tYWludGFpbmFiaWxpdHkiPjxpbWcgc3JjPSJodHRwczovL2FwaS5jb2RlY2xpbWF0ZS5jb20vdjEvYmFkZ2VzL2E2MzQ4MjUzMDYzZTE3OWJhNDRmL21haW50YWluYWJpbGl0eSIgLz48L2E+CjxhIGhyZWY9Imh0dHBzOi8vY29kZWNsaW1hdGUuY29tL2dpdGh1Yi94MW4xM3k4NGlzc21kNDIvb2FzaXMvdGVzdF9jb3ZlcmFnZSI+PGltZyBzcmM9Imh0dHBzOi8vYXBpLmNvZGVjbGltYXRlLmNvbS92MS9iYWRnZXMvYTYzNDgyNTMwNjNlMTc5YmE0NGYvdGVzdF9jb3ZlcmFnZSIgLz48L2E+ClshW0dvIFJlcG9ydCBDYXJkXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS94MW4xM3k4NGlzc21kNDIvb2FzaXMpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20veDFuMTN5ODRpc3NtZDQyL29hc2lzKQoKV29yayBpbiBwcm9ncmVzcy4KCi0gW1VzYWdlXSgjdXNhZ2UpCiAgLSBbTWFudWFsIG1vZGVdKCNtYW51YWwtbW9kZSkKICAtIFtTY3JpcHQgbW9kZV0oI3NjcmlwdC1tb2RlKQoKIyMgVXNhZ2UKT2FzaXMgY2FuIGJlIHVzZWQgZWl0aGVyIGluIG1hbnVhbCBtb2RlLCB3aGljaCBhbGxvd3MgdG8gdGVzdCBzaW5nbGUgb3BlcmF0aW9ucywgb3IgaW4gc2NyaXB0IG1vZGUsIHdoaWNoIGlzIGRlc2lnbmVkIHRvIHRlc3QgY29tcGxleCBpbnRlcmFjdGlvbiBzY2VuYXJpb3MsIGludm9sdmluZyBtdWx0aXBsZSBlbmRwb2ludHMgYW5kIHJldXNpbmcgZGF0YSBhY3Jvc3MgdGhlbS4KCiMjIyDimJHvuI8gTWFudWFsIG1vZGUKYHJ1bi9vYXNpcyBmcm9tIHNwZWMvcGV0c3RvcmUueWFtbCB0ZXN0IGZpbmRQZXRzQnlTdGF0dXNgCgpPYXNpcyB1c2VzIHRoZSBleGFtcGxlIHZhbHVlIGZvciB0aGUgYHN0YXR1c2AgcXVlcnkgcGFyYW1ldGVyIGZvciB0aGUgYGZpbmRQZXRzQnlTdGF0dXNgIG9wZXJhdGlvbiBkZWZpbmVkIGluIHRoZSBzcGVjIGZpbGUuCgpZb3UgY2FuIG92ZXJyaWRlIGFueSBwYXJhbWV0ZXIgZnJvbSBDTEk6CgpgcnVuL29hc2lzIGZyb20gc3BlYy9wZXRzdG9yZS55YW1sIHRlc3QgZ2V0UGV0QnlJZCB1c2UgcGF0aCBwYXJhbWV0ZXJzIHBldElkPTEwYAoKSW5jcmVhc2UgbG9nZ2luZyB2ZXJib3NpdHkgdG8gc2VlIGhvdyBwYXJhbWV0ZXJzIGFyZSB1c2VkLgoKYHJ1bi9vYXNpcyBmcm9tIHNwZWMvcGV0c3RvcmUueWFtbCB0ZXN0IGdldFBldEJ5SWQgbG9nIGF0IGxldmVsIDYgdXNlIHBhdGggcGFyYW1ldGVycyBwZXRJZD0xMGAKCkZvciBhbiBleGFtcGxlIG9mIGVycm9yIHJlcG9ydGluZyB1c2UgdGhlIG1hbGZvcm1lZCBzcGVjIGZpbGUgd2hpY2ggZG9lc24ndCBjb3JyZXNwb25kIHRvIHRoZSBhY3R1YWwgQVBJIHJlc3BvbnNlczoKCmBydW4vb2FzaXMgZnJvbSBzcGVjL2Vycm9ycy55YW1sIHRlc3QgdGFza0xpc3QgbG9nIGF0IGxldmVsIDZgCgpgcnVuL29hc2lzIGZyb20gc3BlYy9lcnJvcnMueWFtbCB0ZXN0IGNvbmZpZ1R5cGVzIGxvZyBhdCBsZXZlbCA2YAoK8J+TliBbTGVhcm4gbW9yZSBhYm91dCBDTEldKGRvYy9DTEkubWQpCgrwn5OWIFtMZWFybiBtb3JlIGFib3V0IG9wZXJhdGlvbiBwYXJhbWV0ZXJzXShkb2MvUGFyYW1ldGVycy5tZCkKCiMjIyDwn5SAIFNjcmlwdCBtb2RlCkZvciBjb21wbGV4IHNjZW5hcmlvcyBpbnZvbHZpbmcgbXVsdGlwbGUgZW5kcG9pbnRzIGFuZCBkYXRhIHJldXNlIGFjcm9zcyB0aGVtIHRoZXJlIGlzIGEgc2NyaXB0IG1vZGU6CgpgcnVuL29hc2lzIGV4ZWN1dGUgc2NyaXB0L3BldHN0b3JlLnlhbWxgCmBydW4vb2FzaXMgZXhlY3V0ZSBzY3JpcHQvbnV4ZW8ueWFtbGAKClNjcmlwdCBpcyBhIGdyYXBoIG9mIGRlcGVuZGVudCBvcGVyYXRpb25zLiBDeWNsZXMgYXJlIG5vdCBhbGxvd2VkOgoKYHJ1bi9vYXNpcyBleGVjdXRlIHNjcmlwdC9jeWNsZS55YW1sYAoK8J+TliBbTGVhcm4gbW9yZSBhYm91dCBzY3JpcHRzXShkb2MvU2NyaXB0Lm1kKQoKIyMgUmVzb3VyY2VzCltPcGVuQVBJIFNwZWNdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4yLm1kI3NlY3VyaXR5U2NoZW1lT2JqZWN0KQoKW0hUVFAgQXV0aCBSZWdpc3RyeV0oaHR0cDovL3d3dy5pYW5hLm9yZy9hc3NpZ25tZW50cy9odHRwLWF1dGhzY2hlbWVzL2h0dHAtYXV0aHNjaGVtZXMueGh0bWwpCgpbU3RvcGxpZ2h0IE9BUyBFZGl0b3JdKGh0dHBzOi8vc3RvcGxpZ2h0LmlvL3Avc3R1ZGlvL3NsL18vNmM1ZmFvZnkpCgpbUmV5ZXNvZnQgQVBJIFBsYXlncm91bmRdKGh0dHBzOi8vanNvbmFwaXBsYXlncm91bmQucmV5ZXNvZnQuY29tLykKCltOdXhlbyBBUEkgUGxheWdyb3VuZF0oaHR0cHM6Ly9udXhlby5naXRodWIuaW8vYXBpLXBsYXlncm91bmQvIy9yZXNvdXJjZXMpCgpbQm9va2VyIEFQSSBQbGF5Z3JvdW5kXShodHRwczovL3Jlc3RmdWwtYm9va2VyLmhlcm9rdWFwcC5jb20vYXBpZG9jL2luZGV4Lmh0bWwpCgpbT2FzaXMgVjJdKGh0dHBzOi8vZ2l0aHViLmNvbS94MW4xM3k4NGlzc21kNDIvb2FzaXMvdHJlZS9jODhjOWExNWUwYTA1YWJiZjczMmY3ZmQ5NWFhMzBmN2NmNDk0N2ZkKQo= readmeEtag: '"7f8c637a8454059f8380bcf780fdc047f5fd7805"' readmeLastModified: Fri, 14 Jan 2022 11:17:29 GMT repositoryId: 199255832 description: 🔀🆗☑️ Automated API testing from Swagger/Open API specifications. created: '2019-07-28T07:20:06Z' updated: '2023-08-08T12:48:41Z' language: Go archived: false stars: 2 watchers: 1 forks: 1 owner: x1n13y84issmd42 logo: https://avatars.githubusercontent.com/u/2878036?v=4 license: GPL-3.0 repoEtag: '"7585da61431503f2ca443dfc57ae74a8d00a4053930ed3d9082a120c421b4509"' repoLastModified: Tue, 08 Aug 2023 12:48:41 GMT foundInMaster: true category: Converters id: c874538e827425f5331ba6bf670d35b1 - source: openapi3 tags repository: https://github.com/unionj-cloud/go-doudou-openapi-ui v3: true id: 066c6ad6c3382351e6ab1b1493a4e4b7 repositoryMetadata: base64Readme: >- IyBnby1kb3Vkb3Utb3BlbmFwaS11aQpUaGlzIGlzIGFuIE9wZW5BUEkzLjAgc3BlY2lmaWNhdGlvbiBVSSBwcm9qZWN0IHdyaXR0ZW4gYnkgdnVlanMrdHlwZXNjcmlwdCBzdGFjayBpbnNwaXJlZCBieSBbcmVkb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKSAuIEN1cnJlbnRseSwgaXQgc3VwcG9ydHMganNvbiBmb3JtYXQgc3BlYyBvbmx5LCB5YW1sIHN1cHBvcnQgaXMgY29taW5nIHNvb24uCgohW2V4YW1wbGVdKC4vZXhhbXBsZS5naWYpCiMjIFVzYWdlCkNsb25lIHRoZSByZXBvLgpgYGAKZ2l0IGNsb25lIGdpdEBnaXRodWIuY29tOnVuaW9uai1jbG91ZC9nby1kb3Vkb3Utb3BlbmFwaS11aS5naXQKYGBgCkluc3RhbGwgZGVwZW5kZW5jaWVzIGFuZCBzdGFydCB0aGUgcHJvamVjdCwgdGhlbiBicm93c2VyIHdpbGwgYmUgb3BlbmVkIGF1dG9tYXRpY2FsbHkuCmBgYHNoZWxsCnlhcm4gaW5zdGFsbCAmJiB5YXJuIHNlcnZlCmBgYAoKQWRkIGA/ZG9jVXJsPSR7T0FTM19TUEVDX0xJTkt9YCB0byBgaHR0cDovL2xvY2FsaG9zdDo5NTI3LyMvaW5kZXhgIGFuZCBjbGljayBgZW50ZXJgLCBvbmxpbmUgZG9jdW1lbnRhdGlvbiB3aWxsIGRpc3BsYXkuICAKCk5PVEU6IGAke09BUzNfU1BFQ19MSU5LfWAgbXVzdCBzdXBwb3J0IGNvcnMgdmlzaXQsIGZvciBleGFtcGxlOiBgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uYAoKIyMgU2NyZWVuc2hvdAohW3NjcmVlbnNob3RdKC4vc2NyZWVuc2hvdC5wbmcpCiFbc2NyZWVuc2hvdDJdKC4vc2NyZWVuc2hvdDIucG5nKQohW3NjcmVlbnNob3QzXSguL3NjcmVlbnNob3QzLnBuZykK readmeEtag: '"db0286fb915f3b25a93ad40199cb625d38560cb7"' readmeLastModified: Sun, 20 Feb 2022 11:01:41 GMT repositoryId: 392144026 description: OpenAPI3 Online Documentation UI created: '2021-08-03T01:12:59Z' updated: '2023-03-01T02:04:02Z' language: Vue archived: false stars: 2 watchers: 0 forks: 3 owner: unionj-cloud logo: https://avatars.githubusercontent.com/u/79033021?v=4 license: MIT repoEtag: '"c1d1984ff0fb306bd050ee8d04822cdb718150fc3a24724f273028a5e100431d"' repoLastModified: Wed, 01 Mar 2023 02:04:02 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/clickermonkey/rez v3: true id: 0ab51b0a493219a5638555ecc871cd31 repositoryMetadata: base64Readme: >- # rez
REST (easy) framework in Go with out of the box OpenAPI generation, validation, dependency injection, and much more. 

> go get github.com/ClickerMonkey/rez

- [Dependency Injection](#dependency-injection) How values are sent to middleware and endpoint functions.
- [Router](#router) Defining the routes, middlware, & documentation.
- [Middleware](#middleware) Code that runs before it reaches the final endpoint
- [Inspection](#inspection) How types are converted into documentation.
- [Validation](#validation) How to control validation.
- [Documentation](#documentation) All the ways to specify documentation.
- [Site](#methods) The main site type and its useful methods.

### Example
```go
type Echo struct {
	Message string `json:"message" api:"desc=A message to echo."`
}
site := rez.New(chi.NewRouter())
// /echo?message=HelloWorld!
site.Get("/echo", func(q rez.Query[Echo]) (*Echo, *rez.NotFound[string]) {
  if q.Value.Message == "" {
    return nil, rez.NewNotFound("message is required")
  }
  return &q.Value, nil
})
site.ServeSwaggerUI("/doc/swagger", nil)
site.ServeRedoc("/doc/redoc", nil)
site.Listen(":3000")
```
More can be found in [examples](examples).

## Dependency Injection

Dependency injection is used to pass arguments to a middleware or route functions. **rez** uses [deps](https://github.com/ClickerMonkey/deps) for dependency injection. There are several types that get injected out of the box:

- `context.Context`: The context of the request.
- `deps.Scope`: The scope which holds all the given values that can be injected for the current request. The root `Site` type has a parent scope which request scopes can inherit values from.
- `http.ResponseWriter`: The outgoing response.
- `http.Request`: The incoming request.
- `rez.Router`: The reference to the router where the middleware was used or the route was defined on.
- `rez.Path[P]`: A generic wrapper which holds the struct that is parsed from the path parameters. If the path is `/task/{taskID}` and the struct is `type TaskPath struct { TaskID int }` the `TaskID` property will be populated from the value in the URL.
- `rez.Query[Q]`: A generic wrapper which holds the struct that is parsed from the query string. If the url is `?message=Hi&times=4` and the struct is `type MyQuery struct { Message string, Times int }` the Message and Times fields will be populated from the query string.
- `rez.Header[H]`: A generic wrapper which holds the struct that is parsed from the headers. 
- `rez.Body[B]`: A generic wrapper which holds the type that is parsed from the request body.
- `rez.Request[B, P, Q]`: A generic wrapper which holds the body, params, and query structs that are to be parsed from the request.
- `rez.Validator`: A validator for the route or middleware.
- `api.Operation`: The operation (route only).
- `rez.MiddlewareNext`: Invoke the next handler (middleware only).

There are a few other methods to get other injectable values.
1. Use `rez.Site.Scope` to set global values and providers.
2. Implement `rez.Injectable`.
3. Use `rez.Router.DefineBody(bodies...)` to define types that will only be used as arguments that should come from the request body.
4. Use `rez.Router.DefinePath(paths...)` to define types that will only be used as arguments that should come from the request path parameters.
5. Use `rez.Router.DefineQuery(queries...)` to define types that will only be used as arguments that should come from the request query parameters.
6. Use `rez.Router.DefineHeader(headers...)` to define types that will only be used as arguments that should come from the request headers.
7. Use `*deps.Scope` as an argument in middleware and `Set` or `Provide` other values that the following handlers will be able to receive.

## Router
The [rez.Router](rez.go) is a wrapper of chi.Router where instead of `http.Hander`s and `http.HandlerFunc` you pass in a `func(args) results` which gets its arguments injected, and in the case of middleware is able to provide injected values for routes in the router. The function argument and result types are also inspected to build the OpenAPI documentation.

## Middleware
Middleware in **rez** is also a dependency injected function. The middleware can return nothing or can return an error which if non-nil will be sent as the response. The middleware has a special injected value `rez.MiddlewareNext` which is a function to call if we want to call the next handler. _Any arguments or return types that are identified as headers, queries, paths, request bodies, or responses are added as those objects in all routes that are in the router using the middleware._

Example:
```go
// site.Use(authMiddleware)
func authMiddleware(next rez.MiddlewareNext, r *http.Request) *rez.Unauthorized[string] {
  // Accessing headers this way doesn't add it as a parameter to all routes that use the middleware.
  auth := r.Header.Get("Authorization")
  if auth == "" {
    return rez.NewUnauthorized("No access")
  }
  next()
  return nil
}
```

You can also pass down injectable values to all middlewares and routes which are defined after middleware by setting the value on the scope.

```go
type User struct { ID int }
// site.Use(authMiddleware)
func authMiddleware(next rez.MiddlewareNext, s *deps.Scope) {
  // authenticate user and return error, if successful apply the user to the scope.
  s.Set(User{ID: 23})
  next()
}
type Task struct { ID int, Name string, Done bool }
// set.Get("/tasks", getTasks)
func getTasks(user User) Task[] {
  // get tasks the user can see, we only get here if authMiddleware called next
  return []Task{}
}
```

As mentioned what you reference with middleware could add to the operations that follow. This middleware is an example of authentication where the token is foolishly sent in the query string. All operations that follow will have the specified security scheme, accept `token` as a query parameter, and could respond with `rez.Unauthorized[string]`.

```go
type Auth struct { Token string }

type AuthMiddleware func(s *deps.Scope, next rez.MiddlewareNext, q rez.Query[Auth]) *rez.Unauthorized[string]
// All routes which use this middleware accept this type of security
func (auth AuthMiddleware) APIOperationUpdate(op *api.Operation) {
  op.Security = append(op.Security, map[string][]string{"queryAuth": {}})
}

var authMiddleware AuthMiddleware = func(s *deps.Scope, next rez.MiddlewareNext, q rez.Query[Auth]) *rez.Unauthorized[string] {
  if q.Value.Token == "" {
    return rez.NewUnauthorized("No access")
  }
  s.Set(q.Value)
  next()
  return nil
}
func echoToken(token Auth) Auth {
  return token
}

// Usage
site.Open.AddSecurity("queryAuth", &api.Security{
  Type: api.SecurityTypeApiKey,
  Name: "token",
  In:   api.ParameterInQuery,
})
site.Use(authMiddleware)
site.Get("/token", echoToken)
```

## Inspection

Function arguments are inspected to determine what path parameters, query parameters, headers, and body is used by a route. See [Dependency Injection](#dependency-injection) for more details on that. The types detected are converted into `api` objects and are added to the OpenAPI document and referenced in the path & operations in the path. The function return arguments are inspected for possible responses - most of the time these return types will be pointers for routes which can have multiple response types (or no specific response type). If the return type implements `rez.HasStatus` that is where the status code is pulled from. If the return type does not it's assumed to be a possible OK (200) result. The schemas built from the argument and return types are built once and can be controlled using various functions and interfaces. If the type is a struct then `json` and `api` tags can control the field visibility or schema options. See [Documentation](#documentation) for additional details on how to control the documentation & validation that is generated.

## Validation

Validation in rez is done if enabled and only for certain schema fields and after the data is marshalled into values. So any invalid type errors will not be triggered by the validation but when the JSON is parsed. General validation options can be applied per type, validation can be enabled or disabled for any router, and types can have custom validation code that takes over the validation process or runs after the validation process. If validation fails the error is returned to the user. How those validations are sent to the user can be controlled by calling `rez.Router.SetErrorHandler`.

- `rez.Router.EnabledValidation(bool)` enables or disables validation in this router and any sub-routers created after this call. By default validation is not enabled.
- `rez.Router.SetValidationOptions(any,ValidationOptions)` sets the validation options for the given type, which controls if validation is skipped, if format is enforced, or if specifying deprecated values triggers a validation error.
- `rez.CanValidateFull` if a type implements this it handles all validation logic.
- `rez.CanValidatePost` if a type implements this it will do additional validation logic after other validation logic has been done.
- `rez.Injectable` if a type implements this it must implement an `APIValidate` method.

The following schema fields are used during validation:
- `MultipleOf`, `Maximum`, `Minimum`, `ExclusiveMaximum`, `ExclusiveMinimum` are used for any int or float types.
- `MaxLength`, `MinLength` are used for string types.
- `Deprecated`, `Nullable`, `Pattern`, `Format`, `Enum`, `OneOf`, `AllOf`, `AnyOf`, `Not` are used for all types.
- `MinItems`, `MaxItems`, `Items`, `UniqueItems` are used for array and slice types.
- `MinProperties`, `MaxProperties`, `AdditionalProperties` are used for map types.
- `Properties`, `Required` are used for struct types.


## Documentation

Documentation is control by various ways on the types themselves or through router methods.

- `api.HasName`
A type's documented name is the name of the type in the GO code, but there might be collisions. If there are collisions the OpenAPI built will have schema names that include the types pkg path to be unique. To avoid those potentially lengthy names you can implement `api.HasName` like so:
```go
// tasks folder
type Search struct { Name string }
func (Search) APIName() string { return "TaskSearch" }
```
- `api.Description`
A request or response's description can be specified on tyhe type by implementing this interface. Using the code above.
```go
func (Search) APIDescription() string { return "This is used to determine what Tasks to return." }
```
- `api.HasBaseSchema`
A type's schema will be dynamically determined, but implementing this interface will provide the schema building logic with a starting point. You can define the preferred schema properties.
```go
func (Search) APIBaseSchema() *api.Schema {
  return &api.Schema{
    Title:       "Task Search",
    Description: "This is used to determine what Tasks to return.",
    Example      api.Any(Search{Name: "homework"}),
  }
}
```
- `api.HasFullSchema`
A type's schema will only be determined by what's returned, no further inspection is done. This is useful if you want to use some of the built-in struct types that support different formats.
```go
type Timestamp time.Time
func (Timestamp) APIFullSchema() *api.Schema {
  return &api.Schema{
    Type:    api.DataTypeString,
    Format:  "date-time",
    Pattern: `\\d{4}-\d\d-\d\dT\d\d:\d\d:\d\d+\d\d:\d\d`,
    Example: api.Any("2018-11-13T20:20:39+00:00"),
  }
}
```
- `api.HasEnum`
A type can accept only a handful of values.
```go
type TodoAction string
const (
  TodoActionArchive  TodoAction = "archive"
  TodoActionDelete   TodoAction = "delete"
  TodoActionComplete TodoAction = "complete"
)
func (TodoAction) APIEnum() []any {
  return []any{TodoActionArchive, TodoActionDelete, TodoActionComplete}
}
```
- `api.HasExamples`
A type can provide several named examples for a given content type.
```go
func (Search) APIExamples(contentType api.ContentType) api.Examples {
  return api.Examples{
    "All tasks": api.Example{
      Summary: "This search will return all tasks the user can see.",
      Value:   api.Any(Search{}),
    },
    "Tasks with 'homework' in the name": api.Example{
      Summary: "This search will return all tasks with 'homework' in the name.",
      Value:   api.Any(Search{Name: "homework"}),
    },
  }
}
```
- `api.HasExample`
A type that can provide a single example for a type.
```go
func (Search) APIExample() *any {
  return api.Any(Search{Name: "homework"})
}
```
- `api.HasOperation`
A route function that has the operation fully defined here and no inspection needs to be done on the arguments or return types.
```go
type GetTask func(id string) *Task
func (GetTask) APIOperation() api.Operation {
  return api.Operation{
    Tags:        []string{"Task"},
    Summary:     "Get the task with the given ID",
    OperationID: "GET_TASK_BY_ID",
    Parameters:  []api.Parameter{{
      Name:     "id",
      In:       api.ParameterInPath,
      Required: true,
      Schema:   &api.Schema{Type: api.TypeString},
      Example:  api.Any("87y34"),
    }},
    Responses:  api.Responses{
      "200":    &api.Response{
        Description: "The task exists and has these values",
        Content:     api.Contents{
          api.ContentTypeJSON: &api.MediaType{
            Schema: site.Open.GetSchema(reflect.TypeOf(Task{})),
          },
        },
      },
    },
  }
}
```
- `api.HasOperationUpdate`
A route function that modifies the operation inspected after its done inspection.
```go
type GetTask func(id string) *Task
func (GetTask) APIOperationUpdate(op *api.Operation) {
  op.OperationID = "GET_TASK_BY_ID"
}
```
- `rez.Site.Open` is a reference to `api.Builder` which has a `Document` field which can be modified. This is the base document to use before building the final `api.Document`.
- `rez.Router` has a few methods to assist in documentation:
  - `GetOperations() *api.Operation` returns a reference to the operation template that has accumulated at this point in the router. Sub routers inherit this. Middlewares add to it.
  - `SetOperations(api.Operation)` sets the operation template in its entirety, overwriting what has been built so far.
  - `UpdateOperations(api.Operation)` merges in the fields set on the given operation into the operation template of this router.
  - `GetPath(pattern) *api.Path` returns a reference to the path with the given pattern. Defining methods will add operations to this path. If the path has not been defined yet nil is returned.
  - `CreatePath(pattern) *api.Path` returns a reference to the path with the given pattern, creating it if need be.
  - `UpdatePath(pattern, api.Path)` merges in the fields set on the given path with the path defined at the given pattern - creating it if need be.
  - `SetTags(tags []string)` sets the tags on the operation template to this value.
  - `AddTags(tags)` adds the tags to the operation template.
  - `SetResponses(api.Responses)` sets the responses for all operations defined after. This overwrites any responses specified previously by the user or middlewares.
  - `AddResponses(api.Responses)` adds the responses to the operation template.
  - `AddResponse(code, api.Response)` adds the response to the operation template.
  - `HandleFunc(pattern, fn, ...api.Operation) *api.Path` can accept zero or more operation definitions to merge into the operations defined at this path - and the reference to the path at the pattern is returned.
  - `"method"(pattern, fn, ...api.Operation) *api.Operation` is a method with the name of any of the HTTP methods which adds this method to the path with the pattern and merges in any given operations with the operation template and then returns the reference to the final built operation for this route.
- Struct tags. Fields on a struct can specify the `api` tag which is a comma-delimited list of key=value or flags. If you need to use a comma in a value you can escape it like `\,`.
  - `title` ex: `api:"title=A person's address"` (see `api.Schema.Title`)
  - `desc` or `description` ex: `api:"desc=The ten digit home phone number."` (see `api.Schema.Description`)
  - `format` ex: `api:"format=email"` (see `api.Schema.Format`)
  - `pattern` ex: `api:"pattern=\d+"` (see `api.Schema.Pattern`)
  - `deprecated` ex: `api:"deprecated"` (see `api.Schema.Deprecated`)
  - `required` ex: `api:"required"` (see `api.Schema.Nullable`)
  - `null` or `nullable` ex: `api:"null"` (see `api.Schema.Nullable`)
  - `readonly` ex: `api:"readonly"` (see `api.Schema.ReadOnly`)
  - `writeonly` ex: `api:"writeonly"` (see `api.Schema.WriteOnly`)
  - `enum` ex: `api:"enum=1|2|3"` (see `api.Schema.Enum`)
  - `minlength` ex: `api:"minlength=6"` (see `api.Schema.MinLength`)
  - `maxlength` ex: `api:"maxlength=6"` (see `api.Schema.MaxLength`)
  - `minitems` ex: `api:"minitems=6"` (see `api.Schema.MinItems`)
  - `maxitems` ex: `api:"maxitems=6"` (see `api.Schema.MaxItems`)
  - `multipleof` ex: `api:"multipleof=2"` (see `api.Schema.MultipleOf`)
  - `min` or `minimum` ex: `api:"min=1"` (see `api.Schema.Minimum`)
  - `max` or `maximum` ex: `api:"max=1"` (see `api.Schema.Maximum`)
  - `exclusivemaximum` or `exclusivemax` ex: `api:"exclusivemax=true"` (see `api.Schema.ExclusiveMaximum`)
  - `exclusiveminimum` or `exclusivemin` ex: `api:"exclusivemin"` (see `api.Schema.ExclusiveMinimum`)

## Site

`rez.Site` is the implementation of router that must be created with `rez.New(chi.Router)`. Site has a few additional methods:
- `BuildDocument() *api.Document` returns the built document based on the routes and middlewares defined thus far.
- `BuildJSON() []byte` calls `BuildDocument` and marshals it to JSON.
- `ServeOpenJSON(patten)` serves the `BuildJSON` to a GET route at the defined pattern. This gets called by the other `Serve` document related endpoints if it was not called yet with a default pattern of `openapi3.json`.
- `ServeSwaggerUI(pattern,options)` serves an HTML page at the given pattern which presents the SwaggerUI which points to the OpenAPI document JSON.
- `ServeRedoc(pattern)` serves an HTML page at the given pattern which presents the Redoc which points to the OpenAPI document JSON.
- `Listen(addr)` starts the site and blocks until it stops.
- `Run()` starts the site but looks at the CLI args for a `--host` argument to specify the port. It defaults to `:80`.
- `PrintPaths()` prints an ASCII grid to the console with the paths described in the site at this point in time. Includes the "Method", "URL", and "About" if any summary or descriptions are given. Example output:
```
┌───────┬───────────┬────────────────────┐
│Method │URL        │About               │
├───────┼───────────┼────────────────────┤
│GET    │/task/{id} │Get task by id      │
├───────┼───────────┼────────────────────┤
│DELETE │/task/{id} │Delete task by id   │
├───────┼───────────┼────────────────────┤
│GET    │/auth      │Get current session │
├───────┼───────────┼────────────────────┤
│POST   │/auth      │Login               │
├───────┼───────────┼────────────────────┤
│DELETE │/auth      │Logout              │
└───────┴───────────┴────────────────────┘
```
 readmeEtag: '"0773d99603a133802817e7f684bd74d335be605b"' readmeLastModified: Fri, 02 May 2025 12:58:46 GMT repositoryId: 569863233 description: >- REST (easy) framework in Go with out of the box OpenAPI generation, validation, generics, and much more created: '2022-11-23T19:38:15Z' updated: '2025-05-06T14:04:11Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: ClickerMonkey logo: https://avatars.githubusercontent.com/u/421233?v=4 license: MIT repoEtag: '"46867457cdf906779599f8bc690c5840d61b777776da3fa9704442bff7a05ed0"' repoLastModified: Tue, 06 May 2025 14:04:11 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/n1215/openapi-http-foundation-validator v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWh0dHAtZm91bmRhdGlvbi12YWxpZGF0b3IKT3BlbkFQSSh2MykgVmFsaWRhdG9ycyBmb3IgU3ltZm9ueSBodHRwLWZvdW5kYXRpb24sIHVzaW5nIGBsZWFndWUvb3BlbmFwaS1wc3I3LXZhbGlkYXRvcmAgYW5kIGBzeW1mb255L3Bzci1odHRwLW1lc3NhZ2UtYnJpZGdlYC4KCiMjIFJlcXVpcmVtZW50cwotIFBIUCA+PSA4LjEKCiMjIEluc3RhbGxhdGlvbgoKYGBgc2hlbGwKY29tcG9zZXIgcmVxdWlyZSBuMTIxNS9vcGVuYXBpLWh0dHAtZm91bmRhdGlvbi12YWxpZGF0b3IKYGBgCgojIyBVc2FnZQoKIyMjIDEuIGluc3RhbGwgUFNSLTE3IEhUVFAgRmFjdG9yeSBpbXBsZW1lbnRhdGlvbi4KLSBZb3UgY2FuIHVzZSBhbnkgaW1wbGVtZW50YXRpb24gb2YgUFNSLTE3IEhUVFAgRmFjdG9yeS4KICAtIGV4LiBgbnlob2xtL3BzcjdgCgpgYGBzaGVsbApjb21wb3NlciByZXF1aXJlIG55aG9sbS9wc3I3CmBgYAoKIyMjIDIuIGNyZWF0ZSBodHRwIG1lc3NhZ2UgZmFjdG9yeQoKYGBgcGhwCiRwc3IxN0ZhY3RvcnkgPSBuZXcgXE55aG9sbVxQc3I3XEZhY3RvcnlcUHNyMTdGYWN0b3J5KCk7Ci8qKiBAdmFyIFxTeW1mb255XEJyaWRnZVxQc3JIdHRwTWVzc2FnZVxIdHRwTWVzc2FnZUZhY3RvcnlJbnRlcmZhY2UgJGh0dHBNZXNzYWdlRmFjdG9yeSAqLwokaHR0cE1lc3NhZ2VGYWN0b3J5ID0gbmV3IFxTeW1mb255XEJyaWRnZVxQc3JIdHRwTWVzc2FnZVxGYWN0b3J5XFBzckh0dHBGYWN0b3J5KAogICAgc2VydmVyUmVxdWVzdEZhY3Rvcnk6ICRwc3IxN0ZhY3RvcnksCiAgICBzdHJlYW1GYWN0b3J5OiAkcHNyMTdGYWN0b3J5LAogICAgdXBsb2FkZWRGaWxlRmFjdG9yeTogJHBzcjE3RmFjdG9yeSwKICAgIHJlc3BvbnNlRmFjdG9yeTogJHBzcjE3RmFjdG9yeQopOwpgYGAKCiMjIyAzLiBjcmVhdGUgdmFsaWRhdG9yIGJ1aWxkZXIKCi0gQSBidWlsZGVyIGNhbiBiZSBjcmVhdGVkIGZyb20gWUFNTCBmaWxlLCBZQU1MIHN0cmluZywgSlNPTiBmaWxlLCBvciBKU09OIHN0cmluZy4KLSBZb3UgY2FuIHVzZSBQU1ItMTYgc2ltcGxlIGNhY2hlIGluc3RlYWQgb2YgUFNSLTYgQ2FjaGUuCgojIyMjIGV4YW1wbGUxCmBgYHBocAovKiogQHZhciBcTjEyMTVcT3BlbkFwaVZhbGlkYXRpb25cSHR0cEZvdW5kYXRpb25cVmFsaWRhdG9yQnVpbGRlciAkdmFsaWRhdG9yQnVpbGRlciAqLwokdmFsaWRhdG9yQnVpbGRlciA9IChuZXcgXE4xMjE1XE9wZW5BcGlWYWxpZGF0aW9uXEh0dHBGb3VuZGF0aW9uXFZhbGlkYXRvckJ1aWxkZXIoJHBzcjE3RmFjdG9yeSkpCiAgICAtPmZyb21ZYW1sRmlsZSgnL3BhdGgvdG8vb3BlbmFwaS55YW1sJykKICAgIC0+c2V0Q2FjaGUobmV3IFlvdXJQc3I2Q2FjaGUoKSwgODY0MDApOwpgYGAKCgojIyMjIGV4YW1wbGUyCmBgYHBocAovKiogQHZhciBcTjEyMTVcT3BlbkFwaVZhbGlkYXRpb25cSHR0cEZvdW5kYXRpb25cVmFsaWRhdG9yQnVpbGRlciAkdmFsaWRhdG9yQnVpbGRlciAqLwokdmFsaWRhdG9yQnVpbGRlciA9IChuZXcgXE4xMjE1XE9wZW5BcGlWYWxpZGF0aW9uXEh0dHBGb3VuZGF0aW9uXFZhbGlkYXRvckJ1aWxkZXIoJHBzcjE3RmFjdG9yeSkpCiAgICAtPmZyb21Kc29uRmlsZSgnL3BhdGgvdG8vb3BlbmFwaS5qc29uJykKICAgIC0+c2V0U2ltcGxlQ2FjaGUobmV3IFlvdXJQc3IxNkNhY2hlKCksIDM2MDApOwpgYGAKCiMjIyA0LiBnZXQgdmFsaWRhdG9ycyBmcm9tIGJ1aWxkZXIKCmBgYHBocAovKiogQHZhciBcTjEyMTVcT3BlbkFwaVZhbGlkYXRpb25cSHR0cEZvdW5kYXRpb25cVmFsaWRhdG9ycyAkdmFsaWRhdG9ycyAqLwokdmFsaWRhdG9ycyA9ICR2YWxpZGF0b3JCdWlsZGVyLT5nZXRWYWxpZGF0b3JzKCk7CmBgYAoKIyMjIDUuIHZhbGlkYXRlIHJlcXVlc3QKCmBgYHBocAovKiogQHZhciBcU3ltZm9ueVxDb21wb25lbnRcSHR0cEZvdW5kYXRpb25cUmVxdWVzdCAkcmVxdWVzdCAqLwovKiogQHZhciBcTjEyMTVcT3BlbkFwaVZhbGlkYXRpb25cSHR0cEZvdW5kYXRpb25cUmVxdWVzdFZhbGlkYXRvckludGVyZmFjZSAkcmVxdWVzdFZhbGlkYXRvciAqLwokcmVxdWVzdFZhbGlkYXRvciA9ICR2YWxpZGF0b3JzLT5nZXRSZXF1ZXN0VmFsaWRhdG9yKCk7CiRyZXF1ZXN0VmFsaWRhdG9yLT52YWxpZGF0ZSgkcmVxdWVzdCk7CmBgYAoKIyMjIDYuIHZhbGlkYXRlIHJlc3BvbnNlCgpgYGBwaHAKLyoqIEB2YXIgXFN5bWZvbnlcQ29tcG9uZW50XEh0dHBGb3VuZGF0aW9uXFJlc3BvbnNlICRyZXNwb25zZSAqLwovKiogQHZhciBcTjEyMTVcT3BlbkFwaVZhbGlkYXRpb25cSHR0cEZvdW5kYXRpb25cUmVzcG9uc2VWYWxpZGF0b3JJbnRlcmZhY2UgJHJlc3BvbnNlVmFsaWRhdG9yICovCiRyZXNwb25zZVZhbGlkYXRvciA9ICR2YWxpZGF0b3JzLT5nZXRSZXNwb25zZVZhbGlkYXRvcigpOwokcmVzcG9uc2VWYWxpZGF0b3ItPnZhbGlkYXRlKAogICAgbmV3IFxOMTIxNVxPcGVuQXBpVmFsaWRhdGlvblxPcGVyYXRpb25BZGRyZXNzKCcvcGF0aCcsICdHRVQnKSwKICAgICRyZXNwb25zZQopOwpgYGAKCiMjIFVzYWdlIGZvciBMYXJhdmVsCnNlZSBodHRwczovL2dpdGh1Yi5jb20vbjEyMTUvb3BlbmFwaS1sYXJhdmVsLXZhbGlkYXRvcgo= readmeEtag: '"92247ff9ffda8c419373a62499a32d5de74e66a2"' readmeLastModified: Fri, 22 Sep 2023 12:59:36 GMT repositoryId: 404792127 description: >- OpenAPI(v3) Validators for Symfony http-foundation, using `league/openapi-psr7-validator` and `symfony/psr-http-message-bridge`. created: '2021-09-09T16:22:40Z' updated: '2024-11-14T05:06:35Z' language: PHP archived: false stars: 2 watchers: 1 forks: 0 owner: n1215 logo: https://avatars.githubusercontent.com/u/2157593?v=4 license: MIT repoEtag: '"468ab79b882deb96492845fe7f3e9773bee0a768bf93629b6cb15cb1481d124a"' repoLastModified: Thu, 14 Nov 2024 05:06:35 GMT foundInMaster: true category: Data Validators id: e5ae30ae45d597e93d4c0950e4132b19 - source: openapi3 tags repository: https://github.com/bhagyas/alfresco-swagger-gen v3: true repositoryMetadata: base64Readme: >- IyBhbGZyZXNjby1zd2FnZ2VyLWdlbgpPcGVuQVBJIChTd2FnZ2VyKSBkZWZpbml0aW9uIGdlbmVyYXRvciBmb3IgQWxmcmVzY28gV2Vic2NyaXB0IEZpbGVzCgojIyBJbnN0YWxsYXRpb24KCiMjIyBWaWEgTlBNIApSdW4gYG5wbSBpbnN0YWxsIC1nIGFsZnJlc2NvLXN3YWdnZXItZ2VuYAoKT2ZmaWNpYWwgTlBNIFBhY2thZ2UgaXMgYXQgaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvYWxmcmVzY28tc3dhZ2dlci1nZW4KCiMjIyBWaWEgQ2xvbmluZyAKQ2xvbmUgdGhlIHJlcG9zaXRvcnkgYW5kIHJ1biBgbnBtIGxpbmtgIGZyb20gdGhlIHByb2plY3Qgcm9vdC4KCiMjIFVzYWdlCgojIyMgR2VuZXJhdGluZyBPcGVuQVBJIChTd2FnZ2VyKSBEb2N1bWVudApSZWZlciB0byB0aGUgY29tbWFuZCBsaW5lIHN5bnRheCBiZWxvdy4KCmBgYAphbGZyZXNjby1zd2FnZ2VyLWdlbiAtLWhlYWRlciAuLi9teS1wcm9qZWN0L215LWhlYWRlci55YW1sIFwKICAgICAgICAgICAgICAgICAgICAgLS1kZXN0aW5hdGlvbiAuL215LXByb2plY3Qvc3dhZ2dlci1kZWZpbml0aW9ucy55YW1sIFwKICAgICAgICAgICAgICAgICAgICAgLS1zY2FuUGF0aCAuL215LXByb2plY3Qvc3JjICAKYGBgICAgICAgICAgICAgICAgICAgCgojIyMgUGFyYW1ldGVyIFJlZmVyZW5jZQp8IFBhcmFtZXRlciB8IERlc2NyaXB0aW9uIHwKfC0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS18CnwgaGVhZGVyICB8IFNwZWNpZnkgYSBjdXN0b20gaGVhZGVyIGZpbGUgdG8gYmUgdXNlZCBmb3IgT3BlbkFQSSAoU3dhZ2dlcikgZGVmaW5pdGlvbiBnZW5lcmF0aW9uIHwKfCBkZXN0aW5hdGlvbiB8IERlc3RpbmF0aW9uIHBhdGggKHRhcmdldCBmaWxlKSBmb3IgdGhlIGdlbmVyYXRlZCBBUEkgZGVmaW5pdGlvbiBmaWxlLiB8Cnwgc2NhblBhdGggfCBSb290IHBhdGggZm9yIHNjYW5uaW5nIGZvciBBbGZyZXNjbyB3ZWJzY3JpcHQgZGVzY3JpcHRvciBmaWxlcyB8CgoqIEFuIGV4YW1wbGUgaGVhZGVyIGZpbGUgaXMgZm91bmQgaW4gYC4vdGVtcGxhdGVzL2RlZmF1bHRfaGVhZGVyLnlhbWxgIHdpdGhpbiB0aGUgc291cmNlLgoKIyMjIFJ1bm5pbmcgU3dhZ2dlci1VSQoKYWxmcmVzY28tc3dhZ2dlci1nZW4gYWxsb3dzIHlvdSB0byBydW4gU3dhZ2dlci1VSSB3aXRoIHlvdXIgZ2VuZXJhdGVkIE9wZW5BUEkgZG9jdW1lbnQuIEl0IGNvcGllcyB0aGUgc3BlY2lmaWVkIE9wZW5BUEkgZG9jdW1lbnQgdG8gYSBjdXN0b20gZm9sZGVyIGFuZCBtb3VudHMgaXQgdG8gYSBkb2NrZXIgaW1hZ2UgcnVubmluZyBTd2FnZ2VyLVVJLgoKVGhlIGZvbGxvd2luZyBjb21tYW5kIHJ1bnMgdGhlIGJ1aWx0LWluIGBkb2NrZXItY29tcG9zZWAgZmlsZSBhbmQgb3BlbnMgdGhlIGJyb3dzZXIgc2hvd2luZyB0aGUgU3dhZ2dlci1VSS4KCmBgYAphbGZyZXNjby1zd2FnZ2VyLWdlbiB1aSAtLWRlc3RpbmF0aW9uIC4vbXktcHJvamVjdC9zd2FnZ2VyLWRlZmluaXRpb25zLnlhbWwgXAomJiBvcGVuIGh0dHA6Ly9sb2NhbGhvc3Q6ODAKYGBgCgoKCiMjIE1hcHBpbmcKIyMjIEhvdyB0byBtYXAgIFJlc3BvbnNlIFNjaGVtYXMKQWRkIGA8eC1yZXNwb25zZS1zY2hlbWE+TXlSZXNwb25zZVNjaGVtYTwveC1yZXNwb25zZS1zY2hlbWE+YCB0byB5b3VyIHdlYnNjcmlwdCBkZXNjcmlwdG9yIGZpbGUuCgpgTXlSZXNwb25zZVNjaGVtYWAgY2FuIGJlIGRlZmluZWQgaW4geW91ciBoZWFkZXIgZmlsZS4KCiMjIEF1dGhvcgotIEJoYWd5YSBTaWx2YSAtIFtAYmhhZ3lhc10oaHR0cHM6Ly9saW5rZWRpbi5jb20vaW4vYmhhZ3lhcyk= readmeEtag: '"caaf2ec66903f2ef37335a39b5dd3b52dd1eabe0"' readmeLastModified: Wed, 04 Aug 2021 15:46:03 GMT repositoryId: 203178879 description: OpenAPI (Swagger) definition generator for Alfresco Webscript Files created: '2019-08-19T13:30:53Z' updated: '2023-10-05T21:23:05Z' language: JavaScript archived: false stars: 2 watchers: 2 forks: 0 owner: bhagyas logo: https://avatars.githubusercontent.com/u/750003?v=4 repoEtag: '"18535d28a2d7cccdc7655e17a17c7c8920ad98045fb55780b0aeea0fdb8aa33a"' repoLastModified: Thu, 05 Oct 2023 21:23:05 GMT foundInMaster: true category: - Description Validators - Parsers id: 57c04bf601ff68939256e9adabb1b734 - source: openapi3 tags repository: https://github.com/vinoth5595/phone-bridge v3: true repositoryMetadata: base64Readme: >- IyBQaG9uZSBCcmlkZ2UgUHJvamVjdCAoTWljcm9zZXJ2aWNlcykKCmRvY2tlci1jb21wb3NlIHVwIChUbyBzdGFydCBNb25nb0RCICYgQXBhY2hlIEthZmthKQoKZG9ja2VyLWNvbXBvc2UgZG93biAoVG8gc3RvcCBNb25nb0RCICYgQXBhY2hlIEthZmthKQoKU3RhcnQgdGhlIGFwcGxpY2F0aW9uIGluIHRoZSBiZWxvdyBvcmRlci4KCnwgQXBwbGljYXRpb24gIHwgRGVzY3JpcHRpb24gfCBQb3J0IHwKfCAtLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0gfAp8IHBob25lYnJpZGdlLWRpc2NvdmVyeSAgfCBQaG9uZSBicmlkZ2UgZXVyZWthIHNlcnZlciAgfCA4NzYxICB8CnwgcGhvbmVicmlkZ2UtYWRtaW4gIHwgUGhvbmUgYnJpZGdlIGFkbWluICB8IDgwOTMgIHwKfCBwaG9uZWJyaWRnZS1jb25maWctc2VydmVyICB8IFBob25lIGJyaWRnZSBjb25maWcgc2VydmVyICB8IDg4ODggIHwKfCBwaG9uZWJyaWRnZS1wb3J0YWwgIHwgUG9ydGFsIHNlcnZpY2UgJiBjYW1wYWlnbiBzZXJ2aWNlICB8IDgwODAgIHwKfCBwaG9uZWJyaWRnZS1hdXRoc2VydmVyICB8IFBob25lIGJyaWRnZSBhdXRoIHNlcnZlciAgfCA4MDgxICB8CnwgcGhvbmVicmlkZ2UtZ2F0ZXdheS1zZXJ2ZXIgIHwgUGhvbmUgYnJpZGdlIGdhdGV3YXkgc2VydmVyICB8IDgwODUgIHwKfCBwaG9uZS1icmlkZ2UtdWkgIHwgVXNlciBpbnRlcmZhY2UgZm9yIHBob25lIGJyaWRnZSBhcHBsaWNhdGlvbiAgfCA0MjAwICB8CnwgcGhvbmVicmlkZ2UtY29tbW9uICB8IFByb2plY3Qgd2l0aCBjb21tb24gZmlsZXMgIHwgTkEgIHw= readmeEtag: '"b71052d9cd74602be4f65147a01d21c446d97ed9"' readmeLastModified: Tue, 03 May 2022 07:07:58 GMT repositoryId: 243821819 description: Phone Bridge Project created: '2020-02-28T17:48:05Z' updated: '2022-05-03T07:07:26Z' language: SCSS archived: false stars: 2 watchers: 1 forks: 1 owner: Vinoth5595 logo: https://avatars.githubusercontent.com/u/32703692?v=4 repoEtag: '"c51cb55bff0631fe47cc699983a8637bc8988db604be989e428fe2ecdb0d0513"' repoLastModified: Tue, 03 May 2022 07:07:26 GMT foundInMaster: true category: - Documentation - Server Implementations id: b9b0dcb83f765a04a5eed22c05c48a46 - source: openapi3 tags repository: https://github.com/lionelsu/store-manager-api v3: true id: 04ee39c98b6167d24883073bc0878fd5 repositoryMetadata: base64Readme: >- PCEtLSBFc3RlIMOpIHVtIGNvbWVudMOhcmlvOiBvbWl0aXIgb3MgdMOzcGlkb3MgcmVkdW5kYW50ZXMgLS0+CjwhLS0gICoqfCBbQnJhemlsXShSRUFETUUubWQpIHwgW2FzZGZdKFJFQURNRV9lbi5tZCkgfCoqIC0tPgoKIyBTdG9yZSBNYW5hZ2VyCgpbRG9jdW1lbnRhw6fDo28gZGEgQVBJIG5vIFBvc3RtYW5dKGh0dHBzOi8vZG9jdW1lbnRlci5nZXRwb3N0bWFuLmNvbS92aWV3LzMwMTU5MzU1LzJzOVlKZ1RMSEIpCgpPIFN0b3JlIE1hbmFnZXIgQVBJIMOpIHVtYSBzb2x1w6fDo28gY29tcGxldGEgZGUgZ2VyZW5jaWFtZW50byBkZSB2ZW5kYXMgcG9yIGRyb3Agc2hpcHBpbmcuIEVzdGEgQVBJIG9mZXJlY2Ugb3BlcmHDp8O1ZXMgZGUgQ1JVRCAoQ3JlYXRlLCBSZWFkLCBVcGRhdGUsIERlbGV0ZSkgcGFyYSBmYWNpbGl0YXIgbyBnZXJlbmNpYW1lbnRvIGRlIHZlbmRhcy4gRGVzZW52b2x2aWRhIGNvbSBmb2NvIG5hIHF1YWxpZGFkZSBlIHNlZ3VpbmRvIGEgbWV0b2RvbG9naWEgVEREIChUZXN0LURyaXZlbiBEZXZlbG9wbWVudCksIGEgQVBJIGFkb3RhIHVtYSBhcnF1aXRldHVyYSBlbSBjYW1hZGFzIGNvbnNpc3RlbnRlIGNvbSBNb2RlbCwgU2VydmljZSBlIENvbnRyb2xsZXIgKE1TQykuCgo8ZGV0YWlscz4KCjxzdW1tYXJ5PkZlcnJhbWVudGFzIFV0aWxpemFkYXM8L3N1bW1hcnk+CgotICoqVGVjbm9sb2dpYXMgUHJpbmNpcGFpczoqKgogIC0gTm9kZS5qcwogIC0gRXhwcmVzcy5qcwogIC0gTXlTUUwgU2VydmVyCgotICoqVGVzdGVzOioqCiAgLSBNb2NoYQogIC0gQ2hhaQogIC0gU2lub24KCi0gKipEb2N1bWVudGHDp8OjbzoqKgogIC0gU3dhZ2dlcgogIC0gUG9zdG1hbgoKPC9kZXRhaWxzPgoKIyMgUHLDqS1SZXF1aXNpdG9zCgpVdGlsaXplIG8gRG9ja2VyOgoKLSBbRG9ja2VyICYgRG9ja2VyIENvbXBvc2VdKGh0dHBzOi8vZG9jcy5kb2NrZXIuY29tL2NvbXBvc2UvKQoKPCEtLSAjIyBGZWF0dXJlcyAtLT4KIyMgSW5zdGFsYcOnw6NvCgoxLiBDbG9uYXIgbyBSZXBvc2l0w7NyaW8KCiAgICBQcmltZWlybywgY29waWUgb3UgY2xvbmUgZXN0ZSByZXBvc2l0w7NyaW8gcGFyYSBvIHNldSBzaXN0ZW1hIGxvY2FsIHVzYW5kbyBvIEdpdDoKCiAgICBgYGBiYXNoCiAgICBnaXQgY2xvbmUgZ2l0QGdpdGh1Yi5jb206bGlvbmVsc3Uvc3RvcmUtbWFuYWdlci1hcGkuZ2l0ICYmIGNkIHN0b3JlLW1hbmFnZXItYXBpCiAgICBgYGAKCjIuIEluaWNpYXIgbyBDb250w6ppbmVyIERvY2tlcgoKICAgIFV0aWxpemUgbyBEb2NrZXIgQ29tcG9zZSBwYXJhIGluaWNpYXIgbyBjb250w6ppbmVyIGRvIFN0b3JlIE1hbmFnZXI6CgogICAgYGBgYmFzaAogICAgZG9ja2VyIGNvbXBvc2UgdXAgLWQKICAgIGBgYAoKMy4gQWNlc3NlIGEgZG9jdW1lbnRhw6fDo28gZGEgQVBJIG5vIG5hdmVnYWRvcjoKCiAgICBgYGBodHRwCiAgICBodHRwOi8vMTI3LjAuMC4xOjMwMDEvdjEvZG9jcy8KICAgIGBgYAoKIyMgVXNvCgpQYXJhIGludGVyYWdpciBjb20gbyBTdG9yZSBNYW5hZ2VyLCB2b2PDqiBwb2RlIHVzYXIgb3Mgc2VndWludGVzIGNvbWFuZG9zOgoKVmlzdWFsaXphw6fDo28gZGUgTG9ncwoKYGBgYmFzaApkb2NrZXIgbG9ncyAtbiAxMCAtZiBzdG9yZV9tYW5hZ2VyCmBgYAoKTGluaGEgZGUgQ29tYW5kbyBJbnRlcmF0aXZhLCB1c2FkYSBwYXJhICoqW3Rlc3Rhcl0oI3Rlc3RlcykqKiBhIGFwbGljYcOnw6NvCgpgYGBiYXNoCmRvY2tlciBleGVjIC1pdCBzdG9yZV9tYW5hZ2VyIGJhc2gKYGBgCgpSZXNldGFyIG8gQmFuY28gZGUgRGFkb3MgKGRldmUgc2VyIGV4ZWN1dGFkbyBkZW50cm8gZGEgKipbbGluaGEgZGUgY29tYW5kbyBpbnRlcmF0aXZhXSgjdXNvKSoqKQoKYGBgYmFzaApucG0gcnVuIGRiOnJlc2V0CmBgYAoKIyMgUm90YXMgZGEgQVBJCgoqKlByb2R1dG9zOioqCgotICoqYFBPU1QgL3Byb2R1Y3RzYCoqOiBDYWRhc3RyYSB1bSBub3ZvIHByb2R1dG8uCi0gKipgR0VUIC9wcm9kdWN0c2AqKjogUmV0b3JuYSB0b2RvcyBvcyBwcm9kdXRvcyBjYWRhc3RyYWRvcy4KLSAqKmBHRVQgL3Byb2R1Y3RzLzppZGAqKjogUmV0b3JuYSB1bSBwcm9kdXRvIGVzcGVjw61maWNvIHBlbG8gSUQuCi0gKipgR0VUIC9wcm9kdWN0cy9zZWFyY2hgKio6IFJldG9ybmEgcHJvZHV0b3MgY29tIGJhc2UgbmEgY29uc3VsdGEgZGUgbm9tZS4KLSAqKmBQVVQgL3Byb2R1Y3RzLzppZGAqKjogQXR1YWxpemEgdW0gcHJvZHV0byBwZWxvIElELgotICoqYERFTEVURSAvcHJvZHVjdHMvOmlkYCoqOiBFeGNsdWkgdW0gcHJvZHV0byBwZWxvIElELgoKKipWZW5kYXM6KioKCi0gKipgUE9TVCAvc2FsZXNgKio6IENhZGFzdHJhIHVtYSBub3ZhIHZlbmRhLgotICoqYEdFVCAvc2FsZXNgKio6IFJldG9ybmEgdG9kYXMgYXMgdmVuZGFzIGNhZGFzdHJhZGFzLgotICoqYEdFVCAvc2FsZXMvOmlkYCoqOiBSZXRvcm5hIHVtYSB2ZW5kYSBlc3BlY8OtZmljYSBwZWxvIElELgotICoqYFBVVCAvc2FsZXMvOnNhbGVJZC9wcm9kdWN0cy86cHJvZHVjdElkL3F1YW50aXR5YCoqOiBBdHVhbGl6YSBhIHF1YW50aWRhZGUgZGUgdW0gcHJvZHV0byBlbSB1bWEgdmVuZGEuCi0gKipgREVMRVRFIC9zYWxlcy86aWRgKio6IEV4Y2x1aSB1bWEgdmVuZGEgcGVsbyBJRC4KCjxkZXRhaWxzPgoKICA8c3VtbWFyeT5Db25maWd1cmHDp8Ojbzwvc3VtbWFyeT4KCiAgVmFyacOhdmVpcyBkZSBhbWJpZW50ZSBkZW50cm8gZG8gY29udGFpbmVyOgoKICBgYGBteXNxbAogIE1ZU1FMX1VTRVI6IHJvb3QKICBNWVNRTF9QQVNTV09SRDogcGFzc3dvcmQKICBNWVNRTF9IT1NUTkFNRTogZGIKICBNWVNRTF9QT1JUOiAzMzA2CiAgUE9SVDogMzAwMQogIGBgYAoKPC9kZXRhaWxzPgoKIyMgVGVzdGVzCgotIERlbnRybyBkYSAqKltsaW5oYSBkZSBjb21hbmRvIGludGVyYXRpdmFdKCN1c28pKiosIHZvY8OqIHBvZGUgZXhlY3V0YXIgb3Mgc2VndWludGVzIHRlc3RlczoKCiAgLSBUZXN0ZXMgVW5pdMOhcmlvczoKCiAgYGBgYmFzaAogIG5wbSBydW4gdGVzdDptb2NoYQogIGBgYAoKICAtIENvYmVydHVyYSBkZSBUZXN0ZXM6CgogIGBgYGJhc2gKICBucG0gcnVuIHRlc3Q6Y292ZXJhZ2UKICBgYGAKCiAgLSBUZXN0ZXMgZGUgTXV0YcOnw6NvOgoKICBgYGBiYXNoCiAgbnBtIHJ1biB0ZXN0Om11dGF0aW9uCiAgYGBgCgojIyBIYWJpbGlkYWRlcyBkZXNlbnZvbHZpZGFzCgpVc2VpIG8gYE5vZGUuanNgIGNvbSBvIGBFeHByZXNzLmpzYCBjb21vIGJhc2UgZGEgbWluaGEgYXBsaWNhw6fDo28sIHBlcm1pdGluZG8gY3JpYXIgZmFjaWxtZW50ZSBlbmRwb2ludHMgYEhUVFBgIHBhcmEgYXRlbmRlciDDoHMgbmVjZXNzaWRhZGVzIGRvIHNpc3RlbWEuCgpPIGJhbmNvIGRlIGRhZG9zIGBNeVNRTCBTZXJ2ZXJgIGZvaSBlc2NvbGhpZG8gcGFyYSBhcm1hemVuYXIgZGFkb3MgcmVsYWNpb25hZG9zIGEgcHJvZHV0b3MsIHZlbmRhcyBlIG91dHJvcyBlbGVtZW50b3MgY3J1Y2lhaXMgZG8gc2lzdGVtYS4KCkEgcXVhbGlkYWRlIGRvIGPDs2RpZ28gZm9pIGdhcmFudGlkYSBwb3IgbWVpbyBkZSB0ZXN0ZXMgcmlnb3Jvc29zIHVzYW5kbyBgTW9jaGFgLCBgQ2hhaWAgZSBgU2lub25gLiBFc3NlcyB0ZXN0ZXMgYXZhbGlhcmFtIG1pbnVjaW9zYW1lbnRlIG9zIGVuZHBvaW50cywgc2VydmnDp29zIGUgZnVuw6fDtWVzIHBhcmEgZ2FyYW50aXIgcXVlIHR1ZG8gZnVuY2lvbmFzc2UgY29ycmV0YW1lbnRlLgoKUGFyYSBkb2N1bWVudGFyIGEgYEFQSWAsIHV0aWxpemVpIG8gYFN3YWdnZXJgLCBxdWUgaW5jbHVpIGluZm9ybWHDp8O1ZXMgc29icmUgcm90YXMsIHBhcsOibWV0cm9zIGUgZXhlbXBsb3MgcHLDoXRpY29zLgoKVGFtYsOpbSBkaXNwb25pYmlsaXplaSB1bWEgY29sZcOnw6NvIG5vIGBQb3N0bWFuYCBwYXJhIGZhY2lsaXRhciB0ZXN0ZXMgZSBpbnRlcmHDp8O1ZXMgY29tIGEgYEFQSWAuCgpDb20gZXNzYXMgZXRhcGFzIGNvbmNsdcOtZGFzLCBlc3RvdSBjb25maWFudGUgbmEgZW50cmVnYSBkZSB1bWEgYEFQSWAgc8OzbGlkYSBlIGZ1bmNpb25hbCwgcHJvbnRhIHBhcmEgYXRlbmRlciDDoHMgbmVjZXNzaWRhZGVzIGRvcyB1c3XDoXJpb3MuCg== readmeEtag: '"9590ad8377f9eebb892c3e2ed7b00347604dae65"' readmeLastModified: Sat, 07 Oct 2023 19:06:01 GMT repositoryId: 692222558 description: >- RESTful API offering comprehensive CRUD functionality for sales management, specifically drop shipping. Developed with TDD methodology. created: '2023-09-15T21:04:38Z' updated: '2023-12-02T18:02:08Z' language: JavaScript archived: false stars: 2 watchers: 1 forks: 0 owner: lionelsu logo: https://avatars.githubusercontent.com/u/50457741?v=4 repoEtag: '"18f836249c016a33683bda68b85a769d5057bc80ec5272a1afe48425e7c6b19c"' repoLastModified: Sat, 02 Dec 2023 18:02:08 GMT category: Code Generators foundInMaster: true - source: openapi3 tags repository: https://github.com/sitmcella/openapi-3-object-converter v3: true id: 9a10040c85e27660dbdc71c3606026f6 repositoryMetadata: base64Readme: >- IyBKYXZhIGNsYXNzIGNvbnZlcnNpb24gdG8gT3BlbkFQSSAzIFlBTUwgU2NoZW1hIE9iamVjdAoKVGhpcyBwcm9qZWN0IGNvbnNpc3RzIG9mIGEgSmF2YSBhcHBsaWNhdGlvbiB1c2VkIGZvciBjb252ZXJ0aW5nIGEgSmF2YSBjbGFzcyB0byBhIGZyZWUtZm9ybSBvYmplY3QgaW4gT3BlbkFQSSAzIFlBTUwgClNjaGVtYSBPYmplY3QgZGVmaW5pdGlvbi4KClRoaXMgaW1wbGVtZW50YXRpb24gaXMgYW4gZXh0ZW5zaW9uIG9mIHRoZSBmcmVlLWZvcm0gcXVlcnkgcGFyYW1ldGVyIGRlZmluaXRpb24gaW4gT3BlbkFQSSAzIHNwZWNpZmljYXRpb24uCgpUaGUgT3BlbkFQSSAzIFlBTUwgU2NoZW1hIE9iamVjdCBpcyBpbnNlcnRlZCBpbnRvIGFuIE9wZW5BUEkgMyBkb2N1bWVudCwgY29tcGxpYW50IHdpdGggdGhlIFlBTUwgdmVyc2lvbiAxLjIgZm9ybWF0LgoKVGhlIG9mZmljaWFsIE9wZW5BUEkgMyBkb2N1bWVudGF0aW9uIGlzOiBbT3BlbkFQSSAzLjAuMCBkb2N1bWVudGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMC5tZCkKCiMjIERldmVsb3BtZW50CgojIyMgU2V0dXAKCkluc3RhbGwgT3BlbkpESyAxMS4KCiMjIyBCdWlsZCBwcm9qZWN0CgpgYGAKZ3JhZGxldyBjbGVhbiBidWlsZApgYGAKCiMjIyBSdW4gcHJvamVjdAoKYGBgCmphdmEgLWphciBidWlsZC9saWJzL29iamVjdC1jb252ZXJ0ZXItMC4wLjItU05BUFNIT1QuamFyIDxKYXZhLWNsYXNzLWNhbm9uaWNhbC1uYW1lPgpgYGAKCl9fRXhhbXBsZTpfXwoKYGBgCmphdmEgLWphciBidWlsZC9saWJzL29iamVjdC1jb252ZXJ0ZXItMC4wLjItU05BUFNIT1QuamFyIGRlLm1jZWxsYS5vcGVuYXBpLnYzLm9iamVjdGNvbnZlcnRlci5FeGFtcGxlCmBgYAoKVGhlIGdlbmVyYXRlZCBPcGVuQVBJIDMgZG9jdW1lbnQgaXMgdGhlIGZpbGUgImJ1aWxkL2xpYnMvb3BlbmFwaS55YW1sIi4KCl9fRXhhbXBsZSB3aXRoIGV4dGVybmFsIGNsYXNzOl9fCgpgYGAKamF2YWMgZXhhbXBsZS1leHRlcm5hbFxzcmNcKiAtZCBleGFtcGxlLWV4dGVybmFsXGNsYXNzZXMKYGBgCgpXaW5kb3dzOgoKYGBgCmphdmEgLWNwIGJ1aWxkXGxpYnNcKjtleGFtcGxlLWV4dGVybmFsXGNsYXNzZXMgZGUubWNlbGxhLm9wZW5hcGkudjMub2JqZWN0Y29udmVydGVyLk9iamVjdENvbnZlcnRlck1haW4gIGRlLm1jZWxsYS5vcGVuYXBpLnYzLm9iamVjdGNvbnZlcnRlci5leGFtcGxlLkV4YW1wbGVGcm9tRXh0ZXJuYWwKYGBgCkxpbnV4IGFuZCBNYWNPUzoKCmBgYApqYXZhIC1jcCAiYnVpbGQvbGlicy8qOmV4YW1wbGUtZXh0ZXJuYWwvY2xhc3NlczouIiBkZS5tY2VsbGEub3BlbmFwaS52My5vYmplY3Rjb252ZXJ0ZXIuT2JqZWN0Q29udmVydGVyTWFpbiBkZS5tY2VsbGEub3BlbmFwaS52My5vYmplY3Rjb252ZXJ0ZXIuZXhhbXBsZS5FeGFtcGxlRnJvbUV4dGVybmFsCmBgYAoKVGhlIGdlbmVyYXRlZCBPcGVuQVBJIDMgZG9jdW1lbnQgaXMgYSBmaWxlIG5hbWVkICJidWlsZC9saWJzL29wZW5hcGkueWFtbCIuCgojIyMgT3BlbkFQSSAzIGRvY3VtZW50IHZhbGlkYXRpb24KClJ1biB0aGUgZm9sbG93aW5nIGdyYWRsZSBjb21tYW5kIHRvIHZhbGlkYXRlIHRoZSBnZW5lcmF0ZWQgT3BlbkFQSSAzIGRvY3VtZW50OgoKYGBgCmdyYWRsZXcgb3BlbkFwaVZhbGlkYXRlCmBgYAoKIyMjIFNvdXJjZSBjb2RlIHN0eWxlCgpUaGlzIHByb2plY3QgZm9sbG93cyB0aGUgR29vZ2xlIEphdmEgU3R5bGUsIGFuZCB1c2VzIHRoZSBbZ29vZ2xlLWphdmEtZm9ybWF0XShodHRwczovL2dpdGh1Yi5jb20vZ29vZ2xlL2dvb2dsZS1qYXZhLWZvcm1hdCkgZm9ybWF0dGVyLgoKIyMgRG9jdW1lbnRhdGlvbgoKVGhpcyBhcHBsaWNhdGlvbiBpcyB1c2VmdWwgdG8gY29udmVydCBhIGNvbXBsZXggSmF2YSBjbGFzcyB0byBhbiBPcGVuQVBJIDMgU2NoZW1hIGZyZWUtZm9ybSBvYmplY3QgZGVmaW5pdGlvbi4KClRoZSBPcGVuQVBJIDMgWUFNTCBTY2hlbWEgT2JqZWN0IGlzIGluc2VydGVkIGludG8gYW4gT3BlbkFQSSAzIGRvY3VtZW50LgpUaGUgZ2VuZXJhdGVkIE9wZW5BUEkgMyBkb2N1bWVudCBpcyBhIHNpbmdsZSBhbmQgY29tcGxldGUgT3BlbkFQSSAzIGRvY3VtZW50IHdpdGggYSBzaW1wbGUgdGVtcGxhdGUuClRoZSBjb252ZXJ0ZWQgSmF2YSBjbGFzcyBpcyBpbnNlcnRlZCBhcyBhIFNjaGVtYSBPYmplY3QgaW50byBhIHNhbXBsZSBQT1NUIHJlcXVlc3Qgd2l0aCBib2R5IGNvbnRlbnQgImFwcGxpY2F0aW9uL2pzb24iLgoKQXNzdW1wdGlvbnM6Ci0gVGhlIFNjaGVtYSBPYmplY3QgZG9lcyBub3QgY29udGFpbiBSZWZlcmVuY2UgT2JqZWN0cwotIFRoZSBTY2hlbWEgT2JqZWN0IGNvbnRhaW5zIGEgc3Vic2V0IG9mIHRoZSBhdmFpbGFibGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIHByb3BlcnRpZXMKLSBUaGUgYXBwbGljYXRpb24gZG9lcyBub3Qgc3VwcG9ydCBjb21wb3NpdGlvbiBhbmQgaW5oZXJpdGFuY2UgdXNpbmcgdGhlICJhbGxPZiIgYW5kICJkaXNjcmltaW5hdG9yIiBwcm9wZXJ0aWVzCgojIyMgRGF0YSBUeXBlcwoKVGhlIHN1cHBvcnRlZCBPcGVuQVBJIDMgZGF0YSB0eXBlcyBhcmUgdGhlIGZvbGxvd2luZzoKCnwgQ29tbW9uIG5hbWUgfCB0eXBlIHwgZm9ybWF0IHwgQ29tbWVudHMgfAp8IDotLS0tLS06IHwgOi0tLS0tLTogfCA6LS0tLS0tOiB8IDotLS0tLS06IHwKfCBpbnRlZ2VyIHwgaW50ZWdlciB8IGludDMyIHwgc2lnbmVkIDMyIGJpdHMgfAp8IGxvbmcgfCBpbnRlZ2VyIHwgaW50NjQgfCBzaWduZWQgNjQgYml0cyB8CnwgZmxvYXQgfCBudW1iZXIgfCBmbG9hdCB8IHwKfCBkb3VibGUgfCBudW1iZXIgfCBkb3VibGUgfCB8Cnwgc3RyaW5nIHwgc3RyaW5nIHwgfCB8CnwgYnl0ZSB8IHN0cmluZyB8IGJ5dGUgfCBiYXNlNjQgZW5jb2RlZCBjaGFyYWN0ZXJzIHwKfCBib29sZWFuIHwgYm9vbGVhbiB8IHwgfAp8IGRhdGUgfCBzdHJpbmcgfCBkYXRlIHwgQXMgZGVmaW5lZCBieSBmdWxsLWRhdGUgLSBbUkZDMzMzOV0oaHR0cHM6Ly94bWwycmZjLnRvb2xzLmlldGYub3JnL3B1YmxpYy9yZmMvaHRtbC9yZmMzMzM5Lmh0bWwjYW5jaG9yMTQpIHwKfCBkYXRlVGltZSB8IHN0cmluZyB8IGRhdGUtdGltZSB8IEFzIGRlZmluZWQgYnkgZGF0ZS10aW1lIC0gW1JGQzMzMzldKGh0dHBzOi8veG1sMnJmYy50b29scy5pZXRmLm9yZy9wdWJsaWMvcmZjL2h0bWwvcmZjMzMzOS5odG1sI2FuY2hvcjE0KSB8CgpUaGUgYXBwbGljYXRpb24gc3VwcG9ydHMgYm90aCBKYXZhIHByaW1pdGl2ZSB0eXBlcyBhbmQgSmF2YSB3cmFwcGVyIGNsYXNzZXMuCgojIyMgRmVhdHVyZXMKClRoZSBhcHBsaWNhdGlvbiBzdXBwb3J0cyB0aGUgY29udmVyc2lvbiBvZjoKLSBKYXZhIHByaW1pdGl2ZSBhbmQgSmF2YSB3cmFwcGVyIGNsYXNzZXMKLSBDdXN0b20gY2xhc3NlcwotIExpc3RzICh3aXRoIGEgZGVmaW5lZCBHZW5lcmljIENsYXNzKQotIERpY3Rpb25hcmllcwo= readmeEtag: '"5ce2a446dee1a422769e431a8c9adba0b417f976"' readmeLastModified: Sat, 05 Dec 2020 09:48:43 GMT repositoryId: 193395986 description: Java class conversion to OpenAPI 3 YAML Schema Object created: '2019-06-23T21:12:21Z' updated: '2025-02-01T17:53:12Z' language: Java archived: false stars: 2 watchers: 0 forks: 0 owner: sitMCella logo: https://avatars.githubusercontent.com/u/2535184?v=4 license: MIT repoEtag: '"54eb4482d28d0c082f6919a049a3270ee2706cc5a584bbae59ab39c2cdf529a1"' repoLastModified: Sat, 01 Feb 2025 17:53:12 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/python-lapidary/lapidary-render v3: true id: 1557ede36cd84f0cf0f651b2bb4e4d62 repositoryMetadata: base64Readme: >- IyBMYXBpZGFyeSByZW5kZXIKClshWy5naXRodWIvd29ya2Zsb3dzL3Rlc3QueW1sXShodHRwczovL2dpdGh1Yi5jb20vcHl0aG9uLWxhcGlkYXJ5L2xhcGlkYXJ5LXJlbmRlci9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vcHl0aG9uLWxhcGlkYXJ5L2xhcGlkYXJ5LXJlbmRlci9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LnltbCkKCkxhcGlkYXJ5LXJlbmRlciBpcyBhIHByb2dyYW0gdGhhdCBnZW5lcmF0ZXMgUHl0aG9uIFdlYiBBUEkgY2xpZW50cyBmcm9tIE9wZW5BUEkgZG9jdW1lbnRzLgoKIyMgV2h5CgpJdCdzIGEgZ29vZCBwcmFjdGljZSB0byBlbmNhcHN1bGF0ZSBXZWIgQVBJIGNsaWVudCBjb2RlIGluIGZ1bmN0aW9ucyBvciBjbGFzc2VzIGFuZCBtZXRob2RzLAoKSWYgdGhlIFdlYiBBUEkgZXhwb3NlcyBhbiBPcGVuQVBJIGRvY3VtZW50LCB5b3UgY2FuIHJlZHVjZSB0aGUgbWFudWFsIGVmZm9ydCBieSBnZW5lcmF0aW5nIHRoZSBjbGllbnQgY29kZS4KCiMjIEhvdwoKSW5zdGFsbCBMYXBpYXJ5LXJlbmRlciwgZm9yIGV4YW1wbGUgd2l0aCBwaXB4CgpgYGBzaGVsbApwaXB4IGluc3RhbGwgbGFwaWRhcnktcmVuZGVyCmBgYAoKU3RhcnQgeW91ciBwcm9qZWN0CgpgYGBzaGVsbApsYXBpZGFyeSBpbml0IC0tc2F2ZSBodHRwczovL2V4YW1wbGUuY29tL29wZW5hcGkuanNvbiBwcm9qZWN0X2RpciBteV9hcGlfY2xpZW50CmBgYAoKR2VuZXJhdGUgY29kZToKYGBgc2hlbGwKY2QgcHJvamVjdF9kaXIKbGFwaWRhcnkgcmVuZGVyCmBgYAoKQ2hlY2sgdGhlIFtkb2N1bWVudGF0aW9uXShodHRwczovL2xhcGlkYXJ5LmRldi9sYXBpZGFyeS1yZW5kZXIvKSBmb3IgbW9yZSBkZXRhaWxzLgo= readmeEtag: '"c19d4a4bd3d4130092a904fa8085c370ed2d94b0"' readmeLastModified: Fri, 05 Dec 2025 23:05:55 GMT repositoryId: 584507821 description: Web API client generator created: '2023-01-02T19:09:02Z' updated: '2026-01-16T00:05:31Z' language: Python archived: false stars: 4 watchers: 1 forks: 1 owner: python-lapidary logo: https://avatars.githubusercontent.com/u/111730600?v=4 license: AGPL-3.0 repoEtag: '"6d1b13077149834ab425dbb8a3035883fddc727fc89a8ac9167fd90ea1a19f6a"' repoLastModified: Fri, 16 Jan 2026 00:05:31 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/keecon/restdocs-openapi3 v3: true repositoryMetadata: base64Readme: >- # Spring REST Docs OpenAPI 3 Specification

[![jitpack-badge]](https://jitpack.io/#keecon/restdocs-openapi3)
[![build-badge]](https://github.com/keecon/restdocs-openapi3/actions/workflows/build.yml)
[![codecov-badge]](https://codecov.io/gh/keecon/restdocs-openapi3)
[![sonarcloud-badge]](https://sonarcloud.io/summary/new_code?id=keecon_restdocs-openapi3)
[![license-badge]](https://github.com/keecon/restdocs-openapi3/blob/main/LICENSE)

A modified version of the [ePages-de/restdocs-api-spec] with class field type and constraint inference.
And only support [OpenAPI 3.0.1] specs.

## Build configuration

### Versions

- The *1.x.x* version is compatible with Spring Boot *3.0.x* and Spring REST Docs *3.0.x*.
- The *0.x.x* version is compatible with Spring Boot *2.7.x* and Spring REST Docs *2.0.x*.

### Gradle

1. Add the plugin

    ```groovy
    buildscript() {
      repositories {
        // ...
        maven { url "https://jitpack.io" }
      }
      dependencies {
        // ...
        classpath 'com.github.keecon:restdocs-openapi3:x.x.x'
      }
    }

    plugins {
      // ...
      id 'com.github.keecon:restdocs-openapi3'
    }
    ```

2. Add required dependencies to your tests

    ```groovy
    repositories {
      // ...
      maven { url 'https://jitpack.io' }
    }

    dependencies {
      //..
      testImplementation 'com.github.keecon:restdocs-openapi3:x.x.x'
    }

    openapi3 {
      server = 'http://localhost:8080'
      title = 'My API'
      description = 'My API description'
      tagDescriptionsPropertiesFile = 'src/test/resources/openapi-tags.yml'
      version = '0.1.0'
      format = 'yaml'
    }
    ```

## Usage with Spring REST Docs

```groovy
when:
def resultActions = mockMvc.perform(
  post('/v1/products/{productId}/result', 1)
    .contentType(MediaType.APPLICATION_JSON)
    .content(objectMapper.writeValueAsBytes(new ProductResultCreateRequestBody(result)))
    .accept(MediaType.APPLICATION_JSON)
)

then:
def reqModel = Constraints.model(ProductResultCreateRequest.class)
def reqBodyModel = Constraints.model(ProductResultCreateRequestBody.class)
def respModel = Constraints.model(ProductResultCreateResponse.class)
resultActions
  .andExpect(status().isOk())
  .andDo(document('products-id-result-post',
    resource(ResourceSnippetParameters.builder()
      .tag('product')
      .summary('Create a product result')
      .description('''
        |Create a product result
        |
        |### Error details
        |
        |`400` BAD_REQUEST
        |- bad request description
        |
        |`401` UNAUTHORIZED
        |- unauthorized description
        |
        |'''.stripMargin())
      .requestSchema(schema('ProductResultCreateRequest'))
      .pathParameters(
        reqModel.withName('productId').description('product id'),
      )
      .requestFields(
        reqBodyModel.withPath('result').description('product result'),
        reqBodyModel.withPath('result.code').description('product result code'),
        reqBodyModel.withPath('result.seq').description('product result seq'),
        reqBodyModel.withPath('result.score').description('product result score'),
        reqBodyModel.withPath('result.assigns[]').description('result assign object list'),
        reqBodyModel.withPath('result.assigns[].code').description('result assign code'),
        reqBodyModel.withPath('result.assigns[].seq').description('result assign seq'),
        reqBodyModel.withPath('result.assigns[].objectId').description('result assign object id'),
        reqBodyModel.withPath('result.assigns[].fileType').description('result assign file type')
          .optional(),
        reqBodyModel.withPath('result.assigns[].fileUrl').description('result assign file url')
          .optional(),
        reqBodyModel.withPath('result.assigns[].comments[]').description('result assign comment list')
          .type(DataType.ARRAY)
          .attributes(Attributes.items(DataType.STRING, null, null))
          .optional(),
      )
      .responseSchema(schema('ProductResultCreateResponse'))
      .responseFields(
        respModel.withPath('status').description('operation status'),
        respModel.withPath('code').description('product result code')
          .optional(),
      )
      .build())))
```

```groovy
when:
def resultActions = mockMvc.perform(
  get('/v1/products/{productId}/result?code={code}', 1, 1)
    .accept(MediaType.APPLICATION_JSON)
)

then:
def reqModel = Constraints.model(ProductResultRequest.class)
def respModel = Constraints.model(ProductResultResponse.class)
resultActions
  .andExpect(status().isOk())
  .andDo(document('products-id-result-code-get',
    resource(ResourceSnippetParameters.builder()
      .tag('product')
      .summary('Get a product result info')
      .description('''
        |Get a product result info
        |
        |### Error details
        |
        |`400` BAD_REQUEST
        |- bad request description
        |
        |`401` UNAUTHORIZED
        |- unauthorized description
        |
        |`404` NOT_FOUND
        |- not found description
        |
        |'''.stripMargin())
      .requestSchema(schema('ProductResultRequest'))
      .pathParameters(
        reqModel.withName('productId').description('product id'),
      )
      .queryParameters(
        reqModel.withName('code').description('product result code'),
        reqModel.withName('seq').description('product result seq')
          .defaultValue(ProductResultRequest.DEFAULT_RESULT_SEQ)
          .optional(),
      )
      .responseSchema(schema('ProductResultResponse'))
      .responseFields(
        respModel.withPath('result').description('product result'),
        respModel.withPath('result.code').description('product result code'),
        respModel.withPath('result.seq').description('product result seq'),
        respModel.withPath('result.score').description('product result score'),
        respModel.withPath('result.assigns[]').description('result assign object list'),
        respModel.withPath('result.assigns[].code').description('result assign code'),
        respModel.withPath('result.assigns[].seq').description('result assign seq'),
        respModel.withPath('result.assigns[].objectId').description('result assign object id'),
        respModel.withPath('result.assigns[].fileType').description('result assign file type')
          .optional(),
        respModel.withPath('result.assigns[].fileUrl').description('result assign file url')
          .optional(),
        respModel.withPath('result.assigns[].comments[]').description('result assign comment list')
          .type(DataType.ARRAY)
          .attributes(Attributes.items(DataType.STRING, null, null))
          .optional(),
      )
      .build())))
```

[jitpack-badge]: https://jitpack.io/v/keecon/restdocs-openapi3.svg

[build-badge]: https://github.com/keecon/restdocs-openapi3/actions/workflows/build.yml/badge.svg

[codecov-badge]: https://codecov.io/gh/keecon/restdocs-openapi3/branch/main/graph/badge.svg?token=TRQZ6GOVK4

[sonarcloud-badge]: https://sonarcloud.io/api/project_badges/measure?project=keecon_restdocs-openapi3&metric=alert_status

[license-badge]: https://img.shields.io/github/license/keecon/restdocs-openapi3.svg

[ePages-de/restdocs-api-spec]: https://github.com/ePages-de/restdocs-api-spec

[OpenAPI 3.0.1]: https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.1.md
 readmeEtag: '"e6aa617fd3dd190f7e1106ba1e281a19d1ce6ed8"' readmeLastModified: Tue, 13 Aug 2024 01:36:57 GMT repositoryId: 462124358 description: >- Clone of ePages-de/restdocs-api-spec with class field type and constraint inference. created: '2022-02-22T03:33:26Z' updated: '2025-04-29T02:01:27Z' language: Kotlin archived: false stars: 2 watchers: 4 forks: 0 owner: keecon logo: https://avatars.githubusercontent.com/u/19371343?v=4 license: MIT repoEtag: '"cc24e6eefe74c77fbdebaa2faf248c0dbc307b048da2f8d7b89ce485c9ff8ea1"' repoLastModified: Tue, 29 Apr 2025 02:01:27 GMT foundInMaster: true category: Server id: 07a8059ee6961dc1e46442f5582d1501 - source: openapi3 tags repository: https://github.com/quantumsheep/swagger-schema-extractor v3: true repositoryMetadata: base64Readme: >- WyFbbnBtXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L3N3YWdnZXItc2NoZW1hLWV4dHJhY3RvcildKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3N3YWdnZXItc2NoZW1hLWV4dHJhY3RvcikKCiMgU3dhZ2dlciBTY2hlbWEgRXh0cmFjdG9yCkV4dHJhY3QgT3BlbkFQSSBzY2hlbWFzIGludG8gVHlwZVNjcmlwdCB0eXBlcy4KCiMjIFVzYWdlCiMjIyBPcGVuQVBJIHYzCmBgYGJhc2gKbnB4IHN3YWdnZXItc2NoZW1hLWV4dHJhY3RvciBodHRwOi8vbG9jYWxob3N0OjMwMDAvb3BlbmFwaS5qc29uID4gdHlwZXMudHMKYGBgCgojIyMgT3BlbkFQSSB2MgpgYGBiYXNoCm5weCBzd2FnZ2VyLXNjaGVtYS1leHRyYWN0b3IgaHR0cDovL2xvY2FsaG9zdDozMDAwL3YyL3N3YWdnZXIuanNvbiA+IHR5cGVzLnRzCmBgYAo= readmeEtag: '"1f8cf473ab0422000c39c5f100696f6b25cb5ae9"' readmeLastModified: Mon, 07 Dec 2020 09:58:02 GMT repositoryId: 318500212 description: Extract Swagger (OpenAPI) schemas into TypeScript types created: '2020-12-04T11:48:15Z' updated: '2020-12-14T23:59:30Z' language: JavaScript archived: false stars: 2 watchers: 1 forks: 0 owner: quantumsheep logo: https://avatars.githubusercontent.com/u/7271496?v=4 license: MIT repoEtag: '"055d4558c53f9f74442ef4a47633ed13ae1c24f3009d3c0a26eb9740af067b73"' repoLastModified: Mon, 14 Dec 2020 23:59:30 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: ed2585a9c3054899a65e93cc6fab281c - source: openapi3 tags repository: https://github.com/jhansenbarreto/pessoa-endereco v3: true id: 67bcb878d92aad2b34e4ef36f01e30c7 repositoryMetadata: base64Readme: >- IyA6cGVuY2lsOiBEb2N1bWVudGHDp8OjbwoKPHA+CiAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvU3ByaW5nJTIwQm9vdC12My4wLjEtYnJpZ2h0Z3JlZW4iLz4KICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9qYXZhLS1qZGstdjE3LjAuNC4xLW9yYW5nZSIvPgogIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL21hdmVuLS1jZW50cmFsLXY0LjAuMC1ibHVlIi8+CiAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvSlVuaXQtdjUuOS4xLXJlZCIvPgo8L3A+CiAgCj4qRXN0ZSBwcm9qZXRvIGZvaSBpbXBsZW1lbnRhZG8gbmEgZXRhcGEgZG8gdGVzdGUgdMOpY25pY28gbm8gcHJvY2Vzc28gc2VsZXRpdm8gZGEgYGBBdHRvcm5hdHVzIFByb2N1cmFkb3JpYSBEaWdpdGFsYGAgcGFyYSBhIHZhZ2EgZGUgYGBEZXNlbnZvbHZlZG9yIEJhY2sgRW5kIC0gSmF2YWBgLiBTZWd1aW5kbyBhcyBlc3BlY2lmaWNhw6fDtWVzIGRhZGFzIG5vIHRlc3RlLCBvIHByb2pldG8gw6kgdW1hIHNpbXBsZXMgQVBJIGRlc3RpbmFkYSBhbyBjYWRhc3RybyBkZSBQZXNzb2FzIGUgRW5kZXJlw6dvcywgc2VydmluZG8gYXMgZnVuY2lvbmFsaWRhZGVzIG9icmlnYXTDs3JpYXMgZXhpZ2lkYXMgbm8gdGVzdGUgZSBvdXRyYXMgaW1wbGVtZW50YWRhcyBwZWxvIGRlc2Vudm9sdmVkb3IgZGEgQVBJIHBvciBkZWNpc8OjbyBkZSBwcm9qZXRvLioKICAKTyBwcm9qZXRvIGNvbnRhIGNvbSBvIENSVUQgY29tcGxldG8gZGFzIHNlZ3VpbnRlcyBlbnRpZGFkZXM6IAoKLSBgYEVzdGFkb3NgYAotIGBgQ2lkYWRlc2BgCi0gYGBQZXNzb2FzYGAKLSBgYEVuZGVyZcOnb3NgYAoKQSBkb2N1bWVudGHDp8OjbyBjb21wbGV0YSBkbyBwcm9qZXRvIGZvaSBmZWl0YSB1dGlsaXphbmRvIGFzIGZlcnJhbWVudGFzIGRhIGJpYmxpb3RlY2EgYGBTcHJpbmcgRG9jYGAsIGRlcGVuZMOqbmNpYSBkbyBwcm9qZXRvLCBhdHJhdsOpcyBkbyBgYFN3YWdnZXIgVUlgYCBjb20gYXMgZXNwZWNpZmljYcOnw7VlcyBkbyBgYE9wZW4gQVBJIDMuMGBgLCBhZmltIGRlIHByb3ZlciBtYWlvciBlbnRlbmRpbWVudG8gcGFyYSBvcyBjb25zdW1pZG9yZXMgZGEgQVBJLiBQYXJhIGFjZXNzYXIgYSBkb2N1bWVudGHDp8OjbywgZmHDp2EgbyBjbG9uZSBkbyBwcm9qZXRvIHBhcmEgYSBzdWEgbcOhcXVpbmEsIGluaWNpZSBhIGFwbGljYcOnw6NvICoocG9kZSBkZW1vcmFyIHVtIHBvdWNvLCBwb2lzIHNlcsOhIGZlaXRvIG8gZG93bmxvYWQgZGFzIGRlcGVuZMOqbmNpYXMgbmEgcHJpbWVpcmEgdmV6KSogZSBhcMOzcyBzdWJpciBvIHNlcnZpw6dvIGFicmEgc2V1IG5hdmVnYWRvciBlIGRpZ2l0ZTogYGBodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS9pbmRleC5odG1sYGAuIFNlIHByZWZlcmlyLCBvcyB0ZXN0ZXMgcG9kZW0gc2VyIGZlaXRvcyBuYSBwcsOzcHJpYSBww6FnaW5hIGRhIGRvY3VtZW50YcOnw6NvLCBvbmRlIGNvbnTDqW0gYXMgZXhwbGljYcOnw7VlcyBkZXZpZGFzIHBhcmEgY29uc3VtaXIgY2FkYSAqZW5kcG9pbnQqLgoKIyMgOmhhbW1lcjogRnVuY2lvbmFsaWRhZGVzCgo6cGFja2FnZTogYGBFc3RhZG9zOmBgCi0gQ2FkYXN0cmEgZXN0YWRvcwotIENvbnN1bHRhIGVzdGFkb3MKLSBMaXN0YSB0b2RvcyBvcyBlc3RhZG9zIGNhZGFzdHJhZG9zCi0gTGlzdGEgdG9kYXMgYXMgY2lkYWRlcyBkZSB1bSBlc3RhZG8KLSBFZGl0YSBlc3RhZG9zCi0gRXhjbHVpIGVzdGFkb3MKLSA+IDpidWxiOiAqKkRldGFsaGVzKio6ICpOw6NvIMOpIHBlcm1pdGlkbyBlc3RhZG9zIGNvbSBkYWRvcyBkdXBsaWNhZG9zLCBub21lcyBzw7MgYWNlaXRhbSBsZXRyYXMgbWFpw7pzY3VsYXMsIG1pbsO6c2N1bGFzLCBhY2VudHVhZGFzIGUgZXNwYcOnb3MsIFVGIHPDsyBhY2VpdGEgMiBsZXRyYXMuKgoKOnBhY2thZ2U6IGBgQ2lkYWRlczpgYAotIENhZGFzdHJhIGNpZGFkZXMKLSBDb25zdWx0YSBjaWRhZGVzCi0gTGlzdGEgdG9kYXMgYXMgY2lkYWRlcyBjYWRhc3RyYWRhcwotIEVkaXRhIGNpZGFkZXMKLSBFeGNsdWkgY2lkYWRlcwotID4gOmJ1bGI6ICoqRGV0YWxoZXMqKjogKk5vbWVzIHPDsyBhY2VpdGFtIGxldHJhcyBtYWnDunNjdWxhcywgbWluw7pzY3VsYXMsIGFjZW50dWFkYXMgZSBlc3Bhw6dvcy4qCgo6cGFja2FnZTogYGBQZXNzb2FzOmBgCi0gQ2FkYXN0cmEgcGVzc29hcwotIENvbnN1bHRhIHBlc3NvYXMKLSBMaXN0YSB0b2RhcyBhcyBwZXNzb2FzIGNhZGFzdHJhZGFzCi0gRWRpdGEgcGVzc29hcwotIEV4Y2x1aSBwZXNzb2FzCi0gPiA6YnVsYjogKipEZXRhbGhlcyoqOiAqTm9tZXMgc8OzIGFjZWl0YW0gbGV0cmFzIG1hacO6c2N1bGFzLCBtaW7DunNjdWxhcywgYWNlbnR1YWRhcyBlIGVzcGHDp29zLiBEYXRhIGRlIG5hc2NpbWVudG8gbm8gcGFkcsOjbyB5eXl5LW1tLWRkKgoKOnBhY2thZ2U6IGBgRW5kZXJlw6dvczpgYAotIENhZGFzdHJhIGVuZGVyZcOnb3MgcGFyYSBQZXNzb2FzCi0gTGlzdGEgdG9kb3Mgb3MgRW5kZXJlw6dvcyBkZSB1bWEgUGVzc29hCi0gRWRpdGEgZW5kZXJlw6dvcwotIEV4Y2x1aSBlbmRlcmXDp29zCi0gUGVybWl0ZSBtYXJjYXIgdW0gZW5kZXJlw6dvIGNvbW8gcHJpbmNpcGFsCi0gPiA6YnVsYjogKipEZXRhbGhlcyoqOiAqQ0VQIGFjZWl0YSBhcGVuYXMgOCBjYXJhY3RlcmVzIG51bcOpcmljb3MuIFRvZG8gcHJpbWVpcm8gZW5kZXJlw6dvIGNhZGFzdHJhZG8gw6kgbWFyY2FkbyBjb21vIHByaW5jaXBhbCBhdXRvbWF0aWNhbWVudGUuIFNlbXByZSBxdWUgdW0gZW5kZXJlw6dvIHByaW5jaXBhbCBmb3IgZXhjbHXDrWRvLCBjYXNvIGFpbmRhIGV4aXN0YW0gb3V0cm9zLCBvIHByaW1laXJvIGRhIGxpc3RhIHNlIHRvcm5hIG8gbm92byBwcmluY2lwYWwuKgoKIyMgOmhlYXZ5X2NoZWNrX21hcms6IFTDqWNuaWNhcyBlIFRlY25vbG9naWFzIFV0aWxpemFkYXMKCi0gYGBKYXZhIDE3YGAKLSBgYFNwcmluZyBCb290IDNgYAotIGBgQXBhY2hlIE5ldEJlYW5zIElERSAxNWBgCi0gYGBCYW5jbyBkZSBEYWRvcyBlbSBNZW3Ds3JpYSAoSDIpYGAKLSBgYFBvc3RtYW4gMTAgKHBhcmEgdGVzdGVzIHByw6F0aWNvcylgYAotIGBgSlVuaXQgNSAocGFyYSB0ZXN0ZXMgcHJvZ3JhbcOhdGljb3MpYGAKLSBgYFN3YWdnZXIgVUkgLyBPcGVuIEFQSSAzIChwYXJhIGRvY3VtZW50YcOnw6NvKWBgCi0gYGBQYWRyw6NvIGRlIFByb2pldG8gRFRPIChEYXRhIFRyYW5zZmVyIE9iamVjdClgYAotIGBgTW9kZWxhZ2VtIGNvbSBEREQgKERvbWFpbi1Ecml2ZW4gRGVzaWduKWBgCgojIyA6cm90YXRpbmdfbGlnaHQ6IE9ic2VydmHDp8O1ZXMKCi0gYGBDb21lbnTDoXJpb3MgZSBKYXZhZG9jOmBgIFNlIGZvciBkZSBzZXUgaW50ZXJlc3NlIGdlcmFyIG8gSmF2YWRvYyBkbyBwcm9qZXRvLCBvcyBjb21lbnTDoXJpb3MgasOhIGZvcmFtIGVzY3JpdG9zIHBhcmEgZmFjaWxpdGFyIG8gZW50ZW5kaW1lbnRvIGRlIGFsZ3VtYXMgZGVjaXPDtWVzIHRvbWFkYXMgZHVyYW50ZSBhIGltcGxlbWVudGHDp8Ojby4KLSBgYERhZG9zIGRlIFRlc3RlOmBgIFBhcmEgZmFjaWxpdGFyIG9zIHRlc3RlcywgZm9pIGRlaXhhZG8gbm8gZGlyZXTDs3JpbyBgYHNyYy9tYWluL3Jlc291cmNlc2BgIHVtIGFycXVpdm8gYGBkYXRhLnNxbGBgLCB1dGlsaXphZG8gcGFyYSBwb3B1bGFyIG8gYmFuY28gZGUgZGFkb3Mgc2VtcHJlIHF1ZSBhIGFwbGljYcOnw6NvIGZvciBpbmljaWFkYS4gQ29uZmlyYSBvIGFycXVpdm8gPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2poYW5zZW5iYXJyZXRvL3Blc3NvYS1lbmRlcmVjby9ibG9iL21hc3Rlci9zcmMvbWFpbi9yZXNvdXJjZXMvZGF0YS5zcWwiPmNsaWNhbmRvIGFxdWk8L2E+LiA6YnVsYjogKipBVEVOw4fDg086KiogKk9zIHRlc3RlcyBkZSBpbnRlZ3Jhw6fDo28gZGVzZW52b2x2aWRvcyBlbSBKVW5pdCA1IHV0aWxpemFtIGVzdGVzIGRhZG9zLioKLSBgYFRyYXRhbWVudG8gZGUgRXJyb3M6YGAgQSBBUEkgY29udGEgY29tIHRyYXRhbWVudG8gZGUgZXJyb3MgZSBmb3JuZWNlIGFvIGNvbnN1bWlkb3IgdW0gbW9kZWxvIGRlIHJlcHJlc2VudGHDp8OjbyBkZSBlcnJvcyBiYXNlYWRvIG5hIGVzcGVjaWZpY2HDp8OjbyA8YSBocmVmPSJodHRwczovL3d3dy5yZmMtZWRpdG9yLm9yZy9yZmMvcmZjNzgwNyI+UkZDIDc4MDcgKFByb2JsZW0gRGV0YWlscyBmb3IgSFRUUCBBUElzKTwvYT4sIHV0aWxpemFuZG8tYSBhcGVuYXMgY29tbyByZWZlcsOqbmNpYSwgbsOjbyBpbXBsZW1lbnRhbmRvIHRvdGFsbWVudGUgw6AgcmlzY2EuCgojIyA6Y29uc3RydWN0aW9uX3dvcmtlcjogQXV0b3IKCnwgOnRlY2hub2xvZ2lzdDogRGVzZW52b2x2ZWRvciB8Omdsb2JlX3dpdGhfbWVyaWRpYW5zOiBMaW5rcyDDmnRlaXN8CnwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLTp8LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwKfDxwIGFsaWduPSJjZW50ZXIiPjxpbWcgc3JjPSJodHRwczovL2F2YXRhcnMuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3UvMTM3OTA2MDg/dj00IiB3aWR0aD0xMTU+PC9icj48c3ViPkpoYW5zZW4gQmFycmV0bzwvc3ViPjwvcD58PHVsPjxsaT48YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vamhhbnNlbmJhcnJldG8/dGFiPXJlcG9zaXRvcmllcyI+R2l0SHViPC9hPjwvbGk+PGxpPjxhIGhyZWY9Imh0dHBzOi8vYnIubGlua2VkaW4uY29tL2luL2poYW5zZW4tYy1iYXJyZXRvIj5MaW5rZWRJbjwvYT48L2xpPjxsaT48YSBocmVmPSJodHRwczovL3d3dy5pbnN0YWdyYW0uY29tL2poYW5zZW5iYXJyZXRvLyI+SW5zdGFncmFtPC9hPjwvbGk+PC91bD58Cg== readmeEtag: '"0e8bc30e12e68bc7188799a4f4361705c0e13c73"' readmeLastModified: Fri, 20 Oct 2023 00:37:13 GMT repositoryId: 590518845 description: >- API implementada para teste técnico no processo seletivo da Attornatus Procuradoria Digital para a vaga de Desenvolvedor Back End - Java. created: '2023-01-18T15:49:30Z' updated: '2025-09-17T03:41:19Z' language: Java archived: false stars: 3 watchers: 1 forks: 2 owner: jhansenbarreto logo: https://avatars.githubusercontent.com/u/13790608?v=4 repoEtag: '"7c10179d376b47481527a5b84274bcd96b40e0e509d55d51c7634a050aafde56"' repoLastModified: Wed, 17 Sep 2025 03:41:19 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/walkersumida/go-api-server v3: true id: 2518a5e96dc9498c8e3e8fd17d5d20f3 repositoryMetadata: repositoryId: 657910869 description: API Server (ogen + ent + golang-migrate + postgres) created: '2023-06-24T07:25:04Z' updated: '2025-10-29T08:14:05Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: walkersumida logo: https://avatars.githubusercontent.com/u/12683375?v=4 license: Apache-2.0 repoEtag: '"1580df92e8f8bb571183a5c10d59503b9bf74faf26cdeaeeec286c94b1d793f9"' repoLastModified: Wed, 29 Oct 2025 08:14:05 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/mbhuman/ravagepdf v3: true id: 88064ff57f84edc19e52519d84c64009 repositoryMetadata: base64Readme: >- IyByYXZhZ2VwZGYKCkNoZWNrIGNvdmVyYWdlIDIKCiMjIEludHJvZHVjdGlvbgoKUmF2YWdlcGRmIGlzIGEgY2xpLCB3aXRoIHlvdSBjYW4gZWFzaWx5IGNvbnZlcnQgeW91ciBvcGVuYXBpLmpzb24gb3Igb3BlbmFwaS55bWwgc3BlY3MgZnJvbSBVUkwgb3IgZnJvbSBmaWxlIHRvIFBERiB3aXRoIHNpbXBsZSBjb21tYW5kLiBJdCB0YWtlcyBmZXcgc2Vjb25kcyB0byBnZW5lcmF0ZSBmaWxlLiBBbGwgbWV0aG9kcyBhcmUgYXN5bmMsIGFuZCBjYW4gYmUgdXNlZCBhc3luYyB3aXRoaW4gcHJvamVjdHMgZm9yIGFzeW5jIFBERiBnZW5lcmF0aW9uLgoKIyMgRmVhdHVyZXMKCiogVVJMIGFwaSBzcGVjIHN1cHBvcnQgKFlNTCBhbmQgSlNPTikKKiBPcGVuQVBJIGZpbGUgc3BlYyBzdXBwb3J0IChZTUwgYW5kIEpTT04pCiogVGFibGUgb2YgQ29udGVudHMKKiBIYW5kbGVyIGRlc2NyaXB0aW9uCiogUmVxdWVzdCBkZXNjcmlwdGlvbgoqIFJlc3BvbnNlIGRlc2NyaXB0aW9uCiogRXhhbXBsZXMgZm9yIFJlc3BvbnNlIGFuZCBSZXF1ZXN0CiogT3BlbkFQSTMuMCBzdXBwb3J0IHdpdGggQW55T2YsIE9uZU9mIGFuZCBvdGhlcnMKKiBFbnVtIGV4YW1wbGVzIHN1cHBvcnQKKiBEZXNjcmlwdGlvbiBmb3IgYWxsIHBhcmFtcyBpbiBib2R5CgojIyBFeGFtcGxlcyBvZiB1c2FnZQoKIyMjIEluc3RhbGxhdGlvbgoKYGBgY29uc29sZQpucG0gaSAtZyByYXZhZ2VwZGYKYGBgCgojIyMgRXhhbXBsZSB3aXRoIFVSTAoKYGBgY29uc29sZQpyYXZhZ2VwZGYgLXMgaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcHJveHkvcmVnaXN0cnkvRGlyZWN0X1dpbmVzL0NhcnRBUElTdWl0ZS8xLjcuMFw/cmVzb2x2ZWRcPXRydWVcJmZsYXR0ZW5cPXRydWVcJnByZXR0eVw9dHJ1ZSAtbyBwZGZEb2NzLnBkZgpgYGAKCiMjIyBFeGFtcGxlIHdpdGggZmlsZQoKIyMjIyBKU09OCgpgYGBjb25zb2xlCnJhdmFnZXBkZiAtcyBvcGVuYXBpLmpzb24gLW8gcGRmRG9jcy5wZGYKYGBgCgojIyMjIFlNTCAKCmBgYGNvbnNvbGUKCnJhdmFnZXBkZiAtcyBvcGVuYXBpLnltbCAtbyBwZGZEb2NzLnBkZgpgYGAKCiMjIFBERiBleGFtcGxlcwoKIyMjIEluZm8KCiFbXShodHRwczovL2dpdGh1Yi5jb20vTUJIdW1hbi9yYXZhZ2VwZGYvYmxvYi9tYWluL2ltYWdlcy9pbmZvLnBuZykKCiMjIyBJbmRleAoKIVtdKGh0dHBzOi8vZ2l0aHViLmNvbS9NQkh1bWFuL3JhdmFnZXBkZi9ibG9iL21haW4vaW1hZ2VzL2luZGV4LnBuZykKCiMjIyBQYXRocyBIZWFkZXIKCiFbXShodHRwczovL2dpdGh1Yi5jb20vTUJIdW1hbi9yYXZhZ2VwZGYvYmxvYi9tYWluL2ltYWdlcy9wYXRoc19oZWFkZXIucG5nKQoKIyMjIFJlc3BvbnNlIERlc2NyaXB0aW9uCgohW10oaHR0cHM6Ly9naXRodWIuY29tL01CSHVtYW4vcmF2YWdlcGRmL2Jsb2IvbWFpbi9pbWFnZXMvcmVzcG9uc2VfZGVzY3JpcHRpb24ucG5nKQoKIyMjIEV4YW1wbGUKCiFbXShodHRwczovL2dpdGh1Yi5jb20vTUJIdW1hbi9yYXZhZ2VwZGYvYmxvYi9tYWluL2ltYWdlcy9leGFtcGxlLnBuZykKCiMjIyBBUEkgTGlzdAoKIVtdKGh0dHBzOi8vZ2l0aHViLmNvbS9NQkh1bWFuL3JhdmFnZXBkZi9ibG9iL21haW4vaW1hZ2VzL2FwaV9saXN0LnBuZykKCiMjIFByb2dyYW0gRGVzaWduCgohW10oLi9pbWFnZXMvcmF2YWdlLnN2Zyk= readmeEtag: '"6b6aebc2c215f16481b330fc211b83791ef95f44"' readmeLastModified: Fri, 23 Jun 2023 11:46:34 GMT repositoryId: 628531125 description: PDF generation from OpenAPI / Swagger Spec Resources URL or JSON/YML file created: '2023-04-16T08:35:08Z' updated: '2025-01-04T12:37:43Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 1 owner: MBHuman logo: https://avatars.githubusercontent.com/u/44523394?v=4 license: Apache-2.0 repoEtag: '"abdf43e5c0bfa4d0f096b6ce6a0920fc8b3b05ac97d0e85eb4a64940adbb8871"' repoLastModified: Sat, 04 Jan 2025 12:37:43 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/kumuluz/kumuluzee-openapi v3: true repositoryMetadata: base64Readme: >- IyBLdW11bHV6RUUgT3BlbkFQSQpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdHJhdmlzL2t1bXVsdXova3VtdWx1emVlLW9wZW5hcGkvbWFzdGVyLnN2Zz9zdHlsZT1mbGF0KV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2t1bXVsdXova3VtdWx1emVlLW9wZW5hcGkpCgo+IEt1bXVsdXpFRSBPcGVuQVBJIHByb2plY3QgcHJvdmlkZXMgcG93ZXJmdWwgdG9vbHMgdG8gaW5jb3Jwb3JhdGUgYW5kIHZpc3VhbGl6ZSB0aGUgT3BlbkFQSSAzIHNwZWNpZmljYXRpb24gdG8geW91ciBtaWNyb3NlcnZpY2UuCgpLdW11bHV6RUUgT3BlbkFQSSBwcm9qZWN0IGFsbG93cyB5b3UgdG8gZG9jdW1lbnQgbWljcm9zZXJ2aWNlIEFQSXMgdXNpbmcgT3BlbkFQSSB2MyBjb21wbGlhbnQgYW5ub3RhdGlvbnMuIFByb2plY3Qgd2lsbCBhdXRvbWF0aWNhbGx5IGhvb2stdXAgc2VydmxldCB0aGF0IHdpbGwgCnNlcnZlIHlvdXIgQVBJIHNwZWNpZmljYXRpb25zIG9uIGVuZHBvaW50IGBgYC9hcGktc3BlY3MvPGpheC1ycyBhcHBsaWNhdGlvbi1iYXNlLXBhdGg+L29wZW5hcGkuW2pzb258eWFtbF1gYGAuIEZ1cnRoZXJtb3JlLCBwcm9qZWN0IGFsbG93cyB5b3UgdG8gaW50ZWdyYXRlIFN3YWdnZXItVUkgaW50byB5b3VyCm1pY3Jvc2VydmljZSB0aGF0IHdpbGwgdmlzdWFsaXplIEFQSXMgZG9jdW1lbnRhdGlvbiBhbmQgYWxsb3cgeW91IHRvIGludGVyYWN0IHdpdGggeW91ciBBUEkgcmVzb3VyY2VzLgogCk1vcmUgZGV0YWlsczogW09wZW5BUEkgdjMgU3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjAubWQpLgoKIyMgVXNhZ2UKCllvdSBjYW4gZW5hYmxlIEt1bXVsdXpFRSBPcGVuQVBJIHN1cHBvcnQgYnkgYWRkaW5nIHRoZSBmb2xsb3dpbmcgZGVwZW5kZW5jeToKYGBgeG1sCjxkZXBlbmRlbmN5PgogICAgPGdyb3VwSWQ+Y29tLmt1bXVsdXouZWUub3BlbmFwaTwvZ3JvdXBJZD4KICAgIDxhcnRpZmFjdElkPmt1bXVsdXplZS1vcGVuYXBpPC9hcnRpZmFjdElkPgogICAgPHZlcnNpb24+JHtrdW11bHV6ZWUtb3BlbmFwaS52ZXJzaW9ufTwvdmVyc2lvbj4KPC9kZXBlbmRlbmN5PgpgYGAKCiMjIE9wZW5BUEkgY29uZmlndXJhdGlvbgoKV2hlbiBrdW11bHV6ZWUtb3BlbmFwaSBkZXBlbmRlbmN5IGlzIGluY2x1ZGVkIGluIHRoZSBwcm9qZWN0LCB5b3UgY2FuIHN0YXJ0IGRvY3VtZW50aW5nIHlvdXIgUkVTVCBBUEkgdXNpbmcgW1N3YWdnZXItQ29yZSBBbm5vdGF0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29yZS93aWtpL0Fubm90YXRpb25zLTIuWCkuCgojIyMgRG9jdW1lbnRpbmcgYXBwbGljYXRpb24gY2xhc3MKYGBgamF2YQpAU2VjdXJpdHlTY2hlbWUobmFtZSA9ICJvcGVuaWQtY29ubmVjdCIsIHR5cGUgPSBTZWN1cml0eVNjaGVtZVR5cGUuT1BFTklEQ09OTkVDVCwgCiAgICAgICAgICAgICAgICBvcGVuSWRDb25uZWN0VXJsID0gImh0dHA6Ly9hdXRoLXNlcnZlci11cmwvLndlbGwta25vd24vb3BlbmlkLWNvbmZpZ3VyYXRpb24iKQpAT3BlbkFQSURlZmluaXRpb24oaW5mbyA9IEBJbmZvKHRpdGxlID0gIlJlc3QgQVBJIiwgdmVyc2lvbiA9ICJ2MSIsIGNvbnRhY3QgPSBAQ29udGFjdCgpLCBsaWNlbnNlID0gQExpY2Vuc2UoKSwgZGVzY3JpcHRpb24gPSAiSmF2YVNJIEFQSSBmb3IgbWFuYWdpbmcgY29uZmVyZW5jZS4iKSwgc2VjdXJpdHkgPSBAU2VjdXJpdHlSZXF1aXJlbWVudChuYW1lID0gIm9wZW5pZC1jb25uZWN0IiksIHNlcnZlcnMgPSBAU2VydmVyKHVybCA9Imh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC92MSIpKQpAQXBwbGljYXRpb25QYXRoKCJ2MSIpCnB1YmxpYyBjbGFzcyBKYXZhU2lBcHBsaWNhdGlvbiBleHRlbmRzIEFwcGxpY2F0aW9uIHsuLi59CmBgYAoKIyMjIERvY3VtZW50aW5nIHJlc291cmNlIGNsYXNzIGFuZCBvcGVyYXRpb25zCmBgYGphdmEKQFBhdGgoInNlc3Npb25zIikKQENvbnN1bWVzKE1lZGlhVHlwZS5BUFBMSUNBVElPTl9KU09OKQpAUHJvZHVjZXMoTWVkaWFUeXBlLkFQUExJQ0FUSU9OX0pTT04pCnB1YmxpYyBjbGFzcyBTZXNzaW9uc1Jlc291cmNlIHsKCiAgICBAT3BlcmF0aW9uKGRlc2NyaXB0aW9uID0gIlJldHVybnMgbGlzdCBvZiBzZXNzaW9ucy4iLCBzdW1tYXJ5ID0gIlNlc3Npb25zIGxpc3QiLCB0YWdzID0gInNlc3Npb25zIiwgcmVzcG9uc2VzID0gewogICAgICAgICAgICBAQXBpUmVzcG9uc2UocmVzcG9uc2VDb2RlID0gIjIwMCIsCiAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb24gPSAiTGlzdCBvZiBzZXNzaW9ucyIsCiAgICAgICAgICAgICAgICAgICAgY29udGVudCA9IEBDb250ZW50KAogICAgICAgICAgICAgICAgICAgICAgICAgICAgYXJyYXkgPSBAQXJyYXlTY2hlbWEoc2NoZW1hID0gQFNjaGVtYShpbXBsZW1lbnRhdGlvbiA9IFNlc3Npb24uY2xhc3MpKSksCiAgICAgICAgICAgICAgICAgICAgaGVhZGVycyA9IHtASGVhZGVyKG5hbWUgPSAiWC1Ub3RhbC1Db3VudCIsIHNjaGVtYSA9IEBTY2hlbWEodHlwZSA9ICJpbnRlZ2VyIikpfQogICAgICAgICAgICAgICAgICAgICl9KQogICAgQFNlY3VyaXR5UmVxdWlyZW1lbnQobmFtZSA9ICJvcGVuaWQtY29ubmVjdCIpCiAgICBAR0VUCiAgICBwdWJsaWMgUmVzcG9uc2UgZ2V0U2Vzc2lvbnMoKSB7Li4ufSAKICAgIC4uLgp9CmBgYAoKIyMgQWNjZXNzaW5nIEFQSSBzcGVjaWZpY2F0aW9uCgpCdWlsZCBhbmQgcnVuIHByb2plY3QgdXNpbmc6CgpgYGBiYXNoCm12biBjbGVhbiBwYWNrYWdlCmphdmEgLWphciB0YXJnZXQvJHtwcm9qZWN0LmJ1aWxkLmZpbmFsTmFtZX0uamFyCmBgYAoKQWZ0ZXIgc3RhcnR1cCBBUEkgc3BlY2lmaWNhdGlvbiB3aWxsIGJlIGF2YWlsYWJsZSBhdDoKCioqaHR0cDovLzwtaG9zdG5hbWUtOjwtcG9ydC0+L2FwaS1zcGVjcy88LWFwcGxpY2F0aW9uLWJhc2UtcGF0aC0+L29wZW5hcGkuW2pzb24seWFtbF0qKgoKRXhhbXBsZToKCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGktc3BlY3MvdjEvb3BlbmFwaS5qc29uCgpgL2FwaS1zcGVjc2AgbWFwcGluZyBjYW4gYmUgcmVuYW1lZCBieSBzZXR0aW5nIHRoZSBjb25maWd1cmF0aW9uIHByb3BlcnR5IGBrdW11bHV6ZWUub3BlbmFwaS5zZXJ2bGV0Lm1hcHBpbmdgLCBGb3IKZXhhbXBsZSB0byBtb3ZlIHRoZSBhYm92ZSBleGFtcGxlIHRvIHVybCBodHRwOi8vbG9jYWxob3N0OjgwODAvb3BlbmFwaS1zcGVjcy92MS9vcGVuYXBpLmpzb24gdXNlIHRoZSBmb2xsb3dpbmcKY29uZmlndXJhdGlvbjoKCmBgYHltbAprdW11bHV6ZWU6CiAgb3BlbmFwaToKICAgIHNlcnZsZXQ6CiAgICAgIG1hcHBpbmc6IG9wZW5hcGktc3BlY3MKYGBgCgpTZXJ2aW5nIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBjYW4gYmUgZGlzYWJsZWQgYnkgc2V0dGluZyBwcm9wZXJ0eSAqKmt1bXVsdXplZS5vcGVuYXBpLnNwZWMuZW5hYmxlZCoqIHRvIGZhbHNlLiBCeSBkZWZhdWx0IHNlcnZpbmcgQVBJIHNwZWMgaXMgZW5hYmxlZC4KCiMjIEFkZGluZyBPcGVuQVBJIFVJCgpUbyBzZXJ2ZSBBUEkgc3BlY2lmaWNhdGlvbiBpbiB2aXN1YWwgZm9ybSBhbmQgdG8gYWxsb3cgQVBJIGNvbnN1bWVycyB0byBpbnRlcmFjdCB3aXRoIEFQSSByZXNvdXJjZXMgeW91IGNhbiBhZGQgT3BlbkFQSSBVSSBieSBpbmNsdWRpbmcgZGVwZW5kZW5jeToKICAKYGBgeG1sCjxkZXBlbmRlbmN5PgogICAgPGdyb3VwSWQ+Y29tLmt1bXVsdXouZWUub3BlbmFwaTwvZ3JvdXBJZD4KICAgIDxhcnRpZmFjdElkPmt1bXVsdXplZS1vcGVuYXBpLXVpPC9hcnRpZmFjdElkPgogICAgPHZlcnNpb24+JHtrdW11bHV6ZWUtb3BlbmFwaS52ZXJzaW9ufTwvdmVyc2lvbj4KPC9kZXBlbmRlbmN5PgpgYGAKCkRlcGVuZGVuY3kgd2lsbCBpbmNsdWRlIE9wZW5BUEkgVUkgYXJ0aWZhY3RzLCBpbiBjYXNlIHlvdSB3YW50IHRvIHRlbXBvcmFyaWx5IGRpc2FibGUgT3BlbkFQSSBVSSB5b3UgY2FuIGRvIHNvIGJ5IHNldHRpbmcgY29uZmlndXJhdGlvbiBwcm9wZXJ0eToKIApgYGB5YW1sCmt1bXVsdXplZToKICBvcGVuYXBpOgogICAgdWk6CiAgICAgIGVuYWJsZWQ6IGZhbHNlCmBgYAoKQWZ0ZXIgc3RhcnR1cCBPcGVuQVBJIFVJIGlzIGF2YWlsYWJsZSBhdDogaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS1zcGVjcy91aS4KCklmIHlvdSB3YW50IHRvIGRpc2FibGUgT3BlbkFQSSBEZXBlbmRlbmN5IHlvdSBjYW4gc2V0IHRoZSBmb2xsb3dpbmcgcHJvcGVydHk6CiAKYGBgeWFtbAprdW11bHV6ZWU6CiAgb3BlbmFwaToKICAgIGVuYWJsZWQ6IGZhbHNlCmBgYAoKIyMgQ2hhbmdlbG9nCgpSZWNlbnQgY2hhbmdlcyBjYW4gYmUgdmlld2VkIG9uIEdpdGh1YiBvbiB0aGUgW1JlbGVhc2VzIFBhZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9rdW11bHV6L2t1bXVsdXplZS1vcGVuYXBpL3JlbGVhc2VzKQoKCiMjIENvbnRyaWJ1dGUKClNlZSB0aGUgW2NvbnRyaWJ1dGluZyBkb2NzXShodHRwczovL2dpdGh1Yi5jb20va3VtdWx1ei9rdW11bHV6ZWUtb3BlbmFwaS9ibG9iL21hc3Rlci9DT05UUklCVVRJTkcubWQpCgpXaGVuIHN1Ym1pdHRpbmcgYW4gaXNzdWUsIHBsZWFzZSBmb2xsb3cgdGhlIApbZ3VpZGVsaW5lc10oaHR0cHM6Ly9naXRodWIuY29tL2t1bXVsdXova3VtdWx1emVlLW9wZW5hcGkvYmxvYi9tYXN0ZXIvQ09OVFJJQlVUSU5HLm1kI2J1Z3MpLgoKV2hlbiBzdWJtaXR0aW5nIGEgYnVnZml4LCB3cml0ZSBhIHRlc3QgdGhhdCBleHBvc2VzIHRoZSBidWcgYW5kIGZhaWxzIGJlZm9yZSBhcHBseWluZyB5b3VyIGZpeC4gU3VibWl0IHRoZSB0ZXN0IAphbG9uZ3NpZGUgdGhlIGZpeC4KCldoZW4gc3VibWl0dGluZyBhIG5ldyBmZWF0dXJlLCBhZGQgdGVzdHMgdGhhdCBjb3ZlciB0aGUgZmVhdHVyZS4KCiMjIExpY2Vuc2UKCk1JVAo= readmeEtag: '"44eaa54e982e4c111f43373f67db6f2eae16a394"' readmeLastModified: Thu, 15 Oct 2020 09:42:30 GMT repositoryId: 108509681 description: >- KumuluzEE OpenAPI extension provides powerful tools to incorporate and visualize the OpenAPI 3 specification to your microservice. created: '2017-10-27T06:46:32Z' updated: '2020-10-15T09:42:33Z' language: Java archived: false stars: 2 watchers: 14 forks: 0 owner: kumuluz logo: https://avatars.githubusercontent.com/u/6859905?v=4 license: NOASSERTION repoEtag: '"3faaed07d4911c9d6f4cc57966da247665dd545e43ade14be08ffe2d593ca2ce"' repoLastModified: Thu, 15 Oct 2020 09:42:33 GMT foundInMaster: true category: Parsers id: 35734194256a144e4a14160acdce2f5a - source: openapi3 tags repository: https://github.com/quetz-al/quetzal-client v3: true repositoryMetadata: base64Readme: >- Li4gTm90ZSB0aGF0IHRoaXMgZmlsZSBpcyBpbmNsdWRlZCBvbiBTcGhpbnggYXMgd2VsbC4KClF1ZXR6YWwgUHl0aG9uIGNsaWVudAo9PT09PT09PT09PT09PT09PT09PT0KClB5dGhvbiBjbGllbnQgZm9yIHRoZSBRdWV0emFsIEFQSS4KCklmIHlvdSBhcmUgbm90IGZhbWlsaWFyIHdpdGggdGhlIFF1ZXR6YWwgQVBJLCByZWFkIGl0cwpgZG9jdW1lbnRhdGlvbiA8aHR0cHM6Ly9xdWV0emFsLWFwaS5yZWFkdGhlZG9jcy5vcmc+YF8gZmlyc3QuIFRoaXMgUHl0aG9uCnBhY2thZ2UgcHJvdmlkZXMgYSBjb21tYW5kLWxpbmUgYW5kIGhlbHBlciBmdW5jdGlvbnMgdG8gaW50ZXJhY3Qgd2l0aCB0aGUKUXVldHphbCBBUEkuCgpOb3RlIHRoYXQgdGhpcyBwYWNrYWdlIGRlcGVuZHMgb24gYW4gYXV0by1nZW5lcmF0ZWQgcGFja2FnZQpgcXVldHphbC5vcGVuYXBpX2NsaWVudCA8aHR0cHM6Ly9naXRodWIuY29tL3F1ZXR6LWFsL3F1ZXR6YWwtb3BlbmFwaS1jbGllbnQ+YF8sCndoaWNoIGlzIGFsc28gYSBjbGllbnQgdG8gdGhpcyBBUEkuIFlvdSBjYW4gdXNlIHRoZSBsYXR0ZXIgYXMgYSBwdXJlIFB5dGhvbgpjbGllbnQsIGJ1dCAqcXVldHphbC5jbGllbnQqIHByb3ZpZGVzIHNvbWUgaGVscGVycyBhbmQgc21hbGwgZml4ZXMuCgpJbnN0YWxsYXRpb24KLS0tLS0tLS0tLS0tCgpZb3UgY2FuIGluc3RhbGwgKnF1ZXR6YWwtY2xpZW50Kiwgd2l0aCBgYHBpcGBgOgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICBwaXAgaW5zdGFsbCBxdWV0emFsLWNsaWVudAoKQWx0ZXJuYXRpdmVseSwgYWRkIHRoaXMgbGluZSB0byB5b3VyIGBgcmVxdWlyZW1lbnRzLnR4dGBgOgoKLi4gY29kZS1ibG9jazo6IG5vbmUKCiAgICBxdWV0emFsLWNsaWVudAoKYW5kIHRoZW4gZG8gYGBwaXAgaW5zdGFsbCAtciByZXF1aXJlbWVudHMudHh0YGAuCgpJZiB5b3UgYXJlIHVzaW5nIGNvbmRhLCBhZGQgdGhlIGZvbGxvd2luZyBzdHJ1Y3R1cmUgdG8geW91ciBgYGVudmlyb25tZW50LnlhbWxgYDoKCi4uIGNvZGUtYmxvY2s6OiB5YW1sCgogICAgLi4uCiAgICBkZXBlbmRlbmNpZXM6CiAgICAgIC0gcGlwCiAgICAgIC0gcGlwOgogICAgICAgIC0gcXVldHphbC1jbGllbnQKCmFuZCBjcmVhdGUgb3IgdXBkYXRlIHlvdXIgZW52aXJvbm1lbnQgd2l0aApgYGNvbmRhIGVudiBjcmVhdGUgLWYgZW52aXJvbm1lbnQueWFtbGBgIG9yCmBgY29uZGEgZW52IHVwZGF0ZSAtZiBlbnZpcm9ubWVudC55YW1sYGAsIHJlc3BlY3RpdmVseS4KCgpHZXR0aW5nIHN0YXJ0ZWQKLS0tLS0tLS0tLS0tLS0tCgpJbiBvcmRlciB0byB1c2UgKnF1ZXR6YWwuY2xpZW50KiwgeW91IG5lZWQgdG8ga25vdyB0aGUgVVJMIG9mIHRoZSBBUEkgc2VydmVyLAp5b3VyIHVzZXJuYW1lIGFuZCBwYXNzd29yZC4gWW91IGNhbiBzZXQgdGhlc2Ugb24gdGhlIGNvbW1hbmQtbGluZSBpbnRlcmZhY2UsCnRocm91Z2ggYSBjb25maWd1cmF0aW9uIG9iamVjdCwgb3IgdXNpbmcgZW52aXJvbm1lbnQgdmFyaWFibGVzOgoKLi4gbGlzdC10YWJsZTo6IEVudmlyb25tZW50IHZhcmlhYmxlcyBjb25zaWRlcmVkIGJ5IGBxdWV0emFsLWNsaWVudGAuCiAgIDpoZWFkZXItcm93czogMQoKICAgKiAtIFZhcmlhYmxlCiAgICAgLSBEZXNjcmlwdGlvbgogICAgIC0gRGVmYXVsdCBpZiBub3Qgc2V0CiAgICogLSBgYFFVRVRaQUxfVVJMYGAKICAgICAtIENvbXBsZXRlIFVSTCBvZiB0aGUgUXVldHphbCBzZXJ2ZXIgKyBBUEkgdmVyc2lvbi4KICAgICAtIGBgJ2h0dHBzOi8vYXBpLnF1ZXR6LmFsL2FwaS92MSdgYAogICAqIC0gYGBRVUVUWkFMX1VTRVJgYAogICAgIC0gUXVldHphbCB1c2VybmFtZS4KICAgICAtIGBgJydgYAogICAqIC0gYGBRVUVUWkFMX1BBU1NXT1JEYGAKICAgICAtIFF1ZXR6YWwgcGFzc3dvcmQuCiAgICAgLSBgYCcnYGAKICAgKiAtIGBgUVVFVFpBTF9BUElfS0VZYGAKICAgICAtIFF1ZXR6YWwgQVBJIGtleS4KICAgICAtIGBgJydgYAoKCkJhc2ljIHVzYWdlCi0tLS0tLS0tLS0tCgpUaGVyZSBhcmUgdHdvIHdheXMgdGhpcyBwYWNrYWdlIGhlbHBzIHlvdTogd2l0aCBhIGNvbW1hbmQtbGluZSBpbnRlcmZhY2Ugb3IgYnkKcHJvdmlkaW5nIHNvbWUgaGVscGVyIG1vZHVsZXMgdGhhdCB5b3UgY2FuIHVzZSBpbiBQeXRob24uCgpDb21tYW5kLWxpbmUgaW50ZXJmYWNlCl5eXl5eXl5eXl5eXl5eXl5eXl5eXl4KClRoZSBjb21tYW5kLWxpbmUgaW50ZXJmYWNlIGlzIGF2YWlsYWJsZSB0aHJvdWdoIHRoZSBgYHF1ZXR6YWwtY2xpZW50YGAgY29tbWFuZC4KVXNlIHRoZSBgYC0taGVscGBgIG9yIGBgLS1oZWxwLWFsbGBgIG9wdGlvbnMgdG8gZ2V0IGEgZGV0YWlsZWQgZGVzY3JpcHRpb24gb2YKZWFjaCBjb21tYW5kLgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICAkIHF1ZXR6YWwtY2xpZW50IC0taGVscAoKICAgIFVzYWdlOiBxdWV0emFsLWNsaWVudCBbR0xPQkFMIE9QVElPTlNdIENPTU1BTkQgW0FSR1NdLi4uCgogICAgICBDb21tYW5kLWxpbmUgdXRpbGl0eSBmb3IgdGhlIFF1ZXR6YWwgQVBJIGNsaWVudC4KCiAgICBPcHRpb25zOgogICAgICAtLXVybCBURVhUICAgICAgIFF1ZXR6YWwgVVJMLiBJZiBub3Qgc2V0LCB1c2VzIGVudmlyb25tZW50IHZhcmlhYmxlCiAgICAgICAgICAgICAgICAgICAgICAgUVVFVFpBTF9VUkwgaWYgdGhpcyB2YXJpYWJsZSBpcyBkZWZpbmVkLiAgW2RlZmF1bHQ6CiAgICAgICAgICAgICAgICAgICAgICAgaHR0cHM6Ly9hcGkucXVldHouYWwvYXBpL3YxXQogICAgICAtLXVzZXJuYW1lIFRFWFQgIFF1ZXR6YWwgdXNlcm5hbWUuIElmIG5vdCBzZXQsIHVzZXMgZW52aXJvbm1lbnQgdmFyaWFibGUKICAgICAgICAgICAgICAgICAgICAgICBRVUVUWkFMX1VTRVIuIE9wdGlvbiBpcyBtdXR1YWxseSBleGNsdXNpdmUgd2l0aCB0b2tlbiwKICAgICAgICAgICAgICAgICAgICAgICBhcGlfa2V5LgogICAgICAtLXBhc3N3b3JkIFRFWFQgIFF1ZXR6YWwgcGFzc3dvcmQuIElmIG5vdCBzZXQsIHVzZXMgZW52aXJvbm1lbnQgdmFyaWFibGUKICAgICAgICAgICAgICAgICAgICAgICBRVUVUWkFMX1BBU1NXT1JELiBPcHRpb24gaXMgbXV0dWFsbHkgZXhjbHVzaXZlIHdpdGggdG9rZW4sCiAgICAgICAgICAgICAgICAgICAgICAgYXBpX2tleS4KICAgICAgLS10b2tlbiBURVhUICAgICBRdWV0emFsIGJlYXJlciB0b2tlbi4gSWYgbm90IHNldCwgdXNlcyBlbnZpcm9ubWVudCB2YXJpYWJsZQogICAgICAgICAgICAgICAgICAgICAgIFFVRVRaQUxfVE9LRU4uIE9wdGlvbiBpcyBtdXR1YWxseSBleGNsdXNpdmUgd2l0aCB1c2VybmFtZSwKICAgICAgICAgICAgICAgICAgICAgICBwYXNzd29yZCwgYXBpX2tleS4KICAgICAgLS1hcGkta2V5IFRFWFQgICBRdWV0emFsIEFQSSBrZXkuIElmIG5vdCBzZXQsIHVzZXMgZW52aXJvbm1lbnQgdmFyaWFibGUKICAgICAgICAgICAgICAgICAgICAgICBRVUVUWkFMX0FQSV9LRVkgT3B0aW9uIGlzIG11dHVhbGx5IGV4Y2x1c2l2ZSB3aXRoIHVzZXJuYW1lLAogICAgICAgICAgICAgICAgICAgICAgIHBhc3N3b3JkLCB0b2tlbi4KICAgICAgLS1pbnNlY3VyZSAgICAgICBEbyBub3QgdmVyaWZ5IEhUVFBTIGNlcnRpZmljYXRlcy4KICAgICAgLXYsIC0tdmVyYm9zZSAgICBWZXJib3NpdHkgbGV2ZWwuIFVzZSAtdiBmb3IgdmVyYm9zZSwgLXZ2IGZvciBldmVuIG1vcmUKICAgICAgICAgICAgICAgICAgICAgICB2ZXJib3NpdHkKICAgICAgLS1oZWxwLWFsbCAgICAgICBTaG93IGEgZGV0YWlsZWQgaGVscCBtZXNzYWdlIHdpdGggYWxsIG9wdGlvbnMgYW5kIGV4aXQuCiAgICAgIC0taGVscCAgICAgICAgICAgU2hvdyBoZWxwIG1lc3NhZ2UgZm9yIHRoaXMgY29tbWFuZCBhbmQgZXhpdC4KICAgICAgLS12ZXJzaW9uICAgICAgICBTaG93IHRoZSB2ZXJzaW9uIGFuZCBleGl0LgoKICAgIENvbW1hbmRzOgogICAgICBhdXRoICAgICAgIEF1dGhlbnRpY2F0aW9uIG9wZXJhdGlvbnMuCiAgICAgIGZpbGUgICAgICAgRmlsZSBvcGVyYXRpb25zLgogICAgICBxdWVyeSAgICAgIFF1ZXJ5IG1ldGFkYXRhLgogICAgICB3b3Jrc3BhY2UgIFdvcmtzcGFjZSBvcGVyYXRpb25zLgoKClB5dGhvbgpeXl5eXl4KClRvIHN0YXJ0IHVzaW5nICpxdWV0emFsLmNsaWVudCogb24gUHl0aG9uIGNvZGUsIHVzZSB0aGUgZm9sbG93aW5nIHRlbXBsYXRlOgoKLi4gY29kZS1ibG9jazo6IHB5dGhvbgoKICAgIGZyb20gcXVldHphbC5jbGllbnQgaW1wb3J0IENsaWVudCwgQ29uZmlndXJhdGlvbiwgUXVldHphbEFQSUV4Y2VwdGlvbgogICAgZnJvbSBxdWV0emFsLmNsaWVudCBpbXBvcnQgaGVscGVycwoKICAgIGNvbmZpZyA9IENvbmZpZ3VyYXRpb24oKQogICAgIyAuLi4gY2hhbmdlIGNvbmZpZyBhcyBuZWVkZWQgb3IgZmFsbGJhY2sgdG8gdGhlIGVudmlyb25tZW50IHZhcmlhYmxlcyAuLi4KICAgICMgY29uZmlnLnZlcmlmeV9zc2wgPSBGYWxzZSAgIyBVc2UgdGhpcyBmb3Igc2VydmVycyB3aXRob3V0IGNlcnRpZmljYXRlcyAoZGV2IHNlcnZlcnMpCiAgICBjbGllbnQgPSBDbGllbnQoY29uZmlnKQoKICAgICMgQSBzaW1wbGUgdGVzdCB1c2luZyB0aGUgaGVscGVycyB0byB2ZXJpZnkgdGhhdCB3ZSBjYW4gbG9naW4gdG8gUXVldHphbAogICAgdHJ5OgogICAgICAgIGhlbHBlcnMuYXV0aC5sb2dpbihjbGllbnQpCiAgICAgICAgcHJpbnQoJ0xvZ2dlZCBpbiBzdWNjZXNzZnVsbHkhJykKICAgIGV4Y2VwdCBRdWV0emFsQVBJRXhjZXB0aW9uIGFzIGV4OgogICAgICAgIHByaW50KGYnT3BlcmF0aW9uIGZhaWxlZC4ge2V4LnRpdGxlfSAtIHtleC5kZXRhaWx9JykKCgpEb2N1bWVudGF0aW9uCi0tLS0tLS0tLS0tLS0KClRoZXJlIGFyZSBtb3JlIGRldGFpbHMgb24gKnF1ZXR6YWwuY2xpZW50KiBvbiBpdHMgb2ZmaWNpYWwgZG9jdW1lbnRhdGlvbiBhdApgcmVhZHRoZWRvY3MgPGh0dHBzOi8vcXVldHphbC1jbGllbnQucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0Lz5gXy4KCkNvbnRyaWJ1dGUKLS0tLS0tLS0tLQoKLSBJc3N1ZSBUcmFja2VyOiBodHRwczovL2dpdGh1Yi5jb20vcXVldHotYWwvcXVldHphbC1jbGllbnQvaXNzdWVzCi0gU291cmNlIENvZGU6IGh0dHBzOi8vZ2l0aHViLmNvbS9xdWV0ei1hbC9xdWV0emFsLWNsaWVudAoKTm90ZSB0byBwYWNrYWdlIG1haW50YWluZXIKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KClRvIGJ1aWxkIGFuZCBzZW5kIHRoaXMgcGFja2FnZSB0byBQeVBJOgoKLi4gY29kZS1ibG9jazo6IGNvbnNvbGUKCiAgICAjIENsZWFuIHByZXZpb3VzIGJ1aWxkcyEKICAgIHJtIC1yZiBkaXN0LwogICAgIyBCdWlsZAogICAgcHl0aG9uIHNldHVwLnB5IHNkaXN0IGJkaXN0X3doZWVsCgogICAgIyBJbnN0YWxsIGhlbHBlciBtb2R1bGUgZm9yIHVwbG9hZGluZyB0byBQeVBJCiAgICBwaXAgaW5zdGFsbCB0d2luZQogICAgIyBGaXJzdCwgc2VuZCB0byB0ZXN0IFB5UEkKICAgIHR3aW5lIHVwbG9hZCAtLXJlcG9zaXRvcnktdXJsIGh0dHBzOi8vdGVzdC5weXBpLm9yZy9sZWdhY3kvIGRpc3QvKgogICAgIyBJZiB0aGF0IHdvcmtzIG9rLCB0aGVuIHNlbmQgdG8gUHlQSQogICAgcHl0aG9uIC1tIHR3aW5lIHVwbG9hZCBkaXN0LyoKCkxpY2Vuc2UKLS0tLS0tLQoKVGhlIHByb2plY3QgaXMgdW5kZXIgdGhlIEJTRCAzLWNsYXVzZSBsaWNlbnNlLgo= readmeEtag: '"ea03b32059fbdfbf802a1fef988008f5a7ba6373"' readmeLastModified: Fri, 05 Jun 2020 13:55:43 GMT repositoryId: 170111505 description: Python client for the Quetzal API created: '2019-02-11T10:41:13Z' updated: '2020-06-05T13:57:52Z' language: Python archived: false stars: 2 watchers: 0 forks: 1 owner: quetz-al logo: https://avatars.githubusercontent.com/u/47522179?v=4 license: BSD-3-Clause repoEtag: '"6399e3323b9d6986d2c3c9a0e1b79b75cadf8ff32a60512b668433108b05dfe0"' repoLastModified: Fri, 05 Jun 2020 13:57:52 GMT foundInMaster: true category: Server Implementations id: 2607f54b49a6ccbc03114f8a47b48a6c - source: openapi3 tags repository: https://github.com/c-classen/oak-dsl v3: true repositoryMetadata: base64Readme: >- # 🌳 OpenAPI Kotlin Domain Specific Language

![Maven Central Badge](https://img.shields.io/maven-central/v/io.github.c-classen.oakdsl/oakdsl)

This Kotlin library offers a concise way to define your OpenAPI specification.
For example, it is able to turn this code which defines some CRUD endpoints to manage a todo list

```kotlin
@file:CompilerOptions("-jvm-target", "1.8")
@file:DependsOn("io.github.c-classen.oakdsl:oakdsl:0.1.3")

import io.github.cclassen.oakdsl.builder.OpenApiBuilder

OpenApiBuilder.file("api.yaml") {
    info {
        title = "Test api"
        version = "1.0.0"
    }

    post<Todo>("createTodo") / "todos" def {
        requestBody<TodoPayload>()
    }
    get<List<Todo>>("readTodos") / "todos" def {}
    get<Todo>("readTodo") / "todos" / longParam("todoId") def {}
    put<Todo>("updateTodo") / "todos" / longParam("todoId") def {}
    patch<Todo>("markAsDone") / "todos" / longParam("todoId") def {
        boolParam("done")
    }
    delete("deleteTodo") / "todos" / longParam("todoId") def {}
}

interface Todo {
    val id: Long
    val text: String
    val done: Boolean
}

interface TodoPayload {
    val text: String
}
```

into the following OpenAPI Spec:

```yaml
# This file is generated by oak-dsl (https://github.com/c-classen/oak-dsl)
openapi: 3.0.3
info:
  version: "1.0.0"
  title: "Test api"
paths:
  "/todos":
    get:
      operationId: "readTodos"
      responses:
        "200":
          description: "Success"
          content:
            "application/json":
              schema:
                type: "array"
                items:
                  "$ref": "#/components/schemas/Todo"
    post:
      operationId: "createTodo"
      requestBody:
        required: true
        content:
          "application/json":
            schema:
              "$ref": "#/components/schemas/TodoPayload"
      responses:
        "200":
          description: "Success"
          content:
            "application/json":
              schema:
                "$ref": "#/components/schemas/Todo"
  "/todos/{todoId}":
    delete:
      operationId: "deleteTodo"
      parameters:
        - in: path
          name: "todoId"
          required: true
          schema:
            type: integer
            format: "int64"
      responses:
        "200":
          description: "Success"
    get:
      operationId: "readTodo"
      parameters:
        - in: path
          name: "todoId"
          required: true
          schema:
            type: integer
            format: "int64"
      responses:
        "200":
          description: "Success"
          content:
            "application/json":
              schema:
                "$ref": "#/components/schemas/Todo"
    patch:
      operationId: "markAsDone"
      parameters:
        - in: path
          name: "todoId"
          required: true
          schema:
            type: integer
            format: "int64"
        - in: query
          name: "done"
          required: true
          schema:
            type: boolean
      responses:
        "200":
          description: "Success"
          content:
            "application/json":
              schema:
                "$ref": "#/components/schemas/Todo"
    put:
      operationId: "updateTodo"
      parameters:
        - in: path
          name: "todoId"
          required: true
          schema:
            type: integer
            format: "int64"
      responses:
        "200":
          description: "Success"
          content:
            "application/json":
              schema:
                "$ref": "#/components/schemas/Todo"
components:
  schemas:
    Todo:
      type: "object"
      required: [ "done", "id", "text" ]
      properties:
        done:
          type: boolean
        id:
          type: integer
          format: "int64"
        text:
          type: string
    TodoPayload:
      type: "object"
      required: [ "text" ]
      properties:
        text:
          type: string
```

Although the resulting yaml could be optimized by hand to become shorter (e.g., by defining the `todoId` parameter in the components section), it would still not be nearly as short as the oak-dsl version.
To achieve this brevity, oak-dsl uses two techniques.

First, it utilizes Kotlin's syntax features to make certain definitions have less boilerplate.
For example, defining a property in a schema is just one line in Kotlin (`val name: Type`) while it takes two or even three lines in yaml:

```yaml
name:
  type: Type
```

Second, it makes assumptions about things that you have to specify explicitly in yaml.
Take the following line as an example.

```kotlin
get<List<Todo>>("readTodos") / "todos" def {}
```

First we specify that the endpoint uses a GET HTTP method.
We then say that it returns a List of Todo objects which are defined at a later point.
After that, we specify the operation id as `readTodos` and the path as `/todos`.
We could have used the block following the `def` keyword to further customize the endpoint, but we did not need to.

This line corresponds to the following OpenAPI definition:

```yaml
  "/todos":
    get:
      operationId: "readTodos"
      responses:
        "200":
          description: "Success"
          content:
            "application/json":
              schema:
                type: "array"
                items:
                  "$ref": "#/components/schemas/Todo"
```

OpenAPI requires more information than we specified, so oak-dsl just filled them with default values.
The first one is the response code which will usually be `200` in case no error happened.
Second is the description which is often not necessary, as the meaning of the response is clear from the context.
Oak-dsl defaults to `"Success"` as a description of the default response.
The third default parameter is the content type.
There probably will be cases, where `application/json` is not appropriate, e.g., because we want to download a file, but most of the time it does the job.

# Usage

To generate your OpenAPI Specification with oak-dsl, you create a file named `api.main.kts` and define your endpoints as seen in the first example above. 

To generate the yaml file, you will need to install the [Kotlin compiler](https://kotlinlang.org/docs/command-line.html).
Once you have done that, execute `kotlinc -script api.main.kts` from the command line in the directory with the oak-dsl definition file.

If you are using IntelliJ, you will have working auto-completion for the `api.main.kts` file.
You can also use IntelliJ to run your script, but this may currently fail in case you have other Kotlin code with errors in your project.



# Features and Documentation

This section explains how the features of oak-dsl can help you quickly write your OpenAPI definition.

## Basics

Your oak-dsl file should usually look as follows:

```kotlin
@file:CompilerOptions("-jvm-target", "1.8")
@file:DependsOn("io.github.cclassen.oakdsl:oakdsl:VERSION")

import io.github.cclassen.oakdsl.builder.OpenApiBuilder

OpenApiBuilder.file("api.yaml") {
    // Endpoints
}

// Types
```

The first and second line configure the Kotlin compiler and declare the dependency on oak-dsl respectively.
Then the script imports the class `OpenApiBuilder` and calls the `file` method on its companion object.
It takes the filename of the OpenAPI YAML file to be generated, and a lambda where the file's contents can be configured.
Within that lambda, first the contents of the info section of the specification are declared.
It is also possible to define more metadata within that section and to declare servers by using the following method call.

```kotlin
server("http://example.com/", "Some Description")
```

## Defining Endpoints

To define an endpoint, you need to call the method corresponding to the HTTP method.
It takes an optional type parameter.
If you specify it, oak-dsl will generate a `200` response with content type `application/json` and a schema that references a type corresponding to the type parameter.
The only value parameter is the operation id.
After the method call, you can define the path of the endpoint using the `/` operator on the resulting `EndpointPathBuilder`.
As second operand you can either use a string literal, or a parameter which is explained later.
After appending all path segments, you can conclude the endpoint definition using the `def` infix function followed by a lambda that allows you to further customize your endpoint.
You can use it to define a description, tags, responses and the request body.

## Defining Types

You can use either interfaces or classes to define the types you use as your schemas for requests and responses.
Interfaces allow for easy inheritance which is represented by a schema using `allOf`.
However, the order of the fields cannot be preserved and is therefore alphabetically.
For classes, you need to put the fields in the primary constructor like this:

```kotlin
class Example(
  firstField: Int,
  secondField: String
)
```

All types that are referenced directly or indirectly by your specification will be translated to a schema.
Primitive types are translated to their json schema counterparts and classes, interfaces and enumerations will be defined in the `components/schemas` section of the resulting YAML.
It is therefore important that they have unique names.

You can also define schemas by constructing the classes in the package `io.github.cclassen.oakdsl.model.schema` directly.
It is also possible to override type resolution for your own or built-in types, as shown in the following example:

```kotlin
OpenApiBuilder.file("api.yaml") {
  val dbId = components.type("DbId", PrimitiveSchema("integer", "int64"))
  customResolve<DbId> { dbId }
  // Endpoints
}

interface DbId
```

By calling `components.type`, you can register your constructed type to be output in the `components/schemas` section of the YAML.

## Defining parameters

For oak-dsl, two kinds of parameters must be distinguished.
Path parameters must be declared when building the path, like in the following example:

```kotlin
get<Todo>("readTodo") / "todos" / longParam("todoId") def {}
```

Query and header parameters are declared within the endpoint definition:

```kotlin
get<Todo>("readTodo") / "todos" def {
  longParam("todoId")
}
```

You can also declare have your parameter declared in the `components/parameters` section of the YAML and then refer to it later by reference:

```kotlin
val idParam = components.parameter("idParam", "id", components.resolveClass(Long::class), kind = "path")
  
get<Todo>("readTodo") / "todos" / idParam def {}

```

## Endpoint Filters

Endpoint filters allow you to have sections in your oak-dsl definition that enhance readability and allow you to apply certain transformations to all endpoints within the section.
They can be used as follows:

```kotlin
val prefix = endpointFilter { path = "/todos$path" }
    
prefix {
    get<List<Todo>>("readTodos") def {}
    get<Todo>("readTodo") / longParam("todoId") def {}
}
```

The two endpoints within the `prefix` section will have the `/todos` path segment prepended to their path.
Therefore, it does not need to be specified when declaring the endpoints.
This reduces verbosity and gives the oak-dsl document more structure.
When declaring an endpoint filter, you specify a lambda receiving an `EndpointItem` which allows you to modify the Http-`method`, the `path` and the other properties within the `endpoint`. 
You can also combine multiple filters by using their `and`-method which returns a new filter that applies first the right-hand filter and then the left-hand one.

### Prefix

There is no need to declare a custom endpoint filter if you want to prepend a prefix to all paths within the filter.
You can just use the built-in `prefix`-method which creates a filter with a prefix you specify:

```kotlin
prefix("/prefix") {
    // endpoints
}
```

### Tagged

The same goes in case you want to add a tag to each endpoint.

```kotlin
tagged("SomeTag") {
    // endpoints
}
```

### Marked

Endpoints can be marked, which has no effect on the generated YAML, but can be used to distinguish certain endpoints in post-processing.

```kotlin
enum class MicroService { A, B }

marked(MicroService.A) {
  post("hello") / "hello" def {}
}
```

## Global filters and custom extensions

Endpoint filters can also be applied to all endpoints by using the `globalFilter` method.
However, you cannot modify the HTTP method or the path this way.
This is a convenient way to put some custom extensions on your endpoints that depend on endpoint metadata.

```kotlin
globalFilterForMarked(MicroService.A) {
    endpoint.additionalYamlProperties["x-my-extension"] = YamlMap.build {
        map("my-map") {
            array("my-array") {
                string("SomeString")
                value("SomeValue")
            }
            shortArray("my-short-array", listOf(1, true, -1.3, "Test"))
            shortStringArray("my-string-array", listOf("a", "b"))
            string("someString", "Teststring")
        }
    }
}
```

In this code, a custom extension is added to all endpoints that were marked with `MicroService.A` (where `MicroService` is an enum).
The extension will be serialized to the following yaml:

```yaml
"x-my-extension":
  "my-map":
    "my-array":
    - "SomeString"
    - SomeValue
    "my-short-array": [ 1, true, -1.3, Test ]
    "my-string-array": [ "a", "b" ]
    someString: "Teststring"
```
 readmeEtag: '"51411dad6c2ae8bc4ef8453a67f13967d62424e8"' readmeLastModified: Thu, 02 Dec 2021 14:29:56 GMT repositoryId: 398049382 description: Create OpenAPI Specifications using Kotlin created: '2021-08-19T19:10:39Z' updated: '2023-04-30T07:30:21Z' language: Kotlin archived: false stars: 2 watchers: 1 forks: 0 owner: c-classen logo: https://avatars.githubusercontent.com/u/58041050?v=4 license: MIT repoEtag: '"09effd44d2c1f1be1065340c837ccc5fc4aba3f9f08198d95e867aeae8ac0202"' repoLastModified: Sun, 30 Apr 2023 07:30:21 GMT foundInMaster: true id: 4373894238bf0f136bc8d79d25df5f69 - source: openapi3 tags repository: https://github.com/37mobileteam/appstoreconnectcerts v3: true id: b8b614c0a3ec633d4b7805b71a1f3f13 repositoryMetadata: base64Readme: >- IyBBcHBTdG9yZUNvbm5lY3RDZXJ0cwpQcm9qZWN0IG9mIENlcnRpZmljYXRlIE1hbmFnZW1lbnQgQXV0b21hdGlvbiAtIEJhY2tlbmQK readmeEtag: '"cb476f86edba5c0a98c02fe2fea0a74407c83993"' readmeLastModified: Mon, 25 Sep 2023 10:14:11 GMT repositoryId: 694456934 description: Project of Certificate Management Automation - Backend created: '2023-09-21T03:17:49Z' updated: '2023-09-27T03:53:43Z' language: C archived: false stars: 2 watchers: 1 forks: 3 owner: 37MobileTeam logo: https://avatars.githubusercontent.com/u/85087913?v=4 license: GPL-3.0 repoEtag: '"8e314d8612f3b444a727ea6d46228aa63de8f7ec6b361e695cd4906dddbcca1f"' repoLastModified: Wed, 27 Sep 2023 03:53:43 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mheap/json-schema-spell-checker v3: true repositoryMetadata: base64Readme: >- IyBqc29uLXNjaGVtYS1zcGVsbC1jaGVja2VyCgpUaGlzIENMSSBhbGxvd3MgeW91IHRvIHByb3ZpZGUgYSBKU09OUGF0aCBleHByZXNzaW9uIGFuZCBydW4gYG1kc3BlbGxgIGFnYWluc3QgYW55IG1hdGNoaW5nIGxpbmVzLgoKPiBUaGlzIHRvb2wgaXMgY3VycmVudGx5IGluIGFscGhhIGFuZCBoYXMgcHJpbWFyaWx5IGJlZW4gdGVzdGVkIGFnYWluc3QgT3BlbkFQSSBmaWxlcy4gUFJzIGFyZSB3ZWxjb21lIQoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCm5wbSBpbnN0YWxsIC1nIGpzb24tc2NoZW1hLXNwZWxsLWNoZWNrZXIKYGBgCgojIyBVc2FnZQoKQ2hlY2sgYWdhaW5zdCBzcGVjaWZpYyBmaWVsZCBuYW1lcyBhdCBhbnkgZGVwdGg6CgpgYGBiYXNoCmpzb24tc2NoZW1hLXNwZWxsLWNoZWNrZXIgLWYgJ2Rlc2NyaXB0aW9uLHRpdGxlJyAvcGF0aC90by9vcGVuYXBpLmpzb24KYGBgCgpBbHRlcm5hdGl2ZWx5LCB5b3UgY2FuIHNwZWNpZnkgYSBKU09OUGF0aCBleHByZXNzaW9uIHlvdXJzZWxmCgpgYGBiYXNoCmpzb24tc2NoZW1hLXNwZWxsLWNoZWNrZXIgLWogIiQuLltkZXNjcmlwdGlvbix0aXRsZV0iIC9wYXRoL3RvL29wZW5hcGkuanNvbgpgYGAKCklmIHlvdSBoYXZlIHdvcmRzIHRoYXQgYXJlbid0IGEgc3BlbGxpbmcgbWlzdGFrZSBidXQgYXJlbid0IGluIHRoZSBkaWN0aW9uYXJ5LCBhZGQgdGhlbSB0byBhIGAuc3BlbGxpbmdgIGZpbGUgKG9uZSB3b3JkIHBlciBsaW5lKSBpbiB0aGUgY3VycmVudCBmb2xkZXIuIEFsdGVybmF0aXZlbHksIHlvdSBjYW4gcHJvdmlkZSBhIHBhdGggd2l0aCBgLXNgCgpgYGBiYXNoCmpzb24tc2NoZW1hLXNwZWxsLWNoZWNrZXIgLXMgL3BhdGgvdG8vLnNwZWxsaW5nIC1mICdkZXNjcmlwdGlvbix0aXRsZScgL3BhdGgvdG8vb3BlbmFwaS5qc29uCmBgYAoKIyMgUHJlLWNvbW1pdCBob29rCgpZb3UgY2FuIHVzZSB0aGlzIHRvb2wgYXMgYSBgcHJlLWNvbW1pdGAgaG9vayBieSBhZGRpbmcgdGhlIGZvbGxvd2luZyB0byB5b3VyIGAucHJlLWNvbW1pdC1jb25maWcueWFtbGA6CgpgYGB5YW1sCnJlcG9zOgogIC0gcmVwbzogaHR0cHM6Ly9naXRodWIuY29tL21oZWFwL2pzb24tc2NoZW1hLXNwZWxsLWNoZWNrZXIKICAgIHJldjogPHZlcnNpb24+ICMgVXNlIHRoZSByZWYgeW91IHdhbnQgdG8gcG9pbnQgYXQKICAgIGhvb2tzOgogICAgICAtIGlkOiBqc29uLXNjaGVtYS1zcGVsbC1jaGVja2VyCiAgICAgICAgZmlsZXM6IDxzY2hlbWE+XC5qc29uJApgYGAK readmeEtag: '"592d46c30d35dc5fe57c42553bf329574068209e"' readmeLastModified: Mon, 20 Nov 2023 09:49:07 GMT repositoryId: 227215167 description: >- This CLI allows you to provide a JSONPath expression and run mdspell against any matching lines. created: '2019-12-10T21:03:06Z' updated: '2023-09-08T18:01:07Z' language: JavaScript archived: false stars: 2 watchers: 2 forks: 1 owner: mheap logo: https://avatars.githubusercontent.com/u/59130?v=4 repoEtag: '"4a273c59060dd5f33a08854870cfc0677727f226090d5ee47c406e1f4de0fdd5"' repoLastModified: Fri, 08 Sep 2023 18:01:07 GMT foundInMaster: true category: Parsers id: 9de99ade60b1432266afaf1e0719633a - source: openapi3 tags repository: https://github.com/speakeasy-sdks/template-sdk v3: true id: 8c17078292c83db2e050b190bc267a99 repositoryMetadata: base64Readme: >- # Template SDK

<div align="left">
    <a href="https://speakeasyapi.dev/"><img src="https://custom-icon-badges.demolab.com/badge/-Built%20By%20Speakeasy-212015?style=for-the-badge&logoColor=FBE331&logo=speakeasy&labelColor=545454" /></a>   
</div>

## How to use this repository

**👀** This template repository is designed to bootstrap a [Speakeasy managed SDK repository](https://speakeasyapi.dev/docs/create-client-sdks/) using Github's repository clone feature. Once this repository is setup it will automatically keep your SDK up to date and published to a package manager. 


### Creating an SDK

1. To get started, simply clone the repository by clicking on the "Use template" button and give it a name.
   
![Screenshot 2023-09-06 at 09 20 52](https://github.com/speakeasy-sdks/template-sdk/assets/68016351/b4cf4e43-db4e-455a-9359-0f09f942b997)

2. Configure the Speakeasy workflow to generate the SDK. Go to the [generation workflow file](https://github.com/speakeasy-sdks/template-sdk/blob/main/.github/workflows/speakeasy_sdk_generation.yml) and configure the `language`, `mode` and `location` of your openapi document. For complete documentation on all the available generation configurations, see [here](https://www.speakeasyapi.dev/docs/advanced-setup/sdk-automation). You will also need to add a `SPEAKEASY_API_KEY` as a repository secret. If you don't already have a key you can get one by making a workspace on Speakeasy [here](https://app.speakeasyapi.dev/).

3. Configure the Speakeasy workflow to publish the SDK. Go to the [publishing workflow file](https://github.com/speakeasy-sdks/template-sdk/blob/main/.github/workflows/speakeasy_sdk_publish.yml) and configure any relevant package manager credentials as repository secrets. For complete documentation on all the available publishing configurations, see [here](https://www.speakeasyapi.dev/docs/productionize-sdks/publish-sdks).

4. Configure the generation by editing the `gen.yaml` file in the root of the repo. This file controls the generator and determines various attributes of the SDK like `packageName`, `sdkClassName`, inlining of parameters, and other ergonomics.

5. Finally go to the Actions tab, choose the generation workflow and click "Force Generate". This will trigger a new generation of your SDK using the configuration you provided above. Depending on whether you configured `pr` or `direct` mode above your updated SDK will appear in PR or in the main branch.

![Screenshot 2023-09-06 at 10 01 46](https://github.com/speakeasy-sdks/template-sdk/assets/68016351/35828982-c6de-4a5c-84f5-ae2b4224cece)

🚀 You should have a working SDK for your API 🙂 . To check out all the features of the SDK please see our docs [site](https://speakeasyapi.dev/docs/create-client-sdks/).

### Local development

Once you have the SDK setup you may want to iterate on the SDK. Speakeasy supports OpenAPI vendor extensions that can be added to your spec to customize the SDK ergonomics (method names, namespacing resources etc.) and functionality (adding retries, pagination, multiple server support etc)

To get started install the Speakeasy CLI.

In your terminal, run:

```bash
brew install speakeasy-api/homebrew-tap/speakeasy
```
Once you annonate your spec with an extension you will want to run `speakeasy validate` to check the spec for correctness and `speakeasy generate` to recreate the SDK locally. More documentation on OpenAPI extensions [here](https://speakeasyapi.dev/docs/customize-sdks/namespaces/). Here's an example of adding a multiple server support to the spec so that your SDK supports production and sandbox versions of your API. 

```yaml
info:
  title: Example
  version: 0.0.1
servers:
  - url: https://prod.example.com # Used as the default URL by the SDK
    description: Our production environment
    x-speakeasy-server-id: prod
  - url: https://sandbox.example.com
    description: Our sandbox environment
    x-speakeasy-server-id: sandbox
```

Once you're finished iterating and happy with the output push only the latest version of spec into the repo and regenerate the SDK using step 6 above.

<!-- Start SDK Installation [installation] -->
## SDK Installation

```bash
pip install git+https://github.com/speakeasy-sdks/template-sdk.git
```
<!-- End SDK Installation [installation] -->

<!-- Start SDK Example Usage [usage] -->
## SDK Example Usage

### Sign in

First you need to send an authentication request to the API by providing your username and password.
In the request body, you should specify the type of token you would like to receive: API key or JSON Web Token.
If your credentials are valid, you will receive a token in the response object: `res.object.token: str`.

```python
import speakeasybar
from speakeasybar.models import operations

s = speakeasybar.Speakeasybar()

req = operations.LoginRequestBody(
    type=operations.Type.API_KEY,
)

res = s.authentication.login(req, operations.LoginSecurity(
    password="<PASSWORD>",
    username="<USERNAME>",
))

if res.object is not None:
    # handle response
    pass

```

### Browse available drinks

Once you are authenticated, you can use the token you received for all other authenticated endpoints.
For example, you can filter the list of available drinks by type.

```python
import speakeasybar
from speakeasybar.models import shared

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)


res = s.drinks.list_drinks(drink_type=shared.DrinkType.SPIRIT)

if res.classes is not None:
    # handle response
    pass

```

### Create an order

When you submit an order, you can include a callback URL along your request.
This URL will get called whenever the supplier updates the status of your order.

```python
import speakeasybar
from speakeasybar.models import shared

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)


res = s.orders.create_order(request_body=[
    shared.OrderInput(
        product_code='APM-1F2D3',
        quantity=26535,
        type=shared.OrderType.DRINK,
    ),
], callback_url='<value>')

if res.order is not None:
    # handle response
    pass

```

### Subscribe to webhooks to receive stock updates

```python
import speakeasybar
from speakeasybar.models import operations, shared

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)

req = [
    operations.RequestBody(),
]

res = s.config.subscribe_to_webhooks(req)

if res is not None:
    # handle response
    pass

```
<!-- End SDK Example Usage [usage] -->

<!-- Start Available Resources and Operations [operations] -->
## Available Resources and Operations

### [authentication](docs/sdks/authentication/README.md)

* [login](docs/sdks/authentication/README.md#login) - Authenticate with the API by providing a username and password.

### [drinks](docs/sdks/drinks/README.md)

* [get_drink](docs/sdks/drinks/README.md#get_drink) - Get a drink.
* [list_drinks](docs/sdks/drinks/README.md#list_drinks) - Get a list of drinks.

### [ingredients](docs/sdks/ingredients/README.md)

* [list_ingredients](docs/sdks/ingredients/README.md#list_ingredients) - Get a list of ingredients.

### [orders](docs/sdks/orders/README.md)

* [create_order](docs/sdks/orders/README.md#create_order) - Create an order.

### [config](docs/sdks/config/README.md)

* [subscribe_to_webhooks](docs/sdks/config/README.md#subscribe_to_webhooks) - Subscribe to webhooks.
<!-- End Available Resources and Operations [operations] -->



<!-- Start Retries [retries] -->
## Retries

Some of the endpoints in this SDK support retries. If you use the SDK without any configuration, it will fall back to the default retry strategy provided by the API. However, the default retry strategy can be overridden on a per-operation basis, or across the entire SDK.

To change the default retry strategy for a single API call, simply provide a `RetryConfig` object to the call:
```python
import speakeasybar
from speakeasybar.models import operations, shared
from speakeasybar.utils import BackoffStrategy, RetryConfig

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)

req = [
    operations.RequestBody(),
]

res = s.config.subscribe_to_webhooks(req,
    RetryConfig('backoff', BackoffStrategy(1, 50, 1.1, 100), False))

if res is not None:
    # handle response
    pass

```

If you'd like to override the default retry strategy for all operations that support retries, you can use the `retry_config` optional parameter when initializing the SDK:
```python
import speakeasybar
from speakeasybar.models import operations, shared
from speakeasybar.utils import BackoffStrategy, RetryConfig

s = speakeasybar.Speakeasybar(
    retry_config=RetryConfig('backoff', BackoffStrategy(1, 50, 1.1, 100), False)
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)

req = [
    operations.RequestBody(),
]

res = s.config.subscribe_to_webhooks(req)

if res is not None:
    # handle response
    pass

```
<!-- End Retries [retries] -->



<!-- Start Error Handling [errors] -->
## Error Handling

Handling errors in this SDK should largely match your expectations.  All operations return a response object or raise an error.  If Error objects are specified in your OpenAPI Spec, the SDK will raise the appropriate Error type.

| Error Object      | Status Code       | Content Type      |
| ----------------- | ----------------- | ----------------- |
| errors.BadRequest | 400               | application/json  |
| errors.APIError   | 5XX               | application/json  |
| errors.SDKError   | 4xx-5xx           | */*               |

### Example

```python
import speakeasybar
from speakeasybar.models import errors, operations, shared

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)

req = [
    operations.RequestBody(),
]

res = None
try:
    res = s.config.subscribe_to_webhooks(req)
except errors.BadRequest as e:
    # handle exception
    raise(e)
except errors.APIError as e:
    # handle exception
    raise(e)
except errors.SDKError as e:
    # handle exception
    raise(e)

if res is not None:
    # handle response
    pass

```
<!-- End Error Handling [errors] -->



<!-- Start Server Selection [server] -->
## Server Selection

### Select Server by Name

You can override the default server globally by passing a server name to the `server: str` optional parameter when initializing the SDK client instance. The selected server will then be used as the default on the operations that use it. This table lists the names associated with the available servers:

| Name | Server | Variables |
| ----- | ------ | --------- |
| `prod` | `https://speakeasy.bar` | None |
| `staging` | `https://staging.speakeasy.bar` | None |
| `customer` | `https://{organization}.{environment}.speakeasy.bar` | `environment` (default is `prod`), `organization` (default is `api`) |

#### Example

```python
import speakeasybar
from speakeasybar.models import shared

s = speakeasybar.Speakeasybar(
    server="customer",
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)


res = s.ingredients.list_ingredients(ingredients=[
    '<value>',
])

if res.classes is not None:
    # handle response
    pass

```

#### Variables

Some of the server options above contain variables. If you want to set the values of those variables, the following optional parameters are available when initializing the SDK client instance:
 * `environment: models.ServerEnvironment`
 * `organization: str`

### Override Server URL Per-Client

The default server can also be overridden globally by passing a URL to the `server_url: str` optional parameter when initializing the SDK client instance. For example:
```python
import speakeasybar
from speakeasybar.models import shared

s = speakeasybar.Speakeasybar(
    server_url="https://speakeasy.bar",
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)


res = s.ingredients.list_ingredients(ingredients=[
    '<value>',
])

if res.classes is not None:
    # handle response
    pass

```

### Override Server URL Per-Operation

The server URL can also be overridden on a per-operation basis, provided a server list was specified for the operation. For example:
```python
import speakeasybar
from speakeasybar.models import shared

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)


res = s.drinks.list_drinks(server_url="https://speakeasy.bar", drink_type=shared.DrinkType.SPIRIT)

if res.classes is not None:
    # handle response
    pass

```
<!-- End Server Selection [server] -->



<!-- Start Custom HTTP Client [http-client] -->
## Custom HTTP Client

The Python SDK makes API calls using the [requests](https://pypi.org/project/requests/) HTTP library.  In order to provide a convenient way to configure timeouts, cookies, proxies, custom headers, and other low-level configuration, you can initialize the SDK client with a custom `requests.Session` object.

For example, you could specify a header for every request that this sdk makes as follows:
```python
import speakeasybar
import requests

http_client = requests.Session()
http_client.headers.update({'x-custom-header': 'someValue'})
s = speakeasybar.Speakeasybar(client=http_client)
```
<!-- End Custom HTTP Client [http-client] -->



<!-- Start Authentication [security] -->
## Authentication

### Per-Client Security Schemes

This SDK supports the following security schemes globally:

| Name          | Type          | Scheme        |
| ------------- | ------------- | ------------- |
| `api_key`     | apiKey        | API key       |
| `bearer_auth` | http          | HTTP Bearer   |

You can set the security parameters through the `security` optional parameter when initializing the SDK client instance. The selected scheme will be used by default to authenticate with the API for all operations that support it. For example:
```python
import speakeasybar
from speakeasybar.models import shared

s = speakeasybar.Speakeasybar(
    security=shared.Security(
        api_key="<YOUR_API_KEY>",
    ),
)


res = s.ingredients.list_ingredients(ingredients=[
    '<value>',
])

if res.classes is not None:
    # handle response
    pass

```

### Per-Operation Security Schemes

Some operations in this SDK require the security scheme to be specified at the request level. For example:
```python
import speakeasybar
from speakeasybar.models import operations

s = speakeasybar.Speakeasybar()

req = operations.LoginRequestBody(
    type=operations.Type.API_KEY,
)

res = s.authentication.login(req, operations.LoginSecurity(
    password="<PASSWORD>",
    username="<USERNAME>",
))

if res.object is not None:
    # handle response
    pass

```
<!-- End Authentication [security] -->

<!-- Placeholder for Future Speakeasy SDK Sections -->



### Maturity

This SDK is in beta, and there may be breaking changes between versions without a major version update. Therefore, we recommend pinning usage
to a specific package version. This way, you can install the same version each time without breaking changes unless you are intentionally
looking for the latest version.

### Contributions

While we value open-source contributions to this SDK, this library is generated programmatically.
Feel free to open a PR or a Github issue as a proof of concept and we'll do our best to include it in a future release !

### SDK Created by [Speakeasy](https://docs.speakeasyapi.dev/docs/using-speakeasy/client-sdks)
 readmeEtag: '"ca4fcb630c438231d9619c6bb544e5b77c0978e2"' readmeLastModified: Mon, 08 Apr 2024 15:05:30 GMT repositoryId: 604771680 description: null created: '2023-02-21T18:56:21Z' updated: '2025-09-10T16:05:14Z' language: TypeScript archived: false stars: 3 watchers: 2 forks: 0 owner: speakeasy-sdks logo: https://avatars.githubusercontent.com/u/121978464?v=4 license: MIT repoEtag: '"c72dd931e0d164462896fa214e4c899841aa0b9077ee782cab5be5c6153888b4"' repoLastModified: Wed, 10 Sep 2025 16:05:14 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/enorganic/oapi v3: true id: 7c7acc9e926f31e4935a1e03abcc82ec repositoryMetadata: base64Readme: >- IyBvYXBpCgpbIVt0ZXN0XShodHRwczovL2dpdGh1Yi5jb20vZW5vcmdhbmljL29hcGkvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC55bWwvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluKV0oaHR0cHM6Ly9naXRodWIuY29tL2Vub3JnYW5pYy9vYXBpL2FjdGlvbnMvd29ya2Zsb3dzL3Rlc3QueW1sKQpbIVtQeVBJIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9weS9vYXBpLnN2Zz9pY29uPXNpJTNBcHl0aG9uKV0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL3B5L29hcGkpCgpgb2FwaWAgaXMgYSBsaWJyYXJ5IGZvciBhdXRob3JpbmcgcHl0aG9uIGNsaWVudCBsaWJyYXJpZXMgZm9yIHdlYiBBUElzCmJhc2VkIG9uIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiAoYm90aApPcGVuQVBJL1N3YWdnZXIgW3ZlcnNpb24gMgpdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8yLjAubWQpCmFuZCBbdmVyc2lvbiAzCl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYWluL3ZlcnNpb25zLzMuMS4xLm1kKQphcmUgc3VwcG9ydGVkKS4KCmBvYXBpYC1nZW5lcmF0ZWQgcGFja2FnZXMvbW9kdWxlcyBkaWZmZXIgZnJvbSB0aG9zZSBnZW5lcmF0ZWQgYnkgb3RoZXIgdG9vbHMKKGluY2x1ZGluZyBidXQgbm90IGxpbWl0ZWQgdG8KW3N3YWdnZXItY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29kZWdlbikpIGluIHRoYXQKYG9hcGlgIGdlbmVyYXRlcyBhIGRhdGEgbW9kZWwgd2hpY2ggZW5mb3JjZXMgdGhlIHNjaGVtYXMgZGVmaW5lZAppbiB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uLiBUaGlzIG1lYW5zIHRoYXQgcmVxdWVzdCBhbmQgcmVzcG9uc2Ugb2JqZWN0cywKYW5kIHRoZWlyIHByb3BlcnRpZXMvaXRlbXMvbWVtYmVycywgY2Fubm90IGNvbXByaXNlIGRhdGEgdHlwZXMgd2hpY2ggZGlmZmVyCmZyb20gdGhvc2Ugc3BlY2lmaWVkIGluIHRoZWlyIE9wZW5BUEkgZG9jdW1lbnQuIEFkZGl0aW9uYWxseSwgcmVxdWVzdCBhbmQKcmVzcG9uc2Ugb2JqZWN0cyBjYW4gYmUgdmFsaWRhdGVkIHVzaW5nIGBzb2IudmFsaWRhdGVgIHRvIHZlcmlmeSB0aGF0IHRoZXkKd2VyZSBub3QgcGFyc2VkIGZyb20gYW4gb2JqZWN0IGhhdmluZyBhdHRyaWJ1dGVzICpub3QqIGRlZmluZWQgaW4gdGhlCmVsZW1lbnQncyBKU09OIHNjaGVtYSwgYW5kICpoYXZlKiBhbGwgcmVxdWlyZWQgYXR0cmlidXRlcy4gYG9hcGlgCmZ1bGx5IHN1cHBvcnRzIHBvbHltb3JwaGlzbSB3aGVuIGEgc2NoZW1hIGhhcyBhbiAiYW55T2YiLCAiYWxsT2YiLCBvcgoib25lT2YiIGF0dHJpYnV0ZSAoYWx0aG91Z2ggYG9hcGlgIGRvZXMgbm90ICpyZXF1aXJlKiBhIFtkaXNjcmltaW5hdG9yCl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYWluL3ZlcnNpb25zLzMuMS4xLm1kI2Rpc2NyaW1pbmF0b3Itb2JqZWN0KQp0byBiZSBkZWZpbmVkKS4KCk5vdGFibGUgZmVhdHVyZXMgb2YgYW4gYG9hcGlgIGdlbmVyYXRlZCBjbGllbnQvbW9kZWwgbGlicmFyeSBpbmNsdWRlOgoKLSBgb2FwaWAgZ2VuZXJhdGVkIGRhdGEgbW9kZWxzIGFuZCBjbGllbnRzIGFyZSBmdWxseSB0eXBlLWFubm90YXRlZAotIGBvYXBpYCBnZW5lcmF0ZWQgZGF0YSBtb2RlbHMgYW5kIGNsaWVudHMgYWRoZXJlIHRvIFBFUC04IGZvcm1hdHRpbmcKLSBgb2FwaWAgZ2VuZXJhdGVkIGRhdGEgbW9kZWxzIGFuZCBjbGllbnRzIGluY2x1ZGUgZG9jc3RyaW5ncyBmb3IgYWxsIG1vZGVscywKICBtb2RlbCBwcm9wZXJ0aWVzLCBjbGllbnQgbWV0aG9kcywgYW5kIGNsaWVudCBtZXRob2QgcGFyYW1ldGVycywgaW5mZXJyZWQKICBmcm9tICJkZXNjcmlwdGlvbiIgYW5kICJzdW1tYXJ5IiBhdHRyaWJ1dGVzIG9mIGVsZW1lbnRzIHdpdGhpbiB0aGUKICBzcGVjaWZpY2F0aW9uCi0gYG9hcGlgIGdlbmVyYXRlZCBjbGllbnRzIHdpbGwgaGFuZGxlIHRoZSBtb3N0IGNvbW1vbiBzZXJ2ZXItdG8tc2VydmVyCiAgYXV0aGVudGljYXRpb24gbWV0aG9kcyBvdXQtb2YtYm94LCBpbmNsdWRpbmcgdGhlIE9BdXRoMiBhdXRob3JpemF0aW9uCiAgY29kZSBmbG93IChjbGllbnRfaWQvY2xpZW50X3NlY3JldCksIEFQSSBrZXlzLCBiZWFyZXIgdG9rZW5zLCBIVFRQIGJhc2ljCiAgYXV0aCwgYW5kIG1vc3QgY29va2llLWJhc2VkIGF1dGhlbnRpY2F0aW9uIHBhdHRlcm5zLiBDdXN0b20gYXV0aGVudGljYXRpb24KICBtZXRob2RzIGFyZSBhbHNvIGVhc2lseSBhY2NvbXBsaXNoZWQgYnkgc3BlY2lmeWluZyBjdXN0b20gYmFzZSBjbGFzc2VzCiAgYW5kL29yIHBhc3NpbmcgaW5pdGlhbGl6YXRpb24gZGVjb3JhdG9ycyB0byBgb2FwaS53cml0ZV9jbGllbnRfbW9kdWxlYC4KLSBgb2FwaWAgZ2VuZXJhdGVkIGNsaWVudHMgYW5kIG1vZGVscyBhcmUgZWFzaWx5IHVwZGF0ZWQuIEZvciBpbnN0YW5jZTogYG9hcGlgCiAgZ2VuZXJhdGVkIGRhdGEgbW9kZWwgY2xhc3NlcyBjYW4gYmUgcmVuYW1lZCB1c2luZyB5b3VyIGZhdm9yaXRlIElERSdzCiAgcmVmYWN0b3JpbmcgdG9vbHMsIGFuZCBzdWJzZXF1ZW50IHVwZGF0ZXMgdG8gdGhhdCBkYXRhIG1vZGVsIChzdWNoIGFzCiAgdG8gcmVmbGVjdCBtaW5vci12ZXJzaW9uIHVwZGF0ZXMgdG8gdGhlIHNwZWMgd2hpY2ggYWRkIGVuZHBvaW50cywgc2NoZW1hcywKICBwYXJhbWV0ZXJzLCBldGMuKSB3aWxsIHJldGFpbiB5b3VyIG5ldyBjbGFzcyBuYW1lcywgc28gbG9uZyBhcyB0aGUKICBzY2hlbWEgZnJvbSB3aGljaCB0aGUgY2xhc3Mgd2FzIGdlbmVyYXRlZCByZW1haW5zIGluIHRoZSBzYW1lIGxvY2F0aW9uCiAgd2l0aGluIHRoZSBzcGVjLiBDbGFzcyBuYW1pbmcgY2FuIGFsc28gYmUgY3VzdG9taXplZCBieSBwYXNzaW5nIGEgY3VzdG9tCiAgaG9vayB0byB0aGUgKmdldF9jbGFzc19uYW1lX2Zyb21fcG9pbnRlciogcGFyYW1ldGVyIGZvcgogIGBvYXBpLndyaXRlX21vZGVsX21vZHVsZWAu readmeEtag: '"e35ffe4f86c1a6ec69681993e2c08957248950a6"' readmeLastModified: Thu, 24 Apr 2025 23:58:53 GMT repositoryId: 190461316 description: A tool for generating client libraries from an OpenAPI document. created: '2019-06-05T20:09:55Z' updated: '2026-01-22T19:26:24Z' language: Python archived: false stars: 2 watchers: 1 forks: 2 owner: enorganic logo: https://avatars.githubusercontent.com/u/91716117?v=4 license: MIT repoEtag: '"0b6be582eb512a34fd8ba94881143b52f0064fb64f4a2e6d225d8d4648713e4f"' repoLastModified: Thu, 22 Jan 2026 19:26:24 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/ankumar/programming-machine-learning v3: true id: 6221a7dad056965aa69ad16ff0132577 repositoryMetadata: base64Readme: >- # Platform Space

1. [**The Shift from Models to Compound AI Systems**](https://bair.berkeley.edu/blog/2024/02/18/compound-ai-systems/) - This article from Berkeley AI Research (BAIR) highlights a growing trend where AI advancements increasingly rely on **compound AI systems**—combinations of multiple models and components—rather than traditional monolithic models. Compound systems offer more flexibility and adaptability, as each component can specialize in a different task or phase of the pipeline.  

Key points:
- **Design, optimization, and operation:** Approaches are still emerging, but compound AI systems are proving more efficient for complex tasks.
- **Maximizing reliability and quality:** These systems promise higher reliability, particularly for large-scale applications, by breaking down tasks into smaller, more manageable units. 
- **Trend for 2024:** BAIR sees this as one of the most important trends, where developers will focus on how to assemble these components in effective ways.
   
![image](https://github.com/user-attachments/assets/d874e82f-9b89-487e-894c-1ed3b247294d)
<p align="center">Increasingly many new AI results are from compound systems.</p>

"_Figuring out the best practices for developing compound AI systems is still an open question, but there are already exciting approaches to aid with design, end-to-end optimization, and operation. We believe that compound AI systems will remain the best way to maximize the quality and reliability of AI applications going forward, and may be one of the most important trends in AI in 2024._"

2. [**Building A Generative AI Platform**](https://huyenchip.com/2024/07/25/genai-platform.html) - This article explores the architecture and development of a generative AI platform, focusing on the challenges of integrating various AI components. It emphasizes:
- **Open Source LLM Tools:** The article references several tools and techniques for building generative AI, especially focusing on model alignment, optimization, and adaptability. The referenced [Open Source LLM Tools](https://huyenchip.com/llama-police) are crucial for understanding the ecosystem surrounding generative AI models like Meta’s LLaMA.
- **Challenges in platform building:** From model integration to ensuring the alignment of generative AI output with intended use cases, there are many hurdles in achieving an operational platform.

Both articles stress that the future of AI will involve **more sophisticated systems**, relying on the cooperation of various specialized models (compound AI) and the integration of diverse tools for generative AI platform development. This aligns with ongoing exploration of **compound AI systems** and **generative AI platform architecture**.
   
## APIs
Started with this one endpoint Completions **/completions** with Text Input/Output (now Legacy)  
  
Then ChatGPT, Chat completions **/chat/completions**  with Messages Input/Output
  
3. Embeddings			**/embeddings**
4. Image generation		**/images/generations**
5. Text to speech			**/audio/speech**
6. Speech to text			**/audio/transcriptions**
7. Moderation			**/moderations**
8. Fine-tuning  **/fine_tuning/jobs**
9. Batch				**/files /batches**
  
REST APIs:  
- Full OpenAPI specification for the [OpenAI API](https://github.com/openai/openai-openapi/blob/master/openapi.yaml)    
- Response Formats - [Building AGI with OpenAI's Structured Outputs API](https://www.youtube.com/watch?v=NjOfH9D8aJo)
- Function Calling - Query Database, Send Alerts etc.
  - [Azure OpenAI API](https://learn.microsoft.com/en-us/azure/ai-services/openai/reference#api-specs)
  - [Bedrock API](https://docs.aws.amazon.com/bedrock/latest/APIReference/API_runtime_InvokeModel.html)
  - [Llama Stack API](https://github.com/meta-llama/llama-stack/blob/main/docs/resources/llama-stack-spec.yaml)
  - Gemini API - ?

Python SDKs:  
- OpenAI Python [API](https://github.com/openai/openai-python/blob/main/api.md) library in the The official Python library for the [OpenAI API](https://github.com/openai/openai-python)

```python  
import os
from openai import OpenAI

client = OpenAI(
    # This is the default and can be omitted
    api_key=os.environ.get("OPENAI_API_KEY"),
)

chat_completion = client.chat.completions.create(
    messages=[
        {
            "role": "user",
            "content": "Say this is a test",
        }
    ],
    model="gpt-3.5-turbo",
)

print(chat_completion.to_json())
# print(chat_completion.choices[0].message.content)

from openai import AzureOpenAI

# gets the API Key from environment variable AZURE_OPENAI_API_KEY
client = AzureOpenAI(
    # https://learn.microsoft.com/azure/ai-services/openai/reference#rest-api-versioning
    api_version="2023-07-01-preview",
    # https://learn.microsoft.com/azure/cognitive-services/openai/how-to/create-resource?pivots=web-portal#create-a-resource
    azure_endpoint="https://example-endpoint.openai.azure.com",
    # azure_deployment="gpt35",
)

chat_completion = client.chat.completions.create(
    # model="deployment-name",  # e.g. gpt-35-instant
    model="gpt35",  # gpt-35-turbo
    messages=[
        {
            "role": "user",
            "content": "How do I output all files in a directory using Python?",
        },
    ],
)
print(chat_completion.to_json())
```
  
- Anthropic Python    

```python
client = Anthropic()  
client = AnthropicBedrock()  
client = AnthropicVertex()  

message = client.messages.create()  
print(message.content)
```

**More APIs:**
  
- [Drop in replacement](https://github.com/datastax/astra-assistants-api) for the OpenAI Assistants API
  - Full coverage of OpenAI endpoints in the repo [here](https://github.com/datastax/astra-assistants-api/blob/main/coverage.md)
 
- [Llama Stack RFC](https://github.com/meta-llama/llama-stack/blob/main/rfcs/RFC-0001-llama-stack.md)
  - https://github.com/meta-llama/llama-stack-apps
  - https://github.com/meta-llama/llama-stack
 
- [Web APIs](https://developer.chrome.com/docs/ai/built-in)
  - https://x.com/addyosmani/status/1843328653155062108  

**More Languages:**
- The official [Go](https://github.com/openai/openai-go) library for the OpenAI API
    
## Secrets

### 1. API Keys
An API key is a simple string (often alphanumeric) used to authenticate requests. It can be included as:
- **URL Parameter**:  
  `https://example.com/api/resource?api_key=YOUR_API_KEY`
- **Header**:  
  `Authorization: ApiKey YOUR_API_KEY`

API keys are typically used for simple authentication and are suited for server-to-server communication but are less secure if exposed in URLs.

---

### 2. Bearer Token
A **Bearer Token** is a security token that is issued as part of OAuth 2.0. This token grants the bearer access to resources. It's usually passed in the request header:
- **Header**:  
  `Authorization: Bearer YOUR_TOKEN`

Bearer tokens offer more security compared to API keys, especially when combined with token expiration and refresh mechanisms.

---

### 3. Microsoft Entra ID (formerly Azure AD)
Entra ID provides **OAuth 2.0** or **OpenID Connect (OIDC)** based authentication and authorization, mostly used for securing enterprise apps. The flow typically involves:
- **Access Token**: Obtained after a user or service authenticates with Entra ID.
- **Header**:  
  `Authorization: Bearer YOUR_ACCESS_TOKEN`

Entra ID is often used in conjunction with Microsoft services or enterprise environments for user-based or service-based authentication.

---

### 4. AWS Signature
**AWS Signature Version 4** is used to securely sign API requests to AWS services. This method calculates a signature based on the request parameters, headers, and the user's secret access key. The signature is added to the request as:
- **Authorization Header**:  
  `Authorization: AWS4-HMAC-SHA256 Credential=ACCESS_KEY/..., SignedHeaders=..., Signature=SIGNATURE`

It is typically more secure because the signature is derived dynamically and is time-limited.


## Guardrails
- Sensitive information filters - PII types, Regex patterns etc.
- Content filters - Configure content filters to detect & block harmful user inputs and model responses
- Denied topics
- Word filters
- Contextual grounding check

## Assistants
- Tune Personality & Capabilities
- Call Models
- Access Tools in parallel
    - Built-in code_interpreter, file_search etc.
    - Function Calling
- Persistent Threads
- File Formats

## Agents
- “Agent is a more overloaded term at this point than node, service, and instance.”
  
-> https://x.com/rakyll/status/1837164761362133057

- “I'm wondering what would be the base requirements of "true agent" (i.e. not just over-hyped marketing). 
For me: 
Can use APIs reliably. 
APIs by other companies, not just ones specifically written for the agent.
The API usage should cover a large subset of the services that the agent is aiming to cover. 

I.e. if your agent is supposed to order food, it should be able to find an open restaurant with take away, figure out how to do the delivery and at least support the 3 large delivery companies.”

-> https://x.com/gwenshap/status/1837167653338681819

## Telemetry
- [OpenTelemetry](https://opentelemetry.io/)
  - [Semantic Conventions for Generative AI systems | OpenTelemetry](https://opentelemetry.io/docs/specs/semconv/gen-ai/)
    - [Introduce semantic conventions for modern AI (LLMs, vector databases, etc.)](https://github.com/open-telemetry/semantic-conventions/issues/327)

# Model Spaces...

![image](https://github.com/user-attachments/assets/6830c307-62ad-4255-ab3a-5d7037176e2b)
<p align="center">Source: https://github.com/rasbt/LLMs-from-scratch</p>


**Foundation Models:** Emphasize the creation and application of large-scale models that can be adapted to a wide range of tasks with minimal task-specific tuning.  

**Predictive Human Preference (PHP):** Leveraging human feedback in the loop of model training to refine outputs or predictions based on what is preferred or desired by humans.   

- Predictive Human Preference - [php](php)

**Fine Tuning:** The process of training an existing pre-trained model on a specific task or dataset to improve its performance on that task.  

- https://llama.meta.com/docs/how-to-guides/fine-tuning
- https://github.com/hiyouga/LLaMA-Factory

**Cross-cutting Themes:**

"Our results show conditioning away risk of attack remains an unsolved problem; for example, all tested models showed between 25% and 50% successful prompt injection tests."

https://ai.meta.com/research/publications/cyberseceval-2-a-wide-ranging-cybersecurity-evaluation-suite-for-large-language-models/

**Personal Identifiable Information (PII) and Security:** These considerations are crucial for ensuring that ML models respect privacy and are secure against potential threats.  

- Personal Identifiable Information - [pii](pii)  

**Code, SQL, Genomics, and More:** These areas highlight the interdisciplinary nature of ML, where knowledge in programming, databases, biology, and other fields converge to advance ML applications.  

**Neural Architecture Search (NAS):** Highlights the automation of the design of neural network architectures to optimize performance for specific tasks.  

- Biology (Collab w/ [Ashish Phal](https://www.linkedin.com/in/ashish-phal-548b37125/)) - [genomics](docs/genomics.md)  

**Few-Shot and Zero-Shot Learning:** Points to learning paradigms that aim to reduce the dependency on large labeled datasets for training models.  

**Federated Learning:** Focuses on privacy-preserving techniques that enable model training across multiple decentralized devices or servers holding local data samples.  
 
**Transformers in Vision and Beyond:** Discusses the application of transformer models, originally designed for NLP tasks, in other domains like vision and audio processing.  

**Reinforcement Learning Enhancements:** Looks at advancements in RL techniques that improve efficiency and applicability in various decision-making contexts.   

**MLOps and AutoML:** Concentrates on the operationalization of ML models and the automation of the ML pipeline to streamline development and deployment processes.  

**Hybrid Models:** Explores the integration of different model types or AI approaches to leverage their respective strengths in solving complex problems.  

**AI Ethics and Bias Mitigation:** Underlines the importance of developing fair and ethical AI systems by addressing and mitigating biases in ML models.  

**Energy-Efficient ML:** Reflects the growing concern and need for environmentally sustainable AI by developing models that require less computational power and energy. 

**Hardware:** Points to the importance of developing and utilizing hardware optimized for ML tasks to improve efficiency and performance.  










 readmeEtag: '"8c6724ee65b223d9763f693d899b9519517d845e"' readmeLastModified: Fri, 24 Jan 2025 19:48:07 GMT repositoryId: 747875001 description: >- The landscape of machine learning (ML) is constantly evolving with new techniques, tools, and frameworks emerging at a rapid pace. created: '2024-01-24T20:13:28Z' updated: '2025-10-05T20:56:11Z' language: Python archived: false stars: 3 watchers: 1 forks: 0 owner: ankumar logo: https://avatars.githubusercontent.com/u/658791?v=4 repoEtag: '"1ee5397f810a9fc91914ec55c6ac39bc5944ffb834539610cc9a54f31368285a"' repoLastModified: Sun, 05 Oct 2025 20:56:11 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/ivorisoutdoors/openapi-stitcher v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFN0aXRjaGVyCgpTdGl0Y2ggdG9nZXRoZXIgT3BlbkFQSSBmaWxlcyBpbnRvIGEgc2luZ2xlIGZpbGUuCgojIyBSZXF1aXJlbWVudHMKCiogTm9kZUpTICg+PSAxMCkKCiMjIFVzYWdlCgpTZWUgaGVscCBmb3IgYWxsIGNvbW1hbmRzIGFuZCBvcHRpb25zCgogICAgbnB4IG9wZW5hcGktc3RpdGNoZXIgLS1oZWxwCiAgICBucHggb3BlbmFwaS1zdGl0Y2hlciBidWlsZCAtLWhlbHAKClN0aXRjaCB0b2dldGhlciBmaWxlcyBtYXRjaGluZyBhIHBhdHRlcm4uCgogICAgbnB4IG9wZW5hcGktc3RpdGNoZXIgYnVpbGQgInNwZWMvKiovKi57eWFtbCx5bWx9IiBvcGVuYXBpLnlhbWwKCldhdGNoIGZvciBjaGFuZ2VzIGFuZCByZWJ1aWxkIHRoZSBmaWxlLgoKICAgIG5weCBvcGVuYXBpLXN0aXRjaGVyIGJ1aWxkIC0td2F0Y2ggInNwZWMvKiovKi57eWFtbCx5bWx9IiBvcGVuYXBpLnlhbWwKClNlcnZlIHRoZSBzcGVjaWZpY2F0aW9uIHVzaW5nIFtTd2FnZ2VyIFVJXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkuCgogICAgbnB4IG9wZW5hcGktc3RpdGNoZXIgc2VydmUgInNwZWMvKiovKi57eWFtbCx5bWx9IgoKU2VlIF9leGFtcGxlL3NwZWNfIGZvciBhbiBleGFtcGxlIG9mIGhvdyB0byBvcmdhbml6ZSB5b3VyIGZpbGVzIHRvIGJlc3Qgd29yawp3aXRoIHRoaXMgdG9vbC4K readmeEtag: '"21be57417b8b9fb00012f03b02cf339d6bdd5fc9"' readmeLastModified: Sun, 01 Mar 2020 04:39:23 GMT repositoryId: 243554283 description: Stitch together multiple files into an OpenAPI specification created: '2020-02-27T15:51:06Z' updated: '2025-11-18T15:24:39Z' language: JavaScript archived: true stars: 2 watchers: 1 forks: 0 owner: ivorisoutdoors logo: https://avatars.githubusercontent.com/u/159041?v=4 license: MIT repoEtag: '"e9acccf1ed07380cd1467242fe6bc000003119260ecf3d0ea2c681fae847007a"' repoLastModified: Tue, 18 Nov 2025 15:24:39 GMT foundInMaster: true category: Parsers id: 861d2419f3913d6a2f851e48d21f562b oldLocations: - https://github.com/mloberg/openapi-stitcher - source: openapi3 tags repository: https://github.com/theogravity/apifire-server v3: true id: b8bff8d1642e3e61b6a9846c60f9ee00 repositoryMetadata: base64Readme: >- IyBnZW5lcmF0b3ItYXBpZmlyZS1zZXJ2ZXIgCgpbIVtOUE0gdmVyc2lvbl0oaHR0cDovL2ltZy5zaGllbGRzLmlvL25wbS92L2dlbmVyYXRvci1hcGlmaXJlLXNlcnZlci5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9nZW5lcmF0b3ItYXBpZmlyZS1zZXJ2ZXIpCiFbYnVpbHQgd2l0aCB0eXBlc2NyaXB0XShodHRwczovL2NhbW8uZ2l0aHVidXNlcmNvbnRlbnQuY29tLzkyZTlmN2IxMjA5YmFiOWUzZTljZDhjZGY2MmYwNzJhNjI0ZGE0NjEvNjg3NDc0NzA3MzNhMmYyZjY2NmM2MTc0MmU2MjYxNjQ2NzY1NmUyZTZlNjU3NDJmNjI2MTY0Njc2NTJmNDI3NTY5NmM3NDI1MzIzMDU3Njk3NDY4MmY1NDc5NzA2NTUzNjM3MjY5NzA3NDJmNjI2Yzc1NjUpIAoKCj4gR2VuZXJhdGVzIGFuIGV4cHJlc3MgQVBJIHNlcnZlciB3cml0dGVuIGluIFR5cGVzY3JpcHQgd2l0aCByb3V0ZXMvY29udHJvbGxlcnMvdmFsaWRhdGlvbi9tb2RlbC9taWdyYXRpb24gZ2VuZXJhdGlvbiBjYXBhYmlsaXRpZXMuIAo+IFVzZSBhbiBPcGVuQVBJIDMgeWFtbCBmaWxlIHRvIGdlbmVyYXRlIGNvbnRyb2xsZXJzIC8gcm91dGVycyAvIHZhbGlkYXRvcnMKPiBVc2UgYSB5YW1sIHRvIGdlbmVyYXRlIG9iamVjdGlvbi5qcyBtb2RlbHMgYW5kIGtuZXggbWlncmF0aW9ucwoKVXNlcyB0aGUgZm9sbG93aW5nIGxpYnJhcmllczoKCi0gW2Bjb25maWd1cml0eWBdKGh0dHBzOi8vZ2l0aHViLmNvbS90aGVvZ3Jhdml0eS9jb25maWd1cml0eSk6IEhhbmRsZXMgY29uZmlndXJhdGlvbgotIFtgb2JqZWN0aW9uLWdlbmVyYXRvcmBdKGh0dHBzOi8vZ2l0aHViLmNvbS90aGVvZ3Jhdml0eS9vYmplY3Rpb24tZ2VuZXJhdG9yKTogR2VuZXJhdGVzIFR5cGVzY3JpcHQgb2JqZWN0aW9uLmpzIG1vZGVscyBhbmQga25leCBtaWdyYXRpb25zCi0gW2BuZXctZXJyb3JgXShodHRwczovL2dpdGh1Yi5jb20vdGhlb2dyYXZpdHkvbmV3LWVycm9yKTogVXNlZCB0byBidWlsZCBhbiBlcnJvci1oYW5kbGluZyBpbmZyYXN0cnVjdHVyZQotIFtgYXBpZmlyZWBdKGh0dHBzOi8vZ2l0aHViLmNvbS90aGVvZ3Jhdml0eS9hcGlmaXJlKTogR2VuZXJhdGVzIFR5cGVzY3JpcHQgY29udHJvbGxlcnMvcm91dGVycy92YWxpZGF0b3JzIGZyb20gYW4gT3BlbkFQSSAzIHNwZWMuCi0gW2BhanZgXShodHRwczovL2Fqdi5qcy5vcmcvKTogVmFsaWRhdGVzIEFQSSByZXF1ZXN0IHBhcmFtZXRlcnMgaW4gcm91dGVycwotIFtgbG9nbGF5ZXJgXShodHRwczovL2dpdGh1Yi5jb20vdGhlb2dyYXZpdHkvbG9nbGF5ZXIpOiBTdHJ1Y3R1cmVkIGxvZ2dpbmcKClNlZSB0aGUgW2BhcGlmaXJlYF0oaHR0cHM6Ly9naXRodWIuY29tL3RoZW9ncmF2aXR5L2FwaWZpcmUpIGZvciBub3RlcyBvbiBjdXJyZW50IGxpbWl0YXRpb25zIHdpdGggdXNpbmcgYW4gT3BlbkFQSSAzIHNwZWMuCgpUaGUgZ2VuZXJhdGVkIHByb2plY3QgY29tZXMgd2l0aCBhIHNhbXBsZSBPcGVuQVBJIGFuZCBgb2JqZWN0aW9uLWdlbmVyYXRvcmAgc3BlYyB0byBxdWlja2x5IGdldCBhIHdvcmtpbmcgc2VydmVyIHVwIGFuZCBydW5uaW5nLgoKIyMgSW5zdGFsbGF0aW9uCgpGaXJzdCwgaW5zdGFsbCBbWWVvbWFuXShodHRwOi8veWVvbWFuLmlvKSBhbmQgZ2VuZXJhdG9yLWFwaS1zZXJ2ZXIgdXNpbmcgW25wbV0oaHR0cHM6Ly93d3cubnBtanMuY29tLykgKHdlIGFzc3VtZSB5b3UgaGF2ZSBwcmUtaW5zdGFsbGVkIFtub2RlLmpzXShodHRwczovL25vZGVqcy5vcmcvKSkuCgpgYGBiYXNoCm5wbSBpbnN0YWxsIC1nIHlvCm5wbSBpbnN0YWxsIC1nIGdlbmVyYXRvci1hcGlmaXJlLXNlcnZlcgpgYGAKClRoZW4gZ2VuZXJhdGUgeW91ciBuZXcgcHJvamVjdDoKCmBgYGJhc2gKeW8gYXBpZmlyZS1zZXJ2ZXIKYGBgCgojIyBHZXR0aW5nIFRvIEtub3cgWWVvbWFuCgogKiBZZW9tYW4gaGFzIGEgaGVhcnQgb2YgZ29sZC4KICogWWVvbWFuIGlzIGEgcGVyc29uIHdpdGggZmVlbGluZ3MgYW5kIG9waW5pb25zLCBidXQgaXMgdmVyeSBlYXN5IHRvIHdvcmsgd2l0aC4KICogWWVvbWFuIGNhbiBiZSB0b28gb3BpbmlvbmF0ZWQgYXQgdGltZXMgYnV0IGlzIGVhc2lseSBjb252aW5jZWQgbm90IHRvIGJlLgogKiBGZWVsIGZyZWUgdG8gW2xlYXJuIG1vcmUgYWJvdXQgWWVvbWFuXShodHRwOi8veWVvbWFuLmlvLykuCgojIyBMaWNlbnNlCgpNSVQgwqkgW1RoZW8gR3Jhdml0eV0oaHR0cHM6Ly9naXRodWIuY29tL3RoZW9ncmF2aXR5KQoKCltucG0taW1hZ2VdOiBodHRwczovL2JhZGdlLmZ1cnkuaW8vanMvZ2VuZXJhdG9yLXRzLW9wZW5hcGktc2VydmVyLnN2ZwpbbnBtLXVybF06IGh0dHBzOi8vbnBtanMub3JnL3BhY2thZ2UvZ2VuZXJhdG9yLXRzLW9wZW5hcGktc2VydmVyClt0cmF2aXMtaW1hZ2VdOiBodHRwczovL3RyYXZpcy1jaS5jb20vdGhlb2dyYXZpdHkvZ2VuZXJhdG9yLXRzLW9wZW5hcGktc2VydmVyLnN2Zz9icmFuY2g9bWFzdGVyClt0cmF2aXMtdXJsXTogaHR0cHM6Ly90cmF2aXMtY2kuY29tL3RoZW9ncmF2aXR5L2dlbmVyYXRvci10cy1vcGVuYXBpLXNlcnZlcgpbZGF2aWRkbS1pbWFnZV06IGh0dHBzOi8vZGF2aWQtZG0ub3JnL3RoZW9ncmF2aXR5L2dlbmVyYXRvci10cy1vcGVuYXBpLXNlcnZlci5zdmc/dGhlbWU9c2hpZWxkcy5pbwpbZGF2aWRkbS11cmxdOiBodHRwczovL2RhdmlkLWRtLm9yZy90aGVvZ3Jhdml0eS9nZW5lcmF0b3ItdHMtb3BlbmFwaS1zZXJ2ZXIK readmeEtag: '"6dbbe9e38f8d94ce34746f157f5302e5fd4be83c"' readmeLastModified: Thu, 11 May 2023 00:21:15 GMT repositoryId: 264778841 description: >- Generates an express-based API server w/ database migrations and models and an error-handling / logging infrastructure written in Typescript created: '2020-05-17T23:51:29Z' updated: '2022-11-21T19:57:06Z' language: TypeScript archived: false stars: 2 watchers: 2 forks: 0 owner: theogravity logo: https://avatars.githubusercontent.com/u/855434?v=4 license: MIT repoEtag: '"9e225ea2a1a326e707687c4e3044fec282544dc2b4ce4c0c0514d37319ab5cd9"' repoLastModified: Mon, 21 Nov 2022 19:57:06 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/luisramos08/cliente_api v3: true repositoryMetadata: base64Readme: >- CiMjIENsaWVudGUgQXBpCgpSRVNULUFQSSBkZXNhcnJvbGxhZGEgZW4gU3ByaW5nIEJvb3QuCgoKCiMjIFNlIHV0aWxpesOzOgoKLSBTcHJpbmcgQm9vdAotIE1hdmVuCi0gT3BlbkFwaSBVSSAtIFN3YWdnZXIKLSBIMiAoQ29tbyBnZXN0b3IgZGUgYmFzZSBkZSBkYXRvcyBlbiBtZW1vcmlhKQotIExvbWJvayAKCgojIyBQcnVlYmFzCgpZYSBxdWUgc2UgaW5jbHV5w7MgdW5hIGludGVyZmF6IGRlIHBydWViYSBxdWUgc2UgYWdyZWdhIGRlc2RlIFN3YWdnZXIuClVuYSB2ZXogZWwgcHJveWVjdG8gZXN0ZSBjb3JyaWVuZG8gc2UgcHVlZGUgdXRpbGl6YXIgZWwgc2lndWllbnRlIGxpbmsgcGFyYSBoYWNlciBsYXMgcHJ1ZWJhcyBkZSBsdWdhcjoKCi0gaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkvaW5kZXguaHRtbAo= readmeEtag: '"027a507b56dcc5133a6fff6e4234d95ccb1a61db"' readmeLastModified: Sat, 28 May 2022 01:49:11 GMT repositoryId: 484739334 description: API RESTFul en Java Spring Boot created: '2022-04-23T12:20:05Z' updated: '2023-01-17T13:10:58Z' language: Java archived: false stars: 2 watchers: 1 forks: 1 owner: LuisRamos08 logo: https://avatars.githubusercontent.com/u/35547126?v=4 repoEtag: '"4277c832fdae2b0bac5527c14cfbbd3ee2263cf33d32d82d9dd0016b86f7ede7"' repoLastModified: Tue, 17 Jan 2023 13:10:58 GMT foundInMaster: true category: - Editors - SDK id: c88018c0c052f8e5547d375b8b058fb4 - source: openapi3 tags repository: https://github.com/ditschedev/swag-ts v3: true id: cf0757432d430e75140f2b0cf4396f07 repositoryMetadata: base64Readme: >- IyBzd2FnLXRzCgpbIVtdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL2RpdHNjaGVkZXYvc3dhZy10cy90ZXN0LnltbD9icmFuY2g9bWFpbiZsb25nQ2FjaGU9dHJ1ZSZsYWJlbD1UZXN0JmxvZ289Z2l0aHViJTIwYWN0aW9ucyZsb2dvQ29sb3I9ZmZmKV0oaHR0cHM6Ly9naXRodWIuY29tL2RpdHNjaGVkZXYvc3dhZy10cy9hY3Rpb25zP3F1ZXJ5PXdvcmtmbG93JTNBVGVzdCkKWyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL2RpdHNjaGVkZXYvc3dhZy10cyldKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9kaXRzY2hlZGV2L3N3YWctdHMpCgpTaW1wbHkgcHJvdmlkZSBhIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBhbmQgc3dhZy10cyB3aWxsIGdlbmVyYXRlIHR5cGVzY3JpcHQgdHlwZXMgZm9yIHlvdS4gWW91IGNhbiBwcm92aWRlIGpzb24gb3IgeWFtbCBkZWZpbml0aW9ucyBvbiB5b3VyIGxvY2FsIGZpbGVzeXN0ZW0gb3IgYSByZW1vdGUgdXJsLgoKIyMgTW90aXZhdGlvbgpXaHkgYW5vdGhlciB0eXBlIGdlbmVyYXRvciBmb3IgT3BlbkFQSSAoU3dhZ2dlcik/IFdlbGwgaXQncyBiZWNhdXNlIEkgY291bGQgbm90IGZpbmQgYSBnZW5lcmF0b3IgdGhhdCBvbmx5IGdlbmVyYXRlcyB0eXBlc2NyaXB0IHR5cGVzLiAKTW9zdCBnZW5lcmF0b3JzIGFsc28gZ2VuZXJhdGUgcnVudGltZSBjb2RlIHdoaWNoIEkgZG9uJ3QgbmVjZXNzYXJpbHkgbmVlZC4gSSBqdXN0IHdhbnQgdG8gaGF2ZSB0aGUgdHlwZXNjcmlwdCB0eXBlcyB0byB1c2UgdGhlbSBpbiBteSBmcm9udGVuZCBhcHBsaWNhdGlvbi4KCklmIHRoYXRzIHNvbWV0aGluZyBmb3IgeW91LCBmZWVsIGZyZWUgdG8gdXNlIGl0LiBJZiB5b3UgbmVlZCBtb3JlIGZ1bmN0aW9uYWxpdHksIGZlZWwgZnJlZSB0byBvcGVuIGFuIGlzc3VlIG9yIGEgcHVsbCByZXF1ZXN0LgoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCmdvIGluc3RhbGwgZ2l0aHViLmNvbS9kaXRzY2hlZGV2L3N3YWctdHNAbGF0ZXN0CmBgYAoKIyMgVXNhZ2UKCmBgYGJhc2gKVXNhZ2U6CiAgc3dhZy10cyBbZmxhZ3NdCgpGbGFnczoKICAtZiwgLS1maWxlIHN0cmluZyAgICAgZmlsZSBwYXRoIG9yIHVybCB0byB0aGUgT3BlbkFQSSBTcGVjaWZpY2F0aW9uCiAgLWgsIC0taGVscCAgICAgICAgICAgIGhlbHAgZm9yIHN3YWctdHMKICAtbywgLS1vdXRwdXQgc3RyaW5nICAgb3V0cHV0IGZpbGUgZm9yIGdlbmVyYXRlZCBkZWZpbml0aW9ucyAoZGVmYXVsdCAiLi90eXBlcy9zd2FnZ2VyLnRzIikKICAtdiwgLS12ZXJzaW9uICAgICAgICAgc2hvd3MgdGhlIHZlcnNpb24gb2YgdGhlIGNsaQpgYGAKCiMjIEZvcm1hdApUaGlzIGxpYnJhcnkgYWltcyB0byBvbmx5IHByb3ZpZGUgdHlwZXNjcmlwdCB0eXBlIGRlZmluaXRpb25zIGZyb20gYSBnaXZlbiBPcGVuQVBJIFNwZWNpZmljYXRpb24uIEl0IGRvZXMgbm90IHByb3ZpZGUgYW55IHJ1bnRpbWUgZnVuY3Rpb25hbGl0eS4KQWxsIHR5cGVzIGFyZSBleHBvcnRlZCBhcyBgaW50ZXJmYWNlYC4KCkZvciBleGFtcGxlLCB0aGUgZm9sbG93aW5nIFNjaGVtYToKYGBgeWFtbApMb2dpblJlc3BvbnNlOgogIHJlcXVpcmVkOgogICAgLSB0b2tlbgogIHR5cGU6IG9iamVjdAogIHByb3BlcnRpZXM6CiAgICB0b2tlbjoKICAgICAgbWluTGVuZ3RoOiAxCiAgICAgIHR5cGU6IHN0cmluZwogIGFkZGl0aW9uYWxQcm9wZXJ0aWVzOiBmYWxzZQoKTG9naW5SZXNwb25zZVdyYXBwZXI6CiAgcmVxdWlyZWQ6CiAgICAtIGRhdGEKICB0eXBlOiBvYmplY3QKICBwcm9wZXJ0aWVzOgogICAgZGF0YToKICAgICAgJHJlZjogJyMvY29tcG9uZW50cy9zY2hlbWFzL0xvZ2luUmVzcG9uc2UnCiAgICBtZXNzYWdlOgogICAgICB0eXBlOiBzdHJpbmcKICAgICAgbnVsbGFibGU6IHRydWUKICBhZGRpdGlvbmFsUHJvcGVydGllczogZmFsc2UKYGBgCgp3aWxsIGJlIGNvbnZlcnRlZCB0byB0aGUgZm9sbG93aW5nIHR5cGVzY3JpcHQgZGVmaW5pdGlvbnM6CmBgYHR5cGVzY3JpcHQKZXhwb3J0IGludGVyZmFjZSBMb2dpblJlc3BvbnNlIHsKICB0b2tlbjogc3RyaW5nOwp9CgpleHBvcnQgaW50ZXJmYWNlIExvZ2luUmVzcG9uc2VXcmFwcGVyIHsKICBkYXRhOiBMb2dpblJlc3BvbnNlOwogIG1lc3NhZ2U/OiBzdHJpbmcgfCBudWxsOwp9CmBgYAoKIyMjIEVudW1zCkVudW1zIGFyZSBjb252ZXJ0ZWQgdG8gdHlwZXNjcmlwdCBlbnVtcy4gU2VlIHRoZSBleGFtcGxlIGJlbG93OgpgYGB5YW1sCkNhcjoKICByZXF1aXJlZDoKICAgIC0gbWFudWZhY3R1cmVyCiAgdHlwZTogb2JqZWN0CiAgcHJvcGVydGllczoKICAgIG1hbnVmYWN0dXJlcjoKICAgICAgJHJlZjogIiMvY29tcG9uZW50cy9zY2hlbWFzL0Nhck1hbnVmYWN0dXJlciIKQ2FyTWFudWZhY3R1cmVyOgogIHR5cGU6IHN0cmluZwogIGVudW06CiAgICAtIEJNVwogICAgLSBNZXJjZWRlcwogICAgLSBBdWRpICAKYGBgCgp3aWxsIGJlIGNvbnZlcnRlZCB0byB0aGUgZm9sbG93aW5nIHR5cGVzY3JpcHQgZGVmaW5pdGlvbnM6CmBgYHR5cGVzY3JpcHQKZXhwb3J0IGludGVyZmFjZSBDYXIgewogICAgbWFudWZhY3R1cmVyOiBDYXJNYW51ZmFjdHVyZXI7Cn0KCmV4cG9ydCBlbnVtIENhck1hbnVmYWN0dXJlciB7CiAgICBCTVcgPSAiQk1XIiwKICAgIE1lcmNlZGVzID0gIk1lcmNlZGVzIiwKICAgIEF1ZGkgPSAiQXVkaSIsCn0KYGBgCgojIyMgRm9ybURhdGEgUmVxdWVzdHMKSWYgeW91IGhhdmUgYSByZXF1ZXN0IHdpdGggYG11bHRpcGFydC9mb3JtLWRhdGFgIGNvbnRlbnQgdHlwZSB0aGUgY2xpIHdpbGwgZ2VuZXJhdGUgYSB0eXBlIGRlZmluaXRpb24gYXMgd2VsbC4KQXMgZm9yIG1vc3QgT3BlbkFQSSBTcGVjcyB0aGUgc2NoZW1hIG9mIHRoZSBmb3JtIGRhdGEgd2lsbCBub3QgYmUgYWRkZWQgdG8gdGhlIGBzY2hlbWFzYCBzZWN0aW9uIG9mIHRoZSBkZWZpbml0aW9uIGl0c2VsZiwgcmF0aGVyIHRoYW4gaW4gdGhlIGByZXF1ZXN0Qm9keWAgc2VjdGlvbiBvZiB0aGUgYHBhdGhgLgoKVGhlIGdlbmVyYXRlZCB0eXBlIHdpbGwgYmUgbmFtZWQgYWZ0ZXIgdGhlIG9wZXJhdGlvbiBpZCB3aXRoIGEgc3VmZml4IG9mIGBGb3JtRGF0YWAuIEZvciBjb252ZXJ0aW5nIHRoaXMgdHlwZSB0byBhIGBGb3JtRGF0YWAgb2JqZWN0IHlvdSBjYW4gdXNlIHRoZSBgY29udmVydFRvRm9ybURhdGFgIGZ1bmN0aW9uIGZyb20gdGhlIGdlbmVyYXRlZCBmaWxlLgo= readmeEtag: '"76497e8daa67d5a2f73d24df7c9466c049e6f09f"' readmeLastModified: Wed, 13 Dec 2023 15:31:12 GMT repositoryId: 641864655 description: >- swag-ts is a simple and fast code generator written in Go that creates typescript interfaces and enums for your openapi specification. created: '2023-05-17T10:22:08Z' updated: '2023-10-05T11:09:54Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: ditschedev logo: https://avatars.githubusercontent.com/u/13885333?v=4 license: MIT repoEtag: '"483db971264523da0abcccd2d1e6853d10f636facda0bfb3a20d9b5fad71e666"' repoLastModified: Thu, 05 Oct 2023 11:09:54 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/cangokceaslan/express-swagger-producer v3: true repositoryMetadata: base64Readme: >- IyBFeHByZXNzIFN3YWdnZXIgUHJvZHVjZXI6CgpUaGlzIG1vZHVsZSBoYXMgc2FtZSBsb2dpYyB3aXRoIFtleHByZXNzLXN3YWdnZXItZ2VuZXJhdG9yXShodHRwczovL25wbWpzLmNvbS9wYWNrYWdlL2V4cHJlc3Mtc3dhZ2dlci1nZW5lcmF0b3IpIGxpYnJhcnkgd2hpY2ggY3VycmVudGx5IGRvZXNuJ3Qgc3VwcG9ydCBUUyAoVHlwZVNjcmlwdCkgZW52aXJvbm1lbnQgb3IgZ2V0dGluZyB1cGRhdGVzLiBUaGlzIG1vZHVsZSBpcyB3cml0dGVuIHdpdGggVFMgKFR5cGVTY3JpcHQpIGFuZCBzdXBwb3J0cyBUUyAoVHlwZVNjcmlwdCkgZW52aXJvbm1lbnQuIAoKSW4gYWRkaXRpb24gdG8gVHlwZVNjcmlwdCBlbnZpcm9ubWVudCBzdXBwb3J0LCB0aGlzIG1vZHVsZSBoYXMgb25lIG1vcmUgZmVhdHVyZSB0aGFuIGV4cHJlc3Mtc3dhZ2dlci1nZW5lcmF0b3IuIFlvdSBjYW4gdXNlIHN3YWdnZXIgJiBvcGVuYXBpIHZlcnNpb24ga2V5cyB3aXRoIG9wdGlvbnMgYXMgZG93biBiZWxvdy4KCllvdSBjYW4gdmlzaXQgW2Nhbmdva2NlYXNsYW4uY29tXShodHRwczovL3d3dy5jYW5nb2tjZWFzbGFuLmNvbSkgZm9yIG1vcmUgZGV0YWlscwoKPGEgaHJlZj0iaHR0cHM6Ly93d3cuYnV5bWVhY29mZmVlLmNvbS9jYW5nb2tjZWFzbGFuIj48aW1nIHNyYz0iaHR0cHM6Ly9jZG4uYnV5bWVhY29mZmVlLmNvbS9idXR0b25zL3YyL2RlZmF1bHQteWVsbG93LnBuZyIgd2lkdGg9IjIwMCIgLz48L2E+CgojIyMjIEluc3RhbGxhdGlvbgoKYGBgCm5wbSBpIGV4cHJlc3Mtc3dhZ2dlci1wcm9kdWNlcgpgYGAKCiMjIyMgVXNhZ2UKCmBgYGphdmFzY3JpcHQKY29uc3QgZXhwcmVzcyA9IHJlcXVpcmUoJ2V4cHJlc3MnKTsKY29uc3QgYXBwID0gZXhwcmVzcygpOwpjb25zdCBFeHByZXNzU3dhZ2dlckZuID0gcmVxdWlyZSgnZXhwcmVzcy1zd2FnZ2VyLXByb2R1Y2VyJyk7CgpsZXQgb3B0aW9ucyA9IHsKICAgIHN3YWdnZXJEZWZpbml0aW9uOiB7CiAgICAgICAgaW5mbzogewogICAgICAgICAgICBkZXNjcmlwdGlvbjogJ1RoaXMgaXMgYSBzZXJ2ZXIgd2l0aCBlbmFibGVkIFN3YWdnZXIgZG9jdW1lbnRhdGlvbiBmZWF0dXJlJywKICAgICAgICAgICAgdGl0bGU6ICdTaW1wbGUgU2VydmVyJywKICAgICAgICAgICAgdmVyc2lvbjogJzEuMC4wJywKICAgICAgICB9LAogICAgICAgIGhvc3Q6ICdsb2NhbGhvc3Q6MzAwMCcsCiAgICAgICAgc3dhZ2dlcjonMi4wJywgLy8gIG9yIG9wZW5hcGk6JzMuMC4wJwogICAgICAgIGJhc2VQYXRoOiAnL3YxJywKICAgICAgICBwcm9kdWNlczogWwogICAgICAgICAgICAiYXBwbGljYXRpb24vanNvbiIsCiAgICAgICAgICAgICJhcHBsaWNhdGlvbi94bWwiCiAgICAgICAgXSwKICAgICAgICBzY2hlbWVzOiBbJ2h0dHAnLCAnaHR0cHMnXSwKCQlzZWN1cml0eURlZmluaXRpb25zOiB7CiAgICAgICAgICAgIEpXVDogewogICAgICAgICAgICAgICAgdHlwZTogJ2FwaUtleScsCiAgICAgICAgICAgICAgICBpbjogJ2hlYWRlcicsCiAgICAgICAgICAgICAgICBuYW1lOiAnQXV0aG9yaXphdGlvbicsCiAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbjogIkJhc2ljIGFwaUtleSBhdXRob3JpemF0aW9uIGluIHRoZSBzeXN0ZW0iLAogICAgICAgICAgICB9CiAgICAgICAgfQogICAgfSwKICAgIGJhc2VkaXI6IF9fZGlybmFtZSwgLy9hcHAgYWJzb2x1dGUgcGF0aAogICAgZmlsZXM6IFsnLi9yb3V0ZXMvKiovKi5qcyddIC8vUGF0aCB0byB0aGUgQVBJIGhhbmRsZSBmb2xkZXIKfTsKCmNvbnN0IEV4cHJlc3NTd2FnZ2VyID0gRXhwcmVzc1N3YWdnZXJGbihhcHApOyAvLyBQbGVhc2UgYWRkIHRoaXMgbGluZSB3aGVyZSB5b3VyIHJvdXRlcyBsYXllciBzdGFydHMKRXhwcmVzc1N3YWdnZXIob3B0aW9ucyk7IC8vIEVuYWJsZSB0aGlzIGlmIHlvdSB3YW50IHRvIGdlbmVyYXRlIFN3YWdnZXIgZG9jdW1lbnQKYXBwLmxpc3RlbigzMDAwKTsKYGBgCgpPcGVuIGh0dHAocyk6Ly88YXBwX2hvc3Q+OjxhcHBfcG9ydD4vYXBpLWRvY3MgaW4geW91ciBicm93c2VyIHRvIHZpZXcgdGhlIGRvY3VtZW50YXRpb24uCllvdSBjYW4gZmluZCB0aGUgc3dhZ2dlci5qc29uIGF0IGh0dHAocyk6Ly88YXBwX2hvc3Q+OjxhcHBfcG9ydD4vYXBpLWRvY3MuanNvbgoKIyMjIyBIb3cgdG8gZG9jdW1lbnQgdGhlIEFQSQoKYGBgamF2YXNjcmlwdAovKioKICogVGhpcyBmdW5jdGlvbiBjb21tZW50IGlzIHBhcnNlZCBieSBkb2N0cmluZQogKiBAcm91dGUgR0VUIC9hcGkKICogQGdyb3VwIGZvbyAtIE9wZXJhdGlvbnMgYWJvdXQgdXNlcgogKiBAcGFyYW0ge3N0cmluZ30gZW1haWwucXVlcnkucmVxdWlyZWQgLSB1c2VybmFtZSBvciBlbWFpbCAtIGVnOiB1c2VyQGRvbWFpbgogKiBAcGFyYW0ge3N0cmluZ30gcGFzc3dvcmQucXVlcnkucmVxdWlyZWQgLSB1c2VyJ3MgcGFzc3dvcmQuCiAqIEByZXR1cm5zIHtvYmplY3R9IDIwMCAtIEFuIGFycmF5IG9mIHVzZXIgaW5mbwogKiBAcmV0dXJucyB7RXJyb3J9ICBkZWZhdWx0IC0gVW5leHBlY3RlZCBlcnJvcgogKi8KZXhwb3J0cy5mb28gPSBmdW5jdGlvbigpIHt9CmBgYAoKRm9yIG1vZGVsIGRlZmluaXRpb25zOgoKYGBgamF2YXNjcmlwdAovKioKICogQHR5cGVkZWYgUHJvZHVjdAogKiBAcHJvcGVydHkge2ludGVnZXJ9IGlkCiAqIEBwcm9wZXJ0eSB7c3RyaW5nfSBuYW1lLnJlcXVpcmVkIC0gU29tZSBkZXNjcmlwdGlvbiBmb3IgcHJvZHVjdAogKiBAcHJvcGVydHkge0FycmF5LjxQb2ludD59IFBvaW50CiAqLwoKLyoqCiAqIEB0eXBlZGVmIFBvaW50CiAqIEBwcm9wZXJ0eSB7aW50ZWdlcn0geC5yZXF1aXJlZAogKiBAcHJvcGVydHkge2ludGVnZXJ9IHkucmVxdWlyZWQgLSBTb21lIGRlc2NyaXB0aW9uIGZvciBwb2ludCAtIGVnOiAxMjM0CiAqIEBwcm9wZXJ0eSB7c3RyaW5nfSBjb2xvcgogKiBAcHJvcGVydHkge2VudW19IHN0YXR1cyAtIFN0YXR1cyB2YWx1ZXMgdGhhdCBuZWVkIHRvIGJlIGNvbnNpZGVyZWQgZm9yIGZpbHRlciAtIGVnOiBhdmFpbGFibGUscGVuZGluZwogKi8KCi8qKgogKiBAdHlwZWRlZiBFcnJvcgogKiBAcHJvcGVydHkge3N0cmluZ30gY29kZS5yZXF1aXJlZAogKi8KCi8qKgogKiBAdHlwZWRlZiBSZXNwb25zZQogKiBAcHJvcGVydHkge1tpbnRlZ2VyXX0gY29kZQogKi8KCgovKioKICogVGhpcyBmdW5jdGlvbiBjb21tZW50IGlzIHBhcnNlZCBieSBkb2N0cmluZQogKiBUZXN0IFJvdXRlCiAqIEByb3V0ZSBQT1NUIC91c2VycwogKiBAcGFyYW0ge1BvaW50Lm1vZGVsfSBwb2ludC5ib2R5LnJlcXVpcmVkIC0gdGhlIG5ldyBwb2ludAogKiBAZ3JvdXAgZm9vIC0gT3BlcmF0aW9ucyBhYm91dCB1c2VyCiAqIEBwYXJhbSB7c3RyaW5nfSBlbWFpbC5xdWVyeS5yZXF1aXJlZCAtIHVzZXJuYW1lIG9yIGVtYWlsCiAqIEBwYXJhbSB7c3RyaW5nfSBwYXNzd29yZC5xdWVyeS5yZXF1aXJlZCAtIHVzZXIncyBwYXNzd29yZC4KICogQHBhcmFtIHtlbnVtfSBzdGF0dXMucXVlcnkucmVxdWlyZWQgLSBTdGF0dXMgdmFsdWVzIHRoYXQgbmVlZCB0byBiZSBjb25zaWRlcmVkIGZvciBmaWx0ZXIgLSBlZzogYXZhaWxhYmxlLHBlbmRpbmcKICogQG9wZXJhdGlvbklkIHJldHJpZXZlRm9vSW5mbwogKiBAcHJvZHVjZXMgYXBwbGljYXRpb24vanNvbiBhcHBsaWNhdGlvbi94bWwKICogQGNvbnN1bWVzIGFwcGxpY2F0aW9uL2pzb24gYXBwbGljYXRpb24veG1sCiAqIEByZXR1cm5zIHtSZXNwb25zZS5tb2RlbH0gMjAwIC0gQW4gYXJyYXkgb2YgdXNlciBpbmZvCiAqIEByZXR1cm5zIHtQcm9kdWN0Lm1vZGVsfSAgZGVmYXVsdCAtIFVuZXhwZWN0ZWQgZXJyb3IKICogQHJldHVybnMge0FycmF5LjxQb2ludD59IFBvaW50IC0gU29tZSBkZXNjcmlwdGlvbiBmb3IgcG9pbnQKICogQGhlYWRlcnMge2ludGVnZXJ9IDIwMC5YLVJhdGUtTGltaXQgLSBjYWxscyBwZXIgaG91ciBhbGxvd2VkIGJ5IHRoZSB1c2VyCiAqIEBoZWFkZXJzIHtzdHJpbmd9IDIwMC5YLUV4cGlyZXMtQWZ0ZXIgLSAJZGF0ZSBpbiBVVEMgd2hlbiB0b2tlbiBleHBpcmVzCiAqIEBzZWN1cml0eSBKV1QKICovCmBgYA== readmeEtag: '"81d793407c3bec6616b4d37a050358a63ec92cc4"' readmeLastModified: Tue, 03 May 2022 19:43:39 GMT repositoryId: 470741285 description: "This repository contains the source codes from npm package: express-swagger-producer |\_express-swagger-producer is TypeScript module that supports generating Swagger Api Documentation File with comments" created: '2022-03-16T20:27:05Z' updated: '2023-12-23T14:51:49Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 0 owner: cangokceaslan logo: https://avatars.githubusercontent.com/u/48398625?v=4 license: MIT repoEtag: '"b509e5b9ac18d1f7155a232f2a41874b097aadddde783d55f039f4b55a42be5b"' repoLastModified: Sat, 23 Dec 2023 14:51:49 GMT foundInMaster: true category: Parsers id: 0461674804ad63872c0efb2049669dbc - source: openapi3 tags repository: https://github.com/ruiaraujo012/openapi-express-codegen v3: true id: b7d1a7754796abe41f1cb0a854b3b89d repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIEV4cHJlc3MgQ29kZWdlbgoKPiBOb2RlLmpzIGxpYnJhcnkgdGhhdCBnZW5lcmF0ZXMgZXhwcmVzcyBUeXBlc2NyaXB0IHR5cGVzIGJhc2VkIG9uIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24uCgojIyBJbnN0YWxsCgpgYGAKbnBtIGluc3RhbGwgb3BlbmFwaS1leHByZXNzLWNvZGVnZW4gLS1zYXZlLWRldgpgYGAKCiMjIFVzYWdlCgpgYGAKJCBleHByZXNzLWdlbiAtLWhlbHAKCiAgVXNhZ2U6IGV4cHJlc3MtZ2VuIFtvcHRpb25zXQoKICBPcHRpb25zOgogICAgLVYsIC0tdmVyc2lvbiAgICAgICAgICAgICBvdXRwdXQgdGhlIHZlcnNpb24gbnVtYmVyCiAgICAtaSwgLS1pbnB1dCA8dmFsdWU+ICAgICAgIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiwgY2FuIGJlIGEgcGF0aCwgdXJsIG9yIHN0cmluZyBjb250ZW50IChyZXF1aXJlZCkKICAgIC1vLCAtLW91dHB1dCA8dmFsdWU+ICAgICAgT3V0cHV0IGRpcmVjdG9yeSAocmVxdWlyZWQpCiAgICAtLXVzZVVuaW9uVHlwZXMgICAgICAgICAgIFVzZSB1bmlvbiB0eXBlcyBpbnN0ZWFkIG9mIGVudW1zCiAgICAtLWV4cG9ydEludGVyZmFjZXMgPHZhbHVlPiAgV3JpdGUgaW50ZXJmYWNlcyB0byBkaXNrIChkZWZhdWx0OiB0cnVlKQogICAgLS1leHBvcnRNb2RlbHMgPHZhbHVlPiAgICBXcml0ZSBtb2RlbHMgdG8gZGlzayAoZGVmYXVsdDogdHJ1ZSkKICAgIC0tZXhwb3J0U2NoZW1hcyA8dmFsdWU+ICAgV3JpdGUgc2NoZW1hcyB0byBkaXNrIChkZWZhdWx0OiBmYWxzZSkKICAgIC0taW5kZW50IDx2YWx1ZT4gICAgICAgICAgSW5kZW50YXRpb24gb3B0aW9ucyBbNCwgMiwgdGFiXSAoZGVmYXVsdDogIjQiKQogICAgLS1wb3N0Zml4IDx2YWx1ZT4gICAgICAgICBJbnRlcmZhY2UgbmFtZSBwb3N0Zml4IChkZWZhdWx0OiAiR2VuZXJhdGVkIikKICAgIC1oLCAtLWhlbHAgICAgICAgICAgICAgICAgZGlzcGxheSBoZWxwIGZvciBjb21tYW5kCgogIEV4YW1wbGVzCiAgICAkIGV4cHJlc3MtZ2VuIC0taW5wdXQgLi9zcGVjLmpzb24gLS1vdXRwdXQgLi9nZW5lcmF0ZWQKYGBgCgojIERvY3VtZW50YXRpb24KCi0gW0Jhc2ljIHVzYWdlXShkb2NzL2Jhc2ljLXVzYWdlLm1kKQoKIyBSZWZlcmVuY2VzCgpUaGlzIHByb2plY3QgaXMgYmFzZWQgb246IFtvcGVuYXBpLXR5cGVzY3JpcHQtY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL2ZlcmRpa29vbWVuL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlZ2VuKQo= readmeEtag: '"f56fe896c3fb60d4ecad1c14aa9c08a6e9504ded"' readmeLastModified: Sat, 25 Mar 2023 23:51:09 GMT repositoryId: 500802549 description: >- Library that generates express Typescript types based on the OpenAPI specification. created: '2022-06-07T10:59:22Z' updated: '2024-01-26T11:08:36Z' language: TypeScript archived: true stars: 2 watchers: 1 forks: 0 owner: ruiaraujo012 logo: https://avatars.githubusercontent.com/u/16304685?v=4 license: MIT repoEtag: '"cedc7edcdbe7788295e9d5792169d52c7b1f4b41850e3f0df345a9a2ae2b08d5"' repoLastModified: Fri, 26 Jan 2024 11:08:36 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/gbarre/capsule-api v3: true repositoryMetadata: base64Readme: >- # Capsule-API Documentation

![python versions](https://img.shields.io/badge/Python-3.6%20%7C%203.7%20%7C%203.8-blue.svg)

| CI | Pipeline / build | Coverage |
|----|------------------|----------|
| Gitlab (internal) | [![pipeline status](https://git.in.ac-versailles.fr/capsule/capsapi/badges/master/pipeline.svg)](https://git.in.ac-versailles.fr/capsule/capsapi/commits/master) | [![coverage report](https://git.in.ac-versailles.fr/capsule/capsapi/badges/master/coverage.svg)](https://git.in.ac-versailles.fr/capsule/capsapi/commits/master) |
| Github (travis) | [![Build Status](https://travis-ci.com/gbarre/capsule-api.svg?branch=master)](https://travis-ci.com/gbarre/capsule-api) | N/A |

## Run the capsule API server in the development environment

**Requirements** : to run the full stack, you need to install:

- docker
- docker-compose
- python3.6 (or higher)
- python3.6-dev
- jq

Then:

```sh
# You have to activate a virtualenv and install the required packages.
python3 -m venv ./venv
. venv/bin/activate
pip install --upgrade pip
pip install --upgrade setuptools
pip install -r requirements.txt
pip install -r test-requirements.txt # If you want to be able to run tests too.

# Run a docker instance of keycloak a valid config file ./config-dev.yml
# for the capsule-api server will be created.
./keycloak/start.sh

# To up the local MySQL server and the NATS server.
docker-compose up -d

# To open a mysql client in the docker.
#
#docker-compose exec db mysql -u root -p'local' capsule_local

# To apply a migration of the database.
FLASK_APP=server.py CAPSULE_API_CONFIG=config-dev.yml python -m flask db upgrade

# And then, to run a dev/test capsule-api server (not relevant for a production server).
python -Wd server.py -c config-dev.yml
```

**Remark:** if the server is running, you can view the API specification
at the address [http://localhost:5000/v1/ui/](http://localhost:5000/v1/ui/).

## To stop capsule API server and remove all docker instances

Type `CTRL+c` to stop the current execution of capsule-api server.
Then, to remove all docker instances (keyloack, MySQL and NATS):

```sh
# Remove the keycloak instance.
docker stop keycloak_dev && docker rm keycloak_dev

# Remove the NATS and MySQL instances.
# Warning, -v option remove the MySQL volume and you will lose all data.
# Don't mention the -v option if you want to keep the MySQL volume.
docker-compose down -v
```

## Database migration

```sh
# To add a new migration :
FLASK_APP=server.py CAPSULE_API_CONFIG=config-dev.yml python -m flask db migrate -m "My new migration"

# To apply a migration.
FLASK_APP=server.py CAPSULE_API_CONFIG=config-dev.yml python -m flask db upgrade
```

## Run tests

```sh
# To run cover (which includes tests)
python -m pytest -v -n8 --cov=. --cov-config=.coveragerc-ci --cov-report html --cov-report term tests/api/

# To run lint
flake8 --extend-exclude=venv,migrations --ignore=E402

# To run secaudit
bandit -n5 -x "./venv/*,./tests/*,./dev-tools/*" -r . -ll
```

## Update API specifications

After updating the API spec, you must rebuild the `openapi.json` file with this command:

```sh
docker run --rm -v "$PWD/spec:/spec" -it jeanberu/swagger-cli swagger-cli bundle -o /spec/openapi.json /spec/index.yaml
```

## Run production server

```sh
gunicorn --access-logfile - --bind 0.0.0.0:5000 -w 4 --preload server:app
```

## Run the docker production server

```sh
docker run --rm \
  -v $PWD/config.yml:/etc/capsule-api/config.yml \
  --net=host \
  -e WORKERS=8 \
  -e DB_MIGRATE=upgrade \
  --name capsule-api \
  harbor.in.ac-versailles.fr/infra/capsule-api:2.0.1-57fb135c-00639
```

To use with HTTPS **in development** run:

```sh
docker run --rm \
  -v $PWD/config.yml:/etc/capsule-api/config.yml \
  --net=host \
  -e WORKERS=8 \
  -e DB_MIGRATE=upgrade \
  -e SSL=true \
  --name capsule-api \
  harbor.in.ac-versailles.fr/infra/capsule-api:2.0.1-57fb135c-00639
```

To use with HTTPS **with your certificate** run:

```sh
docker run --rm \
  -v $PWD/config.yml:/etc/capsule-api/config.yml \
  --net=host \
  -e WORKERS=8 \
  -e DB_MIGRATE=upgrade \
  -e SSL=true \
  -v ${PATH_TO_CRT_FILE}:/capsule-api/cert/capsule.crt \
  -v ${PATH_TO_KEY_FILE}:/capsule-api/cert/capsule.key \
  --name capsule-api \
  harbor.in.ac-versailles.fr/infra/capsule-api:2.0.1-57fb135c-00639
```

## A few usefull commands

### Hack the code and create capsules and users

```sh
curl --location --request POST 'http://localhost:5000/v1/capsules' \
    --header 'Content-Type: application/json' \
    --header "Authorization: Bearer $TOKEN" \
    --data-raw '{ "name": "Test-Capsule-1", "owners": [ "userfoo", "userbar" ] }'
```

### Get keycloak users

```sh
#!/bin/sh

# requires jq

# config
KEYCLOAK_URL=https://keycloak.example.com/auth
KEYCLOAK_REALM=my_realm
KEYCLOAK_CLIENT_ID=my_client_id
KEYCLOAK_CLIENT_SECRET=ffffffff-ffff-ffff-ffff-ffffffffffff

# DO NOT EDIT NEXT THIS LINE
FILTER=""
USER="$1"

# Get valid token
RESULT=`curl -s --data "grant_type=client_credentials&client_id=${KEYCLOAK_CLIENT_ID}&client_secret=${KEYCLOAK_CLIENT_SECRET}" \
  ${KEYCLOAK_URL}/realms/${KEYCLOAK_REALM}/protocol/openid-connect/token`
export TKN=$(echo $RESULT | jq -r '.access_token')

# Look for specific user ?
if [ ! -z $USER ]
then
  FILTER="?username=${USER}"
fi

# Get user(s)
curl -s -X GET "${KEYCLOAK_URL}/admin/realms/${KEYCLOAK_REALM}/users/${FILTER}" \
-H "Accept: application/json" \
-H "Authorization: Bearer $TKN"

```

### Send Nats request to simulate a driver

**Warning**: ensure all private/public keys are setted in the `config.ylm`.

```sh
cd dev-tools
python publish_msg.py --nats=localhost:4222 --subject="capsule.addon.ecea7683-92a8-4e2d-a846-be3c92f01308" --state="?list" --data='{}'
python publish_msg.py --nats=localhost:4222 --subject="capsule.webapp" --state="?state" --data='{"id": "19129f93-b50c-4d06-9c96-d779d1dac467"}'
```

## TODO

- API:
  - add webapp/addon status-flag
- Front:
  - capsule:
    - Send error message before the end of creation
    - Catch owner repetition on creation
    - Check if user exists in keycloak after the local search fails
  - addons:
    - Change "file" option (like webapp)
    - Script to check consistency between sqlite & existing databases
 readmeEtag: '"f9102c13c17ce831f90335e3456d8f89d7cc7c85"' readmeLastModified: Thu, 08 Feb 2024 14:10:36 GMT repositoryId: 247653584 description: null created: '2020-03-16T08:53:52Z' updated: '2023-02-15T08:54:46Z' language: Python archived: false stars: 2 watchers: 4 forks: 2 owner: gbarre logo: https://avatars.githubusercontent.com/u/2445791?v=4 repoEtag: '"d3e8898dd83243fbd63a533c445696bba58ac64fdb43b36d4218728ac3ed4ad7"' repoLastModified: Wed, 15 Feb 2023 08:54:46 GMT foundInMaster: true category: Server Implementations id: d99a6f581aa69091e73283d4a9cb3b72 - source: openapi3 tags repository: https://github.com/dota-showcase/inventory-service v3: true id: 110eab7ad814a12856226121e167c630 repositoryMetadata: base64Readme: >- IyBEb3RhIFNob3djYXNlIC0gSW52ZW50b3J5IFNlcnZpY2UKClJFU1QgQVBJIHNlcnZpY2UgdG8gc3RvcmUgYW5kIHRyYWNrIGNoYW5nZXMgb2Ygc3RlYW0gdXNlcidzIGRvdGEgMiBpbnZlbnRvcnkgaXRlbXMuCgojIyBGZWF0dXJlcwoKKiBJbnZlbnRvcnkgaGlzdG9yeSAtIGNyZWF0ZWQsIHVwZGF0ZWQsIGRlbGV0ZWQgaXRlbXMKKiBSYXRlIGxpbWl0ZXIKKiBVbml0IHRlc3RzCgojIyBBUEkgRG9jcwoKU3dhZ2dlciBPcGVuQVBJIDMgaXMgYXZhaWxhYmxlIGJ5IHBhdGg6CgoqIFVJOiAgIGBob3N0OnBvcnQvc3dhZ2dlci11aS5odG1sYAoqIEpTT046IGBob3N0OnBvcnQvYXBpLWRvY3NgCgojIyMgSW52ZW50b3J5IEVuZHBvaW50cwoKKiBTaG93IEludmVudG9yaWVzOiBgR0VUIC9hcGkvdjEvaW52ZW50b3JpZXNgCiogU2VhcmNoIEludmVudG9yaWVzIChwYWdpbmF0ZWQpOiBgR0VUIC9hcGkvdjEvaW52ZW50b3JpZXMvc2VhcmNoLXBhZ2VgCiogQ3JlYXRlIGFuIEludmVudG9yeSAobG9hZCBpdGVtcyBmcm9tIHN0ZWFtKTogYFBPU1QgL2FwaS92MS9pbnZlbnRvcmllcy86c3RlYW1pZGAKKiBTaG93IGFuIEludmVudG9yeTogYEdFVCAvYXBpL3YxL2ludmVudG9yaWVzLzpzdGVhbWlkYAoqIFVwZGF0ZSBhbiBJbnZlbnRvcnkgKHJlbG9hZCBpdGVtcyBmcm9tIHN0ZWFtKTogYFBVVCAvYXBpL3YxL2ludmVudG9yaWVzLzpzdGVhbWlkYAoqIERlbGV0ZSBhbiBJbnZlbnRvcnk6IGBERUxFVEUgL2FwaS92MS9pbnZlbnRvcmllcy86c3RlYW1pZGAKCiMjIyBJbnZlbnRvcnkgSXRlbXMgRW5kcG9pbnRzCgoqIFNlYXJjaCBJbnZlbnRvcnkgSXRlbXMgKGZpbHRlciB3aXRoIGxhcmdlIHBheWxvYWRzKTogYFBPU1QgL2FwaS92MS9pbnZlbnRvcmllcy86c3RlYW1pZC9pdGVtcy9zZWFyY2hgCiogU2VhcmNoIEludmVudG9yeSBJdGVtcyAocGFnaW5hdGVkOyBmaWx0ZXIgd2l0aCBsYXJnZSBwYXlsb2Fkcyk6IGBQT1NUIC9hcGkvdjEvaW52ZW50b3JpZXMvOnN0ZWFtaWQvaXRlbXMvc2VhcmNoLXBhZ2VgCiogU2hvdyBJbnZlbnRvcnkgSXRlbXMgKHBvc2l0aW9uZWQgcGFnaW5hdGlvbik6IGBHRVQgL2FwaS92MS9pbnZlbnRvcmllcy86c3RlYW1pZC9pdGVtcy9wYWdlLXBvc2l0aW9uZWRgCiogU2hvdyBJbnZlbnRvcnkgSXRlbXMgJ2RlZkluZGV4JyBmaWVsZHM6IGBHRVQgL2FwaS92MS9pbnZlbnRvcmllcy86c3RlYW1JZC9pdGVtcy9kZWYtaW5kZXhlc2AKKiBTaG93IEludmVudG9yeSBJdGVtcyBjaGFuZ2VzOiBgR0VUIC9hcGkvdjEvaW52ZW50b3JpZXMvOnN0ZWFtaWQvY2hhbmdlcy86dmVyc2lvbi86dHlwZWAKCiMjIyBPcGVyYXRpb25zIEVuZHBvaW50cwoKKiBTZWFyY2ggT3BlcmF0aW9ucyAocGFnaW5hdGVkKTogYEdFVCAvYXBpL3YxL2ludmVudG9yaWVzLzpzdGVhbWlkL29wZXJhdGlvbnMvc2VhcmNoLXBhZ2VgCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCiMjIyBQcmVyZXF1aXNpdGVzCgoqIEphdmEgMjUKKiBHcmFkbGUgOQoqIE1vbmdvREIgOC4yCgojIyMgSW5zdGFsbGF0aW9uCgoxLiBDcmVhdGUgYC5lbnZgIGJhc2VkIG9uIGAuZW52LmV4YW1wbGVgCjIuIENyZWF0ZSBgZW52aXJvbm1lbnQucHJvcGVydGllc2AgYmFzZWQgb24gYGVudmlyb25tZW50LnByb3BlcnRpZXMuZXhhbXBsZWAgaW4gYGFwcC9zcmMvbWFpbi9yZXNvdXJjZXNgCjMuIFNldCB0aGUgc3RlYW0gYXBpIGtleSBwcm9wZXJ0eSB0byBhY2Nlc3MgU3RlYW0ncyBBUEk6CiAgICBgYGBiYXNoCiAgICBlbnYuc3RlYW0uYXBpLmtleT15b3VyX2FwaV9rZXkKICAgIGBgYAo0LiBUbyBydW4gaW4gZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnQuIAogICBZb3UgY2FuIHVzZSBhIGRlZmF1bHQgYnVpbGQgZm9yIEludGVsbGlKIElERUEgdG8gcnVuIHRoZSBhcHAgLSBgLnJ1bi9kZXYtYnVpbGQucnVuLnhtbGAuCiAgICBgYGBiYXNoCiAgICBkb2NrZXItY29tcG9zZSB1cAogICAgYGBgCjUuIFRvIHJ1biBpbiBwcm9kdWN0aW9uIGVudmlyb25tZW50CiAgICBgYGBiYXNoCiAgICBkb2NrZXIgY29tcG9zZSAtZiBkb2NrZXItY29tcG9zZS5wcm9kLnltbCB1cCAtZAogICAgYGBgCgojIyBEYXRhIE1vZGVsCgpDb25jZXB0dWFsIGVudGl0eSByZWxhdGlvbiBkaWFncmFtIChCYXJrZXIncyBzeXN0ZW0pOgoKIVtdKGVudGl0eS1yZWxhdGlvbi1kaWFncmFtLnBuZykKCiMjIEJhY2stb2YtdGhlLWVudmVsb3BlIEVzdGltYXRpb25zCgojIyMgIFN0b3JhZ2UgU3BhY2UKCkFzc3VtcHRpb25zOgoKKiBBdmVyYWdlIGludmVudG9yeSBpdGVtIHNpemUgLSA1MDAgYnl0ZXMuCiogQXZlcmFnZSB1c2VyIGhhcyA1MDAwIGludmVudG9yeSBpdGVtcy4KKiBVc2VycyB1cGRhdGUgaW52ZW50b3J5IG9uY2UgcGVyIGRheS4gRWFjaCBkYXkgaXQgaW5jcmVhc2VkIGJ5IHNvbWUgc21hbGwgY29lZmZpY2llbnQgKDAuMDUpLgoqIERhdGEgaXMgc3RvcmVkIGZvciBhIHllYXIuCgpFc3RpbWF0ZToKCiogVXNlciBpbnZlbnRvcnkgc2l6ZTogNTAwIEIgKiA1MDAwID0gMi41IE1CCiogVXNlciBpbnZlbnRvcnkgc2l6ZSBmb3IgYSB5ZWFyOiAyLjUgTUIgKyAoMzY1ICogMC4wNSAqIDIuNSBNQikgPSB+NTAgTUIKKiAqKlN0b3JhZ2UgZm9yIDEwMEsgdXNlcnM6IDUwTUIgKiAxMDAwMDAgPSB+NSBUQioqCiMjIFJlc291cmNlcwoKLSBbU3RlYW0gQVBJIC0gR2V0UGxheWVySXRlbXNdKGh0dHBzOi8vd2lraS50ZWFtZm9ydHJlc3MuY29tL3dpa2kvV2ViQVBJL0dldFBsYXllckl0ZW1zKQoKIyMgSXNzdWVzCgojIyMgUnVuIHRlc3RzIG9uIGxpbnV4IC0gcHJvYmxlbSB3aXRoIGZsYXBkb29kbGUgZW1iZWRkZWQgbW9uZ28gZGIKRG93bmxvYWQgJiBpbnN0YWxsIGxpYnNzbDEuMSBsYXRlc3QgdmVyc2lvbjoKCmBgYGJhc2gKd2dldCBodHRwOi8vbnoyLmFyY2hpdmUudWJ1bnR1LmNvbS91YnVudHUvcG9vbC9tYWluL28vb3BlbnNzbC9saWJzc2wxLjFfMS4xLjFmLTF1YnVudHUyX2FtZDY0LmRlYgpzdWRvIGRwa2cgLWkgbGlic3NsMS4xXzEuMS4xZi0xdWJ1bnR1Ml9hbWQ2NC5kZWIKIGBgYAoKIyMgTGljZW5zZQoKVGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIHYzLjAuIFBsZWFzZSBzZWUgW0xpY2Vuc2UgRmlsZV0oTElDRU5TRSkgZm9yIG1vcmUgaW5mb3JtYXRpb24uCg== readmeEtag: '"06ebdc0a086105c594cfad80f712eeba3c008aec"' readmeLastModified: Wed, 14 Jan 2026 03:35:30 GMT repositoryId: 506423971 description: REST service to store and track changes of Dota 2 inventory items. created: '2022-06-22T22:29:20Z' updated: '2026-01-27T20:27:41Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: dota-showcase logo: https://avatars.githubusercontent.com/u/104716440?v=4 license: GPL-3.0 repoEtag: '"6143db8fc840945a5e9cbaab4ef5e8ace047307f5d312c98fb45b35365c36777"' repoLastModified: Tue, 27 Jan 2026 20:27:41 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sierrasoftworks/rex-rs v3: true repositoryMetadata: base64Readme: >- IyBSYW5keSAKKipLZWVwIHRyYWNrIG9mIHZhcmlvdXMgaWRlYXMgYW5kIHJhbmRvbWx5IHNlbGVjdCBvbmUgb24gZGVtYW5kKioKClshW1RyYXZpcyAoLmNvbSldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdHJhdmlzL2NvbS9TaWVycmFTb2Z0d29ya3MvcmFuZHktcnMuc3ZnP3N0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL3RyYXZpcy1jaS5jb20vU2llcnJhU29mdHdvcmtzL3JhbmR5LXJzKQpbIVtBUEkgZG9jc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9kb2NzLWFwaS1ibHVlLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9yYW5keWFwcC5kb2NzLmFwaWFyeS5pbykKIVtHaXRIdWJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvU2llcnJhU29mdHdvcmtzL3JhbmR5LXJzLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKQoKUmFuZHkgaXMgYSB0b29sIGZvciBrZWVwaW5nIHRyYWNrIG9mIGlkZWFzIGZvciB0aGluZ3MgdG8gZG8gYW5kIHByb3ZpZGluZywgb24gZGVtYW5kLAphIHJhbmRvbSBvbmUgdG8gZG8uCg== readmeEtag: '"7eb11f451b9df79f17b3c2be8b21179a33dc8998"' readmeLastModified: Tue, 23 Jul 2024 17:21:45 GMT repositoryId: 175636823 description: Tool for keeping track of ideas and providing random ones on demand created: '2019-03-14T14:20:18Z' updated: '2026-02-05T18:56:05Z' language: Rust archived: false stars: 2 watchers: 0 forks: 0 owner: SierraSoftworks logo: https://avatars.githubusercontent.com/u/5012716?v=4 license: MIT repoEtag: '"0fa793e6131b4df7e0fe048ff9822dc3a221e3945e624346c41fd79b60cb5c05"' repoLastModified: Thu, 05 Feb 2026 18:56:05 GMT foundInMaster: true category: - Description Validators - Testing id: f6a674399fe0c67c7b8178d1394cdf03 - source: openapi3 tags repository: https://github.com/wfcd/api-spec v3: true repositoryMetadata: base64Readme: >- IyBXYXJmcmFtZVN0YXQudXMgT3BlbkFQSSBTcGVjaWZpY2F0aW9uClshW0dpdEh1YiBwYWdlc10oaHR0cHM6Ly9naXRodWIuY29tL1dGQ0QvYXBpLXNwZWMvYWN0aW9ucy93b3JrZmxvd3MvZG9jcy55YW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9XRkNEL2FwaS1zcGVjL2FjdGlvbnMvd29ya2Zsb3dzL2RvY3MueWFtbCkKCiMjIExpbmtzCgotIFtSZWZlcmVuY2UgRG9jdW1lbnRhdGlvbiAoUmVEb2MpXShodHRwczovL2RvY3Mud2FyZnJhbWVzdGF0LnVzKQotIFtTd2FnZ2VyVUldKGh0dHBzOi8vZG9jcy53YXJmcmFtZXN0YXQudXMvc3dhZ2dlci11aS8pCi0gT3BlbkFQSSBSYXcgRmlsZXM6IFtKU09OXShodHRwczovL2RvY3Mud2FyZnJhbWVzdGF0LnVzL29wZW5hcGkuanNvbikgW1lBTUxdKGh0dHBzOi8vZG9jcy53YXJmcmFtZXN0YXQudXMvb3BlbmFwaS55YW1sKQoKKipXYXJuaW5nOioqIEFsbCBhYm92ZSBsaW5rcyBhcmUgdXBkYXRlZCBvbmx5IGFmdGVyIEFjdGlvbnMgZmluaXNoZXMgZGVwbG95bWVudAoKIyMgV29ya2luZyBvbiBzcGVjaWZpY2F0aW9uCiMjIyBJbnN0YWxsCgoxLiBJbnN0YWxsIFtOb2RlIEpTXShodHRwczovL25vZGVqcy5vcmcvKQoyLiBDbG9uZSByZXBvIGFuZCBydW4gYG5wbSBpbnN0YWxsYCBpbiB0aGUgcmVwbyByb290CgojIyMgVXNhZ2UKCiMjIyMgYG5wbSBzdGFydGAKU3RhcnRzIHRoZSBkZXZlbG9wbWVudCBzZXJ2ZXIgb24gcG9ydCBbYDgwODBgXShodHRwOi8vbG9jYWxob3N0OjgwODApLgoKIyMjIyBgbnBtIHJ1biBidW5kbGVgCkJ1bmRsZXMgdGhlIHNwZWMgYW5kIHByZXBhcmVzIHdlYl9kZXBsb3kgZm9sZGVyIHdpdGggc3RhdGljIGFzc2V0cy4KCiMjIyMgYG5wbSB0ZXN0YApWYWxpZGF0ZXMgdGhlIHNwZWMuCgojIyMjIGBucG0gcnVuIGdoLXBhZ2VzYApEZXBsb3lzIGRvY3MgdG8gR2l0SHViIFBhZ2VzLiBZb3UgZG9uJ3QgbmVlZCB0byBydW4gaXQgbWFudWFsbHkgaWYgeW91IGhhdmUgVHJhdmlzIENJIGNvbmZpZ3VyZWQuCg== readmeEtag: '"00c8f2f94c762cbc94cfbb8b88fb9d448a293275"' readmeLastModified: Fri, 06 Sep 2024 17:42:22 GMT repositoryId: 115034680 description: OpenAPI Specifications for WarframeStat.us API created: '2017-12-21T18:21:32Z' updated: '2026-02-06T01:22:13Z' language: CSS archived: false stars: 3 watchers: 6 forks: 5 owner: WFCD logo: https://avatars.githubusercontent.com/u/24436369?v=4 license: MIT repoEtag: '"6da66e68e390826bb9a578d1a6ddd529151358fbe6b7d40342eafb05ac78caab"' repoLastModified: Fri, 06 Feb 2026 01:22:13 GMT foundInMaster: true category: - Description Validators - Parsers id: b24c18cad2dfc337dc62885cc4203361 - source: openapi3 tags repository: https://github.com/alwaysbemark/spring-boot-openapi-generator v3: true repositoryMetadata: base64Readme: >- IyBTcHJpbmcgQm9vdCBBUEkgRmlyc3QgRGVzaWduIFdpdGggT3BlbkFQSSBHZW5lcmF0b3IKVGhpcyBzaG93Y2FzZXMgYSBzaW1wbGlzdGljIFNwcmluZyBCb290IHByb2plY3Qgd2l0aCBhbiBBUEktZmlyc3QgZGVzaWduIHBhdHRlcm4uICAKVGhlIHJlc3QgcGFja2FnZSBwcm92aWRlcyBhbiBpbXBsZW1lbnRhdGlvbiBvZiBhIFJFU1QgY29udHJvbGxlciB0aGF0IGltcG9ydHMgQVBJIGFuZCBtb2RlbCBmaWxlcyB0aGF0IGdldCBnZW5lcmF0ZWQgYXV0b21hdGljYWxseSBmcm9tIHRoZSBzdXBwbGllZCBbT3BlbkFQSSAzIHNwZWMgZmlsZV0oc3JjL21haW4vcmVzb3VyY2VzL2FwaS55bWwpLiAgCkFwaSBhbmQgbW9kZWwgZmlsZXMgd2lsbCBiZSBnZW5lcmF0ZWQgYmVmb3JlIHRoZSBjb21waWxlSmF2YSB0YXNrIG9mIHRoZSBHcmFkbGUgSmF2YSBwbHVnaW4uIFRoZSBwcm9jZXNzUmVzb3VyY2UgZGVwZW5kZW5jeSBpcyB0byBhcHBlYXNlIEdyYWRsZSdzIGNvbXBpbGUgb3B0aW1pc2F0aW9uLiAgCiAgClBsZWFzZSBydW4gYC4vZ3JhZGxldyBjbGVhbiBidWlsZGAgdG8gd2l0bmVzcyB0aGUgbWFnaWMuCgo= readmeEtag: '"f87944e30aaa9634e4348b62623ec734da1a0bfa"' readmeLastModified: Fri, 27 Aug 2021 07:58:54 GMT repositoryId: 397819522 description: >- This showcases a simplistic Spring Boot project with an API-first design pattern created: '2021-08-19T04:49:13Z' updated: '2023-11-20T20:47:59Z' language: Java archived: false stars: 2 watchers: 1 forks: 2 owner: alwaysbemark logo: https://avatars.githubusercontent.com/u/56749912?v=4 license: MIT repoEtag: '"08a0513a2c39d0f1c2f9be1f30067b0b0224350313d35823f5362b6bc366709f"' repoLastModified: Mon, 20 Nov 2023 20:47:59 GMT foundInMaster: true category: - SDK - Server Implementations id: 40c23804d1858307199a3f4edf5f957a - source: openapi3 tags repository: https://github.com/jonarsli/flask-restapi v3: true repositoryMetadata: base64Readme: >- IyBGbGFzay1SRVNUQVBJCgpbIVtsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL2pvbmFyc2xpL2ZsYXNrLXJlc3RhcGkuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2pvbmFyc2xpL2ZsYXNrLXJlc3RhcGkvYmxvYi9tYXN0ZXIvTElDRU5TRSkKWyFbcHlwaV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9weXBpL3YvZmxhc2stcmVzdGFwaS5zdmcpXShodHRwczovL3B5cGkucHl0aG9uLm9yZy9weXBpL2ZsYXNrLXJlc3RhcGkpCgoKW0ZsYXNrLVJFU1RBUEkgZG9jdW1lbnRdKGh0dHBzOi8vam9uYXJzbGkuZ2l0aHViLmlvL2ZsYXNrLXJlc3RhcGkvKQoKRmxhc2stUkVTVEFQSSBpcyBhbiBleHRlbnNpb24gZm9yIEZsYXNrIHRoYXQgaXMgYSBkYXRhYmFzZS1hZ25vc3RpYyBmcmFtZXdvcmsgbGlicmFyeSBmb3IgY3JlYXRpbmcgUkVTVCBBUElzLiBJdCBpcyBhIGxpZ2h0d2VpZ2h0IGFic3RyYWN0aW9uIHRoYXQgd29ya3Mgd2l0aCB5b3VyIGV4aXN0aW5nIE9STS9saWJyYXJpZXMuCgpJdCB1c2UgcHlkYW50aWMgdG8gdmFsaWRhdGUgYW5kIHNlcmlhbGl6ZSBkYXRhLiBPcGVuQVBJIGRvY3VtZW50IGNhbiBiZSBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlZCB0aHJvdWdoIHRoZSBweXRob24gZGVjb3JhdG9yIGFuZCBpdCBzdXBwb3J0cyBzd2FnZ2VyIHVpIGRpc3BsYXkuCgpQeWRhbnRpYyBhcmUgdXNlZCB0byB2YWxpZGF0ZSBhbmQgc2VyaWFsaXplIHBhcmFtZXRlcnMuIEZvciBkZXRhaWxzLCBwbGVhc2UgcmVmZXIgdG8gdGhlIFtweWRhbnRpYyBkb2N1bWVudGF0aW9uXShodHRwczovL3B5ZGFudGljLWRvY3MuaGVscG1hbnVhbC5pby8pLgoKIyMgSW5zdGFsbGF0aW9uCmBgYGJhc2gKcGlwIGluc3RhbGwgZmxhc2stcmVzdGFwaQpgYGAKCiMjIEV4YW1wbGUKYGBgcHl0aG9uCmZyb20gZmxhc2sgaW1wb3J0IEZsYXNrCmZyb20gZmxhc2sudmlld3MgaW1wb3J0IE1ldGhvZFZpZXcKZnJvbSBweWRhbnRpYyBpbXBvcnQgQmFzZU1vZGVsCgpmcm9tIGZsYXNrX3Jlc3RhcGkgaW1wb3J0IEFwaSwgUmVxdWVzdFBhcmFtZXRlcnNUeXBlCgphcHAgPSBGbGFzayhfX25hbWVfXykKYXBpID0gQXBpKGFwcCkKCgpjbGFzcyBVc2VyR2V0U3BlYyhCYXNlTW9kZWwpOgogICAgbmFtZTogc3RyCgoKY2xhc3MgVXNlclJlc3BvbnNlU3BlYyhCYXNlTW9kZWwpOgogICAgaWQ6IGludAogICAgbmFtZTogc3RyCgoKY2xhc3MgVXNlcihNZXRob2RWaWV3KToKICAgIEBhcGkucXVlcnkoVXNlckdldFNwZWMpCiAgICBAYXBpLnJlc3BvbnNlKFVzZXJSZXNwb25zZVNwZWMpCiAgICBkZWYgZ2V0KHNlbGYsIHBhcmFtZXRlcnM6IFJlcXVlc3RQYXJhbWV0ZXJzVHlwZSk6CiAgICAgICAgIiIiR2V0IGEgdXNlciBuYW1lIGFuZCBpZCIiIgogICAgICAgIHVzZXJfbmFtZSA9IHBhcmFtZXRlcnMucXVlcnkubmFtZQogICAgICAgIHJldHVybiBVc2VyUmVzcG9uc2VTcGVjKGlkPTEsIG5hbWU9dXNlcl9uYW1lKQoKCmFwcC5hZGRfdXJsX3J1bGUoIi91c2VyIiwgdmlld19mdW5jPVVzZXIuYXNfdmlldygidXNlciIpKQoKYGBgCgojIyBTd2FnZ2VyIEFQSSBkb2NzCk5vdyBnbyB0byBodHRwOi8vbG9jYWxob3N0OjUwMDAvZG9jcwohW10oZG9jcy9pbWFnZXMvZXhhbXBsZS5wbmcp readmeEtag: '"0cdc24c8c4d86ae71888d45895dcd9020b8eca7b"' readmeLastModified: Wed, 03 May 2023 06:24:13 GMT repositoryId: 376734803 description: Flask-RESTAPI is an extension for validate and make OpenAPI docs. created: '2021-06-14T07:07:00Z' updated: '2024-05-09T06:19:21Z' language: Python archived: false stars: 2 watchers: 2 forks: 1 owner: jonarsli logo: https://avatars.githubusercontent.com/u/19283610?v=4 license: MIT repoEtag: '"abf077a26b36e57e75b1d9a498184d6038801c01dd70d2fdd4e0e5fda07448df"' repoLastModified: Thu, 09 May 2024 06:19:21 GMT foundInMaster: true category: Server Implementations id: 2c087c6f731350a28eb694b169467349 - source: - openapi3 tags - openapi31 tags repository: https://github.com/swaggerexpert/openapi-path-templating v3: true v3_1: true id: 6c56774d6631897980d8ac8149951454 repositoryMetadata: base64Readme: >- # openapi-path-templating

[![npmversion](https://badge.fury.io/js/openapi-path-templating.svg)](https://www.npmjs.com/package/openapi-path-templating)
[![npm](https://img.shields.io/npm/dm/openapi-path-templating.svg)](https://www.npmjs.com/package/openapi-path-templating)
[![Test workflow](https://github.com/swaggerexpert/openapi-path-templating/actions/workflows/test.yml/badge.svg)](https://github.com/swaggerexpert/openapi-path-templating/actions)
[![Dependabot enabled](https://img.shields.io/badge/Dependabot-enabled-blue.svg)](https://dependabot.com/)
[![try on RunKit](https://img.shields.io/badge/try%20on-RunKit-brightgreen.svg?style=flat)](https://npm.runkit.com/openapi-path-templating)
[![Tidelift](https://tidelift.com/badges/package/npm/openapi-path-templating)](https://tidelift.com/subscription/pkg/npm-openapi-path-templating?utm_source=npm-openapi-path-templating&utm_medium=referral&utm_campaign=readme)

[OpenAPI Path Templating](https://spec.openapis.org/oas/v3.1.1.html#path-templating) refers to the usage of template expressions, delimited by curly braces (`{}`), to mark a section of a URL path as replaceable using path parameters.
Each template expression in the path MUST correspond to a path parameter that is included in the [Path Item](https://spec.openapis.org/oas/v3.1.1.html#path-item-object) itself and/or in each of the Path Item's [Operations](https://spec.openapis.org/oas/v3.1.1.html#operation-object).
An exception is if the path item is empty, for example due to ACL constraints, matching path parameters are not required.

`openapi-path-templating` is a **parser**, **validator**, **resolver** and **matcher** for OpenAPI Path Templating,
which played a [foundational role](https://github.com/OAI/OpenAPI-Specification/pull/4244) in defining the official ANBF grammar for OpenAPI Path Templating.

It supports Path Templating defined in following OpenAPI specification versions:

- [OpenAPI 2.0](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/2.0.md#pathTemplating)
- [OpenAPI 3.0.0](https://spec.openapis.org/oas/v3.0.0.html#path-templating)
- [OpenAPI 3.0.1](https://spec.openapis.org/oas/v3.0.1.html#path-templating)
- [OpenAPI 3.0.2](https://spec.openapis.org/oas/v3.0.2.html#path-templating)
- [OpenAPI 3.0.3](https://spec.openapis.org/oas/v3.0.3.html#path-templating)
- [OpenAPI 3.0.4](https://spec.openapis.org/oas/v3.0.4.html#path-templating)
- [OpenAPI 3.1.0](https://spec.openapis.org/oas/v3.1.0.html#path-templating)
- [OpenAPI 3.1.1](https://spec.openapis.org/oas/v3.1.1.html#path-templating)


<table>
  <tr>
    <td align="right" valign="middle">
        <img src="https://cdn2.hubspot.net/hubfs/4008838/website/logos/logos_for_download/Tidelift_primary-shorthand-logo.png" alt="Tidelift" width="60" />
      </td>
      <td valign="middle">
        <a href="https://tidelift.com/subscription/pkg/npm-openapi-path-templating?utm_source=npm-openapi-path-templating&utm_medium=referral&utm_campaign=readme">
            Get professionally supported openapi-path-templating with Tidelift Subscription.
        </a>
      </td>
  </tr>
</table>

## Table of Contents

- [Getting started](#getting-started)
  - [Installation](#installation)
  - [Usage](#usage)
    - [Parsing](#parsing)
    - [Validation](#validation)
    - [Resolution](#resolution)
    - [Normalization](#normalization)
    - [Matching](#matching)
    - [Grammar](#grammar)
- [More about OpenAPI Path Templating](#more-about-openapi-path-templating)
- [License](#license)


## Getting started

### Installation

You can install `openapi-path-templating` using `npm`:

```sh
 $ npm install openapi-path-templating
```

### Usage

`openapi-path-templating` currently supports **parsing**, **validation**, **resolution** and **matching**.
Both parser and validator are based on a superset of [ABNF](https://www.rfc-editor.org/rfc/rfc5234) ([SABNF](https://github.com/ldthomas/apg-js2/blob/master/SABNF.md))
and use [apg-lite](https://github.com/ldthomas/apg-lite) parser generator.

#### Parsing

Parsing a Path Templating is as simple as importing the **parse** function
and calling it.

```js
import { parse } from 'openapi-path-templating';

const parseResult = parse('/pets/{petId}');
parseResult.result.success; // => true
```

**parseResult** variable has the following shape:

```
{
  result: {
    success: true,
    state: 101,
    stateName: 'MATCH',
    length: 13,
    matched: 13,
    maxMatched: 13,
    maxTreeDepth: 20,
    nodeHits: 232
  },
  ast: fnast {
    callbacks: [
      'path-template': [Function: pathTemplate],
      slash: [Function: slash],
      'path-literal': [Function: pathLiteral],
      'template-expression': [Function: templateExpression],
      'template-expression-param-name': [Function: templateExpressionParamName]
    ],
    init: [Function (anonymous)],
    ruleDefined: [Function (anonymous)],
    udtDefined: [Function (anonymous)],
    down: [Function (anonymous)],
    up: [Function (anonymous)],
    translate: [Function (anonymous)],
    setLength: [Function (anonymous)],
    getLength: [Function (anonymous)],
    toXml: [Function (anonymous)]
  }
}
```

###### Interpreting AST as list of entries

```js
import { parse } from 'openapi-path-templating';

const parseResult = parse('/pets/{petId}');
const parts = [];

parseResult.ast.translate(parts);
```

After running the above code, **parts** variable has the following shape:

```js
[
  [ 'path-template', '/pets/{petId}' ],
  [ 'slash', '/' ],
  [ 'path-literal', 'pets' ],
  [ 'slash', '/' ],
  [ 'template-expression', '{petId}' ],
  [ 'template-expression-param-name', 'petId' ]
]
```

###### Interpreting AST as XML

```js
import { parse } from 'openapi-path-templating';

const parseResult = parse('/pets/{petId}');
const xml = parseResult.ast.toXml();
```

After running the above code, **xml** variable has the following content:

```xml
<?xml version="1.0" encoding="utf-8"?>
<root nodes="6" characters="13">
  <!-- input string -->
  /pets/{petId}
  <node name="path-template" index="0" length="13">
    /pets/{petId}
    <node name="slash" index="0" length="1">
      /
    </node><!-- name="slash" -->
    <node name="path-literal" index="1" length="4">
      pets
    </node><!-- name="path-literal" -->
    <node name="slash" index="5" length="1">
      /
    </node><!-- name="slash" -->
    <node name="template-expression" index="6" length="7">
      {petId}
      <node name="template-expression-param-name" index="7" length="5">
        petId
      </node><!-- name="template-expression-param-name" -->
    </node><!-- name="template-expression" -->
  </node><!-- name="path-template" -->
</root>
```

> NOTE: AST can also be traversed in classical way using [depth first traversal](https://www.tutorialspoint.com/data_structures_algorithms/depth_first_traversal.htm). For more information about this option please refer to [apg-js](https://github.com/ldthomas/apg-js) and [apg-js-examples](https://github.com/ldthomas/apg-js-examples).

#### Validation

Validating a Path Templating is as simple as importing the **test** function and calling it.


```js
import { test } from 'openapi-path-templating';

test('/pets/{petId}'); // => true
test('/a{petId}'); // => true
test('/pets'); // => true
test('/pets', { strict: true }); // => false (doesn't contain any template-expression)
```

#### Resolution

Resolving a Path Templating is as simple as importing the **resolve** function and calling it.

```js
import { resolve } from 'openapi-path-templating';

resolve('/pets/{petId}', { petId: 3 }); // => "/pets/3"
```

Resolved Path Templating is automatically encoded using [encodeURIComponent](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/encodeURIComponent) function.
It is possible to provide a custom encoder.

```js
import { resolve } from 'openapi-path-templating';

resolve('/pets/{petId}', { petId: '/?#' }, {
  encoder: (component) => component, // no encoding
}); // => "/pets//?#"
```

#### Normalization

Normalization of OpenAPI Path Template is a process of transforming the path template into its normalized form.
Normalization is used to compare two path templates for equivalence and is based on **Syntax-Based Normalization** as defined in [RFC 3986, section 6.2.2](https://datatracker.ietf.org/doc/html/rfc3986#section-6.2.2).

##### Syntax-Based normalization

Syntax-Based normalization is a process of transforming the path template into its normalized form using
a fixed set of normalizers in following order:

- [Case normalization](#case-normalization)
- [Percent-Encoding normalization](#percent-encoding-normalization)
- [Path segment normalization](#path-segment-normalization)

```js
import { normalize } from '@swaggerexpert/openapi-path-templating';

normalize('/api/{userId}/profile/../account/%41ccount'); // => '/api/{userId}/account/Account'
```

Syntax-Based normalization is a default normalization strategy used by this library.


###### No normalization

Noop normalizer that acts as identity function. Performs no transformations on the path template
and returns it as is.

```js
import { identityNormalizer } from '@swaggerexpert/openapi-path-templating';

identityNormalizer('/API/%2faPi/%7bsection%7d/./../profile'); // => '/API/%2faPi/%7bsection%7d/./../profile'
```

###### Case normalization

For all path templates, the hexadecimal digits within a percent-encoding
triplet (e.g., `"%3a"` versus `"%3A"`) are case-insensitive and therefore
should be normalized to use uppercase letters for the digits A-F.

```js
import { caseNormalizer } from '@swaggerexpert/openapi-path-templating';

caseNormalizer('/api/{userId}/profile/%7bsection%7d'); // => '/api/{userId}/profile/%7Bsection%7D'
```

###### Percent-Encoding normalization

The percent-encoding mechanism is a frequent source of
variance among otherwise identical path templates. In addition to the case
normalization issue noted above, some path template producers percent-encode
octets that do not require percent-encoding, resulting in path templates that
are equivalent to their non-encoded counterparts. These path templates should
be normalized by decoding any percent-encoded octet that corresponds
to an unreserved character.

```js
import { percentEndingNormalizer } from '@swaggerexpert/openapi-path-templating';

percentEndingNormalizer('/api/%7BuserId%7D/profile/%41ccount/{account}'); // => '/api/%7BuserId%7D/profile/Account/{account}'
```

###### Path segment normalization

The complete path segments `"."` and `".."` are intended for use
within path templates and are removed as part of
the reference resolution process. Path template segment normalizer removes
dot-segments by applying the [remove_dot_segments](https://datatracker.ietf.org/doc/html/rfc3986#section-5.2.4) algorithm to the path template.

```js
import { pathSegmentNormalizer } from '@swaggerexpert/openapi-path-templating';

pathSegmentNormalizer('/api/{userId}/./profile/../account'); // => '/api/{userId}/account'
```

**NOTE**: note that all normalizers acts as identity functions when invalid path template is provided.

#### Matching

Path templating matching in OpenAPI prioritizes concrete paths over parameterized ones,
treats paths with identical structures but different parameter names as invalid,
and considers paths with overlapping patterns that could match the same request as potentially ambiguous.

##### Equivalence

`isIdentical`

Determines whether two path templates are structurally identical, meaning they have the same sequence
of literals and template expressions, regardless of template expression names. In the OpenAPI context,
such identical paths are considered invalid due to potential conflicts in routing.

Testing path templates for equality is normally based on pair comparison of the
characters that make up the strings, starting from the first and
proceeding until both strings are exhausted and all characters are
found to be equal, until a pair of characters compares unequal, or
until one of the strings is exhausted before the other. More in [RFC 3986, section 6.2.1](https://datatracker.ietf.org/doc/html/rfc3986#section-6.2.1).

```js
import { isIdentical } from 'openapi-path-templating';

isIdentical('/pets/{petId}', '/pets/{name}'); // => true
isIdentical('/pets/{petId}', '/animals/{name}'); // => false
```

By default, extensive [normalization](#normalization) prior to comparison is applied to produce a normalized form of the path template.

```js
import { isIdentical, normalize, identityNormalizer } from 'openapi-path-templating';

// default behavior
isIdentical('/API/%2faPi/%7bsection%7d/./../profile', '/API/%2FaPi/profile'); // => true
isIdentical('/api/{userId}/account/Account', '/api/{userId}/profile/../account/%41ccount'); // => true
// equivalent to above
isIdentical('/API/%2faPi/%7bsection%7d/./../profile', '/API/%2FaPi/profile', { normalizer: normalize }); // => true
isIdentical('/api/{userId}/account/Account', '/api/{userId}/profile/../account/%41ccount', { normalizer: normalize }); // => true
// no normalization
isIdentical('/API/%2faPi/%7bsection%7d/./../profile', '/API/%2FaPi/profile', { normalizer: identityNormalizer }); // => false
isIdentical('/api/{userId}/account/Account', '/api/{userId}/profile/../account/%41ccount', { normalizer: identityNormalizer }); // => false
```

#### Grammar

New grammar instance can be created in following way:

```js
import { Grammar } from 'openapi-path-templating';

const grammar = new Grammar();
```

To obtain original ABNF (SABNF) grammar as a string:

```js
import { Grammar } from 'openapi-path-templating';

const grammar = new Grammar();

grammar.toString();
// or
String(grammar);
```

## More about OpenAPI Path Templating

The Path Templating is defined by the following [ABNF](https://tools.ietf.org/html/rfc5234) syntax

```abnf
; OpenAPI Path Templating ABNF syntax
; variant of https://datatracker.ietf.org/doc/html/rfc3986#section-3.3
path-template                  = slash *( path-segment slash ) [ path-segment ]
path-segment                   = 1*( path-literal / template-expression )
slash                          = "/"
path-literal                   = 1*pchar
template-expression            = "{" template-expression-param-name "}"
template-expression-param-name = 1*( %x00-7A / %x7C / %x7E-10FFFF ) ; every UTF8 character except { and } (from OpenAPI)

; https://datatracker.ietf.org/doc/html/rfc3986#section-3.3
pchar               = unreserved / pct-encoded / sub-delims / ":" / "@"
unreserved          = ALPHA / DIGIT / "-" / "." / "_" / "~"
                    ; https://datatracker.ietf.org/doc/html/rfc3986#section-2.3
pct-encoded         = "%" HEXDIG HEXDIG
                    ; https://datatracker.ietf.org/doc/html/rfc3986#section-2.1
sub-delims          = "!" / "$" / "&" / "'" / "(" / ")"
                    / "*" / "+" / "," / ";" / "="
                    ; https://datatracker.ietf.org/doc/html/rfc3986#section-2.2

; https://datatracker.ietf.org/doc/html/rfc5234#appendix-B.1
ALPHA               = %x41-5A / %x61-7A   ; A-Z / a-z
DIGIT               = %x30-39            ; 0-9
HEXDIG              = DIGIT / "A" / "B" / "C" / "D" / "E" / "F"
```

## License

`openapi-path-templating` is licensed under [Apache 2.0 license](https://github.com/swaggerexpert/openapi-path-templating/blob/main/LICENSE).
`openapi-path-templating` comes with an explicit [NOTICE](https://github.com/swaggerexpert/openapi-path-templating/blob/main/NOTICE) file
containing additional legal notices and information.
 readmeEtag: '"dc6d56d650ac95f49d3c4e5197334eb0acf67597"' readmeLastModified: Sun, 09 Mar 2025 21:27:25 GMT repositoryId: 730126319 description: OpenAPI Path Templating parser, validator and resolver. created: '2023-12-11T09:07:36Z' updated: '2026-01-21T18:45:37Z' language: JavaScript archived: false stars: 3 watchers: 1 forks: 1 owner: swaggerexpert logo: https://avatars.githubusercontent.com/u/172408630?v=4 license: Apache-2.0 repoEtag: '"bef3decab4fc1cc96d2fd22c51279df6c823db16e41bd32a9afc4f71e70c27af"' repoLastModified: Wed, 21 Jan 2026 18:45:37 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/senatov/spring-boot-primerfaces-sheduler v3: true repositoryMetadata: base64Readme: >- IyMgU3ByaW5nIEJvb3QgSW50ZWdyYXRpb24gZXhhbXBsZQoKLSBleGFtcGxlIGNvZGUuCi0gYFRoZSBQcm9qZWN0IGJ1aWxkcyB1bmRlciBPcmFjbGUgSkRLMTcgR2VuZXJhbC1BdmFpbGFiaWxpdHkgUmVsZWFzZWAKCiMjIFRhZ3MKCi0tLQoKPiBPcmFjbGUgMTE8YnI+Cj4gSlBBL0hpYmVybmF0ZSA8YnI+Cj4gUG9zdGdyZXM6bGF0ZXN0IHVuZGVyIERvY2tlcjxicj4KPiBTcHJpbmctQm9vdCAyLjYuMjxicj4KPiBKU0YyIE1vamFycmEgPGJyPgo+IFByaW1lRmFjZXMgMTEuMC4wPGJyPgoKLS0tCgotIGltcGxlbWVudGF0aW9uIG9mIEJhc2lzIFByaW1lZmFjZXMgU2NoZWR1bGVyLgotIENvbXBvbmVudHM6IE9yYWNsZSAxMSwgSlBBL0hpYmVybmF0ZSwgRG9ja2VyICsgUG9zdGdyZXMsIFNwcmluZy1Cb290IDIuNi4yLCBKU0YgMi40LjAgTW9qYXJyYSwgUHJpbWVGYWNlcwogIDExLjAuMCwgT3BlbkFQSSx+U3ByaW5nIFNlY3VyaXR5fgotIFVSTHM6IFtodHRwOi8vbG9jYWxob3N0OjE4MDg3L3VpL3Jlc2VydmF0aW9uLnhodG1sXShodHRwOi8vbG9jYWxob3N0OjE4MDg3L3VpL3Jlc2VydmF0aW9uLnhodG1sKQotIEdJVEggSW5mbyBmcm9tIFJlc3QgV1M6IFtodHRwOi8vbG9jYWxob3N0OjE4MDg3L3ZlcnNpb25dKGh0dHA6Ly9sb2NhbGhvc3Q6MTgwODcvdmVyc2lvbikKCi0tLQoKKipbU3RhcnQgTGlua10oaHR0cDovL2xvY2FsaG9zdDoxODA4Ny91aS9yZXNlcnZhdGlvbi54aHRtbCkqKi4KCmNvbW1vbiBwcm9ncmFtbSB0ZXN0IChEQiBjb25uZWN0LCBCb290LCBKU0YgY29ubmVjdGlvbikgaXMgaGVscGZ1bCBvbiA8YnI+CnJlZmFjdG9yaW5nIGFuZCBwb20gdmVyc2lvbiBjaGFuZ2luZwoKKipbRU1haWw6amF2YWVudHdpY2tsZXJAZ21haWwuY29tXShtYWlsdG86Ly9qYXZhZW50d2lja2xlckBnbWFpbC5jb20pKioKCi0tLQoKIVtpbWFnZTFdKGRvYy9yZXNlcnZhdGlvbi5wbmcgIkltYWdlICMxIikKIVtpbWFnZTFdKGRvYy9saXN0LnBuZyAiSW1hZ2UgIzIiKQohW2ltYWdlMV0oZG9jL2Nld2F0ZS5wbmcgIkltYWdlICMzIikKCkFsbCBVUklzIGFyZSByZWxhdGl2ZSB0byAqaHR0cDovL2xvY2FsaG9zdDoxODA4NyoKCk1ldGhvZCB8IEhUVFAgcmVxdWVzdCB8IERlc2NyaXB0aW9uCi0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLQpbKip2ZXJzaW9uSW5mb3JtYXRpb24qKl0oVmVyc2lvblJlc3RDb250cm9sbGVyQXBpLm1kI3ZlcnNpb25JbmZvcm1hdGlvbikgfCAqKkdFVCoqIC92ZXJzaW9uLyB8Cgo8YSBuYW1lPSJ2ZXJzaW9uSW5mb3JtYXRpb24iPjwvYT4KCiMgKip2ZXJzaW9uSW5mb3JtYXRpb24qKgoKPiBTdHJpbmcgdmVyc2lvbkluZm9ybWF0aW9uKCkKCiMjIyBFeGFtcGxlIENhbGwgT3BlbiBBUEkgU2VydmljZToKCmBgYGphdmEKLy8gSW1wb3J0IGNsYXNzZXM6CmltcG9ydCBvcmcub3BlbmFwaXRvb2xzLmNsaWVudC5BcGlDbGllbnQ7CmltcG9ydCBvcmcub3BlbmFwaXRvb2xzLmNsaWVudC5BcGlFeGNlcHRpb247CmltcG9ydCBvcmcub3BlbmFwaXRvb2xzLmNsaWVudC5Db25maWd1cmF0aW9uOwppbXBvcnQgb3JnLm9wZW5hcGl0b29scy5jbGllbnQubW9kZWxzLio7CmltcG9ydCBvcmcub3BlbmFwaXRvb2xzLmNsaWVudC5hcGkuVmVyc2lvblJlc3RDb250cm9sbGVyQXBpOwoKcHVibGljIGNsYXNzIEV4YW1wbGUgewogIHB1YmxpYyBzdGF0aWMgdm9pZCBtYWluKFN0cmluZ1tdIGFyZ3MpIHsKICAgIEFwaUNsaWVudCBkZWZhdWx0Q2xpZW50ID0gQ29uZmlndXJhdGlvbi5nZXREZWZhdWx0QXBpQ2xpZW50KCk7CiAgICBkZWZhdWx0Q2xpZW50LnNldEJhc2VQYXRoKCJodHRwOi8vbG9jYWxob3N0OjE4MDg3Iik7CgogICAgVmVyc2lvblJlc3RDb250cm9sbGVyQXBpIGFwaUluc3RhbmNlID0gbmV3IFZlcnNpb25SZXN0Q29udHJvbGxlckFwaShkZWZhdWx0Q2xpZW50KTsKICAgIHRyeSB7CiAgICAgIFN0cmluZyByZXN1bHQgPSBhcGlJbnN0YW5jZS52ZXJzaW9uSW5mb3JtYXRpb24oKTsKICAgICAgU3lzdGVtLm91dC5wcmludGxuKHJlc3VsdCk7CiAgICB9IGNhdGNoIChBcGlFeGNlcHRpb24gZSkgewogICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIkV4Y2VwdGlvbiB3aGVuIGNhbGxpbmcgVmVyc2lvblJlc3RDb250cm9sbGVyQXBpI3ZlcnNpb25JbmZvcm1hdGlvbiIpOwogICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIlN0YXR1cyBjb2RlOiAiICsgZS5nZXRDb2RlKCkpOwogICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIlJlYXNvbjogIiArIGUuZ2V0UmVzcG9uc2VCb2R5KCkpOwogICAgICBTeXN0ZW0uZXJyLnByaW50bG4oIlJlc3BvbnNlIGhlYWRlcnM6ICIgKyBlLmdldFJlc3BvbnNlSGVhZGVycygpKTsKICAgICAgZS5wcmludFN0YWNrVHJhY2UoKTsKICAgIH0KICB9Cn0KYGBgCgojIyMgUGFyYW1ldGVycwoKVGhpcyBlbmRwb2ludCBkb2VzIG5vdCBuZWVkIGFueSBwYXJhbWV0ZXIuCgojIyMgUmV0dXJuIHR5cGUKCioqU3RyaW5nKioKCiMjIyBBdXRob3JpemF0aW9uCgpObyBhdXRob3JpemF0aW9uIHJlcXVpcmVkCgojIyMgSFRUUCByZXF1ZXN0IGhlYWRlcnMKCi0gKipDb250ZW50LVR5cGUqKjogTm90IGRlZmluZWQKLSAqKkFjY2VwdCoqOiBhcHBsaWNhdGlvbi9qc29uCgojIyMgSFRUUCByZXNwb25zZSBkZXRhaWxzCgp8IFN0YXR1cyBjb2RlIHwgRGVzY3JpcHRpb24gfCBSZXNwb25zZSBoZWFkZXJzIHwKfC0tLS0tLS0tLS0tLS18LS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS18CioqMjAwKiogfCBPSyB8ICAtICB8 readmeEtag: '"f2ae30b5c57ac8eae09d49c5f9f71454b7618b45"' readmeLastModified: Mon, 19 Sep 2022 22:41:14 GMT repositoryId: 140123000 description: Scheduler example implementation(Spring boot 2.6, JSF 2.4, Primefaces 10) created: '2018-07-07T23:15:09Z' updated: '2022-06-16T14:47:08Z' language: Java archived: false stars: 2 watchers: 2 forks: 2 owner: senatov logo: https://avatars.githubusercontent.com/u/4016842?v=4 repoEtag: '"8b2b3ca5231fd47e74c0be802649009c26132dd745f8f25ed1141717668df404"' repoLastModified: Thu, 16 Jun 2022 14:47:08 GMT foundInMaster: true category: - Low-level Tooling - SDK id: b76f2d705b59816aaf9172e4780d7e04 - source: openapi3 tags repository: https://github.com/danielso2007/virtuallibraryapi v3: true repositoryMetadata: base64Readme: >- [![Actions Status](https://github.com/danielso2007/virtualLibraryAPI/workflows/virtualLibraryAPI/badge.svg)](https://github.com/danielso2007/virtualLibraryAPI/actions)
[![codecov](https://codecov.io/gh/danielso2007/virtualLibraryAPI/branch/develop/graph/badge.svg)](https://codecov.io/gh/danielso2007/virtualLibraryAPI)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=danielso2007_virtualLibraryAPI&metric=alert_status)](https://sonarcloud.io/dashboard?id=danielso2007_virtualLibraryAPI)
![GitHub package version](https://img.shields.io/github/package-json/v/danielso2007/virtualLibraryAPI.svg)
[![GitHub pull requests](https://img.shields.io/github/issues-pr-raw/danielso2007/virtualLibraryAPI.svg)](https://github.com/danielso2007/virtualLibraryAPI/pulls)
[![GitHub issues](https://img.shields.io/github/issues/danielso2007/virtualLibraryAPI.svg)](https://github.com/danielso2007/virtualLibraryAPI/issues?q=is%3Aopen+is%3Aissue)
![GitHub last commit](https://img.shields.io/github/last-commit/danielso2007/virtualLibraryAPI.svg)
[![GitHub issue/pull request author](https://img.shields.io/github/issues/detail/u/danielso2007/virtualLibraryAPI/1.svg)](https://github.com/danielso2007/virtualLibraryAPI/pulls)
![GitHub contributors](https://img.shields.io/github/contributors/danielso2007/virtualLibraryAPI.svg)
![GitHub top language](https://img.shields.io/github/languages/top/danielso2007/virtualLibraryAPI.svg)
[![GitHub](https://img.shields.io/github/license/danielso2007/virtualLibraryAPI.svg)](https://github.com/danielso2007/virtualLibraryAPI)
[![GitHub All Releases](https://img.shields.io/github/downloads/danielso2007/virtualLibraryAPI/total.svg)](https://github.com/danielso2007/virtualLibraryAPI/archive/master.zip)
[![Conventional Commits](https://img.shields.io/badge/Conventional%20Commits-1.0.0-yellow.svg)](https://conventionalcommits.org)

# Virtual Library API

Virtual library API. This project is used to study REST Spring Boot projects.

Este projeto usa o [Commons REST API](https://github.com/danielso2007/commons-rest-api) para dependência para as chamadas e padrões REST. Esse projeto foi feito por mim para abstrair todas a chamadas e retornos padrões do princípio RESTFull. Será necessário instalar o projeto localmente ou usar o [package do github](https://github.com/danielso2007/commons-rest-api/packages).

## Getting Started

These instructions will get you a copy of the project up and running on your local machine for development and testing purposes.

### Prerequisites

Utilizar o ambiente linux e ter o mavem e o java 11 para a compilação e execução do projeto.
Este projeto utiliza o MongoDb como base de dados, para ter um banco de dados para esse projeto, siga o projeto
[dockerMongoDB](https://github.com/danielso2007/dockerMongoDB) usando o [Docker](https://www.docker.com/).

### Installing

Recomendado instalar o [sdkman](https://sdkman.io/) que é uma ferramenta para gerenciar versões paralelas de vários kits de desenvolvimento de software na maioria dos sistemas baseados em Unix.

Com o sdkman, instale o java:
```
sdk list java
sdk install java 11.0.5.j9-adpt
```

Instalando o Maven:
```
sdk install maven
```

## Test

O projeto está configurado para não executar os testes quando construído para desenvolvimento. Para executar os testes, execute o comando abaixo:

`mvn clean test -Ptest`

Os testes são executados normalmente quando construído para **produção** e na criação da imagem **Docker**. Há no arquivo do `pom.xml` essa configuração, que pode ser modificada a qualquer momento do desenvolvimento. Inicialmente, para o desenvolvimento, o desenvolvedor pode executar seus testes ao seguir o padrão TDD pela própria IDE.

### Code test coverage

É utilizado o [EclEmma Jacoco](https://www.eclemma.org/jacoco/) para verificação de cobertura de código. Para executar a cobertura:
```
mvn clean test verify -Ptest jacoco:report
```
Na pasta target, é gerado um "site" mostrando toda a cobertura de código. Este projeto utiliza o _Action_ para os testes e build da aplicação. Ao final do teste, é enviado para [codecov.io](https://codecov.io) todo o relatório. Para verificar a cobertura de teste deste código, acesse [codecov.io/gh/danielso2007]([codecov.io](https://codecov.io/gh/danielso2007/virtualLibraryAPI)).

## Running with docker

Há um perfil no `pom.xml` para a criação da imagem do projeto. Ao executar `mvn clean package -Pdocker`, será realizado o teste e criado o arquivo `Dockerfile` na pasta `target`, criando a imagem `virtualLibraryAPI:<project.version>`.
Para ver a imagem criada, digite no terminal o comando `docker images`.

Para "rodar" a imagem, execute:
```
docker run -p 8080:8080 --name swapi  virtualLibraryAPI:<project.version>
```

### Code quality

O sonar é usado para analisar a qualidade do código. Você pode iniciar um servidor Sonar local (acessível em http: // localhost: 9001) com:

```
docker-compose -f src/main/docker/sonar.yml up -d
```

Você pode executar uma análise do Sonar usando o scanner de sonar ou usando o plugin maven.

Em seguida, execute uma análise do sonar:

```
mvn -Ptest clean verify jacoco:report sonar:sonar
```

Se você precisar executar novamente a fase do Sonar, especifique pelo menos a fase de inicialização, já que as propriedades do Sonar são carregadas do arquivo sonar-project.properties.

```
mvn clean test verify -Ptest jacoco:report
mvn initialize sonar:sonar
```

### Documentação da API - swagger (OpenAPI 3):

A aplicação usa o swagger para a exibição da documentação da API. Para verificar, acesse os links [swagger-ui](http://localhost:8080/swagger-ui.html) e [api-docs](http://localhost:8080/api-docs). Documentação [swagger.io](https://swagger.io/docs/open-source-tools/swagger-ui/usage/configuration/) e [springdoc.org](https://springdoc.org/).

Mais exemplos: [documenting-spring-boot-rest-api-springdoc-openapi-3](https://www.dariawan.com/tutorials/spring/documenting-spring-boot-rest-api-springdoc-openapi-3/)

Ou adicione o caminho abaixo:

```
http://localhost:8080/swagger-ui.html

http://localhost:8080/api-docs
```

### Teste o endereço docker (login: admin / password: admin):

```
http://localhost:8080/api/v1/books

http://localhost:8080/api/v1/ratings
```

Inicialmente só é criada a imagem da API. Posteriormente mostrarei como executar o `docker-compose` criado no build, para a execução completa da api com o banco de dados MongoDb.

## Running with docker-compose

Ao executar o maven `mvn clean package -Pdocker`, é gerado o `Dockerfile` e também o `docker-compose.yml`. Com o docker-compose é possível inicar a aplicação já com um container docker com Mongo. Inicialmente esse banco estará vazio.

Execute esse comando dentro da pasta `target`:
```
docker-compose up -d
```

Será iniciado os containers da api e do mongo. Acessando o endereço `http://localhost:8080/api/v1/books`, será retornado uma lista vazia.

Para parar os containers, execute:
```
docker-compose stop
```

Pelo `pom.xml` é possível configurar a criação do arquivo `docker-compose`. Inicialmente a porta do container mongo está exposta, mas é só modificar o arquivo `docker-compose` e remover, pois a comunicação entre a api e o banco é via `network` interno entre os containers.

## Running (No docker)

Dentro da pasta do projeto, execute:
```
mvn clean package install
```
Após compilar o projeto, dentro da pasta `target`, inicie o projeto (x.x.x é a versão atual do projeto):
```
cd target
nohup java -jar virtuallibraryapi-x.x.x.jar & tail -f nohup.out

```

### Stopping project

Para encerrar o projeto, execute o comando abaixo:
```
ps -ef | grep virtuallibraryapi
```
O comando acima exibirá o id do processo, após ver o id do processo, execute:
```
kill -9 <id_processo>
```

### Scripts de execução e encerramento do projeto

Dentro da pasta `target` é criado os arquivos `start.sh` e `stop.sh`. Esses arquivos não são executáveis e precisam ser modificados:
```
cd target
chmod a+x start.sh stop.sh
```

Para iniciar o projeto, execute (dentro da pasta `target`):
```
./start.sh
```
Para encerrar o projeto, execute (dentro da pasta `target`):
```
./stop.sh
```

## Deployment

Nada será necessário, o projeto é executado como `Fat jar`. Mas para produção e homologação, é possível usar as imagens geradas pelo comando: `mvn clean package -Pdocker`. Muito importante para o uso do CI.


# Banco de dados

O projeto usa o mongo para banco de dados. Recomendo que utilize o projeto [danielso2007/dockerMongoDB](https://github.com/danielso2007/dockerMongoDB) para executar o mongo em docker. Há dois arquivos `book.js` e `rating.js` que podem ser usados para importar dados no banco.

Segue a configuração padrão do banco, mas cada profile do `POM` tem seu database.
- host: localhost
- port: 27017
- database: virtuallibraryapi
- username: root
- password: 112358

Execute o arquivo `./mongoimport.sh` e informe:

- Importando arquivo para o MongoDB...
- Infome o nome do banco de dados:
- `virtuallibraryapi`
- Infome collection:
- `book`
- Infome o nome do arquivo JSON:
- `book.json`

Depois para o `rating`:

- Importando arquivo para o MongoDB...
- Infome o nome do banco de dados:
- `virtuallibraryapi`
- Infome collection:
- `rating`
- Infome o nome do arquivo JSON:
- `rating.json`

## Built With

* [Java](https://www.oracle.com/br/java/)
* [Spring Boot](https://spring.io/projects/spring-boot)
* [Maven](https://maven.apache.org/)
* [Undertow - Servidor Web](http://undertow.io/)
* [Modelmapper - Simple, Intelligent, Object Mapping](http://modelmapper.org/)
* [Project Lombok](https://projectlombok.org/)
* [QueryDsl - type-safe SQL-like queries - fluent API](http://www.querydsl.com/)
* [Swagger](https://swagger.io/)
* [Standard-version](https://github.com/conventional-changelog/standard-version)

## Contributing

Please read [CONTRIBUTING.md](CONTRIBUTING.md) for details on our code of conduct, and the process for submitting pull requests to us.

## Versioning

Usamos [SemVer](http://semver.org/) para versionar. Para as versões disponíveis, consulte as [tags neste repositório](https://github.com/danielso2007/virtualLibraryAPI/releases). 

## Authors

* **Daniel Oliveira** - *Initial work* - [danielso2007](https://github.com/danielso2007)

See also the list of [contributors](https://github.com/danielso2007/virtualLibraryAPI/graphs/contributors) who participated in this project.
 readmeEtag: '"34ff87d8c3724949b6189e9bc7882cacf4ab2d2a"' readmeLastModified: Thu, 12 Aug 2021 21:25:41 GMT repositoryId: 236433782 description: Study of Java REST and Spring Boot projects. created: '2020-01-27T06:42:47Z' updated: '2021-08-12T21:25:45Z' language: Java archived: false stars: 2 watchers: 1 forks: 2 owner: danielso2007 logo: https://avatars.githubusercontent.com/u/1641606?v=4 repoEtag: '"75191c8e7734f8829833a2041e0ded36d6fae616305ef565ae6fa9afeeef518a"' repoLastModified: Thu, 12 Aug 2021 21:25:45 GMT foundInMaster: true category: Code Generators id: 09b5ac67cbb6bc0277c4fa1345238597 - source: openapi3 tags repository: https://github.com/joaoduarte19/openapi4d v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQXBpNEQKKipPcGVuQXBpNEQqKiBpcyBhbiBbT3BlbkFQSSAzLjBdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjAuMy5tZCkgZ2VuZXJhdG9yIGZvciBEZWxwaGkuCg== readmeEtag: '"6f4a7c9d416b89e0248aa85bd78a7aadf9a1e962"' readmeLastModified: Thu, 15 Jul 2021 22:31:21 GMT repositoryId: 384214637 description: OpenAPI generator for Delphi created: '2021-07-08T18:24:09Z' updated: '2025-10-31T18:31:40Z' language: Pascal archived: false stars: 3 watchers: 1 forks: 2 owner: joaoduarte19 logo: https://avatars.githubusercontent.com/u/13373632?v=4 license: Apache-2.0 repoEtag: '"7cc00b81d80cebc2a720a1861dc29238a3e0ef6bd828d13fc3cf521ba8b031cd"' repoLastModified: Fri, 31 Oct 2025 18:31:40 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 9c6cbdff33442ef0c97c06e4df283055 - source: openapi3 tags repository: https://github.com/simonit/vcita-client-java-sdk v3: true repositoryMetadata: base64Readme: >- # vcita-client-java-sdk

client_api API
- API version: v2.0

No description provided (generated by Openapi Generator https://github.com/openapitools/openapi-generator)


*Automatically generated by the [OpenAPI Generator](https://openapi-generator.tech)*


## Requirements

Building the API client library requires:
1. Java 1.8+
2. Maven (3.8.3+)/Gradle (7.2+)

## Installation

To install the API client library to your local Maven repository, simply execute:

```shell
mvn clean install
```

To deploy it to a remote Maven repository instead, configure the settings of the repository and execute:

```shell
mvn clean deploy
```

Refer to the [OSSRH Guide](http://central.sonatype.org/pages/ossrh-guide.html) for more information.

### Maven users

Add this dependency to your project's POM:

```xml
<dependency>
  <groupId>com.vcita.client</groupId>
  <artifactId>vcita-client-java-sdk</artifactId>
  <version>v2.0</version>
  <scope>compile</scope>
</dependency>
```

### Gradle users

Add this dependency to your project's build file:

```groovy
  repositories {
    mavenCentral()     // Needed if the 'vcita-client-java-sdk' jar has been published to maven central.
    mavenLocal()       // Needed if the 'vcita-client-java-sdk' jar has been published to the local maven repo.
  }

  dependencies {
     implementation "com.vcita.client:vcita-client-java-sdk:v2.0"
  }
```

### Others

At first generate the JAR by executing:

```shell
mvn clean package
```

Then manually install the following JARs:

* `target/vcita-client-java-sdk-v2.0.jar`
* `target/lib/*.jar`

## Getting Started

Please follow the [installation](#installation) instruction and execute the following Java code:

```java

// Import classes:
import com.vcita.client.client.ApiClient;
import com.vcita.client.client.ApiException;
import com.vcita.client.client.Configuration;
import com.vcita.client.client.models.*;
import com.vcita.client.client.api.AuthenticationApi;

public class Example {
  public static void main(String[] args) {
    ApiClient defaultClient = Configuration.getDefaultApiClient();
    defaultClient.setBasePath("http://api.vcita.biz/client_api/v1");

    AuthenticationApi apiInstance = new AuthenticationApi(defaultClient);
    String businessUid = "businessUid_example"; // String | 
    try {
      InlineResponse200 result = apiInstance.portalsBusinessUidAuthenticationsLogoutPost(businessUid);
      System.out.println(result);
    } catch (ApiException e) {
      System.err.println("Exception when calling AuthenticationApi#portalsBusinessUidAuthenticationsLogoutPost");
      System.err.println("Status code: " + e.getCode());
      System.err.println("Reason: " + e.getResponseBody());
      System.err.println("Response headers: " + e.getResponseHeaders());
      e.printStackTrace();
    }
  }
}

```

## Documentation for API Endpoints

All URIs are relative to *http://api.vcita.biz/client_api/v1*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AuthenticationApi* | [**portalsBusinessUidAuthenticationsLogoutPost**](docs/AuthenticationApi.md#portalsBusinessUidAuthenticationsLogoutPost) | **POST** /portals/{business_uid}/authentications/logout | 
*AuthenticationApi* | [**portalsBusinessUidAuthenticationsSendCodePost**](docs/AuthenticationApi.md#portalsBusinessUidAuthenticationsSendCodePost) | **POST** /portals/{business_uid}/authentications/send_code | 
*AuthenticationApi* | [**portalsBusinessUidAuthenticationsVerifyCodePost**](docs/AuthenticationApi.md#portalsBusinessUidAuthenticationsVerifyCodePost) | **POST** /portals/{business_uid}/authentications/verify_code | 
*ClientApi* | [**portalsBusinessUidClientGet**](docs/ClientApi.md#portalsBusinessUidClientGet) | **GET** /portals/{business_uid}/client | 
*FormsApi* | [**portalsBusinessUidContactGetFormGet**](docs/FormsApi.md#portalsBusinessUidContactGetFormGet) | **GET** /portals/{business_uid}/contact/get_form | Get Leave Details Form
*FormsApi* | [**portalsBusinessUidContactSubmitPost**](docs/FormsApi.md#portalsBusinessUidContactSubmitPost) | **POST** /portals/{business_uid}/contact/submit | Submits Leave Details Form
*FormsApi* | [**portalsBusinessUidShareDocumentsFormGetFormGet**](docs/FormsApi.md#portalsBusinessUidShareDocumentsFormGetFormGet) | **GET** /portals/{business_uid}/share_documents_form/get_form | Get Share Documents Form
*FormsApi* | [**portalsBusinessUidShareDocumentsFormSubmitPost**](docs/FormsApi.md#portalsBusinessUidShareDocumentsFormSubmitPost) | **POST** /portals/{business_uid}/share_documents_form/submit | Submits Share Documents Form


## Documentation for Models

 - [Client](docs/Client.md)
 - [Data](docs/Data.md)
 - [Data1](docs/Data1.md)
 - [Data2](docs/Data2.md)
 - [Data3](docs/Data3.md)
 - [Field](docs/Field.md)
 - [InlineObject](docs/InlineObject.md)
 - [InlineObject1](docs/InlineObject1.md)
 - [InlineObject2](docs/InlineObject2.md)
 - [InlineObject3](docs/InlineObject3.md)
 - [InlineResponse200](docs/InlineResponse200.md)
 - [InlineResponse2001](docs/InlineResponse2001.md)
 - [InlineResponse401](docs/InlineResponse401.md)
 - [Matter](docs/Matter.md)
 - [PortalsBusinessUidAuthenticationsSendCodePost200Response](docs/PortalsBusinessUidAuthenticationsSendCodePost200Response.md)
 - [PortalsBusinessUidClientGet200Response](docs/PortalsBusinessUidClientGet200Response.md)
 - [PortalsBusinessUidContactGetFormGet200Response](docs/PortalsBusinessUidContactGetFormGet200Response.md)
 - [PortalsBusinessUidContactSubmitFields](docs/PortalsBusinessUidContactSubmitFields.md)
 - [PortalsBusinessUidContactSubmitFormData](docs/PortalsBusinessUidContactSubmitFormData.md)
 - [PortalsBusinessUidContactSubmitPost200Response](docs/PortalsBusinessUidContactSubmitPost200Response.md)
 - [PortalsBusinessUidContactSubmitSourceData](docs/PortalsBusinessUidContactSubmitSourceData.md)
 - [PortalsBusinessUidShareDocumentsFormGetFormGet200Response](docs/PortalsBusinessUidShareDocumentsFormGetFormGet200Response.md)
 - [PortalsBusinessUidShareDocumentsFormSubmitFields](docs/PortalsBusinessUidShareDocumentsFormSubmitFields.md)
 - [PortalsBusinessUidShareDocumentsFormSubmitFormData](docs/PortalsBusinessUidShareDocumentsFormSubmitFormData.md)
 - [PortalsBusinessUidShareDocumentsFormSubmitPost200Response](docs/PortalsBusinessUidShareDocumentsFormSubmitPost200Response.md)
 - [Section](docs/Section.md)


## Documentation for Authorization

Authentication schemes defined for the API:
### default

- **Type**: OAuth
- **Flow**: implicit
- **Authorization URL**: https://ignore.myclients.io
- **Scopes**: 
  - client_api: All client_api related operations
  - client_api_create: client_api create scope
  - client_api_delete: client_api delete scope
  - client_api_read: client_api read scope
  - client_api_update: client_api update scope


## Recommendation

It's recommended to create an instance of `ApiClient` per thread in a multithreaded environment to avoid any potential issues.

## Author



 readmeEtag: '"b4af61ead61427c9fbe38ae35630a8344c8658e6"' readmeLastModified: Fri, 01 Apr 2022 21:05:46 GMT repositoryId: 468561776 description: null created: '2022-03-11T01:03:44Z' updated: '2022-03-15T19:45:45Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: SimonIT logo: https://avatars.githubusercontent.com/u/11720038?v=4 repoEtag: '"65e8053d3472d2db3aeb574fbaedbd53bfe33061cb7ecce0b8625735fd5f1bc5"' repoLastModified: Tue, 15 Mar 2022 19:45:45 GMT foundInMaster: true category: SDK id: 3f38f5167e5778fdbf306b848f22197f - source: openapi3 tags repository: https://github.com/detain/myadmin-client-vue v3: true id: 263be96f5acdb8019bb4aa99faf938b6 repositoryMetadata: base64Readme: >- IyBJbnRlclNlcnZlciBNeUFkbWluIENsaWVudCBJbnRlcmZhY2UKCkNoZWNrIG91dCB0aGUgW1dJS0ldKHdpa2kpIGZvciBkb2N1bWVudGF0aW9uLgoKIyMgRkFRCgojIyMgR2V0dGluZyBhIGJsYW5rIHBhZ2Ugd2hlbiBsb2FkaW5nIHRoZSBzaXRlCgpDaGVjayB5b3VyIGFkIGJsb2NrZXIgb3IgcHJpdmFjeSBhZGRvbnMsIHRoZXkgbWlnaHQgbmVlZCB0byBiZSBkaXNhYmxlZCBmb3IgdGhpcyBzaXRlLgoKIyMgRmVhdHVyZXMKCi0gV2ViIHZlcnNpb24gZm9yIGJyb3dzZXJzCi0gRGVza3RvcCBhcHAgdmlhIEVsZWN0cm9uCi0gQXV0by11cGRhdGUgc3VwcG9ydAotIExpbnV4ICguQXBwSW1hZ2UsIC5kZWIsIFNuYXApCi0gV2luZG93cyAoLmV4ZSBpbnN0YWxsZXIsIHBvcnRhYmxlKQotIG1hY09TICguZG1nLCAucGtnKQotIEFuZHJvaWQgKC5hcGspCi0gaVBob25lCgojIyBEb3dubG9hZHMKCjwhLS0gRE9XTkxPQURTLVNUQVJUIC0tPgo8IS0tIERPV05MT0FEUy1FTkQgLS0+CgojIyBJbnN0YWxsYXRpb24KCiMjIyBXZWIKMS4gRG93bmxvYWQgdGhlIFpJUAoyLiBFeHRyYWN0IHRvIGFueSBmb2xkZXIKMy4gT3BlbiBgaW5kZXguaHRtbGAgaW4geW91ciBicm93c2VyCgojIyMgRWxlY3Ryb24KRm9sbG93IGluc3RydWN0aW9ucyBmb3IgeW91ciBwbGF0Zm9ybSBhZnRlciBkb3dubG9hZGluZyB0aGUgcmVzcGVjdGl2ZSBpbnN0YWxsZXIgb3IgWklQLgoKIyMjIExpbnV4CmBgYGJhc2gKIyBBcHBJbWFnZQpjaG1vZCAreCBNeUFwcC0qLkFwcEltYWdlCi4vTXlBcHAtKi5BcHBJbWFnZQpgYGAKCiMjIyBIb3cgaXQgV29ya3MKCldlIHVzZSBWdWUgMyB3aXRoIHRoZSBDb21wb3NpdGlvbiBBUEkgdG8gcHJvdmlkZSBhbiBlbnRpcmUgd2Vic2l0ZS9hcHBsaWNhdGlvbiBpbiBhIHNpbmdsZSBwYWdlLiAgVGhpcyBpcyBkb25lIHVzaW5nIGEgY29tYmluYXRpb24gb2YgVnVlIFNGQydzIChTaW5nbGUgRmlsZSBDb21wb25lbnRzKSB3aGljaCBhbmQgdGhlIFZ1ZSBSb3V0ZXIgdG8gY2hhbmdlIHRoZSB1cmwgaW4gdGhlIGJyb3dzZXIgd2l0aG91dCBhY3R1YWxseSBuYXZpZ2F0aW5nIHRvIGEgbmV3IHBhZ2UuCgpWYXJpYWJsZXMgYXJlIG1vc3RseSBoYW5kbGVkIGJ5IHJlZmVyZW5jZSAobGlrZSBhIHBvaW50ZXIpLiAgVGhpcyBhbGxvd3MgdXMgdG8gcGFzcyBhcm91bmQgdmFyaWFibGVzIHRoYXQgYXJlIHVrZXB0IHVwIHRvIGRhdGUgd2hlbiBjaGFuZ2VkIGluIG90aGVyIHNlY3Rpb25zIG9mIHRoZSBzaXRlLgoK readmeEtag: '"0486a18a6f56fa4464a2d050028167aae656f258"' readmeLastModified: Tue, 06 Jan 2026 02:08:40 GMT repositoryId: 614741193 description: MyAdmin Vue3 Client Frontend created: '2023-03-16T08:15:57Z' updated: '2026-02-06T03:11:04Z' language: Vue archived: false stars: 3 watchers: 1 forks: 1 owner: detain logo: https://avatars.githubusercontent.com/u/1364504?v=4 repoEtag: '"3b061b8521e342003160ec49b2b7083cf1aa9ef4a8f9152bd5d181e2541df9eb"' repoLastModified: Fri, 06 Feb 2026 03:11:04 GMT category: - SDK - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/alimahmoud80/ecommerceapi v3: true id: ad48ed01ce4090f7321cd4c0bb586151 repositoryMetadata: base64Readme: >- IyBJbnRyb2R1Y3Rpb24KCi0tLS0tLS0tLS0tLS0KVGhpcyBwcm9qZWN0IGlzIGFuIGV4YW1wbGUgb2YgZWNvbW1lcmNlIEFQSSBidWlsdCB1c2luZyBUeXBlc2NyaXB0LCBFeHByZXNzLCBNeVNRTCBhbmQgZG9jdW1lbnRlZCB1c2luZyBPcGVuQVBJIHNwZWNpZmljYXRpb24uCgojIyBEb2N1bWVudGF0aW9uCgotLS0tLS0tLS0tLS0tCkFsbCBwcm9qZWN0IGRvY3VtZW50YXRpb24gaXMgcHJlc2VudGVkIHVuZGVyIHRoZSBmb2xkZXIgW2RvY3NdKC4vZG9jcy8pCg== readmeEtag: '"623c07cdd84c1e2f64fbdc6313268701a81f808b"' readmeLastModified: Fri, 13 May 2022 03:28:31 GMT repositoryId: 491522033 description: >- Ecommerce API built using Typescript, Express, MySQL and documented using OpenAPI Specifications created: '2022-05-12T13:19:11Z' updated: '2024-12-16T22:57:04Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 0 owner: AliMahmoud80 logo: https://avatars.githubusercontent.com/u/22307533?v=4 repoEtag: '"16d3414749dfc74a1421808a92aeec957f6d4db42c57ab03a5f89b7e879dbe85"' repoLastModified: Mon, 16 Dec 2024 22:57:04 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/abassel/flask-restglue v3: true repositoryMetadata: base64Readme: >- 

<!-- PROJECT HEADER/LOGO -->
<!-- <br /> -->

<p align="center">
  <!--
  <a href="https://github.com/abassel/Flask-RestGlue">
    <img src="images/logo.png" alt="Logo" width="80" height="80">
  </a>
  -->
  <h1 align="center">Flask-RestGlue(ALPHA)</h1>

  <p align="center">
    Integrates <b>Flask + MongoDB + OpenAPI</b> in a simple and elegant way!
    <br />
    <a href="https://abassel.github.io/Flask-RestGlue/"><strong>Explore the docs »</strong></a>
    <br />
    <br />
    <a href="https://github.com/abassel/Flask-RestGlue#example">View Demo</a>
    ·
    <a href="https://github.com/abassel/Flask-RestGlue/issues">Report Bug</a>
    ·
    <a href="https://github.com/abassel/Flask-RestGlue/issues">Request Feature</a>
  </p>

  <p align="center">
      <a href="https://github.com/abassel/Flask-RestGlue/actions?query=workflow%3Abuild"><img src="https://github.com/abassel/Flask-RestGlue/workflows/build/badge.svg?branch=master&event=push" alt="Stars Badge"/></a>
      <a href="https://github.com/abassel/Flask-RestGlue/pulls"><img src="https://img.shields.io/github/issues-pr/abassel/Flask-RestGlue" alt="Pull Requests Badge"/></a>
      <a href="https://github.com/abassel/Flask-RestGlue/issues"><img src="https://img.shields.io/github/issues/abassel/Flask-RestGlue" alt="Issues Badge"/></a>
  </p>

  <p align="center">
   <a href="https://pypi.org/project/Flask-RestGlue/"><img src="https://img.shields.io/pypi/pyversions/Flask-RestGlue.svg" alt="Python Version"/></a>
   <a href="https://github.com/abassel/Flask-RestGlue/releases"><img src="https://img.shields.io/pypi/v/Flask-RestGlue?color=green&label=version" alt="Version"/></a>
   <a href="https://github.com/abassel/Flask-RestGlue/pulls?utf8=%E2%9C%93&q=is%3Apr%20author%3Aapp%2Fdependabot"><img src="https://img.shields.io/badge/dependencies-up%20to%20date-brightgreen.svg" alt="Dependencies Status"/></a>
   <a href="https://github.com/PyCQA/bandit"><img src="https://img.shields.io/badge/security-bandit-green.svg" alt="Security: bandit"/></a>
   <a href="https://github.com/abassel/Flask-RestGlue/blob/master/.pre-commit-config.yaml"><img src="https://img.shields.io/badge/pre--commit-enabled-brightgreen?logo=pre-commit&logoColor=white" alt="Pre-commit"/></a>
   <a href="https://github.com/abassel/Flask-RestGlue/blob/master/LICENSE"><img src="https://img.shields.io/github/license/abassel/Flask-RestGlue" alt="License"/></a>
  </p></p>


<!-- TABLE OF CONTENTS -->
## Contents
<details>
  <summary>Table of Contents</summary>
  <ol>
    <!--
    <li>
      <a href="#about-the-project">About The Project</a>
      <ul>
        <li><a href="#built-with">Built With</a></li>
      </ul>
    </li>
    <li>
      <a href="#getting-started">Getting Started</a>
      <ul>
        <li><a href="#prerequisites">Prerequisites</a></li>
        <li><a href="#installation">Installation</a></li>
      </ul>
    </li>
    -->
    <li><a href="#example">Example</a></li>
    <li><a href="#quick-start">Quick Start</a></li>
    <li><a href="#references-notebook">References</a></li>
    <!--<li><a href="#roadmap">Roadmap</a></li>
    <li><a href="#contributing">Contributing</a></li>
    <li><a href="#license">License</a></li>
    <li><a href="#contact">Contact</a></li>
    <li><a href="#acknowledgements">Acknowledgements</a></li>-->
  </ol>
</details>



# Example

For a fullstack boilerplate, visit [https://github.com/abassel/Flask_RestGlue_Svelte_Docker](https://github.com/abassel/Flask_RestGlue_Svelte_Docker)

```python
import mongoengine as mongo
from flask_rest_glue import FlaskRestGlue

mongo.connect("pyglue", host='localhost:27017')

api = FlaskRestGlue()


@api.rest_model()
class User(mongo.Document):
  # id = mongo.StringField(primary_key=True)
  email = mongo.StringField(primary_key=True)
  password = mongo.StringField()


api.run()

```

Go to [http://127.0.0.1:5000/spec_doc](http://127.0.0.1:5000/spec_doc) or [http://127.0.0.1:5000/spec_rdoc](http://127.0.0.1:5000/spec_rdoc) to see the documentation below:

![Swagger UI](https://abassel.github.io/Flask-RestGlue/swagger.png)

![ReText UI](https://abassel.github.io/Flask-RestGlue/rdoc.png)

### Expected output:

```bash
curl -v -d '{"email":"a@b.com","password":"xyz"}' \
     -H "Content-Type: application/json" http://localhost:5000/user

#< HTTP/1.0 200 OK
#< Content-Type: application/json
#< Content-Length: 45
#<
#{
#  "_id": "a@b.com",
#  "password": "xyz"
#}


curl -v http://localhost:5000/users
#< HTTP/1.0 200 OK
#< Content-Type: application/json
#< Content-Length: 57
#<
#[
#  {
#    "_id": "a@b.com",
#    "password": "xyz"
#  }
#]


curl -v http://localhost:5000/user/a@b.com
#< HTTP/1.0 200 OK
#< Content-Type: application/json
#< Content-Length: 45
#<
#{
#  "_id": "a@b.com",
#  "password": "xyz"
#}


curl -v -X PUT -d '{"password":"new_pass"}' \
     -H "Content-Type: application/json" http://localhost:5000/user/a@b.com
#< HTTP/1.0 200 OK
#< Content-Type: application/json
#< Content-Length: 50
#<
#{
#  "_id": "a@b.com",
#  "password": "new_pass"
#}


curl -v -X DELETE http://localhost:5000/user/a@b.com
#< HTTP/1.0 200 OK
#< Content-Type: application/json
#< Content-Length: 45
#<
#{
#  "_id": "a@b.com",
#  "password": "xyz"
#}
```


# Quick Start

Requires **docker** and **python 3.9**

### 1 - install local MongoDB
```bash
mkdir -p ~/mongodata

docker run -d --rm -p 27017:27017 -v ~/mongodata:/data/db --name mongodb mongo
```

### 2 - Install this library
```bash
pip install Flask-RestGlue
```

### 3 - Pull the code
```bash
curl -s -O -L  https://github.com/abassel/Flask-RestGlue/blob/master/example/tut01_hello_world.py
curl -s -O -L  https://github.com/abassel/Flask-RestGlue/blob/master/example/tut01_hello_world.sh
```

### 4 - Run the code
```bash
python tut01_hello_world.py
```
In another terminal window

```bash
bash tut01_hello_world.sh
```

## Road Map :car:

- Evaluate https://testdriven.io/blog/fastapi-mongo/
- Evaluate https://github.com/David-Lor/FastAPI-Pydantic-Mongo_Sample_CRUD_API
- Evaluate https://art049.github.io/odmantic/
- Example user model that stores password as hash - https://aaronluna.dev/series/flask-api-tutorial/part-2/
- ORM examples for FastAPI - https://github.com/mjhea0/awesome-fastapi#orms
- ODM examples for FastAPI - https://github.com/mjhea0/awesome-fastapi#odms
- Authorization at row level - https://github.com/mjhea0/awesome-fastapi?tab=readme-ov-file#auth
- Develop pagination
- Develop Pydantic
- Google search about flask api with sqlalchemy https://www.google.com/search?q=flask+api+sqlalchemy+tutorial
- Evaluate flask api with migration https://aaronluna.dev/series/flask-api-tutorial/part-2/
- Develop tools to generate typescript or svelte stubs
- Create a template like https://github.com/tiangolo/full-stack-fastapi-template

## References :notebook:
- [Flask_RestGlue_Svelte_Docker boilerplate](https://github.com/abassel/Flask_RestGlue_Svelte_Docker)
- [OpenAPI 3.0.3](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.3.md)
- [Flask](http://flask.pocoo.org)
- [Mongoengine](https://github.com/MongoEngine/mongoengine)
 readmeEtag: '"e3aa73f56d255abb6ac8c121f50b90ce95f510c5"' readmeLastModified: Mon, 13 Jan 2025 02:31:34 GMT repositoryId: 326479351 description: >- 🚀 Integrates Flask + MongoDB(mongoengine) + OpenAPI in a simple and elegant way! created: '2021-01-03T18:56:36Z' updated: '2025-02-25T19:44:15Z' language: Python archived: false stars: 2 watchers: 1 forks: 0 owner: abassel logo: https://avatars.githubusercontent.com/u/11035784?v=4 license: MIT repoEtag: '"e7610f6c80c5463fd554b7c476a30f53076f7001ce15035c83ce91d533366149"' repoLastModified: Tue, 25 Feb 2025 19:44:15 GMT foundInMaster: true category: Server Implementations id: e16ab98bb1345dc9c96aeb6124d53383 - source: openapi3 tags repository: https://github.com/jshahriddhi/simple-flask-project v3: true repositoryMetadata: base64Readme: >- IyBzaW1wbGUtZmxhc2stcHJvamVjdAoKU2V0dXA6IGNyZWF0ZSBhIHB5dGhvbiB2aXJ0dWFsIGVudmlyb25tZW50IGFuZCBpbnN0YWxsIHRoZSByZXF1aXJlbWVudHM6CgoJdmlydHVhbGVudiAtLXB5dGhvbj1weXRob24zLjYgLnZlbnYKCXNvdXJjZSAudmVudi9iaW4vYWN0aXZhdGUKCXBpcCBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQKCgpHZW5lcmF0ZSBPcGVuQVBJIHNwZWMgZnJvbSBGbGFzayByb3V0ZXM6CgoJcHl0aG9uIGdlbmVyYXRlX29wZW5hcGlfc3BlYy5weQoKICAgIApWaWV3IHRoZSBPcGVuQVBJIHNwZWMgdXNpbmcgU3dhZ2dlci1VSToKCiAgICBkb2NrZXIgcnVuIC0tcm0gLXAgOTAwMDo4MDgwIC0tbmFtZSBzd2FnZ2VyLXVpIC1lIFNXQUdHRVJfSlNPTj0vYXBpX2RvY3Mvb3BlbmFwaS5qc29uIC12ICQoUFdEKS9hcGlfZG9jczovYXBpX2RvY3Mgc3dhZ2dlcmFwaS9zd2FnZ2VyLXVpCg== readmeEtag: '"c52b339bac8c3d27832ffe58c714eed7e434a986"' readmeLastModified: Mon, 06 Aug 2018 01:50:04 GMT repositoryId: 138524469 description: Generate OpenAPI spec from Flask routes created: '2018-06-25T00:08:10Z' updated: '2025-10-06T09:24:51Z' language: Python archived: false stars: 2 watchers: 0 forks: 0 owner: jshahriddhi logo: https://avatars.githubusercontent.com/u/4574087?v=4 repoEtag: '"55ac32adadec8ba785f64799be1940a3360517120363a314d20e0308a3ec0e61"' repoLastModified: Mon, 06 Oct 2025 09:24:51 GMT foundInMaster: true category: - Converters - Server Implementations id: ac92023d1ac30ff1e4781e3e08f380cf oldLocations: - https://github.com/riddhi89/simple-flask-project - source: openapi3 tags repository: https://github.com/talentplatforms/ropen_pi v3: true repositoryMetadata: base64Readme: >- WyFbQ29udmVudGlvbmFsIENvbW1pdHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvQ29udmVudGlvbmFsJTIwQ29tbWl0cy0xLjAuMC15ZWxsb3cuc3ZnKV0oaHR0cHM6Ly9jb252ZW50aW9uYWxjb21taXRzLm9yZykKCiMgUm9wZW5QaQoKVGhpcyBpcyBhIGZvcmsgb2YgdGhlIFtvcGVuLWFwaS1yc3dhZyBnZW1dKGh0dHBzOi8vZ2l0aHViLmNvbS9qZGFuaWVsaWFuL29wZW4tYXBpLXJzd2FnKS4KSXQncyBhIGdvb2QgZ2VtIGluIGFuIGVhcmx5IGRldmVsb3BtZW50IHN0YWdlLiBUaGUgb3JpZ2luYWwgW3Jzd2FnXShodHRwczovL2dpdGh1Yi5jb20vcnN3YWcvcnN3YWcpIGdlbSBhbmQgdGhlIG9wZW4gYXBpIGZvcmsgYSBoYXZlIG9uZSB0aGluZyBpbiBjb21tb246IHRoZXkgZG8gdG8gbXVjaC4KClRoZXJlIGFyZSBlYXNpZXIgd2F5cyB0byBnZXQgYW4gaW5zdGFuY2Ugb2YgdGhlIHN3YWdnZXIgdWkgcnVubmluZyB0aGFuIGJ1aWxkaW5nIGl0IGFzIGEgcmFpbHMgZW5naW5lLgoKSW4gZnV0dXJlIHJlbGVhc2VzIHRoaXMgZ2VtIHdpbGwgcHJvdmlkZSBhIGJhc2ljIGV4YW1wbGUgb2YgaG93IHRoaXMgd2lsbCBsb29rIGxpa2UuCgpJbiB0aGUgbWVhbiB0aW1lIHRoaXMgd2lsbCByZW1haW4gYSBXSVAgOikKCiMjIEluc3RhbGxhdGlvbgoKQWRkIHRoaXMgbGluZSB0byB5b3VyIGFwcGxpY2F0aW9uJ3MgR2VtZmlsZToKCmBgYHJ1YnkKZ2VtICdyb3Blbl9waScKYGBgCgpBbmQgdGhlbiBleGVjdXRlOgoKICAgICQgYnVuZGxlCgpPciBpbnN0YWxsIGl0IHlvdXJzZWxmIGFzOgoKICAgICQgZ2VtIGluc3RhbGwgcm9wZW5fcGkKCiMjIFVuc3VwcG9ydGVkCgpEdWUgdG8gbWFzc2l2ZSBjaGFuZ2VzIGluIHRoZSBzcGVjLCB0aGlzIGdlbSBjdXJyZW50bHkgZG9lcyBub3Qgc3VwcG9ydCBhbnkgdHlwZSBvZiBjb2xsZWN0aW9uIHF1ZXJpZXMuCgojIyBVc2FnZQoKVE9ETzogV3JpdGUgdXNhZ2UgaW5zdHJ1Y3Rpb25zIGhlcmUKCiMjIERldmVsb3BtZW50CgpTbyB5b3UgaGF2ZSBkZWNpZGVkIHRvIGNvbnRyaWJ1dGU/ISBUaGlzIHNvdW5kcyBncmVhdCEKVG8gbWFrZSBpdCBzdXBlciBlYXN5IHRvIHN0YXJ0IHRoZSBvbmx5IHRoaW5nIHlvdSBoYXZlIHRvIGhhdmUgaW5zdGFsbGVkIGlzIERvY2tlci4KVGhlbiB5b3UgY2FuIGp1c3QgdXNlIHRoZSBiYXR0ZXJpZXMtaW5jbHVkZWQgRGV2Q29udGFpbmVyIGZvciBWU0NvZGUuCgpUaGVyZSBpcyBhIGxvdCBvZiBkb2N1bWVudGF0aW9uIG9uIGdldHRpbmcgc3RhcnRlZCB3aXRoIHRoaXMga2luZCBvZiBkZXZlbG9wbWVudC4KSWYgeW91IGFyZSBraW5kIG9mIG5ldyB0byB0aGlzIHBsZWFzZSBtYWtlIHN1cmUgdG8gcmVhZCB0aGUgb2ZmaWNhbCBbZG9jcyBvdmVyIGF0IE1pY3Jvc29mdF0oaHR0cHM6Ly9jb2RlLnZpc3VhbHN0dWRpby5jb20vZG9jcy9yZW1vdGUvY29udGFpbmVycykuCgojIyBDb250cmlidXRpbmcKCkJ1ZyByZXBvcnRzIGFuZCBwdWxsIHJlcXVlc3RzIGFyZSB3ZWxjb21lIG9uIEdpdEh1YiBhdCBodHRwczovL2dpdGh1Yi5jb20vdGFsZW50cGxhdGZvcm1zL3JvcGVuX3BpLgoKIyMgTGljZW5zZQoKVGhlIGdlbSBpcyBhdmFpbGFibGUgYXMgb3BlbiBzb3VyY2UgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbTUlUIExpY2Vuc2VdKGh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUKS4K readmeEtag: '"614a45628426de892e0a70c7be311c561e0fe392"' readmeLastModified: Tue, 29 Nov 2022 14:13:17 GMT repositoryId: 232335523 description: Rspec and OpenAPI Integration created: '2020-01-07T13:55:14Z' updated: '2021-11-03T11:34:42Z' language: Ruby archived: false stars: 2 watchers: 2 forks: 0 owner: talentplatforms logo: https://avatars.githubusercontent.com/u/55129375?v=4 license: MIT repoEtag: '"ef61dbdb195c9b7ff0edc587a1c04e97b35c433a3b71e0143b23f6eaf053de3b"' repoLastModified: Wed, 03 Nov 2021 11:34:42 GMT foundInMaster: true category: Parsers id: 2927c6148c140508477dc6e6b9a038f6 - source: openapi3 tags repository: https://github.com/michaelsauter/go-oas-server v3: true repositoryMetadata: base64Readme: >- IyBnby1vYXMtc2VydmVyCgpHZW5lcmF0ZSBHbyBzZXJ2ZXIgY29kZSBmcm9tIGFuIFtPcGVuQVBJIDMgc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKS4KClRoaXMgcHJvamVjdCBpcyBmb3IgeW91IGlmIHlvdSB3YW50IHRvIHdyaXRlIGEgUkVTVCBBUEkgaW4gR28sIGFuZCB5b3UgdmFsdWUgdGhlIGZvbGxvd2luZzoKCiogRGVzaWduLWZpc3QgYXBwcm9hY2ggdXNpbmcgT3BlbkFQSSAzCiogVHlwZSBzYWZldHk6IGF2b2lkIHVzaW5nIGBjb250ZXh0LkNvbnRleHRgIGFuZCBgaW50ZXJmYWNle31gIGFzIG11Y2ggYXMgcG9zc2libGUKKiBDb21waWxlci1kcml2ZW4gZGV2ZWxvcG1lbnQ6IGxldCB0aGUgZ2VuZXJhdGVkIGNvZGUgZ3VpZGUgeW91IHdoYXQgeW91IG5lZWQgdG8gaW1wbGVtZW50CiogQXBwcm9hY2hlcyBsaWtlIFtIb3cgSSB3cml0ZSBHbyBIVFRQIHNlcnZpY2VzIGFmdGVyIHNldmVuIHllYXJzXShodHRwczovL21lZGl1bS5jb20vc3RhdHVzY29kZS9ob3ctaS13cml0ZS1nby1odHRwLXNlcnZpY2VzLWFmdGVyLXNldmVuLXllYXJzLTM3YzIwODEyMjgzMSkgYnV0IGRvbid0IHdhbnQgdG8gZGVhbCB3aXRoIHJvdXRpbmcsIHBhcmFtZXRlciB2YWxpZGF0aW9uIGFuZCBkb2N1bWVudGF0aW9uCgojIyBVc2FnZQoKMS4gRGVzaWduIHlvdXIgQVBJLCBlLmcuIGluIHRoZSBbU3dhZ2dlciBFZGl0b3JdKGh0dHBzOi8vc3dhZ2dlci5pby90b29scy9zd2FnZ2VyLWVkaXRvci8pCjIuIEV4cG9ydCB0aGUgc3BlY2lmaWNhdGlvbiBhcyBKU09OCjMuIEdlbmVyYXRlIEdvIGNvZGUgdmlhIGBnby1vYXMtc2VydmVyIGdlbmVyYXRlIC0tZmlsZSBhcGkuanNvbiAtLW91dHB1dC1kaXI9Z2VuYAo0LiBJbXBsZW1lbnQgeW91ciBzZXJ2ZXIgYW5kIGl0cyBlbmRwb2ludHM6CgpgYGAKcGFja2FnZSBtYWluCgp0eXBlIG15U2VydmVyIHN0cnVjdCB7fQoKZnVuYyAocyAqbXlTZXJ2ZXIpIE1pZGRsZXdhcmVzKCkgZ2VuLk1pZGRsZXdhcmVzIHsKCXJldHVybiBNaWRkbGV3YXJlc3t9Cn0KCmZ1bmMgKHMgKnNlcnZlcikgSGFuZGxlUGV0SW5kZXgoKSAoZ2VuLk1pZGRsZXdhcmVzLCBnZW4uQVBJT3BlcmF0aW9uUGV0SW5kZXgpIHsKCXJldHVybiBuaWwsIGZ1bmModyBodHRwLlJlc3BvbnNlV3JpdGVyLCByICpodHRwLlJlcXVlc3QsIHAgZ2VuLlBhcmFtZXRlcnNQZXRJbmRleCkgewoJCS8vIEltcGxlbWVudCB5b3VyIGxvZ2ljIGhlcmUKCX0KfQoKZnVuYyBtYWluKCkgewoJcyA6PSBOZXdPcGVuQVBJU2VydmVyKCZteVNlcnZlcnt9KQoJcy5Cb290KCkKCWxvZy5GYXRhbChodHRwLkxpc3RlbkFuZFNlcnZlKCI6ODAwMCIsIHMpCn0KYGBgCgojIyBDdXJyZW50IFN0YXRlCgpUaGlzIHByb2plY3QgaXMgY3VycmVudGx5IG9ubHkgbGl0dGxlIG1vcmUgdGhhbiBhIHByb29mIG9mIGNvbmNlcHQuIFdoaWxlIHRoZSBnZW5lcmFsIGJ1aWxkaW5nIGJsb2NrcyBhcmUgaW4gcGxhY2UsIGEgbG90IG9mIHRoZSBBUEkgbWF5IGNoYW5nZS4gRnVydGhlciwgc29tZSBhcmVhcyBhcmUgbm90IGV2ZW4gY292ZXJlZCB5ZXQgc3VjaCBhcyBnZW5lcmF0aW5nIHN1cHBvcnQgZm9yIHJlc3BvbnNlcy4KCkhlcmUncyBhIChub24tZXhoYXVzdGl2ZSkgbGlzdCBvZiB3aGF0J3MgbGVmdCB0byBkbzoKKiBTdXBwb3J0IG1vcmUgdHlwZXMgYW5kIHZhbGlkYXRpb25zCiogRW5kcG9pbnQtbGV2ZWwgR28gZGVwZW5kZW5jaWVzCiogUmVzcG9uc2VzCiogSGFuZGxpbmcgbW9yZSBjb21wb25lbnRzCiogTWFrZSBpdCBlYXNpZXIgZm9yIHVzZXJzIHRvIGZpZ3VyZSBvdXQgd2hhdCB0byBpbXBsZW1lbnQgYWZ0ZXIgY29kZSBnZW5lcmF0aW9uCiogTW9yZSB0ZXN0aW5nIHN1cHBvcnQKKiBCZXR0ZXIgbmFtaW5nIHN1cHBvcnQgKGF2b2lkIGJhZCBjaGFycywgYW5kIHN1cHBvcnQgW2NvbW1vbiBpbml0aWFsaXNtc10oaHR0cHM6Ly9naXRodWIuY29tL2dvbGFuZy9saW50L2Jsb2IvOGY0NWY3NzZhYWYxOGNlYmM4ZDY1ODYxY2M3MGMzM2M2MDQ3MTk1Mi9saW50LmdvI0w3NzEpKS4KCkFsc28sIGdvLW9hcy1zZXJ2ZXIgY2Fubm90IGdlbmVyYXRlIGNvZGUgZm9yIGV2ZXJ5IHBvc3NpYmxlIHNwZWNpZmljYXRpb24uIEluIHBhcnQsIHRoaXMgbGltaXRhdGlvbiBleGlzdHMgdG8gYXZvaWQgY29tcGxleGl0eS4KCmdvLW9hcy1zZXJ2ZXIgKGN1cnJlbnRseSkgZG9lcyBub3Qgc3VwcG9ydDoKCiogUGFyYW1ldGVyIHNlcmlhbGl6YXRpb24uCiogUGFyYW1ldGVycyBkZWZpbmVkIHdpdGggYGNvbnRlbnRgIGluc3RlYWQgb2YgYHNjaGVtYWAuCiogTGlua3MuCiogQ2FsbGJhY2tzLgoqIFJlc3BvbnNlcyAoaG93ZXZlciBzdXBwb3J0IGZvciB0aGlzIGlzIHBsYW5uZWQpLgoqIFVzZSBvZiBgb25lT2ZgLCBgYW55T2ZgLCBgYWxsT2ZgIGFueXdoZXJlLgoKVGhlcmUgYXJlIGZ1cnRoZXIgZWdkZSBjYXNlcyBmb3Igc3VyZSwgaWYgeW91IHRoaW5rIHlvdSd2ZSBydW4gaW50byBvbmUsIHBsZWFzZSBvcGVuIGFuIGlzc3VlLgoKCiMjIyBXaHkgZ2VuZXJhdGUgY29kZSBmcm9tIHRoZSBzcGVjaWZpY2F0aW9uLCBhbmQgbm90IHRoZSBvdGhlciB3YXkgYXJvdW5kPwoKKiBUeXBpY2FsbHksIGl0IGlzIGZhc3RlciB0byBkZWZpbmUgdGhlIHNwZWNpZmljYXRpb24gb2YgYW4gQVBJIHRoYW4gdG8gaW1wbGVtZW50IGl0LgoqIElmIHlvdSBoYXZlIGFjY2VzcyB0byBmdXR1cmUgY29uc3VtZXJzIG9mIHRoZSBBUEkgZW5kcG9pbnQocykgYmVpbmcgZGVzaWduZWQsIGl0IGlzIGVhc2llciB0byBnYXRoZXIgZmVlZGJhY2sgYmFzZWQgb24gdGhlIHNwZWNpZmljYXRpb24uIEhhdmluZyBvbmx5IGNvZGUgaXMgbm90IGEgZ29vZCBiYXNlIGZvciBkaXNjdXNzaW9uLgoqIEl0IGlzIG5vbi10cml2aWFsIHRvIGNvbnRyb2wgZXZlcnkgYXNwZWN0IG9mIHRoZSBzcGVjaWZpY2F0aW9uIGZyb20gY29kZS4gT2Z0ZW4gdGhpcyBpbnZvbHZlcyB1c2luZyBsb3RzIG9mIGFubm90YXRpb25zIGluIG90aGVyIGxhbmd1YWdlcy4gRm9yIGV4YW1wbGUsIHBhcmFtZXRlciBjb25zdHJhaW50cyBhcmUgZWFzeSB0byBnZW5lcmF0ZSBmcm9tIHNwZWNpZmljYXRpb24sIGJ1dCBoYXJkIHRvIGdlbmVyYXRlIGZyb20gY29kZS4KKiBXaGVuIHRoZSBzcGVjaWZpY2F0aW9uIGlzIGdlbmVyYXRlZCwgaXQgaXMgb2Z0ZW4gbm90IGxvb2tlZCBhdCBieSB0aGUgQVBJIGRldmVsb3BlciwgYW5kIHRoZXJlZm9yZSBub3QgYXMgZGV0YWlsZWQgYXMgaXQgY291bGQgYmUuCiogV3JpdGluZyBBUEkgZW5kcG9pbnRzIGluIEdvIGludm9sdmVzIGEgbG90IG9mIGJvaWxlcnBsYXRlLCB3aGljaCBpcyBib3JpbmcgdG8gd3JpdGUuIEEgZ2VuZXJhdG9yIGZpdHMgbmljZWx5LgoKIyMgV2hhdCBpcyB0aGUgZGlmZmVyZW5jZSB0byBnby1zd2FnZ2VyPwoKKiBnby1zd2FnZ2VyIGlzIGFuIGltcGxlbWVudGF0aW9uIG9mIFN3YWdnZXIgMi4wLCBub3QgT3BlbkFQSSAzLgoqIGdvLXN3YWdnZXIgbG9va3MgaHVnZSBhbmQgSSB3YW50ZWQgc29tZXRoaW5nIHNtYWxsZXIgdGhhdCBmZWVscyBtb3JlIGxpa2Ugd3JpdGluZyBgbmV0L2h0dHBgIGhhbmRsZXJzLgoqIGdvLW9hcy1zZXJ2ZXIgZG9lcyBub3QgZ2VuZXJhdGUgY2xpZW50cy4K readmeEtag: '"8d306d0e9ebf27fd70c6c86bc56bdbad3df99aec"' readmeLastModified: Tue, 06 Aug 2019 19:15:53 GMT repositoryId: 199204368 description: Go server generator from OpenAPI 3 created: '2019-07-27T19:12:45Z' updated: '2026-01-31T12:19:06Z' language: Go archived: false stars: 3 watchers: 1 forks: 0 owner: michaelsauter logo: https://avatars.githubusercontent.com/u/215455?v=4 repoEtag: '"c6d8af9f2c0218c455d094b475ab71e14ca983073980dc3bc7ac7bcc49e595b5"' repoLastModified: Sat, 31 Jan 2026 12:19:06 GMT foundInMaster: true category: Server Implementations id: 88ce19dbf86d450a1530d74394f6eff6 - source: openapi3 tags repository: https://github.com/ebln/guzzle-openapi-middleware v3: true id: 1d17b4d627cb3cf6ea048171fb9654a0 repositoryMetadata: base64Readme: >- T3BlbkFQSSBWYWxpZGF0aW9uIE1pZGRsZXdhcmUgZm9yIEd1enpsZQo9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CgpbVGhpcyBtaWRkbGV3YXJlIG9ubHkgYWRhcHRzIGxlYWd1ZS9vcGVuYXBpLXBzcjctdmFsaWRhdG9yIGZvciBHdXp6bGUsIHBsZWFzZSBzZWUgdGhlaXIgcHJvamVjdCBmb3IgZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL3RoZXBocGxlYWd1ZS9vcGVuYXBpLXBzcjctdmFsaWRhdG9yI3JlYWRtZSkKCiMjIEluc3RhbGxhdGlvbgoKYGBgCmNvbXBvc2VyIHJlcXVpcmUgZWJsbi9ndXp6bGUtb3BlbmFwaS1taWRkbGV3YXJlCmBgYAoKIyMgVXNhZ2UKCmBgYAp1c2UgR3V6emxlSHR0cFxDbGllbnQ7CnVzZSBHdXp6bGVIdHRwXEhhbmRsZXJTdGFjazsKdXNlIExlYWd1ZVxPcGVuQVBJVmFsaWRhdGlvblxQU1I3XFZhbGlkYXRvckJ1aWxkZXI7CgokYnVpbGRlciA9IG5ldyBWYWxpZGF0b3JCdWlsZGVyKCk7Ci8vIGNhbGwgZWl0aGVyIHNldFNjaGVtYUZhY3RvcnkoKSBvciBvbmUgb2YgdGhlIGZyb20qKCkgbWV0aG9kcyBvcHRpb25hbGx5IGFkZCBhIFBTUjYgY2FjaGUKLy8gQHNlZSBodHRwczovL2dpdGh1Yi5jb20vdGhlcGhwbGVhZ3VlL29wZW5hcGktcHNyNy12YWxpZGF0b3IjcmVhZG1lCgokbWlkZGxld2FyZSA9IG5ldyBNaWRkbGV3YXJlKCRidWlsZGVyLT5nZXRSZXF1ZXN0VmFsaWRhdG9yKCksICRidWlsZGVyLT5nZXRSZXNwb25zZVZhbGlkYXRvcigpKTsKCi8vIEBzZWUgaHR0cHM6Ly9kb2NzLmd1enpsZXBocC5vcmcvZW4vc3RhYmxlL2hhbmRsZXJzLWFuZC1taWRkbGV3YXJlLmh0bWwjbWlkZGxld2FyZQokc3RhY2sgPSBIYW5kbGVyU3RhY2s6OmNyZWF0ZSgpOwokc3RhY2stPnB1c2goJG1pZGRsZXdhcmUsICdvcGVuYXBpX3ZhbGlkYXRpb24nKTsKJGNsaWVudCA9IG5ldyBDbGllbnQoWydoYW5kbGVyJyA9PiAkc3RhY2tdKTsKYGBgCg== readmeEtag: '"d5d2c88ab08fa482a5d092955583aab3476ac33f"' readmeLastModified: Sat, 20 Aug 2022 13:33:05 GMT repositoryId: 526515209 description: >- OpenAPI validation middleware for Guzzle, adapting league/openapi-psr7-validator created: '2022-08-19T07:59:26Z' updated: '2024-04-25T08:55:26Z' language: PHP archived: false stars: 2 watchers: 1 forks: 0 owner: ebln logo: https://avatars.githubusercontent.com/u/34722048?v=4 license: MIT repoEtag: '"40ecab528c0fad73cba867a354f848bc6e32a8851ee4c8d5fc23a2b2364f03f8"' repoLastModified: Thu, 25 Apr 2024 08:55:26 GMT category: Data Validators foundInMaster: true - source: openapi3 tags repository: https://github.com/goodluckxu-go/openapi v3: true id: ad35a471acd8b722cf9fe035a6535cbb repositoryMetadata: base64Readme: >- IyBvcGVuYXBpMyDmlofmoaPnlJ/miJAKCuS9v+eUqGFzdOivreazleino+aekOWZqOino+aekOazqOino++8jOagueaNrm9wZW5hcGkz6K+t5rOV55Sf5oiQCgojIyDlvJXnlKgocXVvdGUpCi0gZ2l0aHViLmNvbS9nZXRraW4va2luLW9wZW5hcGkKCgojIyDnlKjms5UodXNhZ2UpCuWkmuihjOazqOmHiuWPr+S7peS9v+eUqCB8LSDnrKblj7fvvIzlkoxA5qCH6aKY5LiA6KGM77yM5LiN6IO95pyJ5YW25LuW5YaF5a6577yM55+l6YGT6YGH5Yiw5LiL5LiA5Liq5Y+v55So5qCH562+5oiW6ICF5Y2V54us5LiA6KGM5Li6LXzjgILkvovlpoLvvJoKfn5+Z28KLy8gQGluZm8udGl0bGU6IHwtCi8vICAg6L+Z5piv5qCH6aKYCi8vIEBpbmZvLmRlc2NyaXB0aW9uOiB8LQovLyAgIOi/meaYr+aPj+i/sAovLyAgLXwKLy8g6L+Z6YeM55qE5rOo6YeK5piv5peg5pWI55qE77yM5LiK6Z2i5bey57uP57uI5q2i77yM5aaC5p6c5LiN5Lqb57uI5q2i56ym5Y+35YiZ5Li65LiK6Z2i5qCH562+55qE5rOo6YeKCi8vIEBpbmZvLnZlcnNpb246IOi/meaYr+eJiOacrOWPtwp+fn4KCiMjIyDlronoo4Xlkb3ku6QKfn5+c2hlbGwKZ28gaW5zdGFsbCBnaXRodWIuY29tL2dvb2RsdWNreHUtZ28vb3BlbmFwaS9jbWQvYXBpZ2VuQGxhdGVzdAp+fn4KCiMjIyBkb2NzLmdv5paH5qGj5rOo6YeK6K+05piOCn5+fmdvCi8vIEBpbmZvLnRpdGxlOiDmoIfpopgKLy8gQGluZm8uZGVzY3JpcHRpb246IOaPj+i/sAovLyBAaW5mby50ZXJtc09mU2VydmljZTog5pyN5Yqh5p2h5qy+Ci8vIEBpbmZvLmNvbnRhY3QubmFtZTog6IGU57O75Lq6Ci8vIEBpbmZvLmNvbnRhY3QudXJsOiDogZTns7vlnLDlnYAKLy8gQGluZm8uY29udGFjdC5lbWFpbDog6IGU57O76YKu566xCi8vIEBpbmZvLmxpY2Vuc2UubmFtZTog6K645Y+v6K+B5ZCN56ewCi8vIEBpbmZvLmxpY2Vuc2UudXJsOiDorrjlj6/or4HlnLDlnYAKLy8gQGluZm8udmVyc2lvbjog6aG555uu54mI5pys5Y+3Ci8vIEBleHRlcm5hbERvY3MuZGVzY3JpcHRpb246IOaJqeWxleaWh+aho+aPj+i/sAovLyBAZXh0ZXJuYWxEb2NzLnVybDog5omp5bGV5paH5qGj5Zyw5Z2ACi8vIEBzZXJ2ZXJzOiB1cmw95pyN5Yqh5Zyw5Z2AOyBkZXNjcmlwdGlvbj3mnI3liqHmj4/ov7AKLy8gQHRhZ3M6IG5hbWU95qCH562+5ZCN56ewOyBkZXNjcmlwdGlvbj3moIfnrb7mj4/ov7AKLy8gQGNvbXBvbmVudHMuc2VjdXJpdHlTY2hlbWVzOiB8LQovLyAgZmllbGQ96aqM6K+B5a2X5q6177yM6Lev55Sx5rOo6YeK5Lit5L2/55SoOwovLyAgdHlwZT3pqozor4HnsbvlnovvvIzlgLzljIXmi6xhcGlLZXksaHR0cCxvYXV0aDI7Ci8vICBzY2hlbWU9aHR0cOexu+Wei+W/heS8oO+8jOS+i+WmgmJhc2ljLGJlYXJlcjsKLy8gIGJlYXJlckZvcm1hdD1zY2hlbWXkuLpiZWFyZXLml7blj6/kvKDvvIznlKjkuo7mj5DnpLrlrqLmiLfnq6/miYDkvb/nlKjnmoRiZWFyZXIgdG9rZW7nmoTmoLzlvI/vvIzkvovlpoJKV1Q7Ci8vICBpbj1hcGlLZXnml7blv4XkvKDvvIzlgLzljIXmi6xxdWVyeSxoZWFkZXIsY29va2llOwovLyAgbmFtZT1hcGlLZXnml7blv4XkvKDvvIznlKjkuo4gaGVhZGVy44CBIHF1ZXJ5IOaIliBjb29raWUg55qE5Y+C5pWw5ZCN5a2XOwovLyAgZmxvd3M9anNvbuWtl+espuS4su+8jOaWh+aho+ivtOaYjuivpuinge+8mmh0dHBzOi8vb3BlbmFwaS5hcGlmb3guY24vI29hdXRoLWZsb3dzLSVFNSVBRiVCOSVFOCVCMSVBMQpwYWNrYWdlIG1haW4Kfn5+CgojIyMjIGRvY3MuZ28g5paH5qGj5Lit5a6a5LmJ5YWs5YWx55qEIHJvdXRlLmdvIOS4reeahOWxnuaAp++8jOWSjCByb3V0ZS5nbyDnmoTms6jph4rkuIDoh7TvvIzlnKhA5ZCO6Z2i5re75YqgZ2xvYmFsLgotIEBnbG9iYWwucmVzICDlhazlhbHov5Tlm57vvIzlkozot6/nlLHnmoTkuIDoh7QKLSBAZ2xvYmFsLnBhcmFtICDlhazlhbHlj4LmlbDvvIzlkozot6/nlLHnmoTkuIDoh7QKfn5+Z28KLy8gQGdsb2JhbC5yZXM6IHN0YXR1cz01MDA7IGluPWFwcGxpY2F0aW9uL2pzb247IGNvbnRlbnQ95pyN5Yqh5Zmo6ZO+5o6l5aSx6LSlOyBkZXNjPeezu+e7n+WGhemDqOmUmeivrwpwYWNrYWdlIG1haW4Kfn5+CgojIyMgcm91dGUuZ2/mlofmoaPms6jph4ror7TmmI4KLSBA5byA5aeL55qE5qCH6aKY77yM55SoIDsg5YiG5Ymy77yM5YiG5Ymy5oiQ5a+56LGh44CC5q+P5Liq5a+56LGh55SoID0g5YiG5Ymy77yM5YiG5Ymy5oiQ6ZSu77yM5YC877yM5aaC5p6c5LiN5a2Y5ZyoID0g5aW95YiZ6KGo56S65YC85piv5a2X56ym5LiyIHRydWUKCi0g5LiK6Z2i5YiG5Ymy55qE5a+56LGh5Lit55qE5YC877yM55SoICwg5YiG5Ymy5oiQ5pWw57uECn5+fmdvCnBhY2thZ2UgbWFpbgoKLy8gQHN1bW1hcnk6IOi3r+eUseaAu+e7kwovLyBAZGVzY3JpcHRpb246IOi3r+eUseaPj+i/sAovLyBAdGFnczog5qCH562+57uE77yM55SoO+WIhuWJsu+8jOS+i+Wmgu+8mnVzZXI7YWRtaW4KLy8gQHBhcmFtOiDlj4LmlbDvvIzlpJrooYzliJnlpJrkuKrlj4LmlbDvvIzor6bnu4bor7TmmI7op4HkuIvpnaJAcGFyYW3or7TmmI4KLy8gQGJvZHk6IOS8oOmAkuWGheWuue+8jOivpue7huivtOaYjuingeS4i+mdokBib2R56K+05piOCi8vIEByZXM6IOi+k+WHuuWGheWuue+8jOivpuino+ivtOaYjuingeS4i+mdokByZXPor7TmmI4KLy8gQHNlY3VyaXR5OiB8LQovLyAg6aqM6K+B5YC877yM5L2/55SoIEBjb21wb25lbnRzLnNlY3VyaXR5U2NoZW1lcyDkuK3lrprkuYnnmoQgZmllbGQg55qE5YC8Ci8vICDkvovlpoLvvJp0b2tlbjtwcm9qZWN0SUQ9d3JpdGU6cGV0cyxyZWFkOnBldHMg6KGo56S6IOWtmOWcqHRva2Vu6aqM6K+B77yMcHJvamVjdElE6aqM6K+B5pWw57uE5pivW3dyaXRlOnBldHMscmVhZDpwZXRzXQovLyBAcm91dGVyOiB8LQovLyAgbWV0aG9kPWdldCxwdXQgLHBvc3QsZGVsZXRlLG9wdGlvbnMsaGVhZCxwYXRjaOS4reeahOWAvDsKLy8gIHBhdGg96Lev55Sx5Zyw5Z2A77yM5L6L5aaC77yaL3VzZXIve2lkfeOAguWFtuS4rXtpZH3ooajnpLpAcGFyYW3kuK3nmoRpbuS4unBhdGjml7bnmoTlhbPogZQKZnVuYyBMb2dpbigpIHsKfQp+fn4KIyMjIyBAcGFyYW3or7TmmI4K5a6e5L6L77yaQHBhcmFtOiBpbj1wYXRoOyBuYW1lPWlkOyB0eXBlPWludGVnZXIoaW50NjQpOyByZXF1aXJlZDsgZGVzYz3kuLvplK4KLSBpbiDooajnpLrlj4LmlbDnsbvlnovvvIzlgLzmnIlxdWVyeSxoZWFkZXIscGF0aCxjb29raWUKLSBuYW1lIOihqOekuuWPguaVsOWQjeensAotIHR5cGUg6KGo56S65Y+C5pWw57G75Z6L77yM5YC85pyJaW50ZWdlcixudW1iZXIsc3RyaW5nLGJvb2xlYW4KLSByZXF1aXJlZCDmmK/lkKblv4XkvKDlj4LmlbDvvIzlrp7kvovvvJpyZXF1aXJlZCDmiJbogIUgcmVxdWlyZWQ9dHJ1ZQotIGRlc2Mg5Y+C5pWw5o+P6L+wCi0gbWluaW11bSB0eXBl57G75Z6L5pivaW50ZWdlcuaXtueahOacgOWwj+WAvAotIG1heGltdW0gdHlwZeexu+Wei+aYr2ludGVnZXLml7bnmoTmnIDlpKflgLwKLSBtaW5MZW5ndGggdHlwZeexu+Wei+aYr3N0cmluZ+aXtueahOacgOWwj+mVv+W6pgotIG1heExlbmd0aCB0eXBl57G75Z6L5pivc3RyaW5n5pe255qE5pyA5aSn6ZW/5bqmCi0gZXhhbXBsZSDlrp7kvovlgLwKLSBkZWZhdWx0IOm7mOiupOWAvAotIGVudW0g5Y+C5pWw5p6a5Li+77yM5pWw57uE77yM55SoLOWIhuWJsu+8jOS+i+Wmgu+8mmVudW09dXNlcixuYW1lCiMjIyMgQGJvZHnor7TmmI4K5a6e5L6L77yaQGJvZHk6IGluPWFwcGxpY2F0aW9uL2pzb247IGNvbnRlbnQ9dGVzdC9wcm9qZWN0L2FwcC9yZXFzLkxvZ2luQWRtaW5SZXE7IGRlc2M955So5oi35L+h5oGvCi0gaW4g5Lyg5YWl57G75Z6L77yM5YC85pyJIGFwcGxpY2F0aW9uL2pzb24sIGFwcGxpY2F0aW9uL3htbCwgYXBwbGljYXRpb24veC13d3ctZm9ybS11cmxlbmNvZGVkCi0gY29udGVudCDkvKDlhaXlhoXlrrnvvIzku6Uu5YiG5Ymy77yM5YmN57yA5Li6Z28ubW9k5p+l5om+55qE5ZG95ZCN56m66Ze05ZCN56ewKOaUr+aMgWdpdGh1Yuetie+8jOW/hemhu+W8leWFpSnvvIzlkI7nvIDkuLrnu5PmnoTkvZPlkI3np7DjgILliY3nvIDlj6/ku6XmmK/nu5PmnoTkvZNwYWNrYWdl55qE5ZCN56ew77yM6L+Z56eN5oOF5Ya15b+F6aG75LiN6IO96YeN5aSNCi0gZGVzYyDkvKDlhaXlhoXlrrnmj4/ov7AKIyMjIyBAcmVz6K+05piOCuWunuS+i++8mkByZXM6IHN0YXR1cz0yMDA7IGluPWFwcGxpY2F0aW9uL2pzb247IGNvbnRlbnQ9dGVzdC9wcm9qZWN0L2FwcC9yZXNwcy5BZG1pbkxvZ2luUmVzcDsgZGVzYz3ov5Tlm57kv6Hmga8KLSBzdGF0dXMgaW50ZWdlcuexu+Wei++8jOacjeWKoeWZqOeahOeKtuaAgeeggQotIGluIOi/lOWbnuexu+Wei++8jOWAvOaciSBhcHBsaWNhdGlvbi9qc29uLCBhcHBsaWNhdGlvbi94bWwKLSBjb250ZW50IOi/lOWbnuWGheWuue+8jOS7pS7liIblibLvvIzliY3nvIDkuLpnby5tb2Tmn6Xmib7nmoTlkb3lkI3nqbrpl7TlkI3np7Ao5pSv5oyBZ2l0aHVi562J77yM5b+F6aG75byV5YWlKe+8jOWQjue8gOS4uue7k+aehOS9k+WQjeensOOAguWJjee8gOWPr+S7peaYr+e7k+aehOS9k3BhY2thZ2XnmoTlkI3np7DvvIzov5nnp43mg4XlhrXlv4XpobvkuI3og73ph43lpI0KLSBkZXNjIOi/lOWbnuWGheWuueaPj+i/sAoKIyMjIOe7k+aehOS9k+azqOmHiuivtOaYjgp+fn5nbwpwYWNrYWdlIG1haW4KCi8vIExvZ2luUmVxdWVzdCDnu5PmnoTkvZPms6jph4oKdHlwZSBMb2dpblJlcXVlc3Qgc3RydWN0IHsKICAgIEFjY291bnQgc3RyaW5nIGBqc29uOiJhY2NvdW50IiB5YW1sOiJhY2NvdW50IiByZXF1aXJlZDoidHJ1ZSIgbWluaW11bToiMTAiYCAvLyDotKblj7fms6jph4oKCVBhc3N3b3JkIHN0cmluZyBganNvbjoicGFzc3dvcmQiIHlhbWw6InBhc3N3b3JkIiBvcGVuYXBpOiJyZXF1aXJlZDttaW5pbXVtPTEwImAgLy8g5a+G56CB5rOo6YeKCn0Kfn5+CgojIyMjIOe7k+aehOS9k+eahOagh+etvuWPr+S7peS9v+eUqCBtaW5pbXVtOiIxMCIg5ZKMIG9wZW5hcGk6Im1pbmltdW09MTAiIOi/meS4pOenjeaWueW8j++8jOWtl+auteaciQotIG1pbmltdW0gdHlwZeexu+Wei+aYr2ludGVnZXLml7bnmoTmnIDlsI/lgLwKLSBtYXhpbXVtIHR5cGXnsbvlnovmmK9pbnRlZ2Vy5pe255qE5pyA5aSn5YC8Ci0gbWluTGVuZ3RoIHR5cGXnsbvlnovmmK9zdHJpbmfml7bnmoTmnIDlsI/plb/luqYKLSBtYXhMZW5ndGggdHlwZeexu+Wei+aYr3N0cmluZ+aXtueahOacgOWkp+mVv+W6pgotIG1pbkl0ZW1zIHR5cGXnsbvlnovmmK9zbGljZeaXtueahOacgOWwj+mVv+W6pgotIG1heEl0ZW1zIHR5cGXnsbvlnovmmK9zbGljZeaXtueahOacgOWkp+mVv+W6pgotIGV4YW1wbGUg5a6e5L6L5YC8Ci0gZGVmYXVsdCDpu5jorqTlgLwKLSBlbnVtIOmZkOWumuWAvAotIHJlcXVpcmVkIOaYr+WQpuW/heS8oOWPguaVsAotIHR5cGUg57G75Z6L6YeN5a6a5LmJCgojIyDmlofku7bkuIrkvKAK5Y+q6ZyA6KaB5bCGaW7orr7nva7kuLogbXVsdGlwYXJ0L2Zvcm0tZGF0Ye+8jCDnsbvlnovorr7nva7kuLogYmFzZTY0IOaIluiAhSBiaW5hcnkg5Y2z5Y+vCgojIyDlhbPkuo4oYWJvdXQpCueBteaEn+S4uiBnaXRodWIuY29tL3N3YWdnby9zd2FnIOeahOmhueebru+8jOWboOS4uui/meS4qumhueebruaXoOazleino+aekCBvcGVuYXBpMyDnmoTmlofmoaPvvIzlm6DmraToh6rlt7Hlrp7njrDkuobkuIDlpZcgb3BlbmFwaTMg55qE5paH5qGj55Sf5oiQ readmeEtag: '"4297397ea091eb818b3f58d4a10c9e7749287206"' readmeLastModified: Fri, 17 May 2024 09:30:16 GMT repositoryId: 753528268 description: 使用ast语法解析器解析注解,根据openapi3语法生成 created: '2024-02-06T09:55:54Z' updated: '2025-05-22T08:51:01Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: goodluckxu-go logo: https://avatars.githubusercontent.com/u/110141236?v=4 license: MIT repoEtag: '"b0e44752f95951cbd36beebedf10ffe202fa2978eea91e4d5d5e4cbdc3fbe010"' repoLastModified: Thu, 22 May 2025 08:51:01 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/sami-akkawi/symfony-open-api-boilerplate v3: true repositoryMetadata: base64Readme: >- IyBXSVA6IFN5bWZvbnkgKDUuMi41KSBPcGVuQXBpICgzLjAuMykgQm9pbGVycGxhdGUKWyFbQmFkZ2U6IE1JVCBMaWNlbnNlXShwdWJsaWMvYXNzZXRzL2ltYWdlcy9taXQtbGljZW5zZS1iYWRnZS5zdmcpXShMSUNFTlNFKQpbIVtCYWRnZTogYWdpbGUuYWtrYXdpLmNoXShwdWJsaWMvYXNzZXRzL2ltYWdlcy9hZ2lsZS1ha2thd2ktY2gtYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9hZ2lsZS5ha2thd2kuY2gvYnAvNWVjMGI1N2ExZWU3ZikKCioqIE5PVElDRSAqKiBQbGVhc2Ugbm90ZSB0aGlzIGlzIHN0aWxsIGEgd29yayBpbiBwcm9ncmVzcywgYnV0IGlmIHlvdSBmaW5kIGFueXRoaW5nIG1pc3Npbmcgb3Igbm90IGNvcnJlY3QsIHBsZWFzZSBkbwogbm90IGhlc2l0YXRlIHRvIGNvbnRhY3QgbWUuCiAKIyMgSW50cm9kdWN0aW9uICMjCgpUaGlzIHByb2plY3QgaXMgYW4gYXR0ZW1wdCB0byBtYWtlIEFQSSBEb2N1bWVudGF0aW9uIGVhc2llciBhbmQgZXJyb3IgZnJlZS4gUmVxdWVzdCBwYXJhbWV0ZXJzIHNoYWxsIGJlIHZhbGlkYXRlZAogd2l0aCB0aGUgZG9jdW1lbnRhdGlvbiBhbmQgcmVzcG9uc2VzIGFzIHdlbGwuIFRoaXMgd2F5LCBkb2N1bWVudGF0aW9uIGlzIGtlcHQgdXAtdG8tZGF0ZSBhbmQgcmVzcG9uc2VzIGFyZQogZ3VhcmFudGVlZCB0byBsb29rIGxpa2UgdGhlIGRvY3VtZW50YXRpb24uIFRoaXMgd2lsbCBkZWNyZWFzZSBmcnVzdHJhdGlvbiBmcm9tIHRoZSBmcm9udGVuZCBhbmQgaW5jcmVhc2UKIHJlbGlhYmlsaXR5IGluIHRoZSBiYWNrZW5kLgogClRoZSBBUEkgRG9jdW1lbnRhdGlvbiBmb2xsb3dzIHRoZSBbT3BlbkFQSSBJbml0aWF0aXZlXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKSBhbmQgaXRzIGxhdGVzdCBbU3BlY2lmaWNhdGlvbnMKICh2My4wLjMpXShodHRwOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMyku readmeEtag: '"0e5aef6de7b68cc0becc9de5833a172ff494d224"' readmeLastModified: Mon, 17 Jun 2024 05:26:04 GMT repositoryId: 263941472 description: >- An attempt to provide an easy to use boilerplate for API documentation using OpenApi and Symfony. created: '2020-05-14T14:42:21Z' updated: '2024-06-17T05:26:09Z' language: PHP archived: false stars: 2 watchers: 1 forks: 0 owner: sami-akkawi logo: https://avatars.githubusercontent.com/u/45228655?v=4 license: MIT repoEtag: '"0f792223f2cbfc7275cd7c612c88349bc925b4e0e2ca5900556768ab03b2d065"' repoLastModified: Mon, 17 Jun 2024 05:26:09 GMT foundInMaster: true category: Parsers id: 80ddb274ded0c22f6ce8260664a579c8 oldLocations: - https://github.com/SamiAkkawi/symfony-open-api-boilerplate - source: openapi3 tags repository: https://github.com/barnuri/openapi-toolkit v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXRvb2xraXQKb3BlbmFwaS10b29sa2l0IGlzIGFuIG9wZW4tc291cmNlIHRvb2wgZGVzaWduZWQgdG8gc3RyZWFtbGluZSB0aGUgaW50ZWdyYXRpb24gb2YgT3BlbkFQSSAoZm9ybWVybHkga25vd24gYXMgU3dhZ2dlcikgc3BlY2lmaWNhdGlvbnMgaW50byB5b3VyIGRldmVsb3BtZW50IHdvcmtmbG93LiBCeSB0YWtpbmcgYW4gT3BlbkFQSS9Td2FnZ2VyIGZpbGUgYXMgaW5wdXQsIHRoZSBPcGVuQVBJIFRvb2xraXQgYXV0b21hdGljYWxseSBnZW5lcmF0ZXMgc2VydmVyIGFuZCBjbGllbnQgY29kZSwgZW5hYmxpbmcgc2VhbWxlc3MgaW50ZWdyYXRpb24gb2YgQVBJcy4gVGhpcyBhdXRvbWF0aW9uIGFjY2VsZXJhdGVzIGRldmVsb3BtZW50IHByb2Nlc3NlcywgZW5zdXJlcyBjb25zaXN0ZW5jeSBhY3Jvc3MgZGlmZmVyZW50IHBsYXRmb3JtcywgYW5kIHJlZHVjZXMgdGhlIHJpc2sgb2YgbWFudWFsIGVycm9ycy4gV2hldGhlciB5b3UncmUgYnVpbGRpbmcgYSBuZXcgc2VydmljZSBvciBpbnRlZ3JhdGluZyB3aXRoIGV4aXN0aW5nIEFQSXMsIE9wZW5BUEkgVG9vbGtpdCBzaW1wbGlmaWVzIHRoZSBwcm9jZXNzIGJ5IHByb3ZpZGluZyByZWFkeS10by11c2UgY29kZSB0YWlsb3JlZCB0byB5b3VyIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMuCgojIEluc3RhbGwKClshW1J1biBUZXN0c10oaHR0cHM6Ly9naXRodWIuY29tL2Jhcm51cmkvb3BlbmFwaS10b29sa2l0L2FjdGlvbnMvd29ya2Zsb3dzL3J1blRlc3RzLnlhbWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2Jhcm51cmkvb3BlbmFwaS10b29sa2l0L2FjdGlvbnMvd29ya2Zsb3dzL3J1blRlc3RzLnlhbWwpIFshW0NyZWF0ZSBUYWcgQW5kIFJlbGVhc2UgQW5kIFB1Ymxpc2ggVG8gTlBNXShodHRwczovL2dpdGh1Yi5jb20vYmFybnVyaS9vcGVuYXBpLXRvb2xraXQvYWN0aW9ucy93b3JrZmxvd3MvY3JlYXRlVGFnQW5kUmVsZWFzZUFuZFB1Ymxpc2gueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vYmFybnVyaS9vcGVuYXBpLXRvb2xraXQvYWN0aW9ucy93b3JrZmxvd3MvY3JlYXRlVGFnQW5kUmVsZWFzZUFuZFB1Ymxpc2gueWFtbCkKCltOUE1dKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29wZW5hcGktdG9vbGtpdCkKCmBgYGJhc2gKbnBtIGkgb3BlbmFwaS10b29sa2l0CmBgYAoKIyBBdXRvIEdlbmVyYXRlIENsaWVudC9TZXJ2ZXIgKENsaSkKCmBgYGJhc2gKbnBtIGkgLWcgb3BlbmFwaS10b29sa2l0CgojIGV4YW1wbGUKb3BlbmFwaS10b29sa2l0IC1pIGh0dHBzOi8vcGV0c3RvcmUzLnN3YWdnZXIuaW8vYXBpL3YzL29wZW5hcGkuanNvbiAtZyB0eXBlc2NyaXB0LWF4aW9zIC1vIC4vc3JjL3NlcnZpY2VzL3BldFN0b3JlIC0tbW9kZWxOYW1lUHJlZml4IE15IC0tbW9kZWxOYW1lU3VmZml4IC5kdG8KCiMgc2hvdyBhbGwgb3B0aW9ucwpvcGVuYXBpLXRvb2xraXQgLWgKCiMgd2l0aCBkb2NrZXIKZG9ja2VyIHJ1biAtLXJtIC0tbmFtZSBvcGVuYXBpLXRvb2xraXQgLXYgIiQocHdkKS9vdXRwdXQ6L291dHB1dCIgLWUgQ0xJX1BBUkFNUz0iLWkgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uIC1nIHR5cGVzY3JpcHQtYXhpb3MgLS1tb2RlbE5hbWVQcmVmaXggTXkgLS1tb2RlbE5hbWVTdWZmaXggLmR0byIgYmFybnVyaS9vcGVuYXBpLXRvb2xraXQKYGBgCgojIEF1dG8gR2VuZXJhdGUgQ2xpZW50L1NlcnZlciAoSlNcVFMpCgpgYGBqcwpjb25zdCB7IG11bHRpcGxlR2VuZXJhdGUsIGdlbmVyYXRlIH0gPSByZXF1aXJlKCdvcGVuYXBpLXRvb2xraXQnKTsKCi8vIHVzZSBtdWx0aXBsZUdlbmVyYXRlIHdoZW4geW91IHdhbnQgbXVsdGlwbGUgb3V0cHV0cwooYXN5bmMgKCkgPT4gewogICAgY29uc3Qgc2hhcmVkQ29uZmlnID0geyBkZWJ1Z0xvZ3M6IGZhbHNlIH07CiAgICBhd2FpdCBtdWx0aXBsZUdlbmVyYXRlKGBodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92My9vcGVuYXBpLmpzb25gLCBbCiAgICAgICAgeyAuLi5zaGFyZWRDb25maWcsIGdlbmVyYXRvcjogJ3R5cGVzY3JpcHQtcmVhY3QtcXVlcnknLCBvdXRwdXQ6IGAuL3R5cGVzY3JpcHQtcmVhY3QtcXVlcnkvc3JjYCB9LAogICAgICAgIHsgLi4uc2hhcmVkQ29uZmlnLCBnZW5lcmF0b3I6ICd0eXBlc2NyaXB0LWF4aW9zJywgb3V0cHV0OiBgLi90eXBlc2NyaXB0LWF4aW9zL3NyY2AgfSwKICAgICAgICB7IC4uLnNoYXJlZENvbmZpZywgZ2VuZXJhdG9yOiAndHlwZXNjcmlwdC1heGlvcycsIG91dHB1dDogYC4vdHlwZXNjcmlwdC1tb2RlbHMvc3JjYCwgbW9kZWxzT25seTogdHJ1ZSB9LAogICAgICAgIHsgLi4uc2hhcmVkQ29uZmlnLCBnZW5lcmF0b3I6ICdjIycsIG91dHB1dDogYC4vYyMvc3JjYCwgfSwKICAgICAgICB7IC4uLnNoYXJlZENvbmZpZywgZ2VuZXJhdG9yOiAnZ28nLCBvdXRwdXQ6IGAuL2dvL3NyY2AgfSwKICAgICAgICB7IC4uLnNoYXJlZENvbmZpZywgZ2VuZXJhdG9yOiAncHl0aG9uJywgb3V0cHV0OiBgLi9weXRob24vc3JjYCB9LAogICAgXSk7Cn0pKCk7CgovLyB1c2UgZ2VuZXJhdGUgd2hlbiB5b3Ugd2FudCBvbmx5IG9uZSBvdXRwdXQKKGFzeW5jICgpID0+IHsKICAgIGF3YWl0IGdlbmVyYXRlKHsgcGF0aE9yVXJsOiBgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uYCwgZ2VuZXJhdG9yOiAndHlwZXNjcmlwdC1yZWFjdC1xdWVyeScsIG91dHB1dDogYC4vdHlwZXNjcmlwdC1yZWFjdC1xdWVyeS9zcmNgIH0pOwp9KSgpOwpgYGAKCiMjIyBIZWxwIG91dHB1dAoKYGBgdGV4dApvcGVuYXBpLXRvb2xraXQgPGNvbW1hbmQ+LCBkZWZhdWx0IGNvbW1hbmQgJ2dlbmVyYXRlJwoKQ29tbWFuZHM6CiAgZ2VuZXJhdGUgICAgYXV0byBnZW5lcmF0ZSBwcm94eSBjbGllbnQgZnJvbSBzd2FnZ2VyIGZpbGUgICAgICAgICAgICAgW2RlZmF1bHRdCiAgZ2VuZXJhdG9ycyAgZ2VuZXJhdG9ycyBsaXN0CiAgY29tcGxldGlvbiAgZ2VuZXJhdGUgY29tcGxldGlvbiBzY3JpcHQKCk9wdGlvbnM6CiAgICAgIC0tdmVyc2lvbiAgICAgICAgICAgICAgICBTaG93IHZlcnNpb24gbnVtYmVyICAgICAgICAgICAgICAgICAgICAgW2Jvb2xlYW5dCiAgLWgsIC0taGVscCAgICAgICAgICAgICAgICAgICBTaG93IGhlbHAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgW2Jvb2xlYW5dCiAgLWksIC0tcGF0aE9yVXJsICAgICAgICAgICAgICBwYXRoIG9yIHVybCBmb3Igc3dhZ2dlciBmaWxlICAgICAgICAgICBbcmVxdWlyZWRdCiAgLW8sIC0tb3V0cHV0ICAgICAgICAgICAgICAgICBvdXRwdXQgcGF0aCAgICAgICAgICAgICAgICAgICAgICAgICAgICBbcmVxdWlyZWRdCiAgLWcsIC0tZ2VuZXJhdG9yICAgICAgICAgICAgICBnZW5lcmF0b3IgbmFtZSAgICAgIFtkZWZhdWx0OiAidHlwZXNjcmlwdC1heGlvcyJdCiAgLXQsIC0tdHlwZSAgICAgICAgICAgICAgICAgICBbY2hvaWNlczogImNsaWVudCIsICJzZXJ2ZXIiXSBbZGVmYXVsdDogImNsaWVudCJdCiAgLW4sIC0tbmFtZXNwYWNlICAgICAgICAgICAgICAgICAgICAgICAgIFtkZWZhdWx0OiAiT3BlbmFwaURlZmluaXRpb25HZW5lcmF0ZSJdCiAgICAgIC0tbW9kZWxzRm9sZGVyTmFtZSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbZGVmYXVsdDogIm1vZGVscyJdCiAgICAgIC0tbW9kZWxOYW1lUHJlZml4ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbZGVmYXVsdDogIiJdCiAgICAgIC0tbW9kZWxOYW1lU3VmZml4ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbZGVmYXVsdDogIiJdCiAgICAgIC0tY29udHJvbGxlcnNGb2xkZXJOYW1lICAgICAgICAgICAgICAgICAgICAgICAgICAgW2RlZmF1bHQ6ICJjb250cm9sbGVycyJdCiAgICAgIC0tY29udHJvbGxlck5hbWVQcmVmaXggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBbZGVmYXVsdDogIiJdCiAgICAgIC0tY29udHJvbGxlck5hbWVTdWZmaXggICAgICAgICAgICAgICAgICAgICAgICAgICAgIFtkZWZhdWx0OiAiQ29udHJvbGxlciJdCmBgYAoKIyMgW1ZzY29kZSBQbHVnaW4gRm9yIEF1dG8gR2VuZXJhdGVdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1CYXIuZ2VuZXJhdG9yLWZyb20tc3dhZ2dlcikKCgojIFtFeGFtcGxlc10oLi9leGFtcGxlcy9SZWFkTWUubWQpCg== readmeEtag: '"5319505243eb822b67223b81f685660e7c5f7bb9"' readmeLastModified: Thu, 15 Aug 2024 07:39:07 GMT repositoryId: 316588971 description: null created: '2020-11-27T19:56:53Z' updated: '2024-10-14T20:56:32Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 0 owner: barnuri logo: https://avatars.githubusercontent.com/u/13019522?v=4 license: BSD-3-Clause repoEtag: '"49564cc9287b402c790893727d70817a71a0756976ee659a0f8a41a062bf3db7"' repoLastModified: Mon, 14 Oct 2024 20:56:32 GMT foundInMaster: true category: SDK id: da3fab8072e31f292b75b5e935da466e - source: openapi3 tags repository: https://github.com/open-banking/scheduledpayments v3: true repositoryMetadata: base64Readme: >- IyBzY2hlZHVsZWRQYXltZW50cwpPcGVuIEJhbmtpbmcgU2NoZWR1bGVkIFBheW1lbnQgQVBJIGJ1aWx0IG9uIHRvcCBvZiBsaWdodC00ago= readmeEtag: '"00da6289de97d9cc67418c9bda5adee8aa563088"' readmeLastModified: Thu, 25 Jul 2024 20:09:54 GMT repositoryId: 235866546 description: Open Banking Scheduled Payment API built on top of light-4j created: '2020-01-23T19:09:28Z' updated: '2026-02-03T01:08:47Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"83b6ede9206deaeb33e43dd3d87c74338f91d6f236f0ecca04ed0551e94a220a"' repoLastModified: Tue, 03 Feb 2026 01:08:47 GMT foundInMaster: true category: Server id: ec401d778fc36b72fe55feae3729d030 - source: openapi3 tags repository: https://github.com/rusagaib/oas-preview.nvim v3: true id: 2d64a612248f3184b377fe072a356b49 repositoryMetadata: base64Readme: >- IVtpbWFnZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9ydXNhZ2FpYi9vYXMtcHJldmlldy5udmltKQoKIyBvYXMtcHJldmlldy5udmltCldJUCBhIHNpbXBsZSBudmltIHBsdWdpbiB0byBwcmV2aWV3IG9hcy9vcGVuYXBpczMgb3Igc3dhZ2dlciBzcGVjLCBub3cgaW5jbHVkZXMgbXVsdGlwbGUgd3JhcHBlciBjb250YWluZXItdWkgdG8gY2hvb3NlIGZvciB5b3VyIHByZWZlcmVuY2UgbGlrZSBzd2FnZ2VyLCByZWRvY2x5L3JlZG9jIGFuZCBzdG9wbGlnaHQgCgohW2ltYWdlXShkb2NzL3ByZXZpZXcucG5nKQoKIyMjIFByZXJlcXVpc2l0ZSA6c3BhcmtsZXM6Ci0tLQojIyMjIEluc3RhbGxpbmcgdGhlIFVJOgoKKipzd2FnZ2VyLXVpIChkb2NrZXIpOioqCgpgYGBzaApkb2NrZXIgcHVsbCBzd2FnZ2VyYXBpL3N3YWdnZXItdWk6bGF0ZXN0CmBgYAoqKnJlZG9jLXVpIChkb2NrZXIpOioqCgpgYGBzaApkb2NrZXIgcHVsbCByZWRvY2x5L3JlZG9jCmBgYAoKKipzdG9wbGlnaHQtdWkgKHVub2ZmaWNpYWwgZG9ja2VyKToqKgoKYGBgc2gKZG9ja2VyIHB1bGwgc2tyaXB0ZmFicmlrL2VsZW1lbnRzLWNsaQpgYGAKCiMjIyBJbnN0YWxsIFBsdWdpbiA6cm9ja2V0OgotLS0KCioqTGF6eS5udmltOioqICpyZWNvbWVuZGVkKiA6aHVnczoKCkRlZmF1bHQ6CgpgYGAKewogICAgJ3J1c2FnYWliL29hcy1wcmV2aWV3Lm52aW0nLAp9LApgYGAKCmJhc2ljIGNvbmZpZyB3aXRoIGF1dG8tc2V0dXAgIDpzcGFya2xlczoKCmBgYAp7CiAgICAncnVzYWdhaWIvb2FzLXByZXZpZXcubnZpbScsCiAgICBjb25maWcgPSBmdW5jdGlvbigpCiAgICAgICAgcmVxdWlyZSgnb2FzLXByZXZpZXcnKS5zZXR1cCh7CiAgICAgICAgICAgIHBvcnQgPSAiMTExMSIsICAgICAgICAtLSB1cC10by15b3UgCiAgICAgICAgICAgIHVpID0gInN3YWdnZXIiLCAgICAgICAtLSAic3dhZ2dlciIsICJyZWRvYyIsICJzdG9wbGlnaHQiCiAgICAgICAgICAgIGF1dG9fb3Blbl91cmwgPSBmYWxzZSAtLSBmYWxzZSB0byBkaXNhYmxlIGl0LCBkZWZhdWx0IGlzIHRydWUKICAgICAgICAgICAgZXhwb3NlID0gZmFsc2UgICAgICAgIC0tIGlmIGl0IHRydWUgd2lsbCBzZXJ2ZSBhcHAgY29udGFpbmVyIHRvIHVzZSBsb2NhbCBuZXR3b3JrIGlwIHdpdGggcG9ydCA4MCwgZGVmYXVsdCBhcmUgZmFsc2UKICAgICAgICAgICAgb3MgPSAibGludXgiICAgICAgICAgIC0tICJsaW51eCIsICJtYWMiLCAid2luIiwgIndzbCIgaWYgbm90IHNldCB3aWxsIHVzZSBkZWZhdWx0ICJsaW51eCIKICAgICAgICB9KSAgCiAgICBlbmQsCn0sCmBgYAoKKipQYWNrZXI6KioKCmFkZCB0aGlzIGxpbmUgb24geW91ciBwYWNrZXIgY29uZmlnLmx1YSBqdXN0IGFkZCBydXNhZ2FpYi9vYXMtcHJldmlldy5udmltCgpgYGAKdXNlIHsncnVzYWdhaWIvb2FzLXByZXZpZXcubnZpbSd9CmBgYAoKKipQbHVnOioqCgphZGQgdGhpcyBsaW5lIG9uIHlvdXIgY29uZmlnCgpgYGAKUGx1ZyAncnVzYWdhaWIvb2FzLXByZXZpZXcubnZpbScKYGBgCgoKIyMjIEFwcGx5L3NldHVwIG9hcy1wcmV2aWV3IHBsdWdpbiA6Ym93OgotLS0KCmlmIHlvdSdyZSBub3QgdXNpbmcgTGF6eS5udmltIGFzIHBsdWdpbiBtYW5hZ2VyIHRoaXMgc2VjdGlvbiBpcyBuZWVkZWQuLgoKSW4geW91ciBpbml0Lmx1YSAoRGVmYXVsdCBudmltIGNvbmZpZyk6CgpgYGAKIyB3aXRoIERlZmF1bHQgY29uZmlnIHlvdSdsbCBnZXQgdGhpcyBjb25mIAojIHBvcnQgPSAiMTExMSIsIAojIHVpID0gInN3YWdnZXIiLCAKIyBhdXRvX29wZW5fdXJsID0gdHJ1ZSwgCiMgZXhwb3NlID0gZmFsc2UKIyBvcyA9ICJsaW51eCIKcmVxdWlyZSgnb2FzLXByZXZpZXcnKS5zZXR1cCh7fSkKYGBgCgp+IE9SIH4KCmBgYAojIGNob29zZSB5b3VyIHByZWZlcmVuY2VzIHVpLCBkaXNhYmxlIGF1dG9fb3Blbl91cmwsIGRpc2FibGUgZXhwb3NlIGhvc3QgY29udGFpbmVyLXVpIGV0Yy4uCnJlcXVpcmUoJ29hcy1wcmV2aWV3Jykuc2V0dXAoewogICAgcG9ydD0iMjIyMiIsICAgICAgICAgLS0gb3IgYW55IHBvcnQgeW91IHdhbnQgCiAgICB1aT0ic3RvcGxpZ2h0IiwgICAgICAtLSBvcHRpb24gdWk6IHN3YWdnZXIsIHJlZG9jLCBzdG9wbGlnaHQgCiAgICBhdXRvX29wZW5fdXJsPWZhbHNlLCAtLSB5b3UgY2FuIGNob29zZSB0cnVlIG9yIGZhbHNlCiAgICBleHBvc2U9ZmFsc2UgICAgICAgICAtLSBkZWZhdWx0IGZhbHNlIHRobywgYnV0IGlmIHlvdSBuZWVkIHRvIGV4cG9zZSBpdCBmb3IgbG9jYWwgbmV0d29yayB0ZXN0IGV0YywgeW91IGNhbiBzZXQgaXQgdG8gdHJ1ZQogICAgb3M9Im1hYyIgICAgICAgICAgICAgLS0gImxpbnV4IiwgIm1hYyIsICJ3aW4iLCAid3NsIiBpZiBub3Qgc2V0IHdpbGwgdXNlIGRlZmF1bHQgImxpbnV4Igp9KQpgYGAKCiMjIyBSdW4gOmZpcmU6Ci0tLQoKCmBgYAo6T0FTUHJldmlldwpgYGAKCgojIyMgU3RvcCBvcHRzIDp0cmlhbmd1bGFyX2ZsYWdfb25fcG9zdDoKLS0tCgp3aWxsIHN0b3AgYnVmZmVyIGNvbW1hbmRzICYgZGVsZXRlIGNvbnRhaW5lci11aQoKYGBgCjpPQVNQcmV2aWV3U3RvcApgYGAKCgojIyMgQ2hlY2sgQ29uZmlnIDpnZWFyOiAKLS0tCgp3aWxsIHByaW50IG91dCB5b3VyIHNldCBjb25maWcgb24gb2FzLXByZXZpZXcgIAoKYGBgCjpPQVNQcmV2aWV3Q29uZgpgYGAKCiMjIyBDaGVjayBSdW5pbmcgU2VydmljZSBjb250YWluZXItdWkgOnBhY2thZ2U6IAotLS0KCndpbGwgcHJpbnQgb3V0IHlvdXIgY29udGFpbmVyLXVpIGluZm8gCgpgYGAKOk9BU1ByZXZpZXdDaGVjawpgYGAKCi0tLQoKCiMjIyBOb3RlIGZvciB3c2wgdXNlcnMgeW91IG5lZWQgaW5zdGFsbCB3c2x2aWV3IHRvIGV4ZWMgeGRnLW9wZW4gYnJvd3NlcjogCi0tLQoKIyMjIyBbVWJ1bnR1IHdzbDJdKGh0dHBzOi8vd3NsdS53ZWRvdHN0dWQuaW8vd3NsdS9pbnN0YWxsLmh0bWwjdWJ1bnR1KSA6CgpgYGBzaApzdWRvIGFkZC1hcHQtcmVwb3NpdG9yeSBwcGE6d3NsdXRpbGl0aWVzL3dzbHUKc3VkbyBhcHQgdXBkYXRlCnN1ZG8gYXB0IGluc3RhbGwgd3NsdQpgYGAKCipmb3IgYW55IGRpc3RybyB3c2wgeW91IGNhbiB2aWV3IG90aGVyIGluc3RhbGxhdGlvbiBndWlkZSBbaGVyZV0oaHR0cHM6Ly93c2x1LndlZG90c3R1ZC5pby93c2x1L2luc3RhbGwuaHRtbCkKCi0tLQoKCiMjIyMgQW55IGNvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWUgOmJlZXI6Cgo= readmeEtag: '"4124b7059198ff148a6ab29946a4d8928b1d6096"' readmeLastModified: Mon, 08 Sep 2025 21:30:28 GMT repositoryId: 545944120 description: WIP a simple nvim/neovim plugin to preview oas/openapi3 or swagger spec. created: '2022-10-05T08:46:31Z' updated: '2025-09-08T21:31:25Z' language: Lua archived: false stars: 2 watchers: 1 forks: 1 owner: rusagaib logo: https://avatars.githubusercontent.com/u/33116863?v=4 license: MIT repoEtag: '"e43e83f4636d22c46eef5d0736164975e204a54a5ae056ff73e62044207ce856"' repoLastModified: Mon, 08 Sep 2025 21:31:25 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mneiferbag/java-spring-boot v3: true id: cca0fb34e191210e1afb499f6117f02b repositoryMetadata: base64Readme: >- IyBKYXZhIFNwcmluZyBFeGFtcGxlCgpMaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSBmaWxlIFtMSUNFTlNFXSguL0xJQ0VOU0UpLgoKSmF2YSBbU3ByaW5nXShodHRwczovL3NwcmluZy5pby8pIGV4YW1wbGUuIEV4YW1wbGUgaW5jbHVkZXMgY29kZSB3cml0dGVuIGluIEphdmEuCgpbIVtDb2RlUUxdKGh0dHBzOi8vZ2l0aHViLmNvbS9tbmVpZmVyYmFnL2phdmEtc3ByaW5nLWJvb3QvYWN0aW9ucy93b3JrZmxvd3MvY29kZXFsLWFuYWx5c2lzLnltbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vbW5laWZlcmJhZy9qYXZhLXNwcmluZy1ib290L2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVxbC1hbmFseXNpcy55bWwpClshW0phdmEgQ0kgd2l0aCBHcmFkbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9tbmVpZmVyYmFnL2phdmEtc3ByaW5nLWJvb3QvYWN0aW9ucy93b3JrZmxvd3MvZ3JhZGxlLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbW5laWZlcmJhZy9qYXZhLXNwcmluZy1ib290L2FjdGlvbnMvd29ya2Zsb3dzL2dyYWRsZS55bWwpCgojIyBTcHJpbmcgQm9vdAoKVGhlIFN3YWdnZXIgVUkgcGFnZSBpcyBhdmFpbGFibGUgYXQKCltodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS5odG1sXShodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS5odG1sKQoKYW5kIHRoZSBPcGVuQVBJIGRlc2NyaXB0aW9uIGlzIGF2YWlsYWJsZSBhdAoKW2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MC92My9hcGktZG9jc10oaHR0cDovL2xvY2FsaG9zdDo4MDgwL3YzL2FwaS1kb2NzKQoKIyMgR3JhZGxlCgpDaGVjayB2ZXJzaW9uIHdpdGggY29tbWFuZCBgZ3JhZGxlIC12YC4KClJ1biBgZ3JhZGxldyB3cmFwcGVyIC0tZ3JhZGxlLXZlcnNpb24gbGF0ZXN0YCB0byB1cGRhdGUgdGhlIHByb2plY3QgdG8gbGF0ZXN0IHZlcnNpb24uCgpSdW4gYGdyYWRsZXcgd3JhcHBlciAtLWdyYWRsZS12ZXJzaW9uIDcuNC4yYCB0byB1cGRhdGUgdGhlIHByb2plY3QgdG8gNy40LjIuCgojIyBMaW5rcwoKKiBPcGVuQVBJCiAgKiBbT3BlbkFQSSAzIExpYnJhcnkgZm9yIFNwcmluZyBCb290XShodHRwczovL3NwcmluZ2RvYy5vcmcvKQogICogW09wZW5BUEkgSW5pdGlhdGl2ZSAoT0FJKV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykKICAqIFtPcGVuQVBJIFNwZWMgMy4wLjFdKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMSkKICAqIFtTd2FnZ2VyIENvZGVnZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWNvZGVnZW4pCiAgKiBbU3dhZ2dlciBFZGl0b3JdKGh0dHA6Ly9lZGl0b3Iuc3dhZ2dlci5pby8pCiAgKiBbU3dhZ2dlciBWYWxpZGF0b3JdKGh0dHBzOi8vdmFsaWRhdG9yLnN3YWdnZXIuaW8vKQoqIFRlc3RpbmcKICAqIFtDb3JudXR1bS90Y2FzZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9Db3JudXR1bS90Y2FzZXMpIC0gQSBtb2RlbC1iYXNlZCB0ZXN0IGNhc2UgZ2VuZXJhdG9yCiAgKiBbUkVTVCBBc3N1cmVkXShodHRwOi8vcmVzdC1hc3N1cmVkLmlvLykgLSBUZXN0aW5nIGFuZCB2YWxpZGF0aW5nIFJFU1Qgc2VydmljZXMgaW4gSmF2YQogICogW1Rlc3RpbmcgdGhlIFdlYiBMYXllcl0oaHR0cHM6Ly9zcHJpbmcuaW8vZ3VpZGVzL2dzL3Rlc3Rpbmctd2ViLykgLSBTcHJpbmcgR3VpZGUKKiBHZW5lcmFsCiAgKiBbR2l0SHViIEFjdGlvbiBTZXR1cCBKYXZhXShodHRwczovL2dpdGh1Yi5jb20vYWN0aW9ucy9zZXR1cC1qYXZhKSBmb3IgdXNlIHdpdGggR2l0SHViIENvZGVRTAogICogW0dyYWRsZSBCdWlsZCBUb29sXShodHRwczovL2dyYWRsZS5vcmcvKQogICogW0dyZXR0eV0oaHR0cHM6Ly9ncmV0dHktZ3JhZGxlLXBsdWdpbi5naXRodWIuaW8vZ3JldHR5LWRvYy9hYm91dC5odG1sKSAtIGEgZmVhdHVyZS1yaWNoIGdyYWRsZSBwbHVnaW4gZm9yIHJ1bm5pbmcgd2ViLWFwcHMgb24gZW1iZWRkZWQgc2VydmxldCBjb250YWluZXJzCiAgKiBbSmF2YSBEZXZlbG9wbWVudCBFeGFtcGxlc10oaHR0cHM6Ly9naXRodWIuY29tL21uZWlmZXJiYWcvamF2YS1leGFtcGxlcykKCiMjIFRhc2tzCgotIFsgXSBVcGdyYWRlIHRvIGxhdGVzdCB2ZXJzaW9ucyBvZiBKYXZhLCBTcHJpbmcgQm9vdCwgR3JhZGxlLCBzZWUgW1VwZ3JhZGUgR3VpZGVdKC4vZG9jL3VwZ3JhZGVfZ3VpZGUuYWRvYykKLSBbIF0gQWRkIEpldHR5IGV4ZWN1dGlvbiBpbiBjaGVjayB0YXNrIGJlZm9yZSBpbnRlZ3JhdGlvbiB0ZXN0cyBhcmUgcnVuCi0gWyBdIENvbXBhcmUgZ2VuZXJhdGVkIG9wZW5hcGkueWFtbCBhbmQgYXBpX3YzLnlhbWwKLSBbIF0gQWRkIEdyYWRsZSBpbnRlZ3JhdGlvbiB0ZXN0cwogIC0gaHR0cHM6Ly9kb2NzLmdyYWRsZS5vcmcvY3VycmVudC91c2VyZ3VpZGUvbWlncmF0aW5nX2Zyb21fbWF2ZW4uaHRtbCNtaWdtdm46YnVpbGRfbGlmZWN5Y2xlCiAgLSBodHRwczovL2RvY3MuZ3JhZGxlLm9yZy9jdXJyZW50L3VzZXJndWlkZS9taWdyYXRpbmdfZnJvbV9tYXZlbi5odG1sI21pZ212bjppbnRlZ3JhdGlvbl90ZXN0cwogIC0gaHR0cHM6Ly9kb2NzLmdyYWRsZS5vcmcvY3VycmVudC91c2VyZ3VpZGUvamF2YV90ZXN0aW5nLmh0bWwjc2VjOmNvbmZpZ3VyaW5nX2phdmFfaW50ZWdyYXRpb25fdGVzdHMgCg== readmeEtag: '"7ef1418ae143b04c96c903ef4c9161a92ce702a2"' readmeLastModified: Fri, 06 Dec 2024 12:48:46 GMT repositoryId: 359082805 description: Java Spring development example. created: '2021-04-18T08:09:17Z' updated: '2025-11-24T17:30:43Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: mneiferbag logo: https://avatars.githubusercontent.com/u/16717541?v=4 license: MIT repoEtag: '"36a9f8bda9b83c4599a8a0a31c56567a65c9a297ba5985fdb7142d40e1fedaac"' repoLastModified: Mon, 24 Nov 2025 17:30:43 GMT category: - SDK - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/kazuki/pyramid-oas3 v3: true repositoryMetadata: base64Readme: >- cHlyYW1pZF9vYXMzCj09PT09PT09PT09PQoKW1B5cmFtaWRdKGh0dHBzOi8vdHJ5cHlyYW1pZC5jb20vKSBXZWLjgqLjg5fjg6rjgrHjg7zjgrfjg6fjg7Pjgavlr77jgZfjgabjgIEKW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pIFszLjBdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL3ZlcnNpb25zLzMuMC4wLm1kKeOCkuWIqeeUqOOBl+OBn+OAgQrjg6rjgq/jgqjjgrnjg4jjga7mpJzoqLzjg7vjg5Hjg7zjgrnjgpLlrp/mlr3jgZnjgovjg6njgqTjg5bjg6njg6rjgafjgZnjgIIKCltweXJhbWlkX3N3YWdnZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9zdHJpZ2xpYS9weXJhbWlkX3N3YWdnZXIp44GuT3BlbkFQSSAzLjDlr77lv5zniYjjga7mp5jjgarkvY3nva7jgaXjgZHjgafjgZnjgIIKCuioreWumumgheebrgotLS0tLS0tLQoKKiBweXJhbWlkX29hczMudmFsaWRhdGVfcmVzcG9uc2U6IGJvb2wKICAqIOODrOOCueODneODs+OCueOBrkpTT07jgoLmpJzoqLzjgZnjgovjgYvjgpLoqK3lrprjgZfjgb7jgZko44OH44OV44Kp44Or44OIOiBGYWxzZSkKKiBweXJhbWlkX29hczMuZmlsbF9ieV9kZWZhdWx0OiBib29sCiAgKiDjg6rjgq/jgqjjgrnjg4jjg4fjg7zjgr/jgavlr77jgZfjgaZPcGVuQVBJ5a6a576p44Gn6Kit5a6a44GV44KM44GfZGVmYXVsdOWApOOBp+Wfi+OCgeOCi+OBi+OCkuioreWumuOBl+OBvuOBmSjjg4fjg5Xjgqnjg6vjg4g6IEZhbHNlKQoqIHB5cmFtaWRfb2FzMy5yZXNwb25zZV9yZXZpdmVyOiBPcHRpb25hbFtDYWxsYWJsZVtbVW5pb25baW50LCBzdHJdLCBKU09OX1RZUEVTXSwgVW5pb25bSlNPTl9UWVBFUywgcHlyYW1pZF9vYXMzLlVOREVGSU5FRF1dXQogICogSlNPTl9UWVBFUzogVW5pb25baW50LCBmbG9hdCwgc3RyLCBib29sLCBOb25lLCBsaXN0LCBkaWN0XQogICogW0pTT04ucGFyc2VdKGh0dHBzOi8vZGV2ZWxvcGVyLm1vemlsbGEub3JnL2phL2RvY3MvV2ViL0phdmFTY3JpcHQvUmVmZXJlbmNlL0dsb2JhbF9PYmplY3RzL0pTT04vcGFyc2Up44GucmV2aXZlcuOBqOWQjOanmOOBruWHpueQhuOCkuOAgQogICAg44Os44K544Od44Oz44K544K544Kt44O844Oe5qSc6Ki85a6f6KGM5YmN44Gr6YGp55So44GZ44KL44KI44GG44Gr44GX44G+44GZCiAgKiDjg4fjg5Xjgqnjg6vjg4g6IE5vbmUgKHJldml2ZXLjgpLpgannlKjjgZfjgarjgYQpCgrkvb/jgYTmlrkKLS0tLS0tCgrku6XkuIvjga7jgojjgYbjgavoqK3lrprjgZfjgaZweXJhbWlk44Gr57WE44G/6L6844G/44G+44GZCgpgYGAKc2V0dGluZ3MgPSB7CiAgICAncHlyYW1pZC5pbmNsdWRlcyc6ICdweXJhbWlkX29hczMnLAogICAgJ3B5cmFtaWRfb2FzMy5zY2hlbWEnOiB5YW1sLmxvYWQob3Blbignc2NoZW1hLnlhbWwnKS5yZWFkKCkpLAogICAgJ3B5cmFtaWRfb2FzMy52YWxpZGF0ZV9yZXNwb25zZSc6IFRydWUsCiAgICAncHlyYW1pZF9vYXMzLmZpbGxfYnlfZGVmYXVsdCc6IFRydWUsCn0KYGBgCgrmpJzoqLzlpLHmlZfmmYLjga/ku6XkuIvjga7kvovlpJbjgYzov5TljbTjgZXjgozjgovjga7jgafjgIEK6YGp5a6cZXhjZXB0aW9uX3ZpZXdfY29uZmln44KS6Kit5a6a44GX44Gm44GP44Gg44GV44GECgoqIHB5cmFtaWRfb2FzMy5WYWxpZGF0aW9uRXJyb3JzOiDjg6rjgq/jgqjjgrnjg4jjg4fjg7zjgr/jga7jgrnjgq3jg7zjg57mpJzoqLzlpLHmlZfmmYIKKiBweXJhbWlkX29hczMuUmVzcG9uc2VWYWxpZGF0aW9uRXJyb3I6IOODrOOCueODneODs+OCueODh+ODvOOCv+OBruOCueOCreODvOODnuaknOiovOWkseaVl+aZggoqIHB5cmFtaWQuaHR0cGV4Y2VwdGlvbnMuSFRUUEJhZFJlcXVlc3Q6IOOCr+OCqOODquaWh+Wtl+WIl+OBruODkeODvOOCueWkseaVl+aZggoqIHB5cmFtaWQuaHR0cGV4Y2VwdGlvbnMuSFRUUE5vdEFjY2VwdGFibGU6IOODquOCr+OCqOOCueODiOOBrkNvbnRlbnRUeXBl44Gr5a++5b+c44GZ44KL44KC44Gu44GM44K544Kt44O844Oe44Gr44Gq44GECu+8inB5cmFtaWQuaHR0cGV4Y2VwdGlvbnMuSFRUUFVuYXV0aG9yaXplZDog44Oq44Kv44Ko44K544OI44Gr6KqN6Ki85oOF5aCx44GM44Gq44GECgrku6XkuIvjgatleGNlcHRpb25fdmlld19jb25maWfjga7kvovjgpLnpLrjgZfjgb7jgZkKCmBgYApmcm9tIHB5cmFtaWQudmlldyBpbXBvcnQgZXhjZXB0aW9uX3ZpZXdfY29uZmlnCgoKQGV4Y2VwdGlvbl92aWV3X2NvbmZpZyhWYWxpZGF0aW9uRXJyb3JzKQpkZWYgZmFpbGVkX3JlcXVlc3RfdmFsaWRhdGlvbihleGMsIHJlcXVlc3QpOgogICAgcmVzID0gUmVzcG9uc2Uoc3RyKGV4YykpCiAgICByZXMuc3RhdHVzX2ludCA9IDQwMAogICAgcmV0dXJuIHJlcwoKCkBleGNlcHRpb25fdmlld19jb25maWcoUmVzcG9uc2VWYWxpZGF0aW9uRXJyb3IpCmRlZiBmYWlsZWRfcmVzcG9uc2VfdmFsaWRhdGlvbihleGMsIHJlcXVlc3QpOgogICAgcmVzID0gUmVzcG9uc2Uoc3RyKGV4YykpCiAgICByZXMuc3RhdHVzX2ludCA9IDUwMAogICAgcmV0dXJuIHJlcwpgYGAK readmeEtag: '"45a75ae5ea6328152c11af6b0ad211d723675ad2"' readmeLastModified: Thu, 25 Apr 2019 13:40:30 GMT repositoryId: 112933976 description: null created: '2017-12-03T14:18:57Z' updated: '2019-04-26T06:43:22Z' language: Python archived: false stars: 2 watchers: 2 forks: 0 owner: kazuki logo: https://avatars.githubusercontent.com/u/163226?v=4 license: MIT repoEtag: '"0f2a2e918ea0fcca55c199547d678bcfebd7cf6e4e808cc3683780a4cc41f462"' repoLastModified: Fri, 26 Apr 2019 06:43:22 GMT foundInMaster: true category: Data Validators id: b74d0bbb2c4a0daba1f64cbd0694fb89 - source: openapi3 tags repository: https://github.com/essentialprogramming/spring-boot-openapi v3: true repositoryMetadata: base64Readme: >- IyMgU2FtcGxlIFNwcmluZyBCb290IFJFU1RmdWwgc2VydmljZSB1c2luZyBPcGVuQVBJIDMKCiMjIyBTdGVwIDEgLSBBZGRpbmcgT3BlbkFQSSB0byBvdXIgcHJvamVjdAoKRmlyc3QsIHdlIG5lZWQgdG8gYWRkIHRoZSBkZXBlbmRlbmN5IHRvIG91ciBwb20ueG1sOgpgYGAKPGRlcGVuZGVuY3k+CiAgICA8Z3JvdXBJZD5vcmcuc3ByaW5nZG9jPC9ncm91cElkPgogICAgPGFydGlmYWN0SWQ+c3ByaW5nZG9jLW9wZW5hcGktdWk8L2FydGlmYWN0SWQ+CiAgICA8dmVyc2lvbj4xLjUuMTA8L3ZlcnNpb24+CjwvZGVwZW5kZW5jeT4KYGBgCgpUaGVuIHdlIG5lZWQgdG8gYWRkIHRoZSBjb25maWd1cmF0aW9uIGNsYXNzIHRvIG91ciBwcm9qZWN0OgpgYGAKQENvbmZpZ3VyYXRpb24KcHVibGljIGNsYXNzIFN3YWdnZXJDb25maWcgewoKICAgIC8qCiAgICBVc2VyIEFQSQogICAgICovCiAgICBAQmVhbgogICAgcHVibGljIEdyb3VwZWRPcGVuQXBpIHVzZXJBcGkoKSB7CiAgICAgICAgZmluYWwgU3RyaW5nW10gcGFja2FnZXNUb1NjYW4gPSB7ImNvbS5jb250cm9sbGVyIn07CiAgICAgICAgcmV0dXJuIEdyb3VwZWRPcGVuQXBpCiAgICAgICAgICAgICAgICAuYnVpbGRlcigpCiAgICAgICAgICAgICAgICAuZ3JvdXAoIlVzZXIgQVBJIikKICAgICAgICAgICAgICAgIC5wYWNrYWdlc1RvU2NhbihwYWNrYWdlc1RvU2NhbikKICAgICAgICAgICAgICAgIC5wYXRoc1RvTWF0Y2goIi91c2Vycy8qKiIpCiAgICAgICAgICAgICAgICAuYWRkT3BlbkFwaUN1c3RvbWlzZXIoc3RhdHVzQXBpQ29zdHVtaXplcigpKQogICAgICAgICAgICAgICAgLmJ1aWxkKCk7CiAgICB9CgogICAgcHJpdmF0ZSBPcGVuQXBpQ3VzdG9taXNlciBzdGF0dXNBcGlDb3N0dW1pemVyKCkgewogICAgICAgIHJldHVybiBvcGVuQVBJIC0+IG9wZW5BUEkKICAgICAgICAgICAgICAgIC5pbmZvKG5ldyBJbmZvKCkKICAgICAgICAgICAgICAgICAgICAgICAgLnRpdGxlKCJTcHJpbmdib290ICYgT3BlbkFQSSIpCiAgICAgICAgICAgICAgICAgICAgICAgIC5kZXNjcmlwdGlvbigiVGhpcyBpcyBhIHNhbXBsZSBTcHJpbmcgQm9vdCBSRVNUZnVsIHNlcnZpY2UgdXNpbmcgT3BlbkFQSSIpCiAgICAgICAgICAgICAgICAgICAgICAgIC52ZXJzaW9uKCIzLjAuMCIpCiAgICAgICAgICAgICAgICAgICAgICAgIC5jb250YWN0KG5ldyBDb250YWN0KCkKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAubmFtZSgiUmF6dmFuIFByaWNoaWNpIikKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAudXJsKCJodHRwczovL2dpdGh1Yi5jb20vZXNzZW50aWFscHJvZ3JhbW1pbmciKQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIC5lbWFpbCgicmF6dmFucGF1bHBAZ21haWwuY29tIikpKTsKICAgIH0KCiAgICBAQmVhbgogICAgcHVibGljIE9wZW5BUEkgY3VzdG9tT3BlbkFQSSgpIHsKICAgICAgICByZXR1cm4gbmV3IE9wZW5BUEkoKQogICAgICAgICAgICAgICAgLmNvbXBvbmVudHMobmV3IENvbXBvbmVudHMoKSkKICAgICAgICAgICAgICAgIC5pbmZvKG5ldyBJbmZvKCkudGl0bGUoIkNvbnRhY3QgQXBwbGljYXRpb24gQVBJIikuZGVzY3JpcHRpb24oCiAgICAgICAgICAgICAgICAgICAgICAgICJUaGlzIGlzIGEgc2FtcGxlIFNwcmluZyBCb290IFJFU1RmdWwgc2VydmljZSB1c2luZyBzcHJpbmdkb2Mtb3BlbmFwaSBhbmQgT3BlbkFQSSAzLiIpKTsKICAgIH0KCn0KYGBgCgojIyMgU3RlcCAyIC0gTG9va2luZyBhdCBTd2FnZ2VyCgpGb2xsb3dpbmcgVVJMJ3MgYXJlIG5vdyBhd2FpbGFibGUuCgpUaGUgZmlyc3Qgb25lIHJlcHJlc2VudHMgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbi4KCltodHRwOi8vbG9jYWxob3N0OjgwODAvdjMvYXBpLWRvY3NdKGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC92My9hcGktZG9jcykKCiFbQVBJIERvY3NdKHNyYy9tYWluL3Jlc291cmNlcy9pbWcvYXBpLWRvY3MucG5nKQoKVGhlIHNlY29uZCBvbmUgcmVwcmVzZW50cyB0aGUgU3dhZ2dlciBVSSwgd2hpY2ggd2UgY2FuIG5vdyB1c2UgdG8gaW52b2tlIGFuZCBleHBsb3JlIG91ciBBUEkuCgpbU3dhZ2dlciBVSV0oaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkuaHRtbCkKCiFbU3dhZ2dlciBVSV0oc3JjL21haW4vcmVzb3VyY2VzL2ltZy9zd2FnZ2VyLXVpLnBuZykKCkZvciBhbiBlYXNpZXIgYWNjZXNzLCB3ZSBjYW4gY2hhbmdlIHRoZSBTd2FnZ2VyIFVJIFVSTCB0byBhbnl0aGluZyB0aGF0IHdlIHdhbnQuCgpGb3IgdGhhdCwgd2UgbmVlZCB0byBhY2Nlc3MgYGFwcGxpY2F0aW9uLnByb3BlcnRpZXNgIGFuZCBhZGQgdGhlIGZvbGxvd2luZyBwcm9wZXJ0eToKCmBzcHJpbmdkb2Muc3dhZ2dlci11aS5wYXRoPS9hcGlkb2NgCgpUaGlzIHdpbGwgc2V0IG91ciBTd2FnZ2VyIFVJIFVSTCB0bzogW2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGlkb2NdKGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGlkb2MpCgojIyMgU3RlcCAzIC0gQWRkIGFubm90YXRpb25zIHRvIGVuZHBvaW50cwoKSW4gb3JkZXIgZm9yIG91ciBlbmRwb2ludHMgdG8gYmUgdmlzaWJsZSBpbiBTd2FnZ2VyLCB3ZSBuZWVkIHRvIG1hcmsgZWFjaCBvZiB0aGVtIHdpdGggYEBPcGVyYXRpb25gIHRhZy4KCkhlcmUgaXMgYW4gZXhhbXBsZToKCmBgYAogICAgQFBvc3RNYXBwaW5nKCIvdXNlcnMiKQogICAgQE9wZXJhdGlvbihzdW1tYXJ5ID0gIlJlZ2lzdGVyIGEgbmV3IHVzZXIiLCB0YWdzID0geyJVc2VyIix9LAogICAgICAgICAgICByZXNwb25zZXMgPSB7CiAgICAgICAgICAgICAgICAgICAgQEFwaVJlc3BvbnNlKHJlc3BvbnNlQ29kZSA9ICIyMDAiLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgZGVzY3JpcHRpb24gPSAiUmV0dXJucyB0aGUgcmVnaXN0ZXJlZCB1c2VyIiwKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGNvbnRlbnQgPSBAQ29udGVudChtZWRpYVR5cGUgPSAiYXBwbGljYXRpb24vanNvbiIsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNjaGVtYSA9IEBTY2hlbWEoaW1wbGVtZW50YXRpb24gPSBVc2VyLmNsYXNzKSkpCiAgICAgICAgICAgIH0pCiAgICBAUmVzcG9uc2VCb2R5CiAgICBwdWJsaWMgVXNlciByZWdpc3RlcihAUmVxdWVzdFBhcmFtKG5hbWUgPSAibmFtZSIsIHJlcXVpcmVkID0gZmFsc2UsIGRlZmF1bHRWYWx1ZSA9ICJTdHJhbmdlciIpIFN0cmluZyBuYW1lKSB7CiAgICAgICAgVXNlciBuZXdVc2VyID0gbmV3IFVzZXIoY291bnRlci5pbmNyZW1lbnRBbmRHZXQoKSwgbmFtZSk7CiAgICAgICAgcmV0dXJuIHVzZXJSZXBvc2l0b3J5LmFkZFVzZXIobmV3VXNlcik7CiAgICB9CmBgYAoKIyMjIFN0ZXAgNCAtIEdlbmVyYXRlIGEgY2xpZW50CgpUaGUgYWR2YW50YWdlIG9mIGhhdmluZyB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGF2YWlsYWJsZSBpcyB0aGF0CmdlbmVyYXRpbmcgYSBjbGllbnQgY29kZSB0byBjYWxsIG91ciBBUEkgaXMgbm93IHF1aXRlIGVhc3kuCgpGb3IgdGhpcywgd2UgY2FuIHVzZSB2YXJpb3VzIHRvb2xzIHN1Y2ggYXM6IFtPcGVuQVBJIEdlbmVyYXRvcl0oaHR0cHM6Ly9vcGVuYXBpLWdlbmVyYXRvci50ZWNoLykK readmeEtag: '"d4b118ef37a6eca8ebc8d3a2e785743e7156776c"' readmeLastModified: Wed, 15 Jun 2022 06:15:51 GMT repositoryId: 408620019 description: Documenting a Spring REST API Using OpenAPI 3.0 created: '2021-09-20T22:38:37Z' updated: '2024-10-10T00:06:56Z' language: Java archived: false stars: 2 watchers: 1 forks: 6 owner: essentialprogramming logo: https://avatars.githubusercontent.com/u/46305342?v=4 license: GPL-3.0 repoEtag: '"1d53aa5ec4761dd65b952a7a9fd0db7d2b227e780938eafe0a65f2b8fc9129c0"' repoLastModified: Thu, 10 Oct 2024 00:06:56 GMT foundInMaster: true category: - SDK - Parsers id: 91740500afb5ab47cf98d36b27a38045 - source: openapi3 tags repository: https://github.com/chunhei2008/swagger-php-annotation v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyLVBIUCBBbm5vdGF0aW9uCgojIyBJbnN0YWxsCgrlronoo4Xmj5Lku7bvvJoKcGhwc3Ryb206IFBIUCBBbm5vdGF0aW9uCgrlhbfkvZNgdGVtcGxhdGVzYOi3r+W+hOWPguiAg1vmlofmoaNdKGh0dHBzOi8vd3d3LmpldGJyYWlucy5jb20vaGVscC9waHBzdG9ybS90dW5pbmctdGhlLWlkZS5odG1sI2RlZmF1bHQtZGlycykKCmBgYHNoCgpnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2NodW5oZWkyMDA4L3N3YWdnZXItcGhwLWFubm90YXRpb24uZ2l0CgpjZCBzd2FnZ2VyLXBocC1hbm5vdGF0aW9uCgojIOWwhmBzd2FnZ2VyLXBocC1hbm5vdGF0aW9uLnhtbGDlpI3liLbliLBgdGVtcGxhdGVzYOebruW9lQpjcCBzd2FnZ2VyLXBocC1hbm5vdGF0aW9uLnhtbCBbcGhwc3Ryb20vdGVtcGxhdGVzL3BhdGhdCgojIHJlc3RhcnQgeW91ciBwaHBzdHJvbQoKYGBgCgojIyBVc2FnZQoKIyMjIOS4iuS4i+aWh+ivtOaYjgoKYHN3YWdnZXItcGhwLWFubm90YXRpb25g5ZyoYHBocHN0cm9tYOS4reWIhuS4uuS4pOenjeS4iuS4i+aWhwoKMS4g5rOo6YeK5LiK5LiL5paH77ya5rOo6YeK5Z2X5LitCjIuIOmdnuazqOmHiuS4iuS4i+aWh++8muazqOmHiuWdl+S5i+WklueahOepuueZveWkhAoK5ZCM5LiA5Liq5rOo6Kej5YiG5Lik56eN5LiK5LiL5paH5L2/55So5Zy65pmv77yM5LiN5ZCM5Zy65pmv5L2/55So5LiN5ZCM55qE5qCH6K+G56ymCgrlpoLvvJoKCmBAR2V0YCDlkowgYEBHZXRfYCDpg73mmK9IVFRQIGBHRVRgIOivt+axggoKYEBHZXRfYCDkuLrms6jph4rkuIrkuIvmlofkvb/nlKgKYEBHZXRgIOS4uumdnuazqOmHiuS4iuS4i+aWh+S9v+eUqAoKCiMjIyDmlK/mjIHnmoTms6jop6PliJfooagKCi0gQEdldCBAR2V0XwotIEBQb3N0IEBQb3N0XwotIEBQdXQgQFB1dF8KLSBARGVsZXRlIEBEZWxldGVfCi0gQEluZm8gQEluZm9fCi0gQFRhZyBAVGFnXwotIEBJdGVtc18KLSBASnNvbkNvbnRlbnRfCi0gQE1lZGlhVHlwZV8KLSBAUGFyYW1ldGVyXwotIEBQcm9wZXJ0eSBAUHJvcGVydHlfCi0gQFJlcXVlc3RCb2R5IEBSZXF1ZXN0Qm9keV8KLSBAUmVzcG9uc2UyMDBfIEBSZXNwb25zZV8KLSBAU2NoZW1hIEBTY2hlbWFfCi0gQFNlY3VyaXR5U2NoZW1lIEBTZWN1cml0eVNjaGVtZV8KCgojIyDlj4LogIPmlofmoaMKCi0gW3N3YWdnZXItcGhwIEFubm90YXRpb25zXShodHRwczovL2dpdGh1Yi5jb20vemlyY290ZS9zd2FnZ2VyLXBocC90cmVlL21hc3Rlci9zcmMvQW5ub3RhdGlvbnMpCi0gW09wZW5BUEkgMy4wIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLyk= readmeEtag: '"6dab04c7726062b623a5482878419c661a3e82ea"' readmeLastModified: Sat, 07 Aug 2021 23:35:46 GMT repositoryId: 389693571 description: Easy to use Swagger-PHP Annotation in PhpStrom created: '2021-07-26T16:09:17Z' updated: '2021-08-07T23:35:56Z' language: null archived: false stars: 2 watchers: 1 forks: 0 owner: chunhei2008 logo: https://avatars.githubusercontent.com/u/4560568?v=4 repoEtag: '"cc5dcc66f021c273da08f43c57907500f0cfc1191a02b1d0a404d78588ea67b0"' repoLastModified: Sat, 07 Aug 2021 23:35:56 GMT foundInMaster: true category: Code Generators id: 30d97397c2ead83a2d582bfbf7b32ae2 - source: openapi3 tags repository: https://github.com/liam-deacon/online-store-rest-api v3: true repositoryMetadata: base64Readme: >- # Online Store Example

<!--lint disable list-item-indent -->
<!--lint disable list-item-spacing -->
<!--lint disable no-inline-padding-->
[![ ](https://img.shields.io/static/v1?label=demo&message=REST%20API&color=darkorchid&logo=Heroku&logoColor=white)](https://online-store-flask-rest-api.herokuapp.com/apidocs/)
[![ ](https://img.shields.io/static/v1?label=docs&message=GitHub%20Pages&color=orange&logo=Read%20The%20Docs&logoColor=white)](https://liam-deacon.github.io/online-store-rest-api/)
[![ ](https://img.shields.io/static/v1?label=docker&message=image&color=0db7ed&logo=Docker&logoColor=white)](https://hub.docker.com/r/lightbytes/online_store_backend)
[![GitHub pull-requests closed](https://img.shields.io/github/issues-pr-closed/liam-deacon/online-store-rest-api.svg?logo=Git&logoColor=white)](https://GitHub.com/liam-deacon/online-store-rest-api/pulls/)
[![Python CI](https://github.com/Liam-Deacon/online-store-rest-api/workflows/Python%20CI/badge.svg)](https://github.com/Liam-Deacon/online-store-rest-api/actions?query=workflow%3A"Python+CI")
[![Frontend Docker CI/CD](https://github.com/Liam-Deacon/online-store-rest-api/workflows/Frontend%20Docker%20CI/CD/badge.svg)](https://github.com/Liam-Deacon/online-store-rest-api/actions?query=workflow%3A%22Frontend+Docker+CI%2FCD%22)
[![Backend Docker CI/CD](https://github.com/Liam-Deacon/online-store-rest-api/workflows/Backend%20Docker%20CI/CD/badge.svg)](https://github.com/Liam-Deacon/online-store-rest-api/actions?query=workflow%3A%22Backend+Docker+CI%2FCD%22)
[![Sphinx Documentation CI](https://github.com/Liam-Deacon/online-store-rest-api/workflows/Sphinx%20Documentation%20CI/badge.svg)](https://github.com/Liam-Deacon/online-store-rest-api/actions?query=workflow%3A%22Sphinx+Documentation+CI%22)
![ ](https://img.shields.io/static/v1?label=python&message=v3.6%20-%203.9&color=blue&logo=python&logoColor=white)
[![ ](https://coveralls.io/repos/github/Liam-Deacon/online-store-rest-api/badge.svg?branch=master)](https://coveralls.io/github/Liam-Deacon/online-store-rest-api?branch=master)
[![ ](https://codecov.io/gh/liam-deacon/online-store-rest-api/branch/master/graph/badge.svg)](https://codecov.io/gh/liam-deacon/online-store-rest-api)
[![ ](https://api.codacy.com/project/badge/Grade/de571d98b5ed4203b6eda5f927c8835d)](https://www.codacy.com/gh/liam-deacon/online-store-rest-api?utm_source=github.com&amp;utm_medium=referral&amp;utm_content=liam-deacon/online-store-rest-api&amp;utm_campaign=Badge_Grade)
[![ ](https://img.shields.io/codefactor/grade/github/liam-deacon/online-store-rest-api?logo=codefactor)](https://www.codefactor.io/repository/github/liam-deacon/online-store-rest-api)
![ ](https://img.shields.io/badge/dev-Open%20in%20Gitpod-blue?logo=gitpod&link=https://gitpod.io/#https://github.com/liam-deacon/online-store-rest-api)
[![ ](https://camo.githubusercontent.com/52feade06f2fecbf006889a904d221e6a730c194/68747470733a2f2f636f6c61622e72657365617263682e676f6f676c652e636f6d2f6173736574732f636f6c61622d62616467652e737667)](https://colab.research.google.com/github/liam-deacon/online-store-rest-api)
[![ ](https://img.shields.io/badge/Binder%20Launch:-Jupyter%20Lab-blue.svg?colorA=&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAABwAAAAcCAYAAAByDd+UAAAABGdBTUEAALGPC/xhBQAAACBjSFJNAAB6JgAAgIQAAPoAAACA6AAAdTAAAOpgAAA6mAAAF3CculE8AAAABmJLR0QA/wD/AP+gvaeTAAAACXBIWXMAAC4jAAAuIwF4pT92AAAAB3RJTUUH4gsEADkvyr8GjAAABQZJREFUSMeVlnlsVFUUh7/7ZukwpQxdoK2yGGgqYFKMQkyDUVBZJECQEERZVLQEa4iKiggiFjfqbkADhVSgEVkETVSiJBATsEIRja1RoCwuU5gC7Qww03Zm3rzrH/dOfJSZUm4y6Xt9957vnnN/55wruI7RVjMNQAA3AiX6bxw4BTQAQQDvnF1pbYjrAAEUAmXADGAQ0AOQwCWgHqgGdgCRdNBrAm2wW4A1wN2ACZwG/gbcQBFwg/Z2I/AS0JoKanQzmoXAamA0cBx4EhgDTAYmAvcArwNhYD6wHHDbNts9D20LlgMrgWPAXKAO/j8rPc8A5uiNAUwH9tjnddfDAn1mFkJWyoRR58hsv8KIfraAz/QvC3golf2UwEBZBYGyCoJfj/LFz/ceDxRJ09Hccbz/6dDu0ozg7lICZRVXrNFQEyWaDmAkkNslMAnSE59x9IrsMVt8awBP4rI3P9acs83hC3+BkFMAd2eoHn8BrdpG77RA2+IiYDPwHnAbEAOkMGQMcAKTdNheBXqmgDoBhw6xda2Q9tGHPhE4hRTlrrxQGRB29IqE3IUtTyDFu9rQC8AiwAiUVdgFNhTIA85oT68G2nb5ODABJf25niL/emfexX1AA0IWeIr8xWbY+yKwBJVzC4FSm71MlFIdwH505UnnYT5KWRawCvgp0eYBCKEqSBwpFuVMqp2a5Q1WO6TcakiZ55DWwyVVKxDC8gLPA1OAJh32q8qcHTgEKEbl2ncAua99lPy2FdgskH2FlFXNI8IVewcO8P+WUyjr8vqPfmvt+plhmVltIJeilLoK+CWVopy250LAgyrELcl/9nB/ixkbF3GKyOJ/rJs8hxNDZx1KDFvsz+9jJvINAQz1EKvxR7OddzrroyXGiRV5zvp1WPlSzN7bJVCmEtKDF38khguQeR5iBRYGFoaZaUUv9YsEc+KGYfq9vssN1qDsP2MDHRZiYBRXpoEMwa1XAe3Gm4A2YDDQ1z7JTbyvG3O1hXEvcNI0xFPzTh5ZueB4HeXH6hoGR1onC2SlhQgD5RnEl7kwXTOqfu4SeBT4Q5/jVIBtL29KfnsUGAecsISY++W+mpohwQujXJYlPAnzh2HBc7Uxw1iGSpU2VAu7C6Az1A68gEr4ZI6NXT78Pkxh9JEwU4JlGsYbO3a+c7g50/esFGIqcBb4fEzgNBlWwgI2AVsAH13V0oL1K5LvNcBOYACwsfb7qiX3n2mcmGXGirPjHf8uPHqw/Xy/IeuAV/TG3gaOAGyfPwJUbm4HosAdpKilzk7vIVT1iAPTTWG8Of5MY/vIFn8Pt2UVZkfbqi0hvFrFlcBaQNo2DKoxt6CqjQ84nzKktkV+YIE+hz1OaUVyou0iKx41BAR02KYB7wMdnWBJm4aOgOz8MWUDTpa6/NazGdUlo8c2ZuVukdBWfOnCtHlffXAwdPsEK2o47Ju0i2MysAt1xxkLtOpwpwzpFd4+sOHXKHDAIa16YNTJrJzS3x9ZVdvoy+WbecNTLfUCs7Xd/aQr3umGy0rgshIhQ8pNhpSmIeVzTZm9pnjNuLDLXT97gKdRKXUWXUvt3qUNqX1oYz2Bj1H3mXPABh22JlRnuBl4DHWPAVgKfAjIzkDntYB6hIHFKPXO0gbLUQp0oO49Xv1eCXySCtYtDzt56kU159moQulDqfEccAD4FDgEJFLBrgtog4I6r36oG0IC1d0DqNZEOhjAfzgw6LulUF3CAAAAJXRFWHRkYXRlOmNyZWF0ZQAyMDE4LTExLTA0VDAwOjU3OjQ3LTA0OjAwLtN9UwAAACV0RVh0ZGF0ZTptb2RpZnkAMjAxOC0xMS0wNFQwMDo1Nzo0Ny0wNDowMF+Oxe8AAAAASUVORK5CYII=)](https://mybinder.org/v2/gh/liam-deacon/online-store-rest-api/master?urlpath=lab)

<!--lint enable no-inline-padding-->

## Components 📲🧩

There is a Vue.js based front-end for displaying reactive web pages to the user
and a Python Flask backend for serving the online store REST API.

### Running The Node.JS Frontend 🏃‍♀️☕📜

The frontend can be started with the following command:

```bash
$ npm run frontend:start  # starts Vue.js app at http://localhost:8080
```

Note that the frontend can be found under `online_store/frontend` (WIP).

### Running The Python Backend 🏃‍♂️🐍🔚

The backend can be started with the following command:

```bash
$ export FLASK_DEBUG=1  # for development with live reload
$ export FLASK_ENVIRONMENT=development  # this is the default
$ PYTHONPATH='.' python3 manage.py run
```

## Background 📖

This repository was created to solve the 
[Prezola Technical Challenge](https://github.com/prezola/technical-challenge),
which requires a solution capable of adding, removing and listing (added) gifts
from a list. It also required a mechanism to purchase a gift from the list and
generate a report (of purchased vs non-purchased) gifts.  

This code project takes that concept and extends it to a generic (gift) store.

## The challenge

_"Write a program to the best of your knowledge which will allow the user to_
_manage a single list of wedding gifts."_

The user must be able to:

- Add a gift to the list
- Remove a gift from the list
- List the already added gifts of the list 
- Purchase a gift from the list
- Generate a report from the list which will print out gifts & their statuses.
  - The report must include two sections:
    - Purchased gifts: each purchased gift with their details.
    - Not purchased gifts: each available gift with their details.

### Implementation Notes 📄

There are two concrete implementations for realising a gift list with
the following classes from `online_store/backend/gift_list.py`:

- `BasicGiftList`, a pure python implementation of a gift list **(Well Tested)**.
- `SqlDatabaseGiftList`, an SQL ORM based implementation of a gift list for
   use within a flask (or Django) REST API app. In this example, the ORM
   models are found in `online_store/backend/models/` and the REST API is
   implemented in `online_store/backend/routes/gifts.py`.

### Development Setup ⚙️

```bash
$ npm install  # needed for Swagger JSON to OAS YAML spec conversion
$ npm run frontend:install  # install packages needed for frontend app
$ python3 -m venv venv
$ source venv/bin/activate
(venv) $ pip install setuptools
(venv) $ pip install -r requirements-dev.txt -r requirements-test.txt
(venv) $ pip install -r requirements.txt
```

### Testing 🧪

`setup.cfg` has configured `pytest` to collect coverage information
and can be run as follows:

```bash
(venv) $ PYTHONPATH='.' py.test 
```

### Basic Gift List Implementation 🎁

The following showcases a simple python implementation of the gift list:

```python
# load store data
>>> import json
>>> store_items = json.load(open('products.json'))
# import implementation
>>> from online_store.backend.gift_list import BasicGiftList
# create a new gift list
>>> gift_list = BasicGiftList('Liam')
>>> gift_list  # show list representation in interpreter
Liam -> []
# add gift item
>>> gift_list.add_item(store_items[0])
>>> gift_list
Liam -> [{'id': 1, 'name': 'Tea pot', 'brand': 'Le Creuset', 'price': '47.00GBP', 'in_stock_quantity': 50}]
# remove gift item
>>> gift_list.add_item(store_items[1], quantity=3)
>>> gift_list.remove_item(store_items[0])
>>> gift_list
Liam -> [{'id': 2, 'name': 'Cast Iron Oval Casserole - 25cm; Volcanic', 'brand': 'Le Creuset', 'price': '210.00GBP', 'in_stock_quantity': 27}]
# purchase gift item
>>> gift_list.purchase_item(store_items[1], quantity=1)
# generate report
>>> gift_list.create_report()
Gift List Report for Liam:
==============================
Purchased items:
   - {'id': 2, 'name': 'Cast Iron Oval Casserole - 25cm; Volcanic', 'brand': 'Le Creuset', 'price': '210.00GBP', 'in_stock_quantity': 27} (quantity: 1)
------------------------------
Available items:
  - {'id': 2, 'name': 'Cast Iron Oval Casserole - 25cm; Volcanic', 'brand': 'Le Creuset', 'price': '210.00GBP', 'in_stock_quantity': 27} (quantity: 2)
```

### Flask REST API + SQL ORM Implementation 🌍🕸️

Alternatively there is a REST API, which can be run with:

```bash
$ source venv/bin/activate
(venv) $ cd online_store
(venv) $ FLASK_DEBUG=1 flask run
```

This will start a flask development server running on http://localhost:5000 and
provides a Swagger-UI at http://localhost:5000/apidocs/. 

#### Examples

```bash
# register a new user
$ curl -X POST -H "Content-Type: application/json" -d '{"username": "me", "password": "test", "email": "me@test.com"}' 'http://localhost:5000/api/v1/auth/register'

{"code":200,"msg":"success","status":"ok"}

# login with user
$ curl -X POST -H "Content-Type: application/json" -d '{"username": "me", "password": "test"}'

{"access_token":"eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJpYXQiOjE2MDIwMTIxMzQsIm5iZiI6MTYwMjAxMjEzNCwianRpIjoiOTA5NTRlZDAtZWIzMy00MTY2LThhODEtZDE5NDI3MjI5NTE0IiwiZXhwIjoxNjAyMDEzMDM0LCJpZGVudGl0eSI6Im1lIiwiZnJlc2giOmZhbHNlLCJ0eXBlIjoiYWNjZXNzIn0.GYVTQK4Xw9JaiJJxa75vlKBS-mho0QjfcM94usPZtSI","code":200,"status":"ok"}

# note access token
$ export TOKEN='<JWT_FROM_LOGIN>'

# access gift list
$ curl -X GET -H  "accept: application/json" -H  "Authorization: Bearer $TOKEN" http://localhost:5000/api/v1/gifts/list

[]

# add one gift item with store id of 1 to list
$ curl -X POST -H  "accept: application/json" -H  "Authorization: Bearer $TOKEN" 'http://localhost:5000/api/v1/gifts/list/add?item_id=1&quantity=1'

{
  "code": 200,
  "msg": "Item added",
  "status": "ok"
}

# remove item with id 1 from gift list
$ curl -X DELETE -H  "accept: application/json" -H  "Authorization: Bearer $TOKEN" http://localhost:5000/api/v1/gifts/list/1

{
  "code": 200,
  "msg": "Item removed",
  "status": "ok"
}

# add then purchase item
$ curl -X POST -H  "accept: application/json" -H  "Authorization: Bearer $TOKEN" 'http://localhost:5000/api/v1/gifts/list/add?item_id=2&quantity=2'

{
  "code": 200,
  "msg": "Item added",
  "status": "ok"
}

$ curl -X POST -H  "accept: application/json" -H  "Authorization: Bearer $TOKEN" http://localhost:5000/api/v1/gifts/list/2/purchase?quantity=1

{
  "code": 200, 
  "msg": "gift purchased", 
  "status": "ok"
}

# produce report
$ curl -X GET -H  "accept: application/json" -H  "Authorization: Bearer $TOKEN" 'http://localhost:5000/api/v1/gifts/list/report'

{
  "available": [
    {
      "brand": "Le Creuset", 
      "currency": "GBP", 
      "id": 2, 
      "in_stock_quantity": 27, 
      "name": "Cast Iron Oval Casserole - 25cm; Volcanic", 
      "price": 210.0, 
      "quantity": 1
    }
  ], 
  "purchased": [
    {
      "brand": "Le Creuset", 
      "currency": "GBP", 
      "id": 2, 
      "in_stock_quantity": 27, 
      "name": "Cast Iron Oval Casserole - 25cm; Volcanic", 
      "price": 210.0, 
      "quantity": 1
    }
  ], 
  "user": 2
}

```

## Bonus Features ✨

There are currently a number of extra features, which help 

- User-friendly backend application logging using `loguru` python package.
- Simple containerisation using Docker - see `DockerFile`
- Authentication using JSON web tokens via [flask-jwt-extended]() middleware.
- [OpenAPI Specification (OAS)](https://swagger.io/specification/) conformant client documentation generated using [flasgger](https://github.com/flasgger/flasgger) and viewable via SwaggerUI [/apidocs](localhost:5000/apidocs) endpoint when running the flask server.
- Persistent data storage using SQL Database modelled using [sqlalchemy](https://docs.sqlalchemy.org/en/13/intro.html) ORM.

### Developer Documentation 📗

The Sphinx documentation builder is currently used to extract python docstrings and the OpenAPI spec of the REST API.

To build the documentation:

```bash
$ cd docs/
$ make openapi_spec.yml
$ make html  # or another end documentation format e.g. epub
```

A live deployment to GitHub Pages can be found at [https://liam-deacon.github.io/online-store-rest-api/]()

## TODO 📝

- [x] Build script / CI using GitHub Actions
- [x] Sphinx documentation support
- [x] Deploy API documentation to GitHub Pages via Actions
- [ ] Test SqlDatabaseGiftList
- [ ] Automated tests for REST API
- [x] shields.io support for README badges
- [ ] Add linting to CI
- [x] Build and deploy docker image(s) to dockerhub via CI/CD
- [x] Deploy flask app to Heroku for
  [demo](https://online-store-flask-rest-api.herokuapp.com/) purposes (follow link)

## Future Improvements 🔮

Given more time, the following improvements could be made:

-   [ ] Write frontend in React (or maybe Vue.js)
-   [ ] Create `docker-compose.yml` multi-container Docker compose script for orchestrating frontend, backend and database (e.g. React/NPM-based, Flask, Postgres).
-   [ ] Implement missing features in the code (i.e. wherever `NotImplementError` is raised)
-   [ ] Code tidy and refactor
-   [ ] Increase overall code coverage (aiming for nirvana at 100%)
 readmeEtag: '"8a192ca048b122c350d207e215c7077bb8f69060"' readmeLastModified: Tue, 27 Oct 2020 08:20:51 GMT repositoryId: 301341448 description: >- An example online store REST API using flask, SQL ORM and OpenAPI Specifications (with WIP Vue.js frontend) created: '2020-10-05T08:29:22Z' updated: '2022-10-28T17:33:53Z' language: Python archived: false stars: 2 watchers: 1 forks: 0 owner: Liam-Deacon logo: https://avatars.githubusercontent.com/u/3184694?v=4 license: GPL-3.0 repoEtag: '"33570fee56463763ad9af06c7a7198b14a00f36b35f5867474027e722551ec11"' repoLastModified: Fri, 28 Oct 2022 17:33:53 GMT foundInMaster: true category: Server Implementations id: dcd91a71c900c43016a01d86e8f15130 - source: openapi3 tags repository: https://github.com/deepakbhalla/spring-boot-basic-auth-security v3: true id: eea19c2e6d65f964d1b757cc3434fdb3 repositoryMetadata: base64Readme: >- # spring-boot-basic-auth-security
Application to demonstrate Spring boot security using Basic Auth (Username and Password) as authentication type. 

This application covers the below functionalities:

1. <b>User Registration (API authentication is not required)</b>
   - New use registration without API authentication
2. <b>User Management (API authentication is required)</b>
   - Get all users
   - Delete a user
3. <b>Account Management (API authentication is required)</b>
    - Get all accounts
    - Get an account
    - Create new account
    - Update existing account
    - Delete existing account
4. <b>Transaction Management (API authentication is required)</b>
    - Deposit an amount to an account
    - Withdraw an amount from an account

### Project Technologies

- Java version: 17.0.8, vendor: Oracle Corporation
- Apache Maven 3.2.3
- Spring boot 3.1.5
- PostgreSql Database
- Spring boot JPA
- Spring security 6.1.5 (Implemented Basic Auth in this project)
- Lombok
- OpenApi 3 Specifications

### Database

PostgreSQL has been used in this project. Also, this project stores account's events in the form of JSON in JSONB column
in the application table.

JSON data types are for storing JSON (JavaScript Object Notation) data. Such data can also be stored as text, but the 
JSON data types have the advantage of enforcing that each stored value is valid according to the JSON rules. There are 
also assorted JSON-specific functions and operators available for data stored in these data types.

PostgreSQL offers two types for storing JSON data: 

- JSON 
- JSONB. 

1. JSON stores white space, and that is why we can see spaces when key "a" is stored, while JSONB does not.
2. JSON stores all the values of a key. This is the reason you can see multiple values (2 and 1) against the key "a", while JSONB only "stores" the last value.
3. JSON maintains the order in which elements are inserted, while JSONB maintains the "sorted" order.
4. JSONB objects are stored as a decompressed binary as opposed to "raw data" in JSON, where no reparsing of data is required during retrieval.
5. JSONB also supports indexing, which can be a significant advantage.

### OpenAPI Specification

- Path

http://localhost:8081/my-application/swagger-ui/index.html

- Swagger Authorization

![24_swagger_authorization.PNG](screenshots%2F24_swagger_authorization.PNG)

- Swagger UI Details

![25_swagger_ui.png](screenshots%2F25_swagger_ui.png)

### Disable OpenAPI Swagger for Production Environment

We can disable OpenAPI swagger for any environment based upon profiles. We can supply a VM argument 
'-Dspring.profiles.active=<environment name>' to the application configurations.
Using spring profile annotation @Profile("prod"), we can control the display of swagger.

- VM Argument

![img.png](screenshots/22_vm_argument_spring_active_profile.png)

If the value of spring profile is 'prod', then swagger won't be available. Please refer the below screenshot:

![img.png](screenshots/23_swagger_not_available.png)

### API Testing

- Postman Collection

![1_database_tables_script.PNG](screenshots%2F20_postman_collection.PNG)

- Database tables script

![1_database_tables_script.PNG](screenshots%2F1_database_tables_script.PNG)

- Users created on load

![2_Users_created_on_service_startup.PNG](screenshots%2F2_Users_created_on_service_startup.PNG)

- Create New User - Validation Error

![3_create_user_input_validation_error.PNG](screenshots%2F3_create_user_input_validation_error.PNG)

- Create User - Password and MatchPassword Does not Match Validation Error

![4_create_user_password_different_from_matching_password.PNG](screenshots%2F4_create_user_password_different_from_matching_password.PNG)

- Create User - Success - No Auth

![5_create_user_success.PNG](screenshots%2F5_create_user_success.PNG)

- Create User - No Auth Shown Under Authorization tab of Postman

![6_create_user_no_auth_required.PNG](screenshots%2F6_create_user_no_auth_required.PNG)

- Create User - New user created in database table 'user_table'

![7_created_user_in_user_table_in_database.PNG](screenshots%2F7_created_user_in_user_table_in_database.PNG)

- Get User - Not Authorized without authentication

![8_get_user_authentication_error.PNG](screenshots%2F8_get_user_authentication_error.PNG)

- Get Users - Successful with Basic Auth

![9_get_users_success_with_basic_auth.PNG](screenshots%2F9_get_users_success_with_basic_auth.PNG)

- Create Account - Not Authorized without authentication

![10_create_account_authentication_error.PNG](screenshots%2F10_create_account_authentication_error.PNG)

- Create Account - Successful with Basic Auth

![11_create_accounts_success_with_basic_auth.PNG](screenshots%2F11_create_accounts_success_with_basic_auth.PNG)

- Get All Accounts - Not Authorized without authentication

![12_get_all_accounts_authentication_error.PNG](screenshots%2F12_get_all_accounts_authentication_error.PNG)

- Get All Accounts - Successful with Basic Auth

![13_get_all_accounts_success_with_basic_auth.PNG](screenshots%2F13_get_all_accounts_success_with_basic_auth.PNG)

- Get Account Details - Not Authorized without authentication

![14_get_account_info_authentication_error.PNG](screenshots%2F14_get_account_info_authentication_error.PNG)

- Get All Accounts - Successful with Basic Auth

![15_get_account_info_success_with_basic_auth.PNG](screenshots%2F15_get_account_info_success_with_basic_auth.PNG)

- Deposit Amount to an Account - Not Authorized without authentication

![16_deposit_amount_authentication_error.PNG](screenshots%2F16_deposit_amount_authentication_error.PNG)

- Deposit Amount to an Account - Successful with Basic Auth

![17_deposit_amount_success_with_basic_auth.PNG](screenshots%2F17_deposit_amount_success_with_basic_auth.PNG)

- Withdraw Amount from an Account - Not Authorized without authentication

![18_withdraw_amount_authentication_error.PNG](screenshots%2F18_withdraw_amount_authentication_error.PNG)

- Withdraw Amount from an Account - Successful with Basic Auth

![19_withdraw_amount_success_with_basic_auth.PNG](screenshots%2F19_withdraw_amount_success_with_basic_auth.PNG)

 readmeEtag: '"e955173e21e8956650cf5c6f6d3010172a602c9d"' readmeLastModified: Tue, 31 Oct 2023 20:47:09 GMT repositoryId: 710441963 description: >- Spring boot Rest APIs secured with Basic Auth Spring Security Implementation created: '2023-10-26T17:42:58Z' updated: '2023-12-19T11:40:05Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: deepakbhalla logo: https://avatars.githubusercontent.com/u/6603640?v=4 repoEtag: '"974657b6a1397fd6014a8b4d4d6226ebca975f51113ecce75e25e2e66690a45a"' repoLastModified: Tue, 19 Dec 2023 11:40:05 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/linx-software/petstore-api v3: true repositoryMetadata: base64Readme: >- IyAgU3dhZ2dlciBQZXRzdG9yZSAtIE9wZW5BUEkgMy4wIHNhbXBsZQoKIyMgRGVzY3JpcHRpb24KVGhpcyBzYW1wbGUgaW1wbGVtZW50cyBhIExpbnggUkVTVCBBUEkgYmFzZWQgb24gdGhlIFN3YWdnZXIgJ1BldHN0b3JlJyBPcGVuQVBJIGRlZmluaXRpb24gYW5kIGlzIGhvc3RlZCBvbiBhIExpbnggQ2xvdWQgU2VydmVyLCBtb3JlIGRldGFpbHMgb24gdGhlIG9yaWdpbmFsICdQZXRzdG9yZScgR2l0SHViIHByb2plY3QgY2FuIGJlIGZvdW5kIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1wZXRzdG9yZSkuIFRoZSBvcGVyYXRpb25zIG9mIHRoZSBMaW54IEFQSSBkbyBub3QgcGVyc2lzdCBhbnkgZGF0YSwgYnV0IHJhdGhlciwgcmVxdWVzdCBkYXRhIGlzIGZvcndhcmRlZCB0byBhY3R1YWwgJ1BldHN0b3JlJyBzZXJ2ZXIgdmlhIEhUVFAgcmVxdWVzdHMuICAKClZpZXcgdGhlIFtsaXZlIGRlbW9dKGh0dHBzOi8vZGVtb2xpbng2LmFwaS5saW54LnR3ZW50eTU3Lm5ldC9wZXRzdG9yZS9zd2FnZ2VyKSBob3N0ZWQgb24gYSBMaW54IENsb3VkIFNlcnZlci4KCgojIyBJbnN0YWxsYXRpb24KCiMjIyBDbG91ZCBzZXJ2ZXIgZGVwbG95bWVudApUaGlzIHNvbHV0aW9uIGNhbiBiZSBkZXBsb3llZCBkaXJlY3RseSB0byB5b3VyIExpbnggQ2xvdWQgc2VydmVyIGluc3RhbmNlLgoKMS4gUmVnaXN0ZXIgZm9yIGEgTGlueCB0cmlhbCBjbG91ZCBzZXJ2ZXIgW2hlcmVdKGh0dHBzOi8vbGlueC5zb2Z0d2FyZS9zZXJ2ZXItYnV5Mi8pLgoyLiBZb3Ugd2lsbCByZWNlaXZlIGFuIGVtYWlsIGNvbnRhaW5pbmcgeW91ciBMaW54IGNsb3VkIHNlcnZlciBjcmVkZW50aWFscyB3aGVuIHlvdXIgdHJpYWwgc2VydmVyIGhhcyBiZWVuIGFjdGl2YXRlZC4KMS4gTG9nIGludG8geW91ciBjbG91ZCBzZXJ2ZXIgaW5zdGFuY2UgYW5kIHVwbG9hZCB0aGUgU29sdXRpb24gKFRvcCBNZW51ID4gU2VydmVyID4gVXBsb2FkKS4KMy4gT24gdGhlIFNvbHV0aW9uJ3Mgc2VydmljZSBkYXNoYm9hcmQgcGFnZSwgX19zdGFydF9fIGFsbCBvZiB0aGUgc2VydmljZXMgZm9yIHRoZSBTb2x1dGlvbi4gICAKNC4gT25jZSB0aGUgc2VydmljZSBoYXMgc3RhcnRlZCwgeW91IGFyZSBhYmxlIHRvIG1ha2UgcmVxdWVzdHMgdXNpbmcgdGhlIGJhc2UgVVJMIG9mOgogICBgYGAKICAgaHR0cHM6Ly97eW91ciBpbnN0YW5jZSBuYW1lfS5hcGkubGlueC50d2VudHk1Ny5uZXQvcGV0c3RvcmUKICAgYGBgCgoKIyMjIExvY2FsIGVudmlyb25tZW50ClRoZSBiZWxvdyBzdGVwcyBkZXNjcmliZSBob3cgdG8gc2V0dXAgdGhlIHNhbXBsZSB0byBydW4gb24geW91ciBsb2NhbCBMaW54IERlc2lnbmVyIGVudmlyb25tZW50LgoKMS4gRG93bmxvYWQgYW5kIGluc3RhbGwgdGhlIExpbnggRGVzaWduZXIgW2hlcmVdKGh0dHBzOi8vbGlueC5zb2Z0d2FyZS9zZXJ2ZXItYnV5Mi8pLgoxLiBPcGVuIHRoZSBzYW1wbGUgU29sdXRpb24gKC5sc296KSBpbiB5b3VyIExpbnggRGVzaWduZXIuCjIuIEFsdGVyIHRoZSBiZWxvdyBTb2x1dGlvbiBTZXR0aW5nczoKICAgIC0gYExpbnhJc0xvY2FsRGV2RW52YCA6IGBUcnVlYAozLiBTZWxlY3QgdGhlIFJFU1RIb3N0IHNlcnZpY2UsIHJpZ2h0IGNsaWNrIGFuZCBzZWxlY3QgX19kZWJ1Z19fLiBPbmNlIGluaXRpYWxpc2VkLCAqKnN0YXJ0KiogdGhlIGRlYnVnZ2VyLgo0LiBPbmNlIHRoZSBkZWJ1Z2dlciBoYXMgc3RhcnRlZCwgeW91IGFyZSBhYmxlIHRvIG1ha2UgcmVxdWVzdHMgbG9jYWxseSB0bzoKICAgYGBgCiAgIGh0dHBzOi8vbG9jYWxob3N0OjgwODAvcGV0c3RvcmUKICAgYGBgCgojIyBVc2FnZQoKIyMjIFN3YWdnZXIgVUkKW1ZpZXcgbGl2ZSBkZW1vXSgJaHR0cHM6Ly9kZW1vbGlueDYuYXBpLmxpbngudHdlbnR5NTcubmV0L3BldHN0b3JlL3N3YWdnZXIpLgoKCiMjIyBQb3N0bWFuCjEuIE9wZW4gUG9zdG1hbiBhbmQgaW1wb3J0IHRoZSBwcm92aWRlZCBbcmVxdWVzdCBjb2xsZWN0aW9uXShodHRwczovL2dpdGh1Yi5jb20vbGlueC1zb2Z0d2FyZS9wZXRzdG9yZS1hcGkvYmxvYi9tYWluL3Rlc3RzL3Bvc3RtYW4tY29sbGVjdGlvbi9Td2FnZ2VyJTIwUGV0c3RvcmUlMjB3aXRoJTIwTGlueC5wb3N0bWFuX2NvbGxlY3Rpb24uanNvbikgaW4gUG9zdG1hbi4KMi4gQWx0ZXIgdGhlIGNvbGxlY3Rpb24gdmFyaWFibGVzIHRvIHJlZmxlY3QgeW91ciBjbG91ZCBlbnZpcm9ubWVudC4KIAozLiBPcGVuIGVhY2ggcmVxdWVzdCBhbmQgcGFzcyBvciBjaGFuZ2UgcGFyYW1ldGVycyBhbmQgYm9keSB2YWx1ZXMgYXMgZGVzY3JpYmVkIGluIGBodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2AKCgoKIyMgQ29udHJpYnV0aW5nCgpGb3IgcXVlc3Rpb25zIHBsZWFzZSBhc2sgdGhlIFtMaW54IGNvbW11bml0eV0oaHR0cHM6Ly9saW54L3NvZnR3YXJlL2NvbW11bml0eSkgb3IgdXNlIHRoZSBbU2xhY2sgY2hhbm5lbF0oaHR0cHM6Ly9saW54c29mdHdhcmUuc2xhY2suY29tL2FyY2hpdmVzL0MwMUZMQkMxWE5YKS4gCgojIyBMaWNlbnNlCgpbTUlUXShodHRwczovL2dpdGh1Yi5jb20vbGlueC1zb2Z0d2FyZS90ZW1wbGF0ZS1yZXBvL2Jsb2IvbWFpbi9MSUNFTlNFLnR4dCkK readmeEtag: '"08f7dade9df5506a5f93ac80d173c2fcb7b3a441"' readmeLastModified: Fri, 11 Nov 2022 10:37:42 GMT repositoryId: 389524066 description: >- This sample implements a Linx REST API based on the Swagger 'Petstore' OpenAPI definition and is hosted on a Linx Cloud Server created: '2021-07-26T06:00:00Z' updated: '2024-01-30T06:07:22Z' language: null archived: false stars: 2 watchers: 2 forks: 0 owner: linx-software logo: https://avatars.githubusercontent.com/u/62884647?v=4 repoEtag: '"0e1ed266a21360e85e2029126fbdd62a5b857dfd81375a415a610c708da9a56c"' repoLastModified: Tue, 30 Jan 2024 06:07:22 GMT foundInMaster: true category: - Converters - Server Implementations id: 4a157f3784bb3a4a47976a95a50701d0 - source: openapi3 tags repository: https://github.com/fourcels/rest v3: true id: c16c94cc33d32d5ea21bdb7d0e975ab1 repositoryMetadata: base64Readme: >- IyBSRVNUIHdpdGggQ2xlYW4gQXJjaGl0ZWN0dXJlIGZvciBHbwoKWyFbR29EZXZEb2NdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZGV2LWRvYy0wMEFERDg/bG9nbz1nbyldKGh0dHBzOi8vcGtnLmdvLmRldi9naXRodWIuY29tL2ZvdXJjZWxzL3Jlc3QpCgpJbnNwaXJlZCBieSBbc3dhZ2dlc3QvcmVzdF0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXN0L3Jlc3QpCgojIyBGZWF0dXJlcwoKLSBCdWlsdCB3aXRoIFtlY2hvXShodHRwczovL2dpdGh1Yi5jb20vbGFic3RhY2svZWNobykuCi0gQXV0b21hdGljIE9wZW5BUEkgMyBkb2N1bWVudGF0aW9uIHdpdGgKICBbb3BlbmFwaS1nb10oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXN0L29wZW5hcGktZ28pLgotIEF1dG9tYXRpYyByZXF1ZXN0IEpTT04gc2NoZW1hIHZhbGlkYXRpb24gd2l0aAogIFtqc29uc2NoZW1hLWdvXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlc3QvanNvbnNjaGVtYS1nbyksCiAgW2pzb25zY2hlbWFdKGh0dHBzOi8vZ2l0aHViLmNvbS9zYW50aG9zaC10ZWt1cmkvanNvbnNjaGVtYSkuCi0gRW1iZWRkZWQgW1N3YWdnZXIgVUldKGh0dHBzOi8vc3dhZ2dlci5pby90b29scy9zd2FnZ2VyLXVpLykuCgojIyBVc2FnZQoKIyMjIFJlcXVlc3QKCkdvIHN0cnVjdCB3aXRoIGZpZWxkIHRhZ3MgZGVmaW5lcyBpbnB1dC4KCmBgYGdvCi8vIERlY2xhcmUgaW5wdXQgcG9ydCB0eXBlLgp0eXBlIGhlbGxvSW5wdXQgc3RydWN0IHsKICAgIExvY2FsZSBzdHJpbmcgYHF1ZXJ5OiJsb2NhbGUiIGRlZmF1bHQ6ImVuLVVTIiBwYXR0ZXJuOiJeW2Etel17Mn0tW0EtWl17Mn0kIiBlbnVtOiJ6aC1DTixlbi1VUyJgCiAgICBOYW1lICAgc3RyaW5nIGBwYXRoOiJuYW1lIiBtaW5MZW5ndGg6IjMiYCAvLyBGaWVsZCB0YWdzIGRlZmluZSBwYXJhbWV0ZXIgbG9jYXRpb24gYW5kIEpTT04gc2NoZW1hIGNvbnN0cmFpbnRzLgoKICAgIF8gICAgICBzdHJ1Y3R7fSBgdGl0bGU6Ik15IFN0cnVjdCIgZGVzY3JpcHRpb246IkhvbGRzIG15IGRhdGEuImAKfQpgYGAKCklucHV0IGRhdGEgY2FuIGJlIGxvY2F0ZWQgaW46CgotIGBwYXRoYCBwYXJhbWV0ZXIgaW4gcmVxdWVzdCBVUkksIGUuZy4gYC91c2Vycy86bmFtZWAsCi0gYHF1ZXJ5YCBwYXJhbWV0ZXIgaW4gcmVxdWVzdCBVUkksIGUuZy4gYC91c2Vycz9sb2NhbGU9ZW4tVVNgLAotIGBmb3JtYCBwYXJhbWV0ZXIgaW4gcmVxdWVzdCBib2R5IHdpdGggYGFwcGxpY2F0aW9uL3gtd3d3LWZvcm0tdXJsZW5jb2RlZGAKICBjb250ZW50LAotIGBmb3JtRGF0YWAgcGFyYW1ldGVyIGluIHJlcXVlc3QgYm9keSB3aXRoIGBtdWx0aXBhcnQvZm9ybS1kYXRhYCBjb250ZW50LAotIGBqc29uYCBwYXJhbWV0ZXIgaW4gcmVxdWVzdCBib2R5IHdpdGggYGFwcGxpY2F0aW9uL2pzb25gIGNvbnRlbnQsCi0gYGNvb2tpZWAgcGFyYW1ldGVyIGluIHJlcXVlc3QgY29va2llLAotIGBoZWFkZXJgIHBhcmFtZXRlciBpbiByZXF1ZXN0IGhlYWRlci4KCkZpZWxkIHRhZ3MKCi0gbnVtYmVyIGBtYXhpbXVtYCwgYGV4Y2x1c2l2ZU1heGltdW1gLCBgbWluaW11bWAsIGBleGNsdXNpdmVNaW5pbXVtYCwKICBgbXVsdGlwbGVPZmAKLSBzdHJpbmcgYG1pbkxlbmd0aGAsIGBtYXhMZW5ndGhgLCBgcGF0dGVybmAsIGBmb3JtYXRgCi0gYXJyYXkgYG1pbkl0ZW1zYCwgYG1heEl0ZW1zYCwgYHVuaXF1ZUl0ZW1zYAotIGFsbCBgdGl0bGVgLCBgZGVzY3JpcHRpb25gLCBgZGVmYXVsdGAsIGBjb25zdGAsIGBlbnVtYAoKQWRkaXRpb25hbCBmaWVsZCB0YWdzIGRlc2NyaWJlIEpTT04gc2NoZW1hIGNvbnN0cmFpbnRzLCBwbGVhc2UgY2hlY2sKW2RvY3VtZW50YXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VzdC9qc29uc2NoZW1hLWdvI2ZpZWxkLXRhZ3MpLgoKIyMgUmVzcG9uc2UKCmBgYGdvCi8vIERlY2xhcmUgb3V0cHV0IHBvcnQgdHlwZS4KdHlwZSBoZWxsb091dHB1dCBzdHJ1Y3QgewogICAgTm93ICAgICB0aW1lLlRpbWUgYGhlYWRlcjoiWC1Ob3ciIGpzb246Ii0iYAogICAgTWVzc2FnZSBzdHJpbmcgICAgYGpzb246Im1lc3NhZ2UiYAogICAgU2VzcyAgICBzdHJpbmcgICAgYGNvb2tpZToic2VzcyxodHRwb25seSxzZWN1cmUsbWF4LWFnZT04NjQwMCxzYW1lc2l0ZT1sYXgiYAp9CmBgYAoKT3V0cHV0IGRhdGEgY2FuIGJlIGxvY2F0ZWQgaW46CgotIGBqc29uYCBmb3IgcmVzcG9uc2UgYm9keSB3aXRoIGBhcHBsaWNhdGlvbi9qc29uYCBjb250ZW50LAotIGBoZWFkZXJgIGZvciB2YWx1ZXMgaW4gcmVzcG9uc2UgaGVhZGVyLAotIGBjb29raWVgIGZvciBjb29raWUgdmFsdWVzLCBjb29raWUgZmllbGRzIGNhbiBoYXZlIGNvbmZpZ3VyYXRpb24gaW4gZmllbGQgdGFnCiAgKHNhbWUgYXMgaW4gYWN0dWFsIGNvb2tpZSwgYnV0IHdpdGggY29tbWEgc2VwYXJhdGlvbikuCgojIyBFeGFtcGxlCgpbQWR2YW5jZSBFeGFtcGxlXSgvZXhhbXBsZXMvYWR2YW5jZS9tYWluLmdvKQoKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgKAoJImZtdCIKCSJsb2ciCgkidGltZSIKCgkiZ2l0aHViLmNvbS9mb3VyY2Vscy9yZXN0IgoJImdpdGh1Yi5jb20vbGFic3RhY2svZWNoby92NCIKKQoKZnVuYyBtYWluKCkgewoJcyA6PSByZXN0Lk5ld1NlcnZpY2UoKQoJcy5PcGVuQVBJLkluZm8uV2l0aFRpdGxlKCJCYXNpYyBFeGFtcGxlIikKCXMuR0VUKCIvaGVsbG8ve25hbWV9IiwgaGVsbG8oKSkKCgkvLyBTd2FnZ2VyIFVJIGVuZHBvaW50IGF0IC9kb2NzLgoJcy5Eb2NzKCIvZG9jcyIpCgoJLy8gU3RhcnQgc2VydmVyLgoJbG9nLlByaW50bG4oImh0dHA6Ly9sb2NhbGhvc3Q6MTMyMy9kb2NzIikKCXMuU3RhcnQoIjoxMzIzIikKfQoKZnVuYyBoZWxsbygpIHJlc3QuSW50ZXJhY3RvciB7CgkvLyBEZWNsYXJlIGlucHV0IHBvcnQgdHlwZS4KCXR5cGUgaW5wdXQgc3RydWN0IHsKCQlOYW1lICAgc3RyaW5nICAgYHBhdGg6Im5hbWUiIG1pbkxlbmd0aDoiMyJgIC8vIEZpZWxkIHRhZ3MgZGVmaW5lIHBhcmFtZXRlcgoJCUxvY2FsZSBzdHJpbmcgICBgcXVlcnk6ImxvY2FsZSIgZGVmYXVsdDoiZW4tVVMiIHBhdHRlcm46Il5bYS16XXsyfS1bQS1aXXsyfSQiIGVudW06InpoLUNOLGVuLVVTImAKCQlfICAgICAgc3RydWN0e30gYHRpdGxlOiJNeSBTdHJ1Y3QiIGRlc2NyaXB0aW9uOiJIb2xkcyBteSBkYXRhLiJgCgl9CgoJLy8gRGVjbGFyZSBvdXRwdXQgcG9ydCB0eXBlLgoJdHlwZSBvdXRwdXQgc3RydWN0IHsKCQlOb3cgICAgIHRpbWUuVGltZSBgaGVhZGVyOiJYLU5vdyIganNvbjoiLSJgCgkJTWVzc2FnZSBzdHJpbmcgICAgYGpzb246Im1lc3NhZ2UiYAoJfQoKCW1lc3NhZ2VzIDo9IG1hcFtzdHJpbmddc3RyaW5newoJCSJlbi1VUyI6ICJIZWxsbywgJXMhIiwKCQkiemgtQ04iOiAi5L2g5aW9LCAlcyEiLAoJfQoJcmV0dXJuIHJlc3QuTmV3SGFuZGxlcihmdW5jKGMgZWNoby5Db250ZXh0LCBpbiBpbnB1dCwgb3V0ICpvdXRwdXQpIGVycm9yIHsKCQltc2cgOj0gbWVzc2FnZXNbaW4uTG9jYWxlXQoJCW91dC5Ob3cgPSB0aW1lLk5vdygpCgkJb3V0Lk1lc3NhZ2UgPSBmbXQuU3ByaW50Zihtc2csIGluLk5hbWUpCgkJcmV0dXJuIG5pbAoJfSkKfQpgYGAK readmeEtag: '"2f6a120cc802345c0e938abb2306ee832edb8e49"' readmeLastModified: Fri, 12 Jul 2024 06:33:17 GMT repositoryId: 631874551 description: RESTful Web Services base on Echo created: '2023-04-24T08:42:46Z' updated: '2024-07-12T06:33:28Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: fourcels logo: https://avatars.githubusercontent.com/u/1042568?v=4 license: MIT repoEtag: '"a61b458ea2b13b84f1132e10fdabcaf3eb3669db2bc2facefde1b08c5a5cc0b9"' repoLastModified: Fri, 12 Jul 2024 06:33:28 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/osvalda/pitaya v3: true repositoryMetadata: base64Readme: >- IyBQaXRheWEKClshW0NpcmNsZUNJXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NpcmNsZWNpL2J1aWxkL2dpdGh1Yi9vc3ZhbGRhL1BpdGF5YS9tYXN0ZXIpXShodHRwczovL2FwcC5jaXJjbGVjaS5jb20vcGlwZWxpbmVzL2dpdGh1Yi9vc3ZhbGRhL1BpdGF5YT9icmFuY2g9bWFzdGVyKQpbIVtDb3ZlcmFnZSBTdGF0dXNdKGh0dHBzOi8vY292ZXJhbGxzLmlvL3JlcG9zL2dpdGh1Yi9vc3ZhbGRhL1BpdGF5YS9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vY292ZXJhbGxzLmlvL2dpdGh1Yi9vc3ZhbGRhL1BpdGF5YT9icmFuY2g9bWFzdGVyKQpbIVtHaXRIdWIgbGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9zcG90aWZ5L3NjaW8uc3ZnKV0oLi9MSUNFTlNFKQpbIVtNYXZlbiBDZW50cmFsXShodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9pby5naXRodWIub3N2YWxkYS9QaXRheWEuc3ZnP2xhYmVsPU1hdmVuJTIwQ2VudHJhbCldKGh0dHBzOi8vY2VudHJhbC5zb25hdHlwZS5jb20vYXJ0aWZhY3QvaW8uZ2l0aHViLm9zdmFsZGEvUGl0YXlhLykKCkFQSSBjb3ZlcmFnZSB2aXN1YWxpemVyIHRvb2wuIENyZWF0ZXMgYW4gZWFzaWx5IHJlYWRhYmxlIGh0bWwgcmVwb3J0IGJhc2VkIHVwb24gdGhlIGV4ZWN1dGVkIEFQSSB0ZXN0IGNhc2VzLgoKPiBDdXJyZW50bHkgW1Rlc3ROR10gYW5kIFtKVW5pdDVdIGFyZSB0aGUgc3VwcG9ydGVkIGZyYW1ld29ya3MKCiMjIFVzYWdlCiMjIyBEZXBlbmRlbmN5CgoqKk1hdmVuKio6CmBgYHhtbAo8ZGVwZW5kZW5jeT4KICAgIDxncm91cElkPmlvLmdpdGh1Yi5vc3ZhbGRhPC9ncm91cElkPgogICAgPGFydGlmYWN0SWQ+UGl0YXlhPC9hcnRpZmFjdElkPgogICAgPHZlcnNpb24+MS4yLjI8L3ZlcnNpb24+CiAgICA8c2NvcGU+dGVzdDwvc2NvcGU+CjwvZGVwZW5kZW5jeT4KYGBgCgoqKkdyYWRsZSoqOgoKYGBgR3Jvb3Z5CmRlcGVuZGVuY2llcyB7CiAgICB0ZXN0Q29tcGlsZSgiaW8uZ2l0aHViLm9zdmFsZGE6UGl0YXlhOjEuMi4yIikKfQpgYGAKCiMjIyBDb25maWd1cmF0aW9uCgojIyMjIElucHV0IGZpbGUKCiMjIyMjIFBpdGF5YSBlbmRwb2ludCBsaXN0IGZpbGUKQ3JlYXRlIGEgdGV4dCBmaWxlIGluIHlvdXIgcmVzb3VyY2UgZGlyZWN0b3J5IGFuZCBhZGQgbGluZXMgaW4gdGhlIGZvbGxvd2luZyBmb3JtYXQ6CgpgYGAKREVMRVRFIC9wb3N0cy97cG9zdF9pZH0sIFBvc3RzCgpHRVQgL3Bvc3RzL3twb3N0X2lkfS9jb21tZW50cywgQ29tbWVudHMKR0VUIC9jb21tZW50cy97cG9zdF9pZH0sIENvbW1lbnRzCgojIGNvbW1lbnQgbGluZQojIGEgc3RhciBtYXJrcyB0aGUgZW5kcG9pbnQgdG8gYmUgaWdub3JlZAoqIEdFVCAvcG9zdHMve3Bvc3RfaWR9L3BpY3MsIFBpY3R1cmVzCgpHRVQgL3Bvc3RzL3twb3N0X2lkfS9waWNzLCBQaWN0dXJlcwouLi4KYGBgCgojIyMjIyBPcGVuQVBJIFZlcnNpb24gMyBmaWxlCkNvcHkgdGhlIG9wZW4gYXBpIGZpbGUgKGpzb24gb3IgeWFtbCkgdG8gcmVzb3VyY2VzIGRpcmVjdG9yeSBvciB1c2UgaXRzIFVSTAoKIyMjIyBQcm9wZXJ0aWVzIGZpbGUKCkNyZWF0ZSBhIGBwaXRheWEucHJvcGVydGllc2AgZmlsZSBpbnRvIHRoZSByb290IG9mIHlvdXIgcmVzb3VyY2VzIGRpcmVjdG9yeSBhbmQKYWRkIHRoZSBmb2xsb3dpbmcgcHJvcGVydGllcyB0byBpdDoKCnwgS2V5IHwgVmFsdWUgfCBNYW5kYXRvcnkgfAp8IC0tLS0tLSB8IC0tLS0tLSB8IC0tLS0tLSB8CnwgYXBwbGljYXRpb24ubmFtZSB8IFRoZSBuYW1lIG9mIHlvdXIgU1VUIHwgWWVzIHwgCnwgZW5kcG9pbnQubGlzdC5pbnB1dCB8IFRoZSBlbmRwb2ludCBsaXN0IGZpbGUncyByZWxhdGl2ZSBwYXRoIG9yIE9wZW4gQVBJIGZpbGUgVVJMfCBZZXMgfAoKIyMjIFRlc3QgY2FzZSBtb2RpZmljYXRpb25zCgojIyMjIFRlc3RORwoKVG8gdXNlIFBpdGF5YSB3aXRoIFRlc3RORyBhZGQgdGhlIHJlcG9ydGVyIHRvIHRoZSBsaXN0ZW5lcnMgaW4geW91ciB0ZXN0IGNsYXNzKGVzKToKYGBgamF2YQpATGlzdGVuZXJzKHtQaXRheWFDb3ZlcmFnZVJlcG9ydGVyLmNsYXNzfSkKcHVibGljIGNsYXNzIFlvdXJUZXN0TkdUZXN0Q2xhc3MgewogICAgLy8gLi4uCn0KYGBgCgojIyMjIEpVbml0IDUKClRvIHVzZSBQaXRheWEgd2l0aCBKVW5pdDUgIGFkZCB0aGUgZXh0ZW5zaW9uIHRvIHRoZSBleHRlbnNpb25zIGluIHlvdXIgdGVzdCBjbGFzcyhlcyk6CmBgYGphdmEKQEV4dGVuZFdpdGgoUGl0YXlhQ292ZXJhZ2VFeHRlbnNpb24uY2xhc3MpCnB1YmxpYyBjbGFzcyBZb3VySlVuaXQ1VGVzdENsYXNzIHsKICAgIC8vIC4uLgp9CmBgYAoKIyMjIyBJbiBUZXN0IE1ldGhvZHMKCkFubm90YXRlIGFsbCB5b3VyIHRlc3QgbWV0aG9kcyB3aGljaCB5b3Ugd2FudCB0byBpbmNsdWRlIGluIHRoZSByZXBvcnQ6CmBgYGphdmEKQFRlc3QKQFRlc3RDYXNlU3VwcGxlbWVudGFyeShhcGkgPSB7R0VUICsgIi9vbmVfb2ZfeW91cl9lbmRwb2ludCJ9KQpwdWJsaWMgdm9pZCB5b3VyVGVzdE1ldGhvZCgpIHsKICAgIC8vIC4uLgp9CmBgYAoKIyMgUHJldmlldwoKPGEgaHJlZj0iaHR0cHM6Ly9pLmliYi5jby9kd01neDFINy9kYXNoLnBuZyIgdGFyZ2V0PSJfYmxhbmsiPgogICAgPGltZyB3aWR0aD0iNDI1IiBoZWlnaHQ9IjIxNyIgYWx0PSJjaGFydHMiIHNyYz0iaHR0cHM6Ly9pLmliYi5jby9kd01neDFINy9kYXNoLnBuZyI+CjwvYT4KPGEgaHJlZj0iaHR0cHM6Ly9pLmliYi5jby9XTlk1V3F6US9lbmRwb2ludC1MaXN0LnBuZyIgdGFyZ2V0PSJfYmxhbmsiPgogICAgPGltZyB3aWR0aD0iNDI1IiBoZWlnaHQ9IjIxNyIgYWx0PSJlbmRwb2ludCBsaXN0IiBzcmM9Imh0dHBzOi8vaS5pYmIuY28vV05ZNVdxelEvZW5kcG9pbnQtTGlzdC5wbmciPgo8L2E+CgojIyBDSSBJc3N1ZXMgYW5kIHNvbHV0aW9ucwoKIyMjIEplbmtpbnMKCklmIEplbmtpbnMgZmFpbHMgdG8gZGlzcGxheSBjc3MsIHRoZW4gcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCBpbiBgTWFuYWdlIEplbmtpbnNgIC8gYFNjcmlwdCBjb25zb2xlYApgYGAKU3lzdGVtLnNldFByb3BlcnR5KCJodWRzb24ubW9kZWwuRGlyZWN0b3J5QnJvd3NlclN1cHBvcnQuQ1NQIiwgIiIpCmBgYAoKCltUZXN0TkddOiA8aHR0cHM6Ly90ZXN0bmcub3JnL2RvYy8+CltKVW5pdDVdOiA8aHR0cHM6Ly9qdW5pdC5vcmcvanVuaXQ1Lz4= readmeEtag: '"d5ded8e84b91130ed210cbf9e83a83fd5b97178b"' readmeLastModified: Sat, 22 Feb 2025 12:50:01 GMT repositoryId: 284689009 description: API test coverage reporter for TestNG and JUnit5 created: '2020-08-03T12:04:28Z' updated: '2025-02-22T12:50:06Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: osvalda logo: https://avatars.githubusercontent.com/u/29924475?v=4 license: Apache-2.0 repoEtag: '"a46222759036b4bc8dea2170c41f106b1753c0bdb4186be7e4e05916c672c722"' repoLastModified: Sat, 22 Feb 2025 12:50:06 GMT foundInMaster: true category: Description Validators id: 9d42f0c86da6d152bdc8bb7a06122b00 - source: openapi3 tags repository: https://github.com/api-stuff/spectral-spelling-grammar v3: true repositoryMetadata: base64Readme: >- IyBTcGVjdHJhbCBTcGVsbGluZyBhbmQgR3JhbW1hciBSdWxlc2V0CgpPbmUgb2YgdGhlIHRoaW5ncyBJIGZpbmQgYWNyb3NzIHRoZSBjb250cmFjdHMgSSB3b3JrIG9uIGlzIHRoZSBsYWNrIG9mIGF0dGVudGlvbiB0byBkZXRhaWwgaW4gZ2V0dGluZyBzcGVsbGluZyBhbmQgZ3JhbW1hciBjb3JyZWN0IGluIE9wZW5BUEkgZG9jdW1lbnRzLiBTdXJlLCBtb3N0IGRldmVsb3BlcnMgZG9uJ3QgY2FyZSBhYm91dCB0aGlzIHN0dWZmLCBidXQgaXQgdGFrZXMgdGhlIHNoaW5lIG9mIHdoYXQgc2hvdWxkIGJlIGEgZnVuZGFtZW50YWwgcGFydCBvZiB0aGUgcHJvZHVjdCBvZmZlcmluZy4KCkkndmUgdGhlcmVmb3JlIGtub2NrZWQgdG9nZXRoZXIgdGhpcyBbU3BlY3RyYWxdKGh0dHBzOi8vbWV0YS5zdG9wbGlnaHQuaW8vZG9jcy9zcGVjdHJhbC9aRzlqT2pZeC1vdmVydmlldykgcnVsZXNldCBhcyBhIGxpdHRsZSBwcm9vZi1vZi1jb25jZXB0LiBJdCB1c2VzIGEgW0N1c3RvbSBGdW5jdGlvbl0oaHR0cHM6Ly9tZXRhLnN0b3BsaWdodC5pby9kb2NzL3NwZWN0cmFsL1pHOWpPakkxTVRrdy1jdXN0b20tZnVuY3Rpb25zKSBhbmQgdGhlIEF0b20gcGFja2FnZSBbYHNwZWxsY2hlY2tlcmBdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3NwZWxsY2hlY2tlciksIHdoaWNoIG9mZmVycyB1cCBzb21lIGJpbmRpbmdzIHRvIHRoZSBzeXN0ZW0gc3BlbGwgY2hlY2tlci4gTm90ZSAqKm5vIGdyYW1tYXIgZmVhdHVyZXMgYXJlIGltcGxlbWVudGVkIHJpZ2h0IG5vdyAtIHRvIGJlIGltcGxlbWVudGVkIGluIGR1ZSBjb3Vyc2UqKi4KCkl0J3MganVzdCBzb21ldGhpbmcgZm9yIGZvbGtzIHRvIHBpY2sgdXAgYW5kIHVzZS4gVGhlIGJpZ2dlc3QgZHJhd2JhY2sgd2l0aCB0aGUgYXBwcm9hY2ggaXMgdGhlIGJpbmRpbmcgdG8gdGhlIHN5c3RlbSBkaWN0aW9uYXJ5LCB3aGljaCBtYXkgb2YgY291cnNlIGJlIGluIGEgZGlmZmVyZW50IGxhbmd1YWdlIHRvIHRoZSBvbmUgdGhlIE9wZW5BUEkgZG9jdW1lbnQgaXMgYmVpbmcgYXV0aG9yZWQgaW4uIEhvd2V2ZXIsIEknbSBzdXJlIGNsZXZlciBmb2xrcyBvdXQgdGhlcmUgY2FuIGlyb24gdGhpcyB3cmlua2xlIG91dCAocHJvYmFibHkgYnkgc3dhcHBpbmcgdGhlIGBzcGVsbGNoZWNrZXJgIGRlcGVuZGVuY3kgZm9yIHNvbWV0aGluZyBhIGJpdCBtb3JlICJmbGV4aWJsZSIpLgoKIyMgRXhhbXBsZQoKQnkgd2F5IG9mIGV4ZW1wbGFyLCBJJ3ZlIGtub2NrZWQgdG9nZXRoZXIgYW4gW09wZW5BUEkgZG9jdW1lbnRdKHRlc3QvbWlzc3BlbHQtb3BlbmFwaS55YW1sKSB3aXRoIGEgYnVuY2ggb2Ygc3BlbGxpbmcgbWlzdGFrZXMuIElmIHlvdSBydW4gdGhlIGZvbGxvd2luZzoKCmBgYGJhc2gKeWFybiBpbnN0YWxsCnlhcm4gcnVuIGxpbnQgcnVsZXNldHMvc3BlbGwtY2hlY2sueWFtbCB0ZXN0L21pc3NwZWx0LW9wZW5hcGkueWFtbCAjwqBGb3IgY29udmVuaWVuY2UsIGlkZWFsbHkgaW5zdGFsbCBTcGVjdHJhbCBnbG9iYWxseQpgYGAKCllvdSBzaG91bGQgZ2V0IHRoZSBmb2xsb3dpbmcgb3V0cHV0OgoKYGBgYmFzaArinpwgIHNwZWN0cmFsLXNwZWxsaW5nLWdyYW1tYXIgZ2l0OihtYXN0ZXIpIOKclyB5YXJuIHJ1biBsaW50IHJ1bGVzZXRzL3NwZWxsLWNoZWNrLnlhbWwgdGVzdC9taXNzcGVsdC1vcGVuYXBpLnlhbWwKeWFybiBydW4gdjEuMjIuMTcKJCBucHggc3BlY3RyYWwgbGludCAtLXJ1bGVzZXQgcnVsZXNldHMvc3BlbGwtY2hlY2sueWFtbCB0ZXN0L21pc3NwZWx0LW9wZW5hcGkueWFtbAoKL1VzZXJzL2NocmlzL0RvY3VtZW50cy9naXQvZ2l0aHViL2FwaS1zdHVmZi9zcGVjdHJhbC1zcGVsbC1jaGVjay90ZXN0L21pc3NwZWx0LW9wZW5hcGkueWFtbAogIDM6MTAgIHdhcm5pbmcgIGNoZWNrLXRpdGxlLXNwZWxsaW5nICAgICAgICBNaXNzcGVsdCB3b3JkcyBmb3VuZCBpbiB0ZXh0OiBjb25jcHRlICAgICAgICAgICBpbmZvLnRpdGxlCiAgNDoxNiAgd2FybmluZyAgY2hlY2stZGVzY3JpcHRpb24tc3BlbGxpbmcgIE1pc3NwZWx0IHdvcmRzIGZvdW5kIGluIHRleHQ6IGJkYWx5LCBzbWVhbHQgICAgIGluZm8uZGVzY3JpcHRpb24KICA5OjE2ICB3YXJuaW5nICBjaGVjay1zdW1tYXJ5LXNwZWxsaW5nICAgICAgTWlzc3BlbHQgd29yZHMgZm91bmQgaW4gdGV4dDogRGVmYXV0bCAgICAgICAgICAgcGF0aHMuL3Rlc3QuZ2V0LnN1bW1hcnkKIDEwOjIwICB3YXJuaW5nICBjaGVjay1kZXNjcmlwdGlvbi1zcGVsbGluZyAgTWlzc3BlbHQgd29yZHMgZm91bmQgaW4gdGV4dDogc3BlbGluZywgdmVlZXJ5eSAgcGF0aHMuL3Rlc3QuZ2V0LmRlc2NyaXB0aW9uCiAxMzoyNCAgd2FybmluZyAgY2hlY2stZGVzY3JpcHRpb24tc3BlbGxpbmcgIE1pc3NwZWx0IHdvcmRzIGZvdW5kIGluIHRleHQ6IERlZmFsdCAgICAgICAgICAgIHBhdGhzLi90ZXN0LmdldC5yZXNwb25zZXNbMjAwXS5kZXNjcmlwdGlvbgoK4pyWIDUgcHJvYmxlbXMgKDAgZXJyb3JzLCA1IHdhcm5pbmdzLCAwIGluZm9zLCAwIGhpbnRzKQrinKggIERvbmUgaW4gMS4zN3MuCuKenCAgc3BlY3RyYWwtc3BlbGxpbmctZ3JhbW1hciBnaXQ6KG1hc3Rlcikg4pyXCmBgYAoKU2ltcGxlIHJlYWxseS4KCiMjIEFuYXRvbXkKClRoZSBhbmF0b215IG9mIHRoaXMgcHJvamVjdCBpcyBwcmV0dHkgYmFzaWMgYW5kIHJlZmxlY3RzIHRoZSB1c2Ugb2YgQ3VzdG9tIEZ1bmN0aW9ucyBpbiBTcGVjdHJhbC4KCj4gSWYgeW91IGFyZSBub3QgZmFtaWxpYXIgd2l0aCBTcGVjdHJhbCBhbmQgd291bGQgbGlrZSBhIGRlZXAgZGl2ZSBjaGVja291dCBteSBbc291cCB0byBudXRzIGV4YW1wbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9hcGktc3R1ZmYvaGFuZHMtb24td2l0aC1zcGVjdHJhbCkgb3IgbXkgW0FTQyBwcmVzZW50YXRpb25dKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9cmRkdlhnbnFlSlEpIGZvciBhIHF1aWNrIG92ZXJ2aWV3LgoKVGhlIGZ1bmRhbWVudGFsIHBhcnRzIGFyZToKCiogQSBbY3VzdG9tIHJ1bGVzZXRdKHJ1bGVzZXRzL3NwZWxsLWNoZWNrLnlhbWwpIHRoYXQgaW1wbGVtZW50cyBydWxlcyBiYXNlZCBvbiBmaW5kaW5nIGFueSBgZGVzY3JpcHRpb25gLCBgc3VtbWFyeWAgb3IgYHRpdGxlYCBwcm9wZXJ0eSBpbiB0aGUgT3BlbkFQSSBkb2N1bWVudC4gVGhlIHNldmVyaXR5IGlzIHNldCBhcyBhIHdhcm5pbmcgaW4gdGhpcyBydWxlc2V0LgoqIEVhY2ggb2YgdGhvc2UgcnVsZXMgaW52b2tlcyB0aGUgc3BlbGwgY2hlY2sgW2N1c3RvbSBmdW5jdGlvbl0ocnVsZXNldHMvZnVuY3Rpb25zL3NwZWxsLWNoZWNrLmpzKSB3aGljaCBzcGxpdHMgdGhlIHN0cmluZyBpbnRvIHdvcmRzIGFuZCB0aGVuIHJ1bnMgdGhlIGBzcGVsbGNoZWNrZXIuaXNNaXNzcGVsbGVkYCBmdW5jdGlvbiBvbiBlYWNoIHRvIGZpbmQgdGhlIG1pc3NwZWx0IHdvcmRzLiBUaGUgd29yZHMgYXJlIHJldHVybmVkIGluIHRoZSBtZXNzYWdlLgoKSWYgeW91IHdhbnQgdG8gdXNlIHRoaXMgcnVsZXNldCBqdXN0IGNsb25lIHRoaXMgcmVwb3NpdG9yeSwgZ3JhYiB0aGUgcnVsZXNldCBhbmQgZnVuY3Rpb24gYW5kIHJlZmVyZW5jZSB0aGVtIHRodXMgaW4geW91ciBydWxlc2V0IChJIG1pZ2h0IHBhY2thZ2UgdGhpcyBzdHVmZiBsYXRlciBpZiBJIGFkZCBtb3JlIGZ1bmN0aW9ucyk6CgpgYGB5YW1sCmV4dGVuZHM6CiAgLSAuL3NwZWxsLWNoZWNrLnlhbWwKYGBgCgpCb2JzIHlvdXIgVW5jbGUgOnRodW1ic3VwOi4= readmeEtag: '"da82af4316efb4b421e8b5d49e5cdd4ad11f1d86"' readmeLastModified: Thu, 03 Mar 2022 18:14:48 GMT repositoryId: 465832120 description: >- A proof-of-concept for using Spectral to drive spelling and grammar checking in OpenAPI documents created: '2022-03-03T18:06:05Z' updated: '2022-10-21T04:52:59Z' language: JavaScript archived: false stars: 2 watchers: 1 forks: 1 owner: api-stuff logo: https://avatars.githubusercontent.com/u/68026188?v=4 repoEtag: '"81dda7cd9608e4bad2c6459f45ccc646505873e2fa8492293a511f75d163dd68"' repoLastModified: Fri, 21 Oct 2022 04:52:59 GMT foundInMaster: true category: Description Validators id: e6a9aa0225aacef9eaa6bbdfadfc63ed - source: openapi3 tags repository: https://github.com/manuzhang/akka-http-petstore v3: true repositoryMetadata: base64Readme: >- IyBha2thLWh0dHAtcGV0c3RvcmUgWyFbQ29udGludW91cyBJbnRlZ3JhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL21hbnV6aGFuZy9ha2thLWh0dHAtcGV0c3RvcmUvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sL2JhZGdlLnN2Zz9icmFuY2g9bWFpbildKGh0dHBzOi8vZ2l0aHViLmNvbS9tYW51emhhbmcvYWtrYS1odHRwLXBldHN0b3JlL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbCkK readmeEtag: '"e3ca7355df147b706a5b46774504ae31b2dc3803"' readmeLastModified: Tue, 16 Jul 2024 06:39:09 GMT repositoryId: 326146968 description: >- An akka-http server which implements OpenAPI 3 Spec of https://petstore3.swagger.io. created: '2021-01-02T09:12:55Z' updated: '2024-07-16T06:39:12Z' language: Scala archived: false stars: 2 watchers: 1 forks: 1 owner: manuzhang logo: https://avatars.githubusercontent.com/u/1191767?v=4 license: Apache-2.0 repoEtag: '"b1fb874d17c40ce82209b0efbbde5450e0ea1a99ff724b7dae5c838b4ee63c7c"' repoLastModified: Tue, 16 Jul 2024 06:39:12 GMT foundInMaster: true category: - Server - SDK id: 31e653c1ce6f0fa9b3fc293a6d8974ab - source: openapi3 tags repository: https://github.com/tada5hi/trapi v3: true id: b61b0e05c005d9efb075eef2edb804e4 repositoryMetadata: base64Readme: >- IyBUUkFQSSDwn6acCgpbIVttYWluXShodHRwczovL2dpdGh1Yi5jb20vVGFkYTVoaS90cmFwaS9hY3Rpb25zL3dvcmtmbG93cy9tYWluLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vVGFkYTVoaS90cmFwaS9hY3Rpb25zL3dvcmtmbG93cy9tYWluLnltbCkKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL1RhZGE1aGkvdHJhcGkvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPVpVSjhGNVRUU1gpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvVGFkYTVoaS90cmFwaSkKWyFbS25vd24gVnVsbmVyYWJpbGl0aWVzXShodHRwczovL3NueWsuaW8vdGVzdC9naXRodWIvVGFkYTVoaS90cmFwaS9iYWRnZS5zdmcpXShodHRwczovL3NueWsuaW8vdGVzdC9naXRodWIvVGFkYTVoaS90cmFwaSkKCiMjIFdoYXQgaXMgaXQ/CioqVCoqeXBlU2NyaXB0ICoqUioqZXN0ICoqQVBJKiogaXMgYSBjb2xsZWN0aW9uIG9mIHBhY2thZ2VzIHRvIGNyZWF0ZS9nZW5lcmF0ZToKLSBNZXRhZGF0YSBmb3IgUkVTVC1BUElzCi0gU3dhZ2dlciBkb2N1bWVudGF0aW9uCgoqKlRhYmxlIG9mIENvbnRlbnRzKioKCi0gW1BhY2thZ2VzXSgjcGFja2FnZXMpCi0gW0RvY3VtZW50YXRpb25dKCNkb2N1bWVudGF0aW9uKQotIFtMaWNlbnNlXSgjbGljZW5zZSkKCiMjIFBhY2thZ2VzClRoZSByZXBvc2l0b3J5IGNvbnRhaW5zIHRoZSBmb2xsb3dpbmcgcGFja2FnZXM6CgotICoqQHRyYXBpL21ldGFkYXRhKio6IEEgcGFja2FnZSBmb3IgZ2VuZXJhdGluZyBtZXRhZGF0YSBpbmZvcm1hdGlvbiBieSBhbmFseXppbmcgcHJlc2VudCBkZWNvcmF0b3JzLgogIFRoZSBtZXRhZGF0YSBjYW4gdGhhbiBiZSB1c2VkIGZvciBnZW5lcmF0aW5nIGEgZG9jdW1lbnRhdGlvbiBhY2NvcmRpbmcgdG8gdGhlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBvciB0byBjcmVhdGUgcm91dGUgc2NoZW1hL2hhbmRsaW5nIGZvciBsaWJyYXJpZXMgbGlrZTogZXhwcmVzcywga29hLCBldGMuCi0gKipAdHJhcGkvc3dhZ2dlcioqOiBBIHBhY2thZ2UgdG8gZ2VuZXJhdGUgYSBmdWxseSBmZWF0dXJlZCBkb2N1bWVudGF0aW9uIGFjY29yZGluZyB0aGUgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIGZyb20gZ2l2ZW4gbWV0YWRhdGEuCgojIyBEb2N1bWVudGF0aW9uCgpUbyByZWFkIHRoZSBkb2NzLCB2aXNpdCBbaHR0cHM6Ly90cmFwaS50YWRhNWhpLm5ldF0oaHR0cHM6Ly90cmFwaS50YWRhNWhpLm5ldCkKCiMjIExpY2Vuc2UKCk1hZGUgd2l0aCDwn5KaCgpQdWJsaXNoZWQgdW5kZXIgW01JVCBMaWNlbnNlXSguL0xJQ0VOU0UpLgo= readmeEtag: '"aef84820ffebd461880e9c40d3019fb4f2289e10"' readmeLastModified: Mon, 24 Jun 2024 09:22:00 GMT repositoryId: 526931733 description: >- TRAPI is a collection of packages to create/generate metadata for REST-APis and generate swagger documentations. created: '2022-08-20T13:17:32Z' updated: '2025-07-29T14:29:00Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 0 owner: tada5hi logo: https://avatars.githubusercontent.com/u/13162758?v=4 license: MIT repoEtag: '"e77639fce3759123c950cedb683ca191966732501c8b3a99a67df4b24fbfafa5"' repoLastModified: Tue, 29 Jul 2025 14:29:00 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/egoodhall/openapi-tooling v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLXRvb2xpbmcgIVtCdWlsZF0oaHR0cHM6Ly9naXRodWIuY29tL2VtbTAzNS9vcGVuYXBpLXRvb2xpbmcvd29ya2Zsb3dzL0J1aWxkL2JhZGdlLnN2ZykK readmeEtag: '"a5566880aaf35c4575b8123f0c4c740237ba2f9b"' readmeLastModified: Tue, 14 Mar 2023 22:07:46 GMT repositoryId: 278534292 description: Java tooling for OpenAPI created: '2020-07-10T04:02:43Z' updated: '2024-05-01T08:02:44Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: egoodhall logo: https://avatars.githubusercontent.com/u/13439972?v=4 license: MIT repoEtag: '"850ccb201e5285aff1337d7c97d518b5370e044718a8ec18e43aadc8f0c9935d"' repoLastModified: Wed, 01 May 2024 08:02:44 GMT foundInMaster: true category: - Low-level Tooling - SDK - Parsers id: 6c7ef7e719292d4c7dfd7cf06e78b634 oldLocations: - https://github.com/emm035/openapi-tooling - source: openapi3 tags repository: https://github.com/sky0621/cv-admin v3: true id: a6100588c4145919226457bbbf7afee3 repositoryMetadata: base64Readme: >- IyBjdi1hZG1pbgoKR2l0SHViIFBhZ2VzIOOBq+OBpuWFrOmWi+OBl+OBpuOBhOOCiyBbQ3VycmljdWx1bS1WaXRhZV0oaHR0cHM6Ly9za3kwNjIxLmdpdGh1Yi5pby9jdi8pIOOBruODh+ODvOOCv+OCveODvOOCueOCkueuoeeQhuOBmeOCi+OAggoK44OH44O844K/44K944O844K5566h55CG44Gr44GvIFtTUUxpdGVdKGh0dHBzOi8vd3d3LnNxbGl0ZS5vcmcvaW5kZXguaHRtbCkg44KS5L2/44GG44CCCgrjg4fjg7zjgr/jgr3jg7zjgrnjgbjjga7jgqLjgq/jgrvjgrnnlKjjgasgV2ViIEFQSSDjgqjjg7Pjg4njg53jgqTjg7Pjg4jjgpLnlKjmhI/jgIIKCkFQSeWumue+qeOBryBbT3BlbkFQSSB2M10oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKSDjgavmupbmi6DjgIIKCllhbWzjga/jgIFb44GT44KMXSguL3NjaGVtYS9vcGVuYXBpLnltbCkg77yI44OX44Os44OT44Ol44O855So44GrZG9ja2VyZmlsZeabuOOBk+OBhuOBqOaAneOBo+OBn+OBkeOBqUlEReOBruODl+ODrOODk+ODpeODvOapn+iDveOBp+WNgeWIhuOBoOOBo+OBn+OBruOBp+ecgeeVpe+8ieOAggoKW3N3YWdnZXIuaW/jga7jgqjjg4fjgqPjgr9dKGh0dHBzOi8vZWRpdG9yLW5leHQuc3dhZ2dlci5pby8pIOOBq+iyvOOBo+OBpuOCguOBhOOBhOOAggoKIVvjgZPjgpPjgarmhJ/jgZhdKHBpY3MvY3ZhcGkucG5nKQoKT29wcy4g5pyq5a6f6KOF44GM55uu56uL44Gk44O744O744O744CCCgojIyBmdW5jdGlvbgoK4oC744Ot44O844Kr44Or5a6f6KGM44GvIFtnbyAodjEuMjQpXShodHRwczovL2dvLmRldi8pIOOCpOODs+OCueODiOODvOODq+a4iOOBv+OBjOWJjeaPkOOAggoKY3YtYWRtaW4g44OH44Kj44Os44Kv44OI44Oq55u05LiL44Gn5Lul5LiL44KS5Y+p44GP44CCCgojIyMgc2VydmVyCgrjg63jg7zjgqvjg6vjgadBUEnjgrXjg7zjg5Djg7zotbfli5XjgIIKCmBgYApnbyBydW4gc3JjL21haW4uZ28gc2VydmVyCmBgYAoKIyMjIG1pZ3JhdGUKCltlbnQuKE9STSldKGh0dHBzOi8vZW50Z28uaW8vamEvKSDjgaflrprnvqnjgZfjgZ/jg4bjg7zjg5bjg6vlrprnvqnjgpIgW1NRTGl0ZV0oaHR0cHM6Ly93d3cuc3FsaXRlLm9yZy9pbmRleC5odG1sKSDjg5XjgqHjgqTjg6vjgavlj43mmKDjgZnjgovjgIIKCmBgYApnbyBydW4gc3JjL21haW4uZ28gbWlncmF0ZQpgYGAKCiMjIyBleHBvcnQKCkFQSeOCteODvOODkOODvOOBq+OCouOCr+OCu+OCueOBl+OBpuWPluW+l+OBl+OBn+ODh+ODvOOCv+OCveODvOOCueOCkkpTT07lvaLlvI/jgafmjIflrprlhYjjgavjgqjjgq/jgrnjg53jg7zjg4jjgZnjgovjgIIKCuOBk+OCjOOCkuS9v+OBo+OBpiBbY3ZdKGh0dHBzOi8vZ2l0aHViLmNvbS9za3kwNjIxL2N2KSDjg6rjg53jgrjjg4jjg6rjga7jg5Xjg63jg7Pjg4jjgqjjg7Pjg4nmqZ/og73jgYzjgq3jg6Pjg6rjgqLjgrfjg7zjg4jjga5XZWLjg5rjg7zjgrjjgpLooajnpLrjgZnjgovjgIIKCuKAu0FQSeOCteODvOODkOODvOOBjOi1t+WLleOBl+OBpuOBhOOCi+OBk+OBqOOBjOWJjeaPkOOAggoKYGBgCmdvIHJ1biBzcmMvbWFpbi5nbyBleHBvcnQgLS11c2VyaWQg44CQ44Om44O844K244O8SUTjgpLmjIflrprjgJEgLS1kaXIg44CQ5Ye65Yqb5YWI44OR44K544KS5oyH5a6a44CRCmBgYAoKIyMjIHN1Ym1pc3Npb24KCkFQSeOCteODvOODkOODvOOBq+OCouOCr+OCu+OCueOBl+OBpuWPluW+l+OBl+OBn+ODh+ODvOOCv+OCveODvOOCueOCkkV4Y2Vs5b2i5byP44Gn44Ko44Kv44K544Od44O844OI44GZ44KL44CCCgrigLvntJnlqpLkvZPjgarjgYTjgZdFeGNlbOODleOCoeOCpOODq+OBp+OBruaPkOWHuuOCkuaxguOCgeOCieOCjOOBn+aZgueUqOOAggoK4oC7QVBJ44K144O844OQ44O844GM6LW35YuV44GX44Gm44GE44KL44GT44Go44GM5YmN5o+Q44CCCgpgYGAKZ28gcnVuIHNyYy9tYWluLmdvIHN1Ym1pc3Npb24gLS11c2VyaWQg44CQ44Om44O844K244O8SUTjgpLmjIflrprjgJEKYGBgCgrigLvmnKrjgrPjg5/jg4Pjg4jjga7jgIxgLnByaXZhdGUuaW5pYOOAjeODleOCoeOCpOODq++8iOS7peS4i+OBruOCiOOBhuOBquWGheWuue+8ieOCkuODl+ODreOCuOOCp+OCr+ODiOODq+ODvOODiOebtOS4i+OBq+e9ruOBj+OAggoKYGBgCnBhc3N3b3JkID0geHh4eHh4eHh4eAprYW5hID0g44Ok44Oe44OA44CA44K/44Ot44KmCm5hbWUgPSDlsbHnlLDjgIDlpKrpg44KbWFpbCA9IGhvZ2Vob2dlQGV4YW1wbGUuY29tCmVkdWNhdGlvbmFsQmFja2dyb3VuZCA9IDIwWFjlubQz5pyI44CA4peL4peL5aSn5a2m4pah4pah5a2m6YOo44CA5Y2S5qWtCmNpdHlPZlJlc2lkZW5jZSA9IOKXi+KXi+ecjOKWoeKWoeW4ggpuZWFyZXN0U3RhdGlvbiA9IEpS5p2x5pel5pys44CA4peL4peL6aeFCmN2V2ViID0gaHR0cHM6Ly9za3kwNjIxLmdpdGh1Yi5pby9jdi8KY3ZBZG1pbiA9IGh0dHBzOi8vZ2l0aHViLmNvbS9za3kwNjIxL2N2LWFkbWluI3N1Ym1pc3Npb24KYGBgCgojIyMjIEV4Y2Vs44K144Oz44OX44OrCgpzYW1wbGUvc2tpbGxfc2hlZXQueGxzeAoKIVtzMV0ocGljcy9za2lsbHNoZWV0MDEucG5nKQohW3MxXShwaWNzL3NraWxsc2hlZXQwMi5wbmcpCiFbczFdKHBpY3Mvc2tpbGxzaGVldDAzLnBuZykKIVtzMV0ocGljcy9za2lsbHNoZWV0MDQucG5nKQohW3MxXShwaWNzL3NraWxsc2hlZXQwNS5wbmcpCgrjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjgIDjg7sKCuOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOOAgOODuwoK44CA44CA44CA44CA44CA44CA44CA44CA44CA44CA44CA44CA44CA44CA44O7CgojIyBlbnYKCiMjIyBPUwoKYGBgCk1hYyBCb29rIEFpciBNMiAyMDIyClNvbm9tYSAxNC40LjEKYGBgCgojIyMgZ28KCmBgYArina8gZ28gdmVyc2lvbgpnbyB2ZXJzaW9uIGdvMS4yMi4wIGRhcndpbi9hcm02NApgYGAKCiMjIHNldHVwCgojIyMgcHJvamVjdAoKYGBgCiQgZ28gbW9kIGluaXQgZ2l0aHViLmNvbS9za3kwNjIxL2N2LWFkbWluCmdvOiBjcmVhdGluZyBuZXcgZ28ubW9kOiBtb2R1bGUgZ2l0aHViLmNvbS9za3kwNjIxL2N2LWFkbWluCmBgYAoKIyMjIG9wZW4tYXBpLWNvZGVnZW4KCmh0dHBzOi8vZ2l0aHViLmNvbS9nZXRraW4va2luLW9wZW5hcGkKCmh0dHBzOi8vZ2l0aHViLmNvbS9kZWVwbWFwL29hcGktY29kZWdlbgoKYGBgCmdvIGluc3RhbGwgZ2l0aHViLmNvbS9kZWVwbWFwL29hcGktY29kZWdlbi9jbWQvb2FwaS1jb2RlZ2VuQGxhdGVzdApgYGAKCiMjIyBlbnQKCmZyb20gZW50aXR5IG1vZGVsIHRvIGRiIHRhYmxlCgpgYGAKY2Qgc3JjL2VudApgYGAKCmBgYApnbyBydW4gLW1vZD1tb2QgZW50Z28uaW8vZW50L2NtZC9lbnQgbmV3IFVzZXIgVXNlckFjdGl2aXR5IFVzZXJRdWFsaWZpY2F0aW9uIFVzZXJDYXJlZXJHcm91cCBVc2VyQ2FyZWVyIFVzZXJDYXJlZXJEZXNjcmlwdGlvbiBVc2VyQ2FyZWVyUGVyaW9kIENhcmVlclRhc2sgQ2FyZWVyU2tpbGxHcm91cCBDYXJlZXJTa2lsbCBVc2VyTm90ZSBVc2VyTm90ZUl0ZW0gU2tpbGxUYWcgU2tpbGwKYGBgCgojIyMgY29icmEKCmh0dHBzOi8vZ2l0aHViLmNvbS9zcGYxMy9jb2JyYQoKaHR0cHM6Ly9naXRodWIuY29tL3NwZjEzL2NvYnJhLWNsaS9ibG9iL21haW4vUkVBRE1FLm1kCgpgYGAKZ28gaW5zdGFsbCBnaXRodWIuY29tL3NwZjEzL2NvYnJhLWNsaUBsYXRlc3QKYGBgCgpgYGAKY29icmEtY2xpIGluaXQKYGBgCgojIyMjIGFkZCBjb21tYW5kCgpgYGAKY2Qgc3JjCmBgYAoKYGBgCmNvYnJhLWNsaSBhZGQgc2VydmVyCmBgYAoKIyMjIG96em8tdmFsaWRhdGlvbgoKaHR0cHM6Ly9naXRodWIuY29tL2dvLW96em8vb3p6by12YWxpZGF0aW9u readmeEtag: '"853176dd61f51dc6d7c7859da9fb720448fdfad9"' readmeLastModified: Sat, 05 Apr 2025 13:40:12 GMT repositoryId: 533311575 description: CVのデータソース管理、Web API提供、Excel出力 created: '2022-09-06T12:24:32Z' updated: '2025-04-05T13:40:18Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: sky0621 logo: https://avatars.githubusercontent.com/u/15807041?v=4 license: MIT repoEtag: '"e9e24b8fb6892da92ea92167634701b887638800aef9439dfacc1973fbba1226"' repoLastModified: Sat, 05 Apr 2025 13:40:18 GMT category: Code Generators foundInMaster: true - source: openapi3 tags repository: https://github.com/roar-skinderviken/vicx-applications v3: true id: 3fdaadd39a480426d9ab38cf0fb25e57 repositoryMetadata: base64Readme: >- IyBWaWN4IEFwcGxpY2F0aW9ucwoKIyMgR2V0dGluZyBzdGFydGVkCgojIyMgQ3JlYXRlIG5leHQtYXBwLy5lbnYubG9jYWwKU2hvdWxkIGhhdmUgdGhlIGZvbGxvd2luZyBjb250ZW50CmBgYApPQVVUSF9DTElFTlRfU0VDUkVUPXNlY3JldApPQVVUSF9CQVNFX1VSTD1odHRwOi8vbG9jYWxob3N0OjkwMDAvYXV0aC1zZXJ2ZXIKTkVYVEFVVEhfU0VDUkVUPXNlY3JldApORVhUQVVUSF9VUkw9aHR0cDovL2xvY2FsaG9zdDozMDAwL2FwaS9hdXRoClJFQ0FQVENIQV9TSVRFX0tFWT02TGVJeEFjVEFBQUFBSmNaVlJxeUhoNzFVTUlFR05RX01YamlaS2hJCk5FWFRfUFVCTElDX0tNRUFOU19CQUNLRU5EX1VSTD1odHRwOi8vbG9jYWxob3N0OjgwMDAvay1tZWFucwpTUFJJTkdfQkFDS0VORF9CQVNFX1VSTD1odHRwOi8vbG9jYWxob3N0OjgwODAvYmFja2VuZC1zcHJpbmctYm9vdApgYGAKVmFsdWUgZm9yIGBSRUNBUFRDSEFfU0lURV9LRVlgIGlzIGEgZGV2IHZhbHVlLiAKCgojIyMgU3RhcnQgdGhlIGF1dGgtc2VydmVyIG9uIHBvcnQgOTAwMApgYGBzaGVsbAouL2dyYWRsZXcgLXAgYXV0aC1zZXJ2ZXIgYm9vdFJ1bgpgYGAKCiMjIyBTdGFydCB0aGUgTmV4dCBhcHAKYGBgc2hlbGwKY2QgbmV4dC1hcHAKYGBgCmBgYHNoZWxsCm5wbSBjaQpgYGAKYGBgc2hlbGwKbnBtIHJ1biBkZXYKYGBgCgpOZXh0IGFwcCB3aWxsIG5vdyBiZSBhdmFpbGFibGUgb24gCmh0dHA6Ly9sb2NhbGhvc3Q6MzAwMAoKVXNlcm5hbWUvcGFzc3dvcmQgZm9yIGxvZ2dpbmcgaW4gd2l0aCBPQXV0aDoKLSB1c2VyMQotIHBhc3N3b3JkCgojIyMgU3RhcnQgdGhlIGJhY2tlbmQtc3ByaW5nLWJvb3QgKG9wdGlvbmFsKQpgYGBzaGVsbAouL2dyYWRsZXcgLXAgYmFja2VuZC1zcHJpbmctYm9vdCBib290UnVuCmBgYAoKIyMjIFN0YXJ0IHRoZSBiYWNrZW5kLXB5dGhvbiAob3B0aW9uYWwsIHJlcXVpcmVzIFBvZXRyeSkKYGBgc2hlbGwKY2QgYmFja2VuZC1weXRob24KYGBgCmBgYHNoZWxsCnBvZXRyeSBpbnN0YWxsCmBgYApgYGBzaGVsbApwb2V0cnkgcnVuIHV2aWNvcm4gc3JjLmFwcDphcHAgLS1yZWxvYWQKYGBgCgojIyBBUEkgRG9jdW1lbnRhdGlvbgoKVGhlIE9wZW5BUEkgZG9jdW1lbnRhdGlvbiBmb3IgdGhlIEFQSSBpcyBhdmFpbGFibGUgYXQgdGhlIGZvbGxvd2luZyBVUkw6CgpbU3dhZ2dlciBVSV0oaHR0cDovL2xvY2FsaG9zdDo4MDgwL2JhY2tlbmQtc3ByaW5nLWJvb3Qvc3dhZ2dlci11aS9pbmRleC5odG1sKQoKWW91IGNhbiB1c2UgdGhpcyBpbnRlcmZhY2UgdG8gZXhwbG9yZSBhbmQgaW50ZXJhY3Qgd2l0aCB0aGUgQVBJIGVuZHBvaW50cywgdmlldyB0aGVpciBkZXNjcmlwdGlvbnMsIAphbmQgdGVzdCByZXF1ZXN0cyBkaXJlY3RseSBmcm9tIHRoZSBVSS4KCiMjIEdyYXBoUUwKCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9iYWNrZW5kLXNwcmluZy1ib290L2dyYXBoaXFsCgoKYGBgCm11dGF0aW9uIHsKICBjcmVhdGVDYWxjdWxhdGlvbigKICAgIGZpcnN0VmFsdWU6IDEsIAogICAgc2Vjb25kVmFsdWU6IDIsIAogICAgb3BlcmF0aW9uOiBQTFVTCiAgKSB7CiAgICBpZAogICAgZmlyc3RWYWx1ZQogICAgc2Vjb25kVmFsdWUKICAgIG9wZXJhdGlvbgogICAgcmVzdWx0CiAgICB1c2VybmFtZQogICAgY3JlYXRlZEF0CiAgfQp9CmBgYAoKYGBgCnF1ZXJ5IHsKICBnZXRBbGxDYWxjdWxhdGlvbnMocGFnZTogMCkgewogICAgY2FsY3VsYXRpb25zIHsKICAgICAgZmlyc3RWYWx1ZQogICAgICBzZWNvbmRWYWx1ZQogICAgICBvcGVyYXRpb24KICAgICAgcmVzdWx0CiAgICB9LAogICAgcGFnZSwKICAgIHRvdGFsUGFnZXMKICB9Cn0KYGBgCgpgYGAKbXV0YXRpb24gewogICAgZGVsZXRlQ2FsY3VsYXRpb25zKGlkczogWzEsMl0pCn0KYGBgCg== readmeEtag: '"ae52c3abf65413f89c755762e3fe07810d0607a1"' readmeLastModified: Tue, 01 Jul 2025 19:54:11 GMT repositoryId: 823250960 description: Repo for jr. and me to play with microk8s, Flux CD and GitHub Actions created: '2024-07-02T17:06:45Z' updated: '2026-02-02T22:12:43Z' language: Kotlin archived: false stars: 3 watchers: 1 forks: 0 owner: roar-skinderviken logo: https://avatars.githubusercontent.com/u/7499642?v=4 repoEtag: '"f6cc3d61442198bfe8df7dd0c1af107fdba68bcfe5d6a128a251844bee0148d7"' repoLastModified: Mon, 02 Feb 2026 22:12:43 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/kumuluz/kumuluzee-openapi-mp v3: true repositoryMetadata: base64Readme: >- # KumuluzEE OpenAPI MicroProfile
[![KumuluzEE CI](https://github.com/kumuluz/kumuluzee-openapi-mp/actions/workflows/kumuluzee-ci.yml/badge.svg)](https://github.com/kumuluz/kumuluzee-openapi-mp/actions/workflows/kumuluzee-ci.yml)

> KumuluzEE OpenAPI MicroProfile project provides powerful tools to incorporate the OpenAPI 3 specification to your
microservices in a standardized way.

KumuluzEE OpenAPI MicroProfile project allows you to document microservice APIs using OpenAPI v3 compliant annotations.
Project will automatically hook-up servlet that will serve your API specifications on endpoint `/openapi`.
The project implements the MicroProfile OpenAPI specification.
 
More details: 

- [OpenAPI v3 Specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md)
- [MicroProfile OpenAPI specification](https://github.com/eclipse/microprofile-open-api/releases)

## Usage

You can enable KumuluzEE OpenAPI MicroProfile support by adding the following dependency:
```xml
<dependency>
    <groupId>com.kumuluz.ee.openapi</groupId>
    <artifactId>kumuluzee-openapi-mp</artifactId>
    <version>${kumuluzee-openapi-mp.version}</version>
</dependency>
```

## OpenAPI configuration

When kumuluzee-openapi-mp dependency is included in the project, you can start documenting your REST API using
MicroProfile OpenAPI annotations.

### Documenting application class

```java
@SecurityScheme(securitySchemeName = "openid-connect", type = SecuritySchemeType.OPENIDCONNECT,
        openIdConnectUrl = "http://auth-server-url/.well-known/openid-configuration")
@ApplicationPath("v2")
@OpenAPIDefinition(info = @Info(title = "CustomerApi", version = "v2.0.0", contact = @Contact(), license = @License(name="something")), servers = @Server(url = "http://localhost:8080"), security
        = @SecurityRequirement(name = "openid-connect"))
public class CustomerApplication extends Application {
}
```

### Documenting resource class and operations

```java
@Path("customers")
@Consumes(MediaType.APPLICATION_JSON)
@Produces(MediaType.APPLICATION_JSON)
public class CustomerResource {

    @GET
    @Operation(summary = "Get customers details", description = "Returns customer details.")
    @APIResponses({
            @APIResponse(description = "Customer details", responseCode = "200", content = @Content(schema = @Schema(implementation =
                    Customer.class)))
    })
    @Path("{customerId}")
    public Response getCustomer(@PathParam("customerId") String customerId) {
        // ...
    }

}
```

## Accessing API specification

Build and run project using:

```bash
mvn clean package
java -jar target/${project.build.finalName}.jar
```

After startup API specification will be available at:

**http://<-hostname->:<-port->/<-optional-context-path->/openapi**

Example:

http://localhost:8080/openapi

Serving OpenAPI specification can be disabled by setting property **kumuluzee.openapi-mp.enabled** to false. By default
serving API spec is enabled.

## Configuration

The KumuluzEE OpenAPI MicroProfile extension can be configured with the standard KumuluzEE configuration mechanism. For
example with the _config.yml_ file:

```yaml
kumuluzee:
  openapi-mp:
    enabled: true
    servlet:
      mapping: /openapi-custom-mapping
    scan:
      packages: com.kumuluz.ee.samples.openapi
    servers: https://example-api.com,https://my-proxy.com
```

Some interesting configuration properties are:

- `kumuluzee.openapi-mp.enabled` - If set to `false` disables the extension (and OpenAPI servlet). Default value: `true`
- `kumuluzee.openapi-mp.servlet.mapping` - The endpoint at which the OpenAPI specification is available. Appended to optional server context path. Default value: `/openapi`
- `kumuluzee.openapi-mp.scan.packages` - Comma separated list of packages which are scanned for the OpenAPI annotations.
  By default, all packages are scanned.

Full list of configuration properties can be found in
[MicroProfile OpenAPI specification](https://github.com/eclipse/microprofile-open-api/releases).

## Scanning

By default KumuluzEE OpenAPI MP uses optimized scanning in order to reduce startup times. This means that only the main
application JAR will be scanned (main artifact). In order to scan additional artifacts you need to specify them using
the [scan-libraries mechanism](https://github.com/kumuluz/kumuluzee/pull/123). You need to include all dependencies
which contain JAX-RS application and resources as well as dependencies containing models returned from JAX-RS resources.
If all your models and resources are in the main artifact you don't need to include anything. For example to include
_my-models_ artifact use the following configuration:

```yaml
kumuluzee:
  dev:
    scan-libraries:
      - my-models
```

If you are unsure if your configuration is correct you can try to disable optimized scanning by using the following
configuration:

```yaml
kumuluzee:
  openapi-mp:
    scanning:
      optimize: false
```

You can also enable scan debugging by setting the following key to `true`: `kumuluzee.openapi-mp.scanning.debug`. This
will output a verbose log of scanning configuration and progress.

## Adding Swagger UI

To serve API specification in visual form and to allow API consumers to interact with API resources you can add
Swagger UI by including the __kumuluzee-swagger-ui__ dependency:

```xml
<dependency>
    <groupId>com.kumuluz.ee.openapi</groupId>
    <artifactId>kumuluzee-openapi-mp-ui</artifactId>
    <version>${kumuluzee-openapi-mp.version}</version>
</dependency>
```

Swagger UI is automatically enabled and is available at __/api-specs/ui__. In order to disable Swagger UI you can set
the configuration key `kumuluzee.openapi-mp.ui.enabled` to `false`. You can also remap the Swagger UI to another
location by setting the `kumuluzee.openapi-mp.ui.mapping` key (default value: `/api-specs/ui`). Path is appended to optional server context path.

Swagger UI needs to know where the OpenAPI specification is served from. It tries to define it from the following
sources:

1. Static: `<protocol>://localhost:<port>` (useful when nothing is defined, lowest priority)
1. From the `servers` parameter in `@OpenAPIDefinition` annotation (useful when OpenAPI specification is available from
   the same hostname as Swagger UI)
1. Configuration property: `kumuluzee.server.base-url` (useful for overriding above values)
1. Configuration property: `kumuluzee.openapi-mp.ui.specification-server` (same as above but in a namespace specific to
   this extension)
   
## Swagger UI server auto configuration

Server auto configuration allows for dynamic resolution of parameters like `kumuluzee.openapi-mp.ui.specification-server` 
instead of need to set it. This is done by retrieving necessary information from request during runtime and based on that 
setting correct redirect. It is especially useful when service is being deployed in multiple environments.
This feature also updates list of servers in Swagger UI and reorder them, so it will be first one.
It is also possible to resolve correct redirect when service is behind for example Ingres (Ingress must have enabled feature which 
forwards original URL).

By default, this feature is disabled for backward compatibility. To enable it set `kumuluzee.openapi-mp.ui.auto-config.enabled`
to `true`. To enable (Ingress) URI check set `kumuluzee.openapi-mp.ui.auto-config.original-uri-check` to `true`.

Default settings for server auto configuration:
```yaml
kumuluzee:
  openapi-mp:
    ui:
      server-auto-config:
        enabled: false
        original-uri-check: false
```

> **note:** localhost server with same port is added only once (if not already specified in @OpenAPIDefinition) and in case in which 
> is used localhost address in different format (for example http://127.0.0.1:8080 instead of http://localhost:8080), 
>no new server will be added to list and because of this Swagger UI will be unable to do API calls (this is due to issue with cors) 

## OpenAPI Maven Plugin

Maven plugin can be used to generate OpenAPI schema when building the application. To use it add the following to
pom.xml:

```xml
<plugin>
    <artifactId>kumuluzee-openapi-mp-maven-plugin</artifactId>
    <groupId>com.kumuluz.ee.openapi</groupId>
    <version>${kumuluzee-openapi-mp.version}</version>
    <executions>
        <execution>
            <goals>
                <goal>generate</goal>
            </goals>
        </execution>
    </executions>
</plugin>
```

By default, the schema is generated in `target/generated/` directory in JSON and YAML formats.

If the schema is defined across multiple artefacts you must define the `<scanLibraries></scanLibraries>` configuration
property with comma-separated list of dependencies in the same format as described in the [Scanning](#scanning) section
above.

Maven plugin uses SmallRye implementation in order to generate the schema. For more configuration options see the
following document: https://github.com/smallrye/smallrye-open-api/tree/master/tools/maven-plugin#configuration-options

## Changelog

Recent changes can be viewed on Github on the [Releases Page](https://github.com/kumuluz/kumuluzee-openapi-mp/releases)


## Contribute

See the [contributing docs](https://github.com/kumuluz/kumuluzee-openapi-mp/blob/master/CONTRIBUTING.md)

When submitting an issue, please follow the 
[guidelines](https://github.com/kumuluz/kumuluzee-openapi-mp/blob/master/CONTRIBUTING.md#bugs).

When submitting a bugfix, write a test that exposes the bug and fails before applying your fix. Submit the test 
alongside the fix.

When submitting a new feature, add tests that cover the feature.

## License

MIT
 readmeEtag: '"45ffb32ef3233290d30b372946fe1fd2f903faaa"' readmeLastModified: Tue, 07 Jun 2022 13:14:17 GMT repositoryId: 172050864 description: >- KumuluzEE OpenAPI MicroProfile project provides powerful tools to incorporate the OpenAPI 3 specification to your microservices in a standardized way. created: '2019-02-22T10:59:42Z' updated: '2022-06-07T13:02:06Z' language: Java archived: false stars: 2 watchers: 12 forks: 1 owner: kumuluz logo: https://avatars.githubusercontent.com/u/6859905?v=4 license: NOASSERTION repoEtag: '"e0a94d675f43d663394dae6e700d417bac76e86c5f9c7fa527b7271b807f1b5c"' repoLastModified: Tue, 07 Jun 2022 13:02:06 GMT foundInMaster: true category: Parsers id: dd2e2327f293e393564372d67494079c - source: openapi3 tags repository: https://github.com/sergiotm87/openfaas-openapi3-flask v3: true repositoryMetadata: base64Readme: >- CiMgW1dJUF0gT3BlbkZhYVMgcHl0aG9uLWZsYXNrLW9wZW5hcGkzIHRlbXBsYXRlCgpFeGFtcGxlIG9mIHB5dGhvbiB0ZW1wbGF0ZSBmb3IgW09wZW5GYWFzXShodHRwczovL2dpdGh1Yi5jb20vb3BlbmZhYXMvZmFhcykgd2l0aCBhdXRvbWF0aWMgZW5kcG9pbnQgdmFsaWRhdGlvbiBiYXNlZCBvbiBPcGVuQXBpMyBzY2hlbWEuCgpUaGlzIHR1dG9yaWFsIG1ha2UgdXNlIG9mIHRoZXNlIHByb2plY3RzOgoKaHR0cHM6Ly9naXRodWIuY29tL29wZW5mYWFzLWluY3ViYXRvci9weXRob24tZmxhc2stdGVtcGxhdGUKCmh0dHBzOi8vZ2l0aHViLmNvbS96YWxhbmRvL2Nvbm5leGlvbgoKIyMjIFJlcXVpcmVtZW50cwoKLSBbZmFhcy1jbGldKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuZmFhcy9mYWFzLWNsaSkKYGBgCmZhYXMtY2xpIG5ldyBwZXRzdG9yZSAtLWxhbmcgcHl0aG9uMy1mbGFzay1zd2FnZ2VyCmBgYAoKIyMjIEJ1aWxkICYgUnVuCgpgYGAKZmFhcy1jbGkgYnVpbGQgLWYgcGV0c3RvcmUueW1sCgpkb2NrZXIgcnVuIC0tcm0gLXAgODA4MDo4MDgwIC0tbmFtZSBwZXRzdG9yZSBwZXRzdG9yZSAvYmluL3NoIC1jICJmcHJvY2Vzcz1cInB5dGhvbjMgL2hvbWUvYXBwL2luZGV4LnB5XCI7IGZ3YXRjaGRvZyIKYGBgCgojIyMgQmFkIHJlcXVlc3QKCmBgYApjdXJsIC1YIFBPU1QgImh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC8iIC1IICAiYWNjZXB0OiBhcHBsaWNhdGlvbi9qc29uIiAtSCAgIkNvbnRlbnQtVHlwZTogYXBwbGljYXRpb24vanNvbiIgLWQgIntcImluYW1lXCI6XCJwZXRpbmFtZVwiLFwidGFnXCI6XCJwZXR0YWdcIn0iCmBgYAoKYGBgCnsKICAiZGV0YWlsIjogIiduYW1lJyBpcyBhIHJlcXVpcmVkIHByb3BlcnR5IiwKICAic3RhdHVzIjogNDAwLAogICJ0aXRsZSI6ICJCYWQgUmVxdWVzdCIsCiAgInR5cGUiOiAiYWJvdXQ6YmxhbmsiCn0KYGBgCgojIyMgR29vZCByZXF1ZXN0LCBiYWQgcmVzcG9uc2UKCmBgYApjdXJsIC1YIFBPU1QgImh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC8iIC1IICAiYWNjZXB0OiBhcHBsaWNhdGlvbi9qc29uIiAtSCAgIkNvbnRlbnQtVHlwZTogYXBwbGljYXRpb24vanNvbiIgLWQgIntcIm5hbWVcIjpcInBldG5hbWVcIixcInRhZ1wiOlwicGV0dGFnXCJ9IgpgYGAKCmBgYAp7CiAgImRldGFpbCI6ICIne1wibmFtZVwiOiBcInBldG5hbWVcIiwgXCJ0YWdcIjogXCJwZXR0YWdcIn0nIGlzIG5vdCBvZiB0eXBlICdvYmplY3QnXG5cbkZhaWxlZCB2YWxpZGF0aW5nICd0eXBlJyBpbiBzY2hlbWFbJ2FsbE9mJ11bMF06XG4gICAgeydwcm9wZXJ0aWVzJzogeyduYW1lJzogeyd0eXBlJzogJ3N0cmluZyd9LCAndGFnJzogeyd0eXBlJzogJ3N0cmluZyd9fSxcbiAgICAgJ3JlcXVpcmVkJzogWyduYW1lJ10sXG4gICAgICd0eXBlJzogJ29iamVjdCcsXG4gICAgICd4LXNjb3BlJzogWycnLCAnIy9jb21wb25lbnRzL3NjaGVtYXMvUGV0J119XG5cbk9uIGluc3RhbmNlOlxuICAgICd7XCJuYW1lXCI6IFwicGV0bmFtZVwiLCBcInRhZ1wiOiBcInBldHRhZ1wifSciLAogICJzdGF0dXMiOiA1MDAsCiAgInRpdGxlIjogIlJlc3BvbnNlIGJvZHkgZG9lcyBub3QgY29uZm9ybSB0byBzcGVjaWZpY2F0aW9uIiwKICAidHlwZSI6ICJhYm91dDpibGFuayIKfQpgYGAKCiMjIyBGaXggY29kZSAmIGdvb2QgcmVzcG9uc2UKCmBgYApzZWQgLWkgLWUgJ3MjIm1hZ2ljX2hhcHBlbmRzX2hlcmUiI3Jlc3VsdFsnIiciJ2lkJyInIiddID0gNDIjZycgcGV0c3RvcmUvaGFuZGxlci5weQpgYGAKCihidWlsZCBhbmQgcnVuIGFnYWluKQoKYGBgwroKY3VybCAtWCBQT1NUICJodHRwOi8vbG9jYWxob3N0OjgwODAvIiAtSCAgImFjY2VwdDogYXBwbGljYXRpb24vanNvbiIgLUggICJDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24iIC1kICJ7XCJuYW1lXCI6XCJwZXRuYW1lXCIsXCJ0YWdcIjpcInBldHRhZ1wifSIKYGBgCgpgYGAKewogICJpZCI6IDQyLAogICJuYW1lIjogInBldG5hbWUiLAogICJ0YWciOiAicGV0dGFnIgp9CmBgYAoKIyMgU3dhZ2dlci11aQoKaHR0cDovL2xvY2FsaG9zdDo4MDgwL3VpLw== readmeEtag: '"36ecf77fac74172cbefb7f1a7567b141e1353a90"' readmeLastModified: Tue, 15 Oct 2019 20:54:16 GMT repositoryId: 215393778 description: OpenFaas flask template with automatic endpoint validation created: '2019-10-15T20:48:00Z' updated: '2019-11-17T14:40:05Z' language: Dockerfile archived: false stars: 2 watchers: 1 forks: 0 owner: sergiotm87 logo: https://avatars.githubusercontent.com/u/12777256?v=4 repoEtag: '"bff212cd8f52857fcf74b209b90d6d02968c363fd061674737f5ba5f62615fef"' repoLastModified: Sun, 17 Nov 2019 14:40:05 GMT foundInMaster: true category: Data Validators id: b34d7cfb88aecc09582d877a336ccc6f - source: openapi3 tags repository: https://github.com/juandepalo/keycloak-aspnetcore v3: true repositoryMetadata: base64Readme: >- IyBLZXljbG9hayAvICBBU1AuTkVUIENvcmUgLyBTd2FnZ2VyIE5zd2FnCkVzdGEgYXBsaWNhY2nDs24gZXMgdW4gZWplbXBsbyBkZSBjb21vIHByb3RlZ2VyIEFTUC5ORVQgY29yZSAyLjIgY29uIHVuIHNlcnZpZG9yIGRlIGF1dGVudGlmaWNhY2lvbiBvYXV0aDIgS2V5Y2xvYWsuClRhbWJpZW4gc2UgaW1wbGVtZW50YSBlbCBjb25zdW1vIGRlIGxhIGFwaSBjb24gTnN3YWcgY29uIE9wZW5BcGkgMy4KCiFbXSguL2ltYWdlcy8yMDE5LTA3LTI5XzgtMDctNDUuZ2lmKQoKIyMgUmVzdW1lbgotIFNlcnZlcgogIC0gQVNQLk5FVCBDb3JlIDIuMgogIC0gRG9ja2VyOiBrZXljbG9hY2sgc2VydmVyIGNvbiBiYXNlIGRlIGRhdG9zIG15c3FsCi0gQ2xpZW50CiAgLSBOc3dhZy4gaW50ZXJmYXogZGUgdXN1YXJpbyB5IGVsIGdlbmVyYWRvciBkZSBTd2FnZ2VyCgoKIyMgU2V0dXAKCjEuIEluc3RhbGFjacOzbjoKICAgLSBbLk5FVCBDb3JlIDIuMl0oaHR0cHM6Ly9kb3RuZXQubWljcm9zb2Z0LmNvbS9sZWFybi9kb3RuZXQvaGVsbG8td29ybGQtdHV0b3JpYWwvaW5zdGFsbCkKICAgLSBbRG9ja2VyXShodHRwczovL2RvY3MuZG9ja2VyLmNvbS9kb2NrZXItZm9yLXdpbmRvd3MvaW5zdGFsbC8pCgoyLiBDbG9uYXIgZWwgcHJveWVjdG86CiAgICBgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9qdWFuZGVwYWxvL0tleWNsb2FrLWFzcG5ldGNvcmUuZ2l0YAoKCjMuIFNpdHVhcnNlIGVuIGVsIGRpcmVjdG9yaW86CgogICBjZCAuL0tleWNsb2FrLWFzcG5ldGNvcmUKNC4gZWplY3V0YXI6CgogICBgZG9ja2VyLWNvbXBvc2UgLWYgImRvY2tlci1jb21wb3NlLnltbCIgIHVwIC1kIC0tYnVpbGQgLS1yZW1vdmUtb3JwaGFuc2AKCjUuIEFicmlyIGVsIG5hdmVnYWRvciBbaHR0cDovL2xvY2FsaG9zdDo4MDgwXShodHRwOi8vbG9jYWxob3N0OjgwODApIFtodHRwOi8vbG9jYWxob3N0OjEwMDAxXShodHRwOi8vbG9jYWxob3N0OjEwMDAxKS4KCgohW2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MF0oLi9pbWFnZXMvMjAxOS0wNy0yOV84LTQwLTMwLnBuZykKCiFbaHR0cDovL2xvY2FsaG9zdDoxMDAwMV0oLi9pbWFnZXMvMjAxOS0wNy0yOV84LTQwLTM5LnBuZykKCiMjIENvbmZpZ3VyYWNpw7NuCgoxLiBNb2RpZmljYWNpb24gZGUgaG9zdC4KICAgMS4gIFBhcmEgZWplY3V0YXIgbG9jYWxtZW50ZSBtb2RpZmljYXJlbW9zIGVsIGZpY2hlcm8gJ0M6XFdpbmRvd3NcU3lzdGVtMzJcZHJpdmVyc1xldGNcaG9zdHMnIGHDsWFkaWVuZG8gZWwgbm9tYnJlIGRlbCBzZXJ2aWNpbyBkZSBLZXljbG9hcmsgcXVlIHNlIHB1ZWRlIG1vZGlmaWNhciBlbiBlbCBbZG9ja2VyLWNvbXBvc2VdKC4vZG9ja2VyLWNvbXBvc2UueW1sKS4KCiAgICAhW25vbWJyZXNlcnZpY2lva2V5Y2xvYWtdKC4vaW1hZ2VzL25vbWJyZVNlcnZpY2lvS2V5Y2xvYWsucG5nKQoKCgoKCiAgICBtb2RpZmljYWNpw7NuIGZpY2hlcm8gaG9zdHM6CgogICAgYDEyNy4wLjAuMSAgICAgICBsb2NhbGhvc3QgIGtleWNsb2Fsb2NhbGhvc3RgCgoyLiBDb25maWd1cmFjaW9uIGRlIEtleWNsb2FrLgogICAxLiBIZSBkZWphZG8gbGEgZXhwb3J0YWNpw7NuIGRlIGNvbmZpZ3VyYWNpw7NuIGRlIHVuIGNsaWVudGUgcGFyYSBlbCBlamVtcGxvIGVuIFtrZXljbG9hay1JbXBvcnQvcmVhbG0tZXhwb3J0Lmpzb25dKC4va2V5Y2xvYWstSW1wb3J0L3JlYWxtLWV4cG9ydC5qc29uKQoKUHVsc2Ftb3Mgc29icmUgQWRtaW5pc3RyYWRvciBjb25zb2xlLiBOb3Mgc29saWNpdGFyYSBsYXMgY3JlZGVuY2lhbGVzLgpQb3IgZGVmZWN0byBjb24gZG9ja2VyLWNvbXBvc2UgaGVtb3MgY3JlYWRvIGVsIHVzdWFyaW8gKiphZG1pbioqLCBjb24gY29udHJhc2XDsWEgKipQYTU1dzByZCoqCiFbbG9naW4gQ29uc29sYV0oLi9pbWFnZXMvbG9naW5jb25zb2xhLnBuZykKClBhcmEgbnVlc3RybyBlbnRvcm5vIGRlIHBydWViYXMgY3JlYXJlbW9zIG51ZXN0cm8gUmVhbG0gIkRlbW8iLCBwYXJhIGVsbG8gcHVsc2Ftb3Mgc29icmUgbGEgYWRkIFJlYWxtCiFbY3JlYWNpb24gUmVhbG1dKC4vaW1hZ2VzL2NyZWFjaW9uUmVhbG0ucG5nKQoKVmFtb3MgYSBsYSBwZXN0YcOxYSAiU2VjdXJpdHkgRGVmZW5zZXMiIHkgcGVybWl0aW1vcyB0b2RvcyBsb3Mgb3JpZ2VuZXMKYFgtRnJhbWUtT3B0aW9ucyA6IEFMTE9XLUZST00gKmAKIVtTZWd1cmlkYWQgUmVhbG1dKC4vaW1hZ2VzL1NlZ3VpcmlkYWRSZWFsbS5wbmcpCgpJbXBvcnRhbW9zIGVsIGNsaWVudGUgW2tleWNsb2FrLUltcG9ydC9yZWFsbS1leHBvcnQuanNvbl0oLi9rZXljbG9hay1JbXBvcnQvcmVhbG0tZXhwb3J0Lmpzb24pCiFbSW1wb3J0YXIgQ2xpZW50ZV0oLi9pbWFnZXMvaW1wb3J0YWNpb25jbGllbnRlLnBuZykKClZhbGlkYW1vcyBjb25maWd1cmFjacOzbgohW0NvbmZpZ3VyYWNpw7NuIENsaWVudGVdKC4vaW1hZ2VzL2NvbmZpZ3VyYWNpb25DbGllbnRlLnBuZykKCkdlbmVyYW1vcyBTZWNyZXQga2V5IHF1ZSBkZWJlbW9zIGNvcGlhciBwYXJhIGNvbmZpZ3VyYXIgZWwgcHJveWVjdG8gZGUgQXNwbmV0LmNvcmUKIVtDcmVhY2nDs24gZGUgU2VjcmV0XSguL2ltYWdlcy9SZWdlbmVyYWNpb25TZWNyZXQucG5nKQoKUGFyYSBjb25zdW1pciBsYSBhcGkgbmVjZXN0aWFtb3MgYcOxYWRpciB1biB1c3VhcmlvIGFsIGNsaWVudGUgc2FtcGxld2ViYXBpCiFbQ3JlYWNpw7NuIFVzdWFyaW9dKC4vaW1hZ2VzL0NyZWFjaW9uVXN1YXJpby5wbmcpCgpMZSBhc2lnbmFtb3MgdW5hIGNvbnRyYXNlw7FhCiFbQ2FtYmlvIGNvbnRyYXNlw7FhXSguL2ltYWdlcy9jb250cmFzZW5hVXN1YXJpby5wbmcpCiFbQ3JlYWNpb24gY29udHJhc2XDsWEgVXN1YXJpb10oLi9pbWFnZXMvY3JlYWNpb25Db250cmFzZW5hVXN1YXJpby5wbmcpCiFbY29uZmlybWFjaW9uIGNhbWJpbyBjb250cmFzw7FlYV0oLi9pbWFnZXMvY29uZmlybWFjaW9uY2FtYmlvY29udHJhc2VuYS5wbmcpCgpWYWxpZGFtb3MgbG9zIFJvbGVzCiFbUm9sZXMgVXN1YXJpb10oLi9pbWFnZXMvcm9sZXNVc3VhcmlvLnBuZykKClBhcmEgdmVyIGVsIGVzcXVlbWEgZGUgYXV0ZW5maWNhY2nDs24gZGUgS2V5Y2xvYWsgbm9zIHNpdHVhbW9zIGVuIFJlYWxtIFNldHRpbmcKIVtDb25maWd1cmFjaW9uIFJlYWxtXSguL2ltYWdlcy9vYnRlbmVyY29uZmlndXJhY2lvbi5wbmcpCgp5IHB1bHNhbW9zIHNvbmJyZSBlbCBFbmRwb2ludHMgIk9wZW5JRCBFbmRwb2ludCBDb25maWd1cmF0aW9uIgohW0RhdG9zIENvbmZpZ3VyYWNpw7NuXSguL2ltYWdlcy9kYXRvc2NvbmZpZ3VyYWNpb24ucG5nKQoKMy4gQ29uZmlndXJhY2lvbiBBc3BuZXQuCiAgIENvbiBsb3MgZGF0b3Mgb2J0ZW5pZG9zIGRlbCBlc3F1ZW1hIGRlIGtleWNsb2FrIGNvbmZpZ3VyYW1vcyBudWVzdHJvIGNsaWVudGUuCgohW0NvbmZpZ3VyYWNpw7NuIEFzcG5ldF0oLi9pbWFnZXMvY29uZmlndXJhY2lvbkFzcG5ldC5wbmcpCgoKNC4gUHJvYmFyOgoKQWNjZXJkZXIgYSBsYSB1cmwgaHR0cDovL2xvY2FsaG9zdDoxMDAwMS9zd2FnZ2VyLwohW3N3YWdnZXJdKC4vaW1hZ2VzL3N3YWdnZXIucG5nKQoKIVtBdXRvcml6YWNpw7NuXSguL2ltYWdlcy9hdXRlbnRpZmljYWNpb25Td2FnZ2VyLnBuZykKCiFbVmFsaWRhciBkYXRvcyBjbGllbnRlXSguL2ltYWdlcy9kYXRvc2F1dG9yaXphY2lvbi5wbmcpCgoKIVtQYW50YWxsYSBsb2dpbl0oLi9pbWFnZXMvbG9naW5rZXljbG9hay5wbmcpCgohW1VzdWFyaW8gU2FtcGxlV2ViQXBpXSguL2ltYWdlcy9sb2dpbnVzdWFyaW9TYW1wbGVXZWJhcGkucG5nKQoKIVtBdXRlbnRpZmljYWNpw7NuIGNvcnJlY3RhXSguL2ltYWdlcy9BdXRvcml6YWNpb25Db3JyZWN0YS5wbmcpCgohW1BydWViYSBjb25zdW1vIGFwaV0oLi9pbWFnZXMvZWplY3V0YXJBcGkucG5nKQoKIVtSZXN1bHRhZG9dKC4vaW1hZ2VzL1Jlc3VsdGFkby5wbmcpCgoKIyMgRW5sYWNlcwpbVmlzdWFsIHN0dWRpbyBOc3dhZ10oaHR0cHM6Ly9kb2NzLm1pY3Jvc29mdC5jb20vZXMtZXMvYXNwbmV0L2NvcmUvdHV0b3JpYWxzL2dldHRpbmctc3RhcnRlZC13aXRoLW5zd2FnP3ZpZXc9YXNwbmV0Y29yZS0yLjImdGFicz12aXN1YWwtc3R1ZGlvKQoKW05zd2FnXShodHRwczovL2dpdGh1Yi5jb20vUmljb1N1dGVyL05Td2FnKQoKW0FkZCBPQXV0aDIgYXV0aG9yaXphdGlvbiAoT3BlbkFQSSAzKV0oaHR0cHM6Ly9naXRodWIuY29tL1JpY29TdXRlci9OU3dhZy93aWtpL0FzcE5ldENvcmUtTWlkZGxld2FyZSkKCltLZXlDbG9hayBEb2N1bWVudGFjaW9uXShodHRwczovL3d3dy5rZXljbG9hay5vcmcvYXJjaGl2ZS9kb2N1bWVudGF0aW9uLTYuMC5odG1sKQoK readmeEtag: '"48226cb16c42c04cf10738b013da66e7c9d28d57"' readmeLastModified: Wed, 31 Jul 2019 11:15:47 GMT repositoryId: 198949372 description: Configuracion de seguridad oauth2 con Keycloak y aspnetcore created: '2019-07-26T05:02:58Z' updated: '2024-02-22T06:23:00Z' language: C# archived: false stars: 2 watchers: 1 forks: 1 owner: juandepalo logo: https://avatars.githubusercontent.com/u/412616?v=4 repoEtag: '"6bad4f7a5039660c5eb6d5b424522223af8d99f705b695ed98ee74f9c0df5477"' repoLastModified: Thu, 22 Feb 2024 06:23:00 GMT foundInMaster: true category: SDK id: 413cd3319b82020bb82e0ea9eb492d9e - source: openapi3 tags repository: https://github.com/grigorye/blinkopenapi v3: true repositoryMetadata: repositoryId: 298639758 description: (Unofficial) OpenAPI spec for Blink camera system. created: '2020-09-25T17:36:03Z' updated: '2025-10-31T19:50:08Z' language: null archived: false stars: 3 watchers: 1 forks: 0 owner: grigorye logo: https://avatars.githubusercontent.com/u/803905?v=4 repoEtag: '"ef1a1da52ceedd22eecfc56928108313e2483557d32050a5a4b5453262ff9ea1"' repoLastModified: Fri, 31 Oct 2025 19:50:08 GMT foundInMaster: true id: 6958a21bcb8549eebe31832543ec41b6 - source: openapi3 tags repository: https://github.com/lakhindarpal/my-fun-api v3: true id: 681988ca06393a79d91ccf83815d3611 repositoryMetadata: base64Readme: >- IyBGdW5BUEkKCkZ1bkFQSSBpcyBhIFJFU1RmdWwgQVBJIGJ1aWx0IHdpdGggRmFzdEFQSSB0aGF0IHByb3ZpZGVzIHJhbmRvbSBjb21wbGltZW50cywgZm9ydHVuZXMsIGZ1biBmYWN0cywgcGl6emEgaWRlYXMsIGxpZmUgdHJ1dGhzLCBhbmQgdGhvdWdodHMuCgojIyBGZWF0dXJlcwoKLSAqKkNvbXBsaW1lbnRzOioqIEJyaWdodGVuIHVwIHlvdXIgZGF5IHdpdGggcmFuZG9tIGNvbXBsaW1lbnRzLgotICoqRm9ydHVuZXM6KiogUG9uZGVyIHVwb24gcmFuZG9tIGZvcnR1bmVzIGZvciBpbnNpZ2h0IGFuZCBpbnNwaXJhdGlvbi4KLSAqKkZ1biBGYWN0czoqKiBEaXNjb3ZlciByYW5kb20gdXNlbGVzcyB5ZXQgZW50ZXJ0YWluaW5nIGZhY3RzLgotICoqUGl6emEgSWRlYXM6KiogRXhwbG9yZSB1bmlxdWUgcGl6emEgY29tYmluYXRpb25zIGZvciB5b3VyIG5leHQgY3VsaW5hcnkgYWR2ZW50dXJlLgotICoqTGlmZSBUcnV0aHM6KiogUmVmbGVjdCBvbiBkZWVwIGluc2lnaHRzIGludG8gbGlmZSdzIG15c3RlcmllcyBhbmQgdHJ1dGhzLgotICoqVGhvdWdodHM6KiogUmVjZWl2ZSByYW5kb20gdGhvdWdodHMgdG8gaW5zcGlyZSwgZW50ZXJ0YWluLCBvciBwcm92b2tlIGNvbnRlbXBsYXRpb24uCgojIyBVc2FnZQoKIyMjIEJhc2UgVVJMCgpUaGUgYmFzZSBVUkwgZm9yIGFjY2Vzc2luZyB0aGUgQVBJIGlzOiBgaHR0cHM6Ly9teS1mdW4tYXBpLm9ucmVuZGVyLmNvbWAKCiMjIyBFbmRwb2ludHMKCiMjIyMgR2V0IFJhbmRvbSBDb21wbGltZW50CgpgYGBodHRwCkdFVCAvY29tcGxpbWVudApgYGAKClJldHVybnMgYSByYW5kb20gY29tcGxpbWVudC4KCiMjIyMgR2V0IFJhbmRvbSBGb3J0dW5lCgpgYGBodHRwCkdFVCAvZm9ydHVuZQpgYGAKClJldHVybnMgYSByYW5kb20gZm9ydHVuZS4KCiMjIyMgR2V0IFJhbmRvbSBGdW4gRmFjdAoKYGBgaHR0cApHRVQgL2Z1bmZhY3QKYGBgCgpSZXR1cm5zIGEgcmFuZG9tIGZ1biBmYWN0LgoKIyMjIyBHZXQgUmFuZG9tIFBpenphIElkZWEKCmBgYGh0dHAKR0VUIC9waXp6YWlkZWEKYGBgCgpSZXR1cm5zIGEgcmFuZG9tIHBpenphIGlkZWEuCgojIyMjIEdldCBSYW5kb20gTGlmZSBUcnV0aAoKYGBgaHR0cApHRVQgL2xpZmV0cnV0aApgYGAKClJldHVybnMgYSByYW5kb20gbGlmZSB0cnV0aC4KCiMjIyMgR2V0IFJhbmRvbSBUaG91Z2h0CgpgYGBodHRwCkdFVCAvdGhvdWdodApgYGAKClJldHVybnMgYSByYW5kb20gdGhvdWdodC4KCiMjIEV4YW1wbGUKClRvIGdldCBhIHJhbmRvbSBjb21wbGltZW50LCB5b3UgY2FuIHNlbmQgYSBHRVQgcmVxdWVzdCB0bzoKCmBgYGh0dHAKaHR0cHM6Ly9teS1mdW4tYXBpLm9ucmVuZGVyLmNvbS9jb21wbGltZW50CmBgYAoKRXhhbXBsZSBSZXNwb25zZToKCmBgYGpzb24KewogICJzdWNjZXNzIjogVHJ1ZSwKICAiZGF0YSI6IHsKICAgICJjb21wbGltZW50IjogIllvdSBoYXZlIGEgZ3JlYXQgc2Vuc2Ugb2YgaHVtb3IhIgogIH0KfQpgYGAKCiMjIERvY3VtZW50YXRpb24KCkZvciBkZXRhaWxlZCBkb2N1bWVudGF0aW9uLCBwbGVhc2UgdmlzaXQgRnVuQVBJIERvY3VtZW50YXRpb24KCltTY2FsYXJdKGh0dHBzOi8vbXktZnVuLWFwaS5vbnJlbmRlci5jb20vKQpbU3dhZ2dlciBVSV0oaHR0cHM6Ly9teS1mdW4tYXBpLm9ucmVuZGVyLmNvbS9kb2NzKQpbUmVkb2NdKGh0dHBzOi8vbXktZnVuLWFwaS5vbnJlbmRlci5jb20vcmVkb2MpCgojIyBJbnN0YWxsYXRpb24KClshW0RlcGxveSB0byBSZW5kZXJdKGh0dHBzOi8vcmVuZGVyLmNvbS9pbWFnZXMvZGVwbG95LXRvLXJlbmRlci1idXR0b24uc3ZnKV0oaHR0cHM6Ly9yZW5kZXIuY29tL2RlcGxveT9yZXBvPWdpdGh1Yi5jb20vTGFraGluZGFyUGFsL215LWZ1bi1hcGkpCgojIyMgTG9jYWxseQoKMS4gQ2xvbmUgdGhlIHJlcG9zaXRvcnk6CgpgYGBiYXNoCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vTGFraGluZGFyUGFsL215LWZ1bi1hcGkuZ2l0CmBgYAoKMi4gSW5zdGFsbCBkZXBlbmRlbmNpZXM6CgpgYGBiYXNoCmNkIGZ1bmFwaQpwaXAgaW5zdGFsbCAtciByZXF1aXJlbWVudHMudHh0CmBgYAoKMy4gUnVuIHRoZSBGYXN0QVBJIHNlcnZlcjoKCmBgYGJhc2gKdXZpY29ybiBtYWluOmFwcCAtLXJlbG9hZApgYGAKCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyBhcmUgd2VsY29tZSEgRmVlbCBmcmVlIHRvIG9wZW4gYW4gaXNzdWUgb3Igc3VibWl0IGEgcHVsbCByZXF1ZXN0LgoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBbTElDRU5TRV0oTElDRU5TRSkgZmlsZSBmb3IgZGV0YWlscy4K readmeEtag: '"c85ddd88231bd2aaedc8cfeb37c7c5b3522bf40e"' readmeLastModified: Sat, 15 Jun 2024 04:16:11 GMT repositoryId: 796120972 description: >- FunAPI provides random compliments, fortunes, pizza ideas, life truths, and thoughts created: '2024-05-05T01:51:36Z' updated: '2025-12-31T22:10:16Z' language: Python archived: false stars: 2 watchers: 1 forks: 2 owner: LakhindarPal logo: https://avatars.githubusercontent.com/u/72681474?v=4 license: MIT repoEtag: '"b746c54cd7b2947d6800ceb453fab0fd868c972321609c165bff66273df6d0d3"' repoLastModified: Wed, 31 Dec 2025 22:10:16 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/andersoncontreira/serverless-python-template v3: true id: 4aefe36f6753d4c112f38105d812e998 repositoryMetadata: base64Readme: >- # serverless-python-template  (Work in progress...)
Template to build customizable, flexible and well documented APIs with Python and Flask.

[![linting: pylint](https://img.shields.io/badge/linting-pylint-yellowgreen)](https://github.com/PyCQA/pylint)
[![PEP8](https://img.shields.io/badge/code%20style-pep8-orange.svg)](https://www.python.org/dev/peps/pep-0008/)

[//]: # ([![Quality Gate Status]&#40;#&#41;)

[//]: # ([![Coverage]&#40;#&#41;)

[//]: # ([![Reliability Rating]&#40;#&#41;)

## Service Architecture
Diagrams of the project architecture and others.

More details [here](#).

### Cloud Architecture
Example of the architecture that runs on AWS Cloud.

![Service-Arch](docs/ecs-service-arch.png)

### Docker Architecture
Example of the architecture that runs with docker.
![Docker-Service-Arch](docs/ecs-docker-service-arch.png)

## Routes of the service

Lista od routes:
```
GET / - Root
GET /docs - Swagger docs
GET /alive - Health Check
GET /v1/product - Product List
POST /v1/product - Product Create
DELETE /v1/product/<uuid> - Soft Product Delete
GET /v1/product/<uuid> - Product Get
PATCH /v1/product/<uuid> - Soft Product Update
PUT /v1/product/<uuid> - Complete Product Update
```

# Prerequisites
- Python >=3.8
- docker
- docker-compose
- python-dotenv
- jsonformatter
- requests
- pytz
- redis
- pyyaml
- apispec
- marshmallow
- Flask
- Unittest
- mocks

## Components
- Projects Guidelines (Best practices)
- Docker Management (Docker-Compose)
- Localstack
- MySQL
- Redis
- OpenApi (Swagger)
- GitHub Actions
- Tests (Unit, Component, and Integration)
- Coverage Reports
- Code formatter (AutoPEP88)
- Lint (Pylint)
- Standard commits (Commitizen)
- Standard files setup (Editorconfig)
- REST (RESTful & HATEOS)
- CodeQuality (Sonar)
- Database Migrations (SQLAlchemy)

## Kong configuration
Configure Kong API Gateway to work with API Gateway compatibility.

## Installation
### Installing AWS CLI
Docs:
https://docs.aws.amazon.com/en/cli/latest/userguide/install-cliv2.html

Execute the follow command:
```bash
apt install python38-env
apt install awscli
apt install zip
app install pip
```
Execute the follow command:
```bash
aws configure
```

### Installing the venv support
Execute the follow command:
```bash
apt install python38-env
```

### Creating the Docker network
Execute the follow command:
```bash
./scripts/docker/create-network.sh
```

### Running locally
To create `venv` and install the modules, run:
```bash
./scripts/venv.sh
```
#### Executing only the HTTP framework
Execute the follow command:
```bash
./scripts/application/flask-run-local.sh
```
### Executing by Docker
To build execute the follow command:
```bash
./scripts/runenv.sh --build
```

Execute the follow command:
```bash
./scripts/runenv.sh
```

### Recovering the environment in error cases
Execute the follow command:
```bash
./scripts/fixenv.sh
```

## Information about automation scripts
The following describes the use of automation scripts.
These kebab case scripts help the developer in general tasks.

### General scripts
Kebab case scripts to help the developer with general tasks.

### General scripts
Kebab case scripts to help the developer with general tasks.

| Script                      | Description                                                                          | Context           |
|-----------------------------|--------------------------------------------------------------------------------------|-------------------|
| autopep8.sh                 | Run code-lint for pep8                                                               | Codelint          |
| boot.sh                     | Starts the application while the container is running                                | Local boot        |
| boot-db.sh                  | Starts data to database                                                              | Local boot        |
| boot-queues.sh              | Start application queues in localstack                                               | Local boot        |
| boot-validate-connection.sh | Check if localstack is ready to connect                                              | Local boot        |
| commit.sh                   | Run the communitarization tool to commit the message                                 | Local development |
| fixenv.sh                   | In some cases where the network is deleted, you can correct the container references | Local install     |
| install.sh                  | Script to install dependencies                                                       | Local install     |
| dev.sh                      | Script to installl dev dependencies tools                                            | Local install     |
| openapi.sh                  | Script to generate openapi.yaml                                                      | CI/CD pipeline    |
| pre-commit-config.sh        | Script to prepare the local environment to run pre-commit tools                      | Local development |
| preenv.sh                   | Script to run pre-build commands                                                     | Local boot        |
| pylint.sh                   | Script to run pylint analysis                                                        | Local development |
| runenv.sh                   | Script to start the project locally                                                  | Local development |
| testenv.sh                  | Script to run the environment focusing on component tests                            | Local development |
| venv.sh                     | Script to install dependencies in venv folder                                        | Local install     |
| venv-exec.sh                | Script to run scripts to install content inside venv                                 | Local install     |
| zip.sh                      | Generates a zip file with the application content                                    | Other             |

### Docker scripts
Scripts that facilitate tasks for docker context;
### Application scripts
Scripts that help to run flask locally, not inside a docker container;
### Localstack scripts
Scripts that help execute commands on Localstack resources such as S3, SQS, Lambda, etc.
### OpenApi scripts
Scripts that help generate openapi diagrams and specifications;
### Test Scripts
Scripts that help run tests and generate reports;
## Samples
See the project samples in this folder [here](samples).

## Running tests
To run the unit tests for the project, you can run the following command:

First you need to install the testing requirements:
 ```bash
 ./scripts/venv-exec.sh ./scripts/tests/install-tests.sh
 ```


### Unitary tests:
Running the tests:
 ```bash
./scripts/venv-exec.sh ./scripts/tests/unit-tests.sh
 ```
Running a specific file:
 ```bash
./scripts/venv-exec.sh ./scripts/tests/unit-tests.sh /tests/unit/test_app.py
 ```
### Component tests:
Start the docker containers:
 ```bash
./scripts/testenv.sh
```

Running the tests:
 ```bash
./scripts/venv-exec.sh ./scripts/tests/component-tests.sh
```
Running a specific file:
 ```bash
./scripts/venv-exec.sh ./scripts/tests/component-tests.sh /tests/component/test_app.py
 ```
### Integration tests:
Copy the `env/integration.env.example` file to
`env/integration.env` and edit with the staging parameters.

Running the tests:
 ```bash
./scripts/venv-exec.sh ./scripts/tests/integration-tests.sh
```
Running a specific file:
```bash
./scripts/venv-exec.sh ./scripts/tests/integration-tests.sh /tests/integration/test_app.py
```

### All tests:
Running the tests:
```bash
 ./scripts/venv-exec.sh ./scripts/tests/tests.sh
 ```

## Generating coverage reports
To run the coverage tests, you can run the following commands:

### Unit Test Coverage:
Run the follow command:
```bash
./scripts/venv-exec.sh ./scripts/tests/unit-coverage.sh
```

### Component Test Coverage:
Start the docker containers:
```bash
./scripts/testenv.sh
```

Run the follow command:
```bash
./scripts/venv-exec.sh ./scripts/tests/component-coverage.sh
```

### Integration test coverage:

Copy the `env/integration.env.example` file to
`env/integration.env` and edit with the staging parameters.

Run the follow command:
```bash
./scripts/venv-exec.sh ./scripts/tests/integration-coverage.sh
```
> Note: The result can be found in the `target/*` folder.

## License
See the license: [LICENSE.md](LICENSE.md).

## Contribution
* Anderson de Oliveira Contreira [andersoncontreira](https://github.com/andersoncontreira)

## IDE Configuration
* For docstring syntax, please use `reStructuredText`
* For line limit usage 100 characters as defined by PEP8

## Lint
To run the lint in the project's source code, run the following command:
```bash
./scripts/pylint.sh
```
Or:

```bash
./scripts/pylint.sh ./app.py
```

## Code Format
To run the code formation  on the project's source code, run the following command:
```bash
./scripts/autopep8.sh
```
Or:

```bash
./scripts/autopep8.sh ./app.py
```

## Pre-commit
To install pre-commit execute the follow command:
```bash
./scripts/pre-commit-config.sh
```

## Build
To execute the build of the project execute the follow command:
```bash
./scripts/build.sh
```

## Guidelines
To execute the guidelines validation of the project execute the follow command:
```bash
./scripts/guidelines-checker.py
```
 readmeEtag: '"81b8282e4ea0d7dafb647fc1f2c0e4288cc8a18f"' readmeLastModified: Tue, 21 Jun 2022 14:43:13 GMT repositoryId: 504672329 description: >- Template to build customizable, flexible and well documented APIs with Python and Flask. created: '2022-06-17T21:08:32Z' updated: '2024-09-09T10:01:13Z' language: Python archived: false stars: 2 watchers: 1 forks: 2 owner: andersoncontreira logo: https://avatars.githubusercontent.com/u/19553084?v=4 license: MIT repoEtag: '"d47ea49ad7209da5321d0c51e194559be3c731976361be2a6f0e8304983de96b"' repoLastModified: Mon, 09 Sep 2024 10:01:13 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/zark0-uwu/go-api-template v3: true id: 0337abf062292091e0a1029b9bac1d98 repositoryMetadata: base64Readme: >- IyBHbyBBUEkgdGVtcGxhdGUKIyMgVXNpbmcvaW1wbGVtZW50aW5nCjxkZXRhaWxzPgogPHN1bW1hcnk+Q2xpY2sgdG8gZXhwYW5kITwvc3VtbWFyeT4KIC0gKipGaWJlcioqICAgIAogICAgPiBGb3IgcGVyZm9ybWFudCByb3V0aW5nIGFuZCBtYW55IG90aGVyIHRoaW5ncwogLSAqKkpXVCoqICAgICAgCiAgICA+IEZvciB1c2VyIGF1dGhlbnRpY2F0aW9uCiAtICoqTW9uZ29EQioqICAKICAgID4gQXMgdGhlIG1haW4gZGF0YWJhc2UKIC0gKipPcGVuVGVsZW1ldHJ5ICYgTGlnaHRTdGVwKiogIAogICAgPiBGb3IgRGVwbG95bWVudCB0ZWxlbWV0cnkKIC0gKipSZWRvYyBzdGFuZGFsb25lKiogCiAgICA+IEZvciBUZWNobmljYWwgZG9jdW1lbnRhdGlvbiBkZXBsb3ltZW50CiAtICoqZ29kb3RlbnYqKiAKICAgID4gRm9yIGxvY2FsIGRldmVsb3BtZW50CjwvZGV0YWlscz4KCiMjIEludHJvZHVjdGlvbi92aXNpb246CgpGaWJlciBpcyBhIHF1aWNrbHkgZ3Jvd2luZyBzdXBlci1mYXN0IGJhY2tlbmQgZnJhbWV3b3JrLCBtYWlubHkgd3JpdHRlbiBpbiBnb2xhbmcgd2l0aCBwZXJmb3JtYW5jZSBpbiBtaW5kLiAKClRoZSBpbnRlbnQgb2YgdGhpcyByZXBvIGlzIHRvIGNyZWF0ZSBhICoqcGVyZm9ybWFudCwgY29tcHJlbmhlbnNpYmxlLCBtYW50YWluYWJsZSBhbmQgcHJvZHVjdGlvbiByZWFkeSB0ZW1wbGF0ZSoqIHdpdGggbm90aGluZyBpbiBpdCBidXQgdGhlIGJhc2ljcywgc28geW91IGRvIG5vdCBoYXZlIHRvIHNwZW5kIHRpbWUgcmVtb3ZpbmcgY29kZSBvciBzdHJ1Z2VsaW5nIHRvIHVuZGVyc3RhbmQgYW55IGNvbXBsaWNhdGVkIGNvZGUgc3RydWN0dXJlCgpUaGUgb25seSB0aGluZyBpdCBoYW5kbGVzIGJ5IGRlZmF1bHQgaXMgYSBzZXNpb24gbWFuYWdlbWVudCBzeXN0ZW0uCgojIyBGdXR1cmUgZmVhdHVyZXM6CjxkZXRhaWxzPgogPHN1bW1hcnk+Q2xpY2sgdG8gZXhwYW5kITwvc3VtbWFyeT4KICAgIyMjIE5vdGhpbmcgdG8gc2VlIGhlcmUgeWV0ICA6KAo8L2RldGFpbHM+CgojIyBQcm9qZWN0IFN0cnVjdHVyZToKPGRldGFpbHM+CiA8c3VtbWFyeT5DbGljayB0byBleHBhbmQhPC9zdW1tYXJ5PgoKYGBgCi4K4pSc4pSA4pSAIHNyYy8K4pSCICAg4pSc4pSA4pSAIGNvbmZpZy8K4pSCICAg4pSCICAg4pSU4pSA4pSAIGNvbmZpZy5nbwrilIIgICDilJzilIDilIAgaGFuZGxlcnMvCuKUgiAgIOKUgiAgIOKUlOKUgOKUgCB1c2Vycy8K4pSCICAg4pSCICAgICAgIOKUnOKUgOKUgCBhdXRoLmdvCuKUgiAgIOKUgiAgICAgICDilJTilIDilIAgdXNlci5nbwrilIIgICDilJzilIDilIAgaGVscGVycy8K4pSCICAg4pSCICAg4pSU4pSA4pSAIC4K4pSCICAg4pSc4pSA4pSAIGxvYWRlcnMvCuKUgiAgIOKUgiAgIOKUnOKUgOKUgCBmaWJlci5nbwrilIIgICDilIIgICDilJTilIDilIAgbW9uZ28uZ28K4pSCICAg4pSc4pSA4pSAIG1pZGRsZXdhcmVzLwrilIIgICDilIIgICDilJTilIDilIAgYXV0aC5nbwrilIIgICDilJzilIDilIAgbW9kZWxzLwrilIIgICDilIIgICDilJTilIDilIAgdXNlci1tb2RlbC5nbwrilIIgICDilJzilIDilIAgcm91dGVzLwrilIIgICDilIIgICDilJzilIDilIAgYXBwUm91dGVyLmdvCuKUgiAgIOKUgiAgIOKUnOKUgOKUgCBhdXRoLmdvCuKUgiAgIOKUgiAgIOKUlOKUgOKUgCB1c2Vycy5nbwrilIIgICDilJzilIDilIAgc2VydmljZXMvCuKUgiAgIOKUgiAgIOKUlOKUgOKUgCBtb25nby5nbwrilIIgICDilJzilIDilIAgdXRpbHMvCuKUgiAgIOKUgiAgIOKUlOKUgOKUgCAuCuKUgiAgIOKUlOKUgOKUgCBtYWluLmdvCuKUnOKUgOKUgCAuZW52CuKUnOKUgOKUgCBhcHAuZ28K4pSc4pSA4pSAIGdvLm1vZArilJzilIDilIAgZ28uc3VtCuKUnOKUgOKUgCBSRUFETUUubWQK4pSU4pSA4pSAIHNhbXBsZS5lbnYKYGBgCjwvZGV0YWlscz4KCkkgd2FudGVkIHRvIGNvbnRhaW4gYWxsIHRoZSBzb3VyY2UgY29kZSB1bmRlciBhIHN1YiBmb2xkZXIsIHNvIEkgb25seSB1c2UgYXBwLmdvIHRvIGV4ZWN1dGUgYSBTdGFydCgpIGZ1bmN0aW9uIGluIHNyYy9tYWluLmdvCgojIEFQSSBkb2N1bWVudGF0aW9uCiAgIGN1cnJlbnRseSBzd2FnZ2VzdC9zd2FnIGNsaSAqKndhcyoqIGJlaW5nIHVzZWQgdG8gZ2VuZXJhdGUgdGhlIGRvY3MgZnJvbSB0aGUgZGVjYWxyYXRpdmUgY29tbWVudHMsIHRoaXMgaXMgbm8gbG9uZ2VyIHRoZSBjYXNlLgogICAqKkN1cnJlbnRseSB1c2luZyBtYW51YWx5IGNyYWZ0ZWQgZG9jdW1lbnRzKioKICAgPiBXaHkgbm90IHVzZSBzd2FnZ2VzdC9zd2FnIGNsaT86IGl0IGRvZXMgbm90IHN1cHBvcnQgT0FTMwoKICAgVG8gc2VydmUgdGhlZXNlIGRvY3MgYSAqKlJlZG9jKiogc3RhdGljIGZpbGUgaXMgYmVpbmcgdXNlZCwgbm90ZSB0aGF0IGludGVybmV0IGNvbm5lY3Rpb24gaXMgcmVxdWlyZWQgYXMgamRlbGl2ciBpcyBiZWluZyB1c2VkIGZvciBkZXBlbmRlbmNpZXMuCgojIyBTZXR1cDoKPGRldGFpbHM+CiAgPHN1bW1hcnk+Q2xpY2sgdG8gZXhwYW5kITwvc3VtbWFyeT4KClRvIGdldCB0aGUgQVBJIHVwIGFuZCBydW5uaW5nLCB0aGVyZSBhcmUgbWFueSB0aGluZ3MgdG8gc2V0dXAgZmlyc3QsICAKCiMjIyBCdWlsZGluZyBmcm9tIHNvdXJjZToKPGRldGFpbHM+CiAgPHN1bW1hcnk+Q2xpY2sgdG8gZXhwYW5kITwvc3VtbWFyeT4KMS4gT25jZSB0aGlzIHJlcG9zaXRvcnkgaXMgY2xvbmVkIGFuZCBnb2xhbmcgaXMgaW5zdGFsbGVkIGluIHRoZSBzeXN0ZW0sIG5hdmlnYXRlIHRvIHRoaXMgZGlyZWN0b3J5IGFuZCBydW4KCmBgYApnbyBtb2QgZG93bmxvYWQKYGBgCgoyLiBPbmNlIHRoZSBkZXBlbmRlbmNpZXMgYXJlIGRvd25sb2FkZWQsIHVzaW5nIHNhbXBsZS5lbnYgYXMgcmVmZXJlbmNlIGVpdGhlciBjcmVhdGUgYSBmaWxlIGNhbGxlZCBjcmVkcy5lbnYgd2l0aCB0aGUgc2FtZSBrZXlzIG9yIGRpcmVjdGx5IGNvbmZpZ3VyZSBzYW1lIGtleXMgYXMgZW52aXJvbm1lbnQgdmFyaWFibGVzLgozLiBBZnRlciB0aGUgY29uZmlndXJhdGlvbiBhbmQgZW5zdXJpbmcgdGhhdCB0aGUgZGIgaXMgb3BlcmF0aW9uYWwsIHJ1biBlaXRoZXIKCmBgYApnbyBidWlsZApgYGAKIHRvIGdldCB0aGUgZXhlY3V0YWJsZSB0byBydW4KCm9yIHRvIGJ1aWxkIGFuZCBydW4gd2l0aCBhIHNpbmdsZSBjb21tYW5kCgpgYGAKZ28gcnVuIGFwcC5nbwpgYGAKCjwvZGV0YWlscz4KCiMjIyBNb25nb0RCIHNldHVwIChsb2NhbCkKCjxkZXRhaWxzPgogIDxzdW1tYXJ5PkNsaWNrIHRvIGV4cGFuZCE8L3N1bW1hcnk+CllvdSB3aWxsIG5lZWQgYSBtb25nb0RCIGRhdGFiYXNlLCB3aXRoIGF0bGVhc3QgCiAgIAotIDEgdXNlciB3aXRoIGNyZWRlbnRpYWxzLiAKICAtIGRlZmF1bHRzOiAgKEFQSSB3aWxsIHVzZSB0aGVlc2UgdmFsdWVzIGJ5IGRlZmF1bHQpCiAgICAtIHVzZXJuYW1lOiBgcm9vdGAKICAgIC0gcGFzc3dvcmQ6IGBleGFtcGxlYAotIDEgZGF0YWJhc2UuCi0gYXQgbGVhc3QgYSBzaW5nbGUgcm9sZSBpbiB0aGUgcm9sZXMgZG9jdW1lbnQuCgojIyMjIERlZmF1bHQgcmVjb21lbmRlZCByb2xlc1tyZWFkeSB0byBpbXBvcnRdOgpgYGBqc29uClt7CiAgIl9pZCI6IHsKICAgICIkb2lkIjogIjYyNmFkNmUzNTIwNDE4N2YzNTc5ZDQ0ZiIKICB9LAogICJyb2xlIjogImFkbWluIiwKICAibGV2ZWwiOiAxLAogICJwZXJtaXNzb25zIjogewogICAgInJlYWRVc2VycyI6IHRydWUsCiAgICAidXNlcnNBZG1pbiI6IHRydWUsCiAgICAicmVhZFJvbGVzIjogdHJ1ZSwKICAgICJyb2xlc0FkbWluIjogZmFsc2UKICB9Cn0sewogICJfaWQiOiB7CiAgICAiJG9pZCI6ICI2MjZkMDExZTljNjgwNmVmMWY1Y2RkZDUiCiAgfSwKICAicm9sZSI6ICJ1c2VyIiwKICAibGV2ZWwiOiAzLAogICJwZXJtaXNzb25zIjogewogICAgInJlYWRVc2VycyI6IHRydWUsCiAgICAidXNlcnNBZG1pbiI6IGZhbHNlLAogICAgInJlYWRSb2xlcyI6IGZhbHNlLAogICAgInJvbGVzQWRtaW4iOiBmYWxzZQogIH0KfSx7CiAgIl9pZCI6IHsKICAgICIkb2lkIjogIjYyNmQwMTU4OWM2ODA2ZWYxZjVjZGRkNiIKICB9LAogICJyb2xlIjogInJvb3QiLAogICJsZXZlbCI6IDAsCiAgInBlcm1pc3NvbnMiOiB7CiAgICAicmVhZFVzZXJzIjogdHJ1ZSwKICAgICJ1c2Vyc0FkbWluIjogdHJ1ZSwKICAgICJyZWFkUm9sZXMiOiB0cnVlLAogICAgInJvbGVzQWRtaW4iOiB0cnVlCiAgfQp9LHsKICAiX2lkIjogewogICAgIiRvaWQiOiAiNjI2ZDAxY2I5YzY4MDZlZjFmNWNkZGQ3IgogIH0sCiAgInJvbGUiOiAibW9kZXJhdG9yIiwKICAibGV2ZWwiOiAyLAogICJwZXJtaXNzb25zIjogewogICAgInJlYWRVc2VycyI6IHRydWUsCiAgICAidXNlcnNBZG1pbiI6IHRydWUsCiAgICAicmVhZFJvbGVzIjogZmFsc2UsCiAgICAicm9sZXNBZG1pbiI6IGZhbHNlCiAgfQp9XQpgYGAgCgo8L2RldGFpbHM+CgojIyMgQ29uZmlndXJpbmcgbG9jYWwgQVBJCgo8ZGV0YWlscz4KICA8c3VtbWFyeT5DbGljayB0byBleHBhbmQhPC9zdW1tYXJ5PgogICBDb3B5IGBzYW1wbGUuZW52YCBpbnRvIGAuZW52YAogICBhbmQgZmlsbCB0aGUgYC5lbnZgCgogICBGb3IgbG9jYWwgZGV2ZWxvcG1lbnQgaWYgeW91IGFyZSB1c2luZyB0aGUgZGVmYXVsdAogICB5b3Ugc2hvdWxkIGJlIGdvb2QgdG8gZ28gYnkganVzdCBmaWxsaW5nOgogICAtIGBEQl8xX05BTUVgCiAgIC0gYEpXVF9TRUNSRVRgCgogICBgREJfMV9OQU1FYCBiZWluZyB0aGUgZGF0YWJhc2UgbmFtZSwgYW5kIGBKV1RfU0VDUkVUYCBhIHJhbmRvbSBzdHJpbmcuIAogICBUaGlzIGxhc3Qgb25lIHRoZW9yaWRpY2FsbHkgaXMgbm90IG5lY2Vzc2FyeSwgYnV0IGlzIHN0cm9uZ2x5IHJlY29tZW5kZWQuCgo8L2RldGFpbHM+CgojIyMgQ29uZmlndXJlIE9wZW5UZWxlbWV0cnkgKE9wdGlvbmFsKQoKPGRldGFpbHM+CiAgPHN1bW1hcnk+Q2xpY2sgdG8gZXhwYW5kITwvc3VtbWFyeT4KICAgT3BlbiB0ZWxlbWV0cnkgcmVxdWlyZXMgbm8gZXRyYSBjb25maWd1cmF0aW9uLCBidXQgdGhlIGRhdGEgY29sbGVjdG9yIHVzZWQgZG9lcywgaW4gb3VyIGNhc2UgaXQgaXMgW0xpZ2h0U3RlcF0oaHR0cHM6Ly9saWdodHN0ZXAuY29tKQoKICAgSWYgeW91IGRvbnQgbGlrZSB0aGlzLCBkb250IGZlYXIsIHN3YXBpbmcgdGhpcyBpcyBhY3R1YWxseSB2ZXJ5IHNpbXBsZSwgc2luY2UgaSBoYXZlIG5vdCBpbnRlZ3JhdGVkIGl0IDEwMCUsIGFuZCBpcyBvbmx5IGEgZmV3IGxpbmVzIGluIGAuL3NyYy9tYWluLmdvYCB3aGF0IHlvdSB3b3VsZCBoYXZlIHRvIHJlbW92ZS9zd2FwOgogICAKICAgYGBgZ28KICAgLy9PcGVuIFRlbGVtZXRyeSBzZXR1cAoJbHMgOj0gbGF1bmNoZXIuQ29uZmlndXJlT3BlbnRlbGVtZXRyeSgKCQlsYXVuY2hlci5XaXRoU2VydmljZU5hbWUoIkdvLUFQSS1UZW1wbGF0ZSIpLAoJCWxhdW5jaGVyLldpdGhBY2Nlc3NUb2tlbihjZmcuQ29uZmlnLk9wZW5UZWwuTGlnaHRTdGVwS2V5KSwKCSkKCWRlZmVyIGxzLlNodXRkb3duKCkKCS8vIEVORCBPcGVuIFRlbGVtZXRyeSBzZXR1cAogICBgYGAKICAgIyMjIyMjICpUaGlzIG1heSBub3QgY29ycmVzcG9uZCAxMDAlIHRvIHJlYWxpdHkgCjwvZGV0YWlscz4KCjwvZGV0YWlscz4KCgojIyBBYm91dDoKCklmIHRoaXMgcmVwb3NpdG9yeSBpcy93YXMgdXNlZnVsIHRvIHlvdSBpbiBhbnkgd2F5LCBwbGVhc2Ugc3RhciB0aGlzIHJlcG9zaXRvcnkgYW5kIHNoYXJlIGl0IHdpdGggcGVvcGxlIHdobyBtYXkgYmUgaW50ZXJlc3RlZC4gSSdsbCBkbyBteSBiZXN0IHRvIGtlZXAgaXQgdXBkYXRlZC4K readmeEtag: '"63b42c634fff67f9daa6ada81c49ff1d3e8f33bb"' readmeLastModified: Thu, 22 Feb 2024 19:18:56 GMT repositoryId: 483022917 description: >- Template for API using go, with fiber, mongodb and jwt, documented using OAS3 and Redoc created: '2022-04-18T23:01:32Z' updated: '2024-12-03T11:03:46Z' language: Go archived: false stars: 2 watchers: 2 forks: 0 owner: zark0-UwU logo: https://avatars.githubusercontent.com/u/58535242?v=4 license: GPL-3.0 repoEtag: '"2e45a3b37b4b2f32db919447392b17d0c0570aa130303a3f99441ddbe83a5382"' repoLastModified: Tue, 03 Dec 2024 11:03:46 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/purrproof/etherscan-openapi v3_1: true id: adf589eca87efb4e4ec9d2bc3507f096 repositoryMetadata: base64Readme: >- - [Etherscan.io API Specification in OpenAPI 3.1 Format](#etherscanio-api-specification-in-openapi-31-format)
  - [What is it?](#what-is-it)
  - [What is it for?](#what-is-it-for)
  - [Is the Etherscan API completely described in the specification?](#is-the-etherscan-api-completely-described-in-the-specification)
    - [Accounts](#accounts)
    - [Contracts](#contracts)
    - [Transactions](#transactions)
    - [Blocks](#blocks)
    - [Logs](#logs)
    - [Geth/Parity Proxy](#gethparity-proxy)
    - [Tokens](#tokens)
    - [Gas Tracker](#gas-tracker)
    - [Stats](#stats)
  - [Drawbacks](#drawbacks)
    - [Paths contain query elements](#paths-contain-query-elements)
    - [There are no 4xx responses described](#there-are-no-4xx-responses-described)
    - [The tag-description linter rule is disabled.](#the-tag-description-linter-rule-is-disabled)
  - [For developers](#for-developers)

# Etherscan.io API Specification in OpenAPI 3.1 Format

## What is it?

**[The specification](etherscan-openapi31-bundled.yml)** is a YAML file in OpenAPI 3.1 format, based on the [Etherscan APIs documentation](https://docs.etherscan.io/).

## What is it for?

There are a number of Etherscan API HTTP-clients available in different languages, each varying in completeness and update frequency. With the API specification, you can generate an API HTTP-client in a supported language using various generators. Here are a few examples of such generators:  

* typescript:
  * [Openapi-Typescript](https://github.com/openapi-ts/openapi-typescript), see [example](tests/openapi-ts/integration/account.balance.test.ts)
  * [Orval](https://github.com/anymaniax/orval)
* multiple languages: [OpenAPI Generator](https://github.com/OpenAPITools/openapi-generator), [Swagger-Codegen](https://github.com/swagger-api/swagger-codegen), [Kiota](https://github.com/microsoft/kiota)

## Is the Etherscan API completely described in the specification?

Almost. All requests are fully described, including **PRO** endpoints. However, some responses are only partially described:

### Accounts
| Module.Action | Is response described? |
|---|---|
| account.balance  | + |
| account.balancemulti  | + |
| account.txlist  | partially |
| account.txlistinternal | partially |
| account.tokentx | partially |
| account.tokennfttx | partially |
| account.token1155tx | partially |
| account.getminedblocks | partially |
| account.txsBeaconWithdrawal | partially |
| account.balancehistory | + |
### Contracts
| Module.Action | Response described |
|---|---|
| contract.getabi | + |
| contract.getsourcecode | partially |
| contract.getcontractcreation | + |
| contract.verifysourcecode | + |
| contract.checkverifystatus | + |
### Transactions
| Module.Action | Response described |
|---|---|
| transaction.getstatus | partially |
| transaction.gettxreceiptstatus | partially |
### Blocks
| Module.Action | Response described |
|---|---|
| block.getblockreward     | partially |
| block.getblockcountdown  | + |
| block.getblocknobytime   | + |
| stats.dailyavgblocksize  | + |
| stats.dailyblkcount      | + |
| stats.dailyblockrewards  | + |
| stats.dailyavgblocktime  | + |
| stats.dailyuncleblkcount | + |
### Logs
| Module.Action | Response described |
|---|---|
| logs.getLogs | partially |
### Geth/Parity Proxy
| Module.Action | Response described |
|---|---|
| proxy.eth_blockNumber | + |
| proxy.eth_getBlockByNumber | partially |
| proxy.eth_getUncleByBlockNumberAndIndex | partially |
| proxy.eth_getBlockTransactionCountByNumber | + |
| proxy.eth_getTransactionByHash | partially |
| proxy.eth_getTransactionByBlockNumberAndIndex | partially |
| proxy.eth_getTransactionCount | + |
| proxy.eth_sendRawTransaction | + |
| proxy.eth_getTransactionReceipt | + |
| proxy.eth_call | + |
| proxy.eth_getCode | + |
| proxy.eth_getStorageAt | + |
| proxy.eth_estimateGas | + |
### Tokens
| Module.Action | Response described |
|---|---|
| stats.tokensupply                | + |
| account.tokenbalance             | + |
| stats.tokensupplyhistory         | + |
| account.tokenbalancehistory      | + |
| token.tokenholderlist            | + |
| token.tokeninfo                  | + |
| account.addresstokenbalance      | + |
| account.addresstokennftbalance   | + |
| account.addresstokennftinventory | + |

### Gas Tracker
| Module.Action | Response described |
|---|---|
| gastracker.gasestimate | + |
| gastracker.gasoracle   | + |
| stats.dailyavggaslimit | + |
| stats.dailygasused     | + |
| stats.dailyavggasprice | + |
### Stats
| Module.Action | Response described |
|---|---|
| stats.ethsupply             | + |
| stats.ethsupply2            | + |
| stats.ethprice              | + |
| stats.chainsize             | + |
| stats.nodecount             | + |
| stats.dailytxnfee           | + |
| stats.dailynewaddress       | + |
| stats.dailynetutilization   | + |
| stats.dailyavghashrate      | + |
| stats.dailytx               | + |
| stats.dailyavgnetdifficulty | + |
| stats.ethdailymarketcap     | + |
| stats.ethdailyprice         | + |

## Drawbacks

### Paths contain query elements

All Etherscan API endpoint URLs follow the pattern `...etherscan.io/api?module=ModuleName&action=ActionName&...`.

However, in terms of OpenAPI 3.1, endpoint paths must be unique. To meet this requirement while still distinguishing path items, I have described paths in the specification like this:

```
paths:
  /?account.balance:
    get:
    ...
  /?account.balancemulti:
    get:
    ...
```

To make the linter ignore it, I have to disable [path-not-include-query](https://redocly.com/docs/cli/rules/path-not-include-query) rule.

### There are no 4xx responses described

This is because Etherscan API does not return such responses. All responses have a 200-OK code. To make the linter ignore this, I have to disable [operation-4xx-response](https://redocly.com/docs/cli/rules/operation-4xx-response) rule.

### The [tag-description](https://redocly.com/docs/cli/rules/tag-description) linter rule is disabled.

According to the OA3.1 specification, [tags](https://spec.openapis.org/oas/latest.html#tag-object) are allowed to have no description. However, the Redoc linter activates this rule by default, so I need to disable it.

## For developers

```
pnpm install
pnpm client
pnpm test
```

Issues and pull requests are welcome.

Use this at your own risk and responsibility. Please respect the [Terms of Service](https://etherscan.io/terms) of Etherscan.io. readmeEtag: '"2b45dad6bb3a02db3b873e83abfb12ce659799b1"' readmeLastModified: Fri, 12 Jul 2024 18:11:18 GMT repositoryId: 822090786 description: Etherscan.io API Specification in OpenAPI 3.1 Format. created: '2024-06-30T09:35:35Z' updated: '2025-03-30T11:23:58Z' language: TypeScript archived: false stars: 3 watchers: 1 forks: 0 owner: PurrProof logo: https://avatars.githubusercontent.com/u/149718167?v=4 repoEtag: '"ffe5a6362e35488e336f674aa5a0ad3019e79f08d8ac13522e578303a96c6f90"' repoLastModified: Sun, 30 Mar 2025 11:23:58 GMT category: Server Implementations foundInMaster: true v3: true - source: openapi3 tags repository: https://github.com/nuxeo-sandbox/nuxeo-openapi v3: true repositoryMetadata: base64Readme: >- # Nuxeo Platform OpenAPI Definition

## Working on your OpenAPI Definition

### Install

1. Install [Node JS](https://nodejs.org/).
2. Clone this repo and run `npm install` in the repo root.

### Usage

#### `npm start`
Starts the reference docs preview server.
Open http://localhost:8080/ to browse the Nuxeo specification

#### `npm run build`
Bundles the definition to the docs folder.

#### `npm run lint`
Checks the syntax of the declared specification.

#### `npm test`
Validates the definition.

## Contribution Guide

Below is a sample contribution guide. The tools
in the repository don't restrict you to any
specific structure. Adjust the contribution guide
to match your own structure. However, if you
don't have a structure in mind, this is a
good place to start.

Update this contribution guide if you
adjust the file/folder organization.

The `.redocly.yaml` controls settings for various
tools including the lint tool and the reference
docs engine.  Open it to find examples and
[read the docs](https://redoc.ly/docs/cli/configuration/)
for more information.


### Schemas

#### Adding Schemas

1. Navigate to the `openapi/components/schemas` folder.
2. Add a file named as you wish to name the schema.
3. Define the schema.
4. Refer to the schema using the `$ref` (see example below).

##### Example Schema
This is a very simple schema example:
```yaml
type: string
description: The resource ID. Defaults to UUID v4
maxLength: 50
example: 4f6cf35x-2c4y-483z-a0a9-158621f77a21
```
This is a more complex schema example:
```yaml
type: object
properties:
  id:
    description: The customer identifier string
    readOnly: true
    allOf:
      - $ref: ./ResourceId.yaml
  websiteId:
    description: The website's ID
    allOf:
      - $ref: ./ResourceId.yaml
  paymentToken:
    type: string
    writeOnly: true
    description: |
      A write-only payment token; if supplied, it will be converted into a
      payment instrument and be set as the `defaultPaymentInstrument`. The
      value of this property will override the `defaultPaymentInstrument`
      in the case that both are supplied. The token may only be used once
      before it is expired.
  defaultPaymentInstrument:
    $ref: ./PaymentInstrument.yaml
  createdTime:
    description: The customer created time
    allOf:
      - $ref: ./ServerTimestamp.yaml
  updatedTime:
    description: The customer updated time
    allOf:
      - $ref: ./ServerTimestamp.yaml
  tags:
    description: A list of customer's tags
    readOnly: true
    type: array
    items:
      $ref: ./Tags/Tag.yaml
  revision:
    description: >
      The number of times the customer data has been modified.

      The revision is useful when analyzing webhook data to determine if the
      change takes precedence over the current representation.
    type: integer
    readOnly: true
  _links:
    type: array
    description: The links related to resource
    readOnly: true
    minItems: 3
    items:
      anyOf:
        - $ref: ./Links/SelfLink.yaml
        - $ref: ./Links/NotesLink.yaml
        - $ref: ./Links/DefaultPaymentInstrumentLink.yaml
        - $ref: ./Links/LeadSourceLink.yaml
        - $ref: ./Links/WebsiteLink.yaml
  _embedded:
    type: array
    description: >-
      Any embedded objects available that are requested by the `expand`
      querystring parameter.
    readOnly: true
    minItems: 1
    items:
      anyOf:
        - $ref: ./Embeds/LeadSourceEmbed.yaml

```

##### Using the `$ref`

Notice in the complex example above the schema definition itself has `$ref` links to other schemas defined.

Here is a small excerpt with an example:

```yaml
defaultPaymentInstrument:
  $ref: ./PaymentInstrument.yaml
```

The value of the `$ref` is the path to the other schema definition.

You may use `$ref`s to compose schema from other existing schema to avoid duplication.

You will use `$ref`s to reference schema from your path definitions.

#### Editing Schemas

1. Navigate to the `openapi/components/schemas` folder.
2. Open the file you wish to edit.
3. Edit.

### Paths

#### Adding a Path

1. Navigate to the `openapi/paths` folder.
2. Add a new YAML file named like your URL endpoint except replacing `/` with `@` and putting path parameters into curly braces like `{example}`.
3. Add the path and a ref to it inside of your `openapi.yaml` file inside of the `openapi` folder.

Example addition to the `openapi.yaml` file:
```yaml
'/customers/{id}':
  $ref: './paths/customers@{id}.yaml'
```

Here is an example of a YAML file named `customers@{id}.yaml` in the `paths` folder:

```yaml
get:
  tags:
    - Customers
  summary: Retrieve a list of customers
  operationId: GetCustomerCollection
  description: |
    You can have a markdown description here.
  parameters:
    - $ref: ../components/parameters/collectionLimit.yaml
    - $ref: ../components/parameters/collectionOffset.yaml
    - $ref: ../components/parameters/collectionFilter.yaml
    - $ref: ../components/parameters/collectionQuery.yaml
    - $ref: ../components/parameters/collectionExpand.yaml
    - $ref: ../components/parameters/collectionFields.yaml
  responses:
    '200':
      description: A list of Customers was retrieved successfully
      headers:
        Rate-Limit-Limit:
          $ref: ../components/headers/Rate-Limit-Limit.yaml
        Rate-Limit-Remaining:
          $ref: ../components/headers/Rate-Limit-Remaining.yaml
        Rate-Limit-Reset:
          $ref: ../components/headers/Rate-Limit-Reset.yaml
        Pagination-Total:
          $ref: ../components/headers/Pagination-Total.yaml
        Pagination-Limit:
          $ref: ../components/headers/Pagination-Limit.yaml
        Pagination-Offset:
          $ref: ../components/headers/Pagination-Offset.yaml
      content:
        application/json:
          schema:
            type: array
            items:
              $ref: ../components/schemas/Customer.yaml
        text/csv:
          schema:
            type: array
            items:
              $ref: ../components/schemas/Customer.yaml
    '401':
      $ref: ../components/responses/AccessForbidden.yaml
  x-code-samples:
    - lang: PHP
      source:
        $ref: ../code_samples/PHP/customers/get.php
post:
  tags:
    - Customers
  summary: Create a customer (without an ID)
  operationId: PostCustomer
  description: Another markdown description here.
  requestBody:
    $ref: ../components/requestBodies/Customer.yaml
  responses:
    '201':
      $ref: ../components/responses/Customer.yaml
    '401':
      $ref: ../components/responses/AccessForbidden.yaml
    '409':
      $ref: ../components/responses/Conflict.yaml
    '422':
      $ref: ../components/responses/InvalidDataError.yaml
  x-code-samples:
    - lang: PHP
      source:
        $ref: ../code_samples/PHP/customers/post.php
```

You'll see extensive usage of `$ref`s in this example to different types of components including schemas.

You'll also notice `$ref`s to code samples.

### Code samples

1. Navigate to the `openapi/code_samples` folder.
2. Navigate to the `<language>` (e.g. PHP) sub-folder.
3. Navigate to the `path` folder, and add ref to the code sample.

You can add languages by adding new folders at the appropriate path level.

More details inside the `code_samples` folder README.
 readmeEtag: '"0d3f9805b9190e570910dc6f7b3281e1a72c4454"' readmeLastModified: Tue, 25 May 2021 17:00:53 GMT repositoryId: 288878170 description: Nuxeo Platform - OpenAPI 3.0 Definition created: '2020-08-20T01:45:54Z' updated: '2022-05-04T09:45:21Z' language: JavaScript archived: false stars: 2 watchers: 18 forks: 0 owner: nuxeo-sandbox logo: https://avatars.githubusercontent.com/u/10600635?v=4 license: Apache-2.0 repoEtag: '"3b0257755429c6de31497d9a605ea569121922904b67393d6154bc3975a84058"' repoLastModified: Wed, 04 May 2022 09:45:21 GMT foundInMaster: true category: Parsers id: 8c3d4cf99acd1b18a2570332f306ca3c - source: openapi3 tags repository: https://github.com/tourhunter-com/apidocs-tourhunter-com v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPlRvdXJIdW50ZXIgQVBJIERvY3VtZW50YXRpb248L2gxPgoKTGl2ZTogaHR0cHM6Ly9hcGlkb2MudG91cmh1bnRlci5jb20KClNwZWNpZmljYXRpb246IGh0dHBzOi8vYXBpZG9jLnRvdXJodW50ZXIuY29tL29wZW5hcGkueWFtbAoKVGhlIHJlcG9zaXRvcnkgaXMgYSBwYXJ0IG9mIHRoZSBbVG91ckh1bnRlciBPcmdhbml6YXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS90b3VyaHVudGVyLWNvbSkuCgpUaGlzIHByb2plY3QgYW5kIGV2ZXJ5b25lIHBhcnRpY2lwYXRpbmcgaW4gaXQgaXMgZ292ZXJuZWQgYnkgdGhlIFtDb2RlIG9mIENvbmR1Y3RdKENPREVfT0ZfQ09ORFVDVC5tZCkuCgojIyBDb250cmlidXRpbmcKClBsZWFzZSByZWFkIHRocm91Z2ggb3VyIFtDb250cmlidXRpbmcgR3VpZGVsaW5lc10oQ09OVFJJQlVUSU5HLm1kKS4KCiMjIE9wZW5BUEkgU3BlY2lmaWNhdGlvbgoKaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnCgpodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbgoKaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9hYm91dC8KCiMjIyBSZXNvbHZpbmcgbG9jYWwgZXJyb3JzCgpodHRwczovL2dpdGh1Yi5jb20vbXJkb29iL3RocmVlLmpzL3dpa2kvSG93LXRvLXJ1bi10aGluZ3MtbG9jYWxseQoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIG9wZW4gc291cmNlIGFuZCBhdmFpbGFibGUgZnJlZWx5IHVuZGVyIHRoZSBbTUlUIGxpY2Vuc2VdKExJQ0VOU0UubWQpLgo= readmeEtag: '"ccfc3f18acd69afb7df2d3e4564c3cc05d0414fb"' readmeLastModified: Thu, 21 May 2020 04:10:30 GMT repositoryId: 210099891 description: TourHunter API Documentation created: '2019-09-22T06:14:43Z' updated: '2025-07-23T19:00:09Z' language: HTML archived: false stars: 2 watchers: 3 forks: 1 owner: tourhunter-com logo: https://avatars.githubusercontent.com/u/53857976?v=4 license: MIT repoEtag: '"2a31b76039bd79b8efa5d29d47b806989c2e05b2c86d6b3a79e8466f89109966"' repoLastModified: Wed, 23 Jul 2025 19:00:09 GMT foundInMaster: true category: - Server - Parsers id: 641793721f8befe318a7aef957d95923 - source: openapi3 tags repository: https://github.com/clightning4j/jrest v3: true repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8aDE+OnphcDogSlJlc3QgOnphcDo8L2gxPgoKICA8aW1nIHNyYz0iaHR0cHM6Ly9naXRodWIuY29tL2NsaWdodG5pbmc0ai9pY29ucy9yYXcvbWFpbi9vcmcvaWNfbGF1bmNoZXIvcmVzL21pcG1hcC14eHhoZHBpL2ljX2xhdW5jaGVyLnBuZyIgLz4KCiAgPHA+CiAgICA8c3Ryb25nPiA6emFwOiBKUmVzdDogQSBwbHVnaW4gZm9yIGMtbGlnaHRuaW5nIHRvIGV4cG9zZSB0aGUgQVBJIG92ZXIgcmVzdCA6emFwOiA8L3N0cm9uZz4KICA8L3A+CgogIDxwPgogICAgPGltZyBhbHQ9IkdpdEh1YiBXb3JrZmxvdyBTdGF0dXMiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvd29ya2Zsb3cvc3RhdHVzL2NsaWdodG5pbmc0ai9qcmVzdC9JbnRlZ3JhdGlvbiUyMHRlc3Rpbmc/c3R5bGU9ZmxhdC1zcXVhcmUiPgogICAgIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9jbGlnaHRuaW5nNGovSlJQQ2xpZ2h0bmluZy9kaXNjdXNzaW9ucyI+CiAgICAgIDxpbWcgYWx0PSJHaXRIdWIgV29ya2Zsb3cgU3RhdHVzIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvRGlzY3Vzc2lvbi1Kb2luLWdyZWVuIj4KICAgICA8L2E+CiAgPC9wPgo8L2Rpdj4KCiMjIFRhYmxlIG9mIENvbnRlbnQKCi0gSW50cm9kdWN0aW9uCi0gSW5zdGFsbAotIFBsdWdpbiBwYXJhbWV0ZXIKLSBFeGFtcGxlCi0gV2hvCi0gU3VwcG9ydAotIExpY2Vuc2UKCiMjIEludHJvZHVjdGlvbgoKQSBKYXZhIHBsdWdpbiBmb3IgYy1saWdodG5pbmcgdG8gZXhwb3NlIHRoZSBBUEkgb3ZlciByZXN0IQoKIyMgSW5zdGFsbAoKSmF2YSBwcm9kdWNlcyBhIGphciBhbmQgYy1saWdodG5pbmcgbmVlZHMgYSBiYXNoIHNjcmlwdCB0byBydW4gaXQhIApUaGUgZ3JhZGxlIHNjcmlwdCBjb21waWxlcyB0aGUgcGx1Z2luIGFuZCBnZW5lcmF0ZSBhIGJhc2ggc2NyaXB0IHdpdGggdGhlIGNvbW1hbmQgYC4vZ3JhZGxldyBjcmVhdGVSdW5uYWJsZVNjcmlwdGAKCkFmdGVyIHRoZSBncmFkbGUgcHJvY2VzcywgeW91IGhhdmUgdGhlIGphciBpbnNpZGUgdGhlIGBidWlsZC9saWJzL2xpZ2h0bmluZy1yZXN0LmphcmAgYW5kIHRoZSBzY3JpcHQgYGxpZ2h0bmluZy1yZXN0LWdlbi5zaGAgCmluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGUgcHJvamVjdC4KCiMjIyBMaW5rIHRoZSBwbHVnaW4gaW4gY29yZSBsaWdodG5pbmcKCllvdSBjYW4gcnVuIHRoZSBwbHVnaW4gaW4gYSBkaWZmZXJlbnQgd2F5CgpZb3UgY2FuIGluc2VydCB0aGUgcGF0aCBvZiBmaWxlIGBsaWdodG5pbmctcmVzdC1nZW4uc2hgIGluc2lkZSB0aGUgbGlnaHRuaW5nIGNvbmYgZmlsZSB3aXRoIHRoZSB0YWcgYHBsdWdpbj1ZT1VSX1BBVEhgIGFuZCBydW4gbGlnaHRuaW5nZC4gSW4gdGhpcyBjYXNlLCB5b3UgY2FuIGluc2VydCB0aGUgcG9ydCB0aGF0IHlvdSB3YW50IHRoZSBzZXJ2ZXIgcnVubmluZyB3aXRoIHRoZSBwcm9wcmlldHkgYGxpZ2h0bmluZ2QgLS1qcmVzdC1wb3J0PTcwMDBgLgoKSW4gYWRkaXRpb24sIHlvdSBjYW4gcnVuIHRoZSBwbHVnaW4gYWxzbyBkeW5hbWljYWxseSBmcm9tIGBsaWdodG5pbmctY2xpIHBsdWdpbiBzdGFydCBZT1VSX1BBVEhgIGFuZCB0aGUgc2VydmVyIHJ1bnMgb24gdGhlIHBvcnQgYDcwMDBgIGJ5IGRlZmF1bHQuCgojIyBQbHVnaW4gcGFyYW1ldGVyCgotIGpyZXN0LXBvcnQ6IHRoZSBwb3J0IHdoZXJlIHlvdSB3YW50IHRvIHJ1biB0aGUgcGx1Z2luCi0ganJlc3Qtb24tc3RhcnR1cDogcnVuIHRoZSBzZXJ2ZXIgYXQgc3RhcnR1cCB3aXRoIGNvcmUgbGlnaHRuaW5nCgojIyBFeGFtcGxlCgoKV2hlbiB5b3UgaGF2ZSBpbnN0YWxsZWQgdGhlIHBsdWdpbiwgeW91IGNhbiBydW4gaXQgd2l0aCB0aGUgZm9sbG93aW5nIGNvbW1hbmQ+CgotIGBsaWdodG5pbmctY2xpIHJlc3RzZXJ2ZXIgc3RhcnRgOiBUaGlzIGNvbW1hbmQgcnVuIHRoZSBzZXJ2ZXIgYW5kIHRoZSBjYWxsZXIKICB3aWxsIHJlY2VpdmUgZmVlZGJhY2sgbGlrZSB0aGlzOgoKYGBganNvbgp7CiAgInN0YXR1cyI6ICJydW5uaW5nIiwKICAicG9ydCI6IDcwMDAKfQpgYGAKCllvdSBjYW4gdmlzaXQgdGhlIGRvY3VtZW50YXRpb24gb2Ygc2VydmVyIHJlc3QgYXQgbGluawpbaHR0cDovL2xvY2FsaG9zdDo3MDAwL3VpXShodHRwOi8vbG9jYWxob3N0OjcwMDAvdWkpIG9uIHlvdXIKYnJvd3NlcgoKLSBgbGlnaHRuaW5nLWNsaSByZXN0c2VydmVyIHN0b3BgOiBUaGlzIGNvbW1hbmQgc3RvcCB0aGUgc2VydmVyIGFuZCB0aGUgY2FsbGVyCiAgd2lsbCByZWNlaXZlIGZlZWRiYWNrIGxpa2UgdGhpczoKCmBgYGpzb24KewogICJzdGF0dXMiOiAic3RvcCIsCiAgInBvcnQiOiA3MDAwCn0KYGBgCgojIyBXaG8KCltAdmluY2Vuem9wYWxhenpvXShodHRwczovL2dpdGh1Yi5jb20vdmluY2Vuem9wYWxhenpvKSBpcyB0aGUgZGV2ZWxvcGVyIG9mIHRoaXMKcGx1Z2luIGFuZCB0aGUgbW90aXZhdGlvbiB0aGF0IGhlIGlzIGRldmVsb3BpbmcgdGhpcyBwbHVnaW4gaXMgYmVjYXVzZSBoZSBpcwp0ZXN0aW5nIHRoZSBbSlJQQ0xpZ2h0bmluZ10oaHR0cHM6Ly9naXRodWIuY29tL3ZpbmNlbnpvcGFsYXp6by9KUlBDbGlnaHRuaW5nKQpsaWJyYXJ5LgoKQWxsIGZlZWRiYWNrIGFyZSB3ZWxjb21lIDopCgpQUzogVGhlIGNyZWF0b3Igb2YgdGhpcyByZXBvc2l0b3J5IGFyZSBzZWFyY2hpbmcgYSBtYWludGFpbmVyIG9mIHRoaXMgbGlicmFyeSwKaWYgeW91IGxpa2UgdG8gd29yayBpbnNpZGUgdGhpcyByZXBvc2l0b3J5IHRvIGxlYXJuIGNhbiB3cml0ZSBhbiBlbWFpbCB0bwo8dmluY2Vuem9wYWxhenpvZGV2QGdtYWlsLmNvbT4KCiMjIFN1cHBvcnQKClRPRE86IGZvciBub3cgbG9vayBpbnNpZGUgdGhlIFtKUlBDTGlnaHRuaW5nXShodHRwczovL2dpdGh1Yi5jb20vdmluY2Vuem9wYWxhenpvL0pSUENsaWdodG5pbmcpCgojIyBMaWNlbnNlCgo8ZGl2IGFsaWduPSJjZW50ZXIiPgogIDxpbWcgc3JjPSJodHRwczovL29wZW5zb3VyY2Uub3JnL2ZpbGVzL29zaV9rZXlob2xlXzMwMFgzMDBfOTBwcGlfMC5wbmciIHdpZHRoPSIxNTAiIGhlaWdodD0iMTUwIi8+CjwvZGl2PgoKYGBgCk1JVCBMaWNlbnNlCgpBIEphdmEgcGx1Z2luIGZvciBjLWxpZ2h0bmluZyB0byBleHBvc2UgdGhlIEFQSSBvdmVyIHJlc3QhCkNvcHlyaWdodCAoYykgMjAyMC0yMSBWaW5jZW56byBQYWxhenpvIDx2aW5jZW56b3BhbGF6em9kZXZAZ21haWwuY29tPgoKUGVybWlzc2lvbiBpcyBoZXJlYnkgZ3JhbnRlZCwgZnJlZSBvZiBjaGFyZ2UsIHRvIGFueSBwZXJzb24gb2J0YWluaW5nIGEgY29weQpvZiB0aGlzIHNvZnR3YXJlIGFuZCBhc3NvY2lhdGVkIGRvY3VtZW50YXRpb24gZmlsZXMgKHRoZSAiU29mdHdhcmUiKSwgdG8gZGVhbAppbiB0aGUgU29mdHdhcmUgd2l0aG91dCByZXN0cmljdGlvbiwgaW5jbHVkaW5nIHdpdGhvdXQgbGltaXRhdGlvbiB0aGUgcmlnaHRzCnRvIHVzZSwgY29weSwgbW9kaWZ5LCBtZXJnZSwgcHVibGlzaCwgZGlzdHJpYnV0ZSwgc3VibGljZW5zZSwgYW5kL29yIHNlbGwKY29waWVzIG9mIHRoZSBTb2Z0d2FyZSwgYW5kIHRvIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIFNvZnR3YXJlIGlzCmZ1cm5pc2hlZCB0byBkbyBzbywgc3ViamVjdCB0byB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnM6CgpUaGUgYWJvdmUgY29weXJpZ2h0IG5vdGljZSBhbmQgdGhpcyBwZXJtaXNzaW9uIG5vdGljZSBzaGFsbCBiZSBpbmNsdWRlZCBpbiBhbGwKY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS4KClRIRSBTT0ZUV0FSRSBJUyBQUk9WSURFRCAiQVMgSVMiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELCBFWFBSRVNTIE9SCklNUExJRUQsIElOQ0xVRElORyBCVVQgTk9UIExJTUlURUQgVE8gVEhFIFdBUlJBTlRJRVMgT0YgTUVSQ0hBTlRBQklMSVRZLApGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULiBJTiBOTyBFVkVOVCBTSEFMTCBUSEUKQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWSBDTEFJTSwgREFNQUdFUyBPUiBPVEhFUgpMSUFCSUxJVFksIFdIRVRIRVIgSU4gQU4gQUNUSU9OIE9GIENPTlRSQUNULCBUT1JUIE9SIE9USEVSV0lTRSwgQVJJU0lORyBGUk9NLApPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRSBTT0ZUV0FSRSBPUiBUSEUgVVNFIE9SIE9USEVSIERFQUxJTkdTIElOIFRIRQpTT0ZUV0FSRS4KYGBgCgo= readmeEtag: '"d7d615d1fdd85382f53089af10cf5472534d0d91"' readmeLastModified: Fri, 10 Jun 2022 08:38:43 GMT repositoryId: 278958272 description: A Java plugin for c-lightning to expose the API over rest! created: '2020-07-11T23:31:41Z' updated: '2025-07-22T07:54:22Z' language: Java archived: false stars: 3 watchers: 1 forks: 1 owner: clightning4j logo: https://avatars.githubusercontent.com/u/67849355?v=4 license: NOASSERTION repoEtag: '"afc348609eb9e4ca6cfe9e26fc9f9957c4ddeaf47fc32950a5ca1e0219ab8974"' repoLastModified: Tue, 22 Jul 2025 07:54:22 GMT foundInMaster: true category: Server id: 0842c1cecab7de55d68080243b5f9b11 - source: openapi3 tags repository: https://github.com/fabidick22/voting-sys-rest-api v3: true repositoryMetadata: base64Readme: >- IyBWb3RpbmcgU3lzdGVtIFJFU1QgQVBJCiFbQ0ktdm90aW5nLXN5c10oaHR0cHM6Ly9naXRodWIuY29tL2ZhYmlkaWNrMjIvdm90aW5nLXN5cy1yZXN0LWFwaS93b3JrZmxvd3MvQ0ktdm90aW5nLXN5cy9iYWRnZS5zdmcpCgpBIFJFU1QgQVBJIGZvciB2b3Rpbmcgc3lzdGVtCgoqKkNoYXJhY3RlcmlzdGljOioqCi0gTVZDIGFzIGEgZGVzaWduIHBhdHRlcm4KLSBCYXNpYyB0b2tlbi1iYXNlZCBhdXRoZW50aWNhdGlvbiAoZG8gbm90IHVzZSB0aGlzIG1ldGhvZCwgaXQgaXMganVzdCBhbiBleGFtcGxlKQoKIyMjIE1haW4gdG9vbHMgdXNlZAotIFtFeHByZXNzXShodHRwczovL2V4cHJlc3Nqcy5jb20vKQotIFtNb25nb2RiXShodHRwczovL3d3dy5tb25nb2RiLmNvbSkKLSBbTW9uZ29vc2VdKCkKLSBbU3dhZ2dlclVJXShodHRwczovL2dpdGh1Yi5jb20vc2NvdHRpZTE5ODQvc3dhZ2dlci11aS1leHByZXNzKQotIFtNb3JnYW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9leHByZXNzanMvbW9yZ2FuKQotIFtEb2NrZXJdKGh0dHBzOi8vd3d3LmRvY2tlci5jb20vKQoKIyMgR2V0dGluZyBTdGFydGVkCkNsb25lIHRoaXMgcmVwb3NpdG9yeSBhbmQgaW5zdGFsbCBkZXBlbmRlbmNpZXMKYGBgCj4gZ2l0IGNsb25lIGdpdEBnaXRodWIuY29tOmZhYmlkaWNrMjIvdm90aW5nLXN5cy1yZXN0LWFwaS5naXQKPiBjZCB2b3Rpbmctc3lzLXJlc3QtYXBpCgo+IG5wbSBpbnN0YWxsCmBgYAojIyMjIFJ1biBkZXZlbG9wbWVudCB3aXRoIG5vZGVtb24KYGBgCj4gbnBtIHJ1biAgc3RhcnQKYGBgCgojIyMjIEJ1aWxkIGFuZCBydW4gZm9yIHByb2R1Y3Rpb24gCmBgYAo+IG5wbSBydW4gc2VydmVyCmBgYAoKIyMjIyBUZXN0cyAoTm90IGltcGxlbWVudGVkIHlldCkKVW5pdCB0ZXN0czoKYGBgCj4gbnBtIHJ1biB0ZXN0OnNwZWNzCmBgYAoKQmxhY2stYm94IGVuZC10by1lbmQgdGVzdHM6CmBgYApucG0gcnVuIHRlc3Q6ZTJlCmBgYAoKCiMjIyBEZXBsb3kgQXBwCk1ha2Ugc3VyZSB5b3UgaGF2ZSB0aGUgYC5lbnZgIGZpbGUgeW91IGNhbiBzZWUgdGhlIHRlbXBsYXRlIGluIHRoZSBgLmVudi50ZW1wbGF0ZWAgZmlsZQpfRXhhbXBsZTpfCmBgYApOT0RFX0VOVj1kZXZlbG9wbWVudApEQl9IT1NUPWRiICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKHRvIGNvbm5lY3QgaW4gZG9ja2VyLWNvbXBvc2UpCkRCX1VTRVI9cm9vdCAgICAgICAgICAgICAgICAgICAgICAgICAgICAoZGF0YWJhc2UgdXNlciBmb3IgdGhlIGFwcCkKREJfUEFTUz1zdXBlclBhc3N3ZCAgICAgICAgICAgICAgICAgICAgIChkYXRhYmFzZSBwYXNzd29yZCBmb3IgdGhlIGFwcCkKU0VDUkVUX1RPS0VOPSIiICAgICAgICAgICAgICAgICAgICAgICAgIChvcHRpb25hbCBmb3IgdGhlIGRldiBlbnZpcm9ubWVudCkKTU9OR09fSU5JVERCX1JPT1RfVVNFUk5BTUU9cm9vdCAgICAgICAgIChmb3IgbW9uZ29EQikKTU9OR09fSU5JVERCX1JPT1RfUEFTU1dPUkQ9c3VwZXJQYXNzd2QgIChmb3IgbW9uZ29EQikKYGBgCkZyb20gdGhlIHJvb3QgZGlyZWN0b3J5OgpgYGAKIyBCdWlsZCB5b3VyIGRvY2tlcgo+IGRvY2tlci1jb21wb3NlIHVwCmBgYAoKKipTdGFydCBsb2NhbCAoZGV2KSBhcHAgYW5kIGRlcGxveSBtb25nbyBpbiBEb2NrZXIqKgpgYGAKIyBCdWlsZCB5b3VyIGRvY2tlcgo+IGRvY2tlci1jb21wb3NlIHVwIGRiCj4gbnBtIHN0YXJ0ICMgZGV2IGVudmlyb25tZW50CmBgYAoKIyMjIEZ1bmN0aW9uYWxpdHkgdGVzdApBZnRlciBoYXZpbmcgdGhlIGRhdGFiYXNlIGFuZCB5b3VyIEFQSSBydW5uaW5nLCB5b3UgY2FuIHRlc3QgdGhlIGZ1bmN0aW9uYWxpdGllcyBmcm9tIHRoZSBVSSBwcm92aWRlZCBieSBzd2FnZ2VyLVVJLgpZb3UgbXVzdCBiZWFyIGluIG1pbmQgdGhhdCBmaXJzdCB5b3UgaGF2ZSB0byBjcmVhdGUgYSB1c2VyIGZyb20gdGhpcyBlbmRwb2ludCAoYC91c2Vycy9zaWduVXBgKSB0aGVuIHlvdSB3aWxsIGhhdmUgYSB0b2tlbiB0aGF0IHlvdSBtdXN0IHNldCBpbiB0aGUgc3dhZ2dlciBhdXRob3JpemVyIHRvIGJlIGFibGUgdG8gdXNlIHRoZSBvdGhlciBlbmRwb2l0cwoKIyMgQVBJIGRvY3VtZW50YXRpb24KQ2hlY2tvdXQgYGxvY2FsaG9zdDozMDAwL2RvY3NgLgpCYXNlIHBhdGg6IGBsb2NhbGhvc3Q6MzAwMC9hcGkvdjFgCgojIyBNb2R1bGUgc3RydWN0dXJlCmBgYAouCuKUnOKUgOKUgCBhcHAuanMK4pSc4pSA4pSAIGNvbmZpZwrilJzilIDilIAgY29uZmlnLmpzCuKUnOKUgOKUgCBjb250cm9sbGVycwrilJzilIDilIAgZG9ja2VyLWNvbXBvc2UueW1sCuKUnOKUgOKUgCBEb2NrZXJmaWxlCuKUnOKUgOKUgCBtaWRkbGV3YXJlcwrilJzilIDilIAgbW9kZWxzCuKUnOKUgOKUgCBub2RlX21vZHVsZXMK4pSc4pSA4pSAIHBhY2thZ2UuanNvbgrilJzilIDilIAgcGFja2FnZS1sb2NrLmpzb24K4pSc4pSA4pSAIFJFQURNRS5tZArilJzilIDilIAgcm91dGVzCuKUnOKUgOKUgCB0ZXN0CuKUnOKUgOKUgCB1dGlscwrilJTilIDilIAgdm90aW5nT0FTLnlhbWwKCjggZGlyZWN0b3JpZXMsIDggZmlsZXMKYGBgCiMjIyBSZWZlcmVuY2VzOgotIGh0dHBzOi8vbWVkaXVtLmNvbS93b2xveC9kb2N1bWVudGluZy1hLW5vZGVqcy1yZXN0LWFwaS13aXRoLW9wZW5hcGktMy1zd2FnZ2VyLTVkZWVlOWY1MDQyMAotIGh0dHBzOi8vdHdtLm1lL2NvcnJlY3Qtd2F5LXRvLXVzZS1tb25nb29zZS8K readmeEtag: '"047209aed7def0296769682b6201c9e8e7adabcc"' readmeLastModified: Wed, 20 Jul 2022 04:12:27 GMT repositoryId: 251323884 description: Base template to create a REST-API created: '2020-03-30T14:06:01Z' updated: '2023-03-05T08:14:26Z' language: JavaScript archived: false stars: 2 watchers: 1 forks: 0 owner: fabidick22 logo: https://avatars.githubusercontent.com/u/8176821?v=4 license: MIT repoEtag: '"ef0838832eb0d72f9ebc26dd2beac95766f0e6e82859f3a3f10d952e2f0acbf1"' repoLastModified: Sun, 05 Mar 2023 08:14:26 GMT foundInMaster: true category: - Testing - Server Implementations id: 360961fe9a73041e997249011d70f3d1 - source: openapi3 tags repository: https://github.com/tidusjar/wiremock.openapivalidator v3: true id: 6e98ac01bba66f60b5283ccbc8395dc3 repositoryMetadata: base64Readme: >- # WireMock OpenAPI Validator GitHub Action

Automatically validate WireMock stub mappings against OpenAPI specifications in your GitHub Actions workflows. Perfect for ensuring your mocks stay in sync with your API contracts in PR pipelines.

## Features

- ✅ **Automated Validation**: Validates WireMock stubs against OpenAPI v3 specs
- 💬 **PR Comments**: Posts detailed validation results directly to pull requests
- 📊 **Rich Reporting**: Summary tables, pass rates, and detailed failure information
- 🔄 **Comment Updates**: Updates existing comments on new commits to avoid spam
- ⚡ **Fast Setup**: Composite action with minimal configuration required
- 🎯 **Flexible Options**: Control warnings, output formats, and more

## Usage

### Basic Example

Add this to your GitHub Actions workflow (e.g., `.github/workflows/validate-mocks.yml`):

```yaml
name: Validate WireMock Mappings

on:
  pull_request:
    paths:
      - 'openapi/**'
      - 'wiremock/mappings/**'

jobs:
  validate:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4

      - name: Validate WireMock Mappings
        uses: tidusjar/wiremock-openapi-validator@v1
        with:
          openapi-path: 'openapi/api-spec.yml'
          wiremock-path: 'wiremock/mappings'
```

### With All Options

```yaml
- name: Validate WireMock Mappings
  uses: tidusjar/wiremock-openapi-validator@v1
  with:
    openapi-path: 'specs/openapi.yml'
    wiremock-path: 'mocks/mappings'
    fail-on-warnings: true
    post-comment: true
    github-token: ${{ secrets.GITHUB_TOKEN }}
```

### Using Outputs

```yaml
- name: Validate WireMock Mappings
  id: validate
  uses: tidusjar/wiremock-openapi-validator@v1
  with:
    openapi-path: 'openapi.yml'
    wiremock-path: 'mappings'

- name: Check Results
  run: |
    echo "Validation passed: ${{ steps.validate.outputs.validation-passed }}"
    echo "Total checks: ${{ steps.validate.outputs.total-checks }}"
    echo "Passed: ${{ steps.validate.outputs.passed-checks }}"
    echo "Failed: ${{ steps.validate.outputs.failed-checks }}"
    echo "Warnings: ${{ steps.validate.outputs.warning-checks }}"
```

## Inputs

| Input | Description | Required | Default |
|-------|-------------|----------|---------|
| `openapi-path` | Path to the OpenAPI specification file (YAML or JSON) | Yes | - |
| `wiremock-path` | Path to the WireMock mappings directory | Yes | - |
| `fail-on-warnings` | Fail the build if warnings are found | No | `false` |
| `post-comment` | Post validation results as a PR comment | No | `true` |
| `github-token` | GitHub token for posting PR comments | No | `${{ github.token }}` |
| `skip-install` | Skip tool installation (for local development/testing) | No | `false` |

## Outputs

| Output | Description |
|--------|-------------|
| `validation-passed` | Whether all validations passed (`true`/`false`) |
| `total-checks` | Total number of validation checks performed |
| `passed-checks` | Number of checks that passed |
| `failed-checks` | Number of checks that failed |
| `warning-checks` | Number of warnings |
| `error-checks` | Number of errors |

## What Gets Validated?

The action validates the following aspects of your WireMock mappings:

### ✅ HTTP Methods
Ensures the HTTP method in your mock matches the OpenAPI operation

### ✅ URL Path Matching
Verifies that mock URL patterns match actual API paths

### ✅ Required Parameters
Checks that all required query/path/header parameters are present

### ✅ Parameter Types
Validates that parameter types match the OpenAPI specification

### ✅ Response Structure
Verifies response properties exist and have correct types

### ✅ Required Response Properties
Ensures all required response fields are present in mocks

## Example PR Comment

The action posts a detailed comment to your PR:

```markdown
## ✅ WireMock OpenAPI Validation Results

**All validations passed!**

### Summary

| Metric | Count |
|--------|-------|
| Total Checks | 45 |
| ✅ Passed | 43 |
| ⚠️ Warnings | 2 |
| ❌ Failed | 0 |
| 🔴 Errors | 0 |
| **Pass Rate** | **95.6%** |
```

## Advanced Usage

### Fail on Warnings

Use `fail-on-warnings: true` to enforce zero warnings in your codebase:

```yaml
- uses: tidusjar/wiremock-openapi-validator@v1
  with:
    openapi-path: 'api.yml'
    wiremock-path: 'mappings'
    fail-on-warnings: true  # Build fails if any warnings
```

### Disable PR Comments

For CI environments outside pull requests or to disable comments:

```yaml
- uses: tidusjar/wiremock-openapi-validator@v1
  with:
    openapi-path: 'api.yml'
    wiremock-path: 'mappings'
    post-comment: false
```

### Multiple OpenAPI Specs

Validate against multiple specifications:

```yaml
- name: Validate User API Mocks
  uses: tidusjar/wiremock-openapi-validator@v1
  with:
    openapi-path: 'specs/users-api.yml'
    wiremock-path: 'mocks/users'

- name: Validate Orders API Mocks
  uses: tidusjar/wiremock-openapi-validator@v1
  with:
    openapi-path: 'specs/orders-api.yml'
    wiremock-path: 'mocks/orders'
```

### Local Development / Testing

When testing the action locally or using a custom-built version:

```yaml
- name: Build Tool Locally
  run: |
    cd src
    dotnet build --configuration Release
    dotnet pack --configuration Release --output ./nupkg

- name: Install Local Tool
  run: |
    dotnet tool install --global --add-source ./src/nupkg Wiremock.OpenAPIValidator

- name: Validate with Local Action
  uses: ./  # or tidusjar/wiremock-openapi-validator@branch-name
  with:
    openapi-path: 'openapi.yml'
    wiremock-path: 'mappings'
    skip-install: true  # Don't overwrite our local build!
```

### Matrix Strategy

Validate multiple environments:

```yaml
strategy:
  matrix:
    environment: [dev, staging, prod]

steps:
  - uses: actions/checkout@v4

  - name: Validate ${{ matrix.environment }} Mocks
    uses: tidusjar/wiremock-openapi-validator@v1
    with:
      openapi-path: 'specs/${{ matrix.environment }}/api.yml'
      wiremock-path: 'mocks/${{ matrix.environment }}'
```

## CLI Tool

This action is built on the WireMock OpenAPI Validator CLI tool. You can also use the CLI locally:

```bash
# Install globally
dotnet tool install --global Wiremock.OpenAPIValidator

# Run validation
wiremockopenapi -o openapi.yml -w mappings/

# With options
wiremockopenapi -o api.yml -w mappings/ --format json --quiet
```

### CLI Options

- `--format`: Output format (`console`, `json`, `junit`, `github`)
- `--output-file`: Write results to a file
- `--quiet`: Suppress banner and charts
- `--no-color`: Disable colored output

## Requirements

- .NET 10.0 SDK (automatically installed by the action)
- WireMock mapping files in JSON format
- OpenAPI 3.x specification (YAML or JSON)

## Troubleshooting

### Action fails with "tool not found"

The action automatically installs the tool. If you encounter issues, ensure:
- Your runner has .NET 10.0 SDK available
- The tool installation step completes successfully

### PR comment not posted

Check that:
- The workflow is triggered by a `pull_request` event
- The `github-token` input has permissions to comment on PRs
- `post-comment` is set to `true` (default)

### Validation fails locally but passes in CI

Ensure you're using the same version of the tool:
```bash
dotnet tool update --global Wiremock.OpenAPIValidator
```

## Contributing

Issues and pull requests are welcome! See [CONTRIBUTING.md](CONTRIBUTING.md) for guidelines.

## License

[MIT License](LICENSE)

---

Built with ❤️ for teams using WireMock and OpenAPI
 readmeEtag: '"1d0ba0e808aa516ff8b20ed3cad381c3bce7a360"' readmeLastModified: Thu, 13 Nov 2025 16:05:49 GMT repositoryId: 508230905 description: Validate your Wiremock mappings against an OpenAPI spec created: '2022-06-28T09:14:13Z' updated: '2025-11-24T14:10:38Z' language: C# archived: false stars: 5 watchers: 1 forks: 0 owner: tidusjar logo: https://avatars.githubusercontent.com/u/6642220?v=4 license: GPL-3.0 repoEtag: '"14cbbf399416a494610b96334ecddaed84f91647fe6e39f588ff6376b817999d"' repoLastModified: Mon, 24 Nov 2025 14:10:38 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/simonit/vcita-platform-java-sdk v3: true repositoryMetadata: base64Readme: >- # vcita-platform-java-sdk

Platform API
- API version: v2.0

No description provided (generated by Openapi Generator https://github.com/openapitools/openapi-generator)


*Automatically generated by the [OpenAPI Generator](https://openapi-generator.tech)*


## Requirements

Building the API client library requires:
1. Java 1.8+
2. Maven (3.8.3+)/Gradle (7.2+)

## Installation

To install the API client library to your local Maven repository, simply execute:

```shell
mvn clean install
```

To deploy it to a remote Maven repository instead, configure the settings of the repository and execute:

```shell
mvn clean deploy
```

Refer to the [OSSRH Guide](http://central.sonatype.org/pages/ossrh-guide.html) for more information.

### Maven users

Add this dependency to your project's POM:

```xml
<dependency>
  <groupId>com.vcita.platform</groupId>
  <artifactId>vcita-platform-java-sdk</artifactId>
  <version>v2.0</version>
  <scope>compile</scope>
</dependency>
```

### Gradle users

Add this dependency to your project's build file:

```groovy
  repositories {
    mavenCentral()     // Needed if the 'vcita-platform-java-sdk' jar has been published to maven central.
    mavenLocal()       // Needed if the 'vcita-platform-java-sdk' jar has been published to the local maven repo.
  }

  dependencies {
     implementation "com.vcita.platform:vcita-platform-java-sdk:v2.0"
  }
```

### Others

At first generate the JAR by executing:

```shell
mvn clean package
```

Then manually install the following JARs:

* `target/vcita-platform-java-sdk-v2.0.jar`
* `target/lib/*.jar`

## Getting Started

Please follow the [installation](#installation) instruction and execute the following Java code:

```java

// Import classes:
import com.vcita.platform.client.ApiClient;
import com.vcita.platform.client.ApiException;
import com.vcita.platform.client.Configuration;
import com.vcita.platform.client.auth.*;
import com.vcita.platform.client.models.*;
import com.vcita.platform.client.api.ApplicationsApi;

public class Example {
  public static void main(String[] args) {
    ApiClient defaultClient = Configuration.getDefaultApiClient();
    defaultClient.setBasePath("http://api.vcita.biz/platform/v1");
    
    // Configure OAuth2 access token for authorization: default
    OAuth default = (OAuth) defaultClient.getAuthentication("default");
    default.setAccessToken("YOUR ACCESS TOKEN");

    ApplicationsApi apiInstance = new ApplicationsApi(defaultClient);
    String authorization = "authorization_example"; // String | API Token. The resulting header should look like this: 'Authorization: \"Bearer API_TOKEN\"'
    try {
      AppsGet201Response result = apiInstance.appsGet(authorization);
      System.out.println(result);
    } catch (ApiException e) {
      System.err.println("Exception when calling ApplicationsApi#appsGet");
      System.err.println("Status code: " + e.getCode());
      System.err.println("Reason: " + e.getResponseBody());
      System.err.println("Response headers: " + e.getResponseHeaders());
      e.printStackTrace();
    }
  }
}

```

## Documentation for API Endpoints

All URIs are relative to *http://api.vcita.biz/platform/v1*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*ApplicationsApi* | [**appsGet**](docs/ApplicationsApi.md#appsGet) | **GET** /apps | List of apps
*ApplicationsApi* | [**appsIdAssignPost**](docs/ApplicationsApi.md#appsIdAssignPost) | **POST** /apps/{id}/assign | Assign an app on directory or a business [Internal]
*ApplicationsApi* | [**appsIdGet**](docs/ApplicationsApi.md#appsIdGet) | **GET** /apps/{id} | Read an app
*ApplicationsApi* | [**appsIdInstallAppPost**](docs/ApplicationsApi.md#appsIdInstallAppPost) | **POST** /apps/{id}/install_app | Install an app [Internal]
*ApplicationsApi* | [**appsIdPut**](docs/ApplicationsApi.md#appsIdPut) | **PUT** /apps/{id} | Updates an app
*ApplicationsApi* | [**appsIdUnassignPost**](docs/ApplicationsApi.md#appsIdUnassignPost) | **POST** /apps/{id}/unassign | Unassign an app on directory or a business [Internal]
*ApplicationsApi* | [**appsIdUninstallAppPost**](docs/ApplicationsApi.md#appsIdUninstallAppPost) | **POST** /apps/{id}/uninstall_app | Uninstall an app [Internal]
*ApplicationsApi* | [**appsPost**](docs/ApplicationsApi.md#appsPost) | **POST** /apps | Creates an app
*AppointmentsApi* | [**globalAppointmentsGet**](docs/AppointmentsApi.md#globalAppointmentsGet) | **GET** /global/appointments | Get appointments list
*AppointmentsApi* | [**schedulingAppointmentsAppointmentIdGet**](docs/AppointmentsApi.md#schedulingAppointmentsAppointmentIdGet) | **GET** /scheduling/appointments/{appointment_id} | Get Appointment
*AppointmentsApi* | [**schedulingAppointmentsGet**](docs/AppointmentsApi.md#schedulingAppointmentsGet) | **GET** /scheduling/appointments | Get Appointments List
*AvailabilityApi* | [**servicesServiceIdAvailabilityGet**](docs/AvailabilityApi.md#servicesServiceIdAvailabilityGet) | **GET** /services/{service_id}/availability | Get Service availability By ID [Alpha]
*BookingsApi* | [**schedulingBookingsBookingUidUpdateRsvpStatePut**](docs/BookingsApi.md#schedulingBookingsBookingUidUpdateRsvpStatePut) | **PUT** /scheduling/bookings/{booking_uid}/update_rsvp_state | Update RSVP state
*BookingsApi* | [**schedulingBookingsCancelPost**](docs/BookingsApi.md#schedulingBookingsCancelPost) | **POST** /scheduling/bookings/cancel | Cancel a Booking [Alpha]
*BookingsApi* | [**schedulingBookingsGet**](docs/BookingsApi.md#schedulingBookingsGet) | **GET** /scheduling/bookings | Get Bookings [Alpha] 
*BookingsApi* | [**schedulingBookingsPost**](docs/BookingsApi.md#schedulingBookingsPost) | **POST** /scheduling/bookings | Create New Booking [Alpha]
*BookingsApi* | [**schedulingWaitlistCancelPut**](docs/BookingsApi.md#schedulingWaitlistCancelPut) | **PUT** /scheduling/waitlist/cancel | Cancel a Waitlist registration
*BookingsApi* | [**schedulingWaitlistPost**](docs/BookingsApi.md#schedulingWaitlistPost) | **POST** /scheduling/waitlist | Create event Waitlist
*BusinessApi* | [**businessesBusinessIdFeaturesGet**](docs/BusinessApi.md#businessesBusinessIdFeaturesGet) | **GET** /businesses/{business_id}/features | Get Business Features
*BusinessApi* | [**businessesBusinessIdGet**](docs/BusinessApi.md#businessesBusinessIdGet) | **GET** /businesses/{business_id} | Get Business
*BusinessApi* | [**businessesBusinessIdPost**](docs/BusinessApi.md#businessesBusinessIdPost) | **POST** /businesses/{business_id} | Update Business
*BusinessApi* | [**businessesGet**](docs/BusinessApi.md#businessesGet) | **GET** /businesses | Get Business filtered by email or external_id
*BusinessApi* | [**businessesPost**](docs/BusinessApi.md#businessesPost) | **POST** /businesses | Create a Business
*BusinessApi* | [**businessesValidateLoginGet**](docs/BusinessApi.md#businessesValidateLoginGet) | **GET** /businesses/validate_login | Validate Login
*BusinessApi* | [**globalBusinessesGet**](docs/BusinessApi.md#globalBusinessesGet) | **GET** /global/businesses | Gets a list of businesses
*CardApi* | [**paymentCardsSyncCardPost**](docs/CardApi.md#paymentCardsSyncCardPost) | **POST** /payment/cards/sync_card | Create sync card
*CategoriesApi* | [**categoriesGet**](docs/CategoriesApi.md#categoriesGet) | **GET** /categories | Get Categories [Alpha]
*ClientPackagesApi* | [**clientsClientIdPaymentClientPackagesGet**](docs/ClientPackagesApi.md#clientsClientIdPaymentClientPackagesGet) | **GET** /clients/{client_id}/payment/client_packages | Get Client Packages List Of Client [Alpha]
*ClientPackagesApi* | [**clientsPaymentClientPackagesUpdateUsagePost**](docs/ClientPackagesApi.md#clientsPaymentClientPackagesUpdateUsagePost) | **POST** /clients/payment/client_packages/update_usage | Use Client Package Credit [Alpha]
*ClientPackagesApi* | [**clientsPaymentClientPackagesValidateGet**](docs/ClientPackagesApi.md#clientsPaymentClientPackagesValidateGet) | **GET** /clients/payment/client_packages/validate | Validate Client Package has redemption [Alpha]
*ClientPackagesApi* | [**paymentClientPackagesPost**](docs/ClientPackagesApi.md#paymentClientPackagesPost) | **POST** /payment/client_packages | Add package to client [Alpha]
*ClientPackagesApi* | [**paymentClientPackagesUpdateUsagePost**](docs/ClientPackagesApi.md#paymentClientPackagesUpdateUsagePost) | **POST** /payment/client_packages/update_usage | Use Client Package Credit [Alpha]
*ClientsApi* | [**clientsClientIdDelete**](docs/ClientsApi.md#clientsClientIdDelete) | **DELETE** /clients/{client_id} | Deletes a Client by Id
*ClientsApi* | [**clientsClientIdGet**](docs/ClientsApi.md#clientsClientIdGet) | **GET** /clients/{client_id} | Retrieves a Client by Id
*ClientsApi* | [**clientsClientIdPut**](docs/ClientsApi.md#clientsClientIdPut) | **PUT** /clients/{client_id} | Updates a Client
*ClientsApi* | [**clientsGet**](docs/ClientsApi.md#clientsGet) | **GET** /clients | Get Clients
*ClientsApi* | [**clientsPost**](docs/ClientsApi.md#clientsPost) | **POST** /clients | Create a Client
*ClientsApi* | [**globalClientsGet**](docs/ClientsApi.md#globalClientsGet) | **GET** /global/clients | Gets a list of clients
*ConversationsApi* | [**clientsClientIdConversationsGet**](docs/ConversationsApi.md#clientsClientIdConversationsGet) | **GET** /clients/{client_id}/conversations | Get Conversations For A Client
*ConversationsApi* | [**conversationsGet**](docs/ConversationsApi.md#conversationsGet) | **GET** /conversations | Get Conversations
*DocumentsApi* | [**clientsClientIdDocumentsGet**](docs/DocumentsApi.md#clientsClientIdDocumentsGet) | **GET** /clients/{client_id}/documents | Get Documents For A Client
*EstimatesApi* | [**clientsClientIdEstimatesGet**](docs/EstimatesApi.md#clientsClientIdEstimatesGet) | **GET** /clients/{client_id}/estimates | Get Estimates Of Client
*EstimatesApi* | [**estimatesGet**](docs/EstimatesApi.md#estimatesGet) | **GET** /estimates | Get Estimates List
*EstimatesApi* | [**estimatesPost**](docs/EstimatesApi.md#estimatesPost) | **POST** /estimates | Create New Estimate
*EventAttendanceApi* | [**globalEventAttendancesGet**](docs/EventAttendanceApi.md#globalEventAttendancesGet) | **GET** /global/event_attendances | Gets a list of event attendances
*EventInstanceApi* | [**globalEventInstancesGet**](docs/EventInstanceApi.md#globalEventInstancesGet) | **GET** /global/event_instances | Gets a list of event instances
*EventInstanceApi* | [**schedulingEventInstanceEventInstanceIdGet**](docs/EventInstanceApi.md#schedulingEventInstanceEventInstanceIdGet) | **GET** /scheduling/event_instance/{event_instance_id} | Get Event Instance
*FieldsApi* | [**fieldsFieldIdDelete**](docs/FieldsApi.md#fieldsFieldIdDelete) | **DELETE** /fields/{field_id} | Deletes a Field
*FieldsApi* | [**fieldsFieldIdGet**](docs/FieldsApi.md#fieldsFieldIdGet) | **GET** /fields/{field_id} | Get Field
*FieldsApi* | [**fieldsFieldIdPut**](docs/FieldsApi.md#fieldsFieldIdPut) | **PUT** /fields/{field_id} | Updates a Field
*FieldsApi* | [**fieldsGet**](docs/FieldsApi.md#fieldsGet) | **GET** /fields | Get all Fields
*FieldsApi* | [**fieldsPost**](docs/FieldsApi.md#fieldsPost) | **POST** /fields | Create a New Field
*FormsApi* | [**formsGet**](docs/FormsApi.md#formsGet) | **GET** /forms | Get Forms
*FormsApi* | [**formsPut**](docs/FormsApi.md#formsPut) | **PUT** /forms | Updates a Form
*GlobalApi* | [**globalClientPackagesGet**](docs/GlobalApi.md#globalClientPackagesGet) | **GET** /global/client_packages | Gets a list of client packages
*GlobalApi* | [**globalConversationsGet**](docs/GlobalApi.md#globalConversationsGet) | **GET** /global/conversations | Gets a list of conversations
*GlobalApi* | [**globalEstimatesGet**](docs/GlobalApi.md#globalEstimatesGet) | **GET** /global/estimates | Gets a list of estimates
*GlobalApi* | [**globalInvoicesGet**](docs/GlobalApi.md#globalInvoicesGet) | **GET** /global/invoices | Gets a list of invoices
*GlobalApi* | [**globalMattersGet**](docs/GlobalApi.md#globalMattersGet) | **GET** /global/matters | Gets a list of matters
*GlobalApi* | [**globalPackagesGet**](docs/GlobalApi.md#globalPackagesGet) | **GET** /global/packages | Gets a list of packages
*GlobalApi* | [**globalProductsGet**](docs/GlobalApi.md#globalProductsGet) | **GET** /global/products | Gets a list of products
*InvoicesApi* | [**clientsClientIdInvoicesGet**](docs/InvoicesApi.md#clientsClientIdInvoicesGet) | **GET** /clients/{client_id}/invoices | Get Invoices Of Client
*InvoicesApi* | [**invoicesGet**](docs/InvoicesApi.md#invoicesGet) | **GET** /invoices | Get Invoices List
*InvoicesApi* | [**invoicesInvoiceIdGet**](docs/InvoicesApi.md#invoicesInvoiceIdGet) | **GET** /invoices/{invoice_id} | Get Invoice By ID
*InvoicesApi* | [**invoicesPost**](docs/InvoicesApi.md#invoicesPost) | **POST** /invoices | Create New Invoice
*LeadGenApi* | [**leadgenPost**](docs/LeadGenApi.md#leadgenPost) | **POST** /leadgen | Creates a lead
*MarketingApi* | [**globalCampaignRecipientsGet**](docs/MarketingApi.md#globalCampaignRecipientsGet) | **GET** /global/campaign_recipients | Gets a list of campaign recipients
*MarketingApi* | [**globalCampaignsGet**](docs/MarketingApi.md#globalCampaignsGet) | **GET** /global/campaigns | Gets a list of campaigns
*MarketingApi* | [**globalScheduledCampaignsGet**](docs/MarketingApi.md#globalScheduledCampaignsGet) | **GET** /global/scheduled_campaigns | Gets a list of scheduled campaigns
*MattersApi* | [**clientsMergesMergeClientsPut**](docs/MattersApi.md#clientsMergesMergeClientsPut) | **PUT** /clients/merges/merge_clients | Merge duplicate clients into a primary client.
*MattersApi* | [**clientsMergesPossibleMergeMasterGet**](docs/MattersApi.md#clientsMergesPossibleMergeMasterGet) | **GET** /clients/merges/possible_merge_master | List of clients that can be used as primary for merge.
*MessagesApi* | [**messagesPost**](docs/MessagesApi.md#messagesPost) | **POST** /messages | Create a Message
*PackagesApi* | [**paymentPackagesGet**](docs/PackagesApi.md#paymentPackagesGet) | **GET** /payment/packages | Get all package of business
*PackagesApi* | [**paymentPackagesPackageIdGet**](docs/PackagesApi.md#paymentPackagesPackageIdGet) | **GET** /payment/packages/{package_id} | Show package info
*PackagesApi* | [**paymentPackagesPackageIdPut**](docs/PackagesApi.md#paymentPackagesPackageIdPut) | **PUT** /payment/packages/{package_id} | Update package [Alpha]
*PackagesApi* | [**paymentPackagesPost**](docs/PackagesApi.md#paymentPackagesPost) | **POST** /payment/packages | Create Package 
*PaymentCheckoutApi* | [**paymentCheckoutPut**](docs/PaymentCheckoutApi.md#paymentCheckoutPut) | **PUT** /payment/checkout/ | Send checkout update by webhook 
*PaymentCheckoutApi* | [**paymentCheckoutUrlKeyGet**](docs/PaymentCheckoutApi.md#paymentCheckoutUrlKeyGet) | **GET** /payment/checkout/{url_key} | Get Open Checkout For Specific url_key
*PaymentSettingsApi* | [**paymentSettingsGet**](docs/PaymentSettingsApi.md#paymentSettingsGet) | **GET** /payment/settings | Get Current Payment Settings
*PaymentSettingsApi* | [**paymentSettingsPost**](docs/PaymentSettingsApi.md#paymentSettingsPost) | **POST** /payment/settings | Update Payment Settings
*PaymentStatusesApi* | [**paymentStatusesIdApplyCouponPut**](docs/PaymentStatusesApi.md#paymentStatusesIdApplyCouponPut) | **PUT** /payment_statuses/{id}/apply_coupon | Apply Coupon
*PaymentStatusesApi* | [**paymentStatusesIdValidateCouponGet**](docs/PaymentStatusesApi.md#paymentStatusesIdValidateCouponGet) | **GET** /payment_statuses/{id}/validate_coupon | Validate Coupon
*PaymentsApi* | [**clientsClientIdPaymentsGet**](docs/PaymentsApi.md#clientsClientIdPaymentsGet) | **GET** /clients/{client_id}/payments | Get Payments of Client
*PaymentsApi* | [**globalPaymentsGet**](docs/PaymentsApi.md#globalPaymentsGet) | **GET** /global/payments | Gets a list of payments
*PaymentsApi* | [**paymentsGet**](docs/PaymentsApi.md#paymentsGet) | **GET** /payments | Get Payments List
*PaymentsApi* | [**paymentsPaymentUidMatchPost**](docs/PaymentsApi.md#paymentsPaymentUidMatchPost) | **POST** /payments/{payment_uid}/match | Match Payment
*PaymentsApi* | [**paymentsPost**](docs/PaymentsApi.md#paymentsPost) | **POST** /payments | Create New Payment
*SchedulingFormsApi* | [**schedulingSchedulingFormsGetFormGet**](docs/SchedulingFormsApi.md#schedulingSchedulingFormsGetFormGet) | **GET** /scheduling/scheduling_forms/get_form | 
*ServicesApi* | [**categoriesCategoryIdServicesGet**](docs/ServicesApi.md#categoriesCategoryIdServicesGet) | **GET** /categories/{category_id}/services | Get Services List by category [Alpha]
*ServicesApi* | [**globalServicesGet**](docs/ServicesApi.md#globalServicesGet) | **GET** /global/services | Gets a list of services
*ServicesApi* | [**servicesGet**](docs/ServicesApi.md#servicesGet) | **GET** /services | Get Services List [Alpha]
*ServicesApi* | [**servicesServiceIdGet**](docs/ServicesApi.md#servicesServiceIdGet) | **GET** /services/{service_id} | Get Service By ID [Alpha]
*StaffApi* | [**businessesBusinessIdStaffsGet**](docs/StaffApi.md#businessesBusinessIdStaffsGet) | **GET** /businesses/{business_id}/staffs | Get Staff Members
*StaffApi* | [**businessesBusinessIdStaffsPost**](docs/StaffApi.md#businessesBusinessIdStaffsPost) | **POST** /businesses/{business_id}/staffs | Create Staff Member
*StaffApi* | [**businessesBusinessIdStaffsStaffIdDelete**](docs/StaffApi.md#businessesBusinessIdStaffsStaffIdDelete) | **DELETE** /businesses/{business_id}/staffs/{staff_id} | Delete Staff Member
*StaffApi* | [**businessesBusinessIdStaffsStaffIdGet**](docs/StaffApi.md#businessesBusinessIdStaffsStaffIdGet) | **GET** /businesses/{business_id}/staffs/{staff_id} | Get Staff Member Details
*StaffApi* | [**globalStaffsGet**](docs/StaffApi.md#globalStaffsGet) | **GET** /global/staffs | Gets a list of staffs
*StaffApi* | [**schedulingStaffGet**](docs/StaffApi.md#schedulingStaffGet) | **GET** /scheduling/staff | Get Staff List [Alpha]
*StaffApi* | [**schedulingStaffStaffIdGet**](docs/StaffApi.md#schedulingStaffStaffIdGet) | **GET** /scheduling/staff/{staff_id} | Get Staff By ID [Alpha]
*TokenManagementApi* | [**tokensGet**](docs/TokenManagementApi.md#tokensGet) | **GET** /tokens | Get tokens
*TokenManagementApi* | [**tokensPost**](docs/TokenManagementApi.md#tokensPost) | **POST** /tokens | Creates a token
*TokenManagementApi* | [**tokensRevokePost**](docs/TokenManagementApi.md#tokensRevokePost) | **POST** /tokens/revoke | Revokes a token
*WebhooksApi* | [**webhookSubscribePost**](docs/WebhooksApi.md#webhookSubscribePost) | **POST** /webhook/subscribe | Subscribe To Webhook
*WebhooksApi* | [**webhookUnsubscribePost**](docs/WebhooksApi.md#webhookUnsubscribePost) | **POST** /webhook/unsubscribe | Unsubscribe from Webhook
*WebhooksApi* | [**webhooksGet**](docs/WebhooksApi.md#webhooksGet) | **GET** /webhooks | Get Webhooks List
*WizardApi* | [**businessesBusinessIdWizardsGet**](docs/WizardApi.md#businessesBusinessIdWizardsGet) | **GET** /businesses/{business_id}/wizards | Gets a list of wizards
*WizardApi* | [**businessesBusinessIdWizardsWizardNameGet**](docs/WizardApi.md#businessesBusinessIdWizardsWizardNameGet) | **GET** /businesses/{business_id}/wizards/{wizard_name} | Gets a wizard by name
*WizardApi* | [**businessesBusinessIdWizardsWizardNamePut**](docs/WizardApi.md#businessesBusinessIdWizardsWizardNamePut) | **PUT** /businesses/{business_id}/wizards/{wizard_name} | Update a wizard


## Documentation for Models

 - [ActionData](docs/ActionData.md)
 - [AdminAccount](docs/AdminAccount.md)
 - [App](docs/App.md)
 - [AppData](docs/AppData.md)
 - [AppData1](docs/AppData1.md)
 - [Application](docs/Application.md)
 - [Appointment](docs/Appointment.md)
 - [AppsDeepLink](docs/AppsDeepLink.md)
 - [AppsDescription](docs/AppsDescription.md)
 - [AppsGet201Response](docs/AppsGet201Response.md)
 - [AppsIdAssignPost201Response](docs/AppsIdAssignPost201Response.md)
 - [AppsIdGet200Response](docs/AppsIdGet200Response.md)
 - [AppsIdInstallAppPost201Response](docs/AppsIdInstallAppPost201Response.md)
 - [AppsIdInstallAppPost422Response](docs/AppsIdInstallAppPost422Response.md)
 - [AppsIdMenuItems](docs/AppsIdMenuItems.md)
 - [AppsIdMenuItemsSubitems](docs/AppsIdMenuItemsSubitems.md)
 - [AppsIdPut200Response](docs/AppsIdPut200Response.md)
 - [AppsIdUnassignPost201Response](docs/AppsIdUnassignPost201Response.md)
 - [AppsIdUninstallAppPost201Response](docs/AppsIdUninstallAppPost201Response.md)
 - [AppsIdUninstallAppPost422Response](docs/AppsIdUninstallAppPost422Response.md)
 - [AppsMenuItems](docs/AppsMenuItems.md)
 - [AppsMenuItemsItemName](docs/AppsMenuItemsItemName.md)
 - [AppsMenuItemsSubitems](docs/AppsMenuItemsSubitems.md)
 - [AppsPost201Response](docs/AppsPost201Response.md)
 - [Assignment](docs/Assignment.md)
 - [Availability](docs/Availability.md)
 - [Body2](docs/Body2.md)
 - [Booking](docs/Booking.md)
 - [Booking1](docs/Booking1.md)
 - [BookingCredit](docs/BookingCredit.md)
 - [BookingRestriction](docs/BookingRestriction.md)
 - [Branding](docs/Branding.md)
 - [Business](docs/Business.md)
 - [Business1](docs/Business1.md)
 - [BusinessesAdminAccount](docs/BusinessesAdminAccount.md)
 - [BusinessesBusiness](docs/BusinessesBusiness.md)
 - [BusinessesBusinessIdBusiness](docs/BusinessesBusinessIdBusiness.md)
 - [BusinessesBusinessIdBusinessAdminAccount](docs/BusinessesBusinessIdBusinessAdminAccount.md)
 - [BusinessesBusinessIdBusinessBusiness](docs/BusinessesBusinessIdBusinessBusiness.md)
 - [BusinessesBusinessIdBusinessIntegrations](docs/BusinessesBusinessIdBusinessIntegrations.md)
 - [BusinessesBusinessIdBusinessIntegrationsPlugins](docs/BusinessesBusinessIdBusinessIntegrationsPlugins.md)
 - [BusinessesBusinessIdBusinessMeta](docs/BusinessesBusinessIdBusinessMeta.md)
 - [BusinessesBusinessIdBusinessMetaPlan](docs/BusinessesBusinessIdBusinessMetaPlan.md)
 - [BusinessesBusinessIdFeaturesGet200Response](docs/BusinessesBusinessIdFeaturesGet200Response.md)
 - [BusinessesBusinessIdGet200Response](docs/BusinessesBusinessIdGet200Response.md)
 - [BusinessesBusinessIdPost200Response](docs/BusinessesBusinessIdPost200Response.md)
 - [BusinessesBusinessIdPost400Response](docs/BusinessesBusinessIdPost400Response.md)
 - [BusinessesBusinessIdStaffsGet200Response](docs/BusinessesBusinessIdStaffsGet200Response.md)
 - [BusinessesBusinessIdStaffsMeta](docs/BusinessesBusinessIdStaffsMeta.md)
 - [BusinessesBusinessIdStaffsPost200Response](docs/BusinessesBusinessIdStaffsPost200Response.md)
 - [BusinessesBusinessIdStaffsStaff](docs/BusinessesBusinessIdStaffsStaff.md)
 - [BusinessesBusinessIdStaffsStaffIdDelete200Response](docs/BusinessesBusinessIdStaffsStaffIdDelete200Response.md)
 - [BusinessesBusinessIdStaffsStaffIdGet200Response](docs/BusinessesBusinessIdStaffsStaffIdGet200Response.md)
 - [BusinessesBusinessIdWizardsGet200Response](docs/BusinessesBusinessIdWizardsGet200Response.md)
 - [BusinessesBusinessIdWizardsWizardNameGet200Response](docs/BusinessesBusinessIdWizardsWizardNameGet200Response.md)
 - [BusinessesBusinessIdWizardsWizardNamePut200Response](docs/BusinessesBusinessIdWizardsWizardNamePut200Response.md)
 - [BusinessesGet200Response](docs/BusinessesGet200Response.md)
 - [BusinessesMeta](docs/BusinessesMeta.md)
 - [BusinessesMetaAnalytics](docs/BusinessesMetaAnalytics.md)
 - [BusinessesPost200Response](docs/BusinessesPost200Response.md)
 - [BusinessesValidateLoginGet200Response](docs/BusinessesValidateLoginGet200Response.md)
 - [Cancellation](docs/Cancellation.md)
 - [Card](docs/Card.md)
 - [CategoriesCategoryIdServicesGet200Response](docs/CategoriesCategoryIdServicesGet200Response.md)
 - [CategoriesGet200Response](docs/CategoriesGet200Response.md)
 - [Category](docs/Category.md)
 - [Client](docs/Client.md)
 - [ClientPackage](docs/ClientPackage.md)
 - [ClientsClientIdConversationsGet200Response](docs/ClientsClientIdConversationsGet200Response.md)
 - [ClientsClientIdDelete200Response](docs/ClientsClientIdDelete200Response.md)
 - [ClientsClientIdDocumentsGet200Response](docs/ClientsClientIdDocumentsGet200Response.md)
 - [ClientsClientIdEstimatesGet200Response](docs/ClientsClientIdEstimatesGet200Response.md)
 - [ClientsClientIdGet200Response](docs/ClientsClientIdGet200Response.md)
 - [ClientsClientIdInvoicesGet201Response](docs/ClientsClientIdInvoicesGet201Response.md)
 - [ClientsClientIdPaymentClientPackagesGet200Response](docs/ClientsClientIdPaymentClientPackagesGet200Response.md)
 - [ClientsClientIdPaymentClientPackagesGet422Response](docs/ClientsClientIdPaymentClientPackagesGet422Response.md)
 - [ClientsClientIdPaymentsGet201Response](docs/ClientsClientIdPaymentsGet201Response.md)
 - [ClientsClientIdPut200Response](docs/ClientsClientIdPut200Response.md)
 - [ClientsGet200Response](docs/ClientsGet200Response.md)
 - [ClientsMergesMergeClientsPut200Response](docs/ClientsMergesMergeClientsPut200Response.md)
 - [ClientsMergesPossibleMergeMasterGet200Response](docs/ClientsMergesPossibleMergeMasterGet200Response.md)
 - [ClientsPaymentClientPackagesUpdateUsagePost200Response](docs/ClientsPaymentClientPackagesUpdateUsagePost200Response.md)
 - [ClientsPaymentClientPackagesUpdateUsagePost422Response](docs/ClientsPaymentClientPackagesUpdateUsagePost422Response.md)
 - [ClientsPaymentClientPackagesValidateGet200Response](docs/ClientsPaymentClientPackagesValidateGet200Response.md)
 - [ClientsPaymentClientPackagesValidateGet422Response](docs/ClientsPaymentClientPackagesValidateGet422Response.md)
 - [ClientsPost201Response](docs/ClientsPost201Response.md)
 - [Contact](docs/Contact.md)
 - [Conversation](docs/Conversation.md)
 - [ConversationsGet200Response](docs/ConversationsGet200Response.md)
 - [CreditItem](docs/CreditItem.md)
 - [Data](docs/Data.md)
 - [Data1](docs/Data1.md)
 - [Data10](docs/Data10.md)
 - [Data11](docs/Data11.md)
 - [Data12](docs/Data12.md)
 - [Data13](docs/Data13.md)
 - [Data14](docs/Data14.md)
 - [Data15](docs/Data15.md)
 - [Data16](docs/Data16.md)
 - [Data17](docs/Data17.md)
 - [Data18](docs/Data18.md)
 - [Data19](docs/Data19.md)
 - [Data2](docs/Data2.md)
 - [Data20](docs/Data20.md)
 - [Data21](docs/Data21.md)
 - [Data22](docs/Data22.md)
 - [Data23](docs/Data23.md)
 - [Data24](docs/Data24.md)
 - [Data25](docs/Data25.md)
 - [Data26](docs/Data26.md)
 - [Data27](docs/Data27.md)
 - [Data28](docs/Data28.md)
 - [Data29](docs/Data29.md)
 - [Data3](docs/Data3.md)
 - [Data30](docs/Data30.md)
 - [Data31](docs/Data31.md)
 - [Data32](docs/Data32.md)
 - [Data33](docs/Data33.md)
 - [Data34](docs/Data34.md)
 - [Data35](docs/Data35.md)
 - [Data36](docs/Data36.md)
 - [Data37](docs/Data37.md)
 - [Data38](docs/Data38.md)
 - [Data4](docs/Data4.md)
 - [Data40](docs/Data40.md)
 - [Data41](docs/Data41.md)
 - [Data42](docs/Data42.md)
 - [Data43](docs/Data43.md)
 - [Data44](docs/Data44.md)
 - [Data45](docs/Data45.md)
 - [Data46](docs/Data46.md)
 - [Data47](docs/Data47.md)
 - [Data48](docs/Data48.md)
 - [Data49](docs/Data49.md)
 - [Data5](docs/Data5.md)
 - [Data6](docs/Data6.md)
 - [Data7](docs/Data7.md)
 - [Data8](docs/Data8.md)
 - [Data9](docs/Data9.md)
 - [Datum](docs/Datum.md)
 - [Datum1](docs/Datum1.md)
 - [Datum10](docs/Datum10.md)
 - [Datum11](docs/Datum11.md)
 - [Datum12](docs/Datum12.md)
 - [Datum13](docs/Datum13.md)
 - [Datum14](docs/Datum14.md)
 - [Datum15](docs/Datum15.md)
 - [Datum16](docs/Datum16.md)
 - [Datum17](docs/Datum17.md)
 - [Datum18](docs/Datum18.md)
 - [Datum19](docs/Datum19.md)
 - [Datum20](docs/Datum20.md)
 - [Datum3](docs/Datum3.md)
 - [Datum4](docs/Datum4.md)
 - [Datum5](docs/Datum5.md)
 - [Datum6](docs/Datum6.md)
 - [Datum7](docs/Datum7.md)
 - [Datum8](docs/Datum8.md)
 - [Datum9](docs/Datum9.md)
 - [DeepLink](docs/DeepLink.md)
 - [Description](docs/Description.md)
 - [Error](docs/Error.md)
 - [Error1](docs/Error1.md)
 - [Estimate](docs/Estimate.md)
 - [EstimatesGet200Response](docs/EstimatesGet200Response.md)
 - [EstimatesItems](docs/EstimatesItems.md)
 - [EstimatesPost201Response](docs/EstimatesPost201Response.md)
 - [EstimatesTaxes](docs/EstimatesTaxes.md)
 - [EventInstance](docs/EventInstance.md)
 - [FieldItem](docs/FieldItem.md)
 - [FieldsFieldIdDelete200Response](docs/FieldsFieldIdDelete200Response.md)
 - [FieldsFieldIdGet200Response](docs/FieldsFieldIdGet200Response.md)
 - [FieldsFieldIdPut200Response](docs/FieldsFieldIdPut200Response.md)
 - [FieldsGet200Response](docs/FieldsGet200Response.md)
 - [FieldsPost201Response](docs/FieldsPost201Response.md)
 - [FormData](docs/FormData.md)
 - [FormsFields](docs/FormsFields.md)
 - [FormsGet200Response](docs/FormsGet200Response.md)
 - [FormsPut200Response](docs/FormsPut200Response.md)
 - [FormsSections](docs/FormsSections.md)
 - [GlobalAppointmentsGet200Response](docs/GlobalAppointmentsGet200Response.md)
 - [GlobalBusinessesGet200Response](docs/GlobalBusinessesGet200Response.md)
 - [GlobalCampaignRecipientsGet200Response](docs/GlobalCampaignRecipientsGet200Response.md)
 - [GlobalCampaignsGet200Response](docs/GlobalCampaignsGet200Response.md)
 - [GlobalClientPackagesGet200Response](docs/GlobalClientPackagesGet200Response.md)
 - [GlobalClientsGet200Response](docs/GlobalClientsGet200Response.md)
 - [GlobalConversationsGet200Response](docs/GlobalConversationsGet200Response.md)
 - [GlobalEstimatesGet200Response](docs/GlobalEstimatesGet200Response.md)
 - [GlobalEventAttendancesGet200Response](docs/GlobalEventAttendancesGet200Response.md)
 - [GlobalEventInstancesGet200Response](docs/GlobalEventInstancesGet200Response.md)
 - [GlobalInvoicesGet200Response](docs/GlobalInvoicesGet200Response.md)
 - [GlobalMattersGet200Response](docs/GlobalMattersGet200Response.md)
 - [GlobalPackagesGet200Response](docs/GlobalPackagesGet200Response.md)
 - [GlobalPaymentsGet200Response](docs/GlobalPaymentsGet200Response.md)
 - [GlobalProductsGet200Response](docs/GlobalProductsGet200Response.md)
 - [GlobalScheduledCampaignsGet200Response](docs/GlobalScheduledCampaignsGet200Response.md)
 - [GlobalServicesGet200Response](docs/GlobalServicesGet200Response.md)
 - [GlobalStaffsGet200Response](docs/GlobalStaffsGet200Response.md)
 - [InlineObject](docs/InlineObject.md)
 - [InlineObject1](docs/InlineObject1.md)
 - [InlineObject10](docs/InlineObject10.md)
 - [InlineObject11](docs/InlineObject11.md)
 - [InlineObject12](docs/InlineObject12.md)
 - [InlineObject13](docs/InlineObject13.md)
 - [InlineObject14](docs/InlineObject14.md)
 - [InlineObject15](docs/InlineObject15.md)
 - [InlineObject16](docs/InlineObject16.md)
 - [InlineObject17](docs/InlineObject17.md)
 - [InlineObject18](docs/InlineObject18.md)
 - [InlineObject19](docs/InlineObject19.md)
 - [InlineObject2](docs/InlineObject2.md)
 - [InlineObject20](docs/InlineObject20.md)
 - [InlineObject21](docs/InlineObject21.md)
 - [InlineObject22](docs/InlineObject22.md)
 - [InlineObject23](docs/InlineObject23.md)
 - [InlineObject24](docs/InlineObject24.md)
 - [InlineObject25](docs/InlineObject25.md)
 - [InlineObject26](docs/InlineObject26.md)
 - [InlineObject27](docs/InlineObject27.md)
 - [InlineObject28](docs/InlineObject28.md)
 - [InlineObject29](docs/InlineObject29.md)
 - [InlineObject3](docs/InlineObject3.md)
 - [InlineObject30](docs/InlineObject30.md)
 - [InlineObject31](docs/InlineObject31.md)
 - [InlineObject32](docs/InlineObject32.md)
 - [InlineObject33](docs/InlineObject33.md)
 - [InlineObject4](docs/InlineObject4.md)
 - [InlineObject5](docs/InlineObject5.md)
 - [InlineObject6](docs/InlineObject6.md)
 - [InlineObject7](docs/InlineObject7.md)
 - [InlineObject8](docs/InlineObject8.md)
 - [InlineObject9](docs/InlineObject9.md)
 - [InlineResponse200](docs/InlineResponse200.md)
 - [InlineResponse200Subscriptions](docs/InlineResponse200Subscriptions.md)
 - [Integrations](docs/Integrations.md)
 - [Invoice](docs/Invoice.md)
 - [InvoicesGet201Response](docs/InvoicesGet201Response.md)
 - [InvoicesInvoiceIdGet201Response](docs/InvoicesInvoiceIdGet201Response.md)
 - [InvoicesItems](docs/InvoicesItems.md)
 - [InvoicesPost201Response](docs/InvoicesPost201Response.md)
 - [InvoicesTaxes](docs/InvoicesTaxes.md)
 - [Item](docs/Item.md)
 - [Item1](docs/Item1.md)
 - [LeadgenPost201Response](docs/LeadgenPost201Response.md)
 - [Matter](docs/Matter.md)
 - [Meetings](docs/Meetings.md)
 - [MenuItems](docs/MenuItems.md)
 - [MessagesMessage](docs/MessagesMessage.md)
 - [MessagesPost201Response](docs/MessagesPost201Response.md)
 - [Meta](docs/Meta.md)
 - [ModelPackage](docs/ModelPackage.md)
 - [Others](docs/Others.md)
 - [PathParams](docs/PathParams.md)
 - [Payment](docs/Payment.md)
 - [PaymentCardsSyncCardDatails](docs/PaymentCardsSyncCardDatails.md)
 - [PaymentCardsSyncCardPost201Response](docs/PaymentCardsSyncCardPost201Response.md)
 - [PaymentCheckoutCard](docs/PaymentCheckoutCard.md)
 - [PaymentCheckoutCardCardInfo](docs/PaymentCheckoutCardCardInfo.md)
 - [PaymentCheckoutPut201Response](docs/PaymentCheckoutPut201Response.md)
 - [PaymentCheckoutUrlKeyGet201Response](docs/PaymentCheckoutUrlKeyGet201Response.md)
 - [PaymentClientPackagesPost201Response](docs/PaymentClientPackagesPost201Response.md)
 - [PaymentClientPackagesPost422Response](docs/PaymentClientPackagesPost422Response.md)
 - [PaymentClientPackagesUpdateUsagePost200Response](docs/PaymentClientPackagesUpdateUsagePost200Response.md)
 - [PaymentClientPackagesUpdateUsagePost422Response](docs/PaymentClientPackagesUpdateUsagePost422Response.md)
 - [PaymentPackagesGet200Response](docs/PaymentPackagesGet200Response.md)
 - [PaymentPackagesGet422Response](docs/PaymentPackagesGet422Response.md)
 - [PaymentPackagesItems](docs/PaymentPackagesItems.md)
 - [PaymentPackagesPackageIdGet200Response](docs/PaymentPackagesPackageIdGet200Response.md)
 - [PaymentPackagesPackageIdGet422Response](docs/PaymentPackagesPackageIdGet422Response.md)
 - [PaymentPackagesPackageIdPackage](docs/PaymentPackagesPackageIdPackage.md)
 - [PaymentPackagesPackageIdPackageProducts](docs/PaymentPackagesPackageIdPackageProducts.md)
 - [PaymentPackagesPackageIdPut200Response](docs/PaymentPackagesPackageIdPut200Response.md)
 - [PaymentPackagesPackageIdPut422Response](docs/PaymentPackagesPackageIdPut422Response.md)
 - [PaymentPackagesPost201Response](docs/PaymentPackagesPost201Response.md)
 - [PaymentPackagesPost422Response](docs/PaymentPackagesPost422Response.md)
 - [PaymentPackagesProducts](docs/PaymentPackagesProducts.md)
 - [PaymentPackagesServices](docs/PaymentPackagesServices.md)
 - [PaymentRequest](docs/PaymentRequest.md)
 - [PaymentSettings](docs/PaymentSettings.md)
 - [PaymentSettingsGet201Response](docs/PaymentSettingsGet201Response.md)
 - [PaymentSettingsPaymentSettings](docs/PaymentSettingsPaymentSettings.md)
 - [PaymentSettingsPost201Response](docs/PaymentSettingsPost201Response.md)
 - [PaymentStatus](docs/PaymentStatus.md)
 - [PaymentStatusesIdApplyCouponPut200Response](docs/PaymentStatusesIdApplyCouponPut200Response.md)
 - [PaymentStatusesIdValidateCouponGet200Response](docs/PaymentStatusesIdValidateCouponGet200Response.md)
 - [PaymentsGet201Response](docs/PaymentsGet201Response.md)
 - [PaymentsPaymentUidMatchPost201Response](docs/PaymentsPaymentUidMatchPost201Response.md)
 - [PaymentsPost201Response](docs/PaymentsPost201Response.md)
 - [Plan](docs/Plan.md)
 - [Plugin](docs/Plugin.md)
 - [Policies](docs/Policies.md)
 - [Product](docs/Product.md)
 - [Rescheduling](docs/Rescheduling.md)
 - [SchedulingAppointmentsAppointmentIdGet200Response](docs/SchedulingAppointmentsAppointmentIdGet200Response.md)
 - [SchedulingAppointmentsGet200Response](docs/SchedulingAppointmentsGet200Response.md)
 - [SchedulingBookingsBookingUidUpdateRsvpStatePut200Response](docs/SchedulingBookingsBookingUidUpdateRsvpStatePut200Response.md)
 - [SchedulingBookingsPost201Response](docs/SchedulingBookingsPost201Response.md)
 - [SchedulingEventInstanceEventInstanceIdGet200Response](docs/SchedulingEventInstanceEventInstanceIdGet200Response.md)
 - [SchedulingSchedulingFormsGetFormGet200Response](docs/SchedulingSchedulingFormsGetFormGet200Response.md)
 - [SchedulingStaffGet200Response](docs/SchedulingStaffGet200Response.md)
 - [SchedulingStaffStaffIdGet200Response](docs/SchedulingStaffStaffIdGet200Response.md)
 - [SchedulingWaitlistCancelPut200Response](docs/SchedulingWaitlistCancelPut200Response.md)
 - [SchedulingWaitlistPost201Response](docs/SchedulingWaitlistPost201Response.md)
 - [SchedulingWaitlistPost422Response](docs/SchedulingWaitlistPost422Response.md)
 - [Section](docs/Section.md)
 - [Service](docs/Service.md)
 - [ServicesGet200Response](docs/ServicesGet200Response.md)
 - [ServicesServiceIdAvailabilityGet200Response](docs/ServicesServiceIdAvailabilityGet200Response.md)
 - [ServicesServiceIdGet200Response](docs/ServicesServiceIdGet200Response.md)
 - [SourceData](docs/SourceData.md)
 - [Staff](docs/Staff.md)
 - [Staff1](docs/Staff1.md)
 - [StaffSettings](docs/StaffSettings.md)
 - [StateSummary](docs/StateSummary.md)
 - [Subitem](docs/Subitem.md)
 - [Tax](docs/Tax.md)
 - [Token](docs/Token.md)
 - [TokensGet201Response](docs/TokensGet201Response.md)
 - [TokensPost201Response](docs/TokensPost201Response.md)
 - [TokensRevokePost201Response](docs/TokensRevokePost201Response.md)
 - [Waitlist](docs/Waitlist.md)


## Documentation for Authorization

Authentication schemes defined for the API:
### default

- **Type**: OAuth
- **Flow**: implicit
- **Authorization URL**: https://ignore.myclients.io
- **Scopes**: 
  - platform: All platform related operations
  - platform_create: platform create scope
  - platform_delete: platform delete scope
  - platform_read: platform read scope
  - platform_update: platform update scope


## Recommendation

It's recommended to create an instance of `ApiClient` per thread in a multithreaded environment to avoid any potential issues.

## Author



 readmeEtag: '"34d440ed47f5805d8534b60b7da9c71051cec7e8"' readmeLastModified: Thu, 24 Mar 2022 00:18:56 GMT repositoryId: 462478185 description: null created: '2022-02-22T21:19:45Z' updated: '2022-03-15T19:45:22Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: SimonIT logo: https://avatars.githubusercontent.com/u/11720038?v=4 repoEtag: '"d1805217c3a123078a84771142ee3b36b9fbbd258a16ca5d6e78b0b4ebcd35aa"' repoLastModified: Tue, 15 Mar 2022 19:45:22 GMT foundInMaster: true category: User Interfaces id: 6faee664636bc265bfefb7cb4cf646c1 - source: openapi3 tags repository: https://github.com/bump-sh/bump-ci-example v3: true repositoryMetadata: base64Readme: >- IyBCdW1wIENJIGludGVncmF0aW9uIGV4YW1wbGVzCgpDSSBpbnRlZ3JhdGlvbiBleGFtcGxlcyBmb3IgR2l0aHViIEFjdGlvbiwgVHJhdmlzIENJLCBDaXJjbGVDSSBhbmQgR2l0bGFiIENJLgoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyB3aWR0aD0iMjAlIiBzcmM9Imh0dHBzOi8vYnVtcC5zaC9pY29uLWRlZmF1bHQtbWFza2FibGUtbGFyZ2UucG5nIiAvPgo8L3A+Cgo8cCBhbGlnbj0iY2VudGVyIj4KICA8YSBocmVmPSJodHRwczovL2hlbHAuYnVtcC5zaC8iPkhlbHA8L2E+IHwKICA8YSBocmVmPSJodHRwczovL2J1bXAuc2gvdXNlcnMvc2lnbl91cCI+U2lnbiB1cDwvYT4KPC9wPgoKQnVtcCBpcyBhIENvbnRpbnVvdXMgRG9jdW1lbnRhdGlvbiBQbGF0Zm9ybTogaXQgbGV0cyB5b3Uga2VlcCB5b3VyIEFQSSBkb2MgYWx3YXlzIHN5bmNocm9uaXplZCB3aXRoIHlvdXIgY29kZWJhc2UuIFdpdGggdGhlc2UgQ0kgaW50ZWdyYXRpb24gZXhhbXBsZXMgeW91IGNhbiBhdXRvbWF0aWNhbGx5IGdlbmVyYXRlIHlvdXIgQVBJIHJlZmVyZW5jZSAod2l0aCBjaGFuZ2Vsb2cgYW5kIGRpZmYpIG9uIFtCdW1wXShodHRwczovL2J1bXAuc2gpIGZyb20gYW55IFtPcGVuQVBJXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikgb3IgW0FzeW5jQVBJXShodHRwczovL2dpdGh1Yi5jb20vYXN5bmNhcGkvYXN5bmNhcGkpIGZpbGUuCgojIyBFeGFtcGxlcwoKSGVyZSBhcmUgZXhhbXBsZXMgZm9yIGludGVncmF0aW5nIEJ1bXAgd2l0aCB0aGUgbW9zdCBrbm93biBDSSBwcm9kdWN0czoKLSBDaXJjbGVDSSA6IGh0dHBzOi8vZ2l0aHViLmNvbS9idW1wLXNoL2J1bXAtY2ktZXhhbXBsZS9ibG9iL21hc3Rlci8uY2lyY2xlY2kvY29uZmlnLnltbAotIEdpdGxhYiBDSTogaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvYnVtcC1jaS1leGFtcGxlL2Jsb2IvbWFzdGVyLy5naXRsYWItY2kueW1sCi0gVHJhdmlzIENJOiAgaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvYnVtcC1jaS1leGFtcGxlL2Jsb2IvbWFzdGVyLy50cmF2aXMueW1sCi0gR2l0SHViIGFjdGlvbjogaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvZ2l0aHViLWFjdGlvbgoKVGhlIEdpdEh1YiBhY3Rpb24gZXhhbXBsZSB1c2VzIGEgZGVkaWNhdGVkIGFjdGlvbiB3ZSBjcmFmdGVkIGVzcGVjaWFsbHkgZm9yIHlvdS4gWW91IG1heSBmaW5kIG1vcmUgaW5mb3JtYXRpb24gb24gb3VyIFtHaXRIdWIgbWFya2V0cGxhY2UgcGFnZV0oaHR0cHM6Ly9naXRodWIuY29tL21hcmtldHBsYWNlL2FjdGlvbnMvYXBpLWRvY3VtZW50YXRpb24tb24tYnVtcCkuCgpOb3RlIHRoYXQgaWYgeW91IGRvbid0IHdhbnQgdG8ga2VlcCB0aGUgcHJpdmF0ZSB0b2tlbiBhbmQgZG9jdW1lbnRhdGlvbiBpZCBpbiB5b3VyIGNvZGUgYmFzZSwgeW91IHNob3VsZCB1c2UgZW52aXJvbm1lbnQgdmFyaWFibGVzLiBPdXIgQ0xJIGF1dG9tYXRpY2FsbHkgcmVjb2duaXplcyB0aGVzZSAzIHZhcmlhYmxlczoKLSBgQlVNUF9JRGA6IHlvdXIgZG9jdW1lbnRhdGlvbiBwdWJsaWMgaWQgb3Igc2x1ZwotIGBCVU1QX1RPS0VOYDogeW91ciBkb2N1bWVudGF0aW9uIHByaXZhdGUgdG9rZW4KLSBgQlVNUF9IVUJfSURgOiBpZiB1c2luZyBodWJzLCB5b3VyIGh1YiBwdWJsaWMgaWQgb3Igc2x1ZwoKIyMgQ29udHJpYnV0aW5nCgpCdWcgcmVwb3J0cyBhbmQgcHVsbCByZXF1ZXN0cyBhcmUgd2VsY29tZSBvbiBHaXRIdWIgYXQgaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvYnVtcC1jaS1leGFtcGxlLiBUaGlzIHByb2plY3QgaXMgaW50ZW5kZWQgdG8gYmUgYSBzYWZlLCB3ZWxjb21pbmcgc3BhY2UgZm9yIGNvbGxhYm9yYXRpb24sIGFuZCBjb250cmlidXRvcnMgYXJlIGV4cGVjdGVkIHRvIGFkaGVyZSB0byB0aGUgW0NvbnRyaWJ1dG9yIENvdmVuYW50XShodHRwOi8vY29udHJpYnV0b3ItY292ZW5hbnQub3JnKSBjb2RlIG9mIGNvbmR1Y3QuCgojIyBMaWNlbnNlCgpUaGUgc2NyaXB0cyBhbmQgZG9jdW1lbnRhdGlvbiBpbiB0aGlzIHByb2plY3QgYXJlIHJlbGVhc2VkIHVuZGVyIHRoZSBbTUlUIExpY2Vuc2VdKExJQ0VOU0UpLgoKIyMgQ29kZSBvZiBDb25kdWN0CgpFdmVyeW9uZSBpbnRlcmFjdGluZyBpbiB0aGUgQnVtcCBgYnVtcC1jaS1leGFtcGxlYCBjb2RlIHJlcG9zaXRvcnksIGlzc3VlIHRyYWNrZXJzLCBjaGF0IHJvb21zIGFuZCBtYWlsaW5nIGxpc3RzIGlzIGV4cGVjdGVkIHRvIGZvbGxvdyB0aGUgW2NvZGUgb2YgY29uZHVjdF0oaHR0cHM6Ly9naXRodWIuY29tL2J1bXAtc2gvLmdpdGh1Yi9ibG9iL21hc3Rlci9DT0RFX09GX0NPTkRVQ1QubWQpLgo= readmeEtag: '"59390932c2415cc7858807576a6c29159cdf8e3a"' readmeLastModified: Wed, 26 Mar 2025 14:03:12 GMT repositoryId: 156218433 description: >- CI integration examples for Github Action, Travis CI, CircleCI and Gitlab CI. created: '2018-11-05T12:59:45Z' updated: '2025-03-26T14:03:17Z' language: Shell archived: false stars: 2 watchers: 1 forks: 3 owner: bump-sh logo: https://avatars.githubusercontent.com/u/33217836?v=4 license: MIT repoEtag: '"513637a9ee9545326b241150bc9af84a86ab55f7f0d198395463de0c1bc15dcc"' repoLastModified: Wed, 26 Mar 2025 14:03:17 GMT foundInMaster: true category: Server Implementations id: 3c6b78bc784ecdf4c641738397de0df3 - source: openapi3 tags repository: https://github.com/igrek8/nestjs-typed-responses v3: true id: e85bbe79ca2140b53b166fbbe87e5350 repositoryMetadata: base64Readme: >- IyBbSGFuZGxlIHBvbHltb3JwaGljIHJlc3BvbnNlc10oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9kYXRhLW1vZGVscy9pbmhlcml0YW5jZS1hbmQtcG9seW1vcnBoaXNtKSBpbiBOZXN0LmpzIE9wZW5BUEkKCkVuYWJsZXMgT3BlbkFQSSB2MyBwb2x5bW9ycGhpc20gdXNpbmcgYF9fdHlwZWAgbWV0YWZpZWxkIHRvIHJlc29sdmUgZGlmZmVyZW50IGV4Y2VwdGlvbnMgb3IgcmVzcG9uc2VzIHdpdGhpbiB0aGUgc2FtZSBIVFRQIGNvZGUgcmVzcG9uc2UgZ3JvdXAuCgpbIVtOUE1dKGh0dHBzOi8vYmFkZ2VuLm5ldC9ucG0vdi9uZXN0anMtdHlwZWQtcmVzcG9uc2VzKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL25lc3Rqcy10eXBlZC1yZXNwb25zZXMpClshW0NvdmVyYWdlXShodHRwczovL2NvZGVjb3YuaW8vZ2gvaWdyZWs4L25lc3Rqcy10eXBlZC1yZXNwb25zZXMvYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2lncmVrOC9uZXN0anMtdHlwZWQtcmVzcG9uc2VzKQohW1JlbGVhc2VdKGh0dHBzOi8vYmFkZ2VuLm5ldC9naXRodWIvY2hlY2tzL2lncmVrOC9uZXN0anMtdHlwZWQtcmVzcG9uc2VzKQohW0xpY2Vuc2VdKGh0dHBzOi8vYmFkZ2VuLm5ldC9naXRodWIvbGljZW5zZS9pZ3JlazgvbmVzdGpzLXR5cGVkLXJlc3BvbnNlcykKCiMjIEluc3RhbGxhdGlvbgoKYGBgYmFzaApucG0gaW5zdGFsbCAtLXNhdmUgbmVzdGpzLXR5cGVkLXJlc3BvbnNlcwoKeWFybiBhZGQgbmVzdGpzLXR5cGVkLXJlc3BvbnNlcwpgYGAKCiMjIE9wZW5BUEkKCiFbU3dhZ2dlcl0oLi9tZWRpYS9zd2FnZ2VyLnBuZykKCiMjIFVzYWdlCgpgYGB0cwppbXBvcnQgeyBDb25zb2xlTG9nZ2VyLCBDb250cm9sbGVyLCBIdHRwQ29kZSwgSHR0cFN0YXR1cywgTW9kdWxlLCBQb3N0LCBWYWxpZGF0aW9uUGlwZSB9IGZyb20gJ0BuZXN0anMvY29tbW9uJzsKaW1wb3J0IHsgQVBQX1BJUEUsIE5lc3RGYWN0b3J5IH0gZnJvbSAnQG5lc3Rqcy9jb3JlJzsKaW1wb3J0IHsKICBBcGlCYWRSZXF1ZXN0UmVzcG9uc2UsCiAgQXBpRXh0cmFNb2RlbHMsCiAgQXBpRm9yYmlkZGVuUmVzcG9uc2UsCiAgQXBpT2tSZXNwb25zZSwKICBBcGlQcm9wZXJ0eSwKICBEb2N1bWVudEJ1aWxkZXIsCiAgcmVmcywKICBTd2FnZ2VyTW9kdWxlLAp9IGZyb20gJ0BuZXN0anMvc3dhZ2dlcic7CmltcG9ydCB7IEV4cG9zZSB9IGZyb20gJ2NsYXNzLXRyYW5zZm9ybWVyJzsKaW1wb3J0IHsgSXNEYXRlU3RyaW5nLCBJc1N0cmluZyB9IGZyb20gJ2NsYXNzLXZhbGlkYXRvcic7CmltcG9ydCB7CiAgQXBpVHlwZU1ldGFmaWVsZCwKICBCYWRSZXF1ZXN0RXhjZXB0aW9uLAogIFNlcnZpY2VVbmF2YWlsYWJsZUV4Y2VwdGlvbiwKICBUeXBlZERhdGFUcmFuc2Zlck9iamVjdCwKICBUeXBlZFJlc3BvbnNlTW9kdWxlLAogIFZhbGlkYXRpb25FeGNlcHRpb24sCn0gZnJvbSAnbmVzdGpzLXR5cGVkLXJlc3BvbnNlcyc7CgpjbGFzcyBNeVJlc3BvbnNlIGltcGxlbWVudHMgVHlwZWREYXRhVHJhbnNmZXJPYmplY3QgewogIC8vIEFwaVR5cGVNZXRhZmllbGQgbXVzdCBiZSBkZWZpbmVkIHRvIHNwZWNpZnkgc2VyaWFsaXphYmxlIHR5cGUKICBAQXBpVHlwZU1ldGFmaWVsZCgnTXlSZXNwb25zZScpCiAgX190eXBlID0gJ015UmVzcG9uc2UnOwoKICAvLyBJdCBpcyBub3QgcmVxdWlyZWQgdG8gYXBwbHkgQEV4cG9zZSBkZWNvcmF0b3IgKHBlciBjb25maWd1cmF0aW9uKQogIEBBcGlQcm9wZXJ0eSgpCiAgQElzU3RyaW5nKCkKICBkYXRhITogc3RyaW5nOwoKICBjb25zdHJ1Y3Rvcihwcm9wczogT21pdDxNeVJlc3BvbnNlLCAnX190eXBlJz4pIHsKICAgIE9iamVjdC5hc3NpZ24odGhpcywgcHJvcHMpOwogIH0KfQoKY2xhc3MgTWFpbnRlbmFuY2VFeGNlcHRpb24gZXh0ZW5kcyBTZXJ2aWNlVW5hdmFpbGFibGVFeGNlcHRpb24gaW1wbGVtZW50cyBUeXBlZERhdGFUcmFuc2Zlck9iamVjdCB7CiAgLy8gQXBpVHlwZU1ldGFmaWVsZCBtdXN0IGJlIGRlZmluZWQgdG8gc3BlY2lmeSBzZXJpYWxpemFibGUgdHlwZQogIEBBcGlUeXBlTWV0YWZpZWxkKCdNYWludGVuYW5jZUV4Y2VwdGlvbicpCiAgb3ZlcnJpZGUgX190eXBlID0gJ01haW50ZW5hbmNlRXhjZXB0aW9uJzsKCiAgLy8gQEV4cG9zZSBtdXN0IGJlIHVzZWQgZXhwbGljaXRseSBpbiBlcnJvciBjbGFzc2VzIGFzIG9ubHkgc3VjaCBmaWVsZHMgd2lsbCBiZSBleHBvc2VkCiAgQEV4cG9zZSgpCiAgQEFwaVByb3BlcnR5KHsKICAgIGV4YW1wbGU6ICcyMDIyLTEyLTMxVDExOjAwOjAwLjAwMFonLAogICAgZGVzY3JpcHRpb246ICdUaGVyZSBpcyBhIG1haW50ZW5hbmNlIG9uIHRoZSBzZXJ2ZXInLAogIH0pCiAgQElzRGF0ZVN0cmluZygpCiAgb3BlcmF0aW9uYWxBdCE6IERhdGU7Cn0KCkBDb250cm9sbGVyKCkKQEFwaUV4dHJhTW9kZWxzKFZhbGlkYXRpb25FeGNlcHRpb24sIEJhZFJlcXVlc3RFeGNlcHRpb24pCmNsYXNzIEFwcENvbnRyb2xsZXIgewogIEBQb3N0KCkKICBASHR0cENvZGUoSHR0cFN0YXR1cy5PSykKICBAQXBpT2tSZXNwb25zZSh7IHR5cGU6IE15UmVzcG9uc2UgfSkKICBAQXBpRm9yYmlkZGVuUmVzcG9uc2UoeyB0eXBlOiBNYWludGVuYW5jZUV4Y2VwdGlvbiB9KQogIC8vIFBvbHltb3JwaGlzbSBodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2RhdGEtbW9kZWxzL2luaGVyaXRhbmNlLWFuZC1wb2x5bW9ycGhpc20vCiAgQEFwaUJhZFJlcXVlc3RSZXNwb25zZSh7IHNjaGVtYTogeyBvbmVPZjogcmVmcyhWYWxpZGF0aW9uRXhjZXB0aW9uLCBCYWRSZXF1ZXN0RXhjZXB0aW9uKSB9IH0pCiAgZGVtbygpOiBNeVJlc3BvbnNlIHsKICAgIHJldHVybiBuZXcgTXlSZXNwb25zZSh7CiAgICAgIGRhdGE6ICdIZWxsbyEnLAogICAgfSk7CiAgfQp9CgpATW9kdWxlKHsKICBwcm92aWRlcnM6IFsKICAgIHsKICAgICAgcHJvdmlkZTogQVBQX1BJUEUsCiAgICAgIHVzZVZhbHVlOiBuZXcgVmFsaWRhdGlvblBpcGUoewogICAgICAgIGV4Y2VwdGlvbkZhY3Rvcnk6IFZhbGlkYXRpb25FeGNlcHRpb24uZXhjZXB0aW9uRmFjdG9yeSwKICAgICAgfSksCiAgICB9LAogIF0sCiAgaW1wb3J0czogW1R5cGVkUmVzcG9uc2VNb2R1bGVdLAogIGNvbnRyb2xsZXJzOiBbQXBwQ29udHJvbGxlcl0sCn0pCmNsYXNzIEFwcE1vZHVsZSB7fQoKYXN5bmMgZnVuY3Rpb24gYm9vdHN0cmFwKCkgewogIGNvbnN0IGFwcCA9IGF3YWl0IE5lc3RGYWN0b3J5LmNyZWF0ZShBcHBNb2R1bGUpOwogIGNvbnN0IGNvbmZpZyA9IG5ldyBEb2N1bWVudEJ1aWxkZXIoKS5idWlsZCgpOwogIGNvbnN0IGRvY3VtZW50ID0gU3dhZ2dlck1vZHVsZS5jcmVhdGVEb2N1bWVudChhcHAsIGNvbmZpZyk7CiAgU3dhZ2dlck1vZHVsZS5zZXR1cCgnLycsIGFwcCwgZG9jdW1lbnQpOwogIGF3YWl0IGFwcC5saXN0ZW4oMzAwMCk7Cn0KYm9vdHN0cmFwKCk7CmBgYAo= readmeEtag: '"a2ea10f602b6371a8fbd3c0eeca1a1e87dd795ab"' readmeLastModified: Thu, 01 Feb 2024 21:04:40 GMT repositoryId: 552389104 description: >- Allows implementation of polymorphism in OAS and exports nest.js exceptions as swagger schemes created: '2022-10-16T13:39:35Z' updated: '2023-06-15T13:09:53Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 0 owner: igrek8 logo: https://avatars.githubusercontent.com/u/7078731?v=4 license: MIT repoEtag: '"00489427455eabb5010a77df2b5d6f382cfa9fe7138958784b6dde3324671f4f"' repoLastModified: Thu, 15 Jun 2023 13:09:53 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/masterbomb/api v3: true repositoryMetadata: base64Readme: >- IyBNYXN0ZXJCb20gQVBJDQpbIVtjaV0oaHR0cHM6Ly9naXRodWIuY29tL01hc3RlcmJvbWIvYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vTWFzdGVyYm9tYi9hcGkvYWN0aW9ucy93b3JrZmxvd3MvY2kueW1sKQ0KWyFbaW1hZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9NYXN0ZXJib21iL2FwaS9hY3Rpb25zL3dvcmtmbG93cy9pbWFnZS55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL01hc3RlcmJvbWIvYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2ltYWdlLnltbCkNCjxwIGFsaWduPSJjZW50ZXIiPg0KICA8aW1nIHNyYz0iL2RvY3MvaW1nL01CLnBuZyI+PC9pbWc+DQo8L3A+DQoNCk1vZGlmaWVkOiAyMDIxLTExDQoNCiMjIE5hdmlnYXRpb24NCjEuIFtEb2N1bWVudGF0aW9uXSguL2RvY3MvUkVBRE1FLm1kKQ0KMi4gW0xpY2Vuc2VdKCNsaWNlbnNlKSANCg0KIyMgTGljZW5zZQ0KVGhpcyByZXBvc2l0b3J5IGlzIGxpY2Vuc2VkIGJ5IEdQTHYzLiBTZWUgdGhlIGZ1bGwgbGljZW5zZSBbaGVyZV0oTElDRU5TRSk= readmeEtag: '"838d63bb1054c8ed4bedccbe8ffe9080e58a0769"' readmeLastModified: Fri, 21 Jan 2022 23:29:26 GMT repositoryId: 362316411 description: Public-facing RESTful API created: '2021-04-28T02:42:30Z' updated: '2024-06-04T23:56:50Z' language: TypeScript archived: false stars: 2 watchers: 0 forks: 0 owner: Masterbomb logo: https://avatars.githubusercontent.com/u/83322983?v=4 license: GPL-3.0 repoEtag: '"af6c29c8479b90f0db4af64e91640d65aa211e97f6c5bb8edbdc9fe8630549a2"' repoLastModified: Tue, 04 Jun 2024 23:56:50 GMT foundInMaster: true category: - Server - Server Implementations id: 2a84fcc559645de77dd307ea57700808 - source: openapi3 tags repository: https://github.com/aquanest/atmos-go v3: true repositoryMetadata: base64Readme: >- IyBhdG1vcy1nbwoKKipETyBOT1QgRURJVCBUSElTIENPREUgTUFOVUFMTFkhKioKCmF0bW9zLWdvIGlzICoqVW5vZmZpY2lhbCoqIEdvIGNsaWVudCB0byB1c2UgW0FUTU9TIFBsYXRmb3JtIEFQSV0oaHR0cHM6Ly93d3cuYXRtb3MuYXBwLykuCgpUaGUgY29kZSB3YXMgZ2VuZXJhdGVkIGF1dG9tYXRpY2FsbHkgYnkgdXNpbmcgW2F0bW9zLW9wZW5hcGktc3JjXShodHRwczovL2dpdGh1Yi5jb20vdW1hdGFyZTUvYXRtb3Mtb3BlbmFwaS1zcmMpLgoKIyMgVXNhZ2UKCmBgYHNoCmdvIGdldCBnaXRodWIuY29tL3VtYXRhcmU1L2F0bW9zLWdvCmBgYAo= readmeEtag: '"66aa1de25effe71335948b817a092afd55e290af"' readmeLastModified: Wed, 26 Jan 2022 12:35:17 GMT repositoryId: 452261817 description: Unofficial client library to use ATMOS Platform API created: '2022-01-26T12:12:15Z' updated: '2025-08-03T07:04:05Z' language: null archived: false stars: 2 watchers: 1 forks: 0 owner: aquanest logo: https://avatars.githubusercontent.com/u/88537514?v=4 repoEtag: '"3e4350aaa0d5b1de932f8b913f0d075b2d072c1f0876e91f020332b78a0d2a2c"' repoLastModified: Sun, 03 Aug 2025 07:04:05 GMT foundInMaster: true category: - SDK - Code Generators id: d91d20d59028e95de5102cd1f2eefe7f oldLocations: - https://github.com/umatare5/atmos-go - source: openapi3 tags repository: https://github.com/cbetta/json-schema-sensitivity-checker v3: true repositoryMetadata: base64Readme: >- IyBqc29uLXNjaGVtYS1zZW5zaXRpdml0eS1jaGVja2VyCgpbIVtucG0KdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL2pzb24tc2NoZW1hLXNlbnNpdGl2aXR5LWNoZWNrZXIuc3ZnKV0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL2pzb24tc2NoZW1hLXNlbnNpdGl2aXR5LWNoZWNrZXIpCiFbY2kgc3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vY2JldHRhL2pzb24tc2NoZW1hLXNlbnNpdGl2aXR5LWNoZWNrZXIvd29ya2Zsb3dzL05vZGUlMjBDSS9iYWRnZS5zdmcpCgpUaGlzIENMSSBhbGxvd3MgeW91IHRvIHByb3ZpZGUgYSBKU09OUGF0aCBleHByZXNzaW9uIGFuZCBydW4KW2BhbGV4YF0oaHR0cHM6Ly9hbGV4anMuY29tKSBhZ2FpbnN0IGFueSBtYXRjaGluZyBsaW5lcy4KCj4gVGhpcyB0b29sIGlzIGN1cnJlbnRseSBpbiBhbHBoYSBhbmQgaGFzIHByaW1hcmlseSBiZWVuIHRlc3RlZCBhZ2FpbnN0IE9wZW5BUEkKPiBmaWxlcy4gUFJzIGFyZSB3ZWxjb21lIQoKVGhpcyBwcm9qZWN0IGlzIHZlcnkgbXVjaCBpbnNwaXJlZCBieSBhbmQgYmFzZWQgb24gdGhlIHdvcmsgZG9uZSBvbiB0aGUKW2Bqc29uLXNjaGVtYS1zcGVsbC1jaGVja2VyYF0oaHR0cHM6Ly9naXRodWIuY29tL21oZWFwL2pzb24tc2NoZW1hLXNwZWxsLWNoZWNrZXIpCmJ5IE1pY2hhZWwgSGVhcC4KCiMjIEluc3RhbGxhdGlvbgoKYGBgYmFzaApucG0gaW5zdGFsbCAtZyBqc29uLXNjaGVtYS1zZW5zaXRpdml0eS1jaGVja2VyCmBgYAoKIyMgVXNhZ2UKCkNoZWNrIGFnYWluc3Qgc3BlY2lmaWMgZmllbGQgbmFtZXMgYXQgYW55IGRlcHRoOgoKYGBgYmFzaApqc29uLXNjaGVtYS1zZW5zaXRpdml0eS1jaGVja2VyIC1mICdkZXNjcmlwdGlvbix0aXRsZScgcGF0aC90by9vcGVuYXBpLmpzb24KYGBgCgpBbHRlcm5hdGl2ZWx5LCB5b3UgY2FuIHNwZWNpZnkgYSBKU09OUGF0aCBleHByZXNzaW9uIHlvdXJzZWxmLgoKYGBgYmFzaApqc29uLXNjaGVtYS1zZW5zaXRpdml0eS1jaGVja2VyIC1qICckLi5bImRlc2NyaXB0aW9uIiwidGl0bGUiXScgcGF0aC90by9vcGVuYXBpLmpzb24KYGBgCgojIyBPcHRpb25zCgpgYGBiYXNoClVzYWdlOiBiaW4gW29wdGlvbnNdIHNvdXJjZS1maWxlCgpPcHRpb25zOgogIC1WLCAtLXZlcnNpb24gICAgICAgICAgIG91dHB1dCB0aGUgdmVyc2lvbiBudW1iZXIKICAtdCwgLS10ZXh0ICAgICAgICAgICAgICB0cmVhdCBpbnB1dCBhcyBwbGFpbi10ZXh0IChub3QgbWFya2Rvd24pCiAgLWwsIC0taHRtbCAgICAgICAgICAgICAgdHJlYXQgaW5wdXQgYXMgaHRtbCAobm90IG1hcmtkb3duKQogIC1kLCAtLWRpZmYgICAgICAgICAgICAgIGlnbm9yZSB1bmNoYW5nZWQgbGluZXMgKGFmZmVjdHMgVHJhdmlzIG9ubHkpCiAgLWosIC0tanNvbi1wYXRoIFtwYXRoXSAgc3BlY2lmeSBhIGpzb25wYXRoIGV4cHJlc3Npb24gdG8gbWF0Y2gKICAtYywgLS1jb25maWcgW3BhdGhdICAgICBzcGVjaWZ5IGEgSlNPTiBmb3JtYXR0ZWQgQWxleCBjb25maWcgdG8gcGFzcyB0byBldmVyeSBtYXRjaAogIC1mLCAtLWZpZWxkcyBbZmllbGRzXSAgIHNwZWNpZnkgYSBjb21tYSBzZXBhcmF0ZWQgbApgYGAK readmeEtag: '"1d5e04c2421f4a435d41773bb9a428301645aa57"' readmeLastModified: Fri, 17 Nov 2023 13:19:57 GMT repositoryId: 227859847 description: Check the sensitivity of your JSON Schema (including OpenAPI!) documents created: '2019-12-13T14:41:20Z' updated: '2025-08-27T12:21:51Z' language: JavaScript archived: false stars: 3 watchers: 1 forks: 0 owner: cbetta logo: https://avatars.githubusercontent.com/u/7718?v=4 repoEtag: '"029de96c1ad6a474f8497d56d418a5f495fef042561d31d9a0794d38c4873dff"' repoLastModified: Wed, 27 Aug 2025 12:21:51 GMT foundInMaster: true category: Parsers id: ca8b847aee3149a89528c19339c49609 - source: openapi3 tags repository: https://github.com/protung/open-api-generator v3: true id: dad021a6042570a6ec3443b27243924c repositoryMetadata: base64Readme: >- T3BlbiBBcGkgR2VuZXJhdG9yCj09PT09PT09PT09PT09PT09PQoKWyFbQnVpbGRdKGh0dHBzOi8vZ2l0aHViLmNvbS9wcm90dW5nL29wZW4tYXBpLWdlbmVyYXRvci93b3JrZmxvd3MvQnVpbGQvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3Byb3R1bmcvb3Blbi1hcGktZ2VuZXJhdG9yL2FjdGlvbnM/cXVlcnk9d29ya2Zsb3clM0FCdWlsZCticmFuY2glM0FtYWluKQpbIVtTb2Z0d2FyZSBMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2xpY2Vuc2UtTUlULWJyaWdodGdyZWVuLnN2ZyldKExJQ0VOU0UubWQpCgojIyBJbnN0YWxsYXRpb24KClJlcXVpcmUgdXNpbmcgY29tcG9zZXI6CgpgYGBzaGVsbAokIGNvbXBvc2VyIHJlcXVpcmUgcHJvdHVuZy9vcGVuLWFwaS1nZW5lcmF0b3IKYGBgCgojIyBMaWNlbnNlCgpUaGlzIHBhY2thZ2UgaXMgcmVsZWFzZWQgdW5kZXIgdGhlIFtNSVQgbGljZW5zZV0oTElDRU5TRS5tZCkuCg== readmeEtag: '"f4eff9d60bd0d86002e7a1209b4d4f5f613291be"' readmeLastModified: Sat, 10 Aug 2024 05:37:59 GMT repositoryId: 265488630 description: Library to generate the OpenApi specification in PHP created: '2020-05-20T07:41:03Z' updated: '2026-02-04T14:28:32Z' language: PHP archived: false stars: 3 watchers: 2 forks: 6 owner: protung logo: https://avatars.githubusercontent.com/u/81255167?v=4 license: MIT repoEtag: '"75d9bc88398e98504bbe4e5a52f9c6b830b0b48ba816e767a439496c80b37562"' repoLastModified: Wed, 04 Feb 2026 14:28:32 GMT category: - Server - Parsers foundInMaster: true oldLocations: - https://github.com/speicher210/open-api-generator - source: openapi3 tags repository: https://github.com/alexbernardi360/guitar-composer v3: true repositoryMetadata: base64Readme: >- IyBndWl0YXItY29tcG9zZXIKaHR0cHM6Ly9ndWl0YXItY29tcG9zZXIuaGVyb2t1YXBwLmNvbS8KClJFU1RmdWwgQVBJIHBlciBjaGl0YXJyaXN0aSwgcGVybWV0dGUgZGkgYWdnaXVuZ2VyZSwgbW9kaWZpY2FyZSBlIGNlcmNhcmUgdGVzdGkgZGkgY2Fuem9uaSBjb24gYWNjb3JkaS4KSSBkYXRpIHJpZ3VhcmRhbnRpIGxlIGNhbnpvbmkgdmVuZ29ubyBhZ2dpdW50aSBkYWdsaSB1dGVudGkuClZlbmdvbm8gaW52aWF0aSBkYXRpIHVsdGVyaW9yaSBvdHRlbnV0aSBkYSBbVGhlQXVkaW9EQl0oaHR0cDovL3d3dy50aGVhdWRpb2RiLmNvbS8gIlRoZUF1ZGlvREIiKS4KCi0tLS0tLS0tLS0tLQoKUHJvZ2V0dG8gZCdlc2FtZSBwZXIgaWwgY29yc28gW1BJQVRUQUZPUk1FIERJR0lUQUxJIFBFUiBMQSBHRVNUSU9ORSBERUwgVEVSUklUT1JJT10oaHR0cHM6Ly93d3cudW5pdXJiLml0L2luc2VnbmFtZW50aS1lLXByb2dyYW1taS8yNTU1NzcgIlBJQVRUQUZPUk1FIERJR0lUQUxJIFBFUiBMQSBHRVNUSU9ORSBERUwgVEVSUklUT1JJTyIpIGRlbGwnVW5pdmVyc2l0w6AgZGVnbGkgU3R1ZGkgZGkgVXJiaW5vIENhcmxvIEJvLlwKU3ZpbHVwcGF0byBkYSAqKkFsZXNzYW5kcm8gQmVybmFyZGkqKiwgbWF0cmljb2xhOiAyODQ5NjguCgotLS0tLS0tLS0tLS0KCi0gW0RvY3VtZW50YXppb25lIEFQSV0oaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcy1kb2NzL2FsZXhiZXJuYXJkaTM2MC9ndWl0YXItY29tcG9zZXIvMS4wICJEb2N1bWVudGF6aW9uZSBBUEkiKQoKLSBbU2NlbHRlIGltcGxlbWVudGF0aXZlXSguL2RvY3VtZW50YXRpb25faXRhL0NIT0lDRVMubWQgIlNjZWx0ZSBpbXBsZW1lbnRhdGl2ZSIpCgotIFtTZXJ2aXppIGVzdGVybmldKC4vZG9jdW1lbnRhdGlvbl9pdGEvRVhURVJOQUxTRVJWSUNFUy5tZCAiU2Vydml6aSBlc3Rlcm5pIikKCi0gW0RlcGxveV0oLi9kb2N1bWVudGF0aW9uX2l0YS9ERVBMT1kubWQgIkRlcGxveSIpCgotIFtHdWlkYSBhbGwndXRpbGl6em9dKC4vZG9jdW1lbnRhdGlvbl9pdGEvR1VJREUubWQgIkd1aWRhIGFsbCd1dGlsaXp6byIpCgo= readmeEtag: '"ca48043484c2ef5a0b7979f7b1bf1bbaa48f3843"' readmeLastModified: Fri, 12 Jun 2020 17:45:10 GMT repositoryId: 255621056 description: >- An openAPI for guitarists: add, edit and search lyrics and chords of your favorite songs. created: '2020-04-14T13:46:04Z' updated: '2020-06-12T17:45:40Z' language: HTML archived: false stars: 2 watchers: 1 forks: 0 owner: alexbernardi360 logo: https://avatars.githubusercontent.com/u/26739236?v=4 license: MIT repoEtag: '"a6cf96438e19d13720a9c2cbe349fce6c4dcda0fd86bf4f1a64d5c1680652132"' repoLastModified: Fri, 12 Jun 2020 17:45:40 GMT foundInMaster: true category: Mock id: 95448c995e4658a9e2b08a21299df7f6 - source: openapi3 tags repository: https://github.com/kasdihacene/global-azure-2022 v3: true id: 8b83f347681bd99eafa353c51ecf0a3e repositoryMetadata: base64Readme: >- IyMgRW5kLXRvLWVuZCBhcHBsaWNhdGlvbiBtb25pdG9yaW5nIHdpdGggQXp1cmUgQXBwIEluc2lnaHRzIDpyb2NrZXQ6CgohW0VuZC10by1lbmRfYXBwbGljYXRpb25fbW9uaXRvcmluZ193aXRoX0F6dXJlX0FwcF9JbnNpZ2h0c10oYXNzZXRzL0VuZC10by1lbmRfYXBwbGljYXRpb25fbW9uaXRvcmluZ193aXRoX0F6dXJlX0FwcF9JbnNpZ2h0cy5qcGVnKQoKClRoZSBwdXJwb3NlIG9mIHRoaXMgcmVwb3NpdG9yeSBpcyB0byBzaG93IHlvdSwgaG93IGRvIHdlIG1vbml0b3IgdGhlIEphdmEgYXBwbGljYXRpb25zIGRlcGxveWVkIG9uIEF6dXJlIEt1YmVybmV0ZXMgU2VydmljZSwgdXNpbmcgYSBmZWF0dXJlIG9mIEF6dXJlIE1vbml0b3IgdGhhdCBwcm92aWRlcyBleHRlbnNpYmxlIGFwcGxpY2F0aW9uIHBlcmZvcm1hbmNlIG1hbmFnZW1lbnQgKEFQTSkgYW5kIG1vbml0b3JpbmcgZm9yIGxpdmUgd2ViIGFwcHMuIERldmVsb3BlcnMgYW5kIERldk9wcyBwcm9mZXNzaW9uYWxzIGNhbiB1c2UgQXBwbGljYXRpb24gSW5zaWdodHMgdG86CgrinIUgQXV0b21hdGljYWxseSBkZXRlY3QgcGVyZm9ybWFuY2UgYW5vbWFsaWVzLgoK4pyFIEhlbHAgZGlhZ25vc2UgaXNzdWVzIGJ5IHVzaW5nIHBvd2VyZnVsIGFuYWx5dGljcyB0b29scy4KCuKchSBTZWUgd2hhdCB1c2VycyBhY3R1YWxseSBkbyB3aXRoIGFwcHMuCgrinIUgSGVscCBjb250aW51b3VzbHkgaW1wcm92ZSBhcHAgcGVyZm9ybWFuY2UgYW5kIHVzYWJpbGl0eS4KClNpbmNlICoqTm92ZW1iZXIgMjAyMCoqLCB1c2luZyBKYXZhIFNESyBpcyBubyBtb3JlIG5lZWRlZCB0byBpbnN0cnVtZW50IGphdmEgYXBwbGljYXRpb25zIGFuZCB0byBjb2xsZWN0IGFwcGxpY2F0aW9uIAp0ZWxlbWV0cnkuIFRoZSBzb2x1dGlvbiByZWNvbW1lbmRlZCBieSBNaWNyb3NvZnQgaXMgYW4gKiphdXRvLWluc3RydW1lbnRhdGlvbioqIHVzaW5nIDMuMCBhZ2VudCB3aGljaCB3aWxsIAp0cmFjayBhbmQgY29ycmVsYXRlIHRoZSBhcHBsaWNhdGlvbi4KCiMjIyBUZWNobmljYWwgc3RhY2sgOmJ1aWxkaW5nX2NvbnN0cnVjdGlvbjoKClRoZSBzdGFjayB1c2VkIG9uIHRoaXMgdHV0b3JpYWwgaXMgOgoKLSBTcHJpbmdCb290IDIuNi54Ci0gSmF2YSAxNwotIE9wZW5BUEkgMy4wIOKAlCBDb250cmFjdC1GSVJTVCAoRG9jdW1lbnRpbmcgdGhlIEFQSSkKLSBUZXJyYWZvcm0KLSBLdWJlcm5ldGVzIGFuZCBIZWxtIGNoYXJ0CgojIyMgSW5zdGFsbGluZyBkZXYgdG9vbHMg8J+SvAoKRm9yIHRoaXMgd29ya3Nob3Agd2UgbmVlZCBzb21lIHRvb2xzIHRvIGludGVyYWN0IHdpdGggdGhlIEF6dXJlIGFjY291bnQgYGF6IGNsaWAsIEt1YmVybmV0ZXMgc2VydmljZXMgKGBrdWJlY3RsYCksIG1heWJlIHRoZSB0b29sIGZvciBJYWFDIHRvIHByb3Zpc2lvbiBhIHJlc291cmNlcyAoYFRlcnJhZm9ybWApCgoxLSBBenVyZSBDbGk6CgogICAgV2luZG93czogaHR0cHM6Ly9kb2NzLm1pY3Jvc29mdC5jb20vZW4tdXMvY2xpL2F6dXJlL2luc3RhbGwtYXp1cmUtY2xpLXdpbmRvd3M/dGFicz1henVyZS1jbGkKICAgIExpbnV4OiBodHRwczovL2RvY3MubWljcm9zb2Z0LmNvbS9mci1mci9jbGkvYXp1cmUvaW5zdGFsbC1henVyZS1jbGktbGludXg/cGl2b3RzPWFwdAogICAgTWFjb3M6IGh0dHBzOi8vZG9jcy5taWNyb3NvZnQuY29tL2ZyLWZyL2NsaS9henVyZS9pbnN0YWxsLWF6dXJlLWNsaS1tYWNvcwoKQ2hlY2sgOiAkIGF6IC12IG9yIGF6IC0tdmVyc2lvbgoKMi0ga3ViZWN0bCAobm90IGt1YmVsZXQpIDoKCiAgICBXaW5kb3dzOiBodHRwczovL2t1YmVybmV0ZXMuaW8vZnIvZG9jcy90YXNrcy90b29scy9pbnN0YWxsLWt1YmVjdGwvI2luc3RhbGxlci1rdWJlY3RsLXN1ci13aW5kb3dzCiAgICBMaW51eDogaHR0cHM6Ly9rdWJlcm5ldGVzLmlvL2ZyL2RvY3MvdGFza3MvdG9vbHMvaW5zdGFsbC1rdWJlY3RsLyNpbnN0YWxsZXIta3ViZWN0bC1zdXItbGludXgKICAgIE1hY29zOiBodHRwczovL2t1YmVybmV0ZXMuaW8vZnIvZG9jcy90YXNrcy90b29scy9pbnN0YWxsLWt1YmVjdGwvI2luc3RhbGxlci1hdmVjLWhvbWVicmV3LXN1ci1tYWNvcwoKQ2hlY2sgOiAkIGt1YmVjdGwgdmVyc2lvbgoKMy0gdGVycmFmb3JtIAogICAgCiAgICBBbGwgZW52aXJvbm1lbnRzOiBodHRwczovL3d3dy50ZXJyYWZvcm0uaW8vZG93bmxvYWRzLmh0bWwKCkNoZWNrIDogJCB0ZXJyYWZvcm0gLXYgb3IgdGVycmFmb3JtIC0tdmVyc2lvbgoKNC0gWW91IG5lZWQgYWxzbyBoZWxtIGNsaSAoSWYgeW91IHdhbnQgdG8gZGVwbG95IGRpcmVjdGx5IHRoZSBjaGFydHMgd2l0aG91dCB1c2luZyB0ZXJyYWZvcm0gLSBoZWxtX3JlbGVhc2UgcmVzb3VyY2Up readmeEtag: '"3bc943ec502b6bb2c2c6399270b3ba37bb88d299"' readmeLastModified: Sun, 01 May 2022 14:11:45 GMT repositoryId: 479689958 description: >- Communities around the world are organizing localized live streams for everyone around the world to join and learn about Azure from the best-in-class community leaders. created: '2022-04-09T10:16:15Z' updated: '2022-05-16T12:46:37Z' language: Java archived: false stars: 2 watchers: 2 forks: 0 owner: kasdihacene logo: https://avatars.githubusercontent.com/u/25183892?v=4 repoEtag: '"c2add7ce962d55754d3d61431d2064e4a99026f5f223b80c4d0c863418e98a77"' repoLastModified: Mon, 16 May 2022 12:46:37 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/gabrielmaialva33/soc-ex-api v3: true id: 39a8671b7b22dece20e934ddcda828bb repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgogIDxicj4KICA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2dhYnJpZWxtYWlhbHZhMzMvc29jLWV4LWFwaS9tYXN0ZXIvLmdpdGh1Yi9hc3NldHMvcG90aW9uLnBuZyIgYWx0PSJTb2MgRXggQXBpIiB3aWR0aD0iMjAwIj4KICA8YnI+CiAgQSBTb2NpYWwgTmV0d29yayBpbiA8YSBocmVmPSJodHRwczovL3d3dy5waG9lbml4ZnJhbWV3b3JrLm9yZy8iPlBob2VuaXg8L2E+CiAgPGJyPgo8L2gxPgoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyBzcmM9Imh0dHBzOi8vd2FrYXRpbWUuY29tL2JhZGdlL3VzZXIvZTYxODQyZDAtYzU4OC00NTg2LTk2YTMtZjA0NDhhNDM0YmU0L3Byb2plY3QvZjlmZTZmMzUtZTQyNS00MDZiLWFkM2QtNDYwZGE4MDg1OGY4LnN2ZyIgYWx0PSJ3YWthdGltZSI+CiAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhbmd1YWdlcy90b3AvZ2FicmllbG1haWFsdmEzMy9zb2MtZXgtYXBpP3N0eWxlPWZsYXQmbG9nbz1hcHB2ZXlvciIgYWx0PSJHaXRIdWIgdG9wIGxhbmd1YWdlIiA+CiAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhbmd1YWdlcy9jb3VudC9nYWJyaWVsbWFpYWx2YTMzL3NvYy1leC1hcGk/c3R5bGU9ZmxhdCZsb2dvPWFwcHZleW9yIiBhbHQ9IkdpdEh1YiBsYW5ndWFnZSBjb3VudCIgPgogIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9yZXBvLXNpemUvZ2FicmllbG1haWFsdmEzMy9zb2MtZXgtYXBpP3N0eWxlPWZsYXQmbG9nbz1hcHB2ZXlvciIgYWx0PSJSZXBvc2l0b3J5IHNpemUiID4KICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9nYWJyaWVsbWFpYWx2YTMzL3NvYy1leC1hcGk/Y29sb3I9MDBiOGQzP3N0eWxlPWZsYXQmbG9nbz1hcHB2ZXlvciIgYWx0PSJMaWNlbnNlIiAvPiAKICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vZ2FicmllbG1haWFsdmEzMy9zb2MtZXgtYXBpL2NvbW1pdHMvbWFzdGVyIj4KICAgIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9sYXN0LWNvbW1pdC9nYWJyaWVsbWFpYWx2YTMzL3NvYy1leC1hcGk/c3R5bGU9ZmxhdCZsb2dvPWFwcHZleW9yIiBhbHQ9IkdpdEh1YiBsYXN0IGNvbW1pdCIgPgogICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbWFkZSUyMGJ5LU1haWEtMTVjM2Q2P3N0eWxlPWZsYXQmbG9nbz1hcHB2ZXlvciIgYWx0PSJNYWlhIiA+ICAKICA8L2E+CjwvcD4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxhIGhyZWY9IiNib29rbWFyay1hYm91dCI+QWJvdXQ8L2E+Jm5ic3A7Jm5ic3A7Jm5ic3A7fCZuYnNwOyZuYnNwOyZuYnNwOwogIDxhIGhyZWY9IiNjb21wdXRlci10ZWNobm9sb2dpZXMiPlRlY2hub2xvZ2llczwvYT4mbmJzcDsmbmJzcDsmbmJzcDt8Jm5ic3A7Jm5ic3A7Jm5ic3A7CiAgPGEgaHJlZj0iI3BhY2thZ2UtaW5zdGFsbGF0aW9uIj5JbnN0YWxsYXRpb248L2E+Jm5ic3A7Jm5ic3A7Jm5ic3A7fCZuYnNwOyZuYnNwOyZuYnNwOwogIDxhIGhyZWY9IiN3cmVuY2gtY29uZmlndXJhdGlvbiI+Q29uZmlndXJhdGlvbjwvYT4mbmJzcDsmbmJzcDsmbmJzcDt8Jm5ic3A7Jm5ic3A7Jm5ic3A7CiAgPGEgaHJlZj0iI21lbW8tZG9jdW1lbnRhdGlvbiI+RG9jdW1lbnRhdGlvbjwvYT4mbmJzcDsmbmJzcDsmbmJzcDt8Jm5ic3A7Jm5ic3A7Jm5ic3A7CiAgPGEgaHJlZj0iI21lbW8tbGljZW5zZSI+TGljZW5zZTwvYT4KPC9wPgoKPGJyPgoKIyMgOmJvb2ttYXJrOiBBYm91dAoKKipTb2MgRXgqKiBpcyBhIHNvY2lhbCBleHBlcmltZW50IHRoYXQgdXNlcyB0aGUgW1Bob2VuaXhdKGh0dHBzOi8vd3d3LnBob2VuaXhmcmFtZXdvcmsub3JnLykgZnJhbWV3b3JrIHRvIGNyZWF0ZSBhbiBBUEkuCgo8YnI+CgojIyA6Y29tcHV0ZXI6IFRlY2hub2xvZ2llcwoKLSAqKltFbGl4aXJdKGh0dHBzOi8vZWxpeGlyLWxhbmcub3JnLykqKgotICoqW1Bob2VuaXhdKGh0dHBzOi8vd3d3LnBob2VuaXhmcmFtZXdvcmsub3JnLykqKgotICoqW0RvY2tlcl0oaHR0cHM6Ly93d3cuZG9ja2VyLmNvbS8pKioKLSAqKltQb3N0Z3JlU1FMXShodHRwczovL3d3dy5wb3N0Z3Jlc3FsLm9yZy8pKioKCiMjIDpwYWNrYWdlOiBJbnN0YWxsYXRpb24KCmBgYGJhc2gKIyBjbG9uZSB0aGUgcmVwb3NpdG9yeQpnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2dhYnJpZWxtYWlhbHZhMzMvc29jLWV4LWFwaS5naXQKIyBlbnRlciB0aGUgZGlyZWN0b3J5CmNkIHNvYy1leC1hcGkKIyBpbnN0YWxsIHRoZSBkZXBlbmRlbmNpZXMKbWl4IGRlcHMuZ2V0ICMgb3IgbWl4IGRlcHMuZ2V0IC0tb25seSBwcm9kCiMgZWRpdCBgY29uZmlnL2Rldi5leHNgIGFuZCBjb25maWd1cmUgeW91ciBkYXRhYmFzZSBvciB1c2UgdGhlIGVudmlyb25tZW50IHZhcmlhYmxlcwpuYW5vIGNvbmZpZy9kZXYuZXhzICMgb3IgdmltIGNvbmZpZy9kZXYuZXhzCiMgcnVuIHRoZSBkYXRhYmFzZSBtaWdyYXRpb25zCm1peCBlY3RvLnNldHVwICMgb3IgbWl4IGVjdG8uc2V0dXAgLS1vbmx5IHByb2QKIyBzdGFydCB0aGUgc2VydmVyCm1peCBwaHguc2VydmVyICMgb3IgbWl4IHBoeC5zZXJ2ZXIgLS1vbmx5IHByb2QKYGBgCgojIyMgOndyZW5jaDogKipDb25maWd1cmF0aW9uKioKCm9wZW4gdGhlIGBjb25maWcvZGV2LmV4c2AgZmlsZSBhbmQgY29uZmlndXJlIHlvdXIgZGF0YWJhc2UKCmBgYGVsaXhpcgojIENvbmZpZ3VyZSB5b3VyIGRhdGFiYXNlCmNvbmZpZyA6c29jX2V4X2FwaSwgU29jRXhBcGkuUmVwbywKICB1c2VybmFtZTogInBvc3RncmVzIiwKICBwYXNzd29yZDogInBvc3RncmVzIiwKICBkYXRhYmFzZTogInNvY19leF9hcGlfZGV2IiwKICBob3N0bmFtZTogImxvY2FsaG9zdCIsCiAgc2hvd19zZW5zaXRpdmVfZGF0YV9vbl9jb25uZWN0aW9uX2Vycm9yOiB0cnVlLAogIHBvb2xfc2l6ZTogMTAKYGBgCgojIyMgOm1lbW86ICoqRG9jdW1lbnRhdGlvbioqCgpgYGBtZAojIFVzZSBpbnNvbW5pYSBvciBwb3N0bWFuIHRvIHRlc3QgdGhlIHJvdXRlcwoKZmlsZSBgc29jLWV4LWFwaS55YW1sYCBpbiByb290IGRpcmVjdG9yeQpgYGAKCiMjIyA6d3JpdGluZ19oYW5kOiAqKkF1dGhvcioqCgp8IFshW01haWFdKGh0dHBzOi8vYXZhdGFycy5naXRodWJ1c2VyY29udGVudC5jb20vdS8yNjczMjA2Nz9zaXplPTEwMCldKGh0dHBzOi8vZ2l0aHViLmNvbS9nYWJyaWVsbWFpYWx2YTMzKSB8CnwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLXwKfCBbTWFpYV0oaHR0cHM6Ly9naXRodWIuY29tL2dhYnJpZWxtYWlhbHZhMzMpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAoKIyMgTGljZW5zZQoKW01JVCBMaWNlbnNlXSguL0xJQ0VOU0UpCg== readmeEtag: '"b1b99a0a95b7edfe8f42ca34bd3a47c8513a137b"' readmeLastModified: Tue, 21 Nov 2023 03:12:28 GMT repositoryId: 675953605 description: >- Soc Ex is a social experiment that uses the Phoenix framework to create an API. created: '2023-08-08T05:29:52Z' updated: '2025-02-22T16:05:50Z' language: Elixir archived: false stars: 3 watchers: 0 forks: 0 owner: gabrielmaialva33 logo: https://avatars.githubusercontent.com/u/26732067?v=4 license: AGPL-3.0 repoEtag: '"c3b4cad4a07f469886eac93cae4acf16e237caa705e0d95e247ed3b7cc8d99bc"' repoLastModified: Sat, 22 Feb 2025 16:05:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/zubedev/deputy v3: true id: aa45746155c55d35e08ef817dd2cde1c repositoryMetadata: base64Readme: >- YGBgdGV4dArilojilojilojilojilojilojilZcg4paI4paI4paI4paI4paI4paI4paI4pWX4paI4paI4paI4paI4paI4paI4pWXIOKWiOKWiOKVlyAgIOKWiOKWiOKVl+KWiOKWiOKWiOKWiOKWiOKWiOKWiOKWiOKVl+KWiOKWiOKVlyAgIOKWiOKWiOKVlwrilojilojilZTilZDilZDilojilojilZfilojilojilZTilZDilZDilZDilZDilZ3ilojilojilZTilZDilZDilojilojilZfilojilojilZEgICDilojilojilZHilZrilZDilZDilojilojilZTilZDilZDilZ3ilZrilojilojilZcg4paI4paI4pWU4pWdCuKWiOKWiOKVkSAg4paI4paI4pWR4paI4paI4paI4paI4paI4pWXICDilojilojilojilojilojilojilZTilZ3ilojilojilZEgICDilojilojilZEgICDilojilojilZEgICAg4pWa4paI4paI4paI4paI4pWU4pWdCuKWiOKWiOKVkSAg4paI4paI4pWR4paI4paI4pWU4pWQ4pWQ4pWdICDilojilojilZTilZDilZDilZDilZ0g4paI4paI4pWRICAg4paI4paI4pWRICAg4paI4paI4pWRICAgICDilZrilojilojilZTilZ0K4paI4paI4paI4paI4paI4paI4pWU4pWd4paI4paI4paI4paI4paI4paI4paI4pWX4paI4paI4pWRICAgICDilZrilojilojilojilojilojilojilZTilZ0gICDilojilojilZEgICAgICDilojilojilZEK4pWa4pWQ4pWQ4pWQ4pWQ4pWQ4pWdIOKVmuKVkOKVkOKVkOKVkOKVkOKVkOKVneKVmuKVkOKVnSAgICAgIOKVmuKVkOKVkOKVkOKVkOKVkOKVnSAgICDilZrilZDilZ0gICAgICDilZrilZDilZ0KPi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIFJFU1QgQVBJIGZvciBQcm94aWVzCmBgYApbIVtkamFuZ29dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvZGphbmdvLTQuMi0lMjMwOTJFMjA/c3R5bGU9ZmxhdC1zcXVhcmUmbG9nbz1kamFuZ28pXShodHRwczovL3d3dy5kamFuZ29wcm9qZWN0LmNvbSkKWyFbcHl0aG9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3B5dGhvbi0zLjEyLSUyMzM3NzZBQj9zdHlsZT1mbGF0LXNxdWFyZSZsb2dvPXB5dGhvbildKGh0dHBzOi8vd3d3LnB5dGhvbi5vcmcpClshW215cHldKGh0dHBzOi8vd3d3Lm15cHktbGFuZy5vcmcvc3RhdGljL215cHlfYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9teXB5LWxhbmcub3JnKQpbIVtibGFja10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlJTIwc3R5bGUtYmxhY2stYmxhY2suc3ZnP3N0eWxlPWZsYXQtc3F1YXJlJmxvZ289c3R5bGVsaW50KV0oaHR0cHM6Ly9naXRodWIuY29tL3BzZi9ibGFjaykKWyFbUnVmZl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9lbmRwb2ludD91cmw9aHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2FzdHJhbC1zaC9ydWZmL21haW4vYXNzZXRzL2JhZGdlL3YyLmpzb24pXShodHRwczovL2dpdGh1Yi5jb20vYXN0cmFsLXNoL3J1ZmYpClshW3ByZS1jb21taXRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvcHJlLS1jb21taXQtZW5hYmxlZC1icmlnaHRncmVlbj9zdHlsZT1mbGF0LXNxdWFyZSZsb2dvPXByZS1jb21taXQpXShodHRwczovL3ByZS1jb21taXQuY29tKQpbIVtsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2xpY2Vuc2UtTUlULWJsdWU/c3R5bGU9ZmxhdC1zcXVhcmUpXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkKWyFbQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS96dWJlZGV2L2RlcHV0eS9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3p1YmVkZXYvZGVwdXR5L2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbCkKCiMjIEZlYXR1cmVzCgotIFt4XSBSZXN0IEFQSSBwcm92aWRpbmcgZnJlZSBwcm94aWVzIHRvIHVzZQotIFt4XSBJbnRlZ3JhdGlvbiB3aXRoIFtzY3JhcHlkb29dKGh0dHBzOi8vZ2l0aHViLmNvbS96dWJlZGV2L3NjcmFweWRvbykgdG8gb2J0YWluIHByb3hpZXMKLSBbeF0gSW50ZWdyYXRpb24gd2l0aCBbaW5zcGVjdG9yXShodHRwczovL2dpdGh1Yi5jb20venViZWRldi9pbnNwZWN0b3IpIHRvIHZhbGlkYXRlIHByb3hpZXMKLSBbeF0gUHJveGllcyB1cGRhdGVkIGFuZCBjaGVja2VkIGhvdXJseQoKIyMgVXNhZ2UKCmBgYGJhc2gKIyBDb3B5IHRoZSBleGFtcGxlIGVudmlyb25tZW50IGZpbGUgdG8gLmVudgojIFNDUkFQWURfVVJMIG11c3QgYmUgc2V0IGZvciB0aGUgd29ya2Zsb3cgdG8gd29yaywKIyBZb3UgY2FuIGdldCBhbiBpbnN0YW5jZSB1cCBhbmQgcnVubmluZyB0aHJvdWdoIGh0dHBzOi8vZ2l0aHViLmNvbS96dWJlZGV2L3NjcmFweWRvbwpjcCAuZW52LmV4YW1wbGUgLmVudgoKIyBCdWlsZCB0aGUgZG9ja2VyIGltYWdlIGFuZCBydW4gdGhlIGNvbnRhaW5lcgpkb2NrZXItY29tcG9zZSB1cCAtLWJ1aWxkIC0tZGV0YWNoCgojIFlvdSBjYW4gc2NhbGUgdXAgdGhlIG51bWJlciBvZiB3b3JrZXJzIGZvciBtb3JlIGNvbmN1cnJlbmN5CmRvY2tlci1jb21wb3NlIHVwIC0tc2NhbGUgd29ya2VyPTQgLS1kZXRhY2gKYGBgCgpbRGVwdXR5IEFQSV0oaHR0cDovL2xvY2FsaG9zdDo4MDAwKSBpcyBub3cgYXZhaWxhYmxlIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMC4gSWYgYERFQlVHPVRydWVgLCB5b3UgY2FuIHNlZSB0aGUgYnJvd3NhYmxlIEFQSS4KCltEZXB1dHkgQWRtaW5dKGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMC9hZG1pbikgaXMgbm93IGF2YWlsYWJsZSBhdCBodHRwOi8vbG9jYWxob3N0OjgwMDAvYWRtaW4uIENyZWRlbnRpYWxzIGFyZSBzZXQgYXV0b21hdGljYWxseSBmcm9tIGAuZW52YCBmaWxlLgoKIyMgRW5kcG9pbnRzCgotIFtyYW5kb21dKGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMC9wcm94aWVzL3JhbmRvbSk6IGAvcHJveGllcy9yYW5kb21gIC0gZ2V0IGEgcmFuZG9tIHByb3h5CgojIyBEZXZlbG9wbWVudAoKYGBgYmFzaAojIFBvZXRyeSBpcyByZXF1aXJlZCBmb3IgaW5zdGFsbGluZyBhbmQgbWFuYWdpbmcgZGVwZW5kZW5jaWVzCiMgaHR0cHM6Ly9weXRob24tcG9ldHJ5Lm9yZy9kb2NzLyNpbnN0YWxsYXRpb24KcG9ldHJ5IGluc3RhbGwKCiMgSWYgeW91IGRvbid0IGxpa2UgZG9pbmcgYHBvZXRyeSBydW5gIGFsbCB0aGUgdGltZQojIHBvZXRyeSBzaGVsbCAgIyBBY3RpdmF0ZSB2aXJ0dWFsIGVudmlyb25tZW50IGluIHRlcm1pbmFsCgojIFJlcXVpcmVzIGEgUG9zdGdyZVNRTCBkYXRhYmFzZSB0byBiZSBydW5uaW5nIGFuZCBjb25maWd1cmVkIGluIC5lbnYKIyBwb2V0cnkgcnVuIHB5dGhvbiBtYW5hZ2UucHkgbWFrZW1pZ3JhdGlvbnMgICMgQ3JlYXRlIG1pZ3JhdGlvbnMKcG9ldHJ5IHJ1biBweXRob24gbWFuYWdlLnB5IG1pZ3JhdGUgICMgUnVuIG1pZ3JhdGlvbnMKCiMgQ29sbGVjdCBzdGF0aWMgZmlsZXMgZm9yIHdoaXRlbm9pc2UKcG9ldHJ5IHJ1biBweXRob24gbWFuYWdlLnB5IGNvbGxlY3RzdGF0aWMKCiMgUnVuIERlcHV0eSBBUEkKcG9ldHJ5IHJ1biBweXRob24gbWFuYWdlLnB5IHJ1bnNlcnZlciAwLjAuMC4wOjgwMDAKCiMgQ3JlYXRlIGEgc3VwZXJ1c2VyCnBvZXRyeSBydW4gcHl0aG9uIG1hbmFnZS5weSBjcmVhdGVzdXBlcnVzZXIKCiMgSW5zdGFsbCBwcmUtY29tbWl0IGhvb2tzCnBvZXRyeSBydW4gcHJlLWNvbW1pdCBpbnN0YWxsCgojIEZvcm1hdHRpbmcgKGlucGxhY2UgZm9ybWF0cyBjb2RlKQpwb2V0cnkgcnVuIGJsYWNrIC4KCiMgTGludGluZyAoYW5kIHRvIGZpeCBhdXRvbWF0aWNhbGx5KQpwb2V0cnkgcnVuIHJ1ZmYgLgpwb2V0cnkgcnVuIHJ1ZmYgLS1maXggLgoKIyBUeXBlIGNoZWNraW5nCnBvZXRyeSBydW4gbXlweSAuCmBgYAoKQ29uZmlndXJhdGlvbiBkZXRhaWxzIGNhbiBiZSBmb3VuZCBpbiBbcHlwcm9qZWN0LnRvbWxdKHB5cHJvamVjdC50b21sKS4KCiMjIFN1cHBvcnQKWyFbUGF5cGFsXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1BheXBhbC1ATWRadWJhaXJCZWctMjUzQjgwPyZsb2dvPXBheXBhbCldKGh0dHBzOi8vcGF5cGFsLm1lL01kWnViYWlyQmVnLzEwKQo= readmeEtag: '"5d04a31331afea7085618a6927f9bf54ade70673"' readmeLastModified: Sat, 13 Jul 2024 02:50:39 GMT repositoryId: 677937436 description: REST API for Proxies created: '2023-08-13T06:20:27Z' updated: '2024-07-13T02:50:44Z' language: Python archived: false stars: 2 watchers: 1 forks: 0 owner: zubedev logo: https://avatars.githubusercontent.com/u/41994356?v=4 license: MIT repoEtag: '"0dfde3177534c0bcc1931780ace442f00c64e3bca5b375c20615853ad057b441"' repoLastModified: Sat, 13 Jul 2024 02:50:44 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/lolamarket/api-specifications v3: true id: 03abc85bdf37bca72d19ee3994fa3970 repositoryMetadata: base64Readme: >- IyBBUEkgU3BlY2lmaWNhdGlvbnMKR2xvdm9YTCdzIFtPcGVuQVBJXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy9sYXRlc3QuaHRtbCkgc3BlY2lmaWNhdGlvbi4KClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZpbGVzIHRoYXQgZG9jdW1lbnQgR2xvdm9YTCdzIFJFU1QgQVBJczoKCi0gW1JldGFpbGVyIEFQSV0oaHR0cHM6Ly9sb2xhbWFya2V0LmdpdGh1Yi5pby9hcGktc3BlY2lmaWNhdGlvbnMvcmV0YWlsZXItb2FzKQo= readmeEtag: '"7ece5a4d6da84497bff32f38e3619107a59738cb"' readmeLastModified: Tue, 16 Apr 2024 15:28:21 GMT repositoryId: 581213163 description: API Specifications created: '2022-12-22T15:15:35Z' updated: '2024-02-16T11:08:40Z' language: HTML archived: false stars: 2 watchers: 4 forks: 0 owner: lolamarket logo: https://avatars.githubusercontent.com/u/63723569?v=4 repoEtag: '"18b65017149681f42e61826b4cc09f9b551f0d029849b5508a440e5e5bf07fe9"' repoLastModified: Fri, 16 Feb 2024 11:08:40 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/flock-community/kotlin-openapi-bindings v3: true id: c5ac96691a194e6c680faa896c8cbbbb repositoryMetadata: base64Readme: >- IVtNYXZlbiBDZW50cmFsIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L2NvbW11bml0eS5mbG9jay5rb3RsaW54Lm9wZW5hcGkuYmluZGluZ3Mva290bGluLW9wZW5hcGktYmluZGluZ3MpClshW0J1aWxkXShodHRwczovL2dpdGh1Yi5jb20vZmxvY2stY29tbXVuaXR5L2tvdGxpbi1vcGVuYXBpLWJpbmRpbmdzL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZmxvY2stY29tbXVuaXR5L2tvdGxpbi1vcGVuYXBpLWJpbmRpbmdzL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnltbCkKWyFbTGljZW5zZTogTUlUXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtTUlULXllbGxvdy5zdmcpXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkKCiMgS290bGluIE9wZW5BUEkgQmluZGluZ3MKCkEgS290bGluIE11bHRpcGxhdGZvcm0gbGlicmFyeSB0aGF0IHByb3ZpZGVzIGJpbmRpbmdzIGZvciBPcGVuQVBJIEpTT04gZmlsZXMuIFRoZSBsaWJyYXJ5IGNhbiBwYXJzZSBhbmQgc2VyaWFsaXplIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgdG8gS290bGluIGRhdGEgY2xhc3NlcywgaGFuZGxpbmcgdW5pb24gdHlwZXMgYW5kIHgtcHJvcGVydGllcyAoZXh0ZW5zaW9ucykuCgojIyBUYWJsZSBvZiBDb250ZW50cwoKLSBbRmVhdHVyZXNdKCNmZWF0dXJlcykKLSBbU3VwcG9ydGVkIFBsYXRmb3Jtc10oI3N1cHBvcnRlZC1wbGF0Zm9ybXMpCi0gW0luc3RhbGxhdGlvbl0oI2luc3RhbGxhdGlvbikKLSBbVXNhZ2VdKCN1c2FnZSkKICAtIFtPcGVuQVBJIHYzXSgjb3BlbmFwaS12MykKICAtIFtPcGVuQVBJIHYyIChTd2FnZ2VyKV0oI29wZW5hcGktdjItc3dhZ2dlcikKICAtIFtIYW5kbGluZyB4LXByb3BlcnRpZXNdKCNoYW5kbGluZy14LXByb3BlcnRpZXMpCi0gW0FQSSBEb2N1bWVudGF0aW9uXSgjYXBpLWRvY3VtZW50YXRpb24pCi0gW0NvbnRyaWJ1dGluZ10oI2NvbnRyaWJ1dGluZykKLSBbTGljZW5zZV0oI2xpY2Vuc2UpCgojIyBGZWF0dXJlcwoKLSBTdXBwb3J0IGZvciBib3RoIE9wZW5BUEkgdjMgYW5kIHYyIChTd2FnZ2VyKSBzcGVjaWZpY2F0aW9ucwotIEtvdGxpbiBNdWx0aXBsYXRmb3JtIHN1cHBvcnQgKEpWTSwgSlMsIE5hdGl2ZSkKLSBTZXJpYWxpemF0aW9uIGFuZCBkZXNlcmlhbGl6YXRpb24gb2YgT3BlbkFQSSBKU09OCi0gUHJvcGVyIGhhbmRsaW5nIG9mIHVuaW9uIHR5cGVzCi0gU3VwcG9ydCBmb3IgeC1wcm9wZXJ0aWVzIChleHRlbnNpb25zKQotIFR5cGUtc2FmZSBLb3RsaW4gZGF0YSBjbGFzc2VzIGZvciBPcGVuQVBJIG1vZGVscwoKIyMgU3VwcG9ydGVkIFBsYXRmb3JtcwoKLSBKVk0KLSBKYXZhU2NyaXB0IChOb2RlSlMpCi0gTmF0aXZlCiAgLSBtYWNPUyAoeDY0LCBhcm02NCkKICAtIExpbnV4ICh4NjQpCiAgLSBXaW5kb3dzICh4NjQpCgojIyBJbnN0YWxsYXRpb24KCkFkZCB0aGUgZGVwZW5kZW5jeSB0byB5b3VyIGJ1aWxkLmdyYWRsZS5rdHMgb3IgYnVpbGQuZ3JhZGxlIGZpbGU6CgpgYGBrb3RsaW4KcmVwb3NpdG9yaWVzIHsKICAgIG1hdmVuQ2VudHJhbCgpCn0KCmRlcGVuZGVuY2llcyB7CiAgICBpbXBsZW1lbnRhdGlvbigiY29tbXVuaXR5LmZsb2NrLmtvdGxpbngub3BlbmFwaS5iaW5kaW5nczprb3RsaW4tb3BlbmFwaS1iaW5kaW5nczowLjEuMSIpCn0KYGBgCgojIyBVc2FnZQoKIyMjIE9wZW5BUEkgdjMKCmBgYGtvdGxpbgovLyBVc2luZyBjb21tdW5pdHkuZmxvY2sua290bGlueC5vcGVuYXBpLmJpbmRpbmdzLk9wZW5BUElWMwovLyBhbmQgY29tbXVuaXR5LmZsb2NrLmtvdGxpbngub3BlbmFwaS5iaW5kaW5ncy5PcGVuQVBJVjNNb2RlbAoKdmFsIGpzb24gPSAiIiIKewogICJvcGVuYXBpIjogIjMuMC4wIiwKICAiaW5mbyI6IHsKICAgICJ0aXRsZSI6ICJNeSBBUEkiLAogICAgInZlcnNpb24iOiAiMS4wLjAiCiAgfSwKICAicGF0aHMiOiB7CiAgICAiLyI6IHsKICAgICAgImdldCI6IHsKICAgICAgICAicmVzcG9uc2VzIjogewogICAgICAgICAgIjIwMCI6IHsKICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIk9LIiwKICAgICAgICAgICAgImNvbnRlbnQiOiB7CiAgICAgICAgICAgICAgImFwcGxpY2F0aW9uL2pzb24iOiB7CiAgICAgICAgICAgICAgICAic2NoZW1hIjogewogICAgICAgICAgICAgICAgICAidHlwZSI6ICJvYmplY3QiLAogICAgICAgICAgICAgICAgICAicHJvcGVydGllcyI6IHsKICAgICAgICAgICAgICAgICAgICAibWVzc2FnZSI6IHsKICAgICAgICAgICAgICAgICAgICAgICJ0eXBlIjogInN0cmluZyIKICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgIH0KICAgICAgICB9CiAgICAgIH0KICAgIH0KICB9Cn0KIiIiCgovLyBQYXJzZSBPcGVuQVBJIEpTT04gdG8gS290bGluIG9iamVjdAp2YWwgb2JqOiBPcGVuQVBJVjNNb2RlbCA9IE9wZW5BUElWMy5kZWNvZGVGcm9tU3RyaW5nKGpzb24pCgovLyBTZXJpYWxpemUgYSBLb3RsaW4gb2JqZWN0IGJhY2sgdG8gSlNPTgp2YWwgc3RyOiBTdHJpbmcgPSBPcGVuQVBJVjMuZW5jb2RlVG9TdHJpbmcob2JqKQpgYGAKCiMjIyBPcGVuQVBJIHYyIChTd2FnZ2VyKQoKYGBga290bGluCi8vIFVzaW5nIGNvbW11bml0eS5mbG9jay5rb3RsaW54Lm9wZW5hcGkuYmluZGluZ3MuT3BlbkFQSVYyCi8vIGFuZCBjb21tdW5pdHkuZmxvY2sua290bGlueC5vcGVuYXBpLmJpbmRpbmdzLk9wZW5BUElWMk1vZGVsCgp2YWwganNvbiA9ICIiIgp7CiAgInN3YWdnZXIiOiAiMi4wIiwKICAiaW5mbyI6IHsKICAgICJ0aXRsZSI6ICJNeSBBUEkiLAogICAgInZlcnNpb24iOiAiMS4wLjAiCiAgfSwKICAicGF0aHMiOiB7CiAgICAiLyI6IHsKICAgICAgImdldCI6IHsKICAgICAgICAicmVzcG9uc2VzIjogewogICAgICAgICAgIjIwMCI6IHsKICAgICAgICAgICAgImRlc2NyaXB0aW9uIjogIk9LIiwKICAgICAgICAgICAgInNjaGVtYSI6IHsKICAgICAgICAgICAgICAidHlwZSI6ICJvYmplY3QiLAogICAgICAgICAgICAgICJwcm9wZXJ0aWVzIjogewogICAgICAgICAgICAgICAgIm1lc3NhZ2UiOiB7CiAgICAgICAgICAgICAgICAgICJ0eXBlIjogInN0cmluZyIKICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgIH0KICAgICAgICB9CiAgICAgIH0KICAgIH0KICB9Cn0KIiIiCgovLyBQYXJzZSBTd2FnZ2VyIEpTT04gdG8gS290bGluIG9iamVjdAp2YWwgb2JqOiBPcGVuQVBJVjJNb2RlbCA9IE9wZW5BUElWMi5kZWNvZGVGcm9tU3RyaW5nKGpzb24pCgovLyBTZXJpYWxpemUgYSBLb3RsaW4gb2JqZWN0IGJhY2sgdG8gSlNPTgp2YWwgc3RyOiBTdHJpbmcgPSBPcGVuQVBJVjIuZW5jb2RlVG9TdHJpbmcob2JqKQpgYGAKCiMjIyBIYW5kbGluZyB4LXByb3BlcnRpZXMKClRoZSBsaWJyYXJ5IGF1dG9tYXRpY2FsbHkgaGFuZGxlcyB4LXByb3BlcnRpZXMgKGV4dGVuc2lvbnMpIGluIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMuIFRoZXNlIHByb3BlcnRpZXMgYXJlIHN0b3JlZCBpbiBhbiBgeFByb3BlcnRpZXNgIGZpZWxkIGR1cmluZyBwYXJzaW5nIGFuZCBhcmUgcmVzdG9yZWQgd2hlbiBzZXJpYWxpemluZyBiYWNrIHRvIEpTT04uCgojIyBBUEkgRG9jdW1lbnRhdGlvbgoKRm9yIGRldGFpbGVkIEFQSSBkb2N1bWVudGF0aW9uLCBwbGVhc2UgcmVmZXIgdG8gdGhlIFtEb2trYSBkb2N1bWVudGF0aW9uXShodHRwczovL2Zsb2NrLWNvbW11bml0eS5naXRodWIuaW8va290bGluLW9wZW5hcGktYmluZGluZ3MvKS4KCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyB0byBrb3RsaW4tb3BlbmFwaS1iaW5kaW5ncyBhcmUgd2VsY29tZSEgVG8gY29udHJpYnV0ZToKCjEuIEZvcmsgdGhlIHJlcG9zaXRvcnkKMi4gQ3JlYXRlIGEgZmVhdHVyZSBicmFuY2ggKGBnaXQgY2hlY2tvdXQgLWIgZmVhdHVyZS9hbWF6aW5nLWZlYXR1cmVgKQozLiBDb21taXQgeW91ciBjaGFuZ2VzIChgZ2l0IGNvbW1pdCAtbSAnQWRkIHNvbWUgYW1hemluZyBmZWF0dXJlJ2ApCjQuIFB1c2ggdG8gdGhlIGJyYW5jaCAoYGdpdCBwdXNoIG9yaWdpbiBmZWF0dXJlL2FtYXppbmctZmVhdHVyZWApCjUuIE9wZW4gYSBQdWxsIFJlcXVlc3QKClBsZWFzZSBtYWtlIHN1cmUgdG8gdXBkYXRlIHRlc3RzIGFzIGFwcHJvcHJpYXRlLgoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZSAtIHNlZSB0aGUgW0xJQ0VOU0VdKExJQ0VOU0UpIGZpbGUgZm9yIGRldGFpbHMuCg== readmeEtag: '"92569b9a92f77749ea38007e56fc236dfb58dd47"' readmeLastModified: Mon, 22 Sep 2025 20:43:41 GMT repositoryId: 629398742 description: Kotlin multiplatform bindings for parsing openapi3 spec created: '2023-04-18T08:29:05Z' updated: '2026-01-28T13:24:53Z' language: Kotlin archived: false stars: 7 watchers: 7 forks: 0 owner: flock-community logo: https://avatars.githubusercontent.com/u/44088861?v=4 repoEtag: '"8f16f5ad7feb3efd2a0622ab96afd29c4020d85e8da1c49667db9a3e495abf33"' repoLastModified: Wed, 28 Jan 2026 13:24:53 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/pradipmudi/expensys v3: true id: a77985f2e5a27065cb6a41d3726ba7bf repositoryMetadata: base64Readme: >- ## Table of Contents
- [ExpenSys](#ExpenSys)
- [Intuition](#intuition)
- [Key Features](#key-features)
    - [Expense Logging](#expense-logging)
    - [Monthly Reports](#monthly-reports)
    - [Categorization](#categorization)
    - [Subcategories](#subcategories)
    - [User-Specific Reports](#user-specific-reports)
    - [User-Centric View](#user-centric-view)
    - [Subcategory Mapping](#subcategory-mapping)
    - [Database Integration](#database-integration)

- [API Contract](#expensys-openapi-contract)
- [How to Use](#how-to-use)
- [Contribution Guidelines](#contribution-guidelines)


# **ExpenSys**

**ExpenSys** is an efficient **expense tracking** application designed to help you manage your **finances** and maintain a detailed record of your **expenses**. Whether you are an individual looking to **budget** your personal **finances** or a **business owner** tracking **expenses**, **ExpenSys** streamlines the process for you.

For this, I've made the UI from scratch as well and it's a responsive design. Few screenshots from the application.
### **Add Expense**
![Screenshot 2024-01-30 at 7 34 01 PM](https://github.com/pradipmudi/expensys/assets/6489613/c7f56fc5-314a-4369-85e4-be69f8e89665)
![Screenshot 2024-01-30 at 7 49 43 PM](https://github.com/pradipmudi/expensys/assets/6489613/9d28d84b-eadc-4d49-868d-21578f8eed33)

### **View expenses**
![Screenshot 2024-01-30 at 7 44 15 PM](https://github.com/pradipmudi/expensys/assets/6489613/09a2ced6-0219-4727-b3be-a613d347fa0e)
![Screen Shot 2024-01-30 at 19 45 36](https://github.com/pradipmudi/expensys/assets/6489613/2f7ef909-50d7-41ad-be4b-77dabe43feb2)

### Monthly Report
#### **By Main category and spent by user**
![Screen Shot 2024-01-30 at 19 52 08](https://github.com/pradipmudi/expensys/assets/6489613/e5d0f9de-fa55-455a-9fcf-8eb4d337ca05)

#### **By Sub category and spent by user**
![Screen Shot 2024-01-30 at 19 33 25](https://github.com/pradipmudi/expensys/assets/6489613/d8e30e90-82fe-41bd-99ce-2da946a357b7)
![Screen Shot 2024-01-30 at 19 38 14](https://github.com/pradipmudi/expensys/assets/6489613/d6827927-375a-458d-a5fc-0ceb7b88e288)


#### **By Sub category and spent by all**
![Screen Shot 2024-01-30 at 19 30 12](https://github.com/pradipmudi/expensys/assets/6489613/2607f71a-ac61-474b-8c43-e7841979747e)
![Screen Shot 2024-01-30 at 19 30 26](https://github.com/pradipmudi/expensys/assets/6489613/21781599-a1aa-4382-8fe4-1bd88475d859)

**Generated the logo for my application from here:** https://app.logo.com/dashboard/

## Intuition
I found it challenging to track my daily **expenses** using traditional methods such as Google Spreadsheets or existing **expense management apps**. They often lacked the flexibility and data organization I needed. As a result, I decided to develop **ExpenSys** to address these issues and create a more user-friendly **expense tracking** solution. Please feel free to submit **issues**, **feature requests** or **pull requests**; I'm open to accommodating them in future releases.

## Key Features

**ExpenSys** offers several key features to enhance your **expense tracking** experience:

### **Expense Logging**
- Users can easily log their daily **expenses** into the system, providing a detailed record of their spending.

### **Monthly Reports**
- Generate **monthly reports** based on your spending, allowing you to track your financial habits over time.

### **Categorization**
- **Expenses** are categorized into main categories, including **"Essential", "Expense", and "Loss of Money".** Users can also view **monthly reports** based on these categories.

### **Subcategories**
- **Subcategories** provide more granularity, such as **"Groceries", "Vegetables & Fruits", "Outside Food", "Salon", "Transport", "Medical", "Shopping", and "Loss of Money"**. These **subcategories** are displayed alongside their respective main categories.

### **User-Specific Reports**
- Users can select a specific month or view records for all months. They can also filter data based on categories (main or sub) and view **expenses** for all users.

### **User-Centric View**
- **ExpenSys** offers the ability to see reports based on main categories, allowing users to track their spending patterns effectively.

### **Subcategory Mapping**
- The application includes a dictionary of mappings for **subcategories**, ensuring consistency and clarity in **expense tracking**.

### **Database Integration**
- Users can add **expenses** directly into the database, ensuring data accuracy and easy retrieval.

### **Sorting**
- We can sort the result by any field based on report

## **How to Use**

To get started with **ExpenseTrackr**, follow these steps:

1. Clone the repository.
2. Set up the necessary **database** for **expense recording**.
3. Update DB configs on **application.yml**
4. Run the application.
5. Begin logging your daily **expenses** and utilize the various reporting and filtering options to manage your **financial data** effectively.

Your feedback and contributions are valuable to me, and I look forward to enhancing **ExpenTrackr** further based on your needs.
### **How to use the app in your local?**
These are the UI endpoints:
- **Add expense:** http://localhost:8080/new
- **View report:** http://localhost:8080/
- **View Expenses** http://localhost:8080/expenses

**Current TODO:** I plan to make the design more user friendly by creating a panel to redirect to the above pages. Please feel free to raise an PR on the same if you've already.

## **ExpenSys OpenAPI Contract**

The ExpenSys OpenAPI contract defines the interface for interacting with the ExpenSys application through its API. This contract allows developers to understand and integrate with ExpenSys seamlessly. You can access the full OpenAPI specification directly through Swagger UI:

[ExpenSys OpenAPI Contract - Swagger UI](https://petstore.swagger.io/?url=https://raw.githubusercontent.com/pradipmudi/expensys/main/src/main/java/com/expensys/openapi/expensys_openapi.yml#/default/get_report)

Simply click on the link above to view and interact with the ExpenSys OpenAPI contract in your web browser.

The ExpenSys OpenAPI contract is the foundation for building applications that interact with ExpenSys. If you have any questions or need further assistance, please don't hesitate to reach out. Happy API development!


## Contribution Guidelines

If you have any **feature requests** or would like to **contribute** to **ExpenSys**, please follow our contribution guidelines and submit **pull requests**. I'm committed to continually improving the application to meet the diverse needs of the users.

Thank you for choosing **ExpenSys** for your **expense tracking** needs!
 readmeEtag: '"cd37a9a0c59467246aaccdcdb5376a9cdad2a952"' readmeLastModified: Sun, 23 Jun 2024 10:25:10 GMT repositoryId: 709144141 description: >- Efficient expense tracking app for individuals and businesses. Log daily expenses, generate monthly reports, categorize spending, and integrate with a database. Check out the API contract: https://petstore.swagger.io/?url=https://raw.githubusercontent.com/pradipmudi/expensys/main/src/main/java/com/expensys/openapi/expensys_openapi.yml created: '2023-10-24T05:33:21Z' updated: '2025-12-25T11:49:50Z' language: Java archived: false stars: 5 watchers: 1 forks: 1 owner: pradipmudi logo: https://avatars.githubusercontent.com/u/6489613?v=4 repoEtag: '"c3a63134977b36b69b6d3745a5dc02f70de680f68bfb47fe1f2a99dc4d19df7e"' repoLastModified: Thu, 25 Dec 2025 11:49:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/vymalo/api-watcher-ui v3: true id: 684700fb41dc7baa6aa5c582db3b7d1f repositoryMetadata: base64Readme: IyBBUEkgV2F0Y2hlciBVSQoKTW9yZSBpbmZvIGNvbWluZyBuZXh0 readmeEtag: '"8fa1b3915a2f7f25879f885ae703db717eac481b"' readmeLastModified: Sat, 11 Nov 2023 08:14:24 GMT repositoryId: 620795170 description: UI App for the SMS API Watcher, a Stub server for APIs created: '2023-03-29T11:44:33Z' updated: '2025-09-15T00:25:03Z' language: TypeScript archived: true stars: 2 watchers: 1 forks: 0 owner: vymalo logo: https://avatars.githubusercontent.com/u/128943481?v=4 repoEtag: '"2ed6bf31c8a54df674beec8b11e8a170349a849762a9678ef9e33b4b6e5b7395"' repoLastModified: Mon, 15 Sep 2025 00:25:03 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/worksapplications/swagger-devkit v3: true repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyLWRldmtpdAoKWyFbR3JlZW5rZWVwZXIgYmFkZ2VdKGh0dHBzOi8vYmFkZ2VzLmdyZWVua2VlcGVyLmlvL1dvcmtzQXBwbGljYXRpb25zL3N3YWdnZXItZGV2a2l0LnN2ZyldKGh0dHBzOi8vZ3JlZW5rZWVwZXIuaW8vKSBbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL3N3YWdnZXItZGV2a2l0LnN2ZyldKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9zd2FnZ2VyLWRldmtpdCkgWyFbQ2lyY2xlQ0ldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL1dvcmtzQXBwbGljYXRpb25zL3N3YWdnZXItZGV2a2l0LnN2Zz9zdHlsZT1zdmcpXShodHRwczovL2NpcmNsZWNpLmNvbS9naC9Xb3Jrc0FwcGxpY2F0aW9ucy9zd2FnZ2VyLWRldmtpdCkKCldyaXRlIGEgc3dhZ2dlciBkb2N1bWVudCB3aXRoIFR5cGVTY3JpcHQvTm9kZS5qcy4KCiMjIEZlYXR1cmVzCgotIFN1cHBvcnRzIE9wZW5BUEkgMyAobm90IGNvbXBsZXRlZCB5ZXQpCi0gV2VsbC10eXBlZCBsaWJyYXJ5IHdpdGggY29tcGxldGlvbnMKLSBEZWNsYXJhdGl2ZSBzdHlsZSwgaW5zcGlyZWQgYnkgW2F3cy1jZGtdKGh0dHBzOi8vZ2l0aHViLmNvbS9hd3NsYWJzL2F3cy1jZGspCi0gRXh0ZW5kIHdpdGggY3VzdG9tIHBsdWdpbnMKCiMjIEV4YW1wbGUKCi0gW3BldHN0b3JlIGV4YW1wbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9Xb3Jrc0FwcGxpY2F0aW9ucy9zd2FnZ2VyLWRldmtpdC9ibG9iL21hc3Rlci9leGFtcGxlL2luZGV4LmpzKQoKUnVuOiBgbm9kZSBpbmRleC5qc2Agb3IgYG5vZGUgaW5kZXguanMgLS1oZWxwYAoKIyMgQ29weXJpZ2h0CgpDb3B5cmlnaHQgMjAxOSAmY29weTsgV29ya3MgQXBwbGljYXRpb25zIENvLixMdGQuCgpMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKeW91IG1heSBub3QgdXNlIHRoaXMgZmlsZSBleGNlcHQgaW4gY29tcGxpYW5jZSB3aXRoIHRoZSBMaWNlbnNlLgpZb3UgbWF5IG9idGFpbiBhIGNvcHkgb2YgdGhlIExpY2Vuc2UgYXQKCmh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4K readmeEtag: '"e1cc3b0edd5c82f514a458793b97a9af2ff08737"' readmeLastModified: Fri, 26 Jul 2019 16:37:54 GMT repositoryId: 180304121 description: Write a swagger document with TypeScript/Node.js. created: '2019-04-09T06:54:27Z' updated: '2023-01-27T20:28:50Z' language: TypeScript archived: true stars: 1 watchers: 2 forks: 2 owner: WorksApplications logo: https://avatars.githubusercontent.com/u/1889132?v=4 license: Apache-2.0 repoEtag: '"c554b7918c7c3f71d647fd1fb4df0e66cdbf511dc26b3dd503eedeb0496eb8a5"' repoLastModified: Fri, 27 Jan 2023 20:28:50 GMT foundInMaster: true category: Server id: 53b3055bf3be9288026a4b57925cf979 - source: openapi3 tags repository: https://github.com/wdes/typescript-openapi-builder v3: true repositoryMetadata: base64Readme: >- IyBUeXBlU2NyaXB0IE9wZW5BUEkgYnVpbGRlcgoKWyFbQWN0aW9ucyBTdGF0dXNdKGh0dHBzOi8vZ2l0aHViLmNvbS93ZGVzL3R5cGVzY3JpcHQtb3BlbmFwaS1idWlsZGVyL3dvcmtmbG93cy9SdW4lMjB0ZXN0cy9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vd2Rlcy90eXBlc2NyaXB0LW9wZW5hcGktYnVpbGRlci9hY3Rpb25zKQpbIVtBY3Rpb25zIFN0YXR1c10oaHR0cHM6Ly9naXRodWIuY29tL3dkZXMvdHlwZXNjcmlwdC1vcGVuYXBpLWJ1aWxkZXIvd29ya2Zsb3dzL0xpbnQlMjBhbmQlMjBidWlsZC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vd2Rlcy90eXBlc2NyaXB0LW9wZW5hcGktYnVpbGRlci9hY3Rpb25zKQpbIVtjb2RlY292XShodHRwczovL2NvZGVjb3YuaW8vZ2gvd2Rlcy90eXBlc2NyaXB0LW9wZW5hcGktYnVpbGRlci9icmFuY2gvbWFzdGVyL2dyYXBoL2JhZGdlLnN2ZyldKGh0dHBzOi8vY29kZWNvdi5pby9naC93ZGVzL3R5cGVzY3JpcHQtb3BlbmFwaS1idWlsZGVyKQpbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL3R5cGVzY3JpcHQtb3BlbmFwaS1idWlsZGVyLnN2ZyldKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy90eXBlc2NyaXB0LW9wZW5hcGktYnVpbGRlcikKCkdlbmVyYXRlIE9wZW5BUEkgc3BlYyB1c2luZyBUeXBlU2NyaXB0IGFubm90YXRpb25zCgpDbGVhbiwgZmFzdCwgc2ltcGxlLgoKIyMgVE9ETwoKLSBbIF0gSW1wcm92ZSBSRUFETUUubWQgdG8gc3RhdGUgY29kZSBpbXBvcnRhdGlvbgotIFsgXSBBZGQgbW9yZSB0ZXN0cwotIFsgXSBIYXZlIDEwMCUgY292ZXJhZ2UKCiMjIEtub3duIGJ1Z3MKCi0gW3hdIERvIE5PVCB1c2UgYWxpYXNlcyBmb3IgaW1wb3J0cyBsaWtlIGBpbXBvcnQgeyBQb3N0IGFzIFBvc3RPQSB9YCBpdCBicmVha3MgcGFyc2luZwoK readmeEtag: '"300805019b27ea637d61521f71f349b066be8688"' readmeLastModified: Mon, 20 Jul 2020 15:28:52 GMT repositoryId: 262324949 description: 'Generate OpenAPI spec using TypeScript annotations ' created: '2020-05-08T13:02:29Z' updated: '2023-01-28T13:00:16Z' language: TypeScript archived: true stars: 1 watchers: 1 forks: 0 owner: wdes logo: https://avatars.githubusercontent.com/u/25723041?v=4 license: MPL-2.0 repoEtag: '"3bc9fa71816ce7998b438beed01d45aa064fcca7e857a6858836039753ecae5b"' repoLastModified: Sat, 28 Jan 2023 13:00:16 GMT foundInMaster: true category: SDK id: 2cf245cb56941d7150f672e0c9a428c9 - source: openapi3 tags repository: https://github.com/freight-trust/open-edi v3: true repositoryMetadata: repositoryId: 297545146 description: >- OAS3 EDI API for Translation and Validation Transactional Service w/ Attestation & Non-Repudiation created: '2020-09-22T05:30:20Z' updated: '2020-11-04T15:00:24Z' language: HTML archived: false stars: 1 watchers: 0 forks: 0 owner: freight-trust logo: https://avatars.githubusercontent.com/u/57275623?v=4 repoEtag: '"2090d7c27e38829a0e1cf9d76ecea4081ff7f5394e68534ab346da0f49f50865"' repoLastModified: Wed, 04 Nov 2020 15:00:24 GMT foundInMaster: true id: b622714fa3da732c6dd0a82371f08ea0 - source: openapi3 tags repository: https://github.com/evanspauliuts/microservice-apis-coffeemesh v3: true id: 873ef7fa45e7f7e0f95052d7fe8c921c repositoryMetadata: base64Readme: >- IyBNaWNyb3NlcnZpY2UgYXBpcyBDb2ZmZWVtZXNoClRoaXMgYXBpIHJlc3RmdWwgYXBwbGljYXRpb24gY29mZmVlbWVzaCB1c2UgRmFzdEFwaSwgRmxhc2stc21vcmVzdCwgQ29kZSByZXBvc2l0b3J5IGFuZCBzZXJ2aWNlcwo= readmeEtag: '"56056ea0cd9ef9af4620f3b8c191dce663a527af"' readmeLastModified: Thu, 23 Feb 2023 18:05:27 GMT repositoryId: 605451015 description: >- This api restful application coffeemesh use FastApi, Flask-smorest, Code repository and services created: '2023-02-23T07:17:56Z' updated: '2023-07-28T06:18:31Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: EvansPauliuts logo: https://avatars.githubusercontent.com/u/15975585?v=4 repoEtag: '"0a05c0b5c293cb0bf841d5f25c1d90c642e1c3383ba5ecd1b98c271c378b829e"' repoLastModified: Fri, 28 Jul 2023 06:18:31 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/rail-shabayev/microservice-project2 v3: true id: 1f6ece8b31bed5851d58504d8bdd6183 repositoryMetadata: base64Readme: >- PGgxPlByb2plY3QncyBkYXRhYmFzZSB0YWJsZSBzdHJ1Y3R1cmU6PC9oMT4KCiFbaW1hZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9SYWlsLVNoYWJheWV2L21pY3Jvc2VydmljZS1wcm9qZWN0Mi9hc3NldHMvOTkxMzMyOTgvNjQzNTdjMWEtZDRkNS00MzIzLWE3NTItNDVjNzAzNWY1NmU2KQoKPGgxPlByb2plY3Qgc3RydWN0dXJlIGFuZCBmZWF0dXJlcyBpbiBkaWFncmFtOjwvaDE+CgohW2ltYWdlXShodHRwczovL2dpdGh1Yi5jb20vUmFpbC1TaGFiYXlldi9taWNyb3NlcnZpY2UtcHJvamVjdDIvYXNzZXRzLzk5MTMzMjk4LzdjZjc5NTBhLWRjZGMtNDFmYy04ZDFjLTg4NWQ5OTA2ZDhhOSkKCjxoMT5FbmRwb2ludHM6PC9oMT4KCiFbaW1hZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9SYWlsLVNoYWJheWV2L21pY3Jvc2VydmljZS1wcm9qZWN0Mi9hc3NldHMvOTkxMzMyOTgvYjAzOWE4MmUtYjNjMS00ZWU4LWEzNmUtOGZjZGNhMWQ4OTYyKQoKIVtpbWFnZV0oaHR0cHM6Ly9naXRodWIuY29tL1JhaWwtU2hhYmF5ZXYvbWljcm9zZXJ2aWNlLXByb2plY3QyL2Fzc2V0cy85OTEzMzI5OC9kOWUwMzVjYS0zMTA5LTRhZWUtODk1OC04ZjcwYTg5NGE4Y2EpCgohW2ltYWdlXShodHRwczovL2dpdGh1Yi5jb20vUmFpbC1TaGFiYXlldi9taWNyb3NlcnZpY2UtcHJvamVjdDIvYXNzZXRzLzk5MTMzMjk4LzI3ZWRlY2ViLTFhNjUtNDViMS05NmIwLWEzYjkyMWQ1YzIzMykKCiFbaW1hZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9SYWlsLVNoYWJheWV2L21pY3Jvc2VydmljZS1wcm9qZWN0Mi9hc3NldHMvOTkxMzMyOTgvOTQ5MGZjMmYtZDBiYS00MzA3LWJlMzMtMmI1YWQ5OTI5MDFiKQo= readmeEtag: '"392511d48020c08ecfb9dc11d708b1e410cfe01b"' readmeLastModified: Sun, 31 Mar 2024 01:56:58 GMT repositoryId: 764974606 description: 'pet project ' created: '2024-02-29T03:34:19Z' updated: '2025-10-26T04:56:53Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Rail-Shabayev logo: https://avatars.githubusercontent.com/u/99133298?v=4 repoEtag: '"c973eeebf9138a0d2aaef4c9173c4e085c9ddadcbf80ee16390b32db1eaa6cfb"' repoLastModified: Sun, 26 Oct 2025 04:56:53 GMT category: Server foundInMaster: true - source: openapi3 tags repository: https://github.com/pevin/laravel-api-boilerplate v3: true repositoryMetadata: base64Readme: >- IyBsYXJhdmVsLWFwaS1ib2lsZXJwbGF0ZQoKUGVyc29uYWwgdGVtcGxhdGUgLyBzdGF0ZXIga2l0LyBib2lsZXJwbGF0ZSB1c2VkIGZvciBjcmVhdGluZyBuZXcgYXBpIGJhc2VkIG9uIGxhcmF2ZWwgYWxvbmcgd2l0aCBvdGhlciBpbXBsZW1lbnRhdGlvbnMgbGlrZToKCi0gU3dhZ2dlci9PcGVuQXBpIDMuMCAKLSB0eW1vbi9qd3QtYXV0aAotIEVycm9yIGhhbmRsaW5nCgpSZWZlcmVuY2VzOgoKLSBbbGFyYXZlbC9mcmFtZXdvcmtdKGh0dHBzOi8vand0LWF1dGgucmVhZHRoZWRvY3MuaW8vZW4vZG9jcy9sYXJhdmVsLWluc3RhbGxhdGlvbi8pCi0gW3R5bW9uL2p3dC1hdXRoXShodHRwczovL2p3dC1hdXRoLnJlYWR0aGVkb2NzLmlvL2VuL2RvY3MvbGFyYXZlbC1pbnN0YWxsYXRpb24vKQotIFtkYXJrYW9ubGluZS9sNS1zd2FnZ2VyXShodHRwczovL2dpdGh1Yi5jb20vRGFya2FPbkxpbmUvTDUtU3dhZ2dlcikK readmeEtag: '"d57b2cf404771dae478b6757f99efed363f8d6bd"' readmeLastModified: Sun, 12 Jul 2020 15:24:33 GMT repositoryId: 279026479 description: >- Used for starting a new api based on Laravel along with jwt, swagger, and api error handler. created: '2020-07-12T08:50:57Z' updated: '2024-06-04T23:59:34Z' language: PHP archived: false stars: 1 watchers: 0 forks: 0 owner: pevin logo: https://avatars.githubusercontent.com/u/22132315?v=4 repoEtag: '"ccbd001980deb53dc292a09f6d4fe0172dc619d313ac2ede59e25d3740f8244f"' repoLastModified: Tue, 04 Jun 2024 23:59:34 GMT foundInMaster: true category: - Server - Server Implementations id: d02f42249534cfcac732404c7de1b93d - source: openapi3 tags repository: https://github.com/grandlinex/swagger-mate v3: true id: 6d9334e44d2e04e21dd95e81045609e3 repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyLU1hdGUKPiBHcmFuZExpbmVYIFN3YWdnZXItTWF0ZSAgcHJvamVjdAoKWyFbR2l0SHViXShodHRwczovL2JhZGdlLmZ1cnkuaW8vZ2gvZ3JhbmRsaW5leCUyRnN3YWdnZXItbWF0ZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vR3JhbmRsaW5lWC9zd2FnZ2VyLW1hdGUpClshW05QTV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9zdGF0aWMvdjE/bGFiZWw9TlBNJm1lc3NhZ2U9UGFja2FnZSZjb2xvcj1yZWQmbG9nbz1OUE0pXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9AZ3JhbmRsaW5leC9zd2FnZ2VyLW1hdGUpCiFbVFNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vc3RhdGljL3YxP2xhYmVsPUxhbmd1YWdlJm1lc3NhZ2U9VHlwZVNjcmlwdCZjb2xvcj1ibHVlJmxvZ289VHlwZVNjcmlwdCkKCiMjIyBTdGF0dXMKIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1HcmFuZGxpbmVYX3N3YWdnZXItbWF0ZSZtZXRyaWM9YWxlcnRfc3RhdHVzKQohW1NlY3VyaXR5IFJhdGluZ10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9R3JhbmRsaW5lWF9zd2FnZ2VyLW1hdGUmbWV0cmljPXNlY3VyaXR5X3JhdGluZykKIVtNYWludGFpbmFiaWxpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1HcmFuZGxpbmVYX3N3YWdnZXItbWF0ZSZtZXRyaWM9c3FhbGVfcmF0aW5nKQohW1JlbGlhYmlsaXR5IFJhdGluZ10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9R3JhbmRsaW5lWF9zd2FnZ2VyLW1hdGUmbWV0cmljPXJlbGlhYmlsaXR5X3JhdGluZykKIVtDb3ZlcmFnZV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9R3JhbmRsaW5lWF9zd2FnZ2VyLW1hdGUmbWV0cmljPWNvdmVyYWdlKQoKIyMjIElzc3VlcwohW0J1Z3NdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PUdyYW5kbGluZVhfc3dhZ2dlci1tYXRlJm1ldHJpYz1idWdzKQohW1Z1bG5lcmFiaWxpdGllc10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9R3JhbmRsaW5lWF9zd2FnZ2VyLW1hdGUmbWV0cmljPXZ1bG5lcmFiaWxpdGllcykKIVtDb2RlIFNtZWxsc10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9R3JhbmRsaW5lWF9zd2FnZ2VyLW1hdGUmbWV0cmljPWNvZGVfc21lbGxzKQoKIyMgRmVhdHVyZXMKCi0gR2VuZXJhdGUgc3dhZ2dlci57anNvbnx5bWx9IGZyb20gY29kZQotIEdlbmVyYXRlIEFwaSBjbGllbnQgZnJvbSBzd2FnZ2VyLntqc29ufHltbH0KCgojIyBRdWljayBTdGFydAoKIyMjIEluc3RhbGwKCjEuIEluc2F0YWwgbnBtIHBhY2thZ2UKYGBgc2hlbGwKICBucG0gaSAgQGdyYW5kbGluZXgvc3dhZ2dlci1tYXRlCmBgYAoKMi4gVXBkYXRlIHBhY2thZ2UuanNvbiAKYGBganNvbgp7CiAgLy8uLi4KICAic2NyaXB0cyI6IHsKICAgICJtYWtlU3BlYyI6ICJzd2FnZ2VyLW1hdGUiLAogICAgInNlcnZlU3BlYyI6ICJzd2FnZ2VyLW1hdGUgLS1zZXJ2ZSIsCiAgICAiYnVpbGRTcGVjTWFpbiI6ICJzd2FnZ2VyLW1hdGUgLS1idWlsZCAtLW1haW4iLAogICAgImJ1aWxkU3BlY0RldiI6ICJzd2FnZ2VyLW1hdGUgLS1idWlsZCAtLWRldiIKICB9LAogICJnbHgiOiB7CiAgICAia2VybmVsIjogImRpc3QvS2VybmVsLmpzIgogIH0KICAvLy4uLgp9CmBgYAoKfENvbW1hbmR8IERlc2NyaXB0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwtLS18LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CnxgbnBtIHJ1biBtYWtlU3BlY2B8IG1ha2UgYG9wZW5hcGkuanNvbmAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnxgbnBtIHJ1biBzZXJ2ZVNwZWNgfCBzZXJ2ZSBgb3BlbmFwaS5qc29uYCB3aXRoIFtzd2FnZ2VyLXVpXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkgfAp8YG5wbSBydW4gYnVpbGRTcGVjTWFpbmB8IGJ1aWxkIGFwaSBjbGllbnQgKHByb2QpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnxgbnBtIHJ1biBidWlsZFNwZWNEZXZgfCBidWlsZCBhcGkgY2xpZW50IChkZXYpICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CgojIyMgU2VydmUgb3B0aW9uCgo+IERlZmF1bHQgcG9ydCA9IDkwMDAKCkVOViB2YXJpYWJsZXMKCnwgRU5WIHwgRGVzY3JpcHRpb24gICAgICAgICAgICAgIHwKfC0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8ICBTV19QT1JUICAgfCBzZXQgY3VzdG9tIHNlcnZlIHBvcnQgICAgfAp8ICBTV19BVVRIICAgfCBzZXQgZGVmYXVsdCBiZWFyZXIgdG9rZW4gfAoKIyMgRGVmaW5lIHR5cGVzCgojIyMgS2VybmVsCgpgYGB0eXBlc2NyaXB0CmltcG9ydCB7IFNQYXRoVXRpbCwgU3dhZ2dlciB9IGZyb20gJ0BncmFuZGxpbmV4L3N3YWdnZXItbWF0ZSc7Ci8vIE9wZW5BcGkgMy4wLjMgLSBSb290IEFwaSBkZWZpbml0aW9uCkBTd2FnZ2VyKHsKICBpbmZvOiB7CiAgICB0aXRsZTogJ0tlcm5lbFRlc3QnLAogICAgdmVyc2lvbjogJzAuMS4wJywgLy8gVmVyc2lvbiAob3B0aW9uYWwpIHdpbGwgYmUgcmVhZCBmcm9tIHBhY2thZ2UuanNvbgogIH0sCiAgb3BlbmFwaTogJzMuMC4zJywKICBzZXJ2ZXJzOiBbCiAgICB7CiAgICAgIHVybDogJ2h0dHA6Ly9sb2NhbGhvc3Q6OTI1NycsCiAgICAgIGRlc2NyaXB0aW9uOiAnTG9jYWxEZXYnLAogICAgfSwKICBdLAogIHBhdGhzOiB7CiAgICAgIC8vIFN0YXRpYyBkZWZpbml0aW9uCiAgICAnL3ZlcnNpb24nOiB7CiAgICAgIGdldDogewogICAgICAgIGRlc2NyaXB0aW9uOiAnR2V0IHZlcnNpb24nLAogICAgICAgIG9wZXJhdGlvbklkOiAnZ2V0VmVyc2lvbicsCiAgICAgICAgcmVzcG9uc2VzOiBTUGF0aFV0aWwuZGVmYXVsdFJlc3BvbnNlKCcyMDAnLCAnNTAwJyksCiAgICAgIH0sCiAgICB9LAogICAvLyBEeW5hbWljIGRlZmluaXRpb24gd2lsbCBiZSByZWFkIGZyb20gQFNQYXRoCiAgfSwKICBzZWN1cml0eTogWwogICAgewogICAgICBiZWFyZXJBdXRoOiBbXSwKICAgIH0sCiAgXSwKICBjb21wb25lbnRzOiB7CiAgICBzZWN1cml0eVNjaGVtZXM6IHsKICAgICAgYmVhcmVyQXV0aDogewogICAgICAgIHR5cGU6ICdodHRwJywKICAgICAgICBzY2hlbWU6ICdiZWFyZXInLAogICAgICAgIGJlYXJlckZvcm1hdDogJ0pXVCcsCiAgICAgIH0sCiAgICB9LAogIH0sCn0pCmV4cG9ydCBkZWZhdWx0IGNsYXNzIFNvbWVLZXJuZWwge30KCmBgYAoKIyMjIEFjdGlvbgoKYGBgdHlwZXNjcmlwdAppbXBvcnQgeyBTUGF0aFV0aWwsIFN3YWdnZXIgfSBmcm9tICdAZ3JhbmRsaW5leC9zd2FnZ2VyLW1hdGUnOwoKLy8gT3BlbkFwaSAzLjAuMyAtIFBhdGNoIGRlZmluaXRpb24KQFNQYXRoKHsKICAgICcvdGVzdCc6IHsKICAgICAgICBnZXQ6IHsKICAgICAgICAgICAgZGVzY3JpcHRpb246ICd0ZXN0JywKICAgICAgICAgICAgb3BlcmF0aW9uSWQ6ICdnZXRUZXN0JywgLy8gbmFtZSBmb3IgdGhlIGpzIGFwaSBjbGllbnQKICAgICAgICAgICAgc3VtbWFyeTogJ0Rlc2NyaXRwdGlvbiBzdW1tYXJ5JywKICAgICAgICAgICAgcmVzcG9uc2VzOiBTUGF0aFV0aWwuZGVmYXVsdFJlc3BvbnNlKCcyMDAnLCc0MDAnLCAnNTAwJykKICAgICAgICB9LAogICAgfSwKfSkKZXhwb3J0IGRlZmF1bHQgY2xhc3MgU29tZUJhc2VBcGlBY3Rpb24ge30KCmBgYAo= readmeEtag: '"bd6271c06b76d5b48f2e22e85f5223c2e0933bbf"' readmeLastModified: Tue, 30 Jul 2024 17:25:44 GMT repositoryId: 655675985 description: null created: '2023-06-19T11:21:42Z' updated: '2026-01-28T13:08:27Z' language: TypeScript archived: false stars: 1 watchers: 0 forks: 0 owner: GrandlineX logo: https://avatars.githubusercontent.com/u/89772740?v=4 license: BSD-3-Clause repoEtag: '"b2e62ba434f590d8fdf64edef79778acccf35cd56715756a0c69dc6a18aea806"' repoLastModified: Wed, 28 Jan 2026 13:08:27 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/cedn/hangman v3: true repositoryMetadata: base64Readme: >- # Hangman
This project is a Rest API to play Hangman. This context must allow me to try to realize a side project by using the clean architecture, the contract first development and TDD with 100% code coverage. It's an ambitious target, but I like to think it's a success :)

## Table of content
1. [Clean Architecture](#clean-architectire)
1. [Some Word about Presenter](#some-word-about-presenter)
1. [Implementation with Spring Boot](#implementation-with-spring-boot)
1. [Contract First Development](#contract-first-development)
1. [TDD](#tdd)
1. [Code Coverage](#code-coverage)
1. [Install, build and run](#install-build-run)
1. [Links](#links)
1. [Status](#status)

<a name="clean-arcitecture"></a>
## Clean Architecture

The Clean Architecture is defined by Robert C. Martin (aka Uncle Bob), and I advise you to read his blog article about it: https://blog.cleancoder.com/uncle-bob/2012/08/13/the-clean-architecture.html. My implementation modelizes the controller and the presenter as 2 different classes outside of Spring Boot framework classes. In the examples, we could find, modelizes the presenter as the return type of the controller call. This is a mistake, and I advise you to read this detailed explication by Jeremiah Flaga: https://softwareengineering.stackexchange.com/questions/357052/clean-architecture-use-case-containing-the-presenter-or-returning-data. Another mistake is to believe that the controller is a framework 'controller' class like in all MVC framework. Another article by Tobias Strandberg to understand this controller confusion: https://adevelopersdiscourse.blogspot.com/2020/06/clean-architecture-demystified.html.

<a name="some-word-about-presenter"></a>
### Some words about Presenter
Using a presenter as a class instead of a return value from controller calls allows to have non-homogeneous returns. In the case of a letter proposal, in the hangman game, the result can be a letter found, a wrong letter, the word to guess discovered or a game over. Having a homogeneous type as a return value that covers all of these possibilities is obviously difficult. And this type may be complex and not respect the SOLID principles. The case of the result of the proposed letter can be considered as one event among several possible. The use case calls the presenter to use the right method that matches the suitable event.

_The 'output boundary' of letter proposal that the presenter must implement:_
```java
public interface LetterProposalOutputBoundary {
	void gameInProgress(ProgressingGame letterProposalResult);
	void lostGame(LostGame lostGame);
	void wonGame(WonGame wonGame);
	void gameIsOver(GameOver gameOver);
}
```
<a name="implementation-with-spring-boot"></a>
### Implementation with Spring Boot
Controller and presenter are adapter classes. Controller calls the interactor ('usecase' in my project's terminology) through the InputBoundary interface. Usecase calls the presenter through the OutputBoundary. It provides the result of the processing to the Presenter. The presenter transforms the result into a DTO. Finally, the Rest controller calls the controller to process the action and then reads the response from the presenter.
![Implementation diagram](images/adapters-usage.png)

The presenter is stateful because it stores the DTO but the Rest Controller is stateless. I choose to provide controller and presenter at the instantiation of the Rest controller. Just annotate the presenter with `@RequestScope` allows to solve this problem.

_Example from [cna.apps.hangman.tech.ApplicationConfiguration](https://github.com/CedN/hangman/blob/f180d630b1a8e6e6a037c6a9d295fef728552771/hangman-api/src/main/java/cna/apps/hangman/tech/ApplicationConfiguration.java#L42-L46)_:
```java
@Bean
@RequestScope
public LetterProposedPresenter getLetterProposalOutputBoundary() {
  return new LetterProposedPresenter();
}
```
<a name="contract-first-development"></a>
## Contract First Development
Contract-first development is about defining the contract of an API before starting to implement it. Using code generation for server and clients from the contract guarantees they are aligned. We can also mock the API very easily during its development.

I use [OpenAPI 3.0.1](https://spec.openapis.org/oas/v3.0.1) to describe the contract of the Hangman API in a dedicated Maven module named 'specifications'. The 'openapi-generator-maven-plugin' allows to generate code. I want to generate only interfaces for the service side and change some package's names.

_Openapi generator maven plugin configuration for Spring server_:
```xml
<build>
  <plugins>
    <plugin>
      <groupId>org.openapitools</groupId>
      <artifactId>openapi-generator-maven-plugin</artifactId>
      <!-- RELEASE_VERSION -->
      <version>5.1.1</version>
      <!-- /RELEASE_VERSION -->
      <executions>
        <execution>
          <goals>
            <goal>generate</goal>
          </goals>
          <configuration>
            <inputSpec>${project.basedir}/src/main/resources/hangman-openapi.yaml</inputSpec>
            <generatorName>spring</generatorName>
            <configOptions>
              <interfaceOnly>true</interfaceOnly>
              <basePackage>cna.apps.hangman.api</basePackage>
              <apiPackage>cna.apps.hangman.api</apiPackage>
              <modelPackage>cna.apps.hangman.api</modelPackage>
            </configOptions>
          </configuration>
        </execution>
      </executions>
    </plugin>
  </plugins>
</build>
```

I use [Random Words API](https://github.com/mcnaveen/Random-Words-API) to generate words to guess. I write the contract of the operation call and generate the client side in native Java by using 'openapi-generator-maven-plugin'

_Openapi generator maven plugin configuration for Java client_:
```xml
<build>
  <plugins>
    <plugin>
      <groupId>org.openapitools</groupId>
      <artifactId>openapi-generator-maven-plugin</artifactId>
      <version>5.1.0</version>
      <executions>
        <execution>
          <goals>
            <goal>generate</goal>
          </goals>
          <configuration>
            <inputSpec>${project.basedir}/src/main/resources/random-words-openapi.yaml</inputSpec>
            <generatorName>java</generatorName>
            <configOptions>
              <apiPackage>cna.apps.hangman.apiclients.randomwords</apiPackage>
              <dateLibrary>java8</dateLibrary>
              <fullJavaUtil>true</fullJavaUtil>
              <library>native</library>
              <modelPackage>cna.apps.hangman.apiclients.randomwords.model</modelPackage>
              <openApiNullable>false</openApiNullable>
              <swaggerAnnotations>false</swaggerAnnotations>
            </configOptions>
          </configuration>
        </execution>
      </executions>
    </plugin>
  </plugins>
</build>
```
<a name="tdd"></a>
## TDD
I start by implementing the use case tests. The progression into these unit tests allows me to design the domain entities. Coverage of use case tests allows the domain entities to emerge naturally. Then, I realize the development of adapters and then the Spring Boot glue.

The implementation of the domain (user case and entities) is realized by using outside-in TDD. However, the complete application is developed from an inside-out TDD angle.

No mock frameworks are used, so fake and spy classes are coded to simulate external dependencies behaviour or to assert external dependency calls.
<a name="code-coverage"></a>
## Code coverage
I coupled Jacoco and the [Coveralls](https://coveralls.io) service to measure code coverage. To achieve 100% code coverage, I excluded Spring Boot technical classes like application and configuration classes. 
<a name="install-build-run"></a>
## Install, build and run
* Prerequisite: have Java 16 installed (openJDK or Oracle JDK)
* Clone the repo
* Build the project with `mvnw clean package` from a terminal
* Run the class `cna.apps.hangman.tech.HangmanApplication` from your IDE or with `java -jar hangman-api/target/hangman-api-0.0.1-SNAPSHOT.jar` from a terminal. 
* Use [the OpenAPI contract](https://raw.githubusercontent.com/CedN/hangman/main/specifications/src/main/resources/hangman-openapi.yaml) to play :)
<a name="links"></a>
## Links
About clean architecture:
* Robert C. Martin's blog: https://blog.cleancoder.com/
* Tobias Strandberg's blog: https://adevelopersdiscourse.blogspot.com/
* Jeremiah Flaga's blog https://jeremiahflaga.github.io/

About OpenAPI:
* OpenAPI v3.0.1 Specifications : https://spec.openapis.org/oas/v3.0.1
* Spring generator options: https://openapi-generator.tech/docs/generators/spring
* Java client generator options: https://openapi-generator.tech/docs/generators/java

About code coverage:
* Jacoco maven plugin: https://www.eclemma.org/jacoco/trunk/doc/maven.html
* Coveralls : https://coveralls.io
<a name="status"></a>
## Status
[![Build Status](https://travis-ci.com/CedN/hangman.svg?branch=main)](https://travis-ci.com/CedN/hangman) [![Coverage Status](https://coveralls.io/repos/github/CedN/hangman/badge.svg?branch=main)](https://coveralls.io/github/CedN/hangman?branch=main)
 readmeEtag: '"d91d17af7cfee848cdd41528d7b6262198f7a604"' readmeLastModified: Mon, 23 Sep 2024 09:42:38 GMT repositoryId: 346851163 description: Hangman game created: '2021-03-11T22:02:48Z' updated: '2024-09-23T09:42:42Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: CedN logo: https://avatars.githubusercontent.com/u/4160253?v=4 license: Apache-2.0 repoEtag: '"b4f68859bfecbe1565821925a6756a25affcb4a9e81bc0721b2346d998f44993"' repoLastModified: Mon, 23 Sep 2024 09:42:42 GMT foundInMaster: true category: SDK id: de58be3df3712f388da92d99c4cc6b08 - source: openapi3 tags repository: https://github.com/ved-asole/crimsonsky v3: true id: caf53dadb58e08fbe8abaa2a27544acf repositoryMetadata: base64Readme: >- IyBDcmltc29uU2t5IC0gQW4gRmxpZ2h0IEJvb2tpbmcgTWljcm9zZXJ2aWNlcyBBcHBsaWNhdGlvbgoKIyBMaXZlIExpbmsgLyBEZW1vIExpbms6IPCflJcKQWNjZXNzIG15IHNpdGUgYXQgKipbbG9jYWxob3N0Ojg3NjVdKGh0dHA6Ly9sb2NhbGhvc3Q6ODc2NSkqKgoKIyBUYWJsZSBvZiBDb250ZW50OiDwn5ORCgotIFtBYm91dCBUaGUgQXBwXSgjYWJvdXQtdGhlLWFwcC0pCi0gW1NjcmVlbnNob3RzXSgjc2NyZWVuc2hvdHMtKQotIFtUZWNobm9sb2dpZXNdKCN0ZWNobm9sb2dpZXMtJUVGJUI4JThGLS0lRUYlQjglOEYpCi0gW1NldHVwXSgjc2V0dXAtKQogIC0gW0phdmEgKyBTcHJpbmcgQm9vdF0oI2phdmEtLXNwcmluZy1ib290KQogICAgLSBbQXZhaWxhYmxlIFNjcmlwdHNdKCNhdmFpbGFibGUtc2NyaXB0cy0tKQotIFtBcHByb2FjaF0oI2FwcHJvYWNoLSkKLSBbU3RhdHVzXSgjc3RhdHVzLSkKLSBbTGljZW5zZV0oI2xpY2Vuc2UtKQoKIyBBYm91dCB0aGUgQXBwOiDwn5OaCkNyaW1zb25Ta3kgaXMgYSBmbGlnaHQgYm9va2luZyBzeXN0ZW0gYnVpbHQgdXNpbmcgU3ByaW5nIEJvb3QgTWljcm9zZXJ2aWNlcyBhbmQgS2Fma2EsIGhvc3RlZCBvbiBBV1MgRUMyLiBUaGUgc3lzdGVtIGlzIGRlc2lnbmVkIHRvIGhhbmRsZSBmbGlnaHQgc2VhcmNoLCBib29raW5nLCBhbmQgcGF5bWVudHMgaW4gYSBzY2FsYWJsZSBhbmQgcmVzaWxpZW50IG1hbm5lci4gVGhlIGFyY2hpdGVjdHVyZSBmb2xsb3dzIGEgbWljcm9zZXJ2aWNlcyBwYXR0ZXJuIHdpdGggZWFjaCBzZXJ2aWNlIGRlY291cGxlZCB0byBlbmhhbmNlIGZsZXhpYmlsaXR5IGFuZCBmYXVsdC10b2xlcmFuY2UuCgojIFNjcmVlbnNob3RzOiDwn5O3CgohW0NyaW1zb25Ta3kgQXJjaGl0ZWN0dXJlIERpYWdyYW1dKC4vaGlnaC1sZXZlbC1kaWFncmFtLnBuZykKCiMgVGVjaG5vbG9naWVzOiDimJXvuI8gIOKam++4jwoKLSBKYXZhCi0gU3ByaW5nIENsb3VkCi0gU3ByaW5nIEJvb3QKLSBTcHJpbmcgV2ViCi0gU3ByaW5nIEFwaSBHYXRld2F5Ci0gU3ByaW5nIERhdGEgSlBBCi0gS2Fma2EKLSBBV1MgRUMyIGFuZCBDbG91ZGZsYXJlCi0gSDIsIE1vbmdvREIgYW5kIFBvc3RncmVzIERCCi0gU3dhZ2dlciBPcGVuQVBJCgojIFNldHVwOiDwn5K7CgojIyBKYXZhICsgU3ByaW5nIEJvb3QgCgojIyMgQXZhaWxhYmxlIFNjcmlwdHMgLSAKCkluIHRoZSBwcm9qZWN0IGRpcmVjdG9yeSwgeW91IGNhbiBydW46CgojIyMgYG12biBpbnN0YWxsYAoKVG8gaW5zdGFsbCBhbGwgdGhlIGRlcGVuZGVuY2llcyByZXF1aXJlZCBmb3IgdGhlIHByb2plY3QuCgojIyMgYG12biBzcHJpbmctYm9vdDpydW5gCgpSdW5zIHRoZSBhcHAgaW4gdGhlIGRldmVsb3BtZW50IG1vZGUuXApPcGVuIFtodHRwOi8vbG9jYWxob3N0Ojg3NjVdKGh0dHA6Ly9sb2NhbGhvc3Q6ODc2NSkgdG8gdmlldyBpdCBpbiB5b3VyIGJyb3dzZXIuCgoKIyMjIGBucG0gY2xlYW4gaW5zdGFsbGAKCkJ1aWxkcyB0aGUgYXBwIGZvciBwcm9kdWN0aW9uIHRvIHRoZSBgdGFyZ2V0YCBmb2xkZXIuXApJdCBjb3JyZWN0bHkgYnVuZGxlcyBTcHJpbmcgQm9vdCBhcHAgaW4gcHJvZHVjdGlvbiBtb2RlIGFuZCBvcHRpbWl6ZXMgdGhlIGJ1aWxkIGZvciB0aGUgYmVzdCBwZXJmb3JtYW5jZS4KCllvdXIgYXBwIGlzIHJlYWR5IHRvIGJlIGRlcGxveWVkIQoKIyBBcHByb2FjaDog8J+atgpXZSBhcmUgdXNpbmcgKipSZWFjdCoqIGZvciBmcm9udGVuZCBVSSBhbmQgKipKYXZhIHdpdGggU3ByaW5nIEJvb3QqKiBmb3IgYmFja2VuZCBBUEkgd2l0aCAqKk1pY3Jvc2VydmljZXMqKiBhcmNoaXRlY3R1cmUuCgojIFN0YXR1czog8J+TtgpXb3JrIGluIFByb2dyZXNzLi4u8J+boO+4jwoKIyBMaWNlbnNlOiDCqe+4jwpNSVQgTGljZW5zZSAoKipbQ2hlY2sgSGVyZV0oTElDRU5TRSkqKik= readmeEtag: '"a177ebc04c6786b3d7a0cf1f9530027876ccc3d2"' readmeLastModified: Sat, 26 Oct 2024 20:26:58 GMT repositoryId: 860100107 description: >- A Real-time Flight Booking microservices application with Java and Spring Cloud created: '2024-09-19T20:25:08Z' updated: '2025-07-28T16:10:45Z' language: null archived: false stars: 1 watchers: 1 forks: 0 owner: ved-asole logo: https://avatars.githubusercontent.com/u/77259804?v=4 license: MIT repoEtag: '"e51d6abda62741faf31a79b441d45c1e4200d1e2d767538b4d8a3312caf7403b"' repoLastModified: Mon, 28 Jul 2025 16:10:45 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sellix/developers-api-specification v3: true repositoryMetadata: base64Readme: >- IyBbZGV2LW9wZW5hcGkuc2VsbGl4LmlvXShodHRwczovL2Rldi1vcGVuYXBpLnNlbGxpeC5pbykKCkRldmVsb3BlcnMgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIHVzZWQgYnkgb3VyIFNES3Mu readmeEtag: '"17a4472e322e54c5a37739ec6617800e5ea897d8"' readmeLastModified: Sun, 13 Feb 2022 09:58:25 GMT repositoryId: 443609714 description: >- OpenApi v3.0.0 schema for the Sellix Developers API. Contains examples, payloads and specifications on requests and responses. created: '2022-01-01T19:33:11Z' updated: '2024-11-20T12:47:19Z' language: null archived: true stars: 1 watchers: 3 forks: 1 owner: Sellix logo: https://avatars.githubusercontent.com/u/66204773?v=4 repoEtag: '"d45e5ab22d010d7c92aa00beaee3a2f02748d09f8fea14d1714bd4a9861eb5da"' repoLastModified: Wed, 20 Nov 2024 12:47:19 GMT foundInMaster: true category: SDK id: 32ab75082c937ea942ad563a87db51a0 - source: openapi3 tags repository: https://github.com/chrisdostert/listener-from-oas3-js v3: true repositoryMetadata: base64Readme: >- WyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvY2hyaXNkb3N0ZXJ0L2xpc3RlbmVyLWZyb20tb2FzMy1qcy5zdmc/YnJhbmNoPW1hc3RlcildKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9jaHJpc2Rvc3RlcnQvbGlzdGVuZXItZnJvbS1vYXMzLWpzKQpbIVtDb3ZlcmFnZV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2NocmlzZG9zdGVydC9saXN0ZW5lci1mcm9tLW9hczMtanMvYnJhbmNoL21hc3Rlci9ncmFwaC9iYWRnZS5zdmcpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvY2hyaXNkb3N0ZXJ0L2xpc3RlbmVyLWZyb20tb2FzMy1qcykKCj4gKkJlIGFkdmlzZWQ6IHRoaXMgcHJvamVjdCBpcyBjdXJyZW50bHkgYXQgTWFqb3IgdmVyc2lvbiB6ZXJvLiBQZXIgdGhlCj4gc2VtYW50aWMgdmVyc2lvbmluZyBzcGVjOiAiTWFqb3IgdmVyc2lvbiB6ZXJvICgwLnkueikgaXMgZm9yIGluaXRpYWwKPiBkZXZlbG9wbWVudC4gQW55dGhpbmcgbWF5IGNoYW5nZSBhdCBhbnkgdGltZS4gVGhlIHB1YmxpYyBBUEkgc2hvdWxkCj4gbm90IGJlIGNvbnNpZGVyZWQgc3RhYmxlLiIqCgpKYXZhc2NyaXB0IGxpYnJhcnkgd2hpY2ggZ2VuZXJhdGVzIGEgbmF0aXZlIGh0dHAgc2VydmVyIGxpc3RlbmVyIGZyb20gYSB2MyBvcGVuIGFwaSBzcGVjCgojIEluc3RhbGxhdGlvbgoKIyMgTlBNCmBgYHNoZWxsCm5wbSBpbnN0YWxsIC0tc2F2ZSBsaXN0ZW5lci1mcm9tLW9hczMKYGBgCgojIyBZYXJuCmBgYHNoZWxsCnlhcm4gYWRkIGxpc3RlbmVyLWZyb20tb2FzMwpgYGAKCiMgSGFuZGxlcnMKClRoZSBzaWduYXR1cmUgZm9yIGhhbmRsZXJzIGlzCmBhc3luYyBteUhhbmRsZXIoY3R4KWAKClRoZSBjdHggb2JqZWN0IGlzIHRoZSBzYW1lIGFzIHRoZSBrb2EgY3R4IG9iamVjdCBidXQgYWxzbyBpbmNsdWRlczoKCiMjIyBjdHgucGFyYW1ldGVycy5wYXRoCk9iamVjdCB3aGVyZSBlYWNoIHByb3BlcnR5IGlzIHRoZSBuYW1lIG9mIGEgcGF0aCBwYXJhbWV0ZXIgYXBwbGljYWJsZSB0byB0aGUgb3BlcmF0aW9uIGFuZCBpdCdzIHByb3ZpZGVkIHZhbHVlLgoKIyMjIGN0eC5wYXJhbWV0ZXJzLnF1ZXJ5Ck9iamVjdCB3aGVyZSBlYWNoIHByb3BlcnR5IGlzIHRoZSBuYW1lIG9mIGEgcXVlcnkgcGFyYW1ldGVyIGFwcGxpY2FibGUgdG8gdGhlIG9wZXJhdGlvbiBhbmQgaXQncyBwcm92aWRlZCB2YWx1ZS4KCnBhcmFtZXRlcnMgZGVjbGFyZWQgdy8gYXBwbGljYXRpb24vanNvbiBjb250ZW50IHdpbGwgYmUgZGVzZXJpYWxpemVkCgojIyBjdHguc2VjdXJpdHkKT2JqZWN0IHdoZXJlIGVhY2ggcHJvcGVydHkgaXMgdGhlIG5hbWUgb2YgYSBzZWN1cml0eSBkZWZpbml0aW9uIGFwcGxpY2FibGUgdG8gdGhlIG9wZXJhdGlvbiBhbmQgaXQncyBwcm92aWRlZCB2YWx1ZS4KCmZvciBvYXV0aDIsIHZhbHVlIHdpbGwgYmUgdGhlIGJlYXJlciB0b2tlbiBzdHJpbmcgIAoKZm9yIGh0dHAgYmVhcmVyLCB2YWx1ZSB3aWxsIGJlIHRoZSBiZWFyZXIgdG9rZW4gc3RyaW5nICAKCmZvciBodHRwIGJhc2ljLCB2YWx1ZSB3aWxsIGJlIGEgYHt1c2VySWQsIHBhc3N3b3JkfWAgb2JqZWN0CgojIEV4YW1wbGVzCgojIEJhc2ljIHVzYWdlCgpgYGBqYXZhc2NyaXB0CmNvbnN0IGdldFJlcXVlc3RMaXN0ZW5lciA9IHJlcXVpcmUoJ2xpc3RlbmVyLWZyb20tb2FzMycpCmNvbnN0IGh0dHAgPSByZXF1aXJlKCdodHRwJykKCmFzeW5jIGZ1bmN0aW9uIGxpc3Rlbihwb3J0KSB7CiAgY29uc3QgcmVxdWVzdExpc3RlbmVyID0gYXdhaXQgZ2V0UmVxdWVzdExpc3RlbmVyKAogICAgYCR7X19kaXJuYW1lfS9vcGVuYXBpLnlhbWxgLAogICAgLy8gcmVzb2x2ZSBoYW5kbGVycyBmcm9tIGAke19fZGlybmFtZX0vJHtvcGVyYXRpb24tcGF0aH0vJHtvcGVyYXRpb24tbWV0aG9kfWAKICAgIF9fZGlybmFtZQogICkKCiAgaHR0cAogICAgLmNyZWF0ZVNlcnZlcihyZXF1ZXN0TGlzdGVuZXIpCiAgICAubGlzdGVuKHBvcnQpCn0KYGBgCgojIFN1cHBvcnQKCltvcGVuIGFuIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vY2hyaXNkb3N0ZXJ0L2xpc3RlbmVyLWZyb20tb2FzMy1qcy9pc3N1ZXMpCgojIFJlbGVhc2VzCgpyZWxlYXNlcyBhcmUgdmVyc2lvbmVkIGFjY29yZGluZyB0bwpbIVtzZW12ZXIgMi4wLjBdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvc2VtdmVyLTIuMC4wLWJyaWdodGdyZWVuLnN2ZyldKGh0dHA6Ly9zZW12ZXIub3JnL3NwZWMvdjIuMC4wLmh0bWwpCmFuZCBbdGFnZ2VkXShodHRwczovL2dpdC1zY20uY29tL2Jvb2svZW4vdjIvR2l0LUJhc2ljcy1UYWdnaW5nKTsgc2VlCltDSEFOR0VMT0cubWRdKENIQU5HRUxPRy5tZCkgZm9yIHJlbGVhc2Ugbm90ZXMKCiMgQ29udHJpYnV0aW5nCgpzZWUgW0NPTlRSSUJVVElORy5tZF0oQ09OVFJJQlVUSU5HLm1kKQo= readmeEtag: '"dc7b9c92e2309906a0295ca8037f11c405b635e0"' readmeLastModified: Mon, 15 Oct 2018 17:17:08 GMT repositoryId: 140190346 description: >- Javascript library which generates a native http server listener from a v3 open api spec created: '2018-07-08T17:56:57Z' updated: '2023-01-28T09:32:25Z' language: JavaScript archived: true stars: 1 watchers: 0 forks: 0 owner: chrisdostert logo: https://avatars.githubusercontent.com/u/2335774?v=4 license: MIT repoEtag: '"8725dcf593aab39fde799652cc9bf9c9c547e122a075951354fe58a63069d2c2"' repoLastModified: Sat, 28 Jan 2023 09:32:25 GMT foundInMaster: true category: - Converters - Parsers id: cdd1ea9a0dde706b0c4b86a1650f2e43 - source: openapi3 tags repository: https://github.com/aqib1/sparknetworks-filtering-matching v3: true repositoryMetadata: base64Readme: >- IyMgU3BhcmstbmV0d29ya3MgRmlsdGVyIEhhbmRsaW5nIGFwcGxpY2F0aW9uClNwcmluZyBjbG91ZCBtaWNyb3NlcnZpY2UgZm9yIGZpbHRlciBoYW5kbGluZwoKIyMjIEhvdyB0byBSdW4KLSBUbyBydW4gYXBwbGljYXRpb24gaXRzIG5lY2Vzc2FyeSB0byBydW4gRXVyZWthU2VydmVyIGFwcGxpY2F0aW9uIGZpcnN0LCBzbyBvdGhlciBib3RoIGFwcGxpY2F0aW9uIHJlZ2lzdGVyIGZpcnN0IHdpdGggc2VydmVyCi0gbXZuIGV4ZWM6amF2YQoKIyMjIFRlY2hub2xvZ2llcwoqIFNwcmluZyBib290IDIKKiBTcHJpbmcgQ2xvdWQgKE5ldGZsaXggZXVyZWthLCBOZXRmbGl4IGZlaWduIChlbWJlZGRlZCByaWJib24gTEIpLCBOZXRmbGl4IEh5c3RyaXgpCiogU3ByaW5nIE1WQwoqIFNwcmluZyBSZXN0CiogU3ByaW5nIGRhdGEgSlBBCiogU3ByaW5nIEFPUAoqIGJvb3RzdHJhcCBhbmQgSlNUTAoqIEp1bml0IDQgKGZvciB1bml0IHRlc3RpbmcpCiogTW9ja2l0byAoZm9yIG1vY2tpbmcgZGF0YWJhc2UgY2FsbHMgYW5kIG1ldGhvZHMgaW4gdW5pdCB0ZXN0aW5nKQoqIE1vY2tNVkMgKGZvciBJbnRlZ3JhdGlvbiB0ZXN0aW5nKQoqIFN3YWdnZXIgKG9wZW5hcGkgMy4wKQoqIEgyIERhdGFiYXNlCiogT3BlbnBvam8KCiMjIyBEZXNpZ24gUGF0dHJlbgoqIERvdWJsZSBjaGVjayBsb2NraW5nIHNpbmdsZXRvbiBwYXR0cmVuCiogQnVpbGRlciBwYXR0cmVuCgojIyMgVGhyZWFkIHNhZmV0eQpGb3IgdGhyZWFkIHNhZmV0eSwgaSBhbSB1c2luZyBqYXZhIDEuOCBTdGFtcGVkTG9jaywgaW4gd2hpY2ggd2UgaGF2ZSBvcHRpbWlzdGljIHJlYWQgbG9jay4gV2hpY2ggbWFrZSBzeW5jaHJvbml6YXRpb24gb3ZlcmhlYWQgaXMgdmVyeSAgIGxvdy4KCiMjIyBBcHBsaWNhdGlvbiBkZXRhaWxzCi0gQWxsIGFwcGxpY2F0aW9uIGlzIGJ1aWxkIHVzaW5nIHNwcmluZyBjbG91ZCwgbWljcm9zZXJ2aWNlIGFyY2hpdGVjdHVyZQotIENsaWVudCBhcHBsaWNhdGlvbiBpcyBidWlsZCB1c2luZyBTcHJpbmcgTVZDLCBDbGllbnQgQXBwbGljdGlvbiByZWdpc3RlciBpdCBzZWxmIGFzIGVydWVrYSBjbGllbnQgaW4gc2VydmVyLgotIENsaWVudCBhcHBsaWNhdGlvbiBjb21tdW5pY2F0aW5nIHdpdGggYmFja2VuZCBhcGkgdXNpbmcgTmV0ZmxpeCBmZWlnbgotIFNpbXBsZSBkZXNpZ24gaXMgbWFkZXVwIHVzaW5nIGJvb3RzdHJhcCBhbmQgSlNUTCAoZm9yIGRhdGEgbWFuaXB1bGF0aW9uKQotIEJhY2tlbmQgYXBwbGljYXRpb24gaXMgYSBldXJha2EgY2xpZW50LCB3aGljaCBpcyBidWlsZCB1c2luZyBzcHJpbmcgcmVzdAotIEgyIGRhdGFiYXNlIGlzIHVzZWQgc2F2ZSBkYXRhIGFuZCBnaXZlbiBkYXRhIGlzIGxvYWRlZCBkdXJpbmcgdGhlIHN0YXJ0LXVwIG9mIGFwcGxpY2F0aW9uCi0gU3ByaW5nIHNwZWNpZmljYXRpb25zIGFyZSB1c2VkIHRvIGNyZWF0ZSBjb21wbGV4IHF1cmllcwotIEVydWVrYSBTZXJ2ZXIgY29uZmlndXJlIG9uIDgwODUKLSBDbGllbnQgYXBwbGljYXRpb24gaXMgY29uZmlndXJlIG9uIDgwODAKLSBCYWNrZW5kIGFwcGxpY2F0aW9uIGlzIGNvbmZpZ3VyZSBvbiA4MDgyCgojIyMgVGVzdGluZwotIFRERCAodW5pdCB0ZXN0aW5nIGRvbmUgdXNpbmcgSlVOSVQgNCkKLSBNb2NraXRvIChmb3IgbW9ja2luZyBpbiB1bml0IHRlc3RpbmcpCi0gb3BlbnBvam8gKGZvciB0ZXN0aW5nIHJlcXVpcmVkIGdldHRlci9zZXR0ZXIgZXhpc3RlbmNlKQotIE1vY2tNVkMgKGZvciBpbnRlZ2VyYXRpb24gdGVzdGluZyBvZiBBUEkgZW5kIHBvaW50cykKCgoKIyMjIEFQSSBVc2FnZQoKSFRUUCBNRVRIT0QgfCBQQVRIIHwgVVNBR0UKLS0tIHwgLS0tIHwgLS0tCkdFVHwgc3BhcmstbmV0d29ya3MtY2xpZW50L2hvbWUgfCBDbGllbnQgaG9tZSBwYWdlClBPU1R8IHNwYXJrLW5ldHdvcmtzLWNsaWVudC9maWx0ZXIgfCBDbGllbnQgZmlsdGVyIGFwaQpHRVR8IHNwYXJrLW5ldHdvcmtzLWNsaWVudC8gfCBDbGllbnQgbG9naW4gcGFnZQpQT1NUfCBzcGFyay1uZXR3b3Jrcy1jbGllbnQvbG9naW4gfCBDbGllbnQgbG9naW4gb3BlcmF0aW9uIGFwaQpQT1NUfCBzcGFyay1uZXR3b3Jrcy1iYWNrZW5kL2ZpbHRlci9sb2dpbiB8IEJhY2tlbmQgbG9naW4gcmVzdCBhcGkKUE9TVHwgc3BhcmstbmV0d29ya3MtYmFja2VuZC9maWx0ZXIgfCBCYWNrZW5kIGZpbHRlciByZXN0IGFwaQpHRVR8IHNwYXJrLW5ldHdvcmtzLWJhY2tlbmQvZmlsdGVyIHwgQmFja2VuZCBnZXRpbmcgYWxsIHJlc3QgYXBpCgoKIyMjIEh0dHBzIFN0YXR1cwotIDIwMCBPSzogVGhlIHJlcXVlc3QgaGFzIHN1Y2NlZWRlZAotIDQwMCBCYWQgUmVxdWVzdDogVGhlIHJlcXVlc3QgY291bGQgbm90IGJlIHVuZGVyc3Rvb2QgYnkgdGhlIHNlcnZlcgotIDQwNCBOb3QgRm91bmQ6IFRoZSByZXF1ZXN0ZWQgcmVzb3VyY2UgY2Fubm90IGJlIGZvdW5kCi0gNDE3IEJhZCBFeGNlcHRpb246IEludGVybmFsIHNlcnZlciBleGNlcHRpb24KLSA1MDAgSW50ZXJuYWwgU2VydmVyIEVycm9yOiBUaGUgc2VydmVyIGVuY291bnRlcmVkIGFuIHVuZXhwZWN0ZWQgY29uZGl0aW9uCg== readmeEtag: '"36d217b49cca85b8cf14817b4a62db19c07d856b"' readmeLastModified: Fri, 20 Mar 2020 14:10:29 GMT repositoryId: 240132698 description: Filter matching task by spark networks created: '2020-02-12T22:58:53Z' updated: '2020-12-14T18:04:21Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: aqib1 logo: https://avatars.githubusercontent.com/u/8742169?v=4 repoEtag: '"298d1aa67e589c633e3f13b3e0c2f5eb07ea94c2cde77e022d1838382c83d5b5"' repoLastModified: Mon, 14 Dec 2020 18:04:21 GMT foundInMaster: true category: Server Implementations id: a9ae36422794afcf4ffd1ea0b1bb086f - source: openapi3 tags repository: https://github.com/pagopa-archive/pdnd-openapi-server v3: true repositoryMetadata: base64Readme: >- IyBQRE5EIG9wZW4gYXBpIHNlcnZlciAocHJlLWFscGhhKQoKVGhpcyBzZXJ2ZXIgZXhwb3NlcyBBUEkgZnJvbSBbUERORF0oaHR0cHM6Ly9kYXRhcG9ydGFsLmRhZi50ZWFtZGlnaXRhbGUuaXQvKSBhY2NvcmRpbmcgdG8gb3BlbmFwaSAzIHNwZWNpZmljYXRpb24gYW5kIHRoZSBpbnRlcm9wZXJhYmlsaXR5IG1vZGVsIGRlZmluZWQgYnkgSXRhbGlhbiBHb3Zlcm5lbW50IG9yZ2FuaXphdGlvbnMuIEl0IHVzZXMgW2Nvbm5leGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL3phbGFuZG8vY29ubmV4aW9uKSBmcmFtZXdvcmsgZm9yIGNyZWF0aW5nIGFuZCBleHBvc2luZyB0aGUgc2VydmVyIGluIGEgY29udHJhY3QgZmlyc3QgYXBwcm9hY2guIFJlbWVtYmVyIGVhY2ggQVBJIGlzIHNlbGYgZG9jdW1lbnRlZCBpZiB5b3Ugd2FudCB0byByZWFkIHRoZSBkZXRhaWxzIG9mIGVhY2ggQVBJIHJlYWQgdGhlIFtvcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vdGVhbWRpZ2l0YWxlL3BkbmQtb3BlbmFwaS1zZXJ2ZXIvYmxvYi9tYXN0ZXIvb3BlbmFwaS9kYWYtb3BlbmFwaS55YW1sKSBmaWxlIG9yIGxhdW5jaCB0aGUgc2VydmljZSBhcyBkZXNjcmliZWQgYmVsb3cuIAoKIyMjIENvbnRleHQKClRoZSBnb2FsIG9mIHRoaXMgcHJvamVjdCBpcyB0byBleHBvc2UgYWNjb3JkaW5nIHRvIG9wZW5hcGkgMyBzdGFuZGFyZHMgYSBzZXQgb2YgQVBJIGRldmVsb3BlZCBmb3IgW1BETkQgKGV4IERhZildKGh0dHA6Ly9kYXRhcG9ydGFsLmRhZi50ZWFtZGlnaXRhbGUuaXQpLiBUaGUgb2xkIEFQSSB3ZXJlIGRldmVsb3BlZCBhY2NvcmRpbmcgdG8gc3dhZ2dlciAyLjAgc3RhbmRhcmRzLCB1c2luZyBTY2FsYS4gV2UgYXJlIG5vdyBwcm94eWluZyB0aG9zZSBhcGkgdGhyb3VnaHQgdGhpcyByZXBvIGZvciBiZWNvbWluZyBjb21wbGlhbnRzIHdpdGggdGhlIG5ldyBzdGFuZGFyZHMuIEJlbG93IHRoZSBsaXN0IG9mIGFwaSB3ZSBhcmUgd3JhcHBpbmc6CgotICBodHRwczovL2dpdGh1Yi5jb20vaXRhbGlhL2RhZi1kYXRhcG9ydGFsLWJhY2tlbmQvYmxvYi9tYXN0ZXIvY29uZi9mdGRfYXBpLnlhbWwKLSBodHRwczovL2dpdGh1Yi5jb20vaXRhbGlhL2RhZi1zcnYtY2F0YWxvZy9ibG9iL2Rldi9jb25mL2NhdGFsb2dfbWFuYWdlci55YW1sCi0gaHR0cHM6Ly9naXRodWIuY29tL2l0YWxpYS9kYWYtc3J2LXNlY3VyaXR5L2Jsb2IvZGV2L2NvbmYvc2VjdXJpdHlfbWFuYWdlci55YW1sCgojIyMgUHJlcmVxdWlzaXRlcwoKT24geW91ciBtYWNoaW5lIHB5dGhvbiAzIHdpdGggcGlwIGFuZCBbdG94XShodHRwczovL3RveC5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvKSBtdXN0IGJlIGluc3RhbGxlZAoKIyMjIFRlc3QKClRvIHRlc3QgdGhlIEFQSSwganVzdCBydW4KCgpgYGAKdG94IAoKYGBgCgpXaGVuIGRlYnVnZ2luZywgeW91IGNhbiBydW4gdGhlIGZvbGxvd2luZyBpbnN0ZWFkLgoKCmBgYAp0b3ggLS0gIC0tcGRiIC0tcGRiLWZhaWx1cmUgLXZzIC0tbm9sb2djYXB0dXJlCgpgYGAKCgojIyMgTGF1bmNoCgpgYGAKdG94IC1lIHJ1bgpgYGAKY29ubmVjdCB0byBbaHR0cDovL2xvY2FsaG9zdDo4MDgwL3BkbmQtb3BlbmFwaS91aS9dKGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9wZG5kLW9wZW5hcGkvdWkvKQoKIyMjIERvY2tlciAKCkRvY2tlciBNVVNUIGJlIGluc3RhbGxlZCBvbiB5b3VyIGxvY2FsIG1hY2hpbmUKCmBgYApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL3RlYW1kaWdpdGFsZS9wZG5kLW9wZW5hcGktc2VydmVyLmdpdApjZCBwZG5kLW9wZW5hcGktc2VydmVyCmRvY2tlciBidWlsZCAtdCBwZG5kLW9wZW5hcGktc2VydmVyIC4KZG9ja2VyIHJ1biAtcCA4MDgwOjgwODAgcGRuZC1vcGVuYXBpLXNlcnZlcgpgYGAKb3BlbiB5b3VyIGJyb3dzZXIgYXQgW2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9wZG5kLW9wZW5hcGkvdWkvXShodHRwOi8vbG9jYWxob3N0OjgwODAvcGRuZC1vcGVuYXBpL3VpLykKCiMjIyBUdXRvcmlhbAoKVGhlIHNlcnZpY2UgZXhwb3NlIHNvbWUgQVBJIGZyb20gUERORC4gSXQgcmVxdWlyZXMgYXV0aGVudGljYXRpb24gdXNpbmcgQmFzaWMgQXV0aCBvciBCZWFyZXIgbWVhbmluZyB5b3UgYXMgdXNlciBtdXN0IGJlIHJlZ2lzdGVyZWQgW2hlcmVdKGh0dHBzOi8vZGF0YXBvcnRhbC5kYWYudGVhbWRpZ2l0YWxlLml0LyMvcmVnaXN0ZXIpLCBhbmQgbXVzdCBiZSB1c2VkIHJlc3BlY3RpbmcgdGhlIHRocm90dGxpbmcgaGVhZGVycyBkZWZpbmVkIGluIGFjY29yZGFuY2Ugd2l0aCB0aGUgTGluZWUgR3VpZGEgb2YgdGhlIG1vZGVsIG9mIGludGVyb3BlcmFiaWxpdHkgdGhhdCB3aWxsIGJlIHJlbGVhc2VkIGJ5IHRoZSBJdGFsaWFuIEdvdmVybm1lbnQgb3JnYW5pemF0aW9ucy4KCkFmdGVyIHJlZ2lzdGVyZCB5b3UgY2FuIGluc2VyIHlvdXIgRW1haWwgYW5kIHBhc3N3b3JkOgoKIVtCYXNpYyBBdXRoXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vdGVhbWRpZ2l0YWxlL3BkbmQtb3BlbmFwaS1zZXJ2ZXIvbWFzdGVyL3R1dG9yaWFsL2ltZy9iYXNpY19hdXRoLnBuZykKCk9uY2UgbG9nZ2VkIGluIHdpdGggYmFzaWMgYXV0aCB5b3UgY2FuIGdldCB0aGUgdG9rZW4gYXMgaW4gdGhlIGltYWdlIGJlbG93IGNvcHkgb25seSB0aGUgand0IHZhbHVlCgohW0dldCBqd3RdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS90ZWFtZGlnaXRhbGUvcGRuZC1vcGVuYXBpLXNlcnZlci9tYXN0ZXIvdHV0b3JpYWwvaW1nL2dldF9qd3QucG5nKQoKQ29weSBhbmQgcGFzdGUgdGhlIGp3dCBhbmQgaW5zZXJ0IGl0IGZvciBjYWxsaW5nIGVhY2ggQVBJOgoKIVtKd3QgQXV0aF0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3RlYW1kaWdpdGFsZS9wZG5kLW9wZW5hcGktc2VydmVyL21hc3Rlci90dXRvcmlhbC9pbWcvand0X2F1dGgucG5nKQoKTm93IHlvdSBjYW4gY2FsbCB0aGUgQVBJIGFzIGluIHRoZSBpbWFnZSBiZWxvdwoKIVtTZWFyY2ggQVBJXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vdGVhbWRpZ2l0YWxlL3BkbmQtb3BlbmFwaS1zZXJ2ZXIvbWFzdGVyL3R1dG9yaWFsL2ltZy9zZWFyY2gucG5nKQoKCgo= readmeEtag: '"d3e9c1b302162c77afc183d00e807750393dae60"' readmeLastModified: Fri, 07 Feb 2020 12:15:43 GMT repositoryId: 179264077 description: >- PDND openapi server is a server for exposing PDND (ex DAF) api following the openapi 3 standard and the interoperability model created: '2019-04-03T10:10:10Z' updated: '2023-03-29T13:34:56Z' language: Python archived: true stars: 1 watchers: 9 forks: 1 owner: pagopa-archive logo: https://avatars.githubusercontent.com/u/127237556?v=4 repoEtag: '"2f498b31ccf5d61c107ffa34e6f063b7de330256dbe3fe184793840f2282ecea"' repoLastModified: Wed, 29 Mar 2023 13:34:56 GMT foundInMaster: true category: Server Implementations id: 4a40f7044edf105fefcc49e471132ff1 oldLocations: - https://github.com/pagopa/pdnd-openapi-server - source: openapi3 tags repository: https://github.com/lsmhun/demo-api-contract v3: true repositoryMetadata: base64Readme: >- IyBTcHJpbmcgQ2xvdWQgQ29udHJhY3QgZGVtbwpbIVtGT1NTQSBTdGF0dXNdKGh0dHBzOi8vYXBwLmZvc3NhLmNvbS9hcGkvcHJvamVjdHMvZ2l0JTJCZ2l0aHViLmNvbSUyRmxzbWh1biUyRmRlbW8tYXBpLWNvbnRyYWN0LnN2Zz90eXBlPXNoaWVsZCldKGh0dHBzOi8vYXBwLmZvc3NhLmNvbS9wcm9qZWN0cy9naXQlMkJnaXRodWIuY29tJTJGbHNtaHVuJTJGZGVtby1hcGktY29udHJhY3Q/cmVmPWJhZGdlX3NoaWVsZCkKWyFbQ2lyY2xlQ0ldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL2xzbWh1bi9kZW1vLWFwaS1jb250cmFjdC5zdmc/c3R5bGU9c2hpZWxkKV0oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvbHNtaHVuL2RlbW8tYXBpLWNvbnRyYWN0KQoKClRoaXMgaXMgYSBzbWFsbCBkZW1vIG9mIGNvbnRyYWN0IGJhc2VkIGFwcGxpY2F0aW9uIGRldmVsb3BtZW50IHdpdGggW1NwcmluZyBDbG91ZCBDb250cmFjdF0oaHR0cHM6Ly9jbG91ZC5zcHJpbmcuaW8vc3ByaW5nLWNsb3VkLWNvbnRyYWN0LykuIAoKIyMgRG9jdW1lbnRhdGlvbgotIFtFbmdsaXNoIGRvY3VtZW50YXRpb25dKC4vZG9jcy9kZXNjcl9lbi5tZCkKLSBbSHVuZ2FyaWFuIGRvY3VtZW50YXRpb25dKC4vZG9jcy9kZXNjcl9odS5tZCkKCgojIyBMaWNlbnNlClshW0ZPU1NBIFN0YXR1c10oaHR0cHM6Ly9hcHAuZm9zc2EuY29tL2FwaS9wcm9qZWN0cy9naXQlMkJnaXRodWIuY29tJTJGbHNtaHVuJTJGZGVtby1hcGktY29udHJhY3Quc3ZnP3R5cGU9bGFyZ2UpXShodHRwczovL2FwcC5mb3NzYS5jb20vcHJvamVjdHMvZ2l0JTJCZ2l0aHViLmNvbSUyRmxzbWh1biUyRmRlbW8tYXBpLWNvbnRyYWN0P3JlZj1iYWRnZV9sYXJnZSkK readmeEtag: '"5995966a29930531ed5b925a42bad290e3d60bc7"' readmeLastModified: Sat, 27 May 2023 21:16:44 GMT repositoryId: 425954792 description: Spring Cloud Contract demo created: '2021-11-08T18:44:29Z' updated: '2021-12-09T05:35:43Z' language: Java archived: false stars: 1 watchers: 1 forks: 2 owner: lsmhun logo: https://avatars.githubusercontent.com/u/49127995?v=4 license: Apache-2.0 repoEtag: '"725dd065fbbd0aef179caf0534c58b8664a5804e61ba6977f2e7b7f12d5bcc51"' repoLastModified: Thu, 09 Dec 2021 05:35:43 GMT foundInMaster: true category: Server id: 63ba35052b44216d4636fc366beaa5f6 - source: openapi3 tags repository: https://github.com/tryptichon/java11-jakarta-rs-swagger-redoc-demows v3: true repositoryMetadata: base64Readme: >- IyBERU1PIFdTCgpBbiBKYWthcnRhLUpBWC1SUyBza2VsZXRvbiB3ZWJhcHAgdXNpbmc6CgoqIEpha2FydGEtSkFYLVJTIHdpdGggYW5ub3RhdGlvbnMuCiogU3dhZ2dlciBhbm5vdGF0aW9ucyBmb3IgYXV0b21hdGljIG9wZW5hcGkuanNvbi8ueWFtbCBnZW5lcmF0aW9uLgoqIFJlRG9jIHN0YW5kYWxvbmUgdG8gZ2VuZXJhdGUgYSBkb2N1bWVudGF0aW9uIHdlYnNpdGUgZm9yIHRoZSBhcGkuCiogRG9ja2VyIHRvIGNyZWF0ZSBhIHNlbGYtY29udGFpbmVkIGltYWdlIHVzaW5nIHRvbWNhdCAxMC4KCkJlc3QgdG8gYmUgdXNlZCB3aXRoIEludGVsbGlKIElERUEgYW5kIEphdmEgMTEuCgojIyBIYW5kbGluZyBkb2NrZXIgaW1hZ2VzCgpVc2luZyBtYXZlbiB0byBoYW5kbGUgZG9ja2VyIGltYWdlIGNyZWF0aW9uIGFuZCBjbGVhbnVwIGlzIGRvbmUgYnkganVzdCBhY3RpdmF0aW5nIHRoZSBwcm9maWxlIGBkb2NrZXJgLiBUaGlzCmNhbiBiZSBhY3RpdmF0ZWQgdmlhIGAtRGRvY2tlcj10cnVlYCBvbiB0aGUgY29tbWFuZGxpbmUuCgpDcmVhdGUgYSBkb2NrZXIgaW1hZ2U6CgpgYGBzaGVsbAptdm4gcGFja2FnZSAtRGRvY2tlcj10cnVlCmBgYAoKQ2xlYW51cCBhbmQgcmVtb3ZlIHRoZSBkb2NrZXIgaW1hZ2U6CgpgYGBzaGVsbAptdm4gY2xlYW4gLURkb2NrZXI9dHJ1ZQpgYGAKClNlZSBbcG9tLnhtbF0ocG9tLnhtbCkgb24gaG93IHRoaXMgaXMgZG9uZS4KCkEgc2FtcGxlIFtkb2NrZXItY29tcG9zZS55YW1sXShkb2NrZXIvZG9ja2VyLWNvbXBvc2UueWFtbCkgaXMgaW5jbHVkZWQgdG8gc3RhcnQgYSBjb250YWluZXIgZWFzaWx5LiBUaGUKcGFyYW1ldGVyIGBpbWFnZWAgdGhlcmUgbmVlZHMgdG8gYmUgYWRqdXN0ZWQgYnkgaGFuZCB3aGVuIHRoZSBwcm9qZWN0IG5hbWUgLyB2ZXJzaW9uIGNoYW5nZXMuCgpBZnRlciB0aGUgY29udGFpbmVyIGhhcyBzdGFydGVkLCBwb2ludCB5b3VyIGJyb3dzZXIgdG8gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9gLgoKIyMgVXNlZCB0b29scyBhbmQgbGljZW5zZXMKCkZvciBtb3JlIGluZm9ybWF0aW9uIGFib3V0IHN3YWdnZXIgYW5kIGFkZGl0aW9uYWwgbGljZW5zZXM6CgojIyMjIFN3YWdnZXIKCiogW1N3YWdnZXIgSU9dKGh0dHBzOi8vc3dhZ2dlci5pby8pCiogW0xpY2Vuc2VdKGh0dHBzOi8vc3dhZ2dlci5pby9saWNlbnNlLykKCiMjIyMgUmVEb2MKCiogW0dpdEh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkpCiogW09mZmljaWFsIHdlYnNpdGVdKGh0dHBzOi8vcmVkb2NseS5jb20vcmVkb2MpCiogW0RlbW8gc2l0ZV0oaHR0cHM6Ly9yZWRvY2x5LmdpdGh1Yi5pby9yZWRvYy8pCiogW0xpY2Vuc2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jL2Jsb2IvbWFzdGVyL0xJQ0VOU0UpCgojIyMjIERvY2tlcgoKKiBbT2ZmaWNpYWwgd2Vic2l0ZV0oaHR0cHM6Ly93d3cuZG9ja2VyLmNvbS8pCiogW0ZBUSBhYm91dCBsaWNlbnNpbmcgYW5kIHByaWNpbmddKGh0dHBzOi8vd3d3LmRvY2tlci5jb20vcHJpY2luZy9mYXEvKQoKIyMjIyBUb21jYXQKCiogW09mZmljaWFsIHdlYnNpdGVdKGh0dHBzOi8vdG9tY2F0LmFwYWNoZS5vcmcvKQoqIFtMaWNlbnNlXShodHRwczovL3RvbWNhdC5hcGFjaGUub3JnL2xlZ2FsLmh0bWwp readmeEtag: '"85c15fdc7e284da06be26a0b28607fb0f56c3a0a"' readmeLastModified: Tue, 03 May 2022 21:06:07 GMT repositoryId: 484558634 description: An Jakarta-JAX-RS webapp integrating Swagger and ReDOC. created: '2022-04-22T20:08:24Z' updated: '2022-04-29T13:15:36Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: tryptichon logo: https://avatars.githubusercontent.com/u/5908199?v=4 license: MIT repoEtag: '"fc5d09b25a9ec8d188f1eb63648eefb40138882854b106c9628d7a11b905c67a"' repoLastModified: Fri, 29 Apr 2022 13:15:36 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 2dc01ffb64e91dcd50e502dea6d3c884 - source: openapi3 tags repository: https://github.com/kwangsing3/sample_express v3: true id: 085152b5a0eecfcecbd5cb325709538f repositoryMetadata: base64Readme: >- IyAqKmdzc19sb2dfZG9vcm1hbioqCuiiq+WLleaOpeaUtuaXpeiqjOioiuaBr+eahOe2sui3r+W4uOmnkOacjeWLme+8jOioreioiOeUqOS+huaOpeaUtuWQhOeoruagvOW8j+eahOaXpeiqjOS4pumHjee1hOi9ieeZvOiHs0dlbmVzaXMgd2ViQVBJ44CCIDxici8+CgoqRG9vcm1hbiDkuIDoqZ7lj5boh6rploDooZsq44CCCjxici8+PGJyLz4KCgo= readmeEtag: '"55d0d9bc958ed69ba9e11dc4f55d6fb4dfecaa22"' readmeLastModified: Thu, 20 Oct 2022 08:24:09 GMT repositoryId: 554607502 description: 擁有OpenAPI3.0 規範自動生成文件的NodeJS專案範例 created: '2022-10-20T04:48:02Z' updated: '2023-02-01T06:02:28Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: kwangsing3 logo: https://avatars.githubusercontent.com/u/18378969?v=4 license: MIT repoEtag: '"7b223cba1eaf8a435c68c9510d341e3891c415b8a8c52dc475cdf2bc38ecd1cb"' repoLastModified: Wed, 01 Feb 2023 06:02:28 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/lugudu/gestor-medico-spring-jpa v3: true id: b0589ff445d8c6b8d15fca17e784c3ab repositoryMetadata: base64Readme: >- IyBnZXN0b3ItbWVkaWNvLXNwcmluZy1qcGEKIEFQSSBSRVNUIGNvbiBKYXZhIFNwcmluZyBCb290IDMsIFNwcmluZyBTZWN1cml0eSB5IEpQQQo= readmeEtag: '"c2988023b245481580179e948199b0dcf74861af"' readmeLastModified: Fri, 10 Mar 2023 10:12:22 GMT repositoryId: 610233940 description: API REST con SpringBoot 3 y JPA created: '2023-03-06T11:14:31Z' updated: '2023-03-14T11:39:04Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: LuGuDu logo: https://avatars.githubusercontent.com/u/70801680?v=4 repoEtag: '"2068d0353d86548cdb25c304b5a3007c200f76f56df3749eb247201a47294404"' repoLastModified: Tue, 14 Mar 2023 11:39:04 GMT category: - Low-level Tooling - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/2697sahil/contactmanager v3: true id: c1e9c2933115577a7f13758972de1b3a repositoryMetadata: base64Readme: >- IyBDb250YWN0TWFuYWdlcgoKVGhlIENvbnRhY3RNYW5hZ2VyIHByb2plY3QgaXMgYSByb2J1c3QgQVBJIGZvciBtYW5hZ2luZyBjb250YWN0cywgYnVpbHQgd2l0aCBPcGVuQVBJIGFuZCB1dGlsaXppbmcgdGhlIHBvd2VyIG9mIGNvZGUgZ2VuZXJhdGlvbiB3aXRoIFNRTExpdGUgYXMgdGhlIHVuZGVybHlpbmcgZGF0YWJhc2UuCgojIyBGZWF0dXJlcwoKLSAqKkNyZWF0ZSBDb250YWN0Kio6IEVhc2lseSBjcmVhdGUgYSBuZXcgY29udGFjdCBieSBwcm92aWRpbmcgdGhlIHJlcXVpcmVkIGRldGFpbHMgdXNpbmcgYSBzaW1wbGUgUE9TVCByZXF1ZXN0LgoKLSAqKlNlYXJjaCBDb250YWN0cyoqOiBFZmZpY2llbnRseSBzZWFyY2ggZm9yIGNvbnRhY3RzIGJhc2VkIG9uIGZpcnN0IG5hbWUsIGxhc3QgbmFtZSwgb3IgYW55IG90aGVyIHBhcmFtZXRlciBvZiB5b3VyIGNob2ljZSB1c2luZyBhIEdFVCByZXF1ZXN0LgoKLSAqKlVwZGF0ZSBDb250YWN0IERldGFpbHMqKjogU2VhbWxlc3NseSB1cGRhdGUgY29udGFjdCBpbmZvcm1hdGlvbiBieSBzZW5kaW5nIGEgUFVUIHJlcXVlc3Qgd2l0aCB0aGUgY29udGFjdCBJRCBhbmQgdGhlIHVwZGF0ZWQgZGV0YWlscy4KCi0gKipEZWxldGUgQ29udGFjdCoqOiBSZW1vdmUgYSBjb250YWN0IGZyb20gdGhlIGRhdGFiYXNlIHdpdGggYSBERUxFVEUgcmVxdWVzdCB1c2luZyB0aGUgY29udGFjdCBJRC4KCiMjIE9wZW5BUEkgU3BlY2lmaWNhdGlvbgoKVGhlIEFQSSBmb2xsb3dzIHRoZSBPcGVuQVBJIDMuMC4wIHNwZWNpZmljYXRpb24gYW5kIGNhbiBiZSBlYXNpbHkgaW50ZWdyYXRlZCBpbnRvIHlvdXIgcHJvamVjdHMuIFRoZSBBUEkgZGVmaW5pdGlvbiBpcyBjbGVhbiBhbmQgY29uY2lzZSwgbWFraW5nIGl0IHN0cmFpZ2h0Zm9yd2FyZCB0byB1bmRlcnN0YW5kIGFuZCB3b3JrIHdpdGguCgpgYGB5YW1sCm9wZW5hcGk6IDMuMC4wCmluZm86CiAgdGl0bGU6IENvbnRhY3RNYW5hZ2VyIEFQSQogIHZlcnNpb246IDEuMC4wCgpzZXJ2ZXJzOgogIC0gdXJsOiBodHRwOi8vbG9jYWxob3N0OjgwODAvCgojIC4uLiAocGF0aHMsIGNvbXBvbmVudHMsIGV0Yy4pCmBgYAoKIyMgQ29kZSBHZW5lcmF0aW9uCgpUaGUgcHJvamVjdCBzaG93Y2FzZXMgdGhlIHVzZSBvZiBPcGVuQVBJIGNvZGUgZ2VuZXJhdGlvbiB0byBzdHJlYW1saW5lIHRoZSBkZXZlbG9wbWVudCBwcm9jZXNzLiBUaGUgQVBJIGRlZmluaXRpb24gc2VydmVzIGFzIGEgc2luZ2xlIHNvdXJjZSBvZiB0cnV0aCwgYWxsb3dpbmcgYXV0b21hdGljIGdlbmVyYXRpb24gb2Ygc2VydmVyLXNpZGUgYW5kIGNsaWVudC1zaWRlIGNvZGUsIHJlZHVjaW5nIGRldmVsb3BtZW50IHRpbWUgYW5kIHBvdGVudGlhbCBlcnJvcnMuCgojIyBEYXRhYmFzZTogU1FMTGl0ZQoKQ29udGFjdE1hbmFnZXIgdXRpbGl6ZXMgU1FMTGl0ZSBhcyBpdHMgZGF0YWJhc2UsIHByb3ZpZGluZyBhIGxpZ2h0d2VpZ2h0LCBlbWJlZGRlZCByZWxhdGlvbmFsIGRhdGFiYXNlIHNvbHV0aW9uLiBTUUxMaXRlIGlzIGtub3duIGZvciBpdHMgc2ltcGxpY2l0eSBhbmQgZWZmaWNpZW5jeSwgbWFraW5nIGl0IGEgcGVyZmVjdCBmaXQgZm9yIHByb2plY3RzIHdpdGggbW9kZXJhdGUgZGF0YSBzdG9yYWdlIG5lZWRzLgoKIyMgR2V0dGluZyBTdGFydGVkCgoxLiBDbG9uZSB0aGUgcmVwb3NpdG9yeS4KMi4gQ29uZmlndXJlIHlvdXIgU1FMTGl0ZSBkYXRhYmFzZS4KMy4gUnVuIHRoZSBzZXJ2ZXIuCjQuIFN0YXJ0IG1hbmFnaW5nIHlvdXIgY29udGFjdHMgd2l0aCBlYXNlLgoKRmVlbCBmcmVlIHRvIGV4cGxvcmUgdGhlIE9wZW5BUEkgZGVmaW5pdGlvbiBpbiB0aGUgYG9wZW5hcGkueWFtbGAgZmlsZSBhbmQgbGV2ZXJhZ2UgdGhlIHBvd2VyIG9mIGNvZGUgZ2VuZXJhdGlvbiBmb3IgYSBzbW9vdGggZGV2ZWxvcG1lbnQgZXhwZXJpZW5jZS4gSWYgeW91IGVuY291bnRlciBhbnkgaXNzdWVzIG9yIGhhdmUgc3VnZ2VzdGlvbnMsIHBsZWFzZSBvcGVuIGFuIGlzc3VlIG9yIHN1Ym1pdCBhIHB1bGwgcmVxdWVzdC4gSGFwcHkgY29kaW5nIQ== readmeEtag: '"978351d12c8d1946e3a5f39f69c9e4f72d7661c7"' readmeLastModified: Sun, 25 Feb 2024 16:03:36 GMT repositoryId: 743109520 description: >- The ContactManager project is a robust API for managing contacts, built with OpenAPI and utilizing the power of code generation with SQLLite as the underlying database. created: '2024-01-14T11:11:12Z' updated: '2024-01-28T07:01:17Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: 2697sahil logo: https://avatars.githubusercontent.com/u/67955445?v=4 repoEtag: '"25362851d15f89fb982cc9c3b5a61da22c74ebd3df359690c82753ced58afc35"' repoLastModified: Sun, 28 Jan 2024 07:01:17 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/zikzakjack/myrefcards v3: true id: 6f5ecf2cbb0044d67eda5706700ca230 repositoryMetadata: base64Readme: >- IyBNeVJlZmNhcmRzCgpJIGludGVuZCB0byB1c2UgdGhpcyByZXBvIHRvIGFyY2hpdmUgbXkgcmVmY2FyZHMgd2hpY2ggd291bGQgYmUgZnJlcXVlbnRseSB1c2VkIGluIGRheSB0byBkYXkgcm91dGluZXMuCgoxLiBbTGludXhdKE15UmVmY2FyZHNfT1NfTGludXgubWQpCjIuIFtHaXRdKE15UmVmY2FyZHNfU0NNX0dpdC5tZCkKMy4gW0JpZ0RhdGEgLSBIaXZlXShNeVJlZmNhcmRzX0JpZ0RhdGFfSGl2ZS5tZCkKNC4gW0JpZ0RhdGEgLSBQaWddKE15UmVmY2FyZHNfQmlnRGF0YV9QaWcubWQpCjUuIFtCaWdEYXRhIC0gU3Fvb3BdKE15UmVmY2FyZHNfQmlnRGF0YV9TcW9vcC5tZCkKNS4gW0JpZ0RhdGEgLSBLYWZrYV0oTXlSZWZjYXJkc19CaWdEYXRhX0thZmthLm1kKQo2LiBbSmF2YSAtIEpQQV0oTXlSZWZjYXJkc19KYXZhX0pQQS5tZCkKNy4gW0phdmEgLSBTcHJpbmddKE15UmVmY2FyZHNfSmF2YV9TcHJpbmcubWQpCjcuIFtKYXZhIC0gT3BlbkFQSV0oTXlSZWZjYXJkc19KYXZhX09wZW5BUEkubWQpCgpXaWtpIFRpcHMgOgoKMS4gaHR0cHM6Ly9naXRodWIuY29tL2FkYW0tcC9tYXJrZG93bi1oZXJlL3dpa2kvTWFya2Rvd24tQ2hlYXRzaGVldAoyLiBodHRwczovL2d1aWRlcy5naXRodWIuY29tL2ZlYXR1cmVzL21hc3RlcmluZy1tYXJrZG93bi8K readmeEtag: '"b58d12aaa351dc0888d534d8c336186d9cdb83ba"' readmeLastModified: Thu, 30 Nov 2023 20:01:47 GMT repositoryId: 66270941 description: My CheatSheets about Java, Spring Boot, BigData .... created: '2016-08-22T12:30:04Z' updated: '2024-09-10T13:32:58Z' language: null archived: false stars: 1 watchers: 1 forks: 2 owner: zikzakjack logo: https://avatars.githubusercontent.com/u/15928270?v=4 repoEtag: '"adea7b29c1657b2d4456b2f244c87c38c38f648a00849c245680bd5ded053aa6"' repoLastModified: Tue, 10 Sep 2024 13:32:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/smartrecruiters/openapi-schemas-validator v3: true repositoryMetadata: base64Readme: >- IyBAc21hcnRyZWNydWl0ZXJzL29wZW5hcGktc2NoZW1hcy12YWxpZGF0b3IKClshW05QTSBWZXJzaW9uXVtucG0taW1hZ2VdXVtucG0tdXJsXQpbIVtOUE0gRG93bmxvYWRzXVtkb3dubG9hZHMtaW1hZ2VdXVtkb3dubG9hZHMtdXJsXQpbIVtOb2RlLmpzIFZlcnNpb25dW25vZGUtdmVyc2lvbi1pbWFnZV1dW25vZGUtdmVyc2lvbi11cmxdClshW0xpY2VuY2VdW2xpY2Vuc2UtaW1hZ2VdXVtsaWNlbnNlLXVybF0KWyFbQnVpbGRdW3RyYXZpcy1pbWFnZV1dW3RyYXZpcy11cmxdCgpDcmVhdGUgc2NoZW1hIHZhbGlkYXRvcnMgZm9yIGFwaSBkb2N1bWVudGF0aW9uIGluIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLnggZm9ybWF0LgoKIyMgdHY0CgpUaGlzIG1vZHVsZSB1c2VzIFt0djRdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3R2NCkgdmFsaWRhdG9yLCBhZGRpdGlvbmFsbHkKY29uZmlndXJlZCB3aXRoIFt0djQtZm9ybWF0c10oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvdHY0LWZvcm1hdHMpLgoKVGhpcyBtb2R1bGUgdXNlcyB0djQgW2B2YWxpZGF0ZU11bHRpcGxlYF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvdHY0I3VzYWdlLTMtbXVsdGlwbGUtZXJyb3JzKSBmdW5jdGlvbiwKd2l0aCBbYGNoZWNrUmVjdXJzaXZlYF0oaHR0cHM6Ly9naXRodWIuY29tL2dlcmFpbnRsdWZmL3R2NCNjeWNsaWNhbC1qYXZhc2NyaXB0LW9iamVjdHMpIHBhcmFtIGFsd2F5cyBwYXNzZWQgYXMgYHRydWVgLgoKWW91IGNhbiBjb25maWd1cmUgdHY0IHZhbGlkYXRvciB3aXRoIGZvbGxvd2luZyBvcHRpb25zOgogKiBjdXN0b21Gb3JtYXRzIC0gd2lsbCBiZSBwYXNzZWQgdG8gW2B0djQuYWRkRm9ybWF0YF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvdHY0I2FkZGZvcm1hdGZvcm1hdC12YWxpZGF0aW9uZnVuY3Rpb24pIGZ1bmN0aW9uCiAqIGJhblVua25vd25Qcm9wZXJ0aWVzIC0gd2lsbCBiZSBwYXNzZWQgdG8gdmFsaWRhdGluZyBmdW5jdGlvbiAoW1RoZSBiYW5Vbmtub3duUHJvcGVydGllcyBmbGFnXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS90djQjdGhlLWJhbnVua25vd25wcm9wZXJ0aWVzLWZsYWcpKQoKVGhlcmUgaXMgZXhwb3J0ZWQgYW4gYEVycm9yQ29kZXNgIG9iamVjdCBjb250YWluaW5nIG1hcHBpbmdzIGJldHdlZW4gZXJyb3IgY29kZSBudW1iZXJzIGFuZCBlcnJvciBuYW1lcyB0aGF0IGFyZSBnZW5lcmF0ZWQgYnkgW3R2NF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvdHY0KSB2YWxpZGF0b3IuCgojIyBPcGVuQVBJIFNwZWNpZmljYXRpb24gMy4wIHNwZWNpZmljIGZlYXR1cmVzCgpDdXJyZW50bHkgQHNtYXJ0cmVjcnVpdGVycy9vcGVuYXBpLXNjaGVtYXMtdmFsaWRhdG9yIHN1cHBvcnRzIGBudWxsYWJsZWAgZmllbGQuCgpQbGVhc2UgcmVmZXIgdG8gaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjEubWQjZml4ZWQtZmllbGRzLTIwLgoKIyMgTGljZW5zZQoKW01JVF0oTElDRU5TRSkKCltucG0taW1hZ2VdOiBodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L0BzbWFydHJlY3J1aXRlcnMvb3BlbmFwaS1zY2hlbWFzLXZhbGlkYXRvci5zdmcKW25wbS11cmxdOiBodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9Ac21hcnRyZWNydWl0ZXJzL29wZW5hcGktc2NoZW1hcy12YWxpZGF0b3IKW2Rvd25sb2Fkcy1pbWFnZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL2RtL0BzbWFydHJlY3J1aXRlcnMvb3BlbmFwaS1zY2hlbWFzLXZhbGlkYXRvci5zdmcKW2Rvd25sb2Fkcy11cmxdOiBodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9Ac21hcnRyZWNydWl0ZXJzL29wZW5hcGktc2NoZW1hcy12YWxpZGF0b3IKW25vZGUtdmVyc2lvbi1pbWFnZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbm9kZS92L0BzbWFydHJlY3J1aXRlcnMvb3BlbmFwaS1zY2hlbWFzLXZhbGlkYXRvci5zdmcKW25vZGUtdmVyc2lvbi11cmxdOiBodHRwczovL25vZGVqcy5vcmcvZW4vZG93bmxvYWQvCltsaWNlbnNlLXVybF06IGh0dHBzOi8vZ2l0aHViLmNvbS9zbWFydHJlY3J1aXRlcnMvb3BlbmFwaS1zY2hlbWFzLXZhbGlkYXRvci9ibG9iL21hc3Rlci9MSUNFTlNFCltsaWNlbnNlLWltYWdlXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vbC9Ac21hcnRyZWNydWl0ZXJzL29wZW5hcGktc2NoZW1hcy12YWxpZGF0b3Iuc3ZnClt0cmF2aXMtdXJsXTogaHR0cHM6Ly90cmF2aXMtY2kub3JnL3NtYXJ0cmVjcnVpdGVycy9vcGVuYXBpLXNjaGVtYXMtdmFsaWRhdG9yClt0cmF2aXMtaW1hZ2VdOiBodHRwczovL2FwaS50cmF2aXMtY2kub3JnL3NtYXJ0cmVjcnVpdGVycy9vcGVuYXBpLXNjaGVtYXMtdmFsaWRhdG9yLnN2Zz9icmFuY2g9bWFzdGVyCiMjIEFQSQoKCiMjIyBNb2R1bGVzCgo8ZGw+CjxkdD48YSBocmVmPSIjbW9kdWxlX0BzbWFydHJlY3J1aXRlcnMvb3BlbmFwaS1zY2hlbWFzLXZhbGlkYXRvciI+QHNtYXJ0cmVjcnVpdGVycy9vcGVuYXBpLXNjaGVtYXMtdmFsaWRhdG9yPC9hPjwvZHQ+CjxkZD48L2RkPgo8L2RsPgoKIyMjIENsYXNzZXMKCjxkbD4KPGR0PjxhIGhyZWY9IiNTY2hlbWFWYWxpZGF0b3IiPlNjaGVtYVZhbGlkYXRvcjwvYT48L2R0Pgo8ZGQ+PC9kZD4KPC9kbD4KCjxhIG5hbWU9Im1vZHVsZV9Ac21hcnRyZWNydWl0ZXJzL29wZW5hcGktc2NoZW1hcy12YWxpZGF0b3IiPjwvYT4KCiMjIyBAc21hcnRyZWNydWl0ZXJzL29wZW5hcGktc2NoZW1hcy12YWxpZGF0b3IKPGEgbmFtZT0iU2NoZW1hVmFsaWRhdG9yIj48L2E+CgojIyMgU2NoZW1hVmFsaWRhdG9yCioqS2luZCoqOiBnbG9iYWwgY2xhc3MgIAoKKiBbU2NoZW1hVmFsaWRhdG9yXSgjU2NoZW1hVmFsaWRhdG9yKQogICAgKiBbbmV3IFNjaGVtYVZhbGlkYXRvcihzcGVjLCBbb3B0aW9uc10pXSgjbmV3X1NjaGVtYVZhbGlkYXRvcl9uZXcpCiAgICAqIFsudmFsaWRhdGUoZW50aXR5LCBzY2hlbWEpXSgjU2NoZW1hVmFsaWRhdG9yK3ZhbGlkYXRlKSDih5IgPGNvZGU+QXJyYXkuJmx0O09iamVjdCZndDs8L2NvZGU+Cgo8YSBuYW1lPSJuZXdfU2NoZW1hVmFsaWRhdG9yX25ldyI+PC9hPgoKIyMjIyBuZXcgU2NoZW1hVmFsaWRhdG9yKHNwZWMsIFtvcHRpb25zXSkKQ3JlYXRlIHNjaGVtYSB2YWxpZGF0b3IgZm9yIHNjaGVtYXMgZGVmaW5lZCBpbiBzcGVjIGF0IGAjL2NvbXBvbmVudHMvc2NoZW1hc2AKCgp8IFBhcmFtIHwgVHlwZSB8IERlZmF1bHQgfCBEZXNjcmlwdGlvbiB8CnwgLS0tIHwgLS0tIHwgLS0tIHwgLS0tIHwKfCBzcGVjIHwgPGNvZGU+T2JqZWN0PC9jb2RlPiB8ICB8IEFQSSBzcGVjaWZpY2F0aW9uIGluIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLjAgZm9ybWF0IHwKfCBbb3B0aW9uc10gfCA8Y29kZT5PYmplY3Q8L2NvZGU+IHwgPGNvZGU+e308L2NvZGU+IHwgb3B0aW9ucyB8CnwgW29wdGlvbnMuY3VzdG9tRm9ybWF0c10gfCA8Y29kZT5PYmplY3Q8L2NvZGU+IHwgIHwgY3VzdG9tIGZvcm1hdCB2YWxpZGF0b3JzIHwKfCBbb3B0aW9ucy5iYW5Vbmtub3duUHJvcGVydGllc10gfCA8Y29kZT5ib29sZWFuPC9jb2RlPiB8IDxjb2RlPmZhbHNlPC9jb2RlPiB8IGRpc2FsbG93IGV4dHJhIHByb3BlcnRpZXMgaW4gdmFsaWRhdGVkIG9iamVjdHMgfAoKPGEgbmFtZT0iU2NoZW1hVmFsaWRhdG9yK3ZhbGlkYXRlIj48L2E+CgojIyMjIHNjaGVtYVZhbGlkYXRvci52YWxpZGF0ZShlbnRpdHksIHNjaGVtYSkg4oeSIDxjb2RlPkFycmF5LiZsdDtPYmplY3QmZ3Q7PC9jb2RlPgpWYWxpZGF0ZSBvYmplY3QgYWdhaW5zdCBzY2hlbWEuIFNjaGVtYSBjYW4gYmUgcGFzc2VkIGV4cGxpY2l0bHkgb3IgcmVmZXJlbmNlIGEgc2NoZW1hIGZyb20gc3BlYy4KCioqS2luZCoqOiBpbnN0YW5jZSBtZXRob2Qgb2YgWzxjb2RlPlNjaGVtYVZhbGlkYXRvcjwvY29kZT5dKCNTY2hlbWFWYWxpZGF0b3IpICAKKipSZXR1cm5zKio6IDxjb2RlPkFycmF5LiZsdDtPYmplY3QmZ3Q7PC9jb2RlPiAtIEFuIGFycmF5IHdpdGggZXJyb3JzIHdoZW4gb2JqZWN0IGlzIGludmFsaWQsIGB1bmRlZmluZWRgIG90aGVyd2lzZSAgCgp8IFBhcmFtIHwgVHlwZSB8IERlc2NyaXB0aW9uIHwKfCAtLS0gfCAtLS0gfCAtLS0gfAp8IGVudGl0eSB8IDxjb2RlPk9iamVjdDwvY29kZT4gfCBvYmplY3QgdG8gdmFsaWRhdGUgfAp8IHNjaGVtYSB8IDxjb2RlPk9iamVjdDwvY29kZT4gXHwgPGNvZGU+c3RyaW5nPC9jb2RlPiB8IG9iamVjdCBzY2hlbWEgfAoK readmeEtag: '"e9b3adc39b6afebd21ef78e97eb6323655f706d4"' readmeLastModified: Tue, 29 Nov 2022 16:15:28 GMT repositoryId: 124135624 description: >- Schema validators for api documentation in OpenAPI Specification 3.x format. created: '2018-03-06T20:39:42Z' updated: '2026-01-30T10:08:03Z' language: JavaScript archived: false stars: 1 watchers: 31 forks: 1 owner: smartrecruiters logo: https://avatars.githubusercontent.com/u/1618540?v=4 license: MIT repoEtag: '"253f3edc7dab7f50d55d4ae72fb9d2e4a0b5c270cadae0d9fb70742c590fa6e1"' repoLastModified: Fri, 30 Jan 2026 10:08:03 GMT foundInMaster: true category: - Data Validators - Parsers id: d77a07afc70fd9f0ff64c98fa79ec7f9 - source: openapi3 tags repository: https://github.com/bbakla/oap_spring_boot3_code_generation v3: true id: 4c8df6edd80e681c509fd53b96094a0d repositoryMetadata: base64Readme: >- VGhlc2UgYXJlIHR3byBTcHJpbmcgQm9vdCBwcm9qZWN0cyBkZXNpZ25lZCB0byBzaG93Y2FzZSBjb2RlIGdlbmVyYXRpb24gYW5kIEFQSSBkb2N1bWVudGF0aW9uIGluIFNwcmluZyBCb290IDMgdXNpbmcgT3BlbkFQSSwgZW1wbG95aW5nIGJvdGggTWF2ZW4gYW5kIEdyYWRsZSBidWlsZCB0b29scy4K readmeEtag: '"8ea764644b2ca5ed1bcfd4d5db9db787cd17bbcc"' readmeLastModified: Sat, 02 Mar 2024 22:46:41 GMT repositoryId: 762329485 description: null created: '2024-02-23T14:55:22Z' updated: '2024-08-21T07:14:56Z' language: Java archived: false stars: 1 watchers: 1 forks: 2 owner: bbakla logo: https://avatars.githubusercontent.com/u/7812609?v=4 repoEtag: '"2f6dd7103d32b9e1cfcb13037a5052e68826029eec61ea76712ac0b7a44db690"' repoLastModified: Wed, 21 Aug 2024 07:14:56 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/puzzle v3: true id: 9d8896537414d70c7c57f9878d6430bb repositoryMetadata: base64Readme: >- PiBbIUlNUE9SVEFOVF0gIAo+IFRoaXMgcmVwb3NpdG9yeSBpcyByZWFkLW9ubHkgLyBhcmNoaXZlZCBhbmQgd2lsbCBub3QgcmVjaWV2ZSB1cGRhdGVzLgoKIyBbVHlwZWRvYyBEb2N1bWVudGF0aW9uIFdlYnNpdGVdKGh0dHBzOi8vc3Vkb2t1cnUuZ2l0aHViLmlvL0JhY2tlbmQvKTxicj4KCiMgVG9kbwoKLSBbIF0gQWRkIGxpY2Vuc2UgZmlsZSBhbmQgZGlzdHJpYnV0ZSB0byBhbGwgcmVwb3MgdXNpbmcgR2l0SHViIEFjdGlvbiAoVGhvbWFzKQotIFsgXSBGaW5pc2ggd3JpdGluZyBpbnRlZ3JhdGlvbiB0ZXN0cyBmb3IgcHV6emxlIGVuZHBvaW50IChUaG9tYXMpCi0gWyBdIFdyaXRlIEdpdEh1YiBob29rIHRvIHJ1biBhbGwgdGVzdHMgYmVmb3JlIFB1c2ggdG8gcmVwbyAoVGhvbWFzKQotIFt4XSBTZXQgdXAgRGV2IGFuZCBQcm9kIExhbWJkYSBlbnZpcm9ubWVudHMgKFRob21hcy9HcmVnb3J5KQotIFsgXSBBZGQgTWVybWFpZCBkb2N1bWVudGF0aW9uIGFuZCBkaXN0cmlidXRlIHRvIGFsbCByZXBvcyB1c2luZyBHaXRIdWIgQWN0aW9uIChUaG9tYXMpCi0gWyBdIENsZWFuIHVwIGRvY2tlciBpbXBsZW1lbnRhdGlvbiAoYXV0by1yZWJ1aWxkKSAoVGhvbWFzKQotIFsgXSBEaXNwbGF5IGludGVncmF0aW9uIHRlc3QgcmVzdWx0cyB3aXRoIHJlcG9ydGVyIChUaG9tYXMpCi0gWyBdIERlY2lkZSBvbiBpbml0aWFsIEpTT04gc3RydWN0dXJlIGZvciByZW1haW5pbmcgZW5kcG9pbnRzIChUZWFtKQotIFsgXSBXcml0ZSBsb2dpYyBmb3IgcmVtYWluaW5nIGVuZHBvaW50cyAoRGFuaWVsKQotIFsgXSBXcml0ZSBzYW5pdGF0aW9uIGFuZCB2YWxpZGF0aW9uIGZvciByZW1haW5pbmcgZW5kcG9pbnRzIChEYW5pZWwpCi0gWyBdIFdyaXRlIFBvc3RtYW4gaW50ZWdyYXRpb24gdGVzdHMgZm9yIHJlbWFpbmluZyBlbmRwb2ludHMgKERhbmllbCkKLSBbIF0gU2V0IHVwIEF1dGgwIHRva2VuIGF1dGhlbnRpY2F0aW9uIChUaG9tYXMvRGFuaWVsKQotIFsgXSBXcml0ZSB1cCBPcGVuQVBJIHNwZWNpZmljYXRpb25zIGZvciBlbmRwb2ludHMgKFRob21hcy9EYW5pZWwpCi0gWyBdIFJlc29sdmUgcmVtYWluaW5nIGBgYC8vdG9kb2BgYCBpdGVtcyAoVGhvbWFzL0RhbmllbCkKLSBbIF0gRGV0ZXJtaW5lIGhvdyB0byBzZXQgUHJvZCBlbnZpcm9ubWVudCB0byB1c2UgdmVyc2lvbmluZyBjb250cm9sIChUaG9tYXMvR3JlZ29yeSkKLSBbIF0gSW1wbGVtZW50IHVuaXQgdGVzdHMgaWYgbmVlZGVkIChUaG9tYXMvRGFuaWVsKQoKIyBEZXZlbG9wZXIgU2V0dXAKCjEuIEluc3RhbGwgRG9ja2VyIG9uIHlvdXIgbWFjaGluZS4gVHV0b3JpYWwgaXMgbGlua2VkIGJlbG93Ojxicj4KICAgWyFbRG9ja2VyIFR1dG9yaWFsXShodHRwczovL2ltZy55b3V0dWJlLmNvbS92aS8yZXpOcXFhU2pxOC8wLmpwZyldKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9MmV6TnFxYVNqcTgpPGJyPgoyLiBUaGUgTW9uZ28gaW1hZ2UgY2FuIGJlIHJ1biB3aXRoIHRoaXMgY29tbWFuZCBpbiB0aGUgcm9vdCBmb2xkZXI6PGJyPgogICBOb3RlIHVzZSBgYGBzdWRvYGBgIG9uIExpbnV4L01hYzxicj4KYGBgY29uc29sZQpucG0gcnVuIGRvY2tlcgpgYGAKMy4gQ3JlYXRlIC5lbnYgZmlsZSB3aXRoIGVudmlyb25tZW50IHZhcmlhYmxlcwo0LiBSdW4gbnBtIGkKNS4gVGhlIGFwcCBjYW4gdGhlbiBiZSBydW4gd2l0aCB0aGUgY29tbWFuZDo8YnI+CmBgYGNvbnNvbGUKbnBtIHJ1biBzdGFydApgYGAKNi4gSW50ZWdyYXRpb24gdGVzdHMgY2FuIGJlIHJ1biB3aGVuIHRoZSBhcHAgaXMgcnVubmluZyB3aXRoIHRoaXMgY29tbWFuZDo8YnI+CmBgYGNvbnNvbGUKbnBtIHJ1biB0ZXN0OmludGVncmF0aW9uCmBgYAoKIyBQb3N0bWFuCgpXZSBhcmUgdXNpbmcgUG9zdG1hbiBmb3IgaW50ZWdyYXRpb24gdGVzdHMuPGJyPgpUaGUgZm9sbG93aW5nIHZpZGVvIGlzIHZlcnkgaGVscGZ1bCBmb3IgdW5kZXJzdGFuZGluZyBob3cgUG9zdG1hbiB3b3JrcyB3aXRoIEdpdEh1Yjo8YnI+ClshW1Bvc3RtYW4gVHV0b3JpYWxdKGh0dHBzOi8vaW1nLnlvdXR1YmUuY29tL3ZpL2NCN21DdVlldUFVLzAuanBnKV0oaHR0cHM6Ly93d3cueW91dHViZS5jb20vd2F0Y2g/dj1jQjdtQ3VZZXVBVSk8YnI+CgojIyMgQmVsb3cgaXMgYW4gZXhhbXBsZSBzdHJ1Y3R1cmUgZm9yIHRlc3RzIGJhc2VkIG9uIFB1enpsZSBlbmRwb2ludApUaGUgUG9zdG1hbiB0ZXN0cyBhcmUgc3RydWN0dXJlZCBiYXNlZCBvbiB0aGUgZW5kcG9pbnQgbmFtZSwgdGhlIHR5cGUgb2YgcmVxdWVzdCwgYW5kIHRoZSBleHBlY3RlZCByZXNwb25zZSBjb2RlPGJyPgoKIVtwb3N0bWFuX3N0cnVjdHVyZS5wbmddKERvY3VtZW50YXRpb24vaW1hZ2VzL3Bvc3RtYW5TdHJ1Y3R1cmUucG5nKTxicj4KClRoaXMgc3RydWN0dXJlIGFsbG93cyB1cyB0byB3cml0ZSB0ZXN0IGNhc2VzIGZvciB0aGUgQ29kZSA0MDAgZm9sZGVyIHdoaWNoIHdvdWxkIGJlIHJ1biBmb3IgYWxsIHRlc3RzIGluIHRoYXQgZm9sZGVyLjxicj4KPGJyPgpBbGwgdGVzdHMgZm9yIHRoZSBTdWRva3VydS1CYWNrZW5kIGhhdmUgYWNjZXNzIHRvIHRoZSBTdWRva3VydS1CYWNrZW5kIFByZS1yZXF1ZXN0IHNjcmlwdHMgYW5kIHRlc3RzLjxicj4KV2UgY3VycmVudGx5IGhhdmUgYSBQcmUtcmVxdWVzdCBzY3JpcHQgdGhhdCBkZWxldGVzIHRoZSB3aG9sZSBkYXRhYmFzZSBiZWZvcmUgZXZlcnkgdGVzdC48YnI+CldlIGFsc28gaGF2ZSBhIGZ1bmN0aW9uIHRoYXQgY2FuIGJlIHVzZWQgZm9yIGFsbCBERUxFVEUgcmVxdWVzdHMgdG8gdmFsaWRhdGUgdGhlIHJlc3BvbnNlLjxicj4KCiFbYmFja2VuZF9wcmVfcmVxdWVzdF9zY3JpcHRzLnBuZ10oRG9jdW1lbnRhdGlvbi9pbWFnZXMvQmFja2VuZC1QcmUtUmVxdWVzdC1TY3JpcHRzLnBuZyk8YnI+CgojIyMjIEluc2lkZSB0aGUgUHV6emxlIGVuZHBvaW50IGZvbGRlciB3ZSBoYXZlIHNoYXJlZCBmdW5jdGlvbnMgdG8gdmFsaWRhdGUgYSBwdXp6bGUgcmVzcG9uc2UgYW5kIHRvIGdlbmVyYXRlIHNoYXJlZCBwdXp6bGUgSlNPTiBmb3IgUHJlLVJlcXVlc3Qgc2NyaXB0cyBmb3Igb3RoZXIgdGVzdHMuPGJyPgoKIVtwdXp6bGVfcmVzcG9uc2VfY2hlY2tlci5wbmddKERvY3VtZW50YXRpb24vaW1hZ2VzL3B1enpsZV9yZXNwb25zZV9jaGVja2VyLnBuZyk8YnI+CiMjIyMgVGhlIGJlbG93IGZ1bmN0aW9ucyB0YWtlIGluIGFuIGludGVnZXIgYW5kIHJldHVybiBlaXRoZXIgdmFsaWQgSlNPTiBvciBhbiBvYmplY3QgZm9yIHJlZmVyZW5jZS48YnI+CiFbc2hhcmVkX3B1enpsZXNfZm9yX3Rlc3RzLnBuZ10oRG9jdW1lbnRhdGlvbi9pbWFnZXMvc2hhcmVkX3B1enpsZXNfZm9yX3Rlc3RzLnBuZyk8YnI+CgojIyMjIEluc2lkZSB0aGUgIkNvZGUgMjAxIiBmb2xkZXIgd2UgaGF2ZSBhIHRlc3QgY2FzZSB0aGF0IHdpbGwgYmUgcnVuIGZvciBhbGwgdGVzdHMgaW5zaWRlIG9mIHRoZSBmb2xkZXIuPGJyPgohW3NoYXJlZF9yZXNwb25zZV9jb2RlX3Rlc3QucG5nXShEb2N1bWVudGF0aW9uL2ltYWdlcy9zaGFyZWRfcmVzcG9uc2VfY29kZV90ZXN0LnBuZyk8YnI+Cg== readmeEtag: '"fc56dbd9e10debb1f3a6ba9bfe9eacafca4cca16"' readmeLastModified: Fri, 23 Aug 2024 00:41:57 GMT repositoryId: 559546165 description: Backend API for Sudokuru created: '2022-10-30T13:07:50Z' updated: '2024-08-23T00:42:40Z' language: TypeScript archived: true stars: 1 watchers: 0 forks: 1 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"b6d29b0ac283f4bf5b43a9c4de01af2c6278646832b6f50e0ea70cde176def59"' repoLastModified: Fri, 23 Aug 2024 00:42:40 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/sudokuru/backend - source: openapi3 tags repository: https://github.com/aosolorzano/city-tasks-lambda-native-with-dynamodb v3: true id: 08b61c6defc33bd561b2ba03a057fa95 repositoryMetadata: base64Readme: >- CiMjIEVEQSB1c2luZyBTcHJpbmcgQm9vdCBhbmQgU3ByaW5nIENsb3VkIEZ1bmN0aW9ucyB0byBkZXBsb3kgbmF0aXZlIGV4ZWN1dGFibGVzIG9uIEVDUyBhbmQgTGFtYmRhIHJlc3BlY3RpdmVseS4KCiogKipBdXRob3IqKjogW0FuZHJlcyBTb2xvcnphbm9dKGh0dHBzOi8vd3d3LmxpbmtlZGluLmNvbS9pbi9hb3NvbG9yemFuby8pLgoqICoqTGV2ZWwqKjogQWR2YW5jZWQuCiogKipUZWNobm9sb2dpZXMqKjogSmF2YSAxNywgU3ByaW5nIEJvb3QgMywgU3ByaW5nIENsb3VkIDQsIFNwcmluZyBOYXRpdmUsIFNwcmluZyBXZWJGbHV4LCBTcHJpbmcgT0F1dGgyLCBRdWFydHosIEZseXdheSwgTG9tYm9rLCBUZXN0Y29udGFpbmVycywgTG9jYWxTdGFjaywgQVdTIENvcGlsb3QgQ0xJLCBBV1MgU0FNLUNMSSwgQW1hem9uIENvZ25pdG8sIEFtYXpvbiBBdXJvcmEgKHdpdGggUG9zdGdyZVNRTCksIER5bmFtb0RCLCBBcHBsaWNhdGlvbiBMb2FkIEJhbGFuY2VyLCBFdmVudEJyaWRnZSwgTGFtYmRhLCBhbmQgRG9ja2VyLgoKIVtdKHV0aWxzL2ltYWdlcy9zb2x1dGlvbnNfYXJjaGl0ZWN0dXJlX3Y1LnBuZykKCllvdSBjYW4gcmVhZCB0aGUgZm9sbG93aW5nIGFydGljbGVzIGZyb20gbXkgKipNZWRpdW0uY29tKiogYWNjb3VudCB0byBnZXQgbW9yZSBwcm9qZWN0IGRldGFpbHM6CgoxLiBbTXVsdGktQWNjb3VudCBlbnZpcm9ubWVudCBvbiBBV1MgdXNpbmcgSUFNIElkZW50aXR5IENlbnRlcl0oaHR0cHM6Ly9hb3NvbG9yemFuby5tZWRpdW0uY29tL2ltcGxlbWVudGluZy1hLW11bHRpLWFjY291bnQtZW52aXJvbm1lbnQtd2l0aC1hd3Mtb3JnYW5pemF0aW9ucy1hbmQtdGhlLWlhbS1pZGVudGl0eS1jZW50ZXItZDFjZGI0MGJkZjRkKS4KMi4gW09BdXRoMiBpbiBTcHJpbmcgQm9vdCBOYXRpdmUgbWljcm9zZXJ2aWNlXShodHRwczovL2Fvc29sb3J6YW5vLm1lZGl1bS5jb20vb2F1dGgyLWluLXNwcmluZy1ib290LW5hdGl2ZS1yZWFjdGl2ZS1taWNyb3NlcnZpY2Utd2l0aC1hbWF6b24tY29nbml0by1hcy1vaWRjLXNlcnZpY2UtYzQ1NGQ4NGE1MjM0KS4KMy4gW0RlcGxveWluZyBTcHJpbmcgQm9vdCBOYXRpdmUgbWljcm9zZXJ2aWNlIHVzaW5nIENyb3NzLUFjY291bnQgZGVwbG95bWVudF0oaHR0cHM6Ly9hb3NvbG9yemFuby5tZWRpdW0uY29tL3NwcmluZy1ib290LW5hdGl2ZS1taWNyb3NlcnZpY2Utb24tZWNzLWZhcmdhdGUtdXNpbmctYXdzLWNvcGlsb3QtY2xpLWZvci1jcm9zcy1hY2NvdW50LWRlcGxveW1lbnQtNzNiMTgzNmYyMWY3KS4KNC4gW0VuZC10by1FbmQgRW5jcnlwdGlvbiB1c2luZyBUTFMgRUNEU0EgY2VydGlmaWNhdGUgYW5kIEFDTSB3aXRoIENvcGlsb3QgQ0xJXShodHRwczovL2Fvc29sb3J6YW5vLm1lZGl1bS5jb20vZW5kLXRvLWVuZC1lbmNyeXB0aW9uLXVzaW5nLXRscy1lY2RzYS1jZXJ0aWZpY2F0ZS1hY20tYW5kLWF3cy1jb3BpbG90LWNsaS02NGY1ZGFhZmU5NzcpLgo1LiBbRURBIHVzaW5nIEFtYXpvbiBFdmVudEJyaWRnZSwgTGFtYmRhLCBhbmQgU0FNLUNMSSwgd2l0aCBGYXJnYXRlIEVDUyBhcyBFdmVudCBTb3VyY2VdKGh0dHBzOi8vYW9zb2xvcnphbm8ubWVkaXVtLmNvbS9lZGEtd2l0aC1ldmVudGJyaWRnZS1hbmQtbGFtYmRhLXVzaW5nLXNhbS1jbGktd2l0aC1zcHJpbmctYm9vdC1tcy1vbi1mYXJnYXRlLWVjcy1hcy1ldmVudC1zb3VyY2UtOWFiZWUyMzdiZTA4KS4KNi4gW1N0b3JpbmcgRURBIGV2ZW50cyBpbiBEeW5hbW9EQiBmcm9tIGEgTmF0aXZlIExhbWJkYSBGdW5jdGlvbiB1c2luZyBTcHJpbmcgQ2xvdWQgRnVuY3Rpb25zXShodHRwczovL2Fvc29sb3J6YW5vLm1lZGl1bS5jb20vc3RvcmluZy1lZGEtZXZlbnRzLWluLWR5bmFtb2RiLXVzaW5nLWEtbmF0aXZlLWxhbWJkYS1mdW5jdGlvbi13aXRoLWdyYWFsdm0tc3ByaW5nLWNsb3VkLWZ1bmN0aW9ucy1jMzJjZWU0Nzc1ZmMpLgoKLS0tCgojIyBEZXNjcmlwdGlvbi4KVGhpcyBwcm9qZWN0IHVzZXMgU3ByaW5nIEJvb3QgdG8gbWFuYWdlIFF1YXJ0eiBKb2JzIHdpdGggdGhlIGhlbHAgb2YgU3ByaW5nIFdlYmZsdXggYW5kIFNwcmluZyBOYXRpdmUuClRoZSBRdWFydHogbGlicmFyeSBpcyBjb25maWd1cmVkIGZvciBhIGNsdXN0ZXJlZCBlbnZpcm9ubWVudCwgc28gaXQgbmVlZHMgUG9zdGdyZXMgdG8gc3RvcmUgYW5kIG1hbmFnZSBKb2JzIGV4ZWN1dGlvbnMuCldoZW4gYSBRdWFydHogSm9iIGlzIGV4ZWN1dGVkLCB0aGUgY2FsbGluZyBtZXRob2QgcmV0cmlldmVzIHRoZSBEZXZpY2UgaXRlbSBhc3NvY2lhdGVkIHdpdGggdGhlIFRhc2tzIGFuZCB1cGRhdGVzIGl0cyBzdGF0ZSBpbiBEeW5hbW9EQi4KVG8gcGVyZm9ybSBhbGwgdGhlc2UgYWN0aXZpdGllcywgdGhlIHVzZXJzIG11c3QgaGF2ZSBhIHZhbGlkIGFjY2VzcyB0b2tlbiAoSldUKSB0byBhY2Nlc3MgdGhlIGVuZHBvaW50cy4KQWxsIHRlc3QgY2FzZXMgdXNlIFRERCBmcm9tIHRoZSBiZWdpbm5pbmcgb2YgdGhlIGRldmVsb3BtZW50LCBhbmQgb25seSBJbnRlZ3JhdGlvbiBUZXN0cyBhcmUgZXhlY3V0ZWQgd2l0aCB0aGUgc3VwcG9ydCBvZiBUZXN0Y29udGFpbmVycyBhbmQgTG9jYWxTdGFjayB0byB0cnkgdG8gY292ZXIgcmVhbC13b3JsZCBzY2VuYXJpb3MuClRoaXMgcHJvamVjdCBhbHNvIHVzZXMgRG9ja2VyIENvbXBvc2UgdG8gZGVwbG95IGEgbG9jYWwgY2x1c3RlciB3aXRoIHRoZSByZXF1aXJlZCBzZXJ2aWNlcyBmb3IgbG9jYWwgdGVzdGluZy4KCi0tLQoKIyMgUHJlcmVxdWlzaXRlcy4KLSBHaXQuCi0gQVdTIENMSSAodmVyc2lvbiAyLjExLispLgotIEFXUyBDb3BpbG90IENMSSAodmVyc2lvbiAxLjI3LispLgotIEFXUyBTQU0gQ0xJICh2ZXJzaW9uIDEuOTAuKykuCi0gT3BlbkpESyAodmVyc2lvbiAxNy4wLispLiBZb3UgY2FuIHVzZSBTREtNQU4uCi0gQXBhY2hlIE1hdmVuICh2ZXJzaW9uIDMuOS4rKSBZb3UgY2FuIHVzZSBTREtNQU4uCi0gU3ByaW5nIEJvb3QgKHZlcnNpb24gMy4xLispCi0gRG9ja2VyIGFuZCBEb2NrZXIgQ29tcG9zZS4KCi0tLQoKIyMgUHJvamVjdCBTdHJ1Y3R1cmUuClRoZSBwcm9qZWN0IGlzIGRpdmlkZWQgaW50byB0aGUgZm9sbG93aW5nIGRpcmVjdG9yaWVzOgotICoqc3JjKio6IENvbnRhaW5zIHNvdXJjZSBjb2RlIG9mIHRoZSByZXF1aXJlZCBwcm9qZWN0cy4KLSAqKmNvcGlsb3QqKjogQ29udGFpbnMgdGhlIENvcGlsb3QgY29uZmlndXJhdGlvbiBmaWxlcy4KLSAqKnRlbXBsYXRlLnlhbWwqKjogQ29udGFpbnMgdGhlIFNBTSBjb25maWd1cmF0aW9uLgotICoqZG9ja2VyLWNvbXBvc2UueW1sKio6IENvbnRhaW5zIHRoZSBEb2NrZXIgQ29tcG9zZSBmaWxlIHRvIGRlcGxveSB0aGUgbG9jYWwgY2x1c3Rlci4KLSAqKnBvbS54bWwqKjogQ29udGFpbnMgdGhlIHByb2plY3QncyBkZXBlbmRlbmNpZXMuCi0gKip1dGlscyoqOiBDb250YWlucyB0aGUgc2NyaXB0cyB0byBkZXBsb3kgdGhlIGFwcGxpY2F0aW9uIGxvY2FsbHkgb3IgaW4gQVdTLgoKLS0tCgojIyBEZXBsb3ltZW50IE9wdGlvbnMuCllvdSBuZWVkIHRvIGV4ZWN1dGUgdGhlIGZvbGxvd2luZyBjb21tYW5kIGZyb20gdGhlIHByb2plY3QncyByb290IGRpcmVjdG9yeToKYGBgYmFzaAouL3J1bi1zY3JpcHRzLnNoCmBgYAoKVGhlIHNjcmlwdCB3aWxsIGFzayB5b3UgZm9yIHRoZSByZXF1aXJlZCBBV1MgcHJvZmlsZXMgdG8gZGVwbG95IHRoZSBhcHBsaWNhdGlvbiBsb2NhbGx5IG9yIGluIEFXUzoKIVtdKHV0aWxzL2ltYWdlcy9iYXNoX3NjcmlwdF9lbnRlcmluZ192YXJpYWJsZXMucG5nKQoKVGhlbiwgdGhlIHNjcmlwdCBzaG93cyBhIG1haW4gbWVudSB3aXRoIHRoZSBmb2xsb3dpbmcgb3B0aW9uczoKIVtdKHV0aWxzL2ltYWdlcy9iYXNoX3NjcmlwdF9tYWluX21lbnUucG5nKQoKSWYgeW91IGNob29zZSBvcHRpb24gJ2gnIGZvciAnSGVscGVyIE1lbnUsJwp5b3UgY2FuIGNyZWF0ZS9kZXBsb3kgZGVwZW5kZW50IHJlc291cmNlcyByZXF1aXJlZCBiZWZvcmUgZGVwbG95aW5nIHRoZSBhcHBsaWNhdGlvbiBvbiBBV1MuCgotLS0KCiMjIEludGVybmFsIERvY3VtZW50YXRpb24uClRoZSBpbnRlcm5hbCBwcm9qZWN0IGRvY3VtZW50YXRpb24gaXMgZGl2aWRlZCBpbnRvIHRoZSBmb2xsb3dpbmcgZGlyZWN0b3JpZXM6Ci0gKip1dGlscy9kb2NzKio6IENvbnRhaW5zIHRoZSBwcm9qZWN0J3MgZG9jdW1lbnRhdGlvbi4KLSAqKnV0aWxzL2RvY3MvaW1hZ2VzKio6IENvbnRhaW5zIHNvbHV0aW9uJ3MgYXJjaGl0ZWN0dXJlIGltYWdlcy4KClRoZSBtYXJrZG93biBkb2N1bWVudGF0aW9uIGZvciBpbmRpdmlkdWFsIHByb2plY3RzIGlzIGF2YWlsYWJsZSBhdDoKLSBbVGFza3MgQVBJIHdpdGggQ29waWxvdCBDTEldKHNyYy9jaXR5LXRhc2tzLWFwaS9SRUFETUUubWQpLgotIFtUYXNrcyBFdmVudCB3aXRoIFNBTSBDTEldKHNyYy9jaXR5LXRhc2tzLWV2ZW50cy1mdW5jdGlvbi9SRUFETUUubWQpLgo= readmeEtag: '"4ca8f19c899c799682d9488dc10224c79415d381"' readmeLastModified: Sun, 19 Nov 2023 19:50:17 GMT repositoryId: 667132451 description: >- Lambda Native Function using Java and GraalVM, which stores EDA events on Amazon DynamoDB. created: '2023-07-16T18:38:56Z' updated: '2024-12-14T04:24:52Z' language: Java archived: false stars: 1 watchers: 2 forks: 1 owner: aosolorzano logo: https://avatars.githubusercontent.com/u/5817871?v=4 license: Apache-2.0 repoEtag: '"7536ca435b2c13c9d93ebae7323a31bac7138d8a12d25764625855fa9d9508c9"' repoLastModified: Sat, 14 Dec 2024 04:24:52 GMT category: Server Implementations foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/goodluckxu-go/goapi v3: true id: 4143f283ad0108cc2ea084b428069d65 repositoryMetadata: base64Readme: >- IyBnb2FwaQrkvb/nlKhPcGVuQVBJMy4x5paH5qGj55qESFRUUOahhuaetgoK5L2/55So6K+05piOOgotIFvkuK3mlofmlofmoaNdKGRvY3MvemgvaW5kZXgubWQpCiMjIOeUqOazlQp+fn5iYXNoCmdvIGdldCBnaXRodWIuY29tL2dvb2RsdWNreHUtZ28vZ29hcGkvdjIKfn5+CiMjIOWKn+iDvQotIOWunueOsOS6hmh0dHDmnI3liqHvvIzot6/nlLHkvb/nlKhnaW7ot6/nlLHmqKHlvI/nmoTliY3nvIDmoJHmlrnlvI/lrp7njrAKLSDpm4bmiJBzd2FnZ2VyK29wZW5hcGkzLjEuMOaWh+aho+eahOiuv+mXrgotIOWunueOsOS6hm9wZW5hcGnkuK3pqozor4Hlkoxnb2Fwaeeoi+W6j+mqjOivgeeahOWQjOatpQotIOWunueOsOS6huiHquWumuS5ieS4remXtOS7tgotIOWunueOsOS6humJtOadg+iupOivgQotIOWunueOsOS6hui3r+eUsee7hOaooeW8jwotIOWunueOsOS6huWkmuS4queoi+W6j+aooeWdl+e7hOeahOaooeW8jwotIC4uLi4uLgojIyDlhbPkuo4K5L2/55So57G75Ly85LqOUHl0aG9u5Lit55qERmFzdEFQSeeahEFQSeeUn+aIkOaWh+ahow== readmeEtag: '"1c635e245e3725d27ab124ade9d659138a2777fb"' readmeLastModified: Mon, 19 Jan 2026 09:15:20 GMT repositoryId: 819191357 description: Go的API服务器可以生成开放的API文档 Go's API server can generate open API documents created: '2024-06-24T02:43:37Z' updated: '2026-02-06T03:40:54Z' language: Go archived: false stars: 2 watchers: 1 forks: 0 owner: goodluckxu-go logo: https://avatars.githubusercontent.com/u/110141236?v=4 repoEtag: '"dd9f4e48e61476ae894fbd62aa990514fd7e94fbf99640f9bc374233eb464ab7"' repoLastModified: Fri, 06 Feb 2026 03:40:54 GMT category: Parsers foundInMaster: true v3_1: true - source: openapi3 tags repository: https://github.com/listendev/pkg v3: true id: 809b021239d9fe78081a04950e153301 repositoryMetadata: base64Readme: >- IyBwa2cKCkEgY29sbGVjdGlvbiBvZiBjb21tb24gcGFja2FnZXMuCgojIyBQYWNrYWdlcwoKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL2FuYWx5c2lzcmVxdWVzdF0oL2FuYWx5c2lzcmVxdWVzdCkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL2FwaXNwZWNdKC9hcGlzcGVjKQotIFtnaXRodWIuY29tL2xpc3RlbmRldi9wa2cvZGV0ZWN0aW9uL3R5cGVdKC9kZXRlY3Rpb24vdHlwZSkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL2Vjb3N5c3RlbV0oL2Vjb3N5c3RlbSkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL2luZm9ybWF0aW9uYWwvdHlwZV0oL2luZm9ybWF0aW9uYWwvdHlwZSkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL2xvY2tmaWxlXSgvbG9ja2ZpbGUpCi0gW2dpdGh1Yi5jb20vbGlzdGVuZGV2L3BrZy9tYW5pZmVzdF0oL21hbmlmZXN0KQotIFtnaXRodWIuY29tL2xpc3RlbmRldi9wa2cvbWFwL3V0aWxdKC9tYXAvdXRpbCkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL21vZGVsc10oL21vZGVscykKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL25wbV0oL25wbSkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL29ic2VydmFiaWxpdHldKC9vYnNlcnZhYmlsaXR5KQotIFtnaXRodWIuY29tL2xpc3RlbmRldi9wa2cvcHlwaV0oL3B5cGkpCi0gW2dpdGh1Yi5jb20vbGlzdGVuZGV2L3BrZy9yYW5kXSgvcmFuZCkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL3N0cmluZy91dGlsXSgvc3RyaW5nL3V0aWwpCi0gW2dpdGh1Yi5jb20vbGlzdGVuZGV2L3BrZy90eXBlXSgvdHlwZSkKLSBbZ2l0aHViLmNvbS9saXN0ZW5kZXYvcGtnL3ZhbGlkYXRlXSgvdmFsaWRhdGUpCi0gW2dpdGh1Yi5jb20vbGlzdGVuZGV2L3BrZy92ZXJkaWN0Y29kZV0oL3ZlcmRpY3Rjb2RlKQoKIyMgR2VuZXJhdGlvbgoKYGBgCmdvIGluc3RhbGwgZ29sYW5nLm9yZy94L3Rvb2xzL2NtZC9zdHJpbmdlckBsYXRlc3QKZ28gaW5zdGFsbCBnaXRodWIuY29tL2RlZXBtYXAvb2FwaS1jb2RlZ2VuL2NtZC9vYXBpLWNvZGVnZW5AbWFzdGVyICMgVXNlIG1hc3RlciBicmFuY2gKZ28gZ2VuZXJhdGUgLXggLi92ZXJkaWN0Y29kZQpnbyBnZW5lcmF0ZSAteCAuL2Vjb3N5c3RlbQpnbyBnZW5lcmF0ZSAteCAuL21vZGVscy9jYXRlZ29yeQpnbyBnZW5lcmF0ZSAteCAuL21vZGVscy9zZXZlcml0eQpnbyBnZW5lcmF0ZSAteCAuL21vZGVscwpnbyBnZW5lcmF0ZSAteCAuL2xvY2tmaWxlCmdvIGdlbmVyYXRlIC14IC4vbWFuaWZlc3QKZ28gZ2VuZXJhdGUgLXggLi9hcGlzcGVjCmdvIGdlbmVyYXRlIC14IC4vZGV0ZWN0aW9uL3R5cGUKZ28gZ2VuZXJhdGUgLXggLi9pbmZvcm1hdGlvbmFsL3R5cGUKYGBgCg== readmeEtag: '"d243c5a4c82f56ce88bf595e42579f3ebf1a817d"' readmeLastModified: Wed, 04 Sep 2024 16:05:57 GMT repositoryId: 665077779 description: >- Models for the analysis requests to listen.dev and the consequent verdicts responses created: '2023-07-11T11:39:58Z' updated: '2025-02-06T10:12:42Z' language: Go archived: false stars: 1 watchers: 4 forks: 0 owner: listendev logo: https://avatars.githubusercontent.com/u/103680976?v=4 license: Apache-2.0 repoEtag: '"43e6f7a3940b90a4947d02ba292cea8932598c49c09c276805eb5490a6dd1793"' repoLastModified: Thu, 06 Feb 2025 10:12:42 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/unjello/openapi-mockk v3: true repositoryMetadata: base64Readme: >- aW1hZ2U6Omh0dHA6Ly91bm1haW50YWluZWQudGVjaC9iYWRnZS5zdmdbbGluaz1odHRwOi8vdW5tYWludGFpbmVkLnRlY2gvXQ0KDQo9IERFUFJFQ0FURUQNCg0KX29wZW5hcGktcGFyc2VyXyBpcyBubyBsb25nZXIgc3VwcG9ydGVkLCBwbGVhc2UgY29uc2lkZXIgdXNpbmcgaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItcGFyc2VyW3N3YWdnZXItcGFyc2VyXS4NCg== readmeEtag: '"5dc94a8d0e902f95a0e80e0d23c492f159b9f7f8"' readmeLastModified: Wed, 01 May 2019 08:48:51 GMT repositoryId: 128423249 description: DEPRECATED. Generates data based on OpenAPI 3.0 specification created: '2018-04-06T17:10:22Z' updated: '2019-05-01T08:50:29Z' language: JavaScript archived: false stars: 1 watchers: 2 forks: 2 owner: unjello logo: https://avatars.githubusercontent.com/u/155669?v=4 license: CC0-1.0 repoEtag: '"846ef8dbe0d9696dafde759155e27e8edb595922ebc7eb306f6a1b747e47b101"' repoLastModified: Wed, 01 May 2019 08:50:29 GMT foundInMaster: true category: Code Generators id: ac51d5abb01feee912cecaa014d2ef7c - source: openapi3 tags repository: https://github.com/aasaam/openapi-preview v3: true repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICA8aDE+CiAgICBPcGVuQVBJIFByZXZpZXcKICA8L2gxPgogIDxwPgogICAgUHJldmlldyBPcGVuQVBJIFNwZWMgdXNpbmcgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkiPlN3YWdnZXIgVUk8L2E+CiAgPC9wPgogIDxwPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2Fhc2FhbS9vcGVuYXBpLXByZXZpZXciPgogICAgICA8aW1nIGFsdD0iR2l0SHViIHJlcG8gc2l6ZSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9yZXBvLXNpemUvYWFzYWFtL29wZW5hcGktcHJldmlldyI+CiAgICA8L2E+CiAgICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vYWFzYWFtL29wZW5hcGktcHJldmlldy9ibG9iL21hc3Rlci9MSUNFTlNFIj4KICAgICAgPGltZyBhbHQ9IkxpY2Vuc2UiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9hYXNhYW0vb3BlbmFwaS1wcmV2aWV3Ij4KICAgIDwvYT4KICA8L3A+CjwvZGl2PgoKPGRpdj4KICA8cCBhbGlnbj0iY2VudGVyIj4KICAgIDxpbWcgYWx0PSJhYXNhYW0gc29mdHdhcmUgZGV2ZWxvcG1lbnQgZ3JvdXAiIHdpZHRoPSI2NCIgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYWFzYWFtL2luZm9ybWF0aW9uL21hc3Rlci9sb2dvL2Fhc2FhbS5zdmciPgogICAgPGJyIC8+CiAgICBhYXNhYW0gc29mdHdhcmUgZGV2ZWxvcG1lbnQgZ3JvdXAKICA8L3A+CjwvZGl2Pgo= readmeEtag: '"3785309248dff3c86536a2d3faba12ba0039c086"' readmeLastModified: Wed, 03 Mar 2021 23:47:43 GMT repositoryId: 214886616 description: Preview OpenAPI Spec using Swagger UI created: '2019-10-13T20:10:24Z' updated: '2021-03-03T23:48:04Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: aasaam logo: https://avatars.githubusercontent.com/u/10144065?v=4 license: MIT repoEtag: '"3fea4cd10d5db06f1908c9c8f4e723a9d92de5584f8d47b358d376a8eb9b6827"' repoLastModified: Wed, 03 Mar 2021 23:48:04 GMT foundInMaster: true category: - Documentation - Testing id: eeb5e994f260453efafd8bd3ae56f690 - source: openapi3 tags repository: https://github.com/elibracha/openapi-diff-ignore v3: true repositoryMetadata: base64Readme: >- OpenAPI-diff-ignore
============
[![Build Status](https://travis-ci.com/elibracha/openapi-diff-ignore.svg?branch=master)](https://travis-ci.com/elibracha/openapi-diff-ignore)
[![codecov](https://codecov.io/gh/elibracha/openapi-diff-ignore/branch/master/graph/badge.svg)](https://codecov.io/gh/elibracha/openapi-diff-ignore)
[![Jion the Slack chat room](https://img.shields.io/badge/Slack-Join%20the%20chat%20room-orange)](https://join.slack.com/t/openapi-diff-ignore/shared_invite/enQtOTM4MzYxNDAxMTEwLTk0ZDlmYzcwYWVlNWRiNDA4YmI1MDIwYjZlMTk4Mjc5OTI3NjY0Zjg0ZTEyNjA0NzBkOWY3MmNhNjQ2NjgxYjY)

Parse a OpenAPI-diff ignore file specifications(1.x). 
Can be use to integrate with OpenAPI-diff and customize your breaking changes.

If you think you have found a bug, please file an issue in the [Github Issues](https://github.com/elibracha/openapi-diff-ignore/issues).

Ignore Specification
------------------------

### Schema
In the following description, if a field is not explicitly REQUIRED or described with a MUST or SHALL, it can be considered OPTIONAL.

##### Fixed Fields

Field Name | Type | Description
---|:---:|---
<a name="extends"></a>extends | `string` | Can be used to inherit a pre defined ignore file functionality. Provides metadata about basic ignore file for your custom API ignore. (Like -> ".default")
<a name="version"></a>version | `string` | **REQUIRED**. This string MUST be the [semantic version number](https://semver.org/spec/v2.0.0.html) of the [Ignore Specification version](#versions) that the OpenAPI ignore document uses. The `version` field SHOULD be used by tooling specifications and clients to interpret the OpenAPI ignore document.
<a name="info"></a>info | `string` | Provides metadata about general info for the API ignore. The metadata MAY be used by tooling as required.
<a name="project"></a>project | `string` | Provides metadata about the Project using the ignore file. The metadata MAY be used by tooling as required.
<a name="paths"></a>paths | [Paths Object](#pathsObject) |  The available paths and operations for the API that needed to be ignored.

#### <a name="pathsObject"></a>Paths Object

Holds the relative paths to the individual endpoints and their operations.

##### Patterned Fields

Field Pattern | Type | Description
---|:---:|---
<a name="pathsPath"></a>/{path} | [Path Item Object](#pathItemObject) | A relative path to an individual endpoint. The field name MUST begin with a slash. Wildcard is allowed. When matching URLs. In case of ambiguous matching, it's up to the tooling to decide the execution order.

##### Path Wildcard Matching

Assuming the following paths, the concrete definition, `/pets/mine`, will be matched if used:

```
  /pets/**        # will match and also match all pets endpoints
  /**             # will match and also will match all endpoints
```
NOTICE! When using wildcards all endpoints matching the wildcard will apply the ignore specified.

##### Paths Object Example

```yaml
paths:
  /pet/**, /store/**:        # this path will match any 'pet' or 'store' endpoint 
    get:                      
      response:
        200:
          new: true
      parameters:
        - username
        - password

    post:
      request:
        content:
          application/json, application/octet-stream: $
      response:
        200:
          new: true
          content:
            application/json:
              schema:
                properties:
                  - petId
                  - quantity
                  - shipDate
                  - complete
      security:
        petstore_auth:
          - 'write:pets'
          - 'read:pets'

  /user/login:               # this path will match only '/user/login' endpoint. 
    post: $        
```

#### <a name="pathItemObject"></a>Path Item Object

Describes the operations available on a single path.
A Path Item MAY be empty, due to [ACL constraints](#securityFiltering).
The path itself is still exposed to the documentation viewer but they will not know which operations and parameters are available.

##### Fixed Fields

Field Name | Type | Description
---|:---:|---
<a name="pathItemGet"></a>get | [Operation Object](#operationObject) | A definition of a GET operation on this path.
<a name="pathItemPut"></a>put | [Operation Object](#operationObject) | A definition of a PUT operation on this path.
<a name="pathItemPost"></a>post | [Operation Object](#operationObject) | A definition of a POST operation on this path.
<a name="pathItemDelete"></a>delete | [Operation Object](#operationObject) | A definition of a DELETE operation on this path.
<a name="pathItemOptions"></a>options | [Operation Object](#operationObject) | A definition of a OPTIONS operation on this path.
<a name="pathItemHead"></a>head | [Operation Object](#operationObject) | A definition of a HEAD operation on this path.
<a name="pathItemPatch"></a>patch | [Operation Object](#operationObject) | A definition of a PATCH operation on this path.
<a name="pathItemTrace"></a>trace | [Operation Object](#operationObject) | A definition of a TRACE operation on this path.

##### Path Item Object Example


```yaml
 post:
      request:
        content:
          application/json, application/octet-stream: $
      response:
        200:
          new: true
          content:
            application/json:
              schema:
                properties:
                  - petId
                  - quantity
                  - shipDate
                  - complete
      security:
        petstore_auth:
          - 'write:pets'
          - 'read:pets'
put: $
delete: $
```

#### <a name="operationObject"></a>Operation Object

Describes a single API ignore operation on a path.

##### Fixed Fields

Field Name | Type | Description
---|:---:|---
<a name="security"></a>security | [Security Object](#parameterObject) | A list of security requirements that are going to be ignored.
<a name="parameters"></a>parameters | [Parameter Object](#parameterObject)| A list of parameters that are going to be ignored.
<a name="request"></a>request | [Request Object](#requestBodyObject) | The request body applicable for this operation. can specify ignore for content-type.
<a name="response"></a>response | [Response Object](#responsesObject) | The list of possible response statuses that are going to be ignored.

##### Operation Object Example

```yaml
request:
  content:
    application/json, application/octet-stream: $

parameters:
  - username
  - password

response:
  200:
    new: true
    content:
      application/json:
        schema:
          properties:
            - petId
            - quantity
            - shipDate
            - complete

security:
  petstore_auth:
    - 'write:pets'
    - 'read:pets'
```
#### <a name="securityObject"></a>Security Object

Describes a security requirements that needed to be ignored.

##### Fixed Fields

Field Name | Type | Description
---|:---:|---
<a name="schmea"></a>{schema} | Map<String, List> | A map of security requirements that are going to be ignored.
##### Security Object Example

```yaml
security:
  petstore_auth:
    - 'write:pets'
    - 'read:pets'
  api_key:
    - 'read:users'
```

Documentation
-------------

More information can be found on [Docs][openapi-diff-ignore-home].


Where can I get the latest release?
-----------------------------------
You can download release source from our release page.

Contributing
------------

If you are interested in the development of OpenAPI-diff-ignore, please consult the 
documentation first and afterwards you are welcome to join the developers 
mailing list to ask question or discuss new ideas / features / bugs etc.

Take a look into the [contribution guidelines](CONTRIBUTING.md).



Quick Build
-------
If you want to bootstrap openapi-diff-ignore, you'll need:
- Java 1.8+
- Maven 3.x.x or later
- Run Maven, specifying a location into which the completed Maven distro should be installed:
```
./mvnw clean package
```

[openapi-diff-ignore-home]: https://maven.apache.org/
 readmeEtag: '"35cb21cd72cd6694e0e922d261b89a3d08d9bb08"' readmeLastModified: Tue, 05 May 2020 14:09:44 GMT repositoryId: 238002799 description: >- This repository describes an ignore specification, and can be used to integrate with OpenAPI-diff and customize your breaking changes. created: '2020-02-03T15:53:55Z' updated: '2020-05-05T14:09:53Z' language: Java archived: false stars: 1 watchers: 3 forks: 1 owner: elibracha logo: https://avatars.githubusercontent.com/u/31411282?v=4 license: MIT repoEtag: '"336d756f25328a384ac2e52fff15916adbaeb25bb5acebe56131cbaa5b574d95"' repoLastModified: Tue, 05 May 2020 14:09:53 GMT foundInMaster: true category: Parsers id: 2e25629dba8381246f6bb9aa2a9c4951 - source: openapi3 tags repository: https://github.com/karthikeyan-ng/e-learnings-certificate_udemy v3: true repositoryMetadata: repositoryId: 243043222 description: Course Completion Certificates + Technology Certificates created: '2020-02-25T16:07:38Z' updated: '2020-07-05T13:26:31Z' language: null archived: false stars: 1 watchers: 1 forks: 0 owner: karthikeyan-ng logo: https://avatars.githubusercontent.com/u/39563373?v=4 repoEtag: '"0b20cc21f8527608cef757a421c6312b9fc6988988280ad420cfd83e0fd21388"' repoLastModified: Sun, 05 Jul 2020 13:26:31 GMT foundInMaster: true id: 2945795c4fef24e25a12f0ba8a8e93e8 - source: openapi3 tags repository: https://github.com/abhinaba-ghosh/postman-coverage v3: true repositoryMetadata: base64Readme: >- IyBQb3N0bWFuLUNvdmVyYWdlCgpHZW5lcmF0ZSBQb3N0bWFuIGNvbGxlY3Rpb24gY292ZXJhZ2UgcmVwb3J0LgoKV29yayBpbiBwcm9ncmVzcyEgS2VlcCB3YXRjaGluZy4K readmeEtag: '"abfc1968a123d3c44e3c4e78bd856e348d1bfad3"' readmeLastModified: Sun, 27 Jun 2021 21:21:43 GMT repositoryId: 379584054 description: derive postman collection coverage against openapi schema created: '2021-06-23T11:45:36Z' updated: '2022-09-05T10:42:44Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: abhinaba-ghosh logo: https://avatars.githubusercontent.com/u/23141842?v=4 license: MIT repoEtag: '"f058de1377d7a986fadf19b76da21cfcd4f85e25a4e3c7fe737d3999bea43cfa"' repoLastModified: Mon, 05 Sep 2022 10:42:44 GMT foundInMaster: true category: - Converters - Testing id: 99d139aa4e0795bc5213ad50c672253a - source: openapi3 tags repository: https://github.com/kba07/ecwid-shiprocket-app v3: true id: 2500014a818e22f012ce48fe3e880483 repositoryMetadata: base64Readme: >- IyBJbnRlZ3JhdGlvbiBiZXR3ZWVuIGVjd2lkIGFuZCBzaGlwcm9ja2V0LgpBIHNpbXBsZSBpbnRlZ3JhdGlvbiBiZXR3ZWVuIGVjd2lkIGFuZCBzaGlwcm9ja2V0LCB3aGljaCB3aWxsIGdlbmVyYXRlIHNoaXByb2NrZXQgc3BlY2lmaWMgYnVsayBvcmRlcnMgY3N2IGFuZCBzZW5kIGl0IG92ZXIgYW4gZW1haWwuCg== readmeEtag: '"81e94b0e64c0c73c552242ecaa9471583f977d31"' readmeLastModified: Tue, 30 May 2023 07:21:02 GMT repositoryId: 354893401 description: >- A simple integration between ecwid and shiprocket, which will generate shiprocket specific bulk orders csv and send it over an email. created: '2021-04-05T16:10:44Z' updated: '2023-05-30T07:17:45Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: KBA07 logo: https://avatars.githubusercontent.com/u/39996841?v=4 repoEtag: '"08b2c6ba0fd0180ca366c0d105939414b8e495b0ef0abda75d91db416d3117b8"' repoLastModified: Tue, 30 May 2023 07:17:45 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/rrjanbiah/legacy2openapi-spec v3: true repositoryMetadata: base64Readme: >- IyBsZWdhY3kyb3BlbmFwaS1zcGVjIC0gTGVnYWN5IHRvIE9wZW5BUEkgc3BlYyBjb252ZXJzaW9ucyBvZiBBUElzIGZvciBlYXN5IFNESyBjb2RlIGdlbmVyYXRpb25zCgpPcGVuQVBJIHNwZWMtY29tcGxpYW50IEFQSXMgaGF2ZSB0aGUgYmVsb3cgYWR2YW50YWdlczoKMS4gQ2FuIGdlbmVyYXRlIFNESyBjb2RlcyBhdXRvbWF0aWNhbGx5IHVzaW5nIHRoZSBbT3BlbkFQSSBHZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IpCjIuIENhbiBnZW5lcmF0ZSBBUEkgcGxheWdyb3VuZCAmIGRvY3VtZW50YXRpb24gdXNpbmcgW29wZW5hcGktdWldKGh0dHBzOi8vZ2l0aHViLmNvbS9kaGNvZGUvb3BlbmFwaS11aSkgYW5kIFtTd2FnZ2VyIFVJXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkKMy4gQ2FuIGdlbmVyYXRlIG5pY2UtbG9va2luZyBkb2N1bWVudGF0aW9uIHVzaW5nIFtSZWRvY10oaHR0cHM6Ly9naXRodWIuY29tL1JlZG9jbHkvcmVkb2MpCgpCdXQsIGZldyBTYWFTIGNvbXBhbmllcyBkb24ndCBvZmZlciBPcGVuQVBJIHNwZWMgZm9yIHRoZWlyIG93biByZWFzb25zLiBUaGlzIHByb2plY3QgaXMgYW4gYXR0ZW1wdCB0byBpZGVudGlmeSB0aG9zZSBsZWdhY3kgQVBJIGRvY3VtZW50YXRpb24gYW5kIGNvbnZlcnQgdGhlbSBpbnRvIE9wZW5BUEkgc3BlYy4K readmeEtag: '"c278f991609423c723641b2ebce18ee07d09c3a2"' readmeLastModified: Sat, 23 Apr 2022 15:25:45 GMT repositoryId: 484774635 description: Legacy to OpenAPI spec conversions of APIs for easy SDK code generations created: '2022-04-23T14:47:43Z' updated: '2025-09-25T14:39:17Z' language: Rust archived: false stars: 2 watchers: 1 forks: 0 owner: rrjanbiah logo: https://avatars.githubusercontent.com/u/4907427?v=4 license: MIT repoEtag: '"7fb1a459bf943d6b886ea0d59c990d227370b4123915abbe7c9cff076e81dfae"' repoLastModified: Thu, 25 Sep 2025 14:39:17 GMT foundInMaster: true category: - Code Generators - Parsers id: 8b6ae7f657ec6327ab502f153fcbea3a - source: openapi3 tags repository: https://github.com/george-the-penguin/render-spring-boot-rest-api-poc v3: true id: 0693d7a953b867eea8a6cccd6b6f02e2 repositoryMetadata: base64Readme: >- IyByZW5kZXItc3ByaW5nLWJvb3QtcmVzdC1hcGktcG9jCgpUaGlzIGlzIGEgc2ltcGxlIFNwcmluZyBCb290IFJFU1QgQVBJIFByb29mIG9mIENvbmNlcHQgKFBPQykgcHJvamVjdC4KClRoZSBnb2FsIG9mIHRoaXMgcHJvamVjdCBpcyB0byBkZW1vbnN0cmF0ZSBob3cgdG8gY3JlYXRlIGEgUkVTVCBBUEkgdXNpbmcgU3ByaW5nIEJvb3QsIGFuZCBpdCBjYW4gcnVuIHByb3Blcmx5IGluIHRoZSAKUmVuZGVyIGVjb3N5c3RlbS4KClRoZSBpZGVhIHRvIGNyZWF0ZSB0aGlzIHByb2plY3QgYXJpc2VzIGluIHJlc3BvbnNlIHRvIHNldmVyYWwgcXVlc3Rpb25zIGFza2VkIGJ5IHBhcnRpY2lwYW50cyBvZiB0aGUgKipEZXZzIExhdGFtKiogCmluaXRpYXRpdmUgZnJvbSB0aGUgKipUcnlDYXRjaCoqIGRldmVsb3BtZW50IGNvbW11bml0eS4KCiMjIFRlY2hub2xvZ2llcwoKVGhlIGZvbGxvd2luZyBpcyB0aGUgbGlzdCBvZiB0ZWNobm9sb2dpZXMgdXNlZCB0byBidWlsZCB0aGlzIHByb2plY3Q6CgotIFtKYXZhIDE3XShodHRwczovL29wZW5qZGsub3JnL3Byb2plY3RzL2pkay8xNy8pCi0gW1NwcmluZyBCb290IDMuMC42XShodHRwczovL3NwcmluZy5pby8pCi0gW0FwYWNoZSBNYXZlbiAzLjkuMV0oaHR0cHM6Ly9tYXZlbi5hcGFjaGUub3JnLykKLSBbTG9tYm9rIDEuMTguMjZdKGh0dHBzOi8vcHJvamVjdGxvbWJvay5vcmcvKQotIFtPcGVuQVBJIDMuMC4zXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKQotIFtQb3N0Z3JlU1FMIDE1XShodHRwczovL3d3dy5wb3N0Z3Jlc3FsLm9yZy8pCi0gW0RvY2tlciAyMC4xMC4yNF0oaHR0cHM6Ly93d3cuZG9ja2VyLmNvbS8pCi0gW1JlbmRlcl0oaHR0cHM6Ly9yZW5kZXIuY29tLykKCiMjIENyZWF0ZSBhIFJlbmRlcidzIFBvc3RncmVTUUwgZGF0YWJhc2UKClRvIGNyZWF0ZSBhIFBvc3RncmVTUUwgZGF0YWJhc2UgaW4gUmVuZGVyLCB5b3UgbmVlZCB0byBmb2xsb3cgdGhlIGZvbGxvd2luZyBzdGVwczoKCjEuIEdvIHRvIHRoZSBbUmVuZGVyJ3MgZGFzaGJvYXJkXShodHRwczovL2Rhc2hib2FyZC5yZW5kZXIuY29tLykuCjIuIENsaWNrIG9uIHRoZSAqKk5ldysqKiBidXR0b24sIGFuZCB0aGVuIGNsaWNrIG9uIHRoZSAqKlBvc3RncmVTUUwqKiBvcHRpb24uCjMuIEluIHRoZSAqKk5ldyBQb3N0Z3JlU1FMKiogZm9ybSwgeW91IG5lZWQgdG8gZmlsbCB0aGUgZm9sbG93aW5nIGZpZWxkczoKICAgLSAqKk5hbWUqKjogVGhlIG5hbWUgb2YgdGhlIGRhdGFiYXNlIHJlc291cmNlIGluIFJlbmRlci4KICAgLSAqKkRhdGFiYXNlKio6IFRoZSBuYW1lIG9mIHRoZSBkYXRhYmFzZSBpbnN0YW5jZS4KICAgLSAqKlVzZXIqKjogVGhlIHVzZXJuYW1lIG9mIHRoZSBkYXRhYmFzZS4KICAgLSAqKlJlZ2lvbioqOiBUaGUgcmVnaW9uIHdoZXJlIHRoZSBkYXRhYmFzZSB3aWxsIGJlIGNyZWF0ZWQuCiAgIC0gKipQb3N0Z3JlU1FMIFZlcnNpb24qKjogVGhlIFBvc3RncmVTUUwgdmVyc2lvbi4gSSByZWNvbW1lbmQgdXNpbmcgdGhlIGxhdGVzdCB2ZXJzaW9uLgogICAtICoqSW5zdGFuY2UgVHlwZSoqOiBUaGUgaW5zdGFuY2UgdHlwZS4gSSByZWNvbW1lbmQgdXNpbmcgdGhlICoqRnJlZSoqIGluc3RhbmNlIHR5cGUgZm9yIGFjYWRlbWljIHB1cnBvc2VzLgo0LiBDbGljayBvbiB0aGUgKipDcmVhdGUgRGF0YWJhc2UqKiBidXR0b24uCgojIyBIb3cgdG8gYnVpbGQKClRvIGJ1aWxkIHRoaXMgcHJvamVjdCwgeW91IG5lZWQgdG8gaGF2ZSB0aGUgZm9sbG93aW5nIGluc3RhbGxlZCBvbiB5b3VyIG1hY2hpbmU6CgotIEphdmEgMTcKLSBBcGFjaGUgTWF2ZW4gMwotIERvY2tlciAyMAoKVG8gYnVpbGQgdGhlIHByb2plY3QsIHlvdSBuZWVkIHRvIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBiYXNoCm12biBjbGVhbiBwYWNrYWdlCmBgYAoKX05PVEU6IFRoaXMgY29tbWFuZCBtdXN0IGJlIGV4ZWN1dGVkIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGUgcHJvamVjdC5fCgpUbyBidWlsZCB0aGUgRG9ja2VyIGltYWdlLCB5b3UgbmVlZCB0byBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgYmFzaApkb2NrZXIgYnVpbGQgLXQgcmVuZGVyLXNwcmluZy1ib290LXJlc3QtYXBpLXBvYyAuCmBgYAoKX05PVEU6IFRoaXMgY29tbWFuZCBtdXN0IGJlIGV4ZWN1dGVkIGluIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGUgcHJvamVjdC5fCgojIyBIb3cgdG8gcnVuCgpUbyBydW4gdGhlIERvY2tlciBjb250YWluZXIsIHlvdSBuZWVkIHRvIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBiYXNoCmRvY2tlciBydW4gLWUgREJfVVJMPSJEQl9VUkwiIC1lIERCX1VTRVI9REJfVVNFUiAtZSBEQl9QQVNTV0Q9REJfUEFTU1dEIC1wIDgwODA6ODA4MCByZW5kZXItc3ByaW5nLWJvb3QtcmVzdC1hcGktcG9jCmBgYAoKV2hlcmU6Ci0gYERCX1VSTGAgaXMgdGhlIFVSTCBvZiB0aGUgUG9zdGdyZVNRTCBkYXRhYmFzZS4KLSBgREJfVVNFUmAgaXMgdGhlIHVzZXIgb2YgdGhlIFBvc3RncmVTUUwgZGF0YWJhc2UuCi0gYERCX1BBU1NXRGAgaXMgdGhlIHBhc3N3b3JkIG9mIHRoZSBQb3N0Z3JlU1FMIGRhdGFiYXNlLgoKVGhlc2UgZGF0YSBtdXN0IGJlIHByb3ZpZGVkIGFzIGVudmlyb25tZW50IHZhcmlhYmxlcyB0byB0aGUgRG9ja2VyIGNvbnRhaW5lciwgYW5kIHRoZXkgY2FuIGJlIG9idGFpbmVkIGZyb20gdGhlIApSZW5kZXIncyBkYXRhYmFzZSByZXNvdXJjZSwgd2l0aGluIHRoZSAqKkNvbm5lY3Rpb25zKiogc2VjdGlvbi4KCkV4YW1wbGU6CgpgYGBiYXNoCmRvY2tlciBydW4gLWUgREJfVVJMPSJqZGJjOnBvc3RncmVzcWw6Ly9wb3N0Z3Jlcy5yZW5kZXIuY29tOjU0MzIvcmVuZGVyLXNwcmluZy1ib290LXJlc3QtYXBpLXBvYyIgLWUgREJfVVNFUj1wb3N0Z3JlcyAtZSBEQl9QQVNTV0Q9cG9zdGdyZXMgLXAgODA4MDo4MDgwIHJlbmRlci1zcHJpbmctYm9vdC1yZXN0LWFwaS1wb2MKYGBgCgojIyBIb3cgdG8gZGVwbG95IG9uIFJlbmRlcgoKVG8gZGVwbG95IHRoaXMgcHJvamVjdCBvbiBSZW5kZXIsIHlvdSBuZWVkIHRvIGZvbGxvdyB0aGUgZm9sbG93aW5nIHN0ZXBzOgoKMS4gR28gdG8gdGhlIFtSZW5kZXIncyBkYXNoYm9hcmRdKGh0dHBzOi8vZGFzaGJvYXJkLnJlbmRlci5jb20vKS4KMi4gQ2xpY2sgb24gdGhlICoqTmV3KyoqIGJ1dHRvbiwgYW5kIHRoZW4gY2xpY2sgb24gdGhlICoqV2ViIFNlcnZpY2UqKiBvcHRpb24uCjMuIEluIHRoZSAqKkNyZWF0ZSBhIG5ldyBXZWIgU2VydmljZSoqIGZvcm0sIHlvdSBuZWVkIHRvIGZpbGwgdGhlIGZvbGxvd2luZyBmaWVsZHM6CiAgIC0gKipDb25uZWN0IGEgcmVwb3NpdG9yeSoqOiBTZWxlY3QgdGhlIHJlcG9zaXRvcnkgd2hlcmUgdGhlIHByb2plY3QgaXMgbG9jYXRlZC4gQ2xpY2sgb24gdGhlICoqQ29ubmVjdCoqIGJ1dHRvbiB0byBjb250aW51ZS4KNC4gSW4gdGhlIG5leHQgZm9ybSwgeW91IG5lZWQgdG8gZmlsbCB0aGUgZm9sbG93aW5nIGZpZWxkczoKICAgLSAqKk5hbWUqKjogVGhlIG5hbWUgb2YgdGhlIHdlYiBzZXJ2aWNlLgogICAtICoqUmVnaW9uKio6IFRoZSByZWdpb24gd2hlcmUgdGhlIHdlYiBzZXJ2aWNlIHdpbGwgYmUgY3JlYXRlZC4KICAgLSAqKkJyYW5jaCoqOiBUaGUgYnJhbmNoIG9mIHRoZSByZXBvc2l0b3J5IHRvIGRlcGxveS4KICAgLSAqKlJ1bnRpbWUqKjogVGhlIHJ1bnRpbWUgb2YgdGhlIHdlYiBzZXJ2aWNlLiBGb3IgdGhpcyBjYXNlLCB5b3UgbmVlZCB0byBzZWxlY3QgdGhlICoqRG9ja2VyKiogb3B0aW9uLgogICAtICoqSW5zdGFuY2UgVHlwZSoqOiBUaGUgaW5zdGFuY2UgdHlwZS4gSSByZWNvbW1lbmQgdXNpbmcgdGhlICoqRnJlZSoqIGluc3RhbmNlIHR5cGUgZm9yIGFjYWRlbWljIHB1cnBvc2VzLgogICAtIEluIHRoZSAqKkFkdmFuY2VkKiogc2VjdGlvbiwgY2xpY2sgdGhlICoqQWRkIEVudmlyb25tZW50IFZhcmlhYmxlKiogYnV0dG9uLCBhbmQgdGhlbiBhZGQgdGhlIGZvbGxvd2luZyBlbnZpcm9ubWVudCB2YXJpYWJsZXM6CiAgICAgLSAqKlBPUlQqKjogVGhlIHBvcnQgd2hlcmUgdGhlIHdlYiBzZXJ2aWNlIHdpbGwgYmUgbGlzdGVuaW5nLiBGb3IgdGhpcyBjYXNlLCB5b3UgbmVlZCB0byB1c2UgdGhlICoqODA4MCoqIHZhbHVlLiAKICAgICAtICoqREJfVVJMKio6IFRoZSBVUkwgb2YgdGhlIFBvc3RncmVTUUwgZGF0YWJhc2UuCiAgICAgLSAqKkRCX1VTRVIqKjogVGhlIHVzZXIgb2YgdGhlIFBvc3RncmVTUUwgZGF0YWJhc2UuCiAgICAgLSAqKkRCX1BBU1NXRCoqOiBUaGUgcGFzc3dvcmQgb2YgdGhlIFBvc3RncmVTUUwgZGF0YWJhc2UuCjUuIENsaWNrIHRoZSAqKkNyZWF0ZSBXZWIgU2VydmljZSoqIGJ1dHRvbi4KCiMjIEhvdyB0byB1c2UKClRvIHVzZSB0aGlzIHByb2plY3QsIHlvdSBuZWVkIHRvIGZvbGxvdyB0aGUgZm9sbG93aW5nIHN0ZXBzOgoKMS4gVXNlIFBvc3RtYW4gb3IgYW55IG90aGVyIHRvb2wgdG8gbWFrZSBIVFRQIHJlcXVlc3RzLiBZb3UgY2FuIGZpbmQgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBvZiB0aGlzIHByb2plY3QgaW4gdGhlIAogICBgL3YzL2FwaS1kb2NzYCBlbmRwb2ludCBvZiB0aGUgd2ViIHNlcnZpY2UuCjIuIFlvdSBjYW4gdXNlIHRoZSBTd2FnZ2VyIFVJIHRvIG1ha2UgSFRUUCByZXF1ZXN0cy4gWW91IGNhbiBmaW5kIHRoZSBTd2FnZ2VyIFVJIGluIHRoZSBgL3N3YWdnZXItdWkvaW5kZXguaHRtbGAgZW5kcG9pbnQgb2YgCiAgIHRoZSB3ZWIgc2VydmljZS4KCiMjIEF1dGhvcgoKLSAqKkpvcmdlIEdhcmNpYSoqIC0gR2VvcmdlIHRoZSBQZW5ndWluCiAgLSBbV2Vic2l0ZV0oaHR0cHM6Ly9nZW9yZ2V0aGVwZW5ndWluLmRldi8pCiAgLSBbR2l0SHViXShodHRwczovL2dpdGh1Yi5jb20vZ2VvcmdlLXRoZS1wZW5ndWluKQogIC0gW1R3aXR0ZXJdKGh0dHBzOi8vdHdpdHRlci5jb20vbXJnZW9yZ2VwZW5ndWluKQogIC0gW0luc3RhZ3JhbV0oaHR0cHM6Ly93d3cuaW5zdGFncmFtLmNvbS9tcmdlb3JnZXBlbmd1aW4vKQogIC0gW1Rpa1Rva10oaHR0cHM6Ly93d3cudGlrdG9rLmNvbS9AZ2VvcmdlX3RoZV9wZW5ndWluKQogIC0gW1lvdVR1YmVdKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL0BHZW9yZ2VUaGVQZW5ndWluKQogIC0gW1R3aXRjaF0oaHR0cHM6Ly93d3cudHdpdGNoLnR2L2dlb3JnZV90aGVfcGVuZ3VpbikKCiMjIExpY2Vuc2UKClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UgLSBzZWUgdGhlIFtMSUNFTlNFXShMSUNFTlNFLm1kKSBmaWxlIGZvciBkZXRhaWxzCgojIyBBY2tub3dsZWRnbWVudHMKCi0gKipUcnlDYXRjaCoqIC0gW1NvY2lhbCBNZWRpYSBMaW5rc10oaHR0cHM6Ly9saW5rdHIuZWUvdHJ5Y2F0Y2gudHYpCgojIyBEaXNjbGFpbWVyCgpUaGlzIHByb2plY3QgaXMgbm90IGFmZmlsaWF0ZWQsIGFzc29jaWF0ZWQsIGF1dGhvcml6ZWQsIGVuZG9yc2VkIGJ5LCBvciBpbiBhbnkgd2F5IG9mZmljaWFsbHkgY29ubmVjdGVkIHdpdGggUmVuZGVyLCAKb3IgYW55IG9mIGl0cyBzdWJzaWRpYXJpZXMgb3IgaXRzIGFmZmlsaWF0ZXMuIFRoZSBvZmZpY2lhbCBSZW5kZXIgd2Vic2l0ZSBjYW4gYmUgZm91bmQgYXQgaHR0cHM6Ly9yZW5kZXIuY29tLiAKVGhlIG5hbWUgIlJlbmRlciIgYXMgd2VsbCBhcyByZWxhdGVkIG5hbWVzLCBtYXJrcywgZW1ibGVtcyBhbmQgaW1hZ2VzIGFyZSByZWdpc3RlcmVkIHRyYWRlbWFya3MgCm9mIHRoZWlyIHJlc3BlY3RpdmUgb3duZXJzLgo= readmeEtag: '"029377c325d684da559a9c5cf9867fa38671a7d0"' readmeLastModified: Fri, 21 Apr 2023 19:20:30 GMT repositoryId: 631014634 description: >- This is a simple Spring Boot REST API Proof of Concept (POC) project. The goal of this project is to demonstrate how to create a REST API using Spring Boot, and it can run properly in the Render ecosystem. created: '2023-04-21T17:45:21Z' updated: '2023-07-08T22:36:20Z' language: Java archived: false stars: 1 watchers: 1 forks: 1 owner: george-the-penguin logo: https://avatars.githubusercontent.com/u/129033766?v=4 license: MIT repoEtag: '"0fe333005f037fe55b3902098006df770aebf1fd791252bea359463fa98cafa7"' repoLastModified: Sat, 08 Jul 2023 22:36:20 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/refitter/generate-code v3: true id: 3e35276edc64b3c205a0484dbf849447 repositoryMetadata: base64Readme: >- IyBHZW5lcmF0ZS1Db2RlCkdlbmVyYXRlIGEgQyMgUmVmaXQgY2xpZW50IGludGVyZmFjZSBhbmQgY29udHJhY3RzIGZyb20gYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBkb2N1bWVudCB1c2luZyBbUmVmaXR0ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9jaHJpc3RpYW5oZWxsZS9yZWZpdHRlcikuCgojIyBJbnB1dHMKCiMjIyBgb3BlbmFwaS1maWxlYApUaGUgcGF0aCB0byB0aGUgT3BlbkFQSSBkb2N1bWVudCAoYm90aCBKU09OIGFuZCBZQU1MIGFyZSBzdXBwb3J0ZWQpLiBEZWZhdWx0cyB0byBgIm9wZW5hcGkuanNvbiJgIChpLmUuIGEgZmlsZSBpbiB0aGUgY3VycmVudCBkaXJlY3RvcnkgY2FsbGVkIG9wZW5hcGkuanNvbikuIFBhdGhzIHRoYXQgZG8gbm90IHN0YXJ0IHdpdGggYC9gIGFyZSBhc3N1bWVkIHRvIGJlIHJlbGF0aXZlIHRvIHRoZSByb290IG9mIHRoZSByZXBvc2l0b3J5LgoKIyMjIGBvcGVuYXBpLXVybGAKVGhlIFVSTCB0byBsb2FkIHRoZSBPcGVuQVBJIGRvY3VtZW50IGZyb20uIElmIHNldCwgYG9wZW5hcGktZmlsZWAgd2lsbCBiZSBpZ25vcmVkLgoKIyMjIGBuYW1lc3BhY2VgClRoZSBkZWZhdWx0IG5hbWVzcGFjZSB1c2VkIGZvciB0aGUgZ2VuZXJhdGVkIHR5cGVzIChkZWZhdWx0OiBgR2VuZXJhdGVkQ29kZWApCgojIyMgYHVzZS1hcGktcmVzcG9uc2VgClJldHVybiBgVGFzazxJQXBpUmVzcG9uc2U8VD4+YCBpbnN0ZWFkIG9mIGBUYXNrPFQ+YAoKIyMjIGBjYW5jZWxsYXRpb24tdG9rZW5zYApVc2UgY2FuY2VsbGF0aW9uIHRva2VucwoKIyMjIGBtdWx0aXBsZS1pbnRlcmZhY2VzYApHZW5lcmF0ZSBhIFJlZml0IGludGVyZmFjZSBmb3IgZWFjaCBlbmRwb2ludC4gTWF5IGJlIG9uZSBvZiBgQnlFbmRwb2ludGAsIGBCeVRhZ2AKCiMjIyBgY29tbWFuZC1hcmdzYApBZGRpdGlvbmFsIGFyZ3VtZW50cyB0byBwYXNzIHRocm91Z2ggdG8gdGhlIFtSZWZpdHRlcl0oaHR0cHM6Ly9naXRodWIuY29tL2NocmlzdGlhbmhlbGxlL3JlZml0dGVyKSBDTEkgdG9vbC4KCiMjIyBgcHVibGlzaC1hcnRpZmFjdHNgClNldHRpbmcgdGhpcyB3aWxsIHB1Ymxpc2ggdGhlIGdlbmVyYXRlZCBjb2RlIGFzIEMjIGZpbGVzIGFzIGJ1aWxkIGFydGlmYWN0cwoKIyMjIGBvdXRwdXQtZmlsZW5hbWVgCk5vIG91dHB1dHMgYXJlIHJldHVybmVkLiBUaGUgZ2VuZXJhdGVkIGNsaWVudCBpcyBwbGFjZWQgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5IGFuZCB1c2luZyB0aGUgYG91dHB1dC1maWxlbmFtZWAgKERlZmF1bHQ6ICoqYE91dHB1dC5jc2AqKikgdmFsdWUuIFRoZSBvdXRwdXQgZmlsZSBjb250YWlucyBib3RoIHRoZSBSZWZpdCBpbnRlcmZhY2UgYW5kIHRoZSBjb250cmFjdCB0eXBlcyB1c2VkIGJ5IHRoZSBBUEkKCgojIEV4YW1wbGVzCgojIyMgRnJvbSBhIEZpbGUKCmBgYHlhbWwKam9iczoKICBzbW9rZS10ZXN0LXVybDoKICAgIHJ1bnMtb246IHVidW50dS1sYXRlc3QKICAgIHN0ZXBzOgogICAgICAtIHVzZXM6IGFjdGlvbnMvY2hlY2tvdXRAdjMKICAgICAgCiAgICAgICMgVXNlIHRoZSBhY3Rpb24gdG8gZ2VuZXJhdGUgYSBSZWZpdCBjbGllbnQgaW50ZXJmYWNlIGZyb20gYSBmaWxlCiAgICAgICMgVGhpcyBwcm9kdWNlcyBhIGZpbGUgY2FsbGVkIE91dHB1dC5jcyBhcyBhIGJ1aWxkIGFydGlmYWN0CiAgICAgIC0gdXNlczogcmVmaXR0ZXIvZ2VuZXJhdGUtY29kZUB2MQogICAgICAgIG5hbWU6IEdlbmVyYXRlIFJlZml0IENsaWVudAogICAgICAgIHdpdGg6ICAgICAgICAKICAgICAgICAgIG9wZW5hcGktZmlsZTogb3BlbmFwaS5qc29uCiAgICAgICAgICBuYW1lc3BhY2U6IENocmlzdGlhbkhlbGxlLkV4YW1wbGVzLlBldHN0b3JlLnYzCiAgICAgIAogICAgICAjIERvIHNvbWV0aGluZyB3aXRoIHRoZSBnZW5lcmF0ZWQgY2xpZW50IChsaWtlIGluY2x1ZGUgaXQgaW4gYW4gZXhpc3RpbmcgcHJvamVjdCkKICAgICAgLSBydW46IEdldC1Db250ZW50IE91dHB1dC5jcyB8IFdyaXRlLUhvc3QKICAgICAgICBzaGVsbDogcHdzaApgYGAKCiMjIyBGcm9tIGEgVVJMCgpgYGB5YW1sCmpvYnM6CiAgc21va2UtdGVzdC11cmw6CiAgICBuYW1lOiBGcm9tIFVSTAogICAgcnVucy1vbjogdWJ1bnR1LWxhdGVzdAogICAgc3RlcHM6CiAgICAgIC0gdXNlczogYWN0aW9ucy9jaGVja291dEB2MwoKICAgICAgIyBVc2UgdGhlIGFjdGlvbiB0byBnZW5lcmF0ZSBhIFJlZml0IGNsaWVudCBpbnRlcmZhY2UgZnJvbSBhIFVSTAogICAgICAjIFRoaXMgcHJvZHVjZXMgYSBmaWxlIGNhbGxlZCBPdXRwdXRGcm9tVXJsLmNzIGFzIGEgYnVpbGQgYXJ0aWZhY3QKICAgICAgLSB1c2VzOiByZWZpdHRlci9nZW5lcmF0ZS1jb2RlQHYxCiAgICAgICAgbmFtZTogR2VuZXJhdGUgUmVmaXQgQ2xpZW50CiAgICAgICAgd2l0aDoKICAgICAgICAgIG9wZW5hcGktdXJsOiBodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92My9vcGVuYXBpLmpzb24KICAgICAgICAgIG5hbWVzcGFjZTogQ2hyaXN0aWFuSGVsbGUuRXhhbXBsZXMuUGV0c3RvcmUudjMKICAgICAgICAgIGNvbW1hbmQtYXJnczogLS1uby1sb2dnaW5nCiAgICAgICAgICBvdXRwdXQtZmlsZW5hbWU6IE91dHB1dEZyb21VcmwuY3MKICAgICAgCiAgICAgICMgRG8gc29tZXRoaW5nIHdpdGggdGhlIGdlbmVyYXRlZCBjbGllbnQgKGxpa2UgaW5jbHVkZSBpdCBpbiBhbiBleGlzdGluZyBwcm9qZWN0KQogICAgICAtIHJ1bjogR2V0LUNvbnRlbnQgT3V0cHV0RnJvbVVybC5jcyB8IFdyaXRlLUhvc3QKICAgICAgICBzaGVsbDogcHdzaApgYGA= readmeEtag: '"cce7698b6ebe00f0eca3b614eef0b61dc141cac5"' readmeLastModified: Tue, 08 Aug 2023 11:23:18 GMT repositoryId: 671843069 description: >- Generate a C# Refit client interface and contracts from an OpenAPI specifications document using Refitter. created: '2023-07-28T09:08:49Z' updated: '2023-12-15T04:34:25Z' language: null archived: false stars: 1 watchers: 1 forks: 0 owner: Refitter logo: https://avatars.githubusercontent.com/u/140486945?v=4 license: GPL-3.0 repoEtag: '"0e7ba862ab9c0509bf3ee97724b445f26d4cbefa78e11b6cc9214c6b3d6227a5"' repoLastModified: Fri, 15 Dec 2023 04:34:25 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/supermomonga/openapi_rails_typed_parameters v3: true id: 0ddf07584c0d7476a6b4c508b4e40824 repositoryMetadata: base64Readme: >- IyBPcGVuYXBpUmFpbHNUeXBlZFBhcmFtZXRlcnMKCmBvcGVuYXBpX3JhaWxzX3R5cGVkX3BhcmFtZXRlcnNgIGlzIGEgbGlicmFyeSBmb3IgcGVyZm9ybWluZyBwYXJhbWV0ZXIgdmFsaWRhdGlvbiBhbmQgY29lcmNpb24gaW4gUmFpbHMgdXNpbmcgT3BlbkFQSSBkZWZpbml0aW9uIGZpbGVzLiBJdCBhbHNvIHN1cHBvcnRzIHByb3ZpZGluZyBzdGF0aWMgdHlwZXMgdGhyb3VnaCBSQlMgZ2VuZXJhdGlvbi4KClRoaXMgZ2VtIGludGVybmFsbHkgdXNlcyB0aGUgW2BvcGVuYXBpX2ZpcnN0YF0oaHR0cHM6Ly9naXRodWIuY29tL2FoeC9vcGVuYXBpX2ZpcnN0KSBnZW0uCgojIyMgTWFpbiBPYmplY3RpdmVzCgotIEVsaW1pbmF0ZSB0aGUgY29kZSBmb3IgcGFyYW1zIHZhbGlkYXRpb24gYW5kIHR5cGUgY29udmVyc2lvbiAod2hpY2ggaXMgYWx3YXlzIGEgYm9yaW5nIGFuZCByb3V0aW5lIHRhc2spCi0gUHJvdmlkZSBhbiBSQlMgZ2VuZXJhdG9yIChzbyB5b3UgZG9uJ3QgaGF2ZSB0byBtZW1vcml6ZSBwYXJhbWV0ZXJzKQoKIyMjIERpZmZlcmVuY2VzIGZyb20gdGhlIGBvcGVuYXBpX2ZpcnN0YCBnZW0KCi0gT2ZmZXJzIG1vcmUgdGlnaHRseSBjb3VwbGVkIGZlYXR1cmVzIHRvIFJhaWxzCi0gQXNzaXN0cyBpbiBzdGF0aWNhbGx5IHR5cGVkIHByb2dyYW1taW5nIGJ5IHByb3ZpZGluZyBhbiBSQlMgZ2VuZXJhdG9yCgojIyA6d2FybmluZzogVGhpcyBnZW0gaXMgdW5kZXIgZGV2ZWxvcG1lbnQuCgpUaGlzIGdlbSBpcyBjdXJyZW50bHkgdW5kZXIgZGV2ZWxvcG1lbnQuIEl0IGlzIG5vdCByZWNvbW1lbmRlZCBmb3IgdXNlIGluIHByb2R1Y3Rpb24gYXBwbGljYXRpb25zLiBCYWNrd2FyZCBjb21wYXRpYmlsaXR5IGlzIG5vdCBndWFyYW50ZWVkIHVudGlsIHZlcnNpb24gMS4wLjAgaXMgcmVsZWFzZWQuCgpPbmNlIGFsbCBvZiB0aGUgZm9sbG93aW5nIFRPRE9zIGFyZSBjb21wbGV0ZWQsIHdlIHdpbGwgcmVsZWFzZSB2ZXJzaW9uIDEuMC4wLgoKIyMjIFRPRE8KCi0gWyBdIENvZXJjaW9uIHRvIG1vcmUgUmFpbHMgYXBwbGljYXRpb24gYXBwcm9wcmlhdGUgdHlwZXMsIHN1Y2ggYXMgYEFjdGl2ZVN1cHBvcnQ6OlRpbWVXaXRoWm9uZWAKLSBbIF0gUHJvdmlkZSBtb3JlIG9waW5pb25hdGVkIG9wdGlvbnMgZm9yIHR5cGUgY29udmVyc2lvbnMsIHN1Y2ggYXMgY29lcmNpb24gdG8gYFN5bWJvbGAgZm9yIEVudW0gdmFsdWUKLSBbIF0gSW5jbHVkZSBhIFJCUyBnZW5lcmF0aW9uIGdlbmVyYXRvciBmb3Igc3RhdGljYWxseSB0eXBlZCBwcm9ncmFtbWluZwotIFsgXSBJbmNsdWRlIFJCUyBpbiB0aGUgZ2VtIGl0c2VsZgoKCiMjIEluc3RhbGxhdGlvbgoKQWRkIGBvcGVuYXBpX3JhaWxzX3R5cGVkX3BhcmFtZXRlcnNgIHRvIHlvdXIgR2VtZmlsZS4KCmBgYHJiCmdlbSAnb3BlbmFwaV9yYWlsc190eXBlZF9wYXJhbWV0ZXJzJwpgYGAKCiMjIFVzYWdlCgpQbGVhc2UgYWRkIGFuIGluaXRpYWxpemVyIHRvIHlvdXIgUmFpbHMgYXBwbGljYXRpb24gYW5kIHNwZWNpZnkgdGhlIHBhdGggdG8gdGhlIE9wZW5BUEkgc2NoZW1hIGZpbGUuCgplLmcuKSBgY29uZmlnL2luaXRpYWxpemVycy9vcGVuYXBpLnJiYAoKYGBgb3BlbmFwaS5yYgpyZXF1aXJlICdvcGVuYXBpX3JhaWxzX3R5cGVkX3BhcmFtZXRlcnMnCgpPcGVuYXBpUmFpbHNUeXBlZFBhcmFtZXRlcnMuY29uZmlndXJlIGRvIHxjb25maWd8CiAgY29uZmlnLnNjaGVtYV9wYXRoID0gUmFpbHMucm9vdC5qb2luKCdzY2hlbWEueW1sJykKZW5kCmBgYAoKVGhlbiwgYWRkIGB1c2luZyBPcGVuYXBpUmFpbHNUeXBlZFBhcmFtZXRlcnNgIHRvIHlvdXIgY29udHJvbGxlciBjbGFzcy4gWW91IGNhbiBhY2Nlc3Mgc3RhdGljYWxseSB0eXBlZCBwYXJhbWV0ZXJzIHZpYSBgdHlwZWRfcGFyYW1ldGVyc2AgbWV0aG9kLgoKIyMgRXhhbXBsZQoKYGBgc2NoZW1hLnltbApvcGVuYXBpOiAzLjAuMwppbmZvOgogIHZlcnNpb246IDEuMC4wCiAgdGl0bGU6IFNhbXBsZSBBcHAKc2VydmVyczoKICAtIHVybDogaHR0cHM6Ly9leGFtcGxlLmNvbS8KcGF0aHM6CiAgL3VzZXJzOgogICAgZ2V0OgogICAgICBwYXJhbWV0ZXJzOgogICAgICAgIC0gbmFtZTogcm9sZQogICAgICAgICAgaW46IHF1ZXJ5CiAgICAgICAgICByZXF1aXJlZDogdHJ1ZQogICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICB0eXBlOiBzdHJpbmcKICAgICAgICAgICAgZW51bTogWyBhZG1pbiwgbWFpbnRhaW5lciBdCiAgICAgICAgLSBuYW1lOiBtaW5pbXVtCiAgICAgICAgICBpbjogcXVlcnkKICAgICAgICAgIHJlcXVpcmVkOiBmYWxzZQogICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICB0eXBlOiBpbnRlZ2VyCiAgICAgICAgLSBuYW1lOiBtYXhpbXVtCiAgICAgICAgICBpbjogcXVlcnkKICAgICAgICAgIHJlcXVpcmVkOiBmYWxzZQogICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICB0eXBlOiBpbnRlZ2VyCmBgYAoKYGBgdXNlcnNfY29udHJvbGxlci5yYgpjbGFzcyBVc2Vyc0NvbnRyb2xsZXIgPCBBcHBsaWNhdGlvbkNvbnRyb2xsZXIKICB1c2luZyBPcGVuYXBpUmFpbHNUeXBlZFBhcmFtZXRlcnMKCiAgZGVmIGluZGV4CiAgICB0eXBlZF9wYXJhbXMudmFsaWRhdGUhCiAgICByZXMgPSB7CiAgICAgIHJvbGU6IHR5cGVkX3BhcmFtcy5xdWVyeV9wYXJhbXMucm9sZSAjICdhZG1pbicgb3IgJ21haW50YWluZXInLAogICAgICBtaW5pbXVtOiB0eXBlZF9wYXJhbXMucXVlcnlfcGFyYW1zLm1pbmltdW0gIyBJbnRlZ2VyIHZhbHVlCiAgICAgIG1heGltdW06IHR5cGVkX3BhcmFtcy5xdWVyeV9wYXJhbXMubWF4aW11bSAjIEludGVnZXIgdmFsdWUKICAgIH0KICAgIHJlbmRlciBqc29uOiByZXMKICByZXNjdWUgT3BlbmFwaUZpcnN0OjpSZXF1ZXN0SW52YWxpZEVycm9yID0+IGUKICAgIHJlcyA9IHsKICAgICAgIyBlLmcuKQogICAgICAjIFF1ZXJ5IHBhcmFtZXRlciBpcyBpbnZhbGlkOiB2YWx1ZSBhdCBgL3JvbGVgIGlzIG5vdCBvbmUgb2Y6IFsiYWRtaW4iLCAibWFpbnRhaW5lciJdCiAgICAgIG1lc3NhZ2U6IGUubWVzc2FnZQogICAgfQogICAgcmVuZGVyIGpzb246IHJlcywgc3RhdHVzOiA6YmFkX3JlcXVlc3QKICBlbmQKZW5kCmBgYAoKCiMjIExpY2Vuc2UKClRoZSBnZW0gaXMgYXZhaWxhYmxlIGFzIG9wZW4gc291cmNlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgW01JVCBMaWNlbnNlXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkuCgojIyBDb2RlIG9mIENvbmR1Y3QKCkV2ZXJ5b25lIGludGVyYWN0aW5nIGluIHRoZSBPcGVuYXBpUmFpbHNUeXBlZFBhcmFtZXRlcnMgcHJvamVjdCdzIGNvZGViYXNlcywgaXNzdWUgdHJhY2tlcnMsIGNoYXQgcm9vbXMgYW5kIG1haWxpbmcgbGlzdHMgaXMgZXhwZWN0ZWQgdG8gZm9sbG93IHRoZSBbY29kZSBvZiBjb25kdWN0XShodHRwczovL2dpdGh1Yi5jb20vc3VwZXJtb21vbmdhL29wZW5hcGlfcmFpbHNfdHlwZWRfcGFyYW1ldGVycy9ibG9iL21haW4vQ09ERV9PRl9DT05EVUNULm1kKS4K readmeEtag: '"dbc7d3281a7677dcb41a994a33da645e14b8b937"' readmeLastModified: Sun, 07 Jan 2024 09:10:53 GMT repositoryId: 739426895 description: >- Eliminate manual params validation code in Rails by using OpenAPI specification. created: '2024-01-05T14:40:52Z' updated: '2024-01-26T20:40:19Z' language: Ruby archived: false stars: 1 watchers: 1 forks: 0 owner: supermomonga logo: https://avatars.githubusercontent.com/u/377137?v=4 license: MIT repoEtag: '"f06ca62518337c1973c3b0f80fd42a783b9c7f4c659b4b3b718827731b295b77"' repoLastModified: Fri, 26 Jan 2024 20:40:19 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/connorjs/swapi-typespec v3: true id: 1def74b05c01bb642f02fdfa6e0fc268 repositoryMetadata: base64Readme: >- IyBTV0FQSSBUeXBlU3BlYwoKW1R5cGVTcGVjXVt0eXBlc3BlY10gcmVwcmVzZW50YXRpb24gb2YgU1dBUEk6IFRoZSBTdGFyIFdhcnMgQVBJLgoK8J+OiSBTaG91dCBvdXQgdG8gW3N3YXBpLmluZm9dW3N3YXBpXSBmb3IgaG9zdGluZyBhIHN0YWJsZSB2ZXJzaW9uIG9mIHRoZSBBUEkuIPCfjokKCltzd2FwaV06IGh0dHBzOi8vc3dhcGkuaW5mbwpbdHlwZXNwZWNdOiBodHRwczovL21pY3Jvc29mdC5naXRodWIuaW8vdHlwZXNwZWMvCgo8IS0tIEJhZGdlczogVXNlIGxpbmsgc2hvcnRoYW5kIGZvciByZWFkYWJpbGl0eSBhbmQgZWFzaWVyIHJlb3JkZXJpbmcgLS0+CgpbIVtDSSBCdWlsZF1bY2ktYnVpbGQtYmFkZ2VdXVtjaS1idWlsZF0KWyFbbnBtIHZlcnNpb25dW25wbS12ZXJzaW9uLWJhZGdlXV1bbnBtLXZlcnNpb25dCgpbY2ktYnVpbGRdOiBodHRwczovL2dpdGh1Yi5jb20vY29ubm9yanMvc3dhcGktdHlwZXNwZWMvYWN0aW9ucy93b3JrZmxvd3MvY2ktYnVpbGQueW1sCltjaS1idWlsZC1iYWRnZV06IGh0dHBzOi8vZ2l0aHViLmNvbS9jb25ub3Jqcy9zd2FwaS10eXBlc3BlYy9hY3Rpb25zL3dvcmtmbG93cy9jaS1idWlsZC55bWwvYmFkZ2Uuc3ZnCltucG0tdmVyc2lvbl06IGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3N3YXBpLXR5cGVzcGVjCltucG0tdmVyc2lvbi1iYWRnZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3Yvc3dhcGktdHlwZXNwZWMKCjwhLS0gRW5kIGJhZGdlcyAtLT4KCiMjIEFib3V0CgpJIGNyZWF0ZWQgdGhpcyBwcm9qZWN0IHRvIGxlYXJuIGFib3V0IFR5cGVTcGVjIHRocm91Z2ggZmlyc3QtaGFuZCB1c2FnZS4KClRoZSBnZW5lcmF0ZWQgT3BlbiBBUEkgMyBzcGVjaWZpY2F0aW9uIGhhcyA4NzggbGluZXMgKHdpdGggbm8gYmxhbmsgbGluZXMpIGluIGEgc2luZ2xlIGZpbGUuClRoZSBUeXBlU3BlYyBzb3VyY2UgaGFzIDQ5MyBsaW5lcyAoYWJvdXQg4oWTIGFyZSBibGFuayBsaW5lcyBmb3IgcmVhZGFiaWxpdHkpIHNwbGl0IGFjcm9zcyA4IGZpbGVzIGZvciBsb2dpY2FsIG9yZ2FuaXphdGlvbi4KVGhpcyBpcyBhIH40NSUgcmVkdWN0aW9uIGluIGxpbmVzIG9mIGNvZGUuCgpJZiB3ZSBpZ25vcmUgdGhlIGltcG9ydHMsIHVzaW5nIHN0YXRlbWVudHMsIGFuZCBibGFuayBsaW5lcyB3aXRoaW4gbW9kZWxzLCB0aGVuIHRoZSBUeXBlU3BlYyBvbmx5IGhhcyAzNTMgbGluZXMuClRoYXQgbWVhbnMgYSA2MCUgcmVkdWN0aW9uIGluIGNvZGUgdG8gbWFpbnRhaW4uCgojIyBJbnN0YWxsYXRpb24KCkZvbGxvdyB0aGVzZSBzdGVwcyB0byBjb25zdW1lIHRoZSBnZW5lcmF0ZWQgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiBmb3IgU1dBUEkuCgoxLiBJbnN0YWxsIHRoaXMgcGFja2FnZSBhcyBhIGRlcGVuZGVuY3kuCgogICBgYGBzaAogICBucG0gaW5zdGFsbCAtRCBzd2FwaS10eXBlc3BlYwogICBgYGAKCjIuIFJlZmVyZW5jZSB0aGUgZ2VuZXJhdGVkIE9wZW4gQVBJIHNwZWNpZmljYXRpb24gZGlyZWN0bHkuCgogICBgYGBzaAogICAuL25vZGVfbW9kdWxlcy9zd2FwaS10eXBlc3BlYy9zd2FwaS5vcGVuYXBpLnlhbWwKICAgYGBgCgojIyBDb250cmlidXRpbmcKClNlZSBbQ09OVFJJQlVUSU5HLm1kXSguL0NPTlRSSUJVVElORy5tZCkgZm9yIGRldGFpbHMuCg== readmeEtag: '"1d5815cc44e78538d16b7e72233a2c08e6acb0f6"' readmeLastModified: Tue, 15 Apr 2025 20:44:47 GMT repositoryId: 689152742 description: 'TypeSpec representation of SWAPI: The Star Wars API.' created: '2023-09-09T00:02:29Z' updated: '2025-04-15T20:44:52Z' language: TypeSpec archived: false stars: 1 watchers: 1 forks: 0 owner: connorjs logo: https://avatars.githubusercontent.com/u/22208536?v=4 license: MIT repoEtag: '"190bd4be700a2a28849179a2c19a27129b1901d12ee49318272bbbc09406b68a"' repoLastModified: Tue, 15 Apr 2025 20:44:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/tinello/tomcat-embedded-openapi v3: true id: 405ea524b68d2becff287386f4e517b0 repositoryMetadata: base64Readme: >- IyBUb21jYXQtRW1iZWRkZWQtT3BlbkFwaQoKIyMgUmVxdWlyZW1lbnRzIDxhIG5hbWU9IlJlcXVpcmVtZW50cyI+PC9hPgojIyMgU29mdHdhcmU6IDxhIG5hbWU9IlNvZnR3YXJlIj48L2E+Ci0gT3BlbkpESyAyMyB3aXRoIE9wZW5KOSAtPiBodHRwczovL2RldmVsb3Blci5pYm0uY29tL2xhbmd1YWdlcy9qYXZhL3NlbWVydS1ydW50aW1lcy9kb3dubG9hZHMvCi0gRG9ja2VyIDQuMjguMCAtPiBodHRwczovL2RvY3MuZG9ja2VyLmNvbS9kZXNrdG9wL2luc3RhbGwvdWJ1bnR1LwoKIyMjIENvbmZpZ3VyZSBKREsgVlNDb2RlOiA8YSBuYW1lPSJDb25maWd1cmVKREt2c2NvZGUiPjwvYT4KCkNoYW5nZSAvLnZzY29kZS9sYXVuY2guanNvbiBmb3IgeW91ciBKYXZhIEhvbWUKCgojIyMgRW52aXJvbm1lbnQgdmFyaWFibGVzOiA8YSBuYW1lPSJFbnZpcm9ubWVudFZhcmlhYmxlcyI+PC9hPgotIERCX1VSTD1sb2NhbGhvc3Q6NTQzMi9wb3N0Z3JlcwotIERCX1VTRVI9cG9zdGdyZXMKLSBEQl9QQVNTPW15c2VjcmV0cGFzc3dvcmQKCiMjIyBTdGFydCBQb3N0Z3JlU1FMIDxhIG5hbWU9IlN0YXJ0UG9zdGdyZVNRTCI+PC9hPgpgYGBiYXNoCmRvY2tlciBydW4gLS1uYW1lIHRvbWNhdC1vcGVuYXBpLXBvc3RncmVzIC1wIDU0MzI6NTQzMiAtZSBQT1NUR1JFU19QQVNTV09SRD1teXNlY3JldHBhc3N3b3JkIC1kIHBvc3RncmVzOjE2LjMtYWxwaW5lMy4xOApgYGAKCgojIyMgU3RhcnQgYXBwbGljYXRpb24gTWF2ZW4gKG5vdCBmb3VuZCkgPGEgbmFtZT0iU3RhcnRBcHBsaWNhdGlvbk1hdmVuIj48L2E+CmBgYGJhc2gKREJfVVJMPSJsb2NhbGhvc3Q6NTQzMi9wb3N0Z3JlcyIgREJfVVNFUj1wb3N0Z3JlcyBEQl9QQVNTPW15c2VjcmV0cGFzc3dvcmQgSkFWQV9IT01FPS9ob21lL2cvREVWL1Rvb2xzL2pkay0yMy4wLjIrNyAuL212bncgY2xlYW4gY29tcGlsZSBleGVjOmphdmEgLVBydW4KYGBgCgojIyMgQnVpbGQgYXBwbGljYXRpb24gTWF2ZW4gPGEgbmFtZT0iQnVpbGRBcHBsaWNhdGlvbk1hdmVuIj48L2E+CmBgYGJhc2gKSkFWQV9IT01FPS9ob21lL2cvREVWL1Rvb2xzL2pkay0yMy4wLjIrNyAuL212bncgY2xlYW4gcGFja2FnZSAtRHNraXBUZXN0cwpgYGAKCgojIyMgU3RhcnQgYXBwbGljYXRpb24gSmF2YSA8YSBuYW1lPSJTdGFydEFwcGxpY2F0aW9uSmF2YSI+PC9hPgpgYGBiYXNoCkRCX1VSTD0ibG9jYWxob3N0OjU0MzIvcG9zdGdyZXMiIERCX1VTRVI9cG9zdGdyZXMgREJfUEFTUz1teXNlY3JldHBhc3N3b3JkIC9ob21lL2cvREVWL1Rvb2xzL2pkay0yMy4wLjIrNy9iaW4vamF2YSAtWFg6K1RpZXJlZFN0b3BBdExldmVsPTEgLWphciB0b21jYXRlbWJlZC0xLjEuMC5qYXIKYGBgCgoKIyMjIE9XQVNQIERlcGVuZGVuY2llcyBDaGVjayA8YSBuYW1lPSJPV0FTUERlcGVuZGVuY2llc0NoZWNrIj48L2E+CmBgYGJhc2gKSkFWQV9IT01FPS9ob21lL2cvREVWL1Rvb2xzL2pkay0yMy4wLjIrNyBtdm4gY2xlYW4gdmVyaWZ5IC1Ec2tpcFRlc3RzPXRydWUgLURkZXBlbmRlbmN5LmNoZWNrLnNraXA9ZmFsc2UgLURtYXZlbi5jbGVhbi5za2lwPXRydWUgLURtYXZlbi5jb21waWxlLnNraXA9dHJ1ZSAtRG1hdmVuLmluc3RhbGwuc2tpcD10cnVlCmBgYAoKCiMjIERldmVsb3BtZW50IDxhIG5hbWU9ImRldmVsb3BtZW50Ij48L2E+CiMjIyBWaXN1YWwgU3R1ZGlvIENvZGUgRXh0ZW5zaW9uczogPGEgbmFtZT0idnNjb2RlLWV4dGVuc2lvbnMiPjwvYT4KIyMjIyBFeHRlbnNpb24gUGFjayBmb3IgSmF2YQoKSW5zdGFsbCAiRXh0ZW5zaW9uIFBhY2sgZm9yIEphdmEiIGZyb20gTWljcm9zb2Z0OiBodHRwczovL21hcmtldHBsYWNlLnZpc3VhbHN0dWRpby5jb20vaXRlbXM/aXRlbU5hbWU9dnNjamF2YS52c2NvZGUtamF2YS1wYWNrCgoKIyMgRW5kcG9pbnRzCgp8IE5hbWUgICAgICAgICAgICAgICAgIHwgRW5kcG9pbnQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAp8IFNlcnZpY2UgSW5mbyAgICAgICAgIHwgaHR0cDovL2xvY2FsaG9zdDo4MDgwLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAoKCiMjIE9ic2VydmFiaWxpdHkgPGEgbmFtZT0ib2JzZXJ2YWJpbGl0eSI+PC9hPgoKIyMjIFN0YXJ0IERvY2tlciBHcmFmYW5hIE9URUwgPGEgbmFtZT0ic3RhcnQtZG9ja2VyLWdyYWZhbmEtb3RlbCI+PC9hPgoKYGBgYmFzaApkb2NrZXIgcnVuIC0tcmVzdGFydCB1bmxlc3Mtc3RvcHBlZCAtLWRldGFjaCAtLXB1Ymxpc2ggMzAwMDozMDAwIC0tcHVibGlzaCA0MzE3OjQzMTcgLS1wdWJsaXNoIDQzMTg6NDMxOCAtLW5hbWUgZ3JhZmFuYV9vdGVsIGdyYWZhbmEvb3RlbC1sZ3RtOjAuMTEuMApgYGAKClJlZjogaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL2dyYWZhbmEvb3RlbC1sZ3RtCgojIyMgVmlldyBHcmFmYW5hIDxhIG5hbWU9InZpZXctZ3JhZmFuYSI+PC9hPgoKTG9nIGluIHRvIGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMCB3aXRoIHVzZXIgYWRtaW4gYW5kIHBhc3N3b3JkIGFkbWluLg== readmeEtag: '"845d46113376f9ec2bb2adfabb57dec7061f0d0d"' readmeLastModified: Thu, 21 Aug 2025 21:03:45 GMT repositoryId: 666518593 description: API Rest with Apache Tomcat, Open Api Validate. created: '2023-07-14T18:20:21Z' updated: '2025-12-20T14:28:25Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: tinello logo: https://avatars.githubusercontent.com/u/38929644?v=4 license: Unlicense repoEtag: '"d70b6a99078c3eaf5417f82a7050830ebf60b7cb890313d13f5f042d174d11bf"' repoLastModified: Sat, 20 Dec 2025 14:28:25 GMT category: SDK foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/saman-barakat/idlgateway v3: true v3_1: true id: 174e0f826611dac751b49f8d735df68d repositoryMetadata: base64Readme: >- # IDLFilter for Spring Cloud Gateway  

## Overview  

`IDLFilter` is a custom filter for **Spring Cloud Gateway**, designed to handle **inter-parameter dependencies** in web APIs. It ensures that API requests comply with constraints defined in an **OpenAPI Specification (OAS) file**, applying specific analysis modes:  

- **Detection** – Identifies constraint violations.  
- **Explanation** – Provides reasons for violations.  
- **None** – Disables validation.  

This document provides instructions on configuring and applying `IDLFilter` in **Spring Cloud Gateway**, using an example service that interacts with the **Yelp API**.  

---

## How IDLFilter Works  

`IDLFilter` extends `AbstractGatewayFilterFactory`, integrating `IDLReasoner` to validate API requests in three steps:  

1. **Initialize the Analyzer** – Create an `IDLReasoner` instance with the **OAS file**, **operation path**, and **operation type**.  
2. **Validate Requests** – Check if the request meets the defined constraints.
3. **Generate Responses** – Return a response message if the request is invalid.

---

## Prerequisites  

Before using `IDLFilter`, ensure the following dependencies are included in your project:  

- **Java Development Kit (JDK) 11 or later**  
- **IDLReasoner 1.0.1**  
- **Spring Boot & Spring Cloud Gateway dependencies**  

---

## Simplified IDLFilter Code  

Below is an example implementation of `IDLFilter`:  

```java
@Component
public class IDLFilter extends AbstractGatewayFilterFactory<IDLFilter.Config> {
    private final Logger logger = LoggerFactory.getLogger(IDLFilter.class);

    public IDLFilter() {
        super(Config.class);
        logger.info("IDLFilter initialized");
    }

    private static final String BASE_SPEC_PATH = "./src/test/resources/GatewayExperiment";

    @Override
    public GatewayFilter apply(Config config) {
        logger.info("Applying IDLFilter with [{}] analysis mode", config.analysis);

        return (exchange, chain) -> {
            String operationPath = config.operationPath;
            String SPEC_URL = BASE_SPEC_PATH + config.specPath;
            String operationType = exchange.getRequest().getMethod().name().toLowerCase();
            Map<String, String> paramMap = exchange.getRequest().getQueryParams().toSingleValueMap();

            try {
                if ("Detection".equals(config.analysis) || "Explanation".equals(config.analysis)) {
                    // Step 1: Initialize the Analyzer
                    Analyzer analyzer = new OASAnalyzer(SPEC_URL, operationPath, operationType);

                    // Step 2: Validate the Request
                    boolean isValid = analyzer.isValidRequest(paramMap);

                    // Step 3: Generate the Response if Invalid
                    if (!isValid) {
                        String responseMessage = "The request is invalid!";
                        if ("Explanation".equalsIgnoreCase(config.analysis)) {
                            responseMessage = analyzer.getExplanationMessage(paramMap).toString();
                        }
                        throw new ResponseStatusException(HttpStatus.BAD_REQUEST, responseMessage);
                    }
                }
            } catch (IDLException e) {
                throw new ResponseStatusException(HttpStatus.BAD_REQUEST, e.getMessage());
            }
            return chain.filter(exchange);
        };
    }

    public static class Config {
        private String analysis;
        private String specPath;
        private String operationPath;

        // Getters and setters
        public String getOperationPath() { return operationPath; }
        public void setOperationPath(String operationPath) { this.operationPath = operationPath; }

        public String getSpecPath() { return specPath; }
        public void setSpecPath(String specPath) { this.specPath = specPath; }

        public String getAnalysis() { return analysis; }
        public void setAnalysis(String analysis) { this.analysis = analysis; }
    }
}
```
## Configuration  

To apply `IDLFilter` in **Spring Cloud Gateway**, update your `application.yaml` file with the appropriate route configuration.  

### Step-by-Step Instructions  

1. Navigate to `src/main/resources` in your Spring Boot project.  
2. Open (or create) `application.yaml`.  
3. Add a route configuration using the following structure:  

```yaml
spring:
  cloud:
    gateway:
      routes:
        - id: <UniqueRouteID>
          uri: <TargetServiceURI>
          predicates:
            - Path=<RequestPathPattern>
          filters:
            - name: IDLFilter
              args:
                analysis: <AnalysisMode>  # Options: Detection, Explanation, None
                serviceName: "<ServiceName>"
                operationPath: "<OperationPath>"
                specPath: "<SPECPath>"
```

### Configuration Parameters  

| Parameter       | Description |
|----------------|------------|
| **id**         | Unique identifier for the route (`<UniqueRouteID>`). |
| **uri**        | Target service URI (`<TargetServiceURI>`). |
| **predicates** | Defines request matching conditions (routes only `<RequestPathPattern>`). |
| **filters**    | Specifies the filter to be applied (`IDLFilter`). |
| **analysis**   | Analysis mode (`Detection`, `Explanation`, or `None`). |
| **operationPath** | API operation path (`<OperationPath>`). |
| **specPath** | Path to the OpenAPI specification file (`<SPECPath>`). |


### Example: Yelp Transactions Search  

Below is an example configuration for applying `IDLFilter` to the **Yelp Transactions Search** API:

```yaml
spring:
  cloud:
    gateway:
      routes:
        - id: YelpTransactionsSearch
          uri: https://api.yelp.com
          predicates:
            - Path=/v3/transactions/**
          filters:
            - name: IDLFilter
              args:
                analysis: Explanation  # Options: Detection, Explanation, None
                serviceName: "YelpTransactionsSearch"
                operationPath: "/transactions/delivery/search"
                specPath: "/YelpBusinessesSearch/openapi.yaml"
```

 readmeEtag: '"c7a6ee718d373fb4c0dc671857bb2569d232d296"' readmeLastModified: Sun, 23 Feb 2025 17:33:51 GMT repositoryId: 582038295 description: >- IDLFilter is a custom filter for Spring Cloud Gateway, specifically designed to manage inter-parameter dependencies in web APIs. created: '2022-12-25T12:12:17Z' updated: '2025-02-23T17:33:54Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: saman-barakat logo: https://avatars.githubusercontent.com/u/3259712?v=4 repoEtag: '"23d5b82ad17707267077f90210ede025152bc952ea1627cdc2a1e8603b87e37c"' repoLastModified: Sun, 23 Feb 2025 17:33:54 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/lams-epfl/gen-rest v3: true repositoryMetadata: base64Readme: >- IyBzZWFtLXRvLW9wZW5hcGkKClRoaXMgcmVwbyBob3N0cyBhIHRvb2wgdGhhdCB0YWtlcyBzZXJ2aWNlIG1vZGVscyBjcmVhdGVkIHdpdGggW1NlYW1DQURdKGh0dHA6Ly9sYW1zLmVwZmwuY2gvc2VhbWNhZC8pIGFuZCBnZW5lcmF0ZXMgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZyb20gdGhlIHdlYiBzZXJ2aWNlcyBhbm5vdGF0aW9ucyBvZiB0aGUgc2VydmljZSBtb2RlbHMuCgoK readmeEtag: '"c2cd348d6a9031fa4af1016e44d3ecbf4ba97bac"' readmeLastModified: Thu, 10 Jan 2019 17:06:05 GMT repositoryId: 122209000 description: >- A tool to generate automatically OpenAPI specification from a service model created: '2018-02-20T14:29:24Z' updated: '2023-06-06T09:11:54Z' language: Java archived: false stars: 1 watchers: 2 forks: 2 owner: lams-epfl logo: https://avatars.githubusercontent.com/u/34246217?v=4 repoEtag: '"3ff750bd1fc8b1741fb547e4bf6285f65711697b4ce8dd3e5e7ffe30944fecd6"' repoLastModified: Tue, 06 Jun 2023 09:11:54 GMT foundInMaster: true category: - Testing - Parsers id: 602854fe530d1f863fdb163da8c595df - source: openapi3 tags repository: https://github.com/teobler/openapi-schema v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJLXNjaGVtYQp0aGlzIHRvb2wgaXMgYnVpbHQgZm9yIGdlbmVyYXRpbmcgdHlwZXNjcmlwdCBpbnRlcmZhY2Ugb2YgT3BlbkFQSSBzY2hlbWEuCgpqc29uIGZpbGUgaXMgZnJvbSBbT3BlbkFQSS1TcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi90cmVlL21hc3Rlci9zY2hlbWFzL3YzLjApLCB5b3UgY2FuIGFsc28gZmluZCB5YW1sIGZpbGUgdGhlcmUsIHRoYW5rcyBmb3IgdGhlIG1haW50ZW5hbmNlLgoKIyMgTk9URQp0aGlzIHRvb2wgb25seSBmb3IgT3BlbkFQSSAqKnYzLjAueCoqIG5vdwoKIyMgaW5zdGFsbApgYGBiYXNoCiAgbnBtIGluc3RhbGwgLUQgQG9wZW5hcGktaW50ZWdyYXRpb24vb3BlbmFwaS1zY2hlbWEKYGBgCgojIyB1c2UKYGBgamF2YXNjcmlwdAogIGltcG9ydCB7IFNwZWMgfSBmcm9tICJAb3BlbmFwaS1pbnRlZ3JhdGlvbi9vcGVuYXBpLXNjaGVtYSI7CgogIGNvbnNvbGUubG9nKFNwZWMpOwpgYGAKCiMjIEhvdyB0byBTdGFydAoxLiBjbG9uZSByZXBvIHRvIHlvdXIgbG9jYWwKMi4gYG5wbSBpYAozLiBtb2RpZnkgY29kZSB0byB3aGF0IHlvdSB3YW50CjQuIGBucG0gcnVuIGdlbmVyYXRlYCB0byBnZW5lcmF0ZSBuZXcgaW50ZXJmYWNlIGZvciB0eXBlc2NyaXB0Cg== readmeEtag: '"61660b3e35a75fc61a783781c30d235ab89f9f8f"' readmeLastModified: Thu, 19 Mar 2020 08:19:33 GMT repositoryId: 247659506 description: A tool for generating OpenAPI schema interface for typescript. created: '2020-03-16T09:22:18Z' updated: '2023-03-07T12:36:25Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: teobler logo: https://avatars.githubusercontent.com/u/26424691?v=4 repoEtag: '"87a3bb24203a353547a000e435755ed2b12b5e8366ed8e8e9a673455ed2c9a68"' repoLastModified: Tue, 07 Mar 2023 12:36:25 GMT foundInMaster: true category: Parsers id: 5ca25503175774d442f849dc5ceae080 - source: openapi3 tags repository: https://github.com/koevskinikola/camunda-open-api-endpoints v3: true repositoryMetadata: base64Readme: >- IyBjYW11bmRhLW9wZW4tYXBpLWVuZHBvaW50cwpTY3JpcHRzIHRvIGNyZWF0ZSB0aGUgcHJvcGVyIHN0cnVjdHVyZSBhbmQgRlRMIHRlbXBsYXRlIGZpbGVzIGZvciB0aGUgQ2FtdW5kYSBPcGVuIEFQSSBzcGVjCgpUaGUgYGVuZHBvaW50cy5zaGAgYmFzaCBzY3JpcHQgaGFzIHRoZSBmb2xsb3dpbmcgZm9ybWF0OgoKYGBgc2gKLi9lbmRwb2ludHMuc2ggW2h0dHBfdmVyYl0gW3NwYWNlX3NlcGFyYXRlZF9lbnBvaW50X3BhdGhzXQpgYGAKCkZvciBleGFtcGxlLCB0aGUgY2FsbCB0byBgLi9lbmRwb2ludHMuc2ggZ2V0IHRhc2sve2lkfS9jb21wbGV0ZSB0YXNrL3tpZH0vYXNzaWduYAppbiBhIGdpdmVuIGRpcmVjdG9yeSwgd2lsbCBwcm9kdWNlIHRoZSBmb2xsb3dpbmcgZm9sZGVyIHN0cnVjdHVyZToKCiogUEFSRU5UX0RJUkVDVE9SWQogICogdGFzawogICAgKiB7aWR9CiAgICAgICogY29tcGxldGUKICAgICAgICAqIGdldC5mdGwKICAgICAgKiBhc3NpZ24KICAgICAgICAqIGdldC5mdGwKClRoZSBgXCouZnRsYCBmaWxlcyBsaXN0ZWQgYWJvdmUgd2lsbCBjb250YWluIHRoZSBmb2xsb3dpbmcgY29udGVudDoKYGBgeG1sCnsKCiAgPEBsaWIuZW5kcG9pbnRJbmZvCiAgICAgIGlkID0gIkFERF9JRCIKICAgICAgdGFnID0gIkFERF9UQUciCiAgICAgIGRlc2MgPSAiIiAvPgoKICAicGFyYW1ldGVycyIgOiBbCgogICAgPEBsaWIucGFyYW1ldGVyCiAgICAgICAgbmFtZSA9ICJpZCIKICAgICAgICBsb2NhdGlvbiA9ICJwYXRoIgogICAgICAgIHR5cGUgPSAic3RyaW5nIgogICAgICAgIHJlcXVpcmVkID0gdHJ1ZQogICAgICAgIGxhc3QgPSB0cnVlCiAgICAgICAgZGVzYyA9ICJUaGUgaWQgb2YgdGhlIEFERF9UWVBFIGZvciB3aGljaCBFTlRFUl9ERVNDIHNob3VsZCBiZSBBRERfQUNUSU9OLiIvPgogICAgPCNhc3NpZ24gbGFzdCA9IHRydWUgPgogICAgPCNpbmNsdWRlICIvbGliL2NvbW1vbnMvcGFnaW5hdGlvbi1wYXJhbXMuZnRsIiA+CgogIF0sCgogIDxAbGliLnJlcXVlc3RCb2R5CiAgICAgIG1lZGlhVHlwZSA9ICJhcHBsaWNhdGlvbi9qc29uIgogICAgICBkdG8gPSAiX0R0byIKICAgICAgZXhhbXBsZXMgPSBbJyJleGFtcGxlLTEiOiB7CiAgICAgICAgICAgICAgICAgICAgICAgInN1bW1hcnkiOiAiIiwKICAgICAgICAgICAgICAgICAgICAgICAiZGVzY3JpcHRpb24iOiAiIiwKICAgICAgICAgICAgICAgICAgICAgICAidmFsdWUiOgogICAgICAgICAgICAgICAgICAgICB9J10gLz4KCiAgInJlc3BvbnNlcyIgOiB7CgogICAgPEBsaWIucmVzcG9uc2UKICAgICAgICBjb2RlID0gIjIwMCIKICAgICAgICBkdG8gPSAiQUREX0RUTyIKICAgICAgICBhcnJheSA9IHRydWUKICAgICAgICBkZXNjID0gIlJlcXVlc3Qgc3VjY2Vzc2Z1bC4iIC8+CgogICAgPEBsaWIucmVzcG9uc2UKICAgICAgICBjb2RlID0gIjQwMCIKICAgICAgICBkdG8gPSAiRXhjZXB0aW9uRHRvIgogICAgICAgIGxhc3QgPSB0cnVlCiAgICAgICAgZGVzYyA9ICJCYWQgUmVxdWVzdC4gU2VlIHRoZQogICAgICAgICAgICAgICAgW0ludHJvZHVjdGlvbl0oJHtkb2NzVXJsfS9yZWZlcmVuY2UvcmVzdC9vdmVydmlldy8jZXJyb3ItaGFuZGxpbmcpCiAgICAgICAgICAgICAgICBmb3IgdGhlIGVycm9yIHJlc3BvbnNlIGZvcm1hdC4iIC8+CgogIH0KfQpgYGA= readmeEtag: '"73cd1d93679e214c83f76be6523e0a0ea19de7f8"' readmeLastModified: Thu, 19 Mar 2020 09:19:57 GMT repositoryId: 247750700 description: Create a proper structure and FTL templates for the Camunda OpenAPI spec created: '2020-03-16T15:42:58Z' updated: '2020-03-19T09:20:07Z' language: Shell archived: false stars: 1 watchers: 1 forks: 1 owner: koevskinikola logo: https://avatars.githubusercontent.com/u/1534291?v=4 license: Apache-2.0 repoEtag: '"e3441c2c4f5b51d41e75d7a091b16bf416e776f12b5c8262d946612cd3d4bcf3"' repoLastModified: Thu, 19 Mar 2020 09:20:07 GMT foundInMaster: true category: - Server Implementations - Parsers id: d26838616c53ebd5121442d05dd4e7df - source: openapi3 tags repository: https://github.com/ruddfawcett/ios-openapi-gen v3: true repositoryMetadata: base64Readme: >- IyBpb3Mtb3BlbmFwaS1nZW4K8J+OgSBXcmFwIFN3YWdnZXIgT3BlbkFQSXMgd2l0aCBhbiBBUEkgcGF0dGVybiBkZXZlbG9wZWQgYnkgW0BrZWFuXShodHRwczovL2tlYW4uZ2l0aHViLmlvL3Bvc3QvYXBpLWNsaWVudCkuIFJpZ2h0IG5vdyB0aGUgcHJvamVjdCB3aWxsIGRvIGV2ZXJ5dGhpbmcgYnV0IGZ1bmN0aW9uIG5hbWVzLCBidXQgd2lsbCB0YWNrbGUgdGhhdCBzb29uLgoKIyMgVXNhZ2UKLSDwn6eR4oCN8J+SuyBncmFiIGxhdGVzdCBgb3BlbmFwaS5qc29uYAotIPCfk50gZWRpdCBzZXR0aW5ncyBpbiBgc2V0dGluZ3MueW1sYAotIPCfm6AgcnVuIGBweXRob24gZ2VuLnB5YAotIPCflI4gZ28gdG8gYG91dHB1dC8qYAotIPCfmI4gdXNlIGdlbmVyYXRlZCBTd2lmdCBBUEkgbGF5ZXIKCiMjIEV4YW1wbGUgT3V0cHV0CgpgYGBzd2lmdApleHRlbnNpb24gQVBJIHsKICAgIGVudW0gU3RvcmUge30KfQoKZXh0ZW5zaW9uIEFQSS5TdG9yZSB7CgogICAgLy8vIFJldHVybnMgcGV0IGludmVudG9yaWVzIGJ5IHN0YXR1cwogICAgc3RhdGljIGZ1bmMgPCMgUmVhZGFibGUgTmFtZSAjPigpIC0+IEVuZHBvaW50PDwjIE1vZGVsIFR5cGUvVm9pZCAjPj4gewogICAgICAgIHJldHVybiBFbmRwb2ludChtZXRob2Q6IC5nZXQsIHBhdGg6ICIvc3RvcmUvaW52ZW50b3J5IikKICAgIH0KCiAgICAvLy8gUGxhY2UgYW4gb3JkZXIgZm9yIGEgcGV0CiAgICBzdGF0aWMgZnVuYyA8IyBSZWFkYWJsZSBOYW1lICM+KCkgLT4gRW5kcG9pbnQ8PCMgTW9kZWwgVHlwZS9Wb2lkICM+PiB7CiAgICAgICAgcmV0dXJuIEVuZHBvaW50KG1ldGhvZDogLnBvc3QsIHBhdGg6ICIvc3RvcmUvb3JkZXIiKQogICAgfQoKICAgIC8vLyBGaW5kIHB1cmNoYXNlIG9yZGVyIGJ5IGlkCiAgICBzdGF0aWMgZnVuYyA8IyBSZWFkYWJsZSBOYW1lICM+KG9yZGVySWQ6IEludCkgLT4gRW5kcG9pbnQ8PCMgTW9kZWwgVHlwZS9Wb2lkICM+PiB7CiAgICAgICAgcmV0dXJuIEVuZHBvaW50KG1ldGhvZDogLmdldCwgcGF0aDogIi9zdG9yZS9vcmRlci9cKG9yZGVySWQpIikKICAgIH0KCiAgICAvLy8gRGVsZXRlIHB1cmNoYXNlIG9yZGVyIGJ5IGlkCiAgICBzdGF0aWMgZnVuYyA8IyBSZWFkYWJsZSBOYW1lICM+KG9yZGVySWQ6IEludCkgLT4gRW5kcG9pbnQ8PCMgTW9kZWwgVHlwZS9Wb2lkICM+PiB7CiAgICAgICAgcmV0dXJuIEVuZHBvaW50KG1ldGhvZDogLmRlbGV0ZSwgcGF0aDogIi9zdG9yZS9vcmRlci9cKG9yZGVySWQpIikKICAgIH0KCn0KYGBgCgojIyBGdXR1cmUgUGxhbnMKLSDwn6ShIHBhcnNlIGZ1bGwgZG9jdW1lbnRhdGlvbiBmb3IgcGF0aHMKLSDwn5OWIGdlbmVyYXRlIHNlbWktcmVhZGFibGUgQVBJIGZ1bmN0aW9uIG5hbWVzCi0g8J+klyBoYW5kbGUgYm9keSBwYXJhbWV0ZXJzCi0g8J+WvCBzdXBwb3J0IGZyYW1ld29yay9wYXR0ZXJucyBiZXlvbmQgQWxhbW9maXJlCgojIyBDb2xvcGhvbgotIFtAcnVkZGZhd2NldHRdKGh0dHBzOi8vZ2l0aHViLmNvbS9ydWRkZmF3Y2V0dCk6IFB5dGhvbiBwYXJzZXIgYW5kIGNvbnZlcnRlciBmb3IgT3BlbkFQSXMuCi0gW0BrZWFuXShodHRwczovL2tlYW4uZ2l0aHViLmlvL3Bvc3QvYXBpLWNsaWVudCk6IEFQSSBsYXllciBjb25jZXB0LCBvcmlnaW5hbGx5IHdyaXR0ZW4gd2l0aCBBbGFtb2ZpcmUgYW5kIFJ4U3dpZnQuCg== readmeEtag: '"927fff6cf8eefbe4e0ed89ccab34774828759936"' readmeLastModified: Thu, 30 Apr 2020 23:53:31 GMT repositoryId: 260319500 description: Generate iOS client API wrappers from Swagger's OpenAPI spec. created: '2020-04-30T21:11:37Z' updated: '2022-09-06T16:54:15Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: ruddfawcett logo: https://avatars.githubusercontent.com/u/3122471?v=4 repoEtag: '"187537452ca69999271ddd9e709876df283244cf9a1d006704025d95a0a5558d"' repoLastModified: Tue, 06 Sep 2022 16:54:15 GMT foundInMaster: true category: - Data Validators - Parsers id: ad552fe08bfc4be2fef6db46332b5238 - source: openapi3 tags repository: https://github.com/opengeospatial/ogcapi-edr-sprint2 v3: true repositoryMetadata: base64Readme: >- IyBPR0MgQVBJIC0gRURSIFNwcmludCAyCgpbPGltZyBzcmM9Imh0dHA6Ly93d3cub3Blbmdlb3NwYXRpYWwub3JnL3B1Yi93d3cvZmlsZXMvT0dDX0xvZ29fMkRfQmx1ZV94XzBfMC5wbmciIHdpZHRoPSIyMDAiLz5dKGh0dHBzOi8vd3d3Lm9wZW5nZW9zcGF0aWFsLm9yZykKClRoaXMgR2l0aHViIHJlcG9zaXRvcnkgaXMgZm9yIHRoZSBzZWNvbmQgT0dDIEFQSSAtIEVEUiBjb2RlIHNwcmludCBmb2N1c2luZyBvbiB0aGUgW09HQyBBUEkgLSBFbnZpcm9ubWVudGFsIERhdGEgUmV0cmlldmFsIGNhbmRpZGF0ZSBzdGFuZGFyZF0oaHR0cHM6Ly93d3cub2djLm9yZy9wcmVzc3Jvb20vcHJlc3NyZWxlYXNlcy8zMjgwKS4KClsjT0dDQVBJXShodHRwczovL3R3aXR0ZXIuY29tL2hhc2h0YWcvT0dDQVBJKQoKCkFib3V0IHRoZSBDb2RlIFNwcmludAotLS0tLS0tLS0tLS0tLS0tCgoKVGhlIE9wZW4gR2Vvc3BhdGlhbCBDb25zb3J0aXVtIChPR0MpIGludml0ZXMgZGV2ZWxvcGVycyB0byB0aGUgT0dDIEFQSSAtIEVEUiBTcHJpbnQgMiB2aXJ0dWFsIGV2ZW50IHRvIGJlIGhlbGQgdGhyb3VnaCByZW1vdGUgcGFydGljaXBhdGlvbi93ZWItY29uZmVyZW5jaW5nIG9uIE5vdmVtYmVyIDktMTAsIDIwMjAsIGZyb20gOTowMGFtIDU6MzBwbSBbRVNUXShodHRwczovL3d3dy50aW1lYW5kZGF0ZS5jb20vd29ybGRjbG9jay91c2Evd2FzaGluZ3Rvbi1kYykuIFJlZ2lzdHJhdGlvbiBmb3IgdGhlIE9HQyBBUEkgLSBFRFIgU3ByaW50IDIgdmlydHVhbCBldmVudCBhbmQgdGhlIGFzc29jaWF0ZWQgcHJlLWV2ZW50IFdlYmluYXIgaXMgW2hlcmVdKGh0dHBzOi8vcG9ydGFsLm9nYy5vcmcvcHVibGljX29nYy9yZWdpc3Rlci8yMDIwMTFxNF9hcGlfZWRyLnBocCkuCgpUaGUgY29kZSBzcHJpbnQgd2lsbCBmb2N1cyBvbiByZWZpbmluZyB0aGUgW09HQyBBUEkgLSBFbnZpcm9ubWVudGFsIERhdGEgUmV0cmlldmFsIGNhbmRpZGF0ZSBzdGFuZGFyZF0oaHR0cDovL2RvY3Mub3Blbmdlb3NwYXRpYWwub3JnL0RSQUZUUy8xOS0wODYuaHRtbCkuIFRoZSBjYW5kaWRhdGUgc3RhbmRhcmQgdXNlcyBjdXJyZW50IHdlYiB0ZWNobm9sb2dpZXMgYW5kIGJlc3QgcHJhY3RpY2VzIHRvIGVuYWJsZSBlbmQtdXNlcnMgLSBvciBhbnlvbmUgd2l0aCB3ZWIgZGV2ZWxvcG1lbnQgZXhwZXJpZW5jZSAtIHRvIGVhc2lseSBpZGVudGlmeSBhbmQgcmV0cmlldmUgYSBzdWJzZXQgb2YgZGF0YSBmcm9tIOKAmGJpZyBkYXRh4oCZIHN0b3Jlcy4gVGhlIGlkZWEgaXMgdG8gc2F2ZSB0aG9zZSB1c2VycyBpbnRlcmVzdGVkIGluIGVudmlyb25tZW50YWwgKG9yIG90aGVyKSBkYXRhIGZyb20gaGF2aW5nIHRvIHRyYW5zZmVyIGFuZCBkZWFsIHdpdGggZGF0YXNldHMgdGhhdCBpbmV2aXRhYmx5IGNvbnRhaW4gZGF0YSBjb25jZXJuaW5nIGFyZWFzIG9yIHRpbWUgcGVyaW9kcyB0aGF0IGFyZSBpcnJlbGV2YW50IHRvIHRoZWlyIGludGVyZXN0cy4KClJlZ2lzdGVyIGF0IGh0dHBzOi8vcG9ydGFsLm9nYy5vcmcvcHVibGljX29nYy9yZWdpc3Rlci8yMDIwMTFxNF9hcGlfZWRyLnBocAoKCiogW1NwcmludCBEZXNjcmlwdGlvbl0oLi9hYm91dC5hZG9jKQoqIFtBUEkgU3BlY3NdKC4vc3BlY3MuYWRvYykKKiBbU3ByaW50IExvZ2lzdGljc10oLi9sb2dpc3RpY3MuYWRvYykKKiBbU2NoZWR1bGUvQWdlbmRhXSguL2FnZW5kYS5hZG9jKQoqIFtJbXBsZW1lbnRhdGlvbnNdKC4vaW1wbGVtZW50YXRpb25zLmFkb2MpCiogW0RhdGFzZXRzXSguL1NoYXJlZF9EYXRhc2V0cy9SRUFETUUubWQpCiogW1doYXQgaXMgZXZlcnlib2R5IGdvaW5nIHRvIGJlIHdvcmtpbmcgb24/XShodHRwczovL2dpdGh1Yi5jb20vb3Blbmdlb3NwYXRpYWwvT0dDQVBJLUVEUi1TcHJpbnQyL2lzc3Vlcy8xKQoqIFtMZXNzb25zIGFuZCBOZXh0IFN0ZXBzXSguL2xlc3NvbnNBbmROZXh0U3RlcHMuYWRvYykKKiBbQWRkaXRpb25hbCBSZXNvdXJjZXNdKC4vYWRkaXRpb25hbFJlc291cmNlcy5hZG9jKQoqIFtGcmVxdWVudGx5IEFza2VkIFF1ZXN0aW9ucyAoRkFRcyldKC4vRkFRLmFkb2MpCgpUaGUgc3ByaW50IHdpbGwgYmVnaW4gYXQgMDk6MDBhbSBFU1Qgb24gdGhlIGZpcnN0IGRheSwgYW5kIGVuZCBhdCAwNTowMHBtIEVTVCBvbiB0aGUgc2Vjb25kIGRheS4K readmeEtag: '"64e778c9e1fe3b9f795690ded2e851a323fb02d4"' readmeLastModified: Fri, 13 Nov 2020 07:46:23 GMT repositoryId: 304570773 description: >- This Github repository is for the second OGC API - EDR code sprint focusing on the OGC API - Environmental Data Retrieval candidate standard. created: '2020-10-16T08:48:36Z' updated: '2023-01-28T11:44:43Z' language: R archived: true stars: 1 watchers: 20 forks: 6 owner: opengeospatial logo: https://avatars.githubusercontent.com/u/1955193?v=4 repoEtag: '"721e68a0a4bb8cc864ec48f034fbb63a8204ee1c47114dac3ec649a2cba36e2c"' repoLastModified: Sat, 28 Jan 2023 11:44:43 GMT foundInMaster: true category: Server Implementations id: a15d24387a553cf2bb6cd4b6a3953574 - source: openapi3 tags repository: https://github.com/shopiqo/bigbuy-oas v3: true id: 7c4655856582db9c2b38a62ae4e156b1 repositoryMetadata: base64Readme: >- IyBCaWdCdXkgQVBJIHNwZWNpZmljYXRpb25zCgojIyMgR2VuZXJhdGUvcmVidWlsZCBzcGVjaWZpY2F0aW9ucwoKYGBgCi4vYmluL2J1aWxkCmBgYAoKIyMjIFN3YWdnZXIgVUkKCmBgYApkb2NrZXIgcnVuIFwKICAtLXJtIFwKICAtcCA4MDgwOjgwODAgXAogIC12IGBwd2RgOi91c3Ivc2hhcmUvbmdpbngvaHRtbC9wdWJsaWMgXAogIC1lIEFQSV9VUkw9cHVibGljL2Rpc3QvdjEvb3BlbmFwaS55YW1sIFwKICAtZSBWQUxJREFUT1JfVVJMPW5vbmUgXAogIC1lIFRSWV9JVF9PVVRfRU5BQkxFRD10cnVlIFwKICAtZSBESVNQTEFZX1JFUVVFU1RfRFVSQVRJT049dHJ1ZSBcCiAgLWUgRElTUExBWV9PUEVSQVRJT05fSUQ9dHJ1ZSBcCiAgc3dhZ2dlcmFwaS9zd2FnZ2VyLXVpCmBgYAoKIyMjIEdlbmVyYXRlIEFQSSBjbGllbnQKCmBgYApkb2NrZXIgcnVuIC0tcm0gLXYgJFBXRDovYXBwIG9wZW5hcGl0b29scy9vcGVuYXBpLWdlbmVyYXRvci1jbGkgXAogICAgZ2VuZXJhdGUgXAogICAgLWkgL2FwcC9kaXN0L3YxL29wZW5hcGkueWFtbCBcCiAgICAtZyBwaHAgXAogICAgLW8gL2FwcC9jbGllbnRzL3YxL3BocCBcCiAgICAtLXNraXAtdmFsaWRhdGUtc3BlYyBcCiAgICAtLWdpdC11c2VyLWlkIGVmc2EtaW8gXAogICAgLS1naXQtcmVwby1pZCBiaWdidXktcGhwLWFwaS1jbGllbnQgXAogICAgLS1hZGRpdGlvbmFsLXByb3BlcnRpZXMgcGFja2FnZU5hbWU9QmlnYnV5QXBpCmBgYAo= readmeEtag: '"d2ae1445e1a188e37ab6c922386409bad8a0f02f"' readmeLastModified: Tue, 01 Feb 2022 10:49:20 GMT repositoryId: 338562018 description: OpenAPI specification for https://api.bigbuy.eu/. created: '2021-02-13T11:44:15Z' updated: '2023-02-05T14:42:22Z' language: Shell archived: false stars: 1 watchers: 0 forks: 2 owner: shopiqo logo: https://avatars.githubusercontent.com/u/79263571?v=4 repoEtag: '"8ce32e7fd4bb3e3e490a4b58479138ba8bf7358e9b87caf724bb05a2b8b4b74b"' repoLastModified: Sun, 05 Feb 2023 14:42:22 GMT category: - Code Generators - Parsers foundInMaster: true oldLocations: - https://github.com/efsa-io/bigbuy-oas - source: openapi3 tags repository: https://github.com/jonasnicoletti/spring-boot-demo-app v3: true repositoryMetadata: repositoryId: 347713788 description: Springboot demo app with some enterprise standards created: '2021-03-14T18:03:51Z' updated: '2023-03-09T00:05:41Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: JonasNicoletti logo: https://avatars.githubusercontent.com/u/53706359?v=4 repoEtag: '"b0db25aea163a7566b7d0e506e5614523a01a48c845b455f8a7358e55804d793"' repoLastModified: Thu, 09 Mar 2023 00:05:41 GMT foundInMaster: true id: 7fe6f1fa80818846ca4357078438118f - source: openapi3 tags repository: https://github.com/xiasf/kancloud-api-parser v3: true id: 998c30cb82434374582d859a37ff7979 repositoryMetadata: base64Readme: >- IyMga2FuY2xvdWQtYXBpLXBhcnNlciDor7TmmI4KCuWwhiBb55yL5LqR5paH5qGjYXBpXShodHRwczovL2hlbHAua2FuY2xvdWQuY24vNjc1MzkpIOagvOW8j+ino+aekOWvvOWHuuS4uiBbT3BlbkFQSSAzLjFdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjEuMC5tZCkg5qC85byP55qEanNvbuaWh+S7tu+8jOaWueS+v+WvvOWFpeWIsCBbQXBpZm94XShodHRwczovL3d3dy5hcGlmb3guY24vKSDkuK3jgIIKCuWvueS6juS5i+WJjeS9v+eUqOS6huS4jeaUr+aMgeagh+WHhuWMlkFQSeaWh+aho+e7k+aehOWvvOWHuueahOaWh+aho+W5s+WPsOadpeivtO+8jOiHquW3seagueaNruinhOWImeino+aekOaYr+WUr+S4gOeahOWKnuazleS6hu+8jOWQpuWImemmluasoei/geenu+aIkOacrOWkqumrmOS6hu+8jOWPr+iDveWwseaUvuW8g+S6huOAggoK5pys6ISa5pys6Kej5p6Q5pWI5p6c5Y+W5Yaz5LqO5paH5qGj5Lmm5YaZ55qE6KeE6IyD56iL5bqm77yM5Zug5Li65piv6YCa6L+H5q2j5YiZ5Yy56YWN6Kej5p6Q55qE77yM5omA5Lul5Lmf5Y+v5Lul5qC55o2u6ZyA6KaB54G15rS755qE5Y675YGa6LCD5pW044CC55uu5YmN5oiR5Lus5L2/55So5rKh5LuA5LmI6Zeu6aKY77yM5aaC5p6c6Zeu6aKY5qyi6L+O5o+QIElzc3VlcyDkuqTmtYHjgIIKCi0tLS0KCiMjIyDkvb/nlKjmlrnms5UKCmBgYHBocApyZXF1aXJlICdEb2NBcGlQYXJzZXIucGhwJzsKCiRkb2NBcGlQYXJzZXIgPSBuZXcgRG9jQXBpUGFyc2VyKCk7CgovLyAuLi4KCmBgYAoKIyMjIyDoh6rlrprkuYnpu5jorqRBUEnliIbnu4TvvIjliY3nvIDvvIkKCuWIhue7hOWNsyBhcGlfcHJlZml4IO+8jOWGs+WumuWvvOWFpeWIsCBBcGlmb3gg5Lit55qE5pyA5LiK5bGC55uu5b2V5ZCN56ewKHgtYXBpZm94LWZvbGRlciDlrZfmrrUpCgpgYGBwaHAKCiRkb2NBcGlQYXJzZXItPmRlZmF1bHRBcGlQcmVmaXggPSAnYWRtaW4nOyAvLyDpu5jorqTkuLogZGVmYXVsdAoKYGBgCgojIyMjIOino+aekOW5tuWvvOWHuuS4uiBPcGVuQVBJIGpzb27moLzlvI/mlofku7YKCmBgYHBocAovLyAwLiDop6PmnpDnnIvkupHmlofmoaPnm67lvZXmlofku7YKJGZpbGVMaXN0ID0gJGRvY0FwaVBhcnNlci0+cGFyc2VEb2NTdW1tYXJ5KCd4eHgtZG9jL1NVTU1BUlkubWQnKTsKCi8vIDEuIOino+aekCDnnIvkupEg54m55pyJ55qEIEFQSSDmoLzlvI/vvIzovazmjaLkuLog5qCH5YeG55qEQVBJ5YWD5L+h5oGvIOaVsOe7hOagvOW8jwokYXBpcyA9ICRkb2NBcGlQYXJzZXItPnBhcnNlRG9jRmlsZSgncGFya2luZ2xvdC/nrqHnkIblkI7lj7Av5YGc6L2m5Zy6566h55CGL+Wll+mkkOeuoeeQhi5tZCcpOwovLyBwcmludF9yKCRhcGlzKTtleGl0OwoKLy8g5Z+65LqOIOagh+WHhueahEFQSeWFg+S/oeaBryDvvIzkvaDlj6/ku6XlrprkuYnlhbbku5bnmoTku7vkvZXovazmjaLlpITnkIblmagKCi8vIDIuIOWGjei9rOaNouS4uiBPcGVuQVBJIOagvOW8j+eahOaVsOe7hAokYXBpcyA9ICRkb2NBcGlQYXJzZXItPk9wZW5BUElGb3JtYXRIYW5kbGUoJGFwaXMpOwovLyBwcmludF9yKCRhcGlzKTtleGl0OwoKLy8gMy4g5YaZ5YWl5Yiw5paH5Lu2IO+8jCDpu5jorqTkuLoganNvbiDmoLzlvI/vvIzkvaDkuZ/lj6/ku6XlrprkuYkg5YW25LuWIOagvOW8j+i9rOaNouWkhOeQhiBmb3JtYXRIYW5kbGUg5bGe5oCn77yM5aaCIFhNTOetieagvOW8jwokZG9jQXBpUGFyc2VyLT53cml0ZUFwaSgna2FuY2xvdWQtYXBpLmpzb24nLCAkYXBpcyk7CgpgYGAKCi0tLS0KCiMjIyDnnIvkupHmlofmoaMgQVBJIOagh+WHhuagvOW8jwoKIVtpbWFnZV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vMTc1MzU3NTcvMTgyMDIyOTU3LWIwNmJmMzI3LTEwNWMtNGE4Yi05M2M1LTEzZmQ5YjFiMjkxOC5wbmcpCgpgYGB0ZXh0CiMjIyAxLiDliJvlu7rjgIHnvJbovpEg5paw6Ze7Cgrkvb/nlKjlnLrmma/vvJrlkI7lj7DnrqHnkIblkZgg5Yib5bu644CB57yW6L6RIOaWsOmXuwoKfn5+W2FwaTphZG1pbl0KcG9zdDovbmV3cy9lZGl0CippbnQ6aWQ9MSPlj4LmlbBpZApzdHJpbmc6bmFtZT3pu5jorqTlgLwj6K+05piO5paH5a2XCjw8PApzdWNjZXNzCnsKICAgICJkYXRhIjogewogICAgICAgICJpZCI6IDEsCiAgICB9LAogICAgIm1zZyI6ICJvayIsCiAgICAiY29kZSI6IDEKfQo8PDwKZXJyb3IKewogICAgImRhdGEiOiB7fSwKICAgICJtc2ciOiAiZXJyb3IgbXNnIiwKICAgICJjb2RlIjogMAp9Cjw8PArlk43lupTnpLrkvosK6L+Z6YeM5aGr5YaZ6ZSZ6K+v55qE6L+U5Zue56CBCuS7peatpOexu+aOqO+8jOavj+S4queKtuaAgeS9v+eUqCA8PDwg5YiG5YmyLArnrKzkuIDooYzmt7vliqDnirbmgIHlkI3np7AKfn5+CmBgYAoKaHR0cHM6Ly9oZWxwLmthbmNsb3VkLmNuLzY3NTM5Cg== readmeEtag: '"22864c57ae6042a7a6eb214d36bd153456414ee0"' readmeLastModified: Sun, 31 Jul 2022 11:25:00 GMT repositoryId: 519744241 description: 将看云文档api格式解析导出为 OpenAPI 3.0 格式的json文件,方便导入到 Apifox 中 created: '2022-07-31T10:28:21Z' updated: '2022-07-31T11:40:45Z' language: PHP archived: false stars: 1 watchers: 1 forks: 0 owner: xiasf logo: https://avatars.githubusercontent.com/u/17535757?v=4 license: MIT repoEtag: '"5bb7624870d7c5ba9801c063b93e3ca3320b04e81460cade693390cb90efe93d"' repoLastModified: Sun, 31 Jul 2022 11:40:45 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/alexandrecpedro/bbb22-votation-system v3: true id: e59a48c9affe1b53bfe48c6218fe7b41 repositoryMetadata: base64Readme: >- <div align = "center">
    <h1> Sistema de Votação Online do BBB 22 </h1>
</div>
<br>

<div align = 'center' justify-content = 'space-around' >
   <img src= './project/logos/Avanade.png' alt = 'Avanade' >
   <img src= './project/logos/RedeGlobo.png' alt = 'Rede Globo' >
</div>

<h3 align = "center">
  Sistema de Votação Online do Big Brother Brasil 22 (2022)
</h3>

<p align="center">
 <a href="#objetivo">Objetivo</a> •
 <a href="#tecnologias">Tecnologias</a> •
 <a href="#percurso">Percurso</a> 
</p>

<br>
<br>

<div id="objetivo">
<h2> 💡 Objetivo </h2>
Construir um sistema que permita ao usuário votar num participante indicado para eliminação (paredão) do reality show Big Brother Brasil 22 (BBB 22), por semana.

E conseguir contar os votos totais (inclusive os feitos por robôs) e auditar os votos válidos (votos únicos por usuário), configurando uma porcentagem de votos válidos por candidato à eliminação.

Cada semana devemos auditar e configurar os 3 candidatos à eliminação para que o usuário possa escolher em qual participante remanescente do programa ele deseja votar.

A cada semana, o participante que obtiver o maior percentual de votos é eliminado do programa.

</div>
<br>
<br>

<div id="tecnologias">
<h2> 🛠 Tecnologias </h2>
As seguintes ferramentas foram usadas na construção do projeto:<br><br>
    
|                                                     Tipo              | Ferramentas                                |                                 Referência                                   |
| :---------------------------------------------------------------------------: | :--------------------------------------------------------------------------: | :--------------------------------------------------------------------------: |
|                    Banco de Dados Não-Relacional       |   MONGODB     |              https://www.mongodb.com/try/download/community                     |
|                           Robo 3T                                  |   ROBO 3T      |             https://robomongo.org/                                         |
|      Framework Java Spring Boot - construção do Back-end       |    SPRING BOOT       |               https://start.spring.io/                                       |
|      API Documentation Library for Spring Boot       |    OPENAPI 3       |              https://springdoc.org/                                       | 
|                           Framework Apache KAFKA                     | KAFKA   |              https://kafka.apache.org/        |    
|                           View and managing objects a Apache KAFKA         |     OFFSET EXPLORER           |              https://kafkatool.com/         |    
|                           Docker                                    |    DOCKER          |               https://hub.docker.com/                                            |      
|              API RESTful using JSON - Swagger UI           |   SWAGGER     |               https://swagger.io/                                            |    
|              Framework Angular - construção da API web           |  ANGULAR       |              https://angular.io/                                            |    
<br>
<div align = 'center'>
    <h3>Backend</h3>
    <img align='center' width =' 80px ' src="https://cdn.jsdelivr.net/gh/devicons/devicon/icons/mongodb/mongodb-original.svg" />
    <img align='center' width =' 80px ' src="./project/logos/robomongo-3T.png" />
    <img align='center' width =' 80px ' src="https://cdn.jsdelivr.net/gh/devicons/devicon/icons/spring/spring-original.svg" />
    <img align='center' width =' 250px ' src="./project/logos/openAPI3.png" />
    <img align='center' width =' 250px ' src="./project/logos/swagger_logo.svg" />
    <br><br>
    <h3>Infraestrutura</h3>
    <img align='center' width =' 80px ' src="https://cdn.jsdelivr.net/gh/devicons/devicon/icons/apachekafka/apachekafka-original.svg" />
    <img align='center' width =' 80px ' src="https://static.macupdate.com/products/53532/m/offset-explorer-logo.png?v=1633349540" />
    <img align='center' width =' 100px ' src="https://cdn.jsdelivr.net/gh/devicons/devicon/icons/docker/docker-original.svg" />
    <br><br>
    <h3>Frontend</h3>
    <img align='center' width =' 80px ' src="https://cdn.jsdelivr.net/gh/devicons/devicon/icons/angularjs/angularjs-original.svg" />
    <br>
</div>
<br>

<div id="percurso">
<h2> 🔎 Percurso </h2>

<ol>
    <li>Parte 1 - Configurando/preparando ambiente (Infraestrutura)
      <ul>
        <li>Instalar Apache Kafka, Robo 3T, IntelliJ, VS Code, Docker Desktop</li>
        <li>Abrir e logar o Docker Desktop</li>
        <li>Abrir o VS Code</li>
        <li>Configurar uma pasta Docker, com as subpastas Kafka e MongoDB</li>
        <li>Criar os arquivos de configuração (docker-compose.yml) em cada subpasta criada</li>
        <!-- Comando docker-compose up -d (o -d é para subir o servidor em 2º plano) -->
        <li>Entrar em cada subpasta criada e subir o serviço para o Docker (docker-compose up -d)</li>
      </ul>
    </li>
    <br>
    <li>Parte 2 - Modelando o projeto no DrawIO</li>
    <br>
    <li>Parte 3 - Criando back-end com Java Spring Boot
      <ul>
        <li>Abrir o Spring Initializr e configurar o Spring Boot</li>
        <li>Gerar o arquivo Spring Boot e abrir no IntelliJ</li>
        <li>Verificar as dependências a serem usadas e configurar o Maven</li>
        <li>Fazer arquivo de configuração de acesso ao MongoDB e ao servidor Apache Kafka</li>
        <li>Povoar mecanicamente o MongoDB com alguns participantes do BBB, incluindo o id (arquivo JSON)</li>
        <li>Construir a API em Java Spring Boot - classes, objetos, datas e interfaces</li>
        <li>Criar a estrutura do microsserviço em Java Spring Boot - classes, objetos, datas e interfaces</li>
        <li>Importar o microsserviço na mesma pasta da API, no IntelliJ</li>
        <li>Construir as classes, datas, objetos e interfaces para o microsserviço</li>
        <li>Buildar os arquivos principais da API e do microsserviço</li>
        <li>Testar as rotas no Postman
        Rota POST 
        URL: http://localhost:8080/api/parametros/salvar
        Aba BODY:
        JSON =
        {
          "chave": "config.votacao",
          "valor": "3"
        }
        </li>
      </ul>
    </li>
    <br>
    <li>Parte 4 - Criando API web com Angular
      <ul>
        <li>Novo projeto: ng new nomeProjeto </li>
        <li>Iniciando projeto: ng server --open </li>
        <li>Novo componente: ng generate component nomeComponente </li>
        <li>Criar interface: src/app/model/participante.ts</li>
        <li>Criando rota: ng generate module app-routing --flat --module=app</li>
        <li>Importar HttpClientModule em app.module.ts [parte de imports]</li>
        <li>Criar serviço: src/app/service/RestService.ts</li>
        <li>Add @CrossOrigin em ParticipanteController.java e VotacaoController.java</li>
        <li>Implementar as rotas</li>
      </ul>
    </li>
    <br>
    <li>Parte 5 - Integrando back-end/front-end e acessando dados</li>  
</ol>
</div>
<br>
<br>
 readmeEtag: '"af7be61366a762908a1aea98766dd02d3d9d82c8"' readmeLastModified: Fri, 16 Sep 2022 15:01:33 GMT repositoryId: 485125297 description: Online Votation System of Big Brother Brazil 22 (2022) created: '2022-04-24T19:51:31Z' updated: '2023-03-04T01:45:55Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: alexandrecpedro logo: https://avatars.githubusercontent.com/u/80993336?v=4 repoEtag: '"1dd8cf3d7ed56c6605727b2de356e3fc2089e58c815bce472c6982d525356c5b"' repoLastModified: Sat, 04 Mar 2023 01:45:55 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/dongyaoo/kes v3: true id: b0830cc10323f7974be6d41b91361c83 repositoryMetadata: base64Readme: >- IyDnn6Xor4bmir3lj5bns7vnu58KCuacrOezu+e7n+WfuuS6jltBbnQgRGVzaWduIFByb10oaHR0cHM6Ly9wcm8uYW50LmRlc2lnbikuIOS7peS4i+Wwj+iKguWwhuS7i+e7jeWmguS9leS9v+eUqOacrOezu+e7ny4KCiMjIOeOr+Wig+mFjee9rgoK6ZSu5YWl5Lul5LiL5ZG95Luk5a6J6KOF56ys5LiJ5pa55L6d6LWW5bqTIGBub2RlX21vZHVsZXNgOgoKYGBgYmFzaApucG0gaW5zdGFsbApgYGAKCuaIluiAhQoKYGBgYmFzaAp5YXJuCmBgYAoKIyMg6aG555uu6ISa5pys5ZG95LukCgrmnKzpobnnm67ln7rkuo4gQW50IERlc2lnbiBQcm/vvIzlm6DmraTlpI3nlKjkuoYgQW50ZFBybyDohJrmnKzlkb3ku6TjgILor6bmg4Xlj6/mn6XnnIsgYHBhY2thZ2UuanNvbmDjgIIKCiMjIyDlkK/liqjns7vnu58KCmBgYGJhc2gKbnBtIHN0YXJ0CmBgYAoKIyMjIOaehOW7uumhueebrgoKYGBgYmFzaApucG0gcnVuIGJ1aWxkCmBgYAoKIyMjIOajgOafpeS7o+eggemjjuagvAoKYGBgYmFzaApucG0gcnVuIGxpbnQKYGBgCgrkvaDkuZ/lj6/ku6XplK7lhaXku6XkuIvlkb3ku6Toh6rliqjkv67lpI3kuIDkupvpo47moLzplJnor6/vvJoKCmBgYGJhc2gKbnBtIHJ1biBsaW50OmZpeApgYGAKCiMjIyDmtYvor5UKCmBgYGJhc2gKbnBtIHRlc3QKYGBgCgojIyDmm7TlpJoKCuabtOWkmiBBUEkg55So5rOV5Y+v5Y+C6ICDIEFudGRQcm8g55qEW+WumOaWueaWh+aho10oaHR0cHM6Ly9wcm8uYW50LmRlc2lnbinjgIIK readmeEtag: '"468ec960c70bb5c424c32c572851399459953f2b"' readmeLastModified: Wed, 04 May 2022 01:08:15 GMT repositoryId: 488405193 description: 知识抽取系统 created: '2022-05-04T00:35:09Z' updated: '2022-05-04T05:55:52Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: dongyaoo logo: https://avatars.githubusercontent.com/u/62202550?v=4 repoEtag: '"55e6bffb3b205dd4fc80af5d0e14f6c840b7be1506e305b3cc451e28e8c3188e"' repoLastModified: Wed, 04 May 2022 05:55:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mmuzammil196/gradadmit-ann-predictor v3: true id: a33ba8d19e2c210216e331c8c7697aff repositoryMetadata: base64Readme: >- IVtNTklTVCBIYW5kd3JpdHRlbiBEaWdpdCBSZWNvZ25pdGlvbiB1c2luZyBBTk5dKGh0dHBzOi8vZ2l0aHViLmNvbS9tbXV6YW1taWwxOTYvR3JhZEFkbWl0LUFOTi1QcmVkaWN0b3IvYXNzZXRzLzc3Mzg5MDQwLzIxMTA5ZmM4LWQ4ODktNDg1MS05YWNlLTM2OGRhZGQwM2QzYykKCiJVbmxvY2sgeW91ciBmdXR1cmUgd2l0aCBHcmFkQWRtaXQtQU5OLVByZWRpY3RvciEg8J+agPCfjq8iCgpUaGlzIHByb2plY3QsIEdyYWRBZG1pdC1BTk4tUHJlZGljdG9yLCBmb2N1c2VzIG9uIHByZWRpY3RpbmcgdGhlIGxpa2VsaWhvb2Qgb2YgYWRtaXNzaW9uIHRvIGEgZ3JhZHVhdGUgcHJvZ3JhbSB1c2luZyBhbiBBcnRpZmljaWFsIE5ldXJhbCBOZXR3b3JrIChBTk4pIGFsZ29yaXRobS4gQnkgdHJhaW5pbmcgdGhlIG1vZGVsIG9uIGEgZGF0YXNldCBjb250YWluaW5nIHZhcmlvdXMgcGFyYW1ldGVycyBzdWNoIGFzIEdSRSBzY29yZXMsIFRPRUZMIHNjb3JlcywgdW5pdmVyc2l0eSByYXRpbmdzLCBhbmQgbGV0dGVyIG9mIHJlY29tbWVuZGF0aW9uIHN0cmVuZ3Rocywgd2Ugd2VyZSBhYmxlIHRvIGFjaGlldmUgYW4gYWNjdXJhY3kgb2YgODAlIGluIHByZWRpY3RpbmcgYWRtaXNzaW9ucyBvdXRjb21lcy4K readmeEtag: '"998599b81d809ec08f7a387054d7b1f18fee4487"' readmeLastModified: Mon, 05 Jun 2023 19:18:00 GMT repositoryId: 649848514 description: >- This project, GradAdmit-ANN-Predictor, focuses on predicting the likelihood of admission to a graduate program using an Artificial Neural Network (ANN) algorithm. created: '2023-06-05T19:15:46Z' updated: '2023-06-06T16:33:48Z' language: Jupyter Notebook archived: false stars: 1 watchers: 1 forks: 0 owner: mmuzammil196 logo: https://avatars.githubusercontent.com/u/77389040?v=4 repoEtag: '"75d53639048bc23a35d49d41edbe1cf3264ed669c7ea96dbf6975f5cb40272f7"' repoLastModified: Tue, 06 Jun 2023 16:33:48 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/abhishekb91/petstore-openapi3 v3: true repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgogICAgUGV0c3RvcmUtT3BlbmFwaTMKPC9oMT4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxzdHJvbmc+Q3JlYXRpbmcgSFRUUCBBUEkgdXNpbmcgZ29sYW5nIGluIG9wZW5hcGkzIHN0YW5kYXJkPC9zdHJvbmc+PGJyPgo8L3A+CgojIyBDb250ZW50cwotIFtHZXR0aW5nIHN0YXJ0ZWRdKCNnZXR0aW5nLXN0YXJ0ZWQpCi0gW0RldmVsb3BtZW50XSgjZGV2ZWxvcG1lbnQpCgojIyBHZXR0aW5nIHN0YXJ0ZWQKVGhlIGludGVudCBvZiB0aGlzIHNhbXBsZSBwcm9qZWN0IGlzIHRvIGRlbW9uc3RyYXRlIHRoZSB1c2FnZSBvZiB2YXJpb3VzIHRvb2xzIHRvIGNyZWF0ZSB3ZWJhcGlzIHVzaW5nIG9wZW5hcGkgMyBzdGFuZGFyZC4gVGhpcyBwcm9qZWN0IHVzZXM6Ci0gW29hcGktY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL2RlZXBtYXAvb2FwaS1jb2RlZ2VuKSB0byBnZW5lcmF0ZSBzZXJ2ZXIgc3R1YnMKLSBbc3dhZ2dlci11aV0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkpIHRvIGRpc3BsYXkgdGhlIGFwaSBkb2N1bWVudAotIFtlY2hvXShodHRwczovL2dpdGh1Yi5jb20vbGFic3RhY2svZWNobykgYXMgSFRUUCByb3V0aW5nIGVuZ2luZQotIFtzdGF0aWtdKGh0dHBzOi8vZ2l0aHViLmNvbS9yYWt5bGwvc3RhdGlrKSB0byBjb21waWxlIHN0YXRpYyBmaWxlcyBpbnRvIGEgR28gYmluYXJ5Ci0gW21vY2tlcnldKGh0dHBzOi8vZ2l0aHViLmNvbS92ZWt0cmEvbW9ja2VyeSkgdG8gZ2VuZXJhdGUgbW9ja3MgZnJvbSBpbnRlcmZhY2VzCgojIyMgUHJlcmVxdWlzaXRlCi0gbG9jYWwgZ29sYW5nIGluc3RhbGxhdGlvbgotIE15U1FMIERhdGFiYXNlLiBVc2UgTWFrZWZpbGUgdG8gZ2VuZXJhdGUgTXlTUUwgY29udGFpbmVyIGxvY2FsbHkuCgojIyBEZXZlbG9wbWVudAoKLSBTdGFydGluZyBBcHBsaWNhdGlvbgoKICBUaGUgc2VydmljZSBleHBlY3RzIGRhdGFiYXNlIGNvbm5lY3Rpb24gdG8gYmUgcGFzc2VkIGFzIGFuIGVudmlyb25tZW50IHZhcmlhYmxlLiBJbiBvcmRlciB0byBtYWtlIGxvY2FsIGRldiBlYXN5LCB0aGVyZSBhcmUgdHdvIG9wdGlvbnMgZnJvbSB3aGljaCBvbmUgY2FuIHJ1biB0aGUgc2VydmljZToKCiAgLSBGcm9tIFZpc3VhbCBTdHVkaW8gQ29kZToKICAgIEluIG9yZGVyIHRvIGRlYnVnIHRoZSBzZXJ2aWNlLCBvbmUgY2FuIHJ1biB0aGUgc2VydmljZSBkaXJlY3RseSBmcm9tIHRoZSBWaXN1YWwgU3R1ZGlvIENvZGUgYERlYnVnIGFuZCBSdW5gID4gYExhdW5jaGAgYW5kIGFkZCBicmVha3BvaW50cwoKICAtIFVzaW5nIE1ha2VmaWxlOgogICAgRnJvbSB0aGUgcHJvamVjdCBSb290OgoKICAgIGBgYGJhc2gKICAgIG1ha2Ugc3RhcnRfc2VydmljZQogICAgYGBgCgotIEFQSSBFbmQtcG9pbnRzCgogIGBgYAogIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9kb2NzLwogIGBgYAoKLSBSdW5uaW5nIFVuaXQgVGVzdHMKICBgYGBiYXNoCiAgZ28gdGVzdCAuLy4uLgogIGBgYA== readmeEtag: '"154d584327b09797c02d12c881e3a7b06acb43b3"' readmeLastModified: Sun, 23 May 2021 03:55:43 GMT repositoryId: 354568182 description: Creating HTTP API using golang in openapi3 standard created: '2021-04-04T14:43:43Z' updated: '2024-02-28T14:37:13Z' language: Go archived: false stars: 1 watchers: 1 forks: 1 owner: abhishekb91 logo: https://avatars.githubusercontent.com/u/11881672?v=4 license: MIT repoEtag: '"3a855b858c659bdf0d960524b9e73dd8a19ee7961454aebf1d5ef5ea0dd73d4f"' repoLastModified: Wed, 28 Feb 2024 14:37:13 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 6cf4808a04d7c970f29d5ef9209dff6f - source: openapi3 tags repository: https://github.com/godpeny/goserv v3: true repositoryMetadata: base64Readme: >- IyBHT1NFUlYKc2VydmVyIHRlbXBsYXRlIGluIEdPCgojIyBSdW4gQVBJIFNlcnZlcgpnbyBydW4gY21kL2dvc2Vydi1hcGkvbWFpbi5nbwoKIyMgUnVuIERCIFNlcnZlcgpnbyBydW4gY21kL2dvc2Vydi1kYi9tYWluLmdvCgojIyBSdW4gUmFiYml0IE1RIHdpdGggRG9ja2VyCmBgYApkb2NrZXIgcnVuIC0tcm0gLWQgLS1uYW1lIHJhYmJpdG1xIC1wIDU2NzI6NTY3MiAtcCAxNTY3MjoxNTY3MiAgLWUgUkFCQklUTVFfREVGQVVMVF9VU0VSPWFkbWluIC1lIFJBQkJJVE1RX0RFRkFVTFRfUEFTUz1hZG1pbiByYWJiaXRtcTozLW1hbmFnZW1lbnQKYGBgCgojIyBHZW5lcmF0aW5nIENvZGUKdXNpbmcgb3BlbmFwaS1nZW5lcmF0b3IgNS4wLjAtYmV0YTMKCiMjIyBBUEkgU2VydmVyCmBgYApqYXZhIC1qYXIgLi9hcGkvb3BlbmFwaS1nZW5lcmF0b3ItY2xpLmphciBnZW5lcmF0ZSAtaSAuL2FwaS9zd2FnZ2VyLXNlcnZlci55YW1sIC1nIGdvLWdpbi1zZXJ2ZXIgLW8gLi9pbnRlcm5hbC9nb3NlcnYtYXBpIC0tcGFja2FnZS1uYW1lIGFwaQpgYGAKCiMjIyBBUEkgQ2xpZW50IChzYW1wbGUpCmBgYApqYXZhIC1qYXIgLi9hcGkvb3BlbmFwaS1nZW5lcmF0b3ItY2xpLmphciBnZW5lcmF0ZSAtaSAuL2FwaS9zd2FnZ2VyLWNsaWVudC55YW1sIC1nIGdvIC1vIC4vaW50ZXJuYWwvY2xpZW50cy9zYW1wbGUgLS1wYWNrYWdlLW5hbWUgY2xpZW50CmBgYApVc2UgYGBhcGkvY29kZ2VuLXNlcnZlci5nb2BgIHRvIGNyZWF0ZSBgYHN3YWdnZXItc2VydmVyLnlhbWxgYAoKIyMjIERCIFNjaGVtYSBmb3IgZW50CmBgYApnbyBydW4gZ2l0aHViLmNvbS9mYWNlYm9vay9lbnQvY21kL2VudCBpbml0ICR7eW91ci1uZXctc2NoZW1hfSAKYGBgCgplZGl0IGBgPHlvdXItbmV3LXNjaGVtYT5gYC4gaW5pdGlhbGx5IGBgVXNlcmBgLGBgUHJvamVjdGBgIFNjaGVtYSBpcyBzZXQgOikKCmBgYApnbyBnZW5lcmF0ZSAuL2VudApgYGAK readmeEtag: '"c41cfac12eda351a04820649724035488e53fc5a"' readmeLastModified: Thu, 28 Jan 2021 13:40:14 GMT repositoryId: 318779942 description: Go Server Template using Rabbit MQ in Micro Service Architecture created: '2020-12-05T12:14:32Z' updated: '2024-09-09T14:48:25Z' language: Go archived: false stars: 1 watchers: 1 forks: 0 owner: godpeny logo: https://avatars.githubusercontent.com/u/37652215?v=4 repoEtag: '"61299322825ac5dd34e2d1720fbd11c2e15ea2656108b12bee7c797d4531838a"' repoLastModified: Mon, 09 Sep 2024 14:48:25 GMT foundInMaster: true category: Code Generators id: 6ea98c8b537f8f13c141dd1ddc5eba2d - source: openapi3 tags repository: https://github.com/stephane-segning/rust-open-api-cli v3: true id: b2032686c1e369c2705519bad664f933 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpX2NsaQoKb3BlbmFwaV9jbGkgaXMgYSBSdXN0LWJhc2VkIGNvbW1hbmQtbGluZSB0b29sIGZvciBnZW5lcmF0aW5nIGNvZGUgZnJvbSBPcGVuQVBJIHNwZWNpZmljYXRpb25zLiBJdCBwcm92aWRlcyBhIHNpbXBsZSB3YXkgdG8gZG93bmxvYWQgdGhlIE9wZW5BUEkgZ2VuZXJhdG9yIGFuZCB0aGVuIGdlbmVyYXRlIGNvZGUgaW4gYSBzcGVjaWZpZWQgbGFuZ3VhZ2UuCgojIyBGZWF0dXJlcwotIERvd25sb2FkcyB0aGUgT3BlbkFQSSBnZW5lcmF0b3IgQ0xJIGlmIGl0J3Mgbm90IGFscmVhZHkgcHJlc2VudAotIEdlbmVyYXRlcyBjb2RlIGluIGEgc3BlY2lmaWVkIGxhbmd1YWdlIGZyb20gYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZpbGUKLSBQcm92aWRlcyBhIGNvbW1hbmQtbGluZSBpbnRlcmZhY2UgZm9yIGVhc3kgdXNhZ2UKCiMjIFByZXJlcXVpc2l0ZXMKLSBSdXN0IHByb2dyYW1taW5nIGxhbmd1YWdlIChZb3UgY2FuIGluc3RhbGwgUnVzdCBmcm9tIFtoZXJlXShodHRwczovL3J1c3R1cC5ycy8pKQotIEphdmEgRGV2ZWxvcG1lbnQgS2l0IChKREspCgojIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKJCBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL3N0ZXBoYW5lLXNlZ25pbmcvcnVzdC1vcGVuLWFwaS1jbGkuZ2l0CiQgY2QgcnVzdC1vcGVuLWFwaS1jbGkKJCBjYXJnbyBidWlsZCAtLXJlbGVhc2UKYGBgCgpUaGUgZXhlY3V0YWJsZSB3aWxsIGJlIGxvY2F0ZWQgaW4gdGhlICoqdGFyZ2V0L3JlbGVhc2UqKiBkaXJlY3RvcnkuCgojIyBVc2FnZQoKVG8gdXNlIHRoZSBhcHBsaWNhdGlvbiwgcnVuIHRoZSBjb21tYW5kIGJlbG93IGZyb20gdGhlIGNvbW1hbmQgbGluZToKCmBgYGJhc2gKJCAuL3RhcmdldC9yZWxlYXNlL29wZW5hcGlfY2xpIGdlbmVyYXRlIC12IDx2ZXJzaW9uPiAtaSA8aW5wdXQ+IC1vIDxvdXRwdXQ+IC1sIDxsYW5ndWFnZT4KYGBgCgpSZXBsYWNlIDx2ZXJzaW9uPiB3aXRoIHRoZSB2ZXJzaW9uIG9mIE9wZW5BUEkgZ2VuZXJhdG9yIENMSSwgYDxpbnB1dD5gIHdpdGggdGhlIHBhdGggdG8geW91ciBPcGVuQVBJIHNwZWNpZmljYXRpb24gZmlsZSwgPG91dHB1dD4gd2l0aCB0aGUgcGF0aCB0byB0aGUgZGlyZWN0b3J5IHdoZXJlIHRoZSBnZW5lcmF0ZWQgY29kZSBzaG91bGQgYmUgc2F2ZWQsIGFuZCA8bGFuZ3VhZ2U+IHdpdGggdGhlIHRhcmdldCBsYW5ndWFnZSBmb3IgdGhlIGdlbmVyYXRlZCBjb2RlLgoKIyMgQ29udHJpYnV0aW5nCgpQdWxsIHJlcXVlc3RzIGFyZSB3ZWxjb21lLiBGb3IgbWFqb3IgY2hhbmdlcywgcGxlYXNlIG9wZW4gYW4gaXNzdWUgZmlyc3QgdG8gZGlzY3VzcyB3aGF0IHlvdSB3b3VsZCBsaWtlIHRvIGNoYW5nZS4KClBsZWFzZSBtYWtlIHN1cmUgdG8gdXBkYXRlIHRlc3RzIGFzIGFwcHJvcHJpYXRlLgoKIyMgTGljZW5zZQoKW01JVF0oaHR0cHM6Ly9jaG9vc2VhbGljZW5zZS5jb20vbGljZW5zZXMvbWl0Lyk= readmeEtag: '"dd6bc2d38426becedbea93511f0c9c5b9366d6fc"' readmeLastModified: Sat, 22 Jul 2023 22:07:26 GMT repositoryId: 669529638 description: >- openapi_cli is a Rust-based command-line tool for generating code from OpenAPI specifications created: '2023-07-22T15:19:39Z' updated: '2025-06-03T09:55:57Z' language: Rust archived: false stars: 1 watchers: 1 forks: 1 owner: stephane-segning logo: https://avatars.githubusercontent.com/u/26783557?v=4 repoEtag: '"84d3dfd14b0617ddf75f4dfa4bc171077741b4317cea529e3587560689434132"' repoLastModified: Tue, 03 Jun 2025 09:55:57 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/refactorian/openapi-swagger-docker v3: true id: b94b2f4c4f7b8e91b61770c90e60f505 repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+PGltZyBzcmM9Imh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy93cC1jb250ZW50L3VwbG9hZHMvc2l0ZXMvMy8yMDE2LzEwL09wZW5BUElfUGFudG9uZS5wbmciIHdpZHRoPSI1MDAiPjwvcD4KCiMgT3BlbkFQSSwgU3dhZ2dlciwgRG9ja2VyCi0gT3BlbkFQSSB2My54Ci0gU3dhZ2dlciBVSSB2NS54Ci0gU3dhZ2dlciBFZGl0b3IgdjQueAotIFJlZG9jIHYyLngKCiMgUmVxdWlyZW1lbnRzCi0gU3RhYmxlIHZlcnNpb24gb2YgW0RvY2tlcl0oaHR0cHM6Ly9kb2NzLmRvY2tlci5jb20vZW5naW5lL2luc3RhbGwvKQotIENvbXBhdGlibGUgdmVyc2lvbiBvZiBbRG9ja2VyIENvbXBvc2VdKGh0dHBzOi8vZG9jcy5kb2NrZXIuY29tL2NvbXBvc2UvaW5zdGFsbC8jaW5zdGFsbC1jb21wb3NlKQoKIyBIb3cgVG8gRGVwbG95Ci0gYGRvY2tlciBjb21wb3NlIHVwIC1kYAoKIyBOb3RlcwoKIyMjIFN3YWdnZXIgVUkgKERvY2tlcikKLSBVUkw6IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MAoKIyMjIEdpdEh1YiBQYWdlcwotIFN3YWdnZXIgVUk6IGh0dHBzOi8vcmVmYWN0b3JpYW4uZ2l0aHViLmlvL29wZW5hcGktc3dhZ2dlci1kb2NrZXIKLSBTd2FnZ2VyIEVkaXRvcjogaHR0cHM6Ly9yZWZhY3Rvcmlhbi5naXRodWIuaW8vb3BlbmFwaS1zd2FnZ2VyLWRvY2tlci9lZGl0b3IKLSBSZWRvYzogaHR0cHM6Ly9yZWZhY3Rvcmlhbi5naXRodWIuaW8vb3BlbmFwaS1zd2FnZ2VyLWRvY2tlci9yZWRvYwoKIyMjIERvY2tlciBjb21wb3NlIGNvbW1hbmRzCi0gQnVpbGQgb3IgcmVidWlsZCBzZXJ2aWNlcwogICAgLSBgZG9ja2VyIGNvbXBvc2UgYnVpbGRgCi0gQ3JlYXRlIGFuZCBzdGFydCBjb250YWluZXJzCiAgICAtIGBkb2NrZXIgY29tcG9zZSB1cCAtZGAKLSBTdG9wIGFuZCByZW1vdmUgY29udGFpbmVycywgbmV0d29ya3MKICAgIC0gYGRvY2tlciBjb21wb3NlIGRvd25gCi0gU3RvcCBhbGwgc2VydmljZXMKICAgIC0gYGRvY2tlciBjb21wb3NlIHN0b3BgCi0gUmVzdGFydCBzZXJ2aWNlIGNvbnRhaW5lcnMKICAgIC0gYGRvY2tlciBjb21wb3NlIHJlc3RhcnRgCi0gUnVuIGEgY29tbWFuZCBpbnNpZGUgYSBjb250YWluZXIKICAgIC0gYGRvY2tlciBjb21wb3NlIGV4ZWMgW2NvbnRhaW5lcl0gW2NvbW1hbmRdYAoKCiMgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIEd1aWRlCgpUaGUgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIChPQVMpIGlzIGEgc3RhbmRhcmQsIGxhbmd1YWdlLW5ldXRyYWwgd2F5IHRvIGRlZmluZSB5b3VyIEFQSS4gSXQgaGVscHMgaHVtYW5zIGFuZCBjb21wdXRlcnMgZGlzY292ZXIgYW5kIHVuZGVyc3RhbmQgd2hhdCB5b3VyIHNlcnZpY2UgY2FuIGRvLCBhbGwgd2l0aG91dCBuZWVkaW5nIHRvIGxvb2sgYXQgdGhlIHNvdXJjZSBjb2RlIG9yIGRvY3VtZW50YXRpb24gb3IgaW5zcGVjdCBuZXR3b3JrIHRyYWZmaWMuIEFuIE9wZW5BUEkgZG9jdW1lbnQgaXMgYSBZQU1MIG9yIEpTT04gZmlsZSB0aGF0IG91dGxpbmVzIHRoZSBBUEnigJlzIGVuZHBvaW50cywgb3BlcmF0aW9ucywgaW5wdXQgYW5kIG91dHB1dCBwYXJhbWV0ZXJzLCByZXF1ZXN0IGFuZCByZXNwb25zZSBmb3JtYXRzLCBhdXRoZW50aWNhdGlvbiBtZXRob2RzLCBhbmQgbW9yZS4KCiMjIyBCYXNpYyBTdHJ1Y3R1cmUKCjEuICoqT3BlbkFQSSBWZXJzaW9uKioKICAgYGBgeWFtbAogICBvcGVuYXBpOiAzLjAuMAogICBgYGAKCjIuICoqSW5mbyBPYmplY3QqKgogICBgYGB5YW1sCiAgIGluZm86CiAgICAgdGl0bGU6IFNhbXBsZSBBUEkKICAgICBkZXNjcmlwdGlvbjogQVBJIGRlc2NyaXB0aW9uIGluIE1hcmtkb3duLgogICAgIHZlcnNpb246IDEuMC4wCiAgIGBgYAoKMy4gKipTZXJ2ZXJzIE9iamVjdCoqCiAgIGBgYHlhbWwKICAgc2VydmVyczoKICAgICAtIHVybDogaHR0cHM6Ly9hcGkuZXhhbXBsZS5jb20vdjEKICAgICAgIGRlc2NyaXB0aW9uOiBQcm9kdWN0aW9uIHNlcnZlcgogICBgYGAKCjQuICoqUGF0aHMgT2JqZWN0KioKICAgYGBgeWFtbAogICBwYXRoczoKICAgICAvdXNlcnM6CiAgICAgICBnZXQ6CiAgICAgICAgIHN1bW1hcnk6IFJldHVybnMgYSBsaXN0IG9mIHVzZXJzLgogICAgICAgICByZXNwb25zZXM6CiAgICAgICAgICAgJzIwMCc6CiAgICAgICAgICAgICBkZXNjcmlwdGlvbjogQSBKU09OIGFycmF5IG9mIHVzZXIgbmFtZXMKICAgICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgICAgICAgICAgc2NoZW1hOiAKICAgICAgICAgICAgICAgICAgIHR5cGU6IGFycmF5CiAgICAgICAgICAgICAgICAgICBpdGVtczoKICAgICAgICAgICAgICAgICAgICAgdHlwZTogc3RyaW5nCiAgIGBgYAoKIyMjIENvbXBvbmVudHMKCjEuICoqU2NoZW1hcyoqCiAgIGBgYHlhbWwKICAgY29tcG9uZW50czoKICAgICBzY2hlbWFzOgogICAgICAgVXNlcjoKICAgICAgICAgdHlwZTogb2JqZWN0CiAgICAgICAgIHByb3BlcnRpZXM6CiAgICAgICAgICAgaWQ6CiAgICAgICAgICAgICB0eXBlOiBpbnRlZ2VyCiAgICAgICAgICAgbmFtZToKICAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICBgYGAKCjIuICoqUGFyYW1ldGVycyoqCiAgIGBgYHlhbWwKICAgY29tcG9uZW50czoKICAgICBwYXJhbWV0ZXJzOgogICAgICAgVXNlcklkOgogICAgICAgICBuYW1lOiB1c2VySWQKICAgICAgICAgaW46IHBhdGgKICAgICAgICAgcmVxdWlyZWQ6IHRydWUKICAgICAgICAgc2NoZW1hOgogICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgYGBgCgozLiAqKlJlc3BvbnNlcyoqCiAgIGBgYHlhbWwKICAgY29tcG9uZW50czoKICAgICByZXNwb25zZXM6CiAgICAgICBOb3RGb3VuZDoKICAgICAgICAgZGVzY3JpcHRpb246IEVudGl0eSBub3QgZm91bmQuCiAgIGBgYAoKNC4gKipTZWN1cml0eSBTY2hlbWVzKioKICAgYGBgeWFtbAogICBjb21wb25lbnRzOgogICAgIHNlY3VyaXR5U2NoZW1lczoKICAgICAgIEFwaUtleUF1dGg6CiAgICAgICAgIHR5cGU6IGFwaUtleQogICAgICAgICBpbjogaGVhZGVyCiAgICAgICAgIG5hbWU6IFgtQVBJLUtleQogICBgYGAKCiMjIyBLZXkgRWxlbWVudHMKCiMjIyBQYXRocwpEZWZpbmVzIHRoZSBhdmFpbGFibGUgcGF0aHMgYW5kIG9wZXJhdGlvbnMgZm9yIHRoZSBBUEkuCmBgYHlhbWwKcGF0aHM6CiAgL3BldHM6CiAgICBnZXQ6CiAgICAgIHN1bW1hcnk6IExpc3QgYWxsIHBldHMKICAgICAgb3BlcmF0aW9uSWQ6IGxpc3RQZXRzCiAgICAgIHJlc3BvbnNlczoKICAgICAgICAnMjAwJzoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBBIHBhZ2VkIGFycmF5IG9mIHBldHMKICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgIGFwcGxpY2F0aW9uL2pzb246ICAgIAogICAgICAgICAgICAgIHNjaGVtYTogCiAgICAgICAgICAgICAgICB0eXBlOiBhcnJheQogICAgICAgICAgICAgICAgaXRlbXM6IAogICAgICAgICAgICAgICAgICAkcmVmOiAnIy9jb21wb25lbnRzL3NjaGVtYXMvUGV0JwpgYGAKCiMjIyBQYXJhbWV0ZXJzCkRlZmluZXMgdGhlIGlucHV0cyB0byBhbiBBUEkgbWV0aG9kLCB3aGljaCBjb3VsZCBiZSBwYXRoLCBxdWVyeSwgaGVhZGVyLCBvciBjb29raWUgcGFyYW1ldGVycy4KYGBgeWFtbApwYXJhbWV0ZXJzOgogIC0gaW46IHF1ZXJ5CiAgICBuYW1lOiBsaW1pdAogICAgc2NoZW1hOgogICAgICB0eXBlOiBpbnRlZ2VyCiAgICBkZXNjcmlwdGlvbjogSG93IG1hbnkgaXRlbXMgdG8gcmV0dXJuIGF0IG9uZSB0aW1lIChtYXggMTAwKQpgYGAKCiMjIyBSZXNwb25zZXMKRGVmaW5lcyB0aGUgcG9zc2libGUgcmVzcG9uc2VzIGZyb20gYW4gQVBJIG1ldGhvZC4KYGBgeWFtbApyZXNwb25zZXM6CiAgJzIwMCc6CiAgICBkZXNjcmlwdGlvbjogQSBwYWdlZCBhcnJheSBvZiBwZXRzCiAgICBoZWFkZXJzOgogICAgICB4LW5leHQ6CiAgICAgICAgZGVzY3JpcHRpb246IEEgbGluayB0byB0aGUgbmV4dCBwYWdlIG9mIHJlc3BvbnNlcwogICAgICAgIHNjaGVtYToKICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgY29udGVudDoKICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICBzY2hlbWE6IAogICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgIGl0ZW1zOiAKICAgICAgICAgICAgJHJlZjogJyMvY29tcG9uZW50cy9zY2hlbWFzL1BldCcKYGBgCgojIyMgQ29tcG9uZW50cwpIb2xkcyByZXVzYWJsZSBvYmplY3RzLCBzdWNoIGFzIHNjaGVtYXMsIHJlc3BvbnNlcywgcGFyYW1ldGVycywgYW5kIHNlY3VyaXR5IGRlZmluaXRpb25zLgpgYGB5YW1sCmNvbXBvbmVudHM6CiAgc2NoZW1hczoKICAgIFBldDoKICAgICAgdHlwZTogb2JqZWN0CiAgICAgIHJlcXVpcmVkOgogICAgICAgIC0gaWQKICAgICAgICAtIG5hbWUKICAgICAgcHJvcGVydGllczoKICAgICAgICBpZDoKICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgICAgICAgIGZvcm1hdDogaW50NjQKICAgICAgICBuYW1lOgogICAgICAgICAgdHlwZTogc3RyaW5nCiAgICAgICAgdGFnOgogICAgICAgICAgdHlwZTogc3RyaW5nCmBgYAoKIyMjIEJlc3QgUHJhY3RpY2VzCgoxLiAqKlVzZSBNZWFuaW5nZnVsIERlc2NyaXB0aW9uczoqKiBQcm92aWRlIGRldGFpbGVkIGRlc2NyaXB0aW9ucyBmb3IgZWFjaCBmaWVsZCwgZW5kcG9pbnQsIGFuZCByZXNwb25zZSB0byBtYWtlIHRoZSBkb2N1bWVudGF0aW9uIHVzZXItZnJpZW5kbHkuCjIuICoqS2VlcCBJdCBEUlkgKERvbid0IFJlcGVhdCBZb3Vyc2VsZik6KiogVXRpbGl6ZSB0aGUgY29tcG9uZW50cyBzZWN0aW9uIHRvIGF2b2lkIHJlZHVuZGFuY3kuCjMuICoqVmFsaWRhdGUgdGhlIFNwZWNpZmljYXRpb246KiogVXNlIHRvb2xzIGxpa2UgU3dhZ2dlciBvciBSZWRvY2x5IHRvIHZhbGlkYXRlIGFuZCB2aXN1YWxpemUgeW91ciBPcGVuQVBJIHNwZWNpZmljYXRpb24uCjQuICoqRG9jdW1lbnQgU2VjdXJpdHk6KiogQ2xlYXJseSBkZWZpbmUgYXV0aGVudGljYXRpb24gYW5kIGF1dGhvcml6YXRpb24gbWVjaGFuaXNtcy4KNS4gKipWZXJzaW9uIFlvdXIgQVBJOioqIEluY2x1ZGUgdmVyc2lvbmluZyBpbiB0aGUgQVBJIHBhdGhzIG9yIGhlYWRlcnMgdG8gaGFuZGxlIGJyZWFraW5nIGNoYW5nZXMgZ3JhY2VmdWxseS4K readmeEtag: '"f4d2d3ecf6febd9cea530d8249ae8386a2507849"' readmeLastModified: Tue, 09 Jul 2024 23:24:34 GMT repositoryId: 826362668 description: >- OpenAPI v3.x, Swagger UI v5.x, Swagger Editor v4.x, Redoc v2.x, Docker Compose, GitHub Pages created: '2024-07-09T15:04:25Z' updated: '2024-07-09T23:32:30Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: refactorian logo: https://avatars.githubusercontent.com/u/110738252?v=4 repoEtag: '"aad1251e2d3f66edfe71ead919272f9e25d82b519c077a3c0e789f58f238b6e8"' repoLastModified: Tue, 09 Jul 2024 23:32:30 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/robert-dzikowski/api-smoke-test v3: true id: e5e5c07daf5797c09a6fbcec3aa5a2f2 repositoryMetadata: base64Readme: >- IyBJbnRyb2R1Y3Rpb24gClRoaXMgcmVwbyBjb250YWlucyBzbW9rZSB0ZXN0IHdoaWNoIHRlc3RzIGFsbCBlbmRwb2ludHMgZm91bmQgaW4gT3BlbkFQSSB2LjMgc3BlYyBmaWxlIChpdCB3YXNuJ3QgdGVzdGVkIHdpdGggb2xkZXIgdmVyc2lvbnMgb2YgT3BlbkFQSSkuIApJdCB3YXMgdGVzdGVkIHdpdGggU3BvdGlmeSBBUEkgKGh0dHBzOi8vZ2l0aHViLmNvbS9zb25hbGx1eC9zcG90aWZ5LXdlYi1hcGkpLgogIAojIEdldHRpbmcgU3RhcnRlZApJbnN0YWxsIFB5dGhvbiAzLjkgb3IgbmV3ZXIuCgpUbyBpbnN0YWxsIHBhY2thZ2VzIGdsb2JhbGx5IChmb3IgYWxsIHVzZXJzKSwKcnVuICdwaXAgaW5zdGFsbCAtciByZXF1aXJlbWVudHMudHh0JyBpbiBwcm9qZWN0IHJvb3QgZGlyZWN0b3J5LgoKIyBSdW4KcHl0aG9uIGFwaS1zbW9rZS10ZXN0XHNtb2tlX3Rlc3QucHkgbmFtZV9vZl9hX3NwZWNfZmlsZSBbLS1hdXRoXSBbLS1sb2NhbGhvc3RdIFstLW9ubHktZ2V0XSBbLS1yZXF1ZXN0LXBhcmFtPV0gCgpvcgoKcHl0aG9uIGFwaS1zbW9rZS10ZXN0XHNtb2tlX3Rlc3QucHkgdXJsX29mX2Ffc3BlY19maWxlIFstLWF1dGhdIC4uLgoKLS1hdXRoIFNtb2tlIHRlc3QgbmVlZHMgYXV0aG9yaXphdGlvbiB0byBtYWtlIEFQSSByZXF1ZXN0cy4gQWxzbyB5b3UgbmVlZCB0byBmaWxsIGNvbmZpZy5weSB3aXRoIGFwcHJvcHJpYXRlIHZhbHVlcy4KCi0tbG9jYWxob3N0IEFQSSBpcyBydW5pbmcgb24gYSBsb2NhbCBtYWNoaW5lLiAKSW4gbXkgY2FzZSBsb2NhbCBBUEkgdXNlcyBIVFRQIGluc3RlYWQgb2YgSFRUUFMsIHNvIHlvdSBzaG91bGQgYWRhcHQgbXkgY29kZSB0byB5b3VyIG5lZWRzLgoKLS1vbmx5LWdldCBUZXN0IG9ubHkgR0VUIG1ldGhvZHMuCgotLXJlcXVlc3QtcGFyYW09IFNldCB2YWx1ZSBmb3IgcmVxdWVzdHMgcGFyYW1ldGVycywgCmUuZy4gaWYgLS1yZXF1ZXN0LXBhcmFtPTcgdGhlbiBmb3IgL2FsYnVtcy97aWR9IGVuZHBvaW50IHRlc3Qgd2lsbCBzZW5kIHJlcXVlc3QgdG8gL2FsYnVtcy83LgoKCiMgVE9ETwpBZGQgYWJpbGl0eSB0byBzZWxlY3QgYXV0aG9yaXphdGlvbiBtZXRob2Q6IAotIGNsaWVudCBpZCBhbmQgc2VjcmV0IChkZWZhdWx0LCB1c2VkIGJ5IFNwb3RpZnkgQVBJKQotIGNsaWVudCBpZCBhbmQgc2VjcmV0LCB1c2VyIG5hbWUgYW5kIHBhc3N3b3JkCg== readmeEtag: '"d6c8e39c9f9e56a2d993b3fa9aed96f7d259aaf9"' readmeLastModified: Sat, 27 May 2023 16:47:59 GMT repositoryId: 330251410 description: Smoke test for APIs that use OpenAPI v.3 specification created: '2021-01-16T20:36:00Z' updated: '2024-10-22T10:11:20Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: robert-dzikowski logo: https://avatars.githubusercontent.com/u/12864860?v=4 license: MIT repoEtag: '"e5a9e31f3dacf9cc5d7295179f12c86488e577b6cc68b2a36ee3a42787beda2c"' repoLastModified: Tue, 22 Oct 2024 10:11:20 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/aeimer/clockify-openapi-spec v3: true id: f2d454ed4a7cade537647e90a9a015fe repositoryMetadata: base64Readme: >- IyBJbi1vZmZpY2lhbCBDbG9ja2lmeSBPcGVuQVBJIHNwZWMKClRoaXMgaXMgYSBpbm9mZmljaWFsIENsb2NraWZ5IE9wZW5BUEkgc3BlYyBiYXNlZCBvbiB0aGUKW29mZmljaWFsIGRvY3NdKGh0dHBzOi8vZG9jcy5jbG9ja2lmeS5tZSkuCgpUaGUgbWFpbiBnb2FsIG9mIHRoaXMgcmVwbyBpcyB0byBoYXZlIGEgYmV0dGVyIHNwZWMgdGhhbiB0aGUgQVBJIGRvY3MuClNhZGx5IENsb2NraWZ5IGRvZXMgbm90IHB1Ymxpc2hlcyBhIE9wZW5BUEkgc3BlYyAob3IgZG8gdGhleT8pIGFuZCB0aGUgQVBJIGRvY3Mgc2VlbSB0byBiZSBpbmNvbXBsZXRlLgpUaGVyZWZvcmUsIHRoaXMgT3BlbkFQSSBzcGVjIHNob3VsZCBoZWxwIHlvdSB0byBlLmcuIGdlbmVyYXRlIFNES3MgaW4gdGhlIGxhbmd1YWdlIHlvdSBuZWVkLgoKU2VlCltTd2FnZ2VyIFVJXShodHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vP3VybD1odHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vYWVpbWVyL2Nsb2NraWZ5LW9wZW5hcGktc3BlYy9yZWZzL2hlYWRzL21haW4vY2xvY2tpZnktb3BlbmFwaS1zcGVjLnlhbWwpLgoKRElTQ0xBSU1FUjoKVGhpcyBpcyAxMDAlIGJlc3QgZWZmb3J0IGFuZCBub3QgcmVsYXRlZCBpbiBhbnkga2luZCB0byBDbG9ja2lmeSEKCkxhc3QgdXBkYXRlZDogMjAyNC0xMS0xOAoKIyMgT2ZmaWNpYWwgQ2xvY2tpZnkgT3BlbkFQSSBzcGVjCgpUaGVyZSBpcyBhbHNvIGFuIE9wZW5BUEkgc3BlYyBhdmFpbGFibGUgdW5kZXIKaHR0cHM6Ly9hcGkuY2xvY2tpZnkubWUvYXBpL3YzL2FwaS1kb2NzCgpTZWUKW1N3YWdnZXIgVUldKGh0dHBzOi8vcGV0c3RvcmUuc3dhZ2dlci5pby8/dXJsPWh0dHBzOi8vYXBpLmNsb2NraWZ5Lm1lL2FwaS92My9hcGktZG9jcykuCgpBIGNvcHkgY2FuIGJlIGZvdW5kIGluIHRoZSBmaWxlIFtgb2ZmaWNpYWwtY2xvY2tpZnktb3BlbmFwaS1zcGVjLnlhbWxgXShvZmZpY2lhbC1jbG9ja2lmeS1vcGVuYXBpLXNwZWMueWFtbCkuClRoZSBjb3B5IHdhcyBjcmVhdGVkIGF0IDIwMjUtMTAtMjMgd2l0aApgaHR0cCBHRVQgaHR0cHM6Ly9hcGkuY2xvY2tpZnkubWUvYXBpL3YzL2FwaS1kb2NzIHwgeXEgLVAgLSA+IG9mZmljaWFsLWNsb2NraWZ5LW9wZW5hcGktc3BlYy55YW1sYAoKRElTQ0xBSU1FUjoKVGhpcyBpcyBhbiB1bmRvY3VtZW50ZWQgZW5kcG9pbnQsIHVzZSB3aXRoIGNhdXRpb24hCgojIyBDb250cmlidXRpb24KCklmIHlvdSB3YW50IHRvIGhlbHAgbWUga2VlcGluZyB0aGlzIEFQSSBzcGVjIHVwdG9kYXRlLCBJIHdvdWxkIGxvdmUgZ2V0IHNvbWUgcHVsbC1yZXF1ZXN0cyDwn6STCgpUaGlzIEFQSSBzcGVjIGlzIGdlbmVyYXRlZCB3aXRoCltBUElibGRyXShodHRwczovL2FwaWJsZHIuY29tKS4KCiMjIFN0YXIgaGlzdG9yeQoKWyFbU3RhciBIaXN0b3J5XShodHRwczovL2FwaS5zdGFyLWhpc3RvcnkuY29tL3N2Zz9yZXBvcz1hZWltZXIvY2xvY2tpZnktb3BlbmFwaS1zcGVjJnR5cGU9RGF0ZSldKGh0dHBzOi8vc3Rhci1oaXN0b3J5LmNvbS8jYWVpbWVyL2Nsb2NraWZ5LW9wZW5hcGktc3BlYyZEYXRlKQo= readmeEtag: '"ea5373da2876c45c4e87cdc872dcea866f307d6d"' readmeLastModified: Thu, 23 Oct 2025 16:03:34 GMT repositoryId: 890375840 description: In-official Clockify OpenAPI Spec v3 created: '2024-11-18T13:15:54Z' updated: '2026-01-05T18:12:34Z' language: null archived: false stars: 3 watchers: 1 forks: 0 owner: aeimer logo: https://avatars.githubusercontent.com/u/25035182?v=4 license: MIT repoEtag: '"baa73bfdef9b238e3c98c73110ae61d145a8f94a42e62b2917c83d671fe1d126"' repoLastModified: Mon, 05 Jan 2026 18:12:34 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/nachonacho-inc/public-api-doc v3: true id: 092df77af6c1ae32ddbda7b3e299888a repositoryMetadata: base64Readme: >- IyBOYWNob05hY2hvIFB1YmxpYyBBUEkgRG9jdW1lbnRhdGlvbgoKIVtJbWFnZSBvZiBOYWNob05hY2hvXShodHRwczovL2FwcC5uYWNob25hY2hvLmNvbS9sb2dvL05hY2hvTmFjaG9Mb2dvLnBuZykKCkFQSSBEb2N1bWVudGF0aW9uIGV4cG9zZWQgdmlhIE9wZW5BUEkgVUkuCgpZb3UgY2FuIGFjY2VzcyBpdCBoZXJlIDogaHR0cHM6Ly9uYWNob25hY2hvLWluYy5naXRodWIuaW8vcHVibGljLWFwaS1kb2MK readmeEtag: '"b52855e831ff7d6d32c9a3b380a838dddbbd50da"' readmeLastModified: Tue, 16 Aug 2022 20:01:59 GMT repositoryId: 524476113 description: NachoNacho Public API Documentation created: '2022-08-13T18:20:20Z' updated: '2022-10-26T18:23:38Z' language: Makefile archived: false stars: 1 watchers: 6 forks: 2 owner: nachonacho-inc logo: https://avatars.githubusercontent.com/u/64567964?v=4 repoEtag: '"913bd1785674ab756165201073ea62cde02253d0667f36fd8567a8f4dacd01d4"' repoLastModified: Wed, 26 Oct 2022 18:23:38 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/wndhydrnt/openapi-go-server v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWdvLXNlcnZlcgoKVGVtcGxhdGVzIGZvciB0aGUgZ28tc2VydmVyIGdlbmVyYXRvciBvZiB0aGUgW09wZW5BUEkgR2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKS4KClRoZSB0ZW1wbGF0ZXMgYXJlIGEgZm9yayBvZiB0aGUgW29yaWdpbmFsIGdvLXNlcnZlciBnZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IvdHJlZS9tYXN0ZXIvbW9kdWxlcy9vcGVuYXBpLWdlbmVyYXRvci9zcmMvbWFpbi9yZXNvdXJjZXMvZ28tc2VydmVyKSB0ZW1wbGF0ZXMuCgojIyBGZWF0dXJlcyBhbmQgQ2hhbmdlcwoKKiBFbmFibGUgW0Rpc2FsbG93VW5rbm93bkZpZWxkc10oaHR0cHM6Ly9nb2xhbmcub3JnL3BrZy9lbmNvZGluZy9qc29uLyNEZWNvZGVyLkRpc2FsbG93VW5rbm93bkZpZWxkcykgb24gSlNPTiBkZWNvZGVyCiogT3B0aW9uYWxseSBzZXQgcmVzcG9uc2UgY29kZSBhbmQgYm9keSBvbiBlcnJvcgoqIFJlbW92ZSByZXF1ZXN0IGxvZ2dpbmcKKiBPcHRpb25hbGx5IHNldCBhIGxvZ2dlciB0byBsb2cgZXJyb3JzIGluIENvbnRyb2xsZXJzCiogUGFzcyBjb250ZXh0IG9mIHRoZSBpbmNvbWluZyBgaHR0cC5SZXF1ZXN0YCB0byBtZXRob2RzIG9mIGEgU2VydmljZQoqIExldCBtZXRob2RzIG9mIGEgU2VydmljZSByZXR1cm4gdHlwZXMKCiMjIEtub3duIENhdmVhdHMKCiogU3VwcG9ydHMgb25seSBvbmUgcmV0dXJuIHR5cGUgb24gbWV0aG9kcyBvZiBhIFNlcnZpY2UKKiBTdXBwb3J0cyBKU09OIGVuY29kaW5nIG9mIHJlc3BvbnNlIGJvZHkgb25seQoKIyMgRXhhbXBsZQoKU2VlIFtleGFtcGxlcy9wZXRzdG9yZV0oZXhhbXBsZXMvcGV0c3RvcmUpLgoKIyMgVXNhZ2UKCmBgYApjdXJsIC1MIC1vIG9wZW5hcGktZ28tc2VydmVyLnppcCBodHRwczovL2dpdGh1Yi5jb20vd25kaHlkcm50L29wZW5hcGktZ28tc2VydmVyL2FyY2hpdmUvbWFzdGVyLnppcAp1bnppcCBvcGVuYXBpLWdvLXNlcnZlci56aXAKcm0gb3BlbmFwaS1nby1zZXJ2ZXIuemlwCm12IG9wZW5hcGktZ28tc2VydmVyLW1hc3Rlci9nby1zZXJ2ZXIgLgpybSAtcmYgb3BlbmFwaS1nby1zZXJ2ZXItbWFzdGVyLwpvcGVuYXBpLWdlbmVyYXRvciBnZW5lcmF0ZSAtaSBodHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjIvc3dhZ2dlci5qc29uIC1vIC4gLS1nZW5lcmF0b3ItbmFtZSBnby1zZXJ2ZXIgLS1hZGRpdGlvbmFsLXByb3BlcnRpZXM9c291cmNlRm9sZGVyPWFwaSxwYWNrYWdlTmFtZT1hcGkgLXQgLi9nby1zZXJ2ZXIKYGBgCg== readmeEtag: '"9ba486b02a59726d0ed36d8a31fd731d61f74351"' readmeLastModified: Sat, 02 Nov 2019 14:48:13 GMT repositoryId: 219166478 description: Templates for the go-server generator of the OpenAPI Generator. created: '2019-11-02T14:36:54Z' updated: '2022-12-30T07:07:23Z' language: HTML archived: false stars: 1 watchers: 1 forks: 1 owner: wndhydrnt logo: https://avatars.githubusercontent.com/u/3942978?v=4 license: MIT repoEtag: '"8f8fc87562d4693b23aa07117c7d8c2aab9fde5a781bfa11c09bd1d45d11cd41"' repoLastModified: Fri, 30 Dec 2022 07:07:23 GMT foundInMaster: true category: - SDK - Parsers id: 061a1147b9de1d975771f791659ca4f6 - source: openapi3 tags repository: https://github.com/akh11k/flask-template v3: true repositoryMetadata: base64Readme: IyBmbGFzay10ZW1wbGF0ZQpBIGZsYXNrIHRlbXBsYXRlIHdpdGggcnEsIGNvbm5leGlvbgo= readmeEtag: '"1497df7bfbd157a003cc5e95ca0ed09358d2e814"' readmeLastModified: Mon, 06 May 2019 06:47:43 GMT repositoryId: 184807083 description: A flask template with rq, connexion created: '2019-05-03T19:05:28Z' updated: '2019-05-06T06:48:28Z' language: Python archived: false stars: 1 watchers: 1 forks: 1 owner: akh11k logo: https://avatars.githubusercontent.com/u/44919026?v=4 repoEtag: '"461f3d78d9e85d2b097124488438784d2a7c3a2415a9b1f20cad8c99bb14cf1c"' repoLastModified: Mon, 06 May 2019 06:48:28 GMT foundInMaster: true category: - Mock - Server Implementations id: 3525950c4113199e2652a648c17b7efd - source: openapi3 tags repository: https://github.com/dzx912/sberapi-mock v3: true id: 2f68ab17ae20e48422266d441520f3b3 repositoryMetadata: base64Readme: >- IyBzYmVyYXBpLW1vY2sgWyFbYnVpbGQgc3RhdHVzXShodHRwczovL2dpdGh1Yi5jb20vcmlnZTEvc2JlcmFwaS1tb2NrL2FjdGlvbnMvd29ya2Zsb3dzL3dvcmtmbG93LnltbC9iYWRnZS5zdmc/YnJhbmNoPW1haW4pXShodHRwczovL2dpdGh1Yi5jb20vcmlnZTEvc2JlcmFwaS1tb2NrL2FjdGlvbnMvd29ya2Zsb3dzL3dvcmtmbG93LnltbCkKCtCh0LXRgNCy0LXRgCDQt9Cw0LPQu9GD0YjQutCwIFNiZXIgQVBJLiDQmNGB0L/QvtC70YzQt9GD0LXRgtGB0Y8g0LTQu9GPINCx0YvRgdGC0YDQvtC5INC40L3RgtC10LPRgNCw0YbQuNC4INC4INGC0LXRgdGC0LjRgNC+0LLQsNC90LjRjyAKCiMjINCn0YLQviDRg9C80LXQtdGCIAoKKiDQk9C10L3QtdGA0LjRgNGD0LXRgiDQuCDQv9GA0L7QstC10YDRj9C10YIg0LfQsNC/0YDQvtGB0Ysg0LjRgdC/0L7Qu9GM0LfRg9GPIE9wZW5BUEkgMwoqINCY0LzQtdC10YIg0L/QvtC00LTQtdGA0LbQutGDIFRMUyDQuCBtVExTCgojIyDQn9GA0LjQvNC10YDRiyDQuNGB0L/QvtC70YzQt9C+0LLQsNC90LjRjyAKCtCX0LDQv9GD0YHQuiDQsdC10Lcg0LDRgNCz0YPQvNC10L3RgtC+0LIuINCh0LXRgNCy0LXRgCDQsdGD0LTQtdGCINGB0LvRg9GI0LDRgtGMINC90LAg0L/QvtGA0YLQtSA4MDgwOiAKCmBgYHNoCnNiZXJhcGktbW9jayBzdGFydApgYGAKCtCX0LDQv9GD0YHQuiDRgSDRg9C60LDQt9Cw0L3QuNC10Lwg0L/QvtGA0YLQsDogCgpgYGBzaApzYmVyYXBpLW1vY2sgc3RhcnQgLS1wb3J0IDgwODQKYGBgCgrQl9Cw0L/Rg9GB0Log0YEgbVRMUzoKCmBgYHNoIApzYmVyYXBpLW1vY2sgc3RhcnQgLS1jZXJ0IHNlcnZlcl9jZXJ0LnBlbSAtLWtleSBzZXJ2ZXJfa2V5LnBlbSAtLWNsaWVudC1jZXJ0IGNsaWVudF9jZXJ0LnBlbSAKYGBgCgrQntGC0LrQu9GO0YfQtdC90LjQtSDQstCw0LvQuNC00LDRhtC40Lgg0LfQsNC/0YDQvtGB0LA6IApgYGBzaApzYmVyYXBpLW1vY2sgc3RhcnQgLS1pZ25vcmUtdmFsaWRhdGlvbgpgYGAKCiMjINCf0YDQuNC80LXRgNGLINC30LDQv9GA0L7RgdC+0LIgCgpD0L/QuNGB0L7QuiDQtNC+0YHRgtGD0L/QvdGL0YUgQVBJOgpgYGBzaApjdXJsIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4NAoKKFBPU1QpIC9jcmVhdGlvbgooUE9TVCkgL3N0YXR1cwpgYGAK0KHQvtC30LTQsNC90LjQtSBRUjogCmBgYHNoCiMg0JfQsNC/0YPRgdC60LDQtdC8INC30LDQs9C70YPRiNC60YMg0LHQtdC3INC/0YDQvtCy0LXRgNC60Lgg0LfQsNC/0YDQvtGB0L7QsiAKc2JlcmFwaS1tb2NrIHN0YXJ0IC0taWdub3JlLXZhbGlkYXRpb24KCiMg0JfQsNC/0YDQvtGBCmN1cmwgLVggUE9TVCBodHRwOi8vbG9jYWxob3N0OjgwODAvY3JlYXRpb24gLWQgJ3t9JyB8IGpxIAoKIyDQntGC0LLQtdGCCnsKICAic3RhdHVzIjogewogICAgImVycm9yX2NvZGUiOiAiMDAwMDAwIiwKICAgICJlcnJvcl9kZXNjcmlwdGlvbiI6ICLQntC/0LjRgdCw0L3QuNC1INC+0YjQuNCx0LrQuCDQstGL0L/QvtC70L3QtdC90LjRjyDQt9Cw0L/RgNC+0YHQsCIsCiAgICAib3JkZXJfZm9ybV91cmwiOiAiaHR0cHM6Ly9zYmVyYmFuay5ydS9xci8/dXVpZD0xMTExMTExMTExMTExMTExMTEiLAogICAgIm9yZGVyX2lkIjogIjEwMDAxMDAwNTE4OTU2NjM3IiwKICAgICJvcmRlcl9udW1iZXIiOiAiNzc0NjM1NTI2NjM3IiwKICAgICJvcmRlcl9zdGF0ZSI6ICJDUkVBVEVEIiwKICAgICJycV90bSI6ICIyMDA1LTA4LTE1VDE1OjUyOjAxWiIsCiAgICAicnFfdWlkIjogImFjMTFjQTFDRWFlMUQxMTExZEFCZjFmRDFCYjBhY0FkIgogIH0KfQpgYGAKCiMjINCh0LHQvtGA0LrQsAoKYGBgc2ggCm1ha2UgYnVpbGQgCmBgYAoKIyMgRG9ja2VyCiMjIyDQodCx0L7RgNC60LAKCmBgYHNoIApkb2NrZXIgaW1hZ2UgYnVpbGQgLS10YWcgc2JlcmFwaS1tb2NrIC4KYGBgCgojIyMg0JfQsNC/0YPRgdC6CgpgYGBzaCAKZG9ja2VyIHJ1biAtLXB1Ymxpc2ggODA4MDo4MDgwIHNiZXJhcGktbW9jawpgYGAKCiMjIyDQl9Cw0L/Rg9GB0Log0YEg0L/QsNGA0LDQvNC10YLRgNCw0LzQuAoKYGBgc2ggCmRvY2tlciBydW4gLS1wdWJsaXNoIDgwODA6ODA4MCBzYmVyYXBpLW1vY2sgLS1pZ25vcmUtdmFsaWRhdGlvbgpgYGAKCmBgYHNoIApkb2NrZXIgcnVuIC0tcHVibGlzaCA4MDgwOjgwODAgLS12b2x1bWUgPGFic29sdW1lX3BhdGhfdG9fY2VydF9kaXI+Oi9hcHAvY2VydC8gc2JlcmFwaS1tb2NrIC0tY2VydCAvYXBwL2NlcnQvc2VydmVyX2NlcnQucGVtIC0ta2V5IC9hcHAvY2VydC9zZXJ2ZXJfa2V5LnBlbSAtLWNsaWVudC1jZXJ0IC9hcHAvY2VydC9jbGllbnRfY2VydC5wZW0gCmBgYA== readmeEtag: '"a7023168cfa7862ebc2f7c52ec9a486798b30ff6"' readmeLastModified: Fri, 18 Feb 2022 22:17:42 GMT repositoryId: 379581037 description: Сервер заглушка Sber API created: '2021-06-23T11:33:27Z' updated: '2023-03-02T18:51:14Z' language: Go archived: true stars: 1 watchers: 0 forks: 0 owner: dzx912 logo: https://avatars.githubusercontent.com/u/3342950?v=4 repoEtag: '"36c5df63b7a211847b49c9b171fc07646338b4eeb8597ffbd4db28a7d4e48beb"' repoLastModified: Thu, 02 Mar 2023 18:51:14 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/himenon/openapi-typescript-code-generator-demo-project v3: true repositoryMetadata: base64Readme: >- IyBEZW1vIHByb2plY3QgZm9yIFtIaW1lbm9uL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlLWdlbmVyYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL0hpbWVub24vb3BlbmFwaS10eXBlc2NyaXB0LWNvZGUtZ2VuZXJhdG9yKQoKW+aXpeacrOiqnl0oLi9SRUFETUVfamEubWQpCgpUaGlzIHNlY3Rpb24gZGVzY3JpYmVzIHRoZSBzcGVjaWZpYyBzdGVwcyBmb3IgY29kZSBnZW5lcmF0aW9uLgoKIyMgU2V0dXAKCkZpcnN0LCB5b3UgbmVlZCB0byBjbG9uZSB0aGlzIHJlcG9zaXRvcnkuIAoKYGBgc2gKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9IaW1lbm9uL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlLWdlbmVyYXRvci1kZW1vLXByb2plY3QKYGBgCgpUaGVuIGdvIHRvIHRoZSBjbG9uZWQgZGlyZWN0b3J5IGFuZCBpbnN0YWxsIHRoZSBwYWNrYWdlLgoKYGBgc2gKY2Qgb3BlbmFwaS10eXBlc2NyaXB0LWNvZGUtZ2VuZXJhdG9yLWRlbW8tcHJvamVjdAp5YXJuIGluc3RhbGwKYGBgCgpUaGUgc2V0dXAgaXMgbm93IGNvbXBsZXRlLgoKIyMgUnVuIGRlbW8KCkxldCdzIGFjdHVhbGx5IGNyZWF0ZSB0aGUgY29kZSBpbiBbVXNhZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9IaW1lbm9uL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlLWdlbmVyYXRvciN1c2FnZSkuClRoZSBPcGVuQVBJIFNjaGVtYSB0byBiZSBsb2FkZWQgaXMgYG9wZW5hcGkueW1sYC4KCkFsc28sIHRoZSBhY3R1YWwgY29kZSB0byBiZSBleGVjdXRlZCB3aWxsIGJlIHdyaXR0ZW4gaW4gcGxhaW4gSmF2YVNjcmlwdC4KCiMjIyBEZW1vMQoKRXhhbXBsZXMgdG8gREVNT++8mltHZW5lcmF0ZSB0eXBlZGVmLW9ubHkgY29kZV0oaHR0cHM6Ly9naXRodWIuY29tL0hpbWVub24vb3BlbmFwaS10eXBlc2NyaXB0LWNvZGUtZ2VuZXJhdG9yI2dlbmVyYXRlLXR5cGVkZWYtb25seS1jb2RlKQoKRXhlY3V0ZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQgdG8gZ2VuZXJhdGUgdGhlIGNvZGUgaW4gdGhlIGRlbW8xIGRpcmVjdG9yeS4KCmBgYHNoCm5vZGUgZ2VuZXJhdGUtZGVtbzEuanMKYGBgCgojIyMgRGVtbzIKCkV4YW1wbGVzIHRvIERFTU/vvJpbR2VuZXJhdGUgY29kZSBjb250YWluaW5nIHRoZSBBUEkgQ2xpZW50XShodHRwczovL2dpdGh1Yi5jb20vSGltZW5vbi9vcGVuYXBpLXR5cGVzY3JpcHQtY29kZS1nZW5lcmF0b3IjZ2VuZXJhdGUtY29kZS1jb250YWluaW5nLXRoZS1hcGktY2xpZW50KQoKRXhlY3V0ZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQgdG8gZ2VuZXJhdGUgdGhlIGNvZGUgaW4gdGhlIGRlbW8yIGRpcmVjdG9yeS4KCmBgYHNoCm5vZGUgZ2VuZXJhdGUtZGVtbzIuanMKYGBgCgojIyBEZW1vMwoKRXhhbXBsZXMgdG8gREVNT++8mltTcGxpdCB0aGUgdHlwZSBkZWZpbml0aW9uIGZpbGUgYW5kIHRoZSBBUEkgQ2xpZW50IGltcGxlbWVudGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vSGltZW5vbi9vcGVuYXBpLXR5cGVzY3JpcHQtY29kZS1nZW5lcmF0b3Ijc3BsaXQtdGhlLXR5cGUtZGVmaW5pdGlvbi1maWxlLWFuZC10aGUtYXBpLWNsaWVudC1pbXBsZW1lbnRhdGlvbikKCkV4ZWN1dGUgdGhlIGZvbGxvd2luZyBjb21tYW5kIHRvIGdlbmVyYXRlIHRoZSBjb2RlIGluIHRoZSBkZW1vMyBkaXJlY3RvcnkuCgpgYGBzaApub2RlIGdlbmVyYXRlLWRlbW8zLmpzCmBgYAoKIyMgQWJvdXQgdGhlIHBhY2thZ2UgeW91IG5lZWQKCltAaGltZW5vbi9vcGVuYXBpLXR5cGVzY3JpcHQtY29kZS1nZW5lcmF0b3JdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL0BoaW1lbm9uL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlLWdlbmVyYXRvcikgcHJvdmlkZXMgYSBtZWNoYW5pc20gZm9yIGdlbmVyYXRpbmcgY29kZSBmcm9tIE9wZW5BUEkgU2NoZW1hLgoKSXQgYWxzbyB1c2VzIFR5cGVTY3JpcHQgaXRzZWxmIGRpcmVjdGx5IGludGVybmFsbHkuIFRoZXJlZm9yZSwgaWYgeW91IGRvIG5vdCBoYXZlIHRoaXMgW3R5cGVzY3JpcHRdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3R5cGVzY3JpcHQpIHBhY2thZ2UgaW5zdGFsbGVkLCB5b3Ugd2lsbCBub3QgYmUgYWJsZSB0byBnZW5lcmF0ZSBjb2RlLgoK readmeEtag: '"4f1dfdd8b3024ad9eecd8ab24613b8f2013dd624"' readmeLastModified: Sun, 12 Sep 2021 14:18:25 GMT repositoryId: 355945487 description: Demo Project for @himenon/openapi-typescript-code-generator created: '2021-04-08T14:44:28Z' updated: '2022-05-22T15:21:37Z' language: TypeScript archived: false stars: 1 watchers: 0 forks: 2 owner: Himenon logo: https://avatars.githubusercontent.com/u/6715229?v=4 repoEtag: '"8f46e3f0cc3af82742c89e9b04c370856189c89d303553987c741e94e83384f0"' repoLastModified: Sun, 22 May 2022 15:21:37 GMT foundInMaster: true category: SDK id: d2f30604362509539598b330fb1bbb68 - source: openapi3 tags repository: https://github.com/bmykyta/cheese-whiz v3: true id: 9aa890b5402d1a9b42ae222e1fed6343 repositoryMetadata: base64Readme: >- IyBDaGVlc2UgV2hpeiAoQVBJIFBsYXRmb3JtKQoKIyMgQWJvdXQgdGhlIHByb2plY3QKClRoZSBwcm9qZWN0IGlzIGJ1aWx0IG9uIHRvcCBvZiB0aGUgU3ltZm9ueSA2IHdpdGggQVBJIFBsYXRmb3JtIDIuNiBidW5kbGUuIEZvbGxvd2VkIGJ5IHRoZSB0dXRvcmlhbCBvZgpbU3ltZm9ueUNhc3RzIC0gQVBJIFBsYXRmb3JtOiBTZXJpb3VzIFJFU1RmdWwgQVBJc10oaHR0cHM6Ly9zeW1mb255Y2FzdHMuY29tL3NjcmVlbmNhc3QvYXBpLXBsYXRmb3JtKS4KCiMjIyBCdWlsZCB3aXRoCgohW1BIUDhdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vc3RhdGljL3YxP3N0eWxlPWZvci10aGUtYmFkZ2UmbWVzc2FnZT1QSFAlMjA4JmNvbG9yPTc3N0JCNCZsb2dvQ29sb3I9d2hpdGUmbG9nbz1waHAmbGFiZWw9KQohW1N5bWZvbnldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vc3RhdGljL3YxP3N0eWxlPWZvci10aGUtYmFkZ2UmbWVzc2FnZT1TeW1mb255JTIwNiZjb2xvcj0wMDAwMDAmbG9nbz1TeW1mb255JmxvZ29Db2xvcj1GRkZGRkYmbGFiZWw9KQohW09wZW5BUEkzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3N0YXRpYy92MT9zdHlsZT1mb3ItdGhlLWJhZGdlJm1lc3NhZ2U9T3BlbkFQSSUyMDMmY29sb3I9NkJBNTM5JmxvZ289T3BlbkFQSStJbml0aWF0aXZlJmxvZ29Db2xvcj1GRkZGRkYmbGFiZWw9KQoKIyMgR2V0dGluZyBTdGFydGVkCgojIyMgUHJlcmVxdWlzaXRlcwoKLSBQSFAgOAotIENvbXBvc2VyIAotIERvY2tlci1jb21wb3NlCgojIyMgSW5zdGFsbGF0aW9uCgoxLiBDbG9uZSB0aGUgcmVwbzogCmBgYGJhc2gKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9ibXlreXRhL2NoZWVzZS13aGl6LmdpdApgYGAKMi4gQ29weSB0aGUgLmVudiBmaWxlIGFuZCBhZGp1c3QgdG8geW91ciBuZWVkcwozLiBSdW4gZG9ja2VyIGNvbXBvc2UgY29tbWFuZApgYGBiYXNoCmRvY2tlci1jb21wb3NlIHVwIC1kCmBgYAo0LiBHbyB0byBwaHAtZnBtIHNlcnZpY2UgCmBgYGJhc2gKZG9ja2VyLWNvbXBvc2UgZXhlYyBwaHAtZnBtIGJhc2gKYGBgCjUuIFJ1biBjb21wb3NlciBpbnN0YWxsLCBjcmVhdGUgREIgYW5kIG1pZ3JhdGlvbnMKYGBgYmFzaApjb21wb3NlciBpbnRhbGwKcGhwIGJpbi9jb25zb2xlIGQ6ZDpjIApwaHAgYmluL2NvbnNvbGUgZDptOm0gLW4KYGBgCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHdoYXQgbWFrZSB0aGUgb3BlbiBzb3VyY2UgY29tbXVuaXR5IHN1Y2ggYW4gYW1hemluZyBwbGFjZSB0byBsZWFybiwgaW5zcGlyZSwgYW5kIGNyZWF0ZS4gCkFueSBjb250cmlidXRpb25zIHlvdSBtYWtlIGFyZSAqKmdyZWF0bHkgYXBwcmVjaWF0ZWQqKi4g8J+RjwoKSWYgeW91IGhhdmUgYSBzdWdnZXN0aW9uIHRoYXQgd291bGQgbWFrZSB0aGlzIGJldHRlciwgcGxlYXNlIGZvcmsgdGhlIHJlcG8gYW5kIGNyZWF0ZSBhIHB1bGwgcmVxdWVzdC4gCllvdSBjYW4gYWxzbyBzaW1wbHkgb3BlbiBhbiBpc3N1ZSB3aXRoIHRoZSB0YWcgIl9lbmhhbmNlbWVudF8iLgpEb24ndCBmb3JnZXQgdG8gZ2l2ZSB0aGUgcHJvamVjdCBhIHN0YXIg4q2QISBUaGFua3MgYWdhaW4hCgoxLiBGb3JrIHRoZSBQcm9qZWN0CjIuIENyZWF0ZSB5b3VyIEZlYXR1cmUgQnJhbmNoIChgZ2l0IGNoZWNrb3V0IC1iIGZlYXR1cmUvQW1hemluZ0ZlYXR1cmVgKQozLiBDb21taXQgeW91ciBDaGFuZ2VzIChgZ2l0IGNvbW1pdCAtbSAnQWRkIHNvbWUgQW1hemluZ0ZlYXR1cmUnYCkKNC4gUHVzaCB0byB0aGUgQnJhbmNoIChgZ2l0IHB1c2ggb3JpZ2luIGZlYXR1cmUvQW1hemluZ0ZlYXR1cmVgKQo1LiBPcGVuIGEgUHVsbCBSZXF1ZXN0 readmeEtag: '"777c0a701335d13f60aac05c53a398da802a314c"' readmeLastModified: Wed, 28 Sep 2022 20:48:35 GMT repositoryId: 516092537 description: >- Built on top of Symfony 6, API Platform enables you to build a rich, JSON-LD-powered, hypermedia API... pretty much instantly! created: '2022-07-20T18:25:23Z' updated: '2022-07-26T11:33:33Z' language: PHP archived: false stars: 1 watchers: 1 forks: 0 owner: bmykyta logo: https://avatars.githubusercontent.com/u/31631183?v=4 repoEtag: '"18c50f5cc925bdf9e1c7559928339bb6ea1615334ac2199be20574152538768c"' repoLastModified: Tue, 26 Jul 2022 11:33:33 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/fugerit-org/yaml-doc-tool v3: true id: 97f4d8f8067f40e02afeeb61b95be57e repositoryMetadata: base64Readme: >- IyB5YW1sLWRvYy10b29sCgo+IOKaoO+4jyAqKldhcm5pbmc6KiogQXMgb2YgMjAyNS0wNy0yNyB0aGlzIHJlcG9zaXRvcnkgaXMgYXJjaGl2ZWQgYW5kIHN1YnN0aXR1dGVkIGJ5IFtvcGVuYXBpLWRvYy10b29sXShodHRwczovL2dpdGh1Yi5jb20vZnVnZXJpdC1vcmcvb3BlbmFwaS1kb2MtdG9vbCkgKHNlZSBbZnVnZXJpdC1vcmcvb3BlbmFwaS1kb2MtdG9vbCMxXShodHRwczovL2dpdGh1Yi5jb20vZnVnZXJpdC1vcmcvb3BlbmFwaS1kb2MtdG9vbC9pc3N1ZXMvMSkpLgoKVG9vbCBmb3IgYXV0byBkb2N1bWVudGF0aW9uIG9mIHlhbWwgLyBvcGVuYXBpCgpbIVtLZWVwIGEgQ2hhbmdlbG9nIHYxLjEuMCBiYWRnZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jaGFuZ2Vsb2ctS2VlcCUyMGElMjBDaGFuZ2Vsb2clMjB2MS4xLjAtJTIzRTA1NzM1KV0oQ0hBTkdFTE9HLm1kKSAKWyFbTWF2ZW4gQ2VudHJhbF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9tYXZlbi1jZW50cmFsL3Yvb3JnLmZ1Z2VyaXQuamF2YS95YW1sLWRvYy10b29sLnN2ZyldKGh0dHBzOi8vbXZucmVwb3NpdG9yeS5jb20vYXJ0aWZhY3Qvb3JnLmZ1Z2VyaXQuamF2YS95YW1sLWRvYy10b29sKQpbIVtsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtQXBhY2hlJTIwTGljZW5zZSUyMDIuMC10ZWFsLnN2ZyldKGh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvQXBhY2hlLTIuMCkKWyFbY29kZSBvZiBjb25kdWN0XShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NvbmR1Y3QtQ29udHJpYnV0b3IlMjBDb3ZlbmFudC1wdXJwbGUuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2Z1Z2VyaXQtb3JnL2ZqLXVuaXZlcnNlL2Jsb2IvbWFpbi9DT0RFX09GX0NPTkRVQ1QubWQpClshW1F1YWxpdHkgR2F0ZSBTdGF0dXNdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PWZ1Z2VyaXQtb3JnX3lhbWwtZG9jLXRvb2wmbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPWZ1Z2VyaXQtb3JnX3lhbWwtZG9jLXRvb2wpClshW0NvdmVyYWdlXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1mdWdlcml0LW9yZ195YW1sLWRvYy10b29sJm1ldHJpYz1jb3ZlcmFnZSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPWZ1Z2VyaXQtb3JnX3lhbWwtZG9jLXRvb2wpCgpbIVtKYXZhIHJ1bnRpbWUgdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9ydW4lMjBvbi1qYXZhJTIwOCstJTIzMTEzMzY2LnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289b3BlbmpkayZsb2dvQ29sb3I9d2hpdGUpXShodHRwczovL3VuaXZlcnNlLmZ1Z2VyaXQub3JnL3NyYy9kb2NzL3ZlcnNpb25zL2phdmExMS5odG1sKQpbIVtKYXZhIGJ1aWxkIHZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvYnVpbGQlMjBvbi1qYXZhJTIwMTErLSUyM0VEOEIwMC5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPW9wZW5qZGsmbG9nb0NvbG9yPXdoaXRlKV0oaHR0cHM6Ly91bml2ZXJzZS5mdWdlcml0Lm9yZy9zcmMvZG9jcy92ZXJzaW9ucy9qYXZhMTEuaHRtbCkKWyFbQXBhY2hlIE1hdmVuXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0FwYWNoZSUyME1hdmVuLTMuOS4wKy1DNzFBMzY/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPUFwYWNoZSUyME1hdmVuJmxvZ29Db2xvcj13aGl0ZSldKGh0dHBzOi8vdW5pdmVyc2UuZnVnZXJpdC5vcmcvc3JjL2RvY3MvdmVyc2lvbnMvbWF2ZW4zXzkuaHRtbCkKCiMjIFF1aWNrc3RhcnQgKGJ1aWxkKQoKYGBgCm12biBjbGVhbiBpbnN0YWxsIC1QIHNpbmdsZXBhY2thZ2UJCQpgYGAKCiMjIFF1aWNrc3RhcnQgKG1vZGU6IHNpbmdsZSk6CgpgYGAKamF2YSAtamFyIGRpc3QteWFtbC1kb2MtdG9vbC0qLmphciAtLWlucHV0LXlhbWwgW3BhdGgtdG8tb3BlbmFwaV0gXAoJCQkJCQkJCQkJLS1vdXRwdXQtZmlsZSBbb3V0cHV0LWZpbGVdIFwKCQkJCQkJCQkJCS0tbGFuZ3VhZ2UgW2xhbmd1YWdlXSBcCgkJCQkJCQkJCQktLWxhYmVscy1vdmVycmlkZSBbcGF0aC10by1sYWJlbHMtcHJvcGVydGllc10KYGBgCgkJCQkJCQkJCQkKKipleGFtcGxlIDoqKiAgCgpgYGAKamF2YSAtamFyIGRpc3QteWFtbC1kb2MtdG9vbC0qLmphciAtLWlucHV0LXlhbWwgc2FtcGxlLnlhbWwgLS1vdXRwdXQtZmlsZSBzYW1wbGUucGRmIC0tbGFuZ3VhZ2UgaXQKYGBgCgoqKm91dHB1dC1maWxlKiogICAKY3VycmVudGx5IHN1cHBvcnRlZCBleHRlbnNpb25zIDogcGRmLCB4bHN4LCB4bWwsIGZvCgoqKmxhbmd1YWdlKiogIApjdXJyZW50bHkgc3VwcG9ydGVkIGxhbmd1YWdlcyA6IGl0LCBlbgoKKipsYWJlbHMtb3ZlcnJpZGVzKiogICAKcGF0aCB0byBhbHRlcm5hdGUgbGFiZWxzIHByb3BlcnRpZXMKY3VycmVudGx5IHN1cHBvcnRlZCBsYWJlbHMgYXJlIGF2YWlsYWJsZSBpbiA6IHNyYy9tYWluL3Jlc291cmNlcy9sYW5nL2xhYmVsLnByb3BlcnRpZXMKCioqbWF2ZW4gcGx1Z2luKiogIApBIFtNYXZlbiBQbHVnaW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9mdWdlcml0LW9yZy95YW1sLWRvYy1tYXZlbi1wbHVnaW4pIGlzIGFsc28gYXZhaWxhYmxlLgoKCiMjIFF1aWNrc3RhcnQgKG1vZGU6IGNvbmZpZyk6CgpgYGAKamF2YSAtamFyIGRpc3QteWFtbC1kb2MtdG9vbC0qLmphciAJCS0tbW9kZSBbY29uZmlnXSBcCgkJCQkJCQkJCQktLWNvbmZpZy1wYXRoIFtwYXRoLXRvLWNvbmZpZ10gXAoJCQkJCQkJCQkJLS1pZC1jYXRhbG9nIFtpZC1jYXRhbG9nLWluLWNvbmZpZ10KYGBgCgoqKmNvbmZpZy1wYXRoKiogICAKcGF0aCB0byBjb25maWd1cmF0aW9uIChzZWUgW3NhbXBsZV0oc3JjL3Rlc3QvcmVzb3VyY2VzL3lhbWwtZG9jLWNvbmZpZy54bWwpKQoKKippZC1jYXRhbG9nKiogICAKaWQgb2YgdGhlIGNhdGFsb2cgaW4gY29uZmlnIHRvIHVzZQoKCiMjIFF1aWNrc3RhcnQgKG1vZGU6IGNoZWNrLW1vZGVsKToKCk5vZGUgOiB0aGUgdHlwZSB0byBjaGVjayBuZWVkIHRvIGJlIGluIGNsYXNzcGF0aC4KCmBgYApqYXZhIC1qYXIgZGlzdC15YW1sLWRvYy10b29sLSouamFyIAkJLS1tb2RlIFtjaGVjay1tb2RlbF0gXAoJCQkJCQkJCQkJLS1pbnB1dC15YW1sIFtwYXRoLXRvLW9wZW5hcGldIFwKCQkJCQkJCQkJCS0tb3V0cHV0LWZpbGUgW291dHB1dC1maWxlXSBcCgkJCQkJCQkJCQktLWNoZWNrLXR5cGUgW2phdmEgdHlwZSB0byBjaGVja10gXAoJCQkJCQkJCQkJLS1jaGVjay1zY2hlbWEgW29wZW5hcGkgc2NoZW1hIHRvIGNoZWNrXQoJCQkJCQkJCQkJLS12ZXJzaW9uIFttb2RlbC12ZXJzaW9uXQpgYGAKCiMjIE1vZGVsIHZlcnNpb24gaW5mbyAocGFyYW0gInZlcnNpb24iKQoKIyMjIFZlcnNpb24gMCAtIGxlZ2FjeSB2ZXJzaW9uICh1cCB0byAxLjAuMCBleGNsdWRlZCkKCiMjIyBWZXJzaW9uIDEKCmhhbmRsaW5nIGhhbmRsaW5nIG9mIGNvbnN0cmFpbnRzIFtPcGVuQVBJIGRhdGEgdHlwZXNdKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vZGF0YS1tb2RlbHMvZGF0YS10eXBlcy8pCiAtIG1pbkxlbmd0aAogLSBtYXhMZW5ndGgKIC0gbWluaW11bQogLSBtYXhpbXVtIAo= readmeEtag: '"1eba063cba25f38294917f952a955f64241ac1f9"' readmeLastModified: Sun, 27 Jul 2025 18:27:31 GMT repositoryId: 471110771 description: Documentation tool for yaml (openapi) created: '2022-03-17T19:11:05Z' updated: '2025-07-27T18:28:10Z' language: Java archived: true stars: 1 watchers: 1 forks: 0 owner: fugerit-org logo: https://avatars.githubusercontent.com/u/37816284?v=4 license: Apache-2.0 repoEtag: '"023c505382462f2f843328e4ffa656eba1aea7b68e2c0e2e4912c435892c62dd"' repoLastModified: Sun, 27 Jul 2025 18:28:10 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/rnhc1000/apichallenge v3: true id: 0efd5e5ff88b9176ef9fdcfa6b55c13e repositoryMetadata: base64Readme: >- IyMgX0FQSSBBZ2dyZWdhdG9yIENoYWxsZW5nZV8gPGJyIC8+CkdvYWw6IEFub3RoZXIgdGFrZS1ob21lIGNoYWxsZW5nZSAKCiMjIF9UYWJsZSBvZiBjb250ZW50c18KCi0gW19PdmVydmlld19dKCNvdmVydmlldykKLSBbX1JlcXVpcmVtZW50c19dKCNyZXF1aXJlbWVudHMpCi0gW19Qcm9qZWN0IFN0cnVjdHVyZV9dKCNyZXF1aXJlbWVudHMpCi0gW19Ib3d0byBCdWlsZCBhbmQgUnVuX10oI3JlcXVpcmVtZW50cykKLSBbX1NjcmVlbnNob3RfXSgjc2NyZWVuc2hvdCkKLSBbX0xpbmtzX10oLi4uKQotIFtfQnVpbHQgd2l0aF9dKCNidWlsdC13aXRoKQotIFtfQ29kZSBTbmlwcGV0X10oI3JlcXVpcmVtZW50cykKLSBbX0NvbnRpbnVlZCBkZXZlbG9wbWVudF9dKCNjb250aW51ZWQtZGV2ZWxvcG1lbnQpCi0gW19Vc2VmdWwgcmVzb3VyY2VzX10oI3VzZWZ1bC1yZXNvdXJjZXMpCi0gW19BdXRob3JfXSgjcmVxdWlyZW1lbnRzKQotIFtfUG9ydGZvbGlvX10oI3JlcXVpcmVtZW50cykKCiMjIF9PdmVydmlld18KClRoaXMgY2hhbGxlbmdlIGRlbGl2ZXJzIGFuIGFwcGxpY2F0aW9uIHRoYXQgY29uc3VtZXMgYW4gZXh0ZXJuYWwgQVBJIGVuZHBvaW50LgpJdCBpcyBhIFJGQzgyODggY29tcGxpYW50IHBhZ2luYXRlZCBlbmRwb2ludCB0byBiZSBjb25zdW1lZCwgcmV0dXJuaW5nIGEgbGlzdCBvZiBjb250YWN0cwphbmQgdGhlIHBhZ2luYXRpb24gaW5mb3JtYXRpb24gdmlhIGxpbmsgaGVhZGVyIGFuZCBzaG91bGQgYmUgYnVpbHQgYWNjb3JkaW5nIHRvIHNvbWUKc3BlY2lmaWMgcmVxdWlyZW1lbnRzLgoKCiMjIF9SZXF1aXJlbWVudHNfCgotIFRoZSByZXNwb25zZSBmcm9tIHRoZSBlbmRwb2ludCBzaG91bGQgbWF0Y2ggdGhlIHNjaGVtYSBwcm92aWRlZC4KPGJyIC8+Ci0gUHJvamVjdCBzdHJ1Y3R1cmUgYW5kIG9yZ2FuaXphdGlvbiBvZiB5b3VyIGNvZGU6IHdlIHdhbnQgdG8gc2VlIGlmIHlvdXIgY29kZSBmb2xsb3dzIGdvb2QgcGF0dGVybnMsIGFuZCBzZXBhcmF0ZXMKY29uY2VybnMgYmV0d2VlbiBjb250cm9sbGVycywgc2VydmljZXMsIGFuZCByZXBvc2l0b3JpZXMuCjxiciAvPgotIENhcGFjaXR5IHRvIGZvbGxvdyBnb29kIHByaW5jaXBsZXMgbGlrZSBDbGVhbiBDb2RlIGFuZCBTT0xJRC4KPGJyIC8+Ci0gQ2FwYWNpdHkgdG8gdXNlIGRlc2lnbiBwYXR0ZXJucyBhbmQga25vd2xlZGdlIGFib3V0IHRoZSBjaG9zZW4gb25lKHMpIHVzYWdlLgo8YnIgLz4KLSBBYmlsaXR5IHRvIGNvbnN1bWUgb3RoZXIgQVBJcyBhbmQgY29ycmVjdGx5IGhhbmRsZSBwYWdpbmF0aW9uLgo8YnIgLz4KLSBLbm93bGVkZ2UgYWJvdXQgdGhlIHRvb2xzIHlvdSBkZWNpZGVkIHRvIHVzZS4KPGJyIC8+Ci0gQWJpbGl0eSB0byBjcmVhdGUgc21hcnQgdGVzdHMgdG8gZW5oYW5jZSB5b3VyIGNvZGUgcXVhbGl0eS4KCkJhc2VkIG9uIHRoZXNlLCB0aGUgYXBwIGNvbnN1bWVzIHNlcnZpY2VzIG9mIGFub3RoZXIgYXBpIC0+IDxhIGhyZWY9Imh0dHBzOi8vay1tZXNzYWdlcy1hcGkuaGVyb2t1YXBwLmNvbSIgdGFyZ2V0PSJfYmxhbmsiPkNvbnRhY3RzIEFQSTwvYT4uCgo8YnIgLz4KWW91IGNhbiBwbGF5IHdpdGggdGhlIGFwcCwgdmlzaXRpbmcgdGhlIGxpbmsgYmVsb3cgYWZ0ZXIgeW91IGJ1aWxkIHRoZSBhcHAKYWNjb3JkaW5nIHRvIHRoZSBpbnN0cnVjdGlvbnMuCgpodHRwczovL2NoYWxsZW5nZS5mZXJyZWlyYXMuZGV2LmJyL3N3YWdnZXItdWkvaW5kZXguaHRtbAo8YnIgLz4KVGhlIGFwcCBoYXMgYmVlbiBjb2RlZCB1c2luZyBKYXZhIDE3LCBTcHJpbmcgQm9vdCAzLjMuNCwgR3JhZGxlLCBKYXZhZG9jLCBTcHJpbmcgU2VjdXJpdHksIFNwcmluZyBKUEEsIApTcHJpbmcgV2ViZmx1eCwgT3BlbkFQSSwgSlVuaXQsIE1vY2tpdG8sIE15U1FMLCBEb2NrZXIgYW5kIG90aGVycy48YnIgLz4KVGFrZSBhIGxvb2sgYXQgdGhlIHZpZGVvIGJlbG93IHRvIHVuZGVyc3RhbmQgaG93IEkgZmFjZWQgdGhlIGNoYWxsZW5nZS4KPGJyIC8+CjxhIGhyZWY9Imh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9bnVMOElLQ3NJVG8iIHRhcmdldD0iX2JsYW5rIj5TaG9ydCBWaWRlbzwvYT4KPGhyIC8+CgojIyBfUHJvamVjdCBTdHJ1Y3R1cmVfCi0gZG9jcwogICAtIGphdmFkb2NzCi0gc3JjCiAgICAtIG1haW4KICAgIC0gamF2YQogICAgICAgIC0gYnIuZGV2LmZlcnJlaXJhcy5jaGFsbGVuZ2UKICAgICAgICAgICAgLSBjb250cmFjdHMKICAgICAgICAgICAgLSBjb25maWcKICAgICAgICAgICAgLSBjb250cm9sbGVyCiAgICAgICAgICAgICAgLSBoYW5kbGVycyAKICAgICAgICAgICAgLSBkdG8KICAgICAgICAgICAgLSBlbnRpdHkKICAgICAgICAgICAgLSBlbnVtcwogICAgICAgICAgICAtIHJlcG9zaXRvcnkKICAgICAgICAgICAgLSBzZXJ2aWNlcwogICAgICAgICAgICAgIC0gZXhjZXB0aW9ucwogICAgLSByZXNvdXJjZXMKICAgICAgICAtIGNlcnRzCiAgICAtIHRlc3QKICAgICAgLSBqYXZhCiAgICAgICAgLSBici5kZXYuZmVycmVpcmFzLmNoYWxsZW5nZQogICAgICAgICAgLSBjb250cm9sbGVyCiAgICAgICAgICAtIHNlcnZpY2UKLQoKIyMgX0hvd3RvIEJ1aWxkIGFuZCBSdW5fCgogIGBgYAogIC0gTXlTUUwgRGF0YWJhc2U6IGh0dHA6Ly8xMjcuMC4wLjE6MzMwNjpjaGFsbGVuZ2UKICAtIGNyZWRlbnRpYWxzIGF2YWlsYWJsZSBhdCBjbGFzc3BhdGg6ZGIucHJvcGVydGllcwogIC0gcHJvZmlsZSBhY3RpdmU6IGRldiBvciBwcm9kCiAgLSBwcm9kdWN0aW9uIHNvY2tldDoxMjcuMC4wLjE6ODA5NwogIC0gdHdlYWsgYSBmZXcga25vYnMgdG8gZ2V0IGl0IHVwIGFuZCBydW5uaW5nIGFjY29yZGluZyB0byB0aGUgaW5zdHJ1Y3Rpb25zCiAgICBwcm92aWRlZCBhdCBjbGFzc3BhdGg6ZG9ja2VyQnVpbGQuc2ggb3IganVzdCBpbiBjYXNlIHlvdSB3YW50IHJvIHJ1bgogICAgbG9jYWxseSBnbyB0byB7UFdEfS9jaGFsbGVuZ2UgYW5kIHJ1biAuL2dyYWRsZXcgcnVuLCBvciBjaGVjayB0aGUgdmlkZW8KICAgIHVybCBzaG93biBiZWxvdy4KCmBgYAoKIyMgX1NjcmVlbnNob3RfCgpbIVtdKC4vam9ic2l0eS5wbmcpXSgpCgojIyBfTGlua3NfCgotIExpdmUgU2l0ZSAtPiBDbGljayBoZXJlIDpjbG91ZDogIDxhIGhyZWY9Imh0dHBzOi8vY2hhbGxlbmdlLmZlcnJlaXJhcy5kZXYuYnIvc3dhZ2dlci11aS9pbmRleC5odG1sIiB0YXJnZXQ9Il9ibGFuayI+QVBJIEFnZ3JlZ2F0b3I8L2E+CgojIyBfQnVpbHQgd2l0aF8KClshW015IFNraWxsc10oaHR0cHM6Ly9za2lsbGljb25zLmRldi9pY29ucz9pPWphdmEsc3ByaW5nLG15c3FsLGdyYWRsZSxkb2NrZXIscmVkaGF0LGF3cyxpZGVhLGdpdCxnaXRodWIsKV0oaHR0cHM6Ly9za2lsbGljb25zLmRldikKCiMjIF9Db2RlIFNuaXBwZXRfCgpgYGBqYXZhCmltcG9ydCBqYXZhLnV0aWwuTGlzdDsKCi8qKgogKiAKICogQGF1dGhvciByaWNhcmRvQGZlcnJlaXJhcy5kZXYuYnIKICogQHZlcnNpb24gMS4xLjEwLjIzLjAxCiAqIEBzaW5jZSAxLjAKICoKICovCgpAUmVzdENvbnRyb2xsZXIKQFJlcXVlc3RNYXBwaW5nKCJhcGkvdjEiKQpwdWJsaWMgY2xhc3MgQ29udGFjdENvbnRyb2xsZXIgewoKICAgIHByaXZhdGUgZmluYWwgQ29udGFjdFNlcnZpY2UgY29udGFjdFNlcnZpY2U7CgogICAgcHVibGljIENvbnRhY3RDb250cm9sbGVyKENvbnRhY3RTZXJ2aWNlIGNvbnRhY3RTZXJ2aWNlKSB7CiAgICAgICAgdGhpcy5jb250YWN0U2VydmljZSA9IGNvbnRhY3RTZXJ2aWNlOwogICAgfQoKICAgIEBPcGVyYXRpb24oc3VtbWFyeSA9ICJGZXRjaCAyMCBjb250YWN0cyBwZXIgcGFnZSIpCiAgICBAQXBpUmVzcG9uc2VzKHZhbHVlID0gewogICAgICAgICAgICBAQXBpUmVzcG9uc2UocmVzcG9uc2VDb2RlID0gIjIwMCIsIGRlc2NyaXB0aW9uID0gIkdldCB1cCB0byAyMCBjb250YWN0cyBwZXIgcGFnZS4iLAogICAgICAgICAgICAgICAgICAgIGNvbnRlbnQgPSB7QENvbnRlbnQobWVkaWFUeXBlID0gImFwcGxpY2F0aW9uL2pzb24iLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgc2NoZW1hID0gQFNjaGVtYSAoaW1wbGVtZW50YXRpb24gPSBSZXNwb25zZUNvbnRhY3RzRHRvLmNsYXNzKSl9KSwKICAgICAgICAgICAgQEFwaVJlc3BvbnNlKHJlc3BvbnNlQ29kZSA9ICI0MDAiLCBkZXNjcmlwdGlvbiA9ICJCYWQgUmVxdWVzdCIsCiAgICAgICAgICAgICAgICAgICAgY29udGVudCA9IHtAQ29udGVudChtZWRpYVR5cGUgPSAiYXBwbGljYXRpb24vanNvbiIpfSksCiAgICAgICAgICAgIEBBcGlSZXNwb25zZShyZXNwb25zZUNvZGUgPSAiNDAxIiwgZGVzY3JpcHRpb24gPSAiQWNjZXNzIERlbmllZCIsCiAgICAgICAgICAgICAgICAgICAgY29udGVudCA9IHtAQ29udGVudChtZWRpYVR5cGUgPSAiYXBwbGljYXRpb24vanNvbiIpfSksCiAgICAgICAgICAgIEBBcGlSZXNwb25zZShyZXNwb25zZUNvZGUgPSAiNDAzIiwgZGVzY3JpcHRpb24gPSAiRk9SQklEREVOIiwKICAgICAgICAgICAgICAgICAgICBjb250ZW50ID0ge0BDb250ZW50KG1lZGlhVHlwZSA9ICJhcHBsaWNhdGlvbi9qc29uIil9KSwKICAgICAgICAgICAgQEFwaVJlc3BvbnNlKHJlc3BvbnNlQ29kZSA9ICI0MDQiLCBkZXNjcmlwdGlvbiA9ICJSZXNvdXJjZSBub3QgZm91bmQhIiwKICAgICAgICAgICAgICAgICAgICBjb250ZW50ID0ge0BDb250ZW50KG1lZGlhVHlwZSA9ICJhcHBsaWNhdGlvbi9qc29uIil9KQogICAgfSkKICAgIEBSZXNwb25zZVN0YXR1cyhIdHRwU3RhdHVzLk9LKQogICAgQEdldE1hcHBpbmcoIi9jb250YWN0cyIpCiAgICBAUHJlQXV0aG9yaXplKCJoYXNBdXRob3JpdHkoJ1NDT1BFX1JPTEVfQURNSU4nKSIpCiAgICBwdWJsaWMgTW9ubzxSZXNwb25zZUNvbnRhY3RzRHRvPiBnZXRDb250YWN0cygKICAgICAgICAgICAgQFJlcXVlc3RQYXJhbShkZWZhdWx0VmFsdWUgPSAiMSIpIGludCBwYWdlLAogICAgICAgICAgICBAUmVxdWVzdFBhcmFtKGRlZmF1bHRWYWx1ZSA9ICIyMCIpIGludCBzaXplCiAgICApIHsKCiAgICAgICAgcmV0dXJuIGNvbnRhY3RTZXJ2aWNlLm1ha2VBcGlSZXF1ZXN0KCkKICAgICAgICAgICAgICAgIC5ib2R5VG9Nb25vKFJlc3BvbnNlQ29udGFjdHNEdG8uY2xhc3MpOwoKICAgIH0KfQoKCmBgYCAKCiMjIF9Db250aW51ZWQgZGV2ZWxvcG1lbnRfCgotIFVuaXQgVGVzdHMgLSBkb25lCi0gU3Vic2NyaWJlciBBdXRoZW50aWNhdGlvbiAtIGRvbmUKLSBTcHJpbmcgSldULU9BdXRoMiAtIGRvbmUKLSBSZWNvcmRzIFBhZ2luYXRpb24gLSBkb25lCgojIyMgX1VzZWZ1bCByZXNvdXJjZXNfCgotIFtodHRwczovL3NwcmluZy5pby9dIEF3ZXNvbWUgSmF2YSBmcmFtZXdvcmshLgotIFtodHRwczovL3N0YXJ0LnNwcmluZy5pby9dICBIYW5keSBzdGFydHVwIHRvb2wuCi0gW2h0dHBzOi8vbXZucmVwb3NpdG9yeS5jb20vXQoKIyMgX0F1dGhvcl8KPGEgaHJlZj0ibWFpbHRvOnJpY2FyZG9AZmVycmVpcmFzLmRldi5iciI+UmljYXJkbyBGZXJyZWlyYTwvYT4KCiMjIC0gX1BvcnRmb2xpb18KPGEgaHJlZj0iaHR0cHM6Ly93d3cuZmVycmVpcmFzLmRldi5iciIgdGFyZ2V0PSJfYmxhbmsiPk15IFBvcnRmb2xpby4uLjwvYT4KCg== readmeEtag: '"5a881b0cf9afb5e132b61fbeab6ab16b6ea17123"' readmeLastModified: Sat, 02 Nov 2024 23:57:04 GMT repositoryId: 877578929 description: API Aggregator created: '2024-10-23T22:35:19Z' updated: '2024-11-02T23:57:08Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: rnhc1000 logo: https://avatars.githubusercontent.com/u/60550902?v=4 repoEtag: '"2df1459b0050e590ff53a0f3626689e84d875909b9f9db05a321cf270c5941db"' repoLastModified: Sat, 02 Nov 2024 23:57:08 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/camptocamp/geoserver-rest-openapi v3: true repositoryMetadata: base64Readme: >- IyBHZW9TZXJ2ZXIgSmF2YSBSRVNUIGNsaWVudAoKIyMgUnVuIGludGVncmF0aW9uIHRlc3RzOgoKYGBgCm12biB2ZXJpZnkgLVBkb2NrZXIKYGBgCgpXaWxsIGZpcmUgdXAgdGhlIGdlb3NlcnZlciBkb2NrZXIgY29udGFpbmVyIGF0IHByZS1pbnRlZ3JhdGlvbi10ZXN0cywgcnVuIHRoZSB0ZXN0cyBhdAppbnRlZ3JhdGlvbi10ZXN0LCBhbmQgc2h1dCB0aGUgY29udGFpbmVyIGRvd24gYXQgcG9zdC1pbnRlZ3JhdGlvbi10ZXN0cy4KCiMjIFJ1biBmcm9tIElERToKCkhhdmUgdGhlIEdlb1NlcnZlciBkb2NrZXIgY29udGFpbmVyIHJ1bm5pbmcgYmVmb3JlIGV4ZWN1dGluZyB0aGUgdGVzdHMgZnJvbSB0aGUgSURFOgoKYGBgCmRvY2tlciBydW4gLWl0IC0tcm0gLS1uYW1lIGdzdGVzdHMgLXA4MDgwOjgwODAgb3NjYXJmb250cy9nZW9zZXJ2ZXI6Mi4xNS40CmBgYAoKUGFzcyB0aGUgZm9sbG93aW5nIGVudmlyb25tZW50IHZhcmlhYmxlIHRvIHRoZSB0ZXN0IHJ1biBjb25maWd1cmF0aW9uIG9uIHRoZSBJREU6CgpgYGAKLURnZW9zZXJ2ZXJfYXBpX3VybD1odHRwOi8vbG9jYWxob3N0OjgwODAvZ2Vvc2VydmVyL3Jlc3QKYGBgCgo= readmeEtag: '"152af872e3611bddd0e7b7a6478a499726385d23"' readmeLastModified: Thu, 18 Feb 2021 01:33:52 GMT repositoryId: 335228268 description: >- GeoServer REST client based on curated OpenAPI 3 document, using openapi-codegen to generate the client, and customized code to simplify it. created: '2021-02-02T09:03:44Z' updated: '2024-11-04T13:14:36Z' language: Java archived: false stars: 1 watchers: 22 forks: 5 owner: camptocamp logo: https://avatars.githubusercontent.com/u/28109?v=4 repoEtag: '"a92f0aad0bf6e01088f825e9cf2c6ddc7016b1e30bd06002973700f4d6180181"' repoLastModified: Mon, 04 Nov 2024 13:14:36 GMT foundInMaster: true category: - Code Generators - Server Implementations id: f1ffebfb85dc3b1b4b7d576a63d13259 - source: openapi3 tags repository: https://github.com/retr0h/osapi v3: true id: 33c7e6d9bdd60e5a6138cb80e6611e5a repositoryMetadata: base64Readme: >- WyFbcmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvcmVsZWFzZS9yZXRyMGgvb3NhcGkuc3ZnP3N0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL2dpdGh1Yi5jb20vcmV0cjBoL29zYXBpL3JlbGVhc2VzL2xhdGVzdCkKWyFbY29kZWNvdl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9jb2RlY292L2MvZ2l0aHViL3JldHIwaC9vc2FwaT90b2tlbj1ORjBUODZCMUVQJnN0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvcmV0cjBoL29zYXBpKQpbIVtnbyByZXBvcnQgY2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vcmV0cjBoL29zYXBpP3N0eWxlPWZvci10aGUtYmFkZ2UpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20vcmV0cjBoL29zYXBpKQpbIVtsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2xpY2Vuc2UtTUlULWJyaWdodGdyZWVuLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oTElDRU5TRSkKWyFbYnVpbGRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3JldHIwaC9vc2FwaS9nby55bWw/c3R5bGU9Zm9yLXRoZS1iYWRnZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9yZXRyMGgvb3NhcGkvYWN0aW9ucy93b3JrZmxvd3MvZ28ueW1sKQpbIVtwb3dlcmVkIGJ5XShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3Bvd2VyZWQlMjBieS1nb3JlbGVhc2VyLWdyZWVuLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9naXRodWIuY29tL2dvcmVsZWFzZXIpClshW2NvbnZlbnRpb25hbCBjb21taXRzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0NvbnZlbnRpb25hbCUyMENvbW1pdHMtMS4wLjAteWVsbG93LnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlKV0oaHR0cHM6Ly9jb252ZW50aW9uYWxjb21taXRzLm9yZykKIVtvcGVuYXBpIGluaXRpYXRpdmVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvb3BlbmFwaWluaXRpYXRpdmUtJTIzMDAwMDAwLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289b3BlbmFwaWluaXRpYXRpdmUmbG9nb0NvbG9yPXdoaXRlKQohW0xpbnV4XShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpbnV4LUZDQzYyND9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289bGludXgmbG9nb0NvbG9yPWJsYWNrKQohW2dpdEh1YiBjb21taXQgYWN0aXZpdHldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2NvbW1pdC1hY3Rpdml0eS9tL3JldHIwaC9vc2FwaT9zdHlsZT1mb3ItdGhlLWJhZGdlKQoKIyBPUyBBUEkKCjxpbWcgc3JjPSJhc3NldC9sb2dvLnBuZyIgYWxpZ249ImxlZnQiIC8+CgoqKE9TQVBJIC/LiG/KinPJkcuQcGkvIC0gT2gtc2FoLXBlZSkqIEEgQ1JVRCBBUEkgZm9yIG1hbmFnaW5nIExpbnV4IHN5c3RlbXMuCgpUaGlzIHByb2plY3QgcHJvdmlkZXMgYmFzaWMgbWFuYWdlbWVudCBjYXBhYmlsaXRpZXMgdG8gTGludXggc3lzdGVtcywgZW5hYmxpbmcKdGhlbSB0byBiZSB1c2VkIGFzIGFwcGxpYW5jZXMuCgo8YnIgY2xlYXI9ImxlZnQiLz4KCiMjIERvY3VtZW50YXRpb24KCltHZXR0aW5nIFN0YXJ0ZWRdW10gfCBbQVBJXVtdIHwgW1VzYWdlXVtdCgpbR2V0dGluZyBTdGFydGVkXTogaHR0cHM6Ly9yZXRyMGguZ2l0aHViLmlvL29zYXBpLwpbQVBJXTogaHR0cHM6Ly9yZXRyMGguZ2l0aHViLmlvL29zYXBpL2NhdGVnb3J5L2FwaQpbVXNhZ2VdOiBodHRwczovL3JldHIwaC5naXRodWIuaW8vb3NhcGkvc2lkZWJhci91c2FnZS8KCiMjIExpY2Vuc2UKClRoZSBbTUlUXVtdIExpY2Vuc2UuCgpbTUlUXTogTElDRU5TRQo= readmeEtag: '"b8fa3957016e79007d28949115520de0b924c23f"' readmeLastModified: Wed, 18 Dec 2024 20:21:46 GMT repositoryId: 843181662 description: A CRUD API for managing Linux systems. created: '2024-08-16T00:37:30Z' updated: '2025-06-17T01:22:42Z' language: Go archived: false stars: 1 watchers: 1 forks: 0 owner: retr0h logo: https://avatars.githubusercontent.com/u/9895?v=4 license: MIT repoEtag: '"d14543a0d9df4a6430fbb5eef075139186268838155ad2e54271f8ff34fc7fc3"' repoLastModified: Tue, 17 Jun 2025 01:22:42 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/vivek-26/koa-openapi3-router v3: true repositoryMetadata: base64Readme: IyMjIFRoaXMgbGlicmFyeSBpcyBpbiBhbHBoYSBzdGFnZQ== readmeEtag: '"3bd885ad877c6caf1065abbe1da2e9418a4411dd"' readmeLastModified: Sat, 17 Mar 2018 05:57:23 GMT repositoryId: 123946298 description: Koa (v2) Router, based on OpenAPI 3 Doc. created: '2018-03-05T16:22:45Z' updated: '2018-04-09T12:18:40Z' language: JavaScript archived: false stars: 1 watchers: 2 forks: 0 owner: vivek-26 logo: https://avatars.githubusercontent.com/u/16881423?v=4 license: MIT repoEtag: '"f24c436d7b002d51ba27c53da7df26362464f44924c9919ae7d049546a2e8ff2"' repoLastModified: Mon, 09 Apr 2018 12:18:40 GMT foundInMaster: true category: - Documentation - Testing - Parsers id: 60f2c3139be2a10f5462502ef6f862d5 - source: openapi3 tags repository: https://github.com/opencars/docs v3: true repositoryMetadata: base64Readme: >- # OpenCars API

API для публічної інформації про транспортні засоби України

## Table of Contents

* [Servers](#servers)
* [Paths](#paths)
  - [`GET` /operations](#op-get-operations) 
  - [`GET` /registrations](#op-get-registrations) 
  - [`GET` /registrations/{code}](#op-get-registrations-code) 
  - [`GET` /wanted/vehicles](#op-get-wanted-vehicles) 
  - [`GET` /wanted/revisions](#op-get-wanted-revisions) 
  - [`GET` /wanted/revisions/{id}](#op-get-wanted-revisions-id) 
  - [`GET` /wanted/revisions/stats](#op-get-wanted-revisions-stats) 
* [Schemas](#schemas)
  - Error
  - Operation
  - Registration
  - WantedVehicle
  - WantedRevision
  - WantedRevisionStatMonth


<a id="servers" />
## Servers

<table>
  <thead>
    <tr>
      <th>URL</th>
      <th>Description</th>
    <tr>
  </thead>
  <tbody>
    <tr>
      <td><a href="https://api.opencars.app/api/v1/" target="_blank">https://api.opencars.app/api/v1/</a></td>
      <td></td>
    </tr>
  </tbody>
</table>

<a name="security"></a>
## Security

<table class="table">
  <thead class="table__head">
    <tr class="table__head__row">
      <th class="table__head__cell">Type</th>
      <th class="table__head__cell">In</th>
      <th class="table__head__cell">Name</th>
      <th class="table__head__cell">Scheme</th>
      <th class="table__head__cell">Format</th>
      <th class="table__head__cell">Description</th>
    </tr>
  </thead>
  <tbody class="table__body">
    <tr class="table__body__row">
      <td class="table__body__cell">apiKey</td>
      <td class="table__body__cell">header</td>
      <td class="table__body__cell">Api-Key</td>
      <td class="table__body__cell"></td>
      <td class="table__body__cell"></td>
      <td class="table__body__cell"></td>
    </tr>

  </tbody>
</table>

## Paths


### `GET` /operations
<a id="op-get-operations" />

> Інформація про операції за реєстраційним номером





#### Query parameters

##### &#9655; number

Реєстраційний номер


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>number  <strong>(required)</strong></td>
        <td>
          string
        </td>
        <td>query</td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### &#9655; limit

Кількість записів


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>limit </td>
        <td>
          integer
        </td>
        <td>query</td>
        <td>Кількість записів</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### &#9655; order

Порядок сортування


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>order </td>
        <td>
          string
        </td>
        <td>query</td>
        <td>Порядок сортування</td>
        <td><code>asc</code>, <code>desc</code></td>
      </tr>
  </tbody>
</table>






#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>Response</td>
        <td>
          array(object)
        </td>
        <td></td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.brand</td>
        <td>
          string
        </td>
        <td>Марка</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.model</td>
        <td>
          string
        </td>
        <td>Модель</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.year</td>
        <td>
          integer
        </td>
        <td>Рік випуску</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.date</td>
        <td>
          string
        </td>
        <td>Дата проведення операції</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.registration</td>
        <td>
          string
        </td>
        <td>Вид реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.registration_code</td>
        <td>
          integer
        </td>
        <td>Код виду реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.fuel</td>
        <td>
          string
        </td>
        <td>Тип пального</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.capacity</td>
        <td>
          number
        </td>
        <td>Об'єм двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.color</td>
        <td>
          string
        </td>
        <td>Колір транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.kind</td>
        <td>
          string
        </td>
        <td>Тип транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.body</td>
        <td>
          string
        </td>
        <td>Тип кузову</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.purpose</td>
        <td>
          string
        </td>
        <td>Призначення</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.own_weight</td>
        <td>
          integer
        </td>
        <td>Маса без навантаження</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.total_weight</td>
        <td>
          integer
        </td>
        <td>Повна маса</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.reg_addr_koatuu</td>
        <td>
          string
        </td>
        <td>Адреса реєстрацiї у форматі КОАТУУ</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.dep_code</td>
        <td>
          integer
        </td>
        <td>Код департаменту</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.dep</td>
        <td>
          string
        </td>
        <td>Назва департаменту</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.person</td>
        <td>
          string
        </td>
        <td>Назва департаменту</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
[
  {
    "number": "АА9359РС",
    "brand": "TESLA",
    "model": "MODEL X",
    "year": 2016,
    "date": "2016-10-13",
    "registration": "РЕЄСТРАЦIЯ ТЗ ПРИВЕЗЕНОГО З-ЗА КОРДОНУ ПО ВМД",
    "registration_code": 70,
    "fuel": "ЕЛЕКТРО",
    "capacity": null,
    "color": "ЧОРНИЙ",
    "kind": "ЛЕГКОВИЙ",
    "body": "УНІВЕРСАЛ-B",
    "purpose": "ЗАГАЛЬНИЙ",
    "own_weight": 2485,
    "total_weight": 3021,
    "reg_addr_koatuu": "8036600000",
    "dep_code": 8044,
    "dep": "Центр 8044",
    "person": "Юридична особа"
  }
]
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

### `GET` /registrations
<a id="op-get-registrations" />

> Інформація про реєстрації транспортного засобу





#### Query parameters

##### &#9655; number

Реєстраційний номер


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>number </td>
        <td>
          string
        </td>
        <td>query</td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### &#9655; vin

VIN


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>vin </td>
        <td>
          string
        </td>
        <td>query</td>
        <td>VIN</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### &#9655; code

Номер свідотства про реєстрацію


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>code </td>
        <td>
          string
        </td>
        <td>query</td>
        <td>Номер свідотства про реєстрацію</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>






#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>Response</td>
        <td>
          array(object)
        </td>
        <td></td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.code</td>
        <td>
          string
        </td>
        <td>Номер свідоцтва про реєстрацію транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.vin</td>
        <td>
          string
        </td>
        <td>VIN-код</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.brand</td>
        <td>
          string
        </td>
        <td>Марка</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.model</td>
        <td>
          string
        </td>
        <td>Модель</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.capacity</td>
        <td>
          number
        </td>
        <td>Об'єм двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.rank_category</td>
        <td>
          string
        </td>
        <td>Категорія транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.num_seating</td>
        <td>
          integer
        </td>
        <td>Кількість стоячих місць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.num_standing</td>
        <td>
          integer
        </td>
        <td>Кількість сидячих місць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.color</td>
        <td>
          string
        </td>
        <td>Колір</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.date</td>
        <td>
          string
        </td>
        <td>Дата реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.first_reg_date</td>
        <td>
          string
        </td>
        <td>Дата першої реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.fuel</td>
        <td>
          string
        </td>
        <td>Тип пального</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.kind</td>
        <td>
          string
        </td>
        <td>Тип</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.own_weight</td>
        <td>
          number
        </td>
        <td>Маса без навантаження</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.total_weight</td>
        <td>
          number
        </td>
        <td>Повна маса</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.year</td>
        <td>
          integer
        </td>
        <td>Рік випуску</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
[
  {
    "number": "АА9359РС",
    "code": "СХН484154",
    "vin": "5YJXCCE40GF010543",
    "brand": "TESLA",
    "model": "MODEL X",
    "capacity": null,
    "rank_category": "B",
    "num_seating": null,
    "num_standing": 7,
    "color": "ЧОРНИЙ",
    "date": "2019-06-05",
    "first_reg_date": "2016-10-13",
    "fuel": "ЕЛЕКТРО",
    "kind": "ЛЕГКОВИЙ УНІВЕРСАЛ-B",
    "own_weight": 2485,
    "total_weight": 3021,
    "year": 2016
  }
]
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

### `GET` /registrations/{code}
<a id="op-get-registrations-code" />

> Реєстрація транспортного засобу за номером свідотства



#### Path parameters

##### &#9655; code

Номер свідотства про реєстрацію


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>code  <strong>(required)</strong></td>
        <td>
          string
        </td>
        <td>path</td>
        <td>Номер свідотства про реєстрацію</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>








#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>code</td>
        <td>
          string
        </td>
        <td>Номер свідоцтва про реєстрацію транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>vin</td>
        <td>
          string
        </td>
        <td>VIN-код</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>brand</td>
        <td>
          string
        </td>
        <td>Марка</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>model</td>
        <td>
          string
        </td>
        <td>Модель</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>capacity</td>
        <td>
          number
        </td>
        <td>Об'єм двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>rank_category</td>
        <td>
          string
        </td>
        <td>Категорія транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>num_seating</td>
        <td>
          integer
        </td>
        <td>Кількість стоячих місць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>num_standing</td>
        <td>
          integer
        </td>
        <td>Кількість сидячих місць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>color</td>
        <td>
          string
        </td>
        <td>Колір</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>date</td>
        <td>
          string
        </td>
        <td>Дата реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>first_reg_date</td>
        <td>
          string
        </td>
        <td>Дата першої реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>fuel</td>
        <td>
          string
        </td>
        <td>Тип пального</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>kind</td>
        <td>
          string
        </td>
        <td>Тип</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>own_weight</td>
        <td>
          number
        </td>
        <td>Маса без навантаження</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>total_weight</td>
        <td>
          number
        </td>
        <td>Повна маса</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>year</td>
        <td>
          integer
        </td>
        <td>Рік випуску</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "number": "АА9359РС",
  "code": "СХН484154",
  "vin": "5YJXCCE40GF010543",
  "brand": "TESLA",
  "model": "MODEL X",
  "capacity": null,
  "rank_category": "B",
  "num_seating": null,
  "num_standing": 7,
  "color": "ЧОРНИЙ",
  "date": "2019-06-05",
  "first_reg_date": "2016-10-13",
  "fuel": "ЕЛЕКТРО",
  "kind": "ЛЕГКОВИЙ УНІВЕРСАЛ-B",
  "own_weight": 2485,
  "total_weight": 3021,
  "year": 2016
}
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

### `GET` /wanted/vehicles
<a id="op-get-wanted-vehicles" />

> Інформація про викрадені транспортні засоби





#### Query parameters

##### &#9655; number

Реєстраційний номер


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>number </td>
        <td>
          string
        </td>
        <td>query</td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### &#9655; vin

VIN-код


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>vin </td>
        <td>
          string
        </td>
        <td>query</td>
        <td>VIN-код</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### &#9655; revision

Номер ревізії


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>revision </td>
        <td>
          integer
        </td>
        <td>query</td>
        <td>Номер ревізії</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>






#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>Response</td>
        <td>
          array(object)
        </td>
        <td></td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.id</td>
        <td>
          string
        </td>
        <td>Унікальний номер викраденого транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.brand</td>
        <td>
          string
        </td>
        <td>Інформація про транспортний засіб</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.color</td>
        <td>
          string
        </td>
        <td>Колір транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.body_number</td>
        <td>
          string
        </td>
        <td>Номер кузова</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.chassis_number</td>
        <td>
          string
        </td>
        <td>Номер шасі</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.engine_number</td>
        <td>
          string
        </td>
        <td>Номeр двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.ovd</td>
        <td>
          string
        </td>
        <td>Орган, що вніс дані у реєстр</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.kind</td>
        <td>
          string
        </td>
        <td>Тип транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.status</td>
        <td>
          string
        </td>
        <td>Статус розшуку транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.theft_date</td>
        <td>
          string
        </td>
        <td>Час викрадення транспортного засобу у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.insert_date</td>
        <td>
          string
        </td>
        <td>Час внесення інформації у рееєстр у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
[
  {
    "id": "00163195157556102859",
    "brand": "CHEVROLET - EXPRESS",
    "color": "ЗЕЛЕНИЙ",
    "number": "AA4686EH",
    "body_number": "1GBSHDC44D1126468",
    "chassis_number": "Ідентифікаційний номер транспортного засобу",
    "engine_number": null,
    "ovd": "ГЕНЕРАЛЬНА ПРОКУРАТУРА УКРАЇНИ",
    "kind": "ЛЕГКОВИЙ",
    "status": "stolen",
    "theft_date": "2016-11-14T00:00:00Z",
    "insert_date": "2016-11-14T14:19:35Z"
  }
]
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

### `GET` /wanted/revisions
<a id="op-get-wanted-revisions" />

> Інформація про оновлення реєстру викрадених транспортних засобів





#### Query parameters

##### &#9655; limit

Кількість елементів у відповіді


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>limit </td>
        <td>
          integer
        </td>
        <td>query</td>
        <td>Кількість елементів у відповіді</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>






#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>Response</td>
        <td>
          array(object)
        </td>
        <td></td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.id</td>
        <td>
          string
        </td>
        <td>Унікальний код оновлення реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.name</td>
        <td>
          string
        </td>
        <td>Назва ресурсу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.url</td>
        <td>
          string
        </td>
        <td>Посилання на оновлений ресурс</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.file_hash_sum</td>
        <td>
          string
        </td>
        <td>Хеш сума контенту файлу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.removed</td>
        <td>
          number
        </td>
        <td>Кількість видалених записів з реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.added</td>
        <td>
          number
        </td>
        <td>Кількість доданих записів до реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.created_at</td>
        <td>
          string
        </td>
        <td>Час оновлення даних у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
[
  {
    "id": "29092018_2",
    "name": "MVSWantedTransport_1.json",
    "url": "https://data.gov.ua/dataset/9b0e87e0-eaa3-4f14-9547-03d61b70abb6/resource/06e65b06-3120-4713-8003-7905a83f95f5/revision/29092018_2",
    "file_hash_sum": "f4ae97fb62c7c9b91d6b950c79deb716",
    "removed": 27019,
    "added": 43365,
    "created_at": "2018-09-29T22:33:33Z"
  }
]
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

### `GET` /wanted/revisions/{id}
<a id="op-get-wanted-revisions-id" />

> Відомості про оновлення реєстру за унікальним кодом



#### Path parameters

##### &#9655; id

Унікальний код оновлення реєстру


<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>In</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>id  <strong>(required)</strong></td>
        <td>
          string
        </td>
        <td>path</td>
        <td>Унікальний код оновлення реєстру</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>








#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>id</td>
        <td>
          string
        </td>
        <td>Унікальний код оновлення реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>name</td>
        <td>
          string
        </td>
        <td>Назва ресурсу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>url</td>
        <td>
          string
        </td>
        <td>Посилання на оновлений ресурс</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>file_hash_sum</td>
        <td>
          string
        </td>
        <td>Хеш сума контенту файлу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>removed</td>
        <td>
          number
        </td>
        <td>Кількість видалених записів з реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>added</td>
        <td>
          number
        </td>
        <td>Кількість доданих записів до реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>created_at</td>
        <td>
          string
        </td>
        <td>Час оновлення даних у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "id": "29092018_2",
  "name": "MVSWantedTransport_1.json",
  "url": "https://data.gov.ua/dataset/9b0e87e0-eaa3-4f14-9547-03d61b70abb6/resource/06e65b06-3120-4713-8003-7905a83f95f5/revision/29092018_2",
  "file_hash_sum": "f4ae97fb62c7c9b91d6b950c79deb716",
  "removed": 27019,
  "added": 43365,
  "created_at": "2018-09-29T22:33:33Z"
}
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

### `GET` /wanted/revisions/stats
<a id="op-get-wanted-revisions-stats" />

> Статистика оновлення реєстру за останні 12 місяців









#### Responses


##### ▶ 200 - OK

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>Response</td>
        <td>
          array(object)
        </td>
        <td></td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.added</td>
        <td>
          integer
        </td>
        <td>Кількість доданих записів за місяць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.removed</td>
        <td>
          integer
        </td>
        <td>Кількість видалених записів за місяць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.month</td>
        <td>
          integer
        </td>
        <td>Місяць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>Response.year</td>
        <td>
          integer
        </td>
        <td>Рік</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
[
  {
    "added": 241,
    "removed": 195,
    "month": 2,
    "year": 2020
  }
]
```
##### ▶ 400 - Помилковий запит

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 401 - Потрібна Авторизація

###### Headers
_No headers specified_

##### ▶ 405 - Помилковий метод

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```
##### ▶ 500 - Помилка

###### Headers
_No headers specified_

###### application/json



<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>


##### Example _(generated)_

```json
{
  "error": "string"
}
```

#### Tags

<div class="tags">
  <div class="tags__tag"></div>
</div>
</div>

## Schemas

<a id="" />

#### Error

<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>error</td>
        <td>
          string
        </td>
        <td>Опис помилки</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>

##### Example _(generated)_

```json
{
  "error": "string"
}
```
<a id="" />

#### Operation

<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>brand</td>
        <td>
          string
        </td>
        <td>Марка</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>model</td>
        <td>
          string
        </td>
        <td>Модель</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>year</td>
        <td>
          integer
        </td>
        <td>Рік випуску</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>date</td>
        <td>
          string
        </td>
        <td>Дата проведення операції</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>registration</td>
        <td>
          string
        </td>
        <td>Вид реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>registration_code</td>
        <td>
          integer
        </td>
        <td>Код виду реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>fuel</td>
        <td>
          string
        </td>
        <td>Тип пального</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>capacity</td>
        <td>
          number
        </td>
        <td>Об'єм двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>color</td>
        <td>
          string
        </td>
        <td>Колір транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>kind</td>
        <td>
          string
        </td>
        <td>Тип транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>body</td>
        <td>
          string
        </td>
        <td>Тип кузову</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>purpose</td>
        <td>
          string
        </td>
        <td>Призначення</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>own_weight</td>
        <td>
          integer
        </td>
        <td>Маса без навантаження</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>total_weight</td>
        <td>
          integer
        </td>
        <td>Повна маса</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>reg_addr_koatuu</td>
        <td>
          string
        </td>
        <td>Адреса реєстрацiї у форматі КОАТУУ</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>dep_code</td>
        <td>
          integer
        </td>
        <td>Код департаменту</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>dep</td>
        <td>
          string
        </td>
        <td>Назва департаменту</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>person</td>
        <td>
          string
        </td>
        <td>Назва департаменту</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>

##### Example _(generated)_

```json
{
  "number": "АА9359РС",
  "brand": "TESLA",
  "model": "MODEL X",
  "year": 2016,
  "date": "2016-10-13",
  "registration": "РЕЄСТРАЦIЯ ТЗ ПРИВЕЗЕНОГО З-ЗА КОРДОНУ ПО ВМД",
  "registration_code": 70,
  "fuel": "ЕЛЕКТРО",
  "capacity": null,
  "color": "ЧОРНИЙ",
  "kind": "ЛЕГКОВИЙ",
  "body": "УНІВЕРСАЛ-B",
  "purpose": "ЗАГАЛЬНИЙ",
  "own_weight": 2485,
  "total_weight": 3021,
  "reg_addr_koatuu": "8036600000",
  "dep_code": 8044,
  "dep": "Центр 8044",
  "person": "Юридична особа"
}
```
<a id="" />

#### Registration

<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>code</td>
        <td>
          string
        </td>
        <td>Номер свідоцтва про реєстрацію транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>vin</td>
        <td>
          string
        </td>
        <td>VIN-код</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>brand</td>
        <td>
          string
        </td>
        <td>Марка</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>model</td>
        <td>
          string
        </td>
        <td>Модель</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>capacity</td>
        <td>
          number
        </td>
        <td>Об'єм двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>rank_category</td>
        <td>
          string
        </td>
        <td>Категорія транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>num_seating</td>
        <td>
          integer
        </td>
        <td>Кількість стоячих місць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>num_standing</td>
        <td>
          integer
        </td>
        <td>Кількість сидячих місць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>color</td>
        <td>
          string
        </td>
        <td>Колір</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>date</td>
        <td>
          string
        </td>
        <td>Дата реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>first_reg_date</td>
        <td>
          string
        </td>
        <td>Дата першої реєстрації</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>fuel</td>
        <td>
          string
        </td>
        <td>Тип пального</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>kind</td>
        <td>
          string
        </td>
        <td>Тип</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>own_weight</td>
        <td>
          number
        </td>
        <td>Маса без навантаження</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>total_weight</td>
        <td>
          number
        </td>
        <td>Повна маса</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>year</td>
        <td>
          integer
        </td>
        <td>Рік випуску</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>

##### Example _(generated)_

```json
{
  "number": "АА9359РС",
  "code": "СХН484154",
  "vin": "5YJXCCE40GF010543",
  "brand": "TESLA",
  "model": "MODEL X",
  "capacity": null,
  "rank_category": "B",
  "num_seating": null,
  "num_standing": 7,
  "color": "ЧОРНИЙ",
  "date": "2019-06-05",
  "first_reg_date": "2016-10-13",
  "fuel": "ЕЛЕКТРО",
  "kind": "ЛЕГКОВИЙ УНІВЕРСАЛ-B",
  "own_weight": 2485,
  "total_weight": 3021,
  "year": 2016
}
```
<a id="" />

#### WantedVehicle

<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>id</td>
        <td>
          string
        </td>
        <td>Унікальний номер викраденого транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>brand</td>
        <td>
          string
        </td>
        <td>Інформація про транспортний засіб</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>color</td>
        <td>
          string
        </td>
        <td>Колір транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>number</td>
        <td>
          string
        </td>
        <td>Реєстраційний номер</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>body_number</td>
        <td>
          string
        </td>
        <td>Номер кузова</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>chassis_number</td>
        <td>
          string
        </td>
        <td>Номер шасі</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>engine_number</td>
        <td>
          string
        </td>
        <td>Номeр двигуна</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>ovd</td>
        <td>
          string
        </td>
        <td>Орган, що вніс дані у реєстр</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>kind</td>
        <td>
          string
        </td>
        <td>Тип транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>status</td>
        <td>
          string
        </td>
        <td>Статус розшуку транспортного засобу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>theft_date</td>
        <td>
          string
        </td>
        <td>Час викрадення транспортного засобу у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>insert_date</td>
        <td>
          string
        </td>
        <td>Час внесення інформації у рееєстр у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>

##### Example _(generated)_

```json
{
  "id": "00163195157556102859",
  "brand": "CHEVROLET - EXPRESS",
  "color": "ЗЕЛЕНИЙ",
  "number": "AA4686EH",
  "body_number": "1GBSHDC44D1126468",
  "chassis_number": "Ідентифікаційний номер транспортного засобу",
  "engine_number": null,
  "ovd": "ГЕНЕРАЛЬНА ПРОКУРАТУРА УКРАЇНИ",
  "kind": "ЛЕГКОВИЙ",
  "status": "stolen",
  "theft_date": "2016-11-14T00:00:00Z",
  "insert_date": "2016-11-14T14:19:35Z"
}
```
<a id="" />

#### WantedRevision

<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>id</td>
        <td>
          string
        </td>
        <td>Унікальний код оновлення реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>name</td>
        <td>
          string
        </td>
        <td>Назва ресурсу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>url</td>
        <td>
          string
        </td>
        <td>Посилання на оновлений ресурс</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>file_hash_sum</td>
        <td>
          string
        </td>
        <td>Хеш сума контенту файлу</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>removed</td>
        <td>
          number
        </td>
        <td>Кількість видалених записів з реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>added</td>
        <td>
          number
        </td>
        <td>Кількість доданих записів до реєстру</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>created_at</td>
        <td>
          string
        </td>
        <td>Час оновлення даних у форматі ISO 8601</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>

##### Example _(generated)_

```json
{
  "id": "29092018_2",
  "name": "MVSWantedTransport_1.json",
  "url": "https://data.gov.ua/dataset/9b0e87e0-eaa3-4f14-9547-03d61b70abb6/resource/06e65b06-3120-4713-8003-7905a83f95f5/revision/29092018_2",
  "file_hash_sum": "f4ae97fb62c7c9b91d6b950c79deb716",
  "removed": 27019,
  "added": 43365,
  "created_at": "2018-09-29T22:33:33Z"
}
```
<a id="" />

#### WantedRevisionStatMonth

<table>
  <thead>
    <tr>
      <th>Name</th>
      <th>Type</th>
      <th>Description</th>
      <th>Accepted values</th>
    </tr>
  </thead>
  <tbody>
      <tr>
        <td>added</td>
        <td>
          integer
        </td>
        <td>Кількість доданих записів за місяць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>removed</td>
        <td>
          integer
        </td>
        <td>Кількість видалених записів за місяць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>month</td>
        <td>
          integer
        </td>
        <td>Місяць</td>
        <td><em>Any</em></td>
      </tr>
      <tr>
        <td>year</td>
        <td>
          integer
        </td>
        <td>Рік</td>
        <td><em>Any</em></td>
      </tr>
  </tbody>
</table>

##### Example _(generated)_

```json
{
  "added": 241,
  "removed": 195,
  "month": 2,
  "year": 2020
}
```
 readmeEtag: '"31f14fb6936244abf7c067c4c62e2fd19a617689"' readmeLastModified: Sat, 31 Dec 2022 17:16:16 GMT repositoryId: 232517461 description: ':memo: Documentation for the OpenCars API ' created: '2020-01-08T08:39:27Z' updated: '2022-12-09T11:36:26Z' language: HTML archived: false stars: 1 watchers: 0 forks: 0 owner: opencars logo: https://avatars.githubusercontent.com/u/48658620?v=4 repoEtag: '"4178048b8dc94a838228d46777aa0ebf95a2cbce216935167f745c08eede03c9"' repoLastModified: Fri, 09 Dec 2022 11:36:26 GMT foundInMaster: true category: Learning id: af4731daa82dd0f855d601ad5531103d - source: openapi3 tags repository: https://github.com/codecentric/reedelk-openapi v3: true repositoryMetadata: base64Readme: >- IyBSZWVkZWxrIE9wZW5BUEkgdjMgc2VyaWFsaXplciAvIGRlc2VyaWFsaXplcgoKWyFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWNlbnNlLUFwYWNoZSUyMDIuMC1ibHVlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9yZWVkZWxrL3JlZWRlbGstcnVudGltZS9ibG9iL21hc3Rlci9MSUNFTlNFKQpbIVtUd2l0dGVyXShodHRwczovL2ltZy5zaGllbGRzLmlvL3R3aXR0ZXIvZm9sbG93L3JlZWRlbGsuc3ZnP3N0eWxlPXNvY2lhbCZsYWJlbD1Gb2xsb3cpXShodHRwczovL3R3aXR0ZXIuY29tL2ludGVudC9mb2xsb3c/c2NyZWVuX25hbWU9cmVlZGVsaykKClRoaXMgcHJvamVjdCByZWZlcnMgdG8gdGhlIFtPcGVuQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uKSBwcm9qZWN0IGZyb20gdGhlIFtPcGVuQVBJIGluaXRpYXRpdmVdKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pLgoKIyMgT3ZlcnZpZXcKVGhlIFJlZWRlbGsgT3BlbkFQSSB2My54IHNlcmlhbGl6ZXIvZGVzZXJpYWxpemVyIGxpYnJhcnkgaXMgYSBsaWdodHdlaWdodCBsaWJyYXJ5IHRvIHNlcmlhbGl6ZS9kZXNlcmlhbGl6ZSBPcGVuQVBJIAp2My54IG1vZGVsIHRvL2Zyb20gSlNPTiBhbmQgdG8vZnJvbSBZQU1MLiAKClRoZSBsaWJyYXJ5IHVzZXMgb25seSBbSlNPTi1qYXZhIChvcmcuanNvbildKGh0dHBzOi8vZ2l0aHViLmNvbS9zdGxlYXJ5L0pTT04tamF2YSkgYW5kIApbU25ha2VZQU1MXShodHRwczovL2dpdGh1Yi5jb20vYXNvbW92L3NuYWtleWFtbCkgYXMgZGVwZW5kZW5jaWVzIG1ha2luZyBpdCB0aGUgcGVyZmVjdCBjaG9pY2UgaWYgeW91IGFyZSBqdXN0IApsb29raW5nIGZvciBhIHZlcnkgbGlnaHR3ZWlnaHQgc29sdXRpb24gdG8gc2VyaWFsaXplL2Rlc2VyaWFsaXplIE9wZW5BUEkgdjMueCBkZWZpbml0aW9ucy4KCiAKIyMgRmVhdHVyZXMKCi0gU2VyaWFsaXplIE9wZW5BUEkgdjMueCBtb2RlbCB0byBKU09OCi0gU2VyaWFsaXplIE9wZW5BUEkgdjMueCBtb2RlbCB0byBZQU1MCi0gRGVzZXJpYWxpemUgT3BlbkFQSSB2My54IG1vZGVsIGZyb20gSlNPTgotIERlc2VyaWFsaXplIE9wZW5BUEkgdjMueCBtb2RlbCBmcm9tIFlBTUwKCiMjIE1hdmVuCkFkZCB0aGUgUmVlZGVsayBSZXBvc2l0b3J5IHRvIHlvdXIgcG9tLnhtbDoKCmBgYHhtbAo8cmVwb3NpdG9yaWVzPgogICAgPHJlcG9zaXRvcnk+CiAgICAgICAgPGlkPnJlZWRlbGstcmVwb3NpdG9yeTwvaWQ+CiAgICAgICAgPG5hbWU+UmVlZGVsayBSZXBvc2l0b3J5PC9uYW1lPgogICAgICAgIDx1cmw+aHR0cDovL3JlcG9zaXRvcnkucmVlZGVsay5jb20vcmVsZWFzZS88L3VybD4KICAgIDwvcmVwb3NpdG9yeT4KPC9yZXBvc2l0b3JpZXM+CmBgYAoKQWRkIHRoZSBmb2xsb3dpbmcgZGVwZW5kZW5jeSB0byB5b3VyIHBvbS54bWwgZmlsZToKYGBgeG1sCjxkZXBlbmRlbmN5PgogICAgPGdyb3VwSWQ+Y29tLnJlZWRlbGs8L2dyb3VwSWQ+CiAgICA8YXJ0aWZhY3RJZD5yZWVkZWxrLW9wZW5hcGk8L2FydGlmYWN0SWQ+CiAgICA8dmVyc2lvbj5YLlkuWjwvdmVyc2lvbj4KPC9kZXBlbmRlbmN5PgpgYGAKCiMjIFVzYWdlCiMjIyBTZXJpYWxpemUKIyMjIyBUbyBKU09OIG9yIFlBTUw6CmBgYGphdmEKSW5mb09iamVjdCBpbmZvTW9kZWwgPSBuZXcgSW5mb09iamVjdCgpOwppbmZvTW9kZWwuc2V0RGVzY3JpcHRpb24oIlRoaXMgaXMgYSBzYW1wbGUgQVBJLiIpOwppbmZvTW9kZWwuc2V0VmVyc2lvbigiMS4wLjIiKTsKCk9wZW5BcGlPYmplY3Qgb3BlbkFwaU1vZGVsID0gbmV3IE9wZW5BcGlPYmplY3QoKTsKb3BlbkFwaU1vZGVsLnNldEluZm8oaW5mbyk7CgovLyB0byBKU09OIHN0cmluZwpTdHJpbmcgb3BlbkFwaUFzSnNvbiA9IE9wZW5BcGkudG9Kc29uKG9wZW5BcGlNb2RlbCk7CgovLyB0byBZQU1MIHN0cmluZwpTdHJpbmcgb3BlbkFwaUFzWWFtbCA9IE9wZW5BcGkudG9ZYW1sKG9wZW5BcGlNb2RlbCk7CmBgYAoKIyMjIERlc2VyaWFsaXplCiMjIyMgRnJvbSBKU09OIG9yIFlBTUw6CmBgYGphdmEKLy8gZnJvbSBKU09OIHN0cmluZwpTdHJpbmcgb3BlbkFwaUFzSnNvbiA9ICJ7Im9wZW5hcGkiOiAiMy4wLjMiLCJpbmZvIjogeyJ0aXRsZSI6ICJBUEkiLCJ2ZXJzaW9uIjogInYxIiB9fSI7Ck9wZW5BcGlPYmplY3Qgb3BlbkFwaU1vZGVsID0gT3BlbkFwaS5mcm9tKG9wZW5BcGlBc0pzb24pOwpJbmZvT2JqZWN0IGluZm9Nb2RlbCA9IGFjdHVhbC5nZXRJbmZvKCk7Ci4uLgoKLy8gZnJvbSBZQU1MIHN0cmluZwpTdHJpbmcgb3BlbkFwaUFzWWFtbCA9ICJvcGVuYXBpOiAzLjAuMAogICAgICAgICAgICAgICAgICAgICAgICBpbmZvOgogICAgICAgICAgICAgICAgICAgICAgICAgIHZlcnNpb246IDEuMC4yCiAgICAgICAgICAgICAgICAgICAgICAgICAgdGl0bGU6IFBldHN0b3JlIEFQSSI7Ck9wZW5BcGlPYmplY3Qgb3BlbkFwaU1vZGVsID0gT3BlbkFwaS5mcm9tKG9wZW5BcGlBc1lhbWwpOwpJbmZvT2JqZWN0IGluZm9Nb2RlbCA9IGFjdHVhbC5nZXRJbmZvKCk7Ci4uLgpgYGAKCiMjIENvbnRyaWJ1dGUKLSBJZiB5b3UgZmluZCBhIGJ1ZyBpbiBSZWVkZWxrIE9wZW5BUEksIHBsZWFzZSBbZmlsZSBhIGJ1ZyByZXBvcnRdKGh0dHBzOi8vZ2l0aHViLmNvbS9yZWVkZWxrL3JlZWRlbGstb3BlbmFwaS9pc3N1ZXMpLgotIElmIHlvdSB3YW50IHRvIGRpc2N1c3MgUmVlZGVsayBPcGVuQVBJLCBzdWdnZXN0IG5ldyBmZWF0dXJlcywgeW91IGhhdmUgaXNzdWVzIGdldHRpbmcgc3RhcnRlZCB3aXRoIHRoZSBsaWJyYXJ5IG9yIGp1c3Qgc2F5ICdIaScsIGxldCB1cyBrbm93IGluIHRoZSBbUmVlZGVsayBTbGFjayBEZXZlbG9wZXJzIENvbW11bml0eV0oaHR0cHM6Ly9qb2luLnNsYWNrLmNvbS90L3JlZWRlbGsvc2hhcmVkX2ludml0ZS96dC1mejN3eDU2Zi1YRHlsWHBxWEVSb29LZU90cmhkWnVnKSBbIVtKb2luIFJlZWRlbGsgU2xhY2sgRGV2ZWxvcGVycyBDb21tdW5pdHldKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvU2xhY2stSm9pbiUyMHRoZSUyMGNoYXQlMjByb29tLWJsdWUpXShodHRwczovL2pvaW4uc2xhY2suY29tL3QvcmVlZGVsay9zaGFyZWRfaW52aXRlL3p0LWZ6M3d4NTZmLVhEeWxYcHFYRVJvb0tlT3RyaGRadWcpLgo= readmeEtag: '"cadc7d835b79219df54f865956f2a1b354c8589b"' readmeLastModified: Wed, 21 Apr 2021 22:17:37 GMT repositoryId: 283149249 description: OpenAPI v3 JSON/YAML serializer/deserializer created: '2020-07-28T08:25:55Z' updated: '2021-04-21T22:17:45Z' language: Java archived: false stars: 1 watchers: 0 forks: 2 owner: codecentric logo: https://avatars.githubusercontent.com/u/1009716?v=4 license: Apache-2.0 repoEtag: '"dbbd2dabf428397671ff9edc63f218cd26dd8ce90c1f994f9e900fb23503eb13"' repoLastModified: Wed, 21 Apr 2021 22:17:45 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 92684132322c194c03d78e0d6adfae5c - source: openapi3 tags repository: https://github.com/thestarivore/ramen_website v3: true id: c87aea0345c6f845a324d50697731cf9 repositoryMetadata: base64Readme: >- IyByYW1lbl93ZWJzaXRlClJhbWVuIEFzc29jaWF0aW9uIFdlYnNpdGUgLSBIeXBlcm1lZGlhIGNvdXJzZQo= readmeEtag: '"7d63b0fe870c62fc871d03a58bdd362d74ba2de9"' readmeLastModified: Mon, 12 Oct 2020 10:35:49 GMT repositoryId: 265383330 description: Ramen Association Website - Hypermedia course created: '2020-05-19T22:31:39Z' updated: '2023-08-24T21:57:54Z' language: HTML archived: false stars: 1 watchers: 2 forks: 0 owner: thestarivore logo: https://avatars.githubusercontent.com/u/22894195?v=4 repoEtag: '"2abaf57950576fb3bc481d241e40d66d02fe28ab461511fce6e8c49002145aa9"' repoLastModified: Thu, 24 Aug 2023 21:57:54 GMT category: Server foundInMaster: true - source: openapi3 tags repository: https://github.com/halilozkan/eurovision-song-contest v3: true repositoryMetadata: base64Readme: >- PSBFdXJvdmlzaW9uLXNvbmctY29udGVzdAo6aWNvbnM6IGZvbnQKCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvdmVydC54LTMuOS4xLXB1cnBsZS5zdmdbbGluaz0iaHR0cHM6Ly92ZXJ0eC5pbyJdCgpUaGlzIGV4ZXJjaXNlIGFwcGxpY2F0aW9uIGltcGxlbWVudHMgYSBzbWFsbCBzeXN0ZW0gdG8gY291bnQgdGhlIHJlc3VsdHMgb2YgdGhlIEV1cm92aXNpb24gU29uZyBDb250ZXN0LgoKKiBBbiBPcGVuQVBJIHNwZWNpZmljYXRpb24gZGVzaWduZWQgdG8gcHJvdmlkZSBBcGkgRGVzaWduIEZpcnN0IGFwcHJvYWNoLgoqIFZlcnQueCBXZWIgQVBJIENvbnRyYWN0IHVzZWQgdG8gZXhwb3NlIGVuZHBvaW50IHBhdGhzIGFuZCB2YWxpZGF0ZSB0aGVtIHdpdGggT3BlbkFQSSBzcGVjLgoqIEFuIEhUVFAgU2VydmVyIGV4cG9zZSBhbiBlbmRwb2ludCB3aXRoIHJlbGV2YW50IHBhdGhzIGFsbGlnbmVkIHdpdGggT3BlbkFQSSBzcGVjIGFuZCBjb250cm9sbGVkIGJ5IFdlYiBBUEkgQ29udHJhY3QuCiogRW5kcG9pbnQgcGF0aHMgY29ubmVjdGVkIHRvIEhUVFAgU2VydmVyIHJvdXRlciBidWlsdCB3aXRoIE9wZW5BUEkgZGlyZWN0bHkgdG8gRXZlbnRCdXMgc2VydmljZXMuCiogQW4gRXZlbnRCdXMgYWxsb3dlZCBtYW55IHBhcnRzIG9mIGFwcGxpY2F0aW9uIHRvIGNvbW11bmljYXRlIGVhY2ggb3RoZXIuCiogUmVhY3RpdmUgcGF0dGVybiBpbXBsZW1lbnRlZCB0byBkZXZlbG9wIHRyYW5zYWN0aW9uIHNlcnZpY2VzLCBtYWlubHkgUnhKYXZhMiwgYXN5Y2hyb25vdXMgYW5kIEphdmEgc3RyZWFtcy4KKiBUd28gdmVydGljbGVzIGRldmVsb3BlZDogYSB2b3RlIHRyYW5zYWN0aW9uIHNlcnZpY2VzIHZlcnRpY2xlKEphdmEpIGFuZCBhIG1ldHJpY3MgcmVwb3J0ZXIoS290bGluKQoqIFRoYW5rcyB0byBWZXJ0LnggcG9seWdsb3Qgc3VwcG9ydCBKYXZhIGFuZCBLb3RsaW4gdXNlZCB3aXRoIHZlcnRpY2xlIGRldmVsb3BtZW50cwoqIEEgbWV0cmljIHJlcG9ydGVyIGRlc2lnbmVkIHRvIHB1c2ggbWV0cmljcyB0byB0aGUgZXZlbnRidXMgdG9waWMKKiBBIGRhc2hib2FyZCBkZXNpZ25lZCB0byBjaGVjayB2b3RlIHJlc3VsdCB0aHJvdWdoIGEgd2Vic2Nva2V0IGFuZCByZWFkIGxhdGVzdCB2b3RlIHJlc3VsdCBtZXRyaWNzIGZyb20gdGhlIGV2ZW50YnVzIHRvcGljCiogUmVhY3RpdmUgYW5kIG5vbiBibG9ja2luZyBQb3N0Z3JlU1FMIGNsaWVudCB1c2VkIHRvIGhhbmRsZSBkYXRhYmFzZSBjb25uZWN0aW9ucywgcXVlcmllcyBhbmQgdHJhbnNhY3Rpb25zLgoqIEFwcGxpY2F0aW9uIGRvY2tlcml6ZWQgZnJvbSBhZG9wdG9wZW5qZGsvb3Blbmpkazgtb3Blbmo5OmFscGluZS1qcmUgaW1hZ2UuCiogQSBQb3N0Z3JlU1FMIGRhdGFiYXNlICBwcmVmZmVyZWQgdG8gc3RvcmUgYXBwbGljYXRpb24gZGF0YSB0aHJvdWdoIHJlYWN0aXZlIGRhdGFiYXNlIGRyaXZlci4gRHVyaW5nIGluc3RhbGxhdGlvbiBhbGwgcmVxdWlyZWQgZGF0YWJhc2UsIHNjaGVtYSwgdGFibGUgYW5kIGluZGV4ZXMgYXJlIGNyZWF0ZWQuCiogQXBwbGljYXRpb24gYW5kIGRhdGFiYXNlIGRvY2tlciBjb250YWluZXJzIGNvbXBvc2VkIGFuZCBydW4gd2l0aCBEb2NrZXIgQ29tcG9zZS4KCj09IFByZXJlcXVpc2l0ZXMKCiogSkRLIDgrCiogRG9ja2VyIEVuZ2luZQoqIERvY2tlciBDb21wb3NlCgo9PSBCdWlsZGluZwoKVG8gcGFja2FnZSB5b3VyIGFwcGxpY2F0aW9uOgpgYGAKLi9ncmFkbGV3IGNsZWFuIGJ1aWxkIC14IHRlc3QKYGBgCgo9PSBSdW4KClRvIGNvbXBvc2UgYW5kIHJ1biB5b3VyIGFwcGxpY2F0aW9uOgpgYGAKZG9ja2VyLWNvbXBvc2UgdXAgLS1idWlsZApgYGAKCk5PVEU6IERvY2tlciBjb21wb3NlcyBhbmQgcnVucyBhbiBhcHBsaWNhdGlvbiBjb250YWluZXIgYW5kIGEgZGF0YWJhc2UgY29udGFpbmVyCgpUbyBzdG9wIGFwcGxpY2F0aW9uIGFuZCBkYiBjb250YWluZXJzOgpgYGAKZG9ja2VyLWNvbXBvc2UgZG93biAtdgpgYGAKCj09IEdldHRpbmcgU3RhcnRlZApQb3N0IGEgdm90ZSB3aXRoIGBjdXJsYDoKCmBgYGJhc2gKY3VybCAtWCBQT1NUICJodHRwOi8vbG9jYWxob3N0Ojg4ODgvdm90ZXMvMjAyMCIgLUggImFjY2VwdDogYXBwbGljYXRpb24vanNvbiIgLUggIkNvbnRlbnQtVHlwZTogYXBwbGljYXRpb24vanNvbiIgLWQgIntcImNvdW50cnlGcm9tXCI6XCJOZXRoZXJsYW5kc1wiLFwidm90ZWRGb3JcIjpcIkJlbGdpdW1cIn0iCmBgYAoKR2V0IHRvcCB0aHJlZSBjb3VudHJpZXMgZm9yIHRoZSBzcGVjaWZpZWQgeWVhcgoKYGBgYmFzaApjdXJsIC1YIEdFVCAiaHR0cDovL2xvY2FsaG9zdDo4ODg4L3ZvdGVzLzIwMjAiIC1IICJhY2NlcHQ6IGFwcGxpY2F0aW9uL2pzb24iCmBgYAoKR2V0IHRoZSB0b3AgdGhyZWUgZmF2b3JpdGUgc29uZ3MgZm9yIHRoZSBzcGVjaWZpZWQgeWVhciBhbmQgY291bnRyeQpgYGBiYXNoCmN1cmwgLVggR0VUICJodHRwOi8vbG9jYWxob3N0Ojg4ODgvdm90ZXMvMjAyMC9OZXRoZXJsYW5kcyIgLUggImFjY2VwdDogYXBwbGljYXRpb24vanNvbiIKYGBgCgpUbyBjaGVjayBvbmxpbmUgcmVzdWx0cyBsYXVuY2ggVm90ZSBSZXN1bHRzIERhc2hib2FyZCBhdCBodHRwOi8vbG9jYWxob3N0Ojk5OTkvCgo9PSBIZWxwCgoqIGh0dHBzOi8vdmVydHguaW8vZG9jcy9bVmVydC54IERvY3VtZW50YXRpb25dCiogaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9xdWVzdGlvbnMvdGFnZ2VkL3ZlcnQueD9zb3J0PW5ld2VzdCZwYWdlU2l6ZT0xNVtWZXJ0LnggU3RhY2sgT3ZlcmZsb3ddCiogaHR0cHM6Ly9ncm91cHMuZ29vZ2xlLmNvbS9mb3J1bS8/ZnJvbWdyb3VwcyMhZm9ydW0vdmVydHhbVmVydC54IFVzZXIgR3JvdXBdCiogaHR0cHM6Ly9naXR0ZXIuaW0vZWNsaXBzZS12ZXJ0eC92ZXJ0eC11c2Vyc1tWZXJ0LnggR2l0dGVyXQoKCg== readmeEtag: '"d66330f74b463c3e4cf93ab1917f089cb009adf0"' readmeLastModified: Tue, 26 May 2020 12:07:34 GMT repositoryId: 266740708 description: >- A sample implementation to count the results of the Eurovision Song Contest created: '2020-05-25T09:38:47Z' updated: '2020-06-04T11:27:55Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: HalilOzkan logo: https://avatars.githubusercontent.com/u/8930446?v=4 license: Apache-2.0 repoEtag: '"e2d8b9bde7c73343c7122502cfe9bd6bae00e8464e73a649810506731bd63717"' repoLastModified: Thu, 04 Jun 2020 11:27:55 GMT foundInMaster: true category: Server Implementations id: 4b91d74f01aeab56d54b13f39582b9bb - source: openapi3 tags repository: https://github.com/gosticks/octoprint-open-api v3: true repositoryMetadata: base64Readme: >- CiMgT2N0b1ByaW50IE9wZW5BUEkgRGVmaW5pdGlvbiAoQkVUQSEhISkKClRoaXMgcHJvamVjdCBwcm92ZWQgYSB1bm9mZmljaWFsIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBmb3IgdGhlIHBvcHVsYXIgb3BlbiBzb3VyY2UgM0QgcHJpbnRlciBjb250cm9sIHNvZnR3YXJlIE9jdG9QcmludC4gVGhlIGRlZmluaXRpb25zIGFyZSBiYXNlZCBvbiB0aGUgY3VycmVudGx5IGF2YWlsYWJsZSBPY3RvUHJpbnQgYXBpIGRvY3VtZW50YXRpb24uIAoKIyMgV29ya2luZyBvbiB5b3VyIE9wZW5BUEkgRGVmaW5pdGlvbgoKIyMjIEluc3RhbGwKCjEuIEluc3RhbGwgW05vZGUgSlNdKGh0dHBzOi8vbm9kZWpzLm9yZy8pLgoyLiBDbG9uZSB0aGlzIHJlcG8gYW5kIHJ1biBgeWFybiBpbnN0YWxsYC9gbnBtIGluc3RhbGxgIGluIHRoZSByZXBvIHJvb3QuCgojIyMgVXNhZ2UKCiMjIyMgYG5wbSBzdGFydGAKU3RhcnRzIHRoZSByZWZlcmVuY2UgZG9jcyBwcmV2aWV3IHNlcnZlci4KCiMjIyMgYG5wbSBydW4gYnVpbGRgCkJ1bmRsZXMgdGhlIGRlZmluaXRpb24gdG8gdGhlIGRpc3QgZm9sZGVyLgoKIyMjIyBgbnBtIHRlc3RgClZhbGlkYXRlcyB0aGUgZGVmaW5pdGlvbi4KCiMjIENvbnRyaWJ1dGlvbiBHdWlkZQoKQmVsb3cgaXMgYSBzYW1wbGUgY29udHJpYnV0aW9uIGd1aWRlLiBUaGUgdG9vbHMKaW4gdGhlIHJlcG9zaXRvcnkgZG9uJ3QgcmVzdHJpY3QgeW91IHRvIGFueQpzcGVjaWZpYyBzdHJ1Y3R1cmUuIEFkanVzdCB0aGUgY29udHJpYnV0aW9uIGd1aWRlCnRvIG1hdGNoIHlvdXIgb3duIHN0cnVjdHVyZS4gSG93ZXZlciwgaWYgeW91CmRvbid0IGhhdmUgYSBzdHJ1Y3R1cmUgaW4gbWluZCwgdGhpcyBpcyBhCmdvb2QgcGxhY2UgdG8gc3RhcnQuCgpVcGRhdGUgdGhpcyBjb250cmlidXRpb24gZ3VpZGUgaWYgeW91CmFkanVzdCB0aGUgZmlsZS9mb2xkZXIgb3JnYW5pemF0aW9uLgoKVGhlIGAucmVkb2NseS55YW1sYCBjb250cm9scyBzZXR0aW5ncyBmb3IgdmFyaW91cwp0b29scyBpbmNsdWRpbmcgdGhlIGxpbnQgdG9vbCBhbmQgdGhlIHJlZmVyZW5jZQpkb2NzIGVuZ2luZS4gIE9wZW4gaXQgdG8gZmluZCBleGFtcGxlcyBhbmQKW3JlYWQgdGhlIGRvY3NdKGh0dHBzOi8vcmVkb2MubHkvZG9jcy9jbGkvY29uZmlndXJhdGlvbi8pCmZvciBtb3JlIGluZm9ybWF0aW9uLgoKIyMjIFNjaGVtYXMKCiMjIyMgQWRkaW5nIFNjaGVtYXMKCjEuIE5hdmlnYXRlIHRvIHRoZSBgb3BlbmFwaS9jb21wb25lbnRzL3NjaGVtYXNgIGZvbGRlci4KMi4gQWRkIGEgZmlsZSBuYW1lZCBhcyB5b3Ugd2lzaCB0byBuYW1lIHRoZSBzY2hlbWEuCjMuIERlZmluZSB0aGUgc2NoZW1hLgo0LiBSZWZlciB0byB0aGUgc2NoZW1hIHVzaW5nIHRoZSBgJHJlZmAgKHNlZSBleGFtcGxlIGJlbG93KS4KCiMjIyMjIFVzaW5nIHRoZSBgJHJlZmAKCk5vdGljZSBpbiB0aGUgY29tcGxleCBleGFtcGxlIGFib3ZlIHRoZSBzY2hlbWEgZGVmaW5pdGlvbiBpdHNlbGYgaGFzIGAkcmVmYCBsaW5rcyB0byBvdGhlciBzY2hlbWFzIGRlZmluZWQuCgpIZXJlIGlzIGEgc21hbGwgZXhjZXJwdCB3aXRoIGFuIGV4YW1wbGU6CgpgYGB5YW1sCmRlZmF1bHRQYXltZW50SW5zdHJ1bWVudDoKICAkcmVmOiAuL1BheW1lbnRJbnN0cnVtZW50LnlhbWwKYGBgCgpUaGUgdmFsdWUgb2YgdGhlIGAkcmVmYCBpcyB0aGUgcGF0aCB0byB0aGUgb3RoZXIgc2NoZW1hIGRlZmluaXRpb24uCgpZb3UgbWF5IHVzZSBgJHJlZmBzIHRvIGNvbXBvc2Ugc2NoZW1hIGZyb20gb3RoZXIgZXhpc3Rpbmcgc2NoZW1hIHRvIGF2b2lkIGR1cGxpY2F0aW9uLgoKWW91IHdpbGwgdXNlIGAkcmVmYHMgdG8gcmVmZXJlbmNlIHNjaGVtYSBmcm9tIHlvdXIgcGF0aCBkZWZpbml0aW9ucy4KCiMjIyMgRWRpdGluZyBTY2hlbWFzCgoxLiBOYXZpZ2F0ZSB0byB0aGUgYG9wZW5hcGkvY29tcG9uZW50cy9zY2hlbWFzYCBmb2xkZXIuCjIuIE9wZW4gdGhlIGZpbGUgeW91IHdpc2ggdG8gZWRpdC4KMy4gRWRpdC4KCiMjIyBQYXRocwoKIyMjIyBBZGRpbmcgYSBQYXRoCgoxLiBOYXZpZ2F0ZSB0byB0aGUgYG9wZW5hcGkvcGF0aHNgIGZvbGRlci4KMi4gQWRkIGEgbmV3IFlBTUwgZmlsZSBuYW1lZCBsaWtlIHlvdXIgVVJMIGVuZHBvaW50IGV4Y2VwdCByZXBsYWNpbmcgYC9gIHdpdGggYEBgIGFuZCBwdXR0aW5nIHBhdGggcGFyYW1ldGVycyBpbnRvIGN1cmx5IGJyYWNlcyBsaWtlIGB7ZXhhbXBsZX1gLgozLiBBZGQgdGhlIHBhdGggYW5kIGEgcmVmIHRvIGl0IGluc2lkZSBvZiB5b3VyIGBvcGVuYXBpLnlhbWxgIGZpbGUgaW5zaWRlIG9mIHRoZSBgb3BlbmFwaWAgZm9sZGVyLgoKRXhhbXBsZSBhZGRpdGlvbiB0byB0aGUgYG9wZW5hcGkueWFtbGAgZmlsZToKYGBgeWFtbAonL2N1c3RvbWVycy97aWR9JzoKICAkcmVmOiAnLi9wYXRocy9jdXN0b21lcnNAe2lkfS55YW1sJwpgYGAKCkhlcmUgaXMgYW4gZXhhbXBsZSBvZiBhIFlBTUwgZmlsZSBuYW1lZCBgY3VzdG9tZXJzQHtpZH0ueWFtbGAgaW4gdGhlIGBwYXRoc2AgZm9sZGVyOgoKYGBgeWFtbApnZXQ6CiAgdGFnczoKICAgIC0gQ3VzdG9tZXJzCiAgc3VtbWFyeTogUmV0cmlldmUgYSBsaXN0IG9mIGN1c3RvbWVycwogIG9wZXJhdGlvbklkOiBHZXRDdXN0b21lckNvbGxlY3Rpb24KICBkZXNjcmlwdGlvbjogfAogICAgWW91IGNhbiBoYXZlIGEgbWFya2Rvd24gZGVzY3JpcHRpb24gaGVyZS4KICBwYXJhbWV0ZXJzOgogICAgLSAkcmVmOiAuLi9jb21wb25lbnRzL3BhcmFtZXRlcnMvY29sbGVjdGlvbkxpbWl0LnlhbWwKICAgIC0gJHJlZjogLi4vY29tcG9uZW50cy9wYXJhbWV0ZXJzL2NvbGxlY3Rpb25PZmZzZXQueWFtbAogICAgLSAkcmVmOiAuLi9jb21wb25lbnRzL3BhcmFtZXRlcnMvY29sbGVjdGlvbkZpbHRlci55YW1sCiAgICAtICRyZWY6IC4uL2NvbXBvbmVudHMvcGFyYW1ldGVycy9jb2xsZWN0aW9uUXVlcnkueWFtbAogICAgLSAkcmVmOiAuLi9jb21wb25lbnRzL3BhcmFtZXRlcnMvY29sbGVjdGlvbkV4cGFuZC55YW1sCiAgICAtICRyZWY6IC4uL2NvbXBvbmVudHMvcGFyYW1ldGVycy9jb2xsZWN0aW9uRmllbGRzLnlhbWwKICByZXNwb25zZXM6CiAgICAnMjAwJzoKICAgICAgZGVzY3JpcHRpb246IEEgbGlzdCBvZiBDdXN0b21lcnMgd2FzIHJldHJpZXZlZCBzdWNjZXNzZnVsbHkKICAgICAgaGVhZGVyczoKICAgICAgICBSYXRlLUxpbWl0LUxpbWl0OgogICAgICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9oZWFkZXJzL1JhdGUtTGltaXQtTGltaXQueWFtbAogICAgICAgIFJhdGUtTGltaXQtUmVtYWluaW5nOgogICAgICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9oZWFkZXJzL1JhdGUtTGltaXQtUmVtYWluaW5nLnlhbWwKICAgICAgICBSYXRlLUxpbWl0LVJlc2V0OgogICAgICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9oZWFkZXJzL1JhdGUtTGltaXQtUmVzZXQueWFtbAogICAgICAgIFBhZ2luYXRpb24tVG90YWw6CiAgICAgICAgICAkcmVmOiAuLi9jb21wb25lbnRzL2hlYWRlcnMvUGFnaW5hdGlvbi1Ub3RhbC55YW1sCiAgICAgICAgUGFnaW5hdGlvbi1MaW1pdDoKICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvaGVhZGVycy9QYWdpbmF0aW9uLUxpbWl0LnlhbWwKICAgICAgICBQYWdpbmF0aW9uLU9mZnNldDoKICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvaGVhZGVycy9QYWdpbmF0aW9uLU9mZnNldC55YW1sCiAgICAgIGNvbnRlbnQ6CiAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgIHNjaGVtYToKICAgICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgICAgaXRlbXM6CiAgICAgICAgICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9zY2hlbWFzL0N1c3RvbWVyLnlhbWwKICAgICAgICB0ZXh0L2NzdjoKICAgICAgICAgIHNjaGVtYToKICAgICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgICAgaXRlbXM6CiAgICAgICAgICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9zY2hlbWFzL0N1c3RvbWVyLnlhbWwKICAgICc0MDEnOgogICAgICAkcmVmOiAuLi9jb21wb25lbnRzL3Jlc3BvbnNlcy9BY2Nlc3NGb3JiaWRkZW4ueWFtbAogIHgtY29kZS1zYW1wbGVzOgogICAgLSBsYW5nOiBQSFAKICAgICAgc291cmNlOgogICAgICAgICRyZWY6IC4uL2NvZGVfc2FtcGxlcy9QSFAvY3VzdG9tZXJzL2dldC5waHAKcG9zdDoKICB0YWdzOgogICAgLSBDdXN0b21lcnMKICBzdW1tYXJ5OiBDcmVhdGUgYSBjdXN0b21lciAod2l0aG91dCBhbiBJRCkKICBvcGVyYXRpb25JZDogUG9zdEN1c3RvbWVyCiAgZGVzY3JpcHRpb246IEFub3RoZXIgbWFya2Rvd24gZGVzY3JpcHRpb24gaGVyZS4KICByZXF1ZXN0Qm9keToKICAgICRyZWY6IC4uL2NvbXBvbmVudHMvcmVxdWVzdEJvZGllcy9DdXN0b21lci55YW1sCiAgcmVzcG9uc2VzOgogICAgJzIwMSc6CiAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvcmVzcG9uc2VzL0N1c3RvbWVyLnlhbWwKICAgICc0MDEnOgogICAgICAkcmVmOiAuLi9jb21wb25lbnRzL3Jlc3BvbnNlcy9BY2Nlc3NGb3JiaWRkZW4ueWFtbAogICAgJzQwOSc6CiAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvcmVzcG9uc2VzL0NvbmZsaWN0LnlhbWwKICAgICc0MjInOgogICAgICAkcmVmOiAuLi9jb21wb25lbnRzL3Jlc3BvbnNlcy9JbnZhbGlkRGF0YUVycm9yLnlhbWwKICB4LWNvZGUtc2FtcGxlczoKICAgIC0gbGFuZzogUEhQCiAgICAgIHNvdXJjZToKICAgICAgICAkcmVmOiAuLi9jb2RlX3NhbXBsZXMvUEhQL2N1c3RvbWVycy9wb3N0LnBocApgYGAKCllvdSdsbCBzZWUgZXh0ZW5zaXZlIHVzYWdlIG9mIGAkcmVmYHMgaW4gdGhpcyBleGFtcGxlIHRvIGRpZmZlcmVudCB0eXBlcyBvZiBjb21wb25lbnRzIGluY2x1ZGluZyBzY2hlbWFzLgoKWW91J2xsIGFsc28gbm90aWNlIGAkcmVmYHMgdG8gY29kZSBzYW1wbGVzLgoK readmeEtag: '"71b48387434e3d2f2606711f48ba1e62b3db0c47"' readmeLastModified: Sun, 22 Nov 2020 00:58:08 GMT repositoryId: 306438975 description: OctoPrint OpenAPI definition created: '2020-10-22T19:27:00Z' updated: '2022-03-28T13:09:07Z' language: null archived: false stars: 1 watchers: 1 forks: 0 owner: gosticks logo: https://avatars.githubusercontent.com/u/9556979?v=4 license: MIT repoEtag: '"77703a0d51545614a44fc735d1dc004cc146de4a0752cd48cfa14f21bdb08652"' repoLastModified: Mon, 28 Mar 2022 13:09:07 GMT foundInMaster: true category: Parsers id: 2ee853734020d1d065fc32dc763f8940 - source: openapi3 tags repository: https://github.com/stephan-mueller/api-documentation-showcase v3: true repositoryMetadata: base64Readme: >- IyBBUEkgRG9jdW1lbnRhdGlvbiBTaG93Y2FzZQoKWyFbR2l0SHViIFdvcmtmbG93XShodHRwczovL2dpdGh1Yi5jb20vc3RlcGhhbi1tdWVsbGVyL2FwaS1kb2N1bWVudGF0aW9uLXNob3djYXNlL2FjdGlvbnMvd29ya2Zsb3dzL21hdmVuLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vc3RlcGhhbi1tdWVsbGVyL2FwaS1kb2N1bWVudGF0aW9uLXNob3djYXNlL2FjdGlvbnMpClshW0dpdEh1YiBsYXN0IGNvbW1pdF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGFzdC1jb21taXQvc3RlcGhhbi1tdWVsbGVyL2FwaS1kb2N1bWVudGF0aW9uLXNob3djYXNlKV0oaHR0cHM6Ly9naXRodWIuY29tL3N0ZXBoYW4tbXVlbGxlci9hcGktZG9jdW1lbnRhdGlvbi1zaG93Y2FzZS9jb21taXRzKQpbIVtHaXRIdWJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2Uvc3RlcGhhbi1tdWVsbGVyL2FwaS1kb2N1bWVudGF0aW9uLXNob3djYXNlKV0oaHR0cHM6Ly9naXRodWIuY29tL3N0ZXBoYW4tbXVlbGxlci9hcGktZG9jdW1lbnRhdGlvbi1zaG93Y2FzZS9ibG9iL21hc3Rlci9MSUNFTlNFKQoKVGhpcyBpcyBhIHNob3djYXNlIGZvciBhcGkgZG9jdW1lbnRhdGlvbi4gSXQgY29udGFpbnMgaGVsbG8gd29ybGQgYXBwbGljYXRpb25zLCB3aGljaCBkZW1vbnN0cmF0ZXMgZmVhdHVyZXMgb2YgdGhlCltPcGVuQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vb3BlbmFwaXMub3JnKSBhbmQgb3RoZXJzLiBTb2Z0d2FyZSByZXF1aXJlbWVudHMgdG8gcnVuIHRoZSBzYW1wbGVzIGFyZSBgbWF2ZW5gLCBgb3Blbmpkay04YCAob3IgYW55IG90aGVyIApKREsgOCkgYW5kIGBkb2NrZXJgLgoKIyMgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIChPQVMpCgpUaGUgW09wZW5BUEkgU3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMC4zKSAoT0FTKSBkZWZpbmVzIGEgc3RhbmRhcmQsIHByb2dyYW1taW5nIGxhbmd1YWdlLWFnbm9zdGljIGludGVyZmFjZSBkZXNjcmlwdGlvbiBmb3IgUkVTVCBBUElzLCB3aGljaCBhbGxvd3MgYm90aCAKaHVtYW5zIGFuZCBjb21wdXRlcnMgdG8gZGlzY292ZXIgYW5kIHVuZGVyc3RhbmQgdGhlIGNhcGFiaWxpdGllcyBvZiBhIHNlcnZpY2Ugd2l0aG91dCByZXF1aXJpbmcgYWNjZXNzIHRvIHNvdXJjZSBjb2RlLCBhZGRpdGlvbmFsIApkb2N1bWVudGF0aW9uLCBvciBpbnNwZWN0aW9uIG9mIG5ldHdvcmsgdHJhZmZpYy4gV2hlbiBwcm9wZXJseSBkZWZpbmVkIHZpYSBPcGVuQVBJLCBhIGNvbnN1bWVyIGNhbiB1bmRlcnN0YW5kIGFuZCBpbnRlcmFjdCB3aXRoIHRoZSByZW1vdGUKc2VydmljZSB3aXRoIGEgbWluaW1hbCBhbW91bnQgb2YgaW1wbGVtZW50YXRpb24gbG9naWMuIFNpbWlsYXIgdG8gd2hhdCBpbnRlcmZhY2UgZGVzY3JpcHRpb25zIGhhdmUgZG9uZSBmb3IgbG93ZXItbGV2ZWwgcHJvZ3JhbW1pbmcsIHRoZSAKT3BlbkFQSSBTcGVjaWZpY2F0aW9uIHJlbW92ZXMgZ3Vlc3N3b3JrIGluIGNhbGxpbmcgYSBzZXJ2aWNlLgoKIyMjIE9BUyB2MiAoZmthIFN3YWdnZXIgUkVTVGZ1bCBBUEkgRG9jdW1lbnRhdGlvbiBTcGVjaWZpY2F0aW9uKQoKW1N3YWdnZXJdKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uL3YyLykgaXMgYW4gSW50ZXJmYWNlIERlc2NyaXB0aW9uIExhbmd1YWdlIGZvciBkZXNjcmliaW5nIFJFU1RmdWwgQVBJcyBleHByZXNzZWQgdXNpbmcgSlNPTi4gU3dhZ2dlciBpcyB1c2VkIHRvZ2V0aGVyIHdpdGggYSBzZXQgb2YgCm9wZW4tc291cmNlIHNvZnR3YXJlIHRvb2xzIHRvIGRlc2lnbiwgYnVpbGQsIGRvY3VtZW50LCBhbmQgdXNlIFJFU1RmdWwgd2ViIHNlcnZpY2VzLiBTd2FnZ2VyIGluY2x1ZGVzIGF1dG9tYXRlZCBkb2N1bWVudGF0aW9uLCBjb2RlIApnZW5lcmF0aW9uIChpbnRvIG1hbnkgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2VzKSwgYW5kIHRlc3QtY2FzZSBnZW5lcmF0aW9uLgoKIyMjIFNob3djYXNlcwoKKiBUaGUgW0VjbGlwc2UgTWljcm9Qcm9maWxlIE9wZW5BUEkgU2hvd2Nhc2VdKG1wLW9wZW5hcGktc2hvd2Nhc2UvUkVBRE1FLm1kKSBzaG93cyBmZWF0dXJlcyBvZiB0aGUgTVAgT3BlbkFQSSBTcGVjaWZpY2F0aW9uLgoqIFRoZSBbU3dhZ2dlciBTaG93Y2FzZV0oc3dhZ2dlci1zaG93Y2FzZS9SRUFETUUubWQpIHNob3dzIGZlYXR1cmVzIG9mIHRoZSBTd2FnZ2VyIFNwZWNpZmljYXRpb24gYW5kIHRoZSBTd2FnZ2VyIENvcmUgZnJhbWV3b3JrLg== readmeEtag: '"7d2b40c246a6a49d24193c019b1e68575b766de7"' readmeLastModified: Wed, 16 Jun 2021 14:42:06 GMT repositoryId: 368658825 description: null created: '2021-05-18T20:30:38Z' updated: '2023-05-24T06:20:59Z' language: Java archived: true stars: 1 watchers: 1 forks: 0 owner: stephan-mueller logo: https://avatars.githubusercontent.com/u/13217222?v=4 license: Apache-2.0 repoEtag: '"22294cc10d1f41cff5528565d53a238cd044351ab32047d9ea4d3050a1c34237"' repoLastModified: Wed, 24 May 2023 06:20:59 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 7d031cad29aed62914f5b9b6a12b9188 - source: openapi3 tags repository: https://github.com/javasks/open-api v3: true repositoryMetadata: repositoryId: 366143641 description: >- This repo is related to the open API generator. You need to come with the first API contract and then Models and endpoint will be exposed based on the open API generator plugin. created: '2021-05-10T18:43:46Z' updated: '2021-05-11T07:51:15Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: javasks logo: https://avatars.githubusercontent.com/u/35063886?v=4 repoEtag: '"34016352aed3ca40b27c6bcbfe8af476abf29bc55a3262844f4608101d197518"' repoLastModified: Tue, 11 May 2021 07:51:15 GMT foundInMaster: true id: c537c00b4db182f81ea02e362da86e06 - source: openapi3 tags repository: https://github.com/scalabli/yadig v3: true repositoryMetadata: base64Readme: >- [![Downloads](https://pepy.tech/badge/citus)](https://pepy.tech/project/citus)
[![PyPI version](https://badge.fury.io/py/citus.svg)](https://badge.fury.io/py/citus)
[![Wheel](https://img.shields.io/pypi/wheel/citus.svg)](https://pypi.com/project/citus)
[![Windows Build Status](https://img.shields.io/appveyor/build/gerrishons/citus/main?logo=appveyor&cacheSeconds=600)](https://ci.appveyor.com/project/gerrishons/citus)
[![pyimp](https://img.shields.io/pypi/implementation/citus.svg)](https://pypi.com/project/citus)
[![RTD](https://readthedocs.org/projects/citus/badge/)](https://citus.readthedocs.io)
[![licence](https://img.shields.io/pypi/l/citus.svg)](https://opensource.org/licenses/MIT)
[![Twitter Follow](https://img.shields.io/twitter/follow/gerrishon_s.svg?style=social)](https://twitter.com/gerrishon_s)


[![Logo](https://raw.githubusercontent.com/secretum-inc/citus/main/docs/images/citus.png)](https://github.com/secretum-inc/citus)


`Forever Scalable`

**𝙲𝚒𝚝𝚞𝚜** is a python based, ultrafast web framework  focusing on composing Web APIs all the more rapidly and with needless baggage. 

Citus requires Python `3.8` or later. 


## Features
The key features are:

- [x] **Fast**: Very high performance, on par with **NodeJS** and **Go** (thanks to Starlette and Pydantic). [One of the fastest Python frameworks available](#performance).

- [x] **Fast to code**: Increase the speed to develop features by about 200% to 300%. *
- [x] **Fewer bugs**: Reduce about 40% of human (developer) induced errors. *
- [x] **Intuitive**: Great editor support. <abbr title="also known as auto-complete, autocompletion, IntelliSense">Completion</abbr> everywhere. Less time debugging.
- [x] **Easy**: Designed to be easy to use and learn. Less time reading docs.
- [x] **Short**: Minimize code duplication. Multiple features from each parameter declaration. Fewer bugs.
- [x] **Robust**: Get production-ready code. With automatic interactive documentation.
- [x] **Standards-based**: Based on (and fully compatible with) the open standards for APIs: <a href="https://github.com/OAI/OpenAPI-Specification" class="external-link" target="_blank">OpenAPI</a> (previously known as Swagger) and <a href="https://json-schema.org/" class="external-link" target="_blank">JSON Schema</a>.

<small>* estimation based on tests on an internal development team, building production applications.</small>
- [x] Support for Ansi, RGB and HTML color models
- [x] Support for tabular presentation of data
- [x] Interactive progressbars
- [x] Code completions
- [x] Nesting of commands
- [x] Automatic help page generation
- [x] Syntax highlighting
- [x] Autosuggestions
- [x] Key Binders

## Getting Started
### Installation
You can install citus via the Python Package Index (PyPI)

```
pip install -U citus
```

## Example

### Create it

* Create a file `main.py` with:

```Python
from typing import Optional

import citus

app = citus.App()


@app.GET("/")
def read_root():
    return "Hello World"


@app.GET("/items/{item_id}")
def read_item(item_id: int, q: Optional[str] = None):
    return {"item_id": item_id, "q": q}
```

<details markdown="1">
<summary>Or use <code>async def</code>...</summary>

If your code uses `async` / `await`, use `async def`:

```Python hl_lines="9  14"
from typing import Optional
import quo

app = citus.App()


@app.GET("/")
async def read_root():
    return "Hello World"


@app.GET("/items/{item_id}")
async def read_item(item_id: int, q: Optional[str] = None):
    return {"item_id": item_id, "q": q}
```


**Note**:

If you don't know, check the _"In a hurry?"_ section about <a href="https://fastapi.tiangolo.com/async/#in-a-hurry" target="_blank">`async` and `await` in the docs</a>.

</details>

### Run it

Run the server with:

<div class="termy">

```console
$ citus main:app --reload

INFO:     Citus running on http://127.0.0.1:8000 (Press CTRL+C to quit)
INFO:     Started reloader process [28720]
INFO:     Started server process [28722]
INFO:     Waiting for application startup.
INFO:     Application startup complete.
```

</div>

<details markdown="1">
<summary>About the command <code>uvicorn main:app --reload</code>...</summary>

The command `citus main:app` refers to:

* `main`: the file `main.py` (the Python "module").
* `app`: the object created inside of `main.py` with the line `app = citus.App()`.
* `--reload` or `-r`: make the server restart after code changes. Only do this for development.

</details>

## Example upgrade

Now modify the file `main.py` to receive a body from a `PUT` request.

Declare the body using standard Python types, thanks to Pydantic.

```Python hl_lines="4  9-12  25-27"
from typing import Optional

import citus

app = citus.App()


class Item(citus.Base):
    name: str
    price: float
    is_offer: Optional[bool] = None


@app.GET("/")
def read_root():
    return "Hello World"


@app.GET("/items/{item_id}")
def read_item(item_id: int, q: Optional[str] = None):
    return {"item_id": item_id, "q": q}


@app.GET("/items/{item_id}")
def update_item(item_id: int, item: Item):
    return {"item_name": item.name, "item_id": item_id}
```

The server should reload automatically (because you added `--reload` to the command above).

### Check it

Open your browser at <a href="http://127.0.0.1:8000/items/33?q=checkuser" class="external-link" target="_blank">http://127.0.0.1:8000/items/33?q=checkuser</a>.

You will see the JSON response as:

```JSON
{"item_id": 33, "q": "checkuser"}
```

You already created an API that:

* Receives HTTP requests in the _paths_ `/` and `/items/{item_id}`.
* Both _paths_ take `GET` <em>operations</em> (also known as HTTP _methods_).
* The _path_ `/items/{item_id}` has a _path parameter_ `item_id` that should be an `int`.
* The _path_ `/items/{item_id}` has an optional `str` _que

## Getting Help

### Community

For discussions about the usage, development, and the future of quo, please join our Google community

* [Community👨‍👩‍👦‍👦](https://groups.google.com/forum/#!forum/secretum)

## Resources

### Bug tracker

If you have any suggestions, bug reports, or annoyances please report them
to our issue tracker at 
[Bug tracker](https://github.com/secretum-inc/citus/issues/) or send an email to:

 📥 secretum@googlegroups.com


## License📑

[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)  
This software is licensed under the `MIT License`. See the [License](https://github.com/secretum-inc/citus/blob/main/LICENSE) file in the top distribution directory for the full license text.


## Code of Conduct
Code of Conduct is adapted from the Contributor Covenant,
version 1.2.0 available at
[Code of Conduct](http://contributor-covenant.org/version/1/2/0/)

 readmeEtag: '"5deadb74a6fd7a18a619c7e5a52e488e5b8f5fa8"' readmeLastModified: Tue, 29 Mar 2022 19:43:26 GMT repositoryId: 444886233 description: >- `𝙲𝚒𝚝𝚞𝚜` is a python based, ultrafast web framework focusing on composing Web APIs all the more rapidly and with needless baggage created: '2022-01-05T17:08:58Z' updated: '2022-10-15T08:01:07Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: scalabli logo: https://avatars.githubusercontent.com/u/34297519?v=4 license: MIT repoEtag: '"cc3ddaa0e88c0fc036bb762a00a535498e7756b7b36eb5fd48f7b95f33030a00"' repoLastModified: Sat, 15 Oct 2022 08:01:07 GMT foundInMaster: true category: Server Implementations id: e230a9681537acc79ecffb5abe8f70d2 oldLocations: - https://github.com/secretum-inc/citus - source: openapi3 tags repository: https://github.com/izobretatel777/ambiglyph-server v3: true id: 2021539a7c6727ac3683c732f2ce6c34 repositoryMetadata: base64Readme: >- IyBBbWJpZ2x5cGggU2VydmVyCjxkaXYgYWxpZ249ImNlbnRlciI+CjxpbWcgc3JjPSJkb2MvaWNvbi5wbmciIGFsdD0iaWNvbiIgd2lkdGg9IjEyOCIvPgo8aDI+QW1iaWdseXBoPC9oMj4KPC9kaXY+CgoKKipBbWJpZ2x5cGggU2VydmVyKiogLSB0aGUgaW1wbGVtZW50YXRpb24gb2YgdGhlIEFtYmlnbHlwaCBwcm9qZWN0IGJhY2tlbmQuIGl0IGlzIGZ1bGx5IHdyaXR0ZW4gaW4gSmF2YSB3aXRoIFNwcmluZyB0ZWNobm9sb2d5IHRoYXRzIHByb3ZpZGUgcG93ZXJmdWwgQVBJLgoKSXQgY2FuIGRldGVjdCBob21vZ2x5cGhzIGluc2lkZSBhIHRleHQgYW5kIGdpdmUgc3VnZ2VzdGlvbnMgdG8gcmVjb3ZlciBpdCB1c2luZyBMZXZlbnNodGVpbiBhbGdvcml0aG0uICBUaGUgbWFpbiBhcHByb2FjaCBpcyB0byB1c2UgdGhlIGFsZ29yaXRobSBhcyBpdCB1c2VkIGZvciBtaXNzcGVsbHMgZml4aW5nLiAgQW4gdXNlciB3aWxsIGJlIHByb3ZpZGVkIHdpdGggc3VnZ2VzdGlvbnMgaG93IHdpdGggd2hpY2ggd29yZCBvYmZ1c2NhdGVkIG9uZSBjYW4gYmUgcmVwbGFjZWQuCgpUaGUgc2VydmVyIHVzZXMgb3BlbiBkYXRhYmFzZSBvZiB3b3JkcyBhbmQgaG9tb2dseXBzOiBodHRwOi8vaG9tb2dseXBocy5uZXQvCgphbmQgZGF0YWJhc2Ugb2Ygd29yZHM6IGh0dHA6Ly93d3cubWllbGllc3Ryb25rLmNvbS9jb3JuY29iX2xvd2VyY2FzZS50eHQKCgojIyBCdWlsZAoKIyMjIE1hbnVhbCBCdWlsZC9SdW4KCioqUmVxdWlyZW1lbnRzKioKCi0gSmF2YSAxNwoKLSBNeVNRTCBTZXJ2ZXIgOAoKLSBNYXZlbiAzLjguKwoKUnVuIGluIHRoZSBwcm9qZWN0IHJvb3QgZGlyZWN0b3J5CgpgYGBiYXNoCm12biBzcHJpbmctYm9vdDpydW4KYGBgCgojIyMgRG9ja2VyIEltYWdlCgoqKlJlcXVpcmVtZW50cyoqCgotIERvY2tlcgoKSnVzdCBydW4gaW4gdGhlIHByb2plY3QgZm9sZGVyOgoKYGBgYGBiYXNoCmRvY2tlciBidWlsZCAtdCBhbWJpZ2x5cGgtc2VydmVyIC1mIERvY2tlcmZpbGUKZG9ja2VyIHJ1biAtcCA4MDgxOjgwODEgLXQgYW1iaWdseXBoLXNlcnZlcgpgYGBgYAoKIyMgVXNhZ2UKClRoaXMgaXMgc2ltcGxlIGJhY2tlbmQgYXBwbGljYXRpb24gdGhhdCB1c2VzIFJFU1QgQVBJLiBIZW5jZSwgaXQgc3VwcG9ydHMgb25seSBSRVNUIHJlcXVlc3RzLiBUbyB0cnkgaXQgb3V0IHdpdGhvdXQgYW55IGZyb250IGFwcGxpY2F0aW9uIGFwcGxpY2F0aW9uIHlvdSBjYW4gdXNlIElub21pbmEgb3IgUG9zdG1hbi4gQWxsIG5lY2Vzc2FyeSBkb2N1bWVudGF0aW9uIHByb3ZpZGVkIGluIFN3YWdnZXIgKE9wZW5BcGkgdjMpLiBZb3UgY2FuIGFjY2VzcyBpdCBsaWtlIGBodHRwOi8vYXBpLWFkZHJlc3M6ODA4MS9hbWJpZ2x5cGgtc2VydmVyL3N3YWdnZXItdWkvaW5kZXguaHRtbGAuCgpPdmVyYWxsLCB0aGUgYXBwbGljYXRpb24gc3VwcG9ydHMgKipDUlVEKiogb3BlcmF0aW9ucyBvbiBkYXRhYmFzZSBvZiB3b3JkcyBhbmQgdXNlcnMuIEl0IGlzIHJlcXVpcmVkIHRvIGJlIGF1dGhlbnRpY2F0ZWQgYW5kIHVzZSBnaXZlbiBKV1QtdG9rZW4gaW4gaGVhZGVycyAgU29tZSBiYXNpYyBvcGVyYXRpb25zOgoKYFBPU1QgL2F1dGhlbnRpY2F0ZS9gIAoKQXV0aGVudGljYXRpb24uIEluIGNhc2Ugb2Ygc3VjY2Vzc2Z1bCBhdXRoZW50aWNhdGlvbiBhIEpXVCB0b2tlbiB3aWxsIGJlIHByb3ZpZGVkIGFzIGEgcmVzcG9uc2UuIEV4YW1wbGU6CgohW2F1dGhdKGRvYy9hdXRoLmpwZykKCkRlZmF1bHQgcGFzc3dvcmRzOgoKfCBVc2VybmFtZSAgICAgfCBQYXNzd29yZCAgICB8CnwgLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0gfAp8IEFETUlOICAgICAgICB8IEFETUlOICAgICAgIHwKfCB0ZXN0ICAgICAgICAgfCB0ZXN0MTIzQkhPUyB8CnwgY2xpZW50ICoqKioqIHwgY2xpZW50ICAgICAgfAoKYCpgIFRoaXMgbG9naW4gY2FuIGJlIHVzZWQgZm9yIHVuYXV0aG9yaXplZCB1c2VycyB0byBjaGVjayB1c2luZyBvbmx5IHRoZSBnZW5lcmFsIHdvcmQgZGF0YWJhc2UuIFJlY29tbWVuZGVkIGZvciBjbGllbnQgYXBwbGljYXRpb25zLiogCgoKCmBQT1NUIC91c2Vycy9gCgpSZWdpc3RlciBhIG5ldyB1c2VyLiBUaGUgb3BlcmF0aW9uIGFjY2Vzc2libGUgb25seSBieSB1c2VycyB3aXRoIHJvbGUgQURNSU4gYW5kIEFQUC4gCgpgUE9TVCAvd29yZHMvYAoKQWRkIGEgbmV3IHdvcmQgdG8gZGF0YWJhc2UKCmBHRVQgL3dvcmRzL2AKCkdldCBhbGwgd29yZHMgYWRkZWQgYnkgY3VycmVudCB1c2VyCgpgR0VUIC93b3Jkcy97aWR9YAoKR2V0IHdvcmQgd2l0aCBge2lkfWAKCmBQT1NUIC9jaGVjay9gCgpTZW5kIHRleHQgdG8gY2hlY2suCgpBbGwgIHdvcmRzIHRoYXQgYXJlIGRldGVjdGVkIHRvIGJlIHNwb29mZWQgd2lsbCBiZSByZXBsYWNlZCB3aXRoIGA8JSVhbWJpZ2x5cGgtZGV0ZWN0ZWQ+YCoqbnVtKipgPGFtYmlnbHlwaC1kZXRlY3RlZCUlPmAsIHdoZXJlICoqbnVtKiogLSB0aGUgbnVtYmVyIG9mIGEgZGV0ZWN0aW9uLgoKQWxzbywgaWYgdGhlIG51bWJlciBvZiBzdWdnZXN0ZWQgd29yZCBpcyBoaWdoZXIgdGhhbiBnaXZlbiBsaW1pdCBhbmQgdGhlcmUgaXMgYSBsZXR0ZXIgdGhhdCBpcyBpbiB0aGUgaG9tb2dseXBoIGRhdGFiYXNlLCB0aGUgbGV0dGVyIHdpbGwgYmUgcmVwbGFjZWQgd2l0aCBgPCUlYW1iaWdseXBoLXdhcm5pbmc+YCoqbnVtKipgPGFtYmlnbHlwaC13YXJuaW5nJSU+YCAsIHdoZXJlICoqbnVtKiogLSB0aGUgbnVtYmVyIG9mIGEgd2FybmluZy4KCkluIGNhc2Ugb2YgYW55IGRldGVjdGlvbiBvciB3YXJuaW5nIGNvcnJlc3BvbmRpbmcgdmFsdWVzIGluIHRoZSByZXNwb25zZSB3aWxsIGJlIG1hcmtlZCBhcyB0cnVlLgoKRXhhbXBsZToKCiFbZGV0ZWN0aW9uXShkb2MvZGV0ZWN0aW9uLmpwZykKCkZvciBtb3JlIHJlcXVlc3RzIGFuZCB0aGVpciBzdHJ1Y3R1cmUgYXMgd2VsbCBhcyB0aGVpciBzdXBwb3NlZCBvdXRwdXQsIHBsZWFzZSB0YWtlIGludG8gY29uc2lkZXJhdGlvbiBjb250cm9sbGVycyBhbmQgRFRPIGluIHNvdXJjZSBjb2RlLgoKVGhlIGtleSBwb2ludCBvZiBzZXBhcmF0aW5nIGRhdGFiYXNlIHVzZXJzIGlzIHRvIGFkZCBhbiBhYmlsaXR5IHRvIGFkZCB3b3JkcyB0aGF0IHRoZXkgdXNlIHJlZ3VsYXJseSwgYnV0IHRoaXMgd29yZHMgc3BlY2lmaWMgYW5kIGl0IGlzIGltcG9zc2libGUgdG8gYWRkIHRoZW0gdG8gdGhlIHNoYXJlZCBkYXRhYmFzZS4gRXhhbXBsZToKCiAhW3NwZWNfbm9fZGV0ZWN0aW9uXShkb2Mvc3BlY19ub19kZXRlY3Rpb24uanBnKQoKVGhlcmUgaXMgbm8gZGV0ZWN0aW9uIHNpbmNlIHRoZSB3b3JkIHdhc24ndCBmb3VuZCBpbiBkYXRhYmFzZSBhdCBhbGwgYnV0IGl0IGl0cyBvYmZ1c2NhdGUgaXQuIEhvd2V2ZXIsIGJ5IGFkZGluZyB0aGUgb3JpZ2luYWwgd29yZCB0byBkYXRhYmFzZSB3aWxsIGhlbHAgdG8gYWRkcmVzcyB0aGUgcHJvYmxlbS4KCiFbZGV0ZWN0aW9uXShkb2Mvc3BlY19kZXRlY3Rpb24uanBnKQo= readmeEtag: '"753a7d5f713aaf3f98963801d9acdc050616ca34"' readmeLastModified: Tue, 28 Jun 2022 19:03:02 GMT repositoryId: 433987867 description: Homoglyph detection tool (backend) created: '2021-12-01T21:17:06Z' updated: '2024-07-24T13:07:02Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: IZOBRETATEL777 logo: https://avatars.githubusercontent.com/u/32099652?v=4 repoEtag: '"db6d6dcf9e95621dccef7bad7ae83778ff8bc20bf043b25f5e92f73e1f3dd17f"' repoLastModified: Wed, 24 Jul 2024 13:07:02 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/parthasarathydnu/protien-data-visualizer v3: true id: c0cb320d6c68c5c061f5b1767b811488 repositoryMetadata: base64Readme: >- WyFbQnVpbGQgYW5kIFB1c2ggR2Vub1F1ZXJ5LUFQSSBJbWFnZV0oaHR0cHM6Ly9naXRodWIuY29tL3BhcnRoYXNhcmF0aHlkTlUvcHJvdGllbi1kYXRhLXZpc3VhbGl6ZXIvYWN0aW9ucy93b3JrZmxvd3MvYnVpbGQtcHVzaC1nZW5vcXVlcnktYXBpLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vcGFydGhhc2FyYXRoeWROVS9wcm90aWVuLWRhdGEtdmlzdWFsaXplci9hY3Rpb25zL3dvcmtmbG93cy9idWlsZC1wdXNoLWdlbm9xdWVyeS1hcGkueW1sKQoKIyBbUHJvdGVpbiBEYXNoYm9hcmRdKGh0dHBzOi8vcHJvdGVpbi1kYXNoYm9hcmQtdW5sb2NrLThkeXJ6MzkuZ2FtbWEuc2l0ZS8pCgpUaGlzIHByb2plY3QgaXMgYSBjb21wcmVoZW5zaXZlIHdlYiBhcHBsaWNhdGlvbiBkZXNpZ25lZCB0byBpbnRlcmFjdCB3aXRoIGEgcHJvdGVpbiBkYXRhYmFzZS4gSXQgaW5jbHVkZXMgZnVuY3Rpb25hbGl0aWVzIGZvciBjcmVhdGluZywgcmVhZGluZywgdXBkYXRpbmcsIGFuZCBkZWxldGluZyBwcm90ZWluIGRhdGEsIGFzIHdlbGwgYXMgYWR2YW5jZWQgZmVhdHVyZXMgbGlrZSBhbiBBSSBjaGF0Ym90IGZvciBTUUwgcXVlcnkgZ2VuZXJhdGlvbiBhbmQgdmlzdWFsaXphdGlvbnMgdXNpbmcgVGhyZWUuanMuCgojIyBDaGVjayBvdXQgdGhlIGFwcGxpY2F0aW9uCi0gVUkgOiBodHRwczovL3d3dy5nZW5vcXVlcnkuY29tCi0gQVBJOiBodHRwczovL2FwaS1nZW4tYmlvbWVkLnBhcnRoYWRocnV2LmNvbS9kb2NzCgojIyBBcmNoaXRlY3R1cmUKCiFbU2FtcGxlIEFyY2hpdGVjdHVyZV0oLi9pbWFnZXMvU2FtcGxlQXJjaGl0ZWN0dXJlLnBuZykKCiMjIEZlYXR1cmVzCgotICoqQ1JVRCBPcGVyYXRpb25zKio6IENyZWF0ZSwgUmVhZCwgVXBkYXRlLCBhbmQgRGVsZXRlIHByb3RlaW4gZGF0YS4KLSAqKkNoYXRib3QqKjogSW50ZXJhY3Qgd2l0aCB0aGUgcHJvdGVpbiBkYXRhYmFzZSB1c2luZyBuYXR1cmFsIGxhbmd1YWdlLgotICoqVmlzdWFsaXphdGlvbnMqKjogM0QgdmlzdWFsaXphdGlvbnMgb2YgcHJvdGVpbiBkYXRhIHVzaW5nIFRocmVlLmpzLgotICoqQ2hhcnRzKio6IFZhcmlvdXMgY2hhcnRzIHRvIHJlcHJlc2VudCBwcm90ZWluIGRhdGEgc3RhdGlzdGljcy4KLSAqKlJlc3BvbnNpdmUgVUkqKjogQnVpbHQgd2l0aCBSZWFjdCwgVHlwZVNjcmlwdCwgVGFpbHdpbmQgQ1NTLCBhbmQgU2hhZGNuIFVJIGNvbXBvbmVudHMuCgojIyBUZWNobm9sb2dpZXMgVXNlZAoKLSAqKkJhY2tlbmQqKjogRmFzdEFQSSwgUG9zdGdyZVNRTCwgU1FMQWxjaGVteSwgUHlkYW50aWMKLSAqKkZyb250ZW5kKio6IFJlYWN0LCBUeXBlU2NyaXB0LCBUYWlsd2luZCBDU1MsICBDaGFydC5qcywgVGhyZWUuanMKLSAqKkFJIEludGVncmF0aW9uKio6IE9wZW5BSSBHUFQtNAotICoqRGVwbG95bWVudCoqOiBEb2NrZXIsIEFXUyBFbGFzdGljIEJlYW5zdGFsaywgQVdTIFJEUwoKIyMgU2NyZWVuc2hvdHMKCioqUHJvdGVpbnMgTGlzdCoqCgpUaGUgbWFpbiBsYW5kaW5nIHBhZ2UgZmVhdHVyZXMgYSBsaXN0IG9mIHByb3RlaW5zIHdpdGggY29sdW1ucyBmb3IgZW50cnkgSUQsIGxlbmd0aCwgZmlyc3Qgc2VlbiBkYXRlLCBsYXN0IHNlZW4gZGF0ZSwgUEZBTSwgU01BUlQsIGFuZCBhdmVyYWdlIGh5ZHJvcGhvYmljaXR5LiBFYWNoIHJvdyBpbmNsdWRlcyAiRWRpdCIgYW5kICJEZWxldGUiIGJ1dHRvbnMgZm9yIG1hbmFnaW5nIHByb3RlaW4gcmVjb3Jkcy4KIVtQcm90ZWlucyBMaXN0XSguL2ltYWdlcy9sYW5kaW5nUGFnZS5wbmcpCgoqKlByb3RlaW4gU3RydWN0dXJlIFZpc3VhbGl6YXRpb246KioKClRoaXMgc2NyZWVuc2hvdCBzaG93cyB0aGUgIjNEIFZpeiIgdGFiLCB3aGVyZSB0aGUgM0QgdmlzdWFsaXphdGlvbiBvZiB0aGUgcHJvdGVpbiBzdHJ1Y3R1cmUgaXMgcmVuZGVyZWQuIFRoZSB2aXN1YWxpemF0aW9uIHVzZXMgZGlmZmVyZW50IGNvbG9ycyB0byByZXByZXNlbnQgdmFyaW91cyBlbGVtZW50cyBhbmQgc3RydWN0dXJlcyB3aXRoaW4gdGhlIHByb3RlaW4sIHByb3ZpZGluZyBhbiBpbnRlcmFjdGl2ZSB2aWV3LgohW1N0cnVjdHVyZVZpel0oLi9pbWFnZXMvcHJvdGVpblN0cnVjdHVyZVZpei5wbmcpCgoqKlNRTCBDaGF0IEJvdDoqKgoKVGhlIGNoYXQgaW50ZXJmYWNlIGFsbG93cyB1c2VycyB0byBpbnRlcmFjdCB3aXRoIGEgU1FMLWJhc2VkIGNoYXRib3QuIFVzZXJzIGNhbiB0eXBlIHF1ZXJpZXMgcmVsYXRlZCB0byBwcm90ZWluIGRhdGEsIGFuZCB0aGUgY2hhdGJvdCByZXNwb25kcyB3aXRoIHRoZSByZXF1ZXN0ZWQgaW5mb3JtYXRpb24uIFRoZSBpbnRlcmZhY2Ugc2hvd3MgYm90aCB0aGUgdXNlcidzIGlucHV0IGFuZCB0aGUgY2hhdGJvdCdzIHJlc3BvbnNlcywgZm9ybWF0dGVkIGluIGEgY29udmVyc2F0aW9uYWwgc3R5bGUuCiFbQUlDaGF0Qm90XSgvaW1hZ2VzL1NRTENoYXRCb3QucG5nKQoKKioyRCBWaXN1YWxpemF0aW9uczoqKgoKVGhlIHNjcmVlbnNob3Qgc2hvd3MgYSB0YWJiZWQgaW50ZXJmYWNlIGZvciB2aXN1YWxpemluZyBwcm90ZWluIGRhdGEuIFRoZSBhY3RpdmUgdGFiLCAiQW1pbm8gQWNpZCBDb21wb3NpdGlvbiwiIGRpc3BsYXlzIGEgYmFyIGNoYXJ0IHJlcHJlc2VudGluZyB0aGUgcGVyY2VudGFnZSBjb21wb3NpdGlvbiBvZiB2YXJpb3VzIGFtaW5vIGFjaWRzIGluIHRoZSBwcm90ZWluLiBPdGhlciB0YWJzIGluY2x1ZGUgIlNlY29uZGFyeSBTdHJ1Y3R1cmUgQ29tcG9zaXRpb24iIGFuZCAiSHlkcm9waG9iaWNpdHksIiBwcm92aWRpbmcgZGlmZmVyZW50IGRhdGEgdmlzdWFsaXphdGlvbnMuCiFbMkQgVmlzdWFsaXphdGlvbnNdKGltYWdlcy8yZFZpc3VhbGl6YXRpb25zLnBuZykKCgoKIyMgR2V0dGluZyBTdGFydGVkCgojIyMgUHJlcmVxdWlzaXRlcwoKLSBEb2NrZXIKLSBOb2RlLmpzCi0gUHl0aG9uIDMuOSsKLSBQb3N0Z3JlU1FMCgojIyMgRm9sZGVyIFN0cnVjdHVyZQoKYGBgc2hlbGwKLgrilJzilIDilIAgTElDRU5TRQrilJzilIDilIAgUkVBRE1FLm1kCuKUnOKUgOKUgCBpbWFnZXMgOiBUaGlzIGZvbGRlciBjb250YWlucyBzY3JlZW5zaG90cyBmb3IgdGhlIHJlYWRtZSBmaWxlCuKUlOKUgOKUgCBwYWNrYWdlCiAgICDilJzilIDilIAgYXBpIDogQ29kZSBmb3IgdGhlIEFQSSBzZXJ2aWNlCiAgICDilJzilIDilIAgZGFzaGJvYXJkcyA6IENvZGUgZm9yIHRoZSBmcm9udGVuZCAKICAgIOKUlOKUgOKUgCBkYXRhTG9hZGluZyA6IFRvb2xzIHRvIHNldCB1cCBsb2NhbCBkYXRhYmFzZQpgYGAKCgojIyMgRW52aXJvbm1lbnQgVmFyaWFibGVzCgojIyMjIEZyb250bmVkCgpDcmVhdGUgYSBgLmVudmAgZmlsZSBpbiB0aGUgZm9sbG93aW5nIHBhdGggYHBhY2thZ2UvZGFzaGJvYXJkcy9wcm90ZWluLWRhc2hib2FyZC8uZW52YCB3aXRoIHRoZSBmb2xsb3dpbmcgY29udGVudDogCgpgYGBlbnYKUkVBQ1RfQVBQX0FQSV9VUkw9aHR0cDovLzEyNy4wLjAuMTo4MDAwCmBgYAoKIyMjIyBCYWNrZW5kCgpDcmVhdGUgYSBgLmVudmAgZmlsZSBpbiB0aGUgZm9sbG93aW5nIHBhdGggYHBhY2thZ2UvYXBpLy5lbnZgIHdpdGggdGhlIGZvbGxvd2luZyBjb250ZW50OgoKYGBgZW52Ck9QRU5BSV9BUElfS0VZPXlvdXJfb3BlbmFpX2FwaV9rZXkKREFUQUJBU0VfVVJMPXlvdXJfZGF0YWJhc2VfdXJsCmBgYAoKIyMjIExvY2FsIERldmVsb3BtZW50CgoxLiAqKkJhY2tlbmQgU2V0dXAqKjoKCiAgICBgYGBzaAogICAgY2QgcGFja2FnZS9hcGkKICAgIGRvY2tlciBidWlsZCAtdCBwcm90ZWluLWRhc2hib2FyZC1iYWNrZW5kIC4KICAgIGRvY2tlciBydW4gLS1lbnYtZmlsZSAuZW52IC1wIDgwMDA6ODAwMCAtdiAuLzovYXBwIHByb3RlaW4tZGFzaGJvYXJkLWJhY2tlbmQKICAgIGBgYAoKMi4gKipGcm9udGVuZCBTZXR1cCoqOgoKICAgIGBgYHNoCiAgICBjZCBwYWNrYWdlL2Rhc2hib2FyZHMvcHJvdGVpbi1kYXNoYm9hcmQKICAgIG5wbSBpbnN0YWxsCiAgICBucG0gc3RhcnQKICAgIGBgYAoKIyMjIEFQSSBFbmRwb2ludHMKCi0gKipHRVQgL3Byb3RlaW5zLyoqOiBSZXRyaWV2ZSBhbGwgcHJvdGVpbnMKLSAqKkdFVCAvcHJvdGVpbnMve2VudHJ5fSoqOiBSZXRyaWV2ZSBhIHByb3RlaW4gYnkgZW50cnkKLSAqKlBPU1QgL3Byb3RlaW5zLyoqOiBDcmVhdGUgYSBuZXcgcHJvdGVpbgotICoqUFVUIC9wcm90ZWlucy97ZW50cnl9Kio6IFVwZGF0ZSBhIHByb3RlaW4KLSAqKkRFTEVURSAvcHJvdGVpbnMve2VudHJ5fSoqOiBEZWxldGUgYSBwcm90ZWluCi0gKipHRVQgL3Byb3RlaW4tc3RhdHMve2VudHJ5fSoqOiBHZXQgc3RhdGlzdGljcyBmb3IgYSBzcGVjaWZpYyBwcm90ZWluCi0gKipQT1NUIC9xdWVyeS8qKjogR2VuZXJhdGUgU1FMIHF1ZXJ5IHVzaW5nIE9wZW5BSQoKIyMjIERlcGxveW1lbnQKCjEuICoqQnVpbGQgYW5kIFB1c2ggRG9ja2VyIEltYWdlKio6CgogICAgYGBgc2gKICAgIGRvY2tlciBidWlsZCAtdCB5b3VyX2RvY2tlcmh1Yl91c2VybmFtZS9wcm90ZWluLWRhc2hib2FyZC1iYWNrZW5kIC4KICAgIGRvY2tlciBwdXNoIHlvdXJfZG9ja2VyaHViX3VzZXJuYW1lL3Byb3RlaW4tZGFzaGJvYXJkLWJhY2tlbmQKICAgIGBgYAoKMi4gKipEZXBsb3kgb24gQVdTIEVsYXN0aWMgQmVhbnN0YWxrKio6CgogICAgLSBDcmVhdGUgYW4gRWxhc3RpYyBCZWFuc3RhbGsgZW52aXJvbm1lbnQKICAgIC0gQ29uZmlndXJlIHRoZSBlbnZpcm9ubWVudCB0byB1c2UgRG9ja2VyCiAgICAtIFNldCB1cCBlbnZpcm9ubWVudCB2YXJpYWJsZXMgaW4gdGhlIEVsYXN0aWMgQmVhbnN0YWxrIGNvbmZpZ3VyYXRpb24KCiMjIyBUcm91Ymxlc2hvb3RpbmcKCi0gKipDT1JTIElzc3VlcyoqOiBFbnN1cmUgQ09SUyBtaWRkbGV3YXJlIGlzIGNvcnJlY3RseSBzZXQgdXAgaW4gRmFzdEFQSS4KLSAqKlNTTCBUZXJtaW5hdGlvbioqOiBNYWtlIHN1cmUgdGhlIGxvYWQgYmFsYW5jZXIgaXMgcHJvcGVybHkgY29uZmlndXJlZCBmb3IgSFRUUFMuCgojIyMgQ29udGFjdAoKRm9yIGFueSBxdWVzdGlvbnMgb3IgaXNzdWVzLCBwbGVhc2UgY29udGFjdCBbcGFydGhhc2FyYXRoeS5kQG5vcnRoZWFzdGVybi5lZHVdIG9yIGZlZWwgZnJlZSB0byByZWFjaCBvdXQgdG8gbWUgb24gW0xpbmtlZEluXShodHRwczovL3d3dy5saW5rZWRpbi5jb20vaW4vcGFydGhhZGhydXYvKS4KCiMjIExpY2Vuc2UKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4K readmeEtag: '"f3f4b5420fd5309dd242800c9fc69d04bbe49f08"' readmeLastModified: Thu, 19 Sep 2024 01:30:42 GMT repositoryId: 806921406 description: >- A comprehensive web application designed to interact with a protein database. It includes functionalities for creating, reading, updating, and deleting protein data, as well as advanced features like an AI chatbot for SQL query generation and visualizations using Three.js. created: '2024-05-28T06:50:21Z' updated: '2025-11-05T19:58:52Z' language: Python archived: false stars: 5 watchers: 1 forks: 3 owner: parthasarathydNU logo: https://avatars.githubusercontent.com/u/113069126?v=4 license: MIT repoEtag: '"f2f1b9c9fe72548218746714905e2b2c26d359089a89ec663352c5f9ebb416a9"' repoLastModified: Wed, 05 Nov 2025 19:58:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/tranvannhat/fastapi-starter v3: true id: ecf41122d4f260bbb4e59408750e1878 repositoryMetadata: base64Readme: >- IyMgUmVxdWlyZW1lbnRzCi0gUHl0aG9uIDMuOQotIFBvc3RncmVzU1FMIDE2Ci0gUmVkaXMgCgoKIyMgSW5zdGFsbApJbnN0YWxsIFBvZXRyeSAtIFRvb2wgZm9yIGRlcGVuZGVuY3kgbWFuYWdlbWVudCBhbmQgcGFja2FnaW5nIGluIFB5dGhvbgoKYGBgCnBvZXRyeSBpbnN0YWxsCnBvZXRyeSBzaGVsbApgYGAKCmBgYAphbGVtYmljIHN0YW1wIGhlYWQKYWxlbWJpYyByZXZpc2lvbiAtLWF1dG9nZW5lcmF0ZSAtbSAidXBkYXRlIGRiIiAKYWxlbWJpYyB1cGdyYWRlIGhlYWQKYGBgCgpgYGAKc2ggc3RhcnQuc2gKYGBgCg== readmeEtag: '"65c9f419b8d4917d2d255323fe1c48a16468b6b2"' readmeLastModified: Wed, 09 Oct 2024 09:14:12 GMT repositoryId: 382431783 description: >- FastAPI Starter for init project with FastAPI, Postgresql, Redis. Base on https://github.com/tiangolo/full-stack-fastapi-postgresql created: '2021-07-02T18:29:11Z' updated: '2024-10-09T09:14:36Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: tranvannhat logo: https://avatars.githubusercontent.com/u/28804941?v=4 repoEtag: '"a545b173f21d195a2efe21ba49a572bea11a3e2e08141347ca406ea77b452936"' repoLastModified: Wed, 09 Oct 2024 09:14:36 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/traefik-workshops/traefik-hub-gitops v3: true id: c363fcdd6ceeb10d9194ee33be5a1f1f repositoryMetadata: base64Readme: >- <br/>

> Since the end of May 2024, Traefik Hub tutorials has moved to [traefik/hub](https://github.com/traefik/hub) repository

<div align="center" style="margin: 30px;">
<a href="https://hub.traefik.io/">
  <img src="https://doc.traefik.io/traefik-hub/img/hub-logo-light.svg" style="width:250px;" align="center" />
</a>
<br />
<br />

<div align="center">
    <a href="https://hub.traefik.io">Log In</a> |
    <a href="https://doc.traefik.io/traefik-hub/">Documentation</a>
</div>
</div>

<br />

<div align="center"><strong>Traefik Hub - GitOps Complete Tutorial</strong>

<br />
<br />
</div>

<div align="center">Welcome to this tutorial!</div>

# About

This tutorial details how to deploy locally a Kubernetes cluster, with GitOps and observability components.

# Prerequisites

If you'd like to follow along with this tutorial on your own machine, you'll need a few things first:

1. [kubectl](https://github.com/kubernetes/kubectl) command-line tool installed and configured to access the cluster
2. [gh](https://cli.github.com/) command-line tool installed and configured with your account
3. [Flux CD](https://fluxcd.io/flux/cmd/) command-line tool installed.
4. A Kubernetes cluster running.

In this tutorial, you'll use [kind](https://kind.sigs.k8s.io). You may also use alternatives like [k3d](https://k3d.io/), cloud providers, or others.

kind requires some configuration to use an IngressController on localhost, see the following example:

```shell
cat kind.config
```

```yaml
kind: Cluster
apiVersion: kind.x-k8s.io/v1alpha4
name: traefik-hub-gitops
nodes:
- role: control-plane
  extraPortMappings:
  - containerPort: 30000
    hostPort: 80
    protocol: TCP
  - containerPort: 30001
    hostPort: 443
    protocol: TCP
```

1. Clone this GitHub repository:

```shell
git clone https://github.com/traefik-workshops/traefik-hub-gitops.git
cd traefik-hub-gitops
```

2. Create the Kubernetes cluster:

**Using kind**

Create the cluster:

```shell
kind create cluster --config=kind.config
kubectl cluster-info
kubectl wait --for=condition=ready nodes traefik-hub-gitops-control-plane
```

Add a load balancer (LB) to this Kubernetes cluster:

```shell
kubectl apply -f https://raw.githubusercontent.com/metallb/metallb/v0.13.11/config/manifests/metallb-native.yaml
kubectl wait --namespace metallb-system --for=condition=ready pod --selector=app=metallb --timeout=90s
kubectl apply -f clusters/kind/metallb-config.yaml
```

**Using k3d**

```shell
k3d cluster create traefik-hub-gitops --port 80:80@loadbalancer --port 443:443@loadbalancer --port 8000:8000@loadbalancer --k3s-arg "--disable=traefik@server:0"
```

# Fork the repo and deploy Flux CD

Flux needs to be able to commit on the repository, this tutorial can only work on a fork that *you* own.
You will also need a [GitHub personal access token](https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/managing-your-personal-access-tokens) (PAT) with administration permissions.

```shell
gh repo fork --remote
kubectl apply -f clusters/kind/flux-system/gotk-components.yaml
```

# Configure Traefik Hub

Login to the [Traefik Hub UI](https://hub.traefik.io), open the page to [generate a new agent](https://hub.traefik.io/agents/new).
**Do not install the agent, but copy your token.**

Now, open a terminal and run these commands to create the secret needed for Traefik Hub.

```shell
export TRAEFIK_HUB_TOKEN=xxx
kubectl create namespace traefik-hub
kubectl create secret generic hub-agent-token --namespace traefik-hub --from-literal=token=${TRAEFIK_HUB_TOKEN}
```

# Deploy the stack on the cluster

Then, you can configure flux and launch it on the fork.
You'll need to create a [Personal Access Token](https://docs.github.com/en/authentication/keeping-your-account-and-data-secure/managing-your-personal-access-tokens#creating-a-personal-access-token-classic)(PAT) with repo access on your fork.

First, export your GitHub username and your newly created PAT into a variable.

```shell
export GITHUB_ACCOUNT=xxx
export GITHUB_TOKEN=yyy
```

Second, configure Flux CD for your fork of this tutorial.

```shell
flux create secret git git-auth  --url="https://github.com/${GITHUB_ACCOUNT}/traefik-hub-gitops" --namespace=flux-system -u git -p "${GITHUB_TOKEN}"
```

Third, adjust the repository on Flux to use your fork.

```shell
sed -i -e "s/traefik-workshops/${GITHUB_ACCOUNT}/g" clusters/kind/flux-system/gotk-sync.yaml
git commit -m "feat: GitOps on my fork" clusters/kind/flux-system/gotk-sync.yaml
git push origin
```

In the next step, deploy the repository.

```shell
kubectl apply -f clusters/kind/flux-system/gotk-sync.yaml
```

This will start the [Kustomization](https://fluxcd.io/flux/components/kustomize/kustomizations/).
Flux will now sync, validate and deploy all components.

This process will take several minutes.

You can track the process from the CLI.

```shell
flux get ks
```

![Kustomizations are ready](./images/kustomizations-ready.png)

If any of the kustomizations are stuck just run the:

```bash
kubectl delete mutatingwebhookconfigurations hub-api
kubectl delete mutatingwebhookconfigurations hub-acp
```

# Configure traffic generation

To generate traffic, create two users, an `admin` and a `support` user and their groups.

The `admin` user needs to be part of the `admin` group and the `support` user needs to be part of the `support` group.

<details>
  <summary>Traefik Hub UI</summary>

Create the `admin` user in the Traefik Hub UI:

![Create user admin](./images/create-user-admin.png)

Create the `support` user:

![Create user support](./images/create-user-support.png)

</details>

When the Kustomization is **ready**, you can open API Portal, following URL available in the UI or in the CRD:

```shell
kubectl get apiportal -n apps
```

Log in to the API Portal:

![API Portal Login](./images/api-portal-login.png)

Now, create API tokens for both users.

![Create API Token](./images/create-api-token.png)

Create a [Kubernetes secret](https://kubernetes.io/docs/concepts/configuration/secret/) containing the tokens to enable the traffic app to generate load:

```shell
export ADMIN_TOKEN="xxx"
export SUPPORT_TOKEN="yyy"
kubectl create secret generic tokens -n traffic --from-literal=admin="${ADMIN_TOKEN}" --from-literal=support="${SUPPORT_TOKEN}"
```

# Use observability stack

Grafana is accessible on http://grafana.docker.localhost

Prometheus is on http://prometheus.docker.localhost

Default credentials are login: admin and password: admin.

By clicking on the left menu, you can access all dashboards:

![Grafana Dashboards](./images/grafana-dashboards.png)

## Enable event correlation

**Flux events**:

- Add a new Grafana service account with a new key at http://grafana.docker.localhost/org/serviceaccounts
- Create a new Service Account with `Editor` role.
- Add a Service account token and copy it to (starting with `glsa_`) to the `apps/base/monitoring/flux-grafana.yaml` file.
- **NOTE**: You will need to apply or commit this to the repo, if it is a public repo beware that the secret will be public.

```shell
export GRAFANA_TOKEN="xxx"
sed -i -e "s/token:\(.*\)/token: \""${GRAFANA_TOKEN}"\"/g" apps/base/monitoring/flux-grafana.yaml
```

Now, Flux can create annotations of reconciliation events in the dashboards.

**GitHub PR merges**: Add a new Grafana connection (GitHub data source) at http://grafana.docker.localhost/connections/datasources/grafana-github-datasource
You must add a Personal Access Token (PAT) to GitHub at https://github.com/settings/tokens/new, as explained in the Grafana connection creation wizard.
When configured, edit the dashboards to add new annotations using the GitHub data source:
- Show in: All panels
- Query type: Pull Requests
- Owner / Repository: your user and the repo name of the fork
- Query: leave it empty
- Time Field: MergedAt
- Annotations:
  - Time field: `merged_at (time)`
  - Title: `title (string)`
  - Text: `url (string)`
  - End time + tags + id: leave them empty

# Clean up

Everything is installed in Kubernetes.
You can delete the Kubernetes cluster with the following command:

```shell
# kind
kind delete cluster --name traefik-hub-gitops
```

```shell
# k3d
k3d delete cluster traefik-hub-gitops
```

You may also want to delete the agent in Traefik Hub UI.

## License

The content in this repository is licensed under the [Apache 2 License](https://www.apache.org/licenses/LICENSE-2.0 "Link to Apache 2 license").
 readmeEtag: '"209925bd54ff1c96dffabe989697235826dc97fc"' readmeLastModified: Mon, 16 Sep 2024 06:40:47 GMT repositoryId: 711843464 description: Complete demo of Traefik Hub usage with FluxCD created: '2023-10-30T09:37:00Z' updated: '2025-01-13T11:36:27Z' language: Go archived: true stars: 1 watchers: 10 forks: 10 owner: traefik-workshops logo: https://avatars.githubusercontent.com/u/98564823?v=4 license: Apache-2.0 repoEtag: '"5936f46b05300fd15db0b2e4adbe6bd7d1b96ca5a49ce899d5417cda73aa7e79"' repoLastModified: Mon, 13 Jan 2025 11:36:27 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/bepopov/autoshowroom v3: true repositoryMetadata: base64Readme: >- IyMgR2V0dGluZyBTdGFydGVkCgpEb3dubG9hZCBzb3VyY2VzOgpgYGAKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9Cb2dkYW5LRlUvYXV0b3Nob3dyb29tLmdpdApgYGAKQ3JlYXRlIGRhdGFiYXNlIGFuZCBhZGQgeW91ciBwcm9wZXJ0aWVzIHRvIGFwcGxpY2F0aW9uLnByb3BlcnRpZXMuCkJ1aWxkIHNlcnZlciBieSB1c2luZyBidWlsZCB0YXNrLiBSdW4gaXQuIFVzZToKYGBgCmphdmEgLWphcgpgYGAKQnVpbGQgc29hcGNsaWVudCBvciByZXN0Y2xpZW50LiBSdW4gaXQuCkNvbm5lY3QgdG8gdGhlIHNlcnZlciBieSB1c2luZyAoYXZhaWxhYmxlIG5hbWVzIGFyZSBpbiB0aGUgY2xpZW50LmNzdiBmaWxlKToKYGBgCmNvbm5lY3QgPG5hbWU+CmBgYApUaGVyZSBhcmUgZmV3IGNvbW1hbmRzOgpgYGAKY3JlYXRlLW9yZGVyIC0tb3JkZXIgIjxtYWtlX25hbWU+IDxtb2RlbF9uYW1lPiA8b3B0aW9uc19uYW1lcyBzZXBhcmF0ZWQgYnkgQU5EIgplZGl0LW9yZGVyIC0tb3JkZXIgPG9yZGVyX2lkPiAtLW1ha2UgPG1ha2VfbmFtZT4gLS1vcHRpb25zIDxvcHRpb25zX25hbWVzIHNlcGFyYXRlZCBieSBBTkQ+CmRlbGV0ZS1vcmRlciA8b3JkZXJfaWQ+CmdldC1vcmRlcnMKZ2V0LW9yZGVycy1ieS1zdGF0dXMgLS1zdGF0dXMgPHN0YXR1cz4KYGBgCkFsbCB2YWx1ZXMgYXJlIGNvbXBsZXRhYmxlIChwcmVzcyBUQUIpCgojIyBSdW5uaW5nIHNlcnZlciB1c2luZyBEb2NrZXIKCkdvIHRvIHRoZSBzZXJ2ZXIgbW9kdWxlIHBhdGguCgpSdW46CmBgYApncmFkbGUgYnVpbGQKYGBgClRoZW4gcnVuOgpgYGAKZG9ja2VyLWNvbXBvc2UgdXAKYGBgCkNoZWNrIHNlcnZlciBpcyBydW5uaW5nIGJ5IGdvaW5nIHRvIHRoZSBwYWdlOiBsb2NhbGhvc3Q6OTAwMAojIyBEZXBsb3kgYWxsCkJlZm9yZSBzdGFydGluZyB0byBkZXBsb3kgY3JlYXRlIGNsdXN0ZXIgaW4gc29tZSBwbGF0Zm9ybSAoSSB1c2VkIEdvb2dsZSBDbG91ZCkgYW5kIGVuc3VyZSB0aGF0CnlvdSBoYXZlIGluc3RhbGxlZCBnY2xvdWQsIGt1YmVjdGwsIGhlbG0gaW4geW91ciBtYWNoaW5lLiBEb24ndCBmb3JnZXQgY29uZmlndXJlIHRoZSB0b29scy4KLSBidWlsZCB0aGUgYXBwbGljYXRpb25zIGFuZCBEb2NrZXIgaW1hZ2VzCihbX2plbmtpbnMtc2xhdmVfXShodHRwczovL2dpdGh1Yi5jb20vYmVwb3Bvdi9hdXRvc2hvd3Jvb20vdHJlZS9tYXN0ZXIvc2VydmVyL29wdC9qZW5raW5zL3NsYXZlKSwgCltfYXV0b3Nob3dyb29tLXNlcnZlcl9dKGh0dHBzOi8vZ2l0aHViLmNvbS9iZXBvcG92L2F1dG9zaG93cm9vbS90cmVlL21hc3Rlci9zZXJ2ZXIpLApbX2F1dG9zaG93cm9vbS1ncm9vdnktc2VydmVyX10oaHR0cHM6Ly9naXRodWIuY29tL2JlcG9wb3YvYXV0b3Nob3dyb29tL3RyZWUvbWFzdGVyL2dyb292eV9zZXJ2ZXIpKS4gVG8gZGVwbG95IApKZW5raW5zIHlvdSBjYW4gYWxzbyB1c2UgW2FsdGVybmF0aXZlIHdheV0oaHR0cHM6Ly9naXRodWIuY29tL2JlcG9wb3YvYXV0b3Nob3dyb29tL3RyZWUvbWFzdGVyL3NlcnZlci9vcHMvamVua2lucy9tYXN0ZXIpCiB0aGF0IGlzIHJlY29tbWVuZGVkIGFzIGl0IGhhcyBiZXR0ZXIgcGVyZm9ybWFuY2U6CmBgYApncmFkbGUgYnVpbGQKZG9ja2VyIGJ1aWxkIC10IFtTT1VSQ0VfSU1BR0VdIC4KYGBgCi0gcHVzaCB0aGUgaW1hZ2VzIHRvIHRoZSBkb2NrZXIgcmVnaXN0cnk7CmBgYApkb2NrZXIgdGFnIFtTT1VSQ0VfSU1BR0VdIFtIT1NUTkFNRV0vW1BST0pFQ1QtSURdL1tJTUFHRV0KZG9ja2VyIHB1c2ggW0hPU1ROQU1FXS9bUFJPSkVDVC1JRF0vW0lNQUdFXQpgYGAKLSBhcHBseSBgdGlsbGVyLXJiYWMueW1sYDoKYGBgCmt1YmVjdGwgYXBwbHkgLWYgdGlsbGVyLXJiYWMueW1sCmhlbG0gaW5pdCAtLXNlcnZpY2UtYWNjb3VudCB0aWxsZXIgLS1oaXN0b3J5LW1heCAyMDAKYGBgCi0gZGVwbG95CltjYXNzYW5kcmFdKGh0dHBzOi8vZ2l0aHViLmNvbS9iZXBvcG92L2F1dG9zaG93cm9vbS90cmVlL21hc3Rlci9ncm9vdnlfc2VydmVyL29wcy9jYXNzYW5kcmEpCmJ5IHJ1bm5pbmcgYGBgaW5zdGFsbC5zaGBgYAotIGRlcGxveSBqZW5raW5zIHVzaW5nIHRoZSAKW2luc3RydWN0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL2JlcG9wb3YvYXV0b3Nob3dyb29tL3RyZWUvbWFzdGVyL3NlcnZlci9vcHQvamVua2lucykKLSBhcHBseQpbYXV0b3Nob3dyb29tLXNlcnZlci55bWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9iZXBvcG92L2F1dG9zaG93cm9vbS9ibG9iL21hc3Rlci9zZXJ2ZXIvYXV0b3Nob3dyb29tLXNlcnZlci55bWwpCmFuZApbYXV0b3Nob3dyb29tLWdyb292eS1zZXJ2ZXIueW1sXShodHRwczovL2dpdGh1Yi5jb20vYmVwb3Bvdi9hdXRvc2hvd3Jvb20vYmxvYi9tYXN0ZXIvZ3Jvb3Z5X3NlcnZlci9hdXRvc2hvd3Jvb20tZ3Jvb3Z5LXNlcnZlci55bWwpCihjaGFuZ2UgaW1hZ2UgbmFtZXMgdG8geW91cnMgYmVmb3JlKQoKVG8gdXNlIGxvY2FsbHkgZGVwbG95ZWQgYXBwbGljYXRpb25zIHlvdSBjYW4gcnVuIGBgYGt1YmVjdGwgcG9ydC1mb3J3YXJkYGBgIGluIHlvdXIgdGVybWluYWwu readmeEtag: '"0788a72833f0c65da988f0954b6d986acd9394be"' readmeLastModified: Sat, 06 May 2023 14:47:50 GMT repositoryId: 179305875 description: >- Autoshowroom is a training project consists in REST and SOAP server, REST-client written using Spring Shell, SOAP-client written on the previous way and gRPC-server written on Groovy. This repository contains examples of usage Docker, Docker Compose, Kubernetes and also shows how to configure and deploy Jenkins-server created: '2019-04-03T14:20:16Z' updated: '2023-05-06T14:21:16Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: bepopov logo: https://avatars.githubusercontent.com/u/20209992?v=4 repoEtag: '"5cf8c3f777405fe986068dbe33a12f115042ad8b61f0d5b5a929b3b23a962df6"' repoLastModified: Sat, 06 May 2023 14:21:16 GMT foundInMaster: true category: Server Implementations id: 60dfc1ab39453d644a9e2cc366575d26 - source: openapi3 tags repository: https://github.com/microenv/node-typescript-api-boilerplate v3: true repositoryMetadata: base64Readme: IyBub2RlLXR5cGVzY3JpcHQtYm9pbGVycGxhdGU= readmeEtag: '"7ac6850a609af59783aa8bada4b581b18d6ca469"' readmeLastModified: Fri, 17 Apr 2020 01:57:13 GMT repositoryId: 256008749 description: >- 🚀 Boilerplate for typescript REST APIs ⭐Create production ready APIs using es7 and jest unit testing. created: '2020-04-15T18:51:33Z' updated: '2020-04-17T04:30:38Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 1 owner: microenv logo: https://avatars.githubusercontent.com/u/39780224?v=4 license: MIT repoEtag: '"f45194c107023795fe13d5eff62083e6b2276e23c8b5c261a693a78797b444ec"' repoLastModified: Fri, 17 Apr 2020 04:30:38 GMT foundInMaster: true category: - Data Validators - Parsers - Server Implementations id: f8a46fc254f10a5309dfb203c41ade0c - source: openapi3 tags repository: https://github.com/zhuzean/bookkeeping-demo v3: true repositoryMetadata: base64Readme: >- IyBCb29ra2VlcGluZyBQcm9qZWN0ClRoaXMgaXMgYSBkZW1vIHByb2plY3QgZm9yIGJvb2trZWVwaW5nIGFwcGxpY2F0aW9uLgpJdCBpcyBiYXNlZCBvbiB0aGUgVnVlIDMsIEZhc3RBUEkgYW5kIE5naW54LgoKIyMgV29ya2Zsb3cKTWFuYWdlZCBieSB0aGUgR2l0SHViIFtLYW5iYW4gYm9hcmRdKGh0dHBzOi8vZ2l0aHViLmNvbS9aaHVaZWFuL2Jvb2trZWVwaW5nLWRlbW8vcHJvamVjdHMvMSkKCiMjIERlcGxveW1lbnQgd2l0aCBLdWJlcm5ldGVzIDpzdGFyMjoKU2VlIG1vcmUgZGV0YWlscyBpbiB0aGlzIFtwYWdlXShodHRwczovL2dpdGh1Yi5jb20vWmh1WmVhbi9ib29ra2VlcGluZy1kZW1vL3RyZWUvbWFpbi9rOHMpCgojIyBQcmV2aWV3CiMjIyBPbmxpbmUgZGVtbwpQcm90b3R5cGUKLSBbRmlnbWEgbGlua10oaHR0cHM6Ly93d3cuZmlnbWEuY29tL2ZpbGUvSVl4N1dCQUc5SE9zTnphZ05WQnVCTi9QZXJzb25hbC1ib29ra2VlcGluZy13ZWI/bm9kZS1pZD0wJTNBMSkKCkZyb250ZW5kOgotIFdlYiBwYWdlOiBodHRwOi8vZGVtby5rOHMuemVhbi5wcm8KCkJhY2tlbmQ6Ci0gT3BlbkFQSSBzY2hlbWE6IGh0dHA6Ly9hcGkuazhzLnplYW4ucHJvL2JpbGwvZG9jcwoKIyMjIEZyb250ZW5kIChWdWUgMykKSXQgc2hvd3MgaG93IHRvIGFkZCBhIG5ldyByZWNvcmQgb24gdGhlIGJvb2trZWVwaW5nIGZvciBjdXJyZW50IG1vbnRoIGR5bmFtaWNhbGx5LgoKIVtpbWFnZV0oaHR0cHM6Ly9naXRodWIuY29tL1podVplYW4vYm9va2tlZXBpbmctZGVtby9ibG9iL21haW4vcHJldmlldy9mcm9udGVuZC9kZW1vLmdpZikKCiMjIyBCYWNrZW5kIChGYXN0QVBJKQpGYXN0QVBJIGlzIHVzZWQgdG8gbWFrZSB0aGUgUmVzdGZ1bCBBUElzLgoKIyMjIyBPcGVuQVBJIHNjaGVtYQohW2ltYWdlXShodHRwczovL2dpdGh1Yi5jb20vWmh1WmVhbi9ib29ra2VlcGluZy1kZW1vL2Jsb2IvbWFpbi9wcmV2aWV3L2JhY2tlbmQvb3BlbmFwaS5wbmcpCgoKIyMgUnVuIGluIHRoZSBsb2NhbCBtYWNoaW5lCkNoZWNrIHRoaXMgW3BhZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9aaHVaZWFuL2Jvb2trZWVwaW5nLWRlbW8vdHJlZS9tYWluL2s4cykgYWJvdXQgaG93IHRvIGRlcGxveSBpdCB3aXRoIG1pbmlrdWJlIGluIHRoZSBsb2NhbCBtYWNoaW5lCg== readmeEtag: '"d952e51d146c816a3a1b176df412a6cdec249448"' readmeLastModified: Sun, 11 Apr 2021 11:17:42 GMT repositoryId: 343094001 description: A demo project for bookkeeping created: '2021-02-28T11:58:42Z' updated: '2022-03-19T12:44:32Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: ZhuZean logo: https://avatars.githubusercontent.com/u/15837299?v=4 repoEtag: '"d1f838595063eeb8604afb969c2086c38983c113040ffa574da1ad3d7a6ba2dd"' repoLastModified: Sat, 19 Mar 2022 12:44:32 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: 6d257cc38757411c1860231b4655e0d7 - source: openapi3 tags repository: https://github.com/mrredbobr/open-api-change-logger v3: true repositoryMetadata: base64Readme: >- IyMjIE9wZW5BcGkgY2hhbmdlLWxvZ2dlcgpgYGB0eXBlc2NyaXB0CmltcG9ydCB7IENoYW5nZUxvZ2dlciB9IGZyb20gIm9wZW4tYXBpLWNoYW5nZS1sb2dnZXIiOwppbXBvcnQgc291cmNlIGZyb20gIi4vc291cmNlLmpzb24iOwppbXBvcnQgZGVzdGluYXRpb24gZnJvbSAiLi9kZXN0aW5hdGlvbi5qc29uIjsKaW1wb3J0IHsgT3BlbkFQSU9iamVjdCB9IGZyb20gIkBuZXN0anMvc3dhZ2dlci9kaXN0L2ludGVyZmFjZXMiOwppbXBvcnQgKiBhcyBwYXRoIGZyb20gInBhdGgiOwoKY29uc3QgY2hhbmdlTG9nZ2VyOiBDaGFuZ2VMb2dnZXIgPSBuZXcgQ2hhbmdlTG9nZ2VyKHsKICBvbGRTY2hlbWE6IHNvdXJjZSBhcyBPcGVuQVBJT2JqZWN0LAogIG5ld1NjaGVtYTogZGVzdGluYXRpb24gYXMgT3BlbkFQSU9iamVjdCwKfSk7CgpjaGFuZ2VMb2dnZXIuc2F2ZUZpbGVzKHsKICBodG1sRm9sZGVyOiBwYXRoLmpvaW4oX19kaXJuYW1lLCAnZmlsZXMnLCAnaHRtbCcpLAogIHN0eWxlc0ZvbGRlcjogcGF0aC5qb2luKF9fZGlybmFtZSwgJ2ZpbGVzJywgJ3N0eWxlJyksCiAgYXBpTmFtZTogJ3Rlc3QtYXBpJwp9KTsKYGBgCg== readmeEtag: '"6266572b2327228f4599bfaa76fcf287b39a2717"' readmeLastModified: Sat, 22 Apr 2023 17:00:10 GMT repositoryId: 418848216 description: Generates HTML-page with api changes. created: '2021-10-19T09:03:57Z' updated: '2023-04-22T17:00:16Z' language: CSS archived: false stars: 1 watchers: 1 forks: 0 owner: MrRedBobr logo: https://avatars.githubusercontent.com/u/46565863?v=4 repoEtag: '"235cce4c0435d56a55e4b1d292087bad76116746828b944d8eae885b25557852"' repoLastModified: Sat, 22 Apr 2023 17:00:16 GMT foundInMaster: true category: - Data Validators - Parsers id: c6c4a1cf5612c2410ec7d22a2b5a10c5 - source: openapi3 tags repository: https://github.com/opticsquid/bellatrix v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+DQogICA8aW1nIHNyYz0iLi9vbGQtdmVyc2lvbi9QdWJsaWMvYmVsbGF0cml4JTIwc29saWQucG5nIiBhbHQ9IkJlbGxhdHJpeCBMb2dvIj4NCjwvcD4NCg0KIyBCRUxMQVRSSVgNCg0KQSBOb2RlIEpTIHByb2plY3QgdG8gcHJvdmlkZSB5b3Ugd2l0aCBhIHN0YXJ0ZXIgY29kZSBmb3IgeW91ciBzZXJ2ZXIgc2lkZSBhcHBsaWNhdGlvbiB3aXRoIGJ1aWx0IGluIHJvdXRlcyBhbmQgZnVuY3Rpb25hbGl0aWVzIGZvciB0aGUgYXV0aGVudGljYXRpb24gYW5kIGF1dGhvcml6YXRpb24gbmVlZHMgb2YgeW91ciBhcHBsaWNhdGlvbi4gV2UgbWFpbmx5IGZvY3VzIG9uIHRoZSAqKnNlY3VyaXR5Kiogb2YgdGhlIGF1dGhlbnRpY2F0aW9uIGFuZCBhdXRob3JpemF0aW9uIHJvdXRlcw0KDQpXZSBrZWVwIGltcGxlbWVudGluZyBhbGwgdGhlIGxhdGVzdCBzZWN1cml0eSBtZWFzdXJlcyBhbmQgYmVzdCBwcmFjdGljZXMgdGhhdCBhcmUgcmVxdWlyZWQgZm9yIHRoZSBhcHBsaWNhdGlvbidzIGRhdGEgdG8gcmVtYWluIHNlY3VyZSBhbmQgbWFrZSBzdXJlIGRhdGEgYW5kIGFjY2VzcyBpcyBhdmFpbGFibGUgdG8gb25seSB0aGUgb25lcyBpbnRlbmRlZCB0by4gV2UgZG8gdGhlIGJvcmluZyBzZWN1cml0eSBzdHVmZiBzbyB0aGF0IHlvdSBjYW4gZW5qb3kgeW91ciB0aW1lIGJ1aWxkaW5nIG90aGVyIGV4Y2l0aW5nIGZ1bmN0aW9uYWxpdGllcyB3aXRoIG91dCB3b3JyeWluZyBhYm91dCB0aGUgc2VjdXJpdHkgb2YgeW91ciBhcHBsaWNhdGlvbg0KDQojIFdlIGhhdmUgMiB2ZXJzaW9ucyBmb3IgdGhpcyBhcHANCg0KMS4gVGhlIG9sZCB2ZXJzaW9uIHdoaWNoIGlzIGJ1aWx0IHVzaW5nIGV4cHJlc3MuanMuIEFzIGV4cHJlc3MganMgaXMgYSBtaW5pbWFsaXN0IGZyYW1ld29yayBhIGxvdCBvZiBzdHJ1Y3R1cmUgYW5kIHNlY3VyaXR5IGNvdWxkIG5vdCBiZSBwcm92aWRlZCB0byB0aGUgYXBwLiBTbywgaXQgbWF5IGJlIHN1c2NlcHRpYmxlIHRvIHNvbWUgYXR0YWNrcw0KMi4gVGhlIG5ldyB2ZXJzaW9uLiBUaGlzIGlzIGJlaW5nIGJ1aWx0IHVzaW5nIGBuZXN0LmpzYC4gV2hpY2ggaXMgYSB2ZXJ5IHN0YWJsZSBvcGluaW9uYXRlZCBmcmFtZXdvcmsgdGhhdCBnaXZlcyBhIGxvdCBvZiBzdHJ1Y3R1cmUgdG8gdGhlIGNvZGUgYW5kIGZvbGxvd3MgT09QIGFuZCBNVkMgbW9kZWwuIEJ5IGRlZmF1bHQgaXQgYnJpbmdzIGEgbG90IG9mIHNlY3VyaXR5IHRvIHRoZSB0YWJsZSBhbmQgaXMgbXVjaCBtb3JlIG9yZ2FuaXplZCB3aGljaCBoZWxwcyB0aGUgY29kZSBzdWl0YWJsZSBmb3IgaW1wbGVtZW50YXRpb24gaW4gbGFyZ2Ugc2NhbGUgYXBwbGljYXRpb25zLg0KDQoqKkJlbG93IGFyZSB0aGUgc3RlcHMgdG8gYnVpbGQgdGhlIG9sZCB2ZXJzaW9uLiBUaGUgZG9jdW1lbnRhdGlvbiBmb3IgdGhlIG5ldyB2ZXJzaW9uIGlzIGluIHRoZSBbbmV3LXZlcnNpb25dKC4vbmV3LXZlcnNpb24vKSBmb2xkZXIuKioNCg0KPiBXZSBmb2xsb3cgc29tZSBiYXNpYyBydWxlcyBmb3IgdGhlIHNlY3VyaXR5IG9mIHRoZSBhcHAuDQoNCi0gV2UgdXNlIHRoZSBbSldUXShodHRwczovL2p3dC5pby8pIGZvciB0aGUgYXV0aGVudGljYXRpb24gYW5kIGF1dGhvcml6YXRpb24gb2YgdGhlIGFwcGxpY2F0aW9uDQoNCi0gV2UgdXNlIGFuIGFwcHJvYWNoIG9mIHRpbWUgYmFzZWQgdG9rZW4gZm9yIHRoZSBhdXRoZW50aWNhdGlvbiBhbmQgYXV0aG9yaXphdGlvbiBvZiB0aGUgYXBwbGljYXRpb24NCi0gRm9yIHRoaXMgdXNlciBnZXRzIDIgdG9rZW5zIGFmdGVyIHN1Y2Nlc3NmdWwgbG9naW4uIE9uZSBpcyBjYWxsZWQgYGFjY2Vzc190b2tlbmAgKioodmFsaWQgZm9yIDE1IG1pbnMpKiogYW5kIHRoZSBvdGhlciBpcyBjYWxsZWQgYHJlZnJlc2hfdG9rZW5gDQotIEV2ZXJ5IHRpbWUgdGhlIHVzZXIgbWFrZXMgYSByZXF1ZXN0IHRvIGEgcHJvdGVjdGVkIHJvdXRlLCB0aGUgYGFjY2Vzc190b2tlbmAgc2hvdWxkIGJlIHNlbnQgaW4gdGhlIGBBdXRob3JpemF0aW9uYCBoZWFkZXINCg0KLSBFdmVyeXRpbWUgdGhlIGBhY2Nlc3NfdG9rZW5gIGV4cGlyZXMsIHRoZSB1c2VyIGNhbiByZXF1ZXN0IGEgbmV3IGBhY2Nlc3NfdG9rZW5gIGJ5IHNlbmRpbmcgYSAqKkdFVCoqIHJlcXVlc3QgdG8gYFx0b2tlbmAgcm91dGUuIFRoZSBgcmVmcmVzaF90b2tlbmAgc2hvdWxkIGJlIHByb3ZpZGVkIGluIHRoZSBgQXV0aG9yaXphdGlvbmAgaGVhZGVyIG9mIHRoZSByZXF1ZXN0DQoNCj4gV2hpbGUga2VlcGluZyBhbiBleWUgb24gdGhlIHNlY3VyaXR5IHdlIGFsc28gZm9jdXMgb24gcGVyZm9ybWFuY2UsIHNjYWxhYmlsaXR5IGFuZCByZXVzYWJsaXR5IG9mIHRoZSBjb2RlIGJ5IGJyZWFraW5nIHRoZSBjb2RlIHRvIHNtYWxsIG1pZGRsZXdhcmVzIGFuZCBhc3luY2hyb25pemluZyB0aGUgcHJvY2Vzc2VzIGFzIG11Y2ggYXMgcG9zc2libGUuIFdlIHBhcmFsbGVsaXplIHRoZSBwcm9taXNlcyBhbmQgbmV0d29yayByZXF1ZXN0cyB3aGVuIGV2ZXIgcG9zc2libGUgdG8gbWFrZSB0aGUgYXBwbGljYXRpb24gbW9yZSBwZXJmb3JtYW50IHNjYWxhYmxlIGFuZCByZXVzYWJsZS4NCg0KIyMgU3RlcHMgdG8gZ2V0IHN0YXJ0ZWQNCg0KMS4gQ2xvbmUgdGhlIF9CZWxsYXRyaXhfIHJlcG9zaXRvcnkuDQoyLiBNYWtlIHN1cmUgdG8gaW5zdGFsbCB0aGUgbGF0ZXN0ICoqTFRTIHZlcnNpb24gb2YgTm9kZSBKUyoqDQozLiBBZnRlciBuYXZpZ2F0aW5nIHRvIHRoZSBwcm9qZWN0IHJvb3QgZGlyZWN0b3J5LCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOg0KDQogICBgYGBzaA0KICAgbnBtIGluc3RhbGwNCiAgIGBgYA0KDQo0LiBDcmVhdGUgYSBgLmVudmAgZmlsZSBpbiB0aGUgcm9vdCBvZiB0aGUgcHJvamVjdC4gVGhpcyBmaWxlIHdpbGwgY29udGFpbiB0aGUgZW52aXJvbm1lbnQgdmFyaWFibGVzIHJlcXVpcmVkIGZvciB0aGUgYXBwbGljYXRpb24gdG8gcnVuLg0KICAgVGhlIGZpbGUgc2hvdWxkIGNvbnRhaW4gdGhlIGZvbGxvd2luZyBmaWVsZHM6DQoNCiAgIC0gYERCX1VSSWAgLSBfVVJJIG9mIHRoZSBNb25nb0RCIGRhdGFiYXNlXy4NCiAgIC0gYEpXVF9BQ1NfU0VDUkVUYCAtIF9TZWNyZXQga2V5IGZvciB0aGUgSldUIGBhY2Nlc3NfdG9rZW5gXw0KICAgLSBgSldUX1JFRl9TRUNSRVRgIC0gX1NlY3JldCBrZXkgZm9yIHRoZSBKV1QgYHJlZnJlc2hfdG9rZW5gXw0KDQo1LiBSdW4gdGhlIGZvbGxvd2luZyBjb21tYW5kIHRvIHN0YXJ0IHRoZSBhcHBsaWNhdGlvbiBpbiBwcm9kdWN0aW9uIG1vZGU6DQoNCiAgIGBgYHNoDQogICBucG0gc3RhcnQNCiAgIGBgYA0KDQo2LiBUbyBzdGFydCB0aGUgYXBwbGljYXRpb24gaW4gZGV2ZWxvcG1lbnQgbW9kZSwgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoNCg0KICAgYGBgc2gNCiAgIG5wbSBydW4gZGV2DQogICBgYGANCg== readmeEtag: '"23438bbeb18b2a21446bb7bdf1125b6a17fea313"' readmeLastModified: Sun, 09 Oct 2022 11:21:44 GMT repositoryId: 437827006 description: >- A prebuilt Express JS Authentication & Authorization Project based on a REST API interface. It has the basic authentication and Authorization routes with the latest security measures implemented so that your application is much more secure from day 1. You are welcome to build upon and extend this project as and when required. created: '2021-12-13T10:21:01Z' updated: '2023-04-30T11:25:55Z' language: JavaScript archived: false stars: 1 watchers: 1 forks: 0 owner: opticSquid logo: https://avatars.githubusercontent.com/u/59192460?v=4 license: GPL-3.0 repoEtag: '"acd58769cf8a0ef99d96cc59652e495b33dfec3a314f16876f0e6f15023a380e"' repoLastModified: Sun, 30 Apr 2023 11:25:55 GMT foundInMaster: true category: Server Implementations id: cc0bf508eefe56e6f6892b0955c5ec0c oldLocations: - https://github.com/soumalyatheking22012001/Bellatrix - source: openapi3 tags repository: https://github.com/thekeogh/loopback-postgresql-graphql v3: true id: 730a635e026c886ae1e32749e8c73735 repositoryMetadata: base64Readme: >- IyBMb29wQmFjayA0IFBvc3RncmVTUUwgR3JhcGhRTCBCb2lsZXJwbGF0ZQoKQSBib2lsZXJwbGF0ZS9zdGFydGVyIHByb2plY3QgZm9yIHF1aWNrbHkgYnVpbGRpbmcgR3JhcGhRTCBBUElzIHVzaW5nIE5vZGUuanMsIExvb3BCYWNrIDQgYW5kIFBvc3RncmVTUUwuCgojIyBQcmVyZXF1aXNpdGVzCgotIE5vZGUgKGJ1aWx0IHdpdGggYDE5YCwgYnV0IHNob3VsZCB3b3JrIG9uIGxvd2VyKQotIFBvc3RncmVTUUwgKGxvY2FsbHksIGV4dGVybmFsbHkgb3IgZG9ja2VyKQoKIyMgU2V0dXAKCkNsb25lIHRoZSByZXBvc2l0b3J5CgpgYGBzaGVsbApnaXQgY2xvbmUgZ2l0QGdpdGh1Yi5jb206dGhla2VvZ2gvbG9vcGJhY2stcG9zdGdyZXNxbC1ncmFwaHFsLmdpdApgYGAKCkluc3RhbGwgdGhlIGRlcGVuZGVuY2llcwoKYGBgc2hlbGwKbnBtIGluc3RhbGwKYGBgCgpDcmVhdGUgYSBsb2NhbCBgLmVudmAgCgpgYGBzaGVsbAp0b3VjaCAuZW52LmRldmVsb3BtZW50LmxvY2FsCmBgYAoKQ29weSB0aGUgdmFsdWVzIChvbmx5IHRoZSBvbmVzIHlvdSBuZWVkIHRvIGNoYW5nZSkgZnJvbSB0aGUgYC5lbnYuZGV2ZWxvcG1lbnRgIGZpbGUgdG8gdGhpcyBuZXcgZmlsZSwgZm9yIGV4YW1wbGU6CgpgYGBzaGVsbAojIERhdGFiYXNlCkRCX1VTRVI9am9lCkRCX1BBU1NXT1JEPW15c3VwZXJwYXNzd29yZApEQl9EQVRBQkFTRT1sb29wYmFjawpgYGAKCj4gTW9yZSBkZXRhaWxzIG9uIHRoZSBlbnZpcm9ubWVudCBjb25maWcgW2JlbG93XSgjZW52aXJvbm1lbnQpLgoKTWlncmF0ZSB0aGUgZGF0YWJhc2UKCj4gU2VlIHRoZSBbZGF0YWJhc2VdKCNkYXRhYmFzZSkgc2VjdGlvbiBmb3IgdGhpcyBzdGVwLgoKU3RhcnQgdGhlIEFQSQoKYGBgc2hlbGwKbnBtIHN0YXJ0CiMgb3IgZm9yIGRldiAodXNpbmcgdHNjLXdhdGNoKQpucG0gcnVuIGRldgpgYGAKCllvdSBzaG91bGQgbm93IHNlZSB0aGUgcmVsZXZhbnQgVVJMcyBpbiB5b3VyIHRlcm1pbmFsLgoKIyMgRGF0YWJhc2UKCkFsdGhvdWdoIGRlc2lnbmVkIHdpdGggUG9zdGdyZVNRTCBpbiBtaW5kLCB0aGVyZSBpcyBubyByZWFzb24gd2h5IHlvdSBjYW5ub3QgY2hhbmdlIHRoZSBkYXRhc291cmNlIHRvIGFub3RoZXIgKGUuZy4gTWFyaWFEQiwgT3JhY2xlLCBNb25nbyBldGMpLCBpdCBpcyBMb29wQmFjayBhZnRlciBhbGwsIGFuZCBoYXMgbWFueSBkcml2ZXJzLiBCdXQgd2UgdXNlIFBvc3RncmVTUUwgZm9yIHRoaXMgQVBJCgpPbmNlIHlvdSBoYXZlIGVudGVyZWQgeW91ciBkYXRhYmFzZSBjcmVkZW50aWFscyBbYWJvdmVdKCNzZXR1cCksIFBvc3RncmVTUUwgaXMgcmVhZHkgdG8gZ28gb3V0LXRoZS1ib3guIFRoZXJlIGlzIGFuIGV4YW1wbGUgYHVzZXJgIG1vZGVsIGluIHRoZSBgbW9kZWxzL2AgZm9sZGVyIHJlYWR5IHRvIGdvLCB0byBtaWdyYXRlIHRoaXM6CgpgYGBzaGVsbApOT0RFX0VOVj1kZXZlbG9wbWVudCBucG0gcnVuIG1pZ3JhdGUKYGBgCgpOb3cgY2hlY2sgeW91ciBkYXRhYmFzZSBhbmQgeW91IHNob3VsZCBzZWUgYSBgdXNlcmAgdGFibGUuCgojIyBHcmFwaFFMCgpHcmFwaFFMIGlzIGJ1aWx0IGluIHRvIHRoZSBzZXJ2aWNlIHZpYSB0aGUgW29wZW5hcGktdG8tZ3JhcGhxbF0oaHR0cHM6Ly9naXRodWIuY29tL0lCTS9vcGVuYXBpLXRvLWdyYXBocWwpIGFuZCBbZ3JhcGhxbC1odHRwXShodHRwczovL2dpdGh1Yi5jb20vZ3JhcGhxbC9ncmFwaHFsLWh0dHApIGxpYnJhcmllcy4gVGhpcyBtZWFucywgdGhhdCBHcmFwaFFMIHdpbGwgcnVuIG9uIHRoZSBzYW1lIHBvcnQvdXJsIGFzIHlvdXIgT0FTMyBSRVNUZnVsIEFQSS4gRm9yIGV4YW1wbGU6CgpgYGAKUkVTVGZ1bDogaHR0cHM6Ly9sb2NhbGhvc3Q6MzIwMApHcmFwaFFMOiBodHRwczovL2xvY2FsaG9zdDozMjAwL2dyYXBocWwKYGBgCgo+IEdyYXBoaVFMIGlzIG5vdCBjdXJyZW50bHkgc3VwcG9ydGVkLgoKIyMgRW52aXJvbm1lbnQKCltkb3RlbnYtZmxvd10oaHR0cHM6Ly9naXRodWIuY29tL2tlcmltZHpoYW5vdi9kb3RlbnYtZmxvdykgaXMgdXNlZCBvdXQgdGhlIGJveCwgc28gcGxlYXNlIHJlYWQgdGhlIGRvY3VtZW50YXRpb24gb3ZlciB0aGVyZSBmb3IgbW9yZSBvbiB0aGlzLgoKSXQncyBhbHNvIGFkdmlzYWJsZSB0byBhZGQgYW55IG5ldyBlbnYgdmFyaWFibGVzIHRvIHRoZSBgc3JjL2Vudi5kLnRzYCBmaWxlIHRvIGFsbG93IFR5cGVTY3JpcHQgaW5zaWdodC4KCi0tLS0KClshW0xvb3BCYWNrXShodHRwczovL2dpdGh1Yi5jb20vbG9vcGJhY2tpby9sb29wYmFjay1uZXh0L3Jhdy9tYXN0ZXIvZG9jcy9zaXRlL2ltZ3MvYnJhbmRpbmcvUG93ZXJlZC1ieS1Mb29wQmFjay1CYWRnZS0oYmx1ZSktQDJ4LnBuZyldKGh0dHA6Ly9sb29wYmFjay5pby8pCg== readmeEtag: '"7de9d53ea1d913a02ffc4032f9cf039c53de59f0"' readmeLastModified: Tue, 11 Apr 2023 10:08:53 GMT repositoryId: 611844881 description: LoopBack 4 PostgreSQL GraphQL Boilerplate created: '2023-03-09T16:56:54Z' updated: '2023-04-21T09:46:44Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: thekeogh logo: https://avatars.githubusercontent.com/u/1387498?v=4 repoEtag: '"7e5bd9cbce22b56e08d5a505f66c7155a482c7efae9e44e5c30a58404815ae86"' repoLastModified: Fri, 21 Apr 2023 09:46:44 GMT category: - Server - Server Implementations foundInMaster: true oldLocations: - https://github.com/thekeogh/loopback-psql-graphql - source: openapi3 tags repository: https://github.com/nzlouislu/nzlouis-openapi v3: true repositoryMetadata: base64Readme: >- IyBVc2VyIFNlcnZpY2UNCkEgc2VydmljZSBmb3IgZmluZGluZyBhbGwgdXNlcnMsIGNyZWF0aW5nLCByZXRyaWV2aW5nLCB1cGRhdGluZywgYW5kIGRlbGV0aW5nIHVzZXJzLg0KDQpzd2FnZ2VyLXVpIHVybCA6IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9kb2MuaHRtbA0KDQpVc2Ugb2YgdGVjaG5vbG9neTogU3ByaW5nYm9vdDIuMyArIFNwcmluZ2RvYy1vcGVuYXBpIDMgKyBKUEEgKyBIMiArIEp1bml0NA0KDQoiIyBvcGVuYXBpIiANCg0KIVtdKHNyYy9tYWluL3Jlc291cmNlcy9vcGVuYXBpLmdpZik= readmeEtag: '"2fc8526d51da4ea6ef9dcaa2023f32a2ad978df6"' readmeLastModified: Tue, 28 Sep 2021 01:02:17 GMT repositoryId: 410410403 description: >- A service for finding all users, creating, retrieving, updating, and deleting users. created: '2021-09-26T00:21:46Z' updated: '2025-08-19T07:55:42Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: NZLouislu logo: https://avatars.githubusercontent.com/u/72653348?v=4 repoEtag: '"524b28ff065511bd25d9122175099acdfb528dfb684f560ce69464af2375285a"' repoLastModified: Tue, 19 Aug 2025 07:55:42 GMT foundInMaster: true category: - Testing - Server Implementations id: a6efeda45f202c4cdf160de794d557e2 oldLocations: - https://github.com/ailouislu/openapi - source: openapi3 tags repository: https://github.com/interserver/mailbaby-mail-api v3: true id: f1aa137f6555ec0919bc1cd98604137a repositoryMetadata: base64Readme: >- # MailBaby API

API service for accessing the Mail.Baby services.

# API Sample Clients

Sample clients for the API are available in many languages

* https://github.com/interserver/mailbaby-api-samples


# Documentation

* Lists
  * [OpenAPI Tools](https://openapi.tools/) Listing of OpenAPI relatd tools by category (Documentors, Parsers, Mockers, etc)
* Framework Docs
  * [WebMan Manual](https://www.workerman.net/doc/webman) WebMan framework documentation
  * [Illuminate Database Docs](https://laravel.com/docs/8.x/queries)
  * [PHPMailer](https://github.com/PHPMailer/PHPMailer/) email sending library for PHP
* API Spec
  * [raw.githubusercontent.com/interserver/mailbaby-mail-api/master/public/spec/openapi.json](https://raw.githubusercontent.com/interserver/mailbaby-mail-api/master/public/spec/openapi.json)
  * [mynew.interserver.net:8787/spec/openapi.json](http://mynew.interserver.net:8787/spec/openapi.json)
* API Client Sample Generators
  * [swagger-api/swagger-codegen: swagger-codegen contains a template-driven engine to generate documentation, API clients and server stubs in different languages by parsing your OpenAPI / Swagger definition.](https://github.com/swagger-api/swagger-codegen)
  * [OpenAPITools/openapi-generator-cli: A node package wrapper for https://github.com/OpenAPITools/openapi-generator](https://github.com/OpenAPITools/openapi-generator-cli)
    * [Generators List | OpenAPI Generator](https://openapi-generator.tech/docs/generators)
  * [Kong/httpsnippet: HTTP Request snippet generator for many languages & libraries](https://github.com/Kong/httpsnippet)
* Code Snippet and Example integration into SPEC
  * [cdwv/oas3-api-snippet-enricher: Enrich your OpenAPI 3.0 JSON with code samples](https://github.com/cdwv/oas3-api-snippet-enricher/)
  * [ErikWittern/openapi-snippet: Generates code snippets for given Swagger / Open API documents](https://github.com/ErikWittern/openapi-snippet)
  * Generated Code
  * [MailBaby generated PHP Client](https://github.com/interserver/mailbaby-client-php) PHP Client generated by the OpenAPI Generator/Swagger Codegen
* Our API Docs
  * [MailBaby Swagger-UI](https://api.mailbaby.net/doc/index.html)
  * [MailBaby API Documentation](https://www.mail.baby/apidoc.html#/operations/addRule)
  * [Swagger UI](http://mynew.interserver.net:8787/doc/index.html)
  * [MailBaby API Documentation](http://mynew.interserver.net:8787/elements.html#/)
  * [mynew.interserver.net:8787/rapidoc.html#overview](http://mynew.interserver.net:8787/rapidoc.html#get-/mail/rules)
  * [ReDoc](http://mynew.interserver.net:8787/redoc.html#tag/Services)
  * [mynew.interserver.net:8787/openapi-explorer.html#?route=get-/mail](http://mynew.interserver.net:8787/openapi-explorer.html#?route=post-/mail/advsend)
* API Editor
  * [SwaggerHub editor for MailBaby API Spec](https://app.swaggerhub.com/apis/InterServer/MailBaby/1.0.0)
  * [Apicurio Studio - API Editor :: MailBaby Email Delivery and Management Service API](https://studio.apicur.io/apis/93974/editor)
  * [Apicurio Studio - API Editor :: my InterServer Services Management API](https://studio.apicur.io/apis/93618/editor)
* PHP OpenAPI Libs
  * [canvural/php-openapi-faker: Library to generate fake data for OpenAPI request/response/schemas](https://github.com/canvural/php-openapi-faker)
  * [sunrise-php/http-router: :tada: Release 2.0 is released! Very fast HTTP router for PHP 7.1+ based on PSR-7 and PSR-15 with support for annotations/attributes and OpenAPI (Swagger) Specification](https://github.com/sunrise-php/http-router)
  * [cebe/php-openapi: Read and write OpenAPI yaml/json files and make the content accessible in PHP objects.](https://github.com/cebe/php-openapi)
  * [lezhnev74/openapi-psr7-validator: It validates PSR-7 messages (HTTP request/response) against OpenAPI specifications](https://github.com/lezhnev74/openapi-psr7-validator)
  * [erasys/openapi-php: 📚Swagger / Open API 3.0 builder and validation library for PHP that helps you write valid specs.](https://github.com/erasys/openapi-php)
  * [OpenAPI PSR-7 Message Validator](https://github.com/thephpleague/openapi-psr7-validator)
* API Doc Settings
  * [Swagger/OpenAPI Spec Documentation](https://swagger.io/docs/specification/describing-responses/)
  * [StopLight Elements](https://github.com/stoplightio/elements) OpenAPI Documentor
  * [rapi-doc/RapiDoc: RapiDoc -WebComponent for OpenAPI Spec](https://github.com/rapi-doc/RapiDoc)
  * [RapiDoc - Web Component based Swagger & OpenAPI Spec Viewer](https://rapidocweb.com/)
  * [stoplightio/studio: The modern editor for API Design and Technical Writing.](https://github.com/stoplightio/studio)
  * [Open-Source API Mocking, Linting, & Documentation | Stoplight](https://stoplight.io/open-source)
  * [RapiDoc - Web Component based Swagger & OpenAPI Spec Viewer](https://rapidocweb.com/api.html)
  * [RapiDoc API Demo](https://rapidocweb.com/examples/api-demo.html#post-/mail/advsend)

# Development

This is built on top of Webman, a high performance HTTP Service Framework for PHP based on [Workerman](https://github.com/walkor/workerman).

## API Specification

We are utilizing the OpenAPI 3 (formerly known as Swagger) spec for this API.  It is basically the next evolution of SOAP API's with well defined functions, parameters, and responses.  While there are many editors out there I'm currently using SwaggerHub to do most of the editing of the spec.

## Webman Framework

After testing *every* PHP library out there dealing with concurrent/asynchronous processing many times over the years I've found [Workerman](https://github.com/walkor/workerman) to be the overall best.  It has proven more stable and by far faster than the alternatives with the one big downside being that its documentation and code comments are all in Chinese.  [Webman](https://github.com/walkor/webman) is a fairly recently created web framework on top of [Workerman](https://github.com/walkor/workerman).  There had been many previously created frameworks based on workerman and while some were good Webman seemed to hit that perfect balance between ease-of-use and power.

The Chinese documentation is easily readable Using either the auto translate in Chrome or an addon like [Translate Web Pages](https://addons.mozilla.org/en-US/firefox/addon/traduzir-paginas-web/) for Firefox.
 readmeEtag: '"289c6508066bb72cc67dc93219ca73b4aa29ce87"' readmeLastModified: Fri, 24 May 2024 15:38:49 GMT repositoryId: 539355108 description: API Server for the MailBaby Email Services. created: '2022-09-21T07:13:45Z' updated: '2026-01-21T22:37:52Z' language: PHP archived: false stars: 2 watchers: 2 forks: 0 owner: interserver logo: https://avatars.githubusercontent.com/u/29833539?v=4 license: MIT repoEtag: '"598b369e25416e57e29ff8906bc1a87de764da3dbbd61d4c24205ab58a1f69c0"' repoLastModified: Wed, 21 Jan 2026 22:37:52 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/zafir100100/task-management-api-with-asp.net-core v3: true id: f9d0c786462c143acee41a905413a04a repositoryMetadata: base64Readme: >- IyBUYXNrLU1hbmFnZW1lbnQtQVBJLXdpdGgtQVNQLk5FVC1Db3JlCgojIyBJbnRyb2R1Y3Rpb24KClRhc2sgTWFuYWdlbWVudCBBUEkgd2l0aCBBU1AuTkVUIENvcmUgaXMgYSByb2J1c3QgYmFja2VuZCBzeXN0ZW0gZGVzaWduZWQgZm9yIG1hbmFnaW5nIHVzZXIgdGFza3MgdGhyb3VnaCBBUEkgZW5kcG9pbnRzLiBUaGlzIEFQSSBmYWNpbGl0YXRlcyB0aGUgY3JlYXRpb24sIHJldHJpZXZhbCwgdXBkYXRpbmcsIGFuZCBkZWxldGlvbiBvZiB0YXNrcywgcHJvdmlkaW5nIGEgY29tcHJlaGVuc2l2ZSBzb2x1dGlvbiBmb3IgdGFzay1yZWxhdGVkIGZ1bmN0aW9uYWxpdGllcy4KCiMjIFRlY2hub2xvZ3kgU3RhY2sKCi0gQVNQLk5FVCBDb3JlCi0gQyMKLSBFbnRpdHkgRnJhbWV3b3JrIENvcmUKLSBTd2FnZ2VyIChmb3IgQVBJIGRvY3VtZW50YXRpb24pCgojIyBGZWF0dXJlcwoKMS4gKipHZXQgQWxsIFRhc2tzKioKICAgLSAqKkVuZHBvaW50OioqIGBHRVQgL2FwaS92MS9Vc2VyVGFza3MvZ2V0LWFsbC10YXNrYAogICAtICoqRGVzY3JpcHRpb246KiogUmV0cmlldmUgYSBsaXN0IG9mIGFsbCB0YXNrcy4KCjIuICoqR2V0IFRhc2sgYnkgSUQqKgogICAtICoqRW5kcG9pbnQ6KiogYFBPU1QgL2FwaS92MS9Vc2VyVGFza3MvZ2V0LXRhc2stYnktaWRgCiAgIC0gKipEZXNjcmlwdGlvbjoqKiBSZXRyaWV2ZSBhIHRhc2sgYnkgaXRzIElELgogICAtICoqUmVxdWVzdCBCb2R5OioqIGBHZXRUYXNrRHRvYCBjb250YWluaW5nIHRoZSB0YXNrIElELgoKMy4gKipVcGRhdGUgVGFzayoqCiAgIC0gKipFbmRwb2ludDoqKiBgUEFUQ0ggL2FwaS92MS9Vc2VyVGFza3MvdXBkYXRlLXRhc2tgCiAgIC0gKipEZXNjcmlwdGlvbjoqKiBVcGRhdGUgYW4gZXhpc3RpbmcgdGFzay4KICAgLSAqKlJlcXVlc3QgQm9keToqKiBgVXNlclRhc2tgIHJlcHJlc2VudGluZyB0aGUgdXBkYXRlZCB0YXNrLgoKNC4gKipDcmVhdGUgVGFzayoqCiAgIC0gKipFbmRwb2ludDoqKiBgUE9TVCAvYXBpL3YxL1VzZXJUYXNrcy9jcmVhdGUtdGFza2AKICAgLSAqKkRlc2NyaXB0aW9uOioqIENyZWF0ZSBhIG5ldyB0YXNrLgogICAtICoqUmVxdWVzdCBCb2R5OioqIGBVc2VyVGFza2AgcmVwcmVzZW50aW5nIHRoZSBuZXcgdGFzay4KCjUuICoqRGVsZXRlIFRhc2sgYnkgSUQqKgogICAtICoqRW5kcG9pbnQ6KiogYERFTEVURSAvYXBpL3YxL1VzZXJUYXNrcy9kZWxldGUtdGFzay1ieS1pZGAKICAgLSAqKkRlc2NyaXB0aW9uOioqIERlbGV0ZSBhIHRhc2sgYnkgaXRzIElELgogICAtICoqUmVxdWVzdCBCb2R5OioqIGBEZWxldGVUYXNrUmVxdWVzdER0b2AgY29udGFpbmluZyB0aGUgdGFzayBJRC4KCiMjIEhvdyB0byBSdW4gdGhlIFByb2plY3QKCjEuICoqQ2xvbmUgdGhlIFByb2plY3Q6KioKICAgYGBgYmFzaAogICBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL3lvdXJ1c2VybmFtZS90YXNrLW1hbmFnZW1lbnQtYXBpLWRvdG5ldC5naXQKICAgYGBgCgoyLiAqKkluc3RhbGwgRGVwZW5kZW5jaWVzOioqCiAgIGBgYGJhc2gKICAgZG90bmV0IHJlc3RvcmUKICAgYGBgCgozLiAqKlJ1biB0aGUgQVBJOioqCiAgIGBgYGJhc2gKICAgZG90bmV0IHJ1bgogICBgYGAKCjQuICoqQWNjZXNzIFN3YWdnZXIgRG9jdW1lbnRhdGlvbjoqKgogICBPcGVuIHlvdXIgYnJvd3NlciBhbmQgbmF2aWdhdGUgdG8gYGh0dHBzOi8vbG9jYWxob3N0OjUwMDEvc3dhZ2dlci9pbmRleC5odG1sYCB0byBleHBsb3JlIHRoZSBBUEkgZW5kcG9pbnRzIGFuZCB0ZXN0IHRoZW0gaW50ZXJhY3RpdmVseS4KCiMjIENvbmNsdXNpb24KClRoZSBUYXNrIE1hbmFnZW1lbnQgQVBJIHdpdGggQVNQLk5FVCBDb3JlIHByb3ZpZGVzIGEgc29saWQgZm91bmRhdGlvbiBmb3IgaGFuZGxpbmcgdGFzay1yZWxhdGVkIG9wZXJhdGlvbnMgaW4gYSBzY2FsYWJsZSBhbmQgZWZmaWNpZW50IG1hbm5lci4gSXQgbGV2ZXJhZ2VzIHRoZSBwb3dlciBvZiBBU1AuTkVUIENvcmUsIEMjLCBhbmQgRW50aXR5IEZyYW1ld29yayBDb3JlIHRvIGRlbGl2ZXIgYSByZWxpYWJsZSBiYWNrZW5kIHNvbHV0aW9uIGZvciB0YXNrIG1hbmFnZW1lbnQuCgpGZWVsIGZyZWUgdG8gZXhwbG9yZSB0aGUgQVBJLCBydW4gdGVzdHMsIGFuZCBpbnRlZ3JhdGUgaXQgaW50byB5b3VyIGFwcGxpY2F0aW9ucy4gSWYgeW91IGhhdmUgYW55IHF1ZXN0aW9ucyBvciBuZWVkIGZ1cnRoZXIgYXNzaXN0YW5jZSwgcGxlYXNlIGNvbnRhY3QgbWUuCg== readmeEtag: '"c3cd0275d8ccc084b397c69bfe49eaaed65c6dc9"' readmeLastModified: Wed, 20 Dec 2023 17:00:15 GMT repositoryId: 733833893 description: >- Revolutionize task management with our ASP.NET Core API. Create, update, and delete tasks effortlessly using C#, Entity Framework Core, and Swagger. Elevate your daily note taking efficiency. created: '2023-12-20T08:30:16Z' updated: '2026-01-05T20:40:32Z' language: C# archived: false stars: 3 watchers: 1 forks: 0 owner: zafir100100 logo: https://avatars.githubusercontent.com/u/55599023?v=4 repoEtag: '"6a3ba19e8f43edef44c1a186f700b69ec9a3318a1c1ee8fd5b02beb80eea8dc4"' repoLastModified: Mon, 05 Jan 2026 20:40:32 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/dwellics/seljmov-aspnet-common v3: true id: b54d15b8d400b825c9b905af95f9f9a7 repositoryMetadata: base64Readme: IyBzZWxqbW92LWFzcG5ldC1jb21tb24= readmeEtag: '"8da36041f28d5cba16159f29a3e19e5d6f00eecd"' readmeLastModified: Wed, 10 Jul 2024 05:26:01 GMT repositoryId: 594170774 description: .NET Core hosting infrastructure for web apps. created: '2023-01-27T19:09:08Z' updated: '2025-03-16T11:54:43Z' language: C# archived: false stars: 1 watchers: 1 forks: 0 owner: dwellics logo: https://avatars.githubusercontent.com/u/190166700?v=4 repoEtag: '"64674d834b9237f85a34357e76f6252038c59657fbe0165ffb68220d985e0668"' repoLastModified: Sun, 16 Mar 2025 11:54:43 GMT category: Server foundInMaster: true oldLocations: - https://github.com/smartdwell/seljmov-aspnet-common - source: openapi3 tags repository: https://github.com/anfibiacreativa/contoso-services-typespec v3: true id: d67595587a50753676fac259324cc6ff repositoryMetadata: base64Readme: >- IyBjb250b3NvLXNlcnZpY2VzLXR5cGVzcGVjClR5cGVTcGVjIGdlbmVyYXRlZCBzZXJ2aWNlcwoKVGhpcyByZXBvIHVzZXMgaHR0cHM6Ly9taWNyb3NvZnQuZ2l0aHViLmlvL3R5cGVzcGVjLwoKClRvIGNyZWF0ZSB5b3VyIG93biBwcm9qZWN0LCBmb2xsb3cgdGhlc2Ugc3RlcHMuIChZb3UgbXVzdCBtYWtlIHN1cmUgdG8gaGF2ZSBbTm9kZS5qcyAxNiBMVFNdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi9ibG9nL3JlbGVhc2UvdjE2LjE2LjApIG9yIHVwLCBpbnN0YWxsZWQgaW4geW91ciBzeXN0ZW0sIGFuZCBhdCBsZWFzdCBbbnBtXShodHRwczovL3d3dy5ucG1qcy5jb20pIHZlcnNpb24gNysuCgojIyBTdGVwIDEgLSBJbnN0YWxsIHRoZSB0c3AgY29tcGlsZXIgcnVubmluZyBpbiB5b3VyIHRlcm1pbmFsCgpgYGAKJCBucG0gaW5zdGFsbCAtZyBAdHlwZXNwZWMvY29tcGlsZXIKYGBgCgojIyBTdGVwIDIgLSBDcmVhdGUgYSBkaXJlY3RvcnkgYW5kIGluaXRpYWxpemUgdHNwCgpgYGAKJCBta2RpciBteS10c3AtcHJvamVjdCAmJiBjZCBteS10c3AtcHJvamVjdCAmJiB0c3AgaW5pdApgYGAKV2hlbiBwcm9tcHRlZCwgc2VsZWN0IHRoZSBgR2VuZXJpYyBSZXN0IEFQSWAgdGVtcGxhdGUsIHlvdXIgcHJvamVjdCBuYW1lIGFuZCB0aGUgYEB0eXBlc3BlYy9vcGVuYXBpM2AgbGlicmFyeS4KCiMjIFN0ZXAgMyAtIEluc3RhbGwgdGhlIHRzcCBkZXBlbmRlbmNpZXMKClJ1biAKCmBgYAokIHRzcCBpbnN0YWxsCmBgYApUaGF0IHdpbGwgY3JlYXRlIGEgYmFzaWMgYHRzcGAgcHJvamVjdCBzdHJ1Y3R1cmUgdGhhdCBsb29rcyBsaWtlIHRoaXM6CgpgYGAKcGFja2FnZS5qc29uICAgICAjIFBhY2thZ2UgbWFuaWZlc3QgZGVmaW5pbmcgeW91ciB0eXBlc3BlYyBwcm9qZWN0IGFzIGEgbm9kZSBwYWNrYWdlLgp0c3Bjb25maWcueWFtbCAgICMgVHlwZVNwZWMgcHJvamVjdCBjb25maWd1cmF0aW9uIGxldHRpbmcgeW91IGNvbmZpZ3VyZSBlbWl0dGVycywgZW1pdHRlciBvcHRpb25zLCBjb21waWxlciBvcHRpb25zLCBldGMuCm1haW4udHNwICAgICAgICAgIyBUeXBlU3BlYyBlbnRyeXBvaW50LgpgYGAKRWRpdCB0aGUgYG1haW4udHNwYCBmaWxlIGFzIG5lY2Vzc2FyeSBmb3IgeW91ciBkZWZpbml0aW9uLgoKYGBgCiQgdHNwIGZvcm1hdCAqKi8qLnRzcCAjIHRoaXMgd2lsbCBmb3JtYXQgYWxsIC50c3AgZmlsZXMgaW4gdGhlIHByb2plY3QKYGBgCgpOb3cgeW91IGNhbiBjb21waWxlIHRoZSBwcm9qZWN0IHJ1bm5pbmcKCmBgYAokIHRzcCBjb21waWxlIC4KYGBgCgooaWYgeW91J3JlIGluIGEgZGlmZmVyZW50IGZvbGRlciB0aGFuIHRoZSBwcm9qZWN0IHJvb3QsIHBhc3MgdGhlIHBhdGggdG8gdGhlIGNvbXBpbGUgY29tbWFuZCkKCiMjIEVtaXQgdGhlIE9wZW5BUEkgc3BlYyBgeWFtbGAgZmlsZQoKVG8gZW1pdCB0aGUgc3BlY2lmaWNhdGlvbiBmaWxlLCBydW4KCmBgYAokIHRzcCBjb21waWxlIC4gLS1lbWl0PUB0eXBlc3BlYy9vcGVuYXBpMwpgYGAKCllvdSBzaG91bGQgaGF2ZSBhIG5ldyBmb2xkZXIgY2FsbGVkIGB0c3Atb3V0cHV0YCBieSBkZWZhdWx0LCB3aXRoIHlvdXIgc3BlYy4KCllvdSdyZSBhbGwgc2V0ISDwn5qACgoK readmeEtag: '"b8bb7f38b10f0ddaf554079ec3e124bb7afa5d6a"' readmeLastModified: Thu, 21 Sep 2023 11:10:13 GMT repositoryId: 688109520 description: >- Example of TypeSpec code to generate Contoso Real Estate API specification. created: '2023-09-06T17:08:34Z' updated: '2025-10-06T09:42:22Z' language: null archived: false stars: 1 watchers: 1 forks: 1 owner: anfibiacreativa logo: https://avatars.githubusercontent.com/u/4014025?v=4 repoEtag: '"b7144849bc393231e5cb3c9cae901c080294f930c943a4456512a0aab3bd6632"' repoLastModified: Mon, 06 Oct 2025 09:42:22 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/v3rmine/legifrance-api-swagger v3: true repositoryMetadata: base64Readme: >- IyBsZWdpZnJhbmNlLWFwaS1zd2FnZ2VyCgpTcMOpY2lmaWNhdGlvbnMgKE9wZW5BUEkgMy4wLjEpIGRlIGwnW0FQSSBCZXRhIDAuMy4xIGRlIEzDqWdpZnJhbmNlXShodHRwczovL2FwaS5nb3V2LmZyL2xlcy1hcGkvRElMQV9hcGlfTGVnaWZyYW5jZSkuCgpBY2Nlc3NpYmxlIGdyw6JjZSDDoCBbUmVkb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKSBpY2kgOiBbbGVnaWZyYW5jZS5yZXRhLnJlXShodHRwczovL2xlZ2lmcmFuY2UucmV0YS5yZS8pCg== readmeEtag: '"546a5a17b5d6d962e8f8a01ffe35d359c5165520"' readmeLastModified: Fri, 19 Feb 2021 13:46:06 GMT repositoryId: 277805259 description: DILA - Légifrance Beta == OpenAPI 3.0.1 created: '2020-07-07T12:04:50Z' updated: '2024-12-31T13:54:30Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: v3rmine logo: https://avatars.githubusercontent.com/u/11839373?v=4 repoEtag: '"eed123c67942e46e517f4883d60846b0021718256e5c3f7c47e84992471b3be6"' repoLastModified: Tue, 31 Dec 2024 13:54:30 GMT foundInMaster: true category: Code Generators id: 50fac8b036c677a3628ed680ccdcfb1c oldLocations: - https://github.com/joxcat/legifrance-api-swagger - source: openapi3 tags repository: https://github.com/newpointe/churchonline-api-docs v3: true repositoryMetadata: base64Readme: >- IyBDaHVyY2ggT25saW5lIFBsYXRmb3JtIEFQSSBEb2NzClVub2ZmaWNpYWwgQVBJIGRvY3MgZm9yIENodXJjaCBPbmxpbmUgUGxhdGZvcm0gc2l0ZXMuCgpXcml0dGVuIHVzaW5nIHRoZSBbT3BlbkFQSV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9hYm91dC8pIHNwZWNpZmljYXRpb24gYW5kIHByZXNlbnRlZCB1c2luZyBbU3dhZ2dlciBVSV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3N3YWdnZXItdWkvKS4KCiMjIFNlZSBpdCBpbiBhY3Rpb24gaGVyZTogaHR0cHM6Ly9uZXdwb2ludGUuZ2l0aHViLmlvL2NodXJjaG9ubGluZS1hcGktZG9jcwo= readmeEtag: '"b93da7d6a2b0e8beeaeacc5e7f72e1e5c512f8c2"' readmeLastModified: Thu, 17 Jan 2019 19:01:25 GMT repositoryId: 166279406 description: Swagger/OpenAPI docs for Church Online Platform sites created: '2019-01-17T18:57:18Z' updated: '2019-08-15T23:41:09Z' language: HTML archived: false stars: 1 watchers: 2 forks: 0 owner: NewPointe logo: https://avatars.githubusercontent.com/u/3997357?v=4 repoEtag: '"e2b313ba329e89e975642ec76f734ecc1326a7126471e087ddfa8b9f7c0151cf"' repoLastModified: Thu, 15 Aug 2019 23:41:09 GMT foundInMaster: true category: - Server - Server Implementations id: 050509852b4d1e8549e9f5a8dccd2288 - source: openapi3 tags repository: https://github.com/talentplatforms/dry_open_api v3: true repositoryMetadata: base64Readme: >- YGRyeV9vcGVuX2FwaWAKClByb3ZpZGluZyBhIFtkcnldKGh0dHBzOi8vZHJ5LXJiLm9yZy8pIHZlcnNpb24gb2YgdGhlIFtvcGVuX2FwaSBnZW1dKGh0dHBzOi8vZ2l0aHViLmNvbS9uZ3RrL29wZW5fYXBpKS4KIyMgSW5zdGFsbGF0aW9uCgpBZGQgdGhpcyBsaW5lIHRvIHlvdXIgYXBwbGljYXRpb24ncyBHZW1maWxlOgoKYGBgcnVieQpnZW0gJ2RyeV9vcGVuX2FwaScKYGBgCgpBbmQgdGhlbiBleGVjdXRlOgoKICAgICQgYnVuZGxlCgpPciBpbnN0YWxsIGl0IHlvdXJzZWxmIGFzOgoKICAgICQgZ2VtIGluc3RhbGwgZHJ5X29wZW5fYXBpCgojIyBVc2FnZQpJdCBoYXMgdHdvIHVzZSBjYXNlOgoKMS4gU2VyaWFsaXppbmcgZnJvbSBQT1JPIHRvIHlhbWwKMi4gRGVzZXJpYWxpemluZyB5YW1sIHRvIFBPUk8KCiMjIyBTZXJpYWxpemluZwoKYGBgcmIKc3BlYyA9IERyeU9wZW5BcGk6OlNwZWNpZmljYXRpb24ubmV3KAogIG9wZW5hcGk6ICIzLjAuMSIsCiAgaW5mbzogRHJ5T3BlbkFwaTo6SW5mby5uZXcoCiAgICB0aXRsZTogIkF3ZXNvbWUgQVBJIiwKICAgIGRlc2NyaXB0aW9uOiAiSXQgcHJvdmlkZXMgc29tZXRoaW5nIGF3ZXNvbWUiLAogICAgdmVyc2lvbjogIjEuMC4wIiwKICApLAogIHBhdGhzOiBEcnlPcGVuQXBpOjpQYXRocy5uZXcoCiAgICAiL3BldHMiOiBEcnlPcGVuQXBpOjpQYXRoSXRlbS5uZXcoCiAgICAgIGdldDogRHJ5T3BlbkFwaTo6T3BlcmF0aW9uLm5ldygKICAgICAgICBkZXNjcmlwdGlvbjogIlJldHVybnMgYWxsIHBldHMgZnJvbSB0aGUgc3lzdGVtIHRoYXQgdGhlIHVzZXIgaGFzIGFjY2VzcyB0byIsCiAgICAgICAgcmVzcG9uc2VzOiBEcnlPcGVuQXBpOjpSZXNwb25zZXMubmV3KAogICAgICAgICAgIjIwMCI6IERyeU9wZW5BcGk6OlJlc3BvbnNlLm5ldygKICAgICAgICAgICAgZGVzY3JpcHRpb246ICJBIGxpc3Qgb2YgcGV0cy4iLAogICAgICAgICAgICBjb250ZW50OiB7CiAgICAgICAgICAgICAgImFwcGxpY2F0aW9uL2pzb24iOiBEcnlPcGVuQXBpOjpNZWRpYVR5cGUubmV3KAogICAgICAgICAgICAgICAgc2NoZW1hOiBEcnlPcGVuQXBpOjpTY2hlbWEubmV3KAogICAgICAgICAgICAgICAgICB0eXBlOiAiYXJyYXkiLAogICAgICAgICAgICAgICAgICBpdGVtczogRHJ5T3BlbkFwaTo6UmVmZXJlbmNlLm5ldyhyZWY6ICIjL2NvbXBvbmVudHMvc2NoZW1hcy9wZXQiKSwKICAgICAgICAgICAgICAgICkKICAgICAgICAgICAgICApCiAgICAgICAgICAgIH0KICAgICAgICAgICkKICAgICAgICApCiAgICAgICkKICAgICkKICApCikKCnlhbWwgPSBEcnlPcGVuQXBpOjpTZXJpYWxpemVyczo6WWFtbFNlcmlhbGl6ZXIuc2VyaWFsaXplKHNwZWMpCkZpbGUud3JpdGUoInNwZWMueW1sIiwgeWFtbCkKYGBgCgojIyMgRGVzZXJpYWxpemluZwoKYGBgcmIKeWFtbCA9IEZpbGUucmVhZCgic3BlYy55bWwiKQpzcGVjID0gRHJ5T3BlbkFwaTo6U2VyaWFsaXplcnM6OllhbWxTZXJpYWxpemVyLmRlc2VyaWFsaXplKHlhbWwpCmBgYAoKIyMgRGV2ZWxvcG1lbnQKCkFmdGVyIGNoZWNraW5nIG91dCB0aGUgcmVwbywgcnVuIGBiaW4vc2V0dXBgIHRvIGluc3RhbGwgZGVwZW5kZW5jaWVzLiBUaGVuLCBydW4gYHJha2Ugc3BlY2AgdG8gcnVuIHRoZSB0ZXN0cy4gWW91IGNhbiBhbHNvIHJ1biBgYmluL2NvbnNvbGVgIGZvciBhbiBpbnRlcmFjdGl2ZSBwcm9tcHQgdGhhdCB3aWxsIGFsbG93IHlvdSB0byBleHBlcmltZW50LgoKVG8gaW5zdGFsbCB0aGlzIGdlbSBvbnRvIHlvdXIgbG9jYWwgbWFjaGluZSwgcnVuIGBidW5kbGUgZXhlYyByYWtlIGluc3RhbGxgLiBUbyByZWxlYXNlIGEgbmV3IHZlcnNpb24sIHJ1biBgYnVuZGxlIGV4ZWMgcmFrZSBidW1wOnBhdGNoYCB0byB1cGRhdGUgdGhlIHZlcnNpb24sIGFuZCB0aGVuIHJ1biBgYnVuZGxlIGV4ZWMgcmFrZSByZWxlYXNlYCwgd2hpY2ggd2lsbCBjcmVhdGUgYSBnaXQgdGFnIGZvciB0aGUgdmVyc2lvbiwgcHVzaCBnaXQgY29tbWl0cyBhbmQgdGFncywgYW5kIHB1c2ggdGhlIGAuZ2VtYCBmaWxlIHRvIFtydWJ5Z2Vtcy5vcmddKGh0dHBzOi8vcnVieWdlbXMub3JnKS4KCiMjIENvbnRyaWJ1dGluZwoKQnVnIHJlcG9ydHMgYW5kIHB1bGwgcmVxdWVzdHMgYXJlIHdlbGNvbWUgb24gR2l0SHViIGF0IGh0dHBzOi8vZ2l0aHViLmNvbS90YWxlbnRwbGF0Zm9ybXMvZHJ5X29wZW5fYXBpLiBUaGlzIHByb2plY3QgaXMgaW50ZW5kZWQgdG8gYmUgYSBzYWZlLCB3ZWxjb21pbmcgc3BhY2UgZm9yIGNvbGxhYm9yYXRpb24sIGFuZCBjb250cmlidXRvcnMgYXJlIGV4cGVjdGVkIHRvIGFkaGVyZSB0byB0aGUgW0NvbnRyaWJ1dG9yIENvdmVuYW50XShodHRwOi8vY29udHJpYnV0b3ItY292ZW5hbnQub3JnKSBjb2RlIG9mIGNvbmR1Y3QuCgojIyBMaWNlbnNlCgpUaGUgZ2VtIGlzIGF2YWlsYWJsZSBhcyBvcGVuIHNvdXJjZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIFtNSVQgTGljZW5zZV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQpLgoKIyMgQ29kZSBvZiBDb25kdWN0CgpFdmVyeW9uZSBpbnRlcmFjdGluZyBpbiB0aGUgT3BlbkFwaSBwcm9qZWN04oCZcyBjb2RlYmFzZXMsIGlzc3VlIHRyYWNrZXJzLCBjaGF0IHJvb21zIGFuZCBtYWlsaW5nIGxpc3RzIGlzIGV4cGVjdGVkIHRvIGZvbGxvdyB0aGUgW2NvZGUgb2YgY29uZHVjdF0oaHR0cHM6Ly9naXRodWIuY29tL3RhbGVudHBsYXRmb3Jtcy9kcnlfb3Blbl9hcGkvYmxvYi9tYXN0ZXIvQ09ERV9PRl9DT05EVUNULm1kKS4K readmeEtag: '"82347c74c617678c45b389659ba82679ef2b544f"' readmeLastModified: Wed, 04 Mar 2020 12:24:07 GMT repositoryId: 232815290 description: Dried up PORO OpenAPI for ruby created: '2020-01-09T13:26:50Z' updated: '2022-05-10T11:13:06Z' language: Ruby archived: false stars: 1 watchers: 1 forks: 0 owner: talentplatforms logo: https://avatars.githubusercontent.com/u/55129375?v=4 license: MIT repoEtag: '"c38b9490bb3a96fae377ee40a2b5f1f141ee92a224d3f44cb9eeb0021d064747"' repoLastModified: Tue, 10 May 2022 11:13:06 GMT foundInMaster: true category: Parsers id: 801120928b2cf065cc52048077e0828f - source: openapi3 tags repository: https://github.com/godknowsiamgood/oapi3gen v3: true repositoryMetadata: base64Readme: >- IyBvYXBpM2dlbgpTaW1wbGUgT3BlbkFQSSAzIGNvZGUgZ2VuZXJhdG9yIHdpdGggemVybyBvciBtaW5pbXVtIHNlcnZlciBib2lsZXJwbGF0ZQoKIyBGZWF0dXJlcwoqIFN0cm9uZ2x5IHR5cGVkIHBhcmFtZXRlcnMgYW5kIHJlc3BvbnNlcwoqIE5vIHNlcnZlciBib2lsZXJwbGF0ZSBpZiBub3QgbmVlZGVkCiogTWluaW11bSBzZXJ2ZXIgYm9pbGVycGxhdGUgLSBqdXN0IHJvdXRlcyBhbmQgcGFyYW1ldGVyIHZhbGlkYXRpb24KKiBFYXN5IG1pZGRsZXdhcmVzIHdpdGggY3VzdG9tIGB4LW1pZGRsZXdhcmVzYCBvYmplY3QKKiBFYXN5IGVycm9yIHJlc3BvbnNlcyAtIGp1c3QgYWRkIGBjb21wb25lbnRzL3NjaGVtYXMvRXJyb3JgIG9iamVjdAoKIyBVc2FnZQpgYGAKZ28gaW5zdGFsbCBnaXRodWIuY29tL2dvZGtub3dzaWFtZ29vZC9vYXBpM2dlbkBsYXRlc3QKb2FwaTNnZW4gWy1zZXJ2ZXIgZWNob10gWy1vdXRwdXQgLi9vdXQuZ29dIHNwZWMueWFtbApgYGAKCiMgQ29kZSBnZW5lcmF0aW9uClByb2dyYW0gZ2VuZXJhdGVzIG9ubHkgb25lIGZpbGUgd2l0aCBzdHJvbmdseSB0eXBlcyBmb3IgcmVxdWVzdCBwYXJhbWV0ZXJzLCByZXF1ZXN0IGJvZGllcyBhbmQgcmVzcG9uc2VzLgpBbHNvIGl0IGdlbmVyYXRlcyBpbnRlcmZhY2UgdHlwZSBmb3IgY29udHJvbGxlciB0aGF0IHlvdSBtYXkgaW1wbGVtZW50LgoKRS5nLiBzcGVjaWZpY2F0aW9uIGxpa2UKYGBgCiAgL3BldHM6CiAgICBnZXQ6CiAgICAgIHN1bW1hcnk6IExpc3QgYWxsIHBldHMKICAgICAgb3BlcmF0aW9uSWQ6IGxpc3RQZXRzCiAgICAgIHRhZ3M6CiAgICAgICAgLSBwZXRzCiAgICAgIHBhcmFtZXRlcnM6CiAgICAgICAgLSBuYW1lOiBsaW1pdAogICAgICAgICAgaW46IHF1ZXJ5CiAgICAgICAgICBkZXNjcmlwdGlvbjogSG93IG1hbnkgaXRlbXMgdG8gcmV0dXJuIGF0IG9uZSB0aW1lIChtYXggMTAwKQogICAgICAgICAgcmVxdWlyZWQ6IGZhbHNlCiAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgIHR5cGU6IGludGVnZXIKICAgICAgICAgICAgZm9ybWF0OiBpbnQzMgogICAgICByZXNwb25zZXM6CiAgICAgICAgJzIwMCc6CiAgICAgICAgICBkZXNjcmlwdGlvbjogQSBwYWdlZCBhcnJheSBvZiBwZXRzCiAgICAgICAgICBjb250ZW50OgogICAgICAgICAgICBhcHBsaWNhdGlvbi9qc29uOgogICAgICAgICAgICAgIHNjaGVtYToKICAgICAgICAgICAgICAgICRyZWY6ICIjL2NvbXBvbmVudHMvc2NoZW1hcy9QZXRzIgogICAgICAgIGRlZmF1bHQ6CiAgICAgICAgICBkZXNjcmlwdGlvbjogdW5leHBlY3RlZCBlcnJvcgogICAgICAgICAgY29udGVudDoKICAgICAgICAgICAgYXBwbGljYXRpb24vanNvbjoKICAgICAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgICAgICAkcmVmOiAiIy9jb21wb25lbnRzL3NjaGVtYXMvRXJyb3IiCmBgYAp3aWxsIGdlbmVyYXRlIGNvZGU6CmBgYAovLyBPYmplY3RzCnR5cGUgRXJyb3JTY2hlbWEgc3RydWN0IHsKCUNvZGUgICAgaW50MzIgIGBqc29uOiJjb2RlImAKCU1lc3NhZ2Ugc3RyaW5nIGBqc29uOiJtZXNzYWdlImAKfQoKdHlwZSBQZXRTY2hlbWEgc3RydWN0IHsKCUlkICAgaW50NjQgICBganNvbjoiaWQiYAoJTmFtZSBzdHJpbmcgIGBqc29uOiJuYW1lImAKCVRhZyAgKnN0cmluZyBganNvbjoidGFnImAKfQoKdHlwZSBQZXRzU2NoZW1hIFtdUGV0U2NoZW1hCgovLyBQYXJhbWV0ZXJzCnR5cGUgTGlzdFBldHNQYXJhbXMgc3RydWN0IHsKCUxpbWl0ICppbnQzMgp9CgovLyBSZXNwb25zZQp0eXBlIExpc3RQZXRzUmVzcG9uc2Ugc3RydWN0IHsKCUNvZGUgICAgICAgIGludAoJSHR0cDIwMCAgICAgUGV0c1NjaGVtYQoJSHR0cERlZmF1bHQgKkVycm9yU2NoZW1hCn0KCi8vIENvbnRyb2xsZXIKdHlwZSBDb250cm9sbGVyIGludGVyZmFjZSB7CglMaXN0UGV0cyhwYXJhbXMgKkxpc3RQZXRzUGFyYW1zLCByZXEgKmh0dHAuUmVxdWVzdCwgcmVzIGh0dHAuUmVzcG9uc2VXcml0ZXIpIExpc3RQZXRzUmVzcG9uc2UKfQoKYGBgCgojIFNlcnZlciBib2lsZXJwbGF0ZQpJZiBuZWVkZWQgbGlicmFyeSBjYW4gZ2VuZXJhdGUgc2VydmVyIHNwZWNpZmljIGNvZGUuIEF0IHRoaXMgbW9tZW50IG9ubHkgZWNobyBmcmFtZXdvcmsgc3VwcG9ydGVkLiAgSXQgd2lsbCBpbmNsdWRlIG9ubHkgcGFyYW1ldGVycyB2YWxpZGF0aW9uLCBkZWZhdWx0cyBhbmQgcm91dGVzLgoKYG9hcGkzZ2VuIC1zZXJ2ZXIgZWNobyBzcGVjLnlhbWxgCgpgYGAKZnVuYyBCdWlsZFJvdXRlcyhlICplY2hvLkdyb3VwLCBjb250cm9sbGVyIENvbnRyb2xsZXIpIHsKCWUuR0VUKCIvcGV0cyIsIGZ1bmMoYyBlY2hvLkNvbnRleHQpIGVycm9yIHsKCQlwYXJhbWV0ZXJzIDo9ICZMaXN0UGV0c1BhcmFtc3t9CgkJaWYgZXJyIDo9IGluaXRQYXJhbWV0ZXJzKGMsIHBhcmFtZXRlcnMsIG5pbCk7IGVyciAhPSBuaWwgewoJCQlyZXR1cm4gZXJyCgkJfQoKCQlyZXNwb25zZSA6PSBjb250cm9sbGVyLkxpc3RQZXRzKHBhcmFtZXRlcnMsIGMuUmVxdWVzdCgpLCBjLlJlc3BvbnNlKCkuV3JpdGVyKQogICAgLi4uCmBgYAo= readmeEtag: '"69da7092e42f31d60542bdebd97289d907fe2594"' readmeLastModified: Thu, 09 Dec 2021 13:58:54 GMT repositoryId: 419007095 description: OpenAPI 3 code generator with zero or minimum server boilerplate created: '2021-10-19T16:33:28Z' updated: '2021-12-21T12:12:56Z' language: Go archived: false stars: 1 watchers: 0 forks: 0 owner: godknowsiamgood logo: https://avatars.githubusercontent.com/u/5710885?v=4 repoEtag: '"2b5cce321dd8f2bd7972dddfe4efd9ad44478dfb1114e6d3a9a0f7714ebaa85c"' repoLastModified: Tue, 21 Dec 2021 12:12:56 GMT foundInMaster: true category: Parsers id: ff0f2c2a5984ca8ed6346684b4530b77 - source: openapi3 tags repository: https://github.com/stackql-labs/registry.stackql.io v3: true id: c20fbd38d53ab24ed762b635f7a52806 repositoryMetadata: base64Readme: >- IyByZWdpc3RyeS5zdGFja3FsLmlvCgpUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgZG9jdW1lbnRhdGlvbiBmb3IgU3RhY2tRTCBwcm92aWRlcnMsIHdoaWNoIGlzIHB1Ymxpc2hlZCB0byBbcmVnaXN0cnkuc3RhY2txbC5pb10oaHR0cHM6Ly9yZWdpc3RyeS5zdGFja3FsLmlvKS4gIAoKWyFbTmV0bGlmeSBTdGF0dXNdKGh0dHBzOi8vYXBpLm5ldGxpZnkuY29tL2FwaS92MS9iYWRnZXMvNmQ0ZGFlNTMtNzQ1Yi00NzBlLWE5NjQtMmMwNjc2YmZlMTY1L2RlcGxveS1zdGF0dXMpXShodHRwczovL2FwcC5uZXRsaWZ5LmNvbS9zaXRlcy9yZWdpc3RyeS1zdGFja3FsLWlvL2RlcGxveXMpCgojIyBXZWJzaXRlCgpUaGlzIHdlYnNpdGUgaXMgYnVpbHQgdXNpbmcgW0RvY3VzYXVydXMgMl0oaHR0cHM6Ly9kb2N1c2F1cnVzLmlvLyksIGEgbW9kZXJuIHN0YXRpYyB3ZWJzaXRlIGdlbmVyYXRvci4KCiMjIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKeWFybgpgYGAKCiMjIyBMb2NhbCBEZXZlbG9wbWVudAoKYGBgYmFzaAp5YXJuIHN0YXJ0CmBgYAoKVGhpcyBjb21tYW5kIHN0YXJ0cyBhIGxvY2FsIGRldmVsb3BtZW50IHNlcnZlciBhbmQgb3BlbnMgdXAgYSBicm93c2VyIHdpbmRvdy4gTW9zdCBjaGFuZ2VzIGFyZSByZWZsZWN0ZWQgbGl2ZSB3aXRob3V0IGhhdmluZyB0byByZXN0YXJ0IHRoZSBzZXJ2ZXIuCgojIyMgQnVpbGQKCmBgYGJhc2gKeWFybiBidWlsZApgYGAKClRoaXMgY29tbWFuZCBnZW5lcmF0ZXMgc3RhdGljIGNvbnRlbnQgaW50byB0aGUgYGJ1aWxkYCBkaXJlY3RvcnkgYW5kIGNhbiBiZSBzZXJ2ZWQgdXNpbmcgYW55IHN0YXRpYyBjb250ZW50cyBob3N0aW5nIHNlcnZpY2UuCgojIyMgRGVwbG95bWVudAoKYGBgYmFzaApHSVRfVVNFUj08WW91ciBHaXRIdWIgdXNlcm5hbWU+IFVTRV9TU0g9dHJ1ZSB5YXJuIGRlcGxveQpgYGAKCklmIHlvdSBhcmUgdXNpbmcgR2l0SHViIHBhZ2VzIGZvciBob3N0aW5nLCB0aGlzIGNvbW1hbmQgaXMgYSBjb252ZW5pZW50IHdheSB0byBidWlsZCB0aGUgd2Vic2l0ZSBhbmQgcHVzaCB0byB0aGUgYGdoLXBhZ2VzYCBicmFuY2guCg== readmeEtag: '"fd8aa2807c119ce0306a0d2992b67533567a3280"' readmeLastModified: Tue, 27 Sep 2022 05:47:35 GMT repositoryId: 504338261 description: REPLACED BY https://github.com/stackql/stackql-registry-docs created: '2022-06-16T23:52:36Z' updated: '2025-12-25T23:06:26Z' language: CSS archived: true stars: 1 watchers: 1 forks: 1 owner: stackql-labs logo: https://avatars.githubusercontent.com/u/251279874?v=4 repoEtag: '"da8ac9d098667c863b861548773cf72589755cba246bec7aab94c78f55d5ce9b"' repoLastModified: Thu, 25 Dec 2025 23:06:26 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/stackql/registry.stackql.io - source: openapi3 tags repository: https://github.com/tanmaykm/openapiclient.jl v3: true id: 545ca30b5ff061cb9e365cffcdcce408 repositoryMetadata: base64Readme: >- # OpenAPIClient

[![Build Status](https://github.com/tanmaykm/OpenAPIClient.jl/workflows/CI/badge.svg)](https://github.com/tanmaykm/OpenAPIClient.jl/actions?query=workflow%3ACI+branch%3Amaster)
[![codecov.io](http://codecov.io/github/tanmaykm/OpenAPIClient.jl/coverage.svg?branch=master)](http://codecov.io/github/tanmaykm/OpenAPIClient.jl?branch=master)

This is the Julia library needed with code generated by the [OpenAPI generator](https://openapi-generator.tech/).

The goal of OpenAPIClient is to define a standard, language-agnostic interface to REST APIs which allows both humans and computers to discover and understand the capabilities of the service without access to source code, documentation, or through network traffic inspection. When properly defined via OpenAPI, a consumer can understand and interact with the remote service with a minimal amount of implementation logic. Similar to what interfaces have done for lower-level programming, OpenAPI removes the guesswork in calling the service.

Check out [OpenAPI-Spec](https://github.com/OAI/OpenAPI-Specification) for additional information about the OpenAPI project, including additional libraries with support for other languages and more.

## How do I use this?

### Code Generation

Use [instructions](https://openapi-generator.tech/docs/generators) provided for the Julia OpenAPI code generator plugin to generate Julia code.

## Generated Code Structure

### APIs

Each API set is generated into a file named `api_<apiname>.jl`. It is represented as a `struct` and the APIs under it are generated as methods. An API set can be constructed by providing the OpenAPI client instance that it can use for communication.

The required API parameters are generated as regular function arguments. Optional parameters are generated as keyword arguments. Method documentation is generated with description, parameter information and return value. Two variants of the API are generated. The first variant is suitable for calling synchronously and returns a single instance of the result struct.

```julia
# example synchronous API that returns an Order instance
getOrderById(api::StoreApi, orderId::Int64)
```

The second variant is suitable for asynchronous calls to methods that return chunked transfer encoded responses, where in the API streams the response objects into an output channel.

```julia
# example asynchronous API that streams matching Pet instances into response_stream
findPetsByStatus(api::PetApi, response_stream::Channel, status::Vector{String})
```

A client context holds common information to be used across APIs. It also holds a connection to the server and uses that across API calls.
The client context needs to be passed as the first parameter of all API calls. It can be created as:

```julia
Client(root::String;
    headers::Dict{String,String}=Dict{String,String}(),
    get_return_type::Function=(default,data)->default,
    timeout::Int=DEFAULT_TIMEOUT_SECS,
    long_polling_timeout::Int=DEFAULT_LONGPOLL_TIMEOUT_SECS,
    pre_request_hook::Function,
)
```

Where:

- `root`: the root URI where APIs are hosted (should not end with a `/`)
- `headers`: any additional headers that need to be passed along with all API calls
- `get_return_type`: optional method that can map a Julia type to a return type other than what is specified in the API specification by looking at the data (this is used only in special cases, for example when models are allowed to be dynamically loaded)
- `timeout`: optional timeout to apply for server methods (default `OpenAPIClient.DEFAULT_TIMEOUT_SECS`)
- `long_polling_timeout`: optional timeout to apply for long polling methods (default `OpenAPIClient.DEFAULT_LONGPOLL_TIMEOUT_SECS`)
- `pre_request_hook`: user provided hook to modify the request before it is sent

The `pre_request_hook` must provide the following two implementations:
- `pre_request_hook(ctx::OpenAPIClient.Ctx) -> ctx`
- `pre_request_hook(resource_path::AbstractString, body::Any, headers::Dict{String,String}) -> (resource_path, body, headers)`

In case of any errors an instance of `ApiException` is thrown. It has the following fields:

- `status::Int`: HTTP status code
- `reason::String`: Optional human readable string
- `resp::Downloads.Response`: The HTTP Response for this call
- `error::Union{Nothing,Downloads.RequestError}`: The HTTP error on request failure

An API call involves the following steps:
- If a pre request hook is provided, it is invoked with an instance of `OpenAPIClient.Ctx` that has the request attributes. The hook method is expected to make any modifications it needs to the request attributes before the request is prepared, and return the modified context.
- The URL to be invoked is prepared by replacing placeholders in the API URL template with the supplied function parameters.
- If this is a POST request, serialize the instance of `OpenAPIModel` provided as the `body` parameter as a JSON document.
- If a pre request hook is provided, it is invoked with the prepared resource path, body and request headers. The hook method is expected to modify and return back a tuple of resource path, body and headers which will be used to make the request.
- Make the HTTP call to the API endpoint and collect the response.
- Determine the response type / model, invoke the optional user specified mapping function if one was provided.
- Convert (deserialize) the response data into the return type and return.
- In case of any errors, throw an instance of `ApiException`

### Models

Each model from the specification is generated into a file named `model_<modelname>.jl`. It is represented as a `mutable struct` that is a subtype of the abstract type `OpenAPIModel`. Models have the following methods defined:

- constructor that takes keyword arguments to fill in values for all model properties.
- [`propertynames`](https://docs.julialang.org/en/v1/base/base/#Base.propertynames)
- [`hasproperty`](https://docs.julialang.org/en/v1/base/base/#Base.hasproperty)
- [`getproperty`](https://docs.julialang.org/en/v1/base/base/#Base.getproperty)
- [`setproperty!`](https://docs.julialang.org/en/v1/base/base/#Base.setproperty!)

In addition to these standard Julia methods, these convenience methods are also generated that help in checking value at a hierarchical path of the model.

- `function haspropertyat(o::T, path...) where {T<:OpenAPIModel}`
- `function getpropertyat(o::T, path...) where {T<:OpenAPIModel}`

E.g:

```julia
# access o.field.subfield1.subfield2
if haspropertyat(o, "field", "subfield1", "subfield2")
    getpropertyat(o, "field", "subfield1", "subfield2")
end

# access nested array elements, e.g. o.field2.subfield1[10].subfield2
if haspropertyat(o, "field", "subfield1", 10, "subfield2")
    getpropertyat(o, "field", "subfield1", 10, "subfield2")
end
```

### Validations

Following validations are incorporated into models:

- maximum value: must be a numeric value less than or equal to a specified value
- minimum value: must be a numeric value greater than or equal to a specified value
- maximum length: must be a string value of length less than or equal to a specified value
- minimum length: must be a string value of length greater than or equal to a specified value
- maximum item count: must be a list value with number of items less than or equal to a specified value
- minimum item count: must be a list value with number of items greater than or equal to a specified value
- enum: value must be from a list of allowed values

Validations are imposed in the constructor and `setproperty!` methods of models.
 readmeEtag: '"ab841d35744db25a492a36c2eeb2434f2c277947"' readmeLastModified: Fri, 09 Sep 2022 04:13:57 GMT repositoryId: 521258468 description: OpenAPI helper for Julia, works with openapi-generator created: '2022-08-04T12:30:35Z' updated: '2022-09-09T06:34:48Z' language: Julia archived: false stars: 1 watchers: 1 forks: 0 owner: tanmaykm logo: https://avatars.githubusercontent.com/u/1010989?v=4 license: NOASSERTION repoEtag: '"db1c6d45f029a05d4c8ba0072f4f1236f4889b6710f0ef1f95eba5a92044ab78"' repoLastModified: Fri, 09 Sep 2022 06:34:48 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/nasrmohammad4804/reactive-programming-demo v3: true id: 6bb58bce20152c0be4fded1ee9a95a03 repositoryMetadata: base64Readme: >- IyByZWFjdGl2ZS1wcm9ncmFtbWluZy1kZW1vCmkgd2FudCB0byBjcmVhdGUgc2FtcGxlIHByb2plY3Qgd2l0aCBzcHJpbmcgd2ViZmx1eCBhbmQgZG9ja2VyaXplIGl0Ci0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiFbZG93bmxvYWRdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzc2MDM4MTQzLzE4NTc2NjE5MS1hYTM3N2QzMC1mZTk0LTRhNTYtYjhjYy0xZTVhZmViNDE0NzcuanBnKQotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCiMgeW91IGNhbiBjbG9uZSBwcm9qZWN0IGFuZCBydW4gd2l0aCBkb2NrZXIgCioqMSoqLiB5b3UgbmVlZCB0byBidWlsZCBpbWFnZSBvZiBwcm9qZWN0IHdpdGggY29tbWFuZCBvZiAgKipkb2NrZXIgYnVpbGQgLXQgdXNlci1teXNxbCAuKiogIChhbmQgYWJsZSB0byBhbHRlcm5hdGl2ZSBkb2NrZXIgaW1hZ2UgaW5zdGVhZCBvZiB1c2VyLW15c3FsKQoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoqKjIqKi4geW91IG5lZWQgdG8gcHVsbCBteXNxbCBpbWFnZSBmcm9tIGRvY2tlciBodWIgd2l0aCBjb21tYW5kIG9mICoqZG9ja2VyIHB1bGwgbXlzcWw6bGF0ZXN0KiogCihub3RlOiBpZiB5b3UgYWxyZWFkeSBoYXZlIGl0IGRvbnQgbmVlZCB0byBleGVjdXRlIHRoaXMgY29tbWFuZCBhbmQgZ2V0IGl0IGZyb20gbG9jYWwgKQoKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoqKjMqKi4gYWZ0ZXIgeW91IG5lZWQgdG8gcnVuIGNvbnRhaW5lciBvZiBteXNxbCB3aXRoIHlvdXIgb3duIHNwZWNpZmljYXRpb24gc3VjaCBhcyB1c2VybmFtZSBhbmQgcGFzc3dvcmQgYW5kIGRhdGFiYXNlIG5hbWUgd2l0aCBjb21tYW5kIG9mCioqZG9ja2VyIHJ1biAtLW5hbWUgbXlzcWwtc3RhbmRhbG9uZSAtZSBNWVNRTF9ST09UX1BBU1NXT1JEPU1vaGFtbWFkTkBzcjEzODA0ODA0IC1lIE1ZU1FMX0RBVEFCQVNFPXJlYWN0aXZlLWRlbW8gIC1lIE1ZU1FMX1BBU1NXT1JEPU1vaGFtbWFkTkBzcjEzODA0ODA0IC1kIG15c3FsKioKIAotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tCioqNCoqLiBhbmQgYXQgdGhlIGxhc3Qgc3RhZ2UgeW91IG5lZWQgdG8gcnVuIHRoaXMgY29udGFpbmVyIGFuZCBsaW5rIHByb2plY3QgaW1hZ2Ugd2l0aCBteXNxbCBmb3IgY29tcGF0aWJpbGl0eSB0b2dldGhlciBhbmQgbWFwIG9uIGV2ZXJ5IHBvcnQgd2hpY2ggeW91IHdhbnQgb24geW91ciBsb2NhbCBtYWNoaW5lCg== readmeEtag: '"dcba8bde5bd66fbba492f9784a9f23dc493f9085"' readmeLastModified: Sat, 20 Aug 2022 21:30:46 GMT repositoryId: 527035744 description: i want to create sample project with spring webflux and dockerize it created: '2022-08-20T20:44:25Z' updated: '2022-11-18T21:53:54Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: nasrmohammad4804 logo: https://avatars.githubusercontent.com/u/76038143?v=4 license: MIT repoEtag: '"e9f406fea3ae82b03696dd481aa282beb3f5cb33bbc16ce21d7a6357e2d8fabb"' repoLastModified: Fri, 18 Nov 2022 21:53:54 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/namsor/namsor-ruby-sdk2 v3: true repositoryMetadata: base64Readme: >- # namsor_client

NamSorClient - the Ruby gem for the NamSor API v2

NamSor API v2 : enpoints to process personal names (gender, cultural origin or ethnicity) in all alphabets or languages. Use GET methods for small tests, but prefer POST methods for higher throughput (batch processing of up to 1000 names at a time). Need something you can't find here? We have many more features coming soon. Let us know, we'll do our best to add it! 

This SDK is automatically generated by the [OpenAPI Generator](https://openapi-generator.tech) project:

- API version: 2.0.2-beta
- Package version: 0.1.0
- Build package: org.openapitools.codegen.languages.RubyClientCodegen
For more information, please visit [http://www.namsor.com/](http://www.namsor.com/)

## Installation

### Build a gem

To build the Ruby code into a gem:

```shell
gem build namsor_client.gemspec
```

Then either install the gem locally:

```shell
gem install ./namsor_client-0.1.0.gem
```
(for development, run `gem install --dev ./namsor_client-0.1.0.gem` to install the development dependencies)

or publish the gem to a gem hosting service, e.g. [RubyGems](https://rubygems.org/).

Finally add this to the Gemfile:

    gem 'namsor_client', '~> 0.1.0'

### Install from Git

If the Ruby gem is hosted at a git repository: https://github.com/namsor/namsor-ruby-sdk2, then add the following in the Gemfile:

    gem 'namsor_client', :git => 'https://github.com/namsor/namsor-ruby-sdk2.git'

### Include the Ruby code directly

Include the Ruby code directly using `-I` as follows:

```shell
ruby -Ilib script.rb
```

## Getting Started

Please follow the [installation](#installation) procedure and then run the following code:
```ruby
# Load the gem
require 'namsor_client'

# Setup authorization
NamSorClient.configure do |config|
  # Configure API key authorization: api_key
  config.api_key['X-API-KEY'] = 'YOUR API KEY'
  # Uncomment the following line to set a prefix for the API key, e.g. 'Bearer' (defaults to nil)
  #config.api_key_prefix['X-API-KEY'] = 'Bearer'
end

api_instance = NamSorClient::AdminApi.new

begin
  #Print current API usage.
  result = api_instance.api_usage
  p result
rescue NamSorClient::ApiError => e
  puts "Exception when calling AdminApi->api_usage: #{e}"
end

```

## Documentation for API Endpoints

All URIs are relative to *https://v2.namsor.com/NamSorAPIv2*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*NamSorClient::AdminApi* | [**api_usage**](docs/AdminApi.md#api_usage) | **GET** /api2/json/apiUsage | Print current API usage.
*NamSorClient::AdminApi* | [**api_usage_history**](docs/AdminApi.md#api_usage_history) | **GET** /api2/json/apiUsageHistory | Print historical API usage.
*NamSorClient::AdminApi* | [**api_usage_history_aggregate**](docs/AdminApi.md#api_usage_history_aggregate) | **GET** /api2/json/apiUsageHistoryAggregate | Print historical API usage (in an aggregated view, by service, by day/hour/min).
*NamSorClient::AdminApi* | [**available_plans**](docs/AdminApi.md#available_plans) | **GET** /api2/json/availablePlans | List all available plans in the default currency (usd).
*NamSorClient::AdminApi* | [**available_plans1**](docs/AdminApi.md#available_plans1) | **GET** /api2/json/availablePlans/{token} | List all available plans in the user's preferred currency.
*NamSorClient::AdminApi* | [**available_services**](docs/AdminApi.md#available_services) | **GET** /api2/json/apiServices | List of API services and usage cost in Units (default is 1=ONE Unit).
*NamSorClient::AdminApi* | [**billing_currencies**](docs/AdminApi.md#billing_currencies) | **GET** /api2/json/billingCurrencies | List possible currency options for billing (USD, EUR, GBP, ...)
*NamSorClient::AdminApi* | [**billing_history**](docs/AdminApi.md#billing_history) | **GET** /api2/json/billingHistory/{token} | Read the history billing information (invoices paid via Stripe or manually).
*NamSorClient::AdminApi* | [**billing_info**](docs/AdminApi.md#billing_info) | **GET** /api2/json/billingInfo/{token} | Read the billing information (company name, address, phone, vat ID)
*NamSorClient::AdminApi* | [**charge**](docs/AdminApi.md#charge) | **POST** /api2/json/charge | Create a Stripe Customer, based on a payment card token (from secure StripeJS) and email.
*NamSorClient::AdminApi* | [**corporate_key**](docs/AdminApi.md#corporate_key) | **GET** /api2/json/corporateKey/{apiKey}/{corporate} | Setting an API Key to a corporate status.
*NamSorClient::AdminApi* | [**debug_level**](docs/AdminApi.md#debug_level) | **GET** /api2/json/debugLevel/{logger}/{level} | Update debug level for a classifier
*NamSorClient::AdminApi* | [**invalidate_cache**](docs/AdminApi.md#invalidate_cache) | **GET** /api2/json/invalidateCache | Invalidate system caches.
*NamSorClient::AdminApi* | [**learnable**](docs/AdminApi.md#learnable) | **GET** /api2/json/learnable/{source}/{learnable} | Activate/deactivate learning from a source.
*NamSorClient::AdminApi* | [**namsor_counter**](docs/AdminApi.md#namsor_counter) | **GET** /api2/json/namsorCounter | Get the overall API counter
*NamSorClient::AdminApi* | [**payment_info**](docs/AdminApi.md#payment_info) | **GET** /api2/json/paymentInfo/{token} | Get the Stripe payment information associated with the current google auth session token.
*NamSorClient::AdminApi* | [**procure_key**](docs/AdminApi.md#procure_key) | **GET** /api2/json/procureKey/{token} | Procure an API Key (sent via Email), based on an auth token. Keep your API Key secret.
*NamSorClient::AdminApi* | [**redeploy_ui**](docs/AdminApi.md#redeploy_ui) | **GET** /api2/json/redeployUI | Redeploy UI from current dev branch.
*NamSorClient::AdminApi* | [**redeploy_ui1**](docs/AdminApi.md#redeploy_ui1) | **GET** /api2/json/redeployUI/{live} | Redeploy UI from current dev branch.
*NamSorClient::AdminApi* | [**remove_user_account**](docs/AdminApi.md#remove_user_account) | **GET** /api2/json/removeUserAccount/{token} | Subscribe to a give API plan, using the user's preferred or default currency.
*NamSorClient::AdminApi* | [**shutdown**](docs/AdminApi.md#shutdown) | **GET** /api2/json/shutdown | Stop learning and shutdown system.
*NamSorClient::AdminApi* | [**software_version**](docs/AdminApi.md#software_version) | **GET** /api2/json/softwareVersion | Get the current software version
*NamSorClient::AdminApi* | [**source_stats**](docs/AdminApi.md#source_stats) | **GET** /api2/json/sourceStats/{source} | Print basic source statistics.
*NamSorClient::AdminApi* | [**stats**](docs/AdminApi.md#stats) | **GET** /api2/json/stats | Print basic system statistics.
*NamSorClient::AdminApi* | [**stripe_connect**](docs/AdminApi.md#stripe_connect) | **GET** /api2/json/stripeConnect | Connects a Stripe Account.
*NamSorClient::AdminApi* | [**subscribe_plan**](docs/AdminApi.md#subscribe_plan) | **GET** /api2/json/subscribePlan/{planName}/{token} | Subscribe to a give API plan, using the user's preferred or default currency.
*NamSorClient::AdminApi* | [**update_billing_info**](docs/AdminApi.md#update_billing_info) | **POST** /api2/json/updateBillingInfo/{token} | Sets or update the billing information (company name, address, phone, vat ID)
*NamSorClient::AdminApi* | [**update_limit**](docs/AdminApi.md#update_limit) | **GET** /api2/json/updateLimit/{usageLimit}/{hardOrSoft}/{token} | Modifies the hard/soft limit on the API plan's overages (default is 0$ soft limit).
*NamSorClient::AdminApi* | [**update_payment_default**](docs/AdminApi.md#update_payment_default) | **GET** /api2/json/updatePaymentDefault/{defautSourceId}/{token} | Update the default Stripe card associated with the current google auth session token.
*NamSorClient::AdminApi* | [**user_info**](docs/AdminApi.md#user_info) | **GET** /api2/json/userInfo/{token} | Get the user profile associated with the current google auth session token.
*NamSorClient::AdminApi* | [**verify_email**](docs/AdminApi.md#verify_email) | **GET** /api2/json/verifyEmail/{emailToken} | Verifies an email, based on token sent to that email
*NamSorClient::AdminApi* | [**verify_remove_email**](docs/AdminApi.md#verify_remove_email) | **GET** /api2/json/verifyRemoveEmail/{emailToken} | Verifies an email, based on token sent to that email
*NamSorClient::AdminApi* | [**vet**](docs/AdminApi.md#vet) | **GET** /api2/json/vetting/{source}/{vetted} | Vetting of a source.
*NamSorClient::ChineseApi* | [**chinese_name_candidates**](docs/ChineseApi.md#chinese_name_candidates) | **GET** /api2/json/chineseNameCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin} | Identify Chinese name candidates, based on the romanized name.
*NamSorClient::ChineseApi* | [**chinese_name_candidates_batch**](docs/ChineseApi.md#chinese_name_candidates_batch) | **POST** /api2/json/chineseNameCandidatesBatch | Identify Chinese name candidates, based on the romanized name (firstName = chineseGivenName; lastName=chineseSurname).
*NamSorClient::ChineseApi* | [**chinese_name_candidates_gender_batch**](docs/ChineseApi.md#chinese_name_candidates_gender_batch) | **POST** /api2/json/chineseNameCandidatesGenderBatch | Identify Chinese name candidates, based on the romanized name (firstName = chineseGivenName; lastName=chineseSurname).
*NamSorClient::ChineseApi* | [**chinese_name_gender_candidates**](docs/ChineseApi.md#chinese_name_gender_candidates) | **GET** /api2/json/chineseNameGenderCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin}/{knownGender} | Identify Chinese name candidates, based on the romanized name - having a known gender ('male' or 'female')
*NamSorClient::PersonalApi* | [**country**](docs/PersonalApi.md#country) | **GET** /api2/json/country/{personalNameFull} | [USES 10 UNITS] Infer the likely country of residence of a personal full name, or one surname. Assumes names as they are in the country of residence OR the country of origin.
*NamSorClient::PersonalApi* | [**country_batch**](docs/PersonalApi.md#country_batch) | **POST** /api2/json/countryBatch | [USES 10 UNITS] Infer the likely country of residence of up to 1000 personal full names, or surnames. Assumes names as they are in the country of residence OR the country of origin.
*NamSorClient::PersonalApi* | [**diaspora**](docs/PersonalApi.md#diaspora) | **GET** /api2/json/diaspora/{countryIso2}/{firstName}/{lastName} | [USES 20 UNITS] Infer the likely ethnicity/diaspora of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*NamSorClient::PersonalApi* | [**diaspora_batch**](docs/PersonalApi.md#diaspora_batch) | **POST** /api2/json/diasporaBatch | [USES 20 UNITS] Infer the likely ethnicity/diaspora of up to 1000 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*NamSorClient::PersonalApi* | [**gender**](docs/PersonalApi.md#gender) | **GET** /api2/json/gender/{firstName}/{lastName} | Infer the likely gender of a name.
*NamSorClient::PersonalApi* | [**gender_batch**](docs/PersonalApi.md#gender_batch) | **POST** /api2/json/genderBatch | Infer the likely gender of up to 1000 names, detecting automatically the cultural context.
*NamSorClient::PersonalApi* | [**gender_full**](docs/PersonalApi.md#gender_full) | **GET** /api2/json/genderFull/{fullName} | Infer the likely gender of a full name, ex. John H. Smith
*NamSorClient::PersonalApi* | [**gender_full_batch**](docs/PersonalApi.md#gender_full_batch) | **POST** /api2/json/genderFullBatch | Infer the likely gender of up to 1000 full names, detecting automatically the cultural context.
*NamSorClient::PersonalApi* | [**gender_full_geo**](docs/PersonalApi.md#gender_full_geo) | **GET** /api2/json/genderFullGeo/{fullName}/{countryIso2} | Infer the likely gender of a full name, given a local context (ISO2 country code).
*NamSorClient::PersonalApi* | [**gender_full_geo_batch**](docs/PersonalApi.md#gender_full_geo_batch) | **POST** /api2/json/genderFullGeoBatch | Infer the likely gender of up to 1000 full names, with a given cultural context (country ISO2 code).
*NamSorClient::PersonalApi* | [**gender_geo**](docs/PersonalApi.md#gender_geo) | **GET** /api2/json/genderGeo/{firstName}/{lastName}/{countryIso2} | Infer the likely gender of a name, given a local context (ISO2 country code).
*NamSorClient::PersonalApi* | [**gender_geo_batch**](docs/PersonalApi.md#gender_geo_batch) | **POST** /api2/json/genderGeoBatch | Infer the likely gender of up to 1000 names, each given a local context (ISO2 country code).
*NamSorClient::PersonalApi* | [**origin**](docs/PersonalApi.md#origin) | **GET** /api2/json/origin/{firstName}/{lastName} | [USES 10 UNITS] Infer the likely country of origin of a personal name. Assumes names as they are in the country of origin. For US, CA, AU, NZ and other melting-pots : use 'diaspora' instead.
*NamSorClient::PersonalApi* | [**origin_batch**](docs/PersonalApi.md#origin_batch) | **POST** /api2/json/originBatch | [USES 10 UNITS] Infer the likely country of origin of up to 1000 names, detecting automatically the cultural context.
*NamSorClient::PersonalApi* | [**parse_name**](docs/PersonalApi.md#parse_name) | **GET** /api2/json/parseName/{nameFull} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. 
*NamSorClient::PersonalApi* | [**parse_name_batch**](docs/PersonalApi.md#parse_name_batch) | **POST** /api2/json/parseNameBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John.
*NamSorClient::PersonalApi* | [**parse_name_geo**](docs/PersonalApi.md#parse_name_geo) | **GET** /api2/json/parseName/{nameFull}/{countryIso2} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. For better accuracy, provide a geographic context.
*NamSorClient::PersonalApi* | [**parse_name_geo_batch**](docs/PersonalApi.md#parse_name_geo_batch) | **POST** /api2/json/parseNameGeoBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. Giving a local context improves precision. 
*NamSorClient::PersonalApi* | [**parsed_gender_batch**](docs/PersonalApi.md#parsed_gender_batch) | **POST** /api2/json/parsedGenderBatch | Infer the likely gender of up to 1000 fully parsed names, detecting automatically the cultural context.
*NamSorClient::PersonalApi* | [**parsed_gender_geo_batch**](docs/PersonalApi.md#parsed_gender_geo_batch) | **POST** /api2/json/parsedGenderGeoBatch | Infer the likely gender of up to 1000 fully parsed names, detecting automatically the cultural context.
*NamSorClient::PersonalApi* | [**us_race_ethnicity**](docs/PersonalApi.md#us_race_ethnicity) | **GET** /api2/json/usRaceEthnicity/{firstName}/{lastName} | [USES 10 UNITS] Infer a US resident's likely race/ethnicity according to US Census taxonomy W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino).
*NamSorClient::PersonalApi* | [**us_race_ethnicity_batch**](docs/PersonalApi.md#us_race_ethnicity_batch) | **POST** /api2/json/usRaceEthnicityBatch | [USES 10 UNITS] Infer up-to 1000 US resident's likely race/ethnicity according to US Census taxonomy.
*NamSorClient::PersonalApi* | [**us_race_ethnicity_zip5**](docs/PersonalApi.md#us_race_ethnicity_zip5) | **GET** /api2/json/usRaceEthnicityZIP5/{firstName}/{lastName}/{zip5Code} | [USES 10 UNITS] Infer a US resident's likely race/ethnicity according to US Census taxonomy, using (optional) ZIP5 code info. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino).
*NamSorClient::PersonalApi* | [**us_zip_race_ethnicity_batch**](docs/PersonalApi.md#us_zip_race_ethnicity_batch) | **POST** /api2/json/usZipRaceEthnicityBatch | [USES 10 UNITS] Infer up-to 1000 US resident's likely race/ethnicity according to US Census taxonomy, with (optional) ZIP code.
*NamSorClient::SocialApi* | [**phone_code**](docs/SocialApi.md#phone_code) | **GET** /api2/json/phoneCode/{firstName}/{lastName}/{phoneNumber} | [USES 11 UNITS] Infer the likely country and phone prefix, given a personal name and formatted / unformatted phone number.
*NamSorClient::SocialApi* | [**phone_code_batch**](docs/SocialApi.md#phone_code_batch) | **POST** /api2/json/phoneCodeBatch | [USES 11 UNITS] Infer the likely country and phone prefix, of up to 1000 personal names, detecting automatically the local context given a name and formatted / unformatted phone number.
*NamSorClient::SocialApi* | [**phone_code_geo**](docs/SocialApi.md#phone_code_geo) | **GET** /api2/json/phoneCodeGeo/{firstName}/{lastName}/{phoneNumber}/{countryIso2} | [USES 11 UNITS] Infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).
*NamSorClient::SocialApi* | [**phone_code_geo_batch**](docs/SocialApi.md#phone_code_geo_batch) | **POST** /api2/json/phoneCodeGeoBatch | [USES 11 UNITS] Infer the likely country and phone prefix, of up to 1000 personal names, with a local context (ISO2 country of residence).


## Documentation for Models

 - [NamSorClient::APIBillingPeriodUsageOut](docs/APIBillingPeriodUsageOut.md)
 - [NamSorClient::APICounterV2Out](docs/APICounterV2Out.md)
 - [NamSorClient::APIKeyOut](docs/APIKeyOut.md)
 - [NamSorClient::APIPeriodUsageOut](docs/APIPeriodUsageOut.md)
 - [NamSorClient::APIPlanOut](docs/APIPlanOut.md)
 - [NamSorClient::APIPlanSubscriptionOut](docs/APIPlanSubscriptionOut.md)
 - [NamSorClient::APIPlansOut](docs/APIPlansOut.md)
 - [NamSorClient::APIServiceOut](docs/APIServiceOut.md)
 - [NamSorClient::APIServicesOut](docs/APIServicesOut.md)
 - [NamSorClient::APIUsageAggregatedOut](docs/APIUsageAggregatedOut.md)
 - [NamSorClient::BatchFirstLastNameDiasporaedOut](docs/BatchFirstLastNameDiasporaedOut.md)
 - [NamSorClient::BatchFirstLastNameGenderIn](docs/BatchFirstLastNameGenderIn.md)
 - [NamSorClient::BatchFirstLastNameGenderedOut](docs/BatchFirstLastNameGenderedOut.md)
 - [NamSorClient::BatchFirstLastNameGeoIn](docs/BatchFirstLastNameGeoIn.md)
 - [NamSorClient::BatchFirstLastNameGeoZippedIn](docs/BatchFirstLastNameGeoZippedIn.md)
 - [NamSorClient::BatchFirstLastNameIn](docs/BatchFirstLastNameIn.md)
 - [NamSorClient::BatchFirstLastNameOriginedOut](docs/BatchFirstLastNameOriginedOut.md)
 - [NamSorClient::BatchFirstLastNamePhoneCodedOut](docs/BatchFirstLastNamePhoneCodedOut.md)
 - [NamSorClient::BatchFirstLastNamePhoneNumberGeoIn](docs/BatchFirstLastNamePhoneNumberGeoIn.md)
 - [NamSorClient::BatchFirstLastNamePhoneNumberIn](docs/BatchFirstLastNamePhoneNumberIn.md)
 - [NamSorClient::BatchFirstLastNameUSRaceEthnicityOut](docs/BatchFirstLastNameUSRaceEthnicityOut.md)
 - [NamSorClient::BatchNameMatchCandidatesOut](docs/BatchNameMatchCandidatesOut.md)
 - [NamSorClient::BatchParsedFullNameGeoIn](docs/BatchParsedFullNameGeoIn.md)
 - [NamSorClient::BatchParsedFullNameIn](docs/BatchParsedFullNameIn.md)
 - [NamSorClient::BatchPersonalNameGenderedOut](docs/BatchPersonalNameGenderedOut.md)
 - [NamSorClient::BatchPersonalNameGeoIn](docs/BatchPersonalNameGeoIn.md)
 - [NamSorClient::BatchPersonalNameGeoOut](docs/BatchPersonalNameGeoOut.md)
 - [NamSorClient::BatchPersonalNameIn](docs/BatchPersonalNameIn.md)
 - [NamSorClient::BatchPersonalNameParsedOut](docs/BatchPersonalNameParsedOut.md)
 - [NamSorClient::BillingHistoryOut](docs/BillingHistoryOut.md)
 - [NamSorClient::BillingInfoInOut](docs/BillingInfoInOut.md)
 - [NamSorClient::ClassifierMetricsOut](docs/ClassifierMetricsOut.md)
 - [NamSorClient::CurrenciesOut](docs/CurrenciesOut.md)
 - [NamSorClient::DeployUIOut](docs/DeployUIOut.md)
 - [NamSorClient::ExpectedClassMetricsOut](docs/ExpectedClassMetricsOut.md)
 - [NamSorClient::FirstLastNameDiasporaedOut](docs/FirstLastNameDiasporaedOut.md)
 - [NamSorClient::FirstLastNameGenderIn](docs/FirstLastNameGenderIn.md)
 - [NamSorClient::FirstLastNameGenderedOut](docs/FirstLastNameGenderedOut.md)
 - [NamSorClient::FirstLastNameGeoIn](docs/FirstLastNameGeoIn.md)
 - [NamSorClient::FirstLastNameGeoZippedIn](docs/FirstLastNameGeoZippedIn.md)
 - [NamSorClient::FirstLastNameIn](docs/FirstLastNameIn.md)
 - [NamSorClient::FirstLastNameOriginedOut](docs/FirstLastNameOriginedOut.md)
 - [NamSorClient::FirstLastNameOut](docs/FirstLastNameOut.md)
 - [NamSorClient::FirstLastNamePhoneCodedOut](docs/FirstLastNamePhoneCodedOut.md)
 - [NamSorClient::FirstLastNamePhoneNumberGeoIn](docs/FirstLastNamePhoneNumberGeoIn.md)
 - [NamSorClient::FirstLastNamePhoneNumberIn](docs/FirstLastNamePhoneNumberIn.md)
 - [NamSorClient::FirstLastNameUSRaceEthnicityOut](docs/FirstLastNameUSRaceEthnicityOut.md)
 - [NamSorClient::InlineObject](docs/InlineObject.md)
 - [NamSorClient::InvoiceItemOut](docs/InvoiceItemOut.md)
 - [NamSorClient::InvoiceOut](docs/InvoiceOut.md)
 - [NamSorClient::NamSorCounterOut](docs/NamSorCounterOut.md)
 - [NamSorClient::NameMatchCandidateOut](docs/NameMatchCandidateOut.md)
 - [NamSorClient::NameMatchCandidatesOut](docs/NameMatchCandidatesOut.md)
 - [NamSorClient::ParsedFullNameGeoIn](docs/ParsedFullNameGeoIn.md)
 - [NamSorClient::ParsedFullNameIn](docs/ParsedFullNameIn.md)
 - [NamSorClient::PersonalNameGenderedOut](docs/PersonalNameGenderedOut.md)
 - [NamSorClient::PersonalNameGeoIn](docs/PersonalNameGeoIn.md)
 - [NamSorClient::PersonalNameGeoOut](docs/PersonalNameGeoOut.md)
 - [NamSorClient::PersonalNameIn](docs/PersonalNameIn.md)
 - [NamSorClient::PersonalNameParsedOut](docs/PersonalNameParsedOut.md)
 - [NamSorClient::RomanizedNameOut](docs/RomanizedNameOut.md)
 - [NamSorClient::SoftwareVersionOut](docs/SoftwareVersionOut.md)
 - [NamSorClient::SourceDetailedMetricsOut](docs/SourceDetailedMetricsOut.md)
 - [NamSorClient::SourceMetricsOut](docs/SourceMetricsOut.md)
 - [NamSorClient::StripeCardOut](docs/StripeCardOut.md)
 - [NamSorClient::StripeCustomerOut](docs/StripeCustomerOut.md)
 - [NamSorClient::SystemMetricsOut](docs/SystemMetricsOut.md)
 - [NamSorClient::UserInfoOut](docs/UserInfoOut.md)


## Documentation for Authorization


### api_key

- **Type**: API key
- **API key parameter name**: X-API-KEY
- **Location**: HTTP header

 readmeEtag: '"e0835eaedddee7813c66e59bb88b4feef948553a"' readmeLastModified: Thu, 14 Mar 2019 09:45:25 GMT repositoryId: 172474353 description: >- NamSor API v2 Ruby SDK - classify personal names accurately by gender, country of origin, or ethnicity. created: '2019-02-25T09:24:00Z' updated: '2019-04-17T09:12:39Z' language: Ruby archived: false stars: 1 watchers: 2 forks: 2 owner: namsor logo: https://avatars.githubusercontent.com/u/6951565?v=4 license: AGPL-3.0 repoEtag: '"6e7365f131656fa741bf8d65a179b08bd4d2ab03d7abde844c9ab75bf943a8d1"' repoLastModified: Wed, 17 Apr 2019 09:12:39 GMT foundInMaster: true category: - Documentation - Description Validators id: 08a9cac08c5c62457c1f6a2d3063164d - source: openapi3 tags repository: https://github.com/gabrielmaialva33/doc-sympla-integration v3: true id: b09f616f911b87055b98bc874c7caa72 repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPgogIDxicj4KICA8aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2dhYnJpZWxtYWlhbHZhMzMvZG9jLXN5bXBsYS1pbnRlZ3JhdGlvbi9tYXN0ZXIvLmdpdGh1Yi9hc3NldHMvZG9jcy5wbmciIGFsdD0iRG9jOiBTeW1wbGEgIiB3aWR0aD0iMjAwIj4KICA8YnI+CiAgQSBkb2N1bWVudGF0aW9uIGZvciA8YSBocmVmPSJodHRwczovL3d3dy5zeW1wbGEuY29tLmJyLyI+U3ltcGxhIEFQSTwvYT4KICA8YnI+CjwvaDE+Cgo8cCBhbGlnbj0iY2VudGVyIj4KICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGFuZ3VhZ2VzL3RvcC9nYWJyaWVsbWFpYWx2YTMzL2RvYy1zeW1wbGEtaW50ZWdyYXRpb24/c3R5bGU9ZmxhdCZsb2dvPWFwcHZleW9yIiBhbHQ9IkdpdEh1YiB0b3AgbGFuZ3VhZ2UiID4KICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGFuZ3VhZ2VzL2NvdW50L2dhYnJpZWxtYWlhbHZhMzMvZG9jLXN5bXBsYS1pbnRlZ3JhdGlvbj9zdHlsZT1mbGF0JmxvZ289YXBwdmV5b3IiIGFsdD0iR2l0SHViIGxhbmd1YWdlIGNvdW50IiA+CiAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3JlcG8tc2l6ZS9nYWJyaWVsbWFpYWx2YTMzL2RvYy1zeW1wbGEtaW50ZWdyYXRpb24/c3R5bGU9ZmxhdCZsb2dvPWFwcHZleW9yIiBhbHQ9IlJlcG9zaXRvcnkgc2l6ZSIgPgogIDxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL2dhYnJpZWxtYWlhbHZhMzMvZG9jLXN5bXBsYS1pbnRlZ3JhdGlvbj9jb2xvcj0wMGI4ZDM/c3R5bGU9ZmxhdCZsb2dvPWFwcHZleW9yIiBhbHQ9IkxpY2Vuc2UiIC8+IAogIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9nYWJyaWVsbWFpYWx2YTMzL2RvYy1zeW1wbGEtaW50ZWdyYXRpb24vY29tbWl0cy9tYXN0ZXIiPgogICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xhc3QtY29tbWl0L2dhYnJpZWxtYWlhbHZhMzMvZG9jLXN5bXBsYS1pbnRlZ3JhdGlvbj9zdHlsZT1mbGF0JmxvZ289YXBwdmV5b3IiIGFsdD0iR2l0SHViIGxhc3QgY29tbWl0IiA+CiAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9tYWRlJTIwYnktTWFpYS0xNWMzZDY/c3R5bGU9ZmxhdCZsb2dvPWFwcHZleW9yIiBhbHQ9Ik1haWEiID4gIAogIDwvYT4KPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iI2Jvb2ttYXJrLWFib3V0Ij5BYm91dDwvYT4mbmJzcDsmbmJzcDsmbmJzcDt8Jm5ic3A7Jm5ic3A7Jm5ic3A7CiAgPGEgaHJlZj0iI2NvbXB1dGVyLXRlY2hub2xvZ2llcyI+VGVjaG5vbG9naWVzPC9hPiZuYnNwOyZuYnNwOyZuYnNwO3wmbmJzcDsmbmJzcDsmbmJzcDsKICA8YSBocmVmPSIjcGFja2FnZS1pbnN0YWxsYXRpb24iPkluc3RhbGxhdGlvbjwvYT4mbmJzcDsmbmJzcDsmbmJzcDt8Jm5ic3A7Jm5ic3A7Jm5ic3A7CiAgPGEgaHJlZj0iI3dyZW5jaC1jb25maWd1cmF0aW9uIj5Db25maWd1cmF0aW9uPC9hPiZuYnNwOyZuYnNwOyZuYnNwO3wmbmJzcDsmbmJzcDsmbmJzcDsKICA8YSBocmVmPSIjbWVtby1kb2N1bWVudGF0aW9uIj5Eb2N1bWVudGF0aW9uPC9hPiZuYnNwOyZuYnNwOyZuYnNwO3wmbmJzcDsmbmJzcDsmbmJzcDsKICA8YSBocmVmPSIjbWVtby1saWNlbnNlIj5MaWNlbnNlPC9hPgo8L3A+Cgo8YnI+CgojIyA6Ym9va21hcms6IEFib3V0CgoqKkRvYzogU3ltcGxhIEFQSSoqIGlzIGEgZG9jdW1lbnRhdGlvbiBmb3IgW1N5bXBsYSBBUEldKGh0dHBzOi8vZGV2ZWxvcGVycy5zeW1wbGEuY29tLmJyLykKCjxicj4KCiMjIDpjb21wdXRlcjogVGVjaG5vbG9naWVzCgotICoqW09wZW5BUEldKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykqKgotICoqW0luc29tbmlhXShodHRwczovL2luc29tbmlhLnJlc3QvKSoqCi0gKipbU3ltcGxhIEFQSV0oaHR0cHM6Ly9kZXZlbG9wZXJzLnN5bXBsYS5jb20uYnIvKSoqCgojIyA6cGFja2FnZTogSW5zdGFsbGF0aW9uCgpgYGBiYXNoCiMgY2xvbmUgdGhlIHJlcG9zaXRvcnkKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9nYWJyaWVsbWFpYWx2YTMzL2RvYy1zeW1wbGEtaW50ZWdyYXRpb24uZ2l0CiMgZW50ZXIgdGhlIGRpcmVjdG9yeQpjZCBkb2Mtc3ltcGxhLWludGVncmF0aW9uCiMgdXNlIGluc28gdG8gZ2VuZXJhdGUgdGhlIGRvY3VtZW50YXRpb24KaW5zbyBnZW5lcmF0ZSAtLXNwZWMgZG9jLXN5bXBsYS1pbnRlZ3JhdGlvbi55YW1sIC0tdHlwZSBvcGVuYXBpIC0tb3V0cHV0IGRvYy1zeW1wbGEtaW50ZWdyYXRpb24uanNvbgpgYGAKCiMjIyA6d3JlbmNoOiAqKkNvbmZpZ3VyYXRpb24qKgoKb3BlbiB0aGUgYGRvYy1zeW1wbGEueWFtbGAgZmlsZSBpbiBpbnNvbW5pYSBvciBwb3N0bWFuIGFuZCBjaGFuZ2UgdGhlIGhvc3QgYW5kIGJhc2VQYXRoCgpgYGB5YW1sCmhvc3Q6ICJhcGkuc3ltcGxhLmNvbS5iciIKYmFzZVBhdGg6ICIvcHVibGljL3YzIgpgYGAKCiMjIyA6bWVtbzogKipEb2N1bWVudGF0aW9uKioKCmBgYG1kCiMgVXNlIGluc29tbmlhIG9yIHBvc3RtYW4gdG8gdGVzdCB0aGUgcm91dGVzCgpmaWxlIGBkb2Mtc3ltcGxhLnlhbWxgIGluIHJvb3QgZGlyZWN0b3J5CmBgYAoKIyMjIDp3cml0aW5nX2hhbmQ6ICoqQXV0aG9yKioKCnwgWyFbTWFpYV0oaHR0cHM6Ly9hdmF0YXJzLmdpdGh1YnVzZXJjb250ZW50LmNvbS91LzI2NzMyMDY3P3NpemU9MTAwKV0oaHR0cHM6Ly9naXRodWIuY29tL2dhYnJpZWxtYWlhbHZhMzMpIHwKfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8IFtNYWlhXShodHRwczovL2dpdGh1Yi5jb20vZ2FicmllbG1haWFsdmEzMykgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CgojIyBMaWNlbnNlCgpbTUlUIExpY2Vuc2VdKC4vTElDRU5TRSkK readmeEtag: '"9abcf7250c81d899050a9ec02e012042b4d2ceba"' readmeLastModified: Sun, 27 Aug 2023 19:06:39 GMT repositoryId: 680255346 description: Documentação de integração com a API da Sympla created: '2023-08-18T18:19:15Z' updated: '2025-02-22T16:05:57Z' language: null archived: false stars: 2 watchers: 0 forks: 0 owner: gabrielmaialva33 logo: https://avatars.githubusercontent.com/u/26732067?v=4 repoEtag: '"3cd7fdecc244561ee2c7199731af39ba8b0dcb493064a3dd80cb0a370b2d1bab"' repoLastModified: Sat, 22 Feb 2025 16:05:57 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/edenreich/definition-driven-development v3: true id: 8f5af38daa82c5a35a07e59347544518 repositoryMetadata: base64Readme: >- # Definition-driven Development

- [Definition-driven Development](#definition-driven-development)
  - [One-Off / Initialize a Project](#one-off--initialize-a-project)
  - [Development Process](#development-process)
  - [Customization](#customization)
  - [Validation](#validation)
  - [Todos](#todos)

It's not a new concept, yet it has gained more popularity over the last few years due to an improvements in existing tooling.

Definition-driven development is the approach where a clear and well-defined understanding of project requirements and goals guides the entire development process. This approach emphasizes the importance of creating a detailed specification or definition of the project before initiating any coding or implementation.

Having a comprehensive definition upfront enables developers to better understand what needs to be achieved, leading to more efficient and focused development efforts. This approach helps minimize misunderstandings, reduces the likelihood of scope changes, and provides a roadmap for the development team to follow. It is often associated with methodologies like Waterfall, where the development process is sequential and relies on a thorough upfront planning phase.

Parallel Development is one of the key benefits of this approach. With a well-established interface, client and server development can proceed concurrently. Different teams or developers can work on the client and server components simultaneously, making the development process more efficient and reducing unclarity.

Once the interface is well-defined, it's also possible to adjust specific requirements and focus on generating only the things that are frequently changing from that interface.

Relying on OAS as the source of truth enables us to leverage the latest OAS version, specifically version 3.0.0 and beyond. As of the time of writing, when employing tools such as Swaggo, which generates OAS from Go code using annotations, there is a constraint limiting compatibility to OAS version 2.0.0. Furthermore, being independent of the annotations tied to a particular programming language enhances our flexibility, making us language-agnostic.

This method is frequently referred to as the "Design-First" approach. By following this strategy, seamless collaboration among various stakeholders becomes easily achievable.

## One-Off / Initialize a Project

All you need is an OpenAPI Specification (OAS). Once you have it well-defined, you can generate the first `client` and `server` projects in any desired programming language using `openapi-generator-cli`. If you're familiar with `Docker`, you can view the commands I placed in the Makefile; those container images already provide the `openapi-generator-cli` and the tooling needed for generating the code.

After the boilerplate code has been generated, it's necessary to review and determine what is really needed and what is frequently changing. The things that are likely never to change should not be generated, so add them to the `.openapi-generator-ignore` file.

Make the adjustments to those files you don't want to generate.

## Development Process

Let's say a new business requirements arrive at the table. They need a new endpoint.

1. Check out a new branch from the `main` upstream.
2. Design the endpoint by adjusting the source of truth, the OAS (`pets_api.yaml`).
3. Run `make generate`. Which will run the `openapi-generator-cli` for generating the `client` and `server`
4. Fill in the todos and write the business logic / implementation.
5. Add the file where you wrote the implementation to the generator ignore file, so it will know not to regenerate it from the template.
6. Create a Pull Request / Merge Request.
7. Let your co-worker review.
8. Once approved, merge it in.

Let's say those who work on the `client` are on a different team than those who work on the `server`. In that scenario, you can still generate the code for the `client` even on a completely separate repository, but the interface (the `OpenAPI Specification`) has to be with the same version. This way, teams can first design the specification and then work in parallel.

## Customization

Every project comes with unique requirements, making this aspect likely the most challenging part of the equation. However, once you successfully establish the setup, the subsequent workflow and minimal adjustments required for implementation can significantly boost the development velocity.

Utilizing pre-existing templates empowers us to select the generated API code seamlessly. There are instances, though, where bespoke templates become necessary, and fortunately, it's entirely feasible to override the existing ones.

By overriding the default templates, we gain the ability to delve into specific implementation details, thereby enhancing the speed of development. This flexibility ensures a tailored approach that aligns precisely with the project's distinctive needs.

Let's modify an existing template:

1. First let's fetch the template:
```bash
make openapi ARGS='author template -g go-server -o templates/go-server'
```
2. Modify it.
3. Modify the generate command by adding `-t templates/go-server`.

A variety of additional templates crafted by the open-source community are available for selection:

- ada
- ada-server
- android
- apache2
- apex
- asciidoc
- aspnetcore
- avro-schema
- bash
- crystal
- c
- clojure
- cwiki
- cpp-qt-client
- cpp-qt-qhttpengine-server
- cpp-pistache-server
- cpp-restbed-server
- cpp-restbed-server-deprecated
- cpp-restsdk
- cpp-tiny
- cpp-tizen
- cpp-ue4
- csharp
- csharp-functions
- dart
- dart-dio
- eiffel
- elixir
- elm
- erlang-client
- erlang-proper
- erlang-server
- fsharp-functions
- fsharp-giraffe-server
- go
- go-echo-server
- go-server
- go-gin-server
- graphql-schema
- graphql-nodejs-express-server
- groovy
- kotlin
- kotlin-server
- kotlin-spring
- kotlin-vertx
- ktorm-schema
- haskell-http-client
- haskell
- haskell-yesod
- java
- jaxrs-cxf-client
- java-helidon-client
- java-helidon-server
- java-inflector
- java-micronaut-client
- java-micronaut-server
- java-msf4j
- java-pkmst
- java-play-framework
- java-undertow-server
- java-vertx
- java-vertx-web
- java-camel
- jaxrs-cxf
- jaxrs-cxf-extended
- jaxrs-cxf-cdi
- jaxrs-jersey
- jaxrs-resteasy
- jaxrs-resteasy-eap
- jaxrs-spec
- javascript
- javascript-apollo-deprecated
- javascript-flowtyped
- javascript-closure-angular
- jetbrains-http-client
- jmeter
- julia-client
- julia-server
- k6
- lua
- markdown
- mysql-schema
- n4js
- nim
- nodejs-express-server
- objc
- ocaml
- openapi
- openapi-yaml
- plantuml
- perl
- php
- php-nextgen
- php-laravel
- php-lumen
- php-slim4
- php-symfony
- php-mezzio-ph
- php-dt
- postman-collection
- powershell
- protobuf-schema
- python
- python-pydantic-v1
- python-fastapi
- python-flask
- python-aiohttp
- python-blueplanet
- r
- ruby
- ruby-on-rails
- ruby-sinatra
- rust
- rust-server
- scalatra
- scala-akka
- scala-akka-http-server
- scala-finch
- scala-gatling
- scala-lagom-server
- scala-play-server
- scala-sttp
- scala-sttp4
- scalaz
- spring
- dynamic-html
- html
- html2
- swift5
- swift-combine
- typescript
- typescript-angular
- typescript-aurelia
- typescript-axios
- typescript-fetch
- typescript-inversify
- typescript-jquery
- typescript-nestjs
- typescript-node
- typescript-redux-query
- typescript-rxjs
- wsdl-schema
- xojo-client
- zapier

## Validation

To validate the schema of the OAS, just run:
```bash
make openapi ARGS='validate -i pets_api.yaml'
```

## Todos

- [ ] cover virtual services / mocked services
- [ ] generate metrics based of customisations attributes in the OAS
 readmeEtag: '"b6c177ef295f7a8297e7e29226b5e9c1ded9322b"' readmeLastModified: Sun, 17 Dec 2023 23:39:22 GMT repositoryId: 732375254 description: >- Definition-driven development is the approach where a clear and well-defined understanding of project requirements and goals guides the entire development process. This approach emphasizes the importance of creating a detailed specification or definition of the project before initiating any coding or implementation. created: '2023-12-16T13:22:24Z' updated: '2023-12-20T18:31:09Z' language: Mustache archived: false stars: 1 watchers: 1 forks: 0 owner: edenreich logo: https://avatars.githubusercontent.com/u/16985712?v=4 repoEtag: '"e57e0bae05fb843c6b570f03cca8a5d45aad58b645f6e10530139c9e7e27a003"' repoLastModified: Wed, 20 Dec 2023 18:31:09 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/lavandejoey/auth-flutter-hmi-coursework v3: true id: d05d1050c4665a1cbbe8c4dbf7e957c9 repositoryMetadata: base64Readme: >- IyBbQXV0aCBGbHV0dGVyIEhNSSBDb3Vyc2V3b3JrXShodHRwczovL2dpdGh1Yi5jb20vbGF2YW5kZWpvZXkvQXV0aC1GbHV0dGVyLUhNSS1Db3Vyc2V3b3JrKQoKWyFbU3RhcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3N0YXJzL2xhdmFuZGVqb2V5L0F1dGgtRmx1dHRlci1ITUktQ291cnNld29yay0uc3ZnKV0oKQpbIVtMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtTUlULWdyZWVuP2xvZ289bWl0KV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQpClshW0ZsdXR0ZXIgVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9GbHV0dGVyLXYzLjEzLjktYmx1ZT9sb2dvPWZsdXR0ZXIpXShodHRwczovL2ZsdXR0ZXIuZGV2LykKWyFbRGFydCBWZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0RhcnQtdjMuMS41LWJsdWU/bG9nbz1kYXJ0KV0oaHR0cHM6Ly9kYXJ0LmRldi8pClshW0FuZHJvaWQgVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9BbmRyb2lkLXYxNC1ibHVlP2xvZ289YW5kcm9pZCldKGh0dHBzOi8vZGV2ZWxvcGVyLmFuZHJvaWQuY29tL3N0dWRpby9yZWxlYXNlcy9wbGF0Zm9ybXMjNC4wKQpbIVtPcGVuQVBJIFZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvT3BlbkFQSS12My4wLjAtYmx1ZT9sb2dvPW9wZW5hcGktaW5pdGlhdGl2ZSldKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykKClRoaXMgaXMgYSBjb3Vyc2V3b3JrIGZvciB0aGUgSHVtYW4gTWFjaGluZSBJbnRlcmFjdGlvbiBjb3Vyc2UgYXQgRUNVU1QuClRoZSBnb2FsIG9mIHRoaXMgcHJvamVjdCBpcyB0byBjcmVhdGUgYSBtb2JpbGUgYXBwbGljYXRpb24gdGhhdCBhbGxvd3MgdXNlcnMgdG8gcmVnaXN0ZXIgYW5kIGxvZ2luIHRvIHRoZSBzeXN0ZW0uCgojIyBHZXR0aW5nIFN0YXJ0ZWQKClRoaXMgcHJvamVjdCBpcyBhIHN0YXJ0aW5nIHBvaW50IGZvciBhIEZsdXR0ZXIgYXBwbGljYXRpb24uCgpBIGZldyByZXNvdXJjZXMgdG8gZ2V0IHlvdSBzdGFydGVkIGlmIHRoaXMgaXMgeW91ciBmaXJzdCBGbHV0dGVyIHByb2plY3Q6CgotIFtMYWI6IFdyaXRlIHlvdXIgZmlyc3QgRmx1dHRlciBhcHBdKGh0dHBzOi8vZG9jcy5mbHV0dGVyLmRldi9nZXQtc3RhcnRlZC9jb2RlbGFiKQotIFtDb29rYm9vazogVXNlZnVsIEZsdXR0ZXIgc2FtcGxlc10oaHR0cHM6Ly9kb2NzLmZsdXR0ZXIuZGV2L2Nvb2tib29rKQoKRm9yIGhlbHAgZ2V0dGluZyBzdGFydGVkIHdpdGggRmx1dHRlciBkZXZlbG9wbWVudCwgdmlldyB0aGUKW29ubGluZSBkb2N1bWVudGF0aW9uXShodHRwczovL2RvY3MuZmx1dHRlci5kZXYvKSwgd2hpY2ggb2ZmZXJzIHR1dG9yaWFscywKc2FtcGxlcywgZ3VpZGFuY2Ugb24gbW9iaWxlIGRldmVsb3BtZW50LCBhbmQgYSBmdWxsIEFQSSByZWZlcmVuY2UuCgojIyBVcGRhdGUgSGlzdG9yeQoKLSBbMjAyMy0xMi0zMF06IEhNSSBjb3Vyc2V3b3JrIHN1Ym1pc3Npb24gdmVyc2lvbiAodjEuMC4wKS4KLSBbMjAyMy0xMS0yOF06IENvbXBsZXRlIGZ1bmN0aW9ucyBhbmQgVUkgb2YgbG9naW4gYW5kIHJlZ2lzdGVyLgotIFsyMDIzLTExLTAzXTogQ29tcGxldGUgZnVuY3Rpb25zIGFuZCBVSSBvZiB1cGRhdGUgdXNlciBpbmZvcm1hdGlvbi4KLSBbMjAyMy0xMS0wMl06IFByb2plY3QgY3JlYXRlZC4gQVBJIHdpdGggT3BlbkFQSSAzLjAuMCBjcmVhdGVkLgoKIyMgQ29weVJpZ2h0CgpUaGlzIHByb2plY3QgaXMgbGljZW5zZWQgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbTUlUIGxpY2Vuc2VdKExJQ0VOU0UpLgo= readmeEtag: '"c557b4c59ccded32ae0978a54f1bad22476acfc5"' readmeLastModified: Sat, 30 Dec 2023 08:15:17 GMT repositoryId: 713267738 description: >- This is a coursework for the Human Machine Interaction course at ECUST. The goal of this project is to create a mobile application that allows users to register and login to the system. created: '2023-11-02T07:11:30Z' updated: '2025-06-10T04:04:50Z' language: Dart archived: false stars: 3 watchers: 1 forks: 0 owner: lavandejoey logo: https://avatars.githubusercontent.com/u/48612765?v=4 license: MIT repoEtag: '"48802473636c8f29853c069274b15df98e3566d3c6e5d542189d2deebdceae2f"' repoLastModified: Tue, 10 Jun 2025 04:04:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/douglanyabasa/universitystudentportal-springboot v3: true id: d689e1ad713dfc5dfe4eb9b74040e974 repositoryMetadata: base64Readme: >- IyBVbml2ZXJzaXR5U3R1ZGVudFBvcnRhbC1zcHJpbmdib290ClVuaXZlcnNpdHkgU3R1ZGVudHMgcG9ydGFsIGluIHNwcmluZ2Jvb3QuCgpvcGVuIHRoZSBtYXN0ZXIgYnJhbmNoIHRvIHZpZXcgdGhlIGNvZGUuCg== readmeEtag: '"840099264a8b3676652d2cf3a7c624ee9e071d2185bb665355b7dfe8553be6c7"' readmeLastModified: Tue, 19 Dec 2023 12:18:30 GMT repositoryId: 670556368 description: 'University Students portal implemented in springboot ' created: '2023-07-25T10:11:13Z' updated: '2025-07-01T09:04:50Z' language: Java archived: false stars: 2 watchers: 1 forks: 0 owner: DouglaNyabasa logo: https://avatars.githubusercontent.com/u/99015779?v=4 repoEtag: '"92853bed76ed42dcb268bf02fa3f0cccf4e371f80408ed011e8b8c25f7d1ec7e"' repoLastModified: Tue, 01 Jul 2025 09:04:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/emiliocliff/payment-polling-service v3: true id: 00071fb45557d0a5fcd6fc99666c5f83 repositoryMetadata: base64Readme: >- IyBQYXltZW50IFBvbGxpbmcgU3lzdGVtIPCfmoAKCldlbGNvbWUgdG8gdGhlICoqUGF5bWVudCBQb2xsaW5nIFN5c3RlbSoqISBUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgYSBjb2xsZWN0aW9uIG9mIG1pY3Jvc2VydmljZXMgYnVpbHQgd2l0aCBHbyBmb3IgdGhlIFtQYXlkIEJhY2tlbmQgQXNzZXNtZW50XSguaHR0cHM6Ly9naXRodWIuY29tL2dldHBheWQtdGVjaC9iYWNrZW5kLWludGVybi1hc3Nlc21lbnQpLiBUaGUgcHJvamVjdCBkZW1vbnN0cmF0ZXMgYSBtaWNyb3NlcnZpY2VzIGFyY2hpdGVjdHVyZSBidWlsdCB3aXRoIEdvLCBEb2NrZXIsIGFuZCBSYWJiaXRNUS4gRWFjaCBzZXJ2aWNlIGlzIHNlbGYtY29udGFpbmVkIGFuZCBzZXJ2ZXMgYSBzcGVjaWZpYyBwdXJwb3NlIGluIHRoZSBsYXJnZXIgYXJjaGl0ZWN0dXJlLiBOYXZpZ2F0ZSB0byBlYWNoIHNlcnZpY2UgdG8gbGVhcm4gbW9yZSEKCiFbTWljcm9zZXJ2aWNlcyBJbWFnZV0oUGF5bWVudHNfUG9sbGluZy5wbmcpCgojIyBTZXJ2aWNlcyBPdmVydmlldyDwn6epCgotIFtBdXRoZW50aWNhdGlvbiBTZXJ2aWNlXSguL2F1dGhlbnRpY2F0aW9uLXNlcnZpY2UvUkVBRE1FLm1kKSAtIFRoZSBhdXRoZW50aWNhdGlvbiBzZXJ2aWNlIGNvbm5lY3RzIHRvIGEgcG9zdGdyZXMgZGF0YWJhc2UuIEl0IHNlcnZlcyB0aGUgcHVycG9zZSBvZiByZWdpc3RlcmluZyBhIG5ldyB1c2VyIGFuZCBsb2dnaW5nIHRoZSB1c2VyLgotIFtHYXRld2F5IFNlcnZpY2VdKC4vZ2F0ZXdheS1zZXJ2aWNlL1JFQURNRS5tZCkgLSBUaGlzIGlzIHRoZSBnYXRld2F5L2VudHJ5IHBvaW50IHRvIG91ciBzeXN0ZW0uIENsaWVudHMgaW50ZXJhY3Qgd2l0aCB0aGlzIHNlcnZpY2UgYW5kIHRoZSByZXF1ZXN0IGlzIGZvcndhcmRlZCB0byB0aGUgcmVzcGVjdGl2ZSBzZXJ2aWNlIGVpdGhlciB0aHJvdWdoIEhUVFAgcmVxdWVzdHMsIGdSUEMgb3IgYSByYWJiaXRNUSBxdWV1ZS4KLSBbUGF5bWVudHMgU2VydmljZV0oLi9wYXltZW50cy1zZXJ2aWNlL1JFQURNRS5tZCkgLSBUaGUgcGF5bWVudHMgc2VydmljZSBjb25uZWN0cyB0byBhIHBvc3RncmVzIGRhdGFiYXNlLiBJdCBzZXJ2ZXMgdGhlIHB1cnBvc2Ugb2YgZGVwb3NpdGluZyBhbmQgd2l0aGRyYXdpbmcgZnVuZHMgZnJvbSB5b3VyIFBheWQgV2FsbGV0LgoKIyMgU2hhcmVkIGdSUEMKCi0gW1NoYXJlZCBnUlBDXSguL3NoYXJlZC1ncnBjL1JFQURNRS5tZCkgLSBUaGUgc2hhcmVkIGdSUEMgY29udGFpbnMgdGhlIHByb2NlZHVyZSBjYWxscyBuZWVkZWQgZm9yIHRoZSBzeXN0ZW0gdG9nZXRoZXIgd2l0aCBpdHMgbW9ja3MgZm9yIHRlc3RpbmcgcHVycG9zZXMuCgojIyBUZWNoIFN0YWNrIPCfm6DvuI8KCi0gKipHbyoqOiBCYWNrZW5kIHNlcnZpY2VzLgotICoqZ1JQQyoqOiBTZXJ2aWNlIGNvbW11bmljYXRpb24uCi0gKipEb2NrZXIqKjogQ29udGFpbmVyaXphdGlvbi4KLSAqKlJhYmJpdE1RKio6IE1lc3NhZ2UgYnJva2VyLgotICoqUG9zdGdyZXNxbCoqOiBTZXJ2aWNlIERhdGFiYXNlLgotICoqVGVzdGNvbnRhaW5lcnMqKjogSW50ZXJncmF0aW9uIHRlc3RpbmcgdG9vbC4KCkV4cGxvcmUgdGhlIHNlcnZpY2VzIGJ5IHZpc2l0aW5nIHRoZWlyIGRpcmVjdG9yaWVzIGZvciBtb3JlIGRldGFpbHMuCgojIyBHZXR0aW5nIFN0YXJ0ZWQg8J+boO+4jwoKVG8gZ2V0IHN0YXJ0ZWQgd2l0aCB0aGUgZW50aXJlIHByb2plY3QsIGZvbGxvdyB0aGVzZSBzdGVwczoKCjEuICoqQ2xvbmUgdGhlIHJlcG9zaXRvcnkqKgoKICAgYGBgYmFzaAogICBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL0VtaWxpb0NsaWZmL3BheW1lbnQtcG9sbGluZy1zZXJ2aWNlLmdpdAogICBgYGAKClRoaXMgbmV4dCBzdGVwIHJlcXVpcmUgeW91IHRvIGhhdmUgZG9ja2VyIGNvbXBvc2UgaW5zdGFsbGVkIGFuZCB0byBzZXQgdXAgYW4gYWNjb3VudCBpbiBbUGF5ZCBBY2NvdW50XSguaHR0cHM6Ly93ZWIubXlwYXlkLmFwcC9sb2dpbikgZm9yIHBheW1lbnRzIHByb2Nlc3NpbmcuCgoyLiAqKkdlbmVyYXRpbmcgUlNBIGtleSBwYWlyKioKCiAgIFdlIHdpbGwgbmVlZCB0byBnZW5lcmF0ZSBSU0EgcHJpdmF0ZSBhbmQgcHVibGljIGtleXMgZm9yIG91ciBzeXN0ZW0uIFRoZSBSU0EgcHJpdmF0ZSBrZXkgaXMgbmVlZGVkIHRvICoqc2lnbiBKV1QgdG9rZW5zKiogdXNpbmcgdGhlIFJTMjU2IGFsZ29yaXRobSwgYW5kIHRoZSBjb3JyZXNwb25kaW5nIHB1YmxpYyBrZXkgaXMgdXNlZCB0byAqKnZlcmlmeSB0aG9zZSB0b2tlbnMqKi4gVGhpcyBpcyBlc3NlbnRpYWwgZm9yIGFzeW1tZXRyaWMgY3J5cHRvZ3JhcGh5LCB3aGVyZSBvbmx5IHRoZSBob2xkZXIgb2YgdGhlIHByaXZhdGUga2V5IGNhbiBnZW5lcmF0ZSB2YWxpZCB0b2tlbnMgKGF1dGhlbnRpY2F0aW9uLXNlcnZpY2UpLCBidXQgYW55b25lIHdpdGggdGhlIHB1YmxpYyBrZXkgY2FuIHZlcmlmeSB0aGVpciBhdXRoZW50aWNpdHkgKGdhdGV3YXkgc2VydmljZSkKCiAgIC0gTmF2aWdhdGUgdG8gdGhlIGAuL2F1dGhlbnRpY2F0aW9uLXNlcnZpY2UvcGtnYCBhbmQgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZAoKICAgYGBgCiAgICMgR2VuZXJhdGUgcHJpdmF0ZSBrZXkKCiAgIG9wZW5zc2wgZ2VucGtleSAtYWxnb3JpdGhtIFJTQSAtb3V0IG15X3JzYV9rZXkucGVtIC1wa2V5b3B0IHJzYV9rZXlnZW5fYml0czoyMDQ4CiAgIGBgYAoKICAgYGBgCiAgICMgR2VuZXJhdGUgcHVibGljIGtleQoKICAgb3BlbnNzbCByc2EgLXB1Ym91dCAtaW4gbXlfcnNhX2tleS5wZW0gLW91dCBteV9yc2Ffa2V5LnB1Yi5wZW0KICAgYGBgCgogICAtIENvcHkgdGhlIHB1YmxpYyBrZXkgdG8gdGhlIGAuL2dhdGV3YXktc2VydmljZS9wa2dgCgozLiAqKlJ1biB0aGUgc2VydmljZXMqKgoKICAgYGBgYmFzaAogICBkb2NrZXIgY29tcG9zZSB1cAogICBgYGAKCjQuICoqTWFraW5nIFJlcXVlc3QqKgogICBBZnRlciBhbGwgdGhlIHNlcnZpY2VzIGFyZSB1cCB5b3UgY2FuIGdvIHRvIHRoZSBgbG9jYWxob3N0OjgwODAvc3dhZ2dlci9pbmRleC5odG1sYC4gU3dhZ2dlclVJIGlzIHVzZWQgZm9yIHRoZSBnYXRld2F5IHNlcnZpY2UgZG9jdW1lbnRhdGlvbiB3aXRoIHJlcXVlc3QgYW5kIHJlc3BvbnNlIGV4YW1wbGVzLCBoZXJlIHlvdSBjYW4gc3RhcnQgaW50ZXJhY3Rpbmcgd2l0aCB0aGUgc3lzdGVtLgoKIVtNaWNyb3NlcnZpY2VzIEltYWdlXShnYXRld2F5LXNlcnZpY2UvZG9jcy9zd2FnZ2VyL3N3YWdnZXItaW1nLnBuZykKCiMjIFJlbWFya3Mg8J+knQoKSGFwcHkgQ29kaW5nCgojIyBMaWNlbnNlIPCfk50KClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuCg== readmeEtag: '"a7ff57653418ee4a24d9fd734b8936e87764fba1"' readmeLastModified: Tue, 01 Oct 2024 22:03:06 GMT repositoryId: 836114430 description: >- This project is an online payment polling service build with microservices-architecture. Integrated with Payd API and was part of a backend assessment for an opening. created: '2024-07-31T07:19:32Z' updated: '2024-10-02T19:38:43Z' language: Go archived: false stars: 1 watchers: 1 forks: 1 owner: EmilioCliff logo: https://avatars.githubusercontent.com/u/115647640?v=4 license: MIT repoEtag: '"8f36b553e01301ace9c78621b5a9d7b2d2fafb9127fc3fd8acac0acc764e77b8"' repoLastModified: Wed, 02 Oct 2024 19:38:43 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/bzdgn/open-api-web-service-example v3: true id: 42b6a048ebc1b81b674d425e2f9e63b0 repositoryMetadata: base64Readme: >- This project is a demonstration of OpenAPI Web Service with Spring Boot.

TOC
---
- [0 Introduction](#0-introduction) <br/>
- [1 Setup](#1-setup) <br/>
- [2 OpenAPI Specs](#2-openapi-specs) <br/>
- [3 OpenAPI Plugin](#3-openapi-plugin) <br/>
- [4 Testing with Postman Collection](#4-testing-with-postman-collection)<br/>

 0 Introduction
---------------
This project is a demonstration of OpenAPI Web Service with Spring Boot.

With OpenAPI, you can define the models and your service specifications in yaml format and build the skeleton of your web service.

There are good and bad parts and I'm going to list my personal opinions;

The good parts of OpenAPI are;

- The boilerplate code is automatically generated
- You can define the models within the yaml files and they are auto-generated
- The endpoint skeleton is also generated. You can either use the generated controller or override it to customize it.

The bad parts of OpenAPI are;

- You need to dig around the documentation and the net for the use of plugins
- It adds complexity when you are using OpenAPI within a multi-module project
- Using the generated models in a distinct shared project and injecting the resources is not easy

In this project, I'm not going to dive into these topics. This project will give you a simple example of OpenAPI web service.


[TOC](#toc)


 1 Setup
--------
We use OpenAPI to generate models, generate a template for the web service and also helper classes. So that within one simple yaml file, the model classes and also the api template is generated for the controller of the web service.

There are two files that does the magic for the auto-generated code/template;

- The OpenAPI spec file(s) in yaml format, that define(s) the web service skeleton and schemas. These file(s) placed in the **src/main/resources/openapi** folder.
- The build plugin that generates the code based on the OpenAPI yaml file. This is defined in the **pom** file.

In the pom file, build/plugin section has reference for spec file and package names for model and service packages. Plugin uses the yaml spec and generates the model/service classes under the defined packages.
Then these generated files are placed in the target/generated-sources/src folder as can be seen below;

![generated_files](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/01_generated_files.PNG)

After importing it to an IDE (I use eclipse), you need to add the OpenAPI generated code into the build path.

1. The files need to be added will be under **target/generated-sources/src/main/java**

![target_generated_files](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/02_target_generated_files.PNG)

2. Right click to this folder and add it to the build path

![right_click_generated_files](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/03_build_path.PNG)

3. See the generated code is visible in the project

![add_generated_files_to_build_path](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/04_added_build_path.PNG)


[TOC](#toc)


 2 OpenAPI Specs
----------------
The spec file defines the endpoints, and the structure of the model classes those are used in endpoints. We have two get operations and one post operation. These operations uses Person model for request or response and one operation returns an array of Person model.

You can see the file in project sources [here](https://github.com/bzdgn/open-api-web-service-example/blob/main/src/main/resources/openapi/person-api.yaml). The content of the file is as follows;

```yaml
openapi: 3.0.3
info:
  title: Person Service
  description: Public service for querying the persons
  version: 1.0
servers:
  - description: Local development environment
    url: http://localhost:8080/v1
  - description: Kubernetes test environment
    url: http://some-link-here.com/v1
  - description: Kubernetes acc environment
    url: http://some-link-here.com/v1
  - description: Kubernetes production environment
    url: http://some-link-here.com/v1

paths:
  /person/{id}:
    get:
      operationId: getPerson
      tags:
        - person
      parameters:
        - name: id
          in: path
          description: person id
          required: true
          schema:
            type: integer
            format: int32
      responses:
        200:
          description: Person results
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Person'
        400:
          description: not a valid person

  /person:
    get:
      operationId: getPersons
      tags:
        - person
      responses:
        200:
          description: Person results
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/PersonList'
        400:
          description: not a valid person

    post:
      operationId: addPerson
      tags:
        - person
      requestBody:
        required: true
        content:
          application/json:
            schema:
              $ref: '#/components/schemas/Person'
      responses:
        200:
          description: Zoekresultaten
          content:
            application/json:
              schema:
                $ref: '#/components/schemas/Person'
        400:
          description: not a valid person

components:
  schemas:
    Person:
      type: object
      properties:
        id:
          required: true
          description: id of the person
          type: integer
          format: int32
        name:
          description: name of the person
          type: string
        age:
          description: age of the person
          type: integer
          format: int32
    PersonList:
      type: array
      items:
        $ref: '#/components/schemas/Person' 

```

Under the **paths** section, the endpoint is defined. OpenAPI plugin will define the service classes based on this section. Under the **components** section there is **schemas** where the model classes are defined. OpenAPI will generate the model classes based on this section.

Important to notify that the service uses these generated models. If you use maven in command line, the generator will notify you if there are inconsistencies.


[TOC](#toc)


 3 OpenAPI Plugin
-----------------
OpenAPI plugin generates the code based on yaml specs. In the build [plugin section](https://github.com/bzdgn/open-api-web-service-example/blob/main/pom.xml);

```xml
<build>
    <plugins>
        <plugin>
            <groupId>org.openapitools</groupId>
            <artifactId>openapi-generator-maven-plugin</artifactId>
            <version>${openapi-generator-maven-plugin.version}</version>
            <executions>
                <execution>
                    <phase>generate-sources</phase>
                    <goals>
                        <goal>generate</goal>
                    </goals>
                    <configuration>
                        <inputSpec>${project.basedir}/src/main/resources/openapi/person-api.yaml</inputSpec>
                        <generatorName>spring</generatorName>
                        <output>${project.build.directory}/generated-sources/</output>
                        <generateModels>true</generateModels>
                        <generateSupportingFiles>true</generateSupportingFiles>
                        <supportingFilesToGenerate>ApiUtil.java</supportingFilesToGenerate>
                        <modelPackage>io.github.bzdgn.open-api-web-service-example.model</modelPackage>
                        <apiPackage>io.github.bzdgn.open-api-web-service-example.service</apiPackage>
                        <skipValidateSpec>true</skipValidateSpec>
                        <configOptions>
                            <interfaceOnly>true</interfaceOnly>
                            <java11>true</java11>
                            <dateLibrary>java8</dateLibrary>
                            <openApiNullable>false</openApiNullable>
                            <documentationProvider>none</documentationProvider>
                            <annotationLibrary>none</annotationLibrary>
                        </configOptions>
                        <configHelp>false</configHelp>
                    </configuration>
                </execution>
            </executions>
        </plugin>
    </plugins>
</build>
```

If you look at the configuration, a few keys are important;

| variable | description |
| -------- | ----------- |
| inputSpec | This is where you reference your Api spec, in our example, it is the person-api.yaml file |
| output | This is the output folder where all the generated code is placed. Under the project build directory, which is target, it's placed in **generated-sources** folder |
| generateModels | This determines if the model classes are generated, set to **true** |
| generateSupportingFiles | This is the utility file that plugin generates, set to **true** |
| supportingFilesToGenerate | The name of the utility file, in our example **ApiUtil.java** |
| modelPackage | The package that the model generated files placed |
| apiPackage | The package that the service generated files  placed |
| skipValidateSpec | Does the validation in the generation. It's set to **true** in this example but sometimes useful to set to **false** when using multimodule projects |


[TOC](#toc)


 4 Testing with Postman Collection
----------------------------------
I've placed a simple postman collection [under the misc folder, here](https://github.com/bzdgn/open-api-web-service-example/blob/main/misc/OpenAPIWebServiceExample.postman_collection.json). You need to import this collection in Postman to test the example web service.

![postman_collection](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/05_postman_collection.PNG)

Initially the **person** repository will be empty. The implemntation is a simple stub for testing, using an ArrayList to store the posted **Person** objects.

You can post via the Postman;

![postman_collection_post_1](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/06_postman_post_1.PNG)
![postman_collection_post_2](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/07_postman_post_2.PNG)

Then you can get **Person** by id;

![postman_collection_get_by_id](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/08_postman_get_by_id.PNG)

Or you can the whole **Person** repository;

![postman_collection_get_all](https://raw.githubusercontent.com/bzdgn/open-api-web-service-example/main/misc/09_postman_get_all.PNG)


[TOC](#toc)

 readmeEtag: '"7ddc82d1146ca2e105d0dee5873078a19768fbed"' readmeLastModified: Sun, 29 Jan 2023 20:38:10 GMT repositoryId: 594733429 description: Demo Project for Open API Web Service with Spring Boot created: '2023-01-29T13:35:15Z' updated: '2024-04-18T22:57:20Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: bzdgn logo: https://avatars.githubusercontent.com/u/1220904?v=4 repoEtag: '"9e59a12712e8d95d8ede0e923ff124f0c53983a0a529f8618fa45be9aae2f32f"' repoLastModified: Thu, 18 Apr 2024 22:57:20 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/egomobile/node-swagger-proxy v3: true id: 29a02c96fca01f0c28e7b062d0452c93 repositoryMetadata: base64Readme: >- WyFbbnBtXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L0BlZ29tb2JpbGUvc3dhZ2dlci1wcm94eS5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9AZWdvbW9iaWxlL3N3YWdnZXItcHJveHkpClshW2xhc3QgYnVpbGRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3dvcmtmbG93L3N0YXR1cy9lZ29tb2JpbGUvbm9kZS1zd2FnZ2VyLXByb3h5L1B1Ymxpc2gpXShodHRwczovL2dpdGh1Yi5jb20vZWdvbW9iaWxlL25vZGUtc3dhZ2dlci1wcm94eS9hY3Rpb25zP3F1ZXJ5PXdvcmtmbG93JTNBUHVibGlzaCkKWyFbUFJzIFdlbGNvbWVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUFJzLXdlbGNvbWUtYnJpZ2h0Z3JlZW4uc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9naXRodWIuY29tL2Vnb21vYmlsZS9ub2RlLXN3YWdnZXItcHJveHkvcHVsbHMpCgojIEBlZ29tb2JpbGUvc3dhZ2dlci1wcm94eQoKPiBFeHRlbnNpb24gZm9yIFtAZWdvbW9iaWxlL2h0dHAtc2VydmVyXShodHRwczovL2dpdGh1Yi5jb20vZWdvbW9iaWxlL25vZGUtaHR0cC1zZXJ2ZXIpLCB3aGljaCBjb25uZWN0cyB0byBtdWx0aXBseSBbU3dhZ2dlciAvIE9wZW5BUEldKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vYWJvdXQvKSBpbnN0YW5jZXMgYW5kIG1lcmdlIHRoZWlyIGRvY3VtZW50cyB0byBvbmUuCgojIyBJbnN0YWxsCgpFeGVjdXRlIHRoZSBmb2xsb3dpbmcgY29tbWFuZCBmcm9tIHlvdXIgcHJvamVjdCBmb2xkZXIsIHdoZXJlIHlvdXIKYHBhY2thZ2UuanNvbmAgZmlsZSBpcyBzdG9yZWQ6CgpgYGBiYXNoCm5wbSBpbnN0YWxsIC0tc2F2ZSBAZWdvbW9iaWxlL3N3YWdnZXItcHJveHkKYGBgCgojIyBVc2FnZQoKYGBgdHlwZXNjcmlwdAppbXBvcnQgY3JlYXRlU2VydmVyIGZyb20gIkBlZ29tb2JpbGUvaHR0cC1zZXJ2ZXIiOwppbXBvcnQgeyBzZXR1cFN3YWdnZXJQcm94eSB9IGZyb20gIkBlZ29tb2JpbGUvc3dhZ2dlci1wcm94eSI7Cgphc3luYyBmdW5jdGlvbiBtYWluKCkgewogIGNvbnN0IGFwcCA9IGNyZWF0ZVNlcnZlcigpOwoKICBzZXR1cFN3YWdnZXJQcm94eShhcHAsIHsKICAgIGJhc2VEb2N1bWVudDogewogICAgICBpbmZvOiB7CiAgICAgICAgdGl0bGU6ICJNeSBtZXJnZWQgQVBJIiwKICAgICAgICB2ZXJzaW9uOiAiMS4wLjAiLAogICAgICB9LAogICAgfSwKCiAgICBzb3VyY2VzOiBbCiAgICAgIHsKICAgICAgICB1cmw6ICJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYWluL2V4YW1wbGVzL3YzLjAvbGluay1leGFtcGxlLmpzb24iLAogICAgICB9LAogICAgICB7CiAgICAgICAgdXJsOiAiaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vbWFpbi9leGFtcGxlcy92My4wL2NhbGxiYWNrLWV4YW1wbGUueWFtbCIsCiAgICAgIH0sCiAgICAgIHsKICAgICAgICB1cmw6ICJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9tYWluL2V4YW1wbGVzL3YzLjAvcGV0c3RvcmUueWFtbCIsCiAgICAgIH0sCiAgICBdLAogIH0pOwoKICBhd2FpdCBhcHAubGlzdGVuKDgwODApOwoKICBjb25zdCBiYXNlVVJMID0gYGh0dHA6Ly9sb2NhbGhvc3Q6JHthcHAucG9ydH0vc3dhZ2dlcmA7CgogIGNvbnNvbGUubG9nKCJZb3Ugc2hvdWxkIG5vdyBiZSBhYmxlIHRvIGFjY2VzcyBkb2N1bWVudGF0aW9uIGF0OiIpOwogIGNvbnNvbGUubG9nKGAtICR7YmFzZVVSTH1gKTsKCiAgY29uc29sZS5sb2coKTsKCiAgY29uc29sZS5sb2coIllvdSBjYW4gZG93bmxvYWQgZG9jdW1lbnRhdGlvbiBhcyBmaWxlcyBmcm9tOiIpOwogIGNvbnNvbGUubG9nKGAtIEpTT046ICR7YmFzZVVSTH0vanNvbmApOwogIGNvbnNvbGUubG9nKGAtIFlBTUw6ICR7YmFzZVVSTH0veWFtbGApOwp9CgptYWluKCkuY2F0Y2goY29uc29sZS5lcnJvcik7CmBgYAoKIyMgRG9jdW1lbnRhdGlvbgoKVGhlIEFQSSBkb2N1bWVudGF0aW9uIGNhbiBiZSBmb3VuZApbaGVyZV0oaHR0cHM6Ly9lZ29tb2JpbGUuZ2l0aHViLmlvL25vZGUtc3dhZ2dlci1wcm94eS8pLgo= readmeEtag: '"3304865457103efae277209b06df8d1cd5ca11a9"' readmeLastModified: Fri, 19 Aug 2022 07:26:36 GMT repositoryId: 504600843 description: >- Extension for @egomobile/http-server, which connects to multiply Swagger / OpenAPI instances and merge their documents to one. created: '2022-06-17T16:21:27Z' updated: '2024-05-22T17:48:19Z' language: TypeScript archived: true stars: 1 watchers: 0 forks: 0 owner: egomobile logo: https://avatars.githubusercontent.com/u/83799725?v=4 license: LGPL-3.0 repoEtag: '"361a3f06abb714c0a5fe97419ed5f7f6688e1c15d1dc34cc15d6b4c3567c20eb"' repoLastModified: Wed, 22 May 2024 17:48:19 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/akeyless-community/akeyless-openapi-spec-extractor v3: true id: 51e923d150168ae32ce124993762196e repositoryMetadata: base64Readme: >- IyBBa2V5bGVzcyBPcGVuQVBJIFNwZWMgRXh0cmFjdG9yCgpbIVtMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtQXBhY2hlJTIwMi4wLWJsdWUuc3ZnKV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9BcGFjaGUtMi4wKQoKVGhlIEFrZXlsZXNzIE9wZW5BUEkgU3BlYyBFeHRyYWN0b3IgaXMgYSBjb21tYW5kLWxpbmUgaW50ZXJmYWNlIChDTEkpIHRvb2wgdGhhdCBhbGxvd3MgeW91IHRvIGV4dHJhY3Qgc3BlY2lmaWMgZW5kcG9pbnRzIGZyb20gYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uIChTd2FnZ2VyKSBhbG9uZyB3aXRoIGFsbCB0aGUgcmVsZXZhbnQgZGV0YWlscyBvZiB0aGUgZW5kcG9pbnQuIEl0IHByb3ZpZGVzIGEgY29udmVuaWVudCB3YXkgdG8gZmV0Y2ggT3BlbkFQSSBzcGVjcyBmcm9tIGEgVVJMIG9yIHByb2Nlc3MgbG9jYWwgT3BlbkFQSSBzcGVjIGZpbGVzLgoKIyMgUHJvamVjdCBHZW5lc2lzIGFuZCBQdXJwb3NlCgpUaGUgQWtleWxlc3MgT3BlbkFQSSBTcGVjIEV4dHJhY3RvciBwcm9qZWN0IHdhcyBib3JuIG91dCBvZiB0aGUgbmVlZCB0byBleHRyYWN0IGNvbXByZWhlbnNpdmUgaW5mb3JtYXRpb24gYWJvdXQgYSBzcGVjaWZpYyBlbmRwb2ludCBmcm9tIGEgbGFyZ2UgbGlzdCBvZiBlbmRwb2ludHMgaW4gYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uLiBUaGUgZ29hbCB3YXMgdG8gbWFrZSBpdCBlYXNpZXIgdG8gc2VuZCB0aGlzIGluZm9ybWF0aW9uIHRvIGEgbGFyZ2UgbGFuZ3VhZ2UgbW9kZWwgdG8gYXNzaXN0IHdpdGggY29kZSBnZW5lcmF0aW9uIGZvciBpbnRlcmFjdGluZyB3aXRoIHRoYXQgZW5kcG9pbnQgdXNpbmcgdG9vbHMgZm9yIGFnZW50aWMgd29ya2Zsb3dzLgoKQnkgZXh0cmFjdGluZyBhbGwgdGhlIHJlbGV2YW50IGRldGFpbHMgb2YgYW4gZW5kcG9pbnQsIGluY2x1ZGluZyBpbnB1dHMgYW5kIG91dHB1dHMsIHRoZSBBa2V5bGVzcyBPcGVuQVBJIFNwZWMgRXh0cmFjdG9yIHNpbXBsaWZpZXMgdGhlIHByb2Nlc3Mgb2YgaW50ZWdyYXRpbmcgd2l0aCBBUElzIGFuZCBmYWNpbGl0YXRlcyB0aGUgZGV2ZWxvcG1lbnQgb2YgaW50ZWxsaWdlbnQgYWdlbnRzIHRoYXQgY2FuIHVuZGVyc3RhbmQgYW5kIGludGVyYWN0IHdpdGggc3BlY2lmaWMgZW5kcG9pbnRzIGJhc2VkIG9uIHRoZSBleHRyYWN0ZWQgaW5mb3JtYXRpb24uCgojIyBGZWF0dXJlcwoKLSBGZXRjaCBPcGVuQVBJIHNwZWNzIGZyb20gYSBVUkwKLSBQcm9jZXNzIGxvY2FsIE9wZW5BUEkgc3BlYyBmaWxlcwotIFByb2Nlc3MgT3BlbkFQSSBzcGVjcyBmcm9tIHN0ZGluCi0gRXh0cmFjdCBzcGVjaWZpYyBlbmRwb2ludHMgYmFzZWQgb24gdGhlIHByb3ZpZGVkIHBhdGgKLSBPdXRwdXQgdGhlIGV4dHJhY3RlZCBkYXRhIGluIEpTT04gb3IgWUFNTCBmb3JtYXQKLSBDdXN0b21pemUgbG9nZ2luZyBsZXZlbHMgZm9yIGJldHRlciB2aXNpYmlsaXR5IGFuZCBkZWJ1Z2dpbmcKCiMjIEluc3RhbGxhdGlvbgoKVG8gaW5zdGFsbCB0aGUgQWtleWxlc3MgT3BlbkFQSSBTcGVjIEV4dHJhY3RvciwgbWFrZSBzdXJlIHlvdSBoYXZlIFtOb2RlLmpzXShodHRwczovL25vZGVqcy5vcmcpIGluc3RhbGxlZCBvbiB5b3VyIHN5c3RlbS4gVGhlbiwgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYGJhc2gKbnBtIGluc3RhbGwgLWcgYWtleWxlc3Mtb3BlbmFwaS1zcGVjLWV4dHJhY3RvcgpgYGAKClRoaXMgd2lsbCBpbnN0YWxsIHRoZSBDTEkgdG9vbCBnbG9iYWxseSBvbiB5b3VyIHN5c3RlbS4KCiMjIFVzYWdlCgpUaGUgQWtleWxlc3MgT3BlbkFQSSBTcGVjIEV4dHJhY3RvciBwcm92aWRlcyB0aHJlZSBjb21tYW5kczogYGZldGNoYCwgYGxvY2FsYCwgYW5kIGBzdGRpbmAuCgojIyMgRmV0Y2ggQ29tbWFuZAoKVGhlIGBmZXRjaGAgY29tbWFuZCBhbGxvd3MgeW91IHRvIGZldGNoIGFuIE9wZW5BUEkgc3BlYyBmcm9tIGEgVVJMIGFuZCBleHRyYWN0IGEgc3BlY2lmaWMgZW5kcG9pbnQuCgpgYGBiYXNoCm9wZW5hcGktZXh0cmFjdG9yIGZldGNoIC11IDx1cmw+IC1wIDxwYXRoPiBbb3B0aW9uc10KYGBgCgotIGAtdSwgLS11cmwgPHVybD5gOiBVUkwgb2YgdGhlIE9wZW5BUEkgc3BlYyAocmVxdWlyZWQpCi0gYC1wLCAtLXBhdGggPHBhdGg+YDogVGhlIHBhdGggdG8gdGhlIGRlc2lyZWQgZW5kcG9pbnQgd2l0aGluIHRoZSBPcGVuQVBJIHNwZWMgKGluY2x1ZGluZyB0aGUgbGVhZGluZyBzbGFzaCksIGZvciBleGFtcGxlLCAiL2F1dGgiIChyZXF1aXJlZCkKLSBgLW8sIC0tb3V0cHV0IDxvdXRwdXQ+YDogT3V0cHV0IHR5cGUgKGRlZmF1bHQ6ICJqc29uIik6ICJqc29uIiBvciAieWFtbCIKLSBgLWwsIC0tbG9nbGV2ZWwgPGxldmVsPmA6IExvZ2dpbmcgbGV2ZWwgKGRlZmF1bHQ6ICJlcnJvciIpOiAiZXJyb3IiLCAid2FybiIsICJpbmZvIiwgImRlYnVnIgoKIyMjIExvY2FsIENvbW1hbmQKClRoZSBgbG9jYWxgIGNvbW1hbmQgYWxsb3dzIHlvdSB0byBwcm9jZXNzIGEgbG9jYWwgT3BlbkFQSSBzcGVjIGZpbGUgYW5kIGV4dHJhY3QgYSBzcGVjaWZpYyBlbmRwb2ludC4KCmBgYGJhc2gKb3BlbmFwaS1leHRyYWN0b3IgbG9jYWwgLWYgPGZpbGU+IC1wIDxwYXRoPiBbb3B0aW9uc10KYGBgCgotIGAtZiwgLS1maWxlIDxmaWxlPmA6IFBhdGggdG8gdGhlIGxvY2FsIE9wZW5BUEkgc3BlYyBmaWxlIChyZXF1aXJlZCkKLSBgLXAsIC0tcGF0aCA8cGF0aD5gOiBUaGUgcGF0aCB0byB0aGUgZGVzaXJlZCBlbmRwb2ludCB3aXRoaW4gdGhlIE9wZW5BUEkgc3BlYyAoaW5jbHVkaW5nIHRoZSBsZWFkaW5nIHNsYXNoKSwgZm9yIGV4YW1wbGUsICIvYXV0aCIgKHJlcXVpcmVkKQotIGAtbywgLS1vdXRwdXQgPG91dHB1dD5gOiBPdXRwdXQgdHlwZSAoZGVmYXVsdDogImpzb24iKTogImpzb24iIG9yICJ5YW1sIgotIGAtbCwgLS1sb2dsZXZlbCA8bGV2ZWw+YDogTG9nZ2luZyBsZXZlbCAoZGVmYXVsdDogImVycm9yIik6ICJlcnJvciIsICJ3YXJuIiwgImluZm8iLCAiZGVidWciCgojIyMgU3RkaW4gQ29tbWFuZAoKVGhlIGBzdGRpbmAgY29tbWFuZCBhbGxvd3MgeW91IHRvIHByb2Nlc3MgYW4gT3BlbkFQSSBzcGVjIGZyb20gc3RkaW4gYW5kIGV4dHJhY3QgYSBzcGVjaWZpYyBlbmRwb2ludC4KCi0gYC1wLCAtLXBhdGggPHBhdGg+YDogVGhlIHBhdGggdG8gdGhlIGRlc2lyZWQgZW5kcG9pbnQgd2l0aGluIHRoZSBPcGVuQVBJIHNwZWMgKGluY2x1ZGluZyB0aGUgbGVhZGluZyBzbGFzaCksIGZvciBleGFtcGxlLCAiL2F1dGgiIChyZXF1aXJlZCkKLSBgLW8sIC0tb3V0cHV0IDxvdXRwdXQ+YDogT3V0cHV0IHR5cGUgKGRlZmF1bHQ6ICJqc29uIik6ICJqc29uIiBvciAieWFtbCIKLSBgLWwsIC0tbG9nbGV2ZWwgPGxldmVsPmA6IExvZ2dpbmcgbGV2ZWwgKGRlZmF1bHQ6ICJlcnJvciIpOiAiZXJyb3IiLCAid2FybiIsICJpbmZvIiwgImRlYnVnIgoKIyMgRXhhbXBsZXMKCkZldGNoIGFuIE9wZW5BUEkgc3BlYyBmcm9tIGEgVVJMIGFuZCBleHRyYWN0IHRoZSAiL2F1dGgiIGVuZHBvaW50OgoKYGBgYmFzaApvcGVuYXBpLWV4dHJhY3RvciBmZXRjaCAtdSBodHRwczovL2FwaS5leGFtcGxlLmNvbS9vcGVuYXBpLmpzb24gLXAgIi9hdXRoIgpgYGAKClByb2Nlc3MgYSBsb2NhbCBPcGVuQVBJIHNwZWMgZmlsZSBhbmQgZXh0cmFjdCB0aGUgIi91c2VycyIgZW5kcG9pbnQ6CgpgYGBiYXNoCm9wZW5hcGktZXh0cmFjdG9yIGxvY2FsIC1mIC4vb3BlbmFwaS55YW1sIC1wICIvdXNlcnMiCmBgYAoKUHJvY2VzcyBhbiBPcGVuQVBJIHNwZWMgZnJvbSBzdGRpbiBhbmQgZXh0cmFjdCB0aGUgIi9wcm9kdWN0cyIgZW5kcG9pbnQ6CgpgYGBiYXNoCmNhdCBvcGVuYXBpLnlhbWwgfCBvcGVuYXBpLWV4dHJhY3RvciBzdGRpbiAtcCAiL3Byb2R1Y3RzIgpgYGAKCiMjIExpY2Vuc2UKClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UgMi4wLiBTZWUgdGhlIFtMSUNFTlNFXShMSUNFTlNFKSBmaWxlIGZvciBtb3JlIGRldGFpbHMuCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWUhIElmIHlvdSBmaW5kIGFueSBpc3N1ZXMgb3IgaGF2ZSBzdWdnZXN0aW9ucyBmb3IgaW1wcm92ZW1lbnQsIHBsZWFzZSBvcGVuIGFuIGlzc3VlIG9yIHN1Ym1pdCBhIHB1bGwgcmVxdWVzdCBvbiB0aGUgW0dpdEh1YiByZXBvc2l0b3J5XShodHRwczovL2dpdGh1Yi5jb20vYWtleWxlc3MtY29tbXVuaXR5L2FrZXlsZXNzLW9wZW5hcGktc3BlYy1leHRyYWN0b3IpLgoKIyMgQWNrbm93bGVkZ2VtZW50cwoKVGhpcyBDTEkgdG9vbCBpcyBidWlsdCB1c2luZyB0aGUgZm9sbG93aW5nIG9wZW4tc291cmNlIGxpYnJhcmllczoKCi0gW29wZW5hcGktZXh0cmFjdF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS1leHRyYWN0KQotIFtjb21tYW5kZXJdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL2NvbW1hbmRlcikKLSBbYXhpb3NdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL2F4aW9zKQotIFt3aW5zdG9uXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS93aW5zdG9uKQotIFtqcy15YW1sXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9qcy15YW1sKQoKV2Ugd291bGQgbGlrZSB0byBleHByZXNzIG91ciBncmF0aXR1ZGUgdG8gdGhlIG1haW50YWluZXJzIGFuZCBjb250cmlidXRvcnMgb2YgdGhlc2UgbGlicmFyaWVzIGZvciB0aGVpciBleGNlbGxlbnQgd29yay4K readmeEtag: '"97141db11360dfacc85f9184c4a0f5fd3b988ce1"' readmeLastModified: Fri, 26 Apr 2024 16:33:35 GMT repositoryId: 745691544 description: >- Akeyless OpenAPI Spec Extractor: A CLI tool to extract specific endpoints with all relevant details from OpenAPI specs, simplifying API integration and aiding in code generation for agentic workflows using large language models. created: '2024-01-19T22:09:05Z' updated: '2024-12-15T07:40:38Z' language: JavaScript archived: false stars: 1 watchers: 2 forks: 0 owner: akeyless-community logo: https://avatars.githubusercontent.com/u/81695164?v=4 license: Apache-2.0 repoEtag: '"790de75b0792a22dc3188a7d03b3526d359c2c3c58a6cb88e1a887b7a4977ca4"' repoLastModified: Sun, 15 Dec 2024 07:40:38 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/approveapi/openapi-spec v3: true repositoryMetadata: base64Readme: >- IyBBcHByb3ZlQVBJIGNvbmZvcm1zIHRvIE9wZW5BUEkKCltBcHByb3ZlQVBJXShodHRwczovL2FwcHJvdmVhcGkuY29tKSBpcyBjb21wYXRpYmxlIHdpdGggdGhlIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24uIAoKVXNpbmcgW29wZW5hcGktc3BlYy0zLjAueWFtbF0ob3BlbmFwaS1zcGVjLTMuMC55YW1sKSBpbiB0aGlzIHJlcG9zaXRvcnksIHlvdSBjYW4gZ2VuZXJhdGUgY2xpZW50IGxpYnJhcnkgYmluZGluZ3MgdG8gQXBwcm92ZUFQSSBmb3IgeW91ciBmYXZvcml0ZSBsYW5ndWFnZS4K readmeEtag: '"1b1a854d3f2e72265068c99998c030758fd7d768"' readmeLastModified: Mon, 11 Mar 2019 22:09:40 GMT repositoryId: 175090530 description: 'OpenAPI Spec 3.0 for ApproveAPI ' created: '2019-03-11T22:02:12Z' updated: '2025-08-11T21:06:02Z' language: null archived: false stars: 2 watchers: 3 forks: 0 owner: approveapi logo: https://avatars.githubusercontent.com/u/46987102?v=4 repoEtag: '"bff2680295c12cfd07dc664c9882e496879272b17ff38f01ca6cd53bf51aaa40"' repoLastModified: Mon, 11 Aug 2025 21:06:02 GMT foundInMaster: true category: - Code Generators - Parsers id: e75714c89bcf4c22e6f058364f834c12 - source: openapi3 tags repository: https://github.com/acme-software/typeswagger v3: true repositoryMetadata: base64Readme: >- VHlwZVN3YWdnZXIgLSBBIFN3YWdnZXIgLyBPcGVuQXBpIFNwZWMgRFNMIGZvciBTY2FsYQo9PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT09CgpbIVtMaWNlbnNlOiBNSVRdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTGljZW5zZS1NSVQteWVsbG93LnN2ZyldKGh0dHBzOi8vb3BlbnNvdXJjZS5vcmcvbGljZW5zZXMvTUlUKSBbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9hY21lLXNvZnR3YXJlL3R5cGVzd2FnZ2VyLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2FjbWUtc29mdHdhcmUvdHlwZXN3YWdnZXIpIFshW0NvZGFjeSBCYWRnZV0oaHR0cHM6Ly9hcGkuY29kYWN5LmNvbS9wcm9qZWN0L2JhZGdlL0dyYWRlLzFhMmQwYzQ5M2JkOTQxNjk5NmE4NDM2ODcxYTE3YjAxKV0oaHR0cHM6Ly93d3cuY29kYWN5LmNvbS9hcHAvZnJuZS90eXBlc3dhZ2dlcj91dG1fc291cmNlPWdpdGh1Yi5jb20mYW1wO3V0bV9tZWRpdW09cmVmZXJyYWwmYW1wO3V0bV9jb250ZW50PWFjbWUtc29mdHdhcmUvdHlwZXN3YWdnZXImYW1wO3V0bV9jYW1wYWlnbj1CYWRnZV9HcmFkZSkKCioqVHlwZVN3YWdnZXIgcHJvdmlkZXMgYSB0eXBlc2FmZSBTY2FsYSBEU0wgdG8gYnVpbGQgT3BlbkFwaSAoU3dhZ2dlcikgU3BlY2lmaWNhdGlvbnMuIEl0IGNhbiBiZSB1c2VkIHdpdGhpbiBhbnkgU2NhbGEgCnByb2plY3QgdG8gZ2VuZXJhdGUgSFRUUCBBUEkgZG9jdW1lbnRhdGlvbnMuKioKCiMjIEV4YW1wbGUKClRoZSBmb2xsb3dpbmcgZXhhbXBsZSBkZWZpbmVzIGEgc2ltcGxlIGFwaSB3aXRoIHR3byBlbmRwb2ludHMgKGBHRVRgIGFuZCBgREVMRVRFYCkgb24gb25lIHBhdGggKGAvdXNlci97aWR9YCkuIFRoZSAKZXhhbXBsZSBjb250YWlucyBtYW5kYXRvcnkgaW5mb3JtYXRpb24sIGFuZCBhbHNvIHNvbWUsIGJ1dCBub3QgYWxsIG9wdGlvbmFsIGZpZWxkcy4gRm9yIGJldHRlciByZWFkYWJpbGl0eSwgb3B0aW9uYWwgCmZpZWxkcyBhcmUgY2FsbGQgYnkgdmFsdWUuCgpgYGBzY2FsYQppbXBvcnQgY2guYWNtZXNvZnR3YXJlLnR5cGVzd2FnZ2VyLnYzLk9wZW5BcGkKaW1wb3J0IGNoLmFjbWVzb2Z0d2FyZS50eXBlc3dhZ2dlci52My5PcGVuQXBpLl8KaW1wb3J0IGNoLmFjbWVzb2Z0d2FyZS50eXBlc3dhZ2dlci52My5JbXBsaWNpdHMuXwoKT3BlbkFwaS5jcmVhdGUoIkFwaURvYyIsICIxLjAuMCIpLgogIC8vIGFkZCBnbG9iYWwgaW5mb3JtYXRpb24KICB3aXRoSW5mbyhsaWNlbnNlID0gIk1JVCIsIGxpY2Vuc2VVcmwgPSAiaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQiKS4KICAvLyBhZGQgYSB0YWcKICB3aXRoVGFnKCJ0ZXN0dGFnIiwgIkEgdGFnIGRlc2NyaXB0aW9uIiwgZXh0ZXJuYWxEb2NzID0gZG9jKCJodHRwOi8vbGluay50by5kb2MiKSkuCiAgLy8gZGVmaW5lIGFuIGFwaSBwYXRoCiAgcGF0aCgiL3VzZXIve2lkfSIsIHN1bW1hcnkgPSAiUGF0aCBzdW1tYXJ5IikgewogICAgLy8gaHR0cCBHRVQgb3BlcmF0aW9uICBvbiB0aGUgcGF0aAogICAgKEdFVCA+PiBvcCgiQSBzdW1tYXJ5IiwgIkdFVCBvcGVyYXRpb24gZm9yIHRoaXMgcm91dGUiKS4KICAgICAgd2l0aFBhcmFtZXRlcigiaWQiLCBQQVRILCBTY2hlbWEuaW50LCBkZXNjcmlwdGlvbiA9ICJUaGUgaWQuLi4iLCByZXF1aXJlZCA9IHRydWUpLgogICAgICB3aXRoUGFyYW1ldGVyKCJjb21tZW50IiwgUVVFUlksIFNjaGVtYS5zdHJpbmcsIGRlc2NyaXB0aW9uID0gIlNvbWUgZGVwcmVjYXRlZCBjb21tZW50IiwgCiAgICAgICAgICAgICAgICAgICAgZGVwcmVjYXRlZCA9IHRydWUpLgogICAgICB3aXRoVGFnKCJ0ZXN0dGFnIikpIH4KICAgIC8vIGh0dHAgREVMRVRFIG9wZXJhdGlvbiBvbiB0aGUgcGF0aAogICAgKERFTEVURSA+PiBvcCgiRGVsZXRlIG9wZXJhdGlvbiIpLgogICAgICB3aXRoUGFyYW1ldGVyKCJpZCIsIFBBVEgsIFNjaGVtYS5pbnQsIGRlc2NyaXB0aW9uID0gIlRoZSBpZC4uLiIsIHJlcXVpcmVkID0gdHJ1ZSkpCiAgfS4KICAvLyBhZGQgZW5kcG9pbnQgc2VydmVyKHMpCiAgd2l0aFNlcnZlcigiaHR0cDovL2xvY2FsaG9zdDo5MDAwL2FwaSIpLgogIHdpdGhTZXJ2ZXIoImh0dHBzOi8vcHJvZHVjdGlvbi50bGQvYXBpIiwgU29tZSgiUHJvZHVjdGlvbiBTZXJ2ZXIiKSkuCiAgLy8gYnVpbGQganNvbiBzdHJpbmcKICB0b0pzb24oKQpgYGAKCiMjIFVzYWdlCgpJbnN0YWxsIHRoZSBJdnkgZGVwZW5lbmN5IHZpYSBTQlQ6CgpgYGBzY2FsYQoiY2guYWNtZXNvZnR3YXJlIiAlJSAidHlwZXN3YWdnZXIiICUgInt2ZXJzaW9ufSIKYGBgCgpBZGQgaW1wb3J0czoKCmBgYHNjYWxhCmltcG9ydCBjaC5hY21lc29mdHdhcmUudHlwZXN3YWdnZXIudjMuT3BlbkFwaQppbXBvcnQgY2guYWNtZXNvZnR3YXJlLnR5cGVzd2FnZ2VyLnYzLk9wZW5BcGkuXwoKLy8gc3ludGFjdGljIHN1Z2FyIGZvciBjb252ZW5pZW50IERTTCwgYnV0IG9wdGlvbmFsCmltcG9ydCBjaC5hY21lc29mdHdhcmUudHlwZXN3YWdnZXIudjMuSW1wbGljaXRzLl8gCmBgYAoKIyMjIEluZm8KCkFkZCBnbG9iYWwgaW5mb3JtYXRpb24gdXNpbmcgdGhlIGB3aXRoSW5mb2AgZnVuY3Rpb24uIEFsbCBwYXJhbWV0ZXJzIGFyZSBvcHRpb25hbC4gSWYgbm90IHVzaW5nIHRoZSBpbXBsaWNpdHMsIAp1c2UgYFNvbWUoInN0ciIpYCBhcyBwYXJhbWV0ZXIgdmFsdWU6CgpgYGBzY2FsYQppbXBvcnQgY2guYWNtZXNvZnR3YXJlLnR5cGVzd2FnZ2VyLnYzLk9wZW5BcGkKaW1wb3J0IGNoLmFjbWVzb2Z0d2FyZS50eXBlc3dhZ2dlci52My5PcGVuQXBpLl8KaW1wb3J0IGNoLmFjbWVzb2Z0d2FyZS50eXBlc3dhZ2dlci52My5JbXBsaWNpdHMuXwoKT3BlbkFwaS5jcmVhdGUoIkFwaURvYyIsICIxLjAuMCIsIGRlc2NyaXB0aW9uID0gIkFQSSBEb2MgYnVpbHQgd2l0aCBUeXBlU3dhZ2dlciIsIHRlcm1zT2ZTZXJ2aWNlID0gIi90b3MiKS4KICB3aXRoSW5mbyhsaWNlbnNlID0gIk1JVCIsIGxpY2Vuc2VVcmwgPSAiaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQiLCAKICAgICAgICAgICBjb250YWN0TmFtZSA9ICJKb2huIERvZSIsIGNvbnRhY3RFbWFpbCA9ICJpbmZvQHNlcnZlci50bGQiLCBjb250YWN0VXJsID0gImh0dHBzOi8vd2VicGFnZS50bGQiKQpgYGAKCiMjIEZlYXR1cmVzCgoqIENvbnZlbmllbnQsIHR5cGVzYWZlIERTTCB0byBkZWZpbmUgT3BlbkFwaSB2MyBzcGVjaWZpY2F0aW9ucwoqIEludGVncmF0ZWQgSlNPTiByZW5kZXJpbmcKKiBJbnRlZ3JhdGVkIFlBTUwgcmVuZGVyaW5nIChUT0RPKQoqIEphdmEgQVBJIChUT0RPKQoqIEludGVncmF0aW9uIHdpdGggY29tbW9uIEhUVFAtRnJhbWV3b3JrcyAoVE9ETykKCiMjIENvbnRyaWJ1dGlvbnMKClBsZWFzZSB1c2UgdGhlIEdpdEh1YiBpc3N1ZSB0cmFja2luZyBhbmQgUHVsbFJlcXVlc3RzLiBBbnkgaGVscCBpcyB2ZXJ5IHdlbGNvbWUuCgojIyBMaWNlbnNlCgpUaGlzIHByb2plY3QgaXMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBsaWNlbnNlLiBTZWUgW0xJQ0VOU0VdKC4vTElDRU5TRSkgZm9yIG1vcmUgaW5mb3JtYXRpb24u readmeEtag: '"121f2535582736884d12541bef0198f1145d9ce6"' readmeLastModified: Mon, 25 Jun 2018 11:26:05 GMT repositoryId: 138474648 description: A Swagger / OpenApi Specification DSL for Scala created: '2018-06-24T11:00:44Z' updated: '2018-06-25T11:54:52Z' language: Scala archived: false stars: 1 watchers: 1 forks: 0 owner: acme-software logo: https://avatars.githubusercontent.com/u/19473705?v=4 license: MIT repoEtag: '"08dc81aed3f44f2df745f3d8f70e4e08ea4fb5819e0c22442d51f7f900064f48"' repoLastModified: Mon, 25 Jun 2018 11:54:52 GMT foundInMaster: true category: Parsers id: f9bdc713a1ff13ba55fa7ea37d38ab44 - source: openapi3 tags repository: https://github.com/svilgelm/oas3-server v3: true repositoryMetadata: base64Readme: >- IyBvYXMzLXNlcnZlcgpbIVtQcm9qZWN0IFN0YXR1czogV0lQXShodHRwczovL3d3dy5yZXBvc3RhdHVzLm9yZy9iYWRnZXMvbGF0ZXN0L3dpcC5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vU1ZpbGdlbG0vb2FzMy1zZXJ2ZXIpClshW1Rlc3RzIHN0YXR1c10oaHR0cHM6Ly9naXRodWIuY29tL1NWaWxnZWxtL29hczMtc2VydmVyL3dvcmtmbG93cy9UZXN0cy9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vU1ZpbGdlbG0vb2FzMy1zZXJ2ZXIvYWN0aW9ucz9xdWVyeT1icmFuY2glM0FtYXN0ZXIrZXZlbnQlM0FwdXNoKQpbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1TVmlsZ2VsbV9vYXMzLXNlcnZlciZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1TVmlsZ2VsbV9vYXMzLXNlcnZlcikKCk9wZW5BUEkgMyBXZWIgU2VydmVyCg== readmeEtag: '"6aaa787397d0a88089ab74dc6d5f80a82e3e4c72"' readmeLastModified: Mon, 23 Dec 2019 16:02:37 GMT repositoryId: 222094495 description: OpenAPI 3 Server created: '2019-11-16T12:15:06Z' updated: '2025-07-24T18:28:33Z' language: Go archived: true stars: 1 watchers: 1 forks: 1 owner: SVilgelm logo: https://avatars.githubusercontent.com/u/523825?v=4 license: MIT repoEtag: '"40b74d6f6ce13b2139ded105be60336cfd87d3f648100b6d58ec75f3f8eb5b02"' repoLastModified: Thu, 24 Jul 2025 18:28:33 GMT foundInMaster: true category: Server id: 23e7038efa6a7dcdc3fdaa7d1cac63a4 - source: openapi3 tags repository: https://github.com/philenius/dockerized-dummy-web-services v3: true repositoryMetadata: base64Readme: >- IyBEdW1teSBXZWIgU2VydmljZXMKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBkdW1teSB3ZWIgc2VydmljZXMgd2hpY2ggYXJlIGNvbnRhaW5lcml6ZWQgKERvY2tlcikgYW5kIHJlYWR5IHRvIGJlIGRlcGxveWVkIG9uIEt1YmVybmV0ZXMgb3IgT3BlblNoaWZ0IC8gT0tELgoKIyMgR29sYW5nClRoZSBkaXJlY3RvcnkgYGdvbGFuZy1kdW1teS13ZWItc2VydmljZS9gIGNvbnRhaW5zIGEgZHVtbXkgd2ViIHNlcnZpY2UgaW1wbGVtZW50ZWQgaW4gR29sYW5nLgpUaGlzIHdlYiBzZXJ2aWNlIGxpc3RlbnMgcGVyIGRlZmF1bHQgb24gcG9ydCA4MDgxLiBUaGUgcG9ydCBjYW4gYmUgY29uZmlndXJlZCB0aHJvdWdoIHRoZSBlbnZpcm9ubWVudCB2YXJpYWJsZSBgU0VSVkVSX1BPUlRgLiBUaGUgSFRUUCByZXNwb25zZSBvZiB0aGlzIGR1bW15IGFwcCBjb250YWlucyB0aGUgdmFsdWUgb2YgdGhlIGVudmlyb25tZW50IHZhcmlhYmxlIGBNWV9FTlZfVkFSYC4KCioqSFRUUCBFbmRwb2ludHM6KioKCiogYEdFVCAvYAoqIGBHRVQgL2hlYWx0aGAKKiBgR0VUIC9zaHV0ZG93bmAKCioqRG9ja2VyIGltYWdlOioqCmBwaGlsZW5pdXMvZ29sYW5nLWR1bW15LXdlYi1zZXJ2aWNlYAoKIyMgSmF2YSAvIFNwcmluZyBCb290CgpUaGUgZGlyZWN0b3J5IGBqYXZhLXNwcmluZy1ib290LWR1bW15LXdlYi1zZXJ2aWNlL2AgY29udGFpbnMgYSBkdW1teSBpbXBsZW1lbnRhdGlvbiBvZiBQZXRzdG9yZSBPcGVuQVBJIDMuMCBzcGVjIHVzaW5nIFNwcmluZyBCb290LiBUaGUgc2VydmVyIHN0dWIgd2FzIGdlbmVyYXRlZCBieSB0aGUgW3N3YWdnZXItY29kZWdlbl0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29kZWdlbikgcHJvamVjdC4gVGhlIHVuZGVybHlpbmcgbGlicmFyeSBpbnRlZ3JhdGluZyBzd2FnZ2VyIHRvIFNwcmluZ0Jvb3QgaXMgW3NwcmluZ2ZveF0oaHR0cHM6Ly9naXRodWIuY29tL3NwcmluZ2ZveC9zcHJpbmdmb3gpLgpUaGlzIHdlYiBzZXJ2aWNlIGxpc3RlbnMgcGVyIGRlZmF1bHQgb24gcG9ydCA4MDgwLiBUaGUgcG9ydCBjYW4gYmUgY29uZmlndXJlZCB0aHJvdWdoIHRoZSBlbnZpcm9ubWVudCB2YXJpYWJsZSBgU0VSVkVSX1BPUlRgLgoKKipIVFRQIEVuZHBvaW50czoqKgoqIGBHRVQgL3YxL3N3YWdnZXItdWkuaHRtbGAKKiBgR0VUIC92MS9wZXRzYAoqIGBHRVQgL3YxL3BldHMve3BldElkfWAKKiBgUE9TVCAvdjEvcGV0c2AKKiBgR0VUIC9oZWFsdGhgCiogYEdFVCAvc2h1dGRvd25gClNlZSBTd2FnZ2VyIFVJIG9yIEFQSSBzcGVjIGZvciBmdXJ0aGVyIGRldGFpbHMuCgoqKkRvY2tlciBpbWFnZToqKgpgcGhpbGVuaXVzL3BldHN0b3JlLWR1bW15LXNwcmluZy1ib290YAoKIyMgTm9kZSAvIEV4cHJlc3NKUwoKVGhlIGRpcmVjdG9yeSBgbm9kZS1leHByZXNzanMtZHVtbXktd2ViLWFwcGAgY29udGFpbnMgYSBkdW1teSB3ZWIgYXBwIHRoYXQgcHJvdmlkZXMgYSB3ZWJzaXRlIHdpdGggbWVtZXMuIFRoZSBzZXJ2ZXIgd2FzIGdlbmVyYXRlZCB1c2luZyB0aGUgTlBNIHBhY2thZ2UgYGV4cHJlc3MtZ2VuZXJhdG9yYC4KVGhpcyB3ZWIgYXBwIGxpc3RlbnMgcGVyIGRlZmF1bHQgb24gcG9ydCAzMDAwLiBUaGUgcG9ydCBjYW4gYmUgY29uZmlndXJlZCB0aHJvdWdoIHRoZSBlbnZpcm9ubWVudCB2YXJpYWJsZSBgU0VSVkVSX1BPUlRgLiBUaGlzIGFwcCBjYW4gZWl0aGVyIHNlcnZlIGNhdCBvciBkb2cgbWVtZXMuIFRoaXMgYmVoYXZpb3VyIGNhbiBiZSBjb25maWd1cmVkIHRocm91Z2ggc2V0dGluZyB0aGUgZW52aXJvbm1lbnQgdmFyaWFibGUgYE1FTUVCT09LX1ZBUklBTlRgIHRvIHRoZSB2YWx1ZSBgZG9nc2AgcmVzcGVjdGl2ZWx5IGBjYXRzYC4KCioqSFRUUCBFbmRwb2ludHM6KioKKiBgR0VUIC9gCiogYEdFVCAvaGVhbHRoYAoqIGBHRVQgL3NodXRkb3duYAoKKipEb2NrZXIgaW1hZ2U6KioKYHBoaWxlbml1cy9tZW1lYm9vay1kdW1teS1ub2RlLWV4cHJlc3Nqc2AK readmeEtag: '"b20debe5b13b8a8086471490bb79615cff6835eb"' readmeLastModified: Tue, 17 Dec 2019 02:36:33 GMT repositoryId: 184478754 description: Dockerized dummy web services created: '2019-05-01T20:40:53Z' updated: '2023-03-28T09:44:07Z' language: Java archived: false stars: 1 watchers: 0 forks: 0 owner: philenius logo: https://avatars.githubusercontent.com/u/9049899?v=4 repoEtag: '"8fbddc4daf35ae1e1cba8f1bb0192755db7ced6e9f261cbabb00d5058cb4c805"' repoLastModified: Tue, 28 Mar 2023 09:44:07 GMT foundInMaster: true category: - Server - Server Implementations id: a9c2f7e1f38d750810d3964ab3d26cf1 - source: openapi3 tags repository: https://github.com/iulian-stan/crud-nestjs-rest v3: true id: 7113451d8675753deedca4ebbcce2434 repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cDovL25lc3Rqcy5jb20vIiB0YXJnZXQ9ImJsYW5rIj48aW1nIHNyYz0iaHR0cHM6Ly9uZXN0anMuY29tL2ltZy9sb2dvLXNtYWxsLnN2ZyIgd2lkdGg9IjEyMCIgYWx0PSJOZXN0IExvZ28iIC8+PC9hPgo8L3A+Cgo8cCBhbGlnbj0iY2VudGVyIj5BIHByb2dyZXNzaXZlIDxhIGhyZWY9Imh0dHA6Ly9ub2RlanMub3JnIiB0YXJnZXQ9ImJsYW5rIj5Ob2RlLmpzPC9hPiBmcmFtZXdvcmsgZm9yIGJ1aWxkaW5nIGVmZmljaWVudCBhbmQgc2NhbGFibGUgc2VydmVyLXNpZGUgYXBwbGljYXRpb25zLjwvcD4KCiMjIERlc2NyaXB0aW9uCkNSVUQgYXBwbGljYXRpb24gdXNpbmcgTmVzdEpTIGZyYW1ld29yayBUeXBlT1JNIGFuZCBTd2FnZ2VyLgoKIyMgSW5zdGFsbGF0aW9uCkluc3RhbGwgb2YgdGhlIGRlcGVuZGVuY2llcyBmcm9tICpwYWNrYWdlLmpzb24qCmBgYGJhc2gKbnBtIGluc3RhbGwKYGBgCgojIyBVc2FnZQpTdGFydCB0aGUgYXBwbGljYXRpb24gYnkgcnVubmluZwpgYGBiYXNoCm5wbSBzdGFydApgYGAKTmF2aWdhdGUgdG8gdGhlIGZvbGxvd2luZyBVUkwgaW4geW91ciB3ZWIgYnJvd3NlciAtIFtodHRwOi8vMTI3LjAuMC4xOjQwMDAvYXBpL10oaHR0cDovLzEyNy4wLjAuMTo0MDAwL2FwaS8pLg== readmeEtag: '"7a01a34414bcf277922e242de18c7f69b556c47e"' readmeLastModified: Sun, 18 Dec 2022 00:04:53 GMT repositoryId: 242532299 description: Simple CRUD application implemented using NestJS framework created: '2020-02-23T14:33:05Z' updated: '2022-12-08T13:32:24Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: iulian-stan logo: https://avatars.githubusercontent.com/u/13657730?v=4 license: MIT repoEtag: '"ddc118c067028f1d1cb294ca2fbeaf64f5f0622ae8d8a0ae1d2b86a681a04046"' repoLastModified: Thu, 08 Dec 2022 13:32:24 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/grandlinex/docs-to-openapi v3: true repositoryMetadata: base64Readme: >- IyBEb2NzIHRvIE9wZW5BcGkgdi4zCgojIyBEb2N1bWVudGF0aW9uCi0gW0RvY3NdKGh0dHBzOi8vZ3JhbmRsaW5leC5naXRodWIuaW8vZG9jcy91dGlscy8jZG9jcy10by1vcGVuYXBpLXYzKQoKClshW0dpdEh1Yl0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2doL0dyYW5kbGluZVglMkZkb2NzLXRvLW9wZW5hcGkuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL0dyYW5kbGluZVgvZG9jcy10by1vcGVuYXBpKQpbIVtOUE1dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vc3RhdGljL3YxP2xhYmVsPU5QTSZtZXNzYWdlPVBhY2thZ2UmY29sb3I9cmVkJmxvZ289TlBNKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQGdyYW5kbGluZXgvZG9jcy10by1vcGVuYXBpKQohW1RTXShodHRwczovL2ltZy5zaGllbGRzLmlvL3N0YXRpYy92MT9sYWJlbD1MYW5ndWFnZSZtZXNzYWdlPVR5cGVTY3JpcHQmY29sb3I9Ymx1ZSZsb2dvPVR5cGVTY3JpcHQpCgoKIyMgRGVzY3JpcHRpb24KClRoaXMgUHJvamVjdCBjb252ZXJ0cyBjb21tZW50cyBpbiB5b3VyIGNvZGUgb3IgaW4geW91ciBkZXBlbmRlbmNpZXMgdG8gYW4gT3BlbkFwaSBTcGVjLgpUaGUgZm9sbG93aW5nIGFubm90YXRpb25zIGFyZSBzdXBwb3J0ZWQuIAotIGBAb3BlbmFwaWAgCi0gYEBzd2FnZ2VyYAoKQWxsIGNvbW1lbnRzIHdpdGggdGhlc2UgYW5ub3RhdGlvbnMgd2lsbCBiZSBwYXN0ZWQgYXMgYSBzaW5nZSBlbGVtZW50IGluIHRoZSBwYXRoIGRlZmluaXRpb24gYXJyYXkuCkZvciBhbGwgb3RoZXIgT3BlbkFwaSBjb25maWdzIHRoZSBgZ2xjb25mLmpzb25gIGlzIHVzZWQuICAgCgojIyBJbnN0YWxsYXRpb24KCi0gYG5wbSBpIC1nIEBncmFuZGxpbmV4L2RvY3MtdG8tb3BlbmFwaWAgCgojIyMgQ29uZmlnIGZpZWxkcwoKLSByb290RGlyIAogIC0gUmVsYXRpdmUgcGF0aCB0byB5b3VyIHNvdXJjZWNvZGUuCi0gZmlsZXR5cGVzIAogIC0gIExpc3Qgb2YgZmlsZSBleHRlbnNpb25zIHRvIG1vbml0b3IuCi0gZXh0ZXJuYWxNb2R1bGVzCiAgLSBJZiB5b3VyIGRlcGVuZGVuY2llcyB1c2UgYWxzbyB0aGVzZSBhbm5vdGF0aW9ucyB5b3UgY2FuIG1lcmdlIHRoZXNlIGluIHlvdXIgY3VycmVuIHNwZWMuCi0gb3V0UHV0RGlyIChvcHRpb25hbCkKICAtIFJlbGF0aXZlIHBhdGggZm9yIHRoZSBTcGVjLiBvdXRwdXQuICAKCiMjIFF1aWNrIFN0YXJ0CgoxLiBDcmVhdGUgYSBgZ2xjb25mLmpzb25gIGluIHRoZSByb290IGRpciBvZiB5b3VyIHByb2plY3QgKHNlZSBjb25maWcgMDEgb3IgMDIpLgoyLiBFZGl0IHlvdXIgYmFzZSBDb25maWcgZmlsZSBsaWtlIGluIHRoZSBvZmZpY2lhbCBbU3dhZ2dlciBEb2NdKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykKMy4gQWRkIGEgQ29tbWVudCB0byB5b3VyIHNvcnVjZSBjb2RlIGluIHltbCBzeW50YXggbGlrZTogCgogICAgICAgICAgICAvKioKICAgICAgICAgICAgICogQG9wZW5hcGkKICAgICAgICAgICAgICogL3Bpbmc6CiAgICAgICAgICAgICAqICAgZ2V0OgogICAgICAgICAgICAgKiAgICAgc3VtbWFyeTogcGluZwogICAgICAgICAgICAgKiAgICAgdGFnczoKICAgICAgICAgICAgICogICAgICAgLSBiYXNpYwogICAgICAgICAgICAgKiAgICAgcmVzcG9uc2VzOgogICAgICAgICAgICAgKiAgICAgICAyMDA6CiAgICAgICAgICAgICAqICAgICAgICAgZGVzY3JpcHRpb246IE9LCiAgICAgICAgICAgICAqICAgICAgIDQwMToKICAgICAgICAgICAgICogICAgICAgICBkZXNjcmlwdGlvbjogbm90IGF1dGhvcml6ZWQKICAgICAgICAgICAgICovCgo0LiBydW4gYG5weCBAZ3JhbmRsaW5leC9kb2NzLXRvLW9wZW5hcGlgCgo+IElmIHlvdSBpbnN0YWxsIHRoZSBwYWNrYWdlIGFzIGEgZGVwZW5kZW5jeSBvciBnbG9iYWwgdGhlbiB5b3UgY2FuIHVzZSB0aGUgYG1ha2VPcGVuQXBpYCBjb21tYW5kLiAKCiMjIENMSSBQYXJhbWV0ZXIKCnxQYXJhbWV0ZXJ8RGVzY3JpcHRpb258CnwtLS18LS0tfAp8YC0taHRtbGB8Q3JlYXRlcyBhIHN3YWdnZXIuaHRtbCBmaWxlIChTdGFuZGFsb25lIFN3YWdnZXJVSSl8CnxgLS1zZXJ2ZWB8U2VydmVzIHRoZSBPcGVuQXBpIFNwZWMgb24gcG9ydCA5MDAwfAoKCiMjIENvbmZpZyBGaWxlIEV4YW1wbGVzOgoKIyMjIENvbmZpZyAwMSBgZ2xjb25mLmpzb25gIChtaW5pbWFsKQoKCiAgICAgIHsKICAgICAgICAicm9vdERpciI6ICIuL3NyYyIsCiAgICAgICAgImZpbGV0eXBlcyI6IFsidHMiLCJqcyJdLAogICAgICAgICJleHRlcm5hbE1vZHVsZXMiOiBbXSwKICAgICAgICAiYmFzZUNvbmZpZyI6IHsKICAgICAgICAgICJpbmZvIjogewogICAgICAgICAgICAidGl0bGUiOiAiSGVsbG8gV29ybGQiLAogICAgICAgICAgICAidmVyc2lvbiI6ICIxLjAuMCIsCiAgICAgICAgICAgICJkZXNjcmlwdGlvbiI6ICJBIHNhbXBsZSBBUEkiCiAgICAgICAgICB9LAogICAgICAgICAgInNlcnZlcnMiOiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAidXJsIjogImh0dHA6Ly9sb2NhbGhvc3Q6OTI1NyIKICAgICAgICAgICAgfQogICAgICAgICAgXQogICAgICAgIH0KICAgICAgfQoKIyMjIENvbmZpZyAwMiBgZ2xjb25mLmpzb25gIChCZWFyZXIgSldUKQoKICAgICAgewogICAgICAgICJyb290RGlyIjogIi4vc3JjIiwKICAgICAgICAiZmlsZXR5cGVzIjogWyJ0cyIsImpzIl0sCiAgICAgICAgImV4dGVybmFsTW9kdWxlcyI6IFsiLi9ub2RlX21vZHVsZXMvbW9kdWxlX2EiLCIuL25vZGVfbW9kdWxlcy9tb2R1bGVfYiJdLAogICAgICAgICJvdXRQdXREaXIiOiAiLi9kaXN0IiwKICAgICAgICAiYmFzZUNvbmZpZyI6IHsKICAgICAgICAgICJpbmZvIjogewogICAgICAgICAgICAidGl0bGUiOiAiSGVsbG8gV29ybGQiLAogICAgICAgICAgICAidmVyc2lvbiI6ICIxLjAuMCIsCiAgICAgICAgICAgICJkZXNjcmlwdGlvbiI6ICJBIHNhbXBsZSBBUEkiCiAgICAgICAgICB9LAogICAgICAgICAgInNlcnZlcnMiOiBbCiAgICAgICAgICAgIHsKICAgICAgICAgICAgICAidXJsIjogImh0dHA6Ly9sb2NhbGhvc3Q6OTI1NyIKICAgICAgICAgICAgfQogICAgICAgICAgXSwKICAgICAgICAgICJzZWN1cml0eSI6IFsKICAgICAgICAgICAgewogICAgICAgICAgICAgICJiZWFyZXJBdXRoIjogW10KICAgICAgICAgICAgfQogICAgICAgICAgXSwKICAgICAgICAgICJjb21wb25lbnRzIjogewogICAgICAgICAgICAic2VjdXJpdHlTY2hlbWVzIjogewogICAgICAgICAgICAgICJiZWFyZXJBdXRoIjogewogICAgICAgICAgICAgICAgInR5cGUiOiAiaHR0cCIsCiAgICAgICAgICAgICAgICAic2NoZW1lIjogImJlYXJlciIsCiAgICAgICAgICAgICAgICAiYmVhcmVyRm9ybWF0IjogIkpXVCIKICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgIH0KICAgICAgICB9CiAgICAgIH0K readmeEtag: '"c08d42c954d8a7b77d88e45eed7b7f6b77455ca2"' readmeLastModified: Thu, 16 Sep 2021 19:39:07 GMT repositoryId: 401494338 description: >- This Project converts comments in your code or in your dependencies to an OpenApi Spec. created: '2021-08-30T21:45:07Z' updated: '2023-06-17T23:58:13Z' language: TypeScript archived: true stars: 1 watchers: 0 forks: 0 owner: GrandlineX logo: https://avatars.githubusercontent.com/u/89772740?v=4 license: BSD-3-Clause repoEtag: '"dd373e1e929d94e3d9b4a43286f9a7a0ca38ccbfcec736f1b2a90ed3fcd564c1"' repoLastModified: Sat, 17 Jun 2023 23:58:13 GMT foundInMaster: true category: Parsers id: 20d5f81e29cc0e716934c1a7c7bcd728 - source: openapi3 tags repository: https://github.com/namsor/namsor-golang-sdk2 v3: true repositoryMetadata: base64Readme: >- # Go API client for namsorapi

NamSor API v2 : enpoints to process personal names (gender, cultural origin or ethnicity) in all alphabets or languages. By default, enpoints use 1 unit per name (ex. Gender), but Ethnicity classification uses 10 to 20 units per name depending on taxonomy. Use GET methods for small tests, but prefer POST methods for higher throughput (batch processing of up to 100 names at a time). Need something you can't find here? We have many more features coming soon. Let us know, we'll do our best to add it! 

## Overview
This API client was generated by the [OpenAPI Generator](https://openapi-generator.tech) project.  By using the [OpenAPI-spec](https://www.openapis.org/) from a remote server, you can easily generate an API client.

- API version: 2.0.15
- Package version: 2.0.15
- Build package: org.openapitools.codegen.languages.GoClientCodegen
For more information, please visit [http://www.namsor.com/](http://www.namsor.com/)

## Installation

Install the following dependencies:
```
go get github.com/stretchr/testify/assert
go get golang.org/x/oauth2
go get golang.org/x/net/context
go get github.com/antihax/optional
```

Put the package under your project folder and add the following in import:
```golang
import "./namsorapi"
```

## Documentation for API Endpoints

All URIs are relative to *https://v2.namsor.com/NamSorAPIv2*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AdminApi* | [**Anonymize**](docs/AdminApi.md#anonymize) | **Get** /api2/json/anonymize/{source}/{anonymized} | Activate/deactivate anonymization for a source.
*AdminApi* | [**ApiStatus**](docs/AdminApi.md#apistatus) | **Get** /api2/json/apiStatus | Prints the current status of the classifiers. A classifier name in apiStatus corresponds to a service name in apiServices.
*AdminApi* | [**ApiUsage**](docs/AdminApi.md#apiusage) | **Get** /api2/json/apiUsage | Print current API usage.
*AdminApi* | [**ApiUsageHistory**](docs/AdminApi.md#apiusagehistory) | **Get** /api2/json/apiUsageHistory | Print historical API usage.
*AdminApi* | [**ApiUsageHistoryAggregate**](docs/AdminApi.md#apiusagehistoryaggregate) | **Get** /api2/json/apiUsageHistoryAggregate | Print historical API usage (in an aggregated view, by service, by day/hour/min).
*AdminApi* | [**AvailableServices**](docs/AdminApi.md#availableservices) | **Get** /api2/json/apiServices | List of classification services and usage cost in Units per classification (default is 1&#x3D;ONE Unit). Some API endpoints (ex. Corridor) combine multiple classifiers.
*AdminApi* | [**Disable**](docs/AdminApi.md#disable) | **Get** /api2/json/disable/{source}/{disabled} | Activate/deactivate an API Key.
*AdminApi* | [**Learnable**](docs/AdminApi.md#learnable) | **Get** /api2/json/learnable/{source}/{learnable} | Activate/deactivate learning from a source.
*AdminApi* | [**SoftwareVersion**](docs/AdminApi.md#softwareversion) | **Get** /api2/json/softwareVersion | Get the current software version
*AdminApi* | [**TaxonomyClasses**](docs/AdminApi.md#taxonomyclasses) | **Get** /api2/json/taxonomyClasses/{classifierName} | Print the taxonomy classes valid for the given classifier.
*ChineseApi* | [**ChineseNameCandidates**](docs/ChineseApi.md#chinesenamecandidates) | **Get** /api2/json/chineseNameCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming
*ChineseApi* | [**ChineseNameCandidatesBatch**](docs/ChineseApi.md#chinesenamecandidatesbatch) | **Post** /api2/json/chineseNameCandidatesBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**ChineseNameCandidatesGenderBatch**](docs/ChineseApi.md#chinesenamecandidatesgenderbatch) | **Post** /api2/json/chineseNameCandidatesGenderBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname) ex. Wang Xiaoming.
*ChineseApi* | [**ChineseNameGenderCandidates**](docs/ChineseApi.md#chinesenamegendercandidates) | **Get** /api2/json/chineseNameGenderCandidates/{chineseSurnameLatin}/{chineseGivenNameLatin}/{knownGender} | Identify Chinese name candidates, based on the romanized name ex. Wang Xiaoming - having a known gender (&#39;male&#39; or &#39;female&#39;)
*ChineseApi* | [**ChineseNameMatch**](docs/ChineseApi.md#chinesenamematch) | **Get** /api2/json/chineseNameMatch/{chineseSurnameLatin}/{chineseGivenNameLatin}/{chineseName} | Return a score for matching Chinese name ex. 王晓明 with a romanized name ex. Wang Xiaoming
*ChineseApi* | [**ChineseNameMatchBatch**](docs/ChineseApi.md#chinesenamematchbatch) | **Post** /api2/json/chineseNameMatchBatch | Identify Chinese name candidates, based on the romanized name (firstName &#x3D; chineseGivenName; lastName&#x3D;chineseSurname), ex. Wang Xiaoming
*ChineseApi* | [**GenderChineseName**](docs/ChineseApi.md#genderchinesename) | **Get** /api2/json/genderChineseName/{chineseName} | Infer the likely gender of a Chinese full name ex. 王晓明
*ChineseApi* | [**GenderChineseNameBatch**](docs/ChineseApi.md#genderchinesenamebatch) | **Post** /api2/json/genderChineseNameBatch | Infer the likely gender of up to 100 full names ex. 王晓明
*ChineseApi* | [**GenderChineseNamePinyin**](docs/ChineseApi.md#genderchinesenamepinyin) | **Get** /api2/json/genderChineseNamePinyin/{chineseSurnameLatin}/{chineseGivenNameLatin} | Infer the likely gender of a Chinese name in LATIN (Pinyin).
*ChineseApi* | [**GenderChineseNamePinyinBatch**](docs/ChineseApi.md#genderchinesenamepinyinbatch) | **Post** /api2/json/genderChineseNamePinyinBatch | Infer the likely gender of up to 100 Chinese names in LATIN (Pinyin).
*ChineseApi* | [**ParseChineseName**](docs/ChineseApi.md#parsechinesename) | **Get** /api2/json/parseChineseName/{chineseName} | Infer the likely first/last name structure of a name, ex. 王晓明 -&gt; 王(surname) 晓明(given name)
*ChineseApi* | [**ParseChineseNameBatch**](docs/ChineseApi.md#parsechinesenamebatch) | **Post** /api2/json/parseChineseNameBatch | Infer the likely first/last name structure of a name, ex. 王晓明 -&gt; 王(surname) 晓明(given name).
*ChineseApi* | [**PinyinChineseName**](docs/ChineseApi.md#pinyinchinesename) | **Get** /api2/json/pinyinChineseName/{chineseName} | Romanize the Chinese name to Pinyin, ex. 王晓明 -&gt; Wang (surname) Xiaoming (given name)
*ChineseApi* | [**PinyinChineseNameBatch**](docs/ChineseApi.md#pinyinchinesenamebatch) | **Post** /api2/json/pinyinChineseNameBatch | Romanize a list of Chinese name to Pinyin, ex. 王晓明 -&gt; Wang (surname) Xiaoming (given name).
*GeneralApi* | [**NameType**](docs/GeneralApi.md#nametype) | **Get** /api2/json/nameType/{properNoun} | Infer the likely type of a proper noun (personal name, brand name, place name etc.)
*GeneralApi* | [**NameTypeBatch**](docs/GeneralApi.md#nametypebatch) | **Post** /api2/json/nameTypeBatch | Infer the likely common type of up to 100 proper nouns (personal name, brand name, place name etc.)
*GeneralApi* | [**NameTypeGeo**](docs/GeneralApi.md#nametypegeo) | **Get** /api2/json/nameTypeGeo/{properNoun}/{countryIso2} | Infer the likely type of a proper noun (personal name, brand name, place name etc.)
*GeneralApi* | [**NameTypeGeoBatch**](docs/GeneralApi.md#nametypegeobatch) | **Post** /api2/json/nameTypeGeoBatch | Infer the likely common type of up to 100 proper nouns (personal name, brand name, place name etc.)
*JapaneseApi* | [**GenderJapaneseNameFull**](docs/JapaneseApi.md#genderjapanesenamefull) | **Get** /api2/json/genderJapaneseNameFull/{japaneseName} | Infer the likely gender of a Japanese full name ex. 王晓明
*JapaneseApi* | [**GenderJapaneseNameFullBatch**](docs/JapaneseApi.md#genderjapanesenamefullbatch) | **Post** /api2/json/genderJapaneseNameFullBatch | Infer the likely gender of up to 100 full names
*JapaneseApi* | [**GenderJapaneseNamePinyin**](docs/JapaneseApi.md#genderjapanesenamepinyin) | **Get** /api2/json/genderJapaneseName/{japaneseSurname}/{japaneseGivenName} | Infer the likely gender of a Japanese name in LATIN (Pinyin).
*JapaneseApi* | [**GenderJapaneseNamePinyinBatch**](docs/JapaneseApi.md#genderjapanesenamepinyinbatch) | **Post** /api2/json/genderJapaneseNameBatch | Infer the likely gender of up to 100 Japanese names in LATIN (Pinyin).
*JapaneseApi* | [**JapaneseNameGenderKanjiCandidatesBatch**](docs/JapaneseApi.md#japanesenamegenderkanjicandidatesbatch) | **Post** /api2/json/japaneseNameGenderKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName &#x3D; japaneseGivenName; lastName&#x3D;japaneseSurname) with KNOWN gender, ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameKanjiCandidates**](docs/JapaneseApi.md#japanesenamekanjicandidates) | **Get** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{knownGender} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae - and a known gender.
*JapaneseApi* | [**JapaneseNameKanjiCandidates1**](docs/JapaneseApi.md#japanesenamekanjicandidates1) | **Get** /api2/json/japaneseNameKanjiCandidates/{japaneseSurnameLatin}/{japaneseGivenNameLatin} | Identify japanese name candidates in KANJI, based on the romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameKanjiCandidatesBatch**](docs/JapaneseApi.md#japanesenamekanjicandidatesbatch) | **Post** /api2/json/japaneseNameKanjiCandidatesBatch | Identify japanese name candidates in KANJI, based on the romanized name (firstName &#x3D; japaneseGivenName; lastName&#x3D;japaneseSurname), ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameLatinCandidates**](docs/JapaneseApi.md#japanesenamelatincandidates) | **Get** /api2/json/japaneseNameLatinCandidates/{japaneseSurnameKanji}/{japaneseGivenNameKanji} | Romanize japanese name, based on the name in Kanji.
*JapaneseApi* | [**JapaneseNameLatinCandidatesBatch**](docs/JapaneseApi.md#japanesenamelatincandidatesbatch) | **Post** /api2/json/japaneseNameLatinCandidatesBatch | Romanize japanese names, based on the name in KANJI
*JapaneseApi* | [**JapaneseNameMatch**](docs/JapaneseApi.md#japanesenamematch) | **Get** /api2/json/japaneseNameMatch/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | Return a score for matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameMatchBatch**](docs/JapaneseApi.md#japanesenamematchbatch) | **Post** /api2/json/japaneseNameMatchBatch | Return a score for matching a list of Japanese names in KANJI ex. 山本 早苗 with romanized names ex. Yamamoto Sanae
*JapaneseApi* | [**JapaneseNameMatchFeedbackLoop**](docs/JapaneseApi.md#japanesenamematchfeedbackloop) | **Get** /api2/json/japaneseNameMatchFeedbackLoop/{japaneseSurnameLatin}/{japaneseGivenNameLatin}/{japaneseName} | [CREDITS 1 UNIT] Feedback loop to better perform matching Japanese name in KANJI ex. 山本 早苗 with a romanized name ex. Yamamoto Sanae
*JapaneseApi* | [**ParseJapaneseName**](docs/JapaneseApi.md#parsejapanesename) | **Get** /api2/json/parseJapaneseName/{japaneseName} | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae
*JapaneseApi* | [**ParseJapaneseNameBatch**](docs/JapaneseApi.md#parsejapanesenamebatch) | **Post** /api2/json/parseJapaneseNameBatch | Infer the likely first/last name structure of a name, ex. 山本 早苗 or Yamamoto Sanae 
*PersonalApi* | [**Corridor**](docs/PersonalApi.md#corridor) | **Get** /api2/json/corridor/{countryIso2From}/{firstNameFrom}/{lastNameFrom}/{countryIso2To}/{firstNameTo}/{lastNameTo} | [USES 20 UNITS PER NAME COUPLE] Infer several classifications for a cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**CorridorBatch**](docs/PersonalApi.md#corridorbatch) | **Post** /api2/json/corridorBatch | [USES 20 UNITS PER NAME PAIR] Infer several classifications for up to 100 cross border interaction between names (ex. remit, travel, intl com)
*PersonalApi* | [**Country**](docs/PersonalApi.md#country) | **Get** /api2/json/country/{personalNameFull} | [USES 10 UNITS PER NAME] Infer the likely country of residence of a personal full name, or one surname. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**CountryBatch**](docs/PersonalApi.md#countrybatch) | **Post** /api2/json/countryBatch | [USES 10 UNITS PER NAME] Infer the likely country of residence of up to 100 personal full names, or surnames. Assumes names as they are in the country of residence OR the country of origin.
*PersonalApi* | [**Diaspora**](docs/PersonalApi.md#diaspora) | **Get** /api2/json/diaspora/{countryIso2}/{firstName}/{lastName} | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of a personal name, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**DiasporaBatch**](docs/PersonalApi.md#diasporabatch) | **Post** /api2/json/diasporaBatch | [USES 20 UNITS PER NAME] Infer the likely ethnicity/diaspora of up to 100 personal names, given a country of residence ISO2 code (ex. US, CA, AU, NZ etc.)
*PersonalApi* | [**Gender**](docs/PersonalApi.md#gender) | **Get** /api2/json/gender/{firstName}/{lastName} | Infer the likely gender of a name.
*PersonalApi* | [**GenderBatch**](docs/PersonalApi.md#genderbatch) | **Post** /api2/json/genderBatch | Infer the likely gender of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**GenderFull**](docs/PersonalApi.md#genderfull) | **Get** /api2/json/genderFull/{fullName} | Infer the likely gender of a full name, ex. John H. Smith
*PersonalApi* | [**GenderFullBatch**](docs/PersonalApi.md#genderfullbatch) | **Post** /api2/json/genderFullBatch | Infer the likely gender of up to 100 full names, detecting automatically the cultural context.
*PersonalApi* | [**GenderFullGeo**](docs/PersonalApi.md#genderfullgeo) | **Get** /api2/json/genderFullGeo/{fullName}/{countryIso2} | Infer the likely gender of a full name, given a local context (ISO2 country code).
*PersonalApi* | [**GenderFullGeoBatch**](docs/PersonalApi.md#genderfullgeobatch) | **Post** /api2/json/genderFullGeoBatch | Infer the likely gender of up to 100 full names, with a given cultural context (country ISO2 code).
*PersonalApi* | [**GenderGeo**](docs/PersonalApi.md#gendergeo) | **Get** /api2/json/genderGeo/{firstName}/{lastName}/{countryIso2} | Infer the likely gender of a name, given a local context (ISO2 country code).
*PersonalApi* | [**GenderGeoBatch**](docs/PersonalApi.md#gendergeobatch) | **Post** /api2/json/genderGeoBatch | Infer the likely gender of up to 100 names, each given a local context (ISO2 country code).
*PersonalApi* | [**Origin**](docs/PersonalApi.md#origin) | **Get** /api2/json/origin/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer the likely country of origin of a personal name. Assumes names as they are in the country of origin. For US, CA, AU, NZ and other melting-pots : use &#39;diaspora&#39; instead.
*PersonalApi* | [**OriginBatch**](docs/PersonalApi.md#originbatch) | **Post** /api2/json/originBatch | [USES 10 UNITS PER NAME] Infer the likely country of origin of up to 100 names, detecting automatically the cultural context.
*PersonalApi* | [**ParseName**](docs/PersonalApi.md#parsename) | **Get** /api2/json/parseName/{nameFull} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. 
*PersonalApi* | [**ParseNameBatch**](docs/PersonalApi.md#parsenamebatch) | **Post** /api2/json/parseNameBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John.
*PersonalApi* | [**ParseNameGeo**](docs/PersonalApi.md#parsenamegeo) | **Get** /api2/json/parseName/{nameFull}/{countryIso2} | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. For better accuracy, provide a geographic context.
*PersonalApi* | [**ParseNameGeoBatch**](docs/PersonalApi.md#parsenamegeobatch) | **Post** /api2/json/parseNameGeoBatch | Infer the likely first/last name structure of a name, ex. John Smith or SMITH, John or SMITH; John. Giving a local context improves precision. 
*PersonalApi* | [**UsRaceEthnicity**](docs/PersonalApi.md#usraceethnicity) | **Get** /api2/json/usRaceEthnicity/{firstName}/{lastName} | [USES 10 UNITS PER NAME] Infer a US resident&#39;s likely race/ethnicity according to US Census taxonomy W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**UsRaceEthnicityBatch**](docs/PersonalApi.md#usraceethnicitybatch) | **Post** /api2/json/usRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#39;s likely race/ethnicity according to US Census taxonomy. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**UsRaceEthnicityZIP5**](docs/PersonalApi.md#usraceethnicityzip5) | **Get** /api2/json/usRaceEthnicityZIP5/{firstName}/{lastName}/{zip5Code} | [USES 10 UNITS PER NAME] Infer a US resident&#39;s likely race/ethnicity according to US Census taxonomy, using (optional) ZIP5 code info. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*PersonalApi* | [**UsZipRaceEthnicityBatch**](docs/PersonalApi.md#uszipraceethnicitybatch) | **Post** /api2/json/usZipRaceEthnicityBatch | [USES 10 UNITS PER NAME] Infer up-to 100 US resident&#39;s likely race/ethnicity according to US Census taxonomy, with (optional) ZIP code. Output is W_NL (white, non latino), HL (hispano latino),  A (asian, non latino), B_NL (black, non latino). Optionally add header X-OPTION-USRACEETHNICITY-TAXONOMY: USRACEETHNICITY-6CLASSES for two additional classes, AI_AN (American Indian or Alaskan Native) and PI (Pacific Islander).
*SocialApi* | [**PhoneCode**](docs/SocialApi.md#phonecode) | **Get** /api2/json/phoneCode/{firstName}/{lastName}/{phoneNumber} | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, given a personal name and formatted / unformatted phone number.
*SocialApi* | [**PhoneCodeBatch**](docs/SocialApi.md#phonecodebatch) | **Post** /api2/json/phoneCodeBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, detecting automatically the local context given a name and formatted / unformatted phone number.
*SocialApi* | [**PhoneCodeGeo**](docs/SocialApi.md#phonecodegeo) | **Get** /api2/json/phoneCodeGeo/{firstName}/{lastName}/{phoneNumber}/{countryIso2} | [USES 11 UNITS PER NAME] Infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).
*SocialApi* | [**PhoneCodeGeoBatch**](docs/SocialApi.md#phonecodegeobatch) | **Post** /api2/json/phoneCodeGeoBatch | [USES 11 UNITS PER NAME] Infer the likely country and phone prefix, of up to 100 personal names, with a local context (ISO2 country of residence).
*SocialApi* | [**PhoneCodeGeoFeedbackLoop**](docs/SocialApi.md#phonecodegeofeedbackloop) | **Get** /api2/json/phoneCodeGeoFeedbackLoop/{firstName}/{lastName}/{phoneNumber}/{phoneNumberE164}/{countryIso2} | [CREDITS 1 UNIT] Feedback loop to better infer the likely phone prefix, given a personal name and formatted / unformatted phone number, with a local context (ISO2 country of residence).


## Documentation For Models

 - [ApiBillingPeriodUsageOut](docs/ApiBillingPeriodUsageOut.md)
 - [ApiClassifierOut](docs/ApiClassifierOut.md)
 - [ApiClassifierTaxonomyOut](docs/ApiClassifierTaxonomyOut.md)
 - [ApiClassifiersStatusOut](docs/ApiClassifiersStatusOut.md)
 - [ApiCounterV2Out](docs/ApiCounterV2Out.md)
 - [ApiKeyOut](docs/ApiKeyOut.md)
 - [ApiPeriodUsageOut](docs/ApiPeriodUsageOut.md)
 - [ApiPlanSubscriptionOut](docs/ApiPlanSubscriptionOut.md)
 - [ApiServiceOut](docs/ApiServiceOut.md)
 - [ApiServicesOut](docs/ApiServicesOut.md)
 - [ApiUsageAggregatedOut](docs/ApiUsageAggregatedOut.md)
 - [ApiUsageHistoryOut](docs/ApiUsageHistoryOut.md)
 - [BatchCorridorIn](docs/BatchCorridorIn.md)
 - [BatchCorridorOut](docs/BatchCorridorOut.md)
 - [BatchFirstLastNameDiasporaedOut](docs/BatchFirstLastNameDiasporaedOut.md)
 - [BatchFirstLastNameGenderIn](docs/BatchFirstLastNameGenderIn.md)
 - [BatchFirstLastNameGenderedOut](docs/BatchFirstLastNameGenderedOut.md)
 - [BatchFirstLastNameGeoIn](docs/BatchFirstLastNameGeoIn.md)
 - [BatchFirstLastNameGeoZippedIn](docs/BatchFirstLastNameGeoZippedIn.md)
 - [BatchFirstLastNameIn](docs/BatchFirstLastNameIn.md)
 - [BatchFirstLastNameOriginedOut](docs/BatchFirstLastNameOriginedOut.md)
 - [BatchFirstLastNamePhoneCodedOut](docs/BatchFirstLastNamePhoneCodedOut.md)
 - [BatchFirstLastNamePhoneNumberGeoIn](docs/BatchFirstLastNamePhoneNumberGeoIn.md)
 - [BatchFirstLastNamePhoneNumberIn](docs/BatchFirstLastNamePhoneNumberIn.md)
 - [BatchFirstLastNameUsRaceEthnicityOut](docs/BatchFirstLastNameUsRaceEthnicityOut.md)
 - [BatchMatchPersonalFirstLastNameIn](docs/BatchMatchPersonalFirstLastNameIn.md)
 - [BatchNameGeoIn](docs/BatchNameGeoIn.md)
 - [BatchNameIn](docs/BatchNameIn.md)
 - [BatchNameMatchCandidatesOut](docs/BatchNameMatchCandidatesOut.md)
 - [BatchNameMatchedOut](docs/BatchNameMatchedOut.md)
 - [BatchPersonalNameGenderedOut](docs/BatchPersonalNameGenderedOut.md)
 - [BatchPersonalNameGeoIn](docs/BatchPersonalNameGeoIn.md)
 - [BatchPersonalNameGeoOut](docs/BatchPersonalNameGeoOut.md)
 - [BatchPersonalNameIn](docs/BatchPersonalNameIn.md)
 - [BatchPersonalNameParsedOut](docs/BatchPersonalNameParsedOut.md)
 - [BatchProperNounCategorizedOut](docs/BatchProperNounCategorizedOut.md)
 - [CorridorIn](docs/CorridorIn.md)
 - [CorridorOut](docs/CorridorOut.md)
 - [FeedbackLoopOut](docs/FeedbackLoopOut.md)
 - [FirstLastNameDiasporaedOut](docs/FirstLastNameDiasporaedOut.md)
 - [FirstLastNameGenderIn](docs/FirstLastNameGenderIn.md)
 - [FirstLastNameGenderedOut](docs/FirstLastNameGenderedOut.md)
 - [FirstLastNameGeoIn](docs/FirstLastNameGeoIn.md)
 - [FirstLastNameGeoZippedIn](docs/FirstLastNameGeoZippedIn.md)
 - [FirstLastNameIn](docs/FirstLastNameIn.md)
 - [FirstLastNameOriginedOut](docs/FirstLastNameOriginedOut.md)
 - [FirstLastNameOut](docs/FirstLastNameOut.md)
 - [FirstLastNamePhoneCodedOut](docs/FirstLastNamePhoneCodedOut.md)
 - [FirstLastNamePhoneNumberGeoIn](docs/FirstLastNamePhoneNumberGeoIn.md)
 - [FirstLastNamePhoneNumberIn](docs/FirstLastNamePhoneNumberIn.md)
 - [FirstLastNameUsRaceEthnicityOut](docs/FirstLastNameUsRaceEthnicityOut.md)
 - [MatchPersonalFirstLastNameIn](docs/MatchPersonalFirstLastNameIn.md)
 - [NameGeoIn](docs/NameGeoIn.md)
 - [NameIn](docs/NameIn.md)
 - [NameMatchCandidateOut](docs/NameMatchCandidateOut.md)
 - [NameMatchCandidatesOut](docs/NameMatchCandidatesOut.md)
 - [NameMatchedOut](docs/NameMatchedOut.md)
 - [PersonalNameGenderedOut](docs/PersonalNameGenderedOut.md)
 - [PersonalNameGeoIn](docs/PersonalNameGeoIn.md)
 - [PersonalNameGeoOut](docs/PersonalNameGeoOut.md)
 - [PersonalNameIn](docs/PersonalNameIn.md)
 - [PersonalNameParsedOut](docs/PersonalNameParsedOut.md)
 - [ProperNounCategorizedOut](docs/ProperNounCategorizedOut.md)
 - [SoftwareVersionOut](docs/SoftwareVersionOut.md)


## Documentation For Authorization

## api_key
- **Type**: API key 

Example
```golang
auth := context.WithValue(context.Background(), sw.ContextAPIKey, sw.APIKey{
	Key: "APIKEY",
	Prefix: "Bearer", // Omit if not necessary.
})
r, err := client.Service.Operation(auth, args)
```

## Author

contact@namsor.com

 readmeEtag: '"bda79e140b0503daa709e3849df196883b6432b2"' readmeLastModified: Sat, 17 Jul 2021 16:37:57 GMT repositoryId: 311337702 description: >- NamSor API v2 GO golang SDK - classify personal names accurately by gender, country of origin, or ethnicity. created: '2020-11-09T12:52:55Z' updated: '2021-07-17T16:38:17Z' language: Go archived: false stars: 1 watchers: 1 forks: 0 owner: namsor logo: https://avatars.githubusercontent.com/u/6951565?v=4 license: AGPL-3.0 repoEtag: '"752eff8861be26c6fdaa3c00b6f8f940ab8fb7365d4e2a2d124758e35f00eede"' repoLastModified: Sat, 17 Jul 2021 16:38:17 GMT foundInMaster: true category: Description Validators id: 82acbfdc33ea8424ebd5e94a6da14642 - source: openapi3 tags repository: https://github.com/nzlouislu/games-api v3: true id: e3828269cbfbbdda24ebad9ce4fc8b01 repositoryMetadata: base64Readme: >- IyBHYW1lcyBBUEkgU2VydmljZQpBbiBBUEkgc2VydmljZSBmb3IgZmluZGluZyBhbGwgZ2FtZXMsIGNyZWF0aW5nLCByZXRyaWV2aW5nLCB1cGRhdGluZywgYW5kIGRlbGV0aW5nIEdhbWVzIGFuZCBwbGF0Zm9ybXMuCgotR2FtZXMgQVBJIFVSTCA6IGh0dHBzOi8vZ2FtZXNhcGkubnpsb3Vpcy5jb20vZG9jLmh0bWwKCi1Mb2NhbCBTd2FnZ2VyIFVJIFVSTCA6IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4Ny9kb2MuaHRtbAoKVXNlIG9mIHRlY2hub2xvZ3k6IFNwcmluZ2Jvb3QyLjMgKyBTcHJpbmdkb2Mtb3BlbmFwaSAzICsgSlBBICsgSDIgKyBKdW5pdDQKCiIjIGdhbWVzLWFwaSIgCgohW10oc3JjL21haW4vcmVzb3VyY2VzL0dhbWVzLmpwZyk= readmeEtag: '"22611c4d40d9189170d6fd282b97dc9c96ceca90"' readmeLastModified: Tue, 25 Apr 2023 08:00:55 GMT repositoryId: 629405490 description: Games API Service created: '2023-04-18T08:46:31Z' updated: '2023-04-25T07:45:59Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: NZLouislu logo: https://avatars.githubusercontent.com/u/72653348?v=4 repoEtag: '"82dfc48157b244d11fc3e105b1d420009c30a268a2441e22f7f2d790d0e78eaf"' repoLastModified: Tue, 25 Apr 2023 07:45:59 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/ailouislu/games-api - source: openapi3 tags repository: https://github.com/voto-vote/api-specification v3: true id: f1bb93898f6a8a4211765e5b6fd62aaf repositoryMetadata: base64Readme: >- IyBWT1RPIEFQSSBTcGVjaWZpY2F0aW9uCgohW01JVF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2ZykKWyFbQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS92b3RvLXZvdGUvYXBpLXNwZWNpZmljYXRpb24vYWN0aW9ucy93b3JrZmxvd3MvcmVkb2MtZ2gtcGFnZXMueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vdm90by12b3RlL2FwaS1zcGVjaWZpY2F0aW9uL2FjdGlvbnMvd29ya2Zsb3dzL3JlZG9jLWdoLXBhZ2VzLnlhbWwpClshW0dpdGh1YiBQYWdlc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby93ZWJzaXRlLXVwLWRvd24tZ3JlZW4tcmVkL2h0dHAvc2hpZWxkcy5pby5zdmcpXShodHRwczovL3ZvdG8tdm90ZS5naXRodWIuaW8vYXBpLXNwZWNpZmljYXRpb24vKQoKVGhpcyBwcm9qZWN0IGNvbnRhaW5zIHRoZSBzcGVjaWZpY2F0aW9uIG9mIHRoZSBWT1RPIEFQSS4gVGhlIHNjaGVtYSBpcyBiYXNlZCBvbiB0aGUgW09wZW5BUEkgU3BlY2lmaWNhdGlvbiB2My4xXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikuCgojIyBJbnRyb2R1Y3Rpb24KCldlIHNhdyB0aGUgbmVlZCB0byBjcmVhdGUgYSBzdGFuZGFyZCBzZXQgb2YgaW5mb3JtYXRpb24gZm9yIHZvdGUgYWR2aWNlIGFwcGxpY2F0aW9ucy4gVGhlcmVmb3JlLCB3ZSBjYW1lIHVwIHdpdGggdGhlIGlkZWEgb2YgZGVmaW5pbmcgYSBzdGFuZGFyZCBBUEkgd2hpbGUgZnVsbGZpbGxpbmcgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBmb3JtYXQuIFRoaXMgYWxsb3dzIGVhc3kgYWRvcHRpb24gd2l0aCBnZW5lcmF0ZWQgY2xpZW50IGFuZCBzZXJ2ZXIgc3R1YnMuCgojIyBVc2FnZQoKWW91IGFyZSBmcmVlIHRvIHVzZSB0aGUgc3BlY2lmaWNhdGlvbiB0byBnZW5lcmF0ZSBhIGNsaWVudCwgd2hpY2ggY2FuIGJlIHVzZWQgZm9yIGFjY2Vzc2luZyB0aGUgQVBJLiBXZSBoaWdobHkgcmVjb21tZW5kIHVzaW5nIHRoZSBhcGktZ2VuZXJhdG9yLCB3aGljaCBpcyBjYWJhYmxlIG9mIGdlbmVyYXRpbmcgZnVsbHkgdXNhYmxlIGNsaWVudHMgaW4gYWxtb3N0IGV2ZXJ5IHByb2dyYW1taW5nIGxhbmd1YWdlLgoKIyMjIERvY2tlciB1c2FnZQoKR28gaW50byBgb3BlbmFwaWAgZGlyZWN0b3J5IGFuZCBzaW1wbHkgYnVpbGQgYW5kIHJ1biBvdXIgZG9ja2VyIGltYWdlOgoKYGBgc2gKZG9ja2VyIGJ1aWxkIC10IG9wZW4tYXBpLWdlbmVyYXRvcjpsYXRlc3QgLiAmJiBkb2NrZXIgcnVuIC12ICR7UFdEfS86L2FwcC8gLWUgTEFOR1VBR0U9Z28gb3Blbi1hcGktZ2VuZXJhdG9yCmBgYAoKWW91IGNhbiBzcGVjaWZ5IHRoZSBMQU5HVUFHRSBmb3IgdGhlIGdlbmVyYXRlZCBjb2RlIHVzaW5nIHRoZSBgTEFOR1VBR0VgIGVudmlyb25tZW50IHZhcmlhYmxlLiBDaGVjayBvdXQgdGhlIGxpc3QgZnJvbSBbaGVyZV0oaHR0cHM6Ly9vcGVuYXBpLWdlbmVyYXRvci50ZWNoL2RvY3MvZ2VuZXJhdG9ycy8pLiBBZnRlciBydW5uaW5nLCB5b3Ugd2lsbCBmaW5kIHRoZSBgb3V0cHV0YCBmb2xkZXIsIHdoZXJlIG9uZSBtZXJnZWQgc3BlY2lmaWNhdGlvbiBmaWxlIGFuZCB0aGUgZ2VuZXJhdGVkIGNvZGUgc3RheXMuCgojIyMgTWFudWFsIHVzYWdlCgpJZiB5b3Ugd2FudCB0byBydW4gZXZlcnl0aGluZyBtYW51YWxseSwgZ28gdG8gYG9wZW5hcGlgIGZvbGRlciBhbmQgcnVuOgoKYGBgc2gKIyBJbnN0YWxsIHJlZG9jbHkgb3BlbmFwaS1jbGk6Cm5wbSBpbnN0YWxsIEByZWRvY2x5L29wZW5hcGktY2xpIC1nCiMgSW5zdGFsbCBvcGVuYXBpLWdlbmVyYXRvcjoKYnJldyBpbnN0YWxsIG9wZW5hcGktZ2VuZXJhdG9yCiMgUnVuIExpbnRlcjoKbnB4IEByZWRvY2x5L29wZW5hcGktY2xpIGxpbnQgc3BlY2lmaWNhdGlvbi55YW1sCiMgTWVyZ2UgZmlsZXM6Cm5weCBAcmVkb2NseS9vcGVuYXBpLWNsaSBidW5kbGUgc3BlY2lmaWNhdGlvbi55YW1sIC1vIC4vb3V0cHV0L3NwZWMtbWVyZ2VkLnlhbWwKIyBSdW4gb3BlbmFwaSBnZW5lcmF0b3I6Cm9wZW5hcGktZ2VuZXJhdG9yLWNsaSBnZW5lcmF0ZSAtaSAuL3NwZWMtbWVyZ2VkLnlhbWwgLWcgJExBTkdVQUdFIC1vIC4vZ2VuZXJhdGVkLwpgYGAKCkZvciBtb3JlIGluZm9ybWF0aW9uLCBwbGVhc2UgdmlzaXQ6IFtodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vT3BlbkFQSVRvb2xzL29wZW5hcGktZ2VuZXJhdG9yKQo= readmeEtag: '"3b71e6c50dee056bfafc2888158531af9f7485aa"' readmeLastModified: Mon, 04 Mar 2024 06:37:11 GMT repositoryId: 413306506 description: 'VOTO API specified in OpenAPI Specification yml. ' created: '2021-10-04T06:52:07Z' updated: '2025-03-03T08:46:38Z' language: Shell archived: false stars: 1 watchers: 2 forks: 0 owner: voto-vote logo: https://avatars.githubusercontent.com/u/63105374?v=4 license: MIT repoEtag: '"83cedc685f0df48fdf541aefe2ed9d4b7a6d8b208ab37dd44e225af9567029e5"' repoLastModified: Mon, 03 Mar 2025 08:46:38 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/robbot-discord/robbot-api v3: true repositoryMetadata: base64Readme: >- Um9iQm90IEFQSQo9PT0KClRoZSBPcGVuQVBJIHYzIFNjaGVtYSBkZWZpbml0aW9uIHVzZWQgYmVoaW5kIHRoZSBzY2VuZXMgaW4gUm9iQm90CgojIFJlcXVpcmVtZW50cwpbIVtGT1NTQSBTdGF0dXNdKGh0dHBzOi8vYXBwLmZvc3NhLmlvL2FwaS9wcm9qZWN0cy9naXQlMkJnaXRodWIuY29tJTJGcm9iYm90LWRpc2NvcmQlMkZSb2JCb3QtQVBJLnN2Zz90eXBlPXNoaWVsZCldKGh0dHBzOi8vYXBwLmZvc3NhLmlvL3Byb2plY3RzL2dpdCUyQmdpdGh1Yi5jb20lMkZyb2Jib3QtZGlzY29yZCUyRlJvYkJvdC1BUEk/cmVmPWJhZGdlX3NoaWVsZCkKCiogW1N0b3BsaWdodCBTdHVkaW9dKGh0dHBzOi8vc3RvcGxpZ2h0LmlvL3N0dWRpby8pCiogW05vZGUuanMgMTArXShodHRwczovL25vZGVqcy5vcmcvZW4vKQoKIyBHZXR0aW5nIFN0YXJ0ZWQKMS4gUnVuIHRoZSBmb2xsb3dpbmcgdG8gc2V0dXAgR2l0IGhvb2tzIGFuZCBpbnN0YWxsIGRlcGVuZGVuY2llcyByZXF1aXJlZCBmb3IgdXRpbGl0eSBzY3JpcHRzOgpgYGBzaGVsbCBzY3JpcHQKbnBtIGluc3RhbGwKYGBgCjEuIE9wZW4gdGhpcyByZXBvc2l0b3J5IGluIFN0b3BsaWdodCBTdHVkaW8gYW5kIHN0YXJ0IGVkaXRpbmcKCiMjIExpbnRpbmcKYGBgc2hlbGwgc2NyaXB0Cm5wbSBydW4gbGludApgYGAKCiMjIExpY2Vuc2UKWyFbRk9TU0EgU3RhdHVzXShodHRwczovL2FwcC5mb3NzYS5pby9hcGkvcHJvamVjdHMvZ2l0JTJCZ2l0aHViLmNvbSUyRnJvYmJvdC1kaXNjb3JkJTJGUm9iQm90LUFQSS5zdmc/dHlwZT1sYXJnZSldKGh0dHBzOi8vYXBwLmZvc3NhLmlvL3Byb2plY3RzL2dpdCUyQmdpdGh1Yi5jb20lMkZyb2Jib3QtZGlzY29yZCUyRlJvYkJvdC1BUEk/cmVmPWJhZGdlX2xhcmdlKQ== readmeEtag: '"8eb02f72bac3d65a6f4081efe26dd9216c2e8781"' readmeLastModified: Mon, 21 Sep 2020 06:37:25 GMT repositoryId: 217418543 description: The OpenAPI v3 Schema definition used behind the scenes in RobBot created: '2019-10-25T00:27:52Z' updated: '2024-06-04T23:52:14Z' language: null archived: false stars: 1 watchers: 0 forks: 1 owner: robbot-discord logo: https://avatars.githubusercontent.com/u/46425294?v=4 license: MIT repoEtag: '"71a8f4471fdbc26f2dbf512e4c1d56ddd5ceb81a551e27f4c698d6c2f8042e5b"' repoLastModified: Tue, 04 Jun 2024 23:52:14 GMT foundInMaster: true category: Server id: 975cd4c8a22463c3d61d7d414b3322c1 - source: openapi3 tags repository: https://github.com/farhadrezvani/http-generator v3: true id: e6df472abbe7ba67700b466005f08cba repositoryMetadata: base64Readme: >- IyBodHRwLWdlbmVyYXRvcgoKR2VuZXJhdGUgYC5odHRwYCBmaWxlcyBmcm9tIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMKCiMjIEZlYXR1cmVzCgotIEdlbmVyYXRlIEhUVFAgZmlsZShzKSBlaXRoZXIgYXMKCiAgLSBBIHNpbmdsZSBmaWxlIGNvbnRhaW5pbmcgYWxsIHJlcXVlc3RzCiAgLSBBIGZpbGUgcGVyIHJlcXVlc3QKCi0gU3VwcG9ydHMgT3BlbkFQSSB2MiBhbmQgdjMKCiAgLSBKU09OIGFuZCBZQU1MIGZvcm1hdHMKICAtIFNjaGVtYSBWYWxpZGF0aW9uCgotIEluY2x1ZGUgYXV0aG9yaXphdGlvbiBoZWFkZXJzCi0gSW5jbHVkZSBzdW1tYXJpZXMgYW5kIGRlc2NyaXB0aW9ucwotIFZhcmlhYmxlcyBmb3Igcm91dGUgcGFyYW1ldGVycwotIFNwZWNpZnkgYmFzZS11cmwgZm9yIGNvbnZlbmllbnQgZW52aXJvbm1lbnQgc3dpdGNoaW5nCgojIyBJbnN0YWxsYXRpb24KCkluc3RhbGwgaW4geW91ciBwcm9qZWN0IGZvbGRlcjoKCmBgYGJhc2gKbnBtIGkgaHR0cC1nZW5lcmF0b3IgLUQKIyBVc2luZyB5YXJuCnlhcm4gYWRkIGh0dHAtZ2VuZXJhdG9yIC0tZGV2CiMgVXNpbmcgcG5wbQpwbnBtIGFkZCBodHRwLWdlbmVyYXRvciAtRApgYGAKCiMjIFVzYWdlCgpSdW4gaHR0cC1nZW5lcmF0b3Igd2l0aCB0aGUgYXBwcm9wcmlhdGUgb3B0aW9uczoKCmBgYGJhc2gKVXNhZ2U6IGh0dHAtZ2VuZXJhdG9yIFtvcHRpb25zXQoKR2VuZXJhdGUgLmh0dHAgZmlsZXMgZnJvbSBPcGVuQVBJIHNwZWNpZmljYXRpb25zCgpPcHRpb25zOgogIC1pLCAtLWlucHV0IDxpbnB1dD4gICAgICAgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBmaWxlIG9yIFVSTAogIC1vLCAtLW91dHB1dCA8b3V0cHV0PiAgICAgT3V0cHV0IGRpcmVjdG9yeSBvciBIVFRQIGZpbGUKICAtYiwgLS1iYXNlLXVybCA8YmFzZVVybD4gIEJhc2UgVVJMIG9mIHRoZSBBUEkKICAtdCwgLS10b2tlbiA8dG9rZW4+ICAgICAgIEF1dGhvcml6YXRpb24gdG9rZW4KICAtcywgLS1za2lwLXZhbGlkYXRpb24gICAgIFNraXAgdmFsaWRhdGlvbiBvZiBPcGVuQVBJIFNwZWNpZmljYXRpb24gKGRlZmF1bHQ6IGZhbHNlKQogIC12LCAtLXZlcnNpb24gICAgICAgICAgICAgRGlzcGxheSB2ZXJzaW9uIG51bWJlcgogIC1oLCAtLWhlbHAgICAgICAgICAgICAgICAgRGlzcGxheSB0aGlzIG1lc3NhZ2UKYGBgCgojIyBDb25maWd1cmF0aW9uIEZpbGUKCllvdSBjYW4gYWxzbyB1c2UgYGh0dHAtZ2VuZXJhdG9yYCB1c2luZyBmaWxlIGNvbmZpZ3VyYXRpb25zIG9yIGluIGEgcHJvcGVydHkgaW5zaWRlIHlvdXIgcGFja2FnZS5qc29uLCBhbmQgeW91IGNhbiBldmVuIHVzZSBUeXBlU2NyaXB0IGFuZCBoYXZlIHR5cGUtc2FmZXR5IHdoaWxlIHlvdSBhcmUgdXNpbmcgaXQuCgpZb3UgY2FuIHVzZSBhbnkgb2YgdGhlc2UgZmlsZXM6CgotIGBodHRwZ2VuLmNvbmZpZy50c2AKLSBgaHR0cGdlbi5jb25maWcuanNgCi0gYGh0dHBnZW4uY29uZmlnLmNqc2AKLSBgaHR0cGdlbmAgcHJvcGVydHkgaW4geW91ciBgcGFja2FnZS5qc29uYAoKQmFzaWMgQ29uZmlndXJhdGlvbjoKCmBgYGpzCmltcG9ydCB7IGRlZmluZUNvbmZpZyB9IGZyb20gImh0dHAtZ2VuZXJhdG9yIjsKCmV4cG9ydCBkZWZhdWx0IGRlZmluZUNvbmZpZyh7CiAgaW5wdXQ6ICJzY2hlbWEuanNvbiIsCiAgb3V0cHV0OiAib3V0cHV0Lmh0dHAiLAogIGJhc2VVcmw6ICJodHRwczovL2V4YW1wbGUuY29tIiwKICBza2lwVmFsaWRhdGlvbjogZmFsc2UsCiAgdG9rZW46ICJCZWFyZXIgVG9rZW4iLAp9KTsKYGBgCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWUhIFBsZWFzZSBzdWJtaXQgaXNzdWVzIG9yIHB1bGwgcmVxdWVzdHMuCgojIyBMaWNlbnNlCgpMaWNlbnNlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSB0aGUgW0xJQ0VOU0VdKExJQ0VOU0UpIGZpbGUgZm9yIGRldGFpbHMuCg== readmeEtag: '"09664103a1687a0b58b549a1dba9d2f85d49e080"' readmeLastModified: Sat, 13 Jul 2024 12:01:59 GMT repositoryId: 815814066 description: Generate .http files from OpenAPI specifications created: '2024-06-16T08:38:48Z' updated: '2025-12-15T23:54:51Z' language: TypeScript archived: false stars: 2 watchers: 1 forks: 0 owner: farhadrezvani logo: https://avatars.githubusercontent.com/u/10855997?v=4 license: MIT repoEtag: '"9e5a3bc79780d5a840062b9b30442d3fdd961ab2cd796c2abfc97a5ae7a15c1c"' repoLastModified: Mon, 15 Dec 2025 23:54:51 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/nagybalint001/swagger-codegen v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIGNvZGVnZW4KU3dhZ2dlci9PcGVuQXBpIGJhc2VkIEFuZ3VsYXIgY2xpZW50IGdlbmVyYXRpb24gc2FtcGxlcwoKQ29kZSBnZW5lcmF0b3I6IFtOU3dhZyBnZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9SaWNvU3V0ZXIvTlN3YWcpCgpOU3dhZyBjb25maWcgZ2VuZXJhdG9yOiBbTlN3YWcgU3R1ZGlvXShodHRwczovL2dpdGh1Yi5jb20vUmljb1N1dGVyL05Td2FnL3dpa2kvTlN3YWdTdHVkaW8pCgojIyAuTkVUL01TQnVpbGQgYmFzZWQgc2FtcGxlcwoKVGhlIEFuZ3VsYXIgY2xpZW50IGlzIGdlbmVyYXRlZCBkdXJpbmcgdGhlIGJ1aWxkIG9mIHRoZSBkb3RuZXQgcHJvamVjdCAoZnJvbSB0aGUgY29udHJvbGxlcnMsIG5vdCBmcm9tIGEgc3dhZ2dlci5qc29uKS4KClJlc3RmdWwgQVBJIFtzYW1wbGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9uYWd5YmFsaW50MDAxL3N3YWdnZXItY29kZWdlbi90cmVlL21hc3Rlci9SZXN0ZnVsKQoKUlBDIHN0eWxlIEFQSSBbc2FtcGxlXShodHRwczovL2dpdGh1Yi5jb20vbmFneWJhbGludDAwMS9zd2FnZ2VyLWNvZGVnZW4vdHJlZS9tYXN0ZXIvUnBjKQoKIyMgTlBNIGJhc2VkIHNhbXBsZQoKVG8gZ2VuZXJhdGUgdGhlIEFuZ3VsYXIgY2xpZW50IHlvdSBzaG91bGQgcnVuIHRoZSBgbnBtIHJ1biBzd2FnZ2VyYCBjb21tYW5kLgoKUlBDIHN0eWxlIEFQSSBbc2FtcGxlXShodHRwczovL2dpdGh1Yi5jb20vbmFneWJhbGludDAwMS9zd2FnZ2VyLWNvZGVnZW4vdHJlZS9tYXN0ZXIvTnBtKQo= readmeEtag: '"f8fb08f8bbb75a784323a3b8b04617015faa30c9"' readmeLastModified: Tue, 16 Jul 2019 17:36:11 GMT repositoryId: 194234991 description: Swagger/OpenApi based Angular client generation samples created: '2019-06-28T08:14:57Z' updated: '2020-03-18T20:57:44Z' language: TypeScript archived: false stars: 1 watchers: 0 forks: 0 owner: nagybalint001 logo: https://avatars.githubusercontent.com/u/17621599?v=4 repoEtag: '"08e16ee6af23032d340875c54178d4a11abbb571423a2082385d935c97a7d6ba"' repoLastModified: Wed, 18 Mar 2020 20:57:44 GMT foundInMaster: true category: SDK id: 6fdb051d7940b2439e25ed564d1349b8 - source: openapi3 tags repository: https://github.com/albankora/decho v3: true repositoryMetadata: base64Readme: >- IyBERUNITwoKIyMgR29hbApUaGUgZ29hbCBpcyB0byBhY2NlbGVyYXRlIHRoZSBwcm9jZXNzIG9mIGJvb3RzdHJhcHBpbmcgYW4gUmVzdGZ1bGwgQVBJIGJhc2UgZ29sYW5nIGFwcGxpY2F0aW9uCgojIyBIb3cKRG9ja2VyIGlzIHVzZWQgdG8gYWNjZWxlcmF0ZSBsb2NhbCBzZXR1cCBmb3IgW0VjaG8gZnJhbWV3b3JrXShodHRwczovL2dpdGh1Yi5jb20vbGFic3RhY2svZWNobykgYW5kIGFsc28geW91IGNhbiBnZW5lcmF0ZSBhIGBzY3JhdGNoYCBpbWFnZSB3aXRoIGEgbWluaW11bSBmb290cHJpbnQgdGhhdCBjYW4gYmUgdXNlZCBpbiBwcm9kdWN0aW9uLgoKIyMgU2V0dXAgbG9jYWwgZW52aXJvbm1lbnQgdmFyaWFibGVzCmBgYGJhc2gKY3AgLmVudi5leGFtcGxlIC5lbnYKYGBgCgojIyBUaGUgY29tbWFuZCBzY3JpcHQKYGBgYmFzaAouL2FwcCB7Y21kfQpgYGAKCiMjIEJ1aWxkIGxvY2FsIGRldgpgYGBiYXNoCiMgYnVpbGQgbG9jYWwgZG9ja2VyIGNvbnRhaW5lciBhbmQgZG93bmxvYWQgcmVxdWlyZWQgbGlicwouL2FwcCBidWlsZApgYGAKCiMjIFJ1biBsb2NhbCBkZXYKYGBgYmFzaAojIHJ1biBsb2NhbCBkb2NrZXIgY29udGFpbmVyCi4vYXBwIHJ1bgpgYGAKCiMjIERldiBzZXJ2ZXIgVVJMCmBodHRwOi8vMTI3LjAuMC4xOjMwMDBgCgojIyBQcm9kdWN0aW9uIGRvY2tlciBpbWFnZQpgYGBiYXNoCiMgY3JlYXRlcyBhIGRvY2tlciBwcm9kdWN0aW9uIGltYWdlCi4vYXBwIHByZApgYGAKCiMjIEZ1bGwgbGlzdCBvZiBjb21tYW5kczoKYGBgYmFzaApDb21tYW5kczoKCiBidWlsZCAgICAgICAgIEJ1aWxkIGxvY2FsIGRvY2tlciBjb250YWluZXIgYW5kIGRvd25sb2FkIHJlcXVpcmVkIGxpYnMKIHJ1biAgICAgICAgICAgUnVuIGxvY2FsIGRvY2tlciBjb250YWluZXIKIHN0b3AgICAgICAgICAgU3RvcCB5b3VyIGxvY2FsIGRvY2tlciBjb250YWluZXIKIHJlbW92ZSAgICAgICAgUmVtb3ZlIGxvY2FsIGNvbnRhaW5lcnMgYW5kIGV2ZXJ5dGhpbmcgcmVsYXRlZCB0byB0aGVtCiB0ZXN0ICAgICAgICAgIFJ1biB1bml0IHRlc3RzCiBjb3ZlciAgICAgICAgIFJ1biB1bml0IHRlc3RzIGFuZCBnZW5lcmF0ZSBjb3ZlcmFnZSBmaWxlCiBmbXQgICAgICAgICAgIEZvcm1hdCB0aGUgY29kZSB3aXRoIGdvIHN0YW5kYXJkcwogdmV0ICAgICAgICAgICBDaGVjayBhbmQgcmVwb3J0IGxpa2VseSBtaXN0YWtlcyBpbiBwYWNrYWdlcwogbW9kIFtDT01NQU5EXSBDb21tYW5kcyBmb3IgZ28gbW9kdWxlIG1haW50ZW5hbmNlCiBiYXNoICAgICAgICAgIEVudGVyIGRvY2tlcnMgYmFzaCBpbnRlcmZhY2UKIHByZCAgICAgICAgICAgQnVpbGQgc2NyYXRjaCBwcm9kdWN0aW9uIGltYWdlCiBycHJkICAgICAgICAgIFJ1biBwcm9kdWN0aW9uIGltYWdlCiBjb2RlZ2VuICAgICAgIEdlbmVyYXRlIGNvZGUgZnJvbSB0aGUgb3BlbiBhcGkgZG9jdW1lbnRhdGlvbgpgYGAKCiMjIFJlLXJ1biBvbiB0aGUgZmx5ClRoZSBhcHAgdXNlcyBbUmVmbGV4XShodHRwczovL2dpdGh1Yi5jb20vY2VzcGFyZS9yZWZsZXgpIGZvciByZS1ydW5uaW5nIHlvdXIgYXBwIHdoZW4geW91IGNoYW5nZSB5b3VyIGNvZGUKCiMjIFByb2plY3QgTGF5b3V0ClRoZSBhcHAgZm9sbG93cyB0aGUgW1N0YW5kYXJkIEdvIFByb2plY3QgTGF5b3V0XShodHRwczovL2dpdGh1Yi5jb20vZ29sYW5nLXN0YW5kYXJkcy9wcm9qZWN0LWxheW91dCkgc28gdGhlIGRpcmVjdG9yeSBzdHJ1Y3R1cmUgaW5jbHVkZXMgZm9sZGVycyBsaWtlIGBjbWRgLCBgaW50ZXJuYWxgLCBgcGtnYCBhbmQgYGJ1aWxkYC4gU3RhbmRhcmQgR28gUHJvamVjdCBMYXlvdXQgaXMgYW4gZW1lcmdpbmcgcHJvamVjdCBsYXlvdXQgYmFzZSBvbiB0aGUgR28gZWNvc3lzdGVtLgoKIyMgQVBJLUZpcnN0IGFwcHJvYWNoCkVkaXQgZmlsZSBgYXBpZG9jLnlhbWxgIGFuZCBnZW5lcmF0ZSBjb2RlIHVzaW5nIFtPQVBJLUNvZGVnZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9kZWVwbWFwL29hcGktY29kZWdlbikKYGBgYmFzaAojIGdlbmVyYXRlIGNvZGUgZnJvbSBvcGVuIGFwaSB5YW1sIGZpbGUKLi9hcHAgY29kZWdlbgpgYGAKVGhlIGBjb2RlZ2VuYCBjb21tYW5kIHJlYWRzIGBhcGlkb2MueWFtbGAgYW5kIGdlbmVyYXRlcyB0d28gZmlsZXMgYHNlcnZlci5nb2AgYW5kIGB0eXBlcy5nb2AgdW5kZXIgYGludGVybmFsL2NvZGVnZW5gLiBgdHlwZXMuZ29gIGhhcyB0aGUgZGF0YSBzdHJ1Y3R1cmVzIGRlZmZpbmVkIG9uIGBhcGlkb2MueWFtbGAgYW5kIHRoZSBgc2VydmVyLmdvYCBmaWxlIGhhcyB0aGUgc2VydmVyIHNpZGUgc2V0dXAgZm9yIHRoZSBBUEku readmeEtag: '"342d3ec71d088a7ce98c0dfa398f1cc3318afa8c"' readmeLastModified: Sat, 28 Mar 2020 22:32:43 GMT repositoryId: 245699994 description: 'DECHO - Docker setup for Echo framework ' created: '2020-03-07T20:22:05Z' updated: '2020-03-28T22:33:48Z' language: Go archived: false stars: 1 watchers: 1 forks: 0 owner: albankora logo: https://avatars.githubusercontent.com/u/771020?v=4 license: MIT repoEtag: '"8ca54cab9a8bb6faafc52674300cea911c33e13906bfaef139a5a8429774855d"' repoLastModified: Sat, 28 Mar 2020 22:33:48 GMT foundInMaster: true category: - Code Generators - Server Implementations id: ac3aa95db2d6a2402543e229a8914f75 - source: openapi3 tags repository: https://github.com/iamfrench/efrei-m2-st2dccc v3: true repositoryMetadata: base64Readme: >- # ST2DCCC
Cloud Integration (M2, M2-APP-LSI, M2-APP-RI, M2-PRO - 2021S9)

## Overview
This repository contains the final project (that includes labs 1 to 3).

Instructions are available in `.pdf` files at the root of the directory.

![Application Diagram](./artefacts/app-diagram.png)

## Prerequisite
This project runs on docker.

You can check if docker is properly installed and running by typing into a terminal a simple docker command such as `docker ps`:

![Test if docker is running using docker ps command](./artefacts/terminal-test-if-docker-is-running-docker-ps.gif)

## Installation
Clone this repository (you can use Git, or use the GUI client GitHub Desktop)

And that's it !

## Run
Now you can run the project. To do so, you will have to open a terminal and execute the following command :

*Optionnal* You can download pre-built images from the GitHub Container Registry using the following command:

```bash
docker pull ghcr.io/iamfrench/efrei-m2-st2dccc_movie
docker pull ghcr.io/iamfrench/efrei-m2-st2dccc_actor
```

Then you can start the application using the following command:

```bash
docker-compose up -d
```

> Hint: To speed up the build process, you can use the following command:

```bash
docker-compose build --parallel
```

This command will read the [`docker-compose.yml`](./docker-compose.yml) file and deploy the application on your local computer.

![Deploy the application is running using docker-compose up commands](./artefacts/terminal-deploy-the-application-docker-compose-up.gif)

You can check if the containers are running using this command:

```bash
docker-compose ps
```

![Check if the application is running using docker-compose ps commands](./artefacts/terminal-check-if-the-application-is-running-using-docker-compose-ps.png)

## Tests
Now that you have installed, and run the application, it's time to play with the project!

Open you web browser to [`http://localhost`](http://localhost)

The interface that is displayed is the default Swagger UI web Client. This web client has been fed with a definition file ([`swagger.yaml`](./swagger.yaml)) that describe our APIs.

![Open the Swagger UI web client using a basic web browser](./artefacts/chrome-open-swagger-web-client.png)

### Available endpoints and methods
Because we have two services running we have chosen to expose them to two different ports as displayed in the solution diagram.
The port `81` is used by the actor service and the port `82` is used by the movie service.

In order to use those endpoints on different ports in Swagger UI we have overwritten the default host (lines `4` and `9`) in the [`docker-compose.yml`](./docker-compose.yml) file:

![Override the default server](./artefacts/swagger-yaml-custom-servers.png)

#### Actor Service (on port 81)

##### `GET` finAll
> Return all movies stored in the movie service with actors

##### `GET` findOne/`{movieId}`
> Return a movie from the movie service, with actors by `movieId`


#### Movie Service (on port 82)

##### `GET` movies
> Return all movies

##### `POST` movies
> Add a new movie to the movie service

##### `GET` movies/`{movieId}`
> Return a single movie by `movieId` 

##### `PUT` movies/`{movieId}`
> Edit a movie by `movieId` 

##### `DELETE` movies/`{movieId}`
> Delete a movie by `movieId` 

### Test the circuit breaker
The circuit breaker is implemented in the actor service. It is used when this service call the movie service.

#### Service is unavailable (no response)
If the requested service is unavailable a fallback method is implemented to prevent a cascade failure.

To create a failure, you just need to kill the container running the movie service. To do so, we will use the Docker Desktop Dashboard, a GUI for Docker:

![Stop the container responsible for running the movie service](./artefacts/docker-desktop-dashboard-stop-movie-container.png)

Now that the container is down, we can test if the circuit breaker is working by executing a REST request to the actor service.

To do so, we will use the Swagger UI:

![Making a REST request when the movie service is down](./artefacts/chrome-swagger-ui-movie-service-down-request.png)

And here is the response:

![REST response from the actor service when the movie service is down](./artefacts/chrome-swagger-ui-movie-service-down-response.png)

As expected, the movie service is unavailable, therefore the fallback method have been called and the response modified accordingly.

Here is the code responsible for that (in the actor service):

![REST Client of the actor service embedded into a circuit breaker](./artefacts/actor-service-rest-client-with-fallback.png)


#### Service is slow or degraded (timelimit) 
If a request is taking too much time, a fallback method is taking care of providing a response to ensure a continuity of service.

To simulate this delay, we have implemented an argument (passed as a query parameter) for the `GET {movie-service}/movies/{id}` route in the movie service:

![Delay added to the query in the Movie Service using a pause statement](./artefacts/movie-service-delay.png)

And thanks to the Swagger UI we can test this:

![Swagger UI query that induce a delay of 600ms](./artefacts/chrome-swagger-ui-delay-query.png)

This produces the following response:

![Swagger UI response after a query inducing a delay of 600ms](./artefacts/chrome-swagger-ui-delay-response.png)

As expected, a fallback method has been triggered.

Here is the code responsible for that (in the actor service):

![REST Client of the actor service embedded into a circuit breaker](./artefacts/actor-service-rest-client-with-fallback.png)

The time limit has been set in the circuit breaker config:

![Circuit breaker config file](./artefacts/actor-service-circuit-breaker-config.png)

## Clean up
Now that you have finished playing with the application. It's time to clean the project.

To do so, open a terminal and type the following command:

```bash
docker-compose down -v
```

This command will stop and remove containers and networks from your computer assigned to our project.

![Cleaning the application using the docker-compose down command](./artefacts/terminal-retire-application-using-docker-compose-down.png)
 readmeEtag: '"d054fe66dbbf4eb50bb3c38e2914af17d53696a5"' readmeLastModified: Wed, 27 Jan 2021 00:33:46 GMT repositoryId: 323942348 description: Cloud Integration (M2, M2-APP-LSI, M2-APP-RI, M2-PRO - 2021S9) created: '2020-12-23T15:50:46Z' updated: '2021-01-27T00:33:55Z' language: Java archived: false stars: 1 watchers: 1 forks: 1 owner: IAmFrench logo: https://avatars.githubusercontent.com/u/10818450?v=4 repoEtag: '"af5a601dc0f873ee481e21cc69a8d16462db6673beee2fcc9e8e0130376e61d7"' repoLastModified: Wed, 27 Jan 2021 00:33:55 GMT foundInMaster: true category: Server Implementations id: 231ebfcfa88388aaeea1401e51d85d85 - source: openapi3 tags repository: https://github.com/theairblow/mojang-api v3: true repositoryMetadata: base64Readme: IyBNb2phbmcgQVBJIGRvY3VtZW50YXRpb24KTm90aGluZyB0byBzZWUgaGVyZQo= readmeEtag: '"960c922f43686516641ab631b98ed53ab83a211e"' readmeLastModified: Wed, 01 Mar 2023 12:31:15 GMT repositoryId: 479952503 description: Mojang API documented with Swagger created: '2022-04-10T08:06:58Z' updated: '2023-01-03T13:13:05Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: TheAirBlow logo: https://avatars.githubusercontent.com/u/68467762?v=4 repoEtag: '"9e34cfa5de3c494c7c3436c4e56980eb423779d630ef9b726893a0aa554b6607"' repoLastModified: Tue, 03 Jan 2023 13:13:05 GMT foundInMaster: true category: - Documentation - Parsers id: 59a847459f9074c871e1306d704a6a13 - source: openapi3 tags repository: >- https://github.com/sandeepbegudem/springboot-jwt-authorization-junit5-mockito v3: true id: 1720dd9011e9ff1060630ab8942bdc2a repositoryMetadata: base64Readme: >- CiMgc3ByaW5nYm9vdC1qd3QgYXV0aG9yaXphdGlvbi1qdW5pdDUtbW9ja2l0bwoKIyMgUHJvamVjdCBEZXRhaWxzOgoKIyMjIyBKYXZhIDE3CiMjIyMgU3ByaW5nIEJvb3QgMy4xLjAsCiMjIyMgU3ByaW5nIERhdGEgSnBhCiMjIyMgTXlzcWwKIyMjIyBTcHJpbmcgU3RhcnRlciBTZWN1cml0eQojIyMjIEpzb24gV2ViIFRva2VuIChKV1QpCiMjIyMgTG9tYm9rCiMjIyMgc3ByaW5nZG9jLW9wZW5hcGktc3RhcnRlci13ZWJtdmMtdWkKIyMjIyBKdW5pdDUKIyMjIyBNb2NraXRvCiMjIyMgSmFjb2NvCiMjIyMgRGV2dG9vbHMKIyMjIyBTdGFydGVyIFdlYgoK readmeEtag: '"62e86dc73a70d1665ca108222c80e33fb4aa0c1a"' readmeLastModified: Thu, 22 Feb 2024 05:43:37 GMT repositoryId: 747440555 description: >- This application is built using Spring Boot, Java17, Spring Security using JWT authentication, MySQL, Spring Data Jpa, maven, Junit5 and mockito framework. created: '2024-01-23T23:40:17Z' updated: '2024-02-22T06:22:23Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: sandeepbegudem logo: https://avatars.githubusercontent.com/u/124546985?v=4 repoEtag: '"e2d790cac87bf2f1e91f807d9b624f28dd1e0c0c8f352c2ef4683ffbeeb2a582"' repoLastModified: Thu, 22 Feb 2024 06:22:23 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/goldenglorys/eden-api v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CjxhIGhyZWY9Imh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9sYXJhdmVsL2ZyYW1ld29yayI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L2wvbGFyYXZlbC9mcmFtZXdvcmsiIGFsdD0iTGljZW5zZSI+PC9hPgo8L3A+CgojIyBFZGVuIFdvcmsgU2FtcGxlCgpXZWxjb21lIHRvIHRoZSBHYXJkZW4gb2YgRWRlbiEKClRoaXMgd29yayBzYW1wbGUgaXMgYSBzaW1wbGUgQVBJIHdpdGggdmFyaW91cyBjb25zdW1hYmxlIGVuZHBvaW50cyB3aGljaCBhcmUgYXZhaWxhYmxlIHRocm91Z2ggdGhlIEFQSSBkb2N1bWVudGF0aW9uIHBhZ2UuCgpUaGUgQVBJIHNpbXBseSBoYXMgYnV0IG5vdCBsaW1pdGVkIHRvOgoKLSBBdXRvbWF0ZWQgVGVzdGluZy4KLSBBUEkgY2FjaGluZyBmb3IgdXAgdG8gMyBtaW51dGVzIG9uIGFueSBnZXQgcmVxdWVzdHMuCi0gRW1iZWRlZCAmJiBpbnRlcmFjdGl2ZSBBUEkgZG9jdW1lbnRhdGlvbi4gCi0gU2ltcGxpY2l0eSBieSBkZXNpZ24gaW4gaXQncyBvd24gbWluaW1hbCBsZXZlbC4KCiMjIEdldHRpbmcgU3RhcnRlZAotLS0tLS0tLS0tLS0tLS0KCiMjIyMgVmlhIENsb25pbmcgVGhlIFJlcG9zaXRvcnk6CgpbUEhQXShodHRwczovL3BocC5uZXQpIDguMCsgYW5kIFtDb21wb3Nlcl0oaHR0cHM6Ly9nZXRjb21wb3Nlci5vcmcpIHBsdXMgYSBkYXRhYnNlIChNeVNRTCBvciBQb3N0Z3Jlc1NRTCkgYXJlIHJlcXVpcmVkLgoKYGBgYmFzaAojIEdldCB0aGUgcHJvamVjdApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2dvbGRlbmdsb3J5cy9lZGVuLXdvcmstc2FtcGxlLmdpdAoKIyBDaGFuZ2UgZGlyZWN0b3J5CmNkIGVkZW4td29yay1zYW1wbGUKCiMgSW5zdGFsbCBDb21wb3NlciBkZXBlbmRlbmNpZXMKY29tcG9zZXIgaW5zdGFsbCBvciBjb21wb3NlciB1cGRhdGUKCiMgQ29weSAuZW52LmV4YW1wbGUgdG8gLmVudgpjcCAuZW52LmV4YW1wbGUgLmVudgoKIyBDcmVhdGUgYSBkYXRhYmFzZSAod2l0aCBteXNxbCBvciBwb3N0Z3Jlc3FsKQojIEFuZCB1cGRhdGUgLmVudiBmaWxlIHdpdGggZGF0YWJhc2UgY3JlZGVudGlhbHMKIyBEQl9DT05ORUNUSU9OPW15c3FsIE9SIHBnc3FsCiMgREJfSE9TVD0xMjcuMC4wLjEKIyBEQl9EQVRBQkFTRT1lZGVuCiMgREJfVVNFUk5BTUU9IyMjIwojIERCX1BBU1NXT1JEPSMjIyMKCiMgUnVuIHRoZSBkYXRhYmFzZSBtaWdyYXRpb24gYW5kIHRoZSBpbml0aWFsIHNlZWRpbmcgZGF0YSB1c2luZwpwaHAgYXJ0aXNhbiBtaWdyYXRlIC0tc2VlZAoKIyBHZW5lcmF0ZSBhcHBsaWNhdGlvbiBzZWN1cmUga2V5IChpbiAuZW52IGZpbGUpCnBocCBhcnRpc2FuIGtleTpnZW5lcmF0ZQoKIyBSdW4gdGhlIGFwcGxpY2F0aW9uIHVzaW5nCnBocCBhcnRpc2FuIHNlcnZlCmBgYAoKIyMgUnVuIFRlc3RzCgpUbyBydW4gdGhlIGF1dG9tYXRlZCB0ZXN0cywgaXNzdWUgdGhlIGNvbW1hbmQ6CgogICAgcGhwIGFydGlzYW4gdGVzdAoKVGhlIHRlc3QgZG9zZW4ndCBnZW5lcmF0ZSB0aGUgY292ZXJhZ2Ugc3RhdGlzdGljcywgZmVlbCBmcmVlIGFkZGluZyB0aGF0LgoKIyMgTGl2ZSBVUkxzCgotIFtBUEkgVVJMXShodHRwczovL2VkZW4tc2FtcGxlLWFwaS5oZXJva3VhcHAuY29tL2FwaS92MSk6IEhpdHMgdGhlIEFQSSBob21lL3dlbGNvbWUgZW5kcG9pbnQuCi0gW0FQSSBEb2N1bWVudGF0aW9uXShodHRwczovL2VkZW4tc2FtcGxlLWFwaS5oZXJva3VhcHAuY29tL2FwaS9kb2N1bWVudGF0aW9uKTogTGFuZHMgb24gdGhlIGVtYmVkZWQvaW50ZXJhdGljZSBBUEkgZG9jdW1lbnRhaW9uLgoKIyMgTGljZW5zZQoKVGhlIGNvZGUgaXMgb3Blbi1zb3VyY2VkLCBsaWNlbnNlZCB1bmRlciB0aGUgW01JVCBsaWNlbnNlXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkuCgogPGgyIGFsaWduPSJjZW50ZXIiPk1hZGUgcG9zc2libGUgYnkgPGI+UEhQPC9iPjxhIGhyZWY9Imh0dHBzOi8vbGFyYXZlbC5jb20iIHRhcmdldD0iX2JsYW5rIj48aW1nIHNyYz0iaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2xhcmF2ZWwvYXJ0L21hc3Rlci9sb2dvLWxvY2t1cC81JTIwU1ZHLzIlMjBDTVlLLzElMjBGdWxsJTIwQ29sb3IvbGFyYXZlbC1sb2dvbG9ja3VwLWNteWstcmVkLnN2ZyIgd2lkdGg9IjE1MCIgaGVpZ2h0PSI1MCI+PC9hPldpdGgg4p2k77iPPC9oMj4KCiBIYXBweSBDb2RpbmchIQ== readmeEtag: '"a162ffa3b31d7e6c1e2445f4e18ea82eedf76bb3"' readmeLastModified: Tue, 01 Mar 2022 21:18:17 GMT repositoryId: 463852136 description: A new day, another opportunity to be a step closer to "Paradise"! created: '2022-02-26T12:53:12Z' updated: '2024-03-31T19:52:44Z' language: PHP archived: false stars: 1 watchers: 1 forks: 0 owner: goldenglorys logo: https://avatars.githubusercontent.com/u/32324941?v=4 repoEtag: '"985e8a8cebd05f1e0f97f4c0f5d5396b3cb458e6c7af27740dcb724dfa4c734e"' repoLastModified: Sun, 31 Mar 2024 19:52:44 GMT foundInMaster: true category: - Testing - Server Implementations id: 63e8352e8255e8dae0c55b639d4a1fff oldLocations: - https://github.com/cloneofglory/eden-api - source: openapi3 tags repository: https://github.com/kaliszando/bs-api-specification v3: true id: b05e8787eea4278f92d72a0301793ae4 repositoryMetadata: base64Readme: >- PCEtLSBQUk9KRUNUIExPR08gLS0+CjxkaXYgYWxpZ249ImNlbnRlciI+CjxoMz48aW1nIHNyYz0iaHR0cHM6Ly9naXRodWIuY29tL0thbGlzemFuZG8vYnMtd2ViLWNsaWVudC9ibG9iL21haW4vc3JjL2Fzc2V0cy9pbWcvZmF2aWNvbi0xNngxNi5wbmciIGFsdD0iTG9nbyIgd2lkdGg9IjE2IiBoZWlnaHQ9IjE2Ij4gQnVnU3RhbGtlciBBUEk8L2gzPgo8L2Rpdj4KCkJ1Z1N0YWxrZXIgcHJvamVjdHM6CiogW2JzLWJ1c2luZXNzXShodHRwczovL2dpdGh1Yi5jb20vS2FsaXN6YW5kby9icy1idXNpbmVzcykgYmFja2VuZAoqIFticy13ZWItY2xpZW50XShodHRwczovL2dpdGh1Yi5jb20vS2FsaXN6YW5kby9icy13ZWItY2xpZW50KSBmcm9udGVuZAoqIFticy1hcGktc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL0thbGlzemFuZG8vYnMtYXBpLXNwZWNpZmljYXRpb24pIEFQSSBzcGVjaWZpY2F0aW9uIGFuZCBnZW5lcmF0aW9uIHRvb2wKCjwhLS0gQUJPVVQgVEhFIFBST0pFQ1QgLS0+CiMjIEFib3V0IFRoZSBQcm9qZWN0CgpUaGUgQVBJIHNwZWNpZmljYXRpb24gcHJvamVjdCBzZXJ2ZXMgYXMgdGhlIGJhY2tib25lIGZvciBib3RoIGZyb250ZW5kIGFuZCBiYWNrZW5kIGRldmVsb3BtZW50IGluIHRoZSBCdWdTdGFsa2VyIGFwcGxpY2F0aW9uLgpJdCBkZWZpbmVzIHRoZSBBUEkgZW5kcG9pbnRzIGFuZCBkYXRhIG1vZGVscywgcmVxdWlyZWQgZm9yIHNlYW1sZXNzIGNvbW11bmljYXRpb24gYmV0d2VlbiAKdGhlIGZyb250ZW5kIGFuZCBiYWNrZW5kIGFwcGxpY2F0aW9ucy4KCkluIGVzc2VuY2UsIHRoZSBBUEkgU3BlY2lmaWNhdGlvbiBwcm9qZWN0IG5vdCBvbmx5IGRvY3VtZW50cyB0aGUgZW5kcG9pbnRzIGJ1dCBhbHNvIHByb3ZpZGVzIGEgbWVjaGFuaXNtIAp0byBnZW5lcmF0ZSBpbXBsZW1lbnRhdGlvbnMgb2YgdGhlIHNwZWNpZmllZCBlbmRwb2ludHMsCmVuaGFuY2luZyBwcm9kdWN0aXZpdHkgYW5kIG1haW50YWluYWJpbGl0eSBvZiB0aGUgQnVnU3RhbGtlciBhcHBsaWNhdGlvbi4KCiMjIyBCdWlsdCBXaXRoCgotIFN3YWdnZXIgKyBPcGVuQVBJIDMKLSBbbmctb3BlbmFwaS1nZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9jeWNsb3Nwcm9qZWN0L25nLW9wZW5hcGktZ2VuKQoKPCEtLSBHRVRUSU5HIFNUQVJURUQgLS0+CiMjIEdldHRpbmcgU3RhcnRlZAoKSW4gb3JkZXIgdG8gZ2VuZXJhdGUgd2ViIHNlcnZpY2VzIGFuZCBtb2RlbHMgeW91IG5lZWQgdG8gaW5zdGFsbCBbbmctb3BlbmFwaS1nZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9jeWNsb3Nwcm9qZWN0L25nLW9wZW5hcGktZ2VuKS4KCiMjIyBQcmVyZXF1aXNpdGVzCgotIEphdmEgMTcKLSBNYXZlbgotIE5vZGUuanMKLSBucG0KLSBbbmctb3BlbmFwaS1nZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9jeWNsb3Nwcm9qZWN0L25nLW9wZW5hcGktZ2VuKQotIHB5dGhvbioKCiMjIyBNYXZlbiBwcm9maWxlcwoKLSBgYXBpLXRlbXBsYXRlLWJ1bmRsZXJgIHJ1bnMgYGFwaS1idW5kbGVyLnB5YCBiZWZvcmUgaW5zdGFsbCBwaGFzZS4gCkJ1bmRsZXIgbWVyZ2VzIGFsbCB5YW1sIG1vZGVscyBmcm9tIGBzcmMvbWFpbi9zY2hlbWFzYCBhbmQgZW5kcG9pbnRzIGZyb20gYHNyYy9tYWluL3NjaGVtYXNgCmludG8gc2luZ2xlIGBvcGVuYXBpLnlhbWxgIGZpbGUuCi0gYGJhY2tlbmQtY29kZS1nZW5gIGVuYWJsZXMgSmF2YSBqYXIgZmlsZSBnZW5lcmF0aW9uLgotIGB3ZWItY29kZS1nZW5gIHJ1bnMgYWZ0ZXIgaW5zdGFsbCBwaGFzZS4gCkl0IGdlbmVyYXRlcyBmcm9udGVuZCBtb2RlbHMgYW5kIGVuZHBvaW50IGltcGxlbWVudGF0aW9ucy4KCgojIyMgSW5zdGFsbGF0aW9uCgoxLiBDbG9uZSB0aGUgcmVwbwogICBgYGBzaAogICBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL0thbGlzemFuZG8vYnMtYXBpLXNwZWNpZmljYXRpb24uZ2l0CiAgIGBgYAoyLiBTZXQgb3V0cHV0IGRpcmVjdG9yeSBmb3IgZnJvbnRlZCBwcm9qZWN0IGluIGBwYWNrYWdlLmpzb25gCiAgIGBgYHNoCiAgICJnZW4td2ViIjogIm5nLW9wZW5hcGktZ2VuIC0tb3V0cHV0IC4uL2JzLXdlYi1jbGllbnQvc3JjL2FwcC9hcGkiCiAgIGBgYAozLiBCdWlsZCBBUEkgcHJvamVjdC4gCiAgIGBgYHNoCiAgIG12biBjbGVhbiBpbnN0YWxsCiAgIGBgYAoKCjwhLS0gQ09OVEFDVCAtLT4KIyMgQ29udGFjdAoKQWRhbSBLYWxpc3oga2FsaXN6YWRhbTk5K2RldkBnbWFpbC5jb20KCkxpbmtlZEluIFtAYWRhbS1rYWxpc3pdKGh0dHBzOi8vd3d3LmxpbmtlZGluLmNvbS9pbi9hZGFtLWthbGlzei8pCgpPdGhlciBsaW5rcyBbbGlua3RyLmVlL2thbGlzemFuZG9dKGh0dHBzOi8vbGlua3RyLmVlL2thbGlzemFuZG8pCgo8cCBhbGlnbj0icmlnaHQiPig8YSBocmVmPSIjLWJ1Z3N0YWxrZXItYXBpIj5iYWNrIHRvIHRvcDwvYT4pPC9wPgo= readmeEtag: '"bc9b6ef8e650e93884cc5149a86b8ee5a2901ab2"' readmeLastModified: Fri, 01 Nov 2024 17:21:36 GMT repositoryId: 492590343 description: >- [BugStalker] OpenAPI 3 specification and generation tool for bs-business and bs-web-client projects created: '2022-05-15T19:59:13Z' updated: '2026-01-24T20:07:06Z' language: YAML archived: false stars: 1 watchers: 1 forks: 0 owner: Kaliszando logo: https://avatars.githubusercontent.com/u/45283134?v=4 repoEtag: '"af260f6637cc034c2c74dd2385dddf52345f8ff464291b3cab48e02bb6961d9a"' repoLastModified: Sat, 24 Jan 2026 20:07:06 GMT foundInMaster: true category: Server Implementations - source: openapi3 tags repository: https://github.com/bbakla/openapi3-with-go v3: true id: dfea0397e3610cd874405cefdbe21ca0 repositoryMetadata: base64Readme: >- VGhpcyBpcyB0aGUgcmVwbyB0aGF0IEkgY3JlYXRlZCBmb3IgdGhpcyBbYXJ0aWNsZV0oaHR0cHM6Ly9tZWRpdW0uY29tL0BiYmFrbGEvb3Blbi1hcGktd2l0aC1nby1kNzVlYjNhZmFjMTkpLiBJdCBkZW1vbnN0cmF0ZXMgaG93IHRvIGdlbmVyYXRlIGJvaWxlcnBsYXRlIGNvZGUgaW4gR28gdXNpbmcgb3Blbi1hcGkgMy4K readmeEtag: '"897118d45c90ff19c0b2a490939ef521b425e58a"' readmeLastModified: Thu, 04 Jul 2024 20:19:26 GMT repositoryId: 809398146 description: open-api code generation in go created: '2024-06-02T15:23:03Z' updated: '2025-01-13T04:39:52Z' language: Go archived: false stars: 1 watchers: 1 forks: 1 owner: bbakla logo: https://avatars.githubusercontent.com/u/7812609?v=4 repoEtag: '"dea15ac03033ae45ff2a4858f7928d2128de8805f66c5c43d79aab8edad81c6d"' repoLastModified: Mon, 13 Jan 2025 04:39:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/flexbase-eng/openapi-generator v3: true id: b508ca585e247699e3128b803debe3a2 repositoryMetadata: base64Readme: >- # openapi-generator

[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=flexbase-eng_openapi-generator&metric=coverage)](https://sonarcloud.io/summary/new_code?id=flexbase-eng_openapi-generator)
[![Quality Gate Status](https://sonarcloud.io/api/project_badges/measure?project=flexbase-eng_openapi-generator&metric=alert_status)](https://sonarcloud.io/summary/new_code?id=flexbase-eng_openapi-generator)

[OpenAPI](https://www.openapis.org/) code generator.

## Getting started

```
yarn add @flexbase/openapi-generator --dev
```

or

```
npm i @flexbase/openapi-generator -D
```

## Usage

```
openapi-generator -i <openapispec>.yaml -o . -t <template>.hbs
```

| Option                   | Argument                                                   | Description                                                          |
| ------------------------ | ---------------------------------------------------------- | -------------------------------------------------------------------- |
| `--include`              | [glob](<https://en.wikipedia.org/wiki/Glob_(programming)>) | Specifies the glob pattern for files to parse                        |
| `--sharedTemplates`      | [glob](<https://en.wikipedia.org/wiki/Glob_(programming)>) | Specifies the glob pattern for shared templates                      |
| `--config`               | file                                                       | Specify a configuration to use. Defaults to `.openapigenerator.json` |
| `--no-prettier`          |                                                            | Disable prettier                                                     |
| `--no-skipempty`         |                                                            | Generate empty files                                                 |
| `--no-tags`              |                                                            | Disable organization by tags                                         |
| `-d` or `--debug [path]` |                                                            | Output the internal representation                                   |

## Configuration

Example configuration

```json
{
  "include": ["./tests/data/*.yaml"],
  "sharedTemplates": ["./templates/server/*.hbs"],
  "generate": {
    "router.ts": {
      "target": "./output/{api}/router.ts",
      "template": "./templates/server/router.hbs"
    },
    "index": {
      "target": "./output/{api}/{name}/generated/{name}.ts",
      "template": "./templates/server/index.hbs"
    },
    "routes": {
      "target": "./output/{api}/{name}/generated/{name}.routes.ts",
      "template": "./templates/server/routes.hbs"
    },
    "models": {
      "target": "./output/{api}/{name}/generated/{name}.models.ts",
      "template": "./templates/server/models.hbs"
    },
    "validations": {
      "target": "./output/{api}/{name}/generated/{name}.validations.ts",
      "template": "./templates/server/validations.hbs"
    }
  },
  "prettier": true,
  "tags": true,
  "debugPath": "./output/debug/{api}/"
}
```

## Templates

Below are some example [handlebars](https://handlebarsjs.com/) templates to generate a typescript output file

### `models.hbs` template

https://github.com/flexbase-eng/openapi-generator/blob/main/templates/server/models.hbs

### `model.declaration.hbs` partial template

https://github.com/flexbase-eng/openapi-generator/blob/main/templates/server/model.declaration.hbs

### `model.expression.hbs` partial template

https://github.com/flexbase-eng/openapi-generator/blob/main/templates/server/model.expression.hbs

## Example

Generating the petstore openapi spec by running the following:

```
openapi-generator
```

or

```
yarn start
```

will generate code under `./output/swagger-petstore-openapi-3-0/`

### Output `{output}/pet/generated/pet.routes.ts`

```ts
/* eslint-disable */
/* ------------------------------------------------------------
 File auto-generated

 @summary pet
 @description Everything about your Pets
 @version 1.0.17
------------------------------------------------------------ */

import * as middleware from '@middleware/index.js';
import * as utilities from '@shared/utilities/openapi.utilities.js';
import * as models from './pet.models.js';
import * as validators from './pet.validations.js';
import * as handlers from '@modules/swagger-petstore-openapi-3-0/routes/pet/pet.handlers.js';
import { Swagger_Petstore_OpenAPI_3_0Router } from '@modules/swagger-petstore-openapi-3-0/routes/router.js';

/**
 * @summary Update an existing pet
 * @description Update an existing pet by Id
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.put<object, object, models.PetModel | models.PetModel | models.PetModel>(
  '/pet',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware(['application/json', 'application/xml', 'application/x-www-form-urlencoded']),
  async (ctx, next) => {
    let body;
    if (ctx.request.type === 'application/json') {
      body = utilities.handleValidation<models.PetModel>(ctx.body, validators.PetModelValidator, utilities.handleRequestValidatorErrors);
    } else if (ctx.request.type === 'application/xml') {
      body = utilities.handleValidation<models.PetModel>(ctx.body, validators.PetModelValidator, utilities.handleRequestValidatorErrors);
    } else {
      body = utilities.handleValidation<models.PetModel>(ctx.body, validators.PetModelValidator, utilities.handleRequestValidatorErrors);
    }
    const response: models.updatePetResponse = await handlers.updatePet(ctx, body);

    utilities.serializeResponse(ctx, response, ['application/xml', 'application/json']);

    await next();
  },
);

/**
 * @summary Add a new pet to the store
 * @description Add a new pet to the store
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.post<object, object, models.PetModel | models.PetModel | models.PetModel>(
  '/pet',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware(['application/json', 'application/xml', 'application/x-www-form-urlencoded']),
  async (ctx, next) => {
    let body;
    if (ctx.request.type === 'application/json') {
      body = utilities.handleValidation<models.PetModel>(ctx.body, validators.PetModelValidator, utilities.handleRequestValidatorErrors);
    } else if (ctx.request.type === 'application/xml') {
      body = utilities.handleValidation<models.PetModel>(ctx.body, validators.PetModelValidator, utilities.handleRequestValidatorErrors);
    } else {
      body = utilities.handleValidation<models.PetModel>(ctx.body, validators.PetModelValidator, utilities.handleRequestValidatorErrors);
    }
    const response: models.addPetResponse = await handlers.addPet(ctx, body);

    utilities.serializeResponse(ctx, response, ['application/xml', 'application/json']);

    await next();
  },
);

/**
 * @summary Find pet by ID
 * @description Returns a single pet
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.get<object, object>(
  '/pet/:petId',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware([]),
  async (ctx, next) => {
    const params = utilities.handleValidation<models.getPetByIdPathParameter>(
      ctx.params,
      validators.getPetByIdPathParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const response: models.getPetByIdResponse = await handlers.getPetById(ctx, params);

    utilities.serializeResponse(ctx, response, ['application/xml', 'application/json']);

    await next();
  },
);

/**
 * @summary Updates a pet in the store with form data
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.post<object, object>(
  '/pet/:petId',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware([]),
  async (ctx, next) => {
    const params = utilities.handleValidation<models.updatePetWithFormPathParameter>(
      ctx.params,
      validators.updatePetWithFormPathParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const query = utilities.handleValidation<models.updatePetWithFormQueryParameter>(
      ctx.request.query,
      validators.updatePetWithFormQueryParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const response: models.updatePetWithFormResponse = await handlers.updatePetWithForm(ctx, params, query);

    utilities.serializeResponse(ctx, response, []);

    await next();
  },
);

/**
 * @summary Deletes a pet
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.delete<object, object>(
  '/pet/:petId',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware([]),
  async (ctx, next) => {
    const params = utilities.handleValidation<models.deletePetPathParameter>(
      ctx.params,
      validators.deletePetPathParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const headers = utilities.handleValidation<models.deletePetHeaderParameter>(
      ctx.request.headers,
      validators.deletePetHeaderParameterValidator,
      utilities.handleRequestValidatorErrors,
    );
    const response: models.deletePetResponse = await handlers.deletePet(ctx, params, headers);

    utilities.serializeResponse(ctx, response, []);

    await next();
  },
);

/**
 * @summary uploads an image
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.post<object, object, models.uploadFileRequestObject>(
  '/pet/:petId/uploadImage',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware(['application/octet-stream']),
  async (ctx, next) => {
    const body = utilities.handleValidation<models.uploadFileRequestObject>(
      ctx.body,
      validators.uploadFileRequestObjectValidator,
      utilities.handleRequestValidatorErrors,
    );

    const params = utilities.handleValidation<models.uploadFilePathParameter>(
      ctx.params,
      validators.uploadFilePathParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const query = utilities.handleValidation<models.uploadFileQueryParameter>(
      ctx.request.query,
      validators.uploadFileQueryParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const response: models.uploadFileResponse = await handlers.uploadFile(ctx, body, params, query);

    utilities.serializeResponse(ctx, response, ['application/json']);

    await next();
  },
);

/**
 * @summary Finds Pets by status
 * @description Multiple status values can be provided with comma separated strings
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.get<object, object>(
  '/pet/findByStatus',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware([]),
  async (ctx, next) => {
    const query = utilities.handleValidation<models.findPetsByStatusQueryParameter>(
      ctx.request.query,
      validators.findPetsByStatusQueryParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const response: models.findPetsByStatusResponse = await handlers.findPetsByStatus(ctx, query);

    utilities.serializeResponse(ctx, response, ['application/xml', 'application/json']);

    await next();
  },
);

/**
 * @summary Finds Pets by tags
 * @description Multiple tags can be provided with comma separated strings. Use tag1, tag2, tag3 for testing.
 * @remarks required scopes: []
 */
Swagger_Petstore_OpenAPI_3_0Router.get<object, object>(
  '/pet/findByTags',
  middleware.routerAuthMiddleware([]),
  middleware.bodyParserMiddleware([]),
  async (ctx, next) => {
    const query = utilities.handleValidation<models.findPetsByTagsQueryParameter>(
      ctx.request.query,
      validators.findPetsByTagsQueryParameterValidator,
      utilities.handleRequestValidatorErrors,
    );

    const response: models.findPetsByTagsResponse = await handlers.findPetsByTags(ctx, query);

    utilities.serializeResponse(ctx, response, ['application/xml', 'application/json']);

    await next();
  },
);
```
 readmeEtag: '"e2d78fb1e6df34c6804e4a820354d0e5852d884e"' readmeLastModified: Fri, 19 Jul 2024 18:23:14 GMT repositoryId: 564335104 description: OpenAPI v3 code generator created: '2022-11-10T13:53:10Z' updated: '2025-09-04T17:09:27Z' language: TypeScript archived: false stars: 1 watchers: 9 forks: 0 owner: flexbase-eng logo: https://avatars.githubusercontent.com/u/68292078?v=4 license: MIT repoEtag: '"2a6f511bd11204d3d63c0bb978165278cb65d0755806143130feb60552803e53"' repoLastModified: Thu, 04 Sep 2025 17:09:27 GMT category: Parsers foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/gcatanese/openapi-native-mock-server v3: true v3_1: true id: 1fe45e2e9d9a8f115bd2328a671d4c7a repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIE5hdGl2ZSBNb2NrIFNlcnZlcgoKTGlnaHR3ZWlnaHQgbW9jayBzZXJ2ZXIgY3JlYXRlZCBmcm9tIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbi4gCgpXaHkgKipuYXRpdmUqKj8gQmVjYXVzZSB0aGUgcmVxdWVzdC1yZXNwb25zZSBpbnRlcmFjdGlvbnMgYXJlIGRlZmluZWQgd2l0aGluIHRoZSBPcGVuQVBJIGZpbGUsIHdpdGhvdXQKdGhlIG5lZWQgZm9yIGFuIGFkZGl0aW9uYWwgYXBwbGljYXRpb24gb3Igc3RvcmFnZSAoYW5kIHRoZXJlZm9yZSBjb21wbGV4aXR5KS4gTm8gZXh0ZXJuYWwgZGVwZW5kZW5jaWVzLgoKVGhlIHJlcXVlc3QtcmVzcG9uc2UgaW50ZXJhY3Rpb25zIChha2EgdGhlIGNvbnRyYWN0cykgYXJlIGRlZmluZWQgYnkgcGFyc2luZyB0aGUgT3BlbkFQSSBleGFtcGxlcywgc2VlIFttYXBwaW5nIHN0cmF0ZWdpZXNdKCNob3ctaXQtd29ya3MpLgpBIGZhbGxiYWNrIHJlc3BvbnNlIGlzIGdlbmVyYXRlZCB0byBlbnN1cmUgZXZlcnkgcmVxdWVzdCByZWNlaXZlcyBhbHdheXMgYSByZXNwb25zZS4KCkl0IGlzOgoqICoqbGlnaHR3ZWlnaHQqKjogaXQgcnVucyBpbiBhIHNpbmdsZSAoc21hbGwpIGNvbnRhaW5lcgoqICoqZmFzdCoqOiBpdCBjcmVhdGVzIGJlaGluZCB0aGUgc2NlbmVzIGEgR28gc2VydmljZSB3aXRoIGEgbWluaW1hbCBmb290cHJpbnQKKiAqKnNpbXBsZSoqOiBpdCBkaXNwbGF5cyB0aGUgbW9jayBleHBlY3RhdGlvbnMgb24gdGhlIGluZGV4IHBhZ2UuIEluc3BlY3QgdGhlIEpTT04gcGF5bG9hZCBvZiBhbGwgbW9ja2VkIHJlc3BvbnNlcy4KCiMjIFVzZSBjYXNlcwoKSXQgY2FuIGJlIHVzZWQgaW4gZGlmZmVyZW50IHNjZW5hcmlvczoKKiB5b3UgYXJlIGRldmVsb3BpbmcgYW4gQVBJIFNESyBhbmQgbmVlZCB0byBpbXBsZW1lbnQgc29saWQgaW50ZWdyYXRpb24gdGVzdGluZzogdHJ5IGl0IG91dCB3aXRoIFtUZXN0Q29udGFpbmVyc10oaHR0cHM6Ly9naXRodWIuY29tL2djYXRhbmVzZS9vcGVuYXBpLXRlc3Rjb250YWluZXJzKS4KKiB5b3UgYXJlIGJ1aWxkaW5nIEFQSXMgYW5kIHdhbnQgdG8gaGVscCB5b3VyIEFQSSBjb25zdW1lcnMgZHVyaW5nIHRoZSBpbnRlZ3JhdGlvbjogbWFrZSBhIG1vY2sgc2VydmVyICh0aGF0IGltcGxlbWVudHMgdGhlIEFQSSBjb250cmFjdCkgcmVhZGlseSBhdmFpbGFibGUuCiogeW91IGFyZSBkZXZlbG9waW5nIGFuIGFwcGxpY2F0aW9uIHRoYXQgdXNlcyBhIHRoaXJkLXBhcnR5IEFQSSBkZXBlbmRlbmN5OiBtb2NrIHRoZSBBUEkgdG8gbWFrZSB5b3VyIGRldmVsb3BtZW50IGZhc3RlciBhbmQgbW9yZSBlZmZpY2llbnQuCgojIyBVc2FnZQoKIyMjIEJ1aWxkIGFuZCBydW4gd2l0aCBEb2NrZXIKCkJ1aWxkIHRoZSBtb2NrIHNlcnZlciBmcm9tIHlvdXIgT3BlbkFQSSBmaWxlOgpgYGBkb2NrZXIKZG9ja2VyIGJ1aWxkIC0tYnVpbGQtYXJnIG9wZW5hcGlmaWxlPS9wYXRoL29wZW5hcGlGaWxlLnlhbWwgLXQgb3BlbmFwaS1uYXRpdmUtbW9jay1zZXJ2ZXIgLgpgYGAKClJ1biB0aGUgY29udGFpbmVyOgpgYGBkb2NrZXIKZG9ja2VyIHJ1biAtLXJtIC1kIC1wIDgwODA6ODA4MCAtLW5hbWUgb3BlbmFwaS1uYXRpdmUtbW9jay1zZXJ2ZXIgb3BlbmFwaS1uYXRpdmUtbW9jay1zZXJ2ZXIKYGBgCgpQb2ludCB5b3VyIEFQSSByZXF1ZXN0cyB0byB0aGUgbW9jayBzZXJ2ZXI6CmBgYHNoZWxsCmN1cmwgLVggUE9TVCAtSCAiQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uIiBcCiAtZCAneyJuYW1lIjoiYWJjIiwiY291bnRyeSI6ImFiYyJ9JyBcCiAgIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC97eW91cl9hcGlfZW5kcG9pbnR9CmBgYAoKIyMjIEJ1aWxkIGFuZCBydW4gZnJvbSBzb3VyY2UKCkJ1aWxkIHRoZSBtb2NrIHNlcnZlciAoKipOb3RlKio6IHJlcXVpcmVzIEphdmEgMTEpCmBgYHNoZWxsCm12biBwYWNrYWdlCmBgYAoKR2VuZXJhdGUgdGhlIG1vY2sgc2VydmVyIHBhc3NpbmcgdGhlIHBhdGggb2YgdGhlIE9wZW5BUEkgZmlsZSAoaS5lLiBgLWkgb3BlbmFwaS90bXAvb3BlbmFwaS55YW1sYCkgYW5kIHRoZQpmb2xkZXIgdGhhdCB3aWxsIGNvbnRhaW4gdGhlIG1vY2sgc2VydmVyIHNvdXJjZSBjb2RlIChpLmUuIGAtbyBvcGVuYXBpL3RtcC9nby1zZXJ2ZXJgKToKYGBgc2hlbGwKamF2YSAtY3AgdGFyZ2V0L29wZW5hcGktbmF0aXZlLW1vY2stc2VydmVyLmphcjpvcGVuYXBpL2NsaS9vcGVuYXBpLWdlbmVyYXRvci1jbGkuamFyIFwKIG9yZy5vcGVuYXBpdG9vbHMuY29kZWdlbi5PcGVuQVBJR2VuZXJhdG9yIGdlbmVyYXRlIC1nIGNvbS50d2Vlc2t5LmNsb3VkdG9vbHMuY29kZWdlbi5OYXRpdmVNb2NrU2VydmVyQ29kZWdlbiBcCiAgLWkgb3BlbmFwaS90bXAvb3BlbmFwaS55YW1sIC1vIG9wZW5hcGkvdG1wL2dvLXNlcnZlcgpgYGAKClJ1biB0aGUgbW9jayBzZXJ2ZXIgKCoqTm90ZSoqOiByZXF1aXJlcyBHbyAxLjE5KToKCmBgYHNoZWxsCmNkIG9wZW5hcGkvdG1wL2dvLXNlcnZlcgpnbyBtb2QgdGlkeQpnbyBydW4gLgpgYGAKClBvaW50IHlvdXIgQVBJIHJlcXVlc3RzIHRvIHRoZSBtb2NrIHNlcnZlcjoKYGBgc2hlbGwKY3VybCAtWCBQT1NUIC1IICJDb250ZW50LVR5cGU6IGFwcGxpY2F0aW9uL2pzb24iIFwKIC1kICd7Im5hbWUiOiJhYmMiLCJjb3VudHJ5IjoiYWJjIn0nIFwKICAgaHR0cDovL2xvY2FsaG9zdDo4MDgwL3t5b3VyX2FwaV9lbmRwb2ludH0KYGBgCgojIyMgTW9jayBzZXJ2ZXIgYnJvd3NlcgoKQWNjZXNzIHRoZSBpbmRleCBwYWdlIHRvIHZpZXcgdGhlIE9wZW5BUEkgZmlsZSAod2hpY2ggdGhlIG1vY2sgc2VydmVyIGhhcyBiZWVuIGNyZWF0ZWQgZnJvbSkgCmFuZCB0aGUgbW9jayBleHBlY3RhdGlvbnM6CmBgYHNoZWxsCmN1cmwgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2luZGV4LwpgYGAKVGhlIGBpbmRleGAgcGFnZSBwcm92aWRlcyBhIHZpZXcgb2YgdGhlIGNvbnRyYWN0IGludGVyYWN0aW9ucywgbWFraW5nIGNsZWFyIHRoZSBleHBlY3RhdGlvbnMuCkNoZWNrIGl0IG91dC4KCiFbTW9jayBzZXJ2ZXIgYnJvd3Nlcl0obW9jay1zZXJ2ZXIucG5nKQoKCiMjIEhvdyBpdCB3b3JrcwoKVGhlIG1vZHVsZSB1c2VzIHRoZSBbT3BlbkFQSSBHZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IpIHRvIGdlbmVyYXRlIGEgbGlnaHR3ZWlnaHQgCm1vY2sgc2VydmVyIGJhc2VkIG9uIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24uIFRoZSByZXF1ZXN0IGFuZCByZXNwb25zZSBleGFtcGxlcyBhcmUgZm91bmQgYW5kIG1hdGNoZWQgdG8gZGVmaW5lIHRoZSAKaW50ZXJhY3Rpb25zIChjb250cmFjdCkgYmV0d2VlbiB0aGUgY29uc3VtZXIgYW5kIHRoZSBwcm9kdWNlciBvZiB0aGUgQVBJLiAgCgpUaGUgZm9sbG93aW5nIHN0cmF0ZWdpZXMgYXJlIGFwcGxpZWQgKGluIG9yZGVyKSB0byBtYXRjaCByZXF1ZXN0cyB3aXRoIHJlc3BvbnNlczoKKiAqKm1hdGNoIGJ5IENvbnRyYWN0IHRhZyoqOiBkbyB0aGUgZXhhbXBsZXMgdXNlIHRoZSB2ZW5kb3IgZXh0ZW5zaW9uIGB4LWNvbnRyYWN0LWlkYD8gSW4gdGhpcyBjYXNlLCBtYXRjaCBhIHJlcXVlc3Qgd2l0aCBhIHJlc3BvbnNlIGV4YW1wbGUgdGhhdCBoYXMgdGhlIHNhbWUgdmFsdWUKKiAqKm1hdGNoIGJ5IFJlZiBuYW1lKio6IG1hdGNoIGAkcmVmYCByZXF1ZXN0IGV4YW1wbGUgd2l0aCBhIGNvcnJlc3BvbmRpbmcgYCRyZWZgIHJlc3BvbnNlIGV4YW1wbGUsIGZvciBleGFtcGxlIApgY3JlYXRlLXVzZXJzLWV4YW1wbGVgIHdvdWxkIG1hdGNoIGBjcmVhdGUtdXNlcnMtZXhhbXBsZS0yMDBgIHRvIGRlZmluZSBhIHN1Y2Nlc3NmdWwgYDIwMGAgc2NlbmFyaW8KKiAqKm1hdGNoIGJ5IEV4YW1wbGUgbmFtZSoqOiBmaW5kIHJlcXVlc3QgYW5kIHJlc3BvbnNlIGV4YW1wbGVzIHRoYXQgaGF2ZSB0aGUgc2FtZSBuYW1lCiogKipnZW5lcmF0ZSBmcm9tIFNjaGVtYSoqOiBmYWxsYmFjayBzdHJhdGVneSAod2hlbiBubyBtYXRjaGluZyBpcyBmb3VuZCk6IGdlbmVyYXRlIHRoZSByZXNwb25zZSBmcm9tIHRoZSBTY2hlbWEgYW5kCmVuc3VyZSBldmVyeSByZXF1ZXN0IGhhcyBhdCBsZWFzdCBhIHJlc3BvbnNlLgoKCiMjIFJlZmVyZW5jZXMKCkNoZWNrIG91dCB0aGUgW0NvbnRyYWN0IFRlc3Rpbmcgd2l0aCBPcGVuQVBJXShodHRwczovL2JlcHBlY2F0YW5lc2UuaGFzaG5vZGUuZGV2L2NvbnRyYWN0LXRlc3Rpbmctd2l0aC1vcGVuYXBpKSB0byB1bmRlcnN0YW5kIGNoYWxsZW5nZXMgYW5kIG9wcG9ydHVuaXRpZXMgb2YgQ29udHJhY3QgVGVzdGluZyB3aXRoIHRoZSBPcGVuQVBJIHN0YW5kYXJkLgoK readmeEtag: '"3b26303f7de7b5d16b3a80424eb517cbb516d402"' readmeLastModified: Wed, 04 Dec 2024 19:48:37 GMT repositoryId: 710680390 description: >- Generate mock responses based on your OpenAPI specification. No dependencies. Lightweight, fast, simple. created: '2023-10-27T07:58:04Z' updated: '2025-03-30T13:40:36Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: gcatanese logo: https://avatars.githubusercontent.com/u/1771700?v=4 license: Apache-2.0 repoEtag: '"3a982c3267251ac10edc100f8a64f4f20c69a24cf454e8167c320f4f4fad861a"' repoLastModified: Sun, 30 Mar 2025 13:40:36 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/uttesh/exude-api v3: true repositoryMetadata: base64Readme: >- IyA8aW1nIHNyYz0iaHR0cDovL3V0dGVzaC5jb20vZXh1ZGUtYXBpL2ltZy9sb2dvLnBuZyIgYWx0PSJkcmF3aW5nIiB3aWR0aD0iODAiLz4gIEV4dWRlLUFQSSA8c3VwIHN0eWxlPSJjb2xvcjpyZWQiPmJldGE8L3N1cD4KCkV4dWRlIEFQSSBpcyBhbiBPcGVuIFNvdXJjZSBwcm9qZWN0LCBJdCBpcyB1c2VkIGZvciB0aGUgcHJpbWFyeSB3YXlzIGZvciBmaWx0ZXJpbmcgdGhlIHN0b3BwaW5nLCBzdGVtbWluZyB3b3JkcyBmcm9tIHRoZSB0ZXh0IGRhdGEuIFRoaXMgQVBJIGlzIGluIGEgdmVyeSBiYXNpYyBsZXZlbCBvZiBkZXZlbG9wbWVudCBuZWVkIHRvIHdvcmsgb24gZm9yIGxhdGVyIGNoYW5nZXMuCgpFeHVkZSBBUEkgdXNlcyB0aGUgaW4taG91c2UgRXh1ZGUgbGlicmFyeSBmb3IgdGhlIGZpbHRlcmluZyBvZiB0aGUgZGF0YSBFeHVkZS1saWJyYXJ5IDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS91dHRlc2gvZXh1ZGUiIHRhcmdldD0iX2JsYW5rIj5SZXBvc2l0b3J5PC9hPgoKRGVtbzogaHR0cHM6Ly9leHVkZS5oZXJva3VhcHAuY29tLyAoaXRzIHRoZSBzYW1wbGUgc2VydmVyKQojIyBGZWF0dXJlcwoKKiBGaWx0ZXIgc3RvcHBpbmcgd29yZHMgZnJvbSBnaXZlbiB0ZXh0L2ZpbGUvbGluay4KKiBGaWx0ZXIgc3RlbW1pbmcgd29yZHMgZnJvbSBnaXZlbiB0ZXh0L2ZpbGUvbGluay4KKiBHZXQgc3dlYXIgd29yZHMgZnJvbSBnaXZlbiB0ZXh0L2ZpbGUvbGluawoKIyMgT3BlbkFQSS1VUkwKCk9wZW4gQVBJIEpTT046IGh0dHA6Ly91dHRlc2guY29tL2V4dWRlLWFwaS9leHVkZS1zZXJ2aWNlLW9wZW4tYXBpLmpzb24KCk9wZW4gQVBJIFlNTDogaHR0cDovL3V0dGVzaC5jb20vZXh1ZGUtYXBpL2V4dWRlLXNlcnZpY2Utb3Blbi1hcGkueW1sCgpIZXJ1a28gU3BlYyBVUkwgOiBodHRwczovL2V4dWRlLWFwaS5oZXJva3VhcHAuY29tL3N3YWdnZXItdWkvaW5kZXguaHRtbD91cmw9L3YzL2FwaS1kb2NzCgojIyBIb3cgdG8gVXNlCgpFaXRoZXIgdXNlIHRoZSBIZXJ1a28gdXJsIG9yIGRlcGxveSBsb2NhbGx5IGJ5IHVzaW5nIHRoZSBkb2NrZXIgdG8gaGl0IHRoZSBBUEksIGZvciBtb3JlIHRlY2huaWNhbCBkZXRhaWxzIG9uIHRoZQpBUEkgc2VlIGV4dWRlLWFwaSBkb2M6IDxhIGhyZWY9Imh0dHA6Ly91dHRlc2guY29tL2V4dWRlLWFwaS9hcGktZXhwbG9yZXIvIiB0YXJnZXQ9Il9ibGFuayI+IERvY3VtZW50YXRpb24gPC9hPgoKUG9zdG1hbiBleGVjdXRpb24gc2FtcGxlOgoKPGZpZ3VyZSBjbGFzcz0idmlkZW9fY29udGFpbmVyIj4KICA8dmlkZW8gY29udHJvbHM9InRydWUiIGFsbG93ZnVsbHNjcmVlbj0idHJ1ZSIgcG9zdGVyPSJodHRwOi8vdXR0ZXNoLmNvbS9leHVkZS1hcGkvaW1nL2xvZ28ucG5nIj4KICAgIDxzb3VyY2Ugc3JjPSJodHRwOi8vdXR0ZXNoLmNvbS9leHVkZS1hcGkvZXh1ZGUtYXBpLXBvc3RtYW4tc2FtcGxlLXZpZW8ud2VibSIgdHlwZT0idmlkZW8vd2VibSI+CiAgPC92aWRlbz4KPC9maWd1cmU+CgoK readmeEtag: '"dcf27cffedaeb14e7ca00d85a95a502d18a3a7fb"' readmeLastModified: Tue, 10 Mar 2020 18:20:35 GMT repositoryId: 245419070 description: >- Exude API provide the service to filter the stopping,swear words on the provided file,data and link and it uses the in house exude jar to filter the data created: '2020-03-06T12:52:40Z' updated: '2020-06-21T04:43:16Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: uttesh logo: https://avatars.githubusercontent.com/u/2218611?v=4 license: Apache-2.0 repoEtag: '"a61513a030c8bb63ae55fbe2719ac4a7ffa8acdb5a108b38e3c3cc455d2df7ac"' repoLastModified: Sun, 21 Jun 2020 04:43:16 GMT foundInMaster: true category: Testing id: 48c791c3f16d34ec574327043da3d1ad - source: openapi3 tags repository: https://github.com/osevohe/oc-p7-api_rest v3: true repositoryMetadata: base64Readme: >- IyBCaWxlTW8gQVBJCgpPcGVuQ2xhc3NSb29tcyA3dGggUHJvamVjdApCdWlsZCBhbiBBUEkgUkVTVCB3aXRoIFN5bWZvbnkKClshW1NvbmFyQ2xvdWRdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9pbWFnZXMvcHJvamVjdF9iYWRnZXMvc29uYXJjbG91ZC13aGl0ZS5zdmcpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPU9TRXZvaGVfT0MtUDctQVBJX1JFU1QpCgpbIVtNYWludGFpbmFiaWxpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1PU0V2b2hlX09DLVA3LUFQSV9SRVNUJm1ldHJpYz1zcWFsZV9yYXRpbmcpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPU9TRXZvaGVfT0MtUDctQVBJX1JFU1QpClshW1F1YWxpdHkgR2F0ZSBTdGF0dXNdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PU9TRXZvaGVfT0MtUDctQVBJX1JFU1QmbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9T1NFdm9oZV9PQy1QNy1BUElfUkVTVCkKWyFbU2VjdXJpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1PU0V2b2hlX09DLVA3LUFQSV9SRVNUJm1ldHJpYz1zZWN1cml0eV9yYXRpbmcpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPU9TRXZvaGVfT0MtUDctQVBJX1JFU1QpClshW0J1Z3NdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PU9TRXZvaGVfT0MtUDctQVBJX1JFU1QmbWV0cmljPWJ1Z3MpXShodHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPU9TRXZvaGVfT0MtUDctQVBJX1JFU1QpClshW0NvZGUgU21lbGxzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1PU0V2b2hlX09DLVA3LUFQSV9SRVNUJm1ldHJpYz1jb2RlX3NtZWxscyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9T1NFdm9oZV9PQy1QNy1BUElfUkVTVCkKWyFbRHVwbGljYXRlZCBMaW5lcyAoJSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PU9TRXZvaGVfT0MtUDctQVBJX1JFU1QmbWV0cmljPWR1cGxpY2F0ZWRfbGluZXNfZGVuc2l0eSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9T1NFdm9oZV9PQy1QNy1BUElfUkVTVCkKCkNvZGUgQ2xpbWF0ZSAgClshW01haW50YWluYWJpbGl0eV0oaHR0cHM6Ly9hcGkuY29kZWNsaW1hdGUuY29tL3YxL2JhZGdlcy8zYzEwZWE4NmZmN2JhOTRiZjAxMy9tYWludGFpbmFiaWxpdHkpXShodHRwczovL2NvZGVjbGltYXRlLmNvbS9naXRodWIvT1NFdm9oZS9PQy1QNy1BUElfUkVTVC9tYWludGFpbmFiaWxpdHkpCgojIyBQcmVyZXF1aXNpdGUKKiBBIFdlYiBTZXJ2ZXIgKEFwYWNoZSwgTmdpbnguLi4pCiogUGhQIDcuMwoqIENvbXBvc2VyCiogQSBkYXRhYmFzZSBlbmdpbmUgKE1hcmlhREIsIE15U3FsLCBQb3N0Z3JlU1FMLi4uKQoqIFtPcGVuU1NMIGNvbW1hbmQgbGluZSB0b29sXShodHRwczovL3d3dy5vcGVuc3NsLm9yZy9kb2NzL21hbjEuMC4yL21hbjEvb3BlbnNzbC5odG1sKSBvciBhbHJlYWR5IGdlbmVyYXRlZCBTU0ggS2V5cwoKIyMgSW5zdGFsbGF0aW9uCiogQ2xvbmUgb3IgZG93bmxvYWQgdGhlIHByb2plY3QKKiBHbyB0byBwcm9qZWN0IGZvbGRlciBpbiBhIHRlcm1pbmFsCiogVHlwZSBgY29tcG9zZXIgaW5zdGFsbGAKKiBDb25maWd1cmUgYSBuZXcgaG9zdCBpbiB5b3VyIHdlYiBzZXJ2ZXIgd2l0aCBgcHVibGljL2AgZm9sZGVyIGFzIERvY3VtZW50Um9vdAoKIyMjIERhdGFiYXNlIHNldHVwCiogQ29weSBgLmVudmAgdG8gYC5lbnYubG9jYWxgIGFuZCBlZGl0IGRhdGFiYXNlIHBhcmFtZXRlcnMKKiBJbml0aWFsaXplIHRoZSBkYXRhYmFzZSA6IAogICogYHBocCBiaW4vY29uc29sZSBkb2N0cmluZTpkYXRhYmFzZTpjcmVhdGVgCiAgKiBgcGhwIGJpbi9jb25zb2xlIG1ha2U6bWlncmF0aW9uYAogICogYHBocCBiaW4vY29uc29sZSBkb2N0cmluZTptaWdyYXRpb25zOm1pZ3JhdGVgCiAgCiMjIyBKV1QgQXV0aGVudGljYXRpb24gU2V0dXAKKiBHZW5lcmF0ZSBwcml2YXRlIGtleSA6YG9wZW5zc2wgZ2VucnNhIC1vdXQgcHJpdmF0ZS5wZW0gLWFlczI1NiA0MDk2YAoqIEdlbmVyYXRlIHB1YmxpYyBrZXkgOiBgb3BlbnNzbCByc2EgLXB1Ym91dCAtaW4gcHJpdmF0ZS5wZW0gLW91dCBwdWJsaWMucGVtYAoqIG1vdmUgU1NIIGtleXMgKGBwcml2YXRlLnBlbWAgYW5kIGBwdWJsaWMucGVtYCkgdG8gYGNvbmZpZy9qd3QvYAoqIHNldCBwYXNzcGhyYXNlIHVzZWQgdG8gZ2VuZXJhdGUgdGhlIGtleXMgaW4geW91ciBlbnYubG9jYWwgZmlsZSAoYEpXVF9QQVNTUEhSQVNFYCkKCiMjIyBJbml0aWFsIGRhdGFiYXNlIGRhdGEKKiBUbyBzdGFydCB3aXRoIG5vIGRhdGEgOiBgcGhwIGJpbi9jb25zb2xlIGRvY3RyaW5lOmZpeHR1cmVzOmxvYWQgLS1ncm91cD1zdGFydGluZ191c2Vyc2AKKiBUbyBzdGFydCB3aXRoIHNhbXBsZXMgZGF0YSA6IGBwaHAgYmluL2NvbnNvbGUgZG9jdHJpbmU6Zml4dHVyZXM6bG9hZGAKCiMjIFNlY3VyZSB5b3VyIHNpdGUKIyMjIEFkbWluIGFjY2VzcwpCeSBkZWZhdWx0LCB5b3UgY2FuIHJldHJpZXZlIGFuIGFkbWluIHRva2VuIDogIApKU09OIDogYHsidXNlcm5hbWUiOiJhZG1pbiIsInBhc3N3b3JkIjoic3VwZXJQYXNzd29yZDM0ISJ9YCAgIApVUkwgOiAgYC9hcGkvbG9naW5fY2hlY2tgICAgCk1FVEhPRCA6ICBgUE9TVGAKCiMjIyMgT3RoZXIgdXNlcnMKU2FtcGxlcyBkYXRhIGNvbWUgd2l0aCAxIGFkZGl0aW9uYWwgdXNlcnMgOiAgCkpTT04gOiBgeyJ1c2VybmFtZSI6InVzZXIxIiwicGFzc3dvcmQiOiJQYXNzV29yZDAxISJ9YCAgIApVUkwgOiAgYC9hcGkvbG9naW5fY2hlY2tgICAgCk1FVEhPRCA6ICBgUE9TVGAKCiMjIERvY3VtZW50YXRpb24KT25jZSBwcm9qZWN0IGlzIGluc3RhbGxlZCB5b3UgY2FuIGZpbmQgZG9jdW1lbnRhdGlvbiA6IApIVE1MIERvYyA6IChpbiBhIGJyb3dzZXIpIDogYC9hcGkvZG9jYApKc29uIERvYyAoT3BlbkFQSSAzLjApIDogYC9hcGkvZG9jLmpzb25gCkEgUGRmIHZlcnNpb24gY2FuIGJlIGZvdW5kIGluIGAvZG9jYCBkaXJlY3Rvcnk= readmeEtag: '"017bcb80e9da3608d05a9552bc25832524ffc048"' readmeLastModified: Fri, 18 Dec 2020 21:29:18 GMT repositoryId: 313562762 description: OpenClassRooms 7th Project. Build an API REST with Symfony created: '2020-11-17T09:03:19Z' updated: '2020-12-18T21:29:46Z' language: PHP archived: false stars: 1 watchers: 1 forks: 0 owner: OSEvohe logo: https://avatars.githubusercontent.com/u/13046894?v=4 repoEtag: '"59067b649bcb80255ea60e98cb0da175d331be5e7f02f266e2fc9764d0a54126"' repoLastModified: Fri, 18 Dec 2020 21:29:46 GMT foundInMaster: true category: Server id: 9845cdf339b0c28f4c61b07e0a5ab024 - source: openapi3 tags repository: https://github.com/boro23-wq/case-manager-api-design v3: true id: 93247bee3a374b56229352c0f3ad044e repositoryMetadata: base64Readme: >- 
# Case Managers API Collection

The API retrieves information about case managers and their cases.

Contact Support:
 - **Name**: Sintu Boro
 - **Email**: sb394@njit.edu

<!--- If we have only one group/collection, then no need for the "ungrouped" heading -->


## Variables

| Key | Value | Type |
| --- | ------|-------------|
| baseUrl (dev) | http://localhost:3000 | string |
| baseUrl (prod/mock) | https://01d9b5a5-eeac-4d0a-82f2-b7666179ebbb.mock.pstmn.io | string |



## Endpoints

* [casemanagers](#casemanagers)
    1. [{id}](#1-id)
    1. [Create a case manager.](#2-create-a-case-manager)
        * [Successfully created case manager.](#i-example-request-successfully-created-case-manager)
        * [Invalid request. Please fill the required input/inputs.](#ii-example-request-invalid-request-please-fill-the-required-inputinputs)
        * [Bad request. Can't find the requested resource.](#iii-example-request-bad-request-cant-find-the-requested-resource)
        * [Internal server error. Can't fulfill the request.](#iv-example-request-internal-server-error-cant-fulfill-the-request)
    1. [Find all case managers.](#3-find-all-case-managers)
        * [Successfully retrieved all the case managers.](#i-example-request-successfully-retrieved-all-the-case-managers)
        * [Invalid request. Please fill the required input/inputs.](#ii-example-request-invalid-request-please-fill-the-required-inputinputs-1)
        * [Bad request. Can't find the requested resource.](#iii-example-request-bad-request-cant-find-the-requested-resource-1)
        * [Internal server error. Can't fulfill the request.](#iv-example-request-internal-server-error-cant-fulfill-the-request-1)
* [cases](#cases)
    1. [{caseId}](#1-caseid)
    1. [Create a case.](#2-create-a-case)
        * [Successfully created case.](#i-example-request-successfully-created-case)
        * [Invalid request. Please fill the required input/inputs.](#ii-example-request-invalid-request-please-fill-the-required-inputinputs-2)
        * [Bad request. Can't find the requested resource.](#iii-example-request-bad-request-cant-find-the-requested-resource-2)
        * [Internal server error. Can't fulfill the request.](#iv-example-request-internal-server-error-cant-fulfill-the-request-2)
    1. [Find all cases.](#3-find-all-cases)
        * [Successfully retrieved all the cases.](#i-example-request-successfully-retrieved-all-the-cases)
        * [Invalid request. Please fill the required input/inputs.](#ii-example-request-invalid-request-please-fill-the-required-inputinputs-3)
        * [Bad request. Can't find the requested resource.](#iii-example-request-bad-request-cant-find-the-requested-resource-3)
        * [Internal server error. Can't fulfill the request.](#iv-example-request-internal-server-error-cant-fulfill-the-request-3)

--------



## casemanagers



### 1. {id}



***Endpoint:***

```bash
Method: 
Type: 
URL: 
```



### 2. Create a case manager.


Create a new case manager using the input provided and add it to the system.


***Endpoint:***

```bash
Method: POST
Type: RAW
URL: {{baseUrl}}/casemanagers
```


***Headers:***

| Key | Value | Description |
| --- | ------|-------------|
| Content-Type | application/json |  |
| Accept | application/json |  |



***Body:***

```js        
{
  "username": "johndoe",
  "firstName": "John",
  "lastName": "Doe",
  "email": "johndoe@cm.io",
  "phone": 8623817665
}
```



***More example Requests/Responses:***


#### I. Example Request: Successfully created case manager.



***Body:***

```js        
{
  "username": "johndoe",
  "firstName": "John",
  "lastName": "Doe",
  "email": "johndoe@cm.io",
  "phone": 8623817665
}
```



#### I. Example Response: Successfully created case manager.
```js
{
  "username": "johndoe",
  "firstName": "John",
  "lastName": "Doe",
  "email": "johndoe@cm.io",
  "phone": 8623817665,
  "caseManagerId": "378cdddc-31af-481c-9000-2ab4eabeebaf",
  "createdAt": "2021-01-30T08:30:00Z",
  "modifiedAt": "2021-01-31T08:30:00Z"
}
```


***Status Code:*** 201

<br>



#### II. Example Request: Invalid request. Please fill the required input/inputs.



***Body:***

```js        
{
//   "username": "johndoe",
  "firstName": "John",
  "lastName": "Doe",
  "email": "johndoe@cm.io",
  "phone": 8623817665
}
```



#### II. Example Response: Invalid request. Please fill the required input/inputs.
```js
{
  "statusCode": 400,
  "description": "Invalid request. Please fill the required input/inputs."
}
```


***Status Code:*** 400

<br>



#### III. Example Request: Bad request. Can't find the requested resource.



***Body:***

```js        
{
  "username": "johndoe",
//   "firstName": "John",
  "lastName": "Doe",
  "email": "johndoe@cm.io",
  "phone": 8623817665
}
```



#### III. Example Response: Bad request. Can't find the requested resource.
```js
{
  "statusCode": 404,
  "description": "Bad request. Can't find the requested resource."
}
```


***Status Code:*** 404

<br>



#### IV. Example Request: Internal server error. Can't fulfill the request.



***Body:***

```js        
// {
//   "username": "johndoe",
//   "firstName": "John",
//   "lastName": "Doe",
//   "email": "johndoe@cm.io",
//   "phone": 8623817665
// }
```



#### IV. Example Response: Internal server error. Can't fulfill the request.
```js
{
  "statusCode": 500,
  "description": "Internal server error. Can't fulfill the request."
}
```


***Status Code:*** 500

<br>



### 3. Find all case managers.


Get all the case managers and their cases recorded in the system.


***Endpoint:***

```bash
Method: GET
Type: 
URL: {{baseUrl}}/casemanagers
```


***Headers:***

| Key | Value | Description |
| --- | ------|-------------|
| Accept | application/json |  |



***More example Requests/Responses:***


#### I. Example Request: Successfully retrieved all the case managers.



***Body: None***



#### I. Example Response: Successfully retrieved all the case managers.
```js
[
  {
    "username": "johndoe",
    "firstName": "John",
    "lastName": "Doe",
    "email": "johndoe@cm.io",
    "phone": 8623817665,
    "caseManagerId": "378cdddc-31af-481c-9000-2ab4eabeebaf",
    "createdAt": "2021-01-30T08:30:00Z",
    "modifiedAt": "2021-01-31T08:30:00Z",
    "cases": [
      {
        "status": "Active",
        "category": "Eye",
        "severity": "0-Critical Impact",
        "subject": "Operation follow-up",
        "patientDetail": {
          "firstName": "Matt",
          "lastName": "Turner",
          "userName": "mattturner10",
          "phone": 9452877662,
          "email": "mattturner@gmail.com",
          "id": "46005a82-a612-4dc1-9279-7215df6895cd"
        },
        "solution": {
          "solutionSubject": "Operate eye disease.",
          "investigation": "Diseases on the left eye and abnormal eye development and function.",
          "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
        },
        "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
        "createdAt": "2021-03-30T08:30:00Z",
        "modifiedAt": "2021-04-28T08:30:00Z",
        "assignedTo": {
          "firstName": "Katherine",
          "lastName": "Marshall",
          "phone": 8452877665,
          "email": "kathmarshall@hm.io",
          "id": "26005a82-a612-4dc1-9279-7215df6895cd",
          "assignedOn": "2021-03-30T08:30:00Z",
          "profession": "Ophthalmologists"
        },
        "milestones": [
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          },
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          }
        ],
        "activities": [
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ],
        "notes": [
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ]
      },
      {
        "status": "Active",
        "category": "Eye",
        "severity": "0-Critical Impact",
        "subject": "Operation follow-up",
        "patientDetail": {
          "firstName": "Matt",
          "lastName": "Turner",
          "userName": "mattturner10",
          "phone": 9452877662,
          "email": "mattturner@gmail.com",
          "id": "46005a82-a612-4dc1-9279-7215df6895cd"
        },
        "solution": {
          "solutionSubject": "Operate eye disease.",
          "investigation": "Diseases on the left eye and abnormal eye development and function.",
          "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
        },
        "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
        "createdAt": "2021-03-30T08:30:00Z",
        "modifiedAt": "2021-04-28T08:30:00Z",
        "assignedTo": {
          "firstName": "Katherine",
          "lastName": "Marshall",
          "phone": 8452877665,
          "email": "kathmarshall@hm.io",
          "id": "26005a82-a612-4dc1-9279-7215df6895cd",
          "assignedOn": "2021-03-30T08:30:00Z",
          "profession": "Ophthalmologists"
        },
        "milestones": [
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          },
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          }
        ],
        "activities": [
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ],
        "notes": [
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ]
      }
    ]
  },
  {
    "username": "johndoe",
    "firstName": "John",
    "lastName": "Doe",
    "email": "johndoe@cm.io",
    "phone": 8623817665,
    "caseManagerId": "378cdddc-31af-481c-9000-2ab4eabeebaf",
    "createdAt": "2021-01-30T08:30:00Z",
    "modifiedAt": "2021-01-31T08:30:00Z",
    "cases": [
      {
        "status": "Active",
        "category": "Eye",
        "severity": "0-Critical Impact",
        "subject": "Operation follow-up",
        "patientDetail": {
          "firstName": "Matt",
          "lastName": "Turner",
          "userName": "mattturner10",
          "phone": 9452877662,
          "email": "mattturner@gmail.com",
          "id": "46005a82-a612-4dc1-9279-7215df6895cd"
        },
        "solution": {
          "solutionSubject": "Operate eye disease.",
          "investigation": "Diseases on the left eye and abnormal eye development and function.",
          "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
        },
        "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
        "createdAt": "2021-03-30T08:30:00Z",
        "modifiedAt": "2021-04-28T08:30:00Z",
        "assignedTo": {
          "firstName": "Katherine",
          "lastName": "Marshall",
          "phone": 8452877665,
          "email": "kathmarshall@hm.io",
          "id": "26005a82-a612-4dc1-9279-7215df6895cd",
          "assignedOn": "2021-03-30T08:30:00Z",
          "profession": "Ophthalmologists"
        },
        "milestones": [
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          },
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          }
        ],
        "activities": [
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ],
        "notes": [
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ]
      },
      {
        "status": "Active",
        "category": "Eye",
        "severity": "0-Critical Impact",
        "subject": "Operation follow-up",
        "patientDetail": {
          "firstName": "Matt",
          "lastName": "Turner",
          "userName": "mattturner10",
          "phone": 9452877662,
          "email": "mattturner@gmail.com",
          "id": "46005a82-a612-4dc1-9279-7215df6895cd"
        },
        "solution": {
          "solutionSubject": "Operate eye disease.",
          "investigation": "Diseases on the left eye and abnormal eye development and function.",
          "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
        },
        "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
        "createdAt": "2021-03-30T08:30:00Z",
        "modifiedAt": "2021-04-28T08:30:00Z",
        "assignedTo": {
          "firstName": "Katherine",
          "lastName": "Marshall",
          "phone": 8452877665,
          "email": "kathmarshall@hm.io",
          "id": "26005a82-a612-4dc1-9279-7215df6895cd",
          "assignedOn": "2021-03-30T08:30:00Z",
          "profession": "Ophthalmologists"
        },
        "milestones": [
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          },
          {
            "description": "Home therapy for left eye before surgery.",
            "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
            "createdAt": "2022-10-21T01:12:23.926Z"
          }
        ],
        "activities": [
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "description": "Uploaded X-Ray report.",
            "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ],
        "notes": [
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          },
          {
            "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
            "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
            "createdAt": "2022-10-21T01:14:27.516Z"
          }
        ]
      }
    ]
  }
]
```


***Status Code:*** 200

<br>



#### II. Example Request: Invalid request. Please fill the required input/inputs.



***Query:***

| Key | Value | Description |
| --- | ------|-------------|
| page | 1 |  |
| perPage | 10 |  |



***Body: None***



#### II. Example Response: Invalid request. Please fill the required input/inputs.
```js
{
  "statusCode": 400,
  "description": "Invalid request. Please fill the required input/inputs."
}
```


***Status Code:*** 400

<br>



#### III. Example Request: Bad request. Can't find the requested resource.



***Query:***

| Key | Value | Description |
| --- | ------|-------------|
| page | 1 |  |
| perPage | 10 |  |



***Body: None***



#### III. Example Response: Bad request. Can't find the requested resource.
```js
{
  "statusCode": 404,
  "description": "Bad request. Can't find the requested resource."
}
```


***Status Code:*** 404

<br>



#### IV. Example Request: Internal server error. Can't fulfill the request.



***Query:***

| Key | Value | Description |
| --- | ------|-------------|
| page | 1 |  |
| perPage | 10 |  |



***Body: None***



#### IV. Example Response: Internal server error. Can't fulfill the request.
```js
{
  "statusCode": 500,
  "description": "Internal server error. Can't fulfill the request."
}
```


***Status Code:*** 500

<br>



## cases



### 1. {caseId}



***Endpoint:***

```bash
Method: 
Type: 
URL: 
```



### 2. Create a case.


Create a new case using the input provided and add it to the system.


***Endpoint:***

```bash
Method: POST
Type: RAW
URL: {{baseUrl}}/cases
```


***Headers:***

| Key | Value | Description |
| --- | ------|-------------|
| Content-Type | application/json |  |
| Accept | application/json |  |



***Body:***

```js        
{
  "status": "Active",
  "category": "Eye",
  "severity": "0-Critical Impact",
  "subject": "Operation follow-up",
  "patientDetail": {
    "firstName": "Matt",
    "lastName": "Turner",
    "userName": "mattturner10",
    "phone": 9452877662,
    "email": "mattturner@gmail.com",
    "id": "46005a82-a612-4dc1-9279-7215df6895cd"
  },
  "solution": {
    "solutionSubject": "Operate eye disease.",
    "investigation": "Diseases on the left eye and abnormal eye development and function.",
    "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
  },
  "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
  "createdAt": "2021-03-30T08:30:00Z",
  "modifiedAt": "2021-04-28T08:30:00Z",
  "assignedTo": {
    "firstName": "Katherine",
    "lastName": "Marshall",
    "phone": 8452877665,
    "email": "kathmarshall@hm.io",
    "id": "26005a82-a612-4dc1-9279-7215df6895cd",
    "assignedOn": "2021-03-30T08:30:00Z",
    "profession": "Ophthalmologists"
  },
  "milestones": [
    {
      "description": "Home therapy for left eye before surgery.",
      "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
      "createdAt": "2022-10-21T01:12:23.926Z"
    },
    {
      "description": "Home therapy for left eye before surgery.",
      "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
      "createdAt": "2022-10-21T01:12:23.926Z"
    }
  ],
  "activities": [
    {
      "description": "Uploaded X-Ray report.",
      "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    },
    {
      "description": "Uploaded X-Ray report.",
      "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    }
  ],
  "notes": [
    {
      "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
      "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    },
    {
      "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
      "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    }
  ]
}
```



***More example Requests/Responses:***


#### I. Example Request: Successfully created case.



***Body:***

```js        
{
  "status": "Active",
  "category": "Eye",
  "severity": "0-Critical Impact",
  "subject": "Operation follow-up",
  "patientDetail": {
    "firstName": "Matt",
    "lastName": "Turner",
    "userName": "mattturner10",
    "phone": 9452877662,
    "email": "mattturner@gmail.com",
    "id": "46005a82-a612-4dc1-9279-7215df6895cd"
  },
  "solution": {
    "solutionSubject": "Operate eye disease.",
    "investigation": "Diseases on the left eye and abnormal eye development and function.",
    "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
  },
  "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
  "createdAt": "2021-03-30T08:30:00Z",
  "modifiedAt": "2021-04-28T08:30:00Z",
  "assignedTo": {
    "firstName": "Katherine",
    "lastName": "Marshall",
    "phone": 8452877665,
    "email": "kathmarshall@hm.io",
    "id": "26005a82-a612-4dc1-9279-7215df6895cd",
    "assignedOn": "2021-03-30T08:30:00Z",
    "profession": "Ophthalmologists"
  },
  "milestones": [
    {
      "description": "Home therapy for left eye before surgery.",
      "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
      "createdAt": "2022-10-21T01:12:23.926Z"
    },
    {
      "description": "Home therapy for left eye before surgery.",
      "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
      "createdAt": "2022-10-21T01:12:23.926Z"
    }
  ],
  "activities": [
    {
      "description": "Uploaded X-Ray report.",
      "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    },
    {
      "description": "Uploaded X-Ray report.",
      "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    }
  ],
  "notes": [
    {
      "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
      "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    },
    {
      "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
      "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
      "createdAt": "2022-10-21T01:14:27.516Z"
    }
  ]
}
```



#### I. Example Response: Successfully created case.
```js
{
  "status": "Active",
  "category": "Eye",
  "severity": "0-Critical Impact",
  "subject": "Operation follow-up",
  "patientDetail": {
    "firstName": "Matt",
    "lastName": "Turner",
    "userName": "mattturner10",
    "phone": 9452877662,
    "email": "mattturner@gmail.com",
    "id": "46005a82-a612-4dc1-9279-7215df6895cd"
  },
  "solution": {
    "solutionSubject": "Operate eye disease.",
    "investigation": "Diseases on the left eye and abnormal eye development and function.",
    "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
  },
  "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
  "createdAt": "2021-03-30T08:30:00Z",
  "modifiedAt": "2021-04-28T08:30:00Z"
}
```


***Status Code:*** 201

<br>



#### II. Example Request: Invalid request. Please fill the required input/inputs.



***Body: None***



#### II. Example Response: Invalid request. Please fill the required input/inputs.
```js
{
  "statusCode": 400,
  "description": "Invalid request. Please fill the required input/inputs."
}
```


***Status Code:*** 400

<br>



#### III. Example Request: Bad request. Can't find the requested resource.



***Body: None***



#### III. Example Response: Bad request. Can't find the requested resource.
```js
{
  "statusCode": 404,
  "description": "Bad request. Can't find the requested resource."
}
```


***Status Code:*** 404

<br>



#### IV. Example Request: Internal server error. Can't fulfill the request.



***Body: None***



#### IV. Example Response: Internal server error. Can't fulfill the request.
```js
{
  "statusCode": 500,
  "description": "Internal server error. Can't fulfill the request."
}
```


***Status Code:*** 500

<br>



### 3. Find all cases.


Get all the cases recorded in the system.


***Endpoint:***

```bash
Method: GET
Type: 
URL: {{baseUrl}}/cases
```


***Headers:***

| Key | Value | Description |
| --- | ------|-------------|
| Accept | application/json |  |



***More example Requests/Responses:***


#### I. Example Request: Successfully retrieved all the cases.



***Body: None***



#### I. Example Response: Successfully retrieved all the cases.
```js
[
  {
    "status": "Active",
    "category": "Eye",
    "severity": "0-Critical Impact",
    "subject": "Operation follow-up",
    "patientDetail": {
      "firstName": "Matt",
      "lastName": "Turner",
      "userName": "mattturner10",
      "phone": 9452877662,
      "email": "mattturner@gmail.com",
      "id": "46005a82-a612-4dc1-9279-7215df6895cd"
    },
    "solution": {
      "solutionSubject": "Operate eye disease.",
      "investigation": "Diseases on the left eye and abnormal eye development and function.",
      "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
    },
    "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
    "createdAt": "2021-03-30T08:30:00Z",
    "modifiedAt": "2021-04-28T08:30:00Z",
    "assignedTo": {
      "firstName": "Katherine",
      "lastName": "Marshall",
      "phone": 8452877665,
      "email": "kathmarshall@hm.io",
      "id": "26005a82-a612-4dc1-9279-7215df6895cd",
      "assignedOn": "2021-03-30T08:30:00Z",
      "profession": "Ophthalmologists"
    },
    "milestones": [
      {
        "description": "Home therapy for left eye before surgery.",
        "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
        "createdAt": "2022-10-21T01:12:23.926Z"
      },
      {
        "description": "Home therapy for left eye before surgery.",
        "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
        "createdAt": "2022-10-21T01:12:23.926Z"
      }
    ],
    "activities": [
      {
        "description": "Uploaded X-Ray report.",
        "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      },
      {
        "description": "Uploaded X-Ray report.",
        "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      }
    ],
    "notes": [
      {
        "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
        "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      },
      {
        "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
        "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      }
    ]
  },
  {
    "status": "Active",
    "category": "Eye",
    "severity": "0-Critical Impact",
    "subject": "Operation follow-up",
    "patientDetail": {
      "firstName": "Matt",
      "lastName": "Turner",
      "userName": "mattturner10",
      "phone": 9452877662,
      "email": "mattturner@gmail.com",
      "id": "46005a82-a612-4dc1-9279-7215df6895cd"
    },
    "solution": {
      "solutionSubject": "Operate eye disease.",
      "investigation": "Diseases on the left eye and abnormal eye development and function.",
      "resolution": "Successfully operate on disease on the left eye and normal eye development and function."
    },
    "caseId": "36005a82-a612-4dc1-9279-7215df6895cd",
    "createdAt": "2021-03-30T08:30:00Z",
    "modifiedAt": "2021-04-28T08:30:00Z",
    "assignedTo": {
      "firstName": "Katherine",
      "lastName": "Marshall",
      "phone": 8452877665,
      "email": "kathmarshall@hm.io",
      "id": "26005a82-a612-4dc1-9279-7215df6895cd",
      "assignedOn": "2021-03-30T08:30:00Z",
      "profession": "Ophthalmologists"
    },
    "milestones": [
      {
        "description": "Home therapy for left eye before surgery.",
        "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
        "createdAt": "2022-10-21T01:12:23.926Z"
      },
      {
        "description": "Home therapy for left eye before surgery.",
        "id": "30027811-1ddb-4135-bb94-3a68aae1491c",
        "createdAt": "2022-10-21T01:12:23.926Z"
      }
    ],
    "activities": [
      {
        "description": "Uploaded X-Ray report.",
        "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      },
      {
        "description": "Uploaded X-Ray report.",
        "id": "a7a32579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      }
    ],
    "notes": [
      {
        "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
        "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      },
      {
        "comment": "Client is at risk of being non-adherent to medications and other appointments. Client needs reminders to assist with keeping appointments.",
        "id": "n7a42579-d7c3-45ef-8b5f-8eac996db599",
        "createdAt": "2022-10-21T01:14:27.516Z"
      }
    ]
  }
]
```


***Status Code:*** 200

<br>



#### II. Example Request: Invalid request. Please fill the required input/inputs.



***Query:***

| Key | Value | Description |
| --- | ------|-------------|
| page | 1 |  |
| perPage | 10 |  |



***Body: None***



#### II. Example Response: Invalid request. Please fill the required input/inputs.
```js
{
  "statusCode": 400,
  "description": "Invalid request. Please fill the required input/inputs."
}
```


***Status Code:*** 400

<br>



#### III. Example Request: Bad request. Can't find the requested resource.



***Query:***

| Key | Value | Description |
| --- | ------|-------------|
| page | 1 |  |
| perPage | 10 |  |



***Body: None***



#### III. Example Response: Bad request. Can't find the requested resource.
```js
{
  "statusCode": 404,
  "description": "Bad request. Can't find the requested resource."
}
```


***Status Code:*** 404

<br>



#### IV. Example Request: Internal server error. Can't fulfill the request.



***Query:***

| Key | Value | Description |
| --- | ------|-------------|
| page | 1 |  |
| perPage | 10 |  |



***Body: None***



#### IV. Example Response: Internal server error. Can't fulfill the request.
```js
{
  "statusCode": 500,
  "description": "Internal server error. Can't fulfill the request."
}
```


***Status Code:*** 500

<br>



---
[Back to top](#case-managers-api-collection)

>Generated at 2022-10-30 21:18:27 by [docgen](https://github.com/thedevsaddam/docgen)
 readmeEtag: '"f5c075ab712ab0be7c3dc60c331f5eaec3f074a6"' readmeLastModified: Mon, 31 Oct 2022 02:20:31 GMT repositoryId: 554357697 description: >- This is the design of a REST API in Postman that performs operation on case managers, cases, and related case notes, which is a part of the project CS 673 - Care Management and Co-ordination. created: '2022-10-19T17:19:00Z' updated: '2022-10-31T18:47:57Z' language: null archived: false stars: 1 watchers: 1 forks: 1 owner: Boro23-wq logo: https://avatars.githubusercontent.com/u/62204944?v=4 license: MIT repoEtag: '"87273b4b83db4ccbb1d5e47d49069c99799364dc59266c034a0f96c49a8f72dd"' repoLastModified: Mon, 31 Oct 2022 18:47:57 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/tdatit/tda-store-restful-api v3: true id: 2871f22095748321dac0e9d7f7cb5e80 repositoryMetadata: base64Readme: >- IyB0ZGEtc3RvcmUtcmVzdGZ1bC1hcGkKVERBIFdlYiBpcyBhbiBlLWNvbW1lcmNlIHdlYnNpdGUgYnVpbHQgb24gdGhlIFJFU1QgQVBJIGFyY2hpdGVjdHVyZSB1c2VkIGZvciByZWxhdGVkIFREQSBlLWNvbW1lcmNlIGFwcGxpY2F0aW9ucyBzdWNoIGFzIG1vYmlsZSBhbmQgd2ViIAojCjxoMj5UZWNoPC9oMj4KPHVsPiAKICA8bGk+SmF2YSAxMTwvbGk+CiAgPGxpPk15U1FMIC8gUG9zdGdyZVNRTDwvbGk+CiAgPGxpPlNwcmluZyBCb290PC9saT4KICA8bGk+U3ByaW5nIFdlYjwvbGk+CiAgPGxpPlNwcmluZyBTZWN1cml0eTwvbGk+CiAgPGxpPlNwcmluZyBEYXRhPC9saT4KICA8bGk+U3ByaW5nIEFPUDwvbGk+CiAgPGxpPlZhbGlkYXRvcjwvbGk+CiAgPGxpPkNsb3VkaW5hcnk8L2xpPgogIDxsaT5SYWJiaXRNUTwvbGk+CjwvdWw+CgoKIVtlcmRdKGh0dHBzOi8vZ2l0aHViLmNvbS90ZGF0SVQvdGRhLXN0b3JlLXJlc3RmdWwtYXBpL2Fzc2V0cy84NTAxODA2Mi9mZGY3YzhkZS01ODgzLTRlMzktYTdlNS1lOGE1NGU2NGM0NzQpCgoKPGg0PkRldmVsb3BtZW50IGJ5IFRyYW4gVGllbiBEYXQ8L2g0Pgo= readmeEtag: '"d1e6b2de5680aea1375d7c068f53bb11379d6e64"' readmeLastModified: Sat, 10 Jun 2023 09:21:12 GMT repositoryId: 619189905 description: Backend for mobile app (TDA Laptop) created: '2023-03-26T14:35:14Z' updated: '2024-01-15T11:21:35Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: tdatIT logo: https://avatars.githubusercontent.com/u/85018062?v=4 repoEtag: '"4507183a05745bc0e2ef01291b1276b425a8f3a1f370b2c27148b0d7d8eac37a"' repoLastModified: Mon, 15 Jan 2024 11:21:35 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/elomagic/yacog4openapi v3: true id: b068e391411b136805dc8897fb4e93f0 repositoryMetadata: base64Readme: >- IyBZYXN0IEFub3RoZXIgQ29kZSBHZW5lcmF0b3IgVG9vbCA0IE9wZW5BUEkKCkN1cnJlbnRseSwgcHJvdG90eXBlIHN0YXR1cwoKLS0tCgpbIVtHaXRIdWIgdGFnXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi90YWcvZWxvbWFnaWMveWFjb2c0b3BlbmFwaS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZWxvbWFnaWMveWFjb2c0b3BlbmFwaS90YWdzLykKWyFbR2l0SHViIGlzc3Vlc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvaXNzdWVzLXJhdy9lbG9tYWdpYy95YWNvZzRvcGVuYXBpKV0oaHR0cHM6Ly9naXRodWIuY29tL2Vsb21hZ2ljL3lhY29nNG9wZW5hcGkvaXNzdWVzKQpbIVtBcGFjaGUgMi4wIGxpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvQXBhY2hlLTIuMC1ibHVlLnN2ZyldKGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvZ3BsLTMuMC1zdGFuZGFsb25lLmh0bWwpClshW21hZGUtd2l0aC1taWNyb3B5dGhvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9NYWRlJTIwd2l0aC1QeXRob24tMWY0MjVmLnN2ZyldKGh0dHBzOi8vd3d3LnB5dGhvbi5vcmcpClshW01haW50ZW5hbmNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL01haW50YWluZWQlM0YteWVzLWdyZWVuLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9lbG9tYWdpYy95YWNvZzRvcGVuYXBpL2dyYXBocy9jb21taXQtYWN0aXZpdHkpClshW0J1eW1lYWNvZmZlZV0oaHR0cHM6Ly9iYWRnZW4ubmV0L2JhZGdlL2ljb24vYnV5bWVhY29mZmVlP2ljb249YnV5bWVhY29mZmVlJmxhYmVsKV0oaHR0cHM6Ly93d3cuYnV5bWVhY29mZmVlLmNvbS9lbG9tYWdpYykKCiMjIFRhYmxlIG9mIENvbnRlbnRzCgotIFtXaGF0IGFib3V0P10oI3doYXQtYWJvdXQ/KQotIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCi0gW0NvbmZpZ3VyYXRpb25dKCNjb25maWd1cmF0aW9uKQotIFtVc2luZ10oI3VzaW5nKQotIFtDb250cmlidXRpb25dKCNjb250cmlidXRpb24pCgojIFdoYXQgQWJvdXQ/CgpUaGlzIHByb2plY3QgaXMgY3VycmVudGx5IGEgc21hbGwgcHJvdG90eXBlIHByb2plY3QgaG93IHRvIHJlYWQgYW4gT3BlbiBBUEkgZmlsZSBhbmQgdXNlIGEgdGVtcGxhdGUgZW5naW5lIGluIFB5dGhvbi4gCgojIEluc3RhbGxhdGlvbgoKQWxsIHJlcXVpcmVkIFB5dGhvbiBkZXBlbmRlbmNpZXMgYXJlIGRlY2xhcmVkIGluIHRoZSAqcmVxdWlyZW1lbnQudHh0KiBmaWxlIGFuZCBtdXN0IGJlIGluc3RhbGxlZCB3aXRoIHRoZSBmb2xsb3dpbmcgCmNvbW1hbmQgYnkgdXNpbmcgdGhlIFB5dGhvbiBwYWNrYWdlIG1hbmFnZXIgKlBpcCo6IAoKYGBgc2hlbGwKcGlwIGluc3RhbGwgLXIgcmVxdWlyZW1lbnRzLnR4dApgYGAKCiMgQ29uZmlndXJhdGlvbgoKYGBganNvbjUKewogICAgImlucHV0IjogewogICAgICAgIC8vIERlZmF1bHQgT3BlbkFQSSBkZWZpbml0aW9uIGZpbGUKICAgICAgICAic291cmNlIjogIi4uL3Rlc3Qvb3BlbmFwaS5qc29uIgogICAgfSwgICAgCiAgICAvLyBUZW1wbGF0ZSB0byB1c2UKICAgICJ0ZW1wbGF0ZSI6ICJyZXNvdXJjZXMvZGVscGhpLXVuaXQuamluamEyIiwKICAgIC8vIEZpbGUgZ2VuZXJhdGVkIG91dHB1dAogICAgIm91dHB1dCI6ICIuLi9vdXQvdVJlc3RTZXJ2aWNlQ2xpZW50LnBhcyIsCiAgICAvLyBPcGVuIEFQSSBkYXRhdHlwZSBtYXBwaW5ncwogICAgImRhdGF0eXBlLW1hcCI6IHsKICAgICAgICAibnVtYmVyLGZsb2F0IjogICJTaW5nbGUiLAogICAgICAgICJudW1iZXIsZG91YmxlIjogIkRvdWJsZSIsCiAgICAgICAgIm51bWJlciwiOiAiU2luZ2xlIiwKCiAgICAgICAgImludGVnZXIsaW4zMiI6ICJJbnRlZ2VyIiwKICAgICAgICAiaW50ZWdlcixpbnQ2NCI6ICJJbnQ2NCIsCiAgICAgICAgImludGVnZXIsIjogIkludGVnZXIiLAoKICAgICAgICAic3RyaW5nLGJ5dGUiOiAiU3RyaW5nIiwKICAgICAgICAic3RyaW5nLGJpbmFyeSI6ICJTdHJpbmciLAogICAgICAgICJzdHJpbmcsIjogIlN0cmluZyIsCgogICAgICAgICJzdHJpbmcsZGF0ZSI6ICJURGF0ZSIsCiAgICAgICAgInN0cmluZyxkYXRlLXRpbWUiOiAiVERhdGVUaW1lIiwKCiAgICAgICAgInN0cmluZyx1dWlkIjogIlN0cmluZyIsCgogICAgICAgICJib29sZWFuLCI6ICJCb29sZWFuIgogICAgfQp9CmBgYAoKdGJjCgojIFVzaW5nCgojIyBDcmVhdGluZyBvdXRwdXQKCiMjIyBFeGFtcGxlCgpgYGBzaGVsbApzcmMvZ2VuY2xpZW50LnB5IC0tc291cmNlPXRlc3Qvb3BlbmFwaS5qc29uIC0tdGVtcGxhdGU9dGVzdC9kZWxwaGktdGVzdC1leGFtcGxlLmppbmphMiAtLW91dHB1dD10ZXN0LW91dHB1dC9kZWxwaGktdW5pdC5wYXMKYGBgCgojIyBTdXBwb3J0ZWQgcGFyYW1ldGVycwoKVG8gZ2V0IGFuIG92ZXJ2aWV3IG9mIHRoZSBzdXBwb3J0ZWQgcGFyYW1ldGVycywgZW50ZXIgdGhlIGZvbGxvd2luZyBjb21tYW5kIGluIHlvdXIgc2hlbGw6CgpgYGBzaGVsbApzcmMvZ2VuY2xpZW50LnB5IC0taGVscApgYGAKCnRiYwoKIyBDb250cmlidXRpb24KCnRiZA== readmeEtag: '"5449f612c93d25e8231a6291b08989be6f8dddd6"' readmeLastModified: Mon, 15 Aug 2022 09:30:44 GMT repositoryId: 523364744 description: Yast Another Code Generator Tool 4 Open API created: '2022-08-10T13:57:05Z' updated: '2024-01-19T10:28:57Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: elomagic logo: https://avatars.githubusercontent.com/u/2555557?v=4 license: Apache-2.0 repoEtag: '"47cd76afcee8cea6c6a77f35c2fa41c87fb011a48fe8691a32584fcebd2addc8"' repoLastModified: Fri, 19 Jan 2024 10:28:57 GMT category: Server Implementations foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/bump-sh-examples/rails-code-first v3: true v3_1: true id: e9bd7b3601526c75c4e63226d8d34a13 repositoryMetadata: base64Readme: >- IyBSYWlscyBIZWxsbyBPcGVuQVBJCgpBIHF1aWNrICJIZWxsbyBXb3JsZCIgYnV0IGZvciBnZW5lcmF0aW5nIE9wZW5BUEksIHVzaW5nIHRoZSBSdWJ5IG9uIFJhaWxzIGV4dGVuc2lvbiBbUlN3YWddKGh0dHBzOi8vZ2l0aHViLmNvbS9yc3dhZy9yc3dhZy8pLgoKVGhpcyByZXBvc2l0b3J5IHdhcyBidWlsdCBhcyBzYW1wbGUgY29kZSBmb3IgdGhlIEJ1bXAuc2ggZ3VpZGUgb24gW0dlbmVyYXRpbmcgT3BlbkFQSSBkb2NzIGZvciBSdWJ5IG9uIFJhaWxzIHdpdGggUlN3YWddKGh0dHBzOi8vZG9jcy5idW1wLnNoL2d1aWRlcy9vcGVuYXBpL2NvZGUtZmlyc3QtcmFpbHMvKS4KCiMjIFVzYWdlCgpDbG9uZSB0aGUgcmVwb3NpdG9yeSBkb3duIHRvIGdpdmUgaXQgYSB0cnkuCgpgYGAKIyBTZXQgZXZlcnl0aGluZyB1cAokIGJ1bmRsZSBpbnN0YWxsCgojIEV4cG9ydCB0aGUgT3BlbkFQSQokIHJha2UgcnN3YWcKCiMgVGFrZSBhIGxvb2sgYXQgdGhlIGdlbmVyYXRlZCBPcGVuQVBJCmNhdCBzd2FnZ2VyL2FwaS9zd2FnZ2VyLnlhbWwKYGBgCgpQcmV2aWV3IHRoZSBBUEkgcmVmZXJlbmNlIGRvY3MgW29uIEJ1bXAuc2hdKGh0dHBzOi8vYnVtcC5zaC9idW1wLWV4YW1wbGVzL2h1Yi9jb2RlLXNhbXBsZXMvZG9jL3JhaWxzLWhlbGxvLW9wZW5hcGkpLgoKIyMgTGljZW5zZQoKVGhlIGNvbnRlbnRzIG9mIHRoaXMgcmVwb3NpdG9yeSBhcmUgbGljZW5zZWQgdW5kZXIgW0NDIEJZLU5DLVNBCiAgNC4wXSguL0xJQ0VOU0VfQ0MtQlktTkMtU0EtNC4wKS4K readmeEtag: '"2694988af4c3e6185be844cb4987decfa1b13f15"' readmeLastModified: Mon, 11 Mar 2024 16:05:51 GMT repositoryId: 737069913 description: Add OpenAPI powered documentation to your Rails API with RSwag. created: '2023-12-29T18:09:14Z' updated: '2024-02-27T16:17:23Z' language: Ruby archived: false stars: 1 watchers: 1 forks: 1 owner: bump-sh-examples logo: https://avatars.githubusercontent.com/u/157144805?v=4 license: NOASSERTION repoEtag: '"1fc2330cb1d209b387dd5dd8356bc2c642200d43d639ad830bfe81eab1f682fc"' repoLastModified: Tue, 27 Feb 2024 16:17:23 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/rakesh5777/oxytrack-turbo v3: true id: ab8d2cd37cb51948ff57c08e473c2da3 repositoryMetadata: base64Readme: >- IyBPeHlUcmFjayBUdXJibyBQcm9qZWN0CgpXZWxjb21lIHRvIHRoZSBPeHlUcmFjayBUdXJibyBwcm9qZWN0ISBUaGlzIG1vbm9yZXBvIGlzIGJ1aWx0IHdpdGggVHVyYm8gUmVwbywgdXRpbGl6aW5nIGEgcm9idXN0IHN0YWNrIHRoYXQgaW5jbHVkZXMgUHJpc21hLCBQb3N0Z3JlcywgRXhwcmVzcywgTm9kZS5qcywgUmVhY3QsIGFuZCBUeXBlU2NyaXB0IGFjcm9zcyBib3RoIHRoZSBmcm9udGVuZCBhbmQgYmFja2VuZCwgZW5zdXJpbmcgYSBzdHJvbmdseSB0eXBlZCBkZXZlbG9wbWVudCBlbnZpcm9ubWVudC4KCiMjIEZlYXR1cmVzCgotICoqTW9ub3JlcG8gU3RydWN0dXJlKio6IE9yZ2FuaXplZCBpbnRvIGBhcHBzL2AgZm9yIGZyb250ZW5kL2JhY2tlbmQgYXBwbGljYXRpb25zIGFuZCBgcGFja2FnZXMvYCBmb3Igc2hhcmVkIGxvZ2ljIG9yIGNvbXBvbmVudHMuCi0gKipPcGVuQVBJIFNwZWNpZmljYXRpb24qKjogVXRpbGl6ZXMgT3BlbkFQSSBzcGVjIGFzIHRoZSBzb3VyY2Ugb2YgdHJ1dGggZm9yIEFQSSBkZXNpZ24sIHdpdGggYXV0by1nZW5lcmF0ZWQgVHlwZVNjcmlwdCBjbGllbnRzIHVzaW5nIGBvcGVuYXBpLXR5cGVzY3JpcHRgIGFuZCBgb3BlbmFwaS10eXBlc2NyaXB0LWF4aW9zYC4KLSAqKlN0cm9uZyBUeXBpbmcqKjogQm90aCBmcm9udGVuZCBhbmQgYmFja2VuZCBhcmUgZnVsbHkgdHlwZWQsIGVuaGFuY2luZyBkZXZlbG9wbWVudCBleHBlcmllbmNlIGFuZCBjb2RlIHF1YWxpdHkuCi0gKipEYXRhYmFzZSBGbGV4aWJpbGl0eSoqOiBTdXBwb3J0cyBsb2NhbCBQb3N0Z3JlcyBzZXR1cHMgYXMgd2VsbCBhcyBEb2NrZXItYmFzZWQgZW52aXJvbm1lbnRzIGZvciBlYXNlIG9mIGRldmVsb3BtZW50LgotICoqRGV2ZWxvcG1lbnQgV29ya2Zsb3cqKjogSW50ZWdyYXRlcyB0b29scyBsaWtlIFByZXR0aWVyLCBFU0xpbnQsIEh1c2t5LCBhbmQgbGludC1zdGFnZWQgZm9yIGNvZGUgZm9ybWF0dGluZyBhbmQgbGludGluZywgZW5zdXJpbmcgY29kZSBjb25zaXN0ZW5jeSBhbmQgcXVhbGl0eS4KCiMjIEdldHRpbmcgU3RhcnRlZAoKIyMjIFByZXJlcXVpc2l0ZXMKCi0gTm9kZS5qcyAodjE4IG9yIGhpZ2hlcikKLSBwbnBtICh2OC45LjApCi0gRG9ja2VyIChmb3IgcnVubmluZyBQb3N0Z3JlcyBpbiBhIGNvbnRhaW5lcikKCiMjIyBTZXR1cCBJbnN0cnVjdGlvbnMKCjEuICoqQ2xvbmUgdGhlIFJlcG9zaXRvcnkqKgoKICAgU3RhcnQgYnkgY2xvbmluZyB0aGlzIHJlcG9zaXRvcnkgdG8geW91ciBsb2NhbCBtYWNoaW5lLgoKMi4gKipFbnZpcm9ubWVudCBTZXR1cCoqCgogICBSZW5hbWUgYC5leGFtcGxlLmVudmAgdG8gYC5lbnZgIGluIGJvdGggYGFwcHMvYmFja2VuZGAgYW5kIGBwYWNrYWdlcy9kYXRhYmFzZWAuIFRoaXMgaXMgY3J1Y2lhbCBmb3IgY29uZmlndXJpbmcgeW91ciBlbnZpcm9ubWVudCB2YXJpYWJsZXMgY29ycmVjdGx5LgoKMy4gKipEYXRhYmFzZSBDb25maWd1cmF0aW9uKioKCiAgIEVuc3VyZSB5b3UgaGF2ZSBhIHJ1bm5pbmcgUG9zdGdyZXMgZGF0YWJhc2UuIFlvdSBjYW4gdXNlIERvY2tlciBhcyBtZW50aW9uZWQgaW4gdGhlIHNjcmlwdHMgc2VjdGlvbiBvciBzZXQgdXAgYSBsb2NhbCBpbnN0YW5jZS4KCjQuICoqSW5zdGFsbCBEZXBlbmRlbmNpZXMqKgoKICAgUnVuIGBwbnBtIGluc3RhbGxgIHRvIGluc3RhbGwgYWxsIHJlcXVpcmVkIGRlcGVuZGVuY2llcyBhY3Jvc3MgdGhlIG1vbm9yZXBvLgoKNS4gKipTdGFydCBEZXZlbG9wbWVudCBTZXJ2ZXIqKgoKICAgVG8gc3RhcnQgdGhlIGRldmVsb3BtZW50IHNlcnZlciwgcnVuIGB0dXJibyBydW4gZGV2YC4KCiMjIFVzZWZ1bCBTY3JpcHRzCgpEZWZpbmVkIHdpdGhpbiB0aGUgYHR1cmJvLmpzb25gIHBpcGVsaW5lLCB0aGVzZSBjb21tYW5kcyBhcmUga2V5IHRvIHRoZSBwcm9qZWN0J3MgZGV2ZWxvcG1lbnQgYW5kIGRlcGxveW1lbnQgcHJvY2Vzc2VzOgoKLSAqKlByaXNtYSBDb21tYW5kcyoqOgoKICAtIGB0dXJibyBydW4gZGI6Z2VuZXJhdGVgOiBHZW5lcmF0ZXMgUHJpc21hIGNsaWVudCBhcnRpZmFjdHMsIGVzc2VudGlhbCBmb3IgaW50ZXJhY3Rpbmcgd2l0aCB5b3VyIGRhdGFiYXNlIGluIGEgdHlwZS1zYWZlIG1hbm5lci4KICAtIGB0dXJibyBydW4gZGI6cHVzaGA6IFVwZGF0ZXMgdGhlIGRhdGFiYXNlIHNjaGVtYSBkdXJpbmcgZGV2ZWxvcG1lbnQgd2l0aG91dCBhcHBseWluZyBtaWdyYXRpb25zLCB1c2VmdWwgZm9yIHF1aWNrIGl0ZXJhdGlvbnMuCiAgLSBgdHVyYm8gcnVuIGRiOm1pZ3JhdGVgOiBBcHBsaWVzIGRhdGFiYXNlIG1pZ3JhdGlvbnMsIGVuc3VyaW5nIHlvdXIgZGF0YWJhc2Ugc2NoZW1hIG1hdGNoZXMgeW91ciBQcmlzbWEgc2NoZW1hLgoKLSAqKkFQSSBDb250cmFjdCBhbmQgQ2xpZW50IEdlbmVyYXRpb24qKjoKCiAgLSBgdHVyYm8gcnVuIGJ1aWxkLWFwaS1jb250cmFjdGA6IEJ1bmRsZXMgdGhlIE9wZW5BUEkgc3BlYyBhbmQgZ2VuZXJhdGVzIFR5cGVTY3JpcHQgQXhpb3MgY2xpZW50cywgZW5zdXJpbmcgZnJvbnRlbmQgYW5kIGJhY2tlbmQgY29tbXVuaWNhdGUgb3ZlciBhIHN0cm9uZ2x5IHR5cGVkIEFQSS4KCi0gKipUdXJibyBDb21tYW5kcyoqOgogIC0gYHR1cmJvIHJ1biBidWlsZGA6IENvbXBpbGVzIHRoZSBwcm9qZWN0IGZvciBwcm9kdWN0aW9uLCBlbnN1cmluZyBhbGwgZGVwZW5kZW5jaWVzIGxpa2UgUHJpc21hIGNsaWVudCBhbmQgQVBJIGNvbnRyYWN0cyBhcmUgZ2VuZXJhdGVkIGFuZCB1cC10by1kYXRlLgogIC0gYHR1cmJvIHJ1biBkZXZgOiBTdGFydHMgdGhlIGRldmVsb3BtZW50IHNlcnZlciwgd2F0Y2hpbmcgZm9yIGNoYW5nZXMgaW4geW91ciBzb3VyY2UgY29kZSBhbmQgYXV0b21hdGljYWxseSByZWNvbXBpbGluZy4KCiMjIFByb2plY3QgU3RydWN0dXJlCgotIGBhcHBzLypgOiBDb250YWlucyB0aGUgZnJvbnRlbmQgYW5kIGJhY2tlbmQgYXBwbGljYXRpb25zLgotIGBwYWNrYWdlcy8qYDogSW5jbHVkZXMgc2hhcmVkIGxpYnJhcmllcywgY29uZmlndXJhdGlvbnMsIGFuZCB1dGlsaXR5IGZ1bmN0aW9ucy4KCiMjIEFkZGl0aW9uYWwgUmVzb3VyY2VzCgotIFtUdXJibyBSZXBvIERvY3VtZW50YXRpb25dKGh0dHBzOi8vdHVyYm9yZXBvLm9yZy9kb2NzKQotIFtQcmlzbWEgRG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly93d3cucHJpc21hLmlvL2RvY3MvKQotIFtPcGVuQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykKLSBbVHlwZVNjcmlwdF0oaHR0cHM6Ly93d3cudHlwZXNjcmlwdGxhbmcub3JnL2RvY3MvKQotIFtQb3N0Z3JlcyBEb2NrZXIgU2V0dXBdKGh0dHBzOi8vZG9jcy5kb2NrZXIuY29tL3NhbXBsZXMvcG9zdGdyZXNxbF9zZXJ2aWNlLykKCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyBhcmUgd2VsY29tZSEgUGxlYXNlIGZvbGxvdyB0aGUgZXN0YWJsaXNoZWQgY29kaW5nIGNvbnZlbnRpb25zIGFuZCBwdWxsIHJlcXVlc3QgcHJvY2Vzcy4KCi0tLQoKVGhhbmsgeW91IGZvciBjaG9vc2luZyBPeHlUcmFjayBUdXJibyBSZXBvIGZvciB5b3VyIGRldmVsb3BtZW50IG5lZWRzLiBIYXBweSBjb2RpbmchCg== readmeEtag: '"02a9c426d98b5c99dec855d7b8b34c296f9e2ef7"' readmeLastModified: Sat, 20 Apr 2024 13:46:30 GMT repositoryId: 760678504 description: >- A robust full stack application that includes Prisma, Postgres, Express, Node.js, React, and TypeScript across both the frontend and backend, ensuring a strongly typed development environment. created: '2024-02-20T18:26:55Z' updated: '2024-04-20T13:47:13Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: Rakesh5777 logo: https://avatars.githubusercontent.com/u/81339756?v=4 repoEtag: '"529310cfa8753b517d116022538eab5005d5cb104c42af7cdcd820b8e9a84bb9"' repoLastModified: Sat, 20 Apr 2024 13:47:13 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/jumaevkova/openapi2postman v3: true id: a435703b9a6189963a308531a67538c0 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpMnBvc3RtYW4gYWN0aW9uCgpHaXRIdWIgYWN0aW9uIHRvIHB1c2ggb3BlbmFwaSAoT3BlbkFQSSAzLjAsIDMuMSBhbmQgU3dhZ2dlciAyLjApIGZpbGUgdG8gUG9zdG1hbiBkaXJlY3RseSBmcm9tIHlvdXIgd29ya2Zsb3cKCiMjIFVzYWdlCgpBZGQgdGhlIG9wZW5hcGkycG9zdG1hbiBhY3Rpb24gYW5kIHNldCB0aGUgcmVxdWlyZWQgaW5wdXRzCgoqIGBhcGkta2V5YDogeW91ciBQb3N0bWFuIEFQSSBrZXkKKiBgd29ya3NwYWNlLWlkYDogeW91ciBQb3N0bWFuIHdvcmtzcGFjZSBpZAoqIGBjb2xsZWN0aW9uLWlkYDogeW91ciBQb3N0bWFuIGNvbGxlY3Rpb24gaWQKKiBgZmlsZWA6IHlvdXIgb3BlbmFwaSBqc29uIGZpbGUKCiMjIyBGaWxlIGZvcm1hdHMgYXZhaWxhYmxlCgpgLmpzb25gICAgCmAueWFtbGAgICAKYC55bWxgCgojIyMgVXBkYXRlIGV4aXN0aW5nIGNvbGxlY3Rpb24KClVwZGF0ZSBleGlzdGluZyBQb3N0bWFuIGNvbGxlY3Rpb24KCmBgYHlhbWwKICAgIC0gbmFtZTogIlVwZGF0ZSBQb3N0bWFuIENvbGxlY3Rpb24iCiAgICAgIHVzZXM6IGp1bWFldmtvdmEwNC9vcGVuYXBpMnBvc3RtYW5AbWFpbgogICAgICB3aXRoOgogICAgICAgIGFwaS1rZXk6ICR7eyBzZWNyZXRzLlBPU1RNQU5fQVBJX0tFWSB9fQogICAgICAgIHdvcmtzcGFjZS1pZDogJHt7IHNlY3JldHMuUE9TVE1BTl9XT1JLU1BBQ0VfSUQgfX0KICAgICAgICBjb2xsZWN0aW9uLWlkOiAke3sgc2VjcmV0cy5QT1NUTUFOX0NPTExFQ1RJT05fSUQgfX0KICAgICAgICBmaWxlOiAuL2RvY3Mvc3dhZ2dlci5qc29uCmBgYAoKIyMjIEV4YW1wbGUgd29ya2Zsb3cgZmlsZQoKVXBkYXRlIFBvc3RtYW4gY29sbGVjdGlvbnMgb24gYHB1c2hgCgpgYGB5YW1sCm5hbWU6ICJVcGRhdGUgUG9zdG1hbiBjb2xsZWN0aW9uIgoKb246CiAgcHVzaDoKICAgIGJyYW5jaGVzOiBbICJtYWluIiBdCgpqb2JzOgogIHN5bmMtZG9jdW1lbnRhdGlvbi13aXRoLXBvc3RtYW46CiAgICBydW5zLW9uOiB1YnVudHUtbGF0ZXN0CiAgICBzdGVwczoKICAgICAgLSBuYW1lOiAiQ2hlY2tvdXQgcmVwb3NpdG9yeSIKICAgICAgICB1c2VzOiBhY3Rpb25zL2NoZWNrb3V0QHYzCgogICAgICAtIG5hbWU6ICJVcGRhdGUgUG9zdG1hbiBDb2xsZWN0aW9uIgogICAgICAgIHVzZXM6IGp1bWFldmtvdmEwNC9vcGVuYXBpMnBvc3RtYW5AbWFpbgogICAgICAgIHdpdGg6CiAgICAgICAgICBhcGkta2V5OiAke3sgc2VjcmV0cy5QT1NUTUFOX0FQSV9LRVkgfX0KICAgICAgICAgIHdvcmtzcGFjZS1pZDogJHt7IHNlY3JldHMuUE9TVE1BTl9XT1JLU1BBQ0VfSUQgfX0KICAgICAgICAgIGNvbGxlY3Rpb24taWQ6ICR7eyBzZWNyZXRzLlBPU1RNQU5fQ09MTEVDVElPTl9JRCB9fQogICAgICAgICAgZmlsZTogLi9kb2NzL3N3YWdnZXIuanNvbgpgYGA= readmeEtag: '"3605027ed82f649140af3e5883f62f5b49823954"' readmeLastModified: Wed, 19 Jun 2024 02:46:33 GMT repositoryId: 712975783 description: Update Postman collection from openapi file created: '2023-11-01T15:33:20Z' updated: '2025-07-31T12:54:39Z' language: Go archived: false stars: 1 watchers: 1 forks: 0 owner: jumaevkova logo: https://avatars.githubusercontent.com/u/68679772?v=4 repoEtag: '"b783cad0b895dbfa52d0b837995d1cbfb87d8b13565a116af2e4b3f959e08a79"' repoLastModified: Thu, 31 Jul 2025 12:54:39 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/jumaevkova04/openapi2postman - source: openapi3 tags repository: https://github.com/edo1z/rust-axum-seaorm-sample v3: true id: 2129199cc4f3c61ca889912962cd6c67 repositoryMetadata: base64Readme: >- IyBydXN0LWF4dW0tc2Vhb3JtLXNhbXBsZQojIyBMb2NhbCBTZXR1cAoKYGBgCj4gY2QgbG9jYWwKPiBkb2NrZXItY29tcG9zZSB1cApgYGAKCi0gQVBJOiBodHRwOi8vbG9jYWxob3N0OjgwODUKLSBTd2FnZ2VyOiBodHRwOi8vbG9jYWxob3N0OjgwMDEK readmeEtag: '"363caea125308295e1deae7d4427c6478ad176e1"' readmeLastModified: Tue, 05 Jul 2022 16:15:43 GMT repositoryId: 504863913 description: Rust Axum+SeaORM Sample created: '2022-06-18T14:21:30Z' updated: '2025-09-26T07:13:02Z' language: Rust archived: false stars: 2 watchers: 1 forks: 0 owner: edo1z logo: https://avatars.githubusercontent.com/u/89882017?v=4 repoEtag: '"8785ee2c786e252a4e727283e8de687401ba849ce0bc4f594148940c37340cd5"' repoLastModified: Fri, 26 Sep 2025 07:13:02 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/net3i/rust-axum-seaorm-sample - source: openapi3 tags repository: https://github.com/nautilustube/swagger2doc v3: true id: 8bb453efb5a2da797fe14a6777929939 repositoryMetadata: base64Readme: >- IyBDb252ZXJ0IFN3YWdnZXIgSlNPTiB0byBNYXJrRG93biwgSHRtbCBBbmQgTVMgV29yZCBEb2N1bWVudCguZG9jeCkKQ29udmVydHMgU3dhZ2dlciBKU09OIGZpbGUgdG8gTWFya0Rvd24sIE1hcmtEb3duIEh0bWwsIE1TIFdvcmQgRG9jdW1lbnQgCgojIyBVc2FnZQoKIyMjIDEuIGZvciBSZWJ1aWxkIHRoZSBleGUKU3dhZ2dlcjJEb2MuZXhlIHdpbGwgYmUgY3JlYXRlZCBpbiAgL3B1Ymxpc2ggZm9sZGVyCgpgYGBQb3dlclNoZWxsCmV4ZWN1dGUgdGhlIGNvbXBpbGUucHMxIGF0IHRoZSBwYXRoCmBgYAojIyMgMi4gY3JlYXRlIGFwaSBmaWxlcwphZGQgeW91ciBzd2FnZ2VyLmpzb24gZmlsZSBpbiAvcHVibGlzaC9SZXNvdXJjZSBmb2xkZXIgYW5kIGV4ZWN1dGUgU3dhZ2dlcjJEb2MuZXhlCgpzd2FnZ2VyLmRvY3jCoUJzd2FnZ2VyLmh0bWzCoUJzd2FnZ2VyLm1kIHdpbGwgYmUgY3JlYXRlZCBpbiBzYW1lIGZvbGRlcgoKIyMjIDMuIHlvdXIgYXBpIHNwZWMgZmlsZSBleGFtcGxlCgpbbWRdKHB1Ymxpc2gvUmVzb3VyY2VzL091dHB1dC9zd2FnZ2VyLm1kKQpbaHRtbF0ocHVibGlzaC9SZXNvdXJjZXMvT3V0cHV0L3N3YWdnZXIuaHRtbCkKW2RvY3hdKHB1Ymxpc2gvUmVzb3VyY2VzL091dHB1dC9zd2FnZ2VyLmRvY3gpCg== readmeEtag: '"a219a168909fff8d6604f682387d87713a70a1a4"' readmeLastModified: Sun, 22 Sep 2024 11:47:21 GMT repositoryId: 835916567 description: Convert Swagger JSON to MarkDown, Html And MS Word Document(.docx) created: '2024-07-30T19:28:54Z' updated: '2026-01-23T10:09:16Z' language: C# archived: false stars: 3 watchers: 1 forks: 0 owner: nautilustube logo: https://avatars.githubusercontent.com/u/101971417?v=4 repoEtag: '"588cf52f257f3e314592845acb606661ac293bc937becc86ce9b4026a68214a8"' repoLastModified: Fri, 23 Jan 2026 10:09:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/sinoroc/rapids v3: true repositoryMetadata: base64Readme: >- Li4KCgouLiBjb250ZW50czo6CgouLiBzZWN0bnVtOjoKCgpJbnRyb2R1Y3Rpb24KPT09PT09PT09PT09CgoqKlJFU1QgQVBJcyBkb2N1bWVudGVkIGFuZCBzZW5zaWJsZSoqCgpCdWlsZCBhdXRvbWF0aWNhbGx5IGRvY3VtZW50ZWQgUkVTVCBBUElzIHdpdGggdGhlIGBQeXJhbWlkIGZyYW1ld29ya2BfLgoKVGhpcyBsaWJyYXJ5IGlzIGF2YWlsYWJsZSBvbiB0aGUgUHl0aG9uIHBhY2thZ2UgaW5kZXggdW5kZXIgdGhlIHByb2plY3QgbmFtZQpgYHJhcGlkc2BgOgoKKiBodHRwczovL3B5cGkucHl0aG9uLm9yZy9weXBpL3JhcGlkcwoKKiBodHRwczovL3B5cGkub3JnL3Byb2plY3QvcmFwaWRzLwoKClVzYWdlCj09PT09CgouLiBjb2RlOjogcHl0aG9uCgogICAgQHJhcGlkcy5kZWNvcmF0b3JzLnJlc291cmNlKCcnLCBOb25lKQogICAgY2xhc3MgUm9vdChyYXBpZHMucmVzb3VyY2VzLkJhc2UpOgogICAgICAgIHBhc3MKCgogICAgQHB5cmFtaWQudmlldy52aWV3X2NvbmZpZyhjb250ZXh0PVJvb3QsIHJhcGlkcz0nJykKICAgIGRlZiByb290X3ZpZXcocmVzb3VyY2UsIHJlcXVlc3QpOgogICAgICAgIHJldHVybiBweXJhbWlkLmh0dHBleGNlcHRpb25zLkhUVFBOb3RGb3VuZCgpCgoKICAgIEByYXBpZHMuZGVjb3JhdG9ycy5yZXNvdXJjZSgnZm9vJywgUm9vdCkKICAgIGNsYXNzIEZvbyhyYXBpZHMucmVzb3VyY2VzLkJhc2UpOgogICAgICAgIHBhc3MKCgogICAgQHB5cmFtaWQudmlldy52aWV3X2RlZmF1bHRzKGNvbnRleHQ9Rm9vLCByYXBpZHM9JycpCiAgICBjbGFzcyBGb29WaWV3OgogICAgICAgIGRlZiBfX2luaXRfXyhzZWxmLCByZXNvdXJjZSwgcmVxdWVzdCk6CiAgICAgICAgICAgIHBhc3MKCiAgICAgICAgQHB5cmFtaWQudmlldy52aWV3X2NvbmZpZyhyZXF1ZXN0X21ldGhvZD0nR0VUJykKICAgICAgICBkZWYgX2dldF92aWV3KHNlbGYpOgogICAgICAgICAgICByZXR1cm4gcHlyYW1pZC5odHRwZXhjZXB0aW9ucy5IVFRQT2soKQoKICAgICAgICBAcHlyYW1pZC52aWV3LnZpZXdfY29uZmlnKHJlcXVlc3RfbWV0aG9kPSdQT1NUJykKICAgICAgICBkZWYgX3Bvc3RfdmlldyhzZWxmKToKICAgICAgICAgICAgcmV0dXJuIHB5cmFtaWQuaHR0cGV4Y2VwdGlvbnMuSFRUUENyZWF0ZWQoKQoKCkhhY2tpbmcKPT09PT09PQoKVGhpcyBwcm9qZWN0IG1ha2VzIGV4dGVuc2l2ZSB1c2Ugb2YgYHRveGBfLCBgcHl0ZXN0YF8sIGFuZCBgR05VIE1ha2VgXy4KCgpEZXZlbG9wbWVudCBlbnZpcm9ubWVudAotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQoKVXNlIGZvbGxvd2luZyBjb21tYW5kIHRvIGNyZWF0ZSBhIFB5dGhvbiB2aXJ0dWFsIGVudmlyb25tZW50IHdpdGggYWxsCm5lY2Vzc2FyeSBkZXBlbmRlbmNpZXM6OgoKICAgIHRveCAtLXJlY3JlYXRlIC1lIGRldmVsb3AKClRoaXMgY3JlYXRlcyBhIFB5dGhvbiB2aXJ0dWFsIGVudmlyb25tZW50IGluIHRoZSBgYC50b3gvZGV2ZWxvcGBgIGRpcmVjdG9yeS4gSXQKY2FuIGJlIGFjdGl2YXRlZCB3aXRoIHRoZSBmb2xsb3dpbmcgY29tbWFuZDo6CgogICAgLiAudG94L2RldmVsb3AvYmluL2FjdGl2YXRlCgoKUnVuIHRlc3Qgc3VpdGUKLS0tLS0tLS0tLS0tLS0KCkluIGEgUHl0aG9uIHZpcnR1YWwgZW52aXJvbm1lbnQgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZDo6CgogICAgbWFrZSByZXZpZXcKCk91dHNpZGUgb2YgYSBQeXRob24gdmlydHVhbCBlbnZpcm9ubWVudCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kOjoKCiAgICB0b3ggLS1yZWNyZWF0ZQoKCkJ1aWxkIGFuZCBwYWNrYWdlCi0tLS0tLS0tLS0tLS0tLS0tCgpJbiBhIFB5dGhvbiB2aXJ0dWFsIGVudmlyb25tZW50IHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6OgoKICAgIG1ha2UgcGFja2FnZQoKT3V0c2lkZSBvZiBhIFB5dGhvbiB2aXJ0dWFsIGVudmlyb25tZW50IHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6OgoKICAgIHRveCAtLXJlY3JlYXRlIC1lIHBhY2thZ2UKCgouLiBMaW5rcwoKLi4gX2BHTlUgTWFrZWA6IGh0dHBzOi8vd3d3LmdudS5vcmcvc29mdHdhcmUvbWFrZS8KLi4gX2BQeXJhbWlkIGZyYW1ld29ya2A6IGh0dHBzOi8vdHJ5cHlyYW1pZC5jb20vCi4uIF9gcHl0ZXN0YDogaHR0cDovL3B5dGVzdC5vcmcvCi4uIF9gdG94YDogaHR0cHM6Ly90b3gucmVhZHRoZWRvY3MuaW8vCgoKLi4gRU9GCg== readmeEtag: '"ed92292f4de45d389528ff53f8694ca541b1ec25"' readmeLastModified: Sun, 14 Apr 2019 15:06:49 GMT repositoryId: 108432117 description: REST APIs documented and sensible created: '2017-10-26T15:44:17Z' updated: '2024-12-11T12:38:46Z' language: Python archived: false stars: 1 watchers: 1 forks: 3 owner: sinoroc logo: https://avatars.githubusercontent.com/u/5529267?v=4 license: Apache-2.0 repoEtag: '"9bbd8d04b745d3fc0f8a3a30549f3a3798d7be12fdcf40db361b0ec477459d4c"' repoLastModified: Wed, 11 Dec 2024 12:38:46 GMT foundInMaster: true category: Server Implementations id: fa2497dec16a48c00bff1a36601fec89 - source: openapi3 tags repository: https://github.com/blueconic/openapi v3: true id: 1bdf662814e8d18010a84f46df9f5114 repositoryMetadata: base64Readme: >- IyBCbHVlQ29uaWMgT3BlbkFQSSBTcGVjaWZpY2F0aW9uClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGZvciB0aGUgQmx1ZUNvbmljIFJFU1QgQVBJIHYyLgogCiAjIFVzaW5nIHRoZSBsYXRlc3QgcHVibGlzaGVkIE9wZW5BUEkgc3BlY2lmaWNhdGlvbgogKiBPcGVuIGh0dHBzOi8vcmVzdC5hcGlkb2MuYmx1ZWNvbmljLmNvbQogCiAjIFVzaW5nIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gb2YgYSBnaXZlbiBCbHVlQ29uaWMgdGVuYW50CiAqIE9wZW4gaHR0cHM6Ly9yZXN0LmFwaWRvYy5ibHVlY29uaWMuY29tLz9ob3N0bmFtZT1teXRlbmFudG5hbWUuYmx1ZWNvbmljLm5ldAogKiBPciBvcGVuIGh0dHBzOi8vbXl0ZW5hbnRuYW1lLmJsdWVjb25pYy5uZXQvb3BlbmFwaQogCiAjIFJ1bm5pbmcgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBvbiBhIGxvY2FsIG1hY2hpbmUKICogQ2xvbmUgdGhlIHJlcG9zaXRvcnkgYW5kIGBjZGAgdG8gdGhlIHJlcG9zaXRvcnkgbG9jYXRpb24KICogUnVuIGBucHggaHR0cC1zZXJ2ZXJgCiAqIE9wZW4gYGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MD9ob3N0bmFtZT1sb2NhbGhvc3RgCiAqIENPUlMgY29uZmlndXJhdGlvbiBvZiBCbHVlQ29uaWMgbmVlZHMgdG8gYmUgYWRqdXN0ZWQgdG8gdGhlIHRoZSBsb2NhbGhvc3Qgb3JpZ2luIChgaHR0cDovL2xvY2FsaG9zdDo4MDgwYCBieSBkZWZhdWx0KQogCiAjIFNwZWNpZnlpbmcgQkMgdmVyc2lvbgogVGhlIGB2ZXJzaW9uYCBxdWVyeXN0cmluZyBwYXJhbWV0ZXIgY2FuIGJlIHVzZWQgdG8gc3BlY2lmeSBhIHNwZWNpZmljIEJsdWVDb25pYyB2ZXJzaW9uLiBOb3RlIHRoYXQgcGFyYW1ldGVyIGlzIGV4Y2x1c2l2ZSB3aXRoIHRoZSBgaG9zdG5hbWVgIHBhcmFtZXRlciwgd2hpY2ggd2lsbCBhbHdheXMgbG9hZCB0aGUgdmVyc2lvbiBzcGVjaWZpZWQgYnkgdGhlIHRlbmFudCBiZWluZyBjb25uZWN0ZWQgdG8uCiAKICogaHR0cHM6Ly9yZXN0LmFwaWRvYy5ibHVlY29uaWMuY29tLz92ZXJzaW9uPTg5IG9yIGh0dHBzOi8vcmVzdC5hcGlkb2MuYmx1ZWNvbmljLmNvbS8/dmVyc2lvbj1yODkgd2lsbCBsb2FkIHZlcnNpb24gODkKICogaHR0cHM6Ly9yZXN0LmFwaWRvYy5ibHVlY29uaWMuY29tLz92ZXJzaW9uPTkxIG9yIGh0dHBzOi8vcmVzdC5hcGlkb2MuYmx1ZWNvbmljLmNvbS8/dmVyc2lvbj1yOTEgd2lsbCBsb2FkIHZlcnNpb24gOTEKICogaHR0cHM6Ly9yZXN0LmFwaWRvYy5ibHVlY29uaWMuY29tLz92ZXJzaW9uPWN1cnJlbnQgd2lsbCBsb2FkIGxhdGVzdCB2ZXJzaW9uIHB1Ymxpc2hlZC4gQmVoYXZlcyB0aGUgc2FtZSBhcyBub3Qgc3BlY2lmeWluZyBhIHZlcnNpb24gYXQgYWxsLgogCg== readmeEtag: '"f8ee2eaa1a945ac1de3fa1f462a49d346d0fdd5d"' readmeLastModified: Fri, 28 Jun 2024 07:49:45 GMT repositoryId: 595013499 description: 'BlueConic REST API v2 ' created: '2023-01-30T08:01:31Z' updated: '2026-01-30T14:06:48Z' language: JavaScript archived: false stars: 1 watchers: 7 forks: 0 owner: blueconic logo: https://avatars.githubusercontent.com/u/5948030?v=4 license: NOASSERTION repoEtag: '"42e8c03c48c1445377a04de7dce047e8d5e78b193094413657df260e7d9276c1"' repoLastModified: Fri, 30 Jan 2026 14:06:48 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/lordent/oad v3: true repositoryMetadata: base64Readme: >- IyBQeXRob24gbGlicmFyeSBmb3IgZ2VuZXJhdGUgdmFsaWQgT3BlbkFQSSB2MyBzcGVjaWZpY2F0aW9uCgpbCiFbdHJhdmlzXShodHRwczovL2FwaS50cmF2aXMtY2kub3JnL2xvcmRlbnQvb2FkLnN2Zz9icmFuY2g9bWFzdGVyKQpdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9sb3JkZW50L29hZCkKClJlYWQgT3BlbkFQSSBzcGVjaWZpY2F0aW9uOgotIFtzd2FnZ2VyXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pCi0gW2dpdGh1Yl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjAubWQpCgpVc2FnZSBleGFtcGxlOgoKYGBgcHl0aG9uCmltcG9ydCBhc3luY2lvCmZyb20gb2FkIGltcG9ydCBvcGVuYXBpLCBPcGVuQVBJRG9jCgoKaW5mbyA9IHsKICAgICd0aXRsZSc6ICdUZXN0JywKICAgICdkZXNjcmlwdGlvbic6ICdUZXN0IGFwaSBkZXNjcmlwdGlvbicsCiAgICAndGVybXNPZlNlcnZpY2UnOiAnVGVzdCB0ZXJtcycsCiAgICAnY29udGFjdCc6IHsKICAgICAgICAnbmFtZSc6ICdUZXN0ZXInLAogICAgICAgICd1cmwnOiAnaHR0cDovL2V4YW1wbGUuY29tJywKICAgICAgICAnZW1haWwnOiAndGVzdEBleGFtcGxlLmNvbScKICAgIH0sCiAgICAnbGljZW5zZSc6IHsKICAgICAgICAnbmFtZSc6ICdBcGFjaGUgMi4wJywKICAgICAgICAndXJsJzogJ2h0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMC5odG1sJwogICAgfSwKICAgICd2ZXJzaW9uJzogJzEuMCcKfQoKZXJyb3Jfc2NoZW1hID0gewogICAgJ3R5cGUnOiAnb2JqZWN0JywKICAgICdwcm9wZXJ0aWVzJzogewogICAgICAgICd0eXBlJzogeyd0eXBlJzogJ3N0cmluZyd9LAogICAgICAgICdtZXNzYWdlJzogeyd0eXBlJzogJ3N0cmluZyd9LAogICAgICAgICdlcnJvcnMnOiB7J3R5cGUnOiAnb2JqZWN0J30KICAgIH0sCn0KCmVycm9yX3Jlc3BvbnNlID0gewogICAgJ2Rlc2NyaXB0aW9uJzogJ0Vycm9yIHJlc3BvbnNlJywKICAgICdjb250ZW50JzogewogICAgICAgICdhcHBsaWNhdGlvbi9qc29uJzogewogICAgICAgICAgICAnc2NoZW1hJzogewogICAgICAgICAgICAgICAgJyRyZWYnOiAnIy9jb21wb25lbnRzL3NjaGVtYXMvRXJyb3InCiAgICAgICAgICAgIH0KICAgICAgICB9CiAgICB9Cn0KCkBvcGVuYXBpLmRvYyh7CiAgICAnc3VtbWFyeSc6ICdUZXN0IHN1bW1hcnkgdGV4dCcsCiAgICAnZGVzY3JpcHRpb24nOiAnVGVzdCBkZXNjcmlwdGlvbicsCiAgICAndGFncyc6IFsndGVzdCddLAogICAgJ3BhcmFtZXRlcnMnOiBbewogICAgICAgICckcmVmJzogJyMvY29tcG9uZW50cy9wYXJhbWV0ZXJzL1Rlc3RQYXJhbWV0ZXInCiAgICB9XSwKfSkKQG9wZW5hcGkucmVxdWVzdCh7CiAgICAnZGVzY3JpcHRpb24nOiAnVGVzdCBkZXNjcmlwdGlvbicsCn0sIGNvbnRlbnRfZG9jdW1lbnRhdGlvbj17CiAgICAnZXhhbXBsZSc6IHsKICAgICAgICAnaWQnOiAnNCcsCiAgICB9LAp9LCBzY2hlbWE9ewogICAgJ3R5cGUnOiAnb2JqZWN0JywKICAgICdwcm9wZXJ0aWVzJzogewogICAgICAgICdpZCc6IHsKICAgICAgICAgICAgJ3R5cGUnOiAnaW50ZWdlcicsCiAgICAgICAgICAgICdmb3JtYXQnOiAnaW50NjQnLAogICAgICAgICAgICAnZXhhbXBsZSc6ICc0JywKICAgICAgICB9LAogICAgfSwKfSkKQG9wZW5hcGkucmVzcG9uc2UoKQpAb3BlbmFwaS5yZXNwb25zZSgKICAgIHN0YXR1cz00MDAsIHNjaGVtYT17JyRyZWYnOiAnIy9jb21wb25lbnRzL3NjaGVtYXMvRXJyb3InfSkKYXN5bmMgZGVmIHRlc3RfaGFuZGxlcigqYXJncywgKiprd2FyZ3MpOgogICAgYXdhaXQgYXN5bmNpby5zbGVlcCgwKQogICAgcmV0dXJuICdPayEnLCBhcmdzLCBrd2FyZ3MKCgpkb2MgPSAoCiAgICBPcGVuQVBJRG9jKHsKICAgICAgICAnaW5mbyc6IGluZm8sCiAgICB9KQogICAgLmFkZF9wYXJhbWV0ZXIoJ1Rlc3RQYXJhbWV0ZXInKQogICAgLmFkZF90YWcoJ3Rlc3QnLCB7J2Rlc2NyaXB0aW9uJzogJ1Rlc3QgdGFnIGRlc2NyaXB0aW9uJ30pCiAgICAuYWRkX3BhdGgoCiAgICAgICAgJy90ZXN0L3tUZXN0UGFyYW1ldGVyfScsICdwb3N0JywKICAgICAgICB0ZXN0X2hhbmRsZXIuX19vcGVuYXBpX18uZG9jdW1lbnRhdGlvbikKICAgIC5hZGRfc2NoZW1hKCdFcnJvcicsIGVycm9yX3NjaGVtYSkKICAgIC5hZGRfcmVzcG9uc2UoJ0Vycm9yJywgZXJyb3JfcmVzcG9uc2UpCiAgICAuYWRkX3NlY3VyaXR5KCdBcGlLZXknLCAnYXBpS2V5JywgewogICAgICAgICdpbic6ICdoZWFkZXInLAogICAgICAgICduYW1lJzogJ1gtQVBJLUtleScsCiAgICB9KQogICAgLnRvX2RpY3QoKQopCmBgYAo= readmeEtag: '"264bf431e976e1dfadca356518ef3d1d27401cd9"' readmeLastModified: Sun, 08 Apr 2018 10:33:42 GMT repositoryId: 128617517 description: Python library for generate valid OpenAPI v3 specification created: '2018-04-08T08:21:11Z' updated: '2019-02-15T19:24:30Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: lordent logo: https://avatars.githubusercontent.com/u/2070895?v=4 license: Apache-2.0 repoEtag: '"45a2f0be596fbbcf86f05810fdfe6f427f92573b0de5af591301dc5a41d1c53f"' repoLastModified: Fri, 15 Feb 2019 19:24:30 GMT foundInMaster: true category: Parsers id: f2ed564d12c5e6d231dcce7953cd3d75 - source: openapi3 tags repository: https://github.com/ndexbio/openapi-specification v3: true repositoryMetadata: base64Readme: >- CiMgT3BlbkFQSS1TcGVjaWZpY2F0aW9uClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBhIGRpZ2VzdCBvZiBPcGVuQVBJIHNwZWNpZmljYXRpb24gZG9jdW1lbnRzIGZvciB0aGUgKipOKipldHdvcmsgKipEKiphdGEgKipFeCoqY2hhbmdlIAoqKihOREV4KSBSRVNUIEFQSSoqIHVzZWQgdG8gY29tbXVuaWNhdGUgd2l0aCB0aGUgW05ERXggUHJvamVjdF0oaHR0cDovL3d3dy5ob21lLm5kZXhiaW8ub3JnL2luZGV4LykgcHVibGljCnNlcnZlciBhbmQgcmVsYXRlZCBOREV4IHByaXZhdGUgc2VydmVycy4gVGhlIE5ERXggUHJvamVjdCBwcm92aWRlcyBhbiBvcGVuLXNvdXJjZSBmcmFtZXdvcmsgYW5kIG9ubGluZSBkYXRhYmFzZSAKY29tbW9ucyB3aGVyZSBzY2llbnRpc3RzIGNhbiB1cGxvYWQsIHNoYXJlLCBhbmQgcHVibGljbHkgZGlzdHJpYnV0ZSBiaW9sb2dpY2FsIG5ldHdvcmtzCiAqKFByYXR0IGV0IGFsLiwgMjAxNSwgQ2VsbCBTeXN0ZW1zIDEsIDMwMi0zMDUsIE9jdG9iZXIgMjgsIDIwMTUgwqkyMDE1IEVsc2V2aWVyIEluYy4KICBbU2NpZW5jZURpcmVjdF0oaHR0cDovL3d3dy5zY2llbmNlZGlyZWN0LmNvbS9zY2llbmNlL2FydGljbGUvcGlpL1MyNDA1NDcxMjE1MDAxNDc3KSkqLgoKT3BlbkFQSSBpcyBhbiBvcGVuLCBwb3J0YWJsZSBhbmQgdmVuZG9yIG5ldXRyYWwgQVBJIGRlc2NyaXB0aW9uIHN0YW5kYXJkLCBhbmQgaXMgZGVmaW5lZCBieSB0aGUgTGludXggCkZvdW5kYXRpb24gW09wZW5BUEkgSW5pdGlhdGl2ZSAoT0FJKV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnLykgZ292ZXJuaW5nIGJvZHkuIEl0IGlzIGVtcGxveWVkIHdvcmxkd2lkZSBieSAKY29kZSBnZW5lcmF0aW9uLCB0ZXN0aW5nIGFuZCBkb2N1bWVudCBjcmVhdGlvbiB0b29scywgbWFueSBvZiB3aGljaCBhcmUgbGlzdGVkIAogW2hlcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFzdGVyL0lNUExFTUVOVEFUSU9OUy5tZCkuCiAKVGhlIE9wZW5BUEkgZG9jdW1lbnRzIGNvbnRhaW5lZCBpbiB0aGlzIHJlcG9zaXRvcnkgYXJlIHNwZWNpZmljYWxseSB1c2VkIGluIGNvbmp1bmN0aW9uIHdpdGggdGhlIApbbmRleHRvb2xzL29wZW5hcGktcmVhZGVyXShodHRwczovL2dpdGh1Yi5jb20vbmRleHRvb2xzL29wZW5hcGktcmVhZGVyKSBhcHBsaWNhdGlvbiwgc28gdGhhdCBOREV4IGRldmVsb3BlcnMgCmNhbiBpbnRlcmFjdGl2ZWx5IHZpZXcgYW5kICAidHJ5IG91dCIgTkRFeCBBUEkgcmVxdWVzdHMsIHRodXMgcHJvdmlkaW5nIHRoZW0gd2l0aCBhIGxvdy1sZXZlbCBBUEkgCmFuZCBkZWJ1Z2dpbmcgZW52aXJvbm1lbnQgYXMgdGhleSBjcmVhdGUgbGFuZ3VhZ2Utc3BlY2lmaWMgc29mdHdhcmUgYXBwbGljYXRpb25zIGFuZCBwcm9ncmFtIGxpYnJhcmllcy4KCkV4Y2VwdCBmb3IgdGhlc2UgZGVyaXZhdGl2ZSBPcGVuQVBJIGRvY3VtZW50cywgYWxsIHJlZmVyZW5jZXMgdG8gdGhlIE5ERXggQVBJLCBzb3VyY2VzIGFuZCBkb2N1bWVudGF0aW9uIAphcmUgY29weXJpZ2h0ZWQ6ICrCqSAyMDEzLTIwMjAsIFRoZSBSZWdlbnRzIG9mIHRoZSBVbml2ZXJzaXR5IG9mIENhbGlmb3JuaWEsIFRoZSBDeXRvc2NhcGUgQ29uc29ydGl1bS4KQWxsIHJpZ2h0cyByZXNlcnZlZC4qICBQbGVhc2UgYWJpZGUgd2l0aCB0aGVpcgpbVGVybXMgb2YgVXNlLCBMaWNlbnNpbmcgYW5kIFNvdXJjZXNdKGh0dHA6Ly93d3cuaG9tZS5uZGV4YmlvLm9yZy9kaXNjbGFpbWVyLWxpY2Vuc2UvKS4KTGlrZXdpc2UsIHRoZSBbU3dhZ2dlci1VSV0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkpIGRvY3VtZW50IHJlYWRlciB0aGF0IGRpc3BsYXlzIHRoaXMgT3BlbkFQSSAKZG9jdW1lbnQgaXMgY29weXJpZ2h0ZWQgYnkgKlNtYXJ0YmVhciBTb2Z0d2FyZSouIEl0cyBvcGVuLXNvdXJjZSBzb2Z0d2FyZSBsaWNlbnNlIGlzIGZvdW5kIApbaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkvYmxvYi9tYXN0ZXIvTElDRU5TRSkuCgpUaGUgY3VycmVudGx5IHN1cHBvcnRlZCBSRVNUIEFQSSBhbmQgc3VwcG9ydGluZyBkb2N1bWVudGF0aW9uIAppcyBmb3VuZCBhdCB0aGUgW05ERXggUHJvamVjdCB3ZWJzaXRlXShodHRwOi8vd3d3LmhvbWUubmRleGJpby5vcmcvaW5kZXgvKS4K readmeEtag: '"03e0423075a14aefecc08c229b22258c02f41532"' readmeLastModified: Thu, 17 Dec 2020 19:16:38 GMT repositoryId: 117026992 description: >- OpenAPI specification for the ndexbio.org rest server api and used by ndextools/openapi-reader. This repository is created by Mike Fidel and transferred to NDEx project in March 2018. created: '2018-01-11T00:10:49Z' updated: '2020-12-17T19:16:41Z' language: null archived: false stars: 1 watchers: 7 forks: 1 owner: ndexbio logo: https://avatars.githubusercontent.com/u/8965609?v=4 license: MIT repoEtag: '"4a6fc65b3fa22d96c910b08da9456b8fef5bc4aa72c2c6222361e77030dab00e"' repoLastModified: Thu, 17 Dec 2020 19:16:41 GMT foundInMaster: true category: Parsers id: 3b65be6a3615646d98a00c8f8617901a - source: openapi3 tags repository: https://github.com/mariotoffia/openapi-senml v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIDMuMCBTZW5NTCBTcGVjaWZpY2F0aW9uCkEgT3BlbiBBUEkgMy54IHNwZWNpZmljYXRpb24gKHN3YWdnZXIgbmV4dCBnZW4pIGZvciBTZW5NTCAoU2Vuc29yIE1lYXN1cmVtZW50IExpc3QpIGZvciBzZW5zb3IgKElvVCkgY29tbXVuaWNhdGlvbiBhbmQgYWN0dWF0aW9uIChodHRwczovL3Rvb2xzLmlldGYub3JnL2h0bWwvcmZjODQyOCkuCgojIyBHb2FsClRoaXMgc3BlY2lmaWNhdGlvbiBzdHJpdmVzIHRvIGVuY2Fwc3VsYXRlIHRoZSBbU2VuTUxdKGh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmM4NDI4KSBmdW5jdGlvbmFsaXR5IHVzaW5nIGEgcGxhaW4gT3BlbkFQSSAzLnggc3BlY2lmaWNhdGlvbiBhbmQgYSBzYW1wbGUgaW1wbGVtZW50YXRpb24gaW4gQyMsIEtvdGxpbiwgQywgYW5kIEphdmFzY3JpcHQgYm90aCBvbiBzZXJ2ZXIgYW5kIGNsaWVudCBzaWRlLgo= readmeEtag: '"e5a309984cfe63cfc070d4327c0b95bb8cb527f9"' readmeLastModified: Mon, 08 Jul 2019 10:38:55 GMT repositoryId: 195179871 description: >- A Open API 3.x specification (swagger next gen) for SenML (Sensor Measurement List) for sensor (IoT) communication and actuation (https://tools.ietf.org/html/rfc8428) created: '2019-07-04T06:14:19Z' updated: '2022-09-12T00:22:21Z' language: null archived: false stars: 1 watchers: 0 forks: 0 owner: mariotoffia logo: https://avatars.githubusercontent.com/u/1716527?v=4 license: Apache-2.0 repoEtag: '"959f46234eb1c37a7224116750c2032f4fb375a061038983b57421de47d00d22"' repoLastModified: Mon, 12 Sep 2022 00:22:21 GMT foundInMaster: true category: Parsers id: 0f79e8030eec5efa7008eb163f8e07a9 - source: openapi3 tags repository: https://github.com/reschoene/taskplanneraws v3: true id: 833dc8c7d444e61f954edb8555cfe7f1 repositoryMetadata: base64Readme: >- # TaskPlanner for AWS
A simple task planner API for listing tasks grouped by lists and managing them

## Stack
- Kotlin, Java with GraalVM (for native compilation)
- Quarkus, the Supersonic Subatomic Java Framework.
- TestContainers, RESTEasy, RESTAssured
- AWS Lambda, AWS API Gateway, AWS Dynamodb
- AWS SAM CLI (for local testing and AWS Deploy): SAM = Serverless Application Model

## Project status - Sonar analysis
[![Quality gate](https://sonarcloud.io/api/project_badges/quality_gate?project=reschoene_taskplanneraws)](https://sonarcloud.io/summary/new_code?id=reschoene_taskplanneraws)

[![Security Rating](https://sonarcloud.io/api/project_badges/measure?project=reschoene_taskplanneraws&metric=security_rating)](https://sonarcloud.io/summary/new_code?id=reschoene_taskplanneraws)

[![Code Smells](https://sonarcloud.io/api/project_badges/measure?project=reschoene_taskplanneraws&metric=code_smells)](https://sonarcloud.io/summary/new_code?id=reschoene_taskplanneraws)

[![Bugs](https://sonarcloud.io/api/project_badges/measure?project=reschoene_taskplanneraws&metric=bugs)](https://sonarcloud.io/summary/new_code?id=reschoene_taskplanneraws)

[![Vulnerabilities](https://sonarcloud.io/api/project_badges/measure?project=reschoene_taskplanneraws&metric=vulnerabilities)](https://sonarcloud.io/summary/new_code?id=reschoene_taskplanneraws)

[![Coverage](https://sonarcloud.io/api/project_badges/measure?project=reschoene_taskplanneraws&metric=coverage)](https://sonarcloud.io/summary/new_code?id=reschoene_taskplanneraws)

## Build and Deploy 
There are two deploy options available: JVM and native executables. Bellow are the instructions for each one

### Option 1 - Building and deploying native executable

#### Creating a native executable
You can create a native executable using:
```shell script
./gradlew build -Dquarkus.package.type=native
```

Or, if you don't have GraalVM installed, you can run the native executable build in a container using:
```shell script
./gradlew build -Dquarkus.package.type=native -Dquarkus.native.container-build=true
```

#### Deploying a native executable
 - You can then test the executable locally with sam local
`sam local start-api --template build/sam.native.yaml`

 - To deploy to AWS Lambda:
`sam deploy -t build/sam.native.yaml -g`

### Option 2 - Building and deploying JVM executable

#### Creating a JVM executable
You can create a native executable using:
```shell script
./gradlew build
```

#### Deploying a JVM executable
- You can then test the executable locally with sam local
  `sam local start-api --template build/sam.jvm.yaml`

- To deploy to AWS Lambda:
  `sam deploy -t build/sam.jvm.yaml -g`


### Deploy error tips
If you get an error similar to this one: "An error occurred (SignatureDoesNotMatch) when calling the CreateChangeSet operation: Signature expired: 20220619T193119Z is now earlier than 20220619T193604Z (20220619T194104Z - 5 min.)"
run following command to sync your PC date and time: 
```shell script
sudo date -s "$(wget -qSO- --max-redirect=0 google.com 2>&1 | grep Date: | cut -d' ' -f5-8)Z"
```


## Running the application in dev mode
As this is a quarkus application, you can run it using its developer mode that enables live coding using:
```shell script
./gradlew quarkusDev
```

or in debug mode
```shell script
./gradlew quarkusDev --debug
```

## Running dynamoDb locally in dev mode
```shell script
docker run --publish 8000:8000 amazon/dynamodb-local:1.11.477 -jar DynamoDBLocal.jar -inMemory -sharedDb
```
This starts a DynamoDB instance that is accessible on port 8000. You can check it’s running by accessing the web shell on http://localhost:8000/shell.
Open "http://localhost:8000/shell" in your browser.

Copy and paste the following code to the shell and run it:

```shell script 
var params = {
TableName: 'TaskLists',
KeySchema: [{ AttributeName: 'id', KeyType: 'HASH' }],
AttributeDefinitions: [{  AttributeName: 'id', AttributeType: 'S', }],
ProvisionedThroughput: { ReadCapacityUnits: 1, WriteCapacityUnits: 1, }
};

dynamodb.createTable(params, function(err, data) {
if (err) ppJson(err);
else ppJson(data);

});

params = {
TableName: 'Tasks',
KeySchema: [{ AttributeName: 'id', KeyType: 'HASH' }],
AttributeDefinitions: [{  AttributeName: 'id', AttributeType: 'S', }],
ProvisionedThroughput: { ReadCapacityUnits: 1, WriteCapacityUnits: 1, }
};

dynamodb.createTable(params, function(err, data) {
if (err) ppJson(err);
else ppJson(data);

});

params = {
TableName: 'Quotations',
KeySchema: [{ AttributeName: 'id', KeyType: 'HASH' }],
AttributeDefinitions: [{  AttributeName: 'id', AttributeType: 'S', }],
ProvisionedThroughput: { ReadCapacityUnits: 1, WriteCapacityUnits: 1, }
};

dynamodb.createTable(params, function(err, data) {
if (err) ppJson(err);
else ppJson(data);

});
```

## accessing swagger UI
http://localhost:8080/q/swagger-ui

## accessing OpenAPI definition
http://localhost:8080/q/openapi

## creating dynamodb table on your aws:
```shell script 
aws dynamodb create-table --table-name TaskLists \
--attribute-definitions AttributeName=id,AttributeType=S \
--key-schema AttributeName=id,KeyType=HASH \
--provisioned-throughput ReadCapacityUnits=1,WriteCapacityUnits=1
```

```shell script 
aws dynamodb create-table --table-name Tasks \
--attribute-definitions AttributeName=id,AttributeType=S \
--key-schema AttributeName=id,KeyType=HASH \
--provisioned-throughput ReadCapacityUnits=1,WriteCapacityUnits=1
```

```shell script 
aws dynamodb create-table --table-name Quotations \
--attribute-definitions AttributeName=id,AttributeType=S \
--key-schema AttributeName=id,KeyType=HASH \
--provisioned-throughput ReadCapacityUnits=1,WriteCapacityUnits=1
```

### SonarCloud Integration:
Change sonarqube properties on build.gradle file to your SonarCloud configs

#### To generate JaCoCo coverage reports, run:
```shell script
./gradlew test
```

#### To upload coverage data to SonarCloud:
```shell script 
export SONAR_TOKEN=YOUR_SONAR_KEY
```

```shell script 
./gradlew sonarqube
```
 readmeEtag: '"77e365d6105adc00337f536b2220f6d77f10bdf2"' readmeLastModified: Sun, 31 Jul 2022 22:08:12 GMT repositoryId: 505209983 description: >- A simple task planner API for listing tasks grouped by lists and managing them. created: '2022-06-19T20:15:01Z' updated: '2022-07-04T20:02:39Z' language: Kotlin archived: false stars: 1 watchers: 1 forks: 0 owner: reschoene logo: https://avatars.githubusercontent.com/u/32646475?v=4 license: MIT repoEtag: '"9868d3e41096918abf71baebbb6d8022c4ac1cfe37542ac42659a4f8e27c4e09"' repoLastModified: Mon, 04 Jul 2022 20:02:39 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/shocklateboy92/erabikata v3: true repositoryMetadata: base64Readme: >- IyBFcmFiaWthdGEgKOmBuOOBs+aWuSkKClRoaXMgaXMgYSBwcm9ncmFtIEkgd3JvdGUgdG8gaGVscCB0ZWFjaCBteXNlbGYgSmFwYW5lc2UgdGhyb3VnaCBhbmltZS5cCldpdGggaXQsIEkgd2VudCBmcm9tICJub3Qga25vd2luZyBhIHNpbmdsZSB3b3JkIG9mIEphcGFuZXNlIiB0byAiYWJsZSB0byBnbyBiYWNrcGFja2luZyB0aHJvdWdoIEphcGFuIiBpbiBfYWJvdXQgOSBtb250aHNfIQoKIVtTY3JlZW5zaG90IDEsIGRlc2t0b3AgbGF5b3V0XSguL3NjcmVlbnNob3QxLnBuZykKCkl0IHN0YXJ0ZWQgYXMgYSBxdWljayBiYXNoIHNjcmlwdCB0byBncmVwIHRoZSBzdWJ0aXRsZSBmaWxlcyBvZiB2YXJpb3VzIGFuaW1lIGZvciBleGFtcGxlcyBvZiB3b3JkcyBJIGhhZCB0byBsZWFybi4gQnV0IG92ZXIgdGltZQo= readmeEtag: '"a290a7d6878d045c974611269a61c8476c0ca393"' readmeLastModified: Tue, 16 Aug 2022 05:37:07 GMT repositoryId: 316345095 description: >- This is a program I wrote to help teach myself Japanese through anime. With it, I went from "not knowing a single word of Japanese" to "able to go backpacking through Japan" in about 9 months! created: '2020-11-26T22:01:38Z' updated: '2023-11-28T20:56:53Z' language: TypeScript archived: true stars: 1 watchers: 2 forks: 0 owner: shocklateboy92 logo: https://avatars.githubusercontent.com/u/856493?v=4 repoEtag: '"a348892dbda0cd3fbbdbede5b8838b5d6acc45f906c309c133ca66ab61000c9b"' repoLastModified: Tue, 28 Nov 2023 20:56:53 GMT foundInMaster: true id: 5093523d516b0eda1560ca8fb9daeef0 category: Server Implementations - source: openapi3 tags repository: https://github.com/1bitrs/cibo v3: true repositoryMetadata: base64Readme: >- IyMgQ2libwohW1B5dGhvbiBWZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3B5dGhvbi12My43LjUtYnJpZ2h0Z3JlZW4pCiFbU3lzdGVtIFBsYXRmb3JtXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3BsYXRmb3JtLXVidW50dS1icmlnaHRncmVlbi5zdmcpCiFbVHJhdmlzIENJXShodHRwczovL2FwcC50cmF2aXMtY2kuY29tL3lhbmdmYW45NzAyL2NpYm8uc3ZnP2JyYW5jaD1tYXN0ZXImc3RhdHVzPXVua25vd24pCiFbQ292ZXJhZ2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY29kZWNvdi9jL2dpdGh1Yi95YW5nZmFuOTcwMi9jaWJvKQpbIVtDb2RlIHN0eWxlOiBibGFja10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlJTIwc3R5bGUtYmxhY2stMDAwMDAwLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9wc2YvYmxhY2spCiFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS95YW5nZmFuOTcwMi9jaWJvKQohW1JlcG8gU2l6ZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvcmVwby1zaXplL3lhbmdmYW45NzAyL2NpYm8pCgoKIyMgSW5zdGFsbGluZwpgYGBzaGVsbApweXRob24gc2V0dXAucHkgaW5zdGFsbApgYGAKCiMjIEEgU2ltcGxlIEV4YW1wbGUKYGBgcHl0aG9uCmZyb20gY2libyBpbXBvcnQgSGFuZGxlciwgU2ltcGxlQ29udGV4dCwgQmx1ZXByaW50LCBCYXNlQXBpUXVlcnksIEJhc2VBcGlCb2R5CgphcGkgPSBCbHVlcHJpbnQoImFwaSIpCgpAYXBpLnBvc3QoIi9lY2hvIikKY2xhc3MgRWNob0hhbmRsZXIoSGFuZGxlcik6CgogICAgZGVjb3JhdG9ycyA9IFt0b2tlbl9hdXRoXQoKICAgIGNsYXNzIFF1ZXJ5KEJhc2VBcGlRdWVyeSk6CiAgICAgICAgYTogc3RyCiAgICAgICAgYjogT3B0aW9uYWxbTGlzdFtpbnRdXQogICAgICAgIGM6IE9wdGlvbmFsW0RpY3Rbc3RyLCBpbnRdXQoKICAgIGNsYXNzIEJvZHkoQmFzZUFwaUJvZHkpOgogICAgICAgIGQ6IFNldFtpbnRdCiAgICAgICAgZTogVHVwbGVbRGljdFtpbnQsIExpc3RdLCBEaWN0W2ludCwgTGlzdF1dCgogICAgZGVmIGhhbmRsZShzZWxmLCBjb250ZXh0OiBTaW1wbGVDb250ZXh0LCBxdWVyeTogUXVlcnksIGJvZHk6IEJvZHkpOgogICAgICAgICIiImVjaG8gdGhlIHJlY2V2aWVkIHBhcmFtcyIiIgogICAgICAgIHJldHVybiBjb250ZXh0LnN1Y2Nlc3MoCiAgICAgICAgICAgIGRhdGE9ZiJhOiB7cXVlcnkuYX0sIGI6IHtxdWVyeS5ifSwgYzoge3F1ZXJ5LmN9LCBkOiB7Ym9keS5kfSwgZToge2JvZHkuZX0iCiAgICAgICAgKQoKYGBgClVzZSBhIGN1c3RvbSBtb2RlbCBhbmQgdmFsaWRhdGUgZnVuY3Rpb24KYGBgcHl0aG9uCkBhcGkucG9zdCgiL3VzZXIiKQpjbGFzcyBVc2VySGFuZGxlcihIYW5kbGVyKToKICAgIGNsYXNzIEJvZHkoQmFzZUFwaUJvZHkpOgogICAgICAgIGNsYXNzIFVzZXIoQmFzZU1vZGVsKToKICAgICAgICAgICAgbmFtZTogc3RyID0gRmllbGQoZGVzY3JpcHRpb249IuWnk+WQjSIpCiAgICAgICAgICAgIGVtYWlsczogT3B0aW9uYWxbTGlzdFtzdHJdXSA9IEZpZWxkKGRlc2NyaXB0aW9uPSLpgq7nrrEiKQoKICAgICAgICAgICAgQGNsYXNzbWV0aG9kCiAgICAgICAgICAgIGRlZiB2YWxpZGF0ZShjbHMsIHZhbHVlOiBBbnkpOgogICAgICAgICAgICAgICAgb2JqID0gY2xzKCoqdmFsdWUpCiAgICAgICAgICAgICAgICBpZiBvYmouZW1haWxzOgogICAgICAgICAgICAgICAgICAgIGlmIG5vdCBhbGwoCiAgICAgICAgICAgICAgICAgICAgICAgIFsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlLm1hdGNoKAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHIiXlswLTlhLXpBLVpfXXswLDE5fUBbMC05YS16QS1aXXsxLDEzfVwuW2NvbSxjbixuZXRdezEsM30kIiwgZW1haWwKICAgICAgICAgICAgICAgICAgICAgICAgICAgICkKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGZvciBlbWFpbCBpbiBvYmouZW1haWxzCiAgICAgICAgICAgICAgICAgICAgICAgIF0KICAgICAgICAgICAgICAgICAgICApOgogICAgICAgICAgICAgICAgICAgICAgICByYWlzZSBWYWx1ZUVycm9yKCJlbWFpbCBpcyBub3QgdmFsaWQiKQogICAgICAgICAgICAgICAgcmV0dXJuIG9iagoKICAgICAgICB1c2VyOiBVc2VyCiAgICAgICAgaW52aXRlcjogc3RyCgogICAgZGVmIGhhbmRsZShzZWxmLCBjb250ZXh0OiBTaW1wbGVDb250ZXh0LCBib2R5OiBCb2R5KToKICAgICAgICAiIiJjdXN0b20gbW9kZWwgYW5kIHZhbGlkYXRlIiIiCiAgICAgICAgcmV0dXJuIGNvbnRleHQuc3VjY2Vzcyh1c2VyPWJvZHkudXNlciwgaW52aXRlcj1ib2R5Lmludml0ZXIpCmBgYAoKIyMgRGV2CnB1bGwgYHN0dWJzYCBmaWxlcwpgYGBzaGVsbApnaXQgc3VibW9kdWxlIHVwZGF0ZSAtLWluaXQgLS1yZWN1cnNpdmUKYGBgCgojIyBEb2NzCltodHRwOi8vMTI3LjAuMC4xOjUwMDAvZG9jc10oaHR0cDovLzEyNy4wLjAuMTo1MDAwL2RvY3MpCgojIyBDb250cmlidXRpbmcgR3VpZGUKIyMjIEZpcnN0IHRpbWUgc2V0dXAKQ3JlYXRlIGEgdmlydHVhbCBlbnZpcm9ubWVudCBhbmQgaW5zdGFsbCByZXF1aXJlbWVudHM6CmBgYGJhc2gKJCBweXRob24zIC1tIHZlbnYgZW52CiQgc291cmNlIGVudi9iaW4vYWN0aXZhdGUKJCBweXRob24gLW0gcGlwIGluc3RhbGwgLS11cGdyYWRlIHBpcCBzZXR1cHRvb2xzCiQgcGlwIGluc3RhbGwgLXIgcmVxdWlyZW1lbnRzL2Rldi50eHQKJCBwaXAgaW5zdGFsbCAtZSAuCiQgcHJlLWNvbW1pdCBpbnN0YWxsCmBgYAo= readmeEtag: '"df307d04a11da549db3a7ae3a4c0dadcb9e16f8f"' readmeLastModified: Tue, 17 May 2022 03:19:28 GMT repositoryId: 359301009 description: A web framework that is composed of flask, pydantic, and openapi 3. created: '2021-04-19T02:07:52Z' updated: '2023-09-12T09:44:57Z' language: Python archived: true stars: 1 watchers: 1 forks: 0 owner: 1bitrs logo: https://avatars.githubusercontent.com/u/41955644?v=4 license: MIT repoEtag: '"48386a3794a9c0099180ce0fceb3b3f94552160fd7cc5a942e9400052704a60a"' repoLastModified: Tue, 12 Sep 2023 09:44:57 GMT foundInMaster: true category: Data Validators id: c790c04a41fae48236e11b8fbfa78f46 oldLocations: - https://github.com/whitewindss/cibo - source: - openapi3 tags - openapi31 tags repository: https://github.com/sv-tools/roas v3_1: true id: 6cefed15ccc85977666e11bdf863d37b repositoryMetadata: base64Readme: >- IyByb2FzCgpSdXN0IE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAodjIuMCwgdjMuMC5YIGFuZCB2My4xLlgpCgpbIVtjcmF0ZXMuaW9dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY3JhdGVzL3Yvcm9hcy5zdmcpXShodHRwczovL2NyYXRlcy5pby9jcmF0ZXMvcm9hcykKWyFbZG9jcy5yc10oaHR0cHM6Ly9kb2NzLnJzL3JvYXMvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9kb2NzLnJzL3JvYXMpCgpQYXJzaW5nIGFuZCBnZW5lcmF0aW5nIE9wZW5BUEkgU3BlY2lmaWNhdGlvbjoKCiogW3hdIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiB2Mi4wICgqKnYyKio6IG9sZCBzcGVjaWZpY2F0aW9uLCBkaXNhYmxlZCBieSBkZWZhdWx0KQoqIFt4XSBPcGVuQVBJIFNwZWNpZmljYXRpb24gdjMuMC54ICgqKnYzXzAqKjogZGVmYXVsdCBmZWF0dXJlKQoqIFt4XSBPcGVuQVBJIFNwZWNpZmljYXRpb24gdjMuMS54ICgqKnYzXzEqKjo7IGV4cGVyaW1lbnRhbCBhbmQgZGlzYWJsZWQgYnkgZGVmYXVsdCkKCj4gWyFDQVVUSU9OXQo+IFRoZSBwcm9qZWN0IGlzIGluIGVhcmx5IGRldmVsb3BtZW50IHN0YWdlLCBzbyB0aGUgQVBJIG1heSBjaGFuZ2UgaW4gdGhlIGZ1dHVyZS4KPiBDb25zaWRlciBhbnkgMC54LnggdmVyc2lvbiBhcyB1bnN0YWJsZSBhbmQgc3ViamVjdCB0byBicmVha2luZyBjaGFuZ2VzLgoKIyMgVXNhZ2UKClRvIHVzZSBgcm9hc2AsIGFkZCBpdCB0byB5b3VyIGBDYXJnby50b21sYDoKCmBgYHNoZWxsCmNhcmdvIGFkZCByb2FzCmBgYAoKb3IgbWFudWFsbHkgYWRkIHRoZSBmb2xsb3dpbmcgbGluZXM6CgpgYGB0b21sCltkZXBlbmRlbmNpZXNdCnJvYXMgPSAiMC40IiAgCmBgYAoKIyMgRXhhbXBsZXMKCmBgYHJ1c3QKdXNlIHJvYXM6OnYzXzE6OnNwZWM6OlNwZWM7CnVzZSByb2FzOjp2YWxpZGF0aW9uOjp7T3B0aW9ucywgVmFsaWRhdGV9OwoKLi4uCgpsZXQgc3BlYyA9IHNlcmRlX2pzb246OmZyb21fc3RyOjo8U3BlYz4ocmF3X2pzb24pLnVud3JhcCgpOwpzcGVjLnZhbGlkYXRlKE9wdGlvbnM6Oklnbm9yZU1pc3NpbmdUYWdzIHwgT3B0aW9uczo6SWdub3JlRXh0ZXJuYWxSZWZlcmVuY2VzKS51bndyYXAoKTsKCi4uLgoKYGBgCg== readmeEtag: '"988182e1a1d2f034d6af467183229ea4d11d85f7"' readmeLastModified: Sat, 05 Jul 2025 03:51:07 GMT repositoryId: 699556146 description: Rust OpenAPI Specification (v2, v3.0, v3.1) created: '2023-10-02T21:40:47Z' updated: '2025-10-27T20:33:22Z' language: Rust archived: false stars: 3 watchers: 1 forks: 1 owner: sv-tools logo: https://avatars.githubusercontent.com/u/59230487?v=4 license: Apache-2.0 repoEtag: '"746089129002482b932bcdda9150da9e56e08694a881175c51a7358cc949e0a2"' repoLastModified: Mon, 27 Oct 2025 20:33:22 GMT category: Parsers foundInMaster: true v3: true - source: openapi3 tags repository: https://github.com/romabilka/lets-go-chat v3: true id: cc00b439672c4bf3161d2400cf7951cc repositoryMetadata: base64Readme: >- IyBsZXRzLWdvLWNoYXQKCiMjIEhvdyB0byBydW4KMS4gQ2xvbmUgdGhlIHByb2plY3QKMi4gYGBgZ28gcnVuIGNtZC9sZXRzLWdvLWNoYXQvbWFpbi5nb2BgYAoKIyMgVG8gdHJ5IGdvIHRvIHRoZSBsaW5rCmh0dHA6Ly9sb2NhbGhvc3QvYXBpLwoKIyMgTGljZW5zZQpbTUlUIGxpY2Vuc2VdKExJQ0VOU0UubWQpLg== readmeEtag: '"7fc18cd6adc78355bc706fa16ad2780010f777be"' readmeLastModified: Tue, 15 Mar 2022 15:10:40 GMT repositoryId: 421817498 description: null created: '2021-10-27T12:53:20Z' updated: '2024-06-19T21:42:30Z' language: Go archived: false stars: 1 watchers: 1 forks: 0 owner: RomaBilka logo: https://avatars.githubusercontent.com/u/28790784?v=4 license: NOASSERTION repoEtag: '"bf7157e23eb3a67d735cbebcbfdc8c98e8c369fe1fdfb290ea2b7e6476d6b073"' repoLastModified: Wed, 19 Jun 2024 21:42:30 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/howmarketing/employee-maintenance-fullstack-sample-app v3: true id: 60f335779c80fc47c5bb01c79e52bc10 repositoryMetadata: base64Readme: >- ## Simple Employee Maintenance Web Application

### Screenshots

![Screenshot](./public/home.png)

![Screenshot](./public/employee-page.png)

### Overview

The Simple Employee Maintenance web application is designed to streamline the management of employee records and department assignments within an organization. This full-stack application provides essential CRUD operations for employees and departments, alongside a user-friendly interface to view and manage employee information.

### Features

1. **Database Management:**
   - A robust database schema designed using Prisma ORM to store and manage employee and department data.
   - Tables include Employees, Departments, and Employee Department History to track changes over time.

2. **Web Service Endpoints:**
   - **Employee Endpoints:**
     - `GetAllEmployees`: Retrieve a list of all employees.
     - `GetEmployeeById`: Fetch detailed information for a specific employee.
     - `UpdateEmployee`: Modify the details of an existing employee.
     - `DeleteEmployee`: Remove an employee from the system.
   - **Department Endpoints:**
     - `GetAllDepartments`: Retrieve a list of all departments.

3. **Employee List Page:**
   - A web page that displays a comprehensive list of employees, including their names, departments, hire dates, and avatars.
   - Hire dates are displayed in a user-friendly format, indicating the duration of employment.
   - Includes a "View Details" button for each employee to access more detailed information.

### Technical Details

1. **Prisma Schema Definition:**
   The Prisma schema has been updated to include three main models: Department, Employee, and EmployeeDepartmentHistory. Here's a brief overview:

   ```prisma:schema.prisma
	// This is your Prisma schema file,
	// learn more about it in the docs: https://pris.ly/d/prisma-schema

	// Define the Prisma client generator
	generator client {
		provider = "prisma-client-js"
	}

	// Define the database connection
	datasource db {
		provider  = "postgresql"
		url       = env("POSTGRES_PRISMA_URL") // uses connection pooling
		directUrl = env("POSTGRES_URL_NON_POOLING") // uses a direct connection
	}

	// Command to run migrations: npx prisma migrate dev

	// Department model
	model Department {
		id                        Int                         @id @default(autoincrement())
		key                       String                      @unique
		label                     String
		Employee                  Employee[]
		EmployeeDepartmentHistory EmployeeDepartmentHistory[]
		createdAt                 DateTime                    @default(now()) @map("created_at")
		updatedAt                 DateTime                    @default(now()) @updatedAt @map("updated_at")

		@@map("departments")
	}

	// Employee model
	model Employee {
		id                        Int                         @id @default(autoincrement())
		publicId                  String                      @unique @default(uuid()) @map("public_id")
		firstName                 String                      @map("first_name")
		lastName                  String                      @map("last_name")
		hireDate                  DateTime?                   @default(now()) @map("hire_date")
		isActive                  Boolean                     @map("is_active")
		department                Department?                 @relation(fields: [departmentKey], references: [key])
		departmentKey             String?                     @map("department_key")
		phone                     String
		address                   String
		EmployeeDepartmentHistory EmployeeDepartmentHistory[]
		createdAt                 DateTime                    @default(now()) @map("created_at")
		updatedAt                 DateTime                    @default(now()) @updatedAt @map("updated_at")

		@@map("employees")
	}

	// EmployeeDepartmentHistory model
	model EmployeeDepartmentHistory {
		id              Int        @id @default(autoincrement())
		employee        Employee   @relation(fields: [employeeId], references: [id])
		employeeId      Int        @map("employee_id")
		department      Department @relation(fields: [departmentKey], references: [key])
		departmentKey   String     @map("department_key")
		departmentLabel String     @default("") @map("department_label")
		createdAt       DateTime   @default(now()) @map("created_at")
		updatedAt       DateTime   @default(now()) @updatedAt @map("updated_at")

		@@map("employee_department_history")
	}
   ```

   - **Department Model:** Contains fields for department ID, unique key, label, and timestamps. It relates to multiple employees and department history records.
   - **Employee Model:** Includes fields for employee ID, unique public ID, first and last names, hire date, department association, phone, address, and timestamps.
   - **Employee Department History Model:** Tracks the history of employees' department assignments, with fields for employee ID, department key, department label, and timestamps.

2. **Repository Pattern Implementation:**
   We've implemented the repository pattern to abstract database operations. Here's an example of the EmployeeRepository:

   ```typescript:src/repositories/employeeRepository.ts
	import { PrismaClient } from "@prisma/client";
	import { CreateEmployeeDTO, EmployeeDTO } from '@/dtos/employeeDTO';

	export class EmployeeRepository {
		constructor(private prisma: PrismaClient) { }

		async createEmployee(data: CreateEmployeeDTO): Promise<EmployeeDTO> {
			const createdEmployee = await this.prisma.employee.create({
				data,
				include: {
					department: true,
					EmployeeDepartmentHistory: true
				}
			});
			if (data.departmentKey) {
				await this.prisma.employeeDepartmentHistory.create({
					data: {
						employeeId: createdEmployee.id,
						departmentKey: data.departmentKey,
						departmentLabel: createdEmployee.department?.label || "",
					}
				});
			}
			return createdEmployee as EmployeeDTO;
		}
	}

   ```

   This repository centralizes database operations related to employees, improving code organization and maintainability.

3. **API Routes Refactoring:**
   API routes have been refactored to use the new repository pattern and DTOs. For example, the employee creation route now looks like this:

   ```typescript:src/app/api/employee/route.ts
	import { NextRequest } from "next/server";
	import { PrismaClient } from "@prisma/client";
	import { z } from 'zod';
	import { EmployeeRepository } from '@/repositories/employeeRepository';
	import { CreateEmployeeDTO } from '@/dtos/employeeDTO';
	import { EmployeeResponseDTO } from '@/dtos/responseDTO';
	import { createEmployeeSchema } from '@/schemas/employeeSchema';
	import { logger } from '@/utils/logger';

	export const dynamic = "force-dynamic";

	// Singleton pattern for PrismaClient to avoid multiple instances
	const prisma = new PrismaClient();
	const employeeRepository = new EmployeeRepository(prisma);

	export type ICreateResponseProperties = EmployeeResponseDTO & { status: number, statusText: string };

	const createResponse = ({
		success = false,
		message = "",
		data = {} as EmployeeResponseDTO["data"],
		status = 200,
		statusText = "OK"
	}: ICreateResponseProperties): Response => {
		const responseObject: EmployeeResponseDTO = { success, message, data };
		return new Response(JSON.stringify(responseObject), {
			status,
			headers: {
				"Content-Type": "application/json",
				"Access-Control-Allow-Origin": "*", // Consider restricting this for security
			},
			statusText,
		});
	};

	export async function POST(request: NextRequest) {
		try {
			const body: CreateEmployeeDTO = await request.json();
			const validatedData = createEmployeeSchema.parse(body);
			const createdEmployee = await employeeRepository.createEmployee(validatedData);
			logger.info(`Employee created: ${createdEmployee.id}`);
			return createResponse({ success: true, message: "Success", data: createdEmployee, status: 201, statusText: "CREATED" });
		} catch (e: any) {
			const errorResponse = {
				success: false,
				message: e?.message || "Something went wrong",
				status: 500,
				statusText: "Internal Server Error",
			} as ICreateResponseProperties;
			if (e instanceof z.ZodError) {
				errorResponse.message = e.errors.map(err => `${err.path}: ${err.message}`).join(', ');
				errorResponse.status = 400;
				errorResponse.statusText = "Bad Request";
				logger.error(`Error creating employee: ${errorResponse.message}`);
				return createResponse(errorResponse);
			}
			logger.error(`Error creating employee: ${errorResponse.message}`);
			return createResponse(errorResponse);
		}
	}

   ```

   This refactoring enhances error handling and implements logging in API routes.

### Usage

1. **Prisma Schema Definition:**
   - **Department Model:** Contains fields for department ID, unique key, label, and timestamps. It relates to multiple employees and department history records.
   - **Employee Model:** Includes fields for employee ID, unique public ID, first and last names, hire date, department association, phone, address, and timestamps.
   - **Employee Department History Model:** Tracks the history of employees' department assignments, with fields for employee ID, department key, department label, and timestamps.

2. **Technologies Used:**
   - **Backend:** Node.js, with server-side for backend RESTful APIs.
   - **Frontend:** Next.js@14 with React.js@19 implemented the server actions.
   - **Database:** Prisma ORM with SQLite for development database environment.
   - **Other Tools:** Git for version control, GitHub for repository hosting, vercel for deployments.

3. **Instructions::**
   - **Database Setup:** Create the necessary tables and relationships using Prisma migrations.
   - **API Development:** Implement the required endpoints to interact with the employee and department data.
   - **UI Implementation:** Develop the Employee List page to display and manage employee records effectively.

4. **Deliverables:**
   - **API Development:** The required endpoints have been implemented using the repository pattern and DTOs. Examples include:
     - CreateEmployee
     - UpdateEmployee
     - UpdateEmployeeDepartment
     - GetAllEmployees
     - GetEmployeeById
     - DeleteEmployee
     - GetAllDepartments
   - **UI Implementation:** 
     The Employee List page has been developed to display and manage employee records effectively, including features like viewing employee details and updating department assignments.

> The Simple Employee Maintenance web application now incorporates the repository pattern, DTOs, and improved error handling, resulting in a more maintainable and robust solution for managing employee records and departmental assignments.
>

### Conclusion

The Simple Employee Maintenance web application is a comprehensive solution for managing employee records and departmental assignments, providing an intuitive interface and robust backend to ensure efficient data handling and retrieval.

---

## STARTING POINT

```bash
npm i --force
npm i --legacy-peer-deps

npm i -D jest @types/jest @testing-library/jest-dom ts-jest @testing-library/user-event jest-environment-jsdom @testing-library/react --force

npx prisma generate
npm run build
npm run test
npm run start
```

## ASSESSMENT README

A simple web application for managing employee records, built as a fullstack assessment to demonstrate proficiency in software architecture, code quality, design patterns, and best practices.

### Fullstack Assessment – Candidate Instructions

**Expectations**
- Demonstrate a solid and robust architecture.
- Highly maintainable and legible code.
- Use and knowledge of design patterns.
- Follow software development best practices.
- Effective and consistent naming conventions.
- Code crafted with modern development methodologies.
- If you choose to simplify your code in the name of speed (i.e. "prototype" style development) be able to explain the shortcuts you took and why.
- Smart UI patterns are not allowed.
- Code snippets from old projects and/or opening any old projects are strictly forbidden.
- Prioritize good code design and architecture over development speed.
- Read the instructions carefully as only complete and working solutions will be reviewed.
- Your code should compile and run with us doing a simple package restore and nothing more.

**Submitting your code**
- Go to [company-repo-management](https://git.company-repo-management.com) and register or login if you already have an account.
- Create a repository to work on and do constant commits as you would normally do.
- You have no time limit to complete the assessment.
- Once you are ready, add a brief explanation of your solution and architecture to the README file and share your git repository URL with your recruiter.

### Simple Employee Maintenance web app

**First Task (Database):**
Create a database to store the following information:
1. First Name
2. Last Name
3. Hire Date
4. Department
5. Phone
6. Address

Create all needed tables and relationships.

**Second Task (Programming Exercise):**
Create a Web Service that exposes the following endpoints:
1. **For Employee Entity:**
   - Should include the employee department in text form.
   - GetAllEmployees
   - GetEmployeeById
   - UpdateEmployee
   - DeleteEmployee
   
2. **For Department Entity:**
   - GetAllDepartments

**Third Task (Employee List page):**
Create a new page to show a list of employees. You can use the following mockup as reference or create your own design if you prefer.

This page should show the following information:
- Employee Name, Department, Hire Date, Employee Avatar
- It should display the date in the following format: May 2, 2021 (2y – 1m – 4d)
  - The date in parenthesis determines how long that employee has been in the company.
- A "View Details" button that will take the user to the following page.


## Modeling the Database with Prisma

To model the database using Prisma for the requirements you've outlined, we need to define the schema that includes tables for `Employee` and `Department`, and establish the relationships between them. Here’s how you can model it:

### Prisma Schema Definition

Create a new Prisma schema file (typically named `schema.prisma`) and define the schema as follows:

```prisma
// This is your Prisma schema file,
// learn more about it in the docs: https://pris.ly/d/prisma-schema

generator client {
}

datasource db {
}

// npx prisma migrate dev

model Department {
}

model Employee {
}

model EmployeeDepartmentHistory {
}

```
The provided schema defines three models: `Department`, `Employee`, and `EmployeeDepartmentHistory`. Here's an explanation of each model:

1. **Department**:
   - This model represents a department within an organization.
   - It has fields such as `id` (auto-incrementing primary key), `key` (unique string identifier), `label` (department name), and timestamps for `createdAt` and `updatedAt`.
   - It has a one-to-many relationship with the `Employee` model, meaning that a department can have multiple employees.
   - It also has a one-to-many relationship with the `EmployeeDepartmentHistory` model, which tracks the history of employees' department assignments.

2. **Employee**:
   - This model represents an employee within the organization.
   - It has fields such as `id` (auto-incrementing primary key), `publicId` (unique string identifier), `firstName`, `lastName`, `hireDate`, `isActive` (boolean indicating if the employee is currently active), `departmentKey` (foreign key referencing the `key` field in the `Department` model), `phone`, `address`, and timestamps for `createdAt` and `updatedAt`.
   - It has a many-to-one relationship with the `Department` model, meaning that an employee can belong to one department.
   - It has a one-to-many relationship with the `EmployeeDepartmentHistory` model, which tracks the history of the employee's department assignments.

3. **EmployeeDepartmentHistory**:
   - This model keeps track of the history of employees' department assignments.
   - It has fields such as `id` (auto-incrementing primary key), `employeeId` (foreign key referencing the `id` field in the `Employee` model), `departmentKey` (foreign key referencing the `key` field in the `Department` model), `departmentLabel` (the label or name of the department at the time of assignment), and timestamps for `createdAt` and `updatedAt`.
   - It has a many-to-one relationship with both the `Employee` and `Department` models, allowing it to associate an employee with a department at a specific point in time.

This schema allows you to store and manage information about departments, employees, and the history of employees' department assignments within your application. The relationships between the models enable you to query and retrieve data efficiently, such as retrieving all employees belonging to a specific department or retrieving the department history for a particular employee.


### Database Setup

After defining the schema, you would typically run the following commands to generate migrations and apply them to your database:

```bash
npx prisma migrate dev
npx prisma migrate dev --name init

npx prisma migrate up --experimental
```

> These commands will create and apply migrations to your database based on the schema you've defined.
>
 readmeEtag: '"2d505bc56c1f57828ca645e415abf39a8e83632c"' readmeLastModified: Sun, 18 Aug 2024 05:19:12 GMT repositoryId: 838831998 description: >- The Simple Employee Maintenance web application is designed to streamline the management of employee records and department assignments within an organization. This full-stack application provides essential CRUD operations for employees and departments, alongside a user-friendly interface to view and manage employee information. created: '2024-08-06T12:35:26Z' updated: '2024-08-18T05:19:15Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: howmarketing logo: https://avatars.githubusercontent.com/u/72627426?v=4 license: MIT repoEtag: '"b5cf32b77e657a7bbecc8d375f2e2aa68843f6e3a5bdfc3fc6fc50ccbbd75b4f"' repoLastModified: Sun, 18 Aug 2024 05:19:15 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/brisinger/openapi v3: true id: 7f58a1495b387662d0e44b3d24f4ce00 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWNzIHdpdGggU3dhZ2dlciBUb29scwoKIyMgUHJlLXJlcXVpc2l0ZXMKCi0gQmFzaWMga25vd2xlZGdlIG9uIEFQSXMKLSBJbnRlcmVzdCB0byBsZWFybiBhbmQgZXhwbG9yZSBhYm91dCBPcGVuQVBJICYgU3dhZ2dlciB0b29scwoKIyBJbXBvcnRhbnQgTGlua3MKCi0gT3BlbiBBUEkgV2Vic2l0ZSAtIGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZwotIFN3YWdnZXIgV2Vic2l0ZSAtIGh0dHBzOi8vc3dhZ2dlci5pbwotIFN3YWdnZXIgRWRpdG9yIC0gaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pbwotIFN3YWdnZXJIdWIgRXhwbG9yZSAtIGh0dHBzOi8vZXhwbG9yZS5zd2FnZ2VyaHViLmNvbS8KLSBPcGVuQVBJIE1hcCAtIGh0dHBzOi8vb3BlbmFwaS1tYXAuYXBpaGFuZHltYW4uaW8vCi0gT3BlbkFQSSAmIFNwcmluZ0Jvb3QgbGlicmFyeSAtIGh0dHBzOi8vc3ByaW5nZG9jLm9yZwotIExpc3Qgb2YgT3BlbkFQSSB0b29scyAtIGh0dHBzOi8vb3BlbmFwaS50b29scy8KLSBQcmlzbSBNb2NrIHNlcnZlciAtIGh0dHBzOi8vc3RvcGxpZ2h0LmlvL29wZW4tc291cmNlL3ByaXNtCi0gU1dBUEkgVGhlIFN0YXIgV2FycyBBUEkgLSBodHRwczovL3N3YXBpLmRldi8KLSBSRVFSRVMgTW9jayBBUElzIC0gaHR0cHM6Ly9yZXFyZXMuaW4= readmeEtag: '"1b581e0ce6d20e6f56a11eba5425139db31a4330"' readmeLastModified: Wed, 21 Aug 2024 05:39:51 GMT repositoryId: 840953515 description: OpenAPI Specs with Swagger Tools created: '2024-08-11T07:44:04Z' updated: '2024-08-21T05:40:29Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: Brisinger logo: https://avatars.githubusercontent.com/u/17980335?v=4 license: Apache-2.0 repoEtag: '"99d7a5a0153a8b8988a44b0235b5efee0b3da4bcc907dc67d2b6aa1d6416c85a"' repoLastModified: Wed, 21 Aug 2024 05:40:29 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/deepakbhalla/springboot-jasypt-encryption-decryption v3: true id: 05121fe4ec0861b0701ed9dabde1661d repositoryMetadata: base64Readme: >- IyBzcHJpbmdib290LWphc3lwdC1lbmNyeXB0aW9uLWRlY3J5cHRpb24NCiANCkFwcGxpY2F0aW9uIHRvIGRlbW9uc3RyYXRlIHRoZSB1c2Ugb2YNCkphc3lwdCB0byBlbmNyeXB0IGFuZCBkZWNyeXB0IGFuZCBzZW5zaXRpdmUgdmFsdWUgaW4gdGhlIGFwcGxpY2F0aW9uLg0KDQojIyMgV2hhdCBpcyBKYXN5cHQNCg0KSmFzeXB0IHN0YW5kcyBmb3IgJ0phdmEgU2ltcGxpZmllZCBFbmNyeXB0aW9uJw0KDQpKYXN5cHQgaXMgYSBqYXZhIGxpYnJhcnkgd2hpY2ggYWxsb3dzIHRoZSBkZXZlbG9wZXIgdG8gYWRkIGJhc2ljIGVuY3J5cHRpb24gY2FwYWJpbGl0aWVzIHRvIGhpcy9oZXIgcHJvamVjdHMgd2l0aCANCm1pbmltdW0gZWZmb3J0LCBhbmQgd2l0aG91dCB0aGUgbmVlZCBvZiBoYXZpbmcgZGVlcCBrbm93bGVkZ2Ugb24gaG93IGNyeXB0b2dyYXBoeSB3b3Jrcy4NCg0KIyMjIFNwcmluZyBCb290IEphc3lwdCBEZXBlbmRlbmN5DQoNCiFbaW1nLnBuZ10oc2NyZWVuc2hvdHMvamFzeXB0X212bl9kZXBlbmRlbmN5LnBuZykNCg0KIyMjIEphc3lwdCBTZWNyZXQgUGFzc3dvcmQNCg0KUHJvdmlkZWQgYXMgVk0gYXJndW1lbnQ6IC1EamFzeXB0LmVuY3J5cHRvci5wYXNzd29yZD1teXBhc3N3b3JkDQoNCiFbamFzeXB0X3Bhc3N3b3JkX3ZtX2FyZ3VtZW50LnBuZ10oc2NyZWVuc2hvdHMlMkZqYXN5cHRfcGFzc3dvcmRfdm1fYXJndW1lbnQucG5nKQ0KDQojIyMgT3BlbkFQSSBTcGVjaWZpY2F0aW9uDQoNCk9wZW5BUEkgc3dhZ2dlciBzaG93cyB0aGUgQVBJIHdoaWNoIHVzZXMgSmFzeXB0IHRvIGVuY3J5cHQgYW5kIGRlY3J5cHQgdGhlIGlucHV0IHByb3ZpZGVkIGJ5IHVzZXIuDQoNCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9teS1hcHBsaWNhdGlvbi9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwNCg0KIVtvcGVuX2FwaV9zd2FnZ2VyLnBuZ10oc2NyZWVuc2hvdHMlMkZvcGVuX2FwaV9zd2FnZ2VyLnBuZykNCg0KIyMjIEVuY3J5cHQgdXNpbmcgSmFzeXB0IGZyb20gQ29tbWFuZCBQcm9tcHQNCg0KLSBDb21tYW5kOg0KYGBgDQpqYXZhIC1jcCBqYXN5cHQtMS45LjMuamFyIG9yZy5qYXN5cHQuaW50Zi5jbGkuSmFzeXB0UEJFU3RyaW5nRW5jcnlwdGlvbkNMSSBpbnB1dD0iRGVlcGFrIiBwYXNzd29yZD0iaGVsbG8iIGFsZ29yaXRobT1QQkVXaXRoTUQ1QW5kREVTDQpgYGANCg0KIVtqYXN5cHRfZW5jcnlwdF9jb21tYW5kX2xpbmUucG5nXShzY3JlZW5zaG90cyUyRmphc3lwdF9lbmNyeXB0X2NvbW1hbmRfbGluZS5wbmcpDQoNCiMjIyBEZWNyeXB0IHVzaW5nIEphc3lwdCBmcm9tIENvbW1hbmQgUHJvbXB0DQoNCi0gQ29tbWFuZDoNCmBgYA0KamF2YSAtY3AgamFzeXB0LTEuOS4zLmphciBvcmcuamFzeXB0LmludGYuY2xpLkphc3lwdFBCRVN0cmluZ0RlY3J5cHRpb25DTEkgaW5wdXQ9Ikk0cEp3WDhFaXRFNlBPMWZMb3VMTXc9PSIgcGFzc3dvcmQ9ImhlbGxvIiBhbGdvcml0aG09UEJFV2l0aE1ENUFuZERFUw0KYGBgDQoNCiFbamFzeXB0X2RlY3J5cHRfY29tbWFuZF9saW5lLnBuZ10oc2NyZWVuc2hvdHMlMkZqYXN5cHRfZGVjcnlwdF9jb21tYW5kX2xpbmUucG5nKQ0KDQpSZWZlcmVuY2VzOg0KDQpodHRwOi8vd3d3Lmphc3lwdC5vcmcvDQoNCg== readmeEtag: '"cd33b1b6b4fc89b2c18024f79aa3ae32fccf363b"' readmeLastModified: Thu, 26 Oct 2023 14:57:00 GMT repositoryId: 710199246 description: >- Application to demonstrate the use of Jasypt to encrypt and decrypt and sensitive value in the application. created: '2023-10-26T08:11:19Z' updated: '2024-10-29T21:41:50Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: deepakbhalla logo: https://avatars.githubusercontent.com/u/6603640?v=4 repoEtag: '"b0dad3f45d9446aca31e1d8908370beb34173556c6db868343363d955f25bdac"' repoLastModified: Tue, 29 Oct 2024 21:41:50 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/otaviotscha/nodejs-api-boilerplate v3: true repositoryMetadata: base64Readme: >- IyBOb2RlSlMgQVBJIEJvaWxlcnBsYXRlCgojIyBSZXF1aXJlbWVudHMKClRvIHJ1biB0aGlzIHByb2plY3QgeW91IHdpbGwgbmVlZDoKCjEuIFtOb2RlSlNdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi9kb3dubG9hZC8pCjIuIFtZYXJuXShodHRwczovL2NsYXNzaWMueWFybnBrZy5jb20vbGFuZy9lbi9kb2NzL2luc3RhbGwpCgojIyBFbnZpcm9ubWVudAoKLSBgLmVudmAgZmlsZSBpcyByZWFkIHdoZW4gaW4gcHJvZHVjdGlvbi4KLSBgLmVudi5kZXZgIGZpbGUgaXMgcmVhZCB3aGVuIGNhbGxpbmcgYHlhcm4gZGV2YCAoZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnQpLgotIGAuZW52LnRlc3RgIGZpbGUgaXMgcmVhZCB3aGVuIGNhbGxpbmcgYHlhcm4gdGVzdGAgKHRlc3QgZW52aXJvbm1lbnQpLgoKWW91IGNhbiB1c2UgYC5lbnYuc2FtcGxlYCB0byBrbm93IGhvdyB0byBjb25maWd1cmUgdGhlIGAuZW52YCB0byB5b3VyIG5lZWRzLgoKYC5lbnYuc2FtcGxlYCBhbHNvIGNvbnRhaW5zIHRoZSBlbnZpcm9ubWVudCBkZWZhdWx0cy4KCiMjIFNjcmlwdHMKCi0gYHlhcm4gcHJpc21hIG1pZ3JhdGUgZGV2YCB0byBnZW5lcmF0ZS9zeW5jIGRhdGFiYXNlIGFjY29yZGluZ2x5IHRvIG1pZ3JhdGlvbnMgKG9yIGdlbmVyYXRlIG1pZ3JhdGlvbiBpZiBgc2NoZW1hLnByaXNtYWAgd2FzIG1vZGlmaWVkKS4KLSBgeWFybiBkZXZgIHRvIHJ1biBhcHAgaW4gZGV2ZWxvcG1lbnQgZW52aXJvbm1lbnQuCi0gYHlhcm4gdGVzdGAgdG8gcnVuIHRlc3RzLgotIGB5YXJuIHByb2RgIHRvIGJ1aWxkIFR5cGVTY3JpcHQgdG8gSmF2YVNjcmlwdCB0aGVuIHJ1biBhcHAgaW4gcHJvZHVjdGlvbiBlbnZpcm9ubWVudCBzaW11bGF0aW9uLgoKIyMgQVBJIERvY3VtZW50YXRpb24KCi0gYGh0dHA6Ly97aG9zdH06e3BvcnR9L2RvY3NgIGRlZmF1bHQ6IDxodHRwOi8vbG9jYWxob3N0OjQwMDAvZG9jcz4KCiMjIE1haW4gcGFja2FnZXMKCi0gW1R5cGVTY3JpcHRdKGh0dHBzOi8vZ2l0aHViLmNvbS9taWNyb3NvZnQvVHlwZVNjcmlwdCkKLSBIVFRQIFNlcnZlcjogW3JvdXRpbmctY29udHJvbGxlcnNdKGh0dHBzOi8vZ2l0aHViLmNvbS90eXBlc3RhY2svcm91dGluZy1jb250cm9sbGVycykgYW5kIFtleHByZXNzXShodHRwczovL2dpdGh1Yi5jb20vZXhwcmVzc2pzL2V4cHJlc3MpCi0gRGF0YWJhc2U6IFtwcmlzbWFdKGh0dHBzOi8vZ2l0aHViLmNvbS9wcmlzbWEvcHJpc21hKQotIFJlcXVlc3RzIHZhbGlkYXRpb25zOiBbY2xhc3MtdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20vdHlwZXN0YWNrL2NsYXNzLXZhbGlkYXRvcikKLSBBUEkgZG9jdW1lbnRhdGlvbjogW3N3YWdnZXItdWktZXhwcmVzc10oaHR0cHM6Ly9naXRodWIuY29tL3Njb3R0aWUxOTg0L3N3YWdnZXItdWktZXhwcmVzcykKLSBUZXN0czogW2plc3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9mYWNlYm9vay9qZXN0KQotIExvZ3M6IFt3aW5zdG9uXShodHRwczovL2dpdGh1Yi5jb20vd2luc3RvbmpzL3dpbnN0b24pCg== readmeEtag: '"100a24355fee76c1cf762e5dcc8d4bff55c8f789"' readmeLastModified: Tue, 14 Dec 2021 14:44:10 GMT repositoryId: 438273922 description: null created: '2021-12-14T14:00:10Z' updated: '2024-10-31T10:59:25Z' language: TypeScript archived: true stars: 1 watchers: 2 forks: 0 owner: otaviotscha logo: https://avatars.githubusercontent.com/u/15371120?v=4 license: MIT repoEtag: '"2bf38bf111ff90e7f38af02357647521bed4185cb30917fa09b02912c58a0ced"' repoLastModified: Thu, 31 Oct 2024 10:59:25 GMT foundInMaster: true category: - Server - Server Implementations id: 109c0b426faa2b9f363ed379578bc857 - source: openapi3 tags repository: https://github.com/open-banking/standingorders v3: true repositoryMetadata: base64Readme: >- IyBzdGFuZGluZ09yZGVycwpPcGVuIEJhbmtpbmcgU3RhbmRpbmcgT3JkZXJzIEFQSSBidWlsdCBvbiB0b3Agb2YgbGlnaHQtNGoK readmeEtag: '"63ad7a532c05ffd67964bdc9bb5be93facca67cf"' readmeLastModified: Thu, 25 Jul 2024 20:09:50 GMT repositoryId: 235862750 description: Open Banking Standing Orders API built on top of light-4j created: '2020-01-23T18:48:43Z' updated: '2026-02-03T01:08:51Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"e90934d76d299e2b446c2cd258ef68807450ed7877e678dde9aab0003ae23f84"' repoLastModified: Tue, 03 Feb 2026 01:08:51 GMT foundInMaster: true category: Server id: 179c04aca24503cf8be7f31c1cd155be - source: openapi3 tags repository: https://github.com/expediagroup/spec-transformer v3: true id: 33e7adea5a5bffe320e564ae655045c1 repositoryMetadata: base64Readme: >- IyBTcGVjIFRyYW5zZm9ybWVyCgpUaGUgQVBJIFNwZWMgVHJhbnNmb3JtZXIgTGlicmFyeQoKIyMgSW5zdGFsbGF0aW9uCgpgYGBiYXNoCm5wbSBpbnN0YWxsIEBleHBlZGlhZ3JvdXAvc3BlYy10cmFuc2Zvcm1lcgpgYGAKCiMjIFVzYWdlCgpQaWNrIG9uZSBvZiB0aGUgZm9sbG93aW5nIHdheXMgdG8gdXNlIHRoZSBsaWJyYXJ5OgoKIyMjIDEuIFVzZSBzcGVjLXRyYW5zZm9ybWVyIGFzIGEgbGlicmFyeQoKRXhhbXBsZToKCmBgYHR5cGVzY3JpcHQKaW1wb3J0IHsgSGVhZGVyUmVtb3ZhbFRyYW5zZm9ybWVyLCBUcmFuc2Zvcm1lckNoYWluLCBZYW1sUmVhZGVyLCBZYW1sV3JpdGVyIH0gZnJvbSAnQHNwZWMtdHJhbnNmb3JtZXInOwoKY29uc3Qgb3BlbmFwaXNwZWNzID0gJy4uLic7IC8vIE9wZW5BUEkgc3BlY3MgaW4gSlNPTiBvciBZQU1MIGZvcm1hdAoKY29uc3QgdHJhbnNmb3JtZXJzID0gbmV3IFRyYW5zZm9ybWVyQ2hhaW4oWwogIG5ldyBIZWFkZXJSZW1vdmFsVHJhbnNmb3JtZXIoKSAvLyBBZGQgbW9yZSB0cmFuc2Zvcm1lcnMgaGVyZQpdKTsKCmNvbnN0IHRyYW5zZm9ybWVkU3BlY3MgPSB0cmFuc2Zvcm1lcnMudHJhbnNmb3JtKG9wZW5hcGlzcGVjcywgWWFtbFJlYWRlciwgWWFtbFdyaXRlcik7Cgpjb25zb2xlLmxvZyh0cmFuc2Zvcm1lZFNwZWNzKTsKYGBgCgojIyMgMi4gVXNlIHNwZWMtdHJhbnNmb3JtZXIgYXMgYSBDTEkKCmBgYGJhc2gKbnB4IC1wIEBleHBlZGlhZ3JvdXAvc3BlYy10cmFuc2Zvcm1lciBjbGkgLS1oZWxwICAjIFNob3cgaGVscCwgYW5kIGxpc3QgYWxsIGF2YWlsYWJsZSBjb21tYW5kcy4KYGBgCgpFeGFtcGxlOgoKYGBgYmFzaApucHggLXAgQGV4cGVkaWFncm91cC9zcGVjLXRyYW5zZm9ybWVyIGNsaSAtLWlucHV0IHNwZWNzLnlhbWwgLS1vdXRwdXQgb3V0LnlhbWwgLS1oZWFkZXJzICAjIFJlYWQgc3BlY3MgZnJvbSBzcGVjcy55YW1sLCByZW1vdmUgaGVhZGVycywgYW5kIHdyaXRlIHRvIG91dC55YW1sCmBgYAoKIyMjIDMuIEJ1aWxkIGFuZCBydW4gc3BlYy10cmFuc2Zvcm1lciBsb2NhbGx5CgpgYGBiYXNoCm5wbSBpbnN0YWxsCm5wbSBydW4gYnVpbGQKYGBgCgojIyMgVGVzdAoKYGBgYmFzaApucG0gdGVzdApgYGAKCi0tLQoKIyMgRGV2ZWxvcG1lbnQgVGVhbQotIFtNb2hhbW1hZCBOb29yIEFidSBLaGxlaWZdKGh0dHBzOi8vZ2l0aHViLmNvbS9tb2hub29yOTQpCi0gW09zYW1hIFNhbG1hbl0oaHR0cHM6Ly9naXRodWIuY29tL29zYW1hLXNhbG1hbjk5KQoK readmeEtag: '"b96799fa76e57d9b9d617ce95137c0dc0f122bf4"' readmeLastModified: Mon, 12 Aug 2024 10:21:45 GMT repositoryId: 651020076 description: The API Spec Transformer Library created: '2023-06-08T10:09:10Z' updated: '2025-09-18T15:14:52Z' language: TypeScript archived: false stars: 1 watchers: 2 forks: 0 owner: ExpediaGroup logo: https://avatars.githubusercontent.com/u/38541875?v=4 license: Apache-2.0 repoEtag: '"e25d0070d6f658b4f80db97db0d9b68fdb3cf7af257fe3f320e840cc02394973"' repoLastModified: Thu, 18 Sep 2025 15:14:52 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/sensiblewood/swaggerui-webpack-boilerplate v3: true repositoryMetadata: base64Readme: >- T3ZlcnZpZXcKPT09CgpUaGlzIGlzIGJvaWxlcnBsYXRlIHByb2plY3QgZm9yIHJ1bm5pbmcgU3dhZ2dlciBVSSB3aXRoIFdlYnBhY2suIEl0IGV4aXN0cyBiZWNhdXNlOgoKKiBJIHdhbnRlZCBhbiBlYXN5IHdheSBvZiBzbGFtbWluZyBtdWx0aXBsZSBBUEkgc3BlY2lmaWNhdGlvbnMgaW50byBhIFN3YWdnZXIgVUktYmFzZWQgcHJvamVjdCBhbmQgcnVubmluZyBpdCBsb2NhbGx5LgoqIEkgZGlkbid0IHdhbnRlZCB0byBtYW51YWxseSBjaGFuZ2UgdGhlIEphdmFzY3JpcHQgZXZlcnkgdGltZSBJIGFkZGVkIGFuIEFQSSBzcGVjaWZpY2F0aW9uLgoqIFVzaW5nIE5QTS9XZWJwYWNrIG1lYW5zIG15IGRlcGVuZGVuY2llcyBjYW4gYmUga2VwdCB1cC10by1kYXRlIGVhc2lseS4KClN1cmUgdGhlcmUncyBvdGhlciB3YXlzIG9mIGRvaW5nIHRoaXMsIGJ1dCBzdGlsbC4uLgoKflRoZXJlJ3MgYWxzbyBhIFJlZG9jIHBhZ2UgLSBhbHRob3VnaCB0aGlzIGlzIENETi1iYXNlZCBhcyBJIGFtIGxhenkufiBJIHJlbW92ZWQgdGhlIFJlZG9jIHZlcnNpb24gZm9yIHRoZSB0aW1lIGJlaW5nLgoKVXNhZ2UKPT09CgpUbyB1c2UgdGhpcyBwcm9qZWN0IGNsb25lIGFuZCBjZCBpbnRvIGl0IHRoZW46CgpgYGBiYXNoCm5wbSBpCmNwIFt5b3VyIFN3YWdnZXIvT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBhcyBZQU1MXSBzcmMvc3BlY3MKbnBtIHJ1biBkZXYKYGBgCgo6dGh1bWJzdXA6CgpJbXByb3ZlbWVudHMKPT09CgoqIFsgXSBQYXJhbWV0ZXJpc2UgQVBJIHNwZWNpZmljYXRpb24gZG9jdW1lbnQgbmFtZS4KKiBbIF0gUmVmYWN0b3IgYHNjcmlwdHMvcmV3cml0ZS11cmxzLmpzYCBhcyBhIFdlYnBhY2sgcGx1Z2luLgoK readmeEtag: '"507aa0c5f03cd0c62a0bfc8ae13d98dda0e01e12"' readmeLastModified: Tue, 16 Oct 2018 11:51:37 GMT repositoryId: 151232364 description: >- Boilerplate for spinning up local Swagger UI server, orchestrated through Webpack created: '2018-10-02T09:41:56Z' updated: '2021-02-15T13:50:01Z' language: JavaScript archived: false stars: 1 watchers: 0 forks: 1 owner: SensibleWood logo: https://avatars.githubusercontent.com/u/2420069?v=4 repoEtag: '"44cb7d75779a973eb4b693bd4b7fb62e0fa91f6185328f05b002c0e7e3dfbee5"' repoLastModified: Mon, 15 Feb 2021 13:50:01 GMT foundInMaster: true category: - Server - Server Implementations id: 3d8bdcdeba005c2ba64cd69ec363f143 - source: openapi3 tags repository: https://github.com/tdbrian/react-openapi-hooks-gen v3: true repositoryMetadata: base64Readme: >- ## react-openapi-hooks-gen: An OpenAPI 3 code generator for React using hooks, redux, and typescript

![Build Status](https://github.com/tdbrian/react-openapi-hooks-gen/workflows/build/badge.svg)
![test](https://github.com/tdbrian/react-openapi-hooks-gen/workflows/test/badge.svg)

This project is a NPM module that generates model interfaces, redux actions and reducers, and hooks from an [OpenApi 3](https://www.openapis.org/) [specification](https://github.com/OAI/OpenAPI-Specification).
The generated classes follow the principles of [React](https://reactjs.org/).
The generated code is compatible with React v16.13.1.

## Highlights

- It should be easy to use and to integrate with Create React App CLI;
- It should support `OpenAPI` specifications in both `JSON` and `YAML` formats;
- `OpenAPI` supports combinations of request body and response content types.
  For each combination, a distinct method is generated;
- It should be possible to specify a subset of reducers to generate.
  Only the models actually used by that subset should be generated;
- It should be easy to specify a root URL for the web service endpoints;
- Generated files should compile using strict `TypeScript` compiler flags, such as `noUnusedLocals` and `noUnusedParameters`.
- It should use the standard fetch api
- It should use redux to easily allow devs to see current state and actions being fired
- It should expose functionality to views through custom hooks

## Limitations

- Only standard OpenAPI 3 descriptions will be generated. `react-openapi-hooks-gen` allows several extensions, specially types from JSON schema, but they are out of scope for `react-openapi-hooks-gen`. There is, however, support for a few [vendor extensions](#Supported_vendor_extensions);
- Servers per operation are not supported;
- Only the first server is used as a default root URL in the configuration;
- No data transformation is ever performed before sending / after returning data.
  This means that a property of type `string` and format `date-time` will always be generated as `string`, not `Date`.
  Otherwise every API call would need to have a processing that would traverse the returned object graph before sending the request
  to replace all date properties by `Date`. The same applies to sent requests. Such operations are out of scope for `react-openapi-hooks-gen`;

## Installing and running

You may want to install `react-openapi-hooks-gen` globally or just on your project. Here is an example for a global setup:

```bash
$ npm install -g react-openapi-hooks-gen
$ react-openapi-hooks-gen --input my-api.yaml --output my-app/src/api
```

This will expect the file `my-api.yaml` to be in the current directory, and will generate the files on `my-app/src/api`.

## Configuration file and CLI arguments

If the file `react-openapi-hooks-gen.json` exists in the current directory, it will be read. Alternatively, you can run `react-openapi-hooks-gen --config my-config.json` (could also be `-c`) to specify a different configuration file, or even specify the input / output as `react-openapi-hooks-gen -i input.yaml` or `react-openapi-hooks-gen -i input.yaml -o /tmp/generation`.
The only required configuration property is `input`, which specified the `OpenAPI` specification file. The default `output` is `src/api`.

For a list with all possible configuration options, see the [JSON schema file](https://raw.githubusercontent.com/tdbrian/react-openapi-hooks-gen/master/react-openapi-hooks-gen-schema.json).
You can also run `react-openapi-hooks-gen --help` to see all available options.
Each option in the JSON schema can be passed in as a CLI argument, both in camel case, like `--includeTags tag1,tag2,tag3`, or in kebab case, like `--exclude-tags tag1,tag2,tag3`.

Here is an example of a configuration file:

```json
{
  "$schema": "node_modules/react-openapi-hooks-gen/react-openapi-hooks-gen-schema.json",
  "input": "my-file.json",
  "output": "out/person-place",
  "ignoreUnusedModels": false
}
```

## Specifying the root URL / web service endpoint

TBD

## Passing request headers / customizing the request

TBD

## Supported vendor extensions

Besides the OpenAPI 3 specification, the following vendor extensions are supported:

- `x-operation-name`: Defined in [LoopBack](https://loopback.io/doc/en/lb4/Decorators_openapi.html), this extension can be used in operations to specify the actual method name. The `operationId` is required to be unique among all tags, but with this extension, a shorter method name can be used per tag (service). Example:

```yaml
paths:
  /users:
    get:
      tags:
        - Users
      operationId: listUsers
      x-operation-name: list
      # ...
  /places:
    get:
      tags:
        - Places
      operationId: listPlaces
      x-operation-name: list
      # ...
```

- `x-enumNames`: Generated by [NSwag](https://github.com/RicoSuter/NSwag), this extension allows schemas which are enumerations to customize the enum names. It must be an array with the same length as the actual enum values. Example:

```yaml
components:
  schemas:
    HttpStatusCode:
      type: integer
      enum:
        - 200
        - 404
        - 500
      x-enumNames:
        - OK
        - NOT_FOUND
        - INTERNAL_SERVER_ERROR
```

## Developing and contributing

The generator itself is written in TypeScript. When building, the code is transpiled to JavaScript in the `dist` folder. And the `dist` folder is the one that gets published to NPM. Even to prevent publishing from the wrong path, the `package.json` file has `"private": true`, which gets replaced by `false` in the build process.

On the other hand, for developing / running tests, `jasmine-ts` is used, so the tests run directly from TypeScript. There's even a committed VisualStudio Code debug configuration for tests.

After developing the changes, to `link` the module and test it with other node projects, run the following:

```bash
npm run build
cd dist
npm link
```

At that point, the globally available react-openapi-hooks-gen will be the one compiled to the `dist` folder.
 readmeEtag: '"f47549a41abff4d4e6f75ff06aa12a1a2ac879fa"' readmeLastModified: Thu, 18 Jun 2020 03:26:33 GMT repositoryId: 273115846 description: An OpenAPI 3 codegen for React Hooks using Typescript created: '2020-06-18T01:42:52Z' updated: '2020-09-20T23:35:07Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: tdbrian logo: https://avatars.githubusercontent.com/u/1131354?v=4 license: MIT repoEtag: '"3a5d0d8cdf880cd7c86b57886b8b709a56934ec236ae36a78ba5b3548b3eecfd"' repoLastModified: Sun, 20 Sep 2020 23:35:07 GMT foundInMaster: true category: - Testing - Parsers id: ae6d3cb6d80559b99967f4f833af10af - source: openapi3 tags repository: https://github.com/weaponsforge/apidocs-swagger-ui v3: true id: ca0080f7abdd181cfa141adbbb44becd repositoryMetadata: base64Readme: >- IyMgYXBpZG9jcy1zd2FnZ2VyLXVpCgoqKmFwaWRvY3Mtc3dhZ2dlci11aSoqIGZlYXR1cmVzIGEgc2ltcGxlIFRvZG8gbm90ZXMtdGFraW5nIENSVVAgQVBJIGJvcnJvd2VkIGZyb20gW2B0b2RvLW5leHRgXShodHRwczovL2dpdGh1Yi5jb20vd2VhcG9uc2ZvcmdlL3RvZG8tbmV4dCkncyBzZXJ2ZXIuCgpUaGlzIHJlcG9zaXRvcnkgYWltcyB0byB0ZXN0IHVzaW5nIFtzd2FnZ2VyLXVpXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci11aSkgZm9yIGNyZWF0aW5nIG1vZGVybiwgcmVzcG9uc2l2ZSBhbmQgaW50ZXJhY3RpdmUgUkVTVCBBUEkgZG9jdW1lbnRhdGlvbnMgYW5kIHRvIGZhbWlsaWFyaXplIG9uZXNlbGYgd2l0aCB0aGUgW09wZW5BUEldKGh0dHBzOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjEuMCkgc3BlY2lmaWNhdGlvbnMsIHYzLjEuMCBhcyBvZiB0aGlzIHdyaXRpbmcuCgohW3NjcmVlbnNob3RdKC9hc3NldHMvdGh1bWJuYWlsLnBuZykKCiMjIFJlcXVpcmVtZW50cwoKVGhlIGZvbGxvd2luZyBkZXBlbmRlbmNpZXMgYXJlIHVzZWQgZm9yIHRoaXMgcHJvamVjdCdzIGxvY2FsaG9zdCBkZXZlbG9wbWVudCBlbnZpcm9ubWVudC4gRmVlbCBmcmVlIHRvIHVzZSBvdGhlciBkZXBlbmRlbmN5IHZlcnNpb25zIGFzIG5lZWRlZC4KCjEuIFdpbmRvd3MgMTAgT1MKMi4gbnZtIGZvciBXaW5kb3dzIHYxLjEuOQozLiBOb2RlSlMgMTYuMTQuMiBpbnN0YWxsZWQgdXNpbmcgbnZtCiAgIC0gbm9kZSB2MTYuMTQuMgogICAtIG5wbSB2OC41LjAKNC4gTW9uZ29EQiBDb21tdW5pdHkgRWRpdGlvbiAoZm9yIFdpbmRvd3MpCiAgIC0gdmVyc2lvbiA0LjQuMAogICAtIGFyY2hpdGVjdHVyZTogeDg2XzY0CgojIyMgQ29yZSBMaWJyYXJpZXMgYW5kIEZyYW1ld29ya3MKCjEuIHN3YWdnZXItdWkgdjQuMTQuMAoyLiB3ZWJwYWNrIHY1Ljc0LjAKMy4gd2VicGFjay1jbGkgdjQuMTAuMAo0LiBtb25nb29zZSB2Ni41LjIKCiMjIEluc3RhbGxhdGlvbgoKMS4gQ2xvbmUgdGhpcyByZXBvc2l0b3J5Ljxicj4KYGh0dHBzOi8vZ2l0aHViLmNvbS93ZWFwb25zZm9yZ2UvdG9kby1uZXh0LmdpdGAKCjIuIEluc3RhbGwgdGhlIGJhY2tlbmQgc2VydmVyIGRlcGVuZGVuY2llcy48YnI+CmBucG0gaW5zdGFsbGAKCjMuIEluc3RhbGwgdGhlICoqZG9jcyoqIGRlcGVuZGVuY2llcy48YnI+CmBucG0gcnVuIGRvY3M6aW5zdGFsbGAKCjQuIFNldCB1cCB0aGUgZW52aXJvbm1lbnQgdmFyaWFibGVzLiBDcmVhdGUgYSBgLmVudiBgZmlsZSBpbnNpZGUgdGhlIHJvb3QgcHJvamVjdCBkaXJlY3Rvcnkgd2l0aCByZWZlcmVuY2UgdG8gdGhlIGAuZW52LmV4YW1wbGVgIGZpbGUuPGJyPgoKICAgfCBWYXJpYWJsZSBOYW1lICAgICAgICAgfCBEZXNjcmlwdGlvbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CiAgIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAogICB8IE1PTkdPX1VSSSAgICAgICAgICAgICB8IE1vbmdvREIgY29ubmVjdGlvbiBzdHJpbmcuPGJyPkRlZmF1bHQgdmFsdWUgdXNlcyB0aGUgbG9jYWxob3N0IE1vbmdvREIgY29ubmVjdGlvbiBzdHJpbmcuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICAgfCBBTExPV0VEX09SSUdJTlMgICAgICAgfCBJUC9kb21haW4gb3JpZ2lucyBpbiBjb21tYS1zZXBhcmF0ZWQgdmFsdWVzIHRoYXQgYXJlIGFsbG93ZWQgdG8gYWNjZXNzIHRoZSBBUEkgaWYgYEFMTE9XX0NPUlM9MWAuPGJyPiBJbmNsdWRlIGBodHRwOi8vbG9jYWxob3N0OjgwODBgIGJ5IGRlZmF1bHQgdG8gYWxsb3cgQ09SUyBhY2Nlc3MgdG8gdGhlIHN3YWdnZXItdWkgZG9jdW1lbnRhdGlvbiBhcHAgaW4gdGhlICoqL2RvY3MqKiBkaXJlY3RvcnkuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CiAgIHwgQUxMT1dfQ09SUyAgICAgICAgICAgIHwgQWxsb3cgQ3Jvc3MtT3JpZ2luIFJlc291cmNlIFNoYXJpbmcgKENPUlMpIG9uIHRoZSBBUEkgZW5kcG9pbnRzLjxicj48YnI+RGVmYXVsdCB2YWx1ZSBpcyBgMWAsIGFsbG93aW5nIGFjY2VzcyB0byBkb21haW5zIGxpc3RlZCBpbiBgQUxMT1dFRF9PUklHSU5TYC48YnI+IFNldHRpbmcgdG8gYDBgIHdpbGwgYmxvY2sgQUpBWCBBUEkgcmVxdWVzdHMgZnJvbSBjbGllbnQgd2Vic2l0ZXMgaG9zdGVkIG9uIG90aGVyIGRvbWFpbnMsIGJ1dCB3aWxsIGFsbG93IHJlcXVlc3RzIGZyb20gUG9zdG1hbiBhbmQgb3RoZXIgdGVzdGluZyB0b29scy4gfAogICB8IEFQSV9XSU5ET1dfTVNfTUlOVVRFUyB8IFRpbWUgaW4gYG1pbnV0ZXNgIHdoZXJlIGBBUElfUkFURV9MSU1JVGAgdGltZXMgb2Ygc3VjY2Vzc2l2ZSBjYWxscyBmcm9tIGFuIElQIGFyZSBhbGxvd2VkIG9uIHRoZSBzZXJ2ZXIuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKICAgfCBBUElfUkFURV9MSU1JVCAgICAgICAgfCBJdCdzIHRoZSBtYXhpbXVtIG51bWJlciBvZiBhbGxvd2VkIEFQSSByZXF1ZXN0cyBvbiB0aGUgc2VydmVyIHBlciBgQVBJX1dJTkRPV19NU19NSU5VVEVTYC4gPGJyPlVzZXJzIHdpbGwgcmVjZWl2ZSBhIGA0MjkgLSBUb28gbWFueSByZXF1ZXN0c2Agc2VydmVyIGVycm9yIGFmdGVyIGhpdHRpbmcgdGhlIGxpbWl0Ljxicj5UaGUgbGltaXQgd2lsbCByZXNldCBhZnRlciBBUElfV0lORE9XX01TX01JTlVURVMgbWludXRlcywgYWZ0ZXIgd2hpY2ggdXNlcnMgY2FuIHJlc3VtZSBtYWtpbmcgQVBJIHJlcXVlc3RzLiAgICAgICAgICAgICAgICB8CgojIyBVc2FnZQoKMS4gUnVuIHRoZSBzZXJ2ZXIuPGJyPgpgbnBtIHN0YXJ0YAoKMS4gUnVuIHRoZSBkb2NzIGFwaSBvbiBkZXZlbG9wbWVudCBtb2RlLjxicj4KYG5wbSBydW4gZG9jczpkZXZgCgojIyBBdmFpbGFibGUgU2NyaXB0cwoKIyMjIGBucG0gc3RhcnRgCgpSdW4gdGhlIGV4cHJlc3Mgc2VydmVyIGZvciBwcm9kdWN0aW9uIG1vZGUuCgojIyMgYG5wbSBydW4gZGV2YAoKUnVuIHRoZSBzZXJ2ZXIgZm9yIGRldmVsb3BtZW50IG1vZGUgdXNpbmcgW25vZGVtb25dKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL25vZGVtb24pLgoKIyMjIGBucG0gcnVuIGxpbnRgCgpMaW50IHRoZSBzZXJ2ZXIgSmF2YVNjcmlwdCBzb3VyY2UgY29kZXMuCgojIyMgYG5wbSBydW4gbGludDpmaXhgCgpGaXggdGhlIEphdmFTY3JpcHQgbGludCBlcnJvcnMuCgojIyMgYG5wbSBydW4gZG9jczppbnN0YWxsYAoKSW5zdGFsbHMgdGhlIChzd2FnZ2VyLXVpICsgd2VicGFjaykgZGVwZW5kZW5jaWVzIG9uIHRoZSAqKi9kb2NzKiogZGlyZWN0b3J5LgoKIyMjIGBucG0gcnVuIGRvY3M6ZGV2YAoKUnVucyB0aGUgKHN3YWdnZXItdWkgKyB3ZWJwYWNrKSBhcHAgaW4gZGV2ZWxvcG1lbnQgbW9kZS4KCiMjIyBgbnBtIHJ1biBkb2NzOmJ1aWxkYAoKQnVpbGRzIGFuZCBidW5kbGVzIHRoZSAoc3dhZ2dlci11aSArIHdlYnBhY2spIGFwcCBpbiB0aGUgKiovZG9jcyoqIGRpcmVjdG9yeSBmb3IgcHJvZHVjdGlvbiBtb2RlIHRvIHRoZSBgL2RvY3MvZGlzdGAgZGlyZWN0b3J5LgoKRml4IEphdmFTY3JpcHQgbGludCBlcnJvcnMuCgpAd2VhcG9uc2ZvcmdlPGJyPgoyMDIyMDgyMQo= readmeEtag: '"341743a18d945d64424aa1b64e75c7ca483c30ae"' readmeLastModified: Thu, 25 Aug 2022 03:32:30 GMT repositoryId: 527131359 description: Testing using swagger ui for api documentation created: '2022-08-21T07:04:33Z' updated: '2025-03-05T21:34:39Z' language: JavaScript archived: false stars: 2 watchers: 1 forks: 0 owner: weaponsforge logo: https://avatars.githubusercontent.com/u/56998001?v=4 repoEtag: '"807cc25b876593be45bfdc6a201cfa5361f506947c8a1dfbc00c22d8167c1fe1"' repoLastModified: Wed, 05 Mar 2025 21:34:39 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/anshulsharma1610/trendr-news_summarization v3: true id: 5eef97e04c181100276e394f5a472198 repositoryMetadata: base64Readme: >- IyMgVHJlbmRyIC0gU3RheSBhaGVhZCBvZiB0aGUgbmV3cyANCg0KDQpQUkVTRU5UQVRJT04gVklERU8gTElOSzogaHR0cHM6Ly9kcml2ZS5nb29nbGUuY29tL2ZpbGUvZC8xQ3JpVk53WFE5anYxbDRoR0Q4dXl0dFNQOWd5MWtSTVIvdmlldz91c3A9c2hhcmluZw0KDQogIyMjIFByb2plY3QgRGVzY3JpcHRpb246IFRyZW5kcg0KDQpUaGlzIHByb2plY3QgaXMgYSBuZXdzIGFwcGxpY2F0aW9uIHRoYXQgYWxsb3dzIHVzZXJzIHRvIGNyZWF0ZSBhbiBhY2NvdW50IGFuZCBsb2cgaW4gd2l0aCBhIHVuaXF1ZSB1c2VybmFtZSBhbmQgcGFzc3dvcmQuIFVzZXJzIGNhbiB0aGVuIHNldCBhbmQgY2hhbmdlIHRoZWlyIHByZWZlcmVuY2VzIGZvciBuZXdzIGNhdGVnb3JpZXMgYW5kIHNvdXJjZXMuDQpUaGUgbmV3cyBmZWVkIGRpc3BsYXlzIG5ld3Mgc3VtbWFyaWVzIGJhc2VkIG9uIHRoZSB1c2VyJ3MgcHJlZmVyZW5jZXMuIEFkZGl0aW9uYWxseSwgdGhlIGFwcGxpY2F0aW9uIHNob3dzIHRvcCB0d2VldHMgYmFzZWQgb24gdGhlIHVzZXIncyBuZXdzIHByZWZlcmVuY2UgYW5kIHRvcCBuZXdzIGhlYWRsaW5lcyBpbiB0aGUgdHJlbmRpbmcgc2VjdGlvbiBpcnJlc3BlY3RpdmUgb2YgdGhlaXIgcHJlZmVyZW5jZXMuDQpVc2VycyBjYW4gbGlrZSwgY29tbWVudCwgYW5kIHNoYXJlIG5ld3MgYXJ0aWNsZXMgYW5kIGFsc28gc2F2ZS9ib29rbWFyayBhcnRpY2xlcyBmb3IgbGF0ZXIgcmVhZGluZy4gVGhlIGFwcGxpY2F0aW9uIGFsc28gYWxsb3dzIHVzZXJzIHRvIHNlYXJjaCBmb3IgbmV3cyBhcnRpY2xlcyBieSBrZXl3b3JkcyBhbmQgY2F0ZWdvcmllcyBhbmQgdHJhbnNsYXRlIGFydGljbGVzIGludG8gb3RoZXIgbGFuZ3VhZ2VzLg0KSW4gdGVybXMgb2Ygc3Vic2NyaXB0aW9uIG1hbmFnZW1lbnQsIHVzZXJzIGNhbiBlbnJvbGwgYW5kIG1hbmFnZSB0aGVpciBhcHAgc3Vic2NyaXB0aW9uLiBQYWlkIHN1YnNjcmliZXJzIGNhbiBiZSB2aWV3ZWQgYnkgdGhlIGFkbWluLg0KVGhlIGFkbWluIGZlYXR1cmVzIGluY2x1ZGUgdGhlIGFiaWxpdHkgdG8gYWRkIG5ld3MgYXJ0aWNsZXMgdG8gdGhlIHdlYnNpdGUsIG1hbmFnZSB1c2VycywgdmlldyBhbmFseXRpY3MgZGFzaGJvYXJkLCBhbmQgYWRkIG90aGVyIGFkbWlucy4gTW9kZXJhdGlvbiBmZWF0dXJlcyBpbmNsdWRlIHRoZSBhYmlsaXR5IGZvciBtb2RlcmF0b3JzIHRvIGRlbGV0ZSBjb21tZW50cyBhbmQgYmxvY2sgdXNlcnMuDQpUaGUgYW5hbHl0aWNzIGRhc2hib2FyZCBwcm92aWRlcyBpbnNpZ2h0cyBvbiB3ZWJzaXRlIHRyYWZmaWMgYW5kIHVzZXIgYmVoYXZpb3IsIHdoaWNoIGNhbiBiZSB1c2VkIHRvIG1ha2UgZGF0YS1kcml2ZW4gZGVjaXNpb25zLg0KVGhpcyBuZXdzIGFwcGxpY2F0aW9uIHByb3ZpZGVzIGEgY29tcHJlaGVuc2l2ZSBuZXdzIGV4cGVyaWVuY2UgZm9yIHVzZXJzIHdoaWxlIG9mZmVyaW5nIHBvd2VyZnVsIG1hbmFnZW1lbnQgdG9vbHMgZm9yIHRoZSBhZG1pbi4NCg0KDQogDQogIyMjIE9iamVjdCBNb2RlbCBEaWFncmFtDQoNCg0KDQoNCiFbVHJlbmRlciBkcmF3aW9dKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzQyNjY4OTc5LzIyNjk5MTQyNi05MDZlZTA5Yi01NDNlLTRhYTgtYTg0Yi1kODM4Y2NjM2Q1YTIucG5nKQ0KDQoNCiMjIyBFeHRlcm5hbCBBUElzDQoNCmh0dHBzOi8vbmV3c2RhdGEuaW8vICA8YnI+DQpodHRwczovL3BsYXRmb3JtLm9wZW5haS5jb20vIDxicj4NCmh0dHBzOi8vc3RyaXBlLmNvbS9kb2NzL3BheW1lbnRzIDxicj4NCmh0dHBzOi8vcmF6b3JwYXkuY29tL2RvY3MvI2hvbWUtcGF5bWVudHMNCg0KLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KDQogIyMjIEluc3RydWN0aW9ucyB0byB1c2UgdGhlIHJlcG8NCg0KLSBZb3UgY2FuIGNsb25lIHRoZSByZXBvIGJ5IHVzaW5nIEhUVFA6IA0KDQpodHRwczovL2dpdGh1Yi5jb20vbmV1LW1pcy1pbmZvLTYxNTAtc3ByaW5nLTIwMjMvZmluYWwtcHJvamVjdC1ncm91cC13ZWJfd2VhdmVycy5naXQNCg0KT1Igc2V0IHVwIHRoZSBTU0ggS2V5IHVzaW5nOiANCg0KZ2l0QGdpdGh1Yi5jb206bmV1LW1pcy1pbmZvLTYxNTAtc3ByaW5nLTIwMjMvZmluYWwtcHJvamVjdC1ncm91cC13ZWJfd2VhdmVycy5naXQNCg0KQ29tbWFuZHMgdG8gdXNlOiANCg0KZ2l0IGNsb25lIGA8dXNlIEhUVFAgTGluayBvciBTU0ggTGluaz5gDQoNCiMjIyMgVG8gcnVuIFN0cmlwZSBXZWJob29rIHRocm91Z2h0IFN0cmlwZSBDTEkgb24gbG9jYWw6IHN0cmlwZSBsaXN0ZW4gLS1mb3J3YXJkLXRvIGxvY2FsaG9zdDo0MjQyL3dlYmhvb2sgDQotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLQ0KDQoNCg== readmeEtag: '"1f9803fc2dc0be72c0ec0e7ebc32518d38045c94"' readmeLastModified: Tue, 23 Sep 2025 19:12:07 GMT repositoryId: 638790054 description: >- News application that provides users with a concise summary of the latest news in short form. Users can view news for free, & for unlimited articles, they can purchase a subscription. It has both an admin panel & user panel, allowing administrators to manage and update news content, and users to customize news preferences and track subscriptions. created: '2023-05-10T05:40:52Z' updated: '2025-09-23T19:13:00Z' language: JavaScript archived: false stars: 1 watchers: 1 forks: 0 owner: anshulsharma1610 logo: https://avatars.githubusercontent.com/u/114267698?v=4 repoEtag: '"c6e61a3d63064e9a267acb6ea49ff7ffd74cb9e4f982ec334651cdeea39b1e25"' repoLastModified: Tue, 23 Sep 2025 19:13:00 GMT category: - Testing - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/didifive/peoplehub v3: true id: 2c68bc982c9ec4a2c8ad771526ffcb10 repositoryMetadata: base64Readme: >- ![Repository language count](https://img.shields.io/github/languages/count/didifive/peoplehub)
![Repository code size](https://img.shields.io/github/languages/code-size/didifive/peoplehub)
[![GitHub last commit](https://img.shields.io/github/last-commit/didifive/peoplehub?color=blue)](https://github.com/didifive/peoplehub/commits/master)

![coverage](https://img.shields.io/badge/coverage-100%25-green)
[![Made by Didi](https://img.shields.io/badge/made%20by-Didi-green)](https://www.linkedin.com/in/luis-carlos-zancanela/)

![Repository license](https://img.shields.io/github/license/didifive/peoplehub)

[![IntelliJ IDEA](https://img.shields.io/badge/IntelliJIDEA-000000.svg?logo=intellij-idea&logoColor=white)](https://www.jetbrains.com/idea/)

![technology Java](https://img.shields.io/static/v1?color=red&label=Technology&message=Java&style=for-the-badge&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAC0AAAAtCAYAAAA6GuKaAAAABGdBTUEAAK/INwWK6QAAABl0RVh0U29mdHdhcmUAQWRvYmUgSW1hZ2VSZWFkeXHJZTwAAAZRSURBVHjaYvz//z8DzQEjoziQnALECQz//3+l1DiAAGJioA/4BsSuQBxADcMAAohejgbZwwPEHtQwDCCA6OVoJSBmBmJ1YFJhptQwgACil6PdoTQb1PEUAYAAYqJTJsyA8l4A8W9KjQQIICYaO5gfSM4GYnmoyGEGKhRXAAEENoMmmIHBBIhPAfF/KH4CxDLUMBsggGjl4BQgfovk4L9AHEAt8wECiNqOFQbiJUiOheEsJDXsUE9lkWsPQABR08EaQHwCzbHfgTgVzbGXoNiBXLsAAohaDpYH4ptoDn4BxO5QeQcgPgYV3w3EopTYBxBA1HAwDxAfRXPwBSDWgcp3A/EfqPhUIGal1E6AAKKGoyvRHLwPiCXAjmNgWIck3gvEjNSIWYAAotTBgkD8GMlhx4GYHyo3GUl8EzUzPEAAUepoL7RiDZaGTYH4N1T8KxBrUtPRAAFEaY3Ij8T+C8TPoWxLIGaBsr8D8UNqVrQAAUSpo28B8T8oG+RISSj7C5IaISCOpqajAQKIUkdfhmJwSwOINaDsC9CQh4nXA9sh0tRyNEAAMVHYcPkFbRDBgAWUvgjE55DEQQ4uppajAQKIGkUeFxAfgWa650AsABUPRisKXwOxCDUyIkAAUatG1AbiV1DHJSCJz0YrXfSpYR9AAFGz7eEExJ+B+Ay8EmFg4ATic1BHv6G0+oZhgABiomI62wckI4FYFYi9oGKg4u4BVMVsIP81js6CFBA7AbEbMRkWIIAYie5IMDJyA0ltINaE9kRkof29M0DHTENSZwUk2YFi+6F67kO7WbZAsY9YumK9QBwC1gMBr4C4DKh2IS6nAAQQMdHOCMSJQHwDqZYDZbhZQBwJxIp49PoA8QFwOxtVnAmIxaE4GojLkDLzf6g9prjMBQggwiHNyAgK0UdoAy8TgHgvEL+BdlRBhvyB0szQioYNPGQA4X+DVjxSQKwIja0PQBwFdMVLqD0gPe1AXAK1Zy5QLgWbkwACiJiQ5gbi7Vh6IyD8D4h/AfEPIP4GxaCG/09oc/QPFj0vgbgLiKWw2GUILWVA6p7hchNAALEQkcG+AkMhCMiKBeJAIDYFYk5oCILSISse3X+gsXEJiI8A8SEgPgk08wsO9d5IFd4lXIYCBBAjyT16yLCAG9QDx4H4JzQpMEId+RPa9ngLxM/ASYuYQUdGRhcguQbaCAMlORegvkPYlAIEEDnlcQUQP4U29nkIqGUGYjkg9gTiGiBeBMS+WNQ5A/EHaLIAJbEYfOYCBBBpIc3IyAEkbwKxHDQ0QJnoKRC/A+If0EYSKMlxAbEgNPOJADEHmkmgDFcNcQFjBnQYGJTcdgJxK1D8MD5nAAQQOSFtB8R7oBnuP5H4J7SHsxvaowEVlSxA3A6t/lfAOxBEYIAAYiR7lIqRURLaqoNVNkLQUP4JLc5ewtM0A8M9IH4MtPEbkn5QhjOCpvlXpFgNEEDkOZqRkRfasNcC4haiLGVkZIM2ZSkGAAHEQqa+OqRKwADooCPgkISUGN+haZsXmvZVoNU/qH0xCUjPQglxMgBAAJHbomslIT2j4/NALE1JKw8ggFiIKJNVoRhUnYMaQAeBuANaNUdAQ5KDQNj8hjaaTgDxHiD+iMM+NWga/4HPMIAAwp6mGRlB0VoLxJ7QogvUm94P7UbtBRp6F6qOHdqWUIW2K0CZkQ+a7EAWv4dmxtvgTvD//5/xBBDInh3gCub//258jgYIIFzRfxApOt+CRzhBLTLqdRi4wK1DYM0KxEVAvAzaHQPZFw5vCeLQDxBAuEK6AdyDRgVfoZntJrQYew6tVD5Dk8pPaDX+D6mVxwHNkPzQSkYKmjlhMYM8/3IAiNvAbRNQBQNspAFduA1bQAMEEL7QCIVWIp8oyHT4KptbQLwAiGOBWB3cFWNgSALiu9BmAs5ZA4AAIqY9rQCtQJSQ2sRcSCHJAm2ZMUEbTb+gGe83NAZAme4TNH2DKhxQl+sJtFElADXbEYj9oLFxDVwH/P9/AZeTAAKI1LaHMJCcBI1yVqiD/0Az3S9o+fwbqUMAw0xQD4IyqRh0HEQKbdwFNE6yDFqOf8bnDIAAItXRrFDLlKFFnSLUAaC+nig07fJC29us0DT7D+qRr9AQfwvND0+h/UdQSXQVXNX///+XGGcABBgAIku3g4p9ZV8AAAAASUVORK5CYII=)
![technology Spring](https://img.shields.io/static/v1?color=6DB33F&label=Technology&message=Spring&style=for-the-badge&logo=spring)

# API People Hub 👩👨

API RESTful construída com Java 21 e Spring Boot 3.2.4.

---

## 🎯 Objetivo 
A API desenvolvida deve permitir:
- Criar, editar e consultar uma ou mais pessoas;
- Criar, editar e consultar um ou mais endereços de uma pessoa; e
- Poder indicar qual endereço será considerado o principal de uma pessoa.

Uma pessoa deve possuir os seguintes dados:
- Nome completo
- Data de nascimento
- Endereços:
  - Logradouro
  - CEP
  - Número
  - Cidade
  - Estado

Outros pontos considerados: 
- Atingir o máximo de cobertura do código.
- Construir API no formato REST.
- Boas práticas de programação

## 🔧 Principais tecnologias utilizadas
- **Java 21**: Versão LTS mais recente do Java para tirar vantagem das últimas inovações que essa linguagem robusta e amplamente utilizada oferece;
- **Spring Boot 3**: Versão do Spring Boot, que maximiza a produtividade do desenvolvedor por meio de sua poderosa premissa de autoconfiguração;
- **Spring Data JPA**: Ferramenta pode simplificar a camada de acesso aos dados, facilitando a integração com bancos de dados SQL;
- **FlyWay**: Ferramenta que permite versionar e migrar seu banco de dados com scripts SQL simples ou Java;

---

## 🔣 Diagrama de Classes 

Diagrama de classes na sintaxe [Mermaid].

```mermaid
classDiagram
    class BasicEntity {
        <<Abstract>>
        - String id
    }

    class Person {
        - String name
        - LocalDate birthDate
        - List~Address~ adresses
        + Optional~Address~ getMainAddress()
    }

    class Address {
        - String publicPlace
        - Integer number
        - String city
        - String zipCode
        - State state
        - AddressType addressType
        - boolean main
    }

    class AddressType {
        <<Enumeration>>
        COMMERCIAL("Comercial")
        HOME("Residencial")
        MAILING("Correspondência")
    }

    BasicEntity <-- Person
    BasicEntity <-- Address
    Person "1" *-- "0..n" Address
    Address --> AddressType
``` 

---

## 📷 Prints do Projeto

Banner do Spring personalizado  
![Spring Banner](docs/banner.PNG?raw=true "Spring Banner")  

**Coverage com 100%** - relatório gerado pelo plugin do Jacoco  
![Jacoco Report](docs/jacoco100.PNG?raw=true "Jacoco Report")  

Swagger documentando endpoints  
![Swagger](docs/swagger.PNG?raw=true "Swagger")  

---

## ✔️ Testes

Os testes foram feitos utilizando JUnit 5, Mockito e MockMVC com Hamcrest.

- Para executar os testes pode executar sua IDE ou
- Utilizando o terminal (PowerShell, Bash ou similiar), basta executar na pasta do projeto o comando abaixo:

    ```shell
    ./mvnw clean test
    ```

_Após o teste finalizado com sucesso, é possível verificar relatório de coverage em: target/site/jacoco/index.html_

## ⚙ Executando o projeto localmente

Antes de mais nada, é preciso Possuir no mínimo JDK 21 LTS instalado na máquina em que irá executar.
A execução do projeto pode ser feita utilizando recurso de sua IDE ou com comando (demonstrado no próximo item).

### ⚡ Executando com perfil DEV

```bash
./mvnw spring-boot:run -Dspring_profiles_active=dev
```

### 📚 OpenApi / Swagger

Swagger OpenAPI v3.1, link para acessar a página do Swagger executando o projeto localmente: http://localhost:8080/swagger-ui/index.html

### 👪 Populate

Para popular dados automaticamente foi criada a migration do FlyWay `V3__Populate.sql`.

---

## ☁️ Deploy na Nuvem

Esse projeto foi implantado no [Railway] que faz integração com o repositório do GitHub.
URL do Swagger na nuvem: https://peoplehub-production.up.railway.app/swagger-ui/index.html

---

📋 Qualquer dúvida, sugestão ou crítica é só entrar em contato ou abrir uma Issue (https://github.com/didifive).  
💚 Feito com muita dedicação. #EnjoyThis

[Mermaid]: https://mermaid.js.org/
[Railway]: https://railway.app/
 readmeEtag: '"1e11af5686bb0b18a852e63610847b9192e7da84"' readmeLastModified: Thu, 11 Apr 2024 12:10:52 GMT repositoryId: 784670275 description: 'PeopleHub: Registration of people and addresses API' created: '2024-04-10T10:11:53Z' updated: '2024-04-15T16:34:35Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: didifive logo: https://avatars.githubusercontent.com/u/78564001?v=4 license: Apache-2.0 repoEtag: '"77485af9f51f518b5f13f54cf32c33233676cdf94499ae69f6d8fd9c95208390"' repoLastModified: Mon, 15 Apr 2024 16:34:35 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/aminya/auth0-openapi v3: true id: 60f931a08da3cd657c9bc563625ec5f0 repositoryMetadata: base64Readme: >- IyBBdXRoMCBBdXRoZW50aWNhdGlvbiBBUEkKClRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gZm9yIHRoZSBBdXRoMCBBdXRoZW50aWNhdGlvbiBBUEkuCgojIyBPcGVuQVBJIFNwZWNpZmljYXRpb24KCkluIHRoaXMgcmVwb3NpdG9yeSwgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiB0aGF0IGRlc2NyaWJlcyB0aGUgQXV0aDAgQXV0aGVudGljYXRpb24gQVBJIGlzIHByb3ZpZGVkLgoKVGhlIEF1dGgwIEF1dGhlbnRpY2F0aW9uIEFQSSBlbmFibGVzIHlvdSB0byBtYW5hZ2UgYWxsIGFzcGVjdHMgb2YgdXNlciBpZGVudGl0eSB3aGVuIHlvdSB1c2UgQXV0aDAuIEl0IG9mZmVycyBlbmRwb2ludHMgc28geW91ciB1c2VycyBjYW4gbG9nIGluLCBzaWduIHVwLCBsb2cgb3V0LCBhY2Nlc3MgQVBJcywgYW5kIG1vcmUuIFRoZSBBUEkgc3VwcG9ydHMgdmFyaW91cyBpZGVudGl0eSBwcm90b2NvbHMsIGxpa2UgT3BlbklEIENvbm5lY3QsIE9BdXRoIDIuMCwgYW5kIFNBTUwuCgpUaGlzIHNwZWNpZmljYXRpb24gY2FuIGJlIHVzZWQgZm9yIHZhcmlvdXMgcHVycG9zZXMsIHN1Y2ggYXM6CgotIEdlbmVyYXRlIGNsaWVudCBsaWJyYXJpZXMgaW4gbWFueSBsYW5ndWFnZXMgKHZpYSBbT3BlbkFQSSBHZW5lcmF0b3JdKGh0dHBzOi8vb3BlbmFwaS1nZW5lcmF0b3IudGVjaC8pKS4gVGhpcyBpcyBwYXJ0aWN1bGFybHkgdXNlZnVsIGZvciB0aGUgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2VzIHRoYXQgYXJlIG5vdCBvZmZpY2lhbGx5IHN1cHBvcnRlZCBieSBBdXRoMC4KCi0gR2VuZXJhdGUgc2VydmVyIHN0dWJzIGluIG1hbnkgbGFuZ3VhZ2VzICh2aWEgW09wZW5BUEkgR2VuZXJhdG9yXShodHRwczovL29wZW5hcGktZ2VuZXJhdG9yLnRlY2gvKSkKCi0gR2VuZXJhdGUgaW50ZXJhY3RpdmUgZG9jdW1lbnRhdGlvbiAodmlhIFtTd2FnZ2VyIFVJXShodHRwczovL3N3YWdnZXIuaW8vdG9vbHMvc3dhZ2dlci11aS8pLCBbUmFwaURvY10oaHR0cHM6Ly9yYXBpZG9jd2ViLmNvbS8pLCBbUmVkb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKSkKCiMjIENvbnRyaWJ1dGluZwoKVGhpcyBzcGVjaWZpY2F0aW9uIGlzIHdyaXR0ZW4gYmFzZWQgb24gdGhlIFtBdXRoMCBBUEldKGh0dHBzOi8vYXV0aDAuY29tL2RvY3MvYXBpKSBkb2N1bWVudGF0aW9uLiBQbGVhc2UgY29udHJpYnV0ZSB0byBpbXByb3ZpbmcgdGhpcyBzcGVjaWZpY2F0aW9uIGJ5IG9wZW5pbmcgcHVsbCByZXF1ZXN0cy4KCllvdSBjYW4gc3BvbnNvciBteSB3b3JrIGhlcmU6CgpodHRwczovL2dpdGh1Yi5jb20vc3BvbnNvcnMvYW1pbnlhCg== readmeEtag: '"d7cf69511301b21e03ee3782bc3f4c3815b22eea"' readmeLastModified: Fri, 26 Apr 2024 04:54:31 GMT repositoryId: 789388710 description: Auth0 OpenAPI specification created: '2024-04-20T12:07:57Z' updated: '2025-05-05T03:19:10Z' language: null archived: false stars: 1 watchers: 1 forks: 0 owner: aminya logo: https://avatars.githubusercontent.com/u/16418197?v=4 license: Apache-2.0 repoEtag: '"7e16cc8923e58d90f8d005d6225637abc16dde9712c5bb04163a746acc6168e6"' repoLastModified: Mon, 05 May 2025 03:19:10 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mwczapski/swagger_editor_3_docker_container v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyX0VkaXRvcl8zX0RvY2tlcl9Db250YWluZXIKClRoZSBpbnRlbnQgb2YgdGhpcyByZXBvc2l0b3J5IGlzIHRvIGhvc3QgcmVzb3VyY2VzIHRoYXQgcHJvdmlkZSB0aGUgbWVhbnMgdG8gY3JlYXRlIGEgc2VsZi1jb250YWluZWQgRG9ja2VyIGNvbnRhaW5lciBmb3IgQVBJLUZpcnN0IGRldmVsb3BtZW50IHVzaW5nIHRoZSBsYXRlc3QgU3dhZ2dlciBFZGl0b3IgKDMueCkgYW5kIE9wZW5BUEkgKDIgb3IgMykuCgpUaGUgZm9sbG93aW5nIHJlc291cmNlcyBhcmUgYXZhaWxhYmxlIGluIHRoaXMgcmVwb3NpdG9yeToKCjEuIFtTdGVwcyB0byBjcmVhdGUgdGhlIHJlYWR5LXRvLWdvIFN3YWdnZXIgRWRpdG9yIDMuMCBEb2NrZXIgSW1hZ2UgYW5kIGEgcmVhZHktdG8tZ28gU3dhZ2dlciBFZGl0b3IgMy4wIERvY2tlciBDb250YWluZXJdKFN3YWdnZXJfRWRpdG9yX0RvY2tlcl9Db250YWluZXJfZm9yX0FQSS1GaXJzdF9EZXZlbG9wbWVudC5tZCkKMi4gW1N0YXJ0IERvY2tlciBDb250YWluZXIgYW5kIHdvcmsgd2l0aCB0aGUgU3dhZ2dlciBFZGl0b3Igc2VydmVkIGZyb20gdGhlIGNvbnRhaW5lcl0oSG93X1RvX1VzZV9Td2FnZ2VyX0VkaXRvcl9Eb2NrZXJfSW1hZ2UubWQpCjMuIFtTZXQgdXAgU3dhZ2dlciBFZGl0b3Igb24gV2luZG93cyAxMCB3aXRoIG5vIERvY2tlciBvciBXaW5kb3dzIFN1YnN5c3RlbSBmb3IgTGludXggZGVwZW5kZW5jaWVzXShzd2FnZ2VyX2VkaXRvcl9sb2NhbC5tZCkKClRoZSBEb2NrZXIgSW1hZ2UgY3JlYXRlZCBpbiAoMSkgYW5kIHVzZWQgaW4gKDIpIGlzIGF2YWlsYWJsZSBhdCBbbXdjemFwc2tpL3N3YWdnZXItZWRpdG9yOjEuMC4wXShodHRwczovL2h1Yi5kb2NrZXIuY29tL3IvbXdjemFwc2tpL3N3YWdnZXJfZWRpdG9yKSBvbiBEb2NrZXIgSHViLgo= readmeEtag: '"49ef209172c2bf86f70df7d8f3082e2102499edd"' readmeLastModified: Sat, 01 Aug 2020 07:59:21 GMT repositoryId: 277014482 description: >- The intent of this repository is to host resources that provide the means to create a self-contained Docker container for API-First development using latest Swagger Editor (3.x) and OpenAPI (2 or 3). created: '2020-07-04T01:06:47Z' updated: '2024-06-04T23:42:23Z' language: Dockerfile archived: false stars: 1 watchers: 1 forks: 0 owner: mwczapski logo: https://avatars.githubusercontent.com/u/57377978?v=4 license: MIT repoEtag: '"1f7337cf54b3a12640bf9f1db6eb4dee3d8e23e693b4a5a789b952c86ff4c1b8"' repoLastModified: Tue, 04 Jun 2024 23:42:23 GMT foundInMaster: true category: - Code Generators - Server Implementations id: c9649f9445377cd8e5eafe406ec84cc5 - source: openapi3 tags repository: https://github.com/codeasashu/oas.nvim v3: true id: 50e06ec9dcbadc2a1f5737d4f78619a9 repositoryMetadata: base64Readme: >- IyMgT3BlbiBBUEkgZm9yIG5lb3ZpbQoKVGhpcyBwbHVnaW4gcHJvdmlkZXMgc3VwcG9ydCBmb3Igb3BlbmFwaSwgc3VjaCBhczoKCi0gW3hdIFByZXZpZXdzICh2aWEgUmVkb2MpIAotIFsgXSBUcmVlIG5hdmlnYXRvciBpbiBuZW92aW0KCiMjIyBJbnN0YWxsYXRpb24KCjEuIEluc3RhbGwgcmVkb2NseSBjbGk6CiAgICBgYGBzaCAKICAgIG5wbSBpbnN0YWxsIC1nIEByZWRvY2x5L2NsaQogICAgYGBgCgoyLiBBZGQgdGhlIGZvbGxvd2luZyBpbiBsYXp5LnZpbToKICAgIGBgYGx1YQogICAgeyAiY29kZWFzYXNodS9vYXMubnZpbSIgfQogICAgYGBgCgoKIyMjIFByZXZpZXcKCk9wZW4gcmV2aWV3IGJ5IG9wZW5pbmcgYW55IG9wZW5hcGkgZmlsZSBhbmQgY2FsbGluZyBjb21tYW5kIGA6T0FTUHJldmlld2AuIApUbyBzdG9wIHRoZSBwcmV2aWV3LCBzaW1wbHkgY2FsbCBjb21tYW5kOiBgOk9BU1ByZXZpZXdTdG9wYAo= readmeEtag: '"6bd899896c24b8c466ce33424439e40c76a89c97"' readmeLastModified: Fri, 07 Jun 2024 19:42:23 GMT repositoryId: 812023049 description: Openapi support in neovim created: '2024-06-07T19:40:34Z' updated: '2025-10-03T03:39:06Z' language: Lua archived: false stars: 2 watchers: 1 forks: 0 owner: codeasashu logo: https://avatars.githubusercontent.com/u/1492350?v=4 repoEtag: '"598d5d943acb1229d1bffc26216e8d2b829f42026c1780450fddcdea1f40fd40"' repoLastModified: Fri, 03 Oct 2025 03:39:06 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/ellizio/kiota.autogen v3: true id: 55ce9e3e33d8b88a9f0d10a826199a80 repositoryMetadata: base64Readme: >- IyA8cCBhbGlnbj0iY2VudGVyIj4gS2lvdGEuQXV0b2dlbiA8L3A+DQoNCjxwIGFsaWduPSJjZW50ZXIiPiBBIHNldCBvZiBsaWJyYXJpZXMgZm9yIGF1dG8tZ2VuZXJhdGluZyBBUEkgY2xpZW50cyB1c2luZyA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vbWljcm9zb2Z0L2tpb3RhIj5LaW90YTwvYT4gPC9wPg0KDQotLS0NCg0KIyMgTGlicmFyaWVzIGxpc3QNCg0KLSBbIVtdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnVnZXQvZHQvS2lvdGEuQXV0b2dlbi5Td2FnZ2VyP2xvZ289bnVnZXQmY29sb3I9MDA3ZWRmKV0oaHR0cHM6Ly93d3cubnVnZXQub3JnL3BhY2thZ2VzL0tpb3RhLkF1dG9nZW4uU3dhZ2dlci8pIFtLaW90YS5BdXRvZ2VuLlN3YWdnZXJdKHNyYy9LaW90YS5BdXRvZ2VuLlN3YWdnZXIpIGZvciBnZW5lcmF0aW5nIEFQSSBjbGllbnQgYmFzZWQgb24gW1N3YXNoYnVja2xlLkFzcE5ldENvcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9kb21haW5kcml2ZW5kZXYvU3dhc2hidWNrbGUuQXNwTmV0Q29yZSkNCg0KIyMgQmFzaWMgVXNhZ2UNCg0KMS4gQWRkIGEgYENsYXNzIExpYnJhcnlgIHByb2plY3QgdG8gdGhlIHNvbHV0aW9uIHRoYXQgY29udGFpbnMgV2ViQXBpIHByb2plY3QNCjIuIEFkZCBXZWJBcGkgcHJvamVjdCByZWZlcmVuY2Ugd2l0aCBgRXhjbHVkZUFzc2V0cz0iQWxsImANCmBgYHhtbA0KPEl0ZW1Hcm91cD4NCiAgICA8UHJvamVjdFJlZmVyZW5jZSBJbmNsdWRlPSIuLlxFeGFtcGxlU2VydmljZS5BcGlcRXhhbXBsZVNlcnZpY2UuQXBpLmNzcHJvaiIgRXhjbHVkZUFzc2V0cz0iQWxsIiAvPg0KPC9JdGVtR3JvdXA+DQpgYGANCjMuIEluc3RhbGwgYEtpb3RhLkF1dG9nZW4uU3dhZ2dlcmAgd2l0aCBgUHJpdmF0ZUFzc2V0cz0iQWxsImANCmBgYHhtbA0KPEl0ZW1Hcm91cD4NCiAgICA8UGFja2FnZVJlZmVyZW5jZSBJbmNsdWRlPSJLaW90YS5BdXRvZ2VuLlN3YWdnZXIiIFZlcnNpb249IjEuMTguMCIgUHJpdmF0ZUFzc2V0cz0iQWxsIiAvPg0KPC9JdGVtR3JvdXA+DQpgYGANCjQuIEluc3RhbGwgZm9sbG93aW5nIGBLaW90YWAgcGFja2FnZXMNCmBgYHhtbA0KPEl0ZW1Hcm91cD4NCiAgICA8UGFja2FnZVJlZmVyZW5jZSBJbmNsdWRlPSJNaWNyb3NvZnQuS2lvdGEuQWJzdHJhY3Rpb25zIiBWZXJzaW9uPSIxLjEyLjQiIC8+DQogICAgPFBhY2thZ2VSZWZlcmVuY2UgSW5jbHVkZT0iTWljcm9zb2Z0Lktpb3RhLkh0dHAuSHR0cENsaWVudExpYnJhcnkiIFZlcnNpb249IjEuMTIuNCIgLz4NCiAgICA8UGFja2FnZVJlZmVyZW5jZSBJbmNsdWRlPSJNaWNyb3NvZnQuS2lvdGEuU2VyaWFsaXphdGlvbi5Gb3JtIiBWZXJzaW9uPSIxLjEyLjQiIC8+DQogICAgPFBhY2thZ2VSZWZlcmVuY2UgSW5jbHVkZT0iTWljcm9zb2Z0Lktpb3RhLlNlcmlhbGl6YXRpb24uSnNvbiIgVmVyc2lvbj0iMS4xMi40IiAvPg0KICAgIDxQYWNrYWdlUmVmZXJlbmNlIEluY2x1ZGU9Ik1pY3Jvc29mdC5LaW90YS5TZXJpYWxpemF0aW9uLk11bHRpcGFydCIgVmVyc2lvbj0iMS4xMi40IiAvPg0KICAgIDxQYWNrYWdlUmVmZXJlbmNlIEluY2x1ZGU9Ik1pY3Jvc29mdC5LaW90YS5TZXJpYWxpemF0aW9uLlRleHQiIFZlcnNpb249IjEuMTIuNCIgLz4NCjwvSXRlbUdyb3VwPg0KYGBgDQo1LiBDcmVhdGUgYSBgZ2Vuc2V0dGluZ3MuanNvbmAgZmlsZSB3aXRoIHRoZSBmb2xsb3dpbmcgc3RydWN0dXJlDQpgYGBqc29uYw0KWw0KICB7DQogICAgIm5hbWUiOiAiV2VhdGhlckNsaWVudCIsIC8vIEFQSSBjbGllbnQgbmFtZSB0byBiZSBnZW5lcmF0ZWQNCiAgICAibmFtZXNwYWNlIjogIldlYXRoZXIuQ2xpZW50IiwgLy8gQVBJIGNsaWVudCBuYW1lc3BhY2UNCiAgICAidmVyc2lvbiI6ICJ2MSIgLy8gV2ViQXBpIFN3YWdnZXIgZG9jdW1lbnQgdmVyc2lvbg0KICB9LA0KICB7DQogICAgIm5hbWUiOiAiV2VhdGhlckNsaWVudE5ldyIsDQogICAgIm5hbWVzcGFjZSI6ICJXZWF0aGVyLkNsaWVudC5OZXciLA0KICAgICJ2ZXJzaW9uIjogInYyIg0KICB9DQpdDQpgYGANCjYuIFNldCB1cCBgQ2xhc3MgbGlicmFyeWAgcHJvamVjdCB0byBidWlsZCBhcyBhIG51Z2V0IHBhY2thZ2UNCmBgYHhtbA0KPFByb3BlcnR5R3JvdXA+DQogICAgLi4uIG90aGVyIHByb3BlcnRpZXMNCg0KICAgIDxJc1BhY2thYmxlPnRydWU8L0lzUGFja2FibGU+DQogICAgPFBhY2thZ2VJZD5FeGFtcGxlU2VydmljZS5DbGllbnQ8L1BhY2thZ2VJZD4NCjwvUHJvcGVydHlHcm91cD4NCmBgYA0KNy4gUGFjayBgQ2xhc3MgbGlicmFyeWAgcHJvamVjdA0KOC4gRW5qb3kgdXNpbmcgQVBJIGNsaWVudA0KYGBgY3NoYXJwDQp1c2luZyBNaWNyb3NvZnQuS2lvdGEuQWJzdHJhY3Rpb25zLkF1dGhlbnRpY2F0aW9uOw0KdXNpbmcgTWljcm9zb2Z0Lktpb3RhLkh0dHAuSHR0cENsaWVudExpYnJhcnk7DQp1c2luZyBXZWF0aGVyLkNsaWVudDsNCg0KdXNpbmcgdmFyIGh0dHBDbGllbnQgPSBuZXcgSHR0cENsaWVudCgpOw0KaHR0cENsaWVudC5CYXNlQWRkcmVzcyA9IG5ldyBVcmkoImh0dHA6Ly95b3VyX3NlcnZpY2VfdXJsIik7DQoNCnZhciBwcm92aWRlciA9IG5ldyBBbm9ueW1vdXNBdXRoZW50aWNhdGlvblByb3ZpZGVyKCk7DQp1c2luZyB2YXIgYWRhcHRlciA9IG5ldyBIdHRwQ2xpZW50UmVxdWVzdEFkYXB0ZXIocHJvdmlkZXIsIGh0dHBDbGllbnQ6IGh0dHBDbGllbnQpOw0KdmFyIGNsaWVudCA9IG5ldyBXZWF0aGVyQ2xpZW50KGFkYXB0ZXIpOw0KDQp2YXIgZm9yZWNhc3RzID0gYXdhaXQgY2xpZW50LldlYXRoZXJmb3JlY2FzdC5HZXRBc3luYygpOw0KYGBgDQoNClNlZSBbZnVsbCBleGFtcGxlXShodHRwczovL2dpdGh1Yi5jb20vZWxsaXppby9LaW90YS5BdXRvZ2VuLUV4YW1wbGVzL3RyZWUvbWFzdGVyLzEuJTIwU3dhZ2dlcikNCg0KIyMgQWR2YW5jZWQgVXNhZ2UNCg0KWW91IGNhbiBwcm92aWRlIHlvdXIgb3duIGltcGxlbWVudGF0aW9ucyBvZiBgTWljcm9zb2Z0Lktpb3RhLkFic3RyYWN0aW9uc2AgZnJvbSB0aGUgYENsYXNzIGxpYnJhcnlgIHByb2plY3QgaWYgeW91IG5lZWRcDQpVc2FnZToNCmBgYGNzaGFycA0KdXNpbmcgTWljcm9zb2Z0Lktpb3RhLkFic3RyYWN0aW9ucy5BdXRoZW50aWNhdGlvbjsNCnVzaW5nIE1pY3Jvc29mdC5LaW90YS5IdHRwLkh0dHBDbGllbnRMaWJyYXJ5Ow0KdXNpbmcgV2VhdGhlci5DbGllbnQ7DQoNCnVzaW5nIHZhciBodHRwQ2xpZW50ID0gbmV3IEh0dHBDbGllbnQoKTsNCmh0dHBDbGllbnQuQmFzZUFkZHJlc3MgPSBuZXcgVXJpKCJodHRwOi8veW91cl9zZXJ2aWNlX3VybCIpOw0KDQp2YXIgcHJvdmlkZXIgPSBuZXcgQW5vbnltb3VzQXV0aGVudGljYXRpb25Qcm92aWRlcigpOw0KdXNpbmcgdmFyIGFkYXB0ZXIgPSBuZXcgSHR0cENsaWVudFJlcXVlc3RBZGFwdGVyKHByb3ZpZGVyLCBuZXcgV2VhdGhlclBhcnNlTm9kZUZhY3RvcnkoKSwgbmV3IFdlYXRoZXJTZXJpYWxpemF0aW9uV3JpdGVyRmFjdG9yeSgpLCBodHRwQ2xpZW50OiBodHRwQ2xpZW50KTsNCnZhciBjbGllbnQgPSBuZXcgV2VhdGhlckNsaWVudChhZGFwdGVyKTsNCg0KdmFyIGZvcmVjYXN0cyA9IGF3YWl0IGNsaWVudC5XZWF0aGVyZm9yZWNhc3QuR2V0QXN5bmMoKTsNCmBgYA0KDQpTZWUgW2Z1bGwgZXhhbXBsZV0oaHR0cHM6Ly9naXRodWIuY29tL2VsbGl6aW8vS2lvdGEuQXV0b2dlbi1FeGFtcGxlcy90cmVlL21hc3Rlci8yLiUyMEFkdmFuY2VkKQ0KDQojIyBNb3JlIEV4YW1wbGVzDQoNClNlZSBtb3JlIGV4YW1wbGVzIFtoZXJlXShodHRwczovL2dpdGh1Yi5jb20vZWxsaXppby9LaW90YS5BdXRvZ2VuLUV4YW1wbGVzKQ0KDQojIyBBZGRpdGlvbmFsIFJlZmVyZW5jZXMNCg0KLSBbQ2hhbmdlbG9nXShDSEFOR0VMT0cubWQpDQotIFtLaW90YV0oaHR0cHM6Ly9naXRodWIuY29tL21pY3Jvc29mdC9raW90YSkNCi0gW1N3YXNoYnVja2xlLkFzcE5ldENvcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9kb21haW5kcml2ZW5kZXYvU3dhc2hidWNrbGUuQXNwTmV0Q29yZSkNCg== readmeEtag: '"ce4f329d27d82655cc4c8ddae04da7d956f20067"' readmeLastModified: Tue, 24 Jun 2025 15:27:02 GMT repositoryId: 814793618 description: A set of libraries for auto-generating API clients using Kiota created: '2024-06-13T18:06:04Z' updated: '2025-06-24T15:27:06Z' language: C# archived: false stars: 1 watchers: 1 forks: 1 owner: ellizio logo: https://avatars.githubusercontent.com/u/48138666?v=4 license: MIT repoEtag: '"08c08c45abd3537616229b8d4ffb295bef1b7d2b1c1cb74c01105a8a7f3fd1c0"' repoLastModified: Tue, 24 Jun 2025 15:27:06 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/zero-to-prod/data-model-openapi30 v3: true id: 78fd9e63b8ce0aabb3f592921ea83c75 repositoryMetadata: base64Readme: >- # Zerotoprod\DataModelOpenapi30

![](./art/logo.png)

[![Repo](https://img.shields.io/badge/github-gray?logo=github)](https://github.com/zero-to-prod/data-model-openapi30)
[![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/zero-to-prod/data-model-openapi30/test.yml?label=test)](https://github.com/zero-to-prod/data-model-openapi30/actions)
[![GitHub Actions Workflow Status](https://img.shields.io/github/actions/workflow/status/zero-to-prod/data-model-openapi30/backwards_compatibility.yml?label=backwards_compatibility)](https://github.com/zero-to-prod/data-model-openapi30/actions)
[![Packagist Downloads](https://img.shields.io/packagist/dt/zero-to-prod/data-model-openapi30?color=blue)](https://packagist.org/packages/zero-to-prod/data-model-openapi30/stats)
[![Packagist Version](https://img.shields.io/packagist/v/zero-to-prod/data-model-openapi30?color=f28d1a)](https://packagist.org/packages/zero-to-prod/data-model-openapi30)
[![License](https://img.shields.io/packagist/l/zero-to-prod/data-model-openapi30?color=red)](https://github.com/zero-to-prod/data-model-openapi30/blob/main/LICENSE.md)
[![wakatime](https://wakatime.com/badge/github/zero-to-prod/data-model-openapi30.svg)](https://wakatime.com/badge/github/zero-to-prod/data-model-openapi30)
[![Hits-of-Code](https://hitsofcode.com/github/zero-to-prod/data-model-openapi30?branch=main)](https://hitsofcode.com/github/zero-to-prod/data-model-openapi30/view?branch=main)

## Contents

- [Introduction](#introduction)
- [Requirements](#requirements)
- [Installation](#installation)
- [Documentation Publishing](#documentation-publishing)
    - [Automatic Documentation Publishing](#automatic-documentation-publishing)
- [Usage](#usage)
- [Publishing DataModels](#publishing-datamodels)
- [Local Development](./LOCAL_DEVELOPMENT.md)
- [Contributing](#contributing)

## Introduction

DataModels for OpenAPI 3.0.* [specification](https://spec.openapis.org/oas/v3.0.4.html).

## Requirements

- PHP 8.1 or higher.

## Installation

Install `Zerotoprod\DataModelOpenapi30` via [Composer](https://getcomposer.org/):

```shell
composer require zero-to-prod/data-model-openapi30
```

This will add the package to your project's dependencies and create an autoloader entry for it.

## Documentation Publishing

You can publish this README to your local documentation directory.

This can be useful for providing documentation for AI agents.

This can be done using the included script:

```bash
# Publish to default location (./docs/zero-to-prod/data-model-openapi30)
vendor/bin/data-model-openapi30

# Publish to custom directory
vendor/bin/data-model-openapi30 /path/to/your/docs
```

### Automatic Documentation Publishing

You can automatically publish documentation by adding the following to your `composer.json`:

```json
{
    "scripts": {
        "post-install-cmd": [
            "data-model-openapi30"
        ],
        "post-update-cmd": [
            "data-model-openapi30"
        ]
    }
}
```

## Usage

```php
use Zerotoprod\DataModelOpenapi30\OpenApi;

$OpenApi = OpenApi::from(json_decode($json, true));

$OpenApi->components->schemas['pet']->description;
```

## Publishing DataModels

You can directly import these files into your project like this:

```shell
./vendor/bin/data-model-openapi30 app/DataModelOpenapi30
```

## Acceptance Tests

| Test                                                                                                                              | Status             |
|-----------------------------------------------------------------------------------------------------------------------------------|--------------------|
| [4.7.2.2 Info Object Example](https://spec.openapis.org/oas/v3.0.4.html#info-object-example)                                      | :white_check_mark: |
| [4.7.3.2 Contact Object Example](https://spec.openapis.org/oas/v3.0.4.html#contact-object-example)                                | :white_check_mark: |
| [4.7.4.2 License Object Example](https://spec.openapis.org/oas/v3.0.4.html#license-object-example)                                | :white_check_mark: |
| [4.7.5.2 Server Object Example](https://spec.openapis.org/oas/v3.0.4.html#server-object-example)                                  | :white_check_mark: |
| [4.7.7.2 Components Object Example](https://spec.openapis.org/oas/v3.0.4.html#components-object-example)                          | :white_check_mark: |
| [4.7.8.3 Paths Object Example](https://spec.openapis.org/oas/v3.0.4.html#paths-object-example)                                    | :white_check_mark: |
| [4.7.9.2 Path Item Object Example](https://spec.openapis.org/oas/v3.0.4.html#path-item-object-example)                            | :white_check_mark: |
| [4.7.10.2 Operation Object Example](https://spec.openapis.org/oas/v3.0.4.html#operation-object-example)                           | :white_check_mark: |
| [4.7.11.2 External Documentation Object Example](https://spec.openapis.org/oas/v3.0.4.html#external-documentation-object-example) | :white_check_mark: |
| [4.7.12.5 Parameter Object Examples](https://spec.openapis.org/oas/v3.0.4.html#parameter-object-examples)                         | :white_check_mark: |
| [4.7.13.2 Request Body Examples](https://spec.openapis.org/oas/v3.0.4.html#request-body-examples)                                 | :white_check_mark: |
| [4.7.14.2 Media Type Examples](https://spec.openapis.org/oas/v3.0.4.html#media-type-examples-0)                                   | :white_check_mark: |

## Properties

# [4.0 Specification](https://spec.openapis.org/oas/v3.0.4.html#specification)

## [4.7 Schema](https://spec.openapis.org/oas/v3.0.4.html#schema-0)

This section describes the structure of the OpenAPI Description format

### [4.7.1 OpenAPI Object](https://spec.openapis.org/oas/v3.0.4.html#openapi-object)

This is the root object of the OpenAPI Description.

#### [4.7.1.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields)

| Field Name   | Type                                                             | Status             |
|--------------|------------------------------------------------------------------|--------------------|
| openapi      | [`string`](src/OpenApi.php)                                      | :white_check_mark: |
| info         | [`Info Object`](src/Info.php)                                    | :white_check_mark: |
| servers      | [`[Server Object]`](src/Server.php)                              | :white_check_mark: |
| paths        | [`Paths Object`](src/PathItem.php)                               | :white_check_mark: |
| components   | [`Components Object`](src/Components.php)                        | :white_check_mark: |
| security     | [`Security Requirement Object`](src/OpenApi.php)                 | :white_check_mark: |
| tags         | [[`Tag Object`](src/Tag.php)]                                    | :white_check_mark: |
| externalDocs | [`External Documentation Object`](src/ExternalDocumentation.php) | :white_check_mark: |

### [4.7.2 Info Object](https://spec.openapis.org/oas/v3.0.4.html#info-object)

The object provides metadata about the API.

#### [4.7.2.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-0)

| Field Name     | Type                                | Status             |
|----------------|-------------------------------------|--------------------|
| title          | [`string`](src/Info.php)            | :white_check_mark: |
| description    | [`string`](src/Info.php)            | :white_check_mark: |
| termsOfService | [`string`](src/Info.php)            | :white_check_mark: |
| contact        | [`Contact Object`](src/Contact.php) | :white_check_mark: |
| license        | [`License Object`](src/License.php) | :white_check_mark: |
| version        | [`string`](src/Info.php)            | :white_check_mark: |

### [4.7.3 Contact Object](https://spec.openapis.org/oas/v3.0.4.html#contact-object)

Contact information for the exposed API.

#### [4.7.3.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-1)

| Field | Type                        | Status             |
|-------|-----------------------------|--------------------|
| name  | [`string`](src/Contact.php) | :white_check_mark: |
| url   | [`string`](src/Contact.php) | :white_check_mark: |
| email | [`email`](src/Contact.php)  | :white_check_mark: |

### [4.7.4 License Object](https://spec.openapis.org/oas/v3.0.4.html#license-object)

License information for the exposed API.

#### [4.7.4.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-2)

| Field | Type                        | Status             |
|-------|-----------------------------|--------------------|
| name  | [`string`](src/License.php) | :white_check_mark: |
| url   | [`string`](src/License.php) | :white_check_mark: |

### [4.7.5 Server Object](https://spec.openapis.org/oas/v3.0.4.html#server-object)

An object representing a Server.

#### [4.7.5.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-3)

| Field Name  | Type                                                                                | Status             |
|-------------|-------------------------------------------------------------------------------------|--------------------|
| url         | [`string`](src/Server.php)                                                          | :white_check_mark: |
| description | [`string`](src/Server.php)                                                          | :white_check_mark: |
| variables   | Map[[`string`](src/Server.php), [`Server Variable Object`](src/ServerVariable.php)] | :white_check_mark: |

### [4.7.6 Server Object](https://spec.openapis.org/oas/v3.0.4.html#server-variable-object)

An object representing a Server Variable for server URL template substitution.

#### [4.7.6.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-4)

| Field Name  | Type                                 | Status             |
|-------------|--------------------------------------|--------------------|
| enum        | [[`string`]](src/ServerVariable.php) | :white_check_mark: |
| default     | [`string`](src/ServerVariable.php)   | :white_check_mark: |
| description | [`string`](src/ServerVariable.php)   | :white_check_mark: |

### [4.7.7 Components Object](https://spec.openapis.org/oas/v3.0.4.html#components-object)

Holds a set of reusable objects for different aspects of the OAS.

### [4.7.7.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-5)

| Field Name      | Type                                                                                                                               | Status             |
|-----------------|------------------------------------------------------------------------------------------------------------------------------------|--------------------|
| schemas         | Map[[`string`](src/Components.php), [`Schema Object`](src/Schema.php) \| [`Reference Object`](src/Reference.php)]                  | :white_check_mark: |
| responses       | Map[[`string`](src/Components.php), [`Response Object`](src/Response.php) \| [`Reference Object`](src/Reference.php)]              | :white_check_mark: |
| parameters      | Map[[`string`](src/Components.php), [`Parameter Object`](src/Parameter.php) \| [`Reference Object`](src/Reference.php)]            | :white_check_mark: |
| examples        | Map[[`string`](src/Components.php), [`Example Object`](src/Example.php) \| [`Reference Object`](src/Reference.php)]                | :white_check_mark: |
| requestBodies   | Map[[`string`](src/Components.php), [`Request Body Object`](src/RequestBody.php) \| [`Reference Object`](src/Reference.php)]       | :white_check_mark: |
| headers         | Map[[`string`](src/Components.php), [`Header Object`](src/Header.php) \| [`Reference Object`](src/Reference.php)]                  | :white_check_mark: |
| securitySchemes | Map[[`string`](src/Components.php), [`Security Scheme Object`](src/SecurityScheme.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |
| links           | Map[[`string`](src/Components.php), [`Link Object`](src/Link.php) \| [`Reference Object`](src/Reference.php)]                      | :white_check_mark: |
| callbacks       | Map[[`string`](src/Components.php), [`Callback Object`](src/PathItem.php) \| [`Reference Object`](src/Reference.php)]              | :white_check_mark: |

### [4.7.8 Paths Object](https://spec.openapis.org/oas/v3.0.4.html#paths-object)

Holds the relative paths to the individual endpoints and their operations.

#### [4.7.8.1 Patterned Fields](https://spec.openapis.org/oas/v3.0.4.html#patterned-fields)

| Field Name | Type                                 | Status             |
|------------|--------------------------------------|--------------------|
| /{path}    | [Path Item Object](src/PathItem.php) | :white_check_mark: |

### [4.7.9 Path Item Object](https://spec.openapis.org/oas/v3.0.4.html#path-item-object)

Describes the operations available on a single path.

#### [4.7.9.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-6)

| Field Name  | Type                                                                                 | Status             |
|-------------|--------------------------------------------------------------------------------------|--------------------|
| $ref        | [`string`](src/PathItem.php)                                                         | :white_check_mark: |
| summary     | [`string`](src/PathItem.php)                                                         | :white_check_mark: |
| description | [`string`](src/PathItem.php)                                                         | :white_check_mark: |
| get         | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| put         | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| post        | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| delete      | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| options     | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| head        | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| patch       | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| trace       | [`Operation Object`](src/Operation.php)                                              | :white_check_mark: |
| servers     | [`Server Object`](src/Server.php)                                                    | :white_check_mark: |
| parameters  | [[`Parameter Object`](src/Parameter.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |

### [4.7.10 Operation Object](https://spec.openapis.org/oas/v3.0.4.html#operation-object)

Describes a single API operation on a path.

#### [4.7.10.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-7)

| Field Name   | Type                                                                                                                 | Status             |
|--------------|----------------------------------------------------------------------------------------------------------------------|--------------------|
| tags         | [[`string`]](src/Operation.php)                                                                                      | :white_check_mark: |
| summary      | [`string`](src/Operation.php)                                                                                        | :white_check_mark: |
| description  | [`string`](src/Operation.php)                                                                                        | :white_check_mark: |
| externalDocs | [`External Documentation Object`](src/ExternalDocumentation.php)                                                     | :white_check_mark: |
| operationId  | [`string`](src/Operation.php)                                                                                        | :white_check_mark: |
| parameters   | [[`Parameter Object`](src/Parameter.php) \| [`Reference Object`](src/Reference.php)]                                 | :white_check_mark: |
| requestBody  | [`Request Body Object`](src/RequestBody.php) \| [`Reference Object`](src/Reference.php)                              | :white_check_mark: |
| responses    | [`Responses Object`](src/Response.php)                                                                               | :white_check_mark: |
| callbacks    | Map[[`string`](src/Operation.php), [`Callback Object`](src/PathItem.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |
| deprecated   | [`boolean`](src/Operation.php)                                                                                       | :white_check_mark: |
| security     | [`Security Requirement Object`]                                                                                      | :white_check_mark: |
| servers      | [[`Server Object`]](src/Server.php)                                                                                  | :white_check_mark: |

### [4.7.11 External Documentation Object](https://spec.openapis.org/oas/v3.0.4.html#external-documentation-object)

Allows referencing an external resource for extended documentation.

#### [4.7.11.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-8)

| Field Name  | Type                                      | Status             |
|-------------|-------------------------------------------|--------------------|
| description | [`string`](src/ExternalDocumentation.php) | :white_check_mark: |
| url         | [`string`](src/ExternalDocumentation.php) | :white_check_mark: |

### [4.7.12 Parameter Object](https://spec.openapis.org/oas/v3.0.4.html#parameter-object)

Describes a single operation parameter.

#### [4.7.12.2 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-9)

The rules for serialization of the parameter are specified in one of two ways. Parameter Objects _MUST_ include either a `content` field or a `schema`
field, but not both.

##### [4.7.12.2.1 Common Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#common-fixed-fields)

These fields _MAY_ be used with either `content` or `schema`.

| Field Name      | Type                             | Status             |
|-----------------|----------------------------------|--------------------|
| name            | [`string`](src/Parameter.php)    | :white_check_mark: |
| in              | [`string`](src/Parameter.php)    | :white_check_mark: |
| description     | [`string`](src/Parameter.php)    | :white_check_mark: |
| required        | [`boolean`](src/Parameter.php)   | :white_check_mark: |
| deprecated      | [`boolean`](src/Parameter.php)   | :white_check_mark: |
| allowEmptyValue | [[`boolean`]](src/Parameter.php) | :white_check_mark: |

##### [4.7.12.2.2 Fixed Fields for use with `schema`](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-for-use-with-schema)

For simpler scenarios, a `schema` and `style` can describe the structure and syntax of the parameter.

| Field Name    | Type                                                                                                             | Status             |
|---------------|------------------------------------------------------------------------------------------------------------------|--------------------|
| style         | [`string`](src/Parameter.php)                                                                                    | :white_check_mark: |
| explode       | [`boolean`](src/Parameter.php)                                                                                   | :white_check_mark: |
| allowReserved | [`boolean`](src/Parameter.php)                                                                                   | :white_check_mark: |
| schema        | [`Schema Object`](src/Schema.php) \| [`Reference Object`](src/Reference.php)                                     | :white_check_mark: |
| example       | [Any](src/Schema.php)                                                                                            | :white_check_mark: |
| examples      | Map[ [`string`](src/Schema.php), [`Example Object`](src/Example.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |

##### [4.7.12.2.3 Fixed Fields for use with `content`](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-for-use-with-content)

| Field Name | Type                                                                   | Status             |
|------------|------------------------------------------------------------------------|--------------------|
| content    | Map[[`string`](src/Schema.php), [`Media Type Object`](src/Schema.php)] | :white_check_mark: |

### [4.7.13 Request Body Object](https://spec.openapis.org/oas/v3.0.4.html#request-body-object)

Describes a single request body.

#### [4.7.13.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-10)

| Field Name  | Type                                                                           | Status             |
|-------------|--------------------------------------------------------------------------------|--------------------|
| description | [`string`](src/RequestBody.php)                                                | :white_check_mark: |
| content     | Map[[`string`](src/RequestBody.php), [`Media Type Object`](src/MediaType.php)] | :white_check_mark: |
| required    | [`boolean`](src/RequestBody.php)                                               | :white_check_mark: |

### [4.7.14 Media Type Object](https://spec.openapis.org/oas/v3.0.4.html#media-type-object)

Each Media Type Object provides schema and examples for the media type identified by its key.

#### [4.7.14.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-11)

| Field Name | Type                                                                                                               | Status             |
|------------|--------------------------------------------------------------------------------------------------------------------|--------------------|
| schema     | [`Schema Object`](src/Schema.php) \| [`Reference Object`](src/Reference.php)                                       | :white_check_mark: |
| example    | [Any](src/MediaType.php)                                                                                           | :white_check_mark: |
| examples   | Map[[`string`](src/MediaType.php), [`Example Object`](src/Example.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |
| encoding   | Map[[`string`](src/MediaType.php), [`Encoding Object`](src/Encoding.php)]                                          | :white_check_mark: |

### [4.7.15 Encoding Object](https://spec.openapis.org/oas/v3.0.4.html#encoding-object)

A single encoding definition applied to a single schema property.

#### [4.7.15.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-12)

##### [4.7.15.1.1 Common Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#common-fixed-fields-0)

These fields _MAY_ be used either with or without the RFC6570-style serialization fields defined in the next section below.

| Field Name  | Type                                                                                                            | Status             |
|-------------|-----------------------------------------------------------------------------------------------------------------|--------------------|
| contentType | [`string`](src/Encoding.php)                                                                                    | :white_check_mark: |
| headers     | Map[[`string`](src/Encoding.php), [`Header Object`](src/Header.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |

##### [4.7.15.1.2 Fixed Fields for RFC6570-style Serialization](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-for-rfc6570-style-serialization)

| Field Name    | Type                          | Status             |
|---------------|-------------------------------|--------------------|
| style         | [`string`](src/Encoding.php)  | :white_check_mark: |
| explode       | [`boolean`](src/Encoding.php) | :white_check_mark: |
| allowReserved | [`boolean`](src/Encoding.php) | :white_check_mark: |

### [4.7.16 Responses Object](https://spec.openapis.org/oas/v3.0.4.html#responses-object)

A container for the expected responses of an operation. The container maps an HTTP response code to the expected response.

#### [4.7.16.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-13)

| Field Name | Type                                                                             | Status             |
|------------|----------------------------------------------------------------------------------|--------------------|
| default    | [`Response Object`](src/Response.php) \| [`Reference Object`](src/Reference.php) | :white_check_mark: |

#### [4.7.16.2 Patterned Fields](https://spec.openapis.org/oas/v3.0.4.html#patterned-fields-0)

| Field Name       | Type                                                                             | Status             |
|------------------|----------------------------------------------------------------------------------|--------------------|
| HTTP Status Code | [`Response Object`](src/Response.php) \| [`Reference Object`](src/Reference.php) | :white_check_mark: |

### [4.7.17 Response Object](https://spec.openapis.org/oas/v3.0.4.html#response-object)

Describes a single response from an API operation, including design-time, static links to operations based on the response.

#### [4.7.17.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-14)

| Field Name  | Type                                                                                                        | Status             |
|-------------|-------------------------------------------------------------------------------------------------------------|--------------------|
| description | [`string`](src/Response.php)                                                                                | :white_check_mark: |
| headers     | Map[[`string`](src/Response.php), [Header Object](src/Header.php) \| [Reference Object](src/Reference.php)] | :white_check_mark: |
| content     | Map[[`string`](src/Response.php), [Media Type Object](src/Header.php)]                                      | :white_check_mark: |
| links       | Map[[`string`](src/Response.php), [Link Object](src/Link.php) \| [Reference Object](src/Reference.php)]     | :white_check_mark: |

### [4.7.19 Example Object](https://spec.openapis.org/oas/v3.0.4.html#example-object)

An object grouping an internal or external example value with basic `summary` and `description` metadata.

#### [4.7.19.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-15)

| Field Name    | Type                        | Status             |
|---------------|-----------------------------|--------------------|
| summary       | [`string`](src/Example.php) | :white_check_mark: |
| description   | [`string`](src/Example.php) | :white_check_mark: |
| value         | [Any](src/Example.php)      | :white_check_mark: |
| externalValue | [`string`](src/Example.php) | :white_check_mark: |

### [4.7.20 Link Object](https://spec.openapis.org/oas/v3.0.4.html#link-object)

The Link Object represents a possible design-time link for a response.

#### [4.7.20.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-16)

| Field Name   | Type                                                                               | Status             |
|--------------|------------------------------------------------------------------------------------|--------------------|
| operationRef | [`string`](src/Link.php)                                                           | :white_check_mark: |
| operationId  | [`string`](src/Link.php)                                                           | :white_check_mark: |
| parameters   | Map[[`string`](src/Link.php), [Any](src/Link.php) \| [{expression}](src/Link.php)] | :white_check_mark: |
| requestBody  | [Any](src/Link.php) \| [{expression}](src/Link.php)                                | :white_check_mark: |
| description  | [`string`](src/Link.php)                                                           | :white_check_mark: |
| server       | [`Server Object`](src/Server.php)                                                  | :white_check_mark: |

### [4.7.21 Header Object](https://spec.openapis.org/oas/v3.0.4.html#header-object)

Describes a single header for HTTP responses and for individual parts in multipart representations.

#### [4.7.21.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-17)

##### [4.7.21.1.1 Common Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#common-fixed-fields-1)

These fields _MAY_ be used with either `content` or `schema`.

| Field Name  | Type                        | Status             |
|-------------|-----------------------------|--------------------|
| description | [`string`](src/Header.php)  | :white_check_mark: |
| required    | [`boolean`](src/Header.php) | :white_check_mark: |
| deprecated  | [`boolean`](src/Header.php) | :white_check_mark: |

###### [4.7.21.1.2 Fixed Fields for use with `schema`](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-for-use-with-schema-0)

For simpler scenarios, a schema and style can describe the structure and syntax of the header.

| Field Name | Type                                                                                                            | Status             |
|------------|-----------------------------------------------------------------------------------------------------------------|--------------------|
| style      | [`string`](src/Header.php)                                                                                      | :white_check_mark: |
| explode    | [`boolean`](src/Header.php)                                                                                     | :white_check_mark: |
| schema     | [`Schema Object`](src/Schema.php) \| [`Reference Object`](src/Reference.php)                                    | :white_check_mark: |
| example    | [Any](src/Header.php)                                                                                           | :white_check_mark: |
| examples   | Map[[`string`](src/Header.php), [`Example Object`](src/Example.php) \| [`Reference Object`](src/Reference.php)] | :white_check_mark: |

##### [4.7.21.1.3 Fixed Fields for use with `content`](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-for-use-with-content-0)

For more complex scenarios, the content field can define the media type and schema of the header, as well as give examples of its use.

| Field Name | Type                                                                      | Status             |
|------------|---------------------------------------------------------------------------|--------------------|
| content    | Map[[`string`](src/Header.php), [`Media Type Object`](src/MediaType.php)] | :white_check_mark: |

### [4.7.22 Tag Object](https://spec.openapis.org/oas/v3.0.4.html#tag-object)

Adds metadata to a single tag that is used by the Operation Object. It is not mandatory to have a Tag Object per tag defined in the Operation Object
instances.

#### [4.7.22.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-18)

| Field       | Type                                                             | Status             |
|-------------|------------------------------------------------------------------|--------------------|
| name        | [`string`](src/Tag.php)                                          | :white_check_mark: |
| description | [`string`](src/Tag.php)                                          | :white_check_mark: |
| description | [`External Documentation Object`](src/ExternalDocumentation.php) | :white_check_mark: |

### [4.7.23 Reference Object](https://spec.openapis.org/oas/v3.0.4.html#reference-object)

A simple object to allow referencing other components in the OpenAPI Description, internally and externally.

#### [4.7.23.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-19)

| Field | Type                          | Status             |
|-------|-------------------------------|--------------------|
| $ref  | [`string`](src/Reference.php) | :white_check_mark: |

### [4.7.24 Schema Object](https://spec.openapis.org/oas/v3.0.4.html#schema-object)

The Schema Object allows the definition of input and output data types.

#### [4.7.24.1 JSON Schema Keywords](https://spec.openapis.org/oas/v3.0.4.html#json-schema-keywords)

| Field Name           | Type                                                                                                          | Status             |
|----------------------|---------------------------------------------------------------------------------------------------------------|--------------------|
| title                | [`string`](src/Schema.php)                                                                                    | :white_check_mark: |
| multipleOf           | [`number`](src/Schema.php)                                                                                    | :white_check_mark: |
| maximum              | [`number`](src/Schema.php)                                                                                    | :white_check_mark: |
| exclusiveMaximum     | [`boolean`](src/Schema.php)                                                                                   | :white_check_mark: |
| minimum              | [`number`](src/Schema.php)                                                                                    | :white_check_mark: |
| exclusiveMinimum     | [`boolean`](src/Schema.php)                                                                                   | :white_check_mark: |
| maxLength            | [`integer`](src/Schema.php)                                                                                   | :white_check_mark: |
| minLength            | [`integer`](src/Schema.php)                                                                                   | :white_check_mark: |
| pattern              | [`string`](src/Schema.php)                                                                                    | :white_check_mark: |
| maxItems             | [`integer`](src/Schema.php)                                                                                   | :white_check_mark: |
| minItems             | [`integer`](src/Schema.php)                                                                                   | :white_check_mark: |
| uniqueItems          | [`boolean`](src/Schema.php)                                                                                   | :white_check_mark: |
| maxProperties        | [`integer`](src/Schema.php)                                                                                   | :white_check_mark: |
| minProperties        | [`integer`](src/Schema.php)                                                                                   | :white_check_mark: |
| required             | [`array`](src/Schema.php)                                                                                     | :white_check_mark: |
| enum                 | [`array`](src/Schema.php)                                                                                     | :white_check_mark: |
| type                 | [`string`](src/Schema.php)                                                                                    | :white_check_mark: |
| allOf                | [`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php)                                  | :white_check_mark: |
| oneOf                | [`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php)                                  | :white_check_mark: |
| anyOf                | [`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php)                                  | :white_check_mark: |
| not                  | [`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php)                                  | :white_check_mark: |
| items                | [`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php)                                  | :white_check_mark: |
| properties           | [`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php)                                  | :white_check_mark: |
| additionalProperties | [`boolean`](src/Reference.php) \|[`Reference Object`](src/Reference.php) \| [`Schema Object`](src/Schema.php) | :white_check_mark: |
| description          | [`string`](src/Schema.php)                                                                                    | :white_check_mark: |
| format               | [`string`](src/Schema.php)                                                                                    | :white_check_mark: |
| format               | [`Any`](src/Schema.php)                                                                                       | :white_check_mark: |

#### [4.7.24.2 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-20)

| Field Name    | Type                                                             | Status             |
|---------------|------------------------------------------------------------------|--------------------|
| nullable      | [`boolean`](src/Schema.php)                                      | :white_check_mark: |
| discriminator | `Discriminator Object`                                           | :white_check_mark: |
| readOnly      | [`boolean`](src/Schema.php)                                      | :white_check_mark: |
| writeOnly     | [`boolean`](src/Schema.php)                                      | :white_check_mark: |
| xml           | [`XML Object`](src/Xml.php)                                      | :white_check_mark: |
| externalDocs  | [`External Documentation Object`](src/ExternalDocumentation.php) | :white_check_mark: |
| example       | [Any](src/Schema.php)                                            | :white_check_mark: |
| deprecated    | [`boolean`](src/Schema.php)                                      | :white_check_mark: |

### [4.7.25 Discriminator Object](https://spec.openapis.org/oas/v3.0.4.html#discriminator-object)

When request bodies or response payloads may be one of a number of different schemas,
a Discriminator Object gives a hint about the expected schema of the document.

#### [4.7.25.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-21)

| Field Name   | Type                                             | Status             |
|--------------|--------------------------------------------------|--------------------|
| propertyName | [`string`](src/Discriminator.php)                | :white_check_mark: |
| mapping      | Map[[`string`](src/Discriminator.php), `string`] | :white_check_mark: |

### [4.7.26 XML Object](https://spec.openapis.org/oas/v3.0.4.html#xml-object)

A metadata object that allows for more fine-tuned XML model definitions.

#### [4.7.26.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-22)

| Field Name | Type                     | Status             |
|------------|--------------------------|--------------------|
| name       | [`string`](src/Xml.php)  | :white_check_mark: |
| namespace  | [`string`](src/Xml.php)  | :white_check_mark: |
| prefix     | [`string`](src/Xml.php)  | :white_check_mark: |
| attribute  | [`boolean`](src/Xml.php) | :white_check_mark: |
| wrapped    | [`boolean`](src/Xml.php) | :white_check_mark: |

### [4.7.27 Security Scheme Object](https://spec.openapis.org/oas/v3.0.4.html#security-scheme-object-0)

Defines a security scheme that can be used by the operations.

#### [4.7.27.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-23)

| Field Name       | Type                               | Status             |
|------------------|------------------------------------|--------------------|
| type             | [`string`](src/SecurityScheme.php) | :white_check_mark: |
| description      | [`string`](src/SecurityScheme.php) | :white_check_mark: |
| name             | [`string`](src/SecurityScheme.php) | :white_check_mark: |
| in               | [`string`](src/SecurityScheme.php) | :white_check_mark: |
| scheme           | [`string`](src/SecurityScheme.php) | :white_check_mark: |
| bearerFormat     | [`string`](src/SecurityScheme.php) | :white_check_mark: |
| flows            | [`string`](src/OAuthFlows.php)     | :white_check_mark: |
| openIdConnectUrl | [`string`](src/SecurityScheme.php) | :white_check_mark: |

### [4.7.28 OAuth Flows Object](https://spec.openapis.org/oas/v3.0.4.html#oauth-flows-object)

Allows configuration of the supported OAuth Flows.

#### [4.7.28.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-24)

| Field Name        | Type                                     | Status             |
|-------------------|------------------------------------------|--------------------|
| implicit          | [`OAuth Flow Object`](src/OAuthFlow.php) | :white_check_mark: |
| password          | [`OAuth Flow Object`](src/OAuthFlow.php) | :white_check_mark: |
| clientCredentials | [`OAuth Flow Object`](src/OAuthFlow.php) | :white_check_mark: |
| authorizationCode | [`OAuth Flow Object`](src/OAuthFlow.php) | :white_check_mark: |

### [4.7.29 OAuth Flow Object](https://spec.openapis.org/oas/v3.0.4.html#security-scheme-object-0)

Defines a security scheme that can be used by the operations.

#### [4.7.29.1 Fixed Fields](https://spec.openapis.org/oas/v3.0.4.html#fixed-fields-25)

| Field Name       | Type                                                              | Status             |
|------------------|-------------------------------------------------------------------|--------------------|
| authorizationUrl | [`string`](src/OAuthFlow.php)                                     | :white_check_mark: |
| tokenUrl         | [`string`](src/OAuthFlow.php)                                     | :white_check_mark: |
| refreshUrl       | [`string`](src/OAuthFlow.php)                                     | :white_check_mark: |
| scopes           | Map[[`string`](src/OAuthFlow.php), [`string`](src/OAuthFlow.php)] | :white_check_mark: |

## Contributing

Contributions, issues, and feature requests are welcome!
Feel free to check the [issues](https://github.com/zero-to-prod/data-model-openapi30/issues) page if you want to contribute.

1. Fork the repository.
2. Create a new branch (`git checkout -b feature-branch`).
3. Commit changes (`git commit -m 'Add some feature'`).
4. Push to the branch (`git push origin feature-branch`).
5. Create a new Pull Request.
 readmeEtag: '"97699662f902c02ff4b96eaed4f1bfcb6d92b5fe"' readmeLastModified: Mon, 01 Sep 2025 22:45:13 GMT repositoryId: 885105129 description: DataModels for OpenAPI 3.0.* created: '2024-11-08T00:54:00Z' updated: '2026-01-18T20:15:51Z' language: PHP archived: false stars: 3 watchers: 1 forks: 0 owner: zero-to-prod logo: https://avatars.githubusercontent.com/u/61474950?v=4 license: MIT repoEtag: '"6ffa505038e9f1f44e050463b31957c46ce31cbc8f4a98a3b153ad78fb58d2c3"' repoLastModified: Sun, 18 Jan 2026 20:15:51 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/norbiox/audiolabapp v3: true repositoryMetadata: base64Readme: >- IyBBdWRpb0xhYkFwcApMYWJvcmF0b3JpdW0gZMW6d2nEmWtvd2UuCgpTZXJ3aXMgbGFiYXBwIHBvendhbGEgbmEgZ3JvbWFkemVuaWUgaSB6YXJ6xIVkemFuaWUgbmFncmFuaWFtaSBkxbp3acSZa293eW1pIHBvY2hvZHrEhWN5bWkgeiBla3NwZXJ5bWVudMOzdy4KCgojIyBDZWwKClByb2pla3QgdGVuIHBvd3N0YcWCIHcgcmFtYWNoIHByb2pla3R1IHN0dWRlbmNraWVnbyB6IHByemVkbWlvdHUgQXBsaWthY2plIEludGVybmV0b3dlLiBDZWxlbSBwcm9qZWt0dSBiecWCbyAob3Byw7NjeiB6YWxpY3plbmlhIHByemVkbWlvdHUgOykgKSB6ZG9ieWNpZSB3aWVkenkgaSBkb8Wbd2lhZGN6ZW5pYSB3IHByYWN5IHogcG9uacW8c3p5bWkgdGVjaG5vbG9naWFtaSBpIG5hcnrEmWR6aWFtaToKCiogcHl0aG9uICsgZmxhc2sKKiBSRVNUIEFQSQoqIE9wZW4gQVBJIDMgKFN3YWdnZXIpCiogZG9ja2VyCiogZG9ja2VyLWNvbXBvc2UKCgojIyBUZXN0eSBpIHVydWNob21pZW5pZSB3IMWbcm9kb3dpc2t1IGRld2Vsb3BlcnNraW0KCjEuIFNrbG9udWogdG8gcmVwb3p5dG9yaXVtCgoyLiBEb2RhaiDFm2NpZcW8a8SZIGZvbGRlcnUgYGBgbGFiYXBwYGBgIGRvIHptaWVubmVqIHN5c3RlbW93ZWogYGBgUFlUSE9OUEFUSGBgYAoKMy4gUG9iaWVyeiBpIHVydWNob20ga29udGVuZXIgTXlTUUwgcG9sZWNlbmllbQoKICAgICAgICBkb2NrZXIgcnVuIC0tbmFtZSBteXNxbCBcCiAgICAgICAgICAgIC1wIDMzMDY6MzMwNiBcCiAgICAgICAgICAgIC1lIE1ZU1FMX1JBTkRPTV9ST09UX1BBU1NXT1JEPXllcyBcCiAgICAgICAgICAgIC1lIE1ZU1FMX0RBVEFCQVNFPWxhYmFwcCBcCiAgICAgICAgICAgIC1lIE1ZU1FMX1VTRVI9bGFiYXBwIFwKICAgICAgICAgICAgLWUgTVlTUUxfUEFTU1dPUkQ9PHBhc3NwaHJhc2U+IFwKICAgICAgICAgICAgLWQgbXlzcWw6NS43Cgo0LiBVc3RhdyBvZHBvd2llZG5pbyB6bWllbm7EhSBEQVRBQkFTRV9VUkw6CgogICAgZXhwb3J0IERBVEFCQVNFX1VSTD1teXNxbDovL2xhYmFwcDo8cGFzc3BocmFzZT5APGRhdGFiYXNlX2FkZHJlc3M6cG9ydD4vbGFiYXBwCgo1LiBQcnplamTFuiBkbyBmb2xkZXJ1IGBgYGxhYmFwcGBgYCBpIHByemVwcm93YWTFuiB0ZXN0eSBhdXRvbWF0eWN6bmUgcG9sZWNlbmllbQoKICAgICAgICAuL3Rlc3Quc2gKCjYuIFpha3R1YWxpenVqIGJhesSZIGRhbnljaCB3eWtvbnVqxIVjIHBvbGVjZW5pYQoKICAgICAgICBmbGFzayBkYiB1cGdyYWRlCgo3LiBVcnVjaG9tIGFwbGlrYWNqxJkgcG9sZWNlbmllbQoKICAgICAgICBmbGFzayBydW4KClVXQUdBOiBhcGxpa2FjamEgZG9tecWbbG5pZSB1cnVjaG9taW9uYSB6b3N0YW5pZSB3IHRyeWJpZSAnZGV2ZWxvcG1lbnQnLCBhYnkgdXJ1Y2hvbWnEhyB3IHRyeWJpZSAncHJvZHVjdGlvbicgem1pZcWEIHdhcnRvxZtjaSB6bWllbm55Y2ggdyBwbGlrdSAuZmxhc2tlbnY6CgogICAgRkxBU0tfQVBQPSJhcHA6Y3JlYXRlX2FwcCgnYXBwLmNvbmZpZy5Qcm9kdWN0aW9uQ29uZmlnJykiCiAgICBGTEFTS19FTlY9InByb2R1Y3Rpb24iCgo4LiBBUEkgbW/FvG5hIHRlc3Rvd2HEhyBkb3dvbG55bSBuYXJ6xJlkemllbSwgbHViIHcgcHJ6ZWdsxIVkYXJrb3d5bSBla3NwbG9yYXRvcnplIFN3YWdnZXIgcG9kIGFkcmVzZW06CgogICAgICAgIGh0dHA6Ly9bdXJsX2FwbGlrYWNqaV0vMS4wL2xhYi91aQoKCiMjIFVydWNob21pZW5pZSBza29udGVuZXJ5em93YW5laiBhcGxpa2FjamkKCklzdG5pZWplIG1vxbxsaXdvxZvEhyB1cnVjaG9taWVuaWEgYXBsaWthY2ppIHcga29udGVuZXJ6ZSB6YSBwb21vY8SFIGRvY2tlci1jb21wb3NlLiBXIHR5bSBjZWx1IG5hbGXFvHkgbmFqcGllcncgemFwZXduacSHIG5hc3TEmXB1asSFY2Ugem1pZW5uZSDFm3JvZG93aXNrb3dlOgoKICAgIGV4cG9ydCBBUFBfUE9SVD0gXAogICAgICAgIERCX1JPT1RfUEFTU1dPUkQ9IFwKICAgICAgICBEQl9OQU1FPSBcCiAgICAgICAgREJfVVNFUj0gXAogICAgICAgIERCX1BBU1NXT1JEPSBcCiAgICAgICAgU0VDUkVUX0tFWT0gXAogICAgICAgIE1FRElBX0RJUj0KCkEgbmFzdMSZcG5pZSBixJlkxIVjIHcgZm9sZGVyemUgYGBgbGFiYXBwYGBgIHd5a29uYcSHIHBvbGVjZW5pZToKCiAgICBkb2NrZXItY29tcG9zZSB1cCAtZCAtLWJ1aWxkCg== readmeEtag: '"8970d9a3e2771ec54982de35e86e582d1d8f00e2"' readmeLastModified: Tue, 07 May 2019 07:10:21 GMT repositoryId: 153651596 description: AudioLabApp - laboratorium dźwiękowe created: '2018-10-18T16:06:10Z' updated: '2020-05-09T23:27:25Z' language: Python archived: false stars: 1 watchers: 0 forks: 0 owner: Norbiox logo: https://avatars.githubusercontent.com/u/22893882?v=4 repoEtag: '"df0bb7f70c373ce2201f0f3decf6f729f9569f96265a76a6317c032a897e4953"' repoLastModified: Sat, 09 May 2020 23:27:25 GMT foundInMaster: true category: Mock id: 15eae4b761323a3653c13f89c4b338b7 - source: openapi3 tags repository: https://github.com/rezsolt/symfony-api-platform-maker-bundle v3: true repositoryMetadata: base64Readme: >- IyBBcGlQbGF0Zm9ybU1ha2VyQnVuZGxlIGZvciBTeW1mb255IDQKUHJvdmlkZSBtYWtlIGNvbW1hbmQgZm9yIFN5bWZvbnkgQVBJIFBsYXRmb3JtIHdoYXQgc3VwcG9ydHMgT3BlbkFQSSAzLjAgc2NoZW1hIHNvdXJjZQo= readmeEtag: '"e0b79d1ab14ee7dad4a23ba59014c21c6b157c1f"' readmeLastModified: Sun, 11 Aug 2019 13:45:25 GMT repositoryId: 168426934 description: >- Provide make command for Symfony API Platform what supports OpenAPI 3.0 schema source created: '2019-01-30T22:42:51Z' updated: '2019-08-11T16:26:55Z' language: PHP archived: false stars: 1 watchers: 1 forks: 0 owner: rezsolt logo: https://avatars.githubusercontent.com/u/17837327?v=4 license: MIT repoEtag: '"bcd6c5fee3a346b12585b3127cd592eac208788a611161e4dac2347d53dac04a"' repoLastModified: Sun, 11 Aug 2019 16:26:55 GMT foundInMaster: true category: - Server - Parsers id: 6ad157fe1ad35f5927ca1e6047894a1a - source: openapi3 tags repository: https://github.com/linoaviii/simple-gpio-api v3: true repositoryMetadata: base64Readme: >- CiMgUmFzcGJlcnJ5UGkgc2ltcGxlIEdQSU8gQVBJCgpTaW1wbGUgV2ViIGFwaSB1c2VkIHRvIHJlYWQgYW5kIHdyaXRlIFJhc3BiZXJyeVBpIEdQSU8gc3RhdGVzLgpDaGFuZ2UgeW91ciBncGlvIHN0YXRlcyBieSBVUkwgKHVzZWZ1bGwgZm9yIGRvbW90aWMgcHJvamVjdCkKTm8gcHJvZ3JhbW1pbmcga25vd2xlZGdlIG5lZWRlZAoKVGhpcyBBUEkgaXMgY29tcG9zZWQgYnl0IGEgR1BJTyB3ZWIgdGVzdCA6CmBgYApodHRwOi8ve0lQX1JBU1BCRVJSWX0KYGBgCiFbd2ViIGFwaSBwcmV2aWV3XShpbWFnZXMvdGVzdC13ZWIuanBnKQoKQW5kIGFuIGFjY2VzIHRvIEdQSU8gYnkgYSBzaW1wbGUgVVJMOgoKIC0gUmVhZCBHUElPIChpbnB1dCBvciBvdXRwdXQpCmBgYApodHRwOi8ve0lQX1JBU1BCRVJSWX0vZ3Bpby97Z3Bpb19udW1iZXJ9CndpbGwgcmV0dXJuIDAgaWYgR1BJTyBpcyBEb3duLCAxIGlmIEdQSU8gaXMgdXAgKC0xIGlmIGVycm9yKQpgYGAKZXhhbXBsZSwgcmVhZCBHUElPIDEyIHN0YXRlIChHUElPIG51bWJlciBpcyBzZXQgaW4gQkNNIG1vZGUpCmBgYApodHRwOi8vMTkyLjE2OC4wLjEvZ3Bpby8xMgpgYGAKLSBXcml0ZSBHUElPIChvdXRwdXQpCmBgYApodHRwOi8ve0lQX1JBU1BCRVJSWX0vZ3Bpby97Z3Bpb19udW1iZXJ9L3tncGlvX3N0YXRlfQp3aWxsIHNldCB0aGUgZ3BpbyB3aXRoIHRoZSBzdGF0ZSBhbmQgcmV0dXJuIDAgaWYgR1BJTyBpcyBEb3duLCAxIGlmIEdQSU8gaXMgdXAgKC0xIGlmIGVycm9yKQpgYGAKZXhhbXBsZSwgc2V0IEdQSU8gMTIgdXAgKEdQSU8gbnVtYmVyIGlzIHNldCBpbiBCQ00gbW9kZSkKYGBgCmh0dHA6Ly8xOTIuMTY4LjAuMS9ncGlvLzEyLzEKYGBgCmV4YW1wbGUsIHNldCBHUElPIDEzIGRvd24gKEdQSU8gbnVtYmVyIGlzIHNldCBpbiBCQ00gbW9kZSkKYGBgCmh0dHA6Ly8xOTIuMTY4LjAuMS9ncGlvLzEzLzAKYGBgCgojIyBHZXR0aW5nIFN0YXJ0ZWQKClRoZXNlIGluc3RydWN0aW9ucyB3aWxsIGdldCB5b3UgYSBjb3B5IG9mIHRoZSBwcm9qZWN0IHVwIGFuZCBydW5uaW5nIG9uIHlvdXIgUmFzcGJlcnJ5UGkuCgojIyMgUHJlcmVxdWlzaXRlcwoKVG8gZmFjaWxpdGF0ZSBpbnN0YWxsYXRpb24sIGV2ZXJ5dGhpbmcgaXMgaW5jbHVkZSBpbiB0aGUgcHJvamVjdC4KWW91IGp1c3QgbmVlZCBhIHJhc3BiZXJyeSBQaSAoZXZlcnkgdmVyc2lvbnMgd29ya2luZykgd2l0aCBSYXNwYmlhbiBPUyBpbnN0YWxsZWQKVG8gaW5zdGFsbCBSYXNwYmlhbiwgc2VlIDogaHR0cHM6Ly93d3cucmFzcGJlcnJ5cGkub3JnL2Rvd25sb2Fkcy8KCgojIyMjIGEpIEF1dG9tYXRpYyBjb3B5ICh1c2luZyBnaXQpCgogLSBPcGVuIGEgdGVybWluYWwgOgogYGBgCiBDdHJsICsgQWx0ICsgVAogYGBgCiAtIENsb25lIHNpbXBsZS1ncGlvLWFwaSBwcm9qZWN0IGluIGhvbWUgZm9sZGVyICgvaG9tZS9waSk6CmBgYApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2xpbm9hdmlpaS9zaW1wbGUtZ3Bpby1hcGkgL2hvbWUvcGkvc2ltcGxlLWdwaW8tYXBpCmBgYAoKIyMjIyBiKSBtYW51YWwgY29weSAKSWYgZ2l0IGlzIG5vdCBpbnN0YWxsZWQgb3IgaWYgeW91IGRvbid0ICB3YW50IHRvIHVzZSBpdCwgeW91IGNhbiBjb3B5IHNpbXBsZS1ncGlvLWFwaSBmb2xkZXIgIG1hbnVhbGx5IGludG8geW91ciBob21lIGZvbGRlciAoL2hvbWUvcGkpCgoKIyMjIEluc3RhbGxpbmcKIC0gT3BlbiBhIHRlcm1pbmFsIDoKIGBgYAogQ3RybCArIEFsdCArIFQKIGBgYAogLSBSdW4gc2V0dXAgc2NyaXB0CmBgYApzdWRvIHNoIHNpbXBsZS1ncGlvLWFwaS9zZXR1cC5zaApgYGAKYW5kIGl0J3MgZG9uZQoKIyMgUnVubmluZwoKIC0gT3BlbiBhIHRlcm1pbmFsIDoKIGBgYAogQ3RybCArIEFsdCArIFQKIGBgYAogLSBSdW4gc3RhcnQgc2NyaXB0CmBgYApzaCBzaW1wbGUtZ3Bpby1hcGkvc3RhcnQuc2gKYGBgCiFbd2Vic2VydmljZSBzdGFydGVkIHByZXZpZXddKGltYWdlcy93ZWJzZXJ2aWNlLXN0YXJ0ZWQuanBnKQoKRG9uJ3QgY2xvc2UgdGhlIHRlcm1pbmFsIHRvIGtlZXAgQVBJIGFsaXZlLgoKQnkgZGVmYXVsdCwgZXZlcnkgR1BJTyBhcmUgc2V0IHRvIE9VVFBVVCwgaWYgeW91IHdhbnQgdG8gc2V0IHNvbWUgR1BJTyBhcyBJbnB1dCwgZWRpdCB0aGUgKipjb25maWcuSU5JKiogZmlsZSBpbiAqKnNpbXBsZS1ncGlvLWFwaSoqIGZvbGRlciAKbGlzdCBhbGwgR1BJTyBudW1iZXIgKGluIEJDTSBtb2RlKSB5b3Ugd2FudCBhcyBJbnB1dCBpbiAqKmdwaW9faW5wdXQqKiBsaW5lLCBzZXBhcmF0ZWQgYnkgc3BhY2UKZXhhbXBsZToKIGBgYApncGlvX2lucHV0ID0gMiAzIDQgMTcgCiBgYGAKUmVzdGFydCB0aGUgc2NyaXB0IHRvIGNoYW5nZSBHUElPIGZ1bmN0aW9ucwoKIyMgVXNpbmcKWW91IGNhbiB1c2UgdGhpcyBBUEkgZGlyZWN0bHkgYnkgeW91ciBicm93c2VyLgpBdmFpbGFibGUgY29tbWFuZHMgOgoKIC0gKipSZWFkIEdQSU8qKiA6IGh0dHA6Ly8qKntJUF9SQVNQQkVSUll9KiovZ3Bpby8qKntncGlvX251bWJlcn0qKgogLSAqKldyaXRlIEdQSU8qKiA6IGh0dHA6Ly8qKntJUF9SQVNQQkVSUll9KiovZ3Bpby8qKntncGlvX251bWJlcn0qKi8qKntncGlvX3N0YXRlfSoqCiAtICoqR1BJTyBUZXN0Kio6IGh0dHA6Ly8qKntJUF9SQVNQQkVSUll9KioKIC0gKipBUEkgZG9jdW1lbnRhdGlvbioqOiBodHRwOi8vKip7SVBfUkFTUEJFUlJZfSoqL2RvYwoKIyMgTGF1bmNoIGF0IHN0YXJ0dXAKSWYgeW91IHdhbnQgdG8gbGF1bmNoIGF1dG9tYXRpY2FsbHkgdGhlIHNjcmlwdCBhdCBSYXNwYmVycnlQaSBzdGFydHVwCiAtIE9wZW4gYSB0ZXJtaW5hbCA6CiBgYGAKIEN0cmwgKyBBbHQgKyBUCiBgYGAKIC0gbGF1bmNoIHRoZXNlcyBjb21tYW5kcyA6CiBgYGAKIHN1ZG8gY3AgL2hvbWUvcGkvc2ltcGxlLWdwaW8tYXBpL3NpbXBsZS1ncGlvLWFwaS5zZXJ2aWNlIC9ldGMvc3lzdGVtZC9zeXN0ZW0KIHN1ZG8gc3lzdGVtY3RsIGVuYWJsZSBzaW1wbGUtZ3Bpby1hcGkuc2VydmljZSAKIGBgYAogLSByZWJvb3QgeW91ciByYXNwYmVycnlQaSAodGhlIEFQSSB3aWxsIHJ1biB3aXRob3V0IHRlcm1pbmFsLiBZb3UgY2FuIHRlc3QgaXQgZGlyZWN0bHkgd2l0aCB5b3UgYnJvd3NlcikKCiMjIEJ1aWx0IFdpdGgKKiBbUHl0aG9uXShodHRwczovL3d3dy5weXRob24ub3JnLykgLSBQeXRob24gMwoqIFtTd2FnZ2VyXShodHRwczovL3N3YWdnZXIuaW8vKSAtIFdlYiBBcGkgZWRpdG9yIChvcGVuQXBpKQoKCiMjIFZlcnNpb25pbmcKCkkgdXNlIFtTZW1WZXJdKGh0dHA6Ly9zZW12ZXIub3JnLykgZm9yIHZlcnNpb25pbmcuIEZvciB0aGUgdmVyc2lvbnMgYXZhaWxhYmxlLCBzZWUgdGhlIFt0YWdzIG9uIHRoaXMgcmVwb3NpdG9yeV0oaHR0cHM6Ly9naXRodWIuY29tL3lvdXIvcHJvamVjdC90YWdzKS4gCgojIyBBdXRob3JzCgoqICoqRGF2aWQgQXVkcmFuKiogLSAqSW5pdGlhbCB3b3JrKiAtIFtXZWJzaXRlXShodHRwOi8vZGF2aWRhdWRyYW4uY29tKQoKCgo= readmeEtag: '"a1ac60a23c543e109738fd909c3bc6e272c733b9"' readmeLastModified: Wed, 20 May 2020 08:40:02 GMT repositoryId: 265502630 description: RaspberryPi GPIO api for domotic created: '2020-05-20T08:37:48Z' updated: '2020-05-20T09:05:00Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: linoaviii logo: https://avatars.githubusercontent.com/u/65229993?v=4 repoEtag: '"9b1031290dc9ebc0935eb38e348a2743398e73c401f008e3fc3b38e4028ffe65"' repoLastModified: Wed, 20 May 2020 09:05:00 GMT foundInMaster: true category: Server Implementations id: d7d492cbec52532f64b0502276ccf9b5 - source: openapi3 tags repository: https://github.com/rtfpessoa/glugen v3: true repositoryMetadata: repositoryId: 237613358 description: glugen generates http clients for an OpenAPI v3 specification created: '2020-02-01T12:43:06Z' updated: '2023-03-09T04:30:44Z' language: JavaScript archived: true stars: 1 watchers: 1 forks: 1 owner: rtfpessoa logo: https://avatars.githubusercontent.com/u/902384?v=4 repoEtag: '"c9adac6e4f2beca4e5aa7ee6c271bd557b82980e783a39fec7ae6b2fe7e584c9"' repoLastModified: Thu, 09 Mar 2023 04:30:44 GMT foundInMaster: true id: f140b1da60e51d3ea90076aa7cbd7cb6 - source: openapi3 tags repository: https://github.com/javabypatel/guice-grizzly-jersey-openapi-swagger-example v3: true repositoryMetadata: base64Readme: >- IyBndWljZS1ncml6emx5LWplcnNleS1vcGVuYXBpLXN3YWdnZXItZXhhbXBsZQoKU2FtcGxlIHByb2plY3QgdG8gc2hvdyBkeW5hbWljIGdlbmVyYXRpb24gb2YgT3BlbkFQSSBTd2FnZ2VyIGRvY3VtZW50YXRpb24gaW4gcHJvamVjdCB1c2luZyBHdWljZSBhbmQgSmVyc2V5LiAgCmh0dHBzOi8vamF2YWJ5cGF0ZWwuYmxvZ3Nwb3QuY29tLzIwMjAvMTEvc3dhZ2dlci1vcGVuYXBpLXJlc3QtamF2YS1leGFtcGxlLXVzaW5nLWd1aWNlLWplcnNleS5odG1sCgojIyMgQnVpbGQgdGhlIGFwcGxpY2F0aW9uCm12biBjbGVhbiBpbnN0YWxsCgojIyMgUnVuIHRoZSBhcHBsaWNhdGlvbgpHbyB0byBgYEFwcC5qYXZhYGAgd2hpY2ggY29udGFpbnMgbWFpbiBtZXRob2QsIHJ1biBpdCBhcyBqYXZhIGFwcGxpY2F0aW9uLgoKIyMjIyBEZW1vIGFwcGxpY2F0aW9uIFVSTCAKaHR0cDovL2xvY2FsaG9zdDo4MDgwL09wZW5BUElFeGFtcGxlL2dyZWV0P25hbWU9SmF2YUJ5UGF0ZWwKCiMjIyMgRm9yIGdldHRpbmcgdGhlIE9wZW5BUEkgcmVzb3VyY2UgZGVzY3JpcHRpb24sCkluIGpzb24gZm9ybWF0OiBodHRwOi8vbG9jYWxob3N0OjgwODAvT3BlbkFQSUV4YW1wbGUvb3BlbmFwaS5qc29uIFwKSW4geWFtbCBmb3JtYXQ6IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9PcGVuQVBJRXhhbXBsZS9vcGVuYXBpLnlhbWwKCiMjIyBzYW1wbGUganNvbgpgYGAKewogICJvcGVuYXBpIiA6ICIzLjAuMSIsCiAgImluZm8iIDogewogICAgInRpdGxlIiA6ICJHdWljZSBHcml6emx5IEplcnNleSBPcGVuYXBpIFN3YWdnZXIgRXhhbXBsZSBBUEkiLAogICAgImRlc2NyaXB0aW9uIiA6ICJPcGVuQVBJIHN3YWdnZXIgY29uZmlndXJhdGlvbiBleGFtcGxlIGluIHNhbXBsZSBwcm9qZWN0IHRoYXQgdXNlcyBHdWljZSwgR3JpenpseSwgSmVyc2V5LiIsCiAgICAiY29udGFjdCIgOiB7CiAgICAgICJlbWFpbCIgOiAiamF5ZXNobWFoZXNocGF0ZWxAZ21haWwuY29tIgogICAgfSwKICAgICJsaWNlbnNlIiA6IHsKICAgICAgIm5hbWUiIDogIk1JVCBMaWNlbnNlIiwKICAgICAgInVybCIgOiAiaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvTUlUX0xpY2Vuc2UiCiAgICB9LAogICAgInZlcnNpb24iIDogIjEuMC4wIgogIH0sCiAgInNlcnZlcnMiIDogWyB7CiAgICAidXJsIiA6ICJodHRwOi8vbG9jYWxob3N0OjgwODAvT3BlbkFQSUV4YW1wbGUvIiwKICAgICJkZXNjcmlwdGlvbiIgOiAiR3VpY2UgR3JpenpseSBKZXJzZXkgT3BlbmFwaSBTd2FnZ2VyIEV4YW1wbGUgQVBJIHNlcnZlciIKICB9IF0sCiAgInBhdGhzIiA6IHsKICAgICIvZ3JlZXQiIDogewogICAgICAiZ2V0IiA6IHsKICAgICAgICAic3VtbWFyeSIgOiAiVGhpcyBpcyBhIHNhbXBsZSB0ZXN0IEFQSSB0byBncmVldCB1c2VyLiIsCiAgICAgICAgIm9wZXJhdGlvbklkIiA6ICJncmVldCIsCiAgICAgICAgInBhcmFtZXRlcnMiIDogWyB7CiAgICAgICAgICAibmFtZSIgOiAibmFtZSIsCiAgICAgICAgICAiaW4iIDogInF1ZXJ5IiwKICAgICAgICAgICJzY2hlbWEiIDogewogICAgICAgICAgICAidHlwZSIgOiAic3RyaW5nIgogICAgICAgICAgfQogICAgICAgIH0gXSwKICAgICAgICAicmVzcG9uc2VzIiA6IHsKICAgICAgICAgICIyMDAiIDogewogICAgICAgICAgICAiZGVzY3JpcHRpb24iIDogIkdyZWV0ZWQgc3VjY2Vzc2Z1bGx5LiIsCiAgICAgICAgICAgICJjb250ZW50IiA6IHsKICAgICAgICAgICAgICAiYXBwbGljYXRpb24vanNvbiIgOiB7CiAgICAgICAgICAgICAgICAic2NoZW1hIiA6IHsKICAgICAgICAgICAgICAgICAgInR5cGUiIDogImFycmF5IiwKICAgICAgICAgICAgICAgICAgIml0ZW1zIiA6IHsKICAgICAgICAgICAgICAgICAgICAiJHJlZiIgOiAiIy9jb21wb25lbnRzL3NjaGVtYXMvR3JlZXRSZXNwb25zZSIKICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgICAgfSwKICAgICAgICAgICI0MDAiIDogewogICAgICAgICAgICAiZGVzY3JpcHRpb24iIDogIkJhZCByZXF1ZXN0LiBSZXF1ZXN0IGlzIG5vdCB3ZWxsIGZvcm1lZC4iCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9CiAgfSwKICAiY29tcG9uZW50cyIgOiB7CiAgICAic2NoZW1hcyIgOiB7CiAgICAgICJHcmVldFJlc3BvbnNlIiA6IHsKICAgICAgICAidHlwZSIgOiAib2JqZWN0IiwKICAgICAgICAicHJvcGVydGllcyIgOiB7CiAgICAgICAgICAibWVzc2FnZSIgOiB7CiAgICAgICAgICAgICJ0eXBlIiA6ICJzdHJpbmciCiAgICAgICAgICB9CiAgICAgICAgfQogICAgICB9CiAgICB9CiAgfQp9CmBgYAoKIyMjIHNhbXBsZSB5YW1sCgpgYGAKb3BlbmFwaTogMy4wLjEKaW5mbzoKICB0aXRsZTogR3VpY2UgR3JpenpseSBKZXJzZXkgT3BlbmFwaSBTd2FnZ2VyIEV4YW1wbGUgQVBJCiAgZGVzY3JpcHRpb246ICJPcGVuQVBJIHN3YWdnZXIgY29uZmlndXJhdGlvbiBleGFtcGxlIGluIHNhbXBsZSBwcm9qZWN0IHRoYXQgdXNlc1wKICAgIFwgR3VpY2UsIEdyaXp6bHksIEplcnNleS4iCiAgY29udGFjdDoKICAgIGVtYWlsOiBqYXllc2htYWhlc2hwYXRlbEBnbWFpbC5jb20KICBsaWNlbnNlOgogICAgbmFtZTogTUlUIExpY2Vuc2UKICAgIHVybDogaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvTUlUX0xpY2Vuc2UKICB2ZXJzaW9uOiAxLjAuMApzZXJ2ZXJzOgotIHVybDogaHR0cDovL2xvY2FsaG9zdDo4MDgwL09wZW5BUElFeGFtcGxlLwogIGRlc2NyaXB0aW9uOiBHdWljZSBHcml6emx5IEplcnNleSBPcGVuYXBpIFN3YWdnZXIgRXhhbXBsZSBBUEkgc2VydmVyCnBhdGhzOgogIC9ncmVldDoKICAgIGdldDoKICAgICAgc3VtbWFyeTogVGhpcyBpcyBhIHNhbXBsZSB0ZXN0IEFQSSB0byBncmVldCB1c2VyLgogICAgICBvcGVyYXRpb25JZDogZ3JlZXQKICAgICAgcGFyYW1ldGVyczoKICAgICAgLSBuYW1lOiBuYW1lCiAgICAgICAgaW46IHF1ZXJ5CiAgICAgICAgc2NoZW1hOgogICAgICAgICAgdHlwZTogc3RyaW5nCiAgICAgIHJlc3BvbnNlczoKICAgICAgICAiMjAwIjoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBHcmVldGVkIHN1Y2Nlc3NmdWxseS4KICAgICAgICAgIGNvbnRlbnQ6CiAgICAgICAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgICAgICAgc2NoZW1hOgogICAgICAgICAgICAgICAgdHlwZTogYXJyYXkKICAgICAgICAgICAgICAgIGl0ZW1zOgogICAgICAgICAgICAgICAgICAkcmVmOiAnIy9jb21wb25lbnRzL3NjaGVtYXMvR3JlZXRSZXNwb25zZScKICAgICAgICAiNDAwIjoKICAgICAgICAgIGRlc2NyaXB0aW9uOiBCYWQgcmVxdWVzdC4gUmVxdWVzdCBpcyBub3Qgd2VsbCBmb3JtZWQuCmNvbXBvbmVudHM6CiAgc2NoZW1hczoKICAgIEdyZWV0UmVzcG9uc2U6CiAgICAgIHR5cGU6IG9iamVjdAogICAgICBwcm9wZXJ0aWVzOgogICAgICAgIG1lc3NhZ2U6CiAgICAgICAgICB0eXBlOiBzdHJpbmcKCmBgYAo= readmeEtag: '"0e69208d84c0f484a16f5058fd2579972aeea3c6"' readmeLastModified: Mon, 09 Nov 2020 08:24:57 GMT repositoryId: 310980518 description: >- Sample project to demonstrate OpenAPI Swagger configuration in Guice grizzly jersey example. created: '2020-11-08T03:51:02Z' updated: '2022-10-18T06:42:02Z' language: Java archived: false stars: 1 watchers: 1 forks: 3 owner: javabypatel logo: https://avatars.githubusercontent.com/u/16166145?v=4 license: MIT repoEtag: '"9c1eab9dca7aa98e7914270aa7f859ff7d48b3d68b8acdcdd4fe5779e891cc50"' repoLastModified: Tue, 18 Oct 2022 06:42:02 GMT foundInMaster: true category: Parsers id: fe1945f9e09307f9cd13e15ee4ba8aab - source: openapi3 tags repository: https://github.com/nekofar/nobitex-api-openapi-specs v3: true repositoryMetadata: base64Readme: >- WyFbR2l0SHViIFZlcnNpb25dW2ljb24tcmVsZWFzZV1dW2xpbmstcmVsZWFzZV0KWyFbV29ya2Zsb3cgU3RhdHVzXVtpY29uLXdvcmtmbG93XV1bbGluay13b3JrZmxvd10KWyFbTGljZW5zZV1baWNvbi1saWNlbnNlXV1bbGluay1saWNlbnNlXQpbIVtUd2l0dGVyOiBuZWtvZmFyXVtpY29uLXR3aXR0ZXJdXVtsaW5rLXR3aXR0ZXJdCgojIE5vYml0ZXggT3BlbiBBUEkgU3BlY2lmaWNhdGlvbgoKPiBUaGlzIGlzIGFuIHVub2ZmaWNpYWwgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiBmb3IgYE5vYml0ZXhgIGJhc2VkIG9uIHRoZSBbQVBJXShodHRwczovL2FwaWRvY3Mubm9iaXRleC5pcikgYW5kIFtQb3N0bWFuXShodHRwczovL2RvY3VtZW50ZXIuZ2V0cG9zdG1hbi5jb20vdmlldy81NzIyMTIyL1N6bWNheWp3P3ZlcnNpb249bGF0ZXN0KSBkb2N1bWVudGF0aW9uLgoKIyMgTGljZW5zZQoKVGhlIE1JVCBMaWNlbnNlIChNSVQpLiBQbGVhc2Ugc2VlIFtMaWNlbnNlIEZpbGVdKExJQ0VOU0UpIGZvciBtb3JlIGluZm9ybWF0aW9uLgoKLS0tCltpY29uLXR3aXR0ZXJdOiBodHRwczovL2ltZy5zaGllbGRzLmlvL3R3aXR0ZXIvZm9sbG93L25la29mYXIuc3ZnP3N0eWxlPWZsYXQKW2ljb24tcmVsZWFzZV06IGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3YvcmVsZWFzZS9uZWtvZmFyL25vYml0ZXgtYXBpLW9wZW5hcGktc3BlY3M/aW5jbHVkZV9wcmVyZWxlYXNlcwpbaWNvbi1saWNlbnNlXTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9uZWtvZmFyL25vYml0ZXgtYXBpLW9wZW5hcGktc3BlY3Muc3ZnCltpY29uLXdvcmtmbG93XTogaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvd29ya2Zsb3cvc3RhdHVzL25la29mYXIvbm9iaXRleC1hcGktb3BlbmFwaS1zcGVjcy9UZXN0cwoKW2xpbmstdHdpdHRlcl06IGh0dHBzOi8vdHdpdHRlci5jb20vbmVrb2ZhcgpbbGluay1yZWxlYXNlXTogaHR0cHM6Ly9naXRodWIuY29tL25la29mYXIvbm9iaXRleC1hcGktb3BlbmFwaS1zcGVjcy9yZWxlYXNlcwpbbGluay1saWNlbnNlXTogaHR0cHM6Ly9naXRodWIuY29tL25la29mYXIvbm9iaXRleC1hcGktb3BlbmFwaS1zcGVjcy9ibG9iL21hc3Rlci9MSUNFTlNFCltsaW5rLXdvcmtmbG93XTogaHR0cHM6Ly9naXRodWIuY29tL25la29mYXIvbm9iaXRleC1hcGktb3BlbmFwaS1zcGVjcy9hY3Rpb25zL3dvcmtmbG93cy90ZXN0cy55bWwKCg== readmeEtag: '"589e1fa523e1c05863ea4b6623ae2a142f80a60e"' readmeLastModified: Wed, 02 Mar 2022 18:33:08 GMT repositoryId: 409305298 description: Nobitex Open API Specification created: '2021-09-22T17:59:12Z' updated: '2023-03-17T12:44:30Z' language: null archived: true stars: 1 watchers: 1 forks: 0 owner: nekofar logo: https://avatars.githubusercontent.com/u/147401?v=4 license: MIT repoEtag: '"1931d4fe1bba2a1061ef8a0f25a2e84a25bb8f4e93ac63440afee4bddf6d4505"' repoLastModified: Fri, 17 Mar 2023 12:44:30 GMT foundInMaster: true category: SDK id: c5dbdb6639d1221cffb84e32986196a1 - source: openapi3 tags repository: https://github.com/gasparyanvazgen/math-cli-api-kit v3: true id: cb5edc4523bd77f232d9590e922fff1a repositoryMetadata: base64Readme: >- IyBNYXRoIENMSSBBUEkgS2l0CgpbIVtHaXRIdWIgTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9nYXNwYXJ5YW52YXpnZW4vbWF0aC1jbGktYXBpLWtpdC9ibG9iL21hc3Rlci9MSUNFTlNFKQpbIVtHaXRIdWIgSXNzdWVzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9pc3N1ZXMvZ2FzcGFyeWFudmF6Z2VuL21hdGgtY2xpLWFwaS1raXQpXShodHRwczovL2dpdGh1Yi5jb20vZ2FzcGFyeWFudmF6Z2VuL21hdGgtY2xpLWFwaS1raXQvaXNzdWVzKQpbIVtHaXRIdWIgU3RhcnNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3N0YXJzL2dhc3BhcnlhbnZhemdlbi9tYXRoLWNsaS1hcGkta2l0KV0oaHR0cHM6Ly9naXRodWIuY29tL2dhc3BhcnlhbnZhemdlbi9tYXRoLWNsaS1hcGkta2l0L3N0YXJnYXplcnMpClshW0dpdEh1YiBGb3Jrc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvZm9ya3MvZ2FzcGFyeWFudmF6Z2VuL21hdGgtY2xpLWFwaS1raXQpXShodHRwczovL2dpdGh1Yi5jb20vZ2FzcGFyeWFudmF6Z2VuL21hdGgtY2xpLWFwaS1raXQvbmV0d29yaykKClRoZSBNYXRoIENMSSBBUEkgS2l0IGlzIGEgUHl0aG9uIHBhY2thZ2UgYW5kIGNvbW1hbmQtbGluZSB0b29sIGZvciBwZXJmb3JtaW5nIGFsZ2VicmFpYyBhbmQgZ2VvbWV0cmljIG9wZXJhdGlvbnMuIEl0IHByb3ZpZGVzIGEgc2V0IG9mIG1hdGhlbWF0aWNhbCBmdW5jdGlvbnMgYW5kIGV4cG9zZXMgdGhlbSB0aHJvdWdoIGJvdGggYSBjb21tYW5kLWxpbmUgaW50ZXJmYWNlIGFuZCBhIFJFU1RmdWwgQVBJLgoKIVtNYXRoIENMSSBBUEkgS2l0IFN3YWdnZXIgVUldKC4vaW1hZ2VzL3N3YWdnZXItc2NyZWVuc2hvdC5wbmcpCgojIyBUYWJsZSBvZiBDb250ZW50cwoKLSBbSW5zdGFsbGF0aW9uXSgjaW5zdGFsbGF0aW9uKQotIFtVc2FnZV0oI3VzYWdlKQotIFtBUEkgRG9jdW1lbnRhdGlvbl0oI2FwaS1kb2N1bWVudGF0aW9uKQotIFtDb250cmlidXRpbmddKCNjb250cmlidXRpbmcpCi0gW0xpY2Vuc2VdKCNsaWNlbnNlKQoKIyMgSW5zdGFsbGF0aW9uCgpUaGUgTWF0aCBDTEkgQVBJIEtpdCBjYW4gYmUgaW5zdGFsbGVkIGRpcmVjdGx5IGZyb20gR2l0SHViIHVzaW5nIGBwaXBgOgoKYGBgYmFzaApwaXAgaW5zdGFsbCBnaXQraHR0cHM6Ly9naXRodWIuY29tL2dhc3BhcnlhbnZhemdlbi9tYXRoLWNsaS1hcGkta2l0LmdpdApgYGAKCkVuc3VyZSB0aGF0IGEgdmlydHVhbCBlbnZpcm9ubWVudCBpcyBjcmVhdGVkIGJlZm9yZSBpbnN0YWxsaW5nIHRvIG1hbmFnZSB5b3VyIHByb2plY3QncyBkZXBlbmRlbmNpZXMuCgojIyBVc2FnZQoKIyMjIENvbW1hbmQtTGluZSBJbnRlcmZhY2UKClRoZSBNYXRoIENMSSBBUEkgS2l0IG9mZmVycyBhIHJhbmdlIG9mIGFsZ2VicmFpYyBhbmQgZ2VvbWV0cmljIGNvbW1hbmRzIHZpYSB0aGUgY29tbWFuZC1saW5lIGludGVyZmFjZS4gT3BlcmF0aW9ucyBzdWNoIGFzIGFkZGl0aW9uLCBzdWJ0cmFjdGlvbiwgbXVsdGlwbGljYXRpb24sIGFuZCBtb3JlIGFyZSBzdXBwb3J0ZWQuCgpIZXJlIGFyZSBzb21lIGV4YW1wbGVzOgoKYGBgYmFzaApweXRob24gLW0gbWF0aF9jbGlfYXBpX2tpdC5jbGkgYWxnZWJyYSBzdW0gLXggNSAteSAzCnB5dGhvbiAtbSBtYXRoX2NsaV9hcGlfa2l0LmNsaSBnZW9tZXRyeSBzdXJmYWNlX29mX3NxdWFyZSAtYSA0CmBgYAoKQWRkaXRpb25hbCBkZXRhaWxzIG9uIHRoZSBhdmFpbGFibGUgY29tbWFuZHMgYW5kIHRoZWlyIHVzYWdlIGNhbiBiZSBmb3VuZCBieSBydW5uaW5nOgoKYGBgYmFzaApweXRob24gLW0gbWF0aF9jbGlfYXBpX2tpdC5jbGkgLS1oZWxwCmBgYAoKIyMjIEFQSQoKVGhlIE1hdGggQ0xJIEFQSSBLaXQgcHJvdmlkZXMgYSBSRVNUZnVsIEFQSSBmb3IgcHJvZ3JhbW1hdGljIGFjY2VzcyB0byBtYXRoZW1hdGljYWwgb3BlcmF0aW9ucy4gVG8gcnVuIHRoZSBBUEkgc2VydmVyLCB1c2UgdGhlIGZvbGxvd2luZyBzY3JpcHQ6CgpgYGBweXRob24KIyBzZXJ2ZXIucHkKCmZyb20gbWF0aF9jbGlfYXBpX2tpdC5hcGkgaW1wb3J0IGNyZWF0ZV9hcHAKCmFwcCA9IGNyZWF0ZV9hcHAoX19uYW1lX18pCgppZiBfX25hbWVfXyA9PSAiX19tYWluX18iOgogICAgYXBwLnJ1bihob3N0PSJsb2NhbGhvc3QiLCBwb3J0PTgwMDApCmBgYAoKVGhlIHNlcnZlciBjYW4gYmUgc3RhcnRlZCB3aXRoIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYGJhc2gKc2FuaWMgc2VydmVyOmFwcApgYGAKCkFsdGVybmF0aXZlbHksIHRoZSBzZXJ2ZXIgY2FuIGJlIHJ1biB1c2luZyBwbGFpbiBQeXRob246CgpgYGBiYXNoCnB5dGhvbiBzZXJ2ZXIucHkKYGBgCgpUaGlzIGFjdGlvbiBsYXVuY2hlcyB0aGUgc2VydmVyLCBtYWtpbmcgaXQgYWNjZXNzaWJsZSBhdCBodHRwOi8vbG9jYWxob3N0OjgwMDAvLiBBUEkgZW5kcG9pbnRzIHByb3ZpZGVkIGJ5IHRoZSBNYXRoIENMSSBBUEkgS2l0IGNhbiBiZSBhY2Nlc3NlZCB0aHJvdWdoIHRoaXMgc2VydmVyLiBJZiB0aGVyZSdzIGEgbmVlZCB0byBjaGFuZ2UgdGhlIHBvcnQgd2hlbiBydW5uaW5nIHRoZSBzZXJ2ZXIsIHlvdSBjYW4gdXNlIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYGJhc2gKc2FuaWMgc2VydmVyOmFwcCAtLXBvcnQ9ODAwNQpgYGAKCkhvd2V2ZXIsIHdoZW4gcnVubmluZyBpdCB1c2luZyBgcHl0aG9uIHNlcnZlci5weWAsIHRoZSBzZXJ2ZXIgb3BlcmF0ZXMgb24gcG9ydCA4MDA1LCBhcyBzcGVjaWZpZWQgaW4gdGhlIGNvZGUgZ2l2ZW4gaW4gYHNlcnZlci5weWA6CgpgYGBweXRob24KIyB0aGUgcmVzdCBvZiB0aGUgY29kZQoKaWYgX19uYW1lX18gPT0gIl9fbWFpbl9fIjoKICAgIGFwcC5ydW4oaG9zdD0ibG9jYWxob3N0IiwgcG9ydD04MDA1KQpgYGAKCllvdSBjYW4gYWNjZXNzIHRoZSBTd2FnZ2VyIFVJIGZvciB0aGUgQVBJIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODAwNS9zd2FnZ2VyLwoKIyMgQVBJIERvY3VtZW50YXRpb24KCkZvciBkZXRhaWxlZCBkb2N1bWVudGF0aW9uIG9mIHRoZSBBUEkgZW5kcG9pbnRzLCBwbGVhc2UgY29uc3VsdCB0aGUgW29mZmljaWFsIE9wZW5BUEkgZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL2dhc3BhcnlhbnZhemdlbi9tYXRoLWNsaS1hcGkta2l0L2Jsb2IvbWFzdGVyL0FQSV9ET0MubWQpLgoKIyMgQ29udHJpYnV0aW5nCgpDb250cmlidXRpb25zIHRvIHRoZSBNYXRoIENMSSBBUEkgS2l0IGFyZSB3ZWxjb21lLiBUbyBjb250cmlidXRlIG9yIHJlcG9ydCBhbiBpc3N1ZSwgcGxlYXNlIHJlZmVyIHRvIHRoZSBbY29udHJpYnV0aW5nIGd1aWRlbGluZXNdKENPTlRSSUJVVElORy5tZCkuCgojIyBMaWNlbnNlCgpUaGUgTWF0aCBDTEkgQVBJIEtpdCBpcyBhbiBvcGVuLXNvdXJjZSBwcm9qZWN0IHJlbGVhc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gRm9yIG1vcmUgZGV0YWlscywgcGxlYXNlIHJldmlldyB0aGUgW0xJQ0VOU0VdKGh0dHBzOi8vZ2l0aHViLmNvbS9nYXNwYXJ5YW52YXpnZW4vbWF0aC1jbGktYXBpLWtpdC9ibG9iL21hc3Rlci9MSUNFTlNFKS4KCi0tLQoKRm9yIGlucXVpcmllcyBvciBhc3Npc3RhbmNlLCBwbGVhc2UgZmVlbCBmcmVlIHRvIGNvbnRhY3QgdGhlIHByb2plY3Qgb3duZXIsIFZhemdlbiBHYXNwYXJ5YW4sIGF0IFtndmF6Z2VuQG91dGxvb2suY29tXShtYWlsdG86Z3ZhemdlbkBvdXRsb29rLmNvbSkuCgoqKlRoYW5rIHlvdSBmb3IgdXNpbmcgdGhlIE1hdGggQ0xJIEFQSSBLaXQhKioK readmeEtag: '"4649905cbf6377b9551ee8b10b93bb4edccbc2a1"' readmeLastModified: Sat, 21 Oct 2023 08:19:34 GMT repositoryId: 707896579 description: >- A versatile Python command-line interface and API for algebraic and geometric operations, streamlining math-related tasks for applications and developers. created: '2023-10-20T23:20:22Z' updated: '2024-01-27T16:41:04Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: gasparyanvazgen logo: https://avatars.githubusercontent.com/u/89462790?v=4 license: MIT repoEtag: '"be6313797f7712f207d3cb3982004e1eb2c74d58b658edac11ef4c8b00a9ebb4"' repoLastModified: Sat, 27 Jan 2024 16:41:04 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/riptl/nimiq-watch-docs v3: true id: a86198fc049b7191ad08b2ecf2d0e8ca repositoryMetadata: base64Readme: >- IyBOaW1pcS5XYXRjaCBPcGVuQVBJIFNwZWNpZmljYXRpb24KWyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvdGVyb3JpZS9uaW1pcS13YXRjaC1kb2NzLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL3Rlcm9yaWUvbmltaXEtd2F0Y2gtZG9jcykKClthcGkubmltaXEud2F0Y2hdKGh0dHBzOi8vYXBpLm5pbWlxLndhdGNoLykgaXMgcG93ZXJpbmcgW3RoZSBibG9ja2NoYWluIGV4cGxvcmVyXShodHRwczovL25pbWlxLndhdGNoLykKb2YgdGhlIG5ldyBfX05pbWlxIGJsb2NrY2hhaW5fXyBwcm9qZWN0LgoKVG8gbGVhcm4gbW9yZSBhYm91dCB0aGUgTmltaXEgYmxvY2tjaGFpbiwKdmlzaXQgdGhlIFtob21lcGFnZV0oaHR0cHM6Ly9uaW1pcS5jb20pLApjaGVjayBvdXQgQG5pbWlxIG9uIFtUd2l0dGVyXShodHRwczovL3R3aXR0ZXIuY29tL25pbWlxKQpvciBbTWVkaXVtXShodHRwczovL21lZGl1bS5jb20vbmltaXEtbmV0d29yaykKYW5kIGpvaW4gdGhlIFtEaXNjb3JkXShodHRwczovL2Rpc2NvcmQuZ2cvY01IZW1nOCkKYW5kIFtUZWxlZ3JhbV0oaHR0cHM6Ly90Lm1lL2pvaW5jaGF0L0FBQUFBRUpXLW96RndvN0VyOWpwSHcpIGNoYW5uZWxzLgoKVG8gdXNlIHRoaXMgQVBJIGZvciB5b3VyIHByb2plY3QsIGNvbnRhY3QgX19AU29lcmVuIzg5OThfXyBpbiBOaW1pcSdzIERpc2NvcmQgb3Igd3JpdGUgdG8gaGVsbG9Ac29lcmVuc2Nod2VydC5kZQoKIyMgTGlua3MKCi0gW05pbWlxLldhdGNoXShodHRwczovL25pbWlxLndhdGNoKQotIFtOaW1pcS5XYXRjaCBGcm9udGVuZCBDb2RlXShodHRwczovL2dpdGh1Yi5jb20vc2lzb3UvbmltaXEtd2F0Y2gpCi0gW05pbWlxIENvcmUvSlNdKGh0dHBzOi8vZ2l0aHViLmNvbS9uaW1pcS1uZXR3b3JrL2NvcmUpCi0gW05pbWlxIENvcmUvUnVzdF0oaHR0cHM6Ly9naXRodWIuY29tL25pbWlxL2NvcmUtcnMpCi0gW2FwaS5uaW1pcS53YXRjaCBSZURvY10oaHR0cHM6Ly90ZXJvcmllLmdpdGh1Yi5pby9uaW1pcS13YXRjaC1kb2NzLykKLSBbU3dhZ2dlclVJXShodHRwczovL3Rlcm9yaWUuZ2l0aHViLmlvL25pbWlxLXdhdGNoLWRvY3Mvc3dhZ2dlci11aS8pCi0gT3BlbkFQSSBSYXcgRmlsZXM6IFtKU09OXShodHRwczovL3Rlcm9yaWUuZ2l0aHViLmlvL25pbWlxLXdhdGNoLWRvY3Mvb3BlbmFwaS5qc29uKSBbWUFNTF0oaHR0cHM6Ly90ZXJvcmllLmdpdGh1Yi5pby9uaW1pcS13YXRjaC1kb2NzL29wZW5hcGkueWFtbCkKCl9fVE9ETzpfXyBBZGQgQVBJIHRvIFtBUElzLmd1cnVdKGh0dHBzOi8vQVBJcy5ndXJ1KSBkaXJlY3RvcnkgdXNpbmcgW3RoaXMgZm9ybV0oaHR0cHM6Ly9hcGlzLmd1cnUvYWRkLWFwaS8pLgo= readmeEtag: '"2d4c6cc85508d10b2ab131ef997292f1c8c91eba"' readmeLastModified: Fri, 22 Feb 2019 02:02:34 GMT repositoryId: 171151671 description: Nimiq.Watch API Documentation created: '2019-02-17T17:38:13Z' updated: '2019-02-22T02:02:40Z' language: HTML archived: false stars: 1 watchers: 1 forks: 0 owner: riptl logo: https://avatars.githubusercontent.com/u/21371810?v=4 license: MIT repoEtag: '"a3a9c499d07f74e0525b981489c0a73ce80d269863638212c1d9ebfcd8bbd768"' repoLastModified: Fri, 22 Feb 2019 02:02:40 GMT category: SDK oldLocations: - https://github.com/terorie/nimiq-watch-docs foundInMaster: true - source: openapi3 tags repository: https://github.com/stariongroup/openapi v3: true id: 205120cda96c57ae4a0aa4bd246f507c repositoryMetadata: base64Readme: >- IyBSSEVBLk9wZW5BcGkKClRoZSBSSEVBLk9wZW5BcGkgbGlicmFyeSBpcyBhIG5haXZlIEpTT04gcmVhZGVyIGltcGxlbWVudGF0aW9uIG9mIHRoZSBbT3BlbkFQSSBTcGVjaWZpY2F0aW9uIHYzLjEuMF0oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvbGF0ZXN0Lmh0bWwjdmVyc2lvbi0zLTEtMCkuIFRoZSBwdXJwb3NlIG9mIHRoaXMgbGlicmFyeSBpcyB0byBzdXBvcnQgY29kZS1nZW5lcmF0aW9uIGluIHRoZSAuTkVUIGVjb3N5c3RlbS4gVGhlIGxpYnJhcnkgd2lsbCBtb3N0IGxpa2VseSBiZSBkZXByZWNhdGVkIG9uY2UgdGhlIFttaWNyb3NvZnQvT3BlbkFQSS5ORVRdKGh0dHBzOi8vZ2l0aHViLmNvbS9taWNyb3NvZnQvT3BlbkFQSS5ORVQpIHN1cHBvcnRzIHZlcnNpb24gMy4xLiBTdXBwb3J0IGZvciB2ZXJzaW9uIDMuMSBpbiBfbWljcm9zb2Z0L09wZW5BUEkuTkVUXyBpcyBiZWluZyB0cmFja2VkIGluIGlzc3VlIGh0dHBzOi8vZ2l0aHViLmNvbS9taWNyb3NvZnQvT3BlbkFQSS5ORVQvaXNzdWVzLzc5NQoKWyFbUXVhbGl0eSBHYXRlIFN0YXR1c10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9UkhFQUdST1VQX09QRU5BUEkmbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPVJIRUFHUk9VUF9PUEVOQVBJKQpbIVtDb2RlIFNtZWxsc10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9UkhFQUdST1VQX09QRU5BUEkmbWV0cmljPWNvZGVfc21lbGxzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9UkhFQUdST1VQX09QRU5BUEkpClshW0NvdmVyYWdlXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1SSEVBR1JPVVBfT1BFTkFQSSZtZXRyaWM9Y292ZXJhZ2UpXShodHRwczovL3NvbmFyY2xvdWQuaW8vc3VtbWFyeS9uZXdfY29kZT9pZD1SSEVBR1JPVVBfT1BFTkFQSSkKWyFbRHVwbGljYXRlZCBMaW5lcyAoJSldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PVJIRUFHUk9VUF9PUEVOQVBJJm1ldHJpYz1kdXBsaWNhdGVkX2xpbmVzX2RlbnNpdHkpXShodHRwczovL3NvbmFyY2xvdWQuaW8vc3VtbWFyeS9uZXdfY29kZT9pZD1SSEVBR1JPVVBfT1BFTkFQSSkKWyFbTGluZXMgb2YgQ29kZV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9UkhFQUdST1VQX09QRU5BUEkmbWV0cmljPW5jbG9jKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9UkhFQUdST1VQX09QRU5BUEkpClshW01haW50YWluYWJpbGl0eSBSYXRpbmddKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PVJIRUFHUk9VUF9PUEVOQVBJJm1ldHJpYz1zcWFsZV9yYXRpbmcpXShodHRwczovL3NvbmFyY2xvdWQuaW8vc3VtbWFyeS9uZXdfY29kZT9pZD1SSEVBR1JPVVBfT1BFTkFQSSkKWyFbUmVsaWFiaWxpdHkgUmF0aW5nXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1SSEVBR1JPVVBfT1BFTkFQSSZtZXRyaWM9cmVsaWFiaWxpdHlfcmF0aW5nKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9UkhFQUdST1VQX09QRU5BUEkpClshW1NlY3VyaXR5IFJhdGluZ10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9UkhFQUdST1VQX09QRU5BUEkmbWV0cmljPXNlY3VyaXR5X3JhdGluZyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPVJIRUFHUk9VUF9PUEVOQVBJKQpbIVtUZWNobmljYWwgRGVidF0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9UkhFQUdST1VQX09QRU5BUEkmbWV0cmljPXNxYWxlX2luZGV4KV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL3N1bW1hcnkvbmV3X2NvZGU/aWQ9UkhFQUdST1VQX09QRU5BUEkpClshW1Z1bG5lcmFiaWxpdGllc10oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9UkhFQUdST1VQX09QRU5BUEkmbWV0cmljPXZ1bG5lcmFiaWxpdGllcyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPVJIRUFHUk9VUF9PUEVOQVBJKQoKIyMgSW5zdGFsbGF0aW9uCgpUaGUgcGFja2FnZSBpcyBhdmFpbGFibGUgb24gTnVnZXQgYXQ6Cgpwcm9qZWN0ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgTnVnZXQKLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18IC0tLS0tLS0tLS0tLQpbUkhFQS5PcGVuQXBpXShodHRwczovL3d3dy5udWdldC5vcmcvcGFja2FnZXMvUkhFQS5PcGVuQXBpKSAgIHwgWyFbTnVHZXQgQmFkZ2VdKGh0dHBzOi8vYnVpbGRzdGF0cy5pbmZvL251Z2V0L1JIRUEuT3BlbkFwaSldKGh0dHBzOi8vYnVpbGRzdGF0cy5pbmZvL251Z2V0L1JIRUEuT3BlbkFwaSkKCiMjIEJ1aWxkIFN0YXR1cwoKR2l0SHViIGFjdGlvbnMgYXJlIHVzZWQgdG8gYnVpbGQgYW5kIHRlc3QgdGhlIGxpYnJhcnkKCkJyYW5jaCAgICAgIHwgQnVpbGQgU3RhdHVzCi0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0KTWFzdGVyICAgICAgfCAhW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9naXRodWIuY29tL1JIRUFHUk9VUC9PUEVOQVBJL2FjdGlvbnMvd29ya2Zsb3dzL0NvZGVRdWFsaXR5LnltbC9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlcikKRGV2ZWxvcG1lbnQgfCAhW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9naXRodWIuY29tL1JIRUFHUk9VUC9PUEVOQVBJL2FjdGlvbnMvd29ya2Zsb3dzL0NvZGVRdWFsaXR5LnltbC9iYWRnZS5zdmc/YnJhbmNoPWRldmVsb3BtZW50KQoKIyBMaWNlbnNlCgpUaGUgUkhFQS5PcGVuQXBpIGxpYnJhcnkgaXMgcHJvdmlkZWQgdG8gdGhlIGNvbW11bml0eSB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UgMi4wLgoKIyBDb250cmlidXRpb25zCgpDb250cmlidXRpb25zIHRvIHRoZSBjb2RlLWJhc2UgYXJlIHdlbGNvbWUuIEhvd2V2ZXIsIGJlZm9yZSB3ZSBjYW4gYWNjZXB0IHlvdXIgY29udHJpYnV0aW9ucyB3ZSBhc2sgYW55IGNvbnRyaWJ1dG9yIHRvIHNpZ24gdGhlIENvbnRyaWJ1dG9yIExpY2Vuc2UgQWdyZWVtZW50IChDTEEpIGFuZCBzZW5kIHRoaXMgZGlnaXRhbHkgc2lnbmVkIHRvIHMuZ2VyZW5lQHJoZWFncm91cC5jb20uIFlvdSBjYW4gZmluZCB0aGUgQ0xBJ3MgaW4gdGhlIENMQSBmb2xkZXIuCg== readmeEtag: '"7660bd5c63950998bd3d9cfc7e1f68f40c5f7610"' readmeLastModified: Sun, 30 Apr 2023 09:54:40 GMT repositoryId: 622194609 description: >- a naive openapi 3.1 json document reader to support convention & template based code generation created: '2023-04-01T12:02:34Z' updated: '2023-04-28T08:13:22Z' language: C# archived: false stars: 1 watchers: 2 forks: 0 owner: STARIONGROUP logo: https://avatars.githubusercontent.com/u/22165742?v=4 license: Apache-2.0 repoEtag: '"3da66e76089979c4b3ade19201a364fe2992e124e88fb465c1b7bd34aec493a0"' repoLastModified: Fri, 28 Apr 2023 08:13:22 GMT category: SDK foundInMaster: true oldLocations: - https://github.com/rheagroup/openapi - source: openapi3 tags repository: https://github.com/iapost/oaql2 v3: true id: b2cedb58c85c3974274b29d193441356 repositoryMetadata: base64Readme: >- T3BlbkFQSSBRdWVyeSBMYW5ndWFnZSAyIFNlcnZpY2UKPT09PT09PT09PT09PT09PT09PT09PT09PT09PT09PT0KT3BlbkFQSSBRdWVyeSBMYW5ndWFnZSAyIChPQVFMMikgaXMgYSBxdWVyeSBsYW5ndWFnZSBmb3IgT3BlbkFQSSBkb2N1bWVudHMuIE9wZW5BUEkgaXMgYSBzdGFuZGFyZCBmb3JtYXQgZm9yIHRoZSBkZXNjcmlwdGlvbiBvZiBSRVNUZnVsIHNlcnZpY2VzLCBiYXNlZCBvbiBKU09OLiBPQVFMMiBpcyBkZXNpZ25lZCB3aXRoIHN5bnRheCBzaW1pbGFyIHRvIFNRTCBhbmQgc3VwcG9ydHMgcXVlcnlpbmcgbW9zdCBvZiB0aGUgZmllbGRzIGluIGFuIE9wZW5BUEkgZG9jdW1lbnQsIGFzIHdlbGwgYXMgdGhlIHNlbWFudGljIGFubm90YXRpb25zIHByb3Bvc2VkIGZvciBPcGVuQVBJLiBUaGlzIGlzIGFuIGltcGxlbWVudGF0aW9uIG9mIGEgd2ViIHNlcnZpY2UgY2FwYWJsZSBvZiBleGVjdXRpbmcgT0FRTDIgcXVlcmllcy4gVGhpcyBzZXJ2aWNlIHN0b3JlcyBtZXRhZGF0YSBmb3IgZWFjaCBPcGVuQVBJIGRlc2NyaXB0aW9uIGFuZCBleGVjdXRlcyB0aGUgcXVlcmllcyBvbiB0aGVtLiBJdCBidWlsZHMgaW5kZXhlcyB0byBzcGVlZCB1cCBxdWVyaWVzLCBjYW4gaGFuZGxlIGNvbXBvc2l0ZSBzY2hlbWEgb2JqZWN0cyBhbmQgdXNlcyByZWFzb25pbmcgdG8gc3VwcG9ydCBzZWFyY2hpbmcgaW4gYSBzZW1hbnRpYyBtb2RlbC4gVGhlIHdlYiBzZXJ2aWNlIGNvbnNpc3RzIG9mIGEgSmF2YSBzZXJ2ZXIgYW5kIGEgTW9uZ29EQiBkYXRhYmFzZSwgYm90aCBydW5uaW5nIGluc2lkZSBEb2NrZXIgY29udGFpbmVycy4KCiMjIEluc3RhbGxhdGlvbgpZb3UgbmVlZCB0byBoYXZlIERvY2tlciBhbmQgRG9ja2VyIENvbXBvc2UgaW5zdGFsbGVkLgoKRXhlY3V0ZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQgaW5zaWRlIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGUgcHJvamVjdDoKCiAgICAkIGRvY2tlci1jb21wb3NlIHVwIC0tYnVpbGQKICAgIApPbmNlIHRoZSBzZXJ2ZXIgaXMgcmVhZHksIGl0IHdpbGwgb3V0cHV0IHRoZSBtZXNzYWdlICJTZXJ2ZXIgc3RhcnRlZCIuCgojIyBVc2FnZQpUaGUgc2VydmVyIGxpc3RlbnMgZm9yIEhUVFAgcmVxdWVzdHMgb24gcG9ydCA4MCBvZiB0aGUgaG9zdCBtYWNoaW5lLiBZb3UgY2FuIGNvbm5lY3QgdG8gaXQgd2l0aCBhbiBpbnRlcm5ldCBicm93c2VyIChlZy4gRmlyZWZveCkgZm9yIGEgZ3JhcGhpY2FsIHVzZXIgaW50ZXJmYWNlLiBBIGxpc3Qgb2YgYWxsIGF2YWlsYWJsZSBlbmRwb2ludHMgaXMgc2hvd24gYmVsb3c6Cgp8IFBhdGggICAgICAgICAgICAgICAgfCBBY2NlcHRzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IFJldHVybnMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfC0tLS0tLS0tLS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfC0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS18CnwgLyAgICAgICAgICAgICAgICAgICB8IEdFVCBSZXF1ZXN0ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgSFRNTCBkb2N1bWVudCBwcm92aWRpbmcgYSBHVUkgZm9yIGluc2VydGluZyBvciByZXRyaWV2aW5nIE9wZW5BUEkgZGVzY3JpcHRpb25zIGFuZCBleGVjdXRpbmcgT0FRTDIgcXVlcmllcyAgICAgICAgICAgIHwKfCAvaW5zZXJ0RGVzY3JpcHRpb24gIHwgUE9TVCByZXF1ZXN0LjxiciAvPiBSZXF1ZXN0IGJvZHkgbXVzdCBiZSBhIHZhbGlkIE9wZW5BUEkgZGVzY3JpcHRpb24gJiAyMDQgY29kZSB3aXRoIG5vIHJlc3BvbnNlIGJvZHkgfCAyMDQgY29kZSB3aXRoIG5vIHJlc3BvbnNlIGJvZHkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgL3F1ZXJ5ICAgICAgICAgICAgICB8IFBPU1QgcmVxdWVzdC48YnIgLz4gUmVxdWVzdCBib2R5IG11c3QgYmUgYSB2YWxpZCBPQVFMMiBxdWVyeSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgMjAwIGNvZGUgd2l0aCB0aGUgcmVzdWx0cyBvZiB0aGUgcXVlcnkgaW4gdGhlIHJlc3BvbnNlIGJvZHkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC9kZXNjcmlwdGlvbi9cPGlkXD4gfCBHRVQgcmVxdWVzdC48YnIgLz4gXDxpZFw+IG11c3QgYmUgMjQgY2hhcmFjdGVycyBsb25nICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8IDIwMCBjb2RlIHdpdGggdGhlIHJlcXVlc3RlZCBPcGVuQVBJIGRlc2NyaXB0aW9uIGluIHRoZSByZXNwb25zZSBib2R5IG9yIDQwNCBjb2RlIGlmIHRoZXJlIGlzIG5vIE9wZW5BUEkgZGVzY3JpcHRpb24gd2l0aCB0aGF0IGlkIHwKCklmIHRoZSBzZXJ2ZXIgZW5jb3VudGVycyBhbiBlcnJvciwgaXQgd2lsbCByZXNwb25kIHdpdGggYSBzdGF0dXMgY29kZSBvZiA0MDAgYW5kIGFuIGVycm9yIG1lc3NhZ2UgaW4gdGhlIHJlc3BvbnNlIGJvZHkuIAoKIyMgRGF0YWJhc2Ugb2YgT3BlbkFQSSBkb2N1bWVudHMKRm9yIGNvbnZlbmllbmNlLCB0aGUgW2BkYXRhYmFzZWBdKGRhdGFiYXNlKSBkaXJlY3RvcnkgY29udGFpbnMgMTAwMDAgT3BlbkFQSSBkZXNjcmlwdGlvbnMsIHRha2VuIGZyb20gU3dhZ2dlcmh1YiwgaW4gY29tcHJlc3NlZCBmb3JtIChzcGxpdCBpbiB0d28gcGFydHMgZHVlIHRvIEdpdGh1YidzIGZpbGUgc2l6ZSByZXN0cmljdGlvbnMpLiBJdCBhbHNvIGNvbnRhaW5zIGEgYmFzaCBzY3JpcHQgdG8gYXV0b21hdGljYWxseSBzZW5kIHRoZXNlIGRvY3VtZW50cyB0byB0aGUgc2VydmljZS4gVG8gdXNlIHRoZSBzY3JpcHQsIGV4ZWN1dGUgdGhlIGZvbGxvd2luZyBjb21tYW5kIGluc2lkZSBbYGRhdGFiYXNlYF0oZGF0YWJhc2UpOgoKICAgICQgLi9pbnNlcnQuc2ggPGhvc3RuYW1lPiAKClw8aG9zdG5hbWVcPiBzaG91bGQgYmUgdGhlIGFkZHJlc3Mgb2YgdGhlIHNlcnZpY2UuIE5vdGUgdGhhdCB5b3UgbmVlZCB0byBoYXZlIHRoZSAqY3VybCogcGFja2FnZSBpbnN0YWxsZWQuCgojIyBMaWNlbnNlCkRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBHUEwtMy4wIExpY2Vuc2UuIFNlZSBbYExJQ0VOU0VgXShMSUNFTlNFKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCiMjIE5vdGVzCi0gQ3VycmVudGx5LCBvbmx5IE9wZW5BUEkgZG9jdW1lbnRzIGNvbmZvcm1pbmcgdG8gT3BlbkFQSSBTcGVjaWZpY2F0aW9uIHYzLjEuMCBhcmUgc3VwcG9ydGVkCgojIyBSZWZlcmVuY2VzCi0gSS4gQXBvc3RvbGFraXMsIE4uIE1haW5hcyBhbmQgRS5HLk0uIFBldHJha2lzLCAiU2ltcGxlIHF1ZXJ5aW5nIHNlcnZpY2UgZm9yIE9wZW5BUEkgZGVzY3JpcHRpb25zIHdpdGggc2VtYW50aWMgZXh0ZW5zaW9ucyIsICpJbmZvcm1hdGlvbiBTeXN0ZW1zKiAxMTcgKDIwMjMpLCAxMDIyNDEsIGh0dHBzOi8vZG9pLm9yZy8xMC4xMDE2L2ouaXMuMjAyMy4xMDIyNDEKLSBJLiBBcG9zdG9sYWtpcywgIlNpbXBsZSBxdWVyeWluZyBzZXJ2aWNlIGZvciBPcGVuQVBJIGRlc2NyaXB0aW9ucyB3aXRoIFNlbWFudGljIFdlYiBleHRlbnNpb25zIiwgRGlwbG9tYSB0aGVzaXMsIFNjaG9vbCBvZiBFbGVjdHJpY2FsIGFuZCBDb21wdXRlciBFbmdpbmVlcmluZywgVGVjaG5pY2FsIFVuaXZlcnNpdHkgb2YgQ3JldGUgKFRVQyksIENoYW5pYSwgQ3JldGUsIEdyZWVjZSAoTWF5IDIwMjIpLCBodHRwczovL2RpYXMubGlicmFyeS50dWMuZ3Ivdmlldy85MjEyMwo= readmeEtag: '"492b5bac3876a384296519419248821e975f477d"' readmeLastModified: Thu, 22 Jun 2023 08:01:17 GMT repositoryId: 480846077 description: A web service to easily query OpenAPI documents created: '2022-04-12T14:29:53Z' updated: '2024-10-12T18:37:35Z' language: Java archived: false stars: 1 watchers: 2 forks: 0 owner: iapost logo: https://avatars.githubusercontent.com/u/81591846?v=4 license: GPL-3.0 repoEtag: '"8ad0a17cae648f3f0836153721cabca5ac800267923a6570cbe1be2e31c2d77e"' repoLastModified: Sat, 12 Oct 2024 18:37:35 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/itsvigneshmurugan/authflow-go-react-jwt-openapi v3: true id: 5742ec0710e55c9b5ff9420276d8382b repositoryMetadata: base64Readme: >- IyBhdXRoZmxvdwoKUmVmZXIgdG8gW0xpbmtlZEluXShodHRwczovL3d3dy5saW5rZWRpbi5jb20vcHVsc2Uvand0LWF1dGhlbnRpY2F0aW9uLXVzaW5nLWdvbGFuZy1yZWFjdC12aWduZXNoLW11cnVnYW4tZWpqYmMpIGFydGljbGUgZm9yIGRvY3VtZW50YXRpb24uCg== readmeEtag: '"96e50c1cc298fda05ae21394ef199714bc7338ca"' readmeLastModified: Wed, 22 Nov 2023 09:20:03 GMT repositoryId: 704745342 description: Building authentication flow. created: '2023-10-14T01:12:52Z' updated: '2024-06-19T23:14:34Z' language: JavaScript archived: false stars: 1 watchers: 1 forks: 0 owner: ItsVigneshMurugan logo: https://avatars.githubusercontent.com/u/30820063?v=4 repoEtag: '"7d78f859922171587477b3de0529cfc0af0c157702428a9194f8f1b909e7b9f2"' repoLastModified: Wed, 19 Jun 2024 23:14:34 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/big-vi/authflow-go-react-jwt-openapi - source: openapi3 tags repository: https://github.com/adolligit/car-shop v3: true id: 1419b8b6544c12a49e08c6805202cf77 repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPkNhciBTaG9wPC9oMT4KCkNhciBTaG9wIMOpIHVtYSAqQVBJIFJlc3QqIHF1ZSBjb25zdWx0YSBvIGJhbmNvIGRlIGRhZG9zIG7Do28gcmVsYWNpb25hbCBNb25nb0RCIGF0cmF2w6lzIGRvICpPRE0qIE1vbmdvb3NlLgoKRXN0YSBBUEkgw6kgdW0gKkNSVUQqIGZlaXRvIGNvbSBvcyBwcmluY8OtcGlvcyBkYSAqUHJvZ3JhbWHDp8OjbyBPcmllbnRhZGEgw6EgT2JqZXRvcyAoUE9PKSogZSBzZXUgcHJvcMOzc2l0byDDqSBzaW11bGFyIG8gZ2VyZW5jaWFtZW50byBkZSB1bSBzaXN0ZW1hIGRlIHVtYSBjb25jZXNzaW9uw6FyaWEgZGUgdmXDrWN1bG9zLgoKRXN0ZSBwcm9qZXRvIGZvaSBkZXNlbnZvbHZpZG8gZW0gKipOb2RlLmpzICsgRXhwcmVzcy5qcyArIFR5cGVTY3JpcHQqKiwgdXRpbGl6YW5kbyAqKk1vbmdvb3NlLmpzKiogcGFyYSBjb211bmljYcOnw6NvIGNvbSBvIGJhbmNvIGRlIGRhZG9zICoqTW9uZ29EQioqLiBBbMOpbSBkaXNzbywgZm9yYW0gcmVhbGl6YWRvcyB0ZXN0ZXMgbmFzIGNhbWFkYXMgKk1vZGVsLCBDb250cm9sbGVyIGUgU2VydmljZSogY29tICoqTW9jaGEgKyBDaGFpICsgU2lub24qKi4KCiMjIENvbW8gZXUgZmHDp28gcGFyYSBleGVjdXRhciBlc3RlIHByb2pldG8/ClByaW1laXJvIGRlIHR1ZG8sIGNsb25lIG8gcHJvamV0byBuYSBzdWEgbcOhcXVpbmEgbG9jYWwgZSBlbnRyZSBuYSBwYXN0YSBkbyBwcm9qZXRvOgpgYGBiYXNoCmdpdCBjbG9uZSBnaXRAZ2l0aHViLmNvbTpBZG9sbGlnaXQvY2FyLXNob3AuZ2l0ICYmIGNkIC4vY2FyLXNob3AKYGBgCkRlcG9pcyBpbnN0YWxlIGFzIGRlcGVuZMOqbmNpYXM6CmBgYGJhc2gKbnBtIGluc3RhbGwKYGBgCiMjIyBJbnN0YWxhw6fDo28K8J+aqCBBcyB2ZXJzw7VlcyBkZXNjcml0YXMgbmFzIGluc3RhbGHDp8O1ZXMgc8OjbyAqKmZvcnRlbWVudGUgcmVjb21lbmRhZGFzKiouIE5vIGVudGFudG8sIGNhc28gcXVlaXJhIHRlbnRhciBlbSB1bWEgdmVyc8OjbyBkaWZlcmVudGUsIGNvbnNpZGVyZSBhIHBvc3PDrXZlbCBhcGFyacOnw6NvIGRlIGVycm9zIGR1cmFudGUgYSBleGVjdcOnw6NvIGRvIHByb2pldG8uCgpFc2NvbGhhIGFiYWl4byB1bSB0aXBvIGRlIGluc3RhbGHDp8OjbzoKPGRldGFpbHM+CiAgPHN1bW1hcnk+SW5zdGFsYXIgY29tIERvY2tlciDwn5CzPC9zdW1tYXJ5Pgo8L2JyPjxiPlJlcXVpc2l0b3M8L2I+CjwvYnI+Vm9jw6ogZGV2ZSBwb3NzdWlyIGFzIHNlZ3VpbnRlcyBmZXJyYW1lbnRhczoKPHVsPgogICAgPGxpPkRvY2tlcjogdjI0LjAuMjwvbGk+CiAgICA8bGk+RG9ja2VyIENvbXBvc2U6IHYyLjE5LjA8L2xpPgo8L3VsPgoKICDimqDvuI8gU2Ugdm9jw6ogZXN0aXZlciB1c2FuZG8gKipMaW51eCoqLCBkZXNhdGl2ZSBvICoqTW9uZ29EQioqIGxvY2FsbWVudGUgcGFyYSBldml0YXIgY29uZmxpdG9zIGRlIHBvcnRhOgogIGBgYGJhc2gKICBzeXN0ZW1jdGwgc3RvcCBtb25nb2QKICBgYGAKCjxocj4KCiAgMS4gQ3JpZSBlIGluaWNpZSBvcyBjb250w6ppbmVyczoKICBgYGBiYXNoCiAgZG9ja2VyIGNvbXBvc2UgdXAgLWQKICBgYGAKICAyLiBFbnRyZSBubyBjb250w6ppbmVyIGRhIGFwbGljYcOnw6NvOgogIGBgYGJhc2gKICBkb2NrZXIgZXhlYyAtaXQgY2FyX3Nob3AgYmFzaAogIGBgYAogIC0tLQo8L2RldGFpbHM+Cgo8ZGV0YWlscz4KICA8c3VtbWFyeT5JbnN0YWxhciBsb2NhbG1lbnRlIPCfkrs8L3N1bW1hcnk+CjwvYnI+PGI+UmVxdWlzaXRvczwvYj4KPC9icj5Wb2PDqiBkZXZlIHBvc3N1aXIgYXMgc2VndWludGVzIGZlcnJhbWVudGFzOgo8dWw+CiAgICA8bGk+bnBtOiB2OS42Ljc8L2xpPgogICAgPGxpPk5vZGU6IHYyMC4zPC9saT4KICAgIDxsaT5Nb25nb0RCOiB2Ni4wPC9saT4KPC91bD4KClNlIHZvY8OqIHVzYSAqKkxpbnV4KiosIHZlcmlmaXF1ZSBzZSBvIHNlcnZpw6dvIGRvICoqTW9uZ29EQioqIGVzdGEgYXRpdm86IApgYGBiYXNoCnN5c3RlbWN0bCBzdGF0dXMgbW9uZ29kCmBgYApDYXNvIGVzdGVqYSBkZXNhdGl2YWRvLCBzdWJzdGl0dWEgKipzdGF0dXMqKiBwb3IgKipzdGFydCoqIG5vIGNvbWFuZG8gYWNpbWEuCjxocj4KCjEuIENyaWUgbyBhcnF1aXZvIDxiPi5lbnY8L2I+IG5hIHBhc3RhIHJhaXoKYGBgYmFzaAp0b3VjaCAuZW52CmBgYAoKMi4gQ29waWUgZSBjb2xlIGFzIGluZm9ybWHDp8O1ZXMgYSBzZWd1aXI6CgpgYGBiYXNoCiMgTW9uZ29EQgpNREJfVVJJPQpNREJfSE9TVD0xMjcuMC4wLjEKTURCX1BPUlQ9MjcwMTcKTURCX05BTUU9Q2FyU2hvcAoKIyBBUEkKQVBJX1BPUlQ9MzAwMQpgYGAKPiDwn5ej77iPKERpY2EpOiBWb2PDqiBwb2RlIGNvbmVjdGFyIGEgYXBsaWNhw6fDo28gY29tIHVtIGJhbmNvIHJlbW90bywgYWRpY2lvbmFkbyBhIFVSSSBkbyBiYW5jbyBlbSAqKk1EQl9VUkkqKi4KCjxocj4KPC9kZXRhaWxzPgoKIyMjIEV4ZWN1w6fDo28K4pqg77iPIE9zIGNvbWFuZG9zIGEgc2VndWlyIHBvZGVtIHNlciBleGVjdXRhZG9zIGRlbnRybyBkbyBDb250YWluZXIgb3UgbG9jYWxtZW50ZS4KCkluaWNpYXIgbyBwcm9qZXRvIGVtIG1vZG8gZGUgZGVzZW52b2x2aW1lbnRvIChjb20gKipub2RlbW9uKiopOgpgYGBiYXNoCm5wbSBydW4gZGV2CmBgYAoKRXhlY3V0YXIgYSBjb2JlcnR1cmEgZGUgdGVzdGVzOgpgYGBiYXNoCm5wbSBydW4gdGVzdDpkZXYKYGBgCgpFeGVjdXRhIG9zIHRlc3RlcyBlIGRlcG9pcyBjcmlhciBhIHBhc3RhICoqLi9jb3ZlcmFnZSoqIHF1ZSBjb250w6ltIHVtIHJlbGF0w7NyaW86CmBgYGJhc2gKbnBtIHJ1biB0ZXN0OmNvdmVyYWdlCmBgYAo+IPCfl6PvuI8oRGljYSk6IFV0aWxpemFuZG8gYSBleHRlbnPDo28gKipMaXZlIFNlcnZlcioqIG5vIFZpc3VhbCBTdHVkaW8gQ29kZSwgdm9jw6ogcG9kZXLDoSB2aXN1YWxpemFyIGEgcMOhZ2luYSBjb20gbyByZWxhdMOzcmlvIG5hdmVnYW5kbyBhdMOpIHBhc3RhIGNyaWFkYSBlIGFicmluZG8gbyBhcnF1aXZvICoqaW5kZXguaHRtbCoqIG5vIG5hdmVnYWRvci4KCkV4ZWN1dGFyIG8gKipFU0xpbnQqKiBwYXJhIHZlcmlmaWNhciBvcyBlcnJvcyBkbyBjw7NkaWdvIGVzdMOhdGljbzoKYGBgYmFzaApucG0gcnVuIGxpbnQKYGBgCgojIyBUZXN0YW5kbyBvIENSVUQKVm9jw6ogcG9kZSB1dGlsaXphciBxdWFscXVlciBDbGllbnQgQVBJIHBhcmEgdGVzdGFyIG8gQ2FyIFNob3AsIGNvbnR1ZG8sIGV1IGRpc3BvbmliaWxpemVpIGEgZG9jdW1lbnRhw6fDo28gY29tIE9wZW5BUEkgKFN3YWdnZXIpIHBhcmEgcXVlIHZvY8OqIHNhaWJhIHF1YWlzIHJvdGFzIGV4aXN0ZW0gZGVudHJvIGRvIHByb2pldG8gZSBjb21vIHRlc3RhLWxhcy4KCkRlc3RhIGZvcm1hLCBxdWFuZG8gYSBBUEkgZXN0aXZlciBlbSBleGVjdcOnw6NvLCB2b2PDqiBwb2RlcsOhIHZpc2l0YXIgaHR0cDovLzEyNy4wLjAuMTozMDAxL3YxL2RvY3MvIGUgdmlzdWFsaXphciBhIHDDoWdpbmEgZGEgZG9jdW1lbnRhw6fDo28gZSB0ZXN0YXIgYSBBUEkuCgo8aW1nIGFsaWduPSJjZW50ZXIiIGFsdD0iTm9kZUpzIiBzcmM9Imh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9BZG9sbGlnaXQvY2FyLXNob3AvbWFpbi9pbWcvc3MxLnBuZyI+Cgo+IPCfl6PvuI8oRGljYSk6IE8gYmFuY28gZGUgZGFkb3MgbsOjbyBlc3RhIHBvcHVsYWRvLiBQb3J0YW50byDDqSB1bWEgYm9hIGlkZWlhIGNvbWXDp2FyIGNhZGFzdHJhbmRvIG9zIGRhZG9zLgoKIyMgIExpbmd1YWdlbnMgZSBmZXJyYW1lbnRhczoKPGRpdj4KICAgIDxhIGhyZWY9Imh0dHBzOi8vbm9kZWpzLm9yZy9lbi8iPjxpbWcgYWxpZ249ImNlbnRlciIgYWx0PSJOb2RlSnMiIGhlaWdodD0iNDAiIHdpZHRoPSI1MCIgc3JjPSJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvZ2gvZGV2aWNvbnMvZGV2aWNvbi9pY29ucy9ub2RlanMvbm9kZWpzLW9yaWdpbmFsLnN2ZyI+PC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9leHByZXNzanMuY29tL3B0LWJyLyI+PGltZyBhbGlnbj0iY2VudGVyIiBhbHQ9IkV4cHJlc3MiIGhlaWdodD0iNDAiIHdpZHRoPSI1MCIgc3JjPSJodHRwczovL2Nkbi5qc2RlbGl2ci5uZXQvZ2gvZGV2aWNvbnMvZGV2aWNvbi9pY29ucy9leHByZXNzL2V4cHJlc3Mtb3JpZ2luYWwuc3ZnIj48L2E+CiAgICA8YSBocmVmPSJodHRwczovL3d3dy50eXBlc2NyaXB0bGFuZy5vcmcvIj48aW1nIGFsaWduPSJjZW50ZXIiIGFsdD0iVHlwZVNjcmlwdCIgaGVpZ2h0PSI0MCIgd2lkdGg9IjUwIiBzcmM9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9naC9kZXZpY29ucy9kZXZpY29uL2ljb25zL3R5cGVzY3JpcHQvdHlwZXNjcmlwdC1vcmlnaW5hbC5zdmciPjwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vbW9uZ29vc2Vqcy5jb20vIj48aW1nIGFsaWduPSJjZW50ZXIiIGFsdD0iTW9uZ29vc2UiIGhlaWdodD0iNDAiIHdpZHRoPSI1MCIgc3JjPSJodHRwczovL2F2YXRhcnMuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3UvNzU1Mjk2NT9zPTI4MCZ2PTQiPjwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vd3d3Lm1vbmdvZGIuY29tLyI+PGltZyBhbGlnbj0iY2VudGVyIiBhbHQ9Ik1vbmdvIiBoZWlnaHQ9IjQwIiB3aWR0aD0iNTAiIHNyYz0iaHR0cHM6Ly9jZG4uanNkZWxpdnIubmV0L2doL2Rldmljb25zL2Rldmljb24vaWNvbnMvbW9uZ29kYi9tb25nb2RiLXBsYWluLXdvcmRtYXJrLnN2ZyI+PC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9tb2NoYWpzLm9yZy8iPjxpbWcgYWxpZ249ImNlbnRlciIgYWx0PSJNb2NoYSIgaGVpZ2h0PSI0MCIgd2lkdGg9IjUwIiBzcmM9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9naC9kZXZpY29ucy9kZXZpY29uL2ljb25zL21vY2hhL21vY2hhLXBsYWluLnN2ZyI+PC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly93d3cuY2hhaWpzLmNvbS8iPjxpbWcgYWxpZ249ImNlbnRlciIgYWx0PSJDaGFpIiBoZWlnaHQ9IjQwIiB3aWR0aD0iNTAiIHNyYz0iaHR0cHM6Ly9jZG4uaWNvbi1pY29ucy5jb20vaWNvbnMyLzI2OTkvUE5HLzUxMi9jaGFpanNfbG9nb19pY29uXzE2ODQzNS5wbmciPjwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vc2lub25qcy5vcmcvIj48aW1nIGFsaWduPSJjZW50ZXIiIGFsdD0iU2lub24iIGhlaWdodD0iNDAiIHdpZHRoPSI1MCIgc3JjPSJodHRwczovL2F2YXRhcnMuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3UvNjU3MDI1Mz9zPTI4MCZ2PTQiPjwvYT4KPC9kaXY+Cg== readmeEtag: '"76e961ea961d1ba0bd6ca7c46448a4e63e80008c"' readmeLastModified: Wed, 29 Oct 2025 17:12:49 GMT repositoryId: 567522183 description: >- Car Shop é é uma API REST desenvolvida em TypeScript que utiliza o ODM Mongoose para consultar o banco de dados não relacional MongoDB. created: '2022-11-18T00:58:04Z' updated: '2025-10-29T17:13:36Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: Adolligit logo: https://avatars.githubusercontent.com/u/50248803?v=4 repoEtag: '"72694c0e9098fa6bd72c1d0a6c229248c35e8eecbf96e0794a6746065abf0d33"' repoLastModified: Wed, 29 Oct 2025 17:13:36 GMT category: SDK foundInMaster: true - source: - openapi3 tags - openapi31 tags repository: https://github.com/saman-barakat/idlplayground v3: true v3_1: true id: f0df870103a1c6cdaed251b264223984 repositoryMetadata: base64Readme: >- IyMgSURMUGxheWdyb3VuZAoKSURMUGxheWdyb3VuZCBpcyBhbiBvbmxpbmUgYXBwbGljYXRpb24gZGV2ZWxvcGVkIHRvIGFsbG93IGVuZCB1c2VycyB0byBleHBsb3JlIHRoZSBjYXBhYmlsaXRpZXMgb2YgSURMLCBJREw0T0FTLCBhbmQgSURMUmVhc29uZXIgd2ViIEFQSS4gVGhlIFBsYXlncm91bmQgcHJvdmlkZXMgYW4gZWRpdG9yIHNwZWNpZmljYWxseSBkZXNpZ25lZCBmb3IgSURMIGxhbmd1YWdlLgoKIyMjIE92ZXJ2aWV3CgpJRExQbGF5Z3JvdW5kIG9mZmVycyBhIHVzZXItZnJpZW5kbHkgaW50ZXJmYWNlIHRoYXQgaW5jbHVkZXMgYW4gSURMIGVkaXRvciBkZXZlbG9wZWQgdXNpbmcgWyoqWHRleHQqKl0oaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvWHRleHQvKSwgYSBwb3B1bGFyIGZyYW1ld29yayBmb3IgdGhlIGRldmVsb3BtZW50IG9mIHByb2dyYW1taW5nIGxhbmd1YWdlcyBhbmQgRFNMcy4gVGhlIGVkaXRvciBoaWdobGlnaHRzIHRoZSBJREwgc3ludGF4IGFuZCBwcm92aWRlcyByZWFsLXRpbWUgZXJyb3IgZGV0ZWN0aW9uIG9mIHRoZSBJREwgbGFuZ3VhZ2UuCgpUaGUgcGxheWdyb3VuZCBpcyBpbnRlZ3JhdGVkIHdpdGggdGhlIElETFJlYXNvbmVyIHdlYiBBUEksIGFsbG93aW5nIHVzZXJzIHRvIGFuYWx5emUgcHJlLWNvbmZpZ3VyZWQgQVBJIG9wZXJhdGlvbnMgd2l0aCBJREwgc3BlY2lmaWNhdGlvbnMgd2l0aG91dCB0aGUgbmVlZCB0byBpbnN0YWxsIGFkZGl0aW9uYWwgc29mdHdhcmUuIEFjY2VzcyB0aGUgYXBwbGljYXRpb24gaGVyZTogWyoqSURMUGxheWdyb3VuZCoqXShodHRwOi8vaWRsLnVzLmVzL3BsYXlncm91bmQvKQoKIyMjIEZlYXR1cmVzCgotICoqSURMIEVkaXRvcioqOiBTeW50YXggaGlnaGxpZ2h0aW5nIGFuZCByZWFsLXRpbWUgZXJyb3IgZGV0ZWN0aW9uLgotICoqSURMUmVhc29uZXIgd2ViIEFQSSBJbnRlZ3JhdGlvbioqOiBBbmFseXplIElETCBzcGVjaWZpY2F0aW9ucyB1c2luZyB0aGUgSURMUmVhc29uZXIgd2ViIEFQSS4KLSAqKlJlYWwtVGltZSBGZWVkYmFjayoqOiBJbW1lZGlhdGUgc3ludGF4IGVycm9yIGRldGVjdGlvbiBhbmQgYW5hbHlzaXMgcmVzdWx0cy4KLSAqKlVzZXItRnJpZW5kbHkgSW50ZXJmYWNlKio6IEVhc3ktdG8tdXNlIGludGVyZmFjZSBmb3IgZXhwbG9yaW5nIElETCBjYXBhYmlsaXRpZXMuCgojIyMgSG93IHRvIFVzZQoKVGhlIElETFBsYXlncm91bmQgcHJvdmlkZXMgYW4gaW50dWl0aXZlIGludGVyZmFjZSBmb3IgZWRpdGluZyBhbmQgZXZhbHVhdGluZyBJREwgc3BlY2lmaWNhdGlvbnMuIEZvbGxvdyB0aGVzZSBzdGVwcyB0byB1c2UgdGhlIGFwcGxpY2F0aW9uIGVmZmVjdGl2ZWx5OgoKMS4gKipTZWxlY3QgYSBTYW1wbGUgT0FTKio6IEZyb20gdGhlIGNvbWJvIGJveCwgY2hvb3NlIGEgcHJlZGVmaW5lZCBzYW1wbGUgT0FTLiBGb3IgdGhpcyBleGFtcGxlLCBzZWxlY3QgIlllbHAgQnVzaW5lc3MiLgoyLiAqKkNob29zZSBhbiBPcGVyYXRpb24qKjogU2VsZWN0IHRoZSBvcGVyYXRpb24geW91IHdhbnQgdG8gYW5hbHl6ZSwgc3VjaCBhcyAiR0VUIC8gWWVscCBCdXNpbmVzcyBTZWFyY2giLiBUaGUgZWRpdG9yIHdpbGwgYXV0b21hdGljYWxseSBpbXBvcnQgdGhlIGNvcnJlc3BvbmRpbmcgSURMIHNwZWNpZmljYXRpb24uCjMuICoqRWRpdCB0aGUgSURMIFNwZWNpZmljYXRpb24qKjogVXNlIHRoZSBlZGl0b3IgdG8gbWFrZSBjaGFuZ2VzIHRvIHRoZSBJREwgc3BlY2lmaWNhdGlvbi4gVGhlIGVkaXRvciBmZWF0dXJlcyBzeW50YXggaGlnaGxpZ2h0aW5nIGFuZCByZWFsLXRpbWUgZXJyb3IgZGV0ZWN0aW9uIHRvIGhlbHAgeW91IHF1aWNrbHkgaWRlbnRpZnkgYW5kIGNvcnJlY3QgYW55IGlzc3Vlcy4KNC4gKipBbmFseXplIHRoZSBTcGVjaWZpY2F0aW9uKio6IENsaWNrIHRoZSAiQW5hbHl6ZSIgYnV0dG9uIHRvIHNlbmQgdGhlIE9BUyBhbmQgdGhlIElETCBzcGVjaWZpY2F0aW9uIHRvIHRoZSBJRExSZWFzb25lciB3ZWIgQVBJIGZvciB2YWxpZGF0aW9uLgogICAgLSBJZiB0aGUgc3BlY2lmaWNhdGlvbiBpcyB2YWxpZCwgeW91IHdpbGwgcmVjZWl2ZSBhIHN1Y2Nlc3MgbWVzc2FnZS4KICAgIC0gSWYgdGhlIHNwZWNpZmljYXRpb24gaXMgaW52YWxpZCwgdGhlIHdlYiBBUEkgd2lsbCBwcm92aWRlIGFuIGV4cGxhbmF0aW9uIGRldGFpbGluZyB0aGUgaXNzdWVzLgogICAgICAKVG8gc2VlIGEgZGV0YWlsZWQgd2Fsa3Rocm91Z2ggb24gaG93IHRvIHVzZSBJRExQbGF5Z3JvdW5kLCBjaGVjayBvdXQgb3VyIFtZb3VUdWJlIHZpZGVvXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PVMzM0MyTUE1Q21NKS4KCiMjIExpY2Vuc2UKClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgW0dQTC0zLjAgTGljZW5zZV0oTElDRU5TRSkuIAoKIyMgQ29udGFjdAoKRm9yIG1vcmUgaW5mb3JtYXRpb24sIHZpc2l0IHRoZSBbSURMIHdlYnNpdGVdKGh0dHA6Ly9pZGwudXMuZXMvKS4K readmeEtag: '"cc52293327f7f3c34c4977c764f1957f318a1567"' readmeLastModified: Sun, 20 Oct 2024 16:43:14 GMT repositoryId: 577533163 description: >- IDLPlayground is an online application developed to allow end users to explore the capabilities of IDL, IDL4OAS, and IDLReasoner web API. The Playground provides an editor specifically designed for IDL language. created: '2022-12-13T00:17:53Z' updated: '2024-12-09T12:34:52Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: saman-barakat logo: https://avatars.githubusercontent.com/u/3259712?v=4 license: GPL-3.0 repoEtag: '"d92a3bfb072f24f6f2f52072a4135ae91a516171f8611e43d5298b9a4bdb2714"' repoLastModified: Mon, 09 Dec 2024 12:34:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/scarych/aom v3: true repositoryMetadata: base64Readme: >- IyBBT006IEFQSSBPdmVyIE1vZGVscwoKIVtUcmF2aXMgKC5jb20pXShodHRwczovL2ltZy5zaGllbGRzLmlvL3RyYXZpcy9jb20vc2NhcnljaC9hb20pCiFbbnBtXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS9kbS9hb20pCiFbTGlicmFyaWVzLmlvIGRlcGVuZGVuY3kgc3RhdHVzIGZvciBsYXRlc3QgcmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9saWJyYXJpZXNpby9yZWxlYXNlL25wbS9hb20pCiFbR2l0SHViXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL3NjYXJ5Y2gvYW9tKQoKYGFvbWAgLSBpdCBpcyBtZXRhLWZyYW1ld29yayBtYWRlIG9mIHR5cGVzY3JpcHQtZGVjb3JhdG9ycywgd2hpY2ggYWxsb3dzIHRvIGZhc3QgYW5kIGNvbWZvcnRhYmxlCmNyZWF0ZSBzYWZlIGFwaS1zZXJ2aWNlcywgdXNpbmcgdGhlIHByaW5jaXBsZSBvZiBhY2N1bXVsYXRpb24gZGF0YSBsYXllcnMsIGVucmljaGVkIHdpdGggYWJzdHJhY3Rpb25zLgoKIyMgSW5zdGFsbGF0aW9uCgpgYGAKbnBtIGkgLXMgYW9tCmBgYAoKb3IKCmBgYAp5YXJuIGFkZCBhb20KYGBgCgojIyBHZXR0aW5nIHN0YXJ0ZWQKClRvIGNoZWNrIG91dCB0aGUgZG9jdW1lbnRhdGlvbiwgdmlzaXQgW2FvbS5qcy5vcmddKGh0dHA6Ly9hb20uanMub3JnKSAoYGVuYCBhbmQgYHJ1YCBhdmFpbGFibGUpCgojIyBDb25jZXB0CgpUaGUgbWFpbiBpZGVhIHNvdW5kcyBsaWtlOiAiZG9uJ3QgZHVwbGljYXRlIHRoZSBjb2RlLCBsaW5rIHRoZSBjb2RlIi4gYGFvbWAgYWxsb3dzIHRvIHVzZSBkYXRhCnByb2NjZXNzaW5nLCBtYWRlIHRvIGNvdmVyIG1vc3QgY2FzZXMgeW91IG5lZWQuIEF0IHRoZSBzYW1lIHRpbWUgYGFvbWAgZG8gbm90IGxpbWl0IHRoZSBkZXZlbG9wZXIKaW4gZnJhbWVzIG9mIHRoZSBvbmx5IGZyYW1ld29yaywgYnV0IGdpdmVzIHRoZSBhYmlsaXR5IHRvIHVzZSB0aGlyZC1wYXJ0eSBsaWJyYXJpZXMgYW5kIHBhY2thZ2VzLgoKYGFvbWAgaXMgbm90IGEgInRoaW5nIGluIGl0c2VsZiAiLSBhIGZyYW1ld29yayB0aGF0IG9wZXJhdGVzIGV4Y2x1c2l2ZWx5IG9uIGl0cyBvd24gY29kZWJhc2UgYW5kIG9ubHkKd29ya3MgaW4gaXRzIG93biBlbnZpcm9ubWVudC4gSXRzIGltcG9ydGFudCBmZWF0dXJlIGlzIHRoZSBhYmlsaXR5IHRvIGNvbWJpbmUgd2l0aCB0aGUgImNsYXNzaWMiIGNvZGUKb24gYGtvYWAsIHdoaWNoIG1ha2VzIGl0IHVzZWZ1bCB3aGVuIG1pZ3JhdGluZyBmdW5jdGlvbmFsaXR5IGFscmVhZHkgZXhpc3RpbmcgcHJvamVjdHMuCgpgYW9tYCBkb2VzIG5vdCBydW4gY29kZSBpbiBhbiBpc29sYXRlZCBlbnZpcm9ubWVudCwgYnV0IGdlbmVyYXRlcyBzdHJ1Y3R1cmVzIHRoYXQgYXJlIGNvbXBhdGlibGUgd2l0aApwb3B1bGFyIGxpYnJhcmllczogYGtvYS1yb3V0ZXJgLCBga29hLXNlc3Npb25gIGFuZCBvdGhlcnMsIHdoaWNoIGFsbG93cywgaWYgbmVjZXNzYXJ5LAprZWVwIHRoZSBleGlzdGluZyBjb2RlLXN0YWNrLCBhbmQgY29tZm9ydGFibHkgZXh0ZW5kIGl0IGluIHRoZSBgYW9tYCArYHR5cGVzY3JpcHRgIG1ldGhvZG9sb2d5LgoKKipDb2RlIHNhbXBsZSoqCgpgYGB0cwpAQnJpZGdlKCIvYXV0aCIsIEF1dGgpCkBCcmlkZ2UoIi9zaG9wIiwgU2hvcCkKQEJyaWRnZSgiL2FjY291bnQiLCBBY2NvdW50KQpAQ29udHJvbGxlcigpCmNsYXNzIFJvb3QgewogIEBHZXQoKQogIHN0YXRpYyBJbmRleCgpIHsKICAgIHJldHVybiBtb2RlbHMuU2V0dGluZ3MuZmluZE9uZSh7IGVuYWJsZWQ6IHRydWUgfSk7CiAgfQp9CgovLyAuLi4KQENvbnRyb2xsZXIoKQpjbGFzcyBBdXRoIHsKICB1c2VyOiBtb2RlbHMuVXNlcnM7CiAgbG9naW46IG1vZGVscy5Vc2VyTG9naW5zOwogIHRva2VuOiBtb2RlbHMuQXV0aFRva2VuczsKCiAgQE1pZGRsZXdhcmUoKQogIHN0YXRpYyBhc3luYyBSZXF1aXJlZCgKICAgIEBIZWFkZXJzKCJhdXRob3JpemF0aW9uIikgdG9rZW4sCiAgICBAVGhpcygpIF90aGlzOiBBdXRoLAogICAgQE5leHQoKSBuZXh0LAogICAgQEVycigpIGVycgogICkgewogICAgY29uc3QgYXV0aFRva2VuID0gYXdhaXQgbW9kZWxzLkF1dGhUb2tlbnMuY2hlY2tUb2tlbih0b2tlbik7CiAgICBpZiAoYXV0aERhdGEpIHsKICAgICAgX3RoaXMudG9rZW4gPSBhdXRoVG9rZW47CiAgICAgIF90aGlzLnVzZXIgPSBhd2FpdCBtb2RlbHMuVXNlcnMuZmluZEJ5SWQoYXV0aFRva2VuLnVzZXJJZCk7CiAgICAgIF90aGlzLmxvZ2luID0gYXdhaXQgbW9kZWxzLlVzZXJMb2dpbnMuZmluZEJ5SWQoYXV0aFRva2VuLmxvZ2luSWQpOwogICAgICByZXR1cm4gbmV4dCgpOwogICAgfSBlbHNlIHsKICAgICAgcmV0dXJuIGVycigiYWNjZXNzIGRlbmllZCIsIDQwMyk7CiAgICB9CiAgfQoKICBAUG9zdCgpCiAgc3RhdGljIGFzeW5jIExvZ2luKEBCb2R5KCkgeyBsb2dpbiwgcGFzc3dvcmQgfSwgQEVycigpIGVycikgewogICAgY29uc3QgYXV0aExvZ2luID0gYXdhaXQgbW9kZWxzLlVzZXJMb2dpbnMuYXV0aExvZ2luKGxvZ2luLCBwYXNzd29yZCk7CiAgICBpZiAoY2hlY2tMb2dpbikgewogICAgICByZXR1cm4gbW9kZWxzLkF1dGhUb2tlbnMuZ2VuZXJhdGVUb2tlbihhdXRoTG9naW4pOwogICAgfSBlbHNlIHsKICAgICAgcmV0dXJuIGVycigid3JvbmcgbG9naW4iLCA0MDMpOwogICAgfQogIH0KfQoKLy8gLi4uCkBDb250cm9sbGVyKCkKY2xhc3MgU2hvcCB7CiAgQEdldCgpCiAgc3RhdGljIEluZGV4KEBRdWVyeSgpIHF1ZXJ5KSB7CiAgICByZXR1cm4gbW9kZWxzLlByb2R1Y3RzLmZpbmQoeyAuLi5xdWVyeSB9KTsKICB9CgogIEBHZXQoIi9jYXRlZ29yaWVzIikKICBzdGF0aWMgQ2F0ZWdvcmllcyhAUXVlcnkoKSBxdWVyeSkgewogICAgcmV0dXJuIG1vZGVscy5DYXRlZ29yaWVzLmZpbmQoeyAuLi5xdWVyeSB9KTsKICB9CgogIEBHZXQoIi9icmFuZHMiKQogIHN0YXRpYyBCcmFuZHMoQFF1ZXJ5KCkgcXVlcnkpIHsKICAgIHJldHVybiBtb2RlbHMuQnJhbmRzLmZpbmQoeyAuLi5xdWVyeSB9KTsKICB9CgogIEBQb3N0KCIvYWRkX3RvX2NhcnQiKQogIEBVc2UoQXV0aC5SZXF1aXJlZCkKICBzdGF0aWMgQWRkVG9DYXJ0KEBCb2R5KCkgeyBwcm9kdWN0SWQsIHF1YW50aXR5IH0sIEBTdGF0ZU1hcChBdXRoKSB7IHVzZXIgfTogQXV0aCkgewogICAgY29uc3QgYWRkVXNlckNhcnQgPSBhd2FpdCB1c2VyLmFkZFByb2R1Y3RUb0NhcnQocHJvZHVjdElkLCBxdWFudGl0eSk7CiAgICByZXR1cm4gdXNlci5nZXRQcm9kdWN0c0NhcnQoKTsKICB9Cn0KCi8vIC4uLgpAQ29udHJvbGxlcigpCkBVc2UoQXV0aC5SZXF1aXJlZCkKY2xhc3MgQWNjb3VudCB7CiAgQEdldCgpCiAgc3RhdGljIGFzeW5jIEluZGV4KEBTdGF0ZU1hcChBdXRoKSB7IHVzZXIsIGxvZ2luIH06IEF1dGgpIHsKICAgIGNvbnN0IG9yZGVycyA9IGF3YWl0IHVzZXIuZ2V0T3JkZXJzKCk7CiAgICByZXR1cm4geyB1c2VyLCBsb2dpbiwgb3JkZXJzIH07CiAgfQoKICBAUG9zdCgiL2xvZ291dCIpCiAgc3RhdGljIGFzeW5jIExvZ291dChAU3RhdGVNYXAoQXV0aCkgeyB0b2tlbiB9OiBBdXRoKSB7CiAgICBhd2FpdCB0b2tlbi5yZW1vdmUoKTsKICAgIHJldHVybiB7IG1lc3NhZ2U6ICJzdWNjZXNzIGxvZ291dCIgfTsKICB9Cn0KYGBgCgojIyBJc3N1ZXMKClVzZSBbR2l0aHViIGlzc3Vlc10oaHR0cHM6Ly9naXRodWIuY29tL3NjYXJ5Y2gvYW9tL2lzc3VlcykgdG8gYXNrIHlvdXIgcXVlc3Rpb24gb3IgcmVwb3J0IGFib3V0CnByb2JsZW0uCgojIyBDb250YWN0cwoKLSBBdXRob3I6IFtHcmlnb3J5IEtob2xzdGlubmlrb3ZdKGh0dHBzOi8vZ2l0aHViLmNvbS9zY2FyeWNoKQotIEVtYWlsOiBbbWFpbEBzY2FyeWNoLnJ1XShtYWlsdG86bWFpbEBzY2FyeWNoLnJ1KQoKIyMgTGljZW5zZQoKQU9NIGlzIFtNSVQgbGljZW5zZWRdKGh0dHBzOi8vZ2l0aHViLmNvbS9zY2FyeWNoL2FvbS9ibG9iL0hFQUQvTElDRU5TRSkuCgojIyBXYXJuaW5nCgpgYW9tYCBpcyBpbiBvcGVuIGJldGEgYW5kIHdpbGwgYmUgZXhwYW5kZWQgd2l0aCBuZXcgZmVhdHVyZXMuIEVycm9ycyBhcmUgbm90IGV4Y2x1ZGVkLCBhcyB3ZWxsCmFzIHJlcGxhY2VtZW50IGFuZCByZW5hbWluZyBvZiBhIG51bWJlciBvZiBmdW5jdGlvbnMgYW5kIGRlY29yYXRvcnMuCg== readmeEtag: '"69652f085b26e13d0fe4a66a88b67f9354f7beb5"' readmeLastModified: Mon, 12 Aug 2024 10:07:01 GMT repositoryId: 152123069 description: Lightweight and powerful, decorator based REST-API framework. created: '2018-10-08T17:57:15Z' updated: '2025-08-23T14:09:22Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 1 owner: scarych logo: https://avatars.githubusercontent.com/u/1063164?v=4 license: MIT repoEtag: '"8bf2a133fa941e01519bd1612c0ce8a6a209a942194d16158997fb74d8f06ccf"' repoLastModified: Sat, 23 Aug 2025 14:09:22 GMT foundInMaster: true category: Data Validators id: 64190460033d905d82ecec7ff3748dbe - source: openapi3 tags repository: https://github.com/fako1024/httpc v3: true repositoryMetadata: base64Readme: >- IyBBIHNpbXBsZSB3cmFwcGVyIGFyb3VuZCB0aGUgZGVmYXVsdCBHbyBodHRwIGNsaWVudCBvcHRpbWl6ZWQgZm9yIGVhc2Utb2YtdXNlCgpbIVtHaXRodWIgUmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvcmVsZWFzZS9mYWtvMTAyNC9odHRwYy5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZmFrbzEwMjQvaHR0cGMvcmVsZWFzZXMpClshW0dvRG9jXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2Zha28xMDI0L2h0dHBjP3N0YXR1cy5zdmcpXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2Zha28xMDI0L2h0dHBjLykKWyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL2Zha28xMDI0L2h0dHBjKV0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL2Zha28xMDI0L2h0dHBjKQpbIVtCdWlsZC9UZXN0IFN0YXR1c10oaHR0cHM6Ly9naXRodWIuY29tL2Zha28xMDI0L2h0dHBjL3dvcmtmbG93cy9Hby9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZmFrbzEwMjQvaHR0cGMvYWN0aW9ucz9xdWVyeT13b3JrZmxvdyUzQUdvKQpbIVtDb2RlUUxdKGh0dHBzOi8vZ2l0aHViLmNvbS9mYWtvMTAyNC9odHRwYy9hY3Rpb25zL3dvcmtmbG93cy9jb2RlcWwtYW5hbHlzaXMueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9mYWtvMTAyNC9odHRwYy9hY3Rpb25zL3dvcmtmbG93cy9jb2RlcWwtYW5hbHlzaXMueW1sKQoKVGhpcyBwYWNrYWdlIHdyYXBzIHRoZSBHbyBzdGFuZGFyZCBodHRwIGNsaWVudCwgcHJvdmlkaW5nIGEgc2ltcGxpZmllZCBpbnRlcmFjdGlvbiBtb2RlbCB1c2luZyBtZXRob2QgY2hhaW5pbmcgYW5kIGFkZGl0aW9uYWwgY2FwYWJpbGl0aWVzIHN1Y2ggYXMgb3B0aW9uYWwgaW4tZmxvdyB2YWxpZGF0aW9uIGFnYWluc3QgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uLgoKIyMgRmVhdHVyZXMKLSBTaW1wbGUsIG1ldGhvZCBjaGFpbmluZyBiYXNlZCBpbnRlcmZhY2UgZm9yIEhUVFAgY2xpZW50IHJlcXVlc3RzCi0gU2ltdWxhdGlvbiBvZiByZXF1ZXN0IGRlbGF5cwotIFZhbGlkYXRpb24gb2YgcmVxdWVzdCArIHJlc3BvbnNlIGFnYWluc3QgT3BlbkFQSSBzcGVjaWZpY2F0aW9uCi0gQ3VzdG9taXphdGlvbiBvZiBIVFRQIGNsaWVudCB2aWEgZnVuY3Rpb25hbCBwYXJhbWV0ZXIKLSBCYWNrLU9mZi1SZXRyeSBjb25jZXB0IHRvIGF1dG9tYXRpY2FsbHkgcmV0cnkgcmVxdWVzdHMgaWYgcmVxdWlyZWQKCiMjIEluc3RhbGxhdGlvbgpgYGBiYXNoCmdvIGdldCAtdSBnaXRodWIuY29tL2Zha28xMDI0L2h0dHBjCmBgYAoKIyMgRXhhbXBsZXMKIyMjIyBQZXJmb3JtIHNpbXBsZSBIVFRQIEdFVCByZXF1ZXN0CmBgYGdvCmVyciA6PSBodHRwYy5OZXcoIkdFVCIsICJodHRwOi8vZXhhbXBsZS5vcmciKS5SdW4oKQppZiBlcnIgIT0gbmlsIHsKCWxvZy5GYXRhbGYoImVycm9yIHBlcmZvcm1pbmcgR0VUIHJlcXVlc3Q6ICVzIiwgZXJyKQp9CmBgYAoKIyMjIyBQZXJmb3JtIEhUVFAgR0VUIHJlcXVlc3QgYW5kIHBhcnNlIHRoZSByZXN1bHQgYXMgSlNPTiBpbnRvIGEgc3RydWN0CmBgYGdvCnZhciByZXMgPSBzdHJ1Y3QgewoJU3RhdHVzIGludAoJTWVzc2FnZSBzdHJpbmcKfXt9CmVyciA6PSBodHRwYy5OZXcoIkdFVCIsICJodHRwOi8vZXhhbXBsZS5vcmciKS4KCVBhcnNlSlNPTigmcmVzKS4KCVJ1bigpCmlmIGVyciAhPSBuaWwgewoJbG9nLkZhdGFsZigiZXJyb3IgcGVyZm9ybWluZyBHRVQgcmVxdWVzdDogJXMiLCBlcnIpCn0KYGBgCgojIyMjIFBlcmZvcm0gSFRUUFMgUE9TVCByZXF1ZXN0IHdpdGggYSBzaW1wbGUgYm9keSwgZGlzYWJsaW5nIGNlcnRpZmljYXRlIHZhbGlkYXRpb24gYW5kIGNvcHlpbmcgdGhlIHJlc3BvbnNlIHRvIGEgYnl0ZXMuQnVmZmVyCmBgYGdvCmJ1ZiA6PSBuZXcoYnl0ZXMuQnVmZmVyKQplcnIgOj0gaHR0cGMuTmV3KCJQT1NUIiwgImh0dHBzOi8vZXhhbXBsZS5vcmciKS4KCVNraXBDZXJ0aWZpY2F0ZVZlcmlmaWNhdGlvbigpLgoJQm9keShbXWJ5dGV7MHgxLCAweDJ9KS4KCVBhcnNlRm4oaHR0cGMuQ29weShidWYpKS4KCVJ1bigpCgppZiBlcnIgIT0gbmlsIHsKICAgIGxvZy5GYXRhbGYoImVycm9yIHBlcmZvcm1pbmcgUE9TVCByZXF1ZXN0OiAlcyIsIGVycikKfQoKZm10LlByaW50bG4oYnVmLlN0cmluZygpKQpgYGAKCiMjIyMgUGVyZm9ybSBIVFRQUyBHRVQgcmVxdWVzdCAod2l0aCBxdWVyeSBwYXJhbWV0ZXJzICsgaGVhZGVycyArIGJhc2ljIGF1dGgpLCB2YWxpZGF0aW5nIHJlcXVlc3QgYW5kIHJlc3BvbnNlIGFnYWluc3QgT3BlbkFQSXYzIHNwZWNpZmljYXRpb24KYGBgZ28Kb3BlbkFQSUZpbGVEYXRhLCBlcnIgOj0gb3MuUmVhZEZpbGUoIi90bXAvb3BlbmFwaS5qc29uIikKaWYgZXJyICE9IG5pbCB7Cglsb2cuRmF0YWxmKCJFcnJvciBvcGVuaW5nIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBmaWxlOiAlcyIsIGVycikKfQoKZXJyID0gaHR0cGMuTmV3KCJHRVQiLCAiaHR0cHM6Ly9leGFtcGxlLm9yZyIpLgoJU2tpcENlcnRpZmljYXRlVmVyaWZpY2F0aW9uKCkuCglRdWVyeVBhcmFtcyhodHRwYy5QYXJhbXN7CgkJInBhcmFtIjogInRlc3QiLAoJfSkuCglIZWFkZXJzKGh0dHBjLlBhcmFtc3sKCQkiWC1IRUFERVItVEVTVCI6ICJ0ZXN0IiwKCX0pLgoJQXV0aEJhc2ljKCJ1c2VybmFtZSIsICJwYXNzd29yZCIpLgoJT3BlbkFQSVZhbGlkYXRpb25GaWxlRGF0YShvcGVuQVBJRmlsZURhdGEpLgoJUnVuKCkKCmlmIGVyciAhPSBuaWwgewoJbG9nLkZhdGFsZigiZXJyb3IgcGVyZm9ybWluZyBHRVQgcmVxdWVzdDogJXMiLCBlcnIpCn0KYGBgCg== readmeEtag: '"dda14c18e7b3b60355aacf3e4988a28a9f258d26"' readmeLastModified: Tue, 02 Jul 2024 15:02:30 GMT repositoryId: 208796894 description: >- A simple wrapper around the default Go http client optimized for ease-of-use created: '2019-09-16T12:42:06Z' updated: '2025-04-24T14:32:29Z' language: Go archived: false stars: 3 watchers: 4 forks: 2 owner: fako1024 logo: https://avatars.githubusercontent.com/u/10483969?v=4 license: Apache-2.0 repoEtag: '"797c78a1b5771e9ceb4200cde3e72b6f3e815020ef500ce48ad68819034cfdcd"' repoLastModified: Thu, 24 Apr 2025 14:32:29 GMT foundInMaster: true category: Data Validators id: 3c4d97378d5b6503a213aded8daac332 - source: openapi3 tags repository: https://github.com/andrewwgordon/ddd-northwind-api v3: true repositoryMetadata: base64Readme: >- IyBEb21haW4tRHJpdmVuIERlc2lnbiBDUVJTIE1pY3Jvc2VydmljZSBPcGVuQVBJIERlbW8KIyMgT3ZlcnZpZXcKVGhpcyBwcm9qZWN0IGlzIGFuIGV4YW1wbGUgW0RvbWFpbi1Ecml2ZW4gRGVzaWduXShodHRwczovL21hcnRpbmZvd2xlci5jb20vYmxpa2kvRG9tYWluRHJpdmVuRGVzaWduLmh0bWwpIFtNaWNyb3NlcnZpY2VdKGh0dHBzOi8vbWljcm9zZXJ2aWNlcy5pby9pbmRleC5odG1sKSBzdXBwb3J0aW5nIFtPcGVuQVBJXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKS4gVGhlIHB1cnBvc2Ugb2YgdGhlIHByb2plY3QgaXMgYW4gZWR1Y2F0aW9uICJ0b3kiIGFwcGxpY2F0aW9uIHRvIGFydGljdWxhdGUgREREIGFuZCBNaWNyb3NlcnZpY2UgYmVzdCBwcmFjdGljZXMuCgpUaGUgcHJvamVjdCBpcyBidWlsdCB1cG9uIHRoZSBbQ29ubmV4aW9uIE9wZW5BUEldKGh0dHBzOi8vY29ubmV4aW9uLnJlYWR0aGVkb2NzLmlvL2VuL2xhdGVzdC8pIGZyYW1ld29yaywgaXRzZWxmIGxldmVyYWdpbmcgW0ZsYXNrXShodHRwczovL2ZsYXNrLnBhbGxldHNwcm9qZWN0cy5jb20vZW4vMi4wLngvKS4gW1dhaXRyZXNzXShodHRwczovL2RvY3MucHlsb25zcHJvamVjdC5vcmcvcHJvamVjdHMvd2FpdHJlc3MvZW4vbGF0ZXN0LykgaXMgY29uZmlndXJlZCBhcyB0aGUgZGVmYXVsdCBbV1NHSV0oaHR0cHM6Ly9lbi53aWtpcGVkaWEub3JnL3dpa2kvV2ViX1NlcnZlcl9HYXRld2F5X0ludGVyZmFjZSkgSFRUUCBTZXJ2ZXIgYW5kIGlzIHN1aXRhYmxlIGZvciBwcm9kdWN0aW9uIGRlcGxveW1lbnQuCiMjIFF1aWNrIFN0YXJ0CgojIyMgUHJlcXVpc2l0ZXMKUHl0aG9uID49IDMuNwojIyMgSW5zdGFsbApDbG9uZSB0aGlzIHJlcG8uCmBgYGJhc2gKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9hbmRyZXd3Z29yZG9uL2RkZC1ub3J0aHdpbmQtYXBpLmdpdApgYGAKQ3JlYXRlIGEgUHl0aG9uIHZpcnR1YWwgZW52aXJvbm1lbnQgYW5kIGFjdGl2YXRlIGl0LgpgYGBiYXNoCnB5dGhvbiAtbSB2ZW52IHZlbnYKLlx2ZW52XFNjcmlwdHNcYWN0aXZhdGUgKFdpbmRvd3MpCi4vdmVudi9iaW4vYWN0aXZhdGUgKExpbnV4KQpgYGAKSW5zdGFsbCBkZXBlbmRlbmNpZXMKYGBgYmFzaApweXRob24gLW0gcGlwIGluc3RhbGwgLXIgcmVxdWlyZW1lbnRzLnR4dApgYGAKIyMjIERlcGxveSB0aGUgTm9ydGh3aW5kIFNRTGl0ZSBEYXRhYmFzZQpEb3dubG9hZCB0aGUgW1NRTGl0ZTMgTm9ydGh3aW5kIGRhdGFiYXNlXShodHRwczovL2dpdGh1Yi5jb20vanB3aGl0ZTMvbm9ydGh3aW5kLVNRTGl0ZTMvYmxvYi9tYXN0ZXIvTm9ydGh3aW5kX2xhcmdlLnNxbGl0ZS56aXApLCB1bnBhY2sgdGhlIGFyY2hpdmUgYW5kIGRlcGxveSB0byAuL25vcnRod2luZC9yZXNvdXJjZXMvZGF0YWJhc2UvTm9ydGh3aW5kLnNxbGl0ZS4KIyMjIENvbmZpZ3VyZSBMb2dnaW5nCk5vdGU6IFlvdSBhbHNvIG1heSBuZWVkIHRvIGNyZWF0ZSBhIC4vbG9nLyBkaXJlY3RvcnkgaW4gdGhlIHJvb3Qgb2YgdGhlIGFwcGxpY2F0aW9uIGRpcmVjdG9yeS4gTG9nZ2luZyBwYXRoIGNhbiBiZSBjb25maWd1cmVkIGluIC4vbm9ydGh3aW5kL3Jlc291cmNlcy9sb2dnaW5nLmNvbmYuCiMjIyBSdW4gdGhlIFRlc3QgU3VpdGUKYGBgYmFzaApweXRob24gLW0gcHl0ZXN0IC1zIC12CmBgYApOb3RlOiBUaGUgY3VycmVudCBlMmUgUkVTVCBBUEkgdGVzdCBzcGlucyB1cCBhIEZsYXNrIEFwcCBpbnN0YW5jZSBhcyBhIE9TIHN1YiBwcm9jZXNzLiBJdCBhdHRlbXB0cyB0byB0ZXJtaW5hdGUgaXQgb24gdGVzdCBjb21wbGV0aW9uLCBidXQgeW91IG1heSBuZWVkIHRvIGtpbGwgdGhlIHByb2Nlc3MuCiMjIyBTdGFydCB0aGUgU2VydmljZQpTdGFydCB0aGUgc2VydmljZQpgYGBiYXNoCnB5dGhvbiBhcHAucHkKYGBgCgoKTmF2aWdhdGUgdG8gdGhlIFN3YWdnZXIgVUkgdmlhIHlvdXIgYnJvd3NlcgpgYGBiYXNoCmh0dHA6Ly9sb2NhbGhvc3Q6ODA5OC92MS91aQpgYGAKTm90ZTogVGhlIGRlZmF1bHQgcG9ydCBmb3IgdGhlIEhUVFAgU2VydmVyIGlzIDgwOTguIFlvdSBjYW4gY2hhbmdlIHRoaXMgc2V0dGluZyBpbiB0aGUgbm9ydGh3aW5kLnByb3BlcnRpZXMgZmlsZSBpbiAuL25vcnRod2luZC9yZXNvdXJjZXMvLgoKQmFzaWMgYXV0aGVudGljYXRpb24gaXMgZW5hYmxlZCBhbmQgYSB1c2VybmFtZSAvIHBhc3N3b3JkIGhhcyB0byBiZSBzdXBwbGllZCBmb3IgYW55IFJFU1QgQVBJIHNlcnZpY2UgdG8gcmVzcG9uZC4gQ3VycmVudCBpbXBsZW1lbnRhdGlvbiBkb2VzIG5vdCB2ZXJpZnkgY3JlZGVudGlhbHMgYW5kIGNhbiBhY2NlcHQgYW55IHVzZXJuYW1lIGFuZCBwYXNzd29yZC4= readmeEtag: '"1c3e1bb741cf7b5c0638cf0e4776fd6e82bd58f6"' readmeLastModified: Sun, 12 Sep 2021 14:54:09 GMT repositoryId: 404850974 description: >- This project is an example Domain-Driven Design, Microservice supporting OpenAPI .The purpose of the project is an education "toy" application to articulate DDD and Microservice best practices. created: '2021-09-09T19:48:05Z' updated: '2023-07-03T11:22:44Z' language: Python archived: false stars: 1 watchers: 1 forks: 1 owner: andrewwgordon logo: https://avatars.githubusercontent.com/u/51337698?v=4 license: MIT repoEtag: '"570cfd69dbe8555c4633c5bf5bb8916823810354e66e36ac283e2d62b8740188"' repoLastModified: Mon, 03 Jul 2023 11:22:44 GMT foundInMaster: true category: Server Implementations id: ccb4b38a335394fa1fe0a6efc48b58e4 - source: openapi3 tags repository: https://github.com/jitunayak/sukuawali-backend v3: true repositoryMetadata: base64Readme: >- IyMgU3VrdWF3YWxpIGJhY2tlbmQgLVJFU1QgUG9pbnRzCgotICMjIyBTaG93IGFsbCBpdGVtcwoKYGBgaHR0cCByZXF1ZXN0CkdFVCBodHRwOi8vbG9jYWxob3N0OjgwODAvc3VrdWFzL2FsbApgYGAKCi0gIyMjIEFkZCBuZXcgaXRlbQoKYGBgaHR0cCByZXF1ZXN0ClBPU1QgaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N1a3Vhcy9hZGQKQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uCmBgYAoKYGBganNvbgp7CiAgIm5hbWUiOiAiV2hpdGUgbGVnIHByYXduIiwKICAiZGVzY3JpcHRpb24iOiAiU3VwZXIgZGVsaWNpb3VzIiwKICAicHJpY2UiOiA0OTAsCiAgImltYWdlVVJMIjogImh0dHBzOi8vZXhhbXBsZS5jb20vaW1hZ2UvMi9qcGciLAogICJxdWFudGl0eSI6IDEKfQpgYGAKCi0gIyMjIFVwZGF0ZSBleGlzdGluZyBpdGVtCgpgYGBodHRwIHJlcXVlc3QKUFVUIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zdWt1YXMvdXBkYXRlCkNvbnRlbnQtVHlwZTogYXBwbGljYXRpb24vanNvbgpgYGAKCmBgYGpzb24KewogICJpZCI6IDEsCiAgIm5hbWUiOiAiVGlnZXIgZmlzaCBsYXJnZSIsCiAgImRlc2NyaXB0aW9uIjogIlN1cGVyIGRlbGljaW91cyIsCiAgInByaWNlIjogMjUwLAogICJpbWFnZVVSTCI6ICJodHRwczovL2V4YW1wbGUuY29tL2ltYWdlLzEvanBnIiwKICAicXVhbnRpdHkiOiAxCn0KYGBgCgotICMjIyBQbGFjZSBhIG5ldyBvcmRlcgoKYGBgaHR0cCByZXF1ZXN0ClBPU1QgaHR0cDovL2xvY2FsaG9zdDo4MDgwL29yZGVycy9hZGQKQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uCmBgYAoKYGBganNvbgp7CiAgImxvY2F0aW9uIjogIlBhdGlhIiwKICAiaXRlbXMiOiBbCiAgICB7CiAgICAgICJpZCI6IDEsCiAgICAgICJuYW1lIjogIldoaXRlIGxlZyBwcmF3biIsCiAgICAgICJkZXNjcmlwdGlvbiI6ICJTdXBlciBkZWxpY2lvdXMiLAogICAgICAicHJpY2UiOiA0OTAsCiAgICAgICJpbWFnZVVSTCI6ICJodHRwczovL2V4YW1wbGUuY29tL2ltYWdlLzIvanBnIiwKICAgICAgInF1YW50aXR5IjogMQogICAgfQogIF0sCiAgImRlbGl2ZXJ5Q2hhcmdlcyI6IDMwLAogICJ0b3RhbEFtb3VudCI6IDUwMAp9CmBgYAoKLSAjIyMgT3JkZXIgZGVsaXZlcmVkCgpgYGBodHRwIHJlcXVlc3QKUFVUIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9vcmRlcnMvZGVsaXZlcmVkL3tpZH0KYGBgCgotICMjIyBVc2VyIHJlZ2lzdHJhdGlvbgoKYGBgaHR0cCByZXF1ZXN0ClBVVCBodHRwOi8vbG9jYWxob3N0OjgwODAvdXNlcnMvcmVnaXN0ZXIKQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uCmBgYAoKYGBganNvbgp7CiAgInVzZXJuYW1lIjogInVsdHJhMiIsCiAgInBhc3N3b3JkIjogInVsdHJhMiIsCiAgImlzQWN0aXZlIjogdHJ1ZSwKICAicm9sZXMiOiBbCiAgICB7CiAgICAgICJyb2xlIjogIlVTRVIiCiAgICB9LAogICAgewogICAgICAicm9sZSI6ICJBRE1JTiIKICAgIH0KICBdCn0KCmBgYApgYGBodHRwIHJlcXVlc3QKREVMRVRFIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC91c2Vycy9kZWxldGUve3VzZXJfaWR9CmBgYAoKIyMjIyBPbmx5IEFETUlOIGNhbiBhZGQgbmV3IHVzZXJzICYmIERFTElWRVJZX0dVWSBjYW4gY29tcGxldGUgdGhlIGRlbGl2ZXJ5 readmeEtag: '"c1783a82446e8d6f3517b33b88c1a5824a9d3771"' readmeLastModified: Wed, 21 Jul 2021 16:21:50 GMT repositoryId: 385692879 description: null created: '2021-07-13T18:01:00Z' updated: '2021-07-21T16:21:53Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: jitunayak logo: https://avatars.githubusercontent.com/u/35754866?v=4 repoEtag: '"bf34e8a4c3489d9f5e8a7c24a62eb670283e9e5515b03aa74b12edec4a07d126"' repoLastModified: Wed, 21 Jul 2021 16:21:53 GMT foundInMaster: true category: Data Validators id: e347e71a8024aa0361c70e88a26d1a0f - source: openapi3 tags repository: https://github.com/marcelofilipov/filipov-food-api v3: true repositoryMetadata: base64Readme: >- IVtHaXRIdWJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvbWFyY2Vsb2ZpbGlwb3YvZmlsaXBvdi1mb29kLWFwaSkKIVtHaXRIdWIgcmVwbyBzaXplXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9yZXBvLXNpemUvbWFyY2Vsb2ZpbGlwb3YvZmlsaXBvdi1mb29kLWFwaSkKCiMgRmlsaXBvdiBGb29kIEFQSQoKQSAqKkZpbGlwb3YgRm9vZCBBUEkqKiDDqSB1bWEgYXBsaWNhw6fDo28gZGVzZW52b2x2aWRhIGVtIEphdmEgY29tIFNwcmluZyBCb290LCBjb21vIHBhcnRlIGRlIHVtIHByb2pldG8gZGUgZXN0dWRvIGUgcHJvdmEgZGUgY29uY2VpdG8gcGFyYSBjb25zdHJ1w6fDo28gZGUgQVBJcyBSRVNUZnVsIG5vIGRvbcOtbmlvIGRlIGdlcmVuY2lhbWVudG8gZGUgYWxpbWVudG9zLgoKIyMg8J+boCBUZWNub2xvZ2lhcyBVdGlsaXphZGFzCgotICoqSmF2YSoqCi0gKipTcHJpbmcgQm9vdCoqCi0gKipTcHJpbmcgRGF0YSBKUEEgKEhpYmVybmF0ZSkqKgotICoqU3ByaW5nIEZveCAoU3dhZ2dlci9PcGVuQVBJIDMpKioKLSAqKlJlc3QgQXNzdXJlZCoqCi0gKipGaXh0dXJlIEZhY3RvcnkqKgoKIyMg4pyFIEZ1bmNpb25hbGlkYWRlcwoKLSBFbmRwb2ludHMgUkVTVCBwYXJhIG9wZXJhw6fDtWVzIENSVUQgZGUgZW50aWRhZGVzIHJlbGFjaW9uYWRhcyDDoCBnZXN0w6NvIGRlIGFsaW1lbnRvcy4KLSBUZXN0ZXMgYXV0b21hdGl6YWRvcyAodW5pdMOhcmlvcyBlIGRlIGludGVncmHDp8OjbykuCi0gRG9jdW1lbnRhw6fDo28gZGEgQVBJIHZpYSBTd2FnZ2VyLgoKIyMg8J+agCBDb21vIEV4ZWN1dGFyCgojIyMgUHLDqS1yZXF1aXNpdG9zCgotIEpESyAxMSsKLSBNYXZlbgoKIyMjIFBhc3NvcwoKYGBgYmFzaAojIENsb25lIG8gcmVwb3NpdMOzcmlvCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vbWFyY2Vsb2ZpbGlwb3YvZmlsaXBvdi1mb29kLWFwaS5naXQKCiMgQWNlc3NlIG8gZGlyZXTDs3JpbyBkbyBwcm9qZXRvCmNkIGZpbGlwb3YtZm9vZC1hcGkKCiMgQ29tcGlsZSBlIGluc3RhbGUgYXMgZGVwZW5kw6puY2lhcwptdm4gY2xlYW4gaW5zdGFsbAoKIyBFeGVjdXRlIGEgYXBsaWNhw6fDo28KbXZuIHNwcmluZy1ib290OnJ1bgpgYGAKCkEgQVBJIGVzdGFyw6EgZGlzcG9uw612ZWwgZW06IGBodHRwOi8vbG9jYWxob3N0OjgwODBgCgojIyDwn5OEIERvY3VtZW50YcOnw6NvCgpBY2Vzc2UgYGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpLmh0bWxgIHBhcmEgdmlzdWFsaXphciBhIGRvY3VtZW50YcOnw6NvIGludGVyYXRpdmEgZ2VyYWRhIHBlbG8gU3dhZ2dlciAoU3ByaW5nIEZveCkuCgojIyDwn6eqIFRlc3RlcwoKTyBwcm9qZXRvIHBvc3N1aSB0ZXN0ZXMgZGUgaW50ZWdyYcOnw6NvIGNvbSBgUmVzdCBBc3N1cmVkYCBlIG9iamV0b3MgZGUgdGVzdGUgY3JpYWRvcyBjb20gYEZpeHR1cmUgRmFjdG9yeWAuCgojIyDwn5OdIExpY2Vuw6dhCgpFc3RlIHByb2pldG8gZXN0w6EgbGljZW5jaWFkbyBzb2IgYSBsaWNlbsOnYSBNSVQuCgojIyDwn5GkIEF1dG9yCgpEZXNlbnZvbHZpZG8gcG9yIFtNYXJjZWxvIEZpbGlwb3ZdKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXJjZWxvZmlsaXBvdikKCg== readmeEtag: '"0a00724065c62a030d27eb1474b32947041d57c7"' readmeLastModified: Tue, 08 Apr 2025 22:32:05 GMT repositoryId: 221706322 description: ':hamburger: :fries: Food API - Java/SpringBoot | Training and PoC' created: '2019-11-14T13:39:17Z' updated: '2025-04-08T22:32:09Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: marcelofilipov logo: https://avatars.githubusercontent.com/u/42516768?v=4 license: MIT repoEtag: '"2fc25bf1f6af7365319bbc5e85d451a395f7e13600eb045c4bbf3ffbef607f26"' repoLastModified: Tue, 08 Apr 2025 22:32:09 GMT foundInMaster: true category: - Description Validators - SDK id: 8728365f77c1420470cdebd745aa6080 - source: openapi3 tags repository: https://github.com/zoeastra/file-path-api v3: true id: 3fd89fddc49f196bd5a6e31f23ad59cd repositoryMetadata: base64Readme: >- IyBmaWxlLXBhdGgtYXBpCgpUaGUgZmlsZS1wYXRoLWFwaSBSRVNUZnVsIEFQSSBjYW4gYmUgc3RhcnRlZCBieSBjYWxsaW5nIGAuXHJ1bi5wczFgIGluIGEgV2luZG93cyBQb3dlclNoZWxsIGVudmlyb25tZW50LCBvciBgYmFzaCBydW4uc2hgIGluIGEgTWFjL0xpbnV4IGVudmlyb25tZW50LiBUaGUgc2NyaXB0IHRha2VzIGFuIGFic29sdXRlIHBhdGggYXMgYW4gYXJndW1lbnQsIGJ1aWxkcyBhIGRvY2tlciBpbWFnZSwgYW5kIHBhc3NlcyB0aGUgcGF0aCAob3IgdGhlIGRlZmF1bHQgInRlc3RkaXIiIGluY2x1ZGVkIGluIHRoZSBwcm9qZWN0LCBpZiBubyBvdGhlciBwYXRoIGlzIHNwZWNpZmllZCkgYWxvbmcgdG8gZG9ja2VyIGNvbXBvc2UgdG8gdXNlIGFzIGEgYmluZCBtb3VudC4KClRoaXMgQVBJIGlzIGRvY3VtZW50ZWQgaW4gdGhlIFN3YWdnZXIvT3BlbkFQSSAzLjAuMyBmb3JtYXQsIGxvY2F0ZWQgaW4gdGhlIGBhcGlkb2NzLnltbGAgZmlsZSBpbiB0aGUgcm9vdCBkaXJlY3RvcnkuIFRoZSBZQU1MIGZpbGUgaGFzIGFsc28gYmVlbiBjb252ZXJ0ZWQgdG8gSlNPTiwgYW5kIHNlcnZlZCB1cCBhcyBhIFN3YWdnZXIgVUkgc3RhdGljIHdlYnBhZ2UsIHdoaWNoIGNhbiBiZSBmb3VuZCBhdCBbaHR0cDovL2xvY2FsaG9zdDo1MDAwL3N0YXRpYy9pbmRleC5odG1sXShodHRwOi8vbG9jYWxob3N0OjUwMDAvc3RhdGljL2luZGV4Lmh0bWwpIHdoZW4gdGhlIHdlYnNpdGUgaXMgcnVubmluZy4KClNvbWUgb2YgdGhlIHRlc3RzIGFyZSBraW5kIG9mIHdvbmt5LiBDb21wYXJpbmcgdGhlIHJlc3VsdHMgaXMgaGFyZCB3aGVuIHNvbWV0aW1lcyB0aGUgb3JkZXIgb2YgdGhlIHJlc3VsdCBjaGFuZ2VzLCBhcyB3ZWxsIGFzIGZvciBzb21lIHJlYXNvbiBpdCByZWZ1c2luZyB0byBwcm9wZXJseSBkbyBQT1NUcywgUFVUcywgYW5kIERFTEVURXMuIEFsbCBtZXRob2RzIHdvcmsgcGVyZmVjdGx5IGluIHBvc3RtYW4sIHNvIGl0IG11c3QgYmUgc29tZXRoaW5nIGluIHRoZSB0ZXN0cy4gSWYgSSB3ZXJlIHdpbGxpbmcgdG8gc3BlbmQgbW9yZSB0aW1lIG9uIHRoaXMsIEknZCBmaW5kIGEgbW9yZSByb2J1c3QvZWFzaWVyIHdheSBvZiB0ZXN0aW5nIEFQSSByZXN1bHRzLiAKClNpbmNlIGl0IG1heSBoZWxwLCBJIGRpZCBpbmNsdWRlIGFuIGV4cG9ydCBvZiBteSBwb3N0bWFuIHRlc3RzIGZvciB0aGlzIHByb2plY3QsIGBmaWxlLXBhdGgtYXBpLnBvc3RtYW5fY29sbGVjdGlvbi5qc29uYCwgaW4gdGhlIHJvb3QgZGlyZWN0b3J5IG9mIHRoaXMgcHJvamVjdC4KCkkgcHJvYmFibHkgc3BlbnQgbG9uZ2VyIHRoYW4gSSBzaG91bGQgaGF2ZSBvbiB0aGlzLCBhYm91dCA4IGhvdXJzLCB0aG91Z2ggYSBmYWlyIGJpdCBvZiB0aGF0IHdhcyBkdWUgdG8gbXkgdW5mYW1pbGlhcml0eSB3aXRoIGNyZWF0aW5nIEFQSXMgaW4gRmxhc2ssIHNldmVyYWwgY29uZnVzaW5nIGJ1Z3MsIGFzIHdlbGwgYXMgYmVpbmcgYSBiaXQgKG9yIGEgbG90KSBvZiBhIHBlcmZlY3Rpb25pc3QuCg== readmeEtag: '"318b96349e9108d161f07cfa20a9b9bf30a85b14"' readmeLastModified: Mon, 03 Oct 2022 17:29:00 GMT repositoryId: 544972237 description: >- A little file manipulation API I made in Flask, dockerized, and documented in OpenAPI/Swagger. created: '2022-10-03T15:01:03Z' updated: '2022-10-03T15:10:59Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: ZoeAstra logo: https://avatars.githubusercontent.com/u/11560387?v=4 repoEtag: '"d0882b189616931408c19d968bc12b4a61041785276be41df0bd3d950c78683e"' repoLastModified: Mon, 03 Oct 2022 15:10:59 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/asera286/file-path-api - source: openapi3 tags repository: https://github.com/alexbugrimov/rented-apartment v3: true repositoryMetadata: base64Readme: >- IyMg0J/RgNC+0LXQutGCIC0gItCh0YrQtdC80L3QsNGPINC60LLQsNGA0YLQuNGA0LAiIC0gUkVTVCBBUEkgc2VydmljZQoK0J/RgNC+0LXQutGCINGB0L7RgdGC0L7QuNGCINC40LcgMi3RhSDQvNC+0LTRg9C70LXQuToKKiBmcm9udGVuZAoqIHJlc3QtYXBpCgrQodC+0LTQtdGA0LbQuNGCINC40L3RhNC+0YDQvNCw0YbQuNGOCiog0YLQsNGA0LjRhNGLINC30LAg0YPRgdC70YPQs9C4INC/0L4g0YHRh9C10YLRh9C40LrQsNC8OgogICAgKiDRjdC70LXQutGC0YDQvi3RjdC90LXRgNCz0LjRjgogICAgKiDRhdC+0LvQvtC00L3QsNGPINCy0L7QtNCwCiAgICAqINCz0L7RgNGP0YfQsNGPINCy0L7QtNCwCiAgICAqINCy0L7QtNC+0L7RgtCy0LXQtNC10L3QuNC1Ciog0LXQttC10LzQtdGB0Y/Rh9C90YvQtSDQv9C+0LrQsNC30LDQvdC40Y8g0YHRh9C10YLRh9C40LrQvtCyICsg0YHQutGA0LjQvdGLOgogICAgKiDRjdC70LXQutGC0YDQvi3RjdC90LXRgNCz0LjRjgogICAgKiDRhdC+0LvQvtC00L3QsNGPINCy0L7QtNCwCiAgICAqINCz0L7RgNGP0YfQsNGPINCy0L7QtNCwCiog0YHRg9C80LzRiyDQtNC70Y8g0YDQsNGB0YfQtdGC0LAg0LfQsCDQvNC10YHRj9GGCiAgICAqINGN0LvQtdC60YLRgNC+LdGN0L3QtdGA0LPQuNGOCiAgICAqINGF0L7Qu9C+0LTQvdCw0Y8g0LLQvtC00LAKICAgICog0LPQvtGA0Y/Rh9Cw0Y8g0LLQvtC00LAKICAgICog0L7QsdGJ0LDRjyDRgdGD0LzQvNCwINC30LAg0LzQtdGB0Y/RhgoqINCy0YvQs9GA0YPQt9C60LAg0LTQsNC90L3Ri9GFINCyIFBERi3QvtGC0YfQtdGCCg== readmeEtag: '"98f985a743cf8c3faa82b474ec2fc9480d2e76a9"' readmeLastModified: Mon, 12 Oct 2020 16:25:09 GMT repositoryId: 248840821 description: Съемная квартира created: '2020-03-20T19:56:47Z' updated: '2020-10-12T16:25:12Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: AlexBugrimov logo: https://avatars.githubusercontent.com/u/30559567?v=4 repoEtag: '"1add0589021da1e935ed6520ba84a92d07243cbaeeb248099242150c9eda190b"' repoLastModified: Mon, 12 Oct 2020 16:25:12 GMT foundInMaster: true category: DSL id: 4aecd85d2c2ac753e61e3b2ea4d58e2d - source: openapi3 tags repository: https://github.com/vbguard/restful-api-boilerplate v3: true repositoryMetadata: base64Readme: >- IyBSRVNUZnVsLUFQSS1ib2lsZXJwbGF0ZQrim5NUZW1wbGF0ZSBmb3IgZmFzdCBzdGFydCBvZiB3cml0aW5nIFJFU1RmdWwgQVBJIPCfmoAK readmeEtag: '"4ef7d7e4c02bd4bc94553e55a18af5adeab08bb7"' readmeLastModified: Fri, 25 Oct 2019 21:59:33 GMT repositoryId: 197930485 description: ⛓Template for fast start of writing RESTful API 🚀 created: '2019-07-20T12:57:55Z' updated: '2024-06-04T23:58:26Z' language: JavaScript archived: false stars: 1 watchers: 0 forks: 2 owner: vbguard logo: https://avatars.githubusercontent.com/u/3513401?v=4 license: Apache-2.0 repoEtag: '"393d5c4c88d2d33d416b49ce828fd12b842a9f7756d842ebcfdb8769897a8865"' repoLastModified: Tue, 04 Jun 2024 23:58:26 GMT foundInMaster: true category: - Server - Parsers - Server Implementations id: 36c9644a6987d511428b5ae00f48dcb4 - source: openapi3 tags repository: https://github.com/allixender/ogcapi-dggs-webdev-python v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIGdlbmVyYXRlZCBzZXJ2ZXIgZm9yIE9HQyBER0dTIEFQSSB0ZXN0aW5nDQoNCiMjIE92ZXJ2aWV3DQoNClRoaXMgc2VydmVyIHdhcyBnZW5lcmF0ZWQgYnkgdGhlIFtzd2FnZ2VyLWNvZGVnZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWNvZGVnZW4pIHByb2plY3QuIEJ5IHVzaW5nIHRoZQ0KW09wZW5BUEktU3BlY10oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29yZS93aWtpKSBmcm9tIGEgcmVtb3RlIHNlcnZlciwgeW91IGNhbiBlYXNpbHkgZ2VuZXJhdGUgYSBzZXJ2ZXIgc3R1Yi4gIFRoaXMNCmlzIGFuIGV4YW1wbGUgb2YgYnVpbGRpbmcgYSBzd2FnZ2VyLWVuYWJsZWQgRmxhc2sgc2VydmVyLg0KDQpUaGlzIGV4YW1wbGUgdXNlcyB0aGUgW0Nvbm5leGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL3phbGFuZG8vY29ubmV4aW9uKSBsaWJyYXJ5IG9uIHRvcCBvZiBGbGFzay4NCg0KVGhlIE9wZW5BUEkgdGFyZ2V0IGRlZmluaXRpb24gaXMgb2YgREdHUyBBUEkgMC4wLjYgb2YgdGhlIERHR1MgU1dHIFt3aXAtb2djX2FwaV9kZ2dzX3pvbmVfcXVlcnlfcHJvY2Vzc19zdHlsZV0oaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcy9nZW9maXp6eWRyaW5rL3dpcC1vZ2NfYXBpX2RnZ3Nfem9uZV9xdWVyeV9wcm9jZXNzX3N0eWxlLzAuMC42KQ0KDQpUaGUgbGF0ZXN0IGl0ZXJhdGlvbiBvZiB0aGF0IHNwZWMgaXMgYWN0dWFsbHkgW29nY19hcGlfZGdncywgdW5mb3J0dW5hdGVseSBhbHNvIDAuMC42XShodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzL2dlb2Zpenp5ZHJpbmsvb2djX2FwaV9kZ2dzLzAuMC42KS4gVGhpcyByZXBvIGlzIG5vdCB5ZXQgaW1wbGVtZW50aW5nIHRoaXMuIEFzIHBhcnQgb2YgdGhlIFtPY3RvYmVyIDIwMjEgb2djYXBpLWNvZGUtc3ByaW50XShodHRwczovL2dpdGh1Yi5jb20vb3Blbmdlb3NwYXRpYWwvb2djYXBpLWNvZGUtc3ByaW50LTIwMjEtMTAvaXNzdWVzLzEpIEkvd2UgYXJlIGFpbWluZyB0byBzdXBwb3J0IHRoZSBuZXcgQVBJIFNQRUMuIFN0YXkgdHVuZWQgb3IgcHVsbCByZXF1ZXN0Lg0KDQojIyBSZXF1aXJlbWVudHMNClB5dGhvbiAzLjUuMisNCg0KIyMgVXNhZ2UNCg0KV2UgbmVlZCBhbiBlbnYgdmFyIG5hbWVkIFRBQkxFU19DT05GSUcNCnRoaXMgVEFCTEVTX0NPTkZJRyB2YXIgc2hvdWxkIGhvbGQgdGhlIHBhdGggdG8gYSBmaWxlIyBiYXNlZCBvbiB0aGUNCnRhYmxlcy50ZW1wbGF0ZS5jb25mIHdpY2ggaG9sZHMgdGhlIGFjY2VzcyBpbmZvIHRvIHRoZSBkYXRhYmFzZSBkZWZpbml0aW9uLg0KDQpUaGUgbW9zdCB1c2VmdWwgaXMgdGhlIGRlbW8gdGVzdC5zYiBTUWxpdGUgZGF0YWJhc2UuDQoNCmBgYA0KIyBmb3IgZXhhbXBsZQ0KZXhwb3J0IFRBQkxFU19DT05GSUc9JEhPTUUvdGFibGVzLmNvbmYNCg0KYGBgDQoNCg0KVG8gcnVuIHRoZSBzZXJ2ZXIsIHBsZWFzZSBleGVjdXRlIHRoZSBmb2xsb3dpbmcgZnJvbSB0aGUgcm9vdCBkaXJlY3Rvcnk6DQoNCmBgYA0KcGlwMyBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQNCg0KIyBjb25maWd1cmUgY2xpY2tob3VzZSBkYiBhY2Nlc3MNCmV4cG9ydCBUQUJMRVNfQ09ORklHPXRhYmxlcy50ZW1wbGF0ZS5jb25mDQoNCnB5dGhvbjMgLW0gZGdnc19hcGlfc2VydmVyDQoNCiMgb3IgZm9yIGRldiBhbmQgdGVzdGluZw0KIyBweXRob24gc2VydmVyLnB5DQpgYGANCg0KYW5kIG9wZW4geW91ciBicm93c2VyIHRvIGhlcmU6DQoNCmBgYA0KaHR0cDovL2xvY2FsaG9zdDo4MDgwL2RnZ3MtYXBpL3VpLw0KYGBgDQoNCllvdXIgU3dhZ2dlciBkZWZpbml0aW9uIGxpdmVzIGhlcmU6DQoNCmBgYA0KaHR0cDovL2xvY2FsaG9zdDo4MDgwL2RnZ3MtYXBpL29wZW5hcGkuanNvbg0KYGBgDQoNClRvIGxhdW5jaCB0aGUgaW50ZWdyYXRpb24gdGVzdHMsIHVzZSB0b3g6DQpgYGANCnN1ZG8gcGlwIGluc3RhbGwgdG94DQp0b3gNCmBgYA0KDQoqUmVtYXJrOiBJIGhhdmVuJ3QgaGFkIHRpbWUgdG8gdXBkYXRlIHRoZSB0ZXN0cywgcGxlYXNlIGZlZWwgZnJlZSB0byBkbyBzbyoNCg0KIyMgUnVubmluZyB3aXRoIERvY2tlcg0KDQpUbyBydW4gdGhlIHNlcnZlciBvbiBhIERvY2tlciBjb250YWluZXIsIHBsZWFzZSBleGVjdXRlIHRoZSBmb2xsb3dpbmcgZnJvbSB0aGUgcm9vdCBkaXJlY3Rvcnk6DQoNCmBgYGJhc2gNCiMgYnVpbGRpbmcgdGhlIGltYWdlDQpkb2NrZXIgYnVpbGQgLXQgZGdnc19hcGlfc2VydmVyIC4NCg0KIyBzdGFydGluZyB1cCBhIGNvbnRhaW5lcg0KZG9ja2VyIHJ1biAtcCA4MDgwOjgwODAgZGdnc19hcGlfc2VydmVyDQpgYGANCg0KQSByZWNlbnQgZGVtbyBpbnN0YW5jZSB3aXRoIHRoZSBzcWxpdGUzIHRlc3QuZGIgaXMgcnVubmluZyBhdDogaHR0cHM6Ly9kZ2dzLWFwaS1ib3plYTNjc3BhLWV3LmEucnVuLmFwcC9kZ2dzLWFwaS8NCg== readmeEtag: '"d9046e6455564ec248f9f4fe71a27b895b4a5bdc"' readmeLastModified: Tue, 02 Sep 2025 09:12:00 GMT repositoryId: 378910222 description: >- Testing Python code generated servers for ogcapi-discrete-global-grid-systems created: '2021-06-21T11:39:33Z' updated: '2025-09-02T09:12:59Z' language: Jupyter Notebook archived: true stars: 1 watchers: 1 forks: 0 owner: allixender logo: https://avatars.githubusercontent.com/u/4483885?v=4 repoEtag: '"2c589fffd407a555c5f5574d4ea2ba50c0f78316961fc70654a14b4c6ba98f76"' repoLastModified: Tue, 02 Sep 2025 09:12:59 GMT foundInMaster: true category: Server Implementations id: a69484dbdfba2f19352bbc6aa2284c34 - source: openapi3 tags repository: https://github.com/dabevlohn/rocketchat-roles v3: true id: e28802ede32b34f02a89bad6dbc94dd9 repositoryMetadata: base64Readme: >- IyBDcmVhdGUgY3VzdG9tIHZpZXdzIGZvciBSb2NrZXRDaGF0IGRhdGFiYXNlIGNvbGxlY3Rpb25zIChNb25nbykKCldlIHVzZSBSb2NrZXQgOikgd2ViIGZyYW1ld29yayB3cml0dGVuIGluIFJ1c3QKCk5vdyB3ZSBoYXZlIE9wZW5BUEkgZm9yIHJvbGVzICh0aGFua3MgdG8gYGp1aGFrdSAvIHV0b2lwYWApIGFuZCB3ZWIgaW50ZXJmYWNlIHVzaW5nIEhUTVggYW5kIFRlcmEgdGVtcGxhdGVzLgoKLS0tCgpgYGBzaGVsbApjdXJsIGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMC91c2Vycy8gfCBqcQpgYGAKCmBgYGpzb24KWwogIHsKICAgICJfaWQiOiAicm9ja2V0LmNhdCIsCiAgICAicm9sZXMiOiBbCiAgICAgICJib3QiCiAgICBdLAogICAgInN0YXR1cyI6ICJvbmxpbmUiLAogICAgInVzZXJuYW1lIjogInJvY2tldC5jYXQiLAogICAgImFjdGl2ZSI6IHRydWUKICB9LAogIHsKICAgICJfaWQiOiAiRzdSWEtYYTN3OXBNbjNSTXQiLAogICAgInJvbGVzIjogWwogICAgICAidXNlciIsCiAgICAgICJhZG1pbiIKICAgIF0sCiAgICAic3RhdHVzIjogIm9mZmxpbmUiLAogICAgInVzZXJuYW1lIjogImRwbCIsCiAgICAiYWN0aXZlIjogdHJ1ZQogIH0KXQpgYGAKCmBgYHNoZWxsCmN1cmwgaHR0cDovL2xvY2FsaG9zdDo4MDAwL3Blcm1pc3Npb25zLyB8IGpxCmBgYAoKYGBganNvbgpbCiAgewogICAgIl9pZCI6ICJhY2Nlc3MtcGVybWlzc2lvbnMiLAogICAgInJvbGVzIjogWwogICAgICAiYWRtaW4iCiAgICBdCiAgfSwKICB7CiAgICAiX2lkIjogImFjY2Vzcy1tYXJrZXRwbGFjZSIsCiAgICAicm9sZXMiOiBbCiAgICAgICJhZG1pbiIsCiAgICAgICJ1c2VyIgogICAgXQogIH0sCiAgewogICAgIl9pZCI6ICJhY2Nlc3Mtc2V0dGluZy1wZXJtaXNzaW9ucyIsCiAgICAicm9sZXMiOiBbCiAgICAgICJhZG1pbiIKICAgIF0KICB9LAogIHsKICAgICJfaWQiOiAiYWRkLW9hdXRoLXNlcnZpY2UiLAogICAgInJvbGVzIjogWwogICAgICAiYWRtaW4iCiAgICBdCiAgfSwKICB7CiAgICAiX2lkIjogImFkZC11c2VyLXRvLWpvaW5lZC1yb29tIiwKICAgICJyb2xlcyI6IFsKICAgICAgImFkbWluIiwKICAgICAgIm93bmVyIiwKICAgICAgIm1vZGVyYXRvciIKICAgIF0KICB9LAogIHsKICAgICJfaWQiOiAiYWRkLXVzZXItdG8tYW55LWMtcm9vbSIsCiAgICAicm9sZXMiOiBbCiAgICAgICJhZG1pbiIKICAgIF0KICB9LAogIHsKICAgICJfaWQiOiAia2ljay11c2VyLWZyb20tYW55LWMtcm9vbSIsCiAgICAicm9sZXMiOiBbCiAgICAgICJhZG1pbiIKICAgIF0KICB9LAogIHsKICAgICJfaWQiOiAiYXBpLWJ5cGFzcy1yYXRlLWxpbWl0IiwKICAgICJyb2xlcyI6IFsKICAgICAgImFkbWluIiwKICAgICAgImJvdCIsCiAgICAgICJhcHAiCiAgICBdCiAgfSwKICB7CiAgICAiX2lkIjogImFyY2hpdmUtcm9vbSIsCiAgICAicm9sZXMiOiBbCiAgICAgICJhZG1pbiIsCiAgICAgICJvd25lciIKICAgIF0KICB9LAogIHsKICAgICJfaWQiOiAiYXNzaWduLWFkbWluLXJvbGUiLAogICAgInJvbGVzIjogWwogICAgICAiYWRtaW4iCiAgICBdCiAgfSwKICB7CiAgICAiX2lkIjogImFzc2lnbi1yb2xlcyIsCiAgICAicm9sZXMiOiBbCiAgICAgICJhZG1pbiIKICAgIF0KICB9LAogIHsKICAgICJfaWQiOiAiYmFuLXVzZXIiLAogICAgInJvbGVzIjogWwogICAgICAiYWRtaW4iLAogICAgICAib3duZXIiLAogICAgICAibW9kZXJhdG9yIgogICAgXQogIH0sCi4uLgpgYGAKCmBgYHNoZWxsCmN1cmwgaHR0cDovL2xvY2FsaG9zdDo4MDAwL3JvbGVzLyB8IGpxCmBgYAoKYGBganNvbgpbCiAgewogICAgIl9pZCI6ICJhZG1pbiIsCiAgICAic2NvcGUiOiAiVXNlcnMiLAogICAgIm5hbWUiOiAiYWRtaW4iCiAgfSwKICB7CiAgICAiX2lkIjogIm1vZGVyYXRvciIsCiAgICAic2NvcGUiOiAiU3Vic2NyaXB0aW9ucyIsCiAgICAibmFtZSI6ICJtb2RlcmF0b3IiCiAgfSwKICB7CiAgICAiX2lkIjogImxlYWRlciIsCiAgICAic2NvcGUiOiAiU3Vic2NyaXB0aW9ucyIsCiAgICAibmFtZSI6ICJsZWFkZXIiCiAgfSwKICB7CiAgICAiX2lkIjogIm93bmVyIiwKICAgICJzY29wZSI6ICJTdWJzY3JpcHRpb25zIiwKICAgICJuYW1lIjogIm93bmVyIgogIH0sCiAgewogICAgIl9pZCI6ICJ1c2VyIiwKICAgICJzY29wZSI6ICJVc2VycyIsCiAgICAibmFtZSI6ICJ1c2VyIgogIH0sCiAgewogICAgIl9pZCI6ICJib3QiLAogICAgInNjb3BlIjogIlVzZXJzIiwKICAgICJuYW1lIjogImJvdCIKICB9LAogIHsKICAgICJfaWQiOiAiYXBwIiwKICAgICJzY29wZSI6ICJVc2VycyIsCiAgICAibmFtZSI6ICJhcHAiCiAgfSwKICB7CiAgICAiX2lkIjogImd1ZXN0IiwKICAgICJzY29wZSI6ICJVc2VycyIsCiAgICAibmFtZSI6ICJndWVzdCIKICB9LAogIHsKICAgICJfaWQiOiAiYW5vbnltb3VzIiwKICAgICJzY29wZSI6ICJVc2VycyIsCiAgICAibmFtZSI6ICJhbm9ueW1vdXMiCiAgfSwKICB7CiAgICAiX2lkIjogImxpdmVjaGF0LWFnZW50IiwKICAgICJzY29wZSI6ICJVc2VycyIsCiAgICAibmFtZSI6ICJsaXZlY2hhdC1hZ2VudCIKICB9LAogIHsKICAgICJfaWQiOiAibGl2ZWNoYXQtbWFuYWdlciIsCiAgICAic2NvcGUiOiAiVXNlcnMiLAogICAgIm5hbWUiOiAibGl2ZWNoYXQtbWFuYWdlciIKICB9LAogIHsKICAgICJfaWQiOiAibGl2ZWNoYXQtbW9uaXRvciIsCiAgICAic2NvcGUiOiAiVXNlcnMiLAogICAgIm5hbWUiOiAibGl2ZWNoYXQtbW9uaXRvciIKICB9Cl0KYGBgCg== readmeEtag: '"a6a406a117a93b8707b897442409c08dc575ca58"' readmeLastModified: Sun, 22 Sep 2024 06:40:25 GMT repositoryId: 795627333 description: RocketChat roles and permissions research created: '2024-05-03T17:22:26Z' updated: '2024-11-18T10:19:47Z' language: Rust archived: false stars: 1 watchers: 1 forks: 0 owner: dabevlohn logo: https://avatars.githubusercontent.com/u/144282650?v=4 repoEtag: '"644e826d6d95695578e008cd6ad4555502312b22023c1f30f919126238bbc19e"' repoLastModified: Mon, 18 Nov 2024 10:19:47 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/traefik/hub v3: true id: 055ce36733aaa4fb20b0e349443e94ea repositoryMetadata: base64Readme: >- IyBUcmFlZmlrIEh1YgoKPGRpdiBhbGlnbj0iY2VudGVyIiBzdHlsZT0ibWFyZ2luOiAzMHB4OyI+CjxhIGhyZWY9Imh0dHBzOi8vaHViLnRyYWVmaWsuaW8vIj4KICA8aW1nIHNyYz0iaHR0cHM6Ly9kb2MudHJhZWZpay5pby90cmFlZmlrLWh1Yi9pbWcvdHJhZWZpay1odWItbG9nby5zdmciIHN0eWxlPSJ3aWR0aDoyNTBweDsiIGFsaWduPSJjZW50ZXIiIC8+CjwvYT4KPGJyIC8+CjxiciAvPgoKPGRpdiBhbGlnbj0iY2VudGVyIj4KICAgIDxhIGhyZWY9Imh0dHBzOi8vaHViLnRyYWVmaWsuaW8iPkxvZyBJbjwvYT4gfAogICAgPGEgaHJlZj0iaHR0cHM6Ly9kb2MudHJhZWZpay5pby90cmFlZmlrLWh1YiI+RG9jdW1lbnRhdGlvbjwvYT4KPC9kaXY+CjwvZGl2PgoKPGJyIC8+Cgo8ZGl2IGFsaWduPSJjZW50ZXIiPjxzdHJvbmc+VHJhZWZpayBIdWI8L3N0cm9uZz4KCjxiciAvPgo8YnIgLz4KPC9kaXY+Cgo8ZGl2IGFsaWduPSJjZW50ZXIiPldlbGNvbWUgdG8gdGhpcyByZXBvc2l0b3J5ITwvZGl2PgoKIyMgOmluZm9ybWF0aW9uX3NvdXJjZTogQWJvdXQKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBzb3VyY2UgY29kZSBzaG93aW5nIGhvdyB0byB1c2U6CgoxLiBUcmFlZmlrIEh1YiBBUEkgR2F0ZXdheQoyLiBUcmFlZmlrIEh1YiBBUEkgTWFuYWdlbWVudAoKCiMjIDphbGVtYmljOiBBUElzIHVzZWQgaW4gdGhpcyByZXBvc2l0b3J5CgpBbGwgQVBJcyBhcmUgaW1wbGVtZW50ZWQgdXNpbmcgYSB0aW55IEpTT04gc2VydmVyIGluIEdvOyB0aGUgc291cmNlIGNvZGUgaXMgW2hlcmVdKC4vc3JjL2FwaS1zZXJ2ZXIpLgoKVGhpcyBKU09OIHNlcnZlciBpcyB1c2VkIHRvIGRlcGxveSBKU09OIEFQSXMgdXNpbmcgYSBjb25maWdtYXAuCgpUaGUgS3ViZXJuZXRlcyBtYW5pZmVzdHMgKFlBTUwpIHRvIGRlcGxveSB0aG9zZSBhcHBzIGFyZSBbaGVyZV0oLi9zcmMvbWFuaWZlc3RzKS4KCiMjIDpjb25zdHJ1Y3Rpb25fd29ya2VyOiBXaGVyZSB0byBzdGFydCA/CgpUaGUgam91cm5leSBjYW4gc3RhcnQgW2hlcmVdKFdBTEtUSFJPVUdILm1kKSBmb3IgYSBxdWlja3N0YXJ0IHdpdGggYSBnbG9iYWwgb3ZlcnZpZXcKCiMjIPCfk5IgUmVwb3NpdG9yeSBTdHJ1Y3R1cmUKCmBgYHNoZWxsCi4K4pSc4pSA4pSAIGFwaS1nYXRld2F5ICAgICAgICAgICAgICAgICAgICAgICAjIFRyYWVmaWsgSHViIEFQSSBHYXRld2F5IHR1dG9yaWFscwrilILCoMKgIOKUnOKUgOKUgCAxLWdldHRpbmctc3RhcnRlZCAgIyBBUEkgR2F0ZXdheSBRdWljayBTdGFydCBHdWlkZQrilILCoMKgIOKUnOKUgOKUgCAyLWV4cG9zZQrilILCoMKgIOKUnOKUgOKUgCAzLXNlY3VyZS1hcHBsaWNhdGlvbnMK4pSc4pSA4pSAIGFwaS1tYW5hZ2VtZW50ICAgICAgICAgICAgICAgICAgICAjIFRyYWVmaWsgSHViIEFQSSBNYW5hZ2VtZW50IHR1dG9yaWFscwrilILCoMKgIOKUnOKUgOKUgCAxLWdldHRpbmctc3RhcnRlZCAjIEFQSSBNYW5hZ2VtZW50IFF1aWNrIFN0YXJ0IEd1aWRlCuKUlOKUgOKUgCBzcmMKIMKgwqAg4pSc4pSA4pSAIGFwaS1zZXJ2ZXIgICAgICAgICAgICAgICAgICAgICMgQVBJIHNlcnZlciBzb3VyY2UgY29kZQogwqDCoCDilJTilIDilIAgbWFuaWZlc3RzICAgICAgICAgICAgICAgICAgICAgIyBZYW1sIHRvIGRlcGxveSBhbGwgYXBwcwpgYGAK readmeEtag: '"80c7e839feb1975f5c5097c7bb95b4d62d63ee01"' readmeLastModified: Thu, 18 Sep 2025 06:30:04 GMT repositoryId: 335312873 description: Traefik Hub Tutorials created: '2021-02-02T14:18:58Z' updated: '2025-09-18T06:30:09Z' language: Go archived: false stars: 4 watchers: 7 forks: 10 owner: traefik logo: https://avatars.githubusercontent.com/u/14280338?v=4 license: Apache-2.0 repoEtag: '"58becca9a409a228ecb7eb309cc4eb9847434502bf8ac84860e4a4ad50c03dea"' repoLastModified: Thu, 18 Sep 2025 06:30:09 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/justedlev/bridgewayhub v3: true id: af0bcffeb9100735d4b9e67241d85444 repositoryMetadata: base64Readme: >- <div id="header" align="center">
    <h1>BridgeWay Hub</h1>
    <h3>🧱 API Gateway</h3>
</div>

<div id="badges" align="center">

[![language](https://img.shields.io/badge/Java%2017-e6892e.svg?logo=openjdk&logoColor=white)](https://github.com/justedlev/bridgewayhub)
[![framework](https://img.shields.io/badge/Spring%20Boot%203-6DB33F.svg?logo=springboot&logoColor=white)](https://docs.spring.io/spring-boot/index.html)
[![Docker Image Version](https://img.shields.io/docker/v/justedlev/bridgewayhub?logo=docker&label=bridgewayhub)](https://hub.docker.com/repository/docker/justedlev/bridgewayhub)
[![license](https://img.shields.io/github/license/justedlev/bridgewayhub)](https://www.apache.org/licenses/LICENSE-2.0.txt)
[![stars](https://img.shields.io/github/stars/justedlev/bridgewayhub)](https://github.com/justedlev/bridgewayhub/star)
[![issues](https://img.shields.io/github/issues/justedlev/bridgewayhub)](https://github.com/justedlev/bridgewayhub/issues)

</div>

## 📋 About

__BridgeWay Hub__ example implementation of `API Gateway` based on `Spring Boot 3`, `Keycloak` as a security layer
and `Eureka Client` as service registry, in microservice architecture, detailed configuration will depend directly on
additional business requirements and is applied in the application
properties. [For more details](https://docs.spring.io/spring-cloud-gateway/docs/current/reference/html/)

## 🧾 References

- [Externalized Configuration](https://docs.spring.io/spring-boot/reference/features/external-config.html#features.external-config.typesafe-configuration-properties.relaxed-binding.environment-variables)
- [Docker CLI](https://docs.docker.com/reference/cli/docker/compose/)
- [Keycloak on Docker](https://www.keycloak.org/getting-started/getting-started-docker)
- [Postgres on Docker](https://hub.docker.com/_/postgres)

## ⚠️ Requirements

Before running the app you need to configure the next services that depends on:

- Keycloak - security layer
- DB for Keycloak `optional`
- Eureka Server, my [solution](https://github.com/Justedlev/simple-eureka-server) `optional`

## ▶️ Run

### <a href="#"><img src="https://github.com/JetBrains/logos/raw/refs/heads/master/web/intellij-idea/intellij-idea.svg" width="20"/></a> Intellij

To run the app you can use simple run configuration (Intellij IDEA), that based on [.env](.env)
and [jvm options](.vmoptions)

```xml
<component name="ProjectRunConfigurationManager">
    <configuration default="false" name="Default" type="SpringBootApplicationConfigurationType"
                   factoryName="Spring Boot">
        <option name="envFilePaths">
            <option value="$PROJECT_DIR$/.env"/>
        </option>
        <option name="FRAME_DEACTIVATION_UPDATE_POLICY" value="UpdateClassesAndResources"/>
        <module name="bridgewayhub"/>
        <selectedOptions>
            <option name="environmentVariables"/>
        </selectedOptions>
        <option name="SPRING_BOOT_MAIN_CLASS" value="io.github.justedlev.bridgeway.BridgeWayHubApplication"/>
        <option name="VM_PARAMETERS" value="@.vmoptions"/>
        <method v="2">
            <option name="Make" enabled="true"/>
        </method>
    </configuration>
</component>
```

> [!WARNING]
>
> Make sure that the next services already started
> 
> - Service Registry (eureka server)
> - Keycloak server
> - Postgres DB

> [!NOTE]
>
> The Service Registry (Discovery Service) can be disabled by using the properties if needed
> 
> ```yml 
> spring:
>   cloud:
>     discovery:
>       enabled: false
> eureka:
>   client:
>     enabled: false
> ```

> [!TIP]
>
> You can also disable it in [.vmoptions](.vmoptions), just adding the envs
> 
> ```
> -Dspring.cloud.discovery.enabled=false
> -Deureka.client.enabled=false
> ```
> 
> OR
>
> You can add the env props to the [.env](.env)
> 
> ```
> SPRING_CLOUD_DISCOVERY_ENABLED=false
> EUREKA_CLIENT_ENABLED=false
> ```

### 🐳 Docker

First build your amazing 😁 `docker compose` file and then run using the below cmd

```shell
docker compose build -d
```

> [!TIP]
> 
> I already prepared some image on [🐳 docker hub](https://hub.docker.com/repository/docker/justedlev/bridgewayhub)

#### 📝 Docker compose

Simple command to run the container:

```shell
docker compose up -d --build
```

The full [compose.yaml](docker/compose.yaml) that I personally use

```yml
name: justedlev-msrv
services:
  # API Gateway
  api-gateway.justedlev:
    tty: true
    env_file:
      - .env
    container_name: bridgewayhub
    image: justedlev/bridgewayhub:latest
    build:
      context: .
    ports:
      - "${SERVER_PORT}:${SERVER_PORT}"
    healthcheck:
      test: [ "CMD", "curl", "-k", "-f", "http://localhost:${SERVER_PORT}/actuator/health" ]
      interval: 30s
      timeout: 10s
      retries: 5
      start_period: 30s
    depends_on:
      auth-server.justedlev:
        condition: service_started
      service-discovery.justedlev:
        condition: service_healthy
      config-server.justedlev:
        condition: service_healthy
    volumes:
      - ./logs:${LOGGING_FILE_PATH}
    deploy:
      resources:
        limits:
          cpus: "2"
          memory: 1GB

  # Service discovery
  service-discovery.justedlev:
    tty: true
    environment:
      - SERVER_PORT=${SERVER_PORT}
      - SPRING_APPLICATION_NAME=service-discovery
      - SPRING_SECURITY_USER_NAME={example}
      - SPRING_SECURITY_USER_PASSWORD={example}
      - SPRING_SECURITY_USER_ROLES={example}
      - EUREKA_INSTANCE_HOSTNAME=service-discovery
      - EUREKA_CLIENT_SERVICE_URL_DEFAULTZONE=http://{example}:{example}@service-discovery.justedlev.com:${SERVER_PORT}/eureka/
      - LOGGING_FILE_PATH=/var/log
    container_name: ${SRVREG_APP_NAME}
    image: justedlev/simple-eureka-server:latest
    ports:
      - "${SERVER_PORT}:${SERVER_PORT}"
    healthcheck:
      test: [ "CMD", "curl", "-k", "-f", "http://localhost:${SERVER_PORT}/actuator/health" ]
      interval: 30s
      timeout: 10s
      retries: 5
      start_period: 10s
    depends_on:
      config-server.justedlev:
        condition: service_healthy
    volumes:
      - ./logs/service-discovery:/var/log
    deploy:
      resources:
        limits:
          cpus: "2"
          memory: 1GB

  # SSO service (keycloak)
  sso:
    container_name: keycloak
    image: quay.io/keycloak/keycloak:25.0.6
    command: [ "start-dev" ]
    environment:
      KEYCLOAK_ADMIN: "{example}"
      KEYCLOAK_ADMIN_PASSWORD: "{example}"
      KC_HEALTH_ENABLED: true
      KC_HOSTNAME: localhost
      KC_DB: postgres
      KC_DB_URL: jdbc:postgresql://postgres:5432/{example}
      KC_DB_USERNAME: "{example}"
      KC_DB_PASSWORD: "{example}"
      KC_DB_SCHEMA: keycloak
    depends_on:
      - postgres
    ports:
      - "9321:8080"

  # Postgres DB
  postgres:
    container_name: postgres
    image: postgres:16.4-alpine
    environment:
      POSTGRES_DB: "{example}"
      POSTGRES_USER: "{example}"
      POSTGRES_PASSWORD: "{example}"
    volumes:
      - db-data:/var/lib/postgresql/data
      - /.db:/docker-entrypoint-initdb.d
    ports:
      - "5432:5432"
    healthcheck:
      test: [ "CMD", "pg_isready", "-U", "${POSTGRES_USER}", "-d" ]
      interval: 15s
      timeout: 10s
      retries: 5
      start_period: 12s
    restart: unless-stopped
    deploy:
      resources:
        limits:
          cpus: "1"
          memory: 250MB

volumes:
  db-data:
```
 readmeEtag: '"9833ff00a5c7aa644a9db84d35d785ca5118c2ce"' readmeLastModified: Sat, 18 Oct 2025 15:02:28 GMT repositoryId: 786112920 description: 🚧 API Gateway created: '2024-04-13T13:22:10Z' updated: '2025-12-28T19:44:38Z' language: Java archived: false stars: 3 watchers: 1 forks: 0 owner: justedlev logo: https://avatars.githubusercontent.com/u/59051661?v=4 license: Apache-2.0 repoEtag: '"44484441afbd9e0e818576097247cef4cb93141d56404ded4a075c0cb47d3bd3"' repoLastModified: Sun, 28 Dec 2025 19:44:38 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/owainlewis/openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJCgpIYXNrZWxsIHR5cGVzIGZvciB3b3JraW5nIHdpdGggYW5kIG1hbmlwdWxhdGluZyBPcGVuQVBJIHNwZWNpZmljYXRpb24gZG9jdW1lbnRzLgo= readmeEtag: '"e6218aeac6d357cd500e5aa2d6ce15bd1664c856"' readmeLastModified: Sat, 02 Nov 2019 20:23:26 GMT repositoryId: 216896190 description: >- Parse, manipulate and construct OpenAPI specification documents using Haskell created: '2019-10-22T19:42:07Z' updated: '2023-09-08T17:59:22Z' language: Haskell archived: false stars: 1 watchers: 1 forks: 0 owner: owainlewis logo: https://avatars.githubusercontent.com/u/733944?v=4 license: BSD-3-Clause repoEtag: '"f218fb66861842fd7a7794c5f4765b2b80277356862a9ba26ccc25dc5b7ffce2"' repoLastModified: Fri, 08 Sep 2023 17:59:22 GMT foundInMaster: true category: Parsers id: 6fc09913925582445df2c5f3949bd738 - source: openapi3 tags repository: https://github.com/karthikairam/account-transaction-service v3: true id: 525b82762c7e108c88a0c9d3d3c42738 repositoryMetadata: base64Readme: >- IVtKYXZhIENJIHdpdGggTWF2ZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9rYXJ0aGlrYWlzZWx2YW4vYWNjb3VudC10cmFuc2FjdGlvbi1zZXJ2aWNlL3dvcmtmbG93cy9KYXZhJTIwQ0klMjB3aXRoJTIwTWF2ZW4vYmFkZ2Uuc3ZnKSAhW01hdmVuIFBhY2thZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9rYXJ0aGlrYWlzZWx2YW4vYWNjb3VudC10cmFuc2FjdGlvbi1zZXJ2aWNlL3dvcmtmbG93cy9NYXZlbiUyMFBhY2thZ2UvYmFkZ2Uuc3ZnKSAhW0RvY2tlcl0oaHR0cHM6Ly9naXRodWIuY29tL2thcnRoaWthaXNlbHZhbi9hY2NvdW50LXRyYW5zYWN0aW9uLXNlcnZpY2Uvd29ya2Zsb3dzL0RvY2tlci9iYWRnZS5zdmcpIAoKIyBNaWNyb3NlcnZpY2U6IGFjY291bnQtdHJhbnNhY3Rpb24tc2VydmljZQpUaGlzIGlzIGEgZGVtbyBtaWNyb3NlcnZpY2UgZm9yIHRoZSBhY2NvdW50IHRyYW5zYWN0aW9uIGFzIHBhcnQgb2YgbW9iaWxlIGNoYWxsZW5nZXJzIGJhbmtpbmcuCgojIyMgVGVjaCBzdGFjayB1c2VkOgoxLiBKYXZhIDExCjIuIFNwcmluZyBCb290IDIuNC4xCjMuIFNwcmluZyBSRVNUIEFQSSAoU3ByaW5nIFdlYikKNC4gTG9tYm9rIDEuMTguMTYKNS4gU3ByaW5nIERhdGEgSlBBICYgSDIgaW4tbWVtb3J5IGRhdGFiYXNlIChleHBvc2VkIHRoZSBjbGllbnQgZGFzaGJvYXJkIHRvIHNlZSB0aGUgcmVjb3JkcykKNi4gc3ByaW5nZG9jLW9wZW5hcGktdWkgLSBBdXRvIGdlbmVyYXRlZCBBUEkgZG9jdW1lbnRhdGlvbiB1c2luZyBPcGVuQVBJIDMuMCBhbmQgZXhwb3NlZCBTd2FnZ2VyIFVJIGZvciB0aGUgc2FtZSAoUmVmZXIgdGhlIGxpbmsgYXQgdGhlIGJvdHRvbSBvZiB0aGlzIHBhZ2UpCjcuIERvY2tlcml6ZWQgKERvY2tlcmZpbGUgaXMgaW5jbHVkZWQgYW5kIHVzZWQgdGhlIGxhdGVzdCBjb25jZXB0IGNhbGxlZCBtdWx0aS1sYXllcmVkIGltYWdlIGJ1aWxkaW5nIGZvciBvcHRpbWl6ZWQgYnVpbGQgdGltZSkKOC4gVEREIGFwcHJvYWNoIHVzaW5nIEpVbml0IDUsIE1vY2tpdG8sIGFuZCBTcHJpbmcgQm9vdCBUZXN0CjkuIEphQ29DbyBmb3IgY29kZSBjb3ZlcmFnZQoxMC4gU3ByaW5nIGJvb3Qgc3RhcnRlciB2YWxpZGF0aW9uIGZvciByZXF1ZXN0IHZhbGlkYXRpb24gIAoKLS0tCiMjIyBKYUNvQ28gY29kZSBjb3ZlcmFnZSBpczogYGBgIDEwMCAlIGBgYCAgCiFbSmFDb0NvX1JlcG9ydCBpbWcgbWlzc2luZ10oaHR0cHM6Ly9naXRodWIuY29tL2thcnRoaWthaXNlbHZhbi9hY2NvdW50LXRyYW5zYWN0aW9uLXNlcnZpY2UvYmxvYi9tYWluL2ltZy9KYUNvQ29fUmVwb3J0LmpwZz9yYXc9dHJ1ZSkgIAoKLS0tCiMjIyBTdGVwcyB0byBidWlsZCB0aGUgRG9ja2VyIEltYWdlOgojIyMjIyBTdGVwIDE6IEVudGVyIHRoZSBwcm9qZWN0IGRpcmVjdG9yeSwgCiAgICBjZCA8cHJvamVjdF9kaXJlY3Rvcnk+CgojIyMjIyBTdGVwIDI6IFRvIGNsZWFuIGFuZCBwYWNrYWdlIHRoZSBtaWNyby1zZXJ2aWNlIGxvY2FsbHkKICAgIG12biBjbGVhbiBwYWNrYWdlCiAKIyMjIyMgU3RlcCAzOiBCdWlsZCB0aGUgZG9ja2VyIGltYWdlIGFuZCBzdG9yZSBpdCBpbiBhIGxvY2FsIHJlcG9zaXRvcnkgKERvY2tlcmZpbGUgaXMgdGhlcmUgaW4gdGhlIHJvb3Qgb2YgdGhlIGRpcmVjdG9yeSBpdHNlbGYpCiAgICBkb2NrZXIgYnVpbGQgLiAtLXRhZyBhY2NvdW50LXRyYW5zYWN0aW9uLXNlcnZpY2UKCiMjIyMjIFN0ZXAgNDogUnVuIHRoZSBtaWNyby1zZXJ2aWNlIGFzIGEgY29udGFpbmVyIGluIHRoZSBkb2NrZXI6CiAgICBkb2NrZXIgcnVuIC1pdCAtcCA4MDgwOjgwODAgYWNjb3VudC10cmFuc2FjdGlvbi1zZXJ2aWNlOmxhdGVzdAoKTm90ZTogSSBoYXZlIHVzZWQgbXVsdGktbGF5ZXJlZCBhcHByb2FjaCBpbiBkb2NrZXIgYW5kIHNwcmluZy4KCi0tLQojIyMgT3BlbkFQSSAzLjAgRG9jdW1lbnRhdGlvbiBVUkw6IApodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS9pbmRleC5odG1sP2NvbmZpZ1VybD0vdjMvYXBpLWRvY3Mvc3dhZ2dlci1jb25maWcgIAohW0FQSV9Eb2N1bWVudGF0aW9uIGltZyBtaXNzaW5nXShodHRwczovL2dpdGh1Yi5jb20va2FydGhpa2Fpc2VsdmFuL2FjY291bnQtdHJhbnNhY3Rpb24tc2VydmljZS9ibG9iL21haW4vaW1nL0FQSV9Eb2N1bWVudGF0aW9uLmpwZz9yYXc9dHJ1ZSkKCk5vdGU6IEFsc28gW1Bvc3RtYW4gQ29sbGVjdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL2thcnRoaWthaXNlbHZhbi9hY2NvdW50LXRyYW5zYWN0aW9uLXNlcnZpY2UvYmxvYi9tYWluL2FjY291bnQtdHJhbnNhY3Rpb24tc2VydmljZS5wb3N0bWFuX2NvbGxlY3Rpb24uanNvbikgaGFzIGJlZW4gcHVzaGVkIGludG8gdGhlIHJlcG9zaXRvcnkgZm9yIHlvdXIgcmVmZXJlbmNlLgoKLS0tCiMjIyBJbi1NZW1vcnkgRGF0YWJhc2UgRGFzaGJvYXJkOgpodHRwOi8vbG9jYWxob3N0OjgwODAvaDItY29uc29sZS9sb2dpbi5qc3AgICAKIVtIMl9EQl9Db25zb2xlIGltZyBtaXNzaW5nXShodHRwczovL2dpdGh1Yi5jb20va2FydGhpa2Fpc2VsdmFuL2FjY291bnQtdHJhbnNhY3Rpb24tc2VydmljZS9ibG9iL21haW4vaW1nL0gyX0RCX0NvbnNvbGUuanBnP3Jhdz10cnVlKQoKTm90ZTogVXNlcm5hbWUgYW5kIFBhc3N3b3JkIGZvciB0aGUgaW4tbWVtb3J5IEgyIGRhdGFiYXNlIGNhbiBiZSBzZXQgZGlmZmVyZW50bHkgYnkgcGFzc2luZyBlbnZpcm9ubWVudCB2YXJpYWJsZXMgbmFtZWx5IGBgYERCX1VTRVJgYGAgYW5kIGBgYERCX1BXRGBgYCBpZiByZXF1aXJlZCwgZWxzZSBIMiBkYiBwcm92aWRlZCBkZWZhdWx0IHVzZXJuYW1lIGFuZCBwYXNzd29yZCB3aWxsIGJlIHVzZWQuIEZvciBtb3JlIGRldGFpbHMgcGxlYXNlIHJlZmVyOiBbYXBwbGljYXRpb24ueW1sXShodHRwczovL2dpdGh1Yi5jb20va2FydGhpa2Fpc2VsdmFuL2FjY291bnQtdHJhbnNhY3Rpb24tc2VydmljZS9ibG9iL21haW4vc3JjL21haW4vcmVzb3VyY2VzL2FwcGxpY2F0aW9uLnltbCNMMykKCi0tLQo= readmeEtag: '"6da22f219b8186d10310c0796142308ed599e23b"' readmeLastModified: Wed, 09 Feb 2022 18:18:26 GMT repositoryId: 327923839 description: >- This is a demo micro-service for account transaction as part of mobile challengers banking. created: '2021-01-08T14:22:37Z' updated: '2022-10-05T11:11:04Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: karthikairam logo: https://avatars.githubusercontent.com/u/10527779?v=4 repoEtag: '"c97ab3cc2db0f824ec8a67a601e4ef2f322251c28779fc995cb7bcea62a46f1e"' repoLastModified: Wed, 05 Oct 2022 11:11:04 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/onmax/openapi-graph-core v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJLWdyYXBoLWNvcmUKCkEgVFMgbGlicmFyeSB0byBtYW5hZ2UgbGFyZ2UgQVBJIHByb2plY3RzIGRlZmluZWQgYnkgT3BlbkFQSXYzIHNwZWNpZmljYXRpb24uCgojIyBJbnN0YWxsYXRpb24KCkp1c3QgcnVuCgo+IG5wbSBpbnN0YWxsIG9wZW5hcGktZ3JhcGgtY29yZQoKYW5kIHlvdSBhcmUgZ29vZCB0byBnby4KCiMjIENMSQoKVGhpcyBgUkVBRE1FLm1kYCBpcyBvbmx5IGZvciB0aGUgQVBJIG9mIHRoaXMgbGlicmFyeS4gWW91IGNhbiBjaGVjayB0aGUgY2xpIHRvIHJ1biBpdCBpbiB0aGUgY29uc29sZSBbaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL29ubWF4L29wZW5BUEktZ3JhcGgtY2xpKQoKIyMgQVBJCgpPcGVuQVBJLWdyYXBoIHByb3ZpZGUgdHdvIG1ham9yIGNsYXNzZXMgdGhhdCBhcmUgYWNjZXNpYmxlIGNhbGxlZDogW2BPcGVuQVBJR3JhcGhzYF0oI09wZW5BUElHcmFwaHMpIGFuZCBbYEFuYWx5emVyYF0oI0FuYWx5emVyKS4KCiMjIyBPcGVuQVBJR3JhcGhzCgpUaGlzIGNsYXNzIHdpbGwgY3JlYXRlIHRoZSBncmFwaHMgb2YgdGhlIEFQSXMuIFlvdSBjYW4gcHJvdmlkZSBhIHJvb3QgcGF0aCwgd2hpY2ggd2lsbCBiZSB1c2VkIHRvIGxvb2sgZm9yIHRoZSBBUEkgc3BlY2lmaWNhdGlvbnMuIEZvciBleGFtcGxlLCBhIHZhbGlkIHJvb3QgcGF0aCBpZiB3ZSB1c2Ugb25lIG9mIHRoZSBleGFtcGxlcyB3b3VsZCBiZToKCmBgYGphdmFzY3JpcHQKY29uc3QgT3BlbkFQSUdyYWhDb3JlID0gcmVxdWlyZSgnb3BlbmFwaS1ncmFwaC1jb3JlJyk7CgooYXN5bmMgKCkgPT4gewogICAgY29uc3QgZ3JhcGhzID0gYXdhaXQgbmV3IE9wZW5BUElHcmFoQ29yZS5PcGVuQVBJR3JhcGhzKCcuL3Rlc3RzL3Jlc291cmNlcy9zb2NpYWwtbmV0d29yaycpLmJ1aWxkKCkKCiAgICAvKiBJdCB3aWxsIHJldHVybiAKICAgIAogICAgewogICAgICAgICJzb2NpYWwtbmV0d29yay55YW1sIjogeyBPbWl0dGluZyBncmFwaCAuLi4gfSwKICAgICAgICAiLi9wb3N0cy9wb3N0cy55YW1sIjogeyBPbWl0dGluZyBncmFwaCAuLi4gfSwKICAgICAgICAiLi91c2Vycy91c2Vycy55YW1sIjogeyBPbWl0dGluZyBncmFwaCAuLi4gfSwKICAgIH0KICAgICovCn0pKCk7CmBgYAoKIyMjIEFuYWx5emVyCgpJdCB3aWxsIGFuYWx5emUgdGhlIGdyYXBocyBjaGVja2luZyBkaWZmZXJlbnQgY29uZGl0aW9ucy4gWW91IGNhbiBpbml0aWF6bGllIGFzIGZvbGxvd3MgKHlvdSBuZWVkIHRvIGluaXRpYWxpemUgYE9wZW5BUElHcmFocHNgKToKCmBgYGphdmFzY3JpcHQKY29uc3QgT3BlbkFQSUdyYWhDb3JlID0gcmVxdWlyZSgnb3BlbmFwaS1ncmFwaC1jb3JlJyk7CgooYXN5bmMgKCkgPT4gewogICAgY29uc3QgZ3JhcGhzID0gYXdhaXQgbmV3IE9wZW5BUElHcmFoQ29yZS5PcGVuQVBJR3JhcGhzKCcuL3Rlc3RzL3Jlc291cmNlcy9zb2NpYWwtbmV0d29yaycpLmJ1aWxkKCkKICAgIGNvbnN0IGFuYWx5emVyID0gT3BlbkFQSUdyYWhDb3JlLkFuYWx5emVyKGdyYXBocykKfSkoKTsKYGBgCkZvciBub3csIHRoZXNlIGZ1bmN0aW9uIGhhdmUgYmVlbiBkZXZlbG9wZWQ6CgojIyMjIFVudXNlZCBzY2hlbWFzCgp8IEZ1bmN0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICB8IERlc2NyaXB0aW9uICB8IFJldHVybnMgIHwgCnwtLS18LS0tfC0tLXwKfCBnZXRVbnVzZWRTY2hlbWFzKCkgICAgICAgICAgICAgICAgfCBJdCB3aWxsIGNoZWNrIGFsbCB0aGUgc2NoZW1hcyBkZWNsYXJlZCBidXQgbm90IGJlaW5nIHVzZWQgaW4gdGhlIGNvbXBvbmVudHMuc2NoZW1hcyBjb250YWluZXIgIHwgW0pTT05dKGh0dHBzOi8vZ2l0aHViLmNvbS9vbm1heC9vcGVuYXBpLWdyYXBoLXR5cGVzL2Jsb2IvbWFpbi9zcmMvbW9kZWwvQW5hbHl6ZXIudHMjTDYpICB8CnwgZ2V0RGVwcmVjYXRlZFNjaGVtYXNCZWluZ1VzZWQoKSAgIHwgSXQgd2lsbCBjaGVjayBhbGwgdGhlIGRlcHJlY2F0ZWQgc2NoZW1hcyBkZWNsYXJlZCBiZWluZyBpbiB1c2VkICB8IFtKU09OXShodHRwczovL2dpdGh1Yi5jb20vb25tYXgvb3BlbmFwaS1ncmFwaC10eXBlcy9ibG9iL21haW4vc3JjL21vZGVsL0FuYWx5emVyLnRzI0w3KSAgfAoK readmeEtag: '"e4974465255ad92f22ddc0e63e698f3197b48eab"' readmeLastModified: Fri, 11 Nov 2022 03:27:27 GMT repositoryId: 349407768 description: >- A TS library to manage large API projects defined by OpenAPIv3 specification. created: '2021-03-19T11:59:15Z' updated: '2022-11-11T03:27:33Z' language: TypeScript archived: false stars: 1 watchers: 0 forks: 0 owner: onmax logo: https://avatars.githubusercontent.com/u/22072217?v=4 license: MIT repoEtag: '"f932341ee7364db986b9b25c4c00821cd195e6ce04f355dc600087ef6737e8b3"' repoLastModified: Fri, 11 Nov 2022 03:27:33 GMT foundInMaster: true category: Parsers id: c162d187112cfc2cc016217575c122c3 - source: openapi3 tags repository: https://github.com/devolite/openapi-fetch-angular v3: true id: 0ce4e31b19ca7cbd393a19141db1ae88 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWZldGNoLWFuZ3VsYXIKClRoaXMgbGlicmFyeSBpcyBhbiBbb3BlbmFwaS1mZXRjaF0oaHR0cHM6Ly9naXRodWIuY29tL2Ryd3Bvdy9vcGVuYXBpLXR5cGVzY3JpcHQvdHJlZS9tYWluL3BhY2thZ2VzL29wZW5hcGktZmV0Y2gpIGNsb25lIHRoYXQgdXNlcyBbQW5ndWxhcidzIEhUVFBDbGllbnQgQVBJXShodHRwczovL2FuZ3VsYXIuaW8vYXBpL2NvbW1vbi9odHRwL0h0dHBDbGllbnQpLgpfVGhhbmtzLCBbQGRyd3Bvd10oaHR0cHM6Ly9naXRodWIuY29tL2Ryd3BvdykhXwoKPiBbIUNBVVRJT05dCj4gVGhpcyBsaWJyYXJ5IGlzIHN0aWxsIGluIGRldmVsb3BtZW50LiBQbGVhc2UgcmVwb3J0IGFueSBpc3N1ZXMgeW91IGVuY291bnRlci4KCj4gWyFDQVVUSU9OXQo+IEN1cnJlbnRseSAqKm9ubHkqKiBzdXBwb3J0cyBKU09OIHJlcXVlc3RzIGFuZCByZXNwb25zZXMuCgojIyBVc2FnZQoKSW4gb3JkZXIgdG8gZ2V0IHN0YXJ0ZWQsIGdlbmVyYXRlIGEgc3BlY2lmaWNhdGlvbiBmaWxlIHVzaW5nIFtvcGVuYXBpLXR5cGVzY3JpcHRdKGh0dHBzOi8vZ2l0aHViLmNvbS9kcndwb3cvb3BlbmFwaS10eXBlc2NyaXB0L3RyZWUvbWFpbi9wYWNrYWdlcy9vcGVuYXBpLXR5cGVzY3JpcHQpLgoKYGBgc2gKIyBMb2NhbCBzY2hlbWEuLi4KbnB4IG9wZW5hcGktdHlwZXNjcmlwdCAuL3BhdGgvdG8vbXkvc2NoZW1hLnlhbWwgLW8gLi9wYXRoL3RvL215L3NjaGVtICMgbnBtCnlhcm4gZGx4IG9wZW5hcGktdHlwZXNjcmlwdCAuL3BhdGgvdG8vbXkvc2NoZW1hLmQudHMgLW8gLi9wYXRoL3RvL215L3NjaGVtYS50cyAjIG9yIHlhcm4KcG5wbSBkbHggb3BlbmFwaS10eXBlc2NyaXB0IC4vcGF0aC90by9teS9zY2hlbWEuZC50cyAtbyAuL3BhdGgvdG8vbXkvc2NoZW1hLnRzICMgb3IgcG5wbQojIPCfmoAgLi9wYXRoL3RvL215L3NjaGVtYS55YW1sIC0+IC4vcGF0aC90by9teS9zY2hlbWEuZC50cyBbN21zXQoKIyBSZW1vdGUgc2NoZW1hLi4uCm5weCBvcGVuYXBpLXR5cGVzY3JpcHQgaHR0cHM6Ly9leGFtcGxlLmNvbS9zY2hlbWEueWFtbCAtbyAuL3BhdGgvdG8vbXkvc2NoZW1hICMgbnBtCnlhcm4gZGx4IG9wZW5hcGktdHlwZXNjcmlwdCBodHRwczovL2V4YW1wbGUuY29tL3NjaGVtYS5kLnRzIC1vIC4vcGF0aC90by9teS9zY2hlbWEudHMgIyBvciB5YXJuCnBucG0gZGx4IG9wZW5hcGktdHlwZXNjcmlwdCBodHRwczovL2V4YW1wbGUuY29tL3NjaGVtYS5kLnRzIC1vIC4vcGF0aC90by9teS9zY2hlbWEudHMgIyBvciBwbnBtCiMg8J+agCBodHRwczovL2V4YW1wbGUuY29tL3NjaGVtYS55YW1sIC0+IC4vcGF0aC90by9teS9zY2hlbWEuZC50cyBbN21zXQpgYGAKClRoZW4sIHV0aWxpemUgdGhlIGdlbmVyYXRlZCBzcGVjaWZpY2F0aW9uIGZpbGUgdG8gbWFrZSByZXF1ZXN0cy4gSW4gb3JkZXIgdG8gZG8gdGhpcywgY3JlYXRlIGEgc2VydmljZSBsaWtlIHNvOgoKYGBgdHMKQEluamVjdGFibGUoewogIHByb3ZpZGVkSW46ICJyb290IiwKfSkKZXhwb3J0IGNsYXNzIFRlc3RBUElTZXJ2aWNlIGV4dGVuZHMgT3BlbkFQSUNsaWVudFNlcnZpY2U8cGF0aHM+IHsKICBjb25zdHJ1Y3RvcihodHRwQ2xpZW50OiBIdHRwQ2xpZW50KSB7CiAgICBzdXBlcihodHRwQ2xpZW50LCB7CiAgICAgIGJhc2VVcmw6ICIvYXBpIiwKICAgICAgLy8gLi4uIG9wdGlvbnMKICAgIH0pOwogIH0KfQpgYGAKCk5vdywgeW91IGNhbiB1c2UgdGhlIHNlcnZpY2UgdG8gbWFrZSByZXF1ZXN0cyB0aGF0IG1hdGNoIHRoZSBzcGVjaWZpY2F0aW9uOgoKYGBgdHMKY29uc3RydWN0b3IocHJpdmF0ZSBhcGk6IFRlc3RBUElTZXJ2aWNlKSB7fQoKY29uc3QgeyBkYXRhLCBlcnJvciAvKiwgcmVzcG9uc2UqLyB9ID0gYXdhaXQgdGhpcy5hcGkuZ2V0KCIvcGF0aC90by9lbmRwb2ludCIpOwpgYGAKCkZvciBtb3JlIGluZm9ybWF0aW9uLCBzZWUgdGhlIFtvcGVuYXBpLWZldGNoIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vb3BlbmFwaS10cy5wYWdlcy5kZXYvb3BlbmFwaS1mZXRjaC8pLgoKIyMgQ3JlZGl0cwoKQmlnIHRoYW5rcyB0byBbQGRyd3Bvd10oaHR0cHM6Ly9naXRodWIuY29tL2Ryd3BvdykgZm9yIGNyZWF0aW5nIHRoZSBvcmlnaW5hbCBvcGVuYXBpLWZldGNoIGFuZCBvcGVuYXBpLXR5cGVzY3JpcHQgbGlicmFyeSEK readmeEtag: '"841ec48899de291a8eb32c39c726927303d49367"' readmeLastModified: Thu, 28 Mar 2024 13:16:35 GMT repositoryId: 777763812 description: >- 💡 Fast, typesafe fetch client for your OpenAPI schema. Works with Angular HTTPClient API. created: '2024-03-26T13:17:17Z' updated: '2025-06-22T08:27:13Z' language: TypeScript archived: true stars: 2 watchers: 2 forks: 0 owner: devolite logo: https://avatars.githubusercontent.com/u/164477852?v=4 license: MIT repoEtag: '"2b651e504f1b34fba685d2655ef58b441245cf77c694d1a5d5a3fbc23c8c0ef7"' repoLastModified: Sun, 22 Jun 2025 08:27:13 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/wahyubudii/spring-boot-boilerplate v3: true id: 26beb171cac6a44beda841391499b157 repositoryMetadata: base64Readme: >- IyMgQVBJIERvY3VtZW50YXRpb24gZnJvbSBTd2FnZ2VyIGluIHlvdXIgbG9jYWwKT3BlbiBpbiBicm93c2VyIGBodHRwOi8vbG9jYWxob3N0OjxzZXJ2ZXItcG9ydD4vc3dhZ2dlci11aS9pbmRleC5odG1sYCA8YnI+CnNlcnZlciBwb3J0IHZhbHVlIGJ5IGRlZmF1bHQgaXM6IGA4MDgwYCwgYnV0IHlvdSBjYW4gYWRqdXN0IGluIHRoZSBgL2FwcGxpY2F0aW9uLnltbGAgZmlsZQoKIyMgRGVwZW5kZW5jeQrigJMgSWYgeW91IHdhbnQgdG8gdXNlIFBvc3RncmVTUUw6CmBgYHhtbAo8ZGVwZW5kZW5jeT4KICA8Z3JvdXBJZD5vcmcucG9zdGdyZXNxbDwvZ3JvdXBJZD4KICA8YXJ0aWZhY3RJZD5wb3N0Z3Jlc3FsPC9hcnRpZmFjdElkPgogIDxzY29wZT5ydW50aW1lPC9zY29wZT4KPC9kZXBlbmRlbmN5PgpgYGAK4oCTIG9yIE15U1FMOgpgYGB4bWwKPGRlcGVuZGVuY3k+CiAgPGdyb3VwSWQ+Y29tLm15c3FsPC9ncm91cElkPgogIDxhcnRpZmFjdElkPm15c3FsLWNvbm5lY3Rvci1qPC9hcnRpZmFjdElkPgogIDxzY29wZT5ydW50aW1lPC9zY29wZT4KPC9kZXBlbmRlbmN5PgpgYGAKIyMgQ29uZmlndXJlIFNwcmluZyBEYXRhc291cmNlLCBKUEEsIEFwcCBwcm9wZXJ0aWVzCk9wZW4gYHNyYy9tYWluL3Jlc291cmNlcy9hcHBsaWNhdGlvbi55bWxgCi0gRm9yIFBvc3RncmVTUUw6CmBgYApzcHJpbmc6CiAgZGF0YXNvdXJjZToKICAgIGRyaXZlci1jbGFzcy1uYW1lOiBvcmcucG9zdGdyZXNxbC5Ecml2ZXIKICAgIHVybDogamRiYzpwb3N0Z3Jlc3FsOi8vbG9jYWxob3N0OjU0MzIvZGJuYW1lCiAgICB1c2VybmFtZTogcm9vdAogICAgcGFzc3dvcmQ6IHJvb3QKICBqcGE6CiAgICBoaWJlcm5hdGU6CiAgICAgIGRkbC1hdXRvOiB1cGRhdGUgIyBIaWJlcm5hdGUgZGRsIGF1dG8gKGNyZWF0ZSwgY3JlYXRlLWRyb3AsIHZhbGlkYXRlLCB1cGRhdGUpCiAgICBzaG93LXNxbDogdHJ1ZQogICAgZGF0YWJhc2U6IHBvc3RncmVzcWwKICAgIGRhdGFiYXNlLXBsYXRmb3JtOiBvcmcuaGliZXJuYXRlLmRpYWxlY3QuUG9zdGdyZVNRTERpYWxlY3QKICBzZWN1cml0eToKICAgIGp3dDoKICAgICAgc2VjcmV0OiBqd3RTZWNyZXRLZXkKICAgICAgZXhwaXJhdGlvbjogODY0MDAwMDAgIyAyNCBob3VycwpgYGAKLSBGb3IgTXlTUUwKYGBgCnNwcmluZzoKICBkYXRhc291cmNlOgogICAgZHJpdmVyLWNsYXNzLW5hbWU6IGNvbS5teXNxbC5jai5qZGJjLkRyaXZlcgogICAgdXJsOiBqZGJjOm15c3FsOi8vbG9jYWxob3N0OjMzMDYvZGJuYW1lCiAgICB1c2VybmFtZTogcm9vdAogICAgcGFzc3dvcmQ6IHJvb3QKICBqcGE6CiAgICBoaWJlcm5hdGU6CiAgICAgIGRkbC1hdXRvOiB1cGRhdGUgIyBIaWJlcm5hdGUgZGRsIGF1dG8gKGNyZWF0ZSwgY3JlYXRlLWRyb3AsIHZhbGlkYXRlLCB1cGRhdGUpCiAgICBzaG93LXNxbDogdHJ1ZQogIHNlY3VyaXR5OgogICAgand0OgogICAgICBzZWNyZXQ6IGp3dFNlY3JldEtleQogICAgICBleHBpcmF0aW9uOiA4NjQwMDAwMCAjIDI0IGhvdXJzCmBgYAoKIyMgUnVuIGZvbGxvd2luZyBTUUwgaW5zZXJ0IHN0YXRlbWVudHMK4oCTIG11bHRpIGRhdGEKYGBgeG1sCklOU0VSVCBJTlRPIHJvbGVzIChpZCwgbmFtZSkgVkFMVUVTIAooZ2VuX3JhbmRvbV91dWlkKCksICdST0xFX0FETUlOJyksCihnZW5fcmFuZG9tX3V1aWQoKSwgJ1JPTEVfVVNFUicpLAooZ2VuX3JhbmRvbV91dWlkKCksICdST0xFX01PREVSQVRPUicpOwpgYGAKCuKAkyBzaW5nbGUgZGF0YQpgYGB4bWwKSU5TRVJUIElOVE8gcm9sZXMgKGlkLCBuYW1lKSBWQUxVRVMgCihnZW5fcmFuZG9tX3V1aWQoKSwgJ1JPTEVfTU9ERVJBVE9SJyk7CmBgYAoKRm9yIG1vcmUgZGV0YWlsLCBwbGVhc2UgdmlzaXQ6Cj4gW1NlY3VyZSBTcHJpbmcgQm9vdCB3aXRoIFNwcmluZyBTZWN1cml0eSAmIEpXVCBBdXRoZW50aWNhdGlvbl0oaHR0cHM6Ly9iZXprb2Rlci5jb20vc3ByaW5nLWJvb3Qtand0LWF1dGhlbnRpY2F0aW9uLykK readmeEtag: '"622f69b33734997e65696e2df50d4cb46965c838"' readmeLastModified: Sat, 01 Feb 2025 14:50:42 GMT repositoryId: 925269588 description: >- Spring Boot v3, Spring Security v6, JWT, Swagger, OpenAPI v3, Roles Based, JPA, Lombok, Validation, PostgreSQL created: '2025-01-31T15:01:24Z' updated: '2025-02-01T14:50:46Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: wahyubudii logo: https://avatars.githubusercontent.com/u/79882296?v=4 repoEtag: '"d26c82815d78ecd8e814dde48d2f91a0bb0ae80b2980a1d6d46829aa368db9da"' repoLastModified: Sat, 01 Feb 2025 14:50:46 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/nitesh232/rest-api_documentation_with_openapi_3 v3: true id: fdadc87cbad758bec90c823d63cf13f5 repositoryMetadata: base64Readme: >- IyBSRVNULUFQSV9Eb2N1bWVudGF0aW9uX3dpdGhfT3BlbkFQSV8zCgpUaGlzIHByb2plY3Qgc2hvd3MgdGhlIGRvY3VtZW50YXRpb24gb2YgUkVTVF9BUElzIGJ5IHVzaW5nIE9wZW5BcGlfMyBzcGVjaWZpY2F0aW9uLgoKRHVyaW5nIHRoaXMgcHJvamVjdCBJIGhhdmUgdXNlZCB0aGUgVGlja2V0IEJvb2tpbmcgUkVTVC1BUEkgd2hpY2ggSSBoYXZlIGNyZWF0ZWQgZWFybGllciB5b3UgY2FuIGZpbmQgW2hlcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9OaXRlc2gyMzIvUkVTVC1BUElfRm9yX1RpY2tldF9Cb29raW5nKSAuCgoKUkVTVCBkb2N1bWVudGF0aW9uIGNhbiBiZSBmb3VuZCB0byBVUkwgOiBodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS9pbmRleC5odG1sCgoKQmVsb3cgSSBhbSBhdHRhY2hpbmcgaW1hZ2Ugb2YgZG9jdW1lbnRhdGlvbi4KCjxpbWcgc3JjPWh0dHBzOi8vZ2l0aHViLmNvbS9OaXRlc2gyMzIvUkVTVC1BUElfRG9jdW1lbnRhdGlvbl93aXRoX09wZW5BUElfMy9ibG9iL21haW4vc3dhZ2dlci5QTkcgd2lkdGg9IjEzMDAiID4KCgpCZWxvdyBpcyBhdHRhY2hlZCBpbWFnZSBmb3IgZW5kcG9pbnQgd2l0aCBQT1NUIG1ldGhvZC4KCgo8aW1nIHNyYz1odHRwczovL2dpdGh1Yi5jb20vTml0ZXNoMjMyL1JFU1QtQVBJX0RvY3VtZW50YXRpb25fd2l0aF9PcGVuQVBJXzMvYmxvYi9tYWluL3Bvc3QuUE5HIHdpZHRoPSIxMzAwIiA+Cg== readmeEtag: '"bbe624cac51cdc4c305f4f4de6b4cdcf1197b122"' readmeLastModified: Sun, 02 Jul 2023 08:56:42 GMT repositoryId: 661241530 description: >- This project shows the documentation of REST_APIs by using OpenApi_3 specification. created: '2023-07-02T08:26:09Z' updated: '2023-07-02T09:06:54Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Nitesh232 logo: https://avatars.githubusercontent.com/u/92843270?v=4 repoEtag: '"08203481297cd1ec0204185fe8f8e4f5c9b4938e0599120962bf646dc1646402"' repoLastModified: Sun, 02 Jul 2023 09:06:54 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/ekoulemaneng/openapi-inputs-validator v3: true id: dfbd7e4a55a13df24b1ec2dc9f31970e repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIElucHV0cyBWYWxpZGF0b3IKQW4gdW5vcGluaW9uYXRlZCBUeXBlU2NyaXB0IGxpYnJhcnkgdGhhdCBwcm92aWRlcyBpbnB1dHMgZGF0YSB2YWxpZGF0b3JzIGluIGNvbXBsaWFuY2Ugd2l0aCBhIGdpdmVuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbi4gRm9yIDMuMS54IGFuZCAzLjAueCBPcGVuQVBJIHZlcnNpb25zLgoKIyMgSW5zdGFsbGF0aW9uCgpgYGAKbnBtIGluc3RhbGwgb3BlbmFwaS1pbnB1dHMtdmFsaWRhdG9yCmBgYApvciAKYGBgCnlhcm4gYWRkIG9wZW5hcGktaW5wdXRzLXZhbGlkYXRvcgpgYGAKCiMjIFVzYWdlCgpgYGB0eXBlc2NyaXB0CmltcG9ydCBzY2hlbWFSZXRyaWV2ZXIgZnJvbSAnb3BlbmFwaS1zY2hlbWEtcmV0cmlldmVyJwppbXBvcnQgeyBjb21waWxlLCB2YWxpZGF0ZSB9IGZyb20gJ29wZW5hcGktaW5wdXRzLXZhbGlkYXRvcicKCmFzeW5jIGZ1bmN0aW9uIG1haW4oKSB7CgogICAgLy8gU2V0dXAgYW4gb3BlbmFwaSBzcGVjaWZpY2F0aW9uCiAgICBjb25zdCBzY2hlbWEgPSBhd2FpdCBzY2hlbWFSZXRyaWV2ZXIoJy4vb3BlbmFwaS55YW1sJywgX19kaXJuYW1lKQoKICAgIC8vIEdlbmVyYXRlIHZhbGlkYXRvcnMgZnJvbSBhbiBvcGVuYXBpIHNwZWNpZmljYXRpb24KICAgIGNvbnN0IHZhbGlkYXRvcnMgPSBhd2FpdCBjb21waWxlKHNjaGVtYSkKCiAgICAvLyBTZXQgcGFyYW1ldGVycwogICAgY29uc3QgcGF0aCA9ICcvdXNlcnMnCiAgICBjb25zdCBvcGVyYXRpb24gPSAnZ2V0JwogICAgY29uc3QgaW5wdXRzID0gewogICAgICAgIHBhdGg6IHsKICAgICAgICAgICAgLy8gLi4uLi4uLi4KICAgICAgICB9LAogICAgICAgIGhlYWRlcnM6IHsKICAgICAgICAgICAgLy8gLi4uLi4uLgogICAgICAgIH0sCiAgICAgICAgY29va2llczogewogICAgICAgICAgICAvLyAuLi4uLi4uCiAgICAgICAgfSwKICAgICAgICBxdWVyeTogewogICAgICAgICAgICAvLyAuLi4uLi4uCiAgICAgICAgfSwKICAgICAgICBib2R5OiAvLyAuLi4uLi4KICAgIH0KCiAgICAvLyBWYWxpZGF0ZQogICAgY29uc3QgdmFsaWRhdGlvblJlc3VsdCA9IHZhbGlkYXRlKHBhdGgsIG9wZXJhdGlvbiwgaW5wdXRzLCB2YWxpZGF0b3JzKQoKICAgIGNvbnNvbGUubG9nKHZhbGlkYXRpb25SZXN1bHQpIC8vID0+IHsgdmFsaWQ6IGJvb2xlYW4sIGVycm9yczogbnVsbCB8IHN0cmluZyB8IHsuLi59IHwgdW5kZWZpbmVkIH0KCn0KYGBgCgojIyBMaWNlbnNlClRoaXMgcGFja2FnZSBpcyBsaWNlbnNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL21pdCkuCgojIyBDb250YWN0CklmIHlvdSBoYXZlIGFueSBxdWVzdGlvbnMgb3IgaXNzdWVzLCBwbGVhc2UgY29udGFjdCB0aGUgcGFja2FnZSBtYWludGFpbmVyIGF0IGVrb3VsZW1hbmVuZ0BnbWFpbC5jb20u readmeEtag: '"3c897dd652961a605e77ecf982cf951221a054de"' readmeLastModified: Mon, 15 May 2023 17:25:55 GMT repositoryId: 633462766 description: >- TypeScript library providing inputs data validators in accordance with a given OpenAPI specification. For 3.1.x and 3.0.x OpenAPI versions. created: '2023-04-27T14:50:04Z' updated: '2023-05-11T12:30:16Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: ekoulemaneng logo: https://avatars.githubusercontent.com/u/12666431?v=4 license: MIT repoEtag: '"228ebb832c6fd9c05d2b9360fc1fe168221948cb102ed8d8f6c19e79407a0e48"' repoLastModified: Thu, 11 May 2023 12:30:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/refitter/generate-nupkg v3: true id: 8e8df31870ab1dad33f850d25cd99557 repositoryMetadata: base64Readme: >- IyBHZW5lcmF0ZS1OdXBrZwpHZW5lcmF0ZSBhIFJlZml0IGNsaWVudCBpbnRlcmZhY2UgYW5kIGNvbnRyYWN0cyBmcm9tIGFuIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgZG9jdW1lbnQgdXNpbmcgW1JlZml0dGVyXShodHRwczovL2dpdGh1Yi5jb20vY2hyaXN0aWFuaGVsbGUvcmVmaXR0ZXIpLgoKIyMgSW5wdXRzCgojIyMgYG9wZW5hcGktZmlsZWAKVGhlIHBhdGggdG8gdGhlIE9wZW5BUEkgZG9jdW1lbnQgKGJvdGggSlNPTiBhbmQgWUFNTCBhcmUgc3VwcG9ydGVkKS4gRGVmYXVsdHMgdG8gIm9wZW5hcGkuanNvbiIgKGkuZS4gYSBmaWxlIGluIHRoZSBjdXJyZW50IGRpcmVjdG9yeSBjYWxsZWQgb3BlbmFwaS5qc29uKS4gUGF0aHMgdGhhdCBkbyBub3Qgc3RhcnQgd2l0aCBgL2AgYXJlIGFzc3VtZWQgdG8gYmUgcmVsYXRpdmUgdG8gdGhlIHJvb3Qgb2YgdGhlIHJlcG9zaXRvcnkuCgojIyMgYG9wZW5hcGktdXJsYApUaGUgVVJMIHRvIGxvYWQgdGhlIE9wZW5BUEkgZG9jdW1lbnQgZnJvbS4gSWYgc2V0LCBgb3BlbmFwaS1maWxlYCB3aWxsIGJlIGlnbm9yZWQuCgojIyMgYG5hbWVzcGFjZWAKVGhlIGRlZmF1bHQgbmFtZXNwYWNlIHVzZWQgZm9yIHRoZSBnZW5lcmF0ZWQgdHlwZXMgKGRlZmF1bHQ6IGBHZW5lcmF0ZWRDb2RlYCkKCiMjIyBgdXNlLWFwaS1yZXNwb25zZWAKUmV0dXJuIGBUYXNrPElBcGlSZXNwb25zZTxUPj5gIGluc3RlYWQgb2YgYFRhc2s8VD5gCgojIyMgYGNhbmNlbGxhdGlvbi10b2tlbnNgClVzZSBjYW5jZWxsYXRpb24gdG9rZW5zCgojIyMgYG11bHRpcGxlLWludGVyZmFjZXNgCkdlbmVyYXRlIGEgUmVmaXQgaW50ZXJmYWNlIGZvciBlYWNoIGVuZHBvaW50LiBNYXkgYmUgb25lIG9mIGBCeUVuZHBvaW50YCwgYEJ5VGFnYAoKIyMjIGBjb21tYW5kLWFyZ3NgCkFkZGl0aW9uYWwgYXJndW1lbnRzIHRvIHBhc3MgdGhyb3VnaCB0byB0aGUgW1JlZml0dGVyXShodHRwczovL2dpdGh1Yi5jb20vY2hyaXN0aWFuaGVsbGUvcmVmaXR0ZXIpIENMSSB0b29sLgoKIyMjIGBwdWJsaXNoLWFydGlmYWN0c2AKUHVibGlzaCB0aGUgZ2VuZXJhdGVkIG51Z2V0IHBhY2thZ2UgYXMgYSBidWlsZCBhcnRpZmFjdAoKIyMjIGBvdXRwdXQtZmlsZW5hbWVgCk5vIG91dHB1dHMgYXJlIHJldHVybmVkLiBUaGUgZ2VuZXJhdGVkIGNsaWVudCBpcyBwbGFjZWQgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5IGFuZCB1c2luZyB0aGUgYG91dHB1dC1maWxlbmFtZWAgKERlZmF1bHQ6ICoqYE91dHB1dC5jc2AqKikgdmFsdWUuIFRoZSBvdXRwdXQgZmlsZSBjb250YWlucyBib3RoIHRoZSBSZWZpdCBpbnRlcmZhY2UgYW5kIHRoZSBjb250cmFjdCB0eXBlcyB1c2VkIGJ5IHRoZSBBUEkKCiMjIyBgdmVyc2lvbmAKVGhlIHZlcnNpb24gbnVtYmVyIHVzZWQgZm9yIHRoZSBOdUdldCBwYWNrYWdlIChkZWZhdWx0OiBgMS4wLiR7eyBnaXRodWIucnVuX251bWJlciB9fWApCgojIyMgYHRhcmdldC1mcmFtZXdvcmtgClRoZSB0YXJnZXQgZnJhbWV3b3JrIHVzZWQgaW4gdGhlIGdlbmVyYXRlZCBDbGllbnQgU0RLIChkZWZhdWx0OiBgbmV0Ni4wYCkKCiMjIyBgcGFja2FnZS1pZGAKVGhlIHZhbHVlIHVzZWQgYXMgYDxQYWNrYWdlSWQ+YCBmb3IgcGFja2FnZSB0aGUgZ2VuZXJhdGVkIGNvZGUgaW50byBhIE51R2V0IHBhY2thZ2UKCiMjIyBgdGl0bGVgClRoZSB2YWx1ZSB1c2VkIGFzIGA8VGl0bGU+YCBmb3IgcGFja2FnZSB0aGUgZ2VuZXJhdGVkIGNvZGUgaW50byBhIE51R2V0IHBhY2thZ2UKCiMjIyBgcm9vdC1uYW1lc3BhY2VgClRoZSB2YWx1ZSB1c2VkIGFzIGA8Um9vdE5hbWVzcGFjZT5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIC0gYSBOdUdldCBwYWNrYWdlCgojIyMgYGFzc2VtYmx5YApUaGUgdmFsdWUgdXNlZCBhcyBgPEFzc2VtYmx5TmFtZT5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgLSBOdUdldCBwYWNrYWdlCgojIyMgYGF1dGhvcnNgClRoZSB2YWx1ZSB1c2VkIGFzIGA8QXV0aG9ycz5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgLSBwYWNrYWdlCgojIyMgYGNvbXBhbnlgClRoZSB2YWx1ZSB1c2VkIGFzIGA8Q29tcGFueT5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgLSBwYWNrYWdlCgojIyMgYHByb2R1Y3RgClRoZSB2YWx1ZSB1c2VkIGFzIGA8UHJvZHVjdD5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgLSBwYWNrYWdlCgojIyMgYGRlc2NyaXB0aW9uYApUaGUgdmFsdWUgdXNlZCBhcyBgPERlc2NyaXB0aW9uPmAgZm9yIHBhY2thZ2UgdGhlIGdlbmVyYXRlZCBjb2RlIGludG8gYSAtIE51R2V0IHBhY2thZ2UKCiMjIyBgbGljZW5zZWAKVGhlIHZhbHVlIHVzZWQgYXMgYDxQYWNrYWdlTGljZW5zZUV4cHJlc3Npb24+YCBmb3IgcGFja2FnZSB0aGUgZ2VuZXJhdGVkIGNvZGUgLSBpbnRvIGEgTnVHZXQgcGFja2FnZQoKIyMjIGBwcm9qZWN0LXVybGAKVGhlIHZhbHVlIHVzZWQgYXMgYDxQYWNrYWdlUHJvamVjdFVybD5gIGFuZCBgPFJlcG9zaXRvcnlVcmw+YCBmb3IgcGFja2FnZSAtIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgcGFja2FnZQoKIyMjIGByZXBvc2l0b3J5LXR5cGVgClRoZSB2YWx1ZSB1c2VkIGFzIGA8UmVwb3NpdG9yeVR5cGU+YCBmb3IgcGFja2FnZSB0aGUgZ2VuZXJhdGVkIGNvZGUgaW50byBhIE51R2V0IHBhY2thZ2UKCgojIEV4YW1wbGVzCgojIyMgVXNpbmcgYSBVUkwgYW5kIHByb2R1Y2luZyBhIENsaWVudCBTREsgTnVHZXQgcGFja2FnZQoKYGBgeWFtbApqb2JzOgogIHNtb2tlLXRlc3QtdXJsLXdpdGgtY2xpZW50LXNkazoKICAgIHJ1bnMtb246IHVidW50dS1sYXRlc3QKICAgIHN0ZXBzOgogICAgICAtIHVzZXM6IGFjdGlvbnMvY2hlY2tvdXRAdjMKICAgICAgCiAgICAgICMgVXNlIHRoZSBhY3Rpb24gdG8gZ2VuZXJhdGUgYSBSZWZpdCBjbGllbnQgaW50ZXJmYWNlCiAgICAgICMgVGhpcyBwcm9kdWNlcyBhIE51R2V0IHBhY2thZ2UgYXMgYSBidWlsZCBhcnRpZmFjdAogICAgICAtIHVzZXM6IHJlZml0dGVyL2dlbmVyYXRlLW51cGtnQHYxCiAgICAgICAgbmFtZTogR2VuZXJhdGUgUmVmaXQgQ2xpZW50IFNESwogICAgICAgIHdpdGg6CiAgICAgICAgICBvcGVuYXBpLXVybDogaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uCiAgICAgICAgICBuYW1lc3BhY2U6IENocmlzdGlhbkhlbGxlLkV4YW1wbGVzLlBldHN0b3JlLnYzCiAgICAgICAgICBjbGllbnQtc2RrOiB0cnVlCiAgICAgICAgICB2ZXJzaW9uOiAzLjAuJHt7IGdpdGh1Yi5ydW5fbnVtYmVyIH19CiAgICAgICAgICBwYWNrYWdlLWlkOiBDaHJpc3RpYW5IZWxsZS5FeGFtcGxlcy5QZXRzdG9yZS52MwogICAgICAgICAgdGl0bGU6IENocmlzdGlhbkhlbGxlLkV4YW1wbGVzLlBldHN0b3JlLnYzCiAgICAgICAgICByb290LW5hbWVzcGFjZTogQ2hyaXN0aWFuSGVsbGUuRXhhbXBsZXMuUGV0c3RvcmUudjMKICAgICAgICAgIGFzc2VtYmx5OiBDaHJpc3RpYW5IZWxsZS5FeGFtcGxlcy5QZXRzdG9yZS52MwogICAgICAgICAgcHJvZHVjdDogQ2hyaXN0aWFuSGVsbGUuRXhhbXBsZXMuUGV0c3RvcmUudjMKICAgICAgICAgIGF1dGhvcnM6IENocmlzdGlhbiBSZXNtYSBIZWxsZQogICAgICAgICAgY29tcGFueTogQ2hyaXN0aWFuIFJlc21hIEhlbGxlCiAgICAgICAgICBkZXNjcmlwdGlvbjogRXhhbXBsZSBnZW5lcmF0ZWQgY29kZSB1c2luZyBSZWZpdHRlciBhbmQgdGhlIFN3YWdnZXIgUGV0c3RvcmUgdjMgZXhhbXBsZSBPcGVuQVBJIHNwZWNpZmljYXRpb25zCiAgICAgICAgICBwcm9qZWN0LXVybDogaHR0cHM6Ly9naXRodWIuY29tL2NocmlzdGlhbmhlbGxlL3JlZml0dGVyCmBgYAo= readmeEtag: '"18958a595b0c2bf07721ee7267c591d3d7fa7224"' readmeLastModified: Tue, 08 Aug 2023 11:30:22 GMT repositoryId: 671892173 description: >- Generate NuGet package containing a Refit client interface and contracts from an OpenAPI specifications document using Refitter. created: '2023-07-28T11:37:10Z' updated: '2023-07-28T11:59:23Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: Refitter logo: https://avatars.githubusercontent.com/u/140486945?v=4 license: GPL-3.0 repoEtag: '"246b079f25682d27edd7f6ced60cc8a67ffaf901538a30dc4d2989ed6cce9d53"' repoLastModified: Fri, 28 Jul 2023 11:59:23 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/noplagiarism/openvk-openapi v3: true id: f7ace7da5bef860cd67f73660240e79f repositoryMetadata: base64Readme: >- IyBvcGVudmstb3BlbmFwaQpPcGVuLUFwaSBvZiBPcGVuVksuIFVzZSBbSW5zb21uaWFdKGh0dHBzOi8vaW5zb21uaWEucmVzdC8pIHRvIGltcG9ydCBpdAo= readmeEtag: '"059278c11c782edcd06cf91f76879100e9cbec07"' readmeLastModified: Tue, 31 Jan 2023 08:26:38 GMT repositoryId: 579141510 description: Open-Api of OpenVK. Use insomnia to use it created: '2022-12-16T19:10:06Z' updated: '2022-12-16T19:23:58Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: NoPlagiarism logo: https://avatars.githubusercontent.com/u/37241775?v=4 repoEtag: '"92c60d0b201d4c5c720f853df49d01eed3a46222586776277499b3cd3dc47f3b"' repoLastModified: Fri, 16 Dec 2022 19:23:58 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/christianhelle/refitter-action v3: true id: 896749e54ed5a10569450d472c645c45 repositoryMetadata: base64Readme: >- IyBSZWZpdHRlciBHaXRodWIgQWN0aW9uCkdlbmVyYXRlIGEgUmVmaXQgY2xpZW50IGludGVyZmFjZSBhbmQgY29udHJhY3RzIGZyb20gYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBkb2N1bWVudCB1c2luZyBbUmVmaXR0ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9jaHJpc3RpYW5oZWxsZS9yZWZpdHRlcikuCgojIyBJbnB1dHMKCiMjIyBgb3BlbmFwaS1maWxlYApUaGUgcGF0aCB0byB0aGUgT3BlbkFQSSBkb2N1bWVudCAoYm90aCBKU09OIGFuZCBZQU1MIGFyZSBzdXBwb3J0ZWQpLiBEZWZhdWx0cyB0byAib3BlbmFwaS5qc29uIiAoaS5lLiBhIGZpbGUgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5IGNhbGxlZCBvcGVuYXBpLmpzb24pLiBQYXRocyB0aGF0IGRvIG5vdCBzdGFydCB3aXRoIGAvYCBhcmUgYXNzdW1lZCB0byBiZSByZWxhdGl2ZSB0byB0aGUgcm9vdCBvZiB0aGUgcmVwb3NpdG9yeS4KCiMjIyBgb3BlbmFwaS11cmxgClRoZSBVUkwgdG8gbG9hZCB0aGUgT3BlbkFQSSBkb2N1bWVudCBmcm9tLiBJZiBzZXQsIGBvcGVuYXBpLWZpbGVgIHdpbGwgYmUgaWdub3JlZC4KCiMjIyBgbmFtZXNwYWNlYApUaGUgZGVmYXVsdCBuYW1lc3BhY2UgdXNlZCBmb3IgdGhlIGdlbmVyYXRlZCB0eXBlcyAoZGVmYXVsdDogYEdlbmVyYXRlZENvZGVgKQoKIyMjIGB1c2UtYXBpLXJlc3BvbnNlYApSZXR1cm4gYFRhc2s8SUFwaVJlc3BvbnNlPFQ+PmAgaW5zdGVhZCBvZiBgVGFzazxUPmAKCiMjIyBgY2FuY2VsbGF0aW9uLXRva2Vuc2AKVXNlIGNhbmNlbGxhdGlvbiB0b2tlbnMKCiMjIyBgY29tbWFuZC1hcmdzYApBZGRpdGlvbmFsIGFyZ3VtZW50cyB0byBwYXNzIHRocm91Z2ggdG8gdGhlIFtSZWZpdHRlcl0oaHR0cHM6Ly9naXRodWIuY29tL2NocmlzdGlhbmhlbGxlL3JlZml0dGVyKSBDTEkgdG9vbC4KCiMjIyBgcHVibGlzaC1hcnRpZmFjdHNgClNldHRpbmcgdGhpcyB3aWxsIHB1Ymxpc2ggdGhlIGdlbmVyYXRlZCBjb2RlIGFzIEMjIGZpbGVzIGFzIGJ1aWxkIGFydGlmYWN0cwoKIyMjIGBjbGllbnQtc2RrYApTZXR0aW5nIHRoaXMgd2lsbCBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgcGFja2FnZSBhbmQgcHVibGlzaCB0aGlzIGFzIGEgYnVpbGQgYXJ0aWZhY3QuIFRoZSBmb2xsb3dpbmcgaW5wdXRzIGFyZSAqKipSZXF1aXJlZCoqKiB3aGVuIGBjbGllbnQtc2RrYCBpcyBlbmFibGVkCgotICMjIyBgY2xpZW50LXNkay12ZXJzaW9uYCAtIFRoZSB2ZXJzaW9uIG51bWJlciB1c2VkIGZvciB0aGUgTnVHZXQgcGFja2FnZSAoZGVmYXVsdDogYDEuMC4ke3sgZ2l0aHViLnJ1bl9udW1iZXIgfX1gKQotICMjIyBgY2xpZW50LXNkay10YXJnZXQtZnJhbWV3b3JrYCAtIFRoZSB0YXJnZXQgZnJhbWV3b3JrIHVzZWQgaW4gdGhlIGdlbmVyYXRlZCBDbGllbnQgU0RLIChkZWZhdWx0OiBgbmV0Ni4wYCkKLSAjIyMgYGNsaWVudC1zZGstcGFja2FnZS1pZGAgLSBUaGUgdmFsdWUgdXNlZCBhcyBgPFBhY2thZ2VJZD5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgcGFja2FnZQotICMjIyBgY2xpZW50LXNkay10aXRsZWAgLSBUaGUgdmFsdWUgdXNlZCBhcyBgPFRpdGxlPmAgZm9yIHBhY2thZ2UgdGhlIGdlbmVyYXRlZCBjb2RlIGludG8gYSBOdUdldCBwYWNrYWdlCi0gIyMjIGBjbGllbnQtc2RrLXJvb3QtbmFtZXNwYWNlYCAtIFRoZSB2YWx1ZSB1c2VkIGFzIGA8Um9vdE5hbWVzcGFjZT5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIC0gYSBOdUdldCBwYWNrYWdlCi0gIyMjIGBjbGllbnQtc2RrLWFzc2VtYmx5YCAtIFRoZSB2YWx1ZSB1c2VkIGFzIGA8QXNzZW1ibHlOYW1lPmAgZm9yIHBhY2thZ2UgdGhlIGdlbmVyYXRlZCBjb2RlIGludG8gYSAtIE51R2V0IHBhY2thZ2UKLSAjIyMgYGNsaWVudC1zZGstYXV0aG9yc2AgLSBUaGUgdmFsdWUgdXNlZCBhcyBgPEF1dGhvcnM+YCBmb3IgcGFja2FnZSB0aGUgZ2VuZXJhdGVkIGNvZGUgaW50byBhIE51R2V0IC0gcGFja2FnZQotICMjIyBgY2xpZW50LXNkay1jb21wYW55YCAtIFRoZSB2YWx1ZSB1c2VkIGFzIGA8Q29tcGFueT5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgTnVHZXQgLSBwYWNrYWdlCi0gIyMjIGBjbGllbnQtc2RrLXByb2R1Y3RgIC0gVGhlIHZhbHVlIHVzZWQgYXMgYDxQcm9kdWN0PmAgZm9yIHBhY2thZ2UgdGhlIGdlbmVyYXRlZCBjb2RlIGludG8gYSBOdUdldCAtIHBhY2thZ2UKLSAjIyMgYGNsaWVudC1zZGstZGVzY3JpcHRpb25gIC0gVGhlIHZhbHVlIHVzZWQgYXMgYDxEZXNjcmlwdGlvbj5gIGZvciBwYWNrYWdlIHRoZSBnZW5lcmF0ZWQgY29kZSBpbnRvIGEgLSBOdUdldCBwYWNrYWdlCi0gIyMjIGBjbGllbnQtc2RrLWxpY2Vuc2VgIC0gVGhlIHZhbHVlIHVzZWQgYXMgYDxQYWNrYWdlTGljZW5zZUV4cHJlc3Npb24+YCBmb3IgcGFja2FnZSB0aGUgZ2VuZXJhdGVkIGNvZGUgLSBpbnRvIGEgTnVHZXQgcGFja2FnZQotICMjIyBgY2xpZW50LXNkay1wcm9qZWN0LXVybGAgLSBUaGUgdmFsdWUgdXNlZCBhcyBgPFBhY2thZ2VQcm9qZWN0VXJsPmAgYW5kIGA8UmVwb3NpdG9yeVVybD5gIGZvciBwYWNrYWdlIC0gdGhlIGdlbmVyYXRlZCBjb2RlIGludG8gYSBOdUdldCBwYWNrYWdlCi0gIyMjIGBjbGllbnQtc2RrLXJlcG9zaXRvcnktdHlwZWAgLSBUaGUgdmFsdWUgdXNlZCBhcyBgPFJlcG9zaXRvcnlUeXBlPmAgZm9yIHBhY2thZ2UgdGhlIGdlbmVyYXRlZCBjb2RlIGludG8gYSBOdUdldCBwYWNrYWdlCgojIyBPdXRwdXRzCk5vIG91dHB1dHMgYXJlIHJldHVybmVkLiBUaGUgZ2VuZXJhdGVkIGNsaWVudCBpcyBwbGFjZWQgaW4gdGhlIGN1cnJlbnQgZGlyZWN0b3J5IGFuZCBjYWxsZWQgKipPdXRwdXQuY3MqKiB3aGljaCBjb250YWlucyBib3RoIHRoZSBSZWZpdCBpbnRlcmZhY2UgYW5kIHRoZSBjb250cmFjdCB0eXBlcyB1c2VkIGJ5IHRoZSBBUEkKCgojIEV4YW1wbGVzCgojIyMgVXNpbmcgYSBGaWxlIGZvciBPcGVuQVBJIHNwZWNpZmljYXRpb25zIGRvY3VtZW50CgpgYGB5YW1sCmpvYnM6CiAgc21va2UtdGVzdC11cmw6CiAgICBydW5zLW9uOiB1YnVudHUtbGF0ZXN0CiAgICBzdGVwczoKICAgICAgLSB1c2VzOiBhY3Rpb25zL2NoZWNrb3V0QHYzCiAgICAgIAogICAgICAjIFVzZSB0aGUgYWN0aW9uIHRvIGdlbmVyYXRlIGEgUmVmaXQgY2xpZW50IGludGVyZmFjZQogICAgICAjIFRoaXMgcHJvZHVjZXMgYSBmaWxlIGNhbGxlZCBPdXRwdXQuY3MgYXMgYSBidWlsZCBhcnRpZmFjdAogICAgICAtIHVzZXM6IGNocmlzdGlhbmhlbGxlL3JlZml0dGVyLWFjdGlvbkBtYWluCiAgICAgICAgbmFtZTogR2VuZXJhdGUgUmVmaXQgQ2xpZW50CiAgICAgICAgd2l0aDogICAgICAgIAogICAgICAgICAgb3BlbmFwaS1maWxlOiBvcGVuYXBpLmpzb24KICAgICAgICAgIG5hbWVzcGFjZTogQ2hyaXN0aWFuSGVsbGUuRXhhbXBsZXMuUGV0c3RvcmUudjMKICAgICAgCiAgICAgICMgRG8gc29tZXRoaW5nIHdpdGggdGhlIGdlbmVyYXRlZCBjbGllbnQgKGxpa2UgaW5jbHVkZSBpdCBpbiBhbiBleGlzdGluZyBwcm9qZWN0KQogICAgICAtIHJ1bjogR2V0LUNvbnRlbnQgT3V0cHV0LmNzIHwgV3JpdGUtSG9zdAogICAgICAgIHNoZWxsOiBwd3NoCmBgYAoKIyMjIFVzaW5nIGEgVVJMIGFuZCBwcm9kdWNpbmcgYSBDbGllbnQgU0RLIE51R2V0IHBhY2thZ2UKCmBgYHlhbWwKam9iczoKICBzbW9rZS10ZXN0LXVybC13aXRoLWNsaWVudC1zZGs6CiAgICBydW5zLW9uOiB1YnVudHUtbGF0ZXN0CiAgICBzdGVwczoKICAgICAgLSB1c2VzOiBhY3Rpb25zL2NoZWNrb3V0QHYzCiAgICAgIAogICAgICAjIFVzZSB0aGUgYWN0aW9uIHRvIGdlbmVyYXRlIGEgUmVmaXQgY2xpZW50IGludGVyZmFjZQogICAgICAjIFRoaXMgcHJvZHVjZXMgYSBOdUdldCBwYWNrYWdlIGFzIGEgYnVpbGQgYXJ0aWZhY3QKICAgICAgLSB1c2VzOiBjaHJpc3RpYW5oZWxsZS9yZWZpdHRlci1hY3Rpb25AbWFpbgogICAgICAgIG5hbWU6IEdlbmVyYXRlIFJlZml0IENsaWVudCBTREsKICAgICAgICB3aXRoOgogICAgICAgICAgb3BlbmFwaS11cmw6IGh0dHBzOi8vcGV0c3RvcmUzLnN3YWdnZXIuaW8vYXBpL3YzL29wZW5hcGkuanNvbgogICAgICAgICAgbmFtZXNwYWNlOiBDaHJpc3RpYW5IZWxsZS5FeGFtcGxlcy5QZXRzdG9yZS52MwogICAgICAgICAgY2xpZW50LXNkazogdHJ1ZQogICAgICAgICAgY2xpZW50LXNkay12ZXJzaW9uOiAzLjAuJHt7IGdpdGh1Yi5ydW5fbnVtYmVyIH19CiAgICAgICAgICBjbGllbnQtc2RrLXBhY2thZ2UtaWQ6IENocmlzdGlhbkhlbGxlLkV4YW1wbGVzLlBldHN0b3JlLnYzCiAgICAgICAgICBjbGllbnQtc2RrLXRpdGxlOiBDaHJpc3RpYW5IZWxsZS5FeGFtcGxlcy5QZXRzdG9yZS52MwogICAgICAgICAgY2xpZW50LXNkay1yb290LW5hbWVzcGFjZTogQ2hyaXN0aWFuSGVsbGUuRXhhbXBsZXMuUGV0c3RvcmUudjMKICAgICAgICAgIGNsaWVudC1zZGstYXNzZW1ibHk6IENocmlzdGlhbkhlbGxlLkV4YW1wbGVzLlBldHN0b3JlLnYzCiAgICAgICAgICBjbGllbnQtc2RrLXByb2R1Y3Q6IENocmlzdGlhbkhlbGxlLkV4YW1wbGVzLlBldHN0b3JlLnYzCiAgICAgICAgICBjbGllbnQtc2RrLWF1dGhvcnM6IENocmlzdGlhbiBSZXNtYSBIZWxsZQogICAgICAgICAgY2xpZW50LXNkay1jb21wYW55OiBDaHJpc3RpYW4gUmVzbWEgSGVsbGUKICAgICAgICAgIGNsaWVudC1zZGstZGVzY3JpcHRpb246IEV4YW1wbGUgZ2VuZXJhdGVkIGNvZGUgdXNpbmcgUmVmaXR0ZXIgYW5kIHRoZSBTd2FnZ2VyIFBldHN0b3JlIHYzIGV4YW1wbGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucwogICAgICAgICAgY2xpZW50LXNkay1wcm9qZWN0LXVybDogaHR0cHM6Ly9naXRodWIuY29tL2NocmlzdGlhbmhlbGxlL3JlZml0dGVyLWFjdGlvbgpgYGA= readmeEtag: '"9494f135d029b64e34399b1799b5c43cc0b2c2ff"' readmeLastModified: Mon, 12 Jun 2023 21:49:28 GMT repositoryId: 640287071 description: Generate Refit API Client using Refitter created: '2023-05-13T15:34:09Z' updated: '2023-05-15T20:33:59Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: christianhelle logo: https://avatars.githubusercontent.com/u/710400?v=4 license: GPL-3.0 repoEtag: '"66a15ed5c38240c6a2b23b4eaa92e8aec61393e35e39bd696c0a6e64959d57fd"' repoLastModified: Mon, 15 May 2023 20:33:59 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/ekoulemaneng/openapi-security-retriever v3: true id: b19330aebed83b7c5d395eacf404f955 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNlY3VyaXR5IFJldHJpZXZlcgoKVHlwZXNjcmlwdCBwYWNrYWdlIHRvIHJldHJpZXZlIHNlY3VyaXR5IG1lY2hhbmlzbSBpbmZvcm1hdGlvbnMgZnJvbSBhbiBvcGVyYXRpb24gaW4gY29tcGxpYW5jZSB3aXRoIGFuIE9wZW5BcGkgc3BlY2lmaWNhdGlvbi4gRm9yIDMuMS54IGFuZCAzLjAueCBPcGVuQVBJIHZlcnNpb25zLgoKIyMgSW5zdGFsbGF0aW9uCmBgYHR5cGVzY3JpcHQKbnBtIGluc3RhbGwgb3BlbmFwaS1zZWN1cml0eS1yZXRyaWV2ZXIgCmBgYApvcgpgYGB0eXBlc2NyaXB0Cnlhcm4gYWRkIG9wZW5hcGktc2VjdXJpdHktcmV0cmlldmVyCmBgYAoKIyMgVXNhZ2UKYGBgdHlwZXNjcmlwdAppbXBvcnQgc2VjdXJpdHlSZXRyaWV2ZXIgZnJvbSAnb3BlbmFwaS1zZWN1cml0eS1yZXRyaWV2ZXInCmltcG9ydCBzY2hlbWFSZXRyaWV2ZXIgZnJvbSAnb3BlbmFwaS1zY2hlbWEtcmV0cmlldmVyJwoKbGV0IHNjaGVtYQoKY29uc3QgbWFpbiA9IGFzeW5jICgpID0+IHsKCiAgICAvKiBGaXJzdCwgYnkgc2NoZW1hUmV0cmlldmVyLCBnZXQsIGNoZWNrIGFuZCBwYXJzZSB0aGUgb3BlbmFwaSBzcGVjaWZpY2F0aW9uIHRoYXQgY2FuIGJlIGFuIG9iamVjdCBvciBhbiBmaWxlIHBhdGggc3RyaW5nLgogICAgICogSWYgdGhlIHNwZWNpZmljYXRpb24gaW5wdXQgaXMgYSBmaWxlIHBhdGggc3RyaW5nLCBhIHNlY29uZCBhcmd1bWVudCBzdGFuZGluZyBmb3IgdGhlIGN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkgaXMgbWFuZGF0b3J5LgogICAgICogV2UgcmVjb21tZW5kIHRvIHVzZSAnX19kaXJuYW1lJyBhcyBzZWNvbmQgYXJndW1lbnQuCiAgICAgKiBUaGUgZmlsZSBtdXN0IGJlIGVpdGhlciBhIGpzb24sIGEgeWFtbCBvciBhIHltbCBmaWxlLiAKICAgICAqIHNwZWNCdWlsZGVyIHJldHVybnMgYW4gb2JqZWN0IGluIGFjY29yZGFuY2Ugd2l0aCBPcGVuQVBJIHNjcGVjaWZpY2F0aW9uLgogICAgKi8KICAgIGNvbnN0IHNjaGVtYSA9IGF3YWl0IHNjaGVtYVJldHJpZXZlcignLi9vcGVuYXBpLnlhbWwnLCBfX2Rpcm5hbWUpCgogICAgLyoKICAgICAqIHNlY3VyaXR5UmV0cmlldmVyIHRha2VzIDMgbWFuZGF0b3J5IGFyZ3VtZW50czogYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uIG9iamVjdCwgYSBzdHJpbmcgYXMgYSBzY2hlbWEgcGF0aCwgYW5kIGEgc3RyaW5nIGFzIGEgaHR0cCByZXF1ZXN0IG1ldGhvZC4KICAgICovCiAgICBjb25zdCBzZWN1cml0eSA9IHNlY3VyaXR5UmV0cmlldmVyKHNjaGVtYSwgJy9wcm9kdWN0cycsICdnZXQnKQoKICAgIGNvbnNvbGUubG9nKHNlY3VyaXR5KSAvLyA9PiB7IHNlY3VyZWQ6IHRydWUsIG9wdGlvbmFsOiBmYWxzZSwgc2VjdXJpdGllczogWyBKd3RPYXV0aDogeyB0eXBlOiAnaHR0cCcsIHNjaGVtZTogJ2JlYXJlcicsIHNjb3BlczogWydhZG1pbiddIH0gXSB9IHwgbnVsbCAKICAgICAgICAgICAgICAgICAgICAgICAgICAvLyAnbnVsbCcgaWYgb3BlcmF0aW9uIGlzIG5vdCBmb3VuZCBpbiB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9uCn0KCm1haW4oKQpgYGAKCiMjIE91dHB1dCBzdHJ1Y3R1cmUKClRoZSBvdXRwdXQgb2YgdGhlIG1vZHVsZSwgb25jZSBjYWxsZWQsIGlzIGFuIG9iamVjdCB3aG9zZSB0aGUgc3RydWN0dXJlIGlzOgpgYGB0eXBlc2NyaXB0CnsKICAgIHNlY3VyZWQ6IGJvb2xlYW4sIC8vIGluZGljYXRlcyB3aGV0aGVyIHRoZSBvcGVyYXRpb24gcmVxdWlyZXMgYXV0aGVudGljYXRpb24KICAgIG9wdGlvbmFsOiBib29sZWFuLCAvLyBpbmRpY2F0ZXMgd2hldGhlciBhdXRoZW50aWNhdGlvbiBpcyBvcHRpb25hbCBvciBtYW5kYXRvcnkKICAgIHNlY3VyaXRpZXM6IFtPYmplY3RdIC8vIGdpdmVzIGxpc3Qgb2Ygc2VjdXJpdHkgbWVjaGFuaXNtcwp9CmBgYApUaGUgYWJvdmUtbWVudGlvbmVkIE9iamVjdCBpcyBzdHJ1Y3R1cmVkIGFzIGZvbGxvd3M6CmBgYHR5cGVzY3JpcHQKewogICAge25hbWV9OiB7CiAgICAgICAgICAgICAgICB0eXBlOiAnYXBpS2V5JywKICAgICAgICAgICAgICAgIG5hbWU6IHN0cmluZywKICAgICAgICAgICAgICAgIGluOiBzdHJpbmcsCiAgICAgICAgICAgICAgICBzY29wZXM6IFtzdHJpbmddCiAgICAgICAgICAgIH0gfCB7CiAgICAgICAgICAgICAgICB0eXBlOiAnaHR0cCcsCiAgICAgICAgICAgICAgICBzY2hlbWU6IHN0cmluZywKICAgICAgICAgICAgICAgIGJlYXJlckZvcm1hdD86IHN0cmluZywKICAgICAgICAgICAgICAgIHNjb3BlczogW3N0cmluZ10KICAgICAgICAgICAgfSB8IHsKICAgICAgICAgICAgICAgIHR5cGU6ICdvYXV0aDInLAogICAgICAgICAgICAgICAgZmxvd3M6IHsKICAgICAgICAgICAgICAgICAgICAgICAgaW1wbGljaXQ6IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgYXV0aG9yaXphdGlvblVybDogc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWZyZXNoVXJsPzogc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzY29wZXM6IFtzdHJpbmddCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgICAgICAgICBwYXNzd29yZDogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b2tlblVybDogc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWZyZXNoVXJsPzogc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzY29wZXM6IFtzdHJpbmddCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgICAgICAgICBjbGllbnRDcmVkZW50aWFsczogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB0b2tlblVybDogc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZWZyZXNoVXJsPzogc3RyaW5nLAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBzY29wZXM6IFtzdHJpbmddCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgICAgICAgICBhdXRob3JpemF0aW9uQ29kZTogewogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBhdXRob3JpemF0aW9uVXJsOiBzdHJpbmcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHRva2VuVXJsOiBzdHJpbmcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlZnJlc2hVcmw/OiBzdHJpbmcsCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNjb3BlczogW3N0cmluZ10KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgc2NvcGVzOiBbc3RyaW5nXQogICAgICAgICAgICB9IHwgewogICAgICAgICAgICAgICAgdHlwZTogJ29wZW5JZENvbm5lY3QnLAogICAgICAgICAgICAgICAgb3BlbklkQ29ubmVjdFVybDogc2VjdXJpdHlTY2hlbWUub3BlbklkQ29ubmVjdFVybCwKICAgICAgICAgICAgICAgIHNjb3BlczogW3N0cmluZ10KICAgICAgICAgICAgfQp9CmBgYAoKIyMgTGljZW5zZQpUaGlzIHBhY2thZ2UgaXMgbGljZW5zZWQgdW5kZXIgdGhlIFtNSVQgTGljZW5zZV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9taXQpLgoKIyMgQ29udGFjdApJZiB5b3UgaGF2ZSBhbnkgcXVlc3Rpb25zIG9yIGlzc3VlcywgcGxlYXNlIGNvbnRhY3QgdGhlIHBhY2thZ2UgbWFpbnRhaW5lciBhdCBla291bGVtYW5lbmdAZ21haWwuY29tLgo= readmeEtag: '"6b55383fbd5b3d8307eb75ca010458989956b94c"' readmeLastModified: Mon, 08 May 2023 15:25:58 GMT repositoryId: 637176290 description: >- Typescript package to retrieve security mechanism information from an operation in compliance with an OpenApi specification. For 3.1.x and 3.0.x OpenAPI versions. created: '2023-05-06T18:38:00Z' updated: '2023-05-12T04:02:56Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: ekoulemaneng logo: https://avatars.githubusercontent.com/u/12666431?v=4 license: MIT repoEtag: '"5b7ea14972274051e9e4d1a7294621c4a95e4763904ca8ca5ed105c6ea4d0652"' repoLastModified: Fri, 12 May 2023 04:02:56 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mermade/openapi2js v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJMmpzCgpDb252ZXJ0IE9wZW5BcGkgKFN3YWdnZXIpIDIuMCAvIDMuMC54IEpTT04vWUFNTCBkZWZpbml0aW9uIHRvIHNpbXBsZSBKYXZhc2NyaXB0IEFQSSwgY29tcGxldGUgd2l0aCBKU0RPQywgc3VpdGFibGUgZm9yIHVzZSB3aXRoIHRoZSBPcGVuIE5pdHJvIFNESwoKIyMgVXNhZ2U6CgpgYGBqYXZhc2NyaXB0CiAgICB2YXIgb2EyanMgPSByZXF1aXJlKCdvcGVuYXBpMmpzJyk7CiAgICBvYTJqcy5vcGVuQVBJMmpzKGluRmlsZW5hbWUsb3V0RmlsZW5hbWUpOwpgYGAKCm9yCgpgYGBqYXZhc2NyaXB0CiAgICB2YXIgb2EyanMgPSByZXF1aXJlKCdvcGVuYXBpMmpzJyk7CiAgICB2YXIganNTdHIgPSBvYTJqcy5vcGVuQVBJMmpzKHN3YWdnZXJPYmplY3QpOwpgYGAKCg== readmeEtag: '"78076fdc01b5658a5434032d359652095c33b8cd"' readmeLastModified: Tue, 28 Apr 2020 14:33:31 GMT repositoryId: 61999590 description: Create simple JS SDK from OpenAPI (Swagger) 2.0/3.0.x json/yaml document created: '2016-06-26T17:09:25Z' updated: '2020-04-28T14:33:42Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"b49ca74ff8294adccfde3da9d9a4f29def49b3d969e3c1f3e15adc47009f747a"' repoLastModified: Tue, 28 Apr 2020 14:33:42 GMT foundInMaster: true category: Parsers id: cc756d5b3fdee2221f89b92f3bb0a204 - source: openapi3 tags repository: https://github.com/arya-oss/swaggerui-addon v3: true repositoryMetadata: base64Readme: >- IyMgU3dhZ2dlciBVSSBGaXJlZm94IEFkZG9uCgpOb3QgcHVibGlzaGVkIHlldCB0byB0ZXN0IGluIGZpcmVmb3ggYnJvd3NlcgoKMS4gT3BlbiBGaXJlZm94IGFuZCBvcGVuIHRoaXMgbGluayBgYWJvdXQ6ZGVidWdnaW5nYC4KMi4gYExvYWQgVGVtcGVyb3J5IGFkZG9uYCBhbmQgc2VsZWN0IGBtYW5pZmVzdC5qc29uYCBmcm9tIGNvZGUgcmVwb3NpdG9yeS4KCiMjIyBPbiBPcGVyYSBvciBDaHJvbWUKMS4gb3BlbiBDaHJvbWUgb3IgT3BlcmEgYW5kIG9wZW4gdGhpcyBsaW5rIGBjaHJvbWU6Ly9leHRlbnNpb25zYAoyLiBgTG9hZCBVbnBhY2tlZCBFeHRlbnNpb25zYCBhbmQgc2VsZWN0IGB0aGlzIHJlcG9zaXRvcnkgZGlyYCBmcm9tIGNvZGUgcmVwby4K readmeEtag: '"502d80f581dee084d169388d0fa396d1eeefa46e"' readmeLastModified: Wed, 04 Apr 2018 06:45:52 GMT repositoryId: 127413705 description: Swagger UI addons to browse OpenAPI spec APIs created: '2018-03-30T10:09:45Z' updated: '2018-04-04T06:45:54Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: arya-oss logo: https://avatars.githubusercontent.com/u/8657040?v=4 license: MIT repoEtag: '"ffea0ed3a197cd6ae4e22ded5d5ca75e61172012c4042043d5f10997f289372c"' repoLastModified: Wed, 04 Apr 2018 06:45:54 GMT foundInMaster: true category: Documentation id: 1b0b9179f7b33d4642a34006cdbd6439 - source: openapi3 tags repository: https://github.com/encima/openape v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBFClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly90cmF2aXMtY2kuY29tL2VuY2ltYS9vcGVuYXBlLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kuY29tL2VuY2ltYS9vcGVuYXBlKQpbIVtHbyBSZXBvcnQgQ2FyZF0oaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL2JhZGdlL2dpdGh1Yi5jb20vZW5jaW1hL29wZW5hcGUpXShodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vcmVwb3J0L2dpdGh1Yi5jb20vZW5jaW1hL29wZW5hcGUpClshW0dvRG9jXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2VuY2ltYS9vcGVuYXBlP3N0YXR1cy5zdmcpXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2VuY2ltYS9vcGVuYXBlKQoKT3BlbkFQRSBpcyBkZXNpZ25lZCB0byBiZSBhIHNlcnZlciBleHRlbnNpb24gb2YgdGhlIGBPcGVuQVBJYCBzcGVjaWZpY2F0aW9uLCByZWFkaW5nIGluIGEgYFN3YWdnZXJgIGZpbGUgYW5kIGEgY29uZmlnIGFuZCBPcGVuQVBFIHdpbGwgZG8gYWxsIHRoZSBgY29kZSBtb25rZXlgIHN0dWZmIGZvciB5b3U6IGJ1aWxkaW5nIHRoZSByb3V0ZXMsIGFkZGluZyB0aGUgbW9kZWxzIHRvIGEgZGF0YWJhc2UsIHZhbGlkYXRpb24gZXRjLgoKTXVjaCBvZiB3ZWIgZGV2ZWxvcG1lbnQgY29uc2lzdHMgb2YgY3JlYXRpbmcgYW5kIG1haW50YWluaW5nIGFuIEFQSSB0aGF0IGlzIG1hcmdpbmFsbHkgZGlmZmVyZW50IGZyb20gdGhlIGxhc3Qgb25lPyBgU3dhZ2dlcmAgYW5kIGBPcGVuQVBJYCBoYXMgbWFkZSB0aGlzIG11Y2ggZWFzaWVyIHdpdGggY29kZSBnZW5lcmF0aW9uIHRvb2xzIGFuZCBmcmFtZXdvcmtzIHN1Y2ggYXMgYExvb3BiYWNrYC4gQWxtb3N0IGFsbCBtb2RlbHMgY3JlYXRlZCByZXF1aXJlIHRoZSBiYXNpYyBIVFRQIHZlcmJzIGFuZCBzdXBwb3J0ZWQgYWN0aW9ucyAoYFBVVGAsIGBERUxFVEVgIGV0YykgYnV0IG1vc3QgZXhpc3RpbmcgdG9vbHMgb25seSBnZW5lcmF0ZSBtZXRob2Qgc3R1YnMgc3RpbGwgcmVxdWlyaW5nIGFkZGl0aW9uIGNvZGUgZnJvbSBkZXZlbG9wZXJzIHRoaXMgYWltcyB0byBzb2x2ZSB0aGF0LgoKLS0tCgojIyBDb250cmlidXRpbmcKClBScyBhcmUgd2VsY29tZSwgdGhpcyBpcyBhIHN1cGVyIGVhcmx5IHZlcnNpb24gYW5kIGlzIGZhciBmcm9tIHBlcmZlY3QuCgpJZiB5b3UgYXJlIGEgYmVnaW5uZXIsIGNoZWNrIG91dCB0aGUgW2xhYmVsbGVkIG9wZW4gaXNzdWVzXShodHRwczovL2dpdGh1Yi5jb20vZW5jaW1hL29wZW5hcGUvaXNzdWVzP3E9aXMlM0Fpc3N1ZStpcyUzQW9wZW4rbGFiZWwlM0ElMjJnb29kK2ZpcnN0K2lzc3VlJTIyKS4KCi0tLQoKIyMgUmVxdWlyZW1lbnRzCgoqIFBvc3RncmVzIChyZW1vdGUgb3IgbG9jYWwpCiogR28KKiBBIGNvbmZpZyBmaWxlIChleGFtcGxlIGNhbiBiZSBmb3VuZCBpbiB0aGUgYGNvbmZpZ2AgZm9sZGVyCgojIyBHRVQgU1RBUlRFRAoKMS4gYGdvIGdldCBnaXRodWIuY29tL2VuY2ltYS9vcGVuYXBlYAoyLiBgYGAKICAgIHBhY2thZ2UgbWFpbgoKICAgIGltcG9ydCAoCiAgICAgICAgImdpdGh1Yi5jb20vZW5jaW1hL29wZW5hcGUiCiAgICApCgogICAgZnVuYyBtYWluKCkgewogICAgICAgIG9wZW5hcGUuTmV3U2VydmVyKCJQQVRIL1RPL0NPTkZJRyIpCiAgICB9CiAgICBgYGAK readmeEtag: '"94b4fe835195b361031bdcb2c21946d996ada7c0"' readmeLastModified: Tue, 30 Oct 2018 10:19:44 GMT repositoryId: 140843698 description: >- Go get rid of your code monkey duties. Generate servers using only an OpenAPI spec created: '2018-07-13T12:20:39Z' updated: '2023-07-25T14:18:31Z' language: Go archived: false stars: 0 watchers: 0 forks: 1 owner: encima logo: https://avatars.githubusercontent.com/u/517923?v=4 license: GPL-3.0 repoEtag: '"9c2e9f98a93445d5566ee034f065087958195b18fd5c698dc3a46859caa7683f"' repoLastModified: Tue, 25 Jul 2023 14:18:31 GMT foundInMaster: true category: Parsers id: 0b24bbe6a0313b94cd166918598a6cb4 - source: openapi3 tags repository: https://github.com/quetz-al/quetzal-openapi-client v3: true repositoryMetadata: base64Readme: >- # quetzal-openapi-client
Quetzal: an API to manage data files and their associated metadata.

This Python package is automatically generated by the [OpenAPI Generator](https://openapi-generator.tech) project:

- API version: 0.5.0
- Package version: 0.5.0
- Build package: org.openapitools.codegen.languages.PythonClientCodegen
For more information, please visit [https://quetz.al](https://quetz.al)

## Requirements.

Python 2.7 and 3.4+

## Installation & Usage
### pip install

You can install directly with ``pip``:

```sh
pip install quetzal-openapi-client
```

Then import the package:
```python
import quetzal.openapi_client 
```

## Getting Started

Please follow the [installation procedure](#installation--usage) and then run the following:

```python
import quetzal.openapi_client
from quetzal.openapi_client.rest import ApiException
from pprint import pprint

configuration = quetzal.openapi_client.Configuration()
# Configure HTTP basic authorization: basic
configuration.username = 'YOUR_USERNAME'
configuration.password = 'YOUR_PASSWORD'

# create an instance of the API class
api_instance = quetzal.openapi_client.AuthenticationApi(quetzal.openapi_client.ApiClient(configuration))

try:
    # Login.
    api_response = api_instance.auth_get_token()
    pprint(api_response)
except ApiException as e:
    print("Exception when calling AuthenticationApi->auth_get_token: %s\n" % e)

```

## Documentation for API Endpoints

All URIs are relative to *https://api.quetz.al/api/v1*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AuthenticationApi* | [**auth_get_token**](docs/AuthenticationApi.md#auth_get_token) | **POST** /auth/token | Login.
*AuthenticationApi* | [**auth_logout**](docs/AuthenticationApi.md#auth_logout) | **POST** /auth/logout | Logout.
*DataApi* | [**public_file_details**](docs/DataApi.md#public_file_details) | **GET** /data/files/{uuid} | Fetch public file.
*DataApi* | [**public_file_fetch**](docs/DataApi.md#public_file_fetch) | **GET** /data/files/ | List public files.
*DataApi* | [**public_query_create**](docs/DataApi.md#public_query_create) | **POST** /data/queries/ | Prepare a query.
*DataApi* | [**public_query_details**](docs/DataApi.md#public_query_details) | **GET** /data/queries/{qid} | Query details.
*DataApi* | [**public_query_fetch**](docs/DataApi.md#public_query_fetch) | **GET** /data/queries/ | List public queries.
*DataApi* | [**workspace_commit**](docs/DataApi.md#workspace_commit) | **PUT** /data/workspaces/{wid}/commit | Commit workspace.
*DataApi* | [**workspace_create**](docs/DataApi.md#workspace_create) | **POST** /data/workspaces/ | Create workspace.
*DataApi* | [**workspace_delete**](docs/DataApi.md#workspace_delete) | **DELETE** /data/workspaces/{wid} | Delete workspace.
*DataApi* | [**workspace_details**](docs/DataApi.md#workspace_details) | **GET** /data/workspaces/{wid} | Workspace details.
*DataApi* | [**workspace_fetch**](docs/DataApi.md#workspace_fetch) | **GET** /data/workspaces/ | List workspaces.
*DataApi* | [**workspace_file_create**](docs/DataApi.md#workspace_file_create) | **POST** /data/workspaces/{wid}/files/ | Upload file.
*DataApi* | [**workspace_file_delete**](docs/DataApi.md#workspace_file_delete) | **DELETE** /data/workspaces/{wid}/files/{uuid} | Delete a file.
*DataApi* | [**workspace_file_details**](docs/DataApi.md#workspace_file_details) | **GET** /data/workspaces/{wid}/files/{uuid} | Fetch file.
*DataApi* | [**workspace_file_fetch**](docs/DataApi.md#workspace_file_fetch) | **GET** /data/workspaces/{wid}/files/ | List files.
*DataApi* | [**workspace_file_set_metadata**](docs/DataApi.md#workspace_file_set_metadata) | **PUT** /data/workspaces/{wid}/files/{uuid} | Rewrite metadata.
*DataApi* | [**workspace_file_update_metadata**](docs/DataApi.md#workspace_file_update_metadata) | **PATCH** /data/workspaces/{wid}/files/{uuid} | Modify metadata.
*DataApi* | [**workspace_query_create**](docs/DataApi.md#workspace_query_create) | **POST** /data/workspaces/{wid}/queries/ | Prepare a query.
*DataApi* | [**workspace_query_details**](docs/DataApi.md#workspace_query_details) | **GET** /data/workspaces/{wid}/queries/{qid} | Query details.
*DataApi* | [**workspace_query_fetch**](docs/DataApi.md#workspace_query_fetch) | **GET** /data/workspaces/{wid}/queries/ | List queries.
*DataApi* | [**workspace_scan**](docs/DataApi.md#workspace_scan) | **PUT** /data/workspaces/{wid}/scan | Update views.
*PublicApi* | [**public_file_details**](docs/PublicApi.md#public_file_details) | **GET** /data/files/{uuid} | Fetch public file.
*PublicApi* | [**public_file_fetch**](docs/PublicApi.md#public_file_fetch) | **GET** /data/files/ | List public files.
*PublicApi* | [**public_query_create**](docs/PublicApi.md#public_query_create) | **POST** /data/queries/ | Prepare a query.
*PublicApi* | [**public_query_fetch**](docs/PublicApi.md#public_query_fetch) | **GET** /data/queries/ | List public queries.
*PublicApi* | [**workspace_create**](docs/PublicApi.md#workspace_create) | **POST** /data/workspaces/ | Create workspace.
*PublicApi* | [**workspace_fetch**](docs/PublicApi.md#workspace_fetch) | **GET** /data/workspaces/ | List workspaces.
*QueryApi* | [**public_query_create**](docs/QueryApi.md#public_query_create) | **POST** /data/queries/ | Prepare a query.
*QueryApi* | [**public_query_details**](docs/QueryApi.md#public_query_details) | **GET** /data/queries/{qid} | Query details.
*QueryApi* | [**public_query_fetch**](docs/QueryApi.md#public_query_fetch) | **GET** /data/queries/ | List public queries.
*QueryApi* | [**workspace_query_create**](docs/QueryApi.md#workspace_query_create) | **POST** /data/workspaces/{wid}/queries/ | Prepare a query.
*QueryApi* | [**workspace_query_details**](docs/QueryApi.md#workspace_query_details) | **GET** /data/workspaces/{wid}/queries/{qid} | Query details.
*QueryApi* | [**workspace_query_fetch**](docs/QueryApi.md#workspace_query_fetch) | **GET** /data/workspaces/{wid}/queries/ | List queries.
*WorkspaceApi* | [**public_query_details**](docs/WorkspaceApi.md#public_query_details) | **GET** /data/queries/{qid} | Query details.
*WorkspaceApi* | [**workspace_commit**](docs/WorkspaceApi.md#workspace_commit) | **PUT** /data/workspaces/{wid}/commit | Commit workspace.
*WorkspaceApi* | [**workspace_details**](docs/WorkspaceApi.md#workspace_details) | **GET** /data/workspaces/{wid} | Workspace details.
*WorkspaceApi* | [**workspace_fetch**](docs/WorkspaceApi.md#workspace_fetch) | **GET** /data/workspaces/ | List workspaces.
*WorkspaceApi* | [**workspace_file_create**](docs/WorkspaceApi.md#workspace_file_create) | **POST** /data/workspaces/{wid}/files/ | Upload file.
*WorkspaceApi* | [**workspace_file_delete**](docs/WorkspaceApi.md#workspace_file_delete) | **DELETE** /data/workspaces/{wid}/files/{uuid} | Delete a file.
*WorkspaceApi* | [**workspace_file_details**](docs/WorkspaceApi.md#workspace_file_details) | **GET** /data/workspaces/{wid}/files/{uuid} | Fetch file.
*WorkspaceApi* | [**workspace_file_fetch**](docs/WorkspaceApi.md#workspace_file_fetch) | **GET** /data/workspaces/{wid}/files/ | List files.
*WorkspaceApi* | [**workspace_file_set_metadata**](docs/WorkspaceApi.md#workspace_file_set_metadata) | **PUT** /data/workspaces/{wid}/files/{uuid} | Rewrite metadata.
*WorkspaceApi* | [**workspace_file_update_metadata**](docs/WorkspaceApi.md#workspace_file_update_metadata) | **PATCH** /data/workspaces/{wid}/files/{uuid} | Modify metadata.
*WorkspaceApi* | [**workspace_query_create**](docs/WorkspaceApi.md#workspace_query_create) | **POST** /data/workspaces/{wid}/queries/ | Prepare a query.
*WorkspaceApi* | [**workspace_query_details**](docs/WorkspaceApi.md#workspace_query_details) | **GET** /data/workspaces/{wid}/queries/{qid} | Query details.
*WorkspaceApi* | [**workspace_query_fetch**](docs/WorkspaceApi.md#workspace_query_fetch) | **GET** /data/workspaces/{wid}/queries/ | List queries.
*WorkspaceApi* | [**workspace_scan**](docs/WorkspaceApi.md#workspace_scan) | **PUT** /data/workspaces/{wid}/scan | Update views.


## Documentation For Models

 - [BaseMetadata](docs/BaseMetadata.md)
 - [Error](docs/Error.md)
 - [FileContents](docs/FileContents.md)
 - [MetadataByFamily](docs/MetadataByFamily.md)
 - [PaginatedFiles](docs/PaginatedFiles.md)
 - [PaginatedQueries](docs/PaginatedQueries.md)
 - [PaginatedWorkspaces](docs/PaginatedWorkspaces.md)
 - [PaginationEnvelope](docs/PaginationEnvelope.md)
 - [Query](docs/Query.md)
 - [QueryNoResults](docs/QueryNoResults.md)
 - [Token](docs/Token.md)
 - [Workspace](docs/Workspace.md)


## Documentation For Authorization


## apiKey

- **Type**: API key
- **API key parameter name**: X-API-KEY
- **Location**: HTTP header


## basic

- **Type**: HTTP basic authentication


## bearer

- **Type**: Bearer authentication


## Author

support@quetz.al


 readmeEtag: '"a307d4ef7f078d19d06e81ef5fa413627c835ccd"' readmeLastModified: Thu, 17 Oct 2019 12:49:40 GMT repositoryId: 172245557 description: Autogenerated Python client for the Quetzal API created: '2019-02-23T17:55:11Z' updated: '2019-10-17T12:50:29Z' language: Python archived: false stars: 0 watchers: 0 forks: 1 owner: quetz-al logo: https://avatars.githubusercontent.com/u/47522179?v=4 license: BSD-3-Clause repoEtag: '"0aea99815a11efc69b3baa3c23d7087842014d786c4dec6966af479ac0118942"' repoLastModified: Thu, 17 Oct 2019 12:50:29 GMT foundInMaster: true category: Description Validators id: 5fbdc491c2818ac4f76e96167dc5fde6 - source: openapi3 tags repository: https://github.com/ucl/isenseflu-openapi v3: true repositoryMetadata: base64Readme: >- IyBpc2Vuc2VmbHUtb3BlbmFwaQoKT3BlbkFQSSBEZWZpbml0aW9ucyBmb3IgaS1zZW5zZSBmbHUgQVBJCgpTcGVjaWZpY2F0aW9uOiBodHRwOi8vc3BlYy5vcGVuYXBpcy5vcmcvb2FzL3YzLjAuMgoKIyMgVmFsaWRhdGlvbgoKVXNlIHRoZSBbT3BlbiBBUEkgRW5mb3JjZXIgQ0xJXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLWVuZm9yY2VyLWNsaS8pIHRvb2wgZm9yIHZhbGlkYXRpb24uIEluc3RhbGwgd2l0aCBgbnBtYDoKCmBgYApucG0gaW5zdGFsbCAtZyBvcGVuYXBpLWVuZm9yY2VyLWNsaQpgYGAKClZhbGlkYXRlIHRoZSBkZWZpbml0aW9ucyBmaWxlOgoKYGBgCm9wZW5hcGktZW5mb3JjZXIgdmFsaWRhdGUgb3BlbmFwaS55YW1sCmBgYAoKIyMgQ29weXJpZ2h0CgomY29weTsgMjAxOSBVQ0wgKFtodHRwczovL3d3dy51Y2wuYWMudWtdKGh0dHBzOi8vd3d3LnVjbC5hYy51aykpLgoKIyMgTGljZW5zZQoKWyFbTGljZW5zZTogR1BMIHYzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtR1BMdjMtYmx1ZS5zdmcpXShodHRwczovL3d3dy5nbnUub3JnL2xpY2Vuc2VzL2dwbC0zLjApCgpHTlUgR0VORVJBTCBQVUJMSUMgTElDRU5TRSAgClZlcnNpb24gMywgMjkgSnVuZSAyMDA3CgpUaGlzIHByb2dyYW0gaXMgZnJlZSBzb2Z0d2FyZTogeW91IGNhbiByZWRpc3RyaWJ1dGUgaXQgYW5kL29yIG1vZGlmeQppdCB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlIGFzIHB1Ymxpc2hlZCBieQp0aGUgRnJlZSBTb2Z0d2FyZSBGb3VuZGF0aW9uLCBlaXRoZXIgdmVyc2lvbiAzIG9mIHRoZSBMaWNlbnNlLCBvcgooYXQgeW91ciBvcHRpb24pIGFueSBsYXRlciB2ZXJzaW9uLgoKVGhpcyBwcm9ncmFtIGlzIGRpc3RyaWJ1dGVkIGluIHRoZSBob3BlIHRoYXQgaXQgd2lsbCBiZSB1c2VmdWwsCmJ1dCBXSVRIT1VUIEFOWSBXQVJSQU5UWTsgd2l0aG91dCBldmVuIHRoZSBpbXBsaWVkIHdhcnJhbnR5IG9mCk1FUkNIQU5UQUJJTElUWSBvciBGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRS4gIFNlZSB0aGUKR05VIEdlbmVyYWwgUHVibGljIExpY2Vuc2UgZm9yIG1vcmUgZGV0YWlscy4KCllvdSBzaG91bGQgaGF2ZSByZWNlaXZlZCBhIGNvcHkgb2YgdGhlIEdOVSBHZW5lcmFsIFB1YmxpYyBMaWNlbnNlCmFsb25nIHdpdGggdGhpcyBwcm9ncmFtLiAgSWYgbm90LCBzZWUgPGh0dHBzOi8vd3d3LmdudS5vcmcvbGljZW5zZXMvPi4K readmeEtag: '"3eaebec6e4b740e9d144d1fa175af428132da576"' readmeLastModified: Thu, 06 Jun 2019 15:43:49 GMT repositoryId: 190232496 description: OpenAPI Definitions for i-sense flu API created: '2019-06-04T15:42:42Z' updated: '2019-06-06T15:43:57Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: UCL logo: https://avatars.githubusercontent.com/u/2951150?v=4 license: GPL-3.0 repoEtag: '"689b59041519b9b2b1d6aafbe7573c216c7af8ae362e6f0321f5bd99ce3768d6"' repoLastModified: Thu, 06 Jun 2019 15:43:57 GMT foundInMaster: true category: - Server - Parsers id: d293478945a898b421f0b65793024fbe - source: openapi3 tags repository: https://github.com/serialberry/openapispec v3: true repositoryMetadata: base64Readme: >- IyBHb2xhbmcgc3RydWN0cyBmb3IgT3BlbiBBUEkgc3BlY2lmaWNhdGlvbiB2MwoKVGhlcmUgYXJlIG1hbnkgb3BlbnNvdXJjZSBnb2xhbmcgcGFja2FnZXMgaW1wbGVtZW50IE9wZW4gQVBJIHYzIHRvb2xpbmcgZm9yIHZhbGlkYXRpb24sIG1hcnNoYWwsIGFuZCB1bm1hcnNoYWwgZnVuY3Rpb25zLgoKTW9zdCBvZiB0aG9zZSBwcm9qZWN0cyBhZGRyZXNzIHNwZWNpZmljIHByb2JsZW0sIG9yIGltcGxlbWVudCBmdW5jdGlvbmFsaXR5IGFyb3VuZCB0b29saW5nIHN1cHBvcnQsIHRoZXJlZm9yZSB0aGUgZGF0YSBzdHJ1Y3R1cmVzIGFyZSBibG9hdGVkIHdpdGggc3BlY2lmaWMgbWV0aG9kIGltcGxlbWVudGF0aW9ucy4KCkFpbSBvZiB0aGlzIHJlcG9zaXRvcnkgdG8gbWFpbnRhaW4gZ29sYW5nIGRhdGEgc3RydWN0dXJlIGZvciBPcGVuIEFQSSBzcGVjaWZpY2F0aW9uIHZlcnNpb24gMy54IHdpdGhvdXQgc3BlY2lmaWMgdG9vbGluZyBmdW5jdGlvbnMuCgpQYXJzaW5nLCB2YWxpZGF0aW9uLCBhbmQgaW1wbGVtZW50YXRpb24gb2YgZnVuY3Rpb25zIGFyZSB1cHRvIHRoZSByZWZlcmVuY2UgcHJvamVjdC4KCkhvd2V2ZXIsIHBsZWFzZSBub3RlIGF0IHRoZSBtb21lbnQgdGhpcyBpcyBhbiBleHBlcmltZW50YWwgcmVwb3NpdG9yeS4KCiMjIFByb2plY3Qgc3R1Y3R1cmUKClByb2plY3Qgc3RydWN0dXJlIGlzIGJhc2VkIG9uIFtnb2xhbmctc3RhbmRhcmRzXShodHRwczovL2dpdGh1Yi5jb20vZ29sYW5nLXN0YW5kYXJkcy9wcm9qZWN0LWxheW91dCkgcmVmZXJlbmNlIGRvY3MuCgojIyMgW3BrZ10oaHR0cHM6Ly9naXRodWIuY29tL01haGlmZXJuYW5kby9vcGVuYXBpc3BlYy90cmVlL21hc3Rlci9wa2cpCgpQcm92aWRlcyBkYXRhIHN0cnVjdHVyZXMgZm9yIFtPcGVuIEFQSSBzcGVjaWZpY2F0aW9uIHZlcnNpb24gMy4wLjNdKGh0dHA6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMC4zKQ== readmeEtag: '"5d5c7832c291597aa140438fb778534906f13778"' readmeLastModified: Sat, 27 Jun 2020 19:35:17 GMT repositoryId: 266426426 description: Open API 3.0 spec Golang package created: '2020-05-23T21:59:30Z' updated: '2020-06-27T19:35:26Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: serialberry logo: https://avatars.githubusercontent.com/u/49234321?v=4 repoEtag: '"ab49624ba5153f761328677bb04634f69f10c11bc5ad0e3626e281a0a509fd4a"' repoLastModified: Sat, 27 Jun 2020 19:35:26 GMT foundInMaster: true category: Parsers id: d4476af9dc7e5b842f16249529c19e64 - source: openapi3 tags repository: https://github.com/b1a9id/open-api-sample v3: true repositoryMetadata: base64Readme: >- IyBTYW1wbGUgZm9yIHNwcmluZ2RvYy1vcGVuYXBpCjxhIGhyZWY9Imh0dHBzOi8vdHJhY2tnaXQuY29tIj4KPGltZyBzcmM9Imh0dHBzOi8vc2Z5LmN4L3UveGRmIiBhbHQ9InRyYWNrZ2l0LXZpZXdzIiAvPgo8L2E+CgojIyBUcnkgaXQKUnVuIEFwcApgYGAKJCAuL2dyYWRsZXcgY2xlYW4gYm9vdFJ1bgpgYGAKClBhdGhzCmBgYApHRVQgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS92My9hcGktZG9jcy8KICBPcGVuQVBJIGRvY3VtZW50YXRpb24gaW4gSlNPTiBmb3JtYXQKICAKR0VUIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGkvdjMvYXBpLWRvY3MueWFtbAogIE9wZW5BUEkgZG9jdW1lbnRhdGlvbiBpbiBZQU1MIGZvcm1hdAogIApHRVQgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS9zd2FnZ2VyLXVpLmh0bWwKICBTd2FnZ2VyIFVJIEhUTUwgZG9jdW1lbnRhdGlvbgpgYGAK readmeEtag: '"4548b91fd02b699b03a721461db516f22a47d4ed"' readmeLastModified: Sun, 25 Oct 2020 07:58:27 GMT repositoryId: 295139675 description: Sample for springdoc-openapi created: '2020-09-13T11:45:33Z' updated: '2020-10-25T07:58:30Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: b1a9id logo: https://avatars.githubusercontent.com/u/19252181?v=4 repoEtag: '"b89a5e302e52e44db7b129a49ecfdca7a246b8b84cd40619ed4349e75d7da4c9"' repoLastModified: Sun, 25 Oct 2020 07:58:30 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: 2d23b950c126fb81e9b8209e48e86ec6 - source: openapi3 tags repository: https://github.com/banedd/api-template v3: true repositoryMetadata: base64Readme: >- IyBFeHByZXNzIEFQSSBUZW1wbGF0ZQoKVGhpcyBpcyBhIHNpbXBsZSBFeHByZXNzIEFQSSB0ZW1wbGF0ZSB0byB1c2UgZm9yIEFQSSBkZXZlbG9wbWVudC4gRmVhdHVyZXM6CgotIENvbnRhaW5lcml6ZWQgZm9yIHVzZSB3aXRoIERvY2tlcgotIE5vZGVtb24gZm9yIGRldmVsb3BtZW50IHRvIGFsbG93IGVhc3kgcmVsb2FkaW5nIG9uIGNoYW5nZQotIEluY2x1ZGVzIFN3YWdnZXIgVUkgYW5kIEpTT05kb2MgYXV0b2dlbmVyYXRpb24KLSBMb2dnaW5nIHNlcnZpY2UgdG8gYSByb2xsaW5nIGZpbGUKLSBSZXF1ZXN0IGxvZ3MKLSBDb25zb2xlIGxvZ2dpbmcgZm9yIGRldmVsb3BtZW50IG1vZGUKLSBTUUxpdGUgYXMgdGhlIGRhdGFiYXNlIGZvciB0aGUgZXhhbXBsZSwgY2FuIGJlIGNoYW5nZWQgdG8gZGlmZmVyZW50IGRhdGEgc291cmNlIGFzIGRlc2lyZWQKLSBWUyBDb2RlIGRlYnVnZ2luZyBjb25maWd1cmF0aW9uCgojIEJ1aWxkaW5nIGFuZCBydW5uaW5nIHRoZSBwcm9qZWN0CgotIFJ1biBgbnBtIGluc3RhbGxgIHRvIGluc3RhbGwgYWxsIHRoZSBwcm9qZWN0IGRlcGVuZGVuY2llcwotIFJ1biBgc3RhcnQuc2hgIHRvIHJ1biB0aGUgcHJvamVjdCBsb2NhbGx5CgpOT1RFOiBUbyBlbmFibGUgU3dhZ2dlciBVSSBzZXQgKipOT0RFX0VOVioqIHRvICoqZGV2ZWxvcG1lbnQqKgo= readmeEtag: '"8226bbe5eb4fba9669a52202e8894e5b1d89826f"' readmeLastModified: Tue, 31 Mar 2020 14:50:13 GMT repositoryId: 240870348 description: Express API Template created: '2020-02-16T10:16:17Z' updated: '2020-03-31T14:50:22Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: baneDD logo: https://avatars.githubusercontent.com/u/661366?v=4 repoEtag: '"bb87c1ff1a1b0f72cf864657e976e3a41f8f15ea4548ef4c263dc18b4095221b"' repoLastModified: Tue, 31 Mar 2020 14:50:22 GMT foundInMaster: true category: Server Implementations id: a406abd521e3706405bf7f0d846b56da - source: openapi3 tags repository: https://github.com/amitvsavant/jinja-template-2-open-api v3: true repositoryMetadata: base64Readme: >- IyBKaW5qYSBUZW1wbGF0ZSB0byBPcGVuIEFQSSAzLjAgWUFNTApUaGlzIHJlcG9zaXRvcnkgcHJvdmlkZXMgc2FtcGxlIFB5dGhvbiAzLjcgY29kZSB0byBnZW5lcmF0ZSBPcGVuIEFQSSAzLjAgc3BlY2lmaWNhdGlvbiBZQU1MIGZpbGVzIHVzaW5nIEppbmphMiB0ZW1wbGF0ZS4gSXQgc2hvd3MgaG93IHRvIGdlbmVyYXRlIG11bHRpcGxlIHNpbWlsYXIgQVBJIHNwZWNpZmljYXRpb25zIGZyb20gYnkgYXBwbHlpbmcgZGF0YSBpbiBKc29uIGZvcm1hdCB0byBhIEppbmphMiB0ZW1wbGF0ZSBvZiBhbiBPcGVuIEFQSSBzcGVjaWZpY2F0aW9uIGZpbGUuIFRoZSBjb2RlIGdlbmVyYXRlcyBBUEkgc3BlY2lmaWNhdGlvbnMgZm9yIHNpbXBsZSBQZXQgU3RvcmUgYW5kIEJvb2sgU3RvcmUgQVBJcyB1c2luZyBhIGdlbmVyaWMgU3RvcmUgdGVtcGxhdGUgZmlsZS4KCiMjIFN0cnVjdHVyZSBvZiB0aGUgcHJvamVjdAoKIyMjIGRhdGEgZm9sZGVyCgoqKmRhdGEuanNvbjoqKiAgVGhlIEpzb24gZmlsZSBjb250YWluaW5nIHRoZSBkYXRhIG9mIG11bHRpcGxlIEFQSSBzcGVjaWZpY2F0aW9ucyB0byBiZSBnZW5lcmF0ZWQuCgojIyMgdGVtcGxhdGVzIGZvbGRlcgoqKnN0b3JlLXRlbXBsYXRlLnlhbWw6KiogR2VuZXJpYyB0ZW1wbGF0ZSBmaWxlIG9mIGEgU3RvcmUgQVBJLgoKIyMjIGdlbmVyYXRlZC1zcGVjIGZvbGRlcgpUaGUgZGVzdGluYXRpb24gZm9sZGVyIHdoZXJlIHRoZSBmaW5hbCBzcGVjaWZpY2F0aW9uIGZpbGVzIHdpbGwgYmUgY3JlYXRlZCBhZnRlciB0aGUgZXhlY3V0aW9uIG9mIHRoZSBwcm9ncmFtLgoqKnN0b3JlLXRlbXBsYXRlLnlhbWw6KiogR2VuZXJpYyB0ZW1wbGF0ZSBmaWxlIG9mIGEgU3RvcmUgQVBJLgoKIyMgTGljZW5zZQotIFRoZSBzb3VyY2UgY29kZSBpbiB0aGlzIHByb2plY3QgaXMgcHVibGlzaGVkIHVuZGVyIFtNSVQgTGljZW5zZV0oaHR0cHM6Ly9naXRodWIuY29tL2FtaXR2c2F2YW50L2ppbmphLXRlbXBsYXRlLTItb3Blbi1hcGkvYmxvYi9tYXN0ZXIvTElDRU5TRS5tZCku readmeEtag: '"bf594ca21dffe3f58f07b4b03466a31d7542c6c3"' readmeLastModified: Wed, 30 Dec 2020 20:39:11 GMT repositoryId: 325632221 description: Jinja Template to Open API 3.0 YAML created: '2020-12-30T19:31:30Z' updated: '2020-12-30T20:41:32Z' language: Python archived: false stars: 0 watchers: 1 forks: 1 owner: amitvsavant logo: https://avatars.githubusercontent.com/u/30230131?v=4 license: MIT repoEtag: '"bc9181c957a3e6a35553560ceed0ebbbe45e0ee3d1efcee2bc8d9e77f2fd9688"' repoLastModified: Wed, 30 Dec 2020 20:41:32 GMT foundInMaster: true category: - Testing - Parsers id: 4b5cc797d55cc42f1877fa0657359f84 - source: openapi3 tags repository: https://github.com/claudiocifuentesalonso/spring-cloud-service-discovery v3: true repositoryMetadata: base64Readme: >- IyBzcHJpbmctY2xvdWQtc2VydmljZS1kaXNjb3ZlcnkKSW4gdGhpcyByZXBvc2l0b3J5IHdlIGhhdmUgdGhyZWUgbWljcm9zZXJ2aWNlcyB0byBpbXBsZW1lbnQgc2VydmljZSBkaXNjb3ZlcnkgdXNpbmcgc3ByaW5nIGNsb3VkLgoKIyMgVXNhZ2UKClR3byBvcHRpb25zIHRvIHJ1biB0aGUgYXBwbGljYXRpb25zOgoxLiBFeGVjdXRlIGVhY2ggYXBwbGljYXRpb24gc2VwYXJldGx5IGluIG9yZGVyIChldXJla2Etc2VydmVyLCBvcGVuLWFwaS1zcHJpbmdkb2MgYW5kIG9wZW4tYXBpLWNsaWVudCkgdXNpbmc6CgpgYGBiYXNoCm12biBzcHJpbmctYm9vdDpydW4KYGBgCgoyLiBCdWlsZCBkb2NrZXIgaW1hZ2VzIChEb2NrZXJmaWxlIHByZXNlbnQgZm9yIGVhY2ggYXBwbGljYXRpb24pIGFuZCBleGVjdXRlIHRoZSBhcHBsaWNhdGlvbnMgdXNpbmcgZG9ja2VyIGNvbXBvc2UgKGBgYGRvY2tlci1jb21wb3NlLXltbGBgYCBmaWxlKToKYGBgCmRvY2tlci1jb21wb3NlIHVwCmBgYAo= readmeEtag: '"fba2f44b0689de19be3bc8bbbf38fd459cd0d9b6"' readmeLastModified: Mon, 01 Feb 2021 01:29:43 GMT repositoryId: 333291815 description: >- In this repository we have three microservices to implement service discovery using spring cloud. created: '2021-01-27T03:28:13Z' updated: '2021-02-01T01:36:08Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: ClaudioCifuentesAlonso logo: https://avatars.githubusercontent.com/u/31392117?v=4 repoEtag: '"672e58c4df42627f381b5e9c596625543a9d756c7539074fb5ff636cc8f0e0fa"' repoLastModified: Mon, 01 Feb 2021 01:36:08 GMT foundInMaster: true category: - Server - Server Implementations id: f59f91b9c34d83fff8f1dd6fc14a1d11 - source: openapi3 tags repository: https://github.com/tanqidong1992/code-doc-tool v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJ5paH5qGj5bel5YW3Cui/meaYr+S4gOS4quWIhuaekFNwcmluZ+mhueebruS4reeahEphdmHmupDku6PnoIEo5rOo6YeKKeS7peWPimNsYXNz5paH5Lu255Sf5oiQT3BlbkFQSeaOpeWPo+aWh+aho+eahOW3peWFty4KIyMg5Yqf6IO954m55oCnCi0g5o+Q5L6b6ZuG5Lit5byP55qE5o6l5Y+j5paH5qGj566h55CG5bel5YW3KOmbhuaIkFN3YWdnZXItVUnvvIxTd2FnZ2VyLUVkaXRvcinjgIIKLSDliIbmnpDms6jph4rnlJ/miJBPcGVuQVBJ5o6l5Y+j5paH5qGj77yM5peg6ZyAU3dhZ2dlcuazqOino+OAggotIOaUr+aMgeWIhuaekE1hdmVu5aSW6YOo5rqQ5Luj56CBSmFy5paH5Lu244CCCi0g5o+Q5L6bTWF2ZW7mj5Lku7bnlKjkuo7pobnnm67pm4bmiJDjgIIKLSDmlK/mjIHnlJ/miJBNYXJrZG93buaWh+aho+OAggotIOaUr+aMgU9wZW5BUEnmoKHpqozjgIIKIyMg5L2/55SoCiMjIyDpg6jnvbLmjqXlj6PmlofmoaPnrqHnkIblt6XlhbcKMS4g57yW6K+R5oiQZG9ja2Vy6ZWc5YOPCiAgIGBgYHNoZWxsCiAgIGNkIG9wZW5hcGktdWkKICAgbXZuIGNvbXBpbGUgamliOmJ1aWxkVGFyCiAgIGBgYAoyLiDlr7zlhaVkb2NrZXLplZzlg48KICAgYGBgc2hlbGwKICAgZG9ja2VyIGltYWdlIGxvYWQgLWkgdGFyZ2V0L2ppYi1pbWFnZS50YXIgCiAgIGBgYAozLiDphY3nva7ov5DooYwKICAgYGBgc2hlbGwKICAgZG9ja2VyIHJ1biAtLXJtIC10aSAtZSBTWVNURU1fTkFNRT0iWFhY57O757uf5o6l5Y+j5paH5qGjIiAtcCA4MDgwOjgwL3RjcCBobnZtbnMvb3BlbmFwaS11aTowLjAuMgogICBgYGAKICAgZG9ja2VyLWNvbXBvc2Xlj4LogIMKICAgYGBgeW1sCiAgIHZlcnNpb246ICIyLjQiCiAgIHNlcnZpY2VzOgogICAgIHRlc3Q6CiAgICAgIGltYWdlOiBobnZtbnMvb3BlbmFwaS11aTowLjAuMgogICAgICByZXN0YXJ0OiBhbHdheXMKICAgICAgcG9ydHM6CiAgICAgICAgLSA5MDAwOjgwL3RjcAogICAgICB2b2x1bWVzOgogICAgICAgIC0gL2RhdGEvYXBwL2FwaWRvYy90ZXN0L2RhdGE6L2RhdGEKICAgICAgICAtIC9kYXRhL2FwcC9hcGlkb2MvdGVzdC9oaXN0b3J5Oi9oaXN0b3J5CiAgICAgIG1lbV9saW1pdDogIjEwMjRNIgogICAgICBlbnZpcm9ubWVudDoKICAgICAgICBTWVNURU1fTkFNRTogIlhYWOezu+e7n+aOpeWPo+aWh+ahoyIKICAgYGBgCjQuIOaJk+W8gOa1j+iniOWZqOiuv+mXruS4u+mhte+8mmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9kb2MvCiMjIyDpobnnm67pm4bmiJBNYXZlbuaPkuS7tgoxLiDliqDlhaXmj5Lku7blnZDmoIcKICAgYGBgeG1sCiAgICAgICAgICAgIDxwbHVnaW4+CiAgICAgICAgICAgICAgICA8Z3JvdXBJZD5jb20uaG5nZC50b29sPC9ncm91cElkPgogICAgICAgICAgICAgICAgPGFydGlmYWN0SWQ+Y29kZWdlbi1tYXZlbi1wbHVnaW48L2FydGlmYWN0SWQ+CiAgICAgICAgICAgICAgICA8dmVyc2lvbj4yLjYuMC1TTkFQU0hPVDwvdmVyc2lvbj4KICAgICAgICAgICAgICAgIDxjb25maWd1cmF0aW9uPgogICAgICAgICAgICAgICAgICAgIDwhLS0gY29udHJvbGxlcuexu+aJgOWcqOWMheWQjeensCAtLT4KICAgICAgICAgICAgICAgICAgICA8cGFja2FnZUZpbHRlcj5jb20uaG5nZC53ZWIuY29udHJvbGxlcjwvcGFja2FnZUZpbHRlcj4KICAgICAgICAgICAgICAgICAgICA8IS0tIOmFjee9rueUn+aIkG9wZW5hcGnmlofmoaPnmoTln7rnoYDkv6Hmga/phY3nva7mlofku7bmiYDlnKjkvY3nva4gCiAgICAgICAgICAgICAgICAgICAgPGNvbmZGaWxlUGF0aD4ke3Byb2plY3QuYmFzZWRpcn0vYnVpbGQtY29uZmlnL29wZW5hcGkuanNvbjwvY29uZkZpbGVQYXRoPiAtLT4KICAgICAgICAgICAgICAgICAgICA8IS0tT3BlbkFQSSBVSSDmnI3liqHlnLDlnYAs6YWN572u5ZCO5o6l5Y+j5paH5qGj5bCG6Ieq5Yqo5LiK5Lyg5Yiw6K+l5pyN5YqhIC0tPgogICAgICAgICAgICAgICAgICAgIDxvcGVuQVBJVUlTZXJ2ZXI+bG9jYWxob3N0OjgwODA8L29wZW5BUElVSVNlcnZlcj4KICAgICAgICAgICAgICAgICAgICA8IS0tIOa6kOeggeWIhuaekOaOkumZpOi3r+W+hCAtLT4KICAgICAgICAgICAgICAgICAgICA8ZXhjbHVkZXM+KiovY29tL2huZ2QvbW9kZWwvKkV4YW1wbGUuamF2YSwqKi9jb20vaG5nZC9kYW8vKi5qYXZhPC9leGNsdWRlcz4KICAgICAgICAgICAgICAgICAgICA8IS0tIOa6kOeggeWIhuaekOWMheWQq+i3r+W+hCAtLT4KICAgICAgICAgICAgICAgICAgICA8aW5jbHVkZXM+KiovY29tL2huZ2QvKiovKi5qYXZhLGNvbS9obmdkLyoqLyouamF2YTwvaW5jbHVkZXM+CiAgICAgICAgICAgICAgICAgICAgPCEtLSBvcGVuYXBp5o6l5Y+j5Z+656GA5Zyw5Z2AIC0tPgogICAgICAgICAgICAgICAgICAgIDxvcGVuQVBJU2VydmVyVVJMPmh0dHBzOi8vbG9jYWxob3N0OjgwODAvYXBpPC9vcGVuQVBJU2VydmVyVVJMPgogICAgICAgICAgICAgICAgPC9jb25maWd1cmF0aW9uPgogICAgICAgICAgICA8L3BsdWdpbj4KICAgYGBgCjIuIOeUn+aIkOaOpeWPo+aWh+aho+W5tuS4iuS8oOWIsOaOpeWPo+aWh+aho+euoeeQhuW3peWFtwogICBgYGBzaGVsbAogICBtdm4gY29tcGlsZSBjb2RlZ2VuOm9wZW5hcGkKICAgYGBgCjMuIOWcqOebruW9lSR7YmFzZWRpcn0vdGFyZ2V0L29wZW5hcGnkuIvlj6/ku6XnnIvliLDovpPlh7rnmoRvcGVuYXBpIGpzb27mlofmoaM= readmeEtag: '"ad59ba1e8e950dd586f32b82e279657f9b4ab208"' readmeLastModified: Tue, 28 Sep 2021 06:36:33 GMT repositoryId: 367232700 description: 这是一个分析Spring项目中的Java源代码(注释)以及class文件生成OpenAPI接口文档的工具. created: '2021-05-14T02:54:23Z' updated: '2021-09-28T06:36:51Z' language: Java archived: false stars: 0 watchers: 2 forks: 0 owner: tanqidong1992 logo: https://avatars.githubusercontent.com/u/9072280?v=4 repoEtag: '"eec6edcca6d6ae5998a17bc106829464f6706ef509e69cc46e3e440bc26441a0"' repoLastModified: Tue, 28 Sep 2021 06:36:51 GMT foundInMaster: true category: Code Generators id: 8a2230e05d970d422958625db817d08c - source: openapi3 tags repository: https://github.com/sylvain12/mven-project v3: true repositoryMetadata: base64Readme: >- IyBNVkVOIChNb25nb0RCLCBWdWUuSlMsIEV4cHJlc3MsIE5vZGUuSlMpIFBST0pFQ1QKCnRoZSBwcm9qZWN0IGltcGxlbWVudCBDUlVEIG9wZXJhdGlvbnMgb24gY2xpZW50cyBhbmQgcHJvdmlkZXJzLgoKLS0tCiMjIDEuIFJlcXVpcmVtZW50cwoKRm9yIGRldmVsb3BtZW50LCB5b3Ugd2lsbCBvbmx5IG5lZWQgYE5vZGUuanNgIGFuZCBhIG5vZGUgZ2xvYmFsIHBhY2thZ2UsIGBucG1gLCBgZ2l0YCwgYG1vbmdvZGJgIChGb3IgZGF0YWJhc2UpIGluc3RhbGxlZCBpbiB5b3VyIGVudmlyb25tZW50LgoKIyMjIGBOb2RlLmpzIC8gbnBtYAotICMjIyMgTm9kZSBpbnN0YWxsYXRpb24gb24gV2luZG93cwoKICBKdXN0IGdvIG9uIFtvZmZpY2lhbCBOb2RlLmpzIHdlYnNpdGVdKGh0dHBzOi8vbm9kZWpzLm9yZy8pIGFuZCBkb3dubG9hZCB0aGUgaW5zdGFsbGVyLgoKLSAjIyMjIE5vZGUgaW5zdGFsbGF0aW9uIG9uIExpbnV4IChVYnVudHUpCgogIFlvdSBjYW4gaW5zdGFsbCBub2RlanMgYW5kIG5wbSBlYXNpbHkgd2l0aCBhcHQgaW5zdGFsbCwganVzdCBydW4gdGhlIGZvbGxvd2luZyBjb21tYW5kcy4KCiAgICAgICQgc3VkbyBhcHQgdXBkYXRlCiAgICAgICQgc3VkbyBhcHQgaW5zdGFsbCBub2RlanMKICAgICAgJCBzdWRvIGFwdCBpbnN0YWxsIG5wbQoKLSAjIyMjIE90aGVyIE9wZXJhdGluZyBTeXN0ZW1zCiAgWW91IGNhbiBmaW5kIG1vcmUgaW5mb3JtYXRpb24gYWJvdXQgdGhlIGluc3RhbGxhdGlvbiBvbiB0aGUgW29mZmljaWFsIE5vZGUuanMgd2Vic2l0ZV0oaHR0cHM6Ly9ub2RlanMub3JnLykgYW5kIHRoZSBbb2ZmaWNpYWwgTlBNIHdlYnNpdGVdKGh0dHBzOi8vbnBtanMub3JnLykuCgpJZiB0aGUgaW5zdGFsbGF0aW9uIHdhcyBzdWNjZXNzZnVsLCB5b3Ugc2hvdWxkIGJlIGFibGUgdG8gcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZC4KCiAgICAkIG5vZGUgLS12ZXJzaW9uCiAgICB2OC4xMS4zCgogICAgJCBucG0gLS12ZXJzaW9uCiAgICA2LjEuMAoKSWYgeW91IG5lZWQgdG8gdXBkYXRlIGBucG1gLCB5b3UgY2FuIG1ha2UgaXQgdXNpbmcgYG5wbWAhIENvb2wgcmlnaHQ/IEFmdGVyIHJ1bm5pbmcgdGhlIGZvbGxvd2luZyBjb21tYW5kLCBqdXN0IG9wZW4gYWdhaW4gdGhlIGNvbW1hbmQgbGluZSBhbmQgYmUgaGFwcHkuCgogICAgJCBucG0gaW5zdGFsbCBucG0gLWcKCiMjIyBgR2l0YAotICMjIyMgR2l0IGluc3RhbGxhdGlvbiBvbiBXaW5kb3dzClZpc2l0OiB0aGlzIHNpdGUgaHR0cHM6Ly9naXQtc2NtLmNvbS8gYW5kIGRvd25sb2FkIGV4ZWN1dGFibGUgZmlsZSBhbmQgcnVuIGl0CgotICMjIyMgR2l0IGluc3RhbGxhdGlvbiBvbiBMaW51eChVYnVudHUpCgogICAgICAgJCBzdWRvIGFwdC1nZXQgaW5zdGFsbCBnaXQKCkFmdGVyIHlvdSBjYW4gcnVuIHRoaXMgY29tbWFuZCB0byBjaGVjawoKICAgICAgJCBnaXQgLS12ZXJzaW9uCiAgICAgIGdpdCB2ZXJzaW9uIDIuMjUuMQoKIyMjIGBNb25nb0RCYAoKVmlzaXQgdGhpcyBsaW5rIGFuZCBmb2xsb3cgdGhlIGluc3RhbGxhdGlvbiBpbnN0cnVjdGlvbnMgZm9yIHlvdXIgT1MKCj0+IFtVYnVudHUgaW5zdGFsbGF0aW9uIGxpbmtdKGh0dHBzOi8vZG9jcy5tb25nb2RiLmNvbS9tYW51YWwvdHV0b3JpYWwvaW5zdGFsbC1tb25nb2RiLW9uLXVidW50dS8pCgo9PiBbV2luZG93cyBpbnN0YWxsYXRpb24gbGlua10oaHR0cHM6Ly9kb2NzLm1vbmdvZGIuY29tL21hbnVhbC90dXRvcmlhbC9pbnN0YWxsLW1vbmdvZGItb24td2luZG93cy8pCgo9PiBbbWFjT1MgaW5zdGFsbGF0aW9uIGxpbmtdKGh0dHBzOi8vZG9jcy5tb25nb2RiLmNvbS9tYW51YWwvdHV0b3JpYWwvaW5zdGFsbC1tb25nb2RiLW9uLW9zLXgtdGFyYmFsbC8pCgpBZnRlciBtb25nb0RCIGluc3RhbGxpbmcgcnVuCgogICAgJCBtb25nbyAtLXZlcnNpb24KCiAgICAgIEJ1aWxkIEluZm86IHsKICAgICAgInZlcnNpb24iOiAiNC40LjQiLAogICAgICAiZ2l0VmVyc2lvbiI6ICI4ZGIzMGE2M2RiMWE5ZDg0YmRjYWQwYzgzMzY5NjIzZjcwOGUwMzk3IiwKICAgICAgIm1vZHVsZXMiOiBbXSwKICAgICAgImFsbG9jYXRvciI6ICJzeXN0ZW0iLAogICAgICAiZW52aXJvbm1lbnQiOiB7CiAgICAgICAgICAiZGlzdGFyY2giOiAieDg2XzY0IiwKICAgICAgICAgICJ0YXJnZXRfYXJjaCI6ICJ4ODZfNjQiCiAgICAgIH0KCkNvbm5lY3QgdG8gbW9uZ29EQiBzaGVsbAoKICAgICQgbW9uZ28KICAgID4gc2hvdyBkYnMKCiAgICBhZG1pbiAgICAgICAgIDAuMDAwR0IKICAgIGNsaWVudC1hcHAgICAgMC4wMDBHQgogICAgY29uZmlnICAgICAgICAwLjAwMEdCCiAgICBsb2NhbCAgICAgICAgIDAuMDAwR0IKCi0tLQojIyAyLiBJbnN0YWxsIHByb2plY3QKCgpDbG9uZSB0aGUgcHJvamVjdCBjb2RlIG9uIHlvdXIgbWFjaGluZQoKICAgICQgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9zeWx2YWluMTIvbXZlbi1wcm9qZWN0CiAgICAkIGNkIG12ZW4tcHJvamVjdAogICAgJCBscwogICAgUmVhZG1lLm1kICAgY2xpZW50ICAgY29uZmlnLmV4YW1wbGUuZW52IG1vZGVscyAgIHBhY2thZ2UtbG9jay5qc29uICByb3V0ZXMgICAgIHV0aWxzIGFwcC5qcyAgICBjb25maWcuZW52ICAgIGNvbnRyb2xsZXJzICAgcGFja2FnZS5qc29uICBzZXJ2ZXIuanMKCiMjIDMuIENvbmZpZ3VyYXRvbgogIyMjIGBOb2RlIENvbmZpZ3VyYXRpb25gCiBBdCB0aGUgcm9vdCBvZiB0aGUgcHJvamVjdCBydW4gOgoKICAgICQgbnBtIGluc3RhbGwKICMjIyBgQ2xpZW50IHNpZGUgKFZ1ZS5KUykgQ29uZmlndXJhdGlvbmAKICAgICQgY2QgY2xpZW50CiAgICAkIG5wbSBpbnN0YWxsCgogIyMjIGBEYXRhYmFzZSBDb25maWd1cmF0aW9uYApDb25uZWN0IHRvIG1vbmdvREIgYW5kIGNyZWF0ZSBhIGRhdGFiYXNlCgogICAgJCBtb25nbwogICAgPiB1c2UgPGRhdGFiYXNlX25hbWU+CiAgICBzd2l0Y2hlZCB0byBkYiA8ZGF0YWJhc2VfbmFtZT4KICAgID4gcXVpdCgpCiAgICAkIAogICAgCiMjIyBgRW52aXJvbm1lbnQgdmFyaWFibGVzIGNvbmZpZ3VyYXRpb25gCm1vdmUgdGhlIGBjb25maWcuZXhhbXBsZS5lbnZgIHRvIGBjb25maWcuZW52YApNYWtlIHN1cmUgdG8gaW4gdGhlIHJvb3Qgb2YgdGhlIHByb2plY3QKCiAgICAkIG12IGNvbmZpZy5leGFtcGxlLmVudiBjb25maWcuZW52CgpvcGVuIHRoZSBgY29uZmlnLmVudmAgZmlsZSBhbmQgc2V0IHZhcmlhYmxlIGVudmlyb25tZW50CgpSZXBsYWNlIGFsbCB0aGUgdmFyaWFibGUgYmV0d2VlbiB3aXRoIHRoZWlyIGNvcnJlc3BvbmRpbmcgdmFsdWUKCiAgICBQT1JUPTxOT0RFX1NFUlZFUl9QT1JUPgogICAgREFUQUJBU0VfTE9DQUw9bW9uZ29kYjovL2xvY2FsaG9zdDo8TU9OR09EQl9QT1JUPi88REFUQUJBU0VfTkFNRT4KICAgIERBVEFCQVNFX1BST0Q9PFBST0RVQ1RJT05fREFUQUJBU0VfVVJJPgogICAgTk9ERV9FTlY9ZGV2ZWxvcG1lbnQKTkI6IGZvciBkZXBsb3ltZW50IHdlIHVzZSBgcHJvZHVjdGlvbmAKCkFmdGVyIHZhcmlhYmxlcyBlbnZpcm9ubWVudCBjb25maWd1cmF0aW9uLCBzZXQgYE5PREVfUE9SVGAgb24gdnVlLmpzIGNvbmZpZyB0byBwcm94eSBub2RlLmpzIHNlcnZlciBBUEkuIE1ha2Ugc3VyZSB0byBiZSBvbiBjbGllbnQgZm9sZGVyLCBhbmQgb3BlbiBgdnVlLmNvbmZpZy5qc2AgZmlsZS4KCiAgICAkIG5hbm8gdnVlLmNvbmZpZy5qcwoKICAgIHByb3h5OiB7CiAgICAgICcvYXBpL3YxJzogewogICAgICAgIHRhcmdldDogYGh0dHA6Ly9sb2NhbGhvc3Q6PE5PREVfUE9SVD5gCiAgICAgIH0KICAgIH0KRG9uJ3QgZm9yZ2V0IHRvIHNhdmUgdGhlIGZpbGUuCgojIyA0LiBSdW5uaW5nIHRoZSBwcm9qZWN0CgojIyMgYGRldmVsb3BtZW50YAogICAgJCBucG0gcnVuIGRldgoKICBgTE9DQUxfU0VSVkVSYCA6IGBsb2NhbGhvc3Q6Tk9ERV9QT1JUYAoKICBgQVBJX1JPVVRFYCA6IGBsb2NhbGhvc3Q6Tk9ERV9QT1JUL2FwaS1kb2NzYAoKICBgQ0xJRU5UX0FQUGAgOiBgbG9jYWxob3N0OjgwMDBgCiAgCgojIyAtIFBST0pFQ1QgQVBJIApWaXNpdCB0aGlzIGFkZHJlc3Mgb24geW91ciB3ZWIgYnJvd3NlciBmb3IgcHJvamVjdCBBUEkgZG9jdW1lbnRhdGlvbiB3aXRoIFtzd2FnZ2VyXShodHRwczovL3N3YWdnZXIuaW8vKSBhbmQgW29wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pCgoKIyBgbG9jYWxob3N0OjxOT0RFX1BPUlQ+L2FwaS1kb2NzYAoKPCEtLSAjIyMgYHByb2R1Y3Rpb25gCiAgICAkIG5wbSBydW4gcHJvZCAtLT4KCgojIyA1LiBUZXN0aW5nCkNvbWluZyBzb29uLi4uIQ== readmeEtag: '"ec76da3d87edfb7e7f608e5bc7d5db08534a2dbe"' readmeLastModified: Fri, 23 Apr 2021 00:58:55 GMT repositoryId: 358070233 description: MVEN STACK project (Mongodb, Vue.JS, Express, Node.JS) created: '2021-04-14T23:35:46Z' updated: '2025-10-15T17:12:01Z' language: Vue archived: false stars: 1 watchers: 1 forks: 0 owner: sylvain12 logo: https://avatars.githubusercontent.com/u/26870715?v=4 repoEtag: '"1a985a0b583ae8447c78e84e23d32570d897ed16628648339a7acfd0462676a8"' repoLastModified: Wed, 15 Oct 2025 17:12:01 GMT foundInMaster: true category: - Server - Server Implementations id: 5e44a9a1f0e2d5629b2e6650fc04a96f - source: openapi3 tags repository: https://github.com/mizyind/saunter v3: true repositoryMetadata: base64Readme: >- IyBzYXVudGVyCgpbIVtHb10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvZ28tbW9kL2dvLXZlcnNpb24vbWl6eWluZC9zYXVudGVyP3N0eWxlPWZvci10aGUtYmFkZ2UmbGFiZWw9JmNvbG9yPTAwYWRkOCZsb2dvPWdvJmxvZ29Db2xvcj1mZmYpXShodHRwczovL2dvbGFuZy5vcmcpCgpTYXVudGVyIGlzIGFuIG9waW5pb25hdGVkIFN3YWdnZXIgMy4wIGRvY3VtZW50YXRpb24gZ2VuZXJhdG9yIGZvciBHbydzIEdpbiBmcmFtZXdvcmsuCgojIyBBdXRob3IKCm1pWnlpbmQgPG1penlpbmRAZ21haWwuY29tPgoKIyMgTElDRU5TRQoKTGljZW5zZWQgdW5kZXIgdGhlIFtNSVRdKExJQ0VOU0UpIExpY2Vuc2UuCg== readmeEtag: '"cae269dbf1d7470871d1cc5d275ed23744c72b13"' readmeLastModified: Thu, 21 Jan 2021 11:10:25 GMT repositoryId: 312517706 description: >- Saunter is an opinionated Swagger 3.0 documentation generator for Go's Gin framework. created: '2020-11-13T08:32:58Z' updated: '2023-01-28T05:33:54Z' language: Go archived: true stars: 0 watchers: 1 forks: 0 owner: miZyind logo: https://avatars.githubusercontent.com/u/21041375?v=4 license: MIT repoEtag: '"b90e5a4ff88859db8a783311a5abb7ccfdbc8c45802e8f11fff8c77c50f47b75"' repoLastModified: Sat, 28 Jan 2023 05:33:54 GMT foundInMaster: true category: - Data Validators - Server Implementations id: d3ff2966a56845108bcd9932be1376d3 - source: openapi3 tags repository: https://github.com/jeanpierm/rest-api-spring-jpa v3: true repositoryMetadata: base64Readme: >- IyBSRVNUZnVsIEFQSSBTcHJpbmcgQm9vdA0KDQohW1NwcmluZyBCb290ICsgSlBBXShodHRwczovL21pcm8ubWVkaXVtLmNvbS9tYXgvMTMzMi8wKmllR3RGZVRBd3FwMGl6azAucG5nKQ0KDQojIyBUZWNobm9sb2dpZXMgdXNlZA0KDQotIEphdmEgMTENCi0gU3ByaW5nIEJvb3QgKFNwcmluZyBXZWIpDQotIFNwcmluZyBTZWN1cml0eSAoc3RhdGVsZXNzIHdpdGggSldUIGFuZCByb2xlcykNCi0gSlBBIChoaWJlcm5hdGUpDQotIEphdmF4IFZhbGlkYXRpb25zIHRvIHZhbGlkYXRlIHRoZSBwcm9wZXJ0aWVzIG9mIHRoZSBlbnRpdGllcyBpbiB0aGUgcmVxdWVzdHMuDQotIFBvc3RncmVTUUwgMTQgKyBwZ0FkbWluICh3aXRoIGRvY2tlcikNCi0gT3BlbkFQSSAzIFtzcHJpbmdkb2NdKGh0dHBzOi8vc3ByaW5nZG9jLm9yZy8pDQotIE1vZGVsTWFwcGVyIChmb3IgbWFwcGluZyBkdG9zIHRvIGpwYSBlbnRpdGllcyBlYXNpbHkpDQoNCiMjIFJ1biBQb3N0Z3JlU1FMICsgcGdBZG1pbiB3aXRoIERvY2tlcg0KDQpgYGBiYXNoDQojIHVwIGNvbnRhaW5lcnMNCiQgZG9ja2VyLWNvbXBvc2UgdXAgLWQgcG9zdGdyZXMgcGdhZG1pbg0KDQojIGRvd24gY29udGFpbmVycw0KJCBkb2NrZXItY29tcG9zZSBkb3duDQpgYGANCg0KIyMgTWljcm9zZXJ2aWNlIChEb2NrZXJpemF0aW9uKQ0KDQpgYGBiYXNoDQpkb2NrZXIgYnVpbGQgLXQgamVhbnAwL3Jlc3QtYXBpLXNwcmluZy1ib290IC4NCmBgYA0KDQpSdW4gYSAqKnBvc3RncmVzKiosICoqcGdhZG1pbioqIGFuZCAqKnNwcmluZy1hcHAqKiBpbWFnZXMgd2l0aCBkb2NrZXItY29tcG9zZS4gVGhlICoqc3ByaW5nLWFwcCoqIGltYWdlIGlzIGJ1aWx0IGF1dG9tYXRpY2FsbHkgaWYgaXQgZG9lc24ndCBleGlzdC4NCg0KYGBgYmFzaA0KIyB1cCBjb250YWluZXJzDQokIGRvY2tlci1jb21wb3NlIHVwIC1kDQoNCiMgZG93biBjb250YWluZXJzDQokIGRvY2tlci1jb21wb3NlIGRvd24NCg0KIyB2aWV3IGxvZ3MNCiQgZG9ja2VyLWNvbXBvc2UgbG9ncyAtZg0KYGBgDQoNCiMjIENhcHR1cmVzDQoNCiMjIyBBcHBsaWNhdGlvbiBydW5uaW5nIHdpdGggRG9ja2VyIHZpYSBkb2NrZXItY29tcG9zZQ0KDQohW0FwcGxpY2F0aW9uIHJ1bm5pbmcgd2l0aCBEb2NrZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9qZWFucGllcm0vcmVzdC1hcGktc3ByaW5nLWJvb3QvYmxvYi9tYWluL2NhcHR1cmVzLzkucG5nP3Jhdz10cnVlKQ0KDQojIyMgU3dhZ2dlciBVSQ0KDQohW1N3YWdnZXIgVUldKGh0dHBzOi8vZ2l0aHViLmNvbS9qZWFucGllcm0vcmVzdC1hcGktc3ByaW5nLWJvb3QvYmxvYi9tYWluL2NhcHR1cmVzLzEucG5nP3Jhdz10cnVlKQ0KDQojIyMgU2NoZW1hcyBieSBPcGVuQVBJIDMNCg0KIVtTY2hlbWFzIGJ5IE9wZW5BUEkgM10oaHR0cHM6Ly9naXRodWIuY29tL2plYW5waWVybS9yZXN0LWFwaS1zcHJpbmctYm9vdC9ibG9iL21haW4vY2FwdHVyZXMvMi5wbmc/cmF3PXRydWUpDQoNCiMjIyBTdWNjZXNzIGxvZ2luDQoNCiFbU3VjY2VzcyBsb2dpbl0oaHR0cHM6Ly9naXRodWIuY29tL2plYW5waWVybS9yZXN0LWFwaS1zcHJpbmctYm9vdC9ibG9iL21haW4vY2FwdHVyZXMvMy5wbmc/cmF3PXRydWUpDQoNCiMjIyBGYWlsIGxvZ2luDQoNCiFbRmFpbCBsb2dpbl0oaHR0cHM6Ly9naXRodWIuY29tL2plYW5waWVybS9yZXN0LWFwaS1zcHJpbmctYm9vdC9ibG9iL21haW4vY2FwdHVyZXMvNC5wbmc/cmF3PXRydWUpDQoNCiMjIyBHRVQgd2l0aCB0b2tlbiBleHBpcmVkDQoNCiFbR0VUIHdpdGggdG9rZW4gZXhwaXJlZF0oaHR0cHM6Ly9naXRodWIuY29tL2plYW5waWVybS9yZXN0LWFwaS1zcHJpbmctYm9vdC9ibG9iL21haW4vY2FwdHVyZXMvNS5wbmc/cmF3PXRydWUpDQoNCiMjIyBHRVQgd2l0aG91dCB0b2tlbg0KDQohW0dFVCB3aXRob3V0IHRva2VuXShodHRwczovL2dpdGh1Yi5jb20vamVhbnBpZXJtL3Jlc3QtYXBpLXNwcmluZy1ib290L2Jsb2IvbWFpbi9jYXB0dXJlcy82LnBuZz9yYXc9dHJ1ZSkNCg0KIyMjIEdFVCB3aXRoIHZhbGlkIHRva2VuDQoNCiFbR0VUIHdpdGggdmFsaWQgdG9rZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9qZWFucGllcm0vcmVzdC1hcGktc3ByaW5nLWJvb3QvYmxvYi9tYWluL2NhcHR1cmVzLzcucG5nP3Jhdz10cnVlKQ0KDQojIyMgUmVmcmVzaCB0b2tlbg0KDQohW1JlZnJlc2ggdG9rZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9qZWFucGllcm0vcmVzdC1hcGktc3ByaW5nLWJvb3QvYmxvYi9tYWluL2NhcHR1cmVzLzgucG5nP3Jhdz10cnVlKQ0K readmeEtag: '"4e3ce46becc95584e38a710017c9203c9a43dc08"' readmeLastModified: Tue, 09 Aug 2022 02:22:10 GMT repositoryId: 410396224 description: >- RESTful API with JWT & Swagger. Built with Spring Boot, JPA, Spring Security, PostgreSQL and dockerized. created: '2021-09-25T22:37:44Z' updated: '2021-12-09T01:48:04Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: jeanpierm logo: https://avatars.githubusercontent.com/u/67121413?v=4 repoEtag: '"c4be7232dc6c35db14eb3b63b6864b4ff64cc7e6bafc945a7113ba28a95ef5e4"' repoLastModified: Thu, 09 Dec 2021 01:48:04 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: f1f18180bc7393a0e6a8d5ee9b252979 - source: openapi3 tags repository: https://github.com/maxstanley/xeffect_backend v3: true repositoryMetadata: base64Readme: IyB4ZWZmZWN0X2JhY2tlbmQKQmFja2VuZCBmb3IgdGhlIFhFZmZlY3QK readmeEtag: '"9d62e1d641c85cbfd5138de1b2c765d26f3bc9df"' readmeLastModified: Wed, 29 Dec 2021 21:48:04 GMT repositoryId: 441542078 description: Backend for the XEffect created: '2021-12-24T19:34:43Z' updated: '2021-12-29T21:48:08Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: maxstanley logo: https://avatars.githubusercontent.com/u/66874003?v=4 license: MIT repoEtag: '"ee852bd1c3b549d239355da15533c92c4ac09110a2083064f27b5d319d4d4694"' repoLastModified: Wed, 29 Dec 2021 21:48:08 GMT foundInMaster: true category: - Server - Server Implementations id: c3aaf516fc90134a965d89b84d0b7c61 - source: openapi3 tags repository: https://github.com/raghav2211/bug-tracker v3: true id: 6c75b861ec7154f50bdd7f1654ac41a2 repositoryMetadata: base64Readme: >- IyBidWctdHJhY2tlcgpodHRwOi8vbG9jYWxob3N0OjgwODAvd2ViamFycy9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwjLwo= readmeEtag: '"dec33c4cb803cd3e538999e841474f6913327102"' readmeLastModified: Sun, 22 Jan 2023 10:50:40 GMT repositoryId: 564781575 description: null created: '2022-11-11T13:38:12Z' updated: '2022-12-19T14:05:14Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Raghav2211 logo: https://avatars.githubusercontent.com/u/7526431?v=4 repoEtag: '"6bc3d5c6429d8b70d9108f56932c6a83eecaa6635ab050299ba52cdcbaea2e7c"' repoLastModified: Mon, 19 Dec 2022 14:05:14 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/hailiangxie/myopenapi_codegen_resourceserver_proj v3: true id: a1222678295875fc2f572b4810f1ce63 repositoryMetadata: base64Readme: >- # OpenAPI Code Generation and OAuth2 Resource Server
This is a demo application to generate the REST APIs with OpenAPI and protect them with OAuth2 Reousrce Server. In this demo application, we generate the customer APIs and implement them. After that we protect those APIs by configuring the OAuth2 Resource Server. To access those APIs, we have to prived the signed JWT access token which had been issued by the Authorization Server.
</br>
</br>
Please read the documents to find more details about [OpenAPI Specficiation](https://swagger.io/specification/) and [OAuth2.0](https://oauth.net/2/).
</br>
</br>
Prerequisites: [Java8 or above](https://openjdk.org/)
## Getting Started
To install this demo application, please run the following command in a terminal window:
```cmd
git clone https://github.com/hailiangxie/myopenapi_codegen_resourceserver_proj.git
cd myopenapi_codegen_resourceserver_proj
```
## Generating APIs
To generate APIs, first we need to create an API Spec file called `api.yaml` and put it to `src/main/reousrces/api`:
```yaml
openapi: 3.0.2
info:
  title: Customer - OpenAPI 3.0
  description: |-
    This is a sample Customer Server based on the OpenAPI 3.0 specification.  You can find out more about
    Swagger at [http://swagger.io](http://swagger.io).

  version: 1.0.0
externalDocs:
  description: Find out more about Swagger
  url: http://swagger.io
servers:
  - url: /api/v3
tags:
  - name: customer
    description: Everything about the customers
    externalDocs:
      description: Find out more
      url: http://swagger.io
```
Second we need to configure the API package in the project `pom` file:
```xml
<plugin>
    			<groupId>org.openapitools</groupId>
    			<artifactId>openapi-generator-maven-plugin</artifactId>
    			<version>4.2.2</version>
    			<executions>
        			<execution>
            			<goals>
                			<goal>generate</goal>
            			</goals>
            			<configuration>
                			<inputSpec>${api.spec.file}</inputSpec>
                			<output>${project.build.directory}/generated/open-api</output>
                			<generatorName>spring</generatorName>
                			<apiPackage>xie.hailiang.resourceserver.api</apiPackage>
                			<modelPackage>xie.hailiang.resourceserver.model</modelPackage>
                			<invokerPackage>xie.hailiang.resourceserver.handler</invokerPackage>
                			<modelNameSuffix>Json</modelNameSuffix>
                			<configOptions>
                    			<delegatePattern>true</delegatePattern>
                			</configOptions>
            			</configuration>
        			</execution>
    			</executions>
			</plugin>
```
And finally we can run the following command in a terminal window:
```cmd
cd myopenapi_codegen_resourceserver_proj
./mvnw install
```
The customer APIs should be generated.
## Implementing APIs
To implement the generated REST APIs, we need to implement the generated delegate interface `CustomerApiDelegate`:
```java
@Component
public class CustomerApiDelegateImpl implements CustomerApiDelegate {
  // implement the methods declared in CustomerApiDelegate
}
```
## Protecting APIs
To protect the generated REST APIs, we need to configure the OAuth2 Resource Server in the project:
</br>
First we add the OAuth2 and Reousrce Server dependencies to the porject `pom`.
```xml
<dependency>
   			<groupId>org.springframework.boot</groupId>
   			<artifactId>spring-boot-starter-oauth2-resource-server</artifactId>
		</dependency>
		<dependency>
   			<groupId>org.springframework.cloud</groupId>
   			<artifactId>spring-cloud-starter-oauth2</artifactId>
   			<version>2.2.5.RELEASE</version>
		</dependency>
```
Second we add the `ResourceServerConfig` class to enable Resource Server functionality.
```java
@Configuration
@EnableResourceServer
public class ResourceServerConfig extends ResourceServerConfigurerAdapter {
}
```
And third we need to configure the cryptographic public key in `src/main/resources/application.yml`.
</br>
The public key was generated by calling the endpoint `/oauth/token_key` from the [Authroization Server](https://github.com/hailiangxie/myoauth2authorization_proj).
```yml
jwtkey: 
  publicKey: "-----BEGIN PUBLIC KEY-----\n...\n-----END PUBLIC KEY-----"
```
The Resource Server protects the APIs by validating that if the client has a valid JWT access token.
## Run the Application
To run this demo application, please run the following command in a terminal window:
```cmd
cd myopenapi_codegen_resourceserver_proj
./mvnw spring-boot:run
```
After everything starts, we should be able to test the Application.
## Test
Now we shoulbe be able to test the customer REST APIs. To view the API docs and test the APIs we can open the swagger-ui by accessing the url `http://localhost:8091/swagger-ui.html` in the web browser. And also we can test the APIs with other tools (e.g., `curl`, `postman`). For example, we test the API `/api/v3/customer` to add a new customer.
- Access the API `http://localhost:8091/api/v3/customer` without a token:
```
Request Method: Post
Accept: application/json
ContentType: application/json
Content: The new customer json string
Expect: It should return status 401 Unauthorized
```
- Access the API `http://localhost:8091/api/v3/customer` without an invalid token:
```
Request Method: Post
Header: Authorization: Bearer itisaninvalidtoken
Accept: application/json
ContentType: application/json
Content: The new customer json string
Expect: It should return status 401 Unauthorized
```
- Access the API `http://localhost:8091/api/v3/customer` with a valid token (by calling the endpoint `/oauth/token` from the [Authorization Server](https://github.com/hailiangxie/myoauth2authorization_proj)):
```
Request Method: Post
Header: Authorization: Bearer <Valid token>
Accept: application/json
ContentType: application/json
Content: The new customer json string
Expect: It should return status 200 and the new customer created
```
## See Also
The following guides may also be helpful:
- [Swagger UI](https://swagger.io/tools/swagger-ui/)
- [JSON Web Tokens](https://jwt.io/)
 readmeEtag: '"7fa28cbe4dc55de21ba18d5d8e7b0fbf40b2ab37"' readmeLastModified: Sat, 01 Oct 2022 15:51:28 GMT repositoryId: 542639581 description: >- This is a demo application to generate and implement REST APIs with OpenAPI and protect them with Oauth2.0 Resource Server created: '2022-09-28T14:42:25Z' updated: '2022-09-29T04:07:27Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: hailiangxie logo: https://avatars.githubusercontent.com/u/113340734?v=4 repoEtag: '"e60c650ac91af4c4916eb07923d8cc9ec43b9ce4063d3104ed1cd8239478de0e"' repoLastModified: Thu, 29 Sep 2022 04:07:27 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/nickfallon/catalyst v3: true id: a05953d17866a75dbaf494b9c80c8b03 repositoryMetadata: base64Readme: >- IyAhW0NhdGFseXN0XShsb2dvLnN2ZykgQ2F0YWx5c3QKCkFuIE9wZW5BUEktY29tcGF0aWJsZSBSRVNUIEFQSSBnZW5lcmF0b3IgZm9yIG5vZGUvcG9zdGdyZXMKCiMjIFdoYXQgaXQgZG9lcwoKV2hlbiBwcm92aWRlZCB3aXRoIGEgUG9zdGdyZXMgZGF0YWJhc2UsIENhdGFseXN0IGRvZXMgdGhlIGZvbGxvd2luZzoKCi0gQ3JlYXRlcyBhbiBPcGVuQVBJMyBzcGVjaWZpY2F0aW9uIGZyb20gdGhlIHRhYmxlcyBhbmQga2V5cyBhbmQgc2F2ZXMgdGhlIHJlc3VsdCB0byBgb3BlbmFwaS4zLjAuMC5qc29uYCBpbiB0aGUgcm9vdCBmb2xkZXIuCgotIENyZWF0ZXMgYSBSRVNUIEFQSSB0aGF0IGNvbnN1bWVzIHRoZSBPcGVuQVBJIHNwZWMgYW5kIHB1dHMgdGhlIGF1dG9nZW5lcmF0ZWQgY29kZSBpbiB0aGUgYC9hcGlgIGZvbGRlci4gQWxsIEFQSSBlbmRwb2ludHMgYXJlIHByZWNlZGVkIHdpdGggdGhlIGNvbW1vbiBwYXRoIGAvYXBpL3YxL2AuCgotIENyZWF0ZXMgaW50ZXJhY3RpdmUgU3dhZ2dlciBkb2NzIGF0IGAvYXBpL3YxL2FwaS1kb2NzYC4KCiMjIFNldHRpbmcgdXAKIyMjIEdlbmVyYXRpbmcgY2VydGlmaWNhdGVzCgpUaGlzIHN0ZXAgZ2VuZXJhdGVzIHR3byBmaWxlcywgYGxvY2FsaG9zdC5wZW1gIGFuZCBgbG9jYWxob3N0LWtleS5wZW1gIGluIHRoZSBmb2xkZXIgYGNlcnRzYC4KVGhpcyBtYWtlcyB0aGUgbG9jYWwgd2ViIGFwcCB3b3JrIG92ZXIgSFRUUFMKCkNyZWF0ZSBhIGZvbGRlciBjYWxsZWQgYGNlcnRzYCBpbiB0aGUgbWFpbiBkaXJlY3RvcnkgYW5kIG1vdmUgdG8gaXQ6CgpgJCBta2RpciBjZXJ0c2AKCmAkIGNkIGNlcnRzYAoKSW5zdGFsbCB0aGUgYG1rY2VydGAgcGFja2FnZToKCmAkIGJyZXcgaW5zdGFsbCBta2NlcnRgCgpgJCBta2NlcnQgLWluc3RhbGxgCgpHZW5lcmF0ZSB0aGUga2V5IGZpbGVzOgoKYCQgbWtjZXJ0IGxvY2FsaG9zdGAKCgojIyMgVXBkYXRlIC5lbnYgZmlsZQoKUmVuYW1lIGAuZW52LmV4YW1wbGVgIHRvIGAuZW52YCBhbmQgc2V0IHRoZSBwb3N0Z3JlcyBjb25uZWN0aW9uIGluZm8uCgojIyBIb3cgdG8gcnVuCgpVc2U6IAoKYG5wbSBydW4gZGV2YAoKIyMjIEdlbmVyYXRpbmcgdGhlIEFQSSAKCipXaGVuIHJ1bm5pbmcgZm9yIHRoZSBmaXJzdCB0aW1lOioKCkluIGEgYnJvd3NlciwgZ28gdG8gW2h0dHBzOi8vbG9jYWxob3N0OjkwMDAvYXBpL2dlbmVyYXRvci9idWlsZF0oaHR0cHM6Ly9sb2NhbGhvc3Q6OTAwMC9hcGkvZ2VuZXJhdG9yL2J1aWxkKQoKVGhlIGdlbmVyYXRvciBjb25uZWN0cyB0byBhIHBvc3RncmVzIGRhdGFiYXNlIHVzaW5nIHRoZSBjcmVkZW50aWFscyBpbiBgLmVudmAgZmlsZS4KSXQgdGhlbiBjcmVhdGVzIGEgZmlsZSBgb3BlbmFwaS4zLjAuMC5qc29uYCBhbmQgd3JpdGVzIGNvZGUgdG8gdGhlIGAvYXBpYCBmb2xkZXIgdG8gcGVyZm9ybSBTUUwgcXVlcmllcy4KClJlLXJ1bm5pbmcgYC9hcGkvZ2VuZXJhdG9yL2J1aWxkYCB3aWxsIG92ZXJ3cml0ZSB0aGUgZmlsZXMgY3JlYXRlZCBpbiBgL2FwaWAgYW5kIHRoZSBgb3BlbmFwaS4zLjAuMC5qc29uYCBmaWxlLgoKKk5vdGU6KiBEb24ndCB1c2Ugbm9kZW1vbiB3aGVuIGdlbmVyYXRpbmcgdGhlIEFQSSwgc2luY2UgZHluYW1pYyBjaGFuZ2VzIHRvIHRoZSBzb3VyY2UgdHJpZ2dlciBhIHJlc3RhcnQgd2hpY2ggc3RvcHMgdGhlIGdlbmVyYXRvci4KCgojIyMgVXNpbmcgdGhlIEFQSSAKClJlLXN0YXJ0IHRoZSBhcHAuIEV4cHJlc3Mgcm91dGVzIGFyZSBhdXRvbWF0aWNhbGx5IG1hcHBlZCB0byB0aGUgZ2VuZXJhdGVkIGAvYXBpYCBmb2xkZXIgY29kZS4KCkluIGEgYnJvd3NlciwgZ28gdG8gIFtodHRwczovL2xvY2FsaG9zdDo5MDAwL2FwaS92MS9hcGktZG9jc10oaHR0cHM6Ly9sb2NhbGhvc3Q6OTAwMC9hcGkvdjEvYXBpLWRvY3MpIHRvIHNlZSBpbnRlcmFjdGl2ZSBBUEkgZG9jcy4KCipOb3RlOiogWW91IG11c3QgcHJvdmlkZSBhbiBBdXRob3JpemF0aW9uIEJlYXJlciBoZWFkZXIgdG8gcGVyZm9ybSBjYWxscy4gKHNlZSBbU2VjdXJpdHkgYW5kIHJlc3RyaWN0aW9uIG9mIGRhdGEgYWNjZXNzXSgjc2VjdXJpdHktYW5kLXJlc3RyaWN0aW9uLW9mLWRhdGEtYWNjZXNzKSBmb3IgZGV0YWlscykuCgoKIyMgRmVhdHVyZXMKCkRhdGEgdHlwZXMgYXJlIHZhbGlkYXRlZCBhbmQgZW5mb3JjZWQgYnkgdGhlIEFQSS4KCkJlYXJlciB0b2tlbiBhdXRob3JpemF0aW9uIGlzIGVuZm9yY2VkIGZvciBhbGwgQVBJIGNhbGxzLgoKYEdFVGAgZW5kcG9pbnRzIGFyZSBhdmFpbGFibGUgZm9yIGFsbCB0YWJsZXMsIGVnLiBgL2ludm9pY2VgCgpgR0VUIGJ5IHV1aWRgIGVuZHBvaW50cyBhcmUgZ2VuZXJhdGVkIGZvciBhbGwgdGFibGVzIHdoaWNoIGhhdmUgYSBgdXVpZGAgY29sdW1uLiAKCmBHRVQgYnkgaWRgIGVuZHBvaW50cyBhcmUgZ2VuZXJhdGVkIGZvciBhbGwgdGFibGVzIHdoaWNoIGhhdmUgYSBgaWRgIGNvbHVtbiBidXQgbm8gYHV1aWRgIGNvbHVtbi4KCmBQT1NUYCAoY3JlYXRlKSBlbmRwb2ludHMgYXJlIGF2YWlsYWJsZSBmb3IgYWxsIHRhYmxlcy4KCmBQVVRgICh1cGRhdGUpIGVuZHBvaW50cyBhcmUgYXZhaWxhYmxlIGZvciB0YWJsZXMgd2l0aCBhIGB1dWlkYCBmaWVsZC4KCmBHZXQgY2hpbGQgY29sbGVjdGlvbnNgIEZvcmVpZ24ga2V5cyBhcmUgdXNlZCB0byBjcmVhdGUgQVBJIHBhdGhzIGluIHRoZSBmb3JtIGAvcGFyZW50L3twYXJlbnRfdXVpZH0vY2hpbGRgLgoKYEVudW1zYCBhcmUgY3JlYXRlZCBpbiBgYXBpL2VudW1zL2AgZm9yIGFsbCB0YWJsZXMgd2l0aCBuYW1lcyBlbmRpbmcgaW4gYF9zdGF0dXNgIG9yIGBfdHlwZWAuCgojIyMgUGFnaW5nIGFuZCBmaWx0ZXJpbmcKCkdFVCBlbmRwb2ludHMgd2hpY2ggdGFrZSBubyBwYXJhbWV0ZXJzIChzby1jYWxsZWQgJ2dldCBhbGwnIGVuZHBvaW50cykgZWcuIGAvaW52b2ljZWAgcHJvdmlkZSBvcHRpb25hbCBwYWdpbmcgYW5kIGZpbHRlcmluZyBxdWVyeXN0cmluZyBwYXJhbWV0ZXJzLCBhcyBmb2xsb3dzOgoKLSBwYWdlc2l6ZSAoaW50ZWdlciwgb3B0aW9uYWwpIDogVGhlIG51bWJlciBvZiByb3dzIHRvIGJlIHJldHVybmVkLiBEZWZhdWx0IGlzIDEwLiBNYXhpbXVtIGlzIDEwMC4KCi0gcGFnZSAoaW50ZWdlciwgb3B0aW9uYWwpIDogVGhlIHBhZ2UgdG8gYmUgcmV0dXJuZWQuIERlZmF1bHQgaXMgMC4gSWYgYSBub24temVybyBwYWdlIGlzIHNwZWNpZmllZCwgKHBhZ2VzaXplICogcGFnZSkgcm93cyBhcmUgc2tpcHBlZCB3aGVuIHBlcmZvcm1pbmcgdGhlIHF1ZXJ5LgoKLSBmaWx0ZXIgKHN0cmluZywgb3B0aW9uYWwpIDogVXNlZCBmb3Igc2VhcmNoaW5nIGZvciBzcGVjaWZpYyBkYXRhLiBJZiBzcGVjaWZpZWQsIG9ubHkgcm93cyB3aGljaCBjb250YWluIHRoZSBmaWx0ZXIgc3RyaW5nIGluIGFueSBvZiB0aGUgYHRleHRgIGNvbHVtbnMgYXJlIHJldHVybmVkLiBJZiBubyBkYXRhIGNhbiBiZSBmb3VuZCwgYW4gZW1wdHkgYXJyYXkgaXMgcmV0dXJuZWQuCgoKCiMjIEFzc3VtcHRpb25zL29waW5pb25zCgotIHRhYmxlcyBhcmUgYXNzdW1lZCB0byBoYXZlIGFuIGBpZGAgZmllbGQgb2YgdHlwZSBzZXJpYWwgKHByaW1hcnkga2V5KSwgCmFuZCBvcHRpb25hbGx5IGEgYHV1aWRgIGZpZWxkLCB3aGljaCBzaG91bGQgYmUgYE5PVCBOVUxMYC4KCi0gdGFibGUgbmFtZXMgZW5kaW5nIGluIGBfc3RhdHVzYCBvciBgX3R5cGVgIGFyZSBhc3N1bWVkIHRvIGhhdmUgYW4gYGlkYCBmaWVsZCBhbmQgYSBkZXNjcmlwdGlvbiBjb2x1bW4gd2hpY2ggYXJlIGl0ZXJhdGVkIGluIGFuIGF1dG9nZW5lcmF0ZWQgZmlsZSBhdCBgYXBpL2VudW1zL2luZGV4LmpzYCBzbyB0aGV5IGNhbiBiZSB1c2VkIGluIGNvZGUgdG8gcmVmZXIgdG8gc3RhdHVzIG9yIHR5cGUgdmFsdWVzLgoKLSBgUFVUYCAodXBkYXRlKSBBUEkgY2FsbHMgYXJlIG5vdCBhdmFpbGFibGUgZm9yIHRhYmxlcyB3aXRob3V0IGEgYHV1aWRgIGZpZWxkIChlZy4gc3RhdHVzIHRhYmxlcykuCgotIGlmIGEgYHdoaXRlbGlzdC5qc29uYCBmaWxlIGlzIGluY2x1ZGVkIGluIHRoZSBgL2FwcGAgZm9sZGVyLCBpdCB3aWxsIGNhdXNlIHRoZSBnZW5lcmF0b3IgdG8gaW5jbHVkZQpvbmx5IHRob3NlIHRhYmxlcyBsaXN0ZWQgaW5zaWRlLiBFeGFtcGxlIGZvcm1hdDoKCmBgYApbCiAgICB7CiAgICAgICAgInRhYmxlIjogImludm9pY2UiCiAgICB9LAogICAgewogICAgICAgICJ0YWJsZSI6ICJpbnZvaWNlX3N0YXR1cyIKICAgIH0sCiAgICB7CiAgICAgICAgInRhYmxlIjogInByb2R1Y3QiCiAgICB9Cl0KYGBgCgojIyMgU2VjdXJpdHkgYW5kIHJlc3RyaWN0aW9uIG9mIGRhdGEgYWNjZXNzCgpBbiBhc3N1bXB0aW9uIGlzIG1hZGUgdGhhdCBhIGB1c2VyYCB0YWJsZSBleGlzdHMgY29udGFpbmluZyBhIHJvdyBmb3IgZWFjaCB1c2VyLCBhbmQgdGhhdCBhIGBiZWFyZXJfdG9rZW5gIGNvbHVtbiBpcyBwcmVzZW50IG9uIHRoYXQgdGFibGUgd2hpY2ggaXMgdXNlZCB0byBpZGVudGlmeSB0aGUgQVBJIGNhbGxlciBieSBtYXRjaGluZyBpdCB3aXRoIHRoZSBwcm92aWRlZCBhdXRob3JpemF0aW9uIGhlYWRlci4gVGhpcyBtZWFucyB0aGF0IHRoZSBBUEkgY2FuIGFsd2F5cyBpZGVudGlmeSB0aGUgdXNlciBwZXJmb3JtaW5nIHRoZSBjYWxsLiBUaGUgYHVzZXJgIHRhYmxlIG5hbWUgY2FuIGJlIGNoYW5nZWQgaW4gdGhlIGAuZW52YCB2YXJpYWJsZSBgSk9JTl9VU0VSX1RBQkxFYC4KCldoZW4gcXVlcnlpbmcgYW55IGVudGl0eSwgdGhlIFNRTCBxdWVyeSB3aWxsIGFsd2F5cyBqb2luIHRvIHRoZSBgdXNlcmAgdGFibGUsIGVpdGhlciBkaXJlY3RseSBvciBpbmRpcmVjdGx5LCBpZiBpdCdzIHBvc3NpYmxlIHRvIGRvIHNvLiBUaGUgQVBJIGNvZGUgZ2VuZXJhdG9yIGRpc2NvdmVycyB3aGljaCB0YWJsZXMgYXJlIG5lZWRlZCB0byBqb2luIGluIG9yZGVyIHRvIHJlYWNoIHRoZSBgdXNlcmAgdGFibGUgdXNpbmcgYHJlY3Vyc2Vfam9pbl9jaGFpbigpYCBpbiBbZ2VuZXJhdG9yLmpzXShodHRwczovL2dpdGh1Yi5jb20vbmlja2ZhbGxvbi9jYXRhbHlzdC9ibG9iL21haW4vYXBwL2dlbmVyYXRvci5qcykuIFRoaXMgbWVhbnMgdGhhdCBmb3IgZGF0YWJhc2VzIHdoZXJlIG11bHRpcGxlIGRvbWFpbnMsIG9yZ2FuaXNhdGlvbnMsIGNvbXBhbmllcywgYWNjb3VudHMgb3Igb3RoZXIga2luZHMgb2Ygc2lsbyBleGlzdCwgcmVndWxhciB1c2VycyB3aWxsIG9ubHkgYmUgYWJsZSB0byByZXRyaWV2ZSB0aGUgZGF0YSBpbiB0aGVpciBvd24gc2lsby4KCgojIyMgQnVncy90by1kbwoKLSBgdXNlcmAgdGFibGUgKG9yIGAuZW52LkpPSU5fVVNFUl9UQUJMRWAgZXF1aXZhbGVudCkgY2FuIGJlIG9wZW5seSBxdWVyaWVkLiByZXN0cmljdCB0byBhZG1pbiBvbmx5IG9yIGFkZCBlbnZpcm9ubWVudCB2YXJpYWJsZS9saXN0IG9mIHJlc3RyaWN0ZWQgdGFibGVzLgoKLSBNYWtlIHN1cmUgeW91ciBmb3JlaWduIGtleXMgaGF2ZSB1bmlxdWUgbmFtZXMsIG90aGVyd2lzZSBlbmRwb2ludHMgYmV0d2VlbiB0YWJsZXMgdGhhdCBkb24ndCBoYXZlIHJlbGF0aW9uc2hpcHMgd2lsbCBiZSBnZW5lcmF0ZWQu readmeEtag: '"147cc7b326ce422b783ee484f258cbef0b570844"' readmeLastModified: Sun, 16 Jun 2024 00:46:26 GMT repositoryId: 675363467 description: >- Catalyst auto-generates an OpenAPI specification, a REST API that consumes it, and interactive Swagger docs from any provided Postgres database. created: '2023-08-06T17:05:01Z' updated: '2024-06-16T00:46:30Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: nickfallon logo: https://avatars.githubusercontent.com/u/15161343?v=4 repoEtag: '"223c076ecb89ac58f2da773e5db95a86b4dfdc7e514c8c63f8b66e695e397867"' repoLastModified: Sun, 16 Jun 2024 00:46:30 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/edmartt/django-task-backend v3: true id: f3e05424d2465ec5e28b82c267a8bf36 repositoryMetadata: base64Readme: >- IyBEamFuZ28gVGFzayBCYWNrZW5kCgpEamFuZ28gVGFzayBCYWNrZW5kIGlzIGEgc21hbGwgcHJvamVjdCBmb3Igc3RvcmluZyBkYWlseSB0YXNrcy4gSSB3YXMgbWFpbmx5IGludGVyZXN0ZWQgaW4gdGVzdGluZyB0aGUgcG93ZXIgb2YgRGphbmdvIHdpdGhvdXQgdXNpbmcgYW55IGV4dGVuc2lvbiB0byBtYWtlIGEgcmVzdGZ1bCBiYWNrZW5kIHNlcnZpY2UuCgojIyMgRmVhdHVyZXMKCi0gUmVnaXN0ZXIgZW5kcG9pbnQgZm9yIG5ldyB1c2VycwotIExvZ2luIGVuZHBvaW50IGZvciBleGlzdGFudCB1c2VycwotIEFmdGVyIGxvZ2dpbmcgaW4sIHRoZSByZXNwb25zZSBpcyBhIEpXVCB0byBiZSBhYmxlIHRvIGNyZWF0ZSBuZXcgdGFza3Mgb3IgY2hlY2sgZXhpc3Rpbmcgb25lcwotIEVhY2ggdXNlciBoYXMgdW5pcXVlIHRhc2tzIHRoYXQgY2FuIGJlIGFjY2Vzc2VkIGJ5IHRoZSBvd25lciBvZiB0aGUgdGFza3MKCiMjIFJlcXVpcmVtZW50cwoKLSBQeXRob24gMy4xMSsKLSBTUUxpdGUKLSBodHRwIGNsaWVudDogUE9TVE1BTiwgSW5zb21uaWEsIGNVUkwKCiMjIyBSdW5uaW5nIExvY2FsbHkKCmBgYApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL0VkbWFydHQvZGphbmdvLXRhc2stYmFja2VuZC5naXQKYGBgCgpvciBzc2ggaW5zdGVhZDoKCmBgYApnaXQgY2xvbmUgZ2l0QGdpdGh1Yi5jb206RWRtYXJ0dC9kamFuZ28tdGFzay1iYWNrZW5kLmdpdApgYGAKCmJyb3dzZSBpbnRvIHByb2plY3QgZGlyZWN0b3J5OgoKYGBgCmNkIGRqYW5nby10YXNrLWJhY2tlbmQvCmBgYAoKY3JlYXRlIHZpcnR1YWwgZW52aXJvbm1lbnQKCmBgYApweXRob24gLW0gdmVudiBlbnYKYGBgCmFjdGl2YXRlIHZpcnR1YWwgZW52aXJvbm1lbnQKCmBgYAouIGVudi9iaW4vYWN0aXZhdGUgCmBgYAppbnN0YWxsIGRlcGVuZGVuY2llcwoKYGBgCnBpcCBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQKYGBgCgpzZXQgZW52aXJvbm1lbnQgdmFyaWFibGVzIGZvbGxvd2luZyB0aGUgWy5lbnZyYy5leGFtcGxlXShodHRwczovL2dpdGh1Yi5jb20vRWRtYXJ0dC9kamFuZ28tdGFzay1iYWNrZW5kL2Jsb2IvbWFpbi8uZW52cmMuZXhhbXBsZSkgZmlsZSBhbmQgcnVuCgphcHBseSBtaWdyYXRpb25zCmBgYApweXRob24gbWFuYWdlLnB5IG1pZ3JhdGUKYGBgCgpydW4KCmBgYApweXRob24gbWFuYWdlLnB5IHJ1bnNlcnZlcgpgYGAKCgojIyMjIE5vdGUKCmFwaSBkb2N1bWVudGF0aW9uIGluOiBgL2FwaS92MS9zd2FnZ2VyYAo= readmeEtag: '"ca701ef7a1f0029300a934c7acd1e42bed8398d9"' readmeLastModified: Sat, 03 Aug 2024 19:31:41 GMT repositoryId: 738866392 description: Django Task Backend is a small project for storing daily tasks created: '2024-01-04T08:20:55Z' updated: '2024-08-03T19:31:45Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: Edmartt logo: https://avatars.githubusercontent.com/u/47486245?v=4 repoEtag: '"df3745e9f4e11de494372044da9114c114e1d644772ddbfe5422251046cba2be"' repoLastModified: Sat, 03 Aug 2024 19:31:45 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/open-banking/products v3: true repositoryMetadata: base64Readme: >- IyBwcm9kdWN0cwpPcGVuIEJhbmtpbmcgUHJvZHVjdHMgQVBJIGJ1aWx0IG9uIHRvcCBvZiBsaWdodC00ago= readmeEtag: '"c2fc536f7babe6a3ca208d30901d2c731eb423cc"' readmeLastModified: Thu, 25 Jul 2024 20:09:52 GMT repositoryId: 235865331 description: Open Banking Products API built on top of light-4j created: '2020-01-23T19:02:29Z' updated: '2026-02-03T01:08:49Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"7e43b04f691c6472a39893c01e6bc508f29626e9d9a39d4e04863ae5ca3f05c5"' repoLastModified: Tue, 03 Feb 2026 01:08:49 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 38262f9c9ca5153fa80b425418311ff0 - source: openapi3 tags repository: https://github.com/flum1025/sql-enum-generator v3: true id: ddb46cb2ffe07f0d6c8ad7c0b45fc64c repositoryMetadata: base64Readme: >- IyBzcWwtZW51bS1nZW5lcmF0b3IKCnNxbC1lbnVtLWdlbmVyYXRvciBpcyBhIHRvb2wgdGhhdCBjb252ZXJ0cyBTUUwgSU5TRVJUIHN0YXRlbWVudHMgZm9yIG1hc3RlciBkYXRhIGludG8gT3BlbkFQSSBzY2hlbWFzLiBUaGlzIGFwcGxpY2F0aW9uIGVuYWJsZXMgZGV2ZWxvcGVycyB0byBlYXNpbHkgZ2VuZXJhdGUgZW51bSByZXByZXNlbnRhdGlvbnMgb2YgZGF0YWJhc2UgbWFzdGVyIGRhdGEgdXNpbmcgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucywgc3RyZWFtbGluaW5nIHRoZSBkZXZlbG9wbWVudCBwcm9jZXNzIGFuZCBlbnN1cmluZyBjb25zaXN0ZW5jeSBiZXR3ZWVuIHRoZSBkYXRhYmFzZSBhbmQgYXBwbGljYXRpb24gY29kZS4KCkN1cnJlbnRseSBvbmx5IHBvc3RncmVzcWwgaXMgc3VwcG9ydGVkLgoKIyMgRmVhdHVyZXMKCi0gUGFyc2VzIFNRTCBJTlNFUlQgc3RhdGVtZW50cyBhbmQgZ2VuZXJhdGVzIGNvcnJlc3BvbmRpbmcgT3BlbkFQSSBzY2hlbWFzCi0gR2VuZXJhdGVkIE9wZW5BUEkgc2NoZW1hcyBjYW4gYmUgdXRpbGl6ZWQgd2l0aCBvdGhlciB0b29scyBmb3IgdHlwZSBnZW5lcmF0aW9uCgojIyBRdWljayBTdGFydAoKMS4gKipDcmVhdGUgYSBjb25maWd1cmF0aW9uIGZpbGUqKiBuYW1lZCBgc3FsZW51bWdlbi55bWxgIHdpdGggdGhlIGZvbGxvd2luZyBjb250ZW50OgoKYGBgeWFtbAp2ZXJzaW9uOiAiMSIKdGFibGVzOgogIC0gbmFtZTogcHJvZHVjdHMKICAgIGtleTogbmFtZQogICAgdmFsdWU6IGlkCmBgYAoKMi4gKipSdW4gdGhlIGFwcGxpY2F0aW9uKiogdXNpbmcgdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgc2gKJCBnbyBydW4gZ2l0aHViLmNvbS9mbHVtMTAyNS9zcWwtZW51bS1nZW5lcmF0b3IgZ2VuZXJhdGUgLS1zb3VyY2UtcGF0aCAuL2V4YW1wbGUvbWFzdGVyLnNxbCAtLW91dHB1dC1wYXRoIC4vZXhhbXBsZS9vcGVuYXBpLmdlbmVyYXRlZC5qc29uIC0tY29uZmlnIC4vZXhhbXBsZS9zcWxlbnVtZ2VuLnltbApgYGAKCjMuICoqVXRpbGl6ZSBsYW5ndWFnZS1zcGVjaWZpYyBnZW5lcmF0aW9uIHRvb2xzKiogdG8gY3JlYXRlIGVudW1zIGZyb20gdGhlIGdlbmVyYXRlZCBPcGVuQVBJIHNjaGVtYS4KCkZvciBhY3R1YWwgZ2VuZXJhdGlvbiBleGFtcGxlcywgcGxlYXNlIHJlZmVyIHRvIHRoZSBgZXhhbXBsZWAgZGlyZWN0b3J5IGluIHRoZSByZXBvc2l0b3J5LgoKIyMgTGFuZ3VhZ2UtU3BlY2lmaWMgVXNhZ2UgRXhhbXBsZXMKCiMjIyBHbwoKRm9yIEdvLCB5b3UgY2FuIHVzZSBbb2FwaS1jb2RlZ2VuXShodHRwczovL2dpdGh1Yi5jb20vb2FwaS1jb2RlZ2VuL29hcGktY29kZWdlbikgdG8gZ2VuZXJhdGUgY29kZSBmcm9tIHRoZSBPcGVuQVBJIHNjaGVtYS4gQ3JlYXRlIGEgY29uZmlndXJhdGlvbiBmaWxlIG5hbWVkIGBvYXBpLWNvZGVnZW4ueW1sYCB3aXRoIHRoZSBmb2xsb3dpbmcgY29udGVudDoKCmBgYHlhbWwKcGFja2FnZTogbWFpbgpvdXRwdXQ6IC4vb3BlbmFwaS5nZW5lcmF0ZWQuZ28KZ2VuZXJhdGU6CiAgbW9kZWxzOiB0cnVlCmNvbXBhdGliaWxpdHk6CiAgYWx3YXlzLXByZWZpeC1lbnVtLXZhbHVlczogdHJ1ZQpvdXRwdXQtb3B0aW9uczoKICBza2lwLXBydW5lOiB0cnVlCmBgYAoKVGhlbiwgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCB0byBnZW5lcmF0ZSB0aGUgR28gY29kZToKCmBgYHNoCiQgZ28gcnVuIGdpdGh1Yi5jb20vb2FwaS1jb2RlZ2VuL29hcGktY29kZWdlbi92Mi9jbWQvb2FwaS1jb2RlZ2VuIC1jb25maWcgLi9leGFtcGxlL29hcGktY29kZWdlbi55bWwgLi9leGFtcGxlL29wZW5hcGkuZ2VuZXJhdGVkLmpzb24KYGBgCgojIyMgVHlwZVNjcmlwdAoKRm9yIFR5cGVTY3JpcHQsIHlvdSBjYW4gdXNlIFtvcGVuYXBpLXR5cGVzY3JpcHRdKGh0dHBzOi8vZ2l0aHViLmNvbS9vcGVuYXBpLXRzL29wZW5hcGktdHlwZXNjcmlwdCkgdG8gZ2VuZXJhdGUgVHlwZVNjcmlwdCBkZWZpbml0aW9ucy4gUnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYHNoCiQgbnB4IG9wZW5hcGktdHlwZXNjcmlwdCAuL2V4YW1wbGUvb3BlbmFwaS5nZW5lcmF0ZWQuanNvbiAtbyAuL2V4YW1wbGUvb3BlbmFwaS5nZW5lcmF0ZWQuZC50cyAtLWVudW0KYGBgCgojIyBGdXR1cmUgUGxhbnMKCi0gWyBdIEFkZCBzdXBwb3J0IGZvciBhZGRpdGlvbmFsIFNRTCBkaWFsZWN0cwoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIHRoZSBgTElDRU5TRWAgZmlsZSBmb3IgZGV0YWlscy4K readmeEtag: '"69fc459a4b1f815edb785b5e9668b6261e210f0e"' readmeLastModified: Mon, 06 Jan 2025 16:44:25 GMT repositoryId: 909331079 description: >- sql-enum-generator is a tool that parses SQL INSERT statements from a file and automatically generates OpenAPI schema files. The generated OpenAPI schema can be used with existing tools to generate type definitions in various programming languages. created: '2024-12-28T11:35:54Z' updated: '2025-09-22T17:56:44Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: flum1025 logo: https://avatars.githubusercontent.com/u/9972700?v=4 license: MIT repoEtag: '"c88d0c403599ae6be0d0bdb03bb3c91a411e1445e60ad28fd458ba4988ceda60"' repoLastModified: Mon, 22 Sep 2025 17:56:44 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/avidee007/weather-app v3: true id: 8678290787f6dd7f0516e3a3e798a276 repositoryMetadata: base64Readme: >- IyBXZWF0aGVyIEFwcAoKIyMgSW50cm9kdWN0aW9uCgpUaGlzIGlzIGEgZnVsbHkgZnVuY3Rpb25hbCwgc2VjdXJlZCB3ZWF0aGVyIGFwcGxpY2F0aW9uIHRoYXQgdXNlcyBwdWJsaWNseQpleHBvc2VkIEFQSXMgdG8gZmV0Y2ggd2VhdGhlciBvZiB0aGUgdmFsaWQgVVNBIHBvc3RhbCBjb2RlLiBUaGlzIHdlYXRoZXIgc2VydmljZSB1c2VzCmEgc2VjdXJlIGFwaSBrZXkgZ2VuZXJhdGVkIGF0IFtXZWF0aGVyQVBJXShodHRwczovL3d3dy53ZWF0aGVyYXBpLmNvbS8pIHRvIGZldGNoIGN1cnJlbnQgd2VhdGhlciBkYXRhLgoKIyMjIEZ1bmN0aW9uYWwgRmVhdHVyZXM6CgoxLiBSZWdpc3RlciB0byBhcHBsaWNhdGlvbiB3aXRoIHVuaXF1ZSB1c2VybmFtZSBhbmQgc3Ryb25nIHBhc3N3b3JkLgoyLiBHZXQgYSB3ZWF0aGVyIHJlcG9ydCBvZiBhbnkgdmFsaWQgVVNBIHBvc3RhbC9aaXAgY29kZS4KMy4gR2V0IGhpc3Rvcnkgb2YgYWxsIHJlcXVlc3RlZCB3ZWF0aGVyIHJlcG9ydHMgZm9yIGFueSBnaXZlbiB1c2VyTmFtZS4KNC4gR2V0IGhpc3Rvcnkgb2YgYWxsIHJlcXVlc3RlZCB3ZWF0aGVyIHJlcG9ydHMgZm9yIGFueSBnaXZlbiBwb3N0YWwgY29kZS4KNS4gRGVhY3RpdmF0ZSBhbnkgZXhpc3RpbmcgdXNlci4KCiMjIyBOb24tRnVuY3Rpb25hbCBGZWF0dXJlczoKCjEuICoqU2VjdXJpdHk6KioKICAgICogU2lnbnVwIEFQSSBwcm9oaWJpdHMgZHVwbGljYXRlIHVzZXJuYW1lIGNyZWF0aW9uLgogICAgKiBTdHJvbmcgcGFzc3dvcmQgcG9saWN5IGlzIGVuZm9yY2VkLiBQYXNzd29yZCBtdXN0IGNvbnRhaW4gT25lIG51bWJlciwgT25lIHVwcGVyY2FzZSBsZXR0ZXJzLCBPbmUgbG93ZXJjYXNlCiAgICAgIGxldHRlcnMsIE9uZSBub24tYWxwaGEgbnVtZXJpYyBudW1iZXIgYW5kIG1pbmltdW0gOOKAkzE2IGNoYXJhY3RlcnMgaW4gbGVuZ3RoLgogICAgKiBBbGwgZW5kcG9pbnRzIGFyZSBzZWN1cmVkIHdpdGggdXNlcm5hbWUvcGFzc3dvcmQgYXV0aGVudGljYXRpb24gZXhjZXB0IFNpZ251cCBBUEkgaXQgaXMgZm9yIHVzZXIgcmVnaXN0cmF0aW9uLgogICAgKiBSb2xlIGJhc2VzIGF1dGhvcml6YXRpb24gaXMgaW1wbGVtZW50ZWQgZm9yIGFsbCBzZWN1cmVkIGVuZHBvaW50cy4KICAgICogRGVhY3RpdmF0ZSBBUEkgaXMgb25seSBhY2Nlc3NpYmxlIGZvciAqKkFETUlOKiogdXNlci4gQWRtaW4gdXNlciBpcyBhdXRvbWF0aWNhbGx5IGNyZWF0ZWQKICAgICAgb25jZSBhcHBsaWNhdGlvbiBydW5zIGZpcnN0IHRpbWUgdGhyb3VnaCBhIGxpcXVpYmFzZSBzY3JpcHQgdXNpbmcgZW5jcnlwdGVkIHZhbHVlcy4KICAgICogKipBRE1JTioqIGNyZWRlbnRpYWxzIHdpbGwgYmUgcHJvdmlkZWQgb24gcmVmZXJlbmNlLgogICAgKiBPdGhlciBBUElzIGFyZSBhc3Nlc3NhYmxlIGZvciBib3RoICoqVVNFUioqIGFuZCAqKkFETUlOKiogdXNlci4KCjIuICoqU2NhbGFiaWxpdHk6KioKICAgICogRG9ja2VyIGltYWdlcyBjYW4gYmUgY3JlYXRlZCB3aXRoIERvY2tlciBzdXBwb3J0LgogICAgKiBIb3Jpem9udGFsIHNjYWxpbmcgdXNpbmcgY29udGFpbmVyIG9yY2hlc3RyYXRpb24gdG9vbHMgbGlrZSBLdWJlcm5ldGVzLgogICAKMy4gKipIaWdoIEF2YWlsYWJpbGl0eToqKgogICAgKiBIaWdoIGF2YWlsYWJpbGl0eSB1c2luZyBjb250YWluZXIgb3JjaGVzdHJhdGlvbiB0b29scyBsaWtlIEt1YmVybmV0ZXMuCgojIyMgVGVjaCBTdGFjayBhbmQgRnJhbWV3b3JrczoKClRoaXMgbWljcm9zZXJ2aWNlIHdhcyBidWlsZCB1c2luZyBTcHJpbmcgQm9vdCB3aXRoIHRoZSBmb2xsb3dpbmcgZnJhbWV3b3JrczoKCiogKipKREsgMTc6KiogVXNlZCBhcyBKVk0gbGFuZ3VhZ2UuCiogKipTcHJpbmcgV2ViOioqIFVzZWQgZm9yIGJ1aWxkaW5nIFJFU1RGdWwgQVBJIGZvciB1c2VyIG1hbmFnZW1lbnQgYW5kIHdlYXRoZXIgb3BlcmF0aW9ucy4KKiAqKlNwcmluZyBEYXRhIEpQQToqKiBVc2VkIGZvciBkYXRhYmFzZSBhY2Nlc3MgYW5kIGRhdGFiYXNlIG9wZXJhdGlvbnMuCiogKipMaXF1aWJhc2U6KiogVXNlZCBmb3IgZGF0YWJhc2UgbWlncmF0aW9uLgoqICoqUG9zdGdyZXNTUUw6KiogQXMgUkRCTVMgdG8gc3RvcmUgdXNlciBhbmQgd2VhdGhlciBlbnRpdGllcy4KKiAqKlNwcmluZyBTZWN1cml0eToqKiBVc2VkIGZvciBzZWN1cmluZyBleHBvc2VkIFJFU1QgQVBJIGJ5IGltcGxlbWVudGluZyBhdXRoZW50aWNhdGlvbiBhbmQgYXV0aG9yaXphdGlvbi4KKiAqKlNwcmluZyBWYWxpZGF0aW9uOioqIFVzZWQgZm9yIGltcGxlbWVudCBiZWFuIHZhbGlkYXRpb24gZm9yIHVzZXIgaW5wdXRzLgoqICoqU3ByaW5nZG9jIE9wZW5BcGk6KiogVXNlZCB0byBpbXBsZW1lbnQgbGl2ZSBBUEkgZG9jdW1lbnRhdGlvbiB1c2luZyBTd2FnZ2VyLgoqICoqSlVuaXQgNToqKiBVc2VkIHRvIGltcGxlbWVudCB1bml0IHRlc3QgY2FzZXMuCiogKipNb2NraXRvOioqIFVzZWQgZm9yIG1vY2tpbmcgZXh0ZXJuYWwgZGVwZW5kZW5jaWVzIGluIHVuaXQgdGVzdCBjYXNlcy4KKiAqKlNwcmluZ0Jvb3RUZXN0OioqIFVzZWQgZm9yIGNvbXBsZXRlIGludGVncmF0aW9uIHRlc3RpbmcuCgojIyMgSG93IHRvIHJ1biB0aGUgcHJvamVjdAoKKipOT1RFOioqIFdlYXRoZXJBcGkgYXBpIGtleSBuZWVkcyB0byBiZSBjb25maWd1cmVkIHRvIGFwcGxpY2F0aW9uLnlhbWwgZmlsZSB0byBmZXRjaCBkYXRhIGZyb20gcHVibGljIHdlYXRoZXIgYXBpLgpDb21wbGV0ZSBzdGVwcyBhcmUgbWVudGlvbmVkIGluIFtIb3ctdG8tcnVuLWd1aWRlXShkb2NzL0hvdy10by1SdW4ubWQpLgpQbGVhc2UgZm9sbG93IHRoZSBpbnN0cnVjdGlvbnMgZG9jdW1lbnRlZCBpbiBbSG93LXRvLXJ1bi1ndWlkZV0oZG9jcy9Ib3ctdG8tUnVuLm1kKS4KCiMjIyBTd2FnZ2VyIGZvciBBUEkgZG9jdW1lbnRhdGlvbiBhbmQgdGVzdGluZwoKT25jZSBhcHBsaWNhdGlvbiBpcyBzdGFydGVkIHdpdGggYW55IG9mIHRoZSBjaG9zZW4gbWV0aG9kcyBmcm9tIHRoZSBhYm92ZSBzdGVwLgpbQ2xpY2sgaGVyZSB0byBvcGVuIFN3YWdnZXIgVUkuXShodHRwOi8vbG9jYWxob3N0OjgwODAvd2VhdGhlci1hcHAvc3dhZ2dlci11aS9pbmRleC5odG1sIy8pClRoaXMgd2lsbCBvcGVuIHN3YWdnZXIgVUkgd2hpY2ggY2FuIGJlIHVzZWQgdG8gdGVzdCB0aGUgQVBJUy4KCkRldGFpbGVkIHN0ZXBzIGZvciBhdXRoZW50aWNhdGlvbiBhbmQgdGVzdGluZyB0aGUgQVBJIGFyZSBtZW50aW9uZWQgaW4gW1J1bmJvb2tdKGRvY3MvU3RlcHMtdG8tdGVzdC1hcGkubWQpLgoKVGhlIEFwcGxpY2F0aW9uIGNvdmVycyBhbGwgbmVnYXRpdmUgYW5kIGVkZ2UgY2FzZXMgb2YgYXV0aGVudGljYXRpb24sIGF1dGhvcml6YXRpb24gYW5kIGFwaSBmdW5jdGlvbmFsaXRpZXMgYXMgcGVyIG15CnRlc3RpbmcuIEkgYW0gb3BlbiB0byBmdXJ0aGVyIGZlZWRiYWNrIGFuZCBpc3N1ZXMsIGlmIGFueS4= readmeEtag: '"9d8a46491c4ba80cfd053f62edb62363968027bf"' readmeLastModified: Wed, 18 Dec 2024 12:51:59 GMT repositoryId: 904673748 description: >- This is a fully functional, secured weather application that uses publicly exposed APIs to fetch weather of the valid USA postal code. created: '2024-12-17T10:41:14Z' updated: '2025-01-28T13:25:41Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: avidee007 logo: https://avatars.githubusercontent.com/u/31031315?v=4 repoEtag: '"11a69edf132ba28ae06d1f709f42dc00cc0ed18c58844f7cb2aff5df86171d02"' repoLastModified: Tue, 28 Jan 2025 13:25:41 GMT category: Server Implementations foundInMaster: true - source: openapi31 tags repository: https://github.com/juhaku/utoipa v3: true repositoryMetadata: base64Readme: >- # utoipa - Auto-generated OpenAPI documentation

[![Utoipa build](https://github.com/juhaku/utoipa/actions/workflows/build.yaml/badge.svg)](https://github.com/juhaku/utoipa/actions/workflows/build.yaml)
[![crates.io](https://img.shields.io/crates/v/utoipa.svg?label=crates.io&color=orange&logo=rust)](https://crates.io/crates/utoipa)
[![docs.rs](https://img.shields.io/static/v1?label=docs.rs&message=utoipa&color=blue&logo=data:image/svg+xml;base64,PHN2ZyByb2xlPSJpbWciIHhtbG5zPSJodHRwOi8vd3d3LnczLm9yZy8yMDAwL3N2ZyIgdmlld0JveD0iMCAwIDUxMiA1MTIiPjxwYXRoIGZpbGw9IiNmNWY1ZjUiIGQ9Ik00ODguNiAyNTAuMkwzOTIgMjE0VjEwNS41YzAtMTUtOS4zLTI4LjQtMjMuNC0zMy43bC0xMDAtMzcuNWMtOC4xLTMuMS0xNy4xLTMuMS0yNS4zIDBsLTEwMCAzNy41Yy0xNC4xIDUuMy0yMy40IDE4LjctMjMuNCAzMy43VjIxNGwtOTYuNiAzNi4yQzkuMyAyNTUuNSAwIDI2OC45IDAgMjgzLjlWMzk0YzAgMTMuNiA3LjcgMjYuMSAxOS45IDMyLjJsMTAwIDUwYzEwLjEgNS4xIDIyLjEgNS4xIDMyLjIgMGwxMDMuOS01MiAxMDMuOSA1MmMxMC4xIDUuMSAyMi4xIDUuMSAzMi4yIDBsMTAwLTUwYzEyLjItNi4xIDE5LjktMTguNiAxOS45LTMyLjJWMjgzLjljMC0xNS05LjMtMjguNC0yMy40LTMzLjd6TTM1OCAyMTQuOGwtODUgMzEuOXYtNjguMmw4NS0zN3Y3My4zek0xNTQgMTA0LjFsMTAyLTM4LjIgMTAyIDM4LjJ2LjZsLTEwMiA0MS40LTEwMi00MS40di0uNnptODQgMjkxLjFsLTg1IDQyLjV2LTc5LjFsODUtMzguOHY3NS40em0wLTExMmwtMTAyIDQxLjQtMTAyLTQxLjR2LS42bDEwMi0zOC4yIDEwMiAzOC4ydi42em0yNDAgMTEybC04NSA0Mi41di03OS4xbDg1LTM4Ljh2NzUuNHptMC0xMTJsLTEwMiA0MS40LTEwMi00MS40di0uNmwxMDItMzguMiAxMDIgMzguMnYuNnoiPjwvcGF0aD48L3N2Zz4K)](https://docs.rs/utoipa/latest/utoipa/)
![MSRV](https://img.shields.io/static/v1?label=MSRV&message=1.75&color=orange&logo=rust)

Pronounced **_/u:ˈtoʊ:i.pɑ/_** or **_/u:ˈtoʊˌaɪ.piˈeɪ/_** whatever works better for you.

Want to have your API documented with OpenAPI? But don't want to be bothered
with manual YAML or JSON tweaking? Would like it to be so easy that it would almost
be utopic? Don't worry: utoipa is here to fill this gap. It aims to do, if not all, then
most of the heavy lifting for you, enabling you to focus on writing the actual API logic instead of
documentation. It aims to be _minimal_, _simple_ and _fast_. It uses simple `proc` macros which
you can use to annotate your code to have items documented.

The `utoipa` crate provides auto-generated OpenAPI documentation for Rust REST APIs. It treats
code-first approach as a first class citizen and simplifies API documentation by providing
simple macros for generating the documentation from your code.

It also contains Rust types of the OpenAPI spec, allowing you to write the OpenAPI spec only using
Rust if auto generation is not your flavor or does not fit your purpose.

Long term goal of the library is to be the place to go when OpenAPI documentation is needed in any Rust
codebase.

Utoipa is framework-agnostic, and could be used together with any web framework, or even without one. While
being portable and standalone, one of its key aspects is simple integration with web frameworks.

## Choose your flavor and document your API with ice-cold IPA

|Flavor|Support|
|--|--|
|[actix-web](https://github.com/actix/actix-web)|Parse path, path parameters and query parameters, recognize request body and response body, [`utoipa-actix-web` bindings](./utoipa-actix-web/README.md). See more at [docs](https://docs.rs/utoipa/latest/utoipa/attr.path.html#actix_extras-feature-support-for-actix-web)|
|[axum](https://github.com/tokio-rs/axum)|Parse path and query parameters, recognize request body and response body, [`utoipa-axum` bindings](./utoipa-axum/README.md). See more at [docs](https://docs.rs/utoipa/latest/utoipa/attr.path.html#axum_extras-feature-support-for-axum)|
|[rocket](https://github.com/SergioBenitez/Rocket)| Parse path, path parameters and query parameters, recognize request body and response body. See more at [docs](https://docs.rs/utoipa/latest/utoipa/attr.path.html#rocket_extras-feature-support-for-rocket)|
|Others*| Plain `utoipa` without extra flavor. This gives you all the basic benefits listed below in **[Features](#features)** section but with little less automation.|

> Others* = For example [warp](https://github.com/seanmonstar/warp) but could be anything.

Refer to the existing [examples](./examples) to find out more.

## Features

* OpenAPI 3.1
* Pluggable, easy setup and integration with frameworks. 
* No bloat, enable what you need.
* Support for generic types
  * **Note!**<br>
    Tuples, arrays and slices cannot be used as generic arguments on types. Types implementing `ToSchema` manually should not have generic arguments, as
    they are not composeable and will result compile error.
* Automatic schema collection from usages recursively. 
  * Request body from either handler function arguments (if supported by framework) or from `request_body` attribute.
  * Response body from response `body` attribute or response `content` attribute.
* Various OpenAPI visualization tools supported out of the box.
* Rust type aliases via [`utoipa-config`](./utoipa-config/README.md).

## What's up with the word play?

The name comes from the words `utopic` and `api` where `uto` are the first three letters of _utopic_
and the `ipa` is _api_ reversed. Aaand... `ipa` is also an awesome type of beer :beer:.

## Crate Features

- **`macros`** Enable `utoipa-gen` macros. **This is enabled by default.**
- **`yaml`**: Enables **serde_norway** serialization of OpenAPI objects.
- **`actix_extras`**: Enhances [actix-web](https://github.com/actix/actix-web/) integration with being able to
  parse `path`, `path` and `query` parameters from actix web path attribute macros. See
  [docs](https://docs.rs/utoipa/latest/utoipa/attr.path.html#actix_extras-feature-support-for-actix-web) or [examples](./examples) for more details.
- **`rocket_extras`**: Enhances [rocket](https://github.com/SergioBenitez/Rocket) framework integration with being
  able to parse `path`, `path` and `query` parameters from rocket path attribute macros. See [docs](https://docs.rs/utoipa/latest/utoipa/attr.path.html#rocket_extras-feature-support-for-rocket)
  or [examples](./examples) for more details.
- **`axum_extras`**: Enhances [axum](https://github.com/tokio-rs/axum) framework integration allowing users to use `IntoParams` without
  defining the `parameter_in` attribute. See [docs](https://docs.rs/utoipa/latest/utoipa/attr.path.html#axum_extras-feature-support-for-axum)
  or [examples](./examples) for more details.
- **`debug`**: Add extra traits such as debug traits to openapi definitions and elsewhere.
- **`chrono`**: Add support for [chrono](https://crates.io/crates/chrono) `DateTime`, `Date`, `NaiveDate`, `NaiveDateTime`, `NaiveTime` and `Duration`
  types. By default these types are parsed to `string` types with additional `format` information.
  `format: date-time` for `DateTime` and `NaiveDateTime` and `format: date` for `Date` and `NaiveDate` according
  [RFC3339](https://www.rfc-editor.org/rfc/rfc3339#section-5.6) as `ISO-8601`. To
  override default `string` representation users have to use `value_type` attribute to override the type.
  See [docs](https://docs.rs/utoipa/latest/utoipa/derive.ToSchema.html) for more details.
- **`time`**: Add support for [time](https://crates.io/crates/time) `OffsetDateTime`, `PrimitiveDateTime`, `Date`, and `Duration` types.
  By default these types are parsed as `string`. `OffsetDateTime` and `PrimitiveDateTime` will use `date-time` format. `Date` will use
  `date` format and `Duration` will not have any format. To override default `string` representation users have to use `value_type` attribute
  to override the type. See [docs](https://docs.rs/utoipa/latest/utoipa/derive.ToSchema.html) for more details.
- **`jiff_0_2`** Add support for [jiff 0.2](https://crates.io/crates/jiff) `Zoned`, and `civil::Date` types.
  By default these types are parsed as `string`. `Zoned` will use `date-time` format. `civil::Date` will use
  `date` format. To override default `string` representation users have to use `value_type` attribute
  to override the type. See [docs](https://docs.rs/utoipa/latest/utoipa/derive.ToSchema.html) for more details.
- **`decimal`**: Add support for [rust_decimal](https://crates.io/crates/rust_decimal) `Decimal` type. **By default**
  it is interpreted as `String`. If you wish to change the format you need to override the type.
  See the `value_type` in [component derive docs](https://docs.rs/utoipa/latest/utoipa/derive.ToSchema.html).
- **`decimal_float`**: Add support for [rust_decimal](https://crates.io/crates/rust_decimal) `Decimal` type. **By default**
  it is interpreted as `Number`. This feature is mutually exclusive with **decimal** and allow to change the default type used in your
  documentation for `Decimal` much like `serde_with_float` feature exposed by rust_decimal.
- **`uuid`**: Add support for [uuid](https://github.com/uuid-rs/uuid). `Uuid` type will be presented as `String` with
  format `uuid` in OpenAPI spec.
- **`ulid`**: Add support for [ulid](https://github.com/dylanhart/ulid-rs). `Ulid` type will be presented as `String` with
  format `ulid` in OpenAPI spec.
- **`url`**: Add support for [url](https://github.com/servo/rust-url). `Url` type will be presented as `String` with
  format `uri` in OpenAPI spec.
- **`smallvec`**: Add support for [smallvec](https://crates.io/crates/smallvec). `SmallVec` will be treated as `Vec`.
- **`openapi_extensions`**: Adds traits and functions that provide extra convenience functions.
  See the [`request_body` docs](https://docs.rs/utoipa/latest/utoipa/openapi/request_body) for an example.
- **`repr`**: Add support for [repr_serde](https://github.com/dtolnay/serde-repr)'s `repr(u*)` and `repr(i*)` attributes to unit type enums for
  C-like enum representation. See [docs](https://docs.rs/utoipa/latest/utoipa/derive.ToSchema.html) for more details.
- **`preserve_order`**: Preserve order of properties when serializing the schema for a component.
  When enabled, the properties are listed in order of fields in the corresponding struct definition.
  When disabled, the properties are listed in alphabetical order.
- **`preserve_path_order`**: Preserve order of OpenAPI Paths according to order they have been
  introduced to the `#[openapi(paths(...))]` macro attribute. If disabled the paths will be
  ordered in alphabetical order. **However** the operations order under the path **will** be always constant according to [specification](https://spec.openapis.org/oas/latest.html#fixed-fields-6)
- **`indexmap`**: Add support for [indexmap](https://crates.io/crates/indexmap). When enabled `IndexMap` will be rendered as a map similar to
  `BTreeMap` and `HashMap`.
- **`non_strict_integers`**: Add support for non-standard integer formats `int8`, `int16`, `uint8`, `uint16`, `uint32`, and `uint64`.
- **`rc_schema`**: Add `ToSchema` support for `Arc<T>` and `Rc<T>` types. **Note!** serde `rc` feature flag must be enabled separately to allow
  serialization and deserialization of `Arc<T>` and `Rc<T>` types. See more about [serde feature flags](https://serde.rs/feature-flags.html).
- **`config`** Enables [`utoipa-config`](./utoipa-config/README.md) for the project which allows defining global configuration options for `utoipa`.

### Default Library Support

* Implicit partial support for `serde` attributes. See [docs](https://docs.rs/utoipa/latest/utoipa/derive.ToSchema.html#partial-serde-attributes-support) for more details.
* Support for [http](https://crates.io/crates/http) `StatusCode` in responses.

## Install

Add dependency declaration to `Cargo.toml`.

```toml
[dependencies]
utoipa = "5"
```

## Examples

_Create type with `ToSchema` and use it in `#[utoipa::path(...)]` that is registered to the `OpenApi`._

```rust
use utoipa::{OpenApi, ToSchema};

#[derive(ToSchema)]
struct Pet {
   id: u64,
   name: String,
   age: Option<i32>,
}

mod pet_api {
    /// Get pet by id
    ///
    /// Get pet from database by pet id
    #[utoipa::path(
        get,
        path = "/pets/{id}",
        responses(
            (status = 200, description = "Pet found successfully", body = Pet),
            (status = NOT_FOUND, description = "Pet was not found")
        ),
        params(
            ("id" = u64, Path, description = "Pet database id to get Pet for"),
        )
    )]
    async fn get_pet_by_id(pet_id: u64) -> Result<Pet, NotFound> {
        Ok(Pet {
            id: pet_id,
            age: None,
            name: "lightning".to_string(),
        })
    }
}

#[derive(OpenApi)]
#[openapi(paths(pet_api::get_pet_by_id))]
struct ApiDoc;

println!("{}", ApiDoc::openapi().to_pretty_json().unwrap());
```

<details>
    <summary><i><b>Above example will produce an OpenAPI doc like this:</b></i></summary>

```json
{
  "openapi": "3.1.0",
  "info": {
    "title": "application name from Cargo.toml",
    "description": "description from Cargo.toml",
    "contact": {
      "name": "author name from Cargo.toml",
      "email": "author email from Cargo.toml"
    },
    "license": {
      "name": "license from Cargo.toml"
    },
    "version": "version from Cargo.toml"
  },
  "paths": {
    "/pets/{id}": {
      "get": {
        "tags": [
          "pet_api"
        ],
        "summary": "Get pet by id",
        "description": "Get pet from database by pet id",
        "operationId": "get_pet_by_id",
        "parameters": [
          {
            "name": "id",
            "in": "path",
            "description": "Pet database id to get Pet for",
            "required": true,
            "schema": {
              "type": "integer",
              "format": "int64",
              "minimum": 0
            }
          }
        ],
        "responses": {
          "200": {
            "description": "Pet found successfully",
            "content": {
              "application/json": {
                "schema": {
                  "$ref": "#/components/schemas/Pet"
                }
              }
            }
          },
          "404": {
            "description": "Pet was not found"
          }
        }
      }
    }
  },
  "components": {
    "schemas": {
      "Pet": {
        "type": "object",
        "required": [
          "id",
          "name"
        ],
        "properties": {
          "age": {
            "type": [
              "integer",
              "null"
            ],
            "format": "int32"
          },
          "id": {
            "type": "integer",
            "format": "int64",
            "minimum": 0
          },
          "name": {
            "type": "string"
          }
        }
      }
    }
  }
}
```

</details>

## Modify OpenAPI at runtime

You can modify generated OpenAPI at runtime either via generated types directly or using
[Modify](https://docs.rs/utoipa/latest/utoipa/trait.Modify.html) trait.

_Modify generated OpenAPI via types directly._

```rust
#[derive(OpenApi)]
#[openapi(
    info(description = "My Api description"),
)]
struct ApiDoc;

let mut doc = ApiDoc::openapi();
doc.info.title = String::from("My Api");
```

_You can even convert the generated [OpenApi](https://docs.rs/utoipa/latest/utoipa/openapi/struct.OpenApi.html) to [OpenApiBuilder](https://docs.rs/utoipa/latest/utoipa/openapi/struct.OpenApiBuilder.html)._

```rust
let builder: OpenApiBuilder = ApiDoc::openapi().into();
```

See [Modify](https://docs.rs/utoipa/latest/utoipa/trait.Modify.html) trait for examples on how to modify generated OpenAPI via it.

## Go beyond the surface

- See how to serve OpenAPI doc via Swagger UI check [utoipa-swagger-ui](https://docs.rs/utoipa-swagger-ui/) crate for more details.
- Browse to [examples](https://github.com/juhaku/utoipa/tree/master/examples) for more comprehensive examples.
- Check [IntoResponses](https://docs.rs/utoipa/latest/utoipa/derive.IntoResponses.html) and [ToResponse](https://docs.rs/utoipa/latest/utoipa/derive.ToResponse.html) for examples on deriving responses.
- More about OpenAPI security in [security documentation](https://docs.rs/utoipa/latest/utoipa/openapi/security/index.html).
- Dump generated API doc to file at build time. See [issue 214 comment](https://github.com/juhaku/utoipa/issues/214#issuecomment-1179589373).

## FAQ

### Swagger UI returns 404 NotFound from built binary

This is highly probably due to `RustEmbed` not embedding the Swagger UI to the executable. This is natural since the `RustEmbed`
library **does not** by default embed files on debug builds. To get around this you can do one of the following.

1. Build your executable in `--release` mode
2. or add `debug-embed` feature flag to your `Cargo.toml` for `utoipa-swagger-ui`. This will enable the `debug-emebed` feature flag for
   `RustEmbed` as well. Read more about this [here](https://github.com/juhaku/utoipa/issues/527#issuecomment-1474219098) and [here](https://github.com/juhaku/utoipa/issues/268).

Find `utoipa-swagger-ui` [feature flags here](https://github.com/juhaku/utoipa/tree/master/utoipa-swagger-ui#crate-features).

### How to implement `ToSchema` for external type?

There are few ways around this that are elaborated [here in detail](https://github.com/juhaku/utoipa/issues/790#issuecomment-1787754185).

### Auto discover for OpenAPI schemas and paths?

Currently there is no build in solution to automatically discover the OpenAPI types but for your luck there is a pretty neat crate that 
just does this for you called [utoipauto](https://github.com/ProbablyClem/utoipauto).

## License

Licensed under either of [Apache 2.0](LICENSE-APACHE) or [MIT](LICENSE-MIT) license at your option.

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in this crate
by you, shall be dual licensed, without any additional terms or conditions.
 readmeEtag: '"a359df34e8ca2d1d90949e4396c85325bab315dc"' readmeLastModified: Wed, 23 Apr 2025 19:35:18 GMT repositoryId: 412240914 description: >- Simple, Fast, Code first and Compile time generated OpenAPI documentation for Rust created: '2021-09-30T22:00:31Z' updated: '2026-02-05T20:08:23Z' language: Rust archived: false stars: 3637 watchers: 11 forks: 327 owner: juhaku logo: https://avatars.githubusercontent.com/u/26358664?v=4 license: Apache-2.0 repoEtag: '"b47a44e177a9b5d8189fb58bf680c6f0b2ccb8a5c63396482f0f5a38889cfc60"' repoLastModified: Thu, 05 Feb 2026 20:08:23 GMT foundInMaster: true category: Parsers id: 14269a89115b3be71493769fca4d995e v3_1: true - source: openapi31 tags repository: https://github.com/a-chacon/oas_rails v3_1: true id: 4f2556a5a26e04f5a4be6c77a48b61ac repositoryMetadata: base64Readme: >- IVtHZW0gVmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9nZW0vdi9vYXNfcmFpbHM/Y29sb3I9RTk1NzNGKQohW0dpdEh1YiBMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL2EtY2hhY29uL29hc19yYWlscz9jb2xvcj1ibHVlKQohW0dpdEh1YiBBY3Rpb25zIFdvcmtmbG93IFN0YXR1c10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvYWN0aW9ucy93b3JrZmxvdy9zdGF0dXMvYS1jaGFjb24vb2FzX3JhaWxzLy5naXRodWIlMkZ3b3JrZmxvd3MlMkZydWJ5b25yYWlscy55bWwpCiFbR2VtIFRvdGFsIERvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9nZW0vZHQvb2FzX3JhaWxzKQohW1N0YXRpYyBCYWRnZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9SYWlscy0lM0UlM0Q3LjAuMC0lMjNFOTU3M0YpCiFbU3RhdGljIEJhZGdlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1J1YnktJTNFJTNEMy4xLjAtJTIzRTk1NzNGKQoKIyDwn5ODT3BlbiBBUEkgU3BlY2lmaWNhdGlvbiBGb3IgUmFpbHMKCk9hc1JhaWxzIGlzIGEgUmFpbHMgZW5naW5lIGZvciBnZW5lcmF0aW5nICoqYXV0b21hdGljIGludGVyYWN0aXZlIGRvY3VtZW50YXRpb24gZm9yIHlvdXIgUmFpbHMgQVBJcyoqLiBJdCBnZW5lcmF0ZXMgYW4gKipPQVMgMy4xKiogZG9jdW1lbnQgYW5kIGRpc3BsYXlzIGl0IHVzaW5nICoqW1JhcGlEb2NdKGh0dHBzOi8vcmFwaWRvY3dlYi5jb20pKiouCgpJdCByZWxpZXMgb24gdGhlIFtPYXNDb3JlXShodHRwczovL2dpdGh1Yi5jb20vYS1jaGFjb24vb2FzX2NvcmUpIGdlbS4KCiMjIyDwn5qAIERlbW8gQXBwCgpFeHBsb3JlIHRoZSBpbnRlcmFjdGl2ZSBkb2N1bWVudGF0aW9uIGxpdmU6Cgrwn5SXICoqW09wZW4gRGVtbyBBcHBdKGh0dHBzOi8vcGFzby5mbHkuZGV2L2FwaS9kb2NzKSoqICAK8J+RpCAqKlVzZXJuYW1lKio6IGBvYXNyYWlsc2AgIArwn5SRICoqUGFzc3dvcmQqKjogYG9hc3JhaWxzYAoK8J+OrCBBIERlbW8gSW5zdGFsbGF0aW9uL1VzYWdlIFZpZGVvOgo8aHR0cHM6Ly92aW1lby5jb20vMTAxMzY4NzMzMj4K8J+OrAoKIVtTY3JlZW5zaG90XShodHRwczovL2EtY2hhY29uLmdpdGh1Yi5pby9vYXNfY29yZS9hc3NldHMvcmFpbHNfdGhlbWUucG5nKQoKIyMgUmVsYXRlZCBQcm9qZWN0cwoKLSAqKltBcGlQaWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9BcGlwaWUvYXBpcGllLXJhaWxzKSoqOiBEb2Vzbid0IHN1cHBvcnQgT0FTIDMuMSwgcmVxdWlyZXMgbGVhcm5pbmcgYSBEU0wsIGxhY2tzIGEgbmljZSBVSQotICoqW3N3YWdnZXJfeWFyZC1yYWlsc10oaHR0cHM6Ly9naXRodWIuY29tL2xpdmluZ3NvY2lhbC9zd2FnZ2VyX3lhcmQtcmFpbHMpKio6IFNlZW1zIGFiYW5kb25lZCwgYnV0IHNlcnZlcyBhcyBpbnNwaXJhdGlvbgotICoqW1Jzd2FnXShodHRwczovL2dpdGh1Yi5jb20vcnN3YWcvcnN3YWcpKio6IE5vdCBhdXRvbWF0aWMsIGRlcGVuZHMgb24gUlNwZWM7IE1hbnkgZGV2ZWxvcGVycyBub3cgdXNlIE1pbml0ZXN0IGFzIGl0J3MgdGhlIGRlZmF1bHQgdGVzdCBmcmFtZXdvcmsKLSAqKltncmFwZS1zd2FnZ2VyXShodHRwczovL2dpdGh1Yi5jb20vcnVieS1ncmFwZS9ncmFwZS1zd2FnZ2VyKSoqOiBSZXF1aXJlcyBHcmFwZQotICoqW3JzcGVjX2FwaV9kb2N1bWVudGF0aW9uXShodHRwczovL2dpdGh1Yi5jb20vemlwbWFyay9yc3BlY19hcGlfZG9jdW1lbnRhdGlvbikqKjogUmVxdWlyZXMgUlNwZWMgYW5kIGEgY29tbWFuZCB0byBnZW5lcmF0ZSB0aGUgZG9jcwoKIyMgV2hhdCBTZXRzIE9hc1JhaWxzIEFwYXJ0PwoKLSAqKkR5bmFtaWMqKjogTm8gY29tbWFuZCByZXF1aXJlZCB0byBnZW5lcmF0ZSBkb2NzCi0gKipTaW1wbGUqKjogQ29tcGxlbWVudCBkZWZhdWx0IGRvY3VtZW50YXRpb24gd2l0aCBhIGZldyBjb21tZW50czsgbm8gbmVlZCB0byBsZWFybiBhIGNvbXBsZXggRFNMCi0gKipQdXJlIFJ1Ynkgb24gUmFpbHMgQVBJcyoqOiBObyBhZGRpdGlvbmFsIGZyYW1ld29ya3MgbmVlZGVkIChlLmcuLCBHcmFwZSwgUlNwZWMpCgojIyDwn5O977iPIE1vdGl2YXRpb24KCkFmdGVyIGV4cGVyaWVuY2luZyB0aGUgaW50ZXJhY3RpdmUgZG9jdW1lbnRhdGlvbiBpbiBQeXRob24ncyBmYXN0LWFwaSBmcmFtZXdvcmssIEkgc291Z2h0IHNpbWlsYXIgZnVuY3Rpb25hbGl0eSBpbiBSdWJ5IG9uIFJhaWxzLiBVbmFibGUgdG8gZmluZCBhIHN1aXRhYmxlIHNvbHV0aW9uLCBJIFthc2tlZCBvbiBTdGFjayBPdmVyZmxvd10oaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9xdWVzdGlvbnMvNzE5NDcwMTgvaXMtdGhlcmUtYS13YXktdG8tZ2VuZXJhdGUtYW4taW50ZXJhY3RpdmUtZG9jdW1lbnRhdGlvbi1mb3ItcmFpbHMtYXBpcykgeWVhcnMgYWdvLiBOb3csIHdpdGggc29tZSBmcmVlIHRpbWUgd2hpbGUgZnJlZWxhbmNpbmcgYXMgYW4gQVBJIGRldmVsb3BlciwgSSBkZWNpZGVkIHRvIGJ1aWxkIG15IG93biB0b29sLgoKKipOb3RlOiBUaGlzIGlzIG5vdCB5ZXQgYSBwcm9kdWN0aW9uLXJlYWR5IHNvbHV0aW9uLiBUaGUgY29kZSBtYXkgYmUgcm91Z2ggYW5kIGJlaGF2ZSB1bmV4cGVjdGVkbHksIGJ1dCBJIGFtIGFjdGl2ZWx5IHdvcmtpbmcgb24gaW1wcm92aW5nIGl0LiBJZiB5b3UgbGlrZSB0aGUgaWRlYSwgcGxlYXNlIGNvbnNpZGVyIGNvbnRyaWJ1dGluZyB0byBpdHMgZGV2ZWxvcG1lbnQuKioKClRoZSBnb2FsIGlzIHRvIG1pbmltaXplIHRoZSBlZmZvcnQgcmVxdWlyZWQgdG8gY3JlYXRlIGNvbXByZWhlbnNpdmUgZG9jdW1lbnRhdGlvbi4gQnkgZm9sbG93aW5nIFJFU1QgcHJpbmNpcGxlcyBpbiBSYWlscywgd2UgYmVsaWV2ZSB0aGlzIGlzIGFjaGlldmFibGUuIFlvdSBjYW4gZW5oYW5jZSB0aGUgZG9jdW1lbnRhdGlvbiB1c2luZyBbWWFyZF0oaHR0cHM6Ly95YXJkb2Mub3JnLykgdGFncy4KCiMjIERvY3VtZW50YXRpb24KCkZvciBzZWUgaG93IHRvIGluc3RhbGwsIGNvbmZpZ3VyZSBhbmQgdXNlIE9hc1JhaWxzIHBsZWFzZSByZWZlcmUgdG8gdGhlIFtPYXNDb3JlIE1EQm9va10oaHR0cHM6Ly9hLWNoYWNvbi5naXRodWIuaW8vb2FzX2NvcmUpCgojIyBDb250cmlidXRpbmcKCkNvbnRyaWJ1dGlvbnMgYXJlIHdoYXQgbWFrZSB0aGUgb3BlbiBzb3VyY2UgY29tbXVuaXR5IHN1Y2ggYW4gYW1hemluZyBwbGFjZSB0byBsZWFybiwgaW5zcGlyZSwgYW5kIGNyZWF0ZS4gQW55IGNvbnRyaWJ1dGlvbnMgeW91IG1ha2UgYXJlICoqZ3JlYXRseSBhcHByZWNpYXRlZCoqLiBJZiB5b3UgaGF2ZSBhIHN1Z2dlc3Rpb24gdGhhdCB3b3VsZCBtYWtlIHRoaXMgYmV0dGVyLCBwbGVhc2UgZm9yayB0aGUgcmVwbyBhbmQgY3JlYXRlIGEgcHVsbCByZXF1ZXN0LiBZb3UgY2FuIGFsc28gc2ltcGx5IG9wZW4gYW4gaXNzdWUgd2l0aCB0aGUgdGFnICJlbmhhbmNlbWVudCIuIERvbid0IGZvcmdldCB0byBnaXZlIHRoZSBwcm9qZWN0IGEgc3RhcuKtkCEgVGhhbmtzIGFnYWluIQoKSWYgeW91IHBsYW4gYSBiaWcgZmVhdHVyZSwgZmlyc3Qgb3BlbiBhbiBpc3N1ZSB0byBkaXNjdXNzIGl0IGJlZm9yZSBhbnkgZGV2ZWxvcG1lbnQuCgoxLiBGb3JrIHRoZSBQcm9qZWN0CjIuIENyZWF0ZSB5b3VyIEZlYXR1cmUgQnJhbmNoIChgZ2l0IGNoZWNrb3V0IC1iIGZlYXR1cmUvQW1hemluZ0ZlYXR1cmVgKQozLiBDb21taXQgeW91ciBDaGFuZ2VzIChgZ2l0IGNvbW1pdCAtbSAnQWRkIHNvbWUgQW1hemluZ0ZlYXR1cmUnYCkKNC4gUHVzaCB0byB0aGUgQnJhbmNoIChgZ2l0IHB1c2ggb3JpZ2luIGZlYXR1cmUvQW1hemluZ0ZlYXR1cmVgKQo1LiBPcGVuIGEgUHVsbCBSZXF1ZXN0CgojIyBMaWNlbnNlCgpPYXNSYWlscyBpcyByZWxlYXNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXShodHRwczovL29wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL01JVCkuCgojIyBTdGFyIEhpc3RvcnkKClshW1N0YXIgSGlzdG9yeSBDaGFydF0oaHR0cHM6Ly9hcGkuc3Rhci1oaXN0b3J5LmNvbS9zdmc/cmVwb3M9YS1jaGFjb24vb2FzX3JhaWxzJnR5cGU9RGF0ZSldKGh0dHBzOi8vd3d3LnN0YXItaGlzdG9yeS5jb20vI2EtY2hhY29uL29hc19yYWlscyZEYXRlKQo= readmeEtag: '"ce0cbc119580705c610b0d733188233b9b7d6e66"' readmeLastModified: Sat, 06 Sep 2025 19:17:15 GMT repositoryId: 831796203 description: Generate Automatic Interactive Documentation for Your Rails API created: '2024-07-21T16:44:36Z' updated: '2026-01-27T13:28:21Z' language: Ruby archived: false stars: 179 watchers: 4 forks: 22 owner: a-chacon logo: https://avatars.githubusercontent.com/u/39093711?v=4 license: MIT repoEtag: '"41bf9d43c7d0c5faa20c5a8be612cc427b9720adba4fee4f30b5d21a61da5c9d"' repoLastModified: Tue, 27 Jan 2026 13:28:21 GMT category: Parsers foundInMaster: true - source: openapi31 tags repository: https://github.com/bump-sh-examples/train-travel-api v3_1: true id: ae6b3dd61fddf7877ad70e617d1625ac repositoryMetadata: base64Readme: >- IyBUcmFpbiBUcmF2ZWwgQVBJCgpUaGlzIEFQSSBkb2VzIG5vdCBleGlzdCwgYnV0IGV4aXN0cyB0byBoZWxwIHBlb3BsZSBsZWFybiBPcGVuQVBJIGJ5IGRlc2NyaWJpbmcgYSByZWFsaXN0aWMgQVBJIGluIGEgcmVhbGlzdGljIHdheS4KCiMjIEJhY2tncm91bmQKClRoZSB3b3JsZCBvZiBPcGVuQVBJIGhhcyBiZWVuIHBsYWd1ZWQgYnkgIlRoZSBQZXQgU3RvcmUiLCBhbiBvbGQgQVBJIGRlc2NyaXB0aW9uIHVzZWQgYXMgYSBkZW1vIGluIGV2ZXJ5IHByb2plY3QgZXZlci4gSXQgZGVzY3JpYmVzIGEgUlBDLWVzcXVlIEFQSSB0aGF0J3MgbWFzY2VyYWRpbmcgYXMgUkVTVCwgYW5kIGRlc2NyaWJlcyBpdCBwb29ybHksIHVzaW5nIG9sZCBPcGVuQVBJIDIuMCB3aGljaCBoYXMgYmVlbiB1cGdyYWRlZCB0byBPcGVuQVBJIHYzLjAgd2l0aG91dCB0YWtpbmcgYWR2YW50YWdlIG9mIGFueSBvZiB0aGUgbmV3IGZlYXR1cmVzLgoKVGhpcyBleGFtcGxlIEFQSSBoYXMgYmVlbiB3cml0dGVuIGJ5IHNvbWVib2R5IHdobyB1c2VzIE9wZW5BUEkgZXZlcnkgZGF5LCBhbmQgaGFzIGRlc2NyaWJlZCAxMDBzIG9mIEFQSXMuIAoKIyMgR29hbHMKCkluc3RlYWQgb2Ygc2h5aW5nIGF3YXkgb2YgdG91Z2ggZGVjaXNpb25zLCB0aGlzIE9wZW5BUEkgZ2V0cyBzdHVjayBpbnRvIGFkdmFuY2VkIHRvcGljcyBsaWtlOgoKLSBIb3cgdG8gc3BsaXQgZmlsZXMgdXAgd2l0aCAkcmVmIHRvIGF2b2lkIHJlcGVhdGluZyBvdXJzZWx2ZXMuCi0gSG93IHRvIHJldXNlIHRoZSBzYW1lIG1vZGVsIGZvciByZWFkIGFuZCB3cml0ZSBvcGVyYXRpb25zLgotIEhvdyB0byB3cmFwIGRhdGEgd2l0aCBhIHdyYXBwZXIgbGlrZSBgeyBkYXRhOiBbXSwgbGlua3M6IHt9IH1gLgoKVGhpcyBtaWdodCBub3QgYmUgdGhlIGVhc2llc3QgQVBJIGRlc2NyaXB0aW9uIHRvIHJlYWQgaW4gYSB0ZXh0IGVkaXRvciwgYnV0IGl0IGxvb2tzIHdvbmRlcmZ1bCB3aGVuIHB1dCB0aHJvdWdoIFtCdW1wLnNoXShodHRwczovL2J1bXAuc2gvKS4KCioqW1ByZXZpZXcgdGhpcyBPcGVuQVBJIG9uIEJ1bXAuc2hdKGh0dHBzOi8vYnVtcC5zaC9idW1wLWV4YW1wbGVzL2RvYy90cmFpbi10cmF2ZWwtYXBpLykqKgoKT3IgW2Rvd25sb2FkIGl0IHlvdXJzZWxmXShodHRwczovL2dpdGh1Yi5jb20vYnVtcC1zaC1leGFtcGxlcy90cmFpbi10cmF2ZWwtYXBpL2FyY2hpdmUvcmVmcy9oZWFkcy9tYWluLnppcCkgYW5kIGhhdmUgYSBwb2tlIGFyb3VuZC4KCiMjIExpY2Vuc2UKClRoZSBjb250ZW50cyBvZiB0aGlzIHJlcG9zaXRvcnkgYXJlIGxpY2Vuc2VkIHVuZGVyIFtDQyBCWS1OQy1TQQogIDQuMF0oLi9MSUNFTlNFX0NDLUJZLU5DLVNBLTQuMCku readmeEtag: '"386afc96834d98a63d55cbaa4803ce63553bb174"' readmeLastModified: Thu, 15 Aug 2024 14:18:03 GMT repositoryId: 751352638 description: >- Sample OpenAPI description to use for whatever you like, as a hopefully more modern and useful alternative to the Petstore. created: '2024-02-01T12:49:32Z' updated: '2025-12-08T10:29:42Z' language: JavaScript archived: false stars: 68 watchers: 6 forks: 10 owner: bump-sh-examples logo: https://avatars.githubusercontent.com/u/157144805?v=4 license: NOASSERTION repoEtag: '"65b467cdffb962f5b0828c0bfe37874ddc7d0e66e1661c41afdc81075b36d28d"' repoLastModified: Mon, 08 Dec 2025 10:29:42 GMT category: Server Implementations foundInMaster: true - source: openapi31 tags repository: https://github.com/x52dev/oas3-rs v3_1: true id: 652ad18d642d9b05bb22ef2bacf1d425 repositoryMetadata: base64Readme: >- IyBgb2FzM2AKCjwhLS0gcHJldHRpZXItaWdub3JlLXN0YXJ0IC0tPgoKWyFbY3JhdGVzLmlvXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NyYXRlcy92L29hczM/bGFiZWw9bGF0ZXN0KV0oaHR0cHM6Ly9jcmF0ZXMuaW8vY3JhdGVzL29hczMpClshW0RvY3VtZW50YXRpb25dKGh0dHBzOi8vZG9jcy5ycy9vYXMzL2JhZGdlLnN2Zz92ZXJzaW9uPTAuMjAuMSldKGh0dHBzOi8vZG9jcy5ycy9vYXMzLzAuMjAuMSkKWyFbZGVwZW5kZW5jeSBzdGF0dXNdKGh0dHBzOi8vZGVwcy5ycy9jcmF0ZS9vYXMzLzAuMjAuMS9zdGF0dXMuc3ZnKV0oaHR0cHM6Ly9kZXBzLnJzL2NyYXRlL29hczMvMC4yMC4xKQohW01JVCBvciBBcGFjaGUgMi4wIGxpY2Vuc2VkXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NyYXRlcy9sL29hczMuc3ZnKQo8YnIgLz4KWyFbQ0ldKGh0dHBzOi8vZ2l0aHViLmNvbS94NTJkZXYvb2FzMy1ycy9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3g1MmRldi9vYXMzLXJzL2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbCkKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL3g1MmRldi9vYXMzLXJzL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj1PcFllNkk3ZGo1KV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL3g1MmRldi9vYXMzLXJzKQohW1ZlcnNpb25dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vY3JhdGVzL21zcnYvb2FzMy5zdmcpClshW0Rvd25sb2FkXShodHRwczovL2ltZy5zaGllbGRzLmlvL2NyYXRlcy9kL29hczMuc3ZnKV0oaHR0cHM6Ly9jcmF0ZXMuaW8vY3JhdGVzL29hczMpCgo8IS0tIHByZXR0aWVyLWlnbm9yZS1lbmQgLS0+Cgo8IS0tIGNhcmdvLXJkbWUgc3RhcnQgLS0+CgpTdHJ1Y3R1cmVzIGFuZCB0b29scyB0byBwYXJzZSwgbmF2aWdhdGUgYW5kIHZhbGlkYXRlIFtPcGVuQVBJIHYzLjEueF0gc3BlY2lmaWNhdGlvbnMuCgpOb3RlIHRoYXQgZHVlIHRvIHYzLjEueCBiZWluZyBhIGJyZWFraW5nIGNoYW5nZSBmcm9tIHYzLjAueCwgeW91IG1heSBoYXZlIHRyb3VibGUgY29ycmVjdGx5IHBhcnNpbmcKc3BlY3MgaW4gdGhlIG9sZGVyIGZvcm1hdC4KCiMjIEV4YW1wbGUKCmBgYHJ1c3QKbGV0IHlhbWwgPSBzdGQ6OmZzOjpyZWFkX3RvX3N0cmluZygicGF0aC90by9vcGVuYXBpLnltbCIpLnVud3JhcCgpOwoKbWF0Y2ggb2FzMzo6ZnJvbV95YW1sKHlhbWwpIHsKICBPayhzcGVjKSA9PiBwcmludGxuISgic3BlYzogezo/fSIsIHNwZWMpLAogIEVycihlcnIpID0+IHByaW50bG4hKCJlcnJvcjoge30iLCBlcnIpCn0KYGBgCgpbT3BlbkFQSSB2My4xLnhdOiBodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4xLjEKCjwhLS0gY2FyZ28tcmRtZSBlbmQgLS0+Cg== readmeEtag: '"7ae6ce8fc8ed0bb7b77bfd24c6e82d48aeaf8f3e"' readmeLastModified: Sun, 16 Nov 2025 22:33:18 GMT repositoryId: 303455500 description: >- Structures and tools to parse, navigate and validate OpenAPI v3.1 specifications. created: '2020-10-12T16:49:47Z' updated: '2026-02-01T02:36:25Z' language: Rust archived: false stars: 69 watchers: 2 forks: 25 owner: x52dev logo: https://avatars.githubusercontent.com/u/140988044?v=4 license: MIT repoEtag: '"b565bb25ddb6c2f64136fcec7b520ffc9d14250b87608fdd4a32e6ddd594057b"' repoLastModified: Sun, 01 Feb 2026 02:36:25 GMT category: Parsers foundInMaster: true - source: openapi31 tags repository: https://github.com/andrewwalsh/at-your-service v3: false id: 4306c9c5ce54843afcd1be9822783721 repositoryMetadata: base64Readme: >- <!-- Improved compatibility of back to top link: See: https://github.com/othneildrew/Best-README-Template/pull/73 -->

<a name="readme-top"></a>

[![MIT License][license-shield]][license-url]
<a href="https://www.npmjs.com/package/at-your-service"><img alt="npm" src="https://img.shields.io/npm/v/at-your-service?style=for-the-badge"></a>
<a href="https://atyourservice.awalsh.io/"><img alt="npm" src="https://img.shields.io/badge/View%20-Live%20Demo-422662?style=for-the-badge"></a>

<a href="https://github.com/AndrewWalsh/at-your-service/actions"><img alt="npm" src="https://github.com/AndrewWalsh/at-your-service/actions/workflows/node.js.yml/badge.svg"></a>
<a href="https://codeclimate.com/github/AndrewWalsh/at-your-service/test_coverage"><img src="https://api.codeclimate.com/v1/badges/56fa1f99da7509735cee/test_coverage" /></a>

<!-- PROJECT LOGO -->
<br />
<div align="center">
  <a href="https://github.com/AndrewWalsh/at-your-service">
    <img src="https://raw.githubusercontent.com/AndrewWalsh/at-your-service/main/resources/logo-floor.png" alt="Logo">
  </a>

  <br />
  <h2 align="center">at-your-service</h2>
  <br />

  <p align="center">
    <blockquote>
        A developer tool for API observability on the browser. Generate OpenAPI specifications and code from network traffic
        <br />
        <br />
        Designed for ease of use. No need to integrate with existing code
      </blockquote>
      <br />
      <br />
      <a href="https://atyourservice.awalsh.io/">View the Live Demo</a>
      <br />
      <br />
      <a href="https://github.com/AndrewWalsh/at-your-service/issues">Report Bug</a>
      ·
      <a href="https://www.npmjs.com/package/at-your-service">View on npm</a>
  </p>
</div>

<br />
<hr />
<br />

<!-- TABLE OF CONTENTS -->

<details open>
  <summary>Table of Contents</summary>
  <ol>
    <li>
      <a href="#about-the-project-and-why">About the Project and Why</a>
    </li>
    <li>
      <a href="#features">Features</a>
    </li>
    <li>
      <a href="#getting-started">Getting Started</a>
    </li>
    <li>
      <a href="#how-it-works">How It Works</a>
    </li>
    <li>
      <a href="#contributing">Contributing</a>
    </li>
    <li>
      <a href="#limitations">Limitations</a>
    </li>
    <li>
      <a href="#license">License</a>
    </li>
  </ol>
</details>

<br />

<!-- ABOUT THE PROJECT AND WHY -->

## About the Project and Why

<a href="https://atyourservice.awalsh.io/"><img alt="npm" src="https://img.shields.io/badge/View%20-Live%20Demo-422662"></a>


This tool is designed to help tackle problems that arise from a lack of awareness on API behaviour.

It records network requests under the hood using a service worker proxy. As the tool makes observations of network traffic over time, it learns the structure of the underlying API.

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- FEATURES -->

## Features

- **Spec gen**: autogenerate [OpenAPI 3.1](https://www.openapis.org/blog/2021/02/18/openapi-specification-3-1-released) specifications from network traffic
- **Code gen**: [convert](https://github.com/quicktype/quicktype) network response bodies into code for 10+ languages including TypeScript, Python, and JSON Schema
- **API Observability**: explore the network layer with enhanced tooling
- **Easy installation**: designed to plug in and go with an existing application

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- GETTING STARTED -->

## Getting Started

`at-your-service` features a CLI tool that places its service worker file into a directory. You likely wish to place this in `public` or `static`. See [more information here](https://mswjs.io/docs/getting-started/integrate/browser#where-is-my-public-directory) on common locations for static files.

The service worker must be served from the root of your site. Once this is installed run the start script in your application code.

1. Install the npm package
   ```sh
   npm install -D at-your-service@latest
   ```
2. Add service worker to your `public`, `static`, or otherwise root directory
   ```sh
   npx at-your-service@latest <directory>
   ```
3. Run the start script in your application

   ```ts
   import { startAtYourService } from "at-your-service";

   startAtYourService();
   ```

4. A button to open the drawer will be visible on your site
5. You can view copied OpenAPI 3.1 specifications in [editor-next.swagger.io](https://editor-next.swagger.io/). At the time of writing, you need to manually change the version from `3.1.0` to `3.0.0` after pasting the specification. Support for the new version of the specifcation is an ongoing process

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- HOW IT WORKS -->

## How It Works

More information on the rationale, functionality, and architecture of the tool [can be found here](https://awalsh.io/posts/developer-tool-api-discovery-observability-frontend/).

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- CONTRIBUTING -->

## Contributing

A development environment exists in `demo`, which when built is the landing page for the library.

If you have a suggestion that would make this better, please fork the repo and create a pull request.

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- LIMITATIONS -->

## Limitations

The library creates specifications that are only as accurate as the underlying observations. If your application relies on a response body that has not been observed, then type information for it will not be available.

Overall the intent is to produce a *best guess* that reveals API behaviour. This will never be a replacement for proper documentation.

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- LICENSE -->

## License

Distributed under the MIT License. See `LICENSE` for more information.

<p align="right">(<a href="#readme-top">back to top</a>)</p>

<!-- MARKDOWN LINKS & IMAGES -->
<!-- https://www.markdownguide.org/basic-syntax/#reference-style-links -->

[contributors-shield]: https://img.shields.io/github/contributors/AndrewWalsh/at-your-service.svg?style=for-the-badge
[contributors-url]: https://github.com/AndrewWalsh/at-your-service/graphs/contributors
[forks-shield]: https://img.shields.io/github/forks/AndrewWalsh/at-your-service.svg?style=for-the-badge
[forks-url]: https://github.com/AndrewWalsh/at-your-service/network/members
[stars-shield]: https://img.shields.io/github/stars/AndrewWalsh/at-your-service.svg?style=for-the-badge
[stars-url]: https://github.com/AndrewWalsh/at-your-service/stargazers
[issues-shield]: https://img.shields.io/github/issues/AndrewWalsh/at-your-service.svg?style=for-the-badge
[issues-url]: https://github.com/AndrewWalsh/at-your-service/issues
[license-shield]: https://img.shields.io/github/license/AndrewWalsh/at-your-service.svg?style=for-the-badge
[license-url]: https://github.com/AndrewWalsh/at-your-service/blob/master/LICENSE
 readmeEtag: '"e0f697045f0006be4901211e12c6cc19aec0a0fe"' readmeLastModified: Wed, 19 Mar 2025 04:24:39 GMT repositoryId: 564572432 description: >- Generate OpenAPI 3.1 specifications from network requests in browser environments created: '2022-11-11T02:02:37Z' updated: '2025-10-06T09:16:31Z' language: TypeScript archived: true stars: 17 watchers: 1 forks: 3 owner: AndrewWalsh logo: https://avatars.githubusercontent.com/u/15863952?v=4 license: MIT repoEtag: '"b534bc75c317d4b6310622a931a87a166280df7be56c29a0fc50165a49aa8d8e"' repoLastModified: Mon, 06 Oct 2025 09:16:31 GMT foundInMaster: true v3_1: true name: at-your-service link: https://atyourservice.awalsh.io/ language: TypeScript source_description: > A developer tool for API observability on the browser. Generate OpenAPI specifications and code from network traffic v2: false category: Server Implementations - source: openapi31 tags repository: https://github.com/up9inc/oas-diff v3_1: true repositoryMetadata: base64Readme: >- WyFbYWNjZXB0YW5jZSB0ZXN0c10oaHR0cHM6Ly9naXRodWIuY29tL3VwOWluYy9vYXMtZGlmZi9hY3Rpb25zL3dvcmtmbG93cy9hY2NlcHRhbmNlX3Rlc3RzLnltbC9iYWRnZS5zdmc/YnJhbmNoPWRldmVsb3ApXShodHRwczovL2dpdGh1Yi5jb20vdXA5aW5jL29hcy1kaWZmL2FjdGlvbnMvd29ya2Zsb3dzL2FjY2VwdGFuY2VfdGVzdHMueW1sKQojIE9BUy1ESUZGIApPQVMgMy4xIFZhbGlkYXRpb24gYW5kIERpZmYgVG9vbAoKIyMgRGVwZW5kZW5jaWVzCi0gR2l0Ci0gTWFrZQotIEdvIDEuMTgrCgojIyBCdWlsZAotIEJ1aWxkCiAgICBgYGBgCiAgICBtYWtlIGJ1aWxkCiAgICBgYGBgCi0gUnVuCiAgICBgYGBgCiAgICAuL2J1aWxkL29hc2RpZmYgLS12ZXJzaW9uCiAgICBgYGBgCiMjIE9wdGlvbnMKLSBWYWxpZGF0ZQogICAgYGBgYAogICAgLS1iYXNlLWZpbGUgdmFsdWUsIC0tZjEgdmFsdWUgICAgcGF0aCBvZiB0aGUgYmFzZSBPQVMgMy4xIGZpbGUKICAgIC0taGVscCwgLWggICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHNob3cgaGVscCAoZGVmYXVsdDogZmFsc2UpCiAgIGBgYGAKLSBEaWZmCiAgICBgYGBgCiAgICAtLWJhc2UtZmlsZSB2YWx1ZSwgLS1mMSB2YWx1ZSAgICBwYXRoIG9mIHRoZSBiYXNlIE9BUyAzLjEgZmlsZQogICAgLS1zZWNvbmQtZmlsZSB2YWx1ZSwgLS1mMiB2YWx1ZSAgcGF0aCBvZiB0aGUgc2Vjb25kIE9BUyAzLjEgZmlsZQogICAgLS10eXBlLWZpbHRlciB2YWx1ZSwgLS10ZiB2YWx1ZSAgY2hhbmdlbG9nIFR5cGUgZmlsdGVyIChjcmVhdGUvdXBkYXRlL2RlbGV0ZSkKICAgIC0tb3V0cHV0LWh0bWwsIC0tb2ggICAgICAgICAgICAgIHNhdmUgYW4gaHRtbCByZXBvcnQgKGRlZmF1bHQ6IGZhbHNlKQogICAgLS1vdXRwdXQtZW5kcG9pbnQsIC0tb2UgICAgICAgICAgZW5kcG9pbnQgYmFzZWQgY2hhbmdlbG9nIG91dHB1dCAoZGVmYXVsdDogZmFsc2UpCiAgICAtLWxvb3NlLCAtbCAgICAgICAgICAgICAgICAgICAgICBsb29zZWx5IGRpZmYsIGlnbm9yZXMgZ2xvYmFsIGNhc2Ugc2Vuc2l0aXZpdHkgZm9yIHN0cmluZ3MgY29tcGFyaXNvbnMgYW5kIGlnbm9yZSBoZWFkZXJzIHRoYXQgc3RhcnQgd2l0aCAneC0nIGFuZCAndXNlci1hZ2VudCcgKGRlZmF1bHQ6IGZhbHNlKQogICAgLS1pbmNsdWRlLWZpbGUtcGF0aCwgLS1pZnAgICAgICAgd2hldGhlciBvciBub3QgdG8gaW5jbHVkZSB0aGUgZnVsbCBmaWxlIHBhdGggZnJvbSB0aGUgZGlmZiBjaGFuZ2Vsb2cgKGRlZmF1bHQ6IGZhbHNlKQogICAgLS1pZ25vcmUtZGVzY3JpcHRpb25zLCAtLWlkICAgICAgd2hldGhlciBvciBub3QgdG8gaWdub3JlIGRlc2NyaXB0aW9ucyB3aGVuIHBlcmZvcm1pbmcgdGhlIGRpZmYgKGRlZmF1bHQ6IGZhbHNlKQogICAgLS1pZ25vcmUtZXhhbXBsZXMsIC0taWUgICAgICAgICAgd2hldGhlciBvciBub3QgdG8gaWdub3JlIGV4YW1wbGVzIHdoZW4gcGVyZm9ybWluZyB0aGUgZGlmZiAoZGVmYXVsdDogZmFsc2UpCiAgICAtLWhlbHAsIC1oICAgICAgICAgICAgICAgICAgICAgICBzaG93IGhlbHAgKGRlZmF1bHQ6IGZhbHNlKQogICAgYGBgYAojIyBFeGFtcGxlcwotIFZlcnNpb24KICAgIGBgYGAKICAgIC4vYnVpbGQvb2FzZGlmZiAtdgogICAgYGBgYAotIEF2YWlsYWJsZSBDb21tYW5kcwogICAgYGBgYAogICAgLi9idWlsZC9vYXNkaWZmCiAgICBgYGBgCi0gVmFsaWRhdGUKICAgIGBgYGAKICAgIC4vYnVpbGQvb2FzZGlmZiB2YWxpZGF0ZSAtLWJhc2UtZmlsZSBleGFtcGxlcy9pbnZhbGlkLmpzb24KICAgIC4vYnVpbGQvb2FzZGlmZiB2YWxpZGF0ZSAtLWYxIGV4YW1wbGVzL2ludmFsaWQuanNvbgogICAgYGBgYAotIERpZmYKICAgIGBgYGAKICAgIC4vYnVpbGQvb2FzZGlmZiBkaWZmIC0tYmFzZS1maWxlIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tc2Vjb25kLWZpbGUgZXhhbXBsZXMvc2ltcGxlMi5qc29uCiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWJhc2UtZmlsZSBleGFtcGxlcy9zaW1wbGUuanNvbiAtLXNlY29uZC1maWxlIGV4YW1wbGVzL3NpbXBsZTIuanNvbiAtLWxvb3NlCiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWYxIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tZjIgZXhhbXBsZXMvc2ltcGxlMi5qc29uIC0tdHlwZS1maWx0ZXIgY3JlYXRlCiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWYxIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tZjIgZXhhbXBsZXMvc2ltcGxlMi5qc29uIC0tdHlwZS1maWx0ZXIgdXBkYXRlCiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWYxIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tZjIgZXhhbXBsZXMvc2ltcGxlMi5qc29uIC0tdHlwZS1maWx0ZXIgZGVsZXRlCiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWYxIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tZjIgZXhhbXBsZXMvc2ltcGxlMi5qc29uIC0taWdub3JlLWRlc2NyaXB0aW9ucwogICAgLi9idWlsZC9vYXNkaWZmIGRpZmYgLS1mMSBleGFtcGxlcy9zaW1wbGUuanNvbiAtLWYyIGV4YW1wbGVzL3NpbXBsZTIuanNvbiAtLWlnbm9yZS1leGFtcGxlcwogICAgLi9idWlsZC9vYXNkaWZmIGRpZmYgLS1mMSBleGFtcGxlcy9zaW1wbGUuanNvbiAtLWYyIGV4YW1wbGVzL3NpbXBsZTIuanNvbiAtLWxvb3NlIC0taWdub3JlLWRlc2NyaXB0aW9ucyAtLWlnbm9yZS1leGFtcGxlcwogICAgLi9idWlsZC9vYXNkaWZmIGRpZmYgLS1mMSBleGFtcGxlcy9zaW1wbGUuanNvbiAtLWYyIGV4YW1wbGVzL3NpbXBsZTIuanNvbiAtLW91dHB1dC1odG1sCiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWYxIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tZjIgZXhhbXBsZXMvc2ltcGxlMi5qc29uIC0tb3V0cHV0LWVuZHBvaW50CiAgICAuL2J1aWxkL29hc2RpZmYgZGlmZiAtLWYxIGV4YW1wbGVzL3NpbXBsZS5qc29uIC0tZjIgZXhhbXBsZXMvc2ltcGxlMi5qc29uIC0tb3V0cHV0LWVuZHBvaW50IC0tb3V0cHV0LWh0bWwgLS10eXBlLWZpbHRlciB1cGRhdGUKICAgIGBgYGAKCiMjIEFycmF5IElkZW50aWZpZXJzCiBBcnJheSBpZGVudGlmaWVycyBhcmUgdXNlZCBvbmx5IGZvciBhcnJheXMgdG8gY29tcGFyZSBhcnJheXMgYnkgYSBtYXRjaGluZyBpZGVudGlmaWVyIGFuZCBub3QgYmFzZWQgb24gb3JkZXIuIElmIGFuIGlkZW50aWZpYWJsZSBlbGVtZW50IGlzIGZvdW5kIGluIGJvdGggdGhlIGZyb20gYW5kIHRvIHN0cnVjdHVyZXMsIHRoZXkgd2lsbCBiZSBkaXJlY3RseSBjb21wYXJlZAoKLSBTZXJ2ZXJzCiAgICBgYGBgCiAgICBVUkwgICAgICAgICBzdHJpbmcgICAgICAgICAgICAgYGpzb246InVybCxvbWl0ZW1wdHkiIGRpZmY6InVybCxpZGVudGlmaWVyImAKICAgIGBgYGAKLSBUYWdzCiAgICBgYGBgCiAgICBOYW1lICAgICAgICAgc3RyaW5nICAgICAgICBganNvbjoibmFtZSxvbWl0ZW1wdHkiIGRpZmY6Im5hbWUsaWRlbnRpZmllciJgCiAgICBgYGBgCi0gUGFyYW1ldGVycwogICAgYGBgYAogICAgTmFtZSAgICAgICAgIHN0cmluZyAgICAgICAgYGpzb246Im5hbWUsb21pdGVtcHR5IiBkaWZmOiJuYW1lLGlkZW50aWZpZXIiYAogICAgYGBgYAoKIyMgQ2hhbmdlbG9nIFJ1bGVzCi0gQXJyYXlzCiAgICBgYGBgCiAgICBDUkVBVEUvREVMRVRFIC0+IEFsd2F5cyB0aGUgZW50aXJlIGVsZW1lbnQKICAgIFVQREFURSAtPiBPbmx5IHRoZSBwcm9wZXJ0eSwgZXhjZXB0aW9uIGlmIHRoZSBwcm9wZXJ0eSBpcyB0aGUgaWRlbnRpZmllcgogICAgYGBgYAoKIyMgTGltaXRhdGlvbnMKLSBgb3BlbmFwaWAgZmllbGQgaXMgbm90IGEgcGFydCBvZiB0aGUgY2hhbmdlbG9nLCB3ZSBvbmx5IHN1cHBvcnQgT0FTIHZlcnNpb24gMy4xLCBzbyBhbnkgY2hhbmdlcyB3aWxsIGNhdXNlIGEgdmFsaWRhdGlvbiBmYWlsdXJlCi0gYGpzb25TY2hlbWFEaWFsZWN0YCBmaWVsZCBpcyBub3QgYSBwYXJ0IG9mIHRoZSBjaGFuZ2Vsb2csIHdlIG9ubHkgc3VwcG9ydCBPQVMgdmVyc2lvbiAzLjEgYW5kIGl0IHVzZXMgYEpTT04gU2NoZW1hIFZhbGlkYXRpb24gRHJhZnQgMjAyMC0xMmAKLSBgU3BlY2lmaWNhdGlvbiBFeHRlbnNpb25zYCBhcmUgbm90IHN1cHBvcnRlZCAoVE9ETykKLSBgcGFyYW1ldGVyc2AgYXJyYXkgb2YgYFJlZmVyZW5jZSBPYmplY3RgIGFyZSBub3Qgc3VwcG9ydGVkIChUT0RPKQ== readmeEtag: '"bbde2a9d823fdc2d4e23883c3402ef9d76eb4542"' readmeLastModified: Sun, 12 Jun 2022 12:25:54 GMT repositoryId: 457402945 description: OAS 3.1 Validation and Diff CLI Tool created: '2022-02-09T14:54:00Z' updated: '2024-10-16T17:36:40Z' language: Go archived: false stars: 10 watchers: 3 forks: 0 owner: up9inc logo: https://avatars.githubusercontent.com/u/51116998?v=4 repoEtag: '"75be76dcf31a261f1accd5aa3ae1c227fc0970eff7c60300bef5caff3c70eb01"' repoLastModified: Wed, 16 Oct 2024 17:36:40 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: d626b1c63ecbfe1be8f891819fb740bf - source: openapi31 tags repository: https://github.com/ghostrider-05/patreon-api-spec v3_1: true id: ec6b22685d3f6b8b1bc770ed5c4f8eaf repositoryMetadata: base64Readme: >- IyBQYXRyZW9uIE9wZW5BUEkKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBhIHByZXZpZXcgb2YgdGhlIFtPcGVuQVBJIDMuMSBzcGVjaWZpY2F0aW9uXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21haW4vdmVyc2lvbnMvMy4xLjAubWQpIGZvciBbUGF0cmVvbnMncyBBUEldKGh0dHBzOi8vZG9jcy5wYXRyZW9uLmNvbS8pLiBDdXJyZW50bHksIHRoZSBzcGVjIGlzIG9ubHkgYXZhaWxhYmxlIGZvciB0aGUgQVBJIHZlcnNpb24gW3YyXShodHRwczovL2RvY3MucGF0cmVvbi5jb20vI2FwaXYyLW9hdXRoKS4KCj4gWyFXQVJOSU5HXQo+IFRoaXMgcmVwb3NpdG9yeSBpcyBub3QgY3JlYXRlZCwgbWFpbnRhaW5lZCBvciBhc3NvY2lhdGVkIGluIGFueSB3YXkgd2l0aCBQYXRyZW9uLiBCb3RoIHNwZWNpZmljYXRpb25zIGFyZSBtYWRlIGZyb20gdGhlIHB1YmxpYyBkb2N1bWVudGVkIEFQSSBhbmQgYXJlIHN1YmplY3QgdG8gYnJlYWtpbmcgY2hhbmdlcyB3aXRob3V0IG5vdGljZSBpZiBQYXRyZW9uIHVwZGF0ZXMgdGhlIHB1YmxpYyBBUEkuCgojIyBVc2FnZQoKIyMjIFNwZWMgRmlsZXMKClR3byB2ZXJzaW9ucyBvZiB0aGUgc3BlYyBhcmUgaW5jbHVkZWTigJR0aGUgc3RhbmRhcmQgc3BlYyBhbmQgdGhlIHByZXZpZXcgc3BlYzoKCi0gW2BvcGVuYXBpLmpzb25gXShzcGVjcy9vcGVuYXBpLmpzb24pIGlzIHRoZSBzdGFuZGFyZCBzcGVjIHRoYXQgY29udGFpbnMgdGhlIGRvY3VtZW50ZWQsIHB1YmxpYyBBUEkuCi0gW2BvcGVuYXBpX3ByZXZpZXcuanNvbmBdKHNwZWNzL29wZW5hcGlfcHJldmlldy5qc29uKSBpcyB0aGUgcHJldmlldyBzcGVjIHdoaWNoIGNvbnRhaW5zIHVuc3RhYmxlIGFuZC9vciB1bmRvY3VtZW50ZWQgQVBJIGZlYXR1cmVzLiAqKlRoaXMgc2hvdWxkIG5vdCBiZSBjb25zaWRlcmVkIHN0YWJsZSoqIG9yIHVzZWQgaW4gcHJvZHVjdGlvbiBlbnZpcm9ubWVudHMuCgojIyMgRG9jdW1lbnRhdGlvbgoKVG8gdmlldyBib3RoIHRoZSBzdGFibGUgYW5kIGV4cGVyaW1lbnRhbCBzcGVjaWZpY2F0aW9ucywgc2VlIHRoZSBbQVBJIHJlZmVyZW5jZV0oaHR0cHM6Ly9wYXRyZW9uLmFwaWRvY3VtZW50YXRpb24uY29tL3YyLXN0YWJsZS9yZWZlcmVuY2UpLgoKVG8gdXNlIHRoZSBQYXRyZW9uIEFQSSB3aXRoIGEgbGlicmFyeSwgZ28gdG8gdGhlIFtkb2N1bWVudGF0aW9uIG9mIGBwYXRyZW9uLWFwaS50c2BdKGh0dHBzOi8vcGF0cmVvbi1hcGkucGFnZXMuZGV2LykuCgo8IS0tIFRPRE86IC0tPgo8IS0tICMjIyBJbnRlZ3JhdGluZyB3aXRoIFBvc3RtYW4gLS0+CgojIyBDb250cmlidXRpbmcKCk9wZW5BUEkgc3BlYyBjb250ZW50cyBhcmUgYXV0b21hdGljYWxseSBnZW5lcmF0ZWQgYnkgdGhlIFtgcGF0cmVvbi1hcGkudHNgXShodHRwczovL2dpdGh1Yi5jb20vZ2hvc3RyaWRlci0wNS9wYXRyZW9uLWFwaS50cykgbGlicmFyeS4KCkZvciBidWcgZml4ZXMgb3IgaW1wcm92ZW1lbnRzIGZvciB0aGUgT3BlbkFQSSBzcGVjLCB5b3UgY2FuIFtvcGVuIGFuIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vZ2hvc3RyaWRlci0wNS9wYXRyZW9uLWFwaS1zcGVjL2lzc3VlcykuCgojIyBLbm93biBpc3N1ZXMKClRPRE86IGFkZCBrbm93biBpc3N1ZXMKCiMjIExpY2Vuc2UKCltNSVRdKGh0dHBzOi8vZ2l0aHViLmNvbS9naG9zdHJpZGVyLTA1L3BhdHJlb24tYXBpLnRzL2Jsb2IvbWFpbi9MSUNFTlNFKQo= readmeEtag: '"4452d74f62c1ff92a9d57f0e571c5c73e770f252"' readmeLastModified: Mon, 09 Dec 2024 14:29:47 GMT repositoryId: 879807162 description: OpenAPI specification for Patreon API created: '2024-10-28T15:24:59Z' updated: '2026-01-14T19:45:32Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: ghostrider-05 logo: https://avatars.githubusercontent.com/u/52837878?v=4 repoEtag: '"cf4082765c2e0af917fa57efa092bbffeb7f2f052c9868ba60e9368ff4a531be"' repoLastModified: Wed, 14 Jan 2026 19:45:32 GMT category: Parsers foundInMaster: true - source: openapi31 tags repository: https://github.com/cedric05/swagger2har v3_1: true id: e5a20892c281663f25c6b50cecd202f7 repositoryMetadata: base64Readme: IyBzd2FnZ2VyMmhhcg== readmeEtag: '"52a2b7aaa2a0ec9d7ff8a336fca1b7ee6edcfccc"' readmeLastModified: Tue, 16 Jul 2024 13:44:05 GMT repositoryId: 829025450 description: null created: '2024-07-15T15:49:19Z' updated: '2024-07-16T13:44:15Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: cedric05 logo: https://avatars.githubusercontent.com/u/11557066?v=4 repoEtag: '"6d41fa9ca7c8b263993b17338a18dd33d0668fdb7d7e3af1d382e5a3d4ddd869"' repoLastModified: Tue, 16 Jul 2024 13:44:15 GMT category: Server Implementations foundInMaster: true - source: openapi31 tags repository: https://github.com/maks11060/openapi v3_1: true id: 05f95a2311cf6fd43cebf9e24ba9e4b6 repositoryMetadata: base64Readme: >- # OpenAPI schemas

Unofficial OpenAPI schemas for some public APIs

- [Danbooru](#danbooru)
- [Moebooru](#moebooru)
- [Shikimori](#shikimori)

> [!NOTE]
>
> - **OpenAPI 3.1**
> - Code examples [`/example`](example)
> - Schemas definitions in [zod](https://zod.dev/)
> - OpenAPI Builder [`jsr:@maks11060/openapi`](https://jsr.io/@maks11060/openapi)

## Danbooru

- [Danbooru](https://danbooru.donmai.us/)
  - [Api Wiki](https://danbooru.donmai.us/wiki_pages/help:api)
  - [Search Cheatsheet](https://danbooru.donmai.us/wiki_pages/help%3Acheatsheet)
  - [Danbooru Help Table of Contents](https://danbooru.donmai.us/wiki_pages/help:toc#dtext-developer_guide)
- [Code example](./example/danbooru.ts)
- [Redoc][danbooru.redoc]
- [Swagger Editor][danbooru.swagger]

### Typescript client with [openapi-fetch](https://openapi-ts.dev/openapi-fetch/)

#### 1. Generating types from OpenAPI schema

```ps
deno run -A npm:openapi-typescript \
  https://github.com/MAKS11060/openapi/releases/latest/download/danbooru.openapi.yaml \
  -o ./danbooru.oas.ts
```

```ps
npx openapi-typescript \
  https://github.com/MAKS11060/openapi/releases/latest/download/danbooru.openapi.yaml \
  -o ./danbooru.oas.ts
```

#### 2. Create openapi-fetch client

```ts
// danbooru.ts
import createClient from 'openapi-fetch'
import type {components, paths} from './danbooru.oas.ts'

export type Danbooru = components['schemas']

// Almost all GET requests do not require authorization.
// To use 'saved searches', you need an ApiKey.
// Register api key: https://danbooru.donmai.us/profile => API Key
// const login = ''
// const apiKey = ''
// const authorization = new TextEncoder().encode(`${login}:${apiKey}`).toBase64()

export const danbooruApi = createClient<paths>({
  baseUrl: 'https://danbooru.donmai.us',
  // headers: {authorization},
  querySerializer,
})

// deep serializer / {search: {id: [1,2]}} => ?search[id]=1,2
function querySerializer(
  obj: Record<string, unknown>,
  params: URLSearchParams = new URLSearchParams(),
  prefix = '',
): string {
  for (const [key, value] of Object.entries(obj)) {
    const encodedKey = encodeURIComponent(key)
    const paramKey = prefix ? `${prefix}[${encodedKey}]` : encodedKey
    if (value == null) continue
    if (Array.isArray(value)) {
      if (value.length === 0) continue
      params.append(paramKey, value.map(String).join(','))
    } else if (typeof value === 'object' && value !== null) {
      querySerializer(value as Record<string, unknown>, params, paramKey)
    } else {
      params.append(paramKey, String(value))
    }
  }

  return params.toString()
}
```

## Moebooru

- [Redoc][moebooru.redoc]
- [Swagger Editor][moebooru.swagger]

#### A group of services based on [moebooru](https://github.com/moebooru/moebooru)

- [yande.re](https://yande.re)
  - [Api doc](https://yande.re/help/api)
- [konachan.com](https://konachan.com) | [konachan.net](https://konachan.net)

### Typescript client with [openapi-fetch](https://openapi-ts.dev/openapi-fetch/)

#### 1. Generating types from OpenAPI schema

```ps
deno run -A npm:openapi-typescript \
  https://github.com/MAKS11060/openapi/releases/latest/download/moebooru.openapi.yaml \
  -o ./moebooru.oas.ts
```

```ps
npx openapi-typescript \
  https://github.com/MAKS11060/openapi/releases/latest/download/moebooru.openapi.yaml \
  -o ./moebooru.oas.ts
```

#### 2. Create openapi-fetch client

```ts
// moebooru.ts
import createClient from 'openapi-fetch'
import type {components, paths} from './moebooru.oas.ts'

export type Moebooru = components['schemas']

export const moebooruApi = createClient<paths>({
  baseUrl: 'https://yande.re',
  // baseUrl: 'https://konachan.com',
  // baseUrl: 'https://konachan.net',
})

// handle '/post.json?api_version=2'
api.use({ // merge two query parameters '?ver=2?tags=id:1' => '?ver=2&tags=id:1'
  onRequest({request}) {
    const url = new URL(request.url)
    if (url.search.slice(1).includes('?')) {
      url.search = url.search.slice(1).replace('?', '&')
    }
    return new Request(url, request)
  },
})
```

## Shikimori

- [Shikimori](https://shikimori.one/)
  - [Developer API](https://shikimori.one/api/doc)
- [Code example](./example/shikimori.ts)
- [Redoc][shikimori.redoc]
- [Swagger Editor][shikimori.swagger]

### Typescript client with [openapi-fetch](https://openapi-ts.dev/openapi-fetch/)

#### 1. Generating types from OpenAPI schema

```ps
deno run -A npm:openapi-typescript \
  https://github.com/MAKS11060/openapi/releases/latest/download/shikimori.openapi.yaml \
  -o ./shikimori.oas.ts
```

```ps
npx openapi-typescript \
  https://github.com/MAKS11060/openapi/releases/latest/download/shikimori.openapi.yaml \
  -o ./shikimori.oas.ts
```

2. Create openapi-fetch client

```ts
// shikimori.ts
import createClient from 'openapi-fetch'
import type {components, paths} from './shikimori.oas.ts'

export type Shikimori = components['schemas']

// Requirements
// Add your Oauth2 Application name to User-Agent requests header.
// Don’t mimic a browser.
// Your IP address may be banned if you use API without properly set User-Agent header.
const shikimoriUserAgent = 'your-oauth2-app-name'

export const shikimoriApi = createClient<paths>({
  baseUrl: 'https://shikimori.one',
  // headers: {'user-agent': shikimoriUserAgent},
  querySerializer: {
    array: {explode: false, style: 'form'},
  },
})
```

[danbooru.redoc]: https://redocly.github.io/redoc/?url=https://github.com/MAKS11060/openapi/releases/latest/download/danbooru.openapi.yaml
[danbooru.swagger]: https://editor.swagger.io/?url=https://no-cors.deno.dev/https://github.com/MAKS11060/openapi/releases/latest/download/danbooru.openapi.yaml
[moebooru.redoc]: https://redocly.github.io/redoc/?url=https://github.com/MAKS11060/openapi/releases/latest/download/moebooru.openapi.yaml
[moebooru.swagger]: https://editor.swagger.io/?url=https://no-cors.deno.dev/https://github.com/MAKS11060/openapi/releases/latest/download/moebooru.openapi.yaml
[shikimori.redoc]: https://redocly.github.io/redoc/?url=https://github.com/MAKS11060/openapi/releases/latest/download/shikimori.openapi.yaml
[shikimori.swagger]: https://editor.swagger.io/?url=https://no-cors.deno.dev/https://github.com/MAKS11060/openapi/releases/latest/download/shikimori.openapi.yaml

## Build

> [!NOTE]
> **Project structure**
>
> - `src/{service}/`
>   - `mod.ts` - Entry point
>   - `openapi.ts` - Config
>   - `schema.ts` - Data models

### Prerequisites

#### Install [Deno](https://github.com/denoland/deno/?tab=readme-ov-file#installation)

> Install dependencies

```ps
deno run init
```

#### Check formatting and build

```ps
deno run ok
```

#### Build OpenAPI Schemas

```ps
deno run build
```

#### Build client types in [example](./example/)

```ps
deno run build:client
```
 readmeEtag: '"4f958d46ea41c665f2a848a247c0beeca597a8ce"' readmeLastModified: Mon, 26 Jan 2026 15:13:18 GMT repositoryId: 890460198 description: OpenAPI schema for Danbooru, Moebooru, Shikimori created: '2024-11-18T15:52:52Z' updated: '2026-01-26T15:17:00Z' language: TypeScript archived: false stars: 1 watchers: 0 forks: 0 owner: MAKS11060 logo: https://avatars.githubusercontent.com/u/31521952?v=4 license: MIT repoEtag: '"f4094a9a436f329b67c308b6a5f003a63d3dbfbcafc480ad5126eacf116a4e7d"' repoLastModified: Mon, 26 Jan 2026 15:17:00 GMT category: Parsers foundInMaster: true - source: https://openapi.tools/ name: OpenApiSpecGeneratorPlugin category: - Auto Generators - Server Implementations language: C# link: >- https://learn.microsoft.com/en-us/microsoft-cloud/dev/dev-proxy/technical-reference/openapispecgeneratorplugin repository: https://github.com/dotnet/dev-proxy source_description: >- Built-in plugin for open-source DevProxy that automatically generates OpenAPI 3.0 Spec from captured network requests v3: true id: 9c3752b2072de3b2a3862b6499e1236c repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPiAKICA8aW1nIGFsdD0iRGV2IFByb3h5IiBzcmM9Ii4vbWVkaWEvaWNvbi5wbmciIHdpZHRoPSIxMjUiIC8+CiAgPHA+RGV2IFByb3h5PC9wPiAgCjwvaDE+Cgo8aDQgYWxpZ249ImNlbnRlciI+CiAgVGVzdCB0aGUgdW50ZXN0YWJsZQo8L2g0PgoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9ic2t5LmFwcC9wcm9maWxlL2RldnByb3h5LmJza3kuc29jaWFsIj4KICAgIDxpbWcgYWx0PSJCbHVlc2t5IiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvYmx1ZXNreS0lNDBkZXZwcm94eS5ic2t5LnNvY2lhbCVFMiU4MCVBQy1ibHVlP3N0eWxlPXNvY2lhbCZsb2dvPWJsdWVza3kmbGluaz1odHRwcyUzQSUyRiUyRmJza3kuYXBwJTJGcHJvZmlsZSUyRmRldnByb3h5LmJza3kuc29jaWFsIiAvPgogIDwvYT4KICA8YnIgLz4KICAgPGEgaHJlZj0iaHR0cHM6Ly95b3V0dWJlLmNvbS9AZGV2cHJveHkiPgogICAgPGltZyBhbHQ9IllvdVR1YmUiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS95b3VUdWJlLSU0MGRldnByb3h5JUUyJTgwJUFDLXJlZD9zdHlsZT1zb2NpYWwmbG9nbz15b3V0dWJlJmxpbms9aHR0cHMlM0ElMkYlMkZ5b3V0dWJlLmNvbSUyRiU0MGRldnByb3h5IiAvPgogIDwvYT4KICA8YnIgLz4KICA8YSBocmVmPSJodHRwczovL2FrYS5tcy9kZXZwcm94eSI+CiAgICBEb2N1bWVudGF0aW9uCiAgPC9hPgo8L3A+CgojIFdoYXQgaXMgRGV2IFByb3h5PwoKRGV2IFByb3h5IGlzIGFuIEFQSSBzaW11bGF0b3IgdGhhdCBoZWxwcyB5b3UgZWZmb3J0bGVzc2x5IHRlc3QgeW91ciBhcHAgYmV5b25kIHRoZSBoYXBweSBwYXRoLiBEZXYgUHJveHkgaXMgYSBjb21tYW5kLWxpbmUgdG9vbCB0aGF0IHdvcmtzIG9uIGFueSBwbGF0Zm9ybS4gQmVjYXVzZSBpdCBpbnRlcmNlcHRzIG5ldHdvcmsgcmVxdWVzdHMsIGl0IHdvcmtzIHdpdGggYW55IHR5cGUgb2YgYXBwIGFuZCB0ZWNoIHN0YWNrLgoKRGV2IFByb3h5IGlzICoqb3BlbiBzb3VyY2UqKiBhbmQgKipmcmVlIHRvIHVzZSoqLgoKIVtEZXYgUHJveHldKC4vbWVkaWEvYmFubmVyLnBuZykKCllvdSB0ZXN0IHlvdXIgYXBwIHRvIG1ha2Ugc3VyZSBpdCB3b3JrcyBhcyBpbnRlbmRlZC4gQnV0IHdoYXQgaWYgdGhlIEFQSXMgeW91IHVzZSBmYWlsPyBXaWxsIHlvdXIgYXBwIGxvc2UgeW91ciBjdXN0b21lcidzIGRhdGE/IEhvdyBkbyB5b3UgdGVzdCBmb3IgdGhpcz8gU2ltdWxhdGluZyBBUEkgZmFpbHVyZXMgaXMgaGFyZC4gWW91IGVuZCB1cCB3cml0aW5nIGNvZGUgdGhhdCB5b3Ugd29uJ3QgYmUgc2hpcHBpbmcgb3Igd29yc2U6IG5vdCB0ZXN0aW5nIGF0IGFsbC4gVGhhdCdzIHdoeSB3ZSBidWlsdCBEZXYgUHJveHksIHRvIHNpbXVsYXRlIEFQSSBlcnJvcnMgc28gdGhhdCB5b3UgY2FuIGVhc2lseSB0ZXN0IHlvdXIgYXBwIHdpdGhvdXQgY2hhbmdpbmcgeW91ciBjb2RlLgoKV2l0aCBEZXYgUHJveHkgeW91OgoKLSAqKlNlZSBob3cgeW91ciBhcHAgcmVzcG9uZHMgdG8gQVBJIGVycm9ycyoqLCB3aXRob3V0IGNoYW5naW5nIHlvdXIgYXBw4oCZcyBjb2RlLCBzbyB0aGF0IHlvdSBjYW4gKipidWlsZCBtb3JlIHJvYnVzdCBhcHBzIGFuZCBkb24ndCBsb3NlIGN1c3RvbWVycycgZGF0YSoqLgotICoqVmVyaWZ5IGhvdyB5b3VyIGFwcCBoYW5kbGVzIEFQSSByYXRlIGxpbWl0cyoqLCBzbyB0aGF0IHlvdSBjYW4gYXZvaWQgZ2V0dGluZyB0aHJvdHRsZWQgYW5kICoqaW1wcm92ZSB0aGUgdXNlciBleHBlcmllbmNlIGZvciB5b3VyIGN1c3RvbWVycyoqLgotICoqU2VlIGhvdyB5b3VyIGFwcCBoYW5kbGVzIHNsb3cgQVBJcyoqLCBzbyB0aGF0IHlvdSBjYW4gaW1wbGVtZW50IHRoZSBuZWNlc3NhcnkgYWZmb3JkYW5jZXMsIGFuZCAqKm1ha2UgeW91ciBhcHAgbW9yZSB1c2VyLWZyaWVuZGx5KiouCi0gKipRdWlja2x5IHN0YW5kLXVwIG1vY2sgQVBJcyoqIHdpdGhvdXQgd3JpdGluZyBhIGxpbmUgb2YgY29kZSwgc28gdGhhdCB5b3UgY2FuICoqZm9jdXMgb24gYnVpbGRpbmcgeW91ciBhcHAgaW5zdGVhZCBvZiB3cml0aW5nIGNvZGUgeW91IHdvbid0IGJlIHNoaXBwaW5nKiouCi0gSW1wcm92ZSB5b3VyIGFwcCB3aXRoIGNvbnRleHR1YWwgZ3VpZGFuY2Ugb24gaG93IHlvdSB1c2UgQVBJcywgdG8gKiptYWtlIHlvdXIgYXBwIGV2ZW4gYmV0dGVyKiouCgojIyBHZXQgc3RhcnRlZAoKVG8gZ2V0IHN0YXJ0ZWQgd2l0aCBEZXYgUHJveHksIGZvbGxvdyBvdXIgW3R1dG9yaWFsXShodHRwczovL2FrYS5tcy9kZXZwcm94eS9zdGFydCkgdG8gaW5zdGFsbCBhbmQgcnVuIERldiBQcm94eSBmb3IgdGhlIGZpcnN0IHRpbWUuCgpbIVtHZXR0aW5nIHN0YXJ0ZWQgd2l0aCBEZXYgUHJveHldKGh0dHBzOi8vaW1nLnlvdXR1YmUuY29tL3ZpL0hWVEpsR1N4aGN3LzAuanBnKV0oaHR0cHM6Ly93d3cueW91dHViZS5jb20vd2F0Y2g/dj1IVlRKbEdTeGhjdykKCiMjIC5ORVQgRm91bmRhdGlvbgoKVGhpcyBwcm9qZWN0IGlzIHN1cHBvcnRlZCBieSB0aGUgWy5ORVQgRm91bmRhdGlvbl0oaHR0cHM6Ly9kb3RuZXRmb3VuZGF0aW9uLm9yZykuCg== readmeEtag: '"664d7cfda4d5585a33b54cb38cfcd6ca0a3b5cbd"' readmeLastModified: Fri, 16 Jan 2026 09:43:21 GMT repositoryId: 534755927 description: Simulate API failures, throttling, and chaos — all from your command line. created: '2022-09-09T18:11:29Z' updated: '2026-02-06T01:54:43Z' language: C# archived: false stars: 763 watchers: 14 forks: 79 owner: dotnet logo: https://avatars.githubusercontent.com/u/9141961?v=4 license: MIT repoEtag: '"d4f0b39da97f42606401eefe07b59c1c84aaf2ac4d949f25e6c16bedad959f03"' repoLastModified: Fri, 06 Feb 2026 01:54:43 GMT foundInMaster: true oldLocations: - https://github.com/microsoft/dev-proxy - source: openapi3 tags repository: https://github.com/vincenzo-racca/spring-boot-openapi v3: true repositoryMetadata: repositoryId: 291784837 description: >- Java project with Spring Boot 2, Swagger 3 with the OpenAPI specification and a self-generated client. created: '2020-08-31T17:48:07Z' updated: '2021-04-10T15:59:33Z' language: Java archived: false stars: 0 watchers: 1 forks: 2 owner: vincenzo-racca logo: https://avatars.githubusercontent.com/u/69170382?v=4 repoEtag: '"2174212c21000ca16011389c751af56ad8f6a3aed37996fdfeca9b8e656e7851"' repoLastModified: Sat, 10 Apr 2021 15:59:33 GMT foundInMaster: true id: 069b27ff6187547339fe168ff01e3981 - source: openapi3 tags repository: https://github.com/agranya99/product-management-system v3: true repositoryMetadata: base64Readme: >- # Product Management System

A scalable REST API developed in Node.Js to facilitate CRUD and utility operations on products, categories, providers (Business Partners).

The system design is such that it supports all kinds of Products (from stationery items to online MOOCs), with the use of nested `attributes` object that supports searching too. Each product can have its own attributes. The model provides generalization and specification as the categories support sub-categories and so on. 

Authorization is performed via **OAuth 2.0 (Client Credentials Flow)** and Request Validation is performed using **Hapi/JOI**. The API uses MongoDB as back-end database with **Mongoose** framework for data modelling, querying and validation. 

**OpenAPI 3.0 Specification** is written for the API. <https://app.swaggerhub.com/apis/agranya99/Product-Management-System/1.0.0>

## Softwares and Frameworks
- OpenAPI 3.0 Specification
- express.js
- OAuth 2.0 (Okta Authorization Server)
- JOI 
- Mongoose
- MongoDB Atlas

## How to Run

### Installation

- Clone this GitHub repo to 'dir'
- Navigate to 'dir'
- Run `npm install` to install the dependencies
- Run `nodemon .\server.js` or `node .\server.js` to start a server instance of the project

### Authorization

- Create a POST Request to `{ ISSUER }/v1/token` == <https://dev-941571.okta.com/oauth2/aus46pynt0MINWHhm4x6/v1/token>  with
  - Pass `Basic` Authorization Header: `{ clientID }` and `{ clientSecret }` from .env
  - Pass Parameters: `scope` from .env and `grant_type = client_credentials`
  ![Postman Auth Image](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/getAccessToken.png "Postman Example")
  
- If credentials are valid, the application will receive back a response with `Bearer` token (around 800 characters)
  - ```
    {
    "access_token": "eyJhbG[...]1LQ",
    "token_type": "Bearer",
    "expires_in": 86400,
    "scope": "pms"
    }
    ```
    
- Pass this `Bearer` token as Request Header of your requests to the REST API
  

## Features and Nitty Gritties

### OpenAPI 3.0 Specification
Detailed API Description - available endpoints and possible operations on the endpoints with request and response details.

The specification is built using re-usable `schemas`, `parameters`, `requestBodies` and `responses` and includes `securitySchemes` for OAuth2.0.

Checkout <https://app.swaggerhub.com/apis/agranya99/Product-Management-System/1.0.0> or openapi-spec.yaml

### Generalization and Specification (scalability!)

The Products feature `attributes` object which can contain any number of attributes with multiple values

Like:
```
attributes: { "colors": ["white", "black"], "sizes": ["13in", "15in"] ... }
```

The Categories feature `parentCategoryID` which allows branching of categories into more specific ones, like peripherals is a sub-category of computers again is a sub-category of electronics.

### qTags - Tags associated with a product

`qTags` help with searching and finding similar products. These are an array of strings with terms loosely associated with a product.

Like:
```
    {
        "qTags": ["laptop", "gaming", "student", "creator"],
        "imageURLs": ["D:/blade/image001.png"],
        "providerID": 1,
        "status": "available",
        "sku": "123abc45",
        "name": "Blade Stealth",
        "categoryID": 1,
        "attributes": { "colors": ["white", "black", "silver"],
                        "sizes": ["13in", "15in"] },
        "price": 115999,
        "launchDate": "2020-03-15T00:00:00.000Z",
        "stock": 3500
    }
```

### Fetching, Searching and Pagination
`limit` and `offset` parameters can be passed as query string to set a limit on number of records to fetch at once while being able to navigate through the whole set. This helps with **scalability**. 

Like:
- `localhost:3000/products?offset=0&limit=25`,
- `localhost:3000/categories?offset=1&limit=10`,
- `localhost:3000/providers?limit=10`

**GET /products**
Fetch a list of Products via **queries**

Like:
- `localhost:3000/products?name=Blade Stealth`,
- `localhost:3000/products?provider=Razer`,
- `localhost:3000/products?qTags=marker&attributes[colors]=black`

**GET /categories**
Fetch List of Category objects using **queries**

Like:
- `localhost:3000/categories?name=Peripherals`

**GET /providers**
Fetch a list of Providers (Partner Businesses) via **queries**

Like:
- `localhost:3000/providers?name=Razer`

| ![searchProductsByqTagsAndAttributes](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/searchProductsByqTagsAndAttributes.PNG "searchProductsByqTagsAndAttributes") | ![searchCategoriesByName](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/searchCategoriesByName.PNG "searchCategoriesByName") |
|---------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------------------------------|
| ![paginateProducts](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/paginateProducts.PNG "paginateProducts")                                                       | ![getSimilarProducts](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/getSimilarProducts.PNG "getSimilarProducts")             |



| Method | Endpoint                               | Description                                                                           |
|--------|----------------------------------------|---------------------------------------------------------------------------------------|
| GET    | /products/{sku}                        | Get the details of a product                                                          |
| GET    | /categories/{categoryID}               | Get the details of a category                                                         |
| GET    | /categories/{categoryID}/subCategories | Get sub categories of a category                                                      |
| GET    | /categories/{categoryID}/products      | Get all products in a category                                                        |
| GET    | /products/{sku}/similar            | Get similar products to a particular product (Based on Query Tags -  qTags attribute) |
| GET    | /providers/{providerID}                | Get the details of a Provider (Partner Business)                                      |


### Records Manipulation
MongoDB database is used for storing records. **Mongoose** is used  to model the data. It includes built-in type casting, validation, query building. Refer OpenAPI Spec for request bodies and parameters.

**Responses are consistent**

POST
- `200` with { `sku`, `status`, `message` } for successful operation
- `400` with { `status`, `message` } for Invalid Request / Duplicate Key Error

PUT
- `200` with updated `product`, `category` or `provider` object for successful operation
- `404` for Not Found Error on the resource to be updated
- `400` with { `status`, `message` } for Invalid Request

DELETE
- `200` with { `sku`, `status`, `message` } for successful operation
- `404` for Not Found Error on the resource to be updated
- `400` with { `status`, `message` } for Invalid Request

Validation Errors
- `400` response with { `status`, `message` } describing what caused the validation to fail

Internal Server Errors
- `500` response with { `status`, `message` }, no description of error

| ![createProduct](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/createProduct.PNG "createProduct") | ![createCategory](https://github.com/agranya99/Product-Management-System/blob/master/screenshots/createCategory.PNG "createCategory") |
|------------------------------------------------------------------------------------------------------------------------------------|---------------------------------------------------------------------------------------------------------------------------------------|


| Method | Endpoint                               | Description                                                                           |
|--------|----------------------------------------|---------------------------------------------------------------------------------------|
| POST   | /products                              | Add a new product via JSON request                                                    |
| PUT    | /products/{sku}                        | Update an existing product                                                            |
| DELETE | /products/{sku}                        | Delete an existing product                                                            |
| POST   | /categories                            | Add a new category via JSON request                                                   |
| PUT    | /categories/{categoryID}               | Update an existing category                                                           |
| DELETE | /categories/{categoryID}               | Delete an existing category and all its subcategories                                 |
| POST   | /providers                             | Add a new Provider (Partner Business) via JSON request                                                   |
| PUT    | /providers/{providerID}                | Update an existing Provider (Partner Business)                                        |
| DELETE | /providers/{providerID}                | Delete an existing Provider                                                           |

### Authorization 
**OAuth 2.0 Client Credentials Grant** is implemented using Okta Authorization server. It helps with granting limited access without user credentials, set rate limits, etc.

The client (daemon) needs to fetch a valid `token` from the okta authorization server by sending a request with `clientID`, `clientSecret`, and `scope`.

The received token (whose validity can be set in the authorization server, max = 1 day) is then passed as header (`bearer`) to the API server with each request.

### Request Validation
**Hapi/JOI** is used to perform schema-based request validation. It helps enforce standards and prevent unexpected errors.

Checkout: <https://github.com/agranya99/Product-Management-System/tree/master/screenshots/errors>


## Miscellaneous

**GET /**
Fetch a list of all valid endpoints

`GET` `POST` `PUT` `DELETE` /* 
Invalid URL handled. List of valid endpoints returned.
 readmeEtag: '"dfbb8d3c1b36efc2371231e499d6d306015222bc"' readmeLastModified: Fri, 20 Mar 2020 17:32:16 GMT repositoryId: 248045592 description: >- A scalable REST API developed in Node.Js to facilitate CRUD and utility operations on products, categories, providers (Business Partners). OpenAPI 3.0 Specification: created: '2020-03-17T18:34:43Z' updated: '2020-03-20T17:32:19Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: agranya99 logo: https://avatars.githubusercontent.com/u/35340618?v=4 repoEtag: '"17ccc307769da15dc16f7aa139be2b147465a5fbc35eca4e1954a8485c0f3ea6"' repoLastModified: Fri, 20 Mar 2020 17:32:19 GMT foundInMaster: true category: Server Implementations id: 95b027c6e559274844c25f5bbc89a71d - source: openapi3 tags repository: https://github.com/mwczapski/swagger_codegen_3_docker_container v3: true repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyX0NvZGVnZW5fMy4wX0RvY2tlcl9Db250YWluZXIKClRoZSBpbnRlbnQgb2YgdGhpcyByZXBvc2l0b3J5IGlzIHRvIGhvc3QgcmVzb3VyY2VzIHRoYXQgcHJvdmlkZSB0aGUgbWVhbnMgdG8gY3JlYXRlIGEgc2VsZi1jb250YWluZWQgRG9ja2VyIGNvbnRhaW5lciBmb3IgQVBJLUZpcnN0IGRldmVsb3BtZW50IHVzaW5nIHRoZSBsYXRlc3QgU3dhZ2dlciBDb2RlZ2VuICgzLngpIGFuZCBPcGVuQVBJICgyIG9yIDMpLgoKVGhlIGZvbGxvd2luZyByZXNvdXJjZXMgYXJlIGF2YWlsYWJsZSBpbiB0aGlzIHJlcG9zaXRvcnk6CgoxLiBbQ3JlYXRlIERvY2tlciBJbWFnZV0oU3dhZ2dlcl9Db2RlZ2VuX0RvY2tlcl9Db250YWluZXJfZm9yX0FQSS1GaXJzdF9EZXZlbG9wbWVudC5tZCkgLSBTdGVwcyB0byBjcmVhdGUgdGhlIHJlYWR5LXRvLWdvIFN3YWdnZXIgQ29kZWdlbiBEb2NrZXIgQ29udGFpbmVyLgoyLiBbVXNlIFN3YWdnZXIgQ29kZWdlbiBEb2NrZXIgQ29udGFpbmVyXShIb3dfVG9fVXNlX1N3YWdnZXJfQ29kZWdlbl9Eb2NrZXJfSW1hZ2UubWQpIC0gQ3JlYXRlIGFuZCB1c2UgdGhlIGNvbnRhaW5lcgozLiBbQ3JlYXRlIFN3YWdnZXIgQ29kZWdlbiBFbnZpcm9ubWVudCBpbiBBbHBpbmUgV1NMXShjcmVhdGVfc3dhZ2dlcl9jb2RlZ2VuX2luX3dzbF9hbHBpbmUubWQpIC0gQ3JlYXRlIFN3YWdnZXIgQ29kZWdlbiBlbnZpcm9ubWVudCBpbiBXaW5kb3dzIFdTTDIgdXNpbmcgQWxwaW5lIExpbnV4IGRpc3RyaWJ1dGlvbgoKVGhlIERvY2tlciBJbWFnZSBjcmVhdGVkIGluICgxKSBhbmQgdXNlZCBpbiAoMikgaXMgYXZhaWxhYmxlIGF0IFttd2N6YXBza2kvc3dhZ2dlci1jb2RlZ2VuOjEuMC4wXShodHRwczovL2h1Yi5kb2NrZXIuY29tL3IvbXdjemFwc2tpL3N3YWdnZXJfY29kZWdlbikgb24gRG9ja2VyIEh1Yi4K readmeEtag: '"1e40b981af605d7a572789fff5a2c7f516a4215b"' readmeLastModified: Sun, 09 Aug 2020 03:27:08 GMT repositoryId: 278782925 description: >- The intent of this repository is to host resources that provide the means to create a self-contained Docker container for API-First development using the latest Swagger Codegen (3.x) and OpenAPI (2 or 3). created: '2020-07-11T03:50:21Z' updated: '2020-08-09T03:28:59Z' language: Shell archived: false stars: 0 watchers: 1 forks: 0 owner: mwczapski logo: https://avatars.githubusercontent.com/u/57377978?v=4 license: MIT repoEtag: '"4cd815081cae4b87a75044435b001b6d606f0c71e4b1e9d767d449a5d9303f32"' repoLastModified: Sun, 09 Aug 2020 03:28:59 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 353287ebae45b064ba880934bb30c6f9 - source: openapi3 tags repository: https://github.com/zit0un/geojson-oas3 v3: true repositoryMetadata: base64Readme: >- IyBHZW9KU09OLU9BUzMKKkdlb0pTT04gZGVmaW5pdGlvbiB3aXRoIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiB2My4wKgoKVGhpcyBkb2N1bWVudCBkZWZpbmVzIHRoZSBHZW9KU09OIGZvcm1hdCBhcyBhbiBPcGVuQVBJIHYzLgpJdCBjb250YWlucyB0aGUgZGVmaW5pdGlvbnMgZm9yICdGZWF0dXJlJyBvYmplY3QgYW5kICdGZWF0dXJlQ29sbGVjdGlvbicKb2JqZWN0cywgYXMgd2VsbCBhcyB0aGUgZGVmaW5pdGlvbnMgZm9yIGFsbCAnR2VvbWV0cnknIG9iamVjdHMuCkl0IGNvbmZvcm1zIHdpdGggdGhlICdSRkMtNzk0Nicgc3RhbmRhcmQgZnJvbSBJRVRGIChBdWd1c3QgMjAxNiB2ZXJzaW9uKQo8YnIvPmh0dHBzOi8vdG9vbHMuaWV0Zi5vcmcvaHRtbC9yZmM3OTQ2CgpLdWRvcyB0byBAYnViYm9ibmUgYW5kIEBpZGt3IHdob3NlIGNvZGUgaGVscGVkIG1lIG5vdCBzdGFydCBmcm9tIHNjcmF0Y2gKPGJyLz5odHRwczovL2dpc3QuZ2l0aHViLmNvbS9idWJib2JuZS9mZTVmMmRiNjVhY2YwMzliZTZhOWZkOTJmYzljNzIzMwo= readmeEtag: '"0673d964ed14f3c99e3e4e9c2192999e1543205b"' readmeLastModified: Wed, 30 Jun 2021 07:19:40 GMT repositoryId: 327877838 description: OAS3 definitions for GeoJSON created: '2021-01-08T11:00:03Z' updated: '2021-06-30T07:21:11Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: zit0un logo: https://avatars.githubusercontent.com/u/23296627?v=4 license: MIT repoEtag: '"6deb67c9e7d9c41fc8b05f198b29924bc431ddec2da876ad1c0f94bcddfa50bb"' repoLastModified: Wed, 30 Jun 2021 07:21:11 GMT foundInMaster: true category: Parsers id: 97e3320b6415a877c8152e14611b7933 - source: openapi3 tags repository: https://github.com/spider101/imagegram v3: true repositoryMetadata: base64Readme: >- IyBJbWFnZWdyYW0KCiMjIE92ZXJ2aWV3CgpUaGlzIGlzIGEgc2ltcGxlIGFwcGxpY2F0aW9uIGV4cG9zaW5nIGEgUkVTVCBBUEkgdG8gbWFuYWdlIHBvc3RzIG1hZGUgYnkgdXNlcnMgd2hpY2ggaW5jbHVkZSBpbWFnZXMgYW5kIGNvbW1lbnRzLiBUaGUgdXNlcnMgY2FuIGNyZWF0ZSB0aGVpciBvd24gYWNjb3VudCwgY3JlYXRlIG5ldyBwb3N0cyBmcm9tIHRoZWlyIGFjY291bnQgYW5kIGV2ZW4gYWRkIGNvbW1lbnRzIG9uIHZhcmlvdXMgcG9zdHMsIGluY2x1ZGluZyB0aGVpciBvd24uIEVhY2ggcG9zdCBjb25zaXN0cyBvZiBhIGNhcHRpb24gYW5kIGFuIGltYWdlIGFuZCB0aGUgdXNlciBjYW4gYWNjZXNzIGFsbCB0aGUgcG9zdHMgaW4gdGhlIGFwcGxpY2F0aW9uIHZpYSBhIHNpbmdsZSBlbmRwb2ludC4gRmluYWxseSwgdG8gc2FmZWd1YXJkIHRoZSB1c2VyJ3MgcHJpdmFjeSwgdGhleSBhcmUgYWxzbyBnaXZlbiB0aGUgY2FwYWJpbGl0eSBvZiBkZWFjdGl2YXRpbmcgdGhlaXIgYWNjb3VudCB3aGljaCBhbHNvIGRlbGV0ZXMgYWxsIGRhdGEgKHBvc3RzIGFuZCBjb21tZW50cykgYXNzb2NpYXRlZCB3aXRoIHRoZWlyIGFjY291bnQuCgojIyBQcmVyZXF1aXNpdGVzCgpZb3UgbXVzdCBoYXZlIERvY2tlciBpbnN0YWxsZWQgb24geW91ciBjb21wdXRlci4gWW91IGNhbiBmb2xsb3cgdGhlc2UgW2luc3RydWN0aW9uc10oaHR0cHM6Ly93d3cuZG9ja2VyLmNvbS9wcm9kdWN0cy9kb2NrZXItZGVza3RvcCkgdG8gaGF2ZSBpdCBxdWlja2x5IGluc3RhbGxlZC4gWW91IG11c3QgYWxzbyBoYXZlIGEgYC5lbnZgIGNyZWF0ZWQgYXQgdGhlIHJvb3Qgb2YgdGhlIHJlcG9zaXRvcnkuIFlvdSBjYW4gcmVmZXIgdG8gdGhlIGAuZW52YCBbZXhhbXBsZSBmaWxlXSguZW52LmV4YW1wbGUpIGZvciB0aGUgdmFyaWFibGVzIHlvdSBuZWVkIHRvIGluY2x1ZGUuCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCk9uY2UgeW91IGhhdmUgRG9ja2VyIHNldHVwLiBSdW4gdGhlIGZvbGxvd2luZyBjb21tYW5kcyBvbiB5b3VyIHRlcm1pbmFsIHRvIHNwaW4gdXAgdGhlIGNvbnRhaW5lcnMgdGhhdCB3aWxsIGNvbXBvc2UgdGhlIGFwcGxpY2F0aW9uLgoKYGBgc2hlbGwKZG9ja2VyLWNvbXBvc2UgYnVpbGQKZG9ja2VyLWNvbXBvc2UgdXAKYGBgCgpUbyB2ZXJpZnkgdGhlIGFwcGxpY2F0aW9uIGlzIHJ1bm5pbmcsIHlvdSBjYW4gaGl0IHRoZSBmb2xsb3dpbmcgZW5kcG9pbnQgdG8gcGVyZm9ybSBhIGBoZWFsdGNoZWNrYDoKCmBgYHNoZWxsCmN1cmwgLS1sb2NhdGlvbiAtLXJlcXVlc3QgR0VUICdsb2NhbGhvc3Q6MzAwMS9oZWFsdGhjaGVjaycKYGBgCgpUaGVyZWFmdGVyLCB5b3UgY2FuIGludGVyYWN0IHdpdGggdGhlIGFwcGxpY2F0aW9uIG9uIHRoZSBmb2xsb3dpbmcgZW5kcG9pbnRzOgoKKiBQT1NUIGAvYWNjb3VudHNgIHRvIGNyZWF0ZSBuZXcgYWNjb3VudHMKKiBERUxFVEUgYC9hY2NvdW50cy86YWNjb3VudElkYCB0byBkZWxldGUgYW4gYWNjb3VudCBhbmQgYWxsIHBvc3QgYW5kIGNvbW1lbnQgZGF0YSBhc3NvY2lhdGVkIHdpdGggaXQKKiBQT1NUIGAvcG9zdHNgIHRvIGNyZWF0ZSBuZXcgcG9zdHMKKiBHRVQgYC9wb3N0c2AgdG8gZmV0Y2ggYWxsIHBvc3RzIGluIHRoZSBhcHBsaWNhdGlvbgoqIFBPU1QgYC9jb21tZW50c2AgdG8gYWRkIG5ldyBjb21tZW50cyBvbiBwb3N0cwoKWW91IGNhbiBmaW5kIG1vcmUgaW5mb3JtYXRpb24gb24gaG93IHRvIGJ1aWxkIHRoZSByZXNwZWN0aXZlIGNVUkwgY2FsbHMgYnkgZm9sbG93aW5nIHRoZSBPcGVuQVBJIHNwZWMgZm91bmQgW2hlcmVdKHNyYy9hcGkvb3BlbmFwaS55YW1sKQoKVG8gc3RvcCB0aGUgYXBwbGljYXRpb24sIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQgdG8gc3RvcCB0aGUgcnVubmluZyBjb250YWluZXJzIGFuZCByZW1vdmUgdGhlbS4KCmBgYHNoZWxsCmRvY2tlci1jb21wb3NlIHN0b3AKYGBg readmeEtag: '"85e0652d8eaade5d29b17255f716763f730c6368"' readmeLastModified: Thu, 18 Nov 2021 15:06:54 GMT repositoryId: 412525316 description: >- A simple application exposing a REST API to manage posts made by users which include images and comments. created: '2021-10-01T15:42:06Z' updated: '2021-11-18T15:07:07Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: Spider101 logo: https://avatars.githubusercontent.com/u/6288118?v=4 repoEtag: '"0687da0b7a3cf69d2a8745843617e26e351b6a48df159bf08dcd569483f67d91"' repoLastModified: Thu, 18 Nov 2021 15:07:07 GMT foundInMaster: true category: Server Implementations id: be2ee1b145265eeb87a69c0a430c8acd - source: openapi3 tags repository: https://github.com/alxael/product-api v3: true id: 616741f2748aee446ee398254bb6e602 repositoryMetadata: base64Readme: >- IyBQcm9kdWN0IEFQSQoKVGhpcyBwcm9qZWN0IGlzIGEgUkVTVGZ1bCBBUEkgZGVzaWduZWQgZm9yIHRoZSBtYW5hZ2VtZW50IG9mIGFuIG9ubGluZSBzaG9wLgoKIyMgVGVjaG5vbG9naWVzIHVzZWQKClRoZSBwcm9qZWN0IHVzZXMgdGhlIGZvbGxvd2luZyB0ZWNobm9sb2dpZXMgdG8gd29yazoKCi0gW05vZGVKU10oaHR0cHM6Ly9ub2RlanMub3JnL2VuLykKLSBbRXhwcmVzc10oaHR0cHM6Ly9leHByZXNzanMuY29tLykKLSBbTW9uZ29EQl0oaHR0cHM6Ly93d3cubW9uZ29kYi5jb20vKQotIFtTd2FnZ2VyXShodHRwczovL3N3YWdnZXIuaW8vdG9vbHMvc3dhZ2dlci1jb2RlZ2VuLykKLSBbQXp1cmUgRGV2T3BzXShodHRwczovL2F6dXJlLm1pY3Jvc29mdC5jb20vZW4tdXMvc2VydmljZXMvZGV2b3BzLykKCiMjIEZlYXR1cmVzCgpBdCB0aGUgbW9tZW50LCB0aGUgQVBJIGhhcyBlbmRwb2ludHMgZm9yIHByb2R1Y3RzLCB1c2VycyBhbmQgdXNlciByb2xlcy4gVGhlIEFQSSB1c2VzIEpXVCB0byBzaWduIGNsYWltcyBzZWN1cmVseS4KCk5vZGVKUyBpcyB1c2VkIHRvIHJ1biB0aGUgd2hvbGUgYXBwbGljYXRpb24uIEV4cHJlc3MgaXMgdXNlZCB0byBwcm92aWRlIHNlcnZlci1zaWRlIGxvZ2ljIGZvciB0aGUgd2ViIGFwcGxpY2F0aW9uLiBNb25nb0RCIGlzIHVzZWQgdG8gaGFuZGxlIHRoZSBzdG9yYWdlIGFuZCBtYW5hZ2VtZW50IG9mIHRoZSBkYXRhLiBTd2FnZ2VyIGlzIHVzZWQgdG8gZG9jdW1lbnQgYW5kIGludGVyYWN0IG1hbnVhbGx5IHdpdGggdGhlIEFQSS4gQXp1cmUgRGV2T3BzIGlzIHVzZWQgd2l0aCBhZ2lsZSBtZXRob2RvbG9naWVzIHRvIGVuc3VyZSBhIHNtb290aCBkZXZlbG9wbWVudCBwcm9jZXNzLgoKIyMgSW5zdGFsbGF0aW9uCgojIyMgU3RlcCAxCgpUaGUgQVBJIHJlcXVpcmVzIFtOb2RlSlNdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi8pIHZlcnNpb24gMTQgb3IgbmV3ZXIgdG8gd29yayBhbmQgW01vbmdvREJdKGh0dHBzOi8vd3d3Lm1vbmdvZGIuY29tLykuIEVuc3VyZSB0aGVzZSB0ZWNob25sb2dpZXMgYXJlIGluc3RhbGxlZC4KCiMjIyBTdGVwIDIKCkNsb25lIHRoZSByZXBvc2l0b3J5LgoKYGBgY29uc29sZQpnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2FseGFlbC9ib2lsZXJwbGF0ZS1iYWNrZW5kLmdpdApgYGAKCiMjIyBTdGVwIDMKCkVudGVyIHRoZSBkaXJlY3RvcnkgaW4gd2hpY2ggdGhlIHJlcG9zaXRvcnkgd2FzIGNsb25lZC4gSW5zdGFsbCB0aGUgcGFja2FnZXMgd2l0aCBgYGBucG0gaW5zdGFsbGBgYC4gVGhlbiwgYWRkIGEgYGBgLmVudmBgYCBmaWxlIGJ5IHJ1bm5pbmcgdGhlIGBgYHRvdWNoIC5lbnZgYGAgY29tbWFuZC4gVGhpcyBmaWxlIHNob3VsZCBoYXZlIHRoZSBmb2xsb3dpbmcgc3RydWN0dXJlOgoKYGBgZG9zaW5pCk5PREVfRU5WPWRldmVsb3BtZW50ClBPUlQ9NDAwMApEQl9MSU5LPW1vbmdvZGI6Ly9sb2NhbGhvc3Q6MjcwMTcvbXlhcHAKVE9LRU5fS0VZPSpyYW5kb20gdGV4dCoKU0lHTlVQX1JPTEU9Y2xpZW50CmBgYAoKIyMjIFN0ZXAgNAoKT3BlbiBhIHRlcm1pbmFsIHdpbmRvdyBhbmQgc3RhcnQgdGhlIE1vbmdvREIgc2VydmVyIGJ5IHJ1bm5pbmcgdGhlIGBgYG1vbmdvZGBgYCBjb21tYW5kLiBJbiBhIHNlcGFyYXRlIHRlcm1pbmFsIHdpbmRvdywgb3BlbiB0aGUgTW9uZ28gQ0xJIGJ5IHJ1bm5pbmcgdGhlIGBgYG1vbmdvYGBgIGNvbW1hbmQuIFRoZW4gcnVuIHRoZSBgYGBzaG93IGRic2BgYCBjb21tYW5kIHRvIHZpZXcgYWxsIGRhdGFiYXNlcy4gTm93LCBydW4gdGhlIGBgYHVzZSBteWFwcGBgYCBjb21tYW5kLgoKIyMjIFN0ZXAgNQoKTm93IHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmQ6CgpgYGBqYXZhc2NyaXB0CmRiLnVzZXJyb2xlcy5pbnNlcnQoCiAgWwogICAge3VzZXJSb2xlOiAiY2xpZW50In0sCiAgICB7dXNlclJvbGU6ICJtb2RlcmF0b3IifSwKICAgIHt1c2VyUm9sZTogImFkbWluaXN0cmF0b3IifQogIF0KKQpgYGAKClRoaXMgd2lsbCBjcmVhdGUgdGhlIHVzZXIgcm9sZXMsIHdoaWNoIHVwb24gcnVubmluZyB0aGUgYGBgZGIudXNlcnJvbGVzLmZpbmQoKS5wcmV0dHkoKWBgYCBjb21tYW5kIHNob3VsZCBsb29rIHNpbWlsYXIgdG8gdGhlIGZvbGxvd2luZzoKCmBgYGpzb24KeyAKICAgICJfaWQiIDogT2JqZWN0SWQoIjYxNTZiZTA0NTZjMDhlY2NkNjI5YzZmOCIpLAogICAgInVzZXJSb2xlIiA6ICJjbGllbnQiCiAgICB9CnsKICAgICJfaWQiIDogT2JqZWN0SWQoIjYxNTZiZTA0NTZjMDhlY2NkNjI5YzZmOSIpLAogICAgInVzZXJSb2xlIiA6ICJtb2RlcmF0b3IiIAp9CnsKICAgICJfaWQiIDogT2JqZWN0SWQoIjYxNTZiZTA0NTZjMDhlY2NkNjI5YzZmYSIpLAogICAgInVzZXJSb2xlIiA6ICJhZG1pbmlzdHJhdG9yIgp9CmBgYAoKIyMjIFN0ZXAgNgoKTm93LCBpbiB0aGUgYXBwbGljYXRpb24gZmlsZXMsIG1vZGlmeSB0aGUgYGBgU0lHTlVQX1JPTEVgYGAgZW52aXJvbm1lbnQgdmFyaWFibGUgdG8gYGBgYWRtaW5pc3RyYXRvcmBgYCBvciBgYGBtb2RlcmF0b3JgYGAgYW5kIGNyZWF0ZSBhbiBhY2NvdW50IHdpdGggdGhvc2UgcGVybWlzc2lvbnMuIE1ha2Ugc3VyZSB0byBjaGFuZ2UgdGhlIGVudmlyb25tZW50IHZhcmlhYmxlIGJhY2sgdG8gYGBgY2xpZW50YGBgIGFmdGVyIGZpbmlzaGluZyB0aGlzIGluaXRpYWwgcHJvY2Vzcy4KCiMjIyBTdGVwIDcKClNpZ24gaW4gaW50byBhbiBhZG1pbmlzdHJhdG9yIGFjY291bnQgdG8gdXNlIHRoZSB3aG9sZSBhcHBsaWNhdGlvbi4gVXNlIHRoZSBgYGBQT1NUIC91c2VyL2xvZ2luYGBgIGVuZHBvaW50IHRvIGxvZyBpbi4gQWZ0ZXIgbG9naW4sIHlvdSB3aWxsIGJlIGdyYW50ZWQgYSBKV1QgdG9rZW4uIENvcHkgc2FpZCB0b2tlbiBpbnRvIHRoZSB3aW5kb3cgdGhhdCBwb3BzIHVwIHdoZW4geW91IHByZXNzIHRoZSBgYGBBdXRob3JpemVgYGAgYnV0dG9uLCBpbiB0aGUgcmlnaHQgaGFuZCBzaWRlLiBLZWVwIHRoaXMgdG9rZW4gc2F2ZWQgc29tZXdoZXJlIGFuZCB1cGRhdGUgaXQgZXZlcnkgdGltZSB5b3UgbG9nIGluLCBhcyB5b3Ugd2lsbCBuZWVkIGl0IHRvIHVzZSBtb3N0IG9mIHRoZSBlbmRwb2ludHMgb2YgdGhlIEFQSS4KCkNvbmdyYXR1bGF0aW9ucyEgWW91IGFyZSBub3cgYWxsIHNldCB1cCB0byB1c2UgdGhlIEFQSS4KCiMjIExpY2Vuc2UKClRoZSBwcm9qZWN0IGlzIGN1cnJlbnRseSB1c2luZyB0aGUgW01JVF0oaHR0cHM6Ly9jaG9vc2VhbGljZW5zZS5jb20vbGljZW5zZXMvbWl0LykgbGljZW5zZS4K readmeEtag: '"e47e3ec3de120c4d12352f90ccdce8d57898f5b3"' readmeLastModified: Sun, 15 Oct 2023 14:15:50 GMT repositoryId: 410784269 description: Product management API. created: '2021-09-27T07:28:57Z' updated: '2023-10-15T14:17:20Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: alxael logo: https://avatars.githubusercontent.com/u/78294861?v=4 license: MIT repoEtag: '"16feb3fe9ca93cde2188cd2bc96adc76c9412ad0c6db529a08a4d1e9c248e531"' repoLastModified: Sun, 15 Oct 2023 14:17:20 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cant-code/parking-app-backend v3: true repositoryMetadata: base64Readme: >- IyBHZXR0aW5nIFN0YXJ0ZWQKCiMjIyBSZXF1aXJlbWVudHMKKiBKYXZhIDExIChNaW4gdmVyc2lvbiA4LCBjaGFuZ2UgdGhlIEphdmEgdmVyc2lvbiBpbiB0aGUgcG9tLnhtbCBmaWxlIGFjY29yZGluZ2x5KQoqIFJlbGF0aW9uYWwgREIgLSBQb3N0Z3JlU1FMCiogSUFNIC0gS2V5Y2xvYWsKCiMjIyBFbnZpcm9ubWVudCBWYXJpYWJsZXMKUmVxdWlyZWQgRW52aXJvbm1lbnQgVmFyaWFibGVzOgoKKiBEQl9BRERSIC0gUHJvdmlkZSB0aGUgY29tcGxldGUgSkRCQyBVUkkgZm9yIHRoZSBEQi4KKiBLRVlDTE9BS19TRVJWRVIgLSBUaGUgVVJJIHRvIHRoZSBhdXRoIGVuZHBvaW50IG9mIHRoZSBLZXljbG9hayBTZXJ2ZXIuCiogS0VZQ0xPQUtfU0VDUkVUIC0gVGhlIGtleWNsb2FrIGNsaWVudCBTZWNyZXQKCk9wdGlvbmFsIEVudmlyb25tZW50IFZhcmlhYmxlczoKCiogREJfVVNFUiAtIERCIHVzZXIgdXNlcm5hbWUuIChEZWZhdWx0IC0gcG9zdGdyZXMpCiogREJfUEFTUyAtIERCIHVzZXIgcGFzc3dvcmQuIChEZWZhdWx0IC0gbXlzZWNyZXRwYXNzd29yZCkKKiBLRVlDTE9BS19SRUFMTSAtIFRoZSBrZXljbG9hayByZWFsbSBmb3IgdGhlIGFwcGxpY2F0aW9uLiAoRGVmYXVsdCAtIGVjb21tZXJjZSkKKiBLRVlDTE9BS19SRVNPVVJDRSAtIFRoZSBjbGllbnQgY3JlYXRlZCBmb3IgdGhlIGFwcGxpY2F0aW9uLiAoRGVmYXVsdCAtIGxvZ2luLWFwcCkKCiMjIyBTdGVwcyB0byBydW4KMS4gQ2xvbmUgdGhlIHJlcG8gb3IgZG93bmxvYWQgdGhlIHNvdXJjZQoyLiBVc2UgYG12biBzcHJpbmctYm9vdDpydW5gCiAgICBvcgogICBgbXZudyBzcHJpbmctYm9vdDpydW5gCiAgICBmcm9tIHRoZSBiYXNlIGRpcmVjdG9yeSBvZiB0aGUgcHJvamVjdCB0byBydW4gdGhlIHByb2plY3QKICAgCiMjIyBTdGVwcyB0byBkZXBsb3kKVXNpbmcgRG9ja2VyOgoxLiBCdWlsZCB0aGUgcHJvamVjdCB1c2luZyBgbXZuIHBhY2thZ2VgIG9yIGBtdm53IHBhY2thZ2VgCjIuIEJ1aWxkIGEgZG9ja2VyIGltYWdlIHVzaW5nOgpgYGAKZG9ja2VyIGJ1aWxkIC10IDxpbWFnZS1uYW1lPjo8dGFnPiAuCmBgYAozLiBSdW4gdGhlIGltYWdlIHVzaW5nOgpgYGAKZG9ja2VyIHJ1biAtcCA4MDgwOjgwODAgLWUgREJfQUREUj08dmFsdWU+IC1lIEtFWUNMT0FLX1NFUlZFUj08dmFsdWU+IC1lIEtFWUNMT0FLX1NFQ1JFVD08dmFsdWU+IC1kIDxpbWFnZS1uYW1lPjo8dGFnPgpgYGAKCkRlcGxveSBqYXIgZmlsZToKMS4gQnVpbGQgdGhlIHByb2plY3QgdXNpbmcgYG12biBwYWNrYWdlYCBvciBgbXZudyBwYWNrYWdlYAoyLiBDb3B5IHRoZSAuamFyIGdlbmVyYXRlZCBmcm9tIHRoZSAvdGFyZ2V0IGZvbGRlciB0byB5b3VyIGRlc2lyZWQgbG9jYXRpb24uCjMuIFJ1biB0aGUgcHJvZ3JhbSB1c2luZyBgamF2YSAtamFyIDxmaWxlbmFtZT4uamFyYAo= readmeEtag: '"cdb4f5ffd4c34ad4e91b0c85c4cd6dc772b28878"' readmeLastModified: Sat, 26 Aug 2023 12:41:27 GMT repositoryId: 449337717 description: null created: '2022-01-18T15:22:52Z' updated: '2022-01-23T11:37:25Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: cant-code logo: https://avatars.githubusercontent.com/u/38918278?v=4 repoEtag: '"70545c3df3e92e22dfb9a6c3d77521ad7c357e6adcadeb32dea8af424b3a3508"' repoLastModified: Sun, 23 Jan 2022 11:37:25 GMT foundInMaster: true category: Server Implementations id: 4dae94b5dd42661aa1c4c79ef6282ff0 - source: openapi3 tags repository: >- https://github.com/marinimau/public_administration_blockchain_certified_document_sharing v3: true repositoryMetadata: base64Readme: >- WyFbTGljZW5zZTogQ0MwLTEuMF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWNlbnNlLUNDMCUyMDEuMC1saWdodGdyZXkuc3ZnKV0oaHR0cDovL2NyZWF0aXZlY29tbW9ucy5vcmcvcHVibGljZG9tYWluL3plcm8vMS4wLykKCiMgUHVibGljIEFkbWluaXN0cmF0aW9uIEJsb2NrY2hhaW4gQ2VydGlmaWVkIERvY3VtZW50IFNoYXJpbmcKCkEgZEFwcCB0byBkaXN0cmlidXRlIGJsb2NrY2hhaW4gY2VydGlmaWVkIGRvY3VtZW50cyBmcm9tIFBBcyB0byBjaXRpemVucy4gQWxsIHRoZSBmZWF0dXJlcyBhcmUgYWNjZXNzaWJsZSB2aWEgUkVTVCBhcGkuCgpDaXRpemVuIHJlbGF0ZWQgZmVhdHVyZXMgYXJlIGFjY2Vzc2libGUgYWxzbyB2aWEgZnJvbnRlbmQuCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCmBgYGJhc2gKJCBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL21hcmluaW1hdS9wdWJsaWNfYWRtaW5pc3RyYXRpb25fYmxvY2tjaGFpbl9jZXJ0aWZpZWRfZG9jdW1lbnRfc2hhcmluZy5naXQKJCBjZCBwdWJsaWNfYWRtaW5pc3RyYXRpb25fYmxvY2tjaGFpbl9jZXJ0aWZpZWRfZG9jdW1lbnRfc2hhcmluZwokIHB5dGhvbjMgLW0gdmVudiB2ZW52CiQgc291cmNlIHZlbnYvYmluL2FjdGl2YXRlCiQgcGlwMyBpbnN0YWxsIC1yIHJlcXVpcmVtZW50cy50eHQKJCBweXRob24zIG1hbmFnZS5weSBtYWtlbWlncmF0aW9ucyB1c2VyCiQgcHl0aG9uMyBtYW5hZ2UucHkgbWFrZW1pZ3JhdGlvbnMgZG9jdW1lbnQKJCBweXRob24zIG1hbmFnZS5weSBtYWtlbWlncmF0aW9ucyB0cmFuc2FjdGlvbgokIHB5dGhvbjMgbWFuYWdlLnB5IG1pZ3JhdGUKJCBweXRob24zIG1hbmFnZS5weSBydW5zZXJ2ZXIKYGBgCgpOb3RlIHRoYXQgeW91IHdpbGwgbmVlZCB0byBjcmVhdGUgYSBzdXBlciB1c2VyICh1c2luZyAiY3JlYXRlc3VwZXJ1c2VyIiBjb21tYW5kKSBhbmQgdGhlbiB5b3Ugd2lsbCBuZWVkIHRvIGNyZWF0ZSBQQSwgb3BlcmF0b3IsIGFuZCBjaXRpemVuIHVzaW5nIHRoZSBhZG1pbiBpbnRlcmZhY2UgIGF0IGBgYGh0dHA6Ly8xMjcuMC4wLjE6ODAwMC9hZG1pbi9gYGAuCgpJTVBPUlRBTlQ6IHVzZSB2YWxpZCBFdGhlcnVtIGNyZWRlbnRpYWwgZm9yIHRoZSBvcGVyYXRvciwgYW5kIGVuc3VyZSB0aGF0IGhlIGFzIHNvbWUgRXRoZXIgaW4gaGlzIGJhbGFuY2UgKGRlZmF1bHQgdGVzdG5ldCBpcyBSaW5rZWJ5LCB5b3UgY2FuIGNoYW5nZSBpdCBpbiBzZXR0aW5ncykKCiMjIFRlc3QgdGhlIHByb2plY3QKCmBgYGJhc2gKJCBweXRob24zIG1hbmFnZS5weSB0ZXN0CmBgYAoKCiMjIEZlYXR1cmVzCgoqIENpdGl6ZW5zIGFuZCBQQSBvcGVyYXRvcnMgbWFuYWdlbWVudAoqIERvY3VtZW50IGNyZWF0aW9uIGFuZCB1cGRhdGUgKGJ5IHZlcnNpb24gY3JlYXRpb24pCiogUHVibGljIGFuZCBwcml2YXRlIGRvY3VtZW50czogaWYgYSBkb2N1bWVudCBpcyBwcml2YXRlIGNpdGl6ZW4gbXVzdCBoYXZlIHJlYWQgcGVybWlzc2lvbgoqIERvY3VtZW50IHZlcnNpb24gaW5oZXJpdHMgcGVybWlzc2lvbnMgZnJvbSBkb2N1bWVudAoqIFBhIG1hbmFnZW1lbnQ6IGFuIG9wZXJhdG9yIGNhbiB1cGRhdGUgKGNyZWF0ZSB2ZXJzaW9ucykgb25seSBmb3IgZG9jdW1lbnRzIG93bmVkIGJ5IGhpcyBQQSwgYW4gb3BlcmF0b3IgY2FuIHZpZXcgb25seSBwdWJsaWMgZG9jdW1lbnRzIG9yIHByaXZhdGUgZG9jdW1lbnRzIChpZiB0aGV5IGFyZSBvd25lZCBieSBoaXMgUEEpCiogQXV0b21hdGljIGRvY3VtZW50IFNDIGFuZCBkb2N1bWVudCB2ZXJzaW9uICh0cmFuc2FjdGlvbiBpbiB0aGUgZG9jdW1lbnQgU0MpIGRlcGxveQoqIEF1dG9tYXRpYyBmaW5nZXJwcmludCB2YWxpZGF0aW9uIGluIGRvd25sb2FkIHBoYXNlLgoqIFNDIG9wZXJhdG9yIGF1dGhlbnRpY2F0aW9uIGJhc2VkIG9uIHdoaXRlbGlzdAoqIENpdGl6ZW4gY2FuIGFkZCB0byBmYXZvcml0ZXMgZG9jdW1lbnRzCgoKIyMgT3BlbkFwaSBkb2N1bWVudGF0aW9uCgpUaGUgZG9jdW1lbnRhdGlvbiBpcyBhdmFpbGFibGUgYXQgYGBgaHR0cDovLzEyNy4wLjAuMTo4MDAwL2RvY3VtZW50YXRpb24vYGBgLgoKTm90ZSB0aGF0IG9ubHkgZW5kcG9pbnRzIGFjY2Vzc2libGUgd2l0aCBjdXJyZW50IGF1dGhvcml6YXRpb24gYXJlIHNob3duOgoqIGlmIHlvdSBhcmUgdW5hdXRob3JpemVkIHlvdSBzZWUgb25seSBwdWJsaWMgZW5kcG9pbnRzCiogaWYgeW91IGFyZSBhdXRob3JpemVkIGFzIGNpdGl6ZW4geW91IHNlZSBwdWJsaWMgKyBjaXRpemVuIGVuZHBvaW50cwoqIGlmIHlvdSBhcmUgYXV0aG9yaXplZCBhcyBwYSBvcGVyYXRvciB5b3Ugc2VlIHB1YmxpYyArIG9wZXJhdG9yIGVuZHBvaW50cwoKIyMgQXV0aG9yCgoqIFtNYXVybyBNYXJpbmldKGh0dHBzOi8vZ2l0aHViLmNvbS9tYXJpbmltYXUpCg== readmeEtag: '"1dee51232b6b7eb25a1c85828f2a0e9f91f1b25e"' readmeLastModified: Tue, 03 Aug 2021 15:08:03 GMT repositoryId: 378353674 description: >- A dApp to distribute blockchain certified documents from PAs to citizens. All the features are accessible via REST api. created: '2021-06-19T07:39:48Z' updated: '2021-08-03T15:08:07Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: marinimau logo: https://avatars.githubusercontent.com/u/37106877?v=4 license: NOASSERTION repoEtag: '"1178f4453d5a864f9081d58492598046834d74adcf5f085b7a4cf208a395eb4c"' repoLastModified: Tue, 03 Aug 2021 15:08:07 GMT foundInMaster: true category: - Parsers - Server Implementations id: e4e2b5afe9011c1dd92de5fc3f9843ac - source: openapi3 tags repository: https://github.com/phuntime/aws-lambda-openapi v3: true repositoryMetadata: base64Readme: >- IyBhd3MtbGFtYmRhLW9wZW5hcGkKQWxsIEFXUyBMYW1iZGEgT3BlbkFQSSBmaWxlcyBjb21iaW5lZC4gCgoKU3dhZ2dlciBVSTogaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby8/dXJsPWh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9waHVudGltZS9hd3MtbGFtYmRhLW9wZW5hcGkvbWFpbi9vcGVuYXBpLnlhbWwKCiMjIFNvdXJjZXM6Ci0gaHR0cHM6Ly9kb2NzLmF3cy5hbWF6b24uY29tL2xhbWJkYS9sYXRlc3QvZGcvcnVudGltZXMtYXBpLmh0bWwKLSBodHRwczovL2RvY3MuYXdzLmFtYXpvbi5jb20vbGFtYmRhL2xhdGVzdC9kZy9ydW50aW1lcy1leHRlbnNpb25zLWFwaS5odG1sCi0gaHR0cHM6Ly9kb2NzLmF3cy5hbWF6b24uY29tL2xhbWJkYS9sYXRlc3QvZGcvcnVudGltZXMtbG9ncy1hcGkuaHRtbAoKIyMgTGljZW5zZQoKaHR0cHM6Ly9jcmVhdGl2ZWNvbW1vbnMub3JnL2xpY2Vuc2VzL2J5LXNhLzQuMC8K readmeEtag: '"35248c14072202ae8e76ba1d2343156afb65d01e"' readmeLastModified: Mon, 09 Aug 2021 22:54:08 GMT repositoryId: 394451824 description: All AWS Lambda OpenAPI files combined created: '2021-08-09T22:06:34Z' updated: '2021-08-09T22:54:11Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: phuntime logo: https://avatars.githubusercontent.com/u/66830795?v=4 repoEtag: '"855f6a34d843270493c9bf33b84f75077dd106e7384f20f90d3ae4d8aa372879"' repoLastModified: Mon, 09 Aug 2021 22:54:11 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 93e78feb1ebe295e4ee84b1f0a83d009 - source: openapi3 tags repository: https://github.com/rosario-fiorella/vuejs-calendar v3: true repositoryMetadata: base64Readme: >- IyB2dWVqcy1jYWxlbmRhcgoKIyMgUHJvamVjdCBzZXR1cApgYGAKbnBtIGluc3RhbGwKYGBgCnVzZSBub2RlanMgdmVyc2lvbiAxNiBMVFMKCiMjIyBDb21waWxlcyBhbmQgaG90LXJlbG9hZHMgZm9yIGRldmVsb3BtZW50CmBgYApucG0gcnVuIHNlcnZlCmBgYAoKIyMjIENvbXBpbGVzIGFuZCBtaW5pZmllcyBmb3IgcHJvZHVjdGlvbgpgYGAKbnBtIHJ1biBidWlsZApgYGAKCiMjIyBMaW50cyBhbmQgZml4ZXMgZmlsZXMKYGBgCm5wbSBydW4gbGludApgYGAKCiMjIyBDdXN0b21pemUgY29uZmlndXJhdGlvbgpTZWUgW0NvbmZpZ3VyYXRpb24gUmVmZXJlbmNlXShodHRwczovL2NsaS52dWVqcy5vcmcvY29uZmlnLykuCgojIyBBUEkgU2VydmljZXMKIyMjIE1vY2sgQVBJIHJlcXVlc3QvcmVzcG9uc2UgbW9kZWwKVGhlIGRlZmF1bHQgc2V0dGluZ3MgdXNlIGEgc2ltdWxhdGVkIGFwaSByZXNwb25zZSB0ZW1wbGF0ZSwgaWYgeW91IHdhbnQgdG8gdXNlIGEgcmVhbCBBUEkgcmVxdWVzdC9yZXNwb25zZSB5b3UgbmVlZCB0byBlZGl0IHRoZSAiLmVudiIgZmlsZSBhbmQgZW50ZXIgdGhlIGFwaSBVUkwgd2hpY2ggaW1wbGVtZW50cyB0aGUgZGF0YSBtb2RlbCBzdWdnZXN0ZWQgYnkgdGhlIGZpbGUgW29wZWFuYXBpM10oZG9jL29wZW5hcGkuMy4wLjIueW1sKS4gCj4gVGhpcyBwcm9qZWN0IGRvZXMgbm90IGltcGxlbWVudCBzZXJ2ZXIgc2lkZSBiYWNrZW5kIGFwaSBzZXJ2aWNlcyAKCiMjIyBFbnZpcm9ubWVudApWVUVfQVBQX05PREVfRU5WOiBzZWUgInNyYy9jb21tb24vZW51bS5qcyIsIHVzZSAiZGVtbyIgdG8gdXNlIG1vY2sgYXBpCgpWVUVfQVBQX1RJVExFPWBgYHN0cmluZ2BgYAoKVlVFX0FQUF9BUElfQkFTRV9VUkw9YGBgdXJsYGBgIChleGFtcGxlOiBodHRwOi8vbG9jYWxob3N0L2FwaS8pCgpWVUVfQVBQX0FQSV9CT09LSU5HPWBgYHN0cmluZ2BgYCAoZXhhbXBsZTogYm9va2luZykKClZVRV9BUFBfQVBJX0NPTlRFTlRTPWBgYHN0cmluZ2BgYCAoZXhhbXBsZTogY29udGVudHMpCgpWVUVfQVBQX0FQSV9EQVRBPWBgYHN0cmluZ2BgYCAoZXhhbXBsZTogYXBwKQoKVlVFX0FQUF9BUElfRU5USVRJRVM9YGBgc3RyaW5nYGBgIChleGFtcGxlOiBlbnRpdGllcykKClZVRV9BUFBfTElOS19QUklWQUNZX1RFUk09YGBgdXJsYGBgCgpWVUVfQVBQX0xJTktfUFJJVkFDWV9DT05ESVRJT049YGBgdXJsYGBgCgpWVUVfQVBQX0xJTktfUFJJVkFDWV9QT0xJQ1k9YGBgdXJsYGBgCgpWVUVfQVBQX0FQSV9QRVJfUEFHRT1gYGBudW1iZXJgYGAKClZVRV9BUFBfQVBJX0tFWT1gYGBzdHJpbmdgYGAKCiMjIEljb25zClttZGldKGh0dHBzOi8vcGljdG9ncmFtbWVycy5naXRodWIuaW8vQG1kaS9mb250LzUuNC41NS8pCgpbbWF0ZXJpYWwtZGVzaWduLWljb25zLWljb25mb250XShodHRwczovL2pvc3NlZi5naXRodWIuaW8vbWF0ZXJpYWwtZGVzaWduLWljb25zLWljb25mb250LykKCiMjIFBvc3QgYnVpbGQKW2NsaS52dWVqcy5vcmddKGh0dHBzOi8vY2xpLnZ1ZWpzLm9yZy9ndWlkZS9kZXBsb3ltZW50Lmh0bWwpCgojIyBDSEFOR0UgVEhFTUUgQ09MT1IKRWRpdCBmaWxlIFtjb2xvcnNdKHNyYy9jb21tb24vY29sb3JzLmpzKQoKIyMgREVNTyBURU1QTEFURQohWzEtaG9tZS1kaWFsb2ddKGRvYy9kZW1vLzEtaG9tZS1kaWFsb2cucG5nKQohWzItaG9tZV0oZG9jL2RlbW8vMi1ob21lLnBuZykKIVszLWhvbWUtc2VsZWN0ZWRdKGRvYy9kZW1vLzMtaG9tZS1zZWxlY3RlZC5wbmcpCiFbNC1ob21lLWJvb2tpbmddKGRvYy9kZW1vLzQtaG9tZS1ib29raW5nLnBuZykK readmeEtag: '"e7fb76e1fb46db26b1b7d0e85bf979a501e2742b"' readmeLastModified: Wed, 09 Aug 2023 11:03:08 GMT repositoryId: 400177668 description: >- Front-end booking system template built with VueJs and VuetifyJs - Single Page Application (SPA), dummy data used to simulate API response. created: '2021-08-26T13:24:32Z' updated: '2025-11-07T05:29:02Z' language: Vue archived: false stars: 0 watchers: 1 forks: 1 owner: rosario-fiorella logo: https://avatars.githubusercontent.com/u/41728059?v=4 license: MIT repoEtag: '"1a5d2cc889676b6abe8a18cdcc6723e8080d186f1bdf8fd72d24e64f92a2d546"' repoLastModified: Fri, 07 Nov 2025 05:29:02 GMT foundInMaster: true category: Testing id: e5071ac76f7c1501aeddbdd50c3cc6d1 oldLocations: - https://github.com/rosario-fiorella/calendar-fe - source: openapi3 tags repository: https://github.com/devalurum/messenger-system v3: true id: eddf19a5eabf7da6555b7ecd8143127d repositoryMetadata: base64Readme: >- IyBNZXNzZW5nZXIgUkVTVGZ1bCBBUEkKCtCh0LXRgNCy0LXRgNC90LDRjyDRh9Cw0YHRgtGMINC00LjQv9C70L7QvNC90L7QuSDRgNCw0LHQvtGC0Ysg0L/QviDRgtC10LzQtSAKItCg0LDQt9GA0LDQsdC+0YLQutCwINGB0L/RgNCw0LLQvtGH0L3Qvi3QutC+0LzQvNGD0L3QuNC60LDRhtC40L7QvdC90L7QuSDRgdC40YHRgtC10LzRiyDQtNC70Y8g0L/QsNGG0LjQtdC90YLQvtCyINGB0YLQsNGG0LjQvtC90LDRgNCwLiIKCiMjIyDQodGC0Y3QuiDRgtC10YXQvdC+0LvQvtCz0LjQuQotIEphdmEgMTEKLSBTcHJpbmcgQm9vdAotIFNwcmluZyBTZWN1cml0eSAoSldUKQotIFNwcmluZyBEYXRhIEpQQQotIFNwcmluZyBXZWIKLSBTd2FnZ2VyIChPcGVuQVBJIDMuMCkKLSBQb3N0Z3Jlc3FsICgrUG9zdEdpcykKLSBEb2NrZXIKLSBNYXBTdHJ1Y3QKLSBMb21ib2sKLSBHcmFkbGUKCjwhLS0KIyMg0KHQsdC+0YDQutCwINC/0YDQuNC70L7QttC10L3QuNGPCmBgYHNoZWxsIHNjcmlwdAojINCh0LrQu9C+0L3QuNGA0L7QstCw0YLRjCDQv9GA0L7QtdC60YIg0Log0YHQtdCx0LUKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9kZXZhbHVydW0vbWVzc2VuZ2VyLXN5c3RlbS5naXQKCiMg0L/QvtC00L3Rj9GC0Ywg0LrQvtC90YLQtdC50L3QtdGAIGMgUG9zdGdyZXNxbCgrUG9zdGdpcykg0LggUGdBZG1pbiAKZG9ja2VyLWNvbXBvc2UgdXAgLWQKCiMg0LfQsNCz0YDRg9C20LDQtdGCIGdyYWRsZSB3cmFwcGVyCmdyYWRsZXcgd3JhcHBlcgoKIyDRgdCx0L7RgNC60LAg0L/RgNC+0LXQutGC0LAKZ3JhZGxldyBjbGVhbiBidWlsZCAKCiMg0LfQsNC/0YPRgdC6IFNwcmluZyDRgdC10YDQstC40YHQsApqYXZhIC1qYXIgYnVpbGQvbGlicy9tZXNzZW5nZXItc3lzdGVtLmphciAKYGBgCi0tPgojIyBEZXBsb3ltZW50CtCf0YDQuNC70L7QttC10L3QuNC1INC30LDQtNC10L/Qu9C+0LXQvdC+INC90LAgW2hlcm9rdV0oaHR0cHM6Ly9tZXNzZW5nZXItcmVzdGZ1bC5oZXJva3VhcHAuY29tL21lc3Nlbmdlci1zeXN0ZW0vc3dhZ2dlci11aS9pbmRleC5odG1sCikg0L/QviDQsNC00YDQtdGB0YM6CgpodHRwczovL21lc3Nlbmdlci1yZXN0ZnVsLmhlcm9rdWFwcC5jb20vbWVzc2VuZ2VyLXN5c3RlbS9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKIyMg0KHQsdC+0YDQutCwINC/0YDQuNC70L7QttC10L3QuNGPCmBgYHNoZWxsIHNjcmlwdAojINCh0LrQu9C+0L3QuNGA0L7QstCw0YLRjCDQv9GA0L7QtdC60YIsINC70LjQsdC+INC/0YDQvtGB0YLQviDRgdC60LDRh9Cw0YLRjCDRhNCw0LnQuyAiZG9ja2VyLWNvbXBvc2UueW1sIgpnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2RldmFsdXJ1bS9tZXNzZW5nZXItc3lzdGVtLmdpdAoKIyDQv9C+0LTQvdGP0YLRjCDQutC+0L3RgtC10LnQvdC10YDRiyBjIFBvc3RncmVTUUwoK1Bvc3RnaXMpINC4IFNwcmluZyBCb290IApkb2NrZXItY29tcG9zZSB1cCAtZApgYGAK0J7RgtC60YDRi9GC0Ywg0LIg0LHRgNCw0YPQt9C10YDQtSBTd2FnZ2VyOiBodHRwOi8vbG9jYWxob3N0OjgwODAvbWVzc2FuZ2VyLXN5c3RlbS9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKCiMjIE9wZW5BUEkg0L7Qv9C40YHQsNC90LjQtQoxLiDQntGC0LrRgNC+0LnRgtC1INCw0LTRgNC10YEg0LIg0LHRgNCw0YPQt9C10YDQtSBodHRwOi8vbG9jYWxob3N0OjgwODAvbWVzc2FuZ2VyLXN5c3RlbS9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKMi4g0JLRi9C/0L7Qu9C90LjRgtC1INGA0LXQs9C40YHRgtGA0LDRhtC40Y4g0L/QvtC70YzQt9C+0LLQsNGC0LXQu9GPIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9tZXNzZW5nZXItc3lzdGVtL3N3YWdnZXItdWkvaW5kZXguaHRtbCMvQXV0aGVudGljYXRpb24vcmVnaXN0ZXIKMy4g0KHQutC+0L/QuNGA0YPQudGC0LUg0YLQvtC60LXQvSDQuCDQsNCy0YLQvtGA0LjQt9C40YDRg9C10YLQtdGB0Ywg0LIgU3dhZ2dlciBVSS4g0J/QvtGB0LvQtSDRh9C10LPQviAqKnRva2VuKiog0LHRg9C00LXRgiDQstGB0YLQsNCy0LvQtdC9INCw0LLRgtC+0LzQsNGC0LjRh9C10YHQutC4INCyINC30LDQv9GA0L7RgdGLLgo0LiBwZ0FkbWluIChHVUkg0LTQu9GPIHBvc3RncmVzcWwpIGh0dHA6Ly9sb2NhbGhvc3Q6NTA1MAogICAxLiBsb2dpbjogYWRtaW5AYWRtaW4uY29tCiAgIDIuIHBhc3N3b3JkOiBhZG1pbgogICAgICAxLiBob3N0OiBob3N0LmRvY2tlci5pbnRlcm5hbCAo0YDQtdCz0LjRgdGC0YDQsNGG0LjRjyDQsdCw0LfRiyDQtNCw0L3QvdGL0YUpCiog0J/QvtC70Y8gdGltZSwgaWQsIHNlbmRlciDQv9GA0LggcG9zdC9wYXRjaCDQt9Cw0L/RgNC+0YHQsNGFINC6INGB0YPRidC90L7RgdGC0Y/QvCDQuNCz0L3QvtGA0LjRgNGD0Y7RgtGB0Y8sINGCLtC6INCz0LXQvdC10YDQuNGA0YPRjtGC0YHRjyDQvdCwINGB0YLQvtGA0L7QvdC1INGB0LXRgNCy0LXRgNCwLgoqINCk0L7RgNC80LDRgiDQstGA0LXQvNC10L3QuCDQsiBPcGVuQVBJINCz0LXQvdC10YDQuNGA0YPRjtGC0YHRjyDQvdC10L/RgNCw0LLQuNC70YzQvdC+ICjQv9C+0LrQsCDQvdC1INC/0L7QvdGP0YLQvdC+INC60LDQuiDQv9C+0LzQtdC90Y/RgtGMINC/0LDRgtGC0LXRgNC9KSwg0L/QvtGN0YLQvtC80YMg0L/QvtC70Y8gdGltZSDQu9GD0YfRiNC1INGD0LTQsNC70Y/RgtGMINC/0YDQuCDQvtGC0L/RgNCw0LLQutC1INC30LDQv9GA0L7RgdC+0LIuCiMjIyBUb2RvOgotINCg0LDQt9C+0LHRgNCw0YLRjNGB0Y8sINC/0L7Rh9C10LzRgyDQvdC1INGA0LDQsdC+0YLQsNC10YIg0LDQvdC90L7RgtCw0YbQuNGPIFJvbGVzQWxsb3dlZCDQsiDQutC+0L3RgtGA0L7Qu9C70LXRgNCw0YUuCi0g0J3QsNC/0LjRgdCw0YLRjCDRgtC10YHRgtGLLiDQn9C+0L/RgNC+0LHQvtCy0LDRgtGMIFRlc3Rjb250YWluZXJzINC70LjQsdC+IEgyR2lzINC00LvRjyDRgtC10YHRgtC40YDQvtCy0LDQvdC40Y8g0YDQtdC/0L7Qt9C40YLQvtGA0LjQtdCyLgotINCg0LDQt9C00LXQu9C40YLRjCBEVE8g0L3QsCBSZXF1ZXN0cyDQuCBSZXNwb25zZXMgKNCS0L7Qt9C80L7QttC90L4pLCDQtNC70Y8g0YPQtNCw0LvQtdC90LjRjyDQvdC10LLQsNC70LjQtNC90YvRhSDQv9C+0LvQtdC5INC/0YDQuCDQt9Cw0L/RgNC+0YHQsNGFLgotINCg0LDQt9C+0LHRgNCw0YLRjNGB0Y8g0L/QvtC00YDQvtCx0L3QtdC1INGBINC80LDQv9C/0LjQvdCz0L7QvCBEVE8g0YfQtdGA0LXQtyBNYXBTdHJ1Y3QrTG9tYm9rLgotINCg0LXRhNCw0LrRgtC+0YDQuNC90LMuCg== readmeEtag: '"920eff866933bb7d04bf03c6e367e9f25bb3fbad"' readmeLastModified: Sun, 19 Jun 2022 06:53:31 GMT repositoryId: 489871474 description: RESTful API для функционирования чата. created: '2022-05-08T07:05:35Z' updated: '2022-05-17T03:29:58Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: devalurum logo: https://avatars.githubusercontent.com/u/33903810?v=4 repoEtag: '"b0d1dc383f2dd1b72c6c815f38c1e7dfbcea623f2b4ebde9937e487a2550e3cf"' repoLastModified: Tue, 17 May 2022 03:29:58 GMT category: Low-level Tooling foundInMaster: true - source: openapi3 tags repository: https://github.com/borjatur/money-master-api v3: true id: 468f50c95f456880d3c26b72d0a577b8 repositoryMetadata: base64Readme: >- IyBtb25leS1tYXN0ZXItYXBpCgpCYWNrZW5kIHNlcnZpY2UgZm9yIE1vbmV5IE1hc3RlciwgYSBzaWRlIHByb2plY3QgY3JlYXRlZCBieSBtZSAoYm9yamEudHVyQGdtYWlsKSBmb3IgbGVhcm5pbmcgcHVycG9zZXMuIEJ1aWx0IGZvbGxvd2luZyAiQ2xlYW4gQXJjaGl0ZWN0dXJlIiBwcmluY2lwbGVzIGRlc2NyaWJlZCBieSBSb2JlcnQgQy4gTWFydGluIChha2EgVW5jbGUgQm9iKSwgaWYgeW91IHdhbnQgdG8ga25vdyBtb3JlIHNlZSBodHRwczovL2dpdGh1Yi5jb20vYm9yamF0dXIvY2xlYW4tYXJjaGl0ZWN0dXJlLWZhc3RpZnktbW9uZ29kYiBhbmQgaHR0cHM6Ly9ib3JqYXR1ci5jb20vMjAyMy8wMy8wNy95ZXQtYW5vdGhlci12aXNpb24tb2YtY2xlYW4tYXJjaGl0ZWN0dXJlLyB0aGF0IHdhcyB1c2VkIGFzIGJhc2UgdGVtcGxhdGUgZm9yIGJ1aWxkaW5nIHRoaXMgcHJvamVjdC4KCiMjIFdoYXQgY2FuIHlvdSBmaW5kIGluIHRoaXMgcHJvamVjdD8KCi0gW3hdICJDbGVhbiBBcmNoaXRlY3R1cmUiIHByaW5jaXBsZXMgaW4gcHJhY3RpY2UKLSBbeF0gQnVpbHQgaW4gVHlwZXNjcmlwdAotIFt4XSBCbGF6aW5nIGZhc3QgTm9kZS5qcyBmcmFtZXdvcmssIEZhc3RpZnkKLSBbeF0gRmFuY3kgYXV0byBnZW5lcmF0ZWQgZG9jcyB1c2luZyBPcGVuQVBJIHNwZWNpZmljYXRpb24gdjMKLSBbeF0gTW9uZ29EQiB3aXRoIE1vbmdvb3NlCi0gW3hdIENvbmZpZ3VyZWQgd2l0aCBFc2xpbnQgYW5kIEplc3Qgb3V0IHRoZSBib3gKLSBbeF0gRWFzeSB0byByZWFzb24gYWJvdXQgZm9sZGVyIHN0cnVjdHVyZQotIFsgXSBUZXN0aW5nIChXSVApCi0gWyBdIEF1dGhlbnRpY2F0aW9uL0F1dGhvcml6YXRpb24gKFdJUCkKCiMjIExpdmUgRGVtbwpWaXNpdCBodHRwczovL21vbmV5LW1hc3Rlci1hcGkuYm9yamF0dXIuY2xvdWQvZG9jcyBmb3IgYSBsaXZlIGRlbW8gb2YgdGhpcyBwcm9qZWN0LCB5b3UgY2FuIGFsc28gZmluZCB0aGUgZnJvbnRlbmQgYXBwbGljYXRpb24gYXZhaWxhYmxlIGF0IGh0dHBzOi8vbW9uZXktbWFzdGVyLmJvcmphdHVyLmNsb3VkIGFzIGEgY29uc3VtZXIgb2YgdGhpcyBBUEkKCkRlbW8gc2VydmljZXMgYXJlIHJ1bm5pbmcgb24gbXkgb3duIHBlcnNvbmFsIGt1YmVybmV0ZXMgY2x1c3RlciBkZXBsb3llZCBpbjoKClshW0RpZ2l0YWxPY2VhbiBSZWZlcnJhbCBCYWRnZV0oaHR0cHM6Ly93ZWItcGxhdGZvcm1zLnNmbzIuZGlnaXRhbG9jZWFuc3BhY2VzLmNvbS9XV1cvQmFkZ2UlMjAyLnN2ZyldKGh0dHBzOi8vd3d3LmRpZ2l0YWxvY2Vhbi5jb20vP3JlZmNvZGU9ZTNhMjdkZWVhMmFjJnV0bV9jYW1wYWlnbj1SZWZlcnJhbF9JbnZpdGUmdXRtX21lZGl1bT1SZWZlcnJhbF9Qcm9ncmFtJnV0bV9zb3VyY2U9YmFkZ2UpCgpUaGVyZSBpcyBzb21lIGRlbW8gZGF0YSBhdmFpbGFibGUgdGhhdCB3aWxsIGJlIHJlc3RvcmVkIGluIGEgc2NoZWR1bGVkIGJhc2UuCgojIyBHZXR0aW5nIHN0YXJ0ZWQKCiogZG9ja2VyLWNvbXBvc2UgdXAgLS1mb3JjZS1yZWNyZWF0ZSAtLWJ1aWxkCgpPciBhbHRlcm5hdGl2ZWx5IGEgbW9uZ29kYiBpbnN0YW5jZSBydW5uaW5nIHNvbWV3aGVyZSwgdXBkYXRlIHRoZSBjb25uZWN0aW9uIHVybCBhY2NvcmRpbmdseSBhbmQ6CgoqIG5wbSBpbnN0YWxsCiogbnBtIHJ1biBidWlsZCAmJiBucG0gc3RhcnQKKiB2aXNpdCBodHRwOi8vbG9jYWxob3N0OjUwNTAvZG9jcyBhbmQgZW5qb3kh readmeEtag: '"e7e16cb40572a8d84eb032e0fbec49c3bb86878c"' readmeLastModified: Thu, 16 Nov 2023 17:59:57 GMT repositoryId: 613068520 description: >- rest api following clean architecture principles for Money Master, side project by Borja Tur created: '2023-03-12T19:40:13Z' updated: '2023-04-25T17:00:19Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: borjatur logo: https://avatars.githubusercontent.com/u/15380585?v=4 repoEtag: '"8b70ea42234b889f426aa335b542b245bc9b9d157c3ddc3e130b25307347e4cb"' repoLastModified: Tue, 25 Apr 2023 17:00:19 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/thenriquedb/todo-list-with-spring v3: true id: 95e8cced8900d0116e77c182daa46c25 repositoryMetadata: base64Readme: >- IVtpbWcucG5nXShoZWFkZXIucG5nKQoKIyBUb2RvIExpc3QgUkVTVCBBUEkKIVtKYXZhXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2phdmEtJTIzRUQ4QjAwLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289b3BlbmpkayZsb2dvQ29sb3I9d2hpdGUpIVtTcHJpbmddKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvc3ByaW5nLSUyMzZEQjMzRi5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPXNwcmluZyZsb2dvQ29sb3I9d2hpdGUpIVtTd2FnZ2VyXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlLy1Td2FnZ2VyLSUyM0Nsb2p1cmU/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPXN3YWdnZXImbG9nb0NvbG9yPXdoaXRlKSFbUmVuZGVyXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1JlbmRlci0lNDZFM0I3LnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289cmVuZGVyJmxvZ29Db2xvcj13aGl0ZSkKCkVzdGUgcHJvamV0byBmb2kgZGVzZW52b2x2aWRvIGNvbW8gcGFydGUgZG8gY3Vyc28gZ3JhdHVpdG8gZGUgSmF2YSBvZmVyZWNpZG8gcGVsYSBSb2NrZXRzZWF0LiBUcmF0YS1zZSBkZSB1bWEgQVBJIFJlc3QgcGFyYSBnZXJlbmNpYW1lbnRvIGRlIGxpc3RhIGRlIHRhcmVmYXMsIGNvbnN0cnXDrWRhIGVtIEphdmEgY29tIG8gZnJhbWV3b3JrIFNwcmluZyBCb290LgoKCiMjIENhcmFjdGVyw61zdGljYXMgZG8gUHJvamV0bwotICoqU2VwYXJhw6fDo28gZGUgUmVzcG9uc2FiaWxpZGFkZXMqKjogSW1wbGVtZW50YW1vcyB1bWEgZXN0cnV0dXJhIGRlIGPDs2RpZ28gcXVlIHNlZ3VlIGFzIGJvYXMgcHLDoXRpY2FzIGRlIHNlcGFyYcOnw6NvIGRlIHJlc3BvbnNhYmlsaWRhZGVzIGVudHJlIGNvbnRyb2xsZXJzIGUgcmVwb3NpdMOzcmlvcy4gSXNzbyB0b3JuYSBvIGPDs2RpZ28gbWFpcyBvcmdhbml6YWRvIGUgZGUgZsOhY2lsIG1hbnV0ZW7Dp8Ojby4KLSAqKkZpbHRybyBwYXJhIEF1dGVudGljYcOnw6NvIGUgVmFsaWRhw6fDo28qKjogQ3JpYW1vcyB1bSBmaWx0cm8gcGFyYSBhIHJvdGEgYC90YXNrc2AgcXVlIGdhcmFudGUgYSBhdXRlbnRpY2HDp8OjbyBkbyB1c3XDoXJpbyBlIHZhbGlkYSBhcyBzb2xpY2l0YcOnw7Vlcy4gSXNzbyBhdW1lbnRhIGEgc2VndXJhbsOnYSBlIGEgaW50ZWdyaWRhZGUgZG9zIGRhZG9zLgotICoqRXJyb3IgSGFuZGxlcioqOiBEZXNlbnZvbHZlbW9zIHVtIG1lY2FuaXNtbyBkZSB0cmF0YW1lbnRvIGRlIGVycm9zIHBhcmEgZm9ybmVjZXIgcmVzcG9zdGFzIGNsYXJhcyBlIGFkZXF1YWRhcyBlbSBjYXNvIGRlIHByb2JsZW1hcyBkdXJhbnRlIGFzIHNvbGljaXRhw6fDtWVzLgoKIyMjIFJlcXVpc2l0b3MKLSAqKkphdmEgPj0gMTcqKjsKLSAqKk1hdmVuKio7Ci0gKipSZXN0IGNsaWVudCoqOwoKIyMjIERlcGVkw6puY2lhcwotICoqU3ByaW5nIEpQQSoqOwotICoqTG9tYm9rKio7Ci0gKipCQ3J5cHQqKjsKLSAqKkgyIERhdGFiYXNlKio7CgojIyBBUEkKCiMjIyBQT1NUIGBgL3VzZXJzYGAKCioqUmVxdWVzdCBib2R5KioKYGBganNvbgogIHsKICAgICAgICAibmFtZSI6IHN0cmluZywKICAgICAgICAidXNlcm5hbWUiOiBzdHJpbmcsCiAgICAgICAgInBhc3N3b3JkIjogc3RyaW5nCiAgfQpgYGAKCioqUmVzcG9uc2UqKgpgYGBqc29uCiAgICB7CiAgICAgICAgImlkIjogVVVJRCwKICAgICAgICAidXNlcm5hbWUiOiBzdHJpbmcsCiAgICAgICAgIm5hbWUiOiBzdHJpbmcsCiAgICAgICAgImNyZWF0ZWRBdCI6IFlZWVktTU0tRERUaGg6bW06c3MKICAgIH0KYGBgCgojIyMgUE9TVCBgYC90YXNrc2BgCgoqKkhlYWRlcioqCmBgYApBdXRob3JpemF0aW9uOiBCYXNpYyBCYXNlNjQgKHVzZXJuYW1lOnBhc3N3b3JkKQpgYGAKCioqUmVxdWVzdCoqCgpgYGBqc29uCiAgewogICAgICAidGl0bGUiOiBzdHJpbmcsCiAgICAgICJkZXNjcmlwdGlvbiI6IHN0cmluZywKICAgICAgInByaW9yaXR5IjogMCB8IDEgfCAyIHwgMyB8IDQsCiAgICAgICJzdGFydEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgImVuZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcwogIH0KCmBgYCAKKipSZXNwb25zZSoqCgpgYGBqc29uCiAgewogICAgICAgICJpZCI6IFVVSUQsCiAgICAgICAgInRpdGxlIjogc3RyaW5nLAogICAgICAgICJkZXNjcmlwdGlvbiI6IHN0cmluZywKICAgICAgICAicHJpb3JpdHkiOiAwIHwgMSB8IDIgfCAzIHwgNCwKICAgICAgICAic3RhcnRBdCI6IFlZWVktTU0tRERUaGg6bW06c3MsCiAgICAgICAgImVuZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAiY3JlYXRlZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAidXNlcklkIjogVVVJRAogIH0KYGBgCgojIyMgUE9TVCBgYC90YXNrc2BgCgoqKkhlYWRlcioqCmBgYApBdXRob3JpemF0aW9uOiBCYXNpYyBCYXNlNjQgKHVzZXJuYW1lOnBhc3N3b3JkKQpgYGAKKipSZXNwb25zZSoqCgpgYGBqc29uCiAgewogICAgICAgICJpZCI6IFVVSUQsCiAgICAgICAgInRpdGxlIjogc3RyaW5nLAogICAgICAgICJkZXNjcmlwdGlvbiI6IHN0cmluZywKICAgICAgICAicHJpb3JpdHkiOiAwIHwgMSB8IDIgfCAzIHwgNCwKICAgICAgICAic3RhcnRBdCI6IFlZWVktTU0tRERUaGg6bW06c3MsCiAgICAgICAgImVuZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAiY3JlYXRlZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAidXNlcklkIjogVVVJRAogIH1bXQpgYGAKCiMjIyBQVVQgYGAvdGFza3Mve2lkfWBgCgoqKkhlYWRlcioqCmBgYApBdXRob3JpemF0aW9uOiBCYXNpYyBCYXNlNjQgKHVzZXJuYW1lOnBhc3N3b3JkKQpgYGAKCgoqKlJlc3BvbnNlKioKCmBgYGpzb24KICB7CiAgICAgICAgImlkIjogVVVJRCwKICAgICAgICAidGl0bGUiOiBzdHJpbmcsCiAgICAgICAgImRlc2NyaXB0aW9uIjogc3RyaW5nLAogICAgICAgICJwcmlvcml0eSI6IDAgfCAxIHwgMiB8IDMgfCA0LAogICAgICAgICJzdGFydEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAiZW5kQXQiOiBZWVlZLU1NLUREVGhoOm1tOnNzCiAgfQpgYGAKKipSZXNwb25zZSoqCgpgYGBqc29uCiAgewogICAgICAgICJpZCI6IFVVSUQsCiAgICAgICAgInRpdGxlIjogc3RyaW5nLAogICAgICAgICJkZXNjcmlwdGlvbiI6IHN0cmluZywKICAgICAgICAicHJpb3JpdHkiOiAwIHwgMSB8IDIgfCAzIHwgNCwKICAgICAgICAic3RhcnRBdCI6IFlZWVktTU0tRERUaGg6bW06c3MsCiAgICAgICAgImVuZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAiY3JlYXRlZEF0IjogWVlZWS1NTS1ERFRoaDptbTpzcywKICAgICAgICAidXNlcklkIjogVVVJRAogIH0KYGBgCgojIyBNZWxob3JpYXMKLSBbWF0gQWRpY2lvbmFyIGVuZHBvaW50IHBhcmEgZXhjbHVpciB0YXJlZmE7Ci0gW1hdIERvY3VtZW50YXIgYSBBUEkgdXRpbGl6YW5kbyBTd2FnZ2VyOwotIFsgXSBBZGljaW9uYXIgdGVzdGVzIHBhcmEgY2FkYSBlbmRwb2ludDsKLSBbIF0gU3Vic3RpdHVpciBvIEgyIHBhcmEgbyBQb3N0Z3Jlc3M7Ci0gWyBdIE1lbGhvcmFyIG8gcHJvY2Vzc28gZGUgYXV0ZW50aWNhw6fDo28uCg== readmeEtag: '"d689f7ab5f5361cc9cea81f1fd44969c3c6e4bf9"' readmeLastModified: Wed, 25 Oct 2023 22:00:18 GMT repositoryId: 702747708 description: ' API Rest para gerenciamento de lista de tarefas, construída em Java com o framework Spring Boot.' created: '2023-10-09T23:55:51Z' updated: '2023-10-18T02:35:18Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: thenriquedb logo: https://avatars.githubusercontent.com/u/39653866?v=4 repoEtag: '"d2c6469ced9d47f62809464e80f6be86b19a93363126207a0d97d69bf7ce8bc7"' repoLastModified: Wed, 18 Oct 2023 02:35:18 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/alfuraydio/openapiparser v3: true id: dd50bbb80a9e29c7c6c386a05faeab08 repositoryMetadata: base64Readme: >- IyBTaW1wbGUgamF2YSBPcGVuQVBJIDIgYW5kIE9wZW5BUEkgMyBQYXJzZXIKc2ltcGxlIGphdmEgcGFyc2VyIGZvciBPcHJuQVBJIDIuCg== readmeEtag: '"6568ff70e6f875bd68dfc67105e931a985e0bba0"' readmeLastModified: Sat, 08 Jun 2024 16:51:11 GMT repositoryId: 809965626 description: Simple Java OpenAPI Parser created: '2024-06-03T19:43:12Z' updated: '2024-06-08T16:51:50Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: AlfuraydiO logo: https://avatars.githubusercontent.com/u/13061733?v=4 repoEtag: '"b671d2578c860a2fcb385986db973347acd1d21087998b742658921c875aa55b"' repoLastModified: Sat, 08 Jun 2024 16:51:50 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/omar122/openapiparser - source: openapi3 tags repository: https://github.com/rebecabl/iagenerativa-pipeline-etlpython v3: true id: 0d59d04f3e10ecfd7831c52b9c92cac7 repositoryMetadata: base64Readme: >- CiMgRGVzY3JpcHRpb246CgpUaGlzIFB5dGhvbiBzY3JpcHQgdXRpbGl6ZXMgdGhlIFBhbmRhcyBsaWJyYXJ5IHRvIGNyZWF0ZSBhIHNhbXBsZSBEYXRhRnJhbWUgcmVwcmVzZW50aW5nIHVzZXIgZGF0YSwgaW5jbHVkaW5nIFVzZXJJRCwgTmFtZSwgQWdlLCBFbWFpbCwgYW5kIFRlbGVwaG9uZS4gVGhlIHNjcmlwdCB0aGVuIHJldHJpZXZlcyBhZGRpdGlvbmFsIHVzZXIgaW5mb3JtYXRpb24gZnJvbSBhIGh5cG90aGV0aWNhbCBBUEkgZW5kcG9pbnQgdXNpbmcgdGhlIHJlcXVlc3RzIGxpYnJhcnkuIFRoZSBvYnRhaW5lZCB1c2VyIGRhdGEgaXMgcHJvY2Vzc2VkIHRocm91Z2ggT3BlbkFJJ3MgR1BULTMgZW5naW5lIHRvIGdlbmVyYXRlIHBlcnNvbmFsaXplZCBtYXJrZXRpbmcgbWVzc2FnZXMgZW5jb3VyYWdpbmcgZmluYW5jaWFsIGludmVzdG1lbnQuIFRoZSBtZXNzYWdlcyBhcmUgdGFpbG9yZWQgdG8gZWFjaCB1c2VyLCBpbmNvcnBvcmF0aW5nIHRoZWlyIG5hbWVzIGluIGEgcG9zaXRpdmUgaW52ZXN0bWVudC1mb2N1c2VkIHByb21wdC4KClRoZSBzY3JpcHQgY29uY2x1ZGVzIGJ5IHVwZGF0aW5nIHVzZXIgcHJvZmlsZXMgb24gdGhlIHNhbWUgaHlwb3RoZXRpY2FsIEFQSSB3aXRoIHRoZSBnZW5lcmF0ZWQgbWFya2V0aW5nIG1lc3NhZ2VzLCBpbmRpY2F0aW5nIHN1Y2Nlc3Mgb3IgZmFpbHVyZSBmb3IgZWFjaCB1c2VyLiBUaGUgaW50ZWdyYXRpb24gb2YgUGFuZGFzLCByZXF1ZXN0cywgYW5kIE9wZW5BSSBzaG93Y2FzZXMgYSB2ZXJzYXRpbGUgYXBwcm9hY2ggZm9yIGRhdGEgbWFuaXB1bGF0aW9uLCBleHRlcm5hbCBBUEkgaW50ZXJhY3Rpb24sIGFuZCBBSS1wb3dlcmVkIGNvbnRlbnQgZ2VuZXJhdGlvbiBpbiB0aGUgY29udGV4dCBvZiBwZXJzb25hbGl6ZWQgZmluYW5jaWFsIG1lc3NhZ2luZy4K readmeEtag: '"c83e2760e49d6d4998e3f46be0c4eaa94dae91a9"' readmeLastModified: Sun, 21 Jan 2024 18:45:53 GMT repositoryId: 694912783 description: >- Python Project uses Pandas, Requests and GPT-3 to create personalized marketing messages encouraging investments. Messages are integrated into user profiles in a dummy API. created: '2023-09-22T00:31:11Z' updated: '2024-01-21T18:50:07Z' language: Jupyter Notebook archived: false stars: 0 watchers: 1 forks: 0 owner: Rebecabl logo: https://avatars.githubusercontent.com/u/107273033?v=4 repoEtag: '"c685ac874a43307320e39ac24ba3a34079aa6fae475a396e2460df7792e3991b"' repoLastModified: Sun, 21 Jan 2024 18:50:07 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mcpride/widdershins-templates-asciidoc v3: true id: cd2d1a7ae3adbcc895d7b0abcaf86aa1 repositoryMetadata: base64Readme: >- IyBDb252ZXJ0IE9wZW5BUEkgMyBkZWZpbml0aW9ucyB0byBBU0NJSURPQyBkb2N1bWVudGF0aW9uIHdpdGggd2lkZGVyc2hpbnMKCiMjIEluZm9ybWF0aW9uCgpUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgdGhlIFt3aWRkZXJzaGluc10oaHR0cHM6Ly9naXRodWIuY29tL01lcm1hZGUvd2lkZGVyc2hpbnMvKSBbb3BlbmFwaTMgdGVtcGxhdGVzXShodHRwczovL2dpdGh1Yi5jb20vTWVybWFkZS93aWRkZXJzaGlucy90cmVlL21haW4vdGVtcGxhdGVzL29wZW5hcGkzKSBvZiB2ZXJzaW9uIDQuMDEgLSBtb2RpZmllZCwgdG8gcHJvZHVjZSBhIHZhbGlkIHNpbmdsZSBbQVNDSUlET0NdKGh0dHBzOi8vYXNjaWlkb2Mub3JnLykgZG9jdW1lbnQsIHdoaWNoIGNhbiBiZSBpbmNsdWRlZCBpbnRvIG90aGVyIGRvY3MgYW5kIGNvbnZlcnRlZCB0byBkaXZlcnNlIHRhcmdldCBmb3JtYXRzLiAKClVuZm9ydHVuYXRlbHkgdGhlIGNvbnZlcnNpb24gcmF0ZSBpcyBub3QgMTAwJSEgRm9yIGV4YW1wbGUgbWFya2Rvd24gaW4gZGVzY3JpcHRpb25zIHdpbGwgbm90IGJlIHByb3Blcmx5IGNvbnZlcnRlZCwgYnV0IGl0IGhlbHBzIG1lIGFuZCBteSB0ZWFtIHRvIHByb2R1Y2UgdmFsaWQgaHVtYW4gcmVhZGFibGUgaW50ZXJmYWNlIGRvY3VtZW50YXRpb24uCgpUaGVyZSBpcyBhIHBldHN0b3JlIDMgZXhhbXBsZSBpbmNsdWRlZCB0byBjaGVjayB0aGUgcmVzdWx0cy4KCkZlZWwgZnJlZSB0byB1c2UgaXQgLi4uIGJ1dCB3aXRob3V0IGd1YXJhbnRlZXMgOy0p readmeEtag: '"96699182b8b81d3ca21ae4f7d2ef573a38c2ac0e"' readmeLastModified: Tue, 14 May 2024 07:27:44 GMT repositoryId: 800360164 description: Widdershins templates for AsciiDoc created: '2024-05-14T07:25:27Z' updated: '2024-05-14T07:32:32Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: mcpride logo: https://avatars.githubusercontent.com/u/1053528?v=4 license: MIT repoEtag: '"e9ce0f510443de1e9fc3022e63475c5aa282226fe2eb31caf049995e1c69118a"' repoLastModified: Tue, 14 May 2024 07:32:32 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/mcpikon/cinemarustback v3: true id: 35dfae36d7a4349b51973290867a72f6 repositoryMetadata: base64Readme: >- IyBDaW5lbWFSdXN0QmFjawoKWyFbUnVzdF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS8yMDIxLWJsYWNrP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1ydXN0JmxvZ29Db2xvcj13aGl0ZSZsYWJlbD1SdXN0JmxhYmVsQ29sb3I9YmxhY2smY29sb3I9d2hpdGUpXShodHRwczovL3d3dy5ydXN0LWxhbmcub3JnL2VzKSAmbmJzcDsKWyFbQWN0aXggV2ViXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlLzQuOC4wKy1ibGFjaz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289YWN0aXgmbG9nb0NvbG9yPXdoaXRlJmxhYmVsPUFjdGl4JTIwV2ViJmxhYmVsQ29sb3I9YmxhY2smY29sb3I9d2hpdGUpXShodHRwczovL2FjdGl4LnJzLykgJm5ic3A7ClshW01vbmdvREJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTW9uZ29EQi02LjArLTAwNjg0QT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289bW9uZ29kYiZsb2dvQ29sb3I9d2hpdGUmbGFiZWxDb2xvcj0xMDEwMTApXShodHRwczovL3d3dy5tb25nb2RiLmNvbSkgJm5ic3A7ClshW1N3YWdnZXJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvU3dhZ2dlci1PQVMzLSUyMzg1RUEyRD9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289c3dhZ2dlciZsb2dvQ29sb3I9JTIzRkZGRkZGJmxhYmVsQ29sb3I9JTIzMDAwMDAwKV0oaHR0cHM6Ly9zd2FnZ2VyLmlvLykgJm5ic3A7ClshW1JlZG9jXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1JlZG9jLU9BUzMtZ3JheT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289c3dhZ2dlciZsb2dvQ29sb3I9JTIzRkZGRkZGJmxhYmVsQ29sb3I9JTIzMDAwMDAwKV0oaHR0cHM6Ly9yZWRvY2x5LmNvbS8pICZuYnNwOwpbIVtTY2FsYXJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvU2NhbGFyLU9BUzMtYmx1ZT9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289c3dhZ2dlciZsb2dvQ29sb3I9JTIzRkZGRkZGJmxhYmVsQ29sb3I9JTIzMDAwMDAwKV0oaHR0cHM6Ly9zY2FsYXIuY29tLykKCiMjIPCfmYvigI3imYLvuI8gQXV0b3IKCiogW0phdmllciBQaWPDs25dKGh0dHBzOi8vZ2l0aHViLmNvbS9NQ1Bpa29uKQoKIyMg4pyoIERlc2NyaXBjacOzbgoKQVBJIFJFU1QgZGUgcGVsw61jdWxhcywgc2VyaWVzIHkgcmVzZcOxYXMgZGUgbGFzIG1pc21hcy4KCj4gWyFOT1RFXQo+IEVzdGUgcHJveWVjdG8gdXRpbGl6YSBbU3dhZ2dlciBVSV0oaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS9zd2FnZ2VyLXVpLyksIFtSZWRvY10oaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS9yZWRvYykgeSBbU2NhbGFyXShodHRwOi8vbG9jYWxob3N0OjgwODAvYXBpL3NjYWxhcikgcGFyYSBsYSBkb2N1bWVudGFjacOzbi4KCiMjIPCfkqEgQ8OzbW8gZnVuY2lvbmEKCkVzdGUgcHJveWVjdG8gcmVhbGl6YSB1biBDUlVEIChDcmVhciwgT2J0ZW5lciwgTW9kaWZpY2FyIHkgRWxpbWluYXIpIGRlIFBlbMOtY3VsYXMsIFNlcmllcyB5IHN1cyBSZXNlw7Fhcy4gRGlzcG9uZW4gZGUgZW50aWRhZGVzLCBEVE9zLCByZXBvc2l0b3Jpb3MsIHNlcnZpY2lvcyBlIGltcGxlbWVudGFjaW9uZXMgZGUgbG9zIG1pc21vcy4KCiMjIPCfm6AgVGVjbm9sb2fDrWFzCgoqIFJ1c3QgKDIwMjEgRWRpdGlvbikKKiBBY3RpeCBXZWIgNC44LjAKKiBfKipEZXBlbmRlbmNpYXMgKENyYXRlcyk6KipfCiAgICAqIGFzeW5jLXRyYWl0ICgwLjEuODApCiAgICAqIGNocm9ubyAoMC40LjM4KQogICAgKiBkZXJpdmVfbW9yZSAoMC45OS4xOCkKICAgICogZG90ZW52ICgwLjE1LjApCiAgICAqIGVudl9sb2dnZXIgKDAuMTEuMykKICAgICogZnV0dXJlcy11dGlsICgwLjMuMzApCiAgICAqIGxhenlfc3RhdGljICgxLjUuMCkKICAgICogbG9nICgwLjQuMjEpCiAgICAqIG1vbmdvZGIgKDIuOC4yKQogICAgKiByZWdleCAoMS4xMC41KQogICAgKiBzZXJkZSAoMS4wLjIwMykKICAgICogc2VyZGVfanNvbiAoMS4wLjExOCkKICAgICogdXRvaXBhICg0LjIuMykKICAgICogdXRvaXBhLXJlZG9jICg0LjAuMCkKICAgICogdXRvaXBhLXNjYWxhciAoMC4xLjApCiAgICAqIHV0b2lwYS1zd2FnZ2VyLXVpICg3LjEuMCkKICAgICogdmFsaWRhdG9yICgwLjE4LjEpCiAgICAqIG1vY2thbGwgKDAuMTIuMSkKCiMjIPCfk4QgTGljZW5jaWEKCkVzdGUgcHJveWVjdG8gZXMgZGUgY8OzZGlnbyBhYmllcnRvIHkgZXN0w6EgZGlzcG9uaWJsZSBiYWpvIGxhIFtMaWNlbmNpYSAyLjAgZGUgQXBhY2hlXShMSUNFTlNFKS4K readmeEtag: '"78f236f109c92b69da7a6fa893d603ecc15d1719"' readmeLastModified: Fri, 19 Jul 2024 13:15:31 GMT repositoryId: 810446094 description: >- REST API related to movies, series and its reviews created with Rust and Actix Web. created: '2024-06-04T18:00:24Z' updated: '2024-07-19T13:15:39Z' language: Rust archived: false stars: 0 watchers: 1 forks: 0 owner: MCPikon logo: https://avatars.githubusercontent.com/u/82212139?v=4 license: Apache-2.0 repoEtag: '"8eaacaf341316ef605b3a3265b54da23e7e47ceaefb4992f54c328c376121e2c"' repoLastModified: Fri, 19 Jul 2024 13:15:39 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/productdock/build-apis-made-easy v3: true id: 7a05367e8ece888a40c50a1d22c84fb3 repositoryMetadata: base64Readme: >- IyBCdWlsZGluZyBhcGlzIG1hZGUgZWFzeQoKIyMgUHJlcmVxdWlzaXRlcwoKLSBUeXBlc3BlYzogYG5wbSBpbnN0YWxsIC1nIEB0eXBlc3BlYy9jb21waWxlcmAKLSBvYXBpLWNvZGVnZW46IGBnbyBpbnN0YWxsIGdpdGh1Yi5jb20vZGVlcG1hcC9vYXBpLWNvZGVnZW4vdjIvY21kL29hcGktY29kZWdlbkBsYXRlc3RgCi0gUG9zdG1hbgoKIyMgUnVuIGV4YW1wbGVzCgpBZnRlciBjbG9uaW5nIHRoaXMgcmVwb3NpdG9yeSwgY2hlY2sgTWFrZWZpbGUgZmlyc3QuIAoKQ2hvb3NlIHNlcnZlciBleGFtcGxlOgoKLSBSdW4gZ29sYW5nIGFwaSAocG9ydCA4MDg0KTogYG1ha2UgcnVuLWdvYAoKLSBSdW4gc3ByaW5nIGJvb3QgYXBpIChwb3J0IDgwODQpOiBgbWFrZSBydW4tanZtYAoKSW1wb3J0IGBldmVudHMtcG9zdG1hbi1jb2xsZWN0aW9uLmpzb25gIHRvIHBvc3RtYW4gdG8gdGVzdCB0aGUgQVBJLg== readmeEtag: '"62ec12ee863507c9778be2e3305ae52db2164600"' readmeLastModified: Tue, 06 Aug 2024 12:16:09 GMT repositoryId: 815105500 description: >- Building APIs Made Easy: From Specification to Code Generation and Testing blog post created: '2024-06-14T11:12:07Z' updated: '2025-08-12T08:22:32Z' language: Java archived: false stars: 0 watchers: 2 forks: 0 owner: ProductDock logo: https://avatars.githubusercontent.com/u/86293668?v=4 license: Apache-2.0 repoEtag: '"af049a23a34935578c9f3926d954f8997c3056dec270f176259cf6b1b1003048"' repoLastModified: Tue, 12 Aug 2025 08:22:32 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/millroy094/task-processor v3: true id: d69c98803676a357eb22ced10ce2e30e repositoryMetadata: base64Readme: >- IyBUYXNrIFByb2Nlc3NvciBBcHBsaWNhdGlvbgoKVGhpcyBpcyBhIHRhc2sgcHJvY2Vzc29yIGFwcGxpY2F0aW9uIHRoYXQgY29uc3VtZXMgdGFza3MgZnJvbSBhIFJhYmJpdE1RIG1lc3NhZ2UgcXVldWUsIHByb2Nlc3NlcyB0aGVtLCBhbmQgc3RvcmVzIHRoZSByZXN1bHRzIGluIE1vbmdvREIuIFRoZSB0YXNrcyBjYW4gYmUgb2YgdmFyaW91cyB0eXBlcywgc3VjaCBhcyBoZWFsdGggY2hlY2tzIG9yIGVtYWlsIHByb2Nlc3NpbmcuIFRoZSBhcHBsaWNhdGlvbiB0cmFja3MgdGhlIHN0YXR1cyBvZiBlYWNoIHRhc2ssIGluY2x1ZGluZyByZXRyaWVzLCBmYWlsdXJlcywgYW5kIHRpbWVzdGFtcHMuCgojIyBGZWF0dXJlcwoKLSAqKlRhc2sgUXVldWUgUHJvY2Vzc2luZyoqOiBDb25zdW1lcyB0YXNrcyBmcm9tIGEgUmFiYml0TVEgcXVldWUuCi0gKipUYXNrIFR5cGVzKio6IFN1cHBvcnRzIGRpZmZlcmVudCB0eXBlcyBvZiB0YXNrcyBsaWtlIGhlYWx0aCBjaGVja3MgYW5kIGVtYWlsIHByb2Nlc3NpbmcuCi0gKipUYXNrIFN0YXR1cyBUcmFja2luZyoqOiBVcGRhdGVzIHRhc2sgc3RhdHVzIChlLmcuLCBpbi1wcm9ncmVzcywgY29tcGxldGVkLCBmYWlsZWQpIGFuZCBzdG9yZXMgcmVzdWx0cyBpbiBNb25nb0RCLgotICoqUmV0cnkgTG9naWMqKjogUmV0cmllcyB0YXNrIGV4ZWN1dGlvbiB1cCB0byBhIGNvbmZpZ3VyYWJsZSBudW1iZXIgb2YgcmV0cmllcy4KLSAqKkdyYWNlZnVsIFNodXRkb3duKio6IEhhbmRsZXMgc3lzdGVtIHNodXRkb3duIGdyYWNlZnVsbHkgYnkgc3RvcHBpbmcgdGhlIHdvcmtlcnMgYW5kIGNsZWFuaW5nIHVwIHJlc291cmNlcy4KLSAqKlRhc2sgUmVzdWx0IFN0b3JhZ2UqKjogU3RvcmVzIHRhc2sgcmVzdWx0cywgaW5jbHVkaW5nIHN1Y2Nlc3Mgb3IgZmFpbHVyZSBzdGF0dXMsIG1lc3NhZ2VzLCBhbmQgdGltZXN0YW1wcyBpbiBNb25nb0RCLgoKIyMgQXJjaGl0ZWN0dXJlCgotICoqUmFiYml0TVEqKjogVXNlZCBhcyB0aGUgbWVzc2FnZSBicm9rZXIgdG8gcXVldWUgdGFza3MgZm9yIHByb2Nlc3NpbmcuCi0gKipNb25nb0RCKio6IFVzZWQgYXMgdGhlIGRhdGFiYXNlIHRvIHN0b3JlIHRhc2sgc3RhdHVzIGFuZCByZXN1bHRzLgotICoqR28qKjogVGhlIGFwcGxpY2F0aW9uIGlzIHdyaXR0ZW4gaW4gR28sIHV0aWxpemluZyB0aGUgYHN0cmVhZHdheS9hbXFwYCBsaWJyYXJ5IHRvIGludGVyZmFjZSB3aXRoIFJhYmJpdE1RIGFuZCB0aGUgYG1vbmdvLWRyaXZlcmAgZm9yIE1vbmdvREIgb3BlcmF0aW9ucy4KCiMjIFJlcXVpcmVtZW50cwoKLSBEb2NrZXIgYW5kIERvY2tlciBDb21wb3NlIChmb3IgcnVubmluZyBSYWJiaXRNUSwgTW9uZ29EQiwgYW5kIHRoZSBhcHAgdG9nZXRoZXIpCi0gR28gMS4xOCsgZm9yIHJ1bm5pbmcgdGhlIGFwcGxpY2F0aW9uIGxvY2FsbHkKLSBNb25nb0RCIHJ1bm5pbmcgKGVpdGhlciBsb2NhbGx5IG9yIHZpYSBEb2NrZXIpCi0gUmFiYml0TVEgcnVubmluZyAoZWl0aGVyIGxvY2FsbHkgb3IgdmlhIERvY2tlcikKCiMjIEVudmlyb25tZW50IFZhcmlhYmxlcwoKVGhlIGFwcGxpY2F0aW9uIHJlcXVpcmVzIHRoZSBmb2xsb3dpbmcgZW52aXJvbm1lbnQgdmFyaWFibGVzIHRvIGZ1bmN0aW9uIHByb3Blcmx5OgoKLSBgUkFCQklUTVFfVVJMYDogVVJMIGZvciBSYWJiaXRNUSBjb25uZWN0aW9uIChkZWZhdWx0OiBgYW1xcDovL2d1ZXN0Omd1ZXN0QHJhYmJpdG1xOjU2NzIvYCkuCi0gYE1PTkdPREJfVVJMYDogTW9uZ29EQiBjb25uZWN0aW9uIFVSTCAoZGVmYXVsdDogYG1vbmdvZGI6Ly9tb25nb2RiOjI3MDE3L2ApLgotIGBNQVhfUkVUUklFU2A6IE1heGltdW0gbnVtYmVyIG9mIHJldHJpZXMgZm9yIGEgZmFpbGVkIHRhc2sgKGRlZmF1bHQ6IGAzYCkuCi0gYEFQSV9QT1JUYDogVGhlIHBvcnQgdGhlIHByb2R1Y2VyIHdpbGwgcmVjZWl2ZSB0YXNrIHJlcXVlc3Qgb24gKGRlZmF1bHQ6IGA4MDgwYCkuCgojIyBJbnN0YWxsYXRpb24KCiMjIyAxLiBDbG9uZSB0aGUgUmVwb3NpdG9yeQoKYGBgYmFzaApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL21pbGxyb3kwOTQvdGFzay1wcm9jZXNzb3IuZ2l0CmNkIHRhc2stcHJvY2Vzc29yCg== readmeEtag: '"d5e558363660d1c7bf9a0c236714a6565c44a88f"' readmeLastModified: Wed, 04 Dec 2024 15:00:22 GMT repositoryId: 892571166 description: null created: '2024-11-22T11:14:53Z' updated: '2024-12-04T18:50:21Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: Millroy094 logo: https://avatars.githubusercontent.com/u/58091953?v=4 repoEtag: '"c56184c41f2990d582931af328e2c7bc5f0cd806af7d136d395c78d5643d0a8f"' repoLastModified: Wed, 04 Dec 2024 18:50:21 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/markjivko/rest-object-tree v3: true id: 364b0c5b54d32febc564dcc5faf89473 repositoryMetadata: base64Readme: >- IyBSRVNUIE9iamVjdCBUcmVlIEdlbmVyYXRvcgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8YSBocmVmPSJodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzL21hcmtqaXZrby9iYm94LzAuMC4xIj4KICAgICAgICA8aW1nIHNyYz0iaHR0cHM6Ly9yZXBvc2l0b3J5LWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vNTE1MDYxMDgzLzEyZGZiYmUwLWE4ZGItNDQxMy05MzJmLWNmM2ExNzFiMTA3NCIvPgogICAgPC9hPgo8L3A+CgojIyAxLiBVc2FnZQoKIyMjIDEuMS4gR2VuZXJhdGUKClByZXBhcmUgdGhlIG5ldyBgb3BlbmFwaS55YW1sYCBzcGVjaWZpY2F0aW9uIGFuZCBnZW5lcmF0ZSB0aGUgUkVTVCBBUEkgc2VydmVyIHNvdXJjZSBjb2RlLgoKYGBgCm5wbSBydW4gZ2VuZXJhdGUKYGBgCgpZb3UgY2FuIGVhc2lseSBkZWZpbmUgb2JqZWN0cyBhbmQgdHJlZSBicmFuY2hlcyBpbiBgY29uZmlnL2NvbmZpZy55YW1sYC4KCkNob29zZSBvbmUgb2YgdGhlIHByZWRlZmluZWQgZ2VuZXJhdG9ycyBvciBjcmVhdGUgYSBuZXcgb25lIGluIGBzcmMvZ2VuZXJhdG9yLyouanNgLgoKVGhpcyBwcm9qZWN0IHVzZXMgYC5tdXN0YWNoZWAgZmlsZXMgYXMgdGVtcGxhdGVzLgoKIyMjIDEuMi4gU2VydmUKCkxhdW5jaCB0aGUgUkVTVCBBUEkgYW5kIGl0cyBPcGVuQVBJIHYzIGRvY3VtZW50YXRpb246CgpgYGAKbnBtIHJ1biBzZXJ2ZQpgYGAKCiMjIDIuIENvbXBvbmVudHMKCiMjIyAyLjEuIE9iamVjdHMKCvCfk5ogVGVtcGxhdGUge2JlY29tZXN9IERyYWZ0IHtwdWJsaXNoZWQgb24tY2hhaW4gdG99IE9iamVjdC4KCllvdSBjYW5ub3QgKipjcmVhdGUqKiBvYmplY3RzIG9yIG9iamVjdCBkcmFmdHMgZGlyZWN0bHkuICAKCjEuICBgUFVUIC9vYmplY3QveC90ZW1wbGF0ZS97dGVtcGxhdGVJZH0ve29iamVjdElkfS9gIHRvIGNyZWF0ZSBhIERyYWZ0IGluc2lkZSBPYmplY3QgKGFkZGVkIGlmIG1pc3NpbmcpCgpZb3UgY2Fubm90ICoqdXBkYXRlKiogb2JqZWN0cyBvciBvYmplY3QgdmVyc2lvbnMgZGlyZWN0bHkuICAKCjEuICBgUFVUIC9vYmplY3QveC97b2JqZWN0SWR9L2RyYWZ0L3tkcmFmdElkfS9gIHRvIHB1Ymxpc2ggRHJhZnQgdG8gT2JqZWN0IChhcHBlbmQgdG8gdmVyc2lvbiBjaGFpbikKMi4gIGBQVVQgL29iamVjdC94L3tvYmplY3RJZH0vdmVyc2lvbi97dmVyc2lvbn0vYCB0byByZXZlcnQgT2JqZWN0IHRvIHZlcnNpb24gKHRyaW0gdmVyc2lvbiBjaGFpbikKCkRyYWZ0cyBhcmUgZGlzY2FyZGVkIGFmdGVyIHB1Ymxpc2hpbmcuICAKU2NoZW1hcyBjYW4gb25seSBiZSBjcmVhdGVkIGFuZCBtb2RpZmllZCBpbiBUZW1wbGF0ZXMuCgp8ICAgICAgICAgIHwgVGVtcGxhdGUgICB8IERyYWZ0ICAgICAgIHwgT2JqZWN0ICAgICAgICAgICAgICAgICAgfAp8LS0tLS0tLS0tLXwtLS0tLS0tLS0tLS18LS0tLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tfAp8IERhdGEgICAgIHwgRWRpdGFibGUgICB8IEVkaXRhYmxlICAgIHwgUmVhZC1vbmx5ICh2ZXJzaW9uaXplZCkgfAp8IFNjaGVtYSAgIHwgRWRpdGFibGUgICB8IFJlYWQtb25seSAgIHwgUmVhZC1vbmx5ICh2ZXJzaW9uaXplZCkgfAoKRGF0YSBtaWdyYXRpb24gaXMgdGhlIHNhbWUgYXMgdXNlciBpbnB1dCBzbyBpdCBtdXN0IGJlIGhhbmRsZWQgYXQgdGhlIGFwcGxpY2F0aW9uIGxldmVsLCBub3QgYXQgdGhlIEFQSSBsZXZlbDogIAoKMS4gIGBQT1NUIHwgR0VUIC9vYmplY3QveC90ZW1wbGF0ZS97dGVtcGxhdGVJZH0vYCAtIChvbmNlKSBQcmVwYXJlIG5ldyBzY2hlbWEgYW5kIGRlZmF1bHQgZGF0YQoyLiAgYEdFVCAvb2JqZWN0L3gve29iamVjdElkfS9gIC0gRmV0Y2ggY3VycmVudCBvYmplY3QgZGF0YSwgc2NoZW1hIGFuZCBzY2hlbWEgSUQKMy4gIEluc2lkZSBhcHBsaWNhdGlvbiAtIG1pZ3JhdGUgZGF0YSB1c2luZyAjMSBhbmQgIzIKNC4gIGBQVVQgL29iamVjdC94L3RlbXBsYXRlL3t0ZW1wbGF0ZUlkfS97b2JqZWN0SWR9L2Agd2l0aCAqKnB1Ymxpc2g9dHJ1ZSoqIC0gVXBkYXRlIE9iamVjdCB3aXRoIG5ldyBkYXRhICYgc2NoZW1hCgpBcHBsaWNhdGlvbnMgc2hvdWxkIGRpc3BsYXkgYW55IHNjaGVtYS4gIApEYXRhIG1pZ3JhdGlvbiBzaG91bGQgYmUgcGVyZm9ybWVkIG9uIHVzZXIgcmVxdWVzdC4KCiMjIyAyLjIuIFRyZWVzCgoqKlRyZWVzKiogZGVmaW5lIHJlbGF0aW9uc2hpcHMgYmV0d2VlbiBPYmplY3RzLiAgCgpDb25maWd1cmUgdGhlIGJyYW5jaCBzdHJ1Y3R1cmUgdXNpbmcgY2FyZGluYWxpdHkuIFVzZSBleGNsYW1hdGlvbiBtYXJrIChgIWApIGZvciAxOgoKMS4gICoqbWFueS10by1tYW55KiogRXhhbXBsZTogYC90cmVlL3RlYW0vdXNlcmAKMi4gICoqb25lLXRvLW1hbnkqKiBFeGFtcGxlOiBgL3RyZWUvb3JnYW5pemF0aW9uIS90ZWFtYAozLiAgKiptYW55LXRvLW9uZSoqIEV4YW1wbGU6IGAvdHJlZS91c2VyLyFsZXZlbGAKNC4gICoqb25lLXRvLW9uZSoqIEV4YW1wbGU6IGAvdHJlZS91c2VyIS8hbG9nb2AKCioqTm90ZSoqOiB0aGUgZXhjbGFtYXRpb24gbWFyayBzeW50YXggaXMgb25seSB1c2VkIGluc2lkZSB0aGUgYGNvbmZpZy55YW1sYCBmaWxlIHRvIGRlZmluZSBjYXJkaW5hbGl0eSEgCkZpbmFsIFJFU1QgQVBJIGVuZHBvaW50cyB1c2Ugb25seSB0aGUgcHJlLWRlZmluZWQgb2JqZWN0IG5hbWVzLCBpLmUuIGAvdHJlZS91c2VyL2xvZ29gIGluc3RlYWQgb2YgYC90cmVlL3VzZXIhLyFsb2dvYC4KCiMjIDMuIFNhbXBsZSBvdXRwdXQKCltUaGlzIE9wZW5BUEkgdjMgc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcy9tYXJraml2a28vYmJveC8wLjAuMSkgaXMgZ2VuZXJhdGVkIGJ5IApkZWZhdWx0IHdoZW4gZmlyc3QgZXhlY3V0aW5nIGBucG0gcnVuIGdlbmVyYXRlYC4K readmeEtag: '"c4fa00c20e727b562f452573a52481926d576109"' readmeLastModified: Thu, 03 Oct 2024 13:57:58 GMT repositoryId: 515061083 description: OpenAPI v3 generator for a simple REST API with Objects placed on a Tree created: '2022-07-18T06:28:45Z' updated: '2024-10-03T13:58:02Z' language: Mustache archived: false stars: 0 watchers: 1 forks: 0 owner: markjivko logo: https://avatars.githubusercontent.com/u/50949149?v=4 license: Apache-2.0 repoEtag: '"edd19fd83729e4d1d27627d344bf0400a559993a460b805f0a2b04d2dedbda01"' repoLastModified: Thu, 03 Oct 2024 13:58:02 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/tinello/jetty-embedded-openapi v3: true id: d10022293be2418556ad039c2a7af748 repositoryMetadata: base64Readme: >- IyBKZXR0eS1FbWJlZGRlZC1PcGVuQXBpCkFQSSBSZXN0IHdpdGggRWNsaXBzZSBKZXR0eSwgT3BlbiBBcGkgVmFsaWRhdGUuIAoKCiMjIFJlcXVpcmVtZW50cyA8YSBuYW1lPSJSZXF1aXJlbWVudHMiPjwvYT4KIyMjIFNvZnR3YXJlOiA8YSBuYW1lPSJTb2Z0d2FyZSI+PC9hPgotIE9wZW5KREsgMjAgd2l0aCBPcGVuSjkgLT4gaHR0cHM6Ly9kZXZlbG9wZXIuaWJtLmNvbS9sYW5ndWFnZXMvamF2YS9zZW1lcnUtcnVudGltZXMvZG93bmxvYWRzLwotIERvY2tlciA0LjI4LjAgLT4gaHR0cHM6Ly9kb2NzLmRvY2tlci5jb20vZGVza3RvcC9pbnN0YWxsL3VidW50dS8KCiMjIyBDb25maWd1cmUgSkRLIFZTQ29kZTogPGEgbmFtZT0iQ29uZmlndXJlSkRLdnNjb2RlIj48L2E+CgpDaGFuZ2UgLy52c2NvZGUvbGF1bmNoLmpzb24gZm9yIHlvdXIgSmF2YSBIb21lCgojIyMgQ29uZmlndXJlIEpESyBHcmFkbGU6IDxhIG5hbWU9IkNvbmZpZ3VyZUpES2dyYWRsZSI+PC9hPgoKQ2hhbmdlIGdyYWRsZS5wcm9wZXJ0aWVzIGZvciB5b3VyIEphdmEgSG9tZQoKIyMjIEVudmlyb25tZW50IHZhcmlhYmxlczogPGEgbmFtZT0iRW52aXJvbm1lbnRWYXJpYWJsZXMiPjwvYT4KLSBEQl9VUkw9bG9jYWxob3N0OjU0MzIvcG9zdGdyZXMKLSBEQl9VU0VSPXBvc3RncmVzCi0gREJfUEFTUz1teXNlY3JldHBhc3N3b3JkCgojIyMgU3RhcnQgUG9zdGdyZVNRTCA8YSBuYW1lPSJTdGFydFBvc3RncmVTUUwiPjwvYT4KYGBgYmFzaApkb2NrZXIgcnVuIC0tbmFtZSBqZXR0eS1vcGVuYXBpLXBvc3RncmVzIC1wIDU0MzI6NTQzMiAtZSBQT1NUR1JFU19QQVNTV09SRD1teXNlY3JldHBhc3N3b3JkIC1kIHBvc3RncmVzOjE2LjMtYWxwaW5lMy4xOApgYGAKCgojIyMgU3RhcnQgYXBwbGljYXRpb24gPGEgbmFtZT0iU3RhcnRBcHBsaWNhdGlvbiI+PC9hPgpgYGBiYXNoCkRCX1VSTD0ibG9jYWxob3N0OjU0MzIvcG9zdGdyZXMiIERCX1VTRVI9cG9zdGdyZXMgREJfUEFTUz1teXNlY3JldHBhc3N3b3JkIC4vZ3JhZGxldyBydW4KYGBgCgojIyBEZXZlbG9wbWVudCA8YSBuYW1lPSJkZXZlbG9wbWVudCI+PC9hPgojIyMgVmlzdWFsIFN0dWRpbyBDb2RlIEV4dGVuc2lvbnM6IDxhIG5hbWU9InZzY29kZS1leHRlbnNpb25zIj48L2E+CiMjIyMgRXh0ZW5zaW9uIFBhY2sgZm9yIEphdmEKCkluc3RhbGwgIkV4dGVuc2lvbiBQYWNrIGZvciBKYXZhIiBmcm9tIE1pY3Jvc29mdDogaHR0cHM6Ly9tYXJrZXRwbGFjZS52aXN1YWxzdHVkaW8uY29tL2l0ZW1zP2l0ZW1OYW1lPXZzY2phdmEudnNjb2RlLWphdmEtcGFjawoKIyMjIyBHcmFkbGUgRXh0ZW5zaW9uIFBhY2sKCkluc3RhbGwgIkdyYWRsZSBFeHRlbnNpb24gUGFjayIgZnJvbSBSaWNoYXJkIFdpbGxpcyBodHRwczovL21hcmtldHBsYWNlLnZpc3VhbHN0dWRpby5jb20vaXRlbXM/aXRlbU5hbWU9cmljaGFyZHdpbGxpcy52c2NvZGUtZ3JhZGxlLWV4dGVuc2lvbi1wYWNrCgoKIyMgRW5kcG9pbnRzCgp8IE5hbWUgICAgICAgICAgICAgICAgIHwgRW5kcG9pbnQgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAp8IFNlcnZpY2UgSW5mbyAgICAgICAgIHwgaHR0cDovL2xvY2FsaG9zdDo4MDgwLyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAo= readmeEtag: '"53a5a6c822827f5767726b806ed18d85e9f1f568"' readmeLastModified: Thu, 25 Jul 2024 22:38:04 GMT repositoryId: 833280887 description: 'API Rest with Eclipse Jetty, Open Api Validate. ' created: '2024-07-24T18:00:50Z' updated: '2025-12-22T22:41:28Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: tinello logo: https://avatars.githubusercontent.com/u/38929644?v=4 license: Unlicense repoEtag: '"f40bbef934a218fb9f4440fe118d0d398e6b5d6a000af679c5e0f4876c11cb48"' repoLastModified: Mon, 22 Dec 2025 22:41:28 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/hackolade/eventbridge v3: true repositoryMetadata: base64Readme: >- IyBBV1MgRXZlbnRCcmlkZ2UgU2NoZW1hIFJlZ2lzdHJ5CgpQbHVnaW4gdG8gZW5hYmxlIEV2ZW50QnJpZGdlIFNjaGVtYSBSZWdpc3RyeSBhcyBhIHRhcmdldCBpbiBbSGFja29sYWRlXShodHRwczovL2hhY2tvbGFkZS5jb20pIGRhdGEgbW9kZWxpbmcuICBSZXF1aXJlcyBwcmlvciBkb3dubG9hZCBvZiB0aGUgSGFja29sYWRlIGFwcGxpY2F0aW9uIGZyb20gb3VyIFtkb3dubG9hZCBwYWdlXShodHRwczovL2hhY2tvbGFkZS5jb20vZG93bmxvYWQuaHRtbCkKClRoaXMgcGx1Z2luIGlzIGZvciBPcGVuQVBJIDMgc2NoZW1hcyBzdG9yZWQgaW4gdGhlIFNjaGVtYSBSZWdpc3RyeSBvZiBBV1MgRXZlbnRCcmlkZ2UuICAKCkhhY2tvbGFkZSBleHBvc2VzIGl0cyBjb3JlIGRhdGEgbW9kZWxpbmcgZW5naW5lIHRocm91Z2ggYSBwbHVnaW4gYXJjaGl0ZWN0dXJlLiAgRWFjaCBwbHVnaW4gYXBwbGllcyB0aGUgSGFja29sYWRlIGRhdGEgbW9kZWxpbmcgY2FwYWJpbGl0aWVzIHRvIGEgc3BlY2lmaWMgdGFyZ2V0IHRlY2hub2xvZ3ksIHdoZXRoZXIgZm9yIGRhdGEtYXQtcmVzdCAoZGF0YWJhc2VzKSBvciBkYXRhLWluLW1vdGlvbiAoY29tbXVuaWNhdGlvbnMuKSAgRWFjaCBwbHVnaW4gbWF0Y2hlcyB0aGUgc3BlY2lmaWMgYXNwZWN0cyBvZiB0aGUgdGFyZ2V0IGluIHRlcm1zIG9mIHRlcm1pbm9sb2d5LCBzdG9yYWdlIG1vZGVsLCBkYXRhIHR5cGVzLCBhbmQgY29tbXVuaWNhdGlvbiBwcm90b2NvbC4KClRvIGVuYWJsZSBkYXRhIG1vZGVsaW5nIGNhcGFiaWxpdGllcyBmb3IgYSB0YXJnZXQsIHlvdSBtdXN0IGZpcnN0IGRvd25sb2FkIGFuZCBpbnN0YWxsIHRoZSBwbHVnaW4sIGZvbGxvd2luZyB0aGVzZSBbaW5zdHJ1Y3Rpb25zXShodHRwczovL2hhY2tvbGFkZS5jb20vaGVscC9Eb3dubG9hZGFkZGl0aW9uYWxEQnRhcmdldHBsdWdpbi5odG1sICJQbHVnaW4gZG93bmxvYWQgaW5zdHJ1Y3Rpb25zIikuCgpQbHVnaW5zIGNhbiBiZSBjdXN0b21pemVkIGJ5IGZvbGxvd2luZyB0aGVzZSBbaW5zdHJ1Y3Rpb25zXShodHRwczovL2hhY2tvbGFkZS5jb20vaGVscC9Vc2VyZGVmaW5lZGN1c3RvbXByb3BlcnRpZXMuaHRtbCAiUGx1Z2luIGN1c3RvbWl6YXRpb24gaW5zdHJ1Y3Rpb25zIikuCg== readmeEtag: '"abfe7ebc8c4b41dc8cea01b071ca978632b79e91"' readmeLastModified: Fri, 09 Aug 2024 17:24:32 GMT repositoryId: 271966005 description: >- Hackolade(https://hackolade.com) plugin for AWS EventBridge Schema Registry created: '2020-06-13T07:58:44Z' updated: '2026-02-04T17:25:05Z' language: JavaScript archived: false stars: 0 watchers: 3 forks: 8 owner: hackolade logo: https://avatars.githubusercontent.com/u/20265734?v=4 license: NOASSERTION repoEtag: '"03a2e9c034a478030046b830971d7fa02e7ef45997c3f31fce5303508ee785b4"' repoLastModified: Wed, 04 Feb 2026 17:25:05 GMT foundInMaster: true category: - Server - Server Implementations id: a0a295c1b32b9cbaeb72282a09c7dcf2 - source: openapi3 tags repository: https://github.com/jayson-fong/khaopiak v3: true id: 78e4ffe126bc0aba2efc320a61599077 repositoryMetadata: base64Readme: >- <div align="center">
   <h1>🍜 Khaopiak</h1>
</div>

An account-less and end-to-end encrypted storage system leveraging Cloudflare Workers with OpenAPI 3.1
using [chanfana](https://github.com/cloudflare/chanfana) and [Hono](https://github.com/honojs/hono).

> [!WARNING]  
> Khaopiak is still undergoing heavy, active development and is currently not in a stable state. It should **not** be
> used in a production environment at this time.

<hr />

<div align="center">

[💼 Purpose](#purpose) | [🪀 Features](#features) | [🛡️ Security](#security-considerations) | [🏁 Get started](#get-started) | [🦺 Contributing](#contributing)

</div>

<hr />

## Purpose

Khaopiak is a temporary, intermediary file storage system for transferring between two devices. It was designed
primarily with printing at hotel business centers in mind, but can cover a range of use cases from file transfers,
viewership verification, and sharing secrets.

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🖨️ Use Case: Printing at hotel business centers</summary>

Hotel business centers often restrict printing to dedicated, shared desktops, resulting in two main options for printing
from a personal device:

<ul>
   <li>Connect personal hardware (ex. USB thumb drives) to shared equipment</li>
   <li>Upload files to a web-based intermediary for download</li>
</ul>

<h3>Personal hardware</h3>

Connecting personal hardware may not always be an option, whether due to port restrictions (such as software
restrictions and the lack of a physical port) or the lack of a physical medium (i.e. cabling or drives). Shared devices
may also harbor malware, stealing files and potentially infecting connected devices.

<h3>Web-based portals</h3>

Logging into web-based portals such as Google Drive, Box, SharePoint, Proton Drive, and Dropbox requires inputting
personal login credentials and grants excessive access to files alongside additional services, such as email. When
passwords are shared, stolen credentials may lead to further compromise, such as enabling logon to financial accounts.

While these web-based services often provide shareable links, they are often too long or complicated to type, such as
this Google Drive share link:

<pre>https://docs.google.com/document/d/t9trB8lKoaB_kIRk6FeFltqm1TGdsCpKolHGwcpVKXPE</pre>

When end-users incorrectly type a character of these URLs, it is often difficult to identify the source of error. To
mitigate this issue, a link shortener can be used; however, it increases the probability of randomly stumbling upon
the document, involves an additional party, and may be predictable.

Shareable links often do not expire, allowing a threat actor to regain access after the initial download, such as
through browser history.

<h3>How Khaopiak helps</h3>

Khaopiak alleviates these issues through enabling the seamless end-to-end encryption of documents accessed through
easily-typed, one-time-use BIP39 mnemonics, such as:

<pre>orchard home picture movie only what believe onion physical defy hole among climb brand million edge anchor upgrade sand awake loop layer panther soda</pre>

This means that end-users need not reference a random character-by-character string, but known words they can quickly
type and remember the spelling of. As all words can be identified using their first three letters, Khaopiak can
automatically correct typos.

During this process, hardware manipulation is not required, users do not need to enter logon credentials besides a
unique, expiring mnemonic, and mnemonics cannot be reused to download the same upload.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🤝 Use Case: Secrets distribution</summary>

Khaopiak can be used to share secrets in an environment where communications integrity is guaranteed; however,
confidentiality is susceptible to compromise, such as communicating vocally in an open office, on the condition that the
secret is not of value immediate value.

This leverages Khaopiak's expiring mnemonics, thus if the intended receiver successfully downloads from the Khaopiak
server, it can be assumed that only they have it. The receiver can then communicate back to the sender to confirm
receipt and use the secret.

If the secret for transmission is of immediate value, two Khaopiak uploads can be used for secure transmission. First,
transmitting a secret of no immediate value: the client-side mnemonic for an intended upcoming transmission. If receipt
is confirmed by the intended recipient, that client-side mnemonic can then be used to encrypt the actual secret
client-side for upload to Khaopiak.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>✅ Use Case: Confirmed file access</summary>

Expiring mnemonics enables confirmed file access through using the Khaopiak server's file existance checking. If
Khaopiak reports that a file exists, it has not yet been downloaded. If it reports that a file does not exist, it
implies that the file has already been downloaded or the file expired.

</details>

## Features

### For end-users

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🔒 End-to-end encryption</summary>

A portion of the mnemonic is never transmitted over the internet and is used to encrypt the file before uploading,
allowing end-to-end encryption. As a result, confidentiality of the original file is protected as it is never made
available to intermediaries.

For all purposes of encryption at rest, Khaopiak uses the Advanced Encryption Standard (AES), with all clients
supporting [Cipher Block Chaining (CBC)](https://csrc.nist.gov/pubs/sp/800/38/a/final) and
recommending [Galois/Counter Mode (GCM)](https://csrc.nist.gov/pubs/sp/800/38/d/final) when possible.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>⌛ Expiring files</summary>

By default, all files uploaded to Khaopiak eventually expire. If an attempt is made to an expired file which has not
been deleted from the Khaopiak server, it will be immediately deleted and a response will be returned as if the file did
not exist.

Note: It is possible for a client to assume that a file existed based on the additional processing time required to
check whether the file expired.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🗄️ Protected file metadata</summary>

File names and content types are included as part of the payload for encryption at both the client and server sides. As
a result, at rest, file content cannot be easily inferred.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>📏 Configure mnemonic lengths</summary>

Mnemonics can range from 24 to 48 words where end-users can specify the amount of words for the client and server
independently. As a result, users can choose to increase encryption key lengths for more sensitive files for increased
assurance that data confidentiality is protected.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>😵‍💫 File obscurity</summary>

End-users can specify a number of randomized padding bytes for appending to their data at both the client and server
levels, altering file sizes and helping obscure the identity of the original file that was uploaded. As a result, the
content clients upload to the Khaopiak server is always more than or equal to the amount bytes of the original file, and
the amount of bytes the download client receives is always more than or equal to the amount of bytes that was uploaded,
making it difficult to identify the original file (and thereby assume the original file's contents) based on strictly
file size.

</details>

### For administrators

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🔑 Restrict access with Cloudflare Access</summary>

Cloudflare Zero Trust customers can optionally require authentication through Cloudflare Access as a self-hosted
application. Khaopiak will check for a `cf-access-authenticated-user-email` header containing a valid email. Cloudflare
prevents impersonating through stripping the header from client requests.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>☁️ Serverless deployment</summary>

Khaopiak is designed for deployment on [Cloudflare Workers](https://workers.cloudflare.com/),
leveraging [Cloudflare R2](https://developers.cloudflare.com/r2/) for file storage
and [Cloudflare Queues](https://developers.cloudflare.com/queues/) for file expiry, allowing deployment and automated
scaling without having to maintain servers.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>💰 Deploy for free</summary>

For small-scale users, Khaopiak can be deployed on [Cloudflare Workers](https://workers.cloudflare.com/)'s free tier;
however, requires disabling Khaopiak's automated expiry-based deletion system as it
uses [Cloudflare Queues](https://developers.cloudflare.com/queues/), a Workers Paid feature. However, automated file
deletion can still be accomplished through
using [Cloudflare R2's object lifecycle rules](https://developers.cloudflare.com/r2/buckets/object-lifecycles/) and
expired files will remain inaccessible to clients.

</details>

### For all

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>📖 Open source</summary>

Unlike commercially available products, Khaopiak is open source. Organizations and end-users need not go solely based on
product claims, but verify them through analyzing both code and infrastructure design. If a provider hosting a Khaopiak
server cannot be trusted, a private instance can quickly be deployed.

</details>

## Security considerations

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🔐 Cryptographic strength of encryption algorithms</summary>

Khaopiak supports AES-CBC and AES-GCM as they are available through
the [SubtleCrypto interface of the Web Crypto API](https://developer.mozilla.org/en-US/docs/Web/API/SubtleCrypto):

<ul>
   <li><strong>RSA-OAEP</strong> is supported and required when encrypting payloads to the server or requesting an encrypted response. It is not supported for at-rest storage as it is a public-key encryption system, where current guidelines recommend a minimum of 2048 key bits. To meet this, 192+ BIP39 words would be required, which is unreasonable for an end-user. While client developers may use it for client-side encryption, server-side mnemonic-based encryption/decryption with RSA-OAEP will not be offered.</li>
   <li><strong>AES-CTR</strong> is not supported as it is malleable, potentially allowing the meaning of the ciphertext to be changed.</li>
   <li><strong>AES-CBC</strong> is supported as a client-side encryption algorithm. While Khaopiak is generally not itself vulnerable to a padding oracle attack, client developers should be aware of the algorithm's vulnerability.</li>
   <li><strong>AES-GCM</strong> is supported as both a client and server-side encryption algorithm. AES-GCM provides authenticated encryption which helps authenticate the ciphertext. Additional design considerations are necessary when it is possible for a key and initialization vector (IV) may potentially be reused; however, Khaopiak generates a random key and IV for each upload.</li>
</ul>

The OpenSSL enc program does not support authenticated encryption modes. As a result, some clients may use AES-CBC
instead, such as uploading from the command line.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>💥 Server-side collisions</summary>

Khaopiak does not generate guaranteed unique mnemonics. As a result, it is theoretically possible for a collision to
occur, which may enable accidental file overwriting or downloading of an alternate file. However, this case is extremely
improbable. Client-side encryption helps protect data confidentiality even in the presence of a server-side failure.
While it is possible for another collision, enabling decryption of the file, this is improbable.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🦹 Mnemonic theft</summary>

To conveniently use Khaopiak, the same portal should both accept a server mnemonic and a client mnemonic. However, the
client mnemonic could accidentally be sent to the server if entered incorrectly, compromising end-to-end encryption. A
portal may also be maliciously designed to explicitly capture client mnemonics.

To avoid mnemonic theft, an end-user should have a means of verifying the portal's legitimacy (such as through TLS
certificates). Further, the portal should have a clear means of distinguishing the client and server mnemonics, such as
using half of a combined mnemonic for each, or using a distinct wordlist for the client and server.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🧮 User-specified entropy length</summary>

When a user specifies an entropy length of 160 or 224, Khaopiak pads the entropy to become 192 or 256 bits,
respectively, as AES only supports 128, 192, and 256-bit keys. As a result, while a longer-bit algorithm is used for
encryption, it does not inherently increase the level of security assurance as the padding is predictable.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🔎 Stored hash of entropy</summary>

To locate the file in [Cloudflare R2](https://developers.cloudflare.com/r2/), the server generates a HMAC-SHA256 of the
entropy used for mnemonic generation. As a result, the hash acts as a heuristic for determining the original entropy,
and thereby the decryption key for server-side encryption; however, this is impractical due to the sheer number of
possible combinations. This risk is further mitigated through leveraging an HMAC with a secret key for the server. As a
result, even if entropy generated between two different Khaopiak servers are identical, the associated object key would
differ given a different secret key.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🛠️ Tampering of server code</summary>

In the event that the Cloudflare Worker is compromised and maliciously altered, it is possible for a threat actor to
bypass server-side encryption and easily associate file upload and download clients, presenting an elevated security
risk. This is mitigated through the Khaopiak client natively not trusting the server and encrypting the file locally,
inclusive of file metadata, enabling end-to-end encryption such that even in the event of a server compromise, end-user
content remains protected. Further, end-users may elect for their client to insert randomized padding into the plaintext
prior to encrypting locally, obfuscating the payload's true size and thereby helping obscure the originally uploaded
file.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>📋️ Unauthorized access to server-side storage</summary>

If a threat actor obtains access to files in [Cloudflare R2](https://developers.cloudflare.com/r2/), they would have
access to files encrypted using AES-CBC, where the plaintext may be padded, and itself a padded ciphertext generated by
the client, which is not of immediate value without substantial computing resources and is generally infeasible. Through
using a publicly accessible server, additional obscurity is provided through mixing files between end users, making it
such that threat actors cannot easily differentiate between ciphertexts with valuable plaintexts.

If a threat actor lacks the ability to list files and only has the ability to download given an object key, they would
need a HMAC-SHA256 hash, and it is generally infeasible to brute-force checking hashes due to the sheer number of
possibilities. To generate a valid HMAC-SHA256 corresponding an object's key, a threat actor would require both the
server's secret and the entropy used to generate the server-side mnemonic. Given a threat actor has this information, it
would be easier to query the Khaopiak API rather than [Cloudflare R2](https://developers.cloudflare.com/r2/).

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🦠 Alteration of client code</summary>

As with all end-to-end encrypted solutions, the integrity of the client is crucial for ensuring secure communications. A
maliciously altered client may send unencrypted data to a third party, generate cryptographically insecure entropy, or
record mnemonics without end-user knowledge.

The native, web-based client Khaopiak ships with works to mitigate these threats through recommending that browsers
verifying script and stylesheet integrity, rejecting certain attributes (such as in-line scripts and styles), upgrade
requests to secure HTTP when possible, and relying on no third party assets.

Additional mitigations are deployment-specific, such as enabling
the [Domain Name System Security Extensions (DNSSEC)](https://www.icann.org/resources/pages/dnssec-what-is-it-why-important-2019-03-05-en)
and configuring [certain response headers](#recommended-headers).

Despite these mitigations, the security of upload and download devices alongside the client's integrity remains the
primary concern to the usage of Khaopiak. Malicious browser extensions (or the browser itself) may alter the native
client's functions. Further, devices may trust untrustworthy certificates, compromising the confidentiality and
integrity benefits
[Transport Layer Security (TLS)](https://www.cloudflare.com/learning/ssl/transport-layer-security-tls/)
encryption provides.

The native Khaopiak client works to combat the risk of compromised download devices through rendering certain content
types, such as PDF files, in browser when possible instead of forcing the client to download them, which can help
prevent threat actors from acquiring downloaded files.

</details>

<details style="border: 1px solid; border-radius: 8px; padding: 8px; margin-top: 4px;">
<summary>🌎 Dependency on third-party DNS resolvers</summary>

The native Khaopiak client relies on a [Domain Name System (DNS)](https://www.cloudflare.com/learning/dns/what-is-dns/)
query to retrieve a public key for the Khaopiak server, stored in text format. The web-based client retrieves DNS
records through employing a [DNS over HTTPS (DoH)](https://datatracker.ietf.org/doc/html/rfc8484) server; however, this
adds a party now capable of identifying the fact that a client is using a Khaopiak server with PKI and potentially have
the ability to alter the key.

The native Khaopiak client mitigates this risk through generating a BIP39 mnemonic of the public key's hash, making it
easier for end-users to visually verify it is correct. Further, end-users may elect to use multiple DNS resolvers for
cross-reference purposes and improve confidence in the public key's correctness.

As the public key is only used for PKI encryption to the Khaopiak server, the client remains dependent on the DNS server
configured for their browser or device to determine the internet location of the Khaopiak server, making it such that
alteration of the public key does not inherently lead to compromises in data confidentiality nor integrity. This is
further reinforced by Khaopiak's client-side mnemonics, providing end-to-end encryption even without leveraging PKI.

</details>

## Get started

To initially deploy Khaopiak:

1. Sign up for [Cloudflare Workers](https://workers.dev). The free tier is more than enough for most use cases.
2. Clone this project and install dependencies with `npm install`
3. Run `npx wrangler login` to login to your Cloudflare account in wrangler
4. Run `npx wrangler r2 bucket create khaopiak` to create a Cloudflare R2 bucket
5. Run `npx wrangler queues create khaopiak` to create a Cloudflare Queues queue 6
    - For free tier users, disable queues in `wrangler.toml`
6. Optionally, set a Cloudflare R2 lifecycle rule to automatically delete files over
   time: `npx wrangler r2 bucket lifecycle add khaopiak --expire-days 50`
7. Run `openssl rand -hex 8 | npx wrangler secret put OBJECT_KEY_SECRET` to generate and set a random object key hmac
   secret
8. Run `npx wrangler secret put PRIVATE_KEY_HEX` and set a private key hex value
9. Run `npx wrangler secret put PUBLIC_KEY_HEX` and set a public key hex value
10. Run `npx wrangler deploy` to publish the API to Cloudflare Workers

To deploy an updated version of Khaopiak:

1. Run `npx wrangler deploy` to publish the API to Cloudflare Workers

### Recommended headers

> [!IMPORTANT]  
> To prevent access issues, including for other applications using the same domain, review headers before setting them.
> You may consider setting them using This may be done
> through [Transform Rules](https://developers.cloudflare.com/rules/transform/response-header-modification/). For
> instance, public installations may consider altering `Clear-Site-Data` to reduce inbound requests. This example will
> be altered at a later date when service workers are supported.

> [!TIP]
> When using [Transform Rules](https://developers.cloudflare.com/rules/transform/response-header-modification/) to set
> the `Access-Control-Allow-Origin` header for multiple sites, consider making it
> dynamically-valued: `concat("https://", http.host)`

```
Access-Control-Allow-Origin: <origin>
Cache-Control: no-store
Clear-Site-Data: "*"
Content-Security-Policy: default-src 'none'; base-uri 'none'; script-src 'self'; form-action 'self'; script-src-attr 'none'; connect-src 'self'; style-src-elem 'self'; style-src 'self'; style-src-attr 'none'; frame-ancestors 'none'; upgrade-insecure-requests
Cross-Origin-Embedder-Policy: require-corp
Cross-Origin-Opener-Policy: same-origin
Cross-Origin-Resource-Policy: same-site
Permissions-Policy: microphone=(), camera=(), geolocation=()
Referrer-Policy: no-referrer
Strict-Transport-Security: max-age=63072000; includeSubDomains; preload
Vary: Origin
X-Content-Type-Options: nosniff
```

### Examples

#### Khaopiak server cURL examples

> [!WARNING]  
> Endpoints that require sending a mnemonic to the server should **only** send server-generated mnemonics, and not ones
> generated locally, which can compromise end-to-end encryption. All examples do not use the Khaopiak server's PKI
> encryption, which may lead to internal data leakage if your device trusts an untrustworthy root certificate.

##### Uploading files

> [!IMPORTANT]  
> This example does not leverage client-side encryption. Encrypt sensitive files before transmitting them using this
> command.

Request:

```shell
curl -X 'POST' \
  'https://khaopiak/api/file/upload' \
  -H 'accept: application/json' \
  -H 'Content-Type: multipart/form-data' \
  -F 'file=@/home/username/Desktop/file.pdf' \
  -F 'entropy=128' \
  -F 'expiry=43200' \
  -F 'padding=1024'
```

Response:

```json
{
	"success": true,
	"mnemonic": "badge knife trim glimpse solution chaos nasty that quarter angle marine sniff"
}
```

##### Downloading files

> [!NOTE]  
> If the file was encrypted client-side before uploading, this command will not fully decrypt it.

Request:

```shell
curl -X 'POST' \
  'https://khaopiak/api/file/download?noRender=false' \
  -H 'accept: application/octet-stream' \
  -H 'Content-Type: multipart/form-data' \
  -F 'mnemonic=badge knife trim glimpse solution chaos nasty that quarter angle marine sniff' \
  --output "/home/username/Desktop/file.pdf"
```

##### Checking if files exist

Request:

```shell
curl -X 'POST' \
  'https://khaopiak/api/file/exists' \
  -H 'accept: application/json' \
  -H 'Content-Type: multipart/form-data' \
  -F 'mnemonic=badge knife trim glimpse solution chaos nasty that quarter angle marine sniff'
```

Response:

```json
{
	"success": true,
	"exists": true
}
```

##### Deleting a file

Request:

```shell
curl -X 'POST' \
  'https://khaopiak/api/file/delete' \
  -H 'accept: application/json' \
  -H 'Content-Type: multipart/form-data' \
  -F 'mnemonic=badge knife trim glimpse solution chaos nasty that quarter angle marine sniff'
```

Response:

```json
{
	"success": true
}
```

## Contributing

### Development environment

> [!NOTE]  
> Khaopiak is in active, heavy development. This guide will be updated at a later time when the project becomes more
> stable.

1. Run `npx wrangler dev` to start a local instance of the API.
2. Open `http://localhost:8787/api` in your browser to see the Swagger interface where you can try the endpoints.
3. Changes made in the `src/` folder will automatically trigger the server to reload, you only need to refresh the
   Swagger interface.

### To do

- [ ] POSIX-based upload/download script
- [ ] Web portal
- [ ] Encrypt-then-MAC CBC client-side
 readmeEtag: '"8e4ceddfecc5f6e42c270fec472135a60b4d9efe"' readmeLastModified: Sat, 11 Jan 2025 19:28:59 GMT repositoryId: 910366977 description: >- An end-to-end encrypted, account-less, and serverless file transfer system. created: '2024-12-31T05:02:01Z' updated: '2025-01-11T19:29:06Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: Jayson-Fong logo: https://avatars.githubusercontent.com/u/32722526?v=4 license: MIT repoEtag: '"241f9396535733805d1e55ad8bcba67f39391362251d8b49081c4a78f005afb6"' repoLastModified: Sat, 11 Jan 2025 19:29:06 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/lordent/openapi-doc v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWRvYwpPcGVuQVBJIHYzIGRvYyBkZWNvcmF0b3JzIGZvciBhc3luYyBzYW5pYyB3ZWJmcmFtZWZvcmsK readmeEtag: '"f3c1d56ecc056bff36106ef7a17268fda52a5c5a"' readmeLastModified: Thu, 29 Mar 2018 14:50:33 GMT repositoryId: 118631521 description: OpenAPI v3 doc decorators created: '2018-01-23T15:44:55Z' updated: '2018-03-29T14:50:45Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: lordent logo: https://avatars.githubusercontent.com/u/2070895?v=4 license: MIT repoEtag: '"b25338d94e593bf8706b66723995b8eb903a00c5d2165a9701c603dd5d691ebf"' repoLastModified: Thu, 29 Mar 2018 14:50:45 GMT foundInMaster: true category: - Server - Parsers id: 58b6dd27f3974aae0e40f151d315bd7a - source: openapi3 tags repository: https://github.com/magnayn/belvedere v3: true repositoryMetadata: base64Readme: >- IyBCZWx2ZWRlcmUKCj4gQSBidWlsZGluZywgb3IgYXJjaGl0ZWN0dXJhbCBmZWF0dXJlIG9mIGEgYnVpbGRpbmcsIGRlc2lnbmVkIGFuZCBzaXR1YXRlZCB0byBsb29rIG91dCB1cG9uIGEgcGxlYXNpbmcgc2NlbmUuCgojIyBXcml0ZSBPcGVuQVBJIHNwZWNpZmljYXRpb25zIGEgYmV0dGVyIHdheQoKKiogUGxlYXNlIG5vdGUgdGhhdCB0aGlzIGlzLCB2ZXJ5IG11Y2gsIGEgd29yayBpbiBwcm9ncmVzcyBhbmQgbm90IGEgZmluaXNoZWQgYXJ0aWNsZSEgKioKCiMjIERvY3VtZW50YXRpb24KW0luZGV4XShkb2NzL1JFQURNRS5tZCkKCgojIyMgT3BlbkFQSQoKT3BlbkFQSSBpcyBhbiBleHRyZW1lbHkgdXNlZnVsIHdheSB0byBkb2N1bWVudCB0aGUgX3NwZWNpZmljYXRpb25fIGZvciBhbiBBUEkgLSBwYXJ0aWN1bGFybHkgYmVjYXVzZSBpdCBjcmVhdGVzIGFuIGltcGxlbWVudGF0aW9uLW5ldXRyYWwgYXJ0aWZhY3QgdGhhdCBkb2N1bWVudHMgYm90aCB0aGUgY2FsbHMgYW5kIHBhcmFtZXRlcnMsIGJ1dCBhbHNvIGV4YW1wbGVzIGFuZCB1c2VmdWwgZG9jdW1lbnRhdGlvbi4KClRoZXJlIGFyZSBtYXkgZ3JlYXQgZ2VuZXJhdG9ycyAtIGRvaW5nIHRoZSB3b3JrIG9mIGltcGxlbWVudGluZyB0byBzcGVjIGluIGEgcmVsaWFibGUgd2F5LCBhbmQgZG9jdW1lbnRhdGlvbiAvIGV4cGxvcmF0aW9uIHRvb2xzLCB0aGF0IGFsbG93IHlvdSB0byBjb21tdW5pY2F0ZSB5b3VyIEFQSSB0byBhIHdpZGVyIGF1ZGllbmNlLgoKKkhvd2V2ZXIqIC0gYXMgYW4gX2F1dGhvcmluZ18gc29sdXRpb24gaXQgbGVhdmVzIG11Y2ggdG8gYmUgZGVzaXJlZC4gCgotIFRoZSB0d28gbmF0aXZlIGZvcm1hdHMgLSBKU09OIGFuZCBZQU1MIC0gYXJlIGhhcmQgdG8gd29yayB3aXRoLgoKWW91IGVuZCB1cCBlaXRoZXIgcGxheWluZyAnaHVudCB0aGUgY3VybHkgYnJhY2Ugb3IgbWlzc2luZyBxdW90ZScgb3IgZXJyb3JzIGluIGdldHRpbmcgdGhlIGluZGVudGF0aW9uIHdyb25nIGluIFlBTUwuCgotIFRoZSBuYXRpdmUgZm9ybWF0cyBhcmUgdmVyeSB2ZXJib3NlCgpUaGlzIGlzIGluZXZpdGFibGUgZ2l2ZW4gdGhhdCB0aGV5IGFyZSBsaWdodHdlaWdodCBzZXJpYWxpemF0aW9ucyB0byBhbiB1bmRlcmx5aW5nIGRhdGEgbW9kZWwuIEJ1dCwgcGFydGljdWxhcmx5IGZvciBkYXRhIHN0cnVjdHVyZXMsIGl0IGZlZWxzIGxpa2UgYSBsb3Qgb2YgbGluZXMgYXJlIHJlcXVpcmVkIHRvIHNwZWNpZnkgZXZlbiB2ZXJ5IHNpbXBsZSBEVE9zLgoKLSB2ZXJ5IGhhcmQgdG8gYnJlYWsgYXBhcnQgQVBJcyBpbnRvIG11bHRpcGxlIGZpbGVzIGFuZC9vciByZS11c2UgZXhpc3RpbmcgY29tcG9uZW50cy4gCgpUaGVyZSBhcmUgc29tZSB3YXlzIG9mIHJlZmVyZW5jaW5nIGl0ZW1zIHRocm91Z2ggWUFNTCBmaWxlIHJlZmVyZW5jZXMsIGJ1dCB0aGlzIGlzIGxpbWl0ZWQgKHBsYWNlcyB3aGVyZSBhICRyZWYgaXMgYWxsb3dlZCkuIEV2ZW4gd2hlbiBub3QgbmVlZGluZyByZS11c2UsIGl0IHdvdW9sZCBiZSBuaWNlIHRvIGJlIGFibGUgdG8gc3BsaXQgdXAgbGFyZ2UgQVBJcyBpbnRvIHNlcGFyYXRlIHNlY3Rpb25zLgoKLSBzdWZmZXJzIGZyb20gYSBsb3Qgb2YgJ0RSWScgLSBkb24ndCByZXBlYXQgeW91cnNlbGYuIAoKRS5nOiBJZiBhbiBBUEkgaGFzIGEgc3RhbmRhcmQgcmVzcG9uc2UgdG8gYSByZXF1ZXN0IHdoaWNoIGlzICc0MDQnLCB0aGVuIHRoaXMgbWF5IGJlIHJlcGVhdGVkIG1hbnkgdGltZXMgb3ZlciAod2hpY2ggbWFrZXMgaXQgaGFyZGVyIHRvIG1ha2UgZ2xvYmFsIGNoYW5nZXMpLgoKLSBoYXJkIGlmIHlvdSB3YW50IG11bHRpcGxlIHZhcmlhbnRzCgpFLmc6IERlcGxveWluZyBpbnRvIGEgZnJvbnQtZW5kIGdhdGV3YXkgKGUuZyBBV1MgQVBJIEdhdGV3YXkpIG1heSByZXF1aXJlIGFkZGl0aW9uYWwgZXh0ZW5zaW9uIHZhbHVlcyBpbiB0aGUgT3BlbkFQSSBkZWZpbml0aW9uLiBJZGVhbGx5IHdlJ2QgbGlrZSB0byBhdXRob3IgdGhlc2UgaW4gb25lIHBsYWNlLCBidXQgd2UgZG9uJ3QgcGFydGljdWxhcmx5IHdhbnQgdGhlc2UgdG8gYmUgYSBwYXJ0IG9mIHRoZSBBUEkgZGVmaW5pdGlvbiB3ZSBnaXZlIHRvIHRoZSBwdWJsaWMuIEFsc28sIHdlIG1heSB3aXNoIHRvIGhhdmUgJ3B1YmxpYycgYW5kICdwcml2YXRlJyBtZXRob2RzIHdpdGhpbiB0aGUgc2FtZSBBUEkgLSBhdXRob3JlZCBpbiB0aGUgc2FtZSBwbGFjZSwgYnV0IHB1Ymxpc2hlZCBhcyB0d28gc2VwYXJhdGUgdmFyaWFudHMuCgojIyMgT25lIGFsdGVybmF0aXZlCgpPbmUgYWx0ZXJuYXRpdmUgd291bGQgYmUgdG8gZGVzaWduIHRoZSBBUEkgaW4gYSBzcGVjaWZpYyBsYW5ndWFnZSAoc2F5LCBKYXZhKSwgdGhlbiB1c2UgdGhlIHRvb2xpbmcgKHdoaWNoIGV4aXN0cyBhbHJlYWR5KSB0byBhdXRvLWRvY3VtZW50ICJ3aGF0IGl0IHNlZXMiLgoKVGhpcyBtYXkgaGF2ZSBpdCdzIHVzZXMgLSBhbmQgaW5kZWVkIHRoaXMgdG9vbCBtYXkgYmUgZXh0ZW5kZWQgdG8gYWxsb3cgZGVmaW5pdGlvbnMgaW4gbXVsdGlwbGUgZm9ybWF0cyAtIGJ1dCwgaWYgeW91IGxvb2sgYXQgdGhlIGNvZGUgZ2VuZXJhdGVkIGZvciwgc2F5LCBKYXZhIC0gaXQgaXMgdmVyeSBhbm5vdGF0aW9uLWhlYXZ5LCBhbmQgaXRzZWxmIG5vdCBuZWNjZXNhcmlseSBhIHBsZWFzYW50IHdheSB0byBfYXV0aG9yXyBzcGVjaWZpY2F0aW9uLgoKIyMgQmVsdmVkZXJlCgpCZWx2ZWRlcmUgbGV2ZXJhZ2VzIGdyb292eSB0byBkZWZpbmUgYSBEU0wgKERvbWFpbiBTcGVjaWZpYyBMYW5ndWFnZSkgZm9yIHNwZWNpZnlpbmcgT3BlbkFQSSBzdHJ1Y3R1cmVzLiBUaHVzLCB0aGUgaW5wdXQgaXMgJ0JlbHZlZGVyZSBEU0wnLCBhbmQgdGhlIG91dHB1dCBpcyAnT3BlbkFQSSBZQU1MJy4gCgpTaW5jZSBpdCBpcyBidWlsdCBhcyBhIERTTCwgdGhpcyBlbmFibGVzIGxhbmd1YWdlIGZlYXR1cmVzIHRvIHNpbXBsaWZ5IEFQSSBzcGVjaWZpY2F0aW9ucywgYW5kIGV2ZW4gYWRkIG1ldGFwcm9ncmFtbWluZyBpbiBtb3JlIGNvbXBsZXggc2NlbmFyaW9zLgoKVGhlIERTTCBpcyBkaXJlY3RseSBtYXBwZWQgdG8gdGhlIHVuZGVybHlpbmcgT3BlbkFQSSBtb2RlbCAtIHNvIGl0IHNob3VsZCBiZSB2ZXJ5IGZhbWlsaWFyIHRvIGV4aXN0aW5nIGF1dGhvcnMuIFRoZXJlIHdpbGwgYWxzbyBiZSBjb252ZXJ0ZXJzIHRvIGFjY2VwdCBPcGVuQVBJIEFQSSBkZWZpbml0aW9ucyBhbmQgY29udmVydCB0aGVzZSB0byB0aGUgRFNMIGF1dGhvcmluZyBmb3JtYXQuCgojIyMgRXhhbXBsZXMKClRoaXMgWUFNTDoKYGBgeWFtbAppbmZvOgogIHZlcnNpb246IDEuMC4wCiAgdGl0bGU6IFN3YWdnZXIgUGV0c3RvcmUKICBsaWNlbnNlOgogICAgbmFtZTogTUlUCmBgYAogICAgCmlzIGVxdWl2YWxlbnQgdG8gdGhpcyBEU0wKCmBgYGdyb292eQppbmZvIHsgICAgICAgIAogICAgICAgIHRpdGxlICJTd2FnZ2VyIFBldHN0b3JlIgogICAgICAgIHZlcnNpb24gIjEuMC4wIgogICAgICAgIGxpY2Vuc2UgewogICAgICAgICAgICBuYW1lICJNSVQiCiAgICAgICAgfQogICAgfQpgYGAgICAgCiAgICAKIyMgRGVmaW5pbmcgYSBwYXRoOgoKYGBgZ3Jvb3Z5CgpwYXRoKCIvcGV0cy97aWR9IikgewogICAgICAgCiAgICAgICAgb3BlcmF0aW9uKE9wZXJhdGlvblR5cGUuR0VULCAnZ2V0UGV0JykgewogICAgICAgICAgICByZXNwb25zZSgnMjAwJykgewogICAgICAgICAgICAgICAgZGVzY3JpcHRpb24gIlBldCBzdWNjZXNzZnVsbHkgZm91bmQiCiAgICAgICAgICAgICAgICBjb250ZW50KCdhcHBsaWNhdGlvbi9qc29uJykgewogICAgICAgICAgICAgICAgICAgIHNjaGVtYSB7CiAgICAgICAgICAgICAgICAgICAgICAgIHJlZiBzY2hlbWE6ICdQZXQnCiAgICAgICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9CiAgICAgICAgfQogIC8qIC4uIG1vcmUgQVBJcyAuLiAqLwp9CmBgYAoKIyMgQXZvaWRpbmcgRFJZCgpZb3UgY2FuIGRlZmluZSBmdW5jdGlvbnMsIHRoZW4gZXZhbHVhdGUgdGhlbSBpbiBldmVyeSBkZWNsYXJlZCBvcGVyYXRpb24gdG8gYXZvaWQgcmVwZWF0ZWQgc3BlY2lmaWNhdGlvbi4gRS5nOgoKYGBgZ3Jvb3Z5CnBhdGgoIi9wZXRzL3tpZH0iKSB7CgogICAgICAgIGNvbW1vbl9wYXJhbWV0ZXJzID0gewogICAgICAgICAgICB0YWdzICJQZXRzIgoKICAgICAgICAgICAgcGFyYW1ldGVyKGlkOiBTdHJpbmcsIGluOiAncGF0aCcpIHsKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uICJUaGUgSUQgb2YgdGhlIHBldCIKICAgICAgICAgICAgfQoKICAgICAgICAgICAgcmVzcG9uc2UoJzQwNCcpIHsKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uICJUaGUgcGV0IHdhcyBub3QgZm91bmQiCiAgICAgICAgICAgIH0KCgogICAgICAgIH0KCiAgICAgICAgb3BlcmF0aW9uKE9wZXJhdGlvblR5cGUuR0VULCAnZ2V0UGV0JykgewogICAgICAgICAgICBldmFsdWF0ZSBjb21tb25fcGFyYW1ldGVyczsgLy8gPC0tIEluY2x1ZGUgZXZlcnl0aGluZyBhYm92ZQoKICAgICAgICAgICAgcmVzcG9uc2UoJzIwMCcpIHsKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uICJQZXQgc3VjY2Vzc2Z1bGx5IGZvdW5kIgogICAgICAgICAgICAgICAgY29udGVudCgnYXBwbGljYXRpb24vanNvbicpIHsKICAgICAgICAgICAgICAgICAgICBzY2hlbWEgewogICAgICAgICAgICAgICAgICAgICAgICByZWYgc2NoZW1hOiAnUGV0JwogICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgIH0gICAKfQpgYGAKCiMjIEluY2x1ZGluZyBmaWxlcwoKRmlsZXMgY2FuIGJlIGluY2x1ZGVkIHRocm91Z2ggdGhlaXIgcmVsYXRpdmUgcGF0aDoKCmBgYGdyb292eQogICBjb21wb25lbnRzIHsKICAgICAgICBpbmNsdWRlICdzY2hlbWEvY29tbW9uL0FzeW5jUmVzcG9uc2Uuc2NoZW1hJyAvLyBJbmNsdWRlZCBmaWxlCiAgICAgICAgCiAgICAgICAgLy8gT3Igc3BlY2lmeSBkaXJlY3RseSBoZXJlCiAgICAgICAgc2NoZW1hKCdUaGluZycpIHsKICAgICAgICAgICAgcmVxdWlyZWQgc2NoZW1hIChpZDpTdHJpbmcpIHsKICAgICAgICAgICAgICAgIGRlc2NyaXB0aW9uICJJZGVudGlmaWVyIGZvciB0aGlzIHRoaW5nIgogICAgICAgICAgICB9CgogICAgICAgICAgICBzY2hlbWEoJ3JlYXNvbic6U3RyaW5nKSB7CiAgICAgICAgICAgICAgICBkZXNjcmlwdGlvbiAiUmVhc29uIGZvciB0aGUgcmVxdWVzdCIKICAgICAgICAgICAgfQoKICAgICAgICAgICAgc2NoZW1hKCdyZXF1aXJlbWVudHMnKSB7CiAgICAgICAgICAgICAgICBzY2hlbWEobmFtZTpTdHJpbmcpCiAgICAgICAgICAgIH0KCiAgICAgICAgfQogICB9CmBgYCAgIAoKIyMjIFF1aWNrc3RhcnQ6CgoKIyMjIyBVc2UgdGhyb3VnaCBkb2NrZXIKCiogQ29udmVydGluZyBhIHNpbmdsZSwgaXNvbGF0ZWQgZmlsZSAobm90IHN1aXRhYmxlIGlmIHlvdSB1c2UgaW5jbHVkZXMpCgpgYGBiYXNoIApkb2NrZXIgcnVuIC1pIG1hZ25heW4vYmVsdmVkZXJlIGNvbnZlcnQgLSA8ICB+L215ZmlsZS5hcGkgID4gfi9teWZpbGUueWFtbApgYGAKCiogQ29udmVydGluZyBhIGZpbGUKCmBgYGJhc2gKZG9ja2VyIHJ1biAtdiB+L2FwaTovYXBpIC1pIG1hZ25heW4vYmVsdmVkZXJlIGNvbnZlcnQgLWYgL2FwaS9jcnVkMS5hcGkgPiB+L2FwaS9jcnVkMS55YW1sCmBgYAoK readmeEtag: '"a94e4fab266465b754d41e2020bed730dff92543"' readmeLastModified: Fri, 21 Feb 2020 11:35:05 GMT repositoryId: 202089566 description: Write OpenAPI specifications an easier way created: '2019-08-13T07:31:50Z' updated: '2020-02-21T11:35:18Z' language: Groovy archived: false stars: 0 watchers: 1 forks: 1 owner: magnayn logo: https://avatars.githubusercontent.com/u/53653?v=4 license: Apache-2.0 repoEtag: '"46ad213dbac56ecda51e5187b6e65e1e5b214a25ea2e650addf292e199681f27"' repoLastModified: Fri, 21 Feb 2020 11:35:18 GMT foundInMaster: true category: Parsers id: 01f3b0bfcfe34c56388b7931774c15a3 - source: openapi3 tags repository: https://github.com/britishgas-engineering/openapi3-joiner v3: true repositoryMetadata: base64Readme: >- IyBPcGVuIEFQSSAzIGpvaW5lcgpOb2RlIHNjcmlwdCB0byBtZXJnZSBtdWx0aXBsZSBPcGVuIEFQSSAzIHNwZWNpZmljYXRpb25zIGludG8gc2luZ2xlIHNwZWNpZmljYXRpb24KCkluIGEgbWljcm9zZXJ2aWNlIGFyY2hpdGVjdHVyZSwgZWFjaCBtaWNyb3NlcnZpY2UgaGFzIGl0cyBvd24gQVBJIGRvY3VtZW50YXRpb24uCk9mdGVuIHRoZSBhcGlzIGFyZSBleHBvc2VkIGJlaGluZCBhbiBBUEkgZ2F0ZXdheSB0byBnaXZlIHNpbmdsZSBpbnRlcmZhY2UgdG8gdGhlIGNsaWVudHMuClRoYXQgY3JlYXRlcyBhIG5lZWQgdG8gaGF2ZSBhIHNpZ2xlIEFQSSBzcGVjaWZpY2F0aW9uLgoKVGhpcyBzY3JpcHQgaXMgYSB0b29sIHRvIG1lcmdlIG11bHRpcGxlIE9wZW4gQVBJIDMgc3BlY2lmaWNhdGlvbnMgKGpzb24gb3IgeWFtbCkgCmludG8gYSBzaW5nbGUgc3BlY2lmaWNhdGlvbgoKIyBIb3cgdG8gcnVuPwpgbnBtIGluc3RhbGxgCgpgbm9kZSBpbnZva2UuanNgCihQbGVhc2UgcmVmZXIgW2ludm9rZS5qc10oLi9pbnZva2UuanMpIGZvciBtb3JlIGRldGFpbHMpCgo= readmeEtag: '"bee7a1f0b50224fa98824fe5cdc94275b5a2d7ec"' readmeLastModified: Fri, 15 Feb 2019 15:19:00 GMT repositoryId: 170882510 description: >- Node script to merge multiple Open API 3 specifications into single specification created: '2019-02-15T15:03:48Z' updated: '2019-11-08T11:17:52Z' language: JavaScript archived: false stars: 0 watchers: 3 forks: 0 owner: britishgas-engineering logo: https://avatars.githubusercontent.com/u/23475527?v=4 license: MIT repoEtag: '"4066a4befd415b976eba288fdec629ca2b79e3a2a48445c7656f54e5d644007b"' repoLastModified: Fri, 08 Nov 2019 11:17:52 GMT foundInMaster: true category: - Documentation - Parsers - Server Implementations id: 902a9f21d76febce5d2241c568a39909 - source: openapi3 tags repository: https://github.com/zemanlx/openapi-spec-validator v3: true repositoryMetadata: base64Readme: >- IyBEb2NrZXIgaW1hZ2UgZm9yIG9wZWFuYXBpLXNwZWMtdmFsaWRhdG9yCgpJbWFnZXMgYXJlIGJhc2VkIG9uIFB5dGhvbiBvbiBBbHBpbmUgYW5kIGNvbnRhaW5zIG9ubHkgYWRkZWQgW29wZW5hcGktc3BlYy12YWxpZGF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9wMWMydS9vcGVuYXBpLXNwZWMtdmFsaWRhdG9yKS4KClRhZ3MgYXJlIGNvcHlpbmcgdmFsaWRhdG9yIHZlcnNpb24uCgojIyBVc2FnZQoKTGV0cyBzYXkgdGhhdCB5b3VyIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBpcyBpbiB5b3VyIGN1cnJlbnQgZm9sZGVyIGluIGZvbGRlciBgc3BlY2AgaW4gZmlsZSBgb3BlbmFwaS55YW1sYC4gVG8gdmFsaWRhdGUgaXQgcnVuCgpgYGBiYXNoCmRvY2tlciBydW4gXAogIC0tcm0gXAogIC0tdm9sdW1lICR7UFdEfS9zcGVjOi9zcGVjIFwKICB6ZW1hbmx4L29wZW5hcGktc3BlYy12YWxpZGF0b3I6MC4yLjAgXAogIC9zcGVjL29wZW5hcGkueWFtbApgYGAK readmeEtag: '"01c77261c4619ab000864863fc22b4e89c22c75b"' readmeLastModified: Fri, 18 May 2018 12:55:41 GMT repositoryId: 133953552 description: Docker image for opeanapi-spec-validator created: '2018-05-18T12:48:32Z' updated: '2018-05-18T14:14:49Z' language: Shell archived: false stars: 0 watchers: 1 forks: 0 owner: zemanlx logo: https://avatars.githubusercontent.com/u/18702153?v=4 license: MIT repoEtag: '"e7d7a147665853b3ad002f9767bfeba80675e2ff3a1c9dcc48e4a26d6379f20b"' repoLastModified: Fri, 18 May 2018 14:14:49 GMT foundInMaster: true category: - Description Validators - Parsers id: 9457be94322926df039e28ee1f8dfa63 - source: openapi3 tags repository: https://github.com/shaswatsaxena/task-management v3: true repositoryMetadata: base64Readme: >- IyMgRGVzY3JpcHRpb24KClRhc2sgTWFuYWdlbWVudCBBUEkuCgojIyBJbnN0YWxsYXRpb24KCmBgYGJhc2gKJCBucG0gaW5zdGFsbApgYGAKCi0gUmVxdWlyZXMgUG9zdGdyZVNRTCBpbnN0YW5jZSBydW5uaW5nCi0gUHJvdmlkZSB2YWx1ZXMgdG8gYWxsIHNhbXBsZS5lbnYgdmFyaWFibGVzCi0gUmVuYW1lIHNhbXBsZS5lbnYgdG8gLmVudgoKIyMgUnVubmluZyB0aGUgYXBwCgpgYGBiYXNoCiMgZGV2ZWxvcG1lbnQKJCBucG0gcnVuIHN0YXJ0CgojIHdhdGNoIG1vZGUKJCBucG0gcnVuIHN0YXJ0OmRldgoKIyBwcm9kdWN0aW9uIG1vZGUKJCBucG0gcnVuIHN0YXJ0OnByb2QKYGBgCgojIyBUZXN0CgpDcmVhdGUgYSBkYXRhYmFzZSBuYW1lZCAiZTJlX3Rlc3QiIHRvIGJlIHVzZWQgZm9yIHRlc3RpbmcuCgpgYGBiYXNoCiMgZW5kIHRvIGVuZCB0ZXN0cwokIG5wbSBydW4gdGVzdApgYGAKCi0gQXV0aG9yIC0gU2hhc3dhdCBTYXhlbmEK readmeEtag: '"852e77d5c4fd229bb6e650486e50919dc3503886"' readmeLastModified: Thu, 04 Feb 2021 16:25:01 GMT repositoryId: 266953057 description: 'Task Management API ' created: '2020-05-26T05:32:20Z' updated: '2021-04-10T14:14:21Z' language: TypeScript archived: false stars: 0 watchers: 0 forks: 0 owner: shaswatsaxena logo: https://avatars.githubusercontent.com/u/12628996?v=4 repoEtag: '"69107ebc576427e2fa67963540ab771a4d09bb1657836c05c6732868d707193a"' repoLastModified: Sat, 10 Apr 2021 14:14:21 GMT foundInMaster: true category: - Converters - Server Implementations id: 01c6c5eceb2b5bf710c27e0da567911f - source: openapi3 tags repository: https://github.com/eoinkane/penny-calc-backend v3: true repositoryMetadata: base64Readme: >- IyBfX1Blbm55LUNhbGMgQmFja2VuZF9fCgpUaGlzIHByb2plY3QgaGVscHMgYSB1c2VyIGNvbXBsZXRlIHRoZSAxcCBzYXZpbmcgY2hhbGxlbmdlczoKCiogVGhlIDFwIFNhdmluZ3MgQ2hhbGxlbmdlClsqbW9yZSBpbmZvKl0oaHR0cHM6Ly9tb256by5jb20vYmxvZy8xcC1zYXZpbmdzLWNoYWxsZW5nZS0yMDIwICdsaW5rJykgIAoqIFRoZSBSZXZlcnNlIDFwIFNhdmluZ3MgQ2hhbGxlbmdlClsqbW9yZSBpbmZvKl0oaHR0cHM6Ly9tb256by5jb20vYmxvZy8yMDE5LzA2LzA0L3JldmVyc2UtMXAtc2F2aW5ncy1jaGFsbGVuZ2UtbW9uem8gJ2xpbmsnKQoKX19UaGlzIHByb2plY3QgaGFzIDIgc2VjdGlvbnM6X18gIAoKKiBfX0JhY2tlbmQgQ29kZV9fCiogX19JbmZyYXN0cnVjdHVyZSBhcyBDb2RlIChDREspX18KCiMjIF9fQmFja2VuZCBDb2RlX18KClRoZSBsb2dpYyBjb2RlIGlzIHdyaXR0ZW4gaW4gcHl0aG9uIGFuZCBzdG9yZWQgaW4gYC4vYXBwYAoKIyMjIF9fTGlzdCBvZiBTY3JpcHRzX18KCiogX19gYW1vdW50L3VudGlsL25leHRfcGF5ZGF5YF9fCgogICAgVGhpcyBsYW1iZGEgY2FsY3VsYXRlcyBob3cgbWFueSBkYXlzIHRpbGwgbmV4dCBwYXlkYXkgdGhlbiByZXR1cm5zIHRoZSBhbW91bnQgdGhhdCBuZWVkcyB0byBiZSBzYXZlZCBiZXR3ZWVuIG5vdyBhbmQgdGhlbiAgCiAgICBUaGlzIGludm9sdmVzIG11bHRpcGx5aW5nIHRoZSBudW1iZXIgZGF5cyBieSB0aGUgZGFpbHkgc2F2aW5nIGFtb3VudCAoMy42NikgIAogICAgVGhlIGZpbGUgaXMgYGFwcC9zcmMvYW1vdW50L3VudGlsL25leHRfcGF5ZGF5L2FwcC5weWAgIAogICAgVGhlIGVudHJ5cG9pbnQgaXMgYGxhbWJkYV9oYW5kbGVyYAoKIyMgX19JbmZyYXN0cnVjdHVyZSBhcyBDb2RlIChDREspX18KClRoZSBDREsgaXMgd3JpdHRlbiBpbiBUeXBlU2NyaXB0ICAKClRoaXMgcHJvamVjdCBoYXMgYmVlbiBidWlsdCB1c2luZyBBV1MgQ0RLIHZlcnNpb24gX18xLjYxLjFfXyAgClRoZSBjZGsgY2xpIHRvb2wgcmVxdWlyZXMgYXdzIGNyZWRlbnRpYWxzLiAgCkkgdXNlIGFuIEFXUyBlZHVjYXRlIGFjY291bnQgW21vcmUgaW5mb10oaHR0cHM6Ly9hd3MuYW1hem9uLmNvbS9lZHVjYXRpb24vYXdzZWR1Y2F0ZS9zdHVkZW50cy8gJ2xpbmsnKSAgCkJ1dCBhbnkgYWNjb3VudCB0aGF0IGhhcyB0aGVzZSBjcmVkZW50aWFsczogKmF3c19hY2Nlc3Nfa2V5X2lkKjsgKmF3c19zZWNyZXRfYWNjZXNzX2tleSo7ICphd3Nfc2Vzc2lvbl90b2tlbio7IHdpbGwgZG8KCklmIHlvdSBoYXZlIG5vdCB1c2VkIENESyBpbiB5b3VyIEFXUyBhY2NvdW50IGJlZm9yZSBwbGVhc2UgcnVuIGBjZGsgYm9vdHN0cmFwYCBiZWZvcmUgc3RhcnRpbmcKCiMjIyBfX0NvbXBvbmVudHNfXwoKVGhlIENESyBpbmNsdWRlcyAyIGNvbXBvbmVudHMKCiogX19BUEkgR2F0ZXdheV9fCiogX19MYW1iZGEgRnVuY3Rpb25fXwoKIyMjIF9fTGFtYmRhIEZ1bmN0aW9uX18KClRoZSBTZXJ2ZXJsZXNzIGZ1bmN0aW9uIGFsbG93cyB1cyB0byBjYWxsIG91ciBTY3JpcHRzCgojIyMjIF9fTGlzdCBvZiBMYW1iZGFzX18gIAoKKiBfX2BwZW5ueS1jYWxjLWFtb3VudC11bnRpbC1uZXh0LXBheWRheS1sYW1iZGFgX18gIAogICAgVGhpcyBsYW1iZGEgcnVucyB0aGUgYGFtb3VudC91bnRpbC9uZXh0X3BheWRheWAgc2NyaXB0ICAKCiMjIyBfX0FQSSBHYXRld2F5X18KClRoZSBBUEkgR2F0ZXdheSBwcm92aWRlcyBhY2Nlc3MgdG8gdGhlIGJhY2tlbmQgdGhyb3VnaCBSZXN0IEFQSSBjYWxscy4gIAoKIyMjIyBfX0VuZHBvaW50c19fCgoqIF9fYC9hbW91bnQvdW50aWwvbmV4dC1wYXlkYXlgX18KICAgIFRoaXMgZW5kcG9pbnQgaXMgYSBHRVQgbWV0aG9kIGVuZHBvaW50ICAKICAgIFVzaW5nIGFuIEFXU19QUk9YWSBsYW1kYmEgaW50ZWdyYXRpb24gIAogICAgVGhpcyBlbmRwb2ludCBsaW5rcyB1cCB0byB0aGUgbGFtYmRhIG5hbWVkIGBwZW5ueS1jYWxjLWFtb3VudC11bnRpbC1uZXh0LXBheWRheS1sYW1iZGFgCgojIyBfX0NvbWFuZHNfXwoKSXQgaXMgcmVjY29tbWVuZGVkIHRvIGluc3RhbGwgdGhlIHBhY2thZ2VzIHJlcXVpcmVkICAKKnJ1biB0aGVzZSBjb21tYW5kcyB0byBnZXQgc3RhcnRlZCogIAoKKiBgUElQRU5WX1ZFTlZfSU5fUFJPSkVDVD0xIHBpcGVudiBpbnN0YWxsIC0tZGV2YCAgCiogYFBJUEVOVl9WRU5WX0lOX1BST0pFQ1Q9MSBwaXBlbnYgaW5zdGFsbGAgIAoqIGBucG0gaW5zdGFsbGAKKiBgZXhwb3J0ICQoY2F0IC5lbnYgfCB4YXJncylgCgpFdmVyeSB0aW1lIHlvdSBjaGFuZ2UgdmFsdWVzIGluIGAvLmVudmAgcnVuIGBleHBvcnQgJChjYXQgLmVudiB8IHhhcmdzKWAKCkEgbnVtYmVyIG9mIHRoZSBjb21tYW5kcyByZWx5IG9uIHRoZSBjZGsgY2xpIGFuZCBzbyBwbGVhc2UgaGF2ZSBhIHZhbGlkIHNlc3Npb24vdG9rZW4gZnJvbSBBV1MKCiMjIyBfX0xpc3Qgb2YgQ29tYW5kc19fCgoqIF9fYG5wbSBydW4gYnVpbGRgX18KICAgIFJ1bnMgdGhlIGZvbGxvd2luZyBjb21tYW5kcwoKICAqIGBidWlsZC1weXRob25gCiAgKiBgYnVpbGQtY2RrYAoKKiBfX2BucG0gcnVuIGJ1aWxkLXB5dGhvbmBfXyAgCiAgICBUaGlzIGNvbW1hbmQgYnVpbGRzIHRoZSBweXRob24gY29kZSB3aXRoIHRoZSBwYWNrYWdlcyBpbnN0YWxsZWQgaW50byB0aGUgYnVpbGQgZGlyZWN0b3J5CiogX19gbnBtIHJ1biBidWlsZC1jZGtgX18gIAogICAgUnVucyB0aGUgZm9sbG93aW5nIGNvbW1hbmRzCgogICogYGNsZWFuLWNka2AKICAqIGBjZGstc3ludGhgCgoqIF9fYG5wbSBydW4gY2RrLXN5bnRoYF9fICAKICAgIFRoaXMgY29tbWFuZCBidWlsZHMgYSBjbG91ZGZvcm1hdGlvbiB0ZW1wbGF0ZSBmcm9tIHRoZSBDREsgc3RhY2sgIAogICAgKlByZXF1aXNpdGU6IGBidWlsZC1weXRob25gKgoqIF9fYG5wbSBydW4gY2RrLWRpZmZgX18gIAogICAgVGhpcyBjb21tYW5kIHNob3dzIHRoZSBkaWZmZXJlbmNlIGJldHdlZW4gdGhlIGxhdGVzdCB0ZW1wbGF0ZSBhbmQgdGhlIHByZXZpb3VzIG9uZQoqIF9fYG5wbSBydW4gZGVwbG95YF9fICAKICAgIFRoaXMgY29tbWFuZCBkZXBsb3lzIHRoZSBjbG91ZGZvcm1hdGlvbiBzdGFjayAgCiAgICAqUHJlcXVpc2l0ZTogYGNrZC1zeW50aGAqCiogX19gbnBtIHJ1biBkZXBsb3ktY2lgX18gIAogICAgRm9sbG93cyB0aGUgc2FtZSBiZWhhdmlvdXIgYXMgYGNkay1kZXBsb3lgIGJ1dCBkb2VzIG5vdCBhc2sgZm9yIGNvbmZpcm1hdGlvbiAgCiAgICAqUHJlcXVpc2l0ZTogYGNrZC1zeW50aGAqCiogX19gbnBtIHJ1biB0ZXN0LWNka2BfXyAgCiAgICBUZXN0cyB0aGUgQ0RLIHN0YWNrCiAgICAqUmVmZXIgdG8gdGhlIFRlc3RpbmcgU2VjdGlvbioKKiBfX2BucG0gcnVuIHRlc3QtcHl0aG9uYF9fICAKICAgIFRlc3RzIHRoZSBQeXRob24gc2NyaXB0cwogICAgKlJlZmVyIHRvIHRoZSBUZXN0aW5nIFNlY3Rpb24qCiogX19gbnBtIHJ1biB0c2NgX18gIAogICAgUnVucyB0aGUgdHlwZXNjcmlwdCBjb21wbGllcgoqIF9fYG5wbSBydW4gY2xlYW5gX18gIAogICAgUnVucyB0aGUgZm9sbG93aW5nIGNvbW1hbmRzCgogICogYGNsZWFuLXRlc3RzYAogICogYGNsZWFuLWJ1aWxkYAogICogYGNsZWFuLWNka2AKCiogX19gbnBtIHJ1biBjbGVhbi10ZXN0c2BfXyAgCiAgICBSZW1vdmVzIHRlbXBvcmFyeSB0ZXN0IGZpbGVzCiogX19gbnBtIHJ1biBjbGVhbi1idWlsZGBfXyAgCiAgICBSZW1vdmVzIHRlbXBvcmFyeSBidWlsZCBmaWxlcwoqIF9fYG5wbSBydW4gY2xlYW4tY2RrYF9fICAKICAgIFJlbW92ZXMgdGVtcG9yYXJ5IGNkayBmaWxlcwoqIF9fYG5wbSBydW4gcnVuLWxhbWJkYS1sb2NhbGBfXyAgCiAgICBSdW5zIHRoZSBsYW1iZGEgbG9jYWxseQoKIyMgX19UZXN0aW5nX18KClRoZXJlIGFyZSB0ZXN0cyBmb3IgYm90aCB0aGUgY2RrIHN0YWNrIGFuZCBiYWNrZW5kIGNvZGUgIAoKKiBfX0NESyB0ZXN0c19fCiAgICBfX2BucG0gcnVuIHRlc3QtY2RrYF9fCiAgICBXcml0dGVuIGluIFR5cGVTY3JpcHQgYW5kIHVzaW5nIGplc3QgYXMgdGhlIHRlc3QgcnVubmVyCiogX19QeXRob24gdGVzdHNfXwogICAgX19gbnBtIHJ1biB0ZXN0LXB5dGhvbmBfXwogICAgV3JpdHRlbiBpbiBweXRob24gYW5kIHVzaW5nIHB5dGVzdCBhcyB0aGUgdGVzdCBydW5uZXIKCgojIyBXZWxjb21lIHRvIHlvdXIgQ0RLIFR5cGVTY3JpcHQgcHJvamVjdAoKVGhpcyBpcyBhIGJsYW5rIHByb2plY3QgZm9yIFR5cGVTY3JpcHQgZGV2ZWxvcG1lbnQgd2l0aCBDREsuCgpUaGUgYGNkay5qc29uYCBmaWxlIHRlbGxzIHRoZSBDREsgVG9vbGtpdCBob3cgdG8gZXhlY3V0ZSB5b3VyIGFwcC4KCiMjIFVzZWZ1bCBjb21tYW5kcwoKKiBgbnBtIHJ1biBidWlsZGAgICBjb21waWxlIHR5cGVzY3JpcHQgdG8ganMKKiBgbnBtIHJ1biB3YXRjaGAgICB3YXRjaCBmb3IgY2hhbmdlcyBhbmQgY29tcGlsZQoqIGBucG0gcnVuIHRlc3RgICAgIHBlcmZvcm0gdGhlIGplc3QgdW5pdCB0ZXN0cwoqIGBjZGsgZGVwbG95YCAgICAgIGRlcGxveSB0aGlzIHN0YWNrIHRvIHlvdXIgZGVmYXVsdCBBV1MgYWNjb3VudC9yZWdpb24KKiBgY2RrIGRpZmZgICAgICAgICBjb21wYXJlIGRlcGxveWVkIHN0YWNrIHdpdGggY3VycmVudCBzdGF0ZQoqIGBjZGsgc3ludGhgICAgICAgIGVtaXRzIHRoZSBzeW50aGVzaXplZCBDbG91ZEZvcm1hdGlvbiB0ZW1wbGF0ZQo= readmeEtag: '"8797870617f94a1a8ebbd2ed544618d2d823e952"' readmeLastModified: Fri, 04 Sep 2020 14:44:27 GMT repositoryId: 292794834 description: 'This project helps a user complete the 1p saving challenges:' created: '2020-09-04T08:33:56Z' updated: '2020-09-04T14:44:51Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: eoinkane logo: https://avatars.githubusercontent.com/u/10391408?v=4 repoEtag: '"6128219ccc8782099b3e06193c51aad5180b99dfd8838d2202306a8f73bb453a"' repoLastModified: Fri, 04 Sep 2020 14:44:51 GMT foundInMaster: true category: Server Implementations id: c22999086a3109365e79e2907ee01912 - source: openapi3 tags repository: https://github.com/nielsbergsma/haskell-api-prototype v3: true repositoryMetadata: base64Readme: >- IyBIYXNrZWxsIEFQSSBwcm90b3R5cGUKCiMjIFdoYXQgaXMgdGhpcz8KVGhpcyBpcyBteSByZXNlYXJjaCBpbnRvIHRoZSB3b3JsZCBvZiBIYXNrZWxsLiBTcGVjaWNpZmljbHkgaG93IHRvIGJ1aWxkIG1vZGVybiBhcHBsaWNhdGlvbnMgKGFyY2hpdGVjdHVyZXMpIHdpdGggSGFza2VsbCwgZnVuY3Rpb25hbCBzdHlsZS4gVG9waWNzIHdpbGwgaW5jbHVkZSwgYnV0IG5vdCBsaW1pdGVkIHRvOgotIHJlc3RmdWwgQVBJCi0gZG9tYWluIGRyaXZlbiBkZXNpZ24KLSBDUVJTIC8gZXZlbnQgc291cmNpbmcKLSBzcGVjaWZpY2F0aW9uIGJ5IGV4YW1wbGUKLSBhbmQgc28gb24gLi4uCgpUaGlzIHJlcG9zaXRvcnkgaXMgKGN1cnJlbnRseSkgbWVyZWx5IGEgcmVjb3JkaW5nIG9mIG15IHByb2dyZXNzLgoKIyMgV2hhdCdzIGRvbmU/IGFuZCB3aGF0J3MgbmV4dD8KLSBbeF0gbGlnaHR3ZWlnaHQgYWJzdHJhY3Rpb24gb24gdG9wIG9mIFdBSSwgYXMgYSBtaWNybyBSZXN0ZnVsIGxpYnJhcnkKLSBbeF0gYXV0by1nZW5lcmF0ZSBBUEkgYmFzZWQgb3AgT3BlbkFQSSB2MyBzcGVjCi0gW3hdIHZhbGlkYXRlIHVybCArIHF1ZXJ5IHBhcmFtZXRlcnMKLSBbeF0gdmFsaWRhdGUganNvbiBib2R5LCBiYXNlZCBvbiBPcGVuQVBJIHYzIHNwZWMKLSBbeF0gdHJhbnNmb3JtIGJvZHkgdG8gdHJhbnNmZXIgb2JqZWN0cyAoZm9yIGJvdGggcHJvZHVjdCBhbmQgc3VtIHR5cGVzKQotIFt4XSBjb21wcmVoZW5zaXZlIHZhbGlkYXRpb24gRFNMIChqc29uIGJvZHksIHVybCArIHF1ZXJ5IHBhcmFtZXRlcnMpCi0gW3hdIHN0cm9uZyB0eXBlZCBlcnJvciBtZXNzYWdlcyAoaHVtYW4gKyBtYWNoaW5lIGludGVycHJldGFibGUpLCBpbmNsdWRlcyBKU09OIHBhdGggZm9yIFJlcXVlc3QgQm9keSBlcnJvciBtZXNzYWdlcwotIFt4XSBKV1QgYXV0aG9yaXphdGlvbiAvIGF1dGhlbnRpY2F0aW9uLCByb2xlIGJhc2VkIHdpdGggSldLIHZhbGlkYXRpb24KLSBbIF0gImF1dG8gd2lyZSIgdHJhbnNmZXIgb2JqZWN0cyB0byBjb21tYW5kcwotIFsgXSBjb21tYW5kIGhhbmRsZXJzLCBzYWdhcywgcHJvY2VzcyBtYW5hZ2VycwotIFsgXSBhZ2dyZWdhdGVzLCBzdGF0ZSwgcGVyc2lzdGVuY2UKLSBbIF0gZXZlbnQsIGV2ZW50IGhhbmRsZXJzCi0gWyBdIC4uLiBtb3JlIHRvIGNvbWUKCiMjIFVzZSBvciBpbXByb3ZlPwpTdXJlLCBqdXN0IHNob290IG1lIGEgbWVzc2FnZS4= readmeEtag: '"68d393938c8d8f6c4815c954c6655f92bf103441"' readmeLastModified: Sun, 12 Apr 2020 18:10:37 GMT repositoryId: 255094421 description: haskell rest api prototype created: '2020-04-12T14:00:16Z' updated: '2020-04-12T18:10:45Z' language: Haskell archived: false stars: 0 watchers: 1 forks: 0 owner: nielsbergsma logo: https://avatars.githubusercontent.com/u/1736958?v=4 repoEtag: '"d78c7fb19b641a2cab763e8625bff2900087329e7b0cb43f2110bf2b7094dfbd"' repoLastModified: Sun, 12 Apr 2020 18:10:45 GMT foundInMaster: true category: - Parsers - Server Implementations id: 7755d8920695da9bde7e7b085bd2f691 - source: openapi3 tags repository: https://github.com/hcloward/casc-bundle-management v3: true repositoryMetadata: repositoryId: 303161948 description: >- This repository includes an OpenAPI file I created using Stoplight, while completing Tom Johson's API documentation course. created: '2020-10-11T16:19:18Z' updated: '2020-10-23T16:27:40Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: HCloward logo: https://avatars.githubusercontent.com/u/17242298?v=4 repoEtag: '"b6ccaf4686556cf438cfc8853753227854d21885f37b17ff1bca7fd4168badbe"' repoLastModified: Fri, 23 Oct 2020 16:27:40 GMT foundInMaster: true id: 61ea4572af4f152e072a62c8539d9795 - source: openapi3 tags repository: https://github.com/hoangnx30/sample-express-with-swagger v3: true repositoryMetadata: base64Readme: >- KipBIGV4YW1wbGUgYWJvdXQgdXNpbmcgc3dhZ2dlci11aSBpbiBub2RlanMgdG8gdGVzdCBhcGkqKgo= readmeEtag: '"b6c28f8d1dcfcc41f123b7c7b15d940efea6107d"' readmeLastModified: Sun, 08 Nov 2020 10:50:14 GMT repositoryId: 311039169 description: Example about express mongodb swagger created: '2020-11-08T10:41:16Z' updated: '2020-11-08T10:50:24Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: hoangnx30 logo: https://avatars.githubusercontent.com/u/43129138?v=4 repoEtag: '"5ee03264184bad13a0780f2ca745ecd646f94954ea6b1138df653bb86d78aa6b"' repoLastModified: Sun, 08 Nov 2020 10:50:24 GMT foundInMaster: true category: - Low-level Tooling - Parsers - Server Implementations id: dbbd8e27197f94d32201d8b6795a5100 - source: openapi3 tags repository: https://github.com/semestry/oas3-lint v3: true repositoryMetadata: base64Readme: >- IyBvYXMzLWxpbnQKR2l0SHViIEFjdGlvbiBmb3IgbGludGluZyBPcGVuQVBJIFNwZWNpZmljYXRpb24gMyBmaWxlcyB1c2luZyBbU3BlY3RyYWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9zdG9wbGlnaHRpby9zcGVjdHJhbCkuCgojIyBVc2FnZQpgYGB5YW1sCmpvYnM6CiAgbGludDoKICAgIG5hbWU6IExpbnQKICAgIHJ1bnMtb246IHVidW50dS1sYXRlc3QKCiAgICBzdGVwczoKICAgICAgLSBuYW1lOiBDaGVja291dAogICAgICAgIHVzZXM6IGFjdGlvbnMvY2hlY2tvdXRAdjIKCiAgICAgIC0gdXNlczogZXZlb2gvb2FzMy1saW50QHYxCiAgICAgICAgd2l0aDoKICAgICAgICAgIHNwZWM6IHBhdGgvdG8vc3BlYy55YW1sCiAgICAgICAgICB0b2tlbjogJHt7IHNlY3JldHMuR0lUSFVCX1RPS0VOIH19CmBgYAoKIyMjIE9wdGlvbnMKCiMjIyMgc3BlYyAocmVxdWlyZWQpClRoZSBwYXRoIHRvIHRoZSBzcGVjaWZpY2F0aW9uIGZpbGUgKE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLCBZQU1MKSB0aGF0IHNob3VsZCBiZSBsaW50ZWQuCgoqRXhhbXBsZToqCmBgYHlhbWwKd2l0aDoKICBzcGVjOiBwYXRoL3RvL3NwZWMueWFtbApgYGAKCiMjIyMgdG9rZW4gKHJlcXVpcmVkKQpUaGUgR2l0SHViIHRva2VuIHRoYXQgaXMgdXNlZCB0byBjcmVhdGUgdGhlIENoZWNrIFJ1biB1c2luZyB0aGUgR2l0SHViIENoZWNrcyBBUEkuClNob3VsZCBiZSBzZXQgdG8gdGhlIGBHSVRIVUJfVE9LRU5gIHNlY3JldCB0aGF0IGlzIGF1dG9tYXRpY2FsbHkgY3JlYXRlZCBieSB0aGUgR2l0SHViIEFjdGlvbnMgcnVubmVyLgogICAgICAgICAgCipFeGFtcGxlOioKYGBgeWFtbAp3aXRoOgogIHRva2VuOiAke3sgc2VjcmV0cy5HSVRIVUJfVE9LRU4gfX0KYGBgCgojIyMjIHJ1bGVzZXQKUGF0aCB0byB0aGUgcnVsZXNldCBmaWxlLgoKKkV4YW1wbGU6KgpgYGB5YW1sCndpdGg6CiAgcnVsZXNldDogcGF0aC90by8uc3BlY3RyYWwueWFtbApgYGAK readmeEtag: '"5bff17f2d76c1d82c3a5850d19327ea036ed6a44"' readmeLastModified: Sun, 01 Jan 2023 12:26:01 GMT repositoryId: 323617929 description: GitHub Action for linting OpenAPI Specification 3 files created: '2020-12-22T12:20:30Z' updated: '2023-01-05T13:18:32Z' language: JavaScript archived: true stars: 0 watchers: 3 forks: 0 owner: semestry logo: https://avatars.githubusercontent.com/u/1628298?v=4 repoEtag: '"aaef3701fbd5c3b04258e09e124c39b12a741efca800f3ef1a11764978dd7d02"' repoLastModified: Thu, 05 Jan 2023 13:18:32 GMT foundInMaster: true category: - Description Validators - Parsers id: 0c3ba285fd8078e1381b58fcabe3b0b0 - source: openapi3 tags repository: https://github.com/zeet-co/go-artifacthub v3: true repositoryMetadata: base64Readme: >- IyBnby1hcnRpZmFjdGh1YgoKQXV0byBnZW5lcmF0ZWQgdXNpbmcgaHR0cHM6Ly9naXRodWIuY29tL2RlZXBtYXAvb2FwaS1jb2RlZ2VuIGZyb20gaHR0cHM6Ly9hcnRpZmFjdGh1Yi5pby9kb2NzL2FwaS8KCiMjIFVzYWdlCgpgYGAKY2xpZW50LCBlcnIgOj0gYXJ0aWZhY3RodWIuTmV3Q2xpZW50V2l0aFJlc3BvbnNlcygiaHR0cHM6Ly9hcnRpZmFjdGh1Yi5pby9hcGkvdjEiLCBhcnRpZmFjdGh1Yi5XaXRoUmVxdWVzdEVkaXRvckZuKGZ1bmMoY3R4IGNvbnRleHQuQ29udGV4dCwgcmVxICpodHRwLlJlcXVlc3QpIGVycm9yIHsKICByZXEuSGVhZGVyLlNldCgiWC1BUEktS0VZLUlEIiwgIktFWV9JRCIpCiAgcmVxLkhlYWRlci5TZXQoIlgtQVBJLUtFWS1TRUNSRVQiLCAiS0VZX1NFQ1JFVCIpCiAgcmV0dXJuIG5pbAp9KSkKaWYgZXJyICE9IG5pbCB7CiAgcmV0dXJuIG5pbCwgZXJyCn0KCnJlc3AsIGVyciA6PSBjbGllbnQuR2V0SGVsbVBhY2thZ2VEZXRhaWxzV2l0aFJlc3BvbnNlKGN0eCwgInJlcG8iLCAicGFja2FnZSIpCmlmIGVyciAhPSBuaWwgewogIHJldHVybiBuaWwsIGVycgp9CmBgYAo= readmeEtag: '"300551ecf0f8255bb007cf0bc91d1746f91829a9"' readmeLastModified: Wed, 22 Sep 2021 19:13:21 GMT repositoryId: 409325296 description: artifacthub go client created: '2021-09-22T19:07:41Z' updated: '2021-09-22T19:16:05Z' language: Makefile archived: false stars: 0 watchers: 2 forks: 0 owner: zeet-co logo: https://avatars.githubusercontent.com/u/74479726?v=4 license: MIT repoEtag: '"724381780123193012199fe35db8e7582470b45f101e1ddc311da83c935450cd"' repoLastModified: Wed, 22 Sep 2021 19:16:05 GMT foundInMaster: true category: Data Validators id: 19d546adc3b9f0775d5d82791295eba2 - source: openapi3 tags repository: https://github.com/romabilka/generate-openapi-go-server v3: true repositoryMetadata: base64Readme: >- IyBHZW5lcmF0ZSBPcGVuYXBpIEdPIFNlcnZlcgoKIyNFeGFtcGxlIEdPIFNlcnZlciAoZ2VuZXJhdGUgYnkgT3BlbmFwaSkKCiMjIExpY2Vuc2UKW01JVCBsaWNlbnNlXShMSUNFTlNFLm1kKS4= readmeEtag: '"baa4ade4d4aed6af409ccf9cbc211f6fb2a27f42"' readmeLastModified: Thu, 03 Mar 2022 08:14:13 GMT repositoryId: 443337799 description: null created: '2021-12-31T12:11:33Z' updated: '2022-03-03T10:02:45Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: RomaBilka logo: https://avatars.githubusercontent.com/u/28790784?v=4 license: NOASSERTION repoEtag: '"7d9eef820c4c4cbbe287bc44c479e66c5d1704dcc7ac3636be0c19d0e844dd27"' repoLastModified: Thu, 03 Mar 2022 10:02:45 GMT foundInMaster: true category: Parsers id: cad28d84ec361611943bc98615e109d5 - source: openapi3 tags repository: https://github.com/himenon/datadog-typescript-openapi v3: true repositoryMetadata: base64Readme: >- IyBAaGltZW5vbi9kYXRhZG9nLXR5cGVzY3JpcHQtb3BlbmFwaQoKYGBgYmFzaAojIHlhcm4KeWFybiBhZGQgLUQgQGhpbWVub24vZGF0YWRvZy10eXBlc2NyaXB0LW9wZW5hcGkKCiMgcG5wbQpwbnBtIGFkZCAtRCBAaGltZW5vbi9kYXRhZG9nLXR5cGVzY3JpcHQtb3BlbmFwaQpgYGAKCiMjIFVzYWdlCgpgYGB0cwppbXBvcnQgeyBDbGllbnQgfSBmcm9tICJAaGltZW5vbi9kYXRhZG9nLXR5cGVzY3JpcHQtb3BlbmFwaS92MS4wLjAtYmV0YS45IjsKYGBgCgojIyBCdWlsZAoKYGBgdHMKcG5wbSBidWlsZApgYGAKCiMjIE9wZW5BUEkgU291cmNlIGZvciBEYXRhRG9nCgotIDxodHRwczovL2dpdGh1Yi5jb20vRGF0YURvZy9kYXRhZG9nLWFwaS1jbGllbnQtdHlwZXNjcmlwdC90cmVlL21hc3Rlci8uZ2VuZXJhdG9yL3NjaGVtYXMvdjI+CgojIyBPcGVuQVBJIFR5cGVTY3JpcHQgQ29kZSBHZW5lcmF0b3IKCi0gW0BoaW1lbm9uL29wZW5hcGktdHlwZXNjcmlwdC1jb2RlLWdlbmVyYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL0hpbWVub24vb3BlbmFwaS10eXBlc2NyaXB0LWNvZGUtZ2VuZXJhdG9yKQoKWW91IGNhbiBhbHNvIGp1c3QgdXNlIHRoZSB0eXBlIGRlZmluaXRpb24KCiMjIFVzZSBBbm90aGVyIFZlcnNpb24KCkVkaXQgW2NvbmZpZy50c10oLi9zY3JpcHRzL2NvbmZpZy50cykKCiMjIExJQ0VOQ0UKCltASGltZW5vbi9kYXRhZG9nLXR5cGVzY3JpcHQtb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL0hpbWVub24vZGF0YWRvZy10eXBlc2NyaXB0LW9wZW5hcGkp44O7TUlUCg== readmeEtag: '"6721b07de5708a13bbab6aac30497c66f7eb5787"' readmeLastModified: Tue, 22 Mar 2022 12:12:29 GMT repositoryId: 472745541 description: Implementation of TypeScript generated from DataDog OpenAPI created: '2022-03-22T11:53:49Z' updated: '2022-03-22T12:13:27Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: Himenon logo: https://avatars.githubusercontent.com/u/6715229?v=4 license: MIT repoEtag: '"fb5608681f7ebd0d0a47af42c3ed076d46cd7262bf6a5e8a1bb0f9bbe47de646"' repoLastModified: Tue, 22 Mar 2022 12:13:27 GMT foundInMaster: true category: SDK id: 62bcee4c699acc36654248a08ffb9e05 - source: openapi3 tags repository: https://github.com/jinxapi/querylizer v3: true repositoryMetadata: base64Readme: >- IyBxdWVyeWxpemVyCgpSdXN0IHNlcmRlIGxpYnJhcnkgZm9yIE9wZW5BUEkgdjMgcGFyYW1ldGVyIHN0eWxlcy4KClVzZSBgc2VyZGVgIHRvIHByb3ZpZGUgdGhlIGRpZmZlcmVudCBzdHlsZXMgc3VwcG9ydGVkIGluIE9wZW5BUEkgcGFyYW1ldGVycy4KCk9wZW5BUEkgcHJvdmlkZXMgbXVsdGlwbGUgc3R5bGVzIGZvciBvcGVyYXRpb24gcGFyYW1ldGVycy4gIGBxdWVyeWxpemVyYApwcm92aWRlcyBhIHNlcmRlIFNlcmlhbGl6ZXIgZm9yIGVhY2ggc3R5bGUuCgpDdXJyZW50bHkgc3VwcG9ydGVkIHN0eWxlcyBhcmU6Ci0gYGRlZXBPYmplY3RgCi0gYGZvcm1gCi0gYHNpbXBsZWAKClRoZXJlIGlzIGFsc28gYSBgZGVlcGZvcm1gIHN0eWxlIHRoYXQgaXMgdXNlZnVsIGZvciBPcGVuQVBJIHJlcXVlc3QgYm9kaWVzIHRoYXQgbWl4IGBmb3JtYCBhbmQgYGRlZXBPYmplY3RgIHN0eWxlcy4K readmeEtag: '"2a7c64b6cff9f61565c49f694f25d66409a442b4"' readmeLastModified: Fri, 25 Mar 2022 12:41:27 GMT repositoryId: 464859030 description: Rust serde serializers for OpenAPI parameter styles created: '2022-03-01T11:06:26Z' updated: '2025-08-25T17:03:55Z' language: Rust archived: false stars: 0 watchers: 1 forks: 0 owner: jinxapi logo: https://avatars.githubusercontent.com/u/100527532?v=4 license: Apache-2.0 repoEtag: '"b708e87ddb768209a856c72e512e57ac989dfe54b044e5ab98e1e82f9bb3dfec"' repoLastModified: Mon, 25 Aug 2025 17:03:55 GMT foundInMaster: true category: - Server - Parsers id: c9bc1750dce1973ff6b28823453ffbe1 - source: openapi3 tags repository: >- https://github.com/jorgealfonsogarcia/uan-spec-sweng-architecture-2-lab-01-layered v3: true id: cd5729218b4c6502196132907ce427f3 repositoryMetadata: base64Readme: >- IyB1YW4tc3BlYy1zd2VuZy1hcmNoaXRlY3R1cmUtMi1sYWItMDEtbGF5ZXJlZAoKIyMgRVMg8J+HqvCfh7gKCiMjIyBDb250ZXh0bwoKUHJveWVjdG8gcmVhbGl6YWRvIHBhcmEgbGEgVW5pdmVyc2lkYWQgQW50b25pbyBOYXJpw7FvLCBFc3BlY2lhbGl6YWNpw7NuIGVuIEluZ2VuaWVyw61hIGRlIFNvZnR3YXJlLCBBcnF1aXRlY3R1cmEgZGUKU29mdHdhcmUgSUksIGNvcnJlc3BvbmRpZW50ZSBhbCBMYWJvcmF0b3JpbyBOby4xIGFjZXJjYSBkZSBBcnF1aXRlY3R1cmEgZGUgQ2FwYXMgeSBlbCBwYXRyw7NuIEJ1c2luZXNzIERlbGVnYXRlLgoKIyMjIEF1dG9yZXMKCiogSm9yZ2UgR2FyY8OtYQoqIERpZWdvIFBvdmVkYQoKIyMjIExpY2VuY2lhCgpbR05VIEdFTkVSQUwgUFVCTElDIExJQ0VOU0VdKExJQ0VOU0UubWQpCgojIyMgUmVxdWlzaXRvcwoKKiBKYXZhIDE3LCBkaXNwb25pYmxlIGVuIGh0dHBzOi8vYWRvcHRpdW0ubmV0L2VzL3RlbXVyaW4vcmVsZWFzZXMgCiogQXBhY2hlIE1hdmVuIDMsIGRpc3BvbmlibGUgZW4gaHR0cHM6Ly9tYXZlbi5hcGFjaGUub3JnL2Rvd25sb2FkLmh0bWwgCiogVGVuZXIgbG9jYWxtZW50ZSBkaXNwb25pYmxlIGVsIHB1ZXJ0byA4MDgwCgojIyMgQ29uc3RydWNjacOzbgoKKiBFamVjdXRhciBlbiBsYSBjb25zb2xhIGRlbCBzaXN0ZW1hIGxhIHNpZ3VpZW50ZSBpbnN0cnVjY2nDs246CgpgYGBzaGVsbAptdm4gY2xlYW4gY29tcGlsZSBwYWNrYWdlIApgYGAKCiogRWwgYXJjaGl2byBKQVIgbG8gZW5jb250cmFyw6EgZW4gbGEgY2FycGV0YSBgdGFyZ2V0YC4KCiMjIyBFamVjdWNpw7NuCgoqIEVqZWN1dGFyIGVuIGxhIGNvbnNvbGEgZGVsIHNpc3RlbWEgbGEgc2lndWllbnRlIGluc3RydWNjacOzbjoKCmBgYHNoZWxsCmphdmEgLWphciB1YW4tc3BlYy1zd2VuZy1hcmNoaXRlY3R1cmUtMi1sYWItMDEtbGF5ZXJlZC0xLjAuMC5qYXIgCmBgYAoKKiBJbmdyZXNlIGEgbGEgVVJMOiBodHRwOi8vbG9jYWxob3N0OjgwODAvbGFiLTAxLWxheWVyZWQvc3dhZ2dlci11aS9pbmRleC5odG1sCiogRW4gbGEgY29uc29sYSB3ZWIgZGUgT3BlbkFQSSBlbmN1ZW50cmEgbGFzIGRpZmVyZW50ZXMgYWNjaW9uZXMgZGlzcG9uaWJsZXMgcGFyYSBpbnRlcmFjdHVhciBjb24gbG9zIHNlcnZpY2lvcyBSRVNUIGRlCiAgbGEgYXBsaWNhY2nDs24uCiogVGFtYmnDqW4gcHVlZGUgaW50ZXJhY3R1YXIgY29uIGxhIGFwbGljYWNpw7NuIG1lZGlhbnRlIHVuIGNsaWVudGUgSFRUUCAoZWouIFBvc3RtYW4pIGluZGljYW5kbyBjb21vIFVSTAogIGJhc2UgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2xhYi0wMS1sYXllcmVkLwoqIFBhcmEgaW50ZXJhY3R1YXIgY29uIGxhIGJhc2UgZGUgZGF0b3MsIGluZ3Jlc2UgYSBsYSBVUkwgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2xhYi0wMS1sYXllcmVkL2gyLWNvbnNvbGUgLiBMb3MgZGF0b3MgZGUKICBhY2Nlc28gc29uIGxvcyBzaWd1aWVudGVzOgoKfCBDYW1wbyAgICB8IFZhbG9yIHwKfC0tLS0tLS0tLS18LS0tLS0tLXwKfCAqKkNvbnRyb2xhZG9yKiogIHwgb3JnLmgyLkRyaXZlciB8CnwgKipVUkwgSkRCQyoqIHwgamRiYzpoMjptZW06bGFiLTAxLWxheWVyZWQgIHwKfCAqKk5vbWJyZSBkZSB1c3VhcmlvKiogfCB1YW4gIHwKfCAqKkNvbnRyYXNlw7FhKiogfCBhcmNoaXRlY3R1cmUgIHwKCiMjIyBEaXNlw7FvCgojIyMjIERpYWdyYW1hIGRlIENsYXNlcwoKRWwgZGlhZ3JhbWEgcmVwcmVzZW50YSBlbCB1c28gZGVsIHBhdHLDs24gKkJ1c2luZXNzIERlbGVnYXRlKiBjb24gYmFzZSBlbiBlbCBwYXRyw7NuICpQcm94eSogcGFyYSBsYSBzZXBhcmFjacOzbiBkZSBsYSBjYXBhCmRlIG5lZ29jaW8uIExvcyBjb250cm9sYWRvcmVzIHkgZWwgYEluaXREYXRhTG9hZGVyYCDDum5pY2FtZW50ZSBpbnRlcmFjdMO6YW4gY29uIGxhIGNsYXNlIGBCdXNpbmVzc0RlbGVnYXRlYCBwYXJhIG9idGVuZXIKYWNjZXNvIGEgbGEgbMOzZ2ljYSBkZSBuZWdvY2lvLiBgQnVzaW5lc3NEZWxlZ2F0ZWAgYSBzdSB2ZXogdXRpbGl6YSBsYSBjbGFzZSBgQnVzaW5lc3NMb29rdXBgIHBhcmEgdGVuZXIgYWNjZXNvIGEgbGEgY2xhc2UKZGUgc2VydmljaW8gY29ycmVzcG9uZGllbnRlIGEgbGEgc29saWNpdHVkLgoKIVtEaWFncmFtYSBkZSBDbGFzZXNdKHVtbC9jbGFzc2VzL2NsYXNzZXMucG5nKQoKIyMjIyBEaWFncmFtYSBkZSBTZWN1ZW5jaWEKCkVsIGRpYWdyYW1hIHJlcHJlc2VudGEgbGEgc2VjdWVuY2lhIGNvcnJlc3BvbmRpZW50ZSBhIGxhIGLDunNxdWVkYSBkZSB0b2RvcyBsb3MgZW1wbGVhZG9zIHJlZ2lzdHJhZG9zIGVuIGxhIGJhc2UgZGUgZGF0b3MKZGUgbGEgYXBsaWNhY2nDs24sIHV0aWxpemFuZG8gbGEgY2xhc2UgYEJ1c2luZXNzRGVsZWdhdGVgIGNvbW8gw7puaWNvIHB1bnRvIGRlIGludGVyYWNjacOzbiBkZXNkZSBsb3MgY29udHJvbGFkb3JlcywgeSBjb21vCmVzdGUgaGFjZSB1c28gZGUgbGEgY2xhc2UgYEJ1c2luZXNzTG9va1VwYCBwYXJhIGVuY29udHJhciBlbCBzZXJ2aWNpbyBjb3JyZXNwb25kaWVudGUuCgohW0RpYWdyYW1hIGRlIFNlY3VlbmNpYV0odW1sL3NlcXVlbmNlL3NlcXVlbmNlLnBuZykKCiMjIyBSZWZlcmVuY2lhcwoKKiBNaXRyYSwgUy4gKDIwMTgpLiBMYXllcmVkIEFyY2hpdGVjdHVyZSBVcCBhbmQgUnVubmluZyBqdXN0IGluIDUgbWludXRlczo6IFNwcmluZyBCb290IFBhcnQgMSB8IE1ha2UgJiBLbm93IEphdmEuCiAgQmxvZ3Nwb3QuY29tLiBodHRwOi8vamF2YW9uZmx5LmJsb2dzcG90LmNvbS8yMDE4LzA5L2xheWVyZWQtYXJjaGl0ZWN0dXJlLXVwLWFuZC1ydW5uaW5nLmh0bWwKKiBNaXRyYSwgUy4gKDIwMTgsIE9jdG9iZXIgMTcpLiBCdWlsZGluZyBMYXllcmVkIEFyY2hpdGVjdHVyZSBpbiBKdXN0IDMgTWludXRlczogRmluYWwgUGFydC4gRHpvbmUuY29tOwogIERab25lLiBodHRwczovL2R6b25lLmNvbS9hcnRpY2xlcy9idWlsZGluZy1sYXllcmVkLWFyY2hpdGVjdHVyZS1qdXN0LWluLTMtbWludXRlcy1maS0xCiogQ29yZSBKMkVFIFBhdHRlcm5zIC0gQnVzaW5lc3MgRGVsZWdhdGUuICgyMDE4KS4KICBPcmFjbGUuY29tLiBodHRwczovL3d3dy5vcmFjbGUuY29tL2phdmEvdGVjaG5vbG9naWVzL2J1c2luZXNzLWRlbGVnYXRlLmh0bWwKKiBEZXNpZ24gUGF0dGVybnMgLSBCdXNpbmVzcyBEZWxlZ2F0ZSBQYXR0ZXJuLiAoMjAyMikuCiAgVHV0b3JpYWxzcG9pbnQuY29tLiBodHRwczovL3d3dy50dXRvcmlhbHNwb2ludC5jb20vZGVzaWduX3BhdHRlcm4vYnVzaW5lc3NfZGVsZWdhdGVfcGF0dGVybi5odG0KKiBiYWVsZHVuZy4gKDIwMTksIEFwcmlsIDE4KS4gU3ByaW5nIEJvb3QgV2l0aCBIMiBEYXRhYmFzZSB8IEJhZWxkdW5nLgogIEJhZWxkdW5nLiBodHRwczovL3d3dy5iYWVsZHVuZy5jb20vc3ByaW5nLWJvb3QtaDItZGF0YWJhc2UKKiBQcm94eS4gKDIwMTQpLiBSZWZhY3RvcmluZy5ndXJ1LiBodHRwczovL3JlZmFjdG9yaW5nLmd1cnUvZXMvZGVzaWduLXBhdHRlcm5zL3Byb3h5Cgo= readmeEtag: '"9227b0098a515d9d30428ab98b3f9d81029428a5"' readmeLastModified: Fri, 26 Aug 2022 03:15:34 GMT repositoryId: 528579446 description: >- Project for the Antonio Nariño University, Specialization in Software Engineering, Software Architecture II, corresponding to Laboratory No.1 about Layer Architecture and the Business Delegate pattern. created: '2022-08-24T20:12:48Z' updated: '2022-08-27T01:49:56Z' language: Java archived: false stars: 0 watchers: 2 forks: 0 owner: jorgealfonsogarcia logo: https://avatars.githubusercontent.com/u/3460617?v=4 license: GPL-3.0 repoEtag: '"722d6c655340883201c6de6f110bb781958a435e9380037781c52cf8479ce85a"' repoLastModified: Sat, 27 Aug 2022 01:49:56 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/calebtracey/rugby-data-api v3: true id: fbb953f9bdb8e6cc5c8e79501e8dc063 repositoryMetadata: base64Readme: >- IyBydWdieS1kYXRhLWFwaQoKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL0NhbGViVHJhY2V5L3J1Z2J5LWRhdGEtYXBpL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj03MTQ5SUlUWjgxKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL0NhbGViVHJhY2V5L3J1Z2J5LWRhdGEtYXBpKQoKIyMjIFtTd2FnZ2VyIERvY3NdKGh0dHBzOi8vY2FsZWJ0cmFjZXkuZ2l0aHViLmlvL3J1Z2J5LWRhdGEtYXBpL3N3YWdnZXItdWkvKQoKW3J1Z2J5LW1vZGVsc10oaHR0cHM6Ly9naXRodWIuY29tL0NhbGViVHJhY2V5L3J1Z2J5LW1vZGVscykgdXNlZCBmb3IgY29tbW9uIGRhdGEgdHlwZXMKCltjb25maWcteWFtbF0oaHR0cHM6Ly9naXRodWIuY29tL0NhbGViVHJhY2V5L2NvbmZpZy15YW1sKSB1c2VkIGZvciBlbnZpcm9ubWVudCBjb25maWdzIGFuZCBzZXJ2aWNlL2RiIGluaXRpYWxpemF0aW9uCgoqKkJhc2ljIHNldHVwKioKMS4gQ3JlYXRlIGEgbG9jYWwgUG9zdGdyZXMgZGF0YWJhc2UgYW5kIHVwZGF0ZSB0aGUgY29uZmlnLnlhbWwgZmlsZSB3aXRoIHRoZSBkZXRhaWxzCjIuIFVwZGF0ZSBnby5tb2QgYW5kIGZpbGUgaW1wb3J0cyB3aXRoIHlvdXIgcmVwbyBuYW1lCjMuIFJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmRzIHRvIHVwZGF0ZSBkZXBlbmRlbmNpZXM6CgogICBgZ28gZ2V0IC11IC4vLi4uYAoKICAgIGBnbyBtb2QgdGlkeWAKNC4gTWFrZSBhIHJ1biBjb25maWd1cmF0aW9uIGFzIHNlZW4gYmVsb3cgd2l0aCB5b3VyIHJlcG8gbmFtZToKCiFbUnVuIENvbmZpZ10oLi9pbWFnZXMvcnVuLWNvbmZpZy5wbmcpCgojIyMjIE5vdyB5b3UgY2FuIHN0YXJ0IHRoZSBBUEkgYW5kIGFjY2VzcyBodHRwOi8vbG9jYWxob3N0OjYwODAvc3dhZ2dlci11aS8gZm9yIHN3YWdnZXIgZG9jdW1lbnRhdGlvbiBhbmQgdGVzdGluZwo= readmeEtag: '"1460d98dfafdca53aa89632802b7421350fa20e9"' readmeLastModified: Sun, 04 Dec 2022 18:45:32 GMT repositoryId: 569336302 description: Go API for accessing rugby stats created: '2022-11-22T15:42:10Z' updated: '2022-11-24T15:10:43Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: CalebTracey logo: https://avatars.githubusercontent.com/u/59851860?v=4 license: MIT repoEtag: '"771629e5692b7e64e1482f25a429ec6b45d6bb92510e91160fcf659ef3d148a0"' repoLastModified: Thu, 24 Nov 2022 15:10:43 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/honzabit/pactum-api-coverage v3: true id: dd33f534d3f7250729ac4ca3fc575c62 repositoryMetadata: base64Readme: >- IyBwYWN0dW0tYXBpLWNvdmVyYWdlCgohW1BsYXRmb3JtXShodHRwczovL2ltZy5zaGllbGRzLmlvL25vZGUvdi9wYWN0dW0pCgpBUEkgY292ZXJhZ2UgcmVwb3J0ZXIgZm9yIFtQYWN0dW1dKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL3BhY3R1bSkgdGVzdHMuIEFjY2VwdHMgU3dhZ2dlci9PQVMgc3BlYyBkYXRhIGluIEpTT04gZm9ybWF0LgoKIyMgSW5zdGFsbGF0aW9uCgpgYGBzaGVsbApucG0gaW5zdGFsbCAtLXNhdmUtZGV2IHBhY3R1bSBwYWN0dW0tYXBpLWNvdmVyYWdlCmBgYAoKIyMgVXNhZ2UKCmBgYGphdmFzY3JpcHQKY29uc3QgcGFjdHVtID0gcmVxdWlyZSgncGFjdHVtJyk7CmNvbnN0IHBzYyA9IHJlcXVpcmUoJ3BhY3R1bS1hcGktY292ZXJhZ2UnKTsKY29uc3QgcmVwb3J0ZXIgPSBwYWN0dW0ucmVwb3J0ZXI7CgovLyBnbG9iYWwgYmVmb3JlIGJsb2NrCmJlZm9yZSgoKSA9PiB7CiAgcHNjLnNwZWNEYXRhID0gcmVxdWlyZSgiLi9zcGVjcy92MS9vYXMuanNvbiIpOwogIHJlcG9ydGVyLmFkZChwc2MpOwp9KTsKCi8vIGdsb2JhbCBhZnRlciBibG9jawphZnRlcigoKSA9PiB7CiAgcmV0dXJuIHJlcG9ydGVyLmVuZCgpOwp9KTsKYGBgCgojIyBSZXBvcnRlciBPcHRpb25zCgpgYGBqYXZhc2NyaXB0CmNvbnN0IHBzYyA9IHJlcXVpcmUoJ3BhY3R1bS1hcGktY292ZXJhZ2UnKTsKCi8vIG5hbWUgb2YgdGhlIHJlcG9ydCBmaWxlIC0gZGVmYXVsdHMgdG8gImFwaS1jb3YtcmVwb3J0Lmpzb24iCnBzYy5maWxlID0gJ3JlcG9ydC1uYW1lLmpzb24nOwoKLy8gZm9sZGVyIHBhdGggZm9yIHRoZSByZXBvcnQgZmlsZSAtIGRlZmF1bHRzIHRvICIuL3JlcG9ydHMiCnBzYy5wYXRoID0gJy4vcmVwb3J0cy1wYXRoJzsKCi8vIExvYWQgYSBsb2NhbCBKU09OIHNwZWMgZmlsZSAob3IgYSByZW1vdGUgb25lKQpwc2Muc3BlY0RhdGEgPSByZXF1aXJlKCIuL3NwZWNzL3YxL29hcy5qc29uIik7CgovLyBJZ25vcmUgc3BlY2lmaWMgcmVzcG9uc2UgY29kZXMKcHNjLmlnbm9yZVJlc3BvbnNlQ29kZXMgPSBbIDQyOSwgNTAwXQpgYGAKCiMjIyBSZXBvcnQgSnNvbiBPdXRwdXQgKGV4YW1wbGUpCmBgYGphdmFzY3JpcHQKewogICJiYXNlUGF0aCI6ICIvdjEiLAogICJjb3ZlcmFnZSI6IDAuNSwKICAiY292ZXJlZEFwaUNvdW50IjogNCwKICAibWlzc2VkQXBpQ291bnQiOiA0LAogICJ0b3RhbEFwaUNvdW50IjogOCwKICAiY292ZXJlZEFwaUxpc3QiOiBbCiAgICAiMjAwfGdldHwvdjEvaGVhbHRoIiwKICAgICIyMDB8Z2V0fC92MS9nZXRhbGxuaW5qYXMiLAogICAgIjIwMHxnZXR8L3YxL2dldG5pbmphcy97cmFua30iLAogICAgIjIwMHxnZXR8L3YxL2dldG5pbmphL3tyYW5rfS97bmFtZX0iCiAgXSwKICAibWlzc2VkQXBpTGlzdCI6IFsKICAgICIyMDB8cG9zdHwvdjEvaGVhbHRoIiwKICAgICIyMDB8Z2V0fC92MS9nZXRuaW5qYXMve2NsYW59L3tyYW5rfSIsCiAgICAiNDAwfGdldHwvdjEvZ2V0bmluamFzL3tjbGFufS97cmFua30iLAogICAgIjIwMHxnZXR8L3YxL2dldG5pbmphL3tuYW1lfSIKICBdCn0KYGBgCgojIyBOb3RlcwoKUmVhZCBtb3JlIGFib3V0IFBhY3R1bSBbaGVyZV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvcGFjdHVtKS4KCgpDb3BpZWQgZnJvbSBbcGFjdHVtLXN3YWdnZXItY292ZXJhZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9wYWN0dW1qcy9wYWN0dW0tc3dhZ2dlci1jb3ZlcmFnZSkgYW5kIGFsdGVyZWQuCg== readmeEtag: '"17f4a8c494f7f7d0b33ba5d804ddd7bd55c85abe"' readmeLastModified: Thu, 26 Jan 2023 00:14:40 GMT repositoryId: 590919446 description: >- API Coverage Reporter for Pactum. Provides operation & response level coverage and supports JSON schemas for swagger/OAS specs. created: '2023-01-19T14:22:05Z' updated: '2023-01-25T19:42:47Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: honzabit logo: https://avatars.githubusercontent.com/u/65507986?v=4 license: MIT repoEtag: '"62f79cb58df0dc7d502e244f2e9639102f49bd1ea38c0680ebd3dfaad2d7914a"' repoLastModified: Wed, 25 Jan 2023 19:42:47 GMT category: Data Validators foundInMaster: true - source: openapi3 tags repository: https://github.com/guillermoleonpy/springboot-demo v3: true id: 91dbc7b2103912a7577b69224b9380aa repositoryMetadata: base64Readme: >- c3ByaW5nIGJvb3QgZGVtbyBhcHAKCmFjdHVhdG9yIHVybDogaHR0cDovL2xvY2FsaG9zdDo4MTgxL2FjdHVhdG9yOyB0byBjaGVjayB0aGUgYXBwbGljYXRpb24gc3RhdHVzIGFuZCBoZWFsdGgKIyMjClJhYmJpdCBNUSB0ZXN0CmNvbnNvbGU6IGh0dHA6Ly9sb2NhbGhvc3Q6MTU2NzIvIy9xdWV1ZXMKCgpTd2FnZ2VyCmh0dHA6Ly9sb2NhbGhvc3Q6ODE4MS9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKCg== readmeEtag: '"cbd0e9e4adc1e25834624293f9cf146df17dd829"' readmeLastModified: Wed, 05 Apr 2023 00:03:41 GMT repositoryId: 622243393 description: >- springboot-demo rest api interaction with rabbit mq, apache kafka and jpa. Exemplifies asynchronous processing. End points documented using OpenAPI 3 specification created: '2023-04-01T14:43:51Z' updated: '2023-04-05T08:57:23Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: GuillermoLeonPy logo: https://avatars.githubusercontent.com/u/75272198?v=4 repoEtag: '"b6181393d6cf68d9811a760918090156b30e7a9cb3c9ef685f52740feac0fbf4"' repoLastModified: Wed, 05 Apr 2023 08:57:23 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/arldka/go-artifacthub v3: true id: 873de07499c4cced37621c9e193642f0 repositoryMetadata: base64Readme: >- IyBnby1hcnRpZmFjdGh1YgoKQXV0byBnZW5lcmF0ZWQgdXNpbmcgaHR0cHM6Ly9naXRodWIuY29tL2RlZXBtYXAvb2FwaS1jb2RlZ2VuIGZyb20gaHR0cHM6Ly9hcnRpZmFjdGh1Yi5pby9kb2NzL2FwaS8KCiMjIFVzYWdlCgpgYGAKY2xpZW50LCBlcnIgOj0gYXJ0aWZhY3RodWIuTmV3Q2xpZW50V2l0aFJlc3BvbnNlcygiaHR0cHM6Ly9hcnRpZmFjdGh1Yi5pby9hcGkvdjEiLCBhcnRpZmFjdGh1Yi5XaXRoUmVxdWVzdEVkaXRvckZuKGZ1bmMoY3R4IGNvbnRleHQuQ29udGV4dCwgcmVxICpodHRwLlJlcXVlc3QpIGVycm9yIHsKICByZXEuSGVhZGVyLlNldCgiWC1BUEktS0VZLUlEIiwgIktFWV9JRCIpCiAgcmVxLkhlYWRlci5TZXQoIlgtQVBJLUtFWS1TRUNSRVQiLCAiS0VZX1NFQ1JFVCIpCiAgcmV0dXJuIG5pbAp9KSkKaWYgZXJyICE9IG5pbCB7CiAgcmV0dXJuIG5pbCwgZXJyCn0KCnJlc3AsIGVyciA6PSBjbGllbnQuR2V0SGVsbVBhY2thZ2VEZXRhaWxzV2l0aFJlc3BvbnNlKGN0eCwgInJlcG8iLCAicGFja2FnZSIpCmlmIGVyciAhPSBuaWwgewogIHJldHVybiBuaWwsIGVycgp9CmBgYAo= readmeEtag: '"300551ecf0f8255bb007cf0bc91d1746f91829a9"' readmeLastModified: Sun, 06 Aug 2023 15:33:01 GMT repositoryId: 674820001 description: artifacthub go client created: '2023-08-04T21:44:40Z' updated: '2023-08-05T21:17:50Z' language: Makefile archived: false stars: 0 watchers: 0 forks: 0 owner: arldka logo: https://avatars.githubusercontent.com/u/43825600?v=4 license: MIT repoEtag: '"f6fd203ca54690c4d5004ab24de4c807c7800d7a0e2239f3d04c8599782255b1"' repoLastModified: Sat, 05 Aug 2023 21:17:50 GMT category: Data Validators foundInMaster: true - source: openapi3 tags repository: https://github.com/jalgos/gaard-api-docs v3: true repositoryMetadata: base64Readme: >- IyBHYWFyZCBBUEkgZG9jdW1lbnRhdGlvbgoKVGhlIEFQSSBkb2N1bWVudGF0aW9uIGlzIGJhc2VkIG9uIFtPcGVuQVBJIDMuMC4wXShodHRwczovL3N3YWdnZXIuaW8vKS4KCi0gQVBJIHNlcnZlcjogW2h0dHBzOi8vdmlzaW9uLmdhYXJkLmFpL3YxXShodHRwczovL3Zpc2lvbi5nYWFyZC5haS92MSkKLSBBUEkgZG9jdW1lbnRhdGlvbiB3ZWJzaXRlIFtodHRwczovL2RvY3MuZ2FhcmQuYWldKGh0dHBzOi8vZG9jcy5nYWFyZC5haSkKCiMjIFdvcmsgd2l0aCBBUEkKClRvIHdvcmsgd2l0aCBBUEkgeW91IG5lZWQgYEFQSV9LRVlgLiBQbGVhc2UgY29udGFjdCB1cyBjb250YWN0QGphbGdvcy5jb20gdG8gaGF2ZSBhY2Nlc3MuCgpUaGUgY2xhc3NpZmljYXRpb24gdXNlcyBgYW5hbHlzZV9pZGAgcGFyYW1ldGVyIHRoYXQgc2hvdWxkIGJlIGEgdW5pcXVlIG51bWJlciBmb3IgZWFjaCByZXF1ZXN0LiBZb3UgY2FuIHVzZSBhIGNvbnRpbnVvdXNseSBncm93aW5nIGNvdW50ZXIuCgojIyBVc2Ugc2NyaXB0cwoKQ2xhc3NpZmljYXRpb24gQVBJIGlzIGFzeW5jaHJvbm91cy4gVGhlIGdlbmVyYWwgZmxvdyBpcyB0byBzZW5kIHZpZGVvIHRvIGNsYXNzaWZpY2F0aW9uIGFuZCByZWNlaXZlIHRoZSByZXN1bHQgYnkgY2FsbGluZyBgL3Jlc3VsdGAgZW5kcG9pbnQgb3IgdXNpbmcgd2ViaG9vawoKQ2xvbmUgcmVwb3NpdG9yeQoKYGBgYmFzaApnaXQgY2xvbmUgZ2l0QGdpdGh1Yi5jb206amFsZ29zL2dhYXJkLWFwaS1kb2NzLmdpdApjZCBnYWFyZC1hcGktZG9jcwpgYGAKCkFkZCBgQVBJX0tFWWAgdG8gZW52aXJvbm1lbnQgdmFyaWFibGVzCgpgYGBiYXNoCmV4cG9ydCBBUElfS0VZPSJZT1VSIEFQSSBLRVkiCmBgYAoKIyMjIFBvc3QgdmlkZW8KClVzYWdlCgpgYGBiYXNoCmJpbi9wb3N0LXZpZGVvIFt2aWRlb19wYXRoXSBbYW5hbHlzZV9pZF0gW21ldGFkYXRhX3BhdGhdCmBgYAoKRXhhbXBsZXMKCmBgYGJhc2gKYmluL3Bvc3QtdmlkZW8KYmluL3Bvc3QtdmlkZW8gYmluL3ZpZGVvLm1vdgpiaW4vcG9zdC12aWRlbyBiaW4vdmlkZW8ubW92IDEyNTQ0NDgKYmluL3Bvc3QtdmlkZW8gYmluL3ZpZGVvLm1vdiAxMjU0NDQ4IGJpbi9tZXRhZGF0YS5qc29uCmBgYAoKIyMjIEdldCByZXN1bHRzCgpVc2FnZQoKYGBgYmFzaApiaW4vZ2V0LXJlc3VsdCBhbmFseXNlX2lkCmBgYAoKRXhhbXBsZQoKYGBgYmFzaApiaW4vZ2V0LXJlc3VsdCAxMjU0NDQ4CmBgYAoKIyMjIEdldCBvcmlnaW5hbCB2aWRlbwoKVXNhZ2UKCmBgYGJhc2gKYmluL2dldC1vcmlnaW5hbC12aWRlbyBmaWxlbmFtZQpgYGAKCkV4YW1wbGUKCmBgYGJhc2gKYmluL2dldC1vcmlnaW5hbC12aWRlbyB2aWRlby5tb3YKYGBgCgojIyMgR2V0IEhpZ2hsaWdodGVkIHZpZGVvCgpVc2FnZQoKYGBgYmFzaApiaW4vZ2V0LWhpZ2hsaWdodGVkLXZpZGVvIGZpbGVuYW1lCmBgYAoKRXhhbXBsZQoKYGBgYmFzaApiaW4vZ2V0LWhpZ2hsaWdodGVkLXZpZGVvIHZpZGVvLm1vdgpgYGAKCiMjIExpY2Vuc2VzCgpBbiBleGFtcGxlIGBiaW4vdmlkZW8ubW92YCBpcyBhIHBhcnQgb2YgW1RoZSBWSVJBVCBWaWRlbyBEYXRhc2V0XShodHRwczovL3ZpcmF0ZGF0YS5vcmcvKQo= readmeEtag: '"14630be1d4fcaedba4c874ddbd30310bf1f58faf"' readmeLastModified: Fri, 06 Sep 2024 08:36:07 GMT repositoryId: 249956431 description: Ukpik API documentation created: '2020-03-25T11:08:38Z' updated: '2024-09-29T09:04:25Z' language: Shell archived: false stars: 0 watchers: 5 forks: 0 owner: jalgos logo: https://avatars.githubusercontent.com/u/8981259?v=4 license: MIT repoEtag: '"ec46c51abf0523b359211a5f54dbfaeadc2d778a43b61804258a63b14444dfdf"' repoLastModified: Sun, 29 Sep 2024 09:04:25 GMT foundInMaster: true category: - Documentation - Server Implementations id: f5a7d4cdc09b86e19a1e7bee7b694ae2 oldLocations: - https://github.com/jalgos/ukpik-api-docs - source: openapi3 tags repository: https://github.com/jayllyz/openapi-fastify-petstore v3: true id: 636893cb11fb040ae0d25aa7a39dd8a6 repositoryMetadata: base64Readme: >- IyMgVGVjaCBTdGFjawoKLSBOb2RlLmpzCi0gVHlwZVNjcmlwdAotIEZhc3RpZnkKLSBvcGVuQVBJIDMKLSBQcmlzbWEKLSBQb3N0Z3JlU1FMCi0gRG9ja2VyCgojIyBSdW4gbG9jYWxseQoKVG8gcnVuIHRoZXNlIGNvbW1hbmRzIHlvdSBuZWVkIHRvIGluc3RhbGwgOgpbdGFza10oaHR0cHM6Ly90YXNrZmlsZS5kZXYvIy9pbnN0YWxsYXRpb24pCgpgYGBiYXNoCiMgU3RhcnQgZG9ja2VyCnRhc2sgZG9ja2VyCgojIEluc3RhbGwgZGVwZW5kZW5jaWVzCnRhc2sgaW5zdGFsbAoKIyBJbml0IGRhdGFiYXNlCnRhc2sgcHJpc21hCgojIFN0YXJ0IHNlcnZlcgp0YXNrIGRldgpgYGAKCiMjIEFQSSBEb2N1bWVudGF0aW9uCgotIDxodHRwOi8vbG9jYWxob3N0OjMwMDAvZG9jPgo= readmeEtag: '"c72e7442bf6f8781d74dc8bbb22ba6411194745d"' readmeLastModified: Mon, 11 Dec 2023 08:32:10 GMT repositoryId: 648663642 description: openAPI & Fastify learning project created: '2023-06-02T13:53:42Z' updated: '2024-01-05T13:58:34Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: Jayllyz logo: https://avatars.githubusercontent.com/u/16305216?v=4 repoEtag: '"a971c1bd599fc6f189571f0a9af4a1ce044d10f6845e16c9e1833672c71265e3"' repoLastModified: Fri, 05 Jan 2024 13:58:34 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/karsei/spring-swagger-snippet v3: true id: 3aa12468a60813b680865beef054f91b repositoryMetadata: base64Readme: >- IyBTd2FnZ2VyIOyCrOyaqSDqtIDroKggc25pcHBldCDrqqjsnYwKCmh0dHBzOi8vc3ByaW5nZG9jLm9yZy8KCiMjIOyEpOyglQoK7J2Y7KG07ISxIOy2lOqwgAoKYGBgZ3Jvb3Z5Ci8vIGh0dHBzOi8vbXZucmVwb3NpdG9yeS5jb20vYXJ0aWZhY3Qvb3JnLnNwcmluZ2RvYy9zcHJpbmdkb2Mtb3BlbmFwaS11aQppbXBsZW1lbnRhdGlvbiBncm91cDogJ29yZy5zcHJpbmdkb2MnLCBuYW1lOiAnc3ByaW5nZG9jLW9wZW5hcGktdWknLCB2ZXJzaW9uOiAnMS42LjEyJwpgYGAKCiog6riw67O4IOqyveuhnDogaHR0cDovL2xvY2FsaG9zdDo4MDgwL3YzL2FwaS1kb2NzLyAKICArIEpTT04g7Jy866GcIO2RnO2YhOuQmOyWtCDsnojsnYwKICArIGBhcHBsaWNhdGlvbi55bWxgIOyXkOyEnCBgc3ByaW5nZG9jLmFwaS1kb2NzLnBhdGhgIOyGjeyEseydhCDthrXtlbQg67OA6rK97ZWgIOyImCDsnojri6QuCiogVUkg6rK966GcOiBodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS5odG1sCiAgKyBgYXBwbGljYXRpb24ueW1sYCDsl5DshJwgYHNwcmluZ2RvYy5zd2FnZ2VyLXVpLnBhdGhgIOyGjeyEseydhCDthrXtlbQg67OA6rK97ZWgIOyImCDsnojri6QuCgoKCgo= readmeEtag: '"fa9bca56aca32530625aaa5edaf164b2e87f8a55"' readmeLastModified: Tue, 25 Oct 2022 11:27:19 GMT repositoryId: 539034967 description: Spring Swagger(springdoc) 사용 관련 snippet created: '2022-09-20T14:34:10Z' updated: '2024-01-06T03:46:39Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Karsei logo: https://avatars.githubusercontent.com/u/10175382?v=4 repoEtag: '"1c24486d4e47b6a9eda29aeee8f4d9fc908ee65748fec6d9008e8c1b600cd9a2"' repoLastModified: Sat, 06 Jan 2024 03:46:39 GMT category: - Low-level Tooling - Server Implementations foundInMaster: true oldLocations: - https://github.com/karsei/spring-swagger-example - source: openapi3 tags repository: https://github.com/ryusuke410/spicy-input v3: true id: ae51cb1bed0966edbcf5af77b2dd939e repositoryMetadata: base64Readme: >- IyMgU3BpY3kgSW5wdXQgKFN3YWdnZXIgVUkgUGx1Z2luKQoKVGhlcmUgaXMgbm8gc3RyYWlnaHRmb3J3YXJkIG1ldGhvZCBpbiB0aGUgc3RhbmRhcmQgZmVhdHVyZXMgb2YgU3dhZ2dlciBVSSB0byBwcm9ncmFtbWF0aWNhbGx5IG1hbmlwdWxhdGUgdXNlciBpbnB1dHMgc3VjaCBhcyBxdWVyeSBwYXJhbWV0ZXJzIGFuZCByZXF1ZXN0IGJvZGllcy4gU3BpY3kgSW5wdXQgcHJvdmlkZXMgYSBjb25jaXNlIGludGVyZmFjZSBmb3IgcmV0cmlldmluZywgc2V0dGluZywgYW5kIHN1YnNjcmliaW5nIHRvIHRoZXNlIHVzZXIgaW5wdXRzLgoKRnVydGhlcm1vcmUsIHVzZXIgaW5wdXRzIGFyZSBjb21wbGV0ZWx5IGxvc3QgdXBvbiByZWxvYWRpbmcgU3dhZ2dlciBVSS4gRHVyaW5nIEFQSSBkZXZlbG9wbWVudCwgdGhlcmUgaXMgb2Z0ZW4gYSBkZXNpcmUgdG8gcmVtZW1iZXIgdGhlIGlucHV0dGVkIGNvbnRlbnQuIFNwaWN5IElucHV0IG1ha2VzIGl0IGVhc3kgdG8gcmV0YWluIHRoZXNlIHVzZXIgaW5wdXRzLiBTd2FnZ2VyIFVJIGludGVsbGlnZW50bHkgc2F2ZXMgdXNlciBpbnB1dHMgYmFzZWQgb24gZWFjaCBBUEkgaW50ZXJmYWNlLCBzbyBpdCB3b3JrcyB3ZWxsIGluIG1vc3QgY2FzZXMgZXZlbiBpbiB1c2UgY2FzZXMgd2hlcmUgbXVsdGlwbGUgc3BlY3MgYXJlIGR5bmFtaWNhbGx5IHN3YXBwZWQuIEZvciB0aGUgc2FtZSByZWFzb24sIGlmIHRoZSBkZXRhaWxzIG9mIHRoZSBpbnRlcmZhY2UsIHN1Y2ggYXMgdGhlIHR5cGUgb2YgaW5wdXQgdmFsdWVzLCBhcmUgY2hhbmdlZCwgdGhlIHVzZXIgaW5wdXRzIGFyZSBub3QgcmV0YWluZWQuCgojIyBCYXNpYyBVc2FnZQoKIyMjIFdpdGggbnBtCgpgYGBzaGVsbApucG0gaSBzcGljeS1pbnB1dApgYGAKClBhc3MgdGhpcyBwbHVnaW4gdG8gb3B0aW9ucy4KCmBgYGpzCmNvbnN0IHNwaWN5SW5wdXQgPSByZXF1aXJlKCdzcGljeS1pbnB1dCcpCgpTd2FnZ2VyVUkoewogIHBsdWdpbnM6IFsKICAgIHNwaWN5SW5wdXQuZ2V0UGx1Z2luKCkKICBdCn0pCmBgYAoKVHlwZVNjcmlwdDoKCmBgYHR5cGVzY3JpcHQKaW1wb3J0ICogYXMgc3BpY3lJbnB1dCBmcm9tICdzcGljeS1pbnB1dCc7CgpTd2FnZ2VyVUkoewogIHBsdWdpbnM6IFsKICAgIHNwaWN5SW5wdXQuZ2V0UGx1Z2luKCkKICBdCn0pCmBgYAoKIyMjIFdpdGggVU5QS0cKCmBgYGh0bWwKPGxpbmsgcmVsPSJzdHlsZXNoZWV0IiB0eXBlPSJ0ZXh0L2NzcyIgaHJlZj0iaHR0cHM6Ly91bnBrZy5jb20vc3dhZ2dlci11aS1kaXN0L3N3YWdnZXItdWkuY3NzIiAvPgoKPHNjcmlwdCBzcmM9Imh0dHBzOi8vdW5wa2cuY29tL3N3YWdnZXItdWktZGlzdC9zd2FnZ2VyLXVpLWJ1bmRsZS5qcyI+PC9zY3JpcHQ+CjxzY3JpcHQgc3JjPSJodHRwczovL3VucGtnLmNvbS9zcGljeS1pbnB1dCI+PC9zY3JpcHQ+Cgo8c2NyaXB0Pgp3aW5kb3cub25sb2FkID0gKCkgPT4gewogIFN3YWdnZXJVSUJ1bmRsZSh7CiAgICBwbHVnaW5zOiBbCiAgICAgIHNwaWN5SW5wdXQuZ2V0UGx1Z2luKCkKICAgIF0KICB9KQp9Cjwvc2NyaXB0PgpgYGAKCiMjIyBXaXRoIGpzRGVsaXZyCgpgYGBodG1sCjxsaW5rIHJlbD0ic3R5bGVzaGVldCIgdHlwZT0idGV4dC9jc3MiIGhyZWY9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vc3dhZ2dlci11aS1kaXN0L3N3YWdnZXItdWkuY3NzIiAvPgoKPHNjcmlwdCBzcmM9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vc3dhZ2dlci11aS1kaXN0L3N3YWdnZXItdWktYnVuZGxlLmpzIj48L3NjcmlwdD4KPHNjcmlwdCBzcmM9Imh0dHBzOi8vY2RuLmpzZGVsaXZyLm5ldC9ucG0vc3BpY3ktaW5wdXQiPjwvc2NyaXB0PgoKPHNjcmlwdD4Kd2luZG93Lm9ubG9hZCA9ICgpID0+IHsKICBTd2FnZ2VyVUlCdW5kbGUoewogICAgcGx1Z2luczogWwogICAgICBzcGljeUlucHV0LmdldFBsdWdpbigpCiAgICBdCiAgfSkKfQo8L3NjcmlwdD4KYGBgCgojIyBPcHRpb25zCgojIyMgcGVyc2lzdFVzZXJJbnB1dHMKCllvdSBjYW4gY2hvb3NlIHRvIGFjdGl2YXRlIHRoZSBtZW1vcnkgZnVuY3Rpb24gZm9yIHVzZXIgaW5wdXRzIGJ5IHNwZWNpZnlpbmcgb3B0aW9ucy4gSXQgaXMgZW5hYmxlZCBieSBkZWZhdWx0LgoKYGBgdHlwZXNjcmlwdAppbXBvcnQgKiBhcyBzcGljeUlucHV0IGZyb20gJ3NwaWN5LWlucHV0JzsKClN3YWdnZXJVSSh7CiAgcGx1Z2luczogWwogICAgLy8gRGlzYWJsZSB0aGUgbWVtb3J5IG9mIHVzZXIgaW5wdXRzLgogICAgc3BpY3lJbnB1dC5nZXRQbHVnaW4oeyBwZXJzaXN0VXNlcklucHV0czogZmFsc2UgfSkKICBdCn0pCmBgYAoKIyMjIHByZWZpeAoKQnkgc3BlY2lmeWluZyBhbiBpbnRlcm5hbCBwcmVmaXgsIFNwaWN5IElucHV0IGNhbiBzZXBhcmF0ZSBuYW1lc3BhY2VzIGZvciBzdG9yaW5nIHZhcmlvdXMgdHlwZXMgb2YgZGF0YS4gVGhlIGRlZmF1bHQgdmFsdWUgaXMgYChkZWZhdWx0KWAuIFRoaXMgZmVhdHVyZSBpcyBvbmx5IG5lY2Vzc2FyeSB3aGVuIGRlcGxveWluZyB0aGUgc2FtZSBzcGVjIEFQSSBhY3Jvc3MgbXVsdGlwbGUgZW52aXJvbm1lbnRzIGFuZCB2aWV3aW5nIHRoZW0gaW5kaXZpZHVhbGx5IHRocm91Z2ggU3dhZ2dlciBVSSBhcnJhbmdlZCB1bmRlciB0aGUgc2FtZSBkb21haW4gZm9yIGVhY2ggZW52aXJvbm1lbnQuIE5vcm1hbGx5LCB0aGVyZSBpcyBubyBuZWVkIHRvIHNwZWNpZnkgdGhpcy4KCmBgYHR5cGVzY3JpcHQKaW1wb3J0ICogYXMgc3BpY3lJbnB1dCBmcm9tICdzcGljeS1pbnB1dCc7CgpTd2FnZ2VyVUkoewogIHBsdWdpbnM6IFsKICAgIHNwaWN5SW5wdXQuZ2V0UGx1Z2luKHsgcHJlZml4OiAibXktZW52IiB9KQogIF0KfSkKYGBgCgojIyBBUElzCgpZb3UgY2FuIG1hbmlwdWxhdGUgdXNlciBpbnB1dHMgdGhyb3VnaCB0aGUgc3lzdGVtLgoKYGBgdHlwZXNjcmlwdApjb25zdCBzeXN0ZW0gPSBTd2FnZ2VyVUkoewogIHBsdWdpbnM6IFsKICAgIHNwaWN5SW5wdXQuZ2V0UGx1Z2luKCkKICBdCn0pCmBgYAoKIyMjIEdldCB1c2VyIGlucHV0cwoKWW91IGNhbiBnZXQgYWxsIGN1cnJlbnQgdXNlciBpbnB1dHMuCgpgYGB0eXBlc2NyaXB0CmNvbnN0IHNwaWN5SW5wdXRTZWxlY3RvcnMgPSBzcGljeUlucHV0LnNlbGVjdG9ycyhzeXN0ZW0pOwpjb25zdCBpbnB1dHMgPSBzcGljeUlucHV0U2VsZWN0b3JzLmlucHV0cygpOwpjb25zb2xlLmxvZyhpbnB1dHMpOwpgYGAKCiMjIyBTZXQgdXNlciBpbnB1dHMKCllvdSBjYW4gc2V0IHVzZXIgaW5wdXRzIHByb2dyYW1hdGljYWxseS4KCmBgYHR5cGVzY3JpcHQKY29uc3Qgc3BpY3lJbnB1dEFjdGlvbnMgPSBzcGljeUlucHV0LmFjdGlvbnMoc3lzdGVtKTsKCi8vIFNldCBwYXJhbWV0ZXIKc3BpY3lJbnB1dEFjdGlvbnMuc2V0UGFyYW1ldGVycygiL2NhdC97aWR9IiwgImdldCIsIHsKICAicGF0aC5pZC5oYXNoLTEwNDg3MDU4ODUiOiB7CiAgICAidmFsdWUiOiAieHh4eHgiCiAgfSwKfSk7Ci8vIE90aGVyIGZ1bmN0aW9ucwpzcGljeUlucHV0QWN0aW9ucy5zZXRSZXF1ZXN0Qm9keVZhbHVlOwpzcGljeUlucHV0QWN0aW9ucy5zZXRSZXF1ZXN0Q29udGVudFR5cGU7CnNwaWN5SW5wdXRBY3Rpb25zLnNldFJlc3BvbnNlQ29udGVudFR5cGU7CmBgYAoKWW91IGNhbiBpbmZlciB3aGF0IHNwZWNpZmljIHZhbHVlcyB0byBzZXQgYmFzZWQgb24gdGhlIGN1cnJlbnQgaW5wdXQgdmFsdWVzLgoKIyMjIFN1YnNjcmlwdGlvbgoKWW91IGNhbiBzdWJzY3JpYmUgdG8gY2hhbmdlcyBpbiB1c2VyIGlucHV0cy4KCmBgYHR5cGVzY3JpcHQKY29uc3Qgc3BpY3lJbnB1dEZuID0gc3BpY3lJbnB1dC5mbihzeXN0ZW0pOwovLyBzdWJzY3JpYmUKY29uc3QgdW5zdWJzY3JpYmUgPSBzcGljeUlucHV0Rm4uc3Vic2NyaWJlKCgpID0+IHsKICBjb25zdCBzcGljeUlucHV0U2VsZWN0b3JzID0gc3BpY3lJbnB1dC5zZWxlY3RvcnMoc3lzdGVtKTsKICBjb25zdCBpbnB1dHMgPSBzcGljeUlucHV0U2VsZWN0b3JzLmlucHV0cygpOwogIGNvbnNvbGUubG9nKGlucHV0cyk7Cn0pOwovLyB1bnN1YnNjcmliZQp1bnN1YnNjcmliZSgpOwpgYGAKCiMjIyBJbXBvcnQKCllvdSBjYW4gaW1wb3J0IHNhdmVkIHVzZXIgaW5wdXRzLgoKYGBgdHlwZXNjcmlwdApjb25zdCBzcGljeUlucHV0U2VsZWN0b3JzID0gc3BpY3lJbnB1dC5zZWxlY3RvcnMoc3lzdGVtKTsKLy8gR2V0IGlucHV0cwpjb25zdCBpbnB1dHMgPSBzcGljeUlucHV0U2VsZWN0b3JzLmlucHV0cygpOwoKLy8gLi4uCgpjb25zdCBzcGljeUlucHV0Rm4gPSBzcGljeUlucHV0LmZuKHN5c3RlbSk7Ci8vIEltcG9ydCBpbnB1dHMKc3BpY3lJbnB1dEZuLmltcG9ydElucHV0cyhpbnB1dHMpOwpgYGAK readmeEtag: '"dc8f5de6d85e6edb6447f2f709d54b4f179f57ec"' readmeLastModified: Mon, 08 Jan 2024 11:04:43 GMT repositoryId: 739820154 description: >- Spicy Input provides a concise interface for retrieving, setting, and subscribing to these user inputs. created: '2024-01-06T16:39:11Z' updated: '2024-01-23T13:16:16Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: ryusuke410 logo: https://avatars.githubusercontent.com/u/23029273?v=4 repoEtag: '"eded01d783489750fb0cf9b1cac912536167c744fca5d4c1e969229fb0b977bc"' repoLastModified: Tue, 23 Jan 2024 13:16:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/dog-egg/django-openapi v3: true id: 723c6dbf9a8a25ea1e4a110ec80b6c42 repositoryMetadata: base64Readme: >- IyMjIERlcHJlY2F0ZWQKClRoZSBsYXRlc3QgY29kZSBoYXMgYmVlbiBtb3ZlZCBpbnRvIHRoaXMgcHJvamVjdCAoW2RqYW5nby1vYXNpc10oaHR0cHM6Ly9naXRodWIuY29tL0RvZy1FZ2cvZGphbmdvLW9hc2lzKSkuCg== readmeEtag: '"a2ff9b7055d11ec3f7c5fc46c1cfa6e878ad6603"' readmeLastModified: Tue, 12 Mar 2024 01:37:47 GMT repositoryId: 507220717 description: null created: '2022-06-25T05:18:36Z' updated: '2024-03-12T01:41:41Z' language: Python archived: true stars: 0 watchers: 1 forks: 0 owner: Dog-Egg logo: https://avatars.githubusercontent.com/u/34474847?v=4 license: MIT repoEtag: '"31fd9457910f17facb47a7995369fb4ef9719c0ecf1b4416199861a63dbc8bbc"' repoLastModified: Tue, 12 Mar 2024 01:41:41 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/faaizz/code_demos v3: true id: d7b43de508370ff60c841810e5859577 repositoryMetadata: base64Readme: >- IyBDb2RlIERlbW9zCkEgY2F0YWxvZyBvZiBkZW1vbnN0cmF0aW9uYWwgYXBwbGljYXRpb25zIGZvciBsZWFybmluZyBwdXJwb3Nlcy4KCi0gW0dvOiBTaW1wbGUgQm9va3MgQVBJXSguL2dvX2Jvb2tzX2FwaSkKLSBbR286IFNpbXBsZSBDaGF0IEFwcCBBUEldKC4vZ29fc2ltcGxlX2NoYXRfYXBwX2FwaSkKLSBbR286IFNpbXBsZSBXZWJzaXRlXSguL2dvX3NpbXBsZV93ZWJzaXRlKQo= readmeEtag: '"b8a4fb92e43fb259378e8288997ef1c5b8a0e135"' readmeLastModified: Thu, 28 Mar 2024 21:41:16 GMT repositoryId: 680496485 description: A catalog of demonstrational applications for learning purposes. created: '2023-08-19T12:39:49Z' updated: '2024-03-28T21:41:20Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: Faaizz logo: https://avatars.githubusercontent.com/u/35711052?v=4 license: MIT repoEtag: '"8659ee29fd7666bda5246b27b38423b7956727e1a23cb03cdc6a8c5bbf0cc689"' repoLastModified: Thu, 28 Mar 2024 21:41:20 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/cdn77/patroni-openapi v3: true id: aa12a36dcec3c4ba970075cd2afb2722 repositoryMetadata: base64Readme: >- IyBQYXRyb25pIE9wZW5BUEkgdjMgc3BlYwoKW1BhdHJvbmkgQVBJIERvY3NdKGh0dHBzOi8vcGF0cm9uaS5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvcmVzdF9hcGkuaHRtbCkgYW5kIGl0cyBbc291cmNlXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vemFsYW5kby9wYXRyb25pL21hc3Rlci9kb2NzL3Jlc3RfYXBpLnJzdCkK readmeEtag: '"c6b76ad790ea9f89c5fcf910affee7b390d993ee"' readmeLastModified: Fri, 26 Apr 2024 09:10:40 GMT repositoryId: 754571505 description: Patroni OpenApi v3 spec created: '2024-02-08T10:42:41Z' updated: '2024-04-26T09:10:48Z' language: null archived: false stars: 0 watchers: 2 forks: 0 owner: cdn77 logo: https://avatars.githubusercontent.com/u/7156390?v=4 repoEtag: '"10b5a1e38e61525977fe92abbc23992fff8b3646ca9677b853484abe25dfb47c"' repoLastModified: Fri, 26 Apr 2024 09:10:48 GMT foundInMaster: true category: Server Implementations - source: openapi3 tags repository: https://github.com/muneebhashone/typescript-backend-toolkit v3: true id: c1dfcc32aa8b5672a42fc929c043b0fc repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGltZyBzcmM9ImxvZ28ucG5nIiB3aWR0aD0iMjAwcHgiIGFsaWduPSJjZW50ZXIiIGFsdD0iVHlwZVNjcmlwdCBCYWNrZW5kIFRvb2xraXQgbG9nbyIgLz4KICA8aDEgYWxpZ249ImNlbnRlciI+VHlwZVNjcmlwdCBCYWNrZW5kIFRvb2xraXQ8L2gxPgogIDxwIGFsaWduPSJjZW50ZXIiPgogICAgPGJyLz4KICAgIDxzdHJvbmc+RGV2ZWxvcGVy4oCRZmlyc3QgYmFja2VuZCB0b29sa2l0PC9zdHJvbmc+PGJyLz4KICAgIFNpbXBsZSBmb3IgYmVnaW5uZXJzLCBwcm9kdWN0aXZlIGZvciBleHBlcnRzIOKAlCBhbmQgc3RydWN0dXJlZCBzbyB0b29scyAoaW5jbHVkaW5nIEFJIGFnZW50cykgY2FuIGZvbGxvdyBpdCB3aXRob3V0IGd1ZXNzd29yay48YnIvPgogICAgUHJlZGljdGFibGUgYXJjaGl0ZWN0dXJlLiBUeXBl4oCRc2FmZSBwYXR0ZXJucy4gUGx1Z2lu4oCRYmFzZWQuIFByb2R1Y3Rpb27igJFyZWFkeS4KICA8L3A+CiAgPHAgYWxpZ249ImNlbnRlciI+CiAgICA8YSBocmVmPSIjcXVpY2stc3RhcnQiPlF1aWNrIFN0YXJ0PC9hPiDigKIKICAgIDxhIGhyZWY9IiNmZWF0dXJlcyI+RmVhdHVyZXM8L2E+IOKAogogICAgPGEgaHJlZj0iI2NsaSI+Q0xJPC9hPiDigKIKICAgIDxhIGhyZWY9IiNhcmNoaXRlY3R1cmUiPkFyY2hpdGVjdHVyZTwvYT4g4oCiCiAgICA8YSBocmVmPSJkb2NzLyI+RG9jdW1lbnRhdGlvbjwvYT4KICA8L3A+CjwvcD4KPGJyLz4KCiMjIFByZXJlcXVpc2l0ZXMKCkJlZm9yZSB5b3UgZ2V0IHN0YXJ0ZWQsIG1ha2Ugc3VyZSB5b3UgaGF2ZSB0aGUgZm9sbG93aW5nIGluc3RhbGxlZCBvbiB5b3VyIG1hY2hpbmU6CgotICoqRG9ja2VyICsgRG9ja2VyIENvbXBvc2UqKgotICoqUE5QTSoqCi0gKipOb2RlLmpzIDIwKyAoTFRTKSoqCgojIyBRdWljayBTdGFydAoKIyMjIFNjYWZmb2xkIGEgTmV3IFByb2plY3QKClVzZSB0aGUgQ0xJIHNjYWZmb2xkZXIgdG8gY3JlYXRlIGEgbmV3IHByb2plY3Qgd2l0aCBvbmx5IHRoZSBmZWF0dXJlcyB5b3UgbmVlZDoKCmBgYGJhc2gKIyBJbnRlcmFjdGl2ZSBtb2RlIChyZWNvbW1lbmRlZCkKbnB4IGNyZWF0ZS10YmstYXBwIG15LWJhY2tlbmQtYXBpCgojIE9yIHdpdGggYSBwcmVzZXQKbnB4IGNyZWF0ZS10YmstYXBwIG15LWFwaSAtLXByZXNldD1zdGFuZGFyZAoKIyBUaGVuIHN0YXJ0IGRldmVsb3BpbmcKY2QgbXktYXBpCnBucG0gZGV2CmBgYAoKU2VlIFtjcmVhdGUtdGJrLWFwcCBSRUFETUVdKHBhY2thZ2VzL2NyZWF0ZS10YmstYXBwL1JFQURNRS5tZCkgZm9yIGFsbCBvcHRpb25zIGFuZCBwcmVzZXRzLgoKKipWaXNpdCBgaHR0cDovL2xvY2FsaG9zdDozMDAwL2RvY3NgKiogZm9yIHlvdXIgYXV0by1nZW5lcmF0ZWQgQVBJIGRvY3VtZW50YXRpb24uCgojIyBGZWF0dXJlcwoKLSAqKkF1dG/igJFnZW5lcmF0ZWQgQWRtaW4gRGFzaGJvYXJkIChEamFuZ2/igJFzdHlsZSkqKiDigJQgTWFuYWdlIGRhdGEgYW5kIG9wcyBvdXQgb2YgdGhlIGJveC4KLSAqKkF1dG/igJFnZW5lcmF0ZWQgT3BlbkFQSSBkb2NzIChGYXN0QVBJ4oCRc3R5bGUpKiog4oCUIExpdmUgZG9jcyBhdCBgL2RvY3NgLCBhbHdheXMgaW4gc3luYy4KLSAqKlBsdWdpbuKAkWJhc2VkIHN5c3RlbSoqIOKAlCBBZGQgY2FwYWJpbGl0aWVzIChhdXRoLCByZWFsdGltZSwgYWRtaW4sIHF1ZXVlcykgYXMgcGx1Z2lucy4KLSAqKkZpcnN04oCRY2xhc3MgQ0xJIChBcnRpc2Fu4oCRbGlrZSkqKiDigJQgR2VuZXJhdGUgbW9kdWxlcywgcGx1Z2lucywgYW5kIHNlZWRlcnMgd2l0aCBjb25zaXN0ZW50LCB0eXBl4oCRc2FmZSBwYXR0ZXJucy4KLSAqKlByb2R1Y3Rpb27igJFyZWFkeSBzdGFjayoqIOKAlCBab2QgdmFsaWRhdGlvbiwgSldUIGF1dGgsIGZpbGUgdXBsb2FkcywgcXVldWVzLCBlbWFpbHMsIGFuZCBtb3JlLgoKIyMjIFdvcmtzIGdyZWF0IHdpdGggQUkgYWdlbnRzIChvcHRpb25hbCkKVGhlIHByb2plY3QgaXMgaW50ZW50aW9uYWxseSBzdHJ1Y3R1cmVkIHNvIGFsbW9zdCBhbnkgQUkgY29kaW5nIGFnZW50IGNhbiB3b3JrIHJlbGlhYmx5OiBwcmVkaWN0YWJsZSBmaWxlcywgY2xlYXIgc2NoZW1hcywgYW5kIGNvbnNpc3RlbnQgY29kZWdlbi4gVXNlIEFJIHRvb2xzIGlmIHlvdSBsaWtl4oCUb3IgaWdub3JlIHRoZW0gYW5kIGJ1aWxkIG5vcm1hbGx5LgoKIyMgQ0xJCgpUaGUgYHRia2AgY29tbWFuZOKAkWxpbmUgdG9vbCBpcyBwcm9kdWN0aXZpdHnigJFmb2N1c2VkLCBzaW1pbGFyIHRvIExhcmF2ZWzigJlzIEFydGlzYW4uIEl0IGhlbHBzIHlvdSBzY2FmZm9sZCBmZWF0dXJlcyBhbmQgcnVuIGNvbW1vbiB0YXNrcyBxdWlja2x5IGFuZCBjb25zaXN0ZW50bHkuCgpgYGBiYXNoCiMgU2NhZmZvbGQgZmVhdHVyZXMgZmFzdCAoQXJ0aXNhbuKAkWxpa2UpCnBucG0gdGJrIGc6bW9kdWxlIHVzZXIgLS1wYXRoIC9hcGkvdjEKcG5wbSB0YmsgZzpwbHVnaW4gYWRtaW4KcG5wbSB0Ymsgc2VlZAoKIyBEaXNjb3ZlciBjb21tYW5kcwpwbnBtIHRiayAtLWhlbHAKcG5wbSB0YmsgZzptb2R1bGUgLS1oZWxwCmBgYAoKIyMgQXJjaGl0ZWN0dXJlCgojIyMgQXQgYSBnbGFuY2UKYGBgCnNyYy8K4pSc4pSA4pSAIGFwcC8gICAgICAgICAgICMgQXBwbGljYXRpb24gc2V0dXAgYW5kIHBsdWdpbiByZWdpc3RyYXRpb24K4pSc4pSA4pSAIG1vZHVsZXMvICAgICAgICMgRG9tYWluIGxvZ2ljICh1c2VycywgcHJvZHVjdHMsIHBheW1lbnRzKQrilJzilIDilIAgcGx1Z2lucy8gICAgICAgIyBFeHRlbnNpYmxlIGZlYXR1cmVzIChhdXRoLCByZWFsdGltZSwgYWRtaW4pCuKUnOKUgOKUgCBsaWIvICAgICAgICAgICAjIEluZnJhc3RydWN0dXJlIGNsaWVudHMgKGRhdGFiYXNlLCBzdG9yYWdlLCBlbWFpbCkK4pSc4pSA4pSAIHV0aWxzLyAgICAgICAgICMgUHVyZSBmdW5jdGlvbnMgKEpXVCwgcGFzc3dvcmRzLCBwYWdpbmF0aW9uKQrilJzilIDilIAgY29uZmlnLyAgICAgICAgIyBUeXBlLXNhZmUgZW52aXJvbm1lbnQgY29uZmlndXJhdGlvbgrilJTilIDilIAgcm91dGVzLyAgICAgICAgIyBIVFRQIHdpcmluZyB3aXRoIE1hZ2ljUm91dGVyCmBgYAoKRWFjaCBmb2xkZXIgaGFzIG9uZSBqb2I6Ci0gYGFwcC9gIGJvb3RzIHRoZSBzZXJ2ZXIgYW5kIHdpcmVzIHBsdWdpbnMuCi0gYG1vZHVsZXMvYCBob2xkcyB5b3VyIGJ1c2luZXNzIGxvZ2ljIGFuZCBzY2hlbWFzLgotIGBwbHVnaW5zL2AgYWRkcyBmZWF0dXJlcyB3aXRob3V0IHRvdWNoaW5nIGNvcmUgY29kZS4KLSBgbGliL2AgY29ubmVjdHMgdG8gaW5mcmEgKERCLCBxdWV1ZXMsIGVtYWlsLCBzdG9yYWdlKS4KLSBgdXRpbHMvYCBjb250YWlucyBzbWFsbCwgcHVyZSBoZWxwZXJzLgotIGBjb25maWcvYCB2YWxpZGF0ZXMgZW52aXJvbm1lbnQgdmFyaWFibGVzLgotIGByb3V0ZXMvYCBkZWNsYXJlcyBIVFRQIGVuZHBvaW50cy4KCkNvcmUgY2FwYWJpbGl0aWVzIGluY2x1ZGVkOgotICoqT3BlbkFQSSBkb2NzKiogYXQgYC9kb2NzYAotICoqQXV0aCoqIChHb29nbGUgU2lnbuKAkUluICsgSldUIHNlc3Npb25zKQotICoqVXNlcnMgJiBSb2xlcyoqIChDUlVELCBSQkFDKQotICoqRmlsZSBVcGxvYWRzKiogKFMzL1IyL0xvY2FsLCBtdWx0aXBhcnQpCi0gKipWYWxpZGF0aW9uKiogKFpvZCwgZW5k4oCRdG/igJFlbmQpCi0gKipRdWV1ZXMqKiAoQnVsbE1RICsgQnVsbEJvYXJkKQotICoqUmVhbHRpbWUqKiAoU29ja2V0LklPKQotICoqUmVhbHRpbWUgVGVzdGVyKiogKFVJIHRvIHNlbmQvcmVjZWl2ZSBldmVudHMgYW5kIGluc3BlY3QgY2hhbm5lbHMpCi0gKipBZG1pbiBQYW5lbCoqIChzZXBhcmF0ZSBhdXRoKQotICoqRW1haWxzKiogKHRyYW5zYWN0aW9uYWwgdGVtcGxhdGVzKQoKIyMgRG9jdW1lbnRhdGlvbgoKLSAqKltDTEkgQ29tbWFuZHNdKGRvY3MvZG9jcy9ndWlkZXMvY2xpLWNvbW1hbmRzKSoqCi0gKipbQ3JlYXRpbmcgTW9kdWxlc10oZG9jcy9kb2NzL2d1aWRlcy9jcmVhdGluZy1tb2R1bGVzKSoqICAKLSAqKltNYWdpY1JvdXRlcl0oZG9jcy9kb2NzL2d1aWRlcy9tYWdpYy1yb3V0ZXIpKioKLSAqKltFbnZpcm9ubWVudCBDb25maWddKGRvY3MvZG9jcy9ndWlkZXMvZW52aXJvbm1lbnQtY29uZmlnKSoqCi0gKipbVGVzdGluZyAmIERlYnVnZ2luZ10oZG9jcy9kb2NzL2d1aWRlcy90ZXN0aW5nLWRlYnVnZ2luZykqKgoKIyMgRGV2ZWxvcG1lbnQgQ29tbWFuZHMKCmBgYGJhc2gKIyBEZXZlbG9wbWVudApwbnBtIGRldiAgICAgICAgICAgICAgIyBTdGFydCB3aXRoIGhvdCByZWxvYWQKcG5wbSBidWlsZCAgICAgICAgICAgICMgQ29tcGlsZSB0byBkaXN0LwpwbnBtIHN0YXJ0ICAgICAgICAgICAgIyBSdW4gcHJvZHVjdGlvbiBidWlsZApwbnBtIHR5cGVjaGVjayAgICAgICAgIyBUeXBlIGNoZWNraW5nCnBucG0gbGludCAgICAgICAgICAgICAjIEVTTGludCArIFByZXR0aWVyCgojIENMSSBUb29scyAgCnBucG0gdGJrIGc6bW9kdWxlIDxuYW1lPiAgICMgR2VuZXJhdGUgY29tcGxldGUgbW9kdWxlCnBucG0gdGJrIGc6cGx1Z2luIDxuYW1lPiAgICMgR2VuZXJhdGUgcGx1Z2luCnBucG0gdGJrIHNlZWQgICAgICAgICAgICAgICMgUnVuIGRhdGFiYXNlIHNlZWRlcnMKcG5wbSB0YmsgZG9jczpvcGVuYXBpICAgICAgIyBHZW5lcmF0ZSBPcGVuQVBJIHNwZWMKcG5wbSB0YmsgZG9jczpzZGsgICAgICAgICAgIyBHZW5lcmF0ZSBUeXBlU2NyaXB0IFNESwoKIyBPdGhlciBUb29scwpwbnBtIGVtYWlsOmRldiAgICAgICAgIyBQcmV2aWV3IGVtYWlsIHRlbXBsYXRlcwpgYGAKCiMjIFByb2R1Y3Rpb24gRGVwbG95bWVudAoKYGBgYmFzaAojIEJ1aWxkIGZvciBwcm9kdWN0aW9uCnBucG0gYnVpbGQKCiMgU2V0IHByb2R1Y3Rpb24gZW52aXJvbm1lbnQKY3AgLmVudi5wcm9kdWN0aW9uIC5lbnYKCiMgU3RhcnQgcHJvZHVjdGlvbiBzZXJ2ZXIKcG5wbSBzdGFydDpwcm9kCmBgYAoKIyMgQ29udHJpYnV0aW5nCgoxLiBGb3JrIHRoZSByZXBvc2l0b3J5CjIuIENyZWF0ZSB5b3VyIGZlYXR1cmUgYnJhbmNoIChgZ2l0IGNoZWNrb3V0IC1iIGZlYXR1cmUvYW1hemluZy1mZWF0dXJlYCkKMy4gQ29tbWl0IHlvdXIgY2hhbmdlcyAoYGdpdCBjb21taXQgLW0gJ0FkZCBhbWF6aW5nIGZlYXR1cmUnYCkKNC4gUHVzaCB0byB0aGUgYnJhbmNoIChgZ2l0IHB1c2ggb3JpZ2luIGZlYXR1cmUvYW1hemluZy1mZWF0dXJlYCkKNS4gT3BlbiBhIFB1bGwgUmVxdWVzdAoKIyMgTGljZW5zZQoKVGhpcyBwcm9qZWN0IGlzIGxpY2Vuc2VkIHVuZGVyIHRoZSBNSVQgTGljZW5zZSAtIHNlZSB0aGUgW0xJQ0VOU0VdKExJQ0VOU0UpIGZpbGUgZm9yIGRldGFpbHMuCgotLS0KCioqQnVpbHQgdG8gbWFrZSBiYWNrZW5kIGRldmVsb3BtZW50IGVmZm9ydGxlc3MqKgo= readmeEtag: '"1310c87697e0a71b68d62b9aa76ac0ea7bb1353c"' readmeLastModified: Mon, 03 Nov 2025 23:59:50 GMT repositoryId: 840890740 description: >- Production-ready Express.js/TypeScript framework with auto-generated OpenAPI, Artisan-style CLI, plugin system, JWT/sessions, BullMQ, React Email, Socket.io, Django-style admin, multi-provider storage/cache/email, Zod validation created: '2024-08-11T02:19:59Z' updated: '2026-02-01T16:11:56Z' language: TypeScript archived: false stars: 319 watchers: 5 forks: 57 owner: muneebhashone logo: https://avatars.githubusercontent.com/u/88892379?v=4 license: MIT repoEtag: '"720956f3eb01f4721998d20003bae397439b035b377aaa7c48c3dec1a0a5f8b7"' repoLastModified: Sun, 01 Feb 2026 16:11:56 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/centeredge/centeredge.textpay v3: true repositoryMetadata: base64Readme: >- IyBDZW50ZXJFZGdlLlRleHRQYXkKCk5vdGU6IFRleHRQYXkgKGEuay5hLiBEYXNoTm93KSBpcyBub3cga25vd24gYXMgSHlmaW4uCgpPcGVuQVBJIDMuMCBzcGVjIGJhc2VkIG9uIFtIeWZpbiBkb2N1bWVudGF0aW9uXShodHRwczovL2NlbnRlcmVkZ2UuaHlmaW4uYXBwL2RvY3MvMi4wYy9pbnRyb2R1Y3Rpb24pLgo= readmeEtag: '"7f5feeb34d530e97cc3e8429d81d05d396f45946"' readmeLastModified: Mon, 20 Nov 2023 19:58:14 GMT repositoryId: 446992249 description: OpenAPI 3 specification and C# SDK for the TextPay API created: '2022-01-11T21:52:37Z' updated: '2025-08-06T12:24:00Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: CenterEdge logo: https://avatars.githubusercontent.com/u/7118733?v=4 repoEtag: '"a853031d1343868ee76450f8174b01fa1798308cd00d38ee1ceefbb949b284b6"' repoLastModified: Wed, 06 Aug 2025 12:24:00 GMT foundInMaster: true category: Server id: af095871df8bb9408585fb1b3445e4cb - source: openapi3 tags repository: https://github.com/renatosouzaan/renatosouza-back-end-mvp v3: true id: 308c1c7d283c0184b97ebf0f381185f6 repositoryMetadata: base64Readme: >- # DMarket Structure

![Dmarket drawio](https://github.com/user-attachments/assets/84537415-65f1-4024-b06d-8765cf9f8681)

# Back-End (API) MVP

This API is a Flask-based RESTful API for managing products in a market database. It includes user authentication, admin functionality, and CRUD operations for products. The API is documented using OpenAPI 3 and is designed to work with a separate front-end application.

The front-end has its own repository; the link is provided below.

Front-end repository: [https://github.com/RenatoSouzaAN/RenatoSouza/front-end-mvp-react](https://github.com/RenatoSouzaAN/renatosouza-front-end-mvp-react)

## Table of Contents

-   [About](#about)
-   [Features](#features)
-   [Technology Stack](#technology-stack)
-   [Setup](#setup)
    -   [Prerequisites](#prerequisites)
    -   [Installation](#installation)
    -   [Explanation](#explanation)
    -   [Docker](#docker)
-   [Running the Server (Locally)](#running-the-server)
-   [API Documentation](#api-documentation)
-   [Key API Endpoints](#key-api-endpoints)
    -   [Get all products](#get-all-products)
    -   [Add a product](#add-a-product)
    -   [Update a product](#update-a-product)
    -   [Delete a product](#delete-a-product)
    -   [Set user as an admin](#set-user-as-an-admin)
    -   [Login into an account](#login-into-an-account)
    -   [Logout from account](#logout-from-account)
    -   [Get all users](#get-all-users)
-   [Responses](#responses)
-   [Authentication](#authentication)
-   [Contributors](#contributors)
-   [License](#license)

## About

This back-end API manages market products and supports user authentication with Auth0. It's built with Flask and supports operations like adding, updating, and deleting products. Admin functionality is also included for managing users. It works alongside a separate front-end, linked above.

## Features

-   User authentication with Auth0
-   Admin user management
-   CRUD operations for products
-   User-specific product management
-   OpenAPI 3 documentation

## Technology Stack

This back-end is built using the following technologies:

-   **Flask**: Micro web framework for Python.
-   **Flask-SQLAlchemy**: SQLAlchemy extension for Flask.
-   **Flask-Migrate**: Database migrations for Flask applications.
-   **Flask-CORS**: CORS support for Flask.
-   **Flask-OpenAPI3**: OpenAPI 3 integration for API documentation.
-   **Authlib**: OAuth and OpenID Connect library for Python.
-   **SQLite**: Lightweight, serverless database engine.
-   **Docker**: Platform for developing, shipping, and running applications in containers.

## Setup

### Prerequisites

-   Python 3.x installed
-   Pip package manager
-   Auth0 account and application setup

### Installation

It's highly recommended to use a virtual environment.

1. Clone the repository:

    ```
    git clone https://github.com/RenatoSouzaAN/RenatoSouza-back-end-mvp.git
    cd RenatoSouza-back-end-mvp
    ```

2. Create a `.env` file in the root directory with the following content:

   ```
   AUTH0_DOMAIN=your_auth0_domain
   API_AUDIENCE=your_api_audience
   CLIENT_ID=your_client_id
   CLIENT_SECRET=your_client_secret
   API_MANAGEMENT_CLIENT_ID=your-auth0-management-client-id
   API_MANAGEMENT_CLIENT_SECRET=your-auth0-management-client-secret
   ```

3.  If you're using docker, there's no need to do the steps below (4 to 7) and you should instead follow [Docker](#docker) instructions

4. Create virtual environment (Optional):

    ```
    python -m venv env
    ```

5. Access virtual environment (Optional):

    On Powershell:

    ```
    .\env\Scripts\activate
    ```

    On bash:

    ```
    source \env\Scripts\activate
    ```

6. Install dependencies:

    ```
    pip install -r requirements.txt
    ```

7. Set up the database:
    ```
    flask db init
    flask db migrate
    flask db upgrade
    ```

### Explanation

These commands ensure that your database schema is synchronized with your models:

-   **`flask db init`**: Initializes a new migration repository in your project.
-   **`flask db migrate`**: Generates a migration script to align the database schema with your models.
-   **`flask db upgrade`**: Applies the migration script to update the database schema.

These steps are crucial whenever you make changes to your database models to keep the schema up to date.

### Docker

To run the back-end through Docker, ensure you have [Docker](https://docs.docker.com/engine/install/) installed and running on your machine.

Navigate to the directory containing the Dockerfile and requirements.txt in your terminal. Execute **as administrator** the following command to build the Docker image:

```
$ docker build -t renatosouza-back-end-mvp .
```

Once the image is created, to run the container, **execute as an administrator** the following command:

```
$ docker run -p 5000:5000 renatosouza-back-end-mvp
```

Once running, to access the API, open [http://localhost:5000/](http://localhost:5000/) in your browser. If you have the front-end repository downloaded, you can also run it now and go to its address (http://localhost:3000/) to test the whole project.

## Running the Server

To start the Flask server:

```
flask run
```

or

```
python run.py
```

The API will be accessible at `http://localhost:5000`.

### API Documentation

OpenAPI 3 is integrated for API documentation. After starting the server, visit http://localhost:5000/openapi to explore the API endpoints and interact with them.

## Key API Endpoints

### Get all products

-   **GET** `/products`
    -   Returns a list of all products in the database.

### Add a product

-   **POST** `/products/create`
    -   Adds a new product to the database. Requires JSON payload with `name`, `price`, and `quantity`. (authenticated)

### Update a product

-   **PUT** `/products/<product_id>/update`
    -   Upgrades the product with the specified `product_id` from the database. Requires a JSON payload with any of the following fields: description, price, or quantity. (authenticated)

### Delete a product

-   **DELETE** `/products/<product_id>/delete`
    -   Deletes the product with the specified `product_id` from the database. (authenticated)

### Set user as an admin
-   **POST** `/admin/set`
    -   Changes the privileges of a user by setting it as an admin (authenticaded) (admin only)

### Login into an account
-   **GET** `/login`
    -   Initiate login process by requesting authentication through Auth0 API

### Logout from account
-   **GET** `/logout`
    -   Logs out the current logged user by requesting Auth0 API

### Get all users
-   **GET** `/admin/users`
    -   Returns a list of all user in the database. (authenticaded) (admin only)

## Responses

-   Successful responses include appropriate HTTP status codes and JSON payloads.
-   Error responses provide meaningful error messages and status codes.

## Authentication

This API uses Auth0 for authentication. Users need to authenticate through the `/login` endpoint, which will redirect to Auth0 for login. After successful authentication, users receive a JWT token which should be included in the `Authorization` header for authenticated requests.

## Contributors

-   Renato Souza de Almeida Neto <renatosouza.an@gmail.com>

## License

This project is licensed under the Custom Code License Agreement. See the [LICENSE](./LICENSE) file for more details.
 readmeEtag: '"f57e959c0091dafcc5d54151f60e799f59037d35"' readmeLastModified: Tue, 10 Dec 2024 14:06:42 GMT repositoryId: 818426871 description: >- Back-end of the MVP developed for the Sprint: Advanced Back-end Development course at PUC-Rio. created: '2024-06-21T20:40:10Z' updated: '2024-12-13T12:27:26Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: RenatoSouzaAN logo: https://avatars.githubusercontent.com/u/85565491?v=4 license: NOASSERTION repoEtag: '"2b6be44b72b07ee5990db6b26a45f880feed44a3c4e66f0c5e9c11328e863e07"' repoLastModified: Fri, 13 Dec 2024 12:27:26 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/open-banking/offers v3: true repositoryMetadata: base64Readme: >- IyBvZmZlcnMKT3BlbiBCYW5raW5nIE9mZmVycyBBUEkgYnVpbHQgb24gdG9wIG9mIGxpZ2h0LTRqCg== readmeEtag: '"3dae9a104ad5caaf69e3ac2e18c1205dc6881761"' readmeLastModified: Thu, 25 Jul 2024 20:09:55 GMT repositoryId: 235865977 description: Open Banking Offers API built on top of light-4j created: '2020-01-23T19:06:03Z' updated: '2026-02-03T01:08:47Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"f16d946b721cbe5f1a19a35bd90bd6c868d4d6cde33beab5329c866f6619e9a3"' repoLastModified: Tue, 03 Feb 2026 01:08:47 GMT foundInMaster: true category: - Documentation - Server - Parsers id: 3d4a34accc0845683d2874d3aef8f9fa - source: https://openapi.tools/ name: Alphadoc category: - Documentation - Description Validators link: https://alphadoc.io language: SaaS source_description: >- Alphadoc is a full featured developer experience platform. API components with granular parameter control and diagrams generated from your OpenAPI. Not only the reference, but the entire set of tutorials and guides automatically updates when the underlying API updates. v2: true v3: true v3_1: true id: 0143b34d223b2f7897e29fce204e72d9 foundInMaster: true - source: openapi3 tags repository: https://github.com/swrlab/swrlab v3: true id: 75fd16ecc0ecd5ba9a0051b517b3bdaa repositoryMetadata: base64Readme: >- IyBTV1IgQXVkaW8gTGFiCgojIyBQcmluY2lwbGVzCgpMZWFybiBhYm91dCBvdXIgRW5naW5lZXJpbmcgYW5kIFByb2plY3QgUHJpbmNpcGxlcyBpbiBbUFJJTkNJUExFUy5tZF0oUFJJTkNJUExFUy5tZCkuCgojIyBNb3JlIGZyb20gdGhlIEF1ZGlvIExhYgoKLSBPdXIgTm9kZUpTIHRvb2xzOiBbYEBzd3JsYWIvbm9kZS11dGlsc2BdKGh0dHBzOi8vZ2l0aHViLmNvbS9zd3JsYWIvbm9kZS11dGlscykgLyBvbiBbbnBtIGBAc3dybGFiL3V0aWxzYF0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2UvQHN3cmxhYi91dGlscykKLSBQdWJsaWMgcmVwb3NpdG9yaWVzIG9uIEdpdEh1YjogW3N3cmxhYi9yZXBvc2l0b3JpZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9vcmdzL3N3cmxhYi9yZXBvc2l0b3JpZXM/cT0mdHlwZT1wdWJsaWMmbGFuZ3VhZ2U9JnNvcnQ9bmFtZSkKLSBQdWJsaWMgcGFja2FnZXMgb24gbnBtOiBbbnBtanMuY29tL29yZy9zd3JsYWJdKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9vcmcvc3dybGFiKQoKIyMgUHVibGljIEtleQoKWW91IGNhbiBmaW5kIG91ciBwdWJsaWMga2V5IGZvciBgbGFiIFthdF0gc3dyLmRlYCBvbiBba2V5cy5vcGVucGdwLm9yZ10oaHR0cHM6Ly9rZXlzLm9wZW5wZ3Aub3JnL3NlYXJjaD9xPWxhYiU0MHN3ci5kZSkuCg== readmeEtag: '"34d717782e4b50612edcaabd42453ab3d4264abc"' readmeLastModified: Tue, 20 Aug 2024 11:29:35 GMT repositoryId: 208029195 description: 🤝 SWR Audio Lab created: '2019-09-12T10:52:47Z' updated: '2024-08-26T11:39:23Z' language: null archived: false stars: 6 watchers: 3 forks: 0 owner: swrlab logo: https://avatars.githubusercontent.com/u/55231245?v=4 repoEtag: '"1a772dd22f618ee2a15094f47012d5f119d077f3516b91323f121326b64b51ee"' repoLastModified: Mon, 26 Aug 2024 11:39:23 GMT category: - Low-level Tooling - Server Implementations foundInMaster: true oldLocations: - https://github.com/swrlab/swr-radiohub-docs - name: Bitstream API Analytics source: Tooling repository issues source_description: >- Bitstream is an API analytics platform designed to provide insights into API usage, performance, and security. It offers tools for monitoring and alerting, collaboration and intelligent features to automate API management.- Repository: https://github.com/bitstreamapis link: https://www.bitstreamapis.com v3_1: true v3: true v2: true sourceIssueMetadata: issueNumber: 158 author: hedleysmith createdAt: '2024-07-30T22:56:42Z' updatedAt: '2024-07-30T22:56:42Z' url: https://github.com/OAI/tools.openapis.org/issues/158 status: open id: b75cd921cb6a42d3f0e6456c2a9e6bd9 foundInMaster: true - source: https://openapi.tools/ name: orval category: - Code Generators - Mock language: - Typescript - React - Vue - Svelte - Angular source_description: >- orval is able to generate client with appropriate type-signatures (TypeScript) from any valid OpenAPI v3 or Swagger v2 specification, either in yaml or json formats. Generate, valid, cache and mock in your frontend applications all with your OpenAPI specification. 🍺 link: https://orval.dev repository: https://github.com/orval-labs/orval v2: true v3: true id: 72b4e3cc7ddcef07aadfe99a81245653 repositoryMetadata: base64Readme: >- [![npm version](https://badge.fury.io/js/orval.svg)](https://badge.fury.io/js/orval)
![NPM Downloads](https://img.shields.io/npm/dm/orval?color=purple)
[![License: MIT](https://img.shields.io/badge/License-MIT-yellow.svg)](https://opensource.org/licenses/MIT)
[![tests](https://github.com/orval-labs/orval/actions/workflows/tests.yaml/badge.svg)](https://github.com/orval-labs/orval/actions/workflows/tests.yaml)
[![orval](https://snyk.io/advisor/npm-package/orval/badge.svg)](https://snyk.io/advisor/npm-package/orval)
[![Gurubase](https://img.shields.io/badge/Gurubase-Ask%20Orval%20Guru-006BFF)](https://gurubase.io/g/orval)

<p align="center">
  <img src="./logo/orval-logo-horizontal.svg?raw=true" width="500" height="160" alt="orval - Restfull Client Generator" />
</p>
<h1 align="center">
  Generate Typescript clients from OpenAPI specification!
</h1>

### Code Generation

`orval` generates type-safe JS clients (TypeScript) from any valid OpenAPI v3 or Swagger v2 specification, either in `yaml` or `json` formats.

> [!IMPORTANT]
> Version [8.0.0+](https://orval.dev/docs/versions/v8) comes with a lot of improvements and changes please see the [Migration Guide](https://orval.dev/docs/versions/v8)

### Supported clients

`generate` models, requests, hooks, [mocks](https://mswjs.io/) and more, for these supported clients:

- [React](https://react.dev/)
- [React Query](https://tanstack.com/query/latest/docs/framework/react/overview)
- [React with swr](https://swr.vercel.app/)
- [Vue Query](https://tanstack.com/query/latest/docs/framework/vue/overview)
- [Svelte Query](https://tanstack.com/query/latest/docs/framework/svelte/overview)
- [Solid Query](https://tanstack.com/query/latest/docs/framework/solid/overview)
- [SolidStart](https://start.solidjs.com/)
- [Angular](https://angular.dev/)
- [Angular Query](https://tanstack.com/query/latest/docs/framework/angular/overview)
- [Hono](https://hono.dev/)
- [zod](https://zod.dev/)
- [native fetch](https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API)
- [mcp](https://modelcontextprotocol.io/introduction)

### Samples

You can find some samples below:

- [react app](https://github.com/orval-labs/orval/tree/master/samples/react-app)
- [react query](https://github.com/orval-labs/orval/tree/master/samples/react-query)
- [svelte query](https://github.com/orval-labs/orval/tree/master/samples/svelte-query)
- [vue query](https://github.com/orval-labs/orval/tree/master/samples/vue-query)
- [react app with swr](https://github.com/orval-labs/orval/tree/master/samples/react-app-with-swr)
- [angular app](https://github.com/orval-labs/orval/tree/master/samples/angular-app)
- [angular query](https://github.com/orval-labs/orval/tree/master/samples/angular-query)
- [hono](https://github.com/orval-labs/orval/tree/master/samples/hono)
- [next app with fetch](https://github.com/orval-labs/orval/tree/master/samples/next-app-with-fetch)
- [mcp server](https://github.com/orval-labs/orval/tree/master/samples/mcp)

### Playground

Try Orval out for yourself using our [Playground](https://orval.dev/playground) application!

## Developers

This project uses [Yarn](https://yarnpkg.com/) for package management and building. Here are the key scripts available for development:

### Prerequisites

Before using Yarn scripts, ensure you have Yarn installed. You can install it globally using npm:

```bash
npm install -g yarn
```

Alternatively, you can enable Corepack (which comes with Node.js 16.10+) to manage Yarn:

```bash
corepack enable
```

### Build Scripts

- **`yarn nuke:all`** - Completely clean your workspace by removing all build artifacts, node_modules, and cached files. Use this when you want to start fresh.

- **`yarn build`** - Build the project and make changes available to the workspace. Run this after making code changes to compile TypeScript and prepare the project for use.

### Test Scripts

- **`yarn test`** - Run unit tests in all packages.

- **`yarn update-samples`** - Generate sample outputs using the newly built version of Orval. This regenerates the sample code based on the current build.

- **`yarn test:samples`** - Run tests in the samples directory using the newly generated output from `update-samples`.

- **`yarn test:cli`** - Test that the generated output (not samples) is valid TypeScript. This validates the TypeScript compilation of the generated code.

### Development Workflow

A typical development workflow would be:

1. Make your code changes
2. Run `yarn build` to compile your changes
3. Run `yarn test` to ensure unit tests pass
4. Run `yarn update-samples` to regenerate sample outputs
5. Run `yarn test:samples` to verify samples work correctly
6. Run `yarn test:cli` to validate TypeScript compilation

If you encounter issues or want to start completely fresh:

1. Run `yarn nuke:all` to clean everything
2. Reinstall dependencies and rebuild from scratch

## Sponsors

Thank you to all our sponsors! 🍻

Support orval development by [Open Collective](https://opencollective.com/orval) and your logo will be displayed here with a link to your website.

<a href="https://opencollective.com/orval">
  <img src="https://orval.dev/images/orval-logo-horizontal.svg?raw=true" width="300" alt="Become a sponsor" />
</a>

## Backers

Thank you to all our backers! 🙏

Support us with a one-time donation and help us continue our activities on [Open Collective](https://opencollective.com/orval).

<a href="https://opencollective.com/orval">
  <img src="https://orval.dev/images/emblem.svg" width="50" height="50" alt="Backer" />
</a>
<a href="https://opencollective.com/orval">
  <img src="https://orval.dev/images/emblem.svg" width="50" height="50" alt="Backer" />
</a>
<a href="https://opencollective.com/orval">
  <img src="https://orval.dev/images/emblem.svg" width="50" height="50" alt="Backer" />
</a>

**Note:** After becoming a sponsor or backer, please contact us on [Discord](https://discord.gg/6fC2sjDU7w) to upload your logo.

## Star History

<a href="https://star-history.com/#orval-labs/orval&Date">
  <picture>
    <source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/svg?repos=orval-labs/orval&type=Date&theme=dark" />
    <source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/svg?repos=orval-labs/orval&type=Date" />
    <img alt="Star History Chart" src="https://api.star-history.com/svg?repos=orval-labs/orval&type=Date" />
  </picture>
</a>

### All Thanks To Our Contributors:

<a href="https://github.com/orval-labs/orval/graphs/contributors">
  <img src="https://contrib.rocks/image?repo=anymaniax/orval" />
</a>
 readmeEtag: '"6db5ee1ebbfef0101af3fd72a50cd59239e70df5"' readmeLastModified: Tue, 27 Jan 2026 16:35:49 GMT repositoryId: 238402553 description: >- orval is able to generate client with appropriate type-signatures (TypeScript) from any valid OpenAPI v3 or Swagger v2 specification, either in yaml or json formats. 🍺 created: '2020-02-05T08:31:50Z' updated: '2026-02-06T04:02:52Z' language: TypeScript archived: false stars: 5321 watchers: 13 forks: 545 owner: orval-labs logo: https://avatars.githubusercontent.com/u/176128704?v=4 license: MIT repoEtag: '"29fe826aa5e21e389497adef2cfe2afee223a6219a98fbdb14e1d0efc938776c"' repoLastModified: Fri, 06 Feb 2026 04:02:52 GMT foundInMaster: true oldLocations: - https://github.com/anymaniax/orval - source: openapi3 tags repository: https://github.com/domapic/domapic-base v3: true repositoryMetadata: base64Readme: >- ![Domapic Base][domapic-base-logo-image]

# Domapic Base

Base for Domapic Node.js packages.

[![Build status][travisci-image]][travisci-url] <!-- [![Coverage Status][coveralls-image]][coveralls-url] [![Quality Gate][quality-gate-image]][quality-gate-url] --> [![js-standard-style][standard-image]][standard-url]

[![NPM dependencies][npm-dependencies-image]][npm-dependencies-url] [![Last commit][last-commit-image]][last-commit-url] <!-- [![Last release][release-image]][release-url] -->

[![NPM downloads][npm-downloads-image]][npm-downloads-url] [![Website][website-image]][website-url] [![License][license-image]][license-url]

---

## Table of Contents

* [Introduction](#introduction)
* [Quick Start](#quick-start)
* [Options](#options)
	* [Get config](#get-config)
	* [Set config](#set-config)
	* [Custom config](#custom-config)
* [Server](#server)
* [Adding API resources](#adding-api-resources)
	* [Open API definitions](#open-api-definitions)
	* [Operations](#operations)
  * [Custom middlewares](#custom-middlewares)
* [Client](#client)
* [Traces](#traces)
* [Errors](#errors)
* [Storage](#storage)
* [Utils](#utils)
* [Info](#info)
* [Authentication](#authentication)
	* [Api Key](#api-key)
	* [Jwt](#jwt)
	* [Authorization](#authorization)
* [CLI](#command-line-interface)
	* [Implementation](#implementation)
	* [Usage](#usage)
	* [Custom options and commands](#custom-options-and-commands)
* [Test suites](#test-suites)

---

## Introduction

This package is used as a base for [__domapic-controller__][domapic-controller-url] and [__domapic-service__][domapic-service-url] packages.

Maybe you´ll better want to read documentation about that pieces and learn how use them directly, because this is an internal dependency. Anyway, if you think it can be useful for you separately...

It provides:

* __Server__ with an extensible API:
	* Optional ssl protocol, just provide ssl certificate and key paths.
	* OAUTH. Jwt and/or apiKey customizable authentication to api resources of choice.
	* Authentication can be disabled in a range of IPs of choice, or completely.
	* Customizable authorization level for each API resource.
	* Add API resources using OpenApi 3.0 definitions.
	* Http OPTIONS method created for each API resource, auto describing it.
	* Automatic api parameters and body validation using the openapi schemas.
	* API operations as Promises.
	* Automatic error handling mapped to HTTP errors.
	* Openapi.json auto generated and served.
	* Built-in _Swagger_ web interface.
  * Allows serving statics under paths of your choice.
* __Client__ to other Domapic services:
	* Automatic authentication if API resource requires it.
	* Requests as Promises.
* __Configuration__
	* Built-in service command line options. (port, host, etc...).
	* Fully extensible with your own options.
	* Storable. Next executions will remember options if `--saveConfig` is specified in one execution.
* __Traces__
	* Six different log levels.
	* Ansi colored, at your choice.
	* Daily file, last ten stored.
* __Errors__
	* Custom errors for easy error handling.
	* Mapping to HTTP errors.
* __Storage__
	* Javascript objects to JSON at file system and viceversa.
	* File system access scoped to unique service instance folder.
* __CLI__. Easy implementable in your own package, it provides:
	* If the service is started using the CLI, the process will be executed in background, and managed using [_PM2_][pm2-url].
	* Multi-instanciable. Start many services instances providing different names.
	* CLI commands to stop or display logs.
	* Extensible with your own commands.

---

## Quick start

```js
// server.js file
const path = require('path')
const domapic = require('domapic-base')

domapic.Service({
  packagePath: path.resolve(__dirname),
  type: 'module'
}).then(service => {
  return service.server.start()
})
```

* The `packagePath` option must be the path where your package.json file is, in order to automatically create the `/api/about` api resource that can provide useful information about the package to other domapic services.
* The `type` option will be exposed in the `api/about` api resource directly, in order to inform about the service type.

```shell
# Start server
node ./server.js
```

Browse to http://localhost:3000 to open _Swagger_ interface and inspect API.

[back to top](#table-of-contents)

---

## Options

```shell
# Display help with detailed information about all options
node ./server.js --help
```

option | type | description | default
--- | --- | --- | ---
`name` | String | Service instance name. If not received, the service name will be the package name | -
`port` | Number | Http port used | 3000
`hostName` | String | Hostname for the server | -
`sslCert` | String | Path to an ssl certificate | -
`sslKey` | String | Path to an ssl key | - 
`authDisabled` | Array | Array of IPs or CIDR IP ranges with authentication disabled | ['127.0.0.1', '::1/128']
`auth` | Boolean | If false, the authentication will be disabled for all api resources and origins | true
`color` | Boolean | Use ANSI colors in traces | true
`logLevel` | String | Tracing level. Choices are 'log', 'trace', 'debug', 'info', 'warn' and 'error' | info
`path` | String | Path to be used as home path, instead of user´s default (.domapic folder will be created inside) | ~
`saveConfig` | Boolean | Save current options for next execution (except `name` and `path`) | false
`rejectUntrusted` | Boolean | Reject untrusted ssl certificates when using built-in client to make requests to another services | false

Example of setting options from command line:
```shell
node ./server.js --name=fooName --authDisabled=192.168.1.1 172.0.0.1 --logLevel=debug --color=false --auth=true
```

### Get config

Options defined from command line are available in the `config` object of the service.

`service.config.get([key])`

```js
domapic.Service({
  packagePath: path.resolve(__dirname)
}).then(service => {
  return service.config.get()
}).then(configuration => {
  console.log(configuration)
})
```

### Set config

It is not recommended, but, it if has sense, `config` properties can be modified programmatically. If you want to store some data, you´ll better want to read about the [storage feature](#storage). 

`service.config.set(key [, value])`

```js
domapic.Service({
  packagePath: path.resolve(__dirname)
}).then(service => {
  return service.config.set('fooKey', 'fooValue')
    .then(value => {
      console.log(value)
      // fooValue
    })
})
```

### Custom config

You can add your own custom configuration options. They will be seteable from command line execution, displayed in help and validated as the rest of options. Use `customConfig` parameter to define them.

[_Yargs_][yargs-url] is used as underlayer to manage options, so you can read its documentation for more details about how to define them:

```js
// Usage of customConfig parameter
domapic.Service({
  packagePath: path.resolve(__dirname),
  customConfig: {
    fooOption: {
      type: 'boolean',
      alias: ['foo-option'],
      describe: 'Testing a custom configuration option',
      default: true
    }
  }
}).then(service => {
  return service.config.get()
}).then(configuration => {
  console.log(configuration)
})
```
```shell
node ./server.js --fooOption=false
```

Custom options defined for a service should be defined in CLI implementation too, to make them available from command line interface. Read the [CLI custom options and commands](#custom-options-and-commands) chapter for further info.

Default options values (or saved values if the `--saveConfig` option is used) are saved into a file at `~/.domapic/<serviceName>/config/service.json`. This file can be edited manually, and the new values will be applied next time the service is started.

[back to top](#table-of-contents)

---

## Server

The `service.server` object has methods:

* `start` - Starts the server. Returns a promise, resolved when the server is running. Once the server is started, it is not possible to add more open api definitions, operations, or authentication implementations.
* `init` - Only initialize the server, adding all internal middlewares and routers, and returns the server instance. This allows you to add more custom middlewares, sockets, etc. This method should be called just before calling to the "start" method.
* `extendOpenApi` - Add open api definitions to the server. Read the [Adding API resources](#adding-api-resources) chapter for further info.
* `addOperations` - Add operations related to api paths. Read [Adding API resources](#adding-api-resources)).
* `addAuthentication` - Add authentication implementations. Read [Authentication](#authentication).
* `addAuthorization` - Add authorization roles. Read [Authentication](#authentication).
* `addMiddleware` - Add custom middlewares to api. Read [Custom middlewares](#custom-middlewares).
* `addStatic` - Serve statics. First argument defines server path, second argument defines fileSystem path. `addStatic("/assets", path.resolve(__dirname, "assets"))`. Statics added with this method will be served using gzip compression.

[back to top](#table-of-contents)

---

## Adding API resources

You can add your own API resources. They will be automatically added to the openapi.json definition, and will be available under the `/api` path of the server.

```js
// server.js file
const path = require('path')
const domapic = require('domapic-base')

const myOpenApi = require('./api/myOpenApi.json')

domapic.Service({
  packagePath: path.resolve(__dirname)
}).then(service => {
  return Promise.all([
    service.server.extendOpenApi(myOpenApi),
    service.server.addOperations({
      myApiOperation: {
        handler: (params, body, res) => {
          return Promise.resolve({
            hello: 'world'
          })
        }
      }
    })
  ]).then(() => {
    return service.server.start()
  })
})
```

### Open API definitions

`service.server.extendOpenApi(openApiDefinition)`

Openapi 3.0 spec is used to define new API paths. Read more about how to define paths in [_Swagger_ specification docs](https://swagger.io/specification/). You can even use the [_Swagger_ editor](https://swagger.io/swagger-editor/) to define and design your API, and afterwards, load the resultant .json files in _domapic base_.

You can add as many openApi definitions as you want, and for each one you can define "components", "tags", or "paths". The resultant `openapi.json` will be the result of extending all of them, after adding all needed base properties.

See here an [openApi definition example](lib/api/about/openapi.json), which is used internally to create the built-in `/about` api resource.

### Operations

`service.server.addOperations(apiOperationsObject)`

Each openApi path should contain a property called `operationId`. This value defines which operation will be executed when the api resource is requested.

Use the `addOperations` server method to add the operations. The operation key should match with the openApi `operationId` property.

Each operation can have properties:

* `handler` - The function that will be executed when the api resource is requested. Mandatory.
	* Arguments:
		* params - Request parameters. `params.path` and `params.query`.
		* body - Request body
		* res - Allows to set custom headers and statusCode to response. Examples: `res.status(201)`, `res.header('location', '/api/books/new-book')`
		* userData - If user is "logged in", The userData returned by the correspondant authentication method `verify` handler will be received in this property.
	* Returns:
		* Can return a Promise. If rejected, the error will be mapped to a correspondant html error. If resolved, the resolved value will be returned as response body.
		* If returns a value, the value will be returned as response body.
		* If throws an error, the error will be mapped to a correspondant html error.
* `parse` - Parse parameters from request.
	* It must be an object, with first level keys as request object where the parameter will be found, and second level keys as parameter name to be parsed. The `parser` function will receive the original value of the parameter as argument, and should return the parsed value.
	* Useful, for example, to convert numeric values from request params or query strings, that are received as strings, to real numbers.
* `auth` - This method will be executed to check if the user has enough permissions to perform an operation. Can be a function, or a string that defines which authorization role function has to be executed. If this method is not defined, the api resource will be available for all users. Read [Authentication](#authentication) for further info.
	* Arguments:
		* userData - The decoded data about the user that is making the request.
		* params - Request parameters. `params.path` and `params.query`.
		* body - Request body
	* Returns: 
		* Promise.resolve, or `true` to validate the user.
		* Any other returned value will result in a "Forbidden" response.

```js
service.server.addOperations({
  myApiOperation: {
    auth: (userData, params, body) => {
      if (userIsAllowed(userData)) {
        return Promise.resolve()
      }
      return Promise.reject()
    },
    handler: (params, body, res, userData) => {
      res.status(201)
      res.header('location', '/api/books/new-book')
      return Promise.resolve({
        hello: 'world'
      })
    },
    parse: {
      params: {
        id: (id) => {
          return parseInt(id, 10)
        }
      },
      query: {
        page: (page) => {
          return parseInt(page, 10)
        }
      }
    }
  }
})
```

### Custom middlewares

`service.server.addMiddleware(expressMiddleware)`

Custom express middlewares can be added. Will be added before all other internal middlewares, such as operation handlers.

```js
service.server.addMiddleware((req, res, next) => {
  console.log('Executing middleware before operation')
  next()
})
```


[back to top](#table-of-contents)

---

## Client

Make requests to other _Domapic_ services. Automatic authentication and error handling is provided.

```js
domapic.Service().then(service => {
  const client = new service.client.Connection('http://localhost:3000')
  return client.get('/about').then(response => {
    console.log(response)
  })
})
```

```js
// Client with two authentication methods example
domapic.Service().then(service => {
  const client = new service.client.Connection('http://localhost:3000',{
    apiKey: 'thisIsaFooApiKey',
    jwt: {
      user: 'fooUserName',
      password: 'fooPassword'
    }
  })
  return client.get('/about').then(response => {
    console.log(response)
  })
})
```

[back to top](#table-of-contents)

---

## Traces

There are six different levels of traces. Depending of the choiced log level when started the service, the trace will be printed to the console and written to the daily file or not.

All traces in a day are saved to a file, into `~/.domapic/<serviceName>/logs/<serviceName>.<date>.log`. Trace files older than ten days are automatically deleted.

When the service is started at background using the built-in CLI, logs are also saved to a file at `~/.domapic/<serviceName>/logs/<serviceName>.pm2.log`. It is recommended to install [_PM2 log rotate_](https://github.com/keymetrics/pm2-logrotate) to avoid this file growing too much.

Sorted tracer levels are: 'log', 'trace', 'debug', 'info', 'warn' and 'error'.

Tracer usage:

```js
domapic.Service().then(service => {
  return service.tracer.debug('testing').then(() => {
    return service.tracer.log('testing log')
  }).then(() => {
    return service.tracer.warn('This is a warning', 'This is part of the same warning')
  }).then(() => {
    return service.tracer.error(new Error('This will print the error stack'))
  }).then(() => {
    return service.tracer.error('Printed with error style, but no stack')
  })
})
```

There is an extra method called `group`, that allows to invoque different levels of tracers at a time:

```js
domapic.Service().then(service => {
  return service.tracer.group([
    {
      log: 'This is a log'
    },
    {
      trace: 'This is a trace'
    },
    {
      warn: 'This is a warn'
    }
  ])
})
```

[back to top](#table-of-contents)

---

## Errors

Custom errors constructors are provided through the `service.errors` object.

Custom errors usage:

```js
const Promise = require('bluebird')

domapic.Service().then(service => {
  return Promise.reject(new service.errors.BadData('Received bad data'))
    .catch(service.errors.BadData, () => {
      console.log('Bad data error caught')
      throw new service.errors.BadImplementation()
    })
})
```

Consult [all available error constructors](lib/bases/core/Errors.js) and its correspondences with html errors.

In addition to error constructors, three methods are provided in the `errors` object. These methods are used internally by _domapic-base_ in order to map the returned errors to HTML errors and viceversa:

* `isControlled` - Allows to know if error has been created with a custom error constructor
	* `service.errors.isControlled(error)`
	
```js
const error = new service.errors.Conflict()
console.log(service.errors.isControlled(error))
// true

error = new Error()
console.log(service.errors.isControlled(error))
// false
```

* `FromCode` - Returns an error created with the constructor correspondent to the provided html error status code:
	* `service.errors.FromCode(code, message [, stack] [, extraData])`
	
```js
return Promise.reject(service.errors.FromCode(403, 'Custom message'))
  .catch(service.errors.Forbidden, error => {
    console.log('Forbidden error caught')
    console.log(error.message)
    // Custom message
  })
```

* `toHTML` - Returns a [_Boomified_](https://www.npmjs.com/package/boom) error correspondent to the used error constructor. Each error constructor is mapped to an specific status code, ready to be returned by the API:
	* `service.errors.toHTML(error)`

```js
const error = service.errors.Forbidden()
console.log( service.errors.toHTML(error).output.payload.statusCode )
// 403
```

[back to top](#table-of-contents)

---

## Storage

Storage methods read and save json data from a file stored as `~/.domapic/<serviceName>/storage/service.json`.

```js
service.storage.set('fooProperty', {test: 'testing'})
  .then(() => {
    return service.storage.get('fooProperty')
  })
  .then(data => {
    console.log(data)
    // {test: 'testing'}
    return service.storage.remove('fooProperty')
  })
```

Methods

* `get` - Read data from storage file.
	* `service.storage.get([key])`
	* Arguments:
		* key - Optional, key of the object to get. If no provided, entire data is returned.
	* Returns: 
		* A promise, resolved with the correspondant data.
* `set` - Save data into the storage file.
	* `service.storage.set([key,] value)`
	* Arguments:
		* key - Optional. Key of the object to set. If no provided, entire data is overwritten by the given value.
		* value - Data to be saved.
* `remove` - Removes a property from the stored object.
	* `service.storage.remove(key)`
	* Arguments:
		* key - Key of the object to be removed.
* `getPath` - Get storage folder.
	* Returns: 
		* A promise, resolved with the absolute path to the storage folder.

[back to top](#table-of-contents)

---

## Utils

Set of utilities:

* `templates`
	* `compile` - Received an object containing a set of `key:'string'`, will use [_Handlebars_](http://handlebarsjs.com/) to compile each string, and return an object with same keys, but containing the compiled templates.
	* `compiled` - Set of precompiled templates, used internally.
```js
const templates = service.utils.templates.compile({
  myTemplate1: 'Value is: {{value}}'
})
console.log(templates.myTemplate1({
  value: 123
}))
// Value is: 123
```
* `cli`
	* `usedCommand` - Used internally by CLI. Returns the command used to start the current process.
* `services` - Set of utilities used internally to normalize services names, etc..

[back to top](#table-of-contents)

---

## Info

Static object containing information about the package, from the `package.json` file.

* `name` - Mandatory. The `package.json` must contain this property.
* `type` - Category of the service. It is defined with an argument when service is created. Possible values when using Domapic packages are `module`, `controller` or `plugin`.
* `version` - Mandatory. The `package.json` must contain this property.
* `description` - Mandatory. The `package.json` must contain this property.
* `homepage`
* `author`
* `license`

```js
console.log(service.info)
```

[back to top](#table-of-contents)

---

## Authentication

The server supports two types of OAUTH authentication methods, with built-in api "login" urls and token validations.

Each authentication strategy need some different methods to be provided in order to be activated. This externally provided methods have the responsibility of checking the user data, and then delegate the rest of the flow into the built-in security modules. In the case of Json Web Token, as the rest of the process is "token-based", this methods will be only invoqued at "login" or "refresh token" points.

To require an authentication method in your API operations, you must define the openapi `security` property in the correspondant API path:

```json
"paths": {
  "/fooOperationPath": {
    "post": {
      "security": [{
        "jwt": []
      }, {
        "apiKey": []
      }]
    }
  }
}
```

Use the server `addAuthentication` method to define your authentication implementations. The parameter must be an object containing keys `jwt`, `apiKey` and/or `disabled`, which will contain the specific configuration for each method:

`service.server.addAuthentication(authConfigObject)`

### Api key

Must contain properties:

* `verify`- Checks if the received api key is still allowed to be used.
	* Arguments:
		* apiKey -  Api key received in the request header.
	* Returns:
		* Promise.resolve(userData) -> Allowed, pass the user data to authorization methods.
		* Rejected promise -> Unauthorized.
* `authenticate` - API operation for requesting a new api key. This api point needs authentication as well, so, if your system  authentication is only api key based, you have to define an initial api key that could be used to request more in case it´s needed. This method supports _Json Web Token_ authentication as well, if it is implemented.
	* `auth` - Authorization method for the `/api/auth/apikey` _POST_ api resource.
		* Arguments:
			* userData - The decoded data about the user that is making the request.
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
	* `handler` - Operation handler for the `/api/auth/apikey` _POST_ api resource.
		* Arguments:
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
			* res - Allows to set custom headers and statusCode to response. Examples: `res.status(201)`, `res.header('location', '/api/books/new-book')`
			* userData - If user is "loged", here will be received the userData returned by the correspondant authentication method `verify` handler.
		* Should return a new api key.
* `revoke` - API operation for removing an api key. This api resource needs authentication as well.
	* `auth` - Authorization method for the `/api/auth/apikey` _DELETE_ api resource.
		* Arguments:
			* userData - The decoded data about the user that is making the request.
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
	* `handler` - Operation handler for the `/api/auth/apikey` _DELETE_ api resource.
		* Arguments:
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
			* res - Allows to set custom headers and statusCode to response. Examples: `res.status(201)`, `res.header('location', '/api/books/new-book')`
			* userData - If user is "loged", here will be received the userData returned by the correspondant authentication method `verify` handler.
		* Any returned value will be ignored, and not exposed to the api response.

Implementation example:

```js
service.server.addAuthentication({
  apiKey: {
    verify: apiKey => {
      // Check if apiKey is allowed, and return correspondant user data, or reject.
      return getUserDataFromApiKey(apiKey)
    },
    authenticate: {
      auth: (userData, params, body) => {
        // Check if user is allowed to create a new api key, resolve or reject
        return checkUserPermissionToManageApiKeys(userData)
      },
      handler: (params, body, res, userData) => {
        // Returns a new api key
        return getNewApiKey()
      }
    },
    revoke: {
      auth: (userData, params, body) => {
        // Check if user is allowed to remove an existant api key, resolve or reject
        return checkUserPermissionToManageApiKeys(userData)
      },
      handler: (params, body, res, userData) => {
        // Remove existant api key
        return removeApiKey(body.apiKey)
      }
    }
  }
})
```

### JWT

Properties:

* `secret` - Optional. String used to generate tokens. If not provided, a random secret is used.
* `expiresIn` - Number. Time of tokens expiration time in miliseconds. By default, 300 will be used if this option is not provided. It is not recommended to use a high value for this option. Tokens should expire in short time, and then, refresh tokens should be used to request a new token again.
* `authenticate` - API operation for requesting a new token. Will receive `user` and `password`, or `refreshToken`. Your implementation should be able to identify the user, and then return the correspondant data that will be stored in the _Json Web Token_. Afterwards, the API will use that data in each request to apply the correspondant authorization policy for each api resource. _Refresh Token_ is used to renew the user credentials without providing the user data again when the token expires.
	* `handler` - Operation handler for the `/api/auth/jwt` _POST_ api resource.
		* Arguments:
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
			* res - Allows to set custom headers and statusCode to response. Examples: `res.status(201)`, `res.header('location', '/api/books/new-book')`
		* Should return an object containing `userData` (an object with all user data that will be passed as argument to the API resources authorization handlers), and `refreshToken` if the request has not received it.
* `revoke` - API operation for removing a refresh token. This api resource needs authentication as well, and it only supports the `jwt` authentication method.
	* `auth` - Authorization method for the `/api/auth/jwt` _DELETE_ api resource.
		* Arguments:
			* userData - The decoded data about the user that is making the request.
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
	* `handler` - Operation handler for the `/api/auth/jwt` _DELETE_ api resource.
		* Arguments:
			* params - Request parameters. `params.path` and `params.query`.
			* body - Request body
			* res - Allows to set custom headers and statusCode to response. Examples: `res.status(201)`, `res.header('location', '/api/books/new-book')`
			* userData - If user is "loged", here will be received the userData returned by the correspondant authentication method `verify` handler.
		* Any returned value will be ignored, and not exposed to the api response.

Implementation example:

```js
service.server.addAuthentication({
  jwt: {
    secret: 'thisIsNotaRealTokenSecretPleaseReplaceIt',
    expiresIn: 180,
    authenticate: {
      handler: (params, body, res) => {
        // Check if user has right credentials, or refresh token. Returns user data (with new refresh token if not provided)
        if (body.refreshToken) {
          return getUserDataFromRefreshToken(body.refreshToken)
        }
        return checkUserData({
          name: body.user,
          password: body.password
        }).then((userData) => {
          return createNewRefreshToken(userData)
            .then((refreshToken) => {
              return Promise.resolve({
                userData: userData,
                refreshToken: refreshToken
              })
            })
        })
      }
    },
    revoke: {
      auth: (userData, params, body) => {
        // Check if user is allowed to remove an existant refresh token
        return checkUserPermissionToManageApiKeys(userData)
      },
      handler: (params, body, res, userData) => {
        // Remove existant refresh token
        return removeRefreshToken(body.refreshToken)
      }
    }
  }
})
```

### Disabled

When authentication is disabled for an specific origin because of the `authDisabled`, or `auth` options, the authorization handlers will not be executed. By default, an user with the format `{user: 'anonymous'}` will be passed to action handlers, but you can define your own "disabled" strategy, and return an user of your convenience to be passed.

Properties:

* `verify`- Function that should return the user to be passed to action handlers when authentication is disabled for an specific origin.

Implementation example:

```js
service.server.addAuthentication({
  disabled: {
    verify: () => Promise.resolve({
      name: 'your-auth-disabled-user',
      role: 'your-anonymous-role'
    })
  }
})
```

### Authorization

About authorization, each operation defined in the API can have its own `auth` method, that will receive the decoded user data as argument for each request, allowing to reject or allow an specific request based on your own security policy implementation.

The authorization method is agnostic in relation with the used authentication method, because it only receives the user data, no matter the method used to store or recover this data from the request.

Read more about how to define and use the `auth` methods in the [operations chapter](#operations).

An operation `auth` method can be defined as a function, or as a string that defines which "authorization role" function has to be executed. This "authorization roles" must to be defined in the server, using the `addAuthorization` method:

`service.server.addAuthorization(roleName, authHandler)`

```js
service.server.addAuthorization('fooRoleName', userData => {
  if (roleIsAllowed(userData.role)) {
    return Promise.resolve()
    // Execute the operation handler
  }
  return false
  // Forbidden response
}).then(() => {
  return service.server.addOperations({
    fooOperation: {
      auth: 'fooRoleName',
      handler: () => {
        return {}
      }
    }
  })
})
```

[back to top](#table-of-contents)

---

## Command Line Interface

A built-in CLI is provided, but it needs some steps in your package in order to expose it. Once it is implemented, it allows to start the service in background, managed using [_PM2_][pm2-url]. It also provides logs displaying, and a command to stop the service.
Also an API is at your disposal for defining new commands.

### Implementation

Follow these steps to implement the built-in CLI in your package:

* Create a `/bin/<your-cli-name>` file in your package. The name of the file should be equal to the command name that you want to use for your CLI. The content of this file must be:

```shell
#!/usr/bin/env node
require('../cli/index')
```

* Add a `bin` property to your `package.json`, and add an npm script to allow using the CLI through npm alternatively:

```json
  "bin": {
    "your-cli-name": "./bin/your-cli-name"
  },
  "scripts": {
    "your-cli-name": "./bin/your-cli-name"
  }
```

* Create a `/cli/index.js` file in your package. It must contains the CLI initialization:

```js
const path = require('path')
const domapic = require('domapic-base')

domapic.cli({
  script: path.resolve(__dirname, '..', 'server.js')
})
```
The `script` parameter must be the path to the file where you have your service initialization. This will be the process that will be started in background.

### Usage

Once you have installed globally your package, you´ll have the CLI available from command line. If not installed globally, all available commands can be executed as well using npm scripts, or executing directly the `/bin/your-cli-name` file. Next, the `start` example include the different invocation methods examples.

Default available commands are:

* `help`

```shell
your-cli-name help
# Displays help

your-cli-name start --help
# Displays help for start command
```

* `start` - Starts the service process in background. A custom name for the process instance can be provided as first argument, or using the `--name` option.

```shell
// global cli
your-cli-name start fooName --logLevel=debug
```

```shell
// npm script
npm run your-cli-name start fooName -- --logLevel=debug
```

```shell
// bin execution
./bin/your-cli-name start fooName --logLevel=debug
```

All available options for the `start` command are described in the [options chapter](#options) of this documentation.

* `stop` - Stops a background service instance:

```shell
your-cli-name stop

## If name was provided when started, if must be provided to stop:
your-cli-name stop fooName
```

* `logs` - Displays logs of a background service instance:

```shell
your-cli-name logs
# Displays logs

your-cli-name logs --lines=300
# Displays 300 last lines of logs (30 by default, if option is not provided)

## If name was provided when started, if must be provided to stop:
your-cli-name logs fooName --lines=300
```

### Custom options and commands

* Custom configuration
	A `customConfig` property can be defined in initialization object in order to define the custom options of your service:

```js
const path = require('path')
const domapic = require('domapic-base')
const customConfig = require('./customConfig')

domapic.cli({
  script: path.resolve(__dirname, '..', 'server.js'),
  customConfig: customConfig
})
```

Read more about how to define them in the [Custom options chapter](#custom-config)

* Custom commands
	A `customCommands` property can be defined in initialization object in order to extend the CLI features. It must be an object, whose keys will be the names of the custom commands. Each command must have properties:

	* `processName` - _String_. A reference name for the core in order to save the command default configuration, etc... As examples: `service`, `logs`, etc...
	* `describe` - Description for the command. Used when displaying help.
	* `cli` - Command name and arguments expression. [_Yargs_][yargs-url] is used as underlayer to manage commands, so you can read its documentation for more details about how to define them.
	* `options` - Available options for the command. All commands will inherit the options `name`, `color`, `logLevel`, `path` and `saveConfig`. Read [_Yargs_][yargs-url] documentation for further info about defining your own options.
	* `command` - Handler function that will be executed when command is invoqued.
		* Arguments:
			* `config` - An object containing default config, extended with stored config and extended with explicitly defined options in the command execution.
			* `cliUtils` - A set of methods:
				* `tracer` - A `tracer` object, as [described here](#traces).
				* `errors` - An `errors` object, as [described here](#errors).
				* `config` - A `config` object, as [described here](#get-config).
				* `utils` - An `utils` object, as [described here](#utils).
				* `process` - An object that allows to manage the related `script` property _pm2_ process instance related to provided mandatory option `--name`. It has methods:
					* `start` - Starts the process.
					* `stop` - Stops the process.
					* `logs` - Displays out logs while are being received.

	Example of custom command definition:

```js
const path = require('path')
const domapic = require('domapic-base')

domapic.cli({
  script: path.resolve(__dirname, '..', 'server.js'),
  customCommands: {
    restart: {
      processName: 'stopCustom',
      describe: 'Example of a custom command',
      cli: 'stopCustom <fooOption1>',
      options: {
        fooOption2: {
          type: 'boolean',
          describe: 'Foo option for command example',
          default: true
        }
      },
      command: (config, cliUtils) => {
        return cliUtils.tracer.info(JSON.stringify(config))
          .then(() => {
            cliUtils.process.stop()
          })
      }
    }
  }
})
```

Example of custom command usage:

```shell
your-cli-name stopCustom value1 --fooOption2=false --name=testing
# Will display configuration for the custom command, and then stop the process of script "server.js" with name "testing"
```

[back to top](#table-of-contents)

---

## Test Suites

This package uses [Narval][narval-url] as test suites runner.

Different test suites are included, categorized in "unit", "integration", and "end-to-end" tests. For further details, read the `.narval.yml`, and, if needed, you can learn more about Narval configuration at [its own documentation][narval-url].

Run tests:

```shell
npm test
```

[back to top](#table-of-contents)

[domapic-base-logo-image]: http://domapic.com/assets/domapic-logo.png

[coveralls-image]: https://coveralls.io/repos/github/domapic/domapic-base/badge.svg
[coveralls-url]: https://coveralls.io/github/domapic/domapic-base
[travisci-image]: https://travis-ci.org/domapic/domapic-base.svg?branch=master
[travisci-url]: https://travis-ci.org/domapic/domapic-base
[last-commit-image]: https://img.shields.io/github/last-commit/domapic/domapic-base.svg
[last-commit-url]: https://github.com/domapic/domapic-base/commits
[license-image]: https://img.shields.io/npm/l/domapic-base.svg
[license-url]: https://github.com/domapic/domapic-base/blob/master/LICENSE
[npm-downloads-image]: https://img.shields.io/npm/dm/domapic-base.svg
[npm-downloads-url]: https://www.npmjs.com/package/domapic-base
[npm-dependencies-image]: https://img.shields.io/david/domapic/domapic-base.svg
[npm-dependencies-url]: https://david-dm.org/domapic/domapic-base
[quality-gate-image]: https://sonarcloud.io/api/project_badges/measure?project=domapic-base&metric=alert_status
[quality-gate-url]: https://sonarcloud.io/dashboard?id=domapic-base
[release-image]: https://img.shields.io/github/release-date/domapic/domapic-base.svg
[release-url]: https://github.com/domapic/domapic-base/releases
[standard-image]: https://img.shields.io/badge/code%20style-standard-brightgreen.svg
[standard-url]: http://standardjs.com/

[website-image]: https://img.shields.io/website-up-down-green-red/http/domapic.com.svg?label=domapic.com
[website-url]: http://domapic.com/

[pm2-url]: http://pm2.keymetrics.io/
[yargs-url]: https://www.npmjs.com/package/yargs
[narval-url]: https://www.npmjs.com/package/narval
[domapic-controller-url]: https://npmjs.com/domapic-controller
[domapic-service-url]: https://npmjs.com/domapic-service
 readmeEtag: '"f13e9614ca50c99137e109ba65ec4340bc85ed97"' readmeLastModified: Sat, 02 Mar 2019 11:30:57 GMT repositoryId: 121413231 description: Base for Domapic Node.js packages created: '2018-02-13T17:31:30Z' updated: '2019-05-26T09:26:31Z' language: JavaScript archived: false stars: 0 watchers: 0 forks: 0 owner: domapic logo: https://avatars.githubusercontent.com/u/36418970?v=4 license: MIT repoEtag: '"33c77f2c1f853880a39ba473e5d704390466fd9b9809d77ad5161e0ae74c3f30"' repoLastModified: Sun, 26 May 2019 09:26:31 GMT foundInMaster: true id: fbd71be97b54eacae0bec76ede6f9a40 - source: openapi3 tags repository: https://github.com/chunghha/lb-poc v3: true repositoryMetadata: base64Readme: >- IyBsYi1wb2MKClshW0xvb3BCYWNrXShodHRwczovL2dpdGh1Yi5jb20vc3Ryb25nbG9vcC9sb29wYmFjay1uZXh0L3Jhdy9tYXN0ZXIvZG9jcy9zaXRlL2ltZ3MvYnJhbmRpbmcvUG93ZXJlZC1ieS1Mb29wQmFjay1CYWRnZS0oYmx1ZSktQDJ4LnBuZyldKGh0dHA6Ly9sb29wYmFjay5pby8pCgohW29hc2dyYXBoXSguL29hc2dyYXBoLnBuZykK readmeEtag: '"136082fa6fe940a6ac5cd14b5207ed75ca3ba56f"' readmeLastModified: Mon, 30 Mar 2020 14:57:28 GMT repositoryId: 153759478 description: Loopback 4 POC created: '2018-10-19T09:33:11Z' updated: '2023-01-28T03:27:42Z' language: TypeScript archived: true stars: 0 watchers: 1 forks: 0 owner: chunghha logo: https://avatars.githubusercontent.com/u/24458075?v=4 repoEtag: '"74a1491a915d134b7088d0e5ea5952120b58973fc77275a13edc698f07fcc83a"' repoLastModified: Sat, 28 Jan 2023 03:27:42 GMT foundInMaster: true category: Server id: 0aeabccd199aadf54cfdb62290aed132 - source: openapi3 tags repository: https://github.com/bepopov/intelligent_agent v3: true repositoryMetadata: repositoryId: 185402874 description: An intelligent agent for course creating created: '2019-05-07T13:00:33Z' updated: '2019-05-07T13:04:19Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: bepopov logo: https://avatars.githubusercontent.com/u/20209992?v=4 repoEtag: '"63f696ee4f5e8aaacd9f763dc0b2c5882903bdd51e0e7aa80e64af3ec21954b8"' repoLastModified: Tue, 07 May 2019 13:04:19 GMT foundInMaster: true id: 2f9a60bc091a5b228cf32f7105aa82c4 - source: openapi3 tags repository: https://github.com/epiphone/express-openapi-typer v3: true repositoryMetadata: base64Readme: >- IyBleHByZXNzLW9wZW5hcGktdHlwZXIKIVtdKGh0dHBzOi8vZ2l0aHViLmNvbS9lcGlwaG9uZS9leHByZXNzLW9wZW5hcGktdHlwZXIvd29ya2Zsb3dzL0NJL2JhZGdlLnN2ZykgWyFbbnBtIHZlcnNpb25dKGh0dHBzOi8vYmFkZ2UuZnVyeS5pby9qcy9leHByZXNzLW9wZW5hcGktdHlwZXIuc3ZnKV0oaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL2V4cHJlc3Mtb3BlbmFwaS10eXBlcikKCiMjIENhdXRpb24hIEFscGhhLWxldmVsIHNvZnR3YXJlIGFoZWFkLiBVc2UgYXQgeW91ciBvd24gcGVyaWwKQ29kZS1nZW5lcmF0aW9uLWZyZWUgY29udmVyc2lvbiBvZiAqKk9wZW5BUEkgdjMuMSoqIHNjaGVtYSBpbnRvICoqdHlwZS1jaGVja2VkIEV4cHJlc3MgcmVxdWVzdCBoYW5kbGVycyoqLgoKRGVyaXZlIEV4cHJlc3MgaGFuZGxlciB0eXBlcyBmcm9tIGFuIE9wZW5BUEkgc2NoZW1hIHRvIGdldAotIHR5cGUgZXJyb3JzIHdoZW4gYSBoYW5kbGVyIGRvZXNuJ3QgbWF0Y2ggdGhlIHNjaGVtYSwgYW5kCi0gYXV0by1jb21wbGV0aW9uIG9uIGhhbmRsZXIgcGF0aCwgYHJlcS5wYXJhbWAsIGByZXEucXVlcnlgLCBgcmVxLmJvZHlgLCBgcmVzLnNlbmQoKWAsIGByZXMuanNvbigpYCBldGMuCgpOb3RlIHRoYXQgdGhlIGxpYnJhcnkgKipkb2VzIG5vdCBwZXJmb3JtIHJ1bnRpbWUgdmFsaWRhdGlvbioqIGFnYWluc3QgdGhlIE9wZW5BUEkgc2NoZW1hOiBhZGQgc29tZXRoaW5nIGxpa2UgaHR0cHM6Ly9naXRodWIuY29tL0hpbHp1L2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0ZSBmb3IgdGhhdCBwdXJwb3NlLgoKKipSZXF1aXJlcyBPcGVuQVBJIHYzLjEqKi4gVGhpcyBsaWJyYXJ5IHJlbGllcyBoZWF2aWx5IG9uIGV4aXN0aW5nIEpTT04gU2NoZW1hIHRvb2xpbmcgd2hlcmVhcyBlYXJsaWVyIE9wZW5BUEkgdmVyc2lvbnMgdXNlIHRoZSBbT3BlbkFQSSBTY2hlbWEgT2JqZWN0XShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMC5tZCNzY2hlbWFPYmplY3QpIGluc3RlYWQgb2YgcHVyZSBKU09OIFNjaGVtYS4gT3BlbkFQSSBgdjMuMWAgaXMgeWV0IHVucHVibGlzaGVkOyB0cmFjayBwcm9ncmVzcyBbaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vaXNzdWVzLzIwMjUpLiBSZWFkIG1vcmUgYWJvdXQgdGhlIE9wZW5BUEkvSlNPTiBTY2hlbWEgZGl2ZXJnZW5jZSBhdCBodHRwczovL2FwaXN5b3V3b250aGF0ZS5jb20vYmxvZy9vcGVuYXBpLWFuZC1qc29uLXNjaGVtYS1kaXZlcmdlbmNlLXBhcnQtMSBhbmQgaG93IGB2My4xYCBzb2x2ZXMgaXQgYXQgaHR0cHM6Ly9waGlsLnRlY2gvMjAxOS8wOS8wNy91cGRhdGUtb3BlbmFwaS1qc29uLXNjaGVtYS8uCgojIyBJbnN0YWxsCgpgeWFybiBhZGQgZXhwcmVzcy1vcGVuYXBpLXR5cGVyYAoKIyMgVXNhZ2UKCkZpcnN0IGRlZmluZSB5b3VyIE9wZW5BUEkgc2NoZW1hIGFzIGEgVHlwZVNjcmlwdCB0eXBlOgoKYGBgdHlwZXNjcmlwdAppbnRlcmZhY2UgUGV0U3RvcmVTY2hlbWEgewogIG9wZW5hcGk6ICczLjEuMCcKICBpbmZvOiB7IC4uLiB9CiAgcGF0aHM6IHsKICAgICcvcGV0cyc6IHsKICAgICAgICBnZXQ6IHsgLi4ufQogICAgfSwKICAgIC4uLgogIH0KfQpgYGAKClRoZW4gb3ZlcnJpZGUgeW91ciBFeHByZXNzIHJvdXRlcidzIHR5cGUgZnJvbQoKYGBgdHlwZXNjcmlwdApjb25zdCByb3V0ZXIgPSBleHByZXNzLlJvdXRlcigpCmBgYAoKaW50byB0aGUgZm9sbG93aW5nOgoKYGBgdHlwZXNjcmlwdAppbXBvcnQgeyBPcGVuQVBJUm91dGVyIH0gZnJvbSAnZXhwcmVzcy1vcGVuYXBpLXR5cGVyJwoKY29uc3Qgcm91dGVyID0gKGV4cHJlc3MuUm91dGVyKCkgYXMgdW5rbm93bikgYXMgT3BlbkFQSVJvdXRlcjxQZXRTdG9yZVNjaGVtYT4KYGBgCgpIYW5kbGVyIGZ1bmN0aW9ucyBpbiBgcm91dGVyYCBub3cgZ2V0IHR5cGUtY2hlY2tlZCBhcyBwZXIgYFBldFN0b3JlU2NoZW1hYCEgRm9yIGV4YW1wbGUgd2hlbiB1c2luZyBbdGhlIGZ1bGwgc2FtcGxlIFBldFN0b3JlIHNjaGVtYV0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvZXhhbXBsZXMvdjMuMC9wZXRzdG9yZS1leHBhbmRlZC55YW1sKSB3ZSBlbmQgdXAgd2l0aCB0aGUgZm9sbG93aW5nOgoKIVtVc2FnZSBzYW1wbGVdKC4vZG9jL3VzYWdlLmdpZikKCiMjIyBBY2Nlc3MgT3BlbkFQSSBzY2hlbWEgdHlwZSBmcm9tIGEgcnVudGltZSB2YWx1ZQoKSXQgY2FuIGJlIHVzZWZ1bCB0byBpbnN0YW50aWF0ZSB0aGUgT3BlbkFQSSBzY2hlbWEgYXMgYSBydW50aW1lIHZhbHVlIGluc3RlYWQgb2YganVzdCBkZWZpbmluZyBhIHR5cGUuIEZvciBleGFtcGxlIHdoZW4gc2VydmluZyB0aGUgc2NoZW1hIGFzIGRvY3VtZW50YXRpb24gb3IgaGFuZGxpbmcgdmFsaWRhdGlvbiB3ZSBuZWVkIHRvIGFjY2VzcyB0aGUgc2NoZW1hIGF0IHJ1bnRpbWUuIEluIGNhc2VzIGxpa2UgdGhlc2UgY29tYmluZSBgdHlwZW9mYCBhbmQgW2BhcyBjb25zdGBdKGh0dHBzOi8vd3d3LnR5cGVzY3JpcHRsYW5nLm9yZy9kb2NzL2hhbmRib29rL3JlbGVhc2Utbm90ZXMvdHlwZXNjcmlwdC0zLTQuaHRtbCNjb25zdC1hc3NlcnRpb25zKSB0byBhY2Nlc3MgdGhlIHNjaGVtYSB2YWx1ZSdzIHR5cGU6CgpgYGB0eXBlc2NyaXB0CmNvbnN0IHBldFN0b3JlU2NoZW1hID0gewogIG9wZW5hcGk6ICczLjEuMCcsCiAgaW5mbzogeyAuLi4gfSwKICBwYXRoczogewogICAgJy9wZXRzJzogewogICAgICAgIGdldDogeyAuLi4gfQogICAgfSwKICAgIC4uLgogIH0KfSBhcyBjb25zdCAvLyA8LS0gaW1wb3J0YW50IQoKdHlwZSBQZXRTdG9yZVNjaGVtYSA9IHR5cGVvZiBwZXRTdG9yZVNjaGVtYQpgYGAKCiMjIyBBbGxvdyBhZGRpdGlvbmFsIHBhdGhzCgpCeSBkZWZhdWx0IGBPcGVuQVBJUm91dGVyYCBkb2Vzbid0IGFsbG93IGFueSBhZGRpdGlvbmFsIGhhbmRsZXJzIG5vdCBkZWZpbmVkIGluIHRoZSBPcGVuQVBJIHNjaGVtYS4gVG8gbG9vc2VuIHRoaXMgcmVzdHJpY3Rpb24geW91IGNhbiBleHBhbmQgdGhlIHR5cGUgYXMgZm9sbG93czoKCmBgYHR5cGVzY3JpcHQKaW1wb3J0ICogYXMgZXhwcmVzcyBmcm9tICdleHByZXNzJwoKY29uc3Qgcm91dGVyID0gZXhwcmVzcy5Sb3V0ZXIoKSBhcyBPcGVuQVBJUm91dGVyPFBldFN0b3JlU2NoZW1hPiAmIGV4cHJlc3MuUm91dGVyCmBgYAoKWW91IGNhbiBhbHNvIHNlbGVjdCBhIHN1YnNldCBvZiBgZXhwcmVzcy5Sb3V0ZXJgIHdpdGggW2BQaWNrYC9gT21pdGBdKGh0dHBzOi8vd3d3LnR5cGVzY3JpcHRsYW5nLm9yZy9kb2NzL2hhbmRib29rL3V0aWxpdHktdHlwZXMuaHRtbCNwaWNrdGspIHdoZW4gYWxsb3dpbmcgYWRkaXRpb25hbCBtZXRob2RzIG9ubHkgZm9yIGEgc3BlY2lmaWMgSFRUUCBtZXRob2QsIGZvciBleGFtcGxlLgoKIyMgVE9ETwotIEFsbCB0aGUgbGltaXRhdGlvbnMgZnJvbSBbYGpzb24tc2NoZW1hLXR5cGUtbWFwcGVyYF0oaHR0cHM6Ly9naXRodWIuY29tL2VwaXBob25lL2pzb24tc2NoZW1hLXR5cGUtbWFwcGVyKSBhcHBseSBoZXJlIGFzIHdlbGwKLSBIYW5kbGUgcmVxdWVzdCBoZWFkZXIgcGFyYW1ldGVycz8KLSBBUEkgY2xpZW50IHR5cGUgY2hlY2tpbmcsIHVzaW5nIEF4aW9zPwotIEZpZ3VyZSBhIHdheSBvdXQgb2YgdGhlIHVuZm9ydHVuYXRlIGBhcyB1bmtub3duYCBjYXN0Ci0gU3VwcG9ydCBwYXRoLWJhc2VkIGAkcmVmYHMsIG5vdCBqdXN0IGAkaWRgLWJhc2VkIG9uZXMKICAtIHJlcXVpcmVzIHNvbWUgc29ydCBvZiBtYW51YWwgbWFwcGluZyBhcyB3ZSBjYW4ndCB0YWtlIGAiIy9jb21wb25lbnRzL3NjaGVtYXMvTmV3VXNlciJgIGFwYXJ0IGF0IHR5cGUtbGV2ZWwKCiMjIFJlbGF0ZWQgcHJvamVjdHMKLSBDaGVjayBvdXQgW3Jlc3R5cGVkXShodHRwczovL2dpdGh1Yi5jb20vcmF3cm1hYW4vcmVzdHlwZWQpIGFuZCBbcmVzdC50c10oaHR0cHM6Ly9naXRodWIuY29tL2htaWwvcmVzdC50cykgZm9yIGRpZmZlcmVudCBhcHByb2FjaGVzIHRvIHR5cGUtc2FmZSBSRVNUIEFQSXMK readmeEtag: '"d0b621a6479949518dfec26510268a2f56d41cd8"' readmeLastModified: Mon, 09 Dec 2019 14:29:43 GMT repositoryId: 222856268 description: >- Code-generation-free conversion of OpenAPI schema into typed Express request handlers created: '2019-11-20T05:15:33Z' updated: '2019-12-09T14:29:50Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: epiphone logo: https://avatars.githubusercontent.com/u/1923531?v=4 license: MIT repoEtag: '"e0cf4803743c5f8e23fd33564cf01eb63a5d07494c2161f60820a8066220f843"' repoLastModified: Mon, 09 Dec 2019 14:29:50 GMT foundInMaster: true category: Parsers id: 3dc9d32a30d29bdcebf1249a3ca227f9 - source: openapi3 tags repository: https://github.com/sudokuru/frontend v3: true id: 69e0ef6db084252188c93ab7716b9309 repositoryMetadata: base64Readme: >- ![Sudokuru Logo](https://sudokuru.s3.amazonaws.com/goldLogoText.png)

# The Official Cross-Platform Frontend for our Free Open Source Sudoku Project

# 💻 Try our DEV site at: https://sudokuru.pages.dev/

# ⬇️ Download our alpha 📱Android, 🐧Linux, and 🪟Windows app builds at: https://sudokuru.itch.io/sudokuru

[![Pipeline](https://github.com/SudoKuru/Frontend/actions/workflows/pipeline.yml/badge.svg?branch=main)](https://github.com/SudoKuru/Frontend/actions/workflows/pipeline.yml)
[![Coveralls Coverage](https://coveralls.io/repos/github/SudoKuru/Frontend/badge.svg?branch=main)](https://coveralls.io/github/SudoKuru/Frontend?branch=main)
[![Codecov Coverage](https://codecov.io/gh/SudoKuru/Frontend/graph/badge.svg?token=XQSTKPTBFF)](https://codecov.io/gh/SudoKuru/Frontend)
[![FOSSA Status](https://app.fossa.com/api/projects/git%2Bgithub.com%2FSudoKuru%2FFrontend.svg?type=shield&issueType=license)](https://app.fossa.com/projects/git%2Bgithub.com%2FSudoKuru%2FFrontend?ref=badge_shield&issueType=license)
[![CodeFactor](https://www.codefactor.io/repository/github/sudokuru/frontend/badge)](https://www.codefactor.io/repository/github/sudokuru/frontend)
[![Codacy Badge](https://app.codacy.com/project/badge/Grade/1342f842a14f40cca856d6e81204f8ac)](https://app.codacy.com/gh/Sudokuru/Frontend/dashboard?utm_source=gh&utm_medium=referral&utm_content=&utm_campaign=Badge_grade)
[![Maintainability](https://api.codeclimate.com/v1/badges/4cd1d1027b7b2532c680/maintainability)](https://codeclimate.com/github/Sudokuru/Frontend/maintainability)

<!--- Note: The following text is duplicated in the AboutUsPage.tsx file -->

## Sudokuru is an open-source project focused on developing a world-class, cross-platform Sudoku app. We aim to provide a delightful user experience while also contributing to the community by building a collection of reusable software modules. These modules are designed to be free, well-documented, modern, and interoperable, allowing the open source community to easily incorporate them into their own Sudoku-related projects.

### This frontend module is the primary user interface for Sudokuru, providing a cross-platform Sudoku experience built with React Native Web and TypeScript. It offers a clean, intuitive design with features such as:

- 🎮 Play Sudoku across 9 different difficulty levels generated by the Sudokuru Clearinghouse module
- 🎓 Learn how to play Sudoku with lessons from the basics all the way to advanced strategies
- 📊 Statistics to track your progress
- ⚙️ Sensible default settings for casual players with options to customize the playing experience for users with different playstyles
- 💡Custom strategy based hints from the Sudokuru npm library module
- 📅 _Upcoming_: drills which let you practice individual strategies also powered by the Sudokuru npm library module

# 🖥️ Supported Platforms

## Tier 1 Support

### 🚢 Web

## Tier 2 Support

### ⚒️ Android

### ⚒️ Windows

### ⚒️ Linux

## Planned Support

### 📅 iOS

### 📅 MacOS

## Legend

### 🚢 Tier 1 Support - We have full end to end test coverage for all new and existing functionality. We do manual testing of new features before deployment.

### ⚒️ Tier 2 Support - We have partial or zero end to end test coverage for new and existing functionality. We may not always do manual testing of new features before deployment.

### 📅 Planned Support - We do not currently support this platform, but we plan on supporting in the future.

# 🛠️ Local development

## ⚙️ General Setup:

1. Git clone this repository
2. Install Node.js `v19.5.0` or later
3. Install npm `9.3.1` or later
4. Run `npm i` in the root folder
5. [Install pre-commit hooks](https://pre-commit.com/#install)

6. Run `pre-commit install` to setup pre-commit hooks. Pre-commit hooks can be run manually with `npm run pre-commit`, but will always run before git commit and git push if setup correctly.

## 📲 Mobile Setup (not needed to run website):

1. Contact Thomas to get added to the Expo organization so that you can log in with your own email and password.
2. Download the expo app on your mobile device.
3. On your development device, e.g., laptop, login by running: `expo login` and fill out the parameters.

## 🏃‍♂️ Running the Application Locally

1. Run `npm run start` from this repositories root folder
2. For iOS, scan the QR code with your camera app, for Android you will need to scan the code from within the Expo app.
3. For Web, hit the w key to start up the website at `localhost:8081`

## 🧭 Navigating the Codebase

<details>
<summary>app</summary>

📂 sudokuru/app/ is the parent folder for all the application code

- 📡 Api/ contains classes to make it easier to interact with LocalStorage state data

- 🖼️ ️Components/ contains the React components displayed on the pages
  - Contains folders to store components for various pages
  - Due to its size and complexity we'll break down SudokuBoard directory further.
    - In addition to the SudokuBoard.tsx component, SudokuBoard contains a folder of subcomponents /Components, a folder of helper functions /Functions, and a sudoku.ts file with even more helper functions.
- ⚛️ Contexts/ contains the React contexts code
  - TODO: explain this folder better
- 🔢 Data/ contains files filled with Sudoku puzzles to serve to the user
- 🔨 Functions/ contains files of helper functions
- 🧭 Navigation/ contains code for Drawer navigation
- 📄 Pages/ contains the applications pages
  - AboutUsPage.tsx lets users learn more about the Sudokuru project
    - <img src="docs/PageScreenshots/AboutUs.png" alt="AboutUsPage Screenshot" width=400>
  - ContactPage.tsx lets users provide us with feedback from inside the app
    - <img src="docs/PageScreenshots/Contact.png" alt="ContactPage Screenshot" width=400>
  - DrillGame.tsx is not currently available to users
  - DrillPage.tsx is not currently available to users
  - HomePage.tsx is the users entry into the app
    - <img src="docs/PageScreenshots/Home.png" alt="HomePage Screenshot" width=400>
  - LearnPage.tsx lets users select lessons to start
    - <img src="docs/PageScreenshots/Learn.png" alt="LearnPage Screenshot" width=400>
  - Lesson.tsx lets users complete lessons
    - <img src="docs/PageScreenshots/Lesson.png" alt="Lesson Screenshot" width=400>
  - PlayPage.tsx lets users select puzzle to play based on difficulty
    - <img src="docs/PageScreenshots/Play.png" alt="Play Screenshot" width=400>
  - ProfilePage.tsx lets users select preferences e.g. theme
    - <img src="docs/PageScreenshots/Profile.png" alt="Profile Screenshot" width=400>
  - ReleaseNotesPage.tsx lets users see what's added to the app with each version update
    - <img src="docs/PageScreenshots/ReleaseNotes.png" alt="ReleaseNotes Screenshot" width=400>
  - StatisticsPage.tsx lets users see their Sudoku playing statistics
    - <img src="docs/PageScreenshots/Statistics.png" alt="Statistics Screenshot" width=400>
  - SudokuPage.tsx lets users play Sudoku!
    - <img src="docs/PageScreenshots/Sudoku.png" alt="Sudoku Screenshot" width=400>
- 🎨 Styling/ contains code for theme and highlighting colors
    - All application colors are defined in `sudokuru/app/Styling/theme.ts`

</details>

<details>
<summary>docs</summary>

📂 docs/ is the parent folder for all the documentation

- 📡 BackendApiCalls/ contains docs to help understand how to make API calls to Sudokuru npm library
- 🖼️ Components/ contains various docs related to Sudoku logic, terminology, and highlighting
- 📜 DesignDecisions/ contains docs explaining some historical design decisions made in the apps development
- 📸 PageScreenshots/ contains screenshots of the apps pages used as reference in this README

</details>

<details>
<summary>e2e/web</summary>

- 🖼️ components/ contains testing components to help assert things about components including the SudokuBoard itself
- 📄 page/ contains testing components to help assert things about pages
- 🎭 specs/ contains the actual Playwright e2e tests
- 🔢 data.ts contains test Sudoku game objects
- 🏗️ fixture.ts contains test fixtures including to automatically get things setup like navigating to a page to be tested

</details>

## 🎭 Playwright E2E Tests

### ⚙️ Setup

2. Run `npm run playwright:init` to install playwright dependencies
3. Inside of e2e/web folder, create a `.env` file in for local development based on values in `.env.example`

### 🏃‍♂️ Running the Tests

- ⚠️ Make sure that the website is running locally (or change baseURL to match where you want to test)
- 💻 Run `npm run playwright:ui` to run tests using playwright ui
- ⌨️ Run `npm run playwright:test` to run tests using playwright cli
- 📋 Run `npm run playwright:report` to view playwright report

### 🔧 Setup to use a Single Puzzle for Debugging

- In the `sudokuru/app/Api/Puzzles.ts` file, the `startGame` function can be modified so that only a single game is used.
- Replace `returnGameOfDifficulty(difficulty)` with `returnGameOfDifficulty("dev")` and the dev puzzle will be retrieved.
- The `returnGameOfDifficulty` function can also be modified to return a desired puzzle. By default, it returns the first `novice` puzzle.

## 🛡️🐶 Run Snyk scans

The command to run a Snyk Open Source scan is `npm run snyk:opensource`

The command to run a Snyk code scan is `npm run snyk:code`

Existing issues in the main branch can be [viewed here](https://app.snyk.io/org/sudokuru)

## 📦 Analyzing Bundle size with Atlas

To see the dev bundle size, run `npm run start:atlas`

To see the production bundle size, run `npm run start:atlas:prod`

Note - I was not able to get the `--no-dev` flag working with android, but it works for web and ios.

Connect to expo with the desired device to see the bundle size and package ratios.

You will need to navigate to the following url: http://localhost:8081/\_expo/atlas

It is possible to switch between multiple bundle types in the atlas UI for comparison.

iOS Example:  
![alt text](docs/img/atlas-ios.png)

Android Example:  
![alt text](docs/img/atlas-android.png)

Web Example:  
![alt text](docs/img/atlas-web.png)

# Tips for updating Expo version

1. Read through the Expo release blog post.
2. When updating dependencies, ensure that there are no nested `node_modules` folders - there should only be one `node_module` folder inside of root. Try running commands `npm dedupe` and `npm list <package_name>` to help debug issues, as a likely cause of multiple `node_modules` folders generating is multiple versions of the same package. 

# 🛣️ Roadmap

## Github Project Kanban Boards

We organize our future work using the [Frontend Github Project Board](https://github.com/orgs/Sudokuru/projects/4/views/1) which we currently have divided into the following columns. We also apply labels which are [documented here](https://github.com/Sudokuru/Frontend/labels) and work towards milestones [listed here](https://github.com/Sudokuru/Frontend/milestones).

### 💡 Backlog

This is the general bucket for everything that we are considering doing someday but does not fit into any of the following boards.

### ⚙️ In development

This is for all the tickets we are currently working. Requires an acceptance criteria to be added to the description at this stage.

### 🚢 Shipped

This is for all the cool things we have already accomplished!
 readmeEtag: '"850d44c2ebbd2cf654bdb74f8618fa0ca0fe4e8a"' readmeLastModified: Fri, 07 Nov 2025 02:10:59 GMT repositoryId: 559579865 description: null created: '2022-10-30T14:59:36Z' updated: '2026-01-23T01:25:02Z' language: TypeScript archived: false stars: 3 watchers: 0 forks: 1 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: AGPL-3.0 repoEtag: '"cbf3caccca4f77e2205020940b1bf544053b721a66496b8f5db54f37301160de"' repoLastModified: Fri, 23 Jan 2026 01:25:02 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/seandavi/cmgd_web v3: true repositoryMetadata: repositoryId: 296348041 description: Web portal and API for the Curated Metagenomics Data project created: '2020-09-17T14:16:44Z' updated: '2024-07-12T06:24:26Z' language: Python archived: false stars: 0 watchers: 2 forks: 0 owner: seandavi logo: https://avatars.githubusercontent.com/u/92435?v=4 repoEtag: '"90aebd6e3188392b7718c63e322d346f844e02f96f796d0234354739da1ba7fb"' repoLastModified: Fri, 12 Jul 2024 06:24:26 GMT foundInMaster: true id: fb4fed9b0a7ae53a122c03126af2f4d0 - source: openapi3 tags repository: https://github.com/aqib1/currencyconvertor v3: true repositoryMetadata: base64Readme: >- IyBDdXJyZW5jeUNvbnZlcnRvcgoKIyBUYXNrCkxldCdzIHdhbGsgdGhyb3VnaCB0aGlzIHRhc2s6IAoxLiBDcmVhdGUgYSBuZXcgSmF2YSAvIEtvdGxpbiBNaWNyby1TZXJ2aWNlIEFwcGxpY2F0aW9uLCB1dGlsaXppbmcgU3ByaW5nIEJvb3QuIAoyLiBTZXJ2ZSB0aGUgZm9sbG93aW5nIEFQSSBlbmRwb2ludHM6IAogICBPdXRwdXQgdGhlIGN1cnJlbnQgdGltZSAoYXMgdmFsaWQgSlNPTikgQ29udmVydCBjdXJyZW5jeSBpbnB1dHMgZW5kcG9pbnQ6ICAvYXBpL2N1cnJlbmN5L2NvbnZlcnQgCiAgIGlucHV0czoKICAgICAgICBhbW91bnQgKGluIHNvdXJjZS1jdXJyZW5jeSksIHNvdXJjZS1jdXJyZW5jeSwgdGFyZ2V0LWN1cnJlbmN5IAogICBvdXRwdXQ6IAogICAgICAgIGFtb3VudCBpbiB0YXJnZXQtY3VycmVuY3kgKGFzIHZhbGlkIEpTT04pIHlvdSBtYXkgdXNlIHN0YXRpYyBjb252ZXJzaW9uIGRhdGEgb3IgY29ubmVjdCB0byBhIHB1YmxpYyBjb252ZXJ0ZXIgKGUuZy4gaHR0cHM6Ly9jdXJyZW5jeWxheWVyLmNvbS9kb2N1bWVudGF0aW9uKSBWYWxpZGF0ZSBhIFZBVCBpbnB1dCBhbmQgcmV0dXJuIGlucHV0OiBwb3RlbnRpYWwgVkFUIG51bWJlciBvdXRwdXQ6IGNvdW50cnkgKGNvZGUpIGZvciBWQVQgbnVtYmVyIChhcyB2YWxpZCBKU09OKSB5b3UgbWF5IHVzZSBhIHN0YXRpYyBtYXBwaW5nIG9yIGNvbm5lY3QgdG8gYSBwdWJsaWMgYXBpIChlLmcuIGh0dHBzOi8vYXBpLmNsb3VkbWVyc2l2ZS5jb20vc3dhZ2dlci9pbmRleC5odG1sPyB1cmxzLnByaW1hcnlOYW1lPVZhbGlkYXRlJTIwQVBJKSAKMy4gQWRkIHVuaXQtdGVzdCBjb3ZlcmFnZSBhcyBhcHByb3ByaWF0ZS4gCjQuIEVuc3VyZSB0aGUgc2VydmljZSBjYW4gYmUgYnVpbHQsIHRlc3RlZCwgYW5kIHJ1bi4gVGhlIHNlcnZpY2Ugc2hvdWxkIHJ1biBvbiBwb3J0IDgwODEgCgpCT05VUzogCmluY29ycG9yYXRlIGFuIGVtYmVkZGVkIERCL2NhY2hpbmcgbGF5ZXIgZm9yIHlvdXIgQVBJIGVuZHBvaW50cyAKQk9OVVM6IApUaGUgc2VydmljZSBpbmNsdWRlcyBhdXRvZ2VuZXJhdGVkIHN3YWdnZXIgZG9jdW1lbnRhdGlvbi4gCgo1LiBBZGQgYSBSRUFETUUgdG86IGRlc2NyaWJlIHRoZSBhcHBsaWNhdGlvbiwgdGhlIHByZS1yZXF1aXNpdGVzLCBob3cgdG8gcnVuIGl0LCBhZGRpdGlvbmFsIG5vdGVzLCBldGMuIAo2LiBQYWNrYWdlIHRoZSBhcHAgYW5kIHNoYXJlIGEgZG93bmxvYWQgbGluayB0byB0aGUgc291cmNlLWNvZGUgb3IgcHJpdmF0ZSBHaXRIdWIgcmVwb3NpdG9yeS4gCgpXQVJOSU5HOiBETyBOT1QgSU5DTFVERSBUSEUgQlVJTEQvT1VUUFVULiBXZSBleHBlY3QgdG8gYmUgYWJsZSB0byBidWlsZCwgdGVzdCwgYW5kIHJ1biBpdCBvdXJzZWx2ZXMuIFBsZWFzZSBkbyBub3QgbWFrZSB0aGUgc291cmNlLWNvZGUgcHVibGljLiAKQk9OVVM6IGRvY2tlcml6ZSB5b3VyIGFwcGxpY2F0aW9uIChpLmUuIGluY2x1ZGUgYSBEb2NrZXLvrIFsZSwgZG9ja2VyLWNvbXBvc2UueW1sLCBhbmQgcmVsZXZhbnQgUkVBRE1FIGluc3RydWN0aW9ucykKCiMgU29sdXRpb24KRm9yIGltcGxlbWVudGF0aW9uIGFzIE1pY3JvLXNlcnZpY2UgYXJjaGl0ZWN0dXJlLCBTcHJpbmcgY2xvdWQgdGVjaG5vbG9neSBpcyB1c2VkLCBOZXRmbGl4IEV1cmVrYSBzZXJ2ZXIgaXMgdXNlZCBmb3Igc2VydmljZSByZWdpc3RyeS4gCkZvciB0aGUgc2FrZSBvZiB0aGlzIHRhc2ssIGEgc2luZ2xlIG1pY3JvLXNlcnZpY2UgYXMgdGhlIGV1cmVrYSBjbGllbnQgaXMgY3JlYXRlZCBhbG9uZyB3aXRoIGEgZXVyZWthIHNlcnZlci4KVGhlIGNvbXBsZXRlIGFwcGxpY2F0aW9uIGlzIGNyZWF0ZWQgdXNpbmcgdGhlIHRlc3QtZHJpdmVuIGFwcHJvYWNoLiBUaGF0J3MgaXMgdGhlIHJlYXNvbiBhcHBsaWNhdGlvbiBjb250YWlucyBhIGxhcmdlIG51bWJlciBvZiB0ZXN0IGNhc2VzLgpJbiBuZXRmbGl4LWNsaWVudCwgdHdvIGNvbnRyb2xsZXJzIGFyZSB3cml0dGVuLCB3aGljaCBkZXRhaWxzIG9uIGFic3RyYWN0aW9uIGxldmVsIGlzIGdpdmVuIGhlcmVpbmFmdGVyLiBBUEkgaXMgY29uZmlndXJlZCBvbiBwb3J0IDgwODEKIC0gQ3VycmVuY3lDb252ZXJ0ZXJDb250cm9sbGVyCiAtIFZBVENvbnRyb2xsZXIKIApDdXJyZW5jeUNvbnZlcnRlckNvbnRyb2xsZXIgaXMgYSByZXN0IGNvbnRyb2xsZXIgd2hpY2ggZnVydGhlciBjb25zaXN0cyBvZiB0d28gdHlwZXMgb2YgQVBJLCBrZWVwIGluIG1pbmQgdGhhdCBlYWNoIHJlc3QgQVBJIGlzIGNyZWF0ZWQgd2hpbGUga2VlcGluZyBpbiBtaW5kIG9mIHRoZSAybmQtZGVncmVlIG1hdHVyaXR5IGxldmVsIG9mICBSZXN0IEFQSSBsZXZlbCBkZWZpbmVkIGJ5IFJpY2hhcmRzb24sIEluIGZ1dHVyZSBjb21taXRzLCBBUEkgd2lsbCBiZSB1cGdyYWRlZCB0byAzcmQgZGVncmVlIHVzaW5nIEhBVEVPQVMuCgoxLSBBUEkgZG9jdW1lbnRhdGlvbnMgYW5kIGRldGFpbHMgYXJlIGNyZWF0ZWQgdXNpbmcgc3dhZ2dlciAzLjAsIGNvbW1vbiBvYmplY3RzIGFyZSB1c2luZyBjcmVhdGVkIGJ5IHN3YWdnZXIgMy4wLiBTd2FnZ2VyIGNvbmZpZ3VyYXRpb24gZmlsZSBpcyBwbGFjZWQgdW5kZXIgcmVzb3VyY2VzIHdpdGggdGhlIG5hbW1lIG9mIGFwaS55YW1sCgoyLSBMb2dnaW5nIGFsc28gaW1wbGVtZW50YXRlZCB3aXRoIHJvbGxpbmcgZmlsZSBwb2xpY3kuIGl0cyBjb25maWd1cmF0aW9uIGlzIGFsc28gYXZhaWxhYmxlIHVuZGVyIHJlc291cmNlIGZvbGRlci4KCjMtIEtlZXAgaW4gbWluZCB0aGF0IHlvdSBhcmUgbm90IGFibGUgdG8gcnVuIGFwcGxpY2F0aW9uIHVudGlsIHlvdSBkaWQgbm90IHN0YXJ0IHRoZSBldXJla2Egc2VydmVyLCBhcyB0aGUgY2xpZW50IG5lZWRzIHRvIHJlZ2lzdGVyIGl0c2VsZiB3aXRoIHRoZSBzZXJ2ZXIgKGNvbmZpZ3VyYXRpb24gcHJvdmlkZWQgaW4gdGhlIGFwcGxpY2F0aW9uKS4KCjQtIEZvciBjdXJyZW5jeSBjb252ZXJzaW9uIGFuZCB2YXQgdmFsaWRhdGlvbiB0d28gZGlmZmVyZW50IGZyZWUgQVBJIHVzZWQsIFRob3NlIEFQSSBhcmUgY29ubmVjdGVkIHVzaW5nIE5ldGZsaXgtRmVpZ24gZGVjbGFyYXRpdmUgSFRUUCBjbGllbnQgd2hpY2ggaXMgYWxzbyB0aGUgcGFydCBvZiBTcHJpbmcgY2xvdWQgdGVjaG5vbG9neS4KCjUtIEZvciBjYWNoZSBIMi1EYXRhYmFzZSB1c2VkIGFzIGluIG1lbW9yeSBkYXRhYmFzZS4gVG8gYXZvaWQgYW55IGRpcnR5IHJlYWRzL3dyaXRlLCBwaGFudG9tIHJlYWRzLCByZXBlYXRlZCByZWFkcywgSmF2YSBSZWFkV3JpdGVSZWVudGVyZW50TG9jayBpcyB1c2VkLgoKNi0gQ29udHJvbGxlciBhZHZpY2UgaXMgd3JpdHRlbiB0byBoYW5kbGUgZXhjZXB0aW9uIGFuZCBzZW5kIHJlc3BlY3RpdmUgcmVzcG9uc2UuCgo3LSBTcHJpbmcgQU9QIGlzIHVzZWQgdG8gbG9nZ2luZyBlYWNoIG1ldGhvZCBlbnRlcmVuY2UgYW5kIGV4aXQuCgo4LSBTcHJpbmcgc2NoZWR1bGVyIGlzIHdyaXR0ZXIgdG8gc2NoZWR1bGUgY2FjaGUgcmVmcmVzaGVyIHdpdGggYSBzcGVjaWZpYyB0aW1lIGludGVydmFsLgoKOS0gU3ByaW5nIGJvb3QgZG9ja2VyaXphdGlvbiBpcyBhZGRlZCB1c2luZyBzcG90aWZ5IG1hdmVuIHBsdWdpbiBmb3IgZG9ja2VyLCBieSBydW5uaW5nIGNvbW1hbmQgbXZuIHBhY2thZ2UgZG9ja2VyOmJ1aWxkLgpBZnRlciBjcmVhdGluZyBpbWFnZSB5b3UgY2FuIHNlZSBpbWFnZSBpbiB5b3VyIGRvY2tlciBieSBkb2NrZXIgaW1hZ2UgbHMuIE5vdyBsZXRzIGFkZCB0YWcgdG8gdGhhdCBpbWFnZSBieSBjb21tYW5kIGRvY2tlciB0YWcgaW1hZ2VfaWQgVEFHX05BTUUuIEFmdGVyIHRoYXQgeW91IGNhbiBwdXNoIGltYWdlIHRvIHlvdXIgZG9ja2VyIGh1YiBieSBkb2NrZXIgcHVzaCBUQUdfTkFNRQoKIyBEb2NrZXIgaW1hZ2UKQXBwbGljYXRpb24gaXMgY29tcGxldGVseSBkb2NrZXJpemUgYW5kIGltYWdlcyBhcmUgdXBsb2FkZWQgdG8gZG9ja2VyLWh1YiB3aXRoIHRoZSBuYW1lIGFxaWJidXR0MzA3OC9jY3ZhdGFwaSBmb3IgbWljcm8tc2VydmljZSBldXJla2EgY2xpZW50IGFuZCBhcWliYnV0dDMwNzgvY2N2YXRhcGlfZXVyZWthIGZvciBldXJrYSBzZXJ2ZXIKIAojIEN1cnJlbmN5IENvbnZlcnRlcgpDdXJyZW5jeSBDb252ZXJ0ZXIgQVBJIGNvbnNpc3RzIG9uIHR3byBBUEkncyAKLSAvYXBpL2N1cnJlbmN5L2NvbnZlcnQ/c291cmNlPVNPVVJDRV9DT1VOVFJZX0NPREUmdGFyZ2V0PVRBUkdFVF9DT1VOVFJZX0NPREUmYW1vdW50PUFNT1VOVF9OVU1CRVIKLSAvYXBpL2N1cnJlbmN5L2NvbnZlcnQvZGV0YWlscz9zb3VyY2U9U09VUkNFX0NPVU5UUllfQ09ERSZ0YXJnZXQ9VEFSR0VUX0NPVU5UUllfQ09ERSZhbW91bnQ9QU1PVU5UX05VTUJFUgoKRmlyc3QgQVBJIHdpbGwgZ29pbmcgdG8gcmV0dXJuIGNvbnZlcnRpb24gZGV0YWlsZXMgYXMgaXQgd2FzIGFza2VkIGluIHRhc2ssIHNlY29uZCBBUEkgcmV0dXJuIGRldGFpbHMgd2l0aCBzb21lIGFkZGl0aW9uYWwgaW5WQVRfSUQKIyBWQVRDb250cm9sbGVyClZBVENvbnRyb2xsZXIgYWxzbyBwcm92aWRlcyB0d28gQVBJJ3MgCi0gL2FwaS92YXQvdmFsaWRhdGU/dmF0PVZBVF9JRAotIC9hcGkvdmF0L3ZhbGlkYXRlL2RldGFpbHM/dmF0PVZBVF9JRAojIFRlc3RpbmcKRm9yIGVhY2ggY2xhc3MsIG1ldGhvZHMgdW5pdCB0ZXN0cyBhcmUgcHJvdmlkZWQuIEFzIGFwcGxpY2F0aW9uIGlzIGJ1aWxkIHVzaW5nIHJlZC1ncmVlbi1yZWZhY3RvciBzb3J0IG9mIHRlc3QgZHJpdmVuIGFwcHJvYWNoLCBzbyB0aGF0cyBhIHJlYXNvbiBpdCBoYXZlIGFsb3Qgb2YgdGVzdCBjYXNlcy4KSW50ZWdyYXRpb24gdGVzdCBmb3IgUmVzdCBBUEkgaXMgcHJvdmlkZWQgc2VwcmVhdGx5LgojIEFsbCBjdXJyZW5jaWVzIHN1cHBvcnRlZApQbGVhc2UgY2xpY2sgdGhlIHVybCB0byBzZWUgYWxsIGN1cnJlbmNpZXMgbGlzdApodHRwczovL2ZyZWUuY3VycmNvbnYuY29tL2FwaS92Ny9jb3VudHJpZXM/YXBpS2V5PTQ3NTRjOGZhOGMwZDhlMDhjNjU4Cg== readmeEtag: '"ddf9047507cbc3aff90980958bcdfe61d4ab89b9"' readmeLastModified: Thu, 26 Sep 2019 07:33:04 GMT repositoryId: 208506319 description: null created: '2019-09-14T21:31:22Z' updated: '2019-10-17T02:23:17Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: aqib1 logo: https://avatars.githubusercontent.com/u/8742169?v=4 repoEtag: '"397d4edfc614a2fabe9ca637595a69a44404fff9adb0f27efa0eaaf4ba4eb6d2"' repoLastModified: Thu, 17 Oct 2019 02:23:17 GMT foundInMaster: true category: Server Implementations id: b21b7ab96e0c1b7078783685d6213fed - source: openapi3 tags repository: https://github.com/nicesi/openapi-ui-single-file v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFVJCgoqIFNpbmdsZSBmaWxlCiogVWx0cmEgbGlnaHQtd2VpZ2h0CgpDYW4gdXNlIGl0IGluIDMgZGlmZmVyZW50IHdheXMKCjEuIEdvIHRvIGh0dHBzOi8vbmljZXNpLmNvbS9vcGVuYXBpLXVpIGFuZCBlbnRlciBvcGVuYXBpKHN3YWdnZXIpIGZpbGUgdXJsCjIuIGh0dHBzOi8vbmljZXNpLmNvbS9vcGVuYXBpLXVpP3VybD1odHRwczovL3BldHN0b3JlLnN3YWdnZXIuaW8vdjIvc3dhZ2dlci5qc29uCjMuIEVkaXQgdXJsIGZpZWxkIGh0dHBzOi8vZ2l0aHViLmNvbS9uaWNlc2kvb3BlbmFwaS11aS1zaW5nbGUtZmlsZS9ibG9iL21hc3Rlci9pbmRleC5odG1sI0wzOSAuL3N3YWdnZXIuanNvbiAoZmlsZXBhdGgpIE9SIGh0dHBzOi8veW91ci1vcGVuYXBpLWZpbGUuanNvbiAoVVJMKQo= readmeEtag: '"b0c9424a417ed1ff99789241acc650421daac773"' readmeLastModified: Mon, 05 Oct 2020 18:00:23 GMT repositoryId: 291342065 description: null created: '2020-08-29T20:13:23Z' updated: '2020-10-05T18:00:49Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: nicesi logo: https://avatars.githubusercontent.com/u/13571949?v=4 license: MIT repoEtag: '"4c45d3fd6eb567f0e489de1bd0f8b43a157c624d379255abddfb497de53e0018"' repoLastModified: Mon, 05 Oct 2020 18:00:49 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: b790e4afd85397b8d62572064ce78d3a - source: openapi3 tags repository: https://github.com/david-rojo/pvp-manager v3: true repositoryMetadata: base64Readme: >- # PVP Manager

This is a Java 8 project that exposes a REST API documented with OpenAPI, developed with SpringBoot 2.3.1 using H2 database for persistency with an initial load of data. It implements the following [scenario](doc/scenario.md). 

It has been done following the MVC design pattern and using [Spring Tool Suite 4](https://spring.io/tools)

## H2 database

For **primary key** of the **PRICES** table has been chosen three fields: BRAND_ID, PRODUCT_ID and PRICE_LIST. This decision has been done thinking that cannot be a row in the table that has the same values for these fields.

Data types for each field:

| FIELD      | DATA TYPE    |
|------------|--------------|
| BRAND_ID   | INTEGER      |
| PRICE_LIST | INTEGER      |
| PRODUCT_ID | INTEGER      |
| PRIORITY   | INTEGER      |
| START_DATE | TIMESTAMP    |
| END_DATE   | TIMESTAMP    |
| PRICE      | DOUBLE       |
| CURR       | VARCHAR(255) |

## Deployment

Once that the repository has been cloned:

```
$ git clone https://github.com/david-rojo/pvp-manager.git
```

import the project in your IDE and to execute it, right click of the mouse in **spring-boot-pvpmanager** folder in Package Explorer window and select *Run as > Spring Boot App*

The project is deployed and ready to verify that implements the requested scenario. You can access to its exposed API here: [http://localhost:8080/pvp-manager-api.html](http://localhost:8080/pvp-manager-api.html)

![PVP MANAGER OPEN API](doc/img/pvp-manager-openapi.png)

To verify the scenario, you can test in the OpenAPI User Interface, the get method in **pvp** group writing manually all the values or you can simply click in the following links that use the same data that is used in each test:

* **Test1**: has to return priceList 1
  * input data: ```brandId = 1, productId=35455, date=2020-06-14-10.00.00```
  * URL: [http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-14-10.00.00](http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-14-10.00.00)
  * output data: 
  ```{"priceList":1,"brandId":1,"productId":35455,"startDate":"2020-06-14T00:00:00","endDate":"2020-12-31T23:59:59","price":35.5,"currency":"EUR"}```

* **Test2**:  has to return priceList 2
  * input data: ```brandId = 1, productId=35455, date=2020-06-14-16.00.00```
  * URL: [http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-14-16.00.00](http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-14-16.00.00)
  * output data: 
  ```{"priceList":2,"brandId":1,"productId":35455,"startDate":"2020-06-14T15:00:00","endDate":"2020-06-14T18:30:00","price":25.45,"currency":"EUR"}```
  
* **Test3**:  has to return priceList 1
  * input data: ```brandId = 1, productId=35455, date=2020-06-14-21.00.00```
  * URL: [http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-14-21.00.00](http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-14-21.00.00)
  * output data: 
  ```{"priceList":1,"brandId":1,"productId":35455,"startDate":"2020-06-14T00:00:00","endDate":"2020-12-31T23:59:59","price":35.5,"currency":"EUR"}```
  
* **Test4**:  has to return priceList 3
  * input data: ```brandId = 1, productId=35455, date=2020-06-15-10.00.00```
  * URL: [http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-15-10.00.00](http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-15-10.00.00)
  * output data: 
  ```{"priceList":3,"brandId":1,"productId":35455,"startDate":"2020-06-15T00:00:00","endDate":"2020-06-15T11:00:00","price":30.5,"currency":"EUR"}```
  
* **Test5**: has to return priceList 4
  * input data: ```brandId = 1, productId=35455, date=2020-06-16-21.00.00```
  * URL: [http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-16-21.00.00](http://localhost:8080/pvp-manager/pvp/1/35455/2020-06-16-21.00.00)
  * output data: 
  ```{"priceList":4,"brandId":1,"productId":35455,"startDate":"2020-06-15T16:00:00","endDate":"2020-12-31T23:59:59","price":38.95,"currency":"EUR"}```

Implementation of the controller can be examinated in [PvpApiController](spring-boot-pvpmanager/src/main/java/com/pvpmanager/springboot/app/controller/PvpApiController.java) class

### Additional work

Going beyond the scope of the scenario, a full CRUD set of operations has been implemented for the application in order to manage **PRICES** table:

- Add a new price.
- Update an existing price.
- Delete an existing price by ID.
- Get all prices in the table.

Implemention can be examinated in [PriceApiController](spring-boot-pvpmanager/src/main/java/com/pvpmanager/springboot/app/controller/PriceApiController.java) class

## Access to h2 console

When the project is deployed, also is available h2 console: [http://localhost:8080/h2-console](http://localhost:8080/h2-console) in order to connect there and verify that the database initially has the requested content. 

![H2 CONSOLE](doc/img/h2-console-login.png)

This data can be used to login:
```
Driver Class:	org.h2.Driver
JDBC URL:	jdbc:h2:mem:pvpmanagerdb;DB_CLOSE_ON_EXIT=FALSE
User Name:	sa
Password:	password
```
Once we are logged, we can check the content of **PRICES** table executing this query:

```
SELECT * FROM PRICES 
```

![PRICES TABLE CONTENT](doc/img/h2-table-content.png)

## Executing JUnit tests

To run the requested tests we can do it executing this command from **spring-boot-pvpmanager** folder:

```
$ mvn test 
```
> **Note:** mvn has to be installed in order to execute the command

Near of the end of the execution log, we can find this line:

```
[INFO] Results:
[INFO] 
[INFO] Tests run: 12, Failures: 0, Errors: 0, Skipped: 0
[INFO] 
```
Detail of each test can be found previously of this text above.

Five tests has been requested in the scenario, testing the rest endpoint, but the project contains also additional tests to check that select operations to database works fine.

JUnitTests classes:

* [PvpApiControllerIntegrationTests](spring-boot-pvpmanager/src/test/java/com/pvpmanager/springboot/app/PvpApiControllerIntegrationTests.java): 
   * test the rest endpoint. 
   * using ```MockMvc```
   * **required** tests.

* [IPriceDaoTests](spring-boot-pvpmanager/src/test/java/com/pvpmanager/springboot/app/IPriceDaoTests.java): 
   * test the select operations to database. 
   * using ```@DataJpaTest```
   * **additional** tests

 readmeEtag: '"158bc5f66afe1158aab583c50c4b6ab8d6ca5b9f"' readmeLastModified: Fri, 14 May 2021 23:47:29 GMT repositoryId: 274145786 description: >- Spring Boot project that exposes a REST API documented with OpenAPI, using H2 database for deployment and testing. created: '2020-06-22T13:32:52Z' updated: '2021-05-14T23:47:32Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: david-rojo logo: https://avatars.githubusercontent.com/u/24654236?v=4 license: MIT repoEtag: '"ce1b894d537ec692debd57130f27211594b7539989aac38eb2ccff889545210d"' repoLastModified: Fri, 14 May 2021 23:47:32 GMT foundInMaster: true category: Server Implementations id: 14028bea2f0274f45664e49ab5ab1f5b - source: openapi3 tags repository: https://github.com/aiveraiva/api-documents v3: true id: d8f6d3f5ec884958073766300cb69d0d repositoryMetadata: base64Readme: >- IyBpbSBsYXp5IHRvIGZpbmlzaCB0aGlzIGJ1dCBpIHdpbGwKCmBgYHlhbWwKdHlwZTogb2JqZWN0CnByb3BlcnRpZXM6CiAgaWQ6IAogICAgZGVzY3JpcHRpb246IFRoZSBjdXN0b21lciBpZGVudGlmaWVyIHN0cmluZwogICAgcmVhZE9ubHk6IHRydWUKICAgIGFsbE9mOgogICAgICAtICRyZWY6IC4vUmVzb3VyY2VJZC55YW1sCiAgd2Vic2l0ZUlkOgogICAgZGVzY3JpcHRpb246IFRoZSB3ZWJzaXRlJ3MgSUQKICAgIGFsbE9mOgogICAgICAtICRyZWY6IC4vUmVzb3VyY2VJZC55YW1sCiAgcGF5bWVudFRva2VuOgogICAgdHlwZTogc3RyaW5nCiAgICB3cml0ZU9ubHk6IHRydWUKICAgIGRlc2NyaXB0aW9uOiB8CiAgICAgIEEgd3JpdGUtb25seSBwYXltZW50IHRva2VuOyBpZiBzdXBwbGllZCwgaXQgd2lsbCBiZSBjb252ZXJ0ZWQgaW50byBhCiAgICAgIHBheW1lbnQgaW5zdHJ1bWVudCBhbmQgYmUgc2V0IGFzIHRoZSBgZGVmYXVsdFBheW1lbnRJbnN0cnVtZW50YC4gVGhlCiAgICAgIHZhbHVlIG9mIHRoaXMgcHJvcGVydHkgd2lsbCBvdmVycmlkZSB0aGUgYGRlZmF1bHRQYXltZW50SW5zdHJ1bWVudGAKICAgICAgaW4gdGhlIGNhc2UgdGhhdCBib3RoIGFyZSBzdXBwbGllZC4gVGhlIHRva2VuIG1heSBvbmx5IGJlIHVzZWQgb25jZQogICAgICBiZWZvcmUgaXQgaXMgZXhwaXJlZC4KICBkZWZhdWx0UGF5bWVudEluc3RydW1lbnQ6CiAgICAkcmVmOiAuL1BheW1lbnRJbnN0cnVtZW50LnlhbWwKICBjcmVhdGVkVGltZToKICAgIGRlc2NyaXB0aW9uOiBUaGUgY3VzdG9tZXIgY3JlYXRlZCB0aW1lCiAgICBhbGxPZjoKICAgICAgLSAkcmVmOiAuL1NlcnZlclRpbWVzdGFtcC55YW1sCiAgdXBkYXRlZFRpbWU6CiAgICBkZXNjcmlwdGlvbjogVGhlIGN1c3RvbWVyIHVwZGF0ZWQgdGltZQogICAgYWxsT2Y6CiAgICAgIC0gJHJlZjogLi9TZXJ2ZXJUaW1lc3RhbXAueWFtbAogIHRhZ3M6CiAgICBkZXNjcmlwdGlvbjogQSBsaXN0IG9mIGN1c3RvbWVyJ3MgdGFncwogICAgcmVhZE9ubHk6IHRydWUKICAgIHR5cGU6IGFycmF5CiAgICBpdGVtczoKICAgICAgJHJlZjogLi9UYWdzL1RhZy55YW1sCiAgcmV2aXNpb246CiAgICBkZXNjcmlwdGlvbjogPgogICAgICBUaGUgbnVtYmVyIG9mIHRpbWVzIHRoZSBjdXN0b21lciBkYXRhIGhhcyBiZWVuIG1vZGlmaWVkLgoKICAgICAgVGhlIHJldmlzaW9uIGlzIHVzZWZ1bCB3aGVuIGFuYWx5emluZyB3ZWJob29rIGRhdGEgdG8gZGV0ZXJtaW5lIGlmIHRoZQogICAgICBjaGFuZ2UgdGFrZXMgcHJlY2VkZW5jZSBvdmVyIHRoZSBjdXJyZW50IHJlcHJlc2VudGF0aW9uLgogICAgdHlwZTogaW50ZWdlcgogICAgcmVhZE9ubHk6IHRydWUKICBfbGlua3M6CiAgICB0eXBlOiBhcnJheQogICAgZGVzY3JpcHRpb246IFRoZSBsaW5rcyByZWxhdGVkIHRvIHJlc291cmNlCiAgICByZWFkT25seTogdHJ1ZQogICAgbWluSXRlbXM6IDMKICAgIGl0ZW1zOgogICAgICBhbnlPZjoKICAgICAgICAtICRyZWY6IC4vTGlua3MvU2VsZkxpbmsueWFtbAogICAgICAgIC0gJHJlZjogLi9MaW5rcy9Ob3Rlc0xpbmsueWFtbAogICAgICAgIC0gJHJlZjogLi9MaW5rcy9EZWZhdWx0UGF5bWVudEluc3RydW1lbnRMaW5rLnlhbWwKICAgICAgICAtICRyZWY6IC4vTGlua3MvTGVhZFNvdXJjZUxpbmsueWFtbAogICAgICAgIC0gJHJlZjogLi9MaW5rcy9XZWJzaXRlTGluay55YW1sCiAgX2VtYmVkZGVkOgogICAgdHlwZTogYXJyYXkKICAgIGRlc2NyaXB0aW9uOiA+LQogICAgICBBbnkgZW1iZWRkZWQgb2JqZWN0cyBhdmFpbGFibGUgdGhhdCBhcmUgcmVxdWVzdGVkIGJ5IHRoZSBgZXhwYW5kYAogICAgICBxdWVyeXN0cmluZyBwYXJhbWV0ZXIuCiAgICByZWFkT25seTogdHJ1ZQogICAgbWluSXRlbXM6IDEKICAgIGl0ZW1zOgogICAgICBhbnlPZjoKICAgICAgICAtICRyZWY6IC4vRW1iZWRzL0xlYWRTb3VyY2VFbWJlZC55YW1sCgpgYGAKCmBgYHlhbWwKZ2V0OgogIHRhZ3M6CiAgICAtIEN1c3RvbWVycwogIHN1bW1hcnk6IFJldHJpZXZlIGEgbGlzdCBvZiBjdXN0b21lcnMKICBvcGVyYXRpb25JZDogR2V0Q3VzdG9tZXJDb2xsZWN0aW9uCiAgZGVzY3JpcHRpb246IHwKICAgIFlvdSBjYW4gaGF2ZSBhIG1hcmtkb3duIGRlc2NyaXB0aW9uIGhlcmUuCiAgcGFyYW1ldGVyczoKICAgIC0gJHJlZjogLi4vY29tcG9uZW50cy9wYXJhbWV0ZXJzL2NvbGxlY3Rpb25MaW1pdC55YW1sCiAgICAtICRyZWY6IC4uL2NvbXBvbmVudHMvcGFyYW1ldGVycy9jb2xsZWN0aW9uT2Zmc2V0LnlhbWwKICAgIC0gJHJlZjogLi4vY29tcG9uZW50cy9wYXJhbWV0ZXJzL2NvbGxlY3Rpb25GaWx0ZXIueWFtbAogICAgLSAkcmVmOiAuLi9jb21wb25lbnRzL3BhcmFtZXRlcnMvY29sbGVjdGlvblF1ZXJ5LnlhbWwKICAgIC0gJHJlZjogLi4vY29tcG9uZW50cy9wYXJhbWV0ZXJzL2NvbGxlY3Rpb25FeHBhbmQueWFtbAogICAgLSAkcmVmOiAuLi9jb21wb25lbnRzL3BhcmFtZXRlcnMvY29sbGVjdGlvbkZpZWxkcy55YW1sCiAgcmVzcG9uc2VzOgogICAgJzIwMCc6CiAgICAgIGRlc2NyaXB0aW9uOiBBIGxpc3Qgb2YgQ3VzdG9tZXJzIHdhcyByZXRyaWV2ZWQgc3VjY2Vzc2Z1bGx5CiAgICAgIGhlYWRlcnM6CiAgICAgICAgUmF0ZS1MaW1pdC1MaW1pdDoKICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvaGVhZGVycy9SYXRlLUxpbWl0LUxpbWl0LnlhbWwKICAgICAgICBSYXRlLUxpbWl0LVJlbWFpbmluZzoKICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvaGVhZGVycy9SYXRlLUxpbWl0LVJlbWFpbmluZy55YW1sCiAgICAgICAgUmF0ZS1MaW1pdC1SZXNldDoKICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvaGVhZGVycy9SYXRlLUxpbWl0LVJlc2V0LnlhbWwKICAgICAgICBQYWdpbmF0aW9uLVRvdGFsOgogICAgICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9oZWFkZXJzL1BhZ2luYXRpb24tVG90YWwueWFtbAogICAgICAgIFBhZ2luYXRpb24tTGltaXQ6CiAgICAgICAgICAkcmVmOiAuLi9jb21wb25lbnRzL2hlYWRlcnMvUGFnaW5hdGlvbi1MaW1pdC55YW1sCiAgICAgICAgUGFnaW5hdGlvbi1PZmZzZXQ6CiAgICAgICAgICAkcmVmOiAuLi9jb21wb25lbnRzL2hlYWRlcnMvUGFnaW5hdGlvbi1PZmZzZXQueWFtbAogICAgICBjb250ZW50OgogICAgICAgIGFwcGxpY2F0aW9uL2pzb246CiAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgIHR5cGU6IGFycmF5CiAgICAgICAgICAgIGl0ZW1zOgogICAgICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvc2NoZW1hcy9DdXN0b21lci55YW1sCiAgICAgICAgdGV4dC9jc3Y6CiAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgIHR5cGU6IGFycmF5CiAgICAgICAgICAgIGl0ZW1zOgogICAgICAgICAgICAgICRyZWY6IC4uL2NvbXBvbmVudHMvc2NoZW1hcy9DdXN0b21lci55YW1sCiAgICAnNDAxJzoKICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9yZXNwb25zZXMvQWNjZXNzRm9yYmlkZGVuLnlhbWwKICB4LWNvZGUtc2FtcGxlczoKICAgIC0gbGFuZzogUEhQCiAgICAgIHNvdXJjZToKICAgICAgICAkcmVmOiAuLi9jb2RlX3NhbXBsZXMvUEhQL2N1c3RvbWVycy9nZXQucGhwCnBvc3Q6CiAgdGFnczoKICAgIC0gQ3VzdG9tZXJzCiAgc3VtbWFyeTogQ3JlYXRlIGEgY3VzdG9tZXIgKHdpdGhvdXQgYW4gSUQpCiAgb3BlcmF0aW9uSWQ6IFBvc3RDdXN0b21lcgogIGRlc2NyaXB0aW9uOiBBbm90aGVyIG1hcmtkb3duIGRlc2NyaXB0aW9uIGhlcmUuCiAgcmVxdWVzdEJvZHk6CiAgICAkcmVmOiAuLi9jb21wb25lbnRzL3JlcXVlc3RCb2RpZXMvQ3VzdG9tZXIueWFtbAogIHJlc3BvbnNlczoKICAgICcyMDEnOgogICAgICAkcmVmOiAuLi9jb21wb25lbnRzL3Jlc3BvbnNlcy9DdXN0b21lci55YW1sCiAgICAnNDAxJzoKICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9yZXNwb25zZXMvQWNjZXNzRm9yYmlkZGVuLnlhbWwKICAgICc0MDknOgogICAgICAkcmVmOiAuLi9jb21wb25lbnRzL3Jlc3BvbnNlcy9Db25mbGljdC55YW1sCiAgICAnNDIyJzoKICAgICAgJHJlZjogLi4vY29tcG9uZW50cy9yZXNwb25zZXMvSW52YWxpZERhdGFFcnJvci55YW1sCiAgeC1jb2RlLXNhbXBsZXM6CiAgICAtIGxhbmc6IFBIUAogICAgICBzb3VyY2U6CiAgICAgICAgJHJlZjogLi4vY29kZV9zYW1wbGVzL1BIUC9jdXN0b21lcnMvcG9zdC5waHAKYGBgCg== readmeEtag: '"c0a4150d348bc6e71584a55135f8b63c6c1b14ae"' readmeLastModified: Sun, 12 Nov 2023 06:57:19 GMT repositoryId: 704369516 description: The documentation for my api created: '2023-10-13T05:48:34Z' updated: '2024-12-10T11:59:50Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: AiverAiva logo: https://avatars.githubusercontent.com/u/43096905?v=4 license: MIT repoEtag: '"436d368dff34b9a8391be39c8eb4b148503000cb47803ea3f59b6df4adbb7be7"' repoLastModified: Tue, 10 Dec 2024 11:59:50 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/aiveraiva/documents - source: openapi3 tags repository: https://github.com/migoya2020/ecommerce-marketplace-api v3: true repositoryMetadata: base64Readme: >- IyBlY29tbWVyY2UtbWFya2V0cGxhY2UtYXBpCmVjb21tZXJjZSBBUEkgZGVzaWduLiBzdG9wbGlnaHQuaW8KCgojIHNjcmVlbnNob290CgohW2Rlc2lnbl0oZGVzaWduLnBuZykKZWNvbW1lcmNlIEFQSSBkZXNpZ24uCg== readmeEtag: '"98e1710fa6863d080593f518463bc4e005f03fa9"' readmeLastModified: Mon, 25 Nov 2019 07:28:51 GMT repositoryId: 217833221 description: ecommerce API design. stoplight.io created: '2019-10-27T09:39:04Z' updated: '2019-11-25T07:31:17Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: migoya2020 logo: https://avatars.githubusercontent.com/u/23479444?v=4 repoEtag: '"68f60c6a882d040fd60eaea18054ddf145e1ab9da1853bed945e25688f211bff"' repoLastModified: Mon, 25 Nov 2019 07:31:17 GMT foundInMaster: true category: - Documentation - Server Implementations id: 6e08ecdc8e0e5279b1e7510d40e29a51 - source: openapi3 tags repository: https://github.com/mrxz/javalin-openapi-apt v3: true repositoryMetadata: base64Readme: >- IyBqYXZhbGluLW9wZW5hcGktYXB0CkFubm90YXRpb24gcHJvY2Vzc2luZyB0byBnZW5lcmF0ZSBPcGVuQVBJIHNwZWNpZmljYXRpb24gZnJvbSBzb3VyY2UgY29kZSB1c2luZyBbSmF2YWxpbl0oaHR0cHM6Ly9qYXZhbGluLmlvKS4KClRoaXMgYW5ub3RhdGlvbiBwcm9jZXNzb3IgdXNlcyBbSmF2YVBhcnNlcl0oaHR0cHM6Ly9naXRodWIuY29tL2phdmFwYXJzZXIvamF2YXBhcnNlcikgdG8gYW5hbHl6ZSB0aGUgd2F5IGEgcHJvamVjdCB1c2VzIEphdmFsaW4gYW5kIHRvIGdlbmVyYXRlIGEgY29ycmVzcG9uZGluZyBPcGVuQVBJIHYzIGRvY3VtZW50LgpJdCBjYW4gZGVyaXZlIHRoZSBwYXRocyAoaW5jbHVkaW5nIHBhdGggcGFyYW1ldGVycyksIHF1ZXJ5IHBhcmFtZXRlcnMsIHJlcXVlc3QgYm9keSBhbmQgdGhlIHJlc3BvbnNlIGJvZHkgKHdpdGggdGhlIGFwcHJvcGlhdGUgc2NoZW1hKS4KU29tZSBzaWduaWZpY2FudCBsaW1pdGF0aW9ucyBhcHBseSAoc2VlIExpbWl0YXRpb25zIHNlY3Rpb24gYmVsb3cpLiBUaGlzIHByb2plY3QgaXMgYSBwcm9vZiBvZiBjb25jZXB0LCBzbyB1c2FnZSBpbiBwcm9kdWN0aW9uIGlzIF9ub3RfIHJlY29tbWVuZGVkLgoKIyMgTGltaXRhdGlvbnMKKiBUaGUgcHJvY2Vzc29yIG9ubHkgd29ya3Mgd2l0aCBgamF2YWNgLgoqIFRoZSBgQXBpQnVpbGRlcmAgb2YgSmF2YWxpbiBtdXN0IGJlIHVzZWQgdG8gY29uc3RydWN0IHRoZSBwYXRocy4gT25seSBhIHNpbmdsZSBtZXRob2QgZm9yIHRoZSBmdWxsIHJvdXRpbmcgdHJlZSBpcyBzdXBwb3J0ZWQuCiogSGFuZGxlcnMgbXVzdCBiZSBwYXNzZWQgYXMgbWV0aG9kIHJlZmVyZW5jZXMgdG8gdGhlIGBBcGlCdWlsZGVyYCBtZXRob2RzLgoqIFBvbHltb3JwaGljIG1vZGVscyB3aWxsIG9ubHkgaGF2ZSB0aGVpciBiYXNlIGNsYXNzIGluIHRoZSByZXN1bHRpbmcgT3BlbkFQSSBkb2N1bWVudC4KKiBSZXF1aXJlZCBwcm9wZXJ0eSBvZiBwYXJhbWV0ZXJzIG9yIGZpZWxkcyBvbiBhIG1vZGVsIGFyZSBub3Qgc3VwcG9ydGVkLgoqIEhhbmRsZXJzIHRoYXQgY29uZGl0aW9uYWxseSByZXNwb25kIHdpdGggZGlmZmVyZW50IHR5cGVzIG9mIGJvZGllcyBvciByZXF1aXJlcyBkaWZmZXJlbnQgcGFyZW1ldGVycyBhcmVuJ3Qgc3VwcG9ydGVkLgoKVGhlIHByb2Nlc3NvciB0cmllcyB0byBncmFjZWZ1bGx5IGhhbmRsZSB1bnN1cHBvcnRlZCBjb25zdHJ1Y3Rpb25zIGJ5IHByb2R1Y2luZyBhIGRlc2NyaXB0aXZlIGVycm9yLgo= readmeEtag: '"7175c25bae3e489006cb92dfadaced8f7db704fc"' readmeLastModified: Sat, 23 Nov 2019 18:00:00 GMT repositoryId: 219295656 description: >- Annotation processing to generate OpenAPI specification from source code using Javalin created: '2019-11-03T12:13:49Z' updated: '2019-11-23T18:00:15Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: mrxz logo: https://avatars.githubusercontent.com/u/8823461?v=4 license: NOASSERTION repoEtag: '"c600d51287bff11dd3d0e72363faa8c5227bf392fdccb10a9dd4f1922489c07d"' repoLastModified: Sat, 23 Nov 2019 18:00:15 GMT foundInMaster: true category: Parsers id: fa727f7dd55a2b01c344fe34370ad87b - source: openapi3 tags repository: https://github.com/dkirrane/ibm-openapi-validator-docker v3: true repositoryMetadata: base64Readme: >- IyBpYm0tb3BlbmFwaS12YWxpZGF0b3IKCkRvY2tlciBJbWFnZSBmb3IgaHR0cHM6Ly9naXRodWIuY29tL0lCTS9vcGVuYXBpLXZhbGlkYXRvcgo= readmeEtag: '"24161c42814b30fde65f6cbcfe58125e82a6d91d"' readmeLastModified: Mon, 21 Oct 2019 14:43:40 GMT repositoryId: 216578889 description: Docker Image for https://github.com/IBM/openapi-validator created: '2019-10-21T13:45:19Z' updated: '2019-10-21T14:43:46Z' language: Dockerfile archived: false stars: 0 watchers: 1 forks: 2 owner: dkirrane logo: https://avatars.githubusercontent.com/u/1280116?v=4 license: Apache-2.0 repoEtag: '"ae7ad27c0876992a6b6047f271d755ce41835d7d92cfdef2336369a1ef97e0cc"' repoLastModified: Mon, 21 Oct 2019 14:43:46 GMT foundInMaster: true category: Description Validators id: 03eba9bbb724766e0c2b82ed9d1a9b71 - source: openapi3 tags repository: https://github.com/aqib1/subscription-api v3: true repositoryMetadata: base64Readme: >- IyBTdWJzY3JpcHRpb24gU2VydmljZQpUaGUgaW1wbGVtZW50YWNpb24gb2YgU3Vic2NyaXB0aW9uIFNlcnZpY2UgaGFzIGJlZW4gbWFkZSB1c2luZyBhIG1pY3Jvc2VydmljZSBhcHByb2FjaC4gVGhpcyBzZXZpY2UgaXMgbWFkZSBvZiAzIHNlcnZpY2VzLiBUaGUgbWFpbiBzZXZpY2UgaXMgdGhlIFN1YnNjcmlwdGlvbiBtaWNyb3NlcnZpY2Ugd2hpY2ggaXMgdGhlIG9ubHkgdGhhdCBpcyByZWFjaGFibGUgc2luY2UgYSBwdWJsaWMgbmV0d29yay4gVGhpcyBzZXZpY2UgaW50ZXJhY3Qgd2l0aCBvbmUgaW50ZXJuYWwgc2VydmljZSwgRW1haWwgbWljcm9zZXZpY2UsIGFuZCBhIGRhdGFiYXNlIHRvIGNyZWF0ZSBzdWJzY3JpcHRpb25zLiBJZiBhbGwgcHJvY2VzcyBpcyBjb3JyZWN0LCB0aGUgbmV3IHN1YnNjcmlwdGlvbiBpZGVudGlmaWVyIGlzIHJldHVybmVkLiBUaGUgRW1haWwgbWljcm9zZXZpY2UgYXJlIG5vdCByZWFjaGFibGUgaWYgeW91IHVzZSBkb2NrZXIgcHJvZmlsZS4gVGhlIERvY2tlci9hcHAueW1sIGlzIG1ha2UgZm9yIHRoZSBjb211bmljYXRpb24gYW1vbmcgdGhlIHJlc3Qgb2YgdGhlIHNlcnZpY2VzLCBFdXJla2EgU2VydmVyLiBJbiB0aGlzLCB0aGUgRW1haWwgTWljcm9zZXJ2aWNlIGlzIHJlZ2lzdGVyZWQgdG8gYmUgYWNjZXNzaWJsZSBieSB0aGUgU3Vic2NyaXB0aW9uIE1pY3Jvc2VydmljZS4KCkluIHRoaXMgZGV2ZWxvcG1lbnQgaGFzIGJlZW4gdXNlZCB0aGUgZm9sbG93aW5nIHRlY2hub2xvZ2llcyBhbmQgbGlicmFyaWVzOgoKLSBTcHJpbmcgQ2xvdWQKaXQgaGFzIGJlZW4gdXNlZCB0byBkZXZlbG9wIHRoZSBhcmNoaXRlY3R1cmUgb2YgbWljcm9zZXJ2aWNlcy4gSXQgbWFrZSBwb3NzaWJsZSB0aGUgZWFzeSBkZXZlbG9wbWVudCBvZiBhbiBFdXJla2EgU2VydmVyIGFuZCB0aGUgcmVnaXN0cmF0aW9uIG9mIGRpZmZlcmVudCBzZXJ2aWNlcyBvbiBpdC4gVXNlZCBOZXRmbGl4IGZlaWduIGZvciBkZWNsYXJhdGl2ZSBjb21taWNhdGlvbiB3aXRoIG1pY3Jvc2VydmljZXMuIE5ldGZsaXggcmliYm9uIGlzIHVzZWQgZm9yCmxvYWQgYmFsYW5jaW5nLiBOZXRmbGl4IGh5c3RyaXggaXMgdXNlZCBmb3IgY2lyY3VpdCBicmVha2VyLgoKLSBTcHJpbmcgQm9vdCAKaXQgbWFrZXMgaXQgZWFzeSB0byBjcmVhdGUgc3RhbmQtYWxvbmUgYXBwbGljYXRpb24sIHdoaWNoIGNhbiBiZSBleGVjdXRlZCBsaWtlIHdpdGhvdXQgbmVjZXNzaXR5IG9mIGEgYXBwbGljYXRpb24gd2ViIHNlcnZlciBsaWtlIHRvbWNhdC4KCi0gU3ByaW5nIERhdGEgSlBBCnRvIHJlZHVjZSB0aGUgYW1vdW50IG9mIGJvaWxlcnBsYXRlIGNvZGUgcnF1aXJlZCB0byBpbXBsZW1lbnQgZGF0YSBhY2Nlc3MgZm9yIHZhcmlvdXMgcGVyc2lzdGVuY2Ugc3RvcmVzLiBJbiB0aGlzIGNhc2UsIGR1cmluZyB0aGUgZGV2ZWxvcG1lbnQgaGFzIGJlZW4gdXNlZCBIMiBkYXRhYmFzZS4KCi0gU3dhZ2dlcgp0byBkZXZlbG9wIHRoZSBBUEkgZG9jdW1lbnRhdGlvbnMgYW5kIHJlYWwgdGltZSBvYmplY3RzIG9mIGNvbW1vbiBvdmVyIG1pY3Jvc2VydmljZXMuCgotIEgyIGRhdGFiYXNlCml0IGlzIHRoZSBkYXRhYmFzZSB1c2VkIGluIHRoZSBkZXZlbG9wbWVudCBwcm9jY2Vzcy4KCi0gU3ByaW5nIEFPUAppcyB1c2VkIGZvciBsb2dnaW5nIG1ldGhvZCBpbm5lciBvdXRlciBzdGF0ZW1lbnRzCgotIEphdmFNYWlsIAppcyB1c2VkIHRvIHNlbmQgcmVhbCB0aW1lIGVtYWlscwoKLSBKdW5pdCAKaXMgdXNlZCB0byBkbyB1bml0IHRlc3RpbmcgYXMgYXBwbGljYXRpb24gaXMgYnVpbGQgdXNpbmcgdGVzdC1kcml2ZW4gYXBwcm9hY2ggCgotIE1vY2tpdG8gZnJhbWV3b3JrIAppcyB1c2VkIHRvIG1vY2sgb2JqZWN0cywgc2VydmljZXMgaW4gdW5pdCB0ZXN0aW5nCgotIE1hcFN0cnVjdAppcyB1c2VkIHRvIHBlcmZvcm0gaW50ZXItbWFwcGluZyBvZiBvYmplY3RzCgotIE9wZW4gUE9KTwppcyB1c2VkIHRvIHBlcmZvcm0gdW5pdCB0ZXN0aW5nIGFnYWluc3QgamF2YSBvYmplY3RzCgojIEJ1aWxkaW5nIGZvciBkZXZlbG9wbWVudApUbyBidWlsZCB0aGUgU3Vic2NyaXB0aW9uIFNlcnZpY2UgZm9yIGRldmVsb3BtZW50IHlvdSBtdXN0IGJ1aWxkIHRoZSBmb3VyIHNlcnZpY2VzLiBUbyBidWlsZCB0aGUgU3Vic2NyaXB0aW9uIE1pY3Jvc2VydmljZSB5b3UgbXVzdCBiZSBpbiB0aGUgU3Vic2NyaXB0aW9uTWljcm9zZXJ2aWNlIGZvbGRlciBhbmQgcnVuIHRoZSBmb2xsb3dpbmcgY29tbWFuZCwKCm12biBjbGVhbiBwYWNrYWdlIC1QIGRldgpUaGUgcHJvY2Nlc3MgaXMgdGhlIHNhbWUgZm9yIHRoZSBvdGhlciBzZXJ2aWNlcy4KClJ1bm5pbmcgdGhlIFN1YnNjcmlwdGlvbiBTZXJ2aWNlIGZvciBkZXZlbG9tZW50Ckl0IGlzIG5lY2Nlc3NhcnkgcnVuIHRoZSBzZXJ2aWNlcyBmb2xsb3dpbmcgYW4gb3JkZXIsCgotIEV1cmVrYSBTZXJ2ZXIKCkl0IGlzIG5lY2Vzc2FyeSB3YWl0IHRvIEV1cmVrYSBTZXJ2ZXMgYXJlIGNvbXBsZXRseSBkZXBsb3llZCBiZWZvcmUgc3RhdGluZyB0byBydW4gdGhlIHJlc3Qgb2Ygc2VydmljZXMuCmphdmEgLWphciB0YXJnZXQvZXVyZWthLXNlcnZlci0wLjAuMS5qYXIKCi0gRW1haWwgTWljcm9zZXJ2aWNlCgpqYXZhIC1qYXIgdGFyZ2V0L2FkaWRhcy1lbWFpbC1hcGktMC4wLjEtU05BUFNIT1QuamFyCgotIFN1YnNjcmlwdGlvbiBNaWNyb3NlcnZpY2UKCmphdmEgLWphciB0YXJnZXQvYWRpZGFzLXN1YnNjcmlwdGlvbi1hcGktMC4wLjEtU05BUFNIT1QuamFyCgpVc2luZyBTdWJzY3JpcHRpb24gU2VydmljZSBpbiBkZXZlbG9wbWVudApJZiB5b3Ugd2FudCB0byBhY2Nlc3MgdG8gRXVyZWthIFNlcnZlciB0byBzZWUgdGhlIHJlZ2lzdHJlZCBzZXJ2aWNlcywgaHR0cDovL2xvY2FsaG9zdDo4NzYxIGluIHlvdXIgYnJvd3Nlci4KClRvIGFkZCBhIG5ldyBTdWJzY3JpcHRpb24sIGl0IGlzIG5lY2Vzc2FyeSB0byBtYWtlIGEgUE9TVCByZXF1ZXN0IHRvIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4My9zdWJzY3JpcHRpb24uIAoKIyBVc2luZyBEb2NrZXIKWW91IGNhbiBkb2NrZXJpY2UgdGhlIFN1YnNjcmlwdGlvbiBTZXJ2aWNlLiBUbyBhY2hpZXZlIHRoaXMsIGZpcnN0IGJ1aWxkIGEgZG9ja2VyIGltYWdlIG9mIGVhY2ggb2YgdGhlIHNlcnZpY2UgcnVubmluZyB0aGUgZm9sbG93aW5nIGNvbW1hbmQgaW4gdGhlIG1haW4gZm9sZGVyIG9mIGVhY2ggc2VydmljZS4KCm12biBjbGVhbiBwYWNrYWdlIC1QIGRvY2tlciBkb2NrZXI6YnVpbGQKQWZ0ZXIgZXhlY3V0aW5nIHRoZSBiZWxvdyBjb21tYW5kLCB5b3UgbXVzdCB0byBnbyB0byB0aGUgZG9ja2VyIGZvbGRlciwgYW5kIHJ1bjoKCmRvY2tlci1jb21wb3NlIC1mIGFwcC55bWwgdXAgLWQKVGhpcyBjb21tYW5kIHdpbGwgY3JlYXRlIGZvdXIgZG9ja2VyIGNvbnRhaW5lciBpbiB3aGljaCB0aGUgdGhyZWUgbWljcm9zZXJ2aWNlcyBhbmQgdGhlIEV1cmVrYSBTZXJ2ZXIgd2lsbCBiZSBleGVjdXRlZC4gSW4gYWRkaXRpb24sIGl0IHdpbGwgYmUgY3JlYXRlIGEgaW50ZXJuYWwgZG9ja2VyIG5ldHdvcmsgd2hpY2ggdG8gcG9zc2libGUgdGhlIGNvbW11bmljYXRpb24gYW1vbmcgYWxsIGNvaW50YWluZXJzLiBGaW5hbGx5LCBhbm90aGVyIGNvaW50YWluZXIgaXMgY3JlYXRlZCB3aGljaCBsaWUgdGhlIFBvc3RncmVTUUwuIE9ubHkgdGhlIHBvcnQgODA4MCBvZiB0aGUgY29udGFpbmVyIHdoZXJlIHdpbGwgYmUgZXhlY3V0ZWQgdGhlIFN1YnNjcmlwdGlvbiBNaWNyb3NlcnZpY2UgYW5kIHRoZSBzYW1lIHBvcnQgb2YgdGhlIEV1cmVrYSBTZXJ2ZXIgaGF2ZSBiZWVuIHB1Ymxpc2hlZCB0byBiZSByZWFjaGFibGUgc2luY2UgYSBleHRlcm5hbCBuZXR3b3JrLiBTbyBpdCBpcyBpbXBvc3NpYmxlIHRoYXQgdGhlIGRhdGFiYXNlLCB0aGUgRW1haWwgTWljcm9zZXJ2aWNlIGFuZCB0aGUgRXZlbnQgTWljcm9zZXJ2aWNlIHdhcyByZWFjaGFibGUgYnkgb3RoZXIgcHJvY2Nlc3MgdGhhdCBkbyBub3QgYmUgdGhlIFN1YmNyaXB0aW9uIE1pY3Jvc2VydmljZS4KClRvIGFjY2VzcyB0byBFdXJla2EgU2VydmVyIHRvIHNlZSB0aGUgcmVnaXN0cmVkIHNlcnZpY2VzLCBodHRwOi8vbG9jYWxob3N0OjgwODEKVG8gYWRkIGEgbmV3IFN1YnNjcmlwdGlvbiwgaXQgaXMgbmVjZXNzYXJ5IHRvIG1ha2UgYSBQT1NUIHJlcXVlc3QgdG8gaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N1YnNjcmlwdGlvbgoKIyBTd2FnZ2VyIERvY3VtZW50YXRpb24KU3dhZ2dlciBkb2N1bWVudGF0aW9uIGlzIGJ1aWxkIGFuZCBjb21taXRlZCBpbiBkb2N1bWVuYXRpb24gZm9sZGVyIHdoaWNoIGNvbnRhaW4gYWxsIGRldGFpbHMgYW5kIGVuZCBwb2ludHMgb2YgYWxsIG1pY3JvLXNlcnZpY2VzLgpJdCBpcyBidWlsZCB1c2luZyBzd2FnZ2VyLWh1Yi4K readmeEtag: '"15cad2ce98fe35576f7b45f301c30ddca2d226c3"' readmeLastModified: Thu, 13 Feb 2020 20:54:11 GMT repositoryId: 214656319 description: 'Micro-service API for Subscriptions and email ' created: '2019-10-12T13:58:31Z' updated: '2020-02-13T20:54:16Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: aqib1 logo: https://avatars.githubusercontent.com/u/8742169?v=4 repoEtag: '"4925407ec7a22ae162eb59f22825325741866d563c1ba5f455e517cc79eda7bc"' repoLastModified: Thu, 13 Feb 2020 20:54:16 GMT foundInMaster: true category: Server Implementations id: 1065ef5a8c3a7843dee99fca1e7c8381 - source: openapi3 tags repository: https://github.com/aqib1/idealotask v3: true repositoryMetadata: base64Readme: >- IyBJZGVhbG9UYXNrClRoZXNlIGFyZSB0aGUgZGV0YWlscyByZWdhcmRpbmcgdGFza3MuIEkgcGljayB0aGUgdGFzayAzIHdoaWNoIGlzIHRoZSBmb2xsb3dpbmcgYmVsb3cuIFRoZSBwcm9qZWN0IGlzIGNvbXBsZXRlbHkgZG9ja2VyaXplIHdpdGggYSBwdWJsaWMgaW1hZ2UgYXZhaWxhYmxlIG9uIGRvY2tlciBodWIgcGxlYXNlIHVzZSB0aGUgY29tbWFuZCA8Yj5kb2NrZXIgcHVsbCBhcWliYnV0dDMwNzgvdG95cm9ib3Q6aW1hZ2U8L2I+IHRvIHB1bGwgdGhlIGltYWdlLCBpZiB5b3UgbGlrZSB0byBydW4gaXQgZnJvbSBkb2NrZXIgc2hlbGwuCiMgVGFzayAzCgpDb2RlIHByb2JsZW0gZGV0YWlsczoKLSBUaGUgYXBwbGljYXRpb24gaXMgYSBzaW11bGF0aW9uIG9mIGEgdG95IHJvYm90IG1vdmluZyBvbiBhIHNxdWFyZSB0YWJsZXRvcCwgb2YKZGltZW5zaW9ucyA1IHggNSB1bml0cy4KLSBUaGVyZSBhcmUgbm8gb3RoZXIgb2JzdHJ1Y3Rpb25zIG9uIHRoZSB0YWJsZSBzdXJmYWNlLgotIFRoZSByb2JvdCBpcyBmcmVlIHRvIHJvYW0gYXJvdW5kIHRoZSBzdXJmYWNlIG9mIHRoZSB0YWJsZSwgYnV0IG11c3QgYmUgcHJldmVudGVkIGZyb20KZmFsbGluZyB0byBkZXN0cnVjdGlvbi4gQW55IG1vdmVtZW50IHRoYXQgd291bGQgcmVzdWx0IGluIHRoZSByb2JvdCBmYWxsaW5nIGZyb20gdGhlIHRhYmxlIG11c3QKYmUgcHJldmVudGVkLCBob3dldmVyIGZ1cnRoZXIgdmFsaWQgbW92ZW1lbnQgY29tbWFuZHMgbXVzdCBzdGlsbCBiZSBhbGxvd2VkLgpDcmVhdGUgYW4gYXBwbGljYXRpb24gdGhhdCBjYW4gcmVhZCBpbiBjb21tYW5kcyBvZiB0aGUgZm9sbG93aW5nIGZvcm06Ci0gUExBQ0UgWCxZLEYKLSBNT1ZFCi0gTEVGVAotIFJJR0hUCi0gUkVQT1JUClBMQUNFIHdpbGwgcHV0IHRoZSB0b3kgcm9ib3Qgb24gdGhlIHRhYmxlIGluIHBvc2l0aW9uIFgsWSBhbmQgZmFjaW5nIE5PUlRILApTT1VUSCwgRUFTVCBvciBXRVNULiBUaGUgb3JpZ2luICgwLDApIGNhbiBiZSBjb25zaWRlcmVkIHRvIGJlIHRoZSBTT1VUSCBXRVNUCm1vc3QgY29ybmVyLgpNT1ZFIHdpbGwgbW92ZSB0aGUgdG95IHJvYm90IG9uZSB1bml0IGZvcndhcmQgaW4gdGhlIGRpcmVjdGlvbiBpdCBpcyBjdXJyZW50bHkgZmFjaW5nLgpMRUZUIGFuZCBSSUdIVCB3aWxsIHJvdGF0ZSB0aGUgcm9ib3QgOTAgZGVncmVlcyBpbiB0aGUgc3BlY2lmaWVkIGRpcmVjdGlvbiB3aXRob3V0CmNoYW5naW5nIHRoZSBwb3NpdGlvbiBvZiB0aGUgcm9ib3QuClJFUE9SVCB3aWxsIGFubm91bmNlIHRoZSBYLFkgYW5kIEYgb2YgdGhlIHJvYm90LiAKQ29uc3RyYWludHM6Ci0gVGhlIGFwcGxpY2F0aW9uIG11c3QgYmUgYSBTcHJpbmctQm9vdC1BcHBsaWNhdGlvbgotIElucHV0IG11c3QgYmUgcmVhbGlzZWQgb3ZlciB0aGUgUkVTVC1BUEksIHRha2UgY2FyZSB3aGVuIGRlc2lnbmluZyB0aGUgUkVTVC1BUEkKLSBUaGUgcm9ib3QgdGhhdCBpcyBub3Qgb24gdGhlIHRhYmxlIGNhbiBjaG9vc2UgdGhlIGlnbm9yZSB0aGUgTU9WRSwgTEVGVCwgUklHSFQKYW5kIFJFUE9SVCBjb21tYW5kcy4KLSBUaGUgcm9ib3QgbXVzdCBub3QgZmFsbCBvZmYgdGhlIHRhYmxlIGR1cmluZyBtb3ZlbWVudC4gVGhpcyBhbHNvIGluY2x1ZGVzIHRoZSBpbml0aWFsCnBsYWNlbWVudCBvZiB0aGUgdG95IHJvYm90LgotIEFueSBtb3ZlIHRoYXQgd291bGQgY2F1c2UgdGhlIHJvYm90IHRvIGZhbGwgbXVzdCBiZSBpZ25vcmVkLgotIEl0IGlzIG5vdCByZXF1aXJlZCB0byBwcm92aWRlIGFueSBncmFwaGljYWwgb3V0cHV0IHNob3dpbmcgdGhlIG1vdmVtZW50IG9mIHRoZSB0b3kKcm9ib3QuIApEZWxpdmVyYWJsZXM6IAotIHRoZSB3aG9sZSBwcm9qZWN0LCBleGFtcGxlIHJlcXVlc3RzIHRvIHRlc3QgYSByb2JvdC1hcHBsaWNhdGlvbiBpbiBmb3JtIG9mIFBvc3RtYW5Db2xsZWN0aW9uIG9yIGluIGZvcm0gb2YgYSB0ZXh0IGZpbGUuCg== readmeEtag: '"57b6a5f60fbbeca3908f6e510e099b5d558ecb3d"' readmeLastModified: Wed, 11 Sep 2019 01:08:14 GMT repositoryId: 203261645 description: null created: '2019-08-19T22:55:03Z' updated: '2019-10-17T02:25:03Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: aqib1 logo: https://avatars.githubusercontent.com/u/8742169?v=4 repoEtag: '"144f88de7338edaae5156b7b47f048108abf85df3920e9ae2f6a3120fef11f57"' repoLastModified: Thu, 17 Oct 2019 02:25:03 GMT foundInMaster: true category: Server Implementations id: 3dd9aa2756dfd980f28501062f14c68a - source: openapi3 tags repository: https://github.com/marcomicera/rns v3: true repositoryMetadata: base64Readme: >- IyBgcm5zYDogUm9hZCBOYXZpZ2F0aW9uIFNlcnZpY2VzCmBybnNgIGlzIGEgd2ViIHNlcnZpY2UgZGV2ZWxvcGVkIGFzIGFuIGFzc2lnbm1lbnQgZm9yIHRoZSBEaXN0cmlidXRlZCBQcm9ncmFtbWluZyBJSSBjb3Vyc2UgYXQgdGhlIFBvbHl0ZWNobmljIFVuaXZlcnNpdHkgb2YgVHVyaW4uCgojIyMjIFVzZWZ1bCBsaW5rcwotIFtDb3Vyc2Ugd2Vic2l0ZV0oaHR0cHM6Ly9wYWQucG9saXRvLml0OjgwODAvZW5naW5mcmFtZS9kcDIvZHAyLnhtbCkKICAtIFtBc3NpZ25tZW50cyBGQVFzXShodHRwczovL3BhZC5wb2xpdG8uaXQ6ODA4MC9lbmdpbmZyYW1lL2RwMi9kcDIueG1sP191cmk9Ly9kcDIvYXNzaWdubWVudF9mYXFzKQogIC0gW0V4YW1wbGVzXShodHRwczovL3BhZC5wb2xpdG8uaXQ6ODA4MC9lbmdpbmZyYW1lL2RwMi9kcDIueG1sP191cmk9Ly9kcDIvbWF0ZXJpYWwpCi0gW2BpdC5wb2xpdG8uZHAyLlJOU2AgcGFja2FnZV0oaHR0cHM6Ly9wYWQucG9saXRvLml0OjgwODAvZW5naW5mcmFtZS9kcDIvYXNzaWdubWVudHMvbGFiMS9kb2MvaW5kZXguaHRtbCkKLSBbU3dhZ2dlciBBUElzXShodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzL21hcmNvbWljZXJhL1Juc1N5c3RlbS8yLjAtb2FzMykKICAtIFtPcGVuQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vYWJvdXQvKQotIFtMaXN0IG9mIEhUVFAgc3RhdHVzIGNvZGVzXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9MaXN0X29mX0hUVFBfc3RhdHVzX2NvZGVzKQotIFtQVVQgdnMuIFBPU1QgaW4gUkVTVF0oaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9xdWVzdGlvbnMvNjMwNDUzL3B1dC12cy1wb3N0LWluLXJlc3QpCgojIyMjIFNvZnR3YXJlCi0gYG9wZW5qZGstOC1qZGtgCi0gW0FwYWNoZSBBbnQgMS45LjZdKGh0dHBzOi8vYXJjaGl2ZS5hcGFjaGUub3JnL2Rpc3QvYW50L2JpbmFyaWVzL2FwYWNoZS1hbnQtMS45LjYtYmluLnRhci5neikgCi0gW05lbzRKIDMuMi4zXShodHRwczovL25lbzRqLmNvbS9kb3dubG9hZC10aGFua3MvP2VkaXRpb249Y29tbXVuaXR5JnJlbGVhc2U9My4yLjMmZmxhdm91cj11bml4KQogIC0gW0Rpc2FibGUgYXV0aGVudGljYXRpb25dKGh0dHBzOi8vbmVvNGouY29tL2RvY3Mvb3BlcmF0aW9ucy1tYW51YWwvY3VycmVudC9hdXRoZW50aWNhdGlvbi1hdXRob3JpemF0aW9uL2VuYWJsZS8pCi0gW0FwYWNoZSBUb21jYXQgOC41LjIwXShodHRwczovL2FyY2hpdmUuYXBhY2hlLm9yZy9kaXN0L3RvbWNhdC90b21jYXQtOC92OC41LjIwL2Jpbi9hcGFjaGUtdG9tY2F0LTguNS4yMC50YXIuZ3opCiAgLSAqKkltcG9ydGFudCoqOiBzZXQgYHNoYXJlZC5sb2FkZXI9L29wdC9kcDIvc2hhcmVkL2xpYi8qLmphcmAgaW4gYHRvbWNhdF9kaXIvY29uZi9jYXRhbGluYS5wcm9wZXJ0aWVzYAo= readmeEtag: '"fb68e949435176fcf46001df4ed55eeb9519f0cf"' readmeLastModified: Sat, 11 Apr 2020 13:40:14 GMT repositoryId: 183401765 description: Road Navigation Services created: '2019-04-25T09:28:59Z' updated: '2023-01-28T15:05:33Z' language: Java archived: true stars: 0 watchers: 1 forks: 0 owner: marcomicera logo: https://avatars.githubusercontent.com/u/13918587?v=4 license: MIT repoEtag: '"d196600ffaa5e024ee073182aba3503f84a2e3055cae3ff547415cae3c3e988a"' repoLastModified: Sat, 28 Jan 2023 15:05:33 GMT foundInMaster: true category: Code Generators id: c697198760415f3a02305c92e624b53c - source: openapi3 tags repository: https://github.com/sumant86/node-rest-api v3: true repositoryMetadata: base64Readme: >- IyBOb2RlIFJFU1QgQVBJCgpBIGtpY2sgc3RhcnQgcHJvamVjdCBmb3IgaW1wbGVtZW50YXRpb24gb2YgUkVTVGZ1bCBBUEkgd2l0aCBOb2RlLmpzIHVzaW5nIEVTNi4KRmVhdHVyZXMgaW5jbHVkZToKCi0gICBVc2Ugb2YgdGhlIEhleGFnb25hbCBBcmNoaXRlY3R1cmUgdG8gYXJyYW5nZSB0aGUgYXBwbGljYXRpb24gaW50byBsb2dpY2FsCiAgICBsYXllcnMsIHdpdGggd2VsbC1kZWZpbmVkIHJlc3BvbnNpYmlsaXRpZXMuCi0gICBSRVNUZnVsIEFQSXMgYXJlIGltcGxlbWVudGVkIHVzaW5nIHRoZSBbRXhwcmVzc10oaHR0cDovL2V4cHJlc3Nqcy5jb20vKQogICAgZnJhbWV3b3JrLgotICAgUGVyc2lzdGVuY2UgaXMgaW1wbGVtZW50ZWQgdXNpbmcgYW4gaW4tbWVtb3J5IHJlcG9zaXRvcnkgbGF5ZXIuIFRoaXMgY2FuIGJlCiAgICBzdWJzdGl0dXRlZCB3aXRoIGFueSBwZXJzaXN0ZW5jZSB0ZWNobm9sb2d5IG9mIHlvdXIgY2hvaWNlLgotICAgTVNTUUwgJiBNeVNxbCBhZGFwdG9ycyBhcmUgcHJvdmlkZWQgZm9yIGZsZXhpYmxpdHkgdG8gY29ubmVjdCB3aXRoIGRhdGFiYXNlLgogICAgRGF0YWJhc2UgY3JlZGVudGlhbHMgbmVlZCB0byBiZSB1cGRhdGVkIHRvIGAuZW52YCBmaWxlCgojIyBEZXYgQnVpbGQKCmBgYGJhc2gKJCBucG0gaW5zdGFsbAokIG5wbSBzdGFydApgYGAKClRoZSBkZXYgYnVpbGQgc3RhcnRzIHRoZSBhcHBsaWNhdGlvbiBpbiB3YXRjaCBtb2RlLiBJZiB5b3UgbWFrZSBhbnkgY2hhbmdlcyB0bwp0aGUgc291cmNlIGZpbGVzLCB0aGUgYXBwbGljYXRpb24gd2lsbCByZWNvbXBpbGUgYW5kIHJlc3RhcnQuCgpUbyB2ZXJpZnkgdGhhdCB0aGUgYXBwbGljYXRpb24gaXMgd29ya2luZyBjb3JyZWN0bHksIHBvaW50IHlvdXIgYnJvd3NlciB0bwpbaHR0cDovL2xvY2FsaG9zdDozMDAwL2FwaS9wcm9kdWN0cy8xXShodHRwOi8vbG9jYWxob3N0OjMwMDAvYXBpL3Byb2R1Y3RzLzEpIC0KeW91IHNob3VsZCBzZWUgYSByZXNwb25zZSB3aXRoIG9uZSBwcm9kdWN0IGluIEpTT04gZm9ybWF0LgoKWW91IGNhbiBzZWUgYSBPcGVuQVBJIChTd2FnZ2VyKSBkZWZpbml0aW9uIG9mIHRoZSBSRVNUIEFQSSBhdApbaHR0cDovL2xvY2FsaG9zdDozMDAwL2FwaS1kb2NzL10oaHR0cDovL2xvY2FsaG9zdDozMDAwL2FwaS1kb2NzLykuIFRoaXMKaW50ZXJmYWNlIGFsc28gYWxsb3dzIHlvdSB0byBpbnRlcmFjdCB3aXRoIHRoZSBBUEkuIFBvc3RtYW4gQ29sbGVjdGlvbiBwcm92aWRlZAp3aXRoIHRoaXMgcmVwb3NpdG9yeS4KClRvIGRlYnVnIHRoZSBhcHBsaWNhdGlvbiBpbiBDaHJvbWUsIHBvaW50IHRoZSBicm93c2VyIHRvIGNocm9tZTovL2luc3BlY3QgYW5kCmNsaWNrIG9uICJPcGVuIGRlZGljYXRlZCBEZXZUb29scyBmb3IgTm9kZSIuCgojIyBQcm9kdWN0aW9uIEJ1aWxkCgpgYGBiYXNoCiQgbnBtIHJ1biBwcmV0dGllcgokIG5wbSBydW4gYnVpbGQKJCBucG0gcnVuIHNlcnZlCmBgYAoKIyMgUHJldHRpZXIKCmBgYGJhc2gKJCBucG0gcnVuIHByZXR0aWVyCmBgYAoKIyMgRm9sZGVyIFN0cnVjdHVyZQoKYGBgCi9zcmMKICAgIC9yb3V0ZXMKICAgIC9zZXJ2aWNlcwogICAgL3JlcG9zaXRvcmllcwogICAgL2Nvbm5lY3RvcnMKICAgIC91dGlscwpgYGAKClRoZSBzb3VyY2UgZm9sZGVyIGNvbnRhaW5zIHN1Yi1mb2xkZXJzIHRoYXQgYXJyYW5nZSB0aGUgYXBwbGljYXRpb24gaW50byBsb2dpY2FsCmxheWVyczoKCi0gICBgcm91dGVzYDogVGhpcyBpcyB0aGUgYWRhcHRlciBsYXllciBvZiB0aGUgSGV4YWdvbmFsIEFyY2hpdGVjdHVyZS4gSXQgYWRhcHRzCiAgICB0aGUgSFRUUCB0cmFuc2Zvcm1zIHRoZSBIVFRQIHJlcXVlc3RzIGZyb20gdGhlIGV4dGVybmFsIHdvcmxkIHRvIHRoZSBzZXJ2aWNlCiAgICBsYXllciBhbmQgdHJhbnNmb3JtcyB0aGUgb2JqZWN0cyByZXR1cm5lZCBieSB0aGUgc2VydmljZSBsYXllciB0byBIVFRQCiAgICByZXNwb25zZXMuCgotICAgYHNlcnZpY2VzYDogVGhlIHNlcnZpY2UgbGF5ZXIgY29vcmRpbmF0ZXMgaGlnaC1sZXZlbCBhY3Rpdml0aWVzIHN1Y2ggYXMKICAgIGNyZWF0aW9uIG9mIGRvbWFpbiBvYmplY3RzIGFuZCBhc2tpbmcgdGhlbSB0byBwZXJmb3JtIHRhc2tzIHJlcXVlc3RlZCBieSB0aGUKICAgIGV4dGVybmFsIHdvcmxkLiBJdCBpbnRlcmFjdHMgd2l0aCB0aGUgcmVwb3NpdG9yeSBsYXllciB0byBzYXZlIGFuZCByZXN0b3JlCiAgICBvYmplY3RzLgoKLSAgIGByZXBvc2l0b3JpZXNgOiBUaGUgcmVwb3NpdG9yeSBsYXllciBpcyByZXNwb25zaWJsZSBmb3IgcGVyc2lzdGluZyBkb21haW4KICAgIG9iamVjdHMgYW5kIHBlcmZvcm1pbmcgQ1JVRCBvcGVyYXRpb25zIG9uIHRoZW0uIFRoaXMgdGVtcGxhdGUgdXNlcyBpbi1tZW1vcnkKICAgIHBlcnNpc3RlbmNlIGZvciBwcm9kdWN0cyAmIE15c3FsICYgTVNTcWwgY29ubmVjdG9yIHRvIGNvbm5lY3Qgd2l0aCBkYXRhYmFzZS4KCi0gICBgY29ubmVjdG9yc2A6IFRoZSBjb25uZWN0b3IgbGF5ZXIgYWxsb3dzIGFwcGxpY2F0aW9uIHRvIGNvbm5lY3Qgd2l0aAogICAgZGF0YWJhc2UgZm9yIGFsbCBxdWVyaWVzLiBDdXJyZW50bHkgY29ubmVjdG9ycyBhcmUgYXZhaWxhYmxlIGZvciBgTXlTcWxgICYKICAgIGBNc3NxbGAuIE15c3FsIGNvbm5lY3RvciBpcyB1c2luZyBgbXlzcWxgIGRyaXZlci4gTVNTcWwgY29ubmVjdG9yIGlzIHVzaW5nCiAgICBgbXNzcWxgICYgYHRlZGlvdXNgIGRyaXZlciwgaXQgaXMgdXBvbiB1c2VyIHRvIGNob29zZSB0aGUgZHJpdmVycyBiYXNlZCBvbgogICAgZmxleGlibGl0eS4KCi0gICBUaGUgYHV0aWxzYCBmb2xkZXIgY29udGFpbnMgdXNlZnVsIHV0aWxpdGllcyBhbmQgaGVscGVycy4K readmeEtag: '"57cf6e528c666d7f4c34a56fc4480cb1bbce580a"' readmeLastModified: Wed, 11 May 2022 08:12:07 GMT repositoryId: 268822901 description: >- A kick start project for implementation of RESTful API with Node.js using ES6. created: '2020-06-02T14:26:38Z' updated: '2022-02-06T09:00:11Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: sumant86 logo: https://avatars.githubusercontent.com/u/3600265?v=4 repoEtag: '"b4f0395e3e8e6bfe4e9b3ac04323fbacce71323aa786e6240a74a45d8bf2e045"' repoLastModified: Sun, 06 Feb 2022 09:00:11 GMT foundInMaster: true category: Server Implementations id: 1eae6eee2c342724345c837c65b692e2 - source: openapi3 tags repository: https://github.com/mtkhawaja/spring-boot-3-openapi-archetype v3: true id: a0cf3aea3870ae92510ec88f50ce5008 repositoryMetadata: base64Readme: >- IyBzcHJpbmctYm9vdC0zLW9wZW5hcGktYXJjaGV0eXBlCgpNYXZlbiBBcmNoZXR5cGUgZm9yIGdlbmVyYXRpbmcgYmFzaWMgSmF2YSAxNyBTcHJpbmcgQm9vdCAzIHByb2plY3RzIHdpdGggT3BlbkFQSSBzdXBwb3J0LgoKIyMgR2V0dGluZyBTdGFydGVkCgohW0phdmEgMTddKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvamF2YS0xNy1vcmFuZ2UpCgojIyMgQnVpbGQKCmBgYGJhc2gKIyEvdXNyL2Jpbi9lbnYgYmFzaAouL212bncgY2xlYW4gaW5zdGFsbApgYGAKCiMjIyBVc2FnZQoKYGBgYmFzaAojIS91c3IvYmluL2VudiBiYXNoCgptdm4gYXJjaGV0eXBlOmdlbmVyYXRlIFwKICAtRGFyY2hldHlwZUdyb3VwSWQ9ImNvbS5tdW5lZWJraGF3YWphIiBcCiAgLURhcmNoZXR5cGVBcnRpZmFjdElkPSJzcHJpbmctYm9vdC0zLW9wZW5hcGktYXJjaGV0eXBlIiBcCiAgLURhcmNoZXR5cGVWZXJzaW9uPSIxLjAuMCIgXAogIC1EZ3JvdXBJZD0iY29tLmV4YW1wbGUiIFwKICAtRGFydGlmYWN0SWQ9ImFwaSIgXAogIC1EcGFja2FnZT0iY29tLmV4YW1wbGUuYXBpIiBcCiAgLUR2ZXJzaW9uPSIwLjAuMS1TTkFQU0hPVCIgXAogIC1CCgpgYGAKCkFsdGVybmF0aXZlbHksIHlvdSBjYW4gdXNlIGRvY2tlciBhbmQgdm9sdW1lIGJpbmRpbmcgdG8gZ2VuZXJhdGUgdGhlIHByb2plY3Q6CgpgYGBiYXNoCiMhL3Vzci9iaW4vZW52IGJhc2gKCmRvY2tlciBidWlsZCAtdCBtdGtoYXdhamEvc3ByaW5nLWJvb3QtMy1vcGVuYXBpLWFyY2hldHlwZTpsYXRlc3QgLgpkb2NrZXIgcnVuIFwKICAtdiAuLzovb3B0L2FwcGxpY2F0aW9uL2dlbmVyYXRlZCBcCiAgLWUgR0VORVJBVEVEX0dST1VQX0lEPSJjb20uZXhhbXBsZSIgXAogIC1lIEdFTkVSQVRFRF9BUlRJRkFDVF9JRD0iYXBpIiBcCiAgLWUgR0VORVJBVEVEX1BBQ0tBR0U9ImNvbS5leGFtcGxlLmFwaSIgXAogIC1lIEdFTkVSQVRFRF9WRVJTSU9OPSIwLjAuMS1TTkFQU0hPVCIgXAogIG10a2hhd2FqYS9zcHJpbmctYm9vdC0zLW9wZW5hcGktYXJjaGV0eXBlOmxhdGVzdAoKYGBgCgpUaGlzIHdpbGwgZ2VuZXJhdGUgYSBwcm9qZWN0IHdpdGggdGhlIGZvbGxvd2luZyBkaXJlY3Rvcnkgc3RydWN0dXJlOgoKYGBgYmFzaAojIS91c3IvYmluL2VudiBiYXNoCgp0cmVlIC1hIC4vYXBpCmBgYAoKYGBgcGxhaW50ZXh0Ci4vYXBpLwrilJzilIDilIAgLmdpdGlnbm9yZQrilJzilIDilIAgLm12bgrilIIgICDilJTilIDilIAgd3JhcHBlcgrilIIgICAgICAg4pSU4pSA4pSAIG1hdmVuLXdyYXBwZXIucHJvcGVydGllcwrilJzilIDilIAgRG9ja2VyZmlsZQrilJzilIDilIAgTElDRU5TRQrilJzilIDilIAgUkVBRE1FLm1kCuKUnOKUgOKUgCBkb2NrZXItY29tcG9zZS55YW1sCuKUnOKUgOKUgCBtdm53CuKUnOKUgOKUgCBtdm53LmNtZArilJzilIDilIAgcG9tLnhtbArilJTilIDilIAgc3JjCiAgICDilJzilIDilIAgbWFpbgogICAg4pSCICAg4pSc4pSA4pSAIGphdmEKICAgIOKUgiAgIOKUgiAgIOKUlOKUgOKUgCBjb20KICAgIOKUgiAgIOKUgiAgICAgICDilJTilIDilIAgZXhhbXBsZQogICAg4pSCICAg4pSCICAgICAgICAgICDilJTilIDilIAgYXBpCiAgICDilIIgICDilIIgICAgICAgICAgICAgICDilJzilIDilIAgQXBwbGljYXRpb24uamF2YQogICAg4pSCICAg4pSCICAgICAgICAgICAgICAg4pSU4pSA4pSAIEV4YW1wbGVDb250cm9sbGVyLmphdmEKICAgIOKUgiAgIOKUlOKUgOKUgCByZXNvdXJjZXMKICAgIOKUgiAgICAgICDilJzilIDilIAgYXBwbGljYXRpb24ueWFtbAogICAg4pSCICAgICAgIOKUnOKUgOKUgCBsb2diYWNrLnhtbAogICAg4pSCICAgICAgIOKUlOKUgOKUgCBzd2FnZ2VyLnlhbWwKICAgIOKUlOKUgOKUgCB0ZXN0CiAgICAgICAg4pSU4pSA4pSAIGphdmEKICAgICAgICAgICAg4pSU4pSA4pSAIGNvbQogICAgICAgICAgICAgICAg4pSU4pSA4pSAIGV4YW1wbGUKICAgICAgICAgICAgICAgICAgICDilJTilIDilIAgYXBpCiAgICAgICAgICAgICAgICAgICAgICAgIOKUnOKUgOKUgCBBcHBsaWNhdGlvblNtb2tlVGVzdC5qYXZhCiAgICAgICAgICAgICAgICAgICAgICAgIOKUlOKUgOKUgCBFeGFtcGxlQ29udHJvbGxlclRlc3QuamF2YQpgYGAKCiMjIExpY2Vuc2UKCltBcGFjaGUgTGljZW5zZSBWZXJzaW9uIDIuMF0oLi9MSUNFTlNFKQ== readmeEtag: '"a662790df6529dfde16e7703612ff5241c810ccc"' readmeLastModified: Wed, 14 Jun 2023 03:20:06 GMT repositoryId: 649925362 description: >- Maven Archetype for generating Spring Boot 3 projects with OpenAPI support. created: '2023-06-06T00:19:12Z' updated: '2024-03-11T10:37:33Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: mtkhawaja logo: https://avatars.githubusercontent.com/u/36654508?v=4 license: Apache-2.0 repoEtag: '"a79866567497b2468ca27fbabad557d707d12dd1b994eb0da332d45d67ce6f88"' repoLastModified: Mon, 11 Mar 2024 10:37:33 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/calebtracey/rugby-crawler-api v3: true id: 2fba72e912abd94c76e3860abaa6215b repositoryMetadata: base64Readme: >- IyBydWdieS1jcmF3bGVyLWFwaQoKWyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL0NhbGViVHJhY2V5L3J1Z2J5LWNyYXdsZXItYXBpL2JyYW5jaC9tYWluL2dyYXBoL2JhZGdlLnN2Zz90b2tlbj04S08xWVQzQ09NKV0oaHR0cHM6Ly9jb2RlY292LmlvL2doL0NhbGViVHJhY2V5L3J1Z2J5LWNyYXdsZXItYXBpKQoKIyMjIEdvIFJFU1QgQVBJIGZvciBzY3JhcGluZyBydWdieSBzdGF0cwojIyMgW1N3YWdnZXIgRG9jc10oaHR0cHM6Ly9jYWxlYnRyYWNleS5naXRodWIuaW8vcnVnYnktY3Jhd2xlci1hcGkvc3dhZ2dlci11aS8pCi0tLQoKW3J1Z2J5LW1vZGVsc10oaHR0cHM6Ly9naXRodWIuY29tL0NhbGViVHJhY2V5L3J1Z2J5LW1vZGVscykgdXNlZCBmb3IgY29tbW9uIGRhdGEgdHlwZXMKCltjb25maWcteWFtbF0oaHR0cHM6Ly9naXRodWIuY29tL0NhbGViVHJhY2V5L2NvbmZpZy15YW1sKSB1c2VkIGZvciBlbnZpcm9ubWVudCBjb25maWdzIGFuZCBzZXJ2aWNlL2RiIGluaXRpYWxpemF0aW9uCgoqKkJhc2ljIHNldHVwKioKMS4gQ3JlYXRlIGEgbG9jYWwgUG9zdGdyZXMgZGF0YWJhc2UgYW5kIHVwZGF0ZSB0aGUgY29uZmlnLnlhbWwgZmlsZSB3aXRoIHRoZSBkZXRhaWxzCjIuIFVwZGF0ZSBnby5tb2QgYW5kIGZpbGUgaW1wb3J0cyB3aXRoIHlvdXIgcmVwbyBuYW1lCjMuIFJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmRzIHRvIHVwZGF0ZSBkZXBlbmRlbmNpZXM6CgogICBgZ28gZ2V0IC11IC4vLi4uYAoKICAgIGBnbyBtb2QgdGlkeWAKNC4gTWFrZSBhIHJ1biBjb25maWd1cmF0aW9uIGFzIHNlZW4gYmVsb3cgd2l0aCB5b3VyIHJlcG8gbmFtZToKCiFbUnVuIENvbmZpZ10oLi9pbWFnZXMvcnVuLWNvbmZpZy5wbmcpCgojIyMjIE5vdyB5b3UgY2FuIHN0YXJ0IHRoZSBBUEkgYW5kIGFjY2VzcyBodHRwOi8vbG9jYWxob3N0OjYwODAvc3dhZ2dlci11aS8gZm9yIHN3YWdnZXIgZG9jdW1lbnRhdGlvbiBhbmQgdGVzdGluZwo= readmeEtag: '"a537cddfbc77d2c961e0f8b1ab325ad877520676"' readmeLastModified: Fri, 09 Dec 2022 00:27:00 GMT repositoryId: 570243924 description: Go API for scraping and storing rugby stats created: '2022-11-24T17:17:01Z' updated: '2022-11-25T05:42:19Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: CalebTracey logo: https://avatars.githubusercontent.com/u/59851860?v=4 repoEtag: '"5342b71b43229be8de73cb014661085fb0195721c8b5bdc6d6f2200be212dcd0"' repoLastModified: Fri, 25 Nov 2022 05:42:19 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mukul273/springboot-react v3: true repositoryMetadata: base64Readme: >- IyBzcHJpbmdib290LXJlYWN0ClNwcmluZyBib290IHdpdGggUmVhY3RKcyBmdWxsIHN0YWNrCgpUaGlzIHByb2plY3QgdXNlcyBTcHJpbmcgYm9vdCAyLjMuNiwgUmVhY3RKcywgTXlTcWwsIFNwcmluZyBEYXRhIEpQQSwgSmF2YSA4LCBsb21ib2sgZXRjLgoKVGhpcyBpcyBFbXBsb3llZSBhZGQsIFVwZGF0ZSwgZGVsZXRlIGFuZCByZWFkIChDUlVEKSBVc2UgY2FzZSBhcHBsaWNhdGlvbiAKClRoaXMgYXBwbGljYXRpb24gaGFzIGJlZW4gdXBkYXRlZCB3aXRoIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gIApHbyB0byAqIFtMb2NhbEhvc3QgT3BlbkFQSV0oaHR0cDovL2xvY2FsaG9zdDo4MDgxL3YzL2FwaS1kb2NzKSAqKiBQbGVhc2UgY2hhbmdlIHlvdXIgcG9ydCAgICAKClRPIGNoYW5nZSB0aGUgQVBJIGRvYyBVUkwgYWJvdmUKUGxlYXNlIHJlZmVyIHRvIHRoZSBhcHBsaWNhdGlvbi5wcm9wZXJ0aWVzIGZpbGUKTG9vayBmb3Igc3ByaW5nZG9jLmFwaS1kb2NzLnBhdGggcGFyYW0gYW5kIGNvbmZpZ3VyZSB5b3VyIHZhbHVlCioqIFBsZWFzZSByZW1lbWJlciB0aGF0IG9uY2UgeW91IGN1c3RvbWl6ZSB0aGUgQVBJIHVybCB0aGVuIHRoZSBkZWZhdWx0IHYzL2FwaS1kb2NzIHdpbGwgIApub3Qgd29yay4KCioqIFBsZWFzZSBhcHBlbmQgLnlhbWwgdG8gdGhlIE9wZW5BUEkgdG8gZG93bmxvYWQgdGhlIEFQSSBzcGVjaWZpY2F0aW9uIGZpbGUsCgpTd2FnZ2VyIGRvY3VtZW50YXRpb24gaXMgc3RpbGwgYXZhaWxhYmxlCkdvIHRvICogW0xvY2FsSG9zdCBTd2FnZ2VyRG9jXShodHRwOi8vbG9jYWxob3N0OjgwODEvc3dhZ2dlci11aS5odG1sKSAqKiBQbGVhc2UgY2hhbmdlIHlvdXIgcG9ydCAKUGxlYXNlIHNwZWNpZnkgL2VtcGxveWVlLWFwaSBpbiB0aGUgZXhwbG9yZSBzZWFyY2gKCgo= readmeEtag: '"1a41b889454036a875abef2b8194c293915552ae"' readmeLastModified: Tue, 01 Dec 2020 16:23:42 GMT repositoryId: 315981861 description: Spring boot with ReactJs full stack created: '2020-11-25T15:35:50Z' updated: '2020-12-01T17:05:49Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: mukul273 logo: https://avatars.githubusercontent.com/u/5340977?v=4 repoEtag: '"d222ecd15ecee055aa652188d96676e3edd73174c95746037a5f0c1086dfef64"' repoLastModified: Tue, 01 Dec 2020 17:05:49 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: 7e6beb62a5584cefa8c1563cd30bd3e9 - source: openapi3 tags repository: https://github.com/akscodebay/announcement-api v3: true repositoryMetadata: base64Readme: >- IyBhbm5vbm5jZW1lbnQtYXBpCkFubm91bmNlbWVudCBBUEkgZm9yIHN0b3JpbmcgYW5kIHJldHJpZXZpbmcgYW5ub3VuY2VtZW50cy4gRGVwbG95YWJsZSBvbiBBV1MuCg== readmeEtag: '"6e97fad2f453b945a07f178371fcd07bd01dbce5"' readmeLastModified: Thu, 10 Dec 2020 16:57:56 GMT repositoryId: 320334399 description: >- Announcement API for storing and retrieving announcements. Deployable on AWS. created: '2020-12-10T16:53:08Z' updated: '2020-12-10T17:00:13Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: akscodebay logo: https://avatars.githubusercontent.com/u/35652014?v=4 repoEtag: '"9367d647cca8aa17771422ca80ff2f38f39965ea83f06f3886d41e03c2599e6f"' repoLastModified: Thu, 10 Dec 2020 17:00:13 GMT foundInMaster: true category: - Server - Server Implementations id: d10619272f684cdb44d5b014ce5092dd - source: openapi3 tags repository: https://github.com/pexmor/demo-openapi3 v3: true repositoryMetadata: base64Readme: >- IyBkZW1vLW9wZW5hcGkzCgpDb25jZXB0dWFsIGRlbW8gb2Ygc2ltcGxlIG9wZW5hcGkzIHVzaW5nIGNvbm5leGlvbiBmb3IgcXVpY2sgcHJvdG90eXBpbmcuCgoqIFtBUEkgYXQgZ2xhbmNlXShodHRwczovL3BleG1vci5naXRodWIuaW8vZGVtby1vcGVuYXBpMy9lbGVtZW50cy5odG1sKQoqIFtHaXRodWIgcGFnZXNdKGh0dHBzOi8vcGV4bW9yLmdpdGh1Yi5pby9kZW1vLW9wZW5hcGkzLykKCiMjIEN1cmwgdGVzdHMKCl9fSlNPTl9fCgpgYGBiYXNoCmN1cmwgLVggJ1BPU1QnIFwKICAnaHR0cDovL2xvY2FsaG9zdDo4MDgwL3YxL2pzb24vdXNlcl8wMDclNDBleGFtcGxlLmNvbScgXAogIC1IICdhY2NlcHQ6ICovKicgXAogIC1IICdYLUF1dGg6IGFiY2QnIFwKICAtSCAnQ29udGVudC1UeXBlOiBhcHBsaWNhdGlvbi9qc29uJyBcCiAgLWQgJ3sKICAiZGF0YSI6ICJJeUV2WW1sdUwySmhjMmdLQ2taTVFWTkxYMFJGUWxWSFBURWdURTlIVEVWV1JVdzlSRVZDVlVjZ2NIbDBhRzl1TXlCaGNHbFRjbll1Y0hrSyIsCiAgImhhc2giOiAiY2U5NGE5ZTg1NWNkYWEwNGQ5NWRhMTkzNjExMjM2YjZhZGVjMTYyNTY4NjU2NjJhM2QyMTA1ODU4YjAwZmI0MiIsCiAgImhhc2hfdHlwZSI6ICJzaGEyNTYiLAogICJ2YXJzIjogewogICAgImluZnJhIjogIm9uZSIsCiAgICAiaXB2NCI6IFsKICAgICAgIjEuMi4zLjQvMzIiCiAgICBdLAogICAgInNzaF9wdWJfa2V5IjogWwogICAgICAic3NoLXJzYSBBQUFBQjNOemFDMXljMkVBQUFBREFRQUJBQUFCZ1FETC4uLj0geW91ci1pZCIsCiAgICAgICJzc2gtcnNhIEFBQUFCM056YUMxeWMyRUFBQUFEQVFBQkFBQUJnUURMLi4uPSB5b3VyLWlkMiIKICAgIF0KICB9Cn0nCmBgYAoKX19NdWx0aXBhcnQgZm9ybV9fCgpgYGBiYXNoCmN1cmwgLVggJ1BPU1QnIFwKICAnaHR0cDovL2xvY2FsaG9zdDo4MDgwL3YxL211bHRpcGFydC91c2VyXzAwNyU0MGV4YW1wbGUuY29tJyBcCiAgLUggJ2FjY2VwdDogKi8qJyBcCiAgLUggJ1gtQXV0aDogYWJjZCcgXAogIC1IICdDb250ZW50LVR5cGU6IG11bHRpcGFydC9mb3JtLWRhdGEnIFwKICAtRiAnZGF0YT1ATElDRU5TRTt0eXBlPXRleHQvcGxhaW4nIFwKICAtRiAnaGFzaD1jZTk0YTllODU1Y2RhYTA0ZDk1ZGExOTM2MTEyMzZiNmFkZWMxNjI1Njg2NTY2MmEzZDIxMDU4NThiMDBmYjQyJyBcCiAgLUYgJ2hhc2hfdHlwZT1zaGEyNTYnIFwKICAtRiAndmFycz17CiAgImluZnJhIjogIm9uZSIsCiAgImlwdjQiOiBbCiAgICAiMS4yLjMuNC8zMiIKICBdLAogICJzc2hfcHViX2tleSI6IFsKICAgICJzc2gtcnNhIEFBQUFCM056YUMxeWMyRUFBQUFEQVFBQkFBQUJnUURMLi4uPSB5b3VyLWlkIiwKICAgICJzc2gtcnNhIEFBQUFCM056YUMxeWMyRUFBQUFEQVFBQkFBQUJnUURMLi4uPSB5b3VyLWlkMiIKICBdCn0nCmBgYAo= readmeEtag: '"0d3c55869a0f236707e216f4567b0f34e0c8f512"' readmeLastModified: Tue, 26 Oct 2021 08:51:16 GMT repositoryId: 421329326 description: Conceptual demo of simple openapi3 using connexion for quick prototyping. created: '2021-10-26T07:45:44Z' updated: '2021-10-26T08:51:22Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: PexMor logo: https://avatars.githubusercontent.com/u/9293161?v=4 license: MIT repoEtag: '"40f3c6e37e8d382ebc4b7ca36eb5b5b83af2698324663e50555e646f35c0c150"' repoLastModified: Tue, 26 Oct 2021 08:51:22 GMT foundInMaster: true category: Data Validators id: eabb62135bdb48adc356243dae621cd9 - source: openapi3 tags repository: https://github.com/txemac/intelistyle v3: true repositoryMetadata: base64Readme: >- IyBpbnRlbGlzdHlsZSB0YXNrOgpZb3VyIHRhc2sgaXMgdG8gYnVpbGQgYSAoc2luZ2xlIHBhZ2UpIHdlYnNpdGUgdGhhdCBJIGNhbiBzZWFyY2ggZm9yIGEgZ2FybWVudCAoZS5nLiBibGFjayBoYXQpIGFuZCBpdCBzaG91bGQgZGlzcGxheSB0aGUgZ2FybWVudHMgdGhhdCBtYXRjaCB0aGUgc2VhcmNoIGNyaXRlcmlhLgoKIyMgVGhlIHNvbHV0aW9uOgpUaGUgd2Vic2l0ZSBzaG91bGQgbG9hZCBnYXJtZW50cyBmcm9tIGEgTW9uZ29EQiBkYXRhYmFzZS4gWW91IGNhbiBsb2FkIGRhdGEgaW50byB5b3VyIGRhdGFiYXNlIGZyb20gdGhpcyBmaWxlOgoKaHR0cHM6Ly9zdHlsci1haS1lbmdpbmUtc3J2LWRhdGEuczMuZXUtd2VzdC0xLmFtYXpvbmF3cy5jb20vL3Nydi9kYXRhL25ld19zY3JhcGVzL3Nob3BzdHlsZS0xNjg5LW1lbi0xOC0wMy0yMDE5L2dhcm1lbnRfaXRlbXMuamwKCllvdSBoYXZlIHRoZSBmcmVlZG9tIHRvIHVzZSBhbnkgYnVpbGQgdG9vbGNoYWluIG9yIGhlbHBlciBsaWJyYXJpZXMgbmVjZXNzYXJ5LCBidXQgeW91IG11c3Qgc3RpY2sgdG8gb3VyIGNvcmUgdGVjaG5vbG9naWVzIG9mIFJlYWN0SlMgZm9yIHRoZSBmcm9udGVuZCBhbmQgTm9kZUpTIG9yIFB5dGhvbiBmb3IgdGhlIGJhY2tlbmQuCgpBcyBwYXJ0IG9mIHRoZSBkZWxpdmVyYWJsZSwgdGhlcmUgc2hvdWxkIGJlIGF1dG9tYXRlZCB0ZXN0cyBmb3IgYWxsIHRoZSBwb3NzaWJsZSB1c2UgY2FzZXMgb2YgdGhlIHRhc2suCgojIyBTdWNjZXNzIENyaXRlcmlhOgotIEFwcGxpY2F0aW9uIGFyY2hpdGVjdHVyZSBhbmQgZnJhbWV3b3JrIGJlc3QgcHJhY3RpY2VzIGZvciB0aGUgZnJhbWV3b3JrIGluIHVzZSBhcmUgZm9sbG93ZWQgYW5kIHVuZGVyc3Rvb2QuCi0gQ29kZSBpcyBmb3JtYXR0ZWQgd2VsbCBhbmQgZWFzeSB0byBmb2xsb3cuIFZhcmlhYmxlIGFuZCBmdW5jdGlvbiBuYW1lcyBtYWtlIHNlbnNlCi0gQXBwbGljYXRpb24gZ3JhY2VmdWxseSBoYW5kbGVzIGRhdGFiYXNlIGVycm9yIGNhc2VzIGFuZCBpcyByZXNpc3RhbnQgdG8gdW5leHBlY3RlZCBtZXNzYWdlcy4KLSBEYXRhYmFzZSBxdWVyaWVzIGFyZSBvcHRpbWlzZWQgZm9yIHBlcmZvcm1hbmNlCi0gVGVzdCBjb3ZlcmFnZSBvZiB0aGUgd2ViIHBhZ2UgaXMgc3VmZmljaWVudCBhbmQgdGhvdWdodCBoYXMgYmVlbiBwdXQgaW50byB3aGF0IGFyZWFzIG9mIHRoZSBhcHBsaWNhdGlvbiBzaG91bGQgYW5kCiAgc2hvdWxkbuKAmXQgYmUgdGVzdGVkLgoKIyMgQm9udXMgUG9pbnRzIChvcHRpb25hbCk6CgotIERlc2lnbiBhbiBpbmZyYXN0cnVjdHVyZSBhcmNoaXRlY3R1cmUgZGlhZ3JhbSB0aGF0IGNhbiBiZSB1c2VkIHRvIHNjYWxlIHRoZSB3ZWJzaXRlIHRvIHRob3VzYW5kcyBvZiBzZWFyY2hlcyBwZXIKICBzZWNvbmQKCiMjIFN1Ym1pc3Npb24gUmVxdWlyZW1lbnRzOgoKLSBTZW5kIGFuIGVtYWlsIHRvIHlvdXIgcmVjcnVpdGVyIHdpdGggYSBsaW5rIHRvIGEgcHVibGljIGdpdCByZXBvc2l0b3J5IChpbiBHaXRIdWIvQml0QnVja2V0L0dpdExhYiwgZXRjKSB3aXRoIHRoZQogIG5hbWUgW2ZpcnN0bmFtZV0tW3N1Ym1pc3Npb24gZGF0ZV0uCi0gUHJvdmlkZSBhIGxpbmsgdG8gYSBob3N0ZWQgdmVyc2lvbiBvZiB0aGUgcHJvamVjdAotIFlvdSB3aWxsIGJlIGFza2VkIHRvIHNjcmVlbiBzaGFyZSBhbmQgd2FsayB0aHJvdWdoIHRoaXMgYXBwICYgY29kZSBpbiB5b3VyIG5leHQgaW50ZXJ2aWV3LCBwbGVhc2UgaGF2ZSBpdCByZWFkeSB0byBiZQogIHJ1biBwcmlvciB0byB0aGUgaW50ZXJ2aWV3LgoKIyMgUnVuCgpTZXQgZW52aXJvbm1lbnQgdmFyaWFibGU6CgpgYGBzaGVsbCBzY3JpcHQKZXhwb3J0IE1PTkdPREJfVVJMPTxtb25nbyBkYiBVUkw+CmV4cG9ydCBNT05HT0RCX0RCX05BTUU9PG1vbmdvIGRiIGRhdGFiYXNlIG5hbWU+CmV4cG9ydCBNT05HT0RCX0NPTExFQ1RJT049Pm1vbmdvIGRiIGNvbGxlY3Rpb24+CmBgYAoKIyMjIFRlcm1pbmFsIHdpdGggdmlydHVhbCBlbnYKCklmIHlvdSB3YW50IHRvIHJ1biB0aGUgYXBwIGluIGEgdGVybWluYWwsIHdyaXRlOgoKYGBgc2hlbGwgc2NyaXB0CmNkIHNyYwp1dmljb3JuIG1haW46YXBwIC0tcmVsb2FkIC0td29ya2VycyAxIC0taG9zdCAwLjAuMC4wIC0tcG9ydCA4MDAwCmBgYAoKVGVzdDoKCmBgYHNoZWxsIHNjcmlwdApweXRlc3QgLXZ2dgpgYGAKCiMjIyBEb2NrZXIKClJ1bjoKCmBgYHNoZWxsIHNjcmlwdAptYWtlIHJ1bgpgYGAKClN0b3A6CgpgYGBzaGVsbCBzY3JpcHQKbWFrZSBzdG9wCmBgYAoKQ2hlY2sgdGhlIEFQSSB3aXRoIGh0dHA6Ly8xMjcuMC4wLjE6ODAwMC9oZWFsdGgKCiMjIERvY3VtZW50YXRpb24KCmh0dHA6Ly8xMjcuMC4wLjE6ODAwMC9kb2NzCg== readmeEtag: '"902d7b76650334d75ed4c6a4ac7e0dd7a7d05049"' readmeLastModified: Thu, 14 Jul 2022 09:09:38 GMT repositoryId: 423989830 description: FastAPI, mongoDB, docker, docker-compose, API created: '2021-11-02T20:30:18Z' updated: '2021-12-09T12:02:03Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: txemac logo: https://avatars.githubusercontent.com/u/10089585?v=4 license: GPL-3.0 repoEtag: '"e8f262e665424010f2635652280ce1ec1f3968f7bdfd2fcd33d278dbdc17e249"' repoLastModified: Thu, 09 Dec 2021 12:02:03 GMT foundInMaster: true category: Server Implementations id: 7cf19b0c1288631db49c5a9670fe1f64 - source: openapi3 tags repository: https://github.com/mdmuradhossain/modern-ecommerce v3: true repositoryMetadata: base64Readme: >- IyBNb2Rlcm4gRWNvbW1lcmNlCgojIyBNb2Rlcm4gRWNvbW1lcmNlIFdlYiBBcHBsaWNhdGlvbgotLS0KIyMjIFRlY2hub2xvZ3kgYW5kIFRvb2xzIFVzZWQKLSBbSkRLIDE1XShodHRwczovL3d3dy5vcmFjbGUuY29tL2phdmEvdGVjaG5vbG9naWVzL2phdmFzZS9qZGsxNS1hcmNoaXZlLWRvd25sb2Fkcy5odG1sKQotIFtTcHJpbmcgQm9vdCAtIDIuNS40XShodHRwczovL3NwcmluZy5pby9wcm9qZWN0cy9zcHJpbmctYm9vdCkKLSBbU3ByaW5nIEJvb3QgRGF0YSBKUEFdKGh0dHBzOi8vc3ByaW5nLmlvL3Byb2plY3RzL3NwcmluZy1kYXRhLWpwYSkKLSBbU3ByaW5nIFNlY3VyaXR5XShodHRwczovL3NwcmluZy5pby9wcm9qZWN0cy9zcHJpbmctc2VjdXJpdHkpCi0gW1RoeW1lbGVhZl0oaHR0cHM6Ly93d3cudGh5bWVsZWFmLm9yZy8pCi0gW0phdmEgTWFpbCBTZW5kZXJdKGh0dHBzOi8vZG9jcy5zcHJpbmcuaW8vc3ByaW5nLWZyYW1ld29yay9kb2NzL2N1cnJlbnQvamF2YWRvYy1hcGkvb3JnL3NwcmluZ2ZyYW1ld29yay9tYWlsL2phdmFtYWlsL0phdmFNYWlsU2VuZGVyLmh0bWwpCi0gW015U1FMIDhdKGh0dHBzOi8vd3d3Lm15c3FsLmNvbS8pCi0gW0xvbWJva10oaHR0cHM6Ly9wcm9qZWN0bG9tYm9rLm9yZy8pCi0gW01hcHN0cnVjdF0oaHR0cHM6Ly9tYXBzdHJ1Y3Qub3JnLykKLSBbSldUXShodHRwczovL2dpdGh1Yi5jb20vand0ay9qand0KQotIFtTdHJpcGVdKGh0dHBzOi8vc3RyaXBlLmNvbS9kb2NzKQotIFtGbHl3YXldKGh0dHBzOi8vZmx5d2F5ZGIub3JnLykKLSBbU3ByaW5nIEJvb3QgQ2FjaGVdKGh0dHBzOi8vc3ByaW5nLmlvL2d1aWRlcy9ncy9jYWNoaW5nLykgIAotIFtPcGVuQVBJIDNdKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykKLS0tCiMjIyBJbnN0YWxsYXRpb24KLSBgbXZuIGNsZWFuIGluc3RhbGxgCi0gYG12biBzcHJpbmctYm9vdDpydW5gCi0tLQojIyMgRmVhdHVyZXMKLSBbeF0gU2lnbiBVcCBhbmQgU2lnbiBJbiBTeXN0ZW0gCi0gW3hdIEFkbWluIGFuZCBVc2VyIHN5c3RlbQotIFt4XSBBZGQgQWRtaW4gYW5kIFJvbGUgYW5kIFByaXZpbGVnZQotIFt4XSBNYW5hZ2UgUHJvZHVjdHMgZnJvbSBBZG1pbgotIFt4XSBNYW5hZ2UgQ2F0ZWdvcnkgZnJvbSBBZG1pbgotIFt4XSBNYW5hZ2UgU3ViQ2F0ZWdvcnkgZnJvbSBBZG1pbgotIFt4XSBNYW5hZ2UgQnJhbmQgZnJvbSBBZG1pbgotIFt4XSBNYW5hZ2UgT3JkZXJzCi0gW3hdIE1hbmFnZSBDYXJ0cwotIFt4XSBNYW5hZ2UgT3JkZXIgSXRlbQotIFt4XSBTdHJpcGUgUGF5bWVudCBTeXN0ZW0KCi0tLQoKIyMjIFN3YWdnZXIgVUkKIVtBUElzXShodHRwczovL2dpdGh1Yi5jb20vbWRtdXJhZGhvc3NhaW4vbW9kZXJuLWVjb21tZXJjZS9ibG9iL21haW4vc3JjL21haW4vcmVzb3VyY2VzL3N0YXRpYy9pbWFnZXMvc3dhZ2dlci11aS11cGRhdGUucG5nKQotLS0KIyMjIFBvc3RtYW4gQVBJIFRlc3RpbmcKIVtQb3N0bWFuIHRlc3RpbmddKGh0dHBzOi8vZ2l0aHViLmNvbS9tZG11cmFkaG9zc2Fpbi9tb2Rlcm4tZWNvbW1lcmNlL2Jsb2IvbWFpbi9zcmMvbWFpbi9yZXNvdXJjZXMvc3RhdGljL2ltYWdlcy9hdXRocmVzcG9uc2UucG5nKQoKLS0tCiMjIyBDb250cmlidXRlCj4gVGhpcyBpcyBvcGVuIHNvdXJjZSBNb2Rlcm4gRWNvbW1lcmNlIHdlYiBhcHBsaWNhdGlvbiwgYW55IGtpbmQgY29udHJpYnV0aW9uIGlzIHdlbGNvbWUuIEkgZGlkbid0IHRlc3QgdGhlIGFwcGxpY2F0aW9uIHlldC4gQ29udHJpYnV0aW9uIG9mIHRlc3RpbmcgY2FuIGJlIGRvbmUgd2l0aCBKdW5pdCBhbmQgTW9ja2l0by4gVGhhbmsgeW91Lg== readmeEtag: '"379b6aeb43530f4868458b9dac091b8cb30da61c"' readmeLastModified: Tue, 21 Dec 2021 15:45:01 GMT repositoryId: 398491331 description: Modern E-commerce Web Application created: '2021-08-21T07:07:38Z' updated: '2023-02-17T08:31:51Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: mdmuradhossain logo: https://avatars.githubusercontent.com/u/20659502?v=4 repoEtag: '"7e82d95b79d87cea1dbf9348165e8a47cc09b8dc03e30b4d6eeddb249bb7cd47"' repoLastModified: Fri, 17 Feb 2023 08:31:51 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: d448d1c32de34d2cdac394afc2b94c90 - source: openapi3 tags repository: https://github.com/umatare5/logbook-api-router-impl v3: true repositoryMetadata: base64Readme: >- IyBsb2dib29rLWFwaS1yb3V0ZXItaW1wbAoKKipETyBOT1QgRURJVCBUSElTIENPREUgTUFOVUFMTFkhKioKCmxvZ2Jvb2stYXBpLWZyYW1ld29yay1pbXBsIGlzIEdvIGltcGxlbWVudGF0aW9uIGZvciBbTG9nYm9vayBBUEldKGh0dHBzOi8vZ2l0aHViLmNvbS91bWF0YXJlNS9sb2dib29rLWFwaSkuCgpUaGUgY29kZSB3YXMgZ2VuZXJhdGVkIGF1dG9tYXRpY2FsbHkgYnkgdXNpbmcgW2xvZ2Jvb2stb3BlbmFwaS1zcmNdKGh0dHBzOi8vZ2l0aHViLmNvbS91bWF0YXJlNS9sb2dib29rLW9wZW5hcGktc3JjKS4KCiMjIFVzYWdlCgpgYGBzaApnbyBnZXQgZ2l0aHViLmNvbS91bWF0YXJlNS9sb2dib29rLWFwaS1yb3V0ZXItaW1wbApgYGAK readmeEtag: '"9f238625480e32825ba8316c2b7e296a8541dcb2"' readmeLastModified: Wed, 26 Jan 2022 17:19:41 GMT repositoryId: 449646146 description: Implementation for routes in my Logbook API created: '2022-01-19T10:29:45Z' updated: '2022-02-15T13:53:41Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: umatare5 logo: https://avatars.githubusercontent.com/u/72645163?v=4 repoEtag: '"d6fe8a2eaf9af3df3e63d0bc10ded6fe46075d37d11413f7c17dd750843588b8"' repoLastModified: Tue, 15 Feb 2022 13:53:41 GMT foundInMaster: true category: Code Generators id: 599d3436d37e07d48888ad50b21f2473 - source: https://openapi.tools/ name: EvoMaster category: Testing language: - Java - Kotlin repository: https://github.com/webfuzzing/evomaster source_description: >- A tool for automatically generating system-level test cases for RESTful APIs, using Evolutionary Algorithms and Dynamic Program Analysis. v2: true v3: true repositoryMetadata: base64Readme: >- # EvoMaster: A Tool For Automatically Generating System-Level Test Cases


![](docs/img/carl-cerstrand-136810_compressed.jpg  "Photo by Carl Cerstrand on Unsplash")

[![Maven Central](https://img.shields.io/maven-central/v/org.evomaster/evomaster-client-java.svg?label=Maven%20Central)](https://central.sonatype.com/artifact/org.evomaster/evomaster-client-java)
[![javadoc](https://javadoc.io/badge2/org.evomaster/evomaster-client-java-controller/javadoc.svg)](https://javadoc.io/doc/org.evomaster/evomaster-client-java-controller)
![CI](https://github.com/WebFuzzing/EvoMaster/workflows/CI/badge.svg)
[![codecov](https://codecov.io/gh/WebFuzzing/EvoMaster/branch/master/graph/badge.svg)](https://codecov.io/gh/WebFuzzing/EvoMaster)
[![DOI](https://zenodo.org/badge/DOI/10.5281/zenodo.5052036.svg)](https://doi.org/10.5281/zenodo.5052036)
[![License: LGPL v3](https://img.shields.io/badge/License-LGPL_v3-blue.svg)](https://www.gnu.org/licenses/lgpl-3.0)
[![Github All Releases](https://img.shields.io/github/downloads/WebFuzzing/evomaster/total.svg)](https://github.com/WebFuzzing/EvoMaster/releases)


### Summary

_EvoMaster_ ([www.evomaster.org](http://evomaster.org)) is the first (2016) open-source AI-driven tool
that automatically *generates* system-level test cases
for web/enterprise applications.
This is related to [Fuzzing](https://en.wikipedia.org/wiki/Fuzzing).
In particular, _EvoMaster_ can fuzz APIs such as REST, GraphQL and RPC.
Not only _EvoMaster_ can generate inputs that find program crashes, but also it generates small effective test suites (e.g., in Python, JavaScript and Java/Kotlin JUnit format) that can be used for _regression testing_.

_EvoMaster_ is an AI driven tool.
In particular, internally it uses an [Evolutionary Algorithm](https://en.wikipedia.org/wiki/Evolutionary_algorithm)
and [Dynamic Program Analysis](https://en.wikipedia.org/wiki/Dynamic_program_analysis)  to be
able to generate effective test cases.
The approach is to *evolve* test cases from an initial population of
random ones, trying to maximize measures like code coverage and fault detection.
_EvoMaster_ uses several kinds of AI heuristics to improve performance even further,
building on decades of research in the field of [Search-Based Software Testing](https://en.wikipedia.org/wiki/Search-based_software_engineering).


### 1-Minute Example 

On a console, copy&paste the following (requires _Docker_ installed).
It will fuzz the PetClinic example API from Swagger, for 30 seconds, as shown in the following video. 

```
docker run -v "$(pwd)/generated_tests":/generated_tests webfuzzing/evomaster  --blackBox true --maxTime 30s  --ratePerMinute 60 --bbSwaggerUrl  https://petstore.swagger.io/v2/swagger.json
```


![](docs/img/evomaster_docker_use.gif)

#### Using Docker on Different Shells

Note that, depending on which shell and operating system you are using, you might need slightly different commands when mounting folders with the `-v` option.  

For example, if run in a MSYS shell on Windows like _Git Bash_, there is the need of an extra / before the $.

```
docker run -v "/$(pwd)/generated_tests":/generated_tests webfuzzing/evomaster  --blackBox true --maxTime 30s  --ratePerMinute 60 --bbSwaggerUrl  https://petstore.swagger.io/v2/swagger.json
```

If you are rather using a Command Prompt (Cmd.exe) terminal, you need to use `%CD%` instead of `$(pwd)` to refer to the current folder:

```
docker run -v %CD%/generated_tests:/generated_tests webfuzzing/evomaster  --blackBox true --maxTime 30s  --ratePerMinute 60 --bbSwaggerUrl  https://petstore.swagger.io/v2/swagger.json
```

On the other hand, on a PowerShell you need `${PWD}`:

```
docker run -v ${PWD}/generated_tests:/generated_tests webfuzzing/evomaster  --blackBox true --maxTime 30s  --ratePerMinute 60 --bbSwaggerUrl  https://petstore.swagger.io/v2/swagger.json
```

#### Troubleshooting

If you encounter issues running the command:

* Ensure Docker is installed and running.
* Check that you have the correct rights/permissions to mount the specified volume.
* Consult the Docker documentation for your shell environment for specific syntax requirements.


#### Generated Output

Once the command is executed, you can inspect the generated files under `generated_tests` folder. 

Note, since version 4.0.0, now _EvoMaster_ by default also creates an interactive web report.

![](docs/img/report_overview.png)

![](docs/img/report_endpoints.png)

### Key features

* _Web APIs_: At the moment, _EvoMaster_ can generate test cases for __REST__, __GraphQL__ and __RPC__ (e.g., __gRPC__ and __Thrift__) APIs.

* _Free_: this is an open-source project, with funding from public research (e.g., professors, postdocs and PhD students).
          There is no monetary cost involved in using this tool, or need to use any paid external services (e.g., external LLM APIs).
          If you are a resident in EU, Norway, Argentina, China, or any associate country involved with ERC, thanks for your tax money supporting this project. 

* _In-house, no telemetry_: currently there is no telemetry in use in _EvoMaster_, and it does not require to connect to any external service on internet, besides the tested application. AFAIK, it can be run in-house without any worry of leaking any IP to external parties (if not, please open a new issue to tell us how that could happen, and we will try to fix it). 

* _Black-Box_ testing mode: can run on any API (regardless of its programming language, e.g., Python and Go).
  However, results for black-box testing will be worse than white-box testing (e.g., due to lack of code analysis).
  Default test case output is in Python, but other formats are available as well. 

* _White-Box_ testing mode: can be used for APIs compiled to
  JVM (e.g., Java and Kotlin). _EvoMaster_ analyses the bytecode of the tested applications, and uses
  several heuristics such as _testability transformations_ and _taint analysis_ to be able to generate
  more effective test cases. We support JDK __8__ and the major LTS versions after that (currently JDK __21__). Might work on other JVM versions, but we provide __NO__ support for it.
  Note: there was initial support for other languages as well, like for example JavaScript/TypeScript and C#, but they were not in a stable, feature-complete state. The support for those languages for white-box testing has been dropped, at least for the time being. 

* _Installation_: we provide installers for the main operating systems: _Windows_ (`.msi`),
  _OSX_ (`.dmg`) and _Linux_ (`.deb`). We also provide an uber-fat JAR file.
  To download them, see the [Release page](https://github.com/WebFuzzing/EvoMaster/releases).
  Release notes are present in the file [release_notes.md](https://github.com/WebFuzzing/EvoMaster/blob/master/release_notes.md).
  If you are using the uber-fat JAR, it should work with any major LTS version (from JDK 8 on).
   Whereas for the client library, needed for white-box testing, we will support JDK 8 likely for a long, long while, be warned that future versions of the executable JAR might start to require higher versions of the JDK in a non-so-distant future.
   If that is going to be higher than your current version of the JVM, if you cannot upgrade or have 2 different JDKs on your machine, then you should not use the uber-jar but rather one of the installers. 
   When you use one of the installers, keep in mind that currently they do not update the `PATH` variable. This needs to be done manually, [see documentation](docs/download.md). 
   Also keep in mind we have not paid the [Microsoft/Apple Tax](docs/download.md). This means that your operating system by default will block the installation, stating it cannot verify it is not a malware. 
   But the block [can be bypassed](docs/download.md).

* _Docker_: _EvoMaster_ is now released via Docker as well, under [webfuzzing/evomaster](https://hub.docker.com/r/webfuzzing/evomaster) on Docker Hub. For more information on how to use _EvoMaster_ via Docker, [see documentation](docs/docker.md).

* _GitHub Action_: it is possible to run _EvoMaster_ in GitHub Actions, as part of Continuous Integration, by using the [following custom action](https://github.com/WebFuzzing/evomaster-action) (which is in a different GitHub repository).

* _Hardware_: although state-of-the-art AI techniques are used, there is no major hardware requirement to be able to run _EvoMaster_. It will work even on old laptops. The main computational bottleneck is running the tested applications, and making network calls to them. 

* _State-of-the-art_: an [independent study (2022)](https://arxiv.org/abs/2204.08348), comparing 10 fuzzers on 20 RESTful APIs, shows that _EvoMaster_ gives the best results. Another [independent study (2024)](https://arxiv.org/abs/2410.12547) done by a different research group confirms these results.

* _Schema_: REST APIs must provide a schema in [OpenAPI/Swagger](https://swagger.io)
  format (either _v2_ or _v3_).

* _Output_: the tool generates _JUnit_ (version 4 or 5) tests, written in either Java or Kotlin, as well as test suites in Python and JavaScript. For a complete list, see the documentation for the CLI parameter [--outputFormat](docs/options.md). 
  Some examples are: PYTHON_UNITTEST, KOTLIN_JUNIT_5, JAVA_JUNIT_4 and JS_JEST.
  Note that the generated tests rely on third-party libraries (e.g., to make HTTP calls). 
  These will need to be setup in your projects, [see documentation](docs/library_dependencies.md).

* _Web Report_: besides generating executable tests in different programming language, an interactive _index.html_ web report is created as well by default, visualizing and summarizing the results of the generated tests. 

* _Fault detection_: _EvoMaster_ can generate tests cases that reveal faults/bugs in the tested applications.
  Different heuristics are employed, like checking for 500 status codes, mismatches from the API schemas and access policy violations.

* _Self-contained tests_: for white-box testing, the generated tests do start/stop the application, binding to an ephemeral port.
  This means that the generated tests can be used for _regression testing_ (e.g., added to the Git repository
  of the application, and run with any build tool such as Maven and Gradle).
  For black-box testing, you will need to make sure the application is up and running before executing the tests. 

* _Database handling_: for white-box testing, _EvoMaster_ can intercept and analyse all communications done with SQL and MongoDB databases, and use
  such information to generate higher code coverage test cases. Furthermore, it can generate data directly
  into the databases, and have such initialization automatically added in the generated tests.
  At the moment, _EvoMaster_ supports _Postgres_, _MySQL_, _H2_  and MongoDB  databases.

* _Authentication_: we support auth based on authentication headers and cookies.
  Besides using fixed HTTP headers, 
  it is also possible to declaratively specify which login endpoint should be used to dynamically obtain authentication info (e.g., auth tokens or cookies) for each test execution. [See documentation](docs/auth.md).

### Known Limitations

* _Driver_: to be used for _white-box_ testing, users need to write a [driver manually](docs/write_driver.md).
  We recommend to try _black-box_ mode first (should just need a few minutes to get it up and running) to get
  an idea of what _EvoMaster_ can do for you.

* _JDK 9+_: white-box testing requires bytecode manipulation. 
            Each new release of the JDK makes doing this harder and harder. 
            Dealing with JDKs above __8__ is doable, but it requires some settings.
            [See documentation](docs/jdks.md).

* _Execution time_: to get good results, you might need to run the search for several hours.
  We recommend to first try the search for 10 minutes, just to get an idea of what type of tests can be generated.
  But, then, you should run _EvoMaster_ for something like between 1 and 24 hours (the longer the better, but
  it is unlikely to get better results after 24 hours).

* _RPC APIs_: for the moment, we do not directly support RPC schema definitions. Fuzzing RPC APIs requires to write a driver, using the client library of the API to make the calls.

* _External services_: (e.g., other RESTful APIs) currently there is no support for them (e.g., to automatically mock them).
  It is work in progress.

* _Failing tests_: the tests generated by _EvoMaster_ should all pass, and not fail, even when they detect a fault.
  In those cases, comments/test-names would point out that a test is revealing a possible fault, while still passing.
  However, in some cases the generated tests might fail. This is due to the so called _flaky_ tests, e.g., when
  a test has assertions based on the time clock (e.g., dates and timestamps).
  There is ongoing effort to address this problem, but it is still not fully solved.

<!--### Videos---> 
<!-- 
<div>Icons made by <a href="https://www.flaticon.com/authors/freepik" title="Freepik">Freepik</a> from <a href="https://www.flaticon.com/" title="Flaticon">www.flaticon.com</a></div> 
-->


### Use in Industry

Several enterprises use _EvoMaster_ to fuzz their Web APIs.
We do few academia-industry collaborations ([see more info here](docs/contribute.md)), where we help test engineers to apply _EvoMaster_ on their systems, as long as we can then report on such experience.
Examples of Fortune 500 companies using _EvoMaster_ are:

* [Meituan](https://www.meituan.com): see [TOSEM'23](docs/publications/2023_tosem_rpc.pdf), [ASE'24](docs/publications/2024_ase.pdf), [SCP'25](docs/publications/2025_scp.pdf).

* [Volkswagen](https://www.volkswagen.com): see [AUSE'24](docs/publications/2024_ause_vw.pdf), [ICST'25](docs/publications/2025_icst.pdf).


### Videos

![](docs/img/video-player-flaticon.png)

* A [45-minute talk given at TestCon'25](https://www.youtube.com/watch?v=uKKRo3LrNiw&list=PLqYhGsQ9iSEoXaRmW9WQjjXJK_1NbLlZ6&index=15) on Fuzz Testing Web APIs gives an overview of what can be expected from this kind of fuzzers. 
 A [shorter version (16 minutes)](https://www.youtube.com/watch?v=iJdhVzGedjM) 
  was given at Nordic APIs 2025 Platform Summit.   

* A [short video](https://youtu.be/3mYxjgnhLEo) (5 minutes)
  shows the use of _EvoMaster_ on one of the
  case studies in [EMB](https://github.com/WebFuzzing/EMB).

* This [13-minute video](https://youtu.be/ORxZoYw7LnM)
  shows how to write a white-box driver for EvoMaster, for the
  [rest-api-example](https://github.com/WebFuzzing/rest-api-example).

* How to [Download and Install EvoMaster on Windows 10](https://youtu.be/uh_XzGxws9o), using its _.msi_ installer.

* [Short presentation](https://youtu.be/iQSAlrr-PZo) (5 minutes) about version 2.0.0. 

* [Demonstration of Docker and GitHub Actions support](https://youtu.be/l1ybs7SjvcA).

### Alternatives

In the last few years, several few tools have been proposed in the academic literature and in the open-source community.
You can read more details in this [2023 survey](docs/publications/2023_tosem_survey.pdf) on REST API testing.

Existing open-source tools for REST API fuzzing, with at least 100 stars on GitHub, are for example (in alphabetic order):
[CATS](https://github.com/Endava/cats),
[Dredd](https://github.com/apiaryio/dredd),
[Fuzz-lightyear](https://github.com/Yelp/fuzz-lightyear),
[ResTest](https://github.com/isa-group/RESTest),
[Restler](https://github.com/microsoft/restler-fuzzer),
[Schemathesis](https://github.com/schemathesis/schemathesis)
and
[WuppieFuzz](https://github.com/TNO-S3/WuppieFuzz).

Apart from WuppieFuzz, all these tools are _black-box_, i.e., they do not analyze the source-code of the tested APIs to generate more effective test data.
As we are the authors of EvoMaster, we are too biased to compare it properly with those other black-box tools.
However, different independent studies (e.g., in [2022](https://arxiv.org/abs/2204.08348) and [2024](https://arxiv.org/abs/2410.12547)) shows that EvoMaster is among the best performant.
Furthermore, if your APIs are running on the JVM (e.g., written in Java or Kotlin), then EvoMaster has clearly an advantage, as it supports _white-box_ testing. 

<!---
### Hiring

Depending on the year, we might have funding for _postdoc_ and _PhD student_ positions to work on this project (in Oslo, Norway).

Current open positions: none.
< !---
* 2023: PhD student positions. No new calls scheduled for the moment.
* 2023: Postdoc positions. No new calls scheduled for the moment.
--- >

For questions on these positions, please contact Prof. Andrea Arcuri.

< !---
For more details on current vacancies, see our group page at [AISE Lab](https://WebFuzzing.github.io/).
--->



### Documentation

If you are trying to use _EvoMaster_, but the instructions in this documentation are not enough to get you started, or they are too unclear, then it means it is a _bug_ in the documentation, which then would need to be clarified and updated. In such cases, please create a new [issue](https://github.com/WebFuzzing/EvoMaster/issues).

Also, feel free to start new discussion topics in the [Discussions forum](https://github.com/WebFuzzing/EvoMaster/discussions).
If you have time, please consider answering the polls there.

If you are working on an open-source API, you can drop us a message if you have problems using EvoMaster on it. 
Otherwise, if you are working in industry on closed-source APIs, we have options for academia-industry collaborations ([see more info here](docs/contribute.md)).  


* [Example of generated tests](docs/example.md)
* [Download and Install EvoMaster](docs/download.md)
* [Build EvoMaster from source](docs/build.md)
* [Command-Line Interface (CLI) options](docs/options.md)
* [OpenApi/Swagger Schema](docs/openapi.md)
* [Using EvoMaster for Black-Box Testing (easier to setup, but worse results)](docs/blackbox.md)
* [Using EvoMaster for White-Box Testing (harder to setup, but better results)](docs/whitebox.md)
  * [Write an EvoMaster Driver for White-Box Testing](docs/write_driver.md)
  * [Dealing with JDKs above version 8](docs/jdks.md)
* [Configuring authentication](docs/auth.md)  
* [Console output](docs/console_output.md)
* [Library dependencies for the generated tests](docs/library_dependencies.md)
* [How to contribute](docs/contribute.md)
  * [Technical notes for developers contributing to EvoMaster](docs/for_developers.md)
* Troubleshooting
  * [Windows and networking](docs/troubleshooting/windows.md)
  * [java.lang.OutOfMemoryError](docs/troubleshooting/outofmemory.md)
* More Info
  * [Academic papers related to EvoMaster](docs/publications.md)
  * [Slides of presentations/seminars](docs/presentations.md)
  * [Replicating studies](docs/replicating_studies.md)






### Funding

_EvoMaster_ has been funded by:
* 2020-2026: a 2 million Euro grant by the European Research Council (ERC),
  as part of the *ERC Consolidator* project
  <i>Using Evolutionary Algorithms to Understand and Secure Web/Enterprise Systems</i>.
*  2018-2021: a 7.8 million Norwegian Kroner grant  by the Research Council of Norway (RCN),
   as part of the Frinatek project <i>Evolutionary Enterprise Testing</i>.


<img src="https://github.com/WebFuzzing/EvoMaster/blob/master/docs/img/LOGO_ERC-FLAG_EU_.jpg?raw=true" width="200" >


This project has received funding from the European Research Council (ERC) under the European Union’s Horizon 2020 research and innovation programme (grant agreement No 864972).


### License
_EvoMaster_'s source code is released under the LGPL (v3) license.
For a list of the used third-party libraries, you can directly see the root [pom.xml](./pom.xml) file.
For a list of code directly imported (and then possibly modified/updated) from
other open-source projects, see [here](./docs/reused_code.md).


<!--
### ![](https://www.yourkit.com/images/yklogo.png)

YourKit supports open source projects with its full-featured Java Profiler.
YourKit, LLC is the creator of
<a href="https://www.yourkit.com/java/profiler/">YourKit Java Profiler</a>
and
<a href="https://www.yourkit.com/.net/profiler/">YourKit .NET Profiler</a>,
innovative and intelligent tools for profiling Java and .NET applications.
-->

 readmeEtag: '"8f023e805a95729ba3a9cc127cdd9c9bd1fe80b9"' readmeLastModified: Thu, 29 Jan 2026 10:49:01 GMT repositoryId: 92385933 description: >- The first open-source AI-driven tool for automatically generating system-level test cases (also known as fuzzing) for web/enterprise applications. Currently targeting whitebox and blackbox testing of Web APIs, like REST, GraphQL and RPC (e.g., gRPC and Thrift). created: '2017-05-25T09:11:23Z' updated: '2026-02-05T18:51:57Z' language: Kotlin archived: false stars: 667 watchers: 21 forks: 103 owner: WebFuzzing logo: https://avatars.githubusercontent.com/u/29148026?v=4 license: LGPL-3.0 repoEtag: '"945ec61f110536bc67c74528aac158f0429646d120ad31b8978324694b3a0822"' repoLastModified: Thu, 05 Feb 2026 18:51:57 GMT foundInMaster: true id: d1d3d85149c4a9437dbdbf157cf3d7c7 oldLocations: - https://github.com/emresearch/evomaster - source: openapi3 tags repository: https://github.com/tiblazy/escribo.inovation.teste_2 v3: true id: 86a37bb88c331b52e3d21038f42efc17 repositoryMetadata: repositoryId: 722276761 description: escribo.inovation database created: '2023-11-22T19:52:38Z' updated: '2024-06-12T06:32:19Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: tiblazy logo: https://avatars.githubusercontent.com/u/96823878?v=4 repoEtag: '"f00538c571b45408cdc41bfc6f90435d8087b49345edbec9903b1c5f7d8939da"' repoLastModified: Wed, 12 Jun 2024 06:32:19 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/nakajima97/letter-box-api v3: true repositoryMetadata: base64Readme: >- IyDjgqLjg5fjg6rmpoLopoEKCiMjIExldHRlciBCb3gg44Gr44Gk44GE44GmCgrpoaflrqLjgYzlvpPmpa3lk6Hjgavlr77jgZfjgabljL/lkI3jgafjg6Hjg4Pjgrvjg7zjgrjjgpLpgIHjgozjgovjgqLjg5fjg6rjgIIgIArjg6Hjg4Pjgrvjg7zjgrjjgpLjgoTjgorlj5bjgorjgZnjgovmqZ/og73jga/jgarjgYTjgYwgIArjg63jgrDjgqTjg7PkuI3opoHjgafoqrDjgavpgIHjgovjga7jgYvjgYxVUkzjgavlj43mmKDjgZXjgozjgovjgIIgIArjgZ3jga7jgZ/jgoHjgIFRUuOCs+ODvOODieOCkuiqreOBv+WPluOCi+OBoOOBkeOBp+ODoeODg+OCu+ODvOOCuOOCkuiqsOOBq+mAgeOCi+OBruOBi+aMh+WumuOBp+OBjeOCi+OAgiAgCgojIyBMZXR0ZXIgQm94IOOCkuani+aIkOOBmeOCi+imgee0oAoKTGV0dGVyIEJveCDjga/ku6XkuIs044Gk44Gu44Oq44Od44K444OI44Oq44Gn5qeL5oiQ44GV44KM44KLCgojIyMg5a6i5YG044OV44Ot44Oz44OI44Ko44Oz44OJCgrpoaflrqLjgYzjg6Hjg4Pjgrvjg7zjgrjjgpLkvJ3jgYjjgovpmpvjgavliKnnlKjjgZnjgovnlLvpnaIgIApodHRwczovL2dpdGh1Yi5jb20veXVkYWktbmFrYWppbWEvbGV0dGVyLWJveC1mcm9udGVuZC1jbGllbnQKCiMjIyDnrqHnkIbnlLvpnaLjg5Xjg63jg7Pjg4jjgqjjg7Pjg4kKCuW+k+alreWToeOBjOODoeODg+OCu+ODvOOCuOOCkueiuuiqjeOBmeOCi+OBn+OCgeOBrueUu+mdoiAgCmh0dHBzOi8vZ2l0aHViLmNvbS95dWRhaS1uYWthamltYS9sZXR0ZXItYm94LWZyb250ZW5kLW1hbmFnZW1lbnQvYmxvYi9tYWluL1JFQURNRS5tZAoKIyMjIOODkOODg+OCr+OCqOODs+ODiQoKTGV0dGVyIEJveCDjga7jg5Djg4Pjgq/jgqjjg7Pjg4kK44OV44Ot44Oz44OI44Ko44Oz44OJ44Gv5a6i5YG044O7566h55CG55S76Z2i44Go5YiG44GL44KM44KL44GM44CB44OQ44OD44Kv44Ko44Oz44OJ44GvIDEg44Gk44Gu44G/ICAKaHR0cHM6Ly9naXRodWIuY29tL3l1ZGFpLW5ha2FqaW1hL2xldHRlci1ib3gtYmFja2VuZAoKIyMjIEFQSSDku5Xmp5jmm7gK4oC75pys44Oq44Od44K444OI44OqICAKaHR0cHM6Ly9naXRodWIuY29tL3l1ZGFpLW5ha2FqaW1hL2xldHRlci1ib3gtYXBpCgojIOS9nOaIkOebrueahArlupflk6Hjgavlr77jgZfjgabmsJfou73jgavjg6Hjg4Pjgrvjg7zjgrjjgpLpgIHjgozjgovjgqLjg5fjg6rjgYzjgYLjgaPjgZ/jgokK5oSf6Kyd44Gu5rCX5oyB44Gh44Go44GL5Lyd44GI44KE44GZ44GE44KI44Gq44Go5oCd44Gj44Gf44Gf44KB44CCCgojIyMgQVBJIOS7leanmOabuArigLvmnKzjg6rjg53jgrjjg4jjg6oKCiMg5L2c5oiQ55uu55qECuW6l+WToeOBq+WvvuOBl+OBpuaEn+isneOCkuawl+i7veOBq+S8neOBiOOCi+aJi+auteOCkuaPkOS+m+OBmeOCi+OCouODl+ODquOCkumWi+eZuuOBl+OBn+OBhOOBqOaAneOBo+OBn+OBn+OCgQo= readmeEtag: '"51dbd667ab4953f28dcd9be7f070cd3667368c0a"' readmeLastModified: Wed, 22 Sep 2021 02:26:07 GMT repositoryId: 380894274 description: letter-boxで開発するAPIの仕様書 created: '2021-06-28T03:27:50Z' updated: '2023-06-12T11:21:48Z' language: null archived: true stars: 0 watchers: 1 forks: 0 owner: nakajima97 logo: https://avatars.githubusercontent.com/u/25247800?v=4 repoEtag: '"99dea8ead526222bd24be5921eec2c1ef0823ee9fae38c398a637c2be77d67e5"' repoLastModified: Mon, 12 Jun 2023 11:21:48 GMT foundInMaster: true category: Testing id: c40c4729753bf9c561b875f1151bcee8 oldLocations: - https://github.com/yudai-nakajima/letter-box-api - source: openapi3 tags repository: https://github.com/hikmatullahehsan/wefox-node-js-assessment v3: true repositoryMetadata: base64Readme: >- IyBXZWZveC1ub2RlLWpzLWFzc2Vzc21lbnQKQSBkZW1vIGFwcGxpY2F0aW9uIGRldmVsb3BlZCB3aXRoIE5vZGVKUwoKIyMgSG93IHRvIGluc3RhbGw/IApgYGBiYXNoCiQgbnBtIGluc3RhbGwgCmBgYAojIyBIb3cgdG8gcnVuPwpgYGBiYXNoCiQgbnBtIHN0YXJ0IApgYGAKCiMjIENyZWF0aW5nIGFuIC5lbnYgZmlsZSB3aXRoIHRoZXNlIHByb3BlcnRpZXMgYW5kIGluaXRpYWxpemUgdGhlaXIgdmFsdWVzIHJlbGF0ZWQgdG8geW91ciBlbnZpcm9ubWVudApgYGBqYXZhc2NyaXB0CkFQSV9QT1JUICAgICAgICAgICAgPSBudWxsCk1PTkdPX1VSSSAgICAgICAgICAgPSBudWxsClRPS0VOX0tFWSAgICAgICAgICAgPSBudWxsCk9QRU5fV0VBVEhFUl9BUElfSUQgPSBudWxsCmBgYAoKIyMgT3BlbkFQSSBTd2FnZ2VyIERvY3VtZW50YXRpb24gCmBgYGJhc2gKaHR0cDovL2xvY2FsaG9zdDozMDAwL2FwaS1kb2NzCmBgYAoKIyMgUmVxdWlyZW1lbnRzCiogTm9kZSB2MC4xMC4zMiBvciBhYm92ZQoqIEV4cHJlc3MgNCBvciBhYm92ZQoqIE1vbmdvREIKCiMjIFRlc3RpbmcKCiogYG5wbSBpbnN0YWxsYAoqIGBucG0gdGVzdAoKCiMjIERvY2tlciBJbWFnZSAKYGBgYmFzaApodHRwczovL2h1Yi5kb2NrZXIuY29tL3IvaGVrbWF0MjEvd2Vmb3gtbm9kZS1hc3Nlc3NtZW50CmBgYAoKCgoKCgoK readmeEtag: '"4db7e322fe5d0d37f998cc7f30ee2b218c23dcc0"' readmeLastModified: Sun, 05 Sep 2021 18:41:43 GMT repositoryId: 403301474 description: A general and simple NodeJS application for a demo purpose created: '2021-09-05T12:22:16Z' updated: '2021-09-05T18:41:46Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: HikmatullahEhsan logo: https://avatars.githubusercontent.com/u/23573173?v=4 repoEtag: '"a9fbbf92440465d2293a3f21be4d75ce47d337f9bd009dd48fc56137fc34c2ee"' repoLastModified: Sun, 05 Sep 2021 18:41:46 GMT foundInMaster: true category: - Documentation - Server Implementations id: 6fa7281b9b0763b069ffb1bd05e51341 - source: openapi3 tags repository: https://github.com/bharathbhargavgb/short-url v3: true repositoryMetadata: base64Readme: >- IyBzaG9ydC11cmwKQSBzaW1wbGUgVVJMIHNob3J0ZW5lciBpbXBsZW1lbnRlZCBpbiBHbyB1c2luZyBBV1MgTGFtYmRhIGFuZCBEeW5hbW9EQi4KClshW0FjdGlvbiBzdGF0dXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9iaGFyYXRoYmhhcmdhdmdiL3Nob3J0LXVybC93b3JrZmxvd3MvR28vYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2JoYXJhdGhiaGFyZ2F2Z2Ivc2hvcnQtdXJsL2FjdGlvbnMpCgojIyBPcGVuQVBJIDMuMCBzcGVjcwpodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzLWRvY3MvcmVzdXJnZW5jZS9zaG9ydGVuLzEuMC4wCg== readmeEtag: '"85efb9dc81c0411a0b4b552231dea96091e5c18c"' readmeLastModified: Sun, 13 Jun 2021 16:27:53 GMT repositoryId: 363583561 description: URL shortener using Go | AWS created: '2021-05-02T06:33:28Z' updated: '2021-06-13T16:28:05Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: bharathbhargavgb logo: https://avatars.githubusercontent.com/u/6514305?v=4 repoEtag: '"9f7898a93dbab194145d2d8102a4131f70a0ef64b6b5b3f00c6eb19889038785"' repoLastModified: Sun, 13 Jun 2021 16:28:05 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: d480043fd92044205b6775531c1bc411 - source: openapi3 tags repository: https://github.com/ivolimmen/apidoc v3: true repositoryMetadata: base64Readme: >- IyBBcGlEb2MKCkkgd2FzIG1pc3NpbmcgYSBzaW1wbGUgdG9vbCB0aGF0IGNhbiBjb252ZXJ0IGFuIE9wZW5BUEkgMyBKU09OIGZpbGUgdG8gYSBuaWNlIGxvb2tpbmcgZG9jdW1lbnQuIFNpbmNlIEkgbGlrZSBBc2NpaURvYyBJIGRlY2lkZWQgdG8gb3V0cHV0IGl0IHRvIHRoYXQgZm9ybWF0LgoKIyMgRXhhbXBsZQoKSWYgeW91IGRvd25sb2FkIHRoZSBPcGVuQVBJIHBldHN0b3JlIGV4bWFwbGUgYW5kIGNvbnZlcnQgaXQgdG8gQXNjaWlEb2MgdXNpbmcgdGhpcyB0b29sIHlvdSBnZXQgdGhlIGZvbGxvd2luZzoKCiFbcGV0c3RvcmUgZXhhbXBsZV0oZG9jcy9leGFtcGxlLW91dHB1dC5wbmcgIlBldHN0b3JlIGV4YW1wbGUiKQoKIyMgUmVxdWlyZW1lbnRzCgpNaW5pbWFsIHJlcXVpcmVtZW50czoKCiogSmF2YSAxMSBvciBoaWdoZXIKKiBNYXZlbiAzLjguKiBvciBoaWdoZXIKCiMjIEJ1aWxkaW5nIHRoZSBhcHBsaWNhdGlvbgoKUnVuOgoKICAgIG12biBjbGVhbiBpbnN0YWxsCgpUaGlzIHdpbGwgY3JlYXRlIGFuIGV4ZWN1dGFibGUgSkFSIGluIGBhcHAvdGFyZ2V0L2FwaWRvYy1hcHAtMC4xLmphcmAKCiMjIFJ1bm5pbmcgdGhlIGFwcGxpY2F0aW9uCgpBZnRlciBjb21waWxpbmcgeW91IGNhbiBydW46CgogICAgamF2YSAtamFyIGFwcC90YXJnZXQvYXBpZG9jLWFwcC0wLjEuamFyCgojIyMgQ29udmVydCBtdWx0aXBsZSBBUEkgSlNPTiBmaWxlcwoKVGhlIGFwcGxpY2F0aW9uIGNhbiBjb252ZXJ0IG11bHRpcGxlIEpTT04gZmlsZXMgaW4gb25lIGdvIGJ5IHNwZWZpY3lpbmcgb25lIG91dHB1dCBhbmQgbXVsdGlwbGUgaW5wdXQgZmlsZXMuCgogICAgamF2YSAtamFyIGFwaWRvYy1hcHAtMC4xLmphciAtLW91dHB1dD1+L0Rvd25sb2FkcyBkYXRhL2V4YW1wbGUxLmpzb24gZGF0YS9leGFtcGxlMi5qc29uIGRhdGEvZXhhbXBsZTMuanNvbiBkYXRhL2V4YW1wbGU0Lmpzb24KClRoaXMgd2lsbCBjcmVhdGU6IGBleGFtcGwxLmFkb2NgLCBgZXhhbXBsMi5hZG9jYCwgYGV4YW1wbDMuYWRvY2AgYW5kIGBleGFtcGw0LmFkb2NgIGluIGB+L0Rvd25sb2Fkc2AKClRoZSBhcHBsaWNhdGlvbiBjYW4gYWxzbyB0YWtlIGluIFVSTCdzOgoKICAgIGphdmEgLWphciBhcGlkb2MtYXBwLTAuMS5qYXIgLS1vdXRwdXQ9fi9Eb3dubG9hZHMgaHR0cHM6Ly9wZXRzdG9yZTMuc3dhZ2dlci5pby9hcGkvdjMvb3BlbmFwaS5qc29uCgpUaGlzIHdpbGwgY3JlYXRlIGBwZXRzdG9yZTMuYWRvY2AgYXMgZmlsZSBhcyBpdCB0YWtlcyB0aGUgaG9zdCBuYW1lIGFzIGZpbGUuCgpJZiB5b3UgbGlrZSB0byBjaGFuZ2UgdGhlIGZpbGUgbmFtZSB5b3UgY2FuIGRvIHRoZSBmb2xsb3dpbmc6CgogICAgamF2YSAtamFyIGFwaWRvYy1hcHAtMC4xLmphciAtLW91dHB1dD1+L0Rvd25sb2FkcyBodHRwczovL3BldHN0b3JlMy5zd2FnZ2VyLmlvL2FwaS92My9vcGVuYXBpLmpzb258b3V0cHV0Lmpzb24KClRoaXMgd2lsbCBjcmVhdGUgYG91dHB1dC5hZG9jYCBhcyBmaWxlLgoKVGhlc2UgYXJndW1lbnRzIGNhbiBiZSBjb21iaW5lZCBhcyB3ZWxsLgoKIyMgV2hhdCBkb2VzIHRoaXMgdG9vbCBjb252ZXJ0PwoKVGhlIGZvbGxvd2luZyBtb2RlbHMgZnJvbSBPcGVuQVBJIGFyZSBiZWluZyBjb252ZXJ0ZWQgdG8gQXNjaWlEb2M6CgoqIEluZm8KICAqIFRpdGxlCiAgKiBEZXNjcmlwdGlvbgogICogVmVyc2lvbgogICogQ29udGFjdCBpbmZvcm1hdGlvbgoqIFBhdGhzIChzb3J0ZWQhKQogICogUGVyIHBhdGggdGhlIG9wZXJhdGlvbnMgaWYgdGhleSBhcmUgcHJlc2VudCBpbiB0aGUgb3JkZXI6CiAgICAqIGRlbGV0ZQogICAgKiBnZXQKICAgICogaGVhZAogICAgKiBvcHRpb25zCiAgICAqIHBhdGNoCiAgICAqIHBvc3QKICAgICogcHV0CiAgICAqIHRyYWNlCiogUGVyIG9wZXJhdGlvbjoKICAqIHRoZSBjdXJyZW50IHBhdGgKICAqIHRoZSBjYWxsIChtZXRob2QgYW5kIHBhdGgpCiAgKiBJZiB0aGUgbWV0aG9kIGlzIGRlcHJlY2F0ZWQgKHdhcm5pbmcpCiAgKiBEZXNjcmlwdGlvbgogICogU3VtbWFyeQogICogUmVxdWVzdEJvZHkKICAgICogRGVzY3JpcHRpb24KICAgICogSWYgdGhlIGJvZHkgaXMgcmVxdWlyZWQKICAgICogU2hvd3Mgb2JqZWN0IHJlZmVyZW5jZXMgcGVyIG1lZGlhIHR5cGUKICAqIFBhcmFtZXRlcnMKICAgICogTGlzdHMgdHlwZSxuYW1lLGRlc2NyaXB0aW9uLHNjaGVtYSBhbmQgZGVmYXVsdAogICAgKiBBbHNvIGFkZHMgZXhhbXBsZXMgaW4gdGhlIGRlZmF1bHQgY29sdW1uCiAgKiBSZXNwb25zZXMKICAgICogU2hvd3Mgb2JqZWN0IHJlZmVyZW5jZXMgcGVyIG1lZGlhIHR5cGUKICAgICogQ3VycmVudGx5IGRvZXMgbm90IGxpc3QgYW5vbnltb3VzIHR5cGVzCgpUaGUgZm9sbG93aW5nIHN0dWZmIGlzIG5vdCAoeWV0PykgaGFuZGxlZDoKICAqIEV4dGVybmFsIGV4YW1wbGVzCiAgKiBIZWFkZXJzCiAgKiBTZWN1cml0eQogICogU2VydmVycwogICogVGFncw== readmeEtag: '"fe1ed7ca1ade0a6835fce2db966af39e34da3330"' readmeLastModified: Tue, 26 Oct 2021 15:21:25 GMT repositoryId: 419874172 description: Generate AsciiDoc using a OpenAPI 3.0 json file created: '2021-10-21T20:55:18Z' updated: '2023-09-14T08:55:44Z' language: Java archived: true stars: 0 watchers: 2 forks: 0 owner: IvoLimmen logo: https://avatars.githubusercontent.com/u/196375?v=4 repoEtag: '"5168081e7f2f4db9e95f0b530780480d6182ec0e5d80ccf60a9306a7c8dbab36"' repoLastModified: Thu, 14 Sep 2023 08:55:44 GMT foundInMaster: true category: Parsers id: 84079117079f066573106b694a50dd9b - source: openapi3 tags repository: https://github.com/umatare5/logbook-openapi-src v3: true repositoryMetadata: base64Readme: >- IyBsb2dib29rLW9wZW5hcGktc3JjCgpsb2dib29rLW9wZW5hcGktc3JjIGlzIGEgc291cmNlIGdlbmVyYXRlcyBsaWJyYXJpZXMgZm9yIFtMb2dib29rIEFQSV0oaHR0cHM6Ly9naXRodWIuY29tL3VtYXRhcmU1L2xvZ2Jvb2stYXBpKS4KClRoaXMgY29kZSBnZW5lcmF0ZXMgZm9sbG93aW5nIHR3byBtb2R1bGVzOwoKLSBbbG9nYm9vay1nb10oaHR0cHM6Ly9naXRodWIuY29tL3VtYXRhcmU1L2xvZ2Jvb2stZ28pCgogIEdvIENsaWVudCB0byB1c2UgTG9nYm9vayBBUEkuCgotIFtsb2dib29rLWFwaS1mcmFtZXdvcmstaW1wbF0oaHR0cHM6Ly9naXRodWIuY29tL3VtYXRhcmU1L2xvZ2Jvb2stYXBpLWZyYW1ld29yay1pbXBsKQoKICBJbXBsZW1lbnRhdGlvbiBmb3IgZnJhbWV3b3JrIGxheWVyIGluIExvZ2Jvb2sgQVBJLgoKIyMgRGV2ZWxvcG1lbnQKCiMjIyBTZXR1cAoKLSBJbnN0YWxsIGBzd2FnZ2VyLWNsaWAgZm9yIGJ1bmRsZSBkaXZpZGVkIE9wZW5BUEkgU3BlY2lmaWNhdGlvbnMuCgogIGBgYHNoCiAgbnBtIGluc3RhbGwKICBgYGAKCi0gSW5zdGFsbCBgb2FwaS1jb2RlZ2VuYCBmb3IgYnVpbGQgdGhlIGNsaWVudC4KCiAgYGBgc2gKICBnbyBpbnN0YWxsIGdpdGh1Yi5jb20vZGVlcG1hcC9vYXBpLWNvZGVnZW4vY21kL29hcGktY29kZWdlbkBsYXRlc3QKICBgYGAKCiMjIyBCdWlsZAoKYGBgc2gKbWFrZSBidWlsZApgYGAKCiMjIyBSZWxlYXNlCgpgYGBzaApnaXQgYnVtcAptYWtlIHJlbGVhc2UKYGBgCg== readmeEtag: '"afe21a21a6cae423c7612bfa5df62f99ddb28aa9"' readmeLastModified: Thu, 27 Jan 2022 11:41:41 GMT repositoryId: 449611422 description: Source of libraries to use Logbook API created: '2022-01-19T08:39:29Z' updated: '2022-02-15T13:53:10Z' language: Shell archived: false stars: 0 watchers: 1 forks: 0 owner: umatare5 logo: https://avatars.githubusercontent.com/u/72645163?v=4 repoEtag: '"7b869d90f4bd63c90bf217d047b9b5e7b6b74f15b7b949623fab5993a695aa66"' repoLastModified: Tue, 15 Feb 2022 13:53:10 GMT foundInMaster: true category: Code Generators id: 75906f5f428a76f4ed3ebdc42f111e54 - source: openapi3 tags repository: https://github.com/takeshi-kishima/map-experiment v3: true id: fc8c64bdcd6d624586eebbb056e26213 repositoryMetadata: base64Readme: >- IyBtYXAtZXhwZXJpbWVudAojIyMgRG9ja2Vy5YaF44Gn6ZaL55m644GZ44KL44Gq44KJCiogRG9ja2Vy55Kw5aKDCiogW1JlbW90ZSAtIENvbnRhaW5lcnNdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1tcy12c2NvZGUtcmVtb3RlLnJlbW90ZS1jb250YWluZXJzKQoK44KS5L2/44GE44G+44GZCgojIyMgW1JlbW90ZSAtIENvbnRhaW5lcnNdKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT1tcy12c2NvZGUtcmVtb3RlLnJlbW90ZS1jb250YWluZXJzKQpWU2NvZGXjga7mi6HlvLXmqZ/og73jgafjgZnjgIJGMSA+CmBgYApSZW1vdGUtQ29udGFpbmVyczogT3BlbiBGb2xkZXIgaW4gQ29udGFpbmVyLi4uCmBgYArjgafjgZPjga7jg5Xjgqnjg6vjg4DjgpLplovjgZHjgbDjgIFEb2NrZXLlhoXjgafplovnmbrnkrDlooPkvb/jgYjjgb7jgZnjgIIgIAoqIOOBk+OBruaLoeW8teapn+iDveOBruioreWumuODleOCoeOCpOODq+e+pOOBry5kZXZjb250YWluZXLjg5Xjgqnjg6vjg4DlhoXjgavlhaXjgaPjgabjgYrjgorjgb7jgZnjgIIgIAoqIOODl+ODreOCreOCt+eSsOWig+OBoOOBo+OBn+OBruOBp+OAgS5lbnbjgafoqK3lrprjgZfjgabjgb7jgZnjgIIgIAoqIOS9v+OBo+OBpuOBquOBhOOCk+OBp+OBmeOBjOOAgVBvc3RncmVz44KC56uL44Gh5LiK44GM44KK44G+44GZ4oCm77yI44GE44Gk44GL57mL44GS44Gm44KE44KK5Y+W44KK44GX44Gf44GE44Gq44Go6ICD44GI44Gm77yJCgojIyMgT3BlbkFQSSAoU3dhZ2dlcikKT3BlbkFQSeOBrkRvY2tlcuOCs+ODs+ODhuODiuOBjOeri+OBoeS4iuOBjOOBo+OBpuOBiuOCiuOBvuOBmQp8IOOCs+ODs+ODhuODinwg44Ob44K544OIOuODneODvOODiCB8CnwgLS0tLSB8IC0tLS0gfAp8IHN3YWdnZXItZWRpdG9yICAgICAgICAgICAgICAgICAgIHwgbG9jYWxob3N0OjgwMDEgfAp8IHN3YWdnZXItdWkgICAgICAgICAgICAgICAgICAgICAgIHwgbG9jYWxob3N0OjgwMDIgfAp8IHN0b3BsaWdodC9wcmlzbShBUEnjg6Ljg4Pjgq/jgrXjg7zjg5ApICB8IGxvY2FsaG9zdDo4MDAzIHwKCuOCs+ODnuODs+ODiQpgYGAKbnBtIHJ1biBvcGVuYXBpCmBgYArjgafjgIEvc3JjL2dlbmVyYXRlZC/phY3kuIvjgatBUElDbGllbnTjgafjgYLjgotUeXBlU2NyaXB044Gu44Kz44O844OJ44KS44KS55Sf5oiQ44GX44G+44GZ44CCCgrjgZnjgbnjgabjgIEvYXBpL29wZW5hcGkueW1s44KS5Y+C54Wn44GX44Gm44G+44GZCg== readmeEtag: '"58516d1fc2ffaffff87607556511980884f5e027"' readmeLastModified: Wed, 10 Aug 2022 02:33:20 GMT repositoryId: 501450034 description: ReactでGoogleMapの実験場。いろいろと付け足してみたい created: '2022-06-09T00:15:08Z' updated: '2022-06-29T00:34:11Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: takeshi-kishima logo: https://avatars.githubusercontent.com/u/63081424?v=4 repoEtag: '"21972407fd0906b81f21073a2d9abb3dc91ee81c872252ec9d6a9f304d46939d"' repoLastModified: Wed, 29 Jun 2022 00:34:11 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/sabo99/laravel-restfulapi v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+PGEgaHJlZj0iaHR0cHM6Ly9sYXJhdmVsLmNvbSIgdGFyZ2V0PSJfYmxhbmsiPjxpbWcgc3JjPSJodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vbGFyYXZlbC9hcnQvbWFzdGVyL2xvZ28tbG9ja3VwLzUlMjBTVkcvMiUyMENNWUsvMSUyMEZ1bGwlMjBDb2xvci9sYXJhdmVsLWxvZ29sb2NrdXAtY215ay1yZWQuc3ZnIiB3aWR0aD0iNDAwIj48L2E+PC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CjxhIGhyZWY9Imh0dHBzOi8vdHJhdmlzLWNpLm9yZy9sYXJhdmVsL2ZyYW1ld29yayI+PGltZyBzcmM9Imh0dHBzOi8vdHJhdmlzLWNpLm9yZy9sYXJhdmVsL2ZyYW1ld29yay5zdmciIGFsdD0iQnVpbGQgU3RhdHVzIj48L2E+CjxhIGhyZWY9Imh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy9sYXJhdmVsL2ZyYW1ld29yayI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L2R0L2xhcmF2ZWwvZnJhbWV3b3JrIiBhbHQ9IlRvdGFsIERvd25sb2FkcyI+PC9hPgo8YSBocmVmPSJodHRwczovL3BhY2thZ2lzdC5vcmcvcGFja2FnZXMvbGFyYXZlbC9mcmFtZXdvcmsiPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL3BhY2thZ2lzdC92L2xhcmF2ZWwvZnJhbWV3b3JrIiBhbHQ9IkxhdGVzdCBTdGFibGUgVmVyc2lvbiI+PC9hPgo8YSBocmVmPSJodHRwczovL3BhY2thZ2lzdC5vcmcvcGFja2FnZXMvbGFyYXZlbC9mcmFtZXdvcmsiPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL3BhY2thZ2lzdC9sL2xhcmF2ZWwvZnJhbWV3b3JrIiBhbHQ9IkxpY2Vuc2UiPjwvYT4KPC9wPgoKPGJyPgoKIyMg8J+agCBBUEkgRG9jdW1lbnRhdGlvbgoKLSAgIFtWaWV3IERvY3VtZW50YXRpb25dKGh0dHBzOi8vYXBwLnN3YWdnZXJodWIuY29tL2FwaXMtZG9jcy9zYWJvOTkvTGFyYXZlbFJFU1RmdWxBUEkvMS4wIy8pCgo8YnI+CgojIyDwn5qAIFF1aWNrIHN0YXJ0IChDbG9uZSBQcm9qZWN0KQoKMS4gICoqRmlyc3QgQ2xvbmUgUHJvamVjdCoqCgogICAgQ2xvbmUgcHJvamVjdCB3aXRoIHRoaXMgbGluayBodHRwczovL2dpdGh1Yi5jb20vc2Fibzk5L2xhcmF2ZWwtcmVzdGZ1bGFwaS5naXQKCjIuICAqKkluc3RhbGwgRGVwZWRlbmNpZXMqKgoKICAgIFJlcXVpcmVkIGBjb21wb3NlcmAKCiAgICBgYGBzaGVsbAogICAgY29tcG9zZXIgaW5zdGFsbAogICAgYGBgCjMuICAqKkNvbmZpZyBFbnZpcm9ubWVudCoqCgogICAgQ29weSBgLmVudi5leGFtcGxlYCB0byBuZXcgZmlsZSBgLmVudmAKICAgIGBgYHNoZWxsCiAgICBjcCAuZW52LmV4YW1wbGUgLmVudgogICAgYGBgCgo0LiAgKipSZWFkeSB0byBsYXVuY2ggb24gbG9jYWwgc2VydmVyKioKCiAgICBgYGBzaGVsbAogICAgcGhwIGFydGlzYW4gc2VydmUKICAgIGBgYAoKICAgIFlvdXIgc2l0ZSBpcyBub3cgcnVubmluZyBhdCBodHRwOi8vMTI3LjAuMC4xOjgwMDAKCjUuICAqKkxlYXJuIG1vcmUqKgoKICAgIC0gICBbTGFyYXZlbCBEb2N1bWVudGF0aW9uXShodHRwczovL2xhcmF2ZWwuY29tL2RvY3MvOS54LykKCiMjIExpY2Vuc2UKClRoZSBMYXJhdmVsIGZyYW1ld29yayBpcyBvcGVuLXNvdXJjZWQgc29mdHdhcmUgbGljZW5zZWQgdW5kZXIgdGhlIFtNSVQgbGljZW5zZV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQpLgo= readmeEtag: '"51d210f64930913796715365723af111e729de94"' readmeLastModified: Thu, 05 May 2022 13:38:03 GMT repositoryId: 463057012 description: >- Backend Restful API with Framework Laravel 9 | API Documentation Link below created: '2022-02-24T07:54:51Z' updated: '2022-02-27T13:19:46Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: sabo99 logo: https://avatars.githubusercontent.com/u/49272368?v=4 repoEtag: '"411a9f1649b26993e4699533b00d5217dea8eec29777460126d20b5398f56ebb"' repoLastModified: Sun, 27 Feb 2022 13:19:46 GMT foundInMaster: true category: - Testing - Server Implementations id: 5b2cfcb738d836f99e8d100b603771ba - source: openapi3 tags repository: https://github.com/dahang/uci_djs v3: true id: 0da9c939d7fb10d818215dd9bd77eb9e repositoryMetadata: base64Readme: >- IyBSQkMgVUNJIERvdyBKb25lcyBJbmRleCBkYXRhIHNldCB3aXRoIHNwcmluZy1ib290LTIgd2ViZmx1eAoKIyMgQnVpbGRpbmcgYXBwbGljYXRpb24KCiMjIyBQcmUtcmVxdWlzaXRlcwotIEpESyAxNysKLSBtYXZlbiAzCi0gZG9ja2VyIENMSQoKIyMgT3B0aW9uIDEgcnVuaW5nIGZyb20gbG9jYWwKCmBgYHNoCiAgICBjZCBiYWNrZW5kCiAgICBtdm4gY2xlYW4gcGFja2FnZQogICAgbXZuIHNwcmluZy1ib290OnJ1bgogICAgZ290byBodHRwOi8vbG9jYWxob3N0OjgwODAvd2ViamFycy9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKYGBgCgojIyBPcHRpb24gMjogVXNpbmcgc2NyaXB0CgpgYGBzaAogICAgLi9iYWNrZW5kL2J1aWxkLnNoCiAgICAuL3N0YXJ0LnNoCgpgYGAKCgojIyBPcHRpb24gMzogQnVpbGRpbmcgRXhlY3V0YWJsZSBKQVIKVG8gY3JlYXRlIGFuIGBleGVjdXRhYmxlIGphcmAsIHNpbXBseSBydW46CgpgYGBzaAogbXZuIGNsZWFuIHBhY2thZ2UKYGBgCgojIyMjIFRvIGNyZWF0ZSBhIG5vbi1uYXRpdmUgZG9ja2VyIGltYWdlLCBzaW1wbHkgcnVuOgoKYGBgc2gKbXZuIGNsZWFuIHNwcmluZy1ib290OmJ1aWxkLWltYWdlCmBgYAoKIyMjIyBUbyBydW4gdGhlIGRlbW8gdXNpbmcgZG9ja2VyLCBpbnZva2UgdGhlIGZvbGxvd2luZzoKCmBgYHNoCmRvY2tlciBydW4gLS1ybSAtcCA4MDgwOjgwODAgZGpzOjEuMC4wLVNOQVBTSE9UCmBgYAo= readmeEtag: '"0040595c1c357acffccd43e8292b2faed633b02a"' readmeLastModified: Mon, 24 Oct 2022 11:46:02 GMT repositoryId: 556718039 description: DEMO for OpenAPI3 Spring boot 2.7.x Webflux. R2DBC created: '2022-10-24T11:37:14Z' updated: '2022-10-24T12:16:24Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: dahang logo: https://avatars.githubusercontent.com/u/1457798?v=4 license: MIT repoEtag: '"b3fea4d65806f71777fd96fba8b9d370bdad4f75a2dd2ccb7d36fa9cac50c075"' repoLastModified: Mon, 24 Oct 2022 12:16:24 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/veselovnd88/websocketroomproject v3: true id: b83f5b1f55deaae47e0ebad1f20d8484 repositoryMetadata: repositoryId: 613807194 description: >- Part of Harvex Project. My part is responsible for Room catalog, room settings, chat and events, and also controlling player settings between all users created: '2023-03-14T10:06:12Z' updated: '2023-07-10T13:42:44Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Veselovnd88 logo: https://avatars.githubusercontent.com/u/63257041?v=4 repoEtag: '"7d382676edf687f9f03bed4e84181d224fa0b2b82534723691ed772911cce9d7"' repoLastModified: Mon, 10 Jul 2023 13:42:44 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/irealworlds/laravel-openapi v3: true id: d955812d6ad9842a4e1dba2cb659e942 repositoryMetadata: repositoryId: 673433766 description: OpenApi generation for Laravel applications. created: '2023-08-01T15:59:35Z' updated: '2023-08-08T13:45:40Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: irealworlds logo: https://avatars.githubusercontent.com/u/28299559?v=4 license: MIT repoEtag: '"1d37e88d1ef2521a86dae327059f4fa3eea7fb188de6fc7f1ad99e545c0613fa"' repoLastModified: Tue, 08 Aug 2023 13:45:40 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/asis2016/amaharjande v3: true id: 514fac05bafc5ae153e267cfd3c2b487 repositoryMetadata: base64Readme: >- IyB3d3cuYW1haGFyamFuLmRlCgojIyBBIHNob3J0IG5vdGUgb24gQ2xvdWQgY29tcHV0aW5nCi0gW1JlYWQgdGhpcyBhcnRpY2xlXShjbG91ZC1jb25jZXB0cy9SRUFETUUubWQpCgojIyBPcmdhbml6ZSBtYXJrZG93biB3aXRoIFB5dGhvbgotIFN0ZXAtYnktc3RlcCBndWlkZSB0byBvcmdhbml6aW5nIHlvdXIgbWFya2Rvd24gZmlsZXMgYW5kIGNvbXBpbGluZyB0aGVtIGludG8gYSBzaW5nbGUgUkVBRE1FLm1kIHVzaW5nIFB5dGhvbi4gW1JlYWQgbW9yZV0ob3JnYW5pemUtbWFya2Rvd24td2l0aC1weXRob24vUkVBRE1FLm1kKQoKIyMgUmV2ZXJzZSBwcm94eQotIFRoaXMgcHJvamVjdCBkZW1vbnN0cmF0ZXMgcmV2ZXJzZSBwcm94eSB1c2luZyBEamFuZ28sIEZsYXNrIGFuZCBMaW51eC4gW1JlYWQgbW9yZV0ocmV2ZXJzZS1wcm94eS9SRUFETUUubWQpCgojIyBTZWxmLXNpZ25lZCBjZXJ0aWZpY2F0ZSBzZXR1cCBvbiBuZ2lueAotIFRoaXMgYXJ0aWNsZSBzaG93cyB5b3UgaG93IHlvdSBjYW4gY3JlYXRlIGEgc2VsZi1zaWduZWQgY2VydGlmaWNhdGUgZm9yIG5naW54LiBbUmVhZCBtb3JlXShzZWxmLXNpZ25lZC1jZXJ0aWZpY2F0ZS1uZ2lueC9SRUFETUUubWQpCgojIyB3aGF0aXNteWlwCi0gVGhpcyBwcm9qZWN0IHNob3dzIHlvdSBob3cgeW91IGNhbiBidWlsZCB5b3VyIG93biBwdWJsaWMgSVAgYWRkcmVzcyB2aWV3ZXIgdXNpbmcgRmxhc2suIFtSZWFkIG1vcmVdKHdoYXRpc215aXAvUkVBRE1FLm1kKQ== readmeEtag: '"943e272826af771c8e7b238224481232cb3ab7b2"' readmeLastModified: Sat, 20 Jan 2024 06:35:50 GMT repositoryId: 451885030 description: >- This repository is a part of www.amaharjan.de blogging which provides tutorial on various topics. created: '2022-01-25T13:19:15Z' updated: '2025-12-28T09:39:27Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: asis2016 logo: https://avatars.githubusercontent.com/u/19593234?v=4 repoEtag: '"c61dad3448fd3bb10bb64b4e86352dc451f6a3b1197a2cae0661f2a073b6d43e"' repoLastModified: Sun, 28 Dec 2025 09:39:27 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/himenon/openapi-parameter-formatter v3: true repositoryMetadata: base64Readme: >- IyBAaGltZW5vbi9vcGVuYXBpLXBhcmFtZXRlci1mb3JtYXR0ZXIKCkEgbGlicmFyeSB0aGF0IHNlcmlhbGl6ZXMgT3BlbkFQSSBwYXJhbWV0ZXIgb2JqZWN0cy4KSXQgaXMgaW1wbGVtZW50ZWQgYWNjb3JkaW5nIHRvIHRoZSBmb2xsb3dpbmcgc3R5bGUgZGVmaW5pdGlvbi4KCi0gaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4xLjAubWQjc3R5bGUtZXhhbXBsZXMKCiMjIFVzYWdlCgpgYGB0cwppbXBvcnQgeyBQYXRoUGFyYW1ldGVyIH0gZnJvbSAiQGhpbWVub24vb3BlbmFwaS1wYXJhbWV0ZXItZm9ybWF0dGVyIjsKClBhdGhQYXJhbWV0ZXIuZ2VuZXJhdGUoImNvbG9yIiwgewogIHZhbHVlOiB7CiAgICBSOiAxMDAsCiAgICBHOiAyMDAsCiAgICBCOiAxNTAsCiAgfSwKICBzdHlsZTogImxhYmVsIiwKICBleHBsb2RlOiB0cnVlLAp9KTsKYGBgCgpgYGB0cwppbXBvcnQgeyBRdWVyeVBhcmFtZXRlciB9IGZyb20gIkBoaW1lbm9uL29wZW5hcGktcGFyYW1ldGVyLWZvcm1hdHRlciI7CgpRdWVyeVBhcmFtZXRlci5nZW5lcmF0ZSgiY29sb3IiLCB7CiAgdmFsdWU6IHsKICAgIFI6ICIjMTAwIiwKICAgIEc6ICIjMjAwIiwKICAgIEI6ICIjMTUwIiwKICB9LAogIHN0eWxlOiAiZm9ybSIsCiAgZXhwbG9kZTogdHJ1ZSwKfSk7CgovLyBSPSUyMzEwMCZHPSUyMzIwMCZCPSUyMzE1MApgYGAKCmBgYHRzCmltcG9ydCB7IFF1ZXJ5UGFyYW1ldGVyIH0gZnJvbSAiQGhpbWVub24vb3BlbmFwaS1wYXJhbWV0ZXItZm9ybWF0dGVyIjsKClF1ZXJ5UGFyYW1ldGVyLmdlbmVyYXRlKCJjb2xvciIsIHsKICB2YWx1ZTogWyIjYmx1ZSIsICIjYmxhY2siLCAiI2Jyb3duIl0sCiAgc3R5bGU6ICJzcGFjZURlbGltaXRlZCIsCiAgZXhwbG9kZTogZmFsc2UsCn0pOwoKLy8gY29sb3I9JTIzYmx1ZSUyMCUyM2JsYWNrJTIwJTIzYnJvd24KYGBgCgpgYGB0cwppbXBvcnQgeyBRdWVyeVBhcmFtZXRlciB9IGZyb20gIkBoaW1lbm9uL29wZW5hcGktcGFyYW1ldGVyLWZvcm1hdHRlciI7CgpRdWVyeVBhcmFtZXRlci5nZW5lcmF0ZSgiY29sb3IiLCB7CiAgdmFsdWU6IFsiI2JsdWUiLCAiI2JsYWNrIiwgIiNicm93biJdLAogIHN0eWxlOiAicGlwZURlbGltaXRlZCIsCiAgZXhwbG9kZTogZmFsc2UsCn0pOwoKLy8gY29sb3I9JTIzYmx1ZSU3QyUyM2JsYWNrJTdDJTIzYnJvd24KYGBgCgpgYGB0cwppbXBvcnQgeyBRdWVyeVBhcmFtZXRlciB9IGZyb20gIkBoaW1lbm9uL29wZW5hcGktcGFyYW1ldGVyLWZvcm1hdHRlciI7CgpRdWVyeVBhcmFtZXRlci5nZW5lcmF0ZSgiY29sb3IiLCB7CiAgdmFsdWU6IHsKICAgIFI6IDEwMCwKICAgIEc6IDIwMCwKICAgIEI6IDE1MCwKICB9LAogIHN0eWxlOiAiZGVlcE9iamVjdCIsCiAgZXhwbG9kZTogdHJ1ZSwKfSk7CgovLyBjb2xvciU1QlIlNUQ9MTAwJmNvbG9yJTVCRyU1RD0yMDAmY29sb3IlNUJCJTVEPTE1MApgYGAKCmBgYHRzCmltcG9ydCB7IEhlYWRlclBhcmFtZXRlciB9IGZyb20gIkBoaW1lbm9uL29wZW5hcGktcGFyYW1ldGVyLWZvcm1hdHRlciI7CgpIZWFkZXJQYXJhbWV0ZXIuZ2VuZXJhdGUoImNvbG9yIiwgewogIHZhbHVlOiBbImJsdWUiLCAiYmxhY2siLCAiYnJvd24iXSwKICBzdHlsZTogInNpbXBsZSIsCiAgZXhwbG9kZTogZmFsc2UsCn0pOwovLyAiYmx1ZSxibGFjayxicm93biIKYGBgCgpgYGB0cwppbXBvcnQgeyBDb29raWVQYXJhbWV0ZXIgfSBmcm9tICJAaGltZW5vbi9vcGVuYXBpLXBhcmFtZXRlci1mb3JtYXR0ZXIiOwoKQ29va2llUGFyYW1ldGVyLmdlbmVyYXRlKCJjb2xvciIsIHsKICB2YWx1ZTogWyJibHVlIiwgImJsYWNrIiwgImJyb3duIl0sCiAgc3R5bGU6ICJmb3JtIiwKICBleHBsb2RlOiBmYWxzZSwKfSk7Ci8vICJjb2xvcj1SLDEwMCxHLDIwMCxCLDE1MCIKYGBgCgojIyBMSUNFTkNFCgpbQGhpbWVub24vb3BlbmFwaS1wYXJhbWV0ZXItZm9ybWF0dGVyXShodHRwczovL2dpdGh1Yi5jb20vSGltZW5vbi9vcGVuYXBpLXBhcmFtZXRlci1mb3JtYXR0ZXIp44O7TUlUCg== readmeEtag: '"6e23c7b3fe5566ce18caf7e0402dc380297c0bb4"' readmeLastModified: Tue, 19 Mar 2024 12:15:25 GMT repositoryId: 328625019 description: OpenAPI Query and Path Parameter Formatter. created: '2021-01-11T10:19:36Z' updated: '2025-06-26T20:08:07Z' language: TypeScript archived: false stars: 1 watchers: 0 forks: 2 owner: Himenon logo: https://avatars.githubusercontent.com/u/6715229?v=4 license: MIT repoEtag: '"c8d39ee77abc30e171746332931fa73ab345ad8bf0d91450c6960de8b30350a7"' repoLastModified: Thu, 26 Jun 2025 20:08:07 GMT foundInMaster: true category: Data Validators id: 2db87929392a80657903864a4b385784 - source: openapi3 tags repository: https://github.com/laurosilveira/sify-flix-api v3: true id: 8e919ac798391627da92325cb87fa471 repositoryMetadata: base64Readme: >- IyBzaWZ5LWZsaXgtYXBpCgojIyBBYm91dCB0aGlzIHByb2plY3QKKiBbQWJvdXRdKCNhYm91dCkKKiBbQXJjaGl0ZWN0dXJlXSgjQXJjaGl0ZWN0dXJlKQoqIFtUZWNobm9sb2dpZXNdKCNUZWNobm9sb2dpZXMpCiogW1N3YWdnZXIgRG9jdW1lbnRhdGlvbl0oI1N3YWdnZXItZG9jdW1lbnRhdGlvbikKKiBbU2VjdXJpdHldKCNTZWN1cml0eSkKKiBbSG93IHRvIHJ1bl0oI2hvdy10by1ydW4pCiAgICAqIFtSdW4gd2l0aCBEb2NrZXJdKCNydW4td2l0aC1kb2NrZXItY29tcG9zZSkKICAgICogW1J1biB3aXRoIG12biBzcHJpbmctYm9vdDpydW5dKCNydW4td2l0aC1tdm4tc3ByaW5nLWJvb3QpCiogW2hvdyB0byBydW4gVGVzdF0oI3J1bi10ZXN0cykKKiBbQ29udHJpYnV0b3JzXSgjY29udHJpYnV0b3JzKQoKIyBBYm91dApUaGlzIGlzIGEgUmVzdCBBUEkgZGV2ZWxvcGVkIGJhc2VkIG9uIHRoZSByZXF1aXJlbWVudHMgZ2l2ZW4gZm9yIHRoZSBzZWxlY3Rpb24gcHJvY2VzcyBieSB0aGUgY29tcGFueSBXMk0uClRoaXMgQVBJIHdhcyBidWlsdCB3aXRoIEphdmEgMjEsIFNwcmluZyBCb290LCBIMiBEYXRhYmFzZSwgRmx5d2F5IE1pZ3JhdGlvbiwgU3ByaW5nIERvYyBPcGVuQVBJLCBEb2NrZXIgYW5kIERvY2tlci1jb21wb3NlLgoKIyBBcmNoaXRlY3R1cmUKVGhlIGFwcGxpY2F0aW9uIGlzIGJ1aWx0IGZvbGxvd2luZyB0aGUgKipNVkMqKiBkZXNpZ24gcGF0dGVybi4KRWFjaCBvZiB0aGUgY29tcG9uZW50cyB3aWxsIGJlIGRlc2NyaWJlZCBiZWxvdzoKKiAqKlZpZXcqKjogY29udGFpbnMgcmVzcG9uc2UgZnJvbSB0aGUgUmVzdCBjb250cm9sbGVyIHRoYXQgd2lsbCBiZSBzaG93biBpbiBKU09OIGZvcm1hdCB0byB0aGUgY2xpZW50LgoqICoqUmVzdCBDb250cm9sbGVyKio6IHJlc3BvbnNpYmxlIGZvciByZWNlaXZpbmcgY2FsbHMgYW5kIGRpcmVjdGluZyB0aGVtIHRvIHRoZSBjb3JyZWN0IHNlcnZpY2UuCiogKipTZXJ2aWNlKio6IGEgZGVzaWduIHBhdHRlcm4gbGF5ZXIgcmVzcG9uc2libGUgZm9yIGNhbGxpbmcgdGhlIHNlcnZpY2UgZnJvbSB0aGUgcmVwb3NpdG9yeSBsYXllci4KKiAqKlJlcG9zaXRvcnkqKjogdGhpcyBsYXllciBpcyByZXNwb25zaWJsZSBmb3IgY29ubmVjdGluZyB0byB0aGUgZGF0YWJhc2UgYW5kIGZvciBwZXJzaXN0aW5nLCByZXRyaWV2aW5nLCB1cGRhdGluZyBhbmQgZGVsZXRpbmcgZGF0YS4KCiMgVGVjaG5vbG9naWVzCi0gSmF2YSAyMQotIFNwcmluZyBCb290IDMuMi4yCi0gU3ByaW5nIFNlY3VyaXR5IC0gQmFzaWMgQXV0aGVudGljYXRpb24KLSBTcHJpbmcgT3BlbkFQSQotIEgyIERhdGFiYXNlCi0gRmx5d2F5IE1pZ3JhdGlvbgotIERvY2tlcgotIERvY2tlci1jb21wb3NlCi0gQXNzZXJ0SiwgTW9ja2l0bwoKIyBTd2FnZ2VyIERvY3VtZW50YXRpb24KClRoZSBkb2N1bWVudGF0aW9uIGlzIGdlbmVyYXRlZCBieSBTcHJpbmcgT3BlbmFwaS4gT25jZSB5b3UgaGF2ZSBzdGFydGVkIHRoZSBhcHBsaWNhdGlvbiB0aGUgdXJsOiBgYGBodHRwOi8vbG9jYWxob3N0OjgwODAvc3dhZ2dlci11aS5odG1sYGBgCgohW3NwYWNlLXNoaXAtcmVzY29udHJvbGxlci5wbmddKGRhdGEvc3BhY2Utc2hpcC1yZXN0Y29udHJvbGxlci5wbmcpCgojIFNlY3VyaXR5ClRoaXMgQVBJIGlzIHByb3RlY3RlZCB3aXRoIFNwcmluZyBTZWN1cml0eSB1c2luZyBiYXNpYyBhdXRob3JpemF0aW9uLgpUaGlzIEFQSSBpcyBjb25maWd1cmVkIHRvIG5vdCBzYXZlIHNlc3Npb25zLgpPbmNlIHlvdSBoYXZlIGFjY2Vzc2VkIFN3YWdnZXIsIHNpbXBseSBhdXRoZW50aWNhdGUgd2l0aCB5b3VyIHVzZXJuYW1lIGFuZCBwYXNzd29yZC4KTm90ZXM6IEJhc2ljIGF1dGhvcml6YXRpb24gbW9kZSBpcyAqKm5vdCByZWNvbW1lbmRlZCoqLCBhcyBpdCBpcyBub3QgdGhlIG1vc3Qgc2VjdXJlLgpXaGVuIHN0YXJ0aW5nIHRoZSBhcHBsaWNhdGlvbiwgYSB1c2VyIGBgYHVzZXI6IGFkbWluIHBhc3N3b3JkOjEyMzRgYGAgd2l0aCB0aGUgYWRtaW4gcm9sZSB3aWxsIGFscmVhZHkgYmUgYXZhaWxhYmxlLgoKIyBIb3cgdG8gUnVuClRoZXJlIHR3byBvcHRpb25zIHRvIHJ1biB0aGlzIGFwcGxpY2F0aW9uLCBydW4gYnkgZG9ja2VyLWNvbXBvc2Ugb3IgcnVuIGFzIHNwcmluZy1ib290LgoKIyMjIFJ1biB3aXRoIGRvY2tlci1jb21wb3NlCk1ha2Ugc3VyZSB5b3UgaGF2ZSAqKkRvY2tlcioqIGluc3RhbGxlZCBhbmQgZXhlY3V0ZSB0aGUgY29tbWFuZC4KCmBgYApkb2NrZXItY29tcG9zZSB1cApgYGAKIyMjIFJ1biB3aXRoIG12biBzcHJpbmctYm9vdApgYGBzaGVsbAptdm4gc3ByaW5nLWJvb3Q6cnVuCmBgYAoKIyBSdW4gdGVzdHMKClRoaXMgYXBwbGljYXRpb24gaGFzIHVuaXR5IHRlc3RzIGFuZCBpbnRlZ3JhdGlvbiB0ZXN0cywgb25jZSBpdCB1c2UgTWF2ZW4ganVzdCBydW46CgpgYGAKbXZuIHRlc3QKYGBgCgojIyBDb250cmlidXRvcnMKW0BMYXVyb1NpbHZlaXJhXShodHRwczovL2dpdGh1Yi5jb20vTGF1cm9TaWx2ZWlyYSkK readmeEtag: '"a58dff38848d87c48aaf24840a06ce946dbabded"' readmeLastModified: Mon, 24 Jun 2024 07:47:59 GMT repositoryId: 761306430 description: >- An API REST built with Spring boot, spring security about movies and series of spaceships created: '2024-02-21T16:14:56Z' updated: '2024-06-24T07:48:03Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: LauroSilveira logo: https://avatars.githubusercontent.com/u/10181673?v=4 repoEtag: '"ba8e138a0fe93fb8b5f67bfe54901c611eadb2224c0d9a5ff974a9d11a4f2a4a"' repoLastModified: Mon, 24 Jun 2024 07:48:03 GMT category: Server Implementations foundInMaster: true - source: https://openapi.tools/ name: Apitive Studio category: - Documentation - GUI Editors - Mock language: - Angular 7.0 - Java - Saas link: https://www.apitive.com source_description: > A platform for Digital Product Managers and API Consultants to design REST APIs with in-built mock and documentation. v2: true v3: true foundInMaster: true id: 2fa64050f1dc133ee8b33c58c25298bf - source: https://openapi.tools/ name: LucyBot api-spec-converter category: Converters language: Node.js link: https://www.npmjs.com/package/api-spec-converter repository: https://github.com/lucybot-inc/api-spec-converter source_description: Convert between API description formats such as OpenAPI and RAML. v2: true v3: true repositoryMetadata: base64Readme: >- # api-spec-converter
> This project is looking for a new maintainer! Let us know if you're interested in taking it over.

[![Share on Twitter][twitter-image]][twitter-link]

[![Chat on gitter][gitter-image]][gitter-link]
[![NPM version][npm-image]][npm-link]
[![Build status][travis-image]][travis-link]

[![Dependency status][deps-image]][deps-link]
[![devDependency status][devdeps-image]][devdeps-link]

Convert between API description formats such as [Swagger](http://swagger.io/) and [RAML](http://raml.org/)

**Currently only supports conversion to OpenAPI(fka Swagger) 2.0 format, and from OpenAPI 2.0 to OpenAPI 3.0.x**

You can also use the online version at https://lucybot-inc.github.io/api-spec-converter/.

## Installation

### Command Line
> Problems? See [issue #132](https://github.com/LucyBot-Inc/api-spec-converter/issues/132)
```bash
npm install -g api-spec-converter
```

### NodeJS/Browser
```bash
npm install --save api-spec-converter
```

## Usage

### Command Line
```bash
$ api-spec-converter -h

  Usage: api-spec-converter [options] <URL|filename>

  Convert API descriptions between popular formats.

  Supported formats:
    * swagger_1
    * swagger_2
    * openapi_3
    * api_blueprint
    * io_docs
    * google
    * raml
    * wadl

  Options:

    -h, --help              output usage information
    -V, --version           output the version number
    -f, --from <format>     Specifies format to convert
    -t, --to <format>       Specifies output format
    -s, --syntax [syntax]   Specifies output data syntax: json or yaml. Defaults to json
    -o, --order [sortOrder] Specifies top fields ordering: openapi or alpha. Defaults to openapi
    -c, --check             Check if result is valid spec
    -d, --dummy             Fill missing required fields with dummy data
```

Example:
```bash
$ api-spec-converter --from=swagger_1 --to=swagger_2 --syntax=yaml --order=alpha https://raw.githubusercontent.com/LucyBot-Inc/api-spec-converter/master/test/input/swagger_1/petstore/pet.json > swagger.json
```

### NodeJS

### Options
* `from` - source format (see formats below)
* `to` - desired format (see formats below)
* `source` - Filename, URL, or JS object for the source
### Simple example:
```js
var Converter = require('api-spec-converter');

Converter.convert({
  from: 'swagger_1',
  to: 'swagger_2',
  source: 'https://api.gettyimages.com/swagger/api-docs',
}, function(err, converted) {
  console.log(converted.stringify());
  // For yaml and/or OpenApi field order output replace above line
  // with an options object like below
  //   var  options = {syntax: 'yaml', order: 'openapi'}
  //   console.log(converted.stringify(options));
})
```
### Callback vs Promises
This library has full support for both callback and promises.
All async functions return promises but also will execute callback if provided.

```js
var Converter = require('api-spec-converter');

Converter.convert({
  from: 'swagger_1',
  to: 'swagger_2',
  source: 'https://api.gettyimages.com/swagger/api-docs',
})
.then(function(converted) {
  console.log(converted.stringify());
});
```
### Advanced features:
```js
var Converter = require('api-spec-converter');
Converter.convert({
  from: 'swagger_1',
  to: 'swagger_2',
  source: 'https://api.gettyimages.com/swagger/api-docs',
})
  .then(function(converted) {
    // [Optional] Fill missing fields with dummy values
    converted.fillMissing();

    // [Optional] Validate converted spec
    return converted.validate()
      .then(function (result) {
        if (result.errors)
          return console.error(JSON.stringify(errors, null, 2));
        if (result.warnings)
          return console.error(JSON.stringify(warnings, null, 2));

        fs.writeFileSync('swagger2.json', converted.stringify());
      });
  });
```

### Browser
```js
<script src="node_modules/api-spec-converter/dist/api-spec-converter.js"></script>
APISpecConverter.convert(...)
```

## Supported Formats

* [Swagger 1.x](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/1.2.md) (swagger_1)
* [OpenAPI(fka Swagger) 2.0](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md) (swagger_2)
* [OpenAPI 3.0.x](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md) (openapi_3)
* [I/O Docs](https://github.com/mashery/iodocs) (io_docs)
* [API Blueprint](https://github.com/apiaryio/api-blueprint/blob/master/API%20Blueprint%20Specification.md) (api_blueprint)
* [Google API Discovery](https://developers.google.com/discovery/v1/reference/apis) (google)
* [RAML](http://raml.org/spec.html) (raml)
* [WADL](http://www.w3.org/Submission/wadl/) (wadl)


## Conversion Table

|from:             |swagger_1|swagger_2|openapi_3|io_docs|api_blueprint|google|raml|wadl|
-------------------|:-------:|:-------:|:-----:|:-----:|:-----------:|:----:|:--:|:--:|
|to swagger_1      |  n/a    |         |       |       |             |      |    |    |
|to swagger_2      | :white_check_mark: |    n/a  | :white_check_mark:  | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: | :white_check_mark: |
|to openapi_3      |   :eight_spoked_asterisk:     | :white_check_mark: |  n/a  |   :eight_spoked_asterisk:  | :eight_spoked_asterisk: |  :eight_spoked_asterisk:   | :eight_spoked_asterisk:  | :eight_spoked_asterisk:  |
|to io_docs        |         |         |       |  n/a  |             |      |    |    |
|to api_blueprint  |         |         |       |       |    n/a      |      |    |    |
|to google         |         |         |       |       |             |  n/a |    |    |
|to raml           |         |         |       |       |             |      | n/a|    |
|to wadl           |         |         |       |       |             |      |    | n/a|

#### Key
* :white_check_mark: - direct conversion
* :eight_spoked_asterisk: - conversion via swagger_2

## Contributing
Contributions are welcome and encouraged.

### Testing
Please add a test case if you're adding features or fixing bugs. To run the tests:

```bash
npm test
```

In case you need to override the expected outputs, due to a justified and verified change, run this:
```bash
WRITE_GOLDEN=true npm test
```
### Releases
```
npm run browserify
git commit -a -m "Build browser distribution"
npm version minor # or major/patch
npm publish
git push --follow-tags
```

[twitter-image]: https://img.shields.io/twitter/url/http/lucybot.github.io/api-spec-converter.svg?style=social
[twitter-link]: https://twitter.com/intent/tweet?text=Convert+between+API+description+formats+such+as+Swagger+and+RAML:&url=http%3A%2F%2Flucybot.github.io%2Fapi-spec-converter
[gitter-image]: https://img.shields.io/gitter/room/lucybot/api-spec-converter.svg
[gitter-link]: https://gitter.im/lucybot/api-spec-converter
[npm-image]: https://img.shields.io/npm/v/api-spec-converter.svg
[npm-link]: https://npmjs.org/package/api-spec-converter
[travis-image]: https://img.shields.io/travis/LucyBot-Inc/api-spec-converter.svg
[travis-link]: https://travis-ci.org/LucyBot-Inc/api-spec-converter
[deps-image]: https://img.shields.io/david/lucybot/api-spec-converter.svg
[deps-link]: https://david-dm.org/lucybot/api-spec-converter
[devdeps-image]: https://img.shields.io/david/dev/lucybot/api-spec-converter.svg
[devdeps-link]: https://david-dm.org/lucybot/api-spec-converter#info=devDependencies
 readmeEtag: '"7dbd3b0e8b5ff05e10f04d7d2e337385db68efd6"' readmeLastModified: Wed, 08 Jun 2022 13:14:35 GMT repositoryId: 35116903 description: >- Convert API descriptions between popular formats such as OpenAPI(fka Swagger), RAML, API Blueprint, WADL, etc. created: '2015-05-05T18:30:05Z' updated: '2026-01-26T11:46:59Z' language: JavaScript archived: false stars: 1167 watchers: 22 forks: 189 owner: LucyBot-Inc logo: https://avatars.githubusercontent.com/u/26313641?v=4 license: MIT repoEtag: '"26e19a7f4b18451355c8dcf407f57be62bf165484c5ce7a8efd040e43aadb35a"' repoLastModified: Mon, 26 Jan 2026 11:46:59 GMT foundInMaster: true id: dccbe92257187d6d136bb4b5f304d6fe - source: https://openapi.tools/ name: LucyBot DocGen category: - Documentation - Server Implementations repository: https://github.com/lucybot-inc/documentation-starter link: https://lucybot.com/docgen language: JavaScript source_description: >- Generate a customizable website, with API documentation, console, and interactive workflows, from an OpenAPI spec v2: true v3: true repositoryMetadata: base64Readme: >- IyBMdWN5Qm90IFN0YXJ0ZXIgQVBJIENvbnNvbGUKVGhpcyBpcyB0aGUgZGVmYXVsdCBidWlsZCBmb3IgW0x1Y3lCb3QncyBBUEkgRG9jdW1lbnRhdGlvbl0oaHR0cDovL2x1Y3lib3QuY29tKS4KSXQgY2FuIGJlIHVzZWQgaW4gbm9uLWNvbW1lcmNpYWwgcHJvamVjdHMsIG9yIGZvciBkZW1vIHB1cnBvc2VzLgoKQ2hlY2sgb3V0IHRoZSBbUGV0IFN0b3JlIGRlbW9dKGh0dHA6Ly9kZW1vLmx1Y3lib3QuY29tKQoKQ29tbWVyY2lhbCBsaWNlbnNlcyBhbmQgYWRkaXRpb25hbCBmZWF0dXJlcyBhcmUgYXZhaWxhYmxlIGF0IFtsdWN5Ym90LmNvbV0oaHR0cDovL2x1Y3lib3QuY29tKQoKIyMgVXNhZ2UKU2ltcGx5IGZvcmsgdGhpcyByZXBvc2l0b3J5IGFuZCByZXBsYWNlIGBvcGVuYXBpLmpzb25gIHdpdGggeW91cgpbT3BlbkFQSSBzcGVjaWZpY2F0aW9uXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKS4KCmBgYGJhc2gKZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS9MdWN5Qm90LUluYy9kb2N1bWVudGF0aW9uLXN0YXJ0ZXIKY3AgL3BhdGgvdG8vbXkvb3BlbmFwaS5qc29uIGRvY3VtZW50YXRpb24tc3RhcnRlci9vcGVuYXBpLmpzb24KYGBgCgo+IEhhdmUgUkFNTCwgV0FETCwgQVBJIEJsdWVwcmludCwgb3IgSS9PIERvY3M/Cj4gQ2hlY2sgb3V0IFthcGktc3BlYy1jb252ZXJ0ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9sdWN5Ym90L2FwaS1zcGVjLWNvbnZlcnRlcikKCiMjIFNlcnZpbmcKWW91IGNhbiBzZXJ2ZSB0aGUgd2Vic2l0ZSB3aXRoIGFueSBzdGF0aWMgSFRUUCBzZXJ2ZXIuCgplLmcuIHdpdGggW2h0dHAtc2VydmVyXShodHRwczovL2dpdGh1Yi5jb20vaW5kZXh6ZXJvL2h0dHAtc2VydmVyKQpgYGAKbnBtIGluc3RhbGwgLWcgaHR0cC1zZXJ2ZXIKaHR0cC1zZXJ2ZXIgLi9kb2N1bWVudGF0aW9uLXN0YXJ0ZXIKYGBgCgpvciB3aXRoIFBIUDoKYGBgCnBocCAtUyAwLjAuMC4wOjgwIC10IC4vZG9jdW1lbnRhdGlvbi1zdGFydGVyCmBgYAoKVGhlIGVhc2llc3Qgd2F5IHRvIHNlcnZlIHRoZSBkb2N1bWVudGF0aW9uIHB1YmxpY2x5IGlzIHdpdGggR2l0SHViIHBhZ2VzOgppbiB5b3VyIGZvcmssIHZpc2l0IHRoZSBTZXR0aW5ncyBwYWdlLCBhbmQgY2hvb3NlICJtYXN0ZXIgYnJhbmNoIiBhcyB0aGUgc291cmNlIGluIHRoZQpHaXRIdWIgUGFnZXMgc2VjdGlvbi4gWW91IGNhbiBhbHNvIHNldCBhIGN1c3RvbSBkb21haW4gdGhlcmUuCgpZb3UgY2FuIGFsc28gc2VydmUgdGhpcyBkaXJlY3Rvcnkgd2l0aCBBcGFjaGUsIE5vZGVKUyBFeHByZXNzLCBldGMuCgojIyBDdXN0b21pemF0aW9uCiMjIyBUaXRsZSBhbmQgRGVzY3JpcHRpb24KVGhlIEFQSSB0aXRsZSBhbmQgZGVzY3JpcHRpb24gYXJlIGNvbnRyb2xsZWQgYnkKdGhlIGBpbmZvYCBmaWVsZCBpbiBgb3BlbmFwaS5qc29uYC4gIFlvdSBjYW4gdXNlCltNYXJrZG93bl0oaHR0cHM6Ly9naXRodWIuY29tL2FkYW0tcC9tYXJrZG93bi1oZXJlL3dpa2kvTWFya2Rvd24tQ2hlYXRzaGVldCkKaW4gdGhlIGRlc2NyaXB0aW9uLgoKIyMjIFRoZW1lcwpZb3UgY2FuIHVzZSB5b3VyIG93biBCb290c3RyYXAgdGhlbWUgdG8gY3VzdG9taXplIGNvbG9ycywgZm9udHMsIHNpemVzLCBhbmQgbW9yZS4KSnVzdCByZXBsYWNlIGBkaXN0L2Jvb3RzdHJhcC5jc3NgIHdpdGggeW91ciBvd24gYm9vdHN0cmFwLmNzcwoKWW91IGNhbiBnZW5lcmF0ZSBhIGJvb3RzdHJhcC5jc3MgZmlsZSB1c2luZzoKKiBbU3RyYXBwaW5nIV0oaHR0cDovL2JvYmJ5LWJyZW5uYW4uZ2l0aHViLmlvL3N0cmFwcGluZykKKiBbQm9vdHN0cmFwIExpdmUgQ3VzdG9taXplcl0oaHR0cDovL2Jvb3RzdHJhcC1saXZlLWN1c3RvbWl6ZXIuY29tLykKCiMjIyBNb3JlClRoZSBmdWxsIGNvbW1lcmNpYWwgdmVyc2lvbiBvZmZlcnMgc2V2ZXJhbCBhZGRpdGlvbmFsIGZlYXR1cmVzOgoqIEFkZGl0aW9uYWwgTWFya2Rvd24vSFRNTCBzZWN0aW9ucwoqIEN1c3RvbSBuYXZiYXIgYW5kIGZvb3RlcgoqIEN1c3RvbSBob21lcGFnZQoqIEV2ZW50IHRyYWNraW5nCiogU0VPIChgPG1ldGE+YCBhbmQgYDx0aXRsZT5gIHRhZ3MsIGBzaXRlbWFwLnhtbGApCiogVXNlciBhdXRoZW50aWNhdGlvbgoqIEdhbGxlcmllcyBmb3IgbXVsdGlwbGUgQVBJcwoKRm9yIGEgZnVsbCBsaXN0IG9mIGZlYXR1cmVzIGF2YWlsYWJsZSBzZWUgW2x1Y3lib3QuY29tXShodHRwOi8vbHVjeWJvdC5jb20pCgojIyBMaWNlbnNlCltDcmVhdGl2ZSBDb21tb25zIDQuMCAtIE5vbi1jb21tZXJjaWFsXShodHRwczovL2NyZWF0aXZlY29tbW9ucy5vcmcvbGljZW5zZXMvYnktbmMvNC4wLykKCkZvciBhIGNvbW1lcmNpYWwgbGljZW5zZSwgW2NvbnRhY3QgdXNdKGh0dHA6Ly9sdWN5Ym90LmNvbS8jQ29udGFjdCkK readmeEtag: '"286e205f17131d81bf83f5514388a05e39e5a11c"' readmeLastModified: Fri, 14 Jul 2017 20:36:34 GMT repositoryId: 36742809 description: Interactive REST API Documentation created: '2015-06-02T15:30:49Z' updated: '2025-11-17T10:45:01Z' language: HTML archived: false stars: 175 watchers: 11 forks: 59 owner: LucyBot-Inc logo: https://avatars.githubusercontent.com/u/26313641?v=4 license: NOASSERTION repoEtag: '"3a00ca90377faf5f33050a02d1c1046782c79f9e00996353f6144846c99f721c"' repoLastModified: Mon, 17 Nov 2025 10:45:01 GMT foundInMaster: true id: 006c5eda16efc55f050049e1c4964881 - source: https://openapi.tools/ name: Senya Editor category: Text Editors language: Java link: https://senya.io source_description: > JetBrains IDE plugin to show Swagger UI as a preview, for visual feedback as you edit. v2: true v3: true foundInMaster: true id: 0c1c17e79f2ccf9a2140cd587e84a115 v3_1_link: >- https://youtrack.jetbrains.com/issue/IDEA-294782/Add-support-for-OpenAPI-31-in-OpenAPI-Specifications-plugin v3_1: true - source: https://openapi.tools/ name: Sandbox category: - Mock - Server language: - SaaS - Java link: https://getsandbox.com/ repository: https://github.com/getsandbox/sandbox source_description: >- SaaS, self-hosted, or CLI tool for turning OpenAPI (and other) descriptions into a mock server, where you can modify behaviour, simulate downtime, and any other nonsense you can think of thanks to a built-in code editor! v2: true v3: true repositoryMetadata: base64Readme: >- IyBTYW5kYm94IFJ1bnRpbWUgKHJlcGxhY2VkIGJ5IGh0dHBzOi8vZ2l0aHViLmNvbS9nZXRzYW5kYm94L3dvcmtlcikK readmeEtag: '"20279c716c1a9c12c8c3c6f23c5a8402ce03c2af"' readmeLastModified: Sat, 23 May 2020 00:06:45 GMT repositoryId: 266231189 description: null created: '2020-05-23T00:06:26Z' updated: '2024-06-04T23:50:57Z' language: null archived: true stars: 6 watchers: 3 forks: 5 owner: getsandbox logo: https://avatars.githubusercontent.com/u/7564336?v=4 repoEtag: '"c960d57efcb5b582962d5507624fa37ab3d0de99c0eb46fb9734883df99797e0"' repoLastModified: Tue, 04 Jun 2024 23:50:57 GMT foundInMaster: true id: e3edf7a2bc59affaf60207c200c1d0fa - source: https://openapi.tools/ name: APIMatic CodeGen category: - Code Generators - SDK language: SaaS link: https://www.apimatic.io/code-generation-as-a-service source_description: >- Bring in your API description (OAI v2/v3, RAML, API Blueprint, WSDL, etc.) to generate fully functional SDKs in over 10 languages. v2: true v3: true v3_1: true id: b8e1a40af7908548779970a8aa9baba1 foundInMaster: true - source: https://openapi.tools/ name: Meta-API link: https://www.meta-api.io language: SaaS source_description: >- A SaaS platform to integrate APIs using OpenAPI documents, and manipulation of data with online code editor, and automating configuration, authentication, deployment and monitoring. v2: true v3: true v3_1: true foundInMaster: true id: af55f5fdf1f9faca3fd81d5b351523d9 - source: openapi3 tags repository: https://github.com/sergiorodenas/laravel-testdoc v3: true repositoryMetadata: base64Readme: >- IyDwn5KO8J+UkiBMYXJhdmVsIEZ1bmN0aW9uYWwgVGVzdHMgRG9jcyBHZW5lcmF0b3IKClshW0J1aWxkIFN0YXR1c10oaHR0cHM6Ly9zY3J1dGluaXplci1jaS5jb20vZy9Sb2RlbmFzdHlsZS9sYXJhdmVsLXRlc3Rkb2MvYmFkZ2VzL2J1aWxkLnBuZz9iPW1hc3RlcildKGh0dHBzOi8vc2NydXRpbml6ZXItY2kuY29tL2cvUm9kZW5hc3R5bGUvbGFyYXZlbC10ZXN0ZG9jL2J1aWxkLXN0YXR1cy9tYXN0ZXIpClshW0xhdGVzdCBWZXJzaW9uIG9uIFBhY2thZ2lzdF0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9wYWNrYWdpc3Qvdi9yb2RlbmFzdHlsZS9sYXJhdmVsLXRlc3Rkb2Muc3ZnP3N0eWxlPWZsYXQtc3F1YXJlKV0oaHR0cHM6Ly9wYWNrYWdpc3Qub3JnL3BhY2thZ2VzL3JvZGVuYXN0eWxlL2xhcmF2ZWwtdGVzdGRvYykKWyFbUXVhbGl0eSBTY29yZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9zY3J1dGluaXplci9nL3JvZGVuYXN0eWxlL2xhcmF2ZWwtdGVzdGRvYy5zdmc/c3R5bGU9ZmxhdC1zcXVhcmUpXShodHRwczovL3NjcnV0aW5pemVyLWNpLmNvbS9nL1JvZGVuYXN0eWxlL2xhcmF2ZWwtdGVzdGRvYy8/YnJhbmNoPW1hc3RlcikKWyFbQ29kZSBDb3ZlcmFnZV0oaHR0cHM6Ly9zY3J1dGluaXplci1jaS5jb20vZy9Sb2RlbmFzdHlsZS9sYXJhdmVsLXRlc3Rkb2MvYmFkZ2VzL2NvdmVyYWdlLnBuZz9iPW1hc3RlcildKGh0dHBzOi8vc2NydXRpbml6ZXItY2kuY29tL2cvUm9kZW5hc3R5bGUvbGFyYXZlbC10ZXN0ZG9jLz9icmFuY2g9bWFzdGVyKQpbIVtMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL3BhY2thZ2lzdC9sL1JvZGVuYXN0eWxlL2xhcmF2ZWwtdGVzdGRvYy5zdmcpXShodHRwczovL3BhY2thZ2lzdC5vcmcvcGFja2FnZXMvUm9kZW5hc3R5bGUvbGFyYXZlbC10ZXN0ZG9jKQoKIyMgTGljZW5zZQpUaGlzIGxpYnJhcnkgaXMgcmVsZWFzZWQgdW5kZXIgW01JVF0oaHR0cDovL3d3dy50bGRybGVnYWwuY29tL2xpY2Vuc2UvbWl0LWxpY2Vuc2UpIGxpY2Vuc2Uu readmeEtag: '"a8914aed3e183ac97d50debbd8a6e39aca1f0441"' readmeLastModified: Thu, 14 Feb 2019 14:43:52 GMT repositoryId: 166667642 description: 💎🔒 Laravel Functional Tests Docs Generator created: '2019-01-20T13:59:16Z' updated: '2019-02-14T14:44:10Z' language: PHP archived: false stars: 0 watchers: 1 forks: 1 owner: sergiorodenas logo: https://avatars.githubusercontent.com/u/2689890?v=4 license: MIT repoEtag: '"b2838751c3444236cd8494498648c16c910cb4a4a0c44777e7b06b42de0e8835"' repoLastModified: Thu, 14 Feb 2019 14:44:10 GMT foundInMaster: true category: Testing id: 70c8bd1fe35383a5fb2454caa5d06703 - source: openapi3 tags repository: https://github.com/pauliorandall/go-qlueless-api v3: true repositoryMetadata: base64Readme: >- IyBHbyBRbHVlbGVzcyBBUEkKClwqXCpcKioqRGlzY29udGludWVkOiBhcmNoaXZlZCBmb3IgcmVmZXJlbmNlIG9ubHkuKipcKlwqXCoKCkEgR28gaW1wbGVtZW50YXRpb24gb2YgYSBzaW1wbGUgQVBJIHRvIHN0b3JlIGFuZCBhY2Nlc3MgS2FuYmFuIHJlbGF0ZWQgZW50aXRpZXMgYW5kIGV2ZW50cy4KCi0gVGhpcyBwcm9qZWN0IGlzIHVuZGVydGFrZW4gd2l0aCB0aGUgYXVkaWJsZSBhaWQgb2YgW0F2YW50YXNpYV0oaHR0cHM6Ly93d3cuYXZhbnRhc2lhLm5ldCkgYW5kIFtEcmVhbSBUaGVhdGVyXShodHRwOi8vZHJlYW10aGVhdGVyLm5ldCkKLSBUaGlzIFJFQURNRSB3YXMgc3RydWN0dXJlZCBvbiBhIHRlbXBsYXRlIGJ5IFtQdXJwbGVCb290aF0oaHR0cHM6Ly9naXN0LmdpdGh1Yi5jb20vUHVycGxlQm9vdGgvMTA5MzExYmIwMzYxZjMyZDg3YTIpCgojIyBXaHkgaXMgdGhlIHByb2plY3QgY2FsbGVkICdRbHVlbGVzcyc/CgpJIGFwb2xvZ2lzZSwgaXQncyBhIHBvb3IgcGxheSBvbiB3b3JkcyBhdHRlbXB0aW5nIHRvIGNvbWJpbmU6CgoxLiBgQ2x1ZWxlc3NgOiBPbmUgb2YgdGhlIHByb2plY3RzIHB1cnBvc2VzIGlzIHRvIGxlYXJuIGFuZCBleHBlcmltZW50IHdpdGggdGVjaG5vbG9naWVzIHN1Y2ggYXMgR28sIFJlYWN0LCBhbmQgcHVibGljIHBpcGVsaW5lIHRvb2xzOyBJJ20gbW9kZXJhdGVseSBjbHVlbGVzcyBhYm91dCB0aGUgbGF0dGVyIHR3by4KMi4gYFF1ZXVlbGVzc2A6IEkgd2FudCB0byBleHBlcmltZW50IHdpdGggd2F5cyBvZiB2aXN1YWxpc2luZyBhbmQgZW1waGFzaXNpbmcgd29yayBpbiBwcm9ncmVzcyB0aGF0IGlzIG5vdCwgaW4gZmFjdCwgYmVpbmcgcHJvZ3Jlc3NlZCwgaS5lLiBoYWxmIGZpbmlzaGVkIHdvcmsgc2l0dGluZyBpbiBxdWV1ZXMgd2FpdGluZyBmb3Igc29tZW9uZSB0byBmaW5pc2ggdGhlbS4gT25jZSB2aXNpYmxlIGFuZCBiZWluZyBtZWFzdXJlZCBJIGNhbiBzdGFydCB0byBhbmFseXplIGl0LCBhbmQgZXhwZXJpbWVudCB3aXRoIHdheXMgb2YgcmVkdWNpbmcgYW5kIGF2b2lkaW5nIGl0IQoKIyMgR2V0dGluZyBTdGFydGVkCgojIyMgUHJlcmVxdWlzaXRlcwoKLSBHbzogW2h0dHBzOi8vZ29sYW5nLm9yZy9kbC9dCi0gR2l0OiBbaHR0cHM6Ly9naXQtc2NtLmNvbV0KLSBBbiBpbnRlcm5ldCBjb25uZWN0aW9uCi0gQSBkZWNlbnQgd2ViIGJyb3dzZXIKCiMjIyBSdW5uaW5nCgpOYXZpZ2F0ZSB0byBhIHN1aXRhYmxlIGRpcmVjdG9yeSwgb3BlbiBhIHRlcm1pbmFsLCBhbmQgY29weStwYXN0ZSB0aGUgZm9sbG93aW5nOgoKYGBgCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vUGF1bGlvUmFuZGFsbC9nby1xbHVlbGVzcy1hcGkuZ2l0CmNkIGdvLXFsdWVsZXNzLWFwaS9zY3JpcHRzCi4vYnVpbGQtdGVzdC1hcGktcnVuLmdvCmBgYAoKSW4gb3JkZXIsIHRoaXMgd2lsbDoKCjEuIENsb25lIHRoZSBzb3VyY2UgY29kZSByZXBvc2l0b3J5CjIuIE5hdmlnYXRlIHRvIHRoZSB1c2VyIGBzY3JpcHRzYCBkaXJlY3RvcnkKMy4gRXhlY3V0ZSBhIGJ1aWxkIG9mIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24KNC4gRXhlY3V0ZSBhIGJ1aWxkIG9mIHRoZSBhcHBsaWNhdGlvbgo1LiBFeGVjdXRlIHVuaXQgdGVzdHMgd2l0aGluIHRoZSBhcHBsaWNhdGlvbgo2LiBFeGVjdXRlIGJsYWNrIGJveCBBUEkgdGVzdHMgb24gdGhlIGFwcGxpY2F0aW9uCjcuIFN0YXJ0cyB0aGUgYXBwbGljYXRpb24KCiMjIyBSdW5uaW5nIHVuaXQgdGVzdHMKCk9wZW4gYSB0ZXJtaW5hbCBhdCB0aGUgcHJvamVjdCByb290OgoKYGBgCmNkIC9zY3JpcHRzCi4vYnVpbGQtdGVzdC5nbwpgYGAKCiMjIyBSdW5uaW5nIEFQSSB0ZXN0cwoKT3BlbiBhIHRlcm1pbmFsIGF0IHRoZSBwcm9qZWN0IHJvb3Q6CgpgYGAKY2QgL3NjcmlwdHMKLi9idWlsZC10ZXN0LWFwaS5nbwpgYGAKCiMjIyBEZXBsb3ltZW50IAoKPiBDb21pbmcgc29vbiEgU2VlICoqUnVubmluZyoqIGluIHRoZSBtZWFudGltZS4KCiMjIEJ1aWx0IFdpdGgKCi0gW09wZW5BUEldKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vYWJvdXQvKQotIFtHb10oaHR0cHM6Ly9nb2xhbmcub3JnKQotIFt0ZXN0aWZ5XShodHRwczovL2dpdGh1Yi5jb20vc3RyZXRjaHIvdGVzdGlmeSkKLSBbbWFwc3RydWN0dXJlXShodHRwczovL2dpdGh1Yi5jb20vbWl0Y2hlbGxoL21hcHN0cnVjdHVyZSkKCiMjIENvbnRyaWJ1dGluZwoKPiBOb3QgYXBwbGljYWJsZS4KCiMjIFZlcnNpb25pbmcKClRoaXMgcHJvamVjdHMgQVBJIFtDSEFOR0VMT0ddKGh0dHBzOi8vZ2l0aHViLmNvbS9QYXVsaW9SYW5kYWxsL2dvLXFsdWVsZXNzLWFwaS9ibG9iL21hc3Rlci9hcGkvQ0hBTkdFTE9HLm1kKSBmb3JtYXQgaXMgYmFzZWQgb24gW0tlZXAgYSBDaGFuZ2Vsb2ddKGh0dHBzOi8va2VlcGFjaGFuZ2Vsb2cuY29tL2VuLzEuMC4wLyksIGFuZCB0aGUgQVBJIGFkaGVyZXMgdG8gW1NlbWFudGljIFZlcnNpb25pbmddKGh0dHBzOi8vc2VtdmVyLm9yZy9zcGVjL3YyLjAuMC5odG1sKS4KCiMjIEF1dGhvcnMKCi0gW01lXShodHRwczovL2dpdGh1Yi5jb20vUGF1bGlvUmFuZGFsbCkKCiMjIExpY2Vuc2UKClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXShodHRwczovL2dpdGh1Yi5jb20vUGF1bGlvUmFuZGFsbC9nby1xbHVlbGVzcy1hcGkvYmxvYi9tYXN0ZXIvTElDRU5TRSkuCgojIyBBY2tub3dsZWRnbWVudHMKCi0gSW5mbHVlbmNlcwogIC0gJ1RoZSBHb2FsJyBieSBFbGl5aGFodSBNLiBHb2xkcmF0dAogIC0gQ29udGludW91cyBJbnRlZ3JhdGlvbgogIC0gQ29udGludW91cyBEZWxpdmVyeQo= readmeEtag: '"01258343092071f176e349ea063f7ac82371e8bb"' readmeLastModified: Thu, 23 Jan 2020 15:01:18 GMT repositoryId: 174692654 description: >- *Discontinued* Experimenting with ideas by building a simple Go web server exposing a TODO lists created: '2019-03-09T12:38:15Z' updated: '2023-01-28T16:01:15Z' language: Go archived: true stars: 0 watchers: 0 forks: 0 owner: PaulioRandall logo: https://avatars.githubusercontent.com/u/29413800?v=4 license: MIT repoEtag: '"d041cabdc8e8ff668b4667d24354333d75609dc53407631df034148760cfa49a"' repoLastModified: Sat, 28 Jan 2023 16:01:15 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 62d045a14fd3d45cc95b98891af82e47 - source: openapi3 tags repository: https://github.com/senor-artemisio/tendershins v3: true repositoryMetadata: base64Readme: >- IyB0ZW5kZXJzaGlucwoKV3JhcHBlciBmb3Igc2hpbnMgZ2VuZXJhdG9yczoKCiogW1dpZGRlcnNoaW5zXShodHRwczovL2dpdGh1Yi5jb20vTWVybWFkZS93aWRkZXJzaGlucykKKiBbU2hpbnNdKGh0dHBzOi8vZ2l0aHViLmNvbS9NZXJtYWRlL3NoaW5zKQoKR2VuZXJhdGVzIEhUTUwgZG9jdW1lbnRhdGlvbiBmcm9tIE9wZW5BUEkgLyBTd2FnZ2VyIC8gQXN5bmNBUEkgLyBTZW1vYXNhIGRlZmluaXRpb24uCgojIyBUbyBpbnN0YWxsCgpDbG9uZSB0aGUgZ2l0IHJlcG9zaXRvcnksIG9yIHVzZSBucG0gaW5zdGFsbApgYGAKbnBtIGluc3RhbGwgdGVuZGVyc2hpbnMgLS1zYXZlCmBgYAoKIyMgVXNhZ2UKClRvIGdlbmVyYXRlIGRvY3VtZW50YXRpb24gY3JlYXRlIGEgc3BlY2lmaWNhdGlvbiBmaWxlIGluIHRoZSBzdXBwb3J0ZWQgZm9ybWF0IGFuZCBydW4gdGVuZGVyc2hpbnM6CgpgYGAKbnB4IHRlbmRlcnNoaW5zIDxzcGVjX2ZpbGU+IDx0YXJnZV9wYXRoPiBbbG9nb19wYXRoXQpgYGAKCkV4YW1wbGVzIG9mIHNwZWNpZmljYXRpb24gZmlsZXMgc2VlIGluIFtPcGVuQVBJIHJlcG9zaXRvcnldKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL3RyZWUvbWFzdGVyL2V4YW1wbGVzL3YzLjApLiAKCg== readmeEtag: '"58461ac1021e0eabc2af0083eb1619128be5ea89"' readmeLastModified: Thu, 25 Jul 2019 08:52:04 GMT repositoryId: 188308754 description: Wrapper for shins documentation generators created: '2019-05-23T21:24:25Z' updated: '2019-07-25T08:52:07Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: senor-artemisio logo: https://avatars.githubusercontent.com/u/1178789?v=4 license: MIT repoEtag: '"c8b7d1be07b9ade65bfbf5d267752948398105e17ea8838cf279aff24bfb5b5e"' repoLastModified: Thu, 25 Jul 2019 08:52:07 GMT foundInMaster: true category: - Documentation - Parsers id: 80bfcd36c0804b44af58f551da173a30 - source: openapi3 tags repository: https://github.com/sloopsight-com/sandbox v3: true repositoryMetadata: base64Readme: >- IyBUaGUgU2FuZGJveAojIyBBUEkgc2FuZGJveCB0byBzcGVlZCB1cCBpbnRlZ3JhdGlvbnMuCgpBUEkgU2FuZG94IHwgRW5hYmxlIHlvdSBjYW4gdG8gY3JlYXRlIEFQSSBzdHViIG9ubGluZSB3aXRoIG9wZW5hcGkgMy4wIHNwZWMuClN1cHBvcnRzIGxkYXAgYXV0aGVudGljYXRpb24uCkJ1aXQgb24gCgoKIyMgTW9kdWxlcwoKLSBCYWNrZW5kCi0gVUkKey5sbmtzLWxpc3R9CgojIyBUZWNobm9sb2d5IFN0YWNrCgotIFtWdWVKcyAqRnJvbnRlbmQgZnJhbWV3b3JrKl0oIykKLSBbU3ByaW5nYm9vdCAqRm9yIGJ1c2luZXNzIEFwaSpdKCMpCi0gW0NhbWVsIFNlcnZsZXQgKkZvciBzYW5kYm94IGFwaSByb3V0aW5nKl0oIykKLSBbTmFzaGhvcm4gKkZvciBBUEkgQnVzaW5lc3MgTG9naWMgRXhlY3V0aW9uKl0oIykKLSBbSFNRTC9NeXNxbCAqRm9yIGZvciBzdG9yaW5nIGRhdGEqXSgjKQotIFtTd2F5ZGIgKlRUTCBiYXNlZCBrZXktdmFsdWUgc3RvcmUqXShodHRwOi8vc3dheWRiLmlvLz9sYW5ndWFnZT1zY2FsYS8pCi0gW1N3YWdnZXIgRG9jdW1lbnRhdGlvbiAqVG8gZ2VuZXJhdGUgbW9ja2VkIGFwaSBkb2N1bWVudGF0aW9uKl0oaHR0cHM6Ly9naXRodWIuY29tL3NvbmdyZ2cvc3dhZ2dlcmRlbW8pCg== readmeEtag: '"914f1619dccb7147fd7956f11c92e55ee70976dd"' readmeLastModified: Wed, 07 Jul 2021 10:50:27 GMT repositoryId: 281899357 description: >- API Sandox | Enable you can to create API stub online with openapi 3.0 spec. created: '2020-07-23T08:46:24Z' updated: '2021-08-11T07:17:06Z' language: Vue archived: false stars: 0 watchers: 0 forks: 0 owner: sloopsight-com logo: https://avatars.githubusercontent.com/u/68685917?v=4 repoEtag: '"61d96fa18980975ae33674f186138b21880a52432a0a95addc43c430c396fe1c"' repoLastModified: Wed, 11 Aug 2021 07:17:06 GMT foundInMaster: true category: - Server - Server Implementations id: 57b7f4b92935a93a909854865e1cb20b - source: openapi3 tags repository: https://github.com/lucasavila00/beff v3: true id: c1cab811e9accd4842010fbcdbfc7bcc repositoryMetadata: base64Readme: >- IyDwn6qEIEJlZmYKCkJlZmYgaXMgYSBoaWdoLXBlcmZvcm1hbmNlIHZhbGlkYXRvciBnZW5lcmF0b3IgdGhhdCBjcmVhdGVzIGVmZmljaWVudCBydW50aW1lIHZhbGlkYXRvcnMgZnJvbSBUeXBlU2NyaXB0IHR5cGVzIHVzaW5nIGEgYmxhemluZy1mYXN0IGNvbXBpbGVyLgoKIyMgV2h5IENob29zZSBCZWZmPwoKLSAqKlR5cGVTY3JpcHQgRmlyc3QqKjogVW5saWtlIHpvZCwgaW8tdHMsIGFuZCBzaW1pbGFyIGxpYnJhcmllcywgQmVmZiBkb2Vzbid0IGFkZCBvdmVyaGVhZCB0byB0aGUgVHlwZVNjcmlwdCBjb21waWxlci4gRXhwZXJpZW5jZSBmYXN0ZXIgZWRpdG9yIHBlcmZvcm1hbmNlIGFuZCBxdWlja2VyIGNvbXBpbGUgdGltZXMuCi0gKipCbGF6aW5nbHkgRmFzdCoqOiBXcml0dGVuIGluIFJ1c3QgYW5kIGNvbXBpbGVkIHRvIFdlYkFzc2VtYmx5LCBCZWZmIGlzIGNyb3NzLXBsYXRmb3JtIGFuZCBsaWdodG5pbmctcXVpY2suIEl0IGNvbXBpbGVzIGEgaGVsbG8td29ybGQgcHJvamVjdCBpbiA1bXMgYW5kIGhhbmRsZXMgbGFyZ2UgcHJvamVjdHMgd2l0aCAyMDArIHR5cGVzIGluIGp1c3QgMjAwbXMuCi0gKipTZWFtbGVzc2x5IENvbXBhdGlibGUqKjogTGV2ZXJhZ2VzIHRoZSBUeXBlU2NyaXB0IGNvbXBpbGVyIGZvciBwYXRoIHJlc29sdXRpb24uIElmIHlvdXIgZWRpdG9yIGNhbiByZXNvbHZlIHRoZSB0eXBlcywgQmVmZiBjYW4gdG9vLgotICoqT3B0aW1pemVkIE91dHB1dCoqOiBHZW5lcmF0ZXMgaGlnaGx5IGVmZmljaWVudCB2YWxpZGF0b3IgY29kZSB3aXRoIGV4dGVuc2l2ZSBjb21waWxlLXRpbWUgb3B0aW1pemF0aW9ucy4KLSAqKkRldmVsb3BlciBGcmllbmRseSoqOiBQcm92aWRlcyBjbGVhciwgYWN0aW9uYWJsZSBlcnJvciBtZXNzYWdlcyBhdCBib3RoIGNvbXBpbGUgdGltZSBhbmQgcnVudGltZS4KLSAqKkZlYXR1cmUgQ29tcGxldGUqKjogU3VwcG9ydHMgYWR2YW5jZWQgVHlwZVNjcmlwdCBmZWF0dXJlcyBpbmNsdWRpbmcgcmVjdXJzaXZlIHR5cGVzLCBnZW5lcmljcywgbWFwcGVkIHR5cGVzLCBjb25kaXRpb25hbCB0eXBlcywgdXRpbGl0eSB0eXBlcyAoYE9taXRgLCBgRXhjbHVkZWAsIGBQYXJ0aWFsYCwgYFJlcXVpcmVkYCwgYFJlY29yZGApLCBhbmQgbW9yZS4gSWYgYSB0eXBlIGNhbiBiZSB2YWxpZGF0ZWQgYXQgcnVudGltZSwgQmVmZiB1bmRlcnN0YW5kcyBpdC4KCiMjIEdldHRpbmcgU3RhcnRlZAoKR2V0IHVwIGFuZCBydW5uaW5nIHdpdGggQmVmZiBpbiBqdXN0IGEgZmV3IHNpbXBsZSBzdGVwczoKCiMjIyAxLiBJbnN0YWxsCgpJbnN0YWxsIHRoZSByZXF1aXJlZCBwYWNrYWdlcyBmcm9tIG5wbToKCmBgYHNoZWxsCm5wbSBpIEBiZWZmL2NsaSBAYmVmZi9jbGllbnQKYGBgCgojIyMgMi4gQ29uZmlndXJlCgpDcmVhdGUgYSBKU09OIGZpbGUgdG8gY29uZmlndXJlIEJlZmYuIFRoZSBmaWxlIGNhbiBoYXZlIGFueSBuYW1lLCBidXQgaXQncyBzdGFuZGFyZCBwcmFjdGljZSB0byBuYW1lIGl0IGBiZWZmLmpzb25gLgoKYGBganNvbgp7CiAgInBhcnNlciI6ICIuL3NyYy9wYXJzZXIudHMiLAogICJvdXRwdXREaXIiOiAiLi9zcmMvZ2VuZXJhdGVkIgp9CmBgYAoKIyMjIDMuIENyZWF0ZSB0aGUgcGFyc2VyIGZpbGUKCkNyZWF0ZSBhIFR5cGVTY3JpcHQgZmlsZSB0aGF0IGV4cG9ydHMgdGhlIHR5cGVzIHlvdSB3YW50IEJlZmYgdG8gZ2VuZXJhdGUgdmFsaWRhdG9ycyBmb3IuCgpCeSBjb252ZW50aW9uLCB0aGlzIGZpbGUgaXMgdHlwaWNhbGx5IG5hbWVkIGBwYXJzZXIudHNgOgoKYGBgdHMKaW1wb3J0IHBhcnNlIGZyb20gIi4vZ2VuZXJhdGVkL3BhcnNlciI7Cgp0eXBlIFVzZXIgPSB7CiAgbmFtZTogc3RyaW5nOwogIGFnZTogbnVtYmVyOwp9OwoKZXhwb3J0IGNvbnN0IFBhcnNlcnMgPSBwYXJzZS5idWlsZFBhcnNlcnM8ewogIFVzZXI6IFVzZXI7Cn0+KCk7CmBgYAoKIyMjIDQuIEdlbmVyYXRlIHRoZSB2YWxpZGF0b3JzCgpSdW4gdGhlIEJlZmYgQ0xJIHRvIGdlbmVyYXRlIHlvdXIgdmFsaWRhdG9yIGNvZGU6CgpgYGBzaGVsbApucHggYmVmZiAtcCBiZWZmLmpzb24KYGBgCgojIyMgNS4gVXNlIHRoZSB2YWxpZGF0b3JzCgpOb3cgeW91IGNhbiB1c2UgdGhlIGdlbmVyYXRlZCB2YWxpZGF0b3JzIGluIHlvdXIgYXBwbGljYXRpb246CgpgYGB0cwppbXBvcnQgeyBQYXJzZXJzIH0gZnJvbSAiLi9wYXJzZXIudHMiOwoKY29uc3QgdXNlcjEgPSBQYXJzZXJzLlVzZXIucGFyc2UoewogIG5hbWU6ICJKb2huIERvZSIsCiAgYWdlOiA0MiwKfSk7Cgpjb25zdCBtYXliZVVzZXIgPSBQYXJzZXJzLlVzZXIuc2FmZVBhcnNlKG51bGwpOwoKY29uc3QgaXNWYWxpZDogYm9vbGVhbiA9IFBhcnNlcnMuVXNlci52YWxpZGF0ZSh7CiAgbmFtZTogIkpvaG4gRG9lIiwKICBhZ2U6IDQyLAp9KTsKCmNvbnN0IGpzb25TY2hlbWEgPSBQYXJzZXJzLlVzZXIuc2NoZW1hKCk7CmBgYAoKIyMgQ0xJIE9wdGlvbnMKClRoZSBgYmVmZmAgYmluYXJ5IGNhbiBhbHNvIHJ1biBpbiB3YXRjaCBtb2RlLgoKYGBgc2hlbGwKJCBucHggYmVmZiAtaApVc2FnZTogYmVmZiBbb3B0aW9uc10KCkdlbmVyYXRlIHZhbGlkYXRvcnMgZnJvbSBUeXBlU2NyaXB0IHR5cGVzCgpPcHRpb25zOgogIC1wLCAtLXByb2plY3QgPHN0cmluZz4gIFBhdGggdG8gdGhlIHByb2plY3QgZmlsZQogIC12LCAtLXZlcmJvc2UgICAgICAgICAgIFByaW50IHZlcmJvc2Ugb3V0cHV0CiAgLXcsIC0td2F0Y2ggICAgICAgICAgICAgV2F0Y2ggZm9yIGZpbGUgY2hhbmdlcwogIC1oLCAtLWhlbHAgICAgICAgICAgICAgIGRpc3BsYXkgaGVscCBmb3IgY29tbWFuZApgYGAKCiMjIEFkdmFuY2VkIEZlYXR1cmVzCgojIyMgQ3VzdG9tIFN0cmluZyBGb3JtYXRzCgpCZWZmIGFsbG93cyB5b3UgdG8gZGVmaW5lIGN1c3RvbSBzdHJpbmcgdmFsaWRhdGlvbiBmb3JtYXRzLiBGaXJzdCwgY29uZmlndXJlIHlvdXIgYGJlZmYuanNvbmA6CgpgYGBqc29uCnsKICAicGFyc2VyIjogIi4vc3JjL3BhcnNlci50cyIsCiAgIm91dHB1dERpciI6ICIuL3NyYy9nZW5lcmF0ZWQiLAogICJzdHJpbmdGb3JtYXRzIjogWwogICAgewogICAgICAibmFtZSI6ICJWYWxpZEN1cnJlbmN5IgogICAgfQogIF0KfQpgYGAKClRoZW4gdXNlIHRoZSBgU3RyaW5nRm9ybWF0YCBoZWxwZXIgdG8gY3JlYXRlIGEgYnJhbmRlZCBUeXBlU2NyaXB0IHR5cGUgYW5kIGRlZmluZSB0aGUgcnVudGltZSB2YWxpZGF0b3I6CgpgYGB0cwppbXBvcnQgcGFyc2UgZnJvbSAiLi9nZW5lcmF0ZWQvcGFyc2VyIjsKaW1wb3J0IHsgU3RyaW5nRm9ybWF0IH0gZnJvbSAiQGJlZmYvY2xpZW50IjsKZXhwb3J0IHR5cGUgVmFsaWRDdXJyZW5jeSA9IFN0cmluZ0Zvcm1hdDwiVmFsaWRDdXJyZW5jeSI+OwoKZXhwb3J0IGNvbnN0IFBhcnNlcnMgPSBwYXJzZS5idWlsZFBhcnNlcnM8ewogIFZhbGlkQ3VycmVuY3k6IFZhbGlkQ3VycmVuY3k7Cn0+KHsKICBzdHJpbmdGb3JtYXRzOiB7CiAgICBWYWxpZEN1cnJlbmN5OiAoaW5wdXQ6IHN0cmluZykgPT4gewogICAgICBpZiAoVkFMSURfQ1VSUkVOQ0lFUy5pbmNsdWRlKGlucHV0KSkgewogICAgICAgIHJldHVybiB0cnVlOwogICAgICB9CiAgICAgIHJldHVybiBmYWxzZTsKICAgIH0sCiAgfSwKfSk7CmBgYAoKIyMjIEN1c3RvbSBOdW1iZXIgRm9ybWF0cwoKU2ltaWxhcmx5LCB5b3UgY2FuIGRlZmluZSBjdXN0b20gbnVtYmVyIHZhbGlkYXRpb24gZm9ybWF0cy4gQ29uZmlndXJlIHlvdXIgYGJlZmYuanNvbmA6CgpgYGBqc29uCnsKICAicGFyc2VyIjogIi4vc3JjL3BhcnNlci50cyIsCiAgIm91dHB1dERpciI6ICIuL3NyYy9nZW5lcmF0ZWQiLAogICJudW1iZXJGb3JtYXRzIjogWwogICAgewogICAgICAibmFtZSI6ICJWYWxpZEN1cnJlbmN5IgogICAgfQogIF0KfQpgYGAKClRoZW4gdXNlIHRoZSBgTnVtYmVyRm9ybWF0YCBoZWxwZXIgdG8gY3JlYXRlIGEgYnJhbmRlZCBUeXBlU2NyaXB0IHR5cGUgYW5kIGRlZmluZSB0aGUgcnVudGltZSB2YWxpZGF0b3I6CgpgYGB0cwppbXBvcnQgcGFyc2UgZnJvbSAiLi9nZW5lcmF0ZWQvcGFyc2VyIjsKaW1wb3J0IHsgTnVtYmVyRm9ybWF0IH0gZnJvbSAiQGJlZmYvY2xpZW50IjsKZXhwb3J0IHR5cGUgTm9uTmVnYXRpdmVOdW1iZXIgPSBOdW1iZXJGb3JtYXQ8Ik5vbk5lZ2F0aXZlTnVtYmVyIj47CgpleHBvcnQgY29uc3QgUGFyc2VycyA9IHBhcnNlLmJ1aWxkUGFyc2Vyczx7CiAgTm9uTmVnYXRpdmVOdW1iZXI6IE5vbk5lZ2F0aXZlTnVtYmVyOwp9Pih7CiAgbnVtYmVyRm9ybWF0czogewogICAgTm9uTmVnYXRpdmVOdW1iZXI6IChpbnB1dDogbnVtYmVyKSA9PiB7CiAgICAgIHJldHVybiBpbnB1dCA+PSAwOwogICAgfSwKICB9LAp9KTsKYGBgCgojIyMgQWQtaG9jIFZhbGlkYXRvciBDcmVhdGlvbgoKQmVmZiBwcm92aWRlcyBhIHJ1bnRpbWUgdHlwZSBjcmVhdGlvbiBBUEkgc2ltaWxhciB0byBgem9kYCBhbmQgYGlvLXRzYCBmb3Igc2ltcGxlIHVzZSBjYXNlcy4KCioqTm90ZToqKiBUaGlzIEFQSSBpcyBpbnRlbnRpb25hbGx5IGxpbWl0ZWQgYW5kIHN1cHBvcnRzIG9ubHkgYmFzaWMgdHlwZXMgdG8gbWFpbnRhaW4gZmFzdCBlZGl0b3IgcGVyZm9ybWFuY2UuIEEgbW9yZSBjb21wcmVoZW5zaXZlIHJ1bnRpbWUgQVBJIHdvdWxkIGVzc2VudGlhbGx5IG1ha2UgQmVmZiBqdXN0IGFub3RoZXIgcmUtaW1wbGVtZW50YXRpb24gb2Ygem9kLCBkZWZlYXRpbmcgaXRzIGNvcmUgcHVycG9zZS4gRm9yIGNvbXBsZXggdHlwZXMsIHVzZSBCZWZmJ3MgcHJpbWFyeSBmZWF0dXJlIG9mIGNvbXBpbGluZyBUeXBlU2NyaXB0IHR5cGVzLgoKVmFsaWRhdG9ycyBjcmVhdGVkIHdpdGggdGhlIGFkLWhvYyBBUEkgaGF2ZSB0aGUgc2FtZSBpbnRlcmZhY2UgYXMgY29tcGlsZWQgdmFsaWRhdG9yczoKCmBgYHRzCmltcG9ydCB7IGIgfSBmcm9tICJAYmVmZi9jbGllbnQiOwoKY29uc3QgQWRIb2NJdGVtID0gYi5PYmplY3QoewogIHN0cjogYi5TdHJpbmcoKSwKICBudW06IGIuTnVtYmVyKCksCiAgYm9vbDogYi5Cb29sZWFuKCksCiAgdW5kZWZpbmVkOiBiLlVuZGVmaW5lZCgpLAogIG51bGw6IGIuTnVsbCgpLAogIGFueTogYi5BbnkoKSwKICB1bmtub3duOiBiLlVua25vd24oKSwKfSk7Cgpjb25zdCBBZEhvY0xpc3QgPSBiLkFycmF5KEFkSG9jSXRlbSk7Cgpjb25zdCBscyA9IEFkSG9jTGlzdC5wYXJzZShbXSk7CmBgYAoKIyMjIFpvZCBDb21wYXRpYmlsaXR5CgpCZWZmIHByb3ZpZGVzIHNlYW1sZXNzIGludGVyb3BlcmFiaWxpdHkgd2l0aCBab2QuIENhbGwgYC56b2QoKWAgb24gYW55IEJlZmYgcGFyc2VyIHRvIGdldCBhIGNvbXBhdGlibGUgWm9kIHNjaGVtYS4KClRoaXMgbWFrZXMgaXQgZWFzeSB0byBncmFkdWFsbHkgbWlncmF0ZSBleGlzdGluZyBab2QtYmFzZWQgY29kZWJhc2VzOgoKYGBgdHMKaW1wb3J0IHsgUGFyc2VycyB9IGZyb20gIi4vcGFyc2VyLnRzIjsKaW1wb3J0IHsgeiB9IGZyb20gInpvZCI7Cgpjb25zdCB1c2VycyA9IHouYXJyYXkoUGFyc2Vycy5Vc2VyLnpvZCgpKS5wYXJzZSh7CiAgbmFtZTogIkpvaG4gRG9lIiwKICBhZ2U6IDQyLAp9KTsKYGBgCgojIyBDb250cmlidXRpbmcKClBsZWFzZSByZWFkIFtDT05UUklCVVRJTkcubWRdKC9DT05UUklCVVRJTkcubWQpCg== readmeEtag: '"8db14553f03f82b6e12974ea74d2d7dc017b1340"' readmeLastModified: Fri, 03 Oct 2025 00:29:36 GMT repositoryId: 683896092 description: >- Efficient validators from Typescript types generated by a blazing fast compiler created: '2023-08-28T02:32:08Z' updated: '2026-01-09T15:37:20Z' language: Rust archived: false stars: 6 watchers: 1 forks: 1 owner: lucasavila00 logo: https://avatars.githubusercontent.com/u/12750442?v=4 repoEtag: '"28ded80821b5e18986400e1eca0fc87dca9255776f4a91db5b5bb7e3a80fe209"' repoLastModified: Fri, 09 Jan 2026 15:37:20 GMT category: - Server Implementations - Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/modyuan/openapitomarkdown v3: true repositoryMetadata: base64Readme: >- IyBvcGVuYXBpVG9NYXJrZG93bgpjb252ZXJ0IG9wZW5BcGkzIGpzb24gZmlsZSB0byBtYXJrZG93biBmaWxlCgrmiopPcGVuQVBJM+eahGpzb27moLzlvI/vvIwg6L2s5o2i5Li6bWFya2Rvd27moLzlvI/vvIzmlrnkvr/pnZnmgIHlgqjlrZjjgIIKCuaehOaIkOaWueazleWPguiAg+S6hltzd2FnZ2VyLWJvb3RzdHJhcC11aV0oaHR0cHM6Ly9naXRodWIuY29tL3hpYW95bWluL3N3YWdnZXItYm9vdHN0cmFwLXVpKQoK readmeEtag: '"8255735b14cf295527465f2296b79918227f7116"' readmeLastModified: Sat, 11 Jul 2020 23:53:38 GMT repositoryId: 278960180 description: convert openApi3 json file to markdown file created: '2020-07-11T23:50:06Z' updated: '2020-07-11T23:55:19Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: modyuan logo: https://avatars.githubusercontent.com/u/18202493?v=4 license: MIT repoEtag: '"5ab932b016f6fe8a23c1202576a6a6d3d64593743e6eab34c5c83b8b98c79188"' repoLastModified: Sat, 11 Jul 2020 23:55:19 GMT foundInMaster: true category: - Documentation - Parsers id: 0c2f2d8d4ff22b3acb1bc1233560000a - source: openapi3 tags repository: https://github.com/hw0k-playground/openapi-codegen-example v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIENvZGVnZW4gZm9yIFR5cGVTY3JpcHQKCiMjIyBTZXR1cAoKMS4gQ2xvbmUgdGhpcyByZXBvCjIuIFJ1biBgeWFybmAKMy4gUnVuIGB5YXJuIGdlbmVyYXRlOm9wZW5hcGktdHlwZXNgCjQuIENoZWNrIGRpcmVjdG9yeSBgX19nZW5lcmF0ZWRfXy9gCg== readmeEtag: '"e49f9264ee329f9b636a9a0468d53d3932deb713"' readmeLastModified: Sun, 14 Mar 2021 17:35:03 GMT repositoryId: 347705066 description: OpenAPI Codegen for TypeScript created: '2021-03-14T17:28:09Z' updated: '2021-03-14T17:35:05Z' language: JavaScript archived: false stars: 0 watchers: 0 forks: 0 owner: hw0k-playground logo: https://avatars.githubusercontent.com/u/67791654?v=4 repoEtag: '"53e4912cadd3a17705bd0507e5a8b6797b2055322729861edb0845dd559ac962"' repoLastModified: Sun, 14 Mar 2021 17:35:05 GMT foundInMaster: true category: - Code Generators - Server Implementations id: b740e2ac3a4f0991e59f7e7e3387ff6f oldLocations: - https://github.com/hw0k-play/openapi-codegen-example - source: openapi3 tags repository: https://github.com/tumanina/reportingsystem v3: true repositoryMetadata: repositoryId: 311774106 description: null created: '2020-11-10T20:15:25Z' updated: '2021-03-07T13:22:56Z' language: C# archived: false stars: 0 watchers: 1 forks: 0 owner: tumanina logo: https://avatars.githubusercontent.com/u/17797666?v=4 repoEtag: '"e35abd9e84efa85c837697d3f8a66091fd911a7f136a2c920636ed90eaeced8f"' repoLastModified: Sun, 07 Mar 2021 13:22:56 GMT foundInMaster: true id: 70ffeb9c48f7540976ec44d120898b27 - source: openapi3 tags repository: https://github.com/mikuhuyo/sms-service v3: true id: 9471c9b57775e278d57d1cf4ac284039 repositoryMetadata: base64Readme: >- IyDnn63kv6HmnI3liqHkvb/nlKjmiYvlhowKClshW0dpdEh1YiBsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL21pa3VodXlvL3Ntcy1zZXJ2aWNlKV0oaHR0cHM6Ly9naXRodWIuY29tL21pa3VodXlvL3Ntcy1zZXJ2aWNlL2Jsb2IvbWFzdGVyL0xJQ0VOU0UpClshW0dpdEh1YiBpc3N1ZXNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2lzc3Vlcy9taWt1aHV5by9zbXMtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9zbXMtc2VydmljZS9pc3N1ZXMpClshW0dpdEh1YiBzdGFyc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvc3RhcnMvbWlrdWh1eW8vc21zLXNlcnZpY2UpXShodHRwczovL2dpdGh1Yi5jb20vbWlrdWh1eW8vc21zLXNlcnZpY2Uvc3RhcmdhemVycykKWyFbR2l0SHViIGZvcmtzXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9mb3Jrcy9taWt1aHV5by9zbXMtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9zbXMtc2VydmljZS9uZXR3b3JrKQohW0phdmEgdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9KZGstMTEteWVsbG93KQohW1NwcmluZ0Jvb3QgdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9TcHJpbmdCb290LTIuMy4xMi5SRUxFQVNFLWJyaWdodGdyZWVuKQoKIyMg5L+u5pS5cmVkaXPnm7jlhbPphY3nva4KCuWwhuS4i+mdomByZWRpc2Dnm7jlhbPphY3nva7kv67mlLnkuLrkvaDnmoRgcmVkaXNg55u45YWz6YWN572uOgoKYGBgeWFtbApzcHJpbmc6CiAgY2FjaGU6CiAgICB0eXBlOiBSRURJUwogIHJlZGlzOgogICAgaG9zdDogMTI3LjAuMC4xCiAgICBwYXNzd29yZDogeXVlbGltaW52Y0BvdXRsb29rLmNvbQogICAgcG9ydDogNjM3OQogICAgZGF0YWJhc2U6IDAKICAgIHRpbWVvdXQ6IDMwMDBtcwogICAgbGV0dHVjZToKICAgICAgcG9vbDoKICAgICAgICBtYXgtaWRsZTogOAogICAgICAgIG1pbi1pZGxlOiAxCiAgICAgICAgbWF4LWFjdGl2ZTogOAogICAgICAgIG1heC13YWl0OiAzMDAwMG1zCiAgICAgIHNodXRkb3duLXRpbWVvdXQ6IDEwMDAwbXMKYGBgCgojIyDkv67mlLnohb7orq/kupHnm7jlhbPphY3nva4KCuWwhmBhcHBsaWNhdGlvbi55bWxg5paH5Lu25Lit6IW+6K6v5LqR6YWN572u6YOo5YiG5L+u5pS55Li65L2g55qE6YWN572uOgoKYGBgeWFtbApzbXM6CiAgZGVmYXVsdEV4cGlyZTogMzAwCiAgcWNsb3VkOgogICAgIyB5b3VyIGFwcCBpZAogICAgYXBwSWQ6IDAwMDAwMAogICAgYXBwS2V5OiAneW91cl9hcHBfa2V5JwogICAgdGVtcGxhdGVJZDogJ3lvdXJfc21zX3RlbXBsYXRlJwogICAgc2lnbjogJ3lvdXJfdGVuY2VudF9jbG91ZF9zbXNfc2lnbicKICAgICMgdGVuY2VudCBzbXMgYXBpCiAgICB1cmw6ICdodHRwczovL3l1bi50aW0ucXEuY29tL3Y1L3Rsc3Ntc3N2ci9zZW5kc21zJwpgYGAKCiMjIOaOpeWPo+ivt+axggoKPiBodHRwOi8vMTI3LjAuMC4xOjUyMDAwL3N3YWdnZXItdWkuaHRtbAoKPiDojrflj5bpqozor4HnoIEgaHR0cDovLzEyNy4wLjAuMTo1MjAwMC9zbXMvdGVuY2VudC9nZW5lcmF0L3tlZmZlY3RpdmVUaW1lfQoK5Y+C5pWwYGVmZmVjdGl2ZVRpbWVgOiDov4fmnJ/ml7bpl7QsIOWNleS9jeenkiwg5b+F6YCJ6aG5Cgror7fmsYLkvZM6CgpgYGBqc29uCnsKICAibW9iaWxlIjogIjE1NzExMTExMTExIgp9CmBgYAoK6L+U5Zue5YC8OgoKYGBganNvbgp7CiAgImNvZGUiOiAwLAogICJtc2ciOiAic3VjY2VzcyIsCiAgInJlc3VsdCI6IHsKICAgICJrZXkiOiAidGVuY2VudDpzbXM6PyIKICB9Cn0KYGBgCgo+IOmqjOivgeeggeagoemqjCBodHRwOi8vMTI3LjAuMC4xOjUyMDAwL3Ntcy90ZW5jZW50L3ZlcmlmaWNhdGlvbi97a2V5fS97Y29kZX0KCuWPguaVsGBrZXlgOiDnlJ/miJDnmoTnrb7lkI0KCuWPguaVsGBjb2RlYDog55Sf5oiQ55qE6aqM6K+B56CBCgrov5Tlm57lgLw6CgpgYGBqc29uCnsKICAiY29kZSI6IDAsCiAgIm1zZyI6ICJzdWNjZXNzIiwKICAicmVzdWx0IjogdHJ1ZSAvLyB0cnVl6KGo56S65oiQ5YqfLCBmYWxzZeihqOekuuWksei0pQp9CmBgYAoKIyMg54m55Yir6bij6LCiCgojIyMg5YWz5rOo6ICFCgpbIVtTdGFyZ2F6ZXJzIHJlcG8gcm9zdGVyIGZvciBAbWlrdWh1eW8vc21zLXNlcnZpY2VdKGh0dHBzOi8vcmVwb3Jvc3Rlci5jb20vc3RhcnMvbWlrdWh1eW8vc21zLXNlcnZpY2UpXShodHRwczovL2dpdGh1Yi5jb20vbWlrdWh1eW8vc21zLXNlcnZpY2Uvc3RhcmdhemVycykKCiMjIyDmlLbol4/ogIUKClshW0ZvcmtlcnMgcmVwbyByb3N0ZXIgZm9yIEBtaWt1aHV5by9zbXMtc2VydmljZV0oaHR0cHM6Ly9yZXBvcm9zdGVyLmNvbS9mb3Jrcy9taWt1aHV5by9zbXMtc2VydmljZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9taWt1aHV5by9zbXMtc2VydmljZS9uZXR3b3JrL21lbWJlcnMpCgojIyDor7fov5nkuKpi5Zad5p2v5rC0PwoKIVtBbGlwYXldKC4vaW1hZ2UvYWxpcGF5cy5wbmcpCgotLS0KCiFbV2VDaGF0UGF5XSguL2ltYWdlL3dlY2hhdHMucG5nKQ== readmeEtag: '"18c610dbecd677759ff12d4083ffb15351cd0d98"' readmeLastModified: Mon, 27 Feb 2023 12:47:19 GMT repositoryId: 400815235 description: 单纯的短信服务, 目前仅仅集成了腾讯云短信(人在电信外包, 差点没了) created: '2021-08-28T14:39:39Z' updated: '2023-02-27T13:14:52Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: mikuhuyo logo: https://avatars.githubusercontent.com/u/42843191?v=4 repoEtag: '"05b07746fe3134ad60e45aade3d47390f2f264100e29a1d75909d0145a7b434e"' repoLastModified: Mon, 27 Feb 2023 13:14:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/aquanest/atmos-openapi-src v3: true repositoryMetadata: base64Readme: >- IyBhdG1vcy1vcGVuYXBpLXNyYwoKYXRtb3Mtb3BlbmFwaS1zcmMgaXMgYSBzb3VyY2UgZ2VuZXJhdGVzIGxpYnJhcmllcyBmb3IgW0FUTU9TIFBsYXRmb3JtIEFQSV0oaHR0cHM6Ly93d3cuYXRtb3MuYXBwLykuCgpUaGlzIGNvZGUgZ2VuZXJhdGVzIGZvbGxvd2luZyBtb2R1bGU7CgotIFthdG1vcy1nb10oaHR0cHM6Ly9naXRodWIuY29tL3VtYXRhcmU1L2F0bW9zLWdvKQoKICBHbyBDbGllbnQgdG8gdXNlIEFUTU9TIFBsYXRmb3JtIEFQSS4KCiMjIFByZXJlcXVpc2l0ZQoKLSBGUUROIG9mIEFUTU9TIFBsYXRmb3JtIEFQSQotIEFjY2Vzcy10b2tlbiB0byB1c2UgQVRNT1MgUGxhdGZvcm0gQVBJCgpCb3RoIHBhcmFtZXRlcnMgYXJlIGhpZGRlbiB5ZXQuIExldCdzIHdhaXQgZm9yIG9mZmljaWFsIHJlbGVhc2UuCgojIyBEZXZlbG9wbWVudAoKIyMjIFNldHVwCgotIEluc3RhbGwgYHN3YWdnZXItY2xpYCBmb3IgYnVuZGxlIGRpdmlkZWQgT3BlbkFQSSBTcGVjaWZpY2F0aW9ucy4KCiAgYGBgc2gKICBucG0gaW5zdGFsbAogIGBgYAoKLSBJbnN0YWxsIGBvYXBpLWNvZGVnZW5gIGZvciBidWlsZCB0aGUgY2xpZW50LgoKICBgYGBzaAogIGdvIGluc3RhbGwgZ2l0aHViLmNvbS9kZWVwbWFwL29hcGktY29kZWdlbi9jbWQvb2FwaS1jb2RlZ2VuQGxhdGVzdAogIGBgYAoKIyMjIEJ1aWxkCgpgYGBzaAptYWtlIGJ1aWxkCmBgYAoKIyMjIFJlbGVhc2UKCmBgYHNoCmdpdCBidW1wCm1ha2UgcmVsZWFzZQpgYGAK readmeEtag: '"dc535c242a98d9868eee28c456b5d3fefbc7f465"' readmeLastModified: Wed, 26 Jan 2022 12:55:55 GMT repositoryId: 446084686 description: Source of libraries to use ATMOS Platform API created: '2022-01-09T12:41:06Z' updated: '2025-08-03T07:03:33Z' language: Shell archived: false stars: 0 watchers: 1 forks: 0 owner: aquanest logo: https://avatars.githubusercontent.com/u/88537514?v=4 repoEtag: '"805d63566a075ac2ffffa747b00b15068018639e4c0af5dfaac5281ed579f7f6"' repoLastModified: Sun, 03 Aug 2025 07:03:33 GMT foundInMaster: true category: Code Generators id: 74cbb358315c83236a78360f50033aa7 oldLocations: - https://github.com/umatare5/atmos-openapi-src - source: openapi3 tags repository: https://github.com/nexys-system/openapi-to-idl v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHRvIGludGVyZmFjZSBkZWZpbml0aW9uIGxhbmd1YWdlIChJREwpCgpbIVtUZXN0XShodHRwczovL2dpdGh1Yi5jb20vbmV4eXMtc3lzdGVtL29wZW5hcGktdG8taWRsL2FjdGlvbnMvd29ya2Zsb3dzL3Rlc3QueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9uZXh5cy1zeXN0ZW0vb3BlbmFwaS10by1pZGwvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC55bWwpClshW0RlcGxveV0oaHR0cHM6Ly9naXRodWIuY29tL25leHlzLXN5c3RlbS9vcGVuYXBpLXRvLWlkbC9hY3Rpb25zL3dvcmtmbG93cy9kZXBsb3kueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9uZXh5cy1zeXN0ZW0vb3BlbmFwaS10by1pZGwvYWN0aW9ucy93b3JrZmxvd3MvZGVwbG95LnltbCkKCkdlbmVyYXRlIGEgdHlwZXNjcmlwdCBmaWxlLiBkaXJlY3RseSBmcm9tIHRoZSBPcGVuQVBJIGRlZmluaXRpb24gZmlsZS4gVGhlIGdlbmVyYXRlZCBmaWxlIGNhbiBiZSBpbnRlZ3JhdGVkIGluIGEgcHJvamVjdCB0aHVzIG9mZmVyaW5nIGFuIGVhc3kgYW5kIHR5cGVzYWZlIHdheSB0byBpbnRlZ3JhdGUgd2l0aCBhbnkgQVBJLgoK readmeEtag: '"28f64c3a265f395ca2d3ab60212c10442e231324"' readmeLastModified: Fri, 04 Mar 2022 10:52:29 GMT repositoryId: 465714656 description: turn OpenAPI specs into an IDL created: '2022-03-03T12:43:50Z' updated: '2022-03-03T21:24:18Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: nexys-system logo: https://avatars.githubusercontent.com/u/71571169?v=4 license: MIT repoEtag: '"ff2c27cf3e0147486fa83e9d1253526daac2d37400355328888f567f439d47e5"' repoLastModified: Thu, 03 Mar 2022 21:24:18 GMT foundInMaster: true category: - Server - Testing id: 43a0f97b9da42fde1cd7e39dd518a21f - source: openapi3 tags repository: https://github.com/da0hn/bossabox-code-challenge v3: true id: 9d18581223a55a2c643b3c7f5d717e2d repositoryMetadata: base64Readme: >- IyBCb3NzYUJveCBjb2RlIGNoYWxsZW5nZQoKKiBJbXBsZW1lbnRhdGlvbiBvZiBWVVRUUiAoVmVyeSBVc2VmdWwgVG9vbHMgdG8gUmVtZW1iZXIpIHVzaW5nIHN0YWNrCiAgYEphdmEgMTdgLCBgU3ByaW5nIEJvb3QgMi43LjVgIGFuZCBgTmVvNGogR3JhcGggRGF0YWJhc2VgLgoKKiBBUEkgRG9jczogYHtiYXNlVXJsfS9hcGkvdnV0dHItc2VydmljZS92My9hcGktZG9jc2AKKiBTd2FnZ2VyIFVJOiBge2Jhc2VVcmx9L2FwaS92dXR0ci1zZXJ2aWNlL3N3YWdnZXItdWkvaW5kZXguaHRtbGAKKiBUbyBzdGFydCBkYXRhYmFzZSB1c2UgY29tbWFuZCBgZG9ja2VyLWNvbXBvc2UgdXAgLS1idWlsZGAgKF9gLS1idWlsZGAgaXMgcmVxdWlyZWRfKQoKCiogVGhlIHJlcXVpcmVtZW50cyBhcmUgZGVzY3JpYmUgaW4gW25vdGlvbl0oaHR0cHM6Ly93d3cubm90aW9uLnNvL0JhY2stZW5kLTBiMmM0NWYxYTAwZTRhODQ5ZWVmZTNiMWQ1N2YyM2M2KS4KCjxkZXRhaWxzPgogIDxzdW1tYXJ5PlJlZmVyZW5jZXM8L3N1bW1hcnk+CiAgICAgPGxpPjxhIGhyZWY9Imh0dHBzOi8vZG9jcy5zcHJpbmcuaW8vc3ByaW5nLWRhdGEvbmVvNGovZG9jcy82LjMuNS9yZWZlcmVuY2UvaHRtbC8iPk5lbzRqIFNwcmluZyBEYXRhIHY2LjMuNS48L2E+PC9saT4KPC9kZXRhaWxzPgo= readmeEtag: '"4865ba45a121a691c156137ffa2c656ebf8bf603"' readmeLastModified: Mon, 29 May 2023 02:13:05 GMT repositoryId: 546942908 description: BossaBox code challenge implementation created: '2022-10-06T22:45:10Z' updated: '2023-05-09T17:23:02Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: da0hn logo: https://avatars.githubusercontent.com/u/40874826?v=4 repoEtag: '"fabdaa9fd5854023a9f0ff1ae6c94af0aeb34239f52c2a4e89deb498795d0d78"' repoLastModified: Tue, 09 May 2023 17:23:02 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/white1027/micro-service-workspace v3: true id: 7f52e71b35c7508746e16559cb6f53e4 repositoryMetadata: base64Readme: >- IyBNaWNyb1NlcnZpY2VXb3Jrc3BhY2UKCiMjIOWwiOahiOebrueahOiIh+iqquaYjgotIOS7peW+ruacjeWLmeeCuuS4u+eahOWwiOahiOaetuaniwotIOeUseaWvOS5i+WJjeeahOaetuaniyDmnIPlsIfmr4/lgIvlvq7mnI3li5npg73plovkuIDloIblsIjmoYjomZXnkIYg6YCg5oiQ566h55CG5LiK55qE5Zuw6ZujIOaJgOS7peW4jOacm+iDveWwh+WkmuWAi+W+ruacjeWLmeaUvuWcqOWQjOS4gOWAi+WwiOahiCDkvb/lhbbovIPlrrnmmJPnrqHnkIYKLSDlupXkuIvlpoLmnpzmnInpnIDopoEg5Y+v5Lul5pS555SoIGdpdCBzdWJtb2R1bGUg566h55CGCi0gKipnYXRld2F5Kiog5piv5YWl5Y+jIOmAmuW4uOWPquacg+acieS4gOWAiyAo5aaC5p6c5pyJ5LiN5ZCM55qE5pyN5YuZ5oOz6KaB5o6l57O757WxIOWPr+S7peiAg+aFruWPpuWklumWi+S4gOWAiyBnYXRld2F5KQotICoqbGliKiog5YWx55So5Ye95byP5bqrIOmAmuW4uOaUvuaJgOacieaooee1hOmDveacg+eUqOWIsOeahCAo5aaC5p6c5piv5L2/55SoIFNwcmluZyBKUEEg5LmL6aGe55qEIOmcgOimgeWPpuWkluWGjemWi+S4gOWAiyDlj6rntaYgbW9kdWxlcyDkvb/nlKgpCi0gKiptb2R1bGVzKiog5ZCE5YCL5b6u5pyN5YuZIOWPr+S7peS6kuebuOmAmuioigotICoqZGF0YWJhc2UqKiDpoJDoqIjmr4/lgIvlvq7mnI3li5npg73mnIPmnInoh6rlt7HnmoRkYXRhYmFzZQotIOacgOW+jOWwseaYryDpgJnpgorpg73mlL7miJHoh6rlt7Hoprrlvpflpb3nlKjnmoQg5omA5Lul5bCx5pW05ZCI5oiQ5LiA5YCL5bCI5qGIIOS4puWwh+S4gOS6m+W4uOeUqOeahOaSsOWvq+aWueazleaUvuWFpQoKIyMg5L2/55So55qEIExpYnJhcnkKCiMjIyBHYXRld2F5Ci0gKipTcHJpbmcgQWN0dWF0b3IqKiDnm6PmjqfnlKgKLSAqKlNwcmluZyBXZWJmbHV4KiogUmVhY3RpdmUgd2ViICjlmJfoqabkvb/nlKgpCi0gKipKZXR0eSoqIOaQremFjSBXZWJmbHV4Ci0gKipEdWJibyoqIOW+ruacjeWLmemAmuioiueUqCAgIAotICoqWm9va2VlcGVyKiog5b6u5pyN5YuZ6Ki75YaK55SoCi0gKipPcGVuQXBpICsgUmVkb2MqKgotICoqU2EtVG9rZW4qKiDnmbvlhaXpqZforYnnlKgKICAgCiMjIyBDb21tb25MaWJyYXJ5Ci0gKipMb21ib2sqKiDlpb3nlKjnmoQgZ2V0dGVyIHNldHRlciDlt6XlhbcKCiMjIyBNb2R1bGVzCi0gKipTcHJpbmcgQWN0dWF0b3IqKiDnm6PmjqfnlKgKLSAqKkR1YmJvKiog5b6u5pyN5YuZ6YCa6KiK55SoCi0gKipab29rZWVwZXIqKiDlvq7mnI3li5noqLvlhornlKgKLSAqKk15YmF0aXMqKiBEQueahE9STeW3peWFtwotICoqUG9zdGdyZXMqKiDnm67liY3lvJXnlKjnmoRKREJDCi0gKipGbHl3YXkqKiBEQueJiOaOp+W3peWFtwoKIyMg6ZaL55m85pa55rOVCjEuIOWVn+WLlSBab29rZWVwZXIKMi4g5ZWf5YuVIOWQhOW+ruacjeWLmQozLiDllZ/li5UgR2F0ZXdheSBQb3J0IOmgkOioreeCuiA4MDk5CjQuIOiHsyBsb2NhbGhvc3Q6ODA5OS9pbmRleC5odG1sIOWPr+S7peeci+WIsCBBcGkg5YiX6KGoCgojIyDkvb/nlKjms6jmhI/kuovpoIUKLSDlsIjmoYjlpKrlpJrlsI7oh7QgZ3JhZGxlIG9vbQotIE1vZGVsIOeahOmhnuWIpeWPr+S7peS9v+eUqCBSZWNvcmQg5oiWIEBEYXRhIOaIliBAQnVpbGRlciwgbXliYXRpcyBkdWJibyDpg73lj6/ku6Xop6PmnpAKLSBTYS1Ub2tlbiDlj6/ku6XmkK3phY0gZHViYm8g5L2/55SoCgojIyDlrpjmlrnmlofku7YKLSBNeWJhdGlzIFvli5XmhYtTUUxdKGh0dHBzOi8vbXliYXRpcy5vcmcvbXliYXRpcy0zL3poL2R5bmFtaWMtc3FsLmh0bWwpCi0gRHViYm8gW+WumOe2sl0oaHR0cHM6Ly9jbi5kdWJiby5hcGFjaGUub3JnL3poLWNuL292ZXJ2aWV3L2hvbWUvKQotIFJlZG9jIFtHaXRodWJdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWRvY2x5L3JlZG9jKQotIFpvb2tlZXBlciBbRG9ja2VyXShodHRwczovL2h1Yi5kb2NrZXIuY29tL18vem9va2VlcGVyKQotIFNhLVRva2VuIFvlrpjntrJdKGh0dHBzOi8vc2EtdG9rZW4uY2MvKQoKIyMjIOWll+S7tuS9v+eUqOazqOaEj+S6i+mghQotICoqRmx5d2F5Kiog55uu5YmN6Kit5a6a54K6IOWwiOahiOWVn+WLleWwseacg+iHquWLleabtOaWsOebruaomURCCgojIyDnt6jora/mjIfku6Tot5/nmbzooYwKCiMjIyDnt6jora/miJBKQVIKLSBgIGdyYWRsZSBidWlsZGAKCiMjIyMgSmFyIOS9jee9rgotIGdhdGV3YXkvY29yZS1nYXRld2F5L2J1aWxkL2xpYnMvY29yZS1nYXRld2F5OjAuMC4xLVNOQVBTSE9UCi0gbW9kdWxlcy9jb3JlLW1vZHVsZS9idWlsZC9saWJzL2NvcmUtbW9kdWxlOjAuMC4xLVNOQVBTSE9UCi0gbW9kdWxlcy91c2VyLW1vZHVsZS9idWlsZC9saWJzL3VzZXItbW9kdWxlOjAuMC4xLVNOQVBTSE9UCgoKIyMjIGdyYWFsdm0gTmF0aXZlCi0gYGdyYWRsZSA6Z2F0ZXdheTpjb3JlLWdhdGV3YXk6Ym9vdEJ1aWxkSW1hZ2UgOm1vZHVsZXM6Y29yZS1tb2R1bGU6Ym9vdEJ1aWxkSW1hZ2UgOm1vZHVsZXM6dXNlci1tb2R1bGU6Ym9vdEJ1aWxkSW1hZ2VgCm9yCi0gYGdyYWRsZSA6Z2F0ZXdheTpjb3JlLWdhdGV3YXk6Ym9vdEJ1aWxkSW1hZ2VgCi0gYGdyYWRsZSA6bW9kdWxlczpjb3JlLW1vZHVsZTpib290QnVpbGRJbWFnZWAKLSBgZ3JhZGxlIDptb2R1bGVzOnVzZXItbW9kdWxlOmJvb3RCdWlsZEltYWdlYAojIyMjIGltYWdlIOmgkOioreS9jee9rgotIGRvY2tlci5pby9saWJyYXJ5L2NvcmUtZ2F0ZXdheTowLjAuMS1TTkFQU0hPVAotIGRvY2tlci5pby9saWJyYXJ5L2NvcmUtbW9kdWxlOjAuMC4xLVNOQVBTSE9UCi0gZG9ja2VyLmlvL2xpYnJhcnkvdXNlci1tb2R1bGU6MC4wLjEtU05BUFNIT1QKCgojIyBUT0RPCi0gWyBdIG15YmF0aXMg5YiH5o+bIHNjaGVtYQotIFsgXSBkdWJibyBzZXJ2aWNlIG1lc2gKLSBbIF0gZHViYm8gYXBpIGRvY3MKLSBbIF0g5Zau5YWD5ris6KmmICjnm67liY0gZHViYm8g5pyD5aSx5pWXKQotIFsgXSBEb2NrZXIgQ29tcG9zZSDmjqXplovnmbw= readmeEtag: '"36a52bd464a7ef2a60a0831c677e1e4256614975"' readmeLastModified: Tue, 07 Nov 2023 04:12:56 GMT repositoryId: 710117294 description: 一個Workspace放多個微服務 created: '2023-10-26T03:51:50Z' updated: '2023-11-14T04:12:56Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: white1027 logo: https://avatars.githubusercontent.com/u/4639053?v=4 license: MIT repoEtag: '"d685e5b770c3eb2bf0a821a59c2281d1e27618472d6825304730e043fb548aec"' repoLastModified: Tue, 14 Nov 2023 04:12:56 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/gogvale/timbre_api v3: true id: 88f762db0b11e5b1426d7c6ca95975e7 repositoryMetadata: base64Readme: >- IyBUaW1icmUgQVBJCiFbUmFpbHMgQ0kgQmFkZ2VdKGh0dHBzOi8vZ2l0aHViLmNvbS9nb2d2YWxlL3RpbWJyZV9hcGkvYWN0aW9ucy93b3JrZmxvd3MvcnVieW9ucmFpbHMueW1sL2JhZGdlLnN2ZykKCiMjIyBEZXBlbmRlbmNpZXM6CiogcnVieSAzLjEuMnAyMAoqIFJhaWxzIDcuMC40CiogUG9zdGdyZVNRTCAxNC41CgpUaGluZ3MgeW91IG1heSB3YW50IHRvIGNvdmVyOgoKIyMjICBEYXRhYmFzZSBjcmVhdGlvbiArIG1pZ3JhdGlvbnMKYGBgc2hlbGwKJCBiaW4vcmFpbHMgZGI6bWlncmF0ZTpyZXNldApgYGAKCiMjIyBSdW5uaW5nIHRlc3RzCmBgYHNoZWxsCiQgYmluL2J1bmRsZSByc3BlYyAoc3BlY3xwYXRoX3RvX2ZpbGU6bGluZSkKYGBgCiMjIyBEZXBsb3ltZW50IGluc3RydWN0aW9ucwooVE9ETykK readmeEtag: '"3b21dac9b1033bd0081128b4bf9e8d170fa74d3b"' readmeLastModified: Wed, 19 Oct 2022 01:10:17 GMT repositoryId: 551710663 description: >- Social app API where clients search and contact musicians for events (work in progress) created: '2022-10-14T23:59:18Z' updated: '2022-10-17T23:58:39Z' language: Ruby archived: false stars: 0 watchers: 1 forks: 0 owner: gogvale logo: https://avatars.githubusercontent.com/u/23729423?v=4 license: GPL-3.0 repoEtag: '"2e852a66935e290905e4765df323ac21bd61c4cb4edd785d3a1806a6fcbd7393"' repoLastModified: Mon, 17 Oct 2022 23:58:39 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/oltionzefi/nestjs-mailer-handlebars v3: true id: 9d609acb6d4b267d9806cffc4d70a11b repositoryMetadata: base64Readme: >- IyBOZXN0anMgTWFpbGVyIEhhbmRsZWJhcnMKCiMjIERlc2NyaXB0aW9uCgpQcm9qZWN0IHdoaWNoIGludGVncmF0ZXMgW25lc3Rqcy5jb21dKGh0dHBzOi8vbmVzdGpzLmNvbSksIFtuZXN0LW1vZHVsZXMvbWFpbGVyL10oaHR0cHM6Ly9uZXN0LW1vZHVsZXMuZ2l0aHViLmlvL21haWxlci8pIAphbmQgW2hhbmRsZWJhcnNdKGh0dHBzOi8vaGFuZGxlYmFyc2pzLmNvbSkKCiMjIFJ1bm5pbmcgdGhlIGFwcAoKVGhpcyBwcm9qZWN0IGlzIGJ1aWxkIHdpdGggRG9ja2VyIGFuZCBkb2NrZXIgY29tcG9zZSwgYW5kIHRvIHJ1biB0aGUgYXBwIGV4ZWN1dGUgdGhlIGZvbGxvd2luZyBjb21tYW5kLgoKYGBgYmFzaApkb2NrZXIgY29tcG9zZSB1cAoKIyBvcgoKZG9ja2VyIGNvbXBvc2UgdXAgLWQKYGBgCgpBZnRlciB0aGF0IG9wZW4gdGhlIE9wZW5BUEkgdXJsOiBgYGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMC9hcGlgYAoKIVtvcGVuYXBpXSguL2ltYWdlcy9zY3JlZW5jYXB0dXJlLWxvY2FsaG9zdC1vcGVuYXBpLnBuZykKCndoZXJlIHlvdSBjYW4gc2VlIHRoZSBhcGkgd2hpY2ggdGhpcyBwcm9qZWN0IGNvbnRhaW5zLgoKSWYgeW91IG9wZW4gaW4gcGFyYWxsZWxsIGFsc28gdGhlIHVybCBmb3IgY2hlY2tpbmcgZW1haWxzOiBgYGh0dHA6Ly9sb2NhbGhvc3Q6MTA4MGBgLCB5b3Ugd2lsbCBzZWUgZW1wdHkgbGlzdDoKCiFbZW1wdHktZW1haWxdKC4vaW1hZ2VzL3NjcmVlbmNhcHR1cmUtbG9jYWxob3N0LTEwODAucG5nKQoKVG8gZG8gYSB0ZXN0LCBnbyBhZ2FpbiB0byBPcGVuQVBJIGFuZCB0cnkgdGhlIGVuZHBvaW50LCBpbW1lZGlhdGVseSB5b3Ugd2lsbCBzZWUgZW1haWwgaW4gdGhlIG1haWxkZXYsIGZvcm1hdHRlZCBhcyBiZWxvdzoKCiFbZW1haWxdKC4vaW1hZ2VzL3NjcmVlbmNhcHR1cmUtbG9jYWxob3N0LWVtYWlsLnBuZykKCiMjIFRlc3QKCmBgYGJhc2gKIyB1bml0IHRlc3RzCm5wbSBydW4gdGVzdApgYGAKCiMjIExpY2Vuc2UKCk5lc3QgaXMgW0JTRCAzLUNsYXVzZSBMaWNlbnNlXShMSUNFTlNFKS4K readmeEtag: '"f7159ed74a6640d17be2a6d29b0de37751983f99"' readmeLastModified: Fri, 16 Dec 2022 22:49:31 GMT repositoryId: 579196028 description: Demo project with nestjs, nest-modules/mailer and handlebars created: '2022-12-16T22:48:47Z' updated: '2022-12-17T17:24:10Z' language: Handlebars archived: false stars: 0 watchers: 1 forks: 0 owner: oltionzefi logo: https://avatars.githubusercontent.com/u/13889145?v=4 license: BSD-3-Clause repoEtag: '"1987c580eb2ad2e556e8a00ae4889beb4ba2735aa1f87add4d9aa43fc5886827"' repoLastModified: Sat, 17 Dec 2022 17:24:10 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/alexscigalszky/openapi.errorcodes.library v3: true id: f3519342b60a1396502e2c74f3798fb6 repositoryMetadata: base64Readme: >- IyBPcGVuQXBpLkVycm9yQ29kZXMuTGlicmFyeQoKVGhpcyBpcyBhIGxpYnJhcnkgdG8gc2hvdyBleGFtcGxlcyBvZiBlcnJvciBjb2RlcyBmcm9tIGEgY29uc3RhbnRzIGZpbGUgdG8gYSBkZWZpbmVkIHN0cnVjdHVyZQoKCiMjIEhvdyB0byB1c2UKKiBDcmVhdGUgYW4gYXR0cmlidXRlIGV4dGVuZGluZyDCtFJlc3BvbnNlQ29kZUZyb21Db25zdGFudHNEYXRhQXR0cmlidXRlwrQgdG8gc2F2ZSB5b3VyIGRhdGEuCkV4YW1wbGUKYGBgY3NoYXJwCltBdHRyaWJ1dGVVc2FnZShBdHRyaWJ1dGVUYXJnZXRzLkZpZWxkKV0KcHVibGljIGNsYXNzIENvZGVTdWJDb2RlRGVzY3JpcHRpb25MaW5rQXR0cmlidXRlIDogUmVzcG9uc2VDb2RlRnJvbUNvbnN0YW50c0RhdGFBdHRyaWJ1dGUKewogICAgcHVibGljIGludCBDb2RlOwogICAgcHVibGljIGludD8gU3ViQ29kZTsKICAgIHB1YmxpYyBzdHJpbmcgRGVzY3JpcHRpb247CiAgICBwdWJsaWMgc3RyaW5nIExpbms7CiAgICBwdWJsaWMgQ29kZVN1YkNvZGVEZXNjcmlwdGlvbkxpbmtBdHRyaWJ1dGUoaW50IGNvZGUsIHN0cmluZyBkZXNjcmlwdGlvbiwgc3RyaW5nIGxpbmspCiAgICB7CiAgICAgICAgQ29kZSA9IGNvZGU7CiAgICAgICAgU3ViQ29kZSA9IG51bGw7CiAgICAgICAgRGVzY3JpcHRpb24gPSBkZXNjcmlwdGlvbjsKICAgICAgICBMaW5rID0gbGluazsKICAgIH0KfQpgYGAKCiogQ3JlYXRlIGEgUmVzcG9uc2UgY2xhc3MKRXhhbXBsZQoKYGBgY3NoYXJwCnB1YmxpYyBjbGFzcyBDb2RlU3ViQ29kZVJlc3BvbnNlCnsKICAgIHB1YmxpYyBpbnQgQ29kZSB7IGdldDsgc2V0OyB9CiAgICBwdWJsaWMgaW50IFN1YkNvZGUgeyBnZXQ7IHNldDsgfQogICAgcHVibGljIHN0cmluZz8gRGVzY3JpcHRpb24geyBnZXQ7IHNldDsgfQogICAgcHVibGljIHN0cmluZz8gTGluayB7IGdldDsgc2V0OyB9Cn0KYGBgCgoqIENyZWF0ZSBhIG1hcHBpbmcgZnVuY3Rpb24KRXhhbXBsZQpgYGBjc2hhcnAKdmFyIE1hcEF0dHJpYnV0ZVRvUmVzcG9uc2UgPSAoQ29kZVN1YkNvZGVEZXNjcmlwdGlvbkxpbmtBdHRyaWJ1dGUgZGF0YSwgaW50IGNvbnN0YW50KSA9Pgp7CiAgICByZXR1cm4gbmV3IENvZGVTdWJDb2RlUmVzcG9uc2UoKQogICAgewogICAgICAgIENvZGUgPSBkYXRhLkNvZGUsCiAgICAgICAgU3ViQ29kZSA9IGNvbnN0YW50LAogICAgICAgIERlc2NyaXB0aW9uID0gZGF0YS5EZXNjcmlwdGlvbiwKICAgICAgICBMaW5rID0gZGF0YS5MaW5rCiAgICB9Owp9OwpgYGAKCiogQWRkIHRoZSBPcGVuQXBpIGZpbHRlciBpbiB5b3VyIHN0YXJ0dXAgcHJvamVjdApgYGBjc2hhcnAKYnVpbGRlci5TZXJ2aWNlcy5BZGRFbmRwb2ludHNBcGlFeHBsb3JlcigpOwpidWlsZGVyLlNlcnZpY2VzLkFkZFN3YWdnZXJHZW4oYyA9Pgp7CiAgICBjLlN3YWdnZXJEb2MoInYxIiwgbmV3KCkKICAgIHsKICAgICAgICBUaXRsZSA9IGJ1aWxkZXIuRW52aXJvbm1lbnQuQXBwbGljYXRpb25OYW1lLAogICAgICAgIFZlcnNpb24gPSAidjEiLAogICAgfSk7CiAgICAvLy8gYWRkIG5leHQgbGluZQogICAgYy5PcGVyYXRpb25GaWx0ZXI8UmVzcG9uc2VDb2RlRnJvbUNvbnN0YW50c09wZXJhdGlvbkZpbHRlcjxDb2RlU3ViQ29kZVJlc3BvbnNlLCBDb2RlU3ViQ29kZURlc2NyaXB0aW9uTGlua0F0dHJpYnV0ZT4+KE1hcEF0dHJpYnV0ZVRvUmVzcG9uc2UpOwp9KTsKYGBgCgoqIEFkZCBhdHRyaWJ1dG8gaW4geW91ciBlbmRwb2ludCBmdW5jdGlvbi4gCkl0IG5lZWRzIGEgc2VjdGlvbiBuYW1lIGFuZCB0aGUgY2xhc3Mgd2hlcmUgdGhlIGNvbnN0YW50IGNvZGVzIGFyZSBkZWZpbmVkCkV4YW1wbGUKYGBgY3NoYXJwCnZhciBIb21lRm4gPSAKICAgIC8vIGFkZCBuZXh0IGxpbmUKICAgIFtSZXNwb25zZUNvZGVGcm9tQ29uc3RhbnRzKCJMaXN0IG9mIGVycm9yIGNvZGVzIiwgdHlwZW9mKFJlc3BvbnNlU3ViQ29kZXMpKV0KKFtGcm9tQm9keV0gSW5wdXRSZXF1ZXN0IGlucHV0KSA9PiB7CgogICAgcmV0dXJuICJIZWxsbyB3b3JkIjsKfTsKYGBgCgojIyBFeGFtcGxlIG9mIGEgU3dhZ2dlciBVSQoKPGltZyBzcmM9InN3YWdnZXJ1aS5wbmciLz4KCiMjIEV4YW1wbGUgb2YgYSBPcGVuQXBpIGpzb24gZmlsZQoKYGBganNvbgp7CiAgIm9wZW5hcGkiOiAiMy4wLjEiLAogICJpbmZvIjogewogICAgInRpdGxlIjogIk9wZW5BcGkuRXJyb3JDb2Rlcy5MaWJyYXJ5LkNvbnNvbGUiLAogICAgInZlcnNpb24iOiAidjEiCiAgfSwKICAicGF0aHMiOiB7CiAgICAiLyI6IHsKICAgICAgInBvc3QiOiB7CiAgICAgICAgInRhZ3MiOiBbCiAgICAgICAgICAiT3BlbkFwaS5FcnJvckNvZGVzLkxpYnJhcnkuQ29uc29sZSIKICAgICAgICBdLAogICAgICAgICJyZXF1ZXN0Qm9keSI6IHsKICAgICAgICAgICJjb250ZW50IjogewogICAgICAgICAgICAiYXBwbGljYXRpb24vanNvbiI6IHsKICAgICAgICAgICAgICAic2NoZW1hIjogewogICAgICAgICAgICAgICAgIiRyZWYiOiAiIy9jb21wb25lbnRzL3NjaGVtYXMvSW5wdXRSZXF1ZXN0IgogICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgICAgfSwKICAgICAgICAgICJyZXF1aXJlZCI6IHRydWUKICAgICAgICB9LAogICAgICAgICJyZXNwb25zZXMiOiB7CiAgICAgICAgICAiMjAwIjogewogICAgICAgICAgICAiZGVzY3JpcHRpb24iOiAiU3VjY2VzcyIsCiAgICAgICAgICAgICJjb250ZW50IjogewogICAgICAgICAgICAgICJ0ZXh0L3BsYWluIjogewogICAgICAgICAgICAgICAgInNjaGVtYSI6IHsKICAgICAgICAgICAgICAgICAgInR5cGUiOiAic3RyaW5nIgogICAgICAgICAgICAgICAgfQogICAgICAgICAgICAgIH0KICAgICAgICAgICAgfQogICAgICAgICAgfSwKICAgICAgICAgICJMaXN0IG9mIGVycm9yIGNvZGVzIjogewogICAgICAgICAgICAiZGVzY3JpcHRpb24iOiBudWxsLAogICAgICAgICAgICAiY29udGVudCI6IHsKICAgICAgICAgICAgICAiYXBwbGljYXRpb24vanNvbiI6IHsKICAgICAgICAgICAgICAgICJzY2hlbWEiOiB7CiAgICAgICAgICAgICAgICAgICIkcmVmIjogIiMvY29tcG9uZW50cy9zY2hlbWFzL0NvZGVTdWJDb2RlUmVzcG9uc2UiCiAgICAgICAgICAgICAgICB9LAogICAgICAgICAgICAgICAgImV4YW1wbGVzIjogewogICAgICAgICAgICAgICAgICAiMTAzMSI6IHsKICAgICAgICAgICAgICAgICAgICAidmFsdWUiOiAie1wiQ29kZVwiOjIwMCxcIlN1YkNvZGVcIjoxMDMxLFwiRGVzY3JpcHRpb25cIjpcIkl0ZW0gY3JlYXRlZCFcIixcIkxpbmtcIjpcImh0dHA6Ly90YmRcIn0iCiAgICAgICAgICAgICAgICAgIH0sCiAgICAgICAgICAgICAgICAgICIyNTMxIjogewogICAgICAgICAgICAgICAgICAgICJ2YWx1ZSI6ICJ7XCJDb2RlXCI6MjAwLFwiU3ViQ29kZVwiOjI1MzEsXCJEZXNjcmlwdGlvblwiOlwiSXRlbSB1cGRhdGVkIVwiLFwiTGlua1wiOlwiaHR0cDovL3RiZFwifSIKICAgICAgICAgICAgICAgICAgfSwKICAgICAgICAgICAgICAgICAgIjkwNTQiOiB7CiAgICAgICAgICAgICAgICAgICAgInZhbHVlIjogIntcIkNvZGVcIjozMDAsXCJTdWJDb2RlXCI6OTA1NCxcIkRlc2NyaXB0aW9uXCI6XCJBIGZpZWxkIGlzIG1pc3NpbmdcIixcIkxpbmtcIjpcImh0dHA6Ly90YmRcIn0iCiAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICB9CiAgICAgICAgICAgIH0KICAgICAgICAgIH0KICAgICAgICB9CiAgICAgIH0KICAgIH0KICB9LAogICJjb21wb25lbnRzIjogewogICAgInNjaGVtYXMiOiB7CiAgICAgICJDb2RlU3ViQ29kZVJlc3BvbnNlIjogewogICAgICAgICJ0eXBlIjogIm9iamVjdCIsCiAgICAgICAgInByb3BlcnRpZXMiOiB7CiAgICAgICAgICAiY29kZSI6IHsKICAgICAgICAgICAgInR5cGUiOiAiaW50ZWdlciIsCiAgICAgICAgICAgICJmb3JtYXQiOiAiaW50MzIiCiAgICAgICAgICB9LAogICAgICAgICAgInN1YkNvZGUiOiB7CiAgICAgICAgICAgICJ0eXBlIjogImludGVnZXIiLAogICAgICAgICAgICAiZm9ybWF0IjogImludDMyIgogICAgICAgICAgfSwKICAgICAgICAgICJkZXNjcmlwdGlvbiI6IHsKICAgICAgICAgICAgInR5cGUiOiAic3RyaW5nIiwKICAgICAgICAgICAgIm51bGxhYmxlIjogdHJ1ZQogICAgICAgICAgfSwKICAgICAgICAgICJsaW5rIjogewogICAgICAgICAgICAidHlwZSI6ICJzdHJpbmciLAogICAgICAgICAgICAibnVsbGFibGUiOiB0cnVlCiAgICAgICAgICB9CiAgICAgICAgfSwKICAgICAgICAiYWRkaXRpb25hbFByb3BlcnRpZXMiOiBmYWxzZQogICAgICB9LAogICAgICAiSW5wdXRSZXF1ZXN0IjogewogICAgICAgICJ0eXBlIjogIm9iamVjdCIsCiAgICAgICAgInByb3BlcnRpZXMiOiB7CiAgICAgICAgICAibmFtZSI6IHsKICAgICAgICAgICAgInR5cGUiOiAic3RyaW5nIiwKICAgICAgICAgICAgIm51bGxhYmxlIjogdHJ1ZQogICAgICAgICAgfSwKICAgICAgICAgICJjb3VudCI6IHsKICAgICAgICAgICAgInR5cGUiOiAiaW50ZWdlciIsCiAgICAgICAgICAgICJmb3JtYXQiOiAiaW50MzIiLAogICAgICAgICAgICAibnVsbGFibGUiOiB0cnVlCiAgICAgICAgICB9CiAgICAgICAgfSwKICAgICAgICAiYWRkaXRpb25hbFByb3BlcnRpZXMiOiBmYWxzZQogICAgICB9CiAgICB9CiAgfQp9CmBgYAo= readmeEtag: '"900e0974216e9f31767a79adf8a4114a03197cf3"' readmeLastModified: Thu, 24 Aug 2023 16:05:56 GMT repositoryId: 682270973 description: >- This is a library to show examples of error codes from a constants file to a defined structure created: '2023-08-23T20:11:26Z' updated: '2023-08-24T16:08:26Z' language: C# archived: false stars: 0 watchers: 1 forks: 0 owner: AlexScigalszky logo: https://avatars.githubusercontent.com/u/20727215?v=4 license: Apache-2.0 repoEtag: '"25314a60a5db1458b0a5fe2fa879402a6a8db11f46998e453606543466715d5b"' repoLastModified: Thu, 24 Aug 2023 16:08:26 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/dashotv/tvdb v3: true id: ecf31ce50b97ac527c6c494b25182994 repositoryMetadata: base64Readme: >- IyBkYXNob3R2L3R2ZGIKCkdvbGFuZyBUVkRCIENsaWVudCAoQWxwaGEpCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vZHJvbmUuZGFzaG8ubmV0L2FwaS9iYWRnZXMvZGFzaG90di90dmRiL3N0YXR1cy5zdmc/cmVmPXJlZnMvaGVhZHMvbWFpbildKGh0dHBzOi8vZHJvbmUuZGFzaG8ubmV0L2Rhc2hvdHYvdHZkYikKWyFbR29Eb2NdKGh0dHBzOi8vZ29kb2Mub3JnL2dpdGh1Yi5jb20vZGFzaG90di90dmRiP3N0YXR1cy5zdmcpXShodHRwczovL2dvZG9jLm9yZy9naXRodWIuY29tL2Rhc2hvdHYvdHZkYikKWyFbR28gUmVwb3J0IENhcmRdKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9iYWRnZS9naXRodWIuY29tL2Rhc2hvdHYvdHZkYildKGh0dHBzOi8vZ29yZXBvcnRjYXJkLmNvbS9yZXBvcnQvZ2l0aHViLmNvbS9kYXNob3R2L3R2ZGIpCiFbTGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLU1JVC1ibHVlLnN2ZykKCiMjIEdlbmVyYXRlZCBDb2RlCgpUaGlzIHBhY2thZ2UgaXMgZ2VuZXJhdGVkIGZyb20gdGhlIFtUVkRCIE9wZW5BUEldKGh0dHBzOi8vYXBpLnRoZXR2ZGIuY29tL3N3YWdnZXIpIHNwZWNpZmljYXRpb24gdXNpbmcgdGhlIFtTcGVha2Vhc3ldKGh0dHBzOi8vc3BlYWtlYXN5YXBpLmRldikgY29kZSBnZW5lcmF0b3IuCgpUaGUgZ2VuZXJhdGVkIGNvZGUgaXMgaW4gdGhlIGBvcGVuYXBpYCBkaXJlY3RvcnksIGFuZCBJJ3ZlIHdyaXR0ZW4gc2NyaXB0cyB0byB3cmFwCml0IHdpdGggYSBtb3JlIGNvbnZlbmllbnQgaW50ZXJmYWNlLgoKU2VlIHRoZSBgR2VuZXJhdGlvbmAgc2VjdGlvbiBiZWxvdyBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KCj4gWyFOT1RFXQo+IEkgaGF2ZSBuZWVkZWQgdG8gbWFrZSBhIGZldyB0d2Vha3MgdG8gdGhlIG9wZW5hcGkgc3BlYyB0byBnZXQgdGhpbmdzIHRvIHdvcmsgY29ycmVjdGx5IGluIGdvLgoKIyMgQWRkaXRpb25hbCBEb2N1bWVudGF0aW9uCgpUaGUgU3BlYWtlYXN5IGdlbmVyYXRvciBhbHNvIGdlbmVyYXRlcyBkb2N1bWVudGF0aW9uIGZvciB0aGUgQVBJLiBUaGlzIGlzIGF2YWlsYWJsZQppbiB0aGUgb3BlbmFwaSBkaXJlY3RvcnkncyBbUkVBRE1FLm1kXShvcGVuYXBpL1JFQURNRS5tZCkgYW5kIHRoZSBbZG9jc10ob3BlbmFwaS9kb2NzKSBkaXJlY3RvcnkuCgojIyBTdGF0dXMKClRoZSBjb2RlIGlzIGN1cnJlbnRseSBpbiBhbHBoYSwgYW5kIGlzIG5vdCByZWFkeSBmb3IgcHJvZHVjdGlvbiB1c2UuIEl0IGlzIGJlaW5nIHVzZWQKYnkgdGhlIERhc2hvVFYgcHJvamVjdCwgYnV0IGlzIG5vdCB5ZXQgc3RhYmxlLgoKIyMgVXNhZ2UKCkluc3RhbGwgdGhlIHBhY2thZ2Ugd2l0aDoKCmBgYGJhc2gKZ28gZ2V0IGdpdGh1Yi5jb20vZGFzaG90di90dmRiCmBgYAoKSW1wb3J0IHRoZSBwYWNrYWdlIHdpdGg6CgpgYGBnbwppbXBvcnQgImdpdGh1Yi5jb20vZGFzaG90di90dmRiIgpgYGAKCkNyZWF0ZSBhIG5ldyBjbGllbnQgd2l0aDoKCmBgYGdvCmNsaWVudCA6PSB0dmRiLk5ldyhhcGlrZXksIHRva2VuKQpgYGAKCklmIHlvdSBkb24ndCBhbHJlYWR5IGhhdmUgYSB0b2tlbiwgeW91IGNhbiBhdXRoZW50aWNhdGUgd2l0aCB5b3VyIGBhcGlrZXlgIHVzaW5nIHRoZSBgTG9naW5gIG1ldGhvZDoKCmBgYGdvCi8vIEF1dGhlbnRpY2F0ZSB3aXRoIHlvdXIgQVBJIGtleS4gVGhpcyB3aWxsIHJldHVybiBhIGNsaWVudCB3aXRoIHRoZQovLyB0b2tlbiBhbHJlYWR5IGNvbmZpZ3VyZWQuCmNsaWVudCwgZXJyIDo9IHR2ZGIuTG9naW4oYXBpa2V5KQppZiBlcnIgIT0gbmlsIHsKICAgIC8vIGhhbmRsZSBlcnJvcgp9CmNsaWVudC5Ub2tlbiAvLyB0aGUgdG9rZW4KYGBgCgo+IFshSU1QT1JUQU5UXQo+IFlvdSBzaG91bGQgc3RvcmUgdGhlIHRva2VuIHNvbWV3aGVyZSwgYnkgZGVmYXVsdCB0aGUgdG9rZW4gaXMgdmlhYmxlIGZvciAzMCBkYXlzLiBUVkRCIGRvZXNuJ3QgYXBwZWFyIHRvIGNhcmUgaWYgeW91IGF1dGggZXZlcnkgY2FsbCwgYnV0IGl0IGFkZHMgYSBsb3Qgb2Ygb3ZlcmhlYWQuCgpJZiB5b3UgYWxyZWFkeSBoYXZlIHRoZSB0b2tlbiwgeW91IGNhbiBjcmVhdGUgYSBjbGllbnQgd2l0aCB5b3VyIGBhcGlrZXlgIGFuZCBgdG9rZW5gOgoKYGBgZ28KY2xpZW50IDo9IHR2ZGIuTmV3KGFwaWtleSwgdG9rZW4pCmBgYAoKIyMgRGV2ZWxvcG1lbnQKCkNyZWF0ZSBhIGxvY2FsIGAuZW52YCBmaWxlIHdpdGggdGhlIGZvbGxvd2luZyBjb250ZW50OgoKYGBgCiMgLmVudgpUVkRCX0FQSV9VUkw9aHR0cHM6Ly9hcGk0LnRoZXR2ZGIuY29tL3Y0ClRWREJfQVBJX0tFWT15b3VyX2FwaV9rZXkKVFZEQl9BUElfVE9LRU49eW91cl9hcGlfdG9rZW4KYGBgCgpSdW4gdGhlIGZvbGxvd2luZyB0byBnZXQgdGhlIG1ha2UgdGFyZ2V0czoKCmBgYAo+IG1ha2UgaGVscAoKVXNhZ2U6CiAgbWFrZSA8dGFyZ2V0PgoKVGFyZ2V0czoKICBHZW5lcmFsOgogICAgZ2VuZXJhdGUgICAgICAgICAgICBHZW5lcmF0ZSBjb2RlIGZyb20gb3BlbmFwaS55bWwgc3BlYwogICAgY2xlYW4gICAgICAgICAgICAgICBSZW1vdmUgYnVpbGQgcmVsYXRlZCBmaWxlCiAgVGVzdDoKICAgIHRlc3QgICAgICAgICAgICAgICAgUnVuIHRoZSB0ZXN0cyBvZiB0aGUgcHJvamVjdAogICAgY292ZXJhZ2UgICAgICAgICAgICBSdW4gdGhlIHRlc3RzIG9mIHRoZSBwcm9qZWN0IGFuZCBleHBvcnQgdGhlIGNvdmVyYWdlCiAgTGludDoKICAgIGxpbnQgICAgICAgICAgICAgICAgUnVuIGFsbCBhdmFpbGFibGUgbGludGVycwogICAgbGludC1nbyAgICAgICAgICAgICBVc2UgZ29saW50Y2ktbGludCBvbiB5b3VyIHByb2plY3QKICBIZWxwOgogICAgaGVscCAgICAgICAgICAgICAgICBTaG93IHRoaXMgaGVscC4KCmBgYAoKIyMjIEdlbmVyYXRpb24KClRvIHVwZGF0ZSB0aGUgZ2VuZXJhdGVkIGNvZGUsIGVuc3VyZSB5b3UgaGF2ZSB0aGUgZGVwZW5kZW5jaWVzIGluc3RhbGxlZDoKCi0gU3BlYWtlYXN5IENMSSAtIFNlZSBbU3BlYWtlYXN5XShodHRwczovL3NwZWFrZWFzeWFwaS5kZXYpIGZvciBtb3JlIGluZm9ybWF0aW9uLgotIFJ1YnkgLSBTZWUgW1J1YnldKGh0dHBzOi8vd3d3LnJ1YnktbGFuZy5vcmcvZW4vZG9jdW1lbnRhdGlvbi9pbnN0YWxsYXRpb24vKSBmb3IgbW9yZSBpbmZvcm1hdGlvbi4KClRoZW4gcnVuOgoKYGBgYmFzaAptYWtlIGdlbmVyYXRlCmBgYAoKVGhpcyB3aWxsIHJ1biB0aGUgYHNjcmlwdHMvZ2VuZXJhdGUuc2hgIHNjcmlwdCwgd2hpY2ggd2lsbDoKCi0gUnVuIHRoZSBzcGVha2Vhc3kgY2xpIGdlbmVyYXRvcgotIFJlYXJyYW5nZSB0aGUgZ2VuZXJhdGVkIGNvZGUKLSBSdW4gYSBydWJ5IHNjcmlwdCB0byBidWlsZCB0aGUgd3JhcHBlciBjbGllbnQgY29kZQoKIyMjIE5vdGVzCgpUaGVyZSBhcmUgc2V2ZXJhbCBvcGVyYXRpb25zIG9uIHRoZSBhcGkgdGhhdCBhcmUgZGlzYWJsZWQgKHRoaXMgaXMgaGFuZGxlZCBpbiB0aGUKcnVieSBzY3JpcHQpLiBUaGVzZSBhcmUgb3BlcmF0aW9ucyB0aGF0IHJlcXVpcmUgYWRkaXRpb25hbCBwcml2ZWxlZ2VzIChhY3Rpb25zIG9uCmJlaGFsZiBvZiB0aGUgdXNlcikgb3IgdGhhdCBkb24ndCB3b3JrIGNvcnJlY3RseS4gVGhlc2UgYXJlIGNvbmZpZ3VyZWQgYXMgYW4gYXJyYXkKaW4gdGhlIHJ1Ynkgc2NyaXB0Lgo= readmeEtag: '"5d37ac2e5ae85f82ecd3f3f8eb7a5289f66662ad"' readmeLastModified: Mon, 08 Jul 2024 23:12:52 GMT repositoryId: 707020724 description: DashoTV tvdb v4 client (alpha) created: '2023-10-19T04:16:17Z' updated: '2024-07-08T23:12:58Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: dashotv logo: https://avatars.githubusercontent.com/u/20273052?v=4 license: MIT repoEtag: '"d01e8d7111843b58a1c6a3ef3d828739eab8c9727f9ff467483a3dc7c306da81"' repoLastModified: Mon, 08 Jul 2024 23:12:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/useractivegames v3: true id: 207175dc82f84c4eeab9c89927311f5d repositoryMetadata: base64Readme: >- PiBbIUlNUE9SVEFOVF0gIAo+IFRoaXMgcmVwb3NpdG9yeSBpcyByZWFkLW9ubHkgLyBhcmNoaXZlZCBhbmQgd2lsbCBub3QgcmVjaWV2ZSB1cGRhdGVzLgoKIyBbVHlwZWRvYyBEb2N1bWVudGF0aW9uIFdlYnNpdGVdKGh0dHBzOi8vc3Vkb2t1cnUuZ2l0aHViLmlvL1VzZXJBY3RpdmVHYW1lcy8pPGJyPgoKCiMgRGV2ZWxvcGVyIFNldHVwCgoxLiBHZXQgdGhlIC5lbnYgZmlsZSBmcm9tIHRoZSBNU0IgYnVpbGRpbmcgbmV4dCB0byB0aGUgd2F0ZXIgZm91bnRhaW4uIAoyLiBJbnN0YWxsIERvY2tlciBvbiB5b3VyIG1hY2hpbmUuIFR1dG9yaWFsIGlzIGxpbmtlZCBiZWxvdzo8YnI+CiAgIFshW0RvY2tlciBUdXRvcmlhbF0oaHR0cHM6Ly9pbWcueW91dHViZS5jb20vdmkvMmV6TnFxYVNqcTgvMC5qcGcpXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PTJlek5xcWFTanE4KTxicj4KMi4gT25jZSBkb2NrZXIgaXMgaW5zdGFsbGVkLCB0aGUgTW9uZ28gaW1hZ2UgY2FuIGJlIHJ1biB3aXRoIHRoaXMgY29tbWFuZDo8YnI+Ck5vdGUgdXNlIGBgYHN1ZG9gYGAgb24gTGludXgvTWFjPGJyPgpgYGBjb25zb2xlCm5wbSBydW4gZG9ja2VyCmBgYAozLiBUaGUgYXBwIGNhbiB0aGVuIGJlIHJ1biB3aXRoIHRoZSBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHN0YXJ0CmBgYAo0LiBJbnRlZ3JhdGlvbiB0ZXN0cyBjYW4gYmUgcnVuIHdoZW4gdGhlIGFwcCBpcyBydW5uaW5nIHdpdGggdGhpcyBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHRlc3Q6aW50ZWdyYXRpb24KYGBgCg== readmeEtag: '"551645e2a891d8f25afe5c10c232d868e7f8455c"' readmeLastModified: Sun, 12 May 2024 16:01:44 GMT repositoryId: 606496910 description: null created: '2023-02-25T17:08:38Z' updated: '2024-05-12T16:03:12Z' language: TypeScript archived: true stars: 0 watchers: 0 forks: 0 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"8147cb594e9e22906c62d6ffb951adbdc7020b7c60c5c09bb9b8dca21ccdc28d"' repoLastModified: Sun, 12 May 2024 16:03:12 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/majid-l/express-ecommerce-api v3: true id: 5b377bebb2f2eb1e1b487665beba3ecc repositoryMetadata: base64Readme: >- IyBFeHByZXNzLCBUeXBlU2NyaXB0LCBbUHJpc21hIE9STV0oaHR0cHM6Ly93d3cucHJpc21hLmlvLykgYW5kIE9wZW5BUEkgcHJvamVjdAoKVGhpcyBwcm9qZWN0IGZlYXR1cmVzIGEgKipbUkVTVGZ1bCBDUlVEIEFQSV0oaHR0cHM6Ly90YWxpcGh1cy52ZXJjZWwuYXBwL2FwaSkqKiB0aGF0IHByb2Nlc3NlcyBkYXRhIGZvciBhbiBlY29tbWVyY2UgYXBwbGljYXRpb24uCgojIyBLZXkgcHJvZHVjdCBmZWF0dXJlcwotIERlc2lnbmVkIGFuZCBmdWxseSBkb2N1bWVudGVkIHVzaW5nIFN3YWdnZXIgdG9vbHMgYW5kIE9wZW5BUEkgU3BlY2lmaWNhdGlvbi4gCiAgLSBGdWxsICoqW0FQSSBjb250cmFjdF0oLi9vcGVuYXBpLnlhbWwpKiogY2FuIGJlIGZvdW5kIFtoZXJlXSguL29wZW5hcGkueWFtbCkuCi0gUG93ZXJlZCBieSBWZXJjZWwncyBzZXJ2ZXJsZXNzIGZ1bmN0aW9ucywgaW5jbHVkaW5nIGEgUFNRTCBkYXRhYmFzZSBzZXJ2ZXIgZm9yIG1hbmFnaW5nIGFwcGxpY2F0aW9uIGRhdGEgKGNvbmZpZ3VyZWQgZm9yIFtzZXNzaW9uIHBlcnNpc3RlbmNlXSguL2F1dGgvc2Vzc2lvbi50cykpLgotIFtTZXNzaW9uIGFuZCBjb29raWUtYmFzZWQgYXV0aGVudGljYXRpb25dKC4vYXV0aC8pIGVuYWJsaW5nIHBlcnNpc3RlbnQgbG9naW5zLgotIFtNaWRkbGV3YXJlIGZ1bmN0aW9uc10oLi9taWRkbGV3YXJlKSBmb3IgZGF0YSB2YWxpZGF0aW9uIGFuZCB1c2VyIGF1dGhlbnRpY2F0aW9uLgogIC0gSW4tZGVwdGggZXJyb3IgaGFuZGxpbmcsIGNhc3RpbmcgYSB3aWRlIG5ldCBvdmVyIHBvdGVudGlhbCBlZGdlIGNhc2VzIGFuZCBzb3VyY2VzIG9mIGVycm9yLgotIFtEYXRhIG1vZGVsbGluZ10oLi9wcmlzbWEvc2NoZW1hLnByaXNtYSkgYW5kIFtkYXRhYmFzZSBtaWdyYXRpb25zXSguL3ByaXNtYS9taWdyYXRpb25zLzIwMjMwNzI4MTA1NDA4Xy9taWdyYXRpb24uc3FsKSB3aXRoIFByaXNtYSBPUk0uCi0gQ29tcHJlaGVuc2l2ZSBbaW50ZWdyYXRpb24gdGVzdGluZ10oLi9hcGlfdGVzdHMvKSwgYWNoaWV2aW5nICoqOTAlIHRlc3QgY292ZXJhZ2UqKiBhcyByZXBvcnRlZCBieSBJc3RhbmJ1bCdzICpueWMqIENMSS4KLSBQcm9ncmFtbWF0aWMgW2RhdGFiYXNlIHJlc2VlZGluZ10oL3ByaXNtYS9zZWVkLnRzKSB1c2luZyBbZHVtbXkgZGF0YV0oL3ByaXNtYS9kZXZfZGF0YS50cykuCgo8cCBmbG9hdD0ibGVmdCI+CiAgPGltZyBzcmM9Ii4vaW1hZ2VzL25vZGUuc3ZnIiB3aWR0aD0iNjAiIC8+CiAgJm5ic3A7Jm5ic3A7Jm5ic3A7Jm5ic3A7Jm5ic3A7CiAgPGltZyBzcmM9Ii4vaW1hZ2VzL3RzLnN2ZyIgd2lkdGg9IjYwIiAvPgogICZuYnNwOyZuYnNwOyZuYnNwOyZuYnNwOyZuYnNwOwogIDxpbWcgc3JjPSIuL2ltYWdlcy9wcmlzbWEuc3ZnIiB3aWR0aD0iMTQwIiAvPgogICZuYnNwOyZuYnNwOyZuYnNwOyZuYnNwOyZuYnNwOwogIDxpbWcgc3JjPSIuL2ltYWdlcy9wc3FsLnN2ZyIgd2lkdGg9IjYwIiAvPiAKICAmbmJzcDsmbmJzcDsmbmJzcDsmbmJzcDsmbmJzcDsKICA8aW1nIHNyYz0iLi9pbWFnZXMvbW9jaGEuc3ZnIiB3aWR0aD0iNjAiIC8+IAogICZuYnNwOyZuYnNwOyZuYnNwOyZuYnNwOyZuYnNwOwogIDxpbWcgc3JjPSIuL2ltYWdlcy9jaGFpLnN2ZyIgd2lkdGg9IjYwIiAvPgo8L3A+CgotLS0KIyMgRW5kcG9pbnRzCkZvciBtb3JlIGluZm9ybWF0aW9uIG9uIGF2YWlsYWJsZSBxdWVyeSBwYXJhbWV0ZXJzIGFuZCByZXF1ZXN0IGJvZHkgcmVxdWlyZW1lbnRzLCB2aXNpdCB0aGUgKipbQVBJIGJhc2UgdXJsXShodHRwczovL3RhbGlwaHVzLnZlcmNlbC5hcHAvYXBpKSoqLgoKTWFueSBvZiB0aGVzZSBlbmRwb2ludHMgcmVxdWlyZSBhdXRoZW50aWNhdGVkIGFjY2Vzcywgd2hpY2ggeW91IGNhbiBhY2NvbXBsaXNoIGJ5IGZpcnN0IHNpZ25pbmcgdXAgYW5kIHRoZW4gbG9nZ2luZyBpbi4KCmBgYGpzb24KLy8gMSkgU2VuZCBhIFBPU1QgcmVxdWVzdCB0byAvYXBpL3NpZ251cAp7CiAgInVzZXJuYW1lIjogLi4uLAogICJwYXNzd29yZCI6IC4uLiwKICAiZW1haWwiOiAuLi4gLAogICJuYW1lIjogLi4uCn0KCi8vIDIpIFNlbmQgYSBQT1NUIHJlcXVlc3QgdG8gL2FwaS9sb2dpbgp7CiAgInVzZXJuYW1lIjogLi4uIC8qIHlvdXIgdXNlcm5hbWUgZnJvbSBzdGVwIDEpICovLAogICJwYXNzd29yZCI6IC4uLiAvKiB5b3VyIHBhc3N3b3JkIGZyb20gc3RlcCAxKSAqLwp9CmBgYAoKfCBIVFRQIG1ldGhvZChzKSB8IFVSTAp8LS0tfC0tLXwKUE9TVCB8IC9hcGkvc2lnbnVwClBPU1QgfCAvYXBpL2xvZ2luClBPU1QgfCAvYXBpL2xvZ291dApHRVQgfCAvYXBpL3Byb2R1Y3RzCkdFVCB8IC9hcGkvcHJvZHVjdHMvYmVzdHNlbGxlcnMKR0VUIHwgL2FwaS9wcm9kdWN0cy86aWQKR0VUIHwgL2FwaS9wcm9kdWN0cy86aWQvcmV2aXdzCkdFVCwgUFVULCBERUxFVEUgfCAvYXBpL2N1c3RvbWVycy86aWQKR0VULCBQVVQgfCAvYXBpL2N1c3RvbWVycy86aWQvY2FydApHRVQsIFBVVCB8IC9hcGkvY3VzdG9tZXJzLzppZC93aXNobGlzdApHRVQsIFBPU1QgfCAvYXBpL2N1c3RvbWVycy86aWQvb3JkZXJzCkdFVCB8IC9hcGkvY3VzdG9tZXJzLzppZC9mYXZvcml0ZXMKR0VUIHwgL2FwaS9jdXN0b21lcnMvOmlkL29yZGVycy86b3JkZXJJZApHRVQgfCAvYXBpL2N1c3RvbWVycy86aWQvcmV2aWV3cwpQT1NUIHwgL2FwaS9jdXN0b21lcnMvOmlkL2FkZHJlc3NlcwpERUxFVEUgfCAvYXBpL2N1c3RvbWVycy86aWQvYWRkcmVzc2VzLzphZGRyZXNzSWQKR0VUIHwgL2FwaS9jYXRlZ29yaWVzCkdFVCB8IC9hcGkvc3VwcGxpZXJzCkdFVCwgUE9TVCB8IC9hcGkvcmV2aWV3cwpHRVQsIFBVVCwgREVMRVRFIHwgL2FwaS9yZXZpZXdzLzppZAoKLS0tCiMjIERhdGEgbW9kZWwKVGhpcyBpcyBhIHNpbXBsaWZpZWQgdmlldyBvZiB0aGUgZW50aXR5IHJlbGF0aW9uc2hpcHMgdGhhdCBleGlzdCB3aXRoaW4gdGhlIGRhdGFiYXNlLiBGb3IgYSBtb3JlIGNvbXBsZXRlIHBpY3R1cmUsIGNvbnN1bHQgdGhlICoqW3NjaGVtYSBjb25maWd1cmF0aW9uXSgvcHJpc21hL3NjaGVtYS5wcmlzbWEpKiogYW5kICoqW21pZ3JhdGlvbl0oL3ByaXNtYS8vbWlncmF0aW9ucy8yMDIzMDcyODEwNTQwOF8vbWlncmF0aW9uLnNxbCkqKiBmaWxlcy4KCjxpbWcgc3JjPSIuL2ltYWdlcy9lcmQtZGFyay5zdmciIHdpZHRoPSI5MDAiIC8+IAoKLS0tCiMjIE1haW4gcHJvamVjdCBkZXBlbmRlbmNpZXMKfCBQYWNrYWdlIHwgUHVycG9zZQp8LS0tfC0tLXwKRXhwcmVzcyB8IFdlYiBBUEkgZnJhbWV3b3JrClByaXNtYSB8IE5vZGUuanMgJiBUeXBlU2NyaXB0IE9STQpTdXBlclRlc3QgfCBJbnRlZ3JhdGlvbiB0ZXN0aW5nCk1vY2hhIHwgVGVzdCBmcmFtZXdvcmsKQ2hhaSB8IEFzc2VydGlvbiBsaWJyYXJ5ClBhc3Nwb3J0LmpzIHwgQXV0aGVudGljYXRpb24gbWlkZGxld2FyZQpleHByZXNzLXNlc3Npb24gfCBTZXNzaW9uIG1pZGRsZXdhcmUKCi0tLQojIyBSdW5uaW5nIHRoZSBwcm9qZWN0IG9uIGxvY2FsaG9zdAo+IFRoaXMgcHJvamVjdCByZXF1aXJlcyBQU1FMIHRvIGJlIGluc3RhbGxlZCBsb2NhbGx5LgoKIyMjIFNldHVwIGluc3RydWN0aW9ucwoxKSBDbG9uZSBhbmQgZm9yayB0aGUgcmVwb3NpdG9yeSBhbmQgaW5zdGFsbCBhbGwgZGVwZW5kZW5jaWVzLgoyKSBDcmVhdGUgYSBsb2NhbCBlbXB0eSBQU1FMIGRhdGFiYXNlIGNhbGxlZCAqKmVjb21tZXJjZV9kYioqLgozKSBDcmVhdGUgYSAqKi5lbnYqKiBmaWxlIGluIHRoZSByb290IG9mIHRoZSByZXBvc2l0b3J5IHdpdGggNCBlbnZpcm9ubWVudCB2YXJpYWJsZXM6CmBgYApEQVRBQkFTRV9QUklTTUFfVVJMPXBvc3RncmVzcWw6Ly88VVNFUj46PHBhc3N3b3JkPkBsb2NhbGhvc3Q6NTQzMi9lY29tbWVyY2VfZGIKREFUQUJBU0VfVVJMX05PTl9QT09MSU5HPXBvc3RncmVzcWw6Ly88VVNFUj46PHBhc3N3b3JkPkBsb2NhbGhvc3Q6NTQzMi9lY29tbWVyY2VfZGIKU0VTU0lPTl9TRUNSRVQ9c2Vzc2lvbnNlY3JldApQT1JUPTMwMDAKYGBgCjQpIEZvciB0aGUgZGF0YWJhc2UgY29ubmVjdGlvbiBzdHJpbmdzLCBSZXBsYWNlIGBgYDxVU0VSPmBgYCB3aXRoIHRoZSBuYW1lIG9mIHlvdXIgbG9jYWwgZGF0YWJhc2UgdXNlciAoZS5nLiAqcG9zdGdyZXMqKSBhbmQgcmVwbGFjZSBgYGA8UEFTU1dPUkQ+YGBgIHdpdGggd2hhdGV2ZXIgcGFzc3dvcmQgeW91IHVzZWQgdG8gc2V0IHVwIHRoZSBsb2NhbCB1c2VyLgoKWW91IGNhbiBub3cgcnVuIHRoZSBzY3JpcHRzIGJlbG93IGFuZCBiZWdpbiB0byBleHBsb3JlIHRoZSBwcm9qZWN0LgoKIyMjIFJ1biB0aGUgRXhwcmVzcyBzZXJ2ZXIgaW4gZGV2ZWxvcG1lbnQgbW9kZS4KYGBgc2gKbnBtIHJ1biBkZXYKCiMgb3IKCm5wbSBydW4gc3RhcnQKYGBgCgojIyMgQ29tcGlsZSBhbmQgcnVuIE1vY2hhIHRlc3Qgc3VpdGUKYGBgc2gKbnBtIHJ1biB0ZXN0CmBgYAoKIyMjIFRlc3QgY292ZXJhZ2UgcmVwb3J0IApgYGBzaApucG0gcnVuIHRlc3Rjb3YKYGBgCgojIyMgVXBkYXRlIGRhdGFiYXNlIHNjaGVtYSAKVGhpcyBjb21tYW5kIHBlcmZvcm1zIDIgYWN0aW9uczogc3luY3MgdGhlIGRhdGFiYXNlIHNjaGVtYSB3aXRoIFByaXNtYSBzY2hlbWEgYW5kIHJlZ2VuZXJhdGVzIFByaXNtYSBDbGllbnQuCmBgYHNoCm5weCBwcmlzbWEgbWlncmF0ZSBkZXYKCiMgb3IKCnByaXNtYSBkYiBwdXNoCmBgYAoKIyMjIFNlZWQgZGF0YWJhc2UKYGBgc2gKbnB4IHByaXNtYSBkYiBzZWVkCmBgYAoKIyMjIFtQcmlzbWEgU3R1ZGlvXShodHRwczovL3d3dy5wcmlzbWEuaW8vc3R1ZGlvKSAoYnJvd3Nlci1iYXNlZCBHVUkgYW5kIGRhdGFiYXNlIHZpc3VhbGlzZXIpCmBgYHNoCm5weCBwcmlzbWEgc3R1ZGlvCmBgYA== readmeEtag: '"d9dd6f49f666a57f1365231b5059367b01abd035"' readmeLastModified: Sun, 18 Feb 2024 18:14:22 GMT repositoryId: 669093937 description: >- Express REST API featuring Prisma ORM, session-based authentication and integration testing with Mocha, Chai + SuperTest. Designed with Swagger tools and documented with OpenAPI 3.0.3. created: '2023-07-21T10:21:46Z' updated: '2023-08-05T14:13:45Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 1 owner: majid-L logo: https://avatars.githubusercontent.com/u/115797459?v=4 repoEtag: '"0d46cb8c2b5a84590958fd38b71c4eccdba84dbe253fe8dc18263f31e29e0ca3"' repoLastModified: Sat, 05 Aug 2023 14:13:45 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/3782291211/express-ecommerce-api - source: openapi3 tags repository: https://github.com/devolite/openapi-fetch-tauri v3: true id: d45ef2392f12b45cacd6fd21edb0594f repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWZldGNoLXRhdXJpCgpUaGlzIGxpYnJhcnkgaXMgYW4gW29wZW5hcGktZmV0Y2hdKGh0dHBzOi8vZ2l0aHViLmNvbS9kcndwb3cvb3BlbmFwaS10eXBlc2NyaXB0L3RyZWUvbWFpbi9wYWNrYWdlcy9vcGVuYXBpLWZldGNoKSBjbG9uZSB0aGF0IHVzZXMgW1RhdXJpJ3MgSFRUUCBBUEldKGh0dHBzOi8vdGF1cmkuYXBwL3YxL2FwaS9qcy9odHRwLykuIEJlY2F1c2Ugb2YgdGhpcywgaXQgaXMgYSBkcm9wLWluIHJlcGxhY2VtZW50IGZvciBvcGVuYXBpLWZldGNoLCBhbmQgY2FuIGJlIHVzZWQgaW4gdGhlIHNhbWUgd2F5LiAgCl9UaGFua3MsIFtAZHJ3cG93XShodHRwczovL2dpdGh1Yi5jb20vZHJ3cG93KSFfCgo+IFshQ0FVVElPTl0KPiBEdWUgdG8gbGltaXRhdGlvbnMgaW4gdGhlIFRhdXJpIEhUVFAgQVBJLCB0aGlzIGxpYnJhcnkgKipvbmx5Kiogc3VwcG9ydHMgSlNPTiByZXF1ZXN0cyBhbmQgcmVzcG9uc2VzLgoKIyMgVXNhZ2UKCkluIG9yZGVyIHRvIGdldCBzdGFydGVkLCBnZW5lcmF0ZSBhIHNwZWNpZmljYXRpb24gZmlsZSB1c2luZyBbb3BlbmFwaS10eXBlc2NyaXB0XShodHRwczovL2dpdGh1Yi5jb20vZHJ3cG93L29wZW5hcGktdHlwZXNjcmlwdC90cmVlL21haW4vcGFja2FnZXMvb3BlbmFwaS10eXBlc2NyaXB0KS4KCmBgYHNoCiMgTG9jYWwgc2NoZW1hLi4uCm5weCBvcGVuYXBpLXR5cGVzY3JpcHQgLi9wYXRoL3RvL215L3NjaGVtYS55YW1sIC1vIC4vcGF0aC90by9teS9zY2hlbSAjIG5wbQp5YXJuIGRseCBvcGVuYXBpLXR5cGVzY3JpcHQgLi9wYXRoL3RvL215L3NjaGVtYS5kLnRzIC1vIC4vcGF0aC90by9teS9zY2hlbWEudHMgIyBvciB5YXJuCnBucG0gZGx4IG9wZW5hcGktdHlwZXNjcmlwdCAuL3BhdGgvdG8vbXkvc2NoZW1hLmQudHMgLW8gLi9wYXRoL3RvL215L3NjaGVtYS50cyAjIG9yIHBucG0KIyDwn5qAIC4vcGF0aC90by9teS9zY2hlbWEueWFtbCAtPiAuL3BhdGgvdG8vbXkvc2NoZW1hLmQudHMgWzdtc10KCiMgUmVtb3RlIHNjaGVtYS4uLgpucHggb3BlbmFwaS10eXBlc2NyaXB0IGh0dHBzOi8vZXhhbXBsZS5jb20vc2NoZW1hLnlhbWwgLW8gLi9wYXRoL3RvL215L3NjaGVtYSAjIG5wbQp5YXJuIGRseCBvcGVuYXBpLXR5cGVzY3JpcHQgaHR0cHM6Ly9leGFtcGxlLmNvbS9zY2hlbWEuZC50cyAtbyAuL3BhdGgvdG8vbXkvc2NoZW1hLnRzICMgb3IgeWFybgpwbnBtIGRseCBvcGVuYXBpLXR5cGVzY3JpcHQgaHR0cHM6Ly9leGFtcGxlLmNvbS9zY2hlbWEuZC50cyAtbyAuL3BhdGgvdG8vbXkvc2NoZW1hLnRzICMgb3IgcG5wbQojIPCfmoAgaHR0cHM6Ly9leGFtcGxlLmNvbS9zY2hlbWEueWFtbCAtPiAuL3BhdGgvdG8vbXkvc2NoZW1hLmQudHMgWzdtc10KYGBgCgpUaGVuLCB1dGlsaXplIHRoZSBnZW5lcmF0ZWQgc3BlY2lmaWNhdGlvbiBmaWxlIHRvIG1ha2UgcmVxdWVzdHMuIEluIG9yZGVyIHRvIGRvIHRoaXMsIGNyZWF0ZSBhIGNsaWVudCBsaWtlIHNvOgoKYGBgdHMKaW1wb3J0IHR5cGUgeyBwYXRocyB9IGZyb20gJy4vcGF0aC90by9teS9zY2hlbWEnOwoKZXhwb3J0IGNvbnN0IGNsaWVudCA9IGNyZWF0ZUNsaWVudDxwYXRocz4oewogIGJhc2VVcmw6ICdodHRwczovL2V4YW1wbGUuY29tJwogIC8vIC4uLiBkZWZhdWx0IG9wdGlvbnMKfSk7CgojIG9yCgpleHBvcnQgY29uc3QgeyBHRVQsIFBPU1QsIERFTEVURSAvKiwgLi4uKi8gfSA9IGNyZWF0ZUNsaWVudDxwYXRocz4oewogIGJhc2VVcmw6ICdodHRwczovL2V4YW1wbGUuY29tJwogIC8vIC4uLiBkZWZhdWx0IG9wdGlvbnMKfSk7CmBgYAoKTm93LCB5b3UgY2FuIHVzZSB0aGUgY2xpZW50IHRvIG1ha2UgcmVxdWVzdHMgdGhhdCBtYXRjaCB0aGUgc3BlY2lmaWNhdGlvbjoKCmBgYHRzCmltcG9ydCB7IGNsaWVudCB9IGZyb20gJy4vY2xpZW50JzsKCmNvbnN0IHsgZGF0YSwgZXJyb3IgLyosIHJlc3BvbnNlKi8gfSA9IGF3YWl0IGNsaWVudC5HRVQoJy9wYXRoL3RvL2VuZHBvaW50Jyk7CgojIG9yCgppbXBvcnQgeyBHRVQgfSBmcm9tICcuL2NsaWVudCc7Cgpjb25zdCB7IGRhdGEsIGVycm9yIC8qLCByZXNwb25zZSovICB9ID0gYXdhaXQgR0VUKCcvcGF0aC90by9lbmRwb2ludCcpOwpgYGAKCkZvciBtb3JlIGluZm9ybWF0aW9uLCBzZWUgdGhlIFtvcGVuYXBpLWZldGNoIGRvY3VtZW50YXRpb25dKGh0dHBzOi8vb3BlbmFwaS10cy5wYWdlcy5kZXYvb3BlbmFwaS1mZXRjaC8pLgoKIyMgQ3JlZGl0cwoKQmlnIHRoYW5rcyB0byBbQGRyd3Bvd10oaHR0cHM6Ly9naXRodWIuY29tL2Ryd3BvdykgZm9yIGNyZWF0aW5nIHRoZSBvcmlnaW5hbCBvcGVuYXBpLWZldGNoIGFuZCBvcGVuYXBpLXR5cGVzY3JpcHQgbGlicmFyeSEK readmeEtag: '"3e7b77599e8ac5cf84dac44d0f04cb3db2bc9726"' readmeLastModified: Tue, 26 Mar 2024 13:23:46 GMT repositoryId: 770410140 description: >- 💡 Fast, typesafe fetch client for your OpenAPI schema. Works with Tauri HTTP API. created: '2024-03-11T14:01:14Z' updated: '2025-06-22T08:27:57Z' language: TypeScript archived: true stars: 0 watchers: 1 forks: 1 owner: devolite logo: https://avatars.githubusercontent.com/u/164477852?v=4 license: MIT repoEtag: '"6cf015753e34372552e75684b5bea183d09a48cde7827b2989c9678517465b0f"' repoLastModified: Sun, 22 Jun 2025 08:27:57 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/bddvlpr/openapi-fetch-tauri - source: openapi3 tags repository: https://github.com/stephane-segning/sma v3: true id: 7a593313d75d782c47e19b215fc70eef repositoryMetadata: base64Readme: IyBTY2hvb2wgTWFuYWdlbWVudCBBcHAKCgo= readmeEtag: '"5b12edf2c5f3ff95a33d253df167e7411cb8fae1"' readmeLastModified: Mon, 01 Apr 2024 10:42:34 GMT repositoryId: 780374108 description: >- This project is a sample project to demonstrate the technologies. We're using ReactJS and Spring Boot. created: '2024-04-01T10:42:51Z' updated: '2024-04-01T10:46:42Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: stephane-segning logo: https://avatars.githubusercontent.com/u/26783557?v=4 repoEtag: '"24710ad89d01a5162503aad3cb4312944305a28761855006ee6482dba5bc9f25"' repoLastModified: Mon, 01 Apr 2024 10:46:42 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/markelca/hui-api v3: true repositoryMetadata: base64Readme: >- IyBodWktYXBpClRoaXMgaXMgdGhlIHNlcnZlciBzaWRlIFJFU1QgQVBJIG1hZGUgZm9yIHRoZSBIVUkgUHJveWVjdC4gQnVpbHQgd2l0aCBTcHJpbmcgQm9vdCBhbmQgSGliZXJuYXRlLgoKCiMjIyBTdGFjayBvZiB0ZWNobm9sb2dpZXMgdXNlZDoKCiFbSmF2YV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9qYXZhLXJlZC5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWphdmEmbG9nb0NvbG9yPXdoaXRlKQohW1NwcmluZ10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9zcHJpbmctJTIzNkRCMzNGLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289c3ByaW5nJmxvZ29Db2xvcj13aGl0ZSkKIVtIaWJlcm5hdGVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvaGliZXJuYXRlLWJyb3duLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289aGliZXJuYXRlJmxvZ29Db2xvcj13aGl0ZSkKIVtQb3N0bWFuXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3Bvc3RtYW4tb3JhbmdlLnN2Zz9zdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289cG9zdG1hbiZsb2dvQ29sb3I9d2hpdGUpCiFbSmlyYV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9qaXJhLWJsdWUuc3ZnP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1qaXJhJmxvZ29Db2xvcj13aGl0ZSkKIVtDb25mbHVlbmNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NvbmZsdWVuY2UtZ3JleS5zdmc/c3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvPWNvbmZsdWVuY2UmbG9nb0NvbG9yPXdoaXRlKQohW1BIUG15YWRtaW5dKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvUEhQbXlhZG1pbi1GRjAwRkYuc3ZnP3N0eWxlPWZvci10aGUtYmFkZ2UmbG9nbz1waHBteWFkbWluJmxvZ29Db2xvcj13aGl0ZSkKCgpQb3N0bWFuIEFQSSBSZWZlcmVuY2U6IGh0dHBzOi8vd3d3LnBvc3RtYW4uY29tL2h1aS1hcHAKCi0tLQoKIyMgRGV2ZWxvcG1lbnQgSW5zdGFsYXRpb24KMS4gQ29weSB0aGUgZmlsZSBgYXBwbGljYXRpb24ucHJvcGVydGllcy5zYW1wbGVgIGluIHRoZSBzYW1lIGZvbGRlciAqKGFwcC9zcmMvbWFpbi9yZXNvdXJjZXMpKiBhbmQgbmFtZSBpdCBgYXBwbGljYXRpb24ucHJvcGVydGllc2AKCjIuIENoYW5nZSB0aGUgZm9sbG93aW5nIHZhbHVlcyBmb3IgeW91ciBlbnZpcm9tZW50IG9uZXMuCi0gc2VydmVyLnBvcnQKLSAgc3ByaW5nLmRhdGFzb3VyY2UudXJsCi0gIHNwcmluZy5kYXRhc291cmNlLnVzZXJuYW1lCi0gIHNwcmluZy5kYXRhc291cmNlLnBhc3N3b3JkCmBgYHByb3BlcnRpZXMKc2VydmVyLnBvcnQ9PHBvcnQ+CnNwcmluZy5qcGEuc2hvdy1zcWw9dHJ1ZQpzcHJpbmcuanBhLnByb3BlcnRpZXMuaGliZXJuYXRlLmZvcm1hdF9zcWw9dHJ1ZQpzcHJpbmcuZGF0YXNvdXJjZS5kcml2ZXItY2xhc3MtbmFtZT1jb20ubXlzcWwuY2ouamRiYy5Ecml2ZXIKc3ByaW5nLmRhdGFzb3VyY2UudXJsPWpkYmM6bXlzcWw6Ly9sb2NhbGhvc3Q6MzMwNi88ZGF0YWJhc2U+P2NyZWF0ZURhdGFiYXNlSWZOb3RFeGlzdD10cnVlJnVzZVNTTD1mYWxzZQpzcHJpbmcuZGF0YXNvdXJjZS51c2VybmFtZT08dXNlcj4Kc3ByaW5nLmRhdGFzb3VyY2UucGFzc3dvcmQ9PHBhc3N3b3JkPgpzZXJ2ZXIuZXJyb3IuaW5jbHVkZS1tZXNzYWdlPWFsd2F5cwpzcHJpbmcuanBhLmhpYmVybmF0ZS5kZGwtYXV0bz1jcmVhdGUtZHJvcApzcHJpbmcuanBhLnByb3BlcnRpZXMuaGliZXJuYXRlLmRpYWxlY3Q9b3JnLmhpYmVybmF0ZS5kaWFsZWN0Lk15U1FMNUlubm9EQkRpYWxlY3QKc3ByaW5nLmphY2tzb24uc2VyaWFsaXphdGlvbi5GQUlMX09OX0VNUFRZX0JFQU5TPWZhbHNlCgpgYGAKMy4gUnVuIHRoZSBhcHBsaWNhdGlvbiB3aXRoIHRoZSBwcm9wZXJ0eSBgc3ByaW5nLmpwYS5oaWJlcm5hdGUuZGRsLWF1dG89Y3JlYXRlLWRyb3BgIHRvIGNyZWF0ZSBhbmQgcG9wdWxhdGUgdGhlIGRhdGFiYXNlLCB0aGVuIGNoYW5nZSB0aGUgdmFsdWUgdG8gYHVwZGF0ZWAgaWYgeW91IGRvbnQgd2FudCB0byBjcmVhdGUgdGhlIGRhdGFiYXNlIGVhY2ggdGltZSBpdCBydW5zLgo0LiBBbHNvIGNvbW1lbnQgdGhpcyB0aHJlZSBsaW5lcyBpbiB0aGUgZmlsZSBgYXBwL3NyYy9tYWluL2phdmEvY29tL2dydXBvNS9odWlhcGkvY29uZmlnL0RCQ29uZmlnLmphdmFgIHNvIGl0IGRvZXNuJ3QgY2F1c2UgYSBjb25mbGljdCBpbnNlcnRpbmcgdGhlIGluaXRpYWwgZGF0YS4KYGBgamF2YQogQEJlYW4KICAgIENvbW1hbmRMaW5lUnVubmVyIGNvbW1hbmRMaW5lUnVubmVyKCkgewogICAgICAgIHJldHVybiBhcmdzIC0+IHsKICAgICAgICAgICAgLy9jYXRlZ29yeVJlcG9zaXRvcnkuc2F2ZUFsbCggZ2V0SW5pdGlhbENhdGVnb3JpZXMoKSApOwogICAgICAgICAgICAvL3VzZXJSZXBvc2l0b3J5LnNhdmVBbGwoIGdldEluaXRpYWxVc2VycygpICk7CiAgICAgICAgICAgIC8vZXZlbnRSZXBvc2l0b3J5LnNhdmVBbGwoIGdldEluaXRpYWxFdmVudHMoKSApOwogICAgICAgIH07CiAgICB9CmBgYAoK readmeEtag: '"42a2b1f05f7bdc249f57cfe2f7aa6e5a2f34467a"' readmeLastModified: Thu, 09 Jun 2022 14:51:42 GMT repositoryId: 463172255 description: >- Backend API using Spring Boot + Hibernate for the HUI social events management proyect. created: '2022-02-24T14:08:30Z' updated: '2024-04-28T20:51:18Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: markelca logo: https://avatars.githubusercontent.com/u/76633510?v=4 repoEtag: '"52a2be7e89fa759fc85b02cab0178b76b9fe0a83b9b944aa83f79ff5914e0a9d"' repoLastModified: Sun, 28 Apr 2024 20:51:18 GMT foundInMaster: true category: - Server - Server Implementations id: d9fe16857ea3f56e142dbfb44540e084 - source: https://openapi.tools/ name: MockLab category: Mock language: SaaS link: https://www.mocklab.io/docs/getting-started/ repository: https://www.mocklab.io/docs/getting-started/ source_description: SaaS platform to upload your spec to create a mock server v2: true v3: true foundInMaster: true id: fbcb114edb9df0275737eace42893e58 - source: https://openapi.tools/ name: oas-normalize category: - Parsers - Description Validators - Converters language: - JavaScript - TypeScript repository: https://github.com/readmeio/oas-normalize source_description: >- Tooling for converting, validating, and parsing OpenAPI, Swagger, and Postman API definitions v2: true v3: true v3_1: true id: 463179f425347a95bac9bee6af02cc94 repositoryMetadata: base64Readme: >- PiAqKldhcm5pbmcqKgo+Cj4gYG9hcy1ub3JtYWxpemVgIGhhcyBtb3ZlZCEgVGhlIHNvdXJjZSBmb3IgdGhpcyBsaWJyYXJ5IG5vdyBsaXZlcyBhdCBodHRwczovL2dpdGh1Yi5jb20vcmVhZG1laW8vb2FzLgogCjxwIGFsaWduPSJjZW50ZXIiPgogIDxhIGhyZWY9Imh0dHBzOi8vbnBtLmltL29hcy1ub3JtYWxpemUiPgogICAgPGltZyBzcmM9Imh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzMzNzYyLzIwMDQzNDYyMi0yMzk0Njg2OS0xOTY1LTQ2ZjgtOGRlYi1mMjg0YjhkMGI5MmMucG5nIiBhbHQ9Im9hcy1ub3JtYWxpemUiIC8+CiAgPC9hPgo8L3A+Cgo8cCBhbGlnbj0iY2VudGVyIj4KICBUb29saW5nIGZvciBjb252ZXJ0aW5nLCB2YWxpZGF0aW5nLCBhbmQgcGFyc2luZyBPcGVuQVBJLCBTd2FnZ2VyLCBhbmQgUG9zdG1hbiBBUEkgZGVmaW5pdGlvbnMKPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9ucG0uaW0vb2FzLW5vcm1hbGl6ZSI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbnBtL3Yvb2FzLW5vcm1hbGl6ZT9zdHlsZT1mb3ItdGhlLWJhZGdlIiBhbHQ9Ik5QTSBWZXJzaW9uIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9ucG0uaW0vb2FzLW5vcm1hbGl6ZSI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbm9kZS92L29hcy1ub3JtYWxpemU/c3R5bGU9Zm9yLXRoZS1iYWRnZSIgYWx0PSJOb2RlIFZlcnNpb24iPjwvYT4KICA8YSBocmVmPSJodHRwczovL25wbS5pbS9vYXMtbm9ybWFsaXplIj48aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vbC9vYXMtbm9ybWFsaXplP3N0eWxlPWZvci10aGUtYmFkZ2UiIGFsdD0iTUlUIExpY2Vuc2UiPjwvYT4KICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vcmVhZG1laW8vb2FzLW5vcm1hbGl6ZSI+PGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2FjdGlvbnMvd29ya2Zsb3cvc3RhdHVzL3JlYWRtZWlvL29hcy1ub3JtYWxpemUvY2kueW1sP2JyYW5jaD1tYWluJnN0eWxlPWZvci10aGUtYmFkZ2UiIGFsdD0iQnVpbGQgc3RhdHVzIj48L2E+CjwvcD4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxhIGhyZWY9Imh0dHBzOi8vcmVhZG1lLmNvbSI+PGltZyBzcmM9Imh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9yZWFkbWVpby8uZ2l0aHViL21haW4vb3NzLWJhZGdlLnN2ZyIgLz48L2E+CjwvcD4KCiMjIEluc3RhbGxhdGlvbgoKYGBgYmFzaApucG0gaW5zdGFsbCBvYXMtbm9ybWFsaXplCmBgYAoKIyMgVXNhZ2UKCmBgYGphdmFzY3JpcHQKaW1wb3J0IE9BU05vcm1hbGl6ZSBmcm9tICdvYXMtbm9ybWFsaXplJzsKLy8gY29uc3QgeyBkZWZhdWx0OiBPQVNOb3JtYWxpemUgfSA9IHJlcXVpcmUoJ29hcy1ub3JtYWxpemUnKTsgLy8gSWYgeW91J3JlIHVzaW5nIENKUy4KCmNvbnN0IG9hcyA9IG5ldyBPQVNOb3JtYWxpemUoCiAgJ2h0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL21hc3Rlci9leGFtcGxlcy92My4wL3BldHN0b3JlLWV4cGFuZGVkLnlhbWwnLAogIC8vIC4uLm9yIGEgc3RyaW5nLCBwYXRoLCBKU09OIGJsb2IsIHdoYXRldmVyIHlvdSd2ZSBnb3QuCik7CgpvYXMKICAudmFsaWRhdGUoKQogIC50aGVuKGRlZmluaXRpb24gPT4gewogICAgLy8gRGVmaW5pdGlvbiB3aWxsIGFsd2F5cyBiZSBKU09OLCBhbmQgdmFsaWQuCiAgICBjb25zb2xlLmxvZyhkZWZpbml0aW9uKTsKICB9KQogIC5jYXRjaChlcnIgPT4gewogICAgY29uc29sZS5sb2coZXJyKTsKICB9KTsKYGBgCgojIyMgYCNidW5kbGUoKWAKCj4gKipOb3RlKioKPgo+IEJlY2F1c2UgUG9zdG1hbiBjb2xsZWN0aW9ucyBkb24ndCBzdXBwb3J0IGAkcmVmYCBwb2ludGVycywgdGhpcyBtZXRob2Qgd2lsbCBhdXRvbWF0aWNhbGx5IHVwY29udmVydCBhIFBvc3RtYW4gY29sbGVjdGlvbiB0byBPcGVuQVBJIGlmIHN1cHBsaWVkIG9uZS4KCkJ1bmRsZSB1cCB0aGUgZ2l2ZW4gQVBJIGRlZmluaXRpb24sIHJlc29sdmluZyBhbnkgZXh0ZXJuYWwgYCRyZWZgIHBvaW50ZXJzIGluIHRoZSBwcm9jZXNzLgoKYGBganMKYXdhaXQgb2FzLmJ1bmRsZSgpLnRoZW4oZGVmaW5pdGlvbiA9PiB7CiAgY29uc29sZS5sb2coZGVmaW5pdGlvbik7Cn0pOwpgYGAKCiMjIyBgI2RlcmVmKClgCgo+ICoqTm90ZSoqCj4KPiBCZWNhdXNlIFBvc3RtYW4gY29sbGVjdGlvbnMgZG9uJ3Qgc3VwcG9ydCBgJHJlZmAgcG9pbnRlcnMsIHRoaXMgbWV0aG9kIHdpbGwgYXV0b21hdGljYWxseSB1cGNvbnZlcnQgYSBQb3N0bWFuIGNvbGxlY3Rpb24gdG8gT3BlbkFQSSBpZiBzdXBwbGllZCBvbmUuCgpEZXJlZmVyZW5jZSB0aGUgZ2l2ZW4gQVBJIGRlZmluaXRpb24sIHJlc29sdmluZyBhbGwgYCRyZWZgIHBvaW50ZXJzIGluIHRoZSBwcm9jZXNzLgoKYGBganMKYXdhaXQgb2FzLmRlcmVmKCkudGhlbihkZWZpbml0aW9uID0+IHsKICBjb25zb2xlLmxvZyhkZWZpbml0aW9uKTsKfSk7CmBgYAoKIyMjIGAjdmFsaWRhdGUoeyBjb252ZXJ0VG9MYXRlc3Q/OiBib29sZWFuIH0pYAoKVmFsaWRhdGUgYW5kIG9wdGlvbmFsbHkgY29udmVydCB0byBPcGVuQVBJLCBhIGdpdmVuIEFQSSBkZWZpbml0aW9uLiBUaGlzIHN1cHBvcnRzIFN3YWdnZXIgMi4wLCBPcGVuQVBJIDMueCBBUEkgZGVmaW5pdGlvbnMgYXMgd2VsbCBhcyBQb3N0bWFuIDIueCBjb2xsZWN0aW9ucy4KClBsZWFzZSBub3RlIHRoYXQgaWYgeW91J3ZlIHN1cHBsaWVkIGEgUG9zdG1hbiBjb2xsZWN0aW9uIHRvIHRoZSBsaWJyYXJ5IGl0IHdpbGwgKiphbHdheXMqKiBiZSBjb252ZXJ0ZWQgdG8gT3BlbkFQSSwgdXNpbmcgW0ByZWFkbWUvcG9zdG1hbi10by1vcGVuYXBpXShodHRwczovL25wbS5pbS9AcmVhZG1lL3Bvc3RtYW4tdG8tb3BlbmFwaSksIGFuZCB3ZSB3aWxsIG9ubHkgdmFsaWRhdGUgcmVzdWx0aW5nIE9wZW5BUEkgZGVmaW5pdGlvbi4KCmBgYGpzCmF3YWl0IG9hcy52YWxpZGF0ZSgpLnRoZW4oZGVmaW5pdGlvbiA9PiB7CiAgY29uc29sZS5sb2coZGVmaW5pdGlvbik7Cn0pOwpgYGAKCiMjIyMgT3B0aW9ucwoKPCEtLSBwcmV0dGllci1pZ25vcmUtc3RhcnQgLS0+CnwgT3B0aW9uIHwgVHlwZSB8IERlc2NyaXB0aW9uIHwKfCA6LS0tIHwgOi0tLSB8IDotLS0gfAp8IGBjb252ZXJ0VG9MYXRlc3RgIHwgQm9vbGVhbiB8IEJ5IGRlZmF1bHQgYCN2YWxpZGF0ZWAgd2lsbCBub3QgdXBjb252ZXJ0IFN3YWdnZXIgQVBJIGRlZmluaXRpb25zIHRvIE9wZW5BUEkgc28gaWYgeW91IHdpc2ggZm9yIHRoaXMgdG8gaGFwcGVuLCBzdXBwbHkgYHRydWVgLiB8CjwhLS0gcHJldHRpZXItaWdub3JlLWVuZCAtLT4KCiMjIyMgRXJyb3IgSGFuZGxpbmcKCkZvciB2YWxpZGF0aW9uIGVycm9ycywgd2hlbiBhdmFpbGFibGUsIHlvdSdsbCBnZXQgYmFjayBhbiBvYmplY3Q6CgpgYGBqcwp7CiAgImRldGFpbHMiOiBbCiAgICAvLyBBanYgcGF0aGluZyBlcnJvcnMuIEZvciBleGFtcGxlOgogICAgLyogewogICAgICAiaW5zdGFuY2VQYXRoIjogIi9jb21wb25lbnRzL3NlY3VyaXR5U2NoZW1lcy90bHNBdXRoIiwKICAgICAgInNjaGVtYVBhdGgiOiAiIy9wcm9wZXJ0aWVzL3NlY3VyaXR5U2NoZW1lcy9wYXR0ZXJuUHJvcGVydGllcy8lNUUlNUJhLXpBLVowLTklNUMuJTVDLV8lNUQlMkIlMjQvb25lT2YiLAogICAgICAia2V5d29yZCI6ICJvbmVPZiIsCiAgICAgICJwYXJhbXMiOiB7ICJwYXNzaW5nU2NoZW1hcyI6IG51bGwgfSwKICAgICAgIm1lc3NhZ2UiOiAibXVzdCBtYXRjaCBleGFjdGx5IG9uZSBzY2hlbWEgaW4gb25lT2YiCiAgICB9LCAqLwogIF0KfQpgYGAKCmBtZXNzYWdlYCBpcyBhbG1vc3QgYWx3YXlzIHRoZXJlLCBidXQgYHBhdGhgIGlzIGxlc3MgZGVwZW5kYWJsZS4KCiMjIyBgI3ZlcnNpb24oKWAKCkxvYWQgYW5kIHJldHJpZXZlIHZlcnNpb24gaW5mb3JtYXRpb24gYWJvdXQgYSBzdXBwbGllZCBBUEkgZGVmaW5pdGlvbi4KCmBgYGpzCmF3YWl0IG9hcy52ZXJzaW9uKCkudGhlbigoeyBzcGVjaWZpY2F0aW9uLCB2ZXJzaW9uIH0pID0+IHsKICBjb25zb2xlLmxvZyhzcGVjaWZpY2F0aW9uKTsgLy8gb3BlbmFwaQogIGNvbnNvbGUubG9nKHZlcnNpb24pOyAvLyAzLjEuMAp9KTsKYGBgCgojIyMgT3B0aW9ucwoKIyMjIyMgRW5hYmxlIGxvY2FsIHBhdGhzCgpGb3Igc2VjdXJpdHkgcmVhc29ucywgeW91IG5lZWQgdG8gb3B0IGludG8gYWxsb3dpbmcgZmV0Y2hpbmcgYnkgYSBsb2NhbCBwYXRoLiBUbyBlbmFibGUgaXQgc3VwcGx5IHRoZSBgZW5hYmxlUGF0aHNgIG9wdGlvbiB0byB0aGUgY2xhc3MgaW5zdGFuY2U6CgpgYGBqcwpjb25zdCBvYXMgPSBuZXcgT0FTTm9ybWFsaXplKCcuL3BldHN0b3JlLmpzb24nLCB7IGVuYWJsZVBhdGhzOiB0cnVlIH0pOwpgYGAKCiMjIyMjIENvbG9yaXplZCBlcnJvcnMKCklmIHlvdSB3aXNoIGVycm9ycyBmcm9tIGAudmFsaWRhdGUoKWAgdG8gYmUgc3R5bGVkIGFuZCBjb2xvcml6ZWQsIHN1cHBseSBgY29sb3JpemVFcnJvcnM6IHRydWVgIHRvIHlvdXIgaW5zdGFuY2Ugb2YgYE9BU05vcm1hbGl6ZWA6CgpgYGBqcwpjb25zdCBvYXMgPSBuZXcgT0FTTm9ybWFsaXplKCdodHRwczovL2V4YW1wbGUuY29tL3BldHN0b3JlLmpzb24nLCB7CiAgY29sb3JpemVFcnJvcnM6IHRydWUsCn0pOwpgYGAKCkVycm9yIG1lc3NhZ2VzIHdpbGwgbG9vayBsaWtlIHN1Y2g6Cgo8aW1nIHNyYz0iaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vMzM3NjIvMTM3Nzk2NjQ4LTdlMTE1N2MyLWNlZTQtNDY2ZS05MTI5LWRkMmE3NDNkZDE2My5wbmciIHdpZHRoPSI2MDAiIC8+Cg== readmeEtag: '"792a400fc9482536a4e6b3899ff273c05ffdf20e"' readmeLastModified: Thu, 05 Oct 2023 17:29:18 GMT repositoryId: 134783835 description: >- Tooling for converting, validating, and parsing OpenAPI, Swagger, and Postman API definitions. created: '2018-05-25T00:48:24Z' updated: '2025-07-18T14:03:24Z' language: TypeScript archived: true stars: 37 watchers: 16 forks: 7 owner: readmeio logo: https://avatars.githubusercontent.com/u/6878153?v=4 license: MIT repoEtag: '"c31977508f096cedbb24696f029f1cd4067e393b940f5e23e06d371e9f21f63b"' repoLastModified: Fri, 18 Jul 2025 14:03:24 GMT foundInMaster: true - source: https://openapi.tools/ name: Fitting category: - Testing - Data Validators - Learning language: Ruby repository: https://github.com/matchtechnologies/fitting link: https://github.com/matchtechnologies/fitting source_description: >- Library add improve test log for RSpec and WebMock, validate its according to API Blueprint and Open API, show the documentation coverage with log. v2: true v3: true v3_1: true id: cc617beb32d7138f1537294767083922 repositoryMetadata: base64Readme: >- # Fitting

<img align="right" width="192" height="192"
alt="Fitting avatar: Documents with hangers"
src="./images/logo.png">

Note: The project has been moved to https://github.com/tuwilof/fitting

Library add improve test log, validate its according to your API documentation, show the documentation coverage with log.

Test log setting supports RSpec test and WebMock stubbing for Ruby On Rails application, API documentation supports API Blueprint and OpenAPI.

This reduces the costs of support, testers and analysts.

Log
```text
FITTING incoming request {"method":"POST","path":"/public/api/v1/inboxes/tEX5JiZyceiwuKMi1oN9Sf8S/contacts","body":{},"response":{"status":200,"content_type":"application/json","body":{"source_id":"00dbf18d-879e-47cb-ac45-e9aece266eb1","pubsub_token":"ktn6YwPus57JDf4e59eFPom5","id":3291,"name":"shy-surf-401","email":null,"phone_number":null}},"title":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb:9","group":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb","host":"www.example.com"}
FITTING outgoing request {"method":"POST","path":"/v1/organizations/org_id/meeting","body":{},"response":{"status":200,"content_type":"application/json","body":{"success":true,"data":{"meeting":{"id":"meeting_id","roomName":"room_name"}}}},"title":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb:50","group":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb","host":"api.cluster.dyte.in"}
```

validation
```console
FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.FFF..FFFFFFFFFF....F.......F...FF.....F...F....F..............................FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF.F..FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF........FFF...FFFF......FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF........FFFFFFFFFFF..FFFFFF..FFFFFFFFFFFFFFFFF.......FFFFFF.............FFFFFFFFFFFF....F........FFF.F...FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF............FF........FFF......FFFFFFFFFFFFFFFFFFFFFF....FFFFFF......F............FFFF........FFFFFFFFFFFFFF.....FFFFFFFFFFFFFFFFFFFFFFF..FF.....FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.....FF..........FFFFFFFFFFFFFFFFFF...FFFF...............F.F....FF..FFFFFFFF

  1) Fitting::Doc::NotFound log error:

host: www.example.com
method: POST
path: /public/api/v1/inboxes/{inbox_identifier}/contacts
code: 200

content-type: application/json

json-schema: {
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "properties": {
    "id": {
      "type": "integer",
      "description": "Id of the contact"
    },
    "source_id": {
      "type": "string",
      "description": "The session identifier of the contact"
    },
    "name": {
      "type": "string",
      "description": "Name of the contact"
    },
    "email": {
      "type": "string",
      "description": "Email of the contact"
    },
    "pubsub_token": {
      "type": "string",
      "description": "The token to be used to connect to chatwoot websocket"
    }
  }
}

body: {
  "source_id": "c9e8c31f-06df-49b4-8fb9-4466457ae65b",
  "pubsub_token": "Zgc7DEvaj5TkgZ1a4C7AvJXo",
  "id": 3293,
  "name": "restless-snowflake-670",
  "email": null,
  "phone_number": null
}

error [
  "The property '#/email' of type null did not match the following type: string in schema e56b7e65-d70c-5f7a-a96c-982df5f8f2f7"
]

...

804 examples, 565 failure, 0 pending

Coverage: 65.51%
```

and cover
![exmaple](images/b1.png)

![exmaple](images/b2.png)

![exmaple](images/w1.png)

![exmaple](images/w2.png)

## Installation
Add this line to your application's Gemfile:
```ruby
gem 'fitting'
```

After that execute:
```bash
$ bundle
```

Or install the gem by yourself:
```bash
$ gem install fitting
```

## Usage
### Log
Firstly, improve `test.log`.

To your `spec_helper.rb`:

```ruby
require 'fitting'

Fitting.logger
```

Delete all files `log/*.log` and run rspec

You get more information about incoming and outgoing request in `log/fitting*.log`.

```text
FITTING incoming request {"method":"POST","path":"/public/api/v1/inboxes/tEX5JiZyceiwuKMi1oN9Sf8S/contacts","body":{},"response":{"status":200,"content_type":"application/json","body":{"source_id":"00dbf18d-879e-47cb-ac45-e9aece266eb1","pubsub_token":"ktn6YwPus57JDf4e59eFPom5","id":3291,"name":"shy-surf-401","email":null,"phone_number":null}},"title":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb:9","group":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb","host":"www.example.com"}
FITTING outgoing request {"method":"POST","path":"/v1/organizations/org_id/meeting","body":{},"response":{"status":200,"content_type":"application/json","body":{"success":true,"data":{"meeting":{"id":"meeting_id","roomName":"room_name"}}}},"title":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb:50","group":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb","host":"api.cluster.dyte.in"}
```

### Validation
Secondly, validate the log to the documentation.

Add this to your `.fitting.yml`:

```yaml
APIs:
  - host: www.example.com
    type: openapi2
    path: swagger/swagger.json
```

Run 
```bash
bundle e rake fitting:validate
```

Console output

```console
FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.FFF..FFFFFFFFFF....F.......F...FF.....F...F....F..............................FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF.F..FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF........FFF...FFFF......FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF........FFFFFFFFFFF..FFFFFF..FFFFFFFFFFFFFFFFF.......FFFFFF.............FFFFFFFFFFFF....F........FFF.F...FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF............FF........FFF......FFFFFFFFFFFFFFFFFFFFFF....FFFFFF......F............FFFF........FFFFFFFFFFFFFF.....FFFFFFFFFFFFFFFFFFFFFFF..FF.....FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.....FF..........FFFFFFFFFFFFFFFFFF...FFFF...............F.F....FF..FFFFFFFF

  1) Fitting::Doc::NotFound log error:

host: www.example.com
method: POST
path: /public/api/v1/inboxes/{inbox_identifier}/contacts
code: 200

content-type: application/json

json-schema: {
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "properties": {
    "id": {
      "type": "integer",
      "description": "Id of the contact"
    },
    "source_id": {
      "type": "string",
      "description": "The session identifier of the contact"
    },
    "name": {
      "type": "string",
      "description": "Name of the contact"
    },
    "email": {
      "type": "string",
      "description": "Email of the contact"
    },
    "pubsub_token": {
      "type": "string",
      "description": "The token to be used to connect to chatwoot websocket"
    }
  }
}

body: {
  "source_id": "c9e8c31f-06df-49b4-8fb9-4466457ae65b",
  "pubsub_token": "Zgc7DEvaj5TkgZ1a4C7AvJXo",
  "id": 3293,
  "name": "restless-snowflake-670",
  "email": null,
  "phone_number": null
}

error [
  "The property '#/email' of type null did not match the following type: string in schema e56b7e65-d70c-5f7a-a96c-982df5f8f2f7"
]

...

804 examples, 565 failure, 0 pending

Coverage: 65.51%
```

### Coverage
And task will create HTML (`coverage/fitting.html`) reports.

![exmaple](images/b1.png)

![exmaple](images/b2.png)

More information on action coverage

![exmaple2](images/w1.png)

![exmaple2](images/w2.png)

## Settings

### APIs

#### type

##### OpenAPI 2.0
Swagger

```yaml
APIs:
  - host: www.example.com
    type: openapi2
    path: doc/api.json
```

##### OpenAPI 3.0
Also OpenAPI

```yaml
APIs:
  - host: www.example.com
    type: openapi3
    path: doc/api.json
```

##### API Blueprint
First you need to install [drafter](https://github.com/apiaryio/drafter) or [crafter](https://github.com/funbox/crafter).
Works after conversion from API Blueprint to API Elements (in YAML file) with Drafter or Crafter.

That is, I mean that you first need to do this

```bash
drafter doc.apib -o doc.yaml
```

or

```bash
node_modules/.bin/crafter doc.apib > doc.yaml
```

and then

```yaml
APIs:
  - host: www.example.com
    type: drafter
    path: doc/api.yaml
```

or

```yaml
APIs:
  - host: www.example.com
    type: crafter
    path: doc/api.yaml
```

##### Tomograph

To use additional features of the pre-converted [tomograph](https://github.com/funbox/tomograph)

example

```bash
bundle exec tomograph -d crafter --exclude-description doc/api.yml doc/api.json
```

and then

```yaml
APIs:
  - host: www.example.com
    type: tomogram
    path: doc/api.json
```

#### prefix

Setting the prefix name is optional. For example, you can do this:

```yaml
APIs:
  - host: www.example.com
    prefix: /api/v3
    type: openapi2
    path: swagger/swagger.json
```

### SkipValidation

#### host

It is not necessary to immediately describe each host in detail, you can only specify its name and skip it until you are ready to documented it

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
```

#### prefix

If you want to skip a specific prefix in the host

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
    prefix: /admin/api
```

#### method and path

If you want to skip a specific request in the host

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
    method: GET
    path: /api/v1/cars
```

### NoCov

It is not necessary to immediately test each doc in detail, you can only specify its name and skip it until you are ready to test it

#### host
```yaml
NoCov:
  - host: sso.test
```

#### method
```yaml
NoCov:
  - host: sso.test
    method: GET
```

#### path
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
```

#### code
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
```

#### content-type
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
```

#### combination
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
    combination: oneOf.0
```

#### combination_next
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
    combination: oneOf.0
    combination_next: oneOf.0.required.users
```

### Debug

If you find bug, you can debug it or create task in this github project  with new file `coverage/fitting.debug.yml`

```yaml
Debug:
  - host: www.example.com
    method: GET
    path: /api/v3/users
    code: 200
    content-type: application/json
```

## Contributing

Bug reports and pull requests are welcome on GitHub at [github.com/funbox/fitting](https://github.com/funbox/fitting).
This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.

## License

The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).

[![Sponsored by FunBox](https://funbox.ru/badges/sponsored_by_funbox_centered.svg)](https://funbox.ru)
 readmeEtag: '"00e2b176ef7d96987a960eeb75f854fc49d188b1"' readmeLastModified: Wed, 03 Apr 2024 20:24:31 GMT repositoryId: 780564538 description: >- Library add improve test log for RSpec and WebMock, validate its according to API Blueprint and Open API, show the documentation coverage with log. created: '2024-04-01T18:32:22Z' updated: '2024-04-03T20:25:02Z' language: Ruby archived: true stars: 0 watchers: 0 forks: 0 owner: matchtechnologies logo: https://avatars.githubusercontent.com/u/147186495?v=4 license: MIT repoEtag: '"16a3992724535acef9508317ddf92438ce826102844a21cb5bbdb7fb97cbede8"' repoLastModified: Wed, 03 Apr 2024 20:25:02 GMT foundInMaster: true - source: https://openapi.tools/ name: kubb category: Code Generators language: Typescript source_description: >- Kubb is a tool that can generate TypeScript types, Zod schemas, react-query hooks and much more. It has support for Tanstack Query(React, Solid, Svelte and Vue), SWR(React), Zod, Zodios and Axios. Kubb is made based on a plugin system, meaning you can create your own plugin and couple it with the Kubb ecosystem. link: https://kubb.dev/ repository: https://github.com/kubb-labs/kubb v2: true v3: true v3_1: true id: 8deb5ec0f25533d10b0d6025d587f55c repositoryMetadata: base64Readme: >- <div align="center">
  <a href="https://kubb.dev/kubb" target="_blank" rel="noopener noreferrer">
    <img width="180" src="https://raw.githubusercontent.com/kubb-labs/kubb/main/assets/logo.png" alt="Kubb logo">
  </a>


[![npm version][npm-version-src]][npm-version-href]
[![npm downloads][npm-downloads-src]][npm-downloads-href]
[![Coverage][coverage-src]][coverage-href]
[![License][license-src]][license-href]
[![Sponsors][sponsors-src]][sponsors-href]
[![smithery badge](https://smithery.ai/badge/@kubb-labs/kubb)](https://smithery.ai/server/@kubb-labs/kubb)
<h4>
<a href="https://codesandbox.io/s/github/kubb-labs/kubb/tree/main/examples/typescript" target="_blank">View Demo</a>
<span> · </span>
<a href="https://kubb.dev/kubb" target="_blank">Documentation</a>
<span> · </span>
<a href="https://github.com/kubb-labs/kubb/issues/" target="_blank">Report Bug</a>
<span> · </span>
<a href="https://github.com/kubb-labs/kubb/issues/" target="_blank">Request Feature</a>
</h4>
</div>

<br />

## Quick Start

Get started with Kubb in seconds:

```bash
npx kubb init
```

The interactive setup will:
- Create a `package.json` (if needed)
- Guide you through plugin selection
- Install packages automatically
- Generate `kubb.config.ts`

Then generate your code:

```bash
npx kubb generate
```

See the [documentation](https://kubb.dev) for detailed usage and advanced features.

## Features
- Works with Node.js 20+.
- Convert Swagger 2.0, OpenAPI 3.0, and OpenAPI 3.1 to TypeScript, Zod, React-Query, ...
- Plugin ecosystem to extend beyond the default plugins we provide.
- CLI support with progress bar and detailed logs.
- Model Context Protocol (MCP) server for AI assistants like [Claude](https://claude.ai), [Cursor](https://cursor.sh), and other MCP-compatible tools.
- Debug tools with React DevTools.
- Generates barrel files (index.ts).
- And so much more ...

## Supporting Kubb

Kubb uses an MIT-licensed open source project with its ongoing development made possible entirely by the support of Sponsors. If you would like to become a sponsor, please consider:

- [Become a Sponsor on GitHub](https://github.com/sponsors/stijnvanhulle)

<p align="center">
  <a href="https://github.com/sponsors/stijnvanhulle">
    <img src="https://raw.githubusercontent.com/stijnvanhulle/sponsors/main/sponsors.svg" alt="My sponsors" />
  </a>
</p>

## Contributors [![Contributors][contributors-src]][contributors-href]

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tbody>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="http://www.stijnvanhulle.be"><img src="https://avatars.githubusercontent.com/u/5904681?v=4?s=100" width="100px;" alt="Stijn Van Hulle"/><br /><sub><b>Stijn Van Hulle</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=stijnvanhulle" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://aluc.io/"><img src="https://avatars.githubusercontent.com/u/15520015?v=4?s=100" width="100px;" alt="Alfred"/><br /><sub><b>Alfred</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=b6pzeusbc54tvhw5jgpyw8pwz2x6gs" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/raveclassic"><img src="https://avatars.githubusercontent.com/u/1743568?v=4?s=100" width="100px;" alt="Kirill Agalakov"/><br /><sub><b>Kirill Agalakov</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=raveclassic" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://wicky.nillia.ms"><img src="https://avatars.githubusercontent.com/u/1091390?v=4?s=100" width="100px;" alt="Nick Williams"/><br /><sub><b>Nick Williams</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=WickyNilliams" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/helt"><img src="https://avatars.githubusercontent.com/u/1732112?v=4?s=100" width="100px;" alt="helt"/><br /><sub><b>helt</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=helt" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Ti-webdev"><img src="https://avatars.githubusercontent.com/u/478565?v=4?s=100" width="100px;" alt="Vasily Mikhaylovsky"/><br /><sub><b>Vasily Mikhaylovsky</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=Ti-webdev" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/chiptus"><img src="https://avatars.githubusercontent.com/u/1381655?v=4?s=100" width="100px;" alt="Chaim Lev-Ari"/><br /><sub><b>Chaim Lev-Ari</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=chiptus" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="http://projects.pafnuty.name"><img src="https://avatars.githubusercontent.com/u/1635679?v=4?s=100" width="100px;" alt="Pavel Belousov"/><br /><sub><b>Pavel Belousov</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=pafnuty" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/dmitry-blackwave"><img src="https://avatars.githubusercontent.com/u/5526543?v=4?s=100" width="100px;" alt="Dmitry Belov"/><br /><sub><b>Dmitry Belov</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=dmitry-blackwave" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/aburgel"><img src="https://avatars.githubusercontent.com/u/341478?v=4?s=100" width="100px;" alt="Alex Burgel"/><br /><sub><b>Alex Burgel</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=aburgel" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/dgarciamuria"><img src="https://avatars.githubusercontent.com/u/8144333?v=4?s=100" width="100px;" alt="Daniel Garcia"/><br /><sub><b>Daniel Garcia</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=dgarciamuria" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/wuyuanyi135"><img src="https://avatars.githubusercontent.com/u/11760870?v=4?s=100" width="100px;" alt="wuyuanyi135"/><br /><sub><b>wuyuanyi135</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=wuyuanyi135" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/cjthompson"><img src="https://avatars.githubusercontent.com/u/1958266?v=4?s=100" width="100px;" alt="Chris Thompson"/><br /><sub><b>Chris Thompson</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=cjthompson" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/hkang1"><img src="https://avatars.githubusercontent.com/u/220971?v=4?s=100" width="100px;" alt="Caleb Hoyoul Kang"/><br /><sub><b>Caleb Hoyoul Kang</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=hkang1" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/grreeenn"><img src="https://avatars.githubusercontent.com/u/13204857?v=4?s=100" width="100px;" alt="Gregory Zhukovsky"/><br /><sub><b>Gregory Zhukovsky</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=grreeenn" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ChilloManiac"><img src="https://avatars.githubusercontent.com/u/3761964?v=4?s=100" width="100px;" alt="Christoffer Nørbjerg"/><br /><sub><b>Christoffer Nørbjerg</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=ChilloManiac" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://codefy.de/de/karriere"><img src="https://avatars.githubusercontent.com/u/122524301?v=4?s=100" width="100px;" alt="CHE1RON"/><br /><sub><b>CHE1RON</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=CHE1RON" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/ekaradon"><img src="https://avatars.githubusercontent.com/u/9439390?v=4?s=100" width="100px;" alt="ekaradon"/><br /><sub><b>ekaradon</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=ekaradon" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://thijmen.dev"><img src="https://avatars.githubusercontent.com/u/383903?v=4?s=100" width="100px;" alt="Thijmen Stavenuiter"/><br /><sub><b>Thijmen Stavenuiter</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=Thijmen" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/bohdanhusak"><img src="https://avatars.githubusercontent.com/u/13829370?v=4?s=100" width="100px;" alt="Bohdan Husak"/><br /><sub><b>Bohdan Husak</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=bohdanhusak" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Ericlm"><img src="https://avatars.githubusercontent.com/u/19361503?v=4?s=100" width="100px;" alt="Éric Le Maître"/><br /><sub><b>Éric Le Maître</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=Ericlm" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/chambber"><img src="https://avatars.githubusercontent.com/u/11406841?v=4?s=100" width="100px;" alt="Rubens Pereira do Nascimento"/><br /><sub><b>Rubens Pereira do Nascimento</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=chambber" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/msutkowski"><img src="https://avatars.githubusercontent.com/u/784953?v=4?s=100" width="100px;" alt="Matt Sutkowski"/><br /><sub><b>Matt Sutkowski</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=msutkowski" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/vitorcamachoo"><img src="https://avatars.githubusercontent.com/u/20595956?v=4?s=100" width="100px;" alt="Vítor Camacho"/><br /><sub><b>Vítor Camacho</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=vitorcamachoo" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/VasekProchazka"><img src="https://avatars.githubusercontent.com/u/13906845?v=4?s=100" width="100px;" alt="Václav Procházka"/><br /><sub><b>Václav Procházka</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=VasekProchazka" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://codx.dev"><img src="https://avatars.githubusercontent.com/u/59735735?v=4?s=100" width="100px;" alt="Luiz Bett"/><br /><sub><b>Luiz Bett</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=heyBett" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/lambdank"><img src="https://avatars.githubusercontent.com/u/5475129?v=4?s=100" width="100px;" alt="Sebastian Andersen"/><br /><sub><b>Sebastian Andersen</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=lambdank" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://akino.icu"><img src="https://avatars.githubusercontent.com/u/64176534?v=4?s=100" width="100px;" alt="Akino"/><br /><sub><b>Akino</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=akinoccc" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/rmachado-studocu"><img src="https://avatars.githubusercontent.com/u/89906313?v=4?s=100" width="100px;" alt="Ricardo Machado"/><br /><sub><b>Ricardo Machado</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=rmachado-studocu" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://simonelnahas.com"><img src="https://avatars.githubusercontent.com/u/29279201?v=4?s=100" width="100px;" alt="Simon El Nahas"/><br /><sub><b>Simon El Nahas</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=simonelnahas" title="Documentation">📖</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/maartenvansambeek"><img src="https://avatars.githubusercontent.com/u/91739524?v=4?s=100" width="100px;" alt="maartenvansambeek"/><br /><sub><b>maartenvansambeek</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=maartenvansambeek" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://sdufresne.info"><img src="https://avatars.githubusercontent.com/u/583851?v=4?s=100" width="100px;" alt="Stefan du Fresne"/><br /><sub><b>Stefan du Fresne</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=SCdF" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://hugofelippe.github.io/"><img src="https://avatars.githubusercontent.com/u/19368365?v=4?s=100" width="100px;" alt="Hugo Felippe de Souza Cruz"/><br /><sub><b>Hugo Felippe de Souza Cruz</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=hugoFelippe" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/belgattitude"><img src="https://avatars.githubusercontent.com/u/259798?v=4?s=100" width="100px;" alt="Sébastien Vanvelthem"/><br /><sub><b>Sébastien Vanvelthem</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=belgattitude" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://bento.me/vitalygashkov"><img src="https://avatars.githubusercontent.com/u/30000398?v=4?s=100" width="100px;" alt="Vitaly Gashkov"/><br /><sub><b>Vitaly Gashkov</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=vitalygashkov" title="Documentation">📖</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://ducduc.nl"><img src="https://avatars.githubusercontent.com/u/9675738?v=4?s=100" width="100px;" alt="Duco Drupsteen"/><br /><sub><b>Duco Drupsteen</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=ducodrupsteen" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/th3l0g4n"><img src="https://avatars.githubusercontent.com/u/326306?v=4?s=100" width="100px;" alt="th3l0g4n"/><br /><sub><b>th3l0g4n</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=th3l0g4n" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://rxliuli.com"><img src="https://avatars.githubusercontent.com/u/24560368?v=4?s=100" width="100px;" alt="rxliuli"/><br /><sub><b>rxliuli</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=rxliuli" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/humarkx"><img src="https://avatars.githubusercontent.com/u/13049940?v=4?s=100" width="100px;" alt="humarkx"/><br /><sub><b>humarkx</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=humarkx" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/Jakub-Cerovsky"><img src="https://avatars.githubusercontent.com/u/141134227?v=4?s=100" width="100px;" alt="Jakub Cerovsky"/><br /><sub><b>Jakub Cerovsky</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=Jakub-Cerovsky" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/yukikwi"><img src="https://avatars.githubusercontent.com/u/66879660?v=4?s=100" width="100px;" alt="Pachara Chantawong"/><br /><sub><b>Pachara Chantawong</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=yukikwi" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://volodymyrkushnir.dev/"><img src="https://avatars.githubusercontent.com/u/10290626?v=4?s=100" width="100px;" alt="Volodymyr Kushnir"/><br /><sub><b>Volodymyr Kushnir</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=volodymyr-kushnir" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/GKNewsrooms"><img src="https://avatars.githubusercontent.com/u/201248633?v=4?s=100" width="100px;" alt="GKNewsrooms"/><br /><sub><b>GKNewsrooms</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=GKNewsrooms" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/stepek"><img src="https://avatars.githubusercontent.com/u/5058678?v=4?s=100" width="100px;" alt="Kamil Stepczuk"/><br /><sub><b>Kamil Stepczuk</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=stepek" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/JoaoBrlt"><img src="https://avatars.githubusercontent.com/u/11065509?v=4?s=100" width="100px;" alt="João Brilhante"/><br /><sub><b>João Brilhante</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=JoaoBrlt" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/kamilzki"><img src="https://avatars.githubusercontent.com/u/27976736?v=4?s=100" width="100px;" alt="Kamil Sieradzki"/><br /><sub><b>Kamil Sieradzki</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=kamilzki" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="https://github.com/EricPierlotIdmog"><img src="https://avatars.githubusercontent.com/u/124898024?v=4?s=100" width="100px;" alt="Eric Pierlot"/><br /><sub><b>Eric Pierlot</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=EricPierlotIdmog" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://atholin.se"><img src="https://avatars.githubusercontent.com/u/33940473?v=4?s=100" width="100px;" alt="Alexander Sjöcrona Tholin"/><br /><sub><b>Alexander Sjöcrona Tholin</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=ATholin" title="Code">💻</a></td>
      <td align="center" valign="top" width="14.28%"><a href="http://hyoban.cc"><img src="https://avatars.githubusercontent.com/u/38493346?v=4?s=100" width="100px;" alt="Stephen Zhou"/><br /><sub><b>Stephen Zhou</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=hyoban" title="Code">💻</a></td>
    </tr>
    <tr>
      <td align="center" valign="top" width="14.28%"><a href="https://choly.ca"><img src="https://avatars.githubusercontent.com/u/943597?v=4?s=100" width="100px;" alt="Ilia Choly"/><br /><sub><b>Ilia Choly</b></sub></a><br /><a href="https://github.com/kubb-labs/kubb/commits?author=icholy" title="Code">💻</a></td>
    </tr>
  </tbody>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

## Star History

<a href="https://star-history.com/#kubb-labs/kubb&Date">
  <picture>
    <source media="(prefers-color-scheme: dark)" srcset="https://api.star-history.com/svg?repos=kubb-labs/kubb&type=Date&theme=dark" />
    <source media="(prefers-color-scheme: light)" srcset="https://api.star-history.com/svg?repos=kubb-labs/kubb&type=Date" />
    <img alt="Star History Chart" src="https://api.star-history.com/svg?repos=kubb-labs/kubb&type=Date" />
  </picture>
</a>


<!-- Badges -->

[npm-version-src]: https://img.shields.io/npm/v/@kubb/core?flat&colorA=18181B&colorB=f58517
[npm-version-href]: https://npmx.dev/package/@kubb/core
[npm-downloads-src]: https://img.shields.io/npm/dm/@kubb/core?flat&colorA=18181B&colorB=f58517
[npm-downloads-href]: https://npmjs.com/package/@kubb/core
[license-src]: https://img.shields.io/github/license/kubb-labs/kubb.svg?flat&colorA=18181B&colorB=f58517
[license-href]: https://github.com/kubb-labs/kubb/blob/main/LICENSE
[build-src]: https://img.shields.io/github/actions/workflow/status/kubb-labs/kubb/ci.yaml?style=flat&colorA=18181B&colorB=f58517
[build-href]: https://www.npmjs.com/package/@kubb/core
[minified-src]: https://img.shields.io/bundlephobia/min/@kubb/core?style=flat&colorA=18181B&colorB=f58517
[minified-href]: https://www.npmjs.com/package/@kubb/core
[coverage-src]: https://img.shields.io/codecov/c/github/kubb-labs/kubb?style=flat&colorA=18181B&colorB=f58517
[coverage-href]: https://www.npmjs.com/package/@kubb/core
[contributors-src]: https://img.shields.io/github/contributors/kubb-labs/kubb?style=flat&colorA=18181B&colorB=f58517&label=%20
[contributors-href]: #contributors-
[sponsors-src]: https://img.shields.io/github/sponsors/stijnvanhulle?style=flat&colorA=18181B&colorB=f58517
[sponsors-href]: https://github.com/sponsors/stijnvanhulle/
 readmeEtag: '"3c427ee88039a6e22f530c45074f0f7250c7b399"' readmeLastModified: Thu, 05 Feb 2026 23:43:13 GMT repositoryId: 586543156 description: >- 🧩 The Ultimate Toolkit for Generating Type-Safe API Clients, Hooks, and Validators. created: '2023-01-08T14:15:23Z' updated: '2026-02-05T23:43:18Z' language: TypeScript archived: false stars: 1585 watchers: 5 forks: 130 owner: kubb-labs logo: https://avatars.githubusercontent.com/u/122118236?v=4 license: MIT repoEtag: '"5498a7d95f4a652c6c3f3afa94d5639bcb9b12ddddf7ca1637a59edeeb87a9a5"' repoLastModified: Thu, 05 Feb 2026 23:43:18 GMT foundInMaster: true oldLocations: - https://github.com/kubb-project/kubb - source: openapi3 tags repository: https://github.com/matchtechnologies/tomograph v3: true id: b656fa0887abd5cc2460fcf686e1ee21 repositoryMetadata: base64Readme: >- # Tomograph 

Note: The project has been moved to https://github.com/tuwilof/tomograph

Convert API Blueprint, Swagger and OpenAPI to minimal routes with JSON Schema. For ease of use and creation of new tools.

Will look like

  ```json
  [
    {
      "path": "/sessions",
      "method": "POST",
      "content-type": "application/json",
      "requests": [{
        "$schema": "http://json-schema.org/draft-04/schema#",
        "type": "object",
        "properties": {
          "login": {
            "type": "string"
          },
          "password": {
            "type": "string"
          },
          "captcha": {
            "type": "string"
          }
        },
        "required": [
          "login",
          "password"
        ]
      }],
      "responses": [
        {
          "status": "401",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "429",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "201",
          "content-type": "application/json",
          "body": {
            "$schema": "http://json-schema.org/draft-04/schema#",
            "type": "object",
            "properties": {
              "confirmation": {
                "type": "object",
                "properties": {
                  "id": {
                    "type": "string"
                  },
                  "type": {
                    "type": "string"
                  },
                  "operation": {
                    "type": "string"
                  }
                },
                "required": [
                  "id",
                  "type",
                  "operation"
                ]
              },
              "captcha": {
                "type": "string"
              },
              "captcha_does_not_match": {
                "type": "boolean"
              }
            }
          }
        }
      ]
    }
  ]
  ```

## Installation

Then add this line to your application's Gemfile:

```ruby
gem 'tomograph'
```

After that execute:

```bash
$ bundle
```

Or install the gem by yourself:

```bash
$ gem install tomograph
```

## Usage

### In code

#### OpenAPI 2.0

Also Swagger

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(openapi2_json_path: '/path/to/doc.json')
```

#### OpenAPI 3.0

Also OpenAPI

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(openapi3_yaml_path: '/path/to/doc.yaml')
```

#### API Blueprint

First you need to install [drafter](https://github.com/apiaryio/drafter).
Works after conversion from API Blueprint to API Elements (in YAML file) with Drafter.

That is, I mean that you first need to do this

```bash
drafter doc.apib -o doc.yaml
```

and then

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(drafter_yaml_path: '/path/to/doc.yaml')
```

#### Tomograph

To use additional features of the pre-converted

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(tomogram_json_path: '/path/to/doc.json')
```

#### prefix
Default: `''`

You can specify API prefix and path to the spec using one of the possible formats:

```ruby
Tomograph::Tomogram.new(prefix: '/api/v2', drafter_yaml_path: '/path/to/doc.yaml')
```

```ruby
Tomograph::Tomogram.new(prefix: '/api/v2', tomogram_json_path: '/path/to/doc.json')
```

#### to_json
Use `to_json` for converting to JSON, example from API Blueprint:

```ruby
tomogram.to_json
```

<details>
  <summary>Example input</summary>

  ```apib
  FORMAT: 1A
  HOST: http://test.local
  
  # project
  
  # Group project
  
  Project
  
  ## Authentication [/sessions]
  
  ### Sign In [POST]
  
  + Request (application/json)
  
      + Attributes
       + login (string, required)
       + password (string, required)
       + captcha (string, optional)
  
  + Response 401 (application/json)
  
  + Response 429 (application/json)
  
  + Response 201 (application/json)
  
      + Attributes
       + confirmation (Confirmation, optional)
       + captcha (string, optional)
       + captcha_does_not_match (boolean, optional)
  
  
  # Data Structures
  
  ## Confirmation (object)
    + id (string, required)
    + type (string, required)
    + operation (string, required)
  ```
</details>

<details>
  <summary>Example output</summary>

  ```json
  [
    {
      "path": "/sessions",
      "method": "POST",
      "content-type": "application/json",
      "requests": [{
        "$schema": "http://json-schema.org/draft-04/schema#",
        "type": "object",
        "properties": {
          "login": {
            "type": "string"
          },
          "password": {
            "type": "string"
          },
          "captcha": {
            "type": "string"
          }
        },
        "required": [
          "login",
          "password"
        ]
      }],
      "responses": [
        {
          "status": "401",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "429",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "201",
          "content-type": "application/json",
          "body": {
            "$schema": "http://json-schema.org/draft-04/schema#",
            "type": "object",
            "properties": {
              "confirmation": {
                "type": "object",
                "properties": {
                  "id": {
                    "type": "string"
                  },
                  "type": {
                    "type": "string"
                  },
                  "operation": {
                    "type": "string"
                  }
                },
                "required": [
                  "id",
                  "type",
                  "operation"
                ]
              },
              "captcha": {
                "type": "string"
              },
              "captcha_does_not_match": {
                "type": "boolean"
              }
            }
          }
        }
      ]
    }
  ]
  ```
</details> 

#### to_a
```ruby
tomogram.to_a
```

#### find_request
```ruby
request = tomogram.find_request(method: 'GET', path: '/status/1?qwe=rty')
```

#### find_request_with_content_type
```ruby
request = tomogram.find_request_with_content_type(method: 'GET', path: '/status/1?qwe=rty', content_type: 'application/json')
```

#### `find_responses`
```ruby
responses = request.find_responses(status: '200')
```

#### prefix_match?
This may be useful if you specify a prefix.

```ruby
tomogram.prefix_match?('http://local/api/v2/users')
```

#### to_resources
Maps resources for API Blueprint with possible requests.

Example output:

```ruby
{
  '/sessions' => ['POST /sessions']
}
```

### Command line tool

CLI allows you to convert files from API Blueprint (API Elements), Swagger and OpenAPI to JSON Schema.

Run CLI with `-h` to get detailed help:

```bash
tomograph -h
```

To specify the handler version use the `-d` flag:

#### OpenAPI 2.0
```bash
tomograph -d openapi2 openapi2.json tomogram.json
```

#### OpenAPI 3.0
```bash
tomograph -d openapi3 openapi3.yaml doc.json
```

#### API Blueprint
```bash
tomograph -d 4 apielemetns.yaml doc.json
```

#### exclude-description

Exclude "description" keys from json-schemas.

```bash
tomograph -d 4 apielemetns.yaml doc.json --exclude-description
```

#### split

Split output into files by method. Output in dir path.

```bash
tomograph -d 4 --split apielemetns.yaml jsons/
```

## License

The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).

[![Sponsored by FunBox](https://funbox.ru/badges/sponsored_by_funbox_centered.svg)](https://funbox.ru)
 readmeEtag: '"7c15795b959498fe0847c9ed366f5a7c4aa3b6e5"' readmeLastModified: Wed, 03 Apr 2024 19:44:07 GMT repositoryId: 780566490 description: >- Convert API Blueprint, Swagger and OpenAPI to JSON Schema and search through it created: '2024-04-01T18:37:02Z' updated: '2024-04-03T19:44:41Z' language: Ruby archived: true stars: 0 watchers: 0 forks: 0 owner: matchtechnologies logo: https://avatars.githubusercontent.com/u/147186495?v=4 license: MIT repoEtag: '"01bb06e91b20ea1e645c53afe1cd6dd8a4678ec4d1d2737800500ca60ae9ac52"' repoLastModified: Wed, 03 Apr 2024 19:44:41 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/accelbyte/extend-codegen-cli v3: true id: 9c1ac73738764bd74303bc666161d687 repositoryMetadata: base64Readme: >- IyBFeHRlbmQgQ29kZWdlbiBDTEkKCkEgdG9vbCBmb3IgZ2VuZXJhdGluZyBhbiBBY2NlbEJ5dGUgU0RLIG1vZHVsZSBvciBwbHVnaW4gZm9yIGN1c3RvbSBzZXJ2aWNlcywKc3VjaCBhcyBzZXJ2aWNlcyBjcmVhdGVkIHVzaW5nIEV4dGVuZCBTZXJ2aWNlIEV4dGVuc2lvbi4KCjpleGNsYW1hdGlvbjogKipUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgdGhlIGNvZGVnZW4gdGVtcGxhdGUgcGFjayB6aXAgCnJlbGVhc2VzIG9ubHkuKioKCiMjIE92ZXJ2aWV3CgpUaGUgRXh0ZW5kIENvZGVnZW4gY29uc2lzdHMgb2YgYSBDTEkgYXBwIGFuZCBzb21lIHRlbXBsYXRlIHBhY2tzLiBUaGUgQ0xJIGFwcCBpcwpyZWxlYXNlZCBhcyBjb250YWluZXIgaW1hZ2VzIGluIApbRG9ja2VyIEh1Yl0oaHR0cHM6Ly9odWIuZG9ja2VyLmNvbS9yL2FjY2VsYnl0ZS9leHRlbmQtY29kZWdlbi1jbGkpIHdoaWxlIAp0aGUgdGVtcGxhdGUgcGFjayB6aXAgZmlsZXMgYXJlIHJlbGVhc2VkIApbaGVyZV0oaHR0cHM6Ly9naXRodWIuY29tL0FjY2VsQnl0ZS9leHRlbmQtY29kZWdlbi1jbGkvcmVsZWFzZXMpIAppbiB0aGlzIHJlcG9zaXRvcnkuCgpBIHRlbXBsYXRlIHBhY2sgY29udGFpbnMgYSBgTWFrZWZpbGVgIGFuZCBgSmluamFgIHRlbXBsYXRlIGZpbGVzLiBXaGVuIHRoZQpgTWFrZWZpbGVgIGNvbW1hbmQgaXMgaW52b2tlZCwgdGhlIEV4dGVuZCBDb2RlZ2VuIENMSSBhcHAgaXMgZXhlY3V0ZWQgd2l0aCB0aGUKYEppbmphYCB0ZW1wbGF0ZSBmaWxlcyBhbmQgYSBnaXZlbiBjdXN0b20gc2VydmljZSBPcGVuQVBJIDIuMCBKU09OIGZpbGUgdG8gCmdlbmVyYXRlIGNvZGUuIFRoZSBgTWFrZWZpbGVgIGFsc28gZmV0Y2hlcyB0aGUgc3BlY2lmaWVkIENMSSBjb250YWluZXIgaW1hZ2UKdmVyc2lvbiBpZiBpdCBpcyBub3QgYXZhaWxhYmxlIGxvY2FsbHkgeWV0LgoKIyMgR2VuZXJhbCBVc2FnZQoKMS4gRG93bmxvYWQgdGhlIHRlbXBsYXRlIHBhY2sgemlwIGZvciB0aGUgY29ycmVzcG9uZGluZyBBY2NlbEJ5dGUgU0RLIG1vZHVsZSBvcgpwbHVnaW4gW2hlcmVdKGh0dHBzOi8vZ2l0aHViLmNvbS9BY2NlbEJ5dGUvZXh0ZW5kLWNvZGVnZW4tY2xpL3JlbGVhc2VzKSAKCjIuIFVuemlwIHRoZSBkb3dubG9hZGVkIHRlbXBsYXRlIHBhY2sgemlwIGZpbGUgYW5kIGZvbGxvdyB0aGUgaW5zdHJ1Y3Rpb24gaW4gCnRoZSBSRUFETUUubWQgaW5zaWRlLgoKICAg readmeEtag: '"54ff72c743cca2d217e202169a710e798ad14dc3"' readmeLastModified: Wed, 29 May 2024 03:38:44 GMT repositoryId: 684878324 description: AccelByte Extend Code Generator created: '2023-08-30T03:20:42Z' updated: '2024-05-29T03:38:50Z' language: null archived: false stars: 0 watchers: 3 forks: 0 owner: AccelByte logo: https://avatars.githubusercontent.com/u/25496952?v=4 repoEtag: '"4088789667291baa63eabebc077d197a3b3f30b7fc23f8a06c976d5e1a4df042"' repoLastModified: Wed, 29 May 2024 03:38:50 GMT category: - Code Generators - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/apicat/apicat v3: true id: c3406002d5ded68b23a583a8173c775e repositoryMetadata: base64Readme: >- PGRpdiBhbGlnbj0iY2VudGVyIj4KICAgIDxpbWcgYWx0PSJBcGlDYXQiIHdpZHRoPSIzNTBweCIgc3JjPSJodHRwczovL2Nkbi5hcGljYXQubmV0L3VwbG9hZHMvMmQwMmZmMmY2YjE5ZDNkNmQzZjEzNGMxODcyNDg0YWEucG5nIi8+CjwvZGl2PgoKPHAgYWxpZ249ImNlbnRlciI+CiAgRW5nbGlzaCB8CiAgPGEgaHJlZj0iLi9SRUFETUVfQ04ubWQiPueugOS9k+S4reaWhzwvYT4KPC9wPgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8YSBocmVmPSJodHRwczovL2FwaWNhdC5haSIgdGFyZ2V0PSJfYmxhbmsiPgogICAgICAgIDxpbWcgYWx0PSJTdGF0aWMgQmFkZ2UiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9haS1hcGljYXQ/bG9nbz1haSZsb2dvQ29sb3I9cmVkJmxhYmVsPWFwaWNhdCZsYWJlbENvbG9yPTQ4OTRGRiZjb2xvcj1FQUVDRjAiPgogICAgPC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9kaXNjb3JkLmdnLzZVRkJHaE51IiB0YXJnZXQ9Il9ibGFuayI+CiAgICAgICAgPGltZyBhbHQ9IlN0YXRpYyBCYWRnZSIgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NoYXQtRGlzY29yZC00RTVBRjA/bG9nbz1EaXNjb3JkIj4KICAgIDwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vZ2l0aHViLmNvbS9hcGljYXQvYXBpY2F0L2Jsb2IvbWFpbi9MSUNFTlNFIj4KICAgICAgICA8aW1nIGFsdD0iU3RhdGljIEJhZGdlIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGljZW5zZS1NSVQtZ3JlZW4iPgogICAgPC9hPgo8L3A+CgpBcGlDYXQgaXMgYW4gQVBJIGRvY3VtZW50YXRpb24gbWFuYWdlbWVudCB0b29sIHRoYXQgaXMgZnVsbHkgY29tcGF0aWJsZSB3aXRoIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24uIFdpdGggQXBpQ2F0LCB5b3UgY2FuIGZyZWVseSBhbmQgZWZmaWNpZW50bHkgbWFuYWdlIHlvdXIgQVBJcy4gSXQgaW50ZWdyYXRlcyB0aGUgY2FwYWJpbGl0aWVzIG9mIExMTSwgd2hpY2ggbm90IG9ubHkgaGVscHMgeW91IGF1dG9tYXRpY2FsbHkgZ2VuZXJhdGUgQVBJIGRvY3VtZW50YXRpb24gYW5kIGRhdGEgbW9kZWxzIGJ1dCBhbHNvIGNyZWF0ZXMgY29ycmVzcG9uZGluZyB0ZXN0IGNhc2VzIGJhc2VkIG9uIHRoZSBBUEkgY29udGVudC4gVXNpbmcgQXBpQ2F0LCB5b3UgY2FuIHF1aWNrbHkgYWNjb21wbGlzaCBhbnl0aGluZyBvdXRzaWRlIG9mIGNvZGluZywgYWxsb3dpbmcgeW91IHRvIGZvY3VzIHlvdXIgZW5lcmd5IG9uIHRoZSBjb2RlIGl0c2VsZi4KCiMjIFVzaW5nIG91ciBDbG91ZCBTZXJ2aWNlcwoKWW91IGNhbiB0cnkgb3V0IFtBcGlDYXRdKGh0dHBzOi8vYXBpY2F0LmFpKSBub3cuIEl0IHByb3ZpZGVzIGFsbCB0aGUgY2FwYWJpbGl0aWVzIG9mIHRoZSBzZWxmLWRlcGxveWVkIHZlcnNpb24uCgojIyBMb2NhbCBJbnN0YWxsYXRpb24KClRoZSBlYXNpZXN0IHdheSB0byBzdGFydCB0aGUgQXBpQ2F0IGlzIHRvIHJ1biBvdXIgZG9ja2VyLWNvbXBvc2UueWFtbCBmaWxlLiBCZWZvcmUgcnVubmluZyB0aGUgaW5zdGFsbGF0aW9uIGNvbW1hbmQsIG1ha2Ugc3VyZSB0aGF0IFtEb2NrZXJdKGh0dHBzOi8vZG9jcy5kb2NrZXIuY29tL2dldC1kb2NrZXIvKSBhbmQgW0RvY2tlciBDb21wb3NlXShodHRwczovL2RvY3MuZG9ja2VyLmNvbS9jb21wb3NlL2luc3RhbGwvKSBhcmUgaW5zdGFsbGVkIG9uIHlvdXIgbWFjaGluZToKCmBgYGJhc2gKZG9ja2VyIGNvbXBvc2UgdXAgLWQKYGBgCgpBZnRlciBydW5uaW5nLCB5b3UgY2FuIHZpc2l0IFtodHRwOi8vbG9jYWxob3N0OjgwMDBdKGh0dHA6Ly9sb2NhbGhvc3Q6ODAwMCkgb24geW91ciBicm93c2VyIHRvIHN0YXJ0IHVzaW5nIEFwaUNhdC4KCklmIHlvdSBuZWVkIHRvIGN1c3RvbWl6ZSB0aGUgY29uZmlndXJhdGlvbiwgcGxlYXNlIHJlZmVyIHRvIG91ciBbZG9ja2VyLWNvbXBvc2UueW1sXSguL2RvY2tlci1jb21wb3NlLnlhbWwpIGZpbGUgYW5kIG1hbnVhbGx5IHNldCB0aGUgZW52aXJvbm1lbnQgY29uZmlndXJhdGlvbi4gQWZ0ZXIgbWFraW5nIHRoZSBjaGFuZ2VzLCBwbGVhc2UgcnVuIGBkb2NrZXItY29tcG9zZSB1cCAtZGAgYWdhaW4uCgojIyBDb21tdW5pdHkKCklmIHlvdSBoYXZlIGFueXRoaW5nIHlvdSB3b3VsZCBsaWtlIHRvIGRpc2N1c3Mgd2l0aCB1cywgcGxlYXNlIGpvaW4gb3VyIGNvbW11bml0eS4KCi0gW0Rpc2NvcmRdKGh0dHBzOi8vZGlzY29yZC5nZy82VUZCR2hOdSkKCiMjIExpY2Vuc2UKCltNSVRdKGh0dHBzOi8vZ2l0aHViLmNvbS9hcGljYXQvYXBpY2F0L2Jsb2IvbWFpbi9MSUNFTlNFKQ== readmeEtag: '"b67461a0437fd05e7fb32d296c9409ded8ecb359"' readmeLastModified: Thu, 27 Jun 2024 08:43:10 GMT repositoryId: 488860709 description: >- An efficient API documentation management tool that fully adheres to the OpenAPI specification and incorporates advanced LLM technology. This tool can automate the generation of API documentation, data models, and test cases, greatly enhancing development efficiency and documentation quality. created: '2022-05-05T06:44:36Z' updated: '2026-01-21T12:58:06Z' language: Go archived: false stars: 290 watchers: 8 forks: 35 owner: apicat logo: https://avatars.githubusercontent.com/u/64158984?v=4 repoEtag: '"01d4a4b2a714b39a2afc9667e2ba91cbeff52996aa4c5302f25a27e17c02a46e"' repoLastModified: Wed, 21 Jan 2026 12:58:06 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/fastapi/full-stack-fastapi-template v3: true repositoryMetadata: base64Readme: >- # Full Stack FastAPI Template

<a href="https://github.com/fastapi/full-stack-fastapi-template/actions?query=workflow%3A%22Test+Docker+Compose%22" target="_blank"><img src="https://github.com/fastapi/full-stack-fastapi-template/workflows/Test%20Docker%20Compose/badge.svg" alt="Test Docker Compose"></a>
<a href="https://github.com/fastapi/full-stack-fastapi-template/actions?query=workflow%3A%22Test+Backend%22" target="_blank"><img src="https://github.com/fastapi/full-stack-fastapi-template/workflows/Test%20Backend/badge.svg" alt="Test Backend"></a>
<a href="https://coverage-badge.samuelcolvin.workers.dev/redirect/fastapi/full-stack-fastapi-template" target="_blank"><img src="https://coverage-badge.samuelcolvin.workers.dev/fastapi/full-stack-fastapi-template.svg" alt="Coverage"></a>

## Technology Stack and Features

- ⚡ [**FastAPI**](https://fastapi.tiangolo.com) for the Python backend API.
  - 🧰 [SQLModel](https://sqlmodel.tiangolo.com) for the Python SQL database interactions (ORM).
  - 🔍 [Pydantic](https://docs.pydantic.dev), used by FastAPI, for the data validation and settings management.
  - 💾 [PostgreSQL](https://www.postgresql.org) as the SQL database.
- 🚀 [React](https://react.dev) for the frontend.
  - 💃 Using TypeScript, hooks, [Vite](https://vitejs.dev), and other parts of a modern frontend stack.
  - 🎨 [Tailwind CSS](https://tailwindcss.com) and [shadcn/ui](https://ui.shadcn.com) for the frontend components.
  - 🤖 An automatically generated frontend client.
  - 🧪 [Playwright](https://playwright.dev) for End-to-End testing.
  - 🦇 Dark mode support.
- 🐋 [Docker Compose](https://www.docker.com) for development and production.
- 🔒 Secure password hashing by default.
- 🔑 JWT (JSON Web Token) authentication.
- 📫 Email based password recovery.
- 📬 [Mailcatcher](https://mailcatcher.me) for local email testing during development.
- ✅ Tests with [Pytest](https://pytest.org).
- 📞 [Traefik](https://traefik.io) as a reverse proxy / load balancer.
- 🚢 Deployment instructions using Docker Compose, including how to set up a frontend Traefik proxy to handle automatic HTTPS certificates.
- 🏭 CI (continuous integration) and CD (continuous deployment) based on GitHub Actions.

### Dashboard Login

[![API docs](img/login.png)](https://github.com/fastapi/full-stack-fastapi-template)

### Dashboard - Admin

[![API docs](img/dashboard.png)](https://github.com/fastapi/full-stack-fastapi-template)

### Dashboard - Items

[![API docs](img/dashboard-items.png)](https://github.com/fastapi/full-stack-fastapi-template)

### Dashboard - Dark Mode

[![API docs](img/dashboard-dark.png)](https://github.com/fastapi/full-stack-fastapi-template)

### Interactive API Documentation

[![API docs](img/docs.png)](https://github.com/fastapi/full-stack-fastapi-template)

## How To Use It

You can **just fork or clone** this repository and use it as is.

✨ It just works. ✨

### How to Use a Private Repository

If you want to have a private repository, GitHub won't allow you to simply fork it as it doesn't allow changing the visibility of forks.

But you can do the following:

- Create a new GitHub repo, for example `my-full-stack`.
- Clone this repository manually, set the name with the name of the project you want to use, for example `my-full-stack`:

```bash
git clone git@github.com:fastapi/full-stack-fastapi-template.git my-full-stack
```

- Enter into the new directory:

```bash
cd my-full-stack
```

- Set the new origin to your new repository, copy it from the GitHub interface, for example:

```bash
git remote set-url origin git@github.com:octocat/my-full-stack.git
```

- Add this repo as another "remote" to allow you to get updates later:

```bash
git remote add upstream git@github.com:fastapi/full-stack-fastapi-template.git
```

- Push the code to your new repository:

```bash
git push -u origin master
```

### Update From the Original Template

After cloning the repository, and after doing changes, you might want to get the latest changes from this original template.

- Make sure you added the original repository as a remote, you can check it with:

```bash
git remote -v

origin    git@github.com:octocat/my-full-stack.git (fetch)
origin    git@github.com:octocat/my-full-stack.git (push)
upstream    git@github.com:fastapi/full-stack-fastapi-template.git (fetch)
upstream    git@github.com:fastapi/full-stack-fastapi-template.git (push)
```

- Pull the latest changes without merging:

```bash
git pull --no-commit upstream master
```

This will download the latest changes from this template without committing them, that way you can check everything is right before committing.

- If there are conflicts, solve them in your editor.

- Once you are done, commit the changes:

```bash
git merge --continue
```

### Configure

You can then update configs in the `.env` files to customize your configurations.

Before deploying it, make sure you change at least the values for:

- `SECRET_KEY`
- `FIRST_SUPERUSER_PASSWORD`
- `POSTGRES_PASSWORD`

You can (and should) pass these as environment variables from secrets.

Read the [deployment.md](./deployment.md) docs for more details.

### Generate Secret Keys

Some environment variables in the `.env` file have a default value of `changethis`.

You have to change them with a secret key, to generate secret keys you can run the following command:

```bash
python -c "import secrets; print(secrets.token_urlsafe(32))"
```

Copy the content and use that as password / secret key. And run that again to generate another secure key.

## How To Use It - Alternative With Copier

This repository also supports generating a new project using [Copier](https://copier.readthedocs.io).

It will copy all the files, ask you configuration questions, and update the `.env` files with your answers.

### Install Copier

You can install Copier with:

```bash
pip install copier
```

Or better, if you have [`pipx`](https://pipx.pypa.io/), you can run it with:

```bash
pipx install copier
```

**Note**: If you have `pipx`, installing copier is optional, you could run it directly.

### Generate a Project With Copier

Decide a name for your new project's directory, you will use it below. For example, `my-awesome-project`.

Go to the directory that will be the parent of your project, and run the command with your project's name:

```bash
copier copy https://github.com/fastapi/full-stack-fastapi-template my-awesome-project --trust
```

If you have `pipx` and you didn't install `copier`, you can run it directly:

```bash
pipx run copier copy https://github.com/fastapi/full-stack-fastapi-template my-awesome-project --trust
```

**Note** the `--trust` option is necessary to be able to execute a [post-creation script](https://github.com/fastapi/full-stack-fastapi-template/blob/master/.copier/update_dotenv.py) that updates your `.env` files.

### Input Variables

Copier will ask you for some data, you might want to have at hand before generating the project.

But don't worry, you can just update any of that in the `.env` files afterwards.

The input variables, with their default values (some auto generated) are:

- `project_name`: (default: `"FastAPI Project"`) The name of the project, shown to API users (in .env).
- `stack_name`: (default: `"fastapi-project"`) The name of the stack used for Docker Compose labels and project name (no spaces, no periods) (in .env).
- `secret_key`: (default: `"changethis"`) The secret key for the project, used for security, stored in .env, you can generate one with the method above.
- `first_superuser`: (default: `"admin@example.com"`) The email of the first superuser (in .env).
- `first_superuser_password`: (default: `"changethis"`) The password of the first superuser (in .env).
- `smtp_host`: (default: "") The SMTP server host to send emails, you can set it later in .env.
- `smtp_user`: (default: "") The SMTP server user to send emails, you can set it later in .env.
- `smtp_password`: (default: "") The SMTP server password to send emails, you can set it later in .env.
- `emails_from_email`: (default: `"info@example.com"`) The email account to send emails from, you can set it later in .env.
- `postgres_password`: (default: `"changethis"`) The password for the PostgreSQL database, stored in .env, you can generate one with the method above.
- `sentry_dsn`: (default: "") The DSN for Sentry, if you are using it, you can set it later in .env.

## Backend Development

Backend docs: [backend/README.md](./backend/README.md).

## Frontend Development

Frontend docs: [frontend/README.md](./frontend/README.md).

## Deployment

Deployment docs: [deployment.md](./deployment.md).

## Development

General development docs: [development.md](./development.md).

This includes using Docker Compose, custom local domains, `.env` configurations, etc.

## Release Notes

Check the file [release-notes.md](./release-notes.md).

## License

The Full Stack FastAPI Template is licensed under the terms of the MIT license.
 readmeEtag: '"a9049b477932d592f01b8c2b88b9bc35aa7dfc9a"' readmeLastModified: Mon, 08 Dec 2025 19:13:01 GMT repositoryId: 172227885 description: >- Full stack, modern web application template. Using FastAPI, React, SQLModel, PostgreSQL, Docker, GitHub Actions, automatic HTTPS and more. created: '2019-02-23T15:08:34Z' updated: '2026-02-06T04:02:06Z' language: TypeScript archived: false stars: 41406 watchers: 289 forks: 8053 owner: fastapi logo: https://avatars.githubusercontent.com/u/156354296?v=4 license: MIT repoEtag: '"b89d7d6d311028da120ba78599f79bbf25a093d543d25fd6638f56e27bef2c2b"' repoLastModified: Fri, 06 Feb 2026 04:02:06 GMT foundInMaster: true category: Server Implementations id: 0354007b562b4efc3f08ce3da7f03afb oldLocations: - https://github.com/tiangolo/full-stack-fastapi-template - source: openapi3 tags repository: https://github.com/istandaarden/iwlz-indicatie v3: true repositoryMetadata: base64Readme: >- # iWlz-Indicatieregister 2
**iWlz-Indicatieregister bevat de [Graphql-schema](/gql-specificatie) koppelvlak specificatie en voorgeschreven [GraphQL-query templates](/gql-query/) voor het raadplegen van Wlz Indicatiegegevens in het indicatieregister en de [notificaties](/notificaties) verzonden vanuit het indicatieregister.**

Het indicatieregister is in beheer bij het CIZ en is onderdeel van het iWlz-netwerkmodel.

## Versies en Status 

Er zijn altijd minimaal twee versies actueel. Een versie die in productie is, status is *Lopend* en een versie die in ontwikkeling is, status is *In ontwikkeling*.

| | LOPEND (*default branch*) | IN ONTWIKKELING | ARCHIEF |
| --: |:-- |:-- | :-- |
| ***Informatiemodel***| [**Indicatieregister 2**](https://informatiemodel.istandaarden.nl/informatiemodel/iwlz/netwerk/indicatieregister-2/) (*huidige branch*) | -- | [Indicatieregister 1](https://informatiemodel.istandaarden.nl/iWlz-Indicatie-1/) |
| ***Koppelvlak specificatie*** | [Documentatie](https://github.com/iStandaarden/iWlz-indicatie/tree/Indicatieregister-2) / [Release v1.6.5] | -- |  [Documentatie](https://github.com/iStandaarden/iWlz-indicatie/tree/Indicatieregister-1) / [Release v1.2](https://github.com/iStandaarden/iWlz-indicatie/releases/tag/v1.2) |
| ***Datum*** | 27-11-2025 - [Changelog](/CHANGELOG.md)| -- | juli 2023 |

### Changelog
Volledige Changelog [Hier](/CHANGELOG.md)

## Inhoudsopgave
- [iWlz-Indicatieregister 2](#iwlz-indicatieregister-2)
  - [Versies en Status](#versies-en-status)
    - [Changelog](#changelog)
  - [Inhoudsopgave](#inhoudsopgave)
  - [Onderdelen](#onderdelen)
    - [Graphql-schema](#graphql-schema)
    - [Graphql-query](#graphql-query)
    - [Open Agent Policy](#open-agent-policy)
    - [Notificaties](#notificaties)
  - [**Raadplegen Indicatieregister**](#raadplegen-indicatieregister)
    - [Autorisatieregels en autorisatiematrix](#autorisatieregels-en-autorisatiematrix)
  - [Aanvullende Documentatie](#aanvullende-documentatie)
    - [Informatiemodel](#informatiemodel)
    - [GraphQL](#graphql)
    - [Open Agent Policy](#open-agent-policy-1)
  - [Meer informatie](#meer-informatie)
  - [Contactpersonen:](#contactpersonen)

## Onderdelen
De koppelvlak specificatie van het Indicatieregister maakt onderdeel uit van de **iStandaard iWlz**. De specificaties van de andere onderdelen, zoals ERD, regels, procesbeschrijving, autorisatieregels, notificatie-typen staan in het [Informatiemodel iWlz](https://informatiemodel.istandaarden.nl/) dat te vinden is via de website: [https://informatiemodel.istandaarden.nl/](https://informatiemodel.istandaarden.nl/)

![onderdelen](/src/Onderdelen_Netwerk.png)
v.l.n.r. Raadpleger doet via GraphQL-query een raadpleging. Open Policy agent controleert of query voldoet aan autorisatie-regels van dat register. GraphQL-schema definieert het data-schema van het register.

### Graphql-schema 
De [Graphql-schema specificatie](/gql-specificatie/) is bedoelt voor implementatie door de bronhouder en beschrijft hoe de data aan elkaar is gerelateerd. 

> [!NOTE]
> De graphql-specificatie is te vinden in de folder [**/gql-specificatie**](/gql-specificatie/). 

### Graphql-query
De [Graphql-queries](/gql-query/) beschrijven het template hoe een raadpleger vanuit zijn rol informatie kan raadplegen. Deze template volgt altijd het GraphQL-schema maar moet op bepaalde momenten aan vaste patronen voldoen vanwege de geldende autorisatie. Gaat een raadpleger buiten dit patroon dan zal de vraag worden afgekeurd en krijgt de raadpleger geen inzicht in de data. 

> [!NOTE]
> Het overzicht van de beschikbare templates inclusief een toelichting voor welke partij de template is en autorisatieflow is te vinden in de folder [**/gql-query**](/gql-query/) staat .

### Open Agent Policy
De Open Agent Policy controleert of een query voldoet aan de daarvoor afgesproken template. De policy is gebaseerd op de autorisatieregels van dat register. 

De policy is beschikbaar in: @@@ nog te bepalen.

> [!NOTE]
> De functionele beschrijving van de toegangscontrole per raadpleging is beschikbaar in de folder **[/raadplegen](/raadplegen/)**

### Notificaties
Met een notificatie wordt een netwerk-deelnemer op de hoogte gebracht door een bronhouder dat er nieuwe (of gewijzigde) informatie is die directe of afgeleide betrekking heeft op die deelnemer. De notificatie bevat informatie die de deelnemer in staat stelt de relevante informatie te raadplegen bij de bron. Een notificatie loopt altijd van bron naar deelnemer.

> [!NOTE]
> De notificaties vanuit het Indicatieregister zijn te vinden in de folder [**/notificaties**](/notificaties/)

## **Raadplegen Indicatieregister**

Het raadplegen van het Indicatieregister is gebonden aan voorwaarden. De raadpleger moet bevoegd zijn én het vastgestelde raadpleegpatroon volgen. Dit patroon is essentieel voor het valideren van de toestemming. 

Als dat patroon niet wordt gevolgd — bijvoorbeeld door ontbrekende autorisatie, onjuiste of incomplete input, of het opvragen van ongeoorloofde gegevens — wordt de toegang geweigerd of het resultaat beperkt.

Use-cases beschrijven hoe een deelnemer het register correct raadpleegt.

### Autorisatieregels en autorisatiematrix
De toegang tot gegevens is vastgelegd doormiddel van **Autorisatieregels** en de **Autorisatiematrix**. De [autorisatieregels](https://informatiemodel.istandaarden.nl/informatiemodel/iwlz/netwerk/indicatieregister-2/regels/autorisatieregel/) zijn te vinden in het Informatiemodel Indicatieregister 2 (via [hier](https://informatiemodel.istandaarden.nl/informatiemodel/iwlz/netwerk/indicatieregister-2/regels/autorisatieregel/)) en de [autorisatiematrix](/raadplegen/autorisatiematrix_indicatieregister.md) is [hier](/raadplegen/autorisatiematrix_indicatieregister.md) te vinden.


> [!NOTE]
> De functionele beschrijving van per raadpleging per deelnemer is beschikbaar in de folder **[/raadplegen](/raadplegen/)**

Meer informatie over de structuur van het raadplegen en het valideren ervan is te lezen in het [Afsprakenstelsel iWlz - Raadplegen](https://wlz.atlassian.net/wiki/x/KgpgAQ)


## Aanvullende Documentatie

### Informatiemodel

![informatiemodel](/src/Informatiemodel-sml.png)

Ondersteunende documentatie is te vinden in het Informatiemodel, via de website [https://informatiemodel.istandaarden.nl/](https://informatiemodel.istandaarden.nl/) en daar de gewenste versie te selecteren (zie ook in de tabel hierboven voor een directe verwijzing).

### GraphQL
![GraphQL](/src/GraphQL-logo-sml.png) 

zie [GraphQL.org](https://graphql.org) 

### Open Agent Policy
![OPA](/src/OPA-logo-sml.png) 

zie [Open Agent Policy](https://www.openpolicyagent.org) en [documentatie](https://www.openpolicyagent.org/docs/latest/)

## Meer informatie
* Actieprogramma iWlz: van keten naar netwerk: [het Actieprogramma iWlz](https://www.istandaarden.nl/iwlz/actieprogramma/index "Over Actieprogramma iWlz")
* Informatiemodel iStandaarden iWlz: [Informatiemodellen](https://informatiemodel.istandaarden.nl)
* Portaal voor iStandaarden in de
Zorg en Ondersteuning: [homepagina iStandaarden](https://www.istandaarden.nl)

## Contactpersonen:
* Dennis de Gouw - [@dennisdegouw](http://github.com/dennisdegouw)
* Remo van Rest - [@rvanrest](https://github.com/rvanrest)


 readmeEtag: '"c803f85eba06e93569344d5fc20b8da24a99cdb8"' readmeLastModified: Tue, 03 Feb 2026 13:45:54 GMT repositoryId: 204430425 description: Koppelvlak specificatie Indicatieregister created: '2019-08-26T08:25:11Z' updated: '2026-02-03T13:45:59Z' language: null archived: false stars: 2 watchers: 5 forks: 4 owner: iStandaarden logo: https://avatars.githubusercontent.com/u/54351663?v=4 repoEtag: '"71e97540c9730e73af76006f449850f72212ce58db36029a83e0f3ecf3975d73"' repoLastModified: Tue, 03 Feb 2026 13:45:59 GMT foundInMaster: true category: Documentation id: 5a079f438418ad2614981480bbbba221 - source: openapi3 tags repository: https://github.com/javathought/vx-api v3: true repositoryMetadata: base64Readme: >- IyB2eC1hcGkKVGVzdGluZyB2ZXJ0eC13ZWItYXBpLWNvbnRyYWN0CgpbIVtDb2RhY3kgQmFkZ2VdKGh0dHBzOi8vYXBpLmNvZGFjeS5jb20vcHJvamVjdC9iYWRnZS9HcmFkZS84MGRlOGI2OTRkY2E0NDAxYmRmZDhjNDFmODU4YmNhNyldKGh0dHBzOi8vd3d3LmNvZGFjeS5jb20vYXBwL2phdmF0aG91Z2h0L3Z4LWFwaT91dG1fc291cmNlPWdpdGh1Yi5jb20mYW1wO3V0bV9tZWRpdW09cmVmZXJyYWwmYW1wO3V0bV9jb250ZW50PWphdmF0aG91Z2h0L3Z4LWFwaSZhbXA7dXRtX2NhbXBhaWduPUJhZGdlX0dyYWRlKQpbIVtDb2RhY3kgQmFkZ2VdKGh0dHBzOi8vYXBpLmNvZGFjeS5jb20vcHJvamVjdC9iYWRnZS9Db3ZlcmFnZS84MGRlOGI2OTRkY2E0NDAxYmRmZDhjNDFmODU4YmNhNyldKGh0dHBzOi8vd3d3LmNvZGFjeS5jb20vYXBwL2phdmF0aG91Z2h0L3Z4LWFwaT91dG1fc291cmNlPWdpdGh1Yi5jb20mdXRtX21lZGl1bT1yZWZlcnJhbCZ1dG1fY29udGVudD1qYXZhdGhvdWdodC92eC1hcGkmdXRtX2NhbXBhaWduPUJhZGdlX0NvdmVyYWdlKQpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9qYXZhdGhvdWdodC92eC1hcGkuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5vcmcvamF2YXRob3VnaHQvdngtYXBpKQo= readmeEtag: '"150f1b56f142d0d9f0b8c8a82898e8ff9c0c60c7"' readmeLastModified: Mon, 20 Nov 2017 01:05:12 GMT repositoryId: 110819342 description: Testing vertx-web-api-contract created: '2017-11-15T10:29:10Z' updated: '2017-11-17T23:00:22Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: javathought logo: https://avatars.githubusercontent.com/u/1075135?v=4 license: Apache-2.0 repoEtag: '"f36696c8638077bbdf451820ba1352a8c8af524dafcc93fc76bfe83a09bcb053"' repoLastModified: Fri, 17 Nov 2017 23:00:22 GMT foundInMaster: true category: Data Validators id: b1645dfb1035ae7d0d778e0cc5534f4b - source: openapi3 tags repository: https://github.com/marcusvc/backend-rest-swagger-buffer-log v3: true repositoryMetadata: base64Readme: >- IyBiYWNrZW5kLXJlc3Qtc3dhZ2dlci1idWZmZXItbG9nClRoaXMgaXMgYSBzaW1wbGUgSmF2YSBSRVNUIEFQSSBleGFtcGxlIGludGVuZGVkIHRvIHJ1biB3aXRoIFdpbGRmbHkuIEl0IGhhcyBPcGVuYXBpIDMuMCBkb2N1bWVudGF0aW9uIChTd2FnZ2VyKSBhbmQgYSBIdHRwU2VydmxldChSZXF1ZXN0L1Jlc3BvbnNlKSBidWZmZXIgaW1wbGVtZW50YXRpb24uCgojIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gKGZvcm1lcmx5IFN3YWdnZXIgc3BlY2lmaWNhdGlvbikKUHJvdmlkZXMgYSBPcGVuQVBJIDMuMCBzcGVjaWZpY2F0aW9uIHdyaXR0ZW4gaW4gWUFNTCBvciBKU09OIGNhbGxpbmcgdGhlIGZvbGxvd2luZyBVUkwsIHJlc3BlY3RpdmlseToKPiBodHRwW3NdOi8vaG9zdC9jb250ZXh0L2Jhc2VfYXBwbGljYXRpb25fcGF0aF9mb3JfYWxsX3Jlc3RfcmVzb3VyY2VzL29wZW5hcGkueWFtbAo+IGh0dHBbc106Ly9ob3N0L2NvbnRleHQvYmFzZV9hcHBsaWNhdGlvbl9wYXRoX2Zvcl9hbGxfcmVzdF9yZXNvdXJjZXMvb3BlbmFwaS5qc29uCgojIFN3YWdnZXIgVUkKU2luZ2xlIHBhZ2UgYXBwbGljYXRpb24gdGhhdCBleHBvc2VzIEFQSSBkb3VtZW50YXRpb24gYW5kIHByb3ZpZGVzIGEgdGVzdCBzdWl0ZSBiYXNlZCBvbiBvcGVuYXBpIHNwZWNpZmljYXRpb24uCgojIEh0dHBTZXJ2bGV0KFJlcXVlc3QvUmVzcG9uc2UpIGJ1ZmZlcgpUaGUgbmVlZCBvZiBhIEh0dHBTZXJ2bGV0KFJlcXVlc3QvUmVzcG9uc2UpIGlzIHRvIGxvZyB0aGUgaW5wdXRzdHJlYW0gYW5kIG91dHB1dHN0cmVhbSwgYWZ0ZXIgb3IgYmVmb3JlIHBhcnNpbmcgdGltZSwgcmVzcGVjdGl2bHkuIFRoZSBpbnB1dHN0cmVhbSBvZiBhIEh0dHBTZXJ2bGV0UmVxdWVzdCBhbmQgdGhlIG91dHB1dHN0cmVhbSBvZiBhIEh0dHBTZXJ2bGV0UmVzcG9uc2UgY2FuIG9ubHkgYmUgY29uc3VtZWQgb25jZS4gSWYgd2UgdHJ5IHRvIGNvbnN1bSBpbnB1dHN0cmVhbSBhZnRlciBwYXJzZXIsIGl0J3MgZ29ubmEgYmUgZW1wdHkuIElmIHdlIHRyeSB0byBjb25zdW0gb3V0cHV0c3RyZWFtIGJlZm9yZSBwYXJzZXIsIHRoZXJlIHdpbGwgYmUgbm8gZGF0YSB0byBwYXJzZS4gVXNpbmcgYSBIdHRwU2VydmxldChSZXF1ZXN0L1Jlc3BvbnNlKSBidWZmZXIgaW1wbGVtZW50YXRpb24sIGl0J3MgcG9zc2libGUgdG8gbG9nIHJlcXVlc3QncyBpbnB1dHN0cmVhbSAgYW5kIHJlc3BvbnNlJ3Mgb3V0cHV0c3RyZWFtIGluIHRoZSBtaWRkbGUgb2YgdGhlIHByb2Nlc3MuCg== readmeEtag: '"de3569af30c131d611a4d6cbb0cd6067d3dbc999"' readmeLastModified: Mon, 21 Jan 2019 16:44:10 GMT repositoryId: 166137544 description: >- Simple Java REST API example using OpenAPI 3.0, Swagger-UI, request and response buffer created: '2019-01-17T01:19:06Z' updated: '2019-01-21T16:45:56Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: marcusvc logo: https://avatars.githubusercontent.com/u/20482843?v=4 license: Apache-2.0 repoEtag: '"5ed0faa923ab025c5e54c1599652d4de6d2a06edf4d4204097f6ccd9642d3acc"' repoLastModified: Mon, 21 Jan 2019 16:45:56 GMT foundInMaster: true category: Parsers id: 0f686a53ed0f1513e85de5295391ef6c - source: openapi3 tags repository: https://github.com/dperez3/oapi-generator v3: true repositoryMetadata: base64Readme: >- IyBvYXBpLWdlbmVyYXRvcgoKWyFbQ29udmVudGlvbmFsIENvbW1pdHNdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvQ29udmVudGlvbmFsJTIwQ29tbWl0cy0xLjAuMC15ZWxsb3cuc3ZnKV0oaHR0cHM6Ly9jb252ZW50aW9uYWxjb21taXRzLm9yZykKWyFbYnVkZHkgcGlwZWxpbmVdKGh0dHBzOi8vYXBwLmJ1ZGR5LndvcmtzL2RwZXJlejNpaWkvb2FwaS1nZW5lcmF0b3IvcGlwZWxpbmVzL3BpcGVsaW5lLzE5NDM5MC9iYWRnZS5zdmc/dG9rZW49M2M3ZDQ4NGJiZTAwMDNhYmExOTE4ODQ0MjQzZjI3NTJhOTA0ZTA2Y2I1MmRjMTJiYmVkZWU5Y2VlYWU3ODY0NiAiYnVkZHkgcGlwZWxpbmUiKV0oaHR0cHM6Ly9hcHAuYnVkZHkud29ya3MvZHBlcmV6M2lpaS9vYXBpLWdlbmVyYXRvci9waXBlbGluZXMvcGlwZWxpbmUvMTk0MzkwKQoKR2VuZXJhdGUgT3BlbkFQSSBkb2N1bWVudHMgZnJvbSBtdWx0aXBsZSBTd2FnZ2VyIHYyIG9yIE9wZW4gQVBJIHYzIGRvY3VtZW50cy4KCltucG1dKGh0dHBzOi8vd3d3Lm5wbWpzLmNvbS9wYWNrYWdlL29hcGktZ2VuZXJhdG9yKQoKIyMgSW5zdGFsbGF0aW9uCgojIyMgSW5zdGFsbAoKYGBgYmFzaApucG0gaW5zdGFsbCAtLXNhdmUtZGV2IG9hcGktZ2VuZXJhdG9yCm5wbSBpbnN0YWxsIC0tc2F2ZS1kZXYgb3BlbmFwaS10eXBlc0AxLjMuNApgYGAKCiMjIyBDb25maWd1cmUKCiMjIyMgQ29uZmlnIEZpbGUKCkphdmFzY3JpcHQKCmBgYGpzCmNvbnN0IHYxR2VuRG9jID0geyAvKi4uLiovIH07Cgpjb25zdCB2MkdlbkRvYyA9IHsgLyouLi4qLyB9OwoKY29uc3QgY29uZmlnID0KewogIGdlbkNvbmZpZ3M6IFt2MUdlbkRvYywgdjJHZW5Eb2NdCn07CmBgYAoKT3IgVHlwZVNjcmlwdAoKYGBgdHMKaW1wb3J0IHsgQ29uZmlndXJhdGlvbiB9IGZyb20gIm9hcGktZ2VuZXJhdG9yL2NvbmZpZ3VyYXRpb24iOwppbXBvcnQgeyBPcGVuQVBJVjMgfSBmcm9tICJvcGVuYXBpLXR5cGVzIjsKCmNvbnN0IHYxR2VuRG9jOiBDb25maWd1cmF0aW9uLklHZW5PcGVuQVBJVjNDb25maWcgPSB7IC8qLi4uKi8gfTsKCmNvbnN0IHYyR2VuRG9jOiBDb25maWd1cmF0aW9uLklHZW5PcGVuQVBJVjNDb25maWcgPSB7IC8qLi4uKi8gfTsKCmNvbnN0IGNvbmZpZzogQ29uZmlndXJhdGlvbi5JTWFueUdlbk9wZW5BUElWM0NvbmZpZ3MgPQp7CiAgZ2VuQ29uZmlnczogW3YxR2VuRG9jLCB2MkdlbkRvY10KfTsKCmV4cG9ydCA9IGNvbmZpZzsKYGBgCgpgb3BlbmFwaS10eXBlc2AgY29tYmluZWQgd2l0aCBUeXBlc1NjcmlwdCBpbnRlbGxpc2Vuc2UgcHJvdmlkZXMgdGhlIGVhc2llc3Qgd2F5IHRvIHVuZGVyc3RhbmQgdGhlIGNvbmZpZ3VyYXRpb24gQVBJLgoKIyMjIyBgcGFja2FnZS5qc29uYAoKYGBganNvbgp7CiAgInNjcmlwdHMiOiB7CiAgICAib2FwaS1nZW46cnVuIjogIm9hcGktZ2VuZXJhdG9yIC0tY29uZmlnIHtwYXRoX3RvX2Fib3ZlX2NvbmZpZ19maWxlfSIKICB9Cn0KYGBgCg== readmeEtag: '"7dae70b56b4c6e72e8570ed8f5f8d23e8d59c9bb"' readmeLastModified: Mon, 20 Apr 2020 16:34:24 GMT repositoryId: 193387986 description: >- Generate OpenAPI documents from multiple Swagger v2 or Open API v3 documents. created: '2019-06-23T19:34:08Z' updated: '2020-04-20T16:34:28Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: dperez3 logo: https://avatars.githubusercontent.com/u/7634621?v=4 license: GPL-3.0 repoEtag: '"77d3c232d117b57554ea961a2df9176c4f458a46ef443366381a3ce0ea586207"' repoLastModified: Mon, 20 Apr 2020 16:34:28 GMT foundInMaster: true category: - SDK - Parsers id: 4e6af3de5a8e564669f91429104028f1 - source: openapi3 tags repository: https://github.com/zaki-yama/kintone-openapi-generator v3: true repositoryMetadata: base64Readme: >- a2ludG9uZSBPcGVuQVBJIFNwZWMgR2VuZXJhdG9yCj09PT09PT09PT09PT09PT09PT09PT09PT09PT09PQoKR2VuZXJhdGUgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIGRvY3VtZW50IGZyb20ga2ludG9uZSdzIFJFU1QgQVBJIFNjaGVtYXMuICAKSXQgY2FuIGFsc28gbGF1bmNoIFJFU1QgQVBJIGRvY3VtZW50IHNlcnZlciBhbmQgbW9jayBzZXJ2ZXIuCgojIyBJbnN0YWxsYXRpb24KCmBgYAokIG5wbSBpbnN0YWxsCmBgYAoKIyMgVXNhZ2UKCmBgYHpzaAojIEZldGNoIGtpbnRvbmUncyBSRVNUIEFQSSBTY2hlbWFzIGFuZCBnZW5lcmF0ZSBPcGVuQVBJIFNwZWMgZG9jdW1lbnQuCiMgSXQgZ2VuZXJhdGVzIGBzcmMvZ2VuZXJhdGVkL2tpbnRvbmUtYXBpLXNjaGVtYXMuanNvbmAgYW5kIGBzcmMvZ2VuZXJhdGVkL29wZW5hcGkueWFtbGAuCiQgbnBtIHJ1biBnZW5lcmF0ZQoKIyBSdW4gUkVTVCBBUEkgZG9jdW1lbnQgc2VydmVyIChodHRwOi8vbG9jYWxob3N0OjMwMDApCiQgbnBtIHJ1biBkb2MKCiMgUnVuIG1vY2sgc2VydmVyIChodHRwOi8vbG9jYWxob3N0OjQwMTApCiQgbnBtIHJ1biBtb2NrCmBgYAo= readmeEtag: '"89d6f317dcde6f5280c576c08ef8c826d6ad5753"' readmeLastModified: Sun, 18 Sep 2022 13:52:40 GMT repositoryId: 220504408 description: Generate OpenAPI Specification Document from kintone's REST API Schema created: '2019-11-08T16:17:27Z' updated: '2023-01-27T21:08:18Z' language: TypeScript archived: true stars: 0 watchers: 0 forks: 0 owner: zaki-yama logo: https://avatars.githubusercontent.com/u/1001444?v=4 repoEtag: '"4552fd6c1e290be0080b2c0d185905fc637735f695f3d24459d692dcdf3eb453"' repoLastModified: Fri, 27 Jan 2023 21:08:18 GMT foundInMaster: true category: - Testing - Parsers id: f25cbbc6b37854cdf79c6df73f6ee104 - source: openapi3 tags repository: https://github.com/funkeyfreak/msgraph-openapi v3: true repositoryMetadata: repositoryId: 310755500 description: A buildable OpenAPI v3 description for Microsoft Graph created: '2020-11-07T03:01:48Z' updated: '2020-12-30T06:32:05Z' language: null archived: false stars: 0 watchers: 1 forks: 1 owner: funkeyfreak logo: https://avatars.githubusercontent.com/u/5327214?v=4 license: Apache-2.0 repoEtag: '"f7d6e2e961ed6f647f0c3403c9cf7394fc29ee1739ea9ac130573d9c9df45bf3"' repoLastModified: Wed, 30 Dec 2020 06:32:05 GMT foundInMaster: true id: da9f2a7deee2c34b5bc25c19350a25a5 - source: openapi3 tags repository: https://github.com/abhishekastoorkar/nodejs-training v3: true repositoryMetadata: repositoryId: 303295453 description: >- This is a training manager app build using nodejs, express and sequelize ORM. for authentication of user AWS cognito service is used. coverage report and openApi 3 documention are also available created: '2020-10-12T06:16:22Z' updated: '2020-12-02T05:47:34Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: abhishekastoorkar logo: https://avatars.githubusercontent.com/u/51363257?v=4 repoEtag: '"52ea2542459c03bd8de001385a999859d8d7db3b37d0575700d138800ca7d9f7"' repoLastModified: Wed, 02 Dec 2020 05:47:34 GMT foundInMaster: true id: 7feb18067ecfc3ef6d4cb1d45cb22d99 - source: openapi3 tags repository: https://github.com/teobler/openapi-swagger-demo v3: true repositoryMetadata: base64Readme: >- IyBXaGF0IGZvcgoKdGhpcyBpcyBhIGRlbW8gYWJvdXQgT3BlbkFQSSAzLjAgdG8gcHJvdmlkZSBhIHNjaGVtYSB0byBnZW5lcmF0ZSBmcm9udC1lbmQgcmVxdWVzdCBjb2RlIGJ5IHVzaW5nIFtyZWR1eC1hY3Rpb24tZ2VuZXJhdG9yXShodHRwczovL2dpdGh1Yi5jb20vdGVvYmxlci9yZWR1eC1hY3Rpb24tZ2VuZXJhdG9yL3RyZWUvb3BlbmFwaS0zLjApCgojIyBTdGFydAoKMS4gY2xvbmUgY29kZSB0byB5b3VyIGxvY2FsCjIuIHJ1biBgLi9ncmFkbGV3IGJvb3QgcnVuYAozLiBhY2Nlc3MgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS1kb2NzIGluIHlvdXIgYnJvd3NlciB5b3UgY2FuIGdldCBPcGVuQVBJIDMuMCBzY2hlbWEKNC4gYWNjZXNzIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpL2luZGV4Lmh0bWw/Y29uZmlnVXJsPS9hcGktZG9jcy9zd2FnZ2VyLWNvbmZpZyBpbiB5b3VyIGJyb3dzZXIgeW91IGNhbiBzZWUgc3dhZ2dlciB2Mwo= readmeEtag: '"c5567ee2eb418c5809364f999c7248df727a9ec3"' readmeLastModified: Thu, 26 Mar 2020 08:13:08 GMT repositoryId: 250197701 description: >- demo about OpenAPI 3.0 to provide a schema to generate front-end request code created: '2020-03-26T08:10:01Z' updated: '2020-03-26T08:14:14Z' language: Java archived: false stars: 0 watchers: 0 forks: 0 owner: teobler logo: https://avatars.githubusercontent.com/u/26424691?v=4 repoEtag: '"4d27d5b4c619fcd001d4a9753c2b3afa01664fd05ae414579a6a0267fdf81c07"' repoLastModified: Thu, 26 Mar 2020 08:14:14 GMT foundInMaster: true category: - Code Generators - Server Implementations id: a241cb9ae9cda5f9ce71402d3b48ddc1 - source: openapi3 tags repository: https://github.com/ankitech/spring-swagger-setup v3: true repositoryMetadata: base64Readme: >- IyBzcHJpbmctc3dhZ2dlci1zZXR1cApUaGlzIGlzIGFuIHJlcG9zaXRvcnkgdGhhdCBzaG93cyBob3cgdG8gc2V0IHVwIHN3YWdnZXIgZG9jdW1lbnRhdGlvbiBpbiBhIHNwcmluZyBwcm9qZWN0Cg== readmeEtag: '"113097c395b1167de8ea097d13fba499caaf576f"' readmeLastModified: Tue, 21 Apr 2020 05:59:58 GMT repositoryId: 257364853 description: >- This is an repository that shows how to set up swagger documentation in a spring project created: '2020-04-20T18:07:03Z' updated: '2020-04-21T19:03:50Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: ankitech logo: https://avatars.githubusercontent.com/u/9432041?v=4 license: Apache-2.0 repoEtag: '"fa023972360295bca777b5ee9e3266f091321cdb67044c063f9b2a07ebc1f68e"' repoLastModified: Tue, 21 Apr 2020 19:03:50 GMT foundInMaster: true category: - Server - Server Implementations id: 8ff20053e9e7616753043f1b4b10a98c - source: openapi3 tags repository: https://github.com/thiagojacinto/bootcamp-backend-spring v3: true repositoryMetadata: base64Readme: >- IyBbQm9vdGNhbXBdKGh0dHBzOi8vZ2l0aHViLmNvbS90aGlhZ29qYWNpbnRvL3Bvcy11bml0LXBvcnRvL3RyZWUvbWFzdGVyL0Jvb3RjYW1wMik6IFByb2pldG8gZS1jb21tZXJjZSAKIEJvb3RjYW1wIEJhY2tlbmQsIHByb2pldG8gZGUgdW0gZS1jb21tZXJjZSB1dGlsaXphbmRvIFNwcmluZyBCb290LgoKICMjIFVzYW5kbwoKIE8gcHJpbWVpcm8gcGFzc28gw6kgY29uZmlndXJhciBhIGNvbmV4w6NvIGNvbSB1bSBiYW5jbyBQb3N0ZWdyZVNRTCwgY29uZmlndXJhbmRvIG8gX2FwcGxpY2F0aW9uLnByb3BlcnRpZXNfIGNvbW8gW2VzdGUgZXhlbXBsb10oaHR0cHM6Ly9kb2NzLm1pY3Jvc29mdC5jb20vcHQtYnIvYXp1cmUvZGV2ZWxvcGVyL2phdmEvc3ByaW5nLWZyYW1ld29yay9jb25maWd1cmUtc3ByaW5nLWRhdGEtanBhLXdpdGgtYXp1cmUtbXlzcWwjY29uZmlndXJlLXNwcmluZy1ib290LXRvLXVzZS1henVyZS1kYXRhYmFzZS1mb3ItbXlzcWwpLgoKIEEgcGFydGlyIGRpc3NvLCDDqSBwb3Nzw612ZWwgcm9kYXIgbyBwcm9qZXRvIHBlbG8gTWF2ZW4gKGNvbW8gaW5zdGFsYXI/IFtBcXVpXShodHRwczovL21hdmVuLmFwYWNoZS5vcmcvaW5zdGFsbC5odG1sKSksIGEgcGFydGlyIGRvIGNvbWFuZG8gYG12bncgc3ByaW5nLWJvb3Q6cnVuYCByZWFsaXphbmRvIG5vIGRpcmV0w7NyaW8uCgogT3V0cmEgb3DDp8OjbyDDqSByZWFsaXphciBvIGJ1aWxkIGNvbSBvIGNvbWFuZG8gYG12bncgY2xlYW4gcGFja2FnZWAgZSBlbnTDo28gZXhlY3V0YW5kbyBvIGFycXVpdm8gSkFSIHJlc3VsdGFudGUsIGVzcGVjaWZpY2FuZG8gYSB2ZXJzw6NvIGRvIHByb2pldG8gb2J0aWRhIG5vIHBvbS54bWwsIGUgYSBVUkwgZGUgY29uZXjDo28gY29tIHVtIFNHQkQgKipQb3N0Z3JlU1FMKiouCgogYGBgCiBqYXZhIC1qYXIgdGFyZ2V0L2Vjb21tZXJjZS17VkVSU0FPfS5qYXIgLS1zcHJpbmcuZGF0YXNvdXJjZS51cmw9JHtKREJDX0RBVEFCQVNFX1VSTH0KIGBgYAoKICMjIFRlc3RhbmRvCgogVW1hIHZleiByb2RhbmRvIG8gc2VydmnDp28sIMOpIHBvc3PDrXZlbCByZWFsaXphciBjaGFtYWRhcyBIVFRQLCBjb21vIG8gZXhlbXBsbyBhYmFpeG86CgpgYGAKIGN1cmwgLVggR0VUICJodHRwOi8vbG9jYWxob3N0OjgwODAvdjEvcHJvZHV0b3MvbWFyY2EvMT9pdGVucz01JnBhZ2luYT0wIiAtSCAiYWNjZXB0OiAqLyoiIHwganNvbl9wcApgYGAKY29tIG8gcXVhbCBzZSBlc3BlcmEgcmVjZWJlciBjb21vIHJlc3Bvc3RhIHVtIEpTT04gY29uZm9ybWUgZXhlbXBsbyBzZWd1aW50ZToKYGBgCiB7CiAgICJzb3J0IiA6IHsKICAgICAgInNvcnRlZCIgOiBmYWxzZSwKICAgICAgImVtcHR5IiA6IHRydWUsCiAgICAgICJ1bnNvcnRlZCIgOiB0cnVlCiAgIH0sCiAgICJsYXN0IiA6IHRydWUsCiAgICJwYWdlYWJsZSIgOiB7IC4uLiB9LAogICAiZmlyc3QiIDogdHJ1ZSwKICAgInRvdGFsUGFnZXMiIDogMSwKICAgIm51bWJlciIgOiAwLAogICAibnVtYmVyT2ZFbGVtZW50cyIgOiAxLAogICAic2l6ZSIgOiA1LAogICAiY29udGVudCIgOiBbCiAgICAgIHsKICAgICAgICAgInByZWNvVW5pdGFyaW8iIDogNTI5Ljk5LAogICAgICAgICAiZGVzY3JpY2FvIiA6ICJVbSBwcm9kdXRvIHBhcmEgZ2VyZW5jaWFyIHBhY290ZXMiLAogICAgICAgICAibm9tZSIgOiAiTWF2ZW4iLAogICAgICAgICAiY2F0ZWdvcmlhIiA6IHsKICAgICAgICAgICAgImlkIiA6IDMsCiAgICAgICAgICAgICJhdGl2byIgOiB0cnVlLAogICAgICAgICAgICAibm9tZSIgOiAiTGl2cmFyaWEiCiAgICAgICAgIH0sCiAgICAgICAgICJtYXJjYSIgOiB7CiAgICAgICAgICAgICJkZXNjcmljYW8iIDogIlVtYSBtYXJjYSBpbm92YWRvcmEiLAogICAgICAgICAgICAiaWQiIDogMSwKICAgICAgICAgICAgIm5vbWUiIDogIlNob3cgZGUgTWFyY2EiCiAgICAgICAgIH0sCiAgICAgICAgICJpZCIgOiAyLAogICAgICAgICAiZm9ybmVjZWRvciIgOiB7CiAgICAgICAgICAgICJjbnBqIiA6ICIqKioqKioqKiIsCiAgICAgICAgICAgICJpZCIgOiAxLAogICAgICAgICAgICAiZW5kZXJlY28iIDogIlJ1YSAxIiwKICAgICAgICAgICAgImVtYWlsIiA6ICJmb3JuZWNlZG9yMUAqKioqKioqKiIsCiAgICAgICAgICAgICJ0ZWxlZm9uZSIgOiAiMTExMTExKioqKiIsCiAgICAgICAgICAgICJub21lIiA6ICJGb3JuZWNlZG9yIEZvckFsbCIKICAgICAgICAgfSwKICAgICAgICAgInVuaWRhZGUiIDogInVuaWRhZGUiCiAgICAgIH0KICAgXSwKICAgInRvdGFsRWxlbWVudHMiIDogMSwKICAgImVtcHR5IiA6IGZhbHNlCn0KIGBgYAogCiAjIyBDb250cmlidWluZG8gY29tIGVzdGUgcHJvamV0bwoKUGFyYSBjb250cmlidWlyIGNvbSBlc3RlIHByb2pldG8sIHNpZ2EgZXN0ZXMgcGFzc29zOgoKMS4gRmHDp2EgdW0gJ0ZvcmsnIGRlc3RlIHJlcG9zaXTDs3JpbzsKMi4gQ3JpZSB1bSAnQnJhbmNoJyBjb20gbyBjw7NkaWdvIG5vIHRlcm1pbmFsOiBgZ2l0IGNoZWNrb3V0IC1iIGZlYXR1cmUvPE5PTUVfRE9fQlJBTkNIPmA7CjMuIEZhw6dhIHN1YXMgYWx0ZXJhw6fDtWVzIGUgcmVhbGl6ZSBvICdhZGQgJiBjb21taXQnIGNvbSBvIHNlZ3VpbnRlIGPDs2RpZ28gbm8gdGVybWluYWw6IGBnaXQgYWRkIDxhcnF1aXZvcz4gJiYgZ2l0IGNvbW1pdCAtbSAibWVuc2FnZW0iYCwgbWVuc2FnZW0gZGUgY29tbWl0IHV0aWxpemFuZG8gW0NvbnZlbnRpb25hbENvbW1pdF0oaHR0cHM6Ly93d3cuY29udmVudGlvbmFsY29tbWl0cy5vcmcvZW4vdjEuMC4wLyk7CjQuIEVudMOjbywgcmVhbGl6ZSBvICdwdXNoJyBwYXJhIG8gb3JpZ2lhbmw6IGBnaXQgcHVzaCBvcmlnaW4gPGJvb3RjYW1wLWJhY2tlbmQtc3ByaW5nPi88TE9DQUw+YDsKNS4gUG9yIGZpbSwgY3JpZSB1bSAncHVsbCByZXF1ZXN0JyBwYXJhIGVzdGUgcmVwb3NpdMOzcmlvLgoKRW0gY2FzbyBkZSBkw7p2aWRhcywgbGVpYSBhIGRvY3VtZW50YcOnw6NvIGRvIEdpdEh1YiBzb2JyZSBbY29tbyBjcmlhciBwdWxsIHJlcXVlc3RzXShodHRwczovL2hlbHAuZ2l0aHViLmNvbS9lbi9naXRodWIvY29sbGFib3JhdGluZy13aXRoLWlzc3Vlcy1hbmQtcHVsbC1yZXF1ZXN0cy9jcmVhdGluZy1hLXB1bGwtcmVxdWVzdCkuCg== readmeEtag: '"b03d3692960f83a8d6953772a9ce077009a83473"' readmeLastModified: Fri, 22 Jan 2021 03:20:43 GMT repositoryId: 304887142 description: 'Bootcamp Backend: projeto de ecommerce usando Spring Boot' created: '2020-10-17T13:45:04Z' updated: '2025-12-13T18:52:17Z' language: Java archived: true stars: 0 watchers: 1 forks: 0 owner: thiagojacinto logo: https://avatars.githubusercontent.com/u/46906069?v=4 license: MPL-2.0 repoEtag: '"eb077e6aedfd6cc494251f14967844da474e33202c66b2fe083d733836da12b0"' repoLastModified: Sat, 13 Dec 2025 18:52:17 GMT foundInMaster: true category: Data Validators id: 032c859199b555a28058a04e38fbb9b3 - source: openapi3 tags repository: https://github.com/mal2-project/fake-shop-detection_database v3: true repositoryMetadata: base64Readme: >- IyBGYWtlLVNob3AgRGV0ZWN0aW9uIERhdGFiYXNlCgojIyBBYm91dCAvIFN5bm9wc2lzCgoqIE1hbmFnZW1lbnQgZm9yIG1hbnVhbGx5IGNsYXNzaWZ5aW5nIHJlcG9ydGVkIHdlYnNpdGVzIChlLmcuIHZpYSBmYWtlIHNob3AgZGV0ZWN0aW9uIFJFU1QgQVBJKS4KKiBQcm9qZWN0IHN0YXR1czogd29ya2luZy9wcm90b3R5cGUKCiMjIFRhYmxlIG9mIGNvbnRlbnRzCgo+ICogW1JlcXVpcmVtZW50c10oI3JlcXVpcmVtZW50cykKPiAqIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCj4gKiBbVXNhZ2VdKCN1c2FnZSkKPiAgKiBbU2NyZWVuc2hvdHNdKCNzY3JlZW5zaG90cykKPiAqIFtSRVNULUFQSV0oI3Jlc3QtYXBpKQo+ICogW1Jlc291cmNlcyAoRG9jdW1lbnRhdGlvbiBhbmQgb3RoZXIgbGlua3MpXSgjcmVzb3VyY2VzLWRvY3VtZW50YXRpb24tYW5kLW90aGVyLWxpbmtzKQo+ICogW0xpY2Vuc2VdKCNsaWNlbnNlKQo+ICogW0Fib3V0IE1BTDJdKCNhYm91dC1tYWwyKQoKIyMgUmVxdWlyZW1lbnRzCgoqIFVidW50dSAxOC4wNAoqIFB5dGhvbiAzLjUKKiBQb3N0Z3JlU1FMIDEwCiogUHl0aG9uLVBhY2thZ2VzIGFzIGRlZmluZWQgaW4gW3JlcXVpcm1lbnRzLnR4dF0oZGIvcmVxdWlyZW1lbnRzLnR4dCkKCiMjIEluc3RhbGxhdGlvbgoKQ3JlYXRlIGEgUHl0aG9uIHZpcnR1YWwgZW52aXJvbm1lbnQgd2l0aCBlLmcuIFt2aXJ0dWFsZW52d3JhcHBlcl0oaHR0cHM6Ly92aXJ0dWFsZW52d3JhcHBlci5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvKS4KVGhlIFB5dGhvbiB2ZXJzaW9uIHVzZWQgaXMgMy41LgoKYGBgc2hlbGwKJCBta3ZpcnR1YWxlbnYgLXAgL3BhdGgvdG8vcHl0aG9uMy41IG1hbDJEQgpgYGAKCkluc3RhbGwgdGhlIHJlcXVpcmVkIFB5dGhvbiBwYWNrYWdlcwoKYGBgc2hlbGwKcGlwIGluc3RhbGwgLXIgZGIvcmVxdWlyZW1lbnRzLnR4dApgYGAKClBvc3RncmVTUUwgMTAgaXMgdXNlZCBhcyBkYXRhYmFzZS4gQ3JlYXRlIGEgZGF0YWJhc2UgYW5kIGNoYW5nZSB0aGUgYGRiL3NldHRpbmdzL2Jhc2UucHlgIGFjY29yZGluZ2x5LgoKYGBgcHl0aG9uCkRCX0hPU1QgPSAibG9jYWxob3N0IgpEQl9VU0VSID0gIm1hbDJkYiIKREJfTkFNRSA9ICJtYWwyZGIiCkRCX1BBU1NXT1JEID0gIk1ZX1NFQ1JFVF9QQVNTV09SRCIKYGBgCgpBZnRlciB0aGF0LCB0aGUgZGF0YWJhc2UgaXMgaW5pdGlhbGl6ZWQgYXMgZm9sbG93czoKCmBgYHNoZWxsCmNkIGRiCi4vbWFuYWdlLnB5IG1pZ3JhdGUKLi9tYW5hZ2UucHkgbG9hZGRhdGEgaW5pdF9ncm91cHMKLi9tYW5hZ2UucHkgbG9hZGRhdGEgaW5pdF93ZWJzaXRlX3Jpc2Nfc2NvcmUKLi9tYW5hZ2UucHkgbG9hZGRhdGEgaW5pdF93ZWJzaXRlX3R5cGVzCi4vbWFuYWdlLnB5IGxvYWRkYXRhIGluaXRfd2Vic2l0ZV9jYXRlZ29yeQouL21hbmFnZS5weSBsb2FkZGF0YSBpbml0X3dlYnNpdGVzX3JlcG9ydGVkX2J5CmBgYAoKVGhlIGFkbWluIHVzZXIgaXMgY3JlYXRlZCB3aXRoIHRoZSBmb2xsb3dpbmcgY29tbWFuZDoKCmBgYHNoZWxsCiQgLi9tYW5hZ2UucHkgY3JlYXRlc3VwZXJ1c2VyCmBgYAoKRW50ZXIgYSBtYWlsIHNlcnZlciBpbiB0aGUgYGRiL3NldHRpbmdzL2Jhc2UucHlgLgoKYGBgcHl0aG9uCkFETUlOUyA9ICgKICAgICgiTmFtZSIsICJuYW1lQGRvbWFpbi50bGQiKSwKKQoKU0VSVkVSX0VNQUlMID0gIm5hbWVAZG9tYWluLnRsZCIKCkVNQUlMX0hPU1QgPSAibWFpbC5kb21haW4udGxkIgpFTUFJTF9IT1NUX1VTRVIgPSAibWFsMkRCIgpFTUFJTF9IT1NUX1BBU1NXT1JEID0gIk1ZX1NFQ1JFVF9QQVNTV09SRCIKCkRFRkFVTFRfRlJPTV9FTUFJTCA9ICJub3JlcGx5QGRvbWFpbi50bGQiCmBgYAoKIyMjIENocm9tZS1UcmVpYmVyIGbDvHIgU2VsZW5pdW0KClRvIGNyZWF0ZSB3ZWIgcGFnZSBzY3JlZW5zaG90IHRoZSBgY2hyb21pdW0tY2hyb21lZHJpdmVyYCBpcyBuZWVkZWQuCgpgYGAKJCBhcHQgaW5zdGFsbCBjaHJvbWl1bS1jaHJvbWVkcml2ZXIKYGBgCgojIyBVc2FnZQoKTm93IHRoZSBzZXJ2ZXIgY2FuIGJlIHN0YXJ0ZWQgd2l0aCB0aGUgRGphbmdvIGNvbW1hbmQgcnVuc2VydmVyLgoKYGBgc2hlbGwKLi9tYW5hZ2UucHkgcnVuc2VydmVyCmBgYAoKSW4geW91ciBicm93c2VyLCB5b3UgY2FuIG5vdyBnbyB0byBodHRwOi8vMTI3LjAuMC4xOjgwMDAvLgoKQXR0ZW50aW9uLCB0aGlzIGlzIGZvciBkZXZlbG9wbWVudCBvbmx5LiBJbiBhIHByb2R1Y3Rpb24gZW52aXJvbm1lbnQsIGZvciBleGFtcGxlLCBbdXdzZ2ldKGh0dHBzOi8vdXdzZ2ktZG9jcy5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvV1NHSXF1aWNrc3RhcnQuaHRtbCkgY2FuIGJlIHVzZWQgd2l0aCBbYXBhY2hlMl0oaHR0cDovL2h0dHBkLmFwYWNoZS5vcmcvKS4KIyMjIFNjcmVlbnNob3RzCgohW21hbDJEQl0oc2NyZWVuc2hvdC5wbmcpCgojIyBSRVNULUFQSQoKVGhlIFJFU1QgQVBJIGRvY3VtZW50YXRpb24gaXMgYXZhaWxhYmxlIGF0IGh0dHA6Ly8xMjcuMC4wLjE6ODAwMC9hcGkuCgojIyBSZXNvdXJjZXMgKERvY3VtZW50YXRpb24gYW5kIG90aGVyIGxpbmtzKQoKKiBbRGphbmdvIFdlYiBGcmFtZXdvcmtdKGh0dHBzOi8vZG9jcy5kamFuZ29wcm9qZWN0LmNvbS9lbi8yLjIvKQoqIFtEamFuZ28gUkVTVCBGcmFtZXdvcmtdKGh0dHBzOi8vd3d3LmRqYW5nby1yZXN0LWZyYW1ld29yay5vcmcvKQoKCiMjIEFib3V0IE1BTDIKClRoZSBNQUwyIHByb2plY3QgYXBwbGllcyBEZWVwIE5ldXJhbCBOZXR3b3JrcyBhbmQgVW5zdXBlcnZpc2VkIE1hY2hpbmUgTGVhcm5pbmcgdG8gYWR2YW5jZSBjeWJlcmNyaW1lIHByZXZlbnRpb24gYnkgYSkgYXV0b21hdGluZyB0aGUgZGlzY292ZXJ5IG9mIGZyYXVkdWxlbnQgZUNvbW1lcmNlIGFuZCBiKSBkZXRlY3RpbmcgUG90ZW50aWFsbHkgSGFybWZ1bCBBcHBzIChQSEFzKSBpbiBBbmRyb2lkLgpUaGUgZ29hbCBvZiB0aGUgTUFMMiBwcm9qZWN0IGlzIHRvIHByb3ZpZGUgKGkpIGFuIE9wZW4gU291cmNlIGZyYW1ld29yayBhbmQgZXhwZXJ0IHRvb2xzIHdpdGggaW50ZWdyYXRlZCBmdW5jdGlvbmFsaXR5IGFsb25nIHRoZSByZXF1aXJlZCBwaXBlbGluZSDigJMgZnJvbSBtYWxpY2lvdXMgZGF0YSBhcmNoaXZpbmcsIGZlYXR1cmUgc2VsZWN0aW9uIGFuZCBleHRyYWN0aW9uLCB0cmFpbmluZyBvZiBNYWNoaW5lIExlYXJuaW5nIGNsYXNzaWZpY2F0aW9uIGFuZCBkZXRlY3Rpb24gbW9kZWxzIHRvd2FyZHMgZXhwbGFpbmFiaWxpdHkgaW4gdGhlIGFuYWx5c2lzIG9mIHJlc3VsdHMgKGlpKSB0byBleGVjdXRlIGl0cyBjb21wb25lbnRzIGF0IHNjYWxlIGFuZCAoaWlpKSB0byBwdWJsaXNoIGFuIGFubm90YXRlZCBHcm91bmQtVHJ1dGggZGF0YXNldCBpbiBib3RoIGFwcGxpY2F0aW9uIGRvbWFpbnMuIFRvIHJhaXNlIGF3YXJlbmVzcyBmb3IgY3liZXJjcmltZSBwcmV2ZW50aW9uIGluIHRoZSBnZW5lcmFsIHB1YmxpYywgdHdvIGRlbW9uc3RyYXRvcnMsIGEgRmFrZS1TaG9wIERldGVjdGlvbiBCcm93c2VyIFBsdWdpbiBhcyB3ZWxsIGFzIGEgQW5kcm9pZCBNYWx3YXJlIERldGVjdGlvbiBBbmRyb2lkIGFwcCBhcmUgcmVsZWFzZWQgdGhhdCBhbGxvdyBsaXZlLWluc3BlY3Rpb24gYW5kIEFJIGJhc2VkIHByZWRpY3Rpb25zIG9uIHRoZSB0cnVzdHdvcnRoaW5lc3Mgb2YgZUNvbW1lcmNlIHNpdGVzIGFuZCBBbmRyb2lkIGFwcHMuCgpUaGUgd29yayBpcyBiYXNlZCBvbiByZXN1bHRzIGNhcnJpZWQgb3V0IGluIHRoZSByZXNlYXJjaCBwcm9qZWN0IFtNQUwyIHByb2plY3RdKGh0dHBzOi8vcHJvamVrdGUuZmZnLmF0L3Byb2pla3QvMzA0NDk3NSksIHdoaWNoIHdhcyBwYXJ0aWFsbHkgZnVuZGVkIGJ5IHRoZSBBdXN0cmlhbiBGZWRlcmFsIE1pbmlzdHJ5IGZvciBDbGltYXRlIEFjdGlvbiwgRW52aXJvbm1lbnQsIEVuZXJneSwgTW9iaWxpdHksIElubm92YXRpb24gYW5kIFRlY2hub2xvZ3kgKEJNSykgdGhyb3VnaCB0aGUgSUNUIG9mIHRoZSBmdXR1cmUgcmVzZWFyY2ggcHJvZ3JhbSAoNnRoIGNhbGwpIG1hbmFnZWQgYnkgdGhlIEF1c3RyaWFuIGZlZGVyYWwgZnVuZGluZyBhZ2VuY3kgKEZGRykuCiogQXVzdHJpYW4gSW5zdGl0dXRlIG9mIFRlY2hub2xvZ3kgR21iSCwgQ2VudGVyIGZvciBEaWdpdGFsIFNhZmV0eSBhbmQgU2VjdXJpdHkgW0FJVF0oaHR0cHM6Ly93d3cuYWl0LmFjLmF0LykKKiBBdXN0cmlhbiBJbnN0aXR1dGUgZm9yIEFwcGxpZWQgVGVsZWNvbW11bmljYXRpb25zIFvDlklBVF0oaHR0cHM6Ly93d3cub2lhdC5hdCkKKiBYLU5FVCBTZXJ2aWNlcyBHbWJIIFtYTkVUXShodHRwczovL3gtbmV0LmF0L2RlLykKKiBLdXJhdG9yaXVtIHNpY2hlcmVzIMOWc3RlcnJlaWNoIFtLU8OWXShodHRwczovL2t1cmF0b3JpdW0tc2ljaGVyZXMtb2VzdGVycmVpY2guYXQvKQoqIElLQVJVUyBTZWN1cml0eSBTb2Z0d2FyZSBbSUtBUlVTXShodHRwczovL3d3dy5pa2FydXNzZWN1cml0eS5jb20vKQoKTW9yZSBpbmZvcm1hdGlvbiBpcyBhdmFpbGFibGUgYXQgW3d3dy5tYWx6d2VpLmF0XShodHRwOi8vd3d3Lm1hbHp3ZWkuYXQpCgojIyBDb250YWN0CkZvciBkZXRhaWxzIG9uIGJlaGFsZiBvZiB0aGUgTUFMMiBjb25zb3J0aXVtIGNvbnRhY3Q6IApBbmRyZXcgTGluZGxleSAocHJvamVjdCBsZWFkKQpSZXNlYXJjaCBFbmdpbmVlciwgRGF0YSBTY2llbmNlICYgQXJ0aWZpY2lhbCBJbnRlbGxpZ2VuY2UKQ2VudGVyIGZvciBEaWdpdGFsIFNhZmV0eSBhbmQgU2VjdXJpdHksIEFJVCBBdXN0cmlhbiBJbnN0aXR1dGUgb2YgVGVjaG5vbG9neSBHbWJICkdpZWZpbmdnYXNzZSA0IHwgMTIxMCBWaWVubmEgfCBBdXN0cmlhClQgKzQzIDUwNTUwLTQyNzIgfCBNICs0MyA2NjQgODE1Nzg0OCB8IEYgKzQzIDUwNTUwLTQxNTAKYW5kcmV3LmxpbmRsZXlAYWl0LmFjLmF0IHwgd3d3LmFpdC5hYy5hdApvcgpXb2ZsZ2FuZyBFaWJuZXIsIFgtTkVUIFNlcnZpY2VzIEdtYkgsIHdlQHgtbmV0LmF0CgojIyBMaWNlbnNlClRoZSBNQUwyIFNvZnR3YXJlIHN0YWNrIGlzIGR1YWwtbGljZW5zZWQgdW5kZXIgY29tbWVyY2lhbCBhbmQgb3BlbiBzb3VyY2UgbGljZW5zZXMuIApUaGUgU29mdHdhcmUgaW4gdGhpcyByZXBvc2l0b3J5IGlzIHN1YmplY3Qgb2YgdGhlIHRlcm1zIGFuZCBjb25kaXRpb25zIGRlZmluZWQgaW4gZmlsZSAnTElDRU5TRS5tZCcK readmeEtag: '"54de9967b8aa28db1a60514bcaeeb716e14a9196"' readmeLastModified: Mon, 22 Mar 2021 22:44:14 GMT repositoryId: 350507329 description: >- MAL2 Fake-Shop Detection Database. Multi-User Application with OpenAPI3 Endpoints for deadling with the inspection of fake-shops created: '2021-03-22T22:25:54Z' updated: '2021-03-23T20:43:16Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: mal2-project logo: https://avatars.githubusercontent.com/u/75363498?v=4 license: NOASSERTION repoEtag: '"aca4726d936bd31e75545fef91740eba5da3277a8048787233375012ecf0410f"' repoLastModified: Tue, 23 Mar 2021 20:43:16 GMT foundInMaster: true category: Server Implementations id: 963af65d9b1e14f4131885f08854968f - source: openapi3 tags repository: https://github.com/brevetech/breve_drf_template v3: true repositoryMetadata: base64Readme: >- # breve_drf_template

[![Breve Template](https://img.shields.io/badge/breve-template-orange?style=for-the-badge&logo=data:image/svg+xml;base64,PHN2ZyB4bWxucz0iaHR0cDovL3d3dy53My5vcmcvMjAwMC9zdmciIHZpZXdCb3g9IjAgMCA3MCA3MCI+PGRlZnM+PHN0eWxlPi5jbHMtMXtmaWxsOiNlMDAwNGQ7fS5jbHMtMntmaWxsOiNmZmY7fTwvc3R5bGU+PC9kZWZzPjx0aXRsZT5SZWN1cnNvIDFsb2dvIHBhcmEgcGVyZmlsIGRlIGNvcnJlbzwvdGl0bGU+PGcgaWQ9IkNhcGFfMiIgZGF0YS1uYW1lPSJDYXBhIDIiPjxnIGlkPSJDYXBhXzEtMiIgZGF0YS1uYW1lPSJDYXBhIDEiPjxyZWN0IGNsYXNzPSJjbHMtMSIgd2lkdGg9IjcwIiBoZWlnaHQ9IjcwIi8+PHBhdGggaWQ9IlN1c3RyYWNjacOzbl8yIiBkYXRhLW5hbWU9IlN1c3RyYWNjacOzbiAyIiBjbGFzcz0iY2xzLTIiIGQ9Ik00Mi41LDU4YTE2LDE2LDAsMCwxLTYuMjUtMS4yN2wtLjY3LS4zdi04LjdIMjUuMzVWMTFIMzUuNThWMjcuMjlsLjY3LS4zYTE2LDE2LDAsMCwxLDE3LjYsMy40NiwxNi4xNywxNi4xNywwLDAsMS01LjEsMjYuMjdBMTYsMTYsMCwwLDEsNDIuNSw1OFptMC0yMi45NGE2LjgxLDYuODEsMCwxLDAsNi43Nyw2LjgzdjBBNi43OSw2Ljc5LDAsMCwwLDQyLjUsMzUuMDVaIi8+PHJlY3QgaWQ9IlJlY3TDoW5ndWxvXzEzIiBkYXRhLW5hbWU9IlJlY3TDoW5ndWxvIDEzIiBjbGFzcz0iY2xzLTIiIHg9IjExLjU2IiB5PSI0Ny43MiIgd2lkdGg9IjEwLjIzIiBoZWlnaHQ9IjEwLjI4Ii8+PC9nPjwvZz48L3N2Zz4=)](https://www.brevetech.com/)
[![forthebadge made-with-python](https://img.shields.io/badge/made_with-python-blue?style=for-the-badge&logo=python)](https://www.python.org/)
[![forthebadge django-rest](https://img.shields.io/badge/django-rest_framework-a30000?style=for-the-badge&logo=django)](https://www.python.org/)
[![forthebadge pre-commit](https://img.shields.io/badge/pre_commit-enabled-green?style=for-the-badge&logo=pre-commit)](https://github.com/pre-commit/pre-commit)
[![forthebadge pre-commit](https://img.shields.io/badge/uses-commitizen-E0004D?style=for-the-badge&logo=git)](https://github.com/commitizen-tools/commitizen)
[![License: MIT](https://img.shields.io/badge/License-MIT-blue.svg?style=for-the-badge)](https://www.gnu.org/licenses/gpl-3.0)

---

A Django Rest Framework base template with custom configurations, intended to save time with some of the boilerplate configuration. Optmized for PyCharm IDE and VSCode.

## Index

- [breve_drf_template](#breve_drf_template)
  - [Index](#index)
  - [Features](#features)
  - [Getting started](#getting-started)
  - [`.env` structure](#env-structure)
  - [Installing additional modules](#installing-additional-modules)
  - [Creating new app](#creating-new-app)
  - [Endpoint clean code architecture](#endpoint-clean-code-architecture)
  - [OpenAPI 3 Schema Documentation](#openapi-3-schema-documentation)
  - [Authentication](#authentication)
  - [CORS Policy](#cors-policy)
  - [pre-commit and QCA flow](#pre-commit-and-qca-flow)
  - [Commitizen](#commitizen)
  - [Github Actions CI/CD Workflow](#github-actions-cicd-workflow)
  - [Contribution](#contribution)

## Features

- Python 3 and Django 4.0.2 (locked to this version due to security issues).
- `.env` managing with relevant safe variables using [`django-environ`](https://django-environ.readthedocs.io/en/latest/) module.
- `pipfile` requirements modules handling.
- Script for enabling/disabling template functioning. **Use when intended to contribute**.
- Environment-sensitive settings (`dev`, `test` and `prod`).
- OpenAPI 3 schema and DocView with full customizable endpoint metadata, using [`drf-spectacular`](https://drf-spectacular.readthedocs.io/en/latest/readme.html).
- JWT Authentication preconfigured using [Simple JWT](https://django-rest-framework-simplejwt.readthedocs.io/en/latest/).
- Default [Github Actions](https://github.com/features/actions) CI workflow.
- `whitenoise` staticfile handling for **Heroku** deploys and similar ones.
- Clean code endpoint architecture.
- Custom error handling for centralized logging and exception behaviour definition
- JSON responses for 404 and 500 errors in non debug environment
- Boilerplate abstract models (Timestamped and Person).
- Heroku procfile default configuration
- Log request middleware

## Getting started

To use this template run the **install shell** provided in this project, `django` and `pipenv` packages are required globally. If you are in Windows, use `git bash`, Powershell script version is not implemented yet:

```sh
# Execute install shell direct from repo
$ bash <(curl -s https://raw.githubusercontent.com/brevetech/breve_drf_template/master/create_project.sh)
```

This will start a new django project with the template structure. To run the project, you need to set up the `.env` file first.

## `.env` structure

This project uses a `.env` file and the [`django-environ`](https://django-environ.readthedocs.io/en/latest/) module to keep all the sensitive configuration, like secret keys, database URI's or some other credentials off version control. The default `.env` structure includes this variables:

```toml
DEBUG=<value>
SECRET_KEY=<secret_key>
DEV_DB_URL=<dev_db_url>
PROD_DB_URL=<prod_db_url>
```

- **DEBUG** (`True`, `False`) will define if debug mode is enabled or not. Use `True` for development and testing purpose and `False` in production environments. This variable also defines which configuration is being used.
- **SECRET_KEY** is the seed used for encryption purposes. Django recommends to keep it off source control. To generate a secret key, you can go to [Djecrety](https://djecrety.ir).
- **DEV_DB_URL** is the development environment database URL. For testing purpose you can use `sqlite:///db.sqlite3` as DEV_DB_URL. If you want to use another database engine, remember to **install** the required package (e.g. `psycopg2` for PostgreSQL databases).
- **PROD_DB_URL** is the production environment database URL. For testing purpose you can use `sqlite:///db.sqlite3` as PROD_DB_URL. If you want to use another database engine, remember to **install** the required package (e.g. `psycopg2` for PostgreSQL databases).

## Installing additional modules

To install a new package, run:

```sh
# Install package
$ pipenv install <package>
```

This will keep all requirements packages into the `Pipfile`, improving project compatibility across teams and allowing easy use of virtual environments.

To remove a package, run:

```sh
# Uninstall package
$ pipenv uninstall <package>
```

This will remove the package from the `Pipfile`.

## Creating new app

This template defines a different folder structure for apps, putting them in an `apps` directory inside the project directory. This is only for comfort purposes. Because of this, to create a new app, you need to run this:

```shell
# Creating new project app
$ mkdir {project_name}/apps/{app_name}
$ python manage.py startapp {app_name} apps/{app_name}
```

After this, you can install the app in the `settings.base`, but using the full app name, not the relative one.

- ~~.app~~
- project_name.apps.app

## Endpoint clean code architecture

This template uses a specific architecture based (kinda) on OOP languages **Clean Architecture**. This is to keep all business logic and validation code off the views code, having cleaner code and more modularization. The files implied here are:

- Serializers: defines all input and output objects for the endpoints. This serializers can be based on database models `ModelSerializer` or can be full custom serializer `Serializer`. If you want to use different serializers for input and output (e.g. create a join query for output that joins many models or have a different structure than the defined in the model for a create endpoint), you will need to use the `ReadWriteSerializerMixin`.
- Views: defines all endpoints. You can use Function Based Views or Class Based Views (`APIView`, `ViewSet`, `ModelViewSet`).
- Handlers: defines functions with all business logic and validations, database queries and database commands. Use a handler only if you need to redefine a complex logic for the endpoint, otherwise, check if `ModelViewSet` can help.
- Entities: defines input and output objects for the handlers.
- ErrorSerializer: defines the basic error structure for endpoint responses.

Entities, Serializers and handlers are not created when using `python manage.py startapp`, you'll need to create them manually.

The **common** package is meant to store all common access objects, like classes, error serializers, mixins, etc.

## OpenAPI 3 Schema Documentation

This template includes OpenAPI 3 schema documentation, using `.md` files to create rich text descriptions. This descriptions are saved in the **apidocs** folder.

The `index.md` file represents the general schema description. Modify it to edit the schema general description value, like instructions, project objectives, etc. The resting general inforation values are stored in the `SPECTACULAR_SETTINGS` dict in `settings.base`; for more information see [official documentation](https://drf-spectacular.readthedocs.io/en/latest/settings.html).

For every other endpoint you can create a `{endpoint}.md` file in the `apidocs/endpoints/{app}` folder. To load it in the corresponding endpoint, read the file via `read_docs_md()` utils function, and load it via `extend_schema()` function (included in [`drf-spectacular`](https://drf-spectacular.readthedocs.io/en/latest/customization.html#step-2-extend-schema)). Example:

```python
@method_decorator(name='list', decorator=extend_schema(
    summary="Location list",
    # list.md file stores the api description for this endpoint
    description=read_docs_md('endpoints/core/list'),
    responses={
        200: LocationSerializer(many=True),
        400: ErrorSerializer(),
        500: ErrorSerializer()
    }
))
class LocationViewSet(viewsets.ViewSet):
```

## Authentication

This template uses `djangorestframework_simple_jwt` module for JWT Bearer token authentication.

This module works with a pair of JWT tokens, a `access` one, that is needed as `Authorization` in header of any request, and a `refresh` one that is used to get a new `access` token. Access token lives for 5 minutes by default, and refresh token lives for 24 hours by default; this configuration can be set up as required, see [official documentation](https://django-rest-framework-simplejwt.readthedocs.io/en/latest/settings.html).

The `auth_create` endpoint, for getting an access and refresh token pair with a django user, and the `auth_create_refresh`, that gets a new access token providing a refresh token, are both already exposed and configured by default.

## CORS Policy

This template uses `django-cors-headers` middleware for cors handling. The configuration is set up in `CORS_ORIGIN_ALLOW_ALL` key at `settings.base`, set `False` as default. Change it to `True` for development purposes when using SPA frontends and third party consumers.

## pre-commit and QCA flow

This template integrates a full QCA flow using `pre-commit` hooks, that automates code quality tasks on every commit. Using this in your project from the beginning deletes the need to worry about issues as unattended warnings, code styles, etc, doing it automatically.

The QCA flow integrates this hooks:

- [`isort`](https://github.com/PyCQA/isort) for import sorting
- [`black`](https://github.com/psf/black) for code formatting
- [`autoflake`](https://github.com/PyCQA/autoflake) for unused imports and variables removal
- [`pylint`](https://readthedocs.org/projects/pylint/) for Python complaint linting
- `commitizen-check` for commit message structure checking

If `pre-commit` is not working for you, try to run `pre-commit install` for installing the hooks. All the hooks configuration is in the `.pre-commit-config.yaml` file and every tool configuration is in its specified config file.

## Commitizen

This template also includes [`commitizen`](https://commitizen-tools.github.io/commitizen/) as a commit writing helper, following the [convetional commits](https://www.conventionalcommits.org/en/v1.0.0/) standards and making commits useful for release tasks. To use it, replace `git commit` command with `git cz commit`. All template specs and configurations are in `pyproject.toml` file.

## Github Actions CI/CD Workflow

This template includes a basic Github Actions CI Workflow, toggled on develop branch (git flow) push and pull request. It's disabled by default; to enable it, go to `.github/workflows/ci.yml` and toggle comment in all file lines. It also includes a CD workflow that simply bumps version tag, and makes a release with changelog, powered by **commitizen**.

## Contribution

**For contribution**, base your branch in `develop`. Disable template features using the provided `toggle_template.py` script. This will find and replace the `py-tpl` strings `{{project_name}}` with simple strings `project_name` in all project files in order to allow it to run and test.

```sh
# "disable" replaces {{project_name}} with project_name, allowing code to run, "enable" discards the find and replace enabling the py-tpl features required to django project template
python toggle_template.py <enable|disable>
```

When you finished your changes, re-enable the template features with previous script, and then pull request to develop. **Pull requests won't be accepted if template features are not enabled**.
 readmeEtag: '"3916d1268dbc9fa6e51c68be2de4f8e4e0e56780"' readmeLastModified: Fri, 08 Jul 2022 04:38:38 GMT repositoryId: 329782725 description: Django Rest base template created: '2021-01-15T01:58:27Z' updated: '2022-01-11T04:49:30Z' language: Python archived: false stars: 0 watchers: 0 forks: 0 owner: brevetech logo: https://avatars.githubusercontent.com/u/61919566?v=4 license: MIT repoEtag: '"9e3fb3f18ce110e8802455b5105c5a47ffd9290b955878d6da63485f780060f3"' repoLastModified: Tue, 11 Jan 2022 04:49:30 GMT foundInMaster: true category: Server Implementations id: 3252a3757181b778457575dd31338472 - source: openapi3 tags repository: https://github.com/josuablejeru/aws-openapi-ui v3: true repositoryMetadata: base64Readme: >- IyBBV1MgT3BlbmFwaSBVSQoKR2VuZXJhdGUgYW5kIHNlcnZlIGEgVUkgZm9yIHlvdXIgb3BlbmFwaSBkZWZpbml0aW9uIHdpdGggQVdTIENESwoKIyMgSW5zdGFsbGF0aW9uCmBgYGJhc2gKJCB5YXJuIGFkZCBhd3Mtb3BlbmFwaS11aQpgYGAKIyMg8J+agCBFeGFtcGxlIHVzYWdlCmBgYHR5cGVzY3JpcHQKbmV3IEF3c09wZW5hcGlVaShzY29wZSwgJ215LXNlcnZpY2UnLCB7CiAgICBvcGVuYXBpUGF0aDogJzxwYXRoLXRvLXNwZWM+JywKICAgIHNlcnZlcnM6IFsKICAgICAgICAnYXBpLmZvby5jb20nLAogICAgICAgICdkZXYuYXBpLmZvby5jb20nCiAgICBdCiAgfSkKYGBgCgoKIyMg4pyM77iPIEdldCBpbiB0b3VjaCB3aXRoIG1lCgo8YSBocmVmPSJodHRwczovL2ludG8tdGhlLWNvZGUuY29tIiB0YXJnZXQ9Il9ibGFuayI+PGltZyBhbHQ9IlBlcnNvbmFsIFdlYnNpdGUiIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9QZXJzb25hbCUyMFdlYnNpdGUtJTIzMTIxMDBFLnN2Zz8mc3R5bGU9Zm9yLXRoZS1iYWRnZSZsb2dvQ29sb3I9d2hpdGUiIC8+PC9hPgo8YSBocmVmPSJodHRwczovL3R3aXR0ZXIuY29tL2pvc3VhYmxlamVydSIgdGFyZ2V0PSJfYmxhbmsiPjxpbWcgYWx0PSJUd2l0dGVyIiBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvdHdpdHRlci0lMjMxREExRjIuc3ZnPyZzdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289dHdpdHRlciZsb2dvQ29sb3I9d2hpdGUiIC8+PC9hPgo8YSBocmVmPSJodHRwczovL3d3dy5saW5rZWRpbi5jb20vaW4vam9zdWEtYmxlamVydS1hMjg3MWExNjQiIHRhcmdldD0iX2JsYW5rIj48aW1nIGFsdD0iTGlua2VkSW4iIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saW5rZWRpbi0lMjMwMDc3QjUuc3ZnPyZzdHlsZT1mb3ItdGhlLWJhZGdlJmxvZ289bGlua2VkaW4mbG9nb0NvbG9yPXdoaXRlIiAvPjwvYT4KWyFbIkJ1eSBNZSBBIENvZmZlZSJdKGh0dHBzOi8vd3d3LmJ1eW1lYWNvZmZlZS5jb20vYXNzZXRzL2ltZy9jdXN0b21faW1hZ2VzL29yYW5nZV9pbWcucG5nKV0oaHR0cHM6Ly93d3cuYnV5bWVhY29mZmVlLmNvbS9qb3N1YWJsZWplcnUpCgojIyDwn5OdIExpY2Vuc2UKCkRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBNSVQgTGljZW5zZS4gU2VlIExJQ0VOU0UgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgojIyBBY2tub3dsZWRnZW1lbnRzCi0gaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWk= readmeEtag: '"7b3c4db91cae4aa8e0d864699ed1419d3dd53f82"' readmeLastModified: Mon, 26 Jul 2021 10:46:30 GMT repositoryId: 388463929 description: serve your api with swagger-ui and CDK created: '2021-07-22T13:01:34Z' updated: '2022-01-28T09:22:48Z' language: HTML archived: false stars: 0 watchers: 1 forks: 0 owner: josuablejeru logo: https://avatars.githubusercontent.com/u/37913833?v=4 license: MIT repoEtag: '"4790a701b9b268ec1f81997dbafcaf8c6953b213b5a428e14ed9c18885f12b50"' repoLastModified: Fri, 28 Jan 2022 09:22:48 GMT foundInMaster: true category: - SDK - Server Implementations id: ee2544503d1967a786a966e774f9ef25 - source: openapi3 tags repository: https://github.com/virtue-dbis/vrem-unity-interface v3: true repositoryMetadata: base64Readme: >- IyBWUkVNIFVuaXR5IEludGVyZmFjZQoKQyMgY2xpZW50IGdlbmVyYXRlZCB3aXRoIFtPcGVuQVBJXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pIHRvIHByb3ZpZGUgYWNjZXNzIHRvCnRoZSBbVmlydHVhbCBFeGhpYml0aW9uIE1hbmFnZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9WSVJUVUUtREJJUy92aXJ0dWFsLWV4aGliaXRpb24tbWFuYWdlcikgc2VydmVyIEFQSQppbiBbVW5pdHldKGh0dHBzOi8vdW5pdHkuY29tLykuCgojIyBTZXR1cAoKVG8gaW5jbHVkZSB0aGUgbGF0ZXN0IHZlcnNpb24gb2YgdGhpcyBwYWNrYWdlIGluIHlvdXIgVW5pdHkgcHJvamVjdCwgYWRkIHRoZSBmb2xsb3dpbmcgbGluZSB0byB5b3VyIGBtYW5pZmVzdC5qc29uYApmaWxlOgoKYGBganNvbgogICAgImNoLnVuaWJhcy5kbWkuZGJpcy52cmVtLmNsaWVudCI6ICJzc2g6Ly9naXRAZ2l0aHViLmNvbTpWSVJUVUUtREJJUy92cmVtLXVuaXR5LWludGVyZmFjZS5naXQjbWFpbiIsCmBgYAoKQWx0ZXJuYXRpdmVseSB5b3UgY2FuIGFsc28gdXNlIHRoZSBVbml0eSBQYWNrYWdlIE1hbmFnZXIgYW5kIGFkZCB0aGUKVVJMIGBnaXRAZ2l0aHViLmNvbTpWSVJUVUUtREJJUy92cmVtLXVuaXR5LWludGVyZmFjZS5naXQjbWFpbmAuCgojIyBEZXZlbG9wbWVudAoKRm9yIGRldmVsb3BtZW50LCBjbG9uZSB0aGlzIHJlcG9zaXRvcnkgaW50byB0aGUgYFBhY2thZ2VzYCBkaXJlY3Rvcnkgb2YgdGhlIFVuaXR5IHByb2plY3QuICAKWW91IG1heSB3YW50IHRvIHJ1biBgLi9ncmFkbGV3IGNsZWFuIGRlcGxveUFuZFRpZHlgIHRvIGdlbmVyYXRlIHRoZSBiaW5hcmllcyBhbmQgKHJlKW9wZW4gdGhlIHByb2plY3QgaW4gVW5pdHkgaW4gb3JkZXIKdG8gZ2VuZXJhdGUgdGhlIGAubWV0YWAgZmlsZXMuCg== readmeEtag: '"797a06e730738e4fcd45887894f591d2943e8201"' readmeLastModified: Mon, 08 Nov 2021 13:56:41 GMT repositoryId: 401296924 description: OpenAPI bindings for VREM in Unity created: '2021-08-30T10:03:47Z' updated: '2021-11-08T13:56:43Z' language: null archived: false stars: 0 watchers: 0 forks: 0 owner: VIRTUE-DBIS logo: https://avatars.githubusercontent.com/u/55091121?v=4 repoEtag: '"efee29359349db60724d7bc5e0e58e15cbcf0fba41695f392215435557da58ff"' repoLastModified: Mon, 08 Nov 2021 13:56:43 GMT foundInMaster: true category: Code Generators id: b4f4604163ff96389f3a929b3d05bf13 - source: openapi3 tags repository: https://github.com/ans-group/laravel-api-docs v3: true id: aa2ac4e8bbb09ed6543330c6b06298ac repositoryMetadata: base64Readme: >- PGltZyBzcmM9Imh0dHBzOi8vaW1hZ2VzLnVrZmFzdC5jby51ay9sb2dvcy91a2Zhc3QvNDQxeDEyNl90cmFuc3BhcmVudF9zdHJhcGxpbmUucG5nIiBhbHQ9IlVLRmFzdCBMb2dvIiB3aWR0aD0iMzUwcHgiIGhlaWdodD0iYXV0byIgLz4KCiMgTGFyYXZlbCBBUEkgRG9jcyBbQmV0YV0KCipQYWNrYWdlIGlzIHN0aWxsIGluIGJldGEgYW5kIGlzIG5vdCByZWFkeSBmb3IgcHJvZHVjdGlvbiB1c2UqCgpBdXRvbWF0aWNhbGx5IGdlbmVyYXRlIEFQSSBkb2N1bWVudGF0aW9uIGZvciB5b3VyIGxhcmF2ZWwgQVBJJ3MgYmFzZWQgb2ZmIHlvdXIgYXBwbGljYXRpb24gcm91dGVzIGFuZCBoYW5keSBQSFAgOCBhdHRyaWJ1dGVzLgoKIyMgSW5zdGFsbGF0aW9uCgpGaXJzdCwgdXNlIGNvbXBvc2VyIHRvIHJlcXVpcmUgdGhlIHBhY2thZ2UgYXMgYmVsb3c6CgpgYGAKY29tcG9zZXIgcmVxdWlyZSB1a2Zhc3QvbGFyYXZlbC1hcGktZG9jcwpgYGAKClRoZW4gYWxsIHdlIG5lZWQgdG8gZG8gaXMgdG8gcmVnaXN0ZXIgdGhlIHNlcnZpY2UgcHJvdmlkZXIgaW4gdGhlIGBwcm92aWRlcnNgIGtleSBpbiBgY29uZmlnL2FwcC5waHBgOgoKYGBgClVLRmFzdFxMYXJhdmVsQXBpRG9jc1xTZXJ2aWNlUHJvdmlkZXI6OmNsYXNzLApgYGAKCiMjIFVzYWdlCgpEb2N1bWVudGF0aW9uIGlzIGdlbmVyYXRlZCBieSBzY2FubmluZyB5b3VyIHJvdXRlcyBmaWxlIGFuZCBmaW5kaW5nIGNvbnRyb2xsZXIgbWV0aG9kcyB3aXRoIHNwZWNpYWwgZW5kcG9pbnQgYXR0cmlidXRlcyB0YWdnZWQgdG8gdGhlbS4KCldoaWxzdCB5b3UgY2FuIGVhc2lseSBkZWZpbmUgeW91ciBvd24gZW5kcG9pbnQgdHlwZXMsIHRoZSBwYWNrYWdlIGNvbWVzIHdpdGggc29tZSBzZW5zaWJsZSBkZWZhdWx0cyB0aGF0IGdlbmVyYWxseSBjb25mb3JtIHRvIGxhcmF2ZWwgZGVmYXVsdHMKCiMjIyBJbmRleAoKVGhlIGluZGV4IGVuZHBvaW50IGlzIGZvciBlbmRwb2ludHMgdGhhdCByZXR1cm4gYSBwYWdpbmF0ZWQgbGlzdCwgZm9yIGV4YW1wbGU6CgpgYGBwaHAKdXNlIFVLRmFzdFxMYXJhdmVsQXBpRG9jc1xFbmRwb2ludHM7CnVzZSBBcHBcSHR0cFxSZXNvdXJjZXNcUGV0UmVzb3VyY2U7CnVzZSBBcHBcTW9kZWxzXFBldDsKCmNsYXNzIFBldENvbnRyb2xsZXIKewogICAgI1tFbmRwb2ludHNcSW5kZXgoUGV0UmVzb3VyY2U6OmNsYXNzKV0KICAgIHB1YmxpYyBmdW5jdGlvbiBpbmRleCgpCiAgICB7CiAgICAgICAgcmV0dXJuIFBldFJlc291cmNlOjpjb2xsZWN0aW9uKFBldDo6cGFnaW5hdGUoKSk7CiAgICB9Cn0KYGBgCgojIyMgQ3JlYXRlCgpUaGUgY3JlYXRlIGVuZHBvaW50IGlzIGZvciBlbmRwb2ludHMgdGhhdCBjcmVhdGUgYSBuZXcgcmVzb3VyY2UuCgpgYGBwaHAKdXNlIFVLRmFzdFxMYXJhdmVsQXBpRG9jc1xFbmRwb2ludHM7CnVzZSBBcHBcSHR0cFxSZXNvdXJjZXNcUGV0UmVzb3VyY2U7CnVzZSBBcHBcTW9kZWxzXFBldDsKCmNsYXNzIFBldENvbnRyb2xsZXIKewogICAgI1tFbmRwb2ludHNcQ3JlYXRlKFBldFJlc291cmNlOjpjbGFzcykKICAgIHB1YmxpYyBmdW5jdGlvbiBzdG9yZSgpCiAgICB7CiAgICB9Cn0KYGBgCgojIyMgU2hvdwoKVGhlIHNob3cgZW5kcG9pbnQgc2hvd3MgYW4gaW5kaXZpZHVhbCByZXNvdXJjZQoKYGBgcGhwCnVzZSBVS0Zhc3RcTGFyYXZlbEFwaURvY3NcRW5kcG9pbnRzOwp1c2UgQXBwXEh0dHBcUmVzb3VyY2VzXFBldFJlc291cmNlOwp1c2UgQXBwXE1vZGVsc1xQZXQ7CgpjbGFzcyBQZXRDb250cm9sbGVyCnsKICAgICNbRW5kcG9pbnRzXFNob3coUGV0UmVzb3VyY2U6OmNsYXNzKQogICAgcHVibGljIGZ1bmN0aW9uIHNob3coKQogICAgewogICAgfQp9CmBgYAoKIyMjIFVwZGF0ZQoKVGhlIHVwZGF0ZSBlbmRwb2ludCB1cGRhdGVzIGEgcmVzb3VyY2UKCmBgYHBocAp1c2UgVUtGYXN0XExhcmF2ZWxBcGlEb2NzXEVuZHBvaW50czsKdXNlIEFwcFxIdHRwXFJlc291cmNlc1xQZXRSZXNvdXJjZTsKdXNlIEFwcFxNb2RlbHNcUGV0OwoKY2xhc3MgUGV0Q29udHJvbGxlcgp7CiAgICAjW0VuZHBvaW50c1xVcGRhdGUoUGV0UmVzb3VyY2U6OmNsYXNzKQogICAgcHVibGljIGZ1bmN0aW9uIHVwZGF0ZSgpCiAgICB7CiAgICB9Cn0KYGBgCgojIyMgRGVzdHJveQoKVGhlIGRlc3Ryb3kgZW5kcG9pbnQgZGVsZXRlcyBhIHJlc291cmNlCgpgYGBwaHAKdXNlIFVLRmFzdFxMYXJhdmVsQXBpRG9jc1xFbmRwb2ludHM7CgpjbGFzcyBQZXRDb250cm9sbGVyCnsKICAgICNbRW5kcG9pbnRzXERlc3Ryb3ldCiAgICBwdWJsaWMgZnVuY3Rpb24gZGVzdHJveSgpCiAgICB7CiAgICB9Cn0KYGBgCgojIyBDdXN0b21pc2luZyBSZXF1ZXN0IGFuZCBSZXNwb25zZSBTdHJ1Y3R1cmUKCllvdXIgQVBJIGxpa2VseSBoYXMgaXRzIG93biBmb3JtYXQgZGlmZmVyZW50IHRvIHRoZSBkZWZhdWx0cyBwcm92aWRlZCBieSB0aGlzIHBhY2thZ2UuIEJ1dCByZWRlZmluaW5nIHRoZXNlIGlzIG5vdCBkaWZmaWN1bHQuCgpIZXJlJ3MgYW4gZXhhbXBsZSBvZiBhIGN1c3RvbSBpbmRleCBlbmRwb2ludDoKCmBgYHBocApuYW1lc3BhY2UgQXBwXERvY3M7Cgp1c2UgVUtGYXN0XExhcmF2ZWxBcGlEb2NzXEVuZHBvaW50Owp1c2UgQXR0cmlidXRlOwoKI1tBdHRyaWJ1dGVdCmNsYXNzIEluZGV4IGV4dGVuZHMgRW5kcG9pbnQKewogICAgcHVibGljIGZ1bmN0aW9uIF9fY29uc3RydWN0KHByb3RlY3RlZCAkcmVzb3VyY2UpCiAgICB7fQoKICAgIHB1YmxpYyBmdW5jdGlvbiByZXNwb25zZSgpCiAgICB7CiAgICAgICAgcmV0dXJuIFsKICAgICAgICAgICAgJ2RhdGEnID0+IFskdGhpcy0+cmVmKCR0aGlzLT5yZXNvdXJjZSldLAogICAgICAgICAgICAnbWV0YScgPT4gWwogICAgICAgICAgICAgICAgJ3Blcl9wYWdlJyA9PiAxNSwKICAgICAgICAgICAgICAgICd0b3RhbF9wYWdlcycgPT4gMTAsCiAgICAgICAgICAgICAgICAnY3VycmVudF9wYWdlJyA9PiAxLAogICAgICAgICAgICBdLAogICAgICAgIF07CiAgICB9Cn0KYGBgCgpFbmRwb2ludCBjbGFzc2VzIGNhbiBkZWZpbmUgdHdvIG1ldGhvZHM6IGByZXF1ZXN0YCBhbmQgYHJlc3BvbnNlYCBlYWNoIHJldHVybiBhIFBIUCBhcnJheSBvdXRsaW5pbmcgdGhlIHJlcXVlc3Qgc3RydWN0dXJlLgoKQ2FsbHMgdG8gYCR0aGlzLT5yZWZgIGNhbiBiZSBwYXNzZWQgYSBjbGFzcyBwYXRoIHRvIGEgYW55IGNsYXNzIHdpdGggdGhlICNbUmVzb3VyY2VdIGF0dHJpYnV0ZSBvbiBpdC4KCkZvciBtb3JlIGV4YW1wbGVzIGxvb2sgaW5zaWRlIHRoZSBgc3JjL0VuZHBvaW50c2AgZm9sZGVyCgojIyBDb250cmlidXRpbmcKCldlIHdlbGNvbWUgY29udHJpYnV0aW9ucyB0byB0aGlzIHBhY2thZ2UgdGhhdCB3aWxsIGJlIGJlbmVmaWNpYWwgdG8gdGhlIGNvbW11bml0eS4KCllvdSBjYW4gcmVhY2ggb3V0IHRvIG91ciBvcGVuLXNvdXJjZSB0ZWFtIHZpYSAqKm9wZW4tc291cmNlQHVrZmFzdC5jby51ayoqIHdobyB3aWxsIGdldCBiYWNrIHRvIHlvdSBhcyBzb29uIGFzIHBvc3NpYmxlLgoKUGxlYXNlIHJlZmVyIHRvIG91ciBbQ09OVFJJQlVUSU5HXShDT05UUklCVVRJTkcubWQpIGZpbGUgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgoKIyMgU2VjdXJpdHkKCklmIHlvdSB0aGluayB5b3UgaGF2ZSBpZGVudGlmaWVkIGEgc2VjdXJpdHkgdnVsbmVyYWJpbGl0eSwgcGxlYXNlIGNvbnRhY3Qgb3VyIHRlYW0gdmlhICoqc2VjdXJpdHlAdWtmYXN0LmNvLnVrKiogd2hvIHdpbGwgZ2V0IGJhY2sgdG8geW91IGFzIHNvb24gYXMgcG9zc2libGUsIHJhdGhlciB0aGFuIHVzaW5nIHRoZSBpc3N1ZSB0cmFja2VyLgoKCiMjIExpY2VuY2UKClRoaXMgcHJvamVjdCBpcyBsaWNlbmNlZCB1bmRlciB0aGUgTUlUIExpY2VuY2UgKE1JVCkuIFBsZWFzZSBzZWUgdGhlIFtMaWNlbmNlXShMSUNFTkNFKSBmaWxlIGZvciBtb3JlIGluZm9ybWF0aW9uLgo= readmeEtag: '"5308cd3d56f09109621c48fa5e43698862a2cc7a"' readmeLastModified: Mon, 04 Jul 2022 06:17:07 GMT repositoryId: 464829158 description: >- A package to automatically generate openapi v3 documentation for your laravel rest apis created: '2022-03-01T09:36:55Z' updated: '2022-07-04T06:15:01Z' language: PHP archived: false stars: 0 watchers: 13 forks: 0 owner: ans-group logo: https://avatars.githubusercontent.com/u/46777488?v=4 repoEtag: '"8e2b88e708b84184675da792d012cce0b6352cc024c7c91cd17e8a224cac88f9"' repoLastModified: Mon, 04 Jul 2022 06:15:01 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/ukfast/laravel-api-docs - source: openapi3 tags repository: https://github.com/lukearranz/challenge-alkemy v3: true id: 5a0746cd17c530f3859600be1dde6a47 repositoryMetadata: base64Readme: >- IyDinKggUGVsaWN1bGFzIENoYWxsZW5nZSAtIFJlc3RBcGkg4pyoCgpBcGkgUmVzdCBmb3IgcHJhY3RpY2UgcHVycG9zZXMKCiMjIEphdmEgKyBTcHJpbmdCb290CgotIE1hdmVuCi0gSGliZXJuYXRlCi0gTXlTcWwKLSBTcHJpbmdTZWN1cml0eQotIEpXVCBmb3Igc2VjdXJpdHkKLSBMb21ib2sKLSBKVW5pdAotIFN3YWdnZXIgJiBPcGVuQXBpIDMuMAoKIyMgSW5zdGFsbGF0aW9uCgpNYXZlbjoKYGBgYmFzaAptdm4gY2xlYW4gaW5zdGFsbApgYGAKCiMjIFVzYWdlCmBgYGphdmEKCi8vIEFwaSBkb2N1bWVudGF0aW9uCnd3dy5sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwjLwogICAgICAgIAovLyBEZWZhdWx0IFVzZXJzOgpVc2VyOiAgIHVzZXJuYW1lIC0gcGFzc3dvcmQKYGBgCgojIyBBdXRob3IKCi0gaHR0cHM6Ly93d3cubHVjYXNhcnJhbnouY29tLmFyLwotIGh0dHBzOi8vd3d3LmxpbmtlZGluLmNvbS9pbi9sdWNhcy1hcnJhbnotZ2FyY2lhLwoK readmeEtag: '"f42177fe8ac3f3bfc9e4a3163df14721cc0168b4"' readmeLastModified: Wed, 28 Dec 2022 20:28:36 GMT repositoryId: 531430694 description: >- Api Rest desarrollada con el fin de continuar aprendiendo el entorno Java+SpringBoot created: '2022-09-01T08:32:01Z' updated: '2022-12-15T12:04:31Z' language: Java archived: false stars: 0 watchers: 2 forks: 0 owner: lukearranz logo: https://avatars.githubusercontent.com/u/83666817?v=4 repoEtag: '"0cf10585c9a04b8bb3deb3c54a0e89f3bf1f885ffee19f1848b6e49e3b0922ce"' repoLastModified: Thu, 15 Dec 2022 12:04:31 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/ynnckth/pizza-time v3: true id: ff9222532fea8d98b39f9133bc9694fc repositoryMetadata: base64Readme: >- IyBQaXp6YSBUaW1lIQoKIVtDSV0oaHR0cHM6Ly9naXRodWIuY29tL3lubmNrdGgvcGl6emEtdGltZS9hY3Rpb25zL3dvcmtmbG93cy9tYWluLnltbC9iYWRnZS5zdmcpCgoqRXhwbG9yaW5nIHN0YXRlLW9mLXRoZS1hcnQgc3RhdGUgbWFuYWdlbWVudCBhbmQgdGVzdGluZyBwcmFjdGljZXMgZm9yIFJlYWN0SlMgaW4gMjAyMioKClRoaXMgcHJvamVjdCB3YXMgYm9vdHN0cmFwcGVkIHdpdGggW0NyZWF0ZSBSZWFjdCBBcHBdKGh0dHBzOi8vZ2l0aHViLmNvbS9mYWNlYm9vay9jcmVhdGUtcmVhY3QtYXBwKSB1c2luZyB0aGUgVHlwZVNjcmlwdCB0ZW1wbGF0ZS4KCiMjIFByZS1yZXF1aXNpdGVzCgotIE9ubHkgdGVzdGVkIG9uIGEgbWFjIGRldmVsb3BtZW50IG1hY2hpbmUgKHNvbWUgc2NyaXB0cyBtaWdodCBuZWVkIGFkYXB0aW9uIHRvIHdvcmsgb24gd2luZG93cykKLSBgTm9kZWAgYW5kIG5wbSBMVFMgbXVzdCBiZSBpbnN0YWxsZWQgb24gZGV2IG1hY2hpbmUKLSBgRG9ja2VyYCBtdXN0IGJlIGluc3RhbGxlZCBvbiBkZXYgbWFjaGluZQoKIyMgQXZhaWxhYmxlIHNjcmlwdHMKCmBgYHNoZWxsCiMgR2VuZXJhdGUgY2xpZW50IGNvZGUgZm9yIG9yZGVyLXNlcnZpY2UgQVBJIChyZXF1aXJlZCBmb3IgZnJvbnRlbmQpOgouL29wZW5hcGktY29kZWdlbi5zaAoKIyBJbnN0YWxsIGRlcGVuZGVuY2llczoKbnBtIGluc3RhbGwKCiMgUnVuIGluIGRldmVsb3BtZW50IG1vZGUgKG9uIHBvcnQgMzAwMCk6Cm5wbSBzdGFydAoKIyBSdW4gdW5pdCBhbmQgaW50ZWdyYXRpb24gdGVzdHM6IApucG0gdGVzdAoKIyBSdW4gZTJlIHRlc3RzOgpucG0gcnVuIGN5cHJlc3M6b3BlbgpgYGAKCiMjIFRvcGljcyBjb3ZlcmVkIGluIHRoaXMgcHJvamVjdAoKLSBDUkEgdXNpbmcgVHlwZVNjcmlwdCB0ZW1wbGF0ZQotIFJvdXRpbmcgdXNpbmcgcmVhY3Qtcm91dGVyLWRvbSBhbmQgQnJvd3NlclJvdXRlcgotIFN0YXRlIG1hbmFnZW1lbnQgdXNpbmcgUmVkdXggVG9vbGtpdAotIFJlZHV4IG1pZGRsZXdhcmUgdXNpbmcgVGh1bmsgKGRlZmF1bHQgZnJvbSBSZWR1eCBUb29sa2l0KQotIEZldGNoaW5nIGRhdGEgdXNpbmcgUlRLIFF1ZXJ5Ci0gUmVhY3QgZXJyb3IgYm91bmRhcnkKLSBGb3JtIHZhbGlkYXRpb24gdXNpbmcgRm9ybWlrCi0gVW5pdCBhbmQgaW50ZWdyYXRpb24gdGVzdGluZyBpbmNsdWRpbmcgYSByZWFsIHN0b3JlIHVzaW5nIHJlYWN0LXRlc3RpbmctbGlicmFyeSBhbmQgbXN3IGZvciBtb2NraW5nIG5ldHdvcmsgcmVxdWVzdHMKLSBFMmUgdGVzdGluZyB1c2luZyBjeXByZXNzIChzdHViYmVkIG5ldHdvcmsgY2FsbHMpCi0gQ2xpZW50IGNvZGUgZ2VuZXJhdGlvbiBiYXNlZCBvbiBPcGVuQVBJIHNwZWNpZmljYXRpb24KLSBDSS9DRCBwaXBlbGluZSB3aXRoIEdpdEh1YiBQYWdlcywgaW5jbHVkaW5nIHRlc3RpbmcgYW5kIGRlcGxveW1lbnQKLSBFbWJlZCBidWlsZCB2ZXJzaW9uIGluZm9ybWF0aW9uIGludG8gZGVwbG95ZWQgZnJvbnRlbmQgYXJ0aWZhY3QgYmFzZWQgb24gY29tbWl0IGhhc2ggYW5kIGJ1aWxkIG51bWJlcgotIEVudmlyb25tZW50IGUyZSB0ZXN0aW5nIGZvciBmdWxsIGludGVncmF0aW9uIHRlc3RzIG9mIGZyb250LSBhbmQgYmFja2VuZAotIFZpc3VhbCByZWdyZXNzaW9uIHRlc3Rpbmcgd2l0aCBDeXByZXNzIHNuYXBzaG90IHBsdWdpbgoKIyMgTGlua3MgYW5kIHJlbGV2YW50IGRvY3VtZW50YXRpb24KLSBbUmVkdXggVG9vbGtpdCBUeXBlU2NyaXB0IHF1aWNrIHN0YXJ0XShodHRwczovL3JlZHV4LXRvb2xraXQuanMub3JnL3R1dG9yaWFscy90eXBlc2NyaXB0KQotIFtPZmZpY2lhbCBDUkEgdGVtcGxhdGUgcHJvamVjdCB1c2luZyByZWR1eCB0b29sa2l0IGFuZCB0eXBlc2NyaXB0XShodHRwczovL2dpdGh1Yi5jb20vcmVkdXhqcy9jcmEtdGVtcGxhdGUtcmVkdXgtdHlwZXNjcmlwdCkKLSBbUmVkdXggYXN5bmMgbG9naWNdKGh0dHBzOi8vcmVkdXguanMub3JnL3R1dG9yaWFscy9lc3NlbnRpYWxzL3BhcnQtNS1hc3luYy1sb2dpYykKLSBbR3VpZGVsaW5lcyBmb3Igd3JpdGluZyB0ZXN0cyBmb3IgUmVkdXhdKGh0dHBzOi8vcmVkdXguanMub3JnL3VzYWdlL3dyaXRpbmctdGVzdHMpCi0gW1JlZHV4IFRvb2xraXQgUXVlcnkgT3ZlcnZpZXddKGh0dHBzOi8vcmVkdXgtdG9vbGtpdC5qcy5vcmcvcnRrLXF1ZXJ5L292ZXJ2aWV3KQoKLS0tCgojIyBDb2RlIEdlbmVyYXRpb24KClRoaXMgcHJvamVjdCB1c2VzIFN3YWdnZXIgQ29kZUdlbiBhbmQgT3BlbkFQSSB0byBnZW5lcmF0ZSBwYXJ0cyBvZiB0aGUgY29uc3VtZXIgY29kZSBmb3IgdGhlIG9yZGVyIHNlcnZpY2UgQVBJLgoKIVtDb2RlIGdlbmVyYXRpb24gcHJvY2Vzc10oLi9kb2Mvb3BlbmFwaS1jb2RlZ2VuLnBuZykKCiMjIyBQcmVyZXF1aXNpdGVzCi0gVGhlIHByb3ZpZGVkIHNjcmlwdCBleHBlY3RzIHlvdSB0byBoYXZlIGBkb2NrZXJgIGluc3RhbGxlZCBvbiB5b3VyIG1hY2hpbmUgdG8gYmUgYWJsZSB0byBleGVjdXRlIHRoZSBjb2RlIGdlbmVyYXRpb24KCiMjIyBHZW5lcmF0aW5nIENsaWVudCBDb2RlCgpgYGBzaGVsbAojIEdlbmVyYXRlIHRoZSBUeXBlU2NyaXB0IGNsaWVudCBjb2RlOgouL29wZW5hcGktY29kZWdlbi5zaAojIFRoaXMgd2lsbCBjcmVhdGUgYSBmb2xkZXIgYGdlbmVyYXRlZGAgY29udGFpbmluZyB0aGUgZ2VuZXJhdGVkIGNsaWVudCBjb2RlCmBgYAoKIyMjIE9wZW4gUG9pbnRzCgotIFdoZXJlIHRvIHBsYWNlIHRoZSBnZW5lcmF0ZWQgY29kZSBpZGVhbGx5PwotIEhvdyB0byBzcGxpdCB0aGUgZ2VuZXJhdGVkIEFQSXMgaW50byBzZXBhcmF0ZSBmaWxlcyAoZS5nLiBieSBjb250cm9sbGVyKT8KLSBDdXJyZW50bHkgb25seSBjb25zdW1pbmcgZ2VuZXJhdGVkIG1vZGVscyAoaW50ZXJmYWNlcykgLT4gY2hlY2sgd2hhdCBlbHNlIGNhbiBiZSB1c2VkIGZyb20gdGhlIGdlbmVyYXRlZCBjb2RlCi0gRGlmZmVyZW5jZXMgdG8gW09wZW5BUEkgVG9vbHMgLyBvcGVuYXBpLWdlbmVyYXRvcl0oaHR0cHM6Ly9naXRodWIuY29tL09wZW5BUElUb29scy9vcGVuYXBpLWdlbmVyYXRvcik/CgojIyMgTGlua3MgYW5kIERvY3VtZW50YXRpb24KCi0gW1N3YWdnZXIgQ29kZUdlbiBHaXRodWIgUmVwb10oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItY29kZWdlbikKLSBbU3dhZ2dlciBFZGl0b3IgLSBPbmxpbmVdKGh0dHBzOi8vZWRpdG9yLnN3YWdnZXIuaW8vKQotIFtTd2FnZ2VyIFBldCBTdG9yZSBHaXRodWIgUmVwbyAocmVmZXJlbmNlIGltcGxlbWVudGF0aW9uKV0oaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItcGV0c3RvcmUpCi0gW25wbSBwYWNrYWdlIHRoYXQgd3JhcHMgU3dhZ2dlciBDb2RlR2VuXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9Ab3BlbmFwaXRvb2xzL29wZW5hcGktZ2VuZXJhdG9yLWNsaSkKCi0tLQoKIyMgVGVzdCBTdHJhdGVneQoKVGhpcyBhcHBsaWNhdGlvbiBkZWZpbmVzIHRoZSBmb2xsb3dpbmcgdGVzdCBib3VuZGFyaWVzOiAKCkZyb250ZW5kOiAKLSBVbml0IGFuZCBpbnRlZ3JhdGlvbiB0ZXN0cyB0aGF0IGNvdmVyIHRoZSBjb2xsYWJvcmF0aW9uIG9mIG9uZSBvciBtb3JlIGNvbXBvbmVudHMgaW4gY29sbGFib3JhdGlvbiB3aXRoIFJlZHV4IChpbmNsdWRpbmcgbWlkZGxld2FyZSkuIE1vY2tpbmcgaGFwcGVucyBvbiB0aGUgbmV0d29yayBsYXllci4KLSBFMmUgdGVzdHMgdXNpbmcgQ3lwcmVzcy4gVGhlc2UgdGVzdHMgY292ZXIgdGhlIGZyb250ZW5kIGUyZSBhbmQgcnVuIGFnYWluc3QgYSBmdWxseSBydW5uaW5nIGZyb250ZW5kLiBOZXR3b3JrIHJlcXVlc3RzIGFyZSBpbnRlcmNlcHRlZCBhbmQgc3R1YmJlZC4KCkJhY2tlbmQ6Ci0gVGJkCgpFMmUgZW52aXJvbm1lbnQgdGVzdHM6IAotIFRoZXNlIHRlc3RzIHJ1biBhZ2FpbnN0IGEgdGFyZ2V0IGVudmlyb25tZW50IChlLmcuIHRlc3QgZW52aXJvbm1lbnQpIGFuZCBjb3ZlciB0aGUgaW50ZWdyYXRpb24gb2YgZnJvbnQtIGFuZCBiYWNrZW5kLgotIFRoZXkgc2hvdWxkIHJ1biBhdXRvbWF0aWNhbGx5IHByaW9yIHRvIGEgcmVsZWFzZS4KCiFbVGVzdGluZyBib3VuZGFyaWVzXSguL2RvYy90ZXN0aW5nLWJvdW5kYXJpZXMucG5nKQoKLS0tCgoKIyMgUnVudGltZSBFbnZpcm9ubWVudHMKClRoZXJlIGlzIGN1cnJlbnRseSBvbmx5IG9uZSB0YXJnZXQgZW52aXJvbm1lbnQgdGhhdCBpcyBjb25maWd1cmVkIGluIHRoZSBDRCBwaXBlbGluZSBhcyBmb2xsb3dzOgoKVGhlIGJhY2tlbmQgaXMgY3VycmVudGx5IGRlcGxveWVkIHRvIEhlcm9rdSAoZnJlZSB0aWVyKTogCj4gaHR0cHM6Ly9waXp6YS10aW1lLWJhY2tlbmQuaGVyb2t1YXBwLmNvbS9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKCk1hbmFnZSB0aHJvdWdoIFtIZXJva3UgZGFzaGJvYXJkXShodHRwczovL2Rhc2hib2FyZC5oZXJva3UuY29tL2FwcHMvcGl6emEtdGltZS1iYWNrZW5kKS4KClRoZSBmcm9udGVuZCBpcyBkZXBsb3llZCBhcyBhIHN0YXRpYyBzaXRlIHRvIFZlcmNlbDogCj4gaHR0cHM6Ly9mcm9udGVuZC15bm5ja3RoLnZlcmNlbC5hcHAvCgpNYW5hZ2UgdGhyb3VnaCBbVmVyY2VsIGRhc2hib2FyZF0oaHR0cHM6Ly92ZXJjZWwuY29tL2Rhc2hib2FyZCkKCioqVGVjaCBkZWJ0KiogCi0gSXQgbG9va3MgbGlrZSBWZXJjZWwgb25seSBhbGxvd3Mgb3ZlcnJpZGluZyBvbmUgc2luZ2xlIGJ1aWxkIGNvbW1hbmQuIElmIHdlIGhhdmUgbXVsdGlwbGUgZW52aXJvbm1lbnRzLCB0aGUgY3VycmVudCBhcHByb2FjaCBvZiBvdmVycmlkaW5nIHRoZSBidWlsZCBjb21tYW5kIGluIGB2ZXJjZWwuanNvbmAgaXMgbm90IHBvc3NpYmxlLgotIFRoZSBjdXJyZW50IGZyZWUgc2VydmljZXMgY29tZSB3aXRoIHRoZWlyIGxpbWl0YXRpb25zLiBDb25zaWRlciBkZXBsb3lpbmcgdG8gYSBjbG91ZCBlbnZpcm9ubWVudCBpbnN0ZWFkLgo= readmeEtag: '"8106c5447c6b542a8741e56fe7998ad3a511aec9"' readmeLastModified: Mon, 20 Mar 2023 14:00:23 GMT repositoryId: 530479305 description: >- Exploring and implementing state-of-the-art practices and tech when using ReactJS created: '2022-08-30T03:14:01Z' updated: '2022-09-21T02:32:16Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: ynnckth logo: https://avatars.githubusercontent.com/u/5628119?v=4 repoEtag: '"f53b333a54d534a2817c1cf9c0efe5e91923a9acae4f9c1892145f3b1158df89"' repoLastModified: Wed, 21 Sep 2022 02:32:16 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/ubhat95/usercrud v3: true id: c8168af1e613d9bbfdabba07f9092b60 repositoryMetadata: base64Readme: >- IyBVc2VyQ3J1ZAoKU3RhcnRlciBTcHJpbmdib290IHByb2plY3QuIFJ1bnMgb24gZG9ja2VyIApUaGlzIHByb2plY3QgaW1wbGVtZW50cyBSRVNUIEFQSXMgdXNpbmcgdGhlIE1WQyBtb2RlbC4gIGNvbnRyb2xsZXItc2VydmljZS1kYW87IGNvbm5lY3RzIHRvIE15U3FsIERiLiAKSGFzIGltcGxlbWVudGVkIEpQQSBNZXRob2RzLCBuYXRpdmVRdWVyaWVzIHRvIHJ1biBxdWVyaWVzLiBMb2dnaW5nLCBhY3R1YXRvciwgYW5kIGV4Y2VwdGlvbiBoYW5kbGluZyBhcmUgaW4gcGxhY2UuIAoKVXNlciByZWdpc3RyYXRpb247IG9uZSB1c2VyIGNhbiBoYXZlIG1hbnkgZGlmZmVyZW50IGFjY291bnRzLiAKQnVzaW5lc3MgTG9naWMgbmVlZHMgd29yazsgZm9jdXNzaW5nIG9uIGRpZmZlcmVudCBhc3BlY3RzIG9mIHNwcmluZyBmb3Igbm93LiAK readmeEtag: '"cd47996c2e65d45d8d8bc7658a2683ab072c3afe"' readmeLastModified: Tue, 14 Nov 2023 18:39:45 GMT repositoryId: 713455048 description: 'Starter SpringBoot project. ' created: '2023-11-02T14:55:30Z' updated: '2023-11-14T18:37:17Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: ubhat95 logo: https://avatars.githubusercontent.com/u/53697553?v=4 repoEtag: '"259cc01deaeb86ab2782ed9ef57ded47a8a00d3229cdbf44a566c06e17536e20"' repoLastModified: Tue, 14 Nov 2023 18:37:17 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/neofoxxo/weather-api-mongodb v3: true id: edcffbe0cba3edc3a3f23c00c793b14d repositoryMetadata: base64Readme: >- IyBXZWF0aGVyIFNlbnNvciBBUEkKClRoaXMgd2VhdGhlciBzZW5zb3IgQVBJIGlzIGEgc2VjdXJlIFJFU1QgQVBJIGJ1aWx0IHdpdGggRXhwcmVzcy5qcywgTW9uZ29EQiwgYW5kIE1vbmdvb3NlLiBJdCBmZWF0dXJlcyBhIGZ1bGx5IGZ1bmN0aW9uYWwgQVBJIHdoZXJlIHVzZXJzIGNhbiBxdWVyeSB3ZWF0aGVyIGluZm9ybWF0aW9uIHN0b3JlZCBpbiBhIE1vbmdvREIgZGF0YWJhc2UuPGJyPgo8YnI+VGhpcyBBUEkgd2FzIGJ1aWx0IHVzaW5nIHRoZSBPcGVuQVBJIDMgc3RhbmRhcmQgYW5kIGl0cyBkb2N1bWVudGF0aW9uIGNhbiBiZSB2aWV3ZWQgYmVsb3c6Cjxicj48YSBocmVmPSJodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzLWRvY3Mvd2ViRGF0YUNsdXN0ZXIvV2VhdGhlckFQSS8xLjAuMCMvIj5WaWV3IEFQSSBEb2N1bWVudGF0aW9uPC9hPjxicj4KCiMjIEZlYXR1cmVzCiog8J+boe+4jyBTZWN1cmUgYXV0aGVudGljYXRpb24gd2l0aCBKV1QncwoqIPCflJIgQ3JlYXRlIGFuZCB1c2UgYWNjb3VudHMgd2l0aCBmdWxsIHBhc3N3b3JkIGVuY3J5dGlvbgoqIPCfmqogUm9sZS1iYXNlZCBhY2Nlc3MgY29udHJvbCB0byBzZWN1cmUgZW5kcG9pbnRzCiog4pyFIFZhbGlkYXRpb24gdG8gcHJldmVudCB1bmV4cGVjdGVkIGRhdGEgZnJvbSBiZWluZyBpbnNlcnRlZAoqIPCfjKHvuI8gQWRkIG5ldyB3ZWF0aGVyIHJlYWRpbmdzCiog8J+RpSBBZGQgbmV3IHVzZXJzIHdpdGggc3BlY2lmaWVkIHJvbGVzCiog8J+TiiBTZXZlbiBkaWZmZXJlbnQgZW5kcG9pbnRzIHRvIHF1ZXJ5LCBtYW5pcHVsYXRlLCBhbmQgYWRkIHdlYXRoZXIgZGF0YQoqIPCflI0gRml2ZSBkaWZmZXJlbnQgZW5kcG9pbnRzIHRvIGZpbHRlciwgY3JlYXRlLCBhbmQgbWFuaXB1bGF0ZSB1c2VyIGFjY291bnRzIAoKIyMgRW5kcG9pbnRzCjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vTmVvRm94eG8vd2VhdGhlci1hcGktbW9uZ29kYi9ibG9iL2M5ZmY3YjFlNzQ5Yzg1ZDA5NzE2ZGUwYjcxZTM5OTc3MTEwMWY3MjEvaW1hZ2VzL2VuZHBvaW50cy5wbmciIGFsdD0iZW5kcG9pbnRzIiB3aWR0aD0iMTAwJSI+CgojIyBIb3cgVG8gUnVuIEl0ClRvIHJ1biB0aGlzIG9uIHlvdXIgbG9jYWwgbWFjaGluZSwgeW91IHdpbGwgbmVlZCB0byBoYXZlIE5vZGUuanMgaW5zdGFsbGVkIGFuZCBhY2Nlc3MgdG8gYSBNb25nb0RCIGRhdGFiYXNlLgoKMS4gQ2xvbmUgdGhlIHJlcG9zaXRvcnkgdG8geW91ciBsb2NhbCBtYWNoaW5lCjIuIEltcG9ydCB0aGUgZGF0YWJhc2UgZm9sZGVyIHRvIHlvdXIgTW9uZ29EQiBzZXJ2ZXIgd2l0aCBgbW9uZ29yZXN0b3JlYAozLiBDcmVhdGUgYSBgLmVudmAgZmlsZSBpbiB0aGUgcm9vdCBkaXJlY3RvcnkgYW5kIGFkZCBhIGBEQVRBQkFTRV9VUkxgIGFuZCBgQUNDRVNTX1RPS0VOX1NFQ1JFVGAKNC4gT3BlbiB5b3VyIHRlcm1pbmFsIGFuZCBydW4gYG5wbSBpbnN0YWxsYAo1LiBSdW4gdGhlIGBucG0gcnVuIHN0YXJ0YCBjb21tYW5kCjYuIEFjY2VzcyB0aGUgQVBJIG9uIHBvcnQgNTAwMCBhbmQgZW5qb3kh readmeEtag: '"eff71d63b8f559c55d1b01e94ab3093d692cc84c"' readmeLastModified: Fri, 06 Oct 2023 03:11:29 GMT repositoryId: 685361937 description: Weather REST API built with Express.js & MongoDB created: '2023-08-31T04:12:43Z' updated: '2025-04-01T01:28:52Z' language: JavaScript archived: false stars: 1 watchers: 1 forks: 0 owner: NeoFoxxo logo: https://avatars.githubusercontent.com/u/104744179?v=4 repoEtag: '"ea9ec1151281045e72378f3172b1cc31949b71fa4b4b4a0a84f70569fec8219f"' repoLastModified: Tue, 01 Apr 2025 01:28:52 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/mariama4/kvazar-tt v3: true id: 879f24a770c18a4705a679ab607098a1 repositoryMetadata: base64Readme: >- # kvazar-tt

## Описание

Данный проект является заключительной частью тестового задания для Квазар.
Суть задания является соединить воедино предыдущие модули:
- [API](https://github.com/Mariama4/kvazar-tt/tree/api)
- [UTILS](https://github.com/Mariama4/kvazar-tt/tree/utils)
- [ML](https://github.com/Mariama4/kvazar-tt/tree/ml)

А так-же добавление:
* Документации к API
* Документации к Проекту

**API реализовано на:**
* Python 3.11
* SQLite

**С использованием следующих библиотек:**
* Flask-SQLAlchemy 3.0.x
* python-dotenv
* flask-swagger-ui

**TODO:**

* Улучшить качество docstring
* Реализовать часть ML
* Добавить обработчики ошибок
* Исправить предупреждение в тестах: test_delete_user - LegacyAPIWarning
* Улучшить качество тестов, убрать повторения кода

**Test coverage:**
* Stmts: 436
* Miss: 46
* Cover: 89%

**Документация к API выполнена на swagger по стандарту openapi 3.0.0**

## Запуск проекта

Чтобы запустить проект, необходимо:

1. Скачать его:

```shell
git clone --branch=master https://github.com/Mariama4/kvazar-tt.git
```

2. Перейти в папку проекта:

```shell
cd kvazar-tt
```

3. Установить зависимости:

```shell
pip install -r requirements.txt
```

4. Запустить проект:

```shell
python app.py
```

# Использование API:

Если вы установили и запустили проект, то более подробную документацию можно увидеть тут [http://127.0.0.1:5000/api/docs/](http://127.0.0.1:5000/api/docs/)

## API App

> Version 1.0.0

API с модулем Users, который поддерживает CRUD и позволяет получить следующую информацию:
- Используя данные из таблицы User, подсчитывать количество пользователей, зарегистрированных за последние 7 дней;
- Получить топ 5 пользователей с самыми длинными именами;
- Получить процент пользователей, которые имеют адрес электронной почты, зарегистрированный в определенном домене (например, "example.com").

## Path Table

| Method | Path | Description |
| --- | --- | --- |
| GET | [/api/users/](#getapiusers) | Получение всех пользователей |
| GET | [/api/users](#getapiusers) | Получение пользователей с пагинацией |
| POST | [/api/users](#postapiusers) | Создание пользователя |
| GET | [/api/users/{id}](#getapiusersid) | Получение польльзователя по id |
| PATCH | [/api/users/{id}](#patchapiusersid) | Обновление пользователя по id |
| DELETE | [/api/users/{id}](#deleteapiusersid) | Удаление пользователя по id |
| POST | [/api/users/info](#postapiusersinfo) | Получение информации о пользователях |

## Reference Table

| Name | Path | Description |
| --- | --- | --- |
| User | [#/components/schemas/User](#componentsschemasuser) |  |

## Path Details

***

### [GET] /api/users/

- Summary  
Получение всех пользователей

#### Responses

- 200 Список пользователей

`application/json`

```json
[
  {
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
  },
  {
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
  }
]
```

- 403 Ошибка на стороне сервера

***

### [GET] /api/users

- Summary  
Получение пользователей с пагинацией

#### Parameters(Query)

```ts
page: integer
```

```ts
per_page: integer
```

#### Responses

- 200 Список пользователей

`application/json`

```json
[
  {
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
  },
  {
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
  }
]
```

- 403 Ошибка на стороне сервера

***

### [POST] /api/users

- Summary  
Создание пользователя

#### RequestBody

- application/json

```json
{
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
}
```

#### Responses

- 200 Пользователь создан

`application/json`

```json
{
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
}
```

- 403 Ошибка на стороне сервера

***

### [GET] /api/users/{id}

- Summary  
Получение пользователя по id

#### Responses

- 200 Пользователь получен

`application/json`

```json
{
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
}
```

- 403 Ошибка на стороне сервера

***

### [PATCH] /api/users/{id}

- Summary  
Обновление пользователя по id

#### RequestBody

- application/json

```json
{
    "email": string,
    "username": string
}
```

#### Responses

- 200 Пользователь получен

`application/json`

```json
{
    "email": string,
    "id": integer,
    "registration_date": datetime,
    "username": string
}
```

- 403 Ошибка на стороне сервера

***

### [DELETE] /api/users/{id}

- Summary  
Удаление пользователя по id

#### Responses

- 200 Пользователь удален

`application/json`

```json
{
  "deleted": boolean
}
```

- 403 Ошибка на стороне сервера

***

### [POST] /api/users/info

- Summary  
Получение информации о пользователях

- Description  
- Пользователи, которые зарегистрировались за последние 7 дней  
- Топ 5 пользователей, которые имеют самые длинные имена (по убыванию)  
- Процент пользователей, которые имеют адрес электронной почты, зарегистрированный в определенном домене (например, "example.com").

#### RequestBody

- application/json

```json
{
  "domain": string
}
```

#### Responses

- 200 Информация получена

`application/json`

```json
{
	"countUsersForWeek": integer,
	"percentOfDomain": {
		"domain": string,
		"percent": number
	},
	"topUsersWithLongestUsernames": [
		{
            "email": string,
            "id": integer,
            "registration_date": datetime,
            "username": string
        },
		{
            "email": string,
            "id": integer,
            "registration_date": datetime,
            "username": string
        }
	]
}
```

- 403 Ошибка на стороне сервера

## References

### #/components/schemas/User

```json
"id": integer
"username": string
"email": string
"registration_date": datetime
```
 readmeEtag: '"3dc25774993bb0ecc10128fedfe6560441c841bf"' readmeLastModified: Mon, 01 May 2023 06:24:07 GMT repositoryId: 631767888 description: Тестовое задание для "Квазар" created: '2023-04-24T02:30:25Z' updated: '2024-01-15T16:58:58Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: Mariama4 logo: https://avatars.githubusercontent.com/u/61371131?v=4 repoEtag: '"b3d516f9c0a1ae2720f46a8a1ece2a738a1c35689900ad312c5c9f1304bd80bb"' repoLastModified: Mon, 15 Jan 2024 16:58:58 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/yoshi2no/openapi-ts v3: true id: 86bb6e0c47584b3ac9228347f46ad408 repositoryMetadata: base64Readme: >- IyB5b3NoaTJuby9vcGVuYXBpLXRzCgo+IFshSU1QT1JUQU5UXQo+VGhpcyByZXBvc2l0b3J5IGlzIGNyZWF0ZWQgZm9yIHBlcnNvbmFsIGxlYXJuaW5nIGFuZCBkZXZlbG9wbWVudCBwdXJwb3Nlcy4gSXQncyBub3QgaW50ZW5kZWQgZm9yIHByb2R1Y3Rpb24gdXNlLgoKCiMjIPCfk6YgUGFja2FnZXMKCjxhIGhyZWY9Ii4vcGFja2FnZXMvb3BlbmFwaS10cyI+PGJyIC8+CvCfmLogb3BlbmFwaS10czxiciAvPgo8L2E+CkdlbmVyYXRlcyBUeXBlU2NyaXB0IHR5cGVzIGZyb20gc3RhdGljIE9wZW5BUEkgc2NoZW1hcwoKPGEgaHJlZj0iLi9wYWNrYWdlcy9vcGVuYXBpLWZldGNoIj48YnIgLz4K8J+YuiBvcGVuYXBpLWZldGNoPGJyIC8+CjwvYT4K8J+apwoKCiMjIOKchSBTdGF0dXMKCuKchSBtZWFucyBpbXBsZW1lbnRlZAoKIyMjIEJhc2ljIEZlYXR1cmVzCgp8IGZlYXR1cmUgICAgICAgICB8IGltcGwgfCBib29rIHwKfCA6LS0tOiB8IDotLS06IHwgOi0tLTogfAp8IExvY2FsIHNjaGVtYSAgICAgICAgICAgICB8ICAgfCAgIHwKfCBSZW1vdGUgc2NoZW1hICAgICAgICAgICAgIHwgICB8ICAgfAp8IE11bHRpcGxlIHNjaGVtYXMgICAgICAgICAgICAgfCAgIHwgICB8CgoKIyMjIENvbmZpZwp8IGZlYXR1cmUgICAgICAgICB8IGltcGwgfCBib29rIHwKfCA6LS0tOiB8IDotLS06IHwgOi0tLTogfAp8IFJlZG9jIGNvbmZpZyAgICAgICAgICAgICB8ICAgfCAgIHwKCiMjIyBDTEkgT3B0aW9ucwp8IGZlYXR1cmUgICAgICAgICB8IGltcGwgfCBib29rIHwKfCA6LS0tOiB8IDotLS06IHwgOi0tLTogfAp8IC0tb3V0cHV0KC1vKSBbbG9jYXRpb25dICAgICAgICAgICAgfCAgIHwgICB8CnwgLS1oZWxwICAgICAgICAgICB8ICAgfCAgIHwKfCAtLXZlcnNpb24KfCAtLXJlZG9jICAgICAgICAgICAgfCAgIHwgICB8CnwgLS1hZGRpdGlvbmFsLXByb3BlcnRpZXMgICAgICAgICAgICB8ICAgfCAgIHwKfCAtLWFscGhhYmV0aXplICAgICAgICAgICAgfCAgIHwgICB8CnwgLS1hcnJheS1sZW5ndGggICAgICAgICAgICB8ICAgfCAgIHwKfCAtLWRlZmF1bHQtbm9uLW51bGxhYmxlICAgICAgICAgICAgfCAgIHwgICB8CnwgLS1lbXB0eS1vYmplY3RzLXVua25vd24gICAgICAgICAgICB8ICAgfCAgIHwKfCAtLWVudW0gICAgICAgICAgICB8ICAgfCAgIHwKfCAtLWV4Y2x1ZGUtZGVwcmVjYXRlZCAgICAgICAgICAgIHwgICB8ICAgfAp8IC0tZXhwb3J0LXR5cGUgICAgICAgICAgICB8ICAgfCAgIHwKfCAtLWltbXV0YWJsZSAgICAgICAgICAgIHwgICB8ICAgfAp8IC0tcGF0aC1wYXJhbXMtYXMtdHlwZXMgICAgICAgICAgICB8ICAgfCAgIHwKCiMjIPCfmY8gQWNrbm93bGVkZ2VtZW50cwoKVGhpcyBsaWJyYXJ5IGlzIGluc3BpcmVkIGJ5IGFuZCBiYXNlZCBvbiB0aGUgY29uY2VwdHMgbGVhcm5lZCBmcm9tIHRoZSBbb3BlbmFwaS10eXBlc2NyaXB0XShodHRwczovL2dpdGh1Yi5jb20vZHJ3cG93L29wZW5hcGktdHlwZXNjcmlwdCkgcmVwb3NpdG9yeS4gSSBleHRlbmQgbXkgZ3JhdGl0dWRlIHRvIHRoZSBjb250cmlidXRvcnMgb2YgYG9wZW5hcGktdHlwZXNjcmlwdGAgZm9yIHRoZWlyIGludmFsdWFibGUgd29yaywgd2hpY2ggaGFzIGJlZW4gYSBzaWduaWZpY2FudCByZWZlcmVuY2UgaW4gdGhlIGRldmVsb3BtZW50IG9mIHRoaXMgcHJvamVjdC4= readmeEtag: '"54d1c5beed9326d2eb5743c8e8f01d1864fe94a8"' readmeLastModified: Sun, 28 Jan 2024 12:48:27 GMT repositoryId: 742514577 description: >- Generate TypeScript Types from OpenAPI Specs: A Personal Learning Project by @yoshi2no. created: '2024-01-12T16:44:54Z' updated: '2024-01-23T15:20:42Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: yoshi2no logo: https://avatars.githubusercontent.com/u/57059705?v=4 repoEtag: '"68feddbc81a02fda7708ed34c9512537ac931a5a31c8ff8309c431f7b96763b4"' repoLastModified: Tue, 23 Jan 2024 15:20:42 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/bryopsida/spring-boot-starter-tomcat v3: true id: f3d752f349e20cdb44043babfb975bb1 repositoryMetadata: base64Readme: >- IyBTcHJpbmcgQm9vdCBTdGFydGVyIEs4UyBUZW1wbGF0ZQoKIyMgV2hhdCBpcyB0aGlzPwoKVGhpcyBpcyBhIHRlbXBsYXRlIHJlcG9zaXRvcnkgZm9yIGtpY2tpbmcgb2ZmIGEgY2xvdWQgbmF0aXZlIHNwcmluZyBib290IGphdmEgbWljcm9zZXJ2aWNlLgoKIyMgV2hhdCBtYWtlcyB0aGlzIGNsb3VkIG5hdGl2ZT8KCkl0J3Mgc29sZWx5IGZvY3VzZWQgb24gZGVwbG95bWVudCB0byBrdWJlcm5ldGVzLCB0aGUgcHJpbWFyeSBidWlsZCBhcnRpZmFjdHMgb2YgdGhlIHJlcG9zaXRvcnkgYXJlIGEgT0NJIGltYWdlIGFuZCBhIGhlbG0gY2hhcnQuIFRoZSBoZWxtIGNoYXJ0IGlzIGR5bmFtaWNhbGx5IGdlbmVyYXRlZCB1c2luZyBqa3ViZSBhbmQgaXQncyBhc3NvY2lhdGVkIGdyYWRsZSBwbHVnaW4uCgojIyBXaGF0IG9waW5pb25zIGhhdmUgYmVlbiBhZGRlZCB0byB0aGlzPwoKLSBHcmFkbGUKLSBTcHJpbmcgQm9vdAotIFNwcmluZyBNVkMKLSBTcHJpbmcgSlBBIGFuZCBKREJDCi0gVG9tY2F0Ci0gSkt1YmUgKEt1YmVybmV0ZXMgRGVwbG95bWVudCBhbmQgRGV2ZWxvcG1lbnQgVG9vbHMpCi0gTGlxdWliYXNlCgojIyBIb3cgZG8gSSBkZXBsb3k/CgpVc2UgYG1ha2UgZGVwbG95YCwgaWYgdGhlIG5hbWVzcGFjZSBkb2VzIG5vdCBleGlzdCBhbHJlYWR5IHJ1biBgbWFrZSBjcmVhdGUtbmFtZXNwYWNlYAoKIyMgSG93IGRvIEkgcnVuIGxvY2FsbHk/CgpVc2UgYG1ha2UgcnVuYC4KCiMjIEhvdyBkbyBJIHJ1biBsb2NhbGx5IG91dHNpZGUgb2YgYSBjb250YWluZXI/CgpJZiB5b3Ugd2lzaCB0byBydW4gZGlyZWN0bHkgaW4geW91ciBJREU6CgoxLiBSdW4gYGRvY2tlci1jb21wb3NlIHVwIC1kYCB0byBzdGFydCB0aGUgcG9zdGdyZXMgZGF0YWJhc2Ugc2VydmVyLgoyLiBSdW4gYC4vZ3JhZGxldyB1cGRhdGUgYm9vdFJ1bmAsIGB1cGRhdGVgIHJ1bnMgdGhlIG1pZ3JhdGlvbnMgYW5kIGBib290UnVuYCBsYXVuY2hlcyB0aGUgc3ByaW5nIGJvb3QgYXBwbGljYXRpb24KCiMjIEF2YWlsYWJsZSBNYWtlIFRhcmdldHMKCnwgVGFyZ2V0ICAgICAgICAgICAgIHwgRGVzY3JpcHRpb24gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCAtLS0tLS0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAp8IGphciAgICAgICAgICAgICAgICB8IEJ1aWxkcyB0aGUgc3RhbmRhbG9uZSBqYXIgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgaW1hZ2UgICAgICAgICAgICAgIHwgQnVpbGRzIHRoZSBPQ0kgaW1hZ2UgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBydW4gICAgICAgICAgICAgICAgfCBSdW5zIHRoZSBPQ0kgaW1hZ2UgdXNpbmcgZG9ja2VyIGxvY2FsbHkgICAgICAgICAgICAgICAgICAgICAgfAp8IGNyZWF0ZSAgICAgICAgICAgICB8IENyZWF0ZXMgdGhlIGt1YmVybmV0ZXMgbmFtZXNwYWNlIGEgZGVwbG95IHdpbGwgZ28gdG8gICAgICAgICB8CnwgYnVpbGQgICAgICAgICAgICAgIHwgUnVucyB0aGUgZ3JhZGxlIHRhc2tzIHRvIGJ1aWxkIHRoZSBoZWxtIGNoYXJ0ICAgICAgICAgICAgICAgIHwKfCBidWlsZC1kZXBlbmRlbmNpZXMgfCBSdW5zIHRoZSBoZWxtIHRhc2tzIHRvIHB1bGwgaW4gc3ViIGNoYXJ0IGRlcGVuZGVuY2llcyAgICAgICAgfAp8IGRlcGxveSAgICAgICAgICAgICB8IERlcGxveXMgdGhlIGNoYXJ0IHRvIHlvdXIgY3VycmVudCBrdWJlcm5ldGVzIGNvbnRleHQgICAgICAgICB8CnwgdGVtcGxhdGUgICAgICAgICAgIHwgUmVuZGVycyB0aGUgY2hhcnQgdGVtcGxhdGVzIHRvIHN0YW5kYXJkIGt1YmVybmV0ZXMgbWFuaWZlc3RzIHwKfCBraWNzICAgICAgICAgICAgICAgfCBTY2FuIHRoZSBjaGFydCByZXNvdXJjZXMgd2l0aCBLSUNTICAgICAgICAgICAgICAgICAgICAgICAgICAgfAoKIyBQb3N0IFRlbXBsYXRlIFVzZSBDaGVja2xpc3QKCi0gWyBdIEZpbmQgYW5kIHJlcGxhY2UgYWxsIHJlZmVyZW5jZXMgdG8gYGlvLmdpdGh1Yi5icnlvcHNpZGFgIHdpdGggYXBwcm9wcmlhdGUgdmFsdWVzIGZvciB5b3VyIHByb2plY3QKLSBbIF0gRmluZCBhbmQgcmVwbGFjZSBhbGwgcmVmZXJlbmNlcyB0byBgc3ByaW5nLWJvb3Qtc3RhcnRlcmAgd2l0aCBhcHByb3ByaWF0ZSB2YWx1ZXMgZm9yIHlvdXIgcHJvamVjdAotIFsgXSBGaW5kIGFuZCByZXBsYWNlIGFsbCByZWZlcmVuY2VzIHRvIGBzcHJpbmctYm9vdC1zdGFydGVyLXRvbWNhdGAgd2l0aCBhcHByb3ByaWF0ZSB2YWx1ZXMgZm9yIHlvdXIgcG9qZWN0Ci0gWyBdIEZpbmQgYW5kIHJlcGxhY2UgYWxsIHJlZmVyZW5jZXMgdG8gYGJyeW9wc2lkYWAgd2l0aCBhcHByb3ByaWF0ZSB2YWx1ZXMgZm9yIHlvdXIgcHJvamVjdAotIFsgXSBBZGp1c3QgcmVub3ZhdGUuanNvbiAob3IgcmVtb3ZlKSBzZXR0aW5ncyB0byBtZWV0IHlvdXIgbmVlZHMKLSBbIF0gQWZ0ZXIgYSBzdWNjZXNzZnVsIGltYWdlIGJ1aWxkLCBjcmVhdGUgeW91ciBmaXJzdCByZWxlYXNlIHRvIHRyaWdnZXIgYSBoZWxtIHB1Ymxpc2gsIHRoaXMgaXMgbmVlZGVkIGZvciB0aGUgdXBncmFkZSB0ZXN0cyB0byBwYXNzCg== readmeEtag: '"bb074d7e6f09c699c0e6f80241e81a542e6ecd25"' readmeLastModified: Wed, 14 Aug 2024 18:07:06 GMT repositoryId: 708514637 description: >- A spring boot starter template that includes a docker image, generate helm chart using jkube, and GitHub action CI created: '2023-10-22T19:25:57Z' updated: '2026-01-31T00:33:12Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: bryopsida logo: https://avatars.githubusercontent.com/u/8363252?v=4 repoEtag: '"41f233e1b5dee205e9f4c9509b4a9eb55fa33b0ef8ec3829364964b88f6e5ff3"' repoLastModified: Sat, 31 Jan 2026 00:33:12 GMT category: Server Implementations foundInMaster: true - name: Ballerina OpenAPI tool source: Tooling repository issues source_description: >- The Ballerina OpenAPI tool enables the generation of Ballerina client and service skeleton from provided OpenAPI specifications (v2, v3) and facilitates the creation of OpenAPI v3 specifications for given Ballerina services.- Repository: https://github.com/ballerina-platform/openapi-tools link: https://ballerina.io/learn/openapi-tool/ v3_1: true v3: true v2: true sourceIssueMetadata: issueNumber: 136 author: lnash94 createdAt: '2024-02-11T07:22:54Z' updatedAt: '2024-02-11T07:22:54Z' url: https://github.com/OAI/Tooling/issues/136 status: open id: 15c33d6fbddbfa6964ba32d76a6119ea foundInMaster: true - name: UI Bakery source: Tooling repository issues source_description: >- A low-code platform to build user interfaces on top of OpenAPI data using drag and drop repository: https://github.com/uibakery/self-hosted link: https://uibakery.io/ v3_1: true v3: true v2: false sourceIssueMetadata: issueNumber: 137 author: lugovsky createdAt: '2024-02-14T15:49:49Z' updatedAt: '2024-02-14T15:49:49Z' url: https://github.com/OAI/Tooling/issues/137 status: open id: 103b16f99e39fe1b7754fa2921b3370b foundInMaster: true repositoryMetadata: base64Readme: >- PGgzIGFsaWduPSJjZW50ZXIiPgogIFVJIEJha2VyeSBpcyBhIGxvdy1jb2RlIHBsYXRmb3JtIHRvIGJ1aWxkIGFwcHMgYW5kIGF1dG9tYXRpb25zIHlvdSBuZXZlciBoYWQgdGltZSBmb3IKPC9oMz4KCjxoMyBhbGlnbj0iY2VudGVyIj4KICA8Yj48YSBocmVmPSJodHRwczovL2Nsb3VkLnVpYmFrZXJ5LmlvL2F1dGgvcmVnaXN0ZXI/dXRtX3NvdXJjZT1naXRodWIiPkdldCBTdGFydGVkPC9hPjwvYj4KICDigKIKICA8YSBocmVmPSJodHRwczovL2RvY3MudWliYWtlcnkuaW8/dXRtX3NvdXJjZT1naXRodWIiPkRvY3M8L2E+CiAg4oCiCiAgPGEgaHJlZj0iaHR0cHM6Ly9kb2NzLnVpYmFrZXJ5LmlvL3N0YXJ0ZXItZ3VpZGUvdHV0b3JpYWxzP3V0bV9zb3VyY2U9Z2l0aHViIj5UdXRvcmlhbHM8L2E+CiAg4oCiCiAgPGEgaHJlZj0iaHR0cHM6Ly9hcHAuZ2V0YmVhbWVyLmNvbS91aWJha2VyeS9lbj91dG1fc291cmNlPWdpdGh1YiI+V2hhdCdzIG5ldzwvYT4gCiAg4oCiCiAgPGEgaHJlZj0iaHR0cHM6Ly9yb2FkbWFwLnVpYmFrZXJ5LmlvP3V0bV9zb3VyY2U9Z2l0aHViIj5Sb2FkbWFwPC9hPiAKPC9oMz4KCjxhIGhyZWY9Imh0dHBzOi8vY2xvdWQudWliYWtlcnkuaW8vYXV0aC9yZWdpc3Rlci8/dXRtX3NvdXJjZT1naXRodWIiPjxpbWcgc3JjPSJhc3NldHMvaGVyby5wbmciIHdpZHRoPSIxMDAlIiBhbHQ9IlVJIEJha2VyeSAtIEludGVybmFsIHRvb2xzIGFuZCB3b3JrZmxvdyBhdXRvbWF0aW9ucyI+PC9hPgoKIyMgRGVwbG95aW5nIFVJIEJha2VyeSBvbi1wcmVtaXNlCgojIyMjIERlcGxveSBVSSBCYWtlcnkgbG9jYWxseSB0byBtYW5hZ2UgeW91ciBkYXRhIGZyb20geW91ciBwcml2YXRlIG5ldHdvcmsKCldlIHVuZGVyc3RhbmQgdGhhdCB5b3UgbWlnaHQgaGF2ZSBsb3RzIG9mIGRhdGEgYWNjZXNzaWJsZSBmcm9tIHlvdXIgcHJpdmF0ZSBuZXR3b3JrLCB0aGF04oCZcyB3aHkgeW91IGNhbiB1c2UgVUkgQmFrZXJ5IHNlbGYtaG9zdGVkIHZlcnNpb24gZm9yIHlvdXIgYmVuZWZpdC4KCk9uLXByZW1pc2UgdmVyc2lvbiBncmFudHMgeW91OgoKLSBBIHF1aWNrIHNldHVwIHByb2Nlc3MKLSBDdXN0b20gYnJhbmRpbmcKLSBDdXN0b20gZG9tYWluIGhvc3RpbmcKLSBPQXV0aDIgU1NPCi0gU0FNTC1iYXNlZCBpZGVudGl0eSBwcm92aWRlcnMKLSBEYXRhIGlzIHN0b3JlZCBzZWN1cmVseSB1bmRlciB5b3VyIG93biBWUFMKCjpoZWF2eV9jaGVja19tYXJrOiBVSSBCYWtlcnkgb24tcHJlbWlzZSB2ZXJzaW9uIGxpY2Vuc2Uga2V5IGNhbiBiZSBvYnRhaW5lZCBbaGVyZV0oaHR0cHM6Ly91aWJha2VyeS5pby9vbi1wcmVtaXNlLXVpLWJha2VyeSkKCjp3YXJuaW5nOiBJZiB5b3UgaGF2ZSBhbHJlYWR5IGluc3RhbGxlZCBVSSBCYWtlcnkgb24tcHJlbWlzZSB2ZXJzaW9uLCBmb2xsb3cgW3RoaXMgZ3VpZGVdKGh0dHBzOi8vZG9jcy51aWJha2VyeS5pby9vbi1wcmVtaXNlL3VwZGF0aW5nLW9uLXByZW1pc2UtdmVyc2lvbikgdG8gdXBkYXRlIHlvdXIgdmVyc2lvbi4KCiMjIFRhYmxlIG9mIGNvbnRlbnRzCgotIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCiAgLSBbUmVxdWlyZW1lbnRzXSgjcmVxdWlyZW1lbnRzKQogIC0gW0Jhc2ljIGluc3RhbGxhdGlvbl0oI2Jhc2ljLWluc3RhbGxhdGlvbikKICAtIFtPdGhlciBpbnN0YWxsYXRpb24gb3B0aW9uc10oI290aGVyLWluc3RhbGxhdGlvbi1vcHRpb25zKQogIAotIFtEb2N1bWVudGF0aW9uXSgjZG9jdW1lbnRhdGlvbikKCiMjIEluc3RhbGxhdGlvbgoKVGhpcyBkb2N1bWVudCBkZXNjcmliZXMgaG93IHRvIGRlcGxveSB1aS1iYWtlcnkgb24tcHJlbSB2aWEgYGluc3RhbGwuc2hgIHNjcmlwdC4KCjp3YXJuaW5nOiBUaGUgc2NyaXB0IGluc3RhbGxzIGRvY2tlciBhbmQgZG9ja2VyLWNvbXBvc2UsIHdoaWNoIG1heSB1cGdyYWRlIHNvbWUgZGVwZW5kZW5jaWVzIHVuZGVyIHRoZSBob29kLiBQbGVhc2UgYmUgYWR2aXNlZCB0aGF0IGlmIHlvdSBydW4gdGhpcyBzY3JpcHQgb24gdGhlIE9TIHVzZWQgYXMgYSBzZXJ2ZXIgZm9yIG90aGVyIGFwcGxpY2F0aW9ucywgdGhvc2UgYXBwbGljYXRpb25zIG1heSBicmVhayBkdWUgdG8gdGhhdCBwb3RlbnRpYWwgZGVwZW5kZW5jaWVzIHVwZ3JhZGUuCgojIyMgUmVxdWlyZW1lbnRzCgotIExpbnV4LWJhc2VkIE9TIChlLmcuIFVidW50dSAxOC4wNCkuCi0gTWluaW11bSAyIGB2Q1BVc2AsIDQgYEdpQmAgbWVtb3J5IGFuZCAyMCBgR2lCYCBvZiBzdG9yYWdlLgotIE11c3QgaGF2ZSBmdWxsIHJpZ2h0cyB0byB1c2UgInN1ZG8iLgotIFRoZXNlIGRvbWFpbnMgYXJlIGFjY2Vzc2libGUgZnJvbSB5b3VyIG5ldHdvcms6CiAgLSBodHRwczovL2NydWliYWtlcnlvbnByZW0ud2VzdGV1cm9wZS5kYXRhLmF6dXJlY3IuaW8KICAtIGh0dHBzOi8vY3J1aWJha2VyeW9ucHJlbS5jZW50cmFsdXMuZGF0YS5henVyZWNyLmlvCiAgLSBodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20KCiMjIyBCYXNpYyBpbnN0YWxsYXRpb24KCjEuIFJ1biB0aGlzIGNvbW1hbmQgcHJlZmVyYWJseSBmcm9tIGAvaG9tZWAgTGludXggZGlyZWN0b3J5IHRvIGRvd25sb2FkLCBpbnN0YWxsIGFuZCBsYXVuY2ggVUkgQmFrZXJ5OgoKICAgYGBgYmFzaAogICBjdXJsIC1rIC1MIC1vIGluc3RhbGwuc2ggaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3VpYmFrZXJ5L3NlbGYtaG9zdGVkL21haW4vaW5zdGFsbC5zaCAmJiBiYXNoIC4vaW5zdGFsbC5zaAogICBgYGAKCjEuIEluIHRoZSBwcm9jZXNzLCB1cG9uIHJlcXVlc3QsIGVudGVyIHRoZSBwcmV2aW91c2x5IHJlY2VpdmVkIGxpY2Vuc2UgY29kZSwgaG9zdGluZyBVUkwgYW5kIHBvcnQuCjEuIE9uY2UgdGhlIGluc3RhbGxhdGlvbiBpcyBjb21wbGV0ZWQsIG9wZW4gdGhlIGJyb3dzZXIgdXNpbmcgVVJMIGFuZCBwb3J0IHByb3ZpZGVkIGVhcmxpZXIuIEJ5IGRlZmF1bHQgaXQgaXMgW2h0dHA6Ly9sb2NhbGhvc3Q6MzAzMC9dKGh0dHA6Ly9sb2NhbGhvc3Q6MzAzMC8pLgoKKipOT1RFKio6IElmIERvY2tlciBvZiB0aGUgdmVyc2lvbiBsZXNzIHRoYW4gdGhlIHJlcXVpcmVkIChtaW5pbXVtIDIwLjEwLjExKSBpcyBhbHJlYWR5IGluc3RhbGxlZCBvbiB0aGUgc2VydmVyLCBhbmQvb3IgRG9ja2VyIENvbXBvc2UgKG1pbmltdW0gMS4yOS4yKSwgdGhlIHNjcmlwdCB3aWxsIGJlIHN0b3BwZWQuIFlvdSBuZWVkIHRvIHVwZGF0ZSB0aGUgdmVyc2lvbnMgb2YgY29tcG9uZW50cyBtYW51YWxseSBhbmQgcnVuIHRoZSBzY3JpcHQgYWdhaW4uCgojIyMgT3RoZXIgaW5zdGFsbGF0aW9uIG9wdGlvbnMKRm9yIGFkZGl0aW9uYWwgaW5zdGFsbGF0aW9uIGluc3RydWN0aW9ucywgc3VjaCBhcyB0aG9zZSBmb3IgQXp1cmUsIEFXUywgR0NQLCBLdWJlcm5ldGVzLCBhbmQgbW9yZSwgcGxlYXNlIHZpc2l0IHRoZSBbZG9jdW1lbnRhdGlvbiB3ZWJzaXRlXShodHRwczovL2RvY3MudWliYWtlcnkuaW8vb24tcHJlbWlzZS9pbnN0YWxsaW5nLW9uLXByZW1pc2UtdmVyc2lvbikuCgojIyBEb2N1bWVudGF0aW9uCgpGb3IgaW5zdHJ1Y3Rpb25zIG9uIGluc3RhbGxpbmcsIHVwZGF0aW5nLCBhbmQgbWFuYWdpbmcgdGhlIG9uLXByZW1pc2UgaW5zdGFuY2UsIHBsZWFzZSByZWZlciB0byB0aGUgW2RvY3VtZW50YXRpb24gd2Vic2l0ZV0oaHR0cHM6Ly9kb2NzLnVpYmFrZXJ5LmlvL29uLXByZW1pc2UvdWktYmFrZXJ5LW9uLXByZW1pc2UpLgoKCg== readmeEtag: '"669f2bb2c39617f9283ebb516d73a534c9de316a"' readmeLastModified: Thu, 03 Jul 2025 11:14:20 GMT repositoryId: 507816571 description: Internal tools and workflow automations created: '2022-06-27T08:01:30Z' updated: '2026-02-03T07:00:15Z' language: Shell archived: false stars: 278 watchers: 6 forks: 28 owner: uibakery logo: https://avatars.githubusercontent.com/u/92528127?v=4 repoEtag: '"fd91396bea2b27e9bba18a3899512626e3c2c1d81db05f31ee49fe9c8cd511e6"' repoLastModified: Tue, 03 Feb 2026 07:00:15 GMT category: Server Implementations - source: openapi3 tags repository: https://github.com/intellifi-nl/brain-rest-api-spec v3: true repositoryMetadata: base64Readme: >- IyBCcmFpbiBSRVNUIEFQSSBzcGVjaWZpY2F0aW9uCgpUaGlzIHJlcG9zaXRvcnkgaG9sZHMgdGhlIEJyYWluIFJFU1QgQVBJIHJlZmVyZW5jZS4KClRoZSBCcmFpbiBBUEkgcmVmZXJlbmNlIHVzZXMgdGhlIFtPcGVuQVBJXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcvKSBzdGFuZGFyZCBhbmQgYSBudW1iZXIgb2YgZG9jdW1lbnQgZ2VuZXJhdG9ycyB0byB2aXN1YWxpemUgdGhlIEFQSSBTcGVjaWZpY2F0aW9uLgoKVGhlIEJyYWluIEFQSSBzcGVjIGNhbiBiZSBmb3VuZCBpbiB0aGUgIFtgb3BlbmFwaS55bWxgXSguL29wZW5hcGkueW1sKSBmaWxlLgoKVGhlIGRvY3VtZW50YXRpb24gaXMgaG9zdGVkIHVzaW5nIFtHaXRodWIgcGFnZXNdKGh0dHBzOi8vcGFnZXMuZ2l0aHViLmNvbS8pIGF0IFtpbnRlbGxpZmktbmwuZ2l0aHViLmlvL2JyYWluLXJlc3QtYXBpLXNwZWNdKGh0dHBzOi8vaW50ZWxsaWZpLW5sLmdpdGh1Yi5pby9icmFpbi1yZXN0LWFwaS1zcGVjLykuCgpBbGwgZG9jdW1lbnRhdGlvbiBpcyBhdmFpbGFibGUgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBbTUlUIExpY2Vuc2VdKC4vTElDRU5TRSkuCg== readmeEtag: '"7fa7261b24262d80b425a8c3ddbb8c1bf26e35c8"' readmeLastModified: Thu, 11 Jul 2024 12:07:18 GMT repositoryId: 149280093 description: Intellifi Brain REST API spec created: '2018-09-18T11:51:24Z' updated: '2024-11-29T13:13:41Z' language: null archived: false stars: 0 watchers: 4 forks: 0 owner: intellifi-nl logo: https://avatars.githubusercontent.com/u/9928538?v=4 license: MIT repoEtag: '"1c29da561263c93fe418b96317c447d786b388b40282bfcfaed8ac5f5a9de69d"' repoLastModified: Fri, 29 Nov 2024 13:13:41 GMT foundInMaster: true category: - Testing - Parsers id: c5bb244b2a8497d8a4a509ba783c5b59 - source: openapi3 tags repository: https://github.com/ragnar-oock/setlist v3: true repositoryMetadata: base64Readme: >- IyBzZXRsaXN0CgpBbiBBUEkgZHJpdmVuIGxpc3Qgb2YgYWxsIHNvbmdzIGF2YWlsYWJsZSBmb3IgcmVxdWVzdHMgb24gaHR0cHM6Ly90d2l0Y2gudHYvd2FydGhzLgoKVGhlIGNvZGUgZm9yIHRoZSBiYWNrZW5kIGlzIGF2YWlsYWJsZSBoZXJlIDogaHR0cHM6Ly9naXRodWIuY29tL1JhZ25hci1Pb2NrL3NldGxpc3RfdjJfY29ubmV4aW9uCgpWZXJzaW9uIDEgaXMgb3V0IQpjaGVjayB0aGUgbGl2ZSB2ZXJzaW9uIGF0IGh0dHBzOi8vc2V0bGlzdC53YXJ0aHMuZnIvCg== readmeEtag: '"3d3fa475b1303f1b7ee454e71f3f9421c12bf990"' readmeLastModified: Tue, 25 Jun 2024 10:06:28 GMT repositoryId: 287072772 description: >- An API-driven list of all songs available for requests on twitch.tv/warths. Currently in development. created: '2020-08-12T17:19:03Z' updated: '2024-06-25T10:06:32Z' language: Vue archived: false stars: 0 watchers: 1 forks: 0 owner: Ragnar-Oock logo: https://avatars.githubusercontent.com/u/38219333?v=4 license: MIT repoEtag: '"b17a7fbeaddb870d5b4691178be5897b63f0b340716edf613181cbd0a71af248"' repoLastModified: Tue, 25 Jun 2024 10:06:32 GMT foundInMaster: true category: - Documentation - Server Implementations id: 749731d9ec54b967c7193370a6d4034e - source: openapi3 tags repository: https://github.com/iliashenkoa/openapi_express_excel_generator v3: true repositoryMetadata: base64Readme: >- # Node.js express simple app which generated by openapi

## Overview
This server was generated using the [OpenAPI Generator](https://openapi-generator.tech) project.  The code generator, and it's generated code allows you to develop your system with an API-First attitude, where the API contract is the anchor and definer of your project, and your code and business-logic aims to complete and comply to the terms in the API contract.

### prerequisites
- NodeJS >= 10.4
- NPM >= 6.10.0

The code was written on a mac, so assuming all should work smoothly on Linux-based computers. However, there is no reason not to run this library on Windows-based machines. If you find an OS-related problem, please open an issue and it will be resolved.

### Running the server  
To run the server, run:  
  
```  
npm start  
```
### View and test the API
You can see the API documentation, and check the available endpoints by going to http://localhost:3000/api-docs/.  Endpoints that require security need to have security handlers configured before they can return a successful response. At this point they will return [ a response code of 401](https://developer.mozilla.org/en-US/docs/Web/HTTP/Status/401).
##### At this stage the server does not support document body sent in xml format. Forms will be supported in the near future. 

### Node version and guidelines 
The code was written using Node version 10.6, and complies to the [Airbnb .eslint guiding rules](https://github.com/airbnb/javascript). 

### Project Files
#### Root Directory:
In the root directory we have (besides package.json, config.js, and log files):
- **logger.js** - where we define the logger for the project. The project uses winston, but the purpose of this file is to enable users to change and modify their own logger behavior.
- **index.js** - This is the project's 'main' file, and from here we launch the application. This is a very short and concise file, and the idea behind launching from this short file is to allow use-cases of launching the server with different parameters (changing config and/or logger) without affecting the rest of the code. 
- **expressServer.js** - The core of the Express.js server. This is where the express server is initialized, together with the OpenAPI validator, OpenAPI UI, and other libraries needed to start our server. If we want to add external links, that's where they would go. Our project uses the [express-openapi-validator](https://www.npmjs.com/package/express-openapi-validator) library that acts as a first step in the routing process - requests that are directed to paths defined in the `openapi.yaml` file are caught by this process, and it's parameters and bodyContent are validated against the schema. A successful result of this validation will be a new 'openapi' object added to the request. If the path requested is not part of the openapi.yaml file, the validator ignores the request and passes it on, as is, down the flow of the Express server.

#### api/
- **openapi.yaml** - This is the OpenAPI contract to which this server will comply. The file was generated using the codegen, and should contain everything needed to run the API Gateway - no references to external models/schemas. 

#### utils/
Currently a single file:

- **openapiRouter.js** - This is where the routing to our back-end code happens. If the request object includes an ```openapi``` object, it picks up the following values (that are part of the ```openapi.yaml``` file): 'x-openapi-router-controller', and 'x-openapi-router-service'. These variables are names of files/classes in the controllers and services directories respectively. The operationId of the request is also extracted. The operationId is a method in the controller and the service that was generated as part of the codegen process. The routing process sends the request and response objects to the controller, which will extract the expected variables from the request, and send it to be processed by the service, returning the response from the service to the caller.

#### controllers/
After validating the request, and ensuring this belongs to our API gateway, we send the request to a `controller`, where the variables and parameters are extracted from the request and sent to the relevant `service` for processing. The `controller` handles the response from the `service` and builds the appropriate HTTP response to be sent back to the user. 

- **index.js** - load all the controllers that were generated for this project, and export them to be used dynamically by the `openapiRouter.js`. If you would like to customize your controller, it is advised that you link to your controller here, and ensure that the codegen does not rewrite this file.

- **Controller.js** - The core processor of the generated controllers. The generated controllers are designed to be as slim and generic as possible, referencing to the `Controller.js` for the business logic of parsing the needed variables and arguments from the request, and for building the HTTP response which will be sent back. The `Controller.js` is a class with static methods. 

- **{{x-openapi-router-controller}}.js** - auto-generated code, processing all the operations. The Controller is a class that is constructed with the service class it will be sending the request to. Every request defined by the `openapi.yaml`  has an operationId. The operationId is the name of the method that will be called. Every method receives the request and response, and calls the `Controller.js` to process the request and response, adding the service method that should be called for the actual business-logic processing.

#### services/
This is where the API Gateway ends, and the unique business-logic of your application kicks in. Every endpoint in the `openapi.yaml` has a variable 'x-openapi-router-service', which is the name of the service class that is generated. The operationID of the endpoint is the name of the method that will be called. The generated code provides a simple promise with a try/catch clause. A successful operation ends with a call to the generic `Service.js` to build a successful response (payload and response code), and a failure will call the generic `Service.js` to build a response with an error object and the relevant response code. It is recommended to have the services be generated automatically once, and after the initial build add methods manually.

- **index.js** - load all the services that were generated for this project, and export them to be used dynamically by the `openapiRouter.js`. If you would like to customize your service, it is advised that you link to your controller here, and ensure that the codegen does not rewrite this file.

- **Service.js** - A utility class, very simple and thin at this point, with two static methods for building a response object for successful and failed results in the service operation. The default response code is 200 for success and 500 for failure. It is recommended to send more accurate response codes and override these defaults when relevant.

- **{{x-openapi-router-service}}.js** - auto-generated code, providing a stub Promise for each operationId defined in the `openapi.yaml`. Each method receives the variables that were defined in the `openapi.yaml` file, and wraps a Promise in a try/catch clause. The Promise resolves both success and failure in a call to the `Service.js` utility class for building the appropriate response that will be sent back to the Controller and then to the caller of this endpoint.

#### tests/
- **serverTests.js** - basic server validation tests, checking that the server is up, that a call to an endpoint within the scope of the `openapi.yaml` file returns 200, that a call to a path outside that scope returns 200 if it exists and a 404 if not.
- **routingTests.js** - Runs through all the endpoints defined in the `openapi.yaml`, and constructs a dummy request to send to the server. Confirms that the response code is 200. At this point requests containing xml or formData fail - currently they are not supported in the router.
- **additionalEndpointsTests.js** - A test file for all the endpoints that are defined outside the openapi.yaml scope. Confirms that these endpoints return a successful 200 response.


Future tests should be written to ensure that the response of every request sent should conform to the structure defined in the `openapi.yaml`. This test will fail 100% initially, and the job of the development team will be to clear these tests.


#### models/
Currently a concept awaiting feedback. The idea is to have the objects defined in the openapi.yaml act as models which are passed between the different modules. This will conform the programmers to interact using defined objects, rather than loosley-defined JSON objects. Given the nature of JavaScript progrmmers, who want to work with their own bootstrapped parameters, this concept might not work. Keeping this here for future discussion and feedback.



 readmeEtag: '"3bdd94837a9dfdc8bc658cffd6d0a117fe325054"' readmeLastModified: Tue, 10 Mar 2020 09:11:19 GMT repositoryId: 246238533 description: null created: '2020-03-10T07:48:11Z' updated: '2020-03-10T09:13:14Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: iliashenkoa logo: https://avatars.githubusercontent.com/u/37848086?v=4 repoEtag: '"fbf69bdd4c5067b6e735e7d56ca337636d216b2a3c004fcc684efa8353a3d487"' repoLastModified: Tue, 10 Mar 2020 09:13:14 GMT foundInMaster: true category: Server Implementations id: 0fbe268671f3d121f90e52c57d78137b - source: openapi3 tags repository: https://github.com/yjmorales/openapi-specifications v3: true id: ad429c52aa9dd2b0889b134646919780 repositoryMetadata: base64Readme: >- IyMgQVBJIERlZmluaXRpb24gYmFzZWQgb24gT3BlbkFwaSBTcGVjaWZpY2F0aW9ucyAodjMpLgoKClRoaXMgcHJvamVjdCBob2xkcyBhbiBleGFtcGxlIG9mIGFuIEFQSSBkZWZpbml0aW9uIGJhc2VkIG9uIE9wZW5BcGkgMyBTcGVjaWZpY2F0aW9ucy4gCgoKVGhlIHNwZWNpZmljYXRpb24gZmlsZSBpcyBzZWxmLWV4cGxhbmF0b3J5IGFuZCBiZWxvdyBpcyBtaW5vciBpbmZvcm1hdGlvbiBvbiBob3cgdG8gaW5zdGFsbCB0aGlzIG9uIGEgcHJvamVjdC4KCiMjIyBJbnN0YWxsYXRpb24KCklmIHlvdSB3YW50IHRvIGFkZCB0aGlzIGN1cnJlbnQgcmVwb3NpdG9yeSB0byB0aGUgcmVwb3NpdG9yaWVzIHNlY3Rpb24gb24geW91ciBvd24gcHJvamVjdCBgY29tcG9zZXIuanNvbmAgZmlsZS4KCmBgYGAKICAgICJyZXBvc2l0b3JpZXMiOiBbCiAgICAgICAgewogICAgICAgICAgICAidHlwZSI6ICJnaXQiLAogICAgICAgICAgICAidXJsIjogImdpdEBnaXRodWIuY29tOnlqbW9yYWxlcy9vcGVuYXBpLXNwZWNpZmljYXRpb25zLmdpdCIKICAgICAgICB9CiAgICBdLApgYGBgCgpGdXJ0aGVyIGluZm9ybWF0aW9uIGFib3V0IE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBwbGVhc2UgdmlzaXQgaXRzIFtzcGVjaWZpY2F0aW9uXShodHRwczovL3N3YWdnZXIuaW8vc3BlY2lmaWNhdGlvbi8pLgoKIyMjIENvbnRhY3QgbWUKClllbmllciBKaW1lbmV6Cjxicj4KaHR0cDovL3llbmllcmppbWVuZXoubmV0Cjxicj4KeWptb3JhbGVzODZAZ21haWwuY29tCgoK readmeEtag: '"fd769a2746c9e7bcfb371a6dd5c07bf0c8b609ea"' readmeLastModified: Tue, 17 Oct 2023 14:45:14 GMT repositoryId: 325404199 description: >- This project holds an example of an API definition based on OpenApi 3 Specifications. created: '2020-12-29T22:58:28Z' updated: '2023-01-22T04:57:44Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: yjmorales logo: https://avatars.githubusercontent.com/u/24995532?v=4 license: MIT repoEtag: '"785daaaf751fa59925c701710cf4e35f7478fe8785865b967bead21709aaaeef"' repoLastModified: Sun, 22 Jan 2023 04:57:44 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/aqib1/idealotask2 v3: true repositoryMetadata: base64Readme: >- IyBJZGVhbG9UYXNrMgpSZXBvc2l0b3J5IGZvciBJZGVhbG8gdGFzayAtIGNoZWNrb3V0IGFsZ29yaXRobS4gcGxlYXNlIHNlZSBkb2N4IHJlYWRtZSBpbiBwcm9qZWN0IGZvciBkZXRhaWwgYWJvdXQgdGhpcyBwcm9qZWN0Cg== readmeEtag: '"c247f17b7cdfc398708c6d933592bc9f461dfe97"' readmeLastModified: Wed, 21 Apr 2021 05:41:17 GMT repositoryId: 329116401 description: 'Repository for Idealo task - checkout algorithm ' created: '2021-01-12T21:25:49Z' updated: '2021-04-21T05:41:19Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: aqib1 logo: https://avatars.githubusercontent.com/u/8742169?v=4 repoEtag: '"472bb9340fdcb9e8c20c9867bce2b738f99dc603d28333044bb71b75c65eba65"' repoLastModified: Wed, 21 Apr 2021 05:41:19 GMT foundInMaster: true category: - Server - Server Implementations id: 13120cf7e53c50b361e754e64ed287cc - source: openapi3 tags repository: https://github.com/guyplusplus/rfc3339-datetimeformatter v3: true repositoryMetadata: base64Readme: >- IyBSRkMzMzM5IERhdGVUaW1lRm9ybWF0dGVyCgpUaGlzIGlzIGEgdmVyeSBzaW1wbGUgSkFWQSBjb2RlIHRvIHRlc3QgYSBEYXRlVGltZUZvcm1hdHRlciBkZXNpZ25lZCB0byAqKnN0cmlja2x5KiogcGFyc2UgW1JGQzMzMzkgSW50ZXJuZXQgZGF0ZS90aW1lIGZvcm1hdF0oaHR0cHM6Ly9kYXRhdHJhY2tlci5pZXRmLm9yZy9kb2MvaHRtbC9yZmMzMzM5I3NlY3Rpb24tNS42KSwgc3VjaCBhcyBgMTk5Ni0xMi0xOVQxNjozOTo1Ny0wODowMGAuCgpUaGlzIHdvcmtzIHZlcnkgd2VsbCwgd2l0aCB0aGUgb25seSAqKmV4Y2VwdGlvbiBvZiBsZWFwIHNlY29uZCoqIGRhdGUgdGltZS4KCmBgYEpBVkEKRGF0ZVRpbWVGb3JtYXR0ZXIgcmZjMzMzOVBhcnNlciA9IG5ldyBEYXRlVGltZUZvcm1hdHRlckJ1aWxkZXIoKQogICAgICAgIC5wYXJzZUNhc2VJbnNlbnNpdGl2ZSgpCiAgICAgICAgLmFwcGVuZFZhbHVlKENocm9ub0ZpZWxkLllFQVIsIDQpCiAgICAgICAgLmFwcGVuZExpdGVyYWwoJy0nKQogICAgICAgIC5hcHBlbmRWYWx1ZShDaHJvbm9GaWVsZC5NT05USF9PRl9ZRUFSLCAyKQogICAgICAgIC5hcHBlbmRMaXRlcmFsKCctJykKICAgICAgICAuYXBwZW5kVmFsdWUoQ2hyb25vRmllbGQuREFZX09GX01PTlRILCAyKQogICAgICAgIC5hcHBlbmRMaXRlcmFsKCdUJykKICAgICAgICAuYXBwZW5kVmFsdWUoQ2hyb25vRmllbGQuSE9VUl9PRl9EQVksIDIpCiAgICAgICAgLmFwcGVuZExpdGVyYWwoJzonKQogICAgICAgIC5hcHBlbmRWYWx1ZShDaHJvbm9GaWVsZC5NSU5VVEVfT0ZfSE9VUiwgMikKICAgICAgICAuYXBwZW5kTGl0ZXJhbCgnOicpCiAgICAgICAgLmFwcGVuZFZhbHVlKENocm9ub0ZpZWxkLlNFQ09ORF9PRl9NSU5VVEUsIDIpCiAgICAgICAgLm9wdGlvbmFsU3RhcnQoKQogICAgICAgIC5hcHBlbmRGcmFjdGlvbihDaHJvbm9GaWVsZC5OQU5PX09GX1NFQ09ORCwgMiwgOSwgdHJ1ZSkgLy8ybmQgcGFyYW1ldGVyOiAyIGZvciBKUkUgKDgsIDExIExUUyksIDEgZm9yIEpSRSAoMTcgTFRTKQogICAgICAgIC5vcHRpb25hbEVuZCgpCiAgICAgICAgLmFwcGVuZE9mZnNldCgiK0hIOk1NIiwiWiIpCiAgICAgICAgLnRvRm9ybWF0dGVyKCkKICAgICAgICAud2l0aFJlc29sdmVyU3R5bGUoUmVzb2x2ZXJTdHlsZS5TVFJJQ1QpCiAgICAgICAgLndpdGhDaHJvbm9sb2d5KElzb0Nocm9ub2xvZ3kuSU5TVEFOQ0UpOwoKRGF0ZVRpbWVGb3JtYXR0ZXIgcmZjMzMzOUZvcm1hdHRlciA9IERhdGVUaW1lRm9ybWF0dGVyLklTT19PRkZTRVRfREFURV9USU1FOwpgYGA= readmeEtag: '"4e6b0136bfbbab7c688f06f2eef18fa7431c9066"' readmeLastModified: Tue, 17 Aug 2021 13:58:30 GMT repositoryId: 385612575 description: JAVA DataTimeFormatter to strictly parse (and format) RFC3339 timestamps created: '2021-07-13T13:25:08Z' updated: '2021-08-17T13:58:36Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: guyplusplus logo: https://avatars.githubusercontent.com/u/18182742?v=4 license: MIT repoEtag: '"0cee3b1cb837103a91fad32e556b0cab455d3c501cb3d068a14dc68221313062"' repoLastModified: Tue, 17 Aug 2021 13:58:36 GMT foundInMaster: true category: Server Implementations id: 3f69fae4e5cc80dba4dd2f67222b4a1f - source: openapi3 tags repository: https://github.com/akiosarkiz/openapi-attributes v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFBIUCBBdHRyaWJ1dGVzCgojIyBGZWF0dXJlcwoKLSBbeF0gR2VuZXJhdGUgaW5mbwotIFt4XSBHZW5lcmF0ZSBzY2hlbWEKLSBbeF0gR2VuZXJhdGUgcGF0aHMKLSBbeF0gR2VuZXJhdGUgc2VjdXJpdHkgc2NoZW1hCi0gW3hdIEdlbmVyYXRlIHNlcnZlcnMKCgpUaGlzIENMSSBUb29sIGlzIGFibGUgdG8gZ2VuZXJhdGUgYW4gT3BlbkFwaSBKU09OIGZpbGUgZGVzY3JpcHRpb24gYWNjb3JkaW5nIHRvIFBIUCBhdHRyaWJ1dGVzIGNvbnRhaW5lZCBpbiB5b3VyIGZpbGVzLgoKCiMjIOKaoO+4jyBNaXNzaW5nIHNvbWV0aGluZyA/Ckp1c3Qgb3BlbiBhbiBpc3N1ZSBzYXlpbmcgd2hhdCdzIG1pc3NpbmcgISBGZWVsIGZyZWUgdG8gb3BlbiBhIFBSIGJ1dCB3ZSByZWNvbW1lbmQgb3BlbmluZyBhbiBpc3N1ZSBiZWZvcmVoYW5kLiAKCgojIyMgRG9jdW1lbnRhdGlvbiBbbGlua10oaHR0cHM6Ly9ha2lvc2Fya2l6LmdpdGh1Yi5pby9vcGVuYXBpLWF0dHJpYnV0ZXMvKQo= readmeEtag: '"9bfb44c9bf4d69db88bfb47dbc6fa79595594022"' readmeLastModified: Wed, 27 Oct 2021 13:32:35 GMT repositoryId: 394191942 description: null created: '2021-08-09T07:22:42Z' updated: '2021-10-27T13:32:43Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: AkioSarkiz logo: https://avatars.githubusercontent.com/u/42392852?v=4 license: MIT repoEtag: '"8c69d9686b39699edd8e8c90c684e3f1d7fe1b3c43455085e9ab03e1fb8db729"' repoLastModified: Wed, 27 Oct 2021 13:32:43 GMT foundInMaster: true category: Parsers id: 2b99b3aec8d7b940c4aa7d20588bfc92 - source: openapi3 tags repository: https://github.com/umatare5/logbook-go v3: true repositoryMetadata: base64Readme: >- IyBsb2dib29rLWdvCgoqKkRPIE5PVCBFRElUIFRISVMgQ09ERSBNQU5VQUxMWSEqKgoKbG9nYm9vay1nbyBpcyBHbyBsaWJyYXJ5IHRvIHVzZSBbTG9nYm9vayBBUEldKGh0dHBzOi8vZ2l0aHViLmNvbS91bWF0YXJlNS9sb2dib29rLWFwaSkuCgpUaGUgY29kZSB3YXMgZ2VuZXJhdGVkIGF1dG9tYXRpY2FsbHkgYnkgdXNpbmcgW2xvZ2Jvb2stb3BlbmFwaS1zcmNdKGh0dHBzOi8vZ2l0aHViLmNvbS91bWF0YXJlNS9sb2dib29rLW9wZW5hcGktc3JjKS4KCiMjIFVzYWdlCgpgYGBzaApnbyBnZXQgZ2l0aHViLmNvbS91bWF0YXJlNS9sb2dib29rLWdvCmBgYAo= readmeEtag: '"c8808e3b60074f663ef61be2f6fe55fb8ae4ad14"' readmeLastModified: Wed, 26 Jan 2022 17:19:37 GMT repositoryId: 449648100 description: Client library to use my Logbook API created: '2022-01-19T10:36:07Z' updated: '2022-02-15T13:53:30Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: umatare5 logo: https://avatars.githubusercontent.com/u/72645163?v=4 repoEtag: '"f14f9e51a19c88bc40c46f67c4ddb3b79fc5310b5f903ae6c3b72f94a78faea4"' repoLastModified: Tue, 15 Feb 2022 13:53:30 GMT foundInMaster: true category: - SDK - Code Generators id: d1bf1d6b555601d0f01dd1173e521e68 - source: openapi3 tags repository: https://github.com/kekos/prest-doc v3: true id: dbfb1dd556b661e6cd7878109faa7a4d repositoryMetadata: base64Readme: >- IyBQcmVzdC1kb2MKClN0YXRpYyBTaXRlIEdlbmVyYXRvciBmb3IgT3BlbkFQSSBkb2N1bWVudGF0aW9uIHdyaXR0ZW4gaW4gUEhQLgoKIyMgSW5zdGFsbAoKYGBgCmNvbXBvc2VyIHJlcXVpcmUga2Vrb3MvcHJlc3QtZG9jCmBgYAoKUmVxdWlyZXMgYXQgbGVhc3QgUEhQIDguMS4KCiMjIFVzYWdlCgpgYGAKLi9iaW4vcHJlc3QtZG9jIDxpbl9kaXJlY3Rvcnk+IDxvdXRfZGlyZWN0b3J5PiA8bGF5b3V0X2ZpbGU+IFs8Y29uZmlnX2ZpbGU+XQpgYGAKCiogYDxpbl9kaXJlY3Rvcnk+YCBwb2ludHMgdG8gYSBkaXJlY3Rvcnkgd2hpY2ggUHJlc3QtZG9jIHNob3VsZCByZWFkIGFuZCBjb252ZXJ0LgoqIGA8b3V0X2RpcmVjdG9yeT5gIHBvaW50cyB0byBhIGRpcmVjdG9yeSB3aGVyZSBjb252ZXJ0ZWQgZmlsZXMgd2lsbCBiZSB3cml0dGVuLiAqKlBsZWFzZSBub3RlIHRoYXQgdGhpcyBkaXJlY3Rvcnkgd2lsbCBiZSBkZWxldGVkIGJ5IFByZXN0LWRvYyBvbiBlYWNoIHJ1bioqCiogYDxsYXlvdXRfZmlsZT5gIGlzIGEgZmlsZSBwYXRoIHRvIGxheW91dCBmaWxlLiBTZWUgdGhlIFtgZXhhbXBsZXMvYF0oZXhhbXBsZXMpIGRpcmVjdG9yeS4KKiBgPGNvbmZpZ19maWxlPmAgaXMgYSBmaWxlIHBhdGggdG8gW2NvbmZpZ3VyYXRpb25dKCNjb25maWd1cmF0aW9uKS4gKk9wdGlvbmFsKi4KCiMjIENvbmZpZ3VyYXRpb24KCiMjIyBBUEkgdGVtcGxhdGVzCgpZb3UgY2FuIG92ZXJyaWRlIHRoZSBkZWZhdWx0IE9wZW5BUEkgTWFya2Rvd24gdGVtcGxhdGVzIGJ5IGNyZWF0aW5nIGEgY29uZmlndXJhdGlvbiBmaWxlCmFuZCBzdXBwbHkgaXQgd2l0aCB5b3VyIG93biBpbnN0YW5jZXMgb2YgdGVtcGxhdGUgaW50ZXJmYWNlczoKCmBgYHBocAo8P3BocAp1c2UgS2Vrb3NcUHJlc3REb2NcQ29uZmlndXJhdGlvbjsKdXNlIEtla29zXFByZXN0RG9jXEFwaVRlbXBsYXRlc1xDb250cmFjdHNcU2NoZW1hVGVtcGxhdGU7CgpyZXR1cm4gbmV3IENvbmZpZ3VyYXRpb24oCiAgICBhcGlfdGVtcGxhdGVzX2NsYXNzX21hcDogWwogICAgICAgIFNjaGVtYVRlbXBsYXRlOjpjbGFzcyA9PiBuZXcgTXlPd25JbXBsZW1lbnRhdGlvbk9mU2NoZW1hVGVtcGxhdGUoKSwKICAgIF0sCik7CmBgYAoKIyMjIFRvcGljcyBtZW51CgpQcmVzdC1kb2MgY2FuIGJlIGNvbmZpZ3VyZWQgd2l0aCBob3cgaXQgc2hvdWxkIGNyZWF0ZSB0aGUgbWFpbiBBUEkgbmF2aWdhdGlvbiwgdGhlICJ0b3BpY3MgbWVudSIuIFR3byBpbXBsZW1lbnRhdGlvbnMKYXJlIGluY2x1ZGVkLCBidXQgeW91IGNhbiBidWlsZCB5b3VyIG93biBieSBpbXBsZW1lbnRpbmcgdGhlIGludGVyZmFjZSBgXEtla29zXFByZXN0RG9jXEFwaUVudGl0aWVzXFRvcGljc1JlcG9zaXRvcnlgLgpVc2UgdGhlIGBDb25maWd1cmF0aW9uYCBjbGFzc8K0IGBhcGlfdGVtcGxhdGVzX2NsYXNzX21hcGAgcHJvcGVydHkgdG8gY29uZmlndXJlIHRoaXMuCgojIyMjIGBUYWdnZWRUb3BpY3NSZXBvc2l0b3J5YCAoZGVmYXVsdCkKCkFsbCBwYXRoIG9wZXJhdGlvbnMgbXVzdCBoYXZlIGEgdGFnIHByZWZpeGVkIHdpdGggYHRvcGljLWAuIFRoYXQgdGFnIHdpbGwgYmUgdXNlZCB0byBncm91cCBvcGVyYXRpb25zIHRvZ2V0aGVyIHVuZGVyIHRoZQpuYW1lIG9mIHRhZywgZXhjbHVkaW5nIGB0b3BpYy1gLgoKIyMjIyBgU2luZ2xlVG9waWNSZXBvc2l0b3J5YAoKQWxsIHBhdGhzIGFyZSBncm91cGVkIHRvIGEgc2luZ2xlIHRvcGljICoiT3BlcmF0aW9ucyIqLgoKIyMgRmVhdHVyZXMKCi0gW3hdIENvbnZlcnRzIE9wZW5BUEkgc3BlY2lmaWNhdGlvbnMgKEpTT04pIHRvIEhUTUwuCi0gW3hdIEFsbG93cyBmb3Igb3duIEhUTUwgbGF5b3V0IHRlbXBsYXRlLCB3cml0dGVuIGluIFBIUC4KLSBbeF0gQWxsb3dzIGZvciBvd24gQ1NTIGFuZCBKYXZhU2NyaXB0LgotIFsgXSBTb3J0IHBhdGhzICh0b3BpY3MpIGluIGFuIG9yZGVyIGxvZ2ljYWwgZm9yIHlvdSBhbmQgeW91ciBhcHAuCgojIyBCdWdzIGFuZCBpbXByb3ZlbWVudHMKClJlcG9ydCBidWdzIGluIEdpdEh1YiBpc3N1ZXMgb3IgZmVlbCBmcmVlIHRvIG1ha2UgYSBwdWxsIHJlcXVlc3QgOi0pCgojIyBMaWNlbnNlCgpNSVQK readmeEtag: '"2599c1f5153a106b0566ea70bca3ec7fe3d1053e"' readmeLastModified: Sat, 24 May 2025 11:05:04 GMT repositoryId: 513455651 description: Static Site Generator for OpenAPI documentation created: '2022-07-13T09:17:04Z' updated: '2026-01-13T20:03:05Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: Kekos logo: https://avatars.githubusercontent.com/u/914782?v=4 license: MIT repoEtag: '"cab3348775425a0cbfd96910d7de98bebc6fdd7b4c3466ba5c66d56089b9ec87"' repoLastModified: Tue, 13 Jan 2026 20:03:05 GMT foundInMaster: true category: Parsers - source: openapi3 tags repository: https://github.com/pavlopolovyi/rick-and-morty v3: true id: d5a18e661bb41a0a8c0fea831070f0c7 repositoryMetadata: base64Readme: >- IyBSaWNrJk1vcnR5IGNoYXJhY3RlcnMgV2lraQoKPGgyPvCfk6MgUHJvamVjdCBkZXNjcmlwdGlvbiDwn5OjPC9oMj4KVGhpcyBpcyBhIHNpbXBsZSBSRVNUIEFQSSB0aGF0IGV4cG9zZXMgdHdvIGVuZG9pbnRzOiA8Yj4vcmFuZG9tPC9iPiAtIHRvIGdldCBpbmZvIGFib3V0IHJhbmRvbSBSaWNrJk1vcnR5IGNoYXJhY3RlcnMgYW5kIDxiPi9ieS1uYW1lP25hbWVMaWtlPXtxdWVyeX08L2I+IC0gdG8gZmluZCBpbmZvIGFib3V0IGNoYXJhY3RlcnMgd2hvc2UgbmFtZSBpcyBsaWtlIHF1ZXJ5IHBhdHRlcm4uIFRoZSBkYXRhIGlzIGR5bmFtaWNhbGx5IHN5bmNocm9uaXplZCB1c2luZyBhbiBleHRlcm5hbCBBUEkgKGh0dHBzOi8vcmlja2FuZG1vcnR5YXBpLmNvbS8pLiBUaGlzIGFwcGxpY2F0aW9uIGZvbGxvd3MgU09MSUQgcHJpbmNpcGxlcyBhbmQgaXMgYnVpbHQgdXAgdXNpbmcgTi10aWVyIGFyY2hpdGVjdHVyZS4KCjxoMj5GZWF0dXJlczwvaDI+CgoqIER5bmFtaWNhbCBkYXRhIHN5bmNocm9uaXphdGlvbiB1c2luZyBidWlsdCBpbiBTcHJpbmcgQm9vdCBmZWF0dXJlCgoqIFNhdmluZyBkYXRhIHRvIHRoZSBkYXRhYmFzZQoKKiAzLWxheWVyIGFyY2hpdGVjdHVyZTogQ29udHJvbGxlcnMsIFNlcnZpY2VzIGFuZCBSZXBvc2l0b3JpZXMKCiogQWxsIGVuZHBvaW50cyBhcmUgZGVzY3JpYmVkIHVzaW5nIE9wZW5BUEkgc3BlY2lmaWNhdGlvbgoKKiBTZW5kaW5nIHJlcXVlc3RzIHRvIGV4dGVybmFsIEFQSQoKKiBVc2luZyBsYXRlc3Qgc3RhYmxlIEphdmEgdmVyc2lvbiBhbmQgU3ByaW5nIEJvb3QgMwoKKiBQcm9qZWN0IGluY2x1ZGVzIHRlc3RzIGZvciBSZXBvc2l0b3J5LCBTZXJ2aWNlIGFuZCBDb250cm9sbGVyIGxheWVycwoKCjxoMj46YnJpY2tzOlByb2plY3Qgc3RydWN0dXJlOmJyaWNrczo8L2gyPgotIENvbnRyb2xsZXJzIC0gdGFrZSBtYWluIHBhcnQgaW4gcmVxdWVzdC9yZXNwb25zZSBjeWNsZSwgcmVjZWl2ZSBkYXRhIGZyb20gdXNlcnMgYW5kIGludm9rZSBidXNpbmVzcyBsb2dpYyBvZiBzZXJ2aWNlcyAKdG8gcHJvY2VzcyBpdCBhbmQgc3RvcmUgaW4gZGF0YWJhc2UuIFNlbmQgYmFjayBkYXRhIHRvIHVzZXJzLCB3aGVuIHRoZXkgcmVxdWVzdCBpdC48YnI+Ci0gU2VydmljZXMgLSB0aGlzIGxheWVyIGNvb3JkaW5hdGVzIHdvcmsgb2YgYWxsIGFwcGxpY2F0aW9uLCBwcm9jZXNzIGNvbW1hbmRzIGFuZCBwZXJmb3JtcyBkYXRhIHN5bmMuPGJyPgotIFJlcG9zaXRvcmllcyAtIGhlcmUgaW5mb3JtYXRpb24gaXMgc3RvcmVkIGFuZCByZXRyaWV2ZWQuPGJyPgoKIyMgPGgyPlRlY2hub2xvZ2llczwvaDI+CiogSmF2YSAxNwoqIFNwcmluZyBCb290CiogU3ByaW5nIERhdGEgSlBBCiogU3ByaW5nIE1WQwoqIFBvc3RncmVTUUwKKiBIaWJlcm5hdGUKKiBEb2NrZXIKKiBMaXF1aWJhc2UKKiBPcGVuQVBJIDMuMAoqIEpVbml0LCBNb2NraXRvLCBUZXN0Y29udGFpbmVycywgU3ByaW5nIEJvb3QgVGVzdGluZwoqIEZlaWduQ2xpZW50IGZvciBzZW5kaW5nIHJlcXVlc3RzIHRvIGV4dGVybmFsIEFQSQoKCiMjIDxoMj46Ym9tYjpJbnN0cnVjdGlvbnMgZm9yIGxhdW5jaGluZyB0aGUgcHJvamVjdDpib21iOjwvaDI+CjxoND5UbyBydW4gdGhpcyBwcm9qZWN0IGxvY2FsbHksIGZvbGxvdyB0aGVzZSBzdGVwczo8L2g0PgoKMe+4j+KDoyBZb3Ugc2hvdWxkIGluc3RhbGwgPGEgaHJlZj0iaHR0cHM6Ly9kb2NzLmRvY2tlci5jb20vZ2V0LWRvY2tlci8iPkRvY2tlcjwvYT4gZm9yIGVhc3kgbGF1bmNoaW5nCgoy77iP4oOjICBDbG9uZSB0aGlzIHByb2plY3QgZnJvbSBHaXRIdWIKYGBgYmFzaApodHRwczovL2dpdGh1Yi5jb20vUGF2bG9Qb2xvdnlpL3JpY2stYW5kLW1vcnR5CmBgYAoz77iP4oOjIE5hdmlnYXRlIHRvIHByb2plY3QgZm9sZGVyIGluIHRlcm1pbmFsIGFuZCBydW4gZm9sbG93aW5nIGNvbW1hbmQ6CmBgYGJhc2gKLi9tdm53IGNsZWFuIHBhY2thZ2UgLURza2lwVGVzdHMKYGBgCmlmIGl0IGhhcyBmYWlsZWQsIGluc3RhbGwgTWF2ZW4gYW5kIHJ1biBmb2xsb3dpbmcgY29tbWFuZDoKYGBgYmFzaAptdm4gcGFja2FnZSAtRHNraXBUZXN0cwpgYGAKNO+4j+KDoyBUaGVuIHJ1biBhbmQgd2FpdCB3aGlsZSBpbWFnZXMgYXJlIGJ1aWxkaW5nOgpgYGBiYXNoCmRvY2tlciBjb21wb3NlIGJ1aWxkCmBgYAo177iP4oOjIEFuZCBmaW5hbGx5IHJ1biBhcHBsaWNhdGlvbi4gU3RhcnQgdGFrZXMgYSB3aGlsZSwgYmVjYXVzZSBvZiBpbml0aWFsIGRhdGEgc3luY2hyb25pemF0aW9uOgpgYGBiYXNoCmRvY2tlciBjb21wb3NlIHVwCmBgYAo6c2l4OiBHbyB0byB0aGUgYnJvd3NlciBhbmQgdXNlIHRoZSBmb2xsb3dpbmcgdXJsIHRvIHRlc3QgdGhlIGFwcGxpY2F0aW9uIGFuZCBzZWUgT3BlbkFQSSBkb2N1bWVudGF0aW9uLiAKYGBgYmFzaApodHRwOi8vbG9jYWxob3N0OjgwODEvc3dhZ2dlci11aS5odG1sCmBgYAo6c2V2ZW46IElmIHBvcnQgODA4MSBpcyBidXN5IG9uIHlvdXIgbWFjaGluZSwganVzdCBjaGFuZ2UgU1BSSU5HX0xPQ0FMX1BPUlQgdmFsdWUgaW4gdGhlIC5lbnYgZmlsZS4KCg== readmeEtag: '"81af7282148faacadb798d1cf8ecae1d4c8a436d"' readmeLastModified: Mon, 09 Jan 2023 15:30:50 GMT repositoryId: 586850314 description: >- This is a simple REST API that exposes two endoints and provides info about characters from Rick&Morty universe. It uses an external API to synchronize data. created: '2023-01-09T11:32:36Z' updated: '2023-01-09T13:45:22Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: PavloPolovyi logo: https://avatars.githubusercontent.com/u/96145406?v=4 repoEtag: '"06e556f25301e8c083eb5b4134b499d1c3d8e5da58cac3601e3252ece4317783"' repoLastModified: Mon, 09 Jan 2023 13:45:22 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/maciek1839/spring-cloud-demo v3: true id: 4db91225301e8a967d0f7ef5ba72cc65 repositoryMetadata: base64Readme: >- # Spring Cloud demo

| Branch |                                                                                         Pipeline                                                                                         |                                                                                      Code coverage                                                                                       |                                       Test report                                        |                                         Spring REST Docs                                         |                                 SonarCloud                                 |
|:------:|:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------:|:----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------:|:----------------------------------------------------------------------------------------:|:------------------------------------------------------------------------------------------------:|:--------------------------------------------------------------------------:|
| master | [![pipeline status](https://gitlab.com/ShowMeYourCodeYouTube/spring-cloud-demo/badges/master/pipeline.svg)](https://gitlab.com/ShowMeYourCodeYouTube/spring-cloud-demo/-/commits/master) | [![coverage report](https://gitlab.com/ShowMeYourCodeYouTube/spring-cloud-demo/badges/master/coverage.svg)](https://gitlab.com/ShowMeYourCodeYouTube/spring-cloud-demo/-/commits/master) | [link](https://showmeyourcodeyoutube.gitlab.io/spring-cloud-demo/test-report/index.html) | [link](https://showmeyourcodeyoutube.gitlab.io/spring-cloud-demo/rest-docs/client-api-docs.html) | [link](https://sonarcloud.io/organizations/showmeyourcodeyoutube/projects) |


## Technology

- JDK (AWS Corretto)
- Spring Boot Cloud
    - OpenFeign
    - Ribbon
    - Eureka
- Spring REST Docs
- Swagger
- gRPC
- GraphQL
- WireMock
- RestAssured & Hamcrest

## Services specification

![img](docs/high-level-design-diagram.drawio.png)

The diagram was created using [https://app.diagrams.net/](https://app.diagrams.net/).

All services use OpenApi3 (OAS3).

- `eureka-server`
  - Technology: Spring MVC
  - Dashboard: http://localhost:8761
  - Port: 8761
- `shop-microservice`
  - Technology: Spring MVC
  - Swagger UI: http://localhost:8100/shop/swagger-ui.html
  - Eureka service name: spring-cloud-eureka-shop
  - Port: 8100 (REST)
  - Context path: /shop
  - OpenFeign using provided/static URLs or hosts from Ribbon/Eureka (warehouse)
- `warehouse-microservice`
  - Technology: Spring MVC
  - Swagger UI: http://localhost:8200/warehouse/swagger-ui.html
  - Eureka service name: spring-cloud-eureka-warehouse
  - Port: 8200 (REST), 7000 (gRPC)
  - Context path: /warehouse
  - Java API Client configuration using a provided/static URL (factory)
- `factory-microservice`
  - Technology: Spring WebFlux
  - Swagger UI: http://localhost:8300/factory/swagger-ui.html
  - Spring REST Docs: factory-microservice/src/main/asciidoc
  - Port: 8300
  - Context path: /factory
- `reporting-microservice`
  - Technology: Spring MVC
  - Swagger UI: http://localhost:8000/reporting/swagger-ui.html
  - Eureka service name: spring-cloud-eureka-reporting
  - GraphQL
    - GraphiQL - http://localhost:8000/reporting/graphiql
    - Altair - http://localhost:8000/reporting/altair
    - Playground - http://localhost:8000/reporting/playground
    - Voyager - http://localhost:8000/reporting/voyager
    - GraphQL endpoint (not accessible with GET) - http://localhost:8000/reporting/graphql
  - Port: 8000
  - Context path: /reporting
  - RestTemplate using Ribbon with Eureka
- `admin-dashboard`
  - Technology: Spring MVC
  - Dashboard: http://localhost:9000/
  - Eureka service name: spring-cloud-eureka-admin
  - Port: 9000
- `tests`
  - System and acceptance tests.

---

Design remarks:
- In order to simplify this example, models are reused by microservices. In real applications it might not be a good choice.
- The warehouse microservice is a proxy forwarding requests to the factory microservice.
- The shop microservice is a proxy forwarding requests to the warehouse microservice.
- The client microservice calls arbitrary other services.

## Getting started

1. (Optional) Enable Lombok annotations in your IDE
2. (Optional) Run all servers using predefined configurations for IntelliJ (.runconfig)
   1. If your IDE throws errors because some classes were not found, mark `target/generated-sources` as `Generated Sources Root` in your IDE.
3. Run the Eureka server
4. Run microservices
5. Run a client which calls microservices using Eureka

![img](./docs/spring-boot-admin.png)

![img](./docs/spring-boot-eureka.png)

## Testing Strategy

> Acceptance Testing is done after the system testing. It is used to check whether the software meets the customer requirements or not. Acceptance testing is used by testers, stakeholders as well as clients.
>
> Reference: https://www.geeksforgeeks.org/difference-between-system-testing-and-acceptance-testing/

![img](docs/testing-strategy.jpg)

[Reference](https://www.geeksforgeeks.org/acceptance-testing-software-testing/)

System and acceptance tests are in the `tests` module.
Run them when all microservices are up.

Thanks to automatic tests you don't need to manually verify is services are fine after making major changes.

Other good articles about testing levels:
- https://www.guru99.com/levels-of-testing.html

## Primary Projects of Spring Cloud

![img](docs/microservices-6.svg)

[Reference](https://spring.io/microservices)


- Spring Cloud Config
  - Centralized external configuration management backed by a git repository. The configuration resources map directly to Spring Environment but could be used by non-Spring applications if desired.
- Spring Cloud Netflix
  - Integration with various Netflix OSS components (Eureka, Hystrix, Zuul, Archaius, etc.):
    - Service Discovery (Eureka)
    - Circuit Breaker (Hystrix)
    - Intelligent Routing (Zuul)
    - Client Side Load Balancing (Ribbon)
- Spring Cloud Bus
  - An event bus for linking services and service instances together with distributed messaging. Useful for propagating state changes across a cluster (e.g. config change events).
- Spring Cloud Open Service Broker
  - Provides a starting point for building a service broker that implements the Open Service Broker API.
- Spring Cloud Consul
  - Service discovery and configuration management with Hashicorp Consul.
- Spring Cloud Sleuth
  - Distributed tracing for Spring Cloud applications, compatible with Zipkin, HTrace and log-based (e.g. ELK) tracing.
- Spring Cloud Data Flow
  - A cloud-native orchestration service for composable microservice applications on modern runtimes. Easy-to-use DSL, drag-and-drop GUI, and REST-APIs together simplifies the overall orchestration of microservice based data pipelines.
- Spring Cloud Stream
  - A lightweight event-driven microservices framework to quickly build applications that can connect to external systems. Simple declarative model to send and receive messages using Apache Kafka or RabbitMQ between Spring Boot apps.
- Spring Cloud Stream Applications
  - Spring Cloud Stream Applications are out of the box Spring Boot applications providing integration with external middleware systems such as Apache Kafka, RabbitMQ etc. using the binder abstraction in Spring Cloud Stream.
- Spring Cloud Task
  - A short-lived microservices framework to quickly build applications that perform finite amounts of data processing.
  - Simple declarative for adding both functional and non-functional features to Spring Boot apps.
- Spring Cloud Task App Starters
  - Spring Cloud Task App Starters are Spring Boot applications that may be any process including Spring Batch jobs that do not run forever, and they end/stop after a finite period of data processing.
- Spring Cloud Zookeeper
  - Service discovery and configuration management with Apache Zookeeper.
- Spring Cloud Contract
  - Spring Cloud Contract is an umbrella project holding solutions that help users in successfully implementing the Consumer Driven Contracts approach.
- Spring Cloud Gateway
  - Spring Cloud Gateway is an intelligent and programmable router based on Project Reactor.
- Spring Cloud OpenFeign
  - Spring Cloud OpenFeign provides integrations for Spring Boot apps through autoconfiguration and binding to the Spring Environment and other Spring programming model idioms.
- Spring Cloud Function
  - Spring Cloud Function promotes the implementation of business logic via functions.

---

- Spring Cloud documentations
  - https://spring.io/projects/spring-cloud
- Spring Cloud releases
  - https://github.com/spring-cloud/spring-cloud-release/wiki
- Spring Boot & Spring Cloud supported versions
  - https://github.com/spring-cloud/spring-cloud-release/wiki/Supported-Versions

### Spring Cloud Netflix components replacements

| CURRENT	                    | REPLACEMENT                                       |
|-----------------------------|---------------------------------------------------|
| Hystrix	                    | Resilience4j                                      |
| Hystrix Dashboard / Turbine | 	Micrometer + Monitoring System                   |
| Ribbon                      | 	Spring Cloud Loadbalancer                        |
| Zuul 1	                     | Spring Cloud Gateway                              |
| Archaius 1	                 | Spring Boot external config + Spring Cloud Config |

## Service Discovery in Microservices

A microservice needs to know the location (IP address and port) of every service it communicates with.
If we don’t employ a Service Discovery mechanism, service locations become coupled,
leading to a system that’s difficult to maintain.
We could wire the locations or inject them via configuration in a traditional application,
but it isn’t recommended in a modern cloud-based application of this kind.

The Service Discovery mechanism helps us know where each instance is located.
In this way, a Service Discovery component acts as a registry in which the addresses of all instances are tracked.

### How Does Service Discovery Works?

![img](docs/Service-Discovery-1-1.png)

Let’s describe the steps illustrated in the diagram:
1. The location of the Service Provider is sent to the Service Registry (a database containing the locations of all available service instances).
2. The Service Consumer asks the Service Discovery Server for the location of the Service Provider.
3. The location of the Service Provider is searched by the Service Registry in its internal database and returned to the Service Consumer.
4. The Service Consumer can now make direct requests to the Service Provider.

There are two main Service Discovery patterns: Client‑Side Discovery and Server‑Side Discovery.

### Client-Side Service Discovery

![img](docs/netflix-eureka-service-discovery.png)

[Reference](https://www.codeprimers.com/client-side-service-discovery-in-spring-boot-with-netflix-eureka/)

When using Client-Side Discovery, the Service Consumer is responsible for determining the network locations of available service instances and load balancing requests between them.
The client queries the Service Register.
Then the client uses a load-balancing algorithm to choose one of the available service instances and performs a request.

![img](docs/Service-Discovery-Client-Side.png)

Giving responsibility for client-side load balancing is both a burden and an advantage.
It’s an advantage because it saves an extra hop that we would’ve had with a dedicated load balancer.
It’s a disadvantage because the Service Consumer must implement the load balancing logic.

We can also point out that the Service Consumer and the Service Registry are quite coupled.
This means that Client-Side Discovery logic must be implemented for each programming language and framework used by the Service Consumers.

### Server-Side Service Discovery

![img](docs/Service-Discovery-Server-Side.png)

The alternate approach to Service Discovery is the Server-Side Discovery model,
which uses an intermediary that acts as a Load Balancer.
The client makes a request to a service via a load balancer that acts as an orchestrator.
The load balancer queries the Service Registry and routes each request to an available service instance.

In this approach, a dedicated actor, the Load Balancer, does the job of load balancing.
This is the main advantage of this approach.
Indeed, creating this level of abstraction makes the Service Consumer lighter, as it doesn’t have to deal with the lookup procedure.
As a matter of fact, there’s no need to implement the discovery logic separately for each language and framework that the Service Consumer uses.

On the other hand, we must set up and manage the Load Balancer, unless it’s already provided in the deployment environment.

### What Is Service Registry?

The Service Register is a crucial part of service identification.
It’s a database containing the network locations of service instances.
A Service Registry must be highly available and up-to-date.
Clients can cache the network paths obtained from the Service Registry; however, this information eventually becomes obsolete, and clients won’t reach the service instances.
Consequently, a Service Registry consists of a cluster of servers that use a replication protocol to maintain consistency.

Service Registration options:
- Self-Registration
- Third-party Registration

### References

- https://www.baeldung.com/cs/service-discovery-microservices

## Spring REST Docs vs Springdoc

Reference: https://www.baeldung.com/spring-rest-docs-vs-openapi

**Spring REST Docs** is a framework developed by the Spring community in order to create accurate documentation for RESTful APIs. The output of running the tests is created as AsciiDoc files which can be put together using Asciidoctor to generate an HTML page describing our APIs.

**Springdoc OpenAPI UI** can generate UI using Swagger UI.

## Swagger 2 vs OpenApi3

![img](docs/swagger2-vs-openapi3.png)

**OpenAPI 3 is the successor of the widely used OpenAPI/Swagger 2.0 format, for machine-readable API definitions.**

OpenAPI Specification (formerly Swagger Specification) is an API description format for REST APIs. An OpenAPI file allows you to describe your entire API, including:
- Available endpoints (/users) and operations on each endpoint (GET /users, POST /users)
- Operation parameters Input and output for each operation
- Authentication methods
- Contact information, license, terms of use and other information.

API specifications can be written in YAML or JSON. The format is easy to learn and readable to both humans and machines. The complete OpenAPI Specification can be found on GitHub: OpenAPI 3.0 Specification

Swagger is a set of open-source tools built around the OpenAPI Specification that can help you design, build, document and consume REST APIs.

- Reference: https://dev.to/frolovdev/openapi-spec-swagger-v2-vs-v3-4o7c
- Official documentation: https://spec.openapis.org/oas/v3.1.0

## Spring REST Docs

Spring REST Docs helps you to document RESTful services.

It combines hand-written documentation written with Asciidoctor and auto-generated snippets produced with Spring MVC Test. This approach frees you from the limitations of the documentation produced by tools like Swagger.

It helps you to produce documentation that is accurate, concise, and well-structured. This documentation then allows your users to get the information they need with a minimum of fuss.

Ref: https://spring.io/projects/spring-restdocs#overview

![img](docs/springs-docs-generated-api-doc.png)

Example implementation - [spring-projects / spring-restdocs](https://github.com/spring-projects/spring-restdocs/blob/2.0.x/samples/rest-notes-spring-data-rest/src/main/asciidoc/api-guide.adoc)

## API architecture styles

### REST

REST, or REpresentational State Transfer, is an architectural style for providing standards between computer systems on the web, making it easier for systems to communicate with each other. REST-compliant systems, often called RESTful systems, are characterized by how they are stateless and separate the concerns of client and server.

An architectural style is a set of principles and patterns that guide the design of software systems

Key features:
- Uniform Interface
  - It is a key constraint that differentiate between a REST API and Non-REST API. It suggests that there should be an uniform way of interacting with a given server irrespective of device or type of application (website, mobile app).
    - There are four guidelines principle of Uniform Interface are:
      - Resource-Based: Individual resources are identified in requests. For example: API/users.
      - Manipulation of Resources Through Representations: Client has representation of resource and it contains enough information to modify or delete the resource on the server, provided it has permission to do so. Example: Usually user get a user id when user request for a list of users and then use that id to delete or modify that particular user.
      - Self-descriptive Messages: Each message includes enough information to describe how to process the message so that server can easily analyses the request.
      - Hypermedia as the Engine of Application State (HATEOAS): It need to include links for each response so that client can discover other resources easily.
- Stateless
  - It means that the necessary state to handle the request is contained within the request itself and server would not store anything related to the session.
- Cacheable
  - Every response should include whether the response is cacheable or not and for how much duration responses can be cached at the client side.
- Client-Server
  - REST application should have a client-server architecture.
- Layered System
  - An application architecture needs to be composed of multiple layers. Each layer doesn’t know any thing about any layer other than that of immediate layer and there can be lot of intermediate servers between client and the end server. Intermediary servers may improve system availability by enabling load-balancing and by providing shared caches.
- (Optional) Code on Demand
- RFC specification:
  - <https://www.w3.org/Protocols/rfc2616/rfc2616-sec9.html#sec9.6>
- References:
  - <https://www.codecademy.com/article/what-is-rest>
  - <https://www.geeksforgeeks.org/rest-api-architectural-constraints/>

#### Good practises for designing REST API

- Return JSON instead of multiple formats.
- Use Nouns instead of Verbs.
  - GET /books/123
- Name the collections using Plural Nouns.
- Use resource nesting to show relations or hierarchy.
- Error Handling / Standard HTTP error code handling is a must.
  - So the `perfect` error message would consist of:
    - HTTP Status Code
    - Code ID - which may be an internal reference, you may also provide a link to the API documentation containing all the code id’s
    - Human readable message shortly explaining the error (its cause, context or possible remedy)
- Filtering, sorting, paging, and field selection
  - Few of the most important features for consuming an API are filtering, sorting and paging. Resource collections are oftentimes enormous, and when some data has to be retrieved from them, it would be simply not very efficient to always get the full list and browse it for specific items.
- API versioning
  - Versioning your REST API is a good approach to take right from the start. This will allow you to introduce changes to the data structure or specific actions, even if they are breaking/non-backward compatible changes. At some point, you might end up managing more than one API versions. But this will allow you to introduce modifications and improve services on one hand, and on another not to lose a part of your API’s users as some of them might be either reluctant to change (their integrations) or are just slow adopters that need time in order to introduce changes on their side.
- API Documentation.
- Always use SSL/TLS to encrypt the communication with your API. No exceptions. Period.
- References
  - <https://www.merixstudio.com/blog/best-practices-rest-api-development/>


#### Idempotent and safe HTTP methods in REST APIs

A method is safe if it does not modify the server's (internal) state. Safe methods are methods that can be cached, prefetched without any repercussions to the resource.

A method is idempotent if calling it multiple times has the same effect as calling it once (it doesn't change anything externally (response)).

Idempotent HTTP method is a HTTP method that can be called many times without different outcomes. The PUT and DELETE methods are defined to be idempotent. However, there is a caveat on DELETE. The problem with DELETE, which if successful would normally return a 200 (OK) or 204 (No Content), will often return a 404 (Not Found) on subsequent calls, unless the service is configured to "mark" resources for deletion without actually deleting them. However, when the service actually deletes the resource, the next call will not find the resource to delete it and return a 404. However, the state on the server is the same after each DELETE call, but the response is different. [Reference](https://stackoverflow.com/questions/56729786/difference-between-idempotent-and-safe-http-methods-in-rest-apis)

PATCH is not idempotent when it involves incremental updates (e.g., adding money, appending data), however it can be idempotent if it sets specific values rather than modifying them relatively.

PUT is always idempotent, as it replaces the entire resource.

| HTTP Method | Idempotent | Safe |
|-------------|------------|------|
| OPTIONS     | yes        | yes  |
| GET         | yes        | yes  |
 | HEAD        | yes        | yes  |
 | PUT         | yes        | no   |
| POST        | no         | no   |
 | DELETE      | yes        | no   |
 | PATCH       | no         | no   |



#### An example of REST API design

Resource objects often exhibit a functional hierarchy or interrelation. For example, in an online store, we have 'users' and 'orders'. Orders are always associated with a specific user. Hence, we might structure our endpoints as follows:

```text
/users               <- list of users
/users/123           <- specific user
/users/123/orders    <- list of orders belonging to a specific user
/users/123/orders/0001 <- specific order of a specific user
```

It's generally advisable to limit the nesting to one level in a REST API. Too many nested levels can make the API structure less elegant. Alternatively, similar results can often be achieved through filtering, e.g., /orders?user=123.

```text
GET /users?country=USA
GET /users?creation_date=2019-11-11
GET /users?creation_date=2019-11-11
```
```text
GET /users?sort=birthdate_date:asc
GET /users?sort=birthdate_date:desc
```
```text
GET /users?limit=100
GET /users?offset=2
```
```text
GET /users?country=USA&creation_date=2019-11-11&sort=birthdate_date:desc&limit=100&offset=2
```

#### Richardson maturity model

Reference: https://martinfowler.com/articles/richardsonMaturityModel.html

```text
Glory of REST
⬆
|
Level 3: Hypermedia Controls
|
Level 2: HTTP Verbs
|
Level 1: Resources
|
Level 0: The Swamp of POX
```

- Level 0: The Swamp of POX
  - At this level, the API uses HTTP as a transport mechanism for remote interactions but does not utilize the features of HTTP.
  - Often, APIs at this level expose a single endpoint, and interactions are handled through a single method (e.g., POST) with payloads that encapsulate both action and data.
  - The communication is typically in XML or JSON format but lacks structure and clear delineation of resources.
- Level 1: Resources
  - This level introduces the concept of resources, with each resource having its unique URI.
  - Resources can be accessed via different URIs, making the API more structured and navigable.
  - While the resources are identified, the interactions still do not fully leverage HTTP methods.
- Level 2: HTTP Verbs
  - At this level, the API uses HTTP methods (GET, POST, PUT, DELETE) appropriately.
    - Each HTTP method is used to perform specific actions on resources, following REST principles.
      - This level brings more standardization and a clearer interaction model with resources.
- Level 3: Hypermedia Controls (HATEOAS)
  - The final level introduces Hypermedia as the Engine of Application State (HATEOAS).
  - The API provides hypermedia links in the responses, guiding clients on available actions dynamically.
  - This level offers the most RESTful experience, allowing clients to discover and navigate the API interactively through links.

#### Miscellanios

- Can we use REST in SOA instead of SOAP?
  - Yes. SOAP and REST – both provide support for building SOA based application
- Does REST Api have to be HTTP protocol based?
  - REST isn't always linked to HTTP. You can use other transfer protocols, such as FTP, SMTP, etc. and your API can still be RESTful. Any API that uses HTTP as its transfer protocol is referred to as an HTTP API.
  - Ref: <https://hevodata.com/learn/http-api-vs-rest-api/>
- How to implement PATCH (partial update)?
  - Send a map of changed attributes only
    - private Map<String, Object> changedAttrs = new HashMap<>();
  - Use GSON instead of Jackson to distinguish null fields.
  - Instead of Jackson class mapping, use JsonNode and manually check keys and associate with attribute. If a class field is missing, it means it was not changed.
  - Add a boolean to each attribute e.g.
    - private String firstName;
    - private boolean isFirstNameDirty;
  - Ref: <https://stackoverflow.com/questions/38424383/how-to-distinguish-between-null-and-not-provided-values-for-partial-updates-in-s>

### gRPC

RPC is a generic protocol for remote procedure calls, while gRPC is a specific implementation of RPC that uses the HTTP/2 protocol for communication.
Also, RPC uses a binary encoding format to transmit data, while gRPC supports several serialization formats, including Protocol Buffers, JSON, and XML.

The gRPC programming API in most languages comes in both synchronous and asynchronous flavors.

![img](docs/grpc_0401.png)

![grpc](docs/grpc.png)

References:
- https://grpc.io/docs/what-is-grpc/introduction/
- https://www.oreilly.com/library/view/grpc-up-and/9781492058328/ch04.html

---

Protocol buffers are a combination of the definition language (created in .proto files),
the code that the proto compiler generates to interface with data, language-specific runtime libraries,
and the serialization format for data that is written to a file (or sent across a network connection).

Advantages of using protocol buffers include:
- Compact data storage
- Fast parsing
- Availability in many programming languages
- Optimized functionality through automatically-generated classes

![img](docs/protocol-buffers-concepts.png)

Ref: https://protobuf.dev/overview/

#### More about RPC

Remote Procedure Call (RPC) is a protocol that one program can use to request a service from a program located in another computer on a network without having to understand the network's details. A procedure call is also sometimes known as a function call or a subroutine call.

RPC uses the client-server model. The requesting program is a client and the service providing program is the server. Like a regular or local procedure call, an RPC is a synchronous operation requiring the requesting program to be suspended until the results of the remote procedure are returned. However, the use of lightweight processes or threads that share the same address space allows multiple RPCs to be performed concurrently.

When program statements that use RPC framework are compiled into an executable program, a stub is included in the compiled code that acts as the representative of the remote procedure code. When the program is run and the procedure call is issued, the stub receives the request and forwards it to a client runtime program in the local computer.

RPC provides both blocking (synchronous) and non-blocking (asynchronous) calls.

![rpc](docs/rpc.jpg)

Ref: https://stackoverflow.com/questions/49628943/rpc-remote-procedure-call-process

#### RPC vs gRPC

In contrast to REST and RPC, gRPC overcomes issues related to speed and weight — and offers greater efficiency when sending messages — by using the Protobuf (protocol buffers) messaging format. Here are a few details about Protobuf:
- Platform and language agnostic like JSON
  Serializes and deserializes structured data to communicate via binary
- As a highly compressed format, it doesn’t achieve JSON’s level of human readability
- Speeds up data transmission by removing many responsibilities JSON manages so it can focus strictly on serializing and deserializing data
- Data transmission is faster because Protobuf reduces the size of messages and serves as a lightweight messaging format

Ref: https://blog.dreamfactory.com/grpc-vs-rest-how-does-grpc-compare-with-traditional-rest-apis/#:~:text=In%20contrast%20to%20REST%20and,and%20language%20agnostic%20like%20JSON

#### JSON-RPC

JSON-RPC uses the lightweight JSON format to encode data, while XML-RPC uses the more verbose XML format. Using JSON-RPC, an application can send a message to another app requesting that it perform a function, such as the processing of data. With JSON’s strong capabilities in data description, this usually works well. It’s an open, globally identified protocol.

A JSON-RPC request message can contain three possible elements: The method, which is a string that names the method to be invoked; params, which are objects or arrays of values that get passed along as parameters to the destination app; and id, a string or number that matches the response with the request that it is replying to.

The app that receives the JSON-RPC request proceeds to issue a response. The response includes a result, which is the data generated by the invoked method, and the request/response ID. It may also include an error if there is a problem with the receiving app.

Ref: https://nonamesecurity.com/learn-what-is-json-rpc

#### gRPC/RPC vs REST

- REST API Is An Architectural Style, While RPC Is A Protocol
- REST Uses A Uniform Interface, While RPC Does Not.
- REST Is Stateless, While RPC Is Stateful.

```
Placing an Order:

RPC: http://MyRestaurant:8080/Orders/PlaceOrder (POST: {Tacos object})
REST: http://MyRestaurant:8080/Orders/Order?OrderNumber=asdf (POST: {Tacos object})

Retrieving an Order:

RPC: http://MyRestaurant:8080/Orders/GetOrder?OrderNumber=asdf (GET)
REST: http://MyRestaurant:8080/Orders/Order?OrderNumber=asdf (GET)

Updating an Order:

RPC: http://MyRestaurant:8080/Orders/UpdateOrder (PUT: {Pineapple Tacos object})
REST: http://MyRestaurant:8080/Orders/Order?OrderNumber=asdf (PUT: {Pineapple Tacos object})
```

Does REST can use binary data instead of text?
Yes absolutely. When you display a web page, your browser makes a GET request for each image on the page, and receives binary data in return.

References:
- https://narasimmantech.com/a-better-way-to-build-apis-rest-api-vs-rpc-and-grpc-vs-graphql/
- https://stackoverflow.com/questions/26830431/web-service-differences-between-rest-and-rpc
- https://aws.amazon.com/compare/the-difference-between-rpc-and-rest/

### GraphQL

GraphQL is a query language and server-side runtime for application programming interfaces (APIs) that prioritizes
giving clients exactly the data they request and no more. As an alternative to REST, GraphQL lets developers construct
requests that pull data from multiple data sources in a single API call.

When developing a GraphQL API there are two popular approaches to create the GraphQL Schema: the schema-first approach and the code-first. The schema-first consists of building the Schema using the Schema Definition Language while the code-first uses a programming language to create the Schema.

References:
- https://www.redhat.com/en/topics/api/what-is-graphql
- https://graphql.org/
- https://formidable.com/blog/2021/graphql-with-nexus/

![Graphql](./docs/GraphQL_Architecture-1024x461.png)

![Graphql](./docs/rest-vs-graphql.png)

---

Development remarks:
- it's not possible to not return anything
  - https://github.com/ardatan/graphql-tools/issues/277
  - https://stackoverflow.com/questions/44737043/is-it-possible-to-not-return-any-data-when-using-a-graphql-mutation
- you always need to specify which data you want back while doing queries

#### Altair vs GraphiQL vs Voyager vs Playground

All mentioned above can be primarily classified as "GraphQL" tools.

- GraphiQL is an in-browser IDE for exploring GraphQL. Ref: https://www.npmjs.com/package/graphql-voyager
- Altair is a GraphQL API client to explore APIs. Altair provides a good alternative to traditional GraphiQL and Playground.
- GraphQL Voyager represents any GraphQL API as an interactive visual.

Below there is a demo fo GraphQL Voyager.
![demo](docs/demo-gif.gif)

#### HTTP vs WebSocket

GraphQL is a specification and it's common to see GraphQL over HTTP for queries and mutations, however with GraphQL
subscriptions we need to receive continuous updates from an API. That's where WebSockets come in.

WebSockets are often used as a transport protocol for GraphQL Subscriptions.

Subscriptions are useful for notifying your client in real time about changes to back-end data, such as the creation of a new object or updates to an important field.

Reference: https://stackoverflow.com/questions/67659937/what-are-differences-between-graphql-subscription-and-websocket-protocol

#### GraphQL vs REST API

A REST API is an `architectural concept` for network-based software.
GraphQL, on the other hand, is a query language and a set of tools that operate over a single endpoint.

References:
- https://hygraph.com/blog/graphql-vs-rest-apis
- https://aws.amazon.com/compare/the-difference-between-graphql-and-rest/


#### How to solve n+1 problem when using GraphQL?

- When querying for nested data, like in the music example, a scaling issue known as the n+1 problem can occur.
- In the example, the query will fetch a list of musicians. Let’s say it finds n musicians in the database. For each musician found, the albums() resolver will be invoked to locate all the albums associated with that musician. This resolver will trigger a database call for each musician, which will be n calls. This means that in total, there are n+1 database calls occurring. This is much less efficient than having two database calls, one for musicians and one for albums.
- There are well-established solutions for handling it. These include using batching or using data loaders on the client.
- When using a data loader, the fetcher responds with a promise, then moves to the next fetch at the same data level rather than moving onto the nested data. A promise is a proxy for the response of a function that allows processing to continue. The code guarantees that a response for the call will come later, making it a non-blocking function. Once all the data at one level is retrieved (or once all the promises have been fulfilled), a single request is made to get all the nested data.
- Reference: https://hygraph.com/blog/graphql-n-1-problem

## HTTP/2

HTTPS is just the HTTP protocol but with data encryption using SSL/TLS.

Secure Sockets Layer (SSL) is an older, less-secure version of the cryptographic protocol, and Transport Layer Security (TLS) is its successor.

*There is `http2` profile which can be used to run locally this protocol as POC.*

=> https://localhost:8443/reporting/swagger-ui/index.html

![HTTP 2](./docs/tls/http1-http2-http3.png)

Reference: https://www.wallarm.com/what/what-is-http-2-and-how-is-it-different-from-http-1

#### Generating a certificate for local development

Reference: https://byte27.com/2020/02/03/using-http-2-in-your-spring-boot-application/

On Windows:
1. Generate a certificate and the private key using openssl with the command (you must be in `docs` folder):
    ```
    openssl req -x509 -out localhost.crt -keyout localhost.key -newkey rsa:2048 -nodes -sha256 -subj '//CN=localhost' -extensions EXT -config ./certificate.cnf
    ```
2. Generate pkcs12.
   ```
   openssl pkcs12 -export -in localhost.crt -inkey localhost.key -name localhost -out localhost.p12
   ```

 readmeEtag: '"274e0fe037546dcb9a1eec3bbcd939d651f795cd"' readmeLastModified: Sun, 02 Feb 2025 11:17:41 GMT repositoryId: 655553383 description: A simple Spring Cloud demo project - Eureka, Feign, Ribbon ... created: '2023-06-19T06:21:22Z' updated: '2025-03-10T00:53:10Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: maciek1839 logo: https://avatars.githubusercontent.com/u/16246708?v=4 license: MIT repoEtag: '"42a803309a4faf32e5389b48ead35ef2879871722d1729359f53ce73e06e8aa4"' repoLastModified: Mon, 10 Mar 2025 00:53:10 GMT category: Server Implementations foundInMaster: true - name: Apidog source: Tooling repository issues source_description: >- Apidog is an integrated collaboration platform for API documentation, API debugging, API mocking, and API automated testing that combines Postman + Swagger + Mock + JMeter to tackle the data synchronization problem among different systems using a set of systems and a set of data. API debugging, API data mocking, and API automated testing can be directly used without redefining as long as the API documentation is well-defined. link: https://www.apidog.com v3_1: true v3: true v2: true sourceIssueMetadata: issueNumber: 107 author: YukioIkeda createdAt: '2023-12-05T08:08:51Z' updatedAt: '2024-01-26T17:28:47Z' url: https://github.com/OAI/Tooling/issues/107 status: open id: 552adcd550190c0876a18aa706dc2f42 foundInMaster: true - source: https://openapi.tools/ name: APIMatic Developer Experience Portal category: Documentation link: https://www.apimatic.io/developer-experience-portal language: SaaS source_description: >- Customizable developer portals packed with language specific documentation, client libraries, code samples, an API console and much more. v2: true v3: true v3_1: true id: 268b5835cde4b94e2507583308b4720d foundInMaster: true - source: https://openapi.tools/ name: APITree category: Documentation repository: https://github.com/apitree link: https://apitree.com/ language: SaaS source_description: >- HUB for managing and sharing APIs. Converts OpenAPI v2 / v3 files into beautiful API documentation. v2: true v3: true foundInMaster: true id: 8ec6d6f8d9a5447a64cb9059253c7196 - source: https://openapi.tools/ name: Atom/linter-swagger category: Text Editors language: JavaScript repository: https://atom.io/packages/linter-swagger license: MIT source_description: >- This plugin for Atom Linter will lint OpenAPI, both JSON and YAML using swagger-parser node package. v2: true v3: true foundInMaster: true id: 184d87fb65f889fa033f623a7b67df1a - source: https://openapi.tools/ name: Atom/linter-openapi category: Text Editors language: JavaScript repository: https://atom.io/packages/linter-openapi license: MIT source_description: >- This plugin for Atom Linter will lint OpenAPI YAML files using openapi-enforcer node package. v2: false v3: true foundInMaster: true id: 60dd32764dfe435b0eaaa2189ab1bc8c - source: https://openapi.tools/ name: Swagger Inspector category: Learning language: SaaS link: https://swagger.io/tools/swagger-inspector/ source_description: >- Run mock requests in a webapp and Swagger Inspector infers your OpenAPI description. v2: true v3: true foundInMaster: true id: 4fe9bd57d14e8597029d05f3e477dc61 - source: https://openapi.tools/ name: Swagger Inspector category: Testing language: - Self-hosted - SaaS link: https://inspector.swagger.io repository: null source_description: >- Swagger Inspector is a free online tool to quickly execute any API request, validate its responses and generate a corresponding OpenAPI Description. v2: true v3: true foundInMaster: true id: 7cb46761fe06ef24609940d2efe877fc - source: openapi3 tags repository: https://github.com/devjack/example-library-specification v3: true repositoryMetadata: base64Readme: >- IyBFeGFtcGxlIExpYnJhcnkgU3BlY2lmaWNhdGlvbgpUaGlzIHJlcG9zaXRvcnkgZGVmaW5lcyBhIHNpbXBsZSBBUEkgdXNpbmcgW09wZW5BUEkgdjMuMC4xXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8zLjAuMS5tZCkuCgpbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLm9yZy9kZXZqYWNrL2V4YW1wbGUtbGlicmFyeS1zcGVjaWZpY2F0aW9uLnN2Zz9icmFuY2g9bWFzdGVyKV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL2RldmphY2svZXhhbXBsZS1saWJyYXJ5LXNwZWNpZmljYXRpb24pCgojIFNldHVwIGFuZCBpbnN0YWxsYXRpb24KClRoZSBBUEkgc3BlYyBpcyBkZWZpbmVkIGluIGBvcGVuYXBpLnltbGAuIEVhY2ggVVJMIGhhcyBpdHMgb3duIGVuZHBvaW50IGluIGBlbmRwb2ludHMvYCBhbmQgSlNPTiByZXF1ZXN0L3Jlc3BvbnNlIHNjaGVtYXMgYXJlIGRlZmluZWQgaW4gYHNjaGVtYXMvYC4gSlNPTiBTY2hlbWEgaXMgdXNlZCB0byBkZWZpbmUgcmVzcG9uc2VzIGFuZCBjYW4gYmUgdXNlZCBmb3IgdmFsaWRhdGlvbi4KCiMjIFNldHVwIGFuZCBydW4gdGhlIGRvY3VtZW50YXRpb24KUnVuIHRoZXNlIGNvbW1hbmRzIHRvIGluc3RhbGwgbnBtIGRlcGVuZGVuY2llcyBhbmQgc2VydmUgdGhlIEFQSSBkb2N1bWVudGF0aW9uIGxvY2FsbHkuCgpgYGBzaApucG0gaW5zdGFsbApucG0gcnVuIHRlc3QKbnBtIHJ1biBzZXJ2ZQpgYGAKClRoZSBBUEkgZG9jdW1lbnRhdGlvbiB3aWxsIGJlIGF2YWlsYWJsZSBhdCBbaHR0cHM6Ly9sb2NhbGhvc3Q6NTAwMF0oaHR0cDovL2xvY2FsaG9zdDo1MDAwKS4KClRvIHJlc29sdmUgYW5kIHB1Ymxpc2ggdGhlIEFQSSBzcGVjLCBydW46CgpgYGBzaApucG0gcnVuIHJlc29sdmUKYGBgCgpBIHNpbXBsZSBIVE1MIGRvY3VtZW50IGluY2x1ZGluZyBbUmVEb2NdKGh0dHBzOi8vZ2l0aHViLmNvbS9SZWJpbGx5L1JlRG9jKSBpcyBhdmFpbGFibGUgaW4gdGhlIGBwdWJsaWMvYCBkaXJlY3RvcnkuIFRvIGNvcHkgYWNyb3NzIHRoZSBwdWJsaXNoZWQgc3BlYyBhbmQgc2NoZW1hcywgcnVuOgoKYGBgc2gKbnBtIHJ1biBwdWJsaXNoCmBgYAoKWW91IGNhbiB0aGVuIHNlcnZlIGAvcHVibGljYCBmcm9tIGFueSBzdGF0aWMgY29udGVudCBob3N0aW5nIHNlcnZpY2UuIFRoZSBjdXJyZW50IGRvY3VtZW50YXRpb24gaXMgaG9zdGVkIG9uIE5ldGxpZnkgYXQgW2V4YW1wbGUtbGlicmFyeS1zcGVjaWZpY2F0aW9uLm5ldGxpZnkuY29tXShodHRwczovL2V4YW1wbGUtbGlicmFyeS1zcGVjaWZpY2F0aW9uLm5ldGxpZnkuY29tKQo= readmeEtag: '"25e85ab53a5680328ba38e6f15c69e46bf9835ed"' readmeLastModified: Sun, 07 Oct 2018 13:20:15 GMT repositoryId: 151935342 description: Using OpenAPI and JSON Schema to build a design-first API created: '2018-10-07T11:41:30Z' updated: '2021-11-30T21:32:02Z' language: HTML archived: false stars: 0 watchers: 0 forks: 0 owner: devjack logo: https://avatars.githubusercontent.com/u/3516066?v=4 repoEtag: '"f72fb5b7f2e30e5d28030858afe1f18dd03ac3327d0b5eeb7cba7bde21b25781"' repoLastModified: Tue, 30 Nov 2021 21:32:02 GMT foundInMaster: true category: Parsers id: ab8fe5700b9eb0a48246ca741696382e - source: openapi3 tags repository: https://github.com/andres-gr/kc-web v3: true repositoryMetadata: repositoryId: 233687856 description: KC Webapp new boilerplate setup created: '2020-01-13T20:33:41Z' updated: '2020-04-07T01:27:00Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: andres-gr logo: https://avatars.githubusercontent.com/u/26035230?v=4 repoEtag: '"af1a176d91014cbe413b7ece82facfc197f8fa99d242802d7ec5f0d3c542394c"' repoLastModified: Tue, 07 Apr 2020 01:27:00 GMT foundInMaster: true id: e4b19b83c7127702991be56913c18374 - source: openapi3 tags repository: https://github.com/pschichtel/betterswagger v3: true repositoryMetadata: repositoryId: 272568695 description: A new code generator for OpenAPI 3 specs created: '2020-06-16T00:00:41Z' updated: '2020-11-04T16:40:44Z' language: Scala archived: false stars: 0 watchers: 1 forks: 0 owner: pschichtel logo: https://avatars.githubusercontent.com/u/526104?v=4 repoEtag: '"7507db75713ad3f51e1f451f9c95b8a07e94902fe39dfeb07f2e454615cbb6bf"' repoLastModified: Wed, 04 Nov 2020 16:40:44 GMT foundInMaster: true id: 3922409f14bb62251229ba475ee4be90 - source: openapi3 tags repository: https://github.com/navjotsrakhra/eventmanagementplatform v3: true id: e52c196c2a8bbc78a26da1a421c09e5e repositoryMetadata: base64Readme: >- IyBFdmVudCBNYW5hZ2VyIEFwcGxpY2F0aW9uCgpUaGlzIGlzIHRoZSBFdmVudCBNYW5hZ2VyIEFwcGxpY2F0aW9uLCBhIEphdmEtYmFzZWQgd2ViIGFwcGxpY2F0aW9uIGRlc2lnbmVkIGZvciBtYW5hZ2luZyBldmVudHMgYW5kIHVzZXIgYWNjb3VudHMuIEl0CnByb3ZpZGVzIGZ1bmN0aW9uYWxpdHkgZm9yIGNyZWF0aW5nIGFuZCB2aWV3aW5nIGV2ZW50IHBvc3RzLCB1c2VyIHJlZ2lzdHJhdGlvbiBhbmQgYXV0aGVudGljYXRpb24sIGFuZCB1c2VyIG1hbmFnZW1lbnQKZm9yIGFkbWluaXN0cmF0b3JzLgoKIyBOb3RlOiBUbyBzZWUgdGhlIGRlbW8gdXNlIHRoZSBmb2xsb3dpbmcgCiAtIHVzZXJuYW1lOiBhZG1pbiwgcGFzc3dvcmQ6IGFkbWluTgogLSBUaGUgQVBJIGVuZHBvaW50cyBjYW4gYmUgZm91bmQgYXQgaHR0cHM6Ly9ldmVudG1hbmFnZW1lbnRwbGF0Zm9ybS5vbnJlbmRlci5jb20vYXBpLWluZm8KCiMjIFRhYmxlIG9mIENvbnRlbnRzCgo8IS0tIFRPQyAtLT4KCiogW0V2ZW50IE1hbmFnZXIgQXBwbGljYXRpb25dKCNldmVudC1tYW5hZ2VyLWFwcGxpY2F0aW9uKQogICAgKiBbRmVhdHVyZXNdKCNmZWF0dXJlcykKICAgICogW0RvY3VtZW50YXRpb25dKCNkb2N1bWVudGF0aW9uKQogICAgKiBbUHJlcmVxdWlzaXRlc10oI3ByZXJlcXVpc2l0ZXMpCiAgICAqIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCiAgICAqIFtVc2FnZV0oI3VzYWdlKQogICAgKiBbTGljZW5zZV0oI2xpY2Vuc2UpCgo8IS0tIFRPQyAtLT4KCiMjIEZlYXR1cmVzCgotIENyZWF0ZSBhbmQgbWFuYWdlIGV2ZW50IHBvc3RzIHdpdGggZGV0YWlscyBzdWNoIGFzIHRpdGxlLCBjb250ZW50LCBsb2NhdGlvbiwgc3RhcnQgZGF0ZSwgYW5kIGVuZCBkYXRlLgotIFVzZXIgcmVnaXN0cmF0aW9uIGFuZCBhdXRoZW50aWNhdGlvbiB3aXRoIHJvbGUtYmFzZWQgYWNjZXNzIGNvbnRyb2wuCi0gVXNlciBtYW5hZ2VtZW50IGZvciBhZG1pbmlzdHJhdG9ycyB0byBtYW5hZ2UgdXNlciByb2xlcy4KLSBTZWN1cmUgcGFzc3dvcmQgaGFzaGluZyB1c2luZyBCQ3J5cHQuCi0gUkVTVGZ1bCBBUEkgZm9yIGV2ZW50IHBvc3RzLgotIEV4Y2VwdGlvbiBoYW5kbGluZyBmb3IgdmFsaWRhdGlvbiBhbmQgZXJyb3IgcmVzcG9uc2VzLgoKIyMgRG9jdW1lbnRhdGlvbgoKaHR0cHM6Ly9uYXZqb3RzcmFraHJhLmdpdGh1Yi5pby9FdmVudE1hbmFnZW1lbnRQbGF0Zm9ybS8KCiMjIFByZXJlcXVpc2l0ZXMKCkJlZm9yZSB5b3UgYmVnaW4sIGVuc3VyZSB5b3UgaGF2ZSBtZXQgdGhlIGZvbGxvd2luZyByZXF1aXJlbWVudHM6CgotIEphdmEgRGV2ZWxvcG1lbnQgS2l0IChKREspIDIwIG9yIGxhdGVyCi0gUG9zdGdyZVNRTCBkYXRhYmFzZQoKIyMgSW5zdGFsbGF0aW9uCgoxLiBDbG9uZSB0aGUgcmVwb3NpdG9yeToKCiAgIGBgYHNoCiAgIGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vTmF2am90U1Jha2hyYS9FdmVudE1hbmFnZW1lbnRQbGF0Zm9ybS5naXQKCjIuIE5hdmlnYXRlIHRvIHRoZSBwcm9qZWN0IGRpcmVjdG9yeToKICAgYGBgc2gKICAgY2QgRXZlbnRNYW5hZ2VtZW50UGxhdGZvcm0KCjMuIENvbmZpZ3VyZSB5b3VyIGRhdGFiYXNlIHNldHRpbmdzIGluIGBzcmMvbWFpbi9qYXZhL2lvL2dpdGh1Yi9uYXZqb3RzcmFraHJhL2V2ZW50bWFuYWdlci9jb25maWcvRGF0YVNvdXJjZUNvbmZpZ2AKCjQuIEJ1aWxkIHRoZSBwcm9qZWN0IHVzaW5nIE1hdmVuOgogICBgYGBzaAogICAuL212bncgY2xlYW4gaW5zdGFsbAoKIyMgVXNhZ2UKCjEuIFJ1biB0aGUgYXBwbGljYXRpb24uCiAgIGBgYHNoCiAgIG12biBzcHJpbmctYm9vdDpydW4KCjIuIEFjY2VzcyB0aGUgYXBwbGljYXRpb24gYXQgJ2h0dHA6Ly9sb2NhbGhvc3Q6ODA4MCcKMy4gWW91IGNhbiByZWdpc3RlciBhcyBhIG5ldyB1c2VyIG9yIHVzZSB0aGUgcHJvdmlkZWQgYWRtaW5pc3RyYXRvciBhY2NvdW50IHRvIGxvZ2luCgojIyBMaWNlbnNlCgpDb3B5cmlnaHQgKGMpIDIwMjMgTmF2am90IFNpbmdoIFJha2hyYS4gQWxsIHJpZ2h0cyByZXNlcnZlZC4K readmeEtag: '"a79e926f0009a7e3a622d3d867240619f964625a"' readmeLastModified: Mon, 12 Feb 2024 20:12:47 GMT repositoryId: 677508545 description: >- Event management platform to notify about events along with the event day, time and location created: '2023-08-11T18:53:21Z' updated: '2026-01-25T11:31:36Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: NavjotSRakhra logo: https://avatars.githubusercontent.com/u/45345158?v=4 repoEtag: '"c3d00b38109c1af2c8ee71595d1ed566fd103f6f6aa43216367922f60f3bd4ae"' repoLastModified: Sun, 25 Jan 2026 11:31:36 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/bwcash/web-api-spring-boot v3: true id: 35bd99787664a0ddccf56293404d6f37 repositoryMetadata: base64Readme: IiMgV2ViLUFQSS1TcHJpbmctQm9vdCIgCiIjIFdlYi1BUEktU3ByaW5nLUJvb3QiIAo= readmeEtag: '"e9b9311710d3e1ca45bddfc0653d07beea21f40a"' readmeLastModified: Fri, 20 Jan 2023 22:49:58 GMT repositoryId: 578707859 description: STS maven web API project created: '2022-12-15T17:35:23Z' updated: '2022-12-15T18:48:45Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Bwcash logo: https://avatars.githubusercontent.com/u/112275746?v=4 repoEtag: '"4db7a362bc6fe60f9099e21547b83429a0cf4f782275e51317a8f74d8f3d857a"' repoLastModified: Thu, 15 Dec 2022 18:48:45 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/dm1st/eatsy-api v3: true id: d13f0bc48a2f7a8abc858920b7c72fe6 repositoryMetadata: base64Readme: >- # Eatsy API

A spring boot REST API that user-facing services can interface with for finding, creating and sharing your favourite
recipes.

[![CircleCI build](https://circleci.com/gh/DM1st/eatsy-api/tree/develop.svg?style=shield)](https://circleci.com/gh/DM1st/eatsy-api/tree/develop)
[![Coverage Status](https://coveralls.io/repos/github/DM1st/eatsy-api/badge.svg?branch=develop)](https://coveralls.io/github/DM1st/eatsy-api?branch=develop)
[![Create and publish a Docker image](https://github.com/DM1st/eatsy-api/actions/workflows/publish.yml/badge.svg)](https://github.com/DM1st/eatsy-api/actions/workflows/publish.yml)

* A live demo version of the Eatsy API is deployed and the Swagger UI Spec for the service can be
  found [here](https://eatsy-api.onrender.com/swagger-ui/index.html#/). Please note, the application uses free-tier
  services, so it will take a few minutes for the API service to spin up.
* To try out the Eatsy API via a UI, you can use the [eatsy-react-ui](https://github.com/DM1st/eatsy-react-ui/) which is
  a user-friendly React application that allows users to interact with the Eatsy API. A live demo version is deployed
  and integrated with the Eatsy API [here](https://eatsy.onrender.com).

Together, these components (along with a Postgres database) make up the end-to-end Eatsy solution and the live demo
instances are hosted on the [Render](https://render.com) platform.

## Getting started

The live demo version of the service is deployed using the dockerfile in the project root. However, you can deploy your
own instance locally:

### Build and deploy the spring boot service locally

The server is run as a [Sprint Boot](https://spring.io/projects/spring-boot) application. For simplicity reasons it
uses [The Gradle Wrapper](https://docs.gradle.org/current/userguide/gradle_wrapper.html) (eliminating the need to
install Gradle locally). This helps ensure version consistency, easy setup, build reproducibility, and improved build
performance.

#### Pre-requisites

* JDK (version 8 or higher) installed and configured on your local machine to be able to complie and run the Java code.
* Before running the service, you will need to have a Postgres instance installed and need to update the Spring Boot
  ```application.properties``` file to configure the connection details (e.g. host, port, username and password) to
  align to your database instance that the application will use to connect to it. For example:

```
##local dev PostgreSQL datasource properties
spring.datasource.url=jdbc:postgresql://localhost:5432/eatsy
spring.datasource.username=<your postgres username>
spring.datasource.password=<your postgres password>
spring.jpa.show-sql=true
```

```application.properties``` can be found at:

```<projectRoot>\eatsyAppService\eatsyAppService-controller\src\main\resources\application.properties```

#### Running the service can then be done from a bash terminal:

```
./gradlew clean build bootrun
```

Once the server is running navigate to http://localhost:8080/swagger-ui.html to see the Swagger UI Spec.

#### Please note:

If initiating through an IDE such as intelliJ (rather than command line), you will need to ensure the "Working
Directory"
is correctly set (to the same as the Spring Boot application runner) otherwise you will get initialisation errors when
trying to locate logging config and other properties. The working directory in your IDE run configuration should be set
to:

```C:\<location to repository root>\eatsyAppService\eatsyAppService-controller```

#### Populating your local instance with random recipe test data

Random recipe generation can be triggered externally and persisted for manual testing and verification purposes. With
the server running, open a new bash terminal (at the project root) and run the following command:

```
./gradlew generateRandomRecipes
```

This executes a custom gradle task that generates a random number of recipes (between 1 and 15) which then get posted to
your local application's `addRecipe` endpoint. This can be executed as many times as needed to reach the level of test
data required.

If there is a need to remove all data in the database, the following commands can be used to drop all the tables:

```
DROP SCHEMA public CASCADE;
CREATE SCHEMA public;
```

This works because all the tables are in a single schema named public. If you are using PostgreSQL 9.3 or later, you may
also need to restore the default grants.

```
GRANT ALL ON SCHEMA public TO postgres;
GRANT ALL ON SCHEMA public TO public;
```

## Project Structure and configuration

This application uses Gradle as the build tool and is structured as Gradle sub-projects. This helps separate the
different responsibilities and concerns of the application into distinct modules. This contributes to keeping the
codebase organized and maintainable, and makes it easier to test, debug and deploy the application.

The structure of the Gradle projects looks like this, with each sub-project focussing on a specific aspect of the
application:

```sh
eatsyAppService          # Umbrella Gradle project
|
|
+-- controller           # Contains the controllers of the application. 
|                        # Responsible for handling HTTP requests and 
|                        # returning the appropriate response. 
|                        # They interact with the services of the application 
|                        # to retrieve data.
|
|
+-- domain               # Contains the domain objects of the application. 
|                        # These represent the entities and data that the 
|                        # application manipulates. 
|                        # Some business logic is also contained here.
|
|
|
+-- model                # Contains the data transfer objects (DTOs) of the application. 
|                        # In this case, they are used to map the domain objects 
|                        # to a representation that can be sent via HTTP requests/responses
|                        # to other independent components that interact with this API.
|                                        
|
|
+-- model-mappers        # Contains the mappers of the application. 
|                        # The Mappers are responsible for mapping between the DTOs, 
|                        # Domain objects and Entity objects (that get persisted to 
|                        # the database) and they provide a way to convert between
|                        # these representations as needed.
|
|
+-- persistence          # Contains the persistence layer of the application. 
|                        # Responsible for storing and retrieving data (via the Entity 
|                         # objects), and interacting with the postgres database.
|
|
|
|
+-- service              # Contains the services of the application. 
|                        # These are responsible for implementing the business logic 
|                        # of the application.
|
|
|
|
+-- test-data-generation # Generates random test data that is used by the unit tests 
|                        # in all the above sub-projects. This sub project helps  
|                        # improve the reliability of the unit tests. 
|                        # By generating random test data, the tests will test edge cases that 
|                        # may not have been considered with hard-coded test data. 
|                        # This helps catch unexpected behavior and ensures that 
|                        # the code works as intended.
```

- Each sub-project has its own build configuration dependencies (in the respective ```build.gradle``` files) and tests
  cases.
- Changes in one sub-project won't affect the others.
- Improved build performance: This Gradle sub-projects structure allows for building only the parts of the application
  that have changed. This can result in faster build times.
- Better modularity: This Gradle sub-projects structure allows you to develop and deploy your application as modular
  components, which makes it easier to reuse code across different projects and create smaller, more focused artifacts.

### Logging

This project uses log4J2 with the logging config (```application.properties``` and ```log4j2.properties```) located in:

```eatsyAppService\eatsyAppService-controller\src\main\resources```

Logs for all Gradle sub-projects write to ```logfile.log``` located in the ```eatsyAppService-controller```  project:

```\eatsyAppService\eatsyAppService-controller\logs```

Logging levels for the logfile can be changed in log4j2.properties file.

### Testing

This project uses [JUnit 5](https://junit.org/junit5/) as the unit test framework. In addition, the below libraries and
frameworks are also used for testing:

#### Faker

[Faker](https://github.com/DiUS/java-faker) has been used for generating random, localised test data in a variety of
formats. The data generated is used in unit tests to validate the behavior of the application. By generating random test
data, the tests will test edge cases that may not have been considered with hard-coded test data. This helps catch
unexpected behavior and ensures that the code works as intended.

As mentioned in the getting started section, random recipe generation can also be triggered externally and persisted for
manual testing and verification purposes.

With the server running, open a new bash terminal (at the project root) and run the following command:

```
./gradlew generateRandomRecipes
```

This executes a custom gradle task that generates a random number of recipes (between 1 and 15) which then get posted to
your local application's `addRecipe` endpoint. This can be executed as many times as needed to reach the level of test
data required

#### Mockito

[Mockito](https://site.mockito.org/) is a mocking framework for Java that has been used in the unit tests to isolate the
code being tested from its dependencies. By using mock objects in the unit tests, the project avoids having to set up a
full test environment, which speeds up test execution and makes the tests more focussed on the unit of code under test.

#### Coveralls

[Coveralls](https://coveralls.io/) is a web-based service that provides this project with test coverage analysis (as
illustrated in the coverage badge at the top of this readme). This allows for codebase test coverage to be analysed and
identify areas where more tests are needed. This has been used to help ensure that the code is thoroughly tested and to
reduce the risk of bugs in production.

The coveralls task can be executed with the following command at the project root:

``` ./gradlew coveralls ```

The html report can be found at ```project root > build > reports > jacoco > jacocoRootReport > html > index.html```

### API Documentation

A JSON Swagger spec can be found at '/v3/api-docs'. Locally this will be 'http://localhost:8080/v3/api-docs'.

A UI version can be seen locally using the [Swagger UI page](http://localhost:8080/swagger-ui.html).

This Swagger specification is a file that specifies the structure of the API. It is a user-friendly interface that
defines the endpoints, requests and response formats and the data models that are used by the API. This makes it easier
to generate client-side code for consuming the API (such as
the [eatsy-react-ui](https://github.com/DM1st/eatsy-react-ui/)) since the specification provides all necessary
information about the API's structure, therefore reducing the risk of defects.

The Swagger Spec is created with the ```springdoc-openapi-ui``` library and configured in the ```openApiConfiguration```
file located in the ```eatsyAppService-controller``` project.

#### Manual inspection of a local API deployment

As mentioned above, this service has unit tests (using [JUnit 5](https://junit.org/junit5/)
, [Mockito](https://site.mockito.org/)
and [Faker](https://github.com/DiUS/java-faker)) with test coverage displayed on the coverage badge in this ReadMe.
However, this Swagger UI provides a way for the API to be tested manually by making requests and seeing the response
directly in the browser. This allows for easier debugging and validation of the API's behaviour. Example below:

* Once the service is running navigate to the [Swagger UI page](http://localhost:8080/swagger-ui.html).
* Select the 'Api Controller'
* Select the 'POST' request for adding a new recipe.
* Select "Try it out" and enter a request body of your choice as a test. E.g.

```
{
  "ingredientSet": [
    "Beans", "Toast"
  ],
  "method": {
    "1": "Mircowave Beans",
    "2": "Toast Bread",
    "3": "Combine Beans and Toast"
  },
  "name": "Beans on Toast"
}
```

* Select "Execute".
* Inspect the server response and the response body will display the new recipe object that has been created with your
  chosen name.

### Deploy your own instance (via Docker & Docker Compose):

**Pre-requisite:** Docker & Docker Compose installed on your machine if you wish to build and run the Docker images in
the repository. The method offers an alternative approach by allowing Postgres, PgAdmin and the Eatsy-API to be deployed
and run as Docker containers.

The built Docker image of the Eatsy API can be found on
DockerHub [here](https://hub.docker.com/r/dm1st/eatsy-api-docker)

- Copy the docker-compose.yml from the repository located at:

```<projectRoot>\alternativeDeploymentOptions\docker-compose.yml```

*_Please note this is not the docker-compose.yml file located at project root (which builds the image from the
dockerfile rather than this config which pulls the latest built image from dockerhub).

- Transfer the alternativeDeploymentOptions\docker-compose.yml file to your machine where you will be running the
  service. (Your machine will need Docker and Docker-Compose installed as detailed in the pre-requisite)
- Run the following command in the directory where you have placed the docker-compose.yml file to run the application:

```
docker-compose up -d
```

This command will run the eatsy API service, the Postgres DB and pgAdmin in Docker Containers.

For future deployments, explicitly check the dockerhub repository to ensure you are building the image with the latest
changes:

```
docker pull dm1st/eatsy-api-docker && docker-compose up -d
```

- The service can be spun down with the ```docker-compose down'``` command.

With Docker-Compose the Postgres container is started and established first as the API service depends on the
postgresSQL container being up and healthy.

- Navigate to the Swagger UI page to see the API service running

```http://<your_host>:8080/swagger-ui.html```

* You can also follow the above steps, but with the ```docker-compose.yml``` file located at the project root to build
  an image from the project dockerfile, rather than pulling the built image hosted on dockerhub. If using this approach,
  you will need the codebase stored locally on your host (rather than just the docker-compose.yml file located in the
  alternativeDeploymentOptions folder).

### The live instance of this application on Render.com

As mentioned above, the Eatsy API is deployed on Render by using the ```dockerfile``` in the project root. As this
project uses the free-tier products on [Render.com](https://render.com), therefore only the most lightweight
infrastructure is available. As a result, in order to make deployment possible, and reduce image spin-up times (when the
service is cold)
every effort has been made to reduce the docker image size. This includes:

* Layering the Spring boot Jar file so that it is built into 4 layers. This is done in the Controller
  project ```build.gradle``` file.
* The decomposition of the application into different layers allows for the creation of an efficient Docker image, by
  creating a multi-stage ```dockerfile``` (located in the project root). Efficiency is improved and start up time of the
  docker image reduced, especially as it is being run on limited infrastructure.

## TODO

* Add Google shopping list integration.
* Add ability to search for recipe.
* Add search capability for best match based on search ingredients.
* Add the concept of users.
* Add component test pack.
 readmeEtag: '"1237c133b1d7a02179fec8ebbf19c82a71a86be8"' readmeLastModified: Sun, 02 Apr 2023 20:27:59 GMT repositoryId: 432201412 description: 'Eatsy App Service for creating and editing your favourite recipies! ' created: '2021-11-26T14:12:29Z' updated: '2023-02-16T20:31:48Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: DM1st logo: https://avatars.githubusercontent.com/u/30046844?v=4 license: MIT repoEtag: '"2752dfa94bf43b2b48e64cdd398605d75c507b54504b3c0088a2cdc6e9bf63e2"' repoLastModified: Thu, 16 Feb 2023 20:31:48 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/dm1st/eatsy - source: openapi3 tags repository: https://github.com/clustermarket/clustermarket-api v3: true repositoryMetadata: base64Readme: >- Q2x1c3Rlcm1hcmtldCBBUEkKPT09PT09PT09PT09PT09PT0KCkZ1bGwgZG9jdW1lbnRhdGlvbiBhdmFpbGFibGUgb246IFthcGlkb2NzLmNsdXN0ZXJtYXJrZXQuY29tXShodHRwczovL2FwaWRvY3MuY2x1c3Rlcm1hcmtldC5jb20vKQo= readmeEtag: '"09a18d4280e5619f03096dcdf808eeb2a05d58e6"' readmeLastModified: Wed, 01 Mar 2023 13:42:34 GMT repositoryId: 419906738 description: Mirror of https://gitlab.com/clustermarket/clustermarket-api/ created: '2021-10-21T23:32:48Z' updated: '2022-06-24T11:32:52Z' language: JavaScript archived: false stars: 0 watchers: 0 forks: 0 owner: clustermarket logo: https://avatars.githubusercontent.com/u/82224373?v=4 repoEtag: '"769598e791f2b361bb8c1787a2b451f3700820ca145ca09461ad961b01e33d68"' repoLastModified: Fri, 24 Jun 2022 11:32:52 GMT foundInMaster: true category: Server id: b6fe55f17fe5c615f86cf6909318a46b - source: openapi3 tags repository: https://github.com/semestry/api-specs v3: true repositoryMetadata: base64Readme: >- IyBhcGktc3BlY3MKClRoaXMgcmVwb3NpdG9yeSBjb250YWlucyBBUEkgc3BlY2lmaWNhdGlvbnMgYW5kIGdlbmVyYXRlZCBkb2N1bWVudGF0aW9uIGZvciBFdmVvaCdzIEFQSXMuCgojIyBQdWJsaXNoaW5nCgpUaGUgT3BlbkFQSSAzLjAgc3BlY2lmaWNhdGlvbnMgYXJlIGdlbmVyYXRlZCB1c2luZyBSZURvYy4gClByb2R1Y3Rpb24gQVBJIGRvY3VtZW50YXRpb24gcGFnZXMgYXJlIGF1dG9tYXRpY2FsbHkgcHVibGlzaGVkIGJ5IEdpdGh1YiBQYWdlcyB3aGVuIGNoYW5nZXMgYXJlIHB1c2hlZCB0byB0aGUgYG1hc3RlcmAgYnJhbmNoLgoKIyMgRGV2ZWxvcG1lbnQKCiMjIyBWaWV3aW5nIGRvY3VtZW50YXRpb24gbG9jYWxseQoKU3BlY2lmaWNhdGlvbnMgY2FuIGJlIHNlcnZlZCBsb2NhbGx5IHVzaW5nIFlhcm4gY29tbWFuZHM6CgpgeWFybiBzZXJ2ZTo8c3BlYz5gCgpGb3IgZXhhbXBsZToKCmB5YXJuIHNlcnZlOmVjaG8td2ViaG9va3NgCgpEb2N1bWVudGF0aW9uIGlzIGF2YWlsYWJsZSBhdCBgaHR0cDovL2xvY2FsaG9zdDozMDAwYC4KCkNoYW5nZXMgdG8gc291cmNlIGZpbGVzIGFyZSBhdXRvbWF0aWNhbGx5IHByb2Nlc3NlZCwgYWx0aG91Z2ggeW91ciBicm93c2VyIHNob3VsZCBiZSBtYW51YWxseSByZWZyZXNoZWQuCgojIyMgTGludGluZyBzcGVjaWZpY2F0aW9ucwoKTGludGluZyBpcyBwZXJmb3JtZWQgdXNpbmcgU3BlY2N5LgoKTGludCB0aGUgc3BlY3MgdXNpbmcgdGhlIGZvbGxvd2luZyBZYXJuIGNvbW1hbmQ6CgpgeWFybiBsaW50YAoKIyMjIEFkZGluZyBhIEFQSSBzcGVjaWZpY2F0aW9uCgotIEFkZCB0aGUgT3BlbkFQSSAzLjAgZmlsZSB0byBgL3NwZWNgLgotIENyZWF0ZSBhIEhUTUwgZmlsZSBpbiBgL2RvY3NgIHNlcnZpbmcgdGhlIHNwZWNpZmljYXRpb24uCi0gQWRkIGEgbGluayB0byB0aGUgc3BlY2lmaWNhdGlvbiBpbiBgL2RvY3MvaW5kZXguaHRtbGAuIAotIEFkZCBhIGBsaW50OjxzcGVjPmAgc2NyaXB0IGluIGBwYWNrYWdlLmpzb25gLgotIEFkZCBhIGBzZXJ2ZTo8c3BlYz5gIHNjcmlwdCBpbiBgcGFja2FnZS5qc29uYC4K readmeEtag: '"c6c5d9b0cfded6b2f34d656acbd3edec7b43aa82"' readmeLastModified: Mon, 10 Aug 2020 07:45:20 GMT repositoryId: 187231129 description: Eveoh API specifications created: '2019-05-17T14:31:35Z' updated: '2024-07-01T14:26:19Z' language: null archived: true stars: 0 watchers: 6 forks: 0 owner: semestry logo: https://avatars.githubusercontent.com/u/1628298?v=4 repoEtag: '"dd8fbb265ca12937b501fc06d3331b9cc3660087ab154344ca190c1030dd2090"' repoLastModified: Mon, 01 Jul 2024 14:26:19 GMT foundInMaster: true category: - Documentation - Parsers - Server Implementations id: da535d66a142ee78c5bc699dd795fd45 - source: openapi3 tags repository: https://github.com/koriit/openapi-matcher v3: true repositoryMetadata: base64Readme: >- PSBPcGVuQVBJIE1hdGNoZXIKCmltYWdlOmh0dHBzOi8vZ2l0aHViLmNvbS9Lb3JpaXQvb3BlbmFwaS1tYXRjaGVyL2FjdGlvbnMvd29ya2Zsb3dzL2J1aWxkLnlhbWwvYmFkZ2Uuc3ZnW0J1aWxkLCBsaW5rPSJodHRwczovL2dpdGh1Yi5jb20vS29yaWl0L29wZW5hcGktbWF0Y2hlci9hY3Rpb25zL3dvcmtmbG93cy9idWlsZC55YW1sIl0KaW1hZ2U6aHR0cHM6Ly93d3cuY29kZWZhY3Rvci5pby9yZXBvc2l0b3J5L2dpdGh1Yi9rb3JpaXQvb3BlbmFwaS1tYXRjaGVyL2JhZGdlW0NvZGVGYWN0b3IsbGluaz1odHRwczovL3d3dy5jb2RlZmFjdG9yLmlvL3JlcG9zaXRvcnkvZ2l0aHViL2tvcmlpdC9vcGVuYXBpLW1hdGNoZXJdCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvY29kZSUyMHN0eWxlLSVFMiU5RCVBNC1GRjQwODEuc3ZnW2t0bGludCxsaW5rPWh0dHBzOi8va3RsaW50LmdpdGh1Yi5pby9dCgppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9jb20ua29yaWl0LmtvdGxpbi9vcGVuYXBpLW1hdGNoZXIuc3ZnP2xhYmVsPU1hdmVuJTIwQ2VudHJhbFtNYXZlbiBDZW50cmFsLCBsaW5rPSJodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9ZzolMjJjb20ua29yaWl0LmtvdGxpbiUyMiUyMEFORCUyMGE6JTIyb3BlbmFwaS1tYXRjaGVyJTIyIl0KaW1hZ2U6aHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9rb3RsaW4tMS42LjEwLWJsdWUuc3ZnP2xvZ289a290bGluW0tvdGxpbiwgbGluaz1odHRwOi8va290bGlubGFuZy5vcmddCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2Uva29yaWl0L29wZW5hcGktbWF0Y2hlcltHaXRIdWJdCgpXQVJOSU5HOiBGcm9tIHZlcnNpb24gMC43LjAgYWxsIHBhY2thZ2UgbmFtZXMgaGF2ZSBiZWVuIHJlbmFtZWQgdG8gbWF0Y2ggbmV3IGFydGlmYWN0IGdyb3VwIGlkLgoKT3BlbkFQSSBtYXRjaGVyIGZvciBLb3RsaW4gYWxsb3dzIGNvbXBhcmluZyAyIHNwZWNpZmljYXRpb25zIGFnYWluc3QgZWFjaCBvdGhlci4KCltXQVJOSU5HXQpUaGlzIGxpYnJhcnkgaW4gKldvcmsgSW4gUHJvZ3Jlc3MqLiBJdCBkb2Vzbid0IHN1cHBvcnQgYWxsIGVsZW1lbnRzIG9mIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiB5ZXQuIFRoZSBzdXBwb3J0IHNjb3BlIGNhbiBiZQpjaGVja2VkIGluZGl2aWR1YWxseSBmb3IgZXZlcnkgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIG9iamVjdCBpbiB0aGUgYGtvcnJpdC5rb3RsaW4ub3BlbmFwaS5tb2RlbGAgcGFja2FnZS4KCgo9PSBSZWFkaW5nIE9wZW5BUEkKVGhpcyBsaWJyYXJ5IGluY2x1ZGVzIGFuIE9wZW5BUEkgcmVhZGVyIHRoYXQgYWxsb3dzIHBhcnNpbmcgeW91ciBPcGVuQVBJIHNwZWNpZmljYXRpb24gaW4gWUFNTCBmb3JtYXQgaW50bwphbiBpbi1tZW1vcnkgb2JqZWN0IHRoYXQgY2FuIGJlIHVzZWQgaW4gYSBtYXRjaGVyLgoKW3NvdXJjZSxrb3RsaW5dCi0tLS0KdmFsIGRvYzogT3BlbkFQSSA9IE9wZW5BUElSZWFkZXIoKS5sb2FkKHt9LmphdmFDbGFzcy5nZXRSZXNvdXJjZUFzU3RyZWFtKCIvb3BlbmFwaS55YW1sIikpCi0tLS0KCltOT1RFXQpVbnN1cHBvcnRlZCBlbGVtZW50cyBhcmUgaWdub3JlZC4KCj09IFZhbGlkYXRpbmcgT3BlbkFQSQpPbmNlIHlvdSBoYXZlIDIgT3BlbkFQSSBvYmplY3RzIHlvdSBjYW4gbWF0Y2ggdGhlbSBjaGVja2luZyBmb3IgYW55IHNpZ25pZmljYW50IGRpZmZlcmVuY2VzLgoKQSBzaWduaWZpY2FudCBkaWZmZXJlbmNlIGlzIHVuZGVyc3Rvb2QgYXMgYSBkaXZlcmdlbmNlIHRoYXQgbGVhZHMgdG8gb3IgbWF5IGxlYWQgdG8gYSBjb21wYXRpYmlsaXR5IGJyZWFrLgoKVGhlIGdlbmVyYWwgaWRlYSBpcyB0byB2YWxpZGF0ZSB5b3VyIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBhZ2FpbnN0IHlvdXIgY29kZSBpbnN0ZWFkIG9mIHNwZWMvY29kZSBnZW5lcmF0aW9uLgoKW3NvdXJjZSxrb3RsaW5dCi0tLS0KdmFsIHNvdXJjZTogT3BlbkFQSSA9IC4uLiAvLyBBbmFseXplIHlvdXIgY29kZQoKTE9HLmluZm8oIlJlYWRpbmcgT3BlbkFQSSBzcGVjLi4uIikKdmFsIGRvYzogT3BlbkFQSSA9IE9wZW5BUElSZWFkZXIoKS5sb2FkKHt9LmphdmFDbGFzcy5nZXRSZXNvdXJjZUFzU3RyZWFtKCIvb3BlbmFwaS55YW1sIikpCgpMT0cuaW5mbygiVmFsaWRhdGluZyBzcGVjLi4uIikKdmFsIGVycm9ycyA9IE9wZW5BUElNYXRjaGVyKCkubWF0Y2goZG9jLCBzb3VyY2UpCgppZiAoZXJyb3JzLmlzTm90RW1wdHkoKSkgewogICAgTE9HLmluZm8oIlJlc3VsdCBvZiBjb2RlIGFuYWx5c2lzOlxue30iLCBzb3VyY2UpCgogICAgZXJyb3JzLmZvckVhY2ggewogICAgICAgIExPRy5lcnJvcihpdCkKICAgIH0KCiAgICBmYWlsKCJUaGVyZSBhcmUgJHtlcnJvcnMuc2l6ZX0gdmFsaWRhdGlvbiBlcnJvcnMhIikKfSBlbHNlIHsKICAgIExPRy5pbmZvKCJPSyEiKQp9Ci0tLS0KCg== readmeEtag: '"8f67f47d94cef85082b804cc49af58678253f961"' readmeLastModified: Sun, 30 Jan 2022 20:15:37 GMT repositoryId: 217255706 description: >- OpenApi matcher for kotlin, allows to compare 2 specifications against each other. created: '2019-10-24T08:58:03Z' updated: '2022-01-30T20:19:36Z' language: Kotlin archived: false stars: 0 watchers: 1 forks: 0 owner: Koriit logo: https://avatars.githubusercontent.com/u/8916393?v=4 license: MIT repoEtag: '"72ab6a75c17eea1ba5a7fb37d13908b7da60ea9bf16b3a67ea87e5cc678fac81"' repoLastModified: Sun, 30 Jan 2022 20:19:36 GMT foundInMaster: true category: Data Validators id: 8a1ce2f4655ce6c57fd36e72479933f1 - source: openapi3 tags repository: https://github.com/lantrix/phpipam-swagger-openapi v3: true repositoryMetadata: base64Readme: >- IyBwaHBpcGFtLXN3YWdnZXItb3BlbmFwaQoKcGhwSVBBTSBPcGVuQVBJIDMgX2FrYV8gU3dhZ2dlciBTcGVjIGJhc2VkIHVwb24gdGhlIFtwaHBJUEFNIEFQSSBEb2N1bWVudGF0aW9uXShodHRwczovL3BocGlwYW0ubmV0L2FwaS9hcGlfZG9jdW1lbnRhdGlvbi8pCgpJbXBvcnQgdGhlIE9wZW5BUEkgW1lBTUwtUmVzb2x2ZWQgc3BlY10oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL2xhbnRyaXgvcGhwaXBhbS1zd2FnZ2VyLW9wZW5hcGkvbWFzdGVyL3lhbWwtcmVzb2x2ZWQvcGhwaXBhbS1zd2FnZ2VyLnlhbWwpIGludG8gdGhlIHRvb2wgb2YgeW91ciBjaG9pY2UsIGUuZy4gW0luc29tbmlhXShodHRwczovL2luc29tbmlhLnJlc3QpIG9yIFtQb3N0bWFuXShodHRwczovL3d3dy5nZXRwb3N0bWFuLmNvbSkKCkV4cG9ydCB0aGUgQ2xpZW50U0RLIG9yIFNlcnZlclN0dWIgYXMgbmVlZGVkIG92ZXIgYXQgdGhlIFtTd2FnZ2VySHViXShodHRwczovL2FwcC5zd2FnZ2VyaHViLmNvbS9hcGlzL2xhbnRyaXgvcGhwaXBhbS8pLgo= readmeEtag: '"1b0c553b7fb6f7a05dd13ba8ee4e2dfad8ce0ac8"' readmeLastModified: Wed, 02 Oct 2019 05:23:55 GMT repositoryId: 212253611 description: PHPIPAM OpenAPI 3 aka Swagger created: '2019-10-02T04:14:43Z' updated: '2019-10-02T05:23:58Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: lantrix logo: https://avatars.githubusercontent.com/u/117008?v=4 license: GPL-3.0 repoEtag: '"2abdffd9ff39de8bc51a98a0c18efde60634391c76100a40692810909b8bde41"' repoLastModified: Wed, 02 Oct 2019 05:23:58 GMT foundInMaster: true category: Code Generators id: 1eb337fca8933c898f918e1eef1ddc47 - source: openapi3 tags repository: https://github.com/ahelord/boilerplate-express-sequelize v3: true repositoryMetadata: base64Readme: >- IyBib2lsZXJwbGF0ZS1leHByZXNzLXNlcXVlbGl6ZQoKIyBSdW4gRG9ja2VyCgogICBgYGBiYXNoCiAgIGRvY2tlci1jb21wb3NlIHN0YXJ0CmBgYAoKIyMgVXNlZnVsIERvY2tlciBjb21tYW5kcwoKMS4gSWYgeW91IHdhbnQgdG8gY2hlY2sgdGhhdCBhbGwgY29udGFpbmVycyBhcmUgdXAgOgoKICAgYGBgYmFzaAogICBkb2NrZXItY29tcG9zZSBwcwogICBgYGAKCjEuIE90aGVyIERvY2tlciBjb21tYW5kcyA6CgogICBgYGBiYXNoCiAgICMgU3RhcnQgRG9ja2VyCiAgIGRvY2tlci1jb21wb3NlIHN0YXJ0CgogICAjIFJlc3RhcnQgRG9ja2VyCiAgIGRvY2tlci1jb21wb3NlIHJlc3RhcnQKCiAgICMgU3RvcCBEb2NrZXIKICAgZG9ja2VyLWNvbXBvc2Ugc3RvcAoKICAgIyBEZWxldGUgYWxsIGNvbnRhaW5lcnMKICAgZG9ja2VyIHJtICQoZG9ja2VyIHBzIC1hcSkKCiAgICMgRGVsZXRlIGFsbCBpbWFnZXMKICAgZG9ja2VyIHJtaSAkKGRvY2tlciBpbWFnZXMgLXEpCiAgIGBgYAoKMS4gSG93IHRvIGdldCBhIERvY2tlciBjb250YWluZXIncyBJUCBhZGRyZXNzIGZyb20gdGhlIGhvc3QgPwoKICAgYGBgYmFzaAogICBkb2NrZXIgaW5zcGVjdCAtZiAne3tyYW5nZSAuTmV0d29ya1NldHRpbmdzLk5ldHdvcmtzfX17ey5JUEFkZHJlc3N9fXt7ZW5kfX0nIDxjb250YWluZXI+CiAgIGRvY2tlciBpbnNwZWN0ICQoZG9ja2VyIHBzIC1mIG5hbWU9PHNlcnZpY2U+IC1xKSB8IGdyZXAgSVBBZGRyZXNzCiAgIGBgYAoKLS0tCgojIFJ1biBMb2NhbCBvbmx5IG5vZGUgMjAKLSBjZCBzcmMKLSBudm0gdXNlIDIwCi0gbnBtIGkKLSBucG0gcnVuIG5vZGVtb24KCg== readmeEtag: '"04288db065a3df71eb99754dbc431e89d682f127"' readmeLastModified: Thu, 12 Sep 2024 05:19:48 GMT repositoryId: 280018723 description: 'boilerplate express sequelize swagger docker eslint husky precommit ' created: '2020-07-16T01:08:26Z' updated: '2024-09-12T05:19:54Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: ahelord logo: https://avatars.githubusercontent.com/u/21299810?v=4 repoEtag: '"f4fb8d23dca44f0314f33c938d4e0fccb86142aebeaff82afef4cc81fef7014a"' repoLastModified: Thu, 12 Sep 2024 05:19:54 GMT foundInMaster: true category: - Code Generators - Server Implementations id: eb31b899b4a7df11b8fd35752f563e92 - source: openapi3 tags repository: https://github.com/rgreinho/pyconsql v3: true repositoryMetadata: base64Readme: >- IyBweWNvbnNxbAoKVGhpcyBpcyBhIHNpbXBsZSBwcm9vZiBvZiBjb25jZXB0IHRvIHNob3cgaG93IHRvIGJ1aWxkIGEgbW9kZXJuIFJFU1QgQVBJIGluIHB5dGhvbi4KCiMjIEFic3RyYWN0CgpUaGlzIFBPQyB1c2VzIHRoZSBwZXRzdG9yZSBBUEkgc2NoZW1hIGFzIGFuIGV4YW1wbGUuCgpUaGUgbWFpbiBjb21wb25lbnRzIGFyZToKCiogW2Nvbm5leGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL3phbGFuZG8vY29ubmV4aW9uKSB3aXRoCiAgW2Fpb2h0dHBdKGh0dHBzOi8vZG9jcy5haW9odHRwLm9yZy9lbi9zdGFibGUvKSBhcyBhIGJhY2tlbmQKKiBbZ3VuaWNvcm5dKGh0dHBzOi8vZ3VuaWNvcm4ub3JnLykgd2l0aCB0aGUKICBbdXZsb29wXShodHRwczovL2dpdGh1Yi5jb20vTWFnaWNTdGFjay91dmxvb3ApIHdvcmtlcgoqIFtPcGVuQWxjaGVteV0oaHR0cHM6Ly9naXRodWIuY29tL2pka2FuZGVyc3Nvbi9PcGVuQWxjaGVteSkgdG8gZ2VuZXJhdGUgdGhlIG1vZGVscyBvbgogIHRoZSBmbHkgZnJvbSB0aGUgc3BlY2lmaWNhdGlvbgoqIFtTUUxBbGNoZW15XShodHRwczovL3d3dy5zcWxhbGNoZW15Lm9yZy8pIGZvciB0aGUgT1JNCiogW2FsZW1iaWNdKGh0dHBzOi8vYWxlbWJpYy5zcWxhbGNoZW15Lm9yZy9lbi9sYXRlc3QvKSB0byBtYW5hZ2UgdGhlIG1pZ3JhdGlvbnMKKiBbc2NoZW1hdGhlc2lzXShodHRwczovL2dpdGh1Yi5jb20va2l3aWNvbS9zY2hlbWF0aGVzaXMpIHRvIHRlc3Qgb3VyIEFQSQoKT3RoZXIgaW50ZXJlc3RpbmcgbGlicmFyaWVzIGFyZToKCiogW2xvZ3VydV0oaHR0cHM6Ly9naXRodWIuY29tL0RlbGdhbi9sb2d1cnUpIGZvciBiZXR0ZXIgbG9nZ2luZwoqIFt0ZW5hY2l0eV0oaHR0cHM6Ly9naXRodWIuY29tL2pkL3RlbmFjaXR5KSB0byBoYW5kbGUgcmV0cmllcwoqIFt1anNvbl0oaHR0cHM6Ly9naXRodWIuY29tL3VsdHJhanNvbi91bHRyYWpzb24pIGZvciB1bHRyYSBmYXN0IEpTT04gcHJvY2Vzc2luZyAodG8gYmUKICBjb21wYXJlZCB3aXRoIFtvcmpzb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9pamwvb3Jqc29uKSB0aG91Z2gsIGFzIHVqc29uIGlzIHNsaWdodGx5CiAgaW5jb21wYXRpYmxlIHdpdGggdGhlIHN0ZGxpYikKCiMjIFNldHVwCgpJbnN0YWxsIHRoZSBQT0M6CgpgYGBiYXNoCnBvZXRyeSBpbnN0YWxsCmBgYAoKUnVuIHRoZSBtaWdyYXRpb25zOgoKYGBgYmFzaAptYWtlIG1pZ3JhdGUKYGBgCgpTdGFydCB0aGUgc2VydmVyOgoKYGBgYmFzaAptYWtlIGxvY2FsLWFwaQpgYGAKCiMjIFBsYXkgd2l0aCBpdAoKRnJvbSBhbm90aGVyIHRlcm1pbmFsLCB5b3UgY2FuIG5vdyBwZXJmb3JtIHRoZSBmb2xsb3dpbmcgYWN0aW9uczoKClBvcHVsYXRlIHRoZSBkYXRhYmFzZToKCmBgYGJhc2gKY3VybCAtWCBQT1NUICJodHRwOi8vMC4wLjAuMDo4MDAwL2FwaS9wZXRzIiBcCiAgLUggImFjY2VwdDogYXBwbGljYXRpb24vanNvbiIgXAogIC1kICd7Im5hbWUiOiAidGlnZXIiLCAidGFnIjogIndpbGQifScKY3VybCAtWCBQT1NUICJodHRwOi8vMC4wLjAuMDo4MDAwL2FwaS9wZXRzIiBcCiAgLUggImFjY2VwdDogYXBwbGljYXRpb24vanNvbiIgXAogIC1kICd7Im5hbWUiOiAibW9sbHkiLCAidGFnIjogImhvbWVidWRkeSJ9JwpgYGAKClJldHJpZXZlIGFsbCB0aGUgcGV0czoKCmBgYGJhc2gKY3VybCAtcyAtWCBHRVQgImh0dHA6Ly8wLjAuMC4wOjgwMDAvYXBpL3BldHMiIC1IICJhY2NlcHQ6IGFwcGxpY2F0aW9uL2pzb24iIHwganEKYGBgCgpSZXRyaWV2ZSBhIHNwZWNpZmljIHBldDoKCmBgYGJhc2gKY3VybCAtcyAtWCBHRVQgImh0dHA6Ly8wLjAuMC4wOjgwMDAvYXBpL3BldHMvMSIgLUggImFjY2VwdDogYXBwbGljYXRpb24vanNvbiIgfCBqcQpgYGAKCiMjIFRlc3QgaXQKCmBgYGJhc2gKcG9ldHJ5IHJ1biBzY2hlbWF0aGVzaXMgcnVuIGh0dHA6Ly8wLjAuMC4wOjgwMDAvYXBpL29wZW5hcGkuanNvbgpgYGAKClRoZSBlbmRwb2ludCBgUE9TVCAvYXBpL3BldHNgIGZhaWxzIGJlY2F1c2UgdGhlcmUgaXMgbm8gdmFsaWQgZXhhbXBsZSBhc3NvY2lhdGVkIHRvIGl0CmluIHRoZSBzcGVjaWZpY2F0aW9uIGZpbGUuCgojIyBDbGVhbiB1cAoKRGVzdHJveSB0aGUgREIgYW5kIGFzc29jaWF0ZWQgbWlncmF0aW9uczoKCmBgYGJhc2gKbWFrZSBjbGVhbnVwCmBgYAoKIyMgV2hhdCdzIG5lZWRlZCB0byBtYWtlIGl0IHByb2R1Y3Rpb24gcmVhZHkKCiogSW1wcm92ZSB0aGUgY29uZmlndXJhdGlvbiBzeXN0ZW0KICAqIFBhcmFtZXRlcnMgc2hvdWxkIGhhdmUgc2Vuc2libGUgZGVmYXVsdCB2YWx1ZXMKICAqIEFsbCB0aGUgcGFyYW1ldGVycyB0aGF0IGNhbiBiZSBpbmZlcnJlZCBvciBjYWxjdWxhdGVkIHNob3VsZCBiZSBpbmZlcnJlZCBvciBjYWxjdWxhdGVkIChpLmUuIHRoZSBzcGVjZmlsZSBwYXRoKQoqIFJlbW92ZSBHVW5pY29ybgogICogQWx0aG91Z2ggaXQgZG9lcyBub3QgYnJpbmcgbXVjaCBvdmVyaGVhZCwgaXQgaXMgbm90IG5lZWRlZCBieSBtb2Rlcm4gYXBwbGljYXRpb24gZnJhbWV3b3JrcyBhbmQgdGhlIHNjYWxpbmcvaGVhbHRoY2hlY2svcmVzdGFydCB3aWxsIGJlIGhhbmRsZWQgYnkga3ViZXJuZXRlcwoqIFNoYXJlIHRoZSBEQiBjb25uZWN0aW9uIGJldHdlZW4gY29udHJvbGxlcnMsIHZpYSB0aGUgY29ubmV4aW9uIGFwcAogICogVGhpcyBpcyBub3Qgb2ZmaWNpYWwgc3VwcG9ydGVkLCBidXQgaGVyZSBpcyBhbmljZSB3b3JrYXJvdW5kOiBbU2hhcmUgYSBEQiBjb25uZXhpb24gd2hlbiB1c2luZyBhaW9odHRwIGJhY2tlbmRdKGh0dHBzOi8vZ2l0aHViLmNvbS96YWxhbmRvL2Nvbm5leGlvbi9pc3N1ZXMvODMzKQoqIFVzZSB1bHRyYSBmYXN0IGxpYnJhcmllcwogICogUmVwbGFjZSB0aGUgZGVmYXVsdCBldmVudCBsb29wIHdpdGggVVZMb29wCiogUmVwbGFjZSB0aGUgTWFrZWZpbGUgd2l0aCBbSW52b2tlXShodHRwczovL3d3dy5weWludm9rZS5vcmcvKSB0YXNrcwoqIENyZWF0ZSBhIG5pY2UgdGVtcGxhdGUKICAqIENvb2tpZWN1dHRlciAoPykKICAqIE9ubHkgdGhlIG5hbWUgb2YgdGhlIGFwcCBzaG91bGQgYmUgcmVxdWlyZWQKCiMjIENhbiBJIHB1c2ggaXQgZXZlbiBmdXJ0aGVyPwoKSXQgc3RhcnRzIHRvIGdldCBoYXJkIHRvIG9idGFpbiBiZXR0ZXIgcGVyZm9ybWFuY2UgdXNpbmcgUHl0aG9uLCBidXQgaGVyZSBhcmUgc29tZSBwb2ludGVycyB0byBwdXNoIGFzeW5jaW8gdG8gaXRzIGxpbWl0czoKCiogVXNlIFthc3luY3BnXShodHRwczovL2dpdGh1Yi5jb20vTWFnaWNTdGFjay9hc3luY3BnKSB0byBoYW5kbGUgUG9zdGdyZVNRTC4KICAqIEl0IG9ubHkgd29ya3Mgd2l0aCBbU1FMIEFsY2hlbXkgQ29yZV0oaHR0cHM6Ly9kb2NzLnNxbGFsY2hlbXkub3JnL2VuLzEzL2NvcmUvKSB0aG91Z2gsIHRoZXJlZm9yZSB0aGVyZSBtYXkgYmUgc29tZSBhZGp1dHNtZW50cyB0byBtYWtlIHRvIHN0aWxsIGJlIGFibGUgdG8gdXNlIE9wZW4gQWxjaGVteS4KKiBVc2UgdGhlIFthaW8tbGlic10oaHR0cHM6Ly9naXRodWIuY29tL2Fpby1saWJzKSB3aGVuZXZlciBwb3NzaWJsZS4KKiBDaGVjayB0aGUgW2Fpb2h0dHAgZXh0ZW5zaW9uc10oaHR0cHM6Ly9kb2NzLmFpb2h0dHAub3JnL2VuL3N0YWJsZS90aGlyZF9wYXJ0eS5odG1sI2Fpb2h0dHAtZXh0ZW5zaW9ucykuCgojIyBJIHN0aWxsIG5lZWQgbW9yZQoKVXNlIEdvIG9yIFJ1c3QhCg== readmeEtag: '"fe582ed3c9efbc4b85ca90f432bc397fdcbab375"' readmeLastModified: Mon, 10 Aug 2020 18:45:06 GMT repositoryId: 273532058 description: A modern REST server setup wih OpenAPI 3 created: '2020-06-19T15:52:29Z' updated: '2020-08-10T18:45:09Z' language: Python archived: false stars: 0 watchers: 0 forks: 0 owner: rgreinho logo: https://avatars.githubusercontent.com/u/6969134?v=4 license: MIT repoEtag: '"6318ca39601d642b81524ab3252d642d66b59402550a834208022aa4cb319866"' repoLastModified: Mon, 10 Aug 2020 18:45:09 GMT foundInMaster: true category: Server Implementations id: ae89adb4231331435d0be335778f0836 - source: openapi3 tags repository: https://github.com/iulian-stan/crud-flask-rest v3: true id: 7bb75db7936fd6fb318762046eddc7f2 repositoryMetadata: base64Readme: >- IyBGbGFzawpTaW1wbGUgQ1JVRCBhcHBsaWNhdGlvbiBpbXBsZW1lbnRlZCB1c2luZyBGbGFzayBmcmFtZXdvcmsKQWRkaXRpb25hbCBkZXBlbmRlbmNpZXMKKiBbY29ubmV4aW9uXShodHRwczovL2dpdGh1Yi5jb20vemFsYW5kby9jb25uZXhpb24pIC0gT3BlbkFQSSBkb2N1bWVudGF0aW9uIGdlbmVyYXRpb24KKiBbRmxhc2stU1FMQWxjaGVteV0oaHR0cHM6Ly9mbGFzay1zcWxhbGNoZW15LnBhbGxldHNwcm9qZWN0cy5jb20pLSBPUk0gKERCIGludGVyYWN0aW9uKQoqIFtGbGFzay1NYXJzaG1hbGxvd10oaHR0cHM6Ly9mbGFzay1tYXJzaG1hbGxvdy5yZWFkdGhlZG9jcy5pbykgLSBEYXRhIHNlcmlhbGl6YXRpb24vZGVzZXJpYWxpemF0aW9uCgojIyBJbnN0YWxsYXRpb24KQ3JlYXRlIGEgcHl0aG9uIHZpcnR1YWwgZW52aXJvbm1lbnQgKG9wdGlvbmFsKQpgYGBiYXNoCnB5dGhvbiAtbSB2ZW52IC5lbnYKLmVudlxTY3JpcHRzXGFjdGl2YXRlCmBgYApJbnN0YWxsIG9mIHRoZSBkZXBlbmRlbmNpZXMgZnJvbSAqcmVxdWlyZW1lbnRzLnR4dCoKYGBgYmFzaApwaXAgaW5zdGFsbCAtciByZXF1aXJlbWVudHMudHh0CmBgYAoKIyMgVXNhZ2UKU3RhcnQgdGhlIGFwcGxpY2F0aW9uIGJ5IHJ1bm5pbmcgKm1haW4ucHkqCmBgYGJhc2gKcHl0aG9uIG1haW4ucHkKYGBgCk5hdmlnYXRlIHRvIHRoZSBmb2xsb3dpbmcgVVJMIGluIHlvdXIgd2ViIGJyb3dzZXIgLSBbaHR0cDovLzEyNy4wLjAuMTo5MDkwL3YxLjAvdWkvXShodHRwOi8vMTI3LjAuMC4xOjkwOTAvdjEuMC91aS8pLiBJdCB3aWxsIG9wZW4gdGhlIE9wZW5BUEkgZ2VuZXJhdGUgQVBJIGRvY3VtZW50YXRpb24gYmFzZWQgb24gKm9wZW5hcGkueWFtbCouIFRoZSBhcHBsaWNhdGlvbiB3aWxsIGNyZWF0ZSBhbiBpbml0aWFsbHkgZW1wdHkgREIgKHNlZSAqaW5pdC5weSopLiBVc2UgdGhlIFBPU1Qgb3IgUFVUIG1ldGhvZHMgdG8gcG9wdWxhdGUgdGhlIERCIHdpdGggc29tZSBzYW1wbGUgZGF0YSBpbiBvcmRlciB0byBiZSBhYmxlIHRvIHRlc3Qgb3RoZXIgSFRUUCBtZXRob2RzLg== readmeEtag: '"32a0c94628b1ac4637750c10a7ba7547a11d341e"' readmeLastModified: Sat, 05 Nov 2022 15:07:37 GMT repositoryId: 241997756 description: Simple CRUD application implemented using Flask framework created: '2020-02-20T21:50:48Z' updated: '2022-11-06T16:24:35Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: iulian-stan logo: https://avatars.githubusercontent.com/u/13657730?v=4 license: MIT repoEtag: '"ca10922d4a31cae4f8283a6c80c25fd81add1689bf3754e7de6d53d061a0db51"' repoLastModified: Sun, 06 Nov 2022 16:24:35 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/iulian-stan/crud-flask - https://github.com/iulian-stan/flask-crud-application - https://github.com/iulian-stan/flask-example - https://github.com/Iulian-Stan/Flask_Example - source: openapi3 tags repository: https://github.com/dernasherbrezon/jsp-openapi v3: true repositoryMetadata: base64Readme: >- IyBqc3Atb3BlbmFwaSBbIVtCdWlsZCBTdGF0dXNdKGh0dHBzOi8vdHJhdmlzLWNpLmNvbS9kZXJuYXNoZXJicmV6b24vanNwLW9wZW5hcGkuc3ZnP2JyYW5jaD1tYXN0ZXIpXShodHRwczovL3RyYXZpcy1jaS5jb20vZGVybmFzaGVyYnJlem9uL2pzcC1vcGVuYXBpKSBbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1ydS5yMmNsb3VkLm9wZW5hcGklM0Fqc3Atb3BlbmFwaSZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1ydS5yMmNsb3VkLm9wZW5hcGklM0Fqc3Atb3BlbmFwaSkKClRhZyBsaWIgZm9yIHJlbmRlcmluZyBvcGVuYXBpIHNwZWNpZmljYXRpb24uIENTUyBhbmQgamF2YXNjcmlwdCBsaWJyYXJpZXMgYXJlIG5vdCBpbmNsdWRlZC4KCiogImJvb3RzdHJhcDQtb3BlbmFwaSIuIFJlbmRlciB1c2luZyBib290c3RyYXA0LgoKIyMgVXNhZ2UKCkZvciBhIGNvbXBsZXRlIHJlZmVyZW5jZSBzZWUgdGhlIFt0ZXN0XShodHRwczovL2dpdGh1Yi5jb20vZGVybmFzaGVyYnJlem9uL2pzcC1vcGVuYXBpL2Jsb2IvbWFzdGVyL3NyYy90ZXN0L3Jlc291cmNlcy93ZWJhcHAvaW5kZXguanNwKS4KCiMjIyBTdGVwIGJ5IHN0ZXAgZ3VpZGUKCkluY2x1ZGUgZGVwZW5kZW5jeToKCmBgYHhtbAo8ZGVwZW5kZW5jeT4KCTxncm91cElkPnJ1LnIyY2xvdWQub3BlbmFwaTwvZ3JvdXBJZD4KCTxhcnRpZmFjdElkPmpzcC1vcGVuYXBpPC9hcnRpZmFjdElkPgoJPHZlcnNpb24+MS4wPC92ZXJzaW9uPgo8L2RlcGVuZGVuY3k+CmBgYAoKU2V0dXAgY29udHJvbGxlciAoZm9yIGV4YW1wbGUgU3ByaW5nIE1WQyk6CgpgYGBqYXZhCkBSZXF1ZXN0TWFwcGluZygiL2FwaSIpCnB1YmxpYyBNb2RlbEFuZFZpZXcgbG9hZCgpIHRocm93cyBFeGNlcHRpb24gewoJT3BlbkFQSSBvcGVuYXBpID0gbmV3IE9wZW5BUElWM1BhcnNlcigpLnJlYWQoTG9hZEFwaS5jbGFzcy5nZXRDbGFzc0xvYWRlcigpLmdldFJlc291cmNlKCJvcGVuYXBpLmpzb24iKS5nZXRGaWxlKCkpOwoJTWFwPFN0cmluZywgT2JqZWN0PiBtb2RlbCA9IG5ldyBIYXNoTWFwPFN0cmluZywgT2JqZWN0PigpOwoJbW9kZWwucHV0KCJlbnRpdHkiLCBvcGVuYXBpKTsKCXJldHVybiBuZXcgTW9kZWxBbmRWaWV3KCJhcGkiLCBtb2RlbCk7Cn0KYGBgCgo+IE5vdGU6IGlmIEFQSSBzcGVjaWZpY2F0aW9uIGNhbm5vdCBiZSBjaGFuZ2VkIGluIHJ1bnRpbWUsIHRoZW4gaXQgaXMgYmV0dGVyIHRvIGNhY2hlIGl0IG9uIHN0YXJ0dXAuCgo+IE5vdGU6IGlvLnN3YWdnZXIudjMucGFyc2VyLk9wZW5BUElWM1BhcnNlciBpcyBhIHN3YWdnZXIgcGFyc2VyIGZvciB2MyBzcGVjaWZpY2F0aW9uLiBZb3UgY2FuIGltcGxlbWVudCB5b3VyIG93biBwYXJzZXIgb3IgdXNlIGFueSBjb21wYXRpYmxlLiBQYXJzZXIgc2hvdWxkIGJlIGFibGUgdG8gY29uc3RydWN0IGlvLnN3YWdnZXIudjMub2FzLm1vZGVscy5PcGVuQVBJIG1vZGVsLiAKCkNvbmZpZ3VyZSB0YWdsaWI6CgpgYGAKPCVAIHRhZ2xpYiBwcmVmaXg9Im9wZW5hcGkiIHVyaT0iaHR0cHM6Ly9naXRodWIuY29tL2Rlcm5hc2hlcmJyZXpvbi9qc3Atb3BlbmFwaSIgJT4KYGBgCgpVc2Ugb24gdGhlIHBhZ2U6CgpgYGBodG1sCjxib2R5PgoJPGRpdiBjbGFzcz0iY29udGFpbmVyIj4KCQk8b3BlbmFwaTpib290c3RyYXA0LW9wZW5hcGkgb3BlbmFwaT0iJHtlbnRpdHl9Ii8+Cgk8L2Rpdj4KPC9ib2R5PgpgYGAK readmeEtag: '"fc58981a9754d43efd8587b653da7f18da53bf4b"' readmeLastModified: Fri, 31 Mar 2023 20:44:12 GMT repositoryId: 224528587 description: JSP tag for rendering openapi created: '2019-11-27T22:37:39Z' updated: '2022-03-27T20:39:23Z' language: HTML archived: false stars: 0 watchers: 2 forks: 0 owner: dernasherbrezon logo: https://avatars.githubusercontent.com/u/1614424?v=4 license: Apache-2.0 repoEtag: '"63a6526f11a45663738eece85cbceca12a8f42e63e55950f355ff48546719185"' repoLastModified: Sun, 27 Mar 2022 20:39:23 GMT foundInMaster: true category: - Low-level Tooling - Parsers id: 5c313e281614ced14ecaf35afee7d90a - source: openapi3 tags repository: https://github.com/efficap-energie/liveobjects-go v3: true repositoryMetadata: base64Readme: >- # Go API client for liveobjects

API description for Live Objects service

## Overview
This API client was generated by the [OpenAPI Generator](https://openapi-generator.tech) project.  By using the [OpenAPI-spec](https://www.openapis.org/) from a remote server, you can easily generate an API client.

- API version: 2.13.3
- Package version: 1.0.0
- Build package: org.openapitools.codegen.languages.GoClientCodegen

## Installation

Install the following dependencies:

```shell
go get github.com/stretchr/testify/assert
go get golang.org/x/oauth2
go get golang.org/x/net/context
```

Put the package under your project folder and add the following in import:

```golang
import sw "./liveobjects"
```

To use a proxy, set the environment variable `HTTP_PROXY`:

```golang
os.Setenv("HTTP_PROXY", "http://proxy_name:proxy_port")
```

## Configuration of Server URL

Default configuration comes with `Servers` field that contains server objects as defined in the OpenAPI specification.

### Select Server Configuration

For using other server than the one defined on index 0 set context value `sw.ContextServerIndex` of type `int`.

```golang
ctx := context.WithValue(context.Background(), sw.ContextServerIndex, 1)
```

### Templated Server URL

Templated server URL is formatted using default variables from configuration or from context value `sw.ContextServerVariables` of type `map[string]string`.

```golang
ctx := context.WithValue(context.Background(), sw.ContextServerVariables, map[string]string{
	"basePath": "v2",
})
```

Note, enum values are always validated and all unused variables are silently ignored.

### URLs Configuration per Operation

Each operation can use different server URL defined using `OperationServers` map in the `Configuration`.
An operation is uniquely identifield by `"{classname}Service.{nickname}"` string.
Similar rules for overriding default operation server index and variables applies by using `sw.ContextOperationServerIndices` and `sw.ContextOperationServerVariables` context maps.

```
ctx := context.WithValue(context.Background(), sw.ContextOperationServerIndices, map[string]int{
	"{classname}Service.{nickname}": 2,
})
ctx = context.WithValue(context.Background(), sw.ContextOperationServerVariables, map[string]map[string]string{
	"{classname}Service.{nickname}": {
		"port": "8443",
	},
})
```

## Documentation for API Endpoints

All URIs are relative to *https://liveobjects.orange-business.com*

Class | Method | HTTP request | Description
------------ | ------------- | ------------- | -------------
*AccountingV1Api* | [**GetDailyStatisticsUsingGET2**](docs/AccountingV1Api.md#getdailystatisticsusingget2) | **Get** /api/v1/accounting/daily | Get daily accounting metrics (Beta).
*AccountingV1Api* | [**GetMonthlyStatisticsUsingGET2**](docs/AccountingV1Api.md#getmonthlystatisticsusingget2) | **Get** /api/v1/accounting/monthly | Get monthly accounting metrics.
*ApiKeysApi* | [**CreateApiKeyUsingPOST**](docs/ApiKeysApi.md#createapikeyusingpost) | **Post** /api/v0/apiKeys | Create an API key
*ApiKeysApi* | [**DeleteApiKeyUsingDELETE**](docs/ApiKeysApi.md#deleteapikeyusingdelete) | **Delete** /api/v0/apiKeys/{apiKeyId} | Delete an API key
*ApiKeysApi* | [**GetApiKeyFromAuthenticationUsingGET3**](docs/ApiKeysApi.md#getapikeyfromauthenticationusingget3) | **Get** /api/v0/apiKeys/current_key | getApiKeyFromAuthentication
*ApiKeysApi* | [**GetApiKeyUsingGET3**](docs/ApiKeysApi.md#getapikeyusingget3) | **Get** /api/v0/apiKeys/{apiKeyId} | Get an API key
*ApiKeysApi* | [**GetApiKeysUsingGET3**](docs/ApiKeysApi.md#getapikeysusingget3) | **Get** /api/v0/apiKeys | List API keys
*ApiKeysApi* | [**SetApiKeyDebugModeUsingPUT3**](docs/ApiKeysApi.md#setapikeydebugmodeusingput3) | **Put** /api/v0/apiKeys/{apiKeyId}/debugMode | Activate/Deactivate the debug mode on an API key
*ApiKeysApi* | [**UpdateApiKeyUsingPOST1**](docs/ApiKeysApi.md#updateapikeyusingpost1) | **Post** /api/v0/apiKeys/{apiKeyId} | Update an API key
*AuditLogApi* | [**SearchUsingGET**](docs/AuditLogApi.md#searchusingget) | **Get** /api/v0/auditlog/messages | Retrieve messages available in your AuditLog
*BusManagementApi* | [**CreateUsingPOST3**](docs/BusManagementApi.md#createusingpost3) | **Post** /api/v0/topics/fifo | Create a FIFO
*BusManagementApi* | [**DeleteUsingDELETE20**](docs/BusManagementApi.md#deleteusingdelete20) | **Delete** /api/v0/topics/fifo/{fifoName} | Delete a FIFO
*BusManagementApi* | [**FifoPublishUsingPOST**](docs/BusManagementApi.md#fifopublishusingpost) | **Post** /api/v0/topics/fifo/{fifoName} | Publish a message into a FIFO
*BusManagementApi* | [**GetUsingGET19**](docs/BusManagementApi.md#getusingget19) | **Get** /api/v0/topics/fifo/{fifoName} | Get a FIFO
*BusManagementApi* | [**ListFifoTopicsUsingGET**](docs/BusManagementApi.md#listfifotopicsusingget) | **Get** /api/v0/topics/fifo | List all FIFOs
*CACertificatesApi* | [**CreateUsingPOST**](docs/CACertificatesApi.md#createusingpost) | **Post** /api/v0/certificates/ca | Upload CA certificate
*CACertificatesApi* | [**DeleteUsingDELETE9**](docs/CACertificatesApi.md#deleteusingdelete9) | **Delete** /api/v0/certificates/ca/{certificateId} | Delete CA certificate
*CACertificatesApi* | [**ListUsingGET11**](docs/CACertificatesApi.md#listusingget11) | **Get** /api/v0/certificates/ca | List CA certificates
*CACertificatesApi* | [**RetrieveUsingGET**](docs/CACertificatesApi.md#retrieveusingget) | **Get** /api/v0/certificates/ca/{certificateId} | Retrieve CA certificate
*CampaignManagementApi* | [**CancelCampaignUsingPUT**](docs/CampaignManagementApi.md#cancelcampaignusingput) | **Put** /api/v0/deviceMgt/campaigns/{campaignId}/cancel | Cancel a campaign
*CampaignManagementApi* | [**CreateCampaignUsingPOST**](docs/CampaignManagementApi.md#createcampaignusingpost) | **Post** /api/v0/deviceMgt/campaigns | Create a campaign
*CampaignManagementApi* | [**DeleteCampaignUsingDELETE**](docs/CampaignManagementApi.md#deletecampaignusingdelete) | **Delete** /api/v0/deviceMgt/campaigns/{campaignId} | Delete a campaign
*CampaignManagementApi* | [**GetCampaignDetailsUsingGET**](docs/CampaignManagementApi.md#getcampaigndetailsusingget) | **Get** /api/v0/deviceMgt/campaigns/{campaignId}/targets | Get the campaign status per device
*CampaignManagementApi* | [**GetCampaignUsingGET**](docs/CampaignManagementApi.md#getcampaignusingget) | **Get** /api/v0/deviceMgt/campaigns/{campaignId} | Get a campaign
*CampaignManagementApi* | [**ListCampaignsUsingGET**](docs/CampaignManagementApi.md#listcampaignsusingget) | **Get** /api/v0/deviceMgt/campaigns | List all campaigns
*CampaignManagementApi* | [**UpdateCampaignUsingPATCH**](docs/CampaignManagementApi.md#updatecampaignusingpatch) | **Patch** /api/v0/deviceMgt/campaigns/{campaignId} | Update a campaign
*DataBulkInjectionApi* | [**AddDataBulkUsingPOST**](docs/DataBulkInjectionApi.md#adddatabulkusingpost) | **Post** /api/v0/data/bulk | Insert a bulk of new Data
*DataManagementCustomPipelinesApi* | [**DeleteUsingDELETE11**](docs/DataManagementCustomPipelinesApi.md#deleteusingdelete11) | **Delete** /api/v0/pipelines/{pipelineId} | Delete a DataMessage pipeline
*DataManagementCustomPipelinesApi* | [**GetUsingGET12**](docs/DataManagementCustomPipelinesApi.md#getusingget12) | **Get** /api/v0/pipelines/{pipelineId} | Retrieve a DataMessage pipeline
*DataManagementCustomPipelinesApi* | [**ListUsingGET13**](docs/DataManagementCustomPipelinesApi.md#listusingget13) | **Get** /api/v0/pipelines | Retrieve the list of DataMessage pipelines, ordered by priorityLevel
*DataManagementCustomPipelinesApi* | [**PostUsingPOST12**](docs/DataManagementCustomPipelinesApi.md#postusingpost12) | **Post** /api/v0/pipelines | Create a DataMessage pipeline
*DataManagementCustomPipelinesApi* | [**UpdateUsingPUT6**](docs/DataManagementCustomPipelinesApi.md#updateusingput6) | **Put** /api/v0/pipelines/{pipelineId} | Update a DataMessage pipeline
*DataManagementDataSearchApi* | [**DslQueryHitsOnlyUsingPOST**](docs/DataManagementDataSearchApi.md#dslqueryhitsonlyusingpost) | **Post** /api/v0/data/search/hits | Query an Elasticsearch Domain Specific Language request and get only hits result
*DataManagementDataSearchApi* | [**DslQueryUsingPOST**](docs/DataManagementDataSearchApi.md#dslqueryusingpost) | **Post** /api/v0/data/search | Query an Elasticsearch Domain Specific Language request
*DataManagementDataStoreApi* | [**AddDataMessageUsingPOST**](docs/DataManagementDataStoreApi.md#adddatamessageusingpost) | **Post** /api/v0/data/streams/{streamId} | Insert a new Data into the stream
*DataManagementDataStoreApi* | [**RetrieveDataUsingGET**](docs/DataManagementDataStoreApi.md#retrievedatausingget) | **Get** /api/v0/data/streams/{streamId} | Retrieve data from the streamId
*DecodersApi* | [**ListUsingGET22**](docs/DecodersApi.md#listusingget22) | **Get** /api/v0/decoders | Retrieve the list of all decoders (binary, csv, js)
*DecodersBinaryApi* | [**ActivateUsingPUT8**](docs/DecodersBinaryApi.md#activateusingput8) | **Put** /api/v0/decoders/binary/{decoderId}/enabled | Activate or deactivate a decoder
*DecodersBinaryApi* | [**DeleteUsingDELETE8**](docs/DecodersBinaryApi.md#deleteusingdelete8) | **Delete** /api/v0/decoders/binary/{decoderId} | Delete a binary decoder
*DecodersBinaryApi* | [**GetUsingGET10**](docs/DecodersBinaryApi.md#getusingget10) | **Get** /api/v0/decoders/binary/{decoderId} | Retrieve a binary decoder
*DecodersBinaryApi* | [**ListUsingGET10**](docs/DecodersBinaryApi.md#listusingget10) | **Get** /api/v0/decoders/binary | Retrieve the list of binary decoders
*DecodersBinaryApi* | [**PostUsingPOST10**](docs/DecodersBinaryApi.md#postusingpost10) | **Post** /api/v0/decoders/binary | Create a binary decoder
*DecodersBinaryApi* | [**PutUsingPUT2**](docs/DecodersBinaryApi.md#putusingput2) | **Put** /api/v0/decoders/binary/{decoderId} | Update a binary decoder
*DecodersBinaryApi* | [**TestUsingPOST2**](docs/DecodersBinaryApi.md#testusingpost2) | **Post** /api/v0/decoders/binary/test | Test a binary decoder format with an encoded payload
*DecodersCSVApi* | [**ActivateUsingPUT9**](docs/DecodersCSVApi.md#activateusingput9) | **Put** /api/v0/decoders/csv/{decoderId}/enabled | Activate or deactivate a decoder
*DecodersCSVApi* | [**DeleteUsingDELETE10**](docs/DecodersCSVApi.md#deleteusingdelete10) | **Delete** /api/v0/decoders/csv/{decoderId} | Delete a csv decoder
*DecodersCSVApi* | [**GetUsingGET11**](docs/DecodersCSVApi.md#getusingget11) | **Get** /api/v0/decoders/csv/{decoderId} | Retrieve a csv decoder
*DecodersCSVApi* | [**ListUsingGET12**](docs/DecodersCSVApi.md#listusingget12) | **Get** /api/v0/decoders/csv | Retrieve the list of csv decoders filtered by tags
*DecodersCSVApi* | [**PostUsingPOST11**](docs/DecodersCSVApi.md#postusingpost11) | **Post** /api/v0/decoders/csv | Create a csv decoder
*DecodersCSVApi* | [**PutUsingPUT3**](docs/DecodersCSVApi.md#putusingput3) | **Put** /api/v0/decoders/csv/{decoderId} | Update a csv decoder
*DecodersCSVApi* | [**TestUsingPOST3**](docs/DecodersCSVApi.md#testusingpost3) | **Post** /api/v0/decoders/csv/test | Test a csv decoder format with an encoded payload
*DecodersPublicApi* | [**ListUsingGET23**](docs/DecodersPublicApi.md#listusingget23) | **Get** /api/v0/decoders/public | Retrieve the list of all public decoders (binary, csv, js)
*DeprecatedBusManagementRouterApi* | [**AddBindingUsingPOST**](docs/DeprecatedBusManagementRouterApi.md#addbindingusingpost) | **Post** /api/v0/bindings | Add a binding for a FIFO queue
*DeprecatedBusManagementRouterApi* | [**DeleteBindingUsingDELETE**](docs/DeprecatedBusManagementRouterApi.md#deletebindingusingdelete) | **Delete** /api/v0/bindings/{routingKeyFilter}/{fifoName} | Delete a binding
*DeprecatedBusManagementRouterApi* | [**ListBindingsUsingGET**](docs/DeprecatedBusManagementRouterApi.md#listbindingsusingget) | **Get** /api/v0/bindings | List registered bindings
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**AddCommandUsingPOST**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#addcommandusingpost) | **Post** /api/v0/assets/{assetNamespace}/{assetId}/commands | Register a new command
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**DeleteCommandUsingDELETE**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#deletecommandusingdelete) | **Delete** /api/v0/commands/{commandId} | Delete a specific command
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**GetAssetCommandsUsingGET**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#getassetcommandsusingget) | **Get** /api/v0/assets/{assetNamespace}/{assetId}/commands | Get a list of commands targeting a specific asset
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**GetCommandStatusUsingGET**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#getcommandstatususingget) | **Get** /api/v0/commands/{commandId}/status | Get the status of a specific command
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**GetCommandUsingGET**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#getcommandusingget) | **Get** /api/v0/commands/{commandId} | Get a specific command
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**ListCommandsUsingGET**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#listcommandsusingget) | **Get** /api/v0/commands | List registered commands
*DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi* | [**SetCommandStatusUsingPUT**](docs/DeprecatedDeviceManagementCommandV0UseDeviceManagementCommandsV1InsteadApi.md#setcommandstatususingput) | **Put** /api/v0/commands/{commandId}/status | Update the status of specific command 
*DeprecatedDeviceManagementDevicesFirmwaresApi* | [**GetDeviceFirmwareUsingGET**](docs/DeprecatedDeviceManagementDevicesFirmwaresApi.md#getdevicefirmwareusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/firmwares/{firmwareId} | Get a specific device firmware (use /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId} instead)
*DeprecatedDeviceManagementDevicesFirmwaresApi* | [**GetDeviceResourceUpdatesUsingGET**](docs/DeprecatedDeviceManagementDevicesFirmwaresApi.md#getdeviceresourceupdatesusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/firmwareUpdates | Get a list of device firmware updates (use /api/v1/deviceMgt/devices/{deviceId}/resources/updates
*DeprecatedDeviceManagementDevicesFirmwaresApi* | [**GetLastResourceUpdateUsingGET**](docs/DeprecatedDeviceManagementDevicesFirmwaresApi.md#getlastresourceupdateusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/firmwares/{firmwareId}/lastUpdate | Get info about last update of this device firmware (use /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId}/updates/latest instead)
*DeprecatedDeviceManagementDevicesFirmwaresApi* | [**ListDeviceResourcesUsingGET**](docs/DeprecatedDeviceManagementDevicesFirmwaresApi.md#listdeviceresourcesusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/firmwares | Get a map of all device firmwares (use /api/v1/deviceMgt/devices/{deviceId}/resources instead)
*DeprecatedDeviceManagementDevicesFirmwaresApi* | [**SetDeviceResourceVersionUsingPOST**](docs/DeprecatedDeviceManagementDevicesFirmwaresApi.md#setdeviceresourceversionusingpost) | **Post** /api/v1/deviceMgt/devices/{deviceId}/firmwares/{firmwareId} | Set device firmware versions (use /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId} instead)
*DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi* | [**CreateAssetUsingPOST**](docs/DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi.md#createassetusingpost) | **Post** /api/v0/assets | Create a device
*DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi* | [**DeleteDeviceStatusUsingDELETE**](docs/DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi.md#deletedevicestatususingdelete) | **Delete** /api/v0/assets/{assetNamespace}/{assetId} | Delete a device
*DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi* | [**GetAssetStatusUsingGET**](docs/DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi.md#getassetstatususingget) | **Get** /api/v0/assets/{assetNamespace}/{assetId} | Get a device status
*DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi* | [**ListAssetNamespacesUsingGET**](docs/DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi.md#listassetnamespacesusingget) | **Get** /api/v0/inventory/namespaces | Enumerates the used asset namespaces
*DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi* | [**ListAssetsUsingGET**](docs/DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi.md#listassetsusingget) | **Get** /api/v0/assets | List registered assets status
*DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi* | [**UpdateAssetUsingPUT**](docs/DeprecatedDeviceManagementInventoryV0UseDeviceManagementInventoryV1InsteadApi.md#updateassetusingput) | **Put** /api/v0/assets/{assetNamespace}/{assetId} | Update a device
*DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi* | [**GetAssetParamUsingGET**](docs/DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi.md#getassetparamusingget) | **Get** /api/v0/assets/{assetNamespace}/{assetId}/params/{paramKey} | Get a specific asset parameter
*DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi* | [**GetAssetParamsUsingGET**](docs/DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi.md#getassetparamsusingget) | **Get** /api/v0/assets/{assetNamespace}/{assetId}/params | Get a specific asset list of parameters
*DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi* | [**SetAssetParamsUsingPOST**](docs/DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi.md#setassetparamsusingpost) | **Post** /api/v0/assets/{assetNamespace}/{assetId}/params | Update a specific asset list of parameters
*DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi* | [**SetDeviceParamUpdateStatusUsingPUT**](docs/DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi.md#setdeviceparamupdatestatususingput) | **Put** /api/v0/assets/{assetNamespace}/{assetId}/params/{paramKey}/status | Update the status of a specific asset parameter update
*DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi* | [**SetDeviceParamsUpdateStatusUsingPUT**](docs/DeprecatedDeviceManagementParameterV0UseDeviceManagementConfigurationV1InsteadApi.md#setdeviceparamsupdatestatususingput) | **Put** /api/v0/assets/{assetNamespace}/{assetId}/params/status | Update the status of a specific asset parameters update
*DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi* | [**CancelResourceUpdateUsingPOST**](docs/DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi.md#cancelresourceupdateusingpost) | **Post** /api/v0/rm/asset/{assetIdNamespace}/{assetId}/cancelUpdate | Cancel asset resource update
*DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi* | [**GetAllAssetResourcesUsingGET**](docs/DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi.md#getallassetresourcesusingget) | **Get** /api/v0/rm/asset/{assetIdNamespace}/{assetId} | List the asset&#39;s resources (use /api/v1/deviceMgt/devices/{deviceId}/resources instead)
*DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi* | [**GetLastUpdateUsingGET**](docs/DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi.md#getlastupdateusingget) | **Get** /api/v0/rm/asset/{assetIdNamespace}/{assetId}/update | Get the asset&#39;s resources update status (use /api/v1/deviceMgt/devices/{deviceId}/firmwares/{firmwareId}/lastUpdate instead)
*DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi* | [**GetUpdateHistoryUsingGET**](docs/DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi.md#getupdatehistoryusingget) | **Get** /api/v0/rm/asset/{assetIdNamespace}/{assetId}/update/history | Get the asset&#39;s resources update history (use /api/v1/deviceMgt/devices/{deviceId}/firmwareUpdates instead)
*DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi* | [**SetAssetTargetResourceVersionUsingPUT**](docs/DeprecatedDeviceManagementResourceV0UseDeviceManagementResourceV1InsteadApi.md#setassettargetresourceversionusingput) | **Put** /api/v0/rm/asset/{assetIdNamespace}/{assetId}/resource/{resourceId}/targetversion | Set asset&#39;s target resource version (use /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId} instead)
*DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi* | [**CreateGroupUsingPOST**](docs/DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi.md#creategroupusingpost) | **Post** /api/v0/groups | Create a group
*DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi* | [**DeleteGroupUsingDELETE**](docs/DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi.md#deletegroupusingdelete) | **Delete** /api/v0/groups/{groupId} | Delete a group
*DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi* | [**GetGroupUsingGET**](docs/DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi.md#getgroupusingget) | **Get** /api/v0/groups/{groupId} | Get a group
*DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi* | [**ListGroupsUsingGET**](docs/DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi.md#listgroupsusingget) | **Get** /api/v0/groups | List registered groups
*DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi* | [**UpdateGroupUsingPUT**](docs/DeprecatedGroupManagementV0UseDeviceManagementGroupsV1InsteadApi.md#updategroupusingput) | **Put** /api/v0/groups/{groupId} | Update a group
*DeprecatedRouterTopicManagementApi* | [**PublishUsingPOST**](docs/DeprecatedRouterTopicManagementApi.md#publishusingpost) | **Post** /api/v0/topics/router/{routingKey} | Publishing a message into a ROUTER topic
*DeprecatedSMSConnectorApi* | [**SendSMSUsingPOST**](docs/DeprecatedSMSConnectorApi.md#sendsmsusingpost) | **Post** /api/v0/sms-connector/sms | send SMS by SMS Connector for a list of MSISDN
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**CreateBusinessSettingsUsingPOST**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#createbusinesssettingsusingpost) | **Post** /api/v0/sms-connector/settings/business | Create a new business settings of SMS Connector (use Device management - Interfaces - V1 instead)
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**DeleteBusinessSettingsByMsiSDNUsingDELETE**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#deletebusinesssettingsbymsisdnusingdelete) | **Delete** /api/v0/sms-connector/settings/business/msisdn | Delete msisdn in business settings of SMS Connector
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**DeleteBusinessSettingsForOneMsiSDNUsingDELETE**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#deletebusinesssettingsforonemsisdnusingdelete) | **Delete** /api/v0/sms-connector/settings/business/{serverPhoneNumber}/msisdn/{msisdnNumber} | Delete one msisdn in business settings of SMS Connector
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**GetBusinessSettingsUsingGET**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#getbusinesssettingsusingget) | **Get** /api/v0/sms-connector/settings/business | Get a business settings of SMS Connector
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**ListMsisdnUsingGET**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#listmsisdnusingget) | **Get** /api/v0/sms-connector/settings/business/msisdn | List msisdn of business settings of SMS Connector
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**ListSettingsUsingGET**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#listsettingsusingget) | **Get** /api/v0/sms-connector/settings | List all the business settings of the SMSConnector for a tenant
*DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi* | [**UpdateBusinessSettingsUsingPOST**](docs/DeprecatedSMSConnectorBusinessSettingsUseDeviceManagementConnectorNodesV1InsteadApi.md#updatebusinesssettingsusingpost) | **Post** /api/v0/sms-connector/settings/business/{serverPhoneNumber} | Update business settings of the SMSConnector
*DeprecatedStatisticsUseAccountingV1AccountingInsteadApi* | [**GetTenantStatisticsUsingGET3**](docs/DeprecatedStatisticsUseAccountingV1AccountingInsteadApi.md#gettenantstatisticsusingget3) | **Get** /api/v0/statistics/tenant/{tenantId} | Get tenant statistics for a specific tenant and a range of dates
*DeprecatedTriggersAndActionsApi* | [**CreateUsingPOST1**](docs/DeprecatedTriggersAndActionsApi.md#createusingpost1) | **Post** /api/v0/event2action/actionPolicies | Create an ActionPolicy
*DeprecatedTriggersAndActionsApi* | [**DeleteUsingDELETE12**](docs/DeprecatedTriggersAndActionsApi.md#deleteusingdelete12) | **Delete** /api/v0/event2action/actionPolicies/{policyId} | Delete an ActionPolicy
*DeprecatedTriggersAndActionsApi* | [**ListUsingGET14**](docs/DeprecatedTriggersAndActionsApi.md#listusingget14) | **Get** /api/v0/event2action/actionPolicies | List ActionPolicies
*DeprecatedTriggersAndActionsApi* | [**RetrieveUsingGET1**](docs/DeprecatedTriggersAndActionsApi.md#retrieveusingget1) | **Get** /api/v0/event2action/actionPolicies/{policyId} | Retrieve an ActionPolicy
*DeprecatedTriggersAndActionsApi* | [**UpsertUsingPUT**](docs/DeprecatedTriggersAndActionsApi.md#upsertusingput) | **Put** /api/v0/event2action/actionPolicies/{policyId} | Create or update an ActionPolicy
*DeprecatedTriggersAndActionsTestApi* | [**TestHttpPushUsingPOST**](docs/DeprecatedTriggersAndActionsTestApi.md#testhttppushusingpost) | **Post** /api/v0/event2action/test/http-push | Post an http request for testing a webhook
*DeviceManagementCommandsV1Api* | [**AddCommandUsingPOST1**](docs/DeviceManagementCommandsV1Api.md#addcommandusingpost1) | **Post** /api/v1/deviceMgt/devices/{deviceId}/commands | Register a new command
*DeviceManagementCommandsV1Api* | [**DeleteCommandUsingDELETE1**](docs/DeviceManagementCommandsV1Api.md#deletecommandusingdelete1) | **Delete** /api/v1/deviceMgt/commands/{commandId} | Delete a specific command
*DeviceManagementCommandsV1Api* | [**GetAssetCommandsUsingGET1**](docs/DeviceManagementCommandsV1Api.md#getassetcommandsusingget1) | **Get** /api/v1/deviceMgt/devices/{deviceId}/commands | List commands targeting a specific device
*DeviceManagementCommandsV1Api* | [**GetCommandStatusUsingGET1**](docs/DeviceManagementCommandsV1Api.md#getcommandstatususingget1) | **Get** /api/v1/deviceMgt/commands/{commandId}/status | Get the status of a specific command
*DeviceManagementCommandsV1Api* | [**GetCommandUsingGET1**](docs/DeviceManagementCommandsV1Api.md#getcommandusingget1) | **Get** /api/v1/deviceMgt/commands/{commandId} | Get a specific command
*DeviceManagementCommandsV1Api* | [**SetCommandStatusUsingPUT1**](docs/DeviceManagementCommandsV1Api.md#setcommandstatususingput1) | **Put** /api/v1/deviceMgt/commands/{commandId}/status | Update the status of specific command 
*DeviceManagementConfigurationV1Api* | [**GetDeviceConfigParameterUsingGET**](docs/DeviceManagementConfigurationV1Api.md#getdeviceconfigparameterusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/config/parameters/{paramKey} | Get state of a specific device configuration parameter
*DeviceManagementConfigurationV1Api* | [**GetDeviceConfigParametersUsingGET**](docs/DeviceManagementConfigurationV1Api.md#getdeviceconfigparametersusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/config/parameters | Get a description of the device configuration parameters
*DeviceManagementConfigurationV1Api* | [**GetDeviceConfigurationUsingGET**](docs/DeviceManagementConfigurationV1Api.md#getdeviceconfigurationusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/config | Get a description of the device configuration
*DeviceManagementConfigurationV1Api* | [**SetDeviceParamUpdateStatusUsingPUT1**](docs/DeviceManagementConfigurationV1Api.md#setdeviceparamupdatestatususingput1) | **Put** /api/v1/deviceMgt/devices/{deviceId}/config/parameters/{paramKey}/status | Update the status of a specific device parameter
*DeviceManagementConfigurationV1Api* | [**SetMultipleDeviceConfigParamsUsingPOST**](docs/DeviceManagementConfigurationV1Api.md#setmultipledeviceconfigparamsusingpost) | **Post** /api/v1/deviceMgt/devices/{deviceId}/config | Set requested values for a set of device configuration parameters
*DeviceManagementConnectorNodesV1Api* | [**DeleteNodeUsingDELETE**](docs/DeviceManagementConnectorNodesV1Api.md#deletenodeusingdelete) | **Delete** /api/v1/deviceMgt/connectors/{connector}/nodes/{nodeId} | Delete a connector node
*DeviceManagementConnectorNodesV1Api* | [**GetNodeUsingGET**](docs/DeviceManagementConnectorNodesV1Api.md#getnodeusingget) | **Get** /api/v1/deviceMgt/connectors/{connector}/nodes/{nodeId} | Get details of connector node
*DeviceManagementConnectorNodesV1Api* | [**ListNodesUsingGET**](docs/DeviceManagementConnectorNodesV1Api.md#listnodesusingget) | **Get** /api/v1/deviceMgt/connectors/{connector}/nodes | List all connector nodes
*DeviceManagementConnectorNodesV1Api* | [**UpdateNodeUsingPATCH**](docs/DeviceManagementConnectorNodesV1Api.md#updatenodeusingpatch) | **Patch** /api/v1/deviceMgt/connectors/{connector}/nodes/{nodeId} | Update a connector node
*DeviceManagementForCOAPLWM2MBetaApi* | [**GetDeviceUsingGET4**](docs/DeviceManagementForCOAPLWM2MBetaApi.md#getdeviceusingget4) | **Get** /api/v0/vendors/lwm2m/identity/{ep} | Find a device
*DeviceManagementForCOAPLWM2MBetaApi* | [**ListDevicesUsingGET4**](docs/DeviceManagementForCOAPLWM2MBetaApi.md#listdevicesusingget4) | **Get** /api/v0/vendors/lwm2m/identities | List lwm2m devices
*DeviceManagementForCOAPLWM2MBetaApi* | [**RegisterDeviceUsingPOST1**](docs/DeviceManagementForCOAPLWM2MBetaApi.md#registerdeviceusingpost1) | **Post** /api/v0/vendors/lwm2m/identities | Register LWM2M/DTLS identity
*DeviceManagementForCOAPLWM2MBetaApi* | [**UnregisterDeviceUsingDELETE1**](docs/DeviceManagementForCOAPLWM2MBetaApi.md#unregisterdeviceusingdelete1) | **Delete** /api/v0/vendors/lwm2m/identity/{ep} | Unregister a device
*DeviceManagementForCOAPLWM2MBetaApi* | [**UpdateDeviceUsingPUT**](docs/DeviceManagementForCOAPLWM2MBetaApi.md#updatedeviceusingput) | **Put** /api/v0/vendors/lwm2m/identity/{ep} | Update a device
*DeviceManagementForLoRaApi* | [**CountDevicesUsingGET2**](docs/DeviceManagementForLoRaApi.md#countdevicesusingget2) | **Get** /api/v0/vendors/lora/devices/count | Device counts
*DeviceManagementForLoRaApi* | [**GetDeviceProfilesUsingGET**](docs/DeviceManagementForLoRaApi.md#getdeviceprofilesusingget) | **Get** /api/v1/deviceMgt/connectors/lora/profiles | Get LoRa device profiles
*DeviceManagementForLoRaApi* | [**GetDeviceProfilesUsingGET1**](docs/DeviceManagementForLoRaApi.md#getdeviceprofilesusingget1) | **Get** /api/v0/vendors/lora/profiles | Get LoRa device profiles (use /api/v1/deviceMgt/connectors/lora/profiles instead)
*DeviceManagementForLoRaApi* | [**GetDeviceUsingGET3**](docs/DeviceManagementForLoRaApi.md#getdeviceusingget3) | **Get** /api/v0/vendors/lora/devices/{devEUI} | Get a device (use Device management - Connector nodes - V1 instead)
*DeviceManagementForLoRaApi* | [**GetGatewayUsingGET**](docs/DeviceManagementForLoRaApi.md#getgatewayusingget) | **Get** /api/v1/deviceMgt/connectors/lora/gateways/{id} | Get a gateway
*DeviceManagementForLoRaApi* | [**GetMessageCountUsingGET**](docs/DeviceManagementForLoRaApi.md#getmessagecountusingget) | **Get** /api/v0/vendors/lora/data/count | List of message counts group by period
*DeviceManagementForLoRaApi* | [**ListCommandUsingGET**](docs/DeviceManagementForLoRaApi.md#listcommandusingget) | **Get** /api/v1/deviceMgt/connectors/lora/nodes/{devEUI}/downlinks | List commands
*DeviceManagementForLoRaApi* | [**ListCommandUsingGET1**](docs/DeviceManagementForLoRaApi.md#listcommandusingget1) | **Get** /api/v0/vendors/lora/devices/{devEUI}/commands | List commands (use /api/v1/deviceMgt/connectors/lora/nodes/{devEUI}/downlinks instead)
*DeviceManagementForLoRaApi* | [**ListConnectivityPlanUsingGET**](docs/DeviceManagementForLoRaApi.md#listconnectivityplanusingget) | **Get** /api/v1/deviceMgt/connectors/lora/connectivity | List connectivity plan
*DeviceManagementForLoRaApi* | [**ListConnectivityPlansUsingGET**](docs/DeviceManagementForLoRaApi.md#listconnectivityplansusingget) | **Get** /api/v1/deviceMgt/connectors/lora/connectivities | List connectivity plans
*DeviceManagementForLoRaApi* | [**ListDevicesUsingGET3**](docs/DeviceManagementForLoRaApi.md#listdevicesusingget3) | **Get** /api/v0/vendors/lora/devices | List LoRa devices (use Device management - Connector nodes - V1 instead)
*DeviceManagementForLoRaApi* | [**ListGatewaysUsingGET**](docs/DeviceManagementForLoRaApi.md#listgatewaysusingget) | **Get** /api/v1/deviceMgt/connectors/lora/gateways | List gateways
*DeviceManagementForLoRaApi* | [**RegisterCommandUsingPOST**](docs/DeviceManagementForLoRaApi.md#registercommandusingpost) | **Post** /api/v1/deviceMgt/connectors/lora/nodes/{devEUI}/downlinks | Register a command
*DeviceManagementForLoRaApi* | [**RegisterCommandUsingPOST1**](docs/DeviceManagementForLoRaApi.md#registercommandusingpost1) | **Post** /api/v0/vendors/lora/devices/{devEUI}/commands | Register a command (use /api/v1/deviceMgt/connectors/lora/nodes/{devEUI}/downlinks instead)
*DeviceManagementForLoRaApi* | [**RegisterDeviceUsingPOST**](docs/DeviceManagementForLoRaApi.md#registerdeviceusingpost) | **Post** /api/v0/vendors/lora/devices | Register a LoRa device (use Device management - Interfaces - V1 instead)
*DeviceManagementForLoRaApi* | [**UnregisterDeviceUsingDELETE**](docs/DeviceManagementForLoRaApi.md#unregisterdeviceusingdelete) | **Delete** /api/v0/vendors/lora/devices/{devEUI} | Unregister a device (use Device management - Connector nodes - V1 instead)
*DeviceManagementForLoRaApi* | [**UpdateDeviceUsingPATCH1**](docs/DeviceManagementForLoRaApi.md#updatedeviceusingpatch1) | **Patch** /api/v0/vendors/lora/devices/{devEUI} | Update a device (use Device management - Connector nodes - V1 instead)
*DeviceManagementForLoRaApi* | [**UpdateGatewayUsingPATCH**](docs/DeviceManagementForLoRaApi.md#updategatewayusingpatch) | **Patch** /api/v1/deviceMgt/connectors/lora/gateways/{id} | Update a gateway.
*DeviceManagementGroupsV1Api* | [**CreateGroupUsingPOST1**](docs/DeviceManagementGroupsV1Api.md#creategroupusingpost1) | **Post** /api/v1/deviceMgt/groups | Create a group
*DeviceManagementGroupsV1Api* | [**DeleteGroupUsingDELETE1**](docs/DeviceManagementGroupsV1Api.md#deletegroupusingdelete1) | **Delete** /api/v1/deviceMgt/groups/{id} | Delete a group
*DeviceManagementGroupsV1Api* | [**GetGroupUsingGET1**](docs/DeviceManagementGroupsV1Api.md#getgroupusingget1) | **Get** /api/v1/deviceMgt/groups/{id} | Get a group
*DeviceManagementGroupsV1Api* | [**ListGroupsUsingGET1**](docs/DeviceManagementGroupsV1Api.md#listgroupsusingget1) | **Get** /api/v1/deviceMgt/groups | List registered groups
*DeviceManagementGroupsV1Api* | [**UpdateGroupUsingPUT1**](docs/DeviceManagementGroupsV1Api.md#updategroupusingput1) | **Put** /api/v1/deviceMgt/groups/{id} | Update a group
*DeviceManagementInterfacesV1Api* | [**AddInterfaceToDeviceUsingPOST**](docs/DeviceManagementInterfacesV1Api.md#addinterfacetodeviceusingpost) | **Post** /api/v1/deviceMgt/devices/{deviceId}/interfaces | Add an interface to a registered device
*DeviceManagementInterfacesV1Api* | [**DeleteInterfaceUsingDELETE**](docs/DeviceManagementInterfacesV1Api.md#deleteinterfaceusingdelete) | **Delete** /api/v1/deviceMgt/devices/{deviceId}/interfaces/{interfaceId} | Delete an interface
*DeviceManagementInterfacesV1Api* | [**GetInterfaceForADeviceUsingGET**](docs/DeviceManagementInterfacesV1Api.md#getinterfaceforadeviceusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/interfaces/{interfaceId} | Get a specific interface for a registered device
*DeviceManagementInterfacesV1Api* | [**ListInterfacesForADeviceUsingGET**](docs/DeviceManagementInterfacesV1Api.md#listinterfacesforadeviceusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/interfaces | Get the interface list for a registered device
*DeviceManagementInterfacesV1Api* | [**UpdateInterfaceUsingPATCH**](docs/DeviceManagementInterfacesV1Api.md#updateinterfaceusingpatch) | **Patch** /api/v1/deviceMgt/devices/{deviceId}/interfaces/{interfaceId} | Update an interface
*DeviceManagementInventoryV1Api* | [**CreateDeviceUsingPOST**](docs/DeviceManagementInventoryV1Api.md#createdeviceusingpost) | **Post** /api/v1/deviceMgt/devices | Create a device
*DeviceManagementInventoryV1Api* | [**DeleteDeviceUsingDELETE**](docs/DeviceManagementInventoryV1Api.md#deletedeviceusingdelete) | **Delete** /api/v1/deviceMgt/devices/{deviceId} | Delete a device
*DeviceManagementInventoryV1Api* | [**GetDeviceStreamsUsingGET**](docs/DeviceManagementInventoryV1Api.md#getdevicestreamsusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/data/streams | Get a device&#39;s streamIds
*DeviceManagementInventoryV1Api* | [**GetDeviceUsingGET2**](docs/DeviceManagementInventoryV1Api.md#getdeviceusingget2) | **Get** /api/v1/deviceMgt/devices/{deviceId} | Get a device
*DeviceManagementInventoryV1Api* | [**ListDevicesUsingGET2**](docs/DeviceManagementInventoryV1Api.md#listdevicesusingget2) | **Get** /api/v1/deviceMgt/devices | List registered devices
*DeviceManagementInventoryV1Api* | [**UpdateDeviceUsingPATCH**](docs/DeviceManagementInventoryV1Api.md#updatedeviceusingpatch) | **Patch** /api/v1/deviceMgt/devices/{deviceId} | Update a device
*DeviceManagementResourcesManagementApi* | [**AddResourceUsingPOST**](docs/DeviceManagementResourcesManagementApi.md#addresourceusingpost) | **Post** /api/v0/rm | Add a resource in Resource Manager
*DeviceManagementResourcesManagementApi* | [**AddResourceVersionUsingPOST**](docs/DeviceManagementResourcesManagementApi.md#addresourceversionusingpost) | **Post** /api/v0/rm/{resourceId}/version | Add a new version to a resource
*DeviceManagementResourcesManagementApi* | [**DeleteResourceUsingDELETE**](docs/DeviceManagementResourcesManagementApi.md#deleteresourceusingdelete) | **Delete** /api/v0/rm/{resourceId} | Delete a resource
*DeviceManagementResourcesManagementApi* | [**DeleteResourceVersionUsingDELETE**](docs/DeviceManagementResourcesManagementApi.md#deleteresourceversionusingdelete) | **Delete** /api/v0/rm/{resourceId}/version/{resourceVersionId} | Delete a resource&#39;s version
*DeviceManagementResourcesManagementApi* | [**GetAllConnectorsUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getallconnectorsusingget) | **Get** /api/v0/rm/connectors | Get all available connectors
*DeviceManagementResourcesManagementApi* | [**GetConnectorMandatoryAndOptionalMetadataUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getconnectormandatoryandoptionalmetadatausingget) | **Get** /api/v0/rm/connectors/{connector}/metadata | Get mandatory and optional metadata of a connector
*DeviceManagementResourcesManagementApi* | [**GetResourceCompatibleVersionsUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getresourcecompatibleversionsusingget) | **Get** /api/v0/rm/{resourceId}/compatibleVersion/{currentVersion} | List the versions from which a resource update to the version can be done
*DeviceManagementResourcesManagementApi* | [**GetResourceUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getresourceusingget) | **Get** /api/v0/rm/{resourceId} | Get a resource
*DeviceManagementResourcesManagementApi* | [**GetResourceVersionUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getresourceversionusingget) | **Get** /api/v0/rm/{resourceId}/version/{resourceVersionId} | Get a resource&#39;s version
*DeviceManagementResourcesManagementApi* | [**GetResourceVersionsUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getresourceversionsusingget) | **Get** /api/v0/rm/{resourceId}/version | List resource&#39;s versions
*DeviceManagementResourcesManagementApi* | [**GetResourcesUsingGET**](docs/DeviceManagementResourcesManagementApi.md#getresourcesusingget) | **Get** /api/v0/rm | List all resources
*DeviceManagementResourcesManagementApi* | [**UpdateResourceUsingPUT**](docs/DeviceManagementResourcesManagementApi.md#updateresourceusingput) | **Put** /api/v0/rm/{resourceId} | Update a resource
*DeviceManagementResourcesManagementApi* | [**UpdateResourceVersionUsingPUT**](docs/DeviceManagementResourcesManagementApi.md#updateresourceversionusingput) | **Put** /api/v0/rm/{resourceId}/version/{resourceVersionId} | Update a resource&#39;s version
*DeviceManagementResourcesV1Api* | [**GetDeviceResourceUpdatesUsingGET1**](docs/DeviceManagementResourcesV1Api.md#getdeviceresourceupdatesusingget1) | **Get** /api/v1/deviceMgt/devices/{deviceId}/resources/updates | Get a list of device resource updates
*DeviceManagementResourcesV1Api* | [**GetDeviceResourceUsingGET**](docs/DeviceManagementResourcesV1Api.md#getdeviceresourceusingget) | **Get** /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId} | Get a specific device resource
*DeviceManagementResourcesV1Api* | [**GetLastResourceUpdateUsingGET1**](docs/DeviceManagementResourcesV1Api.md#getlastresourceupdateusingget1) | **Get** /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId}/updates/latest | Get info about last update of this device resource
*DeviceManagementResourcesV1Api* | [**ListDeviceResourcesUsingGET1**](docs/DeviceManagementResourcesV1Api.md#listdeviceresourcesusingget1) | **Get** /api/v1/deviceMgt/devices/{deviceId}/resources | Get a map of all device resources
*DeviceManagementResourcesV1Api* | [**SetDeviceResourceVersionUsingPOST1**](docs/DeviceManagementResourcesV1Api.md#setdeviceresourceversionusingpost1) | **Post** /api/v1/deviceMgt/devices/{deviceId}/resources/{resourceId} | Set device resource versions
*EventProcessingActivityApi* | [**DeleteUsingDELETE14**](docs/EventProcessingActivityApi.md#deleteusingdelete14) | **Delete** /api/v0/eventprocessing/activity/rules/{activityRuleId} | Delete an ActivityRule
*EventProcessingActivityApi* | [**GetStatesUsingGET**](docs/EventProcessingActivityApi.md#getstatesusingget) | **Get** /api/v0/eventprocessing/activity/states | Retrieve the list of all the ActivityStates linked to a specific device and/or rule
*EventProcessingActivityApi* | [**GetUsingGET13**](docs/EventProcessingActivityApi.md#getusingget13) | **Get** /api/v0/eventprocessing/activity/rules/{activityRuleId} | Retrieve an ActivityRule
*EventProcessingActivityApi* | [**ListUsingGET16**](docs/EventProcessingActivityApi.md#listusingget16) | **Get** /api/v0/eventprocessing/activity/rules | Retrieve the list of all the ActivityRules or get an ActivityRule by its name
*EventProcessingActivityApi* | [**MuteUsingPUT**](docs/EventProcessingActivityApi.md#muteusingput) | **Put** /api/v0/eventprocessing/activity/states/mute | Mute or reset nextAlarm of ActivityStates targeted by a specific deviceId/activityRuleId
*EventProcessingActivityApi* | [**PostUsingPOST13**](docs/EventProcessingActivityApi.md#postusingpost13) | **Post** /api/v0/eventprocessing/activity/rules | Create an ActivityRule
*EventProcessingActivityApi* | [**UpdateUsingPUT7**](docs/EventProcessingActivityApi.md#updateusingput7) | **Put** /api/v0/eventprocessing/activity/rules/{activityRuleId} | Update an ActivityRule
*EventProcessingContextApi* | [**ClearUsingDELETE**](docs/EventProcessingContextApi.md#clearusingdelete) | **Delete** /api/v0/eventprocessing/context | Delete all context entries
*EventProcessingContextApi* | [**DeleteUsingDELETE15**](docs/EventProcessingContextApi.md#deleteusingdelete15) | **Delete** /api/v0/eventprocessing/context/{contextKey} | Delete a context key
*EventProcessingContextApi* | [**GetUsingGET14**](docs/EventProcessingContextApi.md#getusingget14) | **Get** /api/v0/eventprocessing/context/{contextKey} | Retrieve a context
*EventProcessingContextApi* | [**ListUsingGET17**](docs/EventProcessingContextApi.md#listusingget17) | **Get** /api/v0/eventprocessing/context | Retrieve the list of contexts with optional tag filtering
*EventProcessingContextApi* | [**SaveUsingPUT**](docs/EventProcessingContextApi.md#saveusingput) | **Put** /api/v0/eventprocessing/context/{contextKey} | Save a context
*EventProcessingFiringApi* | [**DeleteUsingDELETE16**](docs/EventProcessingFiringApi.md#deleteusingdelete16) | **Delete** /api/v0/eventprocessing/firing-rule/{firingRuleId} | Delete a FiringRule
*EventProcessingFiringApi* | [**GetFiringGuardUsingGET**](docs/EventProcessingFiringApi.md#getfiringguardusingget) | **Get** /api/v0/eventprocessing/firing-guard/{firingGuardId} | Get a FiringGuard
*EventProcessingFiringApi* | [**GetFiringGuardsUsingPOST**](docs/EventProcessingFiringApi.md#getfiringguardsusingpost) | **Post** /api/v0/eventprocessing/firing-guard/search | Get FiringGuards linked to a FiringRule, and where FiringGuards selection criteria match.
*EventProcessingFiringApi* | [**GetUsingGET15**](docs/EventProcessingFiringApi.md#getusingget15) | **Get** /api/v0/eventprocessing/firing-rule/{firingRuleId} | Retrieve a FiringRule
*EventProcessingFiringApi* | [**ListUsingGET18**](docs/EventProcessingFiringApi.md#listusingget18) | **Get** /api/v0/eventprocessing/firing-rule | Retrieve the list of all the FiringRules or get a FiringRule by its name
*EventProcessingFiringApi* | [**PostUsingPOST14**](docs/EventProcessingFiringApi.md#postusingpost14) | **Post** /api/v0/eventprocessing/firing-rule | Create a FiringRule
*EventProcessingFiringApi* | [**RemoveFiringGuardUsingDELETE**](docs/EventProcessingFiringApi.md#removefiringguardusingdelete) | **Delete** /api/v0/eventprocessing/firing-guard/{firingGuardId} | Remove a FiringGuard
*EventProcessingFiringApi* | [**RemoveFiringGuardsUsingDELETE**](docs/EventProcessingFiringApi.md#removefiringguardsusingdelete) | **Delete** /api/v0/eventprocessing/firing-guard | Remove the FiringGuards linked to FiringRule, and where FiringGuards selection criteria match.
*EventProcessingFiringApi* | [**UpdateUsingPUT8**](docs/EventProcessingFiringApi.md#updateusingput8) | **Put** /api/v0/eventprocessing/firing-rule/{firingRuleId} | Update a FiringRule
*EventProcessingGeozoneApi* | [**DeleteUsingDELETE17**](docs/EventProcessingGeozoneApi.md#deleteusingdelete17) | **Delete** /api/v0/eventprocessing/geozones/{zoneId} | delete a geographic zone
*EventProcessingGeozoneApi* | [**GetUsingGET16**](docs/EventProcessingGeozoneApi.md#getusingget16) | **Get** /api/v0/eventprocessing/geozones/{zoneId} | retrieve a geozone from repository
*EventProcessingGeozoneApi* | [**ListUsingGET19**](docs/EventProcessingGeozoneApi.md#listusingget19) | **Get** /api/v0/eventprocessing/geozones | retrieve paginated list of geozones
*EventProcessingGeozoneApi* | [**SaveUsingPUT1**](docs/EventProcessingGeozoneApi.md#saveusingput1) | **Put** /api/v0/eventprocessing/geozones/{zoneId} | Save a geographic zone
*EventProcessingMatchingApi* | [**DeleteUsingDELETE18**](docs/EventProcessingMatchingApi.md#deleteusingdelete18) | **Delete** /api/v0/eventprocessing/matching-rule/{matchingRuleId} | Delete a MatchingRule
*EventProcessingMatchingApi* | [**GetUsingGET17**](docs/EventProcessingMatchingApi.md#getusingget17) | **Get** /api/v0/eventprocessing/matching-rule/{matchingRuleId} | Retrieve a MatchingRule
*EventProcessingMatchingApi* | [**ListUsingGET20**](docs/EventProcessingMatchingApi.md#listusingget20) | **Get** /api/v0/eventprocessing/matching-rule | Retrieve the list of all the MatchingRules or get a MatchingRule by its name
*EventProcessingMatchingApi* | [**PostUsingPOST15**](docs/EventProcessingMatchingApi.md#postusingpost15) | **Post** /api/v0/eventprocessing/matching-rule | Create a MatchingRule
*EventProcessingMatchingApi* | [**TestUsingPOST4**](docs/EventProcessingMatchingApi.md#testusingpost4) | **Post** /api/v0/eventprocessing/matching-rule/test | Test a JsonLogic predicate with some data sample
*EventProcessingMatchingApi* | [**UpdateUsingPUT9**](docs/EventProcessingMatchingApi.md#updateusingput9) | **Put** /api/v0/eventprocessing/matching-rule/{matchingRuleId} | Update a MatchingRule
*EventProcessingStateProcessingApi* | [**DeleteUsingDELETE19**](docs/EventProcessingStateProcessingApi.md#deleteusingdelete19) | **Delete** /api/v0/eventprocessing/stateprocessing-rule/{stateProcessingRuleId} | Delete a StateProcessingRule
*EventProcessingStateProcessingApi* | [**GetUsingGET18**](docs/EventProcessingStateProcessingApi.md#getusingget18) | **Get** /api/v0/eventprocessing/stateprocessing-rule/{stateProcessingRuleId} | Retrieve a StateProcessingRule
*EventProcessingStateProcessingApi* | [**ListUsingGET21**](docs/EventProcessingStateProcessingApi.md#listusingget21) | **Get** /api/v0/eventprocessing/stateprocessing-rule | Retrieve the list of all the StateProcessingRules or get a StateProcessingRule by its name
*EventProcessingStateProcessingApi* | [**PostUsingPOST16**](docs/EventProcessingStateProcessingApi.md#postusingpost16) | **Post** /api/v0/eventprocessing/stateprocessing-rule | Create a StateProcessingRule
*EventProcessingStateProcessingApi* | [**TestUsingPOST5**](docs/EventProcessingStateProcessingApi.md#testusingpost5) | **Post** /api/v0/eventprocessing/stateprocessing-rule/test | test a  State Processing function
*EventProcessingStateProcessingApi* | [**UpdateUsingPUT10**](docs/EventProcessingStateProcessingApi.md#updateusingput10) | **Put** /api/v0/eventprocessing/stateprocessing-rule/{stateProcessingRuleId} | Update a StateProcessingRule
*NotificationApi* | [**SendMessageUsingPOST**](docs/NotificationApi.md#sendmessageusingpost) | **Post** /api/v0/contact | API to send message
*PartnerDataPushApi* | [**DataPushUsingPOST**](docs/PartnerDataPushApi.md#datapushusingpost) | **Post** /api/v0/partners/data/streams | Push data
*PartnersManagementApi* | [**AddOptionToTenantUsingPOST**](docs/PartnersManagementApi.md#addoptiontotenantusingpost) | **Post** /api/v0/partners/tenants/{tenantId}/options | Subscribe an option for a partner&#39;s tenant
*PartnersManagementApi* | [**CreateTenantByPartnerUsingPOST**](docs/PartnersManagementApi.md#createtenantbypartnerusingpost) | **Post** /api/v0/partners/tenants | Create a partner&#39;s tenant
*PartnersTokensManagementApi* | [**CreatePartnerTokenUsingPOST**](docs/PartnersTokensManagementApi.md#createpartnertokenusingpost) | **Post** /api/v0/oauth2/token | Request a new Token
*TenantAccountsApi* | [**GetMyTenantUsingGET**](docs/TenantAccountsApi.md#getmytenantusingget) | **Get** /api/v0/tenants/me | Get details of your account
*TriggersAndActionsApi* | [**CreateUsingPOST2**](docs/TriggersAndActionsApi.md#createusingpost2) | **Post** /api/v1/event2action/actionPolicies | Create an ActionPolicy
*TriggersAndActionsApi* | [**DeleteUsingDELETE13**](docs/TriggersAndActionsApi.md#deleteusingdelete13) | **Delete** /api/v1/event2action/actionPolicies/{policyId} | Delete an ActionPolicy
*TriggersAndActionsApi* | [**ListUsingGET15**](docs/TriggersAndActionsApi.md#listusingget15) | **Get** /api/v1/event2action/actionPolicies | List ActionPolicies
*TriggersAndActionsApi* | [**RetrieveUsingGET2**](docs/TriggersAndActionsApi.md#retrieveusingget2) | **Get** /api/v1/event2action/actionPolicies/{policyId} | Retrieve an ActionPolicy
*TriggersAndActionsApi* | [**UpsertUsingPUT1**](docs/TriggersAndActionsApi.md#upsertusingput1) | **Put** /api/v1/event2action/actionPolicies/{policyId} | Create or update an ActionPolicy
*TriggersAndActionsTestApi* | [**TestHttpPushUsingPOST1**](docs/TriggersAndActionsTestApi.md#testhttppushusingpost1) | **Post** /api/v1/event2action/test/http-push | Post an http request for testing a webhook
*UserAuthenticationApi* | [**AuthenticateUserUsingPOST**](docs/UserAuthenticationApi.md#authenticateuserusingpost) | **Post** /api/v0/auth | Authenticate a user
*UserAuthenticationApi* | [**CookiesDeleteUsingDELETE**](docs/UserAuthenticationApi.md#cookiesdeleteusingdelete) | **Delete** /api/v0/cookies | cookiesDelete
*UserAuthenticationApi* | [**GetTenantIdUsingGET**](docs/UserAuthenticationApi.md#gettenantidusingget) | **Get** /api/v0/whoami | Get your tenant id
*UserAuthenticationApi* | [**LogoutUsingPOST**](docs/UserAuthenticationApi.md#logoutusingpost) | **Post** /api/v0/logout | Log out of the current session
*UserAuthenticationApi* | [**ResetUserPasswordUsingPOST**](docs/UserAuthenticationApi.md#resetuserpasswordusingpost) | **Post** /api/v0/resetpwd | Reset user&#39;s password
*UserAuthenticationApi* | [**UpdateUserEmailWithTokenUsingPOST**](docs/UserAuthenticationApi.md#updateuseremailwithtokenusingpost) | **Post** /api/v0/updateEmail | Update user&#39;s email
*UserAuthenticationApi* | [**UpdateUserPasswordWithTokenUsingPOST**](docs/UserAuthenticationApi.md#updateuserpasswordwithtokenusingpost) | **Post** /api/v0/setpwd | Update user&#39;s password
*UsersManagementApi* | [**ActivateUserUsingPOST3**](docs/UsersManagementApi.md#activateuserusingpost3) | **Post** /api/v0/users/activateUser | Activate a user in a tenant account
*UsersManagementApi* | [**CreateUserAccountUsingPOST3**](docs/UsersManagementApi.md#createuseraccountusingpost3) | **Post** /api/v0/users | Create a user
*UsersManagementApi* | [**DeleteUserUsingDELETE3**](docs/UsersManagementApi.md#deleteuserusingdelete3) | **Delete** /api/v0/users/{userId} | Delete a user in a tenant account
*UsersManagementApi* | [**GetCurrentUserUsingGET**](docs/UsersManagementApi.md#getcurrentuserusingget) | **Get** /api/v0/users/me | Get a \&quot;myself\&quot; user data of Tenant
*UsersManagementApi* | [**GetUserPortalDataUsingGET**](docs/UsersManagementApi.md#getuserportaldatausingget) | **Get** /api/v0/users/me/portaldata | Get the portal data of me
*UsersManagementApi* | [**GetUserUsingGET3**](docs/UsersManagementApi.md#getuserusingget3) | **Get** /api/v0/users/{userId} | Get details of a user in a tenant account
*UsersManagementApi* | [**ListUsersUsingGET1**](docs/UsersManagementApi.md#listusersusingget1) | **Get** /api/v0/users | List all users in a tenant account
*UsersManagementApi* | [**UpdateUserPasswordUsingPOST**](docs/UsersManagementApi.md#updateuserpasswordusingpost) | **Post** /api/v0/users/{userId}/password | Update user password
*UsersManagementApi* | [**UpdateUserPasswordWithTokenWithoutCaptchaUsingPOST**](docs/UsersManagementApi.md#updateuserpasswordwithtokenwithoutcaptchausingpost) | **Post** /api/v0/users/activateTrial | Update user password with a token
*UsersManagementApi* | [**UpdateUserPortalDataUsingPUT**](docs/UsersManagementApi.md#updateuserportaldatausingput) | **Put** /api/v0/users/me/portaldata | Update the portal data of me
*UsersManagementApi* | [**UpdateUserUsingPOST1**](docs/UsersManagementApi.md#updateuserusingpost1) | **Post** /api/v0/users/{userId} | Update a user


## Documentation For Models

 - [ActionPolicy](docs/ActionPolicy.md)
 - [ActionPolicyV0](docs/ActionPolicyV0.md)
 - [ActionTriggers](docs/ActionTriggers.md)
 - [ActionTriggersV0](docs/ActionTriggersV0.md)
 - [Actions](docs/Actions.md)
 - [ActivityRule](docs/ActivityRule.md)
 - [ActivityState](docs/ActivityState.md)
 - [ActivityStateMuteRequest](docs/ActivityStateMuteRequest.md)
 - [AddPartnerTenantRequest](docs/AddPartnerTenantRequest.md)
 - [AdminInfo](docs/AdminInfo.md)
 - [ApiKey](docs/ApiKey.md)
 - [ApiKeyCreationReqWeb](docs/ApiKeyCreationReqWeb.md)
 - [ApiKeySetDebugModeReqWeb](docs/ApiKeySetDebugModeReqWeb.md)
 - [ApiKeyUpdateReqWeb](docs/ApiKeyUpdateReqWeb.md)
 - [Asset](docs/Asset.md)
 - [AssetAlias](docs/AssetAlias.md)
 - [AssetCommandWeb](docs/AssetCommandWeb.md)
 - [AssetCreateReqWeb](docs/AssetCreateReqWeb.md)
 - [AssetParameter](docs/AssetParameter.md)
 - [AssetParameterValue](docs/AssetParameterValue.md)
 - [AssetParamsStatusUpdateReqWeb](docs/AssetParamsStatusUpdateReqWeb.md)
 - [AssetResourceWeb](docs/AssetResourceWeb.md)
 - [AssetSetTargetResourceVersionReqWeb](docs/AssetSetTargetResourceVersionReqWeb.md)
 - [AssetUpdateReqWeb](docs/AssetUpdateReqWeb.md)
 - [AuditLogMessage](docs/AuditLogMessage.md)
 - [AuthReqWeb](docs/AuthReqWeb.md)
 - [AuthResWeb](docs/AuthResWeb.md)
 - [AzureEventHubsAction](docs/AzureEventHubsAction.md)
 - [BinaryPayloadDescription](docs/BinaryPayloadDescription.md)
 - [BinaryPayloadDescriptionTestRequest](docs/BinaryPayloadDescriptionTestRequest.md)
 - [BusinessSettings](docs/BusinessSettings.md)
 - [BusinessUnit](docs/BusinessUnit.md)
 - [CaCertificate](docs/CaCertificate.md)
 - [CaCertificateCreateReqWeb](docs/CaCertificateCreateReqWeb.md)
 - [CaCertificateCreateResWeb](docs/CaCertificateCreateResWeb.md)
 - [Campaign](docs/Campaign.md)
 - [CampaignDefinition](docs/CampaignDefinition.md)
 - [CampaignList](docs/CampaignList.md)
 - [CampaignOperation](docs/CampaignOperation.md)
 - [CampaignOperationState](docs/CampaignOperationState.md)
 - [CampaignOperationStateError](docs/CampaignOperationStateError.md)
 - [CampaignOptions](docs/CampaignOptions.md)
 - [CampaignPerTarget](docs/CampaignPerTarget.md)
 - [CampaignPerTargetList](docs/CampaignPerTargetList.md)
 - [CampaignPlanning](docs/CampaignPlanning.md)
 - [CampaignStatsPerStatus](docs/CampaignStatsPerStatus.md)
 - [CampaignUpdating](docs/CampaignUpdating.md)
 - [ClientCertificatesConfiguration](docs/ClientCertificatesConfiguration.md)
 - [Command](docs/Command.md)
 - [CommandAddRequest](docs/CommandAddRequest.md)
 - [CommandHistory](docs/CommandHistory.md)
 - [CommandPolicy](docs/CommandPolicy.md)
 - [CommandRequest](docs/CommandRequest.md)
 - [CommandResponse](docs/CommandResponse.md)
 - [CommandStatusFilter](docs/CommandStatusFilter.md)
 - [CommandStatusTrigger](docs/CommandStatusTrigger.md)
 - [CommandV0](docs/CommandV0.md)
 - [ConnectorAccounting](docs/ConnectorAccounting.md)
 - [ConnectorNode](docs/ConnectorNode.md)
 - [ConnectorStatistics](docs/ConnectorStatistics.md)
 - [ConnectorStatusResponse](docs/ConnectorStatusResponse.md)
 - [ContactMessage](docs/ContactMessage.md)
 - [ContextContainer](docs/ContextContainer.md)
 - [ContextContainerList](docs/ContextContainerList.md)
 - [CreateFifoBindingRequest](docs/CreateFifoBindingRequest.md)
 - [CsvColumn](docs/CsvColumn.md)
 - [CsvOptions](docs/CsvOptions.md)
 - [CsvPayloadDescription](docs/CsvPayloadDescription.md)
 - [CsvPayloadDescriptionTestRequest](docs/CsvPayloadDescriptionTestRequest.md)
 - [DataBulkItemWeb](docs/DataBulkItemWeb.md)
 - [DataMatchResult](docs/DataMatchResult.md)
 - [DataMatchTest](docs/DataMatchTest.md)
 - [DataMessageFilter](docs/DataMessageFilter.md)
 - [DataMessageTrigger](docs/DataMessageTrigger.md)
 - [DataStoredWeb](docs/DataStoredWeb.md)
 - [DataWeb](docs/DataWeb.md)
 - [DayMetrics](docs/DayMetrics.md)
 - [DeferredListenableFutureResultDslSearchRequestResponseWithHitsOnlyResponseEntityobject](docs/DeferredListenableFutureResultDslSearchRequestResponseWithHitsOnlyResponseEntityobject.md)
 - [DeferredListenableFutureResultstringstring](docs/DeferredListenableFutureResultstringstring.md)
 - [DeferredResultVoid](docs/DeferredResultVoid.md)
 - [DeferredResultboolean](docs/DeferredResultboolean.md)
 - [Device](docs/Device.md)
 - [DeviceActivityFilter](docs/DeviceActivityFilter.md)
 - [DeviceActivityTrigger](docs/DeviceActivityTrigger.md)
 - [DeviceConfigWeb](docs/DeviceConfigWeb.md)
 - [DeviceCreateRequest](docs/DeviceCreateRequest.md)
 - [DeviceCreatedFilter](docs/DeviceCreatedFilter.md)
 - [DeviceCreatedTrigger](docs/DeviceCreatedTrigger.md)
 - [DeviceDeletedFilter](docs/DeviceDeletedFilter.md)
 - [DeviceDeletedTrigger](docs/DeviceDeletedTrigger.md)
 - [DeviceFirmwareVersionValueWeb](docs/DeviceFirmwareVersionValueWeb.md)
 - [DeviceFirmwareWeb](docs/DeviceFirmwareWeb.md)
 - [DeviceGroup](docs/DeviceGroup.md)
 - [DeviceInterface](docs/DeviceInterface.md)
 - [DeviceParameterValue](docs/DeviceParameterValue.md)
 - [DeviceParameterValueWeb](docs/DeviceParameterValueWeb.md)
 - [DeviceParameterWeb](docs/DeviceParameterWeb.md)
 - [DeviceParametersSetRequest](docs/DeviceParametersSetRequest.md)
 - [DeviceResourceVersionValueWeb](docs/DeviceResourceVersionValueWeb.md)
 - [DeviceResourceWeb](docs/DeviceResourceWeb.md)
 - [DeviceSelector](docs/DeviceSelector.md)
 - [DeviceStatusFilter](docs/DeviceStatusFilter.md)
 - [DeviceStatusTrigger](docs/DeviceStatusTrigger.md)
 - [DeviceStreamsResponseWeb](docs/DeviceStreamsResponseWeb.md)
 - [DeviceUpdate](docs/DeviceUpdate.md)
 - [DslSearchRequestResponseWithHitsOnly](docs/DslSearchRequestResponseWithHitsOnly.md)
 - [EmailAction](docs/EmailAction.md)
 - [ErrorResponseWeb](docs/ErrorResponseWeb.md)
 - [ExternalIdentity](docs/ExternalIdentity.md)
 - [FifoBinding](docs/FifoBinding.md)
 - [FifoCreateReqWeb](docs/FifoCreateReqWeb.md)
 - [FifoPublishAction](docs/FifoPublishAction.md)
 - [FifoTopic](docs/FifoTopic.md)
 - [FiringGuard](docs/FiringGuard.md)
 - [FiringGuardGetRequest](docs/FiringGuardGetRequest.md)
 - [FiringGuardResetRequest](docs/FiringGuardResetRequest.md)
 - [FiringRule](docs/FiringRule.md)
 - [FirmwareUpdateWeb](docs/FirmwareUpdateWeb.md)
 - [GeozoneContainer](docs/GeozoneContainer.md)
 - [GeozoneContainerBase](docs/GeozoneContainerBase.md)
 - [Group](docs/Group.md)
 - [GroupCreateRequest](docs/GroupCreateRequest.md)
 - [GroupPath](docs/GroupPath.md)
 - [GroupUpdateRequest](docs/GroupUpdateRequest.md)
 - [GwConfig](docs/GwConfig.md)
 - [GwSystem](docs/GwSystem.md)
 - [HttpPushAction](docs/HttpPushAction.md)
 - [HttpPushTestResult](docs/HttpPushTestResult.md)
 - [HttpPushWebhookTest](docs/HttpPushWebhookTest.md)
 - [InterfaceCapabilities](docs/InterfaceCapabilities.md)
 - [InterfaceCapability](docs/InterfaceCapability.md)
 - [LWM2MDevice](docs/LWM2MDevice.md)
 - [Location](docs/Location.md)
 - [LoraCommand](docs/LoraCommand.md)
 - [LoraCommandWeb](docs/LoraCommandWeb.md)
 - [LoraConnectivityOptions](docs/LoraConnectivityOptions.md)
 - [LoraConnectivityOptionsWeb](docs/LoraConnectivityOptionsWeb.md)
 - [LoraDevice](docs/LoraDevice.md)
 - [LoraDeviceCreateReqWeb](docs/LoraDeviceCreateReqWeb.md)
 - [LoraDeviceLocation](docs/LoraDeviceLocation.md)
 - [LoraDeviceStatsWeb](docs/LoraDeviceStatsWeb.md)
 - [LoraDeviceUpdateReqWeb](docs/LoraDeviceUpdateReqWeb.md)
 - [LoraGatewayData](docs/LoraGatewayData.md)
 - [LoraGatewayInfo](docs/LoraGatewayInfo.md)
 - [LoraGatewayLocation](docs/LoraGatewayLocation.md)
 - [LoraNetworkFilter](docs/LoraNetworkFilter.md)
 - [LoraNetworkSubscriptionDetail](docs/LoraNetworkSubscriptionDetail.md)
 - [LoraNetworkTrigger](docs/LoraNetworkTrigger.md)
 - [Lwm2mDevicePageWeb](docs/Lwm2mDevicePageWeb.md)
 - [MatchingContext](docs/MatchingContext.md)
 - [MatchingFiredFilter](docs/MatchingFiredFilter.md)
 - [MatchingFiredTrigger](docs/MatchingFiredTrigger.md)
 - [MatchingRule](docs/MatchingRule.md)
 - [MessageSelector](docs/MessageSelector.md)
 - [MessageSelectorFilter](docs/MessageSelectorFilter.md)
 - [Metadata](docs/Metadata.md)
 - [MetadataResourceConnectors](docs/MetadataResourceConnectors.md)
 - [MonthMetrics](docs/MonthMetrics.md)
 - [Msisdns](docs/Msisdns.md)
 - [NewData](docs/NewData.md)
 - [NewDeviceInterface](docs/NewDeviceInterface.md)
 - [NewDeviceParameterValue](docs/NewDeviceParameterValue.md)
 - [OAuth2ErrorWeb](docs/OAuth2ErrorWeb.md)
 - [OfferAndOptions](docs/OfferAndOptions.md)
 - [OffersAndOptionsReqWeb](docs/OffersAndOptionsReqWeb.md)
 - [OptionPartnerTenant](docs/OptionPartnerTenant.md)
 - [PageableActionPolicyV0](docs/PageableActionPolicyV0.md)
 - [PageableApiKey](docs/PageableApiKey.md)
 - [PageableAsset](docs/PageableAsset.md)
 - [PageableAssetResourceWeb](docs/PageableAssetResourceWeb.md)
 - [PageableCampaign](docs/PageableCampaign.md)
 - [PageableCampaignPerTarget](docs/PageableCampaignPerTarget.md)
 - [PageableCommandV0](docs/PageableCommandV0.md)
 - [PageableFifoBinding](docs/PageableFifoBinding.md)
 - [PageableFifoTopic](docs/PageableFifoTopic.md)
 - [PageableGroup](docs/PageableGroup.md)
 - [PageableLWM2MDevice](docs/PageableLWM2MDevice.md)
 - [PageableLoraCommand](docs/PageableLoraCommand.md)
 - [PageableLoraDevice](docs/PageableLoraDevice.md)
 - [PageableLoraGatewayInfo](docs/PageableLoraGatewayInfo.md)
 - [PageableResource](docs/PageableResource.md)
 - [PageableResourceUpdateWeb](docs/PageableResourceUpdateWeb.md)
 - [PageableResourceVersion](docs/PageableResourceVersion.md)
 - [PageableUser](docs/PageableUser.md)
 - [PartnerDataItemSwagger](docs/PartnerDataItemSwagger.md)
 - [PartnerTenant](docs/PartnerTenant.md)
 - [PartnerToken](docs/PartnerToken.md)
 - [PayloadDescription](docs/PayloadDescription.md)
 - [PayloadDescriptionMetadata](docs/PayloadDescriptionMetadata.md)
 - [PayloadDescriptionTestResult](docs/PayloadDescriptionTestResult.md)
 - [Pipeline](docs/Pipeline.md)
 - [PipelineFilter](docs/PipelineFilter.md)
 - [PipelineStep](docs/PipelineStep.md)
 - [Polygon](docs/Polygon.md)
 - [RateLimit](docs/RateLimit.md)
 - [ResetPasswordReqWeb](docs/ResetPasswordReqWeb.md)
 - [Resource](docs/Resource.md)
 - [ResourceAddReqWeb](docs/ResourceAddReqWeb.md)
 - [ResourceUpdateReqWeb](docs/ResourceUpdateReqWeb.md)
 - [ResourceUpdateWeb](docs/ResourceUpdateWeb.md)
 - [ResourceVersion](docs/ResourceVersion.md)
 - [ResourceVersionAddReqWeb](docs/ResourceVersionAddReqWeb.md)
 - [ResourceVersionUpdateReqWeb](docs/ResourceVersionUpdateReqWeb.md)
 - [ResponseEntity](docs/ResponseEntity.md)
 - [Rule](docs/Rule.md)
 - [SMSAction](docs/SMSAction.md)
 - [SMSConnectorBusinessSettingsCreationReqWeb](docs/SMSConnectorBusinessSettingsCreationReqWeb.md)
 - [SMSConnectorBusinessSettingsDeleteMsisdnReqWeb](docs/SMSConnectorBusinessSettingsDeleteMsisdnReqWeb.md)
 - [SMSConnectorBusinessSettingsPageWeb](docs/SMSConnectorBusinessSettingsPageWeb.md)
 - [SMSConnectorBusinessSettingsUpdateReqWeb](docs/SMSConnectorBusinessSettingsUpdateReqWeb.md)
 - [SMSConnectorMsisdnPageWeb](docs/SMSConnectorMsisdnPageWeb.md)
 - [SMSConnectorSendSMSReqWeb](docs/SMSConnectorSendSMSReqWeb.md)
 - [SMSConnectorSettings](docs/SMSConnectorSettings.md)
 - [ScopeApplication](docs/ScopeApplication.md)
 - [SearchDataMessage](docs/SearchDataMessage.md)
 - [SelectionCriteria](docs/SelectionCriteria.md)
 - [SendMessageRequest](docs/SendMessageRequest.md)
 - [SettingsSection](docs/SettingsSection.md)
 - [SilentPolicy](docs/SilentPolicy.md)
 - [SimpleStringWeb](docs/SimpleStringWeb.md)
 - [Source](docs/Source.md)
 - [StateChangeFilter](docs/StateChangeFilter.md)
 - [StateChangeTrigger](docs/StateChangeTrigger.md)
 - [StateProcessingFunctionTest](docs/StateProcessingFunctionTest.md)
 - [StateProcessingFunctionTestResult](docs/StateProcessingFunctionTestResult.md)
 - [StateProcessingRule](docs/StateProcessingRule.md)
 - [StoredDataMessage](docs/StoredDataMessage.md)
 - [Targets](docs/Targets.md)
 - [Tenant](docs/Tenant.md)
 - [TenantDayMetrics](docs/TenantDayMetrics.md)
 - [TenantExternalView](docs/TenantExternalView.md)
 - [TenantMonthMetrics](docs/TenantMonthMetrics.md)
 - [TenantSettings](docs/TenantSettings.md)
 - [TenantSettingsUpdateReqWeb](docs/TenantSettingsUpdateReqWeb.md)
 - [TenantStatsWeb](docs/TenantStatsWeb.md)
 - [TenantWithOfferWeb](docs/TenantWithOfferWeb.md)
 - [Thresholds](docs/Thresholds.md)
 - [Traffic](docs/Traffic.md)
 - [UpdateConnectorNodeRequest](docs/UpdateConnectorNodeRequest.md)
 - [UpdateDeviceFirmwareReqWeb](docs/UpdateDeviceFirmwareReqWeb.md)
 - [UpdateDeviceResourceReqWeb](docs/UpdateDeviceResourceReqWeb.md)
 - [UpdateEmailReqWeb](docs/UpdateEmailReqWeb.md)
 - [UpdateInterfaceReqWeb](docs/UpdateInterfaceReqWeb.md)
 - [UpdatePasswordReqWeb](docs/UpdatePasswordReqWeb.md)
 - [UpdatePasswordWithTokenReqWeb](docs/UpdatePasswordWithTokenReqWeb.md)
 - [User](docs/User.md)
 - [UserActivationReqWeb](docs/UserActivationReqWeb.md)
 - [UserCreationReqWeb](docs/UserCreationReqWeb.md)
 - [UserUpdateReqWeb](docs/UserUpdateReqWeb.md)
 - [Vendor](docs/Vendor.md)


## Documentation For Authorization

 Endpoints do not require authorization.


## Documentation for Utility Methods

Due to the fact that model structure members are all pointers, this package contains
a number of utility functions to easily obtain pointers to values of basic types.
Each of these functions takes a value of the given basic type and returns a pointer to it:

* `PtrBool`
* `PtrInt`
* `PtrInt32`
* `PtrInt64`
* `PtrFloat`
* `PtrFloat32`
* `PtrFloat64`
* `PtrString`
* `PtrTime`

## Author

liveobjects.support@orange.com

 readmeEtag: '"ba9e344ed3a20d2dea4f108b23fd997553984730"' readmeLastModified: Sun, 15 Nov 2020 10:24:46 GMT repositoryId: 290238732 description: Go client library for Orange Business Services Live Objects API created: '2020-08-25T14:33:57Z' updated: '2020-11-15T18:11:31Z' language: Shell archived: false stars: 0 watchers: 0 forks: 0 owner: efficap-energie logo: https://avatars.githubusercontent.com/u/63661671?v=4 repoEtag: '"e562b09911fcdac6c14bcfcd9d696af668ae770f6435b95388dfffd5b00c9ed6"' repoLastModified: Sun, 15 Nov 2020 18:11:31 GMT foundInMaster: true category: User Interfaces id: d52e311c4d9c5e91d1cad49a09e2f179 - source: openapi3 tags repository: https://github.com/dmotta/c3alert v3: true id: 2b53e75c670fbcb4565f5088abe0242e repositoryMetadata: base64Readme: >- IyBHbyBBUEkgU2VydmVyIGZvciBzd2FnZ2VyCgpObyBkZXNjcmlwdGlvbiBwcm92aWRlZCAoZ2VuZXJhdGVkIGJ5IFN3YWdnZXIgQ29kZWdlbiBodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1jb2RlZ2VuKQpodHRwczovL3N3YWdnZXIuaW8vZG9jcy9zcGVjaWZpY2F0aW9uL2Rlc2NyaWJpbmctcmVxdWVzdC1ib2R5LwoKCiMjIE92ZXJ2aWV3ClRoaXMgc2VydmVyIHdhcyBnZW5lcmF0ZWQgYnkgdGhlIFtzd2FnZ2VyLWNvZGVnZW5dCihodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1jb2RlZ2VuKSBwcm9qZWN0LiAgCkJ5IHVzaW5nIHRoZSBbT3BlbkFQSS1TcGVjXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbikgZnJvbSBhIHJlbW90ZSBzZXJ2ZXIsIHlvdSBjYW4gZWFzaWx5IGdlbmVyYXRlIGEgc2VydmVyIHN0dWIuICAKLQoKVG8gc2VlIGhvdyB0byBtYWtlIHRoaXMgeW91ciBvd24sIGxvb2sgaGVyZToKCltSRUFETUVdKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWNvZGVnZW4vYmxvYi9tYXN0ZXIvUkVBRE1FLm1kKQoKLSBBUEkgdmVyc2lvbjogMS4wLjAKLSBCdWlsZCBkYXRlOiAyMDIxLTA2LTAyVDA1OjAyOjAxLjgwNFpbR01UXQoKCiMjIyBSdW5uaW5nIHRoZSBzZXJ2ZXIKVG8gcnVuIHRoZSBzZXJ2ZXIsIGZvbGxvdyB0aGVzZSBzaW1wbGUgc3RlcHM6CgpgYGAKZ28gcnVuIG1haW4uZ28KYGBgCgouL2thZmthLWNvbnNvbGUtY29uc3VtZXIgLS1ib290c3RyYXAtc2VydmVyIGxvY2FsaG9zdDo5MDkyIC0tZnJvbS1iZWdpbm5pbmcgLS10b3BpYyBtb25pdG9yLWFsZXJ0 readmeEtag: '"76cc6ce20fe435a1284ccca160b2e037f9248f1a"' readmeLastModified: Fri, 31 Dec 2021 16:55:48 GMT repositoryId: 376087712 description: c3alert created: '2021-06-11T16:47:22Z' updated: '2022-07-01T17:11:47Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: dmotta logo: https://avatars.githubusercontent.com/u/85045?v=4 repoEtag: '"7ca9788c6b2a9f28e10738b4355bcec2389c7cb80a198031ecf5c4902073b59a"' repoLastModified: Fri, 01 Jul 2022 17:11:47 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/marcsechet/good-smash v3: true repositoryMetadata: base64Readme: >- WyFbQ2lyY2xlQ0ldKGh0dHBzOi8vY2lyY2xlY2kuY29tL2doL01hcmNTZWNoZXQvZ29vZC1zbWFzaC5zdmc/c3R5bGU9c2hpZWxkKV0oaHR0cHM6Ly9jaXJjbGVjaS5jb20vZ2gvTWFyY1NlY2hldC9nb29kLXNtYXNoKQpbIVtDcmVhdGVkIEJhZGdlXShodHRwczovL2JhZGdlcy5wdWZsZXIuZGV2L2NyZWF0ZWQvTWFyY1NlY2hldC9nb29kLXNtYXNoKV0oaHR0cHM6Ly9naXRodWIuY29tL01hcmNTZWNoZXQvZ29vZC1zbWFzaCkKWyFbQ3JlYXRlZCBCYWRnZV0oaHR0cHM6Ly9iYWRnZXMucHVmbGVyLmRldi91cGRhdGVkL01hcmNTZWNoZXQvZ29vZC1zbWFzaCldKGh0dHBzOi8vZ2l0aHViLmNvbS9NYXJjU2VjaGV0L2dvb2Qtc21hc2gpCiFbTGluZXMgb2YgY29kZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90b2tlaS9saW5lcy9naXRodWIvTWFyY1NlY2hldC9nb29kLXNtYXNoKQohW0dpdEh1YiByZXBvIHNpemVdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3JlcG8tc2l6ZS9NYXJjU2VjaGV0L2dvb2Qtc21hc2gpCiFbR2l0SHViXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL01hcmNTZWNoZXQvZ29vZC1zbWFzaCkKCiMgR29vZCBTbWFzaAoKUHJvamVjdHMgY29udGFpbmluZyBhIHNpbmdsZSBCTyB1c2VkIGZvciBtdWx0aXBsZSBmcm9udCBlbmQgOgotIFNtYXNoIAotIFBhcGljb2xvCi0gUMOpcGl0ZXMKCk1heWJlIHNvbWUgZGF5IHRoaXMgQk8gd2lsbCBiZSBzcGxpdCBpbnRvIG1pY3Jvc2VydmljZXMgYnV0IG5vdCBzdXJlIHRoZSBmcmVlIGhlcm9rdSBjYW4gbWFuYWdlIHRoYXQuCgpBcmNoaXRlY3R1cmU6ClRoZSBwcm9qZWN0IGhhcyB0aGUgZm9sbG93aW5nIG1vZHVsZXMgOgotIHNtYXNoLWFwcCAoY29udGFpbnMgY29uZmlnLCBjb250cm9sbGVycywgbWFwcGVycyBhbmQgdXRpbHMpCi0gc21hc2gtc2VydmljZQotIHNtYXNoLXBlcnNpc3QgKGNvbnRhaW5zIGRhbyBhbmQgZW50aXRpZXMpCi0gc21hc2gtc3BlYyAoY29udGFpbnMgb3BlbmFwaSBzcGVjcykKCkluc3RhbGwgbm90ZXMgOiAKVGhlIGdlbmVyYXRlZCBzcGVjIG1pZ2h0IG5vdCBiZSBhY2Nlc3NpYmxlIGluIG90aGVyIG1vZHVsZXMgb2YgdGhlIHByb2plY3QuIFRvIGZpeCB0aGlzLCBvbiBJbnRlbGxpSiwgeW91IGNhbiBvcGVuIHByb2plY3Qgc2V0dGluZ3MgLT4gbW9kdWxlcyAtPiBnb29kc21hc2gtYXBwIC0+IGRlcGVuZGVuY2llcyAtPiBhZGQgbW9kdWxlIGRlcGVuZGVuY3kgZ29vZHNtYXNoLXNwZWMuIElmIHlvdSBoYXZlIGFuIGVycm9yIHdoaWxlIHNhdmluZyBiZWNhdXNlIGNvbnRlbnQgcm9vdCBpcyBhbHJlYWR5IGRlZmluZWQgaW4gb3RoZXIgbW9kdWxlIHByb2plY3Qgc2V0dGluZ3MgLT4gbW9kdWxlcyAtPiBnb29kc21hc2ggLT4gc291cmNlcyAtPiByZW1vdmUgcGF0aHMgaW4gdGhlIGNvbnRlbnQgcm9vdCB0YWIgbm90IGJlbG9uZ2luZyB0byBvdGhlciBtb2R1bGVzLgoKSG93IHRvIGZpeCBTcHJpbmcgQm9vdCBNdWx0aS1Nb2R1bGUgbWF2ZW4gcHJvamVjdCByZXBhY2thZ2UgZmFpbGVkIChTcHJpbmcgcHJvamVjdHMgd2l0aCBtdWx0aSBtb2R1bGVzKSA6IHB1dCB0aGUgc3ByaW5nLWJvb3QtbWF2ZW4tcGx1Z2luIGluIHRoZSBwb20gb2YgdGhlIG1vZHVsZSBjb250YWluaW5nIHRoZSBBcHBsaWNhdGlvbiBjbGFzcyAoaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9hLzU3OTgzODAyKQoKCiMgQ29udHJpYnV0b3JzClshW0NvbnRyaWJ1dG9ycyBEaXNwbGF5XShodHRwczovL2JhZGdlcy5wdWZsZXIuZGV2L2NvbnRyaWJ1dG9ycy9NYXJjU2VjaGV0L2dvb2Qtc21hc2g/c2l6ZT01MCZwYWRkaW5nPTUmYm90cz10cnVlKV0oaHR0cHM6Ly9naXRodWIuY29tL01hcmNTZWNoZXQvZ29vZC1zbWFzaCkK readmeEtag: '"4fdca4eb7c38046ec40a86249eb290fc660c7280"' readmeLastModified: Fri, 18 Jun 2021 22:10:27 GMT repositoryId: 366968135 description: Projects containing a single BO used for multiple front end created: '2021-05-13T07:22:40Z' updated: '2021-06-18T22:10:30Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: MarcSechet logo: https://avatars.githubusercontent.com/u/30333108?v=4 license: Apache-2.0 repoEtag: '"18f6c1691444d39030a3330df32b455a679910d2a844fa187166bd350651b673"' repoLastModified: Fri, 18 Jun 2021 22:10:30 GMT foundInMaster: true category: SDK id: 5e62876014bbb73e70451d51165059f0 - source: openapi3 tags repository: https://github.com/romabilka/go-crud v3: true repositoryMetadata: base64Readme: >- IyBHTyBDUlVECgojI0RhdGEgaXMgc3RvcmVkIGluIG1lbW9yeQoKIyMgSG93IHRvIHJ1bgoxLiBDbG9uZSB0aGUgcHJvamVjdAoyLiBgYGBnbyBydW4gY21kL2dvLWNydWQvbWFpbi5nb2BgYAoKIyMgVG8gdHJ5IGdvIHRvIHRoZSBsaW5rCmh0dHA6Ly9sb2NhbGhvc3QvYXBpLwoKIyMgTGljZW5zZQpbTUlUIGxpY2Vuc2VdKExJQ0VOU0UubWQpLg== readmeEtag: '"3da803779915f63984a8b0958c7ddb3d9af605f1"' readmeLastModified: Thu, 03 Mar 2022 09:35:36 GMT repositoryId: 424592171 description: null created: '2021-11-04T12:45:03Z' updated: '2022-03-03T10:01:48Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: RomaBilka logo: https://avatars.githubusercontent.com/u/28790784?v=4 license: NOASSERTION repoEtag: '"0dd175955c33b5d66b698d7ae67732ff8bd206bc42aa1a59dc3e2bd7660b4d9f"' repoLastModified: Thu, 03 Mar 2022 10:01:48 GMT foundInMaster: true category: Server Implementations id: 3d759a8fa4c01f11a2072e83f412475b - source: openapi3 tags repository: https://github.com/nanonets/api-docs v3: true repositoryMetadata: base64Readme: >- IyBOYW5vTmV0cyBBUEkgUmVmZXJlbmNlCgpodHRwczovL25hbm9uZXRzLmNvbS9kb2N1bWVudGF0aW9uCg== readmeEtag: '"e4ae4c014bd99eefd6b2d3153ecf534ac8b176d1"' readmeLastModified: Fri, 12 Nov 2021 08:47:05 GMT repositoryId: 426224312 description: NanoNets API Reference. created: '2021-11-09T12:41:15Z' updated: '2021-11-12T08:47:21Z' language: HTML archived: false stars: 0 watchers: 4 forks: 0 owner: NanoNets logo: https://avatars.githubusercontent.com/u/30330951?v=4 license: MIT repoEtag: '"366230e4a0286646c8fa34242640c66754d10bd257edccd2c64048e4e7046a62"' repoLastModified: Fri, 12 Nov 2021 08:47:21 GMT foundInMaster: true category: - Testing - Server Implementations id: 1ac680c35c2904f43169d32e6428a45d - source: openapi3 tags repository: https://github.com/subodhjenasymbl/symbl-api v3: true id: 8893554b2dfc65220058c10804c4a356 repositoryMetadata: base64Readme: >- IyBTeW1ibCBBUEkgRG9jdW1lbnRhdGlvbgpTeW1ibCBSRVNUIEFQSSBEb2N1bWVudGF0aW9uCg== readmeEtag: '"61e74e83e83e78574e35a436b74ccf0c27c946ca"' readmeLastModified: Fri, 03 Jun 2022 07:33:38 GMT repositoryId: 499417683 description: Symbl OpenAPI3 Spec created: '2022-06-03T07:21:22Z' updated: '2022-06-03T07:32:43Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: subodhjenasymbl logo: https://avatars.githubusercontent.com/u/106313352?v=4 repoEtag: '"29b555bf3a7f8ce208bf0c7248228f84846f5c3b02fdeefce4832b60f3e4a6fb"' repoLastModified: Fri, 03 Jun 2022 07:32:43 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/onmetal/openapi-extractor v3: true id: 8e0801e45b5f14df2617fa79faf4a28c repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWV4dHJhY3RvcgoKWyFbVGVzdF0oaHR0cHM6Ly9naXRodWIuY29tL29ubWV0YWwvb3BlbmFwaS1leHRyYWN0b3IvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL29ubWV0YWwvb3BlbmFwaS1leHRyYWN0b3IvYWN0aW9ucy93b3JrZmxvd3MvdGVzdC55bWwpClshW1BScyBXZWxjb21lXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL1BScy13ZWxjb21lLWJyaWdodGdyZWVuLnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vbWFrZWFwdWxscmVxdWVzdC5jb20pClshW0dpdEh1YiBMaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL3N0YXRpYy92MT9sYWJlbD1MaWNlbnNlJm1lc3NhZ2U9QXBhY2hlLTIuMCZjb2xvcj1ibHVlJnN0eWxlPWZsYXQtc3F1YXJlKV0oTElDRU5TRSkKClRoZSBgb3BlbmFwaS1leHRyYWN0b3JgIGV4dHJhY3RzIHRoZSBPcGVuQVBJIHYyIGFuZCB2MyBzcGVjaWZpY2F0aW9ucyBvZiBhIGdpdmVuIEt1YmVybmV0ZXMgQVBJIHNlcnZlci4KCiMjIEluc3RhbGxhdGlvbgoKIyMjIEZyb20gc291cmNlCgpUbyBpbnN0YWxsIHRoZSBgb3BlbmFwaS1leHRyYWN0b3JgIGJpbmFyeSBpbnRvIHlvdXIgR28gYmluIHBhdGggcnVuCgpgYGBiYXNoCmdvIGluc3RhbGwgZ2l0aHViLmNvbS9vbm1ldGFsL29wZW5hcGktZXh0cmFjdG9yL2NtZC9vcGVuYXBpLWV4dHJhY3RvckBtYWluCmBgYAoKIyMgVXNhZ2UKCiMjIyBDb21tYW5kIGJhc2VkIGV4dHJhY3Rpb24KCkluIGNhc2UgeW91IGhhdmUgdGhlIGFwaSBzZXJ2ZXIgYmluYXJ5IHByZXNlbnQsIHlvdSBjYW4gZXh0cmFjdCB0aGUgT3BlbkFQSSBzcGVjaWZpY2F0aW9ucyBieSBydW5uaW5nCgpgYGBzaGVsbApvcGVuYXBpLWV4dHJhY3RvciAtLWFwaXNlcnZlci1jb21tYW5kPTxQQVRILVRPLUFQSVNFUlZFUi1CSU4+IFwKICAtLWFwaXNlcnZpY2VzPTxQQVRILVRPLUFQSVNFUlZJQ0VTLURJUj4KYGBgCgojIyMgR28gbW9kdWxlIGJhc2VkIGV4dHJhY3Rpb24KClRoZSBbYHNhbXBsZWBdKC9zYW1wbGUpIGZvbGRlciBjb250YWlucyBhbiBleGFtcGxlIG9uIGhvdyB0byBleHRyYWN0IHRoZSBPcGVuIEFQSSBzcGVjIGZyb20gYW4gYXBpIHNlcnZlciBwYWNrYWdlLiBJbiAKb3VyIGV4YW1wbGUgd2UgYXJlIHVzaW5nIHRoZSBbYG9ubWV0YWwtYXBpYF0oaHR0cHM6Ly9naXRodWIuY29tL29ubWV0YWwvb25tZXRhbC1hcGkpIGFnZ3JlZ2F0ZWQgYXBpIHNlcnZlci4KCmBgYHNoZWxsCm9wZW5hcGktZXh0cmFjdG9yIC0tYXBpc2VydmVyLXBhY2thZ2U9Z2l0aHViLmNvbS9vbm1ldGFsL29ubWV0YWwtYXBpL2NtZC9vbm1ldGFsLWFwaXNlcnZlciBcCiAgLS1hcGlzZXJ2ZXItYnVpbGQtb3B0cz1tb2QgXAogIC0tYXBpc2VydmljZXM9PFBBVEgtVE8tQVBJU0VSVklDRVMtRElSPgpgYGAKCkluIGNhc2UgeW91IHdhbnQgdG8gdXNlIHlvdXIgb3duIHBhY2thZ2UsIGZpcnN0IGBnbyBnZXRgIGl0IHNvIHlvdSBoYXZlIHRvIGNvcnJlY3QgZGVwZW5kZW5jaWVzIGluIHlvdXIgYGdvLm1vZGAgZmlsZSBhbmQKYWRqdXN0IHRoZSBgLS1hcGlzZXJ2ZXItcGFja2FnZWAgZmxhZyBhY2NvcmRpbmdseS4KCiMjIyBPdXRwdXQKClRoZSBleHRyYWN0ZWQgT3BlbkFQSSB2MiBhbmQgdjMgZmlsZXMgY2FuIGJlIGZvdW5kIGluIGN1cnJlbnQgZm9sZGVyIHdoZXJlIHRoZSB2MiB2ZXJzaW9uIHdpbGwgYmUgc3RvcmVkIGluIHRoZSBgc3dhZ2dlci5qc29uYApmaWxlIGFuZCB0aGUgdjMgdmVyc2lvbnMgd2lsbCBiZSBzdG9yZWQgaW4gaW5kaXZpZHVhbCBmaWxlcyBwZXIgZ3JvdXAgaW4gdGhlIGAuL3YzYCBmb2xkZXIuCgpUbyBvdmVycmlkZSB0aGUgbG9jYXRpb24gb2YgdGhlIG91dHB1dCBwYXNzIG9uIHRoZSBgLS1vdXRwdXRgIGZsYWcgZS5nLiB2aWEgYC0tb3V0cHV0PWRldmAgc3RvcmUgZXh0cmFjdCB0aGUgZmlsZXMgaW50bwp0aGUgYC4vZGV2YCBmb2xkZXIuCgojIyBDb250cmlidXRpbmcKCldlJ2QgbG92ZSB0byBnZXQgZmVlZGJhY2sgZnJvbSB5b3UuIFBsZWFzZSByZXBvcnQgYnVncywgc3VnZ2VzdGlvbnMgb3IgcG9zdCBxdWVzdGlvbnMgYnkgb3BlbmluZyBhIEdpdEh1YiBpc3N1ZS4KCiMjIExpY2Vuc2UKCkNvcHlyaWdodCAyMDIyLgoKTGljZW5zZWQgdW5kZXIgdGhlIEFwYWNoZSBMaWNlbnNlLCBWZXJzaW9uIDIuMCAodGhlICJMaWNlbnNlIik7CnlvdSBtYXkgbm90IHVzZSB0aGlzIGZpbGUgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0CgogICAgaHR0cDovL3d3dy5hcGFjaGUub3JnL2xpY2Vuc2VzL0xJQ0VOU0UtMi4wCgpVbmxlc3MgcmVxdWlyZWQgYnkgYXBwbGljYWJsZSBsYXcgb3IgYWdyZWVkIHRvIGluIHdyaXRpbmcsIHNvZnR3YXJlCmRpc3RyaWJ1dGVkIHVuZGVyIHRoZSBMaWNlbnNlIGlzIGRpc3RyaWJ1dGVkIG9uIGFuICJBUyBJUyIgQkFTSVMsCldJVEhPVVQgV0FSUkFOVElFUyBPUiBDT05ESVRJT05TIE9GIEFOWSBLSU5ELCBlaXRoZXIgZXhwcmVzcyBvciBpbXBsaWVkLgpTZWUgdGhlIExpY2Vuc2UgZm9yIHRoZSBzcGVjaWZpYyBsYW5ndWFnZSBnb3Zlcm5pbmcgcGVybWlzc2lvbnMgYW5kCmxpbWl0YXRpb25zIHVuZGVyIHRoZSBMaWNlbnNlLgo= readmeEtag: '"9e4b91b063630400194626098da2de0885a9adf8"' readmeLastModified: Tue, 28 Nov 2023 11:08:11 GMT repositoryId: 724646702 description: null created: '2023-11-28T14:05:19Z' updated: '2023-11-28T14:09:15Z' language: Go archived: true stars: 0 watchers: 3 forks: 0 owner: onmetal logo: https://avatars.githubusercontent.com/u/65116310?v=4 repoEtag: '"e9b47044128b2127cf61935ee7931ae67214b61d88629613269a06a2e90bdbde"' repoLastModified: Tue, 28 Nov 2023 14:09:15 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/experienced-dev/java-starter-template v3: true id: b514b5f0860f68c23c937fe5fa4afc00 repositoryMetadata: base64Readme: >- IyBHZXR0aW5nIFN0YXJ0ZWQKCiMjIyBEb2N1bWVudGF0aW9uCgpBdXRvIGdlbmVyYXRlZCBkb2N1bWVudGF0aW9uCgoqIFN3YWdnZXItIFVJIC9zd2FnZ2VyLXVpL2luZGV4Lmh0bWwKKiBPcGVuQXBpIC92My9hcGktZG9jcwo= readmeEtag: '"4f6c54b564e35aa424d4ce74f40e267e4d9502e7"' readmeLastModified: Thu, 30 Nov 2023 20:57:37 GMT repositoryId: 678730576 description: Java Starter Template created: '2023-08-15T08:33:21Z' updated: '2023-08-26T14:47:42Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: experienced-dev logo: https://avatars.githubusercontent.com/u/61656932?v=4 license: MIT repoEtag: '"68ffc778c54654bd3a0188fe0397dd91c67a996de59ce2c03d417068895ffed5"' repoLastModified: Sat, 26 Aug 2023 14:47:42 GMT category: - Parsers - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/ibakchod/oatpp-todo v3: true id: b43c8b235c00ce710c0a040e2c74b19a repositoryMetadata: base64Readme: IyBRYXRwcCBUb2Rv readmeEtag: '"21e9a325a92b37d72af29eaf59f790cb41a2a2f5"' readmeLastModified: Sun, 07 Aug 2022 10:44:38 GMT repositoryId: 505103853 description: A light-weight To-Do restful API created: '2022-06-19T12:29:45Z' updated: '2022-08-21T09:14:06Z' language: C++ archived: false stars: 0 watchers: 1 forks: 0 owner: ibakchod logo: https://avatars.githubusercontent.com/u/107798368?v=4 license: MIT repoEtag: '"c62701190182d482d36c28a4721162fca61b173005221b0f613a2d9047bd4c6b"' repoLastModified: Sun, 21 Aug 2022 09:14:06 GMT category: - Data Validators - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/simonmicro/access-control v3: true id: 8ace88db44940eca607aace7f40c1b59 repositoryMetadata: base64Readme: >- # What is this?

![login](docs/img/login.png)
![dashboard](docs/img/dash.png)

This application was developed to work inside a cloud-native environment (Kubernetes!) and to protect its web-applications with an additional layer of security.

There is no need for e.g. a VPN-client on the _users_ side, instead their requests will simply be rewritten to the authentication layer of this software. This solution is not perfect and only meant as an additional of security: It works by whitelisting _users_ public IPv4 for a set of _scopes_ (like `stuff.example.com`) by providing a list of IPv4-addresses for the use inside configmaps (e.g. for use inside Nginx). It therfore works best for "small" amount of _users_ only.

## Services
* `api` Provides a REST-ful endpoint with WebSockets and documentation
* `config` Applies a given YAML to the database (supports live changes)
* `dashboard` Provides a static set of files as the dashboard, expects the path `/api` to be routed to the `api` service - to change that (e.g. to use an own sub-domain for the api), you have to compile the dashboard yourself...
* `provision` This container uses the `kubectl` command to apply changes to the pods and configmaps

## Getting started

### API, Dashboard & more
All services (except the `dashboard`) need access to a Redis database instance. To run such instance locally, try this (according to its [documentation](https://hub.docker.com/_/redis)):

```bash
docker run -v "$(pwd)/redis:/data" -p 6379:6379 redis redis-server --save 60 1 --loglevel warning
```

To configure the services to use this instance, take a look into the `.gitlab/*/Dockerfile` files for the services environment variables.

As you may have noted, the service `config` expects a YAML file to prepare the _scopes_ and _users_. Take a look into the `config.sample.yaml` to learn more!

You then have to expose the `dashboard` and `api` services to the _users_ by e.g. using a Nginx instace as a reverse proxy. For the `dashboard` just proxy requests to the `dashboard` service and for the `api`, ensure the path `/api` of your Nginx is proxied to the `api` service (take a look at [Uvicorn Deployment](https://www.uvicorn.org/deployment/#running-behind-nginx) and the api environment variables).

**Rate limits** are strongly recommended to add an additional layer of security to your api endpoint! Here is an example for a Nginx configuration of the `dashboard` with the `api` service mapped to `/api`:

```nginx
limit_req_zone $binary_remote_addr zone=access_api_limit:10m rate=4r/s;

server {
    listen 443 ssl;
    server_name access-control.example.com;

    ssl_certificate ...;
    ssl_certificate_key ...;

    location / {
        set $endpoint dashboard-service.default.svc.cluster.local;
        resolver kube-dns.kube-system.svc.cluster.local;
        proxy_pass http://$endpoint;
    }

    location /api/ {
        # -> https://stackoverflow.com/a/33325584
        rewrite ^/api/(.*) /$1  break;
        resolver kube-dns.kube-system.svc.cluster.local;
        proxy_pass http://api-service.default.svc.cluster.local$uri$is_args$args;
        # Limit interaction to the API
        limit_req zone=access_api_limit burst=32 delay=8;
        limit_req_status 429;
        # Inform target host about proxy client...
        proxy_set_header Host $host;
        proxy_set_header X-Real-IP  $remote_addr;
        proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for;
        proxy_set_header X-Forwarded-Proto $scheme;
        proxy_set_header X-Forwarded-Protocol $scheme;
        proxy_set_header X-Forwarded-Host $http_host;
        # Support WebSocket connections...
        proxy_http_version 1.1; # Default is 1.0
        proxy_set_header Upgrade $http_upgrade;
        proxy_set_header Connection "upgrade";
        # Wait for x seconds for the uplink...
        proxy_connect_timeout 1d; # Keep websockets alive!
        proxy_send_timeout 1d; # Keep websockets alive!
        proxy_read_timeout 1d; # Keep websockets alive!
        send_timeout 10;
    }
}
```

### Configure your Nginx
The `provision` service of this project is compartible with Nginx and its `geo_ip` module. This is achieved by updating a configmap inside the Kubernetes cluster, which holds a list of IPs for the configured _scopes_. As you may have noted, Nginx does not reload the configuration files on its own if any change occurs. For this a small sidecar is required (e.g. below). Furthermore, you also have to configure your Nginx to respect the lists from the (then mounted) configmap. See below for an example too.

Nginx with a sidecar for automatic reloads:
```yaml
apiVersion: apps/v1
kind: DaemonSet
metadata:
  name: nginx
spec:
  selector:
    matchLabels:
      app: nginx
  template:
    metadata:
      labels:
        app: nginx
    spec:
      shareProcessNamespace: true # Required for sending the reload signal to Nginx
      containers:
      - name: config-reloader
        image: realsimonmicro/debian-inotifywait
        # Do NOT listen to all events, otherwise you will get stuck in a loop (because NGINX will trigger "close" after reading them with "access").
        args: ["-c", "echo 'Ready to go!'; inotifywait -e modify -e create -e modify -e delete -m -q -r --format '%w%f' '/geo_ips/' | while read -r path; do echo \"$(date): $path changed!\"; echo \"\tSkipping $(timeout 3 cat | wc -l) further changes...\"; kill -s HUP $(pgrep -o nginx); done"]
        volumeMounts:
        - name: nginx-config-geo-ips-volume
          mountPath: /geo_ips/
          readOnly: true
      - name: nginx
        image: nginx
        volumeMounts:
        - name: nginx-config-geo-ips-volume
          mountPath: /geo_ips/
          readOnly: true
        ports:
        - containerPort: 80
        - containerPort: 443
      volumes:
      - name: nginx-config-geo-ips-volume
        configMap:
          name: nginx-config-geo-ips
```

Nginx configuration for `geo_ip` lists from `/geo_ips/dummy.list`:
```nginx
geo $dummy_list {
    include /geo_ips/dummy.list;
}

server {
    listen 443 ssl;
    server_name service.example.com;

    ssl_certificate ...;
    ssl_certificate_key ...;

    location / {
        if ($dummy_list = 1) {
            # root /success;
            # ...or use inline HTML...
            add_header Content-Type text/html;
            return 200 '<html><body><h1>Access granted!</h1></body></html>';
        }
        if ($dummy_list != 1) {
            # You can also use the "rewrite" instead of the "root" (or whatever) statement to directly send the user to the access-control webinterface.
            # Make sure to insert the external reachible hostname below!
            # rewrite ^ $scheme://EXTERNAL_HOSTNAME_OF_INTERFACE/request/$scheme/$host$uri$is_args$args redirect;

            # root /failure;
            # ...or use inline HTML...
            add_header Content-Type text/html;
            return 403 '<html><body><h1>Access denied!</h1></body></html>';
        }
    }
}
``` readmeEtag: '"227a34b411f34342121dee11ad0e650701c70a6a"' readmeLastModified: Wed, 15 Mar 2023 23:11:53 GMT repositoryId: 535276710 description: An access control system for cloud-native applications. created: '2022-09-11T11:28:07Z' updated: '2024-12-30T17:06:47Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: simonmicro logo: https://avatars.githubusercontent.com/u/24966116?v=4 repoEtag: '"85353ab4a8f83717b2f2e8844e6b67818c6ad38ee8daa3e4ec63b172f17d80c8"' repoLastModified: Mon, 30 Dec 2024 17:06:47 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/isaguler/weather-api v3: true id: bbf658d41ec2f229226b7c500a531c82 repositoryMetadata: base64Readme: >- IyMjIGVudiBhbmQgdGVjaHMKLSBKYXZhIDE3ICYgU3ByaW5nQm9vdCAzCi0gRG9ja2VyICYgRG9ja2VyLUNvbXBvc2UKLSBQcm9tZXRoZXVzCi0gR3JhZmFuYQotIE9wZW5BUEkgRG9jCi0gV2VhdGhlclN0YWNrIEFwaQotIEgyIERhdGFiYXNlCi0gU3ByaW5nIENhY2hlCi0gUmVzaWxpZW5jZTRqCi0gU2xmNGoKCiMjIyBlbmRwb2ludHMKLSBncmFmYW5hIC0+IGxvY2FsaG9zdDozMDAwCi0gcHJvbWV0aGV1cyAtPiBsb2NhbGhvc3Q6OTA5MAotIGFjdHVhdG9yIC0+IGxvY2FsaG9zdDo5NTk1L2FjdHVhdG9yCi0gYXBwIC0+IGxvY2FsaG9zdDo5NTk1 readmeEtag: '"8088c523432ab808db31a1221ff33156caf0e32d"' readmeLastModified: Thu, 30 Mar 2023 07:42:41 GMT repositoryId: 621183272 description: Current Weather Information for Cities created: '2023-03-30T06:42:56Z' updated: '2023-04-03T14:30:06Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: isaguler logo: https://avatars.githubusercontent.com/u/96476351?v=4 repoEtag: '"c6c9f3a20dac5a51fb7dc2defdd01f544c4cb91d56bc00976ea897f008fd99dd"' repoLastModified: Mon, 03 Apr 2023 14:30:06 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/interes-group/b-vsa-ls23-project2-solution v3: true id: 316ffa1df03fb79513d3c55058386698 repositoryMetadata: base64Readme: >- # B-VSA LS 22/23 - Semestrálny projekt 2

![Java 1.8](https://img.shields.io/badge/Java-1.8-red)
![EclipseLink 2.7.11](https://img.shields.io/badge/EclipseLink-2.7.11-blue)
![PostgreSQL Driver](https://img.shields.io/badge/PostgreSQL-42.5.4-green)
![Jersey JAX-RS](https://img.shields.io/badge/Jersey-2.39.1-orange)
[![Public domain](https://img.shields.io/badge/License-Unlicense-lightgray)](https://unlicense.org)

Cieľom 2. semestrálneho projektu je implementovať jednoduchú webovú aplikáciu publikujúcu REST API podľa definovanej
špecifikácie. Projekt má nadväzovať na 1. semestrálny projekt, v ktorom ste implementovali spojenie s SQL databázou
použitím technológie JPA, takže aj v tomto projekte majú byť dodržané vzťahy a logika medzi jednotlivými entitami.

V tomto projekte môžte vychádzať z vášho riešenia 1. semestrálneho projektu, alebo využiť poskytnutú [referenčnú implementáciu](https://github.com/Interes-Group/b-vsa-ls23-project1-solution).
Zachovajte asociácie a objekty definované v 1. projekte. Nakoľko účelom tohto zadania je implementácia REST webových služieb,
môžte upraviť riešenie z predchádzajúceho zadania podľa svojho najlepšieho vedomia a svedomia, tak aby splnilo toto zadanie.
Pre vypracovanie sa môžte inšpirovať [projektami z cvičení](https://github.com/Interes-Group/b-vsa-cvicenia). 
Dbajte na dodržiavanie pokynov zadania nakoľko bude opravované automatizovane pomocou JUnit testov. Na konci zadania je návod, 
ako si lokálne spustiť tzv. "sanity-check" test, pre otestovanie splnenia konfigurácie vypracovania, aby bol projekt testovateľný.

## Špecifikácia

Aplikácia musí publikovať **REST webové služby** dodržiavajúce dodanú [**OpenAPI 3 špecifikáciu**](./src/main/resources/openapi3.spec.yaml). 
Aplikácia tak musí zabezpečiť CRUD operácie cez publikované webové služby (s výnimkou update operácie). Aplikácia musí
taktiež poskytovať webové služby pre priradenie a odovzdanie záverečnej práce, a tak isto pre vyhľadanie záverečnej práce
podľa platných kritérií (t.j. podľa študenta, alebo podľa učiteľa).

Pri implementácií použite protokol **HTTP/1.1**, ako formát prenášaných objektov použite **application/json** s UTF-8 kódovaním. 
Pri chybových stavoch vráťte objekt `Message` (ako je uvedené v špecifikácii) s príslušným kódom chyby. 
Kód odpovede nastavte tak, aby najlepšie vystihla povahu odpovede. 
HTTP kódy: [https://www.restapitutorial.com/httpstatuscodes.html](https://www.restapitutorial.com/httpstatuscodes.html).

### Autentifikácia

V rámci projektu implementujte **autentifikačný mechanizmus Basic Access Authentication** [https://en.wikipedia.org/wiki/Basic_access_authentication](https://en.wikipedia.org/wiki/Basic_access_authentication)
(skrátene Basic Auth). Jednotlivé webové služby sú v špecifikácii označené či je potrebná autentifikácia pre ich dopytovanie. 
Používateľom riešenia, ktorým je umožnené prihlásenie, je každý vytvorený študent a učiteľ.
Ako prihlasovacie meno používateľa použite email učiteľa alebo študenta. Entitu študenta a učiteľa rozšírte o atribút `password`. 
Pri vytvorení používateľa musí byť zadané heslo. Heslo v dopyte vytvorenia používateľa nesmie byť poslané v otvorenej podobe 
ale ako string enkódovaný v BASE64 kódovaní.

Pred uložením do databázy musí byť heslo dekódované do pôvodnej hodnoty vypočítaný hash a následne až výsledný hash reťazec
uložený do databázy ako hodnota `password` atribútu. Pri prihlásení používateľa je porovnané heslo z HTTP hlavičky `Authorization`
s uloženým hash reťazcom z databázy. Autentifikačný mechanizmus musí dokázať rozlíšiť či ide o študenta alebo učiteľa.
Pre vytvorenie hash reťazca hesla a následnú verifikáciu pri prihlásení môžte využiť metódy triedy [BCryptServer](./src/main/java/sk/stuba/fei/uim/vsa/pr2/BCryptService.java).

REST služby implementujte ako bezstavové. Nevytvárajte žiadnu používateľskú reláciu (Session), nevytvárajte žiadne autorizačné tokeny, 
či iné spôsoby relácie a udržiavania stavu používateľa (ako napr. cookies). HTTP hlavička `Authorization` musí byť zaslaná s každým dopytom na služby.

Zlyhanie prihlásenia má vrátiť odpoveď s kódom 401.

### Autorizácia

V rámci projektu implementujte **autorizačný mechanizmus**, ktorý bude **rozlišovať medzi prístupom učiteľa a študenta**.
V [dodanej OpenAPI 3 špecifikácii](./src/main/resources/openapi3.spec.yaml) pri niektorých REST webových službách je popísané,
či má byť implementované obmedzenie pre prístup prihláseného používateľa (napr. pri službe `DELETE /api/student/{id}`). 
Ak nie je zmienené žiadne takéto obmedzenie pri službe, môže ju dopytovať učiteľ aj študent a spracovanie ich dopytu musí
byť rovnaké. Rozlíšenie prístupu určte na základe autentifikovanej entity. 

V prípade nedostatočných oprávnení prihláseného používateľa pre vykonanie dopytu vráťte odpoveď s kódom 403.

## Spustenie

Webovú aplikáciu implementujte pomocou frameworku Jersey (referenčná implementácie JAX-RS štandardu) ako tzv. **'standalone'
webovú aplikáciu**. Výstupom riešenia musí byť spustiteľný JAR súbor. HTTP server aplikácie má počúvať na porte 8080.

Konfiguráciu spustenia HTTP servera je možné nájsť v triede [ApplicationConfiguration](./src/main/java/sk/stuba/fei/uim/vsa/pr2/ApplicationConfiguration.java). 
Jej hodnoty je možné meniť pomocou tzv. 'environment variables'. Východzie hodnoty jednotlivých atribútov nemeňte.

Triedy obsahujúce definície jednotlivých služieb (tzv. 'resource class') anotované anotáciou `@Path` a triedy rozširujúce
funkcionalitu HTTP servera anotované anotáciou `@Provider` sú automaticky skenované a pridané do HTTP servera pri štarte
triedou [JAXRSApplicationConfiguration](./src/main/java/sk/stuba/fei/uim/vsa/pr2/JAXRSApplicationConfiguration.java).
Implementáciu tejto triedy nie je nutné meniť.

Embedovaný HTTP server je nakonfigurovaný a zapnutý v hlavnej triede [Project2Application](./src/main/java/sk/stuba/fei/uim/vsa/pr2/Project2Application.java).
Ak je potrebná inicializácia, alebo je potrebné spustiť ľubovolný kód pri štarte aplikácie (tesne po štarte HTTP servera)
je možné využiť metódu `postStart()` v tejto triede. Implementáciu metód `main` a `startServer` nemeňte.

## Hodnotenie

**Zadanie je hodnotené 15 bodmi. Vypracovanie je nutné odovzdať do 10.05.2023 23:59.**

Zadanie si naklonujte z repozitára zadania. Svoje vypracovanie nahrajte do vášho repozitára pre toto zadanie pomocou
programu Git (git commit + git push). Vypracovanie môžete "pusho-vať" priebežne. Názov Java balíčka nemeňte. 
Nepresúvajte ani nemeňte súbory pripravené v zadaní pokiaľ nie je stanovené inak alebo si to implementácia riešenia
zadania vyslovene vyžaduje (napr. doplnenie tried entít (`<class>` tag) v súbore `persistence.xml`).

**Úpravy pom.xml súbory sú zakázané** mimo ďalej uvedených zmien:

- Pridanie závislosti na driver databázy podľa vlastného výberu. Povolené SQL databázy:
    - MySQL
    - OracleDB
    - Derby
    - PostgreSQL (bude použitá ako testovacia databáza pri oprave)
    - H2
- Doplnenie informácií o autorovi (developerovi, tag `<developers><developer>`)

**Názov `persistence-unit`** v súbore [persistence.xml](src/main/resources/META-INF/persistence.xml) **nemeňte**. Pre
vlastné otestovanie aplikácie môžte implementovať vlastné jUnit testy v priečinku `src/test/java` alebo využiť IntelliJ
HTTP klienta v priečinku `src/test/http`.
Použitá databáza sa musí volať **vsa_pr2** a musí mať vytvoreného používateľa s **menom 'vsa' a heslom 'vsa'**. Pre
tieto účeli môžte využiť súbor [starter.postgres.sql](configs/starter.postgres.sql) (určený pre databázu PostgreSQL).

Implementované webové služby musia konzumovať telá dopytov a produkovať odpovede vo formáte JSON (application/json). 
V rámci projektu je spracovanie JSON formátu už nakonfigurované.

Hodnotiť sa bude iba master/main branch. Kvôli testom a zrýchleniu opravovania je nutné dodržať pokyny a štruktúru
projektu, ako je stanovené v zadaní! Iba kód poslednej verzie vypracovania (t.j. z posledného commit-u) do termínu
odovzdania sa berie do úvahy. Okrem testov sa bude kód a funkcionalita kontrolovať aj manuálne. Hodnotiť sa budú iba
skompilovateľné a spustiteľné riešenia!

### Sanity check test

Projekt obsahuje test tzv. "sanity-check", ktorý kontroluje, či váš projekt dodržuje pokyny zadania a tak či bude 
akceptovaný na opravu. Test je napísaný v triede [SanityCheckTest](src/test/java/sk/stuba/fei/uim/vsa/pr2/SanityCheckTest.java).
Pre spustenie testu stačí vykonať **maven** lifecycle goal **test**, resp. spustiť príkaz:

```shell
mvn test
```

Alebo otvoriť testovací súbor a spustiť jUnit test (zelená šípka vedľa názvu triedy). Ak test skončí úspešne (t.j. bez chyby),
váš projekt je pripravený na opravu.

### Logovanie a výpisy

V rámci projektu je nakonfigurovaná knižnica pre logovanie (formátované výpisy) [Logback](https://www.baeldung.com/logback). Túto knižnicu môžte použiť namiesto
Java `System.out.println` pre formátovaný výpis (napríklad ako je v `SanityCheckTest` triede). Táto knižnica je využitá
ako primárny spôsob výpisu pre pripravené triedy projektu.

Knižnicu je potrebné inicializovať v každej triede kde ju chcete použiť riadkom:

```Java
private static final org.slf4j.Logger log=org.slf4j.LoggerFactory.getLogger(<názov triedy ktorú treba logovať>.class);
```

Následne je možné využiť metódy ako `log.info`, `log.warn` či `log.error` v prípade výnimiek.

## Bonus (3b)

V rámci zadania máte možnosť implementovať rozšírenie funkcionality o tzv. stránkovanie odpovedí služieb za 3 bonusové body.
Pre implementáciu bonusu implementujte [OpenAPI 3 špecifikáciu bonusových webových služieb](./src/main/resources/bonus-openapi3.spec.yaml)
s rozšírením stránkovania. Stránkovanie má pracovať s vašou implementáciou rozhraní Page a Pageable z 1. semestrálneho projektu.
Bonusová špecifikácia zahŕňa novú definíciu webovej služby `GET /api/search/theses`, ktorá nahrádza predchádzajúcu definíciu tejto webovej
služby zo zadania.

Bonusové vypracovanie bude uznané jedine v prípade ak je implementované celé zadanie.
 readmeEtag: '"b8ade4eb277e9ce701ea3dbe537ab7589000b52e"' readmeLastModified: Sat, 27 May 2023 18:47:48 GMT repositoryId: 641824574 description: >- Referenčná implementácia a súbor testov pre hodnotenie 2. semestrálneho zadanie z predmetu B-VSA vyučovanom na FEI STU v letnom semestri akademickom roku 22/23 created: '2023-05-17T08:40:28Z' updated: '2023-05-17T21:50:29Z' language: Java archived: false stars: 0 watchers: 0 forks: 0 owner: Interes-Group logo: https://avatars.githubusercontent.com/u/62511873?v=4 license: Unlicense repoEtag: '"775afa39f79006b38a6d860944a0643636592f980ad73e89d72b9cde1a1fd168"' repoLastModified: Wed, 17 May 2023 21:50:29 GMT category: Low-level Tooling foundInMaster: true - source: openapi3 tags repository: https://github.com/wasedatime/wasedatime.github.io v3: true repositoryMetadata: base64Readme: IyB3YXNlZGF0aW1lLW9wZW5hcGk= readmeEtag: '"8760599935f7cd10bb213b376f86ca797a367548"' readmeLastModified: Wed, 13 Dec 2023 04:49:31 GMT repositoryId: 311374383 description: API documentation website using swagger-ui created: '2020-11-09T15:02:09Z' updated: '2023-07-18T09:13:10Z' language: HTML archived: false stars: 0 watchers: 0 forks: 0 owner: wasedatime logo: https://avatars.githubusercontent.com/u/32608719?v=4 repoEtag: '"4e2be299452293b1a165020a23ac13f262418f6caecb2b6db1e0ac63c693909b"' repoLastModified: Tue, 18 Jul 2023 09:13:10 GMT foundInMaster: true category: - Data Validators - Parsers id: c13102582ceb98d5f8b9d4be146b9fb3 - source: openapi3 tags repository: https://github.com/tufanoruk/enkudo v3: true id: c97e3804452bcda11ec8d879c33f5f84 repositoryMetadata: base64Readme: >- IyBTYW1wbGUgRW5rdWRvIE5CSSBBUEkgV29yawoKVGhpcyBwcm9qZWN0IGluY2x1ZGVzIHNhbXBsZSBFbmt1ZG8gc3Vic2NyaXB0aW9uIEFQSSBkZWZpbml0aW9uIGluIE9wZW5BUEkzLjAgIFlBTUwgZG9jdW1lbnRzLgoKLSAiRW5rdWRvX0RTUF9Db21tb25EYXRhLnlhbWwiIGRlZmluZXMgY29tbW9uIGRhdGEgZGVmaW5pdGlvbnMgd2l0aGluIHRoZSBBUEkgYW5kIHJlZmVyZW5jZWQgZnJvbSBvdGhlciBBUEkgKFlBTUwpIGRvY3VtZW50cy4gWW91IGRvIG5vdCBuZWVkIHRvIHByb3ZpZGUgdGhpcyBkb2N1bWVudCBhbnkgb2YgdmFsaWRhdGlvbiBhbmQvb3IgY29kZSBnZW5lcmF0aW9uIHByb2dyYW1zLgotICJFbmt1ZG9fRFNQX1N1YnNjcmlwdGlvbi55YW1sIiBFbmt1ZG8gU3Vic2NyaXB0aW9uIEFQSSBkZWZpbml0aW9uLgoKVGhlcmUgYXJlIGFsc28gYSBjb3VwbGUgb2YgaGVscGVyIHNjcmlwdHMgdG8gdmFsaWRhdGUgdGhlIFlBTUwgZmlsZXMgd2l0aCByZXNwZWN0IHRvIE9wZW5BUEkgMy4wIHNwZWNpZmljYXRpb24gYW5kIGNvZGUgZ2VuZXJhdGlvbi4KCiJ2YWxpZGF0ZS5zaCIgdmFsaWRhdGVzIGdpdmVuIFlBTUwgZG9jdW1lbnQgdXRpbGl6aW5nIHR3byB2YWxpZGF0b3JzLgoKLSAib3BlbmFwaS1zcGVjLXZhbGlkYXRvciIgYW5kCi0gIm9wZW5hcGktZ2VuZXJhdG9yIHZhbGlkYXRlIiAKCiJnZW5jb2RlLnNoIiBpcyBhIHNpbXBsZSB3cmFwcGVyIHRvIG9wZW5hcGktZ2VuZXJhdG9yIGNvZGUgZ2VuZXJhdGlvbi4KCllvdSBuZWVkIHRvIGluc3RhbGwgdGhlc2UgdHdvIHByb2dyYW1zIGluIHlvdXIgZW52aXJvbm1lbnQuICAKClRvIGluc3RhbGwgb3BlbmFwaS1nZW5lcmF0b3IgKGFuZCBvdGhlciBMaW51eCBsaWtlIHRvb2xzKSAgb24gTWFjT1MgSSBwcmVmZXIgW2JyZXddKGh0dHBzOi8vYnJldy5zaCkuIFRvIGluc3RhbGwgaXQgdmlhIGJyZXcsCgpgYGBiYXNoCiUgYnJldyBpbnN0YWxsIG9wZW5hYXBpLWdlbmVyYXRvcgpgYGAKCiJvcGVuYXBpLXNwZWMtdmFsaWRhdG9yIiBpcyBweXRob24gc2NyaXB0LiBZb3UgY2FuIGluc3RhbGwgaXQgdmlhIHBpcC4KCmBgYGJhc2gKJSBwaXAgaW5zdGFsbCBvcGVuYXBpLXNwZWMtdmFsaWRhdG9yCmBgYAoKVG8gdmFsaWRhdGUgYSBZQU1MIGZpbGUsCgpgYGBiYXNoCi4vdmFsaWRhdGUuc2ggRW5rdWRvX0RTUF9TdWJzY3JpcHRpb24ueWFtbApgYGAKClRvIGdlbmVyYXRlIHNheSBnbyBjbGllbnQgY29kZSBmb3IgRW5rdWRvIFN1YnNjcmlwdGlvbiBOQkksCgpgYGBiYXNoCi4vZ2VuY29kZS5zaCBFbmt1ZG9fRFNQX1N1YnNjcmlwdGlvbi55YW1sIGdvCmBgYAoKVGhpcyBjb21tYW5kIGNyZWFydGVzIGEgImdvX3NyYyIgZGlyZWN0b3J5IHdpdGggY2xpZW50IEFQSSBsaWJyYXJ5IGluIGdvbGFuZy4gVGhlcmUgaXMgYWxzbyBhIHNpbXBsZSBnbyBjbGllbnQgcHJvamVjdCAoZ28tZW5rdWRvLWNsaWVudCkgd2hpY2ggdXRpbGl6ZXMgdGhpcyBvcGVuYXBpLWdlbmVyYXRvciBnZW5lcmF0ZWQgY2xpZW50IEFQSSBsaWJyYXJ5LgoKVG8gZ2V0IHRoZSBsaXN0IG9mIGxhbmd1YWdlcyB0aGF0IG9wZW5hcGktZ2VuZXJhdG9yIGNhbiBnZW5lcmF0ZSBjb2RlIGZvciBjbGllbnQgQVBJIGxpYnJhcnkgYW5kIHNlcnZlciBzdGFiLAoKYGBgYmFzaAolIG9wZW5hcGktZ2VuZXJhdG9yIGxpc3QKYGBgCgpUaGVyZSBhcmUgc2FtcGxlIGdvIHNldmVyIGFuZCBjbGllbnQgcHJvZ3JhbXMgdXRpbGl6aW5nIGdlbmVyYXRlZCBjb2RlIGJ5IG9wZW5hcGktZ2VuZXJhdG9yLgoKLSBnby1zZXJ2ZXJfc3JjIGlzIHRoZSBnZW5lcmF0ZWQgc2VydmVyIGNvZGUgZGlyZWN0b3J5LiBJdCBpbmNsdWRlcyBnby5tb2QgYW5kIG1haW4uZ28gZmlsZXMgd2hpY2ggYXJlIG1vZGlmaWVkIGZyb20gdGhlIGdlbmVyYXRlZCBjb2RlIGFuIHByZXZlbnRlZCBmcm9tIG92ZXJ3cml0dGVuIGR1cmluZyB0aGUgZm9sbG93aW5nIGNvZGUgZ2VuZXJhdGlvbnMuCi0gZ28tZW5rdWRvLWNsaWVudCBpcyBjb250YWlucyBhIHZlcnkgc2ltcGxlIGNsaWVudCBpbXBsZW1lbnRhdGlvbiB0aGF0IHVzZXMgZ29fc3JjIG9wZW5hcGktZ2VuZXJhdG9yIGdlbmVyYXRlZCBjbGllbnQgQVBJIGxpYnJhcnkuCiAKU2FtcGxlIGdvIHNlcnZlciBleGVjdXRlYWJsZSBpcyBnZW5lcmF0ZWQgYXV0b21hdGljYWxseSB1cG9uIHNlcnZlciBjb2RlIGdlbmVyYXRpb24sIGFuZCBjYW4gYmUgZXhlY3V0ZWQuIEl0IHNlcnZlcyBBUElzIGJ1dCByZXR1cm5zICJ1bmltcGxlbWVudGVkIiBmYWlsdXJlLgoKYGBgYmFzaAolIC4vZ2VuY29kZS5zaCBFbmt1ZG9fRFNQX1N1YnNjcmlwdGlvbi55YW1sIGdvLXNlcnZlcgolIGNkIGdvLXNlcnZlcl9zcmMKJSAuL2Vua3Vkby1kc3AtZ28tc2VydmVyCmBgYAoKU2FtcGxlIGdvIGZpbGUgaW4gZ28tZW5rdWRvLWNsaWVudCBleHBsYW5pcyBob3cgdG8gcHJlcGFyZSwgYnVpbGQgYW5kIHJ1biB0aGUgY2xpZW50Lgo= readmeEtag: '"ac3fac8faa165008a71d75743bd02f6140abc6b0"' readmeLastModified: Tue, 26 Jul 2022 07:08:07 GMT repositoryId: 511254274 description: null created: '2022-07-06T18:41:58Z' updated: '2025-04-29T05:32:53Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: tufanoruk logo: https://avatars.githubusercontent.com/u/1410205?v=4 repoEtag: '"fcbcfab3918039cc03e8a2f37142233b6c1ecf153b6f66be0417bfb70f25df65"' repoLastModified: Tue, 29 Apr 2025 05:32:53 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/futurum-dev/futurum.openapi-to-plantuml v3: true id: 146775bbabca2a4943d3d333afedf506 repositoryMetadata: base64Readme: >- IyBGdXR1cnVtLk9wZW5BcGkgdG8gUGxhbnRVbWwKCiFbbGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9mdXR1cnVtLWRldi9mdXR1cnVtLm9wZW5hcGktdG8tcGxhbnR1bWw/c3R5bGU9Zm9yLXRoZS1iYWRnZSkKCkNyZWF0ZXMgW1BsYW50VW1sXShodHRwczovL3BsYW50dW1sLmNvbSkgZGlhZ3JhbXMgZnJvbSBbT3BlbkFwaV0oaHR0cHM6Ly93d3cub3BlbmFwaXMub3JnKSBzcGVjIGZpbGVzLgoKIyMgRXhhbXBsZSAxCltPcGVuQXBpIHNwZWNdKC4vb3BlbmFwaS9vcGVuYXBpLXBldHN0b3JlLmpzb24pIHRha2VuIGZyb20gW2hlcmVdKGh0dHBzOi8vcGV0c3RvcmUuc3dhZ2dlci5pbykKCiMjIyBPcGVuQXBpIGRpYWdyYW0KIVtPcGVuQXBpIGRpYWdyYW1dKC4vZG9jcy9vcGVuYXBpLXBldHN0b3JlLW9wZW5hcGkuc3ZnKQoKIyMjIE9wZW5BcGkgVHlwZSBkaWFncmFtCiFbT3BlbkFwaSBUeXBlIGRpYWdyYW1dKC4vZG9jcy9vcGVuYXBpLXBldHN0b3JlLW9wZW5hcGktdHlwZS5zdmcpCgojIyBFeGFtcGxlIDIKW09wZW5BcGkgc3BlY10oLi9vcGVuYXBpL29wZW5hcGktcGV0c3RvcmUtc2ltcGxlLnlhbWwpCgojIyMgT3BlbkFwaSBkaWFncmFtCiFbT3BlbkFwaSBkaWFncmFtXSguL2RvY3Mvb3BlbmFwaS1wZXRzdG9yZS1zaW1wbGUtb3BlbmFwaS5zdmcpCgojIyMgT3BlbkFwaSBUeXBlIGRpYWdyYW0KIVtPcGVuQXBpIFR5cGUgZGlhZ3JhbV0oLi9kb2NzL29wZW5hcGktcGV0c3RvcmUtc2ltcGxlLW9wZW5hcGktdHlwZS5zdmcpCgojIyBIb3cgdG8gdXNlIGl0CiMjIyBDb25zb2xlClVzZSAqZnV0dXJ1bS5vcGVuYXBpLXRvLXBsYW50dW1sLWNvbnNvbGUqIHByb2plY3QKCiMjIyMgRmlsZQpVc2UgKmZpbGUqIGZvbGxvd2VkIGJ5IGZpbGUgcGF0aApgYGAKZmlsZSAtLXBhdGggIi4uLy4uLy4uLy4uL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS5qc29uIgpgYGAKCiMjIyMgRGlyZWN0b3J5ClVzZSAqZGlyZWN0b3J5KiBmb2xsb3dlZCBieSBkaXJlY3RvcnkgcGF0aApgYGAKZGlyZWN0b3J5IC0tcGF0aCAiLi4vLi4vLi4vLi4vb3BlbmFwaSIKYGBgCgojIyMjIERpYWdyYW0gQ29uZmlndXJhdGlvbgojIyMjIFRoZW1lClVzZSAqLS10aGVtZSogdG8gc3BlY2lmeSB0aGUgUGxhbnRVbWwgdGhlbWUKYGBgCi0tdGhlbWUgImJsdWVwcmludCIKYGBgCipOT1RFIC0gZGVmYXVsdHMgdG8gbm8gdGhlbWUqCgojIyMjIFNob3cgbm90ZXMKVXNlICotLXNob3dub3RlcyogdG8gc3BlY2lmeSBpZiBQbGFudFVtbCBzaG91bGQgc2hvdyBub3RlcwpgYGAKLS1zaG93bm90ZXMgInRydWUiCmBgYAoqTk9URSAtIGRlZmF1bHRzIHRvIG5vIG5vdGVzKgoKIyMjIERvY2tlciAtIFVzZSB3aXRoIG11bHRpcGxlIE9wZW5BcGkgc3BlYyBmaWxlcwoKYGBgCmRvY2tlciBydW4gLS1ybSAtaXQgLXYgJChwd2QpL29wZW5hcGk6L29wZW5hcGkgZnV0dXJ1bS5vcGVuYXBpLXRvLXBsYW50dW1sLWRpcmVjdG9yeQpgYGAKCiMjIyMgRGlhZ3JhbSBDb25maWd1cmF0aW9uCiMjIyMgVGhlbWUKVXNlICotLXRoZW1lKiB0byBzcGVjaWZ5IHRoZSBQbGFudFVtbCB0aGVtZQpgYGAKLS10aGVtZSAiYmx1ZXByaW50IgpgYGAKKk5PVEUgLSBkZWZhdWx0cyB0byBubyB0aGVtZSoKCmUuZy4KYGBgCmRvY2tlciBydW4gLS1ybSAtaXQgLXYgJChwd2QpL29wZW5hcGk6L29wZW5hcGkgZnV0dXJ1bS5vcGVuYXBpLXRvLXBsYW50dW1sLWRpcmVjdG9yeSAtLXRoZW1lICJibHVlcHJpbnQiCmBgYAoKIyMjIyBTaG93IG5vdGVzClVzZSAqLS1zaG93bm90ZXMqIHRvIHNwZWNpZnkgaWYgUGxhbnRVbWwgc2hvdWxkIHNob3cgbm90ZXMKYGBgCi0tc2hvd25vdGVzICJ0cnVlIgpgYGAKKk5PVEUgLSBkZWZhdWx0cyB0byBubyBub3RlcyoKCmUuZy4KYGBgCmRvY2tlciBydW4gLS1ybSAtaXQgLXYgJChwd2QpL29wZW5hcGk6L29wZW5hcGkgZnV0dXJ1bS5vcGVuYXBpLXRvLXBsYW50dW1sLWRpcmVjdG9yeSAtLXNob3dub3RlcyAidHJ1ZSIKYGBgCgojIyMgRG9ja2VyIC0gVXNlIHdpdGggaW5kaXZpZHVhbCBPcGVuQXBpIHNwZWMgZmlsZXMKIyMjIyBPcGVuQXBpIGRpYWdyYW0KYGBgCmNhdCAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS5qc29uIHwgZG9ja2VyIHJ1biAtLXJtIC1hIHN0ZGluIC1hIHN0ZG91dCAtaSBmdXR1cnVtLm9wZW5hcGktdG8tcGxhbnR1bWwtc3RkLWluLW91dCBvcGVuYXBpIHw+IC4vb3BlbmFwaS9vcGVuYXBpLXBldHN0b3JlLW9wZW5hcGkucHVtbApgYGAKYGBgCmNhdCAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS1zaW1wbGUueWFtbCB8IGRvY2tlciBydW4gLS1ybSAtYSBzdGRpbiAtYSBzdGRvdXQgLWkgZnV0dXJ1bS5vcGVuYXBpLXRvLXBsYW50dW1sLXN0ZC1pbi1vdXQgb3BlbmFwaSB8PiAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS1zaW1wbGUtb3BlbmFwaS5wdW1sCmBgYAoKIyMjIyBPcGVuQXBpIFR5cGUgZGlhZ3JhbQpgYGAKY2F0IC4vb3BlbmFwaS9vcGVuYXBpLXBldHN0b3JlLmpzb24gfCBkb2NrZXIgcnVuIC0tcm0gLWEgc3RkaW4gLWEgc3Rkb3V0IC1pIGZ1dHVydW0ub3BlbmFwaS10by1wbGFudHVtbC1zdGQtaW4tb3V0IG9wZW5hcGktdHlwZSB8PiAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS1vcGVuYXBpLXR5cGUucHVtbApgYGAKYGBgCmNhdCAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS1zaW1wbGUueWFtbCB8IGRvY2tlciBydW4gLS1ybSAtYSBzdGRpbiAtYSBzdGRvdXQgLWkgZnV0dXJ1bS5vcGVuYXBpLXRvLXBsYW50dW1sLXN0ZC1pbi1vdXQgb3BlbmFwaS10eXBlIHw+IC4vb3BlbmFwaS9vcGVuYXBpLXBldHN0b3JlLXNpbXBsZS1vcGVuYXBpLXR5cGUucHVtbApgYGAKCiMjIyMgRGlhZ3JhbSBDb25maWd1cmF0aW9uCiMjIyMgVGhlbWUKVXNlICotLXRoZW1lKiB0byBzcGVjaWZ5IHRoZSBQbGFudFVtbCB0aGVtZQpgYGAKLS10aGVtZSAiYmx1ZXByaW50IgpgYGAKKk5PVEUgLSBkZWZhdWx0cyB0byBubyB0aGVtZSoKCmUuZy4KYGBgCmNhdCAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS5qc29uIHwgZG9ja2VyIHJ1biAtLXJtIC1hIHN0ZGluIC1hIHN0ZG91dCAtaSBmdXR1cnVtLm9wZW5hcGktdG8tcGxhbnR1bWwtc3RkLWluLW91dCBvcGVuYXBpIC0tdGhlbWUgImJsdWVwcmludCIgfD4gLi9vcGVuYXBpL29wZW5hcGktcGV0c3RvcmUtb3BlbmFwaS5wdW1sCmBgYAoKIyMjIyBTaG93IG5vdGVzClVzZSAqLS1zaG93bm90ZXMqIHRvIHNwZWNpZnkgaWYgUGxhbnRVbWwgc2hvdWxkIHNob3cgbm90ZXMKYGBgCi0tc2hvd25vdGVzICJ0cnVlIgpgYGAKKk5PVEUgLSBkZWZhdWx0cyB0byBubyBub3RlcyoKCmUuZy4KYGBgCmNhdCAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS5qc29uIHwgZG9ja2VyIHJ1biAtLXJtIC1hIHN0ZGluIC1hIHN0ZG91dCAtaSBmdXR1cnVtLm9wZW5hcGktdG8tcGxhbnR1bWwtc3RkLWluLW91dCBvcGVuYXBpIC0tc2hvd25vdGVzICJ0cnVlIiB8PiAuL29wZW5hcGkvb3BlbmFwaS1wZXRzdG9yZS1vcGVuYXBpLnB1bWwKYGBgCgojIyBEb2NrZXIKIyMjIEhvdyB0byBidWlsZCBpdAoKYGBgCmRvY2tlciBidWlsZCAtdCBmdXR1cnVtLm9wZW5hcGktdG8tcGxhbnR1bWwtZGlyZWN0b3J5IC1mIGZ1dHVydW0ub3BlbmFwaS10by1wbGFudHVtbC1kaXJlY3RvcnkvRG9ja2VyZmlsZSAuCmBgYApgYGAKZG9ja2VyIGJ1aWxkIC10IGZ1dHVydW0ub3BlbmFwaS10by1wbGFudHVtbC1zdGQtaW4tb3V0IC1mIGZ1dHVydW0ub3BlbmFwaS10by1wbGFudHVtbC1zdGQtaW4tb3V0L0RvY2tlcmZpbGUgLgpgYGAKCiMjIyBSZW1vdmUgaW1hZ2VzCmBgYApkb2NrZXIgcm1pIGZ1dHVydW0ub3BlbmFwaS10by1wbGFudHVtbC1kaXJlY3RvcnkKYGBgCmBgYApkb2NrZXIgcm1pIGZ1dHVydW0ub3BlbmFwaS10by1wbGFudHVtbC1zdGQtaW4tb3V0CmBgYA== readmeEtag: '"3ef37dc169d123c91a0a9fbde6ea2c89d35e698b"' readmeLastModified: Sat, 06 Aug 2022 09:35:03 GMT repositoryId: 516064302 description: Creates PlantUml diagrams from OpenApi spec files. created: '2022-07-20T16:54:11Z' updated: '2022-07-22T17:42:05Z' language: C# archived: false stars: 0 watchers: 1 forks: 0 owner: futurum-dev logo: https://avatars.githubusercontent.com/u/97805806?v=4 license: MIT repoEtag: '"ffbeff4b5cdfecd6961ca478b75dc2e6ccc94e3c02e46995e4bb2c1cc7806384"' repoLastModified: Fri, 22 Jul 2022 17:42:05 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/deniskyashif/spring-openapi-codegen-demo v3: true id: 153ea28d4d23780b847d4c0c5340782b repositoryMetadata: base64Readme: >- IyBzcHJpbmctb3BlbmFwaS1jb2RlZ2VuLWRlbW8KClRoaXMgcmVwb3NpdG9yeSBkZW1vbnN0cmF0ZXMgaG93IHRvIGF1dG9tYXRpY2FsbHkgY3JlYXRlIGEgV2ViIEFQSSBzcGVjaWZpY2F0aW9uIGRvY3VtZW50IGZyb20gc291cmNlIGNvZGUgYW5kIGhvdyB0byBnZW5lcmF0ZSBhbiBBUEkgY2xpZW50IGFwcGxpY2F0aW9uIChpbiB2YXJpb3VzIGxhbmd1YWdlcykgZnJvbSBpdC4gCgojIyBHZW5lcmF0ZSBPcGVuQXBpIHNwZWMgdXNpbmcgY29kZSBhbm5vdGF0aW9ucwoKVGhpcyBwYXJ0IHJlcXVpcmVzIFtKYXZhIDhdKGh0dHBzOi8vd3d3Lm9yYWNsZS5jb20vamF2YS90ZWNobm9sb2dpZXMvZG93bmxvYWRzLykgYW5kIFtHcmFkbGVdKGh0dHBzOi8vZ3JhZGxlLm9yZy8pLiBbVGhlIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiBFeHBsYWluZWRdKGh0dHBzOi8vb2FpLmdpdGh1Yi5pby9Eb2N1bWVudGF0aW9uL3NwZWNpZmljYXRpb24uaHRtbCkKCjEuIEFkZCBgJ29yZy5zcHJpbmdkb2M6c3ByaW5nZG9jLW9wZW5hcGktdWk6MS42LjknYCB0byBnZW5lcmF0ZSBPcGVuQVBJIFVJIGluIHRoZSBicm93c2VyCjIuIEFkZCBgJ2lvLnN3YWdnZXIuY29yZS52Mzpzd2FnZ2VyLWNvcmU6Mi4yLjInYCB0byBpbmNsdWRlIGNvZGUgYW5ub3RhdGlvbnMgZm9yIGRvY3VtZW50aW5nIEFQSSBlbmRwb2ludHMKMy4gRG9jdW1lbnQgdGhlIEFQSSBlbmRwb2ludHMgdXNpbmcgY29kZSBhbm5vdGF0aW9ucyAoc2VlIFtHcmVldGluZ0NvbnRyb2xsZXIuamF2YV0oaHR0cHM6Ly9naXRodWIuY29tL2Rlbmlza3lhc2hpZi9zcHJpbmctb3BlbmFwaS1jb2RlZ2VuLWRlbW8vYmxvYi9tYXN0ZXIvc3JjL21haW4vamF2YS9jb20vZXhhbXBsZS9yZXN0c2VydmljZS9HcmVldGluZ0NvbnRyb2xsZXIuamF2YSkpLgo0LiBSdW4gdGhlIHdlYiBhcHAgYGdyYWRsZSBib290UnVuYCBhbmQgbmF2aWdhdGUgdG8gYGxvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkvaW5kZXguaHRtbGAgdG8gc2VlIHRoZSBpbnRlcmFjdGl2ZSBkb2N1bWVudGF0aW9uLgo1LiBBY2Nlc3MgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBKU09OIGRvY3VtZW50IG9uIGBodHRwOi8vbG9jYWxob3N0OjgwODAvdjMvYXBpLWRvY3NgCgpUbyBnZW5lcmF0ZSB0aGUgT3BlbkFQSSBzcGVjIGFzIGEgZmlsZSwgYWRkIHRoZSBbc3ByaW5nZG9jLW9wZW5hcGktZ3JhZGxlLXBsdWdpbl0oaHR0cHM6Ly9naXRodWIuY29tL3NwcmluZ2RvYy9zcHJpbmdkb2Mtb3BlbmFwaS1ncmFkbGUtcGx1Z2luKSBhbmQgcnVuOiAgCgpgZ3JhZGxlIGNsZWFuIGdlbmVyYXRlT3BlbkFwaURvY3NgIAoKVGhlIGZpbGUgaXMgbG9jYXRlZCBpbiBgPHBhdGgtdG8tcHJvamVjdD4vYnVpbGQvb3BlbmFwaS5qc29uYC4gTWFrZSBzdXJlIG5vIG90aGVyIHByb2Nlc3MgaXMgcnVubmluZyBvbiBwb3J0IGA6ODA4MGAuIAoKIyMgR2VuZXJhdGUgYW4gQVBJIGNsaWVudCBmcm9tIGFuIE9wZW5BUEkgc3BlYwoKVGhlIE9wZW5BUEkgR2VuZXJhdG9yIHJlcXVpcmVzIFtub2RlLmpzXShodHRwczovL25vZGVqcy5vcmcvZW4vKS4gU2VlIHRoZSBpbnN0cnVjdGlvbnMgZm9yIFtpbnN0YWxsYXRpb25dKGh0dHBzOi8vb3BlbmFwaS1nZW5lcmF0b3IudGVjaC9kb2NzL2luc3RhbGxhdGlvbikgYW5kIFt1c2FnZV0oaHR0cHM6Ly9vcGVuYXBpLWdlbmVyYXRvci50ZWNoL2RvY3MvdXNhZ2UpLgoKTGlzdCB0aGUgYXZhaWxhYmxlIGdlbmVyYXRvcnM6ICAKYG9wZW5hcGktZ2VuZXJhdG9yLWNsaSBsaXN0YAoKVmFsaWRhdGUgdGhlIE9wZW5BcGkgc3BlYyBmaWxlOiAgCmBvcGVuYXBpLWdlbmVyYXRvci1jbGkgdmFsaWRhdGUgLWkgPHBhdGgtdG8+L29wZW5hcGkuanNvbmAKCkdlbmVyYXRlIGEgQyMgQVBJIENsaWVudDogIApgb3BlbmFwaS1nZW5lcmF0b3ItY2xpIGdlbmVyYXRlIC1pIDxwYXRoLXRvPi9vcGVuYXBpLmpzb24gLWcgY3NoYXJwLW5ldGNvcmUgLW8gLi90bXAvQ2xpZW50QXBwL2AK readmeEtag: '"4bbbbb338d2fe68b62dd3f9db9850deac4b6fa5e"' readmeLastModified: Tue, 09 Aug 2022 05:33:15 GMT repositoryId: 522176786 description: >- How to automatically create a Web API specification document from source code and how to generate an API client application (in various languages) from it created: '2022-08-07T10:06:25Z' updated: '2022-08-08T09:15:30Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: deniskyashif logo: https://avatars.githubusercontent.com/u/5999271?v=4 repoEtag: '"6ecf3586821b538e77017b4663ea5659947fdf3d1af99bd668182a3fc209e0f6"' repoLastModified: Mon, 08 Aug 2022 09:15:30 GMT category: - Code Generators - Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/davids13/api-specification-converter v3: true repositoryMetadata: base64Readme: >- IyBvcGVuQVBJQ29udmVydGVyCkEgY29udmVydGVyIEFQSTogU3dhZ2dlciB0byBPcGVuQXBpIDMgb3IgT3BlbkFwaSAzIHRvIFN3YWdnZXIK readmeEtag: '"9f74789c588de42be5f4318d12d93a3e38223df4"' readmeLastModified: Sat, 29 Feb 2020 13:47:40 GMT repositoryId: 243976048 description: 'A converter for API specification: OAS3, swagger, raml' created: '2020-02-29T13:43:38Z' updated: '2020-07-16T22:09:46Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: davids13 logo: https://avatars.githubusercontent.com/u/19639609?v=4 repoEtag: '"492159f983d9f48bf4348730c2c91854a9ad04dfc977c4f1d68c0d98c3ac7f2b"' repoLastModified: Thu, 16 Jul 2020 22:09:46 GMT foundInMaster: true category: Parsers id: 225a855be04f65490701933a0597917c - source: openapi3 tags repository: https://github.com/soxyl/saferpay-openapi v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFNwZWMgZm9yIFNhZmVycGF5IEpTT04gQVBJCgpUaGlzIHJlcG9zaXRvcnkgcHJvdmlkZXMgYW4gT3BlbkFQSSAzLjAgU3BlY2lmaWNhdGlvbiBGaWxlIGRlc2NyaWJpbmcgdGhlIFNhZmVyZXBheSBKU09OIEFQSSBkb2N1bWVudGVkIGF0IGh0dHBzOi8vc2FmZXJwYXkuZ2l0aHViLmlvL2pzb25hcGkgLgoKVGhlIE9wZW5BUEkgc3BlYyBpcyBob3N0ZWQgYXQgaHR0cHM6Ly9hcHAuc3dhZ2dlcmh1Yi5jb20vYXBpcy9zb3h5bC9zYWZlcnBheQoKIyMgUmVnZW5lcmF0ZSBzcGVjCgpgYGAKJCBjb21wb3NlciBpbnN0YWxsIC0td29ya2luZy1kaXIgZ2VuZXJhdG9yCiQgZ2VuZXJhdG9yL2Jpbi9jb25zb2xlIGdlbmVyYXRlID4gb3BlbmFwaS55YW1sCmBgYAoKIyMgR2VuZXJhdGUgYSBDbGllbnQgTG9jYWxseQoKYGBgCmRvY2tlciBydW4gLS1ybSAtLXVzZXIgMTAwMDoxMDAwIC12ICR7UFdEfTovbG9jYWwgb3BlbmFwaXRvb2xzL29wZW5hcGktZ2VuZXJhdG9yLWNsaSBnZW5lcmF0ZSBcCiAgICAtaSAvbG9jYWwvb3BlbmFwaS55YW1sIFwKICAgIC1nIHBocCBcCiAgICAtbyAvbG9jYWwvY2xpZW50LXBocApgYGAK readmeEtag: '"d28f9e28d646e975c1e3ba821ce48bf3f183e8b3"' readmeLastModified: Tue, 14 Aug 2018 10:03:28 GMT repositoryId: 144590172 description: >- This repository provides an OpenAPI 3.0 Specification File describing the Saferepay JSON API. created: '2018-08-13T14:20:57Z' updated: '2018-08-14T10:07:07Z' language: PHP archived: false stars: 0 watchers: 0 forks: 2 owner: soxyl logo: https://avatars.githubusercontent.com/u/7266755?v=4 license: MIT repoEtag: '"aa810bc4321aa72d0cb0970922bb3c670d8ca5df36e603201c49f84e114223e2"' repoLastModified: Tue, 14 Aug 2018 10:07:07 GMT foundInMaster: true category: - Code Generators - Parsers id: 81db3bb9a1951b971b0811a986670f18 - source: openapi3 tags repository: https://github.com/zweidenker/dockerengine v3: true repositoryMetadata: base64Readme: >- IyBEb2NrZXJFbmdpbmUKClRoaXMgaXMgaW1wbGVtZW50ZWQgdXNpbmcgRG9ja2VyIEVuZ2luZSBBUEkgeWFtbCBmaWxlIGZyb20KCmh0dHBzOi8vZG9jcy5kb2NrZXIuY29tL2VuZ2luZS9hcGkvdjEuMzcvIwoKY29udmVydGVkIHRoZSBmaWxlIHRvIGpzb24gd2l0aAoKJCBhbGlhcyB5YW1sLXRvLWpzb249J2RvY2tlciBydW4gLWkgaW5neS95YW1sLXRvLWpzb24nCiQgY2F0IHN3YWdnZXIueWFtbCB8IHlhbWwtdG8tanNvbiB8IGpxIC4gPiBzd2FnZ2VyLmpzb24KCmFuZCBjb252ZXJ0ZWQgaXQgZnJvbSBzd2FnZ2VyIDIuMCB0byBvcGVuYXBpIDMuMC4wIHdpdGgKCmh0dHBzOi8vb3BlbmFwaS1jb252ZXJ0ZXIuaGVyb2t1YXBwLmNvbS8KCgoK readmeEtag: '"5f9cf774599a4b0073c78da91b517cc30489b7b3"' readmeLastModified: Thu, 13 Sep 2018 12:39:27 GMT repositoryId: 140084992 description: Implementation of DockerEngine API created: '2018-07-07T13:04:59Z' updated: '2025-08-11T21:05:35Z' language: Smalltalk archived: false stars: 1 watchers: 2 forks: 1 owner: zweidenker logo: https://avatars.githubusercontent.com/u/20300491?v=4 repoEtag: '"de5bb93cfa43229df5bc03cab218e348c52d0f9e06fcdfd6d76c1a9b27329f5d"' repoLastModified: Mon, 11 Aug 2025 21:05:35 GMT foundInMaster: true category: - Code Generators - Parsers id: e39bc1e71854fd595a53e89dcbc84c4a - source: openapi3 tags repository: https://github.com/sinnerr0/mock-server-generator v3: true repositoryMetadata: base64Readme: >- IyBNb2NrIFNlcnZlciBHZW5lcmF0b3IKCiMjIE1vdGl2YXRpb24KCk9wZW4gQVBJ66W8IOyCrOyaqe2VmOuKlCDqsoPsnbQg7JWE64uI652866m0IOyEnOuyhOyqveqzvCDtgbTrnbzsnbTslrjtirjsqr3snbQg64+Z7Iuc7JeQIOqwnOuwnOydtCDsp4Ttlokg65Cc64ukLiDsnbTrn7Ag6rK97JqwIO2BtOudvOydtOyWuO2KuOyqveyXkCDrtoDri7TsnbQg7Luk7KeA64qU642wIEFQSSDsiqTtjpnsnYQg66+466asIO2ZleyduO2WiOuLpOqzoCDtlZjrjZTrnbzrj4Qg7ISc67KE6rCAIOyVhOyngSDsl4bsnLzrr4DroZwo7J6I642U652864+EIOydvOu2gOunjCkg7YG065287J207Ja47Yq47Kq9IOyekeyXheydtCDsp4Dsl7DrkJjqsbDrgpggTW9ja2luZ+ydhCDtlbTshJwg67CY7ZmY6rCS7J2EIOqwgOygle2VmOyXkCDrkZDqs6Ag6rCc67Cc7J2EIO2VtOyVvCDtlZzri6TripQg7KCQ7J2064ukLiDshJzrsoTsqr3sl5DshJwg7ZiR7J2Y65CcIEFQSSDsiqTtjpnsl5Ag66ee7LaU7Ja0IOyEnOuyhOyXkOyEnCBNb2Nr7ISc67KE66W8IOq1rO2YhO2VtCDspIDri6TrqbQg7IOB7Zmp7J20IOyigCDrgpjslYTsp4jthYzsp4Drp4wg6rCc67Cc7ZWY6riw64+EIOuwlOyBnOuNsCDsnbTrn7Dqsbgg6rWs7ZiE7ZWY64qUIOqygyDsnpDssrTqsIAg7Im97KeAIOyViuuLpC4KCiMjIFN1bW1hcnkKCuyEnOuyhOyZgOuKlCDqtIDqs4Tsl4bsnbQg7KCV7J2Y65CcIOyduO2EsO2OmOydtOyKpOyXkCDrlLDrnbwg66+466asIOqwnOuwnOydhCDsp4TtlontlaAg7IiYIOyeiOuPhOuhnSBNb2NrIFNlcnZlcuulvCDsg53shLHtlZjqs6Ag7J6Q64+Z7Jy866GcIOyDneyEseuQnCDsnZHri7XqsJLsnYQg64+M66Ck7KSA64ukLgoKIyMgQmFzZSBrbm93bGVkZ2UKCk9wZW4gQVBJIFNwZWMgMy4wLjMgaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vCgojIyBPcGVuIEFQSSBTcGVjIEZpbGUKCi0gb3BlbmFwaS55YW1sCgotIFByaXNtIG1vY2s6IER5bmFtaWMgUmVzcG9uc2UgR2VuZXJhdGlvbgoKICBodHRwczovL21ldGEuc3RvcGxpZ2h0LmlvL2RvY3MvcHJpc20vZG9jcy9ndWlkZXMvMDEtbW9ja2luZy5tZCNkeW5hbWljLXJlc3BvbnNlLWdlbmVyYXRpb24KCiMjIENyZWF0ZS9FZGl0IGZpbGUKCmh0dHBzOi8vZWRpdG9yLnN3YWdnZXIuaW8vCgojIyBJbnN0YWxsCgpgYGAKJCBucG0gaW5zdGFsbApgYGAKCiMjIFNwaW4gdXAgYSBtb2NrIEhUVFAgc2VydmVyCgpgYGAKJCBucG0gc3RhcnQKYGBgCgojIyBQcm94eSBSZWFsIFNlcnZlcgoKYGBgCiMgTW9kaWZ5IHJlYWwgc2VydmVyIGFkZHJlc3MgaW4gcGFja2FnZS5qc29uCiQgbnBtIHJ1biBwcm94eQpgYGAKCiMjIERvY3MgVUkKCmBgYAojIHNlcnZlIHBvcnQgNDAwMDogaHR0cDovL2xvY2FsaG9zdDo0MDAwCiQgbnBtIHJ1biBkb2NzCmBgYAo= readmeEtag: '"9a65a43fda4ddeddb9b1f8af13d01b1eb0814a38"' readmeLastModified: Fri, 28 Aug 2020 11:26:30 GMT repositoryId: 289464781 description: mock server generator & docs serve created: '2020-08-22T10:12:57Z' updated: '2022-12-09T14:36:47Z' language: HTML archived: true stars: 0 watchers: 1 forks: 0 owner: sinnerr0 logo: https://avatars.githubusercontent.com/u/6859131?v=4 repoEtag: '"2bb794c59686a51a0ad42b19b8f3b723cd564ef7b1770f2c9fe129a2fc838170"' repoLastModified: Fri, 09 Dec 2022 14:36:47 GMT foundInMaster: true category: - SDK - Server Implementations id: 13a5f59fc7c008ee062cd1ea9cf5aa53 - source: openapi3 tags repository: https://github.com/mikeralphson/openapi-diff v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIERpZmYKPiBBIENMSSB0b29sIHRvIGlkZW50aWZ5IGRpZmZlcmVuY2VzIGJldHdlZW4gU3dhZ2dlci9PcGVuQVBJIHNwZWNzLgoKIyMgUmVxdWlyZW1lbnRzCi0gbm9kZWpzIDYueCBvciBoaWdoZXIgKHRlc3RlZCB1c2luZyA2LngsIDgueCBhbmQgMTAueCkKLSBucG0gMy54IG9yIGhpZ2hlciAodGVzdGVkIHVzaW5nIDMueCBhbmQgNXgpCgojIyBJbnN0YWxsYXRpb24KCkluc3RhbGwgdGhlIHRvb2wgdXNpbmcgbnBtIGFuZCBhZGQgaXQgdG8gdGhlIHBhY2thZ2UuanNvbgpgYGAKbnBtIGluc3RhbGwgb3BlbmFwaS1kaWZmIC0tc2F2ZS1kZXYKYGBgCgpBdm9pZCBpbnN0YWxsaW5nIHRoZSB0b29sIGdsb2JhbGx5IGFzIHRoaXMgd2lsbCBsZWFkIHRvIHByb2JsZW1zIHdoZW4gbXVsdGlwbGUgY29kZWJhc2VzIHRyeSB0byB1c2UgZGlmZmVyZW50IHZlcnNpb25zCm9mIHRoZSB0b29sIG9uIHRoZSBzYW1lIG1hY2hpbmUuCgojIyBVc2FnZQpJbnZva2UgdGhlIHRvb2wgd2l0aCB0d28gcGF0aHMgdG8gU3dhZ2dlci9PcGVuQVBJIGZpbGVzIGluIG9yZGVyIHRvIGZpbmQgZGlmZmVyZW5jZXMgYmV0d2VlbiB0aGVtLCB0aGVzZSBwYXRocyBjYW4KZWl0aGVyIGJlIHBhdGhzIHRvIHRoZSBzcGVjcyBpbiB0aGUgbG9jYWwgZmlsZXN5c3RlbSBvciBVUkxzIHRvIHRoZSBzcGVjcywgYm90aCBKU09OIGFuZCBZQU1MIGFyZSBzdXBwb3J0ZWQuCmBgYAouL25vZGVfbW9kdWxlcy8uYmluL29wZW5hcGktZGlmZiAvcGF0aC90by9zb3VyY2Uvb3BlbmFwaS5qc29uIC9wYXRoL3RvL2Rlc3RpbmF0aW9uL29wZW5hcGkuanNvbgouL25vZGVfbW9kdWxlcy8uYmluL29wZW5hcGktZGlmZiAvcGF0aC90by9zb3VyY2Uvb3BlbmFwaS55bWwgL3BhdGgvdG8vZGVzdGluYXRpb24vb3BlbmFwaS55bWwKYGBgCgpUaGUgdG9vbCdzIG91dHB1dCB3aWxsIGRpc3BsYXkgdGhlIGFtb3VudCBhbmQgdHlwZSBvZiBjaGFuZ2VzLCBhbmQgdGhlbiBsaXN0IHRoZSBjaGFuZ2VzIHdpdGggdGhlIHJlbGV2YW50IGluZm8uCkNoYW5nZXMgYXJlIGNsYXNzaWZpZWQgYXMgZm9sbG93czoKCiogQnJlYWtpbmc6IGNoYW5nZXMgdGhhdCB3b3VsZCBtYWtlIGV4aXN0aW5nIGNvbnN1bWVycyBpbmNvbXBhdGlibGUgd2l0aCB0aGUgQVBJIChkZWxldGlvbiBvZiBwYXRocywgYWRkaW5nIHJlcXVpcmVkCnByb3BlcnRpZXMuLi4pCiogTm9uLWJyZWFraW5nOiBjaGFuZ2VzIHRoYXQgd291bGQgKipub3QqKiBtYWtlIGV4aXN0aW5nIGNvbnN1bWVycyBpbmNvbXBhdGlibGUgd2l0aCB0aGUgQVBJIChhZGRpdGlvbiBvZiBwYXRocywKdHVybmluZyBhIHJlcXVpcmVkIHByb3BlcnR5IGludG8gb3B0aW9uYWwuLi4pCiogVW5jbGFzc2lmaWVkOiBjaGFuZ2VzIHRoYXQgaGF2ZSBiZWVuIGRldGVjdGVkIGJ5IHRoZSB0b29sIGJ1dCBjYW4ndCBiZSBjbGFzc2lmaWVkIChtb2RpZmljYXRpb25zIHRvIFgtUHJvcGVydGllcyBhbmQKb3RoZXIgdW5mb3Jlc2VlbiBjaGFuZ2VzKQoKVGhlIGNvbW1hbmQgd2lsbCBleGl0IHdpdGggYW4gZXhpdCBjb2RlIDEgaWYgYW55IGJyZWFraW5nIGNoYW5nZXMgd2VyZSBmb3VuZCwgc28gdGhhdCB5b3UgY2FuIGZhaWwgYnVpbGRzIGluIENJIHdoZW4KdGhpcyBoYXBwZW5zLgoKIyMgRmVhdHVyZSBzdXBwb3J0ClNlZSBbU1BFQ19TVVBQT1JULm1kXShTUEVDX1NVUFBPUlQubWQpCg== readmeEtag: '"65efaea40e50866d1e2e96f7b078b711b27e8cf3"' readmeLastModified: Tue, 13 Nov 2018 02:34:28 GMT repositoryId: 158676563 description: Mirror of Atlassian's OpenAPI Diff (from bitbucket.org) created: '2018-11-22T09:45:42Z' updated: '2018-11-22T09:53:27Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: MikeRalphson logo: https://avatars.githubusercontent.com/u/21603?v=4 license: NOASSERTION repoEtag: '"3a0b52cadf91419e75b1e939b9a2327cb1b04b5ca017d20e1395e1f26f360b53"' repoLastModified: Thu, 22 Nov 2018 09:53:27 GMT foundInMaster: true category: Parsers id: eb4fff87cf20a81df48de3a6428af5e3 - source: openapi3 tags repository: https://github.com/yoshinariyamanaka/openapi-with-express v3: true id: ec07d6e09dd12e6df639f9f2968f3b88 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHdpdGggRXhwcmVzcyB3cml0dGVuIGluIFR5cGVTY3JpcHQKCk1haW4gbGlicmFyaWVzIGZvciB0aGlzIHJlcG9zaXRvcnkKCiMgQVBJIFNlcnZlcgoKLSAgIFtleHByZXNzXShodHRwOi8vZXhwcmVzc2pzLmNvbS8pCgojIE9wZW5BUEkgVmFsaWRhdGlvbiBmb3IgYm90aCByZXF1ZXN0IGFuZCByZXNwb25zZQoKLSAgIFtleHByZXNzLW9wZW5hcGktdmFsaWRhdG9yXShodHRwczovL2dpdGh1Yi5jb20vY2RpbWFzY2lvL2V4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IjcmVhZG1lKQoKIyBHZW5lcmF0ZSBUeXBlU2NyaXB0IHR5cGVzIGZyb20gT3BlbkFQSSBzcGVjaWZpY2F0aW9uCgotICAgW29wZW5hcGktdHlwZXNjcmlwdF0oaHR0cHM6Ly9naXRodWIuY29tL2Ryd3Bvdy9vcGVuYXBpLXR5cGVzY3JpcHQjcmVhZG1lKQoKIyBVSSB0byBzZWUgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbgoKLSAgIFtzd2FnZ2VyLXVpLWV4cHJlc3NdKGh0dHBzOi8vZ2l0aHViLmNvbS9zY290dGllMTk4NC9zd2FnZ2VyLXVpLWV4cHJlc3MpCg== readmeEtag: '"1aa82d583ae65e15cbca5df6ecbf09e37b3c5e9c"' readmeLastModified: Tue, 11 Apr 2023 21:00:51 GMT repositoryId: 626168947 description: OpenAPI-with-Express implemented in TypeScript created: '2023-04-11T00:02:41Z' updated: '2023-04-11T00:18:07Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: YoshinariYamanaka logo: https://avatars.githubusercontent.com/u/59109232?v=4 repoEtag: '"316a548fb7b8984960d5ea02cd6efcd59db6a07081b5f5a275eefe08019046ac"' repoLastModified: Tue, 11 Apr 2023 00:18:07 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/yo-mah-ya/openapi-with-express - source: openapi3 tags repository: https://github.com/dansilva41/currency-transaction v3: true repositoryMetadata: base64Readme: >- <img align="center" src="docs/logo.png" alt="Currency Transaction">
    
<h4 align="center">
    <a href="https://currency-transaction.herokuapp.com/swagger-ui.html">Live demo</a>
</h4>

<h3 align="center">
    Simple, fast and reactive application to convert values between all currencies.
</h3>

<p align="center">
  <img alt="GitHub language count" src="https://img.shields.io/github/languages/count/dansilva41/currency-transaction?color=%2304D361">

  <img alt="Repository size" src="https://img.shields.io/github/repo-size/dansilva41/currency-transaction">

  <a href="https://github.com/dansilva41/currency-transaction/commits/main">
    <img alt="GitHub last commit" src="https://img.shields.io/github/last-commit/dansilva41/currency-transaction">
  </a>

   <img alt="License" src="https://img.shields.io/badge/license-MIT-brightgreen">
   <a href="https://github.com/dansilva41/currency-transaction/stargazers">
    <img alt="MIT License" src="https://img.shields.io/github/stars/dansilva41/currency-transaction?style=social">
  </a>

  <a href="https://dansilva41.github.io">
    <img alt="made by Danilo Silva" src="https://img.shields.io/badge/made%20by-Danilo%20Silva-blue">
  </a>

  <a href="https://developers-friends.gitbook.io/">
    <img alt="Developer's Friends" src="https://img.shields.io/badge/Blog-Developers%20Friends-orange">
    </a> 
</p>

<p align="center">
 <a href="#dizzy-about">About</a> •
 <a href="#mega-features">Features</a> • 
 <a href="#rocket-technologies-and-motivation">Technologies and motivation</a> •
 <a href="#scroll-structure">Structure</a> •
 <a href="#computer-building-locally">Building Locally</a> •
 <a href="#memo-license">License</a>

</p>

## :dizzy: About

This application is designed with currency-based value conversion in mind, where the focus is to get the conversion rates from an [external api](https://exchangeratesapi.io),
perform the conversion calculations, persist the data and return to consistent data.
In addition to offering performance due to its reactive behavior, the application provides a way to consult the conversions already carried out by the user.

---
## :mega: Features

|          Actions                                  |     Available       |
| --------------------------                        | :-----------------: |
| Converting value between all currencies           |         ✔️           |
| Get all conversions performed by user             |         ✔️           |

---

## :rocket: Technologies and motivation

The project was developed using the following technologies

- [Spring](): It makes programming in Java faster, easier and safer. Compared to other web ecosystems such as Quarkus, Vert.x, Javalin, it has a very simple configuration, with a large and present community, which makes it easier in the midst of development mishaps.
- [Spring WebFlux](): In order to build an application aimed at the REST API, but to bring reactivity, asynchronism and performance, Spring Web Flux is able to provide this, keeping the code common to the conventional synchronous Rest API.
- [Reactive MongoDB Embedded](): Provide a platform-neutral way to run embedded mongodb, not relying on an external server for the application and with a very fast build, dynamic by instance and powerful in processing.
- [Open API](): Pursuing the API first concept, this specification enables independent writing of the REST APIs and the result is a complete documentation of request and response endpoints and objects
- [Wiremock](): In order to perform integration tests in the application validation (check), Wiremock in an embedded way and with dynamic routes, makes it possible to test the flows where there is external communication and to validate error treatments, response bodies and logic.
- [Lombok](): Lombok is used to reduce boilerplate code for model/data objects, e.g., it can generate getters and setters for those object automatically by using Lombok annotations.
- [GitHub Actions](): As CI / CD process GitHub Actions provides a simple, powerful, integrated with various cloud providers and with the processing power and full feedback, therefore, it fulfills the final release of this product.
- [Heroku](): Because it is flexible and with the possibility of deployment either via console or as part of the CI/CD deployment process from GitHub Actions, it is able to publish a new version of the application for free

---

## :scroll: Structure

```text
.
├── code-quality-enforcement                                Directory that stores files related to code quality
├── docs                                                    README.md documentation related files
├── logs                                                    Folder that stores application log records
├── src
│   ├── main
│   │   ├── java
│   │    │    └── tech.jaya.currencytransaction             Application main package
│   │    │        ├── configuration                         Layer where the necessary instances are configured via Dependency Injection   
│   │    │        ├── core                                  Application domain layer. Logic, dependent objects and business rules are found here.
│   │    │        ├── dataprovider                          Implementation layer in data search and return, where there is a specific implementation of each external provider
│   │    │        ├── entrypoint                            Presentation layer and exposure of asynchronous resources, as well as handling requests and errors
│   │    │        ├── CurrencyTransactionApiApplication     Main class
│   │    └── resources
│   │        ├── i18n                                       Directory reserved for storing files differentiated by locale in messages interpolation
│   │        │   └── messages.properties                    Default messages
│   │        ├── static                                     
│   │        │   └── open-api.yaml                          Descriptive file of the application's API specification
│   │        ├── application.properties                     File that stores basic application information
│   │        ├── application.yaml                           Application custom settings definition file
│   │        └── logback.xml                                Configuration file for launching and handling logs
│   └── test
│       ├── java/tech/jaya/currencytransaction              Package where unit testing and integration testing implementations are concentrated
│       └── resources
│           ├── mock                                        Directory reserved for storing JSON files used by Wiremock fake requests
│           └── application-test.xml                        Application custom settings definition file in the context of tests
├── target
│   └──currency-transaction-api-0.0.1.jar                   Generated jar
├── pom.xml                                                 Fundamental unit of work in Maven for defining, resolving dependencies, and building the project
├── LICENSE.md                                              Application license definition
└── README.md                                               Application Documentary File
```

## :computer: Building Locally

### Requirements

- [Maven]()
- [Java 11]()

### Install

1. Clone the repository

   `git clone git@github.com:DanSilva41/currency-transaction.git`

2. Enter directory
   
    `cd currency-transaction`

3. Install dependencies
   
    `mvn clean install`

### Run application

    `mvn springboot:run`

Open http://localhost:8080/swagger-ui.html

---

### Additional

#### Checkstyle

<details>
    <summary>01. Checking the project via maven plugin</summary>

- To run the check in the project, just use this maven command in shell/console or run via IDEA.

```bash
    mvn checkstyle:check
```
- To generate a report from the analysis of code style violations, just use this maven command in shell/console or run via IDEA.

```bash
    mvn checkstyle:checkstyle
```
Report generated in **target/site/checkstyle.html**.

</details>

<details>
    <summary>02. Installing the CheckStyle-IDEA plugin on IntelliJ</summary>

We can use the CheckStyle-IDEA plugin to help formatting code in the IDE.
To configure it is very simple, first install the plugin via the link above or on IntelliJ at
**File > Settings > Plugins**.

![Install plugin Checkstyle IDEA](docs/checkstyle/install-plugin-checkstyle.png)

### 02. Configuring the CheckStyle-IDEA plugin

- After installation, we need to import the settings defined in the checkstyle.xml file into the CheckStyle-IDEA plugin.
  Navigate to **File > Settings > Tools > Checkstyle** and in **Configuration File** click **Add** (+ sign on the right), indicate the path of your checkstyle.xml and click next.

![Configure the Checkstyle IDEA - First](docs/checkstyle/configure-checkstyle-intellij-first.png)

- With the file imported, don't forget to leave it selected as **Active**.

![Configure the Checkstyle IDEA - Second](docs/checkstyle/configure-checkstyle-intellij-second.png)

- Now let's add the same checkstyle file to the IntelliJ settings itself, so when we use the default formatting shortcuts it will automatically look for Checkstyle Main.
  Within settings, go to **Editor > Code Style > Java** and import the file as shown in the image below:

![Set code style look checkstyle](docs/checkstyle/set-code-style-look-checkstyle.png)

- Once these settings are finished, the CheckStyle option will appear at the bottom of IntelliJ and when clicking, the screen below will appear.
  At this point, in Rules select the one you imported in the previous steps and run the verification.

- In Intellij IDEA, select the project, package(s) or class(es) and **Right click > Analyse > Inspect Code... > OK** then plugin will indicate the problems found.

</details>

---

## :memo: License
This project is under the MIT license. See the [LICENSE](https://github.com/dansilva41/currency-transaction/blob/main/LICENSE) for more information.

Developed by Danilo Silva :wave: [Get in touch!](https://www.linkedin.com/in/danilosilvap/)

[Spring]: https://spring.io/
[Spring WebFlux]: https://docs.spring.io/spring-framework/docs/current/reference/html/web-reactive.html
[Reactive MongoDB Embedded]: https://github.com/flapdoodle-oss/de.flapdoodle.embed.mongo
[Project Reactor]: https://projectreactor.io/
[Open API]: https://swagger.io/specification/
[Wiremock]: http://wiremock.org/
[Lombok]: https://projectlombok.org/
[GitHub Actions]: https://github.com/features/actions
[Heroku]: https://www.heroku.com/what
[Maven]: https://maven.apache.org/install.html
[Java 11]: http://www.oracle.com/technetwork/java/javase/downloads/index.html readmeEtag: '"ac061ac9c9ff55a56d1048977aeb238dc1550477"' readmeLastModified: Sat, 20 Nov 2021 04:07:31 GMT repositoryId: 428406436 description: >- Simple, fast and reactive application to convert values between all currencies. created: '2021-11-15T20:07:30Z' updated: '2021-11-20T04:08:28Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: DanSilva41 logo: https://avatars.githubusercontent.com/u/16976267?v=4 license: MIT repoEtag: '"b9bdfc2e880839a382c44c52420cbe7e0a637abed2a634c54b6abcd076923064"' repoLastModified: Sat, 20 Nov 2021 04:08:28 GMT foundInMaster: true category: - Description Validators - Server Implementations id: 4d3b61994b4494edb5ef3547579ccefa - source: openapi3 tags repository: https://github.com/sshehrozali/dart-swaggergen-client v3: true id: 65ced9380360fbf91458408b214cbc69 repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyClRoaXMgaXMgYSBzYW1wbGUgc2VydmVyIFBldHN0b3JlIHNlcnZlci4gIFlvdSBjYW4gZmluZCBvdXQgbW9yZSBhYm91dCAgICAgU3dhZ2dlciBhdCBbaHR0cDovL3N3YWdnZXIuaW9dKGh0dHA6Ly9zd2FnZ2VyLmlvKSBvciBvbiBbaXJjLmZyZWVub2RlLm5ldCwgI3N3YWdnZXJdKGh0dHA6Ly9zd2FnZ2VyLmlvL2lyYy8pLiAgICAgIEZvciB0aGlzIHNhbXBsZSwgeW91IGNhbiB1c2UgdGhlIGFwaSBrZXkgYHNwZWNpYWwta2V5YCB0byB0ZXN0IHRoZSBhdXRob3JpemF0aW9uICAgICBmaWx0ZXJzLgoKVGhpcyBEYXJ0IHBhY2thZ2UgaXMgYXV0b21hdGljYWxseSBnZW5lcmF0ZWQgYnkgdGhlIFtTd2FnZ2VyIENvZGVnZW5dKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWNvZGVnZW4pIHByb2plY3Q6CgotIEFQSSB2ZXJzaW9uOiAxLjAuMAotIEJ1aWxkIHBhY2thZ2U6IGlvLnN3YWdnZXIuY29kZWdlbi5sYW5ndWFnZXMuRGFydENsaWVudENvZGVnZW4KCiMjIFJlcXVpcmVtZW50cwoKRGFydCAxLjIwLjAgb3IgbGF0ZXIgT1IgRmx1dHRlciAwLjAuMjAgb3IgbGF0ZXIKCiMjIEluc3RhbGxhdGlvbiAmIFVzYWdlCgojIyMgR2l0aHViCklmIHRoaXMgRGFydCBwYWNrYWdlIGlzIHB1Ymxpc2hlZCB0byBHaXRodWIsIHBsZWFzZSBpbmNsdWRlIHRoZSBmb2xsb3dpbmcgaW4gcHVic3BlYy55YW1sCmBgYApuYW1lOiBzd2FnZ2VyCnZlcnNpb246IDEuMC4wCmRlc2NyaXB0aW9uOiBTd2FnZ2VyIEFQSSBjbGllbnQKZGVwZW5kZW5jaWVzOgogIHN3YWdnZXI6CiAgICBnaXQ6IGh0dHBzOi8vZ2l0aHViLmNvbS8vLmdpdAogICAgICB2ZXJzaW9uOiAnYW55JwpgYGAKCiMjIyBMb2NhbApUbyB1c2UgdGhlIHBhY2thZ2UgaW4geW91ciBsb2NhbCBkcml2ZSwgcGxlYXNlIGluY2x1ZGUgdGhlIGZvbGxvd2luZyBpbiBwdWJzcGVjLnlhbWwKYGBgCmRlcGVuZGVuY2llczoKICBzd2FnZ2VyOgogICAgcGF0aDogL3BhdGgvdG8vc3dhZ2dlcgpgYGAKCiMjIFRlc3RzCgpUT0RPCgojIyBHZXR0aW5nIFN0YXJ0ZWQKClBsZWFzZSBmb2xsb3cgdGhlIFtpbnN0YWxsYXRpb24gcHJvY2VkdXJlXSgjaW5zdGFsbGF0aW9uLS11c2FnZSkgYW5kIHRoZW4gcnVuIHRoZSBmb2xsb3dpbmc6CgpgYGBkYXJ0CmltcG9ydCAncGFja2FnZTpzd2FnZ2VyL2FwaS5kYXJ0JzsKCi8vIFRPRE8gQ29uZmlndXJlIE9BdXRoMiBhY2Nlc3MgdG9rZW4gZm9yIGF1dGhvcml6YXRpb246IHBldHN0b3JlX2F1dGgKLy9zd2FnZ2VyLmFwaS5Db25maWd1cmF0aW9uLmFjY2Vzc1Rva2VuID0gJ1lPVVJfQUNDRVNTX1RPS0VOJzsKCnZhciBhcGlfaW5zdGFuY2UgPSBuZXcgUGV0QXBpKCk7CnZhciBib2R5ID0gbmV3IFBldCgpOyAvLyBQZXQgfCBQZXQgb2JqZWN0IHRoYXQgbmVlZHMgdG8gYmUgYWRkZWQgdG8gdGhlIHN0b3JlCgp0cnkgewogICAgYXBpX2luc3RhbmNlLmFkZFBldChib2R5KTsKfSBjYXRjaCAoZSkgewogICAgcHJpbnQoIkV4Y2VwdGlvbiB3aGVuIGNhbGxpbmcgUGV0QXBpLT5hZGRQZXQ6ICRlXG4iKTsKfQoKYGBgCgojIyBEb2N1bWVudGF0aW9uIGZvciBBUEkgRW5kcG9pbnRzCgpBbGwgVVJJcyBhcmUgcmVsYXRpdmUgdG8gKmh0dHBzOi8vcGV0c3RvcmUuc3dhZ2dlci5pby92MioKCkNsYXNzIHwgTWV0aG9kIHwgSFRUUCByZXF1ZXN0IHwgRGVzY3JpcHRpb24KLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tCipQZXRBcGkqIHwgWyoqYWRkUGV0KipdKGRvY3MvL1BldEFwaS5tZCNhZGRwZXQpIHwgKipQT1NUKiogL3BldCB8IEFkZCBhIG5ldyBwZXQgdG8gdGhlIHN0b3JlCipQZXRBcGkqIHwgWyoqZGVsZXRlUGV0KipdKGRvY3MvL1BldEFwaS5tZCNkZWxldGVwZXQpIHwgKipERUxFVEUqKiAvcGV0L3twZXRJZH0gfCBEZWxldGVzIGEgcGV0CipQZXRBcGkqIHwgWyoqZmluZFBldHNCeVN0YXR1cyoqXShkb2NzLy9QZXRBcGkubWQjZmluZHBldHNieXN0YXR1cykgfCAqKkdFVCoqIC9wZXQvZmluZEJ5U3RhdHVzIHwgRmluZHMgUGV0cyBieSBzdGF0dXMKKlBldEFwaSogfCBbKipmaW5kUGV0c0J5VGFncyoqXShkb2NzLy9QZXRBcGkubWQjZmluZHBldHNieXRhZ3MpIHwgKipHRVQqKiAvcGV0L2ZpbmRCeVRhZ3MgfCBGaW5kcyBQZXRzIGJ5IHRhZ3MKKlBldEFwaSogfCBbKipnZXRQZXRCeUlkKipdKGRvY3MvL1BldEFwaS5tZCNnZXRwZXRieWlkKSB8ICoqR0VUKiogL3BldC97cGV0SWR9IHwgRmluZCBwZXQgYnkgSUQKKlBldEFwaSogfCBbKip1cGRhdGVQZXQqKl0oZG9jcy8vUGV0QXBpLm1kI3VwZGF0ZXBldCkgfCAqKlBVVCoqIC9wZXQgfCBVcGRhdGUgYW4gZXhpc3RpbmcgcGV0CipQZXRBcGkqIHwgWyoqdXBkYXRlUGV0V2l0aEZvcm0qKl0oZG9jcy8vUGV0QXBpLm1kI3VwZGF0ZXBldHdpdGhmb3JtKSB8ICoqUE9TVCoqIC9wZXQve3BldElkfSB8IFVwZGF0ZXMgYSBwZXQgaW4gdGhlIHN0b3JlIHdpdGggZm9ybSBkYXRhCipQZXRBcGkqIHwgWyoqdXBsb2FkRmlsZSoqXShkb2NzLy9QZXRBcGkubWQjdXBsb2FkZmlsZSkgfCAqKlBPU1QqKiAvcGV0L3twZXRJZH0vdXBsb2FkSW1hZ2UgfCB1cGxvYWRzIGFuIGltYWdlCipTdG9yZUFwaSogfCBbKipkZWxldGVPcmRlcioqXShkb2NzLy9TdG9yZUFwaS5tZCNkZWxldGVvcmRlcikgfCAqKkRFTEVURSoqIC9zdG9yZS9vcmRlci97b3JkZXJJZH0gfCBEZWxldGUgcHVyY2hhc2Ugb3JkZXIgYnkgSUQKKlN0b3JlQXBpKiB8IFsqKmdldEludmVudG9yeSoqXShkb2NzLy9TdG9yZUFwaS5tZCNnZXRpbnZlbnRvcnkpIHwgKipHRVQqKiAvc3RvcmUvaW52ZW50b3J5IHwgUmV0dXJucyBwZXQgaW52ZW50b3JpZXMgYnkgc3RhdHVzCipTdG9yZUFwaSogfCBbKipnZXRPcmRlckJ5SWQqKl0oZG9jcy8vU3RvcmVBcGkubWQjZ2V0b3JkZXJieWlkKSB8ICoqR0VUKiogL3N0b3JlL29yZGVyL3tvcmRlcklkfSB8IEZpbmQgcHVyY2hhc2Ugb3JkZXIgYnkgSUQKKlN0b3JlQXBpKiB8IFsqKnBsYWNlT3JkZXIqKl0oZG9jcy8vU3RvcmVBcGkubWQjcGxhY2VvcmRlcikgfCAqKlBPU1QqKiAvc3RvcmUvb3JkZXIgfCBQbGFjZSBhbiBvcmRlciBmb3IgYSBwZXQKKlVzZXJBcGkqIHwgWyoqY3JlYXRlVXNlcioqXShkb2NzLy9Vc2VyQXBpLm1kI2NyZWF0ZXVzZXIpIHwgKipQT1NUKiogL3VzZXIgfCBDcmVhdGUgdXNlcgoqVXNlckFwaSogfCBbKipjcmVhdGVVc2Vyc1dpdGhBcnJheUlucHV0KipdKGRvY3MvL1VzZXJBcGkubWQjY3JlYXRldXNlcnN3aXRoYXJyYXlpbnB1dCkgfCAqKlBPU1QqKiAvdXNlci9jcmVhdGVXaXRoQXJyYXkgfCBDcmVhdGVzIGxpc3Qgb2YgdXNlcnMgd2l0aCBnaXZlbiBpbnB1dCBhcnJheQoqVXNlckFwaSogfCBbKipjcmVhdGVVc2Vyc1dpdGhMaXN0SW5wdXQqKl0oZG9jcy8vVXNlckFwaS5tZCNjcmVhdGV1c2Vyc3dpdGhsaXN0aW5wdXQpIHwgKipQT1NUKiogL3VzZXIvY3JlYXRlV2l0aExpc3QgfCBDcmVhdGVzIGxpc3Qgb2YgdXNlcnMgd2l0aCBnaXZlbiBpbnB1dCBhcnJheQoqVXNlckFwaSogfCBbKipkZWxldGVVc2VyKipdKGRvY3MvL1VzZXJBcGkubWQjZGVsZXRldXNlcikgfCAqKkRFTEVURSoqIC91c2VyL3t1c2VybmFtZX0gfCBEZWxldGUgdXNlcgoqVXNlckFwaSogfCBbKipnZXRVc2VyQnlOYW1lKipdKGRvY3MvL1VzZXJBcGkubWQjZ2V0dXNlcmJ5bmFtZSkgfCAqKkdFVCoqIC91c2VyL3t1c2VybmFtZX0gfCBHZXQgdXNlciBieSB1c2VyIG5hbWUKKlVzZXJBcGkqIHwgWyoqbG9naW5Vc2VyKipdKGRvY3MvL1VzZXJBcGkubWQjbG9naW51c2VyKSB8ICoqR0VUKiogL3VzZXIvbG9naW4gfCBMb2dzIHVzZXIgaW50byB0aGUgc3lzdGVtCipVc2VyQXBpKiB8IFsqKmxvZ291dFVzZXIqKl0oZG9jcy8vVXNlckFwaS5tZCNsb2dvdXR1c2VyKSB8ICoqR0VUKiogL3VzZXIvbG9nb3V0IHwgTG9ncyBvdXQgY3VycmVudCBsb2dnZWQgaW4gdXNlciBzZXNzaW9uCipVc2VyQXBpKiB8IFsqKnVwZGF0ZVVzZXIqKl0oZG9jcy8vVXNlckFwaS5tZCN1cGRhdGV1c2VyKSB8ICoqUFVUKiogL3VzZXIve3VzZXJuYW1lfSB8IFVwZGF0ZWQgdXNlcgoKCiMjIERvY3VtZW50YXRpb24gRm9yIE1vZGVscwoKIC0gW0FwaVJlc3BvbnNlXShkb2NzLy9BcGlSZXNwb25zZS5tZCkKIC0gW0NhdGVnb3J5XShkb2NzLy9DYXRlZ29yeS5tZCkKIC0gW09yZGVyXShkb2NzLy9PcmRlci5tZCkKIC0gW1BldF0oZG9jcy8vUGV0Lm1kKQogLSBbVGFnXShkb2NzLy9UYWcubWQpCiAtIFtVc2VyXShkb2NzLy9Vc2VyLm1kKQoKCiMjIERvY3VtZW50YXRpb24gRm9yIEF1dGhvcml6YXRpb24KCgojIyBhcGlfa2V5CgotICoqVHlwZSoqOiBBUEkga2V5Ci0gKipBUEkga2V5IHBhcmFtZXRlciBuYW1lKio6IGFwaV9rZXkKLSAqKkxvY2F0aW9uKio6IEhUVFAgaGVhZGVyCgojIyBwZXRzdG9yZV9hdXRoCgotICoqVHlwZSoqOiBPQXV0aAotICoqRmxvdyoqOiBpbXBsaWNpdAotICoqQXV0aG9yaXphdGlvbiBVUkwqKjogaHR0cDovL3BldHN0b3JlLnN3YWdnZXIuaW8vb2F1dGgvZGlhbG9nCi0gKipTY29wZXMqKjogCiAtICoqd3JpdGU6cGV0cyoqOiBtb2RpZnkgcGV0cyBpbiB5b3VyIGFjY291bnQKIC0gKipyZWFkOnBldHMqKjogcmVhZCB5b3VyIHBldHMKCgojIyBBdXRob3IKCmFwaXRlYW1Ac3dhZ2dlci5pbwoKCg== readmeEtag: '"3a144a5d3bd115a160fef34ca633bbcd9e2e1335"' readmeLastModified: Thu, 23 Jun 2022 09:18:23 GMT repositoryId: 506573218 description: >- Dart client-side code generated using Swagger Codegen and OpenAPI 3.0 specifications. created: '2022-06-23T09:17:30Z' updated: '2022-09-30T09:17:03Z' language: Dart archived: false stars: 0 watchers: 1 forks: 0 owner: sshehrozali logo: https://avatars.githubusercontent.com/u/58667536?v=4 repoEtag: '"22867485bee969aa3e83c9f304127e79908dce2772fa00691c2fabdc20ba7ec4"' repoLastModified: Fri, 30 Sep 2022 09:17:03 GMT category: Parsers foundInMaster: true - source: https://openapi.tools/ name: BlocklyAutomation category: - Code Generators - SDK - Documentation - Testing link: https://ignatandrei.github.io/BlocklyAutomation/ repository: http://github.com/ignatandrei/blocklyautomation language: - Javascript - .NET source_description: >- Input any OpenAPI document to have generated Blocks in Blockly form to test and generate documentation. v3: true v3_1: true repositoryMetadata: base64Readme: >- # VisualAPI - Add ( to your |  inside your ) application LowCode Macros for YOUR API / HTTP Calls

<!-- ALL-CONTRIBUTORS-BADGE:START - Do not remove or modify this section -->
[![All Contributors](https://img.shields.io/badge/all_contributors-2-orange.svg?style=flat-square)](#contributors-)
<!-- ALL-CONTRIBUTORS-BADGE:END -->

Blockly Automation :  is a tool for LowCode / automation of sites (Swagger / OpenAPI / RPA ) and PC made of Blockly. 

# How to use if you are

## a backend developer with an REST / OpenAPI / Swagger enabled site and I want to show the use of my site


Use the appropiate package: 

### [NetCore](https://github.com/ignatandrei/netcoreblockly/)  , made by Andrei Ignat

.NET Core: [![Nuget](https://img.shields.io/nuget/dt/NetCore2Blockly)](https://www.nuget.org/packages/NetCore2Blockly/)

More details at https://github.com/ignatandrei/netcoreblockly/


### [Java](https://github.com/eciuca/blockly-automation-starter-webmvc-ui) , Full example by Emanuel Ciuca 

Java Spring : [![Maven Central](https://img.shields.io/maven-central/v/io.github.eciuca.blockly/blockly-automation-starter-webmvc-ui)](https://central.sonatype.com/artifact/io.github.eciuca.blockly/blockly-automation-starter-webmvc-ui/)

The Maven is https://central.sonatype.com/artifact/io.github.eciuca.blockly/blockly-automation-starter-webmvc-ui/

The repo with example is https://github.com/eciuca/blockly-automation-starter-webmvc-ui 


### [Node](https://github.com/ignatandrei/BlocklyAutomation/wiki/node) , made by Andrei Ignat

Node: [![npm](https://img.shields.io/npm/v/node2-blockly)](https://www.npmjs.com/package/node-blockly?)

More details at https://github.com/ignatandrei/BlocklyAutomation/wiki/node

### [PHP](https://github.com/Tynael/laravel-blockly-automation) , made by Carol Pelu



Packagist : [![php](https://img.shields.io/packagist/v/carolpelu/blockly-automation)](https://packagist.org/packages/carolpelu/blockly-automation)

More details at https://github.com/Tynael/laravel-blockly-automation




## Docker Extension

https://open.docker.com/extensions/marketplace?extensionId=ignatandrei/blockly-automation


### Manual

Download the HTML release [BlocklyAutomation](https://github.com/ignatandrei/BlocklyAutomation/releases/latest/download/releaseBlocklyAutomation.zip/) .

Follow the instructions in the wiki file.  

#### [Manual Java](https://github.com/cosminpopescu14/math-operations-swagger) , Manual example by Cosmin Popescu 

Example by Cosmin Popescu at https://github.com/cosminpopescu14/math-operations-swagger



## a frontend  developer that finds a bug in a REST / OpenAPI / Swagger enabled site

You want to show to the backend developer how to reproduce the problem .

If you control the site , then you can use the [BlocklyAutomation](https://github.com/ignatandrei/BlocklyAutomation/releases/latest/download/releaseBlocklyAutomation.zip/) to reproduce the problem.

Follow the instructions in the wiki file.  

If you do not control the site, then install the application from http://ba.serviciipeweb.ro/ and then use it to reproduce the problem. ( wiki site coming with details) 

## a web site application tester
If you control the site , then can use the https://github.com/ignatandrei/BlocklyAutomation/releases/latest/download/releaseBlocklyAutomation.zip/  and then use it to reproduce the problem. ( wiki site coming with details) .

If you does not control the site, then install the application from http://ba.serviciipeweb.ro/ and then use it to make test cases. ( wiki site coming with details) 


## [Docker Extension](https://github.com/ignatandrei/BlocklyAutomation/wiki/DockerExtension)

Please see [Docker Extension](https://github.com/ignatandrei/BlocklyAutomation/wiki/DockerExtension)
See 
## not a programmer.  I want to automate/gather data from several sites ( public or private )

You want to obtain some data from the web. For example, extract the exchange between EUR / RON .

Install the application from http://ba.serviciipeweb.ro/ .

Follow the instructions in the wiki file.


## not a programmer andI want to automate things on my PC. 

You want to obtain some data from your PC. For example, extract and export to CSV the Chrome Bookmarks.

Install the application from http://ba.serviciipeweb.ro/ .

Follow the instructions in the wiki file.



## Suggest a public API for BlocklyAutomation

If you have a public API/site that you want to automate, please send file an issue at https://github.com/ignatandrei/BlocklyAutomation/issues 


## How to see a preview


To see the whole potential , please go to http://ba.serviciipeweb.ro/ and click Launch .

If you want to see some web Demo , please go to https://ignatandrei.github.io/BlocklyAutomation/ and test various HTTP request, Swaggers and more

## How to install on your PC

See [releases tab](https://github.com/ignatandrei/BlocklyAutomation/releases)

You can find the 

1. Angular site , ready to be deployed to any server
2. IIS site   , ready to be deployed on IIS
3. A .NET Core site, ready to be deployed on Linux or Windows, standalone ( as a service )

# You have a site with OpenAPI / Swagger. What should I do  ?

Download from [releases tab](https://github.com/ignatandrei/BlocklyAutomation/releases) the Angular site. Put index.html and all other files into a BLocklyAutomation folder inside your wwwroot  folder ( or in your project root ) and you are good to go.

1. It could be an idea to map everything that start with /BlocklyAutomation to /BlocklyAutomation/index.html - see src\Loaders\SimpleSite to have an C# example 
   
2. Modify assets/settings.json to change the name and the starting blocks
   
3. Modify assets/loadAtStartup/swaggers.json to add your swaggers

4. Modify assets/showUsage/demoBlocks/all.txt to add your demo for the blocks . 
   ( You can construct and then download and save as files)

5. Send me an email to help you  if something  is not working.
## How to customize

To customize the title , introduction and start blocks , see assets/settings.json

To load swaggers at startup, see  assets/loadAtStartup/swaggers.json

To customize demo blocks,download your blocks, put the txt file in  assets/showUsage/demoBlocks/ and modify assets/showUsage/demoBlocks/all.txt

##  How to contribute

If you are a beginner to blockly, see 

https://github.com/ignatandrei/BlocklyAutomation/issues?q=is%3Aissue+is%3Aopen+label%3A%22good+first+issue%22

![GitHub issues by-label](https://img.shields.io/github/issues/ignatandrei/BlocklyAutomation/good%20first%20issue)

You will be mentioned below ;-)


## Contributors ✨

Thanks goes to these wonderful people ([emoji key](https://allcontributors.org/docs/en/emoji-key)):

<!-- ALL-CONTRIBUTORS-LIST:START - Do not remove or modify this section -->
<!-- prettier-ignore-start -->
<!-- markdownlint-disable -->
<table>
  <tr>
    <td align="center"><a href="https://aenyx-designs.com/"><img src="https://avatars.githubusercontent.com/u/33196341?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Bogdan Bobe</b></sub></a><br /><a href="#design-arealshadow" title="Design">🎨</a> <a href="https://github.com/ignatandrei/BlocklyAutomation/commits?author=arealshadow" title="Code">💻</a></td>
    <td align="center"><a href="https://github.com/adrian-badulescu"><img src="https://avatars.githubusercontent.com/u/49490946?v=4?s=100" width="100px;" alt=""/><br /><sub><b>Adrian Badulescu</b></sub></a><br /><a href="#example-adrian-badulescu" title="Examples">💡</a></td>
  </tr>
</table>

<!-- markdownlint-restore -->
<!-- prettier-ignore-end -->

<!-- ALL-CONTRIBUTORS-LIST:END -->

This project follows the [all-contributors](https://github.com/all-contributors/all-contributors) specification. Contributions of any kind welcome! readmeEtag: '"0b8a3b0a75a8247f4625843bda3826f655a7eb35"' readmeLastModified: Tue, 23 Jul 2024 17:48:16 GMT repositoryId: 427082891 description: >- VisualAPI - LowCode Macros for YOUR API / HTTP Calls : https://visualapi.azurewebsites.net/ created: '2021-11-11T17:14:47Z' updated: '2025-03-23T17:11:34Z' language: JavaScript archived: false stars: 40 watchers: 3 forks: 10 owner: ignatandrei logo: https://avatars.githubusercontent.com/u/153982?v=4 license: MIT repoEtag: '"b66fb339c972682afb29c2a3b793dd7f021da2db621a9b899a2fe2e1025a692a"' repoLastModified: Sun, 23 Mar 2025 17:11:34 GMT foundInMaster: true id: aa2b2e89cc4f3ca16559d1e0a1cef89d - source: openapi3 tags repository: https://github.com/fugerit-org/yaml-doc-maven-plugin v3: true id: 45a49daf369a856447a2dd1a0edd8c87 repositoryMetadata: base64Readme: >- IyB5YW1sLWRvYy1tYXZlbi1wbHVnaW4KCj4g4pqg77iPICoqV2FybmluZzoqKiBBcyBvZiAyMDI1LTA3LTI3IHRoaXMgcmVwb3NpdG9yeSBpcyBhcmNoaXZlZCBhbmQgc3Vic3RpdHV0ZWQgYnkgW29wZW5hcGktZG9jLW1hdmVuLXBsdWdpbl0oaHR0cHM6Ly9naXRodWIuY29tL2Z1Z2VyaXQtb3JnL29wZW5hcGktZG9jLW1hdmVuLXBsdWdpbikgKHNlZSBbZnVnZXJpdC1vcmcvb3BlbmFwaS1kb2MtdG9vbCMxXShodHRwczovL2dpdGh1Yi5jb20vZnVnZXJpdC1vcmcvb3BlbmFwaS1kb2MtdG9vbC9pc3N1ZXMvMSkpLgoKU2ltcGxlIG1hdmVuIHBsdWdpbiBmb3IgW3lhbWwtZG9jLXRvb2xdKGh0dHBzOi8vZ2l0aHViLmNvbS9mdWdlcml0LW9yZy95YW1sLWRvYy10b29sKSBwcm9qZWN0LgoKWyFbS2VlcCBhIENoYW5nZWxvZyB2MS4xLjAgYmFkZ2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvY2hhbmdlbG9nLUtlZXAlMjBhJTIwQ2hhbmdlbG9nJTIwdjEuMS4wLSUyM0UwNTczNSldKENIQU5HRUxPRy5tZCkgClshW01hdmVuIENlbnRyYWxdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbWF2ZW4tY2VudHJhbC92L29yZy5mdWdlcml0LmphdmEveWFtbC1kb2MtbWF2ZW4tcGx1Z2luLnN2ZyldKGh0dHBzOi8vbXZucmVwb3NpdG9yeS5jb20vYXJ0aWZhY3Qvb3JnLmZ1Z2VyaXQuamF2YS95YW1sLWRvYy1tYXZlbi1wbHVnaW4pClshW2xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvTGljZW5zZS1BcGFjaGUlMjBMaWNlbnNlJTIwMi4wLXRlYWwuc3ZnKV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9BcGFjaGUtMi4wKQpbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1mdWdlcml0LW9yZ195YW1sLWRvYy1tYXZlbi1wbHVnaW4mbWV0cmljPWFsZXJ0X3N0YXR1cyldKGh0dHBzOi8vc29uYXJjbG91ZC5pby9zdW1tYXJ5L25ld19jb2RlP2lkPWZ1Z2VyaXQtb3JnX3lhbWwtZG9jLW1hdmVuLXBsdWdpbikKWyFbQ292ZXJhZ2VdKGh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PWZ1Z2VyaXQtb3JnX3lhbWwtZG9jLW1hdmVuLXBsdWdpbiZtZXRyaWM9Y292ZXJhZ2UpXShodHRwczovL3NvbmFyY2xvdWQuaW8vc3VtbWFyeS9uZXdfY29kZT9pZD1mdWdlcml0LW9yZ195YW1sLWRvYy1tYXZlbi1wbHVnaW4pCgpBY2NlcHRlZCBjb25maWcgcGFyYW1zIGFyZSA6ICAKKiBjb25maWdQYXRoCiogaWRDYXRhbG9nCgpIZXJlIGEgc2FtcGxlIGNvbmZpZ3VyYXRpb24gIDoKCmBgYAoJCQk8cGx1Z2luPgoJCQkJPGdyb3VwSWQ+b3JnLmZ1Z2VyaXQuamF2YTwvZ3JvdXBJZD4KCQkJCTxhcnRpZmFjdElkPnlhbWwtZG9jLW1hdmVuLXBsdWdpbjwvYXJ0aWZhY3RJZD4KCQkJCTx2ZXJzaW9uPiR7eWFtbC1kb2MtdmVyc2lvbn08L3ZlcnNpb24+CQoJCQkJPGNvbmZpZ3VyYXRpb24+CgkJCQkJPGNvbmZpZ1BhdGg+c3JjL2NvbmZpZy95YW1sLWRvYy1jb25maWcueG1sPC9jb25maWdQYXRoPgoJCQkJCTxpZENhdGFsb2c+b3BlbmFwaTwvaWRDYXRhbG9nPgkJCgkJCQk8L2NvbmZpZ3VyYXRpb24+CQkJCQkJCQoJCQkJPGV4ZWN1dGlvbnM+CgkJCQkJPGV4ZWN1dGlvbj4KCQkJCQkJPGlkPm9wZW5hcGk8L2lkPgoJCQkJCQk8Z29hbHM+CgkJCQkJCQk8Z29hbD5nZW5lcmF0ZTwvZ29hbD4KCQkJCQkJPC9nb2Fscz4KCQkJCQk8L2V4ZWN1dGlvbj4JCQoJCQkJPC9leGVjdXRpb25zPgoJCQk8L3BsdWdpbj4JCmBgYA== readmeEtag: '"be24b0358a5ab30a9051d585c6ec11c044ad773e"' readmeLastModified: Sun, 27 Jul 2025 18:59:46 GMT repositoryId: 473247612 description: Simple maven plugin for the yaml-doc-tool project created: '2022-03-23T15:26:14Z' updated: '2025-07-27T19:00:24Z' language: Java archived: true stars: 0 watchers: 1 forks: 0 owner: fugerit-org logo: https://avatars.githubusercontent.com/u/37816284?v=4 license: Apache-2.0 repoEtag: '"e4ab86716f3deaaa1f00f045402bbb5fc1822acd5d82c8cd2c584eae64ff05ff"' repoLastModified: Sun, 27 Jul 2025 19:00:24 GMT category: SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/potjerodekool/openapi-generator v3: true id: 24275a8d228f2d092e16833a20c857f4 repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWdlbmVyYXRvcgpDb2RlIGdlbmVyYXRpb24gZm9yIFNwcmluZ2Jvb3QgYmFzZWQgZnJvbSBvcGVuIGFwaQoKU2VlIHRoZSBbd2lraV0oaHR0cHM6Ly9naXRodWIuY29tL3BvdGplcm9kZWtvb2wvb3BlbmFwaS1nZW5lcmF0b3Ivd2lraSkgaW5mbyBhbmQgZXhwbGFpbmF0aW9ucy4K readmeEtag: '"ad6bbf76a7e9a49ae43ec7d5b3d8fb65196197ee"' readmeLastModified: Fri, 08 Mar 2024 19:43:18 GMT repositoryId: 518195562 description: Code generation for Springboot based from open api created: '2022-07-26T19:41:26Z' updated: '2022-07-26T20:11:04Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: potjerodekool logo: https://avatars.githubusercontent.com/u/141829?v=4 license: Apache-2.0 repoEtag: '"1de852112a173dff3a65f9b7a5a5f2000ada4c9916e8823330a70ea933624dda"' repoLastModified: Tue, 26 Jul 2022 20:11:04 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/sptrakesh/openapi2latex v3: true id: 04cdcfcdb01f0c2c253c27afe50329f7 repositoryMetadata: base64Readme: >- # OpenAPI2LaTeX
* [Structure](#structure)
* [Usage](#usage)
  * [Install Dependencies](#install-dependencies)
  * [Command Line Options](#command-line-options)
* [Extension](#extensions)
* [Limitations](#limitations)

Utility process to generate a LaTeX file from an OpenAPI specification.

The rationale for this process is to be able to generate a PDF document that can be distributed to interested
parties, when the source specifications are protected by access control (and where the said parties do not need to
be provided with the access credentials).

The workflow is to use this process to generate the target LaTeX file, and run `xelatex` a few times (usually
two times to get cross-references resolved) for the output PDF document.  Also run `makeindex` to generate the
document *index* if desired.

```shell
<path to>/oa2tex -i <path to openapi.yaml> -o <path to output directory> -c --use-cmark
cd <path to output directory>
xelatex -interaction=nonstopmode openapi
makeindex openapi
xelatex -interaction=nonstopmode openapi
```

## Structure
The generated LaTeX files have the following structure (you can of course modify the output files as desired):

* **Preamble** - Preamble for the document (`<path to output>/preamble.tex`).  Sets up various packages that are used.
  Edit the file as desired, especially the *main font* for the document.  The generator sets the main font to
  *Helvetica Neue* (`\setmainfont[Ligatures=TeX,Numbers=OldStyle]{Helvetica Neue}`).  Change to any system
  supported font as desired (through the command line option or by editing file as desired after generating the
  latex sources).
  * Serif fonts like *Helvetica*, *Verdana* etc. are good for viewing the PDF on screen.
  * Use traditional print friendly fonts like the LaTeX default *Computer Modern*, *Times New Roman* etc if
    the primary purpose of the output PDF is print.
* **Frontmatter** - Titlepage and table of contents.
* **Mainmatter** - Contains two or three parts.
  * **Info** - The *info* object is presented as the first chapter. 
  * **Examples** - Any examples that are defined in the `openapi.components.examples` section.
  * **Parameters** - Any parameters that are *referenced* from the various API endpoints. Will only list referenced
    parameters, not those that are defined *in-line*.
  * **Endpoints** - Part with the path operations grouped by tags. Each tag is presented in a *chapter*.
  * **Schemas** - Part with the *schemas* declared and referenced in the specification.  Each *schema* is presented in a
    *chapter*.  Schemas are listed alphabetically by their filename and entity name.
  * **Responses* - Optional part with *responses* declared in the `openapi.components` structure.  If none are defined
    no part is created.
  * **Request Bodies** - Part with request bodies that are *referenced*, not if they are defined inline with *schema* references.
  * **Code Samples** - If the `x-codeSamples` extension exists for operations, these are collected together into a third
    *part*.  Code samples are grouped together under each *tag* group, which is presented as a *chapter*.
* **Backmatter** - List of tables, and optionally index.
  *  **Index** - Operation ids and schema property names are added to index.

See [openapi.pdf](openapi.pdf) for the PDF generated from the sample 
[petstore](https://github.com/SLdragon/example-openapi-spec/blob/main/petstore-official.yaml)
specifications.  Note that the petstore sample has a CommonMark table in the information, which
the embedded converter does not support.  This sample was generated using the `cmark` option.

## Usage
The generator is written in C++ and has a dependency on the [Boost](https://boost.org/) libraries.

**Note:** The old [Julia](https://julialang.org/) version of the utility is still available in the `julia` branch.

### Command Line Options
The following options are supported by the [main.cpp](src/main.cpp) program:

* `--input | -i` - **Required**. The main OpenAPI specification file to parse.
* `--output | -o` - **Required**. The output LaTeX file to generate.  Best to place this at another location than the 
  api specifications.  Program will attempt to create the directory tree if it does not exist.  Since a lot of
  files are generated by this process, and further when running LaTeX, it would be best to write the output to
  a dedicated temporary directory.
* `--author | -a` - The author credit to show on the titlepage.
* `--footer | -f` - The right side footer text to display for the document.
* `--font | -t` - The font to use for the output document.  Default *Helvetica Neue*.  Note the font must be available
  on the system.
* `--operation-summary | -s` - A flag to indicate that **Operation** *summary* should be used as section headings instead of *operationId*.
* `--use-cmark | -m` - Use [cmark](https://github.com/commonmark/cmark) to convert `info.description` to latex.
  Recommended option, since the description can be quite long and complicated, and `cmark` should have much more 
  comprehensive support for converting CommonMark to latex.  Note the program uses the `cmark` utility via
  `std::system`, and not the library.
* `--log-level | -l` - Set the logging level (`critical|warn|info|debug`).  Default `info`.
* `--console | -c` - Flag to indicate logs should also be echoed to `stdout`.
* `--log-dir | -d` - The directory under which the process log output is written.  The directory *must* exist.
  Default `/tmp/` - note the mandatory trailing `/`.

#### Example
The following options were used to generate the sample petstore specifications document.
```shell
/usr/local/spt/bin/oa2tex -i /tmp/petstore-official.yaml -o /tmp/petstore -s -c --use-cmark
```

## Extensions
A few extensions to the specifications developed by [Redocly](https://redocly.com/) are supported. 

* Source code samples are parsed from the `x-codeSamples` array attached to an operation.  All code samples are attached
  to a separate *part* of the output document, and follow the same chapter organisation as the API tags.
* Tag groups are parsed from the `x-tagGroups` array attached to the root of the specification document.  If specified
  an initial chapter **Tag Groups** is added, which lists the groupings with links to the **Tag** chapters.
* 
## Build
Standard `cmake` build system.

```shell
git clone https://github.com/sptrakesh/openapi2latex.git
cd openapi2latex
cmake -DCMAKE_PREFIX_PATH=/usr/local/boost \
  -DCMAKE_BUILD_TYPE=Release \
  -DCMAKE_INSTALL_PREFIX=/usr/local/spt \
  -S . -B build
cmake --build build -j12
sudo cmake --install build
```

## Limitations
Probably too many to list, but the following items should be kept in mind.

* This process is based on the way *I write API specifications*, and markup descriptions.
* Schema objects are assumed to model closely their organisation in a source code implementation.  This in turn implies
  that nested structures are represented as schema references, and not listed in-line in the schema.  Deeply nested in-line
  schemas would be very hard to represent in a printed document in any case.
* Mainly tested with specifications that are split into individual files - representing paths, schemas, parameters etc.
  Most testing has been against large handwritten specifications, which follows the principles laid
  out in [split specifications](https://davidgarcia.dev/posts/how-to-split-open-api-spec-into-multiple-files/).  A few
  simple single file specifications have also been tested.
* Mainly supports OpenAPI specification version [3.0.3](https://spec.openapis.org/oas/v3.0.3), although some properties
  from [3.1.0](https://spec.openapis.org/oas/latest.html) are also included.
* Only supports loading local specification files in YAML format.  JSON is not supported at present.
* Not all properties/aspects of the specification are output in the generated LaTeX file.  I selected what I felt are
  most relevant to be shared.
* Markdown markup may not be fully translated to LaTeX.  See [convert.cpp](test/convert.cpp) for basic tests around the implemented.
  * Bold/italic blocks of text (spanning paragraphs) are not supported.

The output is a wrapping LaTeX file that includes several smaller included files, and hence can be easily modified 
as needed to further customise the final PDF document.

## Dependencies
* **[rapidyaml](https://github.com/biojppm/rapidyaml)** YAML parser library.  Via `cmake fetchcontent`
* **[Boost](https://boost.org/)** expected local installation.
* **[Clara](https://github.com/catchorg/Clara)** - Command line options parser. Included in project.
* **[Catch2](https://github.com/catchorg/Catch2)** - Testing framework. Via `cmake fetchcontent`
* **[NanoLog](https://github.com/Iyengar111/NanoLog)** - Logging framework used for the project.  Modified version included in project. readmeEtag: '"6a05f800d342357b65d93c0177b8d0e091b93e17"' readmeLastModified: Mon, 19 Jan 2026 18:21:50 GMT repositoryId: 579668923 description: Generate LaTeX source from OpenAPI specification. created: '2022-12-18T13:32:59Z' updated: '2026-01-19T18:22:37Z' language: C++ archived: false stars: 1 watchers: 1 forks: 0 owner: sptrakesh logo: https://avatars.githubusercontent.com/u/2648309?v=4 license: Apache-2.0 repoEtag: '"f25886d5deb9e40086b3c77a13f6b422581389fbca4f1b224e0d616f8e92bfd9"' repoLastModified: Mon, 19 Jan 2026 18:22:37 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/saphir-lab/api_data_dictionary v3: true id: a89430440ab1168c48ce33d33170f562 repositoryMetadata: base64Readme: >- IyBhcGlfZGF0YV9kaWN0aW9uYXJ5ClRoaXMgc2NyaXB0IGJ1aWxkcyBhIGRhdGEgZGljdGlvbmFyeSBmcm9tIG9wZW5hcGkvc3dhZ2dlciBkb2N1bWVudGF0aW9uICh2ZXJzaW9uIDMueCkKClRoZSBkaWN0aW9uYXJ5IGNvbnRhaW5zIGFsbCB0aGUgcGFyYW1ldGVycy9maWVsZHMgbmFtZSBkaXNjb3ZlcmVkIGluIHRoZSBvcGVuYXBpIGZpbGUgdG9nZXRoZXIgd2l0aCB0aGVpciBjaGFyYWN0ZXJpc3RpY3MgKFJlcXVpcmVkIChZL04pLCBUeXBlcyAoc3RyaW5nLCBhcnJheSwgYm9vbGVhbiwgbnVtYmVyLCAuLi4pLCBBUEkgcGF0aCB3aGVyZSB1c2VkLCBsaXN0IG9mIGRlc2NyaXB0aW9ucywgLi4uKQoKIyMgVXNhZ2UKWW91IG5lZWQgdG8gc3BlY2lmeSBhdCBsZWFzdCBvbmUgYXJndW1lbnQgaW4gb3JkZXIgdHUgcnVuIHRoZSBzY3JpcHQocGF0aCBvZiB0aGUgb3BlbmFwaSBmaWxlKToKCmBgYGJhc2gKcHl0aG9uIG1haW4ucHkgb3BlbmFwaS5qc29uCmBgYAoKCj4qKk5vdGUqKjogKmZvciBtb3JlIGluZm9ybWF0aW9uIGFib3V0IHBvc3NpYmxlIHBhcmFtZXRlcnMsIHR5cGU6Kgo+Cj4gYGBgYmFzaAo+IHB5dGhvbiBtYWluLnB5IC0taGVscAo+IGBgYAo= readmeEtag: '"80d30964cf9ab7d67215acb7c410573a596c0af1"' readmeLastModified: Fri, 20 Jan 2023 12:35:18 GMT repositoryId: 538355860 description: Build a data dictionary from openapi/swagger documentation created: '2022-09-19T06:04:04Z' updated: '2023-01-19T14:26:41Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: saphir-lab logo: https://avatars.githubusercontent.com/u/99727450?v=4 license: MIT repoEtag: '"4d8f6afcd77808a7f2aab44e2257e26e07f4596a721883166e8bc8b4f46d5815"' repoLastModified: Thu, 19 Jan 2023 14:26:41 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/youssefjn/expense-tracker v3: true id: 62acb770808cc8cedcab29da3a9c9d5d repositoryMetadata: repositoryId: 601972677 description: REST api with swagger documentation created: '2023-02-15T08:18:03Z' updated: '2023-02-15T14:08:25Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: youssefjn logo: https://avatars.githubusercontent.com/u/59120174?v=4 repoEtag: '"13f2e31fec3dc222a006a645842a2271d8cc8ecfd437066fa8e7adacaed1027a"' repoLastModified: Wed, 15 Feb 2023 14:08:25 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/andrii-hrechyn/auto-documentation v3: true id: e97dd2eaec8fa77bafeded0f536da462 repositoryMetadata: base64Readme: >- # Auto Documentation

A fluent, object-oriented library for generating OpenAPI 3.1.0 documentation in Laravel applications.

![PHP 8.3+](https://img.shields.io/badge/PHP-8.3%2B-blue)
![Laravel 7-11](https://img.shields.io/badge/Laravel-7--11-red)
![License: MIT](https://img.shields.io/badge/License-MIT-green)

## Introduction

Auto Documentation lets you describe your API endpoints as PHP classes using a fluent builder API. The library auto-discovers your documentation classes, resolves reusable components via `$ref`, and generates a complete OpenAPI 3.1.0 specification as YAML — viewable through a built-in Redoc UI.

Key features:

- **Fluent PHP API** — describe paths, schemas, parameters, and responses with method chaining
- **Auto-discovery** — path and component classes are automatically found and registered
- **Reusable components** — schemas, parameters, requests, and responses are deduplicated via `$ref`
- **Laravel integration** — built-in artisan commands, route-based request body extraction, Sanctum auth support
- **OpenAPI 3.1.0** compliant output

## Requirements

- PHP 8.3+
- Laravel 7, 8, 9, 10, or 11

## Installation

Install via Composer:

```bash
composer require andrii-hrechyn/auto-documentation
```

Run the install command to scaffold the `docs/` folder with examples and register the `Docs\\` namespace in your `composer.json` autoload:

```bash
php artisan auto-doc:install
```

This creates the following structure:

```
docs/
├── Components/
│   ├── Parameters/
│   ├── Requests/
│   ├── Responses/
│   ├── Schemas/
│   └── Security/
├── Paths/
└── Documentation.php
```

## Quick Start

### 1. Define your Documentation class

```php
// docs/Documentation.php
namespace Docs;

use AutoDocumentation\BaseDocumentation;
use AutoDocumentation\Info;

class Documentation extends BaseDocumentation
{
    protected function info(): Info
    {
        return Info::make('My API', '1.0.0')
            ->description('API documentation');
    }
}
```

### 2. Create a path component

```php
// docs/Paths/UsersPath.php
namespace Docs\Paths;

use AutoDocumentation\Base\PathComponent;
use AutoDocumentation\Paths\Method;
use AutoDocumentation\Paths\Path;
use AutoDocumentation\Properties\IntegerProperty;
use AutoDocumentation\Properties\StringProperty;

class UsersPath extends PathComponent
{
    public function path(): Path
    {
        return $this->make('users');
    }

    public function get(Method $method): Method
    {
        return $method
            ->operationId('listUsers')
            ->summary('List all users')
            ->tag('Users')
            ->jsonResponse([
                IntegerProperty::make('id')->example(1),
                StringProperty::make('name')->example('John'),
                StringProperty::make('email')->format('email'),
            ]);
    }
}
```

### 3. Generate documentation

```bash
php artisan auto-doc:generate
```

The specification is written to `storage/app/auto-docs/documentation.yaml` and served at `/api/doc`.

## Documentation Class

The `Documentation` class extends `BaseDocumentation` and serves as the entry point for your API spec. It has one required method and several optional ones.

### Info (required)

```php
protected function info(): Info
{
    return Info::make('Blog Platform API', '2.0.0')
        ->description('API documentation for the Blog Platform')
        ->termsOfService('https://example.com/terms')
        ->contact(
            (new Contact())
                ->name('API Support')
                ->email('support@example.com')
                ->url('https://example.com/support')
        )
        ->license(
            License::make('MIT')
                ->url('https://opensource.org/licenses/MIT')
        );
}
```

### Servers

```php
protected function servers(): array
{
    return [
        Server::make('https://api.example.com/{version}')
            ->description('Production server')
            ->variable(
                Variable::make('version', 'v2')
                    ->enum(['v1', 'v2'])
                    ->description('API version')
            ),
    ];
}
```

### Default Security

Set a global security scheme applied to all operations:

```php
protected function defaultSecuritySchema(): ?SecurityRequirement
{
    return SanctumAuth::make();
}
```

### Tags, Extensions & Tag Groups

Use `additionalOptions()` to define tags, vendor extensions, external docs, and tag groups:

```php
public function additionalOptions(): void
{
    $this->openApi->externalDocs(
        ExternalDocs::make('https://example.com/docs')
            ->description('Full developer documentation')
    );

    $this->openApi->tag(
        (new Tag())->name('Users')->description('User management operations')
    );
    $this->openApi->tag(
        (new Tag())->name('Posts')->description('Blog post operations')
    );

    // Vendor extensions
    $this->openApi->extension('x-api-id', 'my-api');

    // Tag groups (Redoc x-tagGroups extension)
    $this->openApi->extension('x-tagGroups', [
        new Group('Content', ['Posts', 'Comments']),
        new Group('Access', ['Users', 'Auth']),
    ]);
}
```

## Paths

### PathComponent (recommended)

Path components are auto-discovered from the `docs/Paths/` directory. Extend `PathComponent` and define HTTP methods as class methods:

```php
class PostPath extends PathComponent
{
    public function path(): Path
    {
        return $this->make('posts/{postId}')
            ->parameter(
                Parameter::make('postId', ParameterIn::PATH)
                    ->description('The post ID')
                    ->required()
                    ->example(42)
            );
    }

    public function get(Method $method): Method
    {
        return $method
            ->operationId('getPost')
            ->summary('Get a post by ID')
            ->tag('Posts')
            ->response(
                SuccessfulResponse::make()->content([
                    ApplicationJson::make(PostSchema::make()),
                ])
            );
    }

    public function delete(Method $method): Method
    {
        return $method
            ->operationId('deletePost')
            ->summary('Delete a post')
            ->tag('Posts')
            ->response(NoContentResponse::make());
    }
}
```

Supported HTTP methods: `get`, `post`, `put`, `patch`, `delete`, `head`, `options`. Only define the methods your endpoint supports — the rest are omitted automatically.

### AuthPathComponent

Extends `PathComponent` and automatically applies Sanctum security to all methods:

```php
class AuthPath extends AuthPathComponent
{
    public function path(): Path
    {
        return $this->make('auth/login');
    }

    public function post(Method $method): Method
    {
        return $method
            ->operationId('login')
            ->summary('Authenticate and receive a token')
            ->tag('Auth')
            ->jsonRequest([
                StringProperty::make('email')->required()->format('email'),
                StringProperty::make('password')->required()->format('password'),
            ])
            ->jsonResponse([
                StringProperty::make('token')->example('1|abc123def456'),
                StringProperty::make('token_type')->default('Bearer'),
            ]);
    }
}
```

### Route-based paths

Reference existing Laravel routes by name and extract request bodies from form request validation rules:

```php
Route::make('posts.store', 'Create a new post')
    ->tag('Posts')
    ->requestBodyFromRequestClass()
    ->successfulResponse(PostSchema::make())
    ->secure();
```

### Method options

Methods support the full OpenAPI operation spec:

```php
$method
    ->operationId('updatePost')
    ->summary('Update a post')
    ->description('Full description here')
    ->tag('Posts')                                      // or ->tag((new Tag())->name('Posts'))
    ->deprecated()
    ->externalDocs(ExternalDocs::make('https://...'))
    ->extension('x-sunset', '2025-12-31')
    ->security(ApiKeyAuth::make())                      // per-operation security override
    ->request($request)
    ->response($response);
```

## Properties

Properties describe individual fields within schemas and request/response bodies. All properties use `make(string $name)` and support method chaining.

| Class | Type | Extra methods |
|---|---|---|
| `StringProperty` | `string` | `minLength()`, `maxLength()` |
| `IntegerProperty` | `integer` | `minimum()`, `maximum()` |
| `NumberProperty` | `number` | `minimum()`, `maximum()` |
| `BooleanProperty` | `boolean` | — |
| `ArrayProperty` | `array` | `minItems()`, `maxItems()` |
| `ObjectProperty` | `object` | — |
| `DateTimeProperty` | `string` (format: `date-time`) | `minLength()`, `maxLength()` |
| `FileProperty` | `string` (format: `binary`) | `minLength()`, `maxLength()` |

Common methods available on all properties: `required()`, `description()`, `example()`, `enum()`, `default()`, `format()`, `title()`, `extension()`.

```php
// Examples
StringProperty::make('title')->required()->maxLength(255)->example('My Post'),
IntegerProperty::make('id')->example(42),
StringProperty::make('status')->enum(['draft', 'published', 'archived'])->default('draft'),
DateTimeProperty::make('created_at'),
ArrayProperty::make('tag_ids', IntegerSchema::make()),
```

## Schemas

Schemas define the structure of data objects and are used inside properties, request bodies, and responses.

### ObjectSchema

```php
ObjectSchema::make([
    IntegerProperty::make('id')->example(1),
    StringProperty::make('name')->required(),
    StringProperty::make('email')->format('email'),
]);
```

### ArraySchema

```php
// Array of primitives
ArraySchema::make(IntegerSchema::make())->minItems(1);

// Array of objects
ArraySchema::make(
    ObjectSchema::make([
        StringProperty::make('title'),
    ])
);
```

### Primitive Schemas

`StringSchema`, `IntegerSchema`, `NumberSchema`, `BooleanSchema` — used as items for `ArraySchema` or `ArrayProperty`.

## Reusable Components

Components are self-registering classes that output `$ref` references in the generated spec. Extend the appropriate base class and implement the `content()` method.

### SchemaComponent

```php
class PostSchema extends SchemaComponent
{
    public function content(): ObjectSchema
    {
        return $this->schema([
            IntegerProperty::make('id')->example(42),
            StringProperty::make('title')->required()->maxLength(255)->example('My First Post'),
            StringProperty::make('body')->required(),
            StringProperty::make('status')->enum(['draft', 'published', 'archived'])->default('draft'),
            IntegerProperty::make('author_id')->example(1),
            ArrayProperty::make('comment_ids', IntegerSchema::make()),
            DateTimeProperty::make('published_at'),
            DateTimeProperty::make('created_at'),
        ])->extension('x-model', 'App\\Models\\Post');
    }
}
```

Usage: `PostSchema::make()` — returns a `$ref` to `#/components/schemas/PostSchema`.

### ParameterComponent

```php
class PageParameter extends ParameterComponent
{
    public function content(): Parameter
    {
        return $this->parameter('page', ParameterIn::QUERY)
            ->description('Page number for pagination')
            ->example(1);
    }
}
```

Usage: `PageParameter::make()` — returns a `$ref` to `#/components/parameters/PageParameter`.

### RequestComponent

```php
class CreatePostRequest extends RequestComponent
{
    public function content(): Request
    {
        return $this->request()
            ->required()
            ->description('Data for creating a new blog post')
            ->content([
                ApplicationJson::make(
                    ObjectSchema::make([
                        StringProperty::make('title')->required()->maxLength(255),
                        StringProperty::make('body')->required(),
                        StringProperty::make('status')->enum(['draft', 'published'])->default('draft'),
                    ])
                ),
            ]);
    }
}
```

### ResponseComponent

```php
class UnauthorizedResponse extends ResponseComponent
{
    public function content(): Response
    {
        return $this->response(401)
            ->description('Unauthenticated')
            ->content([
                ApplicationJson::make(ErrorSchema::make()),
            ]);
    }
}
```

## Parameters

Parameters are created with a name and location (`ParameterIn` enum):

```php
use AutoDocumentation\Enums\ParameterIn;

// Inline parameter
Parameter::make('postId', ParameterIn::PATH)
    ->description('The post ID')
    ->required()
    ->example(42);

// Query parameter
Parameter::make('search', ParameterIn::QUERY)
    ->description('Search term')
    ->example('laravel');

// Header parameter
Parameter::make('Accept-Language', ParameterIn::HEADER)
    ->description('Preferred language')
    ->example('en-US');

// Cookie parameter
Parameter::make('session', ParameterIn::COOKIE)
    ->description('Session identifier');
```

Attach parameters to a path:

```php
$this->make('posts')
    ->parameter(PageParameter::make())
    ->parameter(PerPageParameter::make());
```

## Request Bodies

### JSON request (shorthand)

Use `jsonRequest()` on a method for a quick JSON body:

```php
$method->jsonRequest([
    StringProperty::make('title')->required()->maxLength(255),
    StringProperty::make('body')->required(),
]);
```

Or pass a schema directly:

```php
$method->jsonRequest(
    ObjectSchema::make([
        StringProperty::make('title')->required(),
    ])
);
```

### Multipart/form-data

For file uploads, use `MultipartFormData` content type:

```php
$method->request(
    Request::make()
        ->required()
        ->content([
            MultipartFormData::make(FileUploadSchema::make()),
        ])
);
```

### From Laravel validation rules

When using `Route::make()`, extract the request body directly from a form request class:

```php
Route::make('posts.store', 'Create a post')
    ->requestBodyFromRequestClass();
```

This inspects the controller method's type-hinted `Request` class and converts its `rules()` to an OpenAPI schema.

## Responses

### SuccessfulResponse

```php
SuccessfulResponse::make()          // 200
SuccessfulResponse::make(201)       // 201
    ->description('Post created')
    ->content([
        ApplicationJson::make(PostSchema::make()),
    ]);
```

### NoContentResponse

```php
NoContentResponse::make();          // 204
```

### JSON response (shorthand)

```php
$method->jsonResponse([
    StringProperty::make('token')->example('abc123'),
    StringProperty::make('token_type')->default('Bearer'),
]);
```

### Custom responses

```php
Response::make(422)
    ->name('ValidationError')
    ->description('Validation failed')
    ->content([
        ApplicationJson::make(ErrorSchema::make()),
    ]);
```

## Security

### Built-in: SanctumAuth

```php
use AutoDocumentation\Security\SanctumAuth;

// As global default
protected function defaultSecuritySchema(): ?SecurityRequirement
{
    return SanctumAuth::make();
}

// Per-operation
$method->security(SanctumAuth::make());
```

### HTTP Security Scheme

```php
HttpSecurityScheme::make('bearer')
    ->bearerFormat('JWT')
    ->description('JWT Bearer token');
```

### API Key

```php
use AutoDocumentation\Enums\ApiKeySecuritySchemeIn;

ApiKeySecurityScheme::make('X-API-Key', ApiKeySecuritySchemeIn::HEADER)
    ->description('API key passed in the X-API-Key header');
```

### OpenID Connect

```php
OpenIdConnectSecurityScheme::make('https://auth.example.com/.well-known/openid-configuration')
    ->description('OpenID Connect');
```

### Custom security component

Create a reusable security requirement:

```php
class ApiKeyAuth extends SecurityRequirement
{
    public function content(): SecurityScheme
    {
        return ApiKeySecurityScheme::make('X-API-Key', ApiKeySecuritySchemeIn::HEADER)
            ->description('API key passed in the X-API-Key header');
    }
}
```

Use it globally or per-operation:

```php
// Global
protected function defaultSecuritySchema(): ?SecurityRequirement
{
    return ApiKeyAuth::make();
}

// Per-operation override
$method->security(ApiKeyAuth::make());
```

## Configuration

Publish the config file with `php artisan vendor:publish --tag=auto-documentation-config`.

| Option | Default | Description |
|---|---|---|
| `generate_always` | `false` | Auto-regenerate the spec on every page load (dev only) |
| `environment` | `['local', 'development']` | Environments where documentation routes are registered |
| `routes.documentation` | `api/doc` | URL path for the Redoc UI |
| `routes.specification` | `api/doc/spec` | URL path for the raw OpenAPI YAML |
| `paths.source` | `base_path('docs')` | Directory containing your documentation classes |
| `paths.generated-doc` | `storage_path('app/auto-docs')` | Directory where the generated YAML is stored |

Environment variable `AUTO_DOCUMENTATION_GENERATE_ALWAYS` controls the `generate_always` option. `AUTO_DOCUMENTATION_SOURCE` controls the source docs path.

## Generation

### Artisan command

```bash
php artisan auto-doc:generate
```

Generates `documentation.yaml` from your `Documentation` class and all discovered path components.

### Auto-regeneration

Set `generate_always` to `true` (or `AUTO_DOCUMENTATION_GENERATE_ALWAYS=true` in `.env`) to regenerate the spec on every documentation page load. Useful during development.

### Routes

| Route | Description |
|---|---|
| `/api/doc` | Redoc documentation UI |
| `/api/doc/spec` | Raw OpenAPI YAML specification |

Routes are only registered in the environments listed in the `environment` config option.

## License

This library is licensed under the MIT License. See the [LICENSE](LICENSE) file for details.

## Credits

- Developer: Andriy Hrechyn
- Email: andriy.hrechyn@gmail.com
 readmeEtag: '"a4382285589b7192131211a25a5f80168396a6c6"' readmeLastModified: Sun, 01 Feb 2026 13:47:08 GMT repositoryId: 605599739 description: >- Auto Documentation is a library for generating API documentation for Laravel. created: '2023-02-23T14:00:16Z' updated: '2026-02-01T13:47:14Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: andrii-hrechyn logo: https://avatars.githubusercontent.com/u/38326555?v=4 license: MIT repoEtag: '"ae46d588497aadc70d2be44dc7b75e5a4f62edce7b197dc952da1904a8e8126d"' repoLastModified: Sun, 01 Feb 2026 13:47:14 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/usergamepreferences v3: true id: e8162c6421bf8dd56ababb8dca8ea0d3 repositoryMetadata: base64Readme: >- PiBbIUlNUE9SVEFOVF0gIAo+IFRoaXMgcmVwb3NpdG9yeSBpcyByZWFkLW9ubHkgLyBhcmNoaXZlZCBhbmQgd2lsbCBub3QgcmVjaWV2ZSB1cGRhdGVzLgoKIyBbVHlwZWRvYyBEb2N1bWVudGF0aW9uIFdlYnNpdGVdKGh0dHBzOi8vc3Vkb2t1cnUuZ2l0aHViLmlvL1VzZXJBY3RpdmVHYW1lcy8pPGJyPgoKCiMgRGV2ZWxvcGVyIFNldHVwCgoxLiBHZXQgdGhlIC5lbnYgZmlsZSBmcm9tIHRoZSBNU0IgYnVpbGRpbmcgbmV4dCB0byB0aGUgd2F0ZXIgZm91bnRhaW4uIAoyLiBJbnN0YWxsIERvY2tlciBvbiB5b3VyIG1hY2hpbmUuIFR1dG9yaWFsIGlzIGxpbmtlZCBiZWxvdzo8YnI+CiAgIFshW0RvY2tlciBUdXRvcmlhbF0oaHR0cHM6Ly9pbWcueW91dHViZS5jb20vdmkvMmV6TnFxYVNqcTgvMC5qcGcpXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PTJlek5xcWFTanE4KTxicj4KMi4gT25jZSBkb2NrZXIgaXMgaW5zdGFsbGVkLCB0aGUgTW9uZ28gaW1hZ2UgY2FuIGJlIHJ1biB3aXRoIHRoaXMgY29tbWFuZDo8YnI+Ck5vdGUgdXNlIGBgYHN1ZG9gYGAgb24gTGludXgvTWFjPGJyPgpgYGBjb25zb2xlCm5wbSBydW4gZG9ja2VyCmBgYAozLiBUaGUgYXBwIGNhbiB0aGVuIGJlIHJ1biB3aXRoIHRoZSBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHN0YXJ0CmBgYAo0LiBJbnRlZ3JhdGlvbiB0ZXN0cyBjYW4gYmUgcnVuIHdoZW4gdGhlIGFwcCBpcyBydW5uaW5nIHdpdGggdGhpcyBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHRlc3Q6aW50ZWdyYXRpb24KYGBgCg== readmeEtag: '"551645e2a891d8f25afe5c10c232d868e7f8455c"' readmeLastModified: Sun, 12 May 2024 16:01:41 GMT repositoryId: 609680639 description: null created: '2023-03-04T22:48:37Z' updated: '2024-05-12T16:03:54Z' language: TypeScript archived: true stars: 0 watchers: 0 forks: 0 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"7c25b7a58faaad12a3360b583a6ffbe54464606ce708ad2eeda54fec9f1ccdd0"' repoLastModified: Sun, 12 May 2024 16:03:54 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/liankui/openapi-cli v3: true id: 63cf9213dfed8a7350d6e635f45f254e repositoryMetadata: base64Readme: >- IyBvcGVuYXBpLWNsaQoKb3BlbmFwaS1jbGkg5piv5LiA5Liq5b+r6YCf5qOA5p+l44CB5Y2H57qnc3dhZ2dlci9vcGVuQVBJ5paH5qGj55qE5bCP5bel5YW344CCCgovLyBUT0RPIHAxIOa3u+WKoOWvuXN3YWdnZXIy5paH5Lu255qE5b6q546v5L6d6LWW44CB5Lqk5Y+J5L6d6LWW55qE5qOA5rWLCi8vIFRPRE8gcDEg5re75YqgcG9zdG1hbiBqc29u5paH5Lu26L2s5YyW5Li6b3BlbmFwaTMKCiMjIOS9v+eUqOaWueW8j++8mgoKYGBgClVTQUdFOgpvcGVuYXBpLWNsaSBbY29tbWFuZF0gW2NvbW1hbmQgb3B0aW9uc10gW2FyZ3VtZW50cy4uLl0KCkNPTU1BTkRTOgpsaW50LCBsICAgICBsaW50IHN3YWdnZXIvb3BlbmFwaSBkb2N1bWVudAp1cGdyYWRlLCB1ICB1cGdyYWRlIHN3YWdnZXIyIHRvIG9wZW5hcGkzCnZlcnNpb24sIHYgIFNob3cgdmVyc2lvbgpoZWxwLCBoICAgICBTaG93cyBhIGxpc3Qgb2YgY29tbWFuZHMgb3IgaGVscCBmb3Igb25lIGNvbW1hbmQKYGBgCgojIyMg5Z+656GA5ZG95LukCgoxLiBsaW50IDxmaWxlbmFtZT7vvIzmo4Dmn6VvcGVuQVBJ5Lit5pyJ6Zeu6aKY55qE6K+t5rOV6ZSZ6K+v77ybCiAgICBgYGDFkwogICAkIC4vb3BlbmFwaS1jbGktdjAuMi4zLWRhcndpbi1hbWQ2NCBsaW50IHRlc3RkYXRhL3N3YWdnZXIyLXdyb25nLmpzb24KICAgMjAyMy0wNi0xOVQxNzoxNTowNi41OTErMDgwMCAgICBpbmZvICAgIHBrZy9saW50LmdvOjM3ICBhcGkgbGludCAgICAgICAgeyJmaWxlIjogInRlc3RkYXRhL3N3YWdnZXIyLXdyb25nLmpzb24ifQogICAyMDIzLTA2LTE5VDE3OjE1OjA2LjYxMSswODAwICAgIGluZm8gICAgcGtnL2xpbnQuZ286NDggIHZpb2xhdGlvbiAgICAgICB7InJlc3VsdCI6IHsidmFsaWQiOmZhbHNlLCJwYXRoIjoiL3BvbGljeS9jbGFzc2lmeVNhbXBsZS9nZXRMaXN0IiwibWV0aG9kIjoicG9zdCIsInN0YXJ0TGluZSI6MjYsImVuZExpbmUiOjMxLCJkZXNjcmlwdGlvbiI6Ik9wZXJhdGlvbiBwYXJhbWV0ZXJzIGFyZSB1bmlxdWUgYW5kIG5vbi1yZXBlYXRpbmcuIiwiaG93VG9GaXgiOiJNYWtlIHN1cmUgdGhhdCBhbGwgdGhlIG9wZXJhdGlvbiBwYXJhbWV0ZXJzIGFyZSB1bmlxdWUgYW5kIG5vbi1yZXBlYXRpbmcsIGRvbid0IGR1cGxpY2F0ZSBuYW1lcywgZG9uJ3RyZS11c2UgcGFyYW1ldGVyIG5hbWVzIGluIHRoZSBzYW1lIG9wZXJhdGlvbi4ifX0KICAgMjAyMy0wNi0xOVQxNzoxNTowNi42MTErMDgwMCAgICBpbmZvICAgIHBrZy9saW50LmdvOjUyICBhcGkgbGludCBmaW5pc2hlZCAgICAgICB7ImZpbGUiOiAidGVzdGRhdGEvc3dhZ2dlcjItd3JvbmcuanNvbiJ9CiAgIGBgYAoyLiB1cGdyYWRlIDxmaWxlbmFtZT7vvIzljYfnuqdzd2FnZ2VyMuWIsG9wZW5BUEkz77yI55Sf5oiQ5Li65LiA5Liq5pe26Ze05oiz5ZCO57yA55qEanNvbuaWh+S7tu+8ieOAggogICBgYGAKICAgJCAuL29wZW5hcGktY2xpLXYwLjIuMy1kYXJ3aW4tYW1kNjQgdXBncmFkZSB0ZXN0ZGF0YS9zd2FnZ2VyMi13cm9uZy5qc29uCiAgIDIwMjMtMDYtMTlUMTc6MTg6MzMuMzMzKzA4MDAgICAgaW5mbyAgICBwa2cvb3BlbmFwaTIuZ286NjUgICAgICBhcGkgdXBncmFkZSAgICAgeyJmaWxlIjogInRlc3RkYXRhL3N3YWdnZXIyLXdyb25nLmpzb24ifQogICAyMDIzLTA2LTE5VDE3OjE4OjMzLjM0NyswODAwICAgIGluZm8gICAgcGtnL29wZW5hcGkyLmdvOjE0NCAgICAgZGVsZXRlIGludmFsaWQgb3BlcmF0aW9uICAgICAgICB7Im9wZXJhdGlvbiI6IHsidmFsaWQiOmZhbHNlLCJwYXRoIjoiL3BvbGljeS9jbGFzc2lmeVNhbXBsZS9nZXRMaXN0IiwibWV0aG9kIjoicG9zdCIsInN0YXJ0TGluZSI6MSwiZW5kTGluZSI6MSwiZGVzY3JpcHRpb24iOiJPcGVyYXRpb24gcGFyYW1ldGVycyBhcmUgdW5pcXVlIGFuZCBub24tcmVwZWF0aW5nLiIsImhvd1RvRml4IjoiTWFrZSBzdXJlIHRoYXQgYWxsIHRoZSBvcGVyYXRpb24gcGFyYW1ldGVycyBhcmUgdW5pcXVlIGFuZCBub24tcmVwZWF0aW5nLCBkb24ndCBkdXBsaWNhdGUgbmFtZXMsIGRvbid0cmUtdXNlIHBhcmFtZXRlciBuYW1lcyBpbiB0aGUgc2FtZSBvcGVyYXRpb24uIn19CiAgIDIwMjMtMDYtMTlUMTc6MTg6MzMuMzQ3KzA4MDAgICAgaW5mbyAgICBwa2cvb3BlbmFwaTIuZ286OTYgICAgICBhcGkgdXBncmFkZSBzdWNjZXNzZnVsbHkgICAgICAgIHsiZmlsZSI6ICJ0ZXN0ZGF0YS9zd2FnZ2VyMi13cm9uZy0xNjg3MTY2MzEzMzQ3Lmpzb24iLCAidmVyc2lvbiI6ICIzLjAuMyIsICJkdXJhdGlvbiI6ICIxNC4yMzUwODNtcyJ9CiAgIGBgYAoKcmVmOgpodHRwczovL2dpdGh1Yi5jb20vdXJmYXZlL2NsaQpodHRwczovL2dpdGh1Yi5jb20vTHVjeUJvdC1JbmMvYXBpLXNwZWMtY29udmVydGVyCmh0dHBzOi8vZ2l0aHViLmNvbS9nZXRraW4va2luLW9wZW5hcGkK readmeEtag: '"93048ffbb2d2620fbb67e9b5a61d9e85b32ba6f7"' readmeLastModified: Mon, 22 Jan 2024 06:12:38 GMT repositoryId: 631874442 description: 一个快速检查、升级swagger/openAPI文档的小工具。 created: '2023-04-24T08:42:28Z' updated: '2023-06-26T05:41:44Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: liankui logo: https://avatars.githubusercontent.com/u/26359513?v=4 license: MIT repoEtag: '"41bf401779427925ae82e8e04830f0e4ea55ac04230cca2c47c8bdb7c431a818"' repoLastModified: Mon, 26 Jun 2023 05:41:44 GMT category: Parsers foundInMaster: true - source: https://openapi.tools/ name: OpenAPI Explorer category: - Documentation - Parsers source_description: >- Generate and render fully customizable API documentation, then explore and execute API requests via the integrated console. link: https://github.com/Rhosys/openapi-explorer/blob/main/README.md repository: https://github.com/authress-engineering/openapi-explorer language: - Javascript - Custom Element v2: true v3: true v3_1: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIEV4cGxvcmVyCldlYiBDb21wb25lbnQgQ3VzdG9tIEVsZW1lbnQgZm9yIE9wZW4tQVBJIHNwZWMgdmlld2luZywgd2l0aCBhdXRvbWF0aWMgaW50ZWdyYXRpb24gZm9yIFJlYWN0IGFuZCBWdWUuCgpUaGlzIGlzIGFuIG9wZW4gc291cmNlIHByb2plY3QgbWFuYWdlZCBieSB0aGUgW0F1dGhyZXNzIEVuZ2luZWVyaW5nIHRlYW1dKGh0dHBzOi8vYXV0aHJlc3MuaW8pLgoKPHAgYWxpZ249ImNlbnRlciI+CiAgICA8YSBocmVmPSJodHRwczovL2F1dGhyZXNzLmlvIiBhbHQ9IkF1dGhyZXNzIEVuZ2luZWVyaW5nIj4KICAgICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vc3RhdGljL3YxP2xhYmVsPUF1dGhyZXNzK0VuZ2luZWVyaW5nJm1lc3NhZ2U9T3BlbkFQSSUyMEV4cGxvcmVyJmNvbG9yPSUyM0ZCQUYwQiZsb2dvPWFuZHJvaWRhdXRvJmxvZ29Db2xvcj0lMjNGQkFGMEIiPjwvYT4KICAgIDxhIGhyZWY9Ii4vTElDRU5TRSIgYWx0PSJhcGFjaGUgMi4wIGxpY2Vuc2UiPgogICAgICA8aW1nIHNyYz0iaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9saWNlbnNlLUFwYWNoZSUyMDIuMC1ibHVlLnN2ZyI+PC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly9iYWRnZS5mdXJ5LmlvL2pzL29wZW5hcGktZXhwbG9yZXIiIGFsdD0ibnBtIHZlcnNpb24iPgogICAgICAgIDxpbWcgc3JjPSJodHRwczovL2JhZGdlLmZ1cnkuaW8vanMvb3BlbmFwaS1leHBsb3Jlci5zdmciPjwvYT4KICAgIDxhIGhyZWY9Imh0dHBzOi8vYXV0aHJlc3MuaW8vY29tbXVuaXR5IiBhbHQ9Im5wbSB2ZXJzaW9uIj4KICAgICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvY29tbXVuaXR5LURpc2NvcmQtcHVycGxlLnN2ZyI+PC9hPgogICAgPGEgaHJlZj0iaHR0cHM6Ly93d3cud2ViY29tcG9uZW50cy5vcmcvZWxlbWVudC9vcGVuYXBpLWV4cGxvcmVyIiBhbHQ9InB1Ymxpc2hlZCBvbiB3ZWJjb21wb25lbnRzLm9yZyI+CiAgICAgICAgPGltZyBzcmM9Imh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2Uvd2ViY29tcG9uZW50cy5vcmctT3BlbkFQSSUyMEV4cGxvcmVyLWJsdWUuc3ZnP3N0eWxlPXNvY2lhbCI+PC9hPgo8L3A+CgoKIyMgQmVhdXRpZnVsIGFuZCByZXNwb25zaXZlIEFQSSBleHBsb3JlciBhbmQgY29uc29sZQoKPHA+CiAgPGEgaHJlZj0iaHR0cHM6Ly9hdXRocmVzcy1lbmdpbmVlcmluZy5naXRodWIuaW8vb3BlbmFwaS1leHBsb3Jlci8jP3JvdXRlPWdldC0vdjEvdXNlcnMvLXVzZXJJZC0vcmVzb3VyY2VzLy1yZXNvdXJjZVVyaS0vcGVybWlzc2lvbnMvLXBlcm1pc3Npb24tIiB0YXJnZXQ9Il9ibGFuayI+CiAgICA8aW1nIHNyYz0iLi9kb2NzL2Rlc2t0b3Atdmlldy5wbmciIGFsdD0iRGVza3RvcCBkZW1vIGltYWdlIiB3aWR0aD0iODAwcHgiPgogIDwvYT4KPC9wPgoKIyMgQ2hlY2sgb3V0IHRoZSBEZW1vCltPcGVuQVBJIEV4cGxvcmVyIERlbW9dKGh0dHBzOi8vYXV0aHJlc3MtZW5naW5lZXJpbmcuZ2l0aHViLmlvL29wZW5hcGktZXhwbG9yZXIvIz9yb3V0ZT1nZXQtL3YxL3VzZXJzLy11c2VySWQtL3Jlc291cmNlcy8tcmVzb3VyY2VVcmktL3Blcm1pc3Npb25zLy1wZXJtaXNzaW9uLSkKCihDdXJpb3VzIGFib3V0IHRoZSBbZXhhY3Qgc3R5bGluZyBvZiB0aGUgZGVtb10oLi9kb2NzL2F1dGhyZXNzLWV4YW1wbGUudnVlKSkKCiMjIEdldCBzdGFydGVkIG5vdwpgbnBtIGluc3RhbGwgb3BlbmFwaS1leHBsb3JlcmAKCiMjIyBRdWljayBzdGFydCBleGFtcGxlCiogYGltcG9ydCAnb3BlbmFwaS1leHBsb3Jlcic7YAoKYGBgaHRtbAo8b3BlbmFwaS1leHBsb3JlciA6c3BlYy11cmw9Im9wZW5hcGlTcGVjaWZpY2F0aW9uVXJsIj4KICA8ZGl2IHNsb3Q9Im92ZXJ2aWV3Ij4KICAgIDxoMT5UaGUgQVBJPC9oMT4KICA8L2Rpdj4KPC9vcGVuYXBpLWV4cGxvcmVyPgpgYGAKCiMjIEZlYXR1cmVzCi0gT3BlbkFQSSAzLjIrCi0gQnVpbHQgaW4gYXV0b21hdGljIEludGVybmF0aW9uYWxpemF0aW9uIChJMThuKQotIFdvcmtzIHdpdGggYW55IGZyYW1ld29yawotIFZpZXcgcmVzb3VyY2VzLCBtb2RlbHMsIGFuZCBkaXJlY3RseSBtYWtlIEFQSSBjYWxscwotIEJldHRlciBVc2FiaWxpdHksIAogIC0gUmVxdWVzdCBmaWVsZHMgYXJlIHByZS1wb3B1bGF0ZWQgd2l0aCBkZWZhdWx0IGRhdGEKICAtIFRha2VzIG9ubHkgb25lIGNsaWNrIHRvIG1ha2UgYW4gQVBJIGNhbGwKICAtIFJlbmRlcnMgU0RLL2NsaWVudCBjb2RlIHNhbXBsZXMKICAtIEJyYW5kaW5nIGFuZCBQZXJzb25hbGl6YXRpb24gZmVhdHVyZXMgbWFrZXMgaXQgZWFzeSB0byBmb2xsb3cgYW55IHN0eWxlIGd1aWRlCiAgLSBGdWxseSBjdXN0b21pemFibGUgdGhlbWUKLSBQbGVudHkgb2YgY3VzdG9taXphdGlvbiBvcHRpb25zIAogIC0gQWRkIGV4dGVybmFsIGNvbnRlbnRzIHRocm91Z2hvdXQgdGhlIGNvbXBvbmVudCwgZXh0ZW5zaWJsZSB3aXRoIG1hcmtkb3duLCBpbWFnZXMsIGxpbmtzLCBhbmQgdGV4dAogIC0gQWxsIHByb3BlcnRpZXMgYXJlIHJlYWN0aXZlCiAgLSBTdHlsZSB0aGUgZWxlbWVudCB3aXRoIHN0YW5kYXJkIGNzcyAoY2hhbmdlIHBhZGRpbmcsIHBvc2l0aW9uLCBib3JkZXIsIG1hcmdpbiApCiAgLSBTdHlsZXMgdGhhdCByZWZsZWN0IHlvdXIgc2l0ZSBhbmQgeW91ciBVSS9VWCBmcmFtZXdvcmtzIChSZWFjdCwgVnVlLCB2YW5pbGxhIGpzLCBCb290c3RyYXAsIE1hdGVyaWFsLCBhbmQgbWFueSBtb3JlLi4uKQotIFJlc3BvbnNpdmUgc28gaXQgd29ya3Mgb24gbW9iaWxlCi0gTGlnaHR3ZWlnaHQgYW5kIGZhc3QKCgojIyBEb2N1bWVudGF0aW9uCiogW01pZ3JhdGluZyBmcm9tIHYxIHRvIHYyIG9mIG9wZW5hcGktZXhwbG9yZXJdKC4vQ0hBTkdFTE9HLm1kIzIuMCkgLSBSZXZpZXcgdGhlIGJyZWFraW5nIGNoYW5nZXMKKiBbUHJvcGVydHkgYW5kIHZhcmlhYmxlcyBkb2N1bWVudGF0aW9uXSguL2RvY3MvZG9jdW1lbnRhdGlvbi5tZCkKKiBbRXhhbXBsZXMgKFZ1ZSwgUmVhY3QsIEpTLCBhbmQgbW9yZSldKC4vZG9jcy9leGFtcGxlcy5tZCkKKiBHZW5lcmF0ZSB0aGUgb3BlbiBzcGVjaWZpY2F0aW9uIGRvY3VtZW50IG5lY2Vzc2FyeSBmb3IgdGhpcyBsaWJyYXJ5IC0gYnkgdXNpbmcgYW4gZWRpdG9yIG9yIGJ5IGZvbGxvd2luZyB0aGUgW09wZW4gQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vZ2l0aHViLmNvbS9PQUkvT3BlbkFQSS1TcGVjaWZpY2F0aW9uL2Jsb2IvbWFpbi92ZXJzaW9ucy8zLjIuMC5tZCkKKiBbU3R5bGluZyB5b3VyIG9wZW5hcGktZXhwbG9yZXIgVUldKC4vZG9jcy9zdHlsaW5nLm1kKQoqIFtSZWNlbnQgY2hhbmdlcyBhbmQgdXBkYXRlc10oLi9DSEFOR0VMT0cubWQpCgojIyBUcm91Ymxlc2hvb3RpbmcgaW50ZWdyYXRpb24gaXNzdWVzCltDb21tb24gaXNzdWVzXSguL2RvY3MvdHJvdWJsZXNob290aW5nLm1kKQoKCiMjIENvbnRyaWJ1dGlvbgpbQ29udHJpYnV0aW9ucyBHdWlkZV0oLi9DT05UUklCVVRJTkcubWQpCgpgYGBiYXNoCiMgQ2xvbmUgLyBEb3dubG9hZCB0aGUgcHJvamVjdCB0aGVuCmdpdCBjbG9uZQoKIyMgUHVsbCBpbiBkZXBlbmRlbmNpZXMKeWFybgoKIyBidWlsZCB3aWxsIGdlbmVyYXRlIGRpc3Qvb3BlbmFwaS1leHBsb3Jlci5taW4uanMKeWFybiBidWlsZCAKCmltcG9ydCAnb3BlbmFwaS1leHBsb3Jlcic7CmBgYAoKCiMjIENvcHlyaWdodApDb3B5cmlnaHQgUmhvc3lzIEFHCgpMaWNlbnNlZCB1bmRlciB0aGUgQXBhY2hlIExpY2Vuc2UsIFZlcnNpb24gMi4wICh0aGUgIkxpY2Vuc2UiKTsKeW91IG1heSBub3QgdXNlIHRoaXMgcmVwb3NpdG9yeSBzb3VyY2UgZXhjZXB0IGluIGNvbXBsaWFuY2Ugd2l0aCB0aGUgTGljZW5zZS4KWW91IG1heSBvYnRhaW4gYSBjb3B5IG9mIHRoZSBMaWNlbnNlIGF0CgogIGh0dHA6Ly93d3cuYXBhY2hlLm9yZy9saWNlbnNlcy9MSUNFTlNFLTIuMAoKVW5sZXNzIHJlcXVpcmVkIGJ5IGFwcGxpY2FibGUgbGF3IG9yIGFncmVlZCB0byBpbiB3cml0aW5nLCBzb2Z0d2FyZQpkaXN0cmlidXRlZCB1bmRlciB0aGUgTGljZW5zZSBpcyBkaXN0cmlidXRlZCBvbiBhbiAiQVMgSVMiIEJBU0lTLApXSVRIT1VUIFdBUlJBTlRJRVMgT1IgQ09ORElUSU9OUyBPRiBBTlkgS0lORCwgZWl0aGVyIGV4cHJlc3Mgb3IgaW1wbGllZC4KU2VlIHRoZSBMaWNlbnNlIGZvciB0aGUgc3BlY2lmaWMgbGFuZ3VhZ2UgZ292ZXJuaW5nIHBlcm1pc3Npb25zIGFuZApsaW1pdGF0aW9ucyB1bmRlciB0aGUgTGljZW5zZS4K readmeEtag: '"46cb8846a311b48a47d03f99e8f96aa3bd376ca9"' readmeLastModified: Tue, 14 Oct 2025 20:43:25 GMT repositoryId: 351849045 description: OpenAPI Web component to generate a UI from the spec. created: '2021-03-26T16:42:43Z' updated: '2026-02-01T08:40:58Z' language: JavaScript archived: false stars: 343 watchers: 6 forks: 49 owner: Authress-Engineering logo: https://avatars.githubusercontent.com/u/35577654?v=4 license: Apache-2.0 repoEtag: '"60074c65bbaeb4963a43b67c02dfecd6b3da12c0a60e17299a8f8fd765dcb812"' repoLastModified: Sun, 01 Feb 2026 08:40:58 GMT foundInMaster: true id: c8fb58ca4a515c7e0b381f4e63d7c355 oldLocations: - https://github.com/rhosys/openapi-explorer - source: https://openapi.tools/ name: openapi-backend homepage: https://github.com/anttiviljami/openapi-backend language: Node.js + Typescript source_description: >- Build, Validate, Route, and Mock using OpenAPI specification. Framework-agnostic category: Server link: https://www.npmjs.com/package/openapi-backend repository: https://github.com/openapistack/openapi-backend v2: false v3: true v3_1: true repositoryMetadata: base64Readme: >- <span id="npm--fix-hidden-readme-header"></span>

<h1 align="center"><img alt="openapi-backend" src="https://github.com/openapistack/openapi-backend/raw/main/header.png" style="max-width:50rem"></h1>

[![CI](https://github.com/openapistack/openapi-backend/workflows/CI/badge.svg)](https://github.com/openapistack/openapi-backend/actions?query=workflow%3ACI)
[![License](http://img.shields.io/:license-mit-blue.svg)](https://github.com/openapistack/openapi-backend/blob/main/LICENSE)
[![npm version](https://img.shields.io/npm/v/openapi-backend.svg)](https://www.npmjs.com/package/openapi-backend)
[![npm downloads](https://img.shields.io/npm/dw/openapi-backend.svg)](https://www.npmjs.com/package/openapi-backend)
[![Libraries.io dependency status for latest release](https://img.shields.io/librariesio/release/npm/openapi-backend.svg)](https://www.npmjs.com/package/openapi-backend?activeTab=dependencies)
![npm type definitions](https://img.shields.io/npm/types/openapi-backend.svg)
[![Buy me a coffee](https://img.shields.io/badge/donate-buy%20me%20a%20coffee-orange)](https://buymeacoff.ee/anttiviljami)

<p align="center"><b>Build, Validate, Route, Authenticate, and Mock using OpenAPI definitions.</b></p>

<p align="center">OpenAPI Backend is a Framework-agnostic middleware tool for building beautiful APIs with <a href="https://github.com/OAI/OpenAPI-Specification">OpenAPI Specification</a>.</p>

## Features

- [x] Build APIs by describing them in [OpenAPI specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md)
- [x] Register handlers for [operationIds](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#fixed-fields-8)
to route requests in your favourite Node.js backend
- [x] Use [JSON Schema](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#data-types) to validate
API requests and/or responses. OpenAPI Backend uses the [AJV](https://ajv.js.org/) library under the hood for performant validation
- [x] Register Auth / Security Handlers for [OpenAPI Security Schemes](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#securitySchemeObject)
to authorize API requests
- [x] Auto-mock API responses using [OpenAPI examples objects](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#example-object)
or [JSON Schema definitions](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.2.md#schema-object)
- [x] Built with TypeScript, types included
- [x] Optimised runtime routing and validation. **No generated code!**
- [x] OpenAPI 3.1 support

## Documentation

**New!** OpenAPI Backend documentation is now found on [openapistack.co](https://openapistack.co)

https://openapistack.co/docs/openapi-backend/intro

## Quick Start

Full [example projects](https://github.com/openapistack/openapi-backend/tree/main/examples) included in the repo

```
npm install --save openapi-backend
```

```javascript
import OpenAPIBackend from 'openapi-backend';

// create api with your definition file or object
const api = new OpenAPIBackend({ definition: './petstore.yml' });

// register your framework specific request handlers here
api.register({
  getPets: (c, req, res) => res.status(200).json({ result: 'ok' }),
  getPetById: (c, req, res) => res.status(200).json({ result: 'ok' }),
  validationFail: (c, req, res) => res.status(400).json({ err: c.validation.errors }),
  notFound: (c, req, res) => res.status(404).json({ err: 'not found' }),
});

// initalize the backend
api.init();
```

### Express

```javascript
import express from 'express';

const app = express();
app.use(express.json());
app.use((req, res) => api.handleRequest(req, req, res));
app.listen(9000);
```

[See full Express example](https://github.com/openapistack/openapi-backend/tree/main/examples/express)

[See full Express TypeScript example](https://github.com/openapistack/openapi-backend/tree/main/examples/express-typescript)

### AWS Serverless (Lambda)

```javascript
// API Gateway Proxy handler
module.exports.handler = (event, context) =>
  api.handleRequest(
    {
      method: event.httpMethod,
      path: event.path,
      query: event.queryStringParameters,
      body: event.body,
      headers: event.headers,
    },
    event,
    context,
  );
```

[See full AWS SAM example](https://github.com/openapistack/openapi-backend/tree/main/examples/aws-sam)

[See full AWS CDK example](https://github.com/openapistack/openapi-backend/tree/main/examples/aws-cdk)

[See full SST example](https://github.com/openapistack/openapi-backend/tree/main/examples/aws-sst)

[See full Serverless Framework example](https://github.com/openapistack/openapi-backend/tree/main/examples/serverless-framework)

### Azure Function

```javascript
module.exports = (context, req) =>
  api.handleRequest(
    {
      method: req.method,
      path: req.params.path,
      query: req.query,
      body: req.body,
      headers: req.headers,
    },
    context,
    req,
  );
```

[See full Azure Function example](https://github.com/openapistack/openapi-backend/tree/main/examples/azure-function)

### Fastify

```ts
import fastify from 'fastify';

fastify.route({
  method: ['GET', 'POST', 'PUT', 'PATCH', 'DELETE'],
  url: '/*',
  handler: async (request, reply) =>
    api.handleRequest(
      {
        method: request.method,
        path: request.url,
        body: request.body,
        query: request.query,
        headers: request.headers,
      },
      request,
      reply,
    ),
});
fastify.listen();
```

[See full Fastify example](https://github.com/openapistack/openapi-backend/tree/main/examples/fastify)

### Hapi

```javascript
import Hapi from '@hapi/hapi';

const server = new Hapi.Server({ host: '0.0.0.0', port: 9000 });
server.route({
  method: ['GET', 'POST', 'PUT', 'PATCH', 'DELETE'],
  path: '/{path*}',
  handler: (req, h) =>
    api.handleRequest(
      {
        method: req.method,
        path: req.path,
        body: req.payload,
        query: req.query,
        headers: req.headers,
      },
      req,
      h,
    ),
});
server.start();
```

[See full Hapi example](https://github.com/openapistack/openapi-backend/tree/main/examples/hapi-typescript)


### Koa

```javascript
import Koa from 'koa';
import bodyparser from 'koa-bodyparser';

const app = new Koa();

app.use(bodyparser());
app.use((ctx) =>
  api.handleRequest(
    ctx.request,
    ctx,
  ),
);
app.listen(9000);
```

[See full Koa example](https://github.com/openapistack/openapi-backend/tree/main/examples/koa)

## Registering Handlers for Operations

Handlers are registered for [`operationIds`](https://github.com/OAI/OpenAPI-Specification/blob/main/versions/3.0.2.md#fixed-fields-8)
found in the OpenAPI definitions. You can register handlers as shown above with [`new OpenAPIBackend()`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#parameter-optshandlers)
constructor opts, or using the [`register()`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#registeroperationid-handler)
method.

```javascript
async function getPetByIdHandler(c, req, res) {
  const id = c.request.params.id;
  const pet = await pets.getPetById(id);
  return res.status(200).json({ result: pet });
}
api.register('getPetById', getPetByIdHandler);
// or
api.register({
  getPetById: getPetByIdHandler,
});
```

Operation handlers are passed a special [Context object](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#context-object)
as the first argument, which contains the parsed request, the
matched API operation and input validation results. The other arguments in the example above are Express-specific
handler arguments.

## Request validation

The easiest way to enable request validation in your API is to register a [`validationFail`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#validationfail-handler)
handler.

```javascript
function validationFailHandler(c, req, res) {
  return res.status(400).json({ status: 400, err: c.validation.errors });
}
api.register('validationFail', validationFailHandler);
```

Once registered, this handler gets called if any JSON Schemas in either operation parameters (in: path, query, header,
cookie) or requestPayload don't match the request.

The context object `c` gets a `validation` property with the [validation result](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#validationresult-object).

## Response validation

OpenAPIBackend doesn't automatically perform response validation for your handlers, but you can register a
[`postResponseHandler`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#postresponsehandler-handler)
to add a response validation step using [`validateResponse`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#validateresponseres-operation).

```javascript
api.register({
  getPets: (c) => {
    // when a postResponseHandler is registered, your operation handlers' return value gets passed to context.response
    return [{ id: 1, name: 'Garfield' }];
  },
  postResponseHandler: (c, req, res) => {
    const valid = c.api.validateResponse(c.response, c.operation);
    if (valid.errors) {
      // response validation failed
      return res.status(502).json({ status: 502, err: valid.errors });
    }
    return res.status(200).json(c.response);
  },
});
```

It's also possible to validate the response headers using [`validateResponseHeaders`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#validateresponseheadersheaders-operation-opts).

```javascript
api.register({
  getPets: (c) => {
    // when a postResponseHandler is registered, your operation handlers' return value gets passed to context.response
    return [{ id: 1, name: 'Garfield' }];
  },
  postResponseHandler: (c, req, res) => {
    const valid = c.api.validateResponseHeaders(res.headers, c.operation, {
      statusCode: res.statusCode,
      setMatchType: 'exact',
    });
    if (valid.errors) {
      // response validation failed
      return res.status(502).json({ status: 502, err: valid.errors });
    }
    return res.status(200).json(c.response);
  },
});
```

## Auth / Security Handlers

If your OpenAPI definition contains [Security Schemes](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#securitySchemeObject)
you can register security handlers to handle authorization for your API:

```yaml
components:
  securitySchemes:
  - ApiKey:
      type: apiKey
      in: header
      name: x-api-key
security:
  - ApiKey: []
```

```javascript
api.registerSecurityHandler('ApiKey', (c) => {
  const authorized = c.request.headers['x-api-key'] === 'SuperSecretPassword123';
  // truthy return values are interpreted as auth success
  // you can also add any auth information to the return value
  return authorized;
});
```

The authorization status and return values of each security handler can be
accessed via the [Context Object](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#context-object)

You can also register an [`unauthorizedHandler`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#unauthorizedhandler-handler)
to handle unauthorized requests.

```javascript
api.register('unauthorizedHandler', (c, req, res) => {
  return res.status(401).json({ err: 'unauthorized' })
});
```

See examples:
- [API Key auth (express)](https://github.com/openapistack/openapi-backend/tree/main/examples/express-apikey-auth)
- [JWT auth (express)](https://github.com/openapistack/openapi-backend/tree/main/examples/express-jwt-auth)

## Mocking API responses

Mocking APIs just got really easy with OpenAPI Backend! Register a [`notImplemented`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md#notimplemented-handler)
handler and use [`mockResponseForOperation()`](https://github.com/openapistack/openapi-backend/blob/main/DOCS.md##mockresponseforoperationoperationid-opts)
to generate mock responses for operations with no custom handlers specified yet:

```javascript
api.register('notImplemented', (c, req, res) => {
  const { status, mock } = c.api.mockResponseForOperation(c.operation.operationId);
  return res.status(status).json(mock);
});
```

OpenAPI Backend supports mocking responses using both OpenAPI example objects and JSON Schema:
```yaml
paths:
  '/pets':
    get:
      operationId: getPets
      summary: List pets
      responses:
        200:
          $ref: '#/components/responses/PetListWithExample'
  '/pets/{id}':
    get:
      operationId: getPetById
      summary: Get pet by its id
      responses:
        200:
          $ref: '#/components/responses/PetResponseWithSchema'
components:
  responses:
    PetListWithExample:
      description: List of pets
      content:
        'application/json':
          example:
            - id: 1
              name: Garfield
            - id: 2
              name: Odie
    PetResponseWithSchema:
      description: A single pet
      content:
        'application/json':
          schema:
            type: object
            properties:
              id:
                type: integer
                minimum: 1
              name:
                type: string
                example: Garfield
```

The example above will yield:
```javascript
api.mockResponseForOperation('getPets'); // => { status: 200, mock: [{ id: 1, name: 'Garfield' }, { id: 2, name: 'Odie' }]}
api.mockResponseForOperation('getPetById'); // => { status: 200, mock: { id: 1, name: 'Garfield' }}
```

[See full Mock API example on Express](https://github.com/openapistack/openapi-backend/tree/main/examples/express-ts-mock)

## Commercial support

For assistance with integrating openapi-backend in your company, reach out at support@openapistack.co.

## Contributing

OpenAPI Backend is Free and Open Source Software. Issues and pull requests are more than welcome!

 readmeEtag: '"df55388a5f5a2c9104f126870350480ffe5ee948"' readmeLastModified: Tue, 05 Mar 2024 06:58:46 GMT repositoryId: 154916760 description: Build, Validate, Route, Authenticate and Mock using OpenAPI created: '2018-10-27T02:31:25Z' updated: '2026-01-27T08:04:57Z' language: TypeScript archived: false stars: 675 watchers: 4 forks: 95 owner: openapistack logo: https://avatars.githubusercontent.com/u/147298423?v=4 license: MIT repoEtag: '"5e4121e8b46e19910c14400edf2bcee9fe2de6c7d7782c9db164c1bd4edd7080"' repoLastModified: Tue, 27 Jan 2026 08:04:57 GMT foundInMaster: true id: 13cdf61fc6370ba6543e5a21256c2dfb oldLocations: - https://github.com/anttiviljami/openapi-backend - source: openapi3 tags repository: https://github.com/gruppone/stalker-server v3: true repositoryMetadata: base64Readme: >- IyBTdGFsa2VyIC0gU2VydmVyCgpbIVtRdWFsaXR5IEdhdGUgU3RhdHVzXShodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD1HcnVwcE9uZV9zdGFsa2VyLXNlcnZlciZtZXRyaWM9YWxlcnRfc3RhdHVzKV0oaHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD1HcnVwcE9uZV9zdGFsa2VyLXNlcnZlcikKCjwhLS0gcGVyIGNvbnRyb2xsYXJlIGNoZSBpbCBkYiBzaWEgdXA6IGVjaG8gWCB8IHRlbG5ldCAtZSBYIGxvY2FsaG9zdCAzMzA2IC0tPgoKPCEtLSBUT0RPIHNjcml2ZXJlIHJlYWRtZSAtLT4K readmeEtag: '"848fe32bae379a3bd6854b73cf0309cf49984a34"' readmeLastModified: Wed, 17 Jun 2020 22:55:53 GMT repositoryId: 245448066 description: >- Server per il capitolato C5 - Stalker del corso di Ingegneria del Software 2019/2020 @ UniPD created: '2020-03-06T15:01:47Z' updated: '2020-06-17T22:55:58Z' language: Java archived: false stars: 0 watchers: 2 forks: 0 owner: GruppOne logo: https://avatars.githubusercontent.com/u/58079551?v=4 license: GPL-3.0 repoEtag: '"a2123ab94d46c3be62df32c8ff545cacd883248802f383fb97f2cc814d8ab0ea"' repoLastModified: Wed, 17 Jun 2020 22:55:58 GMT foundInMaster: true category: Server id: 55e2df1b151892ca13b3766635d427ec - source: openapi3 tags repository: https://github.com/anupama-sinha/family-asset-management v3: true repositoryMetadata: base64Readme: >- IyMjIEdldCBTdGFydGVkClRoaXMgaXMgYSBzaW1wbGUgU3ByaW5nIEJvb3QgcHJvamVjdCB3aGljaCBkaXNjcmVldGx5IHVzZXMgYmVsb3cgZmVhdHVyZXMuIFdpbGwga2VlcCBlbmhhbmNpbmcgdGhpcyBwcm9qZWN0IGZyb20gbGVhcm5pbmcgYW5kIHByYWN0aWNlIHBlcnNwZWN0aXZlLgoKIyMjIE9wZW5BcGkzIFN3YWdnZXIgRG9jdW1lbnRhdGlvbiAgCiogQWRkIGRlcGVuZGVuY3kgYW5kIEBPcGVuQVBJRGVmaW5pdGlvbiBpbiBtYWluIGNsYXNzLgoqIFRoZW4gaGl0IGJlbG93IFVSTCBhbmQgY29weSB0byBwcm9qZWN0IGFuZCBlZGl0IGFjY29yZGluZ2x5Lgo+IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC92My9hcGktZG9jcy55YW1sCiogVGhlbiBBY2Nlc3MgU3dhZ2dlciBVUkwgYmVsb3cuCj4gaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkvaW5kZXguaHRtbD9jb25maWdVcmw9L3YzL2FwaS1kb2NzL3N3YWdnZXItY29uZmlnCgojIyMgSDIgSW4gTWVtb3J5IERhdGFiYXNlCiogSW4gbWVtb3J5IERhdGFiYXNlIGhhcyBiZWVuIHVzZWQuIEl0IGNhbiBiZSBhY2Nlc3NlZCBiZWxvdy4KPiBodHRwOi8vbG9jYWxob3N0OjgwODAvaDItY29uc29sZQoKIyMjIENhY2hpbmcKKiBTcHJpbmcncyBkZWZhdWx0IGNhY2hpbmcgaGFzIGJlZW4gdXNlZCBpbiB0aGlzIHByb2plY3QgZm9yIGZyZXF1ZW50bHkgYWNjZXNzZWQgQVBJIEVuZHBvaW50cy4KKiBPbmNlIGVtcHR5LCBEQiBub3QgY2hlY2tlZCB1bnRpbCBUVEwoVGltZSB0byBMaXZlKSwgc28gcGVyZm9ybSBEQiBjaGVjayBpZiBlbXB0eS4KKiBVc2luZyBrZXlzIGFuZCBzeW5jIGlzIHJlY29tbWVuZGVkLiBSZWZlciBmdXJ0aGVyIGRldGFpbHMgYmVsb3cKPiBodHRwczovL3d3dy5mb3JlYWNoLmJlL2Jsb2cvc3ByaW5nLWNhY2hlLWFubm90YXRpb25zLXNvbWUtdGlwcy10cmlja3MKCiMjIyBTY2hlZHVsZXJzCiogSGF2ZSBrZXB0IHNjaGVkdWxlciB0byBwZXJmb3JtIHNjaGVkdWxlZCB0YXNrcy4gQ2hhbmdlIGNyb24uZXhwcmVzc2lvbiBhcyByZXF1aXJlZC4KCiMjIyBHbG9iYWwgRXhjZXB0aW9uIEhhbmRsZXIgZm9yIENvbnRyb2xsZXJzCiogVXNlZCBAUmVzdENvbnRyb2xsZXJBZHZpY2UgdG8gaGF2ZSBjZW50cmFsaXplZCBleGNlcHRpb24gaGFuZGxpbmcgYWNyb3NzIGFsbCBDb250cm9sbGVyIGNsYXNzZXMKKiBBdm9pZGVkIHVzZSBvZiBAUmVzcG9uc2VTdGF0dXMgd2hpY2ggbWFrZXMgY29kZSB0aWdodGx5IGNvdXBsZWQKKiBUb21jYXQgU2VydmxldCBDb250YWluZXIgcmVkaXJlY3RzIHRvIFxlcnJvciBieSBkZWZhdWx0CiogQWx3YXlzIGhhdmUgTm9IYW5kbGVyRm91bmRFeGNlcHRpb24gZm9yIEFQSSBFbmRwb2ludHMgbm90IGNyZWF0ZWQgc28gYXMgdG8gc2VncmVnYXRlIGl0IGZyb20gb3RoZXIgaXNzdWVzIG9mIExlZ2FjeSBJbnRlcm5hbCBTZXJ2ZXIgRXJyb3JzCiogTWV0aG9kQXJndW1lbnROb3RWYWxpZEV4Y2VwdGlvbiBmb3IgYXJndW1lbnRzIHdoZW4gZW1wdHksIG51bGwgb3Igbm90IGdpdmVuLiBQT0pPIG11c3QgaGF2ZSBAVmFsaWQgY2hlY2suIFJlZmVyIHRoaXMgW2xpbmtdKGh0dHBzOi8vZG9jcy5zcHJpbmcuaW8vc3ByaW5nL2RvY3MvY3VycmVudC9qYXZhZG9jLWFwaS9vcmcvc3ByaW5nZnJhbWV3b3JrL3dlYi9iaW5kL01ldGhvZEFyZ3VtZW50Tm90VmFsaWRFeGNlcHRpb24uaHRtbCkKKiBDb21tb24gZXhjZXB0aW9ucyBjYW4gYmUgY2x1YmJlZCBhcyBFeGNlcHRpb24gQXJyYXlzIGFzIHVzZWQgaW4gdGhpcyBwcm9qZWN0CgojIyMgU2VxdWVuY2UgR2VuZXJhdG9yCiogVXNlZCBJZGVudGl0eSBTdHJhdGVneSBoZXJlIHdpdGggYWxsb2NhdGlvbiBzaXplIDI1Cj4gaHR0cHM6Ly93d3cub2JqZWN0ZGIuY29tL2phdmEvanBhL2VudGl0eS9nZW5lcmF0ZWQKCiMjIyBUb3AgTGV2ZWwgSlNPTiBBcnJheXMKKiBIYXZlIGF2b2lkZWQgdXNpbmcgZGlyZWN0IEpTT04gQXJyYXlzIGluIEFQSSByZXNwb25zZS4gUGxlYXNlIGZpbmQgcmVhc29uIGJlbG93IGluIGxpbmsuCj4gaHR0cHM6Ly9zdGFja292ZXJmbG93LmNvbS9xdWVzdGlvbnMvMzUwMzEwMi93aGF0LWFyZS10b3AtbGV2ZWwtanNvbi1hcnJheXMtYW5kLXdoeS1hcmUtdGhleS1hLXNlY3VyaXR5LXJpc2sKCiMjIyBKUEEgJiBIaWJlcm5hdGUgUmVsYXRpb25zaGlwIE1hcHBpbmcKKiBPbmUgUHJvZHVjdCBpcyBvd25lZCBieSBvbmUgbWVtYmVyIC0gQE9uZVRvT25lCiogT25lIE1lbWJlciBjYW4gb3duIG11bHRpcGxlIFByb2R1Y3RzIC0gQE9uZVRvTWFueQoqIFJlZmVyIFtOb3Rlc10oaHR0cHM6Ly9naXRodWIuY29tL2FudXBhbWEtc2luaGEvYW51cGFtYS1ub3Rlcy9ibG9iL21hc3Rlci9oaWJlcm5hdGUtbWFwcGluZy5tZCkKCiMjIyBFbWFpbC9TTVMgSW50ZWdyYXRpb24KKiBJbiBQcm9ncmVzcwoqIFtUd2lsaW8gU01TIE5vdGlmaWNhdGlvbl0oaHR0cHM6Ly93d3cudHdpbGlvLmNvbS9kb2NzL3Ntcy9xdWlja3N0YXJ0L2phdmEpCiogW0VtYWlsIEludGVncmF0aW9uXShodHRwczovL3d3dy5iYWVsZHVuZy5jb20vc3ByaW5nLWVtYWlsKQoKIyMjIEhhdGVPQVMoSHlwZXJtZWRpYSBBcyBUaGUgRW5naW5lIG9mIEFwcGxpY2F0aW9uIFN0YXRlKQoqIEluIFByb2dyZXNzCgojIyMgT3BlbkFQSSBTd2FnZ2VyIFZhbGlkYXRpb24KKiBJbiBQcm9ncmVzcwoKIyMjIFJlZmVyZW5jZSBEb2N1bWVudGF0aW9uCkZvciBmdXJ0aGVyIHJlZmVyZW5jZSwgcGxlYXNlIGNvbnNpZGVyIHRoZSBmb2xsb3dpbmcgc2VjdGlvbnM6CgoqIFtPZmZpY2lhbCBBcGFjaGUgTWF2ZW4gZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9tYXZlbi5hcGFjaGUub3JnL2d1aWRlcy9pbmRleC5odG1sKQoqIFtTcHJpbmcgQm9vdCBNYXZlbiBQbHVnaW4gUmVmZXJlbmNlIEd1aWRlXShodHRwczovL2RvY3Muc3ByaW5nLmlvL3NwcmluZy1ib290L2RvY3MvMi4zLjQuUkVMRUFTRS9tYXZlbi1wbHVnaW4vcmVmZXJlbmNlL2h0bWwvKQoqIFtDcmVhdGUgYW4gT0NJIGltYWdlXShodHRwczovL2RvY3Muc3ByaW5nLmlvL3NwcmluZy1ib290L2RvY3MvMi4zLjQuUkVMRUFTRS9tYXZlbi1wbHVnaW4vcmVmZXJlbmNlL2h0bWwvI2J1aWxkLWltYWdlKQoqIFtTcHJpbmcgV2ViXShodHRwczovL2RvY3Muc3ByaW5nLmlvL3NwcmluZy1ib290L2RvY3MvMi4zLjQuUkVMRUFTRS9yZWZlcmVuY2UvaHRtbHNpbmdsZS8jYm9vdC1mZWF0dXJlcy1kZXZlbG9waW5nLXdlYi1hcHBsaWNhdGlvbnMpCiogW1NwcmluZyBCb290IERldlRvb2xzXShodHRwczovL2RvY3Muc3ByaW5nLmlvL3NwcmluZy1ib290L2RvY3MvMi4zLjQuUkVMRUFTRS9yZWZlcmVuY2UvaHRtbHNpbmdsZS8jdXNpbmctYm9vdC1kZXZ0b29scykKCiMjIyBHdWlkZXMKVGhlIGZvbGxvd2luZyBndWlkZXMgaWxsdXN0cmF0ZSBob3cgdG8gdXNlIHNvbWUgZmVhdHVyZXMgY29uY3JldGVseToKCiogW0J1aWxkaW5nIGEgUkVTVGZ1bCBXZWIgU2VydmljZV0oaHR0cHM6Ly9zcHJpbmcuaW8vZ3VpZGVzL2dzL3Jlc3Qtc2VydmljZS8pCiogW1NlcnZpbmcgV2ViIENvbnRlbnQgd2l0aCBTcHJpbmcgTVZDXShodHRwczovL3NwcmluZy5pby9ndWlkZXMvZ3Mvc2VydmluZy13ZWItY29udGVudC8pCiogW0J1aWxkaW5nIFJFU1Qgc2VydmljZXMgd2l0aCBTcHJpbmddKGh0dHBzOi8vc3ByaW5nLmlvL2d1aWRlcy90dXRvcmlhbHMvYm9va21hcmtzLykKKiBbSGliZXJuYXRlIE1hcHBpbmddKGh0dHBzOi8vc3RhY2thYnVzZS5jb20vYS1ndWlkZS10by1qcGEtd2l0aC1oaWJlcm5hdGUtcmVsYXRpb25zaGlwLW1hcHBpbmcvKQoqIFtKUEFdKGh0dHBzOi8vd3d3Lm9iamVjdGRiLmNvbS9qYXZhL2pwYSk= readmeEtag: '"1d8e34fd722e513a7f16c82c50bdf5f4c4db1b3c"' readmeLastModified: Thu, 07 Dec 2023 12:33:35 GMT repositoryId: 298004156 description: >- A simple standalone RESTful web service application done in Spring Boot framework created: '2020-09-23T14:54:07Z' updated: '2023-12-07T12:33:05Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: anupama-sinha logo: https://avatars.githubusercontent.com/u/68496768?v=4 repoEtag: '"c9b5896987c6a3379e2e7f0c1bb5f20c4b7bb0dc824da0ec453308df02e7c541"' repoLastModified: Thu, 07 Dec 2023 12:33:05 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: dda339640f86bc8fa2eee7992d0ae5c2 - source: openapi3 tags repository: https://github.com/rettvis/geo-spain v3: true repositoryMetadata: base64Readme: >- IyBHRU8tU3BhaW4gQVBJCgpHRU8tU3BhaW4gQVBJIGlzIGEgc2VydmljZSBhaW1lZCBhdCBwcm9ncmFtbWVycywgdG8gc2ltcGxpZnkgdGhlIHByb2Nlc3Npbmcgb2YgcG9zdGFsIGRhdGEgd2l0aGluIHRoZSBTcGFuaXNoIHRlcnJpdG9yeS4= readmeEtag: '"afe5d2dcc289e077343a8caad3c9d2e90582714c"' readmeLastModified: Thu, 15 Apr 2021 22:11:13 GMT repositoryId: 357949475 description: >- GEO-Spain API is a project that aims to facilitate the work of programmers when creating different components that require data related to the geography of Spain, such as street names, postal codes, towns, and a long etcetera. created: '2021-04-14T15:14:57Z' updated: '2021-04-16T06:37:50Z' language: Java archived: false stars: 0 watchers: 0 forks: 0 owner: rettvis logo: https://avatars.githubusercontent.com/u/65330314?v=4 repoEtag: '"3d2f8c4acfc875ba7fb6d86eb08b4e4738865456f8de481a25f3a51cb38bfb57"' repoLastModified: Fri, 16 Apr 2021 06:37:50 GMT foundInMaster: true category: - Testing - Server Implementations id: 47286ac434b8a7bcdfb0add17c319fc7 - source: openapi3 tags repository: https://github.com/pilotak/openapi-typescript-jwt v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIDMgVHlwZVNjcmlwdCBKV1QgdGVtcGxhdGUKQW4gZXhhbXBsZSBBUEkgd2l0aCBUeXBlU2NyaXB0IHN1cHBvcnQgYW5kIGJhc2ljIGxvZ2luIG1lY2hhbmlzbSBiYXNlZCBvbiBKV1QKCiMjIEluc3RhbGwKYGBgc2gKbnBtIGluc3RhbGwgJiYgbnBtIHJ1biBkZXYKYGBgCgojIyBSdW4KRmlyc3QgbG9naW4gYW5kIGNvcHkgZ2VuZXJhdGVkIHRva2VuLgoKYGBgc2gKY3VybCAtSCAnQWNjZXB0OiBhcHBsaWNhdGlvbi9qc29uJyAtWCBQT1NUIC1kICd1c2VybmFtZT10ZXN0JyBodHRwOi8vbG9jYWxob3N0OjMwMDAvdjEvbG9naW4KYGBgCgpOb3cgeW91IGNhbiBzZWUgb3BlbmFwaS5qc29uCmBgYHNoCmN1cmwgLUggJ0FjY2VwdDogYXBwbGljYXRpb24vanNvbicgLVggR0VUIC1IICdBdXRob3JpemF0aW9uOiBCZWFyZXIgZXlKaGJHY2lPaUpJVXpJMU5pSXNJblI1Y0NJNklrcFhWQ0o5Li4uJyBodHRwOi8vbG9jYWxob3N0OjMwMDAvdjEvZG9jcwpgYGAKCk9yIHlvdSBjYW4gbGlzdCB1c2VycwpgYGBzaApjdXJsIC1IICdBY2NlcHQ6IGFwcGxpY2F0aW9uL2pzb24nIC1YIEdFVCAtSCAnQXV0aG9yaXphdGlvbjogQmVhcmVyIGV5SmhiR2NpT2lKSVV6STFOaUlzSW5SNWNDSTZJa3BYVkNKOS4uLicgaHR0cDovL2xvY2FsaG9zdDozMDAwL3YxL3VzZXJzCmBgYAoKIyMgR1VJCkp1c3QgdmlzaXQgYHJlZG9jLmh0bWxgIHZpYSB5b3VyIGJyb3dzZXIK readmeEtag: '"64459c5e485f2321503f2801a5170cf89d5f842e"' readmeLastModified: Mon, 21 Jun 2021 09:08:43 GMT repositoryId: 364698867 description: Express.js OpenAPI 3 typescript JWT boilerplate created: '2021-05-05T20:32:46Z' updated: '2021-06-21T09:08:50Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: pilotak logo: https://avatars.githubusercontent.com/u/8101753?v=4 license: MIT repoEtag: '"66de42312b6f2311c9a7767e4b116a760169ec73d32f5c3279cdc3c6138e142d"' repoLastModified: Mon, 21 Jun 2021 09:08:50 GMT foundInMaster: true category: - Data Validators - Server Implementations id: 73c449d2f080f914c6266a97a3693fbd - source: openapi3 tags repository: https://github.com/glongrais/openapi_photo_management v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJX1Bob3RvX01hbmFnZW1lbnQKCiMjIFByaW5jaXBsZQoKQmFzaWMgcGhvdG8gbWFuYWdlbWVudCBBUEkgY3JlYXRlZCB3aXRoIE9wZW5BUEkgMyB0aGF0IHNlcnZlIGJhc2ljIENSVUQgdmlhIEhUVFAKCiMjIyBUYWJsZSBvZiBDb250ZW50cyAgCltSZXNzb3VyY2VzXSgjUmVzc291Y2VzKSAgCltSZXF1ZXN0c10oI1JlcXVlc3RzKSAgCltTY2hlbWFzXSgjU2NoZW1hcykgICAKW0V4YW1wbGVdKCNFeGFtcGxlKSAgIAoKCiMjIFJlc3NvdXJjZXMKCiogT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGNhbiBiZSBmb3VuZCBbaGVyZV0oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKS4KKiBCYXNpY3MgYWJvdXQgSFRUUCByZXF1ZXN0cyBhbmQgcmVzcG9uc2UgY29kZXMgY2FuIGJlIGZvdW5kIFtoZXJlXShodHRwczovL3d3dy5yZXN0YXBpdHV0b3JpYWwuY29tL2xlc3NvbnMvaHR0cG1ldGhvZHMuaHRtbCkuCgoKIyMgUmVxdWVzdHMKCnwgVHlwZSB8IFJvdXRlIHwgUmVxdWVzdCBCb2R5IHwgUmVzcG9uc2UgQm9keSB8CnwgLS0tIHwgLS0tIHwgLS0tfCAtLS0gfAp8IFBPU1QgfCBgL3Bob3RvYHxQaG90b3w8dWw+PGxpPiAqKlN0YXR1cyBDb2RlKio6IDIwMSAtIGBTdWNjZXNzOiB7bWVzc2FnZTog4oCYUGhvdG8gc3VjY2Vzc2Z1bGx5IGNyZWF0ZWTigJksIGlkOiBwaG90b19pZH1gPC9saT48bGk+KipTdGF0dXMgQ29kZSoqOiA0MjIgLSBgRXJyb3I6IHttZXNzYWdlOiBlcnJ9YDwvbGk+PC91bD58CnwgR0VUIHxgL3Bob3Rvc2B8YWNjZXNzfDx1bD48bGk+KipTdGF0dXMgQ29kZSoqOiAyMDAgLSBgUGhvdG9zOiBbe1Bob3RvfSwgLi5dYCAgPC9saT48bGk+KipTdGF0dXMgQ29kZSoqOiA0MDQgLSAgYEVycm9yOiB7bWVzc2FnZTogZXJyfWAgIDwvbGk+PC91bD58CnwgR0VUIHxgL3Bob3RvL3twaG90b19pZH1gfC18PHVsPjxsaT4qKlN0YXR1cyBDb2RlKio6IDIwMCAtIGBQaG90bzoge2lkOiBpZCwgbmFtZTogbmFtZSwgZGVzY3JpcHRpb246IGRlc2NyaXB0aW9uLCBhY2Nlc3M6IGFjY2VzcywgbG9jYXRpb246IGxvY2F0aW9uLCBjcmVhdGVkX2RhdGU6IGNyZWF0ZWRfZGF0ZSwgbW9kaWZpZWRfZGF0ZTogbW9kaWZpZWRfZGF0ZX1gICA8L2xpPjxsaT4qKlN0YXR1cyBDb2RlKio6IDQwNCAtICBgRXJyb3I6IHttZXNzYWdlOiBlcnJ9YCAgPC9saT48L3VsPnwKfCBQVVQgfGAvcGhvdG8ve3Bob3RvX2lkfWB8UGhvdG98PHVsPjxsaT4qKlN0YXR1cyBDb2RlKio6IDIwMCAtIGBTdWNjZXNzOiB7bWVzc2FnZTog4oCYUGhvdG8gc3VjY2Vzc2Z1bGx5IHVwZGF0ZWTigJksIGlkOiBwaG90b19pZH1gICA8L2xpPjxsaT4qKlN0YXR1cyBDb2RlKio6IDQwNCAtICBgRXJyb3I6IHttZXNzYWdlOiBlcnJ9YDwvbGk+PGxpPioqU3RhdHVzIENvZGUqKjogNDIyIC0gIGBFcnJvcjoge21lc3NhZ2U6IGVycn1gICA8L2xpPjwvdWw+fAp8IERFTEVURSB8YC9waG90by97cGhvdG9faWR9YHwtfDx1bD48bGk+KipTdGF0dXMgQ29kZSoqOiAyMDQgIDwvbGk+PGxpPioqU3RhdHVzIENvZGUqKjogNDA0IC0gIGBFcnJvcjoge21lc3NhZ2U6IGVycn1gICA8L2xpPjwvdWw+fAoKIyMgU2NoZW1hcwoKKiBgUGhvdG9gCgogICAgfCBBdHRyaWJ1dGUgfCBUeXBlIHwgQ29uZGl0aW9ucyB8CiAgICB8IC0tLSB8IC0tLSB8IC0tLSB8CiAgICB8IGlkIHwgc3RyaW5nIHwgYHJlYWRPbmx5OiB0cnVlYCB8CiAgICB8IG5hbWUgfCBzdHJpbmcgfCBgcmVxdWlyZWQ6dHJ1ZWA8L2JyPmBtYXhsZW5naHQ6IDIwYCB8CiAgICB8IGRlc2NyaXB0aW9uIHwgc3RyaW5nIHwgYG1heGxlbmdodDogMTAwYCB8CiAgICB8IGFjY2VzcyB8IHN0cmluZyB8IGByZXF1aXJlZDp0cnVlYDwvYnI+YGVudW06IFtwdWJsaWMsIHByaXZhdGVdYCB8CiAgICB8IGxvY2F0aW9uIHwgc3RyaW5nIHwgYHJlcXVpcmVkOnRydWVgIHwKICAgIHwgZmlsZSB8IHN0cmluZyB8IGByZXF1aXJlZDp0cnVlYCB8CiAgICB8IGNyZWF0ZWRfZGF0ZSB8IHN0cmluZyB8IGByZXF1aXJlZDp0cnVlYDwvYnI+YGZvcm1hdDogZGF0ZS10aW1lYCB8CiAgICB8IHVwZGF0ZWRfZGF0ZSB8IHN0cmluZyB8IGByZXF1aXJlZDp0cnVlYDwvYnI+YGZvcm1hdDogZGF0ZS10aW1lYCB8CgoqIGBQaG90b3NgCgogICAgfCBBdHRyaWJ1dGUgfCBUeXBlIHwgRGVzY3JpcHRpb24gfAogICAgfCAtLS0gfCAtLS0gfCAtLS0gfAogICAgfCBbUGhvdG9dIHwgYXJyYXkgfCBBbiBhcnJheSBvZiBwaG90b3MgfAoKKiBgU3VjY2Vzc2AKCiAgICB8IEF0dHJpYnV0ZSB8IFR5cGUgfCBDb25kaXRpb25zIHwKICAgIHwgLS0tIHwgLS0tIHwgLS0tIHwKICAgIHwgbWVzc2FnZSB8IHN0cmluZyB8IC0gfAogICAgfCBpZCB8IHN0cmluZyB8IC0gfAoKKiBgRXJyb3JgCgogICAgfCBBdHRyaWJ1dGUgfCBUeXBlIHwgQ29uZGl0aW9ucyB8CiAgICB8IC0tLSB8IC0tLSB8IC0tLSB8CiAgICB8IG1lc3NhZ2UgfCBzdHJpbmcgfCAtIHwKCgojIyBFeGFtcGxlCgpgUGhvdG9gIHNjaGVtYToKCmBgYGpzb24KewogICJuYW1lIjogIlBob3RvXzMiLAogICJkZXNjcmlwdGlvbiI6ICJNeSBkb2cgc2VsZmllIiwKICAiYWNjZXNzIjogInB1YmxpYyIsCiAgImxvY2F0aW9uIjogIlNrb3BqZSIsCiAgImZpbGUiOiAidGVkZHlfc2VsZmllLmpwZyIsCiAgImNyZWF0ZWRfZGF0ZSI6ICIxOTk2LTA4LTIzVDE3OjMyOjI4WiIsCiAgIm1vZGlmaWVkX2RhdGUiOiAiMjAyMC0xMi0yN1QyMzo0NToyOFoiCn0KYGBgCgoK readmeEtag: '"b18af50cf255678d9a5544f5e644f6a11d4bbe4e"' readmeLastModified: Thu, 11 Nov 2021 10:08:24 GMT repositoryId: 426552432 description: >- Basic photo management API with OpenAPI 3 to serve basic CRUD via HTTP requests created: '2021-11-10T09:02:16Z' updated: '2021-11-17T16:23:05Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: glongrais logo: https://avatars.githubusercontent.com/u/34845137?v=4 repoEtag: '"47ddfe33991b55a7090adc756ca7f488f5bd41d0a9d86e3be9ad7af0dafbd284"' repoLastModified: Wed, 17 Nov 2021 16:23:05 GMT foundInMaster: true category: Description Validators id: 58070799bfdcbcf0b732a5a146acbab6 - source: - https://openapi.tools/ - openapi3 tags repository: https://github.com/funbox/fitting v3: true repositoryMetadata: base64Readme: >- # Fitting

<img align="right" width="192" height="192"
alt="Fitting avatar: Documents with hangers"
src="./images/logo.png">

Library add improve test log, validate its according to your API documentation, show the documentation coverage with log.

Test log setting supports RSpec test and WebMock stubbing for Ruby On Rails application, API documentation supports API Blueprint and OpenAPI.

This reduces the costs of support, testers and analysts.

Log
```text
FITTING incoming request {"method":"POST","path":"/public/api/v1/inboxes/tEX5JiZyceiwuKMi1oN9Sf8S/contacts","body":{},"response":{"status":200,"content_type":"application/json","body":{"source_id":"00dbf18d-879e-47cb-ac45-e9aece266eb1","pubsub_token":"ktn6YwPus57JDf4e59eFPom5","id":3291,"name":"shy-surf-401","email":null,"phone_number":null}},"title":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb:9","group":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb","host":"www.example.com"}
FITTING outgoing request {"method":"POST","path":"/v1/organizations/org_id/meeting","body":{},"response":{"status":200,"content_type":"application/json","body":{"success":true,"data":{"meeting":{"id":"meeting_id","roomName":"room_name"}}}},"title":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb:50","group":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb","host":"api.cluster.dyte.in"}
```

validation
```console
FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.FFF..FFFFFFFFFF....F.......F...FF.....F...F....F..............................FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF.F..FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF........FFF...FFFF......FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF........FFFFFFFFFFF..FFFFFF..FFFFFFFFFFFFFFFFF.......FFFFFF.............FFFFFFFFFFFF....F........FFF.F...FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF............FF........FFF......FFFFFFFFFFFFFFFFFFFFFF....FFFFFF......F............FFFF........FFFFFFFFFFFFFF.....FFFFFFFFFFFFFFFFFFFFFFF..FF.....FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.....FF..........FFFFFFFFFFFFFFFFFF...FFFF...............F.F....FF..FFFFFFFF

  1) Fitting::Doc::NotFound log error:

host: www.example.com
method: POST
path: /public/api/v1/inboxes/{inbox_identifier}/contacts
code: 200

content-type: application/json

json-schema: {
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "properties": {
    "id": {
      "type": "integer",
      "description": "Id of the contact"
    },
    "source_id": {
      "type": "string",
      "description": "The session identifier of the contact"
    },
    "name": {
      "type": "string",
      "description": "Name of the contact"
    },
    "email": {
      "type": "string",
      "description": "Email of the contact"
    },
    "pubsub_token": {
      "type": "string",
      "description": "The token to be used to connect to chatwoot websocket"
    }
  }
}

body: {
  "source_id": "c9e8c31f-06df-49b4-8fb9-4466457ae65b",
  "pubsub_token": "Zgc7DEvaj5TkgZ1a4C7AvJXo",
  "id": 3293,
  "name": "restless-snowflake-670",
  "email": null,
  "phone_number": null
}

error [
  "The property '#/email' of type null did not match the following type: string in schema e56b7e65-d70c-5f7a-a96c-982df5f8f2f7"
]

...

804 examples, 565 failure, 0 pending

Coverage: 65.51%
```

and cover
![exmaple](images/b1.png)

![exmaple](images/b2.png)

![exmaple](images/w1.png)

![exmaple](images/w2.png)

## Installation
Add this line to your application's Gemfile:
```ruby
gem 'fitting'
```

After that execute:
```bash
$ bundle
```

Or install the gem by yourself:
```bash
$ gem install fitting
```

## Usage
### Log
Firstly, improve `test.log`.

To your `spec_helper.rb`:

```ruby
require 'fitting'

Fitting.logger
```

Delete all files `log/*.log` and run rspec

You get more information about incoming and outgoing request in `log/fitting*.log`.

```text
FITTING incoming request {"method":"POST","path":"/public/api/v1/inboxes/tEX5JiZyceiwuKMi1oN9Sf8S/contacts","body":{},"response":{"status":200,"content_type":"application/json","body":{"source_id":"00dbf18d-879e-47cb-ac45-e9aece266eb1","pubsub_token":"ktn6YwPus57JDf4e59eFPom5","id":3291,"name":"shy-surf-401","email":null,"phone_number":null}},"title":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb:9","group":"./spec/controllers/public/api/v1/inbox/contacts_controller_spec.rb","host":"www.example.com"}
FITTING outgoing request {"method":"POST","path":"/v1/organizations/org_id/meeting","body":{},"response":{"status":200,"content_type":"application/json","body":{"success":true,"data":{"meeting":{"id":"meeting_id","roomName":"room_name"}}}},"title":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb:50","group":"./spec/controllers/api/v1/accounts/integrations/dyte_controller_spec.rb","host":"api.cluster.dyte.in"}
```

### Validation
Secondly, validate the log to the documentation.

Add this to your `.fitting.yml`:

```yaml
APIs:
  - host: www.example.com
    type: openapi2
    path: swagger/swagger.json
```

Run 
```bash
bundle e rake fitting:validate
```

Console output

```console
FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.FFF..FFFFFFFFFF....F.......F...FF.....F...F....F..............................FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF.F..FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF..FF........FFF...FFFF......FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF........FFFFFFFFFFF..FFFFFF..FFFFFFFFFFFFFFFFF.......FFFFFF.............FFFFFFFFFFFF....F........FFF.F...FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF............FF........FFF......FFFFFFFFFFFFFFFFFFFFFF....FFFFFF......F............FFFF........FFFFFFFFFFFFFF.....FFFFFFFFFFFFFFFFFFFFFFF..FF.....FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF.....FF..........FFFFFFFFFFFFFFFFFF...FFFF...............F.F....FF..FFFFFFFF

  1) Fitting::Doc::NotFound log error:

host: www.example.com
method: POST
path: /public/api/v1/inboxes/{inbox_identifier}/contacts
code: 200

content-type: application/json

json-schema: {
  "$schema": "http://json-schema.org/draft-04/schema#",
  "type": "object",
  "properties": {
    "id": {
      "type": "integer",
      "description": "Id of the contact"
    },
    "source_id": {
      "type": "string",
      "description": "The session identifier of the contact"
    },
    "name": {
      "type": "string",
      "description": "Name of the contact"
    },
    "email": {
      "type": "string",
      "description": "Email of the contact"
    },
    "pubsub_token": {
      "type": "string",
      "description": "The token to be used to connect to chatwoot websocket"
    }
  }
}

body: {
  "source_id": "c9e8c31f-06df-49b4-8fb9-4466457ae65b",
  "pubsub_token": "Zgc7DEvaj5TkgZ1a4C7AvJXo",
  "id": 3293,
  "name": "restless-snowflake-670",
  "email": null,
  "phone_number": null
}

error [
  "The property '#/email' of type null did not match the following type: string in schema e56b7e65-d70c-5f7a-a96c-982df5f8f2f7"
]

...

804 examples, 565 failure, 0 pending

Coverage: 65.51%
```

### Coverage
And task will create HTML (`coverage/fitting.html`) reports.

![exmaple](images/b1.png)

![exmaple](images/b2.png)

More information on action coverage

![exmaple2](images/w1.png)

![exmaple2](images/w2.png)

## Settings

### APIs

#### type

##### OpenAPI 2.0
Swagger

```yaml
APIs:
  - host: www.example.com
    type: openapi2
    path: doc/api.json
```

##### OpenAPI 3.0
Also OpenAPI

```yaml
APIs:
  - host: www.example.com
    type: openapi3
    path: doc/api.json
```

##### API Blueprint
First you need to install [drafter](https://github.com/apiaryio/drafter) or [crafter](https://github.com/funbox/crafter).
Works after conversion from API Blueprint to API Elements (in YAML file) with Drafter or Crafter.

That is, I mean that you first need to do this

```bash
drafter doc.apib -o doc.yaml
```

or

```bash
node_modules/.bin/crafter doc.apib > doc.yaml
```

and then

```yaml
APIs:
  - host: www.example.com
    type: drafter
    path: doc/api.yaml
```

or

```yaml
APIs:
  - host: www.example.com
    type: crafter
    path: doc/api.yaml
```

##### Tomograph

To use additional features of the pre-converted [tomograph](https://github.com/funbox/tomograph)

example

```bash
bundle exec tomograph -d crafter --exclude-description doc/api.yml doc/api.json
```

and then

```yaml
APIs:
  - host: www.example.com
    type: tomogram
    path: doc/api.json
```

#### prefix

Setting the prefix name is optional. For example, you can do this:

```yaml
APIs:
  - host: www.example.com
    prefix: /api/v3
    type: openapi2
    path: swagger/swagger.json
```

### SkipValidation

#### host

It is not necessary to immediately describe each host in detail, you can only specify its name and skip it until you are ready to documented it

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
```

#### prefix

If you want to skip a specific prefix in the host

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
    prefix: /admin/api
```

#### method and path

If you want to skip a specific request in the host

```yaml
SkipValidation:
  - host: api.cluster.dyte.in
    method: GET
    path: /api/v1/cars
```

### NoCov

It is not necessary to immediately test each doc in detail, you can only specify its name and skip it until you are ready to test it

#### host
```yaml
NoCov:
  - host: sso.test
```

#### method
```yaml
NoCov:
  - host: sso.test
    method: GET
```

#### path
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
```

#### code
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
```

#### content-type
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
```

#### combination
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
    combination: oneOf.0
```

#### combination_next
```yaml
NoCov:
  - host: sso.test
    method: GET
    path: /users/{userId}
    code: 200
    content-type: application/json
    combination: oneOf.0
    combination_next: oneOf.0.required.users
```

### Debug

If you find bug, you can debug it or create task in this github project  with new file `coverage/fitting.debug.yml`

```yaml
Debug:
  - host: www.example.com
    method: GET
    path: /api/v3/users
    code: 200
    content-type: application/json
```

## Contributing

Bug reports and pull requests are welcome on GitHub at [github.com/funbox/fitting](https://github.com/funbox/fitting).
This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [Contributor Covenant](http://contributor-covenant.org) code of conduct.

## License

The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).

[![Sponsored by FunBox](https://funbox.ru/badges/sponsored_by_funbox_centered.svg)](https://funbox.ru)
 readmeEtag: '"14f5079a6ca2e159943ae83eaa2999a55977d34a"' readmeLastModified: Mon, 17 Apr 2023 17:59:53 GMT repositoryId: 701451219 description: >- Library add improve test log for RSpec and WebMock, validate its according to API Blueprint and Open API, show the documentation coverage with log. created: '2023-10-06T16:56:37Z' updated: '2024-04-01T18:31:52Z' language: null archived: false stars: 0 watchers: 0 forks: 0 owner: funbox logo: https://avatars.githubusercontent.com/u/1439359?v=4 license: MIT repoEtag: '"1972fcd1c40a12ec8aa5d68da4b9582c90936b499796efe491636f3094365f6a"' repoLastModified: Mon, 01 Apr 2024 18:31:52 GMT foundInMaster: true category: - Testing - Data Validators - Learning id: 9aafc5713bc07ef13ec4f56258385ca1 name: Fitting language: Ruby link: https://github.com/funbox/fitting source_description: >- Library add improve test log for RSpec and WebMock, validate its according to API Blueprint and Open API, show the documentation coverage with log. v2: true v3_1: true - source: openapi3 tags repository: https://github.com/funbox/tomograph v3: true repositoryMetadata: base64Readme: >- # Tomograph 

Convert API Blueprint, Swagger and OpenAPI to minimal routes with JSON Schema. For ease of use and creation of new tools.

Will look like

  ```json
  [
    {
      "path": "/sessions",
      "method": "POST",
      "content-type": "application/json",
      "requests": [{
        "$schema": "http://json-schema.org/draft-04/schema#",
        "type": "object",
        "properties": {
          "login": {
            "type": "string"
          },
          "password": {
            "type": "string"
          },
          "captcha": {
            "type": "string"
          }
        },
        "required": [
          "login",
          "password"
        ]
      }],
      "responses": [
        {
          "status": "401",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "429",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "201",
          "content-type": "application/json",
          "body": {
            "$schema": "http://json-schema.org/draft-04/schema#",
            "type": "object",
            "properties": {
              "confirmation": {
                "type": "object",
                "properties": {
                  "id": {
                    "type": "string"
                  },
                  "type": {
                    "type": "string"
                  },
                  "operation": {
                    "type": "string"
                  }
                },
                "required": [
                  "id",
                  "type",
                  "operation"
                ]
              },
              "captcha": {
                "type": "string"
              },
              "captcha_does_not_match": {
                "type": "boolean"
              }
            }
          }
        }
      ]
    }
  ]
  ```

## Installation

Then add this line to your application's Gemfile:

```ruby
gem 'tomograph'
```

After that execute:

```bash
$ bundle
```

Or install the gem by yourself:

```bash
$ gem install tomograph
```

## Usage

### In code

#### OpenAPI 2.0

Also Swagger

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(openapi2_json_path: '/path/to/doc.json')
```

#### OpenAPI 3.0

Also OpenAPI

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(openapi3_yaml_path: '/path/to/doc.yaml')
```

#### API Blueprint

First you need to install [drafter](https://github.com/apiaryio/drafter).
Works after conversion from API Blueprint to API Elements (in YAML file) with Drafter.

That is, I mean that you first need to do this

```bash
drafter doc.apib -o doc.yaml
```

and then

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(drafter_yaml_path: '/path/to/doc.yaml')
```

#### Tomograph

To use additional features of the pre-converted

```ruby
require 'tomograph'

tomogram = Tomograph::Tomogram.new(tomogram_json_path: '/path/to/doc.json')
```

#### prefix
Default: `''`

You can specify API prefix and path to the spec using one of the possible formats:

```ruby
Tomograph::Tomogram.new(prefix: '/api/v2', drafter_yaml_path: '/path/to/doc.yaml')
```

```ruby
Tomograph::Tomogram.new(prefix: '/api/v2', tomogram_json_path: '/path/to/doc.json')
```

#### to_json
Use `to_json` for converting to JSON, example from API Blueprint:

```ruby
tomogram.to_json
```

<details>
  <summary>Example input</summary>

  ```apib
  FORMAT: 1A
  HOST: http://test.local
  
  # project
  
  # Group project
  
  Project
  
  ## Authentication [/sessions]
  
  ### Sign In [POST]
  
  + Request (application/json)
  
      + Attributes
       + login (string, required)
       + password (string, required)
       + captcha (string, optional)
  
  + Response 401 (application/json)
  
  + Response 429 (application/json)
  
  + Response 201 (application/json)
  
      + Attributes
       + confirmation (Confirmation, optional)
       + captcha (string, optional)
       + captcha_does_not_match (boolean, optional)
  
  
  # Data Structures
  
  ## Confirmation (object)
    + id (string, required)
    + type (string, required)
    + operation (string, required)
  ```
</details>

<details>
  <summary>Example output</summary>

  ```json
  [
    {
      "path": "/sessions",
      "method": "POST",
      "content-type": "application/json",
      "requests": [{
        "$schema": "http://json-schema.org/draft-04/schema#",
        "type": "object",
        "properties": {
          "login": {
            "type": "string"
          },
          "password": {
            "type": "string"
          },
          "captcha": {
            "type": "string"
          }
        },
        "required": [
          "login",
          "password"
        ]
      }],
      "responses": [
        {
          "status": "401",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "429",
          "content-type": "application/json",
          "body": {}
        },
        {
          "status": "201",
          "content-type": "application/json",
          "body": {
            "$schema": "http://json-schema.org/draft-04/schema#",
            "type": "object",
            "properties": {
              "confirmation": {
                "type": "object",
                "properties": {
                  "id": {
                    "type": "string"
                  },
                  "type": {
                    "type": "string"
                  },
                  "operation": {
                    "type": "string"
                  }
                },
                "required": [
                  "id",
                  "type",
                  "operation"
                ]
              },
              "captcha": {
                "type": "string"
              },
              "captcha_does_not_match": {
                "type": "boolean"
              }
            }
          }
        }
      ]
    }
  ]
  ```
</details> 

#### to_a
```ruby
tomogram.to_a
```

#### find_request
```ruby
request = tomogram.find_request(method: 'GET', path: '/status/1?qwe=rty')
```

#### find_request_with_content_type
```ruby
request = tomogram.find_request_with_content_type(method: 'GET', path: '/status/1?qwe=rty', content_type: 'application/json')
```

#### `find_responses`
```ruby
responses = request.find_responses(status: '200')
```

#### prefix_match?
This may be useful if you specify a prefix.

```ruby
tomogram.prefix_match?('http://local/api/v2/users')
```

#### to_resources
Maps resources for API Blueprint with possible requests.

Example output:

```ruby
{
  '/sessions' => ['POST /sessions']
}
```

### Command line tool

CLI allows you to convert files from API Blueprint (API Elements), Swagger and OpenAPI to JSON Schema.

Run CLI with `-h` to get detailed help:

```bash
tomograph -h
```

To specify the handler version use the `-d` flag:

#### OpenAPI 2.0
```bash
tomograph -d openapi2 openapi2.json tomogram.json
```

#### OpenAPI 3.0
```bash
tomograph -d openapi3 openapi3.yaml doc.json
```

#### API Blueprint
```bash
tomograph -d 4 apielemetns.yaml doc.json
```

#### exclude-description

Exclude "description" keys from json-schemas.

```bash
tomograph -d 4 apielemetns.yaml doc.json --exclude-description
```

#### split

Split output into files by method. Output in dir path.

```bash
tomograph -d 4 --split apielemetns.yaml jsons/
```

## License

The gem is available as open source under the terms of the [MIT License](http://opensource.org/licenses/MIT).

[![Sponsored by FunBox](https://funbox.ru/badges/sponsored_by_funbox_centered.svg)](https://funbox.ru)
 readmeEtag: '"f17fc04aab49a8f7b4ad80f0907855c95335b068"' readmeLastModified: Thu, 02 Feb 2023 13:12:37 GMT repositoryId: 701451827 description: >- Convert API Blueprint, Swagger and OpenAPI to JSON Schema and search through it created: '2023-10-06T16:58:26Z' updated: '2024-04-01T18:36:44Z' language: null archived: false stars: 0 watchers: 0 forks: 0 owner: funbox logo: https://avatars.githubusercontent.com/u/1439359?v=4 license: MIT repoEtag: '"23ca3be7a7796f644800e794e75f203acf8538933a3ce65957eb48a193333ab1"' repoLastModified: Mon, 01 Apr 2024 18:36:44 GMT foundInMaster: true category: Parsers id: 99ad4f595f72485cba5a861dd6812743 - source: openapi3 tags repository: https://github.com/zimmj/simple-user-service v3: true id: 0c3e0d5c0877b9f25539f349454ccc4a repositoryMetadata: base64Readme: >- IyBTaW1wbGUgVXNlciBzZXJ2aWNlCgpUaGlzIGlzIGEgZXhhbXBsZSBwcm9qZWN0IG9mIGFuIHNpbXBsZSB1c2VyIHNlcnZpY2UuCkEgdXNlciBoYXMgZm9sbG93aW5nIGZpZWxkcy4KCmBgYHltbAp1c2VyOgogIGlkOiB1dWlkCiAgbmFtZTogc3RyaW5nCiAgZW1haWw6IHN0cmluZwogIHBhc3N3b3JkOiBzdHJpbmcKYGBgCgpBcyBpdCBhIHNpbXBsZSBzaG93IG9mIGNvbmNlcHQgcHJvamVjdCwgdGhlIHBhc3N3b3JkIGFyZSBzYXZlZCBpbiBwbGFpbiB0ZXh0LgoKVGhlIHNlcnZpY2UgY29uc2lzdCBvZiB0aHJlZSBsYXllcnM6CgotIGNvbnRyb2xsZXJzCi0gc2VydmljZXMKLSBkYXRhIGxheWVyCgpUaGlzIHNlcnZpY2UgaXMgd3JpdHRlbiBpbiB0eXBlLXNjcmlwdC4KSXQgdXNlcyBleHByZXNzIGZvciBpdCdzIHNlcnZlciBmdW5jdGlvbmFsaXR5LgoKVGhlIFtvcGVuLWFwaSB5YW1sIGRlZmluaXRpb25dKC4vY29uZmlnL29wZW5hcGkueW1sKSBpcyB1c2VkIGFzIG1haW4gY29udHJhY3QuCkFsbCB0aGUgZGVmaW5lZCBlbmRwb2ludHMgYXJlIGxpbmtlZCB0byB0aGUgY29ycmVzcG9uZGluZyBmdW5jdGlvbiBpbiB0aGUgY29udHJvbGxlciBsYXllciBieSB0aGUgcGFja2FnZTogc3dhZ2dlci1yb3V0ZXMtZXhwcmVzcy4KVGhlIHJvdXRlcyBhcmUgbWF0Y2hlZCB2aWEgdGhlIGRlZmluZWQgb3BlcmF0aW9uIElkJ3MuCgpWYWxpZGF0aW9uIG9mIHRoZSByZXF1ZXN0IGFuZCByZXNwb25zZSBpcyBtYWRlIHdpdGggdGhlIHBhY2thZ2U6IGV4cHJlc3Mtb3BlbmFwaS12YWxpZGF0b3IuClRoaXMgYXV0b21hdGljYWxseSB2YWxpZGF0ZXMgYWxsIGRhdGEgY29taW5nIGluIGFuZCBnb2luZyBmb3IgdGhlcmUgY29ycmVjdG5lc3MgYXMgZGVzY3JpYmVkIGluIHRoZSBvcGVuLWFwaSB5YW1sLgoKV2l0aCB0aGlzLCBJIGNhbiBndWFyYW50ZWUsIHRoYXQgdGhpcyBjb250cmFjdCBpcyBjb3JyZWN0bHkgaW1wbGVtZW50ZWQgYW5kIG90aGVyIGRldmVsb3BlciBjYW4gdXNlIHRoaXMgeWFtbCB0byBnZW5lcmF0ZSB0aGUgY2xpZW50LgoKSW4gdGhlIGRhdGEgbGF5ZXIsIEkgdXNlIGRyaXp6bGUgYXMgT01SIHRvIGNvbm5lY3QgdG8gdGhlIHBvc3RncmVzIGRhdGFiYXNlLgpJdCdzIGEgZnVsbCBmbGV0Y2hlZCBPTVIgd2l0aCBtaWdyYXRpb24gYW5kIGFsbG93cyB0eXBlIHNhZmUgcXVlcmllcy4KCiMjIFJ1biB3aXRoIGRvY2tlcgoKVGhpcyByZXBvc2l0b3J5IGlzIGRvY2tlcml6ZWQsIHRoZXJlZm9yZSBjYW4gYmUgYnVpbGQgYW5kIHJ1biB3aXRoIGFueSBjb250YWluZXIgdG9vbHMuCgpBcyB0aGVyZSBpcyBhIHdvcmtpbmcgY29tcG9zZSBmaWxlLCB0aGUgc2ltcGxlc3QgaXMgdG8gcnVuIGRvY2tlci1jb21wb3NlIHVwIGluIHRoZSByb290IGRpcmVjdG9yeS4KSXQgd2lsbCBidWlsZCB0aGUgZG9ja2VyIGNvbnRhaW5lciBhbmQgcnVuIHRoZSBhcHBsaWNhdGlvbiB3aXRoIGEgcG9zdGdyZXMgc2VydmVyIGF0dGFjaGVkIHRvIGl0LgoKVGhlIHBvc3RncmVzIGRhdGFiYXNlIGlzIGluaXRpYWxpemVkIHdpdGggdGhlIFtpbml0LnNxbF0oLi9kb2NrZXIvaW5pdC5zcWwpLgpUaGUgY29udGVudCBvZiBpdCBpcyB0aGUgbWlncmF0aW9uIG9mIGRyaXp6bGUKClRoZSB1c2VyIHNlcnZpY2UgaXMgZXhwb3NlZCBvbiB0aGUgcG9ydDogMzAwMC4KCkEgZ29vZCBlbnRyeSBwb2ludCBpcyB0aGUgW3N3YWdnZXIgdWldKGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMC9zd2FnZ2VyKSBhcyB5b3UgY2FuIHVzZSBpdCwgdG8gY2FsbCB0aGUgZGlmZmVyZW50IGVuZHBvaW50cy4KClRvIHJ1biBzZXJ2aWNlIHJ1biBkb2NrZXIgY29tcG9zZSB1cCBpbiB0aGUgcm9vdCBkaXJlY3Rvcnk6CgpgYGBiYXNoCmRvY2tlciBjb21wb3NlIHVwCmBgYAoKQW5vdGhlciBvcHRpb24gdG8gY2FsbCB0aGUgc2VydmljZSBpcyB0byB1c2UgdGhlIHByb3ZpZGVkIFtwb3N0bWFuIGNvbGxlY3Rpb25dKC4vY29udGVudC9zaW1wbGUtdXNlci1zZXJ2aWNlLnBvc3RtYW5fY29sbGVjdGlvbi5qc29uKS4KClRoZSBmbG93IG9mIHRoZSBzZXJ2aWNlIGlzIGFzIGZvbGxvd3MuCgoxLiBDcmVhdGUgYSB1c2VyCjIuIFNpZ24gaW4gd2l0aCB0aGUgdXNlciAtPiBnZXQgSldUIHRva2VuCgpXaXRoIHRoZSBKV1QgdG9rZW4geW91IGNhbjoKCjEuIExpc3QgYWxsIHVzZXJzCjIuIERpc3BsYXkgYSB1c2VyCjMuIE1vZGlmeSB5b3VyIHVzZXIKNC4gRGVsZXRlIHlvdXIgdXNlcgoKIyMgUnVuIGl0IGxvY2FsbHkKClRvIHJ1biB0aGlzIHNlcnZpY2UgbG9jYWxseSBhIHBvc3RncmVzIGRiIG5lZWQgdG8gcnVuLgpUaGUgZGF0YWJhc2UgY29ubmVjdGlvbiBuZWVkIHRvIGJlIGdpdmVuIGluIHRoZSBmaWxlOiBbLmVudi5kZXZdKC4vY29uZmlnLy5lbnYuZGV2KS4KUHJvdmlkZSBhcyB3ZWxsIHRoZSBuYW1lIG9mIHRoZSBkYXRhYmFzZSB3aGljaCBzaG91bGQgYmUgdXNlZC4KCmBgYHlhbWwKREJfSE9TVD1sb2NhbGhvc3QKREJfUE9SVD01NDMyCkRCX1VTRVI9cG9zdGdyZXMKREJfUEFTU1dPUkQ9cm9vdApEQVRBQkFTRV9OQU1FPXNpbXBsZV91c2VyX3NlcnZpY2UKYGBgCgpCZWZvcmUgcnVubmluZyB0aGUgYXBwbGljYXRpb24sIHRoZSBtaWdyYXRpb24gbmVlZCB0byBiZSBydW4uCgpgYGBiYXNoCnlhcm4gcnVuOm1pZ3JhdGlvbgpgYGAKClRoaXMgY3JlYXRlcyB0aGUgbmVlZGVkIHRhYmxlcyBpbiB0aGUgZGF0YWJhc2UgYW5kIHRoZSBzZXJ2aWNlIGNhbiBiZSBzdGFydGVkLgoKVG8gcnVuIHRoZSBhcHBsaWNhdGlvbiBpbiBkZXZlbG9wZXIgbW9kZToKCmBgYGJhc2gKeWFybiBkZXYKYGBgCg== readmeEtag: '"fc150277921030e0cac3fbafaba5d42961f90dfc"' readmeLastModified: Mon, 07 Aug 2023 19:09:30 GMT repositoryId: 675108094 description: A simple user service, which saves password in plain text!. created: '2023-08-05T20:01:16Z' updated: '2023-09-29T06:55:32Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: zimmj logo: https://avatars.githubusercontent.com/u/33086958?v=4 repoEtag: '"d869b142caf884280cd567ceeba24ef9a66503ac2d6008406a4aa197bb4b752d"' repoLastModified: Fri, 29 Sep 2023 06:55:32 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/marszm/shop_backend v3: true id: dcaf755047af73489f9cf0040b0b4c86 repositoryMetadata: base64Readme: >- IyBTaW1wbGUgYmFja2VuZCBzZXJ2aWNlIGZvciBzaG9wIGFwcGxpY2F0aW9uIHdpdGggIENSVUQgb3BlcmF0aW9ucyB3cml0dGVuIGluIEphdmEgMTcsIFNwcmluZyBCb290IDMgYW5kIFNwcmluZyA2IGZyYW1ld29yayBjb25uZWN0ZWQgdG8gTXlTUUwgZGF0YWJhc2UuCiMgRG9jdW1lbnRhdGlvbiBlbmRwb2ludDogaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkvaW5kZXguaHRtbCM= readmeEtag: '"bbe36b0b63ba4b4d13c57648fe40cdcfd375e933"' readmeLastModified: Tue, 17 Jan 2023 20:09:02 GMT repositoryId: 571245518 description: >- Simple backend service for shop application with CRUD operations written in Java 17, Spring Boot 3 and Spring 6 framework connected to MySQL database. created: '2022-11-27T16:19:23Z' updated: '2023-01-10T20:31:56Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: marszm logo: https://avatars.githubusercontent.com/u/23463955?v=4 repoEtag: '"4a59d892e3b4526f5b500564ea07bc47ae222ad9d383c693d3ed1558a1ad9b70"' repoLastModified: Tue, 10 Jan 2023 20:31:56 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/abdullah2993/textsynth-openapi-v3-spec v3: true id: 5fc54a5ed241302e5a8fa86292f6e6f2 repositoryMetadata: base64Readme: >- IyB0ZXh0c3ludGgtb3BlbmFwaS12My1zcGVjCk9wZW5BUEkgVjMgU3BlY2lmaWNhdGlvbiBmb3IgaHR0cHM6Ly90ZXh0c3ludGguY29tCgpUaGlzIHJlcG9zaXRvcnkgaXMgYSBbdGVtcGxhdGUsIHlvdSBjYW4gY3JlYXRlIHlvdXIgcmVwbyBmcm9tIHRoaXMgdG8gZ2VuZXJhdGUgY2xpZW50cyBdKGh0dHBzOi8vZG9jcy5naXRodWIuY29tL2VuL3JlcG9zaXRvcmllcy9jcmVhdGluZy1hbmQtbWFuYWdpbmctcmVwb3NpdG9yaWVzL2NyZWF0aW5nLWEtcmVwb3NpdG9yeS1mcm9tLWEtdGVtcGxhdGUpCg== readmeEtag: '"db53e9b69acda1e1dff3a60bfa996aef3a41855c"' readmeLastModified: Sat, 22 Apr 2023 00:30:46 GMT repositoryId: 631104713 description: OpenAPI V3 Specification for https://textsynth.com created: '2023-04-22T00:22:49Z' updated: '2023-04-22T00:24:01Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: abdullah2993 logo: https://avatars.githubusercontent.com/u/5978905?v=4 license: MIT repoEtag: '"a794807aa5e693e254970f2bd0c613e15a5ff025872cba5e39790a0eb7548f5c"' repoLastModified: Sat, 22 Apr 2023 00:24:01 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/livingdocsio/openapi v3: true id: 84bfb7fcfdecd08b4df5318f3517f75b repositoryMetadata: base64Readme: >- IyBMaXZpbmdkb2NzIE9wZW5BUEkgU3BlY2lmaWNhdGlvbgpPcGVuQVBJIDMgc3BlY2lmaWNhdGlvbiBmb3IgTGl2aW5nZG9jcyBQdWJsaWMgQVBJCgpUaGlzIGZpbGUgbWFrZXMgaXQgZWFzeSB0byB0ZXN0IGFuZCBzaGFyZSBvdXIgQVBJIGVucG9pbnRzLgoKSXQncyBhdmFpbGFibGUgW2hlcmVdKC9saXZpbmdkb2NzLW9wZW5hcGkuanNvbikKCiMjIFN1cHBvcnRlZCBlbnZpcm9ubWVudHMKLSBsb2NhbGhvc3QKLSBMaXZpbmdkb2NzIGRlbW8gYXQgZWRpdC5saXZpbmdkb2NzLmlvLCBkZXByZWNhdGVkIGZyb20gcmVsZWFzZS0yMDIzLTA5CgojIyBIb3cgdG8gdXNlIGl0ClRoZSBPcGVuQVBJIHNwZWMgaXMgaG9zdGVkIGJ5IEdpdGh1YiBwYWdlcyBhdDogCgpbaHR0cHM6Ly9saXZpbmdkb2NzaW8uZ2l0aHViLmlvL29wZW5hcGkvbGl2aW5nZG9jcy1vcGVuYXBpLmpzb25dKGh0dHBzOi8vbGl2aW5nZG9jc2lvLmdpdGh1Yi5pby9vcGVuYXBpL2xpdmluZ2RvY3Mtb3BlbmFwaS5qc29uKQoKVGhpcyBjYW4gYmUgaW1wb3J0ZWQgaW4gSW5zb21uaWEgb3IgUG9zdG1hbiB0byBoYXZlIGEgZnVsbCBjb2xsZWN0aW9ucyBvZiBBUEkgZW5wb2ludHMgdG8gdGVzdC4KCjxpbWcgd2lkdGg9IjE1MTIiIGFsdD0ib3BlbmFwaS1pbnNvbW5pYSIgc3JjPSJodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS8xNjgyODAzLzE5Njg4NjkxMC03NjBhMmRkZC05YjQ2LTQyYzMtOTI5Mi03MTk2OGQ4MGZiYWYucG5nIj4KCgpBbHRlcm5hdGl2ZWx5IHRoZSBzYW1lIGNhbiBiZSBkb25lIHdpdGggb25saW5lIHRvb2xzLiBZb3UganVzdCBuZWVkIHRvIHBhc3RlIHRoZSBPcGVuQVBJIHNwZWMgdXJsIGluIG9uZSBvZiB0aGUgZm9sbG93aW5nIHRvb2xzLgoKW1N3YWdnZXIgVUldKGh0dHBzOi8vcGV0c3RvcmUuc3dhZ2dlci5pby8pCgpbU3dhZ2dlciBFZGl0b3JdKGh0dHBzOi8vZWRpdG9yLnN3YWdnZXIuaW8vKQoKCiMjIyBTb21lIG90aGVyIHRvb2xzCgpbT3BlbkFQSSBUb29sc10oaHR0cHM6Ly9vcGVuYXBpLnRvb2xzLykgbGlzdCBvZiB1c2VmdWwgdG9vbHMgcmVsYXRlZCB0byBPcGVuQVBJCgpbQXBpVHJlZV0oaHR0cHM6Ly93d3cuYXBpdHJlZS5jb20vKSBob3N0ZWQgQVBJIGRvY3VtZW50YXRpb24gYmFzZWQgb24gT3BlbkFQSQoKW09wZW5Eb2N1bWVudGVyXShodHRwczovL291cm9wZW5jb2RlLmdpdGh1Yi5pby9PcGVuRG9jdW1lbnRlci8pOgpmb3IgY3JlYXRpbmcgYSBkb2N1bWVudGF0aW9uIGZyb20gT3BlbkFQSSBzcGVjCgojIyBIb3cgdG8gZGV2ZWxvcApJbXBvcnQgdGhlIGxpdmluZ2RvY3Mtb3BlbmFwaS5qc29uIGZpbGUgaW50byBpbnNvbW5pYSBhcyBhIGRlc2lnbiBkb2N1bWVudCwgaGVyZSB5b3UgY2FuIHRlc3QgdGhlIEFQSXMgYW5kIGFkZCBuZXcgZW5kcG9pbnRzLgpGb3IgYXV0aGVudGljYXRlZCByZXF1ZXN0cyB5b3UgbmVlZCBhbiBBUEkgdG9rZW4gZnJvbSB5b3VyIExpdmluZ2RvY3MgcHJvamVjdC4KCkFsdGVybmF0aXZlbHkgeW91IGNhbiBkbyBldmVyeXRoaW5nIGluIFZTIENvZGUgdXNpbmcgdGhlIFtPcGVuQVBJIChTd2FnZ2VyKSBFeHRlbnNpb25dKGh0dHBzOi8vbWFya2V0cGxhY2UudmlzdWFsc3R1ZGlvLmNvbS9pdGVtcz9pdGVtTmFtZT00MkNydW5jaC52c2NvZGUtb3BlbmFwaSkuCgojIyMgRXhhbXBsZSBlbmRwb2ludApgYGB5YW1sCiAgL2NoYW5uZWxzL3tjaGFubmVsSGFuZGxlfToKICAgICAgZ2V0OgogICAgICAgIHNlY3VyaXR5OgogICAgICAgIC0gYmVhcmVyQXV0aDogW3B1YmxpYy1hcGk6cmVhZF0KICAgICAgICBwYXJhbWV0ZXJzOgogICAgICAgIC0gbmFtZTogY2hhbm5lbEhhbmRsZQogICAgICAgICAgaW46IHBhdGgKICAgICAgICAgIGRlc2NyaXB0aW9uOiBPcHRpb25hbCBjaGFubmVsSGFuZGxlLiBXaWxsIHJldHVybiBmaXJzdCBjaGFubmVsIG9mIGEgcHJvamVjdCBpZiBub25lIGlzIHBhc3NlZC4KICAgICAgICAgIHJlcXVpcmVkOiB0cnVlCiAgICAgICAgICBzY2hlbWE6CiAgICAgICAgICAgIHR5cGU6IHN0cmluZwogICAgICAgICAgICBleGFtcGxlOiAiIgogICAgICAgIHRhZ3M6CiAgICAgICAgICAtIFByb2plY3QKICAgICAgICBzdW1tYXJ5OiBkZXRhaWxzIGFuZCBjb25maWd1cmF0aW9uIG9mIHRoaXMgcHJvamVjdC4KICAgICAgICBvcGVyYXRpb25JZDogZ2V0Q2hhbm5lbAogICAgICAgIHJlc3BvbnNlczoKICAgICAgICAgICIyMDAiOgogICAgICAgICAgIGRlc2NyaXB0aW9uOiBvawpgYGAKIyMgSG93IHRvIGNvbnRyaWJ1dGUKS2VlcCB0aGUgc2FtZSBzdHJ1Y3R1cmUgYXMgb3VyIGRvY3VtZW50YXRpb24gW2RvY3NdKGh0dHBzOi8vZG9jcy5saXZpbmdkb2NzLmlvL3JlZmVyZW5jZS1kb2NzL3B1YmxpYy1hcGkvKS4KTWFueSBBUEkgZW5kcG9pbnRzIGFyZSBzdGlsbCBtaXNzaW5nLiBQbGVhc2UgbWFyayB0aGUgY2F0ZWdvcnkgYXMgY29tcGxldGVkIHdoZW4gYWxsIGl0cyBlbmRwb2ludHMgYXJlIHJlZ2lzdGVyZWQgaW4gdGhlIHNwZWMuCi0gUHJvamVjdCAgICAgICAgICAgICDinIUKLSBDb21wb3NpdGlvbiBBUEkgICAgIOKchQotIFB1YmxpY2F0aW9ucyAgICAgICAg4pyFICAgIAotIFNlYXJjaCAgICAgICAgICAgICAg4pyFCi0gRG9jdW1lbnQgTGlzdHMgICAgICDinIUKLSBEb2N1bWVudCBDYXRlZ29yaWVzIOKchQotIE1lZGlhIExpYnJhcnkgICAgICAg4pyFCi0gSW1wb3J0cyAgICAgICAgICAgICDinIUKLSBTaXRlbWFwcyAgICAgICAgICAgIOKchQotIE1lbnVzICAgICAgICAgICAgICAg4pyFCi0gUm91dGluZyAgICAgICAgICAgICDinIUKLSBBZGQgRGVsaXZlcnkgU3RhdHVzIOKchQotIEhlYWx0aCAgICAgICAgICAgICAg4pyFCgoKIyMgVE9ET3MgYW5kIGZ1dHVyZSBpZGVhcwotIGFkZCBhbGwgdGhlIFB1YmxpYyBBUEkgZW5kcG9pbnRzIGZyb20gb3VyIGRvY3Mg4pyFCi0gbWFrZSBhdmFpbGFibGUgdGhlIE9BUyBwdWJsaWNseSDinIUKLSBlbWJlZGQgdGhlIFN3YWdnZXIgVUkgaW4gb3VyIGRvY3MKLSBzdXBwb3J0IHRvIGV4dGVuZCBkb3duc3RyZWFtIHdpdGggZG93bnN0cmVhbSBkZWNsYXJhdGlvbgotIGdlbmVyYXRlIHRoZSBzcGVjIGJ5IHRoZSBzZXJ2ZXIK readmeEtag: '"b84dc2e5654effb7f8db83e254ca2257cbb122b1"' readmeLastModified: Thu, 18 Jan 2024 16:02:34 GMT repositoryId: 537367140 description: >- Moved to https://docs.livingdocs.io/openapi.json. OpenAPI (Swagger) spec for Livingdocs Public API created: '2022-09-16T08:16:19Z' updated: '2025-03-13T23:32:29Z' language: null archived: true stars: 0 watchers: 5 forks: 2 owner: livingdocsIO logo: https://avatars.githubusercontent.com/u/3775168?v=4 repoEtag: '"14d9558e9d67cb713c5983f897ab3c3f3ff3fbaf93bebf2792070d590557276b"' repoLastModified: Thu, 13 Mar 2025 23:32:29 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/brokenprogrammer/mswg-gen v3: true id: 299afd28415cb2ebe143e6f0e0b5f3c8 repositoryMetadata: base64Readme: >- IyBtc3dnLWdlbgpDdXN0b20gT3BlbkFQSSBnZW5lcmF0b3IgdGhhdCBpcyB1c2VkIHRvIGdlbmVyYXRlIHN3YWdnZXIgZG9jdW1lbnRhdGlvbiB1c2luZyBjb21tZW50cy4KClRoZSBwcm9qZWN0IGN1cnJlbnRseSBoYXMgc3VwcG9ydCBmb3IgQyAmIEdvIGxhbmd1YWdlIGJ1dCBjYW4gdGVjaG5pY2FsbHkgc3VwcG9ydCBhbnkgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2UgCnRoYXQgbWFrZXMgdXNlIG9mIEMgc3R5bGUgY29tbWVudHMuIFNvbWUgcGFyc2luZyBoYXMgdG8gYmUgYWRkZWQgaG93ZXZlci4KClRoZSBnb2FsIG9mIHRoaXMgcHJvamVjdCBpcyB0byBoYXZlIGEgZmFzdCBzd2FnZ2VyIGdlbmVyYXRvciB0aGF0IGlzIGV4Y2x1ZGVkIGZyb20gdGhlIHByb2plY3QgdGhhdCBjYW4gZWFzaWx5IGJlIGludGVncmF0ZWQKaW50byBhbnkgYnVpbGQgcGlwZWxpbmUuCgojIyBVc2FnZQoKRmlyc3QgeW91IG5lZWQgdG8gY3JlYXRlIGEgY29uZmlnLmluaSBmaWxlIHdpdGhpbiB5b3VyIHByb2plY3QgZGlyZWN0b3J5LiBBbiBleGFtcGxlIGNvbmZpZ3VyYXRpb24gbG9va3MgbGlrZSB0aGUgZm9sbG93aW5nOgoKYGBgaW5pCjsgVGhpcyBpcyBhIGNvbW1lbnQKW21zd2dlbl0KdGl0bGUgPSBTYW1wbGUKIyBUaGlzIGlzIGFsc28gYSBjb21tZW50CmRlc2NyaXB0aW9uID0gVGhpcyBpcyBhIHNhbXBsZSBhcHBsaWNhdGlvbgp0ZXJtc29mc2VydmljZSA9IGh0dHBzOi8vZ29vZ2xlLnNlCgpbbXN3Z2VuLmNvbnRhY3RdCm5hbWUgPSBTdXBwb3J0CnVybCA9IGh0dHBzOi8vZ29vZ2xlLnNlCmVtYWlsID0gc3VwcG9ydEBnb29nbGUuc2UKClttc3dnZW4ubGljZW5zZV0KbmFtZSA9IE1JVAp1cmwgPSBodHRwczovL2dvb2dsZS5zZQoKW21zd2dlbi52ZXJzaW9uXQp2ZXJzaW9uPSB2MQpgYGAKCkV4cGxhbmF0aW9uIGZvciB0aGUgZGlmZmVyZW50IGNvbmZpZ3VyYXRpb24gc2V0dGluZ3MgYXJlIGV4cGxhaW5lZCBpbiB0aGUgZm9sbG93aW5nIHRhYmxlLgp8IENvbmZpZ3VyYXRpb24gfCBEZXNjcmlwdGlvbiAgIHwgRXhhbXBsZSAgICAgICB8CnwgLS0tLS0tLS0tLS0tLSB8IC0tLS0tLS0tLS0tLS0gfCAtLS0tLS0tLS0tLS0tIHwKfCB0aXRsZSAgfCBUaGUgdGl0bGUgb2YgeW91ciBhcHBsaWNhdGlvbi4gVGhpcyB3aWxsIGJlIGRpc3BsYXllZCBvbiB5b3VyIHN3YWdnZXIgcGFnZS4gfCBNeSBBcHBsbGljYXRpb24gfAp8IGRlc2NyaXB0aW9uICB8IERlc2NyaXB0aW9uIG9mIHlvdXIgYXBwbGljYXRpb24uIFdpbGwgYmUgZGlzcGxheWVkIG9uIHlvdXIgc3dhZ2dlciBwYWdlLiAgfCBUaGlzIGlzIGEgc2FtcGxlIGFwcGxpY2F0aW9uIHwKfCB0ZXJtc29mc2VydmljZSAgfCBBIGxpbmsgdG8gdGhlIHRlcm1zIG9mIHNlcnZpY2UgZm9yIHlvdXIgQVBJLi4gIHwgaHR0cHM6Ly9leGFtcGxlLmNvbSB8CnwgY29udGFjdC5uYW1lICB8IERpc3BsYXkgbmFtZSBmb3IgeW91ciBjb250YWN0LiAgfCBTdXBwb3J0IHwKfCBjb250YWN0LnVybCAgfCBDb250YWN0IFVSTC4uICB8IGh0dHBzOi8vZXhhbXBsZS5jb20gfAp8IGNvbnRhY3QuZW1haWwgIHwgQ29udGFjdCBlbWFpbC4gIHwgc3VwcG9ydEBleGFtcGxlLmNvbSB8CnwgbGljZW5zZS5uYW1lICB8IE5hbWUgb2YgdGhlIGxpY2Vuc2UgZm9yIHlvdXIgQVBJLiAgfCBNSVQgfAp8IGxpY2Vuc2UudXJsICB8IFVSTCB0byBsaWNlbnNlIG9mIHlvdXIgQVBJLiAgfCBodHRwczovL2V4YW1wbGUuY29tL2xpY2Vuc2UgfAp8IHZlcnNpb24udmVyc2lvbiB8IFZlcnNpb24gb2YgeW91ciBBUEkuIHwgdjEgfAoKQWZ0ZXIgeW91IGhhdmUgc2V0IHVwIHlvdXIgY29uZmlndXJhdGlvbiBmaWxlIHlvdSBjYW4gYnVpbGQgdGhlIGFwcGxpY2F0aW9uIGJ5IGVudGVyaW5nIHRoZSBgc3JjYCBkaXJlY3RvcnkgYW5kIHJ1biBgbWFrZWAuCgpPbmNlIGRvbmUgeW91IHNob3VsZCBoYXZlIGFjY2VzcyB0byB0aGUgZXhlY3V0YWJsZSBgbXN3Z19nZW5gIHRoYXQgeW91IGNhbiBwbGFjZSB3aGVyZSB5b3Ugd2FudCBpdC4KCkZpbmFsbHkgc29tZSBtb2RpZmljYXRpb24gaXMgbmVlZGVkIGluIHRoZSBjb2RlIG9mIHlvdXIgYXBwbGljYXRpb24uIG1zd2dfZ2VuIHVzZXMgdHdvIHR5cGVzIG9mIGNvbW1lbnQgdG8gc3BlY2lmeSB0d28gZGlmZmVyZW50IEFQSSB0eXBlcyByb3V0ZXMgYW5kIHJvdXRlIHR5cGVzLgoKIyMjIFJvdXRlcwoKQSBuZXcgcm91dGUgaXMgc3BlY2lmaWVkIGJ5IHdyaXRpbmcgYSBjb21tZW50IGluIHlvdXIgY29kZSB3aXRoIHRoZSBmb2xsb3dpbmcgZm9ybWF0OgpgYGBDCi8vIEBSb3V0ZShyb3V0ZSwgbWV0aG9kLCByZXR1cm5fdHlwZSkKYGBgCgpBbiBleGFtcGxlIHdvdWxkIGJlCgpgYGBDCi8vIEBSb3V0ZSgvYXBpL2xvZ2luLCBwb3N0LCB1c2VyX29iamVjdCkKYGBgCgpTb21lIHJvdXRlcyBkbyBub3QgcmV0dXJuIGFueSBqc29uIG9iamVjdHMgaGVuY2UgdGhlIGByZXR1cm5fdHlwZWAgY2FuIGFsc28gYmUgZW1pdHRlZWQgbGlrZSBzbzoKCmBgYEMKLy8gQFJvdXRlKC9hcGkvcmVnaXN0ZXIsIHBvc3QpCmBgYAoKIyMjIFJvdXRlIFR5cGVzCgpSb3V0ZSB0eXBlcyBhcmUgYWRkZWQgYnkgYW5ub3RhdGluZyB5b3VyIHN0cnVjdHVyZXMgdXNpbmcgdGhlIGBAUm91dGVUeXBlYCBjb21tYW5kLgpUaGlzIGFsbG93cyBtc3dnX2dlbiB0byBhZGQgdGhlIHR5cGUgdG8gdGhlIGxpc3Qgb2Yga25vd24gYHJldHVybl90eXBlc2AgZm9yIHJvdXRlcyBpbiBvcmRlciB0byBnZW5lcmF0ZSBqc29uIHdpdGhpbiBzd2FnZ2VyLgoKRXhhbXBsZToKYGBgQwovLyBAUm91dGVUeXBlCnR5cGVkZWYgc3RydWN0IHsKICAgIHVfaW50MzJfdCBzdGF0ZVs1XTsKICAgIHVfaW50NjRfdCBjb3VudDsKICAgIHVfaW50OF90IGJ1ZmZlcltTSEExX0JMT0NLX0xFTkdUSF07Cn0gU0hBMV9DVFg7CmBgYAoKIyMjIEludGVncmF0aW5nIGludG8geW91ciBwcm9qZWN0CgpUaGUgb3V0cHV0IG9mIG1zd2dfZ2VuIGNhbiBlYXNpbHkgYmUgb3V0cHV0IGluIGEgZmlsZSBsaWtlIHNvOgpgYGBiYXNoCi4vbXN3Z19nZW4gPiBleGFtcGxlLnR4dApgYGAKClRoZSBnZW5lcmF0ZWQgb3V0cHV0IGNhbiB0aGVuIGJlIHVzZWQgdG9nZXRoZXIgd2l0aCB0aGUgc3RhdGljIHN3YWdnZXIgc2l0ZSBwcm92aWRlZCAKd2l0aGluIHRoaXMgcmVwb3NpdG9yeTogPGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLXVpPgoKVGhpcyBpcyBlYXN5IHRvIHNldCB1cCBieSBkb3dubG9hZGluZyB0aGUgbGF0ZXN0IHJlbGVhc2UgYW5kIHVzaW5nIHRoZSBzdGF0aWMgSFRNTC9KUy9DU1Mgd2l0aGluIHRoZSBkb3dubG9hZGVkIGAvZGlzdGAgZm9sZGVyLgoKIyMgRGlzY2xhaW1lcgoKVGhpcyBhcHBsaWNhdGlvbiB3YXMgd3JpdHRlbiBpbiBhIHZlcnkgc2hvcnQgZHVyYXRpb24gYW5kIGEgbG90IG9mIHNob3J0Y3V0cyB3YXMgdGFrZW4uIElmIHlvdSBmaW5kIGJ1Z3MgcGxlYXNlIGZpbGUgYW4gaXNzdWUgb3IgYSBwdWxsIHJlcXVlc3QuIAoKIyBMaWNlbnNlCgpNSVQgTGljZW5zZQoKQ29weXJpZ2h0IChjKSAyMDIyIE9za2FyIE1lbmRlbAoKUGVybWlzc2lvbiBpcyBoZXJlYnkgZ3JhbnRlZCwgZnJlZSBvZiBjaGFyZ2UsIHRvIGFueSBwZXJzb24gb2J0YWluaW5nIGEgY29weQpvZiB0aGlzIHNvZnR3YXJlIGFuZCBhc3NvY2lhdGVkIGRvY3VtZW50YXRpb24gZmlsZXMgKHRoZSAiU29mdHdhcmUiKSwgdG8gZGVhbAppbiB0aGUgU29mdHdhcmUgd2l0aG91dCByZXN0cmljdGlvbiwgaW5jbHVkaW5nIHdpdGhvdXQgbGltaXRhdGlvbiB0aGUgcmlnaHRzCnRvIHVzZSwgY29weSwgbW9kaWZ5LCBtZXJnZSwgcHVibGlzaCwgZGlzdHJpYnV0ZSwgc3VibGljZW5zZSwgYW5kL29yIHNlbGwKY29waWVzIG9mIHRoZSBTb2Z0d2FyZSwgYW5kIHRvIHBlcm1pdCBwZXJzb25zIHRvIHdob20gdGhlIFNvZnR3YXJlIGlzCmZ1cm5pc2hlZCB0byBkbyBzbywgc3ViamVjdCB0byB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnM6CgpUaGUgYWJvdmUgY29weXJpZ2h0IG5vdGljZSBhbmQgdGhpcyBwZXJtaXNzaW9uIG5vdGljZSBzaGFsbCBiZSBpbmNsdWRlZCBpbiBhbGwKY29waWVzIG9yIHN1YnN0YW50aWFsIHBvcnRpb25zIG9mIHRoZSBTb2Z0d2FyZS4KClRIRSBTT0ZUV0FSRSBJUyBQUk9WSURFRCAiQVMgSVMiLCBXSVRIT1VUIFdBUlJBTlRZIE9GIEFOWSBLSU5ELCBFWFBSRVNTIE9SCklNUExJRUQsIElOQ0xVRElORyBCVVQgTk9UIExJTUlURUQgVE8gVEhFIFdBUlJBTlRJRVMgT0YgTUVSQ0hBTlRBQklMSVRZLApGSVRORVNTIEZPUiBBIFBBUlRJQ1VMQVIgUFVSUE9TRSBBTkQgTk9OSU5GUklOR0VNRU5ULiBJTiBOTyBFVkVOVCBTSEFMTCBUSEUKQVVUSE9SUyBPUiBDT1BZUklHSFQgSE9MREVSUyBCRSBMSUFCTEUgRk9SIEFOWSBDTEFJTSwgREFNQUdFUyBPUiBPVEhFUgpMSUFCSUxJVFksIFdIRVRIRVIgSU4gQU4gQUNUSU9OIE9GIENPTlRSQUNULCBUT1JUIE9SIE9USEVSV0lTRSwgQVJJU0lORyBGUk9NLApPVVQgT0YgT1IgSU4gQ09OTkVDVElPTiBXSVRIIFRIRSBTT0ZUV0FSRSBPUiBUSEUgVVNFIE9SIE9USEVSIERFQUxJTkdTIElOIFRIRQpTT0ZUV0FSRS4K readmeEtag: '"d255a8297272e5e3e5fe8a8c6069408b59852474"' readmeLastModified: Sun, 26 Jun 2022 19:51:25 GMT repositoryId: 507563615 description: >- Custom OpenAPI generator that is used to generate swagger documentation using comments. created: '2022-06-26T12:07:17Z' updated: '2022-06-26T12:39:07Z' language: C archived: false stars: 0 watchers: 1 forks: 0 owner: brokenprogrammer logo: https://avatars.githubusercontent.com/u/16289144?v=4 license: MIT repoEtag: '"d4ef2130acc593b9e82f84d01b11b94e41d416c4f2c53bf04e09d41f5f8aa970"' repoLastModified: Sun, 26 Jun 2022 12:39:07 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/codessintheclassroom/api-reference-solution v3: true repositoryMetadata: repositoryId: 179300089 description: >- A specification for the API used to provide functionality for the animal shelter's web frontend created: '2019-04-03T13:50:20Z' updated: '2019-05-06T09:53:48Z' language: HTML archived: false stars: 0 watchers: 2 forks: 0 owner: codessintheclassroom logo: https://avatars.githubusercontent.com/u/49059753?v=4 repoEtag: '"e0a72a27ed4e18aae3617084186e3e5939dcd04e5aa2b1d59a2de541ed89b16e"' repoLastModified: Mon, 06 May 2019 09:53:48 GMT foundInMaster: true id: 81ff3479a8a482ef3a03bf6af4fb844c - source: openapi3 tags repository: https://github.com/nktks/go-oapi-gen v3: true id: c2a2af59c5cc5e67da902348ba3b5b81 repositoryMetadata: repositoryId: 234057661 description: code generator from openapi3 and go text/template created: '2020-01-15T10:39:27Z' updated: '2022-02-04T05:05:25Z' language: Smarty archived: false stars: 0 watchers: 1 forks: 0 owner: nktks logo: https://avatars.githubusercontent.com/u/7553415?v=4 repoEtag: '"fc05a950eef469f4f0f3384e1d79ad5ac798c875de8e5d4778615de32a78effe"' repoLastModified: Fri, 04 Feb 2022 05:05:25 GMT foundInMaster: true oldLocations: - https://github.com/nakatamixi/go-oapi-gen - source: openapi3 tags repository: https://github.com/silwathge/microservices v3: true repositoryMetadata: base64Readme: >- IyBBUEkgUG9ydGFsOiBTYW1wbGUgQVBJIEltcGxlbWVudGF0aW9uIChGb3IgT3BlbkFQSSBWMykKCiMjIFNhbXBsZSBTb2x1dGlvbiBpcyBhdmFpbGFibGUgb24gQVdTIChFQzIpIGFzIGRvY2tlciBjb250YWluZXJzCgoKQXBwOiBnbG9iYWwtYXBpLW1hcmtldHBsYWNlLWFwcAoKREI6IGFwaS1wb3N0Z3JlcwoKTmV0d29yazogZ2xvYmFsLWFwaS1tYXJrZXRwbGFjZS1hcHAtbmV0d29yayhicmlkZSkKClZvbHVtZTogcG9zdGdyZXMtZGF0YWJhc2UtZGF0YS12b2x1bWUKCgojIyBTZXJ2ZXIgVVJMIGFuZCBQb3J0CgpodHRwOi8vZWMyLTEzLTIyOS02MC0yMDEuYXAtc291dGhlYXN0LTEuY29tcHV0ZS5hbWF6b25hd3MuY29tOjgwODAvCgoKIyMgUG9zdG1hbiBTY3JpcHQgCgp3aXRoIG90aGVyIGZpbGVzIHVzZWQgZm9yIHVwbG9hZGVkaW5nIHdpbGwgYmUgc2VudCBzZXBhcmF0ZWx5IHZpYSBlbWFpbC4gCgpUaGUgemlwIGZpbGUg4oCcUG9zdG1hblNjcmlwdEFuZEZpbGVzVG9VcGxvYWQuemlw4oCdIHdpbGwgYWxzbyBhdmFpbGFibGUgd2l0aCBzb3VyY2UgY29kZSBpbiB0aGUgR2l0SHViIHJlcG8KCgogMS4gVXNlIGVudmlyb25tZW50IHZhcmlhYmxlIArigJxob3N04oCdICB3aXRoIHZhbHVlIGVjMi0xMy0yMjktNjAtMjAxLmFwLXNvdXRoZWFzdC0xLmNvbXB1dGUuYW1hem9uYXdzLmNvbQoKMi4gUGxhY2Ugb3RoZXIgZmlsZXMgaW4gdGhlIFBvc3RtYW4gd29ya2luZyBkaXJlY3RvcnkKCjMuIERlZmF1bHQgdXNlciBpbiBkYiAKCgkgVXNlciA6IHVzZXJuYW1lCgkgCgkgUGFzc3dvcmQ6IHBhc3N3b3JkCgoKCgoKIyMgU291cmNlIGNvZGUgaXMgYXZhaWxhYmxlIG9uIEdpdEh1YiBhcyBhIHB1YmxpYyByZXBvIDoKCmh0dHBzOi8vZ2l0aHViLmNvbS9zaWx3YXRoZ2UvMTAxLWRpZ2l0YWwtY29kaW5nLWNoYWxsZW5nZQoKCgojIyBEb2NrZXIgaW1hZ2UgaXMgYXZhaWxhYmxlIG9uIERvY2tlSHViIDp1c2UgdGhlIG9uZSB3aXRoIGhpZ2hlc3QgdGFnCgpodHRwczovL2h1Yi5kb2NrZXIuY29tL3JlcG9zaXRvcnkvZG9ja2VyL2thcGlsYXMvZ2xvYmFsLWFwaS1tYXJrZXRwbGFjZS1hcHAKCgprYXBpbGFzL2dsb2JhbC1hcGktbWFya2V0cGxhY2UtYXBwOjAuMC4zLVNOQVBTSE9UCgoKCiMjIERvY2tlciBDb21wb3NlIEZpbGUKSXMgYXZhaWxhYmxlIHdpdGggc291cmNlIGNvZGUgaW4gdGhlIHJlcG8uCgoKIyMjIFRvIGRlcGxveSBmcm9tIGltYWdlcyBkaXJlY3RseToKCkNhbiBiZSB1c2VkIGFzIGl0IGlzIHRvIGRlcGxveSBzZXJ2aWNlIHdpdGggZGIgdG8gZG9ja2VyCgoKIyMjIFRvIGJ1aWxkIGltYWdlOgpDb21tZW50IG91dCBpbWFnZSBhbmQgdW5jb21tZW50IGFsbCB0aHJlZSBsaW5lcyB3aGljaCBpcyBjb21tZW50ZWQgb3V0IGluIHRoZSBjdXJyZW50IGZpbGUKRG8gIG12biBjbGVhbiBwYWNrYWdlCgpUaGVuIHJ1biBkb2NrZXItY29tcG9zZSB1cCAtIG5ldyBpbWFnZSB3aWxsIGJlIGNyZWF0ZWQgYW5kIHNlcnZpY2VzIHdpbGwgYmUgZGVwbG95ZWQuCgoKIyMgU2VydmVyIFVSTCBhbmQgUG9ydApodHRwOi8vZWMyLTEzLTIyOS02MC0yMDEuYXAtc291dGhlYXN0LTEuY29tcHV0ZS5hbWF6b25hd3MuY29tOjgwODAvCgoKIyMgU3dhZ2dlciBBUEkgRG9jIDoKaHR0cDovL2VjMi0xMy0yMjktNjAtMjAxLmFwLXNvdXRoZWFzdC0xLmNvbXB1dGUuYW1hem9uYXdzLmNvbTo4MDgwL3YyL2FwaS1kb2NzCgoKIyMgU3dhZ2dlciBVSSA6Cmh0dHA6Ly9lYzItMTMtMjI5LTYwLTIwMS5hcC1zb3V0aGVhc3QtMS5jb21wdXRlLmFtYXpvbmF3cy5jb206ODA4MC9zd2FnZ2VyLXVpLmh0bWwKCgojIyBBY3R1YXRvciAod2l0aCBkZWZhdWx0IGV4cG9zZWQgZW5kcG9pbnRzKSAgIDoKCmh0dHA6Ly9lYzItMTMtMjI5LTYwLTIwMS5hcC1zb3V0aGVhc3QtMS5jb21wdXRlLmFtYXpvbmF3cy5jb206ODA4MC9hY3R1YXRvcgoKCgoKCgoK readmeEtag: '"ed0bea2f80032d89d3b254886f54902b865b976f"' readmeLastModified: Sun, 12 Jul 2020 05:21:56 GMT repositoryId: 278100638 description: >- This has a Spring boot REST API built to upload Swagger API Documents as Json to web service created: '2020-07-08T13:44:12Z' updated: '2020-07-13T23:04:36Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: silwathge logo: https://avatars.githubusercontent.com/u/68016031?v=4 repoEtag: '"aa48026064f3086eb906616dfba305f363d5d8a5dd760b16771588b582c65841"' repoLastModified: Mon, 13 Jul 2020 23:04:36 GMT foundInMaster: true category: Server Implementations id: f4f5e6d9d29f098aaa1f521167659ce5 - source: openapi3 tags repository: https://github.com/jimleuk/uptimerobot-swagger v3: true repositoryMetadata: base64Readme: >- IyBVcHRpbWVyb2JvdC1zd2FnZ2VyCgpUaGlzIHJlcG8gY29udGFpbnMgYW4gdW5vZmZpY2lhbCBzd2FnZ2VyIGRvY3VtZW50IG9mIHRoZSBVcHRpbWVSb2JvdCBBUEkgdjIuIEl0IGlzIHdyaXR0ZW4gdG8gdGhlIFtPcGVuQXBpMyBzcGVjXShbaHR0cHM6Ly9zd2FnZ2VyLmlvL2RvY3Mvc3BlY2lmaWNhdGlvbi9hYm91dC9dKGh0dHBzOi8vc3dhZ2dlci5pby9kb2NzL3NwZWNpZmljYXRpb24vYWJvdXQvKSkgYW5kIHRlc3RlZCB0byByZW5kZXIgY29ycmVjdGx5IGluIFtzd2FnZ2VyLmlvIGVkaXRvcl0oW2h0dHBzOi8vZWRpdG9yLnN3YWdnZXIuaW8vXShodHRwczovL2VkaXRvci5zd2FnZ2VyLmlvLykpLgoKKipOb3RlKio6IFRoaXMgaXMgbm90IGEgMTAwJSByZXByZXNlbnRhdGlvbiBvZiB0aGUgQVBJIFNwZWMgYW5kIHVwZGF0ZXMgYXJlIGJlc3QgZWZmb3J0LiBQbGVhc2Ugc2VlIEtOT1dOX0lTU1VFUy4gUGxlYXNlIHJlZmVyIHRvIFtodHRwczovL3VwdGltZXJvYm90LmNvbS9hcGldKGh0dHBzOi8vdXB0aW1lcm9ib3QuY29tL2FwaSkgZm9yIHRoZSBvZmZpY2lhbCBBUEkgZG9jdW1lbnRhdGlvbi4KCioqTm90ZSoqOiBUaGlzIGRvY3VtZW50IGlzIG5vdCBhZmZsaWF0ZWQgd2l0aCBVcHRpbWUgUm9ib3QgU2VydmljZSBQcm92aWRlciBMdGQuIElzc3VlcyBhbmQvb3Igc3VwcG9ydCBzaG91bGQgcmVsYXRpbmcgdG8gdGhpcyBsaWJyYXJ5IGJlIGRpcmVjdGVkIHRvIHRoZSBwcm9qZWN0J3MgW2lzc3VlIHRyYWNrZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9qaW1sZXVrL3VwdGltZXJvYm90LXN3YWdnZXIvaXNzdWVzKS4KCiMjIFVzYWdlCgpCZXN0IHZpZXdlZCBpbiBTd2FnZ2VyIFVJIChodHRwczovL3N3YWdnZXIuaW8vdG9vbHMvc3dhZ2dlci11aS8pLiBZb3Ugc2hvdWxkIGFsc28gYmUgYWJsZSB0byBpbXBvcnQgdGhpcyBkb2N1bWVudCB3aGVyZSBvcGVuQVBJMyBzcGVjIGlzIHN1cHBvcnRlZC4KCiMjIFJlbGF0ZWQKCiogW09mZmljaWFsIFVwdGltZXJvYm90IEFQSSBkb2N1bWVudGF0aW9uXShodHRwczovL3VwdGltZXJvYm90LmNvbS9hcGkpCiogW1Vub2ZmaWNpYWwgVXB0aW1lcm9ib3QgY2xpZW50IGZvciBub2RlL2Jyb3dzZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS9qaW1sZXVrL3VwdGltZXJvYm90LWpzKQoqIFtVbm9mZmljaWFsIFVwdGltZXJvYm90IEFQSSB0eXBlc2NyaXB0IHR5cGUgZGVmaW50aW9ucyBvbmx5XShodHRwczovL2dpdGh1Yi5jb20vamltbGV1ay91cHRpbWVyb2JvdC10eXBlcykKCiMjIExJQ0VOQ0UKCk1JVA== readmeEtag: '"c7d2980456ae8724fb1e65d8372b55823e32003b"' readmeLastModified: Mon, 06 Jan 2020 23:10:28 GMT repositoryId: 229309997 description: Unofficial swagger 3.0 documentation for the UptimeRobot.com API v2 spec created: '2019-12-20T17:40:36Z' updated: '2020-01-06T23:10:47Z' language: null archived: false stars: 0 watchers: 1 forks: 1 owner: jimleuk logo: https://avatars.githubusercontent.com/u/373596?v=4 license: MIT repoEtag: '"d871d94715c3a744b44e824cd9bdfbdbd5811ae5ddad30a604088dd6ba2666d0"' repoLastModified: Mon, 06 Jan 2020 23:10:47 GMT foundInMaster: true category: SDK id: ecc7a61b63256567835231c06282540b - source: openapi3 tags repository: https://github.com/marcbru107/openapi3-request-finder v3: true repositoryMetadata: base64Readme: >- IyBPcGVuYXBpMy1yZXF1ZXN0LWZpbmRlcgoKIyMjIyMjIE9ubHkgdGVzdGV0IGZvciBvcGVuYXBpOiAzLjAuMQoKIyMgV2h5OgoKIyMjIyBIZWxwcyB5b3UgdG8gZ2V0IHRoZSByZXF1ZXN0IG9iamVjdHMgZnJvbSBhIG9wZW5hcGktc3dhZ2dlci1maWxlICAgCgoKVXNlIGRlLmJydWNrbS5vcGVuYXBpcmVxdWVzdGZpbmRlci5PcGVuQXBpUmVxdWVzdEZpbmRlcjoKLSBwdWJsaWMgSGFzaE1hcDxTdHJpbmcsIFJlcXVlc3RzRnJvbVlhbWw+IGdldFJlcXVlc3RzTWFwKGZpbmFsIFN0cmluZyByZXF1ZXN0VHlwZSkKICAgICAqIFBhcnNlIHRoZSBvcGVuIGFwaSBpbnB1dCBmaWxlIGZvciB0aGUgcmVxdWVzdGVkIHJlcXVlc3QtdHlwZQogICAgICogQHBhcmFtIHJlcXVlc3RUeXBlIC0+IGpzb24scHJvdG9idWYsLi4KICAgICAqIEByZXR1cm4gTWFwIHdpdGggdGhlIHVybCBhcyBrZXkgYW5kIHRoZSByZXF1ZXN0IGFzIHZhbHVlIC0+IGluY2x1ZGluZyBvbmx5IGVudHJpZXMgZnJvbSB0eXAgcmVxdWVzdFR5cGUKICAgIAojIyBEZXBlbmRlbmNpZXM6CiAgICAtIG9yZy55YW1sLnNuYWtleWFtbAogICAgLSBvcmcuanVuaXQuanVwaXRlci5hcGkKICAgIC0gb3JnLnNsZjRqLkxvZ2dlcgogICAgLSBjb20uZ29vZ2xlLnByb3RvYnVmOnByb3RvYnVmLWphdmEKICAgIC0gY29tLmdvb2dsZWNvZGUucHJvdG9idWYtamF2YS1mb3JtYXQK readmeEtag: '"b4c19878a0177ddd210eb7bd56ac32126983ed4c"' readmeLastModified: Mon, 04 May 2020 21:22:59 GMT repositoryId: 260700799 description: null created: '2020-05-02T14:10:17Z' updated: '2020-05-04T21:23:02Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: MarcBru107 logo: https://avatars.githubusercontent.com/u/25484756?v=4 license: Apache-2.0 repoEtag: '"316196408decae4bb5cfa569b1f96df164fa98d684593c31122c3d27066568ad"' repoLastModified: Mon, 04 May 2020 21:23:02 GMT foundInMaster: true category: - Converters - Parsers id: 9f431e17c0f626c479ef790bc88ca437 - source: openapi3 tags repository: https://github.com/mariusconjeaud/nodejs-az-functions-api v3: true repositoryMetadata: base64Readme: >- IyBSZXF1aXJlbWVudHMKWW91IG5lZWQgdG8gaGF2ZSBpbnN0YWxsZWQgOgoqIE5vZGUuanMgdmVyc2lvbiAxMiBMVFMgZnJvbSBbaGVyZV0oaHR0cHM6Ly9ub2RlanMub3JnL2VuL2Rvd25sb2FkLykuCiogQXp1cmUgRnVuY3Rpb25zIENvcmUgVG9vbHMuIFRvIGluc3RhbGwgaXQsIGV4ZWN1dGUgdGhlIGZvbGxvd2luZyBjb21tYW5kIDoKYGBgYmFzaApucG0gaW5zdGFsbCAtZyBhenVyZS1mdW5jdGlvbnMtY29yZS10b29scwpgYGAKCiMgU3RhcnQgQXp1cmUgRnVuY3Rpb25zCkF6dXJlIEZ1bmN0aW9ucyBDb3JlIFRvb2xzIGVuYWJsZXMgeW91IHRvIGV4ZWN1dGUgeW91ciBGdW5jdGlvbnMgcnVudGltZSBsb2NhbGx5LiBUbyBkbyBzbywgZXhlY3V0ZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQgOgpgYGBiYXNoCmZ1bmMgc3RhcnQKYGBgCklmIGV2ZXJ5dGhpbmcgaXMgd29ya2luZyBwcm9wZXJseSwgeW91IHNob3VsZCBnZXQgdGhlIGZvbGxvd2luZyBDTEkgb3V0cHV0IDoKIVtDTEkgb3V0cHV0XShkb2NzL2Z1bmNfc3RhcnRfb3V0cHV0LnBuZyAiU3VjY2Vzc2Z1bCBDTEkgT3V0cHV0IikKVG8gdmVyaWZ5IHRoYXQgdGhlIEFQSSBpcyB3b3JraW5nIHByb3Blcmx5LCBnbyB0byB0aGUgZm9sbG93aW5nIFVSTCA6IDxodHRwOi8vbG9jYWxob3N0OjMwMDEvYXBpL2NhdHM+LgoKIyMgTm90ZSBvbiBwb3RlbnRpYWwgZXJyb3JzCkRlcGVuZGluZyBvbiB0aGUgY29uZmlndXJhdGlvbiBvZiB5b3VyIG1hY2hpbmUsIHlvdSBtaWdodCBuZWVkIHRvIGluc3RhbGwgdGhlIFsuTkVUIENvcmUgMi54IFNES10oaHR0cHM6Ly9kb3RuZXQubWljcm9zb2Z0LmNvbS9kb3dubG9hZCkuCgpUaGlzIGlzIGJlY2F1c2UgRnVuY3Rpb25zIGFyZSBidWlsdCBvbiB0b3Agb24gLk5FVCBDb3JlLiBXaGVuIHlvdSBzdGFydCB0aGUgRnVuY3Rpb25zIHJ1bnRpbWUsIGl0IHNwaW5zIHVwIGEgLk5FVCBDb3JlIHJ1bnRpbWUsIGluIHdoaWNoIHRoZSBkZXNpcmVkIHJ1bnRpbWUgb2YgeW91ciBmdW5jdGlvbnMgLSBOb2RlLmpzIGluIG91ciBjYXNlIC0gcnVucy4KCllvdSBub3JtYWxseSBkb24ndCBuZWVkIHRoaXMgd2l0aCB0aGlzIHByb2plY3QsIGFzIHdlIG1ha2UgdXNlIG9mIHRoZSBleHRlbnNpb25zIGJ1bmRsZS4gQnV0IGp1c3QgaW4gY2FzZSwgaGVyZSdzIGEgd2FybmluZy4uLgo= readmeEtag: '"0c8e8ebb75451d71452af6f766681493132d8373"' readmeLastModified: Mon, 30 Mar 2020 09:58:00 GMT repositoryId: 250509410 description: Template for Azure Functions API with Node.js runtime created: '2020-03-27T10:51:27Z' updated: '2020-03-30T09:58:42Z' language: HTML archived: false stars: 0 watchers: 0 forks: 0 owner: mariusconjeaud logo: https://avatars.githubusercontent.com/u/7679761?v=4 repoEtag: '"3064599f52e9066eaaf9fa2a24129f1ddf0485a7417cacacdc2f484d2e9f4f86"' repoLastModified: Mon, 30 Mar 2020 09:58:42 GMT foundInMaster: true category: - SDK - Server Implementations id: 6f2e7cf6c04a75b2dbea1729ba73de04 - source: openapi3 tags repository: https://github.com/uuf6429/oas-php v3: true repositoryMetadata: base64Readme: >- IyBPQVMtUEhQCgpbIVtNaW5pbXVtIFBIUCBWZXJzaW9uXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3BocC0lM0UlM0QlMjA3LjQtODg5MkJGLnN2ZyldKGh0dHBzOi8vcGhwLm5ldC8pClshW0xpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vYmFkZ2UvbGljZW5zZS1NSVQtYmx1ZS5zdmcpXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vdXVmNjQyOS9vYXMtcGhwL21hc3Rlci9MSUNFTlNFKQpbIVtQYWNrYWdpc3RdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vcGFja2FnaXN0L3YvdXVmNjQyOS9vYXMtcGhwLnN2ZyldKGh0dHBzOi8vcGFja2FnaXN0Lm9yZy9wYWNrYWdlcy91dWY2NDI5L29hcy1waHApCgpPQVMtUEhQOiBQSFAgSW1wbGVtZW50YXRpb24gb2YgdGhlIE9wZW5BUEkgU3BlYy4KClRoaXMgbGlicmFyeSBpcyBhbiBpbXBsZW1lbnRhdGlvbiBvZiB0aGUgW09wZW5BUEkgdjMgc3BlY2lmaWNhdGlvbl0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMy4wLjMubWQpLgoKIyMgVXNhZ2UKClRoaXMgbGlicmFyeSBpcyBqdXN0IGEgYnVuY2ggb2YgdmFsdWUgb2JqZWN0cy4gVGhlIG1haW4gZW50cnkgcG9pbnQsIChhcyBpcyB0aGUgY2FzZSB3aXRoIHRoZSBzcGVjaWZpY2F0aW9uKSwgaXMgdGhlICoqRG9jdW1lbnQqKiBjbGFzczoKCmBgYHBocAokZG9jdW1lbnQgPSBuZXcgXHV1ZjY0MjlcT3BlbkFQSVxTcGVjXERvY3VtZW50KCk7CiRwYXRoID0gbmV3IFx1dWY2NDI5XE9wZW5BUElcUGF0aCgpOwokZG9jdW1lbnQtPnBhdGhzLT5zZXQoJy9yZXNvdXJjZScsICRwYXRoKTsKYGBgCgojIyBSZW5kZXJpbmcKCklkZWFsbHksIGl0IHNob3VsZCBiZSByZW5kZXIgd2l0aCBhIFlBTUwgc2VyaWFsaXplciAoc3VjaCBhcyBbc3ltZm9ueSdzXShodHRwczovL3N5bWZvbnkuY29tL2RvYy9jdXJyZW50L2NvbXBvbmVudHMveWFtbC5odG1sKSkuClNpbmNlIFlBTUwgaXMgYSBzdXBlcnNldCBvZiBKU09OLCBvbmUgY2FuIGFsc28gZG86CmBgYHBocAokZG9jdW1lbnQgPSBuZXcgRG9jdW1lbnQoKTsKJGpzb24gPSBqc29uX2VuY29kZSgkZG9jdW1lbnQsIEpTT05fVEhST1dfT05fRVJST1IgfCBKU09OX1BSRVRUWV9QUklOVCk7CgovLyBzYXZlIHRvIGZpbGUKZmlsZV9wdXRfY29udGVudHMoJ29wZW5hcGkueWFtbCcsICRqc29uKTsKCi8vIG9yIHNlcnZlIGl0IG91dApoZWFkZXIoJ0NvbnRlbnQtVHlwZTogYXBwbGljYXRpb24veC15YW1sJyk7CmVjaG8gJGpzb247CmBgYAoKIyMgV2h5PwpUaGlzIGxpYnJhcnkgaXNuJ3QgYmV0dGVyIHRoYW4gdGhlIG11bHRpdHVkZSBvZiBnZW5lcmF0b3JzIGFuZCB3aGF0bm90IG91dCB0aGVyZS4KVGhlIGlkZWEgaXMgdGhhdCBleGlzdGluZyBhbmQgbmV3IFBIUC1iYXNlZCB0b29scyB0aGF0IGhhbmRsZSBPcGVuQVBJIGNvdWxkL3Nob3VsZCB1c2UgdGhpcyBpbXBsZW1lbnRhdGlvbiBpbnN0ZWFkIG9mIGhhdmluZyB0aGVpciBvd24gdmVyc2lvbiBvZiBPcGVuQVBJIHNwZWMuCg== readmeEtag: '"6c4738d5d4f114a6849a594aaf57cbd135c3d173"' readmeLastModified: Wed, 25 Mar 2020 19:21:33 GMT repositoryId: 250069225 description: OpenAPI Spec - PHP Implementation created: '2020-03-25T19:15:07Z' updated: '2020-03-25T19:21:55Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: uuf6429 logo: https://avatars.githubusercontent.com/u/230049?v=4 license: MIT repoEtag: '"a9dc02017f8b5596282d855a48fc8dcdc5f3953bd7adb3198108dc70b3ffafb5"' repoLastModified: Wed, 25 Mar 2020 19:21:55 GMT foundInMaster: true category: Parsers id: 53c737c5d2078ddb652cc0002ca0d479 - name: Traefik Hub source: Tooling repository issues source_description: >- Traefik Hub is a Kubernetes-native API Management solution for publishing, securing, and managing APIs. Configuration is driven by Kubernetes CRDs, labels, and selectors for effective GitOps.- Repository: https://github.com/traefik/traefik-hub-helm-chart link: https://doc.traefik.io/traefik-hub v3_1: false v3: true v2: true sourceIssueMetadata: issueNumber: 82 author: immanuelfodor createdAt: '2023-08-29T14:38:31Z' updatedAt: '2023-09-01T11:59:16Z' url: https://github.com/OAI/Tooling/issues/82 status: closed id: c2281522c0f2ab5084ba47e41c47d3bb foundInMaster: true - source: https://openapi.tools/ name: Apimatic Transformer category: Converters language: SaaS source_description: >- Transform API Descriptions to and from RAML, API Blueprint, OAI v2/v3, WSDL, etc. link: https://apimatic.io/transformer v2: true v3: true v3_1: true foundInMaster: true id: 4386067b4dd5088f3ab21846cc5d73a9 - source: https://openapi.tools/ name: APIMatic Developer Experience Portal category: Documentation link: https://apimatic.io/developer-experience-portal language: SaaS source_description: >- Customizable developer portals packed with language specific documentation, client libraries, code samples, an API console and much more. v2: true v3: true v3_1: true foundInMaster: true id: d79a72cfc760351d8e6bb30f5f61a750 - source: https://openapi.tools/ name: APIMatic CodeGen category: SDK language: SaaS link: https://apimatic.io/code-generation-as-a-service source_description: >- Bring in your API description (OAI v2/v3, RAML, API Blueprint, WSDL, etc.) to generate fully functional SDKs in over 10 languages. v2: true v3: true v3_1: true foundInMaster: true id: dcbfbcc9cc800167a32a0c0a400f8e89 - source: openapi3 tags repository: https://github.com/aliakkas006/fitness-training-apps v3: true id: 0a2d905098aa9a2f8eb61189712ed9f8 repositoryMetadata: base64Readme: >- IyDwn4+L77iP4oCN4pmC77iPIEZpdG5lc3MgVHJhaW5pbmcgQXBwbGljYXRpb24KClshW0NvZGUgU3R5bGU6IFByZXR0aWVyXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2NvZGVfc3R5bGUtcHJldHRpZXItZmY2OWI0LnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9wcmV0dGllci9wcmV0dGllcikKCiMjIPCfk4wgVGFibGUgb2YgQ29udGVudHMKLSBb8J+TliBPdmVydmlld10oIy1vdmVydmlldykKLSBb4pyoIEZlYXR1cmVzXSgjLWZlYXR1cmVzKQotIFvwn5ugIFRlY2hub2xvZ2llcyBVc2VkXSgjLXRlY2hub2xvZ2llcy11c2VkKQotIFvwn5OCIFByb2plY3QgU3RydWN0dXJlXSgjLXByb2plY3Qtc3RydWN0dXJlKQotIFvwn5qAIFNldHVwICYgSW5zdGFsbGF0aW9uXSgjLXNldHVwLS1pbnN0YWxsYXRpb24pCi0gW/Cfp6ogUnVubmluZyBUZXN0c10oIy1ydW5uaW5nLXRlc3RzKQotIFvwn5OaIFJlc291cmNlc10oIy1yZXNvdXJjZXMpCi0gW/CfpJ0gQ29udHJpYnV0aW5nXSgjLWNvbnRyaWJ1dGluZykKLSBb8J+TnCBMaWNlbnNlXSgjLWxpY2Vuc2UpCgotLS0KCiMjIPCfk5YgT3ZlcnZpZXcKVGhlICoqRml0bmVzcyBUcmFpbmluZyBBcHBsaWNhdGlvbioqIGlzIGEgcGxhdGZvcm0gdGhhdCBwcm92aWRlcyBwZXJzb25hbGl6ZWQgd29ya291dCBwbGFucywgdHJhY2tzIHVzZXIgcHJvZ3Jlc3MsIGFuZCBvZmZlcnMgdHJhaW5pbmcgdGlwcyB0byBoZWxwIGluZGl2aWR1YWxzIGFjaGlldmUgdGhlaXIgZml0bmVzcyBnb2Fscy4KCiMjIyDwn46vIEtleSBPYmplY3RpdmVzOgotIOKchSBFbmFibGUgdXNlcnMgdG8gY3JlYXRlIGN1c3RvbWl6ZWQgd29ya291dCBwbGFucy4KLSDinIUgUHJvdmlkZSB0cmFja2luZyBmZWF0dXJlcyBmb3IgcHJvZ3Jlc3MgbW9uaXRvcmluZy4KLSDinIUgQWxsb3cgdHJhaW5lcnMgdG8gYWRkIHRyYWluaW5nIHRpcHMgZm9yIGVhY2ggd29ya291dC4KLSDinIUgUmVxdWlyZSB1c2VyIGF1dGhlbnRpY2F0aW9uIGZvciBwZXJzb25hbGl6ZWQgZXhwZXJpZW5jZXMuCi0g4pyFIEVuYWJsZSBwcm9maWxlIGNyZWF0aW9uIHRvIHNldCBpbml0aWFsIGZpdG5lc3MgbGV2ZWxzIGFuZCBnb2Fscy4KCi0tLQoKIyMg4pyoIEZlYXR1cmVzCvCfmrQgUGVyc29uYWxpemVkIHdvcmtvdXQgcGxhbnMgIArwn5OKIFVzZXIgcHJvZ3Jlc3MgdHJhY2tpbmcgIArwn5KhIFRyYWluZXIgdGlwcyBmb3IgZXhlcmNpc2VzICAK8J+UkiBTZWN1cmUgYXV0aGVudGljYXRpb24gYW5kIGF1dGhvcml6YXRpb24gIArwn5GkIFByb2ZpbGUgbWFuYWdlbWVudCB3aXRoIGZpdG5lc3MgZ29hbCBzZXR0aW5nICAK8J+MkCBSRVNUZnVsIEFQSSBhcmNoaXRlY3R1cmUgIAoKLS0tCgojIyDwn5ugIFRlY2hub2xvZ2llcyBVc2VkClRoZSBhcHBsaWNhdGlvbiBpcyBidWlsdCB1c2luZyBtb2Rlcm4gdGVjaG5vbG9naWVzIGFuZCBiZXN0IHByYWN0aWNlczoKCi0g4pqhICoqW05vZGUuanNdKGh0dHBzOi8vbm9kZWpzLm9yZy8pOioqIEphdmFTY3JpcHQgcnVudGltZSBmb3IgYmFja2VuZCBkZXZlbG9wbWVudC4KLSDwn5qAICoqW0V4cHJlc3MuanNdKGh0dHBzOi8vZXhwcmVzc2pzLmNvbS8pOioqIEZhc3QgYW5kIG1pbmltYWxpc3Qgd2ViIGZyYW1ld29yayBmb3IgTm9kZS5qcy4KLSDwn4+XICoqW1R5cGVTY3JpcHRdKGh0dHBzOi8vd3d3LnR5cGVzY3JpcHRsYW5nLm9yZy8pOioqIEVuaGFuY2VzIEphdmFTY3JpcHQgd2l0aCBzdGF0aWMgdHlwZXMuCi0g8J+XhCAqKltNb25nb0RCXShodHRwczovL3d3dy5tb25nb2RiLmNvbS8pOioqIE5vU1FMIGRhdGFiYXNlIGZvciBmbGV4aWJsZSBkYXRhIHN0b3JhZ2UuCi0g8J+UlyAqKltNb25nb29zZV0oaHR0cHM6Ly9tb25nb29zZWpzLmNvbS8pOioqIE9ETSBsaWJyYXJ5IGZvciBNb25nb0RCIGFuZCBOb2RlLmpzLgotIPCfkLMgKipbRG9ja2VyXShodHRwczovL3d3dy5kb2NrZXIuY29tLyk6KiogQ29udGFpbmVyaXphdGlvbiBmb3Igc2VhbWxlc3MgZGV2ZWxvcG1lbnQgYW5kIGRlcGxveW1lbnQuCi0g8J+nqiAqKltKZXN0XShodHRwczovL2plc3Rqcy5pby8pOioqIEphdmFTY3JpcHQgdGVzdGluZyBmcmFtZXdvcmsgZW5zdXJpbmcgY29kZSBxdWFsaXR5LgoKLS0tCgojIyDwn5OCIFByb2plY3QgU3RydWN0dXJlCmBgYApiYWNrZW5kLwrilJzilIDilIAgcGFja2FnZS5qc29uCuKUnOKUgOKUgCBkb2NzLyAgICAgICAgICAgICAgICAgIyDwn5OEIERvY3VtZW50YXRpb24gZmlsZXMK4pSc4pSA4pSAIHRlc3RzLyAgICAgICAgICAgICAgICAjIPCfp6ogVW5pdCAmIGludGVncmF0aW9uIHRlc3RzCuKUlOKUgOKUgCBzcmMvCiAgICDilJzilIDilIAgaW5kZXgudHMgICAgICAgICAgIyDwn5qAIEFwcGxpY2F0aW9uIGVudHJ5IHBvaW50CiAgICDilJzilIDilIAgYXBwLnRzICAgICAgICAgICAgIyDimpnvuI8gRXhwcmVzcyBhcHAgY29uZmlndXJhdGlvbgogICAg4pSc4pSA4pSAIG1pZGRsZXdhcmUvICAgICAgICMg8J+UkCBNaWRkbGV3YXJlIGZ1bmN0aW9ucwogICAg4pSCICAg4pSc4pSA4pSAIGF1dGhlbnRpY2F0ZS50cwogICAg4pSCICAg4pSc4pSA4pSAIGF1dGhvcml6ZS50cwogICAg4pSCICAg4pSc4pSA4pSAIG93bmVyc2hpcC50cwogICAg4pSCICAg4pSc4pSA4pSAIGluZGV4LnRzCiAgICDilJzilIDilIAgcm91dGVzLyAgICAgICAgICAgIyDwn4yNIEFQSSByb3V0ZSBoYW5kbGVycwogICAg4pSCICAg4pSc4pSA4pSAIGluZGV4LnRzCiAgICDilIIgICDilJzilIDilIAgcHVibGljLnRzCiAgICDilIIgICDilJzilIDilIAgcHJpdmF0ZS50cwogICAg4pSCICAg4pSc4pSA4pSAIGFkbWluLnRzCiAgICDilJzilIDilIAgYXBpLwogICAg4pSCICAg4pSU4pSA4pSAIHYxLwogICAg4pSCICAgICAgIOKUnOKUgOKUgCBhdXRoLwogICAg4pSCICAgICAgIOKUnOKUgOKUgCB3b3Jrb3V0cy8KICAgIOKUgiAgICAgICDilJzilIDilIAgcHJvZ3Jlc3MvCiAgICDilIIgICAgICAg4pSc4pSA4pSAIHByb2ZpbGUvCiAgICDilIIgICAgICAg4pSc4pSA4pSAIHVzZXIvCiAgICDilIIgICAgICAg4pSc4pSA4pSAIHRva2VuLwogICAg4pSc4pSA4pSAIGxpYi8gICAgICAgICAgICAgICMg8J+UpyBDb3JlIGJ1c2luZXNzIGxvZ2ljCiAgICDilJzilIDilIAgbW9kZWxzLyAgICAgICAgICAgIyDwn5eCIE1vbmdvb3NlIG1vZGVscwogICAg4pSc4pSA4pSAIHV0aWxzLyAgICAgICAgICAgICMg8J+UqCBVdGlsaXR5IGZ1bmN0aW9ucwpgYGAKCi0tLQoKIyMg8J+agCBTZXR1cCAmIEluc3RhbGxhdGlvbgojIyMgMe+4j+KDoyBDbG9uZSB0aGUgcmVwb3NpdG9yeQpgYGBiYXNoCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20veW91cnVzZXJuYW1lL0ZpdG5lc3MtVHJhaW5pbmctQXBwLmdpdApjZCBGaXRuZXNzLVRyYWluaW5nLUFwcApgYGAKCiMjIyAy77iP4oOjIEluc3RhbGwgZGVwZW5kZW5jaWVzCmBgYGJhc2gKeWFybiBpbnN0YWxsCmBgYAoKIyMjIDPvuI/ig6MgQ29uZmlndXJlIGVudmlyb25tZW50IHZhcmlhYmxlcwpDb3B5IGAuZW52LmV4YW1wbGVgIHRvIGAuZW52YCBhbmQgdXBkYXRlIHRoZSByZXF1aXJlZCB2YWx1ZXMuCgojIyMgNO+4j+KDoyBSdW4gdGhlIGFwcGxpY2F0aW9uIHVzaW5nIERvY2tlcgpgYGBiYXNoCmRvY2tlci1jb21wb3NlIHVwCmBgYAoKLS0tCgojIyDwn6eqIFJ1bm5pbmcgVGVzdHMKUnVuIHVuaXQgYW5kIGludGVncmF0aW9uIHRlc3RzIHVzaW5nIEplc3Q6CmBgYGJhc2gKeWFybiB0ZXN0CmBgYAoKLS0tCgojIyDwn5OaIFJlc291cmNlcwotIPCfk4QgKipbUHVibGljIEFQSSBSZXNlYXJjaCBEb2NzXShodHRwczovL2FsaS1ha2thcy5ub3Rpb24uc2l0ZS9SZXNlYXJjaC1vbi10aGUtcHVibGljLUFQSXMtb2YtVHdpdHRlci1HaXRIdWItYW5kLUdvb2dsZS0wNmY3ZTdmNzg4OTY0YzIzOTJlNWIyYmY3NDFiYWJlZT9wdnM9NCkqKgotIPCfk4QgKipbUkVTVGZ1bCBBUEkgRGVzaWduIERvY3VtZW50YXRpb25dKGh0dHBzOi8vYWxpLWFra2FzLm5vdGlvbi5zaXRlL1JFU1RmdWwtQVBJLURlc2lnbi1mb3ItRml0bmVzcy1UcmFpbmluZy1BcHBsaWNhdGlvbi0xMzYwN2QwNjRhYjM0M2RlYjE1OGU5ZDBmN2UxYzIwMj9wdnM9NCkqKgotIPCfjqUgKipbUHJvamVjdCBQcmVzZW50YXRpb25dKGh0dHBzOi8va3VhY2JkLW15LnNoYXJlcG9pbnQuY29tLzpwOi9nL3BlcnNvbmFsLzIwMDkzNF9rdV9hY19iZC9FWVFQX3ZYZjd1TlBwZXgtbkcxUjEzTUJ5ZmcxclpHMlpXYVlNOWdLQzg1dnhBP2U9VjYySmhRKSoqCgotLS0KCiMjIPCfpJ0gQ29udHJpYnV0aW5nCkNvbnRyaWJ1dGlvbnMgYXJlIHdlbGNvbWUhIElmIHlvdSdkIGxpa2UgdG8gY29udHJpYnV0ZSwgcGxlYXNlIGZvcmsgdGhlIHJlcG9zaXRvcnkgYW5kIHN1Ym1pdCBhIHB1bGwgcmVxdWVzdC4KCi0tLQoKIyMg8J+TnCBMaWNlbnNlClRoaXMgcHJvamVjdCBpcyBsaWNlbnNlZCB1bmRlciB0aGUgW01JVCBMaWNlbnNlXShMSUNFTlNFKS4KCg== readmeEtag: '"8d5e8c702fe27ae7a4b4f25ed352ff0ce7fedaad"' readmeLastModified: Sat, 15 Feb 2025 04:56:35 GMT repositoryId: 650022785 description: >- The fitness training REST API application is a collection of public API endpoints that enables users to create, manage and interact with a single builder workout plan and track their progress over time created: '2023-06-06T07:01:34Z' updated: '2025-03-02T05:22:46Z' language: TypeScript archived: false stars: 1 watchers: 1 forks: 0 owner: aliakkas006 logo: https://avatars.githubusercontent.com/u/75372387?v=4 repoEtag: '"b3cdd6d1c064fc795bd603660e414394db0825eadb4f252b32bfa160b8c14ae2"' repoLastModified: Sun, 02 Mar 2025 05:22:46 GMT category: - Testing - Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/unjello/openapi-parser v3: true repositoryMetadata: base64Readme: >- aW1hZ2U6Omh0dHA6Ly91bm1haW50YWluZWQudGVjaC9iYWRnZS5zdmdbbGluaz1odHRwOi8vdW5tYWludGFpbmVkLnRlY2gvXQ0KDQo9IERFUFJFQ0FURUQNCg0KX29wZW5hcGktcGFyc2VyXyBpcyBubyBsb25nZXIgc3VwcG9ydGVkLCBwbGVhc2UgY29uc2lkZXIgdXNpbmcgaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItcGFyc2VyW3N3YWdnZXItcGFyc2VyXS4NCg== readmeEtag: '"5dc94a8d0e902f95a0e80e0d23c492f159b9f7f8"' readmeLastModified: Wed, 01 May 2019 08:49:54 GMT repositoryId: 128420085 description: DEPRECATED. Parser and validator for OpenAPI 3.0 created: '2018-04-06T16:37:59Z' updated: '2019-05-01T08:51:22Z' language: JavaScript archived: false stars: 0 watchers: 2 forks: 0 owner: unjello logo: https://avatars.githubusercontent.com/u/155669?v=4 license: CC0-1.0 repoEtag: '"b8b8b412a001f381e9625b8edc75a687f3077cd2b1b9dc46acfdf0b134fc7ec1"' repoLastModified: Wed, 01 May 2019 08:51:22 GMT foundInMaster: true category: Code Generators id: 4aae1a6e178eff3bcd50762bb078df14 - source: openapi3 tags repository: https://github.com/shavo007/greetings-doc v3: true repositoryMetadata: base64Readme: >- # Greetings OpenAPI Definition

## Working on your OpenAPI Definition

### Install

1. Install [Node JS](https://nodejs.org/).
2. Clone this repo and run `npm install` in the repo root.

### Usage

#### `npm start`
Starts the reference docs preview server.

#### `npm run build`
Bundles the definition to the dist folder.

#### `npm test`
Validates the definition.

## Contribution Guide

Below is a sample contribution guide. The tools
in the repository don't restrict you to any
specific structure. Adjust the contribution guide
to match your own structure. However, if you
don't have a structure in mind, this is a
good place to start.

Update this contribution guide if you
adjust the file/folder organization.

The `.redocly.yaml` controls settings for various
tools including the lint tool and the reference
docs engine.  Open it to find examples and
[read the docs](https://redoc.ly/docs/cli/configuration/)
for more information.


### Schemas

#### Adding Schemas

1. Navigate to the `openapi/components/schemas` folder.
2. Add a file named as you wish to name the schema.
3. Define the schema.
4. Refer to the schema using the `$ref` (see example below).

##### Example Schema
This is a very simple schema example:
```yaml
type: string
description: The resource ID. Defaults to UUID v4
maxLength: 50
example: 4f6cf35x-2c4y-483z-a0a9-158621f77a21
```
This is a more complex schema example:
```yaml
type: object
properties:
  id:
    description: The customer identifier string
    readOnly: true
    allOf:
      - $ref: ./ResourceId.yaml
  websiteId:
    description: The website's ID
    allOf:
      - $ref: ./ResourceId.yaml
  paymentToken:
    type: string
    writeOnly: true
    description: |
      A write-only payment token; if supplied, it will be converted into a
      payment instrument and be set as the `defaultPaymentInstrument`. The
      value of this property will override the `defaultPaymentInstrument`
      in the case that both are supplied. The token may only be used once
      before it is expired.
  defaultPaymentInstrument:
    $ref: ./PaymentInstrument.yaml
  createdTime:
    description: The customer created time
    allOf:
      - $ref: ./ServerTimestamp.yaml
  updatedTime:
    description: The customer updated time
    allOf:
      - $ref: ./ServerTimestamp.yaml
  tags:
    description: A list of customer's tags
    readOnly: true
    type: array
    items:
      $ref: ./Tags/Tag.yaml
  revision:
    description: >
      The number of times the customer data has been modified.

      The revision is useful when analyzing webhook data to determine if the
      change takes precedence over the current representation.
    type: integer
    readOnly: true
  _links:
    type: array
    description: The links related to resource
    readOnly: true
    minItems: 3
    items:
      anyOf:
        - $ref: ./Links/SelfLink.yaml
        - $ref: ./Links/NotesLink.yaml
        - $ref: ./Links/DefaultPaymentInstrumentLink.yaml
        - $ref: ./Links/LeadSourceLink.yaml
        - $ref: ./Links/WebsiteLink.yaml
  _embedded:
    type: array
    description: >-
      Any embedded objects available that are requested by the `expand`
      querystring parameter.
    readOnly: true
    minItems: 1
    items:
      anyOf:
        - $ref: ./Embeds/LeadSourceEmbed.yaml

```

##### Using the `$ref`

Notice in the complex example above the schema definition itself has `$ref` links to other schemas defined.

Here is a small excerpt with an example:

```yaml
defaultPaymentInstrument:
  $ref: ./PaymentInstrument.yaml
```

The value of the `$ref` is the path to the other schema definition.

You may use `$ref`s to compose schema from other existing schema to avoid duplication.

You will use `$ref`s to reference schema from your path definitions.

#### Editing Schemas

1. Navigate to the `openapi/components/schemas` folder.
2. Open the file you wish to edit.
3. Edit.

### Paths

#### Adding a Path

1. Navigate to the `openapi/paths` folder.
2. Add a new YAML file named like your URL endpoint except replacing `/` with `@` and putting path parameters into curly braces like `{example}`.
3. Add the path and a ref to it inside of your `openapi.yaml` file inside of the `openapi` folder.

Example addition to the `openapi.yaml` file:
```yaml
'/customers/{id}':
  $ref: './paths/customers@{id}.yaml'
```

Here is an example of a YAML file named `customers@{id}.yaml` in the `paths` folder:

```yaml
get:
  tags:
    - Customers
  summary: Retrieve a list of customers
  operationId: GetCustomerCollection
  description: |
    You can have a markdown description here.
  parameters:
    - $ref: ../components/parameters/collectionLimit.yaml
    - $ref: ../components/parameters/collectionOffset.yaml
    - $ref: ../components/parameters/collectionFilter.yaml
    - $ref: ../components/parameters/collectionQuery.yaml
    - $ref: ../components/parameters/collectionExpand.yaml
    - $ref: ../components/parameters/collectionFields.yaml
  responses:
    '200':
      description: A list of Customers was retrieved successfully
      headers:
        Rate-Limit-Limit:
          $ref: ../components/headers/Rate-Limit-Limit.yaml
        Rate-Limit-Remaining:
          $ref: ../components/headers/Rate-Limit-Remaining.yaml
        Rate-Limit-Reset:
          $ref: ../components/headers/Rate-Limit-Reset.yaml
        Pagination-Total:
          $ref: ../components/headers/Pagination-Total.yaml
        Pagination-Limit:
          $ref: ../components/headers/Pagination-Limit.yaml
        Pagination-Offset:
          $ref: ../components/headers/Pagination-Offset.yaml
      content:
        application/json:
          schema:
            type: array
            items:
              $ref: ../components/schemas/Customer.yaml
        text/csv:
          schema:
            type: array
            items:
              $ref: ../components/schemas/Customer.yaml
    '401':
      $ref: ../components/responses/AccessForbidden.yaml
  x-code-samples:
    - lang: PHP
      source:
        $ref: ../code_samples/PHP/customers/get.php
post:
  tags:
    - Customers
  summary: Create a customer (without an ID)
  operationId: PostCustomer
  description: Another markdown description here.
  requestBody:
    $ref: ../components/requestBodies/Customer.yaml
  responses:
    '201':
      $ref: ../components/responses/Customer.yaml
    '401':
      $ref: ../components/responses/AccessForbidden.yaml
    '409':
      $ref: ../components/responses/Conflict.yaml
    '422':
      $ref: ../components/responses/InvalidDataError.yaml
  x-code-samples:
    - lang: PHP
      source:
        $ref: ../code_samples/PHP/customers/post.php
```

You'll see extensive usage of `$ref`s in this example to different types of components including schemas.

You'll also notice `$ref`s to code samples.

### Code samples

1. Navigate to the `openapi/code_samples` folder.
2. Navigate to the `<language>` (e.g. PHP) sub-folder.
3. Navigate to the `path` folder, and add ref to the code sample.

You can add languages by adding new folders at the appropriate path level.

More details inside the `code_samples` folder README.
 readmeEtag: '"35125785da8665cd825ff9df7549125133f94bba"' readmeLastModified: Thu, 24 Sep 2020 03:44:02 GMT repositoryId: 289852836 description: >- generated by https://github.com/Redocly/create-openapi-repo to showcase open api spec v3 (swagger) created: '2020-08-24T07:02:05Z' updated: '2020-09-24T03:44:06Z' language: JavaScript archived: false stars: 0 watchers: 0 forks: 0 owner: shavo007 logo: https://avatars.githubusercontent.com/u/5466825?v=4 license: MIT repoEtag: '"99b63ea100a00e95cc5537519b87c9d9f6e05ff0614875531462b2d04da3b241"' repoLastModified: Thu, 24 Sep 2020 03:44:06 GMT foundInMaster: true category: Parsers id: b9070ef2123ac1514346b02aca45a305 - source: openapi3 tags repository: https://github.com/realotz/protoc-gen-openapi v3: true repositoryMetadata: base64Readme: >- CumAgueUqOS4jmtyYXRvcy1nb+eahHByb3RvYnVm55Sf5oiQb3BlbmFwaTPmlofmoaPmianlsZUKcHMg5pqC5LiN5pSv5oyBbWFw57uT5p6E77yMbWFw5a6a5LmJ5Lya6L+U5Zue5LiA5Liq56m655qEb2JqZWN0CmBgYGdvbGFuZwpnbyBnZXQgLXUgZ2l0aHViLmNvbS9yZWFsb3R6L3Byb3RvYy1nZW4tb3BlbmFwaQpgYGAK55Sf5oiQCmBgYGJpZ3F1ZXJ5CnByb3RvYyAtLXByb3RvX3BhdGg9LiAgLS1wcm90b19wYXRoPS4vdGhpcmRfcGFydHkgLS1vcGVuYXBpX291dD1wYXRocz1zb3VyY2VfcmVsYXRpdmU6LiB0ZXN0LnByb3RvCmBgYA== readmeEtag: '"5ff1574ca3fb0d49df444683c5492a48bb0148a4"' readmeLastModified: Wed, 31 Mar 2021 10:01:47 GMT repositoryId: 352942598 description: protoc-gen-openapi created: '2021-03-30T09:23:30Z' updated: '2021-03-31T10:01:55Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: realotz logo: https://avatars.githubusercontent.com/u/78065238?v=4 repoEtag: '"dd23df65d69e90ff1c86ec161c27c1448ab4ebbf33fb3c1c5e548c2bbcdd829e"' repoLastModified: Wed, 31 Mar 2021 10:01:55 GMT foundInMaster: true category: - Data Validators - Parsers id: 25c1e5522b2f773662415fcbca9e279b - source: openapi3 tags repository: https://github.com/mal2-project/fake-shop-detection_detector-api v3: true repositoryMetadata: base64Readme: >- IyBGYWtlLVNob3AgRGV0ZWN0b3IgQVBJCgojIyBBYm91dCAvIFN5bm9wc2lzCgoqIGNlbnRyYWwgc291cmNlIGZvciBzZXJ2aW5nIGJyb3dzZXIgcGx1Z2luCiogaG9tb2dlbmVvdXNseSBpbnRlZ3JhdGVzIG11bHRpcGxlIGxvY2FsIG9yIGFwaSBwcm92aWRlcnMgb2YgdHJ1c3R3b3J0aHkgYW5kIGtub3duIGZyYXVkdWxlbnQgc2l0ZXMKKiBwcm92aWRlcyBtYWwyLW1vZGVsIHByZWRpY3Rpb24gY2FwYWJpbGl0aWVzCiogY2FjaGluZyBhbmQgbG9hZCBiYWxhbmNpbmcKKiBQcm9qZWN0IHN0YXR1czogd29ya2luZy9wcm90b3R5cGUKCiMjIFRhYmxlIG9mIGNvbnRlbnRzCgo+ICogW1JlcXVpcmVtZW50c10oI3JlcXVpcmVtZW50cykKPiAqIFtJbnN0YWxsYXRpb25dKCNpbnN0YWxsYXRpb24pCj4gKiBbVXNhZ2VdKCN1c2FnZSkKPiAqIFtSdW5uaW5nIHdpdGggRG9ja2VyXSgjUnVubmluZy13aXRoLURvY2tlcikKPiAqIFtSRVNULUFQSV0oI3Jlc3QtYXBpKQo+ICogW0Fib3V0IE1BTDJdKCNhYm91dC1tYWwyKQo+ICogW0NvbnRhY3RdKCNjb250YWN0KQo+ICogW0xpY2Vuc2VdKCNsaWNlbnNlKQoKIyMgUmVxdWlyZW1lbnRzCgoqIFVidW50dSAxNi4wNAoqIFB5dGhvbiAzLjgKKiBQb3N0Z3JlU1FMIDEwCiogUHl0aG9uLVBhY2thZ2VzIGFzIGRlZmluZWQgaW4gW3JlcXVpcm1lbnRzLnR4dF0oYmFja2VuZC1hcGktc2VydmVyL3JlcXVpcmVtZW50cy50eHQpCgojIyBJbnN0YWxsYXRpb24KCkNyZWF0ZSBhIFB5dGhvbiB2aXJ0dWFsIGVudmlyb25tZW50IHdpdGggZS5nLiBbdmlydHVhbGVudndyYXBwZXJdKGh0dHBzOi8vdmlydHVhbGVudndyYXBwZXIucmVhZHRoZWRvY3MuaW8vZW4vbGF0ZXN0Lykgb3IgYW5hY29uZGEuClRoZSBQeXRob24gdmVyc2lvbiB1c2VkIGlzIDMuOC4KCmBgYHNoZWxsCiQgbWt2aXJ0dWFsZW52IC1wIC9wYXRoL3RvL3B5dGhvbjMuOCByZXN0LWFwaQpgYGAKCkluc3RhbGwgdGhlIHJlcXVpcmVkIFB5dGhvbiBwYWNrYWdlcwoKYGBgc2hlbGwKcGlwIGluc3RhbGwgLXIgYmFja2VuZC1hcGktc2VydmVyL3JlcXVpcmVtZW50cy50eHQKYGBgCgpQb3N0Z3JlU1FMIDEwIGlzIHVzZWQgYXMgZGF0YWJhc2UuIENyZWF0ZSBhIGRhdGFiYXNlIGFuZCBjaGFuZ2UgdGhlIGBiYWNrZW5kLWFwaS1zZXJ2ZXIvc3dhZ2dlcl9zZXJ2ZXIvbWFscy9kYi9oYW5kbGVyL2RiX2hhbmRsZXIucHlgIGFjY29yZGluZ2x5LgoKVGhlIGRhdGFiYXNlIGluaXRpYWxpemlhdGlvbiBpcyBkb25lIGJ5IGNhbGxpbmcgaW5pdC1tYWwyLWRiLnNoCmBgYHNoZWxsCiMhL2Jpbi9iYXNoCnNldCAtZQpwc3FsIC12IE9OX0VSUk9SX1NUT1A9MSAtLXVzZXJuYW1lICIkUE9TVEdSRVNfVVNFUiIgLS1kYm5hbWUgIiRQT1NUR1JFU19EQiIgPDwtRU9TUUwKICAgIENSRUFURSBST0xFIG1hbDJ1c2VyIFdJVEggTE9HSU4gUEFTU1dPUkQgJ2NoYW5nZV9wYXNzJyBTVVBFUlVTRVIgSU5IRVJJVCBDUkVBVEVEQiBDUkVBVEVST0xFOwogICAgQ1JFQVRFIERBVEFCQVNFIG1hbDJyZXN0ZGIgT1dORVIgbWFsMnVzZXI7CkVPU1FMCmBgYAoKIyMgVXNhZ2UKClRvIHJ1biB0aGUgc2VydmVyLCBwbGVhc2UgZXhlY3V0ZSB0aGUgZm9sbG93aW5nIGZyb20gdGhlIGJhY2tlbmQtYXBpLXNlcnZlciBkaXJlY3Rvcnk6CgpgYGAKcHl0aG9uMyAtbSBzd2FnZ2VyX3NlcnZlcgpgYGAKCmFuZCBwb2ludCB5b3VyIGJyb3dzZXIgdG86CgpgYGAKaHR0cDovL2xvY2FsaG9zdDo4MDgwL21hbHp3ZWkvZWNvbW1lcmNlLzEuMS91aS8KYGBgCgpZb3VyIFN3YWdnZXIgZGVmaW5pdGlvbiBsaXZlcyBoZXJlOgoKYGBgCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9tYWx6d2VpL2Vjb21tZXJjZS8xLjEvc3dhZ2dlci5qc29uCmBgYAoKQXR0ZW50aW9uLCB0aGlzIGlzIGZvciBkZXZlbG9wbWVudCBvbmx5LiBJbiBhIHByb2R1Y3Rpb24gZW52aXJvbm1lbnQsIGZvciBleGFtcGxlLCBbdXdzZ2ldKGh0dHBzOi8vdXdzZ2ktZG9jcy5yZWFkdGhlZG9jcy5pby9lbi9sYXRlc3QvV1NHSXF1aWNrc3RhcnQuaHRtbCkgY2FuIGJlIHVzZWQgd2l0aCBbYXBhY2hlMl0oaHR0cDovL2h0dHBkLmFwYWNoZS5vcmcvKS4KCiMjIFJ1bm5pbmcgd2l0aCBEb2NrZXIKCkZvciBhIGZ1bGwgbG9jYWwgZGVwbG95bWVudCB3aXRoIGFuIGV4dGVybmFsIGRiIGxhdW5jaCB0aGUgZG9ja2VyIGJ1aWxkOgpgYGBzaGVsbAojIHN0YXJ0aW5nIHRoZSBjb250YWluZXJzCmRvY2tlci1jb21wb3NlIC1mIGRvY2tlci1jb21wb3NlLmxvY2FsX2Rldi55bWwgdXAKYGBgCgpGb3IgYSBmdWxsIHNlcnZlciBkZXBsb3ltZW50IGxhdW5jaCB0aGUgZG9ja2VyIGJ1aWxkOgpgYGBzaGVsbAojIGJ1aWxkaW5nIHRoZSBpbWFnZQpkb2NrZXItY29tcG9zZSBidWlsZCAtLWJ1aWxkLWFyZyBFTkRQT0lOVF9CQVNFPXlvdXIuc2VydmVyLmxvY2F0aW9uCgojIHN0YXJ0aW5nIHRoZSBjb250YWluZXJzCmRvY2tlci1jb21wb3NlIHVwCmBgYAoKQWxsIGRldGFpbHMgb24gdGhlIGRvY2tlciBzdGFuZC1hbG9uIG9yIGRvY2tlci1jb21wb3NlIGJ1aWxkIGFyZSBwcm92aWRlZCBpbiB0aGUgZmlsZXMKYGRvY2tlci1jb21wb3NlLnltbGAKYGRvY2tlci1jb21wb3NlLmxvY2FsX2Rldi55bWxgCmFuZApgYmFja2VuZC1hcGktc2VydmVyL2RvY2tlci9Eb2NrZXJmaWxlYAoKVGhlIGNvZGUgYXMgaXMgY29udGFpbnMgc2V0dGluZ3MgZm9yIGxvY2FsIGRldiBkZXBsb3ltZW50LiBUaGUgRG9ja2VyZmlsZSB1c2VzICdzZWQnIHRvIHByb3ZpZGUgdGhlIHByb2R1Y3Rpb24gY29uZmlndXJhdGlvbiBhbmQgZG9ja2VyLWNvbXBvc2UgdXAgdG8gc3RhcnQgdGhlIGludGVncmF0ZWQgc3lzdGVtCmBSVU4gc2VkIC1pICJzfDEyNy4wLjAuMXwkZW52X0VORFBPSU5UX0JBU0V8ZyIgc3dhZ2dlcl9zZXJ2ZXIvc3dhZ2dlci9zd2FnZ2VyLnlhbWxgCgojIyBSRVNULUFQSQoKVGhlIFJFU1QgQVBJIGRvY3VtZW50YXRpb24gaXMgYXZhaWxhYmxlIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MS9tYWx6d2VpL2Vjb21tZXJjZS8xLjEvdWkvLgoKUGxlYXNlIG5vdGUgYWxsIHBvcnQgY29uZmlndXJhdGlvbnMgb2YgdGhlIHNlcnZlciBhcmUgZGVmaW5lZCB3aXRoaW4gdGhlIGRvY2tlci1jb21wb3NlIGJ1aWxkIHN0YWdlLiBUaGVyZWZvcmUgbWFrZSBzdXJlIHdoZW4gYWRqdXN0aW5nIHRoZSBwb3J0cyB3aXRoaW4gZG9ja2VyLWNvbXBvc2UueW1sIHRvIGhhdmUgaWRlbnRpY2FsIHBvcnRzIGZvciBjb250YWluZXIgYW5kIGhvc3Qgd2l0aGluIHRoZSBtYWwyLXJlc3QtYXBpIHNlcnZpY2UgYXMgdGhleSBhcmUgcGFzc2VkIHRvIGNvbm5leGlvbltzd2FnZ2VyLXVpXSBhdCBidWlsZCB0aW1lLgoKIyMgQWJvdXQgTUFMMgoKVGhlIE1BTDIgcHJvamVjdCBhcHBsaWVzIERlZXAgTmV1cmFsIE5ldHdvcmtzIGFuZCBVbnN1cGVydmlzZWQgTWFjaGluZSBMZWFybmluZyB0byBhZHZhbmNlIGN5YmVyY3JpbWUgcHJldmVudGlvbiBieSBhKSBhdXRvbWF0aW5nIHRoZSBkaXNjb3Zlcnkgb2YgZnJhdWR1bGVudCBlQ29tbWVyY2UgYW5kIGIpIGRldGVjdGluZyBQb3RlbnRpYWxseSBIYXJtZnVsIEFwcHMgKFBIQXMpIGluIEFuZHJvaWQuClRoZSBnb2FsIG9mIHRoZSBNQUwyIHByb2plY3QgaXMgdG8gcHJvdmlkZSAoaSkgYW4gT3BlbiBTb3VyY2UgZnJhbWV3b3JrIGFuZCBleHBlcnQgdG9vbHMgd2l0aCBpbnRlZ3JhdGVkIGZ1bmN0aW9uYWxpdHkgYWxvbmcgdGhlIHJlcXVpcmVkIHBpcGVsaW5lIOKAkyBmcm9tIG1hbGljaW91cyBkYXRhIGFyY2hpdmluZywgZmVhdHVyZSBzZWxlY3Rpb24gYW5kIGV4dHJhY3Rpb24sIHRyYWluaW5nIG9mIE1hY2hpbmUgTGVhcm5pbmcgY2xhc3NpZmljYXRpb24gYW5kIGRldGVjdGlvbiBtb2RlbHMgdG93YXJkcyBleHBsYWluYWJpbGl0eSBpbiB0aGUgYW5hbHlzaXMgb2YgcmVzdWx0cyAoaWkpIHRvIGV4ZWN1dGUgaXRzIGNvbXBvbmVudHMgYXQgc2NhbGUgYW5kIChpaWkpIHRvIHB1Ymxpc2ggYW4gYW5ub3RhdGVkIEdyb3VuZC1UcnV0aCBkYXRhc2V0IGluIGJvdGggYXBwbGljYXRpb24gZG9tYWlucy4gVG8gcmFpc2UgYXdhcmVuZXNzIGZvciBjeWJlcmNyaW1lIHByZXZlbnRpb24gaW4gdGhlIGdlbmVyYWwgcHVibGljLCB0d28gZGVtb25zdHJhdG9ycywgYSBGYWtlLVNob3AgRGV0ZWN0aW9uIEJyb3dzZXIgUGx1Z2luIGFzIHdlbGwgYXMgYSBBbmRyb2lkIE1hbHdhcmUgRGV0ZWN0aW9uIEFuZHJvaWQgYXBwIGFyZSByZWxlYXNlZCB0aGF0IGFsbG93IGxpdmUtaW5zcGVjdGlvbiBhbmQgQUkgYmFzZWQgcHJlZGljdGlvbnMgb24gdGhlIHRydXN0d29ydGhpbmVzcyBvZiBlQ29tbWVyY2Ugc2l0ZXMgYW5kIEFuZHJvaWQgYXBwcy4KClRoZSB3b3JrIGlzIGJhc2VkIG9uIHJlc3VsdHMgY2FycmllZCBvdXQgaW4gdGhlIHJlc2VhcmNoIHByb2plY3QgW01BTDIgcHJvamVjdF0oaHR0cHM6Ly9wcm9qZWt0ZS5mZmcuYXQvcHJvamVrdC8zMDQ0OTc1KSwgd2hpY2ggd2FzIHBhcnRpYWxseSBmdW5kZWQgYnkgdGhlIEF1c3RyaWFuIEZlZGVyYWwgTWluaXN0cnkgZm9yIENsaW1hdGUgQWN0aW9uLCBFbnZpcm9ubWVudCwgRW5lcmd5LCBNb2JpbGl0eSwgSW5ub3ZhdGlvbiBhbmQgVGVjaG5vbG9neSAoQk1LKSB0aHJvdWdoIHRoZSBJQ1Qgb2YgdGhlIGZ1dHVyZSByZXNlYXJjaCBwcm9ncmFtICg2dGggY2FsbCkgbWFuYWdlZCBieSB0aGUgQXVzdHJpYW4gZmVkZXJhbCBmdW5kaW5nIGFnZW5jeSAoRkZHKS4KKiBBdXN0cmlhbiBJbnN0aXR1dGUgb2YgVGVjaG5vbG9neSBHbWJILCBDZW50ZXIgZm9yIERpZ2l0YWwgU2FmZXR5IGFuZCBTZWN1cml0eSBbQUlUXShodHRwczovL3d3dy5haXQuYWMuYXQvKQoqIEF1c3RyaWFuIEluc3RpdHV0ZSBmb3IgQXBwbGllZCBUZWxlY29tbXVuaWNhdGlvbnMgW8OWSUFUXShodHRwczovL3d3dy5vaWF0LmF0KQoqIFgtTkVUIFNlcnZpY2VzIEdtYkggW1hORVRdKGh0dHBzOi8veC1uZXQuYXQvZGUvKQoqIEt1cmF0b3JpdW0gc2ljaGVyZXMgw5ZzdGVycmVpY2ggW0tTw5ZdKGh0dHBzOi8va3VyYXRvcml1bS1zaWNoZXJlcy1vZXN0ZXJyZWljaC5hdC8pCiogSUtBUlVTIFNlY3VyaXR5IFNvZnR3YXJlIFtJS0FSVVNdKGh0dHBzOi8vd3d3LmlrYXJ1c3NlY3VyaXR5LmNvbS8pCgpNb3JlIGluZm9ybWF0aW9uIGlzIGF2YWlsYWJsZSBhdCBbd3d3Lm1hbHp3ZWkuYXRdKGh0dHA6Ly93d3cubWFsendlaS5hdCkKCiMjIENvbnRhY3QKRm9yIGRldGFpbHMgb24gYmVoYWxmIG9mIHRoZSBNQUwyIGNvbnNvcnRpdW0gY29udGFjdDogCkFuZHJldyBMaW5kbGV5IChwcm9qZWN0IGxlYWQpClJlc2VhcmNoIEVuZ2luZWVyLCBEYXRhIFNjaWVuY2UgJiBBcnRpZmljaWFsIEludGVsbGlnZW5jZQpDZW50ZXIgZm9yIERpZ2l0YWwgU2FmZXR5IGFuZCBTZWN1cml0eSwgQUlUIEF1c3RyaWFuIEluc3RpdHV0ZSBvZiBUZWNobm9sb2d5IEdtYkgKR2llZmluZ2dhc3NlIDQgfCAxMjEwIFZpZW5uYSB8IEF1c3RyaWEKVCArNDMgNTA1NTAtNDI3MiB8IE0gKzQzIDY2NCA4MTU3ODQ4IHwgRiArNDMgNTA1NTAtNDE1MAphbmRyZXcubGluZGxleUBhaXQuYWMuYXQgfCB3d3cuYWl0LmFjLmF0Cm9yCldvZmxnYW5nIEVpYm5lciwgWC1ORVQgU2VydmljZXMgR21iSCwgd2VAeC1uZXQuYXQKCiMjIExpY2Vuc2UKVGhlIE1BTDIgU29mdHdhcmUgc3RhY2sgaXMgZHVhbC1saWNlbnNlZCB1bmRlciBjb21tZXJjaWFsIGFuZCBvcGVuIHNvdXJjZSBsaWNlbnNlcy4gClRoZSBTb2Z0d2FyZSBpbiB0aGlzIHJlcG9zaXRvcnkgaXMgc3ViamVjdCBvZiB0aGUgdGVybXMgYW5kIGNvbmRpdGlvbnMgZGVmaW5lZCBpbiBmaWxlICdMSUNFTlNFLm1kJwo= readmeEtag: '"68115733d2d46c34131acd001a2f25a70026c7bd"' readmeLastModified: Tue, 23 Mar 2021 20:58:20 GMT repositoryId: 350778457 description: >- MAL2 Fake-Shop Detector API integration of known legit and fraudulent sites, integration of detector-models, prediction, API for serving plugin interaction created: '2021-03-23T16:16:51Z' updated: '2022-02-06T02:00:28Z' language: Python archived: false stars: 0 watchers: 2 forks: 0 owner: mal2-project logo: https://avatars.githubusercontent.com/u/75363498?v=4 license: NOASSERTION repoEtag: '"8076e618867102b5830b50407dec60f57df70d9a65603b6ad5cd8d09350c0f8f"' repoLastModified: Sun, 06 Feb 2022 02:00:28 GMT foundInMaster: true category: Server Implementations id: 182085f26b44efedde138bc7a37bba60 - source: openapi3 tags repository: https://github.com/sharmar0790/spring-redis-docker v3: true repositoryMetadata: base64Readme: >- IyBTcHJpbmcgUmVkaXMKCiMjIFRhYmxlIG9mIENvbnRlbnRzCi0gW1doYXQgaXMgUmVkaXMuXSgjd2hhdC1pcy1yZWRpcykKLSBbVXNlIENhc2VdKCN1c2UtY2FzZSkKLSBbSW5zdGFsbGF0aW9uXSgjaW5zdGFsbGF0aW9uKQogICAgKyBbVHdvIG1ldGhvZHNdKCN0d28tbWV0aG9kcykKICAgICsgW0luc3RhbGxlZCBhbmQgY29uZmlndXJlIG9uIE1BQ10oI2luc3RhbGxlZC1hbmQtY29uZmlndXJlLW9uLW1hYykKLSBbQmFja3VwXSgjYmFja3VwKQotIFtNb25pdG9yXSgjbW9uaXRvcikKLSBbUmVkaXMgU2VudGluZWxdKCNyZWRpcy1zZW50aW5lbCkKLSBbUmVkaXMgQXV0aF0oI3JlZGlzLWF1dGgpCi0gW1N0ZXBzIHRvIHJ1biB0aGUgYXBwbGljYXRpb25dKCNzdGVwcy10by1ydW4tdGhlLWFwcGxpY2F0aW9uKQogICAgKyBbUnVuIGJvdGggYXBwIGFuZCByZWRpcyBsb2NhbGx5XSgjcnVuLWJvdGgtYXBwLWFuZC1yZWRpcy1sb2NhbGx5KQogICAgKyBbUnVuIGJvdGggYXBwIGFuZCByZWRpcyBpbnNpZGUgYSBjb250YWluZXJdKCNydW4tYm90aC1hcHAtYW5kLXJlZGlzLWluc2lkZS1hLWNvbnRhaW5lcikKICAgICsgW1J1biBib3RoIGFwcCBhbmQgcmVkaXMgaW5zaWRlIGEgY29udGFpbmVyIHVzaW5nIGRvY2tlci1jb21wb3NlXSgjcnVuLWJvdGgtYXBwLWFuZC1yZWRpcy1pbnNpZGUtYS1jb250YWluZXItdXNpbmctZG9ja2VyLWNvbXBvc2UpCi0gW09wZW5BUEldKCNPcGVuQVBJKSAgICAKLSBbQVBJIEVuZHBvaW50IHRvIGFjY2VzcyB0aGUgYXBwXSgjYXBpLWVuZHBvaW50LXRvLWFjY2Vzcy10aGUtYXBwKQotIFtUTyBET10oI3RvLWRvKQoKIyMgV2hhdCBpcyBSZWRpcy4KCiAtIFJlZGlzIGlzIGEgcG93ZXJmdWwgYW5kIGV4dHJlbWVseSBmYXN0IGluIG1lbW9yeSBkYi4gICAKIC0gU3RvcmVzIGRhdGEgaW4ga2V5LXZhbHVlLiAgIAogLSBQZXJzaXN0ZW5jZS4gIAogLSBPcHRpb25hbGx5IHlvdSBjYW4gc2F2ZSB0aGUgZGF0YSB0byBkaXNrLiAgIAogLSBNYXN0ZXIgc2xhdmUgcmVwbGljYXRpb24uICAKIC0gSGlnaGx5IEF2YWlsYWJsZS4gICAKIC0gT3BlbiBzb3VyY2UuICAKIC0gSW5kZXhpbmcgKFN0aWxsIGNhbiBjcmVhdGUgb3duIGluZGV4aW5nKS4gIAogLSBObyBxdWVyeSBsYW5ndWFnZS4gIAoKCiMjIFVzZSBDYXNlCiAtIFVzZXIgU2Vzc2lvbiBNYW5hZ2VtZW50CiAtIENhY2hpbmcuCiAtIFB1Yi9TdWIgKFF1ZXVlcyAmIE5vdGlmaWNhdGlvbikKIC0gTGVhZGVyYm9hcmRzIGZvciBnYW1pbmcgYXBwcwogLSBHZW9zcGF0aWFsCgoKIyMgSW5zdGFsbGF0aW9uCiMjIyMgVHdvIG1ldGhvZHMKIC0gRG93bmxvYWQgYW5kIGluc3RhbGxlZC4gKFBsZWFzZSB1c2UgW1JlZGlzIERvd25sb2FkXShodHRwczovL3JlZGlzLmlvL2Rvd25sb2FkKSB0byBzZWUgdGhlIHN0ZXBzIHRvIGNvbmZpZ3VyZSB0aGUgcmVkaXMgaW4gb3RoZXIgT1MpCiAtIFVzaW5nIHBhY2thZ2UgbWFuYWdlciBsaWtlIGJyZXcgKG1hYykuIChJIHdpbGwgYmUgdXNpbmcgdGhpcyB0byBjb25maWd1cmUgcmVkaXMpCgoKIyMjIyBJbnN0YWxsZWQgYW5kIGNvbmZpZ3VyZSBvbiBNQUMKIC0gT3BlbiB0aGUgdGVybWluYWwgYW5kIHR5cGUKICAgYGBgCiAgICQgYnJldyBpbnN0YWxsIHJlZGlzCiAgIGBgYAogLSBPbmNlIGl0IGRvbmUsIHR5cGUgKipyZWRpcy1zZXJ2ZXIqKiB0byBzdGFydCB0aGUgcmVkaXMgc2VydmVyIHdpdGggZGVmYXVsdCBjb25maWd1cmF0aW9uIHByb3ZpZGVkIGJ5IHJlZGlzLgogICBgYGAKICAgJCByZWRpcy1zZXJ2ZXIge3BhdGggdG8gdGhlIHJlZGlzLmNvbmZ9CiAgIGBgYCAgIAogLSBPcGVuIGFub3RoZXIgdGVybWluYWwgYW5kIHR5cGUgKipyZWRpcy1jbGkqKiwgdG8gb3BlbiB0aGUgcmVkaXMgY2xpLgogICBgYGAKICAgJCByZWRpcy1jbGkgLXAgNjM3OQogICBgYGAgIAogICBZb3Ugd2lsbCBzZWUgdGhlIG91dHB1dCBzb21ldGhpbmcgbGlrZSBiZWxvdzoKICAgYGBgCiAgICQgcmVkaXMtY2xpIC1wIDYzNzkKICAgMTI3LjAuMC4xOjYzNzk+CiAgIGBgYAogICBDaGVja2luZyByZWRpcyBzdGF0dXMgZXhlY3V0ZSBiZWxvdwogICBgYGAKICAgJCByZWRpcy1jbGkgcGluZwogICBQT05HCiAgIGBgYAogICBraWxsaW5nIHJlZGlzIHNlcnZlciBleGVjdXRlIGJlbG93CiAgIGBgYAogICAxMjcuMC4wLjE6NjM3OT4gc2h1dGRvd24KICAgYGBgCgogLSBTb21lIHNhbXBsZSBjb21tYW5kcyBvbiByZWRpcy1jbGkKICAgYGBgCiAgICAxMjcuMC4wLjE6NjM3OT4gc2V0IGNvbG9yIHJlZCAgICNzZXR0aW5nIGEgdmFsdWUgaW4gY29sb3IgKGFzIGtleSkgYW5kIHJlZCAoYXMgdmFsdWUpCiAgICBPSwogICAgMTI3LjAuMC4xOjYzNzk+IGtleXMgKiAgICMgdmVyaWZ5aW5nIHRvdGFsIG51bWJlciBvZiBrZXlzCiAgICAxKSAiXHhhY1x4ZWRceDAwXHgwNXRceDAwXHgwNEJvb2siCiAgICAyKSAiY29sb3IiCiAgICAxMjcuMC4wLjE6NjM3OT4KICAgIDEyNy4wLjAuMTo2Mzc5PgogICAgMTI3LjAuMC4xOjYzNzk+IGdldCBjb2xvciAgIyBnZXR0aW5nIHRoZSB2YWx1ZSBhc3NpZ25lZCB0byB0aGUga2V5IC0gY29sb3IKICAgICJyZWQiCiAgICAxMjcuMC4wLjE6NjM3OT4KICAgYGBgICAKCiMjIEJhY2t1cAogICBCZWxvdyBjb21tYW5kIHdpbGwgY3JlYXRlIHRoZSBkdW1wL2JhY2t1cCBhcyBwZXIgdGhlIHBhdGggc3BlY2lmaWVkIGluIHJlZGlzLmNvbmYgZmlsZS4KICAgYGBgCiAgICQgc2F2ZQogICBgYGAKCiMjIE1vbml0b3IKICAgUnVuIHRoZSBiZWxvdyBjb21tYW5kIGZyb20gdGVybWluYWwgYW5kIHlvdSdsbCBiZSBhYmUgdG8gc2VlIGV2ZXJ5dGhpbmcgaGFwcGVuaW5nIG9uIHJlZGlzCiAgIGBgYAogICAgJCByZWRpcy1jbGkgLXAgNjM3OQogICAgMTI3LjAuMC4xOjYzNzk+IE1vbml0b3IKICAgIE9LCiAgIGBgYAoKIyMgUmVkaXMgU2VudGluZWwKICAtIEl0IGlzIGEgc3lzdGVtLCBkZXNpZ25lZCB0byBoZWxwIG1hbmFnaW5nIFJlZGlzIGluc3RhbmNlcy4KICAtIEl0IGlzIHRoZXJlIHRvIHByb3ZpZGUgSEEgYnkgbW9uaXRvcmluZywgbm90aWZ5aW5nLCBhbmQgcHJvdmlkaW5nIGluc3RhbmNlcyBmYWlsb3Zlci4KICAtIEl0IGNoZWNrIHdoZXRoZXIgbWFzdGVyIGFuZCBzbGF2ZSBhcmUgd29ya2luZyBwcm9wZXJseSBvciBub3QuCiAgLSBJZiBNYXN0ZXIgZ29lcyBkb3duLCBpdCdzIHRoZSBzZW50aW5lbCByZXNwb25zaWJpbGl0eSB0byBtYWtlIG9uZSBvZiB0aGUgc2xhdmUgdG8gbWFzdGVyLiAgCgoKIyMgUmVkaXMgQXV0aAogIC0gUmVkaXMgYXV0aCBoZWxwIHRvIHNlY3VyZSBkYXRhYmFzZS4gV2UgY2FuIGRvIHRoYXQgZWl0aGVyIHZpYSByZWRpcy5jb25mIG9yIHZpYSBjbGkuCiAgLSBDbGkKICAgIGBgYAogICAgJCBjb25maWcgc2V0IHJlcXVpcmVwYXNzICR7YXV0aC1rZXl9CiAgICAkIGF1dGggJHthdXRoLWtleX0KCiAgICBgYGAKCgpFeHRyYSBsaW5rczoKIC0gVXNlIFtSZWRpc0xhYnNdKGh0dHBzOi8vcmVkaXNsYWJzLmNvbS8pIHRvIHNldHVwIHJlZGlzIG9uIGNsb3VkIGZyZWUuCiAtIFJlZGlzIEdVSSAtIFtSZWRpcyBEZXNrdG9wXShodHRwczovL3JkbS5kZXYvKS4KCgojIyBTdGVwcyB0byBydW4gdGhlIGFwcGxpY2F0aW9uCiMjIyMgUnVuIGJvdGggYXBwIGFuZCByZWRpcyBsb2NhbGx5CiAqIEZvbGxvdyBbdGhlc2Ugc3RlcHNdKCNpbnN0YWxsZWQtYW5kLWNvbmZpZ3VyZS1vbi1tYWMpIHRvIGNvbmZpZ3VyZSBhbmQgcnVuIFJlZGlzIGxvY2FsbHkuCiAqIEV4ZWN1dGUgYmVsb3cgY29tbWFuZCB0byBydW4gdGhlIGJvb3QgYXBwIGZyb20gYXBwcyBkaXJlY3RvcnkKIGBgYAogICQuL2dyYWRsZXcgYm9vdFJ1bgogYGBgCiAqIEFsdGVybmF0aXZlbHksIGZpcnN0IGJ1aWxkIHRoZSBqYXIgYW5kIHRoZW4gcnVuICoqamF2YSAtamFyKiogY29tbWFuZCBhcyBiZWxvdwogYGBgCiAgJC4vZ3JhZGxldyBidWlsZAogYGBgCiBgYGAKICAkIGphdmEgLWphciB7bmFtZSBvZiB0aGUgZ2VuZXJhdGVkIGphcn0KIGBgYAojIyMjIFJ1biBib3RoIGFwcCBhbmQgcmVkaXMgaW5zaWRlIGEgY29udGFpbmVyCiAgKiBFeGVjdXRlIHRoZSBiZWxvdyBjb21tYW5kIGZyb20gcmVkaXMgZGlyZWN0b3J5CiAgYGBgCiAgJCBkb2NrZXIgbmV0d29yayBjcmVhdGUgcm1vZmZfc2VydmljZXMKICAkIGRvY2tlciBidWlsZCAtdCByZWRpcy1zZXJ2ZXIgLgogICQgZG9ja2VyIHJ1biAtZCAtcCA2Mzc5OjYzNzkgLS1uZXR3b3JrPXJtb2ZmX3NlcnZpY2VzIFwKICAgIC0tbmFtZSByZWRpcyBcCiAgICAtaCByZWRpcyByZWRpcy1zZXJ2ZXIKICBgYGAKICAqIEJ1aWxkIGFuIGltYWdlIG9mIHRoZSBhcHAuIEV4ZWN1dGUgYmVsb3cgY29tbWFuZCBmcm9tIGFwcHMgZGlyZWN0b3J5LgogIGBgYAogICQgZG9ja2VyIGJ1aWxkIC10IGJhY2tlbmQtc2VydmljZSAuCiAgYGBgCiAgKiBWZXJpZnkgdGhlIGltYWdlCiAgYGBgCiAgJCBkb2NrZXIgaW1hZ2UgbHMgLWEKICBgYGAKICAqIFJ1biB0aGUgYXBwCiAgYGBgCiAgJCBkb2NrZXIgcnVuIC1kIC0tbmFtZSBhcHAgLWUgIlJFRElTX0hPU1Q9cmVkaXMiIFwKICAgLXAgODA4MDo4MDgwIC0tbmV0d29yaz1ybW9mZl9zZXJ2aWNlcyBcCiAgICBiYWNrZW5kLXNlcnZpY2UKICBgYGAKIyMjIyBSdW4gYm90aCBhcHAgYW5kIHJlZGlzIGluc2lkZSBhIGNvbnRhaW5lciB1c2luZyBkb2NrZXItY29tcG9zZQogIGBgYAogICQgZG9ja2VyLWNvbXBvc2UgLWYgZG9ja2VyLWNvbXBvc2Utc3RhY2sueWFtbCB1cCAtZAogIGBgYAojIyMjIFJ1biBib3RoIGluIEs4cwogICogRXhlY3V0ZSBiZWxvdyBjb21tYW5kIHRvIGRlcGxveSB0aGUgYXBwbGljYXRpb24gaW4gazhzIGFuZCBleHBvc2UgdGhlIHNhbWUgdmlhIHNlcnZpY2UgdXNpbmcgbm9kZXBvcnQuCiAgYGBgCiAgJCBrdWJlY3RsIGFwcGx5IC1mIGs4cy1kZXBsb3ltZW50LnlhbWwKICBgYGAgIAoKIyMgT3BlbkFQSQogICogaHR0cDovL2xvY2FsaG9zdDo4MDgwL3N3YWdnZXItdWkuaHRtbAogICogaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS1kb2NzICAKCiMjIEFQSSBFbmRwb2ludCB0byBhY2Nlc3MgdGhlIGFwcAogIC0gaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaS9oZWFsdGgKICAtIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9hcGkvYm9vay9zYXZlCiAgLSBodHRwOi8vbG9jYWxob3N0OjgwODAvYXBpL2Jvb2svYWxsCiAgLSBodHRwOi8vbG9jYWxob3N0OjgwODAvYXBpL2Jvb2svZGVsZXRlL3tpZH0gICAgIAoKICMjIFRPIERPCgogKiBBZGQgc3RlcHMgdG8gcnVuIGJvdGggY29udGFpbmVyIGluIEs4cy4K readmeEtag: '"3811ac68249c10669b75264a31fb8d8ef08fbb31"' readmeLastModified: Thu, 01 Apr 2021 19:49:13 GMT repositoryId: 342413578 description: Sample implementation of spring with redis. created: '2021-02-26T00:07:40Z' updated: '2021-04-01T19:49:15Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: sharmar0790 logo: https://avatars.githubusercontent.com/u/30017762?v=4 repoEtag: '"0856f3763024c8bcff63d3f8c9eafe680b57ec3d75124c5e23a3e2b2c5b6165a"' repoLastModified: Thu, 01 Apr 2021 19:49:15 GMT foundInMaster: true category: Server Implementations id: cadbd8f02a695a30bcfd68f30eb6933d - source: openapi3 tags repository: https://github.com/openapi-it/tutorials v3: true repositoryMetadata: base64Readme: >- IyBPcGVuYXBpLml0IC0gVFVUT1JJQUxTCgpbIVtOfFNvbGlkXShodHRwczovL29wZW5hcGkuaXQvcHVibGljL2Fzc2V0cy9pbWcvbG9nby5wbmcpXShodHRwczovL29wZW5hcGkuaXQvKQoKVW5hIGxpc3RhIGNvbXBsZXRhIGRpIHZpZGVvIHR1dG9yaWFsIGFjY29tcGFnbmF0aSBkYSBlc2VtcGkgcHJhdGljaSBuZWkgbGluZ3VhZ2dpIHBpw7kgdXRpbGl6emF0aS4KCiMjIEVsZW5jbyBkZWkgdmlkZW8KCnwgVGVtYSB8IExpbmd1YWdnaW8gfCBMaW5rIHwgRXNlbXBpIHwKfCAtLS0tLS0gfCAtLS0tLS0gfCAtLS0tLS0gfCAtLS0tLS0gfAp8IE90dGVuaW1lbnRvIGRpIHVuIHRva2VuIHwgV0VCIHwgW2h0dHBzOi8veW91dHUuYmUvYTNZU05TaW9DREVdIHwgLS0tLS0tIHwKfCBPdHRlbmltZW50byBkaSB1biB0b2tlbiB1c2FuZG8gY1VSTCB8IFBIUCB8IFtodHRwczovL3lvdXR1LmJlLzg3MWt4VVZEN0JZXSB8IFtwaHAvdG9rZW4ucGhwXVtQSFAxXSB8CgoKCgpbUEhQMV06IDxodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaS1pdC90dXRvcmlhbHMvdHJlZS9tYWluL3BocC90b2tlbi5waHA+Cg== readmeEtag: '"e957520f527148c1613825de1d96871b2371e244"' readmeLastModified: Wed, 03 Mar 2021 09:47:50 GMT repositoryId: 343882485 description: >- Un elenco di video tutorial accompagnati da esempi pratici in vari linguaggi di programmazione created: '2021-03-02T19:04:00Z' updated: '2025-06-12T09:47:30Z' language: PHP archived: false stars: 1 watchers: 1 forks: 2 owner: openapi-it logo: https://avatars.githubusercontent.com/u/67429652?v=4 repoEtag: '"2eaa2c8fea1fc178528b1b23201aa47f1ce60f51c93482eb4cf6ef3f74ea13ef"' repoLastModified: Thu, 12 Jun 2025 09:47:30 GMT foundInMaster: true category: Description Validators id: d07bd686576b56438cb889c400667933 - source: openapi3 tags repository: https://github.com/daimaou92/gate v3: true repositoryMetadata: base64Readme: >- IyBHYXRlCgotLS0KCiMjIyBUaGlzIGlzIGFscGhhIGdyYWRlIHNvZnR3YXJlCgotLS0KCk9waW5pb25hdGVkIGxpYiBiYXNlZCBvbiBbaHR0cHJvdXRlcl0oaHR0cHM6Ly9naXRodWIuY29tL2p1bGllbnNjaG1pZHQvaHR0cHJvdXRlcikgdG8gYnVpbGQgUkVTVCBBUElzLgoKIyMgVXNhZ2UKCi0tLQoKVXNpbmcgYEdhdGVgIGlzIGEgbGl0dGxlIGRpZmZlcmVudCB0aGFuIG1vc3Qgb3RoZXIgZnJhbWV3b3Jrcy4gVXNpbmcgaXQgY29ycmVjdGx5CnJlcXVpcmVzIHRoYXQgb25lIGRlY2lkZSB0aGUgcmVxdWVzdCwgcmVzcG9uc2UgYW5kIHF1ZXJ5IHBheWxvYWQgd2hpbGUgZGVzaWduaW5nCmFuIEFQSSB0byB0aGUgbWF4aW11bSBleHRlbnQgZmVhc2libGUuIFdoZW4gZXhwbGljaXRseSBkZWZpbmVkLCBhcyBpbiB0aGUgZXhhbXBsZQpiZWxvdywgYW5kIGxhdGVyIGZldGNoZWQgdXNpbmcgdGhlIGBHYXRlYCBwcm92aWRlZCBhdHRyaWJ1dGUgLSBvbmUgaXMgc2F2aW5nCm9uIHJlLWFsbG9jYXRpb25zLgoKYEdhdGVgIG1haW50YWlucyBhIHBvb2wgb2YgYWxsIHRoZSBzdHJ1Y3R1cmVzIGV4cGxpY2l0bHkgcHJvdmlkZWQgd2hlbiB3cml0aW5nCnRoZSBlbmRwb2ludHMuIFRoZXNlIHBvb2xzIGFyZSB1c2VkIHRvIHJlZmxlY3QgYmFjayBhIHByZS1hbGxvY2F0ZWQgdmFsdWUsCndoZW4gcHJlc2VudCwgaW4gdGhlIGFyZ3VtZW50IG9mIHR5cGUgYFJlcXVlc3REYXRhYCBsYXRlciByZWNlaXZlZCBpbiB0aGUgaGFuZGxlci4KRnJlc2ggYWxsb2NhdGlvbiBvY2N1cnMgaWYgbmVlZGVkLiBNb3JlIG9uIGdvbGFuZyBzeW5jIHBvb2xzIFtoZXJlXShodHRwczovL3BrZy5nby5kZXYvc3luYyNQb29sKQoKIyMgT3BlbiBBUEkKCi0tLQoKVXNpbmcgYEdhdGVgIGFzIGludGVuZGVkIGxldHMgdXMgZ2VuZXJhdGUgYW4gT3BlbkFQSSBiYXNlZCBkb2MgcmlnaHQgZnJvbSB5b3VyIGNvZGUuCldvcmsgaXMgdW5kZXJ3YXkgdG8gYWNoaWV2ZSB0aGlzLiBUaGlzIGZlYXR1cmUgaXMgaW5zcGlyZWQgYnkgW0Ryb3BzaG90XShodHRwczovL2dpdGh1Yi5jb20vb3hpZGVjb21wdXRlci9kcm9wc2hvdCkuCgojIyBFeGFtcGxlCgotLS0KCkhlcmUncyBob3cgYSBQT1NUIHJlcXVlc3QgbG9va3MgaW4gR2F0ZQoKYGBgZ28KcGFja2FnZSBtYWluCgppbXBvcnQgKAoJImVuY29kaW5nL2pzb24iCgkibG9nIgoKCSJnaXRodWIuY29tL2RhaW1hb3U5Mi9nYXRlIgoJImdpdGh1Yi5jb20vZ2V0a2luL2tpbi1vcGVuYXBpL29wZW5hcGkzIgopCgovKiBnYXRlIHByb3ZpZGVzIGEgUGF5bG9hZCBpbnRlcmZhY2UgdGhhdCBtdXN0IGJlIHVzZWQKaW4gQVBJIGRlZmluaXRpb25zIGFuZCB0aGVpciBoYW5kbGVycyB0byBhY2Nlc3Mgc2FpZCBkYXRhLgpTaW5jZSBnZW5lcmljcyBhcmUgc3RpbGwgYSBsaXR0bGUgYml0IGF3YXkgdGhlIGRlZmluaXRpb25zCm5lZWQgdG8gdXNlIGEgaW50aWFsaXplZCBpbnN0YW5jZSBvZiBzYWlkIHBheWxvYWQuCkhlcmUncyBob3cgYWxsIHRoYXQgbG9va3MqLwoKLy8gRGVmaW5lIGEgcGF5bG9hZCB0eXBlCnR5cGUgU3RyaW5nSlNPTiBzdHJpbmcKCmZ1bmMgKHNqIFN0cmluZ0pTT04pIE1hcnNoYWwoKSAoW11ieXRlLCBlcnJvcikgewoJcmV0dXJuIGpzb24uTWFyc2hhbChzaikKfQpmdW5jIChzaiAqU3RyaW5nSlNPTikgVW5tYXJzaGFsKHNyYyBbXWJ5dGUpIGVycm9yIHsKCXZhciB2IHN0cmluZwoJaWYgZXJyIDo9IGpzb24uVW5tYXJzaGFsKHNyYywgJnYpOyBlcnIgIT0gbmlsIHsKCQlyZXR1cm4gZXJyCgl9Cgkqc2ogPSBTdHJpbmdKU09OKHYpCglyZXR1cm4gbmlsCn0KZnVuYyAoU3RyaW5nSlNPTikgQ29udGVudFR5cGUoKSBnYXRlLkNvbnRlbnRUeXBlIHsKCXJldHVybiBnYXRlLkNvbnRlbnRUeXBlSlNPTgp9Ci8vIFRoZSBhYm92ZSB0aHJlZSBmdW5jdGlvbnMgTWFyc2hhbCwgVW5tYXJzaGFsIGFuZAovLyBDb250ZW50VHlwZSBpbXBsZW1lbnQgdGhlIGdhdGUuUGF5bG9hZCBpbnRlcmZhY2UKCi8vIFRoaXMgaXMganVzdCBhIGhlbHBlciBmdW5jdGlvbiB0byBnZXQgYQovLyBTdHJpbmdKU09OIHBvaW50ZXIgLSBvciBhIGdhdGUuUGF5bG9hZApmdW5jIE5ld1N0cmluZ0pTT04ocyBzdHJpbmcpICpTdHJpbmdKU09OIHsKCXQgOj0gU3RyaW5nSlNPTihzKQoJcmV0dXJuICZ0Cn0KCi8vIGJlbG93IGlzIGEgcmVxdWVzdCBoYW5kbGVyIHRoYXQgYWNjZXB0cyBhCi8vIEpTT04gc3RyaW5nIGFuZCByZXNwb25kcyBiYWNrIHdpdGggYSBKU09OCi8vIHN0cmluZyBieSBwcmVwZW5kaW5nIHRoZSBwYXR0ZXJuICJZT0xPIgpmdW5jIHlvbG9IYW5kbGVyKHJjICpnYXRlLlJlcXVlc3RDdHgsIHJkICpnYXRlLlJlcXVlc3REYXRhKSAoZ2F0ZS5QYXlsb2FkLCBlcnJvcikgewoJc2osIG9rIDo9IHJkLkJvZHkuKCpTdHJpbmdKU09OKQoJaWYgIW9rIHsKCQkvLyByZXR1cm5pbmcgYW4gZXJyb3IgYXV0b21hdGljYWxseSByZXNwb25kcyBiYWNrIHdpdGgKCQkvLyB0aGUgY29ycmVzcG9uZGluZyBjb2RlIG9mIHRoZSBlcnJvcgoJCS8vIEhlcmUgZm9yIGV4YW1wbGUsIHRoZSBjbGllbnQgd2l0aCByZWNlaXZlIHRoZSBlcnJvciBjb2RlCgkJLy8gNDAwIGFuZCBhIHRleHQgbWVzc2FnZSAiQmFkIFJlcXVlc3QiCgkJcmV0dXJuIG5pbCwgZ2F0ZS5FcnJCYWRSZXF1ZXN0Cgl9CglzaiA9IE5ld1N0cmluZ0pTT04oIllPTE8iICsgc3RyaW5nKCpzaikpCglyZXR1cm4gc2osIG5pbAp9CgpmdW5jIG1haW4oKSB7CgkvLyBOb3cgbGV0cyBkZWZpbmUgdGhlIGFwaSBzZXJ2ZXIKCWFwcCwgZXJyIDo9IGdhdGUuTmV3KGdhdGUuQXBwT3B0aW9uc3sKCQlJbmZvOiBvcGVuYXBpMy5JbmZvewoJCQlUaXRsZTogICAic2FtcGxlQVBJIiwKCQkJVmVyc2lvbjogIjAuMC4xIiwKCQl9LAoJCUFkZHI6ICI6ODA4MCIsCgl9KQoJaWYgZXJyICE9IG5pbCB7CgkJcGFuaWMoZXJyKQoJfQoKCXNqIDo9IE5ld1N0cmluZ0pTT04oIiIpCgkvLyBUaGUgdXNhZ2Ugb2YgYHNqYCBiZWxvdyBpcyBvbmx5IHNvIHRoYXQgZ2F0ZSBrbm93cyB3aGF0CgkvLyB0eXBlIHRvIG1hcnNoYWwgYW5kIHVubWFyc2hhbCB0aGUgcGF5bG9hZHMgaW50bwoJLy8gdGhlIGluaXRpYWxpemVkIHZhbHVlIHNqIGFib3ZlIHNlcnZlcyBubyBvdGhlciBwdXJwb3NlCgkvLyBhdCB0aGUgbW9tZW50LiBNYXliZSB0aGlzJ2xsIGJlIGJldHRlciB3aXRoIGdlbmVyaWNzLgoJLy8gQnV0IHRoaXMgaXMgd2hlcmUgd2UncmUgYXQgbm93LgoJYXBwLlBvc3QoZ2F0ZS5FbmRwb2ludENvbmZpZ3sKCQlQYXRoOiAgICAiL3lvbG9meSIsCgkJSGFuZGxlcjogeW9sb0hhbmRsZXIsCgkJUGF5bG9hZDogZ2F0ZS5FbmRwb2ludFBheWxvYWR7CgkJCVJlcXVlc3RQYXlsb2FkOiAgc2osCgkJCVJlc3BvbnNlUGF5bG9hZDogc2osCgkJfSwKCX0pCglsb2cuUHJpbnRsbigiTGlzdGVuaW5nIGF0IDo4MDgwIikKCWlmIGVyciA6PSBhcHAuTGlzdGVuKCk7IGVyciAhPSBuaWwgewoJCWxvZy5GYXRhbChlcnIpCgl9Cn0KCmBgYAoKTm93IHNpbXBseSB1c2UgQ1VSTCB0byB2ZXJpZnk6CmBjdXJsIC1kICciIicgLUggJ0NvbnRlbnQtVHlwZTogYXBwbGljYXRpb24vanNvbicgLVggUE9TVCBodHRwOi8vbG9jYWxob3N0OjgwODAveW9sb2Z5YAoKWW91IHNob3VsZCBzZWUgYCJZT0xPImAgYXMgb3V0cHV0LgoKLS0tCgojIyMgVGhpcyBkb2N1bWVudGF0aW9uIGxpa2UgdGhlIHdob2xlIHByb2plY3QgaXMgYWxzbyBhIFdJUC4gSXQgc2hvdWxkIGJlIHVwZGF0ZWQgYXMgc29vbiBhcyBJIGhhdmUgbW9yZSB0aW1lLiBUaGFuayB5b3UgZm9yIHlvdXIgcGF0aWVuY2Ug8J+ZjwoKLS0tCg== readmeEtag: '"cbd3316a6a581e44d492c16e9a5efe236a0a7be9"' readmeLastModified: Mon, 21 Mar 2022 20:17:11 GMT repositoryId: 402033009 description: Build REST APIs in Golang created: '2021-09-01T11:17:35Z' updated: '2022-03-21T19:16:51Z' language: Go archived: false stars: 0 watchers: 0 forks: 0 owner: daimaou92 logo: https://avatars.githubusercontent.com/u/43323412?v=4 license: MPL-2.0 repoEtag: '"ea6badec6f2d813e41e10708a9f9ea1c8ab5ceda5ba6bd96e1d0eee299200833"' repoLastModified: Mon, 21 Mar 2022 19:16:51 GMT foundInMaster: true category: Data Validators id: 678614fcd2ecbfe2c3d9ca05339975b8 - source: openapi3 tags repository: https://github.com/damirscorner/20220722-dotnet-swagger-null-route-param v3: true id: ccd1442d77f2e85ff14d826b3620a7af repositoryMetadata: repositoryId: 512170763 description: >- Sample project for 'ASP.NET Core nullable route params in Swagger' blogpost created: '2022-07-09T11:40:50Z' updated: '2022-07-09T12:37:01Z' language: C# archived: false stars: 0 watchers: 1 forks: 0 owner: DamirsCorner logo: https://avatars.githubusercontent.com/u/64305124?v=4 license: MIT repoEtag: '"a95976cdc81507f07c376566274617639117aa2faeac59ad51561567e7b2f047"' repoLastModified: Sat, 09 Jul 2022 12:37:01 GMT foundInMaster: true - source: openapi3 tags repository: https://github.com/kevinbalicot/tamia v3: true id: faa19aa5ecfb91e6f2ad0aa3106ef63f repositoryMetadata: base64Readme: >- IVtsb2dvXShkb2MvbWVkaWEvbG9nby5wbmcpCgojIFRBTUlBCgpBIFJFU1QgQVBJIFNlcnZlciBkZXNpZ25lZCB3aXRoIE9wZW4gQVBJIHYzCgojIyBJbnN0YWxsYXRpb24KCmBgYAokIG5wbSBpbnN0YWxsIC0tc2F2ZSBAdGFtaWEvdGFtaWEKYGBgCgpXaXRoIGNvbm5lY3Qgb3IgZXhwcmVzcyBqcwoKYGBganMKY29uc3QgdGFtaWFBcGkgPSByZXF1aXJlKCdAdGFtaWEvdGFtaWEnKTsKCi8vIC4uLgpjb25zdCBjb25maWcgPSB7IC4uLiB9OyAvLyBPcGVuIEFQSSBzY2hlbWEsIHNlZSBkb2N1bWVudGF0aW9uIGZvciBtb3JlIGluZm8KY29uc3Qgb3B0aW9ucyA9IHsgLi4uIH07IC8vIFRhbWlhIEFQSSBvcHRpb25zCmNvbnN0IHRhbWlhID0gdGFtaWFBcGkoY29uZmlnLCBvcHRpb25zKTsKCmFwcC51c2UodGFtaWEucmVxdWVzdCk7CmBgYAoKIyMjIERvY3VtZW50YXRpb24KClN1bW1hcnkKCi0gW0NPTkZJR1VSQVRJT05dKC9kb2MvQ09ORklHVVJBVElPTi5tZCkKLSBbUExVR0lOU10oL2RvYy9QTFVHSU5TLm1kKQoKIyMjIFRlc3RzCgpgYGAKJCBucG0gdGVzdApgYGAK readmeEtag: '"494da88590aa4b0fdbc9a64d1bfb570a0725b1a9"' readmeLastModified: Tue, 29 Aug 2023 10:23:46 GMT repositoryId: 527976381 description: A REST API Server designed with Open API V3 created: '2022-08-23T12:16:29Z' updated: '2024-11-18T09:47:16Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: kevinbalicot logo: https://avatars.githubusercontent.com/u/4362142?v=4 repoEtag: '"5828c421d26fc468c907c8085174d31e9d1784f0eaa69891cee70e26449612b3"' repoLastModified: Mon, 18 Nov 2024 09:47:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/usergamestatistics v3: true id: 5df3ed782b4775aa1b8250e11bac82bf repositoryMetadata: base64Readme: >- PiBbIUlNUE9SVEFOVF0gIAo+IFRoaXMgcmVwb3NpdG9yeSBpcyByZWFkLW9ubHkgLyBhcmNoaXZlZCBhbmQgd2lsbCBub3QgcmVjaWV2ZSB1cGRhdGVzLgoKIyBbVHlwZWRvYyBEb2N1bWVudGF0aW9uIFdlYnNpdGVdKGh0dHBzOi8vc3Vkb2t1cnUuZ2l0aHViLmlvL1VzZXJHYW1lU3RhdGlzdGljcy8pPGJyPgoKCiMgRGV2ZWxvcGVyIFNldHVwCgoxLiBHZXQgdGhlIC5lbnYgZmlsZSBmcm9tIHRoZSBNU0IgYnVpbGRpbmcgbmV4dCB0byB0aGUgd2F0ZXIgZm91bnRhaW4uIAoyLiBJbnN0YWxsIERvY2tlciBvbiB5b3VyIG1hY2hpbmUuIFR1dG9yaWFsIGlzIGxpbmtlZCBiZWxvdzo8YnI+CiAgIFshW0RvY2tlciBUdXRvcmlhbF0oaHR0cHM6Ly9pbWcueW91dHViZS5jb20vdmkvMmV6TnFxYVNqcTgvMC5qcGcpXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PTJlek5xcWFTanE4KTxicj4KMi4gT25jZSBkb2NrZXIgaXMgaW5zdGFsbGVkLCB0aGUgTW9uZ28gaW1hZ2UgY2FuIGJlIHJ1biB3aXRoIHRoaXMgY29tbWFuZDo8YnI+Ck5vdGUgdXNlIGBgYHN1ZG9gYGAgb24gTGludXgvTWFjPGJyPgpgYGBjb25zb2xlCm5wbSBydW4gZG9ja2VyCmBgYAozLiBUaGUgYXBwIGNhbiB0aGVuIGJlIHJ1biB3aXRoIHRoZSBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHN0YXJ0CmBgYAo0LiBJbnRlZ3JhdGlvbiB0ZXN0cyBjYW4gYmUgcnVuIHdoZW4gdGhlIGFwcCBpcyBydW5uaW5nIHdpdGggdGhpcyBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHRlc3Q6aW50ZWdyYXRpb24KYGBgCg== readmeEtag: '"0c379c3519f92f8b16c9985566d524f6422cd956"' readmeLastModified: Sun, 12 May 2024 16:01:47 GMT repositoryId: 609680602 description: null created: '2023-03-04T22:48:20Z' updated: '2024-05-12T16:02:55Z' language: TypeScript archived: true stars: 0 watchers: 0 forks: 0 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"890b4b4e5a71250339e926b54c4b8189cc52d6be2c3bf9aaec835d45cb528963"' repoLastModified: Sun, 12 May 2024 16:02:55 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/azhirov/openapi-3.0-types v3: true id: 971d273e5665ce0122911e7890fc24ea repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIFR5cGVzCgpUaGlzIHBhY2thZ2UgcHJvdmlkZXMgVHlwZVNjcmlwdCB0eXBlcyBmb3IgT3BlbkFQSSAzLjAsIGJhc2VkIG9uIHRoZSBvZmZpY2lhbCBbT3BlbkFQSSBTcGVjaWZpY2F0aW9uIDMuMC4zXShodHRwczovL3NwZWMub3BlbmFwaXMub3JnL29hcy92My4wLjMpLiBUaGVzZSB0eXBlcyBhbGxvdyB5b3UgdG8gdmFsaWRhdGUgYW5kIHR5cGUtY2hlY2sgeW91ciBPcGVuQVBJIGRvY3VtZW50cyBpbiBUeXBlU2NyaXB0LCBwcm92aWRpbmcgaW1wcm92ZWQgZG9jdW1lbnRhdGlvbiBhbmQgY29uZmlkZW5jZSBpbiB5b3VyIEFQSSBjb250cmFjdHMuCgojIyBJbnN0YWxsYXRpb24KCmBgYApucG0gaSAtLXNhdmUtZGV2IG9wZW5hcGktMy4wLXR5cGVzCmBgYAoKIyMgVXNhZ2UKClRvIHVzZSB0aGUgT3BlbkFQSSB0eXBlcyBpbiB5b3VyIFR5cGVTY3JpcHQgcHJvamVjdCwgc2ltcGx5IGltcG9ydCB0aGVtIGFzIG5lZWRlZDoKCmBgYHR5cGVzY3JpcHQKCmltcG9ydCB0eXBlIHsgT3BlbkFwaU9iamVjdDMgfSBmcm9tICdvcGVuYXBpLTMuMC10eXBlcyc7Cgpjb25zdCBteUFwaVNwZWM6IE9wZW5BcGlPYmplY3QzID0gewogIC8vIC4uLgp9OwpgYGAKVGhlIE9wZW5BcGlPYmplY3QzIHR5cGUgcmVwcmVzZW50cyB0aGUgcm9vdCBvYmplY3Qgb2YgYW4gT3BlbkFQSSBzcGVjaWZpY2F0aW9uLiBPdGhlciB0eXBlcyBhcmUgcHJvdmlkZWQgZm9yIGVhY2ggY29tcG9uZW50IG9mIHRoZSBzcGVjaWZpY2F0aW9uLCBzdWNoIGFzIFBhdGhJdGVtT2JqZWN0MywgUGFyYW1ldGVyT2JqZWN0MywgYW5kIFJlc3BvbnNlT2JqZWN0My4KCkZvciBtb3JlIGluZm9ybWF0aW9uIG9uIHRoZSBhdmFpbGFibGUgdHlwZXMsIHBsZWFzZSByZWZlciB0byB0aGUgb2ZmaWNpYWwgW09wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLjAuM10oaHR0cHM6Ly9zcGVjLm9wZW5hcGlzLm9yZy9vYXMvdjMuMC4zKS4KCiMjIEdlbmVyYXRlIGAqLmQudHNgCgoxLiBDbG9uZSB0aGlzIHJlcG9zaXRvcnkKMi4gSW5zdGFsbCB0eXBlc2NyaXB0CjMuIFJ1biBgdHNjYAoKIyMgQ29udHJpYnV0aW5nCgpJZiB5b3UgZmluZCBhIGJ1ZyBvciB3b3VsZCBsaWtlIHRvIGNvbnRyaWJ1dGUgdG8gdGhlIHBhY2thZ2UsIHBsZWFzZSBmZWVsIGZyZWUgdG8gb3BlbiBhbiBpc3N1ZSBvciBzdWJtaXQgYSBwdWxsIHJlcXVlc3Qgb24gdGhlIEdpdEh1YiByZXBvc2l0b3J5LgoKIyMgTGljZW5zZQpUaGlzIHBhY2thZ2UgaXMgbGljZW5zZWQgdW5kZXIgdGhlIE1JVCBMaWNlbnNlLiBTZWUgdGhlIExJQ0VOU0UgZmlsZSBmb3IgZGV0YWlscy4K readmeEtag: '"309999e9e508944cffffaec49fd397a272a5e69c"' readmeLastModified: Mon, 17 Apr 2023 01:14:30 GMT repositoryId: 628614386 description: This package provides TypeScript types for OpenAPI 3.0 created: '2023-04-16T13:59:06Z' updated: '2023-04-16T14:17:48Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: azhirov logo: https://avatars.githubusercontent.com/u/29061184?v=4 license: MIT repoEtag: '"b456aabe5992c5159b5aeb82135eae1ddd45da077c2f1d690e2902d4f16f3d32"' repoLastModified: Sun, 16 Apr 2023 14:17:48 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/jmcleodfoss/simple-rest-tester v3: true repositoryMetadata: base64Readme: >- ![MIT License](https://img.shields.io/badge/license-MIT-green) 
[![Codacy Badge](https://api.codacy.com/project/badge/Grade/02c9182fb9204a709cbf472a895b87eb)](https://app.codacy.com/gh/Jmcleodfoss/simple-rest-tester?utm_source=github.com&utm_medium=referral&utm_content=Jmcleodfoss/simple-rest-tester&utm_campaign=Badge_Grade_Settings) 
![Codacy Security Scan](https://github.com/Jmcleodfoss/simple-rest-tester/workflows/Codacy%20Security%20Scan/badge.svg) 
![CodeQL](https://github.com/Jmcleodfoss/simple-rest-tester/workflows/CodeQL/badge.svg)
![SL Scan](https://github.com/Jmcleodfoss/simple-rest-tester/workflows/SL%20Scan/badge.svg) 
![Xanitizer Security Analysis](https://github.com/Jmcleodfoss/simple-rest-tester/workflows/Xanitizer%20Security%20Analysis/badge.svg) 
# simple-rest-tester

A simplistic test harness for REST APIs which uses [node.js](http://nodejs.org), [Mocha](http://mochajs.node), and [Chai.js](www.chaijs.com), and JSON files defining test parameters 
and expected results to run simple REST API tests with the following characteristics:
*   a deterministic return value
*   a response code

The *simple-rest-tester* package includes three applications:
*   simple-rest-tester, which runs tests on all JSON files in the current directory, handling dependencies (for tests which use the results of other tests)
*   srt-generator, which creates test JSON files from an [OpenAPI 3](https://swagger.io/specification/) specification with a small number of vendor extensions described below
*   srt-expand-macros, which allows you to show what a test file looks like after macro replacement for macros defined at the command line and using environment variables

There is also an API for use if your testing needs are more complicated than what can be handled by the *simple-rest-tester* application (for example, you need to run tests in a 
specific order different from that used by the provided application).

## Installation
### For Command-Line Use
To use the three applications from the command line, install using npm with the -g / --global flag:
```bash
npm install -g simple-rest-tester
```
### To Use the simple-rest-tester Engine Directly
```bash
npm install simple-rest-tester
```

## JSON structure
The test information is stored in a JSON file with the following members:
```javascript
{
	"description": /* Description of the test to be displayed by Mocha */
	"expectation": /* Description of the results of a successful test to be displayed by Mocha */
	"status": /* Expected numeric HTTP status code, used by a Mocha expect test" */
	"path-suffix": /* A string to be appended to the path (typically a query string) */ 
	"responseRegexp": /* A regular expression describing the expected response, used by a Mocha expect test (e.g. 200, 204, 400, 401, 404, etc) */
	"saveResponse": /* A flag indicating whether the result of this test should be saved for later use by another test in the test suite (true or false) */
	"testname": /* Unique (within test suite) identifier */
	"scheme": /* the URL scheme to use for this test (http or https) */

	"payload": /* An object giving the contents of the body of a POST or PUT request, if any */
	"payload-override": /* An object that replaces some items of the payload (normally used if payload is set to an example value but a small number of members need to be changed"

	"options": {
		"host": /* The name or IP address of the host holding the REST API server to be tested */
		"port": /* The port the REST API server is listening on */
		"path": /* The path of the REST API endpoint to be tested */
		"method": /* The HTTP method (POST, GET, PUT, DELETE, etc) */
		"headers": /* An object providing any HTTP headers, e.g. "accept": "application/json" or "Content-Type": "application/json" */
		}
	}

	"prerequisites": [] /* An array of testnames for tests that have to be perfomed before this test, for cases where there are dependencies that are not in the form of returned values */
}
```
*   The value of `description` is passed to the Mocha`describe` function.
*   The value of `expectation` is passed to the Mocha`i` function.
*   The value of `status` is compared to the status code of the service's response to the request in a Chai.js`expec` test.
*   The value of `responseRequest` is compared to the string returned by the service to the rest request in a Chai.js`expec` test.
*   The value of `saveResponse` indicates whether the response from the server to this request should be saved for use in a future test. See "macros" below for more information.
*   The value of `testname` is used in macros to refer to returned results in later tests. See "macros" below for more information. It is only necessary if saveResponse is true. By convention, it has the format METHOD_service_response, e.g. POST_myservice_200.
*   The value of `scheme` is used with a`require` command to create an http or https object.
*   The value of `payload` is written into the POST or PUT request.
*   The `options` object is the passed into the [http.request](https://nodejs.org/api/http.html#http_http_request_options_callback) or [https.request](https://nodejs.org/api/https.html#https_https_request_options_callback) function.
*   The `prerequisites` array is added to the list of prerequisites calculated based on macro definitions to force tests to be performed in a specific order.

## Test File Names
By convention, the base file name is the same as the testname member, with the extension "json": METHOD_service_response.json, 
e.g. POST_myservice_200.json, but you may pick whatever convention you like.
If you are using srt-geneator to generate test files from an openapi specification, you may use the filename key to specify a custom name.

## Creating Tests from an OpenAPI Specification: srt-generator
### Running srt-generator
`srt-generator [--help|-h] | [--overwrite|-O] [--quiet|-q] [--server=#|-s #] openapi.yaml-1 [openapi.yaml-2 ...]`
where
*   -h / --help: shows command help
*   -O / --overwrite: overwrites all existing files without asking
*   -q / --quiet: suppresses informational messages about which files are being over-written, etc
*   -s # / --server=#: pull scheme, host, and port from the definition for server index `#` instead of from server index 0

### Standard Elements Used
*   `servers[0]`: used to find the scheme, hostname, port, and path prefix, if any.
*   `paths.path`: used to create the `testname` member, the filename, the test description, and the URL `path` member for the http/https `options` object.
*   `paths.path.method`: used to create the `testname` member, the filename, and the `method` member of the http/https `options` object.
*   `paths.path.method.requestBody.content.application/json.examples`: used for the payload and as a suffix for the filename for POST and PUT requests.
*   `paths.method.responses.response`: used for the expected returned status from the server

### Vendor Extensions
The OpenAPI specification does not include quite enough information to make tests useful. The vendor extension element `x-srt` allows srt-generator to create tests which are more than just stubs that need to be filled in.
The x-srt element allows any element of the JSON test file to be set, overriding the values that *srt-generator* by default pulls out of the spec file. This is a powerful feature, but can also lead to problems which may
be difficult to debug. In general, a test will need to define the `description`, `expectation`, `response-regexp`, `path-suffix` and `path-subst` members here, but you may find the default values are suitable in some cases.

#### Caveat
It goes without saying that you should not put authentication credentials into your OpenAPI spec file for testing. Use the environment macro facility described below if you need to perform tests requiring authentication.

#### paths.path.method.responses.response.x-srt/description
This provides the `description` member of the test object.

#### paths.path.method.responses.response.x-srt/expectation
This extension is used to provide the `expectation` member of the test object.

#### paths.path.method.responses.response.x-srt/filename
This may be used to provide a custom filename for the test.

#### paths.path.method.responses.response.x-srt.response-regexp
This provides a regexp matching the expected response from the server. If it is not present, all responses are considered matches.

#### paths.path.method.responses.response.x-srt.path-suffix
This extension is used to build the `path` member of the test object.

#### paths.path.method.responses.response.x-srt.skip-test
This extension may be used with responses for which no test case is desired, Starting with version 1.0.1, this is unnecessary as no
tests will be generated for responses with no x-srt member, but it remains and is not deprecated for those who choose to distinguish
between responses which have not been considered and those which have been explicitly ignored.

#### paths.path.method.responses.response.x-srt.payload
This extension should be used to provide a payload for tests which do not use the paths.path.method.requestBody.content.application/json/examples element to provide the body of a request.

#### paths.path.method.responses.response.x-srt.payload-override
This extension should be used with the x-srt.payload extension to override some members of the main payload extension. The use case for this is setting the payload to a reference to an example in the component section, and then changing a member to generate a slightly different test (e.g. changing a member to an invalid value to test code paths in the API service dealing with bad input).

#### paths.path.method.responses.response.x-srt.path-subst
This extension. if present, should be a set of key-value pairs where the key represents a part of the path to be replaced, and the value is the substituted value.

#### paths.path.method.responses.response.x-srt.scheme
This may be used to over-ride the scheme use for testing (http or https) which is by default obtained from the first defined server.

#### paths.path.method.responses.response.x-srt.options.host
This may be used to over-ride the host used for testing which is obtained from the first defined server.

#### paths.path.method.responses.response.x-srt.options.port
This may be used to over-ride the port used for testing which is obtained from the first defined server.

#### paths.path.method.responses.response.x-srt.options.headers
This array can be used to add header data to the options object passed to the http/https request function. The values provided here over-ride any default values for `Content-Type` or `accept`.

#### paths.path.method.responses.response.x-srt.prerequisites
This array lists testnames for tests that must be performed before the current test.

## Example OpenAPI Specification
An example OpenAPI specification that includes the vendor extensions described above can be found in the [examples/openapi](https://github.com/Jmcleodfoss/srt-generator/blob/master/examples/openapil) directory:
along with output JSON files srt-generator generates:
*   openapi.yaml
*   DELETE_point_204.json
*   DELETE_point_401.json
*   DELETE_point_404.json
*   GET_ping_204.json
*   GET_point_200.json
*   GET_point_404.json
*   POST_point_200_pointOrigin.json
*   POST_point_400.json
*   PUT_point_204_point90Degrees.json
*   PUT_point_400.json
*   PUT_point_401.json
*   PUT_point_404.json

**Note** that generating a server using *openapi-generator* and running it, and the running mocha against these tests, will *not* result in the expected behavior for the \*40x.json files as there is no
code backing the point service to return error values.

## Running Tests: simple-rest-tester
Once you have created your test specification JSON files, either from an OpenAPI spec or manually, you can run your tests. First, start the services you are testing, then, from the directory containing the test
specification files, run
`simple-rest-tester`
The application will analyze your test specifications, attempt to determine the order in which to conduct the tests, and run Mocha.js on each defined test.

## Macros
Within a given run of a suite of tests, it is possible to save information returned in one test for use in a later test, for example, saving the returned item index from a POST request to use in a DELETE, GET, 
or PUT request. Macros may appear in any string value in the test object, although they are most useful in the `path` and `payload` values.

### Saving Responses
To save the response from a test, the following elements need to be defined in the test object:
*   `saveResponse: true`
*   `testname: /* A unique name for the test, e.g. POST_tag_200 */`

The test must return an object like `{"addedItemIndex":"193"}`.

### Referring to Saved Data
To refer to saved information in a later test, use a macro with the format `${saved-testname}.member`, e.g. `${POST_item_200}.addedItemIndex`. 
All occurrences of this macro will be replaced with the value returned by the named test.

Here is an example of path containing a macro (in the `options` member of the test specification):
```json
"path": "/v1/item/${POST_item_200}.addedItemIndex?queryParameter=1"
```
After macro replacement, this will become (using our example `addedItemIndex` from above):
```json
"path": "/v1/item/193?queryParameter=1"
```
Similar substitutions can take place in the payload object.

### General-Purpose Javascript Macros
Staring with version 1.0.2, you may define generic macros in a response's x-srt section. These come in two types, distinguished by the `definitionPhase` member. If the
`definitionPhase` is `preRequest`, then the macro is executed before the http/https request is made, and it may modify the request. If the `definitionPhase` is `postResponse`,
then it can use components of the response in its definition.

#### preRequest Javascript Macros
The pre-request macro mechanism may be used, for example, to add a counter to a value which the server being tested against expects to be unique. In the example below, the username
must be unique. This is accomplished by reading a counter which is persisted in a file in the test directory (creating it if necessary), incrementing it, and returning it,
but also adding it to the userName value in the payload (which one might want to do if the payload comes from an example in the OpenAPI definition file).
```yaml
paths:
  ...
  /user:
    ...
    POST:
      ...
      responses:
        "200":
          description: "Successfully added user; returns username and password"
          x-srt:
            expectation: "should return the username and password"
            macroDef:
            - name: 'userNameCounter'
              definitionPhase: 'preRequest'
              definition: |
                (reqDescr) => {
                  const fs = require('fs');
                  const fn = 'persistent-counter.txt';
                  let counter = fs.existsSync(fn) ? JSON.parse(fs.readFileSync(fn)) + 1 : 0;
                  reqDescr.payload.userName = reqDescr.payload.userName + "-${userNameCounter}";
                  fs.writeFileSync(fn, JSON.stringify(counter));
                  return counter;
                }
```

#### postResponse Javascript Macros
The post-response macro mechanism may be used, for example, to save an authentication string for use in a later test.
```yaml
paths:
  ...
  /user:
    ...
    post:
      ...
      responses:
        "200":
          description: "Successfully added user; returns username and password"
          x-srt:
            expectation: "should return the username and password"
            macroDef:
            - name: 'authString'
              definitionPhase: 'postResponse'
              definition: |
                ({consumerId, password}) => {
                  return require(process.cwd() + "/node_modules/base-64").encode(consumerId + ":" + password);
                }
    ...

    put:
      ..
      responses:
        "204":
          description: "Successfully updated user"
          x-srt:
            expectation: "should update the user with the new description"
            path-subst: { "{userId}": "${POST_user_200_exampleUser1}.uid" }
            options:
              headers:
                Authorization: "Basic ${POST_user_200}.authString"
```
Unfortunately, support for Javascript macros is marginal in srt-expand-macros, as actually evaluating a macro might have permanent side-effects (like incrementing a counter) which are undesireable.

## Environment Substitutions
You may refer to environment variable values in macros of the form `${env}.ENVIRONMENT_VARIABLE_NAME}`. A very contrived example is `${emv}.PATH` will be expanded to your path.
Note (a) that case is important, and (b), if you have a test which saves results under the prefix `${env}`, it won't be picked up.

## User-Defined Macros
You may also add macros programmatically using the `addMacros` function. This functionality is not supported when running Mocha.js via the *simple-rest-tester* application because of 
complications arising from Mocha.js command line processing. These are described in more detail in the section "Rolling Your Own", below.

## Debugging
Basic logging to console of the path, payload, status code, and response, is available by setting the environment variable `SRT_TEST=1` (both when running the *simple-rest-tester* 
application and when running the simple-rest-tester engine from your own application).

## Rolling Your Own
Create a test suite using a file like [examples/test.js](https://github.com/Jmcleodfoss/simple-rest-tester/blob/master/examples/test.js):
```javascript
"use strict";

const srt = require('simple-rest-tester');

function test(jsonFile)
{
	var reqDescr = require(jsonFile);

	try {
		srt.mocha(reqDescr);
	} catch (ex) {
		console.log(ex);
	}
}

srt.addMacro('user-defined', 'User-defined macro goes here.');
srt.addMacro('timestamp', new Date().getTime());

// Order of tests is important if you are using macros ore relying on the existence of resources created by earlier tests in later tests
test('./GET_ping_204.json')
test('./POST_service_200.json')
test('./POST_service_400.json')
// etc
```
This will result in any occurrences of `${user-defined}` in the test specification being replaced by the string "User-defined macro goes here.", and `${timestamp}` being replaced by the timestamped 
grabbed by node when the program was executed.

Invoke *mocha* in the directory containing the above script to run the tests.

## Viewing Macro Substitutions: srt-expand-macros
For troubleshooting, it is often convenient to see how macros are expanded. This can be done using the *srt-expand-macros* application.

### Running srt-expand-macros
```bash
srt-expand-macros[-h | [-D macro-1=value-1 [-D macro-2=value2 [...]]] test-definition [test-definition [...]]]
```
where
*   -h / --help              Display this usage guide
*   -D / --define string[]   Define macro as macro=substitution (this can appear multiple times)

Macros can be either response-style macros (${a}.b=value} or simple substitutions (${a}=value).

## Releases
### 1.0.8 2023-04-06
Update dependencies to remove security issue.
Drop unused library to remove security issue.

### 1.0.7 2021-04-19
Add a new element to allow the file name (and test name) created by srt-generator to be specified.
Allow DELETE methods to have a body.
Allow an array of tests to be defined for a response.
Allow an array of path-subst replacements.
When SRT_DEBUG is true, show the name of each filename as it is read in and parsed.
Add verbose command line option to srt-generate, and list name of each file as it is written when verbose mode is active.
When the expected status code is not encountered, list the expected status code after the one actually encountered (to assist with debugging).
Fix handling of payload-override for objects, and revert to using JSON.parse instead of using VM2.

### 1.0.6 2021-02-12
Add new element to allow payload to be over-ridden. (This allows a single example to be used for multiple tests by modifying various of its properties at test time.)

### 1.0.5 2021-01-25
It is now possible to define prerequisite tests to force a test to take place after another test even if it does not have a dependency on the other test's return value.

### 1.0.4 2020-12-23
Use full path to construct test filename rather than just the first component, so e.g. GET /user/{id}/name creates a file named user_id_name.json rather than user.json.
Log scheme, host, and port if debugging is enabled.

### 1.0.3 2020-12-02
Add support for general-purpose Javascript-based macros.
Show headers if debugging is enabled.

### 1.0.2 2020-11-20
Fix issue #2, crash if an OpenAPI spec file contains a header.options section.
Resolve several problems found by Codacy and Xanitizer.

### 1.0.1 2020-10-11
Fix typo in homepage URL in package.json.
Add a keyword to package.json

#### srt-generator
Don't create tests for responses without an x-srt element.

### 1.0.0, 2020-10-07
Initial release. Functionality to be improved as demand requires and time permits.
 readmeEtag: '"543ec626004f281d9bcece3671c8f03bc8f34ada"' readmeLastModified: Sun, 09 Apr 2023 12:05:45 GMT repositoryId: 297208956 description: A simplistic mocha-based test harness for REST APIs created: '2020-09-21T02:33:54Z' updated: '2021-11-03T11:11:06Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: Jmcleodfoss logo: https://avatars.githubusercontent.com/u/55663604?v=4 license: MIT repoEtag: '"a9d176198df080c72b979a99823a810a0bf187cc3923457ce80ddd3108aa7998"' repoLastModified: Wed, 03 Nov 2021 11:11:06 GMT foundInMaster: true category: Testing id: a367f13a405f78c1ddddc5f71f16abc5 - source: openapi3 tags repository: https://github.com/locol23/swagger-github-pages v3: true id: 3d5b4219102b8161b4c37c7c4ef11e30 repositoryMetadata: base64Readme: >- IyBzd2FnZ2VyLWdpdGh1Yi1wYWdlcwoKIyMgRGVtbwoKaHR0cHM6Ly9sb2NvbDIzLmdpdGh1Yi5pby9zd2FnZ2VyLWdpdGh1Yi1wYWdlcy8KCiMjIFJlZmVyZW5jZQoKaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItdWkvYmxvYi9tYXN0ZXIvZG9jcy91c2FnZS9pbnN0YWxsYXRpb24ubWQjdW5wa2cKCg== readmeEtag: '"88098e084e37d36deec176307094b37609bea86b"' readmeLastModified: Tue, 09 May 2023 14:29:17 GMT repositoryId: 638518383 description: Visualize open api yaml with swagger-ui and deploy to GitHub pages created: '2023-05-09T14:22:21Z' updated: '2023-05-09T14:42:13Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: locol23 logo: https://avatars.githubusercontent.com/u/6946766?v=4 repoEtag: '"1f6d07ccf62dc35773570d096ef27f69123a2963fe17bdad7278761a1cfcb37e"' repoLastModified: Tue, 09 May 2023 14:42:13 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/trebol-ecommerce/api v3: true repositoryMetadata: base64Readme: >- IyBUcsOpYm9sIGVDb21tZXJjZSBBUEkKClshW0dpdEh1YiBTdXBlci1MaW50ZXJdKGh0dHBzOi8vZ2l0aHViLmNvbS90cmVib2wtZWNvbW1lcmNlL2FwaS9hY3Rpb25zL3dvcmtmbG93cy9saW50ZXIueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbWFya2V0cGxhY2UvYWN0aW9ucy9zdXBlci1saW50ZXIpCgpUaGUgY29udHJhY3QgdGhhdCBUcsOpYm9sIGFzIGEgd2hvbGUgbXVzdCBjb21wbHkgd2l0aC4KCiMjIERvY3VtZW50YXRpb24KClRoYW5rcyB0byBHaXRIdWIgQWN0aW9ucyBhbmQgW1JlZG9jXShodHRwczovL2dpdGh1Yi5jb20vUmVkb2NseS9yZWRvYyksIHlvdSBjYW4gZXhwbG9yZSB0aGUgbGF0ZXN0IHZlcnNpb24gb2YgdGhpcyBBUEkgdGhyb3VnaCBbYSBtb2Rlcm4gd2Vic2l0ZSBkZXBsb3llZCBpbiBHaXRIdWIgUGFnZXNdKGh0dHBzOi8vdHJlYm9sLWVjb21tZXJjZS5naXRodWIuaW8vYXBpLykuCg== readmeEtag: '"3834d787e90fa0260cf757deac88586bce070e23"' readmeLastModified: Fri, 16 Aug 2024 03:48:42 GMT repositoryId: 312004563 description: eCommerce web service REST specification created: '2020-11-11T15:03:50Z' updated: '2024-08-16T03:49:06Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: trebol-ecommerce logo: https://avatars.githubusercontent.com/u/75125264?v=4 license: GPL-3.0 repoEtag: '"534926a62773839a7b033b3150b7d851cebff5e25ff2669b48aaf2b6af2ee6e8"' repoLastModified: Fri, 16 Aug 2024 03:49:06 GMT foundInMaster: true category: - Testing - Server Implementations id: 9f9f6dc48233b4a6347515e8cd48e4fb - source: openapi3 tags repository: https://github.com/mtheusbrito/api-rest-java-spring-boot-3 v3: true id: f64d96da191ec0c694f02fedfca1496e repositoryMetadata: base64Readme: >- IyMjIyBTcHJpbmcgQm9vdCAzOiBEZXNlbnZvbHZlbmRvIHVtYSBBUEkgUmVzdCBlbSBKYXZhCgoKCiMjIyBSZXF1aXNpdG9zCgktIEphdmEgdmVyc2lvbiAxNwoJLSBtYXZlbgoJLSBNeVNRTDogNS41IH4gMTAuMy4zOC1NYXJpYURCCgkKIyMjIEJ1aWxkIAoKYGBgCmNkIHByb2plY3RfZm9sZGVyICYmIAptdm4gcGFja2FnZQpgYGAKCiMjIyBFeDogUnVuIGluIHByb2QKCmBgYApjZCBwcm9qZWN0X2ZvbGRlci90YXJnZXQgJiYKamF2YSAtRHNwcmluZy5wcm9maWxlcy5hY3RpdmU9cHJvZCAtRERBVEFTT1VSQ0VfVVJMPWpkYmM6bXlzcWw6Ly9pcF9wcm9kL2RhdGFiYXNlX3Byb2QgLUREQVRBU09VUkNFX1VTRVJOQU1FPXVzZXJuYW1lIC1EREFUQVNPVVJDRV9QQVNTV09SRD1wYXNzd29yZCAtamFyIGFwaS0wLjAuMS1TTkFQU0hPVC5qYXIKYGBg readmeEtag: '"331dc387cb79e9152313a0b3724bce99034b6ceb"' readmeLastModified: Thu, 30 Mar 2023 04:59:01 GMT repositoryId: 616289339 description: API rest utilizando Spring boot 3 created: '2023-03-20T04:51:05Z' updated: '2023-07-09T05:48:07Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: mtheusbrito logo: https://avatars.githubusercontent.com/u/13672114?v=4 repoEtag: '"5a56fbbc62cbf7ed8937663f79865261e464f40816b1e2e990b5c979411084ea"' repoLastModified: Sun, 09 Jul 2023 05:48:07 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/udnboss/generator v3: true id: 193b62f2938391dfcb3d229f6fad1a30 repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIEdlbmVyYXRvcg0KDQpUaGUgZm9sbG93aW5nIHNhbXBsZSBpbnB1dCBjYW4gYmUgdXNlZCB0byBnZW5lcmF0ZSBhbiBPcGVuQVBJIGRvY3VtZW50LiANCg0KYGBgDQphcGk6DQogIHRpdGxlOiBJbnZvaWNpbmcgQVBJDQogIGRlc2NyaXB0aW9uOiBJbnZvaWNpbmcgQVBJIFJlZmVyZW5jZQ0KICB2ZXJzaW9uOiAnMS4wJw0KICBwcmVmaXg6IC9hcGkvdjENCnNjcmlwdHM6DQogIG91dHB1dERpcjogLi9vdXRwdXQvaW52b2ljaW5nDQogIGZyYW1ld29yazogZXhwcmVzcyAjZXhwcmVzc3x3ZWJhcGl8ZmFzdGFwaQ0KaW50ZXJmYWNlczoNCiAgY29tcGFueToNCiAgICBpZDogDQogICAgbmFtZTogDQogICAgYWRkcmVzczogDQogICAgY3JuOiANCiAgICB0cm46IA0KICAgIGNvbnRhY3Q6IA0KICAgIG1vYmlsZTogDQogICAgZW1haWw6IHN0cnxlbWFpbA0KDQogIGNhdGVnb3J5Og0KICAgIGlkOiBzdHJpbmcNCiAgICBuYW1lOiBzdHJpbmcNCiAgICBpdGVtcz86IGl0ZW1bXQ0KDQogIGl0ZW06DQogICAgaWQ6IHN0cmluZw0KICAgIG5hbWU6IHN0cmluZw0KICAgIGNhdGVnb3J5X2lkPzogc3RyID4gY2F0ZWdvcnkuaWQNCiAgICBjYXRlZ29yeT86ID1jYXRlZ29yeQ0KYGBgDQoNClNvbWUgc3BlY2lhbCBzeW50YXggaXMgdXNlZDoNCg0KRXhhbXBsZSAxOiBgZW1haWw6IHN0cnxlbWFpbGAgbWVhbnMgdGhlIHByb3BlcnR5IGBlbWFpbGAgaGFzIGEgZGF0YSB0eXBlIGBzdHJpbmdgIGFuZCB0aGUgZm9ybWF0IGlzIGBlbWFpbGAuDQoNCkV4YW1wbGUgMjogYGNhdGVnb3J5X2lkPzogc3RyID4gY2F0ZWdvcnkuaWRgIG1lYW5zIHRoZSBgY2F0ZWdvcnlfaWRgIGlzIG9wdGlvbmFsIHByb3BlcnR5IChkdWUgdG8gZW5kaW5nIHdpdGggYD9gKSwgdGhlIGRhdGEgdHlwZSBpcyBgc3RyaW5nYCBhbmQgaXQgaXMgYSByZWZlcmVuY2UgYD5gIHRvIGFub3RoZXIgZW50aXR5IG9mIHR5cGUgYGNhdGVnb3J5YCBhbmQgdGhhdCBpdHMgdmFsdWUgaXMgaXRzIGBpZGAgcHJvcGVydHkuICh0aGluayBvZiBpdCBsaWtlIGEgZm9yZWlnbiBrZXkgcmVmZXJlbmNlIGluIGEgZGF0YWJhc2UgdGFibGUgdG8gYW5vdGhlciB0YWJsZSkuDQoNCkV4YW1wbGUgMzogYGl0ZW1zPzogaXRlbVtdYDogdGhlIGA/YCBtZWFucyBpdCBpcyBub3QgcmVxdWlyZWQsIGBpdGVtYCBtZWFucyBpdCBpcyBhIHJlZmVyZW5jZSB0byBhbm90aGVyIGtub3duIGVudGl0eSBvZiB0eXBlIGBpdGVtYCBhbmQgYFtdYCBtZWFucyBpdCBpcyBhbiBhcnJheSBvZiBpdGVtcy4NCg0KDQpZb3UgY2FuIHN0YXJ0IGJ5IHNhdmluZyBpbnB1dCBpbiBhIHlhbWwgZmlsZSwgYW5kIHJ1bm5pbmcgdGhlIGNvbW1hbmRzIHRvIHRlc3QgaXQuIEEgZnVsbCBpbnB1dCBleGFtcGxlIGlzIGxvY2F0ZWQgaW4gYHNjaGVtYXMvaW52b2ljaW5nLnltbGANCg0KUnVuIHRoaXMgY29tbWFuZCB0byB0ZXN0IGl0IChnZW5lcmF0ZSBqdXN0IE9wZW5BUEkgZG9jdW1lbnQpOg0KYHB5dGhvbiBnZW4ucHkgZG9jcyAuL3NjaGVtYXMvaW52b2ljaW5nLnltbGANCg0KWW91IGNhbiB2aWV3IGEgc2FtcGxlIG91dHB1dCBpbiBgLi9vdXRwdXQvaW52b2ljaW5nLnltbGAuDQoNClRoZSBvdXRwdXQgd2lsbCBiZSBzdG9yZWQgaW4gYC4vb3V0cHV0L2ludm9pY2luZy55bWxgLiBZb3UgY2FuIGNvcHkgdGhlIG91dHB1dCBhbmQgcGFzdGUgaXQgaW4gW1N3YWdnZXIgRWRpdG9yXShodHRwczovL2VkaXRvci5zd2FnZ2VyLmlvLykgdG8gdGVzdCBpdCBhbmQgdmFsaWRhdGUuDQoNCkV4YW1wbGUgdG8gZ2VuZXJhdGUgZXhwcmVzc0pzIGFwaSB0eXBlc2NyaXB0IGNvZGUsIChjb2RlIHdpbGwgYmUgZ2VuZXJhdGVkIGZvciB0aGUgc3BlY2lmaWVkIGZyYW1ld29yayBpbiBgc2NyaXB0cy5mcmFtZXdvcmtgKSwgcnVuOg0KYHB5dGhvbiBnZW4ucHkgYXBpIC4vc2NoZW1hcy9pbnZvaWNpbmcueW1sYA0KDQpUaGUgb3V0cHV0IHdpbGwgYmUgd3JpdHRlbiBpbiB0aGUgc3BlY2lmaWVkIGBzY3JpcHRzLm91dHB1dERpcmAgaW4gdGhlIHN1cHBsaWVkIGAueW1sYCBmaWxlLg0KDQpUaGUgZ2VuZXJhdGVkIGFwaSB3aWxsIGhhdmUgdGhlIGJhc2ljIGJ1c2luZXNzLCBpbnRlcmZhY2VzLCBjbGFzc2VzLCBhbmQgcm91dGVzLiBCYWNrZW5kIGNhbiBiZSBKU09OIG9yIFNxbGl0ZTMgKGRlZmF1bHQpIGZvciBub3cuDQoNClVwY29taW5nIGlzIHRvIGdlbmVyYXRlIHNxbCBmb3IgOnNxbGl0ZSwgcG9zdGdyZXMsIHNxbCBzZXJ2ZXIsIGV0Yy4= readmeEtag: '"d78c8f93937eafaa2012d94d79b455fb2b7ed31c"' readmeLastModified: Tue, 05 Sep 2023 18:06:12 GMT repositoryId: 675078533 description: >- A Python script to generate OpenAPI document specification for your API, based on a yaml file that is very minimal to describe your entities. created: '2023-08-05T17:44:48Z' updated: '2023-08-29T19:09:16Z' language: C# archived: false stars: 0 watchers: 1 forks: 0 owner: udnboss logo: https://avatars.githubusercontent.com/u/74108279?v=4 repoEtag: '"8ffb4cef76eb9affea17ef05b2138d16f94a6fd8aaa4c4e4b57f45bf4ea20331"' repoLastModified: Tue, 29 Aug 2023 19:09:16 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/glennfaison/mcq v3: true repositoryMetadata: base64Readme: >- IyBtY3EKQSBSRVNUIEFQSSBmb3IgYW4gTUNRIHdlYiBhcHBsaWNhdGlvbiwgd2l0aCBhIE1vbmdvREIgZGF0YWJhc2UuCg== readmeEtag: '"e8e56482a046a6278990af081e911adc61869daa"' readmeLastModified: Tue, 26 May 2020 09:37:20 GMT repositoryId: 254987221 description: A REST API for an MCQ web application, with a MongoDB database. created: '2020-04-12T01:29:19Z' updated: '2020-05-26T09:37:23Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 1 owner: glennfaison logo: https://avatars.githubusercontent.com/u/17485270?v=4 repoEtag: '"13d5cd2e572474074fee71de08535c880cdbc4b386c720c8bc981296bbed5381"' repoLastModified: Tue, 26 May 2020 09:37:23 GMT foundInMaster: true category: - Server - Server Implementations id: 92c056d9988d9c3d6b6801cb474f2cab - source: openapi3 tags repository: https://github.com/authena-ru/courses-organization v3: true repositoryMetadata: base64Readme: >- WyFbY29kZWNvdl0oaHR0cHM6Ly9jb2RlY292LmlvL2doL2F1dGhlbmEtcnUvY291cnNlcy1vcmdhbml6YXRpb24vYnJhbmNoL21haW4vZ3JhcGgvYmFkZ2Uuc3ZnP3Rva2VuPUZZSTk3U0hUM1gpXShodHRwczovL2NvZGVjb3YuaW8vZ2gvYXV0aGVuYS1ydS9jb3Vyc2VzLW9yZ2FuaXphdGlvbikKCiMgQXV0aGVuYSBjb3Vyc2VzIG9yZ2FuaXphdGlvbiBzZXJ2aWNlCgpDb3Vyc2VzIG9yZ2FuaXphdGlvbiBkb21haW4gZm9yIEF1dGhlbmEgY291cnNlIHBhc3NpbmcgcHJvamVjdAoKIyMgQnVpbGQgJiBSdW4gKExvY2FsbHkpCgojIyMgUHJlcmVxdWlzaXRlcwoKLSBnbyAxLjE3Ci0gRG9ja2VyCi0gZ29sYW5nY2ktbGludAoKIyMjIEVudmlyb25tZW50CgpZb3UgY2FuIGNyZWF0ZSAuZW52IGZpbGUgd2l0aCBmb2xsb3dpbmcgZW52aXJvbm1lbnQgdmFyaWFibGVzIG9yIHNldCB0aGVtIG1hbnVhbGx5OgoKYGBgZG90ZW52CkFQUF9FTlZJUk9OTUVOVD1sb2NhbCAjIEVudmlyb25tZW50IG5hbWUgYW5kIGNvbmZpZyBuYW1lIHRvIHBhcnNlCgpNT05HT19VUkk9bW9uZ29kYjovL21vbmdvZGI6MjcwMTcKTU9OR09fVVNFUk5BTUU9YWRtaW4KTU9OR09fUEFTU1dPUkQ9cXdlcnR5CmBgYAoKIyMjIENvbW1hbmRzCgotIGBgbWFrZSBvcGVuYXBpYGAg4oCUIGdlbmVyYXRlcyBib2lsZXJwbGF0ZSBjb2RlLCB0eXBlcyBhbmQgc2VydmVyIGludGVyZmFjZSB0aGF0IGNvbmZvcm1zIHRvIE9wZW5BUEkKLSBgYG1ha2UgZ28tYnVpbGRgYCDigJQgYnVpbGRzIHByb2plY3QgZm9yIEdPT1M9bGludXgKLSBgYG1ha2UgbGludGBgIOKAlCBydW5zIGxpbnRlcnMKLSBgYG1ha2UgZGV2YGAgLSBydW5zIGRldiBlbnZpcm9ubWVudAotIGBgbWFrZSB0ZXN0LXVuaXRgYCDigJQgcnVucyB1bml0IHRlc3RzIGFuZCBzYXZlIGNvdmVyIHByb2ZpbGUKLSBgYG1ha2UgdGVzdC1pbnRlZ3JhdGlvbmBgIOKAlCBydW5zIGludGVncmF0aW9uIHRlc3RzIGFuZCBzYXZlIGNvdmVyIHByb2ZpbGUKLSBgYG1ha2UgdGVzdC1jb3ZlcmBgIOKAlCBidWlsZHMgY29kZSBjb3ZlciByZXBvcnQKLSBgYG1ha2UgcnVuLXRlc3QtZGJgYCDigJQgcnVucyBEb2NrZXIgd2l0aCB0ZXN0IE1vbmdvIERCCi0gYGBtYWtlIHN0b3AtdGVzdC1kYmBgIOKAlCBzdG9wcyBEb2NrZXIgd2l0aCB0ZXN0IE1vbmdvIERCCg== readmeEtag: '"28349a3f30f7bd905cc5bed1d86d2565db602167"' readmeLastModified: Wed, 17 Nov 2021 10:26:43 GMT repositoryId: 370463122 description: Courses organization domain for Authena course passing project created: '2021-05-24T19:26:45Z' updated: '2021-11-17T10:26:47Z' language: Go archived: false stars: 0 watchers: 0 forks: 0 owner: authena-ru logo: https://avatars.githubusercontent.com/u/84408406?v=4 license: MIT repoEtag: '"5f1fb74768ec9d61c3ee0928eb0e025265723d5c2b1bf58f7d60cd3f3d5b5b62"' repoLastModified: Wed, 17 Nov 2021 10:26:47 GMT foundInMaster: true category: Testing id: 25b5b9e0c4ad3abfe66bac824fa1bf10 - source: openapi3 tags repository: https://github.com/juliandecoss/ymltoswaggerhtml v3: true repositoryMetadata: base64Readme: >- IyBTV0FHR0VSIEdFTkVSQVRPUiBIVE1MIEZJTEVTCgpUaGlzIFJlcG8gY3JlYXRlcyBiZWF1dGlmdWwgaHRtbCBzd2FnZ2VyIGZpbGVzLCB1c2luZyBZQU1MIGZpbGVzIGFuZCB2YWxpZGF0ZXMgdGhlbS4KCgojIyBHZXR0aW5nIHN0YXJ0ZWQKCk1ha2Ugc3VyZSB5b3UgaGF2ZSBpbnN0YWxsZWQKCi0gW1B5dGhvbiAzLjcrXShodHRwczovL3d3dy5weXRob24ub3JnL2Rvd25sb2Fkcy8pCgoKIyMgSW5zdGFsbGF0aW9uCgpUbyBpbnN0YWxsIHRoZSBsaWJyYXJpZXMsIHJ1biBpbiB5b3VyIHRlcm1pbmFsLgoKYGBgc2gKbWFrZSBpbnN0YWxsCmBgYAoKVGhpcyB3aWxsIHNldHVwIHlvdXIgZW52aXJvbm1lbnQgdG8gYWRkIHRoZSAyIGxpYnJhcmllcyB0aGlzIHJlcG8gcmVxdWlyZXMuCgoKIyMgQWRkaW5nIHlvdXIgIFlBTUwgZG9jcwoKSW4gb3JkZXIgdG8gdmFsaWRhdGUgeW91ciBmaWxlcyBhZGQgeW91ciBkb2NzIHRvIHRoZSBkaXJlY3RvcnkgZG9jcy9vcGVuYXBpLgoKCiMjIFZhbGlkYXRpbmcgeW91ciB5YW1sIGZpbGVzCgpUbyB2YWxpZGF0ZSB5b3VyIGZpbGVzIGZpcnN0IGFkZCB0aGVtIGluIHRoZSBkaXJlY3Rvcnkgb3BlbmFwaSwgdGhlbiBydW4gaW4geW91ciB0ZXJtaW5hbDoKCmBgYHNoCm1ha2Ugc3dhZ2dlci12YWxpZGF0aW9uCmBgYAoKIyMgQ3JlYXRpbmcgeW91ciBodG1sIGJlYXV0aWZ1bCBzd2FnZ2VyIHN0YXRpYyBkb2NzCkluIG9yZGVyIHRvIGNvbm5lY3QgeW91ciBmaWxlcyBiZXR3ZWVuIGVhY2ggb25lLCBjaGFuZ2UgdGhlIGZpbGUgdGVtcGxhdGUgaW4gc2NyaXB0cy90ZW1wbGF0ZXMvc3dhZ2dlcl90ZW1wbGF0ZSwgeW91IHdpbGwgZmluZCBpbiB0aGUgbGluZSAzNiB0aGF0IHRoZXJlIGlzIGEgInNlbGVjdCIgaHRtbCBlbGVtZW50LCBhZGQgYWxsIHlvdXIgZmlsZXMgdGhlcmUsIGFzIG9wdGlvbnMgIjxvcHRpb24gdmFsdWU9Im5hbWVfb2ZfeW91cl9maWxlIj5uYW1lX29mX3lvdXJfZmlsZTwvb3B0aW9uPiIgaXRzIG5vdCBuZWNlc3NhcnkgdG8gYWRkIHRoZSBzdWZpeCBodG1sLgoKT25jZSB5b3UgZmluaXNoIHRoaXMgc3RlcCBqdXN0IHJ1biBpbiB5b3VyIHRlcm1pbmFsCgpgYGBzaAptYWtlIHN3YWdnZXItc3RhdGljCmBgYAoKCiMjIyMgQ3JlZGl0cyB0bwpUaGlzIHJlcG8gd2FzIGJ1aWx0IHVzaW5nIHRoZSBzY3JpcHQgc2hvd24gYmVsb3c6Ci0gaHR0cHM6Ly9naXN0LmdpdGh1Yi5jb20vb3NlaXNrYXIvZGJkNTFhMzcyN2ZjOTZkY2Y1ZWQxODlmY2E0OTFmYjM= readmeEtag: '"170112849bb6c0858b4811d69f547a5095ef07ff"' readmeLastModified: Sat, 24 Jul 2021 02:26:43 GMT repositoryId: 388936192 description: >- 📖 Generate beautiful html files with a swagger template using YAML/YML files. 🚀🚀 created: '2021-07-23T21:59:03Z' updated: '2021-07-24T22:05:11Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: juliandecoss logo: https://avatars.githubusercontent.com/u/72807525?v=4 license: Apache-2.0 repoEtag: '"8a91cd01bb769b814e0d70f5216015582fe14b6704f4ccddbf1b993b92e54abc"' repoLastModified: Sat, 24 Jul 2021 22:05:11 GMT foundInMaster: true category: - Testing - Server Implementations id: 7cece16cf56bc26a12f14a7356d46509 - source: openapi3 tags repository: https://github.com/sundalei/book-management v3: true id: c4dbc45096493794f18a9442090a78c3 repositoryMetadata: base64Readme: >- IyBib29rLW1hbmFnZW1lbnQKIyMgc3RlcCAxIHZhbGlkYXRlIHRlbXBsYXRlIHByb2plY3Qgd29ya2luZwpgYGBqYXZhCkBSZXN0Q29udHJvbGxlcgpwdWJsaWMgY2xhc3MgSGVsbG9Db250cm9sbGVyIHsKICAgIAogICAgQEdldE1hcHBpbmcoIi8iKQogICAgcHVibGljIFN0cmluZyBncmVldGluZygpIHsKICAgICAgICByZXR1cm4gImhlbGxvIHdvcmxkIjsKICAgIH0KfQpgYGAKCiMjIHN0ZXAgMiB1c2UgbW9uZ29kYiBhcyBiYWNrZW5kIHN0b3JhZ2UgYW5kIHNwcmluZyBkYXRhIG1vbmdvIApCb29rUmVwb3NpdG9yeQ== readmeEtag: '"5d701975d5d429b0f721229d19fe367320dbec87"' readmeLastModified: Sat, 28 May 2022 08:08:46 GMT repositoryId: 493049552 description: Book Management created: '2022-05-17T01:09:02Z' updated: '2022-09-09T00:46:01Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: sundalei logo: https://avatars.githubusercontent.com/u/12974121?v=4 repoEtag: '"d87dc62cb07381cb992cad59e2c95cc5a7e8054cc88e85b6d16a0ce2a8a06c94"' repoLastModified: Fri, 09 Sep 2022 00:46:01 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/a-sync/ocap-odata v3: true id: 8f91cbfa40dea0d3af43a84d56edb4ea repositoryMetadata: base64Readme: >- IyBvY2FwLW9kYXRhCgojIEFib3V0CiAqIHByb3ZpZGVzIE9EYXRhIHY0LjAxIEFQSSBmb3Igb2NhcC1zdGF0cyBkYXRhYmFzZQogKiBidWlsdCBvbiBsYXJhdmVsIDEwICsgbG9hZGF0YSAyCiAqIG9ubHkgYSB0aGluIGxheWVyIG9mIGVsb3F1ZW50IG1vZGVscyBhbmQgbG9kYXRhIHRyYWl0cwoKIyBEZW1vIHNpdGVzCltmbmYtb2RhdGEuZGV2cy5zcGFjZV0oaHR0cHM6Ly9mbmYtb2RhdGEuZGV2cy5zcGFjZSkgIAogJm5ic3A7ICZyZGNhOyB1c2luZyBbZm5mLXN0YXRzXShodHRwczovL2ZuZi1zdGF0cy5kZXZzLnNwYWNlKSBkYXRhYmFzZSAgCiAmbmJzcDsgJm5ic3A7ICZyZGNhOyB1c2luZyBbT0NBUDJdKGh0dHA6Ly9hYXIuZnJpZGF5bmlnaHRmaWdodC5vcmcpIGRhdGEgZnJvbSBbRk5GXShodHRwczovL3d3dy5mcmlkYXluaWdodGZpZ2h0Lm9yZykKCiMgSG9zdGluZyByZXF1aXJlbWVudHMKICogUEhQIDguMSBvciBsYXRlcgogKiBhY2Nlc3MgdG8gb2NhcC1zdGF0cyBkYXRhYmFzZSAobXlzcWwvbWFyaWFkYikKCllvdSBzaG91bGQgdXNlIGVudiB2YXJzIChlbnYuZXhhbXBsZSkgdG8gY29uZmlndXJlIHRoZSBkYXRhYmFzZSBob3N0L3VzZXIvcHcvZGJuYW1lLCBidXQgeW91IG9ubHkgbmVlZCB0byBlZGl0IHRoZSBbREJfVVJMXSguL2NvbmZpZy9kYXRhYmFzZS5waHAjTDQ4KSBnZW5lcmFsbHkuCgojIERldmVsb3BtZW50IHJlcXVpcmVtZW50cwogKiBkb2NrZXItY29tcG9zZSAyIG9yIGxhdGVyCiAqIGNvbXBvc2VyIDIuMiBvciBsYXRlcgoKIyMgU2V0dXAKSW5zdGFsbCBjb21wb3NlciBkZXBlbmRlbmNpZXMuCmBgYApjb21wb3NlciBpbnN0YWxsCmBgYAoKIyMgQ29uZmlnCkNyZWF0ZSBhIC5lbnYgZmlsZSB0byBlbmFibGUgbGFyYXZlbCBkZWJ1ZyBvcHRpb25zLgpgYGAKQVBQX0VOVj1sb2NhbApBUFBfS0VZPQpBUFBfREVCVUc9dHJ1ZQpMT0dfTEVWRUw9ZGVidWcKYGBgCgojIyBSdW5uaW5nIGxvY2FsbHkK4pqgIFlvdSBuZWVkIHRvIGNvbm5uZWN0IHRvIGFuIGV4aXN0aW5nIG9jYXAtc3RhdHMgZGF0YWJhc2UuICAKVG8gcXVpY2tseSBzcGluIHVwIHlvdXIgb3duIGxvY2FsIGRhdGFiYXNlIGhlYWQgb3ZlciB0byBbb2NhcC1zdGF0cy9TRVRVUC5tZF0oaHR0cHM6Ly9naXRodWIuY29tL2Etc3luYy9vY2FwLXN0YXRzL2Jsb2IvbWFzdGVyL1NFVFVQLm1kKSBmb3IgaW5zdHJ1Y3Rpb25zLgoKIyMjIyBTdGFydGluZyB1cCB0aGUgZGV2IGVudjoKYGBgCmRvY2tlci1jb21wb3NlIHVwCmBgYAoKIyMjIyMgU2VydmljZXMKICogV2ViIAogICAgKiBodHRwOi8vbG9jYWxob3N0Ojg1MDAvCgojIyMgQ2hlY2tpbmcgZGIgY29ubmVjdGlvbgpgYGAKcGhwIGFydGlzYW4gZGI6c2hvdwpgYGAKCiMjIyMgU2h1dHRpbmcgZG93biB0aGUgZGV2IGVudgpgYGAKZG9ja2VyLWNvbXBvc2UgZG93bgpgYGAKCiMjIyMgU3RhcnRpbmcgdXAgdGhlIGRldiBlbnYgYWdhaW4KYGBgCmRvY2tlci1jb21wb3NlIHVwIC1kCmBgYAoKIyMjIENsZWFuIHJlc3RhcnQKYGBgCmRvY2tlci1jb21wb3NlIGRvd24gLXYKZG9ja2VyIHN5c3RlbSBwcnVuZSAtYSAtZiAtLXZvbHVtZXMKY29tcG9zZXIgaW5zdGFsbApwaHAgYXJ0aXNhbiByb3V0ZTpjbGVhcgpwaHAgYXJ0aXNhbiBjb25maWc6Y2xlYXIKcGhwIGFydGlzYW4gY2FjaGU6Y2xlYXIKZG9ja2VyLWNvbXBvc2UgYnVpbGQgLS1uby1jYWNoZQpkb2NrZXItY29tcG9zZSB1cCAtZApgYGAK readmeEtag: '"0538b3cfcb7e78f5e4f9e4f37a8eae8a6686ab3c"' readmeLastModified: Thu, 09 May 2024 20:40:38 GMT repositoryId: 450753308 description: OData API for OCAP data (ocap-stats database) created: '2022-01-22T07:55:37Z' updated: '2024-05-09T20:40:30Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: a-sync logo: https://avatars.githubusercontent.com/u/14183614?v=4 repoEtag: '"e888eb0a0de4929def1943f4d76e3fbde6c3a119a7a7271504f5fede56e09979"' repoLastModified: Thu, 09 May 2024 20:40:30 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/alvarosps/digital-account-server v3: true id: 78b0cbe1eaabe3d4034dcc358599c5a7 repositoryMetadata: base64Readme: >- IyBCYW5rIEFQSQoKVGhpcyBpcyBhbiBBUEkgZm9yIGEgYmFua2luZyBzeXN0ZW0gdGhhdCBhbGxvd3MgdXNlcnMgdG8gY3JlYXRlIGFuZCBtYW5hZ2UgYmFuayBhY2NvdW50cy4KCiMjIFRlY2hub2xvZ2llcwoKVGhpcyBwcm9qZWN0IHVzZXMgdGhlIGZvbGxvd2luZyB0ZWNobm9sb2dpZXM6CgotIE5vZGUuanMKLSBFeHByZXNzLmpzCi0gQVdTIER5bmFtbyBEQgoKIyMgU2V0dXAKClRvIHNldCB1cCB0aGlzIHByb2plY3QsIGZvbGxvdyB0aGVzZSBzdGVwczoKCi0gQ2xvbmUgdGhpcyByZXBvc2l0b3J5IHRvIHlvdXIgbG9jYWwgbWFjaGluZS4KLSBJbnN0YWxsIGRlcGVuZGVuY2llcyBieSBydW5uaW5nIG5wbSBpbnN0YWxsLgotIFNldCB0aGUgZW52aXJvbm1lbnQgdmFyaWFibGVzIGRlZmluZWQgaW4gLmVudi5leGFtcGxlLiBZb3UgY2FuIGVpdGhlciBjcmVhdGUgYSAuZW52IGZpbGUgb3Igc2V0IHRoZW0gaW4geW91ciBlbnZpcm9ubWVudC4KLSBTdGFydCB0aGUgc2VydmVyIGJ5IHJ1bm5pbmcgbnBtIHN0YXJ0LgotIFlvdSB3aWxsIG5lZWQgdG8gY3JlYXRlIGEgLmVudiBmaWxlIGluIHRoZSByb290IGZvbGRlciBvZiB0aGUgcHJvamVjdCwgY29udGFpbmluZzoKYGBgCkFXU19BQ0NFU1NfS0VZX0lEPTx5b3VyIGF3cyBhY2Nlc3Mga2V5PgpBV1NfU0VDUkVUX0FDQ0VTU19LRVk9PHlvdXIgYXdzIHNlY3JldCBhY2Nlc3Mga2V5PgpBV1NfUkVHSU9OPTxhd3MgcmVnaW9uIHlvdSBhcmUgdXNpbmc+CkFVVEgwX0FVRElFTkNFPWh0dHBzOi8vc3BzZmFrZWJhbmthY2NvdW50cy5jb20vYXBpCkFVVEgwX0lTU1VFUl9CQVNFX1VSTD1odHRwczovL2Rldi1zeG1nNDA4bnduc2UwNWp1LnVzLmF1dGgwLmNvbS8KQVVUSDBfVE9LRU5fU0lHTklOR19BTEdPUklUSE09UlMyNTYKYGBgCgpTZXJ2ZXIgd2lsbCBydW4gb24gaHR0cDovL2xvY2FsaG9zdDozMDAwCgojIyBBdXRoZW50aWNhdGlvbgoKVGhpcyBBUEkgdXNlcyBhdXRoMCBhdXRoZW50aWNhdGlvbiwgeW91J2xsIG5lZWQgdG8gaGF2ZSB0aGUgaGVhZGVyICdBdXRob3JpemF0aW9uJyBpbiB0aGUgcmVxdWVzdHMgKGFuZCBBUEkgRG9jdW1lbnRhdGlvbikgdG8gdXNlIGl0LgoKIyMjIEdldHRpbmcgdGhlIGFjY2Vzc190b2tlbgoKWW91J2xsIG5lZWQgdG8gbWFrZSBhIFBPU1QgY2FsbCB0byBodHRwczovL2Rldi1zeG1nNDA4bnduc2UwNWp1LnVzLmF1dGgwLmNvbS9vYXV0aC90b2tlbgpJbiB0aGUgQm9keSBvZiB0aGUgcmVxdWVzdCwgeW91IHNob3VsZCBwdXQgKGZpbGxpbmcgaW4gdGhlIGNsaWVudF9pZCBhbmQgY2xpZW50X3NlY3JldCB3aXRoIHlvdXIgY3JlZGVudGlhbHMpOgoKYGBgCnsKICAiY2xpZW50X2lkIjogIiIsCiAgImNsaWVudF9zZWNyZXQiOiAiIiwKICAiYXVkaWVuY2UiOiAiaHR0cHM6Ly9zcHNmYWtlYmFua2FjY291bnRzLmNvbS9hcGkiLAogICJncmFudF90eXBlIjogImNsaWVudF9jcmVkZW50aWFscyIKfQoKYGBgCgpUaGlzIHdpbGwgaGF2ZSBhIHJlc3BvbnNlIGxpa2UgdGhpczoKCmBgYAp7CiAgICAiYWNjZXNzX3Rva2VuIjogIjxhY2Nlc3NfdG9rZW5fZnJvbV9hdXRoMD4iLAogICAgImV4cGlyZXNfaW4iOiA4NjQwMCwKICAgICJ0b2tlbl90eXBlIjogIkJlYXJlciIKfQpgYGAKClRoZW4sIGluIHRoZSByZXF1ZXN0cyB5b3UgbWFrZSwgeW91J2xsIGhhdmUgdGhlIGZvbGxvd2luZyBIZWFkZXI6CgpgYGAKQXV0aG9yaXphdGlvbjogQmVhcmVyIDxhY2Nlc3NfdG9rZW5fZnJvbV9hdXRoMD4KYGBgCgojIyBVc2FnZQoKT25jZSB0aGUgc2VydmVyIGlzIHVwIGFuZCBydW5uaW5nLCB5b3UgY2FuIHVzZSBhbiBBUEkgY2xpZW50IGxpa2UgUG9zdG1hbiB0byBpbnRlcmFjdCB3aXRoIHRoZSBlbmRwb2ludHMuIFRoZSBBUEkgc3VwcG9ydHMgdGhlIGZvbGxvd2luZyBvcGVyYXRpb25zOgoKLSBDcmVhdGluZyBhbiBhY2NvdW50IGhvbGRlcgotIEdldHRpbmcgYWxsIGFjY291bnQgaG9sZGVycwotIEdldHRpbmcgYW4gYWNjb3VudCBob2xkZXIgYnkgSUQKLSBVcGRhdGluZyBhbiBhY2NvdW50IGhvbGRlcgotIERlbGV0aW5nIGFuIGFjY291bnQgaG9sZGVyCi0gQ3JlYXRpbmcgYSBiYW5rIGFjY291bnQKLSBHZXR0aW5nIGFsbCBiYW5rIGFjY291bnRzCi0gR2V0dGluZyBhIGJhbmsgYWNjb3VudCBieSBJRAotIFVwZGF0aW5nIGEgYmFuayBhY2NvdW50Ci0gQ2xvc2luZyBhIGJhbmsgYWNjb3VudAotIEJsb2NraW5nIGEgYmFuayBhY2NvdW50Ci0gVW5ibG9ja2luZyBhIGJhbmsgYWNjb3VudAotIERlcG9zaXRpbmcgbW9uZXkgaW50byBhIGJhbmsgYWNjb3VudAotIFdpdGhkcmF3aW5nIG1vbmV5IGZyb20gYSBiYW5rIGFjY291bnQKCiMjIEFQSSBEb2N1bWVudGF0aW9uCgpUaGUgQVBJIGRvY3VtZW50YXRpb24gaXMgYXZhaWxhYmxlIGluIHRoZSBPcGVuQVBJIGZvcm1hdCBhbmQgY2FuIGJlIGZvdW5kIGluIHRoZSAuL29wZW5hcGkvb3BlbmFwaS55bWwgZmlsZS4KCllvdSBjYW4gYWNjZXNzIGl0IG9uIGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMC9hcGktZG9jcyAoWW91IG5lZWQgdG8gcHV0IHRoZSBBdXRob3JpemF0aW9uIEhlYWRlciBvbiB0aGUgQnJvd3NlciB0byBhY2Nlc3MgdGhlIGRvY3VtZW50YXRpb24gZGlyZWN0bHkpCg== readmeEtag: '"cb870fa93da3693a37c2233549a48b3b0fb5d9c7"' readmeLastModified: Fri, 24 Mar 2023 19:26:39 GMT repositoryId: 618577049 description: >- A NodeJS / Typescript API Server, that uses Express, AWS Dynamo DB, Auth0 Authentication, OpenAPI for documentation created: '2023-03-24T19:22:39Z' updated: '2023-03-24T19:25:03Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: alvarosps logo: https://avatars.githubusercontent.com/u/22122640?v=4 repoEtag: '"35f822b9c085c4f403e4b0cccba8585ca0470d0be217c07a84b1aa0a02eaf56d"' repoLastModified: Fri, 24 Mar 2023 19:25:03 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/fehho/mydirt-api v3: true id: 91bb2ecdeaf81379b325bec141dd9cd2 repositoryMetadata: base64Readme: >- IyBteURpUlQtYXBpCgojIyBHZXR0aW5nIFN0YXJ0ZWQKCkZpcnN0LCBpbnN0YWxsIGRlcGVuZGVuY2llcy4KCmBgYGJhc2ggCmNwYW5tIC0taW5zdGFsbGRlcHMgLgpgYGAKClJ1biB0aGUgZGV2ZWxvcG1lbnQgd2Vic2VydmVyCgpgYGBiYXNoCm1vcmJvIHNjcmlwdC9teV9hcHAKYGBgCgpPcGVuIGh0dHA6Ly9sb2NhbGhvc3Q6MzAwMC9hcGkvaGVhbHRoIHdpdGggeW91ciBicm93c2VyIHRvIHNlZSB0aGUgcmVzdWx0LgoKQVBJIHJvdXRlcyBjYW4gYmUgYWNjZXNzZWQgb24gaHR0cDovL2xvY2FsaG9zdDozMDAwL2FwaQoKIyMgTGVhcm4gTW9yZQoKW01vam9saWNpb3VzXShodHRwczovL21ldGFjcGFuLm9yZy9wb2QvTW9qb2xpY2lvdXMpIC0gd2ViIGZyYW1ld29yawoKW09wZW5BUEldKGh0dHBzOi8vc3dhZ2dlci5pby9zcGVjaWZpY2F0aW9uLykgLSBBUEkgc3RhbmRhcmQKCltEQkl4OjpDbGFzc10oaHR0cHM6Ly9tZXRhY3Bhbi5vcmcvZGlzdC9EQkl4LUNsYXNzL3ZpZXcvbGliL0RCSXgvQ2xhc3MvTWFudWFsL0RvY01hcC5wb2QpIC0gb2JqZWN0IDwtPiByZWxhdGlvbmFsIG1hcHBlci4K readmeEtag: '"df562c3fa60f68fb97e63aa2195be56f5860fcc7"' readmeLastModified: Mon, 09 Jan 2023 16:36:35 GMT repositoryId: 577360080 description: Scorpion Camp winter 2022 devjam participant created: '2022-12-12T15:07:54Z' updated: '2022-12-22T14:26:36Z' language: Perl archived: false stars: 0 watchers: 2 forks: 0 owner: fehho logo: https://avatars.githubusercontent.com/u/116404887?v=4 repoEtag: '"27029157374c85de08ed9cc23ea4e7180ea609798aa3465a047954871d423dbd"' repoLastModified: Thu, 22 Dec 2022 14:26:36 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/open-banking/statements v3: true repositoryMetadata: base64Readme: >- IyBzdGF0ZW1lbnRzCk9wZW4gQmFua2luZyBTdGF0ZW1lbnRzIEFQSSBidWlsdCBvbiB0b3Agb2YgbGlnaHQtNGoK readmeEtag: '"1eefcc9eca4e906f055d6cdc1c0391035cc99499"' readmeLastModified: Thu, 25 Jul 2024 20:09:53 GMT repositoryId: 235867437 description: Open Banking Statements API built on top of light-4j created: '2020-01-23T19:14:04Z' updated: '2026-02-03T01:08:48Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"6ca812c04d2c2b95da551fdebc6415c7d89a2272c1a764c86bae6093fa563a9d"' repoLastModified: Tue, 03 Feb 2026 01:08:48 GMT foundInMaster: true category: - SDK - Server - Parsers id: 6232a03609eccb95b6e2af51c4a66cda - source: openapi3 tags repository: https://github.com/d00nik/hypixel-sdk v3: true id: 91be572de321d556ddfa1314989c374e repositoryMetadata: base64Readme: >- IyBEaXNjbGFpbWVyCgpUaGlzIHBhY2thZ2UgaXNuJ3QgaW4gYW55IG1lYW4gYWN0aXZlbHkgc3VwcG9ydGVkIGFuZCBzaG91bGRuJ3QgYmUgdXNlIGluIGRpc3RyaWJ1dGlvbi4gSW5zdGVhZCB5b3Ugc2hvdWxkIGZldGNoIHRoZSBhcGkgeW91cnNlbGYgb3IgdXNlIFtoeXBpeGVsLnRzXShodHRwczovL2dpdGh1Yi5jb20vaHlwaXhlbHRzL2h5cGl4ZWwudHMpLiBUaGlzIHBhY2thZ2UgaXMgbWVhbnQgdG8gYmUgcHVyZWx5IHVzZWQgZm9yIHRlc3RpbmcuIENvbnRpbnVlIGF0IHlvdXIgb3duIHJpc2suCgpUaGUgcmVhc29uIGZvciAqd2VpcmQqIG5hbWluZyBhbmQgY29kZSBpcyB0aGUgd2F5IGhvdyBbT3BlbkFQSSBHZW5lcmF0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9PcGVuQVBJVG9vbHMvb3BlbmFwaS1nZW5lcmF0b3IpIHdvcmtzLiBUaGlzIGFsbCB3YXMgZ2VuZXJhdGVkIGZyb20gSHlwaXhlbCdzIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAoU2VlIHRoZSBiZWdnaW5uaW5nIG9mIFtkb2NzXShodHRwczovL2FwaS5oeXBpeGVsLm5ldCkpCgojIFVzYWdlCgpJbnN0YWxsIGZyb20gbnBtIHdpdGg6CmBgYGJhc2gKbnBtIGkgaHlwaXhlbC1zZGsKYGBgClwKSW1wb3J0IGBDb25maWd1cmF0aW9uYCBhbmQgZGVzaXJlZCBBUEkgKGUuZy4gYFBsYXllckRhdGFBcGlgKToKYGBganMKaW1wb3J0IHsgQ29uZmlndXJhdGlvbiwgUGxheWVyRGF0YUFwaSB9IGZyb20gImh5cGl4ZWwtc2RrIjsKYGBgClwKQ3JlYXRlIGEgbmV3IGBDb25maWd1cmF0aW9uYCBhbmQgQVBJIE9iamVjdC4gKFJlbWVtYmVyIHRvIHByb3ZpZGUgeW91ciBvd24gYXBpIGtleSkKYGBganMKY29uc3QgY29uZmlnID0gbmV3IENvbmZpZ3VyYXRpb24oewogIGFwaUtleTogcHJvY2Vzcy5lbnYuSFlQSVhFTF9BUElfS0VZLAp9KTsKCmNvbnN0IHBsYXllckFwaSA9IG5ldyBQbGF5ZXJEYXRhQXBpKGNvbmZpZyk7CmBgYApcClVzZSBhbiBBUEkganVzdCBob3cgaXMgaXQgZGVzY3JpYmVkIGluIFtkb2NzXShodHRwczovL2FwaS5oeXBpeGVsLm5ldCkgZS5nLiBnZXQgcGxheWVyIGRhdGEuCmBgYGpzCmNvbnN0IHVzZXIgPSBhd2FpdCBwbGF5ZXJBcGkKICAucGxheWVyR2V0KHsKICAgIHV1aWQ6ICJlOGZlNjViZi0zYjI3LTQ1OGQtODM5Yi02ZDI5ZTE0NTcyNjEiLAogIH0pCiAgLmNhdGNoKChlcnIpID0+IGNvbnNvbGUubG9nKGVycikpOwoKY29uc29sZS5sb2codXNlcik7CmBgYAoKIwo= readmeEtag: '"960f9e55ccdac1b78d4b485501c01557ead133ba"' readmeLastModified: Sat, 12 Aug 2023 06:58:56 GMT repositoryId: 677497517 description: HypixelAPI SDK generated with OpenAPI SDK Generator created: '2023-08-11T18:14:30Z' updated: '2023-08-12T07:03:04Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: D00NIK logo: https://avatars.githubusercontent.com/u/65020944?v=4 repoEtag: '"1ff5a1fd30d694067ba1fb2affe68a492517c4dd39a2a233dbe6a12e86df13d0"' repoLastModified: Sat, 12 Aug 2023 07:03:04 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/alfabravo2013/hyper-id-backend v3: true id: a8a7f95525ed045bfdeed6531cac8b86 repositoryMetadata: base64Readme: >- IyBIeXBlcklECgojIyBIeXBlcnNraWxsIGZ1bGwgc3RhY2sgcHJvamVjdCwgYmFja2VuZCBSRVNUIEFQSQoKWyFbQnVpbGQgRXhlY3V0YWJsZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9hbGZhYnJhdm8yMDEzL2h5cGVyLWlkLWJhY2tlbmQvYWN0aW9ucy93b3JrZmxvd3MvbmF0aXZlLWFydGlmYWN0LnltbC9iYWRnZS5zdmc/YnJhbmNoPW1hc3RlciZldmVudD13b3JrZmxvd19kaXNwYXRjaCldKGh0dHBzOi8vZ2l0aHViLmNvbS9hbGZhYnJhdm8yMDEzL2h5cGVyLWlkLWJhY2tlbmQvYWN0aW9ucy93b3JrZmxvd3MvbmF0aXZlLWFydGlmYWN0LnltbCkKWyFbVGVzdHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9hbGZhYnJhdm8yMDEzL2h5cGVyLWlkLWJhY2tlbmQvYWN0aW9ucy93b3JrZmxvd3MvdGVzdHMueW1sL2JhZGdlLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9hbGZhYnJhdm8yMDEzL2h5cGVyLWlkLWJhY2tlbmQvYWN0aW9ucy93b3JrZmxvd3MvdGVzdHMueW1sKQoKIyMjIEhvdyB0byBydW4KCiMjIyMgT3B0aW9uIDEuIERvd25sb2FkIGFuIGV4ZWN1dGFibGUgZm9yIHlvdXIgT1M6CgotIFtXaW5kb3dzX3g4Nl82NF0oaHR0cHM6Ly9naXRodWIuY29tL2FsZmFicmF2bzIwMTMvaHlwZXItaWQtYmFja2VuZC9yZWxlYXNlcy9kb3dubG9hZC92MC40L2h5cGVyaWQtV2luZG93cy1zbmFwc2hvdC14ODZfNjQuemlwKQotIFtMaW51eF94ODZfNjRdKGh0dHBzOi8vZ2l0aHViLmNvbS9hbGZhYnJhdm8yMDEzL2h5cGVyLWlkLWJhY2tlbmQvcmVsZWFzZXMvZG93bmxvYWQvdjAuNC9oeXBlcmlkLUxpbnV4LXNuYXBzaG90LXg4Nl82NC56aXApCi0gW01hY09TX3g4Nl82NF0oaHR0cHM6Ly9naXRodWIuY29tL2FsZmFicmF2bzIwMTMvaHlwZXItaWQtYmFja2VuZC9yZWxlYXNlcy9kb3dubG9hZC92MC40L2h5cGVyaWQtbWFjT1Mtc25hcHNob3QteDg2XzY0LnppcCkKCmFuZCBydW4gb24geW91ciBjb21wdXRlci4KCiMjIyMgT3B0aW9uIDIuIERvd25sb2FkIHRoZSBleGVjdXRhYmxlIGphcjoKCi0gW2h5cGVyLWlkLWJhY2tlbmQuamFyXShodHRwczovL2dpdGh1Yi5jb20vYWxmYWJyYXZvMjAxMy9oeXBlci1pZC1iYWNrZW5kL3JlbGVhc2VzL2Rvd25sb2FkL3YwLjQvaHlwZXItaWQtYmFja2VuZC0wLjAuNC1TTkFQU0hPVC5qYXIpCgppbnN0YWxsIEpESyAxNyBhbmQgZXhlY3V0ZSB0aGUgZm9sbG93aW5nIGNvbW1hbmQgaW4gdGhlIGRpcmVjdG9yeSB3aGVyZSB0aGUgamFyIGZpbGUgaXMgbG9jYXRlZDoKCmBgYGJhc2gKamF2YSAtamFyIGh5cGVyLWlkLWJhY2tlbmQtMC4wLjQtU05BUFNIT1QuamFyCmBgYAoKIyMjIyBPcHRpb24gMy4gRG93bmxvYWQgdGhlIHNvdXJjZSBjb2RlOgoKLSBbLnppcF0oaHR0cHM6Ly9naXRodWIuY29tL2FsZmFicmF2bzIwMTMvaHlwZXItaWQtYmFja2VuZC9hcmNoaXZlL3JlZnMvdGFncy92MC40LnppcCkgb3IgCi0gWy50YXIuZ3pdKGh0dHBzOi8vZ2l0aHViLmNvbS9hbGZhYnJhdm8yMDEzL2h5cGVyLWlkLWJhY2tlbmQvYXJjaGl2ZS9yZWZzL3RhZ3MvdjAuNC50YXIuZ3opCgp1bnppcCBpdCwgaW5zdGFsbCBKREsgMTcgYW5kIGV4ZWN1dGUgdGhlIGZvbGxvd2luZyBjb21tYW5kIGluIHRoZSBwcm9qZWN0IHJvb3QgZGlyZWN0b3J5OgoKYGBgYmFzaAouL2dyYWRsZXcgYm9vdFJ1bgpgYGAKClRoZSBzZXJ2ZXIgcmVzcG9uZHMgb24gYGxvY2FsaG9zdDo4MDgwYAoKVGhlIGxhbmRpbmcgcGFnZSByZWRpcmVjdHMgdG8gQVBJIGRvY3MgKFN3YWdnZXIpCg== readmeEtag: '"9897c8741cc56e3053f562f20235ad9ffcd05134"' readmeLastModified: Wed, 23 Nov 2022 23:20:10 GMT repositoryId: 563327718 description: User profile REST API created: '2022-11-08T11:41:28Z' updated: '2023-07-30T23:06:38Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: alfabravo2013 logo: https://avatars.githubusercontent.com/u/65685808?v=4 repoEtag: '"05086feaa8151d209a1aa66cc2344bba2e76f5da43d0a7174b697135dcde9cab"' repoLastModified: Sun, 30 Jul 2023 23:06:38 GMT category: Server Implementations foundInMaster: true - name: Spring Cloud Gateway for Kubernetes source: Tooling repository issues source_description: >- Spring Cloud Gateway for Kubernetes provides an implementation of Spring Cloud Gateway, along with integrating other Spring ecosystem projects such as Spring Security, Spring Session, and more. This product includes commercial-only features on top of those open source including OpenAPI auto-generated documentation. link: >- https://docs.vmware.com/en/VMware-Spring-Cloud-Gateway-for-Kubernetes/index.html v3_1: false v3: true v2: true sourceIssueMetadata: issueNumber: 70 author: Albertoimpl createdAt: '2023-03-30T09:37:02Z' updatedAt: '2023-03-30T09:50:49Z' url: https://github.com/OAI/Tooling/issues/70 status: closed id: a2f27ec087f8bf10d44fa8fe1df17518 foundInMaster: true - name: API portal source: Tooling repository issues source_description: >- API portal enables API consumers to find APIs they can use in their applications by assembling a dashboard and detailed API documentation views by ingesting OpenAPI documentation from the source URLs., And allows for managing, creating, and consuming API Keys. link: https://docs.vmware.com/en/API-portal-for-VMware-Tanzu/index.html v3_1: false v3: true v2: true sourceIssueMetadata: issueNumber: 71 author: Albertoimpl createdAt: '2023-03-30T09:37:04Z' updatedAt: '2023-03-30T09:50:37Z' url: https://github.com/OAI/Tooling/issues/71 status: closed id: 714890331a19472487abb54870be2f1d foundInMaster: true - source: openapi3 tags repository: https://github.com/open-banking/directdebits v3: true repositoryMetadata: base64Readme: >- IyBkaXJlY3REZWJpdHMKT3BlbiBCYW5raW5nIERpcmVjdCBEZWJpdHMgQVBJIGJ1aWx0IG9uIHRvcCBvZiBsaWdodC00ago= readmeEtag: '"95993e54d31406cb3edcd5c81eb1ee8c32345e50"' readmeLastModified: Thu, 25 Jul 2024 20:09:49 GMT repositoryId: 235861354 description: Open Banking Direct Debits API built on top of light-4j created: '2020-01-23T18:41:22Z' updated: '2026-02-03T01:08:51Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"c202674d9f5f359204f08734b37c24b7e406915d1c7139ef738c9051d52f3fa1"' repoLastModified: Tue, 03 Feb 2026 01:08:51 GMT foundInMaster: true category: Server id: a64f93855552ec18dea59e28ee58ac21 - source: openapi3 tags repository: https://github.com/open-banking/transactions v3: true repositoryMetadata: base64Readme: >- IyB0cmFuc2FjdGlvbnMKdHJhbnNhY3Rpb25zIEFQSSBidWlsdCBvbiB0b3Agb2YgbGlnaHQtNGoK readmeEtag: '"1cf0a7583f442416ac344feedfde71f94c3b30f4"' readmeLastModified: Thu, 25 Jul 2024 20:09:57 GMT repositoryId: 227717798 description: transactions API built on top of light-4j created: '2019-12-12T23:51:44Z' updated: '2026-02-03T01:08:45Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"05356ca5e0dee6a9a35b53d5eeadc2b5d4ed48ee72f6588311114d64208b7754"' repoLastModified: Tue, 03 Feb 2026 01:08:45 GMT foundInMaster: true category: - Documentation - Parsers id: 92307dc3ff29d63f54325e5d3f821ed6 - source: openapi3 tags repository: https://github.com/saurhkumar/sqlcrud v3: true id: 61d4138ad7d80a7177294995cf211eb4 repositoryMetadata: base64Readme: >- <!-- ABOUT THE PROJECT -->

<div id="top"></div>

## About The Project

A full fledge rest app powered by express. The idea of this project is to bring up your microservices fast, with inbuilt request validations, connection to the database
and fully working rest APIs.  
**Note** This project is generated through [generator-rest-swagger-express](https://www.npmjs.com/package/generator-rest-swagger-express)  
To see all the application features, see [feature](#service-features) section:  
<!-- GETTING STARTED -->

## Getting Started

This is an example of how you may give instructions on setting up your project locally.

### Prerequisites

- You need a MySQL up and running. To do that, use docker  
  You can run it as a [docker](https://hub.docker.com/_/mysql) or install it as a [standalone application](https://www.mysql.com/). Note you have to create two additional databases also to run this project. There is one docker compose file in test folder docker-compose.yml (`/test/docker-compose.yml`), you can use it to run and create the docker image with the required databases. If you want to use the cloud database, then change the setting in the default.json (`/config/default.json`)
  Note: In your local setup, by default service will connect to the test database. To see the database name and database authentication details see default.json file
- Node.js > v14 needed. You can get it [here](https://nodejs.org/en/download/)
- Better install mocha on global level `npm install --global mocha`

<p align="right">(<a href="#top">back to top</a>)</p>

### Installation

1. Go to your command-line interface and type `npm install`. ( here assumption is slq is available)

<p align="right">(<a href="#top">back to top</a>)</p>

<!-- USAGE EXAMPLES -->

## Usage

Type `npm start`. Few lines will comes up and then go to ["http://localhost:3000/user-service/docs"](http://localhost:3000/user-service/docs/#/). You should see swagger doc

<p align="right">(<a href="#top">back to top</a>)</p>
<!-- Explore -->
## Explore The Project

Once you generate this project using [generator](https://www.npmjs.com/package/generator-rest-swagger-express), you will see this kind of directory structure.  
The top-level working of this project is like

```
 |-app.js
 |-.eslintrc.js
 |-.gitignore
 |-.prettierrc.json
 |-logger.js
 |-package.json
 |-Readme.md
 |-api
 | |-controllers
 | | |-controller.js
 | | |-metricController.js
 | |-helpers
 | | |-shortId.js
 | |-services
 | | |-service.js
 | | |-syncService.js
 | |-swagger
 | | |-swagger.json
 |-config
 | |-custom-environment-variables.json
 | |-default.json
 | |-development.json
 | |-production.json
 |-test
 | |-test.js
```

Here is a detailed description of most of the files  

`app.js` This is a starting point of the application, initializing the validations and swagger UI, all of the REST API path (using `/api/swagger/swagger.json`)

`.eslintrc.js` This file contain all [linting](https://eslint.org/) rules for this project  

`.prettierrc.json` This file contain all [Prettier](https://prettier.io/) rules for this project

`logger.js` This is a wrapper around [Winston](https://www.npmjs.com/package/winston)  

`/api/controllers/*` This directory has all the controllers. These controllers are to format your request and response.  

The `metricController` is exposing the [Prometheus](https://prometheus.io/) data via `/metrics` end point. If you want to see some example charts, explore the `metric` directory at the root level

`/api/serivce/*` This directory has all the services. Every rest end point has its own serivce function that can be used to add business logic or to interact with external services.  

The sync service (`/api/services/syncService.js`) initilize the database connecton, apart from that, it can be used to start a background process.  

`/api/helpers/middlewares` This module is initializing Prometheus metrics right now, for every rest end point. In the same way, this middleware can be used for auth and other methods that need to be invoked before any API call.  

`/api/helpers/sqlFilter.pegjs` and `/api/helpers/sqlFilter.js` The pegjs file is the grammar that is empowering GET and DELETE APIs [filter](#filter) parameter. The `sqlFilter.js` is generated using this pegjs file, and this `sqlFilter.js` file converts the filter query into the database query.  
To generate or customize the filters see [Pegjs](https://pegjs.org/)  
To see what kind of queries can be used and how to use this see [filter](#filter) section  

`/api/helpers/queryHooksjs` see [filter](#filter) section.  

Reqest flow diagram  
![Alt text](./flowDiagram.drawio.svg)

<p align="right">(<a href="#top">back to top</a>)</p>

<!-- Service Features -->
## Service Features

This service has out of the box support [sorting](#sorting), [filter](#filter), [projection](#projection) and [pagination](#pagination), background tasks, [Metrics](#metrics)  
Assuming this service have `name, age, address, country` fields in the the schema. Then following features are available  

<!-- Managing configurations-->

## Managing configurations

All the service configurations are managed through [config](https://www.npmjs.com/package/config) npm package. For your local development, all the configurations are under `config/default.json`. You can override these in production or in dev environment using `config/production.json`or `config/development.json`. The current default database configuration looks like

```
...
  "Database": {
    "server": "localhost",  --- database server URL
    "name": "test",                         --- database name
    "user": "admin",
    "password": "admin",
    "logging": false
  },
...
```
<!-- Sorting -->
## Sorting

In the get request, documents can sorted `$sortBy` parameter. Documents can be sorted in both ascending and descending direction. Ex: `+age -name` to sort the documents by age in ascending order and name is descending order.  
To control what fields you can sort, go to `/api/helpers/queryHooks.js` and inside the `mapping` function modify the `sortFields` keys. Assuming, you want to enable `name` and `age` sorting, then the sortField in the mapping function will be like:

```
  ...
  function mapping() {
    return {
      sortFields: ['name', 'age'],
  ...
```

<p align="right">(<a href="#top">back to top</a>)</p>

<!-- Filter -->
## Filter

Documents can be filtered using the parameter `$filter` by writing SQL-like statements. For example, to get all documents where the age is in ['23', '45'] and address = 'address1', the parameter `$filter` should be like `age in ( '23', '45') and address = 'address1'`.  
Here one thing to note down is the quotes (`''`) around all the values, these quotes are always required, for every value. Right now `integer`, `string`, `boolean`, and `date` data types are supported. To add more data type support, see `/api/helpers/sqlFilter.pegjs` file. For more examples see filter test cases.  

To control what fields you can filter, go to `/api/helpers/queryHooks.js` and inside the `mapping` function modify the `queryFields` keys. Assuming, you want to enable `name` and `age` filter, then the queryFields in the mapping function will be like:

```
  ...
  function mapping() {
    ...
      queryFields: [
        { name: 'name', type: 'string' },
        { name: 'age', type: 'int' }
      ],
    ...
  ...
```

<p align="right">(<a href="#top">back to top</a>)</p>

<!-- Projection -->
## Projection

Projection is a way to select only the necessary data rather than selecting every column. Use the `$projection` parameter to define the projection fields. Assuming you want to get only `age` and `name` out of all the available columns in the database, then the `$projection` parameter will be like `age name`. For more examples see filter test cases.  

<p align="right">(<a href="#top">back to top</a>)</p>

<!-- Pagination -->
## Pagination

 By Default applications exposes, pagination parameters [$top](https://github.com/microsoft/api-guidelines/blob/vNext/Guidelines.md#982-client-driven-paging) and [$skip](https://github.com/microsoft/api-guidelines/blob/vNext/Guidelines.md#982-client-driven-paging).

 <p align="right">(<a href="#top">back to top</a>)</p>

<!-- Pagination -->
## Swagger UI

 To visualize and interact with the API’s resources go to ["http://localhost:3000/user-service/docs"](http://localhost:3000/user-service/docs/#/)

 <p align="right">(<a href="#top">back to top</a>)</p>

<!-- Request and response validation -->
## Request and response validation

Define your request body or request parameter in the swagger file (`api/swagger/swagger.json`). To know more about the definitions, see [Swagger Documentation](https://swagger.io/specification/). According to the definitions defined in the swagger, all the validations will be performed.

 <p align="right">(<a href="#top">back to top</a>)</p>

<!-- Application Metrics -->
## Metrics

All the application metrics like CPU usages, event lag delay, TPM, and p95 are available.

The application `/metrics` end point gives the [prometheus](https://prometheus.io/) compatible data. If you want to see what it will look likes, just inside the `metric` folder and docker-compose-up command. It will run Prometheus on the 9090 port and Grafann on the 9000 port. Go to Grafana dashboard. Go to grafana dashboard [locally](http://localhost:9000/). You should see something like:
![Sample Grafan Dashboard](metrics/Capture.PNG "Sample Grafan Dashboard")

<p align="right">(<a href="#top">back to top</a>)</p>

<!-- CONTRIBUTING -->
## Contributing

Contributions are what make the open source community such an amazing place to learn, inspire, and create. Any contributions you make are **greatly appreciated**.

If you have a suggestion that would make this better, please fork the repo and create a pull request. You can also simply open an issue with the tag "enhancement".
Don't forget to give the project a star! Thanks again!

1. Fork the Project
2. Create your Feature Branch (`git checkout -b feature/AmazingFeature`)
3. Commit your Changes (`git commit -m 'Add some AmazingFeature'`)
4. Push to the Branch (`git push origin feature/AmazingFeature`)
5. Open a Pull Request

<p align="right">(<a href="#top">back to top</a>)</p>
 readmeEtag: '"dce681d1136e57d85d0998f7aa3c2a4c76ddf72d"' readmeLastModified: Wed, 24 Apr 2024 00:33:47 GMT repositoryId: 442657761 description: >- A simple rest microservice with Open API spec 3.0 and SQL backend. Supports out of the box filtering, along with sorting and projection support created: '2021-12-29T04:18:26Z' updated: '2025-11-02T03:38:08Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: saurhkumar logo: https://avatars.githubusercontent.com/u/2967058?v=4 license: MIT repoEtag: '"c32b2b31cab2a0d03d9a88e4eef1555dadfbe0b3cc5cec1077a97b4709611300"' repoLastModified: Sun, 02 Nov 2025 03:38:08 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/saurbkumar/sqlcrud - source: openapi3 tags repository: https://github.com/foxfriends/openfetch v3: true repositoryMetadata: base64Readme: >- IyBPcGVuRmV0Y2gKClshW2NpXShodHRwczovL2dpdGh1Yi5jb20vZm94ZnJpZW5kcy9vcGVuZmV0Y2gvYWN0aW9ucy93b3JrZmxvd3MvY2kueWFtbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vZm94ZnJpZW5kcy9vcGVuZmV0Y2gvYWN0aW9ucy93b3JrZmxvd3MvY2kueWFtbCkKCkZldGNoLWxpa2UgT3BlbkFQSSBjbGllbnQgbGlicmFyeS4gU3VwcG9ydHMgW09wZW5BUEkgM10oaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vKSBvbmx5LgoKSW5zdGFsbCBmcm9tIE5QTTogYG5wbSBpbnN0YWxsIG9wZW5mZXRjaGAKCiMjIFVzYWdlCgpgYGBqcwppbXBvcnQgKiBhcyBvcGVuZmV0Y2ggZnJvbSAnb3BlbmZldGNoJzsKCi8vIEJ1aWxkIGFuIEFQSSBiYXNlZCBvbiB0aGUgc3BlYy4gU2hvd24gYmVsb3cgYXJlIHRoZSBkZWZhdWx0IG9wdGlvbnM6CmNvbnN0IGFwaSA9IG9wZW5mZXRjaC5jcmVhdGUoU1BFQywgewogIC8vIFRoZSBiYXNlIFVSTCBmcm9tIHdoaWNoIHRvIG1ha2UgcmVxdWVzdHMuCiAgdXJsLAogIC8vIEVuYWJsZSBsb2dnaW5nLiBWYWxpZGF0aW9uIGVycm9ycyB3aWxsIGJlIGxvZ2dlZCB0byB0aGUgY29uc29sZS4KICBsb2dnaW5nOiBmYWxzZSwKICAvLyBUaGUgY29uc29sZSwgcHJvdmlkaW5nIHRoZSBsb2cgbWV0aG9kcyAob25seSBgd2FybmAgaXMgdXNlZCkuCiAgY29uc29sZTogd2luZG93LmNvbnNvbGUsCiAgLy8gVGhlIGRlZmF1bHQgaW1wbGVtZW50YXRpb24gb2YgZmV0Y2ggdG8gdXNlIGZvciBtYWtpbmcgcmVxdWVzdHMuCiAgZmV0Y2g6IHdpbmRvdy5mZXRjaCwKfSk7CgovLyBJZiB5b3VyIHNwZWMgaXMgaG9zdGVkLCBpdCBjYW4gYmUgcmV0cmlldmVkIGF1dG9tYXRpY2FsbHkuCmNvbnN0IGFwaSA9IGF3YWl0IG9wZW5mZXRjaC5ob3N0ZWQoJ2h0dHA6Ly9leGFtcGxlLmNvbScsIHsKICAvLyAuLi4gc2FtZSBvcHRzIGFzIGFib3ZlCn0pOwoKLy8gQ3JlYXRlIGFuIGludm9jYXRpb24gY29udGV4dCB3aXRoIGNyZWRlbnRpYWxzIGFuZCBzdWNoLiBUaGUga2V5cyBvZiB0aGUgY3JlZGVudGlhbHMKLy8gb2JqZWN0IGFyZSBuYW1lcyBvZiBzZWN1cml0eSBzY2hlbWVzLCBhbmQgdGhlIHZhbHVlcyBhcmUgdGhlaXIgdmFsdWVzLi4uCi8vICogICBIVFRQIEJhc2ljIEF1dGggZXhwZWN0cyB0aGUgdmFsdWUgdG8gYmUgYHsgdXNlciwgcGFzcyB9YAovLyAqICAgSFRUUCBCZWFyZXIgQXV0aCBleHBlY3RzIHRoZSB2YWx1ZSB0byBiZSBqdXN0IHRoZSB0b2tlbiAoaS5lLiBub3QgaW5jbHVkaW5nIHRoZSAiQmVhcmVyIiBwcmVmaXgpCi8vICogICBPdGhlciBIVFRQIGF1dGggZXhwZWN0cyB0aGUgZnVsbCBoZWFkZXIgdmFsdWUgKGkuZS4gaW5jbHVkaW5nIHRoZSBzY2hlbWUgbmFtZSkKLy8gKiAgIE9BdXRoMiB3aWxsIHBhc3MgdGhlIHRva2VuIHZpYSBBdXRvcml6YXRpb24gaGVhZGVyCmNvbnN0IGludm9rZSA9IG9wZW5mZXRjaC5jbGllbnQoewogIC8vIE92ZXJyaWRlIHRoZSBiYXNlIFVSTCBmcm9tIHdoaWNoIHRvIG1ha2UgcmVxdWVzdHMKICB1cmwsCiAgLy8gT3ZlcnJpZGUgdGhlIGltcGxlbWVudGF0aW9uIG9mIGBmZXRjaGAgYWdhaW4uIFRoaXMgdmFsdWUgdGFrZXMgcHJlY2VkZW5jZSBvdmVyIHRoZSBvbmUKICAvLyBwYXNzZWQgdG8gYGNyZWF0ZWAsIGlmIGJvdGggd2VyZSBwcm92aWRlZC4KICBmZXRjaCwKICAvLyBUaGUgY3JlZGVudGlhbHMgdG8gdXNlIHRvIHNhdGlzZnkgc2VjdXJpdHkgcmVxdWlyZW1lbnRzCiAgY3JlZGVudGlhbHM6IHt9LAp9KTsKCi8vIEludm9rZSBhbiBvcGVyYXRpb24gYnkgYG9wZXJhdGlvbklkYCAoaGVyZTogYGdldFVzZXJgKToKLy8gKiAgIFBhcmFtZXRlciBhcmUgc3VwcGxpZWQgYnkgbmFtZQovLyAqICAgT3B0aW9ucyBhcmUgdGhlIHNhbWUgYXMgZmV0Y2gsIHdpdGggYSBmZXcgYmVpbmcgc3VwcGxpZWQgYXV0b21hdGljYWxseToKLy8gICAgICogICBgQ29udGVudC1UeXBlYCB3aWxsIGJlIGRldGVybWluZWQgYXV0b21hdGljYWxseSBpZiB0aGUgc3BlYyBvbmx5IGRlZmluZXMgb25lIHJlcXVlc3QgYm9keQovLyAgICAgICAgIHR5cGUsIG90aGVyd2lzZSBpdCBtdXN0IGJlIHN1cHBsaWVkIHZpYSBgaGVhZGVyc2AuCi8vICAgICAqICAgSWYgYENvbnRlbnQtVHlwZWAgaXMgSlNPTiwgYGJvZHlgIHdpbGwgYmUgcGFzc2VkIHRocm91Z2ggYEpTT04uc3RyaW5naWZ5YC4KLy8gICAgICAgICBObyBwcm9jZXNzaW5nIHdpbGwgYmUgZG9uZSB0byBhbnkgb3RoZXIgYm9kaWVzLgovLyAgICAgKiAgIFRoZSBgQXV0aG9yaXphdGlvbmAgaGVhZGVyIHdpbGwgYmUgc2V0IGF1dG9tYXRpY2FsbHkgYmFzZWQgb24gdGhlIHNlY3VyaXR5IHJlcXVpcmVtZW50cy4KLy8gICAgICogICBTZXQgdGhlIGBBY2NlcHRgIGhlYWRlciBtYW51YWxseSB0byBzcGVjaWZ5IHdoaWNoIHJlc3BvbnNlIGZvcm1hdCB0byByZWNlaXZlLgpjb25zdCByZXNwb25zZSA9IGF3YWl0IGludm9rZShhcGkuZ2V0VXNlcih7IGlkOiAnZm94ZnJpZW5kcycgfSwgeyBoZWFkZXJzLCBib2R5IH0pKTsKCi8vIFRoZSByZXNwb25zZSBpcyB3aGF0ZXZlciBpcyByZXR1cm5lZCBieSB0aGUgcHJvdmlkZWQgaW1wbGVtZW50YXRpb24gb2YgYGZldGNoYC4gUmVmZXIgdG8gdGhlCi8vIHJlbGV2YW50IGRvY3VtZW50YXRpb24gb24gaG93IHRvIGhhbmRsZSB0aGF0IHJlc3BvbnNlLiBJbiBwYXJ0aWN1bGFyOgovLyAqICAgVGhlIHJlc3BvbnNlIGJvZHkgaXMgbm90IGludGVycHJldGVkIGF0IGFsbCAoZS5nLiBKU09OIGlzIG5vdCBwYXJzZWQgYXV0b21hdGljYWxseSkKLy8gKiAgIFRoZSByZXNwb25zZSBzdGF0dXMgaXMgbm90IGludGVycHJldGVkIGF0IGFsbCAoZS5nLiA0WFgvNVhYIHJlcG9uc2VzIGRvIG5vdCB0aHJvdykKY29uc29sZS5hc3NlcnQocmVzcG9uc2UgaW5zdGFuY2VvZiBSZXNwb25zZSkKYGBgCgpQb2ludHMgdG8gbm90ZToKKiAgIFRoaXMgcGFja2FnZSBhc3N1bWVzIHlvdXIgT3BlbkFQSSBzcGVjIGlzIHZhbGlkL2NvcnJlY3QsIGFuZCB0aGF0IHlvdSBhcmUgKGZvciB0aGUgbW9zdCBwYXJ0KQogICAgY2FsbGluZyBpdCB3aXRoIHNlbnNpYmxlIHZhbHVlcy4gVW5kZWZpbmVkIGJlaGF2aW91ciB3aWxsIG9jY3VyIGlmIHlvdSBkZXZpYXRlIGZyb20gc3BlYy4KKiAgIFRoZSBgc2VydmVyc2AgZmllbGQgb2YgdGhlIHNwZWMgaXMgaWdub3JlZC4gUHJvdmlkZSBhIGNvcnJlY3QgYHVybGAgb24geW91ciBvd24uCiogICBUaGVyZSBpcyBjdXJyZW50bHkgbm8gc3VwcG9ydCBmb3IgYW55IGV4dGVuc2lvbnMsIGJ1dCB0byBiZSBhYmxlIHRvIGltcGxlbWVudCB0aG9zZSBhcyBwbHVnaW5zCiAgICBpcyBzb21ldGhpbmcgdGhhdCBpcyBiZWluZyBjb25zaWRlcmVkLgoKIyMgVGVzdGluZwoKU28gZmFyLi4uIHZlcnkgbGl0dGxlIHRlc3RpbmcgaGFzIGJlZW4gZG9uZS4gSnVzdCBhIGJpdCBvZiBtYW51YWwgc3R1ZmYuIFRydXN0IHRoaXMgcHJvamVjdCBhdCB5b3VyCm93biByaXNrIGZvciBub3csIHVudGlsIEkgZmVlbCBsaWtlIHdyaXRpbmcgYSBwcm9wZXIgdGVzdCBzdWl0ZS4KCiMjIENvbnRyaWJ1dGluZwoKQ29udHJpYnV0aW9ucyBhcmUgd2VsY29tZSEgUGxlYXNlIHNlbmQgYSBQUiBvciBjcmVhdGUgaXNzdWVzIGlmIHlvdSB3b3VsZCBsaWtlIHNvbWV0aGluZyBpbXByb3ZlZC4K readmeEtag: '"619eab1536312eb7b6cf628bcde5a720b3f9cc17"' readmeLastModified: Mon, 17 Jun 2024 13:19:34 GMT repositoryId: 349907093 description: Fetch-like OpenAPI client library created: '2021-03-21T05:20:39Z' updated: '2026-01-22T01:40:21Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: foxfriends logo: https://avatars.githubusercontent.com/u/6126521?v=4 repoEtag: '"6266170be892db9f3a7a702950dc93da29dd6e2c288c181ae1815145d0788a8d"' repoLastModified: Thu, 22 Jan 2026 01:40:21 GMT foundInMaster: true category: - Testing - Server Implementations - Parsers id: 029d9dfa2a96b566f6723dddab2c917b - source: openapi3 tags repository: https://github.com/open-banking/parties v3: true repositoryMetadata: base64Readme: IyBwYXJ0aWVzCnBhcnRpZXMgQVBJIGJ1aWx0IG9uIHRvcCBvZiBsaWdodC00ago= readmeEtag: '"4af5d3ee73f4ea1ff7562baf4d82d3f75ab83666"' readmeLastModified: Thu, 25 Jul 2024 20:09:47 GMT repositoryId: 227717632 description: parties API built on top of light-4j created: '2019-12-12T23:50:14Z' updated: '2026-02-03T01:08:54Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: open-banking logo: https://avatars.githubusercontent.com/u/38990562?v=4 license: Apache-2.0 repoEtag: '"585e964547740546231b5faa7ed23cfcbb0b0d0087184b477fc5d0afd3041afc"' repoLastModified: Tue, 03 Feb 2026 01:08:54 GMT foundInMaster: true category: - Documentation - Server - Parsers id: 6f9a462dcee9c3601ba0dfb6aec30ba1 - source: openapi3 tags repository: https://github.com/istandaarden/iwlz-generiek v3: true repositoryMetadata: base64Readme: >- IyBpV2x6LWdlbmVyaWVrCktvcHBlbHZsYWsgc3BlY2lmaWNhdGllcyBtZXQgbmV0d2VyayBicmVkZSBmdW5jdGlvbmFsaXRlaXQuCgoKKiBLb3BwZWx2bGFrIE5ldHdlcmtwdW50CiAgICAqIFZlcnplbmRlbiB2YW4gbm90aWZpY2F0aWUKICAgICogVmVyemVuZGVuIHZhbiAoZm91dCkgbWVsZGluZwoKCiMjIFZlcnNpZXMKCnxWZXJzaWUgfCBTdGF0dXMgfCAgVG9lbGljaHRpbmcgfAp8Oi0tIHw6LS0gfCA6LS0gfAp8IHZlcnNpZSB2MS4wIC0gMTItMDQtMjAyNCAgfCAgZWVyc3RlIHZlcnNpZSB2b29yIGltcGxlbWVudGF0aWUgPGJyPiAtIFtSRkMwMDA4IC0gTm90aWZpY2F0aWVzXShodHRwczovL2dpdGh1Yi5jb20vaVN0YW5kYWFyZGVuL2lXbHotUkZDL2Jsb2IvbWFpbi9SRkMvUkZDMDAwOCUyMC0lMjBOb3RpZmljYXRpZXMubWQpICA8YnI+IC0gW1JGQzAwMTggLSBGb3V0bWVsZGluZ2VuXShodHRwczovL2dpdGh1Yi5jb20vaVN0YW5kYWFyZGVuL2lXbHotUkZDL2Jsb2IvbWFpbi9SRkMvUkZDMDAxOCUyMC0lMjBNZWxkZW4lMjB2YW4lMjBmb3V0ZW4lMjBpbiUyMGdlZ2V2ZW5zJTIwdm9sZ2VucyUyMGlTdGFuZGFhcmQlMjBpV2x6Lm1kKSAgfCBbdjEuMC4wXShodHRwczovL2dpdGh1Yi5jb20vaVN0YW5kYWFyZGVuL2lXbHotZ2VuZXJpZWsvcmVsZWFzZXMvdGFnL2lXbHotZ2VuLXYxLjAuMCkoW21hc3Rlcl0oaHR0cHM6Ly9naXRodWIuY29tL2lTdGFuZGFhcmRlbi9pV2x6LWdlbmVyaWVrKSkgfAoKQHRvZG8KLSBbIF0gQ2hhbmdlbG9nIGJpandlcmtlbgoKICAKCiMjIERvY3VtZW50YXRpZQotIERvY3VtZW50YXRpZSBvdmVyIE5vdGlmaWNhdGllczogW1JGQzAwMDhdKGh0dHBzOi8vZ2l0aHViLmNvbS9pU3RhbmRhYXJkZW4vaVdsei1SRkMvYmxvYi9tYWluL1JGQy9SRkMwMDA4JTIwLSUyME5vdGlmaWNhdGllcy5tZCkKLSBEb2N1bWVudGF0aWUgb3ZlciBNZWxkZW4gdmFuIGZvdXRlbiBpbiBnZWdldmVucyB2b2xnZW5zIGlTdGFuZGFhcmQgaVdsejogW1JGQzAwMThdKGh0dHBzOi8vZ2l0aHViLmNvbS9pU3RhbmRhYXJkZW4vaVdsei1SRkMvYmxvYi9tYWluL1JGQy9SRkMwMDE4JTIwLSUyME1lbGRlbiUyMHZhbiUyMGZvdXRlbiUyMGluJTIwZ2VnZXZlbnMlMjB2b2xnZW5zJTIwaVN0YW5kYWFyZCUyMGlXbHoubWQpCgojIyBCcm9ubmVuCiogQWN0aWVwcm9ncmFtbWEgaVdsejogdmFuIGtldGVuIG5hYXIgbmV0d2VyazogW2xlZXMgaGllciBtZWVyIG92ZXIgaGV0IEFjdGllcHJvZ3JhbW1hIGlXbHpdKGh0dHBzOi8vd3d3LmlzdGFuZGFhcmRlbi5ubC9hY3RpZXByb2dyYW1tYS1pd2x6ICJBY3RpZXByb2dyYW1tYSBpV2x6IikKKiBJbmZvcm1hdGllbW9kZWwgaVN0YW5kYWFyZGVuIDogW0luZm9ybWF0aWVtb2RlbGxlbl0oaHR0cHM6Ly9pbmZvcm1hdGllbW9kZWwuaXN0YW5kYWFyZGVuLm5sLykKKiBQb3J0YWFsIHZvb3IgaVN0YW5kYWFyZGVuIGluIGRlIFpvcmcgZW4gT25kZXJzdGV1bmluZzogW2hvbWVwYWdpbmEgaVN0YW5kYWFyZGVuXShodHRwczovL3d3dy5pc3RhbmRhYXJkZW4ubmwpCgojIyBDb250YWN0cGVyc29uZW46CiogSGlsa28gSmFjb2JzZSAgLSBbQGhpbGtvamFjb2JzZV0oaHR0cHM6Ly9naXRodWIuY29tL0hpbGtvSmFjb2JzZSkKKiBSZW1vIHZhbiBSZXN0ICAtIFtAcnZhbnJlc3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9ydmFucmVzdCkKKiBEZW5uaXMgZGUgR291dyAtIFtAZGVubmlzZGVnb3V3XShodHRwczovL2dpdGh1Yi5jb20vZGdvdXcpCgojIyBMaWNlbnRpZQpDb3B5cmlnaHQgJmNvcHk7IGlTdGFuZGFhcmRlbiAyMDIxCg== readmeEtag: '"5e3e63d3617920ce0a1e02b21caa6ec4aaa56a15"' readmeLastModified: Fri, 12 Apr 2024 13:38:32 GMT repositoryId: 331606386 description: Koppelvlak specificaties met netwerk brede functionaliteit created: '2021-01-21T11:27:03Z' updated: '2026-01-08T15:24:01Z' language: null archived: false stars: 3 watchers: 4 forks: 3 owner: iStandaarden logo: https://avatars.githubusercontent.com/u/54351663?v=4 license: GPL-3.0 repoEtag: '"9ddcdd354844d922a84e681e210fe234a770c52724f7f9f1647d40004c07f726"' repoLastModified: Thu, 08 Jan 2026 15:24:01 GMT foundInMaster: true category: Low-level Tooling id: 64b110c525b84712e1e71d3932e88126 - source: openapi3 tags repository: https://github.com/mermade/semoasa-tools v3: true repositoryMetadata: base64Readme: >- IyBzZW1vYXNhLXRvb2xzCgpUb29scyBmb3IgbWFuaXB1bGF0aW5nIFtTZW1vYXNhXShodHRwczovL2dpdGh1Yi5jb20vUmVwcmVaZW4vU2Vtb2FzYSkgZG9jdW1lbnRzLgoKIyMgc2Vtb2FzYS1jb21iaW5lCgpDb21iaW5lIG11bHRpcGxlIFNlbW9hc2EgZG9jdW1lbnRzIGludG8gb25lLgoKIyMjIFVzZS1jYXNlCgpCdW5kbGluZyBtdWx0aXBsZSBTZW1vYXNhIGRvY3VtZW50cyBpbnRvIGEgY29tYmluZWQgZG9jdW1lbnQgZm9yIHVzZSBpbiBTZW1vYXNhLQpjb21wbGlhbnQgYXBwbGljYXRpb25zLgoKIyMjIFVzYWdlOgoKYGBgCnNlbW9hc2EgW2ZpbGUuLi5dID4gb3V0cHV0LnlhbWwKYGBgCgojIyMgTm90ZXMKClRoZSBvdXRwdXQgU2Vtb2FzYSB2ZXJzaW9uIG51bWJlciAoYG9wZW5hcGlFeHRlbnNpb25Gb3JtYXRgKSB3aWxsIGJlIHNldCB0byB0aGUgaGlnaGVzdCB2ZXJzaW9uIHNlZW4gaW4gdGhlIGlucHV0IGZpbGVzLCBhY2NvcmRpbmcKdG8gW3NlbXZlcl0oaHR0cDovL3NlbXZlci5vcmcvKS4gTm8gYXR0ZW1wdCBpcyAqY3VycmVudGx5KiBtYWRlIHRvIHZhbGlkYXRlIG9yIHVncmFkZSBlYXJsaWVyIHZlcnNpb25zIHRvIHRoZSBsYXRlc3QgdmVyc2lvbi4KCkNsYXNoaW5nIGRlZmluaXRpb25zIGluIGxhdGVyIGlucHV0IGZpbGVzIHdpbGwgYmUgKiptZXJnZWQqKiB3aXRoIHRob3NlIGluIGVhcmxpZXIgZmlsZXMuIFByZWNlZGVuY2UgaXMgaW1wb3J0YW50LgoKIyMgc2Vtb2FzYS1zcGxpdAoKIyMjIFVzZSBjYXNlCgpTcGxpdHRpbmcgYSBidW5kbGVkIFNlbW9hc2EgZG9jdW1lbnQgYmFjayBpbnRvIGl0cyBjb21wb25lbnQgcGFydHMsIGJhc2VkIG9uIG5hbWVzcGFjZS4KCiMjIyBVc2FnZQoKYHNlbW9hc2Etc3BsaXQgW2ZpbGUuLi5dYAoKIyMjIE5vdGVzCgpOYW1lc3BhY2VzIGV4aXN0aW5nIGluIG11bHRpcGxlIGlucHV0IGZpbGVzIHdpbGwgb3ZlcndyaXRlIHRoZSBzYW1lIG91dHB1dCBkb2N1bWVudChzKS4KCiMjIHNlbW9hc2EtdmFsaWRhdGUKCipDb21pbmcgc29vbioKCiMjIHNlbW9hc2EtdXBncmFkZQoKKkNvbWluZyBzb29uKgoK readmeEtag: '"62a641d925fbf1f2d7048300927b2e19ca490553"' readmeLastModified: Mon, 26 Jul 2021 09:12:13 GMT repositoryId: 107004998 description: Tools to manipulate Semoasa documents created: '2017-10-15T11:08:27Z' updated: '2021-07-26T09:12:21Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: Mermade logo: https://avatars.githubusercontent.com/u/15950345?v=4 license: BSD-3-Clause repoEtag: '"d7c99a0815e40fe6fb308b2c8834377d3e44378418789efb1840a1814edc1ab4"' repoLastModified: Mon, 26 Jul 2021 09:12:21 GMT foundInMaster: true category: Parsers id: feef30c4b998bddca2a34ab95d12e192 - source: openapi3 tags repository: https://github.com/unitedeffects/ue-civic-api_srvc v3: true repositoryMetadata: base64Readme: >- IyBVRS1DaXZpYy1BUElfU3J2YwoKQSB3cmFwcGVyIHdpdGggY2FjaGluZyBmb3IgdGhlIGdvb2dsZWFwaSBjaXZpYyBzZXJ2aWNlIHRvIGFsbG93IHlvdSB0byByZXNwb25kIHF1aWNrbHkgYW5kIGFsc28gdG8gZW5mb3JjZSBodHRwcyBkZXNwaXRlIHRoZSBmYWN0IHRoYXQgc29tZSBvZiB0aGUgY2l2aWMgYXBpIGNvbnRlbnQgcHJvdmlkZWQgYnkgZ29vZ2xlIGlzIG9ubHkgaHR0cC4gSSBkbyBtb2RpZnkgdGhlIHJldHVybiBzbGlnaHRseSB0byBtYWtlIGl0IGVhc2llciB0byByZWZlcmVuY2Ugb2ZmaWNpYWxzIGFuZCB0aXRsZXMgdG9nZXRoZXIuCgojIyBEb2N1bWVudGF0aW9uCgpodHRwczovL2NpdmljcWEubWFpbG15dm9pY2UuY29tCgojIyBMb2NhbCBEZXYKCiogY2xvbmUgdGhpcyByZXBvCiogeWFybgoqIGNwIHNyYy9jb25maWdfY2hhbmdlbWUuanMgc3JjL2NvbmZpZy5qcwoqIG1vZGlmeSBjb25maWcuanMgd2l0aCBhcHByb3ByaWF0ZSB2YWx1ZXMKKiB5YXJuIHJ1biBkZXYKCkFsdGVybmF0aXZlbHksIHlvdSBjYW4gYnVpbGQgd2l0aCBiYWJsZSBhbmQgcnVuIGZvcm0gZGlzdAoKKiB5YXJuIGJ1aWxkCiogeWFybiBydW4gZGlzdAoKQXZhaWxhYmxlIGF0IGh0dHA6Ly9sb2NhbGhvc3Q6NDA1MAoKIyMgTGFtYmRhCgpUaGlzIGlzIGFuIGV4cHJlc3Mgc2VydmljZSB3cmFwcGVkIGFzIGEgbGFtYmRhIGZ1bmN0aW9uLiBZb3UgY2FuIGZpbmQgYW4gZXhhbXBsZSBhbmQgZGV0YWlscyBoZXJlOiBodHRwczovL21lZGl1bS5jb20vQHRoZUJvRWZmZWN0L3BvcnRpbmctbm9kZS1leHByZXNzLWs4LXNlcnZpY2UtdG8tbGFtYmRhLXdpdGgtc2VydmVybGVzcy1hLWZldy1leHRyYS1sZXNzb25zLWxlYXJuZWQtYTlkZGVkM2U2ZDExCgoqIEVuc3VyZSB5b3UgaGF2ZSB0aGUgc2VydmVybGVzcy5jb20gZnJhbWV3b3JrIGluc3RhbGxlZAoqIEVuc3VyZSB5b3UgaGF2ZSBhbiBhd3MgYWNjb3VudCBhbmQgY29uZmlndXJlZCB3aXRoIHNlcnZlcmxlc3MKKiBjaGFuZ2UgdGhlIGRpcmVjdG9yeSAuZW52X2NoYW5nZW1lIHRvIC5lbnYKKiBjaGFuZ2UgdGhlIGpzb24gZmlsZSB2YWx1ZXMgaW4gLmVudi9lbnYucWEuanNvbiB0byBtYXRjaCB5b3VyIGNvbmZpZy5qcyBmaWxlCiogaW4gYXdzLCBtYWtlIHN1cmUgeW91J3ZlIGNyZWF0ZWQgYW4gU0xTIGNlcnRpZmljYXRlIGZvciB5b3VyIHJvdXRlNTMgZG9tYWluCiogU0xTX0VOVj1xYSBzbHMgY3JlYXRlX2RvbWFpbiAoaWYgdGhpcyBpcyB0aGUgZmlyc3QgdGltZSkKKiBTTFNfRU5WPXFhIHlhcm4gZGVwbG95CgpJdCBtYXkgdGFrZSB1cCB0byA0MCBtaW51dGVzIGZvciB0aGUgZG9tYWluIHlvdSBjcmVhdGUgdG8gYmUgYXZhaWxhYmxlLgoKIyMgRG9ja2VyCgpUaGlzIHNlcnZpY2UgY2FuIGFsc28gYmUgdXNlZCBhcyBhIGRvY2tlciBjb250YWluZXIuIERvY2tlcmZpbGUgcHJvdmlkZWQuCgoqIGRvY2tlciBidWlsZCAtdCB5b3UvY2l2aWMgLgoqIHNldCBhbGwgZW52IHBhcmFtZXRlcnMgZnJvbSBjb25maWcuanMgb3IganVzdCBjcmVhdGUgYSBkb2NrZXItY29tcG9zZSBmaWxlIChub3QgcHJvdmlkZWQpCiogaWYgeW91IG1hbnVhbGx5IHNldCBwYXJhbWV0ZXJzOiBkb2NrZXIgcnVuIC1wIDQwNTA6NDA1MCB5b3UvY2l2aWMKKiBvdGhlcndpc2UsIGlmIHlvdSBjcmVhdGVkIHRoZSBjb21wb3NlIGZpbGU6IGRvY2tlci1jb21wb3NlIHVwCgpBdmFpbGFibGUgYXQgaHR0cDovL2xvY2FsaG9zdDo0MDUwCgojIyBUb2RvCgoqIDxzdHJpa2U+dXBkYXRlIGNvbmZpZ3MKKiBnZXQgbG9ncyB3aXRoIHN3YWdnZXIgcnVubmluZwoqIGNvbnZlcnQgY29kZSBiYXNlIGFuZCB1cGRhdGUgc3dhZ2dlcgoqIHVwZGF0ZSByb2xlIGJhc2VkIGFjY2VzcwoqIHVwZGF0ZSBhdXRoIHJlcXVpcmVtZW50cyBmb3IgYW5vbiB1c2VycwoqIGltcGxlbWVudCBhbmQgdGVzdAoqIGRlcGxveSB0byBRQQoqIG1lcmdlIHYyCiogZml4IFJFQURNRTwvc3RyaWtlPgoqIHB1c2ggdG8gcHJvZA== readmeEtag: '"ef9e904fa781c96d0454a810a05bad13e90b3b1e"' readmeLastModified: Mon, 10 May 2021 16:35:45 GMT repositoryId: 85435430 description: >- A wrapper with caching for the googleapi civic service to allow you to respond quickly and also to enforce https despite the fact that some of the civic api content provided by google is only http. created: '2017-03-18T22:14:57Z' updated: '2026-01-04T18:09:11Z' language: JavaScript archived: true stars: 0 watchers: 1 forks: 0 owner: UnitedEffects logo: https://avatars.githubusercontent.com/u/15662367?v=4 license: MIT repoEtag: '"375688151c51c17954294816ec3939cc64e9217e93139274f8a1b9f84e805f1e"' repoLastModified: Sun, 04 Jan 2026 18:09:11 GMT foundInMaster: true category: Server Implementations id: 5b1331246b87de0b0f8db0f6ad8139b8 - source: openapi3 tags repository: https://github.com/bradrmarshall/openlms v3: true repositoryMetadata: base64Readme: IyBPcGVuTE1TCkFuIE9wZW4gTGVhcm5pbmcgTWFuYWdlbWVudCBTeXN0ZW0K readmeEtag: '"437ad6f8aebcf946d3e625e3f6fa6885ff2bba6a"' readmeLastModified: Sun, 03 Feb 2019 21:26:23 GMT repositoryId: 168995367 description: An Open Learning Management System created: '2019-02-03T21:26:23Z' updated: '2019-02-03T21:26:57Z' language: null archived: false stars: 0 watchers: 0 forks: 0 owner: bradrmarshall logo: https://avatars.githubusercontent.com/u/47306005?v=4 license: Apache-2.0 repoEtag: '"b09c3f3137a3b6a04d5fda5b5f1da625fe65cf100d213a3182e3edff7473b66d"' repoLastModified: Sun, 03 Feb 2019 21:26:57 GMT foundInMaster: true category: - SDK - Server Implementations id: 275b9cfe1d57dc446054d25ed7baae74 - source: openapi3 tags repository: https://github.com/fibercrypto/fibercryptopy v3: true repositoryMetadata: repositoryId: 219072349 description: Python client library for FiberCrypto wallet API classes created: '2019-11-01T22:33:39Z' updated: '2019-11-01T22:55:42Z' language: null archived: false stars: 0 watchers: 2 forks: 1 owner: fibercrypto logo: https://avatars.githubusercontent.com/u/47309509?v=4 license: GPL-3.0 repoEtag: '"d9fe56c1569c7e647e8724f9509294c5df4bf261acdd1dda1f19c224ba7b5b50"' repoLastModified: Fri, 01 Nov 2019 22:55:42 GMT foundInMaster: true id: 75d4914cf4c403c9f6649e9ee4c3f3c8 - source: openapi3 tags repository: https://github.com/elipzis/aping v3: true repositoryMetadata: base64Readme: >- IyBhUGluZyBbIVtHaXRIdWIgbGljZW5zZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9naXRodWIvbGljZW5zZS9lbGlwemlzL2FQaW5nLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9lbGlwemlzL2FwaW5nL2Jsb2IvbWFzdGVyL0xJQ0VOU0UubWQpIFshW0dpdEh1YiAocHJlLSlyZWxlYXNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL3JlbGVhc2UtMC40LjAteWVsbG93LnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9lbGlwemlzL2FwaW5nL3JlbGVhc2VzL3RhZy8wLjQuMCkgWyFbRG9uYXRlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0RvbmF0ZS1QYXlQYWwtZ3JlZW4uc3ZnKV0oaHR0cHM6Ly93d3cucGF5cGFsLm1lL2VsaXB6aXMpCkEgc2ltcGxlIEFQSSBQaW5nIHRvb2wgdG8gZmVlZCBhIFN3YWdnZXIvT3BlbkFQSSAzLjAgZG9jdW1lbnQgZmlsZSwgY2FsbCBhbGwgcGF0aHMgYW5kIHJlY29yZCB0aW1lIGFuZCByZXNwb25zZXMsIGUuZy4gdG8gYmVuY2htYXJrIGFuIGVuZHBvaW50LgoKIyMgRmVhdHVyZXMKKiBSZWFkIFtTd2FnZ2VyL09wZW5BUEkgMy4wXVsyXSBhcGkgZGVmaW5pdGlvbiBmaWxlcyBhbmQgY2FsbCBhbGwgcGF0aHMKKiBQaW5nIGFsbCBwYXRocyBpbiBwYXJhbGxlbCB3b3JrZXJzIGFuZC9vciBvdmVyIHNldmVyYWwgbG9vcHMKKiBQYXNzIGN1c3RvbSBoZWFkZXJzLCBlLmcuIGBBdXRob3JpemF0aW9uYAoqIENyZWF0ZSByYW5kb20gYGludGVnZXJgIGFuZCBgc3RyaW5nYCBwYXJhbWV0ZXJzIGZvciB1cmxzCiogVHJhY2sgdGhlIHRpbWUgYW5kIHJlc3BvbnNlIGJvZHkgcGVyIHJlcXVlc3QKKiBPdXRwdXQgdGhlIHJlc3VsdHMgdG8gY29uc29sZSwgQ1NWLCBIVE1MLCBKU09OIG9yIE1hcmtkb3duCgojIyBMYXRlc3QgVmVyc2lvbnMKKiAwLjQuMAogICogQWRkZWQgcmVndWxhciBleHByZXNzaW9uIGZpbHRlciBvcHRpb24gZm9yIHBhdGhzCiAgKiBBZGRlZCByZXNwb25zZSB0aW1lIG1pbGxpc2Vjb25kcyB0aHJlc2hvbGQKKiAwLjMuMAogICogQWRkZWQgYXZlcmFnZSBtcyBjYWxjdWxhdGlvbiBmb3IgbXVsdGlwbGUgbG9vcHMKICAqIEFkZGVkIEJvb3RzdHJhcCBIVE1MIHRlbXBsYXRlIHdpdGggc29ydGFibGUgdGFibGUKICAqIEFkZGVkIEpTT04gb3V0cHV0CiogMC4yLjAKICAqIEFkZGVkIGFuIG9wdGlvbiB0byBjb25maWd1cmUgaW5jbHVkZWQgcXVlcnkgbWV0aG9kcwogIApEb3dubG9hZCB0aGUgbGF0ZXN0IFtyZWxlYXNlIGhlcmVdWzNdLgoKIyMgVXNhZ2UKRm9yIGEgcXVpY2sgc3RhcnQgZG93bmxvYWQgYSBbcmVsZWFzZV1bM10sIGNoYW5nZSBpbnRvIHRoZSBkaXJlY3RvcnkgYW5kIGV4ZWN1dGUgdGhlIGJpbmFyeSB3aXRoIHlvdXIgb3B0aW9ucywgZS5nLjoKYGBgc2hlbGwgc2NyaXB0Ci4vYXBpbmcgLWlucHV0PSJjYWxscy5qc29uIiAtaGVhZGVyPSd7XCJBdXRob3JpemF0aW9uXCI6IFwiQmVhcmVyIGV5WFlaXCJ9JyAtcmVzcG9uc2UgLWJhc2U9aHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaSAtb3V0PWh0bWwgLWw9NSAtdz01CmBgYAoKRXhhbXBsZSBwcm9ncmVzcyBvdXRwdXQ6CmBgYApQaW5naW5nICdSRVNUIEFQSSBEb2N1bWVudGF0aW9uIC0gQmFja2VuZCcKUGluZ2luZyAxMjIgcm91dGVzIChSb3VuZCAxKSAuLi4gZG9uZSEgWzEyMiBpbiAxMS41ODZzXQpQaW5naW5nIDEyMiByb3V0ZXMgKFJvdW5kIDIpIC4uLiBkb25lISBbMTIyIGluIDIzLjMxNXNdClBpbmdpbmcgMTIyIHJvdXRlcyAoUm91bmQgMykgLi4uIGRvbmUhIFsxMjIgaW4gMzcuMjYyc10KUGluZ2luZyAxMjIgcm91dGVzIChSb3VuZCA0KSAuLi4gZG9uZSEgWzEyMiBpbiA0OC42MTZzXQpQaW5naW5nIDEyMiByb3V0ZXMgKFJvdW5kIDUpIC4uLiAyNy45JSDilZHilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilojilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilpHilZEgWzM0IGluIDFtMS4zODZzXQpgYGAKCiMjIyBPcHRpb25zCmBgYHNoZWxsIHNjcmlwdApVc2FnZQogIC1pbnB1dCBzdHJpbmcKICAgICAgICAqVGhlIHBhdGgvdXJsIHRvIHRoZSBTd2FnZ2VyL09wZW5BUEkgMy4wIGlucHV0IHNvdXJjZQogIC1iYXNlIHN0cmluZwogICAgICAgIFRoZSBiYXNlIHVybCB0byBxdWVyeQogIC1oZWFkZXIgc3RyaW5nCiAgICAgICAgUGFzcyBhIGN1c3RvbSBoZWFkZXIgYXMgSlNPTiBzdHJpbmcsIGUuZy4gJ3tcIkF1dGhvcml6YXRpb25cIjogXCJCZWFyZXIgVE9LRU5cIn0nIChkZWZhdWx0ICJ7fSIpCiAgLWxvb3AgaW50CiAgICAgICAgSG93IG9mdGVuIHRvIGxvb3AgdGhyb3VnaCBhbGwgY2FsbHMgKGRlZmF1bHQgMSkKICAtb3V0IHN0cmluZwogICAgICAgIFRoZSBvdXRwdXQgZm9ybWF0LiBPcHRpb25zOiBjb25zb2xlLCBjc3YsIGh0bWwsIG1kLCBqc29uIChkZWZhdWx0ICJjb25zb2xlIikKICAtcmVzcG9uc2UKICAgICAgICBJbmNsdWRlIHRoZSByZXNwb25zZSBib2R5IGluIHRoZSBvdXRwdXQKICAtdGltZW91dCBpbnQKICAgICAgICBUaGUgdGltZW91dCBpbiBzZWNvbmRzIHBlciByZXF1ZXN0IChkZWZhdWx0IDUpCiAgLXdvcmtlciBpbnQKICAgICAgICBUaGUgYW1vdW50IG9mIHBhcmFsbGVsIHdvcmtlcnMgdG8gdXNlIChkZWZhdWx0IDEpCiAgLW1ldGhvZHMgc3RyaW5nCiAgICAgICAgQW4gYXJyYXkgb2YgcXVlcnkgbWV0aG9kcyB0byBpbmNsdWRlLCBlLmcuICdbXCJHRVRcIiwgXCJQT1NUXCJdJyAoZGVmYXVsdCAiW1wiR0VUXCIsXCJQT1NUXCJdIikKICAtdGhyZXNob2xkIGludAogICAgICAgIE9ubHkgY29sbGVjdCBwaW5ncyBhYm92ZSB0aGlzIHJlc3BvbnNlIHRocmVzaG9sZCBpbiBtaWxsaXNlY29uZHMgKGRlZmF1bHQgLTEpCiAgLWZpbHRlciBzdHJpbmcKICAgICAgICBBIHJlZ3VsYXIgZXhwcmVzc2lvbiB0byBmaWx0ZXIgcGF0aHMuIE9ubHkgbWF0Y2hlcyB3aWxsIGJlIHBpbmdlZCEKYGBgCgojIyMjIElucHV0ClJlZmVyZW5jZSBhIGZpbGUgaW5wdXQgc29tZXdoZXJlIHJlYWNoYWJsZSBieSB5b3VyIG1hY2hpbmUuIApSZWZlcmVuY2VzIGluIHRoZSBbT3BlbkFQSV1bMl0gc3BlY2lmaWNhdGlvbiBjYW4gYmUgcmVzb2x2ZWQgaWYgYWJzb2x1dGUgb3IgcmVsYXRpdmUgdG8gdGhlIG1haW4gZmlsZS4KCiMjIyMgQmFzZQpQYXNzIGEgYmFzZSB1cmwgc3VjaCBhcyBgaHR0cDovL2xvY2FsaG9zdDo4MDgwL2FwaWAuCklmIG5vbiBpcyBnaXZlbiB0aGUgYHNlcnZlcnNgIGFycmF5IG9mIHRoZSBbT3BlbkFQSV1bMl0gc3BlY2lmaWNhdGlvbiB3aWxsIGJlIHByZXNlbnRlZCB0byBwaWNrIGEgc2VydmVyIGZyb20uCgojIyMjIEhlYWRlcgpQYXNzIGN1c3RvbSBoZWFkZXJzIHRvIHNlbmQgd2l0aCBldmVyeSByZXF1ZXN0IGFzIGFuIGVzY2FwZWQgSlNPTiBzdHJpbmcgc3VjaCBhcyBgJ3tcIkF1dGhvcml6YXRpb25cIjogXCJCZWFyZXIgZXlYWVpcIn0nYC4KClRoZSBkZWZhdWx0IGhlYWRlcnMgYXJlCmBgYAoiQWNjZXB0IjogICAgICAgIiovKiIKIkNvbm5lY3Rpb24iOiAgICJLZWVwLUFsaXZlIgoiQ29udGVudC1UeXBlIjogImFwcGxpY2F0aW9uL2pzb24iCiJVc2VyLUFnZW50IjogICAiYVBpbmciCmBgYAoKWW91IGNhbiBvdmVycmlkZSB0aGVzZSBvcHRpb25zIGJ5IHBhc3NpbmcgdGhlIHNhbWUga2V5LgoKIyMjIyBXb3JrZXIKSG93IG1hbnkgcGFyYWxsZWwgcHJvY2Vzc2VzIHNob3VsZCBiZSBzcGF3bmVkIHRvIHF1ZXJ5IHlvdXIgZW5kcG9pbnRzLgoKKkVuc3VyZSB0aGF0IHlvdXIgZW5kcG9pbnQgY2FuIGhhbmRsZSBtdWx0aXBsZSByZXF1ZXN0cywgb3RoZXJ3aXNlIG11bHRpcGxlIHdvcmtlcnMgbWlnaHQgcnVuIGludG8gdGhlIHRpbWVvdXQuKgoKIyMjIyBPdXRwdXQKRGVmaW5lIGFuIG91dHB1dCBmb3JtYXQuIFRoZSBvdXRwdXQgaXMgd3JpdHRlbiB0byBhIGxvY2FsIGBhcGluZy5YWVpgIGZpbGUsIGRlcGVuZGluZyBvbiB5b3VyIGNob2ljZS4KClRoZSBvdXRwdXQgY29udGFpbnMgKGF0IG1vc3QpOgoqIFRoZSBwaW5nZWQgcGF0aAoqIFRoZSBlZmZlY3RpdmUgVVJMKnMqIChiYXNlICsgcGF0aCkKKiBUaGUgcXVlcnkgbWV0aG9kCiogVGhlIGF2ZXJhZ2UgbWlsbGlzZWNvbmRzCiogVGhlIHJlc3BvbnNlKnMqCgpTb21lIGRhdGEgaXMgb25seSBhdmFpbGFibGUgd2l0aCB0aGVpciBhY2NvcmRpbmcgZmxhZ3MsIGkuZS4gYGxvb3BgIGFuZCBgcmVzcG9uc2VgCgojIyMjIExvb3AKKklmIGBsb29wID4gMWAgaXMgbWl4ZWQgd2l0aCBgcmVzcG9uc2VgIGFsbCByZXNwb25zZXMgYXJlIGxvZ2dlZCwgaWYgdGhlIHBhdGggaGFzIHBhcmFtZXRlcnMhKgoKIyMgQnVpbGQKW0Rvd25sb2FkIGFuZCBpbnN0YWxsXVs1XSBHb2xhbmcgZm9yIHlvdXIgcGxhdGZvcm0uCgpDbG9uZSB0aGlzIHJlcG9zaXRvcnkgYW5kIGJ1aWxkIHlvdXIgb3duIHZlcnNpb246CmBgYHNoZWxsIHNjcmlwdApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL2VsaXBaaXMvYVBpbmcuZ2l0CmdvIGJ1aWxkIC1vIGFwaW5nIGdpdGh1Yi5jb20vZWxpcFppcy9hUGluZwpgYGAKCiMjIyBDb21wYXRpYmlsaXR5CmFQaW5nIGhhcyBiZWVuIHRlc3RlZCB1bmRlciB0aGUgZm9sbG93aW5nIGNvbmRpdGlvbnMKKiBXaW5kb3dzIDEwIFByb2Zlc3Npb25hbCAoNjQtYml0KQoKIyMgTWlzc2luZy9VcGNvbWluZyBGZWF0dXJlcwphUGluZyBpcyBub3QgZnVsbHktZmxlZGdlZCAoeWV0KS4gU29tZSBmdW5jdGlvbmFsaXR5IGlzIG1pc3NpbmcgYW5kIGVycm9ycyBtYXkgb2NjdXIuCgpLbm93biBpc3N1ZXMgYXJlOgoqIFBhdGhzIGhhdmluZyByZXF1ZXN0IGJvZGllcyBhcmUgbm90IHBpbmdlZAoqIFBhcmFtZXRlcnMgYmVzaWRlcyBgaW50ZWdlcmAgYW5kIGBzdHJpbmdgIGFyZSBub3QgcGluZ2VkIAoKIyMgTGljZW5zZSBhbmQgQ3JlZGl0cwphUGluZyBpcyByZWxlYXNlZCB1bmRlciB0aGUgTUlUIGxpY2Vuc2UgYnkgW2VsaXBaaXNdWzFdLgoKVGhpcyBwcm9ncmFtIHVzZXMgbXVsdGlwbGUgb3RoZXIgbGlicmFyaWVzLiBDcmVkaXRzIGFuZCB0aGFua3MgdG8gYWxsIHRoZSBkZXZlbG9wZXJzIHdvcmtpbmcgb24gdGhlc2UgZ3JlYXQgcHJvamVjdHM6CiogU3dhZ2dlci9PcGVuQVBJIDMuMCBwYXJzZXIgW2tpbi1vcGVuYXBpXVs2XQoqIFByZXR0eSBjb25zb2xlIHByaW50ZXIgW2dvLXByZXR0eV1bN10KCiMjIERpc2NsYWltZXIKVGhpcyBzb3VyY2UgYW5kIHRoZSB3aG9sZSBwYWNrYWdlIGNvbWVzIHdpdGhvdXQgYSB3YXJyYW50eS4gCkl0IG1heSBvciBtYXkgbm90IGhhcm0geW91ciBjb21wdXRlci4gUGxlYXNlIHVzZSB3aXRoIGNhcmUuIApBbnkgZGFtYWdlIGNhbm5vdCBiZSByZWxhdGVkIGJhY2sgdG8gdGhlIGF1dGhvci4gClRoZSBzb3VyY2UgaGFzIGJlZW4gdGVzdGVkIG9uIGEgdmlydHVhbCBlbnZpcm9ubWVudCBhbmQgc2Nhbm5lZCBmb3IgdmlydXNlcyBhbmQgaGFzIHBhc3NlZCBhbGwgdGVzdHMuCgogIFsxXTogaHR0cHM6Ly9lbGlwWmlzLmNvbQogIFsyXTogaHR0cHM6Ly9zd2FnZ2VyLmlvL3NwZWNpZmljYXRpb24vCiAgWzNdOiBodHRwczovL2dpdGh1Yi5jb20vZWxpcFppcy9hUGluZy9yZWxlYXNlcwogIFs0XTogaHR0cHM6Ly9naXRodWIuY29tL2VsaXBaaXMvYVBpbmcvd2lraS9WZXJzaW9uLUhpc3RvcnkKICBbNV06IGh0dHBzOi8vZ29sYW5nLm9yZy9kbC8KICBbNl06IGh0dHBzOi8vZ2l0aHViLmNvbS9nZXRraW4va2luLW9wZW5hcGkKICBbN106IGh0dHBzOi8vZ2l0aHViLmNvbS9qZWRpYjB0L2dvLXByZXR0eQo= readmeEtag: '"9aa8d3f94c0c9f487833ccf647394382fe18d40a"' readmeLastModified: Thu, 13 Aug 2020 14:50:01 GMT repositoryId: 281031032 description: >- 📲 A simple API Ping tool to feed a Swagger/OpenAPI 3.0 document file, call all paths and record time and responses created: '2020-07-20T06:17:44Z' updated: '2020-08-13T14:50:04Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: elipZis logo: https://avatars.githubusercontent.com/u/46478237?v=4 license: MIT repoEtag: '"bafc24c745576ea2033af0c7c69e486549dcc7c62712e688a17a837728a15046"' repoLastModified: Thu, 13 Aug 2020 14:50:04 GMT foundInMaster: true category: Server Implementations id: ad419c774f086dbba8ab197d314b40fe - source: openapi3 tags repository: https://github.com/ragnar-oock/setlist_v2_connexion v3: true repositoryMetadata: base64Readme: >- IyBzZXRsaXN0X3YyX2Nvbm5leGlvbgoKIyMgU2V0dXAKCi0gaW5zdGFsbCByZXF1aXJlbWVudDogIApgcGlwZW52IGluc3RhbGxgCi0gVXBkYXRlIGRhdGFiYXNlIGluZm9ybWF0aW9uIGluIGBzZXRpbmcucHlgCi0gUnVuIGBtYWluLnB5YCBvbmNlIHRoZW4gc2h1dCBpdCBkb3duLgotIFJ1biB0aGUgZm9sbG93aW5nIHF1ZXJpZXMgaW4gdGhlIGRhdGFiYXNlIHRvIHNldHVwIHRoZSBkaWZmZXJlbnRzIElOREVYIGFuZCBhdXRvZ2VuZXJhdGVkIGZpZWxkczogIApgYGBzcWwKLS0gYWRkIGZ0c19jb2wgYW5kIGdpc3QgaW5kZXgKQUxURVIgVEFCTEUgc29uZyBBREQgQ09MVU1OIGZ0c19jb2wgVGV4dAogICAgR0VORVJBVEVEIEFMV0FZUyBBUyAoY29hbGVzY2UobmFtZSwgJycpIHx8JyAnfHwgY29hbGVzY2UoYXJ0aXN0LCAnJykgfHwnICd8fCBjb2FsZXNjZShhbGJ1bSwgJycpKSBTVE9SRUQ7CkNPTU1FTlQgT04gQ09MVU1OIHNvbmcuZnRzX2NvbAogICAgSVMgJ2FnZ3JlZ2F0aW9uIG9mIG5hbWUsIGFydGlzdCBhbmQgYWxidW0gY29sdW1ucyB0byBiZSBpbmRleGVkIGZvciBmYXN0ZXIgdHJpZ3JhbSBzZWFyY2gnOwpDUkVBVEUgSU5ERVggT04gc29uZyBVU0lORyBHaVNUIChmdHNfY29sIGdpc3RfdHJnbV9vcHMpOwoKLS0gY3JlYXRlIGF1dG9pbmNyZW1lbnQgb24gc29uZy5pbmRleApBTFRFUiBUQUJMRSBwdWJsaWMuc29uZwogICAgQUxURVIgQ09MVU1OIGluZGV4IEFERCBHRU5FUkFURUQgQUxXQVlTIEFTIElERU5USVRZICggSU5DUkVNRU5UIDEgU1RBUlQgMCBNSU5WQUxVRSAwICk7YGBg readmeEtag: '"e8b48ce073ea7c86f5745068ad25d0046bf6086d"' readmeLastModified: Tue, 02 Mar 2021 15:23:45 GMT repositoryId: 321718670 description: API back-end for Ragnar-Oock/setlist created: '2020-12-15T16:04:36Z' updated: '2021-03-22T17:20:52Z' language: Python archived: false stars: 0 watchers: 1 forks: 0 owner: Ragnar-Oock logo: https://avatars.githubusercontent.com/u/38219333?v=4 license: MIT repoEtag: '"6f8a7144ac7bd6847fe5115a9761c6947b3cec13c92ef58cc0d592584430818b"' repoLastModified: Mon, 22 Mar 2021 17:20:52 GMT foundInMaster: true category: Server Implementations id: 6804b783d69b4dbf63629b5dfbb94ba4 - source: openapi3 tags repository: https://github.com/karthikeyan-ng/learn-open-api v3: true repositoryMetadata: base64Readme: >- IyBsZWFybi1vcGVuLWFwaQpUaGlzIHJlcG9zaXRvcnkgY29udGFpbnMgaG93IGRldmVsb3AgT3BlbkFQSSBzcGVjaWZpY2F0aW9uIGJhc2VkIFJFU1QgQVBJcy4K readmeEtag: '"e73a85d1889595141623015bf0f359ac4a10210b"' readmeLastModified: Tue, 11 Feb 2020 16:36:14 GMT repositoryId: 237742792 description: >- This repository contains how to develop OpenAPI specification based REST APIs. created: '2020-02-02T08:46:32Z' updated: '2020-04-10T11:46:13Z' language: null archived: false stars: 0 watchers: 1 forks: 1 owner: karthikeyan-ng logo: https://avatars.githubusercontent.com/u/39563373?v=4 repoEtag: '"ce3551b34a1edbe371254e0d9e3da781d921d01f6b2936316220cfe4bcc856e9"' repoLastModified: Fri, 10 Apr 2020 11:46:13 GMT foundInMaster: true category: - Server - Server Implementations id: ce2a86eb20bd1ae4de9b956c346fcecd - source: openapi3 tags repository: >- https://github.com/nicolaslopez82/java-springboot-docker-microservices-skeleton v3: true repositoryMetadata: base64Readme: >- IyBKYXZhIDggU3ByaW5nLUJvb3QgMiBNaWNyb3NlcnZpY2UgUHJvamVjdAoKVGhlIGdvYWxzIGhlcmUgYXJlIHRvIHVwZGF0ZSB0ZWNobm9sb2dpZXMgYW5kIG5ldyBjb25jZXB0cyBhYm91dCBtaWNyby1zZXJ2aWNlczoKCiMjIFRoZSBTd2FnZ2VyIFVJIFJFU1QgQVBJIERvY3VtZW50YXRpb24KCgpgPGxpbms+YCA6IDxodHRwczovL25pY29sYXNsb3BlejgyLmdpdGh1Yi5pby9qYXZhLXNwcmluZ2Jvb3QtZG9ja2VyLW1pY3Jvc2VydmljZXMtc2tlbGV0b24vPgoKIVtdKGh0dHBzOi8vZ2l0aHViLmNvbS9uaWNvbGFzbG9wZXo4Mi9qYXZhLXNwcmluZ2Jvb3QtZG9ja2VyLW1pY3Jvc2VydmljZXMtc2tlbGV0b24vYmxvYi9tYXN0ZXIvc3dhZ2dlci11aS5wbmcpCgojIyBUZWNobm9sb2dpZXMgYW5kIG5ldyBjb25jZXB0cyB1cGRhdGVkLgoKKiBKYXZhIDguCiogU3ByaW5nLUJvb3QgMi4KKiBTcHJpbmcgRGF0YSBKUEEuCiogU3ByaW5nIFNlY3VyaXR5LgoqIFNwcmluZyBSRVNULgoqIFNwcmluZyBNVkMuCiogTG9nZ2VyIHdpdGggTG9nQmFjayBBcHBlbmRlciBSb2xsaW5nRmlsZS4KKiBTd2FnZ2VyIFVJLgoqIE9wZW5BUEkgdjMuCiogSlNPTiBXZWIgVG9rZW4gKEpXVCkuCiogUGFzc3dvcmQgRW5jcnlwdGlvbi4KKiBNb2NraXRvIFRlc3RpbmcuCiogSlVuaXQgVGVzdGluZy4KKiBJbnRlZ3JhdGlvbiBUZXN0aW5nLgoqIEh5cGVybWVkaWEgRHJpdmVuIFJFU1RmdWwuCiogUGFnaW5nIGFuZCBTb3J0aW5nLgoqIFNlY3VyaW5nIEFQSXMuCiogQ29udHJvbGxpbmcgQVBJcyBleHBvc3VyZS4KKiBEb2NrZXIgSW1hZ2UuCiogRG9ja2VyIENvbnRhaW5lcnMuCiogRG9ja2VyIENvbnRhaW5lcnMgRGF0YWJhc2VzLgoqIERvY2tlciBNaWdyYXRpb24gRGF0YWJhc2Ugd2l0aCBGbHl3YXkuCiogRG9ja2VyIEh1Yi4K readmeEtag: '"22e2e1f5efee2ad4a8d24d058a6df68ae6053779"' readmeLastModified: Sun, 20 Dec 2020 04:41:41 GMT repositoryId: 278491784 description: >- Java Spring Boot Microservices Project. Spring Data JPA, Spring Security, Spring REST, and Docker. Please, see the README file for more details. created: '2020-07-09T23:23:37Z' updated: '2020-12-20T04:41:44Z' language: Java archived: false stars: 0 watchers: 0 forks: 0 owner: nicolaslopez82 logo: https://avatars.githubusercontent.com/u/6266503?v=4 repoEtag: '"5cd1ee82d372c918aa212b508ebc993060953b4a7cb5b0256015fd3dbf8f0a12"' repoLastModified: Sun, 20 Dec 2020 04:41:44 GMT foundInMaster: true category: - Low-level Tooling - Server Implementations id: 4a412b648722eee99f3b719824e04d0a - source: openapi3 tags repository: >- https://github.com/joergschultzelutter/openapi-to-robot-framework-datadriver-testgenerator v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIHRvIFJvYm90RnJhbWV3b3JrIERhdGFEcml2ZXIgdGVzdCBnZW5lcmF0b3IKClshW0xpY2Vuc2U6IEdQTCB2M10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9MaWNlbnNlLUdQTHYzLWJsdWUuc3ZnKV0oaHR0cHM6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9ncGwtMy4wKSBbIVtDb2RlIHN0eWxlOiBibGFja10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlJTIwc3R5bGUtYmxhY2stMDAwMDAwLnN2ZyldKGh0dHBzOi8vZ2l0aHViLmNvbS9wc2YvYmxhY2spIFshW0NvZGVRTF0oaHR0cHM6Ly9naXRodWIuY29tL2pvZXJnc2NodWx0emVsdXR0ZXIvb3BlbmFwaS10by1yb2JvdC1mcmFtZXdvcmstZGF0YWRyaXZlci10ZXN0Z2VuZXJhdG9yL2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVxbC55bWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2pvZXJnc2NodWx0emVsdXR0ZXIvb3BlbmFwaS10by1yb2JvdC1mcmFtZXdvcmstZGF0YWRyaXZlci10ZXN0Z2VuZXJhdG9yL2FjdGlvbnMvd29ya2Zsb3dzL2NvZGVxbC55bWwpCgpUaGlzIHByb2dyYW0gZ2VuZXJhdGVzIGEgZnVsbHktZmxlZGdlZCBFeGNlbC1iYXNlZCBSb2JvdCBGcmFtZXdvcmsgRGF0YWRyaXZlciB0ZXN0IGNhc2UgZnJvbSBhbiBPcGVuQVBJIGZpbGUuCgohW0RlbW9dKGRvY3MvaW1nL2RlbW8uanBnKQoKIyMgR2VuZXJhbCBmZWF0dXJlcwogIAotIFN1cHBvcnRzIE9wZW5BUEkgVjMgWUFNTCBhbmQgSlNPTiBpbnB1dCBmaWxlIGZvcm1hdHMuIFYyIGZpbGUgZm9ybWF0IHN1cHBvcnQgY2FuIGJlIGFjaGlldmVkIHRocm91Z2ggbGlicmFyeSByZWNvbmZpZ3VyYXRpb24uCi0gVGhlIHByb2dyYW0gd2lsbCByZWFkIGEgdmFsaWQgT3BlbkFQSSBmaWxlIGFuZCB0aGVuIHRyaWVzIHRvIGdlbmVyYXRlIHRoZSBmb2xsb3dpbmcgb3V0cHV0IGZpbGVzIGZvciB5b3U6CiAgLSBBIFJvYm90IEZyYW1ld29yayB0ZXN0IGZpbGUsIGNvbnRhaW5pbmcgYWxsIHRlc3RzIHRoYXQgYXJlIGFzc29jaWF0ZWQgd2l0aCB0aGlzIE9wZW5BUEkgZmlsZQogIC0gQSBnZW5lcmljIFJGICdpbmNsdWRlJyBmaWxlCiAgLSBBbiBFeGNlbCBmaWxlIHdoaWNoIHdpbGwgYmUgdXNlZCBieSB0aGUgUm9ib3QgRnJhbWV3b3JrIHRlc3QgYXMgaW5wdXQgZGF0YQogIC0gT3B0aW9uYWw6IGdlbmVyYXRlIEppcmEgLyBYUmF5IFRlc3QgLyBUZXN0IEV4Y2V1dGlvbiB0aWNrZXRzIGFuZCBhc3NvY2lhdGUgdGhlbSB3aXRoIHRoZSBSb2JvdCBGcmFtZXdvcmsgdGVzdAotIEFsbCBpbnB1dCBkYXRhIGlzIHRlbXBsYXRlLWJhc2VkLCBtZWFuaW5nIHRoYXQgeW91IGNhbiBhcHBseSBzaW1wbGUgbW9kaWZpY2F0aW9ucyB0byBKaXJhIC8gUm9ib3QgdGVtcGxhdGUgZmlsZXMgd2hpY2ggd2lsbCB0aGVuIHJlcHJlc2VudCB0aGUgZm91bmRhdGlvbiBmb3IgdGhlIGZ1dHVyZSBFeGNlbC9Sb2JvdCB0ZXN0IGZpbGUKLSBBbGwgcmVzdWx0aW5nIGZpbGVzIHdpbGwgbm8gbG9uZ2VyIGJlIGRlcGVuZGVudCBvbiB0aGUgaW5pdGlhbCBPcGVuQVBJIGlucHV0IGZpbGUuIFRoZSBpZGVhIGZvciB0aGlzIHByb2dyYW0gaXMgdGhhdCBpdCB3aWxsIGdlbmVyYXRlIHRoZSBtYWpvcml0eSBvZiByZXF1aXJlZCBjb2RlIGZvciB5b3UgYW5kIHRoYXQgeW91IGN1c3RvbWl6ZSB0aGF0IGNvZGUgbGF0ZXIgb24gZm9yIHlvdXIgbmVlZHMuCgojIyBFeGNlbCBmZWF0dXJlcwoKLSBGb3IgZWFjaCBBUEkgY2FsbCwgdGhlIHByb2dyYW0gd2lsbCB0cnkgdG8gZGV0ZXJtaW5lIHRoZSBzdXBwb3J0ZWQgb3B0aW9uYWwgYW5kIHJlcXVpcmVkIHBhcmFtZXRlcnMgZm9yIHRoZSByZXF1ZXN0IGJvZHk6CiAgLSBfX1JlZF9fIGZpZWxkcyBpbmRpY2F0ZSBhIG1hbmRhdG9yeSBwYXJhbWV0ZXIKICAtIF9fR3JlZW5fXyBmaWVsZHMgaW5kaWNhdGUgb3B0aW9uYWwgcGFyYW1ldGVycwogIC0gX19HcmV5X18gZmllbGRzIGluZGljYXRlIGZpZWxkcyB3aGljaCBhcmUgbm90IHN1cHBvcnRlZCBmb3IgdGhpcyBBUEkgY2FsbAogIC0gX19CZWlnZV9fIGZpZWxkcyBpbmRpY2F0ZSB0ZXN0IHBhcmFtZXRlcnMgc3VjaCBhcyB0ZXN0IG5hbWUsIHRhZ3MgZXRjCiAgLSBZb3UgY2FuIGVhc2lseSBjdXN0b21pemUgdGhlc2UgdmFsdWVzIGluIHRoZSBwcm9ncmFtLgotIElmIGEgcGFyYW1ldGVyIGhhcyBiZWVuIGFzc2lnbmVkIHdpdGggYW4gX2VudW1fIGxpc3Qgb2YgdmFsaWQgdmFsdWVzLCB0aGUgcHJvZ3JhbSB3aWxsIGFkZCB0aGVzZSB2YWx1ZXMgYXMgYSBwcmUtcG9wdWxhdGVkIGRyb3Bkb3duIHRvIHRoZSBFeGNlbCBsaXN0LiBVc2VycyBjYW4gc2VsZWN0IGEgdmFsdWUgZnJvbSB0aGF0IGxpc3QgKGJ1dCBhcmUgc3RpbGwgYWJsZSB0byBvdmVyd3JpdGUgdGhhdCB2YWx1ZSkKLSBCeSBkZWZhdWx0LCB0aGUgcHJvZ3JhbSB3aWxsIG9ubHkgaGlnaGxpZ2h0IHRoZSBvcHRpb25hbC9tYW5kYXRvcnkgcGFyYW1ldGVyIGZpZWxkcyAoYnV0IHdpbGwgbm90IHBvcHVsYXRlIHRoZW0gd2l0aCBhbnkgY29udGVudCkuIFlvdSBjYW4gb3ZlcnJpZGUgdGhpcyBzZXR0aW5nIGFuZCB0ZWxsIHRoZSBwcm9ncmFtIHRvIHByZS1wb3B1bGF0ZSB0aGUgcGFyYW1ldGVycyB3aXRoIGV4YW1wbGUgZGF0YSBzdHJhaWdodCBmcm9tIHlvdXIgT3BlbkFQSSBmaWxlLgotIEV4cGVjdGVkIEhUVFAgcmVzcG9uc2UgY29kZXMgYXJlIGV4dHJhY3RlZCBmcm9tIHRoZSBPcGVuQVBJIGZpbGUgYW5kIGFkZGVkIHRvIHRoZSBFeGNlbCBzaGVldCBhcyBkZWZhdWx0IHZhbHVlCi0gQWxsIE9wZW5BUEkgaW5wdXQgcGFyYW1ldGVycyBpbiB0aGUgRXhjZWwgZmlsZSBhcmUgYWxwaGFiZXRpY2FsbHkgc29ydGVkIGZvciBhIGJldHRlciB1c2VyIGNvbnZlbmllbmNlIHdoZW4gaXQgY29tZXMgdG8gcG9wdWxhdGluZyB0aGUgc2hlZXQgd2l0aCBkYXRhLgoKIyMgUm9ib3QgZmVhdHVyZXMKCi0gVGhlIHByb2dyYW0gd2lsbCB0cnkgdG8gY3JlYXRlIGEgZnVsbHktZmxlZGdlZCBSb2JvdCBGcmFtZXdvcmsgdGVzdCBjYXNlIGZvciB5b3UuCiAgLSBFYWNoIEFQSSBjYWxsIHdpbGwgYmUgdGFpbG9yZWQgdG8gdGhlIHJlc3BlY3RpdmUgQVBJIHNlcnZpY2UgbWV0aG9kLCBtZWFuaW5nIHRoYXQKICAgIC0gYSBKU09OIHJlcXVlc3QgYm9keSB3aWxsIG9ubHkgYmUgc2VudCB0byB0aGUgQVBJIGlmIHRoZSByZXNwZWN0aXZlIEFQSSBzZXJ2aWNlIG1ldGhvZCBzdXBwb3J0cyBpdAogICAgLSB0aGUgSlNPTiByZXF1ZXN0IGJvZHkgd2lsbCBvbmx5IGNvbnRhaW4gcGFyYW1ldGVycyB0aGF0IGJlbG9uZyB0byB0aGlzIHNwZWNpZmljIEFQSSBzZXJ2aWNlIG1ldGhvZAogICAgLSBJZiBhIHNlcnZpY2UgbWV0aG9kIHJlcXVpcmVzIHBhcmFtZXRlcnMgdG8gYmUgc2VudCBhcyBwYXJ0IG9mIHRoZSBBUEkncyBVUkwsIHRoZSBwcm9ncmFtIHdpbGwgZ2VuZXJhdGUgdGhlc2UgVVJMcyBmb3IgeW91LgotIFNvbWUgbWlub3IgYWRqdXN0bWVudHMgdG8gdGhlIFJvYm90IGZpbGUocykgbWF5IHN0aWxsIGJlIG5lY2Vzc2FyeSAoZS5nLiBpbml0aWFsIEFQSSBzZXR1cCwgbmVzdGVkIGRhdGEgc3RydWN0dXJlcykgLSBzZWUga25vd24gaXNzdWVzLgoKIyMgRG9jdW1lbnRhdGlvbgoKLSBbSW5zdGFsbGF0aW9uIGFuZCBmaXJzdCBwcm9ncmFtIHJ1bl0oZG9jcy9JTlNUQUxMQVRJT04ubWQpCi0gW0N1c3RvbWl6YXRpb24gYW5kIHByb2dyYW0gY29uZmlndXJhdGlvbl0oZG9jcy9DT05GSUdVUkFUSU9OLm1kKQotIFtDb21tYW5kIGxpbmUgcGFyYW1ldGVycyBhbmQga25vd24gaXNzdWVzXShkb2NzL1VTQUdFLm1kKQo= readmeEtag: '"a25f1e5eed12a0286b50a33926c63c3176576e3e"' readmeLastModified: Wed, 16 Mar 2022 18:28:55 GMT repositoryId: 429881762 description: >- Generates a fully-fledged Robot Framework Datadriver test and (optional) Jira/XRay test tickets from an OpenAPI V3 JSON/YAML schema file. created: '2021-11-19T17:28:35Z' updated: '2022-09-22T22:18:26Z' language: Python archived: false stars: 0 watchers: 3 forks: 1 owner: joergschultzelutter logo: https://avatars.githubusercontent.com/u/76180229?v=4 license: GPL-3.0 repoEtag: '"63732d4284d613a86a1bc982df578ac30f71eaf6fbd1c4769d6f06a8750343f6"' repoLastModified: Thu, 22 Sep 2022 22:18:26 GMT foundInMaster: true category: Testing id: b6921e1b63c62319faa55662e3947a27 - source: openapi3 tags repository: https://github.com/butley/plugin-resolver v3: true id: 06f4819aa00a24869163fcbaddf54ee1 repositoryMetadata: base64Readme: >- IyMgcGx1Z2luLXJlc29sdmVyCgpUaGUgUGx1Z2luUmVzb2x2ZXIgY2xhc3MgaXMgYSB1dGlsaXR5IGZvciByZXNvbHZpbmcgcGx1Z2lucyBpbiB0aGUgY29udGV4dCBvZiBhbiBBSS1iYXNlZCBjb252ZXJzYXRpb24uIEl0IGlzIHJlc3BvbnNpYmxlIGZvciBmaW5kaW5nIGEgc3VpdGFibGUgcGx1Z2luIGJhc2VkIG9uIHRoZSBnaXZlbiBjb250ZXh0IG1lc3NhZ2VzIGFuZCB1c2VyIG1lc3NhZ2UuIFRoZSBjdXJyZW50IGltcGxlbWVudGF0aW9uIG1ha2VzIHVzZSBvZiB0aGUgZ3B0LTMuNS10dXJiby0wMzAxIG1vZGVsIGFuZCBlbXBsb3lzIGN1c3RvbSBwcm9tcHRzIHRvIGV2YWx1YXRlIGRpZmZlcmVudCBPcGVuQVBJIFlBTUwgc3BlY2lmaWNhdGlvbnMuIEl0IHRoZW4gZXh0cmFjdHMgdGhlIHJlbGV2YW50IGluZm9ybWF0aW9uIGZyb20gdGhlIHBsdWdpbidzIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBhbmQgZ2VuZXJhdGVzIGEgcmVxdWVzdCBwYXlsb2FkIHRoYXQgY2FuIGJlIHVzZWQgdG8gY2FsbCB0aGUgY29ycmVzcG9uZGluZyBBUEkuIFRoZSBjbGFzcyBhbHNvIGtlZXBzIHRyYWNrIG9mIE9wZW5BSSB1c2FnZSB0aHJvdWdob3V0IHRoZSBwcm9jZXNzLgoKIyMjIEtleSBmZWF0dXJlcyBvZiB0aGUgUGx1Z2luUmVzb2x2ZXI6CgoqIGBnZW5lcmF0ZV9wYXRoKClgOiBHZW5lcmF0ZXMgYSBwYXRoIGJhc2VkIG9uIHRoZSBnaXZlbiBPcGVuQVBJIHBhdGhzIHNlY3Rpb24sIGNvbnRleHQgbWVzc2FnZXMsIGFuZCB1c2VyIG1lc3NhZ2UuCiogYGdlbmVyYXRlX3BheWxvYWRfZnJvbV9jb21wb25lbnRzX2FuZF90YXJnZXRfcGF0aCgpYDogR2l2ZW4gdGhlIGNvbnRleHQgbWVzc2FnZXMsIHVzZXIgbWVzc2FnZSwgYW5kIGV4dHJhY3RlZCBjb21wb25lbnRzLCB0aGlzIG1ldGhvZCBnZW5lcmF0ZXMgYSByZXF1ZXN0IHBheWxvYWQgZm9yIHRoZSB0YXJnZXQgQVBJLgoqIGByZXNvbHZlKClgOiBSZXNvbHZlcyB0aGUgYXBwcm9wcmlhdGUgcGx1Z2luLCBleHRyYWN0cyB0aGUgcmVsZXZhbnQgcGF0aCBhbmQgY29tcG9uZW50cywgYW5kIGdlbmVyYXRlcyBhIHJlcXVlc3QgcGF5bG9hZCBieSBjYWxsaW5nIHRoZSBhcHByb3ByaWF0ZSBtZXRob2RzLiBJdCBhbHNvIGNhbGN1bGF0ZXMgdGhlIE9wZW5BSSB1c2FnZSBhbmQgdXBkYXRlcyB0aGUgUGx1Z2luUmVzb2x1dGlvblJlc3BvbnNlLgoKVGhlIFBsdWdpblJlc29sdmVyIGNsYXNzIGlzIGRlc2lnbmVkIHRvIGJlIGEgaGlnaC1sZXZlbCBpbnRlcmZhY2UgZm9yIHdvcmtpbmcgd2l0aCBwbHVnaW5zIGluIGFuIEFJLWRyaXZlbiBjb252ZXJzYXRpb24gc3lzdGVtLiBJdHMgbWFpbiBnb2FsIGlzIHRvIHNpbXBsaWZ5IHRoZSBwcm9jZXNzIG9mIGZpbmRpbmcgdGhlIGFwcHJvcHJpYXRlIHBsdWdpbiwgZXh0cmFjdGluZyB0aGUgbmVjZXNzYXJ5IGluZm9ybWF0aW9uLCBhbmQgZ2VuZXJhdGluZyBhIHJlcXVlc3QgcGF5bG9hZCB0aGF0IGNhbiBiZSB1c2VkIHRvIGludGVyYWN0IHdpdGggdGhlIHRhcmdldCBBUEkubyB3b3JrIHdpdGggQUkgcGx1Z2lucyB0aGF0IGZvbGxvdyB0aGUgT3BlbkFQSSBzdGFuZGFyZCwgYW5kIGl0IHV0aWxpemVzIHRoZSBBSSdzIG5hdHVyYWwgbGFuZ3VhZ2UgdW5kZXJzdGFuZGluZyBjYXBhYmlsaXRpZXMgdG8gaWRlbnRpZnkgdGhlIGNvcnJlY3QgQVBJIHBhdGggYW5kIGdlbmVyYXRlIHJlcXVlc3QgcGF5bG9hZHMgYmFzZWQgb24gdXNlciBtZXNzYWdlcwoKVXNhZ2UgZXhhbXBsZToKCmBgYApzeXN0ZW1fbWVzc2FnZSA9IHsicm9sZSI6ICJzeXN0ZW0iLCAiY29udGVudCI6ICJVc2VyIGlkOiA0NSJ9CmNvbnRleHRfbWVzc2FnZXMgPSBbc3lzdGVtX21lc3NhZ2VdCgptZXNzYWdlID0gJ1JlbWVtYmVyIG15IG1vbSBiaXJ0aGRheSBpcyBKYW51YXJ5IDEzdGgnCgpyZXNwb25zZSA9IFBsdWdpblJlc29sdmVyKCkucmVzb2x2ZShjb250ZXh0X21lc3NhZ2VzLCBtZXNzYWdlLCAnaHR0cHM6Ly9ob3N0Ly53ZWxsLWtub3duL2FpLXBsdWdpbi5qc29uJykKcHJpbnQocmVzcG9uc2UpCmBgYAoKIyMjIEV2YWx1YXRpbmcgdGhlIHJlc3BvbnNlCgpUaGUgYFBsdWdpblJlc29sdXRpb25SZXNwb25zZWAgY2xhc3MgaXMgYSBkYXRhIGNsYXNzIHRoYXQgY291bGQgY29udGFpbiBpbmZvcm1hdGlvbiBhYm91dCB0aGUgcGx1Z2luIHRoYXQgd2FzIHJlc29sdmVkLCAKdGhlIHBhdGggdGhhdCB3YXMgZXh0cmFjdGVkLCB0aGUgY29tcG9uZW50cyB0aGF0IHdlcmUgZXh0cmFjdGVkLCBhbmQgdGhlIHJlcXVlc3QgcGF5bG9hZCB0aGF0IHdhcyBnZW5lcmF0ZWQuIEl0IGFsc28gY29udGFpbnMgaW5mb3JtYXRpb24gYWJvdXQgdGhlIE9wZW5BSSB1c2FnZSB0aHJvdWdob3V0IHRoZSBwcm9jZXNzLgoKCiMjIyBGcmFtZXdvcmsKClRoaXMgcHJvamVjdCB3aWxsIHByb3ZpZGUgYmFzZSBtb2RlbHMgdG8gYmUgcmV1c2VkIGluIG90aGVyIHByb2plY3RzLiAKVGhlIGJhc2UgbW9kZWxzIGFyZToKCiogYEJhc2VNZXNzYWdlYDogQmFzZSBjbGFzcyBmb3IgYWxsIG1lc3NhZ2VzCiogYEh1bWFuTWVzc2FnZWA6IENsYXNzIGZvciBtZXNzYWdlcyBmcm9tIGh1bWFucwoqIGBBaU1lc3NhZ2VgOiBDbGFzcyBmb3IgbWVzc2FnZXMgZnJvbSBBSQoqIGBBaUV2YWx1YXRpb25NZXNzYWdlYDogQ2xhc3MgZm9yIG1lc3NhZ2VzIGZyb20gQUkgZXZhbHVhdGlvbiB0aGF0IGFyZSB1c2VkIGluIGludGVybWVkaWF0ZSBzdGVwcy4gCgpUaGUgYEFpRXZhbHVhdGlvbk1lc3NhZ2VgIGFyZSB2ZXJ5IGltcG9ydGFudCBiZWNhdXNlIHRoZXkgaW1wb3NlIGEgZ3JlYXQgd2VpZ2h0IGluIHRoZSBPcGVuQUkgQVBJIHVzYWdlLgoKCgojIyMgTGltaXRhdGlvbnMKCkdpdmVuIHRoZSBkeW5hbWljIG5hdHVyZSBvZiB0aGUgcmVzb2x2ZXIsIGlkZW50aWZ5aW5nIGFuZCB1c2luZyBhbiBhcHByb3ByaWF0ZSBhdXRoZW50aWNhdGlvbiBtZWNoYW5pc20gaXMgYSBjaGFsbGVuZ2UuClNvLCB0aGUgY3VycmVudCBpbXBsZW1lbnRhdGlvbiBvbmx5IHN1cHBvcnRzIHVuYXV0aGVudGljYXRlZCBBUElzLgoKIyMgU2V0dGluZyB1cCB0aGUgcG9ldHJ5IGVudmlyb25tZW50CgpUbyBzZXQgdXAgdGhlIHBvZXRyeSBlbnZpcm9ubWVudCBmb3IgdGhpcyBwcm9qZWN0LCBmb2xsb3cgdGhlc2Ugc3RlcHM6CgoxLiBFbnN1cmUgeW91IGhhdmUgW1BvZXRyeV0oaHR0cHM6Ly9weXRob24tcG9ldHJ5Lm9yZy8pIGluc3RhbGxlZCBvbiB5b3VyIHN5c3RlbS4gSWYgbm90LCB5b3UgY2FuIGluc3RhbGwgaXQgdXNpbmcgdGhlIGZvbGxvd2luZyBjb21tYW5kOgoKYGBgYmFzaApjdXJsIC1zU0wgaHR0cHM6Ly9pbnN0YWxsLnB5dGhvbi1wb2V0cnkub3JnIHwgcHl0aG9uMyAtCmBgYAoKTmF2aWdhdGUgdG8gdGhlIHByb2plY3QncyByb290IGRpcmVjdG9yeSAod2hlcmUgdGhlIHB5cHJvamVjdC50b21sIGZpbGUgaXMgbG9jYXRlZCkuCgpJbnN0YWxsIHRoZSBwcm9qZWN0IGRlcGVuZGVuY2llcyB1c2luZyBQb2V0cnk6CiAgICAKYGBgYmFzaApwb2V0cnkgaW5zdGFsbApgYGAKCkFjdGl2YXRlIHRoZSB2aXJ0dWFsIGVudmlyb25tZW50IGNyZWF0ZWQgYnkgUG9ldHJ5OgoKYGBgYmFzaApwb2V0cnkgc2hlbGwKYGBgCgpZb3Ugc2hvdWxkIG5vdyBoYXZlIGEgZnVsbHkgY29uZmlndXJlZCBwb2V0cnkgZW52aXJvbm1lbnQgd2l0aCBhbGwgbmVjZXNzYXJ5IGRlcGVuZGVuY2llcyBpbnN0YWxsZWQuCgojIyBCdWlsZGluZyB0aGUgd2hlZWwgZmlsZQpUbyBidWlsZCBhIHdoZWVsIGZpbGUgZm9yIHRoaXMgcHJvamVjdCwgZm9sbG93IHRoZXNlIHN0ZXBzOgoKRW5zdXJlIHlvdSBhcmUgaW4gdGhlIHByb2plY3QncyByb290IGRpcmVjdG9yeS4KCkJ1aWxkIHRoZSB3aGVlbCBmaWxlIHVzaW5nIFBvZXRyeToKYGBgYmFzaApwb2V0cnkgYnVpbGQgLWYgd2hlZWwKYGBgCgpUaGlzIHdpbGwgZ2VuZXJhdGUgYSAud2hsIGZpbGUgaW4gdGhlIGRpc3QgZm9sZGVyIHdpdGhpbiB0aGUgcHJvamVjdCdzIHJvb3QgZGlyZWN0b3J5LgoKSW1wb3J0aW5nIHRoZSB3aGVlbCBmaWxlIGludG8gYW5vdGhlciBwcm9qZWN0ClRvIGltcG9ydCB0aGUgZ2VuZXJhdGVkIHdoZWVsIGZpbGUgaW50byBhbm90aGVyIHByb2plY3QsIGZvbGxvdyB0aGVzZSBzdGVwczoKCkNvcHkgdGhlIC53aGwgZmlsZSBmcm9tIHRoZSBkaXN0IGZvbGRlciBvZiB0aGUgY3VycmVudCBwcm9qZWN0IHRvIHRoZSByb290IGRpcmVjdG9yeSBvZiB0aGUgdGFyZ2V0IHByb2plY3QuCgpJbnN0YWxsIHRoZSB3aGVlbCBmaWxlIGluIHRoZSB0YXJnZXQgcHJvamVjdCB1c2luZyBwaXA6CmBgYGJhc2gKcGlwIGluc3RhbGwgPHdoZWVsLWZpbGUtbmFtZT4ud2hsCmBgYAoKWW91IGNhbiBub3cgdXNlIHRoZSBmdW5jdGlvbmFsaXR5IHByb3ZpZGVkIGJ5IHRoaXMgcHJvamVjdCBpbiB0aGUgdGFyZ2V0IHByb2plY3QgYnkgaW1wb3J0aW5nIHRoZSBuZWNlc3NhcnkgbW9kdWxlcyBhbmQgY2xhc3Nlcy4KCg== readmeEtag: '"3c43e51dd1a9b64d4d2b214b67ee46bdf0588658"' readmeLastModified: Sat, 08 Apr 2023 03:55:10 GMT repositoryId: 623116677 description: >- Project provides functionality to receive a message and try to identify which path and operation to use from an openapi spec created: '2023-04-03T18:19:54Z' updated: '2023-04-08T03:27:20Z' language: Python archived: false stars: 0 watchers: 0 forks: 0 owner: butley logo: https://avatars.githubusercontent.com/u/127567134?v=4 repoEtag: '"68f34ec1159389d14f7e7a2cc48d5cd71e18fa17923814439ce29e43d60124d5"' repoLastModified: Sat, 08 Apr 2023 03:27:20 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/userprofile v3: true id: d448cdd7fbb737c0adcff01770e1454e repositoryMetadata: base64Readme: >- PiBbIUlNUE9SVEFOVF0gIAo+IFRoaXMgcmVwb3NpdG9yeSBpcyByZWFkLW9ubHkgLyBhcmNoaXZlZCBhbmQgd2lsbCBub3QgcmVjaWV2ZSB1cGRhdGVzLgoKIyBbVHlwZWRvYyBEb2N1bWVudGF0aW9uIFdlYnNpdGVdKGh0dHBzOi8vc3Vkb2t1cnUuZ2l0aHViLmlvL1VzZXJBY3RpdmVHYW1lcy8pPGJyPgoKCiMgRGV2ZWxvcGVyIFNldHVwCgoxLiBHZXQgdGhlIC5lbnYgZmlsZSBmcm9tIHRoZSBNU0IgYnVpbGRpbmcgbmV4dCB0byB0aGUgd2F0ZXIgZm91bnRhaW4uIAoyLiBJbnN0YWxsIERvY2tlciBvbiB5b3VyIG1hY2hpbmUuIFR1dG9yaWFsIGlzIGxpbmtlZCBiZWxvdzo8YnI+CiAgIFshW0RvY2tlciBUdXRvcmlhbF0oaHR0cHM6Ly9pbWcueW91dHViZS5jb20vdmkvMmV6TnFxYVNqcTgvMC5qcGcpXShodHRwczovL3d3dy55b3V0dWJlLmNvbS93YXRjaD92PTJlek5xcWFTanE4KTxicj4KMi4gT25jZSBkb2NrZXIgaXMgaW5zdGFsbGVkLCB0aGUgTW9uZ28gaW1hZ2UgY2FuIGJlIHJ1biB3aXRoIHRoaXMgY29tbWFuZDo8YnI+Ck5vdGUgdXNlIGBgYHN1ZG9gYGAgb24gTGludXgvTWFjPGJyPgpgYGBjb25zb2xlCm5wbSBydW4gZG9ja2VyCmBgYAozLiBUaGUgYXBwIGNhbiB0aGVuIGJlIHJ1biB3aXRoIHRoZSBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHN0YXJ0CmBgYAo0LiBJbnRlZ3JhdGlvbiB0ZXN0cyBjYW4gYmUgcnVuIHdoZW4gdGhlIGFwcCBpcyBydW5uaW5nIHdpdGggdGhpcyBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHRlc3Q6aW50ZWdyYXRpb24KYGBgCg== readmeEtag: '"551645e2a891d8f25afe5c10c232d868e7f8455c"' readmeLastModified: Sat, 11 May 2024 15:14:39 GMT repositoryId: 609680674 description: null created: '2023-03-04T22:48:51Z' updated: '2024-05-11T15:14:58Z' language: TypeScript archived: true stars: 0 watchers: 0 forks: 0 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"d348f13db76f24cda01be5eb411e40bf9c3394ad7d8bc999536576f1bac7db38"' repoLastModified: Sat, 11 May 2024 15:14:58 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/edavis25/crypto_history_api v3: true repositoryMetadata: base64Readme: >- # Crypto History API
This project exposes crypto price history I personally scraped from various exchanges into a consistent and extendable API. It provides a layer of abstraction for InfluxDB measurements that have varying schemas and data points.


### Just looking for the API documentation?
[View the API Documentation](https://edavis25.github.io/crypto_history_api/)

___ 
### Looking to stand up own version of the project?
Did you randomly feel compelled to scrape a bunch of historical crypto price data? Did you also decide to use an obscure timeseries database, InfluxDB? Did you finally decide you wanted a consistent API to retrieve the hodge-podge data you scraped? Cool, me too. Let's be friends. Also, keep reading to use this project to stand up an API for your very own InfluxDB data hoard.

## Pre Reqs
This project assumes you have an instance of InfluxDB running a single database that contains different "measurements" (Influx terminology) for recording prices from different exchages. For example, my database contains Open/Close/High/Low/etc data for the Poloniex exchange under a single Influx measurement. Measurements can be loosely thought of as tables and in this case, our table contains historic price data from the Poloniex exchange (and only Poloniex). Different measurements can be created for different exchanges and each record should contain a timestamp, whatever price data you want, and a tag for the trading "pair" (btc_usd).

For example, if you wanted to get price history for btc_usd on the Poloniex exchange, you can imagine an SQL query: `SELECT * FROM poloniex_exchange WHERE pair = 'btc_usd'` for this setup.

TLDR; Create a single Influx database with different measurements for every different exchange's price data. Each record needs to contain whatever price data you want and an [Influx tag](https://docs.influxdata.com/influxdb/v1.8/concepts/glossary/#tag-key) referencing the trading pair. This project assumes you named your tag: `pair` but this is customizable if you are a masochist.

## Installation
1. Clone the repository<br>`git clone https://github.com/edavis25/crypto_history_api.git`
2. Install project dependencies<br>`composer install`
3. Create your environment's configuration file
<br>`cp ./.env.example ./.env`
<br><br> The most important part of this step is configuring your connection to InfluxDB:
```
INFLUXDB_HOST=127.0.0.1
INFLUXDB_PORT=8086
INFLUXDB_SSL=false
INFLUXDB_VERIFYSSL=false
INFLUXDB_TIMEOUT=0
INFLUXDB_DBNAME=crypto_price_history # All measurements must be inside single database
```
4. Run the Laravel migrations.
<br>`php artisan migrate`

## Setup InfluxDB measurements for the API
Now that your environment is setup, we can begin mapping our InfluxDB measurements for use in the API. To do this, we will need to do a few things:
1. Create a new service that fulfills the `InfluxDBMeasurement` contract
2. Create [JSON resources](https://laravel.com/docs/8.x/eloquent-resources) for our response data schemas
3. Map our new service class inside the `influxdb.php` config file.

#### 1. Creating the service
Every InfluxDB measurement will need its own service implementing the `InfluxDBMeasurement` contract so that it can be dynamically injected into the controllers. As a starting point, you can create a class that extends the `BaseMeasurementService` which will fulfill a majority of the contract's functionality.
<br>
```
class PoloniexMeasurementService extends BaseMeasurementService
{
    // the name of the measurement inside of InfluxDB
    public function measurement(): string
    {
        return 'poloniex_exchange';
    }

    // a human-friendly version of the measurement's name for display purposes
    public function displayName(): string
    {
        return 'Poloniex';
    }
    
    // responsible for transforming the data from the measurement's schema into a standardized JSON resource.
    public function buildResource(array $data): JsonResource
    {
        return new PoloniexMeasurement($data);
    }

    // responsible for building a collection of multiple JSON resources.
    public function buildResourceCollection(array $data, bool $has_next_page): ResourceCollection
    {
        return new PoloniexMeasurementCollection($data, $has_next_page);
    }
   
}
```

At the very least, each service will need to implement the methods shown above individually. If you mostly follow the InfluxDB setup explained above, you can rely on the `BaseMeasurementService` to fulfill most of the contract. If you need more customization, you can override the functions inside the `BaseMeasurementService` as necessary.

#### 2. Creating the JSON resources
If you noticed in that last step that we were building resources that didn't exist yet, good eyes. Let's do that now.

If you are like me, you may have various measurements each containing varying data points with different names. The main purpose of creating these resources is to transform data from our InfluxDB measurement schemas into consistent API responses. There isn't really anything fancy going on here, we just need to map the data we want from InfluxDB into [Laravel's native API Resources](https://laravel.com/docs/8.x/eloquent-resources).

```
class PoloniexMeasurement extends JsonResource
{
    // Transform the resource into an array used in the response.
    public function toArray($request)
    {
        return [
            'time'   => (string) $this->resource['time'],
            'pair'   => (string) $this->resource['pair'],
            'open'   => (float) $this->resource['open'],
            'close'  => (float) $this->resource['close'],
            'low'    => (float) $this->resource['low'],
            'high'   => (float) $this->resource['high'],
            'volume_from' => (float) $this->resource['volume'],
            'volume_to' => (float) $this->resource['quote_volume'],
        ];
    }
}
```

// todo fill in some info about creating collections with pagination using the traits.

#### 3. Configuration
The final step is mapping our service to an endpoint in the `influxdb.php` config file. Navigate to the file and look for the `measurements` array. This array will contain key=>value pairs where your measurement is the key and the namespace of your service class is the value:
```
/*
 |--------------------------------------------------------------------------
 | Measurements
 |--------------------------------------------------------------------------
 | ...<removed for brevity>...
*/
 'measurements' => [
    'poloniex_exchange' => \App\Services\PoloniexMeasurementService::class,
 ],
```
Boom! Once your mapping has been created, you can use the API to query your data using the key name as your endpoint:
<br> `127.0.0.1/api/poloniex_exchange/pairs`

#### Route aliases
I know what you're thinking, that URL is ugly? Well I thought so too, and instead of `poloniex_exchange` wouldn't it be cleaner just being: `/api/poloniex/pairs`? Of course it would. To do this, you can use the `route_aliases` array in the same config file. This array will again be key=>pair values where the key becomes the new endpoint and the value contains a reference to the measurement name.
```
/*
|--------------------------------------------------------------------------
| Route Overrides
|--------------------------------------------------------------------------
| ...<removed for brevity>...
*/
route_aliases' => [
    'poloniex' => 'poloniex_exchange'
]
```
With this alias, we can now use the API to query the `127.0.0.1/api/poloniex/pairs` endpoint instead and regale in the beauty of the URL.

## The end ...?
This is all still very much a work in progress and I'm hoping to add price history for different exchanges. If you were crazy enough to actually read through this scatter-brained documentation and end up creating your own exchange services, feel free to open a PR!

 readmeEtag: '"bdb2a5065519ca052ad0d93b25ef6a74ebb55c20"' readmeLastModified: Sat, 19 Jun 2021 09:16:24 GMT repositoryId: 375711453 description: >- API for retrieving personally scraped historic crypto price data from various exchanges. created: '2021-06-10T13:41:39Z' updated: '2023-03-13T07:17:39Z' language: PHP archived: false stars: 0 watchers: 1 forks: 1 owner: edavis25 logo: https://avatars.githubusercontent.com/u/14096584?v=4 repoEtag: '"7c93e667aeb1c8df27832a1f290b792a25684592164bbe5b277f0492169b01e6"' repoLastModified: Mon, 13 Mar 2023 07:17:39 GMT foundInMaster: true category: Server Implementations id: 6f7d4c178d5853c9a7a96bc6d029137e - source: openapi3 tags repository: https://github.com/ekoulemaneng/openapi-schema-retriever v3: true id: 5f0055e058a71ad791dc6310d364782b repositoryMetadata: base64Readme: >- IyBPcGVuYXBpIFNjaGVtYSBSZXRyaWV2ZXIKClR5cGVzY3JpcHQgcGFja2FnZSB0byBnZXQgYW4gT3BlbkFwaS1zcGVjaWZpY2F0aW9uLWNvbXBsaWFudCBkZXJlZmVyZW5jZWQgb2JqZWN0IGZyb20gYW4gb2JqZWN0IG9yIGFuIGpzb24sIHlhbWwgb3IgeW1sIGZpbGUgcGF0aCBzdHJpbmcuIEZvciAzLjEueCBhbmQgMy4wLnggT3BlbkFQSSB2ZXJzaW9ucy4KCiMjIEluc3RhbGxhdGlvbgpgYGB0eXBlc2NyaXB0Cm5wbSBpbnN0YWxsIG9wZW5hcGktc2NoZW1hLXJldHJpZXZlciAKYGBgCm9yCmBgYHR5cGVzY3JpcHQKeWFybiBhZGQgb3BlbmFwaS1zY2hlbWEtcmV0cmlldmVyCmBgYAoKIyMgVXNhZ2UKYGBgdHlwZXNjcmlwdAppbXBvcnQgc2NoZW1hUmV0cmlldmVyIGZyb20gJ29wZW5hcGktc2NoZW1hLXJldHJpZXZlcicKCmNvbnN0IG1haW4gPSBhc3luYyAoKSA9PiB7CgogICAgLyogc2NoZW1hUmV0cmlldmVyICBnZXRzLCBjaGVja3MgYW5kIHBhcnNlcyB0aGUgb3BlbmFwaSBzcGVjaWZpY2F0aW9uIHRoYXQgY2FuIGJlIGFuIG9iamVjdCBvciBhbiBmaWxlIHBhdGggc3RyaW5nLgogICAgICogSWYgdGhlIHNwZWNpZmljYXRpb24gaW5wdXQgaXMgYSBmaWxlIHBhdGggc3RyaW5nLCBhIHNlY29uZCBhcmd1bWVudCBzdGFuZGluZyBmb3IgdGhlIGN1cnJlbnQgd29ya2luZyBkaXJlY3RvcnkgaXMgbWFuZGF0b3J5LgogICAgICogV2UgcmVjb21tZW5kIHRvIHVzZSAnX19kaXJuYW1lJyBhcyBzZWNvbmQgYXJndW1lbnQuCiAgICAgKiBUaGUgZmlsZSBtdXN0IGJlIGVpdGhlciBhIGpzb24sIGEgeWFtbCBvciBhIHltbCBmaWxlLiAKICAgICAqIHNjaGVtYVJldHJpZXZlciByZXR1cm5zIGFuIGRlcmVmZXJlbmNlZCBvYmplY3QgaW4gYWNjb3JkYW5jZSB3aXRoIE9wZW5BUEkgc2NwZWNpZmljYXRpb24uCiAgICAqLwogICAgY29uc3Qgc2NoZW1hID0gYXdhaXQgc2NoZW1hUmV0cmlldmVyKCcuL29wZW5hcGkueWFtbCcsIF9fZGlybmFtZSkKCiAgICBjb25zb2xlLmxvZyhzY2hlbWEpIC8qID0+IHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICBvcGVuYXBpOiAnMy4xLjAnCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgLi4uLi4uLi4uLi4uLi4uCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIH0KICAgICAgICAgICAgICAgICAgICAgICAgKi8KfQoKbWFpbigpCmBgYAoKIyMgTGljZW5zZQpUaGlzIHBhY2thZ2UgaXMgbGljZW5zZWQgdW5kZXIgdGhlIFtNSVQgTGljZW5zZV0oaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9taXQpLgoKIyMgQ29udGFjdApJZiB5b3UgaGF2ZSBhbnkgcXVlc3Rpb25zIG9yIGlzc3VlcywgcGxlYXNlIGNvbnRhY3QgdGhlIHBhY2thZ2UgbWFpbnRhaW5lciBhdCBla291bGVtYW5lbmdAZ21haWwuY29tLgo= readmeEtag: '"e7ace22fa28615c3b4c8fe6d8f96c2812af10026"' readmeLastModified: Sat, 06 May 2023 18:52:26 GMT repositoryId: 636995451 description: >- Typescript package to get an OpenApi-specification-compliant dereferenced object from an object or an json, yaml or yml file path string. For 3.1.x and 3.0.x OpenAPI versions. created: '2023-05-06T07:25:03Z' updated: '2023-05-06T12:22:05Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: ekoulemaneng logo: https://avatars.githubusercontent.com/u/12666431?v=4 license: MIT repoEtag: '"787a55da426eda6a845b01fa64cb5cb0b19d3beb79f151afb18d675601c3c0e8"' repoLastModified: Sat, 06 May 2023 12:22:05 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/afshinparhizkari/issuetracker v3: true repositoryMetadata: base64Readme: >- IyBpc3N1ZXRyYWNrZXIKc29tZSBzaW1wbGUgcHJvamVjdCBmb3IgdHJhY2tpbmcgaXNzdWVzKHN0b3JpZXMgYW5kIGJ1Z3MpCg== readmeEtag: '"a8720708490c9e35797e440f01d30d7737e96e5c"' readmeLastModified: Wed, 08 Nov 2023 11:20:54 GMT repositoryId: 446873964 description: some simple project for tracking issues(stories and bugs) created: '2022-01-11T15:22:37Z' updated: '2022-05-05T07:26:46Z' language: Java archived: false stars: 0 watchers: 1 forks: 1 owner: AfshinParhizkari logo: https://avatars.githubusercontent.com/u/32636285?v=4 license: Apache-2.0 repoEtag: '"83664ad7daae6d51fbd342bac222dcbd73d5f9b433e6f7244bb272e9f83995b5"' repoLastModified: Thu, 05 May 2022 07:26:46 GMT foundInMaster: true category: - Low-level Tooling - Server - Server Implementations id: edae04f136ed5f6a77f930858ab72307 - source: https://openapi.tools/ name: ReadMe category: Documentation link: https://readme.com language: SaaS source_description: Build beautiful, personalized, interactive developer hubs. 🦉 v2: true v3: true v3_1: true foundInMaster: true id: 8959152e98349ff9aac5f6154d972854 - source: openapi3 tags repository: https://github.com/yoshinariyamanaka/backendapp-with-openapiv3-postgresql v3: true id: 9ca431d068cdf892e96fa0c46f040510 repositoryMetadata: base64Readme: >- IyBJbnRyb2R1Y3Rpb25zCgppbnN0YWxsIHBhY2thZ2VzIGFuZCBjb21waWxlIGFsbCB3b3Jrc3BhY2UgcGFja2FnZXMKCmBgYHNoCnlhcm4gaW5zdGFsbApgYGAKClRoZSBjb21tb24gcGFja2FnZSBpcyBsaW5rZWQgYXMgYSBzeW1ib2xpYyBsaW5rCmNoZWNrIGl0IG91dAoKYGBgCmNkIC4vbm9kZV9tb2R1bGVzL0BiYWNrZW5kCmxuIC1zIC4uLy4uL3BhY2thZ2VzL2NvbW1vbiAuCmBgYAoKIyBsb2NhbCBkZXZlbG9wbWVudAoKIyMgY29tbW9uCgpjcmVhdGUgdGFibGVzIGFuZCBpbnNlcnQgc2FtcGxlIGRhdGEKCmBgYApzZXJ2aWNlIHBvc3RncmVzcWwgc3RhcnQKLi9yZXNvdXJjZXMvY29tbW9uL2RkbC5zaApgYGAKCiMjIGFwaQoKIyMjIGJ1aWxkIGFuZCBzdGFydCBvciBkbyB0ZXN0cwoKYGBgCnlhcm4gd29ya3NwYWNlIEBiYWNrZW5kL2FwaSBidWlsZAp5YXJuIHdvcmtzcGFjZSBAYmFja2VuZC9hcGkgc3RhcnQKeWFybiB3b3Jrc3BhY2UgQGJhY2tlbmQvYXBpIHRlc3QKYGBgCgojIyMgd2l0aCBlbnZpcm9ubWVudCB2YWx1ZXMKCmBgYAouL3Jlc291cmNlcy9hcGkvbG9jYWxfZGV2LnNoIGNvbXBpbGUKLi9yZXNvdXJjZXMvYXBpL2xvY2FsX2Rldi5zaCBzdGFydAouL3Jlc291cmNlcy9hcGkvbG9jYWxfZGV2LnNoIHRlc3QKYGBgCgojIyMgYWRkIGxpYnJhcmllcyB0byB0aGUgZW50aXJlIHByb2plY3QKCmBgYAp5YXJuIGFkZCBgc29tZSBwYWNrYWdlYCAtVwpvcgp5YXJuIGFkZCAtLWRldiBgc29tZSBwYWNrYWdlYCAtVwpgYGAKCiMjIyBPcGVuQVBJIGRvY3MKCm9wZW4gc2hvd24gYmVsb3cgb24gdGhlIFdlYiBicm93c2VyCgpgYGAKbG9jYWxob3N0OjMwMDAvYXBpLWRvY3MvIwpgYGAKCiMgRGlyZWN0b3JpZXMKCnRyZWUgLUkgIm5vZGVfbW9kdWxlc3xidWlsZCIK readmeEtag: '"73eab30e10700867d04993b37d8b74614f9b7502"' readmeLastModified: Fri, 17 Nov 2023 04:49:23 GMT repositoryId: 437560745 description: BackendApp with OpenAPIV3 PostgreSQL implemented with TypeScript created: '2021-12-12T13:59:56Z' updated: '2023-01-27T05:53:16Z' language: TypeScript archived: false stars: 0 watchers: 0 forks: 0 owner: YoshinariYamanaka logo: https://avatars.githubusercontent.com/u/59109232?v=4 repoEtag: '"e26e79211bb49fd6e3f9c2f79f2ba5949d0f3e11c830b61377a15c84fdd8eb1a"' repoLastModified: Fri, 27 Jan 2023 05:53:16 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/yo-mah-ya/backendapp-with-openapiv3-postgresql - source: openapi3 tags repository: https://github.com/smartdwell/thesis-images v3: true id: 4a207bd8b0b8ec5bd7a11159ceddd7de repositoryMetadata: base64Readme: IyB0aGVzaXMtaW1hZ2Vz readmeEtag: '"47dba36aca3c977235d0b448e1bcab267848b51b"' readmeLastModified: Thu, 19 Jan 2023 06:55:34 GMT repositoryId: 590771118 description: Сервис для работы с изображениями для "Thesis System" created: '2023-01-19T06:55:34Z' updated: '2024-09-24T14:46:34Z' language: null archived: false stars: 0 watchers: 0 forks: 0 owner: SmartDwell logo: https://avatars.githubusercontent.com/u/158821215?v=4 repoEtag: '"83afa598ebc3b8aab949ee5bf4dee0fbe0a4457bbfc852c4aeada53eae021a9f"' repoLastModified: Tue, 24 Sep 2024 14:46:34 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/thesis-system/thesis-images - source: openapi3 tags repository: https://github.com/mulekick/ocr-argent-bank-frontend v3: true id: c15db9c54d4b423c100a74f71b262916 repositoryMetadata: base64Readme: >- IyBDcmVhdGUgUmVhY3QgQXBwIGN1c3RvbWl6YXRpb25zIDoKMS4gdXNlIHRoZSByZWR1eCB0ZW1wbGF0ZSBmb3IgY3JlYXRlLXJlYWN0LWFwcApgYGBiYXNoCm5weCBjcmVhdGUtcmVhY3QtYXBwIG9jci1hcmdlbnQtYmFuay1mcm9udGVuZCAtLXRlbXBsYXRlIHJlZHV4CmBgYAoyLiByZW1vdmUgdXNlbGVzcyBkZXBlbmRlbmNpZXMKYGBgYmFzaApucG0gdW5pbnN0YWxsIC0tc2F2ZSBAdGVzdGluZy1saWJyYXJ5L2plc3QtZG9tIEB0ZXN0aW5nLWxpYnJhcnkvcmVhY3QgQHRlc3RpbmctbGlicmFyeS91c2VyLWV2ZW50IHdlYi12aXRhbHMKYGBgCjMuIGluc3RhbGwgcHJvamVjdCBkZXYgZGVwZW5kZW5jaWVzCmBgYGJhc2gKbnBtIGluc3RhbGwgLS1zYXZlLWRldiBAbXVsZWtpY2svZXNsaW50LWNvbmZpZy1tdWxlc2xpbnQKYGBgCjQuIGluc3RhbGwgcHJvamVjdCBkZXBlbmRlbmNpZXMKYGBgYmFzaApucG0gaW5zdGFsbCAtLXNhdmUgcmVhY3Qtcm91dGVyLWRvbQpgYGAKNS4gdXBkYXRlIHBhY2thZ2UuanNvbiA6CiAgIC0gYWRkIGN1c3RvbSBwYWNrYWdlLmpzb24gc2NyaXB0cwogICAtIHJlbW92ZSBgYGBlc2xpbnRDb25maWdgYGAga2V5CiAgIC0gc3dpdGNoIHRvIEVTTSBtb2R1bGVzIHVzZSA6YGBgInR5cGUiOiAibW9kdWxlImBgYAogICAtIGFkZCBpbmZvcm1hdGlvbmFsIGVudHJpZXMKNi4gYWRkIGN1c3RvbSBgYGAuZXNsaW50cmMuanNvbmBgYCBmaWxlCjcuIGZpbGUgc3lzdGVtIGNsZWFudXAKYGBgYmFzaApybSBzcmMvc2V0dXBUZXN0cy5qcyBzcmMvcmVwb3J0V2ViVml0YWxzLmpzIHNyYy9BcHAudGVzdC5qcyBzcmMvZmVhdHVyZXMvY291bnRlci9jb3VudGVyU2xpY2Uuc3BlYy5qcyBwdWJsaWMvbG9nbyogcHVibGljL21hbmlmZXN0Lmpzb24gcHVibGljL3JvYm90cy50eHQKYGBgCgojIEdldHRpbmcgU3RhcnRlZCB3aXRoIENyZWF0ZSBSZWFjdCBBcHAgYW5kIFJlZHV4CgpUaGlzIHByb2plY3Qgd2FzIGJvb3RzdHJhcHBlZCB3aXRoIFtDcmVhdGUgUmVhY3QgQXBwXShodHRwczovL2dpdGh1Yi5jb20vZmFjZWJvb2svY3JlYXRlLXJlYWN0LWFwcCksIHVzaW5nIHRoZSBbUmVkdXhdKGh0dHBzOi8vcmVkdXguanMub3JnLykgYW5kIFtSZWR1eCBUb29sa2l0XShodHRwczovL3JlZHV4LXRvb2xraXQuanMub3JnLykgdGVtcGxhdGUuCgojIyBBdmFpbGFibGUgU2NyaXB0cwoKSW4gdGhlIHByb2plY3QgZGlyZWN0b3J5LCB5b3UgY2FuIHJ1bjoKCiMjIyBgbnBtIHN0YXJ0YAoKUnVucyB0aGUgYXBwIGluIHRoZSBkZXZlbG9wbWVudCBtb2RlLlwKT3BlbiBbaHR0cDovL2xvY2FsaG9zdDozMDAwXShodHRwOi8vbG9jYWxob3N0OjMwMDApIHRvIHZpZXcgaXQgaW4geW91ciBicm93c2VyLgoKVGhlIHBhZ2Ugd2lsbCByZWxvYWQgd2hlbiB5b3UgbWFrZSBjaGFuZ2VzLlwKWW91IG1heSBhbHNvIHNlZSBhbnkgbGludCBlcnJvcnMgaW4gdGhlIGNvbnNvbGUuCgojIyMgYG5wbSB0ZXN0YAoKTGF1bmNoZXMgdGhlIHRlc3QgcnVubmVyIGluIHRoZSBpbnRlcmFjdGl2ZSB3YXRjaCBtb2RlLlwKU2VlIHRoZSBzZWN0aW9uIGFib3V0IFtydW5uaW5nIHRlc3RzXShodHRwczovL2ZhY2Vib29rLmdpdGh1Yi5pby9jcmVhdGUtcmVhY3QtYXBwL2RvY3MvcnVubmluZy10ZXN0cykgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgojIyMgYG5wbSBydW4gYnVpbGRgCgpCdWlsZHMgdGhlIGFwcCBmb3IgcHJvZHVjdGlvbiB0byB0aGUgYGJ1aWxkYCBmb2xkZXIuXApJdCBjb3JyZWN0bHkgYnVuZGxlcyBSZWFjdCBpbiBwcm9kdWN0aW9uIG1vZGUgYW5kIG9wdGltaXplcyB0aGUgYnVpbGQgZm9yIHRoZSBiZXN0IHBlcmZvcm1hbmNlLgoKVGhlIGJ1aWxkIGlzIG1pbmlmaWVkIGFuZCB0aGUgZmlsZW5hbWVzIGluY2x1ZGUgdGhlIGhhc2hlcy5cCllvdXIgYXBwIGlzIHJlYWR5IHRvIGJlIGRlcGxveWVkIQoKU2VlIHRoZSBzZWN0aW9uIGFib3V0IFtkZXBsb3ltZW50XShodHRwczovL2ZhY2Vib29rLmdpdGh1Yi5pby9jcmVhdGUtcmVhY3QtYXBwL2RvY3MvZGVwbG95bWVudCkgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgojIyMgYG5wbSBydW4gZWplY3RgCgoqKk5vdGU6IHRoaXMgaXMgYSBvbmUtd2F5IG9wZXJhdGlvbi4gT25jZSB5b3UgYGVqZWN0YCwgeW91IGNhbid0IGdvIGJhY2shKioKCklmIHlvdSBhcmVuJ3Qgc2F0aXNmaWVkIHdpdGggdGhlIGJ1aWxkIHRvb2wgYW5kIGNvbmZpZ3VyYXRpb24gY2hvaWNlcywgeW91IGNhbiBgZWplY3RgIGF0IGFueSB0aW1lLiBUaGlzIGNvbW1hbmQgd2lsbCByZW1vdmUgdGhlIHNpbmdsZSBidWlsZCBkZXBlbmRlbmN5IGZyb20geW91ciBwcm9qZWN0LgoKSW5zdGVhZCwgaXQgd2lsbCBjb3B5IGFsbCB0aGUgY29uZmlndXJhdGlvbiBmaWxlcyBhbmQgdGhlIHRyYW5zaXRpdmUgZGVwZW5kZW5jaWVzICh3ZWJwYWNrLCBCYWJlbCwgRVNMaW50LCBldGMpIHJpZ2h0IGludG8geW91ciBwcm9qZWN0IHNvIHlvdSBoYXZlIGZ1bGwgY29udHJvbCBvdmVyIHRoZW0uIEFsbCBvZiB0aGUgY29tbWFuZHMgZXhjZXB0IGBlamVjdGAgd2lsbCBzdGlsbCB3b3JrLCBidXQgdGhleSB3aWxsIHBvaW50IHRvIHRoZSBjb3BpZWQgc2NyaXB0cyBzbyB5b3UgY2FuIHR3ZWFrIHRoZW0uIEF0IHRoaXMgcG9pbnQgeW91J3JlIG9uIHlvdXIgb3duLgoKWW91IGRvbid0IGhhdmUgdG8gZXZlciB1c2UgYGVqZWN0YC4gVGhlIGN1cmF0ZWQgZmVhdHVyZSBzZXQgaXMgc3VpdGFibGUgZm9yIHNtYWxsIGFuZCBtaWRkbGUgZGVwbG95bWVudHMsIGFuZCB5b3Ugc2hvdWxkbid0IGZlZWwgb2JsaWdhdGVkIHRvIHVzZSB0aGlzIGZlYXR1cmUuIEhvd2V2ZXIgd2UgdW5kZXJzdGFuZCB0aGF0IHRoaXMgdG9vbCB3b3VsZG4ndCBiZSB1c2VmdWwgaWYgeW91IGNvdWxkbid0IGN1c3RvbWl6ZSBpdCB3aGVuIHlvdSBhcmUgcmVhZHkgZm9yIGl0LgoKIyMgTGVhcm4gTW9yZQoKWW91IGNhbiBsZWFybiBtb3JlIGluIHRoZSBbQ3JlYXRlIFJlYWN0IEFwcCBkb2N1bWVudGF0aW9uXShodHRwczovL2ZhY2Vib29rLmdpdGh1Yi5pby9jcmVhdGUtcmVhY3QtYXBwL2RvY3MvZ2V0dGluZy1zdGFydGVkKS4KClRvIGxlYXJuIFJlYWN0LCBjaGVjayBvdXQgdGhlIFtSZWFjdCBkb2N1bWVudGF0aW9uXShodHRwczovL3JlYWN0anMub3JnLykuCg== readmeEtag: '"2c85d204e7da428e7a5780e72744622894e3faba"' readmeLastModified: Wed, 22 Feb 2023 21:35:21 GMT repositoryId: 590762652 description: >- 'Connect to a Back End with an API Using React' - OpenClassRooms project #13 created: '2023-01-19T06:23:03Z' updated: '2023-07-08T11:31:44Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: mulekick logo: https://avatars.githubusercontent.com/u/24319237?v=4 license: MIT repoEtag: '"c423389f31c515d4a4ca9b4b5733210fde0b0098b4228c9361280d138680aa37"' repoLastModified: Sat, 08 Jul 2023 11:31:44 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/artemovskiy/pumapi v3: true repositoryMetadata: base64Readme: >- IyBuZXN0LWpzb24tYXBpCkEgdG9vbCBmb3IgZWFzeSBidWlsZGluZyBBUEkgYWNvcmRpbmcgW0pTT05BUEkgU1BFQ10oaHR0cHM6Ly9qc29uYXBpLm9yZy8pLgoKRmVhdHVyZXM6CiogQXV0b21hdGljIHNlcmlhbGl6YXRpb24KKiBBdXRvbWF0aWMgZGVzZXJpYWxpemF0aW9uCiogQXV0b21hdGljIGJ1aWxkaW5nIG9mIEFQSSBzcGVjCg== readmeEtag: '"c30f1c9b70918584d8207b0cf254de9c6072a9e3"' readmeLastModified: Fri, 17 Mar 2023 05:42:27 GMT repositoryId: 475965997 description: A tool for easy building API according the JSONAPI spec created: '2022-03-30T16:37:56Z' updated: '2022-03-30T16:57:42Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: artemovskiy logo: https://avatars.githubusercontent.com/u/16016725?v=4 repoEtag: '"9a71d315f5e623c0edbe6defc7b0545014b74bb0a5e20d51945961c876107c34"' repoLastModified: Wed, 30 Mar 2022 16:57:42 GMT foundInMaster: true category: - Server - Server Implementations id: 7263baf7be8c738f6cbe4ac487c08c7e oldLocations: - https://github.com/xydens/pumapi - source: openapi3 tags repository: https://github.com/mrigankaghosh/crudapp-with-rest-api v3: true id: 87aab7ae0b65208e063a70d8a51187bd repositoryMetadata: base64Readme: >- IyBDcnVkYXBwLXdpdGgtUkVTVC1BUEkNCg0KQ3J1ZGFwcCBpcyBhIGJhc2ljIFJlc3QgQVBJIHRoYXQgY2FuIHBlcmZvcm0gYSBidW5jaCBvZiBjcnVkIG9wZXJhdGlvbnMgYXMgZm9sbG93czoNCjEuIEFkZCBhIFVzZXINCjIuIERlbGV0ZSBhIFVzZXINCjMuIFNob3cgYWxsIFVzZXJzDQo0LiBDb3VudCB0aGUgbnVtYmVyIG9mIFVzZXJzDQo1LiBTaG93IFVzZXJzIGJ5IHRoZWlyIElkcw0KNi4gU2hvdyBVc2VycyBieSB0aGVpciBuYW1lcw0KNy4gU2hvdyBVc2VycyBieSB0aGVpciBwaG9uZSBudW1iZXINCjguIFNob3cgVXNlcnMgYnkgdGhlaXIgZW1haWwNCg0KVGhlIGRvY3VtZW50YXRpb24gZm9yIHRoaXMgUkVTVCBBUEkgaXMgaW4gdGhlIGZpbGUgbmFtZWQgZG9jdW1lbnRhdGlvbi55YW1sIA0KDQpUaGUgbGFuZ3VhZ2UsIGZyYW1ld29ya3MgYW5kIGRldmVsb3BtZW50IHRvb2xzIHVzZWQgYXJlOg0KMS4gSmF2YQ0KMi4gU3ByaW5nICYgU3ByaW5nIEJvb3QgKGFwcGxpY2F0aW9uIGZyYW1ld29yaykNCjMuIFNwcmluZyBNVkMNCjQuIFNwcmluZyBEYXRhIEpQQSB3aXRoIEhpYmVybmF0ZQ0KNS4gTXlTcWwgKHNxbCBkYXRhYmFzZSkNCjYuIEFwYWNoZSBNYXZlbiAoYnVpbGQgYXV0b21hdGlvbiB0b29sKQ0KNy4gQXBhY2hlIFRvbWNhdCAod2ViIGFwcGxpY2F0aW9uIHNlcnZlcikNCg0KSWYgeW91IHdhbnQgdG8gY2xvbmUgdGhpcyByZXBvIGFuZCBydW4gaXQgb24geW91ciBsb2NhbCBtYWNoaW5lLCB5b3Ugc2hvdWxkIGluc3RhbGwgamF2YSAxNyBvbiB5b3VyIHN5c3RlbS4gQWxzbywgeW91J2xsIGhhdmUgdG8gDQppbnN0YWxsIG1hdmVuIGluIG9yZGVyIHRvIGRvd25sb2FkIGFsbCB0aGUgZGVwZW5kZW5jaWVzIGluIHlvdXIgbG9jYWwgbWFjaGluZS4gWW91IHdvbid0IGJlIG5lZWRpbmcgYSB0b21jYXQgc2VydmVyIGluc3RhbGxlZCBiZWNhdXNlIA0KdGhlcmUgaXMgYW4gaW4tYnVpbGQgc2VydmVyIHByb3ZpZGVkIGJ5IHRoZSBzcHJpbmcgYm9vdCBzdGFydGVyIHBhY2suIEJ5IGRlZmF1bHQsIHRoZSBBUEkgd291bGQgYmUgIHJ1bm5pbmcgb24gbG9jYWxob3N0IHBvcnQgODA5MC4NCkhvd2V2ZXIsIGlmIHRoaXMgcG9ydCBpcyBidXN5IGluIHlvdXIgbWFjaGluZSwgeW91IGNhbiBjaGFuZ2UgdGhlIHBvcnQgbnVtYmVyIGJ5IGdvaW5nIHRvIHNyYy9tYWluL3Jlc291cmNlcy9hcHBsaWNhdGlvbi5wcm9wZXJ0aWVzIA0KYW5kIGNoYW5nZSBzZXJ2ZXIucG9ydD04MDkwIHRvIGFueSBhdmFpbGFibGUgcG9ydC4gSGFwcHkgQ3J1ZGRpbmchISENCg0KSSBoYXZlIHBhc3RlZCBzb21lIG9mIHRoZSBzY3JlZW5zaG90cyBzaG93aW5nIHRoZSB2YXJpb3VzIHVyaSBwYXR0ZXJucyBpbiBhY2NvcmRhbmNlIHRvIHRoZSBvcGVuYXBpIHNwZWNpZmljYXRpb25zOg0KDQoxLiBFbnRpdHkgQ2xhc3Nlcw0KIVsyMDIzLTA0LTIzICgxKV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vNTY5NzY2MzUvMjMzODU5MzM4LTFlYzY0NTgwLTFmM2QtNDFiZC1iOWU1LTYzNmZmMTMxOWY2Ny5wbmcpDQoNCjIuIENSVUQgb3BlcmF0aW9uIFVSSSBwYXR0ZXJuczoNCiFbMjAyMy0wNC0yMyAoMildKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzU2OTc2NjM1LzIzMzg1OTM5My1mM2ZhZTUxYy1mMTRkLTQ4NWEtODU3Zi05NTI5NGZhZjc5OTkucG5nKQ0K readmeEtag: '"011f883516f63ecbbe747abe9022570e3e80bb23"' readmeLastModified: Sun, 23 Apr 2023 19:00:37 GMT repositoryId: 630581426 description: >- Crudapp is a basic Rest API that can perform a bunch of crud operations and it follows all the openapi specifications. The entire API is developed with Java language and Spring, Spring Boot, Spring MVC and Spring Data JPA as its frameworks. created: '2023-04-20T17:30:06Z' updated: '2023-04-23T19:08:02Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: MrigankaGhosh logo: https://avatars.githubusercontent.com/u/56976635?v=4 repoEtag: '"7e2e8c7a8313a62d82464c3913cf9607917d2115cbcb1ea54c228ee89ccb3ec7"' repoLastModified: Sun, 23 Apr 2023 19:08:02 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/czasg/pywss v3: true id: 589a36843824543c6b39b9e4a63c23bb repositoryMetadata: base64Readme: >- <div align='center'>

![pywss](./pywss.png)
  
<br/>
  
![Project status](https://img.shields.io/badge/python-3.6+-green.svg)
![PyPI](https://img.shields.io/pypi/v/pywss?color=green)
![Codecov](https://img.shields.io/codecov/c/github/czasg/pywss?token=JSXIQXY1EQ)
[![GitHub issues](https://img.shields.io/github/issues/czasg/pywss)](https://github.com/czasg/pywss/issues)
[![GitHub issues](https://img.shields.io/github/issues-closed/czasg/pywss)](https://github.com/czasg/pywss/issues-closed)
[![GitHub license](https://img.shields.io/github/license/czasg/pywss)](https://github.com/czasg/pywss/blob/main/LICENSE)
  
<br/>
  
</div>

## Pywss 简介

Pywss（发音 /piːwaɪz/，类似 **p~whys**）是一个轻量级的 Python Web 框架，它基于 Python3.6+ 特性构建。

与 Flask、Django 等主流框架不同的是，Pywss 的底层并没有实现 WSGI 接口协议。
其编程风格也更类似于 Gin、Iris 等框架，因此对于熟悉这些框架的开发者来说，Pywss 是一个非常值得探索的项目。

其关键特性有：
- **简单**：拒绝海量参数，减少心智负担。了解上下文 `pywss.Context` 即刻启程。
- **快速**：引入线程池机制，减少并发场景下线程创建/销毁开销。
- **优雅**：`ctx.next` 真的太优雅了。如果你也和我一样喜欢，那我觉得这件事情，**泰裤辣！！**
- **标准化**：集成了部分 OpenAPI（Swagger）能力，方便开发者快速生成 API 文档并进行调试。
- **支持WebSocket**：开箱即用的 **WebSocket** 能力。
- **接口测试**：开箱即用的 **API 测试模块**，不启动服务也能测试接口功能辣！
- **MCP PRO**：一站式集成 **SSE、StreamHTTP 和 MCPO 协议**，助你轻松构建多 MCP 工具🔥

**_在线文档_** [**_https://czasg.github.io/pywss/_**](https://czasg.github.io/pywss/)

<br/>

## 快速开始

### 1、安装 pywss
```shell
pip3 install pywss
```

### 2、搭建 web 应用    
首先创建 `main.py` 文件，并写入以下代码：
```python
import time
import pywss

def log_handler(ctx: pywss.Context):
    start_time = time.time()
    ctx.next()
    print(
        f"Route: {ctx.route}, "
        f"Method: {ctx.method}, "
        f"Status: {ctx.response_status_code}, "
        f"Time: {time.time() - start_time:.3f}s"
    )

def handler(ctx: pywss.Context):
  ctx.write("hello~")

def main():
    app = pywss.App()
    app.get("/hello", handler)  # curl localhost:8080/hello
    app.any("*", log_handler, handler)  # curl -X POST localhost:8080/hello
    app.run()

if __name__ == '__main__':
    main()
```
接着启动服务:
```shell
python3 main.py
```

至此，一个简单的 web 应用服务就完成了。

### 3、搭建 MCP 应用
要快速构建 MCP 服务，只需继承 `pywss.mcp.MCPServer` 并遵循以下规则：  

**核心约束:**  
- 所有接口方法必须以 **`tool_`** 开头（如 `tool_query_user`）  
- 必须使用 **`@pywss.openapi.docs`** 声明请求参数，且 `request` 必须从 pydantic.BaseModel 继承（自动生成 OpenAPI 文档）  

**请求处理:**    
- 通过 **`ctx.data.req`** 直接获取结构化请求体
- 使用 **`self.handle_success(ctx, data)`** 返回成功响应
- 使用 **`self.handle_error(ctx, code, message)`** 返回标准错误

```python
# coding: utf-8
import pywss
from pydantic import BaseModel
from pywss.mcp import MCPServer

class DomainReq(BaseModel):  # 定义 DomainReq 请求，必须从 pydantic.BaseModel 继承
    domain: str

class DomainMCPServer(MCPServer):  # 定义 DomainMCPServer 服务，必须从 pywss.mcp.MCPServer 继承
    @pywss.openapi.docs(description="获取单个域名服务", request=DomainReq)  # required，工具及其参数说明
    def tool_get_domain(self, ctx: pywss.Context):
        req: DomainReq = ctx.data.req  # 框架已经封装好了请求，可以从 ctx.data.req 直接获取使用，异常请求会被拦截
        self.handle_success(ctx, {  # handle_success 封装了 jsonrpc2.0 输出规范
            "domain": req.domain,
            "color": req.color
        })

class LogReq(BaseModel):
    traceId: str

class LogMCPServer(MCPServer):
    @pywss.openapi.docs(description="获取单个trace日志", request=LogReq)
    def tool_get_trace_log(self, ctx: pywss.Context):
        req: LogReq = ctx.data.req
        self.handle_success(ctx, {
            "traceId": req.traceId,
        })

domainMCPServer = DomainMCPServer()
logMCPServer = LogMCPServer()

app = pywss.App()
app.openapi()  # 开启 OpenAPI 文档
domainMCPServer.mount(app.group("/api/v1/domain"))  # 挂载 MCP 服务，同时指定路由
logMCPServer.mount(app.group("/api/v1/log"))  # 挂载 MCP 服务，同时指定路由
app.run()
```
接着启动服务:
```shell
python3 main.py
```
| 协议类型   | 服务类               | 请求方法 | 端点路径格式                          | 示例路径                              |
|------------|----------------------|----------|---------------------------------------|---------------------------------------|
| **SSE**    | `domainMCPServer`    | `GET`    | `/api/v1/domain/sse`                  | `GET /api/v1/domain/sse`              |
|            | `logMCPServer`       | `GET`    | `/api/v1/log/sse`                     | `GET /api/v1/log/sse`                 |
| **StreamHTTP** | `domainMCPServer` | `POST`   | `/api/v1/domain/mcp`                  | `POST /api/v1/domain/mcp`             |
|            | `logMCPServer`       | `POST`   | `/api/v1/log/mcp`                     | `POST /api/v1/log/mcp`                |
| **MCPO**   | `domainMCPServer`    | `POST`   | `/api/v1/domain/tools/{tool_name}`    | `POST /api/v1/domain/tools/get_domain` |
|            | `logMCPServer`       | `POST`   | `/api/v1/log/tools/{tool_name}`       | `POST /api/v1/log/tools/get_trace_log` |


更多功能见[在线文档](https://czasg.github.io/pywss/)。
  
<br/>

## Activity

![Alt](https://repobeats.axiom.co/api/embed/0647dce0c169ba858b3592938376e41d20dc3e6f.svg "Repobeats analytics image")
 readmeEtag: '"a275b5f3fe20dd41e2dc95638466a97203c72643"' readmeLastModified: Tue, 15 Jul 2025 13:08:43 GMT repositoryId: 214977479 description: 一个轻量级的 Python Web 框架,一站式集成 MCP SSE、StreamHTTP 和 MCPO 协议,助你轻松构建MCP Server🔥 created: '2019-10-14T07:33:17Z' updated: '2025-08-29T15:59:21Z' language: Python archived: false stars: 99 watchers: 2 forks: 19 owner: czasg logo: https://avatars.githubusercontent.com/u/44974561?v=4 license: MIT repoEtag: '"07c0a19b047b6a53a1863c45ced6c66dc551697a9d37554ad5ee45e61fb068ec"' repoLastModified: Fri, 29 Aug 2025 15:59:21 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/zero88/bean.x v3: true id: df9f8fc3d0e3cb58fe42f9b34925e1f4 repositoryMetadata: base64Readme: >- PSBCZWFueAp6ZXJvODgKOnJlcG86IHplcm84OC9iZWFuLngKOmFydGlmYWN0OiBpby5naXRodWIuemVybzg4L2JlYW54Cjpzb25hcktleTogemVybzg4X2JlYW54CgppbWFnZTpodHRwczovL2dpdGh1Yi5jb20ve3JlcG99L2FjdGlvbnMvd29ya2Zsb3dzL2NpLnltbC9iYWRnZS5zdmdbe3JlcG99LGxpbms9aHR0cHM6Ly9naXRodWIuY29tL3tyZXBvfS9hY3Rpb25zL3dvcmtmbG93cy9jaS55bWxdCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL3YvcmVsZWFzZS97cmVwb30/c29ydD1zZW12ZXJbR2l0SHViIHJlbGVhc2UgKGxhdGVzdCBTZW1WZXIpXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL25leHVzL3Ive2FydGlmYWN0fT9zZXJ2ZXI9aHR0cHMlM0ElMkYlMkZvc3Muc29uYXR5cGUub3JnW1NvbmF0eXBlIE5leHVzIChSZWxlYXNlcyldCmltYWdlOmh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vbmV4dXMvcy97YXJ0aWZhY3R9P3NlcnZlcj1odHRwcyUzQSUyRiUyRm9zcy5zb25hdHlwZS5vcmdbU29uYXR5cGUgTmV4dXMgKFNuYXBzaG90cyldCgppbWFnZTpodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD17c29uYXJLZXl9Jm1ldHJpYz1uY2xvY1tMaW5lcyBvZiBDb2RlLGxpbms9aHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD17c29uYXJLZXl9XQppbWFnZTpodHRwczovL3NvbmFyY2xvdWQuaW8vYXBpL3Byb2plY3RfYmFkZ2VzL21lYXN1cmU/cHJvamVjdD17c29uYXJLZXl9Jm1ldHJpYz1jb3ZlcmFnZVtDb3ZlcmFnZSxsaW5rPWh0dHBzOi8vc29uYXJjbG91ZC5pby9kYXNoYm9hcmQ/aWQ9e3NvbmFyS2V5fV0KaW1hZ2U6aHR0cHM6Ly9zb25hcmNsb3VkLmlvL2FwaS9wcm9qZWN0X2JhZGdlcy9tZWFzdXJlP3Byb2plY3Q9e3NvbmFyS2V5fSZtZXRyaWM9c3FhbGVfcmF0aW5nW01haW50YWluYWJpbGl0eSBSYXRpbmcsbGluaz1odHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPXtzb25hcktleX1dCmltYWdlOmh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PXtzb25hcktleX0mbWV0cmljPXJlbGlhYmlsaXR5X3JhdGluZ1tSZWxpYWJpbGl0eSBSYXRpbmcsbGluaz1odHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPXtzb25hcktleX1dCmltYWdlOmh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PXtzb25hcktleX0mbWV0cmljPXNlY3VyaXR5X3JhdGluZ1tTZWN1cml0eSBSYXRpbmcsbGluaz1odHRwczovL3NvbmFyY2xvdWQuaW8vZGFzaGJvYXJkP2lkPXtzb25hcktleX1dCmltYWdlOmh0dHBzOi8vc29uYXJjbG91ZC5pby9hcGkvcHJvamVjdF9iYWRnZXMvbWVhc3VyZT9wcm9qZWN0PXtzb25hcktleX0mbWV0cmljPWFsZXJ0X3N0YXR1c1tRdWFsaXR5IEdhdGUgU3RhdHVzLGxpbms9aHR0cHM6Ly9zb25hcmNsb3VkLmlvL2Rhc2hib2FyZD9pZD17c29uYXJLZXl9XQo= readmeEtag: '"80a51293ea37fd846392a4651a499b2d875d9728"' readmeLastModified: Mon, 24 Jul 2023 11:56:16 GMT repositoryId: 618250421 description: vert.x + Java Bean = bean.x (jsr-380 + schema generator) created: '2023-03-24T04:01:58Z' updated: '2023-07-24T04:37:41Z' language: Kotlin archived: false stars: 0 watchers: 1 forks: 0 owner: zero88 logo: https://avatars.githubusercontent.com/u/10863525?v=4 license: Apache-2.0 repoEtag: '"89b53b20e1f5979f501bba3c15d4ff6ff590fff24797ace2218bce9653885d71"' repoLastModified: Mon, 24 Jul 2023 04:37:41 GMT category: Server foundInMaster: true oldLocations: - https://github.com/zero88/beanx - source: openapi3 tags repository: https://github.com/vanch3d/nvl-slim v3: true repositoryMetadata: base64Readme: >- WyFbQnVpbGQgU3RhdHVzXShodHRwczovL3RyYXZpcy1jaS5vcmcvdmFuY2gzZC9udmwtc2xpbS5zdmc/YnJhbmNoPW1hc3RlciZzdHlsZT1mbGF0KV0oaHR0cHM6Ly90cmF2aXMtY2kub3JnL3ZhbmNoM2QvbnZsLXNsaW0pClshW0dpdEh1YiByZWxlYXNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi90YWcvdmFuY2gzZC9udmwtc2xpbS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vdmFuY2gzZC9udmwtc2xpbS90YWdzKQpbIVtsaWNlbnNlXShodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi9saWNlbnNlL3ZhbmNoM2QvbnZsLXNsaW0uc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL3ZhbmNoM2QvbnZsLXNsaW0vYmxvYi9tYXN0ZXIvTElDRU5TRSkKWyFbV2Vic2l0ZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby93ZWJzaXRlLXVwLWRvd24tZ3JlZW4tcmVkL2h0dHAvbnZsLmNhbHF1ZXMzZC5vcmcuc3ZnP2xhYmVsPW52bC5jYWxxdWVzM2Qub3JnJnN0eWxlPWZsYXQpXShodHRwOi8vbnZsLmNhbHF1ZXMzZC5vcmcpCgpudmwtc2xpbQo9PT09PT09PQoKUGVyc29uYWwvcHJvZmVzc2lvbmFsIHdlYnNpdGUKCgojIyBDaGFuZ2Vsb2cKCiMjIFRPRE8KLSBwcm9qZWN0cyBuZWVkIGEgcHJvcGVyIHN1bW1hcnksIHJhdGhlciB0aGF0IHVzaW5nIGRvdGRvdGRvdCAK readmeEtag: '"e72685aa8db8eb29e3a4b663bd474305bea07905"' readmeLastModified: Tue, 06 Mar 2018 15:46:24 GMT repositoryId: 114134828 description: >- Playing around with twig, PHP Slim and several APIs (Zotero, GitHub, LinkedIn, Piwigo, etc.) to get a research projects mashup created: '2017-12-13T15:05:13Z' updated: '2018-04-04T08:53:51Z' language: HTML archived: false stars: 0 watchers: 0 forks: 0 owner: vanch3d logo: https://avatars.githubusercontent.com/u/2743481?v=4 license: MIT repoEtag: '"b4befeb1ca5b8c8cd4f216731f91540b37b1d82dbdb7ca91c3516b5af5e1022b"' repoLastModified: Wed, 04 Apr 2018 08:53:51 GMT foundInMaster: true category: Description Validators id: 55b295a3a4c029013dee377fddb80118 - source: openapi3 tags repository: https://github.com/phyunsj/web-thing-swagger-page v3: true repositoryMetadata: base64Readme: >- IyBNb3ppbGxhIFdlYlRoaW5ncyAtIE9wZW5BUEkgdmVyc2lvbgoKIyMgW1N3YWdnZXIgRWRpdG9yXShodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1lZGl0b3IpCgpDcmVhdGUvZWRpdCBPcGVuQVBJIHZlcnNpb24gb2YgV2ViIENvbm5lY3RlZCBMYW1wIChFeGFtcGxlIDIpIGZyb20gW01vemlsbGEgV29UXShodHRwczovL2lvdC5tb3ppbGxhLm9yZy93b3QvKQoKR2l0SHViIFBhZ2UgOiBbcGh5dW5zai5naXRodWIuaW8vd2ViLXRoaW5nLXN3YWdnZXItcGFnZV0oaHR0cHM6Ly9waHl1bnNqLmdpdGh1Yi5pby93ZWItdGhpbmctc3dhZ2dlci1wYWdlKSBPbmx5IGBHRVQgYCBtZXRob2QgaXMgcGVybWl0dGVkLgoKPHAgYWxpZ249ImxlZnQiPgo8aW1nIHNyYz0iaHR0cHM6Ly9naXRodWIuY29tL3BoeXVuc2ovd2ViLXRoaW5nLXN3YWdnZXItcGFnZS9ibG9iL21hc3Rlci9pbWFnZXMvd290LW9wZW5hcGktc3dhZ2dlci1lZGl0b3IucG5nIiB3aWR0aD0iNzAwcHgiLz4KPC9wPgoKIyMjIyBJbnN0YWxsICYgUnVuIHN3YWdnZXIgZWRpdG9yCgpgYGAKJCBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL3N3YWdnZXItYXBpL3N3YWdnZXItZWRpdG9yLmdpdAokIGNkIHN3YWdnZXItZWRpdG9yCiQgbnBtIGluc3RhbGwKJCBucG0gcnVuIGJ1aWxkCiQgbnBtIHN0YXJ0IC0+IG9wZW4gdGhlIGJyb3dzZXIgOiBodHRwOi8vbG9jYWxob3N0OjMwMDEKYGBgCgojIyBOb2RlLVJFRCBSRVNUIFNlcnZlcgoKPHAgYWxpZ249ImNlbnRlciI+CjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vcGh5dW5zai93ZWItdGhpbmctc3dhZ2dlci1wYWdlL2Jsb2IvbWFzdGVyL2ltYWdlcy93b3Qtbm9kZS1yZWQucG5nIiB3aWR0aD0iNzUwcHgiLz4KPC9wPgoKCiMjIyMgQ29yZSBub2RlIGNoYW5nZXMgZm9yIHRoaXMgZGVtbwoKLSBgbm9kZXMvY29yZS9pby8yMS1odHRwaW4uanNgIDogY29uc29saWRhdGUgYWxsIGh0dHAgbm9kZXMgaW50byBhIHNpbmdsZSBub2RlIHVzaW5nIGBhbGwoKWAgcm91dGluZyBtZXRob2QgZnJvbSBgZXhwcmVzc2AgKGBSRUQuaHR0cE5vZGVgIGlzIGFuIGluc3RhbmNlIG9mIGBleHByZXNzYCkgaW5zdGVhZCBvZiBjcmVhdGluZyBtdXRpcGxlIGh0dHAgbm9kZXMuIAoKYGBgCmZ1bmN0aW9uIEhUVFBJbihuKSB7CiAgLi4uCiAgdGhpcy5jYWxsYmFjayA9IGZ1bmN0aW9uKHJlcSxyZXMpIHsKICAgICB2YXIgbXNnaWQgPSBSRUQudXRpbC5nZW5lcmF0ZUlkKCk7CiAgICAgcmVzLl9tc2dpZCA9IG1zZ2lkOwogICAgIGlmIChub2RlLm1ldGhvZC5tYXRjaCgvXihwb3N0fGRlbGV0ZXxwdXR8b3B0aW9uc3xwYXRjaCkkLykpIHsKICAgICAgICAgbm9kZS5zZW5kKHtfbXNnaWQ6bXNnaWQscmVxOnJlcSxyZXM6Y3JlYXRlUmVzcG9uc2VXcmFwcGVyKG5vZGUscmVzKSxwYXlsb2FkOnJlcS5ib2R5fSk7CiAgICAgfSBlbHNlIGlmIChub2RlLm1ldGhvZCA9PSAiZ2V0IikgewogICAgICAgICBub2RlLnNlbmQoe19tc2dpZDptc2dpZCxyZXE6cmVxLHJlczpjcmVhdGVSZXNwb25zZVdyYXBwZXIobm9kZSxyZXMpLHBheWxvYWQ6cmVxLnF1ZXJ5fSk7CisgICAgfSBlbHNlIGlmIChub2RlLm1ldGhvZCA9PSAiYWxsIikgIHsKKyAgICAgICAgaWYgKCByZXEubWV0aG9kLm1hdGNoKC9eKFBPU1R8REVMRVRFfFBVVHxPUFRJT05TfFBBVENIKSQvKSkgeworICAgICAgICAgICAgbm9kZS5zZW5kKHtfbXNnaWQ6bXNnaWQscmVxOnJlcSxyZXM6Y3JlYXRlUmVzcG9uc2VXcmFwcGVyKG5vZGUscmVzKSxwYXlsb2FkOnJlcS5ib2R5fSk7CisgICAgICAgIH0gZWxzZSBpZiAocmVxLm1ldGhvZCA9PSAiR0VUIikgeworICAgICAgICAgICAgbm9kZS5zZW5kKHtfbXNnaWQ6bXNnaWQscmVxOnJlcSxyZXM6Y3JlYXRlUmVzcG9uc2VXcmFwcGVyKG5vZGUscmVzKSxwYXlsb2FkOnJlcS5xdWVyeX0pOworICAgICAgICB9CiAgICB9IGVsc2UgeyAKICAgICAgICBub2RlLnNlbmQoe19tc2dpZDptc2dpZCxyZXE6cmVxLHJlczpjcmVhdGVSZXNwb25zZVdyYXBwZXIobm9kZSxyZXMpfSk7CiAgICB9CiAgfTsgICAgICAgICAgCiAgLi4uICAgICAgICAKICBpZiAodGhpcy5tZXRob2QgPT0gImdldCIpIHsKICAuLi4gICAgIAorIH0gZWxzZSBpZiAodGhpcy5tZXRob2QgPT0gImFsbCIpIHsKKyAgICAgUkVELmh0dHBOb2RlLmFsbCh0aGlzLnVybCxjb29raWVQYXJzZXIoKSwuLi4sdGhpcy5jYWxsYmFjayx0aGlzLmVycm9ySGFuZGxlcik7CisgfSAKICAuLi4KfQpgYGAKCi0gYG5vZGVzL2NvcmUvaW8vMjEtaHR0cGluLmh0bWxgIDogYWRkICoqQUxMKiogb3B0aW9uLiAKCmBgYAogIDxkaXYgY2xhc3M9ImZvcm0tcm93Ij4KICAgICAgPGxhYmVsIGZvcj0ibm9kZS1pbnB1dC1tZXRob2QiPjxpIGNsYXNzPSJmYSBmYS10YXNrcyI+PC9pPgogICAgICAgICAgPHNwYW4gZGF0YS1pMThuPSJodHRwaW4ubGFiZWwubWV0aG9kIj48L3NwYW4+PC9sYWJlbD4KICAgICAgPHNlbGVjdCB0eXBlPSJ0ZXh0IiBpZD0ibm9kZS1pbnB1dC1tZXRob2QiIHN0eWxlPSJ3aWR0aDo3MCU7Ij4KICAgICAgPG9wdGlvbiB2YWx1ZT0iZ2V0Ij5HRVQ8L29wdGlvbj4KICAgICAgPG9wdGlvbiB2YWx1ZT0icG9zdCI+UE9TVDwvb3B0aW9uPgogICAgICA8b3B0aW9uIHZhbHVlPSJwdXQiPlBVVDwvb3B0aW9uPgogICAgICA8b3B0aW9uIHZhbHVlPSJkZWxldGUiPkRFTEVURTwvb3B0aW9uPgogICAgICA8b3B0aW9uIHZhbHVlPSJwYXRjaCI+UEFUQ0g8L29wdGlvbj4KKyAgICAgPG9wdGlvbiB2YWx1ZT0iYWxsIj5BTEw8L29wdGlvbj4KICAgICAgPC9zZWxlY3Q+CiAgPC9kaXY+CmBgYAoKc2V0IFVSTCBhcyBge2Jhc2VQYXRofS8qYCB3aXRoICoqQUxMKiogb3B0aW9uLiBiYXNlUGF0aCBpc2AvdGhpbmdzL2xhbXAvdjFgLgoKYGBgClJFRC5odHRwTm9kZS5hbGwoICIvdGhpbmdzL2xhbXAvdjEvKiIgLyogPC0gdGhpcy51cmwgKi8gLCAuLi4sIGhhbmRsZXIpOwogLyogIAogICAvdGhpbmdzL2xhbXAvdjEgICAgICAgPC0gd2lsbCBiZSBpZ25vcmVkCiAgCiAgVGhlIGZvbGxvd2luZyBVUkwgd2lsbCBiZSBhY2NlcHRlZC4KICAgL3RoaW5ncy9sYW1wL3YxLwogICAvdGhpbmdzL2xhbXAvdjEvYWN0aW9ucwogICAvdGhpbmdzL2xhbXAvdjEvcHJvcGVydGllcyAKICovCmBgYAoKLSBgbm9kZXMvY29yZS9jb3JlLzgwLWZ1bmN0aW9uLmpzYCA6IGFsbG93IHRvIHVzZSBgbmVkYmAgbW9kdWxlIGRpcmVjdGx5IGluIGBmdW5jdGlvbmAgbm9kZS4gYWNjZXNzIGAkSE9NRVxhY3Rpb25zLmRiYCBhbmQgYCRIT01FXGV2ZW50cy5kYmAgdG8gZ2VuZXJhdGUgR0VUL1BPU1QgcmVzcG9uc2VzLiBUaGlzIGlzIG5vdCBhbiBpZGVhbCBhcHByb2FjaCBidXQgaXQgaXMgZ29vZCBlbm91Z2ggdG8gdGVzdCBSRVNUIEFQSXMuCgpgYGAKZnVuY3Rpb24gRnVuY3Rpb25Ob2RlKG4pIHsKICAgLi4uCiAgIHZhciBzYW5kYm94ID0gewogICAgICAgICAgICBjb25zb2xlOmNvbnNvbGUsCiAgICAgICAgICAgIHV0aWw6dXRpbCwKICAgICAgICAgICAgQnVmZmVyOkJ1ZmZlciwKICAgICAgICAgICAgRGF0ZTogRGF0ZSwKICAgICAgICAgICAgUkVEOiB7CiAgICAgICAgICAgICAgICB1dGlsOiBSRUQudXRpbAogICAgICAgICAgICB9LAorICAgICAgICAgICAgcmVxdWlyZSA6IHJlcXVpcmUsCiAgIC4uLgp9CmBgYAoKKipFeGFtcGxlKiogOiBmdW5jdGlvbiBub2RlICI1NWQ3NjljNC40NjgzODgiCmBgYApjb25zdCBob21lRGlyID0gcmVxdWlyZSgnb3MnKS5ob21lZGlyKCk7CnZhciBEYXRhc3RvcmUgPSByZXF1aXJlKCduZWRiJyk7CnZhciBkYiA9IG5ldyBEYXRhc3RvcmUoaG9tZURpcisnL2FjdGlvbnMuZGInKTsKZGIubG9hZERhdGFiYXNlKCk7CgovLyBjb252ZXJ0ICJwZW5kaW5nIiB0byAiY29tcGxldGUiIGZvciBtc2cucGF5bG9hZCAoIDwtIHRyYW5zYWNpdGlvbi1pZCkgCmRiLnVwZGF0ZSh7IGhyZWY6IG1zZy5wYXlsb2FkIH0sIHsgJHNldDogeyBzdGF0dXM6ICdjb21wbGV0ZScsdGltZUNvbXBsZXRlZCA6IGRhdGVTdHIrJyAnK3RpbWVTdHIgfSB9LCB7IH0sIGZ1bmN0aW9uIChlcnIsIG51bVJlcGxhY2VkKSB7CiAgICAvLyBEbyBzb21ldGhpbmcgaWYgYW55IAp9KTsKYGBgCgojIyBIb3N0IHN3YWdnZXIgZWRpdG9yIChkaXN0IHZlcnNpb24pIG9uIEdpdEh1YiBQYWdlCgoxLiBjcmVhdGUgYSByZXBvc2l0b3J5CjIuIGdlbmVyYXRlIGB3b3QueWFtbGAKMy4gaW5kZXguaHRtbAoKYGBgCiAgICBjb25zdCBlZGl0b3IgPSBTd2FnZ2VyRWRpdG9yQnVuZGxlKHsKICsgICAgdXJsOiAnd290LnlhbWwnLAogICAgICBkb21faWQ6ICcjc3dhZ2dlci1lZGl0b3InLAogICAgICBsYXlvdXQ6ICdTdGFuZGFsb25lTGF5b3V0JywKICAgICAgcHJlc2V0czogWwogICAgICAgIFN3YWdnZXJFZGl0b3JTdGFuZGFsb25lUHJlc2V0CiAgICAgIF0KICAgIH0pCmBgYAo0LiB1cGxvYWQgYGluZGV4Lmh0bWxgICYgYHdvdC55YW1sYCAmIGBkaXN0LypgCjUuIEdpdEh1YiBSZXBvc2l0b3J5IFNldHRpbmdzCgo8cCBhbGlnbj0ibGVmdCI+CjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vcGh5dW5zai93ZWItdGhpbmctc3dhZ2dlci1wYWdlL2Jsb2IvbWFzdGVyL2ltYWdlcy9naXRodWItcGFnZS1zZXR0aW5nLnBuZyIgd2lkdGg9IjMwMHB4Ii8+CjwvcD4KCjYuIHZpc2l0IGBodHRwczovLzx5b3VyLW5hbWU+LmdpdGh1Yi5pby88cmVwb3NpdG9yeS1uYW1lPmAKCiMjIyMgIFJlbGF0ZWQgUG9zdHMKCi0gW01vemlsbGEgV2ViVGhpbmcgQVBJIFNwZWNpZmljYXRpb25dKGh0dHBzOi8vaW90Lm1vemlsbGEub3JnL3dvdC8pCi0gW0RvY3VtZW50aW5nIEFQSXM6IEEgZ3VpZGUgZm9yIHRlY2huaWNhbCB3cml0ZXJzIGFuZCBlbmdpbmVlcnNdKGh0dHBzOi8vaWRyYXRoZXJiZXdyaXRpbmcuY29tL2xlYXJuYXBpZG9jLykKLSBbU3dhZ2dlcl0oaHR0cHM6Ly9zd2FnZ2VyLmlvLykKLSBbc3dhZ2dlci1lZGl0b3JdKGh0dHBzOi8vZ2l0aHViLmNvbS9zd2FnZ2VyLWFwaS9zd2FnZ2VyLWVkaXRvcikKLSBbQXN5bmNBUEkgLSBNZXNzc2FnZSBEcml2ZSBBUEldKGh0dHBzOi8vd3d3LmFzeW5jYXBpLmNvbS8pCi0gW1J1biBDaHJvbWUgYnJvd3NlciB3aXRob3V0IENPUlNdKGh0dHBzOi8vYWxmaWxhdG92LmNvbS9wb3N0cy9ydW4tY2hyb21lLXdpdGhvdXQtY29ycy8pIDogZGlzYWJsZSBDT1JTIGlmIHlvdSB3aXNoIHRvIHJ1biBzd2FnZ2VyLWVkaXRvciAmIE5vZGUtUkVEIHNlcnZlciBvbiB0aGUgc2FtZSBtYWNoaW5lIHRvIHByZXZlbnQgZmV0Y2ggZXJyb3IuCi0gW25lZGIgOiBKU09OIGRhdGFiYXNlXShodHRwczovL2dpdGh1Yi5jb20vbG91aXNjaGF0cmlvdC9uZWRiKQo= readmeEtag: '"c509b97c403bfd405b6ba633a31b81bbffa108bf"' readmeLastModified: Wed, 03 Apr 2019 21:14:36 GMT repositoryId: 178950853 description: Mozilla WebThing Specification - OpenAPI version created: '2019-04-01T21:38:05Z' updated: '2023-01-28T11:52:17Z' language: HTML archived: true stars: 0 watchers: 1 forks: 0 owner: phyunsj logo: https://avatars.githubusercontent.com/u/3989465?v=4 repoEtag: '"c6d7d1d5150b6a0516056b20cf215dee1cc30f3cd2cbcea5c6662334a882a776"' repoLastModified: Sat, 28 Jan 2023 11:52:17 GMT foundInMaster: true category: Data Validators id: 23d57efc8ee051d7d995380e2300b6e2 - source: openapi3 tags repository: https://github.com/neidigsi/mvm1914_backend v3: true repositoryMetadata: base64Readme: >- IVtNdXNpa3ZlcmVpbiAxOTE0IE3DvG5zdGVyIGUuIFYuXShyZWFkbWUvYmFubmVyLnBuZykKIyBXb3JkcHJlc3MtUkVTVC1BUEkgfCBNdXNpa3ZlcmVpbiAxOTE0IE3DvG5zdGVyIGUuIFYuClRoZSBtdXNpYyBjbHViICJNdXNpa3ZlcmVpbiAxOTE0IE3DvG5zdGVyIGUuIFYuIiBwcm92aWRlcyBpdHMgbWVtYmVycyBhbmQgaW50ZXJlc3RlZCBwZXJzb25zIHNpbmNlIDIwMTkgYW4gb3duIGFuZHJvaWQgYXBwLiAKVGhpcyByZXBvc2l0b3J5IGNvbnRhaW5zIHRoZSBjdXN0b20gZGV2ZWxvcGVkIHdvcmRwcmVzcy1wbHVnaW4gd2hpY2ggcHJvdmlkZXMgYSByZXN0LWFwaSBmb3IgdGhpcyBtZW50aW9uZWQgYXBwLiAKVGhlIGFuZHJvaWQgYXBwIGNhbiBiZSBkb3dubG9hZGVkIGZyb20gZ29vZ2xlIHBsYXkgYXQgdGhlIGZvbGxvd2luZyBsaW5rOiA8cD4KW0dvb2dsZSBQbGF5IFN0b3JlXShodHRwczovL3BsYXkuZ29vZ2xlLmNvbS9zdG9yZS9hcHBzL2RldGFpbHM/aWQ9ZGUubXZtMTkxNC5tdXNpa3ZlcmVpbjE5MTRtbnN0ZXJldikKCkluIHRoZSBjdXJyZW50IHZlcnNpb24sIHRoZSBhcHAgYWxsb3dzIHVzZXJzIHRvIGluZm9ybSB0aGVtc2VsdmVzIGFib3V0IG91ciBhc3NvY2lhdGlvbiB0aHJvdWdoIGN1cnJlbnQgYXJ0aWNsZXMgYW5kIGV2ZW50cywgYXMgd2VsbCBhcyBhbiB1cC10by1kYXRlIGdhbGxlcnkuCgojIyBJbnN0YWxsYXRpb24KVG8gaW5zdGFsbCB0aGlzIFdvcmRwcmVzcy1QbHVnaW4gaXQgaXMgbmVjZXNzYXJ5IHRvIGRvd25sb2FkIHRoZSBwcm9ncmFtIGNvZGUgb2YgdGhpcyByZXBvc2l0b3J5IGF0IGZpcnN0LiA8cD4gSW4gdGhlIHNlY29uZCBzdGVwLCB0aGUgZG93bmxvYWRlZApmb2xkZXIgd2l0aCB0aGUgY29kZSBuZWVkcyB0byBiZSBtb3ZlZCB0byB0aGUgZGlyZWN0b3J5ICIvd3AtY29udGVudC9wbHVnaW5zIiBvZiB0aGUgV29yZHByZXNzLUluc3RhbmNlIChlLmcuIHZpYSBhbiBGVFAgY29ubmVjdGlvbikuIDxwPgpGaW5hbGx5LCB0aGUgcGx1Z2luIG5hbWVkICJSRVNUZnVsIEFQSSAtIE11c2lrdmVyZWluIDE5MTQgTcO8bnN0ZXIgZS4gVi4iIG5lZWRzIHRvIGJlIGFjdGl2YXRlZCBpbiB0aGUgc2VjdGlvbiAiUGx1Z2lucyIgb2YgV29yZHByZXNzJ3MgYWRtaW5pc3RyYXRpb24gaW50ZXJmYWNlLgoKIyMgRG9jdW1lbnRhdGlvbgpUaGUgQVBJIGRvY3VtZW50YXRpb24gaXMgd3JpdHRlbiBpbiBPcGVuQVBJIDMgYW5kIGNvdWxkIGJlIGZvdW5kIGluIHRoZSBmb2xkZXIgImRvY3VtZW50YXRpb24iLg== readmeEtag: '"233cfb61ab64d72cfc2d373e19f3faf5ae988e99"' readmeLastModified: Sat, 07 Nov 2020 11:10:06 GMT repositoryId: 247324473 description: >- The music club "Musikverein 1914 Münster e. V." provides its members and interested persons since 2019 an own android app. This repository contains the custom developed wordpress-plugin which provides a rest-api for this mentioned app. The android app can be downloaded from google play. created: '2020-03-14T17:43:07Z' updated: '2021-02-03T13:41:10Z' language: PHP archived: false stars: 0 watchers: 0 forks: 0 owner: neidigsi logo: https://avatars.githubusercontent.com/u/19466013?v=4 repoEtag: '"3b383f9a7e9222d7b5e5b11bef6103e85cacec19eb1a4312db089ebb5006b7ec"' repoLastModified: Wed, 03 Feb 2021 13:41:10 GMT foundInMaster: true category: - Server - Server Implementations id: 74c77a3f133c4710e01284db66670112 - source: openapi3 tags repository: https://github.com/kamo1010/openapigen v3: true repositoryMetadata: repositoryId: 264130346 description: >- This is a sample of how to generate the OpenApi documentation for a RESTful web service implemented in Java 8 created: '2020-05-15T07:46:33Z' updated: '2020-08-26T19:00:11Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: kamo1010 logo: https://avatars.githubusercontent.com/u/43334955?v=4 repoEtag: '"270afac91ef746d8afad708bef78cda99a0b2339766ddf75b3c55dbac0a3ec07"' repoLastModified: Wed, 26 Aug 2020 19:00:11 GMT foundInMaster: true id: 4f9ddc3292fce5d9b3614e500017beb5 oldLocations: - https://github.com/i-mo-k/openapigen - source: openapi3 tags repository: https://github.com/apprexp/govsg2oas v3: true repositoryMetadata: base64Readme: >- IyBHb3ZTRy0yLU9BUwoKPiAqKlRoaXMgcHJvamVjdCBpcyB1bmRlciBoZWF2eSBkZXZlbG9wbWVudC4gRXhwZWN0IG1hbnkgYnJlYWtpbmcgY2hhbmdlcy4qKgoKIVtHaXRIdWJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvYXBwcmV4cC9nb3ZzZzJvYXM/c3R5bGU9ZmxhdC1zcXVhcmUpClshW0NvbnRyaWJ1dG9yIENvdmVuYW50XShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0NvbnRyaWJ1dG9yJTIwQ292ZW5hbnQtdjEuNCUyMGFkb3B0ZWQtZmY2OWI0LnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSldKGNvZGUtb2YtY29uZHVjdC5tZCkKWyFbc2VtYW50aWMtcmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS8lMjAlMjAlRjAlOUYlOTMlQTYlRjAlOUYlOUElODAtc2VtYW50aWMtLXJlbGVhc2UtZTEwMDc5LnN2Zz9zdHlsZT1mbGF0LXNxdWFyZSldKGh0dHBzOi8vZ2l0aHViLmNvbS9zZW1hbnRpYy1yZWxlYXNlL3NlbWFudGljLXJlbGVhc2UpCiFbT1NTIExpZmVjeWNsZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9vc3NsaWZlY3ljbGUvYXBwcmV4cC9nb3ZzZzJvYXM/c3R5bGU9ZmxhdC1zcXVhcmUpClshW0ZPU1NBIFN0YXR1c10oaHR0cHM6Ly9hcHAuZm9zc2EuaW8vYXBpL3Byb2plY3RzL2dpdCUyQmdpdGh1Yi5jb20lMkZhcHByZXhwJTJGZ292c2cyb2FzLnN2Zz90eXBlPXNoaWVsZCldKGh0dHBzOi8vYXBwLmZvc3NhLmlvL3Byb2plY3RzL2dpdCUyQmdpdGh1Yi5jb20lMkZhcHByZXhwJTJGZ292c2cyb2FzP3JlZj1iYWRnZV9zaGllbGQpCgpUaGUgR292U0ctMi1PQVMgYWltcyB0byBkb2N1bWVudCBwdWJsaWNseS1hY2Nlc3NpYmxlIGdvdmVybm1lbnQgQVBJcyBpbnRvIGEgd2VsbC1lc3RhYmxpc2hlZCBmb3JtYXQ6IE9wZW5BUEkgMy4KClRoaXMgaXMgdG8gZW5jb3VyYWdlIGF1dG9tYXRpYyBjb2RlIGdlbmVyYXRpb24gYW5kIGluY3JlYXNlZCByZWFkYWJpbGl0eSBvZiBBUEkgZG9jdW1lbnRhdGlvbiBhY3Jvc3MgZ292ZXJubWVudCBhZ2VuY2llcy4KCiMjIEZBUQoKIyMjIENhbiB5b3UgYWRkIERhdGEuZ292LnNnIGR5bmFtaWMgZGF0YXNldHM/CgpDdXJyZW50bHkgbm8uIFRoZXJlJ3MgYWxyZWFkeSBhbiBbb2ZmaWNpYWwgT3BlbkFQSSBTcGVjaWZpY2F0aW9uIChPQVMpIDMuMC4wXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vZGF0YWdvdnNnL2RhdGFnb3ZzZy1kYXRhc2V0cy9tYXN0ZXIvZG9jdW1lbnRhdGlvbi9zd2FnZ2VyLnltbCkgZm9yIHRob3NlIGRhdGFzZXRzLiBJZiB0aGVyZSdzIGEgbmV3IHJlYXNvbiBmb3IgdXMgdG8gbWFpbnRhaW4gb3VyIG93biB2ZXJzaW9uIChlLmcuIGl0J3Mgbm8gbG9uZ2VyIG1haW50YWluZWQpLCB0aGVuIFtvcGVuIGFuIGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vYXBwcmV4cC9nb3ZzZzJvYXMvaXNzdWVzL25ldykgc28gdGFodCB3ZSBjYW4gbG9vayBpbnRvIGl0LgoKIyMjIENhbiB5b3UgYWRkIFggZGF0YXNldD8KCldlJ3JlIGFsd2F5cyBvcGVuIHRvIGFjY2VwdGluZyBuZXcgZGF0YXNldHMhIFtvcGVuIGEgbmV3IGlzc3VlXShodHRwczovL2dpdGh1Yi5jb20vYXBwcmV4cC9nb3ZzZzJvYXMvaXNzdWVzL25ldykgYW5kIHdlJ2xsIHRha2UgYSBsb29rIGF0IGl0LgoKIyMjIFdoeSBhcmUgeW91IG1haW50YWluaW5nIHRoZXNlPwoKVGhlIG1pc3Npb24gb2YgQXBwUkV4cCBpcyB0byBkZXZlbG9wIHRvb2xzIHRvIHNlcnZlIGEgcHVycG9zZSBvZiBtYWtpbmcgdGhlIGxpdmVzIG9mIHlvdXRocyBpbiBTaW5nYXBvcmUgYmV0dGVyLiBXZSBiZWxpZXZlIHRoYXQgdGhpcyBwcm9qZWN0IHdpbGwgaGVscCB1cyBhY2NvbXBsaXNoIHRoaXMgZ29hbC4KCiMjIyBIb3cgb2Z0ZW4gd2lsbCB5b3UgdXBkYXRlIHRoaXM/CgpXaGlsZSB3ZSBjYW5ub3QgbWFrZSBhbnkgZ3VhcmFudGVlLCB3ZSBwcm9taXNlIGl0IHdpbGwgYmUgdXBkYXRlZCBvbiBhIGJlc3QtZWZmb3J0IGJhc2lzLiBXZSB3aWxsIHVzdWFsbHkgcmVzcG9uZCB0byBuZXcgaXNzdWVzIHdpdGhpbiAzIGRheXMuIE90aGVyd2lzZSwgeW91IG1heSByZWFjaCBvdXQgdG8gW1JpZmEgb24gVHdpdHRlcl0oaHR0cHM6Ly90d2l0dGVyLmNvbS9hY2hyaW56YSkgd2hvc2Ugc3BlYXJoZWFkaW5nIHRoZSBwcm9qZWN0LgoKIyMgQ29udHJpYnV0b3JzCgoqIFtSaWZhIEFjaHJpbnphXShodHRwczovL2dpdGh1Yi5jb20vYWNocmluemEpIChNYWluIGNvbnRyaWJ1dG9yKQoKCiMjIExpY2Vuc2UKWyFbRk9TU0EgU3RhdHVzXShodHRwczovL2FwcC5mb3NzYS5pby9hcGkvcHJvamVjdHMvZ2l0JTJCZ2l0aHViLmNvbSUyRmFwcHJleHAlMkZnb3ZzZzJvYXMuc3ZnP3R5cGU9bGFyZ2UpXShodHRwczovL2FwcC5mb3NzYS5pby9wcm9qZWN0cy9naXQlMkJnaXRodWIuY29tJTJGYXBwcmV4cCUyRmdvdnNnMm9hcz9yZWY9YmFkZ2VfbGFyZ2Up readmeEtag: '"214f549f851d5662d5f6228397936ba05992d1a3"' readmeLastModified: Sun, 10 May 2020 17:18:57 GMT repositoryId: 222053059 description: >- A project which aims to convert existing government public-facing APIs into the OpenAPI 3 specs. created: '2019-11-16T05:42:25Z' updated: '2020-05-10T17:19:01Z' language: TypeScript archived: false stars: 0 watchers: 0 forks: 2 owner: apprexp logo: https://avatars.githubusercontent.com/u/34160960?v=4 license: Apache-2.0 repoEtag: '"579185c0807f638e08c7d70a034bd6336b330a53e2b2d604e3e41227ea2e4d87"' repoLastModified: Sun, 10 May 2020 17:19:01 GMT foundInMaster: true category: Server id: 18443aee0439fc15022dfa20fc623dc9 - source: openapi3 tags repository: https://github.com/sam-kruglov/basic v3: true repositoryMetadata: base64Readme: >- IyBCYXNlClRoaXMgaXMgYSBiYXNlL3RlbXBsYXRlIGZvciBhbnkgYXBwIHRoYXQgcmVxdWlyZXMgdXNlci1yb2xlLWJhc2VkIHNlY3VyaXR5LgpJdCBpcyBidWlsdCBvbiB0b3Agb2YgU3ByaW5nIEJvb3QgMi40IHVzaW5nIHRoZSBmb2xsb3dpbmcgdGVjaDoKLSBTUUwgZGF0YWJhc2UgKEgyKQotIHNjaGVtYSB2ZXJzaW9uaW5nIChGbHl3YXkpCi0gT1JNIChIaWJlcm5hdGUpCi0gY2FjaGUgKEhhemVsY2FzdCkKLSBzZXJ2bGV0cyAoU3ByaW5nIE1WQykKLSBKV1Qgc2VjdXJpdHkgKFNwcmluZyBTZWN1cml0eSBhbmQgYSBjb3VwbGUgb2YgU3ByaW5nIE9BdXRoMiBkZXBlbmRlbmNpZXMpCi0gYXV0b21hdGljIE9wZW5BUEkgdjMgZG9jdW1lbnRhdGlvbiAoU3ByaW5nRG9jKSArIFVJIChTd2FnZ2VyKSArIGdlbmVyYXRlZCBBUEkgQ2xpZW50IChvcGVuYXBpLWdlbmVyYXRvcikKLSBpbnRlZ3JhdGlvbiB0ZXN0cyAobm90aGluZyBtb2NrZWQpIG9uIHRoZSBBUEkgbGV2ZWwgdXNpbmcgdGhlIGdlbmVyYXRlZCBjbGllbnQ= readmeEtag: '"ea43fd27ffd41fe894bf9a99d754baa55b19ce0d"' readmeLastModified: Tue, 22 Dec 2020 11:13:56 GMT repositoryId: 313208050 description: Template for role-based secured backend created: '2020-11-16T06:15:40Z' updated: '2021-06-28T17:16:23Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: Sam-Kruglov logo: https://avatars.githubusercontent.com/u/23556336?v=4 repoEtag: '"6539af2ae7ddb39f5fd4200d0684b909683da5faa0e7f2de2650f694277549aa"' repoLastModified: Mon, 28 Jun 2021 17:16:23 GMT foundInMaster: true category: Server Implementations id: c24e0a659c09346626ade2fee44c8fb7 - source: openapi3 tags repository: https://github.com/vanathin/dubai-task v3: true repositoryMetadata: base64Readme: >- IyBNaWNyb3NlcnZpY2U6IGNvbnN1bWVyLXByb2R1Y2VyLXNlcnZpY2UKVGhpcyBpcyBhIGRlbW8gbWljcm9zZXJ2aWNlIGZvciB0aGUgY29uc3VtZXJzLXByb2R1Y2VycyBjdXJyZW50bHkgYWNjZXNzaW5nIGF0b21pYyBjb3VudGVyLgoKIyMjIFRlY2ggc3RhY2sgdXNlZDoKMS4gSmF2YSA4CjIuIFNwcmluZyBCb290IDIuNC4yCjMuIFNwcmluZyBSRVNUIEFQSSAoU3ByaW5nIFdlYikKNC4gTG9tYm9rIDEuMTguMTYKNS4gU3ByaW5nIERhdGEgSlBBICYgTXlTUUwtOAo3LiBEb2NrZXJpemVkIChkb2NrZXItY29tcG9zZS55bWwgYW5kIERvY2tlcmZpbGUgaXMgaW5jbHVkZWQpCjguIFRERCBhcHByb2FjaCB1c2luZyBKVW5pdCA1LCBNb2NraXRvLCBhbmQgU3ByaW5nIEJvb3QgVGVzdAoxMC4gU3ByaW5nIGJvb3Qgc3RhcnRlciB2YWxpZGF0aW9uIGZvciByZXF1ZXN0IHZhbGlkYXRpb24gIAoKIyMjIFN0ZXBzIHRvIHJ1biBtaWNyby1zZXJ2aWNlIGFsb25nIHdpdGggYE15U1FMYCBhcyBjb250YWluZXJzIGluIGRvY2tlciB1c2luZyBgZG9ja2VyIGNvbXBvc2VgOgojIyMjIyBTdGVwIDE6IE9wZW4gdGhlIHRlcm1pbmFsIChvcikgQ29tbWFuZCBQcm9tcHQKIyMjIyMgU3RlcCAyOiBFbnRlciB0aGUgcHJvamVjdCBkaXJlY3RvcnksIAogICAgY2QgPHByb2plY3RfZGlyZWN0b3J5PgoKIyMjIyMgU3RlcCAzOiBUbyBjbGVhbiBhbmQgcGFja2FnZSB0aGUgbWljcm8tc2VydmljZSBsb2NhbGx5CiAgICBtdm53IGNsZWFuIHBhY2thZ2UKCiMjIyMjIFN0ZXAgNDogVXNlIGBkb2NrZXIgY29tcG9zZWAgdG8gYnVpbGQgYW5kIHJ1biBib3RoIE15U1FMICYgbWljcm8tc2VydmljZSBhcyBhIGRvY2tlciBjb250YWluZXJzOgogICAgZG9ja2VyLWNvbXBvc2UgdXAgLS1idWlsZApOb3RlOiBBYm92ZSBkb2NrZXIgY29tcG9zZSBjb21tYW5kIHdpbGwgcnVuIHRoZSBjb250YWluZXJzIG5hbWVseSwKICAgMS4gYGVrYXItbXlzcWxgIGNvbnRhaW5lciBmb3IgbXlzcWwgYW5kIGl0cyBwb3J0IGlzIGAzMzA2YAogICAyLiBgZWthci1wcm9kdWNlci1jb25zdW1lci1zZXJ2aWNlYCBjb250YWluZXIgZm9yIG1pY3Jvc2VydmljZSBhbmQgaXRzIHBvcnQgaXMgYDgwODBgCgotLS0KCiMjIyBPcGVuQVBJIDMuMCBEb2N1bWVudGF0aW9uIFVSTDoKCmh0dHA6Ly9sb2NhbGhvc3Q6ODA4MC9zd2FnZ2VyLXVpL2luZGV4Lmh0bWw/Y29uZmlnVXJsPS92My9hcGktZG9jcy9zd2FnZ2VyLWNvbmZpZyAgCiFbQVBJX0RvY3VtZW50YXRpb24gaW1nIG1pc3NpbmddKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS92YW5hdGhpbi9la2FyLXByb2R1Y2VyLWNvbnN1bWVyLXNlcnZpY2UvbWFpbi9pbWcvc3dhZ2dlci11aS5qcGcpCk5vdGU6IEFsc28gW1Bvc3RtYW4gQ29sbGVjdGlvbl0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3ZhbmF0aGluL2VrYXItcHJvZHVjZXItY29uc3VtZXItc2VydmljZS9tYWluL0VrYXIucG9zdG1hbl9jb2xsZWN0aW9uLmpzb24pIGhhcyBiZWVuIHB1c2hlZCBpbnRvIHRoZSByZXBvc2l0b3J5IGZvciB5b3VyIHJlZmVyZW5jZS4KCi0tLQoKIyMjIE15U1FMIERhdGFiYXNlIENMSSB0byB2ZXJpZnkgdGhlIHRyYW5zYWN0aW9uIGluZm9ybWF0aW9uOgojIyMjIyBQcmVyZXF1aXNpdGU6IApEb2NrZXIgaGFzIHRvIGJlIHRoZXJlIG9uIHRoZSBob3N0IG1hY2hpbmUKIyMjIyMgU3RlcCAxOiBPcGVuIGEgbmV3IHRlcm1pbmFsIChvcikgbmV3IENvbW1hbmQgUHJvbXB0ICh3aXRob3V0IGRpc3R1cmJpbmcgdGhlIGVhcmxpZXIgb25lKQojIyMjIyBTdGVwIDI6IENvbm5lY3QgdG8gYGVrYXItbXlzcWxgIGNvbnRhaW5lciB1c2luZyBkb2NrZXIgY2xpIGNvbW1hbmQsIAogICAgZG9ja2VyIGV4ZWMgLWl0IGVrYXItbXlzcWwgL2Jpbi9iYXNoCiMjIyMjIFN0ZXAgMzogQ29ubmVjdCB0byBteXNxbCBzZXJ2ZXIKICAgIG15c3FsIC11c2EgLXBwYXNzd29yZCAtaCBsb2NhbGhvc3QgLVAzMzA2CiMjIyMjIFN0ZXAgNDogRGlzcGxheSBhbGwgdGhlIGF2YWlsYWJsZSBkYXRhYmFzZXMsCiAgICBzaG93IGRhdGFiYXNlczsKIyMjIyMgU3RlcCA1OiBTd2l0Y2ggdG8gZWthcmRiCiAgICB1c2UgZWthcmRiOwojIyMjIyBTdGVwIDY6IExpc3QgYWxsIHRoZSBhdmFpbGFibGUgdGFibGVzCiAgICBzaG93IHRhYmxlczsKIyMjIyMgU3RlcCA3OiBSdW4gdGhlIHNlbGVjdCBxdWVyaWVzIHRvIHZlcmlmeSB0aGUgZGF0YQogICAgc2VsZWN0ICogZnJvbSByZXF1ZXN0X2xvZzsgc2VsZWN0ICogZnJvbSBjb3VudGVyX2xvZzsKTm90ZTogUHJlc2VydmUgdGhpcyBzdGF0ZSB1bnRpbCB5b3UgY29tcGxldGUgeW91ciB0ZXN0aW5nLgojIyMjIyBTdGVwIDg6IEV4aXQgZnJvbSB0aGUgbXlzcWwgY2xpCiAgICBleGl0CiMjIyMjIFN0ZXAgOTogRXhpdCBmcm9tIHRoZSBteXNxbCBjb250YWluZXIKICAgIGV4aXQKCiFbTXlTUUwgQ0xJIE91dHB1dF0oaHR0cHM6Ly9yYXcuZ2l0aHVidXNlcmNvbnRlbnQuY29tL3ZhbmF0aGluL2VrYXItcHJvZHVjZXItY29uc3VtZXItc2VydmljZS9tYWluL2ltZy9teXNxbF9kYl9sb2cuanBnKQoKLS0tCgoKIyMjIFByb2dyYW0gb3V0cHV0cwoKIyMjIyBMZXNzIGNvbnN1bWVycyBhbmQgbW9yZSBwcm9kdWNlcnMKIVtMZXNzIGNvbnN1bWVycyBhbmQgbW9yZSBwcm9kdWNlcnNdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS92YW5hdGhpbi9la2FyLXByb2R1Y2VyLWNvbnN1bWVyLXNlcnZpY2UvbWFpbi9pbWcvTW9yZV9Qcm9kdWNlcl9MZXNzX0NvbnN1bWVyLkpQRykKCiMjIyMgTW9yZSBjb25zdW1lcnMgYW5kIGxlc3MgcHJvZHVjZXJzCiFbTW9yZSBjb25zdW1lcnMgYW5kIGxlc3MgcHJvZHVjZXJzXShodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vdmFuYXRoaW4vZWthci1wcm9kdWNlci1jb25zdW1lci1zZXJ2aWNlL21haW4vaW1nL0xlc3NfUHJvZHVjZXJfTW9yZV9Db25zdW1lci5KUEcpCgojIyMjIEVxdWFsIGNvbnN1bWVycyBhbmQgcHJvZHVjZXJzCiFbRXF1YWwgY29uc3VtZXJzIGFuZCBwcm9kdWNlcnNdKGh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS92YW5hdGhpbi9la2FyLXByb2R1Y2VyLWNvbnN1bWVyLXNlcnZpY2UvbWFpbi9pbWcvRXF1YWxfUHJvZHVjZXJfQ29uc3VtZXIuSlBHKQoKLS0tCg== readmeEtag: '"b36de953644fc2470e15a5a8737b8f1c45aaba09"' readmeLastModified: Mon, 18 Jan 2021 00:47:53 GMT repositoryId: 330501920 description: Demo project for producer & consumer with counters, created: '2021-01-17T22:46:03Z' updated: '2021-01-29T14:43:46Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: vanathin logo: https://avatars.githubusercontent.com/u/10528013?v=4 repoEtag: '"be51ad555c2ee012cd42f44f7e2e898bd9d336442fcb12b59bf6cd1ee55ebab9"' repoLastModified: Fri, 29 Jan 2021 14:43:46 GMT foundInMaster: true category: - Code Generators - Server Implementations id: 98c89ef6075b3eb38e72cd151757380a - source: IMPLEMENTATIONS.md name: swagger-models homepage: >- https://github.com/swagger-api/swagger-core/tree/master/modules/swagger-models language: Java source_description: OpenAPI 3.0 Java Pojos category: - Low-level Tooling - Code Generators foundInMaster: true repository: >- https://github.com/swagger-api/swagger-core/tree/master/modules/swagger-models repositoryMetadata: base64Readme: >- # Swagger Core <img src="https://raw.githubusercontent.com/swagger-api/swagger.io/wordpress/images/assets/SW-logo-clr.png" height="50" align="right">

**NOTE:** If you're looking for Swagger Core 1.5.X and OpenAPI 2.0, please refer to [1.5 branch](https://github.com/swagger-api/swagger-core/tree/1.5).

**NOTE:** Since version 2.1.7, Swagger Core also supports the Jakarta namespace. There are a parallel set of artifacts with the `-jakarta` suffix, providing the same functionality as the unsuffixed (i.e.: `javax`) artifacts.
Please see the [Wiki](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---Getting-started) for more details.

**NOTE:** Since version 2.2.0 Swagger Core supports OpenAPI 3.1; see [this page](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---OpenAPI-3.1) for details

![Build Test Deploy](https://github.com/swagger-api/swagger-core/workflows/Build%20Test%20Deploy%20master/badge.svg?branch=master)
[![Maven Central](https://maven-badges.herokuapp.com/maven-central/io.swagger.core.v3/swagger-project/badge.svg?style=plastic)](https://maven-badges.herokuapp.com/maven-central/io.swagger.core.v3/swagger-project)

Swagger Core is a Java implementation of the OpenAPI Specification. Current version supports *JAX-RS2* (`javax` and `jakarta` namespaces).

## Get started with Swagger Core!
See the guide on [getting started with Swagger Core](https://github.com/swagger-api/swagger-core/wiki/Swagger-2.X---Getting-started) to get started with adding Swagger to your API.

## See the Wiki!
The [github wiki](https://github.com/swagger-api/swagger-core/wiki) contains documentation, samples, contributions, etc. Start there.

## Compatibility
The OpenAPI Specification has undergone several revisions since initial creation in 2010.  The Swagger Core project has the following compatibilities with the OpenAPI Specification:

Swagger core Version      | Release Date | OpenAPI Spec compatibility | Notes | Status
------------------------- | ------------ | -------------------------- | ----- | ----
2.2.42 (**current stable**)| 2026-01-19   | 3.x           | [tag v2.2.42](https://github.com/swagger-api/swagger-core/tree/v2.2.42) | Supported
2.2.41                     | 2025-11-24   | 3.x           | [tag v2.2.41](https://github.com/swagger-api/swagger-core/tree/v2.2.41) | Supported
2.2.40                     | 2025-10-28   | 3.x           | [tag v2.2.40](https://github.com/swagger-api/swagger-core/tree/v2.2.40) | Supported
2.2.39                     | 2025-10-13   | 3.x           | [tag v2.2.39](https://github.com/swagger-api/swagger-core/tree/v2.2.39) | Supported
2.2.38                     | 2025-09-29   | 3.x           | [tag v2.2.38](https://github.com/swagger-api/swagger-core/tree/v2.2.38) | Supported
2.2.37                     | 2025-09-16   | 3.x           | [tag v2.2.37](https://github.com/swagger-api/swagger-core/tree/v2.2.37) | Supported
2.2.36                     | 2025-08-18   | 3.x           | [tag v2.2.36](https://github.com/swagger-api/swagger-core/tree/v2.2.36) | Supported
2.2.35                     | 2025-07-31   | 3.x           | [tag v2.2.35](https://github.com/swagger-api/swagger-core/tree/v2.2.35) | Supported
2.2.34                     | 2025-06-20   | 3.x           | [tag v2.2.34](https://github.com/swagger-api/swagger-core/tree/v2.2.34) | Supported
2.2.33                     | 2025-06-12   | 3.x           | [tag v2.2.33](https://github.com/swagger-api/swagger-core/tree/v2.2.33) | Supported
2.2.32                     | 2025-05-14   | 3.x           | [tag v2.2.32](https://github.com/swagger-api/swagger-core/tree/v2.2.32) | Supported
2.2.31                     | 2025-05-13   | 3.x           | [tag v2.2.31](https://github.com/swagger-api/swagger-core/tree/v2.2.31) | Supported
2.2.30                     | 2025-04-07   | 3.x           | [tag v2.2.30](https://github.com/swagger-api/swagger-core/tree/v2.2.30) | Supported
2.2.29                     | 2025-03-10   | 3.x           | [tag v2.2.29](https://github.com/swagger-api/swagger-core/tree/v2.2.29) | Supported
2.2.28                     | 2025-01-16   | 3.x           | [tag v2.2.28](https://github.com/swagger-api/swagger-core/tree/v2.2.28) | Supported
2.2.27                     | 2024-12-11   | 3.x           | [tag v2.2.27](https://github.com/swagger-api/swagger-core/tree/v2.2.27) | Supported
2.2.26                     | 2024-11-18   | 3.x           | [tag v2.2.26](https://github.com/swagger-api/swagger-core/tree/v2.2.26) | Supported
2.2.25                     | 2024-10-02   | 3.x           | [tag v2.2.25](https://github.com/swagger-api/swagger-core/tree/v2.2.25) | Supported
2.2.24                     | 2024-09-23   | 3.x           | [tag v2.2.24](https://github.com/swagger-api/swagger-core/tree/v2.2.24) | Supported
2.2.23                     | 2024-08-28   | 3.x           | [tag v2.2.23](https://github.com/swagger-api/swagger-core/tree/v2.2.23) | Supported
2.2.22                     | 2024-05-15   | 3.x           | [tag v2.2.22](https://github.com/swagger-api/swagger-core/tree/v2.2.22) | Supported
2.2.21                     | 2024-03-20   | 3.x           | [tag v2.2.21](https://github.com/swagger-api/swagger-core/tree/v2.2.21) | Supported
2.2.20                     | 2023-12-19   | 3.x           | [tag v2.2.20](https://github.com/swagger-api/swagger-core/tree/v2.2.20) | Supported
2.2.19                     | 2023-11-10   | 3.x           | [tag v2.2.19](https://github.com/swagger-api/swagger-core/tree/v2.2.19) | Supported
2.2.18                     | 2023-10-25   | 3.x           | [tag v2.2.18](https://github.com/swagger-api/swagger-core/tree/v2.2.18) | Supported
2.2.17                     | 2023-10-12   | 3.x           | [tag v2.2.17](https://github.com/swagger-api/swagger-core/tree/v2.2.17) | Supported
2.2.16                     | 2023-09-18   | 3.x           | [tag v2.2.16](https://github.com/swagger-api/swagger-core/tree/v2.2.16) | Supported
2.2.15                     | 2023-07-08   | 3.x           | [tag v2.2.15](https://github.com/swagger-api/swagger-core/tree/v2.2.15) | Supported
2.2.14                     | 2023-06-26   | 3.x           | [tag v2.2.14](https://github.com/swagger-api/swagger-core/tree/v2.2.14) | Supported
2.2.13                     | 2023-06-24   | 3.x           | [tag v2.2.13](https://github.com/swagger-api/swagger-core/tree/v2.2.13) | Supported
2.2.12                     | 2023-06-13   | 3.x           | [tag v2.2.12](https://github.com/swagger-api/swagger-core/tree/v2.2.12) | Supported
2.2.11                     | 2023-06-01   | 3.x           | [tag v2.2.11](https://github.com/swagger-api/swagger-core/tree/v2.2.11) | Supported
2.2.10                     | 2023-05-15   | 3.x           | [tag v2.2.10](https://github.com/swagger-api/swagger-core/tree/v2.2.10) | Supported
2.2.9                     | 2023-03-20  | 3.x           | [tag v2.2.9](https://github.com/swagger-api/swagger-core/tree/v2.2.9)                                             | Supported
2.2.8                     | 2023-01-06  | 3.x           | [tag v2.2.8](https://github.com/swagger-api/swagger-core/tree/v2.2.8)                                             | Supported
2.2.7                     | 2022-11-15  | 3.0           | [tag v2.2.7](https://github.com/swagger-api/swagger-core/tree/v2.2.7)                                             | Supported
2.2.6                     | 2022-11-02  | 3.0           | [tag v2.2.6](https://github.com/swagger-api/swagger-core/tree/v2.2.6)                                             | Supported
2.2.5                     | 2022-11-02  | 3.0           | [tag v2.2.5](https://github.com/swagger-api/swagger-core/tree/v2.2.5)                                             | Supported
2.2.4                     | 2022-10-16  | 3.0           | [tag v2.2.4](https://github.com/swagger-api/swagger-core/tree/v2.2.4)                                             | Supported
2.2.3                     | 2022-09-27  | 3.0           | [tag v2.2.3](https://github.com/swagger-api/swagger-core/tree/v2.2.3)                                             | Supported
2.2.2                     | 2022-07-20  | 3.0           | [tag v2.2.2](https://github.com/swagger-api/swagger-core/tree/v2.2.2)                                             | Supported
2.2.1                     | 2022-06-15  | 3.0           | [tag v2.2.1](https://github.com/swagger-api/swagger-core/tree/v2.2.1)                                             | Supported
2.2.0                     | 2022-04-04  | 3.0           | [tag v2.2.0](https://github.com/swagger-api/swagger-core/tree/v2.2.0)                                             | Supported
2.1.13                     | 2022-02-07  | 3.0           | [tag v2.1.13](https://github.com/swagger-api/swagger-core/tree/v2.1.13)                                           | Supported
2.1.12                     | 2021-12-23  | 3.0           | [tag v2.1.12](https://github.com/swagger-api/swagger-core/tree/v2.1.12)                                           | Supported
2.1.11                     | 2021-09-29  | 3.0           | [tag v2.1.11](https://github.com/swagger-api/swagger-core/tree/v2.1.11)                                           | Supported
2.1.10                     | 2021-06-28  | 3.0           | [tag v2.1.10](https://github.com/swagger-api/swagger-core/tree/v2.1.10)                                           | Supported
2.1.9                     | 2021-04-20  | 3.0           | [tag v2.1.9](https://github.com/swagger-api/swagger-core/tree/v2.1.9)                                             | Supported
2.1.8                     | 2021-04-18  | 3.0           | [tag v2.1.8](https://github.com/swagger-api/swagger-core/tree/v2.1.8)                                             | Supported
2.1.7                     | 2021-02-18  | 3.0           | [tag v2.1.7](https://github.com/swagger-api/swagger-core/tree/v2.1.7)                                             | Supported
2.1.6                     | 2020-12-04  | 3.0           | [tag v2.1.6](https://github.com/swagger-api/swagger-core/tree/v2.1.6)                                             | Supported
2.1.5                     | 2020-10-01  | 3.0           | [tag v2.1.5](https://github.com/swagger-api/swagger-core/tree/v2.1.5)                                             | Supported
2.1.4                     | 2020-07-24  | 3.0           | [tag v2.1.4](https://github.com/swagger-api/swagger-core/tree/v2.1.4)                                             | Supported
2.1.3                     | 2020-06-27  | 3.0           | [tag v2.1.3](https://github.com/swagger-api/swagger-core/tree/v2.1.3)                                             | Supported
2.1.2                     | 2020-04-01  | 3.0           | [tag v2.1.2](https://github.com/swagger-api/swagger-core/tree/v2.1.2)                                             | Supported
2.1.1                     | 2020-01-02  | 3.0           | [tag v2.1.1](https://github.com/swagger-api/swagger-core/tree/v2.1.1)                                             | Supported
2.1.0                     | 2019-11-16  | 3.0           | [tag v2.1.0](https://github.com/swagger-api/swagger-core/tree/v2.1.0)                                             | Supported
2.0.10                    | 2019-10-11  | 3.0           | [tag v2.0.10](https://github.com/swagger-api/swagger-core/tree/v2.0.10)                                           | Supported
2.0.9                     | 2019-08-22  | 3.0           | [tag v2.0.9](https://github.com/swagger-api/swagger-core/tree/v2.0.9)                                             | Supported
2.0.8                     | 2019-04-24  | 3.0           | [tag v2.0.8](https://github.com/swagger-api/swagger-core/tree/v2.0.8)                                             | Supported
2.0.7                     | 2019-02-18  | 3.0           | [tag v2.0.7](https://github.com/swagger-api/swagger-core/tree/v2.0.7)                                             | Supported
2.0.6                     | 2018-11-27  | 3.0           | [tag v2.0.6](https://github.com/swagger-api/swagger-core/tree/v2.0.6)                                             | Supported
2.0.5                     | 2018-09-19  | 3.0           | [tag v2.0.5](https://github.com/swagger-api/swagger-core/tree/v2.0.5)                                             | Supported
2.0.4                     | 2018-09-05  | 3.0           | [tag v2.0.4](https://github.com/swagger-api/swagger-core/tree/v2.0.4)                                             | Supported
2.0.3                     | 2018-08-09  | 3.0           | [tag v2.0.3](https://github.com/swagger-api/swagger-core/tree/v2.0.3)                                             | Supported
1.6.14 (**current stable**)| 2024-03-19   | 2.0           | [tag v1.6.14](https://github.com/swagger-api/swagger-core/tree/v1.6.14)                                           | Supported
1.6.13                    | 2024-01-26   | 2.0           | [tag v1.6.13](https://github.com/swagger-api/swagger-core/tree/v1.6.13)                                           | Supported
1.6.12                    | 2023-10-14   | 2.0           | [tag v1.6.12](https://github.com/swagger-api/swagger-core/tree/v1.6.12)                                           | Supported
1.6.11                    | 2023-05-15  | 2.0           | [tag v1.6.11](https://github.com/swagger-api/swagger-core/tree/v1.6.11)                                           | Supported
1.6.10                    | 2023-03-21  | 2.0           | [tag v1.6.10](https://github.com/swagger-api/swagger-core/tree/v1.6.10)                                           | Supported
1.6.9                     | 2022-11-15  | 2.0           | [tag v1.6.9](https://github.com/swagger-api/swagger-core/tree/v1.6.9)                                             | Supported
1.6.8                     | 2022-10-16  | 2.0           | [tag v1.6.8](https://github.com/swagger-api/swagger-core/tree/v1.6.8)                                             | Supported
1.6.7                     | 2022-09-27  | 2.0           | [tag v1.6.7](https://github.com/swagger-api/swagger-core/tree/v1.6.7)                                             | Supported
1.6.6                     | 2022-04-04  | 2.0           | [tag v1.6.6](https://github.com/swagger-api/swagger-core/tree/v1.6.6)                                             | Supported
1.6.5                     | 2022-02-07  | 2.0           | [tag v1.6.5](https://github.com/swagger-api/swagger-core/tree/v1.6.5)                                             | Supported
1.6.4                     | 2021-12-23  | 2.0           | [tag v1.6.4](https://github.com/swagger-api/swagger-core/tree/v1.6.4)                                             | Supported
1.6.3                     | 2021-09-29  | 2.0           | [tag v1.6.3](https://github.com/swagger-api/swagger-core/tree/v1.6.3)                                             | Supported
1.6.2                     | 2020-07-01  | 2.0           | [tag v1.6.2](https://github.com/swagger-api/swagger-core/tree/v1.6.2)                                             | Supported
1.6.1                     | 2020-04-01  | 2.0           | [tag v1.6.1](https://github.com/swagger-api/swagger-core/tree/v1.6.1)                                             | Supported
1.6.0                     | 2019-11-16  | 2.0           | [tag v1.6.0](https://github.com/swagger-api/swagger-core/tree/v1.6.0)                                             | Supported
1.5.24                    | 2019-10-11  | 2.0           | [tag v1.5.24](https://github.com/swagger-api/swagger-core/tree/v1.5.24)                                           | Supported
1.5.23                    | 2019-08-22  | 2.0           | [tag v1.5.23](https://github.com/swagger-api/swagger-core/tree/v1.5.23)                                           | Supported
1.5.22                    | 2019-02-18  | 2.0           | [tag v1.5.22](https://github.com/swagger-api/swagger-core/tree/v1.5.22)                                           | Supported
1.5.21                    | 2018-08-09  | 2.0           | [tag v1.5.21](https://github.com/swagger-api/swagger-core/tree/v1.5.21)                                           | Supported
1.5.20                    | 2018-05-23  | 2.0           | [tag v1.5.20](https://github.com/swagger-api/swagger-core/tree/v1.5.20)                                           | Supported
2.0.2                     | 2018-05-23  | 3.0           | [tag v2.0.2](https://github.com/swagger-api/swagger-core/tree/v2.0.2)                                             | Supported
2.0.1                     | 2018-04-16  | 3.0           | [tag v2.0.1](https://github.com/swagger-api/swagger-core/tree/v2.0.1)                                             | Supported
1.5.19                    | 2018-04-16  | 2.0           | [tag v1.5.19](https://github.com/swagger-api/swagger-core/tree/v1.5.19)                                           | Supported
2.0.0                     | 2018-03-20  | 3.0           | [tag v2.0.0](https://github.com/swagger-api/swagger-core/tree/v2.0.0)                                             | Supported
2.0.0-rc4                 | 2018-01-22  | 3.0           | [tag v2.0.0-rc4](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc4)                                     | Supported
2.0.0-rc3                 | 2017-11-21  | 3.0           | [tag v2.0.0-rc3](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc3)                                     | Supported
2.0.0-rc2                 | 2017-09-29  | 3.0           | [tag v2.0.0-rc2](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc2)                                     | Supported
2.0.0-rc1                 | 2017-08-17  | 3.0           | [tag v2.0.0-rc1](https://github.com/swagger-api/swagger-core/tree/v2.0.0-rc1)                                     | Supported
1.5.18                    | 2018-01-22  | 2.0           | [tag v1.5.18](https://github.com/swagger-api/swagger-core/tree/v1.5.18)                                           | Supported
1.5.17                    | 2017-11-21  | 2.0           | [tag v1.5.17](https://github.com/swagger-api/swagger-core/tree/v1.5.17)                                           | Supported
1.5.16                    | 2017-07-15  | 2.0           | [tag v1.5.16](https://github.com/swagger-api/swagger-core/tree/v1.5.16)                                           | Supported
1.3.12                    | 2014-12-23  | 1.2           | [tag v1.3.12](https://github.com/swagger-api/swagger-core/tree/v1.3.12)                                           | Supported
1.2.4                     | 2013-06-19  | 1.1           | [tag swagger-project_2.10.0-1.2.4](https://github.com/swagger-api/swagger-core/tree/swagger-project_2.10.0-1.2.4) | Deprecated
1.0.0                     | 2011-10-16  | 1.0           | [tag v1.0](https://github.com/swagger-api/swagger-core/tree/v1.0)                                                 | Deprecated


### Change History
If you're interested in the change history of swagger and the Swagger Core framework, see [here](https://github.com/swagger-api/swagger-core/releases).

### Prerequisites
You need the following installed and available in your $PATH:

* Java 11
* Apache maven 3.0.4 or greater
* Jackson 2.4.5 or greater


### To build from source (currently 2.2.43-SNAPSHOT)
```
# first time building locally
mvn -N
```

Subsequent builds:
```
mvn install
```

This will build the modules.

Of course if you don't want to build locally you can grab artifacts from maven central:

`https://repo1.maven.org/maven2/io/swagger/core/`

## Sample Apps
The samples have moved to [a new repository](https://github.com/swagger-api/swagger-samples/tree/2.0) and contain various integrations and configurations.

## Security contact

Please disclose any security-related issues or vulnerabilities by emailing [security@swagger.io](mailto:security@swagger.io), instead of using the public issue tracker.
 readmeEtag: '"27cbc7c379b4a3e20ecccaaa8cddff40247bffe5"' readmeLastModified: Mon, 19 Jan 2026 14:58:21 GMT repositoryId: 2003641 description: >- Examples and server integrations for generating the Swagger API Specification, which enables easy access to your REST API created: '2011-07-05T23:44:11Z' updated: '2026-02-06T03:05:35Z' language: Java archived: false stars: 7516 watchers: 307 forks: 2249 owner: swagger-api logo: https://avatars.githubusercontent.com/u/7658037?v=4 license: Apache-2.0 repoEtag: '"2d6e76914a17c8cb9b177d2f09ecbdf7fcccb32aefc4c38e0d55c3fef7ddb502"' repoLastModified: Fri, 06 Feb 2026 03:05:35 GMT id: 266d60df2d48ad8a47539de3c1071f3a - source: IMPLEMENTATIONS.md name: Microsoft.OpenApi.net homepage: https://github.com/microsoft/openapi.net/ language: dotnet source_description: C# based parser with definition validation and migration support from V2 category: - Low-level Tooling - Parsers foundInMaster: true repository: https://github.com/microsoft/openapi.net/ repositoryMetadata: base64Readme: >- <!-- using the raw image URL so it displays correctly on nuget.org -->
![Category overview screenshot](https://raw.githubusercontent.com/microsoft/OpenAPI.NET/main/docs/images/oainet.png "Microsoft + OpenAPI = Love")

# OpenAPI.NET

|Package|Nuget|
|--|--|
|Models and Writers|[![nuget](https://img.shields.io/nuget/v/Microsoft.OpenApi.svg)](https://www.nuget.org/packages/Microsoft.OpenApi/) |
|YamlReader | [![nuget](https://img.shields.io/nuget/v/Microsoft.OpenApi.YamlReader.svg)](https://www.nuget.org/packages/Microsoft.OpenApi.YamlReader/) |
|Hidi|[![nuget](https://img.shields.io/nuget/v/Microsoft.OpenApi.Hidi.svg)](https://www.nuget.org/packages/Microsoft.OpenApi.Hidi/)

The **OpenAPI.NET** SDK contains a useful object model for OpenAPI documents in .NET along with common serializers to extract raw OpenAPI JSON and YAML documents from the model.

**See more information on the OpenAPI specification and its history here: <a href="https://www.openapis.org">OpenAPI Initiative</a>**

Project Objectives:

- Provide a single shared object model in .NET for OpenAPI descriptions.
- Include the most primitive Reader for ingesting OpenAPI JSON and YAML documents in both V2 and V3 formats.
- Provide OpenAPI description writers for both V2 and V3 specification formats.
- Enable developers to create Readers that translate different data formats into OpenAPI descriptions.

# Installation

- Install core Nuget package [**Microsoft.OpenApi**](https://www.nuget.org/packages/Microsoft.OpenApi)
- Install Yaml Reader Nuget package [**Microsoft.OpenApi.YamlReader**](https://www.nuget.org/packages/Microsoft.OpenApi.YamlReader)

> Note: we just released a new major version of the library, which brings support for OpenAPI 3.2!
> You can read more about the changes of this upcoming version [in the upgrade guide](./docs/upgrade-guide-3.md).

# Processors
The OpenAPI.NET project holds the base object model for representing OpenAPI documents as .NET objects. Some developers have found the need to write processors that convert other data formats into this OpenAPI.NET object model. We'd like to curate that list of processors in this section of the readme. 

The base JSON and YAML processors are built into this project. Below is the list of the other supported processor projects.

- [**C# Comment / Annotation Processor**](https://github.com/Microsoft/OpenAPI.NET.CSharpAnnotations) : Converts standard .NET annotations ( /// comments ) emitted from your build (MSBuild.exe) into OpenAPI.NET document object. 

- [**OData CSDL Processor**](https://github.com/Microsoft/OpenAPI.NET.OData) : Converts the XML representation of the Entity Data Model (EDM) describing an OData Service into OpenAPI.NET document object. 

# Example Usage

Creating an OpenAPI Document

```C#
var document = new OpenApiDocument
{
    Info = new OpenApiInfo
    {
        Version = "1.0.0",
        Title = "Swagger Petstore (Simple)",
    },
    Servers = new List<OpenApiServer>
    {
        new OpenApiServer { Url = "http://petstore.swagger.io/api" }
    },
    Paths = new OpenApiPaths
    {
        ["/pets"] = new OpenApiPathItem
        {
            Operations = new()
            {
                [HttpMethod.Get] = new OpenApiOperation
                {
                    Description = "Returns all pets from the system that the user has access to",
                    Responses = new OpenApiResponses
                    {
                        ["200"] = new OpenApiResponse
                        {
                            Description = "OK"
                        }
                    }
                }
            }
        }
    }
};
```

Reading and writing an OpenAPI description

```C#
var (openApiDocument, _) = await OpenApiDocument.LoadAsync("https://raw.githubusercontent.com/OAI/OpenAPI-Specification/refs/heads/main/_archive_/schemas/v3.0/pass/petstore.yaml");

// Write V2 as JSON
var outputString = await openApiDocument.SerializeAsJsonAsync(OpenApiSpecVersion.OpenApi2_0);
```

# Validating/Testing OpenAPI descriptions
In order to test the validity of an OpenApi document, we avail the following tools:
- [Microsoft.OpenApi.Hidi](https://www.nuget.org/packages/Microsoft.OpenApi.Hidi)

    A commandline tool for validating and transforming OpenAPI descriptions. [Installation guidelines and documentation](https://github.com/microsoft/OpenAPI.NET/blob/main/src/Microsoft.OpenApi.Hidi/readme.md)

- Microsoft.OpenApi.Workbench

    A workbench tool consisting of a GUI where you can test and convert OpenAPI descriptions in both JSON and YAML from v2-->v3 and vice versa.

    #### Installation guidelines:
    1. Clone the repo locally by running this command:
        `git clone https://github.com/microsoft/OpenAPI.NET.git`
    2. Open the solution file `(.sln)` in the root of the project with Visual Studio
    3. Navigate to the `src/Microsoft.OpenApi.Workbench` directory and set it as the startup project
    4. Run the project and you'll see a GUI pop up resembling the one below:
    
    
    ![workbench preview](https://raw.githubusercontent.com/microsoft/OpenAPI.NET/main/docs/images/workbench.png "a screenshot of the workbench application")
    
    5. Copy and paste your OpenAPI descriptions in the **Input Content** window or paste the path to the descriptions file in the **Input File** textbox and click on `Convert` to render the results.

# Contributing

This project welcomes contributions and suggestions.  Most contributions require you to agree to a
Contributor License Agreement (CLA) declaring that you have the right to, and actually do, grant us
the rights to use your contribution. For details, visit https://cla.microsoft.com.

When you submit a pull request, a CLA-bot will automatically determine whether you need to provide
a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the instructions
provided by the bot. You will only need to do this once across all repos using our CLA.

This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/).
For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/) or
contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments.

To provide feedback and ask questions you can use Stack Overflow with the [OpenAPI.NET](https://stackoverflow.com/questions/tagged/openapi.net) tag.
 readmeEtag: '"6a7f787f8e6cf6076b83186b54e46c10ddfc4c94"' readmeLastModified: Tue, 11 Nov 2025 21:58:29 GMT repositoryId: 97175798 description: >- The OpenAPI.NET SDK contains a useful object model for OpenAPI documents in .NET along with common serializers to extract raw OpenAPI JSON and YAML documents from the model. created: '2017-07-14T00:24:14Z' updated: '2026-02-05T12:17:03Z' language: C# archived: false stars: 1575 watchers: 62 forks: 282 owner: microsoft logo: https://avatars.githubusercontent.com/u/6154722?v=4 license: MIT repoEtag: '"02e87e29122fea6641bc92d47b69eb0c7a7bdcb9421082dddf2cc5f1bb1d11d0"' repoLastModified: Thu, 05 Feb 2026 12:17:03 GMT id: f3215034d64cfcbcc28a4bdb0205a075 - source: IMPLEMENTATIONS.md name: openapi-validator homepage: https://gitlab.com/mmalawski/openapi-validator language: PHP source_description: Validates response against OpenAPI schema category: Low-level Tooling foundInMaster: true id: efecab7adc90e5b1b9e902939ab0a319 - source: IMPLEMENTATIONS.md name: OpenAPI-Delphi homepage: https://github.com/paolo-rossi/OpenAPI-Delphi language: Delphi source_description: >- Delphi implementation of a generator, parser and validator for the OpenAPI 3 Specification category: - Low-level Tooling - Parsers foundInMaster: true repository: https://github.com/paolo-rossi/openapi-delphi repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIGZvciBEZWxwaGkgLSBPcGVuQVBJIDMuMCBmb3IgRGVscGhpCgo8YnIgLz4KCjxwIGFsaWduPSJjZW50ZXIiPgogIDxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vcGFvbG8tcm9zc2kvT3BlbkFQSS1EZWxwaGkvYmxvYi9tYXN0ZXIvb3BlbmFwaS1kZWxwaGkucG5nIiBhbHQ9Ik9wZW5BUEkgRGVscGhpIExpYnJhcnkiIHdpZHRoPSI0MDAiIC8+CjwvcD4KCiMjIFdoYXQgaXMgT3BlbkFQSS1EZWxwaGkKCioqT3BlbkFQSS1EZWxwaGkqKiBpcyBhbiBPcGVuQVBJIDMuMCBsaWJyYXJ5IGZvciBbRGVscGhpXShodHRwczovL3d3dy5lbWJhcmNhZGVyby5jb20vcHJvZHVjdHMvZGVscGhpKSB0aGF0IGhlbHBzIHlvdSB0byBnZW5lcmF0ZSAoYW5kIGxvYWQpIE9wZW5BUEkgMy4wIGRvY3VtZW50YXRpb24gKGluIEpTT04pIHN0YXJ0aW5nIGZyb20gcGxhaW4gRGVscGhpIGNsYXNzZXMuIERlbHBoaS1PcGVuQVBJIHVzZXMgdGhlIFtOZW9uXShodHRwczovL2dpdGh1Yi5jb20vcGFvbG8tcm9zc2kvZGVscGhpLW5lb24pIHNlcmlhbGl6YXRpb24gbGlicmFyeSB0byB0cmFuc2Zvcm0gdGhlIE9wZW5BUEkgbW9kZWxzIGZyb20gRGVscGhpIGNsYXNzZXMgdG8gSlNPTiBhbmQgdG8gbG9hZCBhIE9wZW5BUEkgZG9jdW1lbnQgaW50byBhIERlbHBoaSAoT3BlbkFQSSkgb2JqZWN0LiBQbGVhc2UgdGFrZSBhIGxvb2sgYXQgdGhlIERlbW8gdG8gc2VlIE9wZW5BUEktRGVscGhpIGluIGFjdGlvbi4KCiMjIEdlbmVyYWwgRmVhdHVyZXMKCi0gT3BlbkFQSSBkb2N1bWVudCBnZW5lcmF0aW9uIChKU09OKSBmcm9tIGEgRGVscGhpIChPcGVuQVBJKSBvYmplY3QgCi0gT3BlbkFQSSBsb2FkaW5nIGFuZCBwYXJzaW5nIGludG8gYSBEZWxwaGkgKE9wZW5BUEkpIG9iamVjdCAoOnN0YXIyOiBuZXcgaW4gMi4wKQotIFVzZSBwbGFpbiBEZWxwaGkgY2xhc3NlcyB0byBzZXQgdGhlIE9wZW5BUEkgc3BlY2lmaWNhdGlvbiBzZWN0aW9ucyAmIGZpZWxkcwotIFN1cHBvcnQgZm9yIEpTT04gU2NoZW1hICh0aGUgT3BlbkFQSSB2ZXJzaW9uKQotIFN1cHBvcnQgZm9yIFNjaGVtYSBmaWVsZCByZWN1cnNpb24gKDpzdGFyMjogbmV3IGluIDIuMCkKLSBGdWxsIFN1cHBvcnQgZm9yIGVudW0gb2YgYW55IHR5cGUgKDpzdGFyMjogbmV3IGluIDIuMCkKLSBVc2UgMS1saW5lIGNvZGUgKHVzaW5nIHRoZSBbTmVvbl0oaHR0cHM6Ly9naXRodWIuY29tL3Bhb2xvLXJvc3NpL2RlbHBoaS1uZW9uKSBsaWJyYXJ5KSB0byB0cmFuc2Zvcm0gZnJvbSBhbmQgdG8gSlNPTiBkb2N1bWVudHMKCiMjIERlbHBoaSBDb21wYXRpYmlsaXR5ClRoaXMgbGlicmFyeSBoYXMgYmVlbiB0ZXN0ZWQgd2l0aCAqKkRlbHBoaSAxMiBBdGhlbnMqKiwgKipEZWxwaGkgMTEgQWxleGFuZHJpYSoqLCAqKkRlbHBoaSAxMC40IFN5ZG5leSoqLCAqKkRlbHBoaSAxMC4zIFJpbyoqLCAqKkRlbHBoaSAxMC4yIFRva3lvKiouCgoKIyMgVG9kbwotIEZ1bGwgdmFsaWRhdGlvbiBmb3IgdGhlIE9wZW5BUEkgbW9kZWxzCgo= readmeEtag: '"ed3c74637bbd0f70dcb7ad950270331fd2793c36"' readmeLastModified: Sat, 04 May 2024 08:36:53 GMT repositoryId: 190047866 description: >- The Delphi-OpenAPI library is an OpenAPI 3.0 document generator and parser for Delphi created: '2019-06-03T17:07:21Z' updated: '2026-02-05T16:05:17Z' language: Pascal archived: false stars: 104 watchers: 14 forks: 34 owner: paolo-rossi logo: https://avatars.githubusercontent.com/u/4686497?v=4 license: Apache-2.0 repoEtag: '"f1e2627f591cd0780a695858b0486af6c7d405f2611e7e8a8c6be814cf5085cc"' repoLastModified: Thu, 05 Feb 2026 16:05:17 GMT id: 29fab7d64d0321c64910ca260c8a92d1 - source: IMPLEMENTATIONS.md name: openapi-runtime-expression homepage: https://github.com/char0n/openapi-runtime-expression language: JavaScript source_description: OpenAPI Runtime Expressions parser and validator. category: - Low-level Tooling - Parsers repository: https://github.com/swaggerexpert/openapi-runtime-expression id: 8a849f2d0d23f7d6d54e1b510e05fd16 repositoryMetadata: base64Readme: >- # @swaggerexpert/openapi-runtime-expression

[![npmversion](https://img.shields.io/npm/v/%40swaggerexpert%2Fopenapi-runtime-expression?style=flat-square&label=npm%20package&color=%234DC81F&link=https%3A%2F%2Fwww.npmjs.com%2Fpackage%2F%40swaggerexpert%2Fopenapi-runtime-expression)](https://www.npmjs.com/package/@swaggerexpert/openapi-runtime-expression)
[![npm](https://img.shields.io/npm/dm/@swaggerexpert/openapi-runtime-expression)](https://www.npmjs.com/package/@swaggerexpert/openapi-runtime-expression)
[![Test workflow](https://github.com/swaggerexpert/openapi-runtime-expression/actions/workflows/test.yml/badge.svg)](https://github.com/swaggerexpert/openapi-runtime-expression/actions)
[![Dependabot enabled](https://img.shields.io/badge/Dependabot-enabled-blue.svg)](https://dependabot.com/)
[![try on RunKit](https://img.shields.io/badge/try%20on-RunKit-brightgreen.svg?style=flat)](https://npm.runkit.com/@swaggerexpert/openapi-runtime-expression)
[![Tidelift](https://tidelift.com/badges/package/npm/@swaggerexpert%2Fopenapi-runtime-expression)](https://tidelift.com/subscription/pkg/npm-.swaggerexpert-openapi-runtime-expression?utm_source=npm-swaggerexpert-openapi-runtime-expression&utm_medium=referral&utm_campaign=readme)

[OpenAPI Runtime Expressions](https://spec.openapis.org/oas/v3.1.1.html#runtime-expressions) allow defining values based on information that will only be available within the HTTP message in an actual API call.
This mechanism is used by [Link Objects](https://spec.openapis.org/oas/v3.1.1.html#link-object) and [Callback Objects](https://spec.openapis.org/oas/v3.1.1.html#callback-object)
of [OpenAPI specification](https://spec.openapis.org/#openapi-specification).

`@swaggerexpert/openapi-runtime-expression` is a **parser**, **validator** and **extractor** for OpenAPI Runtime Expressions.

It supports Runtime Expressions defined in following OpenAPI specification versions:

- [OpenAPI 3.0.0](https://spec.openapis.org/oas/v3.0.0.html#runtime-expressions)
- [OpenAPI 3.0.1](https://spec.openapis.org/oas/v3.0.1.html#runtime-expressions)
- [OpenAPI 3.0.2](https://spec.openapis.org/oas/v3.0.2.html#runtime-expressions)
- [OpenAPI 3.0.3](https://spec.openapis.org/oas/v3.0.3.html#runtime-expressions)
- [OpenAPI 3.0.4](https://spec.openapis.org/oas/v3.0.4.html#runtime-expressions)
- [OpenAPI 3.1.0](https://spec.openapis.org/oas/v3.1.0.html#runtime-expressions)
- [OpenAPI 3.1.1](https://spec.openapis.org/oas/v3.1.1.html#runtime-expressions)

<table>
  <tr>
    <td align="right" valign="middle">
        <img src="https://cdn2.hubspot.net/hubfs/4008838/website/logos/logos_for_download/Tidelift_primary-shorthand-logo.png" alt="Tidelift" width="60" />
      </td>
      <td valign="middle">
        <a href="https://tidelift.com/subscription/pkg/npm-.swaggerexpert-openapi-runtime-expression?utm_source=npm-swaggerexpert-openapi-runtime-expression&utm_medium=referral&utm_campaign=readme">
            Get professionally supported @swaggerexpert/openapi-runtime-expression with Tidelift Subscription.
        </a>
      </td>
  </tr>
</table>

## Table of Contents

- [Getting started](#getting-started)
  - [Installation](#installation)
  - [Usage](#usage)
    - [Extraction](#extraction)
    - [Parsing](#parsing)
    - [Validation](#validation)
    - [Grammar](#grammar)
- [More about OpenAPI runtime expressions](#more-about-openapi-runtime-expressions)
- [License](#license)


## Getting started

### Installation

You can install `@swaggerexpert/openapi-runtime-expression` using `npm`:

```sh
 $ npm install @swaggerexpert/openapi-runtime-expression
```

### Usage

`@swaggerexpert/openapi-runtime-expression` currently supports **extraction**, **parsing** and **validation**.
Both parser and validator are based on a superset of [ABNF](https://www.rfc-editor.org/rfc/rfc5234) ([SABNF](https://cs.github.com/ldthomas/apg-js2/blob/master/SABNF.md))
and use [apg-lite](https://github.com/ldthomas/apg-lite) parser generator.

#### Extraction

OpenAPI embeds Runtime Expressions into string values surrounded with `{}` curly braces.
To extract Runtime Expressions from this embedded form, use the **extract** function.
Extracted Runtime Expression can be used for further parsing of validation.

```js
import { extract, test, parse } from '@swaggerexpert/openapi-runtime-expression';

const expression = extract('{$request.header.accept}'); // => '$request.header.accept'

test(expression); // => true
parse(expression); // => { result, ast }
```

#### Parsing

Parsing a Runtime Expression is as simple as importing the **parse** function and calling it.

```js
import { parse } from '@swaggerexpert/openapi-runtime-expression';

const parseResult = parse('$request.header.accept');
```

`token` non-terminal is by default being normalized to lower case.

```js
import { parse } from '@swaggerexpert/openapi-runtime-expression';

const parseResult = parse('$request.header.Accept');
const parts = [];

parseResult.ast.translate(parts);
// [
//   [ 'expression', '$request.header.Accept' ],
//   [ 'source', 'header.Accept' ],
//   [ 'header-reference', 'header.Accept' ],
//   [ 'token', 'accept' ],
// ]
```

`token` normalization can be overridden by passing token normalizer to the `parse` function.

**Upper case**

```js
import { parse, tokenUpperCaseNormalizer } from '@swaggerexpert/openapi-runtime-expression';

const parseResult = parse('$request.header.Accept', { tokenNormalizer: tokenUpperCaseNormalizer });
const parts = [];
parseResult.ast.translate(parts);
// [
//   [ 'expression', '$request.header.Accept' ],
//   [ 'source', 'header.Accept' ],
//   [ 'header-reference', 'header.Accept' ],
//   [ 'token', 'ACCEPT' ],
// ]`
```

**Lower case**

```js
import { parse, tokenLowerCaseNormalizer } from '@swaggerexpert/openapi-runtime-expression';

const parseResult = parse('$request.header.Accept', { tokenNormalizer: tokenLowerCaseNormalizer });
const parts = [];
parseResult.ast.translate(parts);
// [
//   [ 'expression', '$request.header.Accept' ],
//   [ 'source', 'header.Accept' ],
//   [ 'header-reference', 'header.Accept' ],
//   [ 'token', 'accept' ],
// ]`
```

**parseResult** variable has the following shape:

```
{
  result: {
    success: true,
    state: 101,
    stateName: 'MATCH',
    length: 22,
    matched: 22,
    maxMatched: 22,
    maxTreeDepth: 13,
    nodeHits: 152
  },
  ast: fnast {
    callbacks: [
      expression: [Function: expression],
      source: [Function: source],
      'header-reference': [Function: headerReference],
      'query-reference': [Function: queryReference],
      'path-reference': [Function: pathReference],
      'body-reference': [Function: bodyReference],
      'json-pointer': [Function: jsonPointer],
      'reference-token': [Function: referenceToken],
      name: [Function: name],
      token: [Function: token]
    ],
    init: [Function (anonymous)],
    ruleDefined: [Function (anonymous)],
    udtDefined: [Function (anonymous)],
    down: [Function (anonymous)],
    up: [Function (anonymous)],
    translate: [Function (anonymous)],
    setLength: [Function (anonymous)],
    getLength: [Function (anonymous)],
    toXml: [Function (anonymous)]
  }
}
```

###### Interpreting AST as list of entries

```js
import { parse } from '@swaggerexpert/openapi-runtime-expression';

const parseResult = parse('$request.header.accept');
const parts = [];

parseResult.ast.translate(parts);
```

After running the above code, **parts** variable has the following shape:

```js
[
  [ 'expression', '$request.header.accept' ],
  [ 'source', 'header.accept' ],
  [ 'header-reference', 'header.accept' ],
  [ 'token', 'accept' ],
]
```

###### Interpreting AST as XML

```js
import { parse } from '@swaggerexpert/openapi-runtime-expression';

const parseResult = parse('$request.header.accept');
const xml = parseResult.ast.toXml();
```

After running the above code, **xml** variable has the following content:

```xml
<?xml version="1.0" encoding="utf-8"?>
<root nodes="4" characters="22">
  <!-- input string -->
  $request.header.accept
  <node name="expression" index="0" length="22">
    $request.header.accept
    <node name="source" index="9" length="13">
      header.accept
      <node name="header-reference" index="9" length="13">
        header.accept
        <node name="token" index="16" length="6">
          accept
        </node><!-- name="token" -->
      </node><!-- name="header-reference" -->
    </node><!-- name="source" -->
  </node><!-- name="expression" -->
</root>
```

> NOTE: AST can also be traversed in classical way using [depth first traversal](https://www.tutorialspoint.com/data_structures_algorithms/depth_first_traversal.htm). For more information about this option please refer to [apg-js](https://github.com/ldthomas/apg-js) and [apg-js-examples](https://github.com/ldthomas/apg-js-examples).

#### Validation

Validating a Runtime Expression is as simple as importing the **test** function and calling it.

```js
import { test } from '@swaggerexpert/openapi-runtime-expression';

test('$request.header.accept'); // => true
test('nonsensical string'); // => false
```

#### Grammar

New grammar instance can be created in following way:

```js
import { Grammar } from '@swaggerexpert/openapi-runtime-expression';

const grammar = new Grammar();
```

To obtain original ABNF (SABNF) grammar as a string:

```js
import { Grammar } from '@swaggerexpert/openapi-runtime-expression';

const grammar = new Grammar();

grammar.toString();
// or
String(grammar);
```

## More about OpenAPI runtime expressions

The runtime expression is defined by the following [ABNF](https://tools.ietf.org/html/rfc5234) syntax

```abnf
; OpenAPI runtime expression ABNF syntax
expression       = "$url" / "$method" / "$statusCode" / "$request." source / "$response." source
source           = header-reference / query-reference / path-reference / body-reference
header-reference = "header." token
query-reference  = "query." name
path-reference   = "path." name
body-reference   = "body" ["#" json-pointer ]
name             = *( CHAR )

; https://datatracker.ietf.org/doc/html/rfc6901#section-3
json-pointer     = *( "/" reference-token )
reference-token  = *( unescaped / escaped )
unescaped        = %x00-2E / %x30-7D / %x7F-10FFFF
                 ; %x2F ('/') and %x7E ('~') are excluded from 'unescaped'
escaped          = "~" ( "0" / "1" )
                 ; representing '~' and '/', respectively

; https://datatracker.ietf.org/doc/html/rfc7230#section-3.2.6
token          = 1*tchar
tchar          = "!" / "#" / "$" / "%" / "&" / "'" / "*"
               / "+" / "-" / "." / "^" / "_" / "`" / "|" / "~"
               / DIGIT / ALPHA
               ; any VCHAR, except delimiters

; https://www.rfc-editor.org/rfc/rfc7159#section-7
CHAR = unescape /
    escape (
        %x22 /          ; "    quotation mark  U+0022
        %x5C /          ; \    reverse solidus U+005C
        %x2F /          ; /    solidus         U+002F
        %x62 /          ; b    backspace       U+0008
        %x66 /          ; f    form feed       U+000C
        %x6E /          ; n    line feed       U+000A
        %x72 /          ; r    carriage return U+000D
        %x74 /          ; t    tab             U+0009
        %x75 4HEXDIG )  ; uXXXX                U+XXXX
escape         = %x5C   ; \
unescape       = %x20-21 / %x23-5B / %x5D-10FFFF

; https://datatracker.ietf.org/doc/html/rfc5234#appendix-B.1
HEXDIG         =  DIGIT / "A" / "B" / "C" / "D" / "E" / "F"
DIGIT          =  %x30-39   ; 0-9
ALPHA          =  %x41-5A / %x61-7A   ; A-Z / a-z
```

The `name` identifier is case-sensitive, whereas `token` is not.

The table below provides examples of runtime expressions and examples of their use in a value:

##### Examples

| Source Location | example expression | notes |
| ---- | :---- | :---- |
| HTTP Method | `$method` | The allowable values for the `$method` will be those for the HTTP operation. |
| Requested media type | `$request.header.accept` | |
| Request parameter | `$request.path.id` | Request parameters MUST be declared in the `parameters` section of the parent operation or they cannot be evaluated. This includes request headers. |
| Request body property | `$request.body#/user/uuid` | In operations which accept payloads, references may be made to portions of the `requestBody` or the entire body. |
| Request URL | `$url` | |
| Response value | `$response.body#/status` | In operations which return payloads, references may be made to portions of the response body or the entire body. |
| Response header | `$response.header.Server` | Single header values only are available |

Runtime expressions preserve the type of the referenced value.
Expressions can be embedded into string values by surrounding the expression with `{}` curly braces.

## License

`@swaggerexpert/openapi-runtime-expression` is licensed under [Apache 2.0 license](https://github.com/swaggerexpert/openapi-runtime-expression/blob/main/LICENSE).
`@swaggerexpert/openapi-runtime-expression` comes with an explicit [NOTICE](https://github.com/swaggerexpert/openapi-runtime-expression/blob/main/NOTICE) file
containing additional legal notices and information.
 readmeEtag: '"fc77416d182fef85df4fd8aed436863a498b5e0a"' readmeLastModified: Mon, 17 Feb 2025 22:25:18 GMT repositoryId: 558798413 description: OpenAPI Runtime Expressions parser, validator and extractor. created: '2022-10-28T10:18:43Z' updated: '2026-01-21T20:45:15Z' language: JavaScript archived: false stars: 6 watchers: 1 forks: 0 owner: swaggerexpert logo: https://avatars.githubusercontent.com/u/172408630?v=4 license: Apache-2.0 repoEtag: '"9a42851f1773f47e455558431939d8d08c51b6a1583dc4473ea6cd18169b984c"' repoLastModified: Wed, 21 Jan 2026 20:45:15 GMT foundInMaster: true oldLocations: - https://github.com/char0n/openapi-runtime-expression - source: IMPLEMENTATIONS.md name: Oxygen OpenAPI Editor homepage: https://www.oxygenxml.com/openapi.html language: Java source_description: >- OpenAPI editor with a variety of editing features and helper views. Support for validation and editing OpenAPI 2.0, 3.0, and 3.1 based on JSON Schema specification. Includes a tool for generating documentations and a tool for testing OpenAPIs. category: Editors id: d874e7661794cdbf6243becfabd4d71d foundInMaster: true - source: IMPLEMENTATIONS.md name: Visual Studio Code extension homepage: https://marketplace.visualstudio.com/items?itemName=42Crunch.vscode-openapi language: TypeScript source_description: >- Extends VS Code to provide OpenAPI 2.0 and 3.0 navigation, code snippets, new API creation category: Editors foundInMaster: true id: 43647cd83453a9aa9ecc87f45fb01dfb - source: IMPLEMENTATIONS.md name: RepreZen API Studio homepage: https://www.reprezen.com/OpenAPI language: Java source_description: Commercial desktop IDE for API design, documentation & development category: Editors foundInMaster: true id: a89ec3ac594d929307e2c40ef58a899b - source: IMPLEMENTATIONS.md name: SwaggerHub homepage: https://swaggerhub.com language: - API Design and Documentation Platform - Built For Teams category: Editors foundInMaster: true id: 5c8b1a8e6d59b3b055d5083f29dedfbc - source: IMPLEMENTATIONS.md name: Remain OpenAPI Studio homepage: >- https://remainsoftware.com/extranet/download-type/openapi-studio-download
Or via Eclipse MarketPlace https://marketplace.eclipse.org/content/openapi-studio-rich-oas3-editor language: Java source_description: >- A user-friendly, visually rich studio supporting all features defined by the OpenAPI 3. Easy but powerful UI-based components creation, API testing, import, export, code generation and much more. category: Editors foundInMaster: true id: ebacf52793f49b2084d381fdf64da69f - source: IMPLEMENTATIONS.md name: lincoln homepage: https://github.com/temando/open-api-renderer language: React.js source_description: A React renderer for OpenAPI v3 category: - User Interfaces - Parsers foundInMaster: true repository: https://github.com/temando/open-api-renderer repositoryMetadata: base64Readme: >- IyBMaW5jb2xuCgo8aW1nIHNyYz0nYXNzZXRzL2xpbmNvbG4tbG9nby13aGl0ZS5wbmcnIGFsdD0nTGluY29sbicgaGVpZ2h0PSI5NiIgd2lkdGg9Ijk2IiAvPgoKWyFbTlBNXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS92L3JlYWN0LWxpbmNvbG4uc3ZnKV0oaHR0cHM6Ly9ucG1qcy5vcmcvcGFja2FnZXMvcmVhY3QtbGluY29sbi8pClshW1RyYXZpcyBDSV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby90cmF2aXMvdGVtYW5kby9vcGVuLWFwaS1yZW5kZXJlci5zdmcpXShodHRwczovL3RyYXZpcy1jaS5vcmcvdGVtYW5kby9vcGVuLWFwaS1yZW5kZXJlcikKWyFbTUlUIExpY2Vuc2VdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vZ2l0aHViL2xpY2Vuc2UvdGVtYW5kby9vcGVuLWFwaS1yZW5kZXJlci5zdmcpXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9NSVRfTGljZW5zZSkKWyFbSmF2YVNjcmlwdCBTdHlsZSBHdWlkZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9jb2RlX3N0eWxlLXN0YW5kYXJkLWJyaWdodGdyZWVuLnN2ZyldKGh0dHBzOi8vc3RhbmRhcmRqcy5jb20pCgpNZWV0IExpbmNvbG4sIGEgW1JlYWN0XShodHRwczovL2ZhY2Vib29rLmdpdGh1Yi5pby9yZWFjdC8pIGNvbXBvbmVudCBmb3IgcmVuZGVyaW5nIFtPcGVuQVBJXShodHRwczovL3d3dy5vcGVuYXBpcy5vcmcpIGRvY3VtZW50cy4gVGhlIHByb2plY3QgaXMgdHJhY2tpbmcgYWdhaW5zdCBbdjMuMC4wXShkb2NzL29wZW4tYXBpLXYzLXN1cHBvcnQubWQpIG9mIHRoZSBPcGVuQVBJIHNwZWNpZmljYXRpb24uCgpMaW5jb2xuIGFpbXMgdG8gc3VwcG9ydCBldmVyZ3JlZW4gYnJvd3NlcnMsIHN1Y2ggYXMgQ2hyb21lLCBGaXJlZm94LCBTYWZhcmkgYW5kIElFMTErLiBJdCBpcyByZXNwb25zaXZlIGFuZCBzaG91bGQgYmUgdXNhYmxlIG9uIG1vc3QgbW9kZXJuIGRldmljZXMuCgpbRGVtb10oaHR0cHM6Ly90ZW1hbmRvLmdpdGh1Yi5pby9vcGVuLWFwaS1yZW5kZXJlci9kZW1vLz91cmw9aHR0cHM6Ly90ZW1hbmRvLmdpdGh1Yi5pby9vcGVuLWFwaS1yZW5kZXJlci9wZXRzdG9yZS1vcGVuLWFwaS12My4wLjAtUkMyLmpzb24pCgojIyBJbnN0YWxsYXRpb24KCmBgYHNoCm5wbSBpbnN0YWxsIC0tc2F2ZSByZWFjdCByZWFjdC1kb20gcmVhY3QtbGluY29sbgpgYGAKCiMjIFVzYWdlCgpUbyB1c2UgTGluY29sbiBpbiB5b3VyIFJlYWN0IHByb2plY3Q6CgpgYGBqcwppbXBvcnQgUmVhY3QgZnJvbSAncmVhY3QnCmltcG9ydCBSZWFjdERPTSBmcm9tICdyZWFjdC1kb20nCmltcG9ydCBMaW5jb2xuIGZyb20gJ3JlYWN0LWxpbmNvbG4nCgpSZWFjdERPTS5yZW5kZXIoCiAgPExpbmNvbG4gZGVmaW5pdGlvblVybD0naHR0cHM6Ly90ZW1hbmRvLmdpdGh1Yi5pby9vcGVuLWFwaS1yZW5kZXJlci9wZXRzdG9yZS1vcGVuLWFwaS12My4wLjAtUkMyLmpzb24nIC8+LAogIGRvY3VtZW50LmJvZHkKKQpgYGAKCkFsdGVybmF0aXZlbHksIHlvdSBjYW4gcGFzcyB0aGUgY29udGVudHMgb2YgdGhlIGRlZmluaXRpb24gZGlyZWN0bHkgdG8gTGluY29sbjoKCmBgYGpzCmltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCcKaW1wb3J0IFJlYWN0RE9NIGZyb20gJ3JlYWN0LWRvbScKaW1wb3J0IExpbmNvbG4gZnJvbSAncmVhY3QtbGluY29sbicKaW1wb3J0IGRlZmluaXRpb24gZnJvbSAnLi9teUFwaS55bWwnCgpSZWFjdERPTS5yZW5kZXIoPExpbmNvbG4gZGVmaW5pdGlvbj17ZGVmaW5pdGlvbn0gLz4sIGRvY3VtZW50LmJvZHkpCmBgYAoKIyMgQ29uZmlndXJhdGlvbgoKVGhlIGZvbGxvd2luZyBjb25maWd1cmF0aW9uIG9wdGlvbnMgYXJlIGF2YWlsYWJsZToKCnwgcHJvcGVydHkgICAgICAgICAgICAgICAgICAgICAgICB8IHJlcXVpcmVkIHwgdHlwZSAgICB8IGRlc2NyaXB0aW9uICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCAtLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tIHwgLS0tLS0tLS0gfCAtLS0tLS0tIHwgLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0gfAp8IGBkZWZpbml0aW9uVXJsYCBvciBgZGVmaW5pdGlvbmAgfCDinJQgICAgICAgIHwgc3RyaW5nICB8IENPUlMtZW5hYmxlZCBVUkwgdG8sIG9yIGNvbnRlbnRzIG9mLCBPcGVuQVBJIHYzIGRvY3VtZW50IHRvIHJlbmRlci4gU3VwcG9ydHMgSlNPTiBvciBZQU1MLiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwKfCBgbmF2U29ydGAgICAgICAgICAgICAgICAgICAgICAgIHwgICAgICAgICAgfCBlbnVtICAgIHwgVGhpcyBwcm9wZXJ0eSBhcHBsaWVzIHdoZW4geW91ciBkZWZpbml0aW9uIHVzZXMgYHRhZ3NgLiBWYWxpZCB2YWx1ZXMgYXJlIGBhbHBoYWAgd2hpY2ggc29ydHMgYnkgSFRUUCBtZXRob2QsIHRoZW4gcGF0aCBvciBgZmFsc2VgLCB3aGljaCB3aWxsIGRpc3BsYXkgcGF0aHMgYXMgZGVmaW5lZC4gRGVmYXVsdHMgdG8gYGZhbHNlYC4gICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IGB2YWxpZGF0ZWAgICAgICAgICAgICAgICAgICAgICAgfCAgICAgICAgICB8IGJvb2xlYW4gfCBJZiBgdHJ1ZWAsIHVzZXMgW01lcm1hZGVdKGh0dHBzOi8vb3BlbmFwaS1jb252ZXJ0ZXIuaGVyb2t1YXBwLmNvbS8pIHRvIHZhbGlkYXRlIGRlZmluaXRpb24uIERlZmF1bHRzIHRvIGBmYWxzZWAuICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgYGluaXRpYWxTY2hlbWFUcmVlRGVwdGhgICAgICAgICB8ICAgICAgICAgIHwgbnVtYmVyICB8IFRoZSBpbml0aWFsIG9wZW5lZCB0cmVlIGRlcHRoIGZvciBzY2hlbWEgdmlzdWFsaXNlciB3aGVuIGZpcnN0IHJlbmRlcmVkLiBUaGlzIGlzIHVzZWZ1bCB3aGVuIHRoZSBzY2hlbWEncyBhY3R1YWwgY29udGVudHMgaXMgYSBjb3VwbGUgb2YgbGV2ZWxzIGRlZXAsIGFuZCB5b3Ugd2FudCB0byBleHBhbmQgdGhlIHRyZWUgdG8gdGhlIGNvbnRlbnRzIGF1dG9tYXRpY2FsbHkuIERlZmF1bHRzIHRvIDAuIHwKfCBgbmF2aWdhdGlvbk1ldGhvZERpc3BsYXlUeXBlYCAgIHwgICAgICAgICAgfCBzdHJpbmcgIHwgUmVndWxhdGVzIGhvdyB0aGUgbmF2aWdhdGlvbiBpdGVtcyBhcmUgcmVuZGVyZWQgLSBwYXRoIG9ubHksIHN1bW1hcnkgb25seSwgb3IgYm90aC4gVGhlIHBvc3NpYmxlIHZhbHVlcyBhcmUgYHN1bW1hcnlgIChkZWZhdWx0KSwgYHBhdGhgLCBvciBgYWxsYC4gSW4gY2FzZSBvZiBhbnkgb3RoZXIgdmFsdWUgb25seSB0aGUgc3VtbWFyeSBpcyByZW5kZXJlZC4gICAgICAgICAgICAgICAgICAgICAgICAgfAoKIyMgUGhpbG9zb3BoeQoKV2hpbGUgdGhpcyBwcm9qZWN0IGlzIGN1cnJlbnRseSBmb2N1c2VkIG9uIHZpc3VhbGlzaW5nIE9wZW5BUEkgVjMgc3BlY2lmaWNhdGlvbnMsIGl0IGlzIGFyY2hpdGVjdGVkIGluIHN1Y2ggYSB3YXkgdGhhdCB0aGUgUmVhY3QgY29tcG9uZW50cyBkZWFsIHdpdGggYSBgVUlSZWFkeVNjaGVtYWAsIHdoaWNoIGlzIGEgZ2VuZXJpYyBzcGVjaWZpY2F0aW9uIChhZG1pdHRlZGx5IGhlYXZpbHkgYmFzZWQgb24gT3BlbkFQSSBWMykuCgpUaGUgZHJlYW0gaXMgdGhhdCB0aGlzIHJlbmRlcmVyIGNvdWxkIHZpc3VhbGlzZSBvdGhlciBmb3JtYXRzIGJ5IGludHJvZHVjaW5nIG5ldyBwYXJzZXJzIHdoaWNoIHRyYW5zZm9ybSBkb2N1bWVudHMgaW50byB0aGUgY29tbW9uIGBVSVJlYWR5U2NoZW1hYCBmb3JtYXQuIFRoaXMgYXBwcm9hY2ggYWxsb3dzIHVzIHRvIGJ1aWxkIHNvbWV0aGluZyBzdXN0YWluYWJsZSBhbmQgc2NhbGFibGUsIHdoZXJlIHRoZSBjb21tdW5pdHkgY2FuIGhlbHAgY29udHJpYnV0ZSBuZXcgcGFyc2VycyAoYW1vbmcgb3RoZXIgdGhpbmdzISkgYXMgcmVxdWlyZWQuCgpUaGUgcHJvamVjdCBpcyBkZWZpbml0ZWx5IGluIGl0cyBpbmZhbmN5IGFuZCB3ZSBhcmUgbm90IHJlYWR5IGZvciB0aGF0IHlldCwgYnV0IHdlIGhvcGUgYnkgc2hhcmluZyB0aGlzIHZpc2lvbiBlYXJseSwgb3RoZXJzIGNhbiBoZWxwIG1ha2UgaXQgYSByZWFsaXR5LiBGb3IgY29udHJpYnV0aW5nIGluZm9ybWF0aW9uLCBzZWUgW0NPTlRSSUJVVElORy5tZF0oQ09OVFJJQlVUSU5HLm1kKS4KCiMjIENyZWRpdAoKLSBUaGUgW1JlRG9jXShodHRwczovL2dpdGh1Yi5jb20vUmViaWxseS9SZURvYykgcHJvamVjdCBpbnNwaXJlZCBMaW5jb2xuLiBJZiB5b3UncmUgbG9va2luZyBmb3IgYW4gYWx0ZXJuYXRpdmUgcmVuZGVyZXIsIGdpdmUgUmVEb2MgYSB0cnkhCi0gW3N3YWdnZXIyb3BlbmFwaV0oaHR0cHM6Ly9naXRodWIuY29tL21lcm1hZGUvc3dhZ2dlcjJvcGVuYXBpKSB3aGljaCBMaW5jb2xuIHVzZXMgdG8gdmFsaWRhdGUgZGVmaW5pdGlvbnMuCgojIyBNYWludGFpbmVycwoKTGluY29sbiBpcyBhbiBvcGVuIHNvdXJjZSBwcm9qZWN0IGZyb20gW1RlbWFuZG9dKGh0dHA6Ly90ZW1hbmRvLmNvbS8pJ3MgRGV2ZWxvcGVyIEV4cGVyaWVuY2UgdGVhbS4gVGVtYW5kbyBjb25uZWN0cyBjYXJyaWVycyB3aXRoIHJldGFpbGVycyBhbmQgcmV0YWlsZXJzIHRvIHBlb3BsZS4gVGhlIFRlbWFuZG8gUGxhdGZvcm0gY29tYmluZXMgc2hpcHBpbmcgZXhwZXJpZW5jZXMsIG11bHRpLWNhcnJpZXIgY29ubmVjdGl2aXR5IGFuZCBsaWdodG5pbmcgZmFzdCBmdWxmaWxsbWVudCBpbiBvbmUgc29sdXRpb24uIElmIHRoaXMgc291bmRzIGxpa2UgZnVuLCBbd29yayB3aXRoIHVzXShodHRwOi8vdGVtYW5kby5jb20vZW4vYWJvdXQvY2FyZWVycykhCg== readmeEtag: '"11de7dad148f4d27dd30ee8f3c12fc1ebcc37f80"' readmeLastModified: Tue, 05 Mar 2019 18:56:03 GMT repositoryId: 86389954 description: 🎩 A React renderer for OpenAPI v3. created: '2017-03-27T22:26:14Z' updated: '2026-02-05T01:14:35Z' language: JavaScript archived: false stars: 153 watchers: 9 forks: 10 owner: temando logo: https://avatars.githubusercontent.com/u/2232135?v=4 license: MIT repoEtag: '"113d76b753187713136ca31028502dfa5d0efedf018281e011105c7e5a30545d"' repoLastModified: Thu, 05 Feb 2026 01:14:35 GMT id: 094e7e36be3a7e482c070bf15c50dd04 - source: IMPLEMENTATIONS.md name: WebSphere Liberty homepage: https://developer.ibm.com/wasdev/downloads/ language: - JavaScript - Java EE source_description: - >- Includes a native OpenAPI v3 UI which allows for customization of its banners and URL - Generates OpenAPI v3 documentation from Java EE applications category: - User Interfaces - Code Generators foundInMaster: true id: 8fcbff34d56152d1d872d7c64c002c7e - source: IMPLEMENTATIONS.md name: Tcases for OpenAPI homepage: >- https://github.com/Cornutum/tcases/blob/master/tcases-openapi/README.md#tcases-for-openapi-from-rest-ful-to-test-ful language: Java source_description: >- Generates test cases directly from an OpenAPI 3.0.X definition. Creates tests executable using various test frameworks. Bonus: Semantic linter reports elements that are inconsistent, superfluous, or dubious. category: - Testing Tools - Testing foundInMaster: true repository: >- https://github.com/Cornutum/tcases/blob/master/tcases-openapi/README.md#tcases-for-openapi-from-rest-ful-to-test-ful repositoryMetadata: base64Readme: >- IyBUY2FzZXM6IEEgTW9kZWwtQmFzZWQgVGVzdCBDYXNlIEdlbmVyYXRvciAjCgpbIVtNYXZlbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9tYXZlbi00LjEuMS1ncmVlbi5zdmcpXShodHRwczovL3NlYXJjaC5tYXZlbi5vcmcvc2VhcmNoP3E9dGNhc2VzLXNoZWxsKQpbIVtKYXZhZG9jXShodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL2phdmFkb2MtNC4xLjEtZ3JlZW4uc3ZnKV0oaHR0cHM6Ly9qYXZhZG9jLmlvL2RvYy9vcmcuY29ybnV0dW0udGNhc2VzL3RjYXNlcy1zaGVsbCkKCiMjIFdoYXQncyBOZXc/ICMjCiAgKiBUaGUgbGF0ZXN0IHZlcnNpb24gKFtUY2FzZXMgNC4xLjFdKFJlbGVhc2VOb3Rlcy5tZCM0MTEpKSBpcyBub3cgYXZhaWxhYmxlIGF0IHRoZSBNYXZlbiBDZW50cmFsIFJlcG9zaXRvcnkuCiAgICBTZWUgWypIb3cgVG8gRG93bmxvYWQgVGNhc2VzKl0oSG93VG9Eb3dubG9hZC5tZCkgZm9yIGRvd25sb2FkIGluc3RydWN0aW9ucy4KCiAgKiBIYXZpbmcgdHJvdWJsZSB3aXRoIFRjYXNlcz8gQ2hlY2sgb3V0IFt0aGVzZSB0aXBzXSguL1Ryb3VibGVzaG9vdGluZy1GQVFzLm1kKS4KCiAgKiBHb3QgYSBxdWVzdGlvbj8gTmVlZCBzb21lIGd1aWRhbmNlPyBTdGFydCBhIFtkaXNjdXNzaW9uXShodHRwczovL2dpdGh1Yi5jb20vQ29ybnV0dW0vdGNhc2VzL2Rpc2N1c3Npb25zKS4KCiMjIFdoYXQgRG9lcyBJdCBEbz8gIyMKClRjYXNlcyBpcyBhIHRvb2wgZm9yIGRlc2lnbmluZyB0ZXN0cy4gSXQgZG9lc24ndCBtYXR0ZXIgd2hhdCBraW5kIG9mIHN5c3RlbSB5b3UgYXJlIHRlc3RpbmcgLS0gVUksIGNvbW1hbmQgbGluZSwKW1JFU1QtZnVsIEFQSV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKSwgb3IgYmFja2VuZC4gIE5vciBkb2VzIGl0IG1hdHRlcgp3aGF0IGxldmVsIG9mIHRoZSBzeXN0ZW0geW91IGFyZSB0ZXN0aW5nIC0tIHVuaXQsIHN1YnN5c3RlbSwgb3IgZnVsbCBzeXN0ZW0uIFlvdSBjYW4gdXNlIFRjYXNlcyB0byBkZXNpZ24geW91ciB0ZXN0cyBpbiBhbnkgb2YKdGhlc2Ugc2l0dWF0aW9ucy4gV2l0aCBUY2FzZXMsIHlvdSBkZWZpbmUgdGhlIGlucHV0IHNwYWNlIGZvciB5b3VyIHN5c3RlbS11bmRlci10ZXN0IGFuZCB0aGUgbGV2ZWwgb2YgY292ZXJhZ2UgdGhhdCB5b3UKd2FudC4gVGhlbiBUY2FzZXMgZ2VuZXJhdGVzIGEgbWluaW1hbCBzZXQgb2YgdGVzdCBjYXNlcyB0aGF0IG1lZXRzIHlvdXIgcmVxdWlyZW1lbnRzLgoKVGNhc2VzIGlzIHByaW1hcmlseSBhIHRvb2wgZm9yIGJsYWNrLWJveCB0ZXN0IGRlc2lnbi4gRm9yIHN1Y2ggdGVzdHMsIHRoZSBjb25jZXB0IG9mICJjb3ZlcmFnZSIgaXMgZGlmZmVyZW50IGZyb20gc3RydWN0dXJhbAp0ZXN0aW5nIGNyaXRlcmlhIHN1Y2ggYXMgbGluZSBjb3ZlcmFnZSwgYnJhbmNoIGNvdmVyYWdlLCBldGMuIEluc3RlYWQsIFRjYXNlcyBpcyBndWlkZWQgYnkgY292ZXJhZ2Ugb2YgdGhlIGlucHV0IHNwYWNlIG9mIHlvdXIKc3lzdGVtLgoKVGNhc2VzIGdpdmVzIHlvdSBhIHdheSB0byBkZWZpbmUgdGhlIGlucHV0IHNwYWNlIGZvciB5b3VyIHN5c3RlbSBpbiBhIGZvcm0gdGhhdCBpcyBjb25jaXNlIGJ1dCBjb21wcmVoZW5zaXZlLiBUaGVuIFRjYXNlcyBhbGxvd3MKeW91IHRvIGNvbnRyb2wgdGhlIG51bWJlciBvZiB0ZXN0IGNhc2VzIGluIHlvdXIgc2FtcGxlIHN1YnNldCBieSBzcGVjaWZ5aW5nIHRoZSBsZXZlbCBvZiBjb3ZlcmFnZSB5b3Ugd2FudC4gWW91IGNhbiBzdGFydCB3aXRoIGEKYmFzaWMgbGV2ZWwgb2YgY292ZXJhZ2UsIGFuZCBUY2FzZXMgd2lsbCBnZW5lcmF0ZSBhIHNtYWxsIHNldCBvZiB0ZXN0IGNhc2VzIHRoYXQgdG91Y2hlcyBldmVyeSBzaWduaWZpY2FudCBlbGVtZW50IG9mIHRoZSBpbnB1dApzcGFjZS4gVGhlbiB5b3UgY2FuIGltcHJvdmUgeW91ciB0ZXN0cyBieSBzZWxlY3RpdmVseSBhZGRpbmcgY292ZXJhZ2UgaW4gc3BlY2lmaWMgaGlnaC1yaXNrIGFyZWFzLiBGb3IgZXhhbXBsZSwgeW91IGNhbiBzcGVjaWZ5CnBhaXJ3aXNlIGNvdmVyYWdlIG9yIGhpZ2hlci1vcmRlciBjb21iaW5hdGlvbnMgb2Ygc2VsZWN0ZWQgaW5wdXQgdmFyaWFibGVzLgoKIyMgSG93IERvZXMgSXQgV29yaz8gIyMKCkZpcnN0LCB5b3UgY3JlYXRlIGEgc3lzdGVtIGlucHV0IGRlZmluaXRpb24sIGEgZG9jdW1lbnQgdGhhdCBkZWZpbmVzIHlvdXIgc3lzdGVtIGFzIGEgc2V0IG9mIGZ1bmN0aW9ucy4gRm9yIGVhY2ggc3lzdGVtCmZ1bmN0aW9uLCB0aGUgc3lzdGVtIGlucHV0IGRlZmluaXRpb24gZGVmaW5lcyB0aGUgdmFyaWFibGVzIHRoYXQgY2hhcmFjdGVyaXplIHRoZSBmdW5jdGlvbiBpbnB1dCBzcGFjZS4gSWYgeW91IGFyZSB0ZXN0aW5nIGEgV2ViCnNlcnZpY2UgQVBJLCB5b3UgY2FuIGV2ZW4gW2dlbmVyYXRlIGEgc3lzdGVtIGlucHV0IGRlZmluaXRpb24gYXV0b21hdGljYWxseV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKQpmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbi4KClRoZW4sIHlvdSBjYW4gY3JlYXRlIGEgZ2VuZXJhdG9yIGRlZmluaXRpb24uIFRoYXQncyBhbm90aGVyIGRvY3VtZW50IHRoYXQgZGVmaW5lcyB0aGUgY292ZXJhZ2UgeW91IHdhbnQgZm9yIGVhY2ggc3lzdGVtCmZ1bmN0aW9uLiBUaGUgZ2VuZXJhdG9yIGRlZmluaXRpb24gaXMgb3B0aW9uYWwuIFlvdSBjYW4gc2tpcCB0aGlzIHN0ZXAgYW5kIHN0aWxsIGdldCBhIGJhc2ljIGxldmVsIG9mIGNvdmVyYWdlLgoKRmluYWxseSwgeW91IHJ1biBUY2FzZXMuIFRjYXNlcyBpcyBhIEphdmEgcHJvZ3JhbSB0aGF0IHlvdSBjYW4gcnVuIGZyb20gdGhlIGNvbW1hbmQgbGluZSBvciB1c2luZyB0aGUKW1RjYXNlcyBNYXZlbiBQbHVnaW5dKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL3RjYXNlcy1tYXZlbi1wbHVnaW4vKS4gVGhlIGNvbW1hbmQgbGluZSB2ZXJzaW9uIG9mIFRjYXNlcyBjb21lcyB3aXRoIGJ1aWx0LWluCnN1cHBvcnQgZm9yIHJ1bm5pbmcgdXNpbmcgYSBzaGVsbCBzY3JpcHQgb3IgYW4gYW50IHRhcmdldC4gVXNpbmcgeW91ciBpbnB1dCBkZWZpbml0aW9uIGFuZCB5b3VyIGdlbmVyYXRvciBkZWZpbml0aW9uLCBUY2FzZXMKZ2VuZXJhdGVzIGEgc3lzdGVtIHRlc3QgZGVmaW5pdGlvbi4gVGhlIHN5c3RlbSB0ZXN0IGRlZmluaXRpb24gaXMgYSBkb2N1bWVudCB0aGF0IGxpc3RzLCBmb3IgZWFjaCBzeXN0ZW0gZnVuY3Rpb24sIGEgc2V0IG9mIHRlc3QKY2FzZXMgdGhhdCBwcm92aWRlcyB0aGUgc3BlY2lmaWVkIGxldmVsIG9mIGNvdmVyYWdlLiBFYWNoIHRlc3QgY2FzZSBkZWZpbmVzIGEgc3BlY2lmaWMgdmFsdWUgZm9yIGV2ZXJ5IGZ1bmN0aW9uIGlucHV0CnZhcmlhYmxlLiBUY2FzZXMgZ2VuZXJhdGVzIG5vdCBvbmx5IHZhbGlkIGlucHV0IHZhbHVlcyB0aGF0IGRlZmluZSBzdWNjZXNzZnVsIHRlc3QgY2FzZXMgYnV0IGFsc28gaW52YWxpZCB2YWx1ZXMgZm9yIHRoZSB0ZXN0cwpjYXNlcyB0aGF0IGFyZSBuZWVkZWQgdG8gdmVyaWZ5IGV4cGVjdGVkIGVycm9yIGhhbmRsaW5nLgoKT2YgY291cnNlLCB0aGUgc3lzdGVtIHRlc3QgZGVmaW5pdGlvbiBpcyBub3Qgc29tZXRoaW5nIHlvdSBjYW4gZXhlY3V0ZSBkaXJlY3RseS4gKFVubGVzcyBpdCB3YXMKW2Rlcml2ZWQgYXV0b21hdGljYWxseSBmcm9tIGFuIE9wZW5BUEkgZGVmaW5pdGlvbl0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI2hvdy1kby15b3UtcnVuLWdlbmVyYXRlZC1hcGktdGVzdC1jYXNlcykhKQpCdXQgaXQgZm9sbG93cyBhIHdlbGwtZGVmaW5lZCBzY2hlbWEsIHdoaWNoIG1lYW5zIHlvdSBjYW4gdXNlIGEgdmFyaWV0eSBvZiB0cmFuc2Zvcm1hdGlvbiB0b29scyB0byBjb252ZXJ0IGl0IGludG8gYSBmb3JtIHRoYXQKaXMgc3VpdGFibGUgZm9yIHRlc3RpbmcgeW91ciBzeXN0ZW0uIEZvciBleGFtcGxlLCBUY2FzZXMgY29tZXMgd2l0aCBhIGJ1aWx0LWluIHRyYW5zZm9ybWVyIHRoYXQgY29udmVydHMgYSBzeXN0ZW0gdGVzdApkZWZpbml0aW9uIGludG8gYSBKYXZhIHNvdXJjZSBjb2RlIHRlbXBsYXRlIGZvciBhIEpVbml0IG9yIFRlc3RORyB0ZXN0IGNsYXNzLgoKIyMgR2V0IFN0YXJ0ZWQhICMjCgogICogKipUaGUgTG93ZG93bioqCiAgICAqIFtUY2FzZXM6IFRoZSBDb21wbGV0ZSBHdWlkZV0oLi9UY2FzZXMtR3VpZGUubWQjdGNhc2VzLXRoZS1jb21wbGV0ZS1ndWlkZSkKICAgICogW1RjYXNlcyBmb3IgT3BlbkFQSV0odGNhc2VzLW9wZW5hcGkvUkVBRE1FLm1kI3RjYXNlcy1mb3Itb3BlbmFwaS1mcm9tLXJlc3QtZnVsLXRvLXRlc3QtZnVsKTogVGVzdGluZyBhIFJFU1QtZnVsIEFQST8gR2VuZXJhdGUgdGVzdCBjYXNlcyBkaXJlY3RseSBmcm9tIHlvdXIgT3BlbkFQSSB2MyBkZWZpbml0aW9uLgogICAgKiBbVGhlIFRjYXNlcyBNYXZlbiBQbHVnaW5dKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL3RjYXNlcy1tYXZlbi1wbHVnaW4vKQoKICAqICoqSGVscGZ1bCBHdWlkZXMqKgogICAgKiBbSG93IFRvIERvd25sb2FkIFVzaW5nIE1hdmVuXShIb3dUb0Rvd25sb2FkLm1kKQogICAgKiBbSG93IFRvIFNldHVwIGEgVGNhc2VzIFdlYiBTZXJ2aWNlXSguL1RjYXNlcy1XZWItU2VydmljZS5tZCkKICAgICogW1VzaW5nIFRoZSBUY2FzZXMgQVBJXSguL1VzaW5nLVRjYXNlcy1BUEkubWQpCiAgICAqIFtUcm91Ymxlc2hvb3RpbmcgRkFRXSguL1Ryb3VibGVzaG9vdGluZy1GQVFzLm1kI3Ryb3VibGVzaG9vdGluZy1mYXFzKQogICAgKiBbUmVsZWFzZSBOb3Rlc10oUmVsZWFzZU5vdGVzLm1kKQoKICAqICoqTW9yZSBJbmZvKioKICAgICogW01vZGVsLURyaXZlbiBUZXN0aW5nIFVzaW5nIFRjYXNlc10oTW9kZWxEcml2ZW5UZXN0aW5nRm9yQWdpbGVUZWFtcy5tZCkKICAgICogSmF2YWRvYzogW1RjYXNlcyBBUEldKGh0dHA6Ly93d3cuY29ybnV0dW0ub3JnL3RjYXNlcy9kb2NzL2FwaS9pbmRleC5odG1sKQoKIyMgQ29udHJpYnV0b3JzICMjCgpUaGFua3MgdG8gdGhlIGZvbGxvd2luZyBwZW9wbGUsIHdobyBoYXZlIGNvbnRyaWJ1dGVkIHNpZ25pZmljYW50IGltcHJvdmVtZW50cyB0byBUY2FzZXMuCgogICogW0tlcnJ5IEtpbWJyb3VnaF0oaHR0cHM6Ly9naXRodWIuY29tL2tlcnJ5a2ltYnJvdWdoKSAocHJvamVjdCBmb3VuZGVyKQogICogW0p1Z2xhcl0oaHR0cHM6Ly9naXRodWIuY29tL2p1Z2xhcikKICAqIFtUaGliYXVsdCBLcnVzZV0oaHR0cHM6Ly9naXRodWIuY29tL3RrcnVzZSkK readmeEtag: '"ca9a75dabe602853f00d0e6ee6f9a7326d7f4433"' readmeLastModified: Thu, 27 Nov 2025 22:47:20 GMT repositoryId: 32223373 description: A model-based test case generator created: '2015-03-14T17:22:38Z' updated: '2025-12-18T16:57:03Z' language: Java archived: false stars: 236 watchers: 16 forks: 56 owner: Cornutum logo: https://avatars.githubusercontent.com/u/11477145?v=4 license: MIT repoEtag: '"a1d286eef641716d283e62ff50d46ad41827aaf523f41391a8f106f17a7d330b"' repoLastModified: Thu, 18 Dec 2025 16:57:03 GMT id: 14da7572a7632f9a9fb1c1ab2acef586 - source: IMPLEMENTATIONS.md name: Vert.x Web API Contract homepage: http://vertx.io/docs/#web language: - Java - Kotlin - JavaScript - Groovy - Ruby - Ceylon - Scala source_description: >- Create an API endpoint with Vert.x 3 and OpenAPI 3 with automatic requests validation category: Server Implementations foundInMaster: true id: f991830834ff7a8fdec302b6ff07ec42 - source: IMPLEMENTATIONS.md name: Modern homepage: https://github.com/modern-project/modern-ruby language: Ruby source_description: >- OpenAPI 3-based Rack framework with automatic OAS generation and requests/response validation category: Server Implementations foundInMaster: true repository: https://github.com/modern-project/modern-ruby repositoryMetadata: base64Readme: >- IyBNb2Rlcm4gIwoKIyMgVXNpbmcgTW9kZXJuICMjCkEgZGV0YWlsZWQgZGVzY3JpcHRpb24gaXMgb24gdGhlIHdheS4gRm9yIG5vdywgcGxlYXNlIGZlZWwgZnJlZSB0byBwZXJ1c2Ugb3VyClt0ZXN0IGNhc2VzXSBhbmQgdGhlIGJlZ2lubmluZ3Mgb2Ygb3VyIFttYW51YWxdIHRvIGdldCBhbiBpZGVhIG9mIE1vZGVybiBhbmQgaG93Cml0IHdvcmtzLgoKIyMgVmVyc2lvbmluZyAjIwpUaGlzIHByb2plY3QgZm9sbG93cyB0aGUgW1NlbWFudGljIFZlcnNpb25pbmddLCB2ZXJzaW9uIDIuMC4wLCB3aXRoIG9uZQphZGRlbmR1bTogdW50aWwgdGhlIHByb2plY3QgcmVhY2hlcyB0aGUgMS4wLjAgbWFyaywgdGhlIG1pbm9yIHZlcnNpb24gbnVtYmVyCihpLmUuLCAwLnkuKikgZGVub3RlcyBhbiBBUEktYnJlYWtpbmcgY2hhbmdlLgoKIyMgQ29udHJpYnV0aW5nICMjCkJ1ZyByZXBvcnRzIGFuZCBwdWxsIHJlcXVlc3RzIGFyZSB3ZWxjb21lIFtvbiBHaXRIdWJdLiBUaGlzIHByb2plY3QgaXMgaW50ZW5kZWQKdG8gYmUgYSBzYWZlLCB3ZWxjb21pbmcgc3BhY2UgZm9yIGNvbGxhYm9yYXRpb24sIGFuZCBjb250cmlidXRvcnMgYXJlIGV4cGVjdGVkCnRvIGFkaGVyZSB0byB0aGUgW0NvbnRyaWJ1dG9yIENvdmVuYW50XSBjb2RlIG9mIGNvbmR1Y3QuCgpUaGUgYmVzdCB3YXkgdG8gc3RhcnQgY29udHJpYnV0aW5nIGlzIHRvIGxvb2sgZm9yIGBUT0RPYCBjb21tZW50cyBpbiB0aGUgY29kZTsKdGhleSdyZSBzcHJpbmtsZWQgbGliZXJhbGx5IHdoZXJldmVyIGl0IGRpZG4ndCBxdWl0ZSBtYWtlIHNlbnNlIHRvIGRvIHNvbWV0aGluZwp5ZXQgYnV0IHNvbWUgaWRlYXMgbWlnaHQgYmUgbHVya2luZyB0aGF0IGRlc2NyaWJlIGhvdyBpdCBjb3VsZCBiZSBoYW5kbGVkLiBJZgpub3Q/IEZpbGUgYW4gaXNzdWUgYW5kIGxldCdzIGNoYXQgYWJvdXQgaXQhCgojIyBMaWNlbnNlICMjClRoZSBnZW0gaXMgYXZhaWxhYmxlIGFzIG9wZW4gc291cmNlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgW01JVCBMaWNlbnNlXS4KCiMjIENvZGUgb2YgQ29uZHVjdCAjIwpFdmVyeW9uZSBpbnRlcmFjdGluZyBpbiB0aGUgTW9kZXJuIHByb2plY3TigJlzIGNvZGViYXNlcywgaXNzdWUgdHJhY2tlcnMsIGNoYXQKcm9vbXMgYW5kIG1haWxpbmcgbGlzdHMgaXMgZXhwZWN0ZWQgdG8gZm9sbG93IHRoZSBbY29kZSBvZiBjb25kdWN0XS4KClt0ZXN0IGNhc2VzXTogaHR0cHM6Ly9naXRodWIuY29tL2Vyb3BwbGUvbW9kZXJuL3RyZWUvbWFzdGVyL3NwZWMvbW9kZXJuClttYW51YWxdOiBodHRwczovL2dpdGh1Yi5jb20vZXJvcHBsZS9tb2Rlcm4vdHJlZS9tYXN0ZXIvbWFudWFsCltTZW1hbnRpYyBWZXJzaW9uaW5nXTogaHR0cHM6Ly9zZW12ZXIub3JnLwpbb24gR2l0SHViXTogaHR0cHM6Ly9naXRodWIuY29tL2Vyb3BwbGUvbW9kZXJuCltDb250cmlidXRvciBDb3ZlbmFudF06IGh0dHA6Ly9jb250cmlidXRvci1jb3ZlbmFudC5vcmcKW01JVCBMaWNlbnNlXTogaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQKW2NvZGUgb2YgY29uZHVjdF06IGh0dHBzOi8vZ2l0aHViLmNvbS9lcm9wcGxlL21vZGVybi9ibG9iL21hc3Rlci9DT0RFX09GX0NPTkRVQ1QubWQ= readmeEtag: '"f0baa1b74cb7dfb9be1bc8727966e62d1c6260fc"' readmeLastModified: Tue, 26 Feb 2019 05:25:01 GMT repositoryId: 121304397 description: An OpenAPI-first web server for Ruby/Rack. created: '2018-02-12T21:30:10Z' updated: '2024-10-07T05:14:37Z' language: Ruby archived: false stars: 14 watchers: 3 forks: 1 owner: modern-project logo: https://avatars.githubusercontent.com/u/36420497?v=4 license: MIT repoEtag: '"e9e0eae189e1720adf68f0d2a8647811892998133df91243f0dbeeed80851f11"' repoLastModified: Mon, 07 Oct 2024 05:14:37 GMT id: 8bc7fa880ea28bdf448ee5dfd659e352 - source: IMPLEMENTATIONS.md name: Exegesis homepage: https://github.com/exegesis-js/exegesis language: Node.js source_description: OpenAPI 3 server-side framework for express and other frameworks. category: Server Implementations foundInMaster: true repository: https://github.com/exegesis-js/exegesis repositoryMetadata: base64Readme: >- # Exegesis OpenAPI Engine

[![NPM version](https://badge.fury.io/js/exegesis.svg)](https://npmjs.org/package/exegesis)
![Build Status](https://github.com/exegesis-js/exegesis/workflows/GitHub%20CI/badge.svg)
[![Coverage Status](https://coveralls.io/repos/exegesis-js/exegesis/badge.svg)](https://coveralls.io/r/exegesis-js/exegesis)
[![Greenkeeper badge](https://badges.greenkeeper.io/exegesis-js/exegesis.svg)](https://greenkeeper.io/)
[![semantic-release](https://img.shields.io/badge/%20%20%F0%9F%93%A6%F0%9F%9A%80-semantic--release-e10079.svg)](https://github.com/semantic-release/semantic-release)

> ## _exegesis_
>
> _n._ An explanation or critical interpretation of a text, especially an
> API definition document.
>
> -- No dictionary ever

This library implements a framework-agnostic server side implementation of
[OpenAPI 3.x](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.1.md#requestBodyObject).

## Description

Exegesis is a library for implementing server-side OpenAPI 3.x The library has been
written in such a way that hopefully it will also be used to implement future
versions of OpenAPI, or possibly even other API description standards altogether.

You probably don't want to be using this library directly. Have a look at:

- [exegesis-express](https://github.com/exegesis-js/exegesis-express) - Middleware
  for serving OpenAPI 3.x APIs from [express](https://expressjs.com/) or
  [connect](https://github.com/senchalabs/connect).
- [exegesis-koa](https://github.com/confuser/exegesis-koa) - Middleware
  for serving OpenAPI 3.x APIs from [koa](https://koajs.com/).

## Features

- Full support for OpenAPI 3.x.x (see [issues tagged with conformance](https://github.com/exegesis-js/exegesis/issues?q=is%3Aissue+is%3Aopen+label%3Aconformance) for areas which could use some improvement).
- Built in support for "application/json" and "application/x-www-form-urlencoded" requests
- Can use express [body parser middlewares](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md#mimetypeparsers)
- [Response validation](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md#onresponsevalidationerror)
- [Authentication support](https://github.com/exegesis-js/exegesis/blob/master/docs/OAS3%20Security.md)
- [Plugins](https://github.com/exegesis-js/exegesis/tree/master/docs) allow easy extensibility
- Easy support for [validating custom formats](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md#customformats)

## Tutorial

Check out the tutorial [here](https://github.com/exegesis-js/exegesis/blob/master/docs/Tutorial.md).

## API

### compileApi(openApiDoc, options[, done])

This function takes an API document and a set of
[options](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md),
and returns a connect-style middleware function which will execute the API.

`openApiDoc` is either a path to your openapi.yaml or openapi.json file,
or it can be a JSON object with the contents of your OpenAPI document. This
should have the [`x-exegesis-controller`](https://github.com/exegesis-js/exegesis/blob/master/docs/OAS3%20Specification%20Extensions.md)
extension defined on any paths you want to be able to access.

`options` is described in detail [here](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md). At a
minimum, you'll probably want to provide `options.controllers`, a path to where
your [controller modules](https://github.com/exegesis-js/exegesis/blob/master/docs/Exegesis%20Controllers.md)
can be found. If you have any security requirements defined, you'll also
want to pass in some [authenticators](https://github.com/exegesis-js/exegesis/blob/master/docs/OAS3%20Security.md).
To enable response validation, you'll want to provide a validation callback
function via [`onResponseValidationError()`](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md#onresponsevalidationerror).
Exegesis's functionality can also be extended using [plugins](https://github.com/exegesis-js/exegesis/tree/master/docs),
which run on every request. Plugins let you add functionality like
[role base authorization](https://github.com/exegesis-js/exegesis-plugin-roles),
or CORS.

### compileRunner(openApiDoc, options[, done])

This function is similar to `compileApi`; it takes an API document and a set of
[options](https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md),
and returns a "runner". The runner is a `function runner(req, res)`, which takes
in a standard node HTTP request and response. It will not modify the response,
however. Instead it returns (either via callback or Promise) and `HttpResult`
object. This is a `{headers, status, body}` object, where `body` is a readable
stream, read to be piped to the response.

### writeHttpResult(httpResult, res[, done])

A convenience function for writing an `HttpResult` from a runner out to the
response.

## Example

```js
import * as path from 'path';
import * as http from 'http';
import * as exegesis from 'exegesis';

// See https://github.com/exegesis-js/exegesis/blob/master/docs/Options.md
const options = {
  controllers: path.resolve(__dirname, './src/controllers'),
};

// `compileApi()` can either be used with a callback, or if none is provided,
// will return a Promise.
exegesis.compileApi(
  path.resolve(__dirname, './openapi/openapi.yaml'),
  options,
  (err, middleware) => {
    if (err) {
      console.error('Error creating middleware', err.stack);
      process.exit(1);
    }

    const server = http.createServer((req, res) =>
      middleware(req, res, (err) => {
        if (err) {
          res.writeHead(err.status || 500);
          res.end(`Internal error: ${err.message}`);
        } else {
          res.writeHead(404);
          res.end();
        }
      })
    );

    server.listen(3000);
  }
);
```

## Internal Workings

Internally, when you "compile" an API, Exegesis produces an
[ApiInterface](https://github.com/exegesis-js/exegesis/blob/f5266dfd27cdb40c5ebf8063303acbf483d78ed9/src/types/internal.ts#L50) object.
This is an object that, given a method, url, and headers, returns a
[`resolvedOperation`](https://github.com/exegesis-js/exegesis/blob/f5266dfd27cdb40c5ebf8063303acbf483d78ed9/src/types/internal.ts#L21) -
essentially a collection of functions that will parse and validate the body and
parameters, has the controller that executes the functionality, etc... The only
current implementation for an ApiInterface is the
[`oas3/OpenApi` class](https://github.com/exegesis-js/exegesis/blob/master/src/oas3/OpenApi.ts).
Essentially this class's job is to take in an OpenAPI 3.x.x document, and turn it
an ApiInterface that Exegesis can use. In theory, however, we could parse some
other API document format, produce an ApiInterface, and Exegsis would still be
able to run it.
 readmeEtag: '"03673cbb061ddf9537f09b6542f23203b4f5732b"' readmeLastModified: Thu, 27 Jun 2024 19:51:44 GMT repositoryId: 130414902 description: Tools for implementing server-side OpenAPI 3.0.0 created: '2018-04-20T21:19:56Z' updated: '2025-10-22T09:57:37Z' language: TypeScript archived: false stars: 143 watchers: 7 forks: 37 owner: exegesis-js logo: https://avatars.githubusercontent.com/u/38572273?v=4 license: MIT repoEtag: '"610f37d87f641892fd6a4861dcba34c6526ecbf505fb1fcf7d5088d6578eadf2"' repoLastModified: Wed, 22 Oct 2025 09:57:37 GMT id: 3621b0c839384ca811a54a6d5e4f4c0c - source: IMPLEMENTATIONS.md name: PHP-CRUD-API homepage: https://github.com/mevdschee/php-crud-api language: PHP source_description: Automatic CRUD API with OpenAPI 3 docs category: Server Implementations foundInMaster: true repository: https://github.com/mevdschee/php-crud-api repositoryMetadata: base64Readme: >- # PHP-CRUD-API

Single file PHP script that adds a REST API to a MySQL/MariaDB, PostgreSQL, SQL Server or SQLite database. 

Howto: Upload "`api.php`" to your webserver, configure it to connect to your database, have an instant full-featured REST API.

NB: This is the [TreeQL](https://treeql.org) reference implementation in PHP.

## Requirements

  - PHP 7.2 or higher with PDO drivers enabled for one of these database systems:
    - MySQL 5.7 / MariaDB 10.0 or higher for spatial features in MySQL
    - PostgreSQL 9.5 or higher with PostGIS 2.2 or higher for spatial features
    - SQL Server 2017 or higher (2019 also has Linux support)
    - SQLite 3.16 or higher (spatial features NOT supported)

## Installation

Download the "`api.php`" file from the latest release:

https://github.com/mevdschee/php-crud-api/releases/latest or direct from:  
https://raw.githubusercontent.com/mevdschee/php-crud-api/main/api.php

This is a single file application! Upload "`api.php`" somewhere and enjoy!

For local development you may run PHP's built-in web server:

    php -S localhost:8080

Test the script by opening the following URL:

    http://localhost:8080/api.php/records/posts/1

Don't forget to modify the configuration at the bottom of the file.

Alternatively you can integrate this project into the web framework of your choice, see:

- [Automatic REST API for Laravel](https://tqdev.com/2019-automatic-rest-api-laravel)
- [Automatic REST API for Symfony 4](https://tqdev.com/2019-automatic-rest-api-symfony)
- [Automatic REST API for SlimPHP 4](https://tqdev.com/2019-automatic-api-slimphp-4)

In these integrations [Composer](https://getcomposer.org/) is used to load this project as a dependency.

For people that don't use composer, the file "`api.include.php`" is provided. This file contains everything 
from "`api.php`" except the configuration from "`src/index.php`" and can be used by PHP's "include" function.

## Configuration

Edit the following lines in the bottom of the file "`api.php`":

    $config = new Config([
        'username' => 'xxx',
        'password' => 'xxx',
        'database' => 'xxx',
    ]);

These are all the configuration options and their default value between brackets:

- "driver": `mysql`, `pgsql`, `sqlsrv` or `sqlite` (`mysql`)
- "address": Hostname (or filename) of the database server (`localhost`)
- "port": TCP port of the database server (defaults to driver default)
- "username": Username of the user connecting to the database (no default)
- "password": Password of the user connecting to the database (no default)
- "database": Database the connecting is made to (no default)
- "command": Extra SQL to initialize the database connection (none)
- "tables": Comma separated list of tables to publish (defaults to 'all')
- "mapping": Comma separated list of table/column mappings (no mappping)
- "geometrySrid": SRID assumed when converting from WKT to geometry (`4326`)
- "middlewares": List of middlewares to load (`cors`)
- "controllers": List of controllers to load (`records,geojson,openapi,status`)
- "customControllers": List of user custom controllers to load (no default)
- "openApiBase": OpenAPI info (`{"info":{"title":"PHP-CRUD-API","version":"1.0.0"}}`)
- "cacheType": `TempFile`, `Redis`, `Memcache`, `Memcached` or `NoCache` (`TempFile`)
- "cachePath": Path/address of the cache (defaults to system's temp directory)
- "cacheTime": Number of seconds the cache is valid (`10`)
- "jsonOptions": Options used for encoding JSON (`JSON_UNESCAPED_UNICODE`)
- "debug": Show errors in the "X-Exception" headers (`false`)
- "basePath": URI base path of the API (determined using PATH_INFO by default)

All configuration options are also available as environment variables. Write the config option with capitals, a "PHP_CRUD_API_" prefix and underscores for word breakes, so for instance:

- PHP_CRUD_API_DRIVER=mysql
- PHP_CRUD_API_ADDRESS=localhost
- PHP_CRUD_API_PORT=3306
- PHP_CRUD_API_DATABASE=php-crud-api
- PHP_CRUD_API_USERNAME=php-crud-api
- PHP_CRUD_API_PASSWORD=php-crud-api
- PHP_CRUD_API_DEBUG=1

The environment variables take precedence over the PHP configuration.

## Limitations

These limitation and constrains apply:

  - Primary keys should either be auto-increment (from 1 to 2^53) or UUID
  - Composite primary and composite foreign keys are not supported
  - Complex writes (transactions) are not supported
  - Complex queries calling functions (like "concat" or "sum") are not supported
  - Database must support and define foreign key constraints
  - SQLite cannot have bigint typed auto incrementing primary keys
  - SQLite does not support altering table columns (structure)
    
## Features

The following features are supported:

  - Composer install or single PHP file, easy to deploy.
  - Very little code, easy to adapt and maintain
  - Supports POST variables as input (x-www-form-urlencoded)
  - Supports a JSON object as input
  - Supports a JSON array as input (batch insert)
  - Sanitize and validate input using type rules and callbacks
  - Permission system for databases, tables, columns and records
  - Multi-tenant single and multi database layouts are supported
  - Multi-domain CORS support for cross-domain requests
  - Support for reading joined results from multiple tables
  - Search support on multiple criteria
  - Pagination, sorting, top N list and column selection
  - Relation detection with nested results (belongsTo, hasMany and HABTM)
  - Atomic increment support via PATCH (for counters)
  - Binary fields supported with base64 encoding
  - Spatial/GIS fields and filters supported with WKT and GeoJSON
  - Mapping table and column names to support legacy systems
  - Generate API documentation using OpenAPI tools
  - Authentication via API key, JWT token or username/password
  - Database connection parameters may depend on authentication
  - Support for reading database structure in JSON
  - Support for modifying database structure using REST endpoint
  - Security enhancing middleware is included
  - Standard compliant: PSR-4, PSR-7, PSR-12, PSR-15 and PSR-17

## Related projects and ports

Related projects:

  - [PHP-CRUD-API Quick Start](https://github.com/nik2208/php-crud-api-quick-start): A customizable, ready to go, docker compose file featuring PHP-CRUD-API.
  - [PHP-CRUD-API filter generator](https://thipages.github.io/jca-filter/#): A JavaScript library creating PHP-CRUD-API filters from expressions.
  - [JS-CRUD-API](https://github.com/thipages/js-crud-api): A JavaScript client library for the API of PHP-CRUD-API
  - [PHP-API-AUTH](https://github.com/mevdschee/php-api-auth): Single file PHP script that is an authentication provider for PHP-CRUD-API
  - [PHP-CRUD-UI](https://github.com/mevdschee/php-crud-ui): Single file PHP script that adds a UI to a PHP-CRUD-API project.
  - [PHP-CRUD-ADMIN](https://github.com/mevdschee/php-crud-admin): Single file PHP script that adds a database admin interface to a PHP-CRUD-API project.
  - [PHP-SP-API](https://github.com/mevdschee/php-sp-api): Single file PHP script that adds a REST API to a SQL database.
  - [dexie-mysql-sync](https://github.com/scriptPilot/dexie-mysql-sync): Synchronization between local IndexedDB and MySQL Database. 
  - [ra-data-treeql](https://github.com/nkappler/ra-data-treeql): NPM package that provides a [Data Provider](https://marmelab.com/react-admin/DataProviderIntroduction.html) for [React Admin](https://marmelab.com/react-admin/).
  - [scriptPilot/vueuse](https://github.com/scriptPilot/vueuse/): Vue [Composables](https://vuejs.org/guide/reusability/composables.html) in addition to [VueUse.org](https://vueuse.org/) (that support PHP-CRUD-API).
  - [scriptPilot/add-php-backend](https://github.com/scriptPilot/add-php-backend): Add MySQL, phpMyAdmin and PHP-CRUD-API to your dev environment. 
  - [VUE-CRUD-UI](https://github.com/nlware/vue-crud-ui): Single file Vue.js script that adds a UI to a PHP-CRUD-API project.
  
There are also ports of this script in:

- [Go-CRUD-API](https://github.com/dranih/go-crud-api) (work in progress)
- [Java JDBC by Ivan Kolchagov](https://github.com/kolchagov/java-crud-api) (v1)
- [Java Spring Boot + jOOQ](https://github.com/mevdschee/java-crud-api/tree/master/full) (v2: work in progress)

There are also proof-of-concept ports of this script that only support basic REST CRUD functionality in:
[PHP](https://github.com/mevdschee/php-crud-api/blob/master/extras/core.php),
[Java](https://github.com/mevdschee/java-crud-api/blob/master/core/src/main/java/com/tqdev/CrudApiHandler.java),
[Go](https://github.com/mevdschee/go-crud-api/blob/master/api.go),
[C# .net core](https://github.com/mevdschee/core-data-api/blob/master/Program.cs),
[Node.js](https://github.com/mevdschee/js-crud-api/blob/master/app.js) and
[Python](https://github.com/mevdschee/py-crud-api/blob/master/api.py).

## Compilation

You can install all dependencies of this project using the following command:

    php install.php

You can compile all files into a single "`api.php`" file using:

    php build.php

Note that you don't use compilation when you integrate this project into another project or framework (use Composer instead).

### Development

You can access the non-compiled code at the URL:

    http://localhost:8080/src/records/posts/1

The non-compiled code resides in the "`src`" and "`vendor`" directories. The "`vendor`" directory contains the dependencies.

### Updating dependencies

You can update all dependencies of this project using the following command:

    php update.php

This script will install and run [Composer](https://getcomposer.org/) to update the dependencies.

NB: The update script will patch the dependencies in the vendor directory for PHP 7.0 compatibility.

## TreeQL, a pragmatic GraphQL

[TreeQL](https://treeql.org) allows you to create a "tree" of JSON objects based on your SQL database structure (relations) and your query.

It is loosely based on the REST standard and also inspired by json:api.

### CRUD + List

The example posts table has only a a few fields:

    posts  
    =======
    id     
    title  
    content
    created

The CRUD + List operations below act on this table.

#### Create

If you want to create a record the request can be written in URL format as: 

    POST /records/posts

You have to send a body containing:

    {
        "title": "Black is the new red",
        "content": "This is the second post.",
        "created": "2018-03-06T21:34:01Z"
    }

And it will return the value of the primary key of the newly created record:

    2

#### Read

To read a record from this table the request can be written in URL format as:

    GET /records/posts/1

Where "1" is the value of the primary key of the record that you want to read. It will return:

    {
        "id": 1
        "title": "Hello world!",
        "content": "Welcome to the first post.",
        "created": "2018-03-05T20:12:56Z"
    }

On read operations you may apply joins.

#### Update

To update a record in this table the request can be written in URL format as:

    PUT /records/posts/1

Where "1" is the value of the primary key of the record that you want to update. Send as a body:

    {
        "title": "Adjusted title!"
    }

This adjusts the title of the post. And the return value is the number of rows that are set:

    1

#### Delete

If you want to delete a record from this table the request can be written in URL format as:

    DELETE /records/posts/1

And it will return the number of deleted rows:

    1

#### List

To list records from this table the request can be written in URL format as:

    GET /records/posts

It will return:

    {
        "records":[
            {
                "id": 1,
                "title": "Hello world!",
                "content": "Welcome to the first post.",
                "created": "2018-03-05T20:12:56Z"
            }
        ]
    }

On list operations you may apply filters and joins.

### Filters

Filters provide search functionality, on list calls, using the "filter" parameter. You need to specify the column
name, a comma, the match type, another comma and the value you want to filter on. These are supported match types:

  - "cs": contain string (string contains value)
  - "sw": start with (string starts with value)
  - "ew": end with (string end with value)
  - "eq": equal (string or number matches exactly)
  - "lt": lower than (number is lower than value)
  - "le": lower or equal (number is lower than or equal to value)
  - "ge": greater or equal (number is higher than or equal to value)
  - "gt": greater than (number is higher than value)
  - "bt": between (number is between two comma separated values)
  - "in": in (number or string is in comma separated list of values)
  - "is": is null (field contains "NULL" value)

You can negate all filters by prepending a "n" character, so that "eq" becomes "neq". 
Examples of filter usage are:

    GET /records/categories?filter=name,eq,Internet
    GET /records/categories?filter=name,sw,Inter
    GET /records/categories?filter=id,le,1
    GET /records/categories?filter=id,ngt,1
    GET /records/categories?filter=id,bt,0,1
    GET /records/categories?filter=id,in,0,1

Output:

    {
        "records":[
            {
                "id": 1
                "name": "Internet"
            }
        ]
    }

In the next section we dive deeper into how you can apply multiple filters on a single list call.

### Multiple filters

Filters can be a by applied by repeating the "filter" parameter in the URL. For example the following URL: 

    GET /records/categories?filter=id,gt,1&filter=id,lt,3

will request all categories "where id > 1 and id < 3". If you wanted "where id = 2 or id = 4" you should write:

    GET /records/categories?filter1=id,eq,2&filter2=id,eq,4
    
As you see we added a number to the "filter" parameter to indicate that "OR" instead of "AND" should be applied.
Note that you can also repeat "filter1" and create an "AND" within an "OR". Since you can also go one level deeper
by adding a letter (a-f) you can create almost any reasonably complex condition tree.

NB: You can only filter on the requested table (not on it's included tables) and filters are only applied on list calls.

### Column selection

By default all columns are selected. With the "include" parameter you can select specific columns. 
You may use a dot to separate the table name from the column name. Multiple columns should be comma separated. 
An asterisk ("*") may be used as a wildcard to indicate "all columns". Similar to "include" you may use the "exclude" parameter to remove certain columns:

```
GET /records/categories/1?include=name
GET /records/categories/1?include=categories.name
GET /records/categories/1?exclude=categories.id
```

Output:

```
    {
        "name": "Internet"
    }
```

NB: Columns that are used to include related entities are automatically added and cannot be left out of the output.

### Ordering

With the "order" parameter you can sort. By default the sort is in ascending order, but by specifying "desc" this can be reversed:

```
GET /records/categories?order=name,desc
GET /records/categories?order=id,desc&order=name
```

Output:

```
    {
        "records":[
            {
                "id": 3
                "name": "Web development"
            },
            {
                "id": 1
                "name": "Internet"
            }
        ]
    }
```

NB: You may sort on multiple fields by using multiple "order" parameters. You can not order on "joined" columns.

### Limit size

The "size" parameter limits the number of returned records. This can be used for top N lists together with the "order" parameter (use descending order).

```
GET /records/categories?order=id,desc&size=1
```

Output:

```
    {
        "records":[
            {
                "id": 3
                "name": "Web development"
            }
        ]
    }
```

NB: If you also want to know to the total number of records you may want to use the "page" parameter.

### Pagination

The "page" parameter holds the requested page. The default page size is 20, but can be adjusted (e.g. to 50).

```
GET /records/categories?order=id&page=1
GET /records/categories?order=id&page=1,50
```

Output:

```
    {
        "records":[
            {
                "id": 1
                "name": "Internet"
            },
            {
                "id": 3
                "name": "Web development"
            }
        ],
        "results": 2
    }
```

The element "results" holds to total number of records in the table, which would be returned if no pagination would be used.

NB: Since pages that are not ordered cannot be paginated, pages will be ordered by primary key.

### Joins

Let's say that you have a posts table that has comments (made by users) and the posts can have tags.

    posts    comments  users     post_tags  tags
    =======  ========  =======   =========  ======= 
    id       id        id        id         id
    title    post_id   username  post_id    name
    content  user_id   phone     tag_id
    created  message

When you want to list posts with their comments users and tags you can ask for two "tree" paths:

    posts -> comments  -> users
    posts -> post_tags -> tags

These paths have the same root and this request can be written in URL format as:

    GET /records/posts?join=comments,users&join=tags

Here you are allowed to leave out the intermediate table that binds posts to tags. In this example
you see all three table relation types (hasMany, belongsTo and hasAndBelongsToMany) in effect:

- "post" has many "comments"
- "comment" belongs to "user"
- "post" has and belongs to many "tags"

This may lead to the following JSON data:

    {
        "records":[
            {
                "id": 1,
                "title": "Hello world!",
                "content": "Welcome to the first post.",
                "created": "2018-03-05T20:12:56Z",
                "comments": [
                    {
                        id: 1,
                        post_id: 1,
                        user_id: {
                            id: 1,
                            username: "mevdschee",
                            phone: null,
                        },
                        message: "Hi!"
                    },
                    {
                        id: 2,
                        post_id: 1,
                        user_id: {
                            id: 1,
                            username: "mevdschee",
                            phone: null,
                        },
                        message: "Hi again!"
                    }
                ],
                "tags": []
            },
            {
                "id": 2,
                "title": "Black is the new red",
                "content": "This is the second post.",
                "created": "2018-03-06T21:34:01Z",
                "comments": [],
                "tags": [
                    {
                        id: 1,
                        message: "Funny"
                    },
                    {
                        id: 2,
                        message: "Informational"
                    }
                ]
            }
        ]
    }

You see that the "belongsTo" relationships are detected and the foreign key value is replaced by the referenced object.
In case of "hasMany" and "hasAndBelongsToMany" the table name is used a new property on the object.

### Batch operations

When you want to create, read, update or delete you may specify multiple primary key values in the URL.
You also need to send an array instead of an object in the request body for create and update. 

To read a record from this table the request can be written in URL format as:

    GET /records/posts/1,2

The result may be:

    [
            {
                "id": 1,
                "title": "Hello world!",
                "content": "Welcome to the first post.",
                "created": "2018-03-05T20:12:56Z"
            },
            {
                "id": 2,
                "title": "Black is the new red",
                "content": "This is the second post.",
                "created": "2018-03-06T21:34:01Z"
            }
    ]

Similarly when you want to do a batch update the request in URL format is written as:

    PUT /records/posts/1,2

Where "1" and "2" are the values of the primary keys of the records that you want to update. The body should 
contain the same number of objects as there are primary keys in the URL:

    [   
        {
            "title": "Adjusted title for ID 1"
        },
        {
            "title": "Adjusted title for ID 2"
        }        
    ]

This adjusts the titles of the posts. And the return values are the number of rows that are set:

    [1,1]

Which means that there were two update operations and each of them had set one row. Batch operations use database
transactions, so they either all succeed or all fail (successful ones get rolled back). If they fail the body will
contain the list of error documents. In the following response the first operation succeeded and the second operation
of the batch failed due to an integrity violation:

    [   
        {
            "code": 0,
            "message": "Success"
        },
        {
            "code": 1010,
            "message": "Data integrity violation"
        }
    ]

The response status code will always be 424 (failed dependency) in case of any failure of one of the batch operations.

To insert multiple records into this table the request can be written in URL format as:

    POST /records/posts

The body should contain an array of records to be inserted:

    [
            {
                "title": "Hello world!",
                "content": "Welcome to the first post.",
                "created": "2018-03-05T20:12:56Z"
            },
            {
                "title": "Black is the new red",
                "content": "This is the second post.",
                "created": "2018-03-06T21:34:01Z"
            }
    ]

The return value is also an array containing the primary keys of the newly inserted records:

    [1,2] 

Note that batch operation for DELETE follows the same pattern as PUT, but without a body.

### Spatial support

For spatial support there is an extra set of filters that can be applied on geometry columns and that starting with an "s":

  - "sco": spatial contains (geometry contains another)
  - "scr": spatial crosses (geometry crosses another)
  - "sdi": spatial disjoint (geometry is disjoint from another)
  - "seq": spatial equal (geometry is equal to another)
  - "sin": spatial intersects (geometry intersects another)
  - "sov": spatial overlaps (geometry overlaps another)
  - "sto": spatial touches (geometry touches another)
  - "swi": spatial within (geometry is within another)
  - "sic": spatial is closed (geometry is closed and simple)
  - "sis": spatial is simple (geometry is simple)
  - "siv": spatial is valid (geometry is valid)

These filters are based on OGC standards and so is the WKT specification in which the geometry columns are represented.
Note that the SRID that is assumed when converting from WKT to geometry is specified by the config variable `geometrySrid` and defaults to 4326 (WGS 84).

#### GeoJSON

The GeoJSON support is a read-only view on the tables and records in GeoJSON format. These requests are supported:

    method path                  - operation - description
    ----------------------------------------------------------------------------------------
    GET    /geojson/{table}      - list      - lists records as a GeoJSON FeatureCollection
    GET    /geojson/{table}/{id} - read      - reads a record by primary key as a GeoJSON Feature

The "`/geojson`" endpoint uses the "`/records`" endpoint internally and inherits all functionality, such as joins and filters.
It also supports a "geometry" parameter to indicate the name of the geometry column in case the table has more than one.
For map views it supports the "bbox" parameter in which you can specify upper-left and lower-right coordinates (comma separated).
The following Geometry types are supported by the GeoJSON implementation:

  - Point
  - MultiPoint
  - LineString
  - MultiLineString
  - Polygon
  - MultiPolygon

The GeoJSON functionality is enabled by default, but can be disabled using the "controllers" configuration.

## Mapping names for legacy systems

To support creating an API for (a part of) a legacy system (such as Wordpress) you may want to map the table and column 
names as can not improve them without changing the software, while the names may need some improvement for consistency.
The config allows you to rename tables and columns with a comma separated list of mappings that are split with an 
equal sign, like this:

    'mapping' => 'wp_posts=posts,wp_posts.ID=posts.id',

This specific example will expose the "`wp_posts`" table at a "`posts`" end-point (instead of "`wp_posts`") and the 
column "`ID`" within that table as the "`id`" property (in lower case instead of upper case).

NB: Since these two mappings overlap the first (less specific) mapping may be omitted.

## Middleware

You can enable the following middleware using the "middlewares" config parameter:

- "firewall": Limit access to specific IP addresses
- "sslRedirect": Force connection over HTTPS instead of HTTP
- "cors": Support for CORS requests (enabled by default)
- "xsrf": Block XSRF attacks using the 'Double Submit Cookie' method
- "ajaxOnly": Restrict non-AJAX requests to prevent XSRF attacks
- "apiKeyAuth": Support for "API Key Authentication"
- "apiKeyDbAuth": Support for "API Key Database Authentication"
- "dbAuth": Support for "Database Authentication"
- "wpAuth": Support for "Wordpress Authentication"
- "jwtAuth": Support for "JWT Authentication"
- "basicAuth": Support for "Basic Authentication"
- "reconnect": Reconnect to the database with different parameters
- "authorization": Restrict access to certain tables or columns
- "validation": Return input validation errors for custom rules and default type rules
- "ipAddress": Fill a protected field with the IP address on create
- "sanitation": Apply input sanitation on create and update
- "multiTenancy": Restricts tenants access in a multi-tenant scenario
- "pageLimits": Restricts list operations to prevent database scraping
- "joinLimits": Restricts join parameters to prevent database scraping
- "textSearch": Search in all text fields with a simple parameter
- "customization": Provides handlers for request and response customization
- "json": Support read/write of JSON strings as JSON objects/arrays
- "xml": Translates all input and output from JSON to XML

The "middlewares" config parameter is a comma separated list of enabled middlewares.
You can tune the middleware behavior using middleware specific configuration parameters:

- "firewall.reverseProxy": Set to "true" when a reverse proxy is used ("")
- "firewall.allowedIpAddresses": List of IP addresses that are allowed to connect ("")
- "cors.allowedOrigins": The origins allowed in the CORS headers ("*")
- "cors.allowHeaders": The headers allowed in the CORS request ("Content-Type, X-XSRF-TOKEN, X-Authorization")
- "cors.allowMethods": The methods allowed in the CORS request ("OPTIONS, GET, PUT, POST, DELETE, PATCH")
- "cors.allowCredentials": To allow credentials in the CORS request ("true")
- "cors.exposeHeaders": Whitelist headers that browsers are allowed to access ("")
- "cors.maxAge": The time that the CORS grant is valid in seconds ("1728000")
- "xsrf.excludeMethods": The methods that do not require XSRF protection ("OPTIONS,GET")
- "xsrf.cookieName": The name of the XSRF protection cookie ("XSRF-TOKEN")
- "xsrf.headerName": The name of the XSRF protection header ("X-XSRF-TOKEN")
- "ajaxOnly.excludeMethods": The methods that do not require AJAX ("OPTIONS,GET")
- "ajaxOnly.headerName": The name of the required header ("X-Requested-With")
- "ajaxOnly.headerValue": The value of the required header ("XMLHttpRequest")
- "apiKeyAuth.mode": Set to "optional" if you want to allow anonymous access ("required")
- "apiKeyAuth.header": The name of the API key header ("X-API-Key")
- "apiKeyAuth.keys": List of API keys that are valid ("")
- "apiKeyDbAuth.mode": Set to "optional" if you want to allow anonymous access ("required")
- "apiKeyDbAuth.header": The name of the API key header ("X-API-Key")
- "apiKeyDbAuth.usersTable": The table that is used to store the users in ("users")
- "apiKeyDbAuth.apiKeyColumn": The users table column that holds the API key ("api_key")
- "dbAuth.mode": Set to "optional" if you want to allow anonymous access ("required")
- "dbAuth.usersTable": The table that is used to store the users in ("users")
- "dbAuth.loginTable": The table or view that is used to retrieve the users info for login ("users")
- "dbAuth.usernameColumn": The users table column that holds usernames ("username")
- "dbAuth.passwordColumn": The users table column that holds passwords ("password")
- "dbAuth.returnedColumns": The columns returned on successful login, empty means 'all' ("")
- "dbAuth.usernameFormField": The name of the form field that holds the username ("username")
- "dbAuth.passwordFormField": The name of the form field that holds the password ("password")
- "dbAuth.newPasswordFormField": The name of the form field that holds the new password ("newPassword")
- "dbAuth.registerUser": JSON user data (or "1") in case you want the /register endpoint enabled ("")
- "dbAuth.loginAfterRegistration": 1 or zero if registered users should be logged in after registration ("")
- "dbAuth.passwordLength": Minimum length that the password must have ("12")
- "dbAuth.sessionName": The name of the PHP session that is started ("")
- "wpAuth.mode": Set to "optional" if you want to allow anonymous access ("required")
- "wpAuth.wpDirectory": The folder/path where the Wordpress install can be found (".")
- "wpAuth.usernameFormField": The name of the form field that holds the username ("username")
- "wpAuth.passwordFormField": The name of the form field that holds the password ("password")
- "jwtAuth.mode": Set to "optional" if you want to allow anonymous access ("required")
- "jwtAuth.header": Name of the header containing the JWT token ("X-Authorization")
- "jwtAuth.leeway": The acceptable number of seconds of clock skew ("5")
- "jwtAuth.ttl": The number of seconds the token is valid ("30")
- "jwtAuth.secrets": The shared secret(s) used to sign the JWT token with ("")
- "jwtAuth.algorithms": The algorithms that are allowed, empty means 'all' ("")
- "jwtAuth.audiences": The audiences that are allowed, empty means 'all' ("")
- "jwtAuth.issuers": The issuers that are allowed, empty means 'all' ("")
- "jwtAuth.sessionName": The name of the PHP session that is started ("")
- "basicAuth.mode": Set to "optional" if you want to allow anonymous access ("required")
- "basicAuth.realm": Text to prompt when showing login ("Username and password required")
- "basicAuth.passwordFile": The file to read for username/password combinations (".htpasswd")
- "basicAuth.sessionName": The name of the PHP session that is started ("")
- "reconnect.driverHandler": Handler to implement retrieval of the database driver ("")
- "reconnect.addressHandler": Handler to implement retrieval of the database address ("")
- "reconnect.portHandler": Handler to implement retrieval of the database port ("")
- "reconnect.databaseHandler": Handler to implement retrieval of the database name ("")
- "reconnect.tablesHandler": Handler to implement retrieval of the table names ("")
- "reconnect.mappingHandler": Handler to implement retrieval of the name mapping ("")
- "reconnect.usernameHandler": Handler to implement retrieval of the database username ("")
- "reconnect.passwordHandler": Handler to implement retrieval of the database password ("")
- "authorization.tableHandler": Handler to implement table authorization rules ("")
- "authorization.columnHandler": Handler to implement column authorization rules ("")
- "authorization.pathHandler": Handler to implement path authorization rules ("")
- "authorization.recordHandler": Handler to implement record authorization filter rules ("")
- "validation.handler": Handler to implement validation rules for input values ("")
- "validation.types": Types to enable type validation for, empty means 'none' ("all")
- "validation.tables": Tables to enable type validation for, empty means 'none' ("all")
- "ipAddress.tables": Tables to search for columns to override with IP address ("")
- "ipAddress.columns": Columns to protect and override with the IP address on create ("")
- "sanitation.handler": Handler to implement sanitation rules for input values ("")
- "sanitation.types": Types to enable type sanitation for, empty means 'none' ("all")
- "sanitation.tables": Tables to enable type sanitation for, empty means 'none' ("all")
- "multiTenancy.handler": Handler to implement simple multi-tenancy rules ("")
- "pageLimits.pages": The maximum page number that a list operation allows ("100")
- "pageLimits.records": The maximum number of records returned by a list operation ("1000")
- "joinLimits.depth": The maximum depth (length) that is allowed in a join path ("3")
- "joinLimits.tables": The maximum number of tables that you are allowed to join ("10")
- "joinLimits.records": The maximum number of records returned for a joined entity ("1000")
- "textSearch.parameter": The name of the parameter used for the search term ("search")
- "customization.beforeHandler": Handler to implement request customization ("")
- "customization.afterHandler": Handler to implement response customization ("")
- "json.controllers": Controllers to process JSON strings for ("records,geojson")
- "json.tables": Tables to process JSON strings for ("all")
- "json.columns": Columns to process JSON strings for ("all")
- "xml.types": JSON types that should be added to the XML type attribute ("null,array")

If you don't specify these parameters in the configuration, then the default values (between brackets) are used.

In the sections below you find more information on the built-in middleware.

### Authentication

Currently there are five types of authentication supported. They all store the authenticated user in the `$_SESSION` super global.
This variable can be used in the authorization handlers to decide wether or not somebody should have read or write access to certain tables, columns or records.
The following overview shows the kinds of authentication middleware that you can enable.

| Name       | Middleware   | Authenticated via      | Users are stored in | Session variable        |
| ---------- | ------------ | ---------------------- | ------------------- | ----------------------- |
| API key    | apiKeyAuth   | 'X-API-Key' header     | configuration       | `$_SESSION['apiKey']`   |
| API key DB | apiKeyDbAuth | 'X-API-Key' header     | database table      | `$_SESSION['apiUser']`  |
| Database   | dbAuth       | '/login' endpoint      | database table      | `$_SESSION['user']`     |
| Basic      | basicAuth    | 'Authorization' header | '.htpasswd' file    | `$_SESSION['username']` |
| JWT        | jwtAuth      | 'Authorization' header | identity provider   | `$_SESSION['claims']`   |

Below you find more information on each of the authentication types.

#### API key authentication

API key authentication works by sending an API key in a request header.
The header name defaults to "X-API-Key" and can be configured using the 'apiKeyAuth.header' configuration parameter.
Valid API keys must be configured using the 'apiKeyAuth.keys' configuration parameter (comma separated list).

    X-API-Key: 02c042aa-c3c2-4d11-9dae-1a6e230ea95e

The authenticated API key will be stored in the `$_SESSION['apiKey']` variable.

Note that the API key authentication does not require or use session cookies.

#### API key database authentication

API key database authentication works by sending an API key in a request header "X-API-Key" (the name is configurable).
Valid API keys are read from the database from the column "api_key" of the "users" table (both names are configurable).

    X-API-Key: 02c042aa-c3c2-4d11-9dae-1a6e230ea95e

The authenticated user (with all it's properties) will be stored in the `$_SESSION['apiUser']` variable.

Note that the API key database authentication does not require or use session cookies.

#### Database authentication

The database authentication middleware defines five new routes:

    method path       - parameters                      - description
    ---------------------------------------------------------------------------------------------------
    GET    /me        -                                 - returns the user that is currently logged in
    POST   /register  - username, password              - adds a user with given username and password
    POST   /login     - username, password              - logs a user in by username and password
    POST   /password  - username, password, newPassword - updates the password of the logged in user
    POST   /logout    -                                 - logs out the currently logged in user

A user can be logged in by sending it's username and password to the login endpoint (in JSON format).
The authenticated user (with all it's properties) will be stored in the `$_SESSION['user']` variable.
The user can be logged out by sending a POST request with an empty body to the logout endpoint.
The passwords are stored as hashes in the password column in the users table. You can register a new user
using the register endpoint, but this functionality must be turned on using the "dbAuth.registerUser"
configuration parameter.

It is IMPORTANT to restrict access to the users table using the 'authorization' middleware, otherwise all 
users can freely add, modify or delete any account! The minimal configuration is shown below:

    'middlewares' => 'dbAuth,authorization',
    'authorization.tableHandler' => function ($operation, $tableName) {
        return $tableName != 'users';
    },

Note that this middleware uses session cookies and stores the logged in state on the server.

**Login using views with joined table**

For login operations, it is possible to use a view as the usersTable. Such view can return a filtered result from the users table, e.g., *where active = true* or it may also return a result multiple tables thru a table join. At a minimum, the view should include the ***username*** and ***password*** and a field named ***id***.

However, views with joined tables are not insertable ([see issue 907](https://github.com/mevdschee/php-crud-api/issues/907) ). As a workaround, use the property ***loginTable*** to set a different reference table for login. The **usersTable** will still be set to the normal, insertable users table. 

#### Wordpress authentication

The Wordpress authentication middleware defines three routes:

    method path       - parameters                      - description
    ---------------------------------------------------------------------------------------------------
    GET    /me        -                                 - returns the user that is currently logged in
    POST   /login     - username, password              - logs a user in by username and password
    POST   /logout    -                                 - logs out the currently logged in user

A user can be logged in by sending it's username and password to the login endpoint (in JSON format).
The user can be logged out by sending a POST request with an empty body to the logout endpoint.
You need to specify the Wordpress installation directory using the "wpAuth.wpDirectory" configuration parameter.
The middleware calls "wp-load.php" this allows you to use Wordpress functions in the authorization middleware, like:

- wp_get_current_user()
- is_user_logged_in()
- is_super_admin()
- user_can(wp_get_current_user(),'edit_posts');

Note that the `$_SESSION` variable is not used by this middleware.

#### Basic authentication

The Basic type supports a file (by default '.htpasswd') that holds the users and their (hashed) passwords separated by a colon (':'). 
When the passwords are entered in plain text they will be automatically hashed.
The authenticated username will be stored in the `$_SESSION['username']` variable.
You need to send an "Authorization" header containing a base64 url encoded version of your colon separated username and password, after the word "Basic".

    Authorization: Basic dXNlcm5hbWUxOnBhc3N3b3JkMQ

This example sends the string "username1:password1".

#### JWT authentication

The JWT type requires another (SSO/Identity) server to sign a token that contains claims. 
Both servers share a secret so that they can either sign or verify that the signature is valid.
Claims are stored in the `$_SESSION['claims']` variable. You need to send an "X-Authorization" 
header containing a base64 url encoded and dot separated token header, body and signature after
the word "Bearer" ([read more about JWT here](https://jwt.io/)). The standard says you need to
use the "Authorization" header, but this is problematic in Apache and PHP.

    X-Authorization: Bearer eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJzdWIiOiIxMjM0NTY3ODkwIiwibmFtZSI6IkpvaG4gRG9lIiwiYWRtaW4iOnRydWUsImlhdCI6IjE1MzgyMDc2MDUiLCJleHAiOjE1MzgyMDc2MzV9.Z5px_GT15TRKhJCTHhDt5Z6K6LRDSFnLj8U5ok9l7gw

This example sends the signed claims:

    {
      "sub": "1234567890",
      "name": "John Doe",
      "admin": true,
      "iat": "1538207605",
      "exp": 1538207635
    }

NB: The JWT implementation only supports the RSA and HMAC based algorithms.

##### Configure and test JWT authentication with Auth0

First you need to create an account on [Auth0](https://auth0.com/auth/login).
Once logged in, you have to create an application (its type does not matter). Collect the `Domain`
and `Client ID` and keep them for a later use. Then, create an API: give it a name and fill the
`identifier` field with your API endpoint's URL.

Then you have to configure the `jwtAuth.secrets` configuration in your `api.php` file.
Don't fill it with the `secret` you will find in your Auth0 application settings but with **a
public certificate**. To find it, go to the settings of your application, then in "Extra settings".
You will now find a "Certificates" tab where you will find your Public Key in the Signing
Certificate field.

To test your integration, you can copy the [auth0/vanilla.html](examples/clients/auth0/vanilla.html)
file. Be sure to fill these three variables:

 - `authUrl` with your Auth0 domain
 - `clientId` with your Client ID
 - `audience` with the API URL you created in Auth0

Note that if you don't fill the audience parameter, it will not work because you won't get a valid JWT.
Also note that you should fill `jwtAuth.audiences` (with the value of the `audience`) to ensure the
tokens are validated to be generated for your application.

You can also change the `url` variable, used to test the API with authentication.

[More info](https://auth0.com/docs/api-auth/tutorials/verify-access-token)

##### Configure and test JWT authentication with Firebase

First you need to create a Firebase project on the [Firebase console](https://console.firebase.google.com/).
Add a web application to this project and grab the code snippet for later use.

Then you have to configure the `jwtAuth.secrets` configuration in your `api.php` file. 
This can be done as follows:

a. Log a user in to your Firebase-based app, get an authentication token for that user
b. Go to [https://jwt.io/](https://jwt.io/) and paste the token in the decoding field
c. Read the decoded header information from the token, it will give you the correct `kid`
d. Grab the public key via this [URL](https://www.googleapis.com/robot/v1/metadata/x509/securetoken@system.gserviceaccount.com), which corresponds to your `kid` from previous step
e. Now, just fill `jwtAuth.secrets` with your public key in the `api.php`

Also configure the `jwtAuth.audiences` (fill in the Firebase project ID).

Here is an example of what it should look like in the configuration:

```
...,
'middlewares' => 'cors, jwtAuth, authorization',
        'jwtAuth.secrets' => "ce5ced6e40dcd1eff407048867b1ed1e706686a0:-----BEGIN CERTIFICATE-----\nMIIDHDCCAgSgAwIBAgIIExun9bJSK1wwDQYJKoZIhvcNAQEFBQAwMTEvMC0GA1UE\nAxMmc2VjdXJldG9rZW4uc3lzdGVtLmdzZXJ2aWNlYWNjb3VudC5jb20wHhcNMTkx\nMjIyMjEyMTA3WhcNMjAwMTA4MDkzNjA3WjAxMS8wLQYDVQQDEyZzZWN1cmV0b2tl\nbi5zeXN0ZW0uZ3NlcnZpY2VhY2NvdW50LmNvbTCCASIwDQYJKoZIhvcNAQEBBQAD\nggEPADCCAQoCggEBAKsvVDUwXeYQtySNvyI1/tZAk0sj7Zx4/1+YLUomwlK6vmEd\nyl2IXOYOj3VR7FBA24A9//nnrp+mV8YOYEOdaWX7PQo0PIPFPqdA0r7CqBUWHPfQ\n1WVHVRQY3G0c7upM97UfMes9xOrMqyvecMRk1e5S6eT12Zh2og7yiVs8gP83M1EB\nGqseUaltaadjyT35w5B0Ny0/7NdLYiv2G6Z0S821SxvSo1/wfmilnBBKYYluP0PA\n9NPznWFP6uXnX7gKxyJT9//cYVxTO6+b1TT13Yvrpm1a4EuCOhLrZH6ErHQTccAM\nhAx8mdNtbROsp0dlPKrSfqO82uFz45RXZYmSeP0CAwEAAaM4MDYwDAYDVR0TAQH/\nBAIwADAOBgNVHQ8BAf8EBAMCB4AwFgYDVR0lAQH/BAwwCgYIKwYBBQUHAwIwDQYJ\nKoZIhvcNAQEFBQADggEBACNsJ5m00gdTvD6j6ahURsGrNZ0VJ0YREVQ5U2Jtubr8\nn2fuhMxkB8147ISzfi6wZR+yNwPGjlr8JkAHAC0i+Nam9SqRyfZLqsm+tHdgFT8h\npa+R/FoGrrLzxJNRiv0Trip8hZjgz3PClz6KxBQzqL+rfGV2MbwTXuBoEvLU1mYA\no3/UboJT7cNGjZ8nHXeoKMsec1/H55lUdconbTm5iMU1sTDf+3StGYzTwC+H6yc2\nY3zIq3/cQUCrETkALrqzyCnLjRrLYZu36ITOaKUbtmZhwrP99i2f+H4Ab2i8jeMu\nk61HD29mROYjl95Mko2BxL+76To7+pmn73U9auT+xfA=\n-----END CERTIFICATE-----\n",
        'jwtAuth.audiences' => 'your-project-id',
        'cors.allowedOrigins' => '*',
        'cors.allowHeaders' => 'X-Authorization'
```

Notes:
 - The `kid:key` pair is formatted as a string
 - Do not include spaces before or after the ':'
 - Use double quotation marks (") around the string text
 - The string must contain the linefeeds (\n)
 - `jwtAuth.audiences` should contain your Firebase projectId

To test your integration, you can copy the [firebase/vanilla.html](examples/clients/firebase/vanilla.html)
file and the [firebase/vanilla-success.html](examples/clients/firebase/vanilla-success.html) file,
used as a "success" page and to display the API result.

Replace, in both files, the Firebase configuration (`firebaseConfig` object).

You can also change the `url` variable, used to test the API with authentication.

[More info](https://firebase.google.com/docs/auth/admin/verify-id-tokens#verify_id_tokens_using_a_third-party_jwt_library)

### Authorizing operations

The Authorization model acts on "operations". The most important ones are listed here:

    method path                  - operation - description
    ----------------------------------------------------------------------------------------
    GET    /records/{table}      - list      - lists records
    POST   /records/{table}      - create    - creates records
    GET    /records/{table}/{id} - read      - reads a record by primary key
    PUT    /records/{table}/{id} - update    - updates columns of a record by primary key
    DELETE /records/{table}/{id} - delete    - deletes a record by primary key
    PATCH  /records/{table}/{id} - increment - increments columns of a record by primary key

The "`/openapi`" endpoint will only show what is allowed in your session. It also has a special 
"document" operation to allow you to hide tables and columns from the documentation.
    
For endpoints that start with "`/columns`" there are the operations "reflect" and "remodel". 
These operations can display or change the definition of the database, table or column. 
This functionality is disabled by default and for good reason (be careful!). 
Add the "columns" controller in the configuration to enable this functionality.

### Authorizing tables, columns and records

By default all tables, columns and paths are accessible. If you want to restrict access to some tables you may add the 'authorization' middleware 
and define a 'authorization.tableHandler' function that returns 'false' for these tables.

    'authorization.tableHandler' => function ($operation, $tableName) {
        return $tableName != 'license_keys';
    },

The above example will restrict access to the table 'license_keys' for all operations.

    'authorization.columnHandler' => function ($operation, $tableName, $columnName) {
        return !($tableName == 'users' && $columnName == 'password');
    },

The above example will restrict access to the 'password' field of the 'users' table for all operations.

    'authorization.recordHandler' => function ($operation, $tableName) {
        return ($tableName == 'users') ? 'filter=username,neq,admin' : '';
    },

The above example will disallow access to user records where the username is 'admin'. 
This construct adds a filter to every executed query. 

    'authorization.pathHandler' => function ($path) {
        return $path === 'openapi' ? false : true;
    },

The above example will disabled the `/openapi` route.

NB: You need to handle the creation of invalid records with a validation (or sanitation) handler.

### SQL GRANT authorization

You can alternatively use database permissons (SQL GRANT statements) to define the authorization model. In this case you
should not use the "authorization" middleware, but you do need to use the "reconnect" middleware. The handlers of the
"reconnect" middleware allow you to specify the correct username and password, like this:

    'reconnect.usernameHandler' => function () {
        return 'mevdschee';
    },
    'reconnect.passwordHandler' => function () {
        return 'secret123';
    },

This will make the API connect to the database specifying "mevdschee" as the username and "secret123" as the password.
The OpenAPI specification is less specific on allowed and disallowed operations when you are using database permissions,
as the permissions are not read in the reflection step.

NB: You may want to retrieve the username and password from the session (the "$_SESSION" variable).

### Sanitizing input

By default all input is accepted and sent to the database. If you want to strip (certain) HTML tags before storing you may add 
the 'sanitation' middleware and define a 'sanitation.handler' function that returns the adjusted value.

    'sanitation.handler' => function ($operation, $tableName, $column, $value) {
        return is_string($value) ? strip_tags($value) : $value;
    },

The above example will strip all HTML tags from strings in the input.

### Type sanitation

If you enable the 'sanitation' middleware, then you (automatically) also enable type sanitation. When this is enabled you may:

- send leading and trailing whitespace in a non-character field (it will be ignored).
- send a float to an integer or bigint field (it will be rounded).
- send a base64url encoded string (it will be converted to regular base64 encoding).
- send a time/date/timestamp in any [strtotime accepted format](https://www.php.net/manual/en/datetime.formats.php) (it will be converted).

You may use the config settings "`sanitation.types`" and "`sanitation.tables`"' to define for which types and
in which tables you want to apply type sanitation (defaults to 'all'). Example:

    'sanitation.types' => 'date,timestamp',
    'sanitation.tables' => 'posts,comments',

Here we enable the type sanitation for date and timestamp fields in the posts and comments tables.

### Validating input

By default all input is accepted and sent to the database. If you want to validate the input in a custom way, 
you may add the 'validation' middleware and define a 'validation.handler' function that returns a boolean 
indicating whether or not the value is valid.

    'validation.handler' => function ($operation, $tableName, $column, $value, $context) {
        return ($column['name'] == 'post_id' && !is_numeric($value)) ? 'must be numeric' : true;
    },

When you edit a comment with id 4 using:

    PUT /records/comments/4

And you send as a body:

    {"post_id":"two"}

Then the server will return a '422' HTTP status code and nice error message:

    {
        "code": 1013,
        "message": "Input validation failed for 'comments'",
        "details": {
            "post_id":"must be numeric"
        }
    }

You can parse this output to make form fields show up with a red border and their appropriate error message.

### Type validations

If you enable the 'validation' middleware, then you (automatically) also enable type validation. 
This includes the following error messages:

| error message       | reason                      | applies to types                            |
| ------------------- | --------------------------- | ------------------------------------------- |
| cannot be null      | unexpected null value       | (any non-nullable column)                   |
| illegal whitespace  | leading/trailing whitespace | integer bigint decimal float double boolean |
| invalid integer     | illegal characters          | integer bigint                              |
| string too long     | too many characters         | varchar varbinary                           |
| invalid decimal     | illegal characters          | decimal                                     |
| decimal too large   | too many digits before dot  | decimal                                     |
| decimal too precise | too many digits after dot   | decimal                                     |
| invalid float       | illegal characters          | float double                                |
| invalid boolean     | use 1, 0, true or false     | boolean                                     |
| invalid date        | use yyyy-mm-dd              | date                                        |
| invalid time        | use hh:mm:ss                | time                                        |
| invalid timestamp   | use yyyy-mm-dd hh:mm:ss     | timestamp                                   |
| invalid base64      | illegal characters          | varbinary, blob                             |

You may use the config settings "`validation.types`" and "`validation.tables`"' to define for which types and
in which tables you want to apply type validation (defaults to 'all'). Example:

    'validation.types' => 'date,timestamp',
    'validation.tables' => 'posts,comments',

Here we enable the type validation for date and timestamp fields in the posts and comments tables.

NB: Types that are enabled will be checked for null values when the column is non-nullable.

### Multi-tenancy support

Two forms of multi-tenancy are supported:

 - Single database, where every table has a tenant column (using the "multiTenancy" middleware).
 - Multi database, where every tenant has it's own database (using the "reconnect" middleware).

Below is an explanation of the corresponding middlewares.

#### Multi-tenancy middleware

You may use the "multiTenancy" middleware when you have a single multi-tenant database. 
If your tenants are identified by the "customer_id" column, then you can use the following handler:

    'multiTenancy.handler' => function ($operation, $tableName) {
        return ['customer_id' => 12];
    },

This construct adds a filter requiring "customer_id" to be "12" to every operation (except for "create").
It also sets the column "customer_id" on "create" to "12" and removes the column from any other write operation.

NB: You may want to retrieve the customer id from the session (the "$_SESSION" variable).

#### Reconnect middleware

You may use the "reconnect" middleware when you have a separate database for each tenant.
If the tenant has it's own database named "customer_12", then you can use the following handler:

    'reconnect.databaseHandler' => function () {
        return 'customer_12';
    },

This will make the API reconnect to the database specifying "customer_12" as the database name. If you don't want
to use the same credentials, then you should also implement the "usernameHandler" and "passwordHandler".

NB: You may want to retrieve the database name from the session (the "$_SESSION" variable).

### Prevent database scraping

You may use the "joinLimits" and "pageLimits" middleware to prevent database scraping.
The "joinLimits" middleware limits the table depth, number of tables and number of records returned in a join operation. 
If you want to allow 5 direct direct joins with a maximum of 25 records each, you can specify:

    'joinLimits.depth' => 1,
    'joinLimits.tables' => 5,
    'joinLimits.records' => 25,

The "pageLimits" middleware limits the page number and the number records returned from a list operation. 
If you want to allow no more than 10 pages with a maximum of 25 records each, you can specify:

    'pageLimits.pages' => 10,
    'pageLimits.records' => 25,

NB: The maximum number of records is also applied when there is no page number specified in the request.

### Search all text fields

You may use the "textSearch" middleware to simplify (wildcard) text searches when listing records. 
It allows you to specify a "search" parameter using:

    GET /records/posts?search=Hello

It will return all records from "posts" that contain "Hello" in one of their text (typed) fields:

    {
        "records":[
            {
                "id": 1,
                "title": "Hello world!",
                "content": "Welcome to the first post.",
                "created": "2018-03-05T20:12:56Z"
            }
        ]
    }

The example searches the fields "title" or "content" for the substring "Hello".

### Customization handlers

You may use the "customization" middleware to modify request and response and implement any other functionality.

    'customization.beforeHandler' => function ($operation, $tableName, $request, $environment) {
        $environment->start = microtime(true);
    },
    'customization.afterHandler' => function ($operation, $tableName, $response, $environment) {
        return $response->withHeader('X-Time-Taken', microtime(true) - $environment->start);
    },

The above example will add a header "X-Time-Taken" with the number of seconds the API call has taken.

### JSON encoding options

You can change the way the JSON is encoded by setting the configuration parameter "jsonOptions".

    'jsonOptions' => JSON_PRETTY_PRINT | JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES,

The above example will set JSON options to 128+256+64 = 448, as per the list of options below:

    JSON_HEX_TAG (1)
        All < and > are converted to \u003C and \u003E. 
    JSON_HEX_AMP (2)
        All & are converted to \u0026. 
    JSON_HEX_APOS (4)
        All ' are converted to \u0027. 
    JSON_HEX_QUOT (8)
        All " are converted to \u0022. 
    JSON_FORCE_OBJECT (16)
        Outputs an object rather than an array when a non-associative array is used. 
        Especially useful when the recipient of the output is expecting an object and 
        the array is empty. 
    JSON_NUMERIC_CHECK (32)
        Encodes numeric strings as numbers. 
    JSON_UNESCAPED_SLASHES (64)
        Don't escape /. 
    JSON_PRETTY_PRINT (128)
        Use whitespace in returned data to format it. 
    JSON_UNESCAPED_UNICODE (256)
        Encode multibyte Unicode characters literally (default is to escape as \uXXXX). 
    JSON_PARTIAL_OUTPUT_ON_ERROR (512)
        Substitute some unencodable values instead of failing. 
    JSON_PRESERVE_ZERO_FRACTION (1024)
        Ensures that float values are always encoded as a float value. 
    JSON_UNESCAPED_LINE_TERMINATORS (2048)
        The line terminators are kept unescaped when JSON_UNESCAPED_UNICODE is supplied. 
        It uses the same behaviour as it was before PHP 7.1 without this constant. 
        Available as of PHP 7.1.0. 

Source: [PHP's JSON constants documentation](https://www.php.net/manual/en/json.constants.php) 

### JSON middleware

You may use the "json" middleware to read/write JSON strings as JSON objects and arrays.

JSON strings are automatically detected when the "json" middleware is enabled.

You may limit the scanning of by specifying specific table and/or field names: 

    'json.tables' => 'products',
    'json.columns' => 'properties',

This will change the output of:

    GET /records/products/1

Without "json" middleware the output will be:

    {
        "id": 1,
        "name": "Calculator",
        "price": "23.01",
        "properties": "{\"depth\":false,\"model\":\"TRX-120\",\"width\":100,\"height\":null}",
    }

With "json" middleware the output will be:

    {
        "id": 1,
        "name": "Calculator",
        "price": "23.01",
        "properties": {
            "depth": false,
            "model": "TRX-120",
            "width": 100,
            "height": null
        },
    }

This also applies when creating or modifying JSON string fields (also when using batch operations).

Note that JSON string fields cannot be partially updated and that this middleware is disabled by default.
You can enable the "json" middleware using the "middlewares" configuration setting.

### XML middleware

You may use the "xml" middleware to translate input and output from JSON to XML. This request:

    GET /records/posts/1

Outputs (when "pretty printed"):

    {
        "id": 1,
        "user_id": 1,
        "category_id": 1,
        "content": "blog started"
    }

While (note the "format" query parameter):

    GET /records/posts/1?format=xml

Outputs:

    <root>
        <id>1</id>
        <user_id>1</user_id>
        <category_id>1</category_id>
        <content>blog started</content>
    </root>

This functionality is disabled by default and must be enabled using the "middlewares" configuration setting.

### File uploads

File uploads are supported through the [FileReader API](https://caniuse.com/#feat=filereader), check out the [example](https://github.com/mevdschee/php-crud-api/blob/master/examples/clients/upload/vanilla.html).

## OpenAPI specification

On the "/openapi" end-point the OpenAPI 3.0 (formerly called "Swagger") specification is served. 
It is a machine readable instant documentation of your API. To learn more, check out these links:

- [Swagger Editor](https://editor.swagger.io/) can be used to view and debug the generated specification.
- [OpenAPI specification](https://swagger.io/specification/) is a manual for creating an OpenAPI specification.
- [Swagger Petstore](https://petstore.swagger.io/) is an example documentation that is generated using OpenAPI.

## Cache

There are 4 cache engines that can be configured by the "cacheType" config parameter:

- TempFile (default)
- Redis
- Memcache
- Memcached

You can install the dependencies for the last three engines by running:

    sudo apt install php-redis redis
    sudo apt install php-memcache memcached
    sudo apt install php-memcached memcached

The default engine has no dependencies and will use temporary files in the system "temp" path.

You may use the "cachePath" config parameter to specify the file system path for the temporary files or
in case that you use a non-default "cacheType" the hostname (optionally with port) of the cache server.

## Types

These are the supported types with their length, category, JSON type and format:

| type       | length | category  | JSON type | format              |
| ---------- | ------ | --------- | --------- | ------------------- |
| varchar    | 255    | character | string    |                     |
| clob       |        | character | string    |                     |
| boolean    |        | boolean   | boolean   |                     |
| integer    |        | integer   | number    |                     |
| bigint     |        | integer   | number    |                     |
| float      |        | float     | number    |                     |
| double     |        | float     | number    |                     |
| decimal    | 19,4   | decimal   | string    |                     |
| date       |        | date/time | string    | yyyy-mm-dd          | 
| time       |        | date/time | string    | hh:mm:ss            |
| timestamp  |        | date/time | string    | yyyy-mm-dd hh:mm:ss |
| varbinary  | 255    | binary    | string    | base64 encoded      |
| blob       |        | binary    | string    | base64 encoded      |
| geometry   |        | other     | string    | well-known text     |

Note that geometry is a non-jdbc type and thus has limited support.

## Data types in JavaScript

Javascript and Javascript object notation (JSON) are not very well suited for reading database records. Decimal, date/time, binary and geometry types must be represented as strings in JSON (binary is base64 encoded, geometries are in WKT format). Below are two more serious issues described.

### 64 bit integers

JavaScript does not support 64 bit integers. All numbers are stored as 64 bit floating point values. The mantissa of a 64 bit floating point number is only 53 bit and that is why all integer numbers bigger than 53 bit may cause problems in JavaScript.

### Inf and NaN floats

The valid floating point values 'Infinite' (calculated with '1/0') and 'Not a Number' (calculated with '0/0') cannot be expressed in JSON, as they are not supported by the [JSON specification](https://www.json.org). When these values are stored in a database then you cannot read them as this script outputs database records as JSON.

## Errors

The following errors may be reported:

| Error | HTTP response code        | Message
| ----- | ------------------------- | --------------
| 1000  | 404 Not found             | Route not found 
| 1001  | 404 Not found             | Table not found 
| 1002  | 422 Unprocessable entity  | Argument count mismatch 
| 1003  | 404 Not found             | Record not found 
| 1004  | 403 Forbidden             | Origin is forbidden 
| 1005  | 404 Not found             | Column not found 
| 1006  | 409 Conflict              | Table already exists 
| 1007  | 409 Conflict              | Column already exists 
| 1008  | 422 Unprocessable entity  | Cannot read HTTP message 
| 1009  | 409 Conflict              | Duplicate key exception 
| 1010  | 409 Conflict              | Data integrity violation 
| 1011  | 401 Unauthorized          | Authentication required 
| 1012  | 403 Forbidden             | Authentication failed 
| 1013  | 422 Unprocessable entity  | Input validation failed 
| 1014  | 403 Forbidden             | Operation forbidden 
| 1015  | 405 Method not allowed    | Operation not supported 
| 1016  | 403 Forbidden             | Temporary or permanently blocked 
| 1017  | 403 Forbidden             | Bad or missing XSRF token 
| 1018  | 403 Forbidden             | Only AJAX requests allowed 
| 1019  | 403 Forbidden             | Pagination Forbidden 
| 1020  | 409 Conflict              | User already exists
| 1021  | 422 Unprocessable entity  | Password too short
| 1022  | 422 Unprocessable entity  | Username is empty
| 1023  | 404 Not found             | Primary key not found
| 9999  | 500 Internal server error | Unknown error 

The following JSON structure is used:

    {
        "code":1002,
        "message":"Argument count mismatch in '1'"
    }

NB: Any non-error response will have status: 200 OK

## Status

To connect to your monitoring there is a 'ping' endpoint:

    GET /status/ping

And this should return status 200 and as data:

    {
        "db": 42,
        "cache": 9
    }

These can be used to measure the time (in microseconds) to connect and read data from the database and the cache.

## Custom controller

You can add your own custom REST API endpoints by writing your own custom controller class. 
The class must provide a constructor that accepts five parameters. With these parameters you can register
your own endpoint to the existing router. This endpoint may use the database and/or the reflection class
of the database.

Here is an example of a custom controller class:

```
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Tqdev\PhpCrudApi\Cache\Cache;
use Tqdev\PhpCrudApi\Column\ReflectionService;
use Tqdev\PhpCrudApi\Controller\Responder;
use Tqdev\PhpCrudApi\Database\GenericDB;
use Tqdev\PhpCrudApi\Middleware\Router\Router;

class MyHelloController {

    private $responder;

    public function __construct(Router $router, Responder $responder, GenericDB $db, ReflectionService $reflection, Cache $cache)
    {
        $router->register('GET', '/hello', array($this, 'getHello'));
        $this->responder = $responder;
    }

    public function getHello(ServerRequestInterface $request): ResponseInterface
    {
        return $this->responder->success(['message' => "Hello World!"]);
    }
}
```

And then you may register your custom controller class in the config object like this:

```
$config = new Config([
    ...
    'customControllers' => 'MyHelloController',
    ...
]);
```

The `customControllers` config supports a comma separated list of custom controller classes.

## Tests

I am testing mainly on Ubuntu and I have the following test setups:

  - (Docker) Debian 10 with PHP 7.3, MariaDB 10.3, PostgreSQL 11.4 (PostGIS 2.5) and SQLite 3.27
  - (Docker) Debian 11 with PHP 7.4, MariaDB 10.5, PostgreSQL 13.4 (PostGIS 3.1) and SQLite 3.34
  - (Docker) Debian 12 with PHP 8.2, MariaDB 10.11, PostgreSQL 15.3 (PostGIS 3.3) and SQLite 3.40
  - (Docker) RockyLinux 8 with PHP 7.2, MariaDB 10.3 and SQLite 3.26
  - (Docker) RockyLinux 9 with PHP 8.0, MariaDB 10.5 and SQLite 3.34
  - (Docker) Ubuntu 18.04 with PHP 7.2, MySQL 5.7, PostgreSQL 10.4 (PostGIS 2.4) and SQLite 3.22
  - (Docker) Ubuntu 20.04 with PHP 7.4, MySQL 8.0, PostgreSQL 12.15 (PostGIS 3.0) and SQLite 3.31 and SQL Server 2019
  - (Docker) Ubuntu 22.04 with PHP 8.1, MySQL 8.0, PostgreSQL 14.2 (PostGIS 3.2) and SQLite 3.37 
  - (Docker) Ubuntu 24.04 with PHP 8.3, MySQL 8.0, PostgreSQL 16.2 (PostGIS 3.4) and SQLite 3.45

This covers not all environments (yet), so please notify me of failing tests and report your environment. 
I will try to cover most relevant setups in the "docker" folder of the project.

### Running

To run the functional tests locally you may run the following commands:

    php build.php
    php test.php

This runs the functional tests from the "tests" directory. It uses the database dumps (fixtures) and
database configuration (config) from the corresponding subdirectories.

## Pretty URL

You may "rewrite" the URL to remove the "api.php" from the URL.

### Apache config example

Enable mod_rewrite and add the following to your ".htaccess" file:

```
RewriteEngine On
RewriteCond %{REQUEST_FILENAME} !-d
RewriteCond %{REQUEST_FILENAME} !-f
RewriteRule ^(.*)$ api.php/$1 [L,QSA]
```

The ".htaccess" file needs to go in the same folder as "api.php".

### Nginx config example

Use the following config to serve the API under Nginx and PHP-FPM:

```
server {
    listen 80 default_server;
    listen [::]:80 default_server;

    root /var/www/html;
    index index.php index.html index.htm index.nginx-debian.html;
    server_name server_domain_or_IP;

    location / {
        try_files $uri $uri/ /api.php?$args;  
    }

    location ~ [^/]\.php(/|$) {
        fastcgi_split_path_info ^(.+\.php)(/.+)$;
        try_files $fastcgi_script_name =404;
        set $path_info $fastcgi_path_info;
        fastcgi_param PATH_INFO $path_info;
        fastcgi_index index.php;
        include fastcgi.conf;
        fastcgi_pass unix:/run/php/php7.0-fpm.sock;
    }

    location ~ /\.ht {
        deny all;
    }
}
```

### Docker tests

Install docker using the following commands and then logout and login for the changes to take effect:

    sudo apt install docker.io docker-buildx
    sudo usermod -aG docker ${USER}

To run the docker tests run "build_all.sh" and "run_all.sh" from the docker directory. The output should be:

    ================================================
    Debian 10 (PHP 7.3)
    ================================================
    [1/4] Starting MariaDB 10.3 ..... done
    [2/4] Starting PostgreSQL 11.4 .. done
    [3/4] Starting SQLServer 2017 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 921 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 1058 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 752 ms, 13 skipped, 0 failed
    ================================================
    Debian 11 (PHP 7.4)
    ================================================
    [1/4] Starting MariaDB 10.5 ..... done
    [2/4] Starting PostgreSQL 13.4 .. done
    [3/4] Starting SQLServer 2017 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 914 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 997 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 735 ms, 13 skipped, 0 failed
    ================================================
    Debian 12 (PHP 8.2)
    ================================================
    [1/4] Starting MariaDB 10.11 .... done
    [2/4] Starting PostgreSQL 15.3 .. done
    [3/4] Starting SQLServer 2019 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 1016 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 1041 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 733 ms, 13 skipped, 0 failed
    ================================================
    RockyLinux 8 (PHP 7.2)
    ================================================
    [1/4] Starting MariaDB 10.3 ..... done
    [2/4] Starting PostgreSQL 11 .... skipped
    [3/4] Starting SQLServer 2017 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 935 ms, 1 skipped, 0 failed
    pgsql: skipped, driver not loaded
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 746 ms, 13 skipped, 0 failed
    ================================================
    RockyLinux 9 (PHP 8.0)
    ================================================
    [1/4] Starting MariaDB 10.5 ..... done
    [2/4] Starting PostgreSQL 12 .... skipped
    [3/4] Starting SQLServer 2017 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 928 ms, 1 skipped, 0 failed
    pgsql: skipped, driver not loaded
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 728 ms, 13 skipped, 0 failed
    ================================================
    Ubuntu 18.04 (PHP 7.2)
    ================================================
    [1/4] Starting MySQL 5.7 ........ done
    [2/4] Starting PostgreSQL 10.4 .. done
    [3/4] Starting SQLServer 2017 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 1296 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 1056 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 772 ms, 13 skipped, 0 failed
    ================================================
    Ubuntu 20.04 (PHP 7.4)
    ================================================
    [1/4] Starting MySQL 8.0 ........ done
    [2/4] Starting PostgreSQL 12.2 .. done
    [3/4] Starting SQLServer 2019 ... done
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 1375 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 868 ms, 1 skipped, 0 failed
    sqlsrv: 120 tests ran in 5713 ms, 1 skipped, 0 failed
    sqlite: 120 tests ran in 733 ms, 13 skipped, 0 failed
    ================================================
    Ubuntu 22.04 (PHP 8.1)
    ================================================
    [1/4] Starting MySQL 8.0 ........ done
    [2/4] Starting PostgreSQL 14.2 .. done
    [3/4] Starting SQLServer 2019 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 1372 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 1064 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 727 ms, 13 skipped, 0 failed
    ================================================
    Ubuntu 24.04 (PHP 8.3)
    ================================================
    [1/4] Starting MySQL 8. ........ done
    [2/4] Starting PostgreSQL 16.2 .. done
    [3/4] Starting SQLServer 2019 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 1344 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 856 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 722 ms, 13 skipped, 0 failed

The above test run (including starting up the databases) takes less than 5 minutes on my slow laptop.

    $ ./run.sh
    1) debian10
    2) debian11
    3) debian12
    4) rockylinux8
    5) rockylinux9
    6) ubuntu18
    7) ubuntu20
    8) ubuntu22
    > 6
    ================================================
    Ubuntu 18.04 (PHP 7.2)
    ================================================
    [1/4] Starting MySQL 5.7 ........ done
    [2/4] Starting PostgreSQL 10.4 .. done
    [3/4] Starting SQLServer 2017 ... skipped
    [4/4] Cloning PHP-CRUD-API v2 ... skipped
    ------------------------------------------------
    mysql: 120 tests ran in 1296 ms, 1 skipped, 0 failed
    pgsql: 120 tests ran in 1056 ms, 1 skipped, 0 failed
    sqlsrv: skipped, driver not loaded
    sqlite: 120 tests ran in 772 ms, 13 skipped, 0 failed
    root@b7ab9472e08f:/php-crud-api# 

As you can see the "run.sh" script gives you access to a prompt in the chosen docker environment.
In this environment the local files are mounted. This allows for easy debugging on different environments.
You may type "exit" when you are done.

### Docker image

There is a `Dockerfile` in the repository that is used to build an image at:

[https://hub.docker.com/r/mevdschee/php-crud-api](https://hub.docker.com/r/mevdschee/php-crud-api)

It will be automatically build on every release. The "latest" tag points to the last release.

The docker image accepts the environment variable parameters from the configuration.

### Docker compose

This repository also contains a `docker-compose.yml` file that you can install/build/run using:

    sudo apt install docker-compose
    docker-compose build
    docker-compose up

This will setup a database (MySQL) and a webserver (Apache) and runs the application using the blog example data used in the tests.

Test the script (running in the container) by opening the following URL:

    http://localhost:8080/records/posts/1

### Star History

[![Star History Chart](https://api.star-history.com/svg?repos=mevdschee/php-crud-api&type=Date)](https://star-history.com/#mevdschee/php-crud-api&Date)

Enjoy!
 readmeEtag: '"0af65dacf7530c69d7dfeaef6b537e3344f594af"' readmeLastModified: Fri, 22 Nov 2024 07:05:44 GMT repositoryId: 30567860 description: Single file PHP script that adds a REST API to a SQL database created: '2015-02-10T01:08:52Z' updated: '2026-02-05T14:26:24Z' language: PHP archived: false stars: 3725 watchers: 168 forks: 1039 owner: mevdschee logo: https://avatars.githubusercontent.com/u/1288217?v=4 license: MIT repoEtag: '"f2632393687c602aede7ede4c05970e4548f349b73a1a9ff5988553bb162bda0"' repoLastModified: Thu, 05 Feb 2026 14:26:24 GMT id: 84a7be428932f7789a8ad1a220b22216 - source: IMPLEMENTATIONS.md name: Fastify OpenAPI v3 homepage: https://gitlab.com/m03geek/fastify-oas language: Node.JS source_description: >- Fastify OpenAPI v3+ plugin. Generates OpenAPI specification from fastify schemas and routes. Also serves swagger ui and spec in json/yaml formats. category: Server Implementations foundInMaster: true id: c78fe82304e392d33668f4af87119380 - source: IMPLEMENTATIONS.md name: JSONSchema::Validator homepage: >- https://github.com/skbkontur/perl-jsonschema-validator](https://github.com/skbkontur/perl-jsonschema-validator language: Perl source_description: OpenAPI 3 request/response validation category: Server Implementations foundInMaster: true repository: >- https://github.com/skbkontur/perl-jsonschema-validator](https://github.com/skbkontur/perl-jsonschema-validator repositoryMetadata: base64Readme: >- IyBOQU1FCgpKU09OU2NoZW1hOjpWYWxpZGF0b3IgLSBWYWxpZGF0b3IgZm9yIEpTT04gU2NoZW1hIERyYWZ0NC9EcmFmdDYvRHJhZnQ3IGFuZCBPcGVuQVBJIFNwZWNpZmljYXRpb24gMy4wCgojIFZFUlNJT04KCnZlcnNpb24gMC4wMTEKCiMgU1lOT1BTSVMKCiAgICAjIHRvIGdldCBPcGVuQVBJIHZhbGlkYXRvciBpbiBZQU1MIGZvcm1hdAogICAgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHJlc291cmNlID0+ICdmaWxlOi8vL3NvbWUvcGF0aC90by9vYXMzMC55bWwnKTsKICAgIG15ICgkcmVzdWx0LCAkZXJyb3JzLCAkd2FybmluZ3MpID0gJHZhbGlkYXRvci0+dmFsaWRhdGVfcmVxdWVzdCgKICAgICAgICBtZXRob2QgPT4gJ0dFVCcsCiAgICAgICAgb3BlbmFwaV9wYXRoID0+ICcvdXNlci97aWR9L3Byb2ZpbGUnLAogICAgICAgIHBhcmFtZXRlcnMgPT4gewogICAgICAgICAgICBwYXRoID0+IHsKICAgICAgICAgICAgICAgIGlkID0+IDEyMzQKICAgICAgICAgICAgfSwKICAgICAgICAgICAgcXVlcnkgPT4gewogICAgICAgICAgICAgICAgZGV0YWlscyA9PiAnc2hvcnQnCiAgICAgICAgICAgIH0sCiAgICAgICAgICAgIGhlYWRlciA9PiB7CiAgICAgICAgICAgICAgICBoZWFkZXIgPT4gJ2hlYWRlciB2YWx1ZScKICAgICAgICAgICAgfSwKICAgICAgICAgICAgY29va2llID0+IHsKICAgICAgICAgICAgICAgIG5hbWUgPT4gJ3ZhbHVlJwogICAgICAgICAgICB9LAogICAgICAgICAgICBib2R5ID0+IFskaXNfZXhpc3RzLCAkY29udGVudF90eXBlLCAkZGF0YV0KICAgICAgICB9CiAgICApOwogICAgbXkgKCRyZXN1bHQsICRlcnJvcnMsICR3YXJuaW5ncykgPSAkdmFsaWRhdG9yLT52YWxpZGF0ZV9yZXNwb25zZSgKICAgICAgICBtZXRob2QgPT4gJ0dFVCcsCiAgICAgICAgb3BlbmFwaV9wYXRoID0+ICcvdXNlci97aWR9L3Byb2ZpbGUnLAogICAgICAgIHN0YXR1cyA9PiAnMjAwJywKICAgICAgICBwYXJhbWV0ZXJzID0+IHsKICAgICAgICAgICAgaGVhZGVyID0+IHsKICAgICAgICAgICAgICAgIGhlYWRlciA9PiAnaGVhZGVyIHZhbHVlJwogICAgICAgICAgICB9LAogICAgICAgICAgICBib2R5ID0+IFskaXNfZXhpc3RzLCAkY29udGVudF90eXBlLCAkZGF0YV0KICAgICAgICB9CiAgICApCgogICAgIyB0byBnZXQgSlNPTiBTY2hlbWEgRHJhZnQ0L0RyYWZ0Ni9EcmFmdDcgdmFsaWRhdG9yIGluIEpTT04gZm9ybWF0CiAgICAkdmFsaWRhdG9yID0gSlNPTlNjaGVtYTo6VmFsaWRhdG9yLT5uZXcocmVzb3VyY2UgPT4gJ2h0dHA6Ly9leGFtcGxlLmNvbS9kcmFmdDQvc2NoZW1hLmpzb24nKQogICAgbXkgKCRyZXN1bHQsICRlcnJvcnMpID0gJHZhbGlkYXRvci0+dmFsaWRhdGVfc2NoZW1hKCRvYmplY3RfdG9fdmFsaWRhdGUpCgojIERFU0NSSVBUSU9OCgpPcGVuQVBJIHNwZWNpZmljYXRpb24gYW5kIEpTT04gU2NoZW1hIERyYWZ0NC9EcmFmdDYvRHJhZnQ3IHZhbGlkYXRvcnMgd2l0aCBtaW5pbXVtIGRlcGVuZGVuY2llcy4KCiMgTUVUSE9EUwoKIyMgbmV3CgpDcmVhdGVzIG9uZSBvZiB0aGUgZm9sbG93aW5nIHZhbGlkYXRvcnM6IEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6RHJhZnQ0LCBKU09OU2NoZW1hOjpWYWxpZGF0b3I6OkRyYWZ0NiwgSlNPTlNjaGVtYTo6VmFsaWRhdG9yOjpEcmFmdDcsIEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6T0FTMzAuCgogICAgbXkgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHJlc291cmNlID0+ICdmaWxlOi8vL3NvbWUvcGF0aC90by9vYXMzMC55bWwnKTsKICAgIG15ICR2YWxpZGF0b3IgPSBKU09OU2NoZW1hOjpWYWxpZGF0b3ItPm5ldyhyZXNvdXJjZSA9PiAnaHR0cDovL2V4YW1wbGUuY29tL2RyYWZ0NC9zY2hlbWEuanNvbicpOwogICAgbXkgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHNjaGVtYSA9PiB7JyRzY2hlbWEnID0+ICdwYXRoL3RvL3NjaGVtYScsIC4uLn0pOwogICAgbXkgJHZhbGlkYXRvciA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+bmV3KHNjaGVtYSA9PiB7Li4ufSwgc3BlY2lmaWNhdGlvbiA9PiAnRHJhZnQ0Jyk7CgppZiBwYXJhbWV0ZXIgYHNwZWNpZmljYXRpb25gIGlzIG5vdCBzcGVjaWZpZWQgdGhlbiB0eXBlIG9mIHZhbGlkYXRvciB3aWxsIGJlIGRldGVybWluZWQgYnkgYCRzY2hlbWFgIGtleQpmb3IgSlNPTiBTY2hlbWEgRHJhZnQ0L0RyYWZ0Ni9EcmFmdDcgYW5kIGJ5IGBvcGVuYXBpYCBrZXkgZm9yIE9wZW5BUEkgU3BlY2lmaWNhdGlvbiAzLjAgaW4gYHNjaGVtYWAgcGFyYW1ldGVyLgoKUGFyYW1ldGVyczoKCi0gcmVzb3VyY2VzCgogICAgVG8gZ2V0IHNjaGVtYSBieSB1cmkKCi0gc2NoZW1hCgogICAgVG8gZ2V0IGV4cGxpY2l0bHkgc3BlY2lmaWVkIHNjaGVtYQoKLSBzcGVjaWZpY2F0aW9uCgogICAgVG8gc3BlY2lmeSBzcGVjaWZpY2F0aW9uIG9mIHNjaGVtYQoKLSB2YWxpZGF0ZVxfc2NoZW1hCgogICAgRG8gbm90IHZhbGlkYXRlIHNwZWNpZmllZCBzY2hlbWEKCi0gYmFzZVxfdXJpCgogICAgVG8gc3BlY2lmeSBiYXNlIHVyaSBvZiBzY2hlbWEuCiAgICBUaGlzIHBhcmFtZXRlciB1c2VkIHRvIGJ1aWxkIGFic29sdXRlIHBhdGggYnkgcmVsYXRpdmUgcmVmZXJlbmNlIGluIHNjaGVtYS4KICAgIEJ5IGRlZmF1bHQgYGJhc2VfdXJpYCBpcyBlcXVhbCB0byB0aGUgcmVzb3VyY2UgcGF0aCBpZiB0aGUgcmVzb3VyY2UgcGFyYW1ldGVyIGlzIHNwZWNpZmllZCBvdGhlcndpc2UgdGhlIGAkaWRgIGtleSBpbiB0aGUgc2NoZW1hLgoKQWRkaXRpb25hbCBwYXJhbWV0ZXJzIG5lZWQgdG8gYmUgbG9va2VkIGF0IGluIGEgc3BlY2lmaWMgdmFsaWRhdG9yIGNsYXNzLgpDdXJyZW50bHkgdGhlcmUgYXJlIHZhbGlkYXRvcnM6IEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6RHJhZnQ0LCBKU09OU2NoZW1hOjpWYWxpZGF0b3I6OkRyYWZ0NiwgSlNPTlNjaGVtYTo6VmFsaWRhdG9yOjpEcmFmdDcsIEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6T0FTMzAuCgojIyB2YWxpZGF0ZVxfcGF0aHMKClZhbGlkYXRlcyBhbGwgZmlsZXMgc3BlY2lmaWVkIGJ5IHBhdGggZ2xvYnMuCgogICAgbXkgJHJlc3VsdCA9IEpTT05TY2hlbWE6OlZhbGlkYXRvci0+dmFsaWRhdGVfcGF0aHMoWycvc29tZS9wYXRoL3RvL29wZW5hcGkuKi55YW1sJywgJy9zb21lL3BhdGgvdG8vanNvbnNjaGVtYS4qLmpzb24nXSk7CiAgICBmb3IgbXkgJGZpbGUgKGtleXMgJSRyZXN1bHQpIHsKICAgICAgICBteSAoJHJlcywgJGVycm9ycykgPSBAeyRyZXN1bHQtPnskZmlsZX19OwogICAgfQoKIyMgdmFsaWRhdGVcX3Jlc291cmNlCgojIyB2YWxpZGF0ZVxfcmVzb3VyY2VcX3NjaGVtYQoKIyBDQVZFQVRTCgojIyBZQU1MICYgYm9vbGVhbnMKCldoZW4gcmVhZGluZyBzY2hlbWEgZGVmaW5pdGlvbnMgZnJvbSBZQU1MLCBwbGVhc2Ugbm90ZSB0aGF0IHRoZSBzdGFuZGFyZApiZWhhdmlvdXIgb2YgW1lBTUw6OlBQXShodHRwczovL21ldGFjcGFuLm9yZy9wb2QvWUFNTCUzQSUzQVBQKSBhbmQgW1lBTUw6OlhTXShodHRwczovL21ldGFjcGFuLm9yZy9wb2QvWUFNTCUzQSUzQVhTKSBpcyB0byByZWFkIHZhbHVlcyB3aGljaCBldmFsdWF0ZQp0byBgdHJ1ZWAgb3IgYGZhbHNlYCBpbiBhIHBlcmwgY29udGV4dC4gVGhlc2UgdmFsdWVzIGhhdmUgbm8gcmVjb2duaXphYmxlCidib29sZWFuIHR5cGUnLiBUaGlzIGlzIGluc3VmZmljaWVudCBmb3IgSlNPTiBzY2hlbWEgdmFsaWRhdGlvbi4KClRvIG1ha2UgdGhlIFlBTUwgcmVhZGVycyBhbmQgYm9vbGVhbnMgd29yayB3aXRoIGBKU09OU2NoZW1hOjpWYWxpZGF0b3JgLAp5b3UgbmVlZCB0byB1c2UgdGhlIGBKU09OOjpQUGAgKGluY2x1ZGVkIGluIFBlcmwncyBzdGFuZGFyZCBsaWJyYXJ5KSBtb2R1bGUKYXMgZm9sbG93czoKCiAgICAjIGZvciBZQU1MOjpQUAogICAgdXNlIFlBTUw6OlBQOwoKICAgIG15ICRyZWFkZXIgPSBZQU1MOjpQUC0+bmV3KCBib29sZWFuID0+ICdKU09OOjpQUCcgKTsKICAgICMgZnJvbSBoZXJlLCB5b3UgY2FuIGZyZWVseSB1c2UgdGhlIHJlYWRlciB0bwogICAgIyByZWFkICYgd3JpdGUgYm9vbGVhbnMgYXMgJ3RydWUnIGFuZCAnZmFsc2UnCgoKICAgICMgZm9yIFlBTUw6OlhTCiAgICB1c2UgWUFNTDo6WFM7CgogICAgbXkgJHJlYWRlciA9IFlBTUw6OlhTLT5uZXc7CgogICAgIyBhbmQgd2hlbmV2ZXIgeW91IHJlYWQgWUFNTCB3aXRoIHRoaXMgcmVhZGVyLCBkbzoKICAgIG15ICR5YW1sID0gZG8gewogICAgICBsb2NhbCAkWUFNTDo6WFM6OkJvb2xlYW4gPSAnSlNPTjo6UFAnOwogICAgICAkcmVhZGVyLT5Mb2FkKCRzdHJpbmcpOyAjIG9yICRyZWFkZXItPkxvYWRGaWxlKCdmaWxlbmFtZScpOwogICAgfTsKClRoaXMgaXNuJ3QgYSBwcm9ibGVtIHdoZW4geW91IHVzZSB0aGUgYHJlc291cmNlYCBhcmd1bWVudCB0byB0aGUKYEpTT05TY2hlbWE6OlZhbGlkYXRvcjo6bmV3YCBjb25zdHJ1Y3RvciwgYnV0IGlmIHlvdSByZWFkIHlvdXIgb3duCnNjaGVtYSBhbmQgdXNlIHRoZSBgc2NoZW1hYCBhcmd1bWVudCwgdGhpcyBpcyBzb21ldGhpbmcgdG8gYmUgYXdhcmUgb2YuCgojIyBhbGxvd1xfYmlnbnVtID0+IDEKClRoZSBgYWxsb3dfYmlnbnVtID1gIDE+IHNldHRpbmcgKGF2YWlsYWJsZSBvbiBbSlNPTjo6WFNdKGh0dHBzOi8vbWV0YWNwYW4ub3JnL3BvZC9KU09OJTNBJTNBWFMpIGFuZApbQ3BhbmVsOjpKU09OOjpYU10oaHR0cHM6Ly9tZXRhY3Bhbi5vcmcvcG9kL0NwYW5lbCUzQSUzQUpTT04lM0ElM0FYUykpIG9uIGRlc2VyaWFsaXplcnMgaXMgbm90IHN1cHBvcnRlZC4KCldoZW4gZGVzZXJpYWxpemluZyBhIHJlcXVlc3QgYm9keSB3aXRoIGEgSlNPTiBwYXJzZXIgY29uZmlndXJlZCB3aXRoCmBhbGxvd19iaWdudW0gPWAgMT4sIGZsb2F0cyAtIGV2ZW4gb25lcyB3aGljaCBmaXQgaW50byB0aGUgcmVndWxhcgpmbG9hdCByYW5nZXMgLSB3aWxsIGJlIGRlc2VyaWFsaXplZCBhcyBgTWF0aDo6QmlnRmxvYXRgLiBTaW1pbGFybHksCmludGVnZXJzIG91dHNpZGUgb2YgdGhlIGludGVybmFsIGludGVnZXIgcmFuZ2UgYXJlIGRlc2VyaWFsaXplZCBhcwpgTWF0aDo6QmlnSW50YC4gTnVtYmVycyByZXByZXNlbnRlZCBhcyBgTWF0aDo6QmlnKmAgb2JqZWN0cyBhcmUgbm90CnJlY29nbml6ZWQgYXMgYWN0dWFsIG51bWJlcnMgYW5kIHdpbGwgZmFpbCB2YWxpZGF0aW9uLgoKIyBBVVRIT1JTCgotIEFsZXhleSBTdGF2cm92IDxsb2dpb25pekB5YS5ydT4KLSBJdmFuIFB1dGludHNldiA8dWlkQHJ5ZGxhYi5ydT4KLSBBbnRvbiBGZWRvdG92IDx0b3NoYS5mZWRvdG92LjIwMDBAZ21haWwuY29tPgotIERlbmlzIEliYWV2IDxkaW9ueXNAZ21haWwuY29tPgotIEFuZHJleSBLaG96b3YgPGFuZHJleUByeWRsYWIucnU+CgojIENPTlRSSUJVVE9SUwoKLSBFcmlrIEh1ZWxzbWFubiA8ZWh1ZWxzQGdtYWlsLmNvbT4KLSBKYW1lcyBXYXRlcnMgPGphbWVzQGpjd2F0ZXJzLmNvLnVrPgoKIyBDT1BZUklHSFQgQU5EIExJQ0VOU0UKClRoaXMgc29mdHdhcmUgaXMgQ29weXJpZ2h0IChjKSAyMDIxIGJ5IEFsZXhleSBTdGF2cm92LgoKVGhpcyBpcyBmcmVlIHNvZnR3YXJlLCBsaWNlbnNlZCB1bmRlcjoKCiAgICBUaGUgTUlUIChYMTEpIExpY2Vuc2UK readmeEtag: '"32131c216eadce854b63f2683e969d148b8f6a5b"' readmeLastModified: Fri, 09 Sep 2022 20:43:45 GMT repositoryId: 382338785 description: JSON Schema and OpenAPI data validator for Perl created: '2021-07-02T12:25:14Z' updated: '2025-08-22T15:27:27Z' language: Perl archived: false stars: 7 watchers: 7 forks: 4 owner: skbkontur logo: https://avatars.githubusercontent.com/u/5087073?v=4 license: NOASSERTION repoEtag: '"6bff7480de4e1f12236d8fbd8da528b861b6c3fdf205d487bac073d7c142f5fe"' repoLastModified: Fri, 22 Aug 2025 15:27:27 GMT id: 79a86285955c77cec16d44dfcef0121d - source: IMPLEMENTATIONS.md name: openapi-client-axios homepage: https://github.com/anttiviljami/openapi-client-axios language: - JavaScript - TypeScript source_description: >- JavaScript client library for consuming OpenAPI-enabled APIs with axios. Types included. category: - Client Implementations - Parsers foundInMaster: true repository: https://github.com/openapistack/openapi-client-axios repositoryMetadata: base64Readme: >- PGgxIGFsaWduPSJjZW50ZXIiPjxpbWcgYWx0PSJvcGVuYXBpLWNsaWVudC1heGlvcyIgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaXN0YWNrL29wZW5hcGktY2xpZW50LWF4aW9zL3Jhdy9tYWluL2hlYWRlci5wbmc/cmF3PXRydWUiIHN0eWxlPSJtYXgtd2lkdGg6NTByZW0iPjwvaDE+CgpbIVtDSV0oaHR0cHM6Ly9naXRodWIuY29tL29wZW5hcGlzdGFjay9vcGVuYXBpLWNsaWVudC1heGlvcy93b3JrZmxvd3MvQ0kvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL29wZW5hcGlzdGFjay9vcGVuYXBpLWNsaWVudC1heGlvcy9hY3Rpb25zP3F1ZXJ5PXdvcmtmbG93JTNBQ0kpClshW0xpY2Vuc2VdKGh0dHA6Ly9pbWcuc2hpZWxkcy5pby86bGljZW5zZS1taXQtYmx1ZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vb3BlbmFwaXN0YWNrL29wZW5hcGktY2xpZW50LWF4aW9zL2Jsb2IvbWFpbi9MSUNFTlNFKQpbIVtucG0gdmVyc2lvbl0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vdi9vcGVuYXBpLWNsaWVudC1heGlvcy5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLWNsaWVudC1heGlvcykKWyFbbnBtIGRvd25sb2Fkc10oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9ucG0vZHcvb3BlbmFwaS1jbGllbnQtYXhpb3Muc3ZnKV0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS1jbGllbnQtYXhpb3MpClshW2J1bmRsZSBzaXplXShodHRwczovL2ltZy5zaGllbGRzLmlvL2J1bmRsZXBob2JpYS9taW56aXAvb3BlbmFwaS1jbGllbnQtYXhpb3Muc3ZnP2xhYmVsPWd6aXAlMjBidW5kbGUpXShodHRwczovL2J1bmRsZXBob2JpYS5jb20vcGFja2FnZS9vcGVuYXBpLWNsaWVudC1heGlvcykKWyFbTGlicmFyaWVzLmlvIGRlcGVuZGVuY3kgc3RhdHVzIGZvciBsYXRlc3QgcmVsZWFzZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9saWJyYXJpZXNpby9yZWxlYXNlL25wbS9vcGVuYXBpLWNsaWVudC1heGlvcy5zdmcpXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLWNsaWVudC1heGlvcz9hY3RpdmVUYWI9ZGVwZW5kZW5jaWVzKQohW25wbSB0eXBlIGRlZmluaXRpb25zXShodHRwczovL2ltZy5zaGllbGRzLmlvL25wbS90eXBlcy9vcGVuYXBpLWNsaWVudC1heGlvcy5zdmcpClshW0J1eSBtZSBhIGNvZmZlZV0oaHR0cHM6Ly9pbWcuc2hpZWxkcy5pby9iYWRnZS9kb25hdGUtYnV5JTIwbWUlMjBhJTIwY29mZmVlLW9yYW5nZSldKGh0dHBzOi8vYnV5bWVhY29mZi5lZS9hbnR0aXZpbGphbWkpCgo8cCBhbGlnbj0iY2VudGVyIj5KYXZhU2NyaXB0IGNsaWVudCBsaWJyYXJ5IGZvciBjb25zdW1pbmcgT3BlbkFQSS1lbmFibGVkIEFQSXMgd2l0aCA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vYXhpb3MvYXhpb3MiIHRhcmdldD0iX2JsYW5rIj5heGlvczwvYT4uIFR5cGVzIGluY2x1ZGVkLjwvcD4KCiMjIEZlYXR1cmVzCgotIFt4XSBDcmVhdGUgQVBJIGNsaWVudHMgZnJvbSBbT3BlbkFQSSB2MyBkZWZpbml0aW9uc10oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24pCi0gW3hdIENsaWVudCBpcyBjb25maWd1cmVkIGluIHJ1bnRpbWUuICoqTm8gZ2VuZXJhdGVkIGNvZGUhKioKLSBbeF0gR2VuZXJhdGUgVHlwZVNjcmlwdCBkZWZpbml0aW9ucyAoLmQudHMpIGZvciB5b3VyIEFQSXMgd2l0aCBmdWxsIEludGVsbGlTZW5zZSBzdXBwb3J0Ci0gW3hdIEVhc3kgdG8gdXNlIEFQSSB0byBjYWxsIEFQSSBvcGVyYXRpb25zIHVzaW5nIEphdmFTY3JpcHQgbWV0aG9kcwogIC0gYGNsaWVudC5nZXRQZXQoMSlgCiAgLSBgY2xpZW50LnNlYXJjaFBldHMoKWAKICAtIGBjbGllbnQuc2VhcmNoUGV0cyh7IGlkczogWzEsIDIsIDNdIH0pYAogIC0gYGNsaWVudC51cGRhdGVQZXQoMSwgcGF5bG9hZClgCi0gW3hdIEJ1aWx0IG9uIHRvcCBvZiB0aGUgcm9idXN0IFtheGlvc10oaHR0cHM6Ly9naXRodWIuY29tL2F4aW9zL2F4aW9zKSBKYXZhU2NyaXB0IGxpYnJhcnkKLSBbeF0gSXNvbW9ycGhpYywgd29ya3MgYm90aCBpbiBicm93c2VyIGFuZCBOb2RlLmpzCgojIyBEb2N1bWVudGF0aW9uCgoqKk5ldyEqKiBPcGVuQVBJIENsaWVudCBBeGlvcyBkb2N1bWVudGF0aW9uIGlzIG5vdyBmb3VuZCBvbiBbb3BlbmFwaXN0YWNrLmNvXShodHRwczovL29wZW5hcGlzdGFjay5jbykKCmh0dHBzOi8vb3BlbmFwaXN0YWNrLmNvL2RvY3Mvb3BlbmFwaS1jbGllbnQtYXhpb3MvaW50cm8KCiMjIFF1aWNrIFN0YXJ0CgpgYGAKbnBtIGluc3RhbGwgLS1zYXZlIGF4aW9zIG9wZW5hcGktY2xpZW50LWF4aW9zCmBgYAoKYGBgCnlhcm4gYWRkIGF4aW9zIG9wZW5hcGktY2xpZW50LWF4aW9zCmBgYAoKV2l0aCBwcm9taXNlcyAvIENvbW1vbkpTIHN5bnRheDoKCmBgYGphdmFzY3JpcHQKY29uc3QgT3BlbkFQSUNsaWVudEF4aW9zID0gcmVxdWlyZSgnb3BlbmFwaS1jbGllbnQtYXhpb3MnKS5kZWZhdWx0OwoKY29uc3QgYXBpID0gbmV3IE9wZW5BUElDbGllbnRBeGlvcyh7IGRlZmluaXRpb246ICdodHRwczovL2V4YW1wbGUuY29tL2FwaS9vcGVuYXBpLmpzb24nIH0pOwphcGkuaW5pdCgpCiAgLnRoZW4oY2xpZW50ID0+IGNsaWVudC5nZXRQZXRCeUlkKDEpKQogIC50aGVuKHJlcyA9PiBjb25zb2xlLmxvZygnSGVyZSBpcyBwZXQgaWQ6MSBmcm9tIHRoZSBhcGknLCByZXMuZGF0YSkpOwpgYGAKCldpdGggYXN5bmMtYXdhaXQgLyBFUzYgc3ludGF4OgoKYGBgamF2YXNjcmlwdAppbXBvcnQgT3BlbkFQSUNsaWVudEF4aW9zIGZyb20gJ29wZW5hcGktY2xpZW50LWF4aW9zJzsKCmNvbnN0IGFwaSA9IG5ldyBPcGVuQVBJQ2xpZW50QXhpb3MoeyBkZWZpbml0aW9uOiAnaHR0cHM6Ly9leGFtcGxlLmNvbS9hcGkvb3BlbmFwaS5qc29uJyB9KTsKYXBpLmluaXQoKTsKCmFzeW5jIGZ1bmN0aW9uIGNyZWF0ZVBldCgpIHsKICBjb25zdCBjbGllbnQgPSBhd2FpdCBhcGkuZ2V0Q2xpZW50KCk7CiAgY29uc3QgcmVzID0gYXdhaXQgY2xpZW50LmNyZWF0ZVBldChudWxsLCB7IG5hbWU6ICdHYXJmaWVsZCcgfSk7CiAgY29uc29sZS5sb2coJ1BldCBjcmVhdGVkJywgcmVzLmRhdGEpOwp9CmBgYAoKIyMgVHlwZXNhZmUgQ2xpZW50cwoKIVtUeXBlU2NyaXB0IEludGVsbGlTZW5zZV0oaHR0cHM6Ly9naXRodWIuY29tL29wZW5hcGlzdGFjay9vcGVuYXBpLWNsaWVudC1heGlvcy9ibG9iL21haW4vcGFja2FnZXMvdHlwZWdlbi9pbnRlbGxpc2Vuc2UuZ2lmKQoKYG9wZW5hcGktY2xpZW50LWF4aW9zYCBjb21lcyB3aXRoIGEgQ0xJIGNvbW1hbmQgYG9wZW5hcGljbWQgdHlwZWdlbmAgdG8gZ2VuZXJhdGUgVHlwZXNjcmlwdCB0eXBlcyBmb3IgdHlwZSBzYWZldHkgYW5kIGNvZGUgYXV0b2NvbXBsZXRlLgoKYGBgCm5weCBvcGVuYXBpY21kIHR5cGVnZW4gLi9vcGVuYXBpLnlhbWwgPiBzcmMvdHlwZXMvb3BlbmFwaS5kLnRzCmBgYAoKVGhlIG91dHB1dCBvZiBgdHlwZWdlbmAgZXhwb3J0cyBhIHR5cGUgY2FsbGVkIGBDbGllbnRgLCB3aGljaCBjYW4gYmUgdXNlZCBmb3IgaW5zdGFuY2VzIGNyZWF0ZWQgd2l0aCBgT3BlbkFQSUNsaWVudEF4aW9zYC4KCkJvdGggdGhlIGBhcGkuZ2V0Q2xpZW50KClgIGFuZCBgYXBpLmluaXQoKWAgbWV0aG9kcyBzdXBwb3J0IHBhc3NpbmcgaW4gYSBDbGllbnQgdHlwZS4KCmBgYHR5cGVzY3JpcHQKaW1wb3J0IHsgQ2xpZW50IGFzIFBldFN0b3JlQ2xpZW50IH0gZnJvbSAnLi9jbGllbnQuZC50cyc7Cgpjb25zdCBjbGllbnQgPSBhd2FpdCBhcGkuaW5pdDxQZXRTdG9yZUNsaWVudD4oKTsKY29uc3QgY2xpZW50ID0gYXdhaXQgYXBpLmdldENsaWVudDxQZXRTdG9yZUNsaWVudD4oKTsKYGBgCgpgb3BlbmFwaWNtZCB0eXBlZ2VuYCBzdXBwb3J0cyB1c2luZyBib3RoIGxvY2FsIGFuZCByZW1vdGUgVVJMcyBmb3IgT3BlbkFQSSBkZWZpbml0aW9uIGZpbGVzLgoKYGBgCiQgbnB4IG9wZW5hcGljbWQgdHlwZWdlbiAuL3BldHN0b3JlLnlhbWwKJCBucHggb3BlbmFwaWNtZCB0eXBlZ2VuIGh0dHBzOi8vcGV0c3RvcmUzLnN3YWdnZXIuaW8vYXBpL3YzL29wZW5hcGkuanNvbgpgYGAKCiMjIENvbW1lcmNpYWwgc3VwcG9ydAoKRm9yIGFzc2lzdGFuY2Ugd2l0aCBvcGVuYXBpLWNsaWVudC1heGlvcyBpbiB5b3VyIGNvbXBhbnksIHJlYWNoIG91dCBhdCBzdXBwb3J0QG9wZW5hcGlzdGFjay5jby4KCiMjIENvbnRyaWJ1dGluZwoKT3BlbkFQSSBDbGllbnQgQXhpb3MgaXMgRnJlZSBhbmQgT3BlbiBTb3VyY2UgU29mdHdhcmUuIElzc3VlcyBhbmQgcHVsbCByZXF1ZXN0cyBhcmUgbW9yZSB0aGFuIHdlbGNvbWUhCg== readmeEtag: '"4f973e3fa064f50ede67e7b355607b79a6cdcde7"' readmeLastModified: Wed, 12 Jun 2024 09:33:11 GMT repositoryId: 157976175 description: JavaScript client library for consuming OpenAPI-enabled APIs with axios created: '2018-11-17T11:04:15Z' updated: '2026-01-23T05:15:22Z' language: TypeScript archived: false stars: 613 watchers: 5 forks: 71 owner: openapistack logo: https://avatars.githubusercontent.com/u/147298423?v=4 license: MIT repoEtag: '"5219bcb0484cb6c30db2079fe8f6ea74afd280d2609d1a59694df7ceb49e36b0"' repoLastModified: Fri, 23 Jan 2026 05:15:22 GMT id: a3b46b452de775b27f5ce966a69ccfa3 oldLocations: - https://github.com/anttiviljami/openapi-client-axios - source: IMPLEMENTATIONS.md name: serverless-openapi-documentation homepage: https://github.com/temando/serverless-openapi-documentation language: TypeScript source_description: >- Serverless 1.0 plugin to generate OpenAPI V3 documentation from serverless configuration category: - Code Generators - Parsers foundInMaster: true repository: https://github.com/temando/serverless-openapi-documentation repositoryMetadata: base64Readme: >- # Serverless OpenAPI Documentation Plugin

[![NPM](https://img.shields.io/npm/v/serverless-openapi-documentation.svg)](https://npmjs.org/packages/serverless-openapi-documentation/)
[![Travis CI](https://img.shields.io/travis/temando/serverless-openapi-documentation.svg)](https://travis-ci.org/temando/serverless-openapi-documentation)
[![JavaScript Style Guide](https://img.shields.io/badge/code_style-standard-brightgreen.svg)](https://standardjs.com)

Generates [**OpenAPI 3.0.0**](https://github.com/OAI/OpenAPI-Specification/blob/3.0.0/versions/3.0.0.md) documentation from serverless configuration files. OpenAPI is formerly known as Swagger. The configuration is inspired by the format used in [serverless-aws-documentation](https://www.npmjs.com/package/serverless-aws-documentation).

Works well with [Lincoln OpenAPI Renderer](https://github.com/temando/open-api-renderer).

---

- [Usage](#usage)
  - [Options](#options)
  - [Configuration](#configuration)
    - [Models](#models)
    - [Functions](#functions)
    - [`queryParams`](#queryparams)
    - [`pathParams`](#pathparams)
    - [`cookieParams`](#cookieparams)
    - [`requestModels`](#requestmodels)
    - [`methodResponses`](#methodresponses)
- [Example Configuration](#example-configuration)
- [Install](#install)

---

## Usage

This plugin requires additional configuration to use, see the "[Configuration](#configuration)" section for how to configure the plugin to generate documentation.

Below are the commandline options to run the generator:

```bash
serverless openapi generate [options]
```

### Options

```bash
Plugin: ServerlessOpenAPIDocumentation
openapi generate  ...................... Generate OpenAPI v3 Documentation
    --output / -o ...................... Output file location [default: openapi.yml|json]
    --format / -f ...................... OpenAPI file format (yml|json) [default: yml]
    --indent / -i ...................... File indentation in spaces [default: 2]
    --help / -h   ...................... Help
```

### Configuration

To configure this plugin to generate valid OpenAPI documentation there are two places you'll need to modify in your `serverless.yml` file, the `custom` variables section and the `http` event section for each given function in your service.

This plugin is compatible with the same documentation configuration structure in [serverless-aws-documentation](https://www.npmjs.com/package/serverless-aws-documentation) and can run beside it.

The `custom` section of your `serverless.yml` can be configured as below:

```yml
custom:
  documentation:
    version: '1'
    title: 'My API'
    description: 'This is my API'
    models: {}
```

These configurations can be quite verbose; you can separate it out into it's own file, such as `serverless.doc.yml` as below:

```yml
custom:
  documentation: ${file(serverless.doc.yml):documentation}

functions:
  myFunc:
    events:
      - http:
          path: getStuff
          method: get
          documentation: ${file(serverless.doc.yml):endpoints.myFunc}
```

For more info on `serverless.yml` syntax, see their docs.

#### Models

Models contain additional information that you can use to define schemas for endpoints.  You must define the *content type* for each schema that you provide in the models.

The *required* directives for the models section are as follow:

* `name`: the name of the schema
* `description`: a description of the schema
* `contentType`: the content type of the described request/response (ie. `application/json` or `application/xml`).
* `schema`: The JSON Schema ([website](http://json-schema.org/)) that describes the model. You can either use inline `YAML` to define these, or refer to an external schema file as below

```yml
custom:
  documentation:
    models:
      - name: "ErrorResponse"
        description: "This is an error"
        contentType: "application/json"
        schema: ${file(models/ErrorResponse.json)}
      - name: "PutDocumentResponse"
        description: "PUT Document response model (external reference example)"
        contentType: "application/json"
        schema: ${file(models/PutDocumentResponse.json)}
      - name: "PutDocumentRequest"
        description: "PUT Document request model (inline example)"
        contentType: "application/json"
        schema:
          $schema: "http://json-schema.org/draft-04/schema#"
          properties:
            SomeObject:
              type: "object"
              properties:
                SomeAttribute:
                  type: "string"
```

#### Functions

To define the documentation for a given function event, you need to create a `documentation` attribute for your http event in your `serverless.yml` file.

The `documentation` section of the event configuration can contain the following attributes:

* `summary`: a short description of the method
* `description`: a detailed description of the method
* `tags`: an array of tags for this event
* `deprecated`: boolean indicator that indicates clients should migrate away from this function
* `requestBody`: contains description of the request
    * `description`: a description of the request body
* `requestModels`: a list of models to describe the request bodies (see [requestModels](#requestmodels) below)
* `queryParams`: a list of query parameters (see [queryParams](#queryparams) below)
* `pathParams`: a list of path parameters (see [pathParams](#pathparams) below)
* `cookieParams`: a list of cookie parameters (see [cookieParams](#cookieparams) below)
* `methodResponses`: an array of response models and applicable status codes
  * `statusCode`: applicable http status code (ie. 200/404/500 etc.)
  * `responseBody`: contains description of the response
    * `description`: a description of the body response
  * `responseHeaders`: a list of response headers (see [responseHeaders](#responseheaders) below)
  * `responseModels`: a list of models to describe the request bodies (see [responseModels](#responsemodels) below) for each `Content-Type`

```yml
functions:
  createUser:
    handler: "handler.create"
    events:
      - http:
        path: "create"
        method: "post"
        documentation:
          summary: "Create User"
          description: "Creates a user and then sends a generated password email"
          requestBody:
            description: "A user information object"
          requestModels:
            application/json: "PutDocumentRequest"
          pathParams:
            - name: "username"
              description: "The username for a user to create"
              schema:
                type: "string"
                pattern: "^[-a-z0-9_]+$"
          queryParams:
            - name: "membershipType"
              description: "The user's Membership Type"
              schema:
                type: "string"
                enum:
                  - "premium"
                  - "standard"
          cookieParams:
            - name: "SessionId"
              description: "A Session ID variable"
              schema:
                type: "string"
          methodResponses:
            - statusCode: 201
              responseBody:
                description: "A user object along with generated API Keys"
              responseModels:
                application/json: "PutDocumentResponse"
            - statusCode: 500
              responseBody:
                description: "An error message when creating a new user"
              responseModels:
                application/json: "ErrorResponse"
```

#### `queryParams`

Query parameters can be described as follow:

* `name`: the name of the query variable
* `description`: a description of the query variable
* `required`: whether the query parameter is mandatory (boolean)
* `schema`: JSON schema (inline or file)

```yml
queryParams:
  - name: "filter"
    description: "The filter parameter"
    required: true
    schema:
      type: "string"
```

#### `pathParams`

Path parameters can be described as follow:

* `name`: the name of the query variable
* `description`: a description of the query variable
* `schema`: JSON schema (inline or file)

```yml
pathParams:
  - name: "usernameId"
    description: "The usernameId parameter"
    schema:
      type: "string"
```

#### `cookieParams`

Cookie parameters can be described as follow:

* `name`: the name of the query variable
* `description`: a description of the query variable
* `required`: whether the query parameter is mandatory (boolean)
* `schema`: JSON schema (inline or file)

```yml
cookieParams:
  - name: "sessionId"
    description: "The sessionId parameter"
    required: true
    schema:
      type: "string"
```

#### `requestModels`

The `requestModels` property allows you to define models for the HTTP Request of the function event. You can define a different model for each different `Content-Type`. You can define a reference to the relevant request model named in the `models` section of your configuration (see [Defining Models](#models) section).

```yml
requestModels:
  application/json: "CreateRequest"
  application/xml: "CreateRequestXML"
```

#### `methodResponses`

You can define the response schemas by defining properties for your function event.

For an example of a `methodResponses` configuration for an event see below:

```yml
methodResponse:
  - statusCode: 200
    responseHeaders:
      - name: "Content-Type"
        description: "Content Type header"
        schema:
          type: "string"
    responseModels:
      application/json: "CreateResponse"
      application/xml: "CreateResponseXML"
```

##### `responseModels`

The `responseModels` property allows you to define models for the HTTP Response of the function event. You can define a different model for each different `Content-Type`. You can define a reference to the relevant response model named in the `models` section of your configuration (see [Defining Models](#models) section).

```yml
responseModels:
  application/json: "CreateResponse"
  application/xml: "CreateResponseXML"
```

##### `responseHeaders` and `requestHeaders`

The `responseHeaders/requestHeaders` section of the configuration allows you to define the HTTP headers for the function event.

The attributes for a header are as follow:

* `name`: the name of the HTTP Header
* `description`: a description of the HTTP Header
* `schema`: JSON schema (inline or file)

```yml
responseHeaders:
  - name: "Content-Type"
    description: "Content Type header"
    schema:
      type: "string"
requestHeaders:
  - name: "Content-Type"
    description: "Content Type header"
    schema:
      type: "string"
```

## Example configuration

Please view the example [serverless.yml](test/project/serverless.yml).

## Install

This plugin works for Serverless 1.x and up. Serverless 0.5 is not supported.

To add this plugin to your package.json:

**Using npm:**
```bash
npm install serverless-openapi-documentation --save-dev
```

**Using Yarn:**
```bash
yarn add serverless-openapi-documentation --dev
```

Next you need to add the plugin to the `plugins` section of your `serverless.yml` file.

```yml
plugins:
  - serverless-openapi-documentation
```

You can confirm the plugin is correctly installed by running:

```bash
serverless | grep -i "ServerlessOpenAPIDocumentation"
```

It should return `ServerlessOpenAPIDocumentation` as one of the plugins on the list.

> Note: Add this plugin _after_ `serverless-offline` to prevent issues with `String.replaceAll` being overridden incorrectly.

## License

MIT
 readmeEtag: '"1ba5c61b0e0ab2906c7c7041c4ccdcce5fb7f172"' readmeLastModified: Fri, 22 Jun 2018 11:16:06 GMT repositoryId: 96071001 description: >- Serverless 1.0 plugin to generate OpenAPI V3 documentation from serverless configuration created: '2017-07-03T04:44:51Z' updated: '2025-10-06T21:38:58Z' language: TypeScript archived: false stars: 96 watchers: 6 forks: 124 owner: temando logo: https://avatars.githubusercontent.com/u/2232135?v=4 license: MIT repoEtag: '"e798d4ad27b8dbaff9a60286f88f9642246fe1488568c2ecc9db869ed1c27255"' repoLastModified: Mon, 06 Oct 2025 21:38:58 GMT id: 6b703add76932c2d3d042f15bd0fe577 - source: IMPLEMENTATIONS.md name: zero-rails_openapi homepage: https://github.com/zhandao/zero-rails_openapi language: Ruby source_description: >- Provide concise DSL for generating the OpenAPI Specification 3 documentation file for Rails application category: Code Generators foundInMaster: true repository: https://github.com/zhandao/zero-rails_openapi repositoryMetadata: base64Readme: >- # ZRO: OpenApi 3 JSON-Doc Generator for Rails

  [![Gem Version](https://badge.fury.io/rb/zero-rails_openapi.svg)](https://badge.fury.io/rb/zero-rails_openapi)
  [![Build Status](https://travis-ci.org/zhandao/zero-rails_openapi.svg?branch=master)](https://travis-ci.org/zhandao/zero-rails_openapi)
  [![Maintainability](https://api.codeclimate.com/v1/badges/471fd60f6eb7b019ceed/maintainability)](https://codeclimate.com/github/zhandao/zero-rails_openapi/maintainability)
  [![Test Coverage](https://api.codeclimate.com/v1/badges/471fd60f6eb7b019ceed/test_coverage)](https://codeclimate.com/github/zhandao/zero-rails_openapi/test_coverage)

  Concise DSL for generating OpenAPI Specification 3 (**OAS3**, formerly Swagger3) JSON documentation for Rails application.
  
  ```ruby
  class Api::ExamplesController < ApiController
    api :update, 'POST update some thing' do
      path  :id, Integer
      query :token, String, desc: 'api token', length: 16
      form data: { phone: String }
    end
  end
  ```

## Contributing

  **Hi, here is ZhanDao = ▽ =  
  It may be a very useful tool if you want to write API document clearly.  
  I'm looking forward to your issue and PR!**
  
  (Test cases are rich, like: [api DSL](spec/api_spec.rb) and [schema Obj](spec/oas_objs/schema_obj_spec.rb))

## Table of Contents

- [About OAS](#about-oas) (OpenAPI Specification)
- [Installation](#installation)
- [Configure](#configure)
- [DSL Usage](#dsl-usage)
  - [a.Basic DSL](#basic-dsl)
    - [a.1. route_base](#1-route_base-required-if-youre-not-writing-dsl-in-controller)
    - [a.2. doc_tag](#2-doc_tag-optional)
    - [a.3. components](#3-components-optional)
    - [a.4. api](#4-api-required)
    - [a.5. api_dry](#5-api_dry-optional)
  - [b. DSLs written inside `api` and `api_dry`'s block](#dsls-written-inside-api-and-api_drys-block)
    - [b.1. this_api_is_invalid!](#1-this_api_is_invalid-and-its-aliases)
    - [b.2. desc](#2-desc-description-for-the-current-api)
    - [b.3. param family methods](#3-param-family-methods-oas---parameter-object)
    - [b.4. request_body family methods](#4-request_body-family-methods-oas---request-body-object)
    - [b.5. response family methods](#5-response-family-methods-oas---response-object)
    - [b.6. callback](#6-callback-oas---callback-object)
    - [b.7. Authentication and Authorization](#7-authentication-and-authorization)
    - [b.8. server](#8-overriding-global-servers-by-server)
    - [b.9. dry](#9-dry)
  - [c. DSLs written inside `components`'s block](#dsls-written-inside-componentss-block)
  - [d. Schema and Type](#schema-and-type)
    - [d.1. (Schema) Type](#schema-type)
    - [d.2. Schema](#schema)
    - [d.3. Combined Schema](#combined-schema)
- [Run! - Generate JSON documentation file](#run---generate-json-documentation-file)
- [Use Swagger UI(very beautiful web page) to show your Documentation](#use-swagger-uivery-beautiful-web-page-to-show-your-documentation)
- [Tricks](#tricks)
    - [Write DSL somewhere else](#trick1---write-the-dsl-somewhere-else)
    - [Global DRYing](#trick2---global-drying)
    - [Auto generate description form enum](#trick3---auto-generate-description-form-enum)
    - [Skip or Use parameters define in `api_dry`](#trick4---skip-or-use-parameters-define-in-api_dry)
    - [Atuo Generate index/show Actions's Responses Based on DB Schema](#trick5---auto-generate-indexshow-actionss-response-types-based-on-db-schema)
- [Troubleshooting](#troubleshooting)
- [About `OpenApi.docs` and `OpenApi.routes_index`](#about-openapidocs-and-openapiroutes_index)

## About OAS

  Everything about OAS3 is on [OpenAPI Specification](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md)

  You can getting started from [swagger.io](https://swagger.io/docs/specification/basic-structure/)

  **I suggest you should understand the basic structure of OAS3 at least.**
  such as component (can help you reuse DSL code, when your apis are used with the
  same data structure).

## Installation

  Add this line to your Rails's Gemfile:

  ```ruby
  gem 'zero-rails_openapi'
  # or
  gem 'zero-rails_openapi', github: 'zhandao/zero-rails_openapi'
  ```

  And then execute:

      $ bundle

## Configure

  Create an initializer, configure ZRO and define your OpenApi documents.

  This is the simplest example:

  ```ruby
  # in config/initializers/open_api.rb
  require 'open_api'

  OpenApi::Config.class_eval do
    # Part 1: configs of this gem
    self.file_output_path = 'public/open_api'

    # Part 2: config (DSL) for generating OpenApi info
    open_api :doc_name, base_doc_classes: [ApiDoc]
    info version: '1.0.0', title: 'Homepage APIs'#, description: ..
    # server 'http://localhost:3000', desc: 'Internal staging server for testing'
    # bearer_auth :Authorization
  end
  ```
  
### Part 1: configs of this gem

  1. `file_output_path`(required): The location where .json doc file will be output.
  2. `default_run_dry`: defaults to run dry blocks even if the `dry` method is not called in the (Basic) DSL block. defaults to `false`.
  3. `doc_location`: give regular expressions for file or folder paths. `Dir[doc_location]` will be `require` before document generates.
      this option is only for not writing spec in controllers.
  4. `rails_routes_file`: give a txt's file path (which's content is the copy of `rails routes`'s output). This will speed up document generation. 
  5. `model_base`: The parent class of models in your application. This option is for auto loading schema from database.
  6. `file_format`

### Part 2: config (DSL) for generating OpenApi info

  ```ruby
  # 1. open_api
  #   base_doc_classes should be an array of base classes of the classes you write the OpenApi spec in, 
  #   like [ActionController::Base] (if you write spec in controllers).
  open_api doc_name, base_doc_classes: []
  
  # 2. info
  info version:, title:, desc: '', **addition
  
  # 3. server
  # 4. security_scheme / base_auth / bearer_auth / api_key
  # 5. global_security_require
  ```

  See all the DSLs: [config_dsl.rb](lib/open_api/config_dsl.rb)

## DSL Usage

  There are two kinds of DSL for this gem: **basic** and **inside basic**.
  1. Basic DSLs are class methods which is for declaring your APIs, components, and spec code DRYing ...
  2. DSLs written inside the block of Basic DSLs, is for declaring the parameters, responses (and so on) of the specified API and component.

### First of all, `include OpenApi::DSL` in your base class (which is for writing spec):

  For example:
  ```ruby
  # in app/controllers/api/api_controller.rb
  class ApiController < ActionController::API
    include OpenApi::DSL
  end
  ```

### DSL Usage Example

  Here is the simplest usage:

  ```ruby
  class Api::ExamplesController < ApiController
    api :index, 'GET list' do
      query :page, Integer#, range: { ge: 1 }, default: 1
      query :rows, Integer#, desc: 'per page', range: { ge: 1 }, default: 10
    end
  end
  ```

### Basic DSL

  [source code](lib/open_api/dsl.rb)

#### (1) `route_base` [required if you're not writing DSL in controller]

  ```ruby
  # ** Method Signature
  route_base path
  # ** Usage
  route_base 'api/v1/examples'
  ```

  [Usage](#trick1---write-the-dsl-somewhere-else): write the DSL somewhere else to simplify the current controller.

#### (2) `doc_tag` [optional]

  ```ruby
  # ** Method Signature
  doc_tag name: nil, **tag_info
  # ** Usage
  doc_tag name: 'ExampleTagName', description: "ExamplesController's APIs"#, externalDocs: ...
  ```
  This method allows you to set the Tag (which is a node of [OpenApi Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#openapi-object))
  of all the APIs in the class.

  Tag's name defaults to controller_name.

#### (3) `components` [optional]

  ```ruby
  # ** Method Signature
  components(&block)
  # ** Usage
  components do
    # (block inside) DSL for defining components
    schema :DogSchema => [ { id: Integer, name: String }, dft: { id: 1, name: 'pet' } ]
    query! :UidQuery  => [ :uid, String, desc: 'uid' ]
    response :BadRqResp => [ 'bad request', :json ]
  end

  # to use component
  api :action do
    query :doge, :DogSchema # to use a Schema component
    param_ref :UidQuery     # to use a Parameter component
    response_ref :BadRqResp # to use a Response component
  end
  ```
  Each RefObj is associated with components through component key.

  We suggest that component keys should be camelized, and **must be Symbol**.

#### (4) `api` [required]

  For defining API (or we could say controller action).

  ```ruby
  # ** Method Signature
  api action_name, summary = '', id: nil, tag: nil, http: nil, dry: Config.default_run_dry, &block
  # ** Usage
  api :index, '(SUMMARY) this api blah blah ...', # block ...
  ```
  
  Parameters explanation:
  1. action_name: must be the same as controller action name
  2. id: operationId
  3. http: HTTP method (like: 'GET' or 'GET|POST')
  
#### (5) `api_dry` [optional]

  This method is for DRYing.
  The blocks passed to `api_dry` will be executed to the specified APIs which are having the actions or tags in the class.

  ```ruby
  # ** Method Signature
  api_dry action_or_tags = :all, &block
  # ** Usage
  api_dry :all, 'common response' # block ...
  api_dry :index # block ...
  api_dry :TagA # block ...

  api_dry [:index, :show] do
    query #...
  end
  ```
  
  And then you should call `dry` method ([detailed info](#9-dry)) for executing the declared dry blocks:
  ```ruby
  api :index do
    dry
  end
  ```

### DSLs written inside [api](#4-api-required) and [api_dry](#5-api_dry-optional)'s block

  [source code](lib/open_api/dsl/api.rb)

  These following methods in the block describe the specified API action: description, valid?,
  parameters, request body, responses, securities and servers.

  (Here corresponds to OAS [Operation Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#operationObject))

#### (1) `this_api_is_invalid!`, and its aliases:
  ```
  this_api_is_expired!
  this_api_is_unused!
  this_api_is_under_repair!
  ```

  ```ruby
  # ** Method Signature
  this_api_is_invalid!(*)
  # ** Usage
  this_api_is_invalid! 'cause old version'
  ```

  After that, `deprecated` field of this API will be set to true.

#### (2) `desc`: description for the current API

  ```ruby
  # ** Method Signature
  desc string
  # ** Usage
  desc "current API's description"
  ```

#### (3) `param` family methods (OAS - [Parameter Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#parameterObject))

  To define parameter for APIs.
  ```
  param                              # 1. normal usage
  param_ref                          # 2. links sepcified RefObjs (by component keys) to current parameters.
  header,  path,  query,  cookie     # 3. passes specified parameter location (like header) to `param`
  header!, path!, query!, cookie!    # 4. bang method of above methods
  in_* by: { parameter_definations } # 5. batch definition, such as `in_path`, `in_query`
  examples                           # 6. examples of parameters
  ```
  **The bang method and param_name (which's name is end of a exclamation point `!`) means this param is required. Without `!` means optional. THE SAME BELOW.**

  ```ruby
  # Part 1
  # param_type:  location of parameter, like: query, path [A]
  # param_name:  name of parameter, it can be Symbol or String [B]
  # schema_type: type of parameter, like: String, Integer (must be a constant). see #schema-and-type
  # required:    :required / :req OR :optional / :opt
  # schema:      see #schema-and-type (including combined schema)
  # ** Method Signature
  param param_type, param_name, schema_type, required, schema = { }
  # ** Usage
  param :query, :page, Integer, :req,  range: { gt: 0, le: 5 }, desc: 'page number'

  # Part 2
  # ** Method Signature
  param_ref *component_key # should pass at least 1 key
  # ** Usage
  param_ref :IdPath#, :NameQuery, :TokenHeader

  # Part 3 & 4
  # ** Method Signature
  header param_name, schema_type = nil, **schema
  query! param_name, schema_type = nil, **schema
  # ** Usage
  header :'X-Token', String
  query! :readed, Boolean, default: false
  # The same effect as above, but not concise
  param :query, :readed, Boolean, :req, default: false

  # Part 5 
  # ** Method Signature
  in_query **params_and_schema
  # ** Usage
  in_query(
    search_type: String,
     search_val: String,
        export!: { type: Boolean, desc: 'export as pdf' }
  )
  # The same effect as above
  query  :search_type, String
  query  :search_val, String
  query! :export, Boolean, desc: 'export as pdf'

  # Part 6
  # ** Method Signature
  examples exp_params = :all, examples_hash
  # ** Usage
  # Suppose we have three parameters: id, name, age
  # * normal
  examples(
    right_input: [ 1, 'user', 26 ],
    wrong_input: [ 2, 'resu', 35 ]
  )
  # * using exp_params
  examples [:id, :name], {
    right_input: [ 1, 'user' ],
    wrong_input: [ 2, 'resu' ]
  }
  ```

  [A] OpenAPI 3.0 distinguishes between the following parameter types based on the parameter location: 
      **header, path, query, cookie**. [more info](https://swagger.io/docs/specification/describing-parameters/)

  [B] If `param_type` is path, for example: if the API path is `/good/:id`, you have to declare a path parameter named `id`

#### (4) `request_body` family methods (OAS - [Request Body Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#requestBodyObject))

  OpenAPI 3.0 uses the requestBody keyword to distinguish the payload from parameters.
  
  Notice: Each API has only ONE request body object. Each request body object can has multiple media types.
  It means: call `request_body` multiple times, (schemas) will be deeply merged (let's call it [fusion](#fusion)) into a request body object.
  ```
  request_body  # 1. normal usage
  body_ref      # 2. it links sepcified RefObjs (by component keys) to the body.
  body, body!   # 3. alias of request_body
  form, form!   # 4. to define a multipart/form-data request_body
  json, json!   # 5. to define a application/json request_body
  data          # 5. to define [a] property in the form-data request_body
  ```
  Bang methods(!) means the specified media-type body is required.

  ```ruby
  # Part 1
  # ** Method Signature
  # a. `data` contains the attributes (params, or properties) and their schemas required by the request body
  # b. `attr_name!` means it is required, without '!' means optional
  # c. options: desc / exp_params and examples
  # d. available `media_type` see: 
  #   https://github.com/zhandao/zero-rails_openapi/blob/master/lib/oas_objs/media_type_obj.rb#L29
  request_body required, media_type, data: { }, desc: '', **options
  # ** Usage
  request_body :opt, :form, data: {
    id!: Integer,
    name: { type: String, desc: 'name' }
  }, desc: 'a form-data'

  # Part 2
  # ** Method Signature
  body_ref component_key
  # ** Usage
  body_ref :UpdateUserBody

  # Part 3
  # ** Method Signature
  body! media_type, data: { }, **options
  # ** Usage
  body :json

  # Part 4
  # ** method Implement
  def form data:, **options # or `form!`
    body :form, data: data, **options
  end
  # ** Usage
  form! data: {
         name!: String,
      password: { type: String, pattern: /[0-9]{6,10}/ },
  }
  json data: { name!: String }

  # Part 5
  # ** Method Signature
  data name, type = nil, schema = { }
  # ** Usage
  data :password!, String, pattern: /[0-9]{6,10}/
  ```

  <a name="fusion"></a> 
  How **fusion** works:
  1. Difference media types will be merged into `requestBody["content"]`

  ```ruby
  form data: { }
  body :json, data: { }
  # will generate: "content": { "multipart/form-data": { }, "application/json": { } }
  ```

  2. The same media-types will be deeply merged together, including their `required` array:  
     (So that you can call `form` multiple times)

  ```ruby
  data :param_a!, String
  data :param_b,  Integer
  # or same as:
  form data: { :param_a! => String }
  form data: { :param_b  => Integer }
  # will generate: { "param_a": { "type": "string" }, "param_b": { "type": "integer" } } (call it X)
  # therefore:
  #   "content": { "multipart/form-data":
  #     { "schema": { "type": "object", "properties": { X }, "required": [ "param_a" ] }
  #   }
  ```

#### (5) `response` family methods (OAS - [Response Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#response-object))

  To define the response for APIs.
  ```
  response      # 1. aliases: `resp` and `error`
  response_ref  # 2. it links sepcified RefObjs (by component keys) to the response.
  ```

  ```ruby
  # ** Method Signature
  response code, desc, media_type = nil, headers: { }, data: { }, **options
  # ** Usage
  resp 200, 'success', :json, data: { name: 'test' }
  response 200, 'query result', :pdf, data: File
  response :success, 'succ', :json, headers: { 'X-Request-Start': String }, data: { }

  # ** Method Signature
  response_ref code_and_compkey_hash
  # ** Usage
  response_ref 700 => :AResp, 800 => :BResp
  ```

### (6) Callback (OAS - [Callback Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#callback-object))

  [About Callbacks](https://swagger.io/docs/specification/callbacks/)
  > In OpenAPI 3 specs, you can define callbacks – asynchronous, out-of-band requests that your service will send to some other service in response to certain events. This helps you improve the workflow your API offers to clients.  
    A typical example of a callback is a subscription functionality ... you can define the format of the “subscription” operation as well as the format of callback messages and expected responses to these messages.  
    This description will simplify communication between different servers and will help you standardize use of webhooks in your API.  
  [Complete YAML Example](https://github.com/OAI/OpenAPI-Specification/blob/master/examples/v3.0/callback-example.yaml)
  
  The structure of Callback Object:
  ```
  callbacks:
    Event1:
      path1:
        ...
      path2:
       ...
    Event2:
      ...
  ```
  
  `callback` method is for defining callbacks.
  ```ruby
  # ** Method Signature
  callback event_name, http_method, callback_url, &block
  # ** Usage
  callback :myEvent, :post, 'localhost:3000/api/goods' do
    query :name, String
    data :token, String
    response 200, 'success', :json, data: { name: String, description: String }
  end
  ```
  
  Use runtime expressions in callback_url:
  ```ruby
  callback :myEvent, :post, '{body callback_addr}/api/goods/{query id}'
  # the final URL will be: {$request.body#/callback_addr}/api/goods/{$request.query.id}
  # Note: Other expressions outside "$request" are not supported yet
  ```

#### (7) Authentication and Authorization

  First of all, please make sure that you have read one of the following documents:
  [OpenApi Auth](https://swagger.io/docs/specification/authentication/)
  or [securitySchemeObject](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#securitySchemeObject)

  ##### Define Security Scheme

  Use these DSL **in your initializer config or `components` block**:
  ```
  security_scheme # alias `auth_scheme`
  base_auth       # will call `security_scheme`
  bearer_auth     # will call `security_scheme`
  api_key         # will call `security_scheme`
  ```
  It's very simple to use (if you understand the above document)
  ```ruby
  # ** Method Signature
  security_scheme scheme_name, other_info
  # ** Usage
  security_scheme :BasicAuth, { type: 'http', scheme: 'basic', desc: 'basic auth' }

  # ** Method Signature
  base_auth scheme_name, other_info = { }
  bearer_auth scheme_name, format = 'JWT', other_info = { }
  api_key scheme_name, field:, in:, **other_info
  # ** Usage
  base_auth :BasicAuth, desc: 'basic auth' # the same effect as above
  bearer_auth :Token
  api_key :ApiKeyAuth, field: 'X-API-Key', in: 'header', desc: 'pass api key to header'
  ```

  ##### Apply Security

  ```
  # Use in initializer (Global effectiveness)
  global_security_require # alias: global_security & global_auth

  # Use in `api`'s block (Only valid for the current controller)
  security_require # alias security & auth_with
  ```
  ```ruby
  # ** Method Signature
  security_require scheme_name, scopes: [ ]
  # ** Usage
  global_auth :Token
  auth_with   :OAuth, scopes: %w[ read_example admin ]
  ```

#### (8) Overriding Global Servers by `server`

  ```ruby
  # ** Method Signature
  server url, desc: ''
  # ** Usage
  server 'http://localhost:3000', desc: 'local'
  ```
  
#### (9) `dry`

  You have to call `dry` method inside `api` block, or pass `dry: true` as parameter of `api`,
  for executing the dry blocks you declared before. Otherwise nothing will happen.
  
  ```ruby
  # ** Method Signature
  dry only: nil, skip: nil, none: false
  
  # ** Usage
  # In general, just:
  dry
  # To skip some params declared in dry blocks:
  dry skip: [:id, :name]
  # `only` is used to specify which parameters will be taken from dry blocks
  dry only: [:id]
  ```

### DSLs written inside [components](#3-components-optional)'s block
  [code source](lib/open_api/dsl/components.rb) (Here corresponds to OAS [Components Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#componentsObject))

  Inside `components`'s block,
  you can use the same DSLs as [DSLs written inside `api` and `api_dry`'s block](#dsls-written-inside-api-and-api_drys-block).  
  But notice there are two differences:

  (1) Each method needs to pass one more parameter `component_key` (as the first parameter),
      it will be used as the reference name for the component.

  ```ruby
  query! :UidQuery, :uid, String， desc: 'it is a component'
  #         ↑         ↑
  # component_key  param_name
  
  # You can also use "arrow writing", it may be easier to understand
  query! :UidQuery => [:uid, String, desc: '']
  ```

  (2) You can use `schema` to define a Schema Component.

  ```ruby
  # ** Method Signature
  schema component_key, type = nil, **schema
  # ** Usage
  schema :Dog  => [ String, desc: 'doge' ]
  # advance usage
  schema :Dog => [
      {
           id!: Integer,
          name: { type: String, desc: 'doge name' }
      }, default: { id: 1, name: 'pet' }
  ]
  # or flatten writing
  schema :Dog, { id!: Integer, name: String }, default: { id: 1, name: 'pet' }
  #
  # pass a ActiveRecord class constant as `component_key`,
  #   it will automatically load schema from database and then generate the component.
  schema User # easy! And the component_key will be :User
  ```
  To enable load schema from database, you must set [model base](#part-1-configs-of-this-gem) correctly.
  
### Schema and Type

  schema and type -- contain each other

#### (Schema) Type

  Support all [data types](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#dataTypes) in OAS.   

  1. String / 'binary' / 'base64' / 'uri'
  2. Integer / Long / 'int32' / 'int64' / Float / Double
  3. File (it will be converted to `{ type: 'string', format: Config.file_format }`)
  4. Date / DateTime
  5. 'boolean'
  6. Array / Array[\<Type\>] (like: `Array[String]`, `[String]`)
  7. Nested Array (like: `[[[Integer]]]`)
  8. Object / Hash (Object with properties)  
     Example: `{ id!: Integer, name: String }`
  9. Nested Hash: `{ id!: Integer, name: { first: String, last: String } }`
  10. Nested Array[Nested Hash]: `[[{ id!: Integer, name: { first: String, last: String } }]]`
  11. Symbol Value: it will generate a Schema Reference Object link to the component correspond to ComponentKey, like: :IdPath, :NameQuery
  
  **Notice** that Symbol is not allowed in all cases except 11.
  
#### Schema

  [OAS Schema Object](https://github.com/OAI/OpenAPI-Specification/blob/OpenAPI.next/versions/3.0.0.md#schemaObject)
  and [source code](https://github.com/zhandao/zero-rails_openapi/blob/master/lib/oas_objs/schema_obj.rb)
  
  Schema (Hash) is for defining properties of parameters, responses and request bodies.
  
  The following property keys will be process slightly:
  1. desc / description / d
  2. enum / in / values / allowable_values  
     should be Array or Range
  3. range: allow value in this continuous range  
     should be Range or like `{ gt: 0, le: 5 }`
  4. length / size / lth  
     should be an Integer, Integer Array, Integer Range, 
     or the following format Symbol: `:gt_`, `:ge_`, `:lt_`, `:le_` (:ge_5 means "greater than or equal 5"; :lt_9 means "lower than 9")
  5. pattern / regxp
  6. additional_properties / add_prop / values_type
  7. example
  8. examples
  9. format
  10. default: default value
  11. type

  The other keys will be directly merged. Such as:
  1. `title: 'Property Title'`
  2. `myCustomKey: 'Value'`

#### Combined Schema

  Very easy to use:
  ```ruby
  query :combination, one_of: [ :GoodSchema, String, { type: Integer, desc: 'integer input' } ]

  form data: {
      :combination_in_form => { any_of: [ Integer, String ] }
  }

  schema :PetSchema => [ not: [ Integer, Boolean ] ]
  ```

  OAS: [link1](https://swagger.io/docs/specification/data-models/oneof-anyof-allof-not/),
  [link2](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/3.0.0.md#schemaObject)

## Run! - Generate JSON Documentation File

  Use `OpenApi.write_docs`:

  ```ruby
  OpenApi.write_docs# if: !Rails.env.production?
  ```

  `if` option is used to control whether a JSON document is generated or not.
  
  Then the JSON files will be written to the directories you set. (Each API a file.)

## Use Swagger UI(very beautiful web page) to show your Documentation

  Download [Swagger UI](https://github.com/swagger-api/swagger-ui) (version >= 2.3.0 support the OAS3)
  to your project,
  change the default JSON file path(url) in index.html.
  In order to use it, you may have to enable CORS, [see](https://github.com/swagger-api/swagger-ui#cors-support)

## Tricks

### Trick1 - Write the DSL Somewhere Else

  Does your documentation take too many lines?  
  Do you want to separate documentation from controller to simplify both?  
  Very easy! Just follow

  ```ruby
  # config/initializers/open_api.rb
  # in your configuration
  base_doc_classes: [ApiDoc]

  # app/api_doc/api_doc.rb
  require 'open_api/dsl'

  class ApiDoc < Object
    include OpenApi::DSL
  end

  # app/api_doc/v1/examples_doc.rb
  class V1::ExamplesDoc < ApiDoc
    route_base 'api/v1/examples'

    api :index do
      # ...
    end
  end
  ```

  Explain: These four steps are necessary:
  1. create a class, like ApiDoc, and make it include OpenApi::DSL (then it could be the base class for writing Api spec).
  2. set the specified Api spec's base_doc_classes to ApiDoc.
  3. let your doc class (like V1::ExamplesDoc) inherit the base_doc_classes (ApiDoc).
  4. set the route_base (to route path api/v1/examples of that controller Api::V1::ExamplesController) inside V1::ExamplesDoc.
  
  Notes: file name ends in `_doc.rb` by default, but you can change it by setting `Config.doc_location`
    (it should be file paths, defaults to `./app/**/*_doc.rb`).

### Trick2 - Global DRYing

  Method `api_dry` is for DRY but its scope is limited to the current controller.

  I have no idea of best practices, But you can look at this [file](examples/auto_gen_doc.rb).  
  The implementation of the file is: do `api_dry` when inherits the base controller inside `inherited` method.

  You can use `sort` to specify the order of parameters.

### Trick3 - Auto Generate Description from Enum

  Just use `enum!`:
  ```ruby
  query :search_type, String, desc: 'search field, allows：<br/>', enum!: %w[name creator category price]
  # it will generate: 
  "search field, allows：<br/>1/ name<br/>2/ creator,<br/>3/ category<br/>4/ price<br/>"
  ```
  Or Hash `enum!`:
  ```ruby
  query :view, String, desc: 'allows values<br/>', enum!: {
          'all goods (default)': :all,
                  'only online': :online,
                 'only offline': :offline,
              'expensive goods': :get,
                  'cheap goods': :borrow,
  }
  ```

## Troubleshooting

- **You wrote document of the current API, but not find in the generated json file?**  
  Check your routing settings.

- **Report error when require `routes.rb`?***
  1. Run `rails routes`.
  2. Copy the output to a file, for example `config/routes.txt`.
     Ignore the file `config/routes.txt`.
  3. Put `c.rails_routes_file = 'config/routes.txt'` to your ZRO config.


## About `OpenApi.docs` and `OpenApi.routes_index`

  After `OpenApi.write_docs`, the above two module variables will be generated.

  `OpenApi.docs`: A Hash with API names as keys, and documents of each APIs as values.  
  documents are instances of ActiveSupport::HashWithIndifferentAccess.

  `OpenApi.routes_index`: Inverted index of controller path to API name mappings.  
  Like: `{ 'api/v1/examples' => :homepage_api }`  
  It's useful when you want to look up a document based on a controller and do something.

## Development

  After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.

  To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).

## License

  The gem is available as open source under the terms of the [MIT License](https://opensource.org/licenses/MIT).

## Code of Conduct

  Everyone interacting in the Zero-RailsOpenApi project’s codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](CODE_OF_CONDUCT.md).
 readmeEtag: '"0c9f9b900b7b94594b7b271215105f3bb1081310"' readmeLastModified: Mon, 26 Feb 2024 09:23:03 GMT repositoryId: 103143995 description: >- Concise DSL for generating OpenAPI Specification 3 (OAS3) JSON documentation for Ruby application. created: '2017-09-11T14:09:24Z' updated: '2025-05-13T01:54:35Z' language: Ruby archived: false stars: 170 watchers: 3 forks: 22 owner: zhandao logo: https://avatars.githubusercontent.com/u/24402029?v=4 license: MIT repoEtag: '"f6e642d2f9b51c7f438b534b871b56275d7365d663944f74d877fbda350a9502"' repoLastModified: Tue, 13 May 2025 01:54:35 GMT id: b6b6454891a26e88286d7f79814c63b0 - source: IMPLEMENTATIONS.md name: slush-vertx homepage: https://github.com/pmlopes/slush-vertx language: - Java - Kotlin - Groovy source_description: >- Generate server skeleton for [Vert.x Web API Contract](http://vertx.io/docs/#web) and API Client based on [Vert.x 3 Web Client](http://vertx.io/docs/#web) category: Code Generators foundInMaster: true repository: https://github.com/pmlopes/slush-vertx repositoryMetadata: base64Readme: >- IyBUaGlzIHByb2plY3QgaGFzIGJlZW4gcmVwbGFjZWQgd2l0aCBodHRwczovL2dpdGh1Yi5jb20vcG1sb3Blcy92ZXJ0eC1zdGFydGVyCgoKIyBzbHVzaC12ZXJ0eAoKWyFbTlBNXShodHRwczovL25vZGVpLmNvL25wbS9zbHVzaC12ZXJ0eC5wbmc/ZG93bmxvYWRzPXRydWUmZG93bmxvYWRSYW5rPXRydWUmc3RhcnM9dHJ1ZSldKGh0dHBzOi8vbm9kZWkuY28vbnBtL3NsdXNoLXZlcnR4LykKClshW1R3aXR0ZXJdKGh0dHBzOi8vaW1nLnNoaWVsZHMuaW8vdHdpdHRlci91cmwvaHR0cHMvZ2l0aHViLmNvbS9wbWxvcGVzL3NsdXNoLXZlcnR4LnN2Zz9zdHlsZT1zb2NpYWwpXShodHRwczovL3R3aXR0ZXIuY29tL2ludGVudC90d2VldD90ZXh0PVdvdzomdXJsPSU1Qm9iamVjdCUyME9iamVjdCU1RCkKClNjYWZmb2xkIHdpdGggW1NsdXNoXVtzbHVzaC11cmxdIHlvdXIgbmV3IFZlcnQueCBhcHBsaWNhdGlvbiEKCnNsdXNoLXZlcnR4IGlzIGEgY29sbGVjdGlvbiBvZiB0ZW1wbGF0ZSBkcml2ZW4gY29kZSBnZW5lcmF0b3JzIHRvIHNjYWZmb2xkIFZlcnQueCBwcm9qZWN0cyBiYXNlZCBvbiBsYW5ndWFnZSBhbmQgYnVpbGQgdG9vbCB1c2VkLgoKIyMgR2VuZXJhdG9ycwoKfCBHZW5lcmF0b3IgfCBEZXNjcmlwdGlvbiB8IEphdmEgfCBLb3RsaW4gfCBKYXZhc2NyaXB0IHwgR3Jvb3Z5IHwgUnVieSB8CnwtLS0tLS0tLS0tLXwtLS0tLS0tLS0tLS0tfC0tLS0tLXwtLS0tLS0tLXwtLS0tLS0tLS0tLS18LS0tLS0tLS18LS0tLS0tfAp8ICoqVmVydC54IFN0YXJ0ZXIgcHJvamVjdCoqIHwgR2VuZXJhdGUgYW4gZW1wdHkgcHJvamVjdCBjb25maWd1cmVkIGZvciBWZXJ0LnggMyBGcmFtZXdvcmsgfDpoZWF2eV9jaGVja19tYXJrOiB8IDpoZWF2eV9jaGVja19tYXJrOiB8IDpoZWF2eV9jaGVja19tYXJrOiB8IDpoZWF2eV9jaGVja19tYXJrOiB8IDpoZWF2eV9jaGVja19tYXJrOiB8CnwgKipWZXJ0LnggV2ViIFNlcnZlciBTdGFydGVyKiogfCBHZW5lcmF0ZSBhIHNrZWxldG9uIHdpdGggc291cmNlcyBhbmQgdGVzdHMgZm9yIFZlcnQueCAzIFdlYiBwb3dlcmVkIFJFU1Qgc2VydmVyIHwgOmhlYXZ5X2NoZWNrX21hcms6IHwgOmhlYXZ5X2NoZWNrX21hcms6IHwgOmhlYXZ5X2NoZWNrX21hcms6IHwgOng6IHwgOng6IHwKfCAqKlZlcnQueCBXZWIgU2VydmVyIE9wZW5BUEkgcHJvamVjdCoqIHwgR2VuZXJhdGUgYSBza2VsZXRvbiBiYXNlZCBvbiBTd2FnZ2VyIDIvT3BlbkFQSSAzIHNwZWNpZmljYXRpb24gd2l0aCBzb3VyY2VzIGFuZCB0ZXN0cyBmb3IgVmVydC54IDMgV2ViIHBvd2VyZWQgUkVTVCBzZXJ2ZXIgfCA6aGVhdnlfY2hlY2tfbWFyazogfCA6aGVhdnlfY2hlY2tfbWFyazogfCA6eDogfCA6aGVhdnlfY2hlY2tfbWFyazogfCA6eDogfAp8ICoqVmVydC54IFdlYiBDbGllbnQgT3BlbkFQSSBwcm9qZWN0KiogfCBHZW5lcmF0ZSBhIGNsaWVudCBiYXNlZCBvbiBhIFN3YWdnZXIgMi9PcGVuQVBJIDMgc3BlY2lmaWNhdGlvbiB8IDpoZWF2eV9jaGVja19tYXJrOiB8IDpoZWF2eV9jaGVja19tYXJrOiB8IDp4OiB8IDpoZWF2eV9jaGVja19tYXJrOiB8IDp4OiB8CgpMaXN0IG9mIHN1cHBvcnRlZCBidWlsZCB0b29sczoKCiogTWF2ZW4KKiBHcmFkbGUKKiBOUE0KKiBOUE0gd2l0aCBqYXIgcGFja2FnaW5nICh0aGFua3MgdG8gW1ZlcnQueCBXZWJwYWNrIHBsdWdpbl0oaHR0cHM6Ly9naXRodWIuY29tL3BtbG9wZXMvd2VicGFjay12ZXJ0eC1wbHVnaW4pKQoKSWYgeW91IHdhbnQgdG8gYWRkIHlvdXIgb3duIGdlbmVyYXRvciBhbmQvb3IgYnVpbGQgdG9vbCBnaXZlIGEgbG9vayBhdCBbaG93IHRvIGNvbnRyaWJ1dGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9wbWxvcGVzL3NsdXNoLXZlcnR4L3dpa2kvSG93LXRvLWNvbnRyaWJ1dGUpCgojIyBJbnN0YWxsCgoxLiBJbnN0YWxsIFtgZ3VscC5qc2BdW2d1bHAtdXJsXQoyLiBJbnN0YWxsIFtTbHVzaF1bc2x1c2gtdXJsXQozLiBJbnN0YWxsIHRoZSBbYHNsdXNoLXZlcnR4YF1bZ2VuZXJhdG9yLXVybF0gZ2VuZXJhdG9yCgpgYGBzaAokIG5wbSBpbnN0YWxsIC1nIGd1bHAgc2x1c2ggc2x1c2gtdmVydHgKYGBgCgojIyBVc2FnZQoKRXhhbXBsZSBvZiBWZXJ0LnggU3RhcnRlciBnZW5lcmF0b3IKWyFbYXNjaWljYXN0XSguL2RlbW8uZ2lmKV0oaHR0cHM6Ly9hc2NpaW5lbWEub3JnL2EvbFIyM09Rck1uRFk2elpOZ2doOGtnVUFFNCkKCkV4YW1wbGUgb2YgVmVydC54IFdlYiBTdGFydGVyIGdlbmVyYXRvcgpbIVthc2NpaWNhc3RdKC4vZGVtbzIuZ2lmKV0oaHR0cHM6Ly9hc2NpaW5lbWEub3JnL2EvRE9aNjM5elV4a3NvTW80UHhEVWMxMnNOTykKCiMjIENvbnRyaWJ1dGUKCklmIHlvdSB3YW50IHRvIGFkZCBhIGdlbmVyYXRvciwgYWRkIHRlbXBsYXRlcyB0byBhbiBleGlzdGluZyBnZW5lcmF0b3Igb3Igc29tZXRoaW5nIGVsc2UgZ2l2ZSBhIGxvb2sgYXQgW2hvdyB0byBjb250cmlidXRlXShodHRwczovL2dpdGh1Yi5jb20vcG1sb3Blcy9zbHVzaC12ZXJ0eC93aWtpL0hvdy10by1jb250cmlidXRlKQoKIyMgQ2hhbmdlbG9nCgojIyMgMC4wLjMKCiogRml4ZWQgdGVtcGxhdGVzIGZvciBWZXJ0LnggMy41LjAgcmVsZWFzZQoqIFVwZGF0ZWQgYHN3YWdnZXIyb3BlbmFwaWAKCiMjIyAwLjAuMgoKKiBOZXcgbGlmZSB0byB0aGUgcHJvamVjdCEKCiMjIExpY2Vuc2UKCk1JVCDCqSBbUGF1bG8gTG9wZXNdKGh0dHA6Ly9qZXRkcm9uZS54eXopIGFuZCBbRnJhbmNlc2NvIEd1YXJkaWFuaV0oaHR0cDovL3NsaW5reWRldmVsb3Blci5naXRodWIuaW8pCgpbc2x1c2gtdXJsXTogaHR0cDovL3NsdXNoanMuZ2l0aHViLmlvCltndWxwLXVybF06IGh0dHA6Ly9ndWxwanMuY29tCltnZW5lcmF0b3ItdXJsXTogaHR0cHM6Ly9naXRodWIuY29tL3BtbG9wZXMvc2x1c2gtdmVydHgK readmeEtag: '"03060079455dd3a6a0461f53b538a4eb20e65fa9"' readmeLastModified: Tue, 26 Mar 2019 10:34:45 GMT repositoryId: 88743692 description: null created: '2017-04-19T12:42:00Z' updated: '2023-01-28T18:43:48Z' language: JavaScript archived: true stars: 36 watchers: 9 forks: 4 owner: pmlopes logo: https://avatars.githubusercontent.com/u/849467?v=4 repoEtag: '"771bdcae091b8ba0920e41dde10f1cc2cd70f534d4ad73f98e11ad75036c7902"' repoLastModified: Sat, 28 Jan 2023 18:43:48 GMT id: d98d45414234c8d035261bf960a46a04 - source: IMPLEMENTATIONS.md name: swac homepage: https://github.com/swaggest/swac language: - PHP - Go source_description: Generates clients for Go and PHP from OpenAPI 2/3. category: Code Generators foundInMaster: true repository: https://github.com/swaggest/swac repositoryMetadata: base64Readme: >- # OpenAPI 3.0 / Swagger 2.0 compiler

A tool to render API spec as code.

[![Build Status](https://travis-ci.org/swaggest/swac.svg?branch=master)](https://travis-ci.org/swaggest/swac)
[![codecov](https://codecov.io/gh/swaggest/swac/branch/master/graph/badge.svg)](https://codecov.io/gh/swaggest/swac)
[![Image Size](https://images.microbadger.com/badges/image/swaggest/swac.svg)](https://microbadger.com/images/swaggest/swac)
![Code lines](https://sloc.xyz/github/swaggest/swac/?category=code)
![Comments](https://sloc.xyz/github/swaggest/swac/?category=comments)

## Installation

### Phar

Download `swac` from [releases](https://github.com/swaggest/swac/releases) page.

### Docker

```bash
docker run --rm swaggest/swac swac --help
```

```
v0.1.28 swac
OpenAPI/Swagger compiler, https://github.com/swaggest/swac
Usage: 
   swac <action>
   action   Action name                                                      
            Allowed values: php-guzzle-client, go-client, js-client, markdown
...
```

Example

```bash
mkdir petstore && cd petstore
docker run -v $(pwd):/code -u 1000:1000 swaggest/swac swac php-guzzle-client https://raw.githubusercontent.com/OAI/OpenAPI-Specification/master/examples/v2.0/json/petstore.json --namespace MyApp\\Petstore
```

Schema can be passed by STDIN using `-` as file path.

```bash
cat ./openapi.json | docker run -i --rm swaggest/swac swac markdown -
```

### Composer

[Install PHP Composer](https://getcomposer.org/doc/00-intro.md)

```bash
composer global require swaggest/swac
```

## Usage

### PHP Client

```
swac php-guzzle-client --help
```

```
v0.1.0 swac php-guzzle-client
OpenAPI/Swagger compiler, https://github.com/swaggest/swac
Usage: 
   swac php-guzzle-client <schemaPath> --namespace <namespace>
   schemaPath   Path/URL to OpenAPI/Swagger schema
   
Options: 
   --operations <operations>      Operations filter in form of comma-separated list of method/path, default empty
   --project-path <projectPath>   Path to project root, default ./                                               
   --namespace <namespace>        Project namespace
```

The generated client depends on [`swaggest/rest-client`](https://github.com/swaggest/php-rest-client),
`guzzlehttp/guzzle` 6, and [`swaggest/json-schema`](https://github.com/swaggest/php-json-schema).

[Examples](/examples/php-guzzle-client).

### Go Client

```
swac go-client --help 
```

```
v0.1.14 swac go-client
OpenAPI/Swagger compiler, https://github.com/swaggest/swac
Usage: 
   swac go-client <schemaPath>
   schemaPath   Path/URL to OpenAPI/Swagger schema
   
Options: 
   --operations <operations>               Operations filter in form of comma-separated list of method/path, default empty                  
   --out <out>                             Path to output package, default ./client                                                         
   --pkg-name <pkgName>                    Output package name, default "client"                                                            
   --skip-default-additional-properties    Do not add field property for undefined `additionalProperties`                                   
   --skip-do-not-edit                      Skip adding "DO NOT EDIT" comments                                                               
   --add-request-tags                      Add field tags with name and location to request structure properties, e.g. 'ID int `query:"id"`'
   --show-const-properties                 Show properties with constant values, hidden by default                                          
   --keep-parent-in-property-names         Keep parent prefix in property name, removed by default                                          
   --ignore-nullable                       Add `omitempty` to nullable properties, removed by default                                       
   --ignore-xgo-type                       Ignore `x-go-type` in schema to skip generation                                                  
   --with-zero-values                      Use pointer types to avoid zero value ambiguity                                                  
   --fluent-setters                        Add fluent setters to struct fields                                                              
   --ignore-required                       Ignore if property is required when deciding on pointer type or omitempty                        
   --renames <renames...>                  Map of exported symbol renames, example From:To                                                  
   --with-tests                            Generate (un)marshaling tests for entities (experimental feature)                                
   --require-xgenerate                     Generate properties with `x-generate: true` only                                                 
   --validate-required                     Generate validation code to check required properties during unmarshal                           
```

[Examples](/examples/go-client).

The generated client is a single package without external dependencies.

### JavaScript Client

```
swac js-client --help 
```

```
v0.1.21 swac js-client
OpenAPI/Swagger compiler, https://github.com/swaggest/swac
Usage: 
   swac js-client <schema>
   schema   Path/URL to OpenAPI/Swagger schema
   
Options: 
   --operations <operations>      Operations filter in form of comma-separated list of method/path, default empty         
   --ignore-operation-id          Ignore operationId and always name operations using method and path                     
   --client-name <clientName>     Name of generated client class, default APIClient                                       
   --types-prefix <typesPrefix>   Prefix generated jsdoc class names                                                      
   --out <out>                    Path to output files, default ./client                                                  
   --patches <patches...>         JSON patches to apply to schema file before processing, merge patches are also supported
```

[Examples](/examples/js-client).

```
swac js-client openapi.json --out ./ --client-name Backend --types-prefix xh
```

The generated client is a ES5 class using `XMLHttpRequest` and `jsdoc` type definitions without external dependencies
suitable for direct usage in browsers.

### Markdown

```
swac markdown --help
```
```
v0.1.24 swac markdown
OpenAPI/Swagger compiler, https://github.com/swaggest/swac
Usage: 
   swac markdown <schema>
   schema   Path/URL to OpenAPI/Swagger schema
   
Options: 
   --operations <operations>         Operations filter in form of comma-separated list of method/path, default empty         
   --ignore-operation-id             Ignore operationId and always name operations using method and path                     
   --client-name <clientName>        Name of generated client class, default APIClient                                       
   --add-schema-url <addSchemaUrl>   Add schema link to the document                                                         
   --types-prefix <typesPrefix>      Prefix generated type names                                                             
   --out <out>                       Path to output files, default ./client                                                  
   --patches <patches...>            JSON patches to apply to schema file before processing, merge patches are also supported
 
```

[Examples](/examples/).

```
swac markdown openapi.json --out ./API-Docs.md --types-prefix xh
```

 readmeEtag: '"e29647ec6c25983479f1116c193104ad3253a656"' readmeLastModified: Mon, 19 Feb 2024 17:36:15 GMT repositoryId: 215669955 description: 🤖 OpenAPI/Swagger client generator for PHP, Go and JavaScript (ES5) created: '2019-10-17T00:32:59Z' updated: '2025-10-26T16:01:03Z' language: PHP archived: false stars: 10 watchers: 1 forks: 3 owner: swaggest logo: https://avatars.githubusercontent.com/u/19609628?v=4 license: MIT repoEtag: '"64d3e5c626249c2f5c9b6ae3d60a6ef6c92b48e5331c8aedce92eb90f7a2b890"' repoLastModified: Sun, 26 Oct 2025 16:01:03 GMT id: 1486b541e99b5e3e5827ff4adf62f77a - source: openapi3 tags repository: https://github.com/silohub/product-apis v3: true repositoryMetadata: base64Readme: >- WyFbUHVibGlzaCBBUEkgUGFja2FnZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9zaWxvaHViL3Byb2R1Y3QtYXBpcy9hY3Rpb25zL3dvcmtmbG93cy9wYWNrYWdlcy1wdWJsaXNoLnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vc2lsb2h1Yi9wcm9kdWN0LWFwaXMvYWN0aW9ucy93b3JrZmxvd3MvcGFja2FnZXMtcHVibGlzaC55bWwpClshW3BhZ2VzLWJ1aWxkLWRlcGxveW1lbnRdKGh0dHBzOi8vZ2l0aHViLmNvbS9zaWxvaHViL3Byb2R1Y3QtYXBpcy9hY3Rpb25zL3dvcmtmbG93cy9wYWdlcy9wYWdlcy1idWlsZC1kZXBsb3ltZW50L2JhZGdlLnN2Zz9icmFuY2g9Z2gtcGFnZXMpXShodHRwczovL2dpdGh1Yi5jb20vc2lsb2h1Yi9wcm9kdWN0LWFwaXMvYWN0aW9ucy93b3JrZmxvd3MvcGFnZXMvcGFnZXMtYnVpbGQtZGVwbG95bWVudCkKCiMjIFNldHVwIGRlbCByZXBvc2l0b3JpbwpQYXJhIGNvbXBpbGFyIGVzdGUgcmVwb3NpdG9yaW8gc2UgbmVjZXNpdGFuIHVuIHBhciBkZSBjb25maWd1cmFjaW9uZXMgYWRpY2lvbmFsZXM6CiMjIyBDb25zZWd1aXIgdW4gUGVyc29uYWwgYWNjZXNzIHRva2VuIChQQVQpIGRlIEdpdEh1YgpDb21vIG5vcyB0ZW5lbW9zIHF1ZSBjb25lY3RhciBhbCByZXBvc2l0b3JpbyBkZSBwYXF1ZXRlcyBwcml2YWRvIGRlIEdpdGh1YiwgbmVjZXNpdGFzIHR1IHVzdWFyaW8geSB1biBQQVQgcGFyYSBhdXRlbnRpY2FydGUuCi0gQWPDoSBzZSBnZW5lcmFuIGxvcyBQZXJzb25hbCBhY2Nlc3MgdG9rZW5zOiAoaHR0cHM6Ly9naXRodWIuY29tL3NldHRpbmdzL3Rva2VucykKLSBIYXkgcXVlIGdlbmVyYXIgdW4gdG9rZW4gY29uIGVsIHNjb3BlICoqcmVhZDpwYWNrYWdlcyoqIHBvciBsbyBtZW5vcwotIExhIGV4cGlyYWNpw7NuIHB1ZWRlbiBkZWphcmxhIGVuICoqIk5vIGV4cGlyYXRpb24iKiogc2kgZXMgcXVlIGRlamFuIHNvbG8gZWwgcGVybWlzbyBwYXJhIGxlZXIKLSBsdWVnbyB0ZW7DqXMgcXVlIGNyZWFyIGRvcyBwcm9waWVkYWRlcyBlbiB0dSBzaGVsbAogIC0gKipTSEdIX1VTRVIqKjogdHUgdXN1YXJpbyBHaXRIdWIKICAtICoqU0hHSF9UT0tFTioqOiBlbCBQQVQgcXVlIHRlIGJhamFzdGUgcmVjacOpbgotIEVzdGUgbWlzbW8gdG9rZW4gdmEgYSBzZXJ2aXIgcGFyYSBjb25lY3RhcnNlIGVuIGVsIHJlcG8gZGUgZnJvbnRlbmRzLCBiYWNrZW5kcy4uLiA6LSkKCiMgRXN0cnVjdHVyYSBkZSBkaXJlY3RvcmlvcwotIFsqKm9wZW5hcGkqKl0ob3BlbmFwaS9SRUFETUUubWQpOiB0aWVuZSBsYXMgZnVlbnRlcyBkZSBsYXMgQVBJLiBTZXBhcmFkYXMgZW4gbGEgZXN0cnVjdHVyYSBzdWdlcmlkYSBwb3IgT3BlbkFQSS4KICAtIF9fKi55YW1sX186IGxhIHBhcGE6IGFjw6EgZXN0w6FuIGxhcyBBUEkgZGVjbGFyYWRhcywgcXVlIHVzYW4gcGF0aHMgeSBjb21wb25lbnRzCiAgLSBbKipwYXRocyoqXShvcGVuYXBpL3BhdGhzL1JFQURNRS5tZCk6IHRpZW5lIGxhIGRlZmluaWNpw7NuIGRlIGxhcyBvcGVyYWNpb25lcwogIC0gWyoqY29tcG9uZW50cyoqXShvcGVuYXBpL2NvbXBvbmVudHMvUkVBRE1FLm1kKTogdGllbmUgcGFyw6FtZXRyb3MsIHJlcXVlc3RzLCByZXNwb25zZXMsIGV0Yy4uLiBBc29jaWFkYXMgYSBsb3MgcGF0aHMuCgoqKlVuYSBkZWZpbmljacOzbiBpbXBvcnRhbnRlLi4uKio6IGxvcyBmdWVudGVzIGFkZW50cm8gZGUgb3BlbmFwaSBmdW5jaW9uYW4gZGlyZWN0bywgbm8gaGF5IHF1ZSBnZW5lcmFyIG5hZGEsIHNlIHB1ZWRlbiB2ZXIgeSBuYXZlZ2FyIHNpbiBwcm9ibGVtYS4gCkxhcyBBUEkgZW4geWFtbCBubyBzZSB0b2NhbiBlbiBlbCBidWlsZC4gU2UgdG9jYW4gcG9yIGFmdWVyYSBjb24gc2NyaXB0cywgcGVybyBsb3MgeWFtbCBubyBjYW1iaWFuIGR1cmFudGUgZWwgYnVpbGQuCgojIyBRdWUgZ2VuZXJhbW9zCi0gdW4gb3BlbmFwaS55YW1sIGNvbXBsZXRvIHBvciBjYWRhIEFQSSwgcXVlIG5vIGRlcGVuZGUgZGUgb3Ryb3MgYXJjaGl2b3MsIGNvbW8gZXN0w6FuIGxvcyBvcmlnaW5hbGVzCi0gdW5hIGNsYXNlIGFic3RyYWN0YSBjb21vIHNlcnZlciBwdWJsaWNhZGEgY29tbyBwYXF1ZXRlIGVuIEdpdGh1YiAKLSB1bmEgY2xpZW50ZSBqYXZhc2NyaXB0IGNvbiBBeGlvcyBwYXJhIGVsIGZyb250LiBwdWJsaWNhZG8gY29tbyBwYXF1ZXRlIGVuIEdpdGh1YgotIGVsIHNpdGlvIGRlIGRvY3VtZW50YWNpw7NuLCBwb3IgYWhvcmEgc3ViaWRvIGEgZ2l0aHViIHBhZ2VzCgojIFZlcnNpb25hZG8gZGUgQVBJcwpMYXMgQVBJIG5vIHNlIHZlcnNpb25hbiBjb21vIGVsIGPDs2RpZ28sIHNpbm8gcXVlIG5vc290cm9zIGRlY2lkaW1vcyBjYW1iaWFyIGxhIHZlcnNpb24uIAoKIyBEZXBlbmRlbmNpYXMKQWxndW5hcyBkZSBsYXMgY29zYXMgcXVlIHVzYW1vcyBwYXJhIHB1YmxpY2FyIGxhcyBBUEkKLSBbUmFwaURvY10oaHR0cHM6Ly9tcmluOS5naXRodWIuaW8vUmFwaURvYy8pIGVzdG8gc2lydmEgcGFyYSBtb3N0cmFyIGxhIGFwaSBZYW1sIGVuIGZvcm1hdG8gZW50ZW5kaWJsZS4uLiDCoWVzdMOhIG11eSBidWVuYSEKCiMgTW9kaWZpY2FyIHRlbXBsYXRlcwotIHBhcmEgZXh0cmFlciBsb3MgdGVtcGxhdGVzIGRlIHNlcnZlcjogYHBucG0gZXhlYyBvcGVuYXBpLWdlbmVyYXRvci1jbGkgYXV0aG9yIHRlbXBsYXRlIC0tZ2VuZXJhdG9yLW5hbWUgamF2YS1taWNyb25hdXQtc2VydmVyIC0tb3V0cHV0IC5zaWxvaHViL3RlbXBsYXRlcy9zZXJ2ZXItcGFja2FnZXMvZ2VuZXJhdG9yYAotIHBhcmEgZXh0cmFlciBsb3MgdGVtcGxhdGVzIGRlIGNsaWVudCBqczogYHBucG0gZXhlYyBvcGVuYXBpLWdlbmVyYXRvci1jbGkgYXV0aG9yIHRlbXBsYXRlIC0tZ2VuZXJhdG9yLW5hbWUgdHlwZXNjcmlwdC1heGlvcyAtLW91dHB1dCAuc2lsb2h1Yi90ZW1wbGF0ZXMvY2xpZW50LWphdmFzY3JpcHQtcGFja2FnZXMvZ2VuZXJhdG9yYAotIHBhcmEgZXh0cmFlciBsb3MgdGVtcGxhdGVzIGRlIGNsaWVudCBqYXZhOiBgcG5wbSBleGVjIG9wZW5hcGktZ2VuZXJhdG9yLWNsaSBhdXRob3IgdGVtcGxhdGUgLS1nZW5lcmF0b3ItbmFtZSBqYXZhLW1pY3JvbmF1dC1jbGllbnQgLS1vdXRwdXQgLnNpbG9odWIvdGVtcGxhdGVzL2NsaWVudC1qYXZhLXBhY2thZ2VzL2dlbmVyYXRvcmAKLSBwYXJhIGV4dHJhZXIgbG9zIHRlbXBsYXRlcyBkZSBhcGktZmlsZXM6IGBwbnBtIGV4ZWMgb3BlbmFwaS1nZW5lcmF0b3ItY2xpIGF1dGhvciB0ZW1wbGF0ZSAtLW91dHB1dCBidWlsZC90ZW1wbGF0ZXMvYXBpLWZpbGVzIC0tZ2VuZXJhdG9yLW5hbWUgb3BlbmFwaS15YW1sYAo= readmeEtag: '"fb466307aa679925c9623e311f08e44d66c95e99"' readmeLastModified: Thu, 02 Jun 2022 14:03:18 GMT repositoryId: 395638198 description: APIs del producto created: '2021-08-13T12:00:44Z' updated: '2022-02-02T22:09:01Z' language: Mustache archived: false stars: 0 watchers: 1 forks: 0 owner: silohub logo: https://avatars.githubusercontent.com/u/88662712?v=4 license: MIT repoEtag: '"3712aadd9f8c7c79876c84ef0d19c8eafe60c1e8c998897a446b2b79585fdb81"' repoLastModified: Wed, 02 Feb 2022 22:09:01 GMT foundInMaster: true category: SDK id: a0ff9d2b34f7567c2fbbc3d4cfd79238 - source: openapi3 tags repository: https://github.com/unjello/openapi-mock-server v3: true repositoryMetadata: base64Readme: >- aW1hZ2U6Omh0dHA6Ly91bm1haW50YWluZWQudGVjaC9iYWRnZS5zdmdbbGluaz1odHRwOi8vdW5tYWludGFpbmVkLnRlY2gvXQ0KDQo9IERFUFJFQ0FURUQNCg0KX29wZW5hcGktbW9jay1zZXJ2ZXJfIGlzIG5vIGxvbmdlciBzdXBwb3J0ZWQsIHBsZWFzZSBjb25zaWRlciB1c2luZyBodHRwczovL2dpdGh1Yi5jb20vc3dhZ2dlci1hcGkvc3dhZ2dlci1wYXJzZXJbc3dhZ2dlci1wYXJzZXJdLg0K readmeEtag: '"4dbd01d7084b7379956c760183af4a64ada7198f"' readmeLastModified: Wed, 01 May 2019 08:49:23 GMT repositoryId: 128439257 description: DEPRECATED. Create mock server based on Open API 3.0 specification file created: '2018-04-06T18:39:09Z' updated: '2019-05-01T08:50:53Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 1 owner: unjello logo: https://avatars.githubusercontent.com/u/155669?v=4 license: CC0-1.0 repoEtag: '"78fe0466e487696955982b37cf4f1d8254bf1612e76a1354122873451daeea83"' repoLastModified: Wed, 01 May 2019 08:50:53 GMT foundInMaster: true category: - SDK - Code Generators id: a25664802eb7f4926581fd1e48ca435b - source: openapi3 tags repository: https://github.com/marioruiz/api-testing-example v3: true repositoryMetadata: base64Readme: >- IyBBUEkgdGVzdGluZyBleGFtcGxlCgpUaGUgcHVycG9zZSBvZiB0aGlzIGRvY3VtZW50IGlzIHRvIGJlIGFuIGVhc3kgZ3VpZGUgZm9yIGRldnMgYW5kIHRlc3RlcnMgdG8gc3RhcnQgdGVzdGluZyBSRVNUIEFQSXMganVzdCBpbiBtaW51dGVzLgoKRm9yIHRoZSBleGFtcGxlcyB3ZSB3aWxsIGJlIHRlc3RpbmcgdGhlIFsqKlViZXIqKl0oaHR0cHM6Ly9hcGkudWJlci5jb20pIEFQSSB1c2luZyBhIFlBTUwgU3dhZ2dlciAvIE9wZW4gQVBJIGZpbGUgYW5kIFsqKlJlcXJlcyoqXShodHRwczovL3JlcXJlcy5pbi8pIEFQSSB1c2luZyBhIG5vcm1hbCBSZXF1ZXN0IEhhc2ggZmlsZS4KClRvIGdlbmVyYXRlIHRoZSBSZXF1ZXN0IEhhc2hlcyBmcm9tIGEgKipTd2FnZ2VyKiogLyAqKk9wZW4gQVBJIGZpbGUqKiwgcnVuIHRoZSBmaWxlIC4vdXRpbHMvaW1wb3J0X3N3YWdnZXIucmIgb3IgdXNlIHRoZSAqKm9wZW5fYXBpX2ltcG9ydCoqIGNvbW1hbmQgbGluZSBleGVjdXRhYmxlLgoKSWYgeW91IHdhbnQgdG8gYXV0b21hdGljYWxseSBnZW5lcmF0ZSBSU3BlYyB0ZXN0cyBmb3IgZXZlcnkgZW5kIHBvaW50IG9mIHlvdXIgU3dhZ2dlciBmaWxlLCB1c2UgdGhlIGNyZWF0ZV90ZXN0cyBnZW06IGh0dHBzOi8vZ2l0aHViLmNvbS9NYXJpb1J1aXovY3JlYXRlX3Rlc3RzCgojIEluc3RhbGxhdGlvbgoKMS4gSW5zdGFsbCBSdWJ5ID49Mi40LiBSZWNvbW1lbmRlZCBsYXN0IDIuNSBzdGFibGUgcmVsZWFzZS4gVG8gc2VlIHdoaWNoIHZlcnNpb24geW91IGFscmVhZHkgaGF2ZSBpbnN0YWxsZWQ6IGBydWJ5IC12YAogICAgKiBbUnVieSBJbnN0YWxsYXRpb25dKGh0dHBzOi8vd3d3LnJ1YnktbGFuZy5vcmcvZW4vZG9jdW1lbnRhdGlvbi9pbnN0YWxsYXRpb24vKQoKMi4gSW5zdGFsbCB0aGUgbGlicmFyaWVzIHdlIHVzZSBieSBydW5uaW5nIGZyb20gcm9vdCBmb2xkZXI6CiAgICBgYGBiYXNoCiAgICBidW5kbGUgaW5zdGFsbAogICAgYGBgICAgIAogICAgCiMgRG9jdW1lbnRhdGlvbgoKIyMgR2VuZXJhbCAKICAqIFtSdWJ5IGluIDIwIG1pbnV0ZXNdKGh0dHBzOi8vd3d3LnJ1YnktbGFuZy5vcmcvZW4vZG9jdW1lbnRhdGlvbi9xdWlja3N0YXJ0LykKICAqIFtSdWJ5IGZyb20gb3RoZXIgbGFuZ3VhZ2VzXShodHRwczovL3d3dy5ydWJ5LWxhbmcub3JnL2VuL2RvY3VtZW50YXRpb24vcnVieS1mcm9tLW90aGVyLWxhbmd1YWdlcy8pCiAgKiBbUnVieSBDaGVhdCBTaGVldF0oaHR0cHM6Ly9sZWFybnhpbnltaW51dGVzLmNvbS9kb2NzL3J1YnkvKQoKIyMgTGlicmFyaWVzCiAgKiBbUlNwZWNdKGh0dHA6Ly9yc3BlYy5pbmZvLyksIFtSU3BlYyBDaGVhdFNoZWV0XShodHRwczovL2RldmhpbnRzLmlvL3JzcGVjKQogICogW25pY2VfaHR0cF0oaHR0cHM6Ly9naXRodWIuY29tL01hcmlvUnVpei9uaWNlX2h0dHApCiAgKiBbbmljZV9oYXNoXShodHRwczovL2dpdGh1Yi5jb20vTWFyaW9SdWl6L25pY2VfaGFzaCkKICAqIFtvcGVuX2FwaV9pbXBvcnRdKGh0dHBzOi8vZ2l0aHViLmNvbS9NYXJpb1J1aXovb3Blbl9hcGlfaW1wb3J0KQogICogW2NyZWF0ZV90ZXN0c10oaHR0cHM6Ly9naXRodWIuY29tL01hcmlvUnVpei9jcmVhdGVfdGVzdHMpCgojIFJ1bm5pbmcgdGVzdHMKClRvIHJ1biBhbGwgdGhlIHRlc3RzOgpgYGBiYXNoCnJzcGVjCmBgYAoKVG8gcnVuIGEgcGFydGljdWxhciB0ZXN0IGZpbGU6CmBgYGJhc2gKcnNwZWMgLi9zcGVjL215X3Rlc3Rfc3BlYy5yYgpgYGAKClRvIHJ1biBhIHBhcnRpY3VsYXIgdGVzdCAoZXhhbXBsZSkgaW5zaWRlIGEgdGVzdCBmaWxlLCBhZGQgdGhlIGxpbmUgbnVtYmVyIHdoZXJlIHRoZSB0ZXN0IGlzOgpgYGBiYXNoCnJzcGVjIC4vc3BlYy9teV90ZXN0X3NwZWMucmI6NDIKYGBgCgpUaGUgZGVmYXVsdCB2YWx1ZXMgdG8gc2V0IGFsbCB0ZXN0cyBhcmUgb24gYHNldHRpbmdzYCBmb2xkZXIuIFlvdSBjYW4gcGFzcyBwYXJhbWV0ZXJzIHRvIG92ZXJ3cml0ZSB0aGUgc2V0dGluZ3MgaW4gY29tbWFuZCBsaW5lIG9yIEVOViB2YXJpYWJsZXMuCgpgYGBiYXNoCkhPU1Q9MTAuMjAuMzAuNTAgcnNwZWMgLi9zcGVjL215X3Rlc3Rfc3BlYy5yYgpgYGAK readmeEtag: '"a98c8c59ec2fc91e4612b66728595907eb251818"' readmeLastModified: Thu, 14 Mar 2019 11:36:47 GMT repositoryId: 170314956 description: >- Example using RSpec and nice_http to test REST APIs. (Uber API and Reqres API) created: '2019-02-12T12:33:54Z' updated: '2019-03-14T11:37:03Z' language: Ruby archived: false stars: 0 watchers: 1 forks: 0 owner: MarioRuiz logo: https://avatars.githubusercontent.com/u/459564?v=4 repoEtag: '"c57729e620f8c530d98bc017b3ac017804b58d9d9d295280a0b6f47f274a0479"' repoLastModified: Thu, 14 Mar 2019 11:37:03 GMT foundInMaster: true category: Testing id: cf282c3053c42bb7de2bfa9100ac7b1f - source: openapi3 tags repository: https://github.com/tomi77/sent-geo-openapi v3: true repositoryMetadata: base64Readme: IyBzZW50LWdlby1vcGVuYXBpClNFTlQgR0VPIE9wZW5BUEl2MyBzY2hlbWEK readmeEtag: '"c9374ddec23048acf9518419e6959bf796b6837b"' readmeLastModified: Fri, 15 May 2020 09:28:00 GMT repositoryId: 263333160 description: SENT GEO OpenAPIv3 schema created: '2020-05-12T12:42:15Z' updated: '2020-05-15T09:29:44Z' language: null archived: false stars: 0 watchers: 1 forks: 0 owner: tomi77 logo: https://avatars.githubusercontent.com/u/490094?v=4 license: MIT repoEtag: '"536f5c9d4f34f6700e8168ea4431251238ef94812ffc41faf48d426ed7308742"' repoLastModified: Fri, 15 May 2020 09:29:44 GMT foundInMaster: true category: - Data Validators - Parsers id: b9dba61fa79179577c35d58901bca143 - source: openapi3 tags repository: https://github.com/cakirmuha/auction-bid-tracker v3: true repositoryMetadata: base64Readme: >- IyBhdWN0aW9uLWJpZC10cmFja2VyCgpTZXJ2ZXIgZm9yIEF1Y3Rpb24gQmlkIFRyYWNrZXIKCiMjIyBCdWlsZAoKICAgICMgQ2xvbmUKICAgIGdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vY2FraXJtdWhhL2F1Y3Rpb24tYmlkLXRyYWNrZXIuZ2l0CiAgICBjZCBhdWN0aW9uLWJpZC10cmFja2VyCgogICAgZXhwb3J0IEdPMTExTU9EVUxFPW9uCgogICAgIyBHZW5lcmF0ZSBhc3NldHMKICAgIGdvIGdlbmVyYXRlIC4vY21kL2F1Y3Rpb24tYmlkLXRyYWNrZXIvYXNzZXRzL2dlbi5nbyAKCiAgICAjIENvbXBpbGUKICAgIGdvIGJ1aWxkIC4vY21kL2F1Y3Rpb24tYmlkLXRyYWNrZXIKCiMjIyBSdW4KICAgCiAgICBTZXQgdXAgZW52IHZhcnMsIHRoZW46CiAgIAogICAgLi9hdWN0aW9uLWJpZC10cmFja2VyCgojIyMgT3B0aW9ucwoKICAgIFVzYWdlIG9mIC4vYXVjdGlvbi1iaWQtdHJhY2tlcjoKICAgICAgLWxpc3RlbiBzdHJpbmcKICAgICAgICAgICAgTGlzdGVuIGFkZHIgKGRlZmF1bHQgIjo4MTgxIikKICAgICAgLWxvZyBzdHJpbmcKICAgICAgICAgICAgTG9nIGxldmVsIChkZWJ1ZywgaW5mbywgd2FybiwgZXJyb3IpIChkZWZhdWx0ICJkZWJ1ZyIpCiAgICAgICAgICAgIAojIyMgQ29uY3VycmVuY3kgVGVzdAoKICAgICBnbyB0ZXN0IC4vY21kL2JpZC50cmFja2VyL2NvbmN1cnJlbmN5X3Rlc3QuZ28KICAgICAgICAgICAgCiMjIyBHZW5lcmF0ZQoKR2VuZXJhdGVkIEFQSSBzcGVjIChvcGVuYXBpLnltbCkgaXMgc2VydmVkIGF0IGAvYXBpL3YxL2Fzc2V0cy9vcGVuYXBpLnltbGAKCiMjIyBHbyBWZXQKCiAgICBnbyB2ZXQgLi8uLi4KCiMjIFNvbHV0aW9uIEluc3RydWN0aW9uCgpJbiB0aGlzIHRhc2ssIHdlIGhhdmUgMyBvYmplY3RzKFVzZXIsIEl0ZW0sIEJpZCkuIAoxMDAgbW9jayB1c2VycyBhbmQgNCBtb2NrIGl0ZW1zIGFyZSBjcmVhdGVkIGJ5IHN0YXJ0aW5nIHRoZSBhcHBsaWNhdGlvbi4gTWFwIGRhdGEgc3RydWN0dXJlIGlzIHVzZWQgdG8gYWNjZXNzIGFsbCAzIG9iamVjdHMuIApNYXBzIGFyZSBub3QgdGhyZWFkLXNhZmUgYnkgZGVmYXVsdCBpbiBnb2xhbmcsIHNvIHN5bmMuTWFwIGNhbiBiZSB1c2VkIHRvIGhhbmRsZSBjb25jdXJyZW5jeSBwcm9ibGVtcy4gRGVmYXVsdCBtYXAgaXMgcHJlZmVyZWQgdG8gaGFuZGxlIGNvbmN1cnJlbmN5IGlzc3VlcyBieSBjb2RlIGluIHNvbWUgcGFydCBvZiB0aGUgY29kZS4KLSBVc2VyIG1hcCh0byBhY2Nlc3MgdXNlciBlbGVtZW50cyBmb3IgYSB1c2VyKSAgICAtIGRlZmF1bHQgbWFwOiBzeW5jLlJXTXV0ZXggaXMgdXNlZCB0byByZWFkIHVzZXIgZGF0YS4KICAtIGtleTogdXNlcmlkLCB2YWx1ZTogVXNlciBvYmplY3QKLSBJdGVtIG1hcCh0byBhY2Nlc3MgaXRlbSBlbGVtZW50cyBmb3IgYW4gaXRlbSkgICAtIGRlZmF1bHQgbWFwOiBzeW5jLlJXTXV0ZXggaXMgdXNlZCB0byByZWFkIGl0ZW0gZGF0YS4KICAtIGtleTogaXRlbWlkLCB2YWx1ZTogSXRlbSBvYmplY3QKLSBJdGVtQmlkIG1hcCh0byBhY2Nlc3MgYmlkIGVsZW1lbnRzIGZvciBhbiBpdGVtKSAtIHN5bmMuTWFwOiBObyBuZWVkIHRvIGNvbnNpZGVyIGNvbmN1cnJlbmN5IHByb2JsZW1zIGxpa2UgZGVhZGxvY2ssIHN5bmNocm9uaXphdGlvbi4uLgogIC0ga2V5OiBpdGVtaWQsIHZhbHVlOiBMaXN0IG9mIGJpZHMKICAtIExpc3Qgb2YgYmlkczogTGlua2VkIGxpc3QgaXMgY2hvc2VuIHRvIGFjY2VzcyBiaWRzIGZvciBhbiBpdGVtLiBXaGVuIGEgbmV3IGJpZCBjb21lLCBiaWQgaXMgcHJlcGVuZGVkIHRvIGxpc3QsIHNvIGhlYWQgb2YgbGlzdCB3aWxsIGJlIHRoZSBsYXN0IGJpZCBhbmQvb3IgdGhlIGJpZCBoYXZpbmcgdGhlIGhpZ2hlc3QgYW1vdW50LiBTbGljZSBjYW4gYWxzbyBiZSB1c2VkIGZvciBsaXN0IG9mIGJpcmRzLCBidXQgaXQgd2lsbCBiZSBzb3J0ZWQgYnkgYXNjZW5kaW5nKG5vdCBzbyBpbXBvcnRhbnQgc2luY2Ugd2UgYWx3YXlzIG5lZWQgdG8gYWNjZXNzIHRoZSBoaWdoZXN0IGVsZW1lbnQgdG8gY2hlY2sgYmlkIGFtb3VudCwgYW5kIGl0IGlzIGVhc2lseSBhY2Nlc3NpYmxlIGJ5IGJvdGggbGlua2VkIGxpc3QgYW5kIHNsaWNlIHdpdGhvdXQgdmlzaXRpbmcgbGlzdCkuIElmIHdlIHdhbnQgdG8gZGVsZXRlIGEgdXNlciBhbmQvb3IgcmVtb3ZlIGFsbCBiaWRzIG9mIHVzZXIgZm9yIGFuIGl0ZW0sIHRoZSBjb3N0IG9mIGRlbGV0ZS9hZGQgb2YgbGlua2VkIGxpc3QgaXMgbG93ZXIgdGhhbiBzbGljZShpdCBjYW4gYmUgdGhvdWdodCBhcyB0aGUgcGx1cyBvZiBsaW5rZWQgbGlzdCkuCgojIyMgU29sdXRpb24gdGVzdAoKQXBwbGljYXRpb24gY2FuIGJlIHRlc3RlZCB1c2luZyBHb3JvdXRpbmVzKGVnLiBgZ28gdGVzdCAuL2NtZC9iaWQudHJhY2tlci9jb25jdXJyZW5jeV90ZXN0LmdvYCkuIEJpZCByZXF1ZXN0IGlzIHNlbnQgYnkgYWxsIDEwMCBtb2NrIHVzZXJzIGZvciBhbiBpdGVtLCB0aGVyZSB3aWxsIGJlIG5vIHJ1bnRpbWUgZXJyb3IsIGFuZCB3aW5uaW5nIGJpZCB3aWxsIGJlIGVxdWFsIHRvIHRoZSBoaWdoZXN0IGJpZCBhbW91bnQgb2YgdXNlcnMuCg== readmeEtag: '"c467d2d3031390bb4adffc3c2e046fda6651bb60"' readmeLastModified: Sun, 05 Apr 2020 10:40:19 GMT repositoryId: 252986832 description: null created: '2020-04-04T12:09:02Z' updated: '2020-04-16T07:26:20Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: cakirmuha logo: https://avatars.githubusercontent.com/u/6494452?v=4 license: MIT repoEtag: '"2571c474014cb776e28507d88dced788ade54f5529cca9c31e96397f055586d0"' repoLastModified: Thu, 16 Apr 2020 07:26:20 GMT foundInMaster: true category: Testing id: 7e4760fa3b2c99c3838842cebd762686 - source: openapi3 tags repository: https://github.com/ideadapt/intellij-swagger-annotations v3: true repositoryMetadata: repositoryId: 295204974 description: >- Intellij plugin providing editor features related to swagger and openapi annotations created: '2020-09-13T17:44:50Z' updated: '2025-10-17T08:08:49Z' language: Java archived: false stars: 1 watchers: 1 forks: 0 owner: ideadapt logo: https://avatars.githubusercontent.com/u/1071386?v=4 repoEtag: '"d8ed63c089e9fdb24cba04ec209bd034a2b6c08fb8d55fe419c6eba576e877a1"' repoLastModified: Fri, 17 Oct 2025 08:08:49 GMT foundInMaster: true id: 8ac9057379e79f6ded4e8a73116a1e47 - source: openapi3 tags repository: https://github.com/witguild/spring-boot-rest-api-h2-jpa-security v3: true repositoryMetadata: repositoryId: 339035926 description: null created: '2021-02-15T10:09:18Z' updated: '2021-02-15T11:07:41Z' language: Java archived: false stars: 0 watchers: 1 forks: 0 owner: witguild logo: https://avatars.githubusercontent.com/u/62769457?v=4 repoEtag: '"ef1b42f783b96cc1100479584c41e96c7faf3e0e1ef15f064d01a9dad3f59d61"' repoLastModified: Mon, 15 Feb 2021 11:07:41 GMT foundInMaster: true id: 877a6f167d8f020d30a4e974c750aa08 - source: openapi3 tags repository: https://github.com/nexys-system/postman-openapi-ui v3: true id: 44b760c78f35f82d9790c9f325a67a64 repositoryMetadata: base64Readme: >- IyBQb3N0bWFuIHRvIE9wZW5BUEkgVUkKClRoaXMgaXMgYSB3cmFwcGVyIGFyb3VuZCBbcG9zdG1hbi10by1vcGVuYXBpXShodHRwczovL2dpdGh1Yi5jb20vam9vbGZlL3Bvc3RtYW4tdG8tb3BlbmFwaSkgc28gdGhhdCB0aGUgb3BlbiBhcGkgZmlsZSBjYW4gYmUgZ2VuZXJhdGVkIGRpcmVjdGx5IGZyb20gd2l0aGluIHRoZSBicm93c2VyCgpbIVtUZXN0XShodHRwczovL2dpdGh1Yi5jb20vbmV4eXMtc3lzdGVtL3Bvc3RtYW4tb3BlbmFwaS11aS9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbmV4eXMtc3lzdGVtL3Bvc3RtYW4tb3BlbmFwaS11aS9hY3Rpb25zL3dvcmtmbG93cy90ZXN0LnltbCkKWyFbLmdpdGh1Yi93b3JrZmxvd3MvZGVwbG95LnltbF0oaHR0cHM6Ly9naXRodWIuY29tL25leHlzLXN5c3RlbS9wb3N0bWFuLW9wZW5hcGktdWkvYWN0aW9ucy93b3JrZmxvd3MvZGVwbG95LnltbC9iYWRnZS5zdmcpXShodHRwczovL2dpdGh1Yi5jb20vbmV4eXMtc3lzdGVtL3Bvc3RtYW4tb3BlbmFwaS11aS9hY3Rpb25zL3dvcmtmbG93cy9kZXBsb3kueW1sKQoKIyMgR2V0IHN0YXJ0ZWQKCmh0dHBzOi8vbmV4eXMtc3lzdGVtLmdpdGh1Yi5pby9wb3N0bWFuLW9wZW5hcGktdWkvCg== readmeEtag: '"e9ef6c792b3c639b5f17070cd0f2005294c2ff35"' readmeLastModified: Thu, 21 Jul 2022 07:08:50 GMT repositoryId: 513859855 description: generate Open API from Postman collection created: '2022-07-14T10:39:45Z' updated: '2022-07-16T11:07:42Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: nexys-system logo: https://avatars.githubusercontent.com/u/71571169?v=4 repoEtag: '"66571162ff2ea53f0c5ddb3d0501cca20ec5cb29495baea2a398aaac2a5b5046"' repoLastModified: Sat, 16 Jul 2022 11:07:42 GMT category: Testing foundInMaster: true - source: openapi3 tags repository: https://github.com/fromsi/example_laravel_rest_api v3: true id: 24d47d3e0ba1cc7f2b8064a1bc0a3f79 repositoryMetadata: base64Readme: >- IyMg0KPRgdGC0LDQvdC+0LLQutCwINC/0YDQvtC10LrRgtCwCi0g0KHQutC70L7QvdC40YDQvtCy0LDRgtGMINC/0YDQvtC10LrRggotINCX0LDQv9GD0YHRgtC40YLRjCDQtNC+0LrQtdGACi0g0JfQsNC/0YPRgdGC0LjRgtGMINC60L7QvNCw0L3QtNGDINCyINC60L7RgNC90LUg0L/RgNC+0LXQutGC0LAgYG1ha2UgaW5zdGFsbGAKCiMjINCX0LDQtNCw0YfQsArQldGB0YLRjCDQu9C+0LrQsNGG0LjQuC4g0JIg0LvQvtC60LDRhtC40Y/RhSDRhdGA0LDQvdC40LvQuNGJ0LAuINCSINGF0YDQsNC90LjQu9C40YnQsNGFINGC0LXQvNC/0LXRgNCw0YLRg9GA0LAg0Lgg0L7Qv9GA0LXQtNC10LvQtdC90L3Ri9C5INC+0LHRitC10Lwg0LLQvNC10YHRgtC40LzQvtGB0YLQuC4g0J/QvtC70L3QvtC1INC+0L/QuNGB0LDQvdC40LUgW0lORk8ubWRdKElORk8ubWQpCgojIyDQndGD0LbQvdC+INC+0LHRj9C30LDRgtC10LvRjNC90L4g0LjQvNC10YLRjAotIERvY2tlciAodjIwKykKLSBEb2NrZXItQ29tcG9zZSAodjEuMjkrKQotIEdOVSBNYWtlICh2MyspCgojIyDQltC10LvQsNGC0LXQu9GM0L3QviDQuNC80LXRgtGMINGB0LLQvtCx0L7QtNC90YvQtSDQv9C+0YDRgtGLCtCV0YHQu9C4INC/0L7RgNGCINC30LDQvdGP0YIsINC80L7QttC90L4g0LjQt9C80LXQvdC40YLRjCDQv9C+0YDRgtGLINCyIGBkb2NrZXIvZG9ja2VyLWNvbXBvc2Uub3ZlcnJpZGUueW1sYCDQuCBgLmVudmAg0L/QvtGB0LvQtSDQutC+0LzQsNC90LTRiyBgbWFrZSBpbnN0YWxsYC4KLSA4MAotIDQ0MwotIDU0MzIKLSA4MDgwCgojIyBYZGVidWcg0LTQu9GPIFBocFN0b3JtCtCSIGBQaHBTdG9ybSA+IFByZWZlcmVuY2VzID4gU2VydmVycyA+ICtgCi0g0JIgTmFtZSDQvdGD0LbQvdC+INC30LDQv9C40YHQsNGC0YwgYGNoYXRgCi0g0JIgSG9zdCDQvdGD0LbQvdC+INC30LDQv9C40YHQsNGC0YwgYGxvY2FsaG9zdGAKLSDQndCw0LnRgtC4INCyIGBQcm9qZWN0IGZpbGVzYCDQtNC40YDQtdC60YLQvtGA0LjRjiDQv9GA0L7QtdC60YLQsCDQuCDQv9GA0L7Qv9C40YHQsNGC0Ywg0Y3RgtC+0Lkg0LTQuNGA0LXQutGC0L7RgNC40LggYC92YXIvd3d3YAoKIyMg0KTQuNC70YzRgtGA0LDRhtC40Y8g0LIgUkVTVCBBUEkK0JjRgdC/0L7Qu9GM0LfRg9C10YLRgdGPINCx0LjQsdC70LjQvtGC0LXQutCwIFtzcGF0aWUvbGFyYXZlbC1xdWVyeS1idWlsZGVyXShodHRwczovL2dpdGh1Yi5jb20vc3BhdGllL2xhcmF2ZWwtcXVlcnktYnVpbGRlcikKCtCf0YDQuNC80LXRgNGLOgotINCh0L7RgNGC0LjRgNC+0LLQutCwIOKAkyBgL2VuZHBvaW50P3NvcnQ9aWRgCi0g0KTQuNC70YzRgtGA0LDRhtC40Y8g4oCTIGAvZW5kcG9pbnQ/ZmlsdGVyW25hbWVdPUZyaWRnZWAKLSDQn9C+0LTQutC70Y7Rh9C10L3QuNC1INGB0YPRidC90L7RgdGC0LXQuSDigJMgYC9lbmRwb2ludD9pbmNsdWRlPXN1YmplY3RgCgojIyDQpdC+0YHRgtGLINC00LvRjyDRgNCw0LHQvtGC0Ysg0YEg0L/RgNC+0LXQutGC0L7QvAotIGh0dHA6Ly9sb2NhbGhvc3Q6ODA4MCDigJMg0LTQu9GPINGA0LDQsdC+0YLRiyDRgSDQkdCUCi0gaHR0cDovL2xvY2FsaG9zdCDigJMg0LTQu9GPINGA0LDQsdC+0YLRiyDRgSBSRVNUIEFQSQoKIyMg0KPQv9GA0LDQstC70LXQvdC40LUg0L/RgNC+0LXQutGC0L7QvAotIGBtYWtlIGluc3RhbGxgIOKAkyDRg9GB0YLQsNC90L7QstC60LAg0L/RgNC+0LXQutGC0LAg0YfQtdGA0LXQtyBEb2NrZXIKLSBgbWFrZSBydW5gIOKAkyDQt9Cw0L/Rg9GB0Log0L/RgNC+0LXQutGC0LAg0YfQtdGA0LXQtyBEb2NrZXIKLSBgbWFrZSBzdG9wYCDigJMg0L7RgdGC0LDQvdC+0LLQutCwINC/0YDQvtC10LrRgtCwINGH0LXRgNC10LcgRG9ja2VyCi0gYG1ha2UgcGhwYCDigJMg0L/QtdGA0LXQudGC0Lgg0LIg0LrQvtC90YLQtdC50L3QtdGAINGBIHBocCDRh9C10YDQtdC3IERvY2tlcgoKIyMg0JLQvtC30LzQvtC20L3QvtGB0YLQuCDQv9GA0L7QtdC60YLQsCwg0LrQvtGC0L7RgNGL0LUg0YLQtdCx0Y8g0LfQsNC40L3RgtC10YDQtdGB0YPRjtGCCi0g0J/QvtC00L3Rj9GC0LjQtSDQv9GA0L7QtdC60YLQsCDQvdCw0L/QuNGB0LDQsiDQstGB0LXQs9C+INC70LjRiNGMINC+0LTQvdGDINC60L7QvNCw0L3QtNGDCi0g0JfQsNC/0YDQvtGB0Ysg0L7QsdGA0LDQsdCw0YLRi9Cy0LDRjtGC0YHRjyDQv9C+INCw0YDRhdC40YLQtdC60YLRg9GA0L3QvtC80YMg0YHRgtC40LvRjiBSRVNUIEFQSQotIEhUVFAgZW5kcG9pbnQn0Ysg0L/QvtC00LTQtdGA0LbQuNCy0LDRjtGCINCy0LXRgNGB0LjQvtC90L3QvtGB0YLRjCwg0YHQvtGA0YLQuNGA0L7QstC60YMsINGE0LjQu9GM0YLRgNCw0YbQuNGOINC4INC/0L7QtNC60LvRjtGH0LXQvdC40LUg0YHRg9GJ0L3QvtGB0YLQtdC5Ci0g0JTQvtC60YPQvNC10L3RgtCw0YbQuNGPINC/0YDQvtC10LrRgtCwLCDQv9C+0LTQtNC10YDQttC60LAg0YLQuNC/0LjQt9Cw0YbQuNC4INC4INC00LLQuNC20LXQvdC40Y8g0LIg0L3QvtCz0YMg0YHQviDQstGA0LXQvNC10L3QtdC8IFBIUDhcTGFyYXZlbDkKLSDQodC40LTQtdGA0Ysg0YEg0YTQsNCx0YDQuNC60LDQvNC4INC00LvRjyDQt9Cw0L/QvtC70L3QtdC90LjRjyDQtNCw0L3QvdGL0LzQuCDRh9C10YDQtdC3IGZha2VyCi0g0KDQtdCz0LjRgdGC0YDQsNGG0LjRjy/QkNGD0YLQtdC90YLQuNGE0LjQutCw0YbQuNGPL9CS0L7RgdGB0YLQsNC90L7QstC70LXQvdC40LUv0JjQt9C80LXQvdC10L3QuNC1INC/0L7Qu9GM0LfQvtCy0LDRgtC10LvRjwotINCb0L7QutCw0YbQuNC4INCz0L7RgNC+0LTQvtCyINC4INGB0YLRgNCw0L0K readmeEtag: '"c928334b69760d3685ef9e7d75877cbf282f0537"' readmeLastModified: Sat, 23 Jul 2022 14:43:49 GMT repositoryId: 514539373 description: Example PHP Laravel created: '2022-07-16T09:54:41Z' updated: '2023-04-02T17:42:12Z' language: PHP archived: true stars: 0 watchers: 1 forks: 0 owner: FromSi logo: https://avatars.githubusercontent.com/u/22871855?v=4 repoEtag: '"460e0d1785dad6493dd936a32146f010129f3c3b45eff4c066e33f7cc5d89823"' repoLastModified: Sun, 02 Apr 2023 17:42:12 GMT category: Mock foundInMaster: true - source: openapi3 tags repository: https://github.com/romabilka/blogo v3: true id: a9c8b26cad59924f6aa6fffaf25640f4 repositoryMetadata: base64Readme: >- IyBCbG9HbwoKIyMjIEV4YW1wbGVzIGRpZmZlcmVudCBjb2RlIGZvciBibG9nCgoqIERpZmZlcmVudCBwcm9qZWN0cyBzdHJ1Y3R1cmUgW3Byb2plY3QtbGF5b3V0XSgvcHJvamVjdC1sYXlvdXQvKQoqIEV4YW1wbGVzIFt0ZXN0c10oL3Rlc3RzLykKKiBFeGFtcGxlIFtXZWJTb2NrZXQgdGVzdF0oL3Rlc3Qtd2Vic29ja2V0LykKKiBBdXRvbWF0ZWQgSW5pdGlhbGl6YXRpb24gaW4gR28gd2l0aCBbd2lyZV0oL2V4YW1wbGUtd2lyZS8pCiogRXhhbXBsZSB1c2Ugc3RydWN0IFt0YWdzXSgvc3RydWN0LXRhZ3MvKQoKCiMjIExpY2Vuc2UKW01JVCBsaWNlbnNlXShMSUNFTlNFLm1kKS4= readmeEtag: '"616329cb0211a6ce089620d4fb7fd28625ffa4e0"' readmeLastModified: Fri, 27 May 2022 11:48:23 GMT repositoryId: 462227574 description: null created: '2022-02-22T09:35:04Z' updated: '2022-12-13T19:47:59Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: RomaBilka logo: https://avatars.githubusercontent.com/u/28790784?v=4 license: NOASSERTION repoEtag: '"5a2020748934f0b3ec27d2e0d62b14429cf253b404d081a4ae95985cfd5e4bb6"' repoLastModified: Tue, 13 Dec 2022 19:47:59 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/sudokuru/useractivegamesbff v3: true id: 40f15de6f7736dbdb83303d3d0c2f28f repositoryMetadata: base64Readme: >- PiBbIUlNUE9SVEFOVF0gIAo+IFRoaXMgcmVwb3NpdG9yeSBpcyByZWFkLW9ubHkgLyBhcmNoaXZlZCBhbmQgd2lsbCBub3QgcmVjaWV2ZSB1cGRhdGVzLgoKIyBbVHlwZWRvYyBEb2N1bWVudGF0aW9uIFdlYnNpdGVdKGh0dHBzOi8vc3Vkb2t1cnUuZ2l0aHViLmlvL0JhY2tlbmQvKTxicj4KCiMgVG9kbwoKLSBbIF0gQWRkIGxpY2Vuc2UgZmlsZSBhbmQgZGlzdHJpYnV0ZSB0byBhbGwgcmVwb3MgdXNpbmcgR2l0SHViIEFjdGlvbiAoVGhvbWFzKQotIFsgXSBGaW5pc2ggd3JpdGluZyBpbnRlZ3JhdGlvbiB0ZXN0cyBmb3IgcHV6emxlIGVuZHBvaW50IChUaG9tYXMpCi0gWyBdIFdyaXRlIEdpdEh1YiBob29rIHRvIHJ1biBhbGwgdGVzdHMgYmVmb3JlIFB1c2ggdG8gcmVwbyAoVGhvbWFzKQotIFt4XSBTZXQgdXAgRGV2IGFuZCBQcm9kIExhbWJkYSBlbnZpcm9ubWVudHMgKFRob21hcy9HcmVnb3J5KQotIFsgXSBBZGQgTWVybWFpZCBkb2N1bWVudGF0aW9uIGFuZCBkaXN0cmlidXRlIHRvIGFsbCByZXBvcyB1c2luZyBHaXRIdWIgQWN0aW9uIChUaG9tYXMpCi0gWyBdIENsZWFuIHVwIGRvY2tlciBpbXBsZW1lbnRhdGlvbiAoYXV0by1yZWJ1aWxkKSAoVGhvbWFzKQotIFsgXSBEaXNwbGF5IGludGVncmF0aW9uIHRlc3QgcmVzdWx0cyB3aXRoIHJlcG9ydGVyIChUaG9tYXMpCi0gWyBdIERlY2lkZSBvbiBpbml0aWFsIEpTT04gc3RydWN0dXJlIGZvciByZW1haW5pbmcgZW5kcG9pbnRzIChUZWFtKQotIFsgXSBXcml0ZSBsb2dpYyBmb3IgcmVtYWluaW5nIGVuZHBvaW50cyAoRGFuaWVsKQotIFsgXSBXcml0ZSBzYW5pdGF0aW9uIGFuZCB2YWxpZGF0aW9uIGZvciByZW1haW5pbmcgZW5kcG9pbnRzIChEYW5pZWwpCi0gWyBdIFdyaXRlIFBvc3RtYW4gaW50ZWdyYXRpb24gdGVzdHMgZm9yIHJlbWFpbmluZyBlbmRwb2ludHMgKERhbmllbCkKLSBbIF0gU2V0IHVwIEF1dGgwIHRva2VuIGF1dGhlbnRpY2F0aW9uIChUaG9tYXMvRGFuaWVsKQotIFsgXSBXcml0ZSB1cCBPcGVuQVBJIHNwZWNpZmljYXRpb25zIGZvciBlbmRwb2ludHMgKFRob21hcy9EYW5pZWwpCi0gWyBdIFJlc29sdmUgcmVtYWluaW5nIGBgYC8vdG9kb2BgYCBpdGVtcyAoVGhvbWFzL0RhbmllbCkKLSBbIF0gRGV0ZXJtaW5lIGhvdyB0byBzZXQgUHJvZCBlbnZpcm9ubWVudCB0byB1c2UgdmVyc2lvbmluZyBjb250cm9sIChUaG9tYXMvR3JlZ29yeSkKLSBbIF0gSW1wbGVtZW50IHVuaXQgdGVzdHMgaWYgbmVlZGVkIChUaG9tYXMvRGFuaWVsKQoKIyBEZXZlbG9wZXIgU2V0dXAKCjEuIEluc3RhbGwgRG9ja2VyIG9uIHlvdXIgbWFjaGluZS4gVHV0b3JpYWwgaXMgbGlua2VkIGJlbG93Ojxicj4KICAgWyFbRG9ja2VyIFR1dG9yaWFsXShodHRwczovL2ltZy55b3V0dWJlLmNvbS92aS8yZXpOcXFhU2pxOC8wLmpwZyldKGh0dHBzOi8vd3d3LnlvdXR1YmUuY29tL3dhdGNoP3Y9MmV6TnFxYVNqcTgpPGJyPgoyLiBMb2dpbiB0byBkb2NrZXIgd2l0aCB0aGUgY29tbWFuZCBgYGBkb2NrZXIgbG9naW4gLS11c2VybmFtZSA8R2l0SHViX1VzZXJuYW1lPmBgYDxicj4KICAgWW91IHdpbGwgYmUgYXNrZWQgZm9yIHlvdXIgcGFzc3dvcmQsIHdoaWNoIGlzIHlvdXIgR2l0SHViIFRva2VuLiBNYWtlIHN1cmUgeW91ciBHaXRIdWIgVG9rZW4gaGFzIHBlcm1pc3Npb25zIHRvIGFjY2VzcyBHaXRIdWIncyBDb250YWluZXIgUmVnaXN0cnkhPGJyPgogICBUaGUgbmVlZGVkIHNjb3BlIGlzIGBgYHJlYWQ6cGFja2FnZXNgYGA8YnI+CiAgIFRoaXMgY29tbWFuZCBzaG91bGQgYmUgcnVuIGluIHRoZSB0ZXJtaW5hbCBpbiB0aGUgcm9vdCBmb2xkZXIgb2YgdGhpcyBwcm9qZWN0Ljxicj4KMy4gRm9sbG93IHRoaXMgdHV0b3JpYWwgaGVyZSBmb3IgZW5zdXJpbmcgZG9ja2VyIGltYWdlcyBhcmUgdXAgdG8gZGF0ZTogW0RvY2tlciBpbWFnZSB0dXRvcmlhbF0oaHR0cHM6Ly9waG9lbml4bmFwLmNvbS9rYi91cGRhdGUtZG9ja2VyLWltYWdlLWNvbnRhaW5lcik8YnI+CjQuIFRoZSBNb25nbyBpbWFnZSBjYW4gYmUgcnVuIHdpdGggdGhpcyBjb21tYW5kIGluIHRoZSByb290IGZvbGRlcjo8YnI+CiAgIE5vdGUgdXNlIGBgYHN1ZG9gYGAgb24gTGludXgvTWFjPGJyPgpgYGBjb25zb2xlCm5wbSBydW4gZG9ja2VyCmBgYAo1LiBDcmVhdGUgLmVudiBmaWxlIHdpdGggZW52aXJvbm1lbnQgdmFyaWFibGVzCjYuIFJ1biBucG0gaQo3LiBUaGUgYXBwIGNhbiB0aGVuIGJlIHJ1biB3aXRoIHRoZSBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHN0YXJ0CmBgYAo4LiBJbnRlZ3JhdGlvbiB0ZXN0cyBjYW4gYmUgcnVuIHdoZW4gdGhlIGFwcCBpcyBydW5uaW5nIHdpdGggdGhpcyBjb21tYW5kOjxicj4KYGBgY29uc29sZQpucG0gcnVuIHRlc3Q6aW50ZWdyYXRpb24KYGBgCgojIEVuZHBvaW50IERvY3VtZW50YXRpb24KCiMjIyBHRVQgbmV3R2FtZSBlbmRwb2ludDo8YnI+CkNyZWF0ZXMgYSBuZXdHYW1lIGZvciB0aGUgdXNlci48YnI+CkV4YW1wbGU6PGJyPgptZXRob2Q6IEdFVDxicj4KdXJsOiBgYGBodHRwOi8vbG9jYWxob3N0OjI5MDEvYXBpL3YxL25ld0dhbWU/ZGlmZmljdWx0eT0xYGBgIDxicj4KSGVhZGVyOiBgYGBBdXRob3JpemF0aW9uOiAiQmVhcmVyICIgKyBhY2Nlc3NUb2tlbmBgYCA8YnI+CgpSZXR1cm5zOiA8YnI+CmBgYGpzb24KWwogICAgewogICAgICAgICJ1c2VySUQiOiAiYXV0aDB8NjM5NDQwYzhmMjkwNjc3NTA3OWM3MjU0IiwKICAgICAgICAicHV6emxlIjogIjMxMDA4NDAwMjIwMDE1MDAwNjU3MDAwMzAxMDQyMzcwODA5NTc2MDAzMDAwMDAwOTU2MjAzMDA1MDAwNjA3MDAwNzAwMDkwMDAwMDAwMTUwMCIsCiAgICAgICAgImN1cnJlbnRUaW1lIjogMCwKICAgICAgICAibnVtSGludHNBc2tlZEZvciI6IDAsCiAgICAgICAgIm51bVdyb25nQ2VsbHNQbGF5ZWQiOiAwLAogICAgICAgICJfaWQiOiAiNjQxMTA5YTE5YTE1NWI1NzEyNmZlNjQ3IiwKICAgICAgICAibW92ZXMiOiBbXSwKICAgICAgICAiX192IjogMAogICAgfQpdCmBgYApwdXp6bGUgY2FuIGJlIGFjY2Vzc2VkIGJ5IHRoZSBmb2xsb3dpbmc6PGJyPgpgYGByZXNwb25zZVswXS5wdXp6bGVgYGAgPGJyPgoKCgo= readmeEtag: '"7f4fe4972ec8b3b817658530d4042b28f9675f75"' readmeLastModified: Sun, 12 May 2024 16:01:49 GMT repositoryId: 599639630 description: null created: '2023-02-09T15:12:56Z' updated: '2024-05-12T16:02:32Z' language: TypeScript archived: true stars: 0 watchers: 0 forks: 0 owner: Sudokuru logo: https://avatars.githubusercontent.com/u/114212382?v=4 license: GPL-3.0 repoEtag: '"ef37893e5c6baccde0157be77a9107a11d68d4f08c720c356f018ce536a010b3"' repoLastModified: Sun, 12 May 2024 16:02:32 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/sudokuru/bff - source: openapi3 tags repository: https://github.com/av1m/uberoo-openapi v3: true id: eee1268d6e1d07582ae9e9b773d757b4 repositoryMetadata: base64Readme: >- IyBVYmVyb28gT3BlbkFQSQoKWyFbRGVwbG95IFViZXJvb0FQSSB0byBHaXRIdWIgUGFnZXNdKGh0dHBzOi8vZ2l0aHViLmNvbS9hdjFtL3ViZXJvby1vcGVuYXBpL2FjdGlvbnMvd29ya2Zsb3dzL2doLXBhZ2VzLnlhbWwvYmFkZ2Uuc3ZnKV0oaHR0cHM6Ly9naXRodWIuY29tL2F2MW0vdWJlcm9vLW9wZW5hcGkvYWN0aW9ucy93b3JrZmxvd3MvZ2gtcGFnZXMueWFtbCkKWyFbTGluayBvZiBkb2NzXShodHRwczovL2ltZy5zaGllbGRzLmlvL3N0YXRpYy92MT9sYWJlbD1kb2NzJm1lc3NhZ2U9UmVkb2NseSZjb2xvcj1ncmVlbildKGh0dHBzOi8vYXYxbS5naXRodWIuaW8vdWJlcm9vLW9wZW5hcGkvKQoKVGFibGUgb2YgQ29udGVudHM6ICAKCi0gW1ViZXJvbyBPcGVuQVBJXSgjdWJlcm9vLW9wZW5hcGkpCiAgLSBbR2V0IHN0YXJ0ZWRdKCNnZXQtc3RhcnRlZCkKICAtIFtQYWNrYWdlLmpzb25dKCNwYWNrYWdlanNvbikKICAgIC0gW0J1aWxkIGRpcmVjdG9yeV0oI2J1aWxkLWRpcmVjdG9yeSkKICAgIC0gW0NvbW1hbmRzXSgjY29tbWFuZHMpCiAgLSBbQ0kvQ0RdKCNjaWNkKQoKIyMgR2V0IHN0YXJ0ZWQKClRoZSBwcmVyZXF1aXNpdGVzIGFyZToKCi0gW25vZGUuanNdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi8pCi0gW2RvY2tlcl0oaHR0cHM6Ly93d3cuZG9ja2VyLmNvbS8pCi0gW2RvY2tlci1jb21wb3NlXShodHRwczovL2RvY3MuZG9ja2VyLmNvbS9jb21wb3NlL2luc3RhbGwvKQotIFtweXRob25dKGh0dHBzOi8vd3d3LnB5dGhvbi5vcmcvKSAob25seSBpZiB5b3UgdXNlIGBucG0gcnVuIHN3YWdnZXItcHl0aG9uYCkKCk9uY2UgeW91IGhhdmUgdGhlIHByZXJlcXVpc2l0ZXMgaW5zdGFsbGVkLCB5b3UgY2FuIHJ1biB0aGUgZm9sbG93aW5nIGNvbW1hbmRzOgoKYGBgYmFzaApucG0gcnVuIHN0YXJ0CmBgYAoKVGhpcyBjb21tYW5kIHdpbGwgZ2VuZXJhdGUgYG9wZW5hcGkuanNvbmAgZmlsZSBhbmQgc3RhcnQgdGhlIHNlcnZlciAodGhyb3VnaCBEb2NrZXIpLgpUaGUgc2VydmVyIGlzIGF2YWlsYWJsZSBhdCBbaHR0cDovL2xvY2FsaG9zdDo4MDgwXShodHRwOi8vbG9jYWxob3N0OjgwODApLgoKIyMgUGFja2FnZS5qc29uCgojIyMgQnVpbGQgZGlyZWN0b3J5CgpBbGwgdGhlIGZpbGVzIGFyZSBnZW5lcmF0ZWQgaW4gdGhlIGAuYnVpbGRgIGRpcmVjdG9yeS4KVGhlIGAuYnVpbGRgIGRpcmVjdG9yeSBpcyBjcmVhdGVkIGF1dG9tYXRpY2FsbHkgd2hlbiB5b3UgcnVuIGBucG0gcnVuIGJ1aWxkYC4KClRoZSBmaW5hbCBmaWxlIGFyZSBnZW5lcmF0ZWQgZnJvbSB0aGUgYHNyY2AgZGlyZWN0b3J5OyBUaGUgb3VwdXQgZm9ybWF0IGlzIGRldGVybWluZWQgYnkgdGhlIGBjb25maWcuZXh0ZW5zaW9uYCBvcHRpb24gaW4gdGhlIGBwYWNrYWdlLmpzb25gIGZpbGUuCkFjdHVlbGx5LCB0aGUgb3V0cHV0IGZvcm1hdCBhdmFpbGFibGUgYXJlIGBqc29uYCBhbmQgYHlhbWxgLgpUbyBjaGFuZ2UgdGhlIG91dHB1dCBmb3JtYXQsIHlvdSBjYW4gY2hhbmdlIHRoZSBgY29uZmlnLmV4dGVuc2lvbmAgb3B0aW9uIGluIHRoZSBgcGFja2FnZS5qc29uYCBmaWxlIG9yIGNoZWNrIHRoZSBuZXh0IHNlY3Rpb24uCgojIyMgQ29tbWFuZHMKClRoZSBbcGFja2FnZS5qc29uXShwYWNrYWdlLmpzb24pIGZpbGUgY29udGFpbnMgbWFueSBjb21tYW5kIGxpbmUgdG9vbHMgZm9yIHdvcmtpbmcgd2l0aCB0aGUgZG9jdW1lbnRhdGlvbi4KClRvIGdldCBhIHF1aWNrIGRldGFpbHMgb2YgdGhlIGF2YWlsYWJsZSBjb21tYW5kczoKCi0gKipgYnVpbGRgKiogLSBHZW5lcmF0ZSB0aGUgZG9jdW1lbnRhdGlvbiBmcm9tIHRoZSBgc3JjYCBkaXJlY3RvcnkgaW4gdGhlIGAuYnVpbGQgZGlyZWN0b3J5YC4gVGhlIGZpbGVzIHdpbGwgYmUgYC5qc29uYCBvciBgLnlhbWxgIGRlcGVuZGluZyBvbiB0aGUgYGNvbmZpZy5leHRlbnNpb25gIG9wdGlvbiBpbiB0aGUgYHBhY2thZ2UuanNvbmAgZmlsZS4KLSAqKmB3YXRjaGAqKiAtIFJlZ2VuZXJhdGUgdGhlIGRvY3VtZW50YXRpb24gd2hlbiB0aGUgYHNyY2AgZGlyZWN0b3J5IGNoYW5nZXMuCi0gKipgdGVzdGAqKiAtIEJ1aWxkIHRoZSBkb2N1bWVudGF0aW9uIGFuZCB0ZXN0IHRoZSBvdXB1dCBmaWxlICh0aGF0IGlzIGluIHRoZSBgLmJ1aWxkYCBkaXJlY3RvcnkpCi0gKipgY2xlYW5gKiogLSBSZW1vdmUgdGhlIGAuYnVpbGRgIGRpcmVjdG9yeQotICoqYHN3YWdnZXItZG9ja2VyYCoqICAtIEJ1aWxkcywgKHJlKWNyZWF0ZXMsIHN0YXJ0cywgYW5kIGF0dGFjaGVzIGNvbnRhaW5lcnMuIFRoZXJlIGlzIHR3byBjb250YWluZXJzIGluIHRoZSBEb2NrZXJmaWxlOiBgc3dhZ2dlci11aWAgYW5kIGBzd2FnZ2VyLWFwaWAuIFRoZSBgc3dhZ2dlci11aWAgY29udGFpbmVyIGlzIHVzZWQgdG8gZGlzcGxheSB0aGUgZG9jdW1lbnRhdGlvbiBhbmQgdGhlIGBzd2FnZ2VyLWFwaWAgY29udGFpbmVyIGlzIHVzZWQgdG8gc2VydmUgYSBzYW1wbGUgb2YgdGhlIEFQSSAodGhyb3VnaCBbYXBpc3Byb3V0XShodHRwczovL2h1Yi5kb2NrZXIuY29tL3IvZGFuaWVsZ3RheWxvci9hcGlzcHJvdXQpKS4KLSAqKmBzd2FnZ2VyLXB5dGhvbmAqKiAgLSBDcmVhdGUgYSBzZXJ2ZXIgd2l0aCB0aGUgU3dhZ2dlciBBUEkgaW4gUHl0aG9uICh1c2UgW2Nvbm5leGlvbl0oaHR0cHM6Ly9weXBpLm9yZy9wcm9qZWN0L2Nvbm5leGlvbi8pIG1vZHVsZSkuCi0gKipgcmVkb2MtcHJldmlld2AqKiAgLSBSdW4gYSBzZXJ2ZXIgd2l0aCB0aGUgUmVkb2MgQVBJIGluIHRoZSBicm93c2VyLiBUaGUgc2VydmVyIGlzIGF2YWlsYWJsZSBhdCBbaHR0cDovL2xvY2FsaG9zdDo1MDAwXShodHRwOi8vbG9jYWxob3N0OjUwMDApLgotICoqYHJlZG9jLWh0bWxgKiogIC0gR2VuZXJhdGUgdGhlIGRvY3VtZW50YXRpb24gZnJvbSB0aGUgYHNyY2AgZGlyZWN0b3J5IGluIHRoZSBgLmJ1aWxkIGRpcmVjdG9yeWAgaW4gdGhlIGBodG1sYCBmb3JtYXQgd2l0aCBSZWRvY2x5LiBUaGUgb3V0cHV0IGZpbGUgaXMgW2AuYnVpbGQvaW5kZXguaHRtbGBdKC5idWlsZC9pbmRleC5odG1sKS4KLSAqKmB1c2UteWFtbGAqKiAgLSBDaGFuZ2UgdGhlIG91dHB1dCBmb3JtYXQgdG8gYHlhbWxgLiBUaGUgZmlsZSB3aWxsIGJlWy5idWlsZC9vcGVuYXBpLnlhbWxdKC5idWlsZC9vcGVuYXBpLnlhbWwpLgotICoqYHVzZS1qc29uYCoqICAtIENoYW5nZSB0aGUgb3V0cHV0IGZvcm1hdCB0byBganNvbmAuIFRoZSBmaWxlIHdpbGwgYmUgW2AuYnVpbGQvb3BlbmFwaS5qc29uYF0oLmJ1aWxkL29wZW5hcGkuanNvbikuCi0gKipgc3RhcnRgKiogLSBDaGFuZ2UgdGhlIG91dHB1dCBmb3JtYXQgdG8gYGpzb25gLCBydW4gdGhlIGBzd2FnZ2VyLWRvY2tlcmAgY29tbWFuZCBhbmQgcnVuIHRoZSB3YXRjaGVyICh0aHJvdWdoIGBucG0gcnVuIHdhdGNoYCkuCi0gKipgc3RvcGAqKiAtIFN0b3BzIGNvbnRhaW5lcnMgYW5kIHJlbW92ZXMgY29udGFpbmVycwoKIyMgQ0kvQ0QKCkFzIG1lbnRpb25uZWQgYWJvdmUsIHlvdSBjYW4gdGVzdCB0aGUgZG9jdW1lbnRhdGlvbiB3aXRoIHRoZSBgbnBtIHRlc3RgIGNvbW1hbmQuCgpPbmNlIHlvdSBwdXNoIHlvdXIgY29kZSB0byB0aGUgcmVwb3NpdG9yeSwgYSBHaXRodWIgQWN0aW9ucyBpcyB0cmlnZ2VyZWQuCgpGaXJzdCwgdGhlIGNvZGUgaXMgdGVzdGVkIGJlZm9yZSBjcmVhdGluZyB0aGUgZG9jdW1lbnRhdGlvbi4KClRoZW4sIHRoZSBBY3Rpb24gd2lsbCBidWlsZCB0aGUgZG9jdW1lbnRhdGlvbiBhbmQgcHVzaCBpdCB0byB0aGUgcmVwb3NpdG9yeSAoW2doLXBhZ2VzXShodHRwczovL2dpdGh1Yi5jb20vYXYxbS91YmVyb28tb3BlbmFwaS90cmVlL2doLXBhZ2VzKSBicmFuY2hlcykuClNvLCB0aGUgZG9jdW1lbnRhdGlvbiB3aWxsIGJlIGF2YWlsYWJsZSBhdCBbYXYxbS5naXRodWIuaW8vdWJlcm9vLW9wZW5hcGldKGh0dHBzOi8vYXYxbS5naXRodWIuaW8vdWJlcm9vLW9wZW5hcGkvKS4KCkZpbmFsbHksIHRoZSBhY3Rpb24gd2lsbCBwdXNoIGFuIGFydGlmYWN0IChvcGVuYXBpLnlhbWwpIHRvIHRoZSByZXBvc2l0b3J5Lgo= readmeEtag: '"63309943e70fdcf86d8cc290b141da95916beed3"' readmeLastModified: Sun, 26 Jun 2022 21:25:16 GMT repositoryId: 500824612 description: This repository contains OpenAPI specifications for the Uberoo API created: '2022-06-07T12:09:18Z' updated: '2023-01-23T10:04:29Z' language: JavaScript archived: true stars: 0 watchers: 1 forks: 1 owner: av1m logo: https://avatars.githubusercontent.com/u/36456709?v=4 repoEtag: '"27ae1e6ec115c52e986afe728988c632510ceeac29e5e06f33281c57625f45ac"' repoLastModified: Mon, 23 Jan 2023 10:04:29 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/rosario-fiorella/http-dispatcher-framework v3: true id: 9597cb2ff22d9709642a2c56942cad72 repositoryMetadata: base64Readme: >- IyBIVFRQIERpc3BhdGNoZXIgRnJhbWV3b3JrCgojIyBBYm91dAo+ICoqIUlNUE9SVEFOVDoqKiB0aGlzIGlzIGFuIGV4cGVyaW1lbnRhbCB2ZXJzaW9uIG5vdCB0byBiZSB1c2VkIGZvciBwcm9kdWN0aW9uIGRldmVsb3BtZW50CgpUaGlzIGh0dHAtYmFzZWQgZnJhbWV3b3JrIGltcGxlbWVudHMgdGhlIFtEaXNwYXRjaGVyIHBhdHRlcm4gKG1vcmUgaW5mbyBmcm9tIE9yYWNsZSBKYXZhKV0oaHR0cHM6Ly93d3cub3JhY2xlLmNvbS9qYXZhL3RlY2hub2xvZ2llcy9mcm9udC1jb250cm9sbGVyLmh0bWwpIGFuZCBjYW4gYmUgY29uZmlndXJlZCBhcyBhIGZyb250LWNvbnRyb2xsZXIgb3IgZGVsZWdhdGUgdGhlIHJlcXVlc3QgdG8gYW4gYXBwbGljYXRpb24uIFN1aXRhYmxlIGZvciBSRVNUIEFQSSBzZXJ2aWNlcyBvciBtb25vbGl0aGljIHNlcnZlci1zaWRlIHJlbmRlcmluZy4KCiMjIEtleSBBcmNoaXRlY3R1cmFsIENvbmNlcHRzCi0gKipEaXNwYXRjaGVyIFBhdHRlcm4qKjogQ2VudHJhbGl6ZXMgSFRUUCByZXF1ZXN0IGhhbmRsaW5nLiBTZWUgYENvcmUvSHR0cC9EaXNwYXRjaGVyLnBocGAgYW5kIGV4YW1wbGUgdXNhZ2UgaW4gYDAxLWV4YW1wbGUtZGlzcGF0Y2hlci1hcy1mcm9udC1jb250cm9sbGVyL2AgYW5kIGAwMi1leGFtcGxlLWRpc3BhdGNoZXItYXMtcHJveHktdXNpbmctYXBwbGljYXRpb24vYC4KLSAqKkZyb250IENvbnRyb2xsZXIgdnMgUHJveHkqKjogVHdvIG1haW4gbW9kZXMuIEZyb250IGNvbnRyb2xsZXIgZGlyZWN0bHkgaGFuZGxlcyByZXF1ZXN0czsgcHJveHkgZGVsZWdhdGVzIHRvIGFuIGFwcGxpY2F0aW9uLiBFeGFtcGxlIGRpcmVjdG9yaWVzIGlsbHVzdHJhdGUgYm90aCBwYXR0ZXJucy4KLSAqKkZpbHRlcnMgJiBJbnRlcmNlcHRvcnMqKjogQ3VzdG9tIEhUVFAgZmlsdGVycyAoYENvcmUvSHR0cC9GaWx0ZXIucGhwYCkgYW5kIGludGVyY2VwdG9ycyAoYENvcmUvSHR0cC9JbnRlcmNlcHRvci5waHBgKSBjYW4gYmUgcmVnaXN0ZXJlZCBmb3IgcmVxdWVzdC9yZXNwb25zZSBtYW5pcHVsYXRpb24uCi0gKipOZWdvdGlhdGlvbioqOiBSZXF1ZXN0IGNvbnRlbnQgdmFsaWRhdGlvbiBpcyBoYW5kbGVkIHZpYSBuZWdvdGlhdGlvbiBjbGFzc2VzIChgQ29yZS9IdHRwL05lZ290aWF0aW9uLnBocGApLgotICoqTG9jYWxpemF0aW9uKio6IFVzZXMgUE9FRElUIGFuZCBgLnBvLy5tb2AgZmlsZXMgdW5kZXIgYGxvY2FsZS9gLiBSZXF1aXJlcyBQSFAgZXh0ZW5zaW9uczogYGdldHRleHRgLCBgaW50bGAsIGBtYnN0cmluZ2AuCi0gKipEZXBlbmRlbmN5IEluamVjdGlvbiAmIFNlcnZpY2UgTG9jYXRvcioqOiBESSBpcyB1c2VkIGZvciBjb25maWd1cmF0aW9uLiBTZWUgYENvcmUvVXRpbHMvT2JqZWN0U3RvcmFnZS5waHBgIGZvciBzZXJ2aWNlIGxvY2F0b3IsIGFuZCBgQ29yZS9Cb290L1JlZ2lzdHJ5LnBocGAgZm9yIHNpbmdsZXRvbiB1c2FnZS4KCiMjIERldmVsb3BlciBXb3JrZmxvd3MKLSAqKk5vIGJ1aWxkIHN0ZXAgcmVxdWlyZWQqKjsgUEhQID49IDguMiBpcyBtYW5kYXRvcnkuCi0gKipFbmFibGUgcmVxdWlyZWQgUEhQIGV4dGVuc2lvbnMqKjogYGZpbHRlcmAsIGBnZXR0ZXh0YCwgYGljb252YCwgYGludGxgLCBganNvbmAsIGBtYnN0cmluZ2AsIGByZWZsZWN0aW9uYCwgYHNwbGAuCi0gKipEZWJ1Z2dpbmcqKjogRGVidWdnaW5nIGlzIGVuYWJsZWQgZm9yIGxvY2FsIGRldmVsb3BtZW50IG9ubHkuIFNlbnNpdGl2ZSBpbmZvIHNob3VsZCBiZSBoaWRkZW4gaW4gSFRUUCByZXF1ZXN0cy9yZXNwb25zZXMuCi0gKipTZWN1cml0eSoqOiBTZXQgcmVjb21tZW5kZWQgSFRUUCBoZWFkZXJzIHVzaW5nIGAuaHRhY2Nlc3NgIChBcGFjaGUpLCBgZGVmYXVsdC5jb25maWdgIChOZ2lueCksIG9yIHZpYSBgQ29yZS9IdHRwL1Jlc3BvbnNlOjpzZXRIZWFkZXJgIGluIGNvbnRyb2xsZXJzLgoKIyMgUHJvamVjdC1TcGVjaWZpYyBDb252ZW50aW9ucwotICoqU2luZ2xldG9ucyoqOiBPbmx5IGBDb3JlL0Jvb3QvUmVnaXN0cnkucGhwYCB1c2VzIHNpbmdsZXRvbjsgYXZvaWQgZWxzZXdoZXJlLgotICoqTG9jYWxpemF0aW9uKio6IEFkZCBuZXcgbGFuZ3VhZ2VzIGJ5IGNyZWF0aW5nIGAucG8vLm1vYCBmaWxlcyBpbiBgbG9jYWxlL2AgYW5kIHVwZGF0aW5nIGNvbmZpZ3VyYXRpb24uCi0gKipDb250cm9sbGVycyAmIFZpZXdzKio6IE9yZ2FuaXplIHVuZGVyIGBDb250cm9sbGVycy9gIGFuZCBgVmlld3MvYCBpbiBleGFtcGxlIGRpcmVjdG9yaWVzLiBGb2xsb3cgbmFtaW5nIGNvbnZlbnRpb25zIGFzIHNob3duLgotICoqQ29uZmlndXJhdGlvbioqOiBVc2UgREkgYW5kIGNvbmZpZ3VyYXRpb24gY2xhc3NlcyB1bmRlciBgQ29uZmlnL2AgYW5kIGBCb290L2AuCgojIyBJbnRlZ3JhdGlvbiBQb2ludHMKLSAqKkV4dGVybmFsIGRlcGVuZGVuY2llcyoqOiBObyBjb21wb3NlciBvciBwYWNrYWdlIG1hbmFnZXI7IGFsbCBkZXBlbmRlbmNpZXMgYXJlIFBIUCBleHRlbnNpb25zLgotICoqQ3Jvc3MtY29tcG9uZW50IGNvbW11bmljYXRpb24qKjogVXNlIERJIGFuZCBzZXJ2aWNlIGxvY2F0b3IgZm9yIHNoYXJpbmcgc3RhdGUgYW5kIHNlcnZpY2VzLgoKIyMgUmVmZXJlbmNlcwotIERpc3BhdGNoZXI6IGBDb3JlL0h0dHAvRGlzcGF0Y2hlci5waHBgCi0gU2VydmljZSBMb2NhdG9yOiBgQ29yZS9VdGlscy9PYmplY3RTdG9yYWdlLnBocGAKLSBTaW5nbGV0b246IGBDb3JlL0Jvb3QvUmVnaXN0cnkucGhwYAotIEV4YW1wbGUgdXNhZ2U6IGAwMS1leGFtcGxlLWRpc3BhdGNoZXItYXMtZnJvbnQtY29udHJvbGxlci9gLCBgMDItZXhhbXBsZS1kaXNwYXRjaGVyLWFzLXByb3h5LXVzaW5nLWFwcGxpY2F0aW9uL2AKLSBMb2NhbGl6YXRpb246IGBsb2NhbGUvYAoKLS0tCgojIyMgRGlzcGF0Y2hlciBsaWZlY3ljbGUgbWFuYWdlbWVudCAKIVt3b3JrZmxvd10oaHR0cHM6Ly9naXRodWIuY29tL3Jvc2FyaW8tZmlvcmVsbGEvaHR0cC1kaXNwYXRjaGVyLWZyYW1ld29yay9hc3NldHMvNDE3MjgwNTkvOGI3MjcxNWMtNWJjNi00MWVkLWI3YzgtODYxYmIyMjEwMDIxKQo= readmeEtag: '"ee5307b872b15304673f34f2b61c7f8fbbe83fa0"' readmeLastModified: Thu, 16 Oct 2025 20:25:05 GMT repositoryId: 591982605 description: >- The http-based PHP framework implements the Dispatcher pattern and can be configured as a front-controller or delegate the request to an application. Suitable for REST API services or monolithic server-side rendering. created: '2023-01-22T15:08:59Z' updated: '2025-10-22T21:34:07Z' language: PHP archived: false stars: 0 watchers: 1 forks: 0 owner: rosario-fiorella logo: https://avatars.githubusercontent.com/u/41728059?v=4 license: MIT repoEtag: '"3ba1164c84fb6d4471a5dfdecd9313b7ec2c4056c34618817fbd06b42096eb07"' repoLastModified: Wed, 22 Oct 2025 21:34:07 GMT category: Server Implementations foundInMaster: true oldLocations: - https://github.com/rosario-fiorella/micro-framework-http - source: https://openapi.tools/ name: BOATS category: - DSL - SDK link: https://www.npmjs.com/package/boats repository: https://github.com/j-d-carmichael/boats language: Node.js source_description: > BOATS allows for larger teams to contribute to multi-file OpenAPI definitions by writing Nunjucks tpl syntax in yaml with a few important helpers to ensure stricter consistency, eg operationId: <$ uniqueOpId() $>. v2: true v3: true repositoryMetadata: base64Readme: >- IyBCT0FUUwoKIVtCb2F0c10oYm9hdHMuanBnKQoKQW4gT3BlbkFQSSAmIEFzeW5jQVBJIHRlbXBsYXRpbmcgc3lzdGVtIHdpdGggTnVuanVja3MuLi4gd3JpdGUgbGVzcyBZQU1MLi4uIGRvIG1vcmUuCl9fXwoKIyMgV2hhdCBpcyBpdD8KCk9wZW5BUEkgYW5kIEFzeW5jQVBJIGFyZSBncmVhdCwgd3JpdGluZyB5YW1sIGlzIGZhc3QuLi4gaG93ZXZlciwgdGhlcmUgaXMgYSBsb3Qgb2YgY29weS9wYXN0ZSByZXF1aXJlZC4gQWRkaXRpb25hbGx5LCBtYW5hZ2luZyBtYW55IHJvdXRlcyBpbiBhIHNpbmdsZSBmaWxlIGlzIHBhaW5mdWwuIEJPQVRTIGFsbG93cyB5b3UgdG8gcmVkdWNlIHRoZSBjb3B5IGFuZCBwYXN0aW5nIHdpdGggbWFueSBidWlsdCBpbiBoZWxwZXJzIHdoaWxlIGF0IHRoZSBzYW1lIHRpbWUgYnJlYWtpbmcgdGhlIDEgbGFyZ2VyIGZpbGUgZG93biBpbnRvIG1hbnkgc21hbGwgZmlsZXMuIFRoZSBvdXRwdXQgZnJvbSBCT0FUUyBpcyBhbHNvIHZhbGlkYXRlZCB2aWEgW0BhcGlkZXZ0b29sc10oaHR0cHM6Ly9naXRodWIuY29tL0FQSURldlRvb2xzKSBvciBbQGFzeW5jYXBpXShodHRwczovL2dpdGh1Yi5jb20vYXN5bmNhcGkpLgoKIyMgUXVpY2sgc3RhcnQKMS4gSW5pdGlhbGl6ZSBhIG5ldyBwcm9qZWN0OiBucG0gaW5pdCAteQoyLiBTZXQgdXAgQk9BVFM6IG5weCBib2F0cyAtLWluaXQgKGZvbGxvdyB0aGUgcHJvbXB0cykKMy4gQnVpbGQgdGhlIHByb2plY3Q6IG5wbSBydW4gYnVpbGQgKG91dHB1dHMgaW4gLi9idWlsZCkKCiMjIERvY3MgJiBDaGFuZ2Vsb2cKW0Z1bGwgZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9qLWQtY2FybWljaGFlbC5naXRodWIuaW8vYm9hdHMpCgpbQ2hhbmdlbG9nXShodHRwczovL2otZC1jYXJtaWNoYWVsLmdpdGh1Yi5pby9ib2F0cy8jLz9pZD1jaGFuZ2Vsb2cpCgojIyBCT0FUUyBDTEkKCldyaXRpbmcgeWFtbCBmaWxlcyBmb3IgQk9BVFMgaXMgZWFzaWVyIHRoYW4gbWFuYWdpbmcgYSBzaW5nbGUgZmlsZSwgYnV0IHRvIG1ha2Ugd3JpdGluZyBCT0FUUyB5bWwgZmlsZXMgZXZlbiBlYXNpZXIuLi4gdGhlcmUgaXMgbm93IGEgQk9BVFMgQ0xJIHRvb2w6CgpodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9AYWNyb250dW0vYm9hdHMtY2xpCgojIyBFeGFtcGxlcwpTaW1wbGUgZXhhbXBsZXMgY2FuIGJlIGZvdW5kIGhlcmU6Ci0gW0FzeW5jIEFQSV0oaHR0cHM6Ly9naXRodWIuY29tL2otZC1jYXJtaWNoYWVsL2JvYXRzL3RyZWUvbWFpbi9pbml0LWZpbGVzL2FzeW5jYXBpKQotIFtPcGVuIEFQSSAyXShodHRwczovL2dpdGh1Yi5jb20vai1kLWNhcm1pY2hhZWwvYm9hdHMvdHJlZS9tYWluL2luaXQtZmlsZXMvb2EyKQotIFtPcGVuIEFQSSAzXShodHRwczovL2dpdGh1Yi5jb20vai1kLWNhcm1pY2hhZWwvYm9hdHMvdHJlZS9tYXN0ZXIvc3JjT0EzKQoKKFJlZmVyIHRvIHRoZSBbZG9jdW1lbnRhdGlvbl0oaHR0cHM6Ly9qLWQtY2FybWljaGFlbC5naXRodWIuaW8vYm9hdHMpIGZvciBhZGRpdGlvbmFsIGZlYXR1cmVzIGFuZCBkZXRhaWxzLikKCiMjIFRoYW5rcyBUbwpCT0FUUyBpcyBub3RoaW5nIHdpdGhvdXQgdGhlIHN1cHBvcnQgb2Y6Ci0gRXZlcnkgW2NvbnRyaWJ1dG9yXShodHRwczovL2dpdGh1Yi5jb20vai1kLWNhcm1pY2hhZWwvYm9hdHMvZ3JhcGhzL2NvbnRyaWJ1dG9ycykgJiBbaXNzdWVdKGh0dHBzOi8vZ2l0aHViLmNvbS9qLWQtY2FybWljaGFlbC9ib2F0cy9pc3N1ZXMpIQotIFtAYXBpZGV2dG9vbHNdKGh0dHBzOi8vZ2l0aHViLmNvbS9BUElEZXZUb29scykKLSBbQGFzeW5jYXBpXShodHRwczovL2dpdGh1Yi5jb20vYXN5bmNhcGkpCi0gW2pzLXlhbWxdKGh0dHBzOi8vZ2l0aHViLmNvbS9ub2RlY2EvanMteWFtbCkKLSBNb3ppbGxhIFtOdWp1Y2tzXShodHRwczovL2dpdGh1Yi5jb20vbW96aWxsYS9udW5qdWNrcykKLSBKZXRicmFpbnMgW09wZW4gU291cmNlIERldmVsb3BtZW50IC0gQ29tbXVuaXR5IFN1cHBvcnRdKGh0dHBzOi8vd3d3LmpldGJyYWlucy5jb20vY29tbXVuaXR5L29wZW5zb3VyY2UvI3N1cHBvcnQpIQoKClshW0pldGJyYWluc10oaHR0cHM6Ly9yZXNvdXJjZXMuamV0YnJhaW5zLmNvbS9zdG9yYWdlL3Byb2R1Y3RzL2NvbXBhbnkvYnJhbmQvbG9nb3MvamJfYmVhbS5zdmcpXShodHRwczovL3d3dy5qZXRicmFpbnMuY29tL2NvbW11bml0eS9vcGVuc291cmNlLyNzdXBwb3J0KSBbIVtHaXRIdWJdKGh0dHBzOi8vZ2l0aHViLmdpdGh1YmFzc2V0cy5jb20vaW1hZ2VzL21vZHVsZXMvZGFzaGJvYXJkL29uYm9hcmRpbmcvZ2gtZGVza3RvcC5wbmcpXShodHRwczovL2dpdGh1Yi5jb20vKSBbIVthcGlkZXZ0b29sc10oaHR0cHM6Ly9hdmF0YXJzLmdpdGh1YnVzZXJjb250ZW50LmNvbS91LzQzNzUwMDc0P3M9MjAwJnY9NCldKGh0dHBzOi8vZ2l0aHViLmNvbS9BUElEZXZUb29scykgWyFbYXN5bmNhcGldKGh0dHBzOi8vYXZhdGFycy5naXRodWJ1c2VyY29udGVudC5jb20vdS8xNjQwMTMzND9zPTIwMCZ2PTQpXShodHRwczovL2dpdGh1Yi5jb20vYXN5bmNhcGkpIFshW051anVja3NdKGh0dHBzOi8vYXZhdGFycy5naXRodWJ1c2VyY29udGVudC5jb20vdS8xMzE1MjQ/cz0yMDAmdj00KV0oaHR0cHM6Ly9naXRodWIuY29tL21vemlsbGEvbnVuanVja3MpCgo= readmeEtag: '"16162014ee30fce0badddab4f3e8e74a914d4515"' readmeLastModified: Thu, 08 Aug 2024 17:13:04 GMT repositoryId: 179890990 description: Beautiful Open Api Template System created: '2019-04-06T21:56:58Z' updated: '2025-12-06T16:42:36Z' language: TypeScript archived: false stars: 57 watchers: 2 forks: 8 owner: j-d-carmichael logo: https://avatars.githubusercontent.com/u/49351986?v=4 license: MIT repoEtag: '"c0e496d67387e0ccc54222bfd711ac70a1820b7c312c524ff8e7e111eedf584e"' repoLastModified: Sat, 06 Dec 2025 16:42:36 GMT foundInMaster: true id: 8049f9d9f009ed969457a47b97bb22a5 oldLocations: - https://github.com/johndcarmichael/boats - source: https://openapi.tools/ name: APIClarity category: - Learning - Server Implementations language: - Golang - JavaScript link: https://apiclarity.io/ repository: https://github.com/openclarity/apiclarity source_description: >- Reconstruct Open API Specifications from real-time workload traffic seamlessly. v2: true v3: true v3_link: https://github.com/apiclarity/apiclarity/issues/39 repositoryMetadata: base64Readme: >- # APIClarity

![APIClarity](API_clarity.svg "APIClarity")

APIClarity is a modular tool that addresses several aspects of API Security, focusing specifically on [OpenAPI](https://spec.openapis.org/oas/latest.html) based APIs.

APIClarity approaches API Security in 2 different ways:
- Captures all API traffic in a given environment and performs a set of security analysis to discover all potential security problems with detected APIs
- Actively tests API endpoints to detect security issues in the implementation of such APIs.

## OpenAPI automatic reconstruction
Both approaches described above are way more effective when APIClarity is primed with the OpenAPI specifications of the APIs analyzed or tested. However, not all applications have an OpenAPI specification available. For this reason one of the main functionality of APIClarity is the automatic reconstruction of OpenAPI specifications based on observed API traffic. In this case, users have the ability to review and approve the reconstructed specifications.

## Security Modules
APIClarity is structured in a modular architecture, which allows to easily add new functionalities.

In the following a brief description of the modules currently implemented:

- **Spec Diffs** This module compares the API traces with the OAPI specifications provided by the user or previously reconstructed. The result of this comparison provides:
    - List of API endpoints that are observed but not documented in the specs, i.e. _Shadow APIs_;
    - List of API endpoints that are observed but marked as deprecated in the specs, i.e. _Zombie APIs_;
    - List of difference between of the APIs observed and their documented specification.
- [**Trace Analyzer**](./backend/pkg/modules/internal/traceanalyzer/README.md) This module analyzes path, headers and body of API requests and responses to discover potential security issues, such as weak authentications, exposure of sensitive information, potential Broken Object Level Authorizations (BOLA) etc.
- [**BFLA Detector**](./backend/pkg/modules/internal/bfla/README.md) This module detects potential Broken Function Level Authorization. In particular it observes the API interactions and build an authorization model that captures what clients are supposed to be authorized to make the various API calls. Based on such authorization model it then signals violations which may represent potential issues in the API authorization procedures.
- **Fuzzer** This module actively tests API endpoints based on their specification attempting in discovering security issues in the API server implementation.

## High level architecture

![High level architecture](diagram.png "High level architecture")

## Getting started

### Supported traffic source integrations

APIClarity supports integrating with the following traffic sources. Install APIClarity and follow the instructions per required integration.

* Istio Service Mesh
  * Make sure that Istio 1.10+ is installed and running in your cluster.
  See the [Official installation instructions](https://istio.io/latest/docs/setup/getting-started/#install)
  for more information.

* Tap via a DaemonSet
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/taper)

* Kong API Gateway
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/gateway/kong)

* Tyk API Gateway
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/gateway/tyk)

* OpenTelemetry Collector (traces only)
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/otel-collector)

The integrations (plugins) for the supported traffic sources above are located in the [plugins directory within the codebase](https://github.com/openclarity/apiclarity/tree/master/plugins) and implement the [plugins API](https://github.com/openclarity/apiclarity/tree/master/plugins/api) to export the API events to APIClarity. To enable and configure the supported traffic sources, please check the ```trafficSource:``` section in [Helm values](https://github.com/openclarity/apiclarity/blob/master/charts/apiclarity/values.yaml).
Contributions of integrations with additional traffic sources are more than welcome!

### Install APIClarity in a K8s cluster using Helm:

1. Add Helm repo

   ```shell
   helm repo add apiclarity https://openclarity.github.io/apiclarity
   ```

2. Save APIClarity default chart values

    ```shell
    helm show values apiclarity/apiclarity > values.yaml
    ```

3. Update `values.yaml` with the required traffic source values

4. Deploy APIClarity with Helm for the selected traffic source

   ```shell
   helm install --values values.yaml --create-namespace apiclarity apiclarity/apiclarity -n apiclarity
   ```

5. Port forward to APIClarity UI:

   ```shell
   kubectl port-forward -n apiclarity svc/apiclarity-apiclarity 9999:8080
   ```

6. Open APIClarity UI in the browser: <http://localhost:9999/>
7. Generate some traffic in the traced applications and check the APIClarity UI :)

### Uninstall APIClarity from a K8s cluster using Helm:

1. Helm uninstall

   ```shell
   helm uninstall apiclarity -n apiclarity
   ```

2. Clean resources

    By default, Helm will not remove the PVCs and PVs for the StatefulSets. Run the following command to delete them all:

    ```shell
    kubectl delete pvc -l app.kubernetes.io/instance=apiclarity -n apiclarity
    ```

## Configurations

The file [values.yaml](https://github.com/openclarity/apiclarity/blob/master/charts/apiclarity/values.yaml) is used to deploy and configure APIClarity on your cluster via Helm.
[This ConfigMap](https://github.com/openclarity/apiclarity/blob/master/charts/apiclarity/templates/configmap.yaml) is used to define the list of headers to ignore when reconstructing the spec.

## Testing with a demo application

A good demo application to try APIClarity with is the [Sock Shop Demo](https://microservices-demo.github.io/).

To deploy the Sock Shop Demo, follow these steps:

1. Create the `sock-shop` namespace and enable Istio injection:

   ```shell
   kubectl create namespace sock-shop
   kubectl label namespaces sock-shop istio-injection=enabled
   ```

2. Deploy the Sock Shop Demo to your cluster:

   ```shell
   kubectl apply -f https://raw.githubusercontent.com/microservices-demo/microservices-demo/master/deploy/kubernetes/complete-demo.yaml
   ```

3. Deploy APIClarity in the `sock-shop` namespace (e.g. Istio service-mesh traffic source):

   ```shell
   helm repo add apiclarity https://openclarity.github.io/apiclarity
   ```

   ```shell
   helm install --set 'trafficSource.envoyWasm.enabled=true' --set 'trafficSource.envoyWasm.namespaces={sock-shop}' --create-namespace apiclarity apiclarity/apiclarity -n apiclarity
   ```

4. Port forward to Sock Shop's front-end service to access the Sock Shop Demo App:

   ```shell
   kubectl port-forward -n sock-shop svc/front-end 7777:80
   ```

   Open the Sock Shop Demo App UI in the browser (<http://localhost:7777/>) and run
   some transactions to generate data to review on the APIClarity dashboard.

## Building

### Building from source:

Build and push the image to your repo:

```shell
DOCKER_IMAGE=<your docker registry>/apiclarity DOCKER_TAG=<your tag> make push-docker
```

Update [values.yaml](https://github.com/openclarity/apiclarity/blob/master/charts/apiclarity/values.yaml) accordingly.

## Running locally with demo data

1. Build UI & backend locally as described above:

   ```shell
   make ui && make backend
   ```

2. Copy the built site:

   ```shell
   cp -r ./ui/build ./site
   ```

3. Run backend and frontend locally using demo data:

   Note: You might need to delete the old local state file and local db:

   ```shell
   rm state.gob; rm db.db
   ```

   ```shell
   DATABASE_DRIVER=LOCAL K8S_LOCAL=true FAKE_TRACES=true FAKE_TRACES_PATH=./backend/pkg/test/trace_files \
   ENABLE_DB_INFO_LOGS=true ./backend/bin/backend run
   ```
   Note: this command requires a proper KUBECONFIG in your environment when __K8S_LOCAL=true__ is used. If you want to run without k8s, use __ENABLE_K8S=false__ instead.

4. Open APIClarity UI in the browser: <http://localhost:8080/>

## Enabling External Trace Sources Support
Enabling external trace sources support, APIClarity can receive the trace sources from the entitites that are external to the K8s cluster. External trace sources such as Gateways, Load balancers, etc. can communicate with APIClarity to report APIs and send the traces. 

The following section describes how to deploy APIClarity with the support for external trace sources
1.	Add Helm Repo
```shell
helm repo add apiclarity https://openclarity.github.io/apiclarity
```

2.	Update values.yaml with 
```shell
Apiclarity -> tls -> enabled as true
supportExternalTraceSource -> enabled as true
```

3.	Deploy APIClarity with the Helm enabling external traffic sources
```shell
helm install --values values.yaml --create-namespace apiclarity apiclarity/apiclarity -n apiclarity
```

4.	Port forward to APIClarity UI:
```shell
kubectl port-forward -n apiclarity svc/apiclarity-apiclarity 9999:8080
```

5.	Open APIClarity UI in the browser:
```shell
http://localhost:9999 
```

The following section describes how to register a new external trace source. And this section includes how to access the service, register a new trace source, and how to receive the token and certificate.
1.	Port forward for service at 8443
```shell
kubectl port-forward -n apiclarity svc/apiclarity-apiclarity 8443:8443
```

2.	Register a new external trace source and receive the token 
```shell
TRACE_SOURCE_TOKEN=$(curl --http1.1 --insecure -s -H 'Content-Type: application/json' -d '{"name":"apigee_gateway","type":"APIGEE_X"}' https://localhost:8443/api/control/traceSources|jq -r '.token')
```

3. Receive the certificate
To receive the certificate, get the External-IP for the service named as apiclarity-external
```shell
kubectl get services -n apiclarity
```
Then, use the External-IP address with the following command, then extract the certificate with -----BEGIN CERTIFICATE----- and -----END CERTIFICATE----- and save to server.crt
```shell
openssl s_client -showcerts -connect <External-IP>:10443
```

Use the above extracted token at the step-2 and certificate at step-3 for configuring subsequent external trace sources such as Apigee X Gateway and BIG-IP LTM Load balancer

## Supported Trace Sources
APIClarity can support with the following trace sources and follow the instructions per required integration.

* Apigee X Gateway
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/gateway/apigeex)
* BIG-IP LTM Load balancer
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/gateway/f5-bigip)
* Kong
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/gateway/kong)
* Tyk
  * [Integration instructions](https://github.com/openclarity/apiclarity/tree/master/plugins/gateway/tyk)

## Contributing

Pull requests and bug reports are welcome.

For larger changes please create an Issue in GitHub first to discuss your
proposed changes and possible implications.

## Contributors

https://panoptica.app

## License

[Apache License, Version 2.0](https://www.apache.org/licenses/LICENSE-2.0)
 readmeEtag: '"e8e7d2fe52e0a9d51ecf4a445a3c52482e9f6883"' readmeLastModified: Wed, 03 Jul 2024 09:43:36 GMT repositoryId: 402435051 description: "An API security tool to capture and analyze API traffic, test API endpoints, reconstruct Open API specification, and identify API security risks.\_" created: '2021-09-02T13:41:19Z' updated: '2026-01-26T17:44:35Z' language: Go archived: false stars: 563 watchers: 19 forks: 71 owner: openclarity logo: https://avatars.githubusercontent.com/u/89657786?v=4 license: Apache-2.0 repoEtag: '"5fd625984b4c067f608fdbfa9bff7b1b91ab18bd93405068d9a46e2144b6c52f"' repoLastModified: Mon, 26 Jan 2026 17:44:35 GMT foundInMaster: true id: 68b981acc13af75d4eb8c04143885e13 oldLocations: - https://github.com/apiclarity/apiclarity - source: https://openapi.tools/ name: Unmock category: - Mock - Testing link: https://unmock.io language: Node.js repository: https://github.com/meeshkan/unmock-js source_description: >- API integration testing library that intercepts outgoing requests and serves back mock data based on the OpenAPI descriptions. v2: false v3: true repositoryMetadata: base64Readme: >- # [Unmock](https://www.unmock.io/) (JavaScript SDK)

[![CircleCI](https://circleci.com/gh/meeshkan/unmock-js.svg?style=svg)](https://circleci.com/gh/meeshkan/unmock-js)
[![codecov](https://codecov.io/gh/unmock/unmock-js/branch/dev/graph/badge.svg)](https://codecov.io/gh/unmock/unmock-js)
[![Known Vulnerabilities](https://snyk.io/test/github/unmock/unmock-js/badge.svg?targetFile=package.json)](https://snyk.io/test/github/unmock/unmock-js?targetFile=package.json)
[![Chat on Gitter](https://badges.gitter.im/gitterHQ/gitter.png)](https://gitter.im/unmock/community)

Fuzz test your REST API calls.

* [Install](#install)
* [What Does Unmock Do](#what-does-unmock-do)
* [When To Use Unmock](#when-to-use-unmock)
* [The Docs](#the-docs)
* [Usage](#usage)
  + [Specifying hostname](#specifying-hostname)
  + [Specifying the path](#specifying-the-path)
  + [Specifying request body](#specifying-request-body)
  + [Specifying request query string](#specifying-request-query-string)
  + [Specifying Request Headers](#specifying-request-headers)
  + [Specifying replies](#specifying-replies)
  + [Specifying reply headers](#specifying-reply-headers)
  + [Chaining](#chaining)
  + [Ignorable API calls](#ignorable-api-calls)
* [Expectations](#expectations)
* [Initializing Mocks](#initializing-mocks)
* [Faker API](#faker-api)
* [Runner](#runner)
* [OpenAPI](#openapi)
* [Tutorials](#tutorials)
* [Contributing](#contributing)
* [Development](#development)
* [License](#license)

## Install

```sh
$ npm install --save-dev unmock
```

## What Does Unmock Do

Unmock helps you fuzz test REST API calls. Fuzz testing, or fuzzing, is a form of testing where you verify the correctness of code by asserting that it behaves correctly with variable and unexpected input. The response of a REST API is most often variable and unexpected. So, in most cases, fuzz testing is a useful way to test integrations with APIs.

## When To Use Unmock

Below are some questions to ask when determining if Unmock is a good fit for your development process.

- Is my code base in JavaScript or TypeScript?
- Does my code base have tests written in Jest, Mocha, Jasmine, Tap or Ava?
- Do I make a network call from my codebase to a REST API?

If the answer is yes to all of these questions, Unmock could help you test code paths in your code that make REST API calls and use the responses from those API.

## The Docs

If you don't like reading READMEs, check out our [docs](https://unmock.io/docs/introduction)!

## Usage

Here is a mock defined using unmock, a function to be tested, and a test written in jest. The commented numbers are explained in the text below this code example.

```js
const fetch = require("node-fetch");
const unmock = require("unmock");
const { runner, transform, u } = unmock;
const { withCodes } = transform;

unmock
  .mock("https://zodiac.com", "zodiac")
  .get("/horoscope/{sign}") // 1
  .reply(200, {
    horoscope: u.string(), // 2
    ascendant: u.opt(u.string()) // 3
  })
  .reply(404, { message: "Not authorized" }); // 4

async function getHoroscope(sign) {
  // use unmock.fetch, request, fetch, axios or any similar library
  const result = await fetch("https://zodiac.com/horoscope/" + sign);
  const json = await result.json();
  return { ...json, seen: false };
}

let zodiac;
beforeAll(() => {
  zodiac = unmock.default.on().services.zodiac;
});
afterAll(() => unmock.default.off());

describe("getHoroscope", () => {
  it(
    "augments the API call with seen=false",
    runner(async () => { // 5
      zodiac.state(withCodes(200)); // 6
      const res = await getHoroscope();
      expect(res).toMatchObject(JSON.parse(zodiac.spy.getResponseBody())); // 7
      expect(res.seen).toBe(false);
    }),
  );
});
```

With unmock, you can (1) override a REST endpoint to provide (2) variable and (3) optional responses in addition to (4) different status codes. Then, one uses the (5) runner to automatically run a test multiple times with subtly different responses from the API. One can also (6) initialize the API to a given state and (7) make assertions about how an API was used.

### Specifying hostname

The request hostname should be a string.

```js
unmock.mock('http://www.example.com')
  .get('/resource')
  .reply(200, u.integer())
```

Unmock will then refer to the service as `example`. To specify the name of the service as `foo`, we would write.

```js
unmock.mock('http://www.example.com', 'foo')
  .get('/resource')
  .reply(200, u.string())
```

To match multiple protocols or URLs, use `associate`.

```js
unmock.default.associate('https://www2.example.com', 'foo')
```

### Specifying the path

The request path should be a string, and you can use any [HTTP verb](#http-verbs). Wild-cards in the string should be enclosed in curly braces.

```js
unmock.mock('http://www.example.com')
  .get('/resource/{id}')
  .reply(200, u.string())
```

Alternatively, you can use an array of path segments, where each segment is either a string, a string in curly-braces for an open parameter, or a key/value pair associating a name of a path parameter to a regex matching it.

```js
unmock.mock('http://www.example.com')
  .get(["users", /[0-9]+/, "status"]) // "/users/{id}/status"
  .reply(200, u.string())
```

### Specifying request body

You can specify the request body to be matched as the second argument to the `get`, `post`, `put` or `delete` specifications. The argument can be any valid JSON, [`json-schema-poet`](https://github.com/unmock/json-schema-poet) using the `u` syntax(`u.integer()`, `u.string()`) or any combination thereof.

Here is an example of a post request that will only validate if it contains a token.

```js
unmock.mock('http://www.example.com')
  .post('/login', u.type({ token: u.string()}, {expires: u.integer()}))
  .reply(200, { id: u.string() })
```

> Unmock does not support body encodings other than "application/json" at this point.

### Specifying request query string

Unmock automatically ignores query strings. However, it understands query strings if you would like to match against them.

> When query strings are included in Unmock, they will act as if they are required. Most APIs do not have required query strings, so make sure to double-check the API documentation before indicating a required query string.

These parameters can be included as part of the path:

```js
unmock.mock('http://example.com')
  .get('/users?foo=bar')
  .reply(200)
```

Instead of placing the entire URL, you can specify the query part as an object:

```js
unmock.mock('http://example.com')
  .get('/users')
  .query({ name:u.string(), surname: u.string() })
  .reply(200, { results: u.array({id: u.integer() }) })
```

### Specifying Request Headers

You can specify the request headers like this:

```js
unmock.mock('http://www.example.com', {
  reqheaders: {
    authorization: 'Basic Auth',
  },
})
  .get('/')
  .reply(200)
```

Or you can use a regular expression to check the header values.

```js
unmock.mock('http://www.example.com', {
  reqheaders: {
    'X-My-Awesome-Header': /Awesome/i,
  },
})
  .get('/')
  .reply(200)
```

Headers in unmock are always a partial match, meaning that additional headers are ignored. This means that you don't need to worry about matching against common headers like `Content-Type` and `Host`.

### Specifying replies

You can specify the return status code for a path on the first argument of reply like this:

```js
unmock.mock('http://myapp.iriscouch.com')
  .get('/users/1')
  .reply(404)
```

You can also specify the reply body as valid JSON, `json-schema-poet`, or any combination thereof.

```js
unmock.mock('http://www.google.com')
  .get('/')
  .reply(200, u.stringEnum(['Hello from Google!', 'Do no evil']))
```

If you would like to transform any part of a constant reply (ie a fixture recorded from real API traffic) into a variable version of itself, use `u.fuzz`. This command infers the type of its input and produces output following the same schema.

```js
unmock.mock('http://www.foo.com')
  .get('/')
  // produces { firstName: "a random string", lastName: "another random string" }
  .reply(200, u.fuzz({ firstName: "Bob", lastName: "Smith" }))
```

### Specifying reply headers

You can specify the reply headers like this:

```js
unmock.mock('https://api.github.com')
  .get('/repos/atom/atom/license')
  .reply(200, { license: 'MIT' }, { 'X-RateLimit-Remaining': u.integer() })
```

### Chaining

You can chain behavior like this:

```js
unmock.mock('http://myapp.iriscouch.com')
  .get('/users/1')
  .reply(404)
  .post('/users', {
    username: u.string(),
    email: u.string(),
  })
  .reply(201, {
    ok: u.boolean(),
    id: u.string(),
    rev: u.string(),
  })
  .get('/users/123ABC')
  .reply(200, {
    _id: u.string(),
    _rev: u.string(),
    user: u.string(),
    name: u.string()
  })
```

### Ignorable API calls

For ignorable API calls where you are passing through information but don't care about a response, you can instruct unmock to serve random `200` responses to those requests using `tldr`.

```js
unmock
   .mock("https://my-analytics-api.vendor.com)
  .tldr();
```

## Expectations

Unmock uses [`sinon`](https://sinonjs.org) spies to help you compose (great) expectations.

> In general, try to write expectations using spies instead of hardcoded values. Instead of `expect(res).toEqual({foo: "bar", baz: true })`, favor `expect(res).toEqual({ ...spy.getRequestBody(), baz: true })`

Assuming you have defined a service using unmock, you can use spies following the `<verb><Request|Response><Thing>` convention. For example, `getResponseBody` or `deleteRequestPath`. Nonsensical things like `getRequsetBody` are not defined.

```js
test("analytics API was called", async () => {
  await myFunction()
  expect(analyticsService.spy.postRequestBody())
    .toMatchObject({ event: "VISITED" })
})
```

The following getter functions are defined on the spy object.

- `getRequestPathname`
- `getRequestPath`
- `getRequestHeaders`
- `getRequestQuery`
- `getRequestProtocol`
- `getResponseBody`
- `getResponseCode`
- `getResponseHeaders`
- `postRequestBody`
- `postRequestPathname`
- `postRequestPath`
- `postRequestHeaders`
- `postRequestQuery`
- `postRequestProtocol`
- `postResponseBody`
- `postResponseCode`
- `postResponseHeaders`
- `putRequestBody`
- `putRequestPathname`
- `putRequestPath`
- `putRequestHeaders`
- `putRequestQuery`
- `putRequestProtocol`
- `putResponseBody`
- `putResponseCode`
- `putResponseHeaders`
- `deleteRequestPathname`
- `deleteRequestPath`
- `deleteRequestHeaders`
- `deleteRequestQuery`
- `deleteRequestProtocol`
- `deleteResponseBody`
- `deleteResponseCode`
- `deleteResponseHeaders`

In addition, spies are full-fledged sinon spies. More about their usage in Unmock can be found [here](https://www.unmock.io/docs/expectations), and more information on sinon can be found [here](https://sinonjs.org/).

## Initializing Mocks

Unmock supports the initialization of services to arbitrary states. This is helpful, for example, if you want to test how your code behaves when a given service returns *exactly* five items or when a particiular field in an object is missing or present.

To do this, set the `state` property of a service. The state property is a function that takes a request and an OpenAPI schema as input and returns an OpenAPI schema and output.  Many utility functions have been created for the most common state manipulations. For example, to test the outcome with only certain codes, use `withCodes`.

```js
const unmock = require('unmock');
const withCodes = unmock.transform.withCodes;
const github = unmock.default.on().services.github;
github.state(withCodes([200, 201, 404]));
```

Because `withCodes` returns a function, the same thing could have been written.

```js
const unmock = require('unmock');
const withCodes = unmock.transform.withCodes;
const github = unmock.default.on().services.github;
github.state((req, schema) => withCodes([200, 201, 404])(req, schema));
```

This is useful, for example, if you want to test a certain code only if a given header is present.

```js
const unmock = require('unmock');
const withCodes = unmock.transform.withCodes;
const github = unmock.default.on().services.github;
github.state((req, schema) =>
  withCodes([200, ...(req.headers.MyHeader ? [404] : [])])(req, schema));
```

The unmock [documentation](https://www.unmock.io/docs/setting-state) contains more information about initializing the state.

## Faker API

`UnmockFaker` class provides a lower-level API for working with mocks. You can create a new `UnmockFaker` via `unmock.faker()`:

```ts
const unmock, { Service, ISerializedRequest} = require("unmock");
// import unmock from "unmock";  // ES6

const faker = unmock.faker();
```

To use the faker for mocking, you need to add services. The first option is to use the `mock` method:

```ts
faker
  .mock("http://petstore.swagger.io", "petstore")
  .get("/v1/pets")
  .reply(200, { foo: u.string() });
```

Alternatively, you can create a service from OpenAPI specification with `Service.fromOpenAPI`:

```ts
const { Service } = require("unmock");

const schema: OpenAPIObject = ...; // Load OpenAPIObject
const petstoreService = Service.fromOpenAPI({ schema, name: "petstore" })

// Add service to `faker`
faker.add(petstoreService);
```

You can then also modify the state of the petstore service via `state` property:

```ts
const { transform } = require("unmock");
// Service should always return code 200
petstore.state(transform.withCodes(200));
```

Once you have added a service, you can use `faker.generate` method to create a mock for any `Request` object:

```ts
const { UnmockRequest, UnmockResponse } = require("unmock");

const req: UnmockRequest = {
  host: "petstore.swagger.io",
  protocol: "http",
  method: "get",
  path: "/v1/pets",
  pathname: "/v1/pets",
  headers: {},
  query: {},
}

const res: UnmockResponse = faker.generate(req);

// Access res.statusCode, res.headers, res.body, etc.
expect(res.statusCode).toBe(200);
```

## Runner

With the [Unmock `runner`](packages/unmock-runner), you can run any test multiple times with different potential outcomes from the API. All of your unmock tests should use the `runner` unless you are absolutely certain that the API response will be the same every time.

### Default

By default, the `runner` is set to run your test 20 times. If you want to change this value, please refer to our docs on [changing the `runner` default](https://github.com/Meeshkan/unmock-js/blob/unmock-runner-docs/packages/unmock-runner/README.md#changing-the-default).

### Jest

A Jest configuration for the `runner` is available through a separate [`unmock-jest-runner`](https://github.com/meeshkan/unmock-jest-runner) package. While the standard `unmock-runner` is available via NPM, you'll want to use the `unmock-jest-runner` when executing your tests to ensure proper error handling. 

The `unmock-jest-runner` can be installed via NPM or Yarn:

```
npm install -D unmock-jest-runner
yarn add unmock-jest-runner
```

Once installed, the `runner` can be imported as a default and used as a wrapper for your tests:

```js
const runner = require("unmock-jest-runner").default;

test("my API always works as expected", runner(async () => {
  const res = await myApiFunction();
  // some expectations
}));
```

### Other configurations

As of now, Jest is the only package we have available. 

However, we're currently building out support for [Mocha](https://github.com/Meeshkan/unmock-js/issues/299) and [QUnit](https://github.com/Meeshkan/unmock-js/issues/300). You can follow the progress of those implementations in the corresponding issues.

## OpenAPI

Unmock supports the reading of OpenAPI specifications out of the box. Place your specification in a folder at the root of your project called `__unmock__/<myspecname>`, where `<myspecname>` is the name of the spec on the `unmock.on().services` object.  Several examples of this exist on the internet, most notably [here](https://github.com/unmock/unmock-examples/tree/master/using-service-repository).

## Tutorials

- [Unmock ts koans](https://github.com/meeshkan/unmock-ts-koans)
- [Unmock js katacoda](https://www.katacoda.com/unmock/scenarios/introduction)

## Contributing

Thanks for wanting to contribute! Take a look at our [Contributing Guide](CONTRIBUTING.md) for notes on our commit message conventions and how to run tests.

Please note that this project is governed by the [Meeshkan Community Code of Conduct](https://github.com/meeshkan/code-of-conduct). By participating in this project, you agree to abide by its terms.

## Development

- See [publish.md](./publish.md) for instructions how to make a new release

## License

[MIT](LICENSE)

Copyright (c) 2018–2019 [Meeshkan](http://meeshkan.com) and other [contributors](https://github.com/unmock/unmock-js/graphs/contributors).
 readmeEtag: '"b12af4fce603397889647bea7d9ab4fe5823e80f"' readmeLastModified: Mon, 23 Nov 2020 10:05:08 GMT repositoryId: 158103154 description: Fuzz test your REST API calls created: '2018-11-18T16:24:30Z' updated: '2023-10-19T15:30:47Z' language: TypeScript archived: true stars: 93 watchers: 6 forks: 8 owner: meeshkan logo: https://avatars.githubusercontent.com/u/32298527?v=4 repoEtag: '"58313482dbc208be81c3e95bc3dbd60c5a7b992b29b877dbd28c6bdb04df75ca"' repoLastModified: Thu, 19 Oct 2023 15:30:47 GMT foundInMaster: true id: e200bd53331c006b83cf719eb34ec292 oldLocations: - https://github.com/unmock/unmock-js - source: https://openapi.tools/ name: OpenAPI Enforcer Middleware category: - Server - Parsers language: Node.js link: https://www.npmjs.com/package/openapi-enforcer-middleware repository: https://github.com/gi60s/openapi-enforcer-middleware source_description: >- An express middleware that makes it easy to write web services that follow an OpenAPI specification by leveraging the tools provided in the openapi-enforcer package. v2: true v3: true repositoryMetadata: base64Readme: >- IyBPcGVuIEFQSSBFbmZvcmNlciBNaWRkbGV3YXJlCgpBbiBleHByZXNzIG1pZGRsZXdhcmUgdGhhdCBtYWtlcyBpdCBlYXN5IHRvIHdyaXRlIHdlYiBzZXJ2aWNlcyB0aGF0IGZvbGxvdyBhbiBPcGVuIEFQSSBzcGVjaWZpY2F0aW9uIGJ5IGxldmVyYWdpbmcgdGhlIHRvb2xzIHByb3ZpZGVkIGluIHRoZSBbb3BlbmFwaS1lbmZvcmNlcl0oaHR0cHM6Ly93d3cubnBtanMuY29tL3BhY2thZ2Uvb3BlbmFwaS1lbmZvcmNlcikgcGFja2FnZS4KCiMjIEZlYXR1cmVzCgotIFN1cHBvcnRzIE9wZW5BUEkgKFN3YWdnZXIpIDIuMCBhbmQgMy54Ci0gRXhwcmVzcyBtaWRkbGV3YXJlCi0gQXV0b21hdGljYWxseSBsaW5rIEphdmFTY3JpcHQgZnVuY3Rpb25zIHRvIHBhdGggZW5kcG9pbnRzCi0gUGFyc2VzIGFuZCB2YWxpZGF0ZXMgaW5jb21pbmcgcmVxdWVzdHMKLSBWYWxpZGF0ZXMgcmVzcG9uc2VzIHByaW9yIHRvIHNlbmRpbmcKLSBBdXRvbWF0aWMgcmVzcG9uc2UgbW9ja2luZyBpbiBkZXZlbG9wbWVudAotIE9wdGlvbiBmb3IgbWFudWFsIHJlc3BvbnNlIG1vY2tpbmcgaW4gcHJvZHVjdGlvbgotIEhpZ2hseSBjb25maWd1cmFibGUKLSBBY2NlcHRzIG1pZGRsZXdhcmUgcGx1Z2lucwoKIyMgSW5zdGFsbGF0aW9uCgpUaGlzIHBhY2thZ2UgaGFzIFtvcGVuYXBpLWVuZm9yY2VyXShodHRwczovL3d3dy5ucG1qcy5jb20vcGFja2FnZS9vcGVuYXBpLWVuZm9yY2VyKSBhcyBhIHBlZXIgZGVwZW5kZW5jeSwgc28gYm90aCBtdXN0IGJlIGluc3RhbGxlZC4KCmBgYGJhc2gKbnBtIGluc3RhbGwgb3BlbmFwaS1lbmZvcmNlciBvcGVuYXBpLWVuZm9yY2VyLW1pZGRsZXdhcmUKYGBgCgojIyBEb2N1bWVudGF0aW9uCgpodHRwczovL2J5dS1vaXQuZ2l0aHViLmlvL29wZW5hcGktZW5mb3JjZXItbWlkZGxld2FyZS8K readmeEtag: '"09e2cf7abd1e26e27ccbf007cbd2d88ed08ee946"' readmeLastModified: Thu, 17 Feb 2022 21:06:17 GMT repositoryId: 120667667 description: >- An express middleware that makes it easy to write web services that follow an Open API specification. created: '2018-02-07T20:28:20Z' updated: '2022-12-05T04:32:04Z' language: JavaScript archived: false stars: 17 watchers: 3 forks: 10 owner: Gi60s logo: https://avatars.githubusercontent.com/u/714117?v=4 license: Apache-2.0 repoEtag: '"8fe6b6f6d3d3c4b6f4fad634a39ea581cd4e0f4ae70e63f5c0bb10c69551a63c"' repoLastModified: Mon, 05 Dec 2022 04:32:04 GMT foundInMaster: true id: b7b2f24374c433724155eceeced5fb7c oldLocations: - https://github.com/byu-oit/openapi-enforcer-middleware - source: https://openapi.tools/ name: BigstickCarpet/swagger-cli category: - Description Validators - Parsers repository: https://github.com/apidevtools/swagger-cli language: - Node.js - CLI source_description: >- Simple validation for OpenAPI files, supporting JSON/YAML and v2/v3 description documents. v2: true v3: true repositoryMetadata: base64Readme: >- Swagger/OpenAPI CLI
============================

[![npm](https://img.shields.io/npm/v/@apidevtools/swagger-cli.svg)](https://www.npmjs.com/package/@apidevtools/swagger-cli)
[![License](https://img.shields.io/npm/l/@apidevtools/swagger-cli.svg)](LICENSE)
[![Buy us a tree](https://img.shields.io/badge/Treeware-%F0%9F%8C%B3-lightgreen)](https://plant.treeware.earth/APIDevTools/swagger-cli)


> ⚠️ Swagger CLI has been deprecated, due to the maintenance burnden of trying to keep up with expectations of a huge userbase with little to no pull requests or support. [Redocly CLI](https://redocly.com/redocly-cli/) covers all of the same functionality, and has more advanced linting with custom rules, and we highly recommend using that instead. They have conveniently provided a [migration guide](https://redocly.com/docs/cli/guides/migrate-from-swagger-cli/) for existing Swagger CLI users. Read the review of [Redocly CLI from APIs You Won't Hate](https://apisyouwonthate.com/blog/redocly-cli/).

Features
--------------------------
- Validate Swagger/OpenAPI files in **JSON or YAML** format
- Supports multi-file API definitions via `$ref` pointers
- Bundle multiple Swagger/OpenAPI files into one combined file



Related Projects
--------------------------
- [Swagger Parser](https://github.com/APIDevTools/swagger-parser)
- [Swagger Express Middleware](https://github.com/APIDevTools/swagger-express-middleware)



Installation
--------------------------
Install using [npm](https://docs.npmjs.com/about-npm/):

```bash
npm install -g @apidevtools/swagger-cli
```



Usage
--------------------------

```
swagger-cli <command> [options] <file>

Commands:
    validate                Validates an API definition in Swagger 2.0 or OpenAPI 3.0 format

    bundle                  Bundles a multi-file API definition into a single file

Options:
    -h, --help              Show help for any command
    -v, --version           Output the CLI version number
    -d, --debug [filter]    Show debug output, optionally filtered (e.g. "*", "swagger:*", etc.)
```


### Validate an API

The `swagger-cli validate` command will validate your Swagger/OpenAPI definition against the [Swagger 2.0 schema](https://github.com/OAI/OpenAPI-Specification/blob/master/schemas/v2.0/schema.json) or [OpenAPI 3.0 Schema](https://github.com/OAI/OpenAPI-Specification/blob/master/schemas/v3.0/schema.json).  It also performs additional validations against the [specification](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md), which will catch some things that aren't covered by the schema, such as duplicate parameters, invalid MIME types, etc.

The command will exit with a non-zero code if the API is invalid.

```
swagger-cli validate [options] <file>

Options:
    --no-schema             Do NOT validate against the Swagger/OpenAPI JSON schema

    --no-spec               Do NOT validate against the Swagger/OpenAPI specification
```

#### Git pre-commit hook

There is a useful Python tool called [pre-commit](https://pre-commit.com/) that can be used to execute a wide suite of pre-commit checks. The `swagger-cli validate` command can be integrated as part of a git pre-commit hook by adding the following configuration to the `repos` entry of an existing `.pre-commit-config.yaml` file.

```
-   repo: https://github.com/APIDevTools/swagger-cli
    rev: v2.2.1
    hooks:
    - id: swagger-validation
      args: ["validate", "<path to root swagger>"]
```

The intention is to point to single root swagger that references multiple swagger definitions. The above hook will execute the `swagger-cli validation` against the root swagger anytime that a file matching the pattern `.*swagger.*\.(json|yaml|yml)` is modified. Any failures in this validation will prevent the git commit from being processed.

### Combine Multiple Files

The Swagger and OpenAPI specs allows you to split your API definition across multiple files using [`$ref` pointers](https://github.com/OAI/OpenAPI-Specification/blob/master/versions/2.0.md#reference-object) to reference each file. You can use the `swagger-cli bundle` command to combine all of those referenced files into a single file, which is useful for distribution or interoperation with other tools.

By default, the `swagger-cli bundle` command tries to keep the output file size as small as possible, by only embedding each referenced file _once_.  If the same file is referenced multiple times, then any subsequent references are simply modified to point to the _single_ inlined copy of the file.  If you want to produce a bundled file without _any_ `$ref` pointers, then add the `--dereference` option.  This will result in a larger file size, since multiple references to the same file will result in that file being embedded multiple times.

If you don't specify the `--outfile` option, then the bundled API will be written to stdout, which means you can pipe it to other commands.

The result of this method by default is written as JSON. It can be changed to YAML with the `--type` option, by passing the `yaml` value.

```
swagger-cli bundle [options] <file>

Options:
    -o, --outfile <file>        The output file

    -r, --dereference           Fully dereference all $ref pointers

    -f, --format <spaces>       Formats the output using the given number of spaces
                                (the default is 2 spaces)

    -t, --type <filetype>       Defines the output file type. The valid values are: json, yaml
                                (the default is JSON)

    -w, --wrap <column>         Set the line length for YAML strings
                                (the default is no wrapping)
```



Contributing
--------------------------
I welcome any contributions, enhancements, and bug-fixes.  [Open an issue](https://github.com/APIDevTools/swagger-cli/issues) on GitHub and [submit a pull request](https://github.com/APIDevTools/swagger-cli/pulls).

#### Building/Testing
To build/test the project locally on your computer:

1. **Clone this repo**<br>
`git clone https://github.com/APIDevTools/swagger-cli.git`

2. **Install dependencies**<br>
`npm install`

3. **Run the tests**<br>
`npm test`



License
--------------------------
Swagger CLI is 100% free and open-source, under the [MIT license](LICENSE). Use it however you want.

This package is [Treeware](http://treeware.earth). If you use it in production, then we ask that you [**buy the world a tree**](https://plant.treeware.earth/APIDevTools/swagger-cli) to thank us for our work. By contributing to the Treeware forest you’ll be creating employment for local families and restoring wildlife habitats.



Big Thanks To
--------------------------
Thanks to these awesome companies for their support of Open Source developers ❤

[![Travis CI](https://jstools.dev/img/badges/travis-ci.svg)](https://travis-ci.com)
[![SauceLabs](https://jstools.dev/img/badges/sauce-labs.svg)](https://saucelabs.com)
[![Coveralls](https://jstools.dev/img/badges/coveralls.svg)](https://coveralls.io)
 readmeEtag: '"b6366f698af1bbc5531a38cc611e28aa3c438a1b"' readmeLastModified: Wed, 15 Nov 2023 11:23:27 GMT repositoryId: 36670476 description: Swagger 2.0 and OpenAPI 3.0 command-line tool created: '2015-06-01T15:36:25Z' updated: '2025-10-27T07:46:59Z' language: JavaScript archived: true stars: 518 watchers: 11 forks: 69 owner: APIDevTools logo: https://avatars.githubusercontent.com/u/43750074?v=4 license: MIT repoEtag: '"3a7b18010917df6b0b92d582cdc503ee89e79009b4ecba4f4e0105cb572b7cde"' repoLastModified: Mon, 27 Oct 2025 07:46:59 GMT foundInMaster: true id: 1861975d6734eb6c3e0ab0980e241853 oldLocations: - https://github.com/bigstickcarpet/swagger-cli - source: https://openapi.tools/ name: OpenAPI Enforcer category: - Data Validators - Description Validators - Server - Testing - Parsers language: Node.js link: https://www.npmjs.com/package/openapi-enforcer repository: https://github.com/gi60s/openapi-enforcer source_description: >- Validate your OpenAPI document, serialize, deserialize, and validate incoming requests and outgoing responses, and simplify response building. You can even produce mock data. v2: true v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJLUVuZm9yY2VyCgpUb29scyBmb3IgdXNpbmcgdGhlIE9wZW4gQVBJIFNwZWNpZmljYXRpb24gKE9BUykKCioqU3VwcG9ydHMgT0FTIDIuMCAoZm9ybWVybHkgU3dhZ2dlcikgYW5kIE9BUyAzLngueCoqCgojIyBGZWF0dXJlcwoKLSBDcmVhdGUgYW4gQVBJLgotIFZhbGlkYXRlIHlvdXIgT0FTIGRvY3VtZW50cy4KLSBTZXJpYWxpemUsIGRlc2VyaWFsaXplLCBhbmQgdmFsaWRhdGUgdmFsdWVzIGFnYWluc3QgT0FTIHNjaGVtYXMuCi0gSWRlbnRpZnkgdGhlIG9wZXJhdGlvbiBhc3NvY2lhdGVkIHdpdGggYSByZXF1ZXN0LgotIFBhcnNlLCBkZXNlcmlhbGl6ZSwgYW5kIHZhbGlkYXRlIHJlcXVlc3QgcGFyYW1ldGVycy4KLSBGYWNpbGl0YXRlZCByZXNwb25zZSBidWlsZGluZy4KLSBHZW5lcmF0ZSByYW5kb20gdmFsaWQgdmFsdWVzIGZvciBhIHNjaGVtYS4KLSBQbHVnaW4gZW52aXJvbm1lbnQgZm9yIGN1c3RvbSBkb2N1bWVudCB2YWxpZGF0aW9uIGFuZCBleHRlbmRlZCBmdW5jdGlvbmFsaXR5IGluY2x1ZGluZyBjdXN0b20gZGF0YSB0eXBlIGZvcm1hdHMuCgojIyBXZWJzaXRlIC0gW29wZW5hcGktZW5mb3JjZXIuY29tXShodHRwczovL29wZW5hcGktZW5mb3JjZXIuY29tLykKCiMjIEluc3RhbGxhdGlvbgoKYGBgc2hlbGwKbnBtIGluc3RhbGwgb3BlbmFwaS1lbmZvcmNlcgpgYGAKCiMjIEV4YW1wbGVzCgojIyMgTG9hZGluZyBhbmQgVmFsaWRhdGluZyBhIERvY3VtZW50CgpVc2UgdGhlIEVuZm9yY2VyIHRvIGxvYWQgYW5kIHJlc29sdmUgYWxsICRyZWYgdmFsdWVzIGFuZCB0aGVuIHRvIHZhbGlkYXRlIHRoZSBjb21wbGV0ZSBkb2N1bWVudC4KCmBgYGpzCmNvbnN0IEVuZm9yY2VyID0gcmVxdWlyZSgnb3BlbmFwaS1lbmZvcmNlcicpCgphc3luYyBmdW5jdGlvbiBydW4gKCkgewogIGNvbnN0IFtvcGVuYXBpLCBlcnJvciwgd2FybmluZ10gPSBhd2FpdCBFbmZvcmNlcignLi9wYXRoL3RvL29wZW5hcGkueW1sJywgewogICAgZnVsbFJlc3VsdDogdHJ1ZQogIH0pCiAgaWYgKGVycm9yICE9PSB1bmRlZmluZWQpIGNvbnNvbGUuZXJyb3IoZXJyb3IpCiAgaWYgKHdhcm5pbmcgIT09IHVuZGVmaW5lZCkgY29uc29sZS53YXJuKHdhcm5pbmcpCiAgaWYgKG9wZW5hcGkgIT09IHVuZGVmaW5lZCkgY29uc29sZS5sb2coJ0RvY3VtZW50IGlzIHZhbGlkJykKfQoKcnVuLmNhdGNoKGNvbnNvbGUuZXJyb3IpCmBgYAoKIyMjIFByb2Nlc3NpbmcgYW4gSW5jb21pbmcgUmVxdWVzdAoKYGBganMKY29uc3QgRW5mb3JjZXIgPSByZXF1aXJlKCdvcGVuYXBpLWVuZm9yY2VyJykKCmFzeW5jIGZ1bmN0aW9uIHJ1biAoKSB7CiAgLy8gQmVjYXVzZSB3ZSBkb24ndCBzcGVjaWZ5IGBmdWxsUmVzdWx0OiB0cnVlYCwgYW55IGVycm9ycyB3aWxsIHRocm93IGFuIGV4Y2VwdGlvbiBhbmQKICAvLyB3YXJuaW5ncyB3aWxsIGJlIGxvZ2dlZCB0byB0aGUgY29uc29sZS4KICBjb25zdCBvcGVuYXBpID0gYXdhaXQgRW5mb3JjZXIoJy4vcGF0aC90by9vcGVuYXBpLnltbCcpCgogIC8vIElmIHRoZSByZXF1ZXN0IGlzIHZhbGlkIHRoZW4gdGhlIHJlcSBvYmplY3Qgd2lsbCBjb250YWluIHRoZSBwYXJzZWQgYW5kIHZhbGlkYXRlZCByZXF1ZXN0LgogIC8vIElmIGl0IGlzIGludmFsaWQgdGhlbiB0aGUgZXJyb3Igd2lsbCBjb250YWluIGRldGFpbHMgYWJvdXQgd2hhdCB3YXMgd3Jvbmcgd2l0aCB0aGUKICAvLyByZXF1ZXN0IGFuZCB0aGVzZSBkZXRhaWxzIGFyZSBzYWZlIHRvIHJldHVybiB0byB0aGUgY2xpZW50IHRoYXQgbWFkZSB0aGUgcmVxdWVzdC4KICBjb25zdCBbIHJlcSwgZXJyb3IgXSA9IG9wZW5hcGkucmVxdWVzdCh7CiAgICBtZXRob2Q6ICdQT1NUJywKICAgIHBhdGg6ICcvdGFza3MnLAogICAgLy8gdGhlIGJvZHkgc2hvdWxkIGJlIHBhcnNlZCBieSBhIEpTT04ucGFyc2UoKSBwcmlvciB0byBwYXNzaW5nIGluIChpZiBhcHBsaWNhYmxlKS4KICAgIGJvZHk6IHsgdGFzazogJ0J1eSBNaWxrJywgcXVhbnRpdHk6IDIgfQogIH0pCgogIC8vIFlvdSBjYW4gdXNlIHRoZSByZXEub3BlcmF0aW9uIHByb3BlcnR5IHRvIGxvb2sgYXQgdGhlIHByb3BlcnRpZXMgZnJvbSB5b3VyIE9wZW5BUEkgZG9jdW1lbnQuCiAgLy8gQSBnb29kIHVzZSBvZiB0aGlzIGlzIHRvIGxvb2sgYXQgdGhlIG9wZXJhdGlvbklkIHlvdSBkZWZpbmVkIHRoZXJlIHRvIGRldGVybWluZSB3aGljaCBwYXRoCiAgLy8gaXMgYmVpbmcgdXNlZCB0byBoYW5kbGUgdGhlIHJlcXVlc3QuCiAgaWYgKHJlcS5vcGVyYXRvbi5vcGVyYXRpb25JZCA9PT0gJ215LW9wZXJhdGlvbi1pZCcpIHsKICAgIC8vIC4uLiBhZGRpdGlvbmFsIHJlcXVlc3QgcHJvY2Vzc2luZwogIH0KfQoKcnVuLmNhdGNoKGNvbnNvbGUuZXJyb3IpCmBgYAoKIyMjIFByb2R1Y2luZyBhIFZhbGlkIFJlc3VsdAoKYGBganMKY29uc3QgRW5mb3JjZXIgPSByZXF1aXJlKCdvcGVuYXBpLWVuZm9yY2VyJykKCmFzeW5jIGZ1bmN0aW9uIHJ1biAoKSB7CiAgY29uc3Qgb3BlbmFwaSA9IGF3YWl0IEVuZm9yY2VyKCcuL3BhdGgvdG8vb3BlbmFwaS55bWwnKQoKICBjb25zdCBbIHJlcSBdID0gb3BlbmFwaS5yZXF1ZXN0KHsKICAgIG1ldGhvZDogJ1BPU1QnLAogICAgcGF0aDogJy90YXNrcycsCiAgICAvLyB0aGUgYm9keSBzaG91bGQgYmUgcGFyc2VkIGJ5IGEgSlNPTi5wYXJzZSgpIHByaW9yIHRvIHBhc3NpbmcgaW4gKGlmIGFwcGxpY2FibGUpLgogICAgYm9keTogeyB0YXNrOiAnQnV5IE1pbGsnLCBxdWFudGl0eTogMiB9CiAgfSkKCiAgY29uc3QgYm9keSA9IHsgaWQ6IDEsIHRhc2s6ICdCdXkgTWlsaycsIHF1YW50aXR5OiAyLCBkYXRlQ29tcGxldGVkOiBudWxsIH0KICBjb25zdCBoZWFkZXJzID0ge30KCiAgLy8gVGhpcyB3aWxsIHZhbGlkYXRlIHRoZSByZXNwb25zZSBjb2RlLCBib2R5LCBhbmQgaGVhZGVycy4gSXQgd2lsbCBhbHNvIGNvcnJlY3RseSBzZXJpYWxpemUKICAvLyB0aGUgYm9keSBhbmQgaGVhZGVycyBmb3Igc2VuZGluZyB0byB0aGUgY2xpZW50IHRoYXQgbWFkZSB0aGUgcmVxdWVzdC4gVXNpbmcgdGhpcyBtZXRob2QKICAvLyB5b3UnbGwgbmV2ZXIgc2VuZCBiYWNrIGEgcmVzcG9uc2UgdGhhdCBkb2VzIG5vdCBtYXRjaCB3aGF0IHlvdXIgT3BlbkFQSSBkb2N1bWVudCBkZWZpbmVzLgogIGNvbnN0IFsgcmVzLCBlcnJvciBdID0gcmVxLnJlc3BvbnNlKDIwMCwgYm9keSwgaGVhZGVycykKICBjb25zb2xlLmxvZyhyZXMuYm9keSwgcmVzLmhlYWRlcnMpCn0KCnJ1bi5jYXRjaChjb25zb2xlLmVycm9yKQpgYGAK readmeEtag: '"379b08da4f96adbeb6c3864a5d5536393476a12b"' readmeLastModified: Sat, 02 Mar 2024 00:36:43 GMT repositoryId: 120373530 description: null created: '2018-02-05T23:06:16Z' updated: '2025-12-11T11:06:21Z' language: JavaScript archived: false stars: 97 watchers: 6 forks: 25 owner: Gi60s logo: https://avatars.githubusercontent.com/u/714117?v=4 license: Apache-2.0 repoEtag: '"6dd30aac5cfecaf749ff244330525608aef055fb163d31dccfcd53feae4160c9"' repoLastModified: Thu, 11 Dec 2025 11:06:21 GMT foundInMaster: true id: d154dd3ce5cfba35ca50effbfd386d73 oldLocations: - https://github.com/byu-oit/openapi-enforcer - source: https://openapi.tools/ name: guardrail category: SDK link: https://github.com/twilio/guardrail repository: https://github.com/guardrail-dev/guardrail language: - Scala - Java - ... source_description: Principled code generation from OpenAPI descriptions v2: true v3: true repositoryMetadata: base64Readme: >- guardrail [![Build Status](https://github.com/guardrail-dev/guardrail/workflows/CI/badge.svg)](https://github.com/guardrail-dev/guardrail/actions?query=workflow%3A%22CI%22) | [![codecov](https://codecov.io/gh/guardrail-dev/guardrail/branch/master/graph/badge.svg?token=ssLYYkVBgv)](https://codecov.io/gh/guardrail-dev/guardrail) | [![Matrix chat](https://img.shields.io/matrix/guardrail:matrix.org.svg?label=matrix&server_fqdn=matrix.org)](https://matrix.to/#/#guardrail:matrix.org) | [![Join the chat at https://gitter.im/guardrail-dev/guardrail](https://badges.gitter.im/guardrail-dev/guardrail.svg)](https://gitter.im/guardrail-dev/guardrail?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge) | [![Scala Steward badge](https://img.shields.io/badge/Scala_Steward-helping-blue.svg?style=flat&logo=data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAA4AAAAQCAMAAAARSr4IAAAAVFBMVEUAAACHjojlOy5NWlrKzcYRKjGFjIbp293YycuLa3pYY2LSqql4f3pCUFTgSjNodYRmcXUsPD/NTTbjRS+2jomhgnzNc223cGvZS0HaSD0XLjbaSjElhIr+AAAAAXRSTlMAQObYZgAAAHlJREFUCNdNyosOwyAIhWHAQS1Vt7a77/3fcxxdmv0xwmckutAR1nkm4ggbyEcg/wWmlGLDAA3oL50xi6fk5ffZ3E2E3QfZDCcCN2YtbEWZt+Drc6u6rlqv7Uk0LdKqqr5rk2UCRXOk0vmQKGfc94nOJyQjouF9H/wCc9gECEYfONoAAAAASUVORK5CYII=)](https://scala-steward.org)
===

guardrail is a code generation tool, capable of reading from OpenAPI/Swagger specification files and generating both Scala and Java source code, targeting various libraries and frameworks listed here:

- Scala: `akka-http` and `http4s`, both backed by `circe`, as well as `dropwizard` backed by `jackson`
- Java: `dropwizard` and `spring-mvc`, both backed by `jackson`

Build tool plugins
------------------

| Plugins | versions | docs |
|---|---|---|
| [`guardrail-dev/sbt-guardrail`](https://github.com/guardrail-dev/sbt-guardrail) | [![sbt-guardrail](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/sbt-guardrail/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:sbt-guardrail) | [docs](docs/plugins/sbt.md) |
| [`guardrail-dev/guardrail-maven-plugin`](https://github.com/guardrail-dev/guardrail-maven-plugin) | [![guardrail-maven-plugin](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-maven-plugin/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-maven-plugin) | [docs](docs/plugins/maven.md) |
| [`guardrail-dev/guardrail-gradle-plugin`](https://github.com/guardrail-dev/guardrail-gradle-plugin) | `guardrail-gradle-plugin` | [Plugin Portal](https://plugins.gradle.org/plugin/dev.guardrail) | [docs](docs/plugins/gradle.md) |
| [CLI support](./modules/cli) | Latest | [`cs install guardrail`](https://get-coursier.io/docs/cli-install), [docs](docs/plugins/make.md) |

New to guardrail?
---

Check out the [docs](https://guardrail.dev/)!

Compatible library versions are listed in [COMPATIBILITY.md](COMPATIBILITY.md)

guardrail module versions
---

guardrail is modularized, using [`sbt-version-policy`](https://github.com/scalacenter/sbt-version-policy) to ensure binary compatibility between dependent modules.

The dependency chain and versions of published modules are listed below for reference:

| module  | version  | depends on |
|-----|-----|-----|
| guardrail-core | [![guardrail-core](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-core_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-core_2.12) |  |
| guardrail-java-support | [![guardrail-java-support](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-java-support_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-java-support_2.12) | core |
| guardrail-java-async-http | [![guardrail-java-async-http](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-java-async-http_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-java-async-http_2.12) | java-support |
| guardrail-java-dropwizard | [![guardrail-java-dropwizard](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-java-dropwizard_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-java-dropwizard_2.12) | java-support, java-async-http |
| guardrail-java-spring-mvc | [![guardrail-java-spring-mvc](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-java-spring-mvc_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-java-spring-mvc_2.12) | java-support |
| guardrail-scala-support | [![guardrail-scala-support](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-scala-support_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-scala-support_2.12) | core |
| guardrail-scala-akka-http | [![guardrail-scala-akka-http](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-scala-akka-http_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-scala-akka-http_2.12) | scala-support |
| guardrail-scala-dropwizard | [![guardrail-scala-dropwizard](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-scala-dropwizard_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-scala-dropwizard_2.12) | scala-support |
| guardrail-scala-http4s | [![guardrail-scala-http4s](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-scala-http4s_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-scala-http4s_2.12) | scala-support |
| guardrail-cli | [![guardrail-cli](https://maven-badges.herokuapp.com/maven-central/dev.guardrail/guardrail-cli_2.12/badge.svg)](https://search.maven.org/search?q=g:dev.guardrail%20a:guardrail-cli_2.12) | guardrail-core |

Interested in contributing?
---

[CONTRIBUTING.md](CONTRIBUTING.md) provides an overview of how the project is structured, expectations, and information around writing new integration tests.
The [issue tracker](https://github.com/guardrail-dev/guardrail/issues) also has tags for [`help wanted`](https://github.com/guardrail-dev/guardrail/issues?q=is%3Aissue+is%3Aopen+label%3A%22help+wanted%22) and [`good first issue`](https://github.com/guardrail-dev/guardrail/issues?q=is%3Aissue+is%3Aopen+label%3A%22good+first+issue%22).

Adopters
========

- [Avast](https://www.avast.com/)
- [Twilio](https://www.twilio.com/)

Contributors
============

We used to have a list of contributors here, but [github's Contributors page](https://github.com/guardrail-dev/guardrail/graphs/contributors) is much more accurate. Thanks to those who contributed before the project was open sourced!
 readmeEtag: '"f6b23074c7d0e522514219d62aa01792c01b934b"' readmeLastModified: Tue, 30 Jul 2024 13:56:35 GMT repositoryId: 102150798 description: Principled code generation from OpenAPI specifications created: '2017-09-01T20:31:36Z' updated: '2026-02-05T03:40:59Z' language: Scala archived: false stars: 540 watchers: 24 forks: 137 owner: guardrail-dev logo: https://avatars.githubusercontent.com/u/77855402?v=4 license: MIT repoEtag: '"2913bd0643a98aa412eb61d91e82612c61e65a2ed44e7bef1de56d484f951f8b"' repoLastModified: Thu, 05 Feb 2026 03:40:59 GMT foundInMaster: true id: 5a3fd6a978abf3303c9cce3950582502 oldLocations: - https://github.com/twilio/guardrail - source: https://openapi.tools/ name: LoopBack 4 category: Server language: Node.js + TypeScript link: https://loopback.io/ repository: https://github.com/loopbackio/loopback-next source_description: > A highly extensible object-oriented Node.js and TypeScript framework for building APIs and microservices with tight OpenAPI 3 integration. Serves Swagger UI and OpenAPI 3 spec out of the box. Generate code to interact with other OpenAPI-compliant APIs, or generate new API endpoints based on existing OpenAPI specs. repositoryMetadata: base64Readme: >- <img src="https://raw.githubusercontent.com/loopbackio/loopback.io/gh-pages/images/branding/logo/blue/loopback-sm.png" alt="LoopBack4 logo" width="400"/>

[![Continuous Integration Status](https://github.com/loopbackio/loopback-next/actions/workflows/continuous-integration.yml/badge.svg)](https://github.com/loopbackio/loopback-next/actions/workflows/continuous-integration.yml)
[![Coverage Status](https://coveralls.io/repos/github/loopbackio/loopback-next/badge.svg?branch=master)](https://coveralls.io/github/loopbackio/loopback-next?branch=master)
[![CodeQL Status](https://github.com/loopbackio/loopback-next/workflows/CodeQL/badge.svg)](https://github.com/loopbackio/loopback-next/actions?query=workflow%3ACodeQL)

[![Twitter](https://img.shields.io/twitter/follow/strongloop.svg?style=social&label=Follow%20%40strongloop)](https://twitter.com/strongloop)
[![LinkedIn](https://img.shields.io/badge/Follow%20us-white?logo=linkedIn&color=0077B5&logoColor=white)](https://www.linkedin.com/groups/5046525/)
[![Slack](https://img.shields.io/badge/slack-Join%20workspace-%234A154B?logo=slack)](https://join.slack.com/t/loopbackio/shared_invite/zt-8lbow73r-SKAKz61Vdao~_rGf91pcsw)

LoopBack makes it easy to build modern applications that require complex
integrations.

- Fast, small, powerful, extensible core
- Generate real APIs with a single command
- Define your data and endpoints with OpenAPI
- No maintenance of generated code

## Status: General Availability

LoopBack 4 GA (General Availability) has been released in October 2018, read
more in [the announcement post](http://strongloop.com/strongblog/loopback-4-ga).

The documentation website is https://loopback.io/doc/en/lb4/.

Learn about the latest and greatest
[features and technologies in LoopBack 4](https://loopback.io/doc/en/lb4/Crafting-LoopBack-4.html)
by using it for your next project. Start by having a look at
[Getting Started](https://loopback.io/doc/en/lb4/Getting-started.html).

Check the [API documentation](https://loopback.io/doc/en/lb4/apidocs.index.html)
for all the API usages in each package.

### Long Term Support

We don't provide any LTS version for LoopBack 4 yet. Please join the discussion
in [loopback-next#4398](https://github.com/loopbackio/loopback-next/issues/4398)
if you are interested in a version that's less frequently changed.

| Version    | Status      | Published | EOL                  |
| ---------- | ----------- | --------- | -------------------- |
| LoopBack 4 | Current     | Oct 2018  | Apr 2028 _(minimum)_ |
| LoopBack 3 | End-of-Life | Dec 2016  | Dec 2020             |
| LoopBack 2 | End-of-Life | Jul 2014  | Apr 2019             |

Please refer to our
[Long Term Support Policy](https://loopback.io/doc/en/contrib/Long-term-support.html)
for more details.

## Installation

Make sure you have the following installed:

| Package                                    | Version/-s                                                             | Link                                  | Note                                                                                                  |
| ------------------------------------------ | ---------------------------------------------------------------------- | ------------------------------------- | ----------------------------------------------------------------------------------------------------- |
| [Node.js](https://nodejs.org/en/download/) | Maintenance LTS (_v20_) <br/> Active LTS (_v22_) <br/> Current (_v24_) | https://nodejs.org/en/about/releases/ | <span style="color: yellow;">The use of the current version for production is not recommended</span>. |

You can generate a project with our generator or with the CLI as follows:

```shell
npm create loopback
npx -p @loopback/cli lb app
```

To create your first LoopBack 4 application, see
[Getting Started](http://loopback.io/doc/en/lb4/Getting-started.html).

## Documentation

- [Official documentation](http://loopback.io/doc/en/lb4/)
- [API documentation](https://loopback.io/doc/en/lb4/apidocs.index.html)
- [FAQ](http://loopback.io/doc/en/lb4/FAQ.html)
- [LoopBack 3 vs LoopBack 4](http://loopback.io/doc/en/lb4/migration-overview.html)
- [Tutorials](http://loopback.io/doc/en/lb4/Tutorials.html)
- [Examples](http://loopback.io/doc/en/lb4/Examples.html)

## Contributing

See the following resources to get you started:

- [Contributing Guidelines](./docs/CONTRIBUTING.md)
- [Monorepo overview](./docs/site/MONOREPO.md)
- [Developing LoopBack](./docs/site/DEVELOPING.md)

You can join the team by posting a comment to
[issue #110](https://github.com/loopbackio/loopback-next/issues/110).

### Security

If you think you have discovered a new security issue with any LoopBack package,
**please do not report it on GitHub**. Instead, send an email to
[security@loopback.io](mailto:security@loopback.io) with a full description and
steps to reproduce.

See [SECURITY.md](SECURITY.md) for more details.

## Team

### Project Architect

|                  Raymond Feng                   |
| :---------------------------------------------: |
| [![raymondfeng]](http://github.com/raymondfeng) |

### Technical Steering Committee

|                  Raymond Feng                   |               Diana Lau               |                       Mario Estrada                        |               Rifa Achrinza                |              Francisco Buceta              |            Samarpan Bhattacharya             |
| :---------------------------------------------: | :-----------------------------------: | :--------------------------------------------------------: | :----------------------------------------: | :----------------------------------------: | :------------------------------------------: |
| [![raymondfeng]](http://github.com/raymondfeng) | [![dhmlau]](http://github.com/dhmlau) | [![marioestradarosa]](https://github.com/marioestradarosa) | [![achrinza]](https://github.com/achrinza) | [![frbuceta]](https://github.com/frbuceta) | [![samarpanB]](https://github.com/samarpanB) |

### Other Project Maintainers

|                 Nora Abdelgadir                  |               Matthew Schnee               |                    Hage Yaapa                    |
| :----------------------------------------------: | :----------------------------------------: | :----------------------------------------------: |
| [![nabdelgadir]](https://github.com/nabdelgadir) |  [![mschnee]](https://github.com/mschnee)  | [![hacksparrow]](https://github.com/hacksparrow) |
|                  **Agnes Lin**                   |                 **Madaky**                 |                 **Hugo Da Roit**                 |
|    [![agnes512]](https://github.com/agnes512)    |   [![madaky]](https://github.com/madaky)   |        [![yaty]](https://github.com/yaty)        |
|                  **Nico Flaig**                  |             **Denny Bartelt**              |             **Douglas McConnachie**              |
|      [![nflaig]](https://github.com/nflaig)      |  [![derdeka]](https://github.com/derdeka)  |    [![dougal83]](https://github.com/dougal83)    |
|            **Samarpan Bhattacharya**             |             **Muhammad Aaqil**             |                                                  |
|   [![samarpanB]](https://github.com/samarpanB)   | [![aaqilniz]](https://github.com/aaqilniz) |                                                  |

See
[all contributors](https://github.com/loopbackio/loopback-next/graphs/contributors).

### Alumni

- [@ritch](http://github.com/ritch)
- [@superkhau](https://github.com/superkhau)
- [@rashmihunt](https://github.com/rashmihunt)
- [@kjdelisle](https://github.com/kjdelisle)
- [@virkt25](https://github.com/virkt25)
- [@shimks](https://github.com/shimks)
- [@b-admike](https://github.com/b-admike)
- [@deepakrkris](https://github.com/deepakrkris)
- [@bajtos](http://github.com/bajtos)
- [@jannyhou](http://github.com/jannyHou)
- [@emonddr](https://github.com/emonddr)

## License

[MIT](LICENSE)

[raymondfeng]: https://avatars0.githubusercontent.com/u/540892?v=3&s=60
[ritch]: https://avatars2.githubusercontent.com/u/462228?v=3&s=60
[dhmlau]: https://avatars2.githubusercontent.com/u/25489897?v=3&s=60
[jannyhou]: https://avatars2.githubusercontent.com/u/12554153?v=3&s=60
[hacksparrow]: https://avatars2.githubusercontent.com/u/950112?v=3&s=60
[nabdelgadir]: https://avatars0.githubusercontent.com/u/42985749?v=3&s=60
[marioestradarosa]: https://avatars2.githubusercontent.com/u/4633823?v=3&s=60
[yaty]: https://avatars3.githubusercontent.com/u/11981803?v=3&s=60
[emonddr]: https://avatars0.githubusercontent.com/u/6864736??v=3&s=60
[agnes512]: https://avatars3.githubusercontent.com/u/50331796?v=3&s=60
[deepakrkris]: https://avatars0.githubusercontent.com/u/7688315?v=3&s=60
[derdeka]: https://avatars3.githubusercontent.com/u/13640166?v=3&s=60
[dougal83]: https://avatars0.githubusercontent.com/u/2735881?v=3&s=60
[achrinza]: https://avatars3.githubusercontent.com/u/25147899?v=3&s=60
[frbuceta]: https://avatars2.githubusercontent.com/u/13822438?v=4&s=60
[mschnee]: https://avatars0.githubusercontent.com/u/1375316?v=4&s=60
[madaky]: https://avatars3.githubusercontent.com/u/17172989?v=4&s=60
[nflaig]: https://avatars3.githubusercontent.com/u/38436224?v=4&s=60
[samarpanb]: https://avatars.githubusercontent.com/u/13620435?v=4&s=60
[aaqilniz]: https://avatars.githubusercontent.com/u/25802906?v=4&s=60
 readmeEtag: '"a345b09da793acb478f2d9e663c3863b3356ce15"' readmeLastModified: Fri, 20 Jun 2025 13:44:51 GMT repositoryId: 78452015 description: >- LoopBack makes it easy to build modern API applications that require complex integrations. created: '2017-01-09T17:27:14Z' updated: '2026-02-05T19:27:38Z' language: TypeScript archived: false stars: 5102 watchers: 130 forks: 1066 owner: loopbackio logo: https://avatars.githubusercontent.com/u/22173486?v=4 license: NOASSERTION repoEtag: '"1bee0e3a274d7b7d213b7dbb381cb3f5524f6571923a508d7dc79f62bbf869ab"' repoLastModified: Thu, 05 Feb 2026 19:27:38 GMT foundInMaster: true id: 9132c25ad468bed25b9dbbe3a352abc2 v3: true oldLocations: - https://github.com/strongloop/loopback-next - source: https://openapi.tools/ name: Karate-IDE category: - Converters - Testing - Mock link: https://marketplace.visualstudio.com/items?itemName=KarateIDE.karate-ide repository: https://github.com/zenwave360/karate-ide language: VSCode Extension source_description: >- Generates KarateDSL Tests and Mocks from OpenAPI 3.0 documents and so you can quickly test/explore your API. v3: true v3_1: false id: 3f0c3a748d56c96cc857afa94b46a9e7 repositoryMetadata: base64Readme: >- # Karate IDE

> After one year **KarateIDE have reached Version 1.0.0**.
> The best user experience for KarateDSL, by far!!

KarateIDE is:

-   A **Test Runner/Debugger** and **REST Client** that uses [KarateDSL](https://github.com/karatelabs/karate) to explore your API, import/export from cURL and generate tests/mocks from OpenAPI.
-   **OpenAPI Generator** that generates:
    -   **Karate Tests** you can immediately run, with validation, inline payload examples and scenario outlines for each response code of your API.
    -   **Stateful Mocks**: combining OpenAPI schemas for validation and examples to load mock datasets.
    -   **Mock Validation Tests**: simpler tests to validate your mocks.
    -   **Business Flow Tests**: you can even generate tests that spans multiple API calls reusing generated karate tests

KarateIDE is by far the best user experience for KarateDSL and Contract Testing!!

Getting Started:

-   [karate-openapi-petstore](https://github.com/ZenWave360/karate-openapi-petstore.git) companion project for a complete example with auto-generated tests, mocks, business flow tests and JUnit tests.
-   KarateIDE [Classpath Configuration](#karate-classpath) section, for configuring this vscode extension.
-   Watch Generating KarateDSL tests from OpenAPI definition

[![KarateIDE: Generate KarateDSL Tests from OpenAPI in VSCode](resources/screenshots/generating-tests-from-openapi-youtube-embed.png)](https://www.youtube.com/watch?v=pYyRvly4cG8)

If you are interested on a deep dive into Contract Testing with KarateDSL, checkout:

-   [From Manual to Contract Testing with KarateDSL and KarateIDE](https://medium.com/@ivangsa/from-manual-to-contract-testing-with-karatedsl-and-karateide-i-5884f1732680) white paper in Medium (15 min read)
-   [High Fidelity Stateful Mocks (Consumer Contracts) with OpenAPI and KarateDSL](https://medium.com/@ivangsa/high-fidelity-stateful-mocks-consumer-contracts-with-openapi-and-karatedsl-85a7f31cf84e) Medium article (10 min read).

<!-- TOC -->

- [Karate IDE](#karate-ide)
  - [Features](#features)
    - [Blazing Fast tests Startup](#blazing-fast-tests-startup)
    - [Generate Karate Tests from OpenAPI definitions](#generate-karate-tests-from-openapi-definitions)
    - [Generate Stateful Mocks and Start them from the Editor (or Tests Explorer side bar)](#generate-stateful-mocks-and-start-them-from-the-editor-or-tests-explorer-side-bar)
    - [Generate Simpler Tests for Mock Validation](#generate-simpler-tests-for-mock-validation)
    - [Generate Tests that spans multiple API calls simulating Business Flows](#generate-tests-that-spans-multiple-api-calls-simulating-business-flows)
    - [Generate Karate Project](#generate-karate-project)
    - [SmartPaste sample payload into new files in scenario outline examples](#smartpaste-sample-payload-into-new-files-in-scenario-outline-examples)
    - [Many options to explore your logs and response payloads](#many-options-to-explore-your-logs-and-response-payloads)
    - [Replacing old Tests Explorer with native _Test API_ from VSCode](#replacing-old-tests-explorer-with-native-test-api-from-vscode)
    - [Auto Configuration](#auto-configuration)
    - [OpenAPI schemas and examples meets Karate Mocks](#openapi-schemas-and-examples-meets-karate-mocks)
    - [Debug Karate Scripts](#debug-karate-scripts)
  - [Configuration Options](#configuration-options)
    - [.vscode/launch.json](#vscodelaunchjson)
    - [Karate classpath](#karate-classpath)
      - [Manual configuration: Using karate.jar (Karate fat jar)](#manual-configuration-using-karatejar-karate-fat-jar)
      - [Manual configuration: Using maven repository dependencies](#manual-configuration-using-maven-repository-dependencies)
    - [Run/Debug command templates](#rundebug-command-templates)
    - [Multimodule projects](#multimodule-projects)
  - [Other functionality](#other-functionality)
    - [Karate.env switcher, Karate Options and MockServer Options](#karateenv-switcher-karate-options-and-mockserver-options)
    - [SmartPaste from cURL in Karate files](#smartpaste-from-curl-in-karate-files)
    - [Code Navigation and Definition Peek](#code-navigation-and-definition-peek)
    - [Auto-Completion](#auto-completion)
    - [Kill vscode.KarateTestProcess command](#kill-vscodekaratetestprocess-command)

<!-- /TOC -->

## Features

### Blazing Fast tests Startup

Save a few seconds on each test startup time. With `vscode.KarateTestProcess` we reuse the java process for running your Karate tests and debugging sessions.

While developing, debugging and manual exploring your APIs saving a few seconds on each run makes a huge difference!!

![Karate-IDE](resources/screenshots/KarareIDE-Blazing-Fast.gif)

**Please Note:** that classpath resources like compiled java classes (and also logback.xml) are cached by this java process. If you need to refresh any of those cached resouces just run `KarateIDE: Stop/Kill Karate Tests/Debug Process` from command palette.

If you are experiencing any trouble or want to rollback to standard process just set `karateIDE.karateCli.useKarateTestServer` setting to `false`.

### Generate Karate Tests from OpenAPI definitions

You can generate Karate tests from OpenAPI definitions including one feature per OpenAPI endpoint. Each feature includes four scenarios: one for validation, one for http call, one inline example payload you can edit and run immediately and one scenario outline for each response code.

![Karate-IDE](resources/screenshots/KarateIDE-Generate-Tests.gif)

This is how one autogenerated feature looks like:

![Karate-IDE](resources/screenshots/addPet.feature.png)

### Generate Stateful Mocks and Start them from the Editor (or Tests Explorer side bar)

You can generate stateful mocks from OpenAPI definitions:

![Karate-IDE](resources/screenshots/KarateIDE-Generate-Mocks.gif)

### Generate Simpler Tests for Mock Validation

Validate your mocks with a set of simple generated Karate Tests. Because we delegate payload validation to [ZenWave ApiMock](https://github.com/ZenWave360/zenwave-apimock) OpenAPI wrapper these tests are much simpler than regular E2E tests.

You should also validate your live API with this same set of tests to make sure your mocks are working as expected. See [VerifyMocksTest.java](https://github.com/ZenWave360/karate-openapi-petstore/blob/master/src/test/java/com/petstore/karate/VerifyMocksTest.java) for a JUnit example you can use in your pipeline.

This is how you can generate them from your OpenAPI definition:

![Karate-IDE](resources/screenshots/KarateIDE-Validate-Mocks.gif)

### Generate Tests that spans multiple API calls simulating Business Flows

And yes, you can reuse these karate features, generated from OpenAPI, and compose Business Flow tests with them. Just select in order the API calls you want to chain and Right-Click to select "KarateIDE Generate Business Flow Tests":

![Karate-IDE](resources/screenshots/KarateIDE-Generate-BusinessFlowTest.gif)

This is how an autogenerated CRUD tests looks like (with payloads collapsed for simplicity). Now you only need edit how your data is chained from one call to the other (see also [PetCRUD.feature](https://github.com/ZenWave360/karate-openapi-petstore/blob/master/src/test/resources/apis/PetApi/PetCRUD.feature) for how a complete working example compares to this one):

![Karate-IDE](resources/screenshots/CRUD.png)

### Generate Karate Project

If you are starting from scratch you can generate a fresh karate project base line: with pom.xml, karate-config.js, environment specific and credentials config files...

Just run `KarateIDE: Generate Karate Project` from View > Command Palette

### SmartPaste sample payload into new files in scenario outline examples

Copy sample payloads and paste with Ctrl+Shift+V into scenario outline examples row _filename like_ column and and it will create a new file + example row for you.

![Karate-IDE](resources/screenshots/KarateIDE-Paste-NewRow.gif)

### Many options to explore your logs and response payloads

KarateIDE offers you many options to explore your response data and output logs. Use Executions and Network Logs tree view to explore them.

![Karate-IDE](resources/screenshots/Karate-IDE-data-at-your-fingerprints.gif)

### Replacing old Tests Explorer with native _Test API_ from VSCode

With the new Tests API, Visual Studio Code supports richer displays of outputs and diffs than was previously possible. This brings a lot of goodies to your user experience:

-   Better looking standard Run/Debug showing in the gutter (replacing old _code lenses_)
-   Test Explorer view allows you navigate and filter tests by name, tags, status, etc... To filter by tag use karate prefix in this way `@karate:@yourtag`
-   Failing tests error messages are presented in multiple and useful ways (a tooltips, in the gutter, in the tests explorer...)
-   Start your Mock Server from Editor and Tests Explorer sidebar

We also have replaced _good old_ terminal with Output Channels for greater display flexibility of output logs and response payload.

![Karate-IDE](resources/screenshots/API-Tests-Explorer.gif)

### Auto Configuration

You can configure this extension `classpath setting` installing [KarateIDE Classpath Jar](https://marketplace.visualstudio.com/items?itemName=KarateIDE.karate-classpath-jar) and running `KarateIDE: Configure Classpath` from Command Palette (View > Command Palette or Ctrl+Shift+P).

Karate IDE Classpath Jar will update automatically to latest Karate version.

![Karate-IDE](resources/screenshots/KarateJar-classpath-config.gif)

For further configuration options see [Configuration Section](#configuration-options)

### OpenAPI schemas and examples meets Karate Mocks

You can now:

-   Leverage OpenAPI schemas and examples for request/response validation and declarative stateless mocks.
-   Use KarateDSL for powerful yet simple stateful mocks.
-   Use openapi examples to populate your karate mocks initial data.

Navigate to [ZenWave ApiMock](https://github.com/ZenWave360/zenwave-apimock) for more details about this integration.

### Debug Karate Scripts

You can also Debug Karate scripts inside KarateIDE. Karate Debug Server is **provided by karate-core** and we are also contributors to.

You can:

-   set breakpoints
-   step-by-step debugging
-   navigate scenario call stack with their variables
-   inspect and copy variables, values or their json path expression
-   interactive debug console where you can print, update variable values or test jsonPath expressions
-   hot reloading (with caveats)

https://twitter.com/KarateDSL/status/1167533484560142336

## Configuration Options

### .vscode/launch.json

When you click `Karate Debug` for the first time if `.vscode/launch.js` does not exist one will be created for you with this contents. This is a one time step, after this file is created you can start debugging normally.

```json
{
    "version": "0.2.0",
    "configurations": [
        {
            "type": "karate-ide",
            "name": "Karate IDE (debug)",
            "request": "launch"
        }
    ]
}
```

### Karate classpath

You have currently three options: Install [KarateIDE Classpath Jar extension](https://marketplace.visualstudio.com/items?itemName=KarateIDE.karate-classpath-jar), manually download Karate "fat" jar from [Karate Release](https://github.com/intuit/karate/releases) or reuse local maven repo artifacts.

If unsure just install [KarateIDE Classpath Jar extension](https://marketplace.visualstudio.com/items?itemName=KarateIDE.karate-classpath-jar) as it will automatically upgrade to each latest karate version.

Please use `KarateIDE: Configure Classpath` command from Command Palette (View > Command Palette or Ctrl+Shift+P) for configuring your classpath.

Karate-IDE will honor your classpath settings when autocompletion and navigating/peeking code with `classpath:` prefix.

#### Manual configuration: Using karate.jar (Karate fat jar)

```json
{
    "karateIDE.karateCli.classpath": "src/test/resources;<path to your file>/karate.jar"
}
```

#### Manual configuration: Using maven repository dependencies

If you are already using maven and karate dependencies are already present in your maven local repository. KarateIDE will replace `${m2.repo}` with the value of `${home}/.m2/repository` or `${MAVEN_HOME}/.m2/repository` if `MAVEN_HOME` env variable is available, but you can configure `karateIDE.karateCli.m2Repo` setting to a different folder.

```json
{
    // full classpath example for for karate 1.1.0 version in windows
    "karateIDE.karateCli.classpath": "src/test/java;src/test/resources;target/classes;target/test-classes;${m2.repo}/com/intuit/karate/karate-core/1.1.0/karate-core-1.1.0.jar;${m2.repo}/org/graalvm/js/js-scriptengine/21.2.0/js-scriptengine-21.2.0.jar;${m2.repo}/org/graalvm/sdk/graal-sdk/21.2.0/graal-sdk-21.2.0.jar;${m2.repo}/org/graalvm/js/js/21.2.0/js-21.2.0.jar;${m2.repo}/org/graalvm/regex/regex/21.2.0/regex-21.2.0.jar;${m2.repo}/org/graalvm/truffle/truffle-api/21.2.0/truffle-api-21.2.0.jar;${m2.repo}/com/ibm/icu/icu4j/69.1/icu4j-69.1.jar;${m2.repo}/ch/qos/logback/logback-classic/1.2.3/logback-classic-1.2.3.jar;${m2.repo}/ch/qos/logback/logback-core/1.2.3/logback-core-1.2.3.jar;${m2.repo}/org/slf4j/slf4j-api/1.7.25/slf4j-api-1.7.25.jar;${m2.repo}/org/slf4j/jcl-over-slf4j/1.7.25/jcl-over-slf4j-1.7.25.jar;${m2.repo}/com/jayway/jsonpath/json-path/2.6.0/json-path-2.6.0.jar;${m2.repo}/net/minidev/json-smart/2.4.7/json-smart-2.4.7.jar;${m2.repo}/net/minidev/accessors-smart/2.4.7/accessors-smart-2.4.7.jar;${m2.repo}/org/ow2/asm/asm/9.1/asm-9.1.jar;${m2.repo}/info/cukes/cucumber-java/1.2.5/cucumber-java-1.2.5.jar;${m2.repo}/info/cukes/cucumber-core/1.2.5/cucumber-core-1.2.5.jar;${m2.repo}/org/yaml/snakeyaml/1.29/snakeyaml-1.29.jar;${m2.repo}/de/siegmar/fastcsv/2.0.0/fastcsv-2.0.0.jar;${m2.repo}/info/picocli/picocli/4.6.1/picocli-4.6.1.jar"
}
```

If you need to add **extra classpath jars** you can use `mvn dependency:build-classpath` for generating a compatible extended classpath.

### Run/Debug command templates

For advanced users, Karate-IDE offers _template_ based configurations for both Run and Debug commands. Variables with _${}_ will be replaced by KarateIDE runtime with actual values.

```json
{
    "karateIDE.karateCli.runCommandTemplate": "java '-Dkarate.env=${karateEnv}' '-Dvscode.port=${vscodePort}' -cp '${classpath}' com.intuit.karate.Main ${karateOptions} '${feature}'",
    "karateIDE.karateCli.debugCommandTemplate": "java '-Dkarate.env=${karateEnv}' '-Dvscode.port=${vscodePort}' -cp '${classpath}' com.intuit.karate.Main -d"
}
```

### Multimodule projects

For multimodule project, you may need to configure `karateIDE.multimodule.rootModuleMarkerFile`. Use pom.xml, build.gradle, package.json or any other file that sits on the root of each module.

Karate java process will be started on that folder (first parent folder of current feature file containing a marker file) so classpath will be relative to that folder.

## Other functionality

### Karate.env switcher, Karate Options and MockServer Options

You can switch `karate.env`, Karate options and MockServer options from Executions View title bar. When using Karate-IDE for manual testing or exploring APIs you will find very handy this options switcher.

![Karate-IDE Options Buttons](resources/screenshots/options.png)

### SmartPaste from cURL in Karate files

![Karate-IDE](resources/screenshots/KarateIDE-SmartPaste_From_Curl.gif)

### Code Navigation and Definition Peek

You can navigate between files, features and scenario @tags using `Control-Click` or _peek_ definitions with `Alt+F12`

You can also navigate to scenarios by _@tag_ in the same or in different feature file.

It honors your [classpath](#karate-classpath) setting when navigating to files with `classpath:` prefix.

### Auto-Completion

When reading yml/json files are calling other features you can autocomplete their names with teh list of local and classpath files.

It honors your [classpath](#karate-classpath) setting when navigating to files with `classpath:` prefix.

### Kill vscode.KarateTestProcess command

If you are experiencing trouble with vscode.KarateTestProcess you can always run command `Stop/Kill Karate Tests/Debug Process` to stop a misbehaving process, from `View > Command Palette` or just `Ctrl+P`.

**Enjoy!**
 readmeEtag: '"a7710b0d11187d4baaf42b075671319a77d88c75"' readmeLastModified: Wed, 14 Aug 2024 09:23:39 GMT repositoryId: 318240592 description: The Best OpenSource IDE for KarateDSL. created: '2020-12-03T15:41:05Z' updated: '2025-11-10T16:47:49Z' language: TypeScript archived: false stars: 14 watchers: 3 forks: 5 owner: ZenWave360 logo: https://avatars.githubusercontent.com/u/93670347?v=4 license: NOASSERTION repoEtag: '"73adf228392abbf7ef9e2111a55b47b63df3d8bdc9501f94cc6c35258d757e89"' repoLastModified: Mon, 10 Nov 2025 16:47:49 GMT foundInMaster: true oldLocations: - https://github.com/ivangsa/karate-ide - source: https://openapi.tools/ name: RESTler category: - Security - Testing language: Any repository: https://github.com/microsoft/restler-fuzzer source_description: >- RESTler is the first stateful REST API fuzzing tool for automatically testing cloud services through their REST APIs and finding security and reliability bugs in these services. RESTler analyzes the OpenAPI description of a cloud service, and then generates and executes tests that exercise the service through its REST API. During testing, it checks for specific classes of bugs and dynamically learns how the service behaves from prior service responses. v2: true v3: true id: 90727a339634b533b35035d035eb4316 repositoryMetadata: base64Readme: >- # RESTler

## What is RESTler?

RESTler is the *first stateful REST API fuzzing tool* for automatically
testing cloud services through their REST APIs and finding security and
reliability bugs in these services. For a given cloud service
with an OpenAPI (formerly Swagger) specification, RESTler analyzes its entire specification,
and then generates and executes tests that exercise the service through its REST API.

RESTler intelligently infers producer-consumer dependencies among request types from the OpenAPI definition.
During testing, it checks for specific classes of bugs
and dynamically learns how the service behaves from prior service responses.
This intelligence allows RESTler to explore deeper service states
reachable only through specific request sequences and to find more bugs.

RESTler is described in these peer-reviewed research papers:

1. [RESTler: Stateful REST API Fuzzing](https://patricegodefroid.github.io/public_psfiles/icse2019.pdf) (ICSE'2019)
2. [Checking Security Properties of Cloud Service REST APIs](https://patricegodefroid.github.io/public_psfiles/icst2020.pdf) (ICST'2020)
3. [Differential Regression Testing for REST APIs​](https://patricegodefroid.github.io/public_psfiles/issta2020.pdf) (ISSTA'2020)
4. [Intelligent REST API Data Fuzzing​​](https://patricegodefroid.github.io/public_psfiles/fse2020.pdf) (FSE'2020)

If you use RESTler in your research, please cite the (default) ICSE'2019 paper ([BibTeX](./docs/user-guide/icse2019.bib)).

RESTler includes multiple test generation strategies.  In order to get a comprehensive comparative view w.r.t. to (i) efficiency 
(i.e., how quickly can RESTler find crashes) and (ii) effectiveness 
(i.e., how many crashes can RESTler find in a give time frame), 
we recommend comparing against all documented `fuzzing_mode(s)`
because each one provides a different trade-off between breadth and depth of state space exploration.
We also recommend running `test` mode before any fuzzing, as described below, 
to discover and fix setup issues (e.g. adding required pre-requisite parameter values to the dictionary) prior to fuzzing.

RESTler was created at Microsoft Research and is still under active development.

For an overview and demo on how to get started, see [Webinar - Fuzzing to Improve the Security and Reliability of Cloud Services](https://www.youtube.com/watch?v=FYmiPoRwEbE).

![RESTler architecture](./docs/user-guide/RESTler-arch.png)


## Build Status

[![Build Status](https://dev.azure.com/ms/restler-fuzzer/_apis/build/status/restler-build-validation?branchName=main)](https://dev.azure.com/ms/restler-fuzzer/_build/latest?definitionId=422&branchName=main)


## Setting up RESTler

RESTler was designed to run on 64-bit machines with Windows or Linux.  Experimental support for macOS is also enabled.

### **Build instructions**

#### Docker

In the root of this repo, run

```shell
docker build -t restler .
```

The resulting docker container will have RESTler available in directory `/RESTler/restler` with main binary `Restler`. 

You can then use this docker image as basis to add the application under test to execute fuzzing inside isolated docker containers.

#### Local

Prerequisites: Install [Python 3.12.8](https://www.python.org/downloads/) and
[.NET 8.0](https://dotnet.microsoft.com/download/dotnet-core?utm_source=getdotnetcorecli&utm_medium=referral), for your appropriate OS.

Create a directory where you'd like to place the RESTler binaries:

```mkdir restler_bin```

Switch to the repo root directory and run the following Python script:

```python ./build-restler.py --dest_dir <full path to restler_bin above>```

Note: if you get nuget error NU1403 when building, a quick workaround is to clear your cache with this command

```dotnet nuget locals all --clear```


## Using RESTler

RESTler runs in 4 main modes (in order):

1. **Compile:** from an OpenAPI JSON or YAML definition (and optionally examples), generate a RESTler grammar. See [Compiling](./docs/user-guide/Compiling.md).
2. **Test:** execute quickly all of the endpoints+methods in a compiled RESTler grammar for debugging the test setup and compute what parts of the OpenAPI definition are covered. This mode is also called a *smoketest*.
See [Testing](./docs/user-guide/Testing.md). To use custom test engine settings, see [Test Engine Settings](./docs/user-guide/SettingsFile.md).
3. **Fuzz-lean:** execute once every endpoint+method in a compiled RESTler grammar with a default set of checkers to see if bugs can be found quickly. See [Fuzzing](./docs/user-guide/Fuzzing.md).
4. **Fuzz:** bug hunting - explore a RESTler fuzzing grammar in smart breadth-first-search mode (deeper search mode) for finding more bugs.
**Warning:** This type of fuzzing is more aggressive and may create outages in the service under test if the service is poorly implemented (e.g., fuzzing might create resource leaks, perf degradation, backend corruptions, etc.).
See [Fuzzing](./docs/user-guide/Fuzzing.md).

## Quick Start

For a quick intro with simple examples, see this [Tutorial](./docs/user-guide/TutorialDemoServer.md).

To quickly try RESTler on your API, see [Quick Start](./docs/user-guide/QuickStart.md).

## Bugs found by RESTler
There are currently two categories of bugs found by RESTler.

- **Error code**: currently, any time a response with status code ```500``` ("Internal Server Error") is received, a bug is reported.
- **Checkers**: each checker tries to trigger specific bugs by executing targeted additional requests or sequences of requests at certain points during fuzzing, determined by context.  Some checkers try to find additional 500s, while other checkers try to find specific logic bugs such as resource leaks or hierarchy violations.  For a full description of checkers, see [Checkers](./docs/user-guide/Checkers.md).

When a bug is found, RESTler reports bugs triaged in bug buckets, and provides a replay log that can be used to reproduce the bug (see [Replay](./docs/user-guide/Replay.md)).


## Advanced Topics

For tips on using RESTler effectively, please see [Best Practices](./docs/user-guide/BestPractices.md) and [Improving API Coverage](./docs/user-guide/ImprovingCoverage.md).

See also these [Frequently Asked Questions](./docs/user-guide/FAQ.md).

If you're interested in using RESTler at scale as part of your CI/CD pipeline, check out
the [REST API Fuzz Testing self-hosted service](https://github.com/microsoft/rest-api-fuzz-testing).

## Questions

If you have a request/suggestion/question, please file an issue.
See [Contributing.md](./docs/contributor-guide/Contributing.md) for instructions.

## Contributing

This project welcomes contributions and suggestions. Most contributions require you to
agree to a Contributor License Agreement (CLA) declaring that you have the right to,
and actually do, grant us the rights to use your contribution. For details, visit
https://cla.microsoft.com.

When you submit a pull request, a CLA-bot will automatically determine whether you need
to provide a CLA and decorate the PR appropriately (e.g., label, comment). Simply follow the
instructions provided by the bot. You will only need to do this once across all repositories using our CLA.

This project has adopted the [Microsoft Open Source Code of Conduct](https://opensource.microsoft.com/codeofconduct/).
For more information see the [Code of Conduct FAQ](https://opensource.microsoft.com/codeofconduct/faq/)
or contact [opencode@microsoft.com](mailto:opencode@microsoft.com) with any additional questions or comments.

For more information, see [Contributing.md](./docs/contributor-guide/Contributing.md).

## Trademarks

This project may contain trademarks or logos for projects, products, or services. Authorized use of Microsoft trademarks or logos is subject to and must follow Microsoft's Trademark & Brand Guidelines. Use of Microsoft trademarks or logos in modified versions of this project must not cause confusion or imply Microsoft sponsorship. Any use of third-party trademarks or logos are subject to those third-party's policies.

## Data collection

The software may collect information about you and your use of the software and send it to Microsoft. Microsoft may use this information to provide services and improve our products and services. You may turn off the telemetry as described in the repository. There are also some features in the software that may enable you and Microsoft to collect data from users of your applications. If you use these features, you must comply with applicable law, including providing appropriate notices to users of your applications together with a copy of Microsoft's privacy statement. Our privacy statement is located at https://go.microsoft.com/fwlink/?LinkID=824704. You can learn more about data collection and use in the help documentation and our privacy statement. Your use of the software operates as your consent to these practices.

For more information, see [Telemetry.md](./docs/user-guide/Telemetry.md).

## Reporting Security Issues

Security issues and bugs should be reported privately, via email, to the
Microsoft Security Response Center (MSRC) at
[secure@microsoft.com](mailto:secure@microsoft.com). You should receive a
response within 24 hours. If for some reason you do not, please follow up via
email to ensure we received your original message. Further information,
including the [MSRC PGP](https://technet.microsoft.com/en-us/security/dn606155)
key, can be found in the [Security TechCenter](https://technet.microsoft.com/en-us/security/default).

For additional details, see [Security.md](SECURITY.md).


# Privacy & Cookies

https://go.microsoft.com/fwlink/?LinkId=521839
 readmeEtag: '"ebc06359596abda3a0ad485b60c0529ca58447d7"' readmeLastModified: Mon, 13 Oct 2025 23:15:45 GMT repositoryId: 282320787 description: >- RESTler is the first stateful REST API fuzzing tool for automatically testing cloud services through their REST APIs and finding security and reliability bugs in these services. created: '2020-07-24T21:40:11Z' updated: '2026-02-06T00:05:02Z' language: Python archived: false stars: 2867 watchers: 37 forks: 324 owner: microsoft logo: https://avatars.githubusercontent.com/u/6154722?v=4 license: MIT repoEtag: '"613f635bfe4ff68035f421cd7ed6655a0b67f75537a26dae75930622e575c55c"' repoLastModified: Fri, 06 Feb 2026 00:05:02 GMT foundInMaster: true oldLocations: - https://github.com/microsoft/restler - source: openapi3 tags repository: https://github.com/xiaoymin/swagger-bootstrap-ui v3: true id: 0ccf3516a99b6fdb818a427fe48ed72c repositoryMetadata: base64Readme: >- Cgpgc3dhZ2dlci1ib290c3RyYXAtdWlg6aG555uu5bey57uP5pu05ZCN5Li6S25pZmU0au+8jOWkp+WutuWOu+aWsOS7k+W6k+iOt+WPluacgOaWsOS7o+egge+8jOaIluiAheaPkGlzc3Vlc+OAgXByIAoK5paw5LuT5bqT5Zyw5Z2A77yaW2h0dHBzOi8vZ2l0aHViLmNvbS94aWFveW1pbi9rbmlmZTRqXShodHRwczovL2dpdGh1Yi5jb20veGlhb3ltaW4va25pZmU0aikgCg== readmeEtag: '"0a2482f90afe381c6c63863e71c415e3dd5f4b97"' readmeLastModified: Fri, 24 Feb 2023 06:40:18 GMT repositoryId: 528459414 description: Knife4j is a set of Swagger2 and OpenAPI3 All-in-one enhancement solution created: '2022-08-24T14:26:52Z' updated: '2025-11-13T05:49:17Z' language: HTML archived: false stars: 96 watchers: 2 forks: 21 owner: xiaoymin logo: https://avatars.githubusercontent.com/u/7894406?v=4 license: Apache-2.0 repoEtag: '"e50116ab16b2a4892222edfcd81810521e69f7474887b24da01a9d8d230f3fe4"' repoLastModified: Thu, 13 Nov 2025 05:49:17 GMT category: - Low-level Tooling - SDK foundInMaster: true - source: openapi3 tags repository: https://github.com/specify/open_api_tools v3: true repositoryMetadata: base64Readme: >- # OpenAPI Testing Framework

A framework for testing an API based on its OpenAPI schema.

This framework allows to create anything between a manual test and a fully
automated test.

## Installation

Install Python (versions 3.6-3.9 where tested to work)

Clone this repository

```bash
git clone https://github.com/specify/open_api_tools
cd open_api_tools
```

Configure a virtual environment

```bash
python -m venv venv
```

Install the dependencies

```bash
./venv/bin/pip install -r requirements.txt
```

Install this package locally

```bash
pip install -e .
```

## Usage

There are three main use cases, each going from more automated to more manual.

The most automated is the `full_test`, which, by default, generates test URLs
with some parameters based on OpenAPI schema, then sends those requests and
makes sure that responses match the schema definition. By default, this method
only tests `GET` endpoints and it does not generate request body object.
However, this can be changed by providing additional parameters. Also, you can
define parameter constraints (e.x if 'a' is set to True, then response must
contain 'b') to further improve the quality of this test.

Next, there is a Chain test that allows to test a chain of requests and make
sure that each request correctly influenced the response of the next request.

Finally, for those that need complete control, there is a `make_request` method
which facilitates validating the request parameters, sending a single request,
validating the response parameters and returning the result.

## Full test

The handler function should return a boolean value saying validating
whether the response object is as expected

### Basic usage

Run the test

```python
from open_api_tools.test.full_test import full_test
from open_api_tools.common.load_schema import load_schema

schema = load_schema('open_api.yaml')

# Error message schema is defined in open_api_tools.validate.index
def after_error_occurred(*error_message):
    print(error_message)

full_test(
    schema=schema,
    max_urls_per_endpoint=50,
    failed_request_limit=10,
    after_error_occurred=after_error_occurred,
)
```

This script would automatically generate test URLs based on
your API schema.

All requests would be sent to the first server
specified in the `servers` part of the API schema.

`max_urls_per_endpoint` parameter defines the limit of queries to send to a
single endpoint.

`failed_request_limit` makes sure that the `full_test` function quits with an
exception if a certain number of requests failed validation. This is useful for
preventing needless server load when all requests are failing for the same
reason.

### Supplying test values for parameters

By default, the test reads the `examples` object
[in the schema](https://swagger.io/specification/#example-object) to generate
request parameters. If `examples` weren't provided, it would try to create some
test values based on the parameter type.

If you would like more customization, an optional `after_examples_generated`
hook can be provided to the `full_test` method.

`after_examples_generated` must be a function that accepts an endpoint name as
the first parameter and
[the parameter object](https://swagger.io/specification/#parameter-object)
as the second parameter (the parameter object would vary depending on how it is
defined in your schema). In turn, the function must return a list of valid
examples.

Example usage:

```python
from open_api_tools.test.full_test import full_test
from open_api_tools.common.load_schema import load_schema

schema = load_schema('open_api.yaml')

def after_error_occurred(*error_message):
    print(error_message)

def after_examples_generated(
    endpoint_name,
    parameter,
    autogenerated_examples
):
    if endpoint_name == '/api/posts/' and parameter.name == 'post_id':
        return [1, 2, 3, 4, 5]
    elif parameter.schema.type == 'string':
        # Some naughty strings
        return ["ÅÍÎÏ˝ÓÔÒÚÆ☃", "Ω≈ç√∫˜µ≤≥÷", "⅛⅜⅝⅞"]
    else:
        return autogenerated_examples

full_test(
    schema=schema,
    max_urls_per_endpoint=50,
    failed_request_limit=10,
    after_error_occurred=after_error_occurred,
    after_examples_generated=after_examples_generated
)
```

The third parameter in `after_examples_generated` is the list of examples that
were generated by this framework. If you don't want to change the generated
examples, the function can return back this value.

Note that `after_examples_generated` would also get called with
`requestBody` as a parameter.name. This allows you to to provide a list of
request objects that would be used in testing. Each request object should be of
type `(str, str)`, where the first string is the MIME type and the second one is
the serialized payload that would be send with the request.

### Handling authentication and amending the request object

`full_test` also supports a `before_request_send` hook that allows you to modify
the request object before a request is sent. This is useful if you want to edit
the headers or add authentication cookies.

Example usage:

```python
from open_api_tools.test.full_test import full_test
from open_api_tools.common.load_schema import load_schema

schema = load_schema('open_api.yaml')

def before_request_send(endpoint_name, request_object):
    if endpoint_name == '/api/main/{id}/':
        if request_object.headers is None:
            request_object.headers = {}
        request_object.headers['Authorization']='Basic YWxhZGRpbjpvcGVuc2VzYW1l'
    return request_object

full_test(
    schema=schema,
    max_urls_per_endpoint=50,
    failed_request_limit=10,
    before_request_send=before_request_send
)
```

The schema for the request object [is defined
here](https://docs.python-requests.org/en/master/api/).

### Defining parameter constrains

If response object depends on the query parameters, you can
test for these relationships by adding your parameter names
and handler functions to the `parameter_constraints` dictionary and passing it
to `full_test`.

Each handler function would receive the following arguments:

* parameter_value (any): the value of the parameter this handler
  works with
* path (str): name of the current endpoint (useful if the same
  parameter is shared between multiple endpoints)
* response (any): [response
  object](https://docs.python-requests.org/en/master/api/#requests.Response).

Example usage:

```python
from open_api_tools.test.full_test import full_test
from open_api_tools.common.load_schema import load_schema

schema = load_schema('open_api.yaml')

def get_popular_posts(
    parameter_value: int,
    endpoint: str,
    response: object
):
    for post in response.json():
        if post.popularity < parameter_value:
            raise Exception(
                f'{endpoint} failed to filter the posts by popularity'
            )

full_test(
    schema=schema,
    max_urls_per_endpoint=50,
    failed_request_limit=10,
    parameter_constraints={

    },
)
```

## Chain test

For more fine-grained testing, there is a `chain` method that allows to test
a chain of request URLs with request/response object validation and assurance
that each request produced expected results.

Example usage:

```python
from open_api_tools.test.chain import chain, Request, Validate
from open_api_tools.common.load_schema import load_schema
import json

schema = load_schema('open_api.yaml')

post_id = 345

def create_post(_arguments, _response, _previous_values):
    return {
        "requestBody": [
            'application/json',
            json.dumps(dict(
                id=post_id,
                name='Post Name',
                body='Post content'
            ))
        ]
    }

chain(
    schema=schema,
    definition=[
        Request(method='GET', endpoint='/api/posts/'),
        Validate(
            validate=\
                lambda response: post_id not in response.json().posts
        ),
        Request(
            method='POST',
            endpoint='/api/posts/',
            parameters=create_post
        ),
        Validate(
            validate=lambda response: post_id in response.json().posts
        ),
        Request(
            method='DELETE',
            endpoint='/api/posts/',
            parameters={'id': post_id}
        ),
        Validate(
            validate= \
                lambda response: post_id not in response.json().posts
        ),
    ],
)
```

The response object for the validation function [is described
here](https://docs.python-requests.org/en/master/api/#requests.Response).

Keep in mind that the `endpoint` string in the `Request` class should be
identical to one of the endpoints in your OpenAPI schema. For example, you
should specify `/api/user/{user_id}/` instead of `/api/user/1/`. Both path
parameters and query parameters should be supplied in the `parameters`
dictionary or returned by the `parameters` function. If `parameters` is a
function, it would get called with these arguments:
(list_of_parameter_objects, previous_response, previous_parameter_values).
Alternatively, you can omit the `parameters` key altogether if the endpoint
doesn't expect any.

Also, `Request` can omit `parameters` if there aren't any to define.
Alternatively, you can supply a dictionary, or a function that would get called
with three arguments:
(list_of_parameter_objects, previous_response, previous_parameter_values).

Additionally, you can
supply a `requestBody` parameter. Unlike most parameters, `requestBody` must be
a `Tuple[str,str]` where the first string is a MIME type and the second string
is a serialized version of the request body.

The `Validate` class expects a function that takes a response object and
returns a boolean saying whether a value is valid. On false, the chain stops.
Note, if `Validate` returned false, an exception is not thrown, but you can
throw one on your own if you need to.

The `chain` method also accepts a `before_request_send` parameter, which is
described in detail in [the previous section
](#handling-authentication-and-amending-the-request-object)

## Manual test

`make_request` method is most useful when you need complete control over the
requests that get send, but still need the assurance that request/response
objects confirm to schema.

Example usage:

```python
from open_api_tools.validate.index import make_request
from open_api_tools.common.load_schema import load_schema
import json

schema = load_schema('open_api.yaml')

response = make_request(
  schema=schema,
  request_url='http://localhost/api/posts/1/?update_indexes=true',
  endpoint_name='/api/posts/<post_id>',
  method='POST',
  body=('application/json', json.dumps({"name": 'New post name'})),
)

```

For additional control, the `make_request` method also accepts a
`before_request_send` parameter, which is described in detail in [the previous
section](#handling-authentication-and-amending-the-request-object).

If you want to only verify the request object, or want to execute some
additional code before executing the request, the `make_request` can be
broken down into `prepare_request` and `file_request` methods. They are defined
in `open_api_tools/validate/index.py`.
 readmeEtag: '"a7af0734c202b1529b5714e250e6704e83ad128d"' readmeLastModified: Sun, 03 Dec 2023 21:12:15 GMT repositoryId: 348397827 description: 'A collection of useful tools powered by Open API schema ' created: '2021-03-16T15:25:48Z' updated: '2025-12-09T08:18:12Z' language: Python archived: false stars: 1 watchers: 1 forks: 0 owner: specify logo: https://avatars.githubusercontent.com/u/2906014?v=4 license: MIT repoEtag: '"534ee95be1e68fa9138e22bfefb4a1cacba759a7d8fd8cabbfb9086717162288"' repoLastModified: Tue, 09 Dec 2025 08:18:12 GMT foundInMaster: true category: Server Implementations id: c1acd58744726314a1310f44952bae71 - source: openapi3 tags repository: https://github.com/cmwylie19/gloo-portal-demo v3: true repositoryMetadata: base64Readme: >- IyBJbnN0YWxsIEdsb28gUG9ydGFsCgojIyBBZGQgdGhlIGhlbG0gcmVwbwpgYGAKaGVsbSByZXBvIGFkZCBkZXYtcG9ydGFsIGh0dHBzOi8vc3RvcmFnZS5nb29nbGVhcGlzLmNvbS9kZXYtcG9ydGFsLWhlbG0KaGVsbSByZXBvIHVwZGF0ZQpgYGAKCiMjIENyZWF0ZSBoZWxtIHZhbHVlcyBvdmVycmlkZQpgYGAKY2F0IDw8IEVPRiA+IGdsb28tdmFsdWVzLnlhbWwKZ2xvbzoKICBlbmFibGVkOiB0cnVlICMgRW5hYmxlcyBpbnRlZ3JhdGlvbiB3aXRoIEdsb28gRWRnZSBFbnRlcnByaXNlCmxpY2Vuc2VLZXk6CiAgc2VjcmV0UmVmOgogICAgbmFtZTogbGljZW5zZQogICAgbmFtZXNwYWNlOiBnbG9vLXN5c3RlbQogICAga2V5OiBsaWNlbnNlLWtleQpFT0YKYGBgCgojIyBDcmVhdGUgdGhlIG5hbWVzcGFjZSBhbmQgaW5zdGFsbCB0aGUgaGVsbSBjaGFydApgYGAKayBjcmVhdGUgbnMgZGV2LXBvcnRhbApoZWxtIGluc3RhbGwgZGV2LXBvcnRhbCBkZXYtcG9ydGFsL2Rldi1wb3J0YWwgLW4gZGV2LXBvcnRhbCAtLXZhbHVlcyBnbG9vLXZhbHVlcy55YW1sCmBgYA== readmeEtag: '"b96bbff7b3b3226566129df502833bc7785e80bf"' readmeLastModified: Wed, 30 Jun 2021 13:16:07 GMT repositoryId: 381358694 description: >- Tasks backend service with OpenAPI v3 doc that stores tasks in NoSQL database. created: '2021-06-29T12:34:52Z' updated: '2021-06-30T13:16:16Z' language: Go archived: false stars: 0 watchers: 0 forks: 0 owner: cmwylie19 logo: https://avatars.githubusercontent.com/u/1096507?v=4 repoEtag: '"6f94fb7f16bf4d8dfbaea6736c8501c34976229ea52c86f606de85ea4a1c88b6"' repoLastModified: Wed, 30 Jun 2021 13:16:16 GMT foundInMaster: true category: - Documentation - Server Implementations id: bca57bb7f6cf16e67cd66a8722405ea1 - source: openapi3 tags repository: https://github.com/satabratapaul-gitac/managex v3: true repositoryMetadata: base64Readme: >- IyBNYW5hZ2VYCkFwcGxpY2F0aW9uIHdoaWNoIGhlbHBzIGFuIEhSIG9yIE1hbmFnZXIgLCB0byBtYW5hZ2UgdGhlIGRldGFpbHMgYW5kIGluZm9ybWF0aW9uIGZvciB2YXJpb3VzIGVtcGxveWVlcyBvZiBhbiBvcmdhbml6YXRpb24KCkl0IHByb3ZpZGVzIHRoZXNlIGZ1bmN0aW9uYWxpdGllczoKClJlZ2lzdHJhdGlvbiBvZiBhIFVzZXIgKEhSIG9yIE1hbmFnZXIpPGJyIC8+CkxvZ2dpbmcgSW4gdGhlIHVzZXI8YnIgLz4KRmV0Y2hpbmcgYWxsIEVtcGxveWVlIERldGFpbHM8YnIgLz4KQWRkaW5nIGEgTmV3IEVtcGxveWVlIHRvIHRoZSBzeXN0ZW08YnIgLz4KVXBkYXRpbmcgYW4gRXhpc3RpbmcgRW1wbG95ZWUncyBkZXRhaWxzPGJyIC8+CkRlbGV0aW5nIGFuIEV4aXN0aW5nIEVtcGxveWVlPGJyIC8+CkxvZ2lnbmcgT3V0IHRoZSBVc2VyPGJyIC8+Cg== readmeEtag: '"5025a0e7c38052d84dffac24d4d085a19209be01"' readmeLastModified: Sat, 29 Jan 2022 09:44:14 GMT repositoryId: 452796275 description: >- Application which helps an HR or Manager , to manage the details and information for various employees of an organization created: '2022-01-27T18:22:35Z' updated: '2022-02-11T08:36:53Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 2 owner: SatabrataPaul-GitAc logo: https://avatars.githubusercontent.com/u/69891965?v=4 repoEtag: '"26f342b0f2955f5f8b352c4a378de9315f33907314ae1fa31754594f4ef3e351"' repoLastModified: Fri, 11 Feb 2022 08:36:53 GMT foundInMaster: true category: - Parsers - Server Implementations id: 377aa09b014636fdf56671807bdf027b - source: openapi3 tags repository: https://github.com/weikangchia/op-converter v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIDMuMCB0byBQb3N0bWFuIDIuMSBDb2xsZWN0aW9ucyBDb252ZXJ0ZXIKCioqb3AtY29udmVydGVyKiogaXMgYSBjdXN0b21pemFibGUgY29udmVydGVyIGZvciBPcGVuQVBJIDMuMCB0byBQb3N0bWFuIDIuMSBDb2xsZWN0aW9ucy4KCiMjIEdldHRpbmcgU3RhcnRlZAoKIyMjIFByZS1yZXF1aXNpdGVzCgotIEluc3RhbGwgW25vZGVdKGh0dHBzOi8vbm9kZWpzLm9yZy9lbi9kb3dubG9hZC8pCi0gSW5zdGFsbCBvcC1jb252ZXJ0ZXIKICBgYGAKICBucG0gaW5zdGFsbCAtZyBvcC1jb252ZXJ0ZXIKICBgYGAKCiMjIyBVc2FnZQpgYGBzaC1zZXNzaW9uClVTQUdFOgogICAgb3AtY29udmVydGVyCgpPUFRJT05TOgogICAgLSBmLCAtLW9wZW5BcGlGaWxlPW9wZW5BcGlGaWxlICAgIHBhdGggdG8geW91ciBPcGVuQVBJIDMuMCBKU09OIGZpbGUKICAgIC0gaCwgLS1oZWxwICAgICAgICAgICAgICAgICAgICAgICBzaG93IENMSSBoZWxwCiAgICAtIG4sIC0tbmFtZT1uYW1lICAgICAgICAgICAgICAgICAgbmFtZSBmb3IgeW91ciBwb3N0bWFuIGNvbGxlY3Rpb24KICAgIC0tYmFzZVVybD1iYXNlVXJsICAgICAgICAgICAgICAgICBiYXNlIHVybAogICAgLS1jb25maWc9Y29uZmlnICAgICAgICAgICAgICAgICAgIGN1c3RvbSBjb25maWcgZmlsZSBuYW1lIChkZWZhdWx0IGlzIGNvbmZpZy5qc29uKQpgYGAKCioqRXhhbXBsZXMqKgpgYGAKb3AtY29udmVydGVyIC1mIC4vb3BlbmFwaS5qc29uIC1uICJNeSBDb2xsZWN0aW9uIiAtLWJhc2VVcmw9Imh0dHA6Ly9sb2NhbGhvc3QiID4gcG9zdG1hbi5qc29uCmBgYAoKIyMgQ29uZmlndXJhdGlvbiBPcHRpb25zCgpCeSBkZWZhdWx0ICoqb3AtY29udmVydGVyKiogcmVhZHMgYWxsIGNvbmZpZ3VyYWJsZSBvcHRpb25zIGZyb20gYGNvbmZpZy5qc29uYCBhdCB0aGUgZm9sbG93aW5nIGxvY2F0aW9uCgpgYGAKVW5peDogfi8uY29uZmlnL29wLWNvbnZlcnRlcgpXaW5kb3dzOiAlTE9DQUxBUFBEQVRBJVxnaXRjZwpgYGAKCkJlbG93IGFyZSB0aGUgYXZhaWxhYmxlIGNvbmZpZ3VyYWJsZSBvcHRpb25zLgoKLSBbcGF0aF0oI3BhdGgpCiAgLSBbZW5hYmxlUmVwbGFjZVByZWZpeF0oI2VuYWJsZVJlcGxhY2VQcmVmaXgpCiAgLSBbcmVwbGFjZVByZWZpeF0oI3JlcGxhY2VQcmVmaXgpCiAgLSBbcmVwbGFjZVByZWZpeFdpdGhdKCNyZXBsYWNlUHJlZml4V2l0aCkKLSBbYXV0aF0oI2F1dGgpCi0gW2V2ZW50c10oI2V2ZW50cykKCiMjIyBwYXRoCgp8IE5hbWUgICAgICAgIHwgVmFsdWUgICAgICAgICAgIHwKfCAtLS0tLS0tLS0tLS0tIHwtLS0tLS0tLS0tLS0tfAp8IHR5cGUgICAgICB8IG9iamVjdCB8CnwgbWFuZGF0b3J5IHwgZmFsc2UgfAoKRXhhbXBsZQpgYGBqc29uCnsKICAicGF0aCI6IHt9Cn0KYGBgCgojIyMjIGVuYWJsZVJlcGxhY2VQcmVmaXgKCgp8IE5hbWUgICAgICAgIHwgVmFsdWUgICAgICAgICAgIHwKfCAtLS0tLS0tLS0tLS0tIHwtLS0tLS0tLS0tLS0tfAp8IHR5cGUgICAgICB8IGJvb2xlYW4gfAp8IHBhcmVudCAgICB8IHBhdGggfAp8IG1hbmRhdG9yeSB8IGZhbHNlIHwKfCBtYW5kYXRvcnkgc2libGluZ3MgfCByZXBsYWNlUHJlZml4LCByZXBsYWNlUHJlZml4V2l0aCB8CgpFeGFtcGxlCmBgYGpzb24KewogICJwYXRoIjogewogICAgImVuYWJsZVJlcGxhY2VQcmVmaXgiOiB0cnVlLAogICAgInJlcGxhY2VQcmVmaXgiOiAiL3YxIiwKICAgICJyZXBsYWNlUHJlZml4V2l0aCI6ICIiCiAgfQp9CmBgYAoKIyMjIyByZXBsYWNlUHJlZml4CgoKfCBOYW1lICAgICAgICB8IFZhbHVlICAgICAgICAgICB8CnwgLS0tLS0tLS0tLS0tLSB8LS0tLS0tLS0tLS0tLXwKfCB0eXBlICAgICAgfCBzdHJpbmcgfAp8IHBhcmVudCAgICB8IHBhdGggfAp8IG1hbmRhdG9yeSB8IGZhbHNlIHwKfCBtYW5kYXRvcnkgc2libGluZ3MgfCBlbmFibGVSZXBsYWNlUHJlZml4LCByZXBsYWNlUHJlZml4V2l0aCB8CgpFeGFtcGxlCmBgYGpzb24KewogICJwYXRoIjogewogICAgImVuYWJsZVJlcGxhY2VQcmVmaXgiOiB0cnVlLAogICAgInJlcGxhY2VQcmVmaXgiOiAiL3YxIiwKICAgICJyZXBsYWNlUHJlZml4V2l0aCI6ICIiCiAgfQp9CmBgYAoKIyMjIyByZXBsYWNlUHJlZml4V2l0aAoKCnwgTmFtZSAgICAgICAgfCBWYWx1ZSAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0gfC0tLS0tLS0tLS0tLS18CnwgdHlwZSAgICAgIHwgc3RyaW5nIHwKfCBwYXJlbnQgICAgfCBwYXRoIHwKfCBtYW5kYXRvcnkgfCBmYWxzZSB8CnwgbWFuZGF0b3J5IHNpYmxpbmdzIHwgZW5hYmxlUmVwbGFjZVByZWZpeCwgcmVwbGFjZVByZWZpeCB8CgpFeGFtcGxlCmBgYGpzb24KewogICJwYXRoIjogewogICAgImVuYWJsZVJlcGxhY2VQcmVmaXgiOiB0cnVlLAogICAgInJlcGxhY2VQcmVmaXgiOiAiL3YxIiwKICAgICJyZXBsYWNlUHJlZml4V2l0aCI6ICIiCiAgfQp9CmBgYAoKIyMjIGF1dGgKClBvc3RtYW4gQXV0aCBmaWVsZAoKfCBOYW1lICAgICAgICB8IFZhbHVlICAgICAgICAgICB8CnwgLS0tLS0tLS0tLS0tLSB8LS0tLS0tLS0tLS0tLXwKfCB0eXBlICAgICAgfCBvYmplY3QgfAp8IG1hbmRhdG9yeSB8IGZhbHNlIHwKCkV4YW1wbGUKYGBganNvbgp7CiAgImF1dGgiOiB7CiAgICAidHlwZSI6ICJiZWFyZXIiLAogICAgImJlYXJlciI6IFsKICAgICAgewogICAgICAgICJrZXkiOiAidG9rZW4iLAogICAgICAgICJ2YWx1ZSI6ICJ7e0NVUlJFTlRfQUNDRVNTX1RPS0VOfX0iLAogICAgICAgICJ0eXBlIjogInN0cmluZyIKICAgICAgfQogICAgXQogIH0KfQpgYGAKCiMjIyBldmVudHMKClBvc3RtYW4gZXZlbnQgZmllbGQKCnwgTmFtZSAgICAgICAgfCBWYWx1ZSAgICAgICAgICAgfAp8IC0tLS0tLS0tLS0tLS0gfC0tLS0tLS0tLS0tLS18CnwgdHlwZSAgICAgIHwgYXJyYXkgfAp8IG1hbmRhdG9yeSB8IGZhbHNlIHwKCkV4YW1wbGUKYGBganNvbgp7CiAgImV2ZW50cyI6IFsKICAgIHsKICAgICAgImxpc3RlbiI6ICJ0ZXN0IiwKICAgICAgInNjcmlwdCI6IHsKICAgICAgICAiZXhlYyI6IFsKICAgICAgICAgICIiCiAgICAgICAgXSwKICAgICAgICAidHlwZSI6ICJ0ZXh0L2phdmFzY3JpcHQiCiAgICAgIH0KICAgIH0KICBdCn0KYGBgCgojIyBMaWNlbnNlCgpEaXN0cmlidXRlZCB1bmRlciB0aGUgTUlUIExpY2Vuc2UuIFNlZSBbTElDRU5TRV0obGljZW5zZSkgZm9yIG1vcmUgaW5mb3JtYXRpb24uCgojIyBDb250cmlidXRpbmcKQ29udHJpYnV0aW9ucyBhcmUgd2hhdCBtYWtlIHRoZSBvcGVuIHNvdXJjZSBjb21tdW5pdHkgc3VjaCBhbiBhbWF6aW5nIHBsYWNlIHRvIGJlIGxlYXJuLCBpbnNwaXJlLCBhbmQgY3JlYXRlLiBBbnkgY29udHJpYnV0aW9ucyB5b3UgbWFrZSBhcmUgZ3JlYXRseSBhcHByZWNpYXRlZC4KCiMjIENvbnRhY3QKCldlaSBLYW5nIC0gd2Vpa2FuZ2NoaWFbQF1nbWFpbC5jb20K readmeEtag: '"3f1969736fa631639b4b6f1f49b0fb3f9518e7a2"' readmeLastModified: Sun, 24 Apr 2022 11:05:58 GMT repositoryId: 392223452 description: Customizable Converter for OpenAPI 3.0 to Postman 2.1 Collections created: '2021-08-03T07:04:49Z' updated: '2021-11-20T15:06:15Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: weikangchia logo: https://avatars.githubusercontent.com/u/2174882?v=4 license: MIT repoEtag: '"ed84942d897ffae6b850a6b122913144df3feef696aa3852582d89d2fc3fb1d5"' repoLastModified: Sat, 20 Nov 2021 15:06:15 GMT foundInMaster: true category: - Description Validators - Parsers id: 90a8911f0b5f8cc9b0cb4c10ffc39819 - source: openapi3 tags repository: https://github.com/midouest/curdle v3: true id: cc9688d1cc4322f8aa6fadf53aadbcd9 repositoryMetadata: base64Readme: >- IyBjdXJkbGUKCkEgY2hlZXN5IHdvcmQgZ2FtZSBidWlsdCB0byBkZW1vbnN0cmF0ZSBEamFuZ28gUkVTVCBGcmFtZXdvcmsgYW5kIFJlZHV4IGNvZGUgZ2VuZXJhdGlvbgoKIVtdKC9pbWFnZXMvc2NyZWVuc2hvdC5wbmcpCgojIyBSZXF1aXJlbWVudHMKCi0gUHl0aG9uIDMuMTAKLSBQaXBlbnYKLSBOb2RlLmpzIDE4LjguMAotIG5wbSA4LjE5LjEKCiMjIFNldHVwCgojIyMgQ2hlY2sgb3V0IHRoZSByZXBvc2l0b3J5CgpgYGBiYXNoCmdpdCBjbG9uZSBodHRwczovL2dpdGh1Yi5jb20vbWlkb3Vlc3QvY3VyZGxlLmdpdApjZCBjdXJkbGUKYGBgCgojIyMgSW5zdGFsbCBQeXRob24gZGVwZW5kZW5jaWVzCgpgYGBiYXNoCnBpcGVudiBpbnN0YWxsIC0tZGV2CnBpcGVudiBzaGVsbApgYGAKCiMjIyBJbml0aWFsaXplIERqYW5nbyBkYXRhYmFzZQoKYGBgYmFzaApweXRob24gbWFuYWdlLnB5IG1pZ3JhdGUKcHl0aG9uIG1hbmFnZS5weSBjcmVhdGVzdXBlcnVzZXIKcHl0aG9uIG1hbmFnZS5weSBsb2FkX3dvcmRzIGRhdGEvYW5zd2Vycy50eHQgZGF0YS9ndWVzc2VzLnR4dApgYGAKCiMjIyBJbnN0YWxsIEphdmFTY3JpcHQgZGVwZW5kZW5jaWVzCgpgYGBiYXNoCmNkIHdlYgpucG0gaW5zdGFsbApgYGAKCiMjIExhdW5jaAoKIyMjIEJhY2tlbmQKCmBgYGJhc2gKcHl0aG9uIG1hbmFnZS5weSBydW5zZXJ2ZXIgODAwMApgYGAKCk9wZW4gaHR0cDovL2xvY2FsaG9zdDo4MDAwL2RvY3MvIHRvIHZpZXcgdGhlIE9wZW5BUEkgc2NoZW1hCgojIyMgRnJvbnRlbmQKCmBgYGJhc2gKY2Qgd2ViCm5wbSBzdGFydApgYGAKCk9wZW4gaHR0cDovL2xvY2FsaG9zdDozMDAwIHRvIHZpZXcgdGhlIGFwcAoKIyMgQ29tbW9uIERldmVsb3BtZW50IFRhc2tzCgojIyMgQ29kZSBHZW5lcmF0aW9uCgpUbyByZWdlbmVyYXRlIHRoZSBPcGVuQVBJIHNjaGVtYSBhbmQgUmVkdXggQVBJIHNsaWNlLCBydW4gdGhlIFtjb2RlZ2VuXSgvc2NyaXB0cy9jb2RlZ2VuLnNoKSBzY3JpcHQgaW4gdGhlIHJvb3Qgb2YgdGhlIHJlcG9zaXRvcnk6CgpgYGBiYXNoCi4vc2NyaXB0cy9jb2RlZ2VuLnNoCmBgYAoKIyMjIFJlc2V0IEdhbWUgRGF0YQoKVG8gY2xlYXIgYWxsIGdhbWUgZGF0YSBhbmQgcmVsb2FkIHRoZSB3b3JkIGRhdGFiYXNlLCBydW4gdGhlIFtyZXNldF0oL3NjcmlwdHMvcmVzZXQuc2gpIHNjcmlwdCBpbiB0aGUgcm9vdCBvZiB0aGUgcmVwb3NpdG9yeToKCmBgYGJhc2gKLi9zY3JpcHRzL3Jlc2V0LnNoCmBgYAo= readmeEtag: '"8382f75ca2bef8fc0ce8d2409259392dbf1d48d5"' readmeLastModified: Mon, 26 Sep 2022 00:46:21 GMT repositoryId: 534877345 description: >- A cheesy word game built to demonstrate Django REST Framework and Redux code generation created: '2022-09-10T03:18:17Z' updated: '2022-09-10T16:03:37Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: midouest logo: https://avatars.githubusercontent.com/u/67558040?v=4 repoEtag: '"72e02ad2489a5741725d1d5e09c0ab22842065cba9f7da620f1f34635f262077"' repoLastModified: Sat, 10 Sep 2022 16:03:37 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/erickyudha/car-rental-backend-api_tdd v3: true id: 28f91e549f3517bccca172feb0674357 repositoryMetadata: base64Readme: >- IyBCQ1IgQmFja2VuZCBBUEkgKFRlc3QgRHJpdmVuIERldmVsb3BtZW50ICYgRGVwbG95bWVudCBDaGFsbGVuZ2UpCgpUaGlzIHByb2plY3QgaXMgYW4gaW5kaXZpZHVhbCBjaGFsbGVuZ2UgdG8gYnVpbGQgdW5pdCBhbmQgaW50ZWdyYXRpb24gdGVzdHMgZnJvbSBhIGdpdmVuIFtleHByZXNzIGJhY2tlbmQgYXBpIHByb2plY3RdKGh0dHBzOi8vZ2l0aHViLmNvbS9lcmlja3l1ZGhhL2Nhci1yZW50YWwtYmFja2VuZC1hcGlfcmF3KSBhbmQgZGVwbG95aW5nIHRoaXMgYXBwIHVzaW5nIFtyYWlsd2F5LmFwcF0ocmFpbHdheS5hcHApLiAgCgpVbml0IGFuZCBpbnRlZ3JhdGlvbiB0ZXN0cyBmb3IgdGhpcyBwcm9qZWN0IGlzIGJ1aWx0IHVzaW5nIGBqZXN0YCBhbmQgYHN1cGVydGVzdGAuIFRlc3QgY292ZXJhZ2UgZGV0YWlscyBjYW4gYmUgc2VlbiBbaGVyZV0oI3Rlc3QtY292ZXJhZ2UpLiAgClN0eWxpbmcgZm9yIHRoaXMgamF2YXNjcmlwdCBwcm9qZWN0IGlzIGVuZm9yY2VkIHVzaW5nIEVTTGludCBbR29vZ2xlIEphdmFzY3JpcHQgc3R5bGUgZ3VpZGVdKGh0dHBzOi8vZ2l0aHViLmNvbS9nb29nbGUvZXNsaW50LWNvbmZpZy1nb29nbGUpLgoKVGhpcyBwcm9qZWN0IGlzIG1hZGUgYXMgcGFydCBvZiBCaW5hciBBY2FkZW15IGZ1bGxzdGFjayB3ZWIgZGV2ZWxvcG1lbnQgY291cnNlLgoKIyMgTGl2ZSBBcHBsaWNhdGlvbgoKVGhpcyBhcHBsaWNhdGlvbiBpcyBjdXJyZW50bHkgZGVwbG95ZWQgdXNpbmcgW3JhaWx3YXkuYXBwXShyYWlsd2F5LmFwcCkuICAKWW91IGNhbiBhY2Nlc3MgdGhpcyBhcHBsaWNhdGlvbiBvbiBbaHR0cHM6Ly9lcmljay1yZW50YWwudXAucmFpbHdheS5hcHAvXShodHRwczovL2VyaWNrLXJlbnRhbC51cC5yYWlsd2F5LmFwcC8pICAKICAKSW5mb3JtYXRpb24gYWJvdXQgc2VydmVyIHJvdXRlcyBhbmQgZG9jdW1lbnRhdGlvbiBjYW4gYmUgc2VlbiBbaGVyZV0oI3JvdXRlcy1hbmQtZG9jdW1lbnRhdGlvbikuCgojIyBSdW5uaW5nIExvY2FsIEFwcAoKVG8gcnVuIHRoaXMgYXBwbGljYXRpb24gaW4geW91ciBjb21wdXRlciB5b3UgY2FuIGZvbGxvdyB0aGVzZSBzdGVwcyBiZWxvdy4KCiMjIyBTZXR1cCBMb2NhbCBTZXJ2ZXIgYW5kIERhdGFiYXNlCgogICAgbnBtIGluc3RhbGwKClJ1biB0aGlzIGNvbW1hbmQgdG8gaW5zdGFsbCBhbGwgZGVwZW5kZW5jaWVzIG5lZWRlZCBieSB0aGlzIHByb2plY3QuCgogICAgbnBtIHJ1biBkYjpzZXR1cAoKQWZ0ZXIgdGhhdCwgcnVuIHRoaXMgY29tbWFuZCB0byBzZXR1cCBkYXRhYmFzZSB3aXRoIHNlcXVlbGl6ZS4gUmVtZW1iZXIgdG8gc2V0dXAgYC5lbnZgIGZpbGUgb24gdGhpcyBwcm9qZWN0IGZpcnN0IHdpdGggeW91ciBwb3N0Z3JlcyBkYXRhYmFzZSBkYXRhLCB5b3UgY2FuIGZvbGxvdyB0aGlzIGV4YW1wbGUgZmlsZSBbaGVyZV0oLmVudi5leGFtcGxlKS4gUmVtZW1iZXIgdG8gaGF2ZSB5b3VyIHBvc3RncmVzIGRhdGFiYXNlIGluc3RhbGxlZCBhbmQgcnVubmluZyBmaXJzdC4gCgoqKlJ1biB0aGlzIGRhdGFiYXNlIHNldHVwIGNvbW1hbmQgdHdpY2UsIHdpdGggZGlmZmVyZW50IGBOT0RFX0VOVmAgc2V0dGluZ3MgaW4gYC5lbnZgIGZpbGUuKiogICAKCjEuIGBOT0RFX0VOVj1kZXZlbG9wbWVudGAgKHRoaXMgaXMgdGhlIGRlZmF1bHQgc2V0dGluZyksIHVuaXQgdGVzdHMgaXMgdXNpbmcgZGF0YWJhc2UgaW4gdGhpcyBlbnZpcm9ubWVudC4KMi4gYE5PREVfRU5WPXRlc3RgLCBpbnRlZ3JhdGlvbiB0ZXN0cyBpcyB1c2luZyBkYXRhYmFzZSBpbiB0aGlzIGVudmlyb25tZW50LgoKKlRoaXMgaGF2ZSB0byBiZSBkb25lIHRvIGVuc3VyZSBib3RoIHVuaXQgYW5kIGludGVncmF0aW9uIHRlc3RzIHdvcmtpbmcgY29ycmVjdGx5KiAgCgojIyMgUnVubmluZyBTZXJ2ZXIKCiAgICBucG0gc3RhcnQKCnJ1biB0aGlzIGNvbW1hbmQgdG8gc3RhcnQgZXhwcmVzcyBzZXJ2ZXIgbm9ybWFsbHkuIFRvIHJ1biB0aGUgc2VydmVyIGluIGRldmVsb3BtZW50IG1vZGUsIHVzZSB0aGlzIGNvbW1hbmQgaW5zdGVhZDoKCiAgICBucG0gcnVuIGRldmVsb3AKClRvIHJ1biBib3RoIHVuaXQgYW5kIGludGVncmF0aW9uIHRlc3RzLCB1c2UgdGhpcyBjb21tYW5kOgoKICAgIG5wbSB0ZXN0CgojIyBSb3V0ZXMsIERvY3VtZW50YXRpb24sIGFuZCBUZXN0IENvdmVyYWdlCgojIyMgUm91dGVzIGFuZCBEb2N1bWVudGF0aW9uCgpEb2N1bWVudGF0aW9uIGFib3V0IEFQSSBpcyBhdmFpbGFibGUgaGVyZToKCnwgUm91dGUgfCBEZWZhdWx0IExvY2FsIExpbmsgfCBMaXZlIExpbmsgfAp8LS18LS18LS18CnwgYC9kb2N1bWVudGF0aW9uYCB8IGxvY2FsaG9zdDo4MDAwL2RvY3VtZW50YXRpb24gfCBodHRwczovL2VyaWNrLXJlbnRhbC51cC5yYWlsd2F5LmFwcC9kb2N1bWVudGF0aW9uLyB8CgpEZWZhdWx0IGFkbWluIGFjY291bnQgZGF0YSBpcyBhdmFpbGFibGUgYW5kIGNhbiBiZSBjaGFuZ2VkIGluIGxvY2FsIG9uIFtgY29uZmlnL2FkbWluLmpzYF0oY29uZmlnL2FkbWluLmpzKSB3aXRoIHRoaXMgYXMgZGVmYXVsdCBzZXR0aW5nczoKCiAgICB7CiAgICAgIG5hbWU6ICdtci5hZG1pbicsCiAgICAgIGVtYWlsOiAnYWRtaW5AYmluYXIuY28uaWQnLAogICAgICBwYXNzd29yZDogJ2FkbWluJywKICAgIH0KCiMjIyBUZXN0IENvdmVyYWdlCgpVbml0IGFuZCBpbnRlZ3JhdGlvbiB0ZXN0cyBjb3ZlcmFnZSBvZiB0aGlzIHByb2plY3QgaXMgKioqPjk3LjclKioqLCBkZXRhaWxzIGNhbiBiZSBzZWVuIG9uIHRoaXMgaW1hZ2UgYmVsb3cuCgohW2NvdmVyYWdlIHRlc3QgZGV0YWlsc10oZG9jcy9jb3ZlcmFnZS10ZXN0LnBuZykKCiFbY292ZXJhZ2UgdGVzdCBjb25zb2xlXShkb2NzL2NvdmVyYWdlLXRlc3QtY29uc29sZS5wbmcpCg== readmeEtag: '"9effb2962159e3313705293ef3b8a787fc1f93b5"' readmeLastModified: Thu, 17 Nov 2022 14:06:47 GMT repositoryId: 566986244 description: >- Test driven development and deployment challenge. Making unit and integration tests for an express backend api project using jest and supertest. Deployment using railway app. created: '2022-11-16T20:42:20Z' updated: '2022-11-17T05:32:47Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 1 owner: erickyudha logo: https://avatars.githubusercontent.com/u/72851075?v=4 license: MIT repoEtag: '"b6f17c9d269869296e88d2e7c7ae1779a1c7e6421a063b3cfd9d3e1835d869ac"' repoLastModified: Thu, 17 Nov 2022 05:32:47 GMT category: Server Implementations foundInMaster: true - source: openapi3 tags repository: https://github.com/api-stuff/master-exploder v3: true id: b9307621f36f0c545fa02028ca188d86 repositoryMetadata: base64Readme: >- IyBNYXN0ZXIgRXhwbG9kZXIgKE9wZW5BUEkgZG9jdW1lbnQgdG8gInN0dWZmIikKClRoaXMgdXRpbGl0eSB0YWtlcyBhbiBPcGVuQVBJIGRvY3VtZW50IGFzIGlucHV0IGFuZCBkZXJlZmVyZW5jZXMgaXQsIGFuZCB0aGVuIHdyaXRlcyBhIGxvYWQgb2YgZGF0YSB0byBhbm90aGVyIGZvcm1hdC4KClJpZ2h0IG5vdyB0aGUgb25seSBzdXBwb3J0ZWQgZm9ybWF0IGlzIGRlbGltaXRlZCBkYXRhIHlvdSBjYW4gY29weSBpbnRvIGEgc3ByZWFkc2hlZXQgZXRjLCBidXQgaXQnbGwgZ2V0IGJldHRlciA6d2luazouCgojIyBJbnN0YWxsCgpUaGlzIGlzIGEgcHVibGljIE5QTSBwYWNrYWdlIG9uIHRoZSBHaXRIdWIgcmVnaXN0cnkuIFlvdSdsbCBuZWVkIGEgbmljZSBzZXR0aW5nIGluIHlvdXIgaG9tZSBgLm5wbXJjYCBmaWxlIGFzIGZvbGxvd3M6CgpgYGBiYXNoCkBhcGktc3R1ZmY6cmVnaXN0cnk9aHR0cHM6Ly9ucG0ucGtnLmdpdGh1Yi5jb20KLy9ucG0ucGtnLmdpdGh1Yi5jb20vOl9hdXRoVG9rZW49JHtHSVRIVUJfTlBNX1RPS0VOfQpgYGAKCldpdGggdGhhdCBpbiBoYW5kIHlvdSBjYW4gZG86CgpgYGBiYXNoCm5wbSBpbnN0YWxsIC1nIEBhcGktc3R1ZmYvbWFzdGVyLWV4cGxvZGVyQDEuMS4wCmBgYAoKYW5kIEJvYiBpcyB5b3VyIG1vdGhlcidzIGJyb3RoZXIuCgojIyBFeGVjdXRpb24KClRoZSBzY3JpcHQgdGFrZXMgYSBsaXN0IG9mIE9wZW5BUEkgZGVzY3JpcHRpb24gZG9jdW1lbnRzIGFuZCBsb29wcyBvdmVyIHRoZSBjb250ZW50cywgZ2VuZXJhdGluZyBvbmUgb3V0cHV0IGZvciB0aGUgc2NyaXB0OgoKYGBgYmFzaAptYXN0ZXItZXhwbG9kZXIgXAotLW91dHB1dCBleHBsb2RlZC50eHQgXAotLXJlcXVlc3QtY29udGVudC10eXBlICJhcHBsaWNhdGlvbi9qc29uIiBcCi0tcmVzcG9uc2UtY29udGVudC10eXBlICJhcHBsaWNhdGlvbi9qc29uIiBcCn4vRG93bmxvYWRzL2FjY291bnQtaW5mby1vcGVuYXBpLmpzb24KYGBgCgpUaGUgbG9nZ2luZyBpcyBgcGlub2AgZm9ybWF0dGluZywgc28gaWYgd2FudCBpdCBwcmV0dGllciB0aGVuIHBpcGUgdG8gYHBpbm8tcHJldHR5YCBpLmUuIGB8IG5weCBwaW5vLXByZXR0eWAuCgo6dGh1bWJzdXA6Cg== readmeEtag: '"ebb51ad35c33b516f0ee2c9c7a4d105c97897ba0"' readmeLastModified: Thu, 25 Jul 2024 11:52:10 GMT repositoryId: 503245695 description: >- A tool for exploding OpenAPI documents into various different formats (CSV, Gherkin, etc.) created: '2022-06-14T07:03:11Z' updated: '2025-07-30T15:10:38Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: api-stuff logo: https://avatars.githubusercontent.com/u/68026188?v=4 repoEtag: '"fb2a267d47c5747324c31817845124158923c9441df6b2c8fb0dd7d094109dc7"' repoLastModified: Wed, 30 Jul 2025 15:10:38 GMT category: Parsers foundInMaster: true - source: https://openapi.tools/ name: Kusk link: https://kusk.io repository: https://github.com/kubeshop/kusk language: Kubernetes source_description: >- Kusk makes your OpenAPI definition the source of truth for API resources in your cluster. v3: true repositoryMetadata: base64Readme: >- # Kusk CLI (archived)

**This repository is now deprecated** and have moved to [Kusk Gateway's repository](https://github.com/kubeshop/kusk-gateway).

----

Kusk is a CLI tool designed to help you manage common tasks required when running Kusk Gateway.

Currently we support the following commands:

- `install` - installs Kusk Gateway and all its components with a single command. (Requires a helm installation)
- `api generate` - for creating Kusk Gateway API resources from your OpenAPI specification document.

---

# Table of contents

- [Usage](#usage)
  - [install](#install)
  - [api generate](#api-generate)
  - [dashboard](#dashboard)
- [Installation](#installation)
- [Updating](#updating)
- [Uninstallation](#uninstallation)
- [Contributing](#contributing)
- [License](#license)

# Usage

## Install

### Flags

|         Flag         |                                                     Description                                                     | Required? |
| :------------------: | :-----------------------------------------------------------------------------------------------------------------: | :-------: |
|       `--name`       | the prefix of the name to give to the helm releases for each of the kusk gateway components (default: kusk-gateway) |     ❌     |
| `--namespace` / `-n` |  the namespace to install kusk gateway into. Will create the namespace if it doesn't exist (default: kusk-system)   |     ❌     |
|   `--no-dashboard`   |                               when set, will not install the kusk gateway dashboard.                                |     ❌     |
|      `--no-api`      |                      when set, will not install the kusk gateway api. implies --no-dashboard.                       |     ❌     |
|  `--no-envoy-fleet`  |                                     when set, will not install any envoy fleets                                     |     ❌     |

### Examples

```sh
$ kusk install

Will install kusk-gateway, a public (for your APIS) and private (for the kusk dashboard and api)
envoy-fleet, api, and dashboard in the kusk-system namespace using helm.

$ kusk install --name=my-release --namespace=my-namespace

Will create a helm release named with --name in the namespace specified by --namespace.

$ kusk install --no-dashboard --no-api --no-envoy-fleet

Will install kusk-gateway, but not the dashboard, api, or envoy-fleet.
```

## Api generate

Generate accepts your OpenAPI spec file as input either as a local file or a URL pointing to your file
and generates a Kusk Gateway compatible API resource that you can apply directly into your cluster.

Configuration of the API resource is done via the x-kusk extension.

If the OpenAPI spec doesn't have a top-level x-kusk annotation set, it will add them for you and set
the upstream service, namespace and port to the flag values passed in respectively and set the rest of the settings to defaults.
This is enough to get you started

If the x-kusk extension is already present, it will override the the upstream service, namespace and port to the flag values passed in respectively
and leave the rest of the settings as they are.

You must specify the name of the envoyfleet you wish to use to expose your API. This is because Kusk Gateway could be managing more than one.
In the future, we will add the notion of a default envoyfleet which kusk gateway will use when none is specified. i.e. kusk-gateway-envoy-fleet.

If you do not specify the envoyfleet namespace, it will default to kusk-system.

Sample usage

No name specified

```sh
kusk api generate \
  -i spec.yaml \
  --envoyfleet.name kusk-gateway-envoy-fleet \
  --envoyfleet.namespace kusk-system
```

In the above example, kusk will use the openapi spec info.title to generate a manifest name and leave the existing
x-kusk extension settings

No api namespace specified

```sh
kusk api generate \
  -i spec.yaml \
  --name httpbin-api \
  --upstream.service httpbin \
  --upstream.port 8080 \
  --envoyfleet.name kusk-gateway-envoy-fleet
```

In the above example, as --namespace isn't defined, it will assume the default namespace.

Namespace specified

```sh
kusk api generate \
  -i spec.yaml \
  --name httpbin-api \
  --upstream.service httpbin \
  --upstream.namespace my-namespace \
  --upstream.port 8080 \
  --envoyfleet.name kusk-gateway-envoy-fleet
```

OpenAPI spec at URL

```sh
kusk api generate \
    -i https://raw.githubusercontent.com/$ORG_OR_USER/$REPO/myspec.yaml \
    --name httpbin-api \
    --upstream.service httpbin \
    --upstream.namespace my-namespace \
    --upstream.port 8080 \
    --envoyfleet.name kusk-gateway-envoy-fleet
```

This will fetch the OpenAPI document from the provided URL and generate a Kusk Gateway API resource

### Flags

|          Flag          |                                             Description                                             | Required? |
| :--------------------: | :-------------------------------------------------------------------------------------------------: | :-------: |
|        `--name`        | the name to give the API resource e.g. --name my-api. Otherwise taken from OpenAPI info title field |     ❌     |
|  `--namespace` / `-n`  | the namespace of the API resource e.g. --namespace my-namespace, -n my-namespace (default: default) |     ❌     |
|     `--in` / `-i`      |       file path or URL to OpenAPI spec file to generate mappings from. e.g. --in apispec.yaml       |     ✅     |
|  `--upstream.service`  |                                 name of upstream Kubernetes service                                 |     ❌     |
| `--upstream.namespace` |                          namespace of upstream service (default: default)                           |     ❌     |
|   `--upstream.port`    |                       port that upstream service is exposed on (default: 80)                        |     ❌     |
|  `--envoyfleet.name`   |                               name of envoyfleet to use for this API                                |     ✅     |
| `envoyfleet.namespace` |                  namespace of envoyfleet to use for this API. Default: kusk-system                  |     ❌     |

### Example

Take a look at the [`./examples/httpbin-spec.yaml`](./examples/httpbin-spec.yaml)

```sh
kusk api generate -i ./examples/httpbin-spec.yaml --name httpbin-api --upstream.service httpbin --upstream.port 8080 --envoyfleet.name kusk-gateway-envoy-fleet
```

The output should contain the following x-kusk extension at the top level

```sh
...
x-kusk:
  cors: {}
  path:
    rewrite:
      pattern: ""
      substitution: ""
  upstream:
    service:
	name: httpbin
	namespace: default
	port: 8080
```

## dashboard
Access the kusk dashboard. kusk dashboard will start a port-forward session on port 8080 to the envoyfleet
serving the dashboard and will open the dashboard in the browser. By default this is kusk-gateway-private-envoy-fleet.kusk-system.

The flags --envoyfleet.namespace and --envoyfleet.name can be used to change the envoyfleet.

### Flags
|           Flag           |                                         Description                                          | Required? |
| :----------------------: | :------------------------------------------------------------------------------------------: | :-------: |
|      `--kubeconfig`      |                                 absolute path to kube config                                 |     ❌     |
|   `--envoyfleet.name`    | kusk gateway dashboard envoy fleet service name. (default: kusk-gateway-private-envoy-fleet) |     ❌     |
| `--envoyfleet.namespace` |         kusk gateway dashboard envoy fleet service namespace. (default: kusk-system)         |     ❌     |
|    `--external-port`     |                    external port to access dashboard at. (default: 8080)                     |     ❌     |

### Examples

```sh
$ kusk dashboard
```

Opens the kusk gateway dashboard in the browser by exposing the default private envoy fleet on port 8080

```sh
$ kusk dashboard --envoyfleet.namespace=other-namespace --envoyfleet.name=other-envoy-fleet
```

Specify other envoyfleet and namespace that is serving the dashboard

```sh
$ kusk dashboard --external-port=9090
```

Expose dashboard on port 9090

```sh
$ kusk dashboard --kubeconfig=/path/to/kube/config
```

Specify path to kube config. $HOME/.kube/config is used by default.

# Installation

## Homebrew

```sh
brew install kubeshop/kusk/kusk
```

## Go install the latest release on Github

```sh
go install github.com/kubeshop/kusk@latest
```

To install a particular version: replace `latest` with the version number

You can get a list of the available kusk versions from our [releases page](https://github.com/kubeshop/kusk/releases)

## Easy install script

This will install `kusk` into `/usr/local/bin/kusk`

```sh
$ curl -sSLf https://raw.githubusercontent.com/kubeshop/kusk/main/scripts/install.sh | sh
...
kusk installed in /usr/local/bin/kusk
```

## From source

```sh
git clone git@github.com:kubeshop/kusk.git && \
  cd kusk && \
  go install
```

## Alternative installation method (manual)

If you don't like automatic scripts you can always use the manual install:

1. Download binary with version of your choice (recent one is recommended).
2. Unpack it (`tar -zxvf kusk_0.1.0_Linux_arm64.tar.gz`).
3. Move it to a location in the `PATH`. For example `mv kusk_0.1.0_Linux_arm64/kusk /usr/local/bin/kusk`.

For Windows, download the binary from [here](https://github.com/kubeshop/kusk/releases), unpack the binary and add it to `%PATH%`.

# Updating

## Homebrew

```sh
brew upgrade kubeshop/kusk/kusk
```

## Latest release on Github

```sh
go install github.com/kubeshop/kusk@latest
```

## From source

Inside of the kusk repository directory

```sh
git pull && go install
```

# CLI Reference 

For detailed command line reference visit [docs](docs/kusk.md)
# Contributing

Your contributions are always welcome! Please have a look at [How to contribute](https://github.com/kubeshop/.github/blob/main/CONTRIBUTING.md) first.

### Development

```sh
make
./kusk --help
```

# License

[The MIT License](https://mit-license.org/).

Please have a look at the [`./LICENSE`](./LICENSE) for more details.
 readmeEtag: '"b3b36f64d95a7b67e18e776f08517bd167f37126"' readmeLastModified: Wed, 12 Oct 2022 11:05:43 GMT repositoryId: 425900868 description: CLI for Kusk Gateway related functionality created: '2021-11-08T15:54:35Z' updated: '2024-02-24T22:05:47Z' language: Go archived: false stars: 7 watchers: 13 forks: 1 owner: kubeshop logo: https://avatars.githubusercontent.com/u/82541796?v=4 license: MIT repoEtag: '"3a974591c7845c53e47798c2217f914f8ddfd17f66e2889eb72eec2e1da60e4a"' repoLastModified: Sat, 24 Feb 2024 22:05:47 GMT foundInMaster: true id: 5a0e82ad543d0aaa0ce624ce0a747f6b category: Server Implementations - source: openapi3 tags repository: https://github.com/yoaquim/zemi v3: true id: b1513f51d7a1d3ba8db7b3fb1d92c0d7 repositoryMetadata: base64Readme: >- # zemi

[![build](https://github.com/yoaquim/zemi/actions/workflows/ci.yml/badge.svg)](https://github.com/yoaquim/zemi/actions/workflows/ci.yml)
[![Code Climate Coverage](https://codeclimate.com/github/yoaquim/zemi/badges/coverage.svg)](https://codeclimate.com/github/yoaquim/zemi/coverage)
[![Code Climate Maintainability](https://codeclimate.com/github/yoaquim/zemi/badges/gpa.svg)](https://codeclimate.com/github/yoaquim/zemi)
[![Snyk.io Vulnerabilities](https://snyk.io/test/github/yoaquim/zemi/badge.svg?targetFile=package.json)](https://snyk.io/test/github/yoaquim/zemi?targetFile=package.json)

[![npm Version](https://img.shields.io/npm/v/zemi?color=137dc2&logo=npm)](https://www.npmjs.com/package/zemi)
[![Types](https://badgen.net/npm/types/zemi)](https://github.com/yoaquim/zemi/tree/main/src/types)
[![Dependencies](https://img.shields.io/badge/dependencies-2-blue)](https://github.com/yoaquim/zemi/blob/main/package.json#L27-L30)
[![Install Size](https://packagephobia.com/badge?p=zemi)](https://packagephobia.com/result?p=zemi)
[![License](https://badgen.net/npm/license/zemi)](https://github.com/yoaquim/zemi/blob/main/LICENSE.md)

<br>

zemi is a [data-driven](#data-driven) routing library for [Express](https://expressjs.com/), built with Typescript.

Features:

- [reverse-routing](#reverse-routing)
- [path-parameter inheritance](#parameter-inheritance)
- route-level [middleware support](#middleware)

<br>

# Table of Contents

1. [Routing](#routing)
    1. [Data-driven](#data-driven)
    2. [Reverse-routing](#reverse-routing)
    3. [Middleware](#middleware)
    4. [Parameter Inheritance](#parameter-inheritance)
2. [Types](#types)
    1. [ZemiMethod](#zemimethod)
    2. [ZemiRequestHandler](#zemirequesthandler)
    3. [ZemiRequest](#zemirequest)
    4. [ZemiResponse](#zemiresponse)
    5. [ZemiRouteDefinition](#zemiroutedefinition)
    6. [ZemiRoute](#zemiroute)
3. [Examples](#examples)
   1. [Simple](https://github.com/yoaquim/zemi/blob/main/examples/simple.ts)
   2. [With Middleware](https://github.com/yoaquim/zemi/blob/main/examples/with-middleware.ts)
   3. [Using Named Routes For Redirects](https://github.com/yoaquim/zemi/blob/main/examples/using-named-routes-for-redirect.ts)
   4. [Using Reverse Routing](https://github.com/yoaquim/zemi/blob/main/examples/using-reverse-routing.ts)
   5. [With Param Inheritance from Parent Routes](https://github.com/yoaquim/zemi/blob/main/examples/nested-route-param-inheritance.ts)
4. [Limitations](#limitations)

<br>

## Routing

### Data-driven

Assume you have the following functions defined: `petsHandler`, `dogBreedHandler`, `dogBreedsIdHandler`, `catsByIdHandler` ; e.g.:

```ts
const petsHandler = (request: ZemiRequest, response: ZemiResponse) => {
  // do something with this request and respond
  response.status(200).json({ pets: ["dogs", "cats"] });
};

const dogBreedHandler = (request: ZemiRequest, response: ZemiResponse) => {
   //...
};

const dogBreedsIdHandler = (request: ZemiRequest, response: ZemiResponse) => {
   //...
};

const catsByIdHandler = (request: ZemiRequest, response: ZemiResponse) => {
   //...
};
```

Then the following code:

```ts
import express from "express";
import zemi, { ZemiRoute, ZemiMethod } from "zemi";

const { GET } = ZemiMethod;

const routes: Array<ZemiRoute> = [
  {
    name: "pets",
    path: "/pets",
    [GET]: petsHandler,
    routes: [
      {
        name: "dogBreeds",
        path: "/dogs/:breed",
        [GET]: dogBreedHandler,
        routes: [
          {
            name: "dogsByBreedById",
            path: "/:id",
            [GET]: dogBreedsIdHandler
          }
        ]
      },
      {
        name: "catsById",
        path: "/cats/:id",
        [GET]: catsByIdHandler
      }
    ]
  }
];

const app = express();
app.use(express.json());
app.use("/", zemi(routes));
app.listen(3000);
```

Generates an API like:

| routes                  | response                                            |
|-------------------------|-----------------------------------------------------|
| `/pets`                 | `{pets: ['dogs', 'cats', 'rabbits']}`               |
| `/pets/dogs`            | `Cannot GET /pets/dogs/` (since it was not defined) |
| `/pets/dogs/labrador`   | `{"result":["Fred","Barney","Wilma"]}`              |
| `/pets/dogs/labrador/1` | `{"result":"Barney"}`                               |
| `/pets/cats`            | `Cannot GET /pets/cats/` (since it was not defined) |
| `/pets/cats/2`          | `{"result":"Daphne"}`                               |

<br>

### Reverse-routing

zemi builds route-definitions for all routes and adds them to the [`ZemiRequest`](#zemirequest) passed to the handler function.

All route-definitions are named (index-accessible) and follow the same naming convention: `[ancestor route names]-[parent route name]-[route name]`, e.g. `basePath-greatGrandparent-grandparent-parent-myRoute`, `pets-dogsBreeds-dogsByBreedById`.

Each route-definition contains the name, path, and path-parameters (if present) of the route.
It also contains a reverse function which — when invoked with an object mapping path-parameters to values — will return the interpolated path with values.

E.g. a handler like this:

```ts
import { ZemiRequest, ZemiResponse, ZemiRouteDefinition } from "zemi";

const petsHandler = (request: ZemiRequest, response: ZemiResponse) => {
  const routeDefinitions: Record<string, ZemiRouteDefinition> = request.routeDefinitions;
  const { path, name, parameters, reverse } = routeDefinitions["pets-dogBreeds-dogsByBreedById"];
  response.status(200).json({ path, name, parameters, reverse: reverse({ breed: 'Corgi', id: '99' }) });
};
```

Returns:

```json
  {
  "path": "/pets/dogs/:breed/:id",
  "name": "pets-dogBreeds-dogsByBreedById",
  "parameters": [
    "breed",
    "id"
  ],
  "reverse": "/pets/dogs/corgi/99"
}
```

This allows you to generate links, redirect, and change path values without having to hardcode strings and change them later.

<br>

### Middleware

zemi lets you define middleware functions at the route level:

Retaking and tweaking our example from the beginning:

```ts
import { ZemiRequest, ZemiResponse } from "zemi";
import { NextFunction } from "express";

const routes: Array<ZemiRoute> = [
  {
    name: "pets",
    path: "/pets",
    [GET]: petsHandler,
    routes: [
      {
        name: "dogBreeds",
        path: "/dogs/:breed",
        [GET]: dogBreedHandler,
        middleware: [
          function logRouteDefs(request: ZemiRequest, response: ZemiResponse, next: NextFunction) {
            console.log(JSON.stringify(request.routeDefinitions));
            next();
          }
        ],
        routes: [
          {
            name: "dogsByBreedById",
            path: "/:id",
            [GET]: dogBreedsIdHandler
          }
        ]
      },
      {
        name: "catsById",
        path: "/cats/:id",
        [GET]: { handler: catsByIdHandler }
      }
    ]
  }
];
```

The middleware function `logRouteDefs` defined at the `dogBreeds` level will be applied to all the methods at that level and all nested routes — which means our `dogsByBreedById` route will gain that functionality also.

<br>

### Parameter Inheritance

As show in previous examples, parameters defined at parent routes are passed and available to nested routes.

E.g. in this purposefully convoluted example:

```ts
const routes: Array<ZemiRoute> = [
  {
    name: "pets",
    path: "/pets",
    [GET]: petsHandler,
    routes: [
      {
        name: "dogBreeds",
        path: "/dogs/:breed",
        [GET]: dogBreedHandler,
        routes: [
          {
            name: "dogsByBreedById",
            path: "/:id",
            [GET]: dogBreedsIdHandler,
            routes: [
              {
                name: "dogsByBreedByIdDetailsSection",
                path: "/details/:section",
                [GET]: dogBreedsIdDetailsSectionHandler,
                routes: [
                  {
                    name: "newDogsByBreedByIdDetailsSection",
                    path: "/new",
                    [POST]: newDogsByBreedByIdDetailsSectionHandler
                  }
                ]
              }
            ]
          }
        ]
      }
    ]
  }
];
```

The `newDogsByBreedByIdDetailsSection` route (path: `/pets/dogs/:breed/:id/details/:section/new`) will have `breed`, `id`, and `section` available as request parameters in the ZemiRequest object.

<br>

## Types

### `ZemiMethod`

*Enum*

The HTTP methods supported by [`ZemiRoute`](#zemiroute).

| Member    | Value     |
|-----------|-----------|
| `GET`     | `get`     |
| `POST`    | `post`    |
| `PUT`     | `put`     |
| `DELETE`  | `delete`  |
| `OPTIONS` | `options` |

<br>

### `ZemiRequestHandler`

How to handle incoming requests for this route method; basically `express.RequestHandler`, but gets passed its own request and response versions, plus adds that routes [`ZemiRouteDefinition`](#zemiroutedefinition) as an optional fourth param.

```ts
(
  request: ZemiRequest,
  response: ZemiResponse,
  next: express.NextFunction,
  routeDef: ZemiRouteDefinition
) => void
```

<br>

### `ZemiRequest`

*extends `express.Request`*

A wrapper for `express.Request`; adds `routeDefinitions` and `allowedResponseHttpCodes` to it.

```ts
{
  routeDefinitions: Record<string, ZemiRouteDefinition>;
  // all other members from express.Request
}

```

<br>

### `ZemiResponse`

*extends `express.Response`*

Just a wrapper for future-proofing; same as `express.Response`.

<br>

### `ZemiRouteDefinition`

Route definition for a given [`ZemiRoute`](#zemiroute).
Contains the name, path, and path-parameters (if present) of the route it's defining.
Also provides a `reverse` function that, when invoked with an object that has parameter-values, will return the resolved path.

```ts
{
  name: string;
  path: string;
  parameters: Array<string>;
  reverse: (parameterValues: object) => string;
}
```

<br>

### `ZemiRoute`

It must be provided a `name: string` and `path: string`; a [[`ZemiMethod`](#zemimethod)]:[`ZemiRequestHandler`](#zemirequesthandler) needs to be provided if that path should have functionality, but doesn't need to be if the path is just present as a path-prefix for nested routes.

```
{
   [ZemiMethod]: ZemiRequestHandler;
   name: string;
   path: string;
   middleware?: Array<RequestHandler>;
   routes?: Array<ZemiRoute>;
}
```

<br>

## Examples

Examples are available in the [examples](https://github.com/yoaquim/zemi/blob/main/examples) dir:

1. [Simple](https://github.com/yoaquim/zemi/blob/main/examples/simple.ts)

2. [With Middleware](https://github.com/yoaquim/zemi/blob/main/examples/with-middleware.ts)

3. [Using Named Routes For Redirects](https://github.com/yoaquim/zemi/blob/main/examples/using-named-routes-for-redirect.ts)
 
4. [Using Reverse Routing](https://github.com/yoaquim/zemi/blob/main/examples/using-reverse-routing.ts)

5. [With Param Inheritance from Parent Routes](https://github.com/yoaquim/zemi/blob/main/examples/nested-route-param-inheritance.ts)

<br>

## Limitations

zemi is a recursive library: it uses recursion across a number of operations in order to facilitate a low footprint and straightforward, declarative definitions.

Recursive operations can break the call-stack by going over its limit, generating `Maximum call stack size exceeded` errors. This means that the recursive function was called too many times, and exceeded the limit placed on it by Node.

While recursive functions _can_ be optimized via [tail call optimization](https://stackoverflow.com/questions/310974/what-is-tail-call-optimization) (TCO), that feature _has_ to be present in the environment being run for optimization to work.

Unfortunately — as of Node 8.x — TCO is [no](https://stackoverflow.com/questions/23260390/node-js-tail-call-optimization-possible-or-not) [longer](https://stackoverflow.com/questions/42788139/es6-tail-recursion-optimisation-stack-overflow/42788286#42788286) [supported](https://bugs.chromium.org/p/v8/issues/detail?id=4698).

This means that, depending on what you're building and the size of your API, zemi might not be the right fit for you. zemi uses recursion when dealing with nested routes, so if your application has a very high number of nested-routes within nested-routes, chances are you might exceed the call stack.
 readmeEtag: '"aad25ee30ed3e6bcf649789d5afe288234cc13bd"' readmeLastModified: Sat, 13 Jan 2024 12:49:58 GMT repositoryId: 512997745 description: zemi is a data-driven and reverse-routing library for Express JS. created: '2022-07-12T04:26:47Z' updated: '2025-12-31T11:59:36Z' language: TypeScript archived: true stars: 6 watchers: 1 forks: 0 owner: yoaquim logo: https://avatars.githubusercontent.com/u/702200?v=4 license: ISC repoEtag: '"daf166ea10c1ba07deb77c1996a1cf369ac86d701c4392f01db3b61bc1ab9c30"' repoLastModified: Wed, 31 Dec 2025 11:59:36 GMT category: Parsers foundInMaster: true - source: openapi3 tags repository: https://github.com/rpstreef/openapi-node-example v3: true repositoryMetadata: base64Readme: >- IyBPcGVuQVBJIE5vZGUgY29kZSBleGFtcGxlCgojIEFib3V0OgoKVGhpcyByZXBvIGdvZXMgdG9nZXRoZXIgd2l0aCB0aGUgW09wZW5BUEkgVGVycmFmb3JtIHJlcG9dKGh0dHBzOi8vZ2l0aHViLmNvbS9ycHN0cmVlZi9vcGVuYXBpLXRmLWV4YW1wbGUpLgoKVGhpcyBqdXN0IGhvbGRzIHRoZSBOb2RlSlMgY29kZSB0aGF0IGdldHMgZGVwbG95ZWQgd2l0aCB0aGUgQ29kZVBpcGVsaW5lIHNldHVwLg== readmeEtag: '"cd3f3ed1687207a83b1b87b52a0e85d7c6094cce"' readmeLastModified: Thu, 27 Aug 2020 05:57:04 GMT repositoryId: 244096270 description: OpenAPI Node code example created: '2020-03-01T06:02:39Z' updated: '2020-08-27T05:57:07Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 2 owner: rpstreef logo: https://avatars.githubusercontent.com/u/15830262?v=4 license: Apache-2.0 repoEtag: '"311bae09b0e30d40bebe142488be0e85edca24453b7e1add1e433c5cc3ab6b2d"' repoLastModified: Thu, 27 Aug 2020 05:57:07 GMT foundInMaster: true category: - Server - Server Implementations id: f1f8b2f0640eb6e3b082bdd243da2680 - source: openapi3 tags repository: https://github.com/reekoheek/xin-openapi v3: true repositoryMetadata: base64Readme: >- IyB4aW4tZXhhbXBsZQoKU3RhcnQgdG8gZGV2ZWxvcCBpcyBlYXN5LCBkb3dubG9hZCBleGFtcGxlIGZyb20gYGh0dHBzOi8vZ2l0aHViLmNvbS9yZWVrb2hlZWsveGluLWV4YW1wbGVgIG9yIGNsb25lLgoKYGBgYmFzaApnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL3JlZWtvaGVlay94aW4tZXhhbXBsZS5naXQgbXktYXBwCmNkIG15LWFwcApucG0gcnVuIGRldgpgYGAKCiMjIFJFRkVSRU5DRVMKCi0gaHR0cDovL3BldHN0b3JlLnN3YWdnZXIuaW8vCi0gaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vdHJlZS9PcGVuQVBJLm5leHQvZXhhbXBsZXMvdjMuMAotIGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZwotIGh0dHBzOi8va291bW91bC5jb20vb3BlbmFwaS12aWV3ZXIvP2NvbmZpZz1odHRwczovL2tvdW1vdWwuY29tL3MvZ2VvY29kZXIvYXBpL3YyL2FwaS1kb2NzLmpzb24KLSBodHRwczovL29wZW5hcGktZ3VpLmhlcm9rdWFwcC5jb20vCi0gaHR0cHM6Ly90ZW1hbmRvLmdpdGh1Yi5pby9vcGVuLWFwaS1yZW5kZXJlci9kZW1vLz91cmw9aHR0cHM6Ly90ZW1hbmRvLmdpdGh1Yi5pby9vcGVuLWFwaS1yZW5kZXJlci9wZXRzdG9yZS1vcGVuLWFwaS12My4wLjAtUkMyLmpzb24jL3BldC9wdXQKLSBodHRwczovL21pa2VyYWxwaHNvbi5naXRodWIuaW8vaW9kb2N0b3IvCgojIyBUT0RPCgotIFNob3cgcm91dGUgZGF0YSBmcm9tIHNwZWMK readmeEtag: '"cfbae69f75fc2686de3b5d005bb569329a91fa59"' readmeLastModified: Mon, 17 Jul 2017 11:48:15 GMT repositoryId: 96982618 description: null created: '2017-07-12T08:15:16Z' updated: '2017-07-12T08:24:59Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: reekoheek logo: https://avatars.githubusercontent.com/u/299394?v=4 license: MIT repoEtag: '"ce17dffb4200cf79ac194b483134bb474529a774794828f49434544b4d59455f"' repoLastModified: Wed, 12 Jul 2017 08:24:59 GMT foundInMaster: true category: - Code Generators - Parsers id: 89a09d094b623be98be9de35f0c24ddb - source: openapi3 tags repository: https://github.com/aristat/http-server v3: true repositoryMetadata: base64Readme: >- IyBIdHRwIFNlcnZlcgoKSXQncyBzbWFsbCBleGFtcGxlIGh0dHAgc2VydmVyIGJhc2VkIG9uIG9wZW5hcGkuIE5vdCBzdWl0YWJsZSBmb3IgbWljcm9zZXJ2aWNlcywgaXQncyBiZXR0ZXIgdG8gbG9vayB0byB0aGUgW2tpdF0oaHR0cHM6Ly9naXRodWIuY29tL2dvLWtpdC9raXQpIAoKIyBHZW5lcmF0ZSBhcGkuZ2VuLmdvCgpgYGAKb2FwaS1jb2RlZ2VuIC1wYWNrYWdlIGFwaSAtZ2VuZXJhdGUgdHlwZXMsY2hpLXNlcnZlcixzcGVjIC1vIGludGVybmFsL2FwcC9hcGkvYXBpLmdlbi5nbyBvcGVuYXBpLnlhbWwKYGBgCgojIFJ1bgoKYGBgCmdvIG1vZCBkb3dubG9hZApnbyBydW4gbWFpbi5nbyBzCmN1cmwgbG9jYWxob3N0OjMwMDAvYXBpL3YxL3Byb2R1Y3RzLzEKYGBgCg== readmeEtag: '"6790e11e256924c464b88c11c4bfe59c53592e85"' readmeLastModified: Sun, 28 Mar 2021 08:22:29 GMT repositoryId: 352156965 description: Example http server created: '2021-03-27T19:16:25Z' updated: '2021-10-06T23:45:40Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: Aristat logo: https://avatars.githubusercontent.com/u/4936284?v=4 repoEtag: '"0bf1f640a3e9313c4293947748060caae97bfecd85605db14bbcf615b64e2f3e"' repoLastModified: Wed, 06 Oct 2021 23:45:40 GMT foundInMaster: true category: - Code Generators - Server Implementations id: c3ba39fc99eb3dedefa20d63a670abc0 - source: openapi3 tags repository: https://github.com/thavorath/oas v3: true repositoryMetadata: base64Readme: >- IyMgRXhwZXJpbWVudGFsIE9wZW5BUEkgMyBkb2N1bWVudGF0aW9uIGdlbmVyYXRvcgoKIyMjIFVzYWdlCgpTaW1wbHkgZ28gZ2V0IGFuZCBydW4gdGhpcyBpbiB5b3VyIHByb2plY3QgZm9sZGVyLiBJdCB3aWxsIGdlbmVyYXRlIGEgYXBpLnNwZWMgZmlsZS4gVXNlIHRoaXMgZmlsZSBpbiB5b3VyIFN3YWdnZXIzIFVJLgoKYGBgCmdvIGdldCBnaXRodWIuY29tL3RoYXZvcmF0aC9vYXMKZ28gaW5zdGFsbCBnaXRodWIuY29tL3RoYXZvcmF0aC9vYXMKY2Qgfi9nby9zcmMvZ2l0aHViLmNvbS90aGF2b3JhdGgvbXktcHJvamVjdApvYXMKYGBgCgo= readmeEtag: '"75b5f9b61f96f2fa9d69f30359b2983bfda0fed9"' readmeLastModified: Wed, 23 Jan 2019 05:55:49 GMT repositoryId: 163516751 description: An experimental OpenAPI 3 spec generator created: '2018-12-29T14:20:42Z' updated: '2019-01-23T05:56:11Z' language: Go archived: false stars: 0 watchers: 1 forks: 0 owner: thavorath logo: https://avatars.githubusercontent.com/u/4705578?v=4 license: MIT repoEtag: '"994324055584445f51161289698a561a69f3c41a7bbc45472e854f3547b95079"' repoLastModified: Wed, 23 Jan 2019 05:56:11 GMT foundInMaster: true category: - SDK - Parsers id: f92bfce6656d9a106c977eb131d93178 - source: openapi3 tags repository: https://github.com/wa-craft/generator-deno v3: true id: 9d24f32b166443ca6d133c30fe8f0838 repositoryMetadata: base64Readme: >- IyBnZW5lcmF0b3IKQW4gb3BlbmFwaSBnZW5lcmF0b3IgZm9yIHdhLWNyYWZ0L2VkaXRvciB3cml0dGVuIGluIERlbm8gdGhhdCBpbiByZXBsYWNlIG9mIHRoZSBvZmZpY2lhbCBvcGVuYXBpLWdlbmVyYXRvci4KCiMgV2FybmluZwpUaGlzIHJlcG9zaXRvcnkgaXMgc3RpbGwgdW5kZXIgZGV2ZWxvcGluZywgRE8gTk9UIHRyeSB0byB1c2UgaXQgaW4gYSByZWFsIHByb2plY3QhCklmIHlvdSBoYXZlIGFueSBxdWVzdGlvbiBwbGVhc2UgY3JlYXRlIGEgbmV3IGlzc3VlLg== readmeEtag: '"7ac9ace8daba83d6f26f76d764c5265966155bb4"' readmeLastModified: Thu, 02 Jun 2022 06:32:05 GMT repositoryId: 458208409 description: >- An openapi generator for wa-craft/editor written in Deno that in replace of the official openapi-generator. created: '2022-02-11T13:58:45Z' updated: '2025-06-23T02:59:25Z' language: TypeScript archived: true stars: 0 watchers: 1 forks: 0 owner: wa-craft logo: https://avatars.githubusercontent.com/u/99487287?v=4 license: MIT repoEtag: '"fa3ab48866e8f04dbb04077a4ce67cdfb57403730d8c55c3895f5995edcab9c7"' repoLastModified: Mon, 23 Jun 2025 02:59:25 GMT category: - Parsers - Server Implementations foundInMaster: true oldLocations: - https://github.com/wa-craft/generator - source: openapi3 tags repository: https://github.com/neo-f/soda v3: true id: 207bdae87d0101f1d3d6c8b7ef8a7dd6 repositoryMetadata: base64Readme: >- # Soda

[![codecov](https://codecov.io/github/neo-f/soda/branch/master/graph/badge.svg?token=uYHY9DCbNe)](https://codecov.io/github/neo-f/soda)

**Soda** is a powerful Go library that seamlessly integrates [OpenAPI 3](https://swagger.io/specification) documentation with the [Fiber](https://github.com/gofiber/fiber) web framework. It automatically generates comprehensive API documentation from your Go structs and route definitions, eliminating the need for manual OpenAPI specification writing.

## 🚀 Features

- **Automatic OpenAPI 3 Generation**: Generate complete OpenAPI specifications from Go structs
- **Fiber Integration**: Built specifically for the Fiber web framework
- **Zero Configuration**: Works out of the box with sensible defaults
- **Type Safety**: Leverage Go's type system for API documentation
- **Interactive Documentation**: Built-in support for Swagger UI, ReDoc, RapiDoc, and Stoplight Elements
- **Request/Response Binding**: Automatic binding of HTTP requests to Go structs
- **Security Schemes**: Support for JWT, API Key, and custom security schemes
- **Validation**: Built-in validation using struct tags
- **Extensible**: Custom hooks and middleware support

## 📦 Installation

```bash
go get github.com/neo-f/soda/v3
```

## 🏁 Quick Start

```go
package main

import (
    "github.com/gofiber/fiber/v2"
    "github.com/neo-f/soda/v3"
)

// Define your request/response structs
type CreateUserRequest struct {
    Name     string `json:"name" validate:"required"`
    Email    string `json:"email" validate:"required,email"`
    Password string `json:"password" validate:"required,min=8"`
}

type UserResponse struct {
    ID    int    `json:"id"`
    Name  string `json:"name"`
    Email string `json:"email"`
}

func main() {
    // Create a new soda engine
    engine := soda.New()

    // Set API info
    engine.OpenAPI().Info.Title = "User API"
    engine.OpenAPI().Info.Version = "1.0.0"
    engine.OpenAPI().Info.Description = "A simple user management API"

    // Define routes with automatic OpenAPI documentation
    engine.Post("/users", createUser).
        SetSummary("Create a new user").
        SetDescription("Creates a new user account with the provided information").
        SetInput(CreateUserRequest{}).
        AddJSONResponse(201, UserResponse{}, "User created successfully").
        AddJSONResponse(400, nil, "Invalid input data").
        AddJSONResponse(500, nil, "Internal server error").
        OK()

    // Serve interactive documentation
    engine.ServeDocUI("/docs", soda.UISwaggerUI)
    engine.ServeSpecJSON("/openapi.json")
    engine.ServeSpecYAML("/openapi.yaml")

    // Start the server
    engine.App().Listen(":3000")
}

func createUser(c *fiber.Ctx) error {
    input := soda.GetInput[CreateUserRequest](c)

    // Your business logic here
    response := UserResponse{
        ID:    1,
        Name:  input.Name,
        Email: input.Email,
    }

    return c.Status(201).JSON(response)
}
```

## 📖 Documentation

### Struct Tags

Soda uses struct tags to generate OpenAPI specifications. Here are the supported tags:

#### Basic Tags

```go
type Example struct {
    // Path parameters
    ID int `path:"id"`
    // Query parameters
    Search string `query:"search"`
    // Header parameters
    Authorization string `header:"Authorization"`
    // Cookie parameters
    Session string `cookie:"session"`
    // Request body
    Profile Profile `body:"application/json"`
    // OpenAPI specific tags
    Name string `json:"name" oai:"minLength=3,maxLength=50,description=The user's full name"`
    Age  int    `json:"age" oai:"minimum=18,maximum=120,description=User's age in years"`
}
```

#### Validation Tags

```go
type ValidatedRequest struct {
    Email    string `json:"email" oai:"format=email,example=user@example.com"`
    Phone    string `json:"phone" oai:"format=phone,example=+1234567890"`
    Website  string `json:"website" oai:"format=uri,example=https://example.com"`
    // Enum values
    Role     string `json:"role" oai:"enum=admin,user,guest"`
    // String constraints
    Username string `json:"username" oai:"minLength=3,maxLength=20,pattern=^[a-zA-Z0-9]+$"`
    // Numeric constraints
    Score    int    `json:"score" oai:"minimum=0,maximum=100,multipleOf=5"`
    // Array constraints
    Tags     []string `json:"tags" oai:"minItems=1,maxItems=10,uniqueItems=true"`
    // Date/time
    Birthday time.Time `json:"birthday" oai:"format=date-time"`
}
```

### Route Definition

#### Basic Routes

```go
// GET /users/:id
engine.Get("/users/:id", getUser).
    SetSummary("Get user by ID").
    AddJSONResponse(200, UserResponse{}).
    AddJSONResponse(404, nil, "User not found").OK()

// POST /users
engine.Post("/users", createUser).
    SetInput(CreateUserRequest{}).
    AddJSONResponse(201, UserResponse{}).OK()

// PUT /users/:id
engine.Put("/users/:id", updateUser).
    SetInput(UpdateUserRequest{}).
    AddJSONResponse(200, UserResponse{}).OK()

// DELETE /users/:id
engine.Delete("/users/:id", deleteUser).
    AddJSONResponse(204, nil).OK()
```

#### Route Groups

```go
// Create a route group with common settings
api := engine.Group("/api/v1")

// Add common tags
api.AddTags("User Management")

// Add common responses
api.AddJSONResponse(400, ErrorResponse{}, "Bad Request")
api.AddJSONResponse(500, ErrorResponse{}, "Internal Server Error")

// Define routes within the group
api.Get("/users", listUsers).OK()
api.Post("/users", createUser).OK()
api.Get("/users/:id", getUser).OK()
api.Put("/users/:id", updateUser).OK()
api.Delete("/users/:id", deleteUser).OK()
```

#### Security

```go
// Add JWT security to the entire API
engine.AddSecurity("jwt", soda.NewJWTSecurityScheme("JWT authentication"))
// Add API key security
engine.AddSecurity("api-key", soda.NewAPIKeySecurityScheme("header", "X-API-Key", "API key authentication"))
// Apply security to specific routes
engine.Post("/admin/users", createAdminUser).
    AddSecurity("jwt", soda.NewJWTSecurityScheme()).
    AddJSONResponse(201, AdminUserResponse{}).OK()
```

### Advanced Features

#### Custom Hooks

```go
engine.Post("/users", createUser).
    OnBeforeBind(func(c *fiber.Ctx) error {
        // Validate request before binding
        contentType := c.Get("Content-Type")
        if contentType != "application/json" {
            return fiber.ErrBadRequest
        }
        return nil
    }).
    OnAfterBind(func(c *fiber.Ctx, input any) error {
        // Process input after binding
        userInput := input.(*CreateUserRequest)
        userInput.Email = strings.ToLower(userInput.Email)
        return nil
    })
```

#### Custom Schema Generation

```go
// Implement custom JSON schema generation
type CustomType struct {
    Value string
}

func (c CustomType) JSONSchema(doc *openapi3.T) *openapi3.SchemaRef {
    return openapi3.NewStringSchema().
        WithFormat("custom").
        WithDescription("Custom type with special validation").
        NewRef()
}
```

## 🎯 Usage Examples

### Complete RESTful API

```go
package main

import (
    "github.com/gofiber/fiber/v2"
    "github.com/neo-f/soda/v3"
)

type (
    CreateTodoRequest struct {
        Title       string `json:"title" oai:"minLength=1,maxLength=100"`
        Description string `json:"description" oai:"maxLength=500"`
        Completed   bool   `json:"completed"`
    }

    UpdateTodoRequest struct {
        Title       *string `json:"title,omitempty" oai:"minLength=1,maxLength=100"`
        Description *string `json:"description,omitempty" oai:"maxLength=500"`
        Completed   *bool   `json:"completed,omitempty"`
    }

    TodoResponse struct {
        ID          int    `json:"id"`
        Title       string `json:"title"`
        Description string `json:"description"`
        Completed   bool   `json:"completed"`
        CreatedAt   string `json:"created_at" format:"date-time"`
        UpdatedAt   string `json:"updated_at" format:"date-time"`
    }

    TodoListResponse struct {
        Todos []TodoResponse `json:"todos"`
        Total int            `json:"total"`
    }

    ErrorResponse struct {
        Error string `json:"error"`
    }
)

func main() {
    engine := soda.New()

    // Configure API
    engine.OpenAPI().Info = &openapi3.Info{
        Title:       "Todo API",
        Version:     "1.0.0",
        Description: "A comprehensive todo management API",
    }

    // Add common security
    engine.AddSecurity("bearer", soda.NewJWTSecurityScheme("JWT Bearer Token"))

    // Add common responses
    engine.AddJSONResponse(400, ErrorResponse{}).
        AddJSONResponse(401, ErrorResponse{}).
        AddJSONResponse(500, ErrorResponse{})

    // API routes
    api := engine.Group("/api/v1")
    api.AddTags("Todos")

    // List todos
    api.Get("/todos", listTodos).
        SetSummary("List all todos").
        SetDescription("Retrieve a paginated list of todos").
        AddJSONResponse(200, TodoListResponse{}).OK()

    // Create todo
    api.Post("/todos", createTodo).
        SetSummary("Create a new todo").
        SetInput(CreateTodoRequest{}).
        AddJSONResponse(201, TodoResponse{}).OK()

    // Get todo
    api.Get("/todos/:id", getTodo).
        SetSummary("Get todo by ID").
        AddJSONResponse(200, TodoResponse{}).
        AddJSONResponse(404, ErrorResponse{}).OK()

    // Update todo
    api.Put("/todos/:id", updateTodo).
        SetSummary("Update todo").
        SetInput(UpdateTodoRequest{}).
        AddJSONResponse(200, TodoResponse{}).
        AddJSONResponse(404, ErrorResponse{}).OK()

    // Delete todo
    api.Delete("/todos/:id", deleteTodo).
        SetSummary("Delete todo").
        AddJSONResponse(204, nil).
        AddJSONResponse(404, ErrorResponse{}).OK()

    // Serve documentation
    engine.ServeDocUI("/docs", soda.UISwaggerUI)
    engine.ServeSpecJSON("/openapi.json")
    engine.ServeSpecYAML("/openapi.yaml")

    engine.App().Listen(":3000")
}

// Handler implementations would go here...
```

## 📊 Available UI Options

Soda provides several built-in options for serving interactive API documentation:

```go
// Swagger UI (most popular)
engine.ServeDocUI("/swagger", soda.UISwaggerUI)

// ReDoc (clean and modern)
engine.ServeDocUI("/redoc", soda.UIRedoc)

// RapiDoc (feature-rich)
engine.ServeDocUI("/rapidoc", soda.UIRapiDoc)

// Stoplight Elements (elegant design)
engine.ServeDocUI("/elements", soda.UIStoplightElement)
```

## 🔧 Configuration

### OpenAPI Information

```go
engine.OpenAPI().Info = &openapi3.Info{
    Title:          "Your API Title",
    Version:        "1.0.0",
    Description:    "Detailed API description",
    TermsOfService: "https://example.com/terms",
    Contact: &openapi3.Contact{
        Name:  "API Support",
        Email: "support@example.com",
        URL:   "https://example.com/support",
    },
    License: &openapi3.License{
        Name: "MIT",
        URL:  "https://opensource.org/licenses/MIT",
    },
}

// Add servers
engine.OpenAPI().Servers = openapi3.Servers{
    &openapi3.Server{
        URL:         "https://api.example.com/v1",
        Description: "Production server",
    },
    &openapi3.Server{
        URL:         "https://staging-api.example.com/v1",
        Description: "Staging server",
    },
}
```

### Custom UIRender

You can implement your own documentation UI:

```go
type CustomUIRender struct{}

func (c CustomUIRender) Render(doc *openapi3.T) string {
    // Your custom HTML/JS implementation
    return `<!DOCTYPE html><html>...</html>`
}

// Use it
engine.ServeDocUI("/custom", CustomUIRender{})
```

## 🧪 Testing

Soda includes comprehensive test coverage. Run the tests with:

```bash
go test ./...
```

## 🤝 Contributing

Contributions are welcome! Please feel free to submit a Pull Request. For major changes, please open an issue first to discuss what you would like to change.

Please make sure to update tests as appropriate.

## 📄 License

This project is licensed under the MIT License - see the [LICENSE](LICENSE) file for details.

## 🙏 Acknowledgments

- [Fiber](https://github.com/gofiber/fiber) - Fast HTTP framework for Go
- [kin-openapi](https://github.com/getkin/kin-openapi) - OpenAPI 3 specification implementation
- [Swagger UI](https://swagger.io/tools/swagger-ui/) - Interactive API documentation
- [ReDoc](https://github.com/Redocly/redoc) - OpenAPI/Swagger-generated API Reference Documentation
- [RapiDoc](https://github.com/rapi-doc/RapiDoc) - Web Component for OpenAPI spec
- [Stoplight Elements](https://github.com/stoplightio/elements) - Beautiful API documentation

## 📈 Changelog

See [CHANGELOG.md](CHANGELOG.md) for a detailed history of changes.

## 📞 Support

If you have any questions or need help, please open an issue on the [GitHub repository](https://github.com/neo-f/soda/issues).

 readmeEtag: '"3679b5eacea84d272ced24b6b50a756e621027ac"' readmeLastModified: Wed, 23 Jul 2025 13:58:11 GMT repositoryId: 581433714 description: null created: '2022-12-23T07:08:24Z' updated: '2025-11-26T03:19:08Z' language: Go archived: false stars: 6 watchers: 2 forks: 2 owner: neo-f logo: https://avatars.githubusercontent.com/u/20187646?v=4 license: MIT repoEtag: '"56c4a9a4c9714f016bfe5fd9b3ece429c56b11cccb9db4afa3adc17915e73cbd"' repoLastModified: Wed, 26 Nov 2025 03:19:08 GMT category: Parsers foundInMaster: true oldLocations: - https://github.com/captain-neo/soda - source: openapi3 tags repository: https://github.com/jonatandb/weatherapp_api v3: true repositoryMetadata: base64Readme: >- ### Características generales y técnicas:

- Desarrollado con JavaScript utilizando Node.js.
- Se utiliza el API del servicio de clima [Open Weather Map](https://openweathermap.org/).
- La ciudad actual se detecta con la IP del usuario, utilizando el servicio de [IP-API](https://ip-api.com/).

---

<center>

![desktop](OpenApi_Screenshot.png)

</center>

---

### Esta API provee en formato JSON el estado del tiempo por medio de diferentes endpoints:

- Ruta base:
  - **/v1**
- Endpoints:

  - **/location** (_Devuelve nombre de la ciudad actual según la IP del usuario._)
  - **/current/_city_** (_Devuelve clima según "city", o de la ciudad actual si no se especificó una._)
  - **/forecast/_city_** (_Devuelve pronóstico a 5 días según "city", o de la ciudad actual si no se especificó una._)

    - **Nota**: _El parámetro "city" es opcional. Si no está presente se usa la cuidad correspondiente a la IP del usuario._

  - **/openapi** (_Swagger (openapi): Interfaz web que muestra documentación de la API y permite interactuar con los endpoints_)

---

### Requerimientos:

- [NodeJs v.16](https://nodejs.org/es/)
- API KEY de [Open Weather Map](https://openweathermap.org/) (_Requiere crear cuenta gratuita_)

---

### Miscelaneos:

- Utilizar el archivo **_.env.example_** como modelo para crear un archivo llamado **_.env_** que tenga el siguiente contenido:
  - `PORT` (_Especifica el puerto en que se desea que se ejecute la API._)
  - `IP_API_SERVICE_BASE_URL` _(Especifica la url de la API que provee información sobre la ubicación según la IP proporcionada, viene pre-configurada.)_
  - `WEATHER_SERVICE_API_KEY` _(Especifica la API KEY de Open Weather Map. Requiere registro gratuito.)_

---

### Ejecución:

> - npm install
> - npm start
> - Navegar a http://localhost:3001/v1/openapi si se desea ver e interactuar con la lista de endpoints (_**3001** es el puerto sugerido, se debe configurar en el archivo .env -> **PORT**_)

---

### Tests/Coverage:

> - npm test
> - npm run test:coverage (_Luego abrir el archivo: **backend\coverage\lcov-report\index.html**_)
> - ~~**NOTA** Debido a que tuve problemas para ejecutar los tests, ya que estoy usando módulos de ES con [_Dynamic imports_](https://javascript.info/modules-dynamic-imports) en un proyecto CommonJS, tuve que crear dos nuevos scripts para correr los tests, los cuales utilizan características experimentales de Node JS. (_[Fuente](https://stackoverflow.com/a/61653104/10752198)_). Los nuevos scripts son los siguientes:~~
>   - ~~_npm run test:experimental_~~
>   - ~~_npm run test:coverage:experimental_~~
>     - _(Solucionado instalando versiones anteriores de node-fetch y public-ip totalmente compatibles con CommonJS)_

---

### Pendientes/Bugs:

- [https://github.com/Jonatandb/weather-app-with-fastify/issues](https://github.com/Jonatandb/weather-app-with-fastify/issues)

---

### Sitios investigados:

- https://stackoverflow.com/questions/45053974/requiring-an-async-function-in-nodejs
- https://stackoverflow.com/questions/50974313/node-js-async-module-require
- https://github.com/fastify/fastify/issues/267 Weird behavior with fastify.register, async/await
- https://stackoverflow.com/questions/27906551/node-js-logging-use-morgan-and-winston
- https://www.fastify.io/docs/latest/Reference/Plugins/
- https://www.fastify.io/docs/latest/Guides/Getting-Started/#your-first-plugin
- https://futurestud.io/tutorials/retrieve-a-requests-ip-address-in-node-js
- https://openweathermap.org/api/one-call-api
- https://openweathermap.org/faq#:~:text=OpenWeather%20uses%20Unix%20time%20and,forecast%20and%20historical%20weather%20data
- https://stackoverflow.com/questions/62376115/how-to-obtain-open-weather-api-date-time-from-city-being-fetched
- https://github.com/testing-library/eslint-plugin-testing-library/blob/main/docs/rules/prefer-screen-queries.md
- https://www.youtube.com/watch?v=KYjjtRgg_H0&ab_channel=midudev TESTING en REACT 🧪 ¡Aprende DESDE CERO! Con react-testing-library y Jest (FullStack Bootcamp JS)
- https://code.visualstudio.com/docs/languages/jsconfig What is jsconfig.json?
- https://github.com/microsoft/TypeScript/issues/46700#issuecomment-968084329 Auto import doesn't work unless it already imported from another file #46700
- https://github.com/microsoft/vscode/issues/132299 Auto Import not working (Typescript ) after latest update
- https://openweathermap.org/weather-conditions
- https://www.freecodecamp.org/news/how-to-create-a-react-app-with-a-node-backend-the-complete-guide/
- https://github.com/fastify/fastify-cors
- https://github.com/OAI/OpenAPI-Specification/issues/93 - Optional path parameters 💔
- https://www.fastify.io/docs/latest/Reference/Validation-and-Serialization/
- https://editor.swagger.io/
- https://swagger.io/docs/specification/2-0/describing-parameters/
- https://github.com/fastify/fastify/pull/2607#discussion_r501120298 - The optional chain operator /:opt?
- https://ajv.js.org/json-schema.html
- https://www.fastify.io/docs/latest/Reference/Validation-and-Serialization/#validation
- https://github.com/fastify/fastify/blob/main/docs/Reference/Request.md
- https://stackoverflow.com/questions/60372790/node-v13-jest-es6-native-support-for-modules-without-babel-or-esm
- https://stackoverflow.com/questions/31629389/how-to-use-eslint-with-jest
- https://jackfiallos.com/entendiendo-code-covereage-usando-jest
- https://jestjs.io/es-ES/search?q=coverage
- https://github.com/micromatch/micromatch
- https://jestjs.io/docs/configuration#collectcoveragefrom-array
- https://joshtronic.com/2017/10/24/configuring-jest-to-show-code-coverage-for-all-of-your-files/
- https://www.fastify.io/docs/latest/Guides/Testing/#benefits-of-using-fastifyinject
- https://github.com/fastify/fastify/blob/main/test/listen.test.js
- https://www.fastify.io/docs/latest/Reference/Server/#server
- https://jestjs.io/es-ES/docs/getting-started
- https://github.com/zentered/demo-async-fastify-with-jest ♥
- https://zentered.co/articles/setup-async-fastify-with-jest-test/ ♥♥
- https://javascript.plainenglish.io/how-to-build-a-reliable-authentication-api-with-fastify-c5a24bf8cd41 mongodb jwt bcrypt
- https://stackoverflow.com/questions/tagged/fastify+jestjs?tab=Newest
- https://stackoverflow.com/questions/71545883/how-to-setup-and-teardown-a-web-server-with-jest
- https://fettblog.eu/void-in-javascript-and-typescript/
- https://daily-dev-tips.com/posts/common-jest-matchers/
- https://dev.to/wolfejw86/setup-a-fastify-app-with-jest-tests-the-right-way-43ih
- https://dev.to/itsrennyman/how-i-structure-my-fastify-application-1j93#fifth-unit-testing
- https://node-tap.org/docs/getting-started/
- https://www.fastify.io/docs/latest/Guides/Testing/#testing-with-a-running-server
- https://ajv.js.org/json-schema.html#type
- https://stackoverflow.com/questions/38933973/how-to-provide-example-value-to-a-response-body-of-content-type-text-html-in-sw
- https://github.com/testdouble/teenytest
- https://stackoverflow.com/questions/40795836/how-do-you-use-the-files-and-directories-properties-in-package-json
- https://stackoverflow.com/questions/38009315/redirecting-from-aws-api-gateway-using-response-302
- https://stackoverflow.com/questions/60332687/how-to-define-two-different-responses-for-same-response-code-with-302-redirect
- https://github.com/fastify/fastify-swagger
- https://lenguajejs.com/automatizadores/introduccion/commonjs-vs-es-modules/
- https://www.youtube.com/watch?v=JD6VNRdGl98&ab_channel=LeonardoKuffo REST y RESTful APIs | Te lo explico en 5 minutos!
- https://www.youtube.com/watch?v=99YMeCBk3jw&ab_channel=LazyLoading Arquitectura monolítica vs microservicios
- https://www.youtube.com/watch?v=QDXMoDbOd5s&ab_channel=MakeitReal Microservicios vs Monolito
- https://www.youtube.com/watch?v=EKseAAm4pvY Creando aplicaciones web con Node.js y Fastify
- https://www.youtube.com/watch?v=YfN9hElekuM&ab_channel=LeiferMendez CORS
- https://eslint.org/docs/user-guide/getting-started
- https://github.com/prettier/eslint-config-prettier
- https://lenguajejs.com/javascript/caracteristicas/eslint/
- https://github.com/Cristiandi/demo-fastify/blob/master/src/environment.js --------- Interesante manejo de diferentes .env (local, development, etc)
- https://platzi.com/tutoriales/1339-fundamentos-javascript/2181-como-usar-eslint-y-prettier-para-mejorar-tu-codigo-javascript/
- https://daily-dev-tips.com/posts/lets-talk-about-software-testing/
- https://daily-dev-tips.com/posts/adding-jest-test-to-a-project/?utm_source=Daily+Dev+Tips&utm_medium=email&utm_campaign=mailinglist
- https://www.conventionalcommits.org/es/v1.0.0/#especificaci%c3%b3n
- https://github.com/petervanderdoes/gitflow-avh#git-flow-usage
- https://docs.github.com/es/get-started/quickstart/github-flow
- https://tech-wiki.online/es/node-setimmediate.html
- https://www.cual-es-mi-ip.net/
- https://www.npmjs.com/package/ip-address
- https://stackoverflow.com/questions/3162457/how-to-check-with-javascript-if-connection-is-local-host
- https://stackoverflow.com/questions/20553554/node-js-return-hostname
- https://stackoverflow.com/questions/21987311/check-is-nodejs-connection-come-from-localhost
- https://www.npmjs.com/package/is-localhost-url
- https://blog.devgenius.io/server-side-development-with-fastify-async-and-await-and-route-prefixes-d5704eb5206
- https://jaywolfe.dev/rapidly-build-a-nodejs-rest-api-with-fastify-postgresql-and-swagger-documentation/
- https://github.com/fastify/example -------------- Collection of Fastify projects
- https://github.com/fastify/fastify
- https://stackoverflow.com/questions/70478820/unable-to-deploy-nodejs-fastify-app-in-azure-app-service-linux
- https://dev.to/ruanmartinelli/how-to-use-schemas-on-fastify-for-fun-and-profit-25e9
- https://dev.to/dsalinasgardon/the-6-best-javascript-frameworks-to-use-in-2022-4k5k
- https://dev.to/eomm/why-should-i-prefer-fastify-to-expressjs-44c4
- https://www.fastify.io/docs/latest/Reference/Routes/#route-prefixing
- https://github.com/useaurora/api/blob/main/app.js --------------------- Buena división entre app.js y server.js \*\*
- https://github.com/fastify/fastify-autoload
- https://dev.to/itsrennyman/how-i-structure-my-fastify-application-1j93
- https://dev.to/thomasbnt/perspective-api-20al
- https://dev.to/thomasbnt/create-a-fastify-server-23lg
- https://dev.to/hypeddev/es-modules-in-fastify-349f
- https://github.com/fastify/fastify-express
- https://json-schema.org/learn/getting-started-step-by-step
- https://dev.to/cristiandi/demo-api-using-fastify-48jo
- https://dev.to/whitep4nth3r/how-i-massively-improved-my-website-performance-by-using-the-right-tool-for-the-job-23cl
- https://www.freecodecamp.org/espanol/news/que-es-jamstack/
- https://dev.to/dailydevtips1/building-a-fastify-nodejs-server-296g
- https://www.fastify.io/docs/latest/Guides/Plugins-Guide/
- https://github.com/gquittet/graceful-server
- https://github.com/fastify/fastify-sensible
- https://github.com/fastify/fastify-nextjs
- https://github.com/fastify/fastify-helmet
- https://github.com/fastify/fastify-env
- https://github.com/fastify/fastify-swagger
- https://www.fastify.io/ecosystem/
- https://www.fastify.io/docs/latest/Guides/Testing/
- https://www.fastify.io/docs/v3.0.x/Routes/
- https://www.fastify.io/docs/latest/Guides/Getting-Started/
- https://www.fastify.io/
- https://xp123.com/articles/3a-arrange-act-assert/
- https://martinfowler.com/bliki/GivenWhenThen.html
- https://dev.to/stealthmusic/everything-thats-not-tested-will-break-1adg
- https://dev.to/codingpizza/what-is-a-unit-test-1e1m
- https://github.com/Huachao/vscode-restclient
- https://stackoverflow.com/questions/69063074/rest-client-vscode-extention-is-not-sending-json-data
- https://dev.to/tusharpandey13/getting-on-with-es6-nodejs-eslint-without-babel-4ip7
- https://www.npmjs.com/package/morgan
- https://ichi.pro/es/funciones-emocionantes-de-javascript-es2021-32885188220171
- https://babeljs.io/docs/en/babel-cli
- https://www.freecodecamp.org/news/setup-babel-in-nodejs/
- https://www.freecodecamp.org/espanol/news/que-es-babel/
- https://hacks.mozilla.org/2015/08/es6-in-depth-modules/
- https://blog.logrocket.com/how-to-use-ecmascript-modules-with-node-js/
- https://stackoverflow.com/questions/69041454/error-require-of-es-modules-is-not-supported-when-importing-node-fetch
- https://es.stackoverflow.com/questions/484949/problema-con-fetch-en-node
- https://stackoverflow.com/questions/6784753/passing-route-control-with-optional-parameter-after-root-in-express
- https://stackoverflow.com/questions/60205891/import-json-extension-in-es6-node-js-throws-an-error
- https://kinsta.com/es/base-de-conocimiento/http-304/
- https://es.stackoverflow.com/questions/199109/obtener-la-ip-del-cliente-en-node-js
- https://expressjs.com/en/4x/api.html#trust.proxy.options.table
- https://www.mickpatterson.com.au/blog/api-versioning-with-nodejs-and-express/
- https://stackoverflow.com/questions/51513715/node-js-rest-api-versioning-the-right-way/51514184#51514184
- https://www.atlassian.com/es/git/tutorials/comparing-workflows/gitflow-workflow
 readmeEtag: '"1b28795dd53d40f74fa1de911a8b366dea9c458a"' readmeLastModified: Mon, 05 Jun 2023 03:58:31 GMT repositoryId: 479925177 description: API of https://github.com/Jonatandb/weatherapp_frontend created: '2022-04-10T05:46:27Z' updated: '2022-04-13T23:06:43Z' language: JavaScript archived: false stars: 0 watchers: 1 forks: 0 owner: Jonatandb logo: https://avatars.githubusercontent.com/u/8082172?v=4 repoEtag: '"8b09f8318de6bb71e1f2a602c79c20d36eee377b7e57bdf6af34632a2599bd09"' repoLastModified: Wed, 13 Apr 2022 23:06:43 GMT foundInMaster: true category: Testing id: d3f2cdb54b053544ddcfca6ed3337e13 - source: openapi3 tags repository: https://github.com/manju4ever/fastity-swagger-generate v3: true repositoryMetadata: base64Readme: >- IyBmYXN0aWZ5LXN3YWdnZXItZ2VuZXJhdGUKCkdlbmVyYXRlIFN3YWdnZXIvT3BlbkFQSSBkZWZpbml0aW9ucyB3aXRob3V0IHJ1bm5pbmcgZmFzdGlmeSBhcHAgIQoKPCEtLSB0b2MgLS0+CgotIFtmYXN0aWZ5LXN3YWdnZXItZ2VuZXJhdGVdKCNmYXN0aWZ5LXN3YWdnZXItZ2VuZXJhdGUpCiAgLSBbSW5zdGFsbF0oI2luc3RhbGwpCiAgLSBbRGlzY2xhaW1lcl0oI2Rpc2NsYWltZXIpCiAgLSBbVXNhZ2VdKCN1c2FnZSkKICAtIFtBUEldKCNhcGkpCiAgICAtIFtPcHRpb25zXSgjb3B0aW9ucykKCjwhLS0gdG9jc3RvcCAtLT4KCiMjIEluc3RhbGwKCmBucG0gaSBmYXN0aWZ5LXN3YWdnZXItZ2VuZXJhdGVgCgojIyBEaXNjbGFpbWVyCgotIE9ubHkgc3VwcG9ydHMgY29uZmlndXJhdGlvbiBiYXNlZCByb3V0ZXMsIGkuZSBjb2xsZWN0aW9uIG9mIHJvdXRlIGRlZmluaXRpb25zCi0gVGhpcyBsaWJyYXJ5IGhhcyBubyBuZXcgY29udGVudCBhZGRlZAotIFRoaXMgbGlicmFyeSBoYXMganVzdCB0d2Vha2VkIHRoZSBvcmlnaW5hbCBbZmFzdGlmeS1zd2FnZ2VyXShodHRwczovL2dpdGh1Yi5jb20vZmFzdGlmeS9mYXN0aWZ5LXN3YWdnZXIpCgojIyBVc2FnZQoKMS4gQ3JlYXRlIGEgYGdlbmVyYXRlLmpzYCBmaWxlIHdoaWNoIGxvb2tzIGxpa2UKCmBgYGphdmFzY3JpcHQKY29uc3QgZmFzdGlmeVN3YWdHZW4gPSByZXF1aXJlKCJmYXN0aWZ5LXN3YWdnZXItZ2VuZXJhdGUiKTsKY29uc3QgUm91dGVzID0gcmVxdWlyZSgiLi9yb3V0ZXMiKTsKY29uc3Qgb3B0cyA9IHt9OwoKLy8gZ2VuZXJhdGUgc3dhZ2dlciBkZWZpbml0aW9ucwpmYXN0aWZ5U3dhZ0dlbihvcHRzLCBSb3V0ZXMsIChlcnIsIGRlZmluaXRpb25zKSA9PiB7CiAgLy8gV3JpdGUgdG8gYSBmaWxlCiAgcmVxdWlyZSgiZnMiKQogICAgLmNyZWF0ZVdyaXRlU3RyZWFtKCIuL2FwcC5zd2FnLmpzb24iKQogICAgLndyaXRlKEpTT04uc3RyaW5naWZ5KGRlZmluaXRpb25zKSk7Cn0pOwpgYGAKCjIuIExldCdzIHNheSB5b3UgaGF2ZSByb3V0ZXMgZGVmaW5lZCBpbiBgcm91dGVzLmpzYAoKYGBgamF2YXNjcmlwdApjb25zdCByb3V0ZXMgPSBbCiAgewogICAgdXJsOiAiL3VzZXJzLzppZCIsCiAgICBtZXRob2Q6ICJHRVQiLAogICAgc2NoZW1hOiB7CiAgICAgIHBhcmFtczogewogICAgICAgIHR5cGU6ICJvYmplY3QiLAogICAgICAgIHByb3BlcnRpZXM6IHsKICAgICAgICAgIGlkOiB7CiAgICAgICAgICAgIHR5cGU6ICJudW1iZXIiLAogICAgICAgICAgfSwKICAgICAgICB9LAogICAgICB9LAogICAgfSwKICB9LApdOwoKbW9kdWxlLmV4cG9ydHMgPSByb3V0ZXM7IC8vIHNob3VsZCBiZSBhbiBpdGVyYWJsZSByb3V0ZXMKYGBgCgojIyBBUEkKCiMjIyBPcHRpb25zCgpbUmVmZXIgaGVyZSBmb3IgbW9yZV0oaHR0cHM6Ly9naXRodWIuY29tL2Zhc3RpZnkvZmFzdGlmeS1zd2FnZ2VyI3JlZ2lzdGVyLW9wdGlvbnMpCg== readmeEtag: '"b7f266e89a466436937bac711a6b40b680145d91"' readmeLastModified: Thu, 22 Dec 2022 14:50:27 GMT repositoryId: 475559400 description: >- Generate Swagger or OpenAPI Specification (JSON/YML) for Fastify server routes without running the server created: '2022-03-29T17:56:37Z' updated: '2023-12-30T08:19:11Z' language: JavaScript archived: false stars: 2 watchers: 1 forks: 0 owner: manju4ever logo: https://avatars.githubusercontent.com/u/9355984?v=4 license: MIT repoEtag: '"0b65432784ef920d54793991bfaa1c3969e9df1cbdc2fb01e583a724d8075632"' repoLastModified: Sat, 30 Dec 2023 08:19:11 GMT foundInMaster: true category: Parsers id: ddff5eb57bb20f920637d0ffcb74b972 oldLocations: - https://github.com/manju4ever/fastify-swagger-generate - source: https://openapi.tools/ name: Bump category: Documentation link: https://bump.sh language: SaaS source_description: >- Bump generates elegant documentation and changelogs from your OpenAPI specifications. Git diff, for your API. v2: true v3: true v3_1: true foundInMaster: true id: aa06bbbc595e3401e9dd7b0a8ac92d63 - source: https://openapi.tools/ name: RepreZen API Studio category: GUI Editors language: Java link: https://www.reprezen.com/ repository: null source_description: > RepreZen API Studio is an integrated workbench that brings API-first design into focus for your whole team, harmonizes your API designs, and generates APIs that click into client apps. v2: true v3: true foundInMaster: true id: 824559c965aaed65843131864b0adb5d - source: https://openapi.tools/ name: APIGit category: - Documentation - Mock language: SaaS link: https://apigit.com source_description: >- native Git based collaboration platform for API document, Design, Mock and Sharing! v2: true v3: true v3_1: true id: 15353b0357b3f4e9edd7657cd4dd73e3 foundInMaster: true - source: openapi3 tags repository: https://github.com/boro23-wq/case-manager-api v3: true id: 35c8cc9043f70001d15390bc18b7b21f repositoryMetadata: base64Readme: >- IyMgQ2FzZSBNYW5hZ2VyIEFQSQoKQVBJIGJ1aWx0IHVzaW5nICoqTmVzdC5qcyoqLCAqKlR5cGVzY3JpcHQqKiwgKipQcmlzbWEqKiBmb3IgT1JNLCBhbmQgKipQbGFuZXRTY2FsZSoqIChNeVNRTCkgdGhhdCBwZXJmb3JtcyBvcGVyYXRpb24gb24gY2FzZSBtYW5hZ2VycywgY2FzZXMsIGFuZCByZWxhdGVkIGNhc2Ugbm90ZXMuIFRoaXMgQVBJIGlzIGEgcGFydCBvZiB0aGUgcHJvamVjdCBDUyA2NzMgLSBDYXJlIE1hbmFnZW1lbnQgYW5kIENvLW9yZGluYXRpb24uCgoKPHAgIGFsaWduPSJjZW50ZXIiPgogIDxhICBocmVmPSJodHRwOi8vbmVzdGpzLmNvbS8iICB0YXJnZXQ9ImJsYW5rIj48aW1nICAJc3JjPSJodHRwczovL25lc3Rqcy5jb20vaW1nL2xvZ29fdGV4dC5zdmciICB3aWR0aD0iMzIwIiAgYWx0PSJOZXN0IExvZ28iIC8+PC9hPgo8L3A+CgpOZXN0LmpzIGlzIGEgcHJvZ3Jlc3NpdmUgPGEgIGhyZWY9Imh0dHA6Ly9ub2RlanMub3JnIiAgdGFyZ2V0PSJfYmxhbmsiPk5vZGUuanM8L2E+IGZyYW1ld29yayBmb3IgYnVpbGRpbmcgZWZmaWNpZW50IGFuZCBzY2FsYWJsZSBzZXJ2ZXItc2lkZSBhcHBsaWNhdGlvbnMuCgojIyBBY3RpdmUgUmVjb3JkIFBhdHRlcm4KVGhlIENhc2UgTWFuYWdlciBBUEkgaXMgYnVpbHQgdXNpbmcgdGhlIEFjdGl2ZSBSZWNvcmQgUGF0dGVybi4KClVzaW5nIHRoZSBBY3RpdmUgUmVjb3JkIGFwcHJvYWNoLCBvbmUgY2FuIGRlZmluZSBhbGwgdGhlaXIgcXVlcnkgbWV0aG9kcyBpbnNpZGUgdGhlIG1vZGVsIGl0c2VsZiwgYW5kICBzYXZlLCByZW1vdmUsIGFuZCBsb2FkIG9iamVjdHMgdXNpbmcgbW9kZWwgbWV0aG9kcy4KClNpbXBseSwgdGhlIEFjdGl2ZSBSZWNvcmQgcGF0dGVybiBpcyBhbiBhcHByb2FjaCB0byBhY2Nlc3MgZGF0YWJhc2Ugd2l0aGluIHRoZSBkZWZpbmVkIG1vZGVscy4gWW91IGNhbiByZWFkIG1vcmUgYWJvdXQgdGhlIEFjdGl2ZSBSZWNvcmQgcGF0dGVybiBvbiBbV2lraXBlZGlhXShodHRwczovL2VuLndpa2lwZWRpYS5vcmcvd2lraS9BY3RpdmVfcmVjb3JkX3BhdHRlcm4pLgoKIyMgSW5zdGFsbGF0aW9uCgoxLiBDbG9uZSB0aGUgcmVwb3NpdG9yeSB0byB5b3VyIGxvY2FsIG1hY2hpbmUuCmBgYGJhc2gKJCBnaXQgY2xvbmUgaHR0cHM6Ly9naXRodWIuY29tL0Jvcm8yMy13cS9jYXNlLW1hbmFnZXItYXBpLmdpdApgYGAKMi4gQ2hhbmdlIGRpcmVjdG9yeSBpbnRvIHRoZSByZXBvc2l0b3J5IHlvdSBqdXN0IGNsb25lZC4KYGBgYmFzaAokIGNkIGNhc2UtbWFuYWdlci1hcGkKYGBgCjMuIE9wZW4gdGhlIHJlcG9zaXRvcnkgaW4geW91ciBmYXZvcml0ZSBjb2RlIGVkaXRvci4gSSdtIHVzaW5nIFZTQ29kZS4gVGhlIHNob3J0Y3V0IHRvIG9wZW4gYSBkaXJlY3RvcnkgaW4gVlNDb2RlIGlzOgpgYGBiYXNoCiQgY29kZSAuCmBgYApQbGVhc2UgbWFrZSBzdXJlIHlvdSBhcmUgaW5zaWRlIHRoZSBkaXJlY3RvcnkuCgo0LiBGaW5hbGx5LCBydW4gdGhlIGNvbW1hbmQgYmVsb3cgYmFzZWQgb24geW91ciBwYWNrYWdlIG1hbmFnZXI6CmBgYGJhc2gKIyB5YXJuIAokIHlhcm4KT1IKIyBucG0KJCBucG0gaW5zdGFsbApgYGAKCiMjIEVudmlyb25tZW50IHZhcmlhYmxlcwoKUGxlYXNlIGNyZWF0ZSBhIGAuZW52YCBvciBgLmVudi5sb2NhbGAgZmlsZSB0byBzZXQgdGhlIFtQbGFuZXRTY2FsZV0oaHR0cHM6Ly9wbGFuZXRzY2FsZS5jb20vKSBkYXRhYmFzZSBzdHJpbmcuIFRoZSBzdHJpbmcgd291bGQgbG9vayBzb21ldGhpbmcgbGlrZSB0aGlzOgoKYGBgYmFzaApEQVRBQkFTRV9VUkw9J215c3FsOioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKicKYGBgCgojIyBSdW5uaW5nIHRoZSBhcHAKYGBgYmFzaAojIGRldmVsb3BtZW50CiQgbnBtIHJ1biBzdGFydAoKIyB3YXRjaCBtb2RlCiMgV2F0Y2ggbW9kZSB3aWxsIGJhc2ljYWxseSB3YXRjaCBmb3IgYW55IGNoYW5nZXMgdG8geW91ciBjb2RlIGFuZCByZXN0YXJ0IHRoZSBzZXJ2ZXIgYWNjb3JkaW5nbHkgd2hpY2ggaXMgdmVyeSBjb252ZW5pZW50LgokIG5wbSBydW4gc3RhcnQ6ZGV2CgojIHByb2R1Y3Rpb24gbW9kZQokIG5wbSBydW4gc3RhcnQ6cHJvZApgYGAKCiMjIFRlc3QKSSBoYXZlbid0IGltcGxlbWVudGVkIGFueSB0ZXN0cyB5ZXQsIGJ1dCBvbmNlIHRoZSB0ZXN0cyBhcmUgYWRkZWQgeW91J2QgaGF2ZSBtb3JlIGluZm9ybWF0aW9uIG9uIGhvdyB0byBydW4gaXQuCiAgCgpgYGBiYXNoCiMgdW5pdCB0ZXN0cwokIG5wbSBydW4gdGVzdAoKIyBlMmUgdGVzdHMKJCBucG0gcnVuIHRlc3Q6ZTJlCgojIHRlc3QgY292ZXJhZ2UKJCBucG0gcnVuIHRlc3Q6Y292CmBgYAoKIyMgUmVzb3VyY2VzCjEuIFtOZXN0LmpzIERvY3VtZW50YXRpb25dKGh0dHBzOi8vZG9jcy5uZXN0anMuY29tLykKMi4gW1ByaXNtYSBEb2N1bWVudGF0aW9uXShodHRwczovL3d3dy5wcmlzbWEuaW8vZG9jcy8pCjMuIFtQbGFuZXRTY2FsZSBEb2N1bWVudGF0aW9uXShodHRwczovL3BsYW5ldHNjYWxlLmNvbS9kb2NzKQoKIyMgU3RheSBpbiB0b3VjaAoKLSBBdXRob3IgLSBbU2ludHUgQm9yb10oaHR0cHM6Ly9zYm9yby52ZXJjZWwuYXBwLykKLSBCbG9nIC0gW2h0dHBzOi8vc2Jvcm8udmVyY2VsLmFwcC9ibG9nXShodHRwczovL3Nib3JvLnZlcmNlbC5hcHAvYmxvZykKLSBMaW5rZWRpbiAtIFtTaW50dSBCb3JvXShodHRwczovL3d3dy5saW5rZWRpbi5jb20vaW4vc2ludHUtYm9yby8pCgojIyBMaWNlbnNlClRoaXMgcHJvamVjdCBpcyBbTUlUIGxpY2Vuc2VkXShMSUNFTlNFKS4= readmeEtag: '"f8457def7596468713b8073dbfa3c42c62cb052e"' readmeLastModified: Thu, 03 Nov 2022 18:57:37 GMT repositoryId: 560106244 description: >- Implementation of the REST API using Nest.js, Typescript, Prisma, and PlanetScale that performs operation on case managers, cases, and related case notes, for the project CS 673 - Care Management and Co-ordination. created: '2022-10-31T18:46:18Z' updated: '2022-11-04T04:04:24Z' language: TypeScript archived: false stars: 0 watchers: 1 forks: 0 owner: Boro23-wq logo: https://avatars.githubusercontent.com/u/62204944?v=4 license: MIT repoEtag: '"10a215b2d20439904e2b12ae6dc2c9d9ff125238af24d3f8c61843789bce8b56"' repoLastModified: Fri, 04 Nov 2022 04:04:24 GMT category: - Code Generators - Server Implementations foundInMaster: true - source: https://openapi.tools/ name: MicroTS category: - Server - Server Implementations language: Node.js link: https://www.npmjs.com/package/microts repository: https://github.com/tomi-vanek/microts source_description: Take an OpenAPI description and generate TypeScript projects via Docker. v2: true v3: false repositoryMetadata: base64Readme: >- IyBNaWNyb1RTIGNvZGUgZ2VuZXJhdG9yIGZvciBtaWNyb3NlcnZpY2VzCgpNaWNyb3NlcnZpY2UgY29kZSBnZW5lcmF0b3Igd2l0aCBpbnRlcmZhY2UtZmlyc3QgYXBwcm9hY2g6IGZyb20gKipPcGVuQVBJIC0gU3dhZ2dlcioqIFJFU1QgQVBJIHNwZWNpZmljYXRpb24gaXMgZ2VuZXJhdGVkIGNvbXBsZXRlIHByb2plY3Qgc2tlbGV0b24gd2l0aCBfVHlwZVNjcmlwdF8gY29kZSwgdGVzdHMgYW5kIF9Eb2NrZXJfIGNvbmZpZ3VyYXRpb24uCgpHZW5lcmF0ZWQgY29kZSBoYXMgdGhlIGFtYml0aW9uIHRvIG1pbmltaXplIGltcGxlbWVudGF0aW9uIHRpbWUgZm9yIG5ldyBtaWNyb3NlcnZpY2VzLgoKVGhlIF9vcGVuYXBpLW1pY3JvLXRzXyBnZW5lcmF0b3IgaXMgYSBzaW1wbGUgIm9uZS1zaG90IiBwcm9qZWN0IGluaXRpYWxpemF0aW9uIHRvb2wgLSBhZnRlciB0aGUgY29kZSBpcyBnZW5lcmF0ZWQsIHRoZSBzZXJ2aWNlIGZ1bmN0aW9uYWxpdHkgaXMgaW1wbGVtZW50ZWQgd2l0aCB0cmFkaXRpb25hbCBtYW51YWwgY29kaW5nLgoKIyMgUXVpY2sgc3RhcnQKCjEuIEluc3RhbGwgdGhlIGdlbmVyYXRvciB3aXRoIGBucG0gaSAtZyBtaWNyb3RzYAoxLiBDcmVhdGUgYSBuZXcgcHJvamVjdCBkaXJlY3RvcnkuIEdvIHRvIHRoZSBuZXcgcHJvamVjdCB3b3JrIGRpcmVjdG9yeSB3aXRoIGBjZCBbTkVXLVBST0pFQ1RdYC4KMS4gQ3JlYXRlIG5ldyBtaWNyb3NlcnZpY2Ugc2NoZW1hIHdpdGggZGVmYXVsdCBuYW1lIGBzd2FnZ2VyLnlhbWxgIGluIHJvb3Qgb2YgdGhlIHByb2plY3QuCjEuIEdlbmVyYXRlIG1pY3Jvc2VydmljZSBjb2RlIHdpdGggY29tbWFuZCBgbWljcm90c2Agd2l0aCBkZWZhdWx0IHBvcnQgMzAwMC4KMS4gTnN0YWxsIGRlcGVuZGVuY2llcyB3aXRoIGBucG0gaWAgYW5kIHN0YXJ0IHRoZSBtaWNyb3NlcnZpY2Ugd2l0aCBgbnBtIHN0YXJ0YC4KMS4gT3BlbiB0aGUgbWljcm9zZXJ2aWNlIGRlYnVnZ2luZyB1c2VyIGludGVyZmFjZSBpbiBicm93c2VyIHdpdGggVVJMIGBsb2NhbGhvc3Q6MzAwMC9bYmFzZS1wYXRoXS91aWAgKGJhc2UgcGF0aCBpcyBkZWZpbmVkIGJ5IHNjaGVtYSkuCgojIyBDb2RlIGdlbmVyYXRpb24gaW4gZGV0YWlsCgoxLiBDcmVhdGUgYSBuZXcgcHJvamVjdCBkaXJlY3RvcnkgKGNyZWF0ZSBhIHByb2plY3QgaW4gR2l0SHViIG9yIG90aGVyIFZDUyBhbmQgY2xvbmUpLiBHbyB0byB0aGUgbmV3IHByb2plY3Qgd29yayBkaXJlY3Rvcnkgd2l0aCBgY2QgW05FVy1QUk9KRUNUXWAuCjEuIENyZWF0ZSBuZXcgbWljcm9zZXJ2aWNlIHNjaGVtYSBpbiByb290IG9mIHRoZSBwcm9qZWN0LiBTdXBwcm90ZWQgc2NoZW1hIGZvcm1hdHMgYXJlIFtPcGVuQVBJIDIuMF0oaHR0cHM6Ly9naXRodWIuY29tL09BSS9PcGVuQVBJLVNwZWNpZmljYXRpb24vYmxvYi9tYXN0ZXIvdmVyc2lvbnMvMi4wLm1kKSBpbiBib3RoIFlBTUwgYW5kIEpTT04gZm9ybWF0LgoxLiBHZW5lcmF0ZSBtaWNyb3NlcnZpY2UgY29kZSB3aXRoIGNvbW1hbmQgYG1pY3JvdHMgLXAgUE9SVCAtcyBTQ0hFTUFgLiBQYXJhbWV0ZXIgUE9SVCBkZWZpbmVzIHRoZSBkZWZhdWx0IHBvcnQgb24gd2hpY2ggdGhlIHNlcnZlciB3aWxsIGxpc3RlbiAoaWYgbm90IHNldCwgZGVmYXVsdCBwb3J0IDMwMDAgd2lsbCBiZSB1c2VkIGZvciBjb2RlIGdlbmVyYXRpb24pLiBQYXJhbWV0ZXIgU0NIRU1BIGlzIHRoZSBuYW1lIG9mIHRoZSBzY2hlbWEgLSBtYXkgYmUgd2l0aCBhYnNvbHV0ZSBvciByZWxhdGl2ZSBwYXRoLCBpZiBzY2hlbWEgaXMgbm90IGluIHdvcmtpbmcgZGlyZWN0b3J5LiBJZiBzY2hlbWEgaXMgbm90IHNldCwgZ2VuZXJhdG9yIHRyaWVzIHRvIG9wZW4gYHN3YWdnZXIueWFtbGAgZmlsZSBmb3IgQVBJIGRlZmluaXRpb24uCjEuIFJlYWQgdGhlIF9OZXh0IHN0ZXBzXyBpbiB0aGUgY29uc29sZSBhbmQgZmFtaWxpYXJpemUgd2l0aCB0aGUgZ2VuZXJhdGVkIG1pY3Jvc2VydmljZSBzZXJ2ZXIuCjEuIE1vcmUgaW5mb3JtYXRpb24gYWJvdXQgdGhlIG1pY3Jvc2VydmljZSBpcyBpbiB0aGUgZ2VuZXJhdGVkIGBSRUFETUUubWRgIGZpbGUuCjEuIEFkZCByZXBvc2l0b3J5IGFuZCBsaWNlbnNlIGZpZWxkcyB0byB0aGUgZ2VuZXJhdGVkIGBwYWNrYWdlLmpzb25gLgoxLiBUaGUgc291cmNlIHNjaGVtYSB3YXMgY29waWVkIChhbmQgaWYgbmVlZGVkIC0gY29udmVydGVkKSB0byBgc3JjL2NvbmYvc3dhZ2dlci55YW1sYCBmaWxlLiBUaGUgc291cmNlIHNjaGVtYSBjYW4gYmUgZGVsZXRlZCAtIGFzIGl0IGlzIG5vdCB1c2VkIGJ5IHRoZSBzZXJ2ZXIuCjEuIFNlYXJjaCBmb3IgYFRPRE9gIGluIGNvZGUsIGFuZCBpbXBsZW1lbnQgdGhlIGZ1bmN0aW9uYWxpdHkuCgpGb3IgYWxsIGNvbW1hbmQgbGluZSBwcm9wZXJ0aWVzIG9mIHRoZSBgbWljcm90c2AgY29kZSBnZW5lcmF0b3IgdXNlIHRoZSBjb21tYW5kIGBtaWNyb3RzIC1oYC4KCiMjIE1pY3Jvc2VydmljZSBkZXZlbG9wbWVudAoKMS4gRm9yIGRlYnVnZ2luZyBydW4gdGhlIG1pY3Jvc2VydmljZSB3aXRoIGBucG0gcnVuIGRldmAuCjEuIEFmdGVyIHNhdmluZyBhbnkgY2hhbmdlIHRoZSBzb3VyY2UgY29kZSBpcyBjb21waWxlZCB0byBydW50aW1lIGZvcm0gaW4gYC9kaXN0YCBkaXJlY3RvcnkgYW5kIHRoZSBzZXJ2ZXIgaXMgcmVzdGFydGVkLgoxLiBUaGUgbWljcm9zZXJ2aWNlIFVJIGhlbHBzIGJ5IGRlYnVnZ2luZyB0aGUgc2VydmljZS4gSXQgc2hvd3MgYWxzbyBgY3VybGAgY29tbWFuZHMgdG8gY2FsbCB0aGUgYWN0aW9ucyBpbiB0aGUgc2VydmljZS4KCiMjIE9wZW5BUEkgLyBTd2FnZ2VyIHNjaGVtYSBhdXRob3JpbmcKCk1pY3Jvc2VydmljZSBpcyBkZWNsYXJlZCB3aXRoIE9wZW5BUEkgMi4wIChTd2FnZ2VyKSBgc3dhZ2dlci55YW1sYCBzY2hlbWEgd2l0aCBBUEkgZGVmaW5pdGlvbiAtIHlvdSBjYW4gdXNlIFN3Z2dlciBlZGl0b3IgZm9yIHNjaGVtYSBkZWZpbml0aW9uLiBUaGlzIGRlc2lnbiBzdGVwIGlzIGNydWNpYWwgZm9yIGZ1cnRoZXIgcXVhbGl0eSBhbmQgdXNhZ2Ugc2ltcGxpY2l0eSBvZiB0aGUgbmV3IEFQSS4gRGVzaWduIGlzIGJlc3QgbWFkZSBpbiBkaXNjdXNzaW9uLiBJdCBtYXkgYmUgdXNlZnVsIHRvIGF1dGhvciB0aGUgc2NoZW1hIHdpdGggW09ubGluZSBTd2FnZ2VyIEVkaXRvcl0oaHR0cHM6Ly9lZGl0b3Iuc3dhZ2dlci5pby8pLgoKRm9yIGNsb3VkIGRlcGxveW1lbnRzIGNvZGUgZ2VuZXJhdG9yIGdlbmVyYXRlcyBjb2RlIGZvciBoZWFsdGggY2hlY2ssIGlmIGluIHNjaGVtYSBpcyBkZWZpbmVkIGFjdGlvbiBgR0VUIC9oZWFsdGhgLgoKIyMgR2VuZXJhdG9yIGNvZGUgZnJvbSBHaXRIdWIKCjEuIERvd25sb2FkIHRoZSBnZW5lcmF0b3Igd2l0aCBgZ2l0IGNsb25lIGh0dHBzOi8vZ2l0aHViLmNvbS90b21pLXZhbmVrL29wZW5hcGktbWljcm8tdHMuZ2l0YCBhbmQgZ28gdG8gcHJvamVjdCByZXBvc2l0b3J5IHdpdGggYGNkIG1pY3JvdHNgCjEuIERvd25sb2FkIGdlbmVyYXRvciBkZXBlbmRlbmNpZXMgYW5kIHRvb2xzIHdpdGggYG5wbSBpYC4KMS4gUmVnaXN0ZXIgdGhlIHRvb2wgaW4gbG9jYWwgTlBNIHdpdGggYG5wbSBsaW5rYCBjb21tYW5kLCBzbyB5b3UgY2FuIHVzZSBpdCBmcm9tIGNvbW1hbmQgbGluZSBpbiBhbnkgZGlyZWN0b3J5IHdpdGggY29tbWFuZCBgbWljcm90c2AuCgojIyBHZW5lcmF0b3IgaW4gZGV0YWlsCgpUaGUgbWljcm9zZXJ2aWNlIGludGVyZmFjZSBpcyBkZWZpbmVkIGluIGZvcm0gb2YgW09wZW5BUEkgMi4wXShodHRwczovL2dpdGh1Yi5jb20vT0FJL09wZW5BUEktU3BlY2lmaWNhdGlvbi9ibG9iL21hc3Rlci92ZXJzaW9ucy8yLjAubWQpIHNjaGVtYSwgYXMgdGhlIGxpYnJhcmllcyAvIHRvb2xzIHVzZWQgaW4gZ2VuZXJhdGVkIGNvZGUgZG8gbm90IHN1cHBvcnQgdGhlIGN1cnJlbnQgdmVyc2lvbiBvZiBPcGVuQVBJIHlldC4KCkdlbmVyYXRlZCBhcHBsaWNhdGlvbiBjb2RlIGlzIGluIFR5cGVTY3JpcHQgbGFuZ3VhZ2UuCgpCYXNpYyBmZWF0dXJlcyBvZiB0aGUgZ2VuZXJhdGVkIGNvZGU6CgoqIFR5cGVTY3JpcHQgbGFuZ3VhZ2UKKiBBcHBsaWNhdGlvbiBjb25maWd1cmF0aW9uIGluIGRpcmVjdG9yeSBgL3NyYy9jb25mYAoqIE5vZGUgJiBFeHByZXNzIHNlcnZlciBzZXR1cCBpbiBgL3NyY2AKKiBDb252ZW50aW9uLWJhc2VkIHJvdXRpbmcgYW5kIHJlcXVlc3QgaGFuZGxlcnMgaW4gYC9zcmMvaGFuZGxlcnNgIC0gcmVxdWVzdCBwYXRoIGNvcnJlc3BvbmRzIHRoZSBkaXJlY3RvcnkgcGF0aCwgbm8gZXhwbGljaXQgcm91dGluZyBsb2dpYyBpcyBuZWVkZWQKKiBaZXJvLWNvZGUgYXV0b21hdGljIGlucHV0IHZhbGlkYXRpb24gZGVmaW5lZCBieSBydWxlcyBpbiBPcGVuQVBJIC8gU3dhZ2dlciBzY2hlbWEKKiBVc2VyIGludGVyZmFjZSBmb3IgbWljcm9zZXJ2aWNlIHRlc3RpbmcgYW5kIGFkbWluaXN0cmF0aW9uIGluIGAvc3JjL3VpYAoqIGBEb2NrZXJmaWxlYCBmb3IgZGVwbG95bWVudCBpbWFnZSBhbmQgYGRvY2tlci1jb21wb3NlLnlhbWxgIGFzIGFuIGV4YW1wbGUgdXNhZ2UgaW4gYXBwbGljYXRpb24gaW50ZWdyYXRpb24KKiBFbmQtdG8tZW5kIHRlc3RzICBpbiBgL3Rlc3RgCgojIyBBcmNoaXRlY3R1cmUgc2hhcGUKCkdlbmVyYXRvciBkb2VzIG5vdCBvZmZlciByaWNoIHNldCBvZiBvcHRpb25zIHRvIHRhaWxvciB0aGUgcmVzdWx0IGludG8gZGlmZmVyZW50IGZvcm1zLiBUaGlzIGFwcHJvYWNoIGV4cHJlc3NlcyBhdXRob3IncyBhcmNoaXRlY3R1cmUgZXhwZXJpZW5jZTogZ2VuZXJhdG9yIGlzIGEgd2F5IHRvIGRlZmluZSBhcmNoaXRlY3R1cmUgd2l0aG91dCBjb21wbGV4IGRvY3VtZW50YXRpb24sIHRoYXQgZ2VudGx5IGRpcmVjdHMgZGV2ZWxvcGVycyBpbiB0aGUgYXJjaGl0ZWN0dXJlLWVudmlzaW9uZWQgZGlyZWN0aW9uOgoKKiBEZXZlbG9wZXJzIGFyZSBpbXBsZW1lbnRpbmcgdGhlIGFwcGxpY2F0aW9uIGxvZ2ljIGluIHJlcXVlc3QgaGFuZGxlcnMgYW5kIHRlc3RzLgoqIE5vbi1mdW5jdGlvbmFsIGNvbmNlcm5zIGFyZSBoaWRkZW4gaW4gdGhlIGdlbmVyYXRlZCBjb2RlLgoqIEltcGxlbWVudGF0aW9uIG9mIG1pY3Jvc2VydmljZSAiZnJvbSB6ZXJvIiBpcyB2ZXJ5IGZhc3QgLSByZW1vdmVzIHRpbWUgJiBlZmZvcnQgY29uY2VybnMgYnkgaW50cm9kdWNpbmcgbmV3IG9yIHJhZGljYWwgcmVmYWN0b3Jpbmcgb2YgZXhpc3Rpbmcgc2VydmljZXMgaW4gc3lzdGVtIGVjb3N5c3RlbS4KCkdlbmVyYXRvciBpbiBhcmNoaXRlY3R1cmU6CgoqIE9wZXJhdGlvbmFsIGRlZmluaXRpb24gb2YgdGhlIGFyY2hpdGVjdHVyZSAoYXMgYSByZXBsYWNlbWVudCBvZiB3cml0ZS1vbmx5IG9ic29sZXRlIGRvY3VtZW50YXRpb24gOi0pCiogQ29uc2lzdGVuY3kgb2YgcHJvamVjdCBzdHJ1Y3R1cmUgLSBzaW1wbGUgZ2xvYmFsIHJlZmFjdG9yaW5nIGJ5IGNoYW5nZXMgb2YgdGhlIHJ1bnRpbWUgZW52aXJvbm1lbnQKKiBEZXZlbG9wZXIgZm9jdXMgb24gYXBwbGljYXRpb24gY29kZSAobWluaW1pemVzIGRldmVsb3BlcidzIGNyZWF0aXZpdHkgaW4gbm9uLWZ1bmN0aW9uYWwgcnVudGltZSBhbmQgc2VjdXJpdHkgY29uY2VybnMpCgpBcyBhbiBhcmNoaXRlY3QgeW91IGhhdmUgeW91ciBvd24gdGVjaG5pY2FsIG9waW5pb24sIHRlY2hub2xvZ3kgY29uc3RyYWludHMgLyBwcmVmZXJlbmNlcyBhbmQgaW5mcmFzdHJ1Y3R1cmUgJsKgc2VjdXJpdHkgc2VydmljZXMgdGhhdCBoYXZlIHRvIGJlIGludGVncmF0ZWQgaW50byB0aGUgKG1pY3JvKXNlcnZpY2Ugc2VydmVycy4KCkp1c3QgZm9yayB0aGlzIHByb2plY3QsIG9yIHRha2UgYW4gaW5zcGlyYXRpb24gYW5kIGJ1aWxkIHlvdXIgb3duIGdlbmVyYXRvciBmcm9tIGFuIHByb29mLW9mLWNvbmNlcHQgc2VydmljZSB0aGF0IGJlc3QgZml0cyB5b3VyIGV4cGVjdGF0aW9ucy4K readmeEtag: '"0b3e48e33f2a2d147ea9a0419cf01a632d968170"' readmeLastModified: Sun, 30 Apr 2023 06:19:22 GMT repositoryId: 165400292 description: >- Microservice code generator: from OpenAPI (Swagger) REST API specification to TypeScript project with Docker. created: '2019-01-12T15:00:44Z' updated: '2025-11-26T11:55:48Z' language: JavaScript archived: false stars: 31 watchers: 1 forks: 3 owner: tomi-vanek logo: https://avatars.githubusercontent.com/u/10799822?v=4 license: MIT repoEtag: '"305b5c34d8bd675c92fb58888dde077d1094857b8ad2929cfd5a7648bfc9a900"' repoLastModified: Wed, 26 Nov 2025 11:55:48 GMT foundInMaster: true id: 87f9aea977df5e49cac64a831bd0299b - source: https://openapi.tools/ name: Meeshkan category: - Learning - Mock - Server Implementations language: Python link: https://github.com/meeshkan/meeshkan repository: https://github.com/meeshkan/meeshkan source_description: >- Mock HTTP APIs through a combination of API definitions, recorded traffic and code. Used for sandboxes, as well as automated and exploratory testing. v2: false v3: true repositoryMetadata: base64Readme: >- IyBNZWVzaGthbiDigJQgdGhlIGZyb250ZW5kIG1vbm8tcmVwbwoKVGhpcyByZXBvIGNvbnRhaW5zIDQgYXBwcyB0aGF0IGhhdmUgc29tZSAnZnJvbnRlbmQnIHV0aWxpdHkgdG8gdGhlIGNvbXBhbnkuCgoxLiBgY3VzdG9tLWdyYXBocWxgIHRoaXMgaXMgd2hlcmUgb3VyIDhiYXNlIGN1c3RvbSBsb2dpYyBsaXZlcy4gWzhiYXNlIGxvZ2ljXShodHRwczovL2RvY3MuOGJhc2UuY29tL2RvY3MvOGJhc2UtY29uc29sZS9wbGF0Zm9ybS10b29scy9mdW5jdGlvbnMvKSBhbGxvd3MgeW91IHRvIGRlcGxveSBjdXN0b20gZnVuY3Rpb25hbGl0eSBhbmQgaXMgdGhlIGJhc2VsaW5lIGZvciBjb21taXR0aW5nIGRhdGFiYXNlIG1pZ3JhdGlvbnMuIEluIG91ciBjYXNlIHdlIGhhdmUgYW4gaW52aXRlIGxpbmsgW3RyaWdnZXIgYW5kIHVwZGF0ZXJdKC9Vc2Vycy9tYWtlbm5hc211dHovRG9jdW1lbnRzL0dpdEh1Yi9tZWVzaGthbi9hcHBzL2N1c3RvbS1ncmFwaHFsLzhiYXNlLnltbCkuIEluIG9yZGVyIHRvIHVzZSB0aGlzIHJlcG8sIHlvdSdsbCBuZWVkIHRvIGJlIGZhbWlsaWFyIHdpdGggdGhlIFs4YmFzZSBDTEldKGh0dHBzOi8vZG9jcy44YmFzZS5jb20vZG9jcy9kZXZlbG9wbWVudC10b29scy9jbGkpLgoyLiBgb2ctY2FyZHNgIGlzIGEgZnJvbnRlbmQgbWljcm8tc2VydmljZSB0aGF0IGdlbmVyYXRlcyBkeW5hbWljIGltYWdlcyBmb3IgdGhlIHNoYXJpbmcgY2FyZHMgc2hvd24gb24gc29jaWFscyBmb3Igb3VyIHdlYnNpdGUuIEl0IGlzIGRlcGxveWVkIG9uIFZlcmNlbCBhbmQgc28gZmFyIGhhcyBuZWVkZWQgbGl0dGxlIHRvIG5vIG1haW50YWluYW5jZS4KMy4gYHdlYmFwcGAgdGhpcyBpcyB0aGUgbW9zdCBhY3RpdmUgYXBwIGluIHRoaXMgcmVwby4gSXQgaXMgdGhlIG1lZXNoa2FuIHdlYmFwcCAoaHR0cHM6Ly9hcHAubWVlc2hrYW4uY29tKSB3aGljaCBpcyBkZXBsb3llZCBvbiBWZXJjZWwuCjQuIGB3ZWJzaXRlYCB0aGlzIGlzIGhlcmUgaW4gYSBob3BlZnVsIG1lYXN1cmUgYnV0IG5vdCBjdXJyZW50bHkgaW4gdXNlLiBDdXJyZW50bHkgb3VyIGxpdmUgd2Vic2l0ZSByZXBvIGlzIGh0dHBzOi8vZ2l0aHViLmNvbS9tZWVzaGthbi93ZWJzaXRlLiBXZSdkIGxpa2UgdG8gdHJhbnNpdGlvbiB0aGlzIGhlcmUgdG8gc2hhcmUgdGhlIGNoYWtyYS11aSBsaWJyYXJ5LgoKVGhpcyByZXBvIGFsc28gY29udGFpbnMgMyBgbGlic2Agb3IgTlggbGlicmFyaWVzIHdoaWNoIGFyZSBzaGFyZWQgY29kZSBiZXR3ZWVuIGFsbCBvZiB0aGUgZnJvbnRlbmQgcHJvamVjdHMuCgoxLiBgY2hha3JhLXRoZW1lYCBzZXJ2ZXMgYXMgdGhlIGN1c3RvbSBbQ2hha3JhIFVJXShodHRwczovL2NoYWtyYS11aS5jb20vKSBzZXR1cCB0aGF0IHdlIHVzZSBhY2Nyb3NzIGFueSBmcm9udGVuZCBwcm9qZWN0cy4KMi4gYGRvd25sb2FkYWJsZS1zY3JpcHRgIGlzIHRoZSBtaWNyb3NlcnZpY2UgdGhhdCBjb252ZXJ0cyB0aGUgdXNlciBzdG9yeSBldmVudHMvY29tbWFuZHMgaW4gOGJhc2UgaW50byBhIFB1cHBldGVlciBzY3JpcHQuCjMuIGBtZWVzaGthbi10eXBlc2AgaXMgYSBjZW50cmFsaXplZCBwbGFjZSBmb3IgY3VzdG9tIFR5cGVTY3JpcHQgaW50ZXJmYWNlcyB3ZSB1c2UuIFNvbWUgYXJlIGJ1aWx0IGN1c3RvbSBidXQgbW9zdCBjb21lIGZyb20gOGJhc2UgdXNpbmcgYSBwYWNrYWdlIGNhbGxlZCBgZ3JhcGhxbC1jb2RlLWdlbmAuCgojIyBHZXR0aW5nIHRoZSBgd2ViYXBwYCBwcm9qZWN0IHdvcmtpbmcgbG9jYWxseQoKVGhpcyBndWlkZSB3aWxsIHVzZSBgeWFybmAuIEZlZWwgZnJlZSB0byB1c2UgdGhlIGBucG1gIGVxdWl2YWxlbnQgdG8gbXkgaW5zdHJ1Y3Rpb25zIGlmIHRoYXQgd29ya3MgYmV0dGVyIGZvciB5b3UhCgoxLiBBZnRlciBjbG9uaW5nIHRoaXMgcmVwbyB0byB5b3VyIGNvbXB1dGVyIOKAlCBgY2RgIGludG8gdGhlIGJhc2UgcmVwb3NpdG9yeSAoYG1lZXNoa2FuYCkuCjIuIER1cGxpY2F0ZSB0aGUgYC5lbnYudGVtcGxhdGVgIGZpbGUsIHJlbmFtaW5nIGl0IHRvIGAuZW52YC4KICAgLSBgQVVUSDBfQ0xJRU5UX0lEYCwgYEFVVEgwX0NMSUVOVF9TRUNSRVRgLCBhbmQgYFNFU1NJT05fQ09PS0lFX1NFQ1JFVGAgY2FuIGFsbCBiZSBmb3VuZCBpbiBBdXRoMC4gSWYgeW91IGRvIG5vdCBoYXZlIGFjY2VzcyB0byBBdXRoMCwgY29udGFjdCBAazRtNCBvciBAS2Vuem9CZW56byB0byBnZXQgeW91IHRoZSB0b2tlbnMuIFRoZSB2YWx1ZSBvZiBgQVVUSDBfRE9NQUlOYCBpcyAibWVlc2hrYW4uZXUuYXV0aDAuY29tIi4KICAgLSBUaGUgdG9rZW4gYE5FWFRfUFVCTElDX0VJR0hUQkFTRV9FTkRQT0lOVGAgY2hhbmdlcyB0aGUgOGJhc2UgZW52aXJvbm1lbnQgYmV0d2VlbiBgc3RhZ2luZ2AgImh0dHBzOi8vYXBpLjhiYXNlLmNvbS9ja2hxZHo1bXUwMXIzMDdtbjZzemNiZGtlX3N0YWdpbmciIGFuZCBgTWFzdGVyYCAiaHR0cHM6Ly9hcGkuOGJhc2UuY29tL2NraHFkejVtdTAxcjMwN21uNnN6Y2Jka2UiLgogICAtIFRoZSB0b2tlbiBgRUlHSFRfQkFTRV9BVVRIX1BST0ZJTEVfSURgIGNhbiBiZSBmb3VuZCBpbiA4YmFzZSAobG9nIGluIGZvdW5kIGluIDFwdykuCiAgIC0gVGhlIGBURVNUX1VSTGAgZGVwZW5kcyBvbiBob3cgeW91IHNlcnZlIHlvdXIgd2ViYXBwIGxvY2FsbHkuIElmIHlvdSB1c2UgdGhlIGNvbW1hbmQgYHlhcm4gc3RhcnQgd2ViYXBwYCB0aGUgdmFsdWUgd2lsbCBiZSAibG9jYWxob3N0OjMwMDAiLiBDaGFuZ2UgYWNjb3JkaW5nbHkgZm9yIGEgZGlmZmVyZW50IHBvcnQuCiAgIC0gVGhlIGBDT09LSUVgIHRva2VuIGNhbiBiZSBmb3VuZCBpbiBBdXRoMCwgb3IgaHR0cHM6Ly9hcHAubWVlc2hrYW4uY29tIHVuZGVyIG91ciBpbnRlcm5hbCBgTWVlc2hrYW4gd2ViYXBwYCBwcm9qZWN0LCBhdXRoIHRva2Vucy4KICAgLSBUaGUgYE1JWFBBTkVMX1RPS0VOYCBjYW4gYmUgZm91bmQgaW4gTWl4cGFuZWwsIGJ1dCBmb3IgbG9jYWwgZGV2ZWxvcG1lbnQgKHRvIG5vdCBwb2xsdXRlIHByb2R1Y3Rpb24gYW5hbHl0aWNzKSBJJ2Qgc3VnZ2VzdCB5b3UgdXNlIHRoZSBgVGVzdCBkYXRhYCBwcm9qZWN0IHRva2VuIG9mICJkZDdmZWJjOGMyNjk3ZmNhOGVkMmQ1NTIzNDA5YTI4MSIuCjMuIFdoaWxlIHN0aWxsIGluIHRoZSBiYXNlIHJlcG9zaXRvcnksIGluc3RhbGwgdGhlIHByb2plY3QgZGVwZW5kZW5jaWVzIHdpdGggYHlhcm5gLgo0LiBTdGFydCBhIGRldmVsb3BtZW50IHNlcnZlciBvZiB0aGUgd2ViYXBwIHVzaW5nIHRoZSBjb21tYW5kIGB5YXJuIHN0YXJ0IHdlYmFwcGAuIFlvdSBzaG91bGQgdGhlbiBzZWUgYSBtZXNzYWdlIHRoYXQgdGhlIHBvcnQgMzAwMCBpcyBub3cgb3BlbiBhbmQgeW91IGNhbiB2aXNpdCBgbG9jYWxob3N0OjMwMDBgLgo= readmeEtag: '"e02599f5aa9525c9d2b715310334bcd13631cc2b"' readmeLastModified: Sun, 05 Sep 2021 00:37:40 GMT repositoryId: 312140836 description: The Meeshkan frontend stack monorepo. created: '2020-11-12T02:01:32Z' updated: '2023-01-27T19:33:58Z' language: TypeScript archived: true stars: 2 watchers: 1 forks: 1 owner: meeshkan logo: https://avatars.githubusercontent.com/u/32298527?v=4 repoEtag: '"ad1ff456329869cf55947d199086ed159807fcb405ac6a6cb7933c5d3441d10a"' repoLastModified: Fri, 27 Jan 2023 19:33:58 GMT foundInMaster: true id: 7e8e0d0c7a10ec695d4736865a7f573b - source: https://openapi.tools/ name: Vert.x Web API Service category: Server language: Java link: https://vertx.io/docs/vertx-web-api-service/java/ repository: https://github.com/vert-x3/vertx-web/ source_description: Create API service proxies using event bus with request/response validation v2: false v3: true repositoryMetadata: base64Readme: >- PSBWZXJ0LngtV2ViCgppbWFnZTpodHRwczovL2dpdGh1Yi5jb20vdmVydC14My92ZXJ0eC13ZWIvYWN0aW9ucy93b3JrZmxvd3MvY2ktNS54LnltbC9iYWRnZS5zdmdbIkJ1aWxkIFN0YXR1cyAoNS54KSIsbGluaz0iaHR0cHM6Ly9naXRodWIuY29tL3ZlcnQteDMvdmVydHgtd2ViL2FjdGlvbnMvd29ya2Zsb3dzL2NpLTUueC55bWwiXQppbWFnZTpodHRwczovL2dpdGh1Yi5jb20vdmVydC14My92ZXJ0eC13ZWIvYWN0aW9ucy93b3JrZmxvd3MvY2ktNC54LnltbC9iYWRnZS5zdmdbIkJ1aWxkIFN0YXR1cyAoNC54KSIsbGluaz0iaHR0cHM6Ly9naXRodWIuY29tL3ZlcnQteDMvdmVydHgtd2ViL2FjdGlvbnMvd29ya2Zsb3dzL2NpLTQueC55bWwiXQppbWFnZTpodHRwczovL2ltZy5zaGllbGRzLmlvL21hdmVuLWNlbnRyYWwvdi9pby52ZXJ0eC92ZXJ0eC13ZWIuc3ZnWyJNYXZlbiBDZW50cmFsIl0KaW1hZ2U6aHR0cHM6Ly9iZXN0cHJhY3RpY2VzLmNvcmVpbmZyYXN0cnVjdHVyZS5vcmcvcHJvamVjdHMvNTQwL2JhZGdlWyJDSUkgQmVzdCBQcmFjdGljZXMiLGxpbms9Imh0dHBzOi8vYmVzdHByYWN0aWNlcy5jb3JlaW5mcmFzdHJ1Y3R1cmUub3JnL3Byb2plY3RzLzU0MCJdCgpWZXJ0LngtV2ViIGlzIGEgc2V0IG9mIGJ1aWxkaW5nIGJsb2NrcyBmb3IgYnVpbGRpbmcgd2ViIGFwcGxpY2F0aW9ucyB3aXRoIFZlcnQueC4gVGhpbmsgb2YgaXQgYXMgYSBTd2lzcyBBcm15IEtuaWZlIGZvciBidWlsZGluZwptb2Rlcm4sIHNjYWxhYmxlLCB3ZWIgYXBwcy4KClBsZWFzZSBzZWUgdGhlIG1haW4gZG9jdW1lbnRhdGlvbiBvbiB0aGUgd2ViLXNpdGUgZm9yIGEgZnVsbCBkZXNjcmlwdGlvbjoKCiogaHR0cHM6Ly92ZXJ0eC5pby9kb2NzLyN3ZWJbV2ViLXNpdGUgZG9jdW1lbnRhdGlvbl0KCj09IFRlbXBsYXRlIGVuZ2luZXMKClRlbXBsYXRlIGVuZ2luZSBpbXBsZW1lbnRhdGlvbnMgYXJlIGluIHRoZSB0ZW1wbGF0ZSBlbmdpbmUgc3ViLXByb2plY3QuCg== readmeEtag: '"269b52e98cea5813787f90c67bef329932b70760"' readmeLastModified: Fri, 16 Aug 2024 21:55:56 GMT repositoryId: 26628954 description: HTTP web applications for Vert.x created: '2014-11-14T08:15:20Z' updated: '2026-02-05T20:01:22Z' language: Java archived: false stars: 1144 watchers: 71 forks: 556 owner: vert-x3 logo: https://avatars.githubusercontent.com/u/8124623?v=4 license: Apache-2.0 repoEtag: '"27b312e88d5724b891cdb6068adcfe907cdb8f8c8a312ae3851967e80ed62edc"' repoLastModified: Thu, 05 Feb 2026 20:01:22 GMT foundInMaster: true v3_link: https://github.com/vert-x3/vertx-web/issues/1872 id: 649e1f8cd9ccaccf681527e6984f2785 v3_1_link: https://github.com/vert-x3/vertx-web/issues/1872 - source: https://openapi.tools/ name: Mayhem for API category: Testing link: https://forallsecure.com/mayhem-for-api language: Any source_description: >- Probe your REST API with an infinite stream of test cases generated automatically from your OpenAPI specification. v2: true v3: true foundInMaster: true id: f0d53d8ed081f28f47065108d2abdf80 - source: openapi3 tags repository: https://github.com/unionj-cloud/go-doudou v3: true repositoryMetadata: base64Readme: >- PHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9nby1kb3Vkb3UuZ2l0aHViLmlvIiB0YXJnZXQ9Il9ibGFuayIgcmVsPSJub29wZW5lciBub3JlZmVycmVyIj4KICAgIDxpbWcgd2lkdGg9IjE4MCIgc3JjPSJodHRwczovL2dvLWRvdWRvdS5naXRodWIuaW8vaGVyby5wbmciIGFsdD0iVml0ZSBsb2dvIj4KICA8L2E+CjwvcD4KPGJyLz4KPHAgYWxpZ249ImNlbnRlciI+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL2F2ZWxpbm8vYXdlc29tZS1nbyI+PGltZyBzcmM9Imh0dHBzOi8vYXdlc29tZS5yZS9tZW50aW9uZWQtYmFkZ2Uuc3ZnIiBhbHQ9Ik1lbnRpb25lZCBpbiBBd2Vzb21lIEdvIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9nb2RvYy5vcmcvZ2l0aHViLmNvbS91bmlvbmotY2xvdWQvZ28tZG91ZG91Ij48aW1nIHNyYz0iaHR0cHM6Ly9nb2RvYy5vcmcvZ2l0aHViLmNvbS91bmlvbmotY2xvdWQvZ28tZG91ZG91P3N0YXR1cy5wbmciIGFsdD0iR29Eb2MiPjwvYT4KICA8YSBocmVmPSJodHRwczovL2dpdGh1Yi5jb20vdW5pb25qLWNsb3VkL2dvLWRvdWRvdS9hY3Rpb25zL3dvcmtmbG93cy9nby55bWwiPjxpbWcgc3JjPSJodHRwczovL2dpdGh1Yi5jb20vdW5pb25qLWNsb3VkL2dvLWRvdWRvdS9hY3Rpb25zL3dvcmtmbG93cy9nby55bWwvYmFkZ2Uuc3ZnP2JyYW5jaD1tYWluIiBhbHQ9IkdvIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9jb2RlY292LmlvL2doL3VuaW9uai1jbG91ZC9nby1kb3Vkb3UiPjxpbWcgc3JjPSJodHRwczovL2NvZGVjb3YuaW8vZ2gvdW5pb25qLWNsb3VkL2dvLWRvdWRvdS9icmFuY2gvbWFpbi9ncmFwaC9iYWRnZS5zdmc/dG9rZW49UVJMUFJBWDg4NSIgYWx0PSJjb2RlY292Ij48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9nb3JlcG9ydGNhcmQuY29tL3JlcG9ydC9naXRodWIuY29tL3VuaW9uai1jbG91ZC9nby1kb3Vkb3UiPjxpbWcgc3JjPSJodHRwczovL2dvcmVwb3J0Y2FyZC5jb20vYmFkZ2UvZ2l0aHViLmNvbS91bmlvbmotY2xvdWQvZ28tZG91ZG91IiBhbHQ9IkdvIFJlcG9ydCBDYXJkIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9naXRodWIuY29tL3VuaW9uai1jbG91ZC9nby1kb3Vkb3UiPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2dpdGh1Yi92L3JlbGVhc2UvdW5pb25qLWNsb3VkL2dvLWRvdWRvdT9zdHlsZT1mbGF0LXNxdWFyZSIgYWx0PSJSZWxlYXNlIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly9vcGVuc291cmNlLm9yZy9saWNlbnNlcy9NSVQiPjxpbWcgc3JjPSJodHRwczovL2ltZy5zaGllbGRzLmlvL2JhZGdlL0xpY2Vuc2UtTUlULXllbGxvdy5zdmciIGFsdD0iTGljZW5zZTogTUlUIj48L2E+CiAgPGEgaHJlZj0iaHR0cHM6Ly93YWthdGltZS5jb20vYmFkZ2UvdXNlci84NTJiY2YyMi04YTM3LTQ2MGEtYThlMi0xMTU4MzMxNzRlYmEvcHJvamVjdC81N2M4MzBmNy1lNTA3LTRjYjEtOWZkMS1mZWVkZDk2Njg1ZjYiPjxpbWcgc3JjPSJodHRwczovL3dha2F0aW1lLmNvbS9iYWRnZS91c2VyLzg1MmJjZjIyLThhMzctNDYwYS1hOGUyLTExNTgzMzE3NGViYS9wcm9qZWN0LzU3YzgzMGY3LWU1MDctNGNiMS05ZmQxLWZlZWRkOTY2ODVmNi5zdmciIGFsdD0iTGljZW5zZTogTUlUIj48L2E+CjwvcD4KPGJyLz4KCiMgZ28tZG91ZG91Cgo+IExpZ2h0d2VpZ2h0IEdvbGFuZyBNaWNyb3NlcnZpY2UgRnJhbWV3b3JrCgotIPCfkqEgU3RhcnRzIGZyb20gZ29sYW5nIGludGVyZmFjZSwgbm8gbmVlZCB0byBsZWFybiBuZXcgSURMKGludGVyZmFjZSBkZWZpbml0aW9uIGxhbmd1YWdlKS4KLSDwn5SpIFBvd2VyZnVsIGNvZGUgZ2VuZXJhdG9yIGNsaSBidWlsdC1pbi4gQWZ0ZXIgZGVmaW5pbmcgeW91ciBpbnRlcmZhY2UgbWV0aG9kcywgeW91ciBvbmx5IGpvYiBpcyBpbXBsZW1lbnRpbmcgeW91ciBhd2Vzb21lIGlkZWEuCi0g4pqhIEJvcm4gZnJvbSB0aGUgY2xvdWQtbmF0aXZlIGVyYS4gQnVpbHQtaW4gQ0xJIGNhbiBzcGVlZCB1cCB5b3VyIHByb2R1Y3QgaXRlcmF0aW9uLgotIPCflJEgQnVpbHQtaW4gc2VydmljZSBnb3Zlcm5hbmNlIHN1cHBvcnQgaW5jbHVkaW5nIHJlbW90ZSBjb25maWd1cmF0aW9uIG1hbmFnZW1lbnQsIGNsaWVudC1zaWRlIGxvYWQgYmFsYW5jZXIsIHJhdGUgbGltaXRlciwgY2lyY3VpdCBicmVha2VyLCBidWxraGVhZCwgdGltZW91dCwgcmV0cnkgYW5kIG1vcmUuCi0g8J+Tpu+4jyBTdXBwb3J0aW5nIGJvdGggbW9ub2xpdGggYW5kIG1pY3Jvc2VydmljZSBhcmNoaXRlY3R1cmVzIGdpdmVzIHlvdSBmbGV4aWJpbGl0eSB0byBkZXNpZ24geW91ciBzeXN0ZW0uCgpHby1kb3Vkb3XvvIhkb3Vkb3UgcHJvbm91bmNlIC9kyZl1ZMmZdS/vvIlpcyBPcGVuQVBJIDMuMCAoZm9yIFJFU1QpIHNwZWMgYW5kIFByb3RvYnVmIHYzIChmb3IgZ3JwYykgYmFzZWQgbGlnaHR3ZWlnaHQgbWljcm9zZXJ2aWNlIGZyYW1ld29yay4gSXQgc3VwcG9ydHMgbW9ub2xpdGggc2VydmljZSBhcHBsaWNhdGlvbiBhcyB3ZWxsLiAgCgpSZWFkIHRoZSBEb2NzIFtodHRwczovL2dvLWRvdWRvdS5naXRodWIuaW8vZ3VpZGVdKGh0dHBzOi8vZ28tZG91ZG91LmdpdGh1Yi5pby9ndWlkZSkgdG8gTGVhcm4gTW9yZS4KCiMjIEJlbmNobWFyawoKIVtiZW5jaG1hcmtdKC4vYmVuY2htYXJrLnBuZykKCk1hY2hpbmU6IGBNYWNCb29rIFBybyAoMTYtaW5jaCwgMjAxOSlgICAKQ1BVOiBgMi4zIEdIeiA4IGNvcmVzIEludGVsIENvcmUgaTlgICAKTWVtb3J5OiBgMTYgR0IgMjY2NyBNSHogRERSNGAgIApQcm9jZXNzaW5nVGltZTogYDBtcywgMTBtcywgMTAwbXMsIDUwMG1zYCAgCkNvbmN1cnJlbmN5OiBgMTAwMGAgIApEdXJhdGlvbjogYDMwc2AgIApnby1kb3Vkb3UgVmVyc2lvbjogYHYxLjMuN2AgIAoKW0NoZWNrb3V0IHRoZSB0ZXN0IGNvZGVdKGh0dHBzOi8vZ2l0aHViLmNvbS93dWJpbjE5ODkvZ28td2ViLWZyYW1ld29yay1iZW5jaG1hcmspCgojIyBDcmVkaXRzCgpHaXZlIGNyZWRpdHMgdG8gZm9sbG93aW5nIHJlcG9zaXRvcmllcyBhbmQgYWxsIHRoZWlyIGNvbnRyaWJ1dG9yczoKLSBbZ28tcmVkaXMvcmVkaXNfcmF0ZV0oZ2l0aHViLmNvbS9nby1yZWRpcy9yZWRpc19yYXRlKTogZ28tZG91ZG91IGlzIHJlbHlpbmcgb24gaXQgdG8gaW1wbGVtZW50IHJlZGlzIGJhc2VkIHJhdGUgbGltaXQgZmVhdHVyZQotIFthcG9sbG9jb25maWcvYWdvbGxvXShodHRwczovL2dpdGh1Yi5jb20vYXBvbGxvY29uZmlnL2Fnb2xsbyk6IGdvLWRvdWRvdSBpcyByZWx5aW5nIG9uIGl0IHRvIGltcGxlbWVudCByZW1vdGUgY29uZmlndXJhdGlvbiBtYW5hZ2VtZW50IHN1cHBvcnQgZm9yIFtBcG9sbG9dKGh0dHBzOi8vZ2l0aHViLmNvbS9hcG9sbG9jb25maWcvYXBvbGxvKQotIFtuYWNvcy1ncm91cC9uYWNvcy1zZGstZ29dKGh0dHBzOi8vZ2l0aHViLmNvbS9uYWNvcy1ncm91cC9uYWNvcy1zZGstZ28pOiBnby1kb3Vkb3UgaXMgcmVseWluZyBvbiBpdCB0byBpbXBsZW1lbnQgc2VydmljZSBkaXNjb3ZlcnkgYW5kIHJlbW90ZSBjb25maWd1cmF0aW9uIG1hbmFnZW1lbnQgc3VwcG9ydCBmb3IgW05hY29zXShodHRwczovL2dpdGh1Yi5jb20vYWxpYmFiYS9uYWNvcykKCiMjIENvbW11bml0eQoKV2VsY29tZSB0byBjb250cmlidXRlIHRvIGdvLWRvdWRvdSBieSBmb3JraW5nIGl0IGFuZCBzdWJtaXR0aW5nIHByIG9yIGlzc3Vlcy4gSWYgeW91IGxpa2UgZ28tZG91ZG91LCBwbGVhc2UgZ2l2ZSBpdCBhCnN0YXIhCgpXZWxjb21lIHRvIGNvbnRhY3QgbWUgZnJvbQoKLSBGYWNlYm9vazogW2h0dHBzOi8vd3d3LmZhY2Vib29rLmNvbS9iaW4ud3UuOTQ2MTc5OTkvXShodHRwczovL3d3dy5mYWNlYm9vay5jb20vYmluLnd1Ljk0NjE3OTk5LykKLSBUd2l0dGVyOiBbaHR0cHM6Ly90d2l0dGVyLmNvbS9CSU5XVTQ5MjA1NTEzXShodHRwczovL3R3aXR0ZXIuY29tL0JJTldVNDkyMDU1MTMpCi0gRW1haWw6IDMyODQ1NDUwNUBxcS5jb20KLSBXZUNoYXQ6ICAKICA8aW1nIHNyYz0iLi9xcmNvZGUucG5nIiBhbHQ9IndlY2hhdC1ncm91cCIgd2lkdGg9IjI0MCI+Ci0gV2VDaGF0IEdyb3VwOiAgCiAgPGltZyBzcmM9Ii4vZ28tZG91ZG91LXdlY2hhdC1ncm91cC5wbmciIGFsdD0id2VjaGF0LWdyb3VwIiB3aWR0aD0iMjQwIj4KLSBRUSBncm91cDogIAogIDxpbWcgc3JjPSIuL2dvLWRvdWRvdS1xcS1ncm91cC5wbmciIGFsdD0icXEtZ3JvdXAiIHdpZHRoPSIyNDAiPgoKIyMg8J+UiyBKZXRCcmFpbnMgT3BlbiBTb3VyY2UgTGljZW5zZQoKR28tZG91ZG91IGhhcyBiZWVuIGJlaW5nIGRldmVsb3BlZCB3aXRoIEdvTGFuZCB1bmRlciB0aGUgKipmcmVlIEpldEJyYWlucyBPcGVuIFNvdXJjZSBsaWNlbnNlKHMpKiogZ3JhbnRlZCBieSBKZXRCcmFpbnMgcy5yLm8uLCBoZW5jZSBJIHdvdWxkIGxpa2UgdG8gZXhwcmVzcyBteSBncmF0aXR1ZGUgaGVyZS4KCjxhIGhyZWY9Imh0dHBzOi8vamIuZ2cvT3BlblNvdXJjZVN1cHBvcnQiIHRhcmdldD0iX2JsYW5rIj48aW1nIHNyYz0iaHR0cHM6Ly9yZXNvdXJjZXMuamV0YnJhaW5zLmNvbS9zdG9yYWdlL3Byb2R1Y3RzL2NvbXBhbnkvYnJhbmQvbG9nb3MvamJfYmVhbS5wbmciIGFsdD0iSmV0QnJhaW5zIExvZ28gKE1haW4pIGxvZ28uIiB3aWR0aD0iMzAwIj48L2E+CgojIyBMaWNlbnNlCgpNSVQK readmeEtag: '"c5717e52cb52d29d98515fcc4aefed1bd696629c"' readmeLastModified: Sun, 11 Aug 2024 06:39:27 GMT repositoryId: 341813963 description: >- go-doudou(doudou pronounce /dəudəu/)is OpenAPI 3.0 (for REST) spec and Protobuf v3 (for grpc) based lightweight microservice framework. It supports monolith service application as well. created: '2021-02-24T07:21:40Z' updated: '2026-02-01T14:56:18Z' language: Go archived: false stars: 1207 watchers: 70 forks: 203 owner: unionj-cloud logo: https://avatars.githubusercontent.com/u/79033021?v=4 license: MIT repoEtag: '"b59891df1f78186b310cf175f006e0753ac6b8e8fa20d92688ca102eaea9692b"' repoLastModified: Sun, 01 Feb 2026 14:56:18 GMT foundInMaster: true category: SDK id: aab01f8bda6c542fdacf9ee8dfa7d20d - source: https://openapi.tools/ name: pyswagger category: - Converters - Server Implementations repository: https://github.com/pyopenapi/pyswagger language: Python source_description: Client & converter in Python, which is type-safe, dynamic, spec-compliant. v2: true v3_progress_link: https://github.com/mission-liao/pyswagger/blob/develop/docs/md/news.md repositoryMetadata: base64Readme: >- pyswagger
=========

[![Build Status](https://travis-ci.org/mission-liao/pyswagger.svg?branch=master)](https://travis-ci.org/mission-liao/pyswagger)
[![Coverage Status](https://coveralls.io/repos/mission-liao/pyswagger/badge.svg?branch=master&style=flat)](https://coveralls.io/r/mission-liao/pyswagger?branch=master)

A python client for [Swagger](https://helloreverb.com/developers/swagger) enabled REST API. It wouldn't be easier to
try Swagger REST API by [Swagger-UI](https://github.com/wordnik/swagger-ui). However, when it's time to **unittest**
your API, the first option you find would be [Swagger-codegen](https://github.com/wordnik/swagger-codegen), but the better option is us.

This project is developed after [swagger-py](https://github.com/digium/swagger-py), which is a nicely implemented one, and inspired many aspects of this project. Another project is [flex](https://github.com/pipermerriam/flex), which focuses on parameter validation, try it if you can handle other parts by yourselves. For other projects related to Swagger tools in python, check [here](https://github.com/swagger-api/swagger-spec#python).

**pyswagger** is much easier to use (compared to swagger-codegen, you don't need to prepare a scala environment) and tries hard to **fully supports** [Swagger Spec](https://helloreverb.com/developers/swagger) in all aspects.

- [NEWs: upcoming support for OpenAPI 3.0](docs/md/news.md)
- [Features](README.md#features)
- [Tutorial](README.md#tutorial)
- [Quick Start](README.md#quick-start)
- [Installation](README.md#installation)
- [Reference](README.md#reference)
- [Contributors](README.md#contributors)
- [Contribution Guideline](README.md#contribution-guildeline)
- [FAQ](docs/md/faq.md)
- [Changes](CHANGES.md)

---------

## Features
- convert Swagger Document from older version to newer one. (ex. convert from 1.2 to 2.0)
- support Swagger **1.2**, **2.0** on python ~~2.6~~, **2.7**, **3.3**, **3.5**, **3.6**
- support YAML via [Pretty-YAML](https://github.com/mk-fg/pretty-yaml)
- support $ref to **External Document**, multiple swagger.json will be organized into a group of App. And external document with self-describing resource is also supported (refer to [issue](https://github.com/swagger-api/swagger-spec/issues/219)).
- type safe, input/output are converted to python types according to [Data Type](https://github.com/wordnik/swagger-spec/blob/master/versions/1.2.md#43-data-types) described in Swagger. You don't need to touch any json schema when using pyswagger. Limitations like **minimum/maximum** or **enum** are also checked. **Model inheritance** also supported.
- provide function **App.validate** to check validity of the loaded API definition according to spec.
- builtin client implementation based on various http clients in python. For usage of these clients, please refer to `pyswagger.tests.contrib.client` for details
  - [requests](https://github.com/kennethreitz/requests)
  - [tornado.httpclient.AsyncHTTPClient](http://tornado.readthedocs.org/en/latest/httpclient.html)
  - [flask.testing.FlaskClient](http://flask.pocoo.org/docs/0.10/api/#flask.testing.FlaskClient)
  - [webapp2](http://webapp2.readthedocs.io/en/latest/guide/testing.html)
- not implemented parts, fire me a bug if you need it
  - [ ] Swagger 2.0
    - [ ] Schema.pattern
    - [ ] Scheme.patternProperties
    - [ ] Schema.readonly
    - [ ] Schema.allowEmptyValue
    - [ ] A scanner to validate schema
  - [ ] A WebSocket client
  - [ ] dump extension field

---------

## Tutorial

- [Initialization](docs/md/tutorial/init.md)
- [Making a Request](docs/md/tutorial/request.md)
- [Access the Response](docs/md/tutorial/response.md)
- [Testing a Local Server](docs/md/tutorial/local.md)
- [Converting Document into another version](docs/md/tutorial/converter.md)
- [Exntending Primitive Factory for user-defined primitives](docs/md/tutorial/extend_prim.md)
- [Rendering Random Requests for BlackBox Testing](docs/md/tutorial/render.md)
- [Operation MIME Support](docs/md/tutorial/mime.md)
- [Test with Invalid Input](docs/md/tutorial/invalid.md)
- [Load Spec from a Restricted Service](docs/md/tutorial/restricted_service.md)
- [Customized Headers](docs/md/tutorial/customized_headers.md)

---------

## Quick Start

Before running this script, please make sure [requests](https://github.com/kennethreitz/requests) is installed on your environment.

```python
from pyswagger import App, Security
from pyswagger.contrib.client.requests import Client
from pyswagger.utils import jp_compose

# load Swagger resource file into App object
app = App._create_('http://petstore.swagger.io/v2/swagger.json')

auth = Security(app)
auth.update_with('api_key', '12312312312312312313q') # api key
auth.update_with('petstore_auth', '12334546556521123fsfss') # oauth2

# init swagger client
client = Client(auth)

# a dict is enough for representing a Model in Swagger
pet_Tom=dict(id=1, name='Tom', photoUrls=['http://test']) 
# a request to create a new pet
client.request(app.op['addPet'](body=pet_Tom))

# - access an Operation object via App.op when operationId is defined
# - a request to get the pet back
req, resp = app.op['getPetById'](petId=1)
# prefer json as response
req.produce('application/json')
pet = client.request((req, resp)).data
assert pet.id == 1
assert pet.name == 'Tom'

# new ways to get Operation object corresponding to 'getPetById'.
# 'jp_compose' stands for JSON-Pointer composition
req, resp = app.resolve(jp_compose('/pet/{petId}', base='#/paths')).get(petId=1)
req.produce('application/json')
pet = client.request((req, resp)).data
assert pet.id == 1
```

---------

## Installation
We support pip installtion.
```bash
pip install pyswagger
```

Additional dependencies must be prepared before firing a request. If you are going to access a remote/local web server, you must install [requests](https://github.com/kennethreitz/requests) first.
```bash
pip install requests
```

If you want to test a local tornado server, please make sure tornado is ready on your environment
``` bash
pip install tornado
```

We also provide native client for flask app, but to use it, flask is also required
``` bash
pip install flask
```


---------

## Reference
All exported API are described in following sections. ![A diagram about relations between components](https://docs.google.com/drawings/d/1DZiJgl4i9L038UJJp3kpwkWRvcNQktf5h-e4m96_C-k/pub?w=849&h=530)

- [App](docs/md/ref/app.md)
- [SwaggerClient](docs/md/ref/client.md)
- [Security](docs/md/ref/security.md)

---------

## Contributors
- [Marcin Goliński](https://github.com/mjgolinski)
- [Andrey Mikhailov](https://github.com/zlovred)
- [Telepenin Nikolay](https://github.com/prefer)
- [WangJiannan](https://github.com/WangJiannan)

---------

## Contribution Guildeline

#### report an issue:
- issues can be reported [here](https://github.com/mission-liao/pyswagger/issues)
- include swagger.json if possible
- turn on logging and report with messages on console
```python
import logging
logger = logging.getLogger('pyswagger')

formatter = logging.Formatter('%(asctime)s - %(name)s - %(levelname)s - %(message)s')

console = logging.StreamHandler()
console.setLevel(logging.DEBUG)
console.setFormatter(formatter)

logger.addHandler(console)
logger.setLevel(logging.DEBUG)

... your stuff

```

- describe expected behavior, or more specific, the input/output

#### submit a PR
- test included
- only PR to `develop` would be accepted

env preparation
```bash
pip install -r requirement-dev.txt
```

unit testing
```bash
python -m pytest -s -v --cov=pyswagger --cov-config=.coveragerc pyswagger/tests
```

 readmeEtag: '"048a9c7059dbcfa9cf05a33ad2a1e65643ed48bf"' readmeLastModified: Mon, 20 Aug 2018 13:44:49 GMT repositoryId: 22305196 description: >- An OpenAPI (fka Swagger) client & converter in python, which is type-safe, dynamic, spec-compliant. created: '2014-07-27T06:47:30Z' updated: '2025-12-10T21:09:03Z' language: Python archived: false stars: 392 watchers: 13 forks: 90 owner: pyopenapi logo: https://avatars.githubusercontent.com/u/35984678?v=4 license: MIT repoEtag: '"1942323c268c1852b657aba13dc291f96f5f8901372f04e52b341590da448893"' repoLastModified: Wed, 10 Dec 2025 21:09:03 GMT foundInMaster: true v3_link: https://github.com/mission-liao/pyswagger/blob/develop/docs/md/news.md id: 83585666ce7c5a42bc3e5463ac9b3c47 oldLocations: - https://github.com/mission-liao/pyswagger - source: https://openapi.tools/ name: JSON Designer category: GUI Editors link: http://jsondesigner.com language: - iOS - Swift source_description: >- Visualize JSON models from imported OpenAPI YAML. Edit models and export OpenAPI YAML. v3: true foundInMaster: true id: 9440025e6c756188ef68d4f144bd42bb - source: openapi3 tags repository: https://github.com/gomarag/siphoon-note v3: true id: 89de05b5bf22f09b2fca45404443c647 repositoryMetadata: base64Readme: >- IyBJbnRybwohWzEw67aE64W47Yq4XShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS84NjEzOTAxMy8xNjI3MTQ3NjUtNzNkZTZkZTItZGE2NC00MjAyLWE1NzItNTE0MzBlYWMxMjczLmdpZikKCj4g4pyP77iPICoq7ZSE66Gc7KCd7Yq466qFKiogOiBgU2lwaG9vbiBOb3RlYAo+IAoKPiDwn5OU77iPICoq7ZSE66Gc7KCd7Yq4IO2Yle2DnCoqIDrrtoDtirjsuqDtlIQg7YyM7J2064SQIO2UhOuhnOygne2KuAo+IAoKPiDwn4yQICoq67Cw7Y+sIOunge2BrCA6IFvwn5Od77iPXShodHRwOi8vZGV2LWRpYXJ5LWJ1Y2tldC5zMy13ZWJzaXRlLmFwLW5vcnRoZWFzdC0yLmFtYXpvbmF3cy5jb20vKSoqW1NpcGhvb24gTm90ZeuhnCDsoJHsho3tlZjquLBdKGh0dHBzOi8vd3d3LnNpcGhvb24tbm90ZS5hcHAvKQo+IAoKPiDwn5GA77iPICoq6riw7ZqN7J2Y64+EKiogOiDquIDsk7DquLDqsIAg7KKL7J2AIOqxtCDslYzqsqDripTrjbAsIOq4tCDquIDsnYQg7JOw64qUIOqyg+ydtCDrtoDri7TsiqTrn6zsm4wg7Iuc7J6R7J20IOyWtOugpOyatCDsgqzrnozrk6TsnYQg7JyE7ZWcIDEw67aEIO2DgOydtOuouCDrhbjtirguCj4KCiMg8J+Tne+4j1NpcGhvb24gTm90ZSDquLDriqUg7IaM6rCcCgrij7DvuI8g6riA7JOw6riwIOuqsOyeheydhCDrj5XripQgKirtg4DsnbTrqLjsmYAg7J6Q64+ZIOyggOyepSDquLDriqUqKuydhCDsoJzqs7XtlbTsmpQhCgrwn5eT77iPIOuFuO2KuOulvCDsnpHshLHtlZwg64Kg7Kec7JeQIOyXsO2VhCDsiqTtg6ztlITrpbwg7LCN7Ja0ICoq7Iq16rSAIOuLrOyEseydvCoq7J2EIO2VnCDriIjsl5Ag7JWMIOyImCDsnojslrTsmpQhCgrwn6WHIOyekeyEsSDquIDsnZgg6rCc7IiY66GcICoq7J20IOyjvOydmCDrnq3tgrkg7Jyg7KCAKirrpbwg7ISg67Cc7ZW07IScIOuztOyXrOykmOyalCEKCvCfjoHvuI8g6riA6rCQ7J20IOuWoOyYpOultOyngCDslYrripQg64Kg7J2EIOychO2VtCAqKuyYpOuKmOydmCDtgqTsm4zrk5wqKuulvCDsoJzqs7XtlbTsmpQhCgrinKjvuI8g6rO16rCcIOyEpOygleycvOuhnCDsnpHshLHtlZwg64W47Yq466W8ICoq66mU7J247Y6Y7J207KeA7JeQIOyGjOqwnCoq7ZWgIOyImCDsnojslrTsmpQhCgrwn5GA77iPIOuFuO2KuOydmCDquIDqsJDsnYQg6rSA66as7ZWY64qUICoq7YOc6re4IOq4sOuKpSoq7J2EIOygnOqzte2VtOyalCEKCiMg8J+boO+4jyDsgqzsmqkg7Iqk7YOdIOuwjyDsi5zsiqTthZwg7JWE7YKk7YWN7LKYCiFbaW1hZ2VdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzIyNDYxMTA4OS1mZDBlNzJiYS1mZjc5LTRmZGUtYTg5ZS1mYWNjMTMwZTFiYmIucG5nKQoKIyDwn5SO77iPIOyjvOyalCDquLDriqUg7Iuc7JewIEdJRgojIOuenOuUqSDtjpjsnbTsp4AKIVvrnpzrlKntjpjsnbTsp4AxXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS84NjEzOTAxMy8xNjQxNzQ0MzMtYjNiMGIxNGYtMWFiYi00YWJjLTg5OWEtYmY3MmI0ZGQ5MTJkLmdpZikKCiMg656c65SpIO2OmOydtOyngCAo7Iqk7YGs66Gk64uk7Jq0KQohW+yKpO2BrOuhpOuLpOyatF0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc1NzE0LTk2NjdkOGU3LWI0OTAtNDI2MS1hNTQ4LWYyN2Q5YTMxYTgxOC5naWYpCgojIOq4gOyTsOq4sCDsg4HsnIQg656t7YK5IOycoOyggCDshozqsJwKIVvsg4HsnITrnq3tgrldKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE3NzMxOC01ZDUxOGQwNy0zMzBhLTQ2ODItYWE2MC03ZWVlNjhhN2Q3ZWEuZ2lmKQoKIyDqs7XqsJzrhbjtirggCiFb6rO16rCc6riAIDRdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE3NzY0NS05YzQ1YmNiZS0xMjM4LTQ1YTItODlkMC00M2MyMTc5MTljODUuZ2lmKQoKIyDrhbjtirgg7Y6Y7J207KeACiFb66mU7J24LSDrhbjtirhdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE3NzczNi04MjY3YzUxMS1hMWM0LTQ0YmMtOTc2Ni00ZThiMjE5ZmVjMmQuZ2lmKQoKIyDsu6zrn6wg7YWM66eIIDXsooUKIVsxOCAg7Lus65+s7YWM66eIIOuzgOqyvV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4NDIyLTA4MzkwNWJiLWY2OTQtNGMyNC1hZGQ3LTUzMDg4MjMxNWM3Ny5naWYpCgojIOq4gOyTsOq4sCDsirXqtIDrtoTshJ0g7Y6Y7J207KeACiFb64us66ClIOyKpO2DrO2UhF0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc3ODcwLTUwYjBkNjMyLWE2MmYtNDI0MC1iMDQ0LWNhYWY4Y2VhM2Y4YS5naWYpCgojIOuFuO2KuCDstpzroKXssL0gKOyVnuuptC/rkrfrqbQpCiFbNyDsubTrk5zruJTroZ0g6re466as65OcXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS84NjEzOTAxMy8xNjQxNzgxMTgtNWM1NTE5M2UtZTliMS00Y2YwLTllNGItZjUyNjczYzA2NzM1LmdpZikKIVsxNSDsubTrk5wg65Kk7KeR6riwXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS84NjEzOTAxMy8xNjQxNzg3MTEtMzFkNjlmZGMtZGI5Ny00NWRlLTk1ZDUtM2FhMDM5NDUwYjE5LmdpZikKCiMg64W47Yq4IOyeheugpeywvQohWzkg7YOA7J2066i4IOyLnOyekV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4MTUxLTNmZjZjYmM2LTEyMWItNGIwOS1iNWU4LTM3NWE1Y2EwMWZmYS5naWYpCgojIO2DgOydtOuouCDquLDriqUo64W47Yq4IOyekOuPmeyggOyepSkKIVsxMSDquIAg7J6Q64+ZIOyggOyepV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4MjE1LWRhMWU4ZjM4LWQzNmMtNDUwYS05MDI2LTNiY2QzZmEyMTVhYy5naWYpCgojIO2DgOydtOuouCDslYjrgrQg66eQ7ZKN7ISgCiFbMTAg7YOA7J2066i4IO2ItO2MgV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc5NjY1LTQ4NzM1ZjFiLTg0MjgtNDQ0YS04M2FlLWM2NGNmNzc0NDAzZi5naWYpCgojIOyYpOuKmOydmCDtgqTsm4zrk5wgCiFbMTMg7Jik64qY7J2YIO2CpOybjOuTnF0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4Mjg0LTdiODlmNDgzLTAwYzAtNGUwMy1iN2Q5LThkMTM0MDM0MTliMS5naWYpCgojIOuFuO2KuCDsl5DrlJTthLAv7KCE7LK067O06riwCiFbMTYg6riAIOyImOyglV0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4MzY3LWRkOWM1ZTEwLTQ1NjEtNDkwMC1hMmJiLTM4YjE4NGQ4NzhmMi5naWYpCiFbMTQg642U67O06riwXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS84NjEzOTAxMy8xNjQxNzgzOTUtMTdjMjIzYTgtNzRiMy00NzU2LWEzY2UtOTVjY2Q3Zjc4Njk3LmdpZikKCiMg7Zy07KeA7Ya1CiFbMTcg6riAIOyCreygnF0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4NTIwLTE5MWYxYTllLTkwNmYtNDkxNy1iZGVhLWM2ZDgyMGQ3MmMzNS5naWYpCgojIOqygOyDieywvQohWzIwIOqygOyDieuwlF0oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vODYxMzkwMTMvMTY0MTc4NTUxLTQ0YzRjYWY0LTM1YTAtNGNkOC1iMTk0LThkZjNhN2QyMjNkOC5naWYpCgojIO2DnOq3uCDqtIDrpqwKIVsxOSDtg5zqt7hdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE3ODYwNy0yZmM2ZjEzZi04YzMyLTQyN2ItYjhkNC0wYWNlMDE4OTUxN2UuZ2lmKQoKIyDtmozsm5Ag6rSA66asIO2OmOydtOyngAojIyDtmozsm5Dsmqkg65Oc66Gt64uk7Jq0CiFb656c65SpKOuhnOq3uOyduCldKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE3NTAyNy00MTg2NjE5Yi0yY2UwLTQ1MzAtYTY5ZS01MDlhMWYzODdkMDAuZ2lmKQojIyDroZzqt7jsnbgKIVvroZzqt7jsnbhdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE4MTU1MS01MzdhZTdlZC0yMTM4LTQ2NWEtYTQ1YS1lOTUzOWE0ZjU2NDMuZ2lmKQojIyDtmozsm5DqsIDsnoUKIVvtmozsm5DqsIDsnoVdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE4MTU2NS1mNmQ5NmE4OC1mYTVlLTQ1YTMtYmQwMy00ZDg3NDc1YTlkNGUuZ2lmKQoKIyDsl5Drn6ztjpjsnbTsp4AKIVsyMiDsv6DtgqRdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzg2MTM5MDEzLzE2NDE3ODc2Ny1jMmY3MDA4NS04MmU2LTQ1YWMtYmEwOC01YzAxMWE5MmQxOGMuZ2lmKQoKCiMg8J+MnSDrp4zrk6Ag7IKs656M65OkCnwg7J2066aEKOq5g+2Xmeunge2BrCkgfCBQb3NpdGlvbiAmIFJvbGV8CnwgLS0tLS0tLSB8IC0tLS0tIHwKfCBb7JWI6rCA656MXShodHRwczovL2dpdGh1Yi5jb20vZ29tYXJhZykgfCDwn5K7IEJhY2stZW5kLCDwn4yfIFRlYW0gbWVtYmVyfAp8IFvsnbTsooXsl7RdKGh0dHBzOi8vZ2l0aHViLmNvbS9qb25neWVvbDEyKSB8IPCfkrsgRnJvbnQtZW5kLCDij7AgVGVhbSBsZWFkZXIgfAp8IFvsnbTqsr3rr7hdKGh0dHBzOi8vZ2l0aHViLmNvbS9MZWVreWVvbmdtaSkgfCDwn5K7IEZyb250LWVuZCwg8J+TnSBUZWFtIG1lbWJlciB8CnwgW+yghOykgO2YlV0oaHR0cHM6Ly9naXRodWIuY29tL2xpbmRpc3QxMikgfCDwn5K7IEZyb250LWVuZCwg8J+XoyBUZWFtIG1lbWJlciB8CgoKCgoK readmeEtag: '"3e324c7bb715543159196f2ccd7a06a42f22b003"' readmeLastModified: Wed, 05 Jul 2023 13:25:51 GMT repositoryId: 470622280 description: 하루 10분! 글쓰기 앱 🚀 10분 노트 created: '2022-03-16T14:37:35Z' updated: '2023-07-07T19:27:17Z' language: JavaScript archived: false stars: 0 watchers: 0 forks: 0 owner: gomarag logo: https://avatars.githubusercontent.com/u/66699849?v=4 repoEtag: '"0642248193af189974b8cd989d1f080707822a80e6f5998d1f2a6bbcdde36f03"' repoLastModified: Fri, 07 Jul 2023 19:27:17 GMT category: Parsers foundInMaster: true - source: https://openapi.tools/ name: Stoplight Docs category: Documentation link: https://stoplight.io/docs/ language: SaaS source_description: >- Create beautiful, customizable, interactive API documentation generated from OpenAPI, integrated with Stoplight Studio. v2: true v3: true v3_1: true foundInMaster: true id: 96eb5e8de02914d4daa8d3910533aa63 - source: openapi3 tags repository: https://github.com/charlyjazz/prettyrestdoc v3: true repositoryMetadata: base64Readme: >- IyBQcmV0dHlSZXN0RG9jCgpbQ2hlY2sgaXQgb3V0IHRoZSBkZW1vIV0oaHR0cHM6Ly9jaGFybHlqYXp6LmdpdGh1Yi5pby9wcmV0dHlyZXN0ZG9jLyNVc2VyKQoKVGhpcyBpcyBhIHRlbXBsYXRlIGNyZWF0ZWQgaW4gUmVhY3QgdG8gZWFzaWx5IGN1c3RvbWl6ZSBhbmQgYnVpbGQgYSBSRVNUIEFQSSBEb2N1bWVudGF0aW9uIGZvciBjb21tZXJjaWFsIG9yIGRldmVsb3BtZW50IHB1cnBvc2VzLiBQcm9qZWN0IGNyZWF0ZWQgdXNpbmcgdXNpbmcgY3JlYXRlLXJlYWN0LWFwcCB3dGggdGhlIHR5cGVzY3JpcHQgdGVtcGxhdGUKCnwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIFJvbGVzIGFuZCBQZXJtaXNzaW9ucyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfCAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgUHJldmlldyBTY2hlbWEgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICB8CnwgOi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLTogfCA6LS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tOiB8CnwgIVtdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzEyNDg5MzMzLzEyNDAxMzk5MS01OTk0NGIwMC1kOWI5LTExZWItODI1ZS01YTAyYTliOTk0ODcuanBlZykgfCAhW10oaHR0cHM6Ly91c2VyLWltYWdlcy5naXRodWJ1c2VyY29udGVudC5jb20vMTI0ODkzMzMvMTI0MDEzOTgzLTU3Y2E4NzgwLWQ5YjktMTFlYi05MGI2LTZhMDIyN2U4YmFjMy5qcGVnKSB8Cgp8ICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgU2VhcmNoICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgRW5kcG9pbnQgU2VjdGlvbiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgfAp8IDotLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS06IHwgOi0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLS0tLTogfAp8ICFbXShodHRwczovL3VzZXItaW1hZ2VzLmdpdGh1YnVzZXJjb250ZW50LmNvbS8xMjQ4OTMzMy8xMjQwMTQwMDItNWI1ZTBlODAtZDliOS0xMWViLTkwYWUtOTBlNDJmNzk4MTQ5LmpwZWcpIHwgIVtdKGh0dHBzOi8vdXNlci1pbWFnZXMuZ2l0aHVidXNlcmNvbnRlbnQuY29tLzEyNDg5MzMzLzEyNDAxNDY5OC0zYTQ5ZWQ4MC1kOWJhLTExZWItOTM1Ny00Njc4ZTRlYmUxN2MuanBlZykgfAoKIyMjIE1lcmdlIEJldHdlZW4gY3VzdG9tIGNvbnRlbnQgYW5kIHlvdXIgQVBJIGNvbnRyYWN0CgpZb3UgY2FuIGVhc2lseSBzeW5jIHlvdSBPcGVuQVBJIDMgSlNPTiBGaWxlIHVzaW5nIHRoZSAiU2VjdGlvbkl0ZW0iIHN0cnVjdHVyZS4gSW4gdGhlIG5leHQgZXhhbXBsZSB5b3UgY2FuIHNlZSBhIFNlY3Rpb25JdGVtIGNyZWF0ZSBmb3IgdGhlIFN0b3JlIHRhZyBmcm9tIHRoZSBQZXRTdG9yZSBTd2FnZ2VyIGV4YW1wbGUuIFRoaXMgd2F5IGNhbiBhZGQgaW1hZ2VzLCBkZXNjcmlwdGlvbnMsIGFuZCBpbnRyb2R1Y3Rvcnkgc2VjdGlvbnMgbGlrZSAiR2V0dGluZyBTdGFydGVkIgoKYGBgamF2YXNjcmlwdApjb25zdCBkb2N1bWVudGF0aW9uOiBTZWN0aW9uSXRlbSA9IHsKICBpZDogInN0b3JlIiwKICB0YWc6ICJzdG9yZSIsCiAgc2NoZW1hOiAiT3JkZXIiLAogIHRpdGxlOiAiU3RvcmUiLAogIGNvbnRlbnQ6IHsKICAgIGxlZnRfc2VjdGlvbl90aXRsZTogIlN0b3JlcyIsCiAgICBsZWZ0X3NlY3Rpb25fcGFyYWdyYXBoczogWyJUaGlzIGlzIGEgU3RvcmUgcmVsYXRlZCBvcGVyYXRpb24iXSwKICAgIHJpZ2h0X2JveF9zbmlwcGV0X3doaXRlOiB0cnVlLAogICAgcmlnaHRfYm94X3NuaXBwZXRfdGl0bGU6ICJFeGFtcGxlIG9mIGEgT3JkZXIgb2JqZWN0IiwKICB9LAogIGlzX2NvcmVfcmVzb3VyY2U6IHRydWUsCn07CmBgYAoKU3RydWN0dXJlIG9mIGEgYFNlY3Rpb25JdGVtYDoKCmBpZGA6IEFuY2hvciBmb3IgVVJMCgpgdGFnYDogVGFnIHRvIGZvdW5kIGNvbnRlbnQgaW5zaWRlIHRoZSBPcGVuQVBJMyBKU09OIGZpbGUKCmB0aXRsZWA6IFRpdGxlIGZvciBVSQoKYGNvbnRlbnRgOiBUaXRsZXMgYW5kIGRlc2NyaXB0aW9ucy4gQWxzbyB5b3UgY2FuIGNyZWF0ZSBhIG9iamVjdCBsaWtlIGEgZXhhbXBsZSBvZiBhIHJlc3BvbnNlL3JlcXVlc3QuCgpgaXNfY29yZV9yZXNvdXJjZWA6IFRoaXMgYm9vbGVhbiBwYXJhbWV0ZXIgaGVscCB0byBzcGxpdCB0aGUgY29udGVudCBiZXR3ZWVuIHNlY3Rpb25zIGxpa2UgYSAiR2V0dGluZyBTdGFydGVkIiBhbmQgdGhlIEFQSSBlbmRwb2ludHMKCldpdGggYSBBcnJheSBvZiBTZWN0aW9uSXRlbXMgYW5kIHRoZSBKU09OIG9mIHRoZSBPcGVuQVBJIFNjaGVtYSB5b3UgY2FuIHJlbmRlciB0aGUgZG9jdW1lbnRhdGlvbjoKCmBgYGphdmFzY3JpcHQKaW1wb3J0IGRvY1N3YWdnZXIgZnJvbSAiLi9wZXRzdG9yZS5qc29uIjsgLy8gT3BlbkFQSSBKU09OCmltcG9ydCBBUElEb2MgZnJvbSAiLi9kb2MiOyAvLyBBcnJheSBvZiBTZWN0aW9uSXRlbQoKPFByZXR0eVJlc3REb2MgZG9jQ3VzdG9tPXtBUElEb2N9IGRvY1N3YWdnZXI9e2RvY1N3YWdnZXJ9IC8+OwpgYGAKCiMjIyBTZWN0aW9uIGZvciBSb2xlcyBhbmQgUGVybWlzc2lvbnMKCllvdSBjYW4gY3JlYXRlIGVhc2lseSBhIHRhYmxlIGZvciByb2xlcyBhbmQgcGVybWlzc2lvbnMgc2VuZGluZyB0aGUgcHJvcCByb2xlcyB0byB0aGUgY29tcG9uZW50LiAqKlRoZSBgU2VjdGlvbkl0ZW1gIG5lZWQgdGhlIGByb2xlcy1wZXJtaXNzaW9uc2AgaWQgZm9yIHRoaXMgc3BlY2lhbCBjYXNlLioqCgpgYGB4bWwKPFByZXR0eVJlc3REb2MKICBkb2NDdXN0b209e0FQSURvY30KICBkb2NTd2FnZ2VyPXtkb2NTd2FnZ2VyIGFzIHVua25vd24gYXMgT3BlbkFQSVYzLkRvY3VtZW50fQogIHJvbGVzPXtbCiAgICB7CiAgICAgIGxhYmVsOiAiQ2xpZW50IiwKICAgICAgdmFsdWU6ICdjbGllbnQnCiAgICB9LAogICAgewogICAgICBsYWJlbDogIkFkbWluIiwKICAgICAgdmFsdWU6ICdhZG1pbicKICAgIH1dfQovPgpgYGAKClRoZW4geW91IHdpbGwgbmVlZCBzZXQgdGhlIHJvbGVzIHRvIGVhY2ggZW5kcG9pbnQgaW4gdGhlIE9wZW5BUEkzIEpTT04uCgpgYGBqc29uCnsKICAicHV0IjogewogICAgInRhZ3MiOiBbInVzZXIiXSwKICAgICJyb2xlcyI6IFsiYWRtaW4iXSwKICAgICJzdW1tYXJ5IjogIlVwZGF0ZSB1c2VyIiwKICAgICJkZXNjcmlwdGlvbiI6ICJUaGlzIGNhbiBvbmx5IGJlIGRvbmUgYnkgdGhlIGxvZ2dlZCBpbiB1c2VyLiIsCiAgICAib3BlcmF0aW9uSWQiOiAidXBkYXRlVXNlciIKICB9Cn0KYGBgCgojIyMgQ3VzdG9taXphdGlvbiBhbmQgU3R5bGVzCgpUaGUgc3RydWN0dXJlIG9mIHRoZSBwcm9qZWN0IGlzIHZlcnkgZWFzeSB0byB1bmRlcnN0YW5kLiBUaGUgY29kZSBvZiB0aGUgZG9jdW1lbnRhdGlvbiBpcyBpbnNpZGUgdGhlIGBzcmMvbGliYCBmb2xkZXI6CgotIGBzcmMvbGliL2FkYXB0ZXItYW8zYDogTG9naWMgdG8gZ2V0IGNvbnRlbnQgZnJvbSBPcGVuQVBJMwotIGBzcmMvbGliL2NvbXBvbmVudHNgOiBBbGwgVUkgY29tcG9uZW50cywgeW91IGNhbiBjaGFuZ2Ugc3R5bGVzIG9yIGxvZ2ljIGVhc2lseSAoTW9kdWxlcyBDU1MgaW1wbGVtZW50ZWQpCi0gYHNyYy9saWIvaG9va3NgOiBIb29rcyEKLSBgc3JjL2xpYi9QcmV0dHlSZXN0RG9jLmpzeGA6IFJvb3QgQ29tcG9uZW50CgojIyMgQ3VycmVudCBleGFtcGxlIGluIHRoaXMgUmVwb3NpdG9yeQoKVGhpcyBSZXBvc2l0b3J5IGhhdmUgYSBleGFtcGxlIHVzaW5nIHRoZSBQZXRTdG9yZSBmcm9tIFN3YWdnZXIgd2l0aCB0aGUgc2VjdGlvbnMgUGV0LCBTdG9yZSBhbmQgVXNlci4gQnV0IHdpdGggdGhyZWUgc2VjdGlvbnMgaW50cm9kdWN0b3JpZXMgdXNpbmcgdGhlIHBvd2VyIG9mIGBTZWN0aW9uc0l0ZW1gIHN0cnVjdHVyZXMuIEFsc28gdGhlcmUgYXJlIGEgc2VjdGlvbiBmb3IgKipSb2xlcyBhbmQgUGVybWlzc2lvbnMqKgoKIyMjIFJ1biBEb2N1bWVudGF0aW9uIGluIHlvdXIgbG9jYWwgQnJvd3NlcgoKYGBgYmFzaApucG0gaW5zdGFsbCAmJiBucG0gcnVuIHN0YXJ0CmBgYAoKIyMjIFJ1biBUZXN0cwoKYGBgYmFzaApucG0gaW5zdGFsbCAmJiBucG0gcnVuIHRlc3QKYGBgCgojIyMgQ2hyb21lIEV4dGVuc2lvbgoKIyMjIENyZWRpdHMKCi0gW1N3YWdnZXJdKGh0dHBzOi8vc3dhZ2dlci5pby8pCi0gW09wZW5BUEldKGh0dHBzOi8vd3d3Lm9wZW5hcGlzLm9yZy8pCg== readmeEtag: '"5038875ae95281b144709eeffe468707ff6aa3a3"' readmeLastModified: Mon, 22 Jan 2024 00:49:03 GMT repositoryId: 381797429 description: >- A chrome extension and also a web app. Swagger UI Alternative. Get a pretty rest documentation app easily cloning this stuff :book: created: '2021-06-30T18:26:03Z' updated: '2025-04-19T07:32:23Z' language: TypeScript archived: false stars: 14 watchers: 1 forks: 0 owner: CharlyJazz logo: https://avatars.githubusercontent.com/u/12489333?v=4 license: Apache-2.0 repoEtag: '"6751b353ab2bfe72123d9722c5fa605e60d44b445af513c550f4b97b19ed143e"' repoLastModified: Sat, 19 Apr 2025 07:32:23 GMT foundInMaster: true category: Parsers id: 8c3f8ed37b463f0b95c1e3f78a9bc2c2 - source: https://openapi.tools/ name: odata2openapi category: Converters repository: https://github.com/elasticio/odata2openapi language: Node.js source_description: OData 4.0 to OpenAPI v2.0 converter v2: true repositoryMetadata: base64Readme: >- IyBPRGF0YSB0byBPcGVuQVBJIGNvbnZlcnRlcgoKVGhpcyBub2RlIG1vZHVsZSBjb252ZXJ0cyBhbiBleGlzdGluZyBPRGF0YSBtZXRhZGF0YSB0byBPcGVuQVBJIGZvcm1hdC4KCiMjIEluc3RhbGwKClJ1biBgbnBtIGluc3RhbGwgLS1zYXZlIG9kYXRhMm9wZW5hcGlgCgojIyBVc2FnZQoKIyMjIENvbnZlcnRpbmcgZXhpc3RpbmcgWE1MIHN0cmluZwoKVXNlIHRoZSBgcGFyc2VgIGFuZCBgY29udmVydGAgbWV0aG9kcyBpZiB5b3UgaGF2ZSB0aGUgbWV0YWRhdGEgYXMgWE1MLgoKIyMjIyBKYXZhU2NyaXB0CgpgYGBqcwpjb25zdCB7IHBhcnNlLCBjb252ZXJ0IH0gPSByZXF1aXJlKCdvZGF0YTJvcGVuYXBpJyk7CgovLyBHZXQgdGhlIE9EYXRhIG1ldGFkYXRhIGFzIGEgc3RyaW5nLgpjb25zdCB4bWwgPSAnJzsKCmNvbnN0IG9wdGlvbnMgPSB7CiAgaG9zdDogJ3NlcnZpY2VzLm9kYXRhLm9yZycsCiAgcGF0aDogJy9WNC9Ob3J0aHdpbmQvTm9ydGh3aW5kLnN2YycKfTsKCnBhcnNlKHhtbCkKICAudGhlbihzZXJ2aWNlID0+IGNvbnZlcnQoc2VydmljZS5lbnRpdHlTZXRzLCBvcHRpb25zLCBzZXJ2aWNlLnZlcnNpb24pKQogIC50aGVuKHN3YWdnZXIgPT4gY29uc29sZS5sb2coSlNPTi5zdHJpbmdpZnkoc3dhZ2dlciwgbnVsbCwgMikpKQogIC5jYXRjaChlcnJvciA9PiBjb25zb2xlLmVycm9yKGVycm9yKSkKYGBgCgojIyMjIFR5cGVTY3JpcHQKYGBgVHlwZVNjcmlwdAppbXBvcnQgeyBwYXJzZSwgY29udmVydCwgT3B0aW9ucyB9IGZyb20gJ29kYXRhMm9wZW5hcGknOwoKY29uc3Qgb3B0aW9uczogT3B0aW9ucyA9IHsKICBob3N0OiAnc2VydmljZXMub2RhdGEub3JnJywKICBwYXRoOiAnL1Y0L05vcnRod2luZC9Ob3J0aHdpbmQuc3ZjJwp9OwoKLy8gR2V0IHRoZSBPRGF0YSBtZXRhZGF0YSBhcyBhIHN0cmluZy4KY29uc3QgeG1sID0gJyc7CgpwYXJzZSh4bWwpCiAgLnRoZW4oc2VydmljZSA9PiBjb252ZXJ0KHNlcnZpY2UuZW50aXR5U2V0cywgb3B0aW9ucywgc2VydmljZS52ZXJzaW9uKSkKICAudGhlbihzd2FnZ2VyID0+IGNvbnNvbGUubG9nKEpTT04uc3RyaW5naWZ5KHN3YWdnZXIsIG51bGwsIDIpKSkKICAuY2F0Y2goZXJyb3IgPT4gY29uc29sZS5lcnJvcihlcnJvcikpCmBgYAo= readmeEtag: '"4923b5af883d3513a48dedd1723c91524b5b1fca"' readmeLastModified: Thu, 20 Feb 2020 13:38:58 GMT repositoryId: 64212637 description: OData to OpenAPI Converter created: '2016-07-26T10:29:14Z' updated: '2025-06-10T08:39:30Z' language: TypeScript archived: false stars: 37 watchers: 10 forks: 24 owner: elasticio logo: https://avatars.githubusercontent.com/u/1560253?v=4 license: MIT repoEtag: '"f8911f5a997a01d1c3d432dcd5c8ec9caea532da83695b84c3f74b40238266fb"' repoLastModified: Tue, 10 Jun 2025 08:39:30 GMT foundInMaster: true id: deb9defcd95175a7029de6b81cf1ca78